ZGlmZiAtLWdpdCBhL01PRFVMRV9MSUNFTlNFX0FQQUNIRTIgYi9NT0RVTEVfTElDRU5TRV9BUEFDSEUyCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlNjlkZTI5Li4wMDAwMDAwCi0tLSBhL01PRFVMRV9MSUNFTlNFX0FQQUNIRTIKKysrIC9kZXYvbnVsbApkaWZmIC0tZ2l0IGEvTk9USUNFIGIvTk9USUNFCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyNjdhNmFhLi4wMDAwMDAwCi0tLSBhL05PVElDRQorKysgL2Rldi9udWxsCkBAIC0xLDIyMiArMCwwIEBACi0gICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gICA9PSAgTk9USUNFIGZpbGUgY29ycmVzcG9uZGluZyB0byB0aGUgc2VjdGlvbiA0IGQgb2YgICAgICAgICAgICAgICAgICAgID09Ci0gICA9PSAgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID09Ci0gICA9PSAgaW4gdGhpcyBjYXNlIGZvciB0aGUgQW5kcm9pZC1zcGVjaWZpYyBjb2RlLiAgICAgICAgICAgICAgICAgICAgICAgID09Ci0gICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0KLUFuZHJvaWQgQ29kZQotQ29weXJpZ2h0IDIwMDUtMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0KLVRoaXMgcHJvZHVjdCBpbmNsdWRlcyBzb2Z0d2FyZSBkZXZlbG9wZWQgYXMgcGFydCBvZgotVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdCAoaHR0cDovL3NvdXJjZS5hbmRyb2lkLmNvbSkuCi0KLSAgID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAgID09ICBOT1RJQ0UgZmlsZSBjb3JyZXNwb25kaW5nIHRvIHRoZSBzZWN0aW9uIDQgZCBvZiAgICAgICAgICAgICAgICAgICAgPT0KLSAgID09ICB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wLCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT0KLSAgID09ICBpbiB0aGlzIGNhc2UgZm9yIEFwYWNoZSBDb21tb25zIGNvZGUuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT0KLSAgID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLQotQXBhY2hlIENvbW1vbnMKLUNvcHlyaWdodCAxOTk5LTIwMDQgVGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uCi0KLVRoaXMgcHJvZHVjdCBpbmNsdWRlcyBzb2Z0d2FyZSBkZXZlbG9wZWQgYXQKLVRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoaHR0cDovL3d3dy5hcGFjaGUub3JnLykuCi0KLSAgID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAgID09ICBOT1RJQ0UgZmlsZSBjb3JyZXNwb25kaW5nIHRvIHRoZSBzZWN0aW9uIDQgZCBvZiAgICAgICAgICAgICAgICAgICAgPT0KLSAgID09ICB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wLCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT0KLSAgID09ICBpbiB0aGlzIGNhc2UgZm9yIEpha2FydGEgQ29tbW9ucyBMb2dnaW5nLiAgICAgICAgICAgICAgICAgICAgICAgICAgPT0KLSAgID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLQotSmFrYXJ0YSBDb21tb25zIExvZ2dpbmcgKEpDTCkKLUNvcHlyaWdodCAyMDA1LDIwMDYgVGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uLgotCi1UaGlzIHByb2R1Y3QgaW5jbHVkZXMgc29mdHdhcmUgZGV2ZWxvcGVkIGF0Ci1UaGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKGh0dHA6Ly93d3cuYXBhY2hlLm9yZy8pLgotCi0gICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gICA9PSAgTk9USUNFIGZpbGUgY29ycmVzcG9uZGluZyB0byB0aGUgc2VjdGlvbiA0IGQgb2YgICAgICAgICAgICAgICAgICAgID09Ci0gICA9PSAgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID09Ci0gICA9PSAgaW4gdGhpcyBjYXNlIGZvciB0aGUgTnVhbmNlIGNvZGUuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID09Ci0gICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0KLVRoZXNlIGZpbGVzIGFyZSBDb3B5cmlnaHQgMjAwNyBOdWFuY2UgQ29tbXVuaWNhdGlvbnMsIGJ1dCByZWxlYXNlZCB1bmRlcgotdGhlIEFwYWNoZTIgTGljZW5zZS4KLQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEFwYWNoZSBMaWNlbnNlCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBWZXJzaW9uIDIuMCwgSmFudWFyeSAyMDA0Ci0gICAgICAgICAgICAgICAgICAgICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvCi0KLSAgIFRFUk1TIEFORCBDT05ESVRJT05TIEZPUiBVU0UsIFJFUFJPRFVDVElPTiwgQU5EIERJU1RSSUJVVElPTgotCi0gICAxLiBEZWZpbml0aW9ucy4KLQotICAgICAgIkxpY2Vuc2UiIHNoYWxsIG1lYW4gdGhlIHRlcm1zIGFuZCBjb25kaXRpb25zIGZvciB1c2UsIHJlcHJvZHVjdGlvbiwKLSAgICAgIGFuZCBkaXN0cmlidXRpb24gYXMgZGVmaW5lZCBieSBTZWN0aW9ucyAxIHRocm91Z2ggOSBvZiB0aGlzIGRvY3VtZW50LgotCi0gICAgICAiTGljZW5zb3IiIHNoYWxsIG1lYW4gdGhlIGNvcHlyaWdodCBvd25lciBvciBlbnRpdHkgYXV0aG9yaXplZCBieQotICAgICAgdGhlIGNvcHlyaWdodCBvd25lciB0aGF0IGlzIGdyYW50aW5nIHRoZSBMaWNlbnNlLgotCi0gICAgICAiTGVnYWwgRW50aXR5IiBzaGFsbCBtZWFuIHRoZSB1bmlvbiBvZiB0aGUgYWN0aW5nIGVudGl0eSBhbmQgYWxsCi0gICAgICBvdGhlciBlbnRpdGllcyB0aGF0IGNvbnRyb2wsIGFyZSBjb250cm9sbGVkIGJ5LCBvciBhcmUgdW5kZXIgY29tbW9uCi0gICAgICBjb250cm9sIHdpdGggdGhhdCBlbnRpdHkuIEZvciB0aGUgcHVycG9zZXMgb2YgdGhpcyBkZWZpbml0aW9uLAotICAgICAgImNvbnRyb2wiIG1lYW5zIChpKSB0aGUgcG93ZXIsIGRpcmVjdCBvciBpbmRpcmVjdCwgdG8gY2F1c2UgdGhlCi0gICAgICBkaXJlY3Rpb24gb3IgbWFuYWdlbWVudCBvZiBzdWNoIGVudGl0eSwgd2hldGhlciBieSBjb250cmFjdCBvcgotICAgICAgb3RoZXJ3aXNlLCBvciAoaWkpIG93bmVyc2hpcCBvZiBmaWZ0eSBwZXJjZW50ICg1MCUpIG9yIG1vcmUgb2YgdGhlCi0gICAgICBvdXRzdGFuZGluZyBzaGFyZXMsIG9yIChpaWkpIGJlbmVmaWNpYWwgb3duZXJzaGlwIG9mIHN1Y2ggZW50aXR5LgotCi0gICAgICAiWW91IiAob3IgIllvdXIiKSBzaGFsbCBtZWFuIGFuIGluZGl2aWR1YWwgb3IgTGVnYWwgRW50aXR5Ci0gICAgICBleGVyY2lzaW5nIHBlcm1pc3Npb25zIGdyYW50ZWQgYnkgdGhpcyBMaWNlbnNlLgotCi0gICAgICAiU291cmNlIiBmb3JtIHNoYWxsIG1lYW4gdGhlIHByZWZlcnJlZCBmb3JtIGZvciBtYWtpbmcgbW9kaWZpY2F0aW9ucywKLSAgICAgIGluY2x1ZGluZyBidXQgbm90IGxpbWl0ZWQgdG8gc29mdHdhcmUgc291cmNlIGNvZGUsIGRvY3VtZW50YXRpb24KLSAgICAgIHNvdXJjZSwgYW5kIGNvbmZpZ3VyYXRpb24gZmlsZXMuCi0KLSAgICAgICJPYmplY3QiIGZvcm0gc2hhbGwgbWVhbiBhbnkgZm9ybSByZXN1bHRpbmcgZnJvbSBtZWNoYW5pY2FsCi0gICAgICB0cmFuc2Zvcm1hdGlvbiBvciB0cmFuc2xhdGlvbiBvZiBhIFNvdXJjZSBmb3JtLCBpbmNsdWRpbmcgYnV0Ci0gICAgICBub3QgbGltaXRlZCB0byBjb21waWxlZCBvYmplY3QgY29kZSwgZ2VuZXJhdGVkIGRvY3VtZW50YXRpb24sCi0gICAgICBhbmQgY29udmVyc2lvbnMgdG8gb3RoZXIgbWVkaWEgdHlwZXMuCi0KLSAgICAgICJXb3JrIiBzaGFsbCBtZWFuIHRoZSB3b3JrIG9mIGF1dGhvcnNoaXAsIHdoZXRoZXIgaW4gU291cmNlIG9yCi0gICAgICBPYmplY3QgZm9ybSwgbWFkZSBhdmFpbGFibGUgdW5kZXIgdGhlIExpY2Vuc2UsIGFzIGluZGljYXRlZCBieSBhCi0gICAgICBjb3B5cmlnaHQgbm90aWNlIHRoYXQgaXMgaW5jbHVkZWQgaW4gb3IgYXR0YWNoZWQgdG8gdGhlIHdvcmsKLSAgICAgIChhbiBleGFtcGxlIGlzIHByb3ZpZGVkIGluIHRoZSBBcHBlbmRpeCBiZWxvdykuCi0KLSAgICAgICJEZXJpdmF0aXZlIFdvcmtzIiBzaGFsbCBtZWFuIGFueSB3b3JrLCB3aGV0aGVyIGluIFNvdXJjZSBvciBPYmplY3QKLSAgICAgIGZvcm0sIHRoYXQgaXMgYmFzZWQgb24gKG9yIGRlcml2ZWQgZnJvbSkgdGhlIFdvcmsgYW5kIGZvciB3aGljaCB0aGUKLSAgICAgIGVkaXRvcmlhbCByZXZpc2lvbnMsIGFubm90YXRpb25zLCBlbGFib3JhdGlvbnMsIG9yIG90aGVyIG1vZGlmaWNhdGlvbnMKLSAgICAgIHJlcHJlc2VudCwgYXMgYSB3aG9sZSwgYW4gb3JpZ2luYWwgd29yayBvZiBhdXRob3JzaGlwLiBGb3IgdGhlIHB1cnBvc2VzCi0gICAgICBvZiB0aGlzIExpY2Vuc2UsIERlcml2YXRpdmUgV29ya3Mgc2hhbGwgbm90IGluY2x1ZGUgd29ya3MgdGhhdCByZW1haW4KLSAgICAgIHNlcGFyYWJsZSBmcm9tLCBvciBtZXJlbHkgbGluayAob3IgYmluZCBieSBuYW1lKSB0byB0aGUgaW50ZXJmYWNlcyBvZiwKLSAgICAgIHRoZSBXb3JrIGFuZCBEZXJpdmF0aXZlIFdvcmtzIHRoZXJlb2YuCi0KLSAgICAgICJDb250cmlidXRpb24iIHNoYWxsIG1lYW4gYW55IHdvcmsgb2YgYXV0aG9yc2hpcCwgaW5jbHVkaW5nCi0gICAgICB0aGUgb3JpZ2luYWwgdmVyc2lvbiBvZiB0aGUgV29yayBhbmQgYW55IG1vZGlmaWNhdGlvbnMgb3IgYWRkaXRpb25zCi0gICAgICB0byB0aGF0IFdvcmsgb3IgRGVyaXZhdGl2ZSBXb3JrcyB0aGVyZW9mLCB0aGF0IGlzIGludGVudGlvbmFsbHkKLSAgICAgIHN1Ym1pdHRlZCB0byBMaWNlbnNvciBmb3IgaW5jbHVzaW9uIGluIHRoZSBXb3JrIGJ5IHRoZSBjb3B5cmlnaHQgb3duZXIKLSAgICAgIG9yIGJ5IGFuIGluZGl2aWR1YWwgb3IgTGVnYWwgRW50aXR5IGF1dGhvcml6ZWQgdG8gc3VibWl0IG9uIGJlaGFsZiBvZgotICAgICAgdGhlIGNvcHlyaWdodCBvd25lci4gRm9yIHRoZSBwdXJwb3NlcyBvZiB0aGlzIGRlZmluaXRpb24sICJzdWJtaXR0ZWQiCi0gICAgICBtZWFucyBhbnkgZm9ybSBvZiBlbGVjdHJvbmljLCB2ZXJiYWwsIG9yIHdyaXR0ZW4gY29tbXVuaWNhdGlvbiBzZW50Ci0gICAgICB0byB0aGUgTGljZW5zb3Igb3IgaXRzIHJlcHJlc2VudGF0aXZlcywgaW5jbHVkaW5nIGJ1dCBub3QgbGltaXRlZCB0bwotICAgICAgY29tbXVuaWNhdGlvbiBvbiBlbGVjdHJvbmljIG1haWxpbmcgbGlzdHMsIHNvdXJjZSBjb2RlIGNvbnRyb2wgc3lzdGVtcywKLSAgICAgIGFuZCBpc3N1ZSB0cmFja2luZyBzeXN0ZW1zIHRoYXQgYXJlIG1hbmFnZWQgYnksIG9yIG9uIGJlaGFsZiBvZiwgdGhlCi0gICAgICBMaWNlbnNvciBmb3IgdGhlIHB1cnBvc2Ugb2YgZGlzY3Vzc2luZyBhbmQgaW1wcm92aW5nIHRoZSBXb3JrLCBidXQKLSAgICAgIGV4Y2x1ZGluZyBjb21tdW5pY2F0aW9uIHRoYXQgaXMgY29uc3BpY3VvdXNseSBtYXJrZWQgb3Igb3RoZXJ3aXNlCi0gICAgICBkZXNpZ25hdGVkIGluIHdyaXRpbmcgYnkgdGhlIGNvcHlyaWdodCBvd25lciBhcyAiTm90IGEgQ29udHJpYnV0aW9uLiIKLQotICAgICAgIkNvbnRyaWJ1dG9yIiBzaGFsbCBtZWFuIExpY2Vuc29yIGFuZCBhbnkgaW5kaXZpZHVhbCBvciBMZWdhbCBFbnRpdHkKLSAgICAgIG9uIGJlaGFsZiBvZiB3aG9tIGEgQ29udHJpYnV0aW9uIGhhcyBiZWVuIHJlY2VpdmVkIGJ5IExpY2Vuc29yIGFuZAotICAgICAgc3Vic2VxdWVudGx5IGluY29ycG9yYXRlZCB3aXRoaW4gdGhlIFdvcmsuCi0KLSAgIDIuIEdyYW50IG9mIENvcHlyaWdodCBMaWNlbnNlLiBTdWJqZWN0IHRvIHRoZSB0ZXJtcyBhbmQgY29uZGl0aW9ucyBvZgotICAgICAgdGhpcyBMaWNlbnNlLCBlYWNoIENvbnRyaWJ1dG9yIGhlcmVieSBncmFudHMgdG8gWW91IGEgcGVycGV0dWFsLAotICAgICAgd29ybGR3aWRlLCBub24tZXhjbHVzaXZlLCBuby1jaGFyZ2UsIHJveWFsdHktZnJlZSwgaXJyZXZvY2FibGUKLSAgICAgIGNvcHlyaWdodCBsaWNlbnNlIHRvIHJlcHJvZHVjZSwgcHJlcGFyZSBEZXJpdmF0aXZlIFdvcmtzIG9mLAotICAgICAgcHVibGljbHkgZGlzcGxheSwgcHVibGljbHkgcGVyZm9ybSwgc3VibGljZW5zZSwgYW5kIGRpc3RyaWJ1dGUgdGhlCi0gICAgICBXb3JrIGFuZCBzdWNoIERlcml2YXRpdmUgV29ya3MgaW4gU291cmNlIG9yIE9iamVjdCBmb3JtLgotCi0gICAzLiBHcmFudCBvZiBQYXRlbnQgTGljZW5zZS4gU3ViamVjdCB0byB0aGUgdGVybXMgYW5kIGNvbmRpdGlvbnMgb2YKLSAgICAgIHRoaXMgTGljZW5zZSwgZWFjaCBDb250cmlidXRvciBoZXJlYnkgZ3JhbnRzIHRvIFlvdSBhIHBlcnBldHVhbCwKLSAgICAgIHdvcmxkd2lkZSwgbm9uLWV4Y2x1c2l2ZSwgbm8tY2hhcmdlLCByb3lhbHR5LWZyZWUsIGlycmV2b2NhYmxlCi0gICAgICAoZXhjZXB0IGFzIHN0YXRlZCBpbiB0aGlzIHNlY3Rpb24pIHBhdGVudCBsaWNlbnNlIHRvIG1ha2UsIGhhdmUgbWFkZSwKLSAgICAgIHVzZSwgb2ZmZXIgdG8gc2VsbCwgc2VsbCwgaW1wb3J0LCBhbmQgb3RoZXJ3aXNlIHRyYW5zZmVyIHRoZSBXb3JrLAotICAgICAgd2hlcmUgc3VjaCBsaWNlbnNlIGFwcGxpZXMgb25seSB0byB0aG9zZSBwYXRlbnQgY2xhaW1zIGxpY2Vuc2FibGUKLSAgICAgIGJ5IHN1Y2ggQ29udHJpYnV0b3IgdGhhdCBhcmUgbmVjZXNzYXJpbHkgaW5mcmluZ2VkIGJ5IHRoZWlyCi0gICAgICBDb250cmlidXRpb24ocykgYWxvbmUgb3IgYnkgY29tYmluYXRpb24gb2YgdGhlaXIgQ29udHJpYnV0aW9uKHMpCi0gICAgICB3aXRoIHRoZSBXb3JrIHRvIHdoaWNoIHN1Y2ggQ29udHJpYnV0aW9uKHMpIHdhcyBzdWJtaXR0ZWQuIElmIFlvdQotICAgICAgaW5zdGl0dXRlIHBhdGVudCBsaXRpZ2F0aW9uIGFnYWluc3QgYW55IGVudGl0eSAoaW5jbHVkaW5nIGEKLSAgICAgIGNyb3NzLWNsYWltIG9yIGNvdW50ZXJjbGFpbSBpbiBhIGxhd3N1aXQpIGFsbGVnaW5nIHRoYXQgdGhlIFdvcmsKLSAgICAgIG9yIGEgQ29udHJpYnV0aW9uIGluY29ycG9yYXRlZCB3aXRoaW4gdGhlIFdvcmsgY29uc3RpdHV0ZXMgZGlyZWN0Ci0gICAgICBvciBjb250cmlidXRvcnkgcGF0ZW50IGluZnJpbmdlbWVudCwgdGhlbiBhbnkgcGF0ZW50IGxpY2Vuc2VzCi0gICAgICBncmFudGVkIHRvIFlvdSB1bmRlciB0aGlzIExpY2Vuc2UgZm9yIHRoYXQgV29yayBzaGFsbCB0ZXJtaW5hdGUKLSAgICAgIGFzIG9mIHRoZSBkYXRlIHN1Y2ggbGl0aWdhdGlvbiBpcyBmaWxlZC4KLQotICAgNC4gUmVkaXN0cmlidXRpb24uIFlvdSBtYXkgcmVwcm9kdWNlIGFuZCBkaXN0cmlidXRlIGNvcGllcyBvZiB0aGUKLSAgICAgIFdvcmsgb3IgRGVyaXZhdGl2ZSBXb3JrcyB0aGVyZW9mIGluIGFueSBtZWRpdW0sIHdpdGggb3Igd2l0aG91dAotICAgICAgbW9kaWZpY2F0aW9ucywgYW5kIGluIFNvdXJjZSBvciBPYmplY3QgZm9ybSwgcHJvdmlkZWQgdGhhdCBZb3UKLSAgICAgIG1lZXQgdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOgotCi0gICAgICAoYSkgWW91IG11c3QgZ2l2ZSBhbnkgb3RoZXIgcmVjaXBpZW50cyBvZiB0aGUgV29yayBvcgotICAgICAgICAgIERlcml2YXRpdmUgV29ya3MgYSBjb3B5IG9mIHRoaXMgTGljZW5zZTsgYW5kCi0KLSAgICAgIChiKSBZb3UgbXVzdCBjYXVzZSBhbnkgbW9kaWZpZWQgZmlsZXMgdG8gY2FycnkgcHJvbWluZW50IG5vdGljZXMKLSAgICAgICAgICBzdGF0aW5nIHRoYXQgWW91IGNoYW5nZWQgdGhlIGZpbGVzOyBhbmQKLQotICAgICAgKGMpIFlvdSBtdXN0IHJldGFpbiwgaW4gdGhlIFNvdXJjZSBmb3JtIG9mIGFueSBEZXJpdmF0aXZlIFdvcmtzCi0gICAgICAgICAgdGhhdCBZb3UgZGlzdHJpYnV0ZSwgYWxsIGNvcHlyaWdodCwgcGF0ZW50LCB0cmFkZW1hcmssIGFuZAotICAgICAgICAgIGF0dHJpYnV0aW9uIG5vdGljZXMgZnJvbSB0aGUgU291cmNlIGZvcm0gb2YgdGhlIFdvcmssCi0gICAgICAgICAgZXhjbHVkaW5nIHRob3NlIG5vdGljZXMgdGhhdCBkbyBub3QgcGVydGFpbiB0byBhbnkgcGFydCBvZgotICAgICAgICAgIHRoZSBEZXJpdmF0aXZlIFdvcmtzOyBhbmQKLQotICAgICAgKGQpIElmIHRoZSBXb3JrIGluY2x1ZGVzIGEgIk5PVElDRSIgdGV4dCBmaWxlIGFzIHBhcnQgb2YgaXRzCi0gICAgICAgICAgZGlzdHJpYnV0aW9uLCB0aGVuIGFueSBEZXJpdmF0aXZlIFdvcmtzIHRoYXQgWW91IGRpc3RyaWJ1dGUgbXVzdAotICAgICAgICAgIGluY2x1ZGUgYSByZWFkYWJsZSBjb3B5IG9mIHRoZSBhdHRyaWJ1dGlvbiBub3RpY2VzIGNvbnRhaW5lZAotICAgICAgICAgIHdpdGhpbiBzdWNoIE5PVElDRSBmaWxlLCBleGNsdWRpbmcgdGhvc2Ugbm90aWNlcyB0aGF0IGRvIG5vdAotICAgICAgICAgIHBlcnRhaW4gdG8gYW55IHBhcnQgb2YgdGhlIERlcml2YXRpdmUgV29ya3MsIGluIGF0IGxlYXN0IG9uZQotICAgICAgICAgIG9mIHRoZSBmb2xsb3dpbmcgcGxhY2VzOiB3aXRoaW4gYSBOT1RJQ0UgdGV4dCBmaWxlIGRpc3RyaWJ1dGVkCi0gICAgICAgICAgYXMgcGFydCBvZiB0aGUgRGVyaXZhdGl2ZSBXb3Jrczsgd2l0aGluIHRoZSBTb3VyY2UgZm9ybSBvcgotICAgICAgICAgIGRvY3VtZW50YXRpb24sIGlmIHByb3ZpZGVkIGFsb25nIHdpdGggdGhlIERlcml2YXRpdmUgV29ya3M7IG9yLAotICAgICAgICAgIHdpdGhpbiBhIGRpc3BsYXkgZ2VuZXJhdGVkIGJ5IHRoZSBEZXJpdmF0aXZlIFdvcmtzLCBpZiBhbmQKLSAgICAgICAgICB3aGVyZXZlciBzdWNoIHRoaXJkLXBhcnR5IG5vdGljZXMgbm9ybWFsbHkgYXBwZWFyLiBUaGUgY29udGVudHMKLSAgICAgICAgICBvZiB0aGUgTk9USUNFIGZpbGUgYXJlIGZvciBpbmZvcm1hdGlvbmFsIHB1cnBvc2VzIG9ubHkgYW5kCi0gICAgICAgICAgZG8gbm90IG1vZGlmeSB0aGUgTGljZW5zZS4gWW91IG1heSBhZGQgWW91ciBvd24gYXR0cmlidXRpb24KLSAgICAgICAgICBub3RpY2VzIHdpdGhpbiBEZXJpdmF0aXZlIFdvcmtzIHRoYXQgWW91IGRpc3RyaWJ1dGUsIGFsb25nc2lkZQotICAgICAgICAgIG9yIGFzIGFuIGFkZGVuZHVtIHRvIHRoZSBOT1RJQ0UgdGV4dCBmcm9tIHRoZSBXb3JrLCBwcm92aWRlZAotICAgICAgICAgIHRoYXQgc3VjaCBhZGRpdGlvbmFsIGF0dHJpYnV0aW9uIG5vdGljZXMgY2Fubm90IGJlIGNvbnN0cnVlZAotICAgICAgICAgIGFzIG1vZGlmeWluZyB0aGUgTGljZW5zZS4KLQotICAgICAgWW91IG1heSBhZGQgWW91ciBvd24gY29weXJpZ2h0IHN0YXRlbWVudCB0byBZb3VyIG1vZGlmaWNhdGlvbnMgYW5kCi0gICAgICBtYXkgcHJvdmlkZSBhZGRpdGlvbmFsIG9yIGRpZmZlcmVudCBsaWNlbnNlIHRlcm1zIGFuZCBjb25kaXRpb25zCi0gICAgICBmb3IgdXNlLCByZXByb2R1Y3Rpb24sIG9yIGRpc3RyaWJ1dGlvbiBvZiBZb3VyIG1vZGlmaWNhdGlvbnMsIG9yCi0gICAgICBmb3IgYW55IHN1Y2ggRGVyaXZhdGl2ZSBXb3JrcyBhcyBhIHdob2xlLCBwcm92aWRlZCBZb3VyIHVzZSwKLSAgICAgIHJlcHJvZHVjdGlvbiwgYW5kIGRpc3RyaWJ1dGlvbiBvZiB0aGUgV29yayBvdGhlcndpc2UgY29tcGxpZXMgd2l0aAotICAgICAgdGhlIGNvbmRpdGlvbnMgc3RhdGVkIGluIHRoaXMgTGljZW5zZS4KLQotICAgNS4gU3VibWlzc2lvbiBvZiBDb250cmlidXRpb25zLiBVbmxlc3MgWW91IGV4cGxpY2l0bHkgc3RhdGUgb3RoZXJ3aXNlLAotICAgICAgYW55IENvbnRyaWJ1dGlvbiBpbnRlbnRpb25hbGx5IHN1Ym1pdHRlZCBmb3IgaW5jbHVzaW9uIGluIHRoZSBXb3JrCi0gICAgICBieSBZb3UgdG8gdGhlIExpY2Vuc29yIHNoYWxsIGJlIHVuZGVyIHRoZSB0ZXJtcyBhbmQgY29uZGl0aW9ucyBvZgotICAgICAgdGhpcyBMaWNlbnNlLCB3aXRob3V0IGFueSBhZGRpdGlvbmFsIHRlcm1zIG9yIGNvbmRpdGlvbnMuCi0gICAgICBOb3R3aXRoc3RhbmRpbmcgdGhlIGFib3ZlLCBub3RoaW5nIGhlcmVpbiBzaGFsbCBzdXBlcnNlZGUgb3IgbW9kaWZ5Ci0gICAgICB0aGUgdGVybXMgb2YgYW55IHNlcGFyYXRlIGxpY2Vuc2UgYWdyZWVtZW50IHlvdSBtYXkgaGF2ZSBleGVjdXRlZAotICAgICAgd2l0aCBMaWNlbnNvciByZWdhcmRpbmcgc3VjaCBDb250cmlidXRpb25zLgotCi0gICA2LiBUcmFkZW1hcmtzLiBUaGlzIExpY2Vuc2UgZG9lcyBub3QgZ3JhbnQgcGVybWlzc2lvbiB0byB1c2UgdGhlIHRyYWRlCi0gICAgICBuYW1lcywgdHJhZGVtYXJrcywgc2VydmljZSBtYXJrcywgb3IgcHJvZHVjdCBuYW1lcyBvZiB0aGUgTGljZW5zb3IsCi0gICAgICBleGNlcHQgYXMgcmVxdWlyZWQgZm9yIHJlYXNvbmFibGUgYW5kIGN1c3RvbWFyeSB1c2UgaW4gZGVzY3JpYmluZyB0aGUKLSAgICAgIG9yaWdpbiBvZiB0aGUgV29yayBhbmQgcmVwcm9kdWNpbmcgdGhlIGNvbnRlbnQgb2YgdGhlIE5PVElDRSBmaWxlLgotCi0gICA3LiBEaXNjbGFpbWVyIG9mIFdhcnJhbnR5LiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IKLSAgICAgIGFncmVlZCB0byBpbiB3cml0aW5nLCBMaWNlbnNvciBwcm92aWRlcyB0aGUgV29yayAoYW5kIGVhY2gKLSAgICAgIENvbnRyaWJ1dG9yIHByb3ZpZGVzIGl0cyBDb250cmlidXRpb25zKSBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICAgICAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yCi0gICAgICBpbXBsaWVkLCBpbmNsdWRpbmcsIHdpdGhvdXQgbGltaXRhdGlvbiwgYW55IHdhcnJhbnRpZXMgb3IgY29uZGl0aW9ucwotICAgICAgb2YgVElUTEUsIE5PTi1JTkZSSU5HRU1FTlQsIE1FUkNIQU5UQUJJTElUWSwgb3IgRklUTkVTUyBGT1IgQQotICAgICAgUEFSVElDVUxBUiBQVVJQT1NFLiBZb3UgYXJlIHNvbGVseSByZXNwb25zaWJsZSBmb3IgZGV0ZXJtaW5pbmcgdGhlCi0gICAgICBhcHByb3ByaWF0ZW5lc3Mgb2YgdXNpbmcgb3IgcmVkaXN0cmlidXRpbmcgdGhlIFdvcmsgYW5kIGFzc3VtZSBhbnkKLSAgICAgIHJpc2tzIGFzc29jaWF0ZWQgd2l0aCBZb3VyIGV4ZXJjaXNlIG9mIHBlcm1pc3Npb25zIHVuZGVyIHRoaXMgTGljZW5zZS4KLQotICAgOC4gTGltaXRhdGlvbiBvZiBMaWFiaWxpdHkuIEluIG5vIGV2ZW50IGFuZCB1bmRlciBubyBsZWdhbCB0aGVvcnksCi0gICAgICB3aGV0aGVyIGluIHRvcnQgKGluY2x1ZGluZyBuZWdsaWdlbmNlKSwgY29udHJhY3QsIG9yIG90aGVyd2lzZSwKLSAgICAgIHVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyAoc3VjaCBhcyBkZWxpYmVyYXRlIGFuZCBncm9zc2x5Ci0gICAgICBuZWdsaWdlbnQgYWN0cykgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNoYWxsIGFueSBDb250cmlidXRvciBiZQotICAgICAgbGlhYmxlIHRvIFlvdSBmb3IgZGFtYWdlcywgaW5jbHVkaW5nIGFueSBkaXJlY3QsIGluZGlyZWN0LCBzcGVjaWFsLAotICAgICAgaW5jaWRlbnRhbCwgb3IgY29uc2VxdWVudGlhbCBkYW1hZ2VzIG9mIGFueSBjaGFyYWN0ZXIgYXJpc2luZyBhcyBhCi0gICAgICByZXN1bHQgb2YgdGhpcyBMaWNlbnNlIG9yIG91dCBvZiB0aGUgdXNlIG9yIGluYWJpbGl0eSB0byB1c2UgdGhlCi0gICAgICBXb3JrIChpbmNsdWRpbmcgYnV0IG5vdCBsaW1pdGVkIHRvIGRhbWFnZXMgZm9yIGxvc3Mgb2YgZ29vZHdpbGwsCi0gICAgICB3b3JrIHN0b3BwYWdlLCBjb21wdXRlciBmYWlsdXJlIG9yIG1hbGZ1bmN0aW9uLCBvciBhbnkgYW5kIGFsbAotICAgICAgb3RoZXIgY29tbWVyY2lhbCBkYW1hZ2VzIG9yIGxvc3NlcyksIGV2ZW4gaWYgc3VjaCBDb250cmlidXRvcgotICAgICAgaGFzIGJlZW4gYWR2aXNlZCBvZiB0aGUgcG9zc2liaWxpdHkgb2Ygc3VjaCBkYW1hZ2VzLgotCi0gICA5LiBBY2NlcHRpbmcgV2FycmFudHkgb3IgQWRkaXRpb25hbCBMaWFiaWxpdHkuIFdoaWxlIHJlZGlzdHJpYnV0aW5nCi0gICAgICB0aGUgV29yayBvciBEZXJpdmF0aXZlIFdvcmtzIHRoZXJlb2YsIFlvdSBtYXkgY2hvb3NlIHRvIG9mZmVyLAotICAgICAgYW5kIGNoYXJnZSBhIGZlZSBmb3IsIGFjY2VwdGFuY2Ugb2Ygc3VwcG9ydCwgd2FycmFudHksIGluZGVtbml0eSwKLSAgICAgIG9yIG90aGVyIGxpYWJpbGl0eSBvYmxpZ2F0aW9ucyBhbmQvb3IgcmlnaHRzIGNvbnNpc3RlbnQgd2l0aCB0aGlzCi0gICAgICBMaWNlbnNlLiBIb3dldmVyLCBpbiBhY2NlcHRpbmcgc3VjaCBvYmxpZ2F0aW9ucywgWW91IG1heSBhY3Qgb25seQotICAgICAgb24gWW91ciBvd24gYmVoYWxmIGFuZCBvbiBZb3VyIHNvbGUgcmVzcG9uc2liaWxpdHksIG5vdCBvbiBiZWhhbGYKLSAgICAgIG9mIGFueSBvdGhlciBDb250cmlidXRvciwgYW5kIG9ubHkgaWYgWW91IGFncmVlIHRvIGluZGVtbmlmeSwKLSAgICAgIGRlZmVuZCwgYW5kIGhvbGQgZWFjaCBDb250cmlidXRvciBoYXJtbGVzcyBmb3IgYW55IGxpYWJpbGl0eQotICAgICAgaW5jdXJyZWQgYnksIG9yIGNsYWltcyBhc3NlcnRlZCBhZ2FpbnN0LCBzdWNoIENvbnRyaWJ1dG9yIGJ5IHJlYXNvbgotICAgICAgb2YgeW91ciBhY2NlcHRpbmcgYW55IHN1Y2ggd2FycmFudHkgb3IgYWRkaXRpb25hbCBsaWFiaWxpdHkuCi0KLSAgIEVORCBPRiBURVJNUyBBTkQgQ09ORElUSU9OUwotCmRpZmYgLS1naXQgYS9hd3QvQW5kcm9pZC5tayBiL2F3dC9BbmRyb2lkLm1rCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjNzQ4MGY1Li4wMDAwMDAwCi0tLSBhL2F3dC9BbmRyb2lkLm1rCisrKyAvZGV2L251bGwKQEAgLTEsMzEgKzAsMCBAQAotIwotIyBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotIwotIyBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSMgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotIyBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSMKLSMgICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSMKLSMgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotIyBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotIyBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSMgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotIyBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSMKLQotTE9DQUxfUEFUSDo9ICQoY2FsbCBteS1kaXIpCi0KLWluY2x1ZGUgJChDTEVBUl9WQVJTKQotCi1MT0NBTF9TUkNfRklMRVMgOj0gJChjYWxsIGFsbC1zdWJkaXItamF2YS1maWxlcykKLQotTE9DQUxfSkFWQV9SRVNPVVJDRV9ESVJTIDo9IHJlc291cmNlcwotCi1MT0NBTF9KQVZBX0xJQlJBUklFUyA6PSBjb3JlIGZyYW1ld29yawotCi1MT0NBTF9NT0RVTEU6PSBhbmRyb2lkLmF3dAotCi1MT0NBTF9EWF9GTEFHUyA6PSAtLWNvcmUtbGlicmFyeQotCi1pbmNsdWRlICQoQlVJTERfSkFWQV9MSUJSQVJZKQpkaWZmIC0tZ2l0IGEvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9BbmRyb2lkR3JhcGhpY3MyRC5qYXZhIGIvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9BbmRyb2lkR3JhcGhpY3MyRC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5YThhZTAyLi4wMDAwMDAwCi0tLSBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvQW5kcm9pZEdyYXBoaWNzMkQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEzNTQgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGNvbS5hbmRyb2lkLmludGVybmFsLmF3dDsKLQotaW1wb3J0IGNvbS5hbmRyb2lkLmludGVybmFsLmF3dC5BbmRyb2lkR3JhcGhpY3NDb25maWd1cmF0aW9uOwotaW1wb3J0IGNvbS5hbmRyb2lkLmludGVybmFsLmdyYXBoaWNzLk5hdGl2ZVV0aWxzOwotCi1pbXBvcnQgamF2YS5hd3QuQWxwaGFDb21wb3NpdGU7Ci1pbXBvcnQgamF2YS5hd3QuQmFzaWNTdHJva2U7Ci1pbXBvcnQgamF2YS5hd3QuQ29sb3I7Ci1pbXBvcnQgamF2YS5hd3QuQ29tcG9zaXRlOwotaW1wb3J0IGphdmEuYXd0LkZvbnQ7Ci1pbXBvcnQgamF2YS5hd3QuRm9udE1ldHJpY3M7Ci1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3M7Ci1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3MyRDsKLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljc0NvbmZpZ3VyYXRpb247Ci1pbXBvcnQgamF2YS5hd3QuSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuUG9seWdvbjsKLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci1pbXBvcnQgamF2YS5hd3QuUmVuZGVyaW5nSGludHM7Ci1pbXBvcnQgamF2YS5hd3QuU2hhcGU7Ci1pbXBvcnQgamF2YS5hd3QuU3Ryb2tlOwotaW1wb3J0IGphdmEuYXd0LmZvbnQuRm9udFJlbmRlckNvbnRleHQ7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5HbHlwaFZlY3RvcjsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFyZWE7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5HZW5lcmFsUGF0aDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLk5vbmludmVydGlibGVUcmFuc2Zvcm1FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5QYXRoSXRlcmF0b3I7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQWZmaW5lVHJhbnNmb3JtT3A7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlT3A7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGF0YUJ1ZmZlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EaXJlY3RDb2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkltYWdlT2JzZXJ2ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmFzdGVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJlbmRlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Xcml0YWJsZVJhc3RlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5yZW5kZXJhYmxlLlJlbmRlcmFibGVJbWFnZTsKLWltcG9ydCBqYXZhLnRleHQuQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yOwotaW1wb3J0IGphdmEudXRpbC5NYXA7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLkltYWdlU3VyZmFjZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLk11bHRpUmVjdEFyZWE7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5TdXJmYWNlOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5BbmRyb2lkR2x5cGhWZWN0b3I7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250LkZvbnRNZXRyaWNzSW1wbDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmltYWdlLk9mZnNjcmVlbkltYWdlOwotCi1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5CaXRtYXA7Ci1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5DYW52YXM7Ci1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5NYXRyaXg7Ci1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5QYWludDsKLWltcG9ydCBhbmRyb2lkLmdyYXBoaWNzLlBhdGg7Ci0KLWltcG9ydCBhbmRyb2lkLmdyYXBoaWNzLlJlY3Q7Ci1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5SZWN0RjsKLWltcG9ydCBhbmRyb2lkLmdyYXBoaWNzLlJlZ2lvbjsKLWltcG9ydCBhbmRyb2lkLmdyYXBoaWNzLlR5cGVmYWNlOwotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuUGl4ZWxYb3JYZmVybW9kZTsKLWltcG9ydCBhbmRyb2lkLnZpZXcuRGlzcGxheTsKLWltcG9ydCBhbmRyb2lkLnZpZXcuV2luZG93TWFuYWdlcjsKLWltcG9ydCBhbmRyb2lkLmNvbnRlbnQuQ29udGV4dDsKLQotcHVibGljIGNsYXNzIEFuZHJvaWRHcmFwaGljczJEIGV4dGVuZHMgR3JhcGhpY3MyRCB7Ci0gICAgCi0gICAgcHJpdmF0ZSBpbnQgZGlzcGxheVdpZHRoLCBkaXNwbGF5SGVpZ2h0OwotCi0gICAgcHJvdGVjdGVkIFN1cmZhY2UgZHN0U3VyZiA9IG51bGw7Ci0gICAgcHJvdGVjdGVkIE11bHRpUmVjdEFyZWEgY2xpcCA9IG51bGw7Ci0KLSAgICBwcm90ZWN0ZWQgQ29tcG9zaXRlIGNvbXBvc2l0ZSA9IEFscGhhQ29tcG9zaXRlLlNyY092ZXI7Ci0gICAgcHJvdGVjdGVkIEFmZmluZVRyYW5zZm9ybSB0cmFuc2Zvcm0gPSBuZXcgQWZmaW5lVHJhbnNmb3JtKCk7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBBbmRyb2lkR3JhcGhpY3MyRCBtQWc7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgQ2FudmFzIG1DOwotCi0gICAgLy8gQW5kcm9pZCBQYWludAotICAgIHB1YmxpYyBzdGF0aWMgUGFpbnQgbVA7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBqYXZhLmF3dC5Gb250IG1GbnQ7Ci0KLSAgICAvLyBDYWNoZWQgTWF0cml4Ci0gICAgcHVibGljIHN0YXRpYyBNYXRyaXggbU07Ci0gICAgcHJpdmF0ZSBzdGF0aWMgRm9udE1ldHJpY3MgbUZtOwotICAgIHByaXZhdGUgc3RhdGljIFJlbmRlcmluZ0hpbnRzIG1SaDsKLSAgICBwcml2YXRlIHN0YXRpYyBDb2xvciBtQmM7Ci0KLSAgICBwcml2YXRlIEFyZWEgbUN1cnJDbGlwOwotICAgIAotICAgIHB1YmxpYyBmaW5hbCBzdGF0aWMgZG91YmxlIFJBRF8zNjAgPSBNYXRoLlBJIC8gMTgwICogMzYwOwotICAgIAotICAgIC8vIEltYWdlIGRyYXdpbmcKLSAgICBwcml2YXRlIEFuZHJvaWRKYXZhQmxpdHRlciBibGl0dGVyOwotICAgIHByaXZhdGUgRGlyZWN0Q29sb3JNb2RlbCBjbTsKLSAgICBwcml2YXRlIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc207Ci0gICAgcHJpdmF0ZSBXcml0YWJsZVJhc3RlciB3cjsKLQotCi0gICAgcHVibGljIHN0YXRpYyBBbmRyb2lkR3JhcGhpY3MyRCBnZXRJbnN0YW5jZSgpIHsKLSAgICAgICAgaWYgKG1BZyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiQW5kcm9pZEdyYXBoaWNzMkQgbm90IGluc3RhbnRpYXRlZCEiKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbUFnOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgQW5kcm9pZEdyYXBoaWNzMkQgZ2V0SW5zdGFuY2UoQ29udGV4dCBjdHgsIENhbnZhcyBjLCBQYWludCBwKSB7Ci0gICAgICAgIGlmIChjID09IG51bGwgfHwgY3R4ID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKAotICAgICAgICAgICAgICAgICAgICAiSWxsZWdhbCBhcmd1bWVudCwgQ2FudmFzIGNhbm5vdCBiZSBudWxsISIpOwotICAgICAgICB9Ci0gICAgICAgIG1BZyA9IG5ldyBBbmRyb2lkR3JhcGhpY3MyRChjdHgsIGMsIHApOwotICAgICAgICByZXR1cm4gbUFnOwotICAgIH0KLQotICAgIHByaXZhdGUgQW5kcm9pZEdyYXBoaWNzMkQoQ29udGV4dCBjdHgsIENhbnZhcyBjLCBQYWludCBwKSB7Ci0gICAgICAgIHN1cGVyKCk7Ci0gICAgICAgIG1DID0gYzsKLSAgICAgICAgbVAgPSBwOwotICAgICAgICBtTSA9IG5ldyBNYXRyaXgoKTsKLSAgICAgICAgbU0ucmVzZXQoKTsKLSAgICAgICAgbU0gPSBtQy5nZXRNYXRyaXgoKTsKLSAgICAgICAgUmVjdCByID0gbUMuZ2V0Q2xpcEJvdW5kcygpOwotICAgICAgICBpbnQgY2xbXSA9IHstMSwgci50b3AsIHIubGVmdCwgLTIsIHIudG9wLCByLnJpZ2h0LCAtMiwgci5ib3R0b20sIHIucmlnaHQsIC0yLCByLmJvdHRvbSwgci5sZWZ0fTsKLSAgICAgICAgbUN1cnJDbGlwID0gbmV3IEFyZWEoY3JlYXRlU2hhcGUoY2wpKTsKLSAgICAgICAgaWYoY3R4ICE9IG51bGwpIHsKLSAgICAgICAgICAgIFdpbmRvd01hbmFnZXIgd20gPSAoV2luZG93TWFuYWdlciljdHguZ2V0U3lzdGVtU2VydmljZShDb250ZXh0LldJTkRPV19TRVJWSUNFKTsKLSAgICAgICAgICAgIERpc3BsYXkgZCA9IHdtLmdldERlZmF1bHREaXNwbGF5KCk7Ci0gICAgICAgICAgICBkaXNwbGF5V2lkdGggPSBkLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBkaXNwbGF5SGVpZ2h0ID0gZC5nZXRIZWlnaHQoKTsKLSAgICAgICAgfQotICAgICAgICBibGl0dGVyID0gbmV3IEFuZHJvaWRKYXZhQmxpdHRlcihjKTsKLSAgICAgICAgY20gPSBuZXcgRGlyZWN0Q29sb3JNb2RlbCgzMiwgMHhmZjAwMDAsIDB4ZmYwMCwgMHhmZiwgMHhmZjAwMDAwMCk7Ci0gICAgICAgIHNtID0gbmV3IFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwoCi0gICAgICAgICAgICAgICAgRGF0YUJ1ZmZlci5UWVBFX0lOVCwgZGlzcGxheVdpZHRoLCBkaXNwbGF5SGVpZ2h0LCBjbS5nZXRNYXNrcygpKTsKLSAgICAgICAgd3IgPSBSYXN0ZXIuY3JlYXRlV3JpdGFibGVSYXN0ZXIoc20sIG51bGwpOwotICAgICAgICBkc3RTdXJmID0gbmV3IEltYWdlU3VyZmFjZShjbSwgd3IpOyAgICAgICAKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBhZGRSZW5kZXJpbmdIaW50cyhNYXA8PywgPz4gaGludHMpIHsKLSAgICAgICAgaWYgKG1SaCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBtUmggPSAoUmVuZGVyaW5nSGludHMpIGhpbnRzOwotICAgICAgICB9Ci0gICAgICAgIG1SaC5hZGQoKFJlbmRlcmluZ0hpbnRzKSBoaW50cyk7Ci0gICAgfQotCi0gICAgcHVibGljIGZsb2F0W10gZ2V0TWF0cml4KCkgewotICAgICAgICBmbG9hdFtdIGYgPSBuZXcgZmxvYXRbOV07Ci0gICAgICAgIG1DLmdldE1hdHJpeCgpLmdldFZhbHVlcyhmKTsKLSAgICAgICAgcmV0dXJuIGY7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogCi0gICAgICogQHJldHVybiBhIE1hdHJpeCBpbiBBbmRyb2lkIGZvcm1hdAotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldEludmVyc2VNYXRyaXgoKSB7Ci0gICAgICAgIEFmZmluZVRyYW5zZm9ybSBhZiA9IG5ldyBBZmZpbmVUcmFuc2Zvcm0oY3JlYXRlQVdUTWF0cml4KGdldE1hdHJpeCgpKSk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBhZiA9IGFmLmNyZWF0ZUludmVyc2UoKTsKLSAgICAgICAgfSBjYXRjaCAoTm9uaW52ZXJ0aWJsZVRyYW5zZm9ybUV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGNyZWF0ZU1hdHJpeChhZik7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBQYXRoIGdldFBhdGgoU2hhcGUgcykgewotICAgICAgICBQYXRoIHBhdGggPSBuZXcgUGF0aCgpOwotICAgICAgICBQYXRoSXRlcmF0b3IgcGkgPSBzLmdldFBhdGhJdGVyYXRvcihudWxsKTsKLSAgICAgICAgd2hpbGUgKHBpLmlzRG9uZSgpID09IGZhbHNlKSB7Ci0gICAgICAgICAgICBnZXRDdXJyZW50U2VnbWVudChwaSwgcGF0aCk7Ci0gICAgICAgICAgICBwaS5uZXh0KCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHBhdGg7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIGdldEN1cnJlbnRTZWdtZW50KFBhdGhJdGVyYXRvciBwaSwgUGF0aCBwYXRoKSB7Ci0gICAgICAgIGZsb2F0W10gY29vcmRpbmF0ZXMgPSBuZXcgZmxvYXRbNl07Ci0gICAgICAgIGludCB0eXBlID0gcGkuY3VycmVudFNlZ21lbnQoY29vcmRpbmF0ZXMpOwotICAgICAgICBzd2l0Y2ggKHR5cGUpIHsKLSAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX01PVkVUTzoKLSAgICAgICAgICAgIHBhdGgubW92ZVRvKGNvb3JkaW5hdGVzWzBdLCBjb29yZGluYXRlc1sxXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0xJTkVUTzoKLSAgICAgICAgICAgIHBhdGgubGluZVRvKGNvb3JkaW5hdGVzWzBdLCBjb29yZGluYXRlc1sxXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX1FVQURUTzoKLSAgICAgICAgICAgIHBhdGgucXVhZFRvKGNvb3JkaW5hdGVzWzBdLCBjb29yZGluYXRlc1sxXSwgY29vcmRpbmF0ZXNbMl0sCi0gICAgICAgICAgICAgICAgICAgIGNvb3JkaW5hdGVzWzNdKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfQ1VCSUNUTzoKLSAgICAgICAgICAgIHBhdGguY3ViaWNUbyhjb29yZGluYXRlc1swXSwgY29vcmRpbmF0ZXNbMV0sIGNvb3JkaW5hdGVzWzJdLAotICAgICAgICAgICAgICAgICAgICBjb29yZGluYXRlc1szXSwgY29vcmRpbmF0ZXNbNF0sIGNvb3JkaW5hdGVzWzVdKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfQ0xPU0U6Ci0gICAgICAgICAgICBwYXRoLmNsb3NlKCk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIHByaXZhdGUgU2hhcGUgY3JlYXRlU2hhcGUoaW50W10gYXJyKSB7Ci0gICAgICAgIFNoYXBlIHMgPSBuZXcgR2VuZXJhbFBhdGgoKTsKLSAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IGFyci5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgaW50IHR5cGUgPSBhcnJbaV07ICAgIAotICAgICAgICAgICAgc3dpdGNoICh0eXBlKSB7Ci0gICAgICAgICAgICBjYXNlIC0xOgotICAgICAgICAgICAgICAgIC8vTU9WRVRPCi0gICAgICAgICAgICAgICAgKChHZW5lcmFsUGF0aClzKS5tb3ZlVG8oYXJyWysraV0sIGFyclsrK2ldKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgLTI6Ci0gICAgICAgICAgICAgICAgLy9MSU5FVE8KLSAgICAgICAgICAgICAgICAoKEdlbmVyYWxQYXRoKXMpLmxpbmVUbyhhcnJbKytpXSwgYXJyWysraV0pOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSAtMzoKLSAgICAgICAgICAgICAgICAvL1FVQURUTwotICAgICAgICAgICAgICAgICgoR2VuZXJhbFBhdGgpcykucXVhZFRvKGFyclsrK2ldLCBhcnJbKytpXSwgYXJyWysraV0sCi0gICAgICAgICAgICAgICAgICAgICAgICBhcnJbKytpXSk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIC00OgotICAgICAgICAgICAgICAgIC8vQ1VCSUNUTwotICAgICAgICAgICAgICAgICgoR2VuZXJhbFBhdGgpcykuY3VydmVUbyhhcnJbKytpXSwgYXJyWysraV0sIGFyclsrK2ldLAotICAgICAgICAgICAgICAgICAgICAgICAgYXJyWysraV0sIGFyclsrK2ldLCBhcnJbKytpXSk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIC01OgotICAgICAgICAgICAgICAgIC8vQ0xPU0UKLSAgICAgICAgICAgICAgICByZXR1cm4gczsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHM7Ci0gICAgfQotICAgIC8qCi0gICAgcHVibGljIGludFtdIGdldFBpeGVscygpIHsKLSAgICAgICAgcmV0dXJuIG1DLmdldFBpeGVscygpOwotICAgIH0qLwotCi0gICAgcHVibGljIHN0YXRpYyBmbG9hdCBnZXRSYWRpYW4oZmxvYXQgZGVncmVlKSB7Ci0gICAgICAgIHJldHVybiAoZmxvYXQpICgoTWF0aC5QSSAvIDE4MCkgKiBkZWdyZWUpOwotICAgIH0KLSAgICAKLSAgICBwcml2YXRlIFNoYXBlIGdldFNoYXBlKCkgewotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZsb2F0IGdldERlZ3JlZShmbG9hdCByYWRpYW4pIHsKLSAgICAgICAgcmV0dXJuIChmbG9hdCkgKCgxODAgLyBNYXRoLlBJKSAqIHJhZGlhbik7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBEZWdyZWUgaW4gcmFkaWFuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmbG9hdCBnZXRFbGxpcHNpc1goZmxvYXQgZGVncmVlLCBmbG9hdCBwcmluY0F4aXMpIHsKLSAgICAgICAgcmV0dXJuIChmbG9hdCkgTWF0aC5jb3MoZGVncmVlKSAqIHByaW5jQXhpczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZsb2F0IGdldEVsbGlwc2lzWShmbG9hdCBkZWdyZWUsIGZsb2F0IGNvbkF4aXMpIHsKLSAgICAgICAgcmV0dXJuIChmbG9hdCkgTWF0aC5zaW4oZGVncmVlKSAqIGNvbkF4aXM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgY2xpcChTaGFwZSBzKSB7Ci0gICAgICAgIG1DLmNsaXBQYXRoKGdldFBhdGgocykpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldENhbnZhcyhDYW52YXMgYykgewotICAgICAgICBtQyA9IGM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhdyhTaGFwZSBzKSB7Ci0gICAgICAgIGlmIChtUCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBtUCA9IG5ldyBQYWludCgpOwotICAgICAgICB9Ci0gICAgICAgIFBhaW50LlN0eWxlIHRtcCA9IG1QLmdldFN0eWxlKCk7Ci0gICAgICAgIG1QLnNldFN0eWxlKFBhaW50LlN0eWxlLlNUUk9LRSk7Ci0gICAgICAgIG1DLmRyYXdQYXRoKGdldFBhdGgocyksIG1QKTsKLSAgICAgICAgbVAuc2V0U3R5bGUodG1wKTsKLSAgICB9Ci0vKgotICAgIHByaXZhdGUgQXJyYXlMaXN0IGdldFNlZ21lbnRzKFNoYXBlIHMpIHsKLSAgICAgICAgQXJyYXlMaXN0IGFyciA9IG5ldyBBcnJheUxpc3QoKTsKLSAgICAgICAgUGF0aEl0ZXJhdG9yIHBpID0gcy5nZXRQYXRoSXRlcmF0b3IobnVsbCk7Ci0gICAgICAgIHdoaWxlIChwaS5pc0RvbmUoKSA9PSBmYWxzZSkgewotICAgICAgICAgICAgZ2V0Q3VycmVudFNlZ21lbnQocGksIGFycik7Ci0gICAgICAgICAgICBwaS5uZXh0KCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGFycjsKLSAgICB9Ci0KLSAgICBwcml2YXRlIHZvaWQgZ2V0Q3VycmVudFNlZ21lbnQoUGF0aEl0ZXJhdG9yIHBpLCBBcnJheUxpc3QgYXJyKSB7Ci0gICAgICAgIGZsb2F0W10gY29vcmRpbmF0ZXMgPSBuZXcgZmxvYXRbNl07Ci0gICAgICAgIGludCB0eXBlID0gcGkuY3VycmVudFNlZ21lbnQoY29vcmRpbmF0ZXMpOwotICAgICAgICBzd2l0Y2ggKHR5cGUpIHsKLSAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX01PVkVUTzoKLSAgICAgICAgICAgIGFyci5hZGQobmV3IEludGVnZXIoLTEpKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfTElORVRPOgotICAgICAgICAgICAgYXJyLmFkZChuZXcgSW50ZWdlcigtMikpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19RVUFEVE86Ci0gICAgICAgICAgICBhcnIuYWRkKG5ldyBJbnRlZ2VyKC0zKSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0NVQklDVE86Ci0gICAgICAgICAgICBhcnIuYWRkKG5ldyBJbnRlZ2VyKC00KSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0NMT1NFOgotICAgICAgICAgICAgYXJyLmFkZChuZXcgSW50ZWdlcigtNSkpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgIH0KLSovCi0gICAgLyoKLSAgICAgKiBDb252ZW5pZW5jZSBtZXRob2QsIG5vdCBzdGFuZGFyZCBBV1QKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBkcmF3KFBhdGggcykgewotICAgICAgICBpZiAobVAgPT0gbnVsbCkgewotICAgICAgICAgICAgbVAgPSBuZXcgUGFpbnQoKTsKLSAgICAgICAgfQotICAgICAgICBQYWludC5TdHlsZSB0bXAgPSBtUC5nZXRTdHlsZSgpOwotICAgICAgICBtUC5zZXRTdHlsZShQYWludC5TdHlsZS5TVFJPS0UpOwotICAgICAgICBzLnRyYW5zZm9ybShtTSk7Ci0gICAgICAgIG1DLmRyYXdQYXRoKHMsIG1QKTsKLSAgICAgICAgbVAuc2V0U3R5bGUodG1wKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3R2x5cGhWZWN0b3IoR2x5cGhWZWN0b3IgZywgZmxvYXQgeCwgZmxvYXQgeSkgewotICAgICAgICAvLyBUT0RPIGRyYXcgYXQgeCwgeQotICAgICAgICAvLyBkcmF3KGcuZ2V0T3V0bGluZSgpKTsKLSAgICAgICAgLyoKLSAgICAgICAgTWF0cml4IG1hdHJpeCA9IG5ldyBNYXRyaXgoKTsKLSAgICAgICAgbWF0cml4LnNldFRyYW5zbGF0ZSh4LCB5KTsKLSAgICAgICAgUGF0aCBwdGggPSBnZXRQYXRoKGcuZ2V0T3V0bGluZSgpKTsKLSAgICAgICAgcHRoLnRyYW5zZm9ybShtYXRyaXgpOwotICAgICAgICBkcmF3KHB0aCk7Ci0gICAgICAgICovCi0gICAgICAgIFBhdGggcGF0aCA9IG5ldyBQYXRoKCk7Ci0gICAgICAgIGNoYXJbXSBjID0gKChBbmRyb2lkR2x5cGhWZWN0b3IpZykuZ2V0R2x5cGhzKCk7Ci0gICAgICAgIG1QLmdldFRleHRQYXRoKGMsIDAsIGMubGVuZ3RoLCB4LCB5LCBwYXRoKTsKLSAgICAgICAgbUMuZHJhd1BhdGgocGF0aCwgbVApOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRyYXdSZW5kZXJhYmxlSW1hZ2UoUmVuZGVyYWJsZUltYWdlIGltZywgQWZmaW5lVHJhbnNmb3JtIHhmb3JtKSB7Ci0gICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQhIik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd1JlbmRlcmVkSW1hZ2UoUmVuZGVyZWRJbWFnZSBpbWcsIEFmZmluZVRyYW5zZm9ybSB4Zm9ybSkgewotICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRyYXdTdHJpbmcoQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIGl0ZXJhdG9yLCBmbG9hdCB4LAotICAgICAgICAgICAgZmxvYXQgeSkgewotICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIG5vdCBzdXBwb3J0ZWQhIik7Ci0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3U3RyaW5nKEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvciBpdGVyYXRvciwgaW50IHgsIGludCB5KSB7Ci0gICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3Igbm90IHN1cHBvcnRlZCEiKTsKLQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRyYXdTdHJpbmcoU3RyaW5nIHMsIGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgICAgIGlmIChtUCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgbVAgPSBuZXcgUGFpbnQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIFBhaW50LlN0eWxlIHRtcCA9IG1QLmdldFN0eWxlKCk7Ci0KLSAgICAgICAgICAgIG1QLnNldFN0eWxlKFBhaW50LlN0eWxlLkZJTEwpOwotICAgICAgICAgICAgUGF0aCBwdGggPSBuZXcgUGF0aCgpOwotICAgICAgICAgICAgbVAuZ2V0VGV4dFBhdGgocywgMCwgcy5sZW5ndGgoKSwgeCwgeSwgcHRoKTsKLSAgICAgICAgICAgIG1DLmRyYXdQYXRoKHB0aCwgbVApOwotICAgICAgICAgICAgbVAuc2V0U3R5bGUodG1wKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3U3RyaW5nKFN0cmluZyBzdHIsIGludCB4LCBpbnQgeSkgewotICAgICAgICAgICAgaWYgKG1QID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBtUCA9IG5ldyBQYWludCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgUGFpbnQuU3R5bGUgdG1wID0gbVAuZ2V0U3R5bGUoKTsKLSAgICAgICAgICAgIG1QLnNldFN0cm9rZVdpZHRoKDApOwotCi0gICAgICAgICAgICBtQy5kcmF3VGV4dChzdHIudG9DaGFyQXJyYXkoKSwgMCwgc3RyLnRvQ2hhckFycmF5KCkubGVuZ3RoLCB4LCB5LAotICAgICAgICAgICAgICAgICAgICBtUCk7Ci0gICAgICAgICAgICBtUC5zZXRTdHlsZSh0bXApOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGZpbGwoU2hhcGUgcykgewotICAgICAgICAgICAgaWYgKG1QID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBtUCA9IG5ldyBQYWludCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgUGFpbnQuU3R5bGUgdG1wID0gbVAuZ2V0U3R5bGUoKTsKLSAgICAgICAgICAgIG1QLnNldFN0eWxlKFBhaW50LlN0eWxlLkZJTEwpOwotICAgICAgICAgICAgbUMuZHJhd1BhdGgoZ2V0UGF0aChzKSwgbVApOwotICAgICAgICAgICAgbVAuc2V0U3R5bGUodG1wKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgQ29sb3IgZ2V0QmFja2dyb3VuZCgpIHsKLSAgICAgICAgcmV0dXJuIG1CYzsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgQ29tcG9zaXRlIGdldENvbXBvc2l0ZSgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIkNvbXBvc2l0ZSBub3QgaW1wbGVtZW50ZWQhIik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBnZXREZXZpY2VDb25maWd1cmF0aW9uKCkgewotICAgICAgICByZXR1cm4gbmV3IEFuZHJvaWRHcmFwaGljc0NvbmZpZ3VyYXRpb24oKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgRm9udFJlbmRlckNvbnRleHQgZ2V0Rm9udFJlbmRlckNvbnRleHQoKSB7Ci0gICAgICAgIHJldHVybiBuZXcgRm9udFJlbmRlckNvbnRleHQoZ2V0VHJhbnNmb3JtKCksIG1QLmlzQW50aUFsaWFzKCksIHRydWUpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBqYXZhLmF3dC5QYWludCBnZXRQYWludCgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIkFXVCBQYWludCBub3QgaW1wbGVtZW50ZWQgaW4gQW5kcm9pZCEiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIENhbnZhcyBnZXRBbmRyb2lkQ2FudmFzKCkgewotICAgICAgICByZXR1cm4gbUM7Ci0gICAgfQotICAgIAotICAgIHB1YmxpYyBzdGF0aWMgUGFpbnQgZ2V0QW5kcm9pZFBhaW50KCkgewotICAgICAgICByZXR1cm4gbVA7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFJlbmRlcmluZ0hpbnRzIGdldFJlbmRlcmluZ0hpbnRzKCkgewotICAgICAgICByZXR1cm4gbVJoOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJva2UgZ2V0U3Ryb2tlKCkgewotICAgICAgICBpZiAobVAgIT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBCYXNpY1N0cm9rZShtUC5nZXRTdHJva2VXaWR0aCgpLCBtUC5nZXRTdHJva2VDYXAoKQotICAgICAgICAgICAgICAgICAgICAub3JkaW5hbCgpLCBtUC5nZXRTdHJva2VKb2luKCkub3JkaW5hbCgpKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgQWZmaW5lVHJhbnNmb3JtIGdldFRyYW5zZm9ybSgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBBZmZpbmVUcmFuc2Zvcm0oY3JlYXRlQVdUTWF0cml4KGdldE1hdHJpeCgpKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaGl0KFJlY3RhbmdsZSByZWN0LCBTaGFwZSBzLCBib29sZWFuIG9uU3Ryb2tlKSB7Ci0gICAgICAgIC8vID8/P0FXVCBUT0RPIGNoZWNrIGlmIG9uIHN0cm9rZQotICAgICAgICByZXR1cm4gcy5pbnRlcnNlY3RzKHJlY3QuZ2V0WCgpLCByZWN0LmdldFkoKSwgcmVjdC5nZXRXaWR0aCgpLCByZWN0Ci0gICAgICAgICAgICAgICAgLmdldEhlaWdodCgpKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCByb3RhdGUoZG91YmxlIHRoZXRhKSB7Ci0gICAgICAgIG1NLnByZVJvdGF0ZSgoZmxvYXQpIEFuZHJvaWRHcmFwaGljczJECi0gICAgICAgICAgICAgICAgLmdldERlZ3JlZSgoZmxvYXQpIChSQURfMzYwIC0gdGhldGEpKSk7Ci0gICAgICAgIG1DLmNvbmNhdChtTSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgcm90YXRlKGRvdWJsZSB0aGV0YSwgZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0gICAgICAgIG1NLnByZVJvdGF0ZSgoZmxvYXQpIEFuZHJvaWRHcmFwaGljczJELmdldERlZ3JlZSgoZmxvYXQpIHRoZXRhKSwKLSAgICAgICAgICAgICAgICAoZmxvYXQpIHgsIChmbG9hdCkgeSk7Ci0gICAgICAgIG1DLmNvbmNhdChtTSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2NhbGUoZG91YmxlIHN4LCBkb3VibGUgc3kpIHsKLSAgICAgICAgbU0uc2V0U2NhbGUoKGZsb2F0KSBzeCwgKGZsb2F0KSBzeSk7Ci0gICAgICAgIG1DLmNvbmNhdChtTSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0QmFja2dyb3VuZChDb2xvciBjb2xvcikgewotICAgICAgICBtQmMgPSBjb2xvcjsKLSAgICAgICAgbUMuY2xpcFJlY3QobmV3IFJlY3QoMCwgMCwgbUMuZ2V0V2lkdGgoKSwgbUMuZ2V0SGVpZ2h0KCkpKTsKLSAgICAgICAgLy8gVE9ETyBkb24ndCBsaW1pdCB0byBjdXJyZW50IGNsaXAKLSAgICAgICAgbUMuZHJhd0FSR0IoY29sb3IuZ2V0QWxwaGEoKSwgY29sb3IuZ2V0UmVkKCksIGNvbG9yLmdldEdyZWVuKCksIGNvbG9yCi0gICAgICAgICAgICAgICAgLmdldEJsdWUoKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0Q29tcG9zaXRlKENvbXBvc2l0ZSBjb21wKSB7Ci0gICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJDb21wb3NpdGUgbm90IGltcGxlbWVudGVkISIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldFNwYWludChQYWludCBwYWludCkgewotICAgICAgICBtUCA9IHBhaW50OwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFBhaW50KGphdmEuYXd0LlBhaW50IHBhaW50KSB7Ci0gICAgICAgIHNldENvbG9yKChDb2xvcilwYWludCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBnZXRSZW5kZXJpbmdIaW50KFJlbmRlcmluZ0hpbnRzLktleSBrZXkpIHsKLSAgICAgICAgaWYgKG1SaCA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbVJoLmdldChrZXkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFJlbmRlcmluZ0hpbnQoUmVuZGVyaW5nSGludHMuS2V5IGhpbnRLZXksIE9iamVjdCBoaW50VmFsdWUpIHsKLSAgICAgICAgaWYgKG1SaCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBtUmggPSBuZXcgUmVuZGVyaW5nSGludHMoaGludEtleSwgaGludFZhbHVlKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG1SaC5wdXQoaGludEtleSwgaGludFZhbHVlKTsKLSAgICAgICAgfQotICAgICAgICBhcHBseUhpbnRzKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UmVuZGVyaW5nSGludHMoTWFwPD8sID8+IGhpbnRzKSB7Ci0gICAgICAgIG1SaCA9IChSZW5kZXJpbmdIaW50cykgaGludHM7Ci0gICAgICAgIGFwcGx5SGludHMoKTsKLSAgICB9Ci0KLSAgICBwcml2YXRlIHZvaWQgYXBwbHlIaW50cygpIHsKLSAgICAgICAgT2JqZWN0IG87Ci0KLSAgICAgICAgLy8gVE9ETyBkbyBzb21ldGhpbmcgbGlrZSB0aGlzOgotICAgICAgICAvKgotICAgICAgICAgKiBTZXQgcyA9IG1SaC5rZXlTZXQoKTsgSXRlcmF0b3IgaXQgPSBzLml0ZXJhdG9yKCk7IHdoaWxlKGl0Lmhhc05leHQoKSkgewotICAgICAgICAgKiBvID0gaXQubmV4dCgpOyB9Ci0gICAgICAgICAqLwotCi0gICAgICAgIC8vIC8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLwotICAgICAgICAvLyBub3Qgc3VwcG9ydGVkIGluIHNraWEKLSAgICAgICAgLyoKLSAgICAgICAgICogbyA9IG1SaC5nZXQoUmVuZGVyaW5nSGludHMuS0VZX0FMUEhBX0lOVEVSUE9MQVRJT04pOyBpZgotICAgICAgICAgKiAoby5lcXVhbHMoUmVuZGVyaW5nSGludHMuVkFMVUVfQUxQSEFfSU5URVJQT0xBVElPTl9ERUZBVUxUKSkgeyB9IGVsc2UKLSAgICAgICAgICogaWYgKG8uZXF1YWxzKFJlbmRlcmluZ0hpbnRzLlZBTFVFX0FMUEhBX0lOVEVSUE9MQVRJT05fUVVBTElUWSkpIHsgfQotICAgICAgICAgKiBlbHNlIGlmIChvLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9BTFBIQV9JTlRFUlBPTEFUSU9OX1NQRUVEKSkgeyB9Ci0gICAgICAgICAqIAotICAgICAgICAgKiBvID0gbVJoLmdldChSZW5kZXJpbmdIaW50cy5LRVlfQ09MT1JfUkVOREVSSU5HKTsgaWYKLSAgICAgICAgICogKG8uZXF1YWxzKFJlbmRlcmluZ0hpbnRzLlZBTFVFX0NPTE9SX1JFTkRFUl9ERUZBVUxUKSkgeyB9IGVsc2UgaWYKLSAgICAgICAgICogKG8uZXF1YWxzKFJlbmRlcmluZ0hpbnRzLlZBTFVFX0NPTE9SX1JFTkRFUl9RVUFMSVRZKSkgeyB9IGVsc2UgaWYKLSAgICAgICAgICogKG8uZXF1YWxzKFJlbmRlcmluZ0hpbnRzLlZBTFVFX0NPTE9SX1JFTkRFUl9TUEVFRCkpIHsgfQotICAgICAgICAgKiAKLSAgICAgICAgICogbyA9IG1SaC5nZXQoUmVuZGVyaW5nSGludHMuS0VZX0RJVEhFUklORyk7IGlmCi0gICAgICAgICAqIChvLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9ESVRIRVJfREVGQVVMVCkpIHsgfSBlbHNlIGlmCi0gICAgICAgICAqIChvLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9ESVRIRVJfRElTQUJMRSkpIHsgfSBlbHNlIGlmCi0gICAgICAgICAqIChvLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9ESVRIRVJfRU5BQkxFKSkgeyB9Ci0gICAgICAgICAqIAotICAgICAgICAgKiBvID0gbVJoLmdldChSZW5kZXJpbmdIaW50cy5LRVlfRlJBQ1RJT05BTE1FVFJJQ1MpOyBpZgotICAgICAgICAgKiAoby5lcXVhbHMoUmVuZGVyaW5nSGludHMuVkFMVUVfRlJBQ1RJT05BTE1FVFJJQ1NfREVGQVVMVCkpIHsgfSBlbHNlCi0gICAgICAgICAqIGlmIChvLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9GUkFDVElPTkFMTUVUUklDU19PRkYpKSB7IH0gZWxzZSBpZgotICAgICAgICAgKiAoby5lcXVhbHMoUmVuZGVyaW5nSGludHMuVkFMVUVfRlJBQ1RJT05BTE1FVFJJQ1NfT04pKSB7IH0KLSAgICAgICAgICogCi0gICAgICAgICAqIG8gPSBtUmguZ2V0KFJlbmRlcmluZ0hpbnRzLktFWV9JTlRFUlBPTEFUSU9OKTsgaWYKLSAgICAgICAgICogKG8uZXF1YWxzKFJlbmRlcmluZ0hpbnRzLlZBTFVFX0lOVEVSUE9MQVRJT05fQklDVUJJQykpIHsgfSBlbHNlIGlmCi0gICAgICAgICAqIChvLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9JTlRFUlBPTEFUSU9OX0JJTElORUFSKSkgeyB9IGVsc2UgaWYKLSAgICAgICAgICogKG8gLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9JTlRFUlBPTEFUSU9OX05FQVJFU1RfTkVJR0hCT1IpKSB7IH0KLSAgICAgICAgICogCi0gICAgICAgICAqIG8gPSBtUmguZ2V0KFJlbmRlcmluZ0hpbnRzLktFWV9SRU5ERVJJTkcpOyBpZgotICAgICAgICAgKiAoby5lcXVhbHMoUmVuZGVyaW5nSGludHMuVkFMVUVfUkVOREVSX0RFRkFVTFQpKSB7IH0gZWxzZSBpZgotICAgICAgICAgKiAoby5lcXVhbHMoUmVuZGVyaW5nSGludHMuVkFMVUVfUkVOREVSX1FVQUxJVFkpKSB7IH0gZWxzZSBpZgotICAgICAgICAgKiAoby5lcXVhbHMoUmVuZGVyaW5nSGludHMuVkFMVUVfUkVOREVSX1NQRUVEKSkgeyB9Ci0gICAgICAgICAqIAotICAgICAgICAgKiBvID0gbVJoLmdldChSZW5kZXJpbmdIaW50cy5LRVlfU1RST0tFX0NPTlRST0wpOyBpZgotICAgICAgICAgKiAoby5lcXVhbHMoUmVuZGVyaW5nSGludHMuVkFMVUVfU1RST0tFX0RFRkFVTFQpKSB7IH0gZWxzZSBpZgotICAgICAgICAgKiAoby5lcXVhbHMoUmVuZGVyaW5nSGludHMuVkFMVUVfU1RST0tFX05PUk1BTElaRSkpIHsgfSBlbHNlIGlmCi0gICAgICAgICAqIChvLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9TVFJPS0VfUFVSRSkpIHsgfQotICAgICAgICAgKi8KLQotICAgICAgICBvID0gbVJoLmdldChSZW5kZXJpbmdIaW50cy5LRVlfQU5USUFMSUFTSU5HKTsKLSAgICAgICAgaWYgKG8gIT0gbnVsbCkgewotICAgICAgICAgICAgaWYgKG8uZXF1YWxzKFJlbmRlcmluZ0hpbnRzLlZBTFVFX0FOVElBTElBU19ERUZBVUxUKSkgewotICAgICAgICAgICAgICAgIG1QLnNldEFudGlBbGlhcyhmYWxzZSk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKG8uZXF1YWxzKFJlbmRlcmluZ0hpbnRzLlZBTFVFX0FOVElBTElBU19PRkYpKSB7Ci0gICAgICAgICAgICAgICAgbVAuc2V0QW50aUFsaWFzKGZhbHNlKTsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoby5lcXVhbHMoUmVuZGVyaW5nSGludHMuVkFMVUVfQU5USUFMSUFTX09OKSkgewotICAgICAgICAgICAgICAgIG1QLnNldEFudGlBbGlhcyh0cnVlKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIG8gPSBtUmguZ2V0KFJlbmRlcmluZ0hpbnRzLktFWV9URVhUX0FOVElBTElBU0lORyk7Ci0gICAgICAgIGlmIChvICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChvLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9URVhUX0FOVElBTElBU19ERUZBVUxUKSkgewotICAgICAgICAgICAgICAgIG1QLnNldEFudGlBbGlhcyhmYWxzZSk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKG8uZXF1YWxzKFJlbmRlcmluZ0hpbnRzLlZBTFVFX1RFWFRfQU5USUFMSUFTX09GRikpIHsKLSAgICAgICAgICAgICAgICBtUC5zZXRBbnRpQWxpYXMoZmFsc2UpOwotICAgICAgICAgICAgfSBlbHNlIGlmIChvLmVxdWFscyhSZW5kZXJpbmdIaW50cy5WQUxVRV9URVhUX0FOVElBTElBU19PTikpIHsKLSAgICAgICAgICAgICAgICBtUC5zZXRBbnRpQWxpYXModHJ1ZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTdHJva2UoU3Ryb2tlIHMpIHsKLSAgICAgICAgaWYgKG1QID09IG51bGwpIHsKLSAgICAgICAgICAgIG1QID0gbmV3IFBhaW50KCk7Ci0gICAgICAgIH0KLSAgICAgICAgQmFzaWNTdHJva2UgYnMgPSAoQmFzaWNTdHJva2UpIHM7Ci0gICAgICAgIG1QLnNldFN0eWxlKFBhaW50LlN0eWxlLlNUUk9LRSk7Ci0gICAgICAgIG1QLnNldFN0cm9rZVdpZHRoKGJzLmdldExpbmVXaWR0aCgpKTsKLQotICAgICAgICBpbnQgY2FwID0gYnMuZ2V0RW5kQ2FwKCk7Ci0gICAgICAgIGlmIChjYXAgPT0gMCkgewotICAgICAgICAgICAgbVAuc2V0U3Ryb2tlQ2FwKFBhaW50LkNhcC5CVVRUKTsKLSAgICAgICAgfSBlbHNlIGlmIChjYXAgPT0gMSkgewotICAgICAgICAgICAgbVAuc2V0U3Ryb2tlQ2FwKFBhaW50LkNhcC5ST1VORCk7Ci0gICAgICAgIH0gZWxzZSBpZiAoY2FwID09IDIpIHsKLSAgICAgICAgICAgIG1QLnNldFN0cm9rZUNhcChQYWludC5DYXAuU1FVQVJFKTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBqb2luID0gYnMuZ2V0TGluZUpvaW4oKTsKLSAgICAgICAgaWYgKGpvaW4gPT0gMCkgewotICAgICAgICAgICAgbVAuc2V0U3Ryb2tlSm9pbihQYWludC5Kb2luLk1JVEVSKTsKLSAgICAgICAgfSBlbHNlIGlmIChqb2luID09IDEpIHsKLSAgICAgICAgICAgIG1QLnNldFN0cm9rZUpvaW4oUGFpbnQuSm9pbi5ST1VORCk7Ci0gICAgICAgIH0gZWxzZSBpZiAoam9pbiA9PSAyKSB7Ci0gICAgICAgICAgICBtUC5zZXRTdHJva2VKb2luKFBhaW50LkpvaW4uQkVWRUwpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyBmbG9hdFtdIGNyZWF0ZU1hdHJpeChBZmZpbmVUcmFuc2Zvcm0gVHgpIHsKLSAgICAgICAgZG91YmxlW10gYXQgPSBuZXcgZG91YmxlWzldOwotICAgICAgICBUeC5nZXRNYXRyaXgoYXQpOwotICAgICAgICBmbG9hdFtdIGYgPSBuZXcgZmxvYXRbYXQubGVuZ3RoXTsKLSAgICAgICAgZlswXSA9IChmbG9hdCkgYXRbMF07Ci0gICAgICAgIGZbMV0gPSAoZmxvYXQpIGF0WzJdOwotICAgICAgICBmWzJdID0gKGZsb2F0KSBhdFs0XTsKLSAgICAgICAgZlszXSA9IChmbG9hdCkgYXRbMV07Ci0gICAgICAgIGZbNF0gPSAoZmxvYXQpIGF0WzNdOwotICAgICAgICBmWzVdID0gKGZsb2F0KSBhdFs1XTsKLSAgICAgICAgZls2XSA9IDA7Ci0gICAgICAgIGZbN10gPSAwOwotICAgICAgICBmWzhdID0gMTsKLSAgICAgICAgcmV0dXJuIGY7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBmbG9hdFtdIGNyZWF0ZUFXVE1hdHJpeChmbG9hdFtdIG1hdHJpeCkgewotICAgICAgICBmbG9hdFtdIGF0ID0gbmV3IGZsb2F0WzldOwotICAgICAgICBhdFswXSA9IG1hdHJpeFswXTsKLSAgICAgICAgYXRbMV0gPSBtYXRyaXhbM107Ci0gICAgICAgIGF0WzJdID0gbWF0cml4WzFdOwotICAgICAgICBhdFszXSA9IG1hdHJpeFs0XTsKLSAgICAgICAgYXRbNF0gPSBtYXRyaXhbMl07Ci0gICAgICAgIGF0WzVdID0gbWF0cml4WzVdOwotICAgICAgICBhdFs2XSA9IDA7Ci0gICAgICAgIGF0WzddID0gMDsKLSAgICAgICAgYXRbOF0gPSAxOwotICAgICAgICByZXR1cm4gYXQ7Ci0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyBNYXRyaXggY3JlYXRlTWF0cml4T2JqKEFmZmluZVRyYW5zZm9ybSBUeCkgewotICAgICAgICBNYXRyaXggbSA9IG5ldyBNYXRyaXgoKTsKLSAgICAgICAgbS5yZXNldCgpOwotICAgICAgICBtLnNldFZhbHVlcyhjcmVhdGVNYXRyaXgoVHgpKTsKLSAgICAgICAgcmV0dXJuIG07Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0VHJhbnNmb3JtKEFmZmluZVRyYW5zZm9ybSBUeCkgewotICAgICAgICBtTS5yZXNldCgpOwotICAgICAgICAvKgotICAgICAgICAgKiBpZihUeC5pc0lkZW50aXR5KCkpIHsgbU0gPSBuZXcgTWF0cml4KCk7IH0KLSAgICAgICAgICovCi0gICAgICAgIG1NLnNldFZhbHVlcyhjcmVhdGVNYXRyaXgoVHgpKTsKLSAgICAgICAgTWF0cml4IG0gPSBuZXcgTWF0cml4KCk7Ci0gICAgICAgIG0uc2V0VmFsdWVzKGdldEludmVyc2VNYXRyaXgoKSk7Ci0gICAgICAgIG1DLmNvbmNhdChtKTsKLSAgICAgICAgbUMuY29uY2F0KG1NKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzaGVhcihkb3VibGUgc2h4LCBkb3VibGUgc2h5KSB7Ci0gICAgICAgIG1NLnNldFNrZXcoKGZsb2F0KSBzaHgsIChmbG9hdCkgc2h5KTsKLSAgICAgICAgbUMuY29uY2F0KG1NKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB0cmFuc2Zvcm0oQWZmaW5lVHJhbnNmb3JtIFR4KSB7Ci0gICAgICAgIE1hdHJpeCBtID0gbmV3IE1hdHJpeCgpOwotICAgICAgICBtLnNldFZhbHVlcyhjcmVhdGVNYXRyaXgoVHgpKTsKLSAgICAgICAgbUMuY29uY2F0KG0pOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHRyYW5zbGF0ZShkb3VibGUgdHgsIGRvdWJsZSB0eSkgewotICAgICAgICBtTS5zZXRUcmFuc2xhdGUoKGZsb2F0KSB0eCwgKGZsb2F0KSB0eSk7Ci0gICAgICAgIG1DLmNvbmNhdChtTSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgdHJhbnNsYXRlKGludCB4LCBpbnQgeSkgewotICAgICAgICBtTS5zZXRUcmFuc2xhdGUoKGZsb2F0KSB4LCAoZmxvYXQpIHkpOwotICAgICAgICBtQy5jb25jYXQobU0pOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGNsZWFyUmVjdChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICBtQy5jbGlwUmVjdCh4LCB5LCB4ICsgd2lkdGgsIHkgKyBoZWlnaHQpOwotICAgICAgICBpZiAobUJjICE9IG51bGwpIHsKLSAgICAgICAgICAgIG1DLmRyYXdBUkdCKG1CYy5nZXRBbHBoYSgpLCBtQmMuZ2V0Qmx1ZSgpLCBtQmMuZ2V0R3JlZW4oKSwgbUJjCi0gICAgICAgICAgICAgICAgICAgIC5nZXRSZWQoKSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBtQy5kcmF3QVJHQigweGZmLCAweGZmLCAweGZmLCAweGZmKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGNsaXBSZWN0KGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIGludCBjbFtdID0gey0xLCB4LCB5LCAtMiwgeCwgeSArIHdpZHRoLCAtMiwgeCArIGhlaWdodCwgeSArIHdpZHRoLCAtMiwgeCArIGhlaWdodCwgeX07Ci0gICAgICAgIFNoYXBlIHNocCA9IGNyZWF0ZVNoYXBlKGNsKTsKLSAgICAgICAgbUN1cnJDbGlwLmludGVyc2VjdChuZXcgQXJlYShzaHApKTsKLSAgICAgICAgbUMuY2xpcFJlY3QobmV3IFJlY3QoeCwgeSwgeCArIHdpZHRoLCB5ICsgaGVpZ2h0KSwgUmVnaW9uLk9wLklOVEVSU0VDVCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgY29weUFyZWEoaW50IHN4LCBpbnQgc3ksIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGR4LCBpbnQgZHkpIHsKLSAgICAgICAgY29weUFyZWEobUMsIHN4LCBzeSwgd2lkdGggKyBkeCwgaGVpZ2h0ICsgZHksIGR4LCBkeSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEdyYXBoaWNzIGNyZWF0ZSgpIHsKLSAgICAgICAgcmV0dXJuIHRoaXM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZGlzcG9zZSgpIHsKLSAgICAgICAgICAgIG1DID0gbnVsbDsKLSAgICAgICAgICAgIG1QID0gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3QXJjKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgc2EsIGludCBlYSkgewotICAgICAgICAgICAgaWYgKG1QID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBtUCA9IG5ldyBQYWludCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgbVAuc2V0U3Ryb2tlV2lkdGgoMCk7Ci0gICAgICAgICAgICBtQy5kcmF3QXJjKG5ldyBSZWN0Rih4LCB5LCB4ICsgd2lkdGgsIHkgKyBoZWlnaHQpLCAzNjAgLSAoZWEgKyBzYSksCi0gICAgICAgICAgICAgICAgICAgICAgIGVhLCB0cnVlLCBtUCk7Ci0gICAgfQotCi0gICAgCi0gICAgLy8gPz8/QVdUOiBvbmx5IHVzZWQgZm9yIGRlYnVnaW5nLCBkZWxldGUgaW4gZmluYWwgdmVyc2lvbgotICAgIHB1YmxpYyB2b2lkIGRyYXdCaXRtYXAoQml0bWFwIGJtLCBmbG9hdCB4LCBmbG9hdCB5LCBQYWludCBwKSB7Ci0gICAgICAgIG1DLmRyYXdCaXRtYXAoYm0sIHgsIHksIG51bGwpOwotICAgIH0KLSAgICAKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1hZ2UsIGludCB4LCBpbnQgeSwgQ29sb3IgYmdjb2xvciwKLSAgICAgICAgICAgIEltYWdlT2JzZXJ2ZXIgaW1hZ2VPYnNlcnZlcikgewotCi0gICAgICAgIGlmKGltYWdlID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0KLSAgICAgICAgYm9vbGVhbiBkb25lID0gZmFsc2U7Ci0gICAgICAgIGJvb2xlYW4gc29tZWJpdHMgPSBmYWxzZTsKLSAgICAgICAgU3VyZmFjZSBzcmNTdXJmID0gbnVsbDsKLSAgICAgICAgaWYoaW1hZ2UgaW5zdGFuY2VvZiBPZmZzY3JlZW5JbWFnZSl7Ci0gICAgICAgICAgICBPZmZzY3JlZW5JbWFnZSBvaSA9IChPZmZzY3JlZW5JbWFnZSkgaW1hZ2U7Ci0gICAgICAgICAgICBpZigob2kuZ2V0U3RhdGUoKSAmIEltYWdlT2JzZXJ2ZXIuRVJST1IpICE9IDApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkb25lID0gb2kucHJlcGFyZUltYWdlKGltYWdlT2JzZXJ2ZXIpOwotICAgICAgICAgICAgc29tZWJpdHMgPSAob2kuZ2V0U3RhdGUoKSAmIEltYWdlT2JzZXJ2ZXIuU09NRUJJVFMpICE9IDA7Ci0gICAgICAgICAgICBzcmNTdXJmID0gb2kuZ2V0SW1hZ2VTdXJmYWNlKCk7Ci0gICAgICAgIH1lbHNlewotICAgICAgICAgICAgZG9uZSA9IHRydWU7Ci0gICAgICAgICAgICBzcmNTdXJmID0gU3VyZmFjZS5nZXRJbWFnZVN1cmZhY2UoaW1hZ2UpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYoZG9uZSB8fCBzb21lYml0cykgewotICAgICAgICAgICAgaW50IHcgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgaCA9IHNyY1N1cmYuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIGJsaXR0ZXIuYmxpdCgwLCAwLCBzcmNTdXJmLCB4LCB5LCBkc3RTdXJmLCB3LCBoLCAoQWZmaW5lVHJhbnNmb3JtKSB0cmFuc2Zvcm0uY2xvbmUoKSwKLSAgICAgICAgICAgICAgICAgICAgY29tcG9zaXRlLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZG9uZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1hZ2UsIGludCB4LCBpbnQgeSwgSW1hZ2VPYnNlcnZlciBpbWFnZU9ic2VydmVyKSB7Ci0gICAgICAgIHJldHVybiBkcmF3SW1hZ2UoaW1hZ2UsIHgsIHksIG51bGwsIGltYWdlT2JzZXJ2ZXIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGRyYXdJbWFnZShJbWFnZSBpbWFnZSwgaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQsCi0gICAgICAgICAgICBDb2xvciBiZ2NvbG9yLCBJbWFnZU9ic2VydmVyIGltYWdlT2JzZXJ2ZXIpIHsKLQotICAgICAgICBpZihpbWFnZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgICAgICBpZih3aWR0aCA9PSAwIHx8IGhlaWdodCA9PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGJvb2xlYW4gZG9uZSA9IGZhbHNlOwotICAgICAgICBib29sZWFuIHNvbWViaXRzID0gZmFsc2U7Ci0gICAgICAgIFN1cmZhY2Ugc3JjU3VyZiA9IG51bGw7Ci0KLSAgICAgICAgaWYoaW1hZ2UgaW5zdGFuY2VvZiBPZmZzY3JlZW5JbWFnZSl7Ci0gICAgICAgICAgICBPZmZzY3JlZW5JbWFnZSBvaSA9IChPZmZzY3JlZW5JbWFnZSkgaW1hZ2U7Ci0gICAgICAgICAgICBpZigob2kuZ2V0U3RhdGUoKSAmIEltYWdlT2JzZXJ2ZXIuRVJST1IpICE9IDApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkb25lID0gb2kucHJlcGFyZUltYWdlKGltYWdlT2JzZXJ2ZXIpOwotICAgICAgICAgICAgc29tZWJpdHMgPSAob2kuZ2V0U3RhdGUoKSAmIEltYWdlT2JzZXJ2ZXIuU09NRUJJVFMpICE9IDA7Ci0gICAgICAgICAgICBzcmNTdXJmID0gb2kuZ2V0SW1hZ2VTdXJmYWNlKCk7Ci0gICAgICAgIH1lbHNlewotICAgICAgICAgICAgZG9uZSA9IHRydWU7Ci0gICAgICAgICAgICBzcmNTdXJmID0gU3VyZmFjZS5nZXRJbWFnZVN1cmZhY2UoaW1hZ2UpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYoZG9uZSB8fCBzb21lYml0cykgewotICAgICAgICAgICAgaW50IHcgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgaCA9IHNyY1N1cmYuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICBpZih3ID09IHdpZHRoICYmIGggPT0gaGVpZ2h0KXsKLSAgICAgICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgeCwgeSwgZHN0U3VyZiwgdywgaCwKLSAgICAgICAgICAgICAgICAgICAgICAgIChBZmZpbmVUcmFuc2Zvcm0pIHRyYW5zZm9ybS5jbG9uZSgpLAotICAgICAgICAgICAgICAgICAgICAgICAgY29tcG9zaXRlLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgIEFmZmluZVRyYW5zZm9ybSB4Zm9ybSA9IG5ldyBBZmZpbmVUcmFuc2Zvcm0oKTsKLSAgICAgICAgICAgICAgICB4Zm9ybS5zZXRUb1NjYWxlKChmbG9hdCl3aWR0aCAvIHcsIChmbG9hdCloZWlnaHQgLyBoKTsKLSAgICAgICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgeCwgeSwgZHN0U3VyZiwgdywgaCwKLSAgICAgICAgICAgICAgICAgICAgICAgIChBZmZpbmVUcmFuc2Zvcm0pIHRyYW5zZm9ybS5jbG9uZSgpLAotICAgICAgICAgICAgICAgICAgICAgICAgeGZvcm0sIGNvbXBvc2l0ZSwgYmdjb2xvciwgY2xpcCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGRvbmU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZHJhd0ltYWdlKEltYWdlIGltYWdlLCBpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCwKLSAgICAgICAgICAgIEltYWdlT2JzZXJ2ZXIgaW1hZ2VPYnNlcnZlcikgewotICAgICAgICByZXR1cm4gZHJhd0ltYWdlKGltYWdlLCB4LCB5LCB3aWR0aCwgaGVpZ2h0LCBudWxsLCBpbWFnZU9ic2VydmVyKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1hZ2UsIGludCBkeDEsIGludCBkeTEsIGludCBkeDIsIGludCBkeTIsCi0gICAgICAgICAgICBpbnQgc3gxLCBpbnQgc3kxLCBpbnQgc3gyLCBpbnQgc3kyLCBDb2xvciBiZ2NvbG9yLAotICAgICAgICAgICAgSW1hZ2VPYnNlcnZlciBpbWFnZU9ic2VydmVyKSB7Ci0KLSAgICAgICAgaWYoaW1hZ2UgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgaWYoZHgxID09IGR4MiB8fCBkeTEgPT0gZHkyIHx8IHN4MSA9PSBzeDIgfHwgc3kxID09IHN5MikgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICBib29sZWFuIGRvbmUgPSBmYWxzZTsKLSAgICAgICAgYm9vbGVhbiBzb21lYml0cyA9IGZhbHNlOwotICAgICAgICBTdXJmYWNlIHNyY1N1cmYgPSBudWxsOwotICAgICAgICBpZihpbWFnZSBpbnN0YW5jZW9mIE9mZnNjcmVlbkltYWdlKXsKLSAgICAgICAgICAgIE9mZnNjcmVlbkltYWdlIG9pID0gKE9mZnNjcmVlbkltYWdlKSBpbWFnZTsKLSAgICAgICAgICAgIGlmKChvaS5nZXRTdGF0ZSgpICYgSW1hZ2VPYnNlcnZlci5FUlJPUikgIT0gMCkgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGRvbmUgPSBvaS5wcmVwYXJlSW1hZ2UoaW1hZ2VPYnNlcnZlcik7Ci0gICAgICAgICAgICBzb21lYml0cyA9IChvaS5nZXRTdGF0ZSgpICYgSW1hZ2VPYnNlcnZlci5TT01FQklUUykgIT0gMDsKLSAgICAgICAgICAgIHNyY1N1cmYgPSBvaS5nZXRJbWFnZVN1cmZhY2UoKTsKLSAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICBkb25lID0gdHJ1ZTsKLSAgICAgICAgICAgIHNyY1N1cmYgPSBTdXJmYWNlLmdldEltYWdlU3VyZmFjZShpbWFnZSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZihkb25lIHx8IHNvbWViaXRzKSB7Ci0KLSAgICAgICAgICAgIGludCBkc3RYID0gZHgxOwotICAgICAgICAgICAgaW50IGRzdFkgPSBkeTE7Ci0gICAgICAgICAgICBpbnQgc3JjWCA9IHN4MTsKLSAgICAgICAgICAgIGludCBzcmNZID0gc3kxOwotCi0gICAgICAgICAgICBpbnQgZHN0VyA9IGR4MiAtIGR4MTsKLSAgICAgICAgICAgIGludCBkc3RIID0gZHkyIC0gZHkxOwotICAgICAgICAgICAgaW50IHNyY1cgPSBzeDIgLSBzeDE7Ci0gICAgICAgICAgICBpbnQgc3JjSCA9IHN5MiAtIHN5MTsKLQotICAgICAgICAgICAgaWYoc3JjVyA9PSBkc3RXICYmIHNyY0ggPT0gZHN0SCl7Ci0gICAgICAgICAgICAgICAgYmxpdHRlci5ibGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIGRzdFgsIGRzdFksIGRzdFN1cmYsIHNyY1csIHNyY0gsCi0gICAgICAgICAgICAgICAgICAgICAgICAoQWZmaW5lVHJhbnNmb3JtKSB0cmFuc2Zvcm0uY2xvbmUoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbXBvc2l0ZSwgYmdjb2xvciwgY2xpcCk7Ci0gICAgICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0gPSBuZXcgQWZmaW5lVHJhbnNmb3JtKCk7Ci0gICAgICAgICAgICAgICAgeGZvcm0uc2V0VG9TY2FsZSgoZmxvYXQpZHN0VyAvIHNyY1csIChmbG9hdClkc3RIIC8gc3JjSCk7Ci0gICAgICAgICAgICAgICAgYmxpdHRlci5ibGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIGRzdFgsIGRzdFksIGRzdFN1cmYsIHNyY1csIHNyY0gsCi0gICAgICAgICAgICAgICAgICAgICAgICAoQWZmaW5lVHJhbnNmb3JtKSB0cmFuc2Zvcm0uY2xvbmUoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHhmb3JtLCBjb21wb3NpdGUsIGJnY29sb3IsIGNsaXApOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBkb25lOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGRyYXdJbWFnZShJbWFnZSBpbWFnZSwgaW50IGR4MSwgaW50IGR5MSwgaW50IGR4MiwgaW50IGR5MiwKLSAgICAgICAgICAgIGludCBzeDEsIGludCBzeTEsIGludCBzeDIsIGludCBzeTIsIEltYWdlT2JzZXJ2ZXIgaW1hZ2VPYnNlcnZlcikgewotCi0gICAgICAgIHJldHVybiBkcmF3SW1hZ2UoaW1hZ2UsIGR4MSwgZHkxLCBkeDIsIGR5Miwgc3gxLCBzeTEsIHN4Miwgc3kyLCBudWxsLAotICAgICAgICAgICAgICAgIGltYWdlT2JzZXJ2ZXIpOwotICAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3SW1hZ2UoQnVmZmVyZWRJbWFnZSBidWZJbWFnZSwgQnVmZmVyZWRJbWFnZU9wIG9wLAotICAgICAgICAgICAgaW50IHgsIGludCB5KSB7Ci0KLSAgICAgICAgaWYoYnVmSW1hZ2UgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYob3AgPT0gbnVsbCkgewotICAgICAgICAgICAgZHJhd0ltYWdlKGJ1ZkltYWdlLCB4LCB5LCBudWxsKTsKLSAgICAgICAgfSBlbHNlIGlmKG9wIGluc3RhbmNlb2YgQWZmaW5lVHJhbnNmb3JtT3ApewotICAgICAgICAgICAgQWZmaW5lVHJhbnNmb3JtT3AgYXRvcCA9IChBZmZpbmVUcmFuc2Zvcm1PcCkgb3A7Ci0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0gPSBhdG9wLmdldFRyYW5zZm9ybSgpOwotICAgICAgICAgICAgU3VyZmFjZSBzcmNTdXJmID0gU3VyZmFjZS5nZXRJbWFnZVN1cmZhY2UoYnVmSW1hZ2UpOwotICAgICAgICAgICAgaW50IHcgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgaCA9IHNyY1N1cmYuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgeCwgeSwgZHN0U3VyZiwgdywgaCwKLSAgICAgICAgICAgICAgICAgICAgKEFmZmluZVRyYW5zZm9ybSkgdHJhbnNmb3JtLmNsb25lKCksIHhmb3JtLAotICAgICAgICAgICAgICAgICAgICBjb21wb3NpdGUsIG51bGwsIGNsaXApOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgYnVmSW1hZ2UgPSBvcC5maWx0ZXIoYnVmSW1hZ2UsIG51bGwpOwotICAgICAgICAgICAgU3VyZmFjZSBzcmNTdXJmID0gU3VyZmFjZS5nZXRJbWFnZVN1cmZhY2UoYnVmSW1hZ2UpOwotICAgICAgICAgICAgaW50IHcgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgaCA9IHNyY1N1cmYuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgeCwgeSwgZHN0U3VyZiwgdywgaCwKLSAgICAgICAgICAgICAgICAgICAgKEFmZmluZVRyYW5zZm9ybSkgdHJhbnNmb3JtLmNsb25lKCksCi0gICAgICAgICAgICAgICAgICAgIGNvbXBvc2l0ZSwgbnVsbCwgY2xpcCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1hZ2UsIEFmZmluZVRyYW5zZm9ybSB0cmFucywKLSAgICAgICAgICAgIEltYWdlT2JzZXJ2ZXIgaW1hZ2VPYnNlcnZlcikgewotCi0gICAgICAgIGlmKGltYWdlID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgICAgIGlmKHRyYW5zID09IG51bGwgfHwgdHJhbnMuaXNJZGVudGl0eSgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZHJhd0ltYWdlKGltYWdlLCAwLCAwLCBpbWFnZU9ic2VydmVyKTsKLSAgICAgICAgfQotCi0gICAgICAgIGJvb2xlYW4gZG9uZSA9IGZhbHNlOwotICAgICAgICBib29sZWFuIHNvbWViaXRzID0gZmFsc2U7Ci0gICAgICAgIFN1cmZhY2Ugc3JjU3VyZiA9IG51bGw7Ci0gICAgICAgIGlmKGltYWdlIGluc3RhbmNlb2YgT2Zmc2NyZWVuSW1hZ2UpewotICAgICAgICAgICAgT2Zmc2NyZWVuSW1hZ2Ugb2kgPSAoT2Zmc2NyZWVuSW1hZ2UpIGltYWdlOwotICAgICAgICAgICAgaWYoKG9pLmdldFN0YXRlKCkgJiBJbWFnZU9ic2VydmVyLkVSUk9SKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZG9uZSA9IG9pLnByZXBhcmVJbWFnZShpbWFnZU9ic2VydmVyKTsKLSAgICAgICAgICAgIHNvbWViaXRzID0gKG9pLmdldFN0YXRlKCkgJiBJbWFnZU9ic2VydmVyLlNPTUVCSVRTKSAhPSAwOwotICAgICAgICAgICAgc3JjU3VyZiA9IG9pLmdldEltYWdlU3VyZmFjZSgpOwotICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgIGRvbmUgPSB0cnVlOwotICAgICAgICAgICAgc3JjU3VyZiA9IFN1cmZhY2UuZ2V0SW1hZ2VTdXJmYWNlKGltYWdlKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmKGRvbmUgfHwgc29tZWJpdHMpIHsKLSAgICAgICAgICAgIGludCB3ID0gc3JjU3VyZi5nZXRXaWR0aCgpOwotICAgICAgICAgICAgaW50IGggPSBzcmNTdXJmLmdldEhlaWdodCgpOwotICAgICAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHhmb3JtID0gKEFmZmluZVRyYW5zZm9ybSkgdHJhbnNmb3JtLmNsb25lKCk7Ci0gICAgICAgICAgICB4Zm9ybS5jb25jYXRlbmF0ZSh0cmFucyk7Ci0gICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgMCwgMCwgZHN0U3VyZiwgdywgaCwgeGZvcm0sIGNvbXBvc2l0ZSwKLSAgICAgICAgICAgICAgICAgICAgbnVsbCwgY2xpcCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGRvbmU7Ci0gICAgfQotICAgICAgICAKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3TGluZShpbnQgeDEsIGludCB5MSwgaW50IHgyLCBpbnQgeTIpIHsKLSAgICAgICAgaWYgKG1QID09IG51bGwpIHsKLSAgICAgICAgICAgIG1QID0gbmV3IFBhaW50KCk7Ci0gICAgICAgIH0KLSAgICAgICAgICAgIG1DLmRyYXdMaW5lKHgxLCB5MSwgeDIsIHkyLCBtUCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd092YWwoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgICAgIGlmIChtUCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgbVAgPSBuZXcgUGFpbnQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1QLnNldFN0eWxlKFBhaW50LlN0eWxlLlNUUk9LRSk7Ci0gICAgICAgICAgICBtQy5kcmF3T3ZhbChuZXcgUmVjdEYoeCwgeSwgeCArIHdpZHRoLCB5ICsgaGVpZ2h0KSwgbVApOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRyYXdQb2x5Z29uKGludFtdIHhwb2ludHMsIGludFtdIHlwb2ludHMsIGludCBucG9pbnRzKSB7Ci0gICAgICAgICAgICBpZiAobVAgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIG1QID0gbmV3IFBhaW50KCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBtQy5kcmF3TGluZSh4cG9pbnRzW25wb2ludHMgLSAxXSwgeXBvaW50c1tucG9pbnRzIC0gMV0sIHhwb2ludHNbMF0sCi0gICAgICAgICAgICAgICAgICAgIHlwb2ludHNbMF0sIG1QKTsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnBvaW50cyAtIDE7IGkrKykgewotICAgICAgICAgICAgICAgIG1DLmRyYXdMaW5lKHhwb2ludHNbaV0sIHlwb2ludHNbaV0sIHhwb2ludHNbaSArIDFdLAotICAgICAgICAgICAgICAgICAgICAgICAgeXBvaW50c1tpICsgMV0sIG1QKTsKLSAgICAgICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3UG9seWxpbmUoaW50W10geHBvaW50cywgaW50W10geXBvaW50cywgaW50IG5wb2ludHMpIHsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBucG9pbnRzIC0gMTsgaSsrKSB7Ci0gICAgICAgICAgICBkcmF3TGluZSh4cG9pbnRzW2ldLCB5cG9pbnRzW2ldLCB4cG9pbnRzW2kgKyAxXSwgeXBvaW50c1tpICsgMV0pOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3Um91bmRSZWN0KGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LAotICAgICAgICAgICAgaW50IGFyY1dpZHRoLCBpbnQgYXJjSGVpZ2h0KSB7Ci0gICAgICAgICAgICBpZiAobVAgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIG1QID0gbmV3IFBhaW50KCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBtQy5kcmF3Um91bmRSZWN0KG5ldyBSZWN0Rih4LCB5LCB3aWR0aCwgaGVpZ2h0KSwgYXJjV2lkdGgsCi0gICAgICAgICAgICAgICAgICAgIGFyY0hlaWdodCwgbVApOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGZpbGxBcmMoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQsIGludCBzYSwgaW50IGVhKSB7Ci0gICAgICAgICAgICBpZiAobVAgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIG1QID0gbmV3IFBhaW50KCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIFBhaW50LlN0eWxlIHRtcCA9IG1QLmdldFN0eWxlKCk7Ci0gICAgICAgICAgICBtUC5zZXRTdHlsZShQYWludC5TdHlsZS5GSUxMX0FORF9TVFJPS0UpOwotICAgICAgICAgICAgbUMuZHJhd0FyYyhuZXcgUmVjdEYoeCwgeSwgeCArIHdpZHRoLCB5ICsgaGVpZ2h0KSwgMzYwIC0gKHNhICsgZWEpLAotICAgICAgICAgICAgICAgICAgICBlYSwgdHJ1ZSwgbVApOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBtUC5zZXRTdHlsZSh0bXApOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGZpbGxPdmFsKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgICAgICBpZiAobVAgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIG1QID0gbmV3IFBhaW50KCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBQYWludC5TdHlsZSB0bXAgPSBtUC5nZXRTdHlsZSgpOwotICAgICAgICAgICAgbVAuc2V0U3R5bGUoUGFpbnQuU3R5bGUuRklMTCk7Ci0gICAgICAgICAgICBtQy5kcmF3T3ZhbChuZXcgUmVjdEYoeCwgeSwgeCArIHdpZHRoLCB5ICsgaGVpZ2h0KSwgbVApOwotICAgICAgICAgICAgbVAuc2V0U3R5bGUodG1wKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmaWxsUG9seWdvbihpbnRbXSB4cG9pbnRzLCBpbnRbXSB5cG9pbnRzLCBpbnQgbnBvaW50cykgewotICAgICAgICAgICAgaWYgKG1QID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBtUCA9IG5ldyBQYWludCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgUGFpbnQuU3R5bGUgdG1wID0gbVAuZ2V0U3R5bGUoKTsKLSAgICAgICAgICAgIG1DLnNhdmUoQ2FudmFzLkNMSVBfU0FWRV9GTEFHKTsKLQotICAgICAgICAgICAgbVAuc2V0U3R5bGUoUGFpbnQuU3R5bGUuRklMTCk7Ci0KLSAgICAgICAgICAgIEdlbmVyYWxQYXRoIGZpbGxlZFBvbHlnb24gPSBuZXcgR2VuZXJhbFBhdGgoCi0gICAgICAgICAgICAgICAgICAgIEdlbmVyYWxQYXRoLldJTkRfRVZFTl9PREQsIG5wb2ludHMpOwotICAgICAgICAgICAgZmlsbGVkUG9seWdvbi5tb3ZlVG8oeHBvaW50c1swXSwgeXBvaW50c1swXSk7Ci0gICAgICAgICAgICBmb3IgKGludCBpbmRleCA9IDE7IGluZGV4IDwgeHBvaW50cy5sZW5ndGg7IGluZGV4KyspIHsKLSAgICAgICAgICAgICAgICBmaWxsZWRQb2x5Z29uLmxpbmVUbyh4cG9pbnRzW2luZGV4XSwgeXBvaW50c1tpbmRleF0pOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZmlsbGVkUG9seWdvbi5jbG9zZVBhdGgoKTsKLSAgICAgICAgICAgIFBhdGggcGF0aCA9IGdldFBhdGgoZmlsbGVkUG9seWdvbik7Ci0gICAgICAgICAgICBtQy5jbGlwUGF0aChwYXRoKTsKLSAgICAgICAgICAgIG1DLmRyYXdQYXRoKHBhdGgsIG1QKTsKLQotICAgICAgICAgICAgbVAuc2V0U3R5bGUodG1wKTsKLSAgICAgICAgICAgIG1DLnJlc3RvcmUoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmaWxsUmVjdChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICAgICAgaWYgKG1QID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBtUCA9IG5ldyBQYWludCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgUGFpbnQuU3R5bGUgdG1wID0gbVAuZ2V0U3R5bGUoKTsKLSAgICAgICAgICAgIG1QLnNldFN0eWxlKFBhaW50LlN0eWxlLkZJTEwpOwotICAgICAgICAgICAgbUMuZHJhd1JlY3QobmV3IFJlY3QoeCwgeSwgeCArIHdpZHRoLCB5ICsgaGVpZ2h0KSwgbVApOwotICAgICAgICAgICAgbVAuc2V0U3R5bGUodG1wKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3UmVjdChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICBpbnRbXSB4cG9pbnRzID0geyB4LCB4LCB4ICsgd2lkdGgsIHggKyB3aWR0aCB9OwotICAgICAgICBpbnRbXSB5cG9pbnRzID0geyB5LCB5ICsgaGVpZ2h0LCB5ICsgaGVpZ2h0LCB5IH07Ci0gICAgICAgIGRyYXdQb2x5Z29uKHhwb2ludHMsIHlwb2ludHMsIDQpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGZpbGxSb3VuZFJlY3QoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQsCi0gICAgICAgICAgICBpbnQgYXJjV2lkdGgsIGludCBhcmNIZWlnaHQpIHsKLSAgICAgICAgICAgIGlmIChtUCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgbVAgPSBuZXcgUGFpbnQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1QLnNldFN0eWxlKFBhaW50LlN0eWxlLkZJTEwpOwotICAgICAgICAgICAgbUMuZHJhd1JvdW5kUmVjdChuZXcgUmVjdEYoeCwgeSwgeCArIHdpZHRoLCB5ICsgaGVpZ2h0KSwgYXJjV2lkdGgsCi0gICAgICAgICAgICAgICAgICAgIGFyY0hlaWdodCwgbVApOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTaGFwZSBnZXRDbGlwKCkgewotICAgICAgICByZXR1cm4gbUN1cnJDbGlwOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Q2xpcEJvdW5kcygpIHsKLSAgICAgICAgICAgIFJlY3QgciA9IG1DLmdldENsaXBCb3VuZHMoKTsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlKHIubGVmdCwgci50b3AsIHIud2lkdGgoKSwgci5oZWlnaHQoKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIENvbG9yIGdldENvbG9yKCkgewotICAgICAgICBpZiAobVAgIT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBDb2xvcihtUC5nZXRDb2xvcigpKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgRm9udCBnZXRGb250KCkgewotICAgICAgICByZXR1cm4gbUZudDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgRm9udE1ldHJpY3MgZ2V0Rm9udE1ldHJpY3MoRm9udCBmb250KSB7Ci0gICAgICAgIG1GbSA9IG5ldyBGb250TWV0cmljc0ltcGwoZm9udCk7Ci0gICAgICAgIHJldHVybiBtRm07Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0Q2xpcChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICBpbnQgY2xbXSA9IHstMSwgeCwgeSwgLTIsIHgsIHkgKyB3aWR0aCwgLTIsIHggKyBoZWlnaHQsIHkgKyB3aWR0aCwgLTIsIHggKyBoZWlnaHQsIHl9OwotICAgICAgICBtQ3VyckNsaXAgPSBuZXcgQXJlYShjcmVhdGVTaGFwZShjbCkpOwotICAgICAgICBtQy5jbGlwUmVjdCh4LCB5LCB4ICsgd2lkdGgsIHkgKyBoZWlnaHQsIFJlZ2lvbi5PcC5SRVBMQUNFKTsKLQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldENsaXAoU2hhcGUgY2xpcCkgewotICAgICAgICBtQ3VyckNsaXAgPSBuZXcgQXJlYShjbGlwKTsKLSAgICAgICAgbUMuY2xpcFBhdGgoZ2V0UGF0aChjbGlwKSwgUmVnaW9uLk9wLlJFUExBQ0UpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldENvbG9yKENvbG9yIGMpIHsKLSAgICAgICAgaWYgKG1QID09IG51bGwpIHsKLSAgICAgICAgICAgIG1QID0gbmV3IFBhaW50KCk7Ci0gICAgICAgIH0KLSAgICAgICAgbVAuc2V0Q29sb3IoYy5nZXRSR0IoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRm9udCBtYXBwaW5nOgotICAgICAqIAotICAgICAqIEZhbWlseToKLSAgICAgKiAKLSAgICAgKiBBbmRyb2lkICAgICAgICAgQVdUCi0gICAgICogLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICAgICAqIHNlcmlmICAgICAgICAgICBTZXJpZiAvIFRpbWVzUm9tYW4KLSAgICAgKiBzYW5zLXNlcmlmICAgICAgU2Fuc1NlcmlmIC8gSGVsdmV0aWNhCi0gICAgICogbW9ub3NwYWNlICAgICAgIE1vbm9zcGFjZWQgLyBDb3VyaWVyCi0gICAgICogCi0gICAgICogU3R5bGU6Ci0gICAgICogCi0gICAgICogQW5kcm9pZCAgICAgICAgICAgIEFXVAotICAgICAqIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSAgICAgKiBub3JtYWwgICAgICAgICAgUGxhaW4KLSAgICAgKiBib2xkICAgICAgICAgICAgYm9sZAotICAgICAqIGl0YWxpYyAgICAgICAgICBpdGFsaWMKLSAgICAgKiAKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRGb250KEZvbnQgZm9udCkgewotICAgICAgICBpZiAoZm9udCA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG1QID09IG51bGwpIHsKLSAgICAgICAgICAgIG1QID0gbmV3IFBhaW50KCk7Ci0gICAgICAgIH0KLQotICAgICAgICBtRm50ID0gZm9udDsKLSAgICAgICAgVHlwZWZhY2UgdGYgPSBudWxsOwotICAgICAgICBpbnQgc3R5ID0gZm9udC5nZXRTdHlsZSgpOwotICAgICAgICBTdHJpbmcgbmFtID0gZm9udC5nZXROYW1lKCk7Ci0gICAgICAgIFN0cmluZyBhRiA9ICIiOwotICAgICAgICBpZiAobmFtICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChuYW0uZXF1YWxzSWdub3JlQ2FzZSgiU2VyaWYiKQotICAgICAgICAgICAgICAgICAgICB8fCBuYW0uZXF1YWxzSWdub3JlQ2FzZSgiVGltZXNSb21hbiIpKSB7Ci0gICAgICAgICAgICAgICAgYUYgPSAic2VyaWYiOwotICAgICAgICAgICAgfSBlbHNlIGlmIChuYW0uZXF1YWxzSWdub3JlQ2FzZSgiU2Fuc1NlcmlmIikKLSAgICAgICAgICAgICAgICAgICAgfHwgbmFtLmVxdWFsc0lnbm9yZUNhc2UoIkhlbHZldGljYSIpKSB7Ci0gICAgICAgICAgICAgICAgYUYgPSAic2Fucy1zZXJpZiI7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKG5hbS5lcXVhbHNJZ25vcmVDYXNlKCJNb25vc3BhY2VkIikKLSAgICAgICAgICAgICAgICAgICAgfHwgbmFtLmVxdWFsc0lnbm9yZUNhc2UoIkNvdXJpZXIiKSkgewotICAgICAgICAgICAgICAgIGFGID0gIm1vbm9zcGFjZSI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBzd2l0Y2ggKHN0eSkgewotICAgICAgICBjYXNlIEZvbnQuUExBSU46Ci0gICAgICAgICAgICB0ZiA9IFR5cGVmYWNlLmNyZWF0ZShhRiwgVHlwZWZhY2UuTk9STUFMKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEZvbnQuQk9MRDoKLSAgICAgICAgICAgIHRmID0gVHlwZWZhY2UuY3JlYXRlKGFGLCBUeXBlZmFjZS5CT0xEKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEZvbnQuSVRBTElDOgotICAgICAgICAgICAgdGYgPSBUeXBlZmFjZS5jcmVhdGUoYUYsIFR5cGVmYWNlLklUQUxJQyk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBGb250LkJPTEQgfCBGb250LklUQUxJQzoKLSAgICAgICAgICAgIHRmID0gVHlwZWZhY2UuY3JlYXRlKGFGLCBUeXBlZmFjZS5CT0xEX0lUQUxJQyk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHRmID0gVHlwZWZhY2UuREVGQVVMVDsKLSAgICAgICAgfQotCi0gICAgICAgIG1QLnNldFRleHRTaXplKGZvbnQuZ2V0U2l6ZSgpKTsKLSAgICAgICAgbVAuc2V0VHlwZWZhY2UodGYpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRyYXdCeXRlcyhieXRlW10gZGF0YSwgaW50IG9mZnNldCwgaW50IGxlbmd0aCwgaW50IHgsIGludCB5KSB7Ci0gICAgICAgIGRyYXdTdHJpbmcobmV3IFN0cmluZyhkYXRhLCBvZmZzZXQsIGxlbmd0aCksIHgsIHkpOwotICAgIH0KLSAgICAKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3UG9seWdvbihQb2x5Z29uIHApIHsKLSAgICAgICAgZHJhd1BvbHlnb24ocC54cG9pbnRzLCBwLnlwb2ludHMsIHAubnBvaW50cyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZmlsbFBvbHlnb24oUG9seWdvbiBwKSB7Ci0gICAgICAgIGZpbGxQb2x5Z29uKHAueHBvaW50cywgcC55cG9pbnRzLCBwLm5wb2ludHMpOwotICAgIH0KLSAgICAKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldENsaXBCb3VuZHMoUmVjdGFuZ2xlIHIpIHsKLSAgICAgICAgU2hhcGUgY2xpcCA9IGdldENsaXAoKTsKLSAgICAgICAgaWYgKGNsaXAgIT0gbnVsbCkgewotICAgICAgICAgICAgUmVjdGFuZ2xlIGIgPSBjbGlwLmdldEJvdW5kcygpOwotICAgICAgICAgICAgci54ID0gYi54OwotICAgICAgICAgICAgci55ID0gYi55OwotICAgICAgICAgICAgci53aWR0aCA9IGIud2lkdGg7Ci0gICAgICAgICAgICByLmhlaWdodCA9IGIuaGVpZ2h0OwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiByOwotICAgIH0KLSAgICAKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBoaXRDbGlwKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIHJldHVybiBnZXRDbGlwQm91bmRzKCkuaW50ZXJzZWN0cyhuZXcgUmVjdGFuZ2xlKHgsIHksIHdpZHRoLCBoZWlnaHQpKTsKLSAgICB9Ci0gICAgCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd0NoYXJzKGNoYXJbXSBkYXRhLCBpbnQgb2Zmc2V0LCBpbnQgbGVuZ3RoLCBpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgbUMuZHJhd1RleHQoZGF0YSwgb2Zmc2V0LCBsZW5ndGgsIHgsIHksIG1QKTsKLSAgICB9Ci0gICAgCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGFpbnRNb2RlKCkgewotICAgICAgICBpZiAobVAgPT0gbnVsbCkgewotICAgICAgICAgICAgbVAgPSBuZXcgUGFpbnQoKTsKLSAgICAgICAgfQotICAgICAgICBtUC5zZXRYZmVybW9kZShudWxsKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRYT1JNb2RlKENvbG9yIGNvbG9yKSB7Ci0gICAgICAgIGlmIChtUCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBtUCA9IG5ldyBQYWludCgpOwotICAgICAgICB9Ci0gICAgICAgIG1QLnNldFhmZXJtb2RlKG5ldyBQaXhlbFhvclhmZXJtb2RlKGNvbG9yLmdldFJHQigpKSk7Ci0gICAgfQotICAgIAotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGZpbGwzRFJlY3QoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQsIGJvb2xlYW4gcmFpc2VkKSB7Ci0gICAgICAgIENvbG9yIGNvbG9yID0gZ2V0Q29sb3IoKTsKLSAgICAgICAgQ29sb3IgY29sb3JVcCwgY29sb3JEb3duOwotICAgICAgICBpZiAocmFpc2VkKSB7Ci0gICAgICAgICAgICBjb2xvclVwID0gY29sb3IuYnJpZ2h0ZXIoKTsKLSAgICAgICAgICAgIGNvbG9yRG93biA9IGNvbG9yLmRhcmtlcigpOwotICAgICAgICAgICAgc2V0Q29sb3IoY29sb3IpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgY29sb3JVcCA9IGNvbG9yLmRhcmtlcigpOwotICAgICAgICAgICAgY29sb3JEb3duID0gY29sb3IuYnJpZ2h0ZXIoKTsKLSAgICAgICAgICAgIHNldENvbG9yKGNvbG9yVXApOwotICAgICAgICB9Ci0KLSAgICAgICAgd2lkdGgtLTsKLSAgICAgICAgaGVpZ2h0LS07Ci0gICAgICAgIGZpbGxSZWN0KHgrMSwgeSsxLCB3aWR0aC0xLCBoZWlnaHQtMSk7Ci0KLSAgICAgICAgc2V0Q29sb3IoY29sb3JVcCk7Ci0gICAgICAgIGZpbGxSZWN0KHgsIHksIHdpZHRoLCAxKTsKLSAgICAgICAgZmlsbFJlY3QoeCwgeSsxLCAxLCBoZWlnaHQpOwotCi0gICAgICAgIHNldENvbG9yKGNvbG9yRG93bik7Ci0gICAgICAgIGZpbGxSZWN0KHgrd2lkdGgsIHksIDEsIGhlaWdodCk7Ci0gICAgICAgIGZpbGxSZWN0KHgrMSwgeStoZWlnaHQsIHdpZHRoLCAxKTsKLSAgICB9Ci0gICAgCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhdzNEUmVjdChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCwgYm9vbGVhbiByYWlzZWQpIHsKLSAgICAgICAgQ29sb3IgY29sb3IgPSBnZXRDb2xvcigpOwotICAgICAgICBDb2xvciBjb2xvclVwLCBjb2xvckRvd247Ci0gICAgICAgIGlmIChyYWlzZWQpIHsKLSAgICAgICAgICAgIGNvbG9yVXAgPSBjb2xvci5icmlnaHRlcigpOwotICAgICAgICAgICAgY29sb3JEb3duID0gY29sb3IuZGFya2VyKCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBjb2xvclVwID0gY29sb3IuZGFya2VyKCk7Ci0gICAgICAgICAgICBjb2xvckRvd24gPSBjb2xvci5icmlnaHRlcigpOwotICAgICAgICB9Ci0KLSAgICAgICAgc2V0Q29sb3IoY29sb3JVcCk7Ci0gICAgICAgIGZpbGxSZWN0KHgsIHksIHdpZHRoLCAxKTsKLSAgICAgICAgZmlsbFJlY3QoeCwgeSsxLCAxLCBoZWlnaHQpOwotCi0gICAgICAgIHNldENvbG9yKGNvbG9yRG93bik7Ci0gICAgICAgIGZpbGxSZWN0KHgrd2lkdGgsIHksIDEsIGhlaWdodCk7Ci0gICAgICAgIGZpbGxSZWN0KHgrMSwgeStoZWlnaHQsIHdpZHRoLCAxKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBjb3B5QXJlYShDYW52YXMgY2FudmFzLCBpbnQgc3gsIGludCBzeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgZHgsIGludCBkeSkgewotICAgICAgICBzeCArPSBnZXRUcmFuc2Zvcm0oKS5nZXRUcmFuc2xhdGVYKCk7Ci0gICAgICAgIHN5ICs9IGdldFRyYW5zZm9ybSgpLmdldFRyYW5zbGF0ZVkoKTsKLQotICAgICAgICBOYXRpdmVVdGlscy5uYXRpdmVTY3JvbGxSZWN0KGNhbnZhcywKLSAgICAgICAgICAgICAgICBuZXcgUmVjdChzeCwgc3ksIHN4ICsgd2lkdGgsIHN5ICsgaGVpZ2h0KSwKLSAgICAgICAgICAgICAgICBkeCwgZHkpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvQW5kcm9pZEdyYXBoaWNzQ29uZmlndXJhdGlvbi5qYXZhIGIvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9BbmRyb2lkR3JhcGhpY3NDb25maWd1cmF0aW9uLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBjODg4Y2QuLjAwMDAwMDAKLS0tIGEvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9BbmRyb2lkR3JhcGhpY3NDb25maWd1cmF0aW9uLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw5NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgY29tLmFuZHJvaWQuaW50ZXJuYWwuYXd0OwotCi1pbXBvcnQgY29tLmFuZHJvaWQuaW50ZXJuYWwuYXd0LkFuZHJvaWRHcmFwaGljczJEOwotCi1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3NDb25maWd1cmF0aW9uOwotaW1wb3J0IGphdmEuYXd0LkdyYXBoaWNzRGV2aWNlOwotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuVm9sYXRpbGVJbWFnZTsKLQotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuQ2FudmFzOwotCi1wdWJsaWMgY2xhc3MgQW5kcm9pZEdyYXBoaWNzQ29uZmlndXJhdGlvbiBleHRlbmRzIEdyYXBoaWNzQ29uZmlndXJhdGlvbiB7Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgQnVmZmVyZWRJbWFnZSBjcmVhdGVDb21wYXRpYmxlSW1hZ2UoaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIC8vIFRPRE8gQXV0by1nZW5lcmF0ZWQgbWV0aG9kIHN0dWIKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UgY3JlYXRlQ29tcGF0aWJsZUltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCwKLSAgICAgICAgICAgIGludCB0cmFuc3BhcmVuY3kpIHsKLSAgICAgICAgLy8gVE9ETyBBdXRvLWdlbmVyYXRlZCBtZXRob2Qgc3R1YgotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgVm9sYXRpbGVJbWFnZSBjcmVhdGVDb21wYXRpYmxlVm9sYXRpbGVJbWFnZShpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgLy8gVE9ETyBBdXRvLWdlbmVyYXRlZCBtZXRob2Qgc3R1YgotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgVm9sYXRpbGVJbWFnZSBjcmVhdGVDb21wYXRpYmxlVm9sYXRpbGVJbWFnZShpbnQgd2lkdGgsIGludCBoZWlnaHQsCi0gICAgICAgICAgICBpbnQgdHJhbnNwYXJlbmN5KSB7Ci0gICAgICAgIC8vIFRPRE8gQXV0by1nZW5lcmF0ZWQgbWV0aG9kIHN0dWIKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFJlY3RhbmdsZSBnZXRCb3VuZHMoKSB7Ci0gICAgICAgIENhbnZhcyBjID0gQW5kcm9pZEdyYXBoaWNzMkQuZ2V0QW5kcm9pZENhbnZhcygpOwotICAgICAgICBpZihjICE9IG51bGwpCi0gICAgICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZSgwLCAwLCBjLmdldFdpZHRoKCksIGMuZ2V0SGVpZ2h0KCkpOwotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgQ29sb3JNb2RlbCBnZXRDb2xvck1vZGVsKCkgewotICAgICAgICAvLyBUT0RPIEF1dG8tZ2VuZXJhdGVkIG1ldGhvZCBzdHViCi0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoaW50IHRyYW5zcGFyZW5jeSkgewotICAgICAgICAvLyBUT0RPIEF1dG8tZ2VuZXJhdGVkIG1ldGhvZCBzdHViCi0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0gZ2V0RGVmYXVsdFRyYW5zZm9ybSgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBBZmZpbmVUcmFuc2Zvcm0oKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR3JhcGhpY3NEZXZpY2UgZ2V0RGV2aWNlKCkgewotICAgICAgICAvLyBUT0RPIEF1dG8tZ2VuZXJhdGVkIG1ldGhvZCBzdHViCi0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0gZ2V0Tm9ybWFsaXppbmdUcmFuc2Zvcm0oKSB7Ci0gICAgICAgIC8vIFRPRE8gQXV0by1nZW5lcmF0ZWQgbWV0aG9kIHN0dWIKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvY29tL2FuZHJvaWQvaW50ZXJuYWwvYXd0L0FuZHJvaWRHcmFwaGljc0ZhY3RvcnkuamF2YSBiL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvQW5kcm9pZEdyYXBoaWNzRmFjdG9yeS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjYTI1NWI1Li4wMDAwMDAwCi0tLSBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvQW5kcm9pZEdyYXBoaWNzRmFjdG9yeS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsODcgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGNvbS5hbmRyb2lkLmludGVybmFsLmF3dDsKLQotaW1wb3J0IGphdmEuYXd0LkZvbnQ7Ci1pbXBvcnQgamF2YS5hd3QuRm9udE1ldHJpY3M7Ci1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3MyRDsKLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljc0Vudmlyb25tZW50OwotaW1wb3J0IGphdmEuYXd0LnBlZXIuRm9udFBlZXI7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLk11bHRpUmVjdEFyZWE7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250LkFuZHJvaWRGb250OwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5Gb250TWFuYWdlcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuRm9udE1ldHJpY3NJbXBsOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5BbmRyb2lkRm9udE1hbmFnZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuTmF0aXZlV2luZG93OwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLldpbmRvd0ZhY3Rvcnk7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5Db21tb25HcmFwaGljczJERmFjdG9yeTsKLQotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuQ2FudmFzOwotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuUGFpbnQ7Ci1pbXBvcnQgYW5kcm9pZC5jb250ZW50LkNvbnRleHQ7Ci0KLXB1YmxpYyBjbGFzcyBBbmRyb2lkR3JhcGhpY3NGYWN0b3J5IGV4dGVuZHMgQ29tbW9uR3JhcGhpY3MyREZhY3RvcnkgewotICAgIAotICAgIHB1YmxpYyBHcmFwaGljc0Vudmlyb25tZW50IGNyZWF0ZUdyYXBoaWNzRW52aXJvbm1lbnQoV2luZG93RmFjdG9yeSB3ZikgewotICAgICAgICAvLyBUT0RPIEF1dG8tZ2VuZXJhdGVkIG1ldGhvZCBzdHViCi0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIHB1YmxpYyBGb250IGVtYmVkRm9udChTdHJpbmcgZm9udEZpbGVQYXRoKSB7Ci0gICAgICAgIC8vIFRPRE8gQXV0by1nZW5lcmF0ZWQgbWV0aG9kIHN0dWIKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgcHVibGljIEZvbnRNYW5hZ2VyIGdldEZvbnRNYW5hZ2VyKCkgewotICAgICAgICByZXR1cm4gQW5kcm9pZEZvbnRNYW5hZ2VyLmluc3Q7Ci0gICAgfQotCi0gICAgcHVibGljIEZvbnRNZXRyaWNzIGdldEZvbnRNZXRyaWNzKEZvbnQgZm9udCkgewotICAgICAgICByZXR1cm4gbmV3IEZvbnRNZXRyaWNzSW1wbChmb250KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgRm9udFBlZXIgZ2V0Rm9udFBlZXIoRm9udCBmb250KSB7Ci0gICAgICAgIC8vcmV0dXJuIGdldEZvbnRNYW5hZ2VyKCkuZ2V0Rm9udFBlZXIoZm9udC5nZXROYW1lKCksIGZvbnQuZ2V0U3R5bGUoKSwgZm9udC5nZXRTaXplKCkpOwotICAgICAgICByZXR1cm4gbmV3IEFuZHJvaWRGb250KGZvbnQuZ2V0TmFtZSgpLCBmb250LmdldFN0eWxlKCksIGZvbnQuZ2V0U2l6ZSgpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgR3JhcGhpY3MyRCBnZXRHcmFwaGljczJEKE5hdGl2ZVdpbmRvdyB3aW4sIGludCB0cmFuc2xhdGVYLAotICAgICAgICAgICAgaW50IHRyYW5zbGF0ZVksIE11bHRpUmVjdEFyZWEgY2xpcCkgewotICAgICAgICAvLyBUT0RPIEF1dG8tZ2VuZXJhdGVkIG1ldGhvZCBzdHViCi0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIHB1YmxpYyBHcmFwaGljczJEIGdldEdyYXBoaWNzMkQoTmF0aXZlV2luZG93IHdpbiwgaW50IHRyYW5zbGF0ZVgsCi0gICAgICAgICAgICBpbnQgdHJhbnNsYXRlWSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIC8vIFRPRE8gQXV0by1nZW5lcmF0ZWQgbWV0aG9kIHN0dWIKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgcHVibGljIEdyYXBoaWNzMkQgZ2V0R3JhcGhpY3MyRChDb250ZXh0IGN0eCwgQ2FudmFzIGMsIFBhaW50IHApIHsKLSAgICAgICAgcmV0dXJuIEFuZHJvaWRHcmFwaGljczJELmdldEluc3RhbmNlKGN0eCwgYywgcCk7Ci0gICAgfQotCi0gICAgcHVibGljIEdyYXBoaWNzMkQgZ2V0R3JhcGhpY3MyRChDYW52YXMgYywgUGFpbnQgcCkgewotICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCEiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgR3JhcGhpY3MyRCBnZXRHcmFwaGljczJEKCkgewotICAgICAgICByZXR1cm4gQW5kcm9pZEdyYXBoaWNzMkQuZ2V0SW5zdGFuY2UoKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvQW5kcm9pZEltYWdlRGVjb2Rlci5qYXZhIGIvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9BbmRyb2lkSW1hZ2VEZWNvZGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDgxYjJlMWEuLjAwMDAwMDAKLS0tIGEvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9BbmRyb2lkSW1hZ2VEZWNvZGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNzQgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotcGFja2FnZSBjb20uYW5kcm9pZC5pbnRlcm5hbC5hd3Q7Ci0KLWltcG9ydCBhbmRyb2lkLmdyYXBoaWNzLkJpdG1hcDsKLWltcG9ydCBhbmRyb2lkLmdyYXBoaWNzLkJpdG1hcEZhY3Rvcnk7Ci0KLWltcG9ydCBqYXZhLmF3dC5UcmFuc3BhcmVuY3k7Ci1pbXBvcnQgamF2YS5hd3QuY29sb3IuQ29sb3JTcGFjZTsKLS8vaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db21wb25lbnRDb2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGlyZWN0Q29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbWFnZUNvbnN1bWVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkluZGV4Q29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS51dGlsLkhhc2h0YWJsZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2UuRGVjb2RpbmdJbWFnZVNvdXJjZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmltYWdlLkltYWdlRGVjb2RlcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotcHVibGljIGNsYXNzIEFuZHJvaWRJbWFnZURlY29kZXIgZXh0ZW5kcyBJbWFnZURlY29kZXIgewotICAgIAotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBoaW50ZmxhZ3MgPQotICAgICAgICBJbWFnZUNvbnN1bWVyLlNJTkdMRUZSQU1FIHwgLy8gUE5HIGlzIGEgc3RhdGljIGltYWdlCi0gICAgICAgIEltYWdlQ29uc3VtZXIuVE9QRE9XTkxFRlRSSUdIVCB8IC8vIFRoaXMgb3JkZXIgaXMgb25seSBvbmUgcG9zc2libGUKLSAgICAgICAgSW1hZ2VDb25zdW1lci5DT01QTEVURVNDQU5MSU5FUzsgLy8gRG9uJ3QgZGVsaXZlciBpbmNvbXBsZXRlIHNjYW5saW5lcwotICAgIAotICAgIC8vIEVhY2ggcGl4ZWwgaXMgYSBncmF5c2NhbGUgc2FtcGxlLgotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBQTkdfQ09MT1JfVFlQRV9HUkFZID0gMDsKLSAgICAvLyBFYWNoIHBpeGVsIGlzIGFuIFIsRyxCIHRyaXBsZS4KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgUE5HX0NPTE9SX1RZUEVfUkdCID0gMjsKLSAgICAvLyBFYWNoIHBpeGVsIGlzIGEgcGFsZXR0ZSBpbmRleCwgYSBQTFRFIGNodW5rIG11c3QgYXBwZWFyLgotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBQTkdfQ09MT1JfVFlQRV9QTFRFID0gMzsKLSAgICAvLyBFYWNoIHBpeGVsIGlzIGEgZ3JheXNjYWxlIHNhbXBsZSwgZm9sbG93ZWQgYnkgYW4gYWxwaGEgc2FtcGxlLgotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBQTkdfQ09MT1JfVFlQRV9HUkFZX0FMUEhBID0gNDsKLSAgICAvLyBFYWNoIHBpeGVsIGlzIGFuIFIsRyxCIHRyaXBsZSwgZm9sbG93ZWQgYnkgYW4gYWxwaGEgc2FtcGxlLgotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBQTkdfQ09MT1JfVFlQRV9SR0JBID0gNjsKLSAgICAKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgTkJfT0ZfTElORVNfUEVSX0NIVU5LID0gMTsgIC8vIDAgPSBmdWxsIGltYWdlCi0gICAgCi0gICAgQml0bWFwIGJtOyAgLy8gVGhlIGltYWdlIGFzIGRlY29kZWQgYnkgQW5kcm9pZAotICAgIAotICAgIC8vIEhlYWRlciBpbmZvcm1hdGlvbgotICAgIGludCBpbWFnZVdpZHRoOyAvLyBJbWFnZSBzaXplCi0gICAgaW50IGltYWdlSGVpZ2h0OyAgCi0gICAgaW50IGNvbG9yVHlwZTsgIC8vIE9uZSBvZiB0aGUgUE5HXyBjb25zdGFudHMgZnJvbSBhYm92ZQotICAgIGludCBiaXREZXB0aDsgICAvLyBOdW1iZXIgb2YgYml0cyBwZXIgY29sb3IKLSAgICBieXRlIGNtYXBbXTsgICAgLy8gVGhlIGNvbG9yIHBhbGV0dGUgZm9yIGluZGV4IGNvbG9yIG1vZGVscwotICAgIENvbG9yTW9kZWwgbW9kZWw7ICAvLyBUaGUgY29ycmVzcG9uZGluZyBBV1QgY29sb3IgbW9kZWwKLSAgICAKLSAgICBib29sZWFuIHRyYW5zZmVySW50czsgLy8gSXMgdHJhbnNmZXIgb2YgdHlwZSBpbnQgb3IgYnl0ZT8KLSAgICBpbnQgZGF0YUVsZW1lbnRzUGVyUGl4ZWw7Ci0KLSAgICAvLyBCdWZmZXJzIGZvciBkZWNvZGVkIGltYWdlIGRhdGEKLSAgICBieXRlIGJ5dGVPdXRbXTsKLSAgICBpbnQgaW50T3V0W107Ci0KLSAgICAKLSAgICBwdWJsaWMgQW5kcm9pZEltYWdlRGVjb2RlcihEZWNvZGluZ0ltYWdlU291cmNlIHNyYywgSW5wdXRTdHJlYW0gaXMpIHsKLSAgICAgICAgc3VwZXIoc3JjLCBpcyk7Ci0gICAgICAgIGRhdGFFbGVtZW50c1BlclBpeGVsID0gMTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICAvKioKLSAgICAgKiBBbGwgdGhlIGRlY29kaW5nIGlzIGRvbmUgaW4gQW5kcm9pZAotICAgICAqIAotICAgICAqIEFXVD8/PzogTWV0aG9kIHJldHVybnMgb25seSBvbmNlIHRoZSBpbWFnZSBpcyBjb21wbGV0bHkgCi0gICAgICogZGVjb2RlZDsgZGVjb2RpbmcgaXMgbm90IGRvbmUgYXN5bmNocm9ub3VzbHkKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBkZWNvZGVJbWFnZSgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBibSA9IEJpdG1hcEZhY3RvcnkuZGVjb2RlU3RyZWFtKGlucHV0U3RyZWFtKTsKLSAgICAgICAgICAgIGlmIChibSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElPRXhjZXB0aW9uKCJJbnB1dCBzdHJlYW0gZW1wdHkgYW5kIG5vIGltYWdlIGNhY2hlZCIpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBDaGVjayBzaXplCi0gICAgICAgICAgICBpbWFnZVdpZHRoID0gYm0uZ2V0V2lkdGgoKTsKLSAgICAgICAgICAgIGltYWdlSGVpZ2h0ID0gYm0uZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICBpZiAoaW1hZ2VXaWR0aCA8IDAgfHwgaW1hZ2VIZWlnaHQgPCAwICkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJJbGxlZ2FsIGltYWdlIHNpemU6ICIgCi0gICAgICAgICAgICAgICAgICAgICAgICArIGltYWdlV2lkdGggKyAiLCAiICsgaW1hZ2VIZWlnaHQpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyBXZSBnb3QgdGhlIGltYWdlIGZ1bGx5IGRlY29kZWQ7IG5vdyBzZW5kIGFsbCBpbWFnZSBkYXRhIHRvIEFXVAotICAgICAgICAgICAgc2V0RGltZW5zaW9ucyhpbWFnZVdpZHRoLCBpbWFnZUhlaWdodCk7Ci0gICAgICAgICAgICBtb2RlbCA9IGNyZWF0ZUNvbG9yTW9kZWwoKTsKLSAgICAgICAgICAgIHNldENvbG9yTW9kZWwobW9kZWwpOwotICAgICAgICAgICAgc2V0SGludHMoaGludGZsYWdzKTsKLSAgICAgICAgICAgIHNldFByb3BlcnRpZXMobmV3IEhhc2h0YWJsZTxPYmplY3QsIE9iamVjdD4oKSk7IC8vIEVtcHR5Ci0gICAgICAgICAgICBzZW5kUGl4ZWxzKE5CX09GX0xJTkVTX1BFUl9DSFVOSyAhPSAwID8gTkJfT0ZfTElORVNfUEVSX0NIVU5LIDogaW1hZ2VIZWlnaHQpOwotICAgICAgICAgICAgaW1hZ2VDb21wbGV0ZShJbWFnZUNvbnN1bWVyLlNUQVRJQ0lNQUdFRE9ORSk7ICAgICAgICAKLSAgICAgICAgfSBjYXRjaCAoSU9FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgdGhyb3cgZTsKLSAgICAgICAgfSBjYXRjaCAoUnVudGltZUV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICBpbWFnZUNvbXBsZXRlKEltYWdlQ29uc3VtZXIuSU1BR0VFUlJPUik7Ci0gICAgICAgICAgICB0aHJvdyBlOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgY2xvc2VTdHJlYW0oKTsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICAvKioKLSAgICAgKiBDcmVhdGUgdGhlIEFXVCBjb2xvciBtb2RlbAotICAgICAqCi0gICAgICogPz8/QVdUOiBBbmRyb2lkIEJpdG1hcHMgYXJlIGFsd2F5cyBvZiB0eXBlOiBBUkdCLTg4ODgtRGlyZWN0IGNvbG9yIG1vZGVsCi0gICAgICogCi0gICAgICogSG93ZXZlciwgd2UgbGVhdmUgdGhlIGNvZGUgaGVyZSBmb3IgYSBtb3JlIHBvd2VyZnVsbCBkZWNvZGVyIAotICAgICAqIHRoYXQgcmV0dXJucyBhIG5hdGl2ZSBtb2RlbCwgYW5kIHRoZSBjb252ZXJzaW9uIGlzIHRoZW4gaGFuZGxlZAotICAgICAqIGluIEFXVC4gV2l0aCBzdWNoIGEgZGVjb2Rlciwgd2Ugd291bGQgbmVlZCB0byBnZXQgdGhlIGNvbG9yVHlwZSwgCi0gICAgICogdGhlIGJpdERlcHRoLCAoYW5kIHRoZSBjb2xvciBwYWxldHRlIGZvciBhbiBpbmRleCBjb2xvciBtb2RlbCkKLSAgICAgKiBmcm9tIHRoZSBpbWFnZSBhbmQgY29uc3RydWN0IHRoZSBjb3JyZWN0IGNvbG9yIG1vZGVsIGhlcmUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBDb2xvck1vZGVsIGNyZWF0ZUNvbG9yTW9kZWwoKSB7Ci0gICAgICAgIENvbG9yTW9kZWwgY20gPSBudWxsOwotICAgICAgICBpbnQgYm1Nb2RlbCA9IDU7IC8vIFRPRE8gVGhpcyBkb2Vzbid0IGV4aXN0OiBibS5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIGNtYXAgPSBudWxsOwotICAgICAgICAgICAKLSAgICAgICAgc3dpdGNoIChibU1vZGVsKSB7Ci0gICAgICAgIC8vIEExX01PREVMCi0gICAgICAgIGNhc2UgMTogCi0gICAgICAgICAgICBjb2xvclR5cGUgPSBQTkdfQ09MT1JfVFlQRV9HUkFZOwotICAgICAgICAgICAgYml0RGVwdGggPSAxOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAKLSAgICAgICAgLy8gQThfTU9ERUwKLSAgICAgICAgY2FzZSAyOgotICAgICAgICAgICAgY29sb3JUeXBlID0gUE5HX0NPTE9SX1RZUEVfR1JBWV9BTFBIQTsKLSAgICAgICAgICAgIGJpdERlcHRoID0gODsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgCi0gICAgICAgIC8vIElOREVYOF9NT0RFTAotICAgICAgICAvLyBSR0JfNTY1X01PREVMCi0gICAgICAgIC8vIEFSR0JfODg4OF9NT0RFTAotICAgICAgICBjYXNlIDM6Ci0gICAgICAgIGNhc2UgNDogCi0gICAgICAgIGNhc2UgNTogCi0gICAgICAgICAgICBjb2xvclR5cGUgPSBibS5oYXNBbHBoYSgpID8gUE5HX0NPTE9SX1RZUEVfUkdCQSA6IFBOR19DT0xPUl9UWVBFX1JHQjsKLSAgICAgICAgICAgIGJpdERlcHRoID0gODsKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAvLyBhd3QuM0M9VW5rbm93biBQTkcgY29sb3IgdHlwZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4zQyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBzd2l0Y2ggKGNvbG9yVHlwZSkgewotICAgICAgICAKLSAgICAgICAgICAgIGNhc2UgUE5HX0NPTE9SX1RZUEVfR1JBWTogewotICAgICAgICAgICAgICAgIGlmIChiaXREZXB0aCAhPSA4ICYmIGJpdERlcHRoICE9IDQgJiYgYml0RGVwdGggIT0gMiAmJiAgYml0RGVwdGggIT0gMSkgewotICAgICAgICAgICAgICAgICAgICAvLyBhd3QuM0M9VW5rbm93biBQTkcgY29sb3IgdHlwZQotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjNDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgLy8gQ3JlYXRlIGdyYXkgY29sb3IgbW9kZWwKLSAgICAgICAgICAgICAgICBpbnQgbnVtRW50cmllcyA9IDEgPDwgYml0RGVwdGg7Ci0gICAgICAgICAgICAgICAgaW50IHNjYWxlRmFjdG9yID0gMjU1IC8gKG51bUVudHJpZXMtMSk7Ci0gICAgICAgICAgICAgICAgYnl0ZSBjb21wc1tdID0gbmV3IGJ5dGVbbnVtRW50cmllc107Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1FbnRyaWVzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgY29tcHNbaV0gPSAoYnl0ZSkgKGkgKiBzY2FsZUZhY3Rvcik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGNtID0gbmV3IEluZGV4Q29sb3JNb2RlbChiaXREZXB0aCwgbnVtRW50cmllcywgY29tcHMsIGNvbXBzLCBjb21wcyk7Ci0KLSAgICAgICAgICAgICAgICB0cmFuc2ZlckludHMgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY2FzZSBQTkdfQ09MT1JfVFlQRV9SR0I6IHsKLSAgICAgICAgICAgICAgICBpZiAoYml0RGVwdGggIT0gOCkgewotICAgICAgICAgICAgICAgICAgICAvLyBhd3QuM0M9VW5rbm93biBQTkcgY29sb3IgdHlwZQotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjNDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIGNtID0gbmV3IERpcmVjdENvbG9yTW9kZWwoMjQsIDB4ZmYwMDAwLCAweEZGMDAsIDB4RkYpOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIHRyYW5zZmVySW50cyA9IHRydWU7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNhc2UgUE5HX0NPTE9SX1RZUEVfUExURTogewotICAgICAgICAgICAgICAgIGlmIChiaXREZXB0aCAhPSA4ICYmIGJpdERlcHRoICE9IDQgJiYgYml0RGVwdGggIT0gMiAmJiBiaXREZXB0aCAhPSAxKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIGF3dC4zQz1Vbmtub3duIFBORyBjb2xvciB0eXBlCi0gICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuM0MiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBpZiAoY21hcCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oIlBhbGV0dGUgY29sb3IgdHlwZSBpcyBub3Qgc3VwcG9ydGVkIik7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgY20gPSBuZXcgSW5kZXhDb2xvck1vZGVsKGJpdERlcHRoLCBjbWFwLmxlbmd0aCAvIDMsIGNtYXAsIDAsIGZhbHNlKTsKLQotICAgICAgICAgICAgICAgIHRyYW5zZmVySW50cyA9IGZhbHNlOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIFBOR19DT0xPUl9UWVBFX0dSQVlfQUxQSEE6IHsKLSAgICAgICAgICAgICAgICBpZiAoYml0RGVwdGggIT0gOCkgewotICAgICAgICAgICAgICAgICAgICAvLyBhd3QuM0M9VW5rbm93biBQTkcgY29sb3IgdHlwZQotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjNDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgY20gPSBuZXcgQ29tcG9uZW50Q29sb3JNb2RlbChDb2xvclNwYWNlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1NfR1JBWSksCi0gICAgICAgICAgICAgICAgICAgICAgICB0cnVlLCBmYWxzZSwKLSAgICAgICAgICAgICAgICAgICAgICAgIFRyYW5zcGFyZW5jeS5UUkFOU0xVQ0VOVCwKLSAgICAgICAgICAgICAgICAgICAgICAgIERhdGFCdWZmZXIuVFlQRV9CWVRFKTsKLQotICAgICAgICAgICAgICAgIHRyYW5zZmVySW50cyA9IGZhbHNlOwotICAgICAgICAgICAgICAgIGRhdGFFbGVtZW50c1BlclBpeGVsID0gMjsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY2FzZSBQTkdfQ09MT1JfVFlQRV9SR0JBOiB7Ci0gICAgICAgICAgICAgICAgaWYgKGJpdERlcHRoICE9IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjNDPVVua25vd24gUE5HIGNvbG9yIHR5cGUKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4zQyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGNtID0gQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCk7Ci0KLSAgICAgICAgICAgICAgICB0cmFuc2ZlckludHMgPSB0cnVlOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAvLyBhd3QuM0M9VW5rbm93biBQTkcgY29sb3IgdHlwZQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuM0MiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgcmV0dXJuIGNtOwotICAgIH0KLSAgICAKLSAgICBwcml2YXRlIHZvaWQgc2VuZFBpeGVscyhpbnQgbmJPZkxpbmVzUGVyQ2h1bmspIHsKLSAgICAgICAgaW50IHcgPSBpbWFnZVdpZHRoOwotICAgICAgICBpbnQgaCA9IGltYWdlSGVpZ2h0OwotICAgICAgICBpbnQgbiA9IDE7Ci0gICAgICAgIGlmIChuYk9mTGluZXNQZXJDaHVuayA+IDAgJiYgbmJPZkxpbmVzUGVyQ2h1bmsgPD0gaCkgewotICAgICAgICAgICAgbiA9IG5iT2ZMaW5lc1BlckNodW5rOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBpZiAodHJhbnNmZXJJbnRzKSB7Ci0gICAgICAgICAgICAvLyBDcmVhdGUgb3V0cHV0IGJ1ZmZlcgotICAgICAgICAgICAgaW50T3V0ID0gbmV3IGludFt3ICogbl07Ci0gICAgICAgICAgICBmb3IgKGludCB5aSA9IDA7IHlpIDwgaDsgeWkgKz0gbikgewotICAgICAgICAgICAgICAgIC8vIExhc3QgY2h1bmsgbWlnaHQgY29udGFpbiBsZXNzIGxpbmVzcwotICAgICAgICAgICAgICAgIGlmIChuID4gMSAmJiBoIC0geWkgPCBuICkgewotICAgICAgICAgICAgICAgICAgICBuID0gaCAtIHlpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBibS5nZXRQaXhlbHMoaW50T3V0LCAwLCB3LCAwLCB5aSwgdywgbik7Ci0gICAgICAgICAgICAgICAgc2V0UGl4ZWxzKDAsIHlpLCB3LCBuLCBtb2RlbCwgaW50T3V0LCAwLCB3KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIEFuZHJvaWQgYml0bWFwcyBhbHdheXMgc3RvcmUgaW50cyAoQVJHQi04ODg4IGRpcmVjdCBtb2RlbCkKLSAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJCeXRlIHRyYW5zZmVyIG5vdCBzdXBwb3J0ZWQiKTsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLX0KZGlmZiAtLWdpdCBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvQW5kcm9pZEphdmFCbGl0dGVyLmphdmEgYi9hd3QvY29tL2FuZHJvaWQvaW50ZXJuYWwvYXd0L0FuZHJvaWRKYXZhQmxpdHRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0MjNiNTM0Li4wMDAwMDAwCi0tLSBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvQW5kcm9pZEphdmFCbGl0dGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1MzYgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGNvbS5hbmRyb2lkLmludGVybmFsLmF3dDsKLQotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuQml0bWFwOwotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuQ2FudmFzOwotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuTWF0cml4OwotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuUGFpbnQ7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5NdWx0aVJlY3RBcmVhOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuU3VyZmFjZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLlhPUkNvbXBvc2l0ZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLnJlbmRlci5CbGl0dGVyOwotCi1pbXBvcnQgamF2YS5hd3QuKjsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGF0YUJ1ZmZlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EYXRhQnVmZmVySW50OwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJhc3RlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Xcml0YWJsZVJhc3RlcjsKLQotcHVibGljIGNsYXNzIEFuZHJvaWRKYXZhQmxpdHRlciBpbXBsZW1lbnRzIEJsaXR0ZXIgewotCi0gICAgcHJpdmF0ZSBDYW52YXMgY2FudmFzOwotICAgIHByaXZhdGUgUGFpbnQgcGFpbnQ7Ci0gICAgcHJpdmF0ZSBpbnQgY29sb3JDYWNoZTsKLSAgICAgICAgCi0gICAgcHVibGljIEFuZHJvaWRKYXZhQmxpdHRlcihDYW52YXMgYykgewotICAgICAgICB0aGlzLmNhbnZhcyA9IGM7Ci0gICAgICAgIHRoaXMucGFpbnQgPSBuZXcgUGFpbnQoKTsKLSAgICAgICAgdGhpcy5wYWludC5zZXRTdHJva2VXaWR0aCgxKTsKLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogSW5zdGVhZCBvZiBtdWx0aXBsaWNhdGlvbiBhbmQgZGl2aXNpb24gd2UgYXJlIHVzaW5nIHZhbHVlcyBmcm9tCi0gICAgICogTG9va3VwIHRhYmxlcy4KLSAgICAgKi8KLSAgICBzdGF0aWMgYnl0ZSBtdWxMVVRbXVtdOyAvLyBMb29rdXAgdGFibGUgZm9yIG11bHRpcGxpY2F0aW9uCi0gICAgc3RhdGljIGJ5dGUgZGl2TFVUW11bXTsgLy8gTG9va3VwIHRhYmxlIGZvciBkaXZpc2lvbgotCi0gICAgc3RhdGljewotICAgICAgICBtdWxMVVQgPSBuZXcgYnl0ZVsyNTZdWzI1Nl07Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCAyNTY7IGkrKyl7Ci0gICAgICAgICAgICBmb3IoaW50IGogPSAwOyBqIDwgMjU2OyBqKyspewotICAgICAgICAgICAgICAgIG11bExVVFtpXVtqXSA9IChieXRlKSgoZmxvYXQpKGkgKiBqKS8yNTUgKyAwLjVmKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBkaXZMVVQgPSBuZXcgYnl0ZVsyNTZdWzI1Nl07Ci0gICAgICAgIGZvcihpbnQgaSA9IDE7IGkgPCAyNTY7IGkrKyl7Ci0gICAgICAgICAgICBmb3IoaW50IGogPSAwOyBqIDwgaTsgaisrKXsKLSAgICAgICAgICAgICAgICBkaXZMVVRbaV1bal0gPSAoYnl0ZSkoKChmbG9hdClqIC8gMjU1KSAvICgoZmxvYXQpaS8gMjU1KSAqIDI1NSArIDAuNWYpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZm9yKGludCBqID0gaTsgaiA8IDI1NjsgaisrKXsKLSAgICAgICAgICAgICAgICBkaXZMVVRbaV1bal0gPSAoYnl0ZSkyNTU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBmaW5hbCBzdGF0aWMgaW50IEFscGhhQ29tcG9zaXRlTW9kZSA9IDE7Ci0gICAgZmluYWwgc3RhdGljIGludCBYT1JNb2RlID0gMjsKLQotICAgIHB1YmxpYyB2b2lkIGJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBBZmZpbmVUcmFuc2Zvcm0gc3lzeGZvcm0sCi0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0sIENvbXBvc2l0ZSBjb21wLCBDb2xvciBiZ2NvbG9yLAotICAgICAgICAgICAgTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0gICAgICAgIAotICAgICAgICBpZih4Zm9ybSA9PSBudWxsKXsKLSAgICAgICAgICAgIGJsaXQoc3JjWCwgc3JjWSwgc3JjU3VyZiwgZHN0WCwgZHN0WSwgZHN0U3VyZiwgd2lkdGgsIGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgc3lzeGZvcm0sIGNvbXAsIGJnY29sb3IsIGNsaXApOwotICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgIGRvdWJsZSBzY2FsZVggPSB4Zm9ybS5nZXRTY2FsZVgoKTsKLSAgICAgICAgICAgIGRvdWJsZSBzY2FsZVkgPSB4Zm9ybS5nZXRTY2FsZVkoKTsKLSAgICAgICAgICAgIGRvdWJsZSBzY2FsZWRYID0gZHN0WCAvIHNjYWxlWDsKLSAgICAgICAgICAgIGRvdWJsZSBzY2FsZWRZID0gZHN0WSAvIHNjYWxlWTsKLSAgICAgICAgICAgIEFmZmluZVRyYW5zZm9ybSBhdCA9IG5ldyBBZmZpbmVUcmFuc2Zvcm0oKTsKLSAgICAgICAgICAgIGF0LnNldFRvVHJhbnNsYXRpb24oc2NhbGVkWCwgc2NhbGVkWSk7Ci0gICAgICAgICAgICB4Zm9ybS5jb25jYXRlbmF0ZShhdCk7Ci0gICAgICAgICAgICBzeXN4Zm9ybS5jb25jYXRlbmF0ZSh4Zm9ybSk7Ci0gICAgICAgICAgICBibGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIDAsIDAsIGRzdFN1cmYsIHdpZHRoLCBoZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgIHN5c3hmb3JtLCBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgcHVibGljIHZvaWQgYmxpdChpbnQgc3JjWCwgaW50IHNyY1ksIFN1cmZhY2Ugc3JjU3VyZiwgaW50IGRzdFgsIGludCBkc3RZLAotICAgICAgICAgICAgU3VyZmFjZSBkc3RTdXJmLCBpbnQgd2lkdGgsIGludCBoZWlnaHQsIEFmZmluZVRyYW5zZm9ybSBzeXN4Zm9ybSwKLSAgICAgICAgICAgIENvbXBvc2l0ZSBjb21wLCBDb2xvciBiZ2NvbG9yLCBNdWx0aVJlY3RBcmVhIGNsaXApIHsKLSAgICAgICAgCi0gICAgICAgIGlmKHN5c3hmb3JtID09IG51bGwpIHsKLSAgICAgICAgICAgIHN5c3hmb3JtID0gbmV3IEFmZmluZVRyYW5zZm9ybSgpOwotICAgICAgICB9Ci0gICAgICAgIGludCB0eXBlID0gc3lzeGZvcm0uZ2V0VHlwZSgpOwotICAgICAgICBzd2l0Y2godHlwZSl7Ci0gICAgICAgICAgICBjYXNlIEFmZmluZVRyYW5zZm9ybS5UWVBFX1RSQU5TTEFUSU9OOgotICAgICAgICAgICAgICAgIGRzdFggKz0gc3lzeGZvcm0uZ2V0VHJhbnNsYXRlWCgpOwotICAgICAgICAgICAgICAgIGRzdFkgKz0gc3lzeGZvcm0uZ2V0VHJhbnNsYXRlWSgpOwotICAgICAgICAgICAgY2FzZSBBZmZpbmVUcmFuc2Zvcm0uVFlQRV9JREVOVElUWToKLSAgICAgICAgICAgICAgICBzaW1wbGVCbGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIGRzdFgsIGRzdFksIGRzdFN1cmYsCi0gICAgICAgICAgICAgICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgaW50IHNyY1cgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgaW50IHNyY0ggPSBzcmNTdXJmLmdldEhlaWdodCgpOwotCi0gICAgICAgICAgICAgICAgaW50IHcgPSBzcmNYICsgd2lkdGggPCBzcmNXID8gd2lkdGggOiBzcmNXIC0gc3JjWDsKLSAgICAgICAgICAgICAgICBpbnQgaCA9IHNyY1kgKyBoZWlnaHQgPCBzcmNIID8gaGVpZ2h0IDogc3JjSCAtIHNyY1k7Ci0KLSAgICAgICAgICAgICAgICBDb2xvck1vZGVsIHNyY0NNID0gc3JjU3VyZi5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgICAgICAgICAgUmFzdGVyIHNyY1IgPSBzcmNTdXJmLmdldFJhc3RlcigpLmNyZWF0ZUNoaWxkKHNyY1gsIHNyY1ksCi0gICAgICAgICAgICAgICAgICAgICAgICB3LCBoLCAwLCAwLCBudWxsKTsKLQotICAgICAgICAgICAgICAgIENvbG9yTW9kZWwgZHN0Q00gPSBkc3RTdXJmLmdldENvbG9yTW9kZWwoKTsKLSAgICAgICAgICAgICAgICBXcml0YWJsZVJhc3RlciBkc3RSID0gZHN0U3VyZi5nZXRSYXN0ZXIoKTsKLQotICAgICAgICAgICAgICAgIHRyYW5zZm9ybWVkQmxpdChzcmNDTSwgc3JjUiwgMCwgMCwgZHN0Q00sIGRzdFIsIGRzdFgsIGRzdFksIHcsIGgsCi0gICAgICAgICAgICAgICAgICAgICAgICBzeXN4Zm9ybSwgY29tcCwgYmdjb2xvciwgY2xpcCk7Ci0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNpbXBsZUJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBDb21wb3NpdGUgY29tcCwKLSAgICAgICAgICAgIENvbG9yIGJnY29sb3IsIE11bHRpUmVjdEFyZWEgY2xpcCkgewotCi0gICAgICAgIC8vIFRPRE8gSXQncyBwb3NzaWJsZSwgdGhvdWdoIHVubGlrZWx5IHRoYXQgd2UgbWlnaHQgZW5jb3VudGVyIG5vbi1pbnRbXQotICAgICAgICAvLyBkYXRhIGJ1ZmZlcnMuIEluIHRoaXMgY2FzZSB0aGUgZm9sbG93aW5nIGNvZGUgbmVlZHMgdG8gaGF2ZSBzZXZlcmFsCi0gICAgICAgIC8vIGJyYW5jaGVzIHRoYXQgdGFrZSB0aGlzIGludG8gYWNjb3VudC4KLSAgICAgICAgZGF0YSA9IChEYXRhQnVmZmVySW50KXNyY1N1cmYuZ2V0UmFzdGVyKCkuZ2V0RGF0YUJ1ZmZlcigpOwotICAgICAgICBpbnRbXSBwaXhlbHMgPSBkYXRhLmdldERhdGEoKTsKLSAgICAgICAgaWYgKCFzcmNTdXJmLmdldENvbG9yTW9kZWwoKS5oYXNBbHBoYSgpKSB7Ci0gICAgICAgICAgICAvLyBUaGlzIHdvdWxkbid0IGJlIG5lY2Vzc2FyeSBpZiBBbmRyb2lkIHN1cHBvcnRlZCBSR0JfODg4LgotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBwaXhlbHMubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgICAgICBwaXhlbHNbaV0gPSBwaXhlbHNbaV0gfCAweGZmMDAwMDAwOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGJtYXAgPSBCaXRtYXAuY3JlYXRlQml0bWFwKHBpeGVscywgd2lkdGgsIGhlaWdodCwgQml0bWFwLkNvbmZpZy5BUkdCXzg4ODgpOwotICAgICAgICBjYW52YXMuZHJhd0JpdG1hcChibWFwLCBkc3RYLCBkc3RZLCBwYWludCk7Ci0gICAgfQotICAgIAotICAgIHB1YmxpYyB2b2lkIGJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBDb21wb3NpdGUgY29tcCwKLSAgICAgICAgICAgIENvbG9yIGJnY29sb3IsIE11bHRpUmVjdEFyZWEgY2xpcCkgewotCi0gICAgICAgIGphdmFCbHQoc3JjWCwgc3JjWSwgc3JjU3VyZi5nZXRXaWR0aCgpLCBzcmNTdXJmLmdldEhlaWdodCgpLAotICAgICAgICAgICAgICAgIHNyY1N1cmYuZ2V0Q29sb3JNb2RlbCgpLCBzcmNTdXJmLmdldFJhc3RlcigpLCBkc3RYLCBkc3RZLAotICAgICAgICAgICAgICAgIGRzdFN1cmYuZ2V0V2lkdGgoKSwgZHN0U3VyZi5nZXRIZWlnaHQoKSwKLSAgICAgICAgICAgICAgICBkc3RTdXJmLmdldENvbG9yTW9kZWwoKSwgZHN0U3VyZi5nZXRSYXN0ZXIoKSwKLSAgICAgICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLQotICAgIH0KLSAgICAKLSAgICBwdWJsaWMgdm9pZCBqYXZhQmx0KGludCBzcmNYLCBpbnQgc3JjWSwgaW50IHNyY1csIGludCBzcmNILAotICAgICAgICAgICAgQ29sb3JNb2RlbCBzcmNDTSwgUmFzdGVyIHNyY1Jhc3QsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIGludCBkc3RXLCBpbnQgZHN0SCwgQ29sb3JNb2RlbCBkc3RDTSwgV3JpdGFibGVSYXN0ZXIgZHN0UmFzdCwKLSAgICAgICAgICAgIGludCB3aWR0aCwgaW50IGhlaWdodCwgQ29tcG9zaXRlIGNvbXAsIENvbG9yIGJnY29sb3IsCi0gICAgICAgICAgICBNdWx0aVJlY3RBcmVhIGNsaXApewotICAgICAgICAKLSAgICAgICAgaW50IHNyY1gyID0gc3JjVyAtIDE7Ci0gICAgICAgIGludCBzcmNZMiA9IHNyY0ggLSAxOwotICAgICAgICBpbnQgZHN0WDIgPSBkc3RXIC0gMTsKLSAgICAgICAgaW50IGRzdFkyID0gZHN0SCAtIDE7Ci0KLSAgICAgICAgaWYoc3JjWCA8IDApewotICAgICAgICAgICAgd2lkdGggKz0gc3JjWDsKLSAgICAgICAgICAgIHNyY1ggPSAwOwotICAgICAgICB9Ci0gICAgICAgIGlmKHNyY1kgPCAwKXsKLSAgICAgICAgICAgIGhlaWdodCArPSBzcmNZOwotICAgICAgICAgICAgc3JjWSA9IDA7Ci0gICAgICAgIH0KLQotICAgICAgICBpZihkc3RYIDwgMCl7Ci0gICAgICAgICAgICB3aWR0aCArPSBkc3RYOwotICAgICAgICAgICAgc3JjWCAtPSBkc3RYOwotICAgICAgICAgICAgZHN0WCA9IDA7Ci0gICAgICAgIH0KLSAgICAgICAgaWYoZHN0WSA8IDApewotICAgICAgICAgICAgaGVpZ2h0ICs9IGRzdFk7Ci0gICAgICAgICAgICBzcmNZIC09IGRzdFk7Ci0gICAgICAgICAgICBkc3RZID0gMDsKLSAgICAgICAgfQotCi0gICAgICAgIGlmKHNyY1ggPiBzcmNYMiB8fCBzcmNZID4gc3JjWTIpIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBpZihkc3RYID4gZHN0WDIgfHwgZHN0WSA+IGRzdFkyKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpZihzcmNYICsgd2lkdGggPiBzcmNYMikgewotICAgICAgICAgICAgd2lkdGggPSBzcmNYMiAtIHNyY1ggKyAxOwotICAgICAgICB9Ci0gICAgICAgIGlmKHNyY1kgKyBoZWlnaHQgPiBzcmNZMikgewotICAgICAgICAgICAgaGVpZ2h0ID0gc3JjWTIgLSBzcmNZICsgMTsKLSAgICAgICAgfQotICAgICAgICBpZihkc3RYICsgd2lkdGggPiBkc3RYMikgewotICAgICAgICAgICAgd2lkdGggPSBkc3RYMiAtIGRzdFggKyAxOwotICAgICAgICB9Ci0gICAgICAgIGlmKGRzdFkgKyBoZWlnaHQgPiBkc3RZMikgewotICAgICAgICAgICAgaGVpZ2h0ID0gZHN0WTIgLSBkc3RZICsgMTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmKHdpZHRoIDw9IDAgfHwgaGVpZ2h0IDw9IDApIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBjbGlwUmVjdHNbXTsKLSAgICAgICAgaWYoY2xpcCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBjbGlwUmVjdHMgPSBjbGlwLnJlY3Q7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBjbGlwUmVjdHMgPSBuZXcgaW50W117NSwgMCwgMCwgZHN0VyAtIDEsIGRzdEggLSAxfTsKLSAgICAgICAgfQotCi0gICAgICAgIGJvb2xlYW4gaXNBbHBoYUNvbXAgPSBmYWxzZTsKLSAgICAgICAgaW50IHJ1bGUgPSAwOwotICAgICAgICBmbG9hdCBhbHBoYSA9IDA7Ci0gICAgICAgIGJvb2xlYW4gaXNYT1JDb21wID0gZmFsc2U7Ci0gICAgICAgIENvbG9yIHhvcmNvbG9yID0gbnVsbDsKLSAgICAgICAgQ29tcG9zaXRlQ29udGV4dCBjb250ID0gbnVsbDsKLQotICAgICAgICBpZihjb21wIGluc3RhbmNlb2YgQWxwaGFDb21wb3NpdGUpewotICAgICAgICAgICAgaXNBbHBoYUNvbXAgPSB0cnVlOwotICAgICAgICAgICAgQWxwaGFDb21wb3NpdGUgYWMgPSAoQWxwaGFDb21wb3NpdGUpIGNvbXA7Ci0gICAgICAgICAgICBydWxlID0gYWMuZ2V0UnVsZSgpOwotICAgICAgICAgICAgYWxwaGEgPSBhYy5nZXRBbHBoYSgpOwotICAgICAgICB9ZWxzZSBpZihjb21wIGluc3RhbmNlb2YgWE9SQ29tcG9zaXRlKXsKLSAgICAgICAgICAgIGlzWE9SQ29tcCA9IHRydWU7Ci0gICAgICAgICAgICBYT1JDb21wb3NpdGUgeGNvbXAgPSAoWE9SQ29tcG9zaXRlKSBjb21wOwotICAgICAgICAgICAgeG9yY29sb3IgPSB4Y29tcC5nZXRYT1JDb2xvcigpOwotICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgIGNvbnQgPSBjb21wLmNyZWF0ZUNvbnRleHQoc3JjQ00sIGRzdENNLCBudWxsKTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvcihpbnQgaSA9IDE7IGkgPCBjbGlwUmVjdHNbMF07IGkgKz0gNCl7Ci0gICAgICAgICAgICBpbnQgX3N4ID0gc3JjWDsKLSAgICAgICAgICAgIGludCBfc3kgPSBzcmNZOwotCi0gICAgICAgICAgICBpbnQgX2R4ID0gZHN0WDsKLSAgICAgICAgICAgIGludCBfZHkgPSBkc3RZOwotCi0gICAgICAgICAgICBpbnQgX3cgPSB3aWR0aDsKLSAgICAgICAgICAgIGludCBfaCA9IGhlaWdodDsKLQotICAgICAgICAgICAgaW50IGN4ID0gY2xpcFJlY3RzW2ldOyAgICAgICAgICAvLyBDbGlwcGluZyBsZWZ0IHRvcCBYCi0gICAgICAgICAgICBpbnQgY3kgPSBjbGlwUmVjdHNbaSArIDFdOyAgICAgIC8vIENsaXBwaW5nIGxlZnQgdG9wIFkKLSAgICAgICAgICAgIGludCBjeDIgPSBjbGlwUmVjdHNbaSArIDJdOyAgICAgLy8gQ2xpcHBpbmcgcmlnaHQgYm90dG9tIFgKLSAgICAgICAgICAgIGludCBjeTIgPSBjbGlwUmVjdHNbaSArIDNdOyAgICAgLy8gQ2xpcHBpbmcgcmlnaHQgYm90dG9tIFkKLQotICAgICAgICAgICAgaWYoX2R4ID4gY3gyIHx8IF9keSA+IGN5MiB8fCBkc3RYMiA8IGN4IHx8IGRzdFkyIDwgY3kpIHsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYoY3ggPiBfZHgpewotICAgICAgICAgICAgICAgIGludCBzaHggPSBjeCAtIF9keDsKLSAgICAgICAgICAgICAgICBfdyAtPSBzaHg7Ci0gICAgICAgICAgICAgICAgX2R4ID0gY3g7Ci0gICAgICAgICAgICAgICAgX3N4ICs9IHNoeDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYoY3kgPiBfZHkpewotICAgICAgICAgICAgICAgIGludCBzaHkgPSBjeSAtIF9keTsKLSAgICAgICAgICAgICAgICBfaCAtPSBzaHk7Ci0gICAgICAgICAgICAgICAgX2R5ID0gY3k7Ci0gICAgICAgICAgICAgICAgX3N5ICs9IHNoeTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYoX2R4ICsgX3cgPiBjeDIgKyAxKXsKLSAgICAgICAgICAgICAgICBfdyA9IGN4MiAtIF9keCArIDE7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmKF9keSArIF9oID4gY3kyICsgMSl7Ci0gICAgICAgICAgICAgICAgX2ggPSBjeTIgLSBfZHkgKyAxOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZihfc3ggPiBzcmNYMiB8fCBfc3kgPiBzcmNZMikgewotICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZihpc0FscGhhQ29tcCl7Ci0gICAgICAgICAgICAgICAgYWxwaGFDb21wb3NlKF9zeCwgX3N5LCBzcmNDTSwgc3JjUmFzdCwgX2R4LCBfZHksCi0gICAgICAgICAgICAgICAgICAgICAgICBkc3RDTSwgZHN0UmFzdCwgX3csIF9oLCBydWxlLCBhbHBoYSwgYmdjb2xvcik7Ci0gICAgICAgICAgICB9ZWxzZSBpZihpc1hPUkNvbXApewotICAgICAgICAgICAgICAgIHhvckNvbXBvc2UoX3N4LCBfc3ksIHNyY0NNLCBzcmNSYXN0LCBfZHgsIF9keSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdENNLCBkc3RSYXN0LCBfdywgX2gsIHhvcmNvbG9yKTsKLSAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgIFJhc3RlciBzciA9IHNyY1Jhc3QuY3JlYXRlQ2hpbGQoX3N4LCBfc3ksIF93LCBfaCwgMCwgMCwgbnVsbCk7Ci0gICAgICAgICAgICAgICAgV3JpdGFibGVSYXN0ZXIgZHIgPSBkc3RSYXN0LmNyZWF0ZVdyaXRhYmxlQ2hpbGQoX2R4LCBfZHksCi0gICAgICAgICAgICAgICAgICAgICAgICBfdywgX2gsIDAsIDAsIG51bGwpOwotICAgICAgICAgICAgICAgIGNvbnQuY29tcG9zZShzciwgZHIsIGRyKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICAKLSAgICB9Ci0KLSAgICBEYXRhQnVmZmVySW50IGRhdGE7Ci0gICAgQml0bWFwIGJtYXAsIGJtcDsKLSAgICAKLSAgICB2b2lkIGFscGhhQ29tcG9zZShpbnQgc3JjWCwgaW50IHNyY1ksIENvbG9yTW9kZWwgc3JjQ00sIFJhc3RlciBzcmNSYXN0LAotICAgICAgICAgICAgaW50IGRzdFgsIGludCBkc3RZLCBDb2xvck1vZGVsIGRzdENNLCBXcml0YWJsZVJhc3RlciBkc3RSYXN0LAotICAgICAgICAgICAgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgcnVsZSwgZmxvYXQgYWxwaGEsIENvbG9yIGJnY29sb3IpewotICAgICAgICAKLSAgICAgICAgT2JqZWN0IHNyY1BpeGVsID0gZ2V0VHJhbnNmZXJBcnJheShzcmNSYXN0LCAxKTsKLSAgICAgICAgZGF0YSA9IChEYXRhQnVmZmVySW50KXNyY1Jhc3QuZ2V0RGF0YUJ1ZmZlcigpOwotICAgICAgICBpbnQgcGl4W10gPSBkYXRhLmdldERhdGEoKTsKLSAgICAgICAgYm1hcCA9IEJpdG1hcC5jcmVhdGVCaXRtYXAocGl4LCB3aWR0aCwgaGVpZ2h0LCBCaXRtYXAuQ29uZmlnLlJHQl81NjUpOwotICAgICAgICBjYW52YXMuZHJhd0JpdG1hcChibWFwLCBkc3RYLCBkc3RZLCBwYWludCk7Ci0gICAgfQotICAgIAotICAgIHZvaWQgcmVuZGVyKGludFtdIGltZywgaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgY2FudmFzLmRyYXdCaXRtYXAoQml0bWFwLmNyZWF0ZUJpdG1hcChpbWcsIHdpZHRoLCBoZWlnaHQsIEJpdG1hcC5Db25maWcuQVJHQl84ODg4KSwgeCwgeSwgcGFpbnQpOwotICAgIH0KLQotICAgIHZvaWQgeG9yQ29tcG9zZShpbnQgc3JjWCwgaW50IHNyY1ksIENvbG9yTW9kZWwgc3JjQ00sIFJhc3RlciBzcmNSYXN0LAotICAgICAgICAgICAgaW50IGRzdFgsIGludCBkc3RZLCBDb2xvck1vZGVsIGRzdENNLCBXcml0YWJsZVJhc3RlciBkc3RSYXN0LAotICAgICAgICAgICAgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBDb2xvciB4b3Jjb2xvcil7Ci0KLSAgICAgICAgZGF0YSA9IChEYXRhQnVmZmVySW50KXNyY1Jhc3QuZ2V0RGF0YUJ1ZmZlcigpOwotICAgICAgICBpbnQgcGl4W10gPSBkYXRhLmdldERhdGEoKTsKLSAgICAgICAgYm1hcCA9IEJpdG1hcC5jcmVhdGVCaXRtYXAocGl4LCB3aWR0aCwgaGVpZ2h0LCBCaXRtYXAuQ29uZmlnLlJHQl81NjUpOwotICAgICAgICBjYW52YXMuZHJhd0JpdG1hcChibWFwLCBkc3RYLCBkc3RZLCBwYWludCk7Ci0gICAgfQotICAgIAotICAgIHByaXZhdGUgdm9pZCB0cmFuc2Zvcm1lZEJsaXQoQ29sb3JNb2RlbCBzcmNDTSwgUmFzdGVyIHNyY1IsIGludCBzcmNYLCBpbnQgc3JjWSwKLSAgICAgICAgICAgIENvbG9yTW9kZWwgZHN0Q00sIFdyaXRhYmxlUmFzdGVyIGRzdFIsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIGludCB3aWR0aCwgaW50IGhlaWdodCwgQWZmaW5lVHJhbnNmb3JtIGF0LCBDb21wb3NpdGUgY29tcCwKLSAgICAgICAgICAgIENvbG9yIGJnY29sb3IsIE11bHRpUmVjdEFyZWEgY2xpcCkgewotICAgICAgICAKLSAgICAgICAgZGF0YSA9IChEYXRhQnVmZmVySW50KXNyY1IuZ2V0RGF0YUJ1ZmZlcigpOwotICAgICAgICBpbnRbXSBwaXhlbHMgPSBkYXRhLmdldERhdGEoKTsKLSAgICAgICAgaWYgKCFzcmNDTS5oYXNBbHBoYSgpKSB7Ci0gICAgICAgICAgICAvLyBUaGlzIHdvdWxkbid0IGJlIG5lY2Vzc2FyeSBpZiBBbmRyb2lkIHN1cHBvcnRlZCBSR0JfODg4LgotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBwaXhlbHMubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgICAgICBwaXhlbHNbaV0gPSBwaXhlbHNbaV0gfCAweGZmMDAwMDAwOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGJtYXAgPSBCaXRtYXAuY3JlYXRlQml0bWFwKHBpeGVscywgd2lkdGgsIGhlaWdodCwgQml0bWFwLkNvbmZpZy5BUkdCXzg4ODgpOwotICAgICAgICAKLSAgICAgICAgTWF0cml4IHRtID0gbmV3IE1hdHJpeCgpOwotICAgICAgICB0bS5zZXRDb25jYXQoY2FudmFzLmdldE1hdHJpeCgpLCBBbmRyb2lkR3JhcGhpY3MyRC5jcmVhdGVNYXRyaXhPYmooYXQpKTsKLSAgICAgICAgaWYoYXQuZ2V0VHlwZSgpID4gMSkgewotICAgICAgICAgICAgYm1wID0gQml0bWFwLmNyZWF0ZUJpdG1hcChibWFwLCAwLCAwLCB3aWR0aCwgaGVpZ2h0LCB0bSwgdHJ1ZSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBibXAgPSBCaXRtYXAuY3JlYXRlQml0bWFwKGJtYXAsIDAsIDAsIHdpZHRoLCBoZWlnaHQsIHRtLCBmYWxzZSk7Ci0gICAgICAgIH0KLSAgICAgICAgY2FudmFzLmRyYXdCaXRtYXAoYm1wLCBkc3RYICsgKGZsb2F0KWF0LmdldFRyYW5zbGF0ZVgoKSwgZHN0WSArIChmbG9hdClhdC5nZXRUcmFuc2xhdGVZKCksIHBhaW50KTsKLSAgICB9Ci0KLSAgICBwcml2YXRlIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKEFmZmluZVRyYW5zZm9ybSBhdCwgUmVjdGFuZ2xlIHIpIHsKLSAgICAgICAgaW50IHggPSByLng7Ci0gICAgICAgIGludCB5ID0gci55OwotICAgICAgICBpbnQgd2lkdGggPSByLndpZHRoOwotICAgICAgICBpbnQgaGVpZ2h0ID0gci5oZWlnaHQ7Ci0KLSAgICAgICAgZmxvYXRbXSBjb3JuZXJzID0gewotICAgICAgICAgICAgeCwgeSwKLSAgICAgICAgICAgIHggKyB3aWR0aCwgeSwKLSAgICAgICAgICAgIHggKyB3aWR0aCwgeSArIGhlaWdodCwKLSAgICAgICAgICAgIHgsIHkgKyBoZWlnaHQKLSAgICAgICAgfTsKLQotICAgICAgICBhdC50cmFuc2Zvcm0oY29ybmVycywgMCwgY29ybmVycywgMCwgNCk7Ci0KLSAgICAgICAgUmVjdGFuZ2xlMkQuRmxvYXQgYm91bmRzID0gbmV3IFJlY3RhbmdsZTJELkZsb2F0KGNvcm5lcnNbMF0sIGNvcm5lcnNbMV0sIDAgLCAwKTsKLSAgICAgICAgYm91bmRzLmFkZChjb3JuZXJzWzJdLCBjb3JuZXJzWzNdKTsKLSAgICAgICAgYm91bmRzLmFkZChjb3JuZXJzWzRdLCBjb3JuZXJzWzVdKTsKLSAgICAgICAgYm91bmRzLmFkZChjb3JuZXJzWzZdLCBjb3JuZXJzWzddKTsKLQotICAgICAgICByZXR1cm4gYm91bmRzOwotICAgIH0KLQotICAgIHByaXZhdGUgaW50IGNvbXBvc2UoaW50IHNyY1JHQiwgYm9vbGVhbiBpc1NyY0FscGhhUHJlLAotICAgICAgICAgICAgaW50IGRzdFJHQiwgYm9vbGVhbiBkc3RIYXNBbHBoYSwgYm9vbGVhbiBpc0RzdEFscGhhUHJlLAotICAgICAgICAgICAgaW50IHJ1bGUsIGludCBzcmNDb25zdEFscGhhKXsKLQotICAgICAgICBpbnQgc2EsIHNyLCBzZywgc2IsIGRhLCBkciwgZGcsIGRiOwotCi0gICAgICAgIHNhID0gKHNyY1JHQiA+PiAyNCkgJiAweGZmOwotICAgICAgICBzciA9IChzcmNSR0IgPj4gMTYpICYgMHhmZjsKLSAgICAgICAgc2cgPSAoc3JjUkdCID4+IDgpICYgMHhmZjsKLSAgICAgICAgc2IgPSBzcmNSR0IgJiAweGZmOwotCi0gICAgICAgIGlmKGlzU3JjQWxwaGFQcmUpewotICAgICAgICAgICAgc2EgPSBtdWxMVVRbc3JjQ29uc3RBbHBoYV1bc2FdICYgMHhmZjsKLSAgICAgICAgICAgIHNyID0gbXVsTFVUW3NyY0NvbnN0QWxwaGFdW3NyXSAmIDB4ZmY7Ci0gICAgICAgICAgICBzZyA9IG11bExVVFtzcmNDb25zdEFscGhhXVtzZ10gJiAweGZmOwotICAgICAgICAgICAgc2IgPSBtdWxMVVRbc3JjQ29uc3RBbHBoYV1bc2JdICYgMHhmZjsKLSAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICBzYSA9IG11bExVVFtzcmNDb25zdEFscGhhXVtzYV0gJiAweGZmOwotICAgICAgICAgICAgc3IgPSBtdWxMVVRbc2FdW3NyXSAmIDB4ZmY7Ci0gICAgICAgICAgICBzZyA9IG11bExVVFtzYV1bc2ddICYgMHhmZjsKLSAgICAgICAgICAgIHNiID0gbXVsTFVUW3NhXVtzYl0gJiAweGZmOwotICAgICAgICB9Ci0KLSAgICAgICAgZGEgPSAoZHN0UkdCID4+IDI0KSAmIDB4ZmY7Ci0gICAgICAgIGRyID0gKGRzdFJHQiA+PiAxNikgJiAweGZmOwotICAgICAgICBkZyA9IChkc3RSR0IgPj4gOCkgJiAweGZmOwotICAgICAgICBkYiA9IGRzdFJHQiAmIDB4ZmY7Ci0KLSAgICAgICAgaWYoIWlzRHN0QWxwaGFQcmUpewotICAgICAgICAgICAgZHIgPSBtdWxMVVRbZGFdW2RyXSAmIDB4ZmY7Ci0gICAgICAgICAgICBkZyA9IG11bExVVFtkYV1bZGddICYgMHhmZjsKLSAgICAgICAgICAgIGRiID0gbXVsTFVUW2RhXVtkYl0gJiAweGZmOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IEZzID0gMDsKLSAgICAgICAgaW50IEZkID0gMDsKLSAgICAgICAgc3dpdGNoKHJ1bGUpewotICAgICAgICBjYXNlIEFscGhhQ29tcG9zaXRlLkNMRUFSOgotICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgY2FzZSBBbHBoYUNvbXBvc2l0ZS5EU1Q6Ci0gICAgICAgICAgICBGZCA9IDI1NTsKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGNhc2UgQWxwaGFDb21wb3NpdGUuRFNUX0FUT1A6Ci0gICAgICAgICAgICBGcyA9IDI1NSAtIGRhOwotICAgICAgICAgICAgRmQgPSBzYTsKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGNhc2UgQWxwaGFDb21wb3NpdGUuRFNUX0lOOgotICAgICAgICAgICAgRmQgPSBzYTsKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGNhc2UgQWxwaGFDb21wb3NpdGUuRFNUX09VVDoKLSAgICAgICAgICAgIEZkID0gMjU1IC0gc2E7Ci0gICAgICAgICAgICBicmVhazsKLQotICAgICAgICBjYXNlIEFscGhhQ29tcG9zaXRlLkRTVF9PVkVSOgotICAgICAgICAgICAgRnMgPSAyNTUgLSBkYTsKLSAgICAgICAgICAgIEZkID0gMjU1OwotICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgY2FzZSBBbHBoYUNvbXBvc2l0ZS5TUkM6Ci0gICAgICAgICAgICBGcyA9IDI1NTsKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGNhc2UgQWxwaGFDb21wb3NpdGUuU1JDX0FUT1A6Ci0gICAgICAgICAgICBGcyA9IGRhOwotICAgICAgICAgICAgRmQgPSAyNTUgLSBzYTsKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGNhc2UgQWxwaGFDb21wb3NpdGUuU1JDX0lOOgotICAgICAgICAgICAgRnMgPSBkYTsKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGNhc2UgQWxwaGFDb21wb3NpdGUuU1JDX09VVDoKLSAgICAgICAgICAgIEZzID0gMjU1IC0gZGE7Ci0gICAgICAgICAgICBicmVhazsKLQotICAgICAgICBjYXNlIEFscGhhQ29tcG9zaXRlLlNSQ19PVkVSOgotICAgICAgICAgICAgRnMgPSAyNTU7Ci0gICAgICAgICAgICBGZCA9IDI1NSAtIHNhOwotICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgY2FzZSBBbHBoYUNvbXBvc2l0ZS5YT1I6Ci0gICAgICAgICAgICBGcyA9IDI1NSAtIGRhOwotICAgICAgICAgICAgRmQgPSAyNTUgLSBzYTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgICAgIGRyID0gKG11bExVVFtzcl1bRnNdICYgMHhmZikgKyAobXVsTFVUW2RyXVtGZF0gJiAweGZmKTsKLSAgICAgICAgZGcgPSAobXVsTFVUW3NnXVtGc10gJiAweGZmKSArIChtdWxMVVRbZGddW0ZkXSAmIDB4ZmYpOwotICAgICAgICBkYiA9IChtdWxMVVRbc2JdW0ZzXSAmIDB4ZmYpICsgKG11bExVVFtkYl1bRmRdICYgMHhmZik7Ci0KLSAgICAgICAgZGEgPSAobXVsTFVUW3NhXVtGc10gJiAweGZmKSArIChtdWxMVVRbZGFdW0ZkXSAmIDB4ZmYpOwotCi0gICAgICAgIGlmKCFpc0RzdEFscGhhUHJlKXsKLSAgICAgICAgICAgIGlmKGRhICE9IDI1NSl7Ci0gICAgICAgICAgICAgICAgZHIgPSBkaXZMVVRbZGFdW2RyXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgZGcgPSBkaXZMVVRbZGFdW2RnXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgZGIgPSBkaXZMVVRbZGFdW2RiXSAmIDB4ZmY7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgaWYoIWRzdEhhc0FscGhhKSB7Ci0gICAgICAgICAgICBkYSA9IDB4ZmY7Ci0gICAgICAgIH0KLSAgICAgICAgZHN0UkdCID0gKGRhIDw8IDI0KSB8IChkciA8PCAxNikgfCAoZGcgPDwgOCkgfCBkYjsKLQotICAgICAgICByZXR1cm4gZHN0UkdCOwotCi0gICAgfQotICAgIAotICAgIC8qKgotICAgICAqIEFsbG9jYXRlIGFuIGFycmF5IHRoYXQgY2FuIGJlIHVzZSB0byBzdG9yZSB0aGUgcmVzdWx0IGZvciBhIAotICAgICAqIFJhc3Rlci5nZXREYXRhRWxlbWVudHMgY2FsbC4KLSAgICAgKiBAcGFyYW0gcmFzdGVyICBSYXN0ZXIgKHR5cGUpIHdoZXJlIHRoZSBnZXREYXRhRWxlbWVudHMgY2FsbCB3aWxsIGJlIG1hZGUuIAotICAgICAqIEBwYXJhbSBuYlBpeGVscyAgSG93IG1hbnkgcGl4ZWxzIHRvIHN0b3JlIGluIHRoZSBhcnJheSBhdCBtb3N0Ci0gICAgICogQHJldHVybiB0aGUgcmVzdWx0IGFycmF5IG9yIG51bGwKLSAgICAgKi8KLSAgICBwcml2YXRlIE9iamVjdCBnZXRUcmFuc2ZlckFycmF5KFJhc3RlciByYXN0ZXIsIGludCBuYlBpeGVscykgewotICAgICAgICBpbnQgdHJhbnNmZXJUeXBlID0gcmFzdGVyLmdldFRyYW5zZmVyVHlwZSgpOwotICAgICAgICBpbnQgbmJEYXRhRWxlbWVudHMgPSByYXN0ZXIuZ2V0U2FtcGxlTW9kZWwoKS5nZXROdW1EYXRhRWxlbWVudHMoKTsKLSAgICAgICAgaW50IG4gPSBuYkRhdGFFbGVtZW50cyAqIG5iUGl4ZWxzOwotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgcmV0dXJuIG5ldyBieXRlW25dOwotICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9TSE9SVDoKLSAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgcmV0dXJuIG5ldyBzaG9ydFtuXTsKLSAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgcmV0dXJuIG5ldyBpbnRbbl07Ci0gICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FUOgotICAgICAgICAgICAgcmV0dXJuIG5ldyBmbG9hdFtuXTsKLSAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFOgotICAgICAgICAgICAgcmV0dXJuIG5ldyBkb3VibGVbbl07Ci0gICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VOREVGSU5FRDoKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIC8qKgotICAgICAqIERyYXcgYSBwaXhlbAotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBkb3QoaW50IHgsIGludCB5LCBpbnQgY2xyKSB7Ci0gICAgICAgIGlmIChjb2xvckNhY2hlICE9IGNscikgewotICAgICAgICAgICAgcGFpbnQuc2V0Q29sb3IoY2xyKTsgIAotICAgICAgICAgICAgY29sb3JDYWNoZSA9IGNscjsKLSAgICAgICAgfQotICAgICAgICBjYW52YXMuZHJhd0xpbmUoeCwgeSwgeCArIDEsIHkgKyAxLCBwYWludCk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9BbmRyb2lkTmF0aXZlRXZlbnRRdWV1ZS5qYXZhIGIvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9BbmRyb2lkTmF0aXZlRXZlbnRRdWV1ZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmYzMwNjE0Li4wMDAwMDAwCi0tLSBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvQW5kcm9pZE5hdGl2ZUV2ZW50UXVldWUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDc1ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgMjAwNywgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotcGFja2FnZSBjb20uYW5kcm9pZC5pbnRlcm5hbC5hd3Q7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5OYXRpdmVFdmVudFF1ZXVlOwotCi1wdWJsaWMgY2xhc3MgQW5kcm9pZE5hdGl2ZUV2ZW50UXVldWUgZXh0ZW5kcyBOYXRpdmVFdmVudFF1ZXVlIHsKLSAgICAKLSAgICBwcml2YXRlIE9iamVjdCBldmVudE1vbml0b3I7Ci0gICAgCi0gICAgcHVibGljIEFuZHJvaWROYXRpdmVFdmVudFF1ZXVlKCkgewotICAgICAgICBzdXBlcigpOwotICAgICAgICBldmVudE1vbml0b3IgPSBnZXRFdmVudE1vbml0b3IoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBhd2FrZSgpIHsKLSAgICAgICAgc3luY2hyb25pemVkIChldmVudE1vbml0b3IpIHsKLSAgICAgICAgICAgIGV2ZW50TW9uaXRvci5ub3RpZnkoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRpc3BhdGNoRXZlbnQoKSB7Ci0gICAgICAgIC8vPz8/QVdUCi0gICAgICAgIFN5c3RlbS5vdXQucHJpbnRsbihnZXRDbGFzcygpKyI6IGVtcHR5IG1ldGhvZCBjYWxsZWQiKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgbG9uZyBnZXRKYXZhV2luZG93KCkgewotICAgICAgICAvLz8/P0FXVAotICAgICAgICBTeXN0ZW0ub3V0LnByaW50bG4oZ2V0Q2xhc3MoKSsiOiBlbXB0eSBtZXRob2QgY2FsbGVkIik7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHBlcmZvcm1MYXRlcihUYXNrIHRhc2spIHsKLSAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgU3lzdGVtLm91dC5wcmludGxuKGdldENsYXNzKCkrIjogZW1wdHkgbWV0aG9kIGNhbGxlZCIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHBlcmZvcm1UYXNrKFRhc2sgdGFzaykgewotICAgICAgICAvLz8/P0FXVAotICAgICAgICBTeXN0ZW0ub3V0LnByaW50bG4oZ2V0Q2xhc3MoKSsiOiBlbXB0eSBtZXRob2QgY2FsbGVkIik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gd2FpdEV2ZW50KCkgewotICAgICAgICB3aGlsZSAoaXNFbXB0eSgpICkgewotICAgICAgICAgICAgc3luY2hyb25pemVkIChldmVudE1vbml0b3IpIHsKLSAgICAgICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgICAgICBldmVudE1vbml0b3Iud2FpdCgxMDAwKTsKLSAgICAgICAgICAgICAgICB9IGNhdGNoIChJbnRlcnJ1cHRlZEV4Y2VwdGlvbiBpZ25vcmUpIHsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9BbmRyb2lkV1RLLmphdmEgYi9hd3QvY29tL2FuZHJvaWQvaW50ZXJuYWwvYXd0L0FuZHJvaWRXVEsuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTYwOWQxMS4uMDAwMDAwMAotLS0gYS9hd3QvY29tL2FuZHJvaWQvaW50ZXJuYWwvYXd0L0FuZHJvaWRXVEsuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDg4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgMjAwNywgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotcGFja2FnZSBjb20uYW5kcm9pZC5pbnRlcm5hbC5hd3Q7Ci0KLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljc0RldmljZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLkN1cnNvckZhY3Rvcnk7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuR3JhcGhpY3NGYWN0b3J5OwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLk5hdGl2ZUV2ZW50UXVldWU7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuTmF0aXZlSU07Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuTmF0aXZlTW91c2VJbmZvOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLk5hdGl2ZVJvYm90OwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLlN5c3RlbVByb3BlcnRpZXM7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuV1RLOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLldpbmRvd0ZhY3Rvcnk7Ci0KLXB1YmxpYyBjbGFzcyBBbmRyb2lkV1RLIGV4dGVuZHMgV1RLIHsKLQotICAgIHByaXZhdGUgQW5kcm9pZEdyYXBoaWNzRmFjdG9yeSBtQWdmOwotICAgIHByaXZhdGUgQW5kcm9pZE5hdGl2ZUV2ZW50UXVldWUgbUFuZXE7Ci0gICAgCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEN1cnNvckZhY3RvcnkgZ2V0Q3Vyc29yRmFjdG9yeSgpIHsKLSAgICAgICAgLy8gVE9ETyBBdXRvLWdlbmVyYXRlZCBtZXRob2Qgc3R1YgotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR3JhcGhpY3NGYWN0b3J5IGdldEdyYXBoaWNzRmFjdG9yeSgpIHsKLSAgICAgICAgaWYobUFnZiA9PSBudWxsKSB7Ci0gICAgICAgICAgICBtQWdmID0gbmV3IEFuZHJvaWRHcmFwaGljc0ZhY3RvcnkoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbUFnZjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgTmF0aXZlRXZlbnRRdWV1ZSBnZXROYXRpdmVFdmVudFF1ZXVlKCkgewotICAgICAgICBpZihtQW5lcSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBtQW5lcSA9IG5ldyBBbmRyb2lkTmF0aXZlRXZlbnRRdWV1ZSgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBtQW5lcTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgTmF0aXZlSU0gZ2V0TmF0aXZlSU0oKSB7Ci0gICAgICAgIC8vIFRPRE8gQXV0by1nZW5lcmF0ZWQgbWV0aG9kIHN0dWIKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE5hdGl2ZU1vdXNlSW5mbyBnZXROYXRpdmVNb3VzZUluZm8oKSB7Ci0gICAgICAgIC8vIFRPRE8gQXV0by1nZW5lcmF0ZWQgbWV0aG9kIHN0dWIKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE5hdGl2ZVJvYm90IGdldE5hdGl2ZVJvYm90KEdyYXBoaWNzRGV2aWNlIHNjcmVlbikgewotICAgICAgICAvLyBUT0RPIEF1dG8tZ2VuZXJhdGVkIG1ldGhvZCBzdHViCi0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTeXN0ZW1Qcm9wZXJ0aWVzIGdldFN5c3RlbVByb3BlcnRpZXMoKSB7Ci0gICAgICAgIC8vIFRPRE8gQXV0by1nZW5lcmF0ZWQgbWV0aG9kIHN0dWIKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFdpbmRvd0ZhY3RvcnkgZ2V0V2luZG93RmFjdG9yeSgpIHsKLSAgICAgICAgLy8gVE9ETyBBdXRvLWdlbmVyYXRlZCBtZXRob2Qgc3R1YgotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvQXd0RmFjdG9yeS5qYXZhIGIvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9Bd3RGYWN0b3J5LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDZlNjY3YjIuLjAwMDAwMDAKLS0tIGEvYXd0L2NvbS9hbmRyb2lkL2ludGVybmFsL2F3dC9Bd3RGYWN0b3J5LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1MiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgY29tLmFuZHJvaWQuaW50ZXJuYWwuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3MyRDsKLWltcG9ydCBqYXZhLmF3dC5Ub29sa2l0OwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuR3JhcGhpY3NGYWN0b3J5OwotCi1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5DYW52YXM7Ci1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5QYWludDsKLQotcHVibGljIGNsYXNzIEF3dEZhY3RvcnkgewotICAgIAotICAgIHByaXZhdGUgc3RhdGljIEdyYXBoaWNzRmFjdG9yeSBnZjsKLSAgICAKLSAgICAvKioKLSAgICAgKiBVc2UgdGhpcyBtZXRob2QgdG8gZ2V0IGFjY2VzIHRvIEFXVCBkcmF3aW5nIHByaW1pdGl2ZXMgYW5kIHRvCi0gICAgICogcmVuZGVyIGludG8gdGhlIHN1cmZhY2UgYXJlYSBvZiBhIEFuZHJvaWQgd2lkZ2V0LiBPcmlnaW4gYW5kIAotICAgICAqIGNsaXAgb2YgdGhlIHJldHVybmVkIGdyYXBoaWNzIG9iamVjdCBhcmUgdGhlIHNhbWUgYXMgaW4gdGhlCi0gICAgICogY29ycmVzcG9uZGluZyBBbmRyb2lkIHdpZGdldC4gCi0gICAgICogCi0gICAgICogQHBhcmFtIGMgQ2FudmFzIG9mIHRoZSBhbmRyb2lkIHdpZGdldCB0byBkcmF3IGludG8KLSAgICAgKiBAcGFyYW0gcCBUaGUgZGVmYXVsdCBkcmF3aW5nIHBhcmFtZXRlcnMgc3VjaCBhcyBmb250LCAKLSAgICAgKiBzdHJva2UsIGZvcmVncm91bmQgYW5kIGJhY2tncm91bmQgY29sb3JzLCBldGMuCi0gICAgICogQHJldHVybiBUaGUgQVdUIEdyYXBoaWNzIG9iamVjdCB0aGF0IG1ha2VzIGFsbCBBV1QgCi0gICAgICogZHJhd2luZyBwcmltaXRpdmVzIGF2YWlsYWJsZSBpbiB0aGUgYW5kcm9pbmQgd29ybGQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBHcmFwaGljczJEIGdldEF3dEdyYXBoaWNzKENhbnZhcyBjLCBQYWludCBwKSB7Ci0gICAgICAgIC8vIEFXVD8/IFRPRE86IHRlc3QgaXQhCi0gICAgICAgIGlmIChudWxsID09IGdmKSB7Ci0gICAgICAgICAgICBUb29sa2l0IHRrID0gVG9vbGtpdC5nZXREZWZhdWx0VG9vbGtpdCgpOwotICAgICAgICAgICAgZ2YgPSB0ay5nZXRHcmFwaGljc0ZhY3RvcnkoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZ2YuZ2V0R3JhcGhpY3MyRChjLCBwKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvSW1hZ2VPdXRwdXRTdHJlYW1XcmFwcGVyLmphdmEgYi9hd3QvY29tL2FuZHJvaWQvaW50ZXJuYWwvYXd0L0ltYWdlT3V0cHV0U3RyZWFtV3JhcHBlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5MjE4NWZkLi4wMDAwMDAwCi0tLSBhL2F3dC9jb20vYW5kcm9pZC9pbnRlcm5hbC9hd3QvSW1hZ2VPdXRwdXRTdHJlYW1XcmFwcGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgY29tLmFuZHJvaWQuaW50ZXJuYWwuYXd0OwotCi1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLk91dHB1dFN0cmVhbTsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uc3RyZWFtLkltYWdlT3V0cHV0U3RyZWFtOwotCi1wdWJsaWMgY2xhc3MgSW1hZ2VPdXRwdXRTdHJlYW1XcmFwcGVyIGV4dGVuZHMgT3V0cHV0U3RyZWFtIHsKLQkKLQlwcm90ZWN0ZWQgSW1hZ2VPdXRwdXRTdHJlYW0gbUlvczsKLQkKLQlwcml2YXRlIGJ5dGVbXSBtQnVmZjsKLQkKLQlwdWJsaWMgSW1hZ2VPdXRwdXRTdHJlYW1XcmFwcGVyKEltYWdlT3V0cHV0U3RyZWFtIGlvcykgewotCQlpZiAobnVsbCA9PSBpb3MpIHsKLQkJCXRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkltYWdlT3V0cHV0U3RyZWFtIG11c3Qgbm90IGJlIG51bGwiKTsKLQkJfQotCQl0aGlzLm1Jb3MgPSBpb3M7Ci0JCXRoaXMubUJ1ZmYgPSBuZXcgYnl0ZVsxXTsKLQl9Ci0KLQlwdWJsaWMgSW1hZ2VPdXRwdXRTdHJlYW0gZ2V0SW1hZ2VPdXRwdXRTdHJlYW0oKSB7Ci0JCXJldHVybiBtSW9zOwotCX0KLQkKLQlAT3ZlcnJpZGUKLQlwdWJsaWMgdm9pZCB3cml0ZShpbnQgb25lQnl0ZSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLQkJbUJ1ZmZbMF0gPSAoYnl0ZSlvbmVCeXRlOwotCQltSW9zLndyaXRlKG1CdWZmLCAwLCAxKTsKLQl9Ci0KLQlwdWJsaWMgdm9pZCB3cml0ZShieXRlW10gYikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLQkJbUlvcy53cml0ZShiLCAwLCBiLmxlbmd0aCk7Ci0JfQotCQotCXB1YmxpYyB2b2lkIHdyaXRlKGJ5dGVbXSBiLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotCQltSW9zLndyaXRlKGIsIG9mZiwgbGVuKTsKLQl9Ci0JCi0JcHVibGljIHZvaWQgZmx1c2goKSB0aHJvd3MgSU9FeGNlcHRpb24gewotCQltSW9zLmZsdXNoKCk7Ci0JfQotCQotICAgIHB1YmxpYyB2b2lkIGNsb3NlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAJaWYgKG1Jb3MgPT0gbnVsbCkgewotICAgIAkJdGhyb3cgbmV3IElPRXhjZXB0aW9uKCJTdHJlYW0gYWxyZWFkeSBjbG9zZWQiKTsKLSAgICAJfQotICAgICAgICBtSW9zID0gbnVsbDsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvQVdURXZlbnQuamF2YSBiL2F3dC9qYXZhL2F3dC9BV1RFdmVudC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBhOGRjODNhLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9BV1RFdmVudC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNjgxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRG1pdHJ5IEEuIER1cm5ldiwgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEudXRpbC5FdmVudE9iamVjdDsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaHRhYmxlOwotaW1wb3J0IGphdmEudXRpbC5FdmVudExpc3RlbmVyOwotCi1pbXBvcnQgamF2YS5hd3QuZXZlbnQuKjsKLQotLyoqCi0gKiBUaGUgYWJzdHJhY3QgY2xhc3MgQVdURXZlbnQgaXMgdGhlIGJhc2UgY2xhc3MgZm9yIGFsbCBBV1QgZXZlbnRzLiBUaGlzIGNsYXNzCi0gKiBhbmQgaXRzIHN1YmNsYXNzZXMgc3VwZXJzZWRlIHRoZSBvcmlnaW5hbCBqYXZhLmF3dC5FdmVudCBjbGFzcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBBV1RFdmVudCBleHRlbmRzIEV2ZW50T2JqZWN0IHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC0xODI1MzE0Nzc5MTYwNDA5NDA1TDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDT01QT05FTlRfRVZFTlRfTUFTSyBpbmRpY2F0ZXMgdGhlIGV2ZW50IHJlbGF0ZXMgdG8gYQotICAgICAqIGNvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGxvbmcgQ09NUE9ORU5UX0VWRU5UX01BU0sgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENPTlRBSU5FUl9FVkVOVF9NQVNLIGluZGljYXRlcyB0aGUgZXZlbnQgcmVsYXRlcyB0byBhCi0gICAgICogY29udGFpbmVyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgbG9uZyBDT05UQUlORVJfRVZFTlRfTUFTSyA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRk9DVVNfRVZFTlRfTUFTSyBpbmRpY2F0ZXMgdGhlIGV2ZW50IHJlbGF0ZXMgdG8gdGhlIGZvY3VzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgbG9uZyBGT0NVU19FVkVOVF9NQVNLID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBLRVlfRVZFTlRfTUFTSyBpbmRpY2F0ZXMgdGhlIGV2ZW50IHJlbGF0ZXMgdG8gYSBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBsb25nIEtFWV9FVkVOVF9NQVNLID0gODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBNT1VTRV9FVkVOVF9NQVNLIGluZGljYXRlcyB0aGUgZXZlbnQgcmVsYXRlcyB0byB0aGUgbW91c2UuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBsb25nIE1PVVNFX0VWRU5UX01BU0sgPSAxNjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBNT1VTRV9NT1RJT05fRVZFTlRfTUFTSyBpbmRpY2F0ZXMgdGhlIGV2ZW50IHJlbGF0ZXMgdG8gYQotICAgICAqIG1vdXNlIG1vdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGxvbmcgTU9VU0VfTU9USU9OX0VWRU5UX01BU0sgPSAzMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBXSU5ET1dfRVZFTlRfTUFTSyBpbmRpY2F0ZXMgdGhlIGV2ZW50IHJlbGF0ZXMgdG8gYSB3aW5kb3cuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBsb25nIFdJTkRPV19FVkVOVF9NQVNLID0gNjQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQUNUSU9OX0VWRU5UX01BU0sgaW5kaWNhdGVzIHRoZSBldmVudCByZWxhdGVzIHRvIGFuIGFjdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGxvbmcgQUNUSU9OX0VWRU5UX01BU0sgPSAxMjg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQURKVVNUTUVOVF9FVkVOVF9NQVNLIGluZGljYXRlcyB0aGUgZXZlbnQgcmVsYXRlcyB0byBhbgotICAgICAqIGFkanVzdG1lbnQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBsb25nIEFESlVTVE1FTlRfRVZFTlRfTUFTSyA9IDI1NjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBJVEVNX0VWRU5UX01BU0sgaW5kaWNhdGVzIHRoZSBldmVudCByZWxhdGVzIHRvIGFuIGl0ZW0uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBsb25nIElURU1fRVZFTlRfTUFTSyA9IDUxMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBURVhUX0VWRU5UX01BU0sgaW5kaWNhdGVzIHRoZSBldmVudCByZWxhdGVzIHRvIHRleHQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBsb25nIFRFWFRfRVZFTlRfTUFTSyA9IDEwMjQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgSU5QVVRfTUVUSE9EX0VWRU5UX01BU0sgaW5kaWNhdGVzIHRoZSBldmVudCByZWxhdGVzIHRvIGFuCi0gICAgICogaW5wdXQgbWV0aG9kLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgbG9uZyBJTlBVVF9NRVRIT0RfRVZFTlRfTUFTSyA9IDIwNDg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUEFJTlRfRVZFTlRfTUFTSyBpbmRpY2F0ZXMgdGhlIGV2ZW50IHJlbGF0ZXMgdG8gYSBwYWludAotICAgICAqIG1ldGhvZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGxvbmcgUEFJTlRfRVZFTlRfTUFTSyA9IDgxOTI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgSU5WT0NBVElPTl9FVkVOVF9NQVNLIGluZGljYXRlcyB0aGUgZXZlbnQgcmVsYXRlcyB0byBhCi0gICAgICogbWV0aG9kIGludm9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBsb25nIElOVk9DQVRJT05fRVZFTlRfTUFTSyA9IDE2Mzg0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEhJRVJBUkNIWV9FVkVOVF9NQVNLIGluZGljYXRlcyB0aGUgZXZlbnQgcmVsYXRlcyB0byBhCi0gICAgICogaGllcmFyY2h5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgbG9uZyBISUVSQVJDSFlfRVZFTlRfTUFTSyA9IDMyNzY4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEhJRVJBUkNIWV9CT1VORFNfRVZFTlRfTUFTSyBpbmRpY2F0ZXMgdGhlIGV2ZW50IHJlbGF0ZXMgdG8KLSAgICAgKiBoaWVyYXJjaHkgYm91bmRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgbG9uZyBISUVSQVJDSFlfQk9VTkRTX0VWRU5UX01BU0sgPSA2NTUzNjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBNT1VTRV9XSEVFTF9FVkVOVF9NQVNLIGluZGljYXRlcyB0aGUgZXZlbnQgcmVsYXRlcyB0byB0aGUKLSAgICAgKiBtb3VzZSB3aGVlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGxvbmcgTU9VU0VfV0hFRUxfRVZFTlRfTUFTSyA9IDEzMTA3MjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBXSU5ET1dfU1RBVEVfRVZFTlRfTUFTSyBpbmRpY2F0ZXMgdGhlIGV2ZW50IHJlbGF0ZXMgdG8gYQotICAgICAqIHdpbmRvdyBzdGF0ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGxvbmcgV0lORE9XX1NUQVRFX0VWRU5UX01BU0sgPSAyNjIxNDQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgV0lORE9XX0ZPQ1VTX0VWRU5UX01BU0sgaW5kaWNhdGVzIHRoZSBldmVudCByZWxhdGVzIHRvIGEKLSAgICAgKiB3aW5kb3cgZm9jdXMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBsb25nIFdJTkRPV19GT0NVU19FVkVOVF9NQVNLID0gNTI0Mjg4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFJFU0VSVkVEX0lEX01BWCBpbmRpY2F0ZXMgdGhlIG1heGltdW0gdmFsdWUgZm9yIHJlc2VydmVkIEFXVAotICAgICAqIGV2ZW50IElEcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBSRVNFUlZFRF9JRF9NQVggPSAxOTk5OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGV2ZW50c01hcC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBIYXNodGFibGU8SW50ZWdlciwgRXZlbnREZXNjcmlwdG9yPiBldmVudHNNYXAgPSBuZXcgSGFzaHRhYmxlPEludGVnZXIsIEV2ZW50RGVzY3JpcHRvcj4oKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb252ZXJ0ZXIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgRXZlbnRDb252ZXJ0ZXIgY29udmVydGVyOwotCi0gICAgLyoqCi0gICAgICogVGhlIElEIG9mIHRoZSBldmVudC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IGlkOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbnN1bWVkIGluZGljYXRlcyB3aGV0aGVyIG9yIG5vdCB0aGUgZXZlbnQgaXMgc2VudCBiYWNrIGRvd24gdG8gdGhlCi0gICAgICogcGVlciBvbmNlIHRoZSBzb3VyY2UgaGFzIHByb2Nlc3NlZCBpdCAoZmFsc2UgbWVhbnMgaXQncyBzZW50IHRvIHRoZSBwZWVyLAotICAgICAqIHRydWUgbWVhbnMgaXQncyBub3QpLgotICAgICAqLwotICAgIHByb3RlY3RlZCBib29sZWFuIGNvbnN1bWVkOwotCi0gICAgLyoqCi0gICAgICogVGhlIGRpc3BhdGNoZWQgYnkga2ZtLgotICAgICAqLwotICAgIGJvb2xlYW4gZGlzcGF0Y2hlZEJ5S0ZNOwotCi0gICAgLyoqCi0gICAgICogVGhlIGlzIHBvc3RlZC4KLSAgICAgKi8KLSAgICB0cmFuc2llbnQgYm9vbGVhbiBpc1Bvc3RlZDsKLQotICAgIHN0YXRpYyB7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoS2V5RXZlbnQuS0VZX1RZUEVEKSwgbmV3IEV2ZW50RGVzY3JpcHRvcihLRVlfRVZFTlRfTUFTSywKLSAgICAgICAgICAgICAgICBLZXlMaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKEtleUV2ZW50LktFWV9QUkVTU0VEKSwgbmV3IEV2ZW50RGVzY3JpcHRvcihLRVlfRVZFTlRfTUFTSywKLSAgICAgICAgICAgICAgICBLZXlMaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKEtleUV2ZW50LktFWV9SRUxFQVNFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoS0VZX0VWRU5UX01BU0ssCi0gICAgICAgICAgICAgICAgS2V5TGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihNb3VzZUV2ZW50Lk1PVVNFX0NMSUNLRUQpLCBuZXcgRXZlbnREZXNjcmlwdG9yKE1PVVNFX0VWRU5UX01BU0ssCi0gICAgICAgICAgICAgICAgTW91c2VMaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKE1vdXNlRXZlbnQuTU9VU0VfUFJFU1NFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoTU9VU0VfRVZFTlRfTUFTSywKLSAgICAgICAgICAgICAgICBNb3VzZUxpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoTW91c2VFdmVudC5NT1VTRV9SRUxFQVNFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoTU9VU0VfRVZFTlRfTUFTSywKLSAgICAgICAgICAgICAgICBNb3VzZUxpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoTW91c2VFdmVudC5NT1VTRV9NT1ZFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgTU9VU0VfTU9USU9OX0VWRU5UX01BU0ssIE1vdXNlTW90aW9uTGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihNb3VzZUV2ZW50Lk1PVVNFX0VOVEVSRUQpLCBuZXcgRXZlbnREZXNjcmlwdG9yKE1PVVNFX0VWRU5UX01BU0ssCi0gICAgICAgICAgICAgICAgTW91c2VMaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKE1vdXNlRXZlbnQuTU9VU0VfRVhJVEVEKSwgbmV3IEV2ZW50RGVzY3JpcHRvcihNT1VTRV9FVkVOVF9NQVNLLAotICAgICAgICAgICAgICAgIE1vdXNlTGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihNb3VzZUV2ZW50Lk1PVVNFX0RSQUdHRUQpLCBuZXcgRXZlbnREZXNjcmlwdG9yKAotICAgICAgICAgICAgICAgIE1PVVNFX01PVElPTl9FVkVOVF9NQVNLLCBNb3VzZU1vdGlvbkxpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoTW91c2VFdmVudC5NT1VTRV9XSEVFTCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgTU9VU0VfV0hFRUxfRVZFTlRfTUFTSywgTW91c2VXaGVlbExpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoQ29tcG9uZW50RXZlbnQuQ09NUE9ORU5UX01PVkVEKSwgbmV3IEV2ZW50RGVzY3JpcHRvcigKLSAgICAgICAgICAgICAgICBDT01QT05FTlRfRVZFTlRfTUFTSywgQ29tcG9uZW50TGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihDb21wb25lbnRFdmVudC5DT01QT05FTlRfUkVTSVpFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgQ09NUE9ORU5UX0VWRU5UX01BU0ssIENvbXBvbmVudExpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoQ29tcG9uZW50RXZlbnQuQ09NUE9ORU5UX1NIT1dOKSwgbmV3IEV2ZW50RGVzY3JpcHRvcigKLSAgICAgICAgICAgICAgICBDT01QT05FTlRfRVZFTlRfTUFTSywgQ29tcG9uZW50TGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihDb21wb25lbnRFdmVudC5DT01QT05FTlRfSElEREVOKSwgbmV3IEV2ZW50RGVzY3JpcHRvcigKLSAgICAgICAgICAgICAgICBDT01QT05FTlRfRVZFTlRfTUFTSywgQ29tcG9uZW50TGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihGb2N1c0V2ZW50LkZPQ1VTX0dBSU5FRCksIG5ldyBFdmVudERlc2NyaXB0b3IoRk9DVVNfRVZFTlRfTUFTSywKLSAgICAgICAgICAgICAgICBGb2N1c0xpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoRm9jdXNFdmVudC5GT0NVU19MT1NUKSwgbmV3IEV2ZW50RGVzY3JpcHRvcihGT0NVU19FVkVOVF9NQVNLLAotICAgICAgICAgICAgICAgIEZvY3VzTGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihQYWludEV2ZW50LlBBSU5UKSwgbmV3IEV2ZW50RGVzY3JpcHRvcihQQUlOVF9FVkVOVF9NQVNLLCBudWxsKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoUGFpbnRFdmVudC5VUERBVEUpLCBuZXcgRXZlbnREZXNjcmlwdG9yKFBBSU5UX0VWRU5UX01BU0ssIG51bGwpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihXaW5kb3dFdmVudC5XSU5ET1dfT1BFTkVEKSwgbmV3IEV2ZW50RGVzY3JpcHRvcigKLSAgICAgICAgICAgICAgICBXSU5ET1dfRVZFTlRfTUFTSywgV2luZG93TGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihXaW5kb3dFdmVudC5XSU5ET1dfQ0xPU0lORyksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgV0lORE9XX0VWRU5UX01BU0ssIFdpbmRvd0xpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoV2luZG93RXZlbnQuV0lORE9XX0NMT1NFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgV0lORE9XX0VWRU5UX01BU0ssIFdpbmRvd0xpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoV2luZG93RXZlbnQuV0lORE9XX0RFSUNPTklGSUVEKSwgbmV3IEV2ZW50RGVzY3JpcHRvcigKLSAgICAgICAgICAgICAgICBXSU5ET1dfRVZFTlRfTUFTSywgV2luZG93TGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihXaW5kb3dFdmVudC5XSU5ET1dfSUNPTklGSUVEKSwgbmV3IEV2ZW50RGVzY3JpcHRvcigKLSAgICAgICAgICAgICAgICBXSU5ET1dfRVZFTlRfTUFTSywgV2luZG93TGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihXaW5kb3dFdmVudC5XSU5ET1dfU1RBVEVfQ0hBTkdFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgV0lORE9XX1NUQVRFX0VWRU5UX01BU0ssIFdpbmRvd1N0YXRlTGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihXaW5kb3dFdmVudC5XSU5ET1dfTE9TVF9GT0NVUyksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgV0lORE9XX0ZPQ1VTX0VWRU5UX01BU0ssIFdpbmRvd0ZvY3VzTGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihXaW5kb3dFdmVudC5XSU5ET1dfR0FJTkVEX0ZPQ1VTKSwgbmV3IEV2ZW50RGVzY3JpcHRvcigKLSAgICAgICAgICAgICAgICBXSU5ET1dfRk9DVVNfRVZFTlRfTUFTSywgV2luZG93Rm9jdXNMaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKFdpbmRvd0V2ZW50LldJTkRPV19ERUFDVElWQVRFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgV0lORE9XX0VWRU5UX01BU0ssIFdpbmRvd0xpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoV2luZG93RXZlbnQuV0lORE9XX0FDVElWQVRFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgV0lORE9XX0VWRU5UX01BU0ssIFdpbmRvd0xpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoSGllcmFyY2h5RXZlbnQuSElFUkFSQ0hZX0NIQU5HRUQpLCBuZXcgRXZlbnREZXNjcmlwdG9yKAotICAgICAgICAgICAgICAgIEhJRVJBUkNIWV9FVkVOVF9NQVNLLCBIaWVyYXJjaHlMaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKEhpZXJhcmNoeUV2ZW50LkFOQ0VTVE9SX01PVkVEKSwgbmV3IEV2ZW50RGVzY3JpcHRvcigKLSAgICAgICAgICAgICAgICBISUVSQVJDSFlfQk9VTkRTX0VWRU5UX01BU0ssIEhpZXJhcmNoeUJvdW5kc0xpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoSGllcmFyY2h5RXZlbnQuQU5DRVNUT1JfUkVTSVpFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgSElFUkFSQ0hZX0JPVU5EU19FVkVOVF9NQVNLLCBIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKENvbnRhaW5lckV2ZW50LkNPTVBPTkVOVF9BRERFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgQ09OVEFJTkVSX0VWRU5UX01BU0ssIENvbnRhaW5lckxpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoQ29udGFpbmVyRXZlbnQuQ09NUE9ORU5UX1JFTU9WRUQpLCBuZXcgRXZlbnREZXNjcmlwdG9yKAotICAgICAgICAgICAgICAgIENPTlRBSU5FUl9FVkVOVF9NQVNLLCBDb250YWluZXJMaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKElucHV0TWV0aG9kRXZlbnQuSU5QVVRfTUVUSE9EX1RFWFRfQ0hBTkdFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgSU5QVVRfTUVUSE9EX0VWRU5UX01BU0ssIElucHV0TWV0aG9kTGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihJbnB1dE1ldGhvZEV2ZW50LkNBUkVUX1BPU0lUSU9OX0NIQU5HRUQpLCBuZXcgRXZlbnREZXNjcmlwdG9yKAotICAgICAgICAgICAgICAgIElOUFVUX01FVEhPRF9FVkVOVF9NQVNLLCBJbnB1dE1ldGhvZExpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoSW52b2NhdGlvbkV2ZW50LklOVk9DQVRJT05fREVGQVVMVCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgSU5WT0NBVElPTl9FVkVOVF9NQVNLLCBudWxsKSk7Ci0gICAgICAgIGV2ZW50c01hcC5wdXQobmV3IEludGVnZXIoSXRlbUV2ZW50LklURU1fU1RBVEVfQ0hBTkdFRCksIG5ldyBFdmVudERlc2NyaXB0b3IoCi0gICAgICAgICAgICAgICAgSVRFTV9FVkVOVF9NQVNLLCBJdGVtTGlzdGVuZXIuY2xhc3MpKTsKLSAgICAgICAgZXZlbnRzTWFwLnB1dChuZXcgSW50ZWdlcihUZXh0RXZlbnQuVEVYVF9WQUxVRV9DSEFOR0VEKSwgbmV3IEV2ZW50RGVzY3JpcHRvcigKLSAgICAgICAgICAgICAgICBURVhUX0VWRU5UX01BU0ssIFRleHRMaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKEFjdGlvbkV2ZW50LkFDVElPTl9QRVJGT1JNRUQpLCBuZXcgRXZlbnREZXNjcmlwdG9yKAotICAgICAgICAgICAgICAgIEFDVElPTl9FVkVOVF9NQVNLLCBBY3Rpb25MaXN0ZW5lci5jbGFzcykpOwotICAgICAgICBldmVudHNNYXAucHV0KG5ldyBJbnRlZ2VyKEFkanVzdG1lbnRFdmVudC5BREpVU1RNRU5UX1ZBTFVFX0NIQU5HRUQpLCBuZXcgRXZlbnREZXNjcmlwdG9yKAotICAgICAgICAgICAgICAgIEFESlVTVE1FTlRfRVZFTlRfTUFTSywgQWRqdXN0bWVudExpc3RlbmVyLmNsYXNzKSk7Ci0gICAgICAgIGNvbnZlcnRlciA9IG5ldyBFdmVudENvbnZlcnRlcigpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBBV1QgZXZlbnQgZnJvbSB0aGUgc3BlY2lmaWVkIEV2ZW50IG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXZlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBFdmVudCBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIEFXVEV2ZW50KEV2ZW50IGV2ZW50KSB7Ci0gICAgICAgIHRoaXMoZXZlbnQudGFyZ2V0LCBldmVudC5pZCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEFXVCBldmVudCB3aXRoIHRoZSBzcGVjaWZpZWQgb2JqZWN0IGFuZCB0eXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgT2JqZWN0LgotICAgICAqIEBwYXJhbSBpZAotICAgICAqICAgICAgICAgICAgdGhlIGV2ZW50J3MgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQVdURXZlbnQoT2JqZWN0IHNvdXJjZSwgaW50IGlkKSB7Ci0gICAgICAgIHN1cGVyKHNvdXJjZSk7Ci0gICAgICAgIHRoaXMuaWQgPSBpZDsKLSAgICAgICAgY29uc3VtZWQgPSBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBldmVudCdzIHR5cGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZXZlbnQgdHlwZSBJRC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldElEKCkgewotICAgICAgICByZXR1cm4gaWQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhIG5ldyBzb3VyY2UgZm9yIHRoZSBBV1RFdmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbmV3U291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHNvdXJjZSBPYmplY3QgZm9yIHRoZSBBV1RFdmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRTb3VyY2UoT2JqZWN0IG5ld1NvdXJjZSkgewotICAgICAgICBzb3VyY2UgPSBuZXdTb3VyY2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIFN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgQVdURXZlbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgU3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBBV1RFdmVudC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICAvKgotICAgICAgICAgKiBUaGUgZm9ybWF0IGlzIGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW9yIHdoaWNoIGNhbiBiZSByZXZlYWxlZCBieQotICAgICAgICAgKiB0aGUgZm9sbG93aW5nIGNvZGU6IEFXVEV2ZW50IGV2ZW50ID0gbmV3IEFXVEV2ZW50KG5ldyBDb21wb25lbnQoKXt9LAotICAgICAgICAgKiAxKXt9OyBTeXN0ZW0ub3V0LnByaW50bG4oZXZlbnQpOwotICAgICAgICAgKi8KLSAgICAgICAgU3RyaW5nIG5hbWUgPSAiIjsgLy8kTk9OLU5MUy0xJAotCi0gICAgICAgIGlmIChzb3VyY2UgaW5zdGFuY2VvZiBDb21wb25lbnQgJiYgKHNvdXJjZSAhPSBudWxsKSkgewotICAgICAgICAgICAgQ29tcG9uZW50IGNvbXAgPSAoQ29tcG9uZW50KWdldFNvdXJjZSgpOwotICAgICAgICAgICAgbmFtZSA9IGNvbXAuZ2V0TmFtZSgpOwotICAgICAgICAgICAgaWYgKG5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIG5hbWUgPSAiIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIChnZXRDbGFzcygpLmdldE5hbWUoKSArICJbIiArIHBhcmFtU3RyaW5nKCkgKyAiXSIgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgKyAiIG9uICIgKyAobmFtZS5sZW5ndGgoKSA+IDAgPyBuYW1lIDogc291cmNlKSk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBBV1RFdmVudCBzdGF0ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBBV1RFdmVudCBzdGF0ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIHBhcmFtU3RyaW5nKCkgewotICAgICAgICAvLyBub3RoaW5nIHRvIGltcGxlbWVudDogYWxsIGV2ZW50IHR5cGVzIG11c3Qgb3ZlcnJpZGUgdGhpcyBtZXRob2QKLSAgICAgICAgcmV0dXJuICIiOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IHRoaXMgQVdURXZlbnQgaGFzIGJlZW4gY29uc3VtZWQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIEFXVEV2ZW50IGhhcyBiZWVuIGNvbnN1bWVkLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGJvb2xlYW4gaXNDb25zdW1lZCgpIHsKLSAgICAgICAgcmV0dXJuIGNvbnN1bWVkOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbnN1bWVzIHRoZSBBV1RFdmVudC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBjb25zdW1lKCkgewotICAgICAgICBjb25zdW1lZCA9IHRydWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29udmVydCBBV1RFdmVudCBvYmplY3QgdG8gYSBjb3JyZXNwb25kaW5nIChkZXByZWNhdGVkKSBFdmVudCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiBuZXcgRXZlbnQgb2JqZWN0IHdoaWNoIGlzIGEgY29udmVydGVkIEFXVEV2ZW50IG9iamVjdCBvciBudWxsIGlmCi0gICAgICogICAgICAgICB0aGUgY29udmVyc2lvbiBpcyBub3QgcG9zc2libGUKLSAgICAgKi8KLSAgICBFdmVudCBnZXRFdmVudCgpIHsKLQotICAgICAgICBpZiAoaWQgPT0gQWN0aW9uRXZlbnQuQUNUSU9OX1BFUkZPUk1FRCkgewotICAgICAgICAgICAgQWN0aW9uRXZlbnQgYWUgPSAoQWN0aW9uRXZlbnQpdGhpczsKLSAgICAgICAgICAgIHJldHVybiBjb252ZXJ0ZXIuY29udmVydEFjdGlvbkV2ZW50KGFlKTsKLQotICAgICAgICB9IGVsc2UgaWYgKGlkID09IEFkanVzdG1lbnRFdmVudC5BREpVU1RNRU5UX1ZBTFVFX0NIQU5HRUQpIHsKLSAgICAgICAgICAgIEFkanVzdG1lbnRFdmVudCBhZSA9IChBZGp1c3RtZW50RXZlbnQpdGhpczsKLSAgICAgICAgICAgIHJldHVybiBjb252ZXJ0ZXIuY29udmVydEFkanVzdG1lbnRFdmVudChhZSk7Ci0KLSAgICAgICAgICAgIC8vID8/P0FXVAotICAgICAgICAgICAgLy8gfSBlbHNlIGlmIChpZCA9PSBDb21wb25lbnRFdmVudC5DT01QT05FTlRfTU9WRUQKLSAgICAgICAgICAgIC8vICYmIHNvdXJjZSBpbnN0YW5jZW9mIFdpbmRvdykgewotICAgICAgICAgICAgLy8gLy90aGUgb25seSB0eXBlIG9mIENvbXBvbmVudCBldmVudHMgaXMgQ09NUE9ORU5UX01PVkVEIG9uIHdpbmRvdwotICAgICAgICAgICAgLy8gQ29tcG9uZW50RXZlbnQgY2UgPSAoQ29tcG9uZW50RXZlbnQpIHRoaXM7Ci0gICAgICAgICAgICAvLyByZXR1cm4gY29udmVydGVyLmNvbnZlcnRDb21wb25lbnRFdmVudChjZSk7Ci0KLSAgICAgICAgfSBlbHNlIGlmIChpZCA+PSBGb2N1c0V2ZW50LkZPQ1VTX0ZJUlNUICYmIGlkIDw9IEZvY3VzRXZlbnQuRk9DVVNfTEFTVCkgewotICAgICAgICAgICAgLy8gbm90aGluZyB0byBjb252ZXJ0Ci0KLSAgICAgICAgICAgIC8vID8/P0FXVAotICAgICAgICAgICAgLy8gfSBlbHNlIGlmIChpZCA9PSBJdGVtRXZlbnQuSVRFTV9TVEFURV9DSEFOR0VEKSB7Ci0gICAgICAgICAgICAvLyBJdGVtRXZlbnQgaWUgPSAoSXRlbUV2ZW50KSB0aGlzOwotICAgICAgICAgICAgLy8gcmV0dXJuIGNvbnZlcnRlci5jb252ZXJ0SXRlbUV2ZW50KGllKTsKLQotICAgICAgICB9IGVsc2UgaWYgKGlkID09IEtleUV2ZW50LktFWV9QUkVTU0VEIHx8IGlkID09IEtleUV2ZW50LktFWV9SRUxFQVNFRCkgewotICAgICAgICAgICAgS2V5RXZlbnQga2UgPSAoS2V5RXZlbnQpdGhpczsKLSAgICAgICAgICAgIHJldHVybiBjb252ZXJ0ZXIuY29udmVydEtleUV2ZW50KGtlKTsKLSAgICAgICAgfSBlbHNlIGlmIChpZCA+PSBNb3VzZUV2ZW50Lk1PVVNFX0ZJUlNUICYmIGlkIDw9IE1vdXNlRXZlbnQuTU9VU0VfTEFTVCkgewotICAgICAgICAgICAgTW91c2VFdmVudCBtZSA9IChNb3VzZUV2ZW50KXRoaXM7Ci0gICAgICAgICAgICByZXR1cm4gY29udmVydGVyLmNvbnZlcnRNb3VzZUV2ZW50KG1lKTsKLSAgICAgICAgfSBlbHNlIGlmIChpZCA9PSBXaW5kb3dFdmVudC5XSU5ET1dfQ0xPU0lORyB8fCBpZCA9PSBXaW5kb3dFdmVudC5XSU5ET1dfSUNPTklGSUVECi0gICAgICAgICAgICAgICAgfHwgaWQgPT0gV2luZG93RXZlbnQuV0lORE9XX0RFSUNPTklGSUVEKSB7Ci0gICAgICAgICAgICAvLyBub3RoaW5nIHRvIGNvbnZlcnQKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBuZXcgRXZlbnQoc291cmNlLCBpZCwgbnVsbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIGNsYXNzIEV2ZW50RGVzY3JpcHRvci4KLSAgICAgKi8KLSAgICBzdGF0aWMgZmluYWwgY2xhc3MgRXZlbnREZXNjcmlwdG9yIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGV2ZW50IG1hc2suCi0gICAgICAgICAqLwotICAgICAgICBmaW5hbCBsb25nIGV2ZW50TWFzazsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGxpc3RlbmVyIHR5cGUuCi0gICAgICAgICAqLwotICAgICAgICBmaW5hbCBDbGFzczw/IGV4dGVuZHMgRXZlbnRMaXN0ZW5lcj4gbGlzdGVuZXJUeXBlOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZXZlbnQgZGVzY3JpcHRvci4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBldmVudE1hc2sKLSAgICAgICAgICogICAgICAgICAgICB0aGUgZXZlbnQgbWFzay4KLSAgICAgICAgICogQHBhcmFtIGxpc3RlbmVyVHlwZQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBsaXN0ZW5lciB0eXBlLgotICAgICAgICAgKi8KLSAgICAgICAgRXZlbnREZXNjcmlwdG9yKGxvbmcgZXZlbnRNYXNrLCBDbGFzczw/IGV4dGVuZHMgRXZlbnRMaXN0ZW5lcj4gbGlzdGVuZXJUeXBlKSB7Ci0gICAgICAgICAgICB0aGlzLmV2ZW50TWFzayA9IGV2ZW50TWFzazsKLSAgICAgICAgICAgIHRoaXMubGlzdGVuZXJUeXBlID0gbGlzdGVuZXJUeXBlOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY2xhc3MgRXZlbnRUeXBlTG9va3VwLgotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBjbGFzcyBFdmVudFR5cGVMb29rdXAgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbGFzdCBldmVudC4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgQVdURXZlbnQgbGFzdEV2ZW50ID0gbnVsbDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGxhc3QgZXZlbnQgZGVzY3JpcHRvci4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgRXZlbnREZXNjcmlwdG9yIGxhc3RFdmVudERlc2NyaXB0b3IgPSBudWxsOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBHZXRzIHRoZSBldmVudCBkZXNjcmlwdG9yLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGV2ZW50Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGV2ZW50LgotICAgICAgICAgKiBAcmV0dXJuIHRoZSBldmVudCBkZXNjcmlwdG9yLgotICAgICAgICAgKi8KLSAgICAgICAgRXZlbnREZXNjcmlwdG9yIGdldEV2ZW50RGVzY3JpcHRvcihBV1RFdmVudCBldmVudCkgewotICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGV2ZW50ICE9IGxhc3RFdmVudCkgewotICAgICAgICAgICAgICAgICAgICBsYXN0RXZlbnQgPSBldmVudDsKLSAgICAgICAgICAgICAgICAgICAgbGFzdEV2ZW50RGVzY3JpcHRvciA9IGV2ZW50c01hcC5nZXQobmV3IEludGVnZXIoZXZlbnQuaWQpKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICByZXR1cm4gbGFzdEV2ZW50RGVzY3JpcHRvcjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBHZXRzIHRoZSBldmVudCBtYXNrLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGV2ZW50Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGV2ZW50LgotICAgICAgICAgKiBAcmV0dXJuIHRoZSBldmVudCBtYXNrLgotICAgICAgICAgKi8KLSAgICAgICAgbG9uZyBnZXRFdmVudE1hc2soQVdURXZlbnQgZXZlbnQpIHsKLSAgICAgICAgICAgIGZpbmFsIEV2ZW50RGVzY3JpcHRvciBlZCA9IGdldEV2ZW50RGVzY3JpcHRvcihldmVudCk7Ci0gICAgICAgICAgICByZXR1cm4gZWQgPT0gbnVsbCA/IC0xIDogZWQuZXZlbnRNYXNrOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIGNsYXNzIEV2ZW50Q29udmVydGVyLgotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBjbGFzcyBFdmVudENvbnZlcnRlciB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjb25zdGFudCBPTERfTU9EX01BU0suCi0gICAgICAgICAqLwotICAgICAgICBzdGF0aWMgZmluYWwgaW50IE9MRF9NT0RfTUFTSyA9IEV2ZW50LkFMVF9NQVNLIHwgRXZlbnQuQ1RSTF9NQVNLIHwgRXZlbnQuTUVUQV9NQVNLCi0gICAgICAgICAgICAgICAgfCBFdmVudC5TSElGVF9NQVNLOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDb252ZXJ0IGFjdGlvbiBldmVudC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBhZQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBhZS4KLSAgICAgICAgICogQHJldHVybiB0aGUgZXZlbnQuCi0gICAgICAgICAqLwotICAgICAgICBFdmVudCBjb252ZXJ0QWN0aW9uRXZlbnQoQWN0aW9uRXZlbnQgYWUpIHsKLSAgICAgICAgICAgIEV2ZW50IGV2dCA9IG5ldyBFdmVudChhZS5nZXRTb3VyY2UoKSwgYWUuZ2V0SUQoKSwgYWUuZ2V0QWN0aW9uQ29tbWFuZCgpKTsKLSAgICAgICAgICAgIGV2dC53aGVuID0gYWUuZ2V0V2hlbigpOwotICAgICAgICAgICAgZXZ0Lm1vZGlmaWVycyA9IGFlLmdldE1vZGlmaWVycygpICYgT0xEX01PRF9NQVNLOwotCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgICogaWYgKHNvdXJjZSBpbnN0YW5jZW9mIEJ1dHRvbikgeyBhcmcgPSAoKEJ1dHRvbikKLSAgICAgICAgICAgICAqIHNvdXJjZSkuZ2V0TGFiZWwoKTsgfSBlbHNlIGlmIChzb3VyY2UgaW5zdGFuY2VvZiBDaGVja2JveCkgeyBhcmcKLSAgICAgICAgICAgICAqID0gbmV3IEJvb2xlYW4oKChDaGVja2JveCkgc291cmNlKS5nZXRTdGF0ZSgpKTsgfSBlbHNlIGlmIChzb3VyY2UKLSAgICAgICAgICAgICAqIGluc3RhbmNlb2YgQ2hlY2tib3hNZW51SXRlbSkgeyBhcmcgPSAoKENoZWNrYm94TWVudUl0ZW0pCi0gICAgICAgICAgICAgKiBzb3VyY2UpLmdldExhYmVsKCk7IH0gZWxzZSBpZiAoc291cmNlIGluc3RhbmNlb2YgQ2hvaWNlKSB7IGFyZyA9Ci0gICAgICAgICAgICAgKiAoKENob2ljZSkgc291cmNlKS5nZXRTZWxlY3RlZEl0ZW0oKTsgfSBlbHNlIGlmIChzb3VyY2UgaW5zdGFuY2VvZgotICAgICAgICAgICAgICogTGlzdCkgeyBhcmcgPSAoKExpc3QpIHNvdXJjZSkuZ2V0U2VsZWN0ZWRJdGVtKCk7IH0gZWxzZSBpZgotICAgICAgICAgICAgICogKHNvdXJjZSBpbnN0YW5jZW9mIE1lbnVJdGVtKSB7IGFyZyA9ICgoTWVudUl0ZW0pCi0gICAgICAgICAgICAgKiBzb3VyY2UpLmdldExhYmVsKCk7IH0gZWxzZSBpZiAoc291cmNlIGluc3RhbmNlb2YgVGV4dEZpZWxkKSB7IGFyZwotICAgICAgICAgICAgICogPSAoKFRleHRGaWVsZCkgc291cmNlKS5nZXRUZXh0KCk7IH0KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgcmV0dXJuIGV2dDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDb252ZXJ0IGFkanVzdG1lbnQgZXZlbnQuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gYWUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgYWUuCi0gICAgICAgICAqIEByZXR1cm4gdGhlIGV2ZW50LgotICAgICAgICAgKi8KLSAgICAgICAgRXZlbnQgY29udmVydEFkanVzdG1lbnRFdmVudChBZGp1c3RtZW50RXZlbnQgYWUpIHsKLSAgICAgICAgICAgIC8vIFRPRE86IEV2ZW50LlNDUk9MTF9CRUdJTi9TQ1JPTExfRU5ECi0gICAgICAgICAgICByZXR1cm4gbmV3IEV2ZW50KGFlLnNvdXJjZSwgYWUuaWQgKyBhZS5nZXRBZGp1c3RtZW50VHlwZSgpIC0gMSwgbmV3IEludGVnZXIoYWUKLSAgICAgICAgICAgICAgICAgICAgLmdldFZhbHVlKCkpKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDb252ZXJ0IGNvbXBvbmVudCBldmVudC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBjZQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBjZS4KLSAgICAgICAgICogQHJldHVybiB0aGUgZXZlbnQuCi0gICAgICAgICAqLwotICAgICAgICBFdmVudCBjb252ZXJ0Q29tcG9uZW50RXZlbnQoQ29tcG9uZW50RXZlbnQgY2UpIHsKLSAgICAgICAgICAgIENvbXBvbmVudCBjb21wID0gY2UuZ2V0Q29tcG9uZW50KCk7Ci0gICAgICAgICAgICBFdmVudCBldnQgPSBuZXcgRXZlbnQoY29tcCwgRXZlbnQuV0lORE9XX01PVkVELCBudWxsKTsKLSAgICAgICAgICAgIGV2dC54ID0gY29tcC5nZXRYKCk7Ci0gICAgICAgICAgICBldnQueSA9IGNvbXAuZ2V0WSgpOwotICAgICAgICAgICAgcmV0dXJuIGV2dDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vID8/P0FXVAotICAgICAgICAvKgotICAgICAgICAgKiBFdmVudCBjb252ZXJ0SXRlbUV2ZW50KEl0ZW1FdmVudCBpZSkgeyBpbnQgb2xkSWQgPSBpZS5pZCArCi0gICAgICAgICAqIGllLmdldFN0YXRlQ2hhbmdlKCkgLSAxOyBPYmplY3Qgc291cmNlID0gaWUuc291cmNlOyBpbnQgaWR4ID0gLTE7IGlmCi0gICAgICAgICAqIChzb3VyY2UgaW5zdGFuY2VvZiBMaXN0KSB7IExpc3QgbGlzdCA9IChMaXN0KSBzb3VyY2U7IGlkeCA9Ci0gICAgICAgICAqIGxpc3QuZ2V0U2VsZWN0ZWRJbmRleCgpOyB9IGVsc2UgaWYgKHNvdXJjZSBpbnN0YW5jZW9mIENob2ljZSkgewotICAgICAgICAgKiBDaG9pY2UgY2hvaWNlID0gKENob2ljZSkgc291cmNlOyBpZHggPSBjaG9pY2UuZ2V0U2VsZWN0ZWRJbmRleCgpOyB9Ci0gICAgICAgICAqIE9iamVjdCBhcmcgPSBpZHggPj0gMCA/IG5ldyBJbnRlZ2VyKGlkeCkgOiBudWxsOyByZXR1cm4gbmV3Ci0gICAgICAgICAqIEV2ZW50KHNvdXJjZSwgb2xkSWQsIGFyZyk7IH0KLSAgICAgICAgICovCi0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENvbnZlcnQga2V5IGV2ZW50LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGtlCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGtlLgotICAgICAgICAgKiBAcmV0dXJuIHRoZSBldmVudC4KLSAgICAgICAgICovCi0gICAgICAgIEV2ZW50IGNvbnZlcnRLZXlFdmVudChLZXlFdmVudCBrZSkgewotICAgICAgICAgICAgaW50IG9sZElkID0ga2UuaWQ7Ci0gICAgICAgICAgICAvLyBsZWF2ZSBvbmx5IG9sZCBFdmVudCdzIG1vZGlmaWVycwotCi0gICAgICAgICAgICBpbnQgbW9kID0ga2UuZ2V0TW9kaWZpZXJzKCkgJiBPTERfTU9EX01BU0s7Ci0gICAgICAgICAgICBDb21wb25lbnQgY29tcCA9IGtlLmdldENvbXBvbmVudCgpOwotICAgICAgICAgICAgY2hhciBrZXlDaGFyID0ga2UuZ2V0S2V5Q2hhcigpOwotICAgICAgICAgICAgaW50IGtleUNvZGUgPSBrZS5nZXRLZXlDb2RlKCk7Ci0gICAgICAgICAgICBpbnQga2V5ID0gY29udmVydEtleShrZXlDaGFyLCBrZXlDb2RlKTsKLSAgICAgICAgICAgIGlmIChrZXkgPj0gRXZlbnQuSE9NRSAmJiBrZXkgPD0gRXZlbnQuSU5TRVJUKSB7Ci0gICAgICAgICAgICAgICAgb2xkSWQgKz0gMjsgLy8gbm9uLUFTQ0lJIGtleSAtPiBhY3Rpb24ga2V5Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gbmV3IEV2ZW50KGNvbXAsIGtlLmdldFdoZW4oKSwgb2xkSWQsIDAsIDAsIGtleSwgbW9kKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDb252ZXJ0IG1vdXNlIGV2ZW50LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIG1lCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIG1lLgotICAgICAgICAgKiBAcmV0dXJuIHRoZSBldmVudC4KLSAgICAgICAgICovCi0gICAgICAgIEV2ZW50IGNvbnZlcnRNb3VzZUV2ZW50KE1vdXNlRXZlbnQgbWUpIHsKLSAgICAgICAgICAgIGludCBpZCA9IG1lLmlkOwotICAgICAgICAgICAgaWYgKGlkICE9IE1vdXNlRXZlbnQuTU9VU0VfQ0xJQ0tFRCkgewotICAgICAgICAgICAgICAgIEV2ZW50IGV2dCA9IG5ldyBFdmVudChtZS5zb3VyY2UsIGlkLCBudWxsKTsKLSAgICAgICAgICAgICAgICBldnQueCA9IG1lLmdldFgoKTsKLSAgICAgICAgICAgICAgICBldnQueSA9IG1lLmdldFkoKTsKLSAgICAgICAgICAgICAgICBpbnQgbW9kID0gbWUuZ2V0TW9kaWZpZXJzKCk7Ci0gICAgICAgICAgICAgICAgLy8gaW4gRXZlbnQgbW9kaWZpZXJzIG1lYW4gYnV0dG9uIG51bWJlciBmb3IgbW91c2UgZXZlbnRzOgotICAgICAgICAgICAgICAgIGV2dC5tb2RpZmllcnMgPSBtb2QgJiAoRXZlbnQuQUxUX01BU0sgfCBFdmVudC5NRVRBX01BU0spOwotICAgICAgICAgICAgICAgIGlmIChpZCA9PSBNb3VzZUV2ZW50Lk1PVVNFX1BSRVNTRUQpIHsKLSAgICAgICAgICAgICAgICAgICAgZXZ0LmNsaWNrQ291bnQgPSBtZS5nZXRDbGlja0NvdW50KCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBldnQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDb252ZXJ0IGtleS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBrZXlDaGFyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGtleSBjaGFyLgotICAgICAgICAgKiBAcGFyYW0ga2V5Q29kZQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBrZXkgY29kZS4KLSAgICAgICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAgICAgKi8KLSAgICAgICAgaW50IGNvbnZlcnRLZXkoY2hhciBrZXlDaGFyLCBpbnQga2V5Q29kZSkgewotICAgICAgICAgICAgaW50IGtleTsKLSAgICAgICAgICAgIC8vIEYxIC0gRjEyCi0gICAgICAgICAgICBpZiAoa2V5Q29kZSA+PSBLZXlFdmVudC5WS19GMSAmJiBrZXlDb2RlIDw9IEtleUV2ZW50LlZLX0YxMikgewotICAgICAgICAgICAgICAgIGtleSA9IEV2ZW50LkYxICsga2V5Q29kZSAtIEtleUV2ZW50LlZLX0YxOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBzd2l0Y2ggKGtleUNvZGUpIHsKLSAgICAgICAgICAgICAgICAgICAgZGVmYXVsdDogLy8gbm9uLWFjdGlvbiBrZXkKLSAgICAgICAgICAgICAgICAgICAgICAgIGtleSA9IGtleUNoYXI7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgLy8gYWN0aW9uIGtleXM6Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgS2V5RXZlbnQuVktfSE9NRToKLSAgICAgICAgICAgICAgICAgICAgICAgIGtleSA9IEV2ZW50LkhPTUU7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSBLZXlFdmVudC5WS19FTkQ6Ci0gICAgICAgICAgICAgICAgICAgICAgICBrZXkgPSBFdmVudC5FTkQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSBLZXlFdmVudC5WS19QQUdFX1VQOgotICAgICAgICAgICAgICAgICAgICAgICAga2V5ID0gRXZlbnQuUEdVUDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlIEtleUV2ZW50LlZLX1BBR0VfRE9XTjoKLSAgICAgICAgICAgICAgICAgICAgICAgIGtleSA9IEV2ZW50LlBHRE47Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSBLZXlFdmVudC5WS19VUDoKLSAgICAgICAgICAgICAgICAgICAgICAgIGtleSA9IEV2ZW50LlVQOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgS2V5RXZlbnQuVktfRE9XTjoKLSAgICAgICAgICAgICAgICAgICAgICAgIGtleSA9IEV2ZW50LkRPV047Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSBLZXlFdmVudC5WS19MRUZUOgotICAgICAgICAgICAgICAgICAgICAgICAga2V5ID0gRXZlbnQuTEVGVDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlIEtleUV2ZW50LlZLX1JJR0hUOgotICAgICAgICAgICAgICAgICAgICAgICAga2V5ID0gRXZlbnQuUklHSFQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSBLZXlFdmVudC5WS19QUklOVFNDUkVFTjoKLSAgICAgICAgICAgICAgICAgICAgICAgIGtleSA9IEV2ZW50LlBSSU5UX1NDUkVFTjsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlIEtleUV2ZW50LlZLX1NDUk9MTF9MT0NLOgotICAgICAgICAgICAgICAgICAgICAgICAga2V5ID0gRXZlbnQuU0NST0xMX0xPQ0s7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSBLZXlFdmVudC5WS19DQVBTX0xPQ0s6Ci0gICAgICAgICAgICAgICAgICAgICAgICBrZXkgPSBFdmVudC5DQVBTX0xPQ0s7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSBLZXlFdmVudC5WS19OVU1fTE9DSzoKLSAgICAgICAgICAgICAgICAgICAgICAgIGtleSA9IEV2ZW50Lk5VTV9MT0NLOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgS2V5RXZlbnQuVktfUEFVU0U6Ci0gICAgICAgICAgICAgICAgICAgICAgICBrZXkgPSBFdmVudC5QQVVTRTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlIEtleUV2ZW50LlZLX0lOU0VSVDoKLSAgICAgICAgICAgICAgICAgICAgICAgIGtleSA9IEV2ZW50LklOU0VSVDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBrZXk7Ci0gICAgICAgIH0KLQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0FXVEV4Y2VwdGlvbi5qYXZhIGIvYXd0L2phdmEvYXd0L0FXVEV4Y2VwdGlvbi5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2NTkwYjczLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9BV1RFeGNlcHRpb24uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotLyoqCi0gKiBUaGUgQVdURXhjZXB0aW9uIGNsYXNzIGlzIHVzZWQgdG8gcHJvdmlkZSBub3RpZmljYXRpb24gYW5kIGluZm9ybWF0aW9uIGFib3V0Ci0gKiBBV1QgZXJyb3JzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEFXVEV4Y2VwdGlvbiBleHRlbmRzIEV4Y2VwdGlvbiB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtMTkwMDQxNDIzMTE1MTMyMzg3OUw7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQVdUIGV4Y2VwdGlvbiB3aXRoIHRoZSBzcGVjaWZpZWQgbWVzc2FnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXNnCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWMgbWVzc2FnZSBmb3IgY3VycmVudCBleGNlcHRpb24uCi0gICAgICovCi0gICAgcHVibGljIEFXVEV4Y2VwdGlvbihTdHJpbmcgbXNnKSB7Ci0gICAgICAgIHN1cGVyKG1zZyk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvQVdUS2V5U3Ryb2tlLmphdmEgYi9hd3QvamF2YS9hd3QvQVdUS2V5U3Ryb2tlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGYwMWY2ZjAuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0FXVEtleVN0cm9rZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNzEyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRG1pdHJ5IEEuIER1cm5ldgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LktleUV2ZW50OwotaW1wb3J0IGphdmEuaW8uT2JqZWN0U3RyZWFtRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uU2VyaWFsaXphYmxlOwotaW1wb3J0IGphdmEubGFuZy5yZWZsZWN0LkNvbnN0cnVjdG9yOwotaW1wb3J0IGphdmEubGFuZy5yZWZsZWN0LkZpZWxkOwotaW1wb3J0IGphdmEudXRpbC5IYXNoTWFwOwotaW1wb3J0IGphdmEudXRpbC5NYXA7Ci1pbXBvcnQgamF2YS51dGlsLk5vU3VjaEVsZW1lbnRFeGNlcHRpb247Ci1pbXBvcnQgamF2YS51dGlsLlN0cmluZ1Rva2VuaXplcjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBBV1RLZXlTdHJva2UgaG9sZHMgYWxsIG9mIHRoZSBpbmZvcm1hdGlvbiBmb3IgdGhlIGNvbXBsZXRlIGFjdCBvZiAKLSAqIHR5cGluZyBhIGNoYXJhY3Rlci4gVGhpcyBpbmNsdWRlcyB0aGUgZXZlbnRzIHRoYXQgYXJlIGdlbmVyYXRlZCB3aGVuIAotICogdGhlIGtleSBpcyBwcmVzc2VkLCByZWxlYXNlZCwgb3IgdHlwZWQgKHByZXNzZWQgYW5kIHJlbGVhc2VkIGdlbmVyYXRpbmcKLSAqIGEgVW5pY29kZSBjaGFyYWN0ZXIgcmVzdWx0KSB3aGljaCBhcmUgYXNzb2NpYXRlZCB3aXRoIHRoZSBldmVudAotICogb2JqZWN0cyBLZXlFdmVudC5LRVlfUFJFU1NFRCwgS2V5RXZlbnQuS0VZX1JFTEVBU0VELCBvciBLZXlFdmVudC5LRVlfVFlQRUQuCi0gKiBJdCBhbHNvIGhvbGRzIGluZm9ybWF0aW9uIGFib3V0IHdoaWNoIG1vZGlmaWVycyAoc3VjaCBhcyBjb250cm9sIG9yIAotICogc2hpZnQpIHdlcmUgdXNlZCBpbiBjb25qdW5jdGlvbiB3aXRoIHRoZSBrZXlzdHJva2UuIFRoZSBmb2xsb3dpbmcgbWFza3MgCi0gKiBhcmUgYXZhaWxhYmxlIHRvIGlkZW50aWZ5IHRoZSBtb2RpZmllcnM6Ci0gKiA8dWw+Ci0gKiA8bGk+amF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5BTFRfR1JBUEhfRE9XTl9NQVNLPC9saT4KLSAqIDxsaT5qYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkFMVF9ET1dOX01BU0s8L2xpPgotICogPGxpPmphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuQ1RSTF9ET1dOX01BU0s8L2xpPgotICogPGxpPmphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuTUVUQV9ET1dOX01BU0s8L2xpPgotICogPGxpPmphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuU0hJRlRfRE9XTl9NQVNLPC9saT4KLSAqIDxsaT5qYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkFMVF9HUkFQSF9NQVNLPC9saT4KLSAqIDxsaT5qYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkFMVF9NQVNLPC9saT4KLSAqIDxsaT5qYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkNUUkxfTUFTSzwvbGk+Ci0gKiA8bGk+amF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5NRVRBX01BU0s8L2xpPiAgCi0gKiA8bGk+amF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5TSElGVF9NQVNLPC9saT4KLSAqIDwvdWw+ICAKLSAqIDxicj4KLSAqICBUaGUgQVdUS2V5U3Ryb2tlIGlzIHVuaXF1ZSwgYW5kIGFwcGxpY2F0aW9ucyBzaG91bGQgbm90IGNyZWF0ZSB0aGVpciBvd24gCi0gKiAgaW5zdGFuY2VzIG9mIEFXVEtleVN0cm9rZS4gQWxsIGFwcGxpY2F0aW9ucyBzaG91bGQgdXNlIGdldEFXVEtleVN0cm9rZSAKLSAqICBtZXRob2RzIGZvciBvYnRhaW5pbmcgaW5zdGFuY2VzIG9mIEFXVEtleVN0cm9rZS4KLSAqICAKLSAqICBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEFXVEtleVN0cm9rZSBpbXBsZW1lbnRzIFNlcmlhbGl6YWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtNjQzMDUzOTY5MTE1NTE2MTg3MUw7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgY2FjaGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgTWFwPEFXVEtleVN0cm9rZSwgQVdUS2V5U3Ryb2tlPiBjYWNoZSA9IG5ldyBIYXNoTWFwPEFXVEtleVN0cm9rZSwgQVdUS2V5U3Ryb2tlPigpOyAvLyBNYXAKLQotICAgIC8vIDwKLSAgICAvLyBBV1RLZXlTdHJva2UKLSAgICAvLyAsCi0gICAgLy8gPwotICAgIC8vIGV4dGVuZHMKLSAgICAvLyBBV1RLZXlTdHJva2UKLSAgICAvLyA+Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQga2V5RXZlbnRUeXBlc01hcC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBNYXA8SW50ZWdlciwgU3RyaW5nPiBrZXlFdmVudFR5cGVzTWFwID0gbmV3IEhhc2hNYXA8SW50ZWdlciwgU3RyaW5nPigpOyAvLyBNYXAKLQotICAgIC8vIDwKLSAgICAvLyBpbnQKLSAgICAvLyAsCi0gICAgLy8gU3RyaW5nCi0gICAgLy8gPgotCi0gICAgcHJpdmF0ZSBzdGF0aWMgQ29uc3RydWN0b3I8Pz4gc3ViQ29uc3RydWN0b3I7Ci0KLSAgICBzdGF0aWMgewotICAgICAgICBrZXlFdmVudFR5cGVzTWFwLnB1dChuZXcgSW50ZWdlcihLZXlFdmVudC5LRVlfUFJFU1NFRCksICJwcmVzc2VkIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAga2V5RXZlbnRUeXBlc01hcC5wdXQobmV3IEludGVnZXIoS2V5RXZlbnQuS0VZX1JFTEVBU0VEKSwgInJlbGVhc2VkIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAga2V5RXZlbnRUeXBlc01hcC5wdXQobmV3IEludGVnZXIoS2V5RXZlbnQuS0VZX1RZUEVEKSwgInR5cGVkIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUga2V5IGNoYXIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBjaGFyIGtleUNoYXI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUga2V5IGNvZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQga2V5Q29kZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBtb2RpZmllcnMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgbW9kaWZpZXJzOwotCi0gICAgLyoqCi0gICAgICogVGhlIG9uIGtleSByZWxlYXNlLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBvbktleVJlbGVhc2U7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQVdUS2V5U3Ryb2tlLiBnZXRBV1RLZXlTdHJva2UgbWV0aG9kIHNob3VsZCBiZSB1c2VkIGJ5Ci0gICAgICogYXBwbGljYXRpb25zIGNvZGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGtleUNoYXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBrZXkgY2hhci4KLSAgICAgKiBAcGFyYW0ga2V5Q29kZQotICAgICAqICAgICAgICAgICAgdGhlIGtleSBjb2RlLgotICAgICAqIEBwYXJhbSBtb2RpZmllcnMKLSAgICAgKiAgICAgICAgICAgIHRoZSBtb2RpZmllcnMuCi0gICAgICogQHBhcmFtIG9uS2V5UmVsZWFzZQotICAgICAqICAgICAgICAgICAgdHJ1ZSBpZiBBV1RLZXlTdHJva2UgaXMgZm9yIGEga2V5IHJlbGVhc2UsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgQVdUS2V5U3Ryb2tlKGNoYXIga2V5Q2hhciwgaW50IGtleUNvZGUsIGludCBtb2RpZmllcnMsIGJvb2xlYW4gb25LZXlSZWxlYXNlKSB7Ci0gICAgICAgIHNldEFXVEtleVN0cm9rZShrZXlDaGFyLCBrZXlDb2RlLCBtb2RpZmllcnMsIG9uS2V5UmVsZWFzZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgQVdUIGtleSBzdHJva2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGtleUNoYXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBrZXkgY2hhci4KLSAgICAgKiBAcGFyYW0ga2V5Q29kZQotICAgICAqICAgICAgICAgICAgdGhlIGtleSBjb2RlLgotICAgICAqIEBwYXJhbSBtb2RpZmllcnMKLSAgICAgKiAgICAgICAgICAgIHRoZSBtb2RpZmllcnMuCi0gICAgICogQHBhcmFtIG9uS2V5UmVsZWFzZQotICAgICAqICAgICAgICAgICAgdGhlIG9uIGtleSByZWxlYXNlLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBzZXRBV1RLZXlTdHJva2UoY2hhciBrZXlDaGFyLCBpbnQga2V5Q29kZSwgaW50IG1vZGlmaWVycywgYm9vbGVhbiBvbktleVJlbGVhc2UpIHsKLSAgICAgICAgdGhpcy5rZXlDaGFyID0ga2V5Q2hhcjsKLSAgICAgICAgdGhpcy5rZXlDb2RlID0ga2V5Q29kZTsKLSAgICAgICAgdGhpcy5tb2RpZmllcnMgPSBtb2RpZmllcnM7Ci0gICAgICAgIHRoaXMub25LZXlSZWxlYXNlID0gb25LZXlSZWxlYXNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBBV1RLZXlTdHJva2Ugd2l0aCBkZWZhdWx0IHBhcmFtZXRlcnM6Ci0gICAgICogS2V5RXZlbnQuQ0hBUl9VTkRFRklORUQga2V5IGNoYXIsIEtleUV2ZW50LlZLX1VOREVGSU5FRCBrZXkgY29kZSwgd2l0aG91dAotICAgICAqIG1vZGlmaWVycyBhbmQgZmFsc2Uga2V5IHJlYWxpemVkIHZhbHVlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBBV1RLZXlTdHJva2UoKSB7Ci0gICAgICAgIHRoaXMoS2V5RXZlbnQuQ0hBUl9VTkRFRklORUQsIEtleUV2ZW50LlZLX1VOREVGSU5FRCwgMCwgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIHVuaXF1ZSBudW1iZXIgdmFsdWUgZm9yIEFXVEtleVN0cm9rZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaW50ZWdlciB1bmlxdWUgdmFsdWUgb2YgdGhlIEFXVEtleVN0cm9rZSBvYmplY3QuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgcmV0dXJuIG1vZGlmaWVycyArIChrZXlDb2RlICE9IEtleUV2ZW50LlZLX1VOREVGSU5FRCA/IGtleUNvZGUgOiBrZXlDaGFyKQotICAgICAgICAgICAgICAgICsgKG9uS2V5UmVsZWFzZSA/IC0xIDogMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2V0IG9mIG1vZGlmaWVycyBmb3IgdGhlIEFXVEtleVN0cm9rZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaW50ZWdlciB2YWx1ZSB3aGljaCBjb250YWlucyBtb2RpZmllcnMuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGludCBnZXRNb2RpZmllcnMoKSB7Ci0gICAgICAgIHJldHVybiBtb2RpZmllcnM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgdGhpcyBBV1RLZXlTdHJva2Ugb2JqZWN0IHRvIHRoZSBzcGVjaWZpZWQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBhbk9iamVjdAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBBV1RLZXlTdHJva2Ugb2JqZWN0IHRvIGNvbXBhcmUgd2l0aCB0aGlzCi0gICAgICogICAgICAgICAgICBpbnN0YW5jZS4KLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgb2JqZWN0cyBhcmUgaWRlbnRpY2FsLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZpbmFsIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBhbk9iamVjdCkgewotICAgICAgICBpZiAoYW5PYmplY3QgaW5zdGFuY2VvZiBBV1RLZXlTdHJva2UpIHsKLSAgICAgICAgICAgIEFXVEtleVN0cm9rZSBrZXkgPSAoQVdUS2V5U3Ryb2tlKWFuT2JqZWN0OwotICAgICAgICAgICAgcmV0dXJuICgoa2V5LmtleUNvZGUgPT0ga2V5Q29kZSkgJiYgKGtleS5rZXlDaGFyID09IGtleUNoYXIpCi0gICAgICAgICAgICAgICAgICAgICYmIChrZXkubW9kaWZpZXJzID09IG1vZGlmaWVycykgJiYgKGtleS5vbktleVJlbGVhc2UgPT0gb25LZXlSZWxlYXNlKSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgQVdUS2V5U3Ryb2tlLiBUaGlzIHN0cmluZyBzaG91bGQKLSAgICAgKiBjb250YWluIGtleSBzdHJva2UgcHJvcGVydGllcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIEFXVEtleVN0cm9rZS4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICBpbnQgdHlwZSA9IGdldEtleUV2ZW50VHlwZSgpOwotICAgICAgICByZXR1cm4gSW5wdXRFdmVudC5nZXRNb2RpZmllcnNFeFRleHQoZ2V0TW9kaWZpZXJzKCkpICsgIiAiICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgIGtleUV2ZW50VHlwZXNNYXAuZ2V0KG5ldyBJbnRlZ2VyKHR5cGUpKSArICIgIiArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAodHlwZSA9PSBLZXlFdmVudC5LRVlfVFlQRUQgPyBuZXcgU3RyaW5nKG5ldyBjaGFyW10gewotICAgICAgICAgICAgICAgICAgICBrZXlDaGFyCi0gICAgICAgICAgICAgICAgfSkgOiBLZXlFdmVudC5nZXRLZXlUZXh0KGtleUNvZGUpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBrZXkgY29kZSBmb3IgdGhlIEFXVEtleVN0cm9rZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUga2V5IGNvZGUgZm9yIHRoZSBBV1RLZXlTdHJva2Ugb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0S2V5Q29kZSgpIHsKLSAgICAgICAgcmV0dXJuIGtleUNvZGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUga2V5IGNoYXJhY3RlciBmb3IgdGhlIEFXVEtleVN0cm9rZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUga2V5IGNoYXJhY3RlciBmb3IgdGhlIEFXVEtleVN0cm9rZSBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGNoYXIgZ2V0S2V5Q2hhcigpIHsKLSAgICAgICAgcmV0dXJuIGtleUNoYXI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQVdUIGtleSBzdHJva2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGtleUNoYXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBrZXkgY2hhci4KLSAgICAgKiBAcGFyYW0ga2V5Q29kZQotICAgICAqICAgICAgICAgICAgdGhlIGtleSBjb2RlLgotICAgICAqIEBwYXJhbSBtb2RpZmllcnMKLSAgICAgKiAgICAgICAgICAgIHRoZSBtb2RpZmllcnMuCi0gICAgICogQHBhcmFtIG9uS2V5UmVsZWFzZQotICAgICAqICAgICAgICAgICAgdGhlIG9uIGtleSByZWxlYXNlLgotICAgICAqIEByZXR1cm4gdGhlIEFXVCBrZXkgc3Ryb2tlLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIEFXVEtleVN0cm9rZSBnZXRBV1RLZXlTdHJva2UoY2hhciBrZXlDaGFyLCBpbnQga2V5Q29kZSwgaW50IG1vZGlmaWVycywKLSAgICAgICAgICAgIGJvb2xlYW4gb25LZXlSZWxlYXNlKSB7Ci0gICAgICAgIEFXVEtleVN0cm9rZSBrZXkgPSBuZXdJbnN0YW5jZShrZXlDaGFyLCBrZXlDb2RlLCBtb2RpZmllcnMsIG9uS2V5UmVsZWFzZSk7Ci0KLSAgICAgICAgQVdUS2V5U3Ryb2tlIHZhbHVlID0gY2FjaGUuZ2V0KGtleSk7Ci0gICAgICAgIGlmICh2YWx1ZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB2YWx1ZSA9IGtleTsKLSAgICAgICAgICAgIGNhY2hlLnB1dChrZXksIHZhbHVlKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdmFsdWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogTmV3IGluc3RhbmNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBrZXlDaGFyCi0gICAgICogICAgICAgICAgICB0aGUga2V5IGNoYXIuCi0gICAgICogQHBhcmFtIGtleUNvZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBrZXkgY29kZS4KLSAgICAgKiBAcGFyYW0gbW9kaWZpZXJzCi0gICAgICogICAgICAgICAgICB0aGUgbW9kaWZpZXJzLgotICAgICAqIEBwYXJhbSBvbktleVJlbGVhc2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBvbiBrZXkgcmVsZWFzZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBBV1Qga2V5IHN0cm9rZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBBV1RLZXlTdHJva2UgbmV3SW5zdGFuY2UoY2hhciBrZXlDaGFyLCBpbnQga2V5Q29kZSwgaW50IG1vZGlmaWVycywKLSAgICAgICAgICAgIGJvb2xlYW4gb25LZXlSZWxlYXNlKSB7Ci0gICAgICAgIEFXVEtleVN0cm9rZSBrZXk7Ci0gICAgICAgIC8vID8/P0FXVAotICAgICAgICAvLyBpZiAoc3ViQ29uc3RydWN0b3IgPT0gbnVsbCkgewotICAgICAgICBrZXkgPSBuZXcgQVdUS2V5U3Ryb2tlKCk7Ci0gICAgICAgIC8vID8/P0FXVAotICAgICAgICAvLyB9IGVsc2UgewotICAgICAgICAvLyB0cnkgewotICAgICAgICAvLyBrZXkgPSAoQVdUS2V5U3Ryb2tlKSBzdWJDb25zdHJ1Y3Rvci5uZXdJbnN0YW5jZSgpOwotICAgICAgICAvLyB9IGNhdGNoIChFeGNlcHRpb24gZSkgewotICAgICAgICAvLyB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbihlKTsKLSAgICAgICAgLy8gfQotICAgICAgICAvLyB9Ci0gICAgICAgIGludCBhbGxNb2RpZmllcnMgPSBnZXRBbGxNb2RpZmllcnMobW9kaWZpZXJzKTsKLSAgICAgICAga2V5LnNldEFXVEtleVN0cm9rZShrZXlDaGFyLCBrZXlDb2RlLCBhbGxNb2RpZmllcnMsIG9uS2V5UmVsZWFzZSk7Ci0gICAgICAgIHJldHVybiBrZXk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgbWFzay4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbW9kCi0gICAgICogICAgICAgICAgICB0aGUgbW9kLgotICAgICAqIEBwYXJhbSBtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgbWFzay4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgaW50IGFkZE1hc2soaW50IG1vZCwgaW50IG1hc2spIHsKLSAgICAgICAgcmV0dXJuICgobW9kICYgbWFzaykgIT0gMCkgPyAobW9kIHwgbWFzaykgOiBtb2Q7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJuIGFsbCAob2xkICYgbmV3KSBtb2RpZmllcnMgY29ycmVzcG9uZGluZyB0by4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbW9kCi0gICAgICogICAgICAgICAgICBvbGQgb3IgbmV3IG1vZGlmaWVycy4KLSAgICAgKiBAcmV0dXJuIG9sZCBhbmQgbmV3IG1vZGlmaWVycyB0b2dldGhlci4KLSAgICAgKi8KLSAgICBzdGF0aWMgaW50IGdldEFsbE1vZGlmaWVycyhpbnQgbW9kKSB7Ci0gICAgICAgIGludCBhbGxNb2QgPSBtb2Q7Ci0gICAgICAgIGludCBzaGlmdCA9IChJbnB1dEV2ZW50LlNISUZUX01BU0sgfCBJbnB1dEV2ZW50LlNISUZUX0RPV05fTUFTSyk7Ci0gICAgICAgIGludCBjdHJsID0gKElucHV0RXZlbnQuQ1RSTF9NQVNLIHwgSW5wdXRFdmVudC5DVFJMX0RPV05fTUFTSyk7Ci0gICAgICAgIGludCBtZXRhID0gKElucHV0RXZlbnQuTUVUQV9NQVNLIHwgSW5wdXRFdmVudC5NRVRBX0RPV05fTUFTSyk7Ci0gICAgICAgIGludCBhbHQgPSAoSW5wdXRFdmVudC5BTFRfTUFTSyB8IElucHV0RXZlbnQuQUxUX0RPV05fTUFTSyk7Ci0gICAgICAgIGludCBhbHRHciA9IChJbnB1dEV2ZW50LkFMVF9HUkFQSF9NQVNLIHwgSW5wdXRFdmVudC5BTFRfR1JBUEhfRE9XTl9NQVNLKTsKLSAgICAgICAgLy8gYnV0dG9uIG1vZGlmaWVycyBhcmUgbm90IGNvbnZlcnRlZCBiZXR3ZWVuIG9sZCAmIG5ldwotCi0gICAgICAgIGFsbE1vZCA9IGFkZE1hc2soYWxsTW9kLCBzaGlmdCk7Ci0gICAgICAgIGFsbE1vZCA9IGFkZE1hc2soYWxsTW9kLCBjdHJsKTsKLSAgICAgICAgYWxsTW9kID0gYWRkTWFzayhhbGxNb2QsIG1ldGEpOwotICAgICAgICBhbGxNb2QgPSBhZGRNYXNrKGFsbE1vZCwgYWx0KTsKLSAgICAgICAgYWxsTW9kID0gYWRkTWFzayhhbGxNb2QsIGFsdEdyKTsKLQotICAgICAgICByZXR1cm4gYWxsTW9kOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gaW5zdGFuY2Ugb2YgQVdUS2V5U3Ryb2tlIGZvciBwYXJzZWQgc3RyaW5nLiBUaGUgc3RyaW5nIG11c3QKLSAgICAgKiBoYXZlIHRoZSBmb2xsb3dpbmcgc3ludGF4OgotICAgICAqPHA+Ci0gICAgICogJmx0O21vZGlmaWVycyZndDsqICgmbHQ7dHlwZWRJRCZndDsgfCAmbHQ7cHJlc3NlZFJlbGVhc2VkSUQmZ3Q7KQotICAgICAqPHA+Ci0gICAgICogbW9kaWZpZXJzIDo9IHNoaWZ0IHwgY29udHJvbCB8IGN0cmwgfCBtZXRhIHwgYWx0IHwgYWx0R3JhcGggPGJyPgotICAgICAqIHR5cGVkSUQgOj0gdHlwZWQgPHR5cGVkS2V5PiA8YnI+Ci0gICAgICogdHlwZWRLZXkgOj0gc3RyaW5nIG9mIGxlbmd0aCAxIGdpdmluZyB0aGUgVW5pY29kZSBjaGFyYWN0ZXIuIDxicj4KLSAgICAgKiBwcmVzc2VkUmVsZWFzZWRJRCA6PSAocHJlc3NlZCB8IHJlbGVhc2VkKSA8a2V5PiA8YnI+Ci0gICAgICoga2V5IDo9IEtleUV2ZW50IGtleSBjb2RlIG5hbWUsIGkuZS4gdGhlIG5hbWUgZm9sbG93aW5nICJWS18iLgotICAgICAqIDxwPgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgU3RyaW5nIHdoaWNoIGNvbnRhaW5zIGtleSBzdHJva2UgcGFyYW1ldGVycy4KLSAgICAgKiBAcmV0dXJuIHRoZSBBV1RLZXlTdHJva2UgZm9yIHN0cmluZy4KLSAgICAgKiBAdGhyb3dzIElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHN0cmluZyBoYXMgaW5jb3JyZWN0IGZvcm1hdCBvciBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgQVdUS2V5U3Ryb2tlIGdldEFXVEtleVN0cm9rZShTdHJpbmcgcykgewotICAgICAgICBpZiAocyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjU9bnVsbCBhcmd1bWVudAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42NSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgU3RyaW5nVG9rZW5pemVyIHRva2VuaXplciA9IG5ldyBTdHJpbmdUb2tlbml6ZXIocyk7Ci0KLSAgICAgICAgQm9vbGVhbiByZWxlYXNlID0gbnVsbDsKLSAgICAgICAgaW50IG1vZGlmaWVycyA9IDA7Ci0gICAgICAgIGludCBrZXlDb2RlID0gS2V5RXZlbnQuVktfVU5ERUZJTkVEOwotICAgICAgICBjaGFyIGtleUNoYXIgPSBLZXlFdmVudC5DSEFSX1VOREVGSU5FRDsKLSAgICAgICAgYm9vbGVhbiB0eXBlZCA9IGZhbHNlOwotICAgICAgICBsb25nIG1vZGlmaWVyID0gMDsKLSAgICAgICAgU3RyaW5nIHRva2VuID0gbnVsbDsKLSAgICAgICAgZG8gewotICAgICAgICAgICAgdG9rZW4gPSBnZXROZXh0VG9rZW4odG9rZW5pemVyKTsKLSAgICAgICAgICAgIG1vZGlmaWVyID0gcGFyc2VNb2RpZmllcih0b2tlbik7Ci0gICAgICAgICAgICBtb2RpZmllcnMgfD0gbW9kaWZpZXI7Ci0gICAgICAgIH0gd2hpbGUgKG1vZGlmaWVyID4gMCk7Ci0KLSAgICAgICAgdHlwZWQgPSBwYXJzZVR5cGVkSUQodG9rZW4pOwotCi0gICAgICAgIGlmICh0eXBlZCkgewotICAgICAgICAgICAgdG9rZW4gPSBnZXROZXh0VG9rZW4odG9rZW5pemVyKTsKLSAgICAgICAgICAgIGtleUNoYXIgPSBwYXJzZVR5cGVkS2V5KHRva2VuKTsKLQotICAgICAgICB9Ci0gICAgICAgIGlmIChrZXlDaGFyID09IEtleUV2ZW50LkNIQVJfVU5ERUZJTkVEKSB7Ci0gICAgICAgICAgICByZWxlYXNlID0gcGFyc2VQcmVzc2VkUmVsZWFzZWRJRCh0b2tlbik7Ci0gICAgICAgICAgICBpZiAocmVsZWFzZSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdG9rZW4gPSBnZXROZXh0VG9rZW4odG9rZW5pemVyKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGtleUNvZGUgPSBwYXJzZUtleSh0b2tlbik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHRva2VuaXplci5oYXNNb3JlVG9rZW5zKCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Nj1JbnZhbGlkIGZvcm1hdAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42NiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGdldEFXVEtleVN0cm9rZShrZXlDaGFyLCBrZXlDb2RlLCBtb2RpZmllcnMsIHJlbGVhc2UgPT0gQm9vbGVhbi5UUlVFKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBuZXh0IHRva2VuLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0b2tlbml6ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSB0b2tlbml6ZXIuCi0gICAgICogQHJldHVybiB0aGUgbmV4dCB0b2tlbi4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBTdHJpbmcgZ2V0TmV4dFRva2VuKFN0cmluZ1Rva2VuaXplciB0b2tlbml6ZXIpIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiB0b2tlbml6ZXIubmV4dFRva2VuKCk7Ci0gICAgICAgIH0gY2F0Y2ggKE5vU3VjaEVsZW1lbnRFeGNlcHRpb24gZXhjZXB0aW9uKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjY9SW52YWxpZCBmb3JtYXQKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGtleSBjb2RlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgcy4KLSAgICAgKiBAcmV0dXJuIHRoZSBrZXkgY29kZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgaW50IGdldEtleUNvZGUoU3RyaW5nIHMpIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIEZpZWxkIHZrID0gS2V5RXZlbnQuY2xhc3MuZ2V0RmllbGQoIlZLXyIgKyBzKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgcmV0dXJuIHZrLmdldEludChudWxsKTsKLSAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIGlmIChzLmxlbmd0aCgpICE9IDEpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuNjY9SW52YWxpZCBmb3JtYXQKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjY2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gS2V5RXZlbnQuVktfVU5ERUZJTkVEOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBpbnN0YW5jZSBvZiB0aGUgQVdUS2V5U3Ryb2tlIGZvciBzcGVjaWZpZWQgY2hhcmFjdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBrZXlDaGFyCi0gICAgICogICAgICAgICAgICB0aGUga2V5Ym9hcmQgY2hhcmFjdGVyIHZhbHVlLgotICAgICAqIEByZXR1cm4gYSBBV1RLZXlTdHJva2UgZm9yIHNwZWNpZmllZCBjaGFyYWN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBBV1RLZXlTdHJva2UgZ2V0QVdUS2V5U3Ryb2tlKGNoYXIga2V5Q2hhcikgewotICAgICAgICByZXR1cm4gZ2V0QVdUS2V5U3Ryb2tlKGtleUNoYXIsIEtleUV2ZW50LlZLX1VOREVGSU5FRCwgMCwgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gaW5zdGFuY2Ugb2YgQVdUS2V5U3Ryb2tlIGZvciBhIGdpdmVuIGtleSBjb2RlLCBzZXQgb2YKLSAgICAgKiBtb2RpZmllcnMsIGFuZCBzcGVjaWZpZWQga2V5IHJlbGVhc2VkIGZsYWcgdmFsdWUuIFRoZSBrZXkgY29kZXMgYXJlCi0gICAgICogZGVmaW5lZCBpbiBqYXZhLmF3dC5ldmVudC5LZXlFdmVudCBjbGFzcy4gVGhlIHNldCBvZiBtb2RpZmllcnMgaXMgZ2l2ZW4KLSAgICAgKiBhcyBhIGJpdHdpc2UgY29tYmluYXRpb24gb2YgbWFza3MgdGFrZW4gZnJvbSB0aGUgZm9sbG93aW5nIGxpc3Q6Ci0gICAgICogPHVsPgotICAgICAqIDxsaT5qYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkFMVF9HUkFQSF9ET1dOX01BU0s8L2xpPiA8bGk+Ci0gICAgICogamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5BTFRfRE9XTl9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuQ1RSTF9ET1dOX01BU0s8L2xpPiA8bGk+Ci0gICAgICogamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5NRVRBX0RPV05fTUFTSzwvbGk+IDxsaT4KLSAgICAgKiBqYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LlNISUZUX0RPV05fTUFTSzwvbGk+IDxsaT4KLSAgICAgKiBqYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkFMVF9HUkFQSF9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuQUxUX01BU0s8L2xpPiA8bGk+Ci0gICAgICogamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5DVFJMX01BU0s8L2xpPiA8bGk+Ci0gICAgICogamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5NRVRBX01BU0s8L2xpPiA8bGk+Ci0gICAgICogamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5TSElGVF9NQVNLPC9saT4KLSAgICAgKiA8L3VsPgotICAgICAqIDxicj4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0ga2V5Q29kZQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBrZXkgY29kZSBvZiBrZXlib2FyZC4KLSAgICAgKiBAcGFyYW0gbW9kaWZpZXJzCi0gICAgICogICAgICAgICAgICB0aGUgYml0IHNldCBvZiBtb2RpZmllcnMuCi0gICAgICogQHBhcmFtIG9uS2V5UmVsZWFzZQotICAgICAqICAgICAgICAgICAgdGhlIHZhbHVlIHdoaWNoIHJlcHJlc2VudHMgd2hldGhlciB0aGlzIEFXVEtleVN0cm9rZSBzaGFsbAotICAgICAqICAgICAgICAgICAgcmVwcmVzZW50cyBhIGtleSByZWxlYXNlLgotICAgICAqIEByZXR1cm4gdGhlIEFXVEtleVN0cm9rZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEFXVEtleVN0cm9rZSBnZXRBV1RLZXlTdHJva2UoaW50IGtleUNvZGUsIGludCBtb2RpZmllcnMsIGJvb2xlYW4gb25LZXlSZWxlYXNlKSB7Ci0gICAgICAgIHJldHVybiBnZXRBV1RLZXlTdHJva2UoS2V5RXZlbnQuQ0hBUl9VTkRFRklORUQsIGtleUNvZGUsIG1vZGlmaWVycywgb25LZXlSZWxlYXNlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIEFXVEtleVN0cm9rZSBmb3IgYSBzcGVjaWZpZWQgY2hhcmFjdGVyIGFuZCBzZXQgb2YgbW9kaWZpZXJzLiBUaGUKLSAgICAgKiBzZXQgb2YgbW9kaWZpZXJzIGlzIGdpdmVuIGFzIGEgYml0d2lzZSBjb21iaW5hdGlvbiBvZiBtYXNrcyB0YWtlbiBmcm9tCi0gICAgICogdGhlIGZvbGxvd2luZyBsaXN0OgotICAgICAqIDx1bD4KLSAgICAgKiA8bGk+amF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5BTFRfR1JBUEhfRE9XTl9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuQUxUX0RPV05fTUFTSzwvbGk+IDxsaT4KLSAgICAgKiBqYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkNUUkxfRE9XTl9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuTUVUQV9ET1dOX01BU0s8L2xpPiA8bGk+Ci0gICAgICogamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5TSElGVF9ET1dOX01BU0s8L2xpPiA8bGk+Ci0gICAgICogamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5BTFRfR1JBUEhfTUFTSzwvbGk+IDxsaT4KLSAgICAgKiBqYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkFMVF9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuQ1RSTF9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuTUVUQV9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuU0hJRlRfTUFTSzwvbGk+Ci0gICAgICogPC91bD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0ga2V5Q2hhcgotICAgICAqICAgICAgICAgICAgdGhlIENoYXJhY3RlciBvYmplY3Qgd2hpY2ggcmVwcmVzZW50cyBrZXlib2FyZCBjaGFyYWN0ZXIKLSAgICAgKiAgICAgICAgICAgIHZhbHVlLgotICAgICAqIEBwYXJhbSBtb2RpZmllcnMKLSAgICAgKiAgICAgICAgICAgIHRoZSBiaXQgc2V0IG9mIG1vZGlmaWVycy4KLSAgICAgKiBAcmV0dXJuIHRoZSBBV1RLZXlTdHJva2Ugb2JqZWN0LgotICAgICAqIEB0aHJvd3MgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYga2V5Q2hhciB2YWx1ZSBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgQVdUS2V5U3Ryb2tlIGdldEFXVEtleVN0cm9rZShDaGFyYWN0ZXIga2V5Q2hhciwgaW50IG1vZGlmaWVycykgewotICAgICAgICBpZiAoa2V5Q2hhciA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMDE9J3swfScgcGFyYW1ldGVyIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMDEiLCAia2V5Q2hhciIpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGdldEFXVEtleVN0cm9rZShrZXlDaGFyLmNoYXJWYWx1ZSgpLCBLZXlFdmVudC5WS19VTkRFRklORUQsIG1vZGlmaWVycywgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gaW5zdGFuY2Ugb2YgQVdUS2V5U3Ryb2tlIGZvciBhIHNwZWNpZmllZCBrZXkgY29kZSBhbmQgc2V0IG9mCi0gICAgICogbW9kaWZpZXJzLiBUaGUga2V5IGNvZGVzIGFyZSBkZWZpbmVkIGluIGphdmEuYXd0LmV2ZW50LktleUV2ZW50IGNsYXNzLgotICAgICAqIFRoZSBzZXQgb2YgbW9kaWZpZXJzIGlzIGdpdmVuIGFzIGEgYml0d2lzZSBjb21iaW5hdGlvbiBvZiBtYXNrcyB0YWtlbgotICAgICAqIGZyb20gdGhlIGZvbGxvd2luZyBsaXN0OgotICAgICAqIDx1bD4KLSAgICAgKiA8bGk+amF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5BTFRfR1JBUEhfRE9XTl9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuQUxUX0RPV05fTUFTSzwvbGk+IDxsaT4KLSAgICAgKiBqYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkNUUkxfRE9XTl9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuTUVUQV9ET1dOX01BU0s8L2xpPiA8bGk+Ci0gICAgICogamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5TSElGVF9ET1dOX01BU0s8L2xpPiA8bGk+Ci0gICAgICogamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudC5BTFRfR1JBUEhfTUFTSzwvbGk+IDxsaT4KLSAgICAgKiBqYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50LkFMVF9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuQ1RSTF9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuTUVUQV9NQVNLPC9saT4gPGxpPgotICAgICAqIGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQuU0hJRlRfTUFTSzwvbGk+Ci0gICAgICogPC91bD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0ga2V5Q29kZQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBrZXkgY29kZSBvZiBrZXlib2FyZC4KLSAgICAgKiBAcGFyYW0gbW9kaWZpZXJzCi0gICAgICogICAgICAgICAgICB0aGUgYml0IHNldCBvZiBtb2RpZmllcnMuCi0gICAgICogQHJldHVybiB0aGUgQVdUS2V5U3Ryb2tlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgQVdUS2V5U3Ryb2tlIGdldEFXVEtleVN0cm9rZShpbnQga2V5Q29kZSwgaW50IG1vZGlmaWVycykgewotICAgICAgICByZXR1cm4gZ2V0QVdUS2V5U3Ryb2tlKGtleUNvZGUsIG1vZGlmaWVycywgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEFXVEtleVN0cm9rZSBmb3IgYSBrZXkgZXZlbnQuIFRoaXMgbWV0aG9kIG9idGFpbnMgdGhlIGtleSBjaGFyCi0gICAgICogYW5kIGtleSBjb2RlIGZyb20gdGhlIHNwZWNpZmllZCBrZXkgZXZlbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGFuRXZlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBrZXkgZXZlbnQgd2hpY2ggaWRlbnRpZmllcyB0aGUgZGVzaXJlZCBBV1RLZXlTdHJva2UuCi0gICAgICogQHJldHVybiB0aGUgQVdUS2V5U3Ryb2tlIGZvciB0aGUga2V5IGV2ZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgQVdUS2V5U3Ryb2tlIGdldEFXVEtleVN0cm9rZUZvckV2ZW50KEtleUV2ZW50IGFuRXZlbnQpIHsKLSAgICAgICAgaW50IGlkID0gYW5FdmVudC5nZXRJRCgpOwotICAgICAgICBjaGFyIHVuZGVmID0gS2V5RXZlbnQuQ0hBUl9VTkRFRklORUQ7Ci0gICAgICAgIGNoYXIga2V5Q2hhciA9IChpZCA9PSBLZXlFdmVudC5LRVlfVFlQRUQgPyBhbkV2ZW50LmdldEtleUNoYXIoKSA6IHVuZGVmKTsKLSAgICAgICAgaW50IGtleUNvZGUgPSAoa2V5Q2hhciA9PSB1bmRlZiA/IGFuRXZlbnQuZ2V0S2V5Q29kZSgpIDogS2V5RXZlbnQuVktfVU5ERUZJTkVEKTsKLSAgICAgICAgcmV0dXJuIGdldEFXVEtleVN0cm9rZShrZXlDaGFyLCBrZXlDb2RlLCBhbkV2ZW50LmdldE1vZGlmaWVyc0V4KCksCi0gICAgICAgICAgICAgICAgaWQgPT0gS2V5RXZlbnQuS0VZX1JFTEVBU0VEKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBrZXkgZXZlbnQgdHlwZSBmb3IgdGhlIEFXVEtleVN0cm9rZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUga2V5IGV2ZW50IHR5cGU6IEtleUV2ZW50LktFWV9QUkVTU0VELCBLZXlFdmVudC5LRVlfVFlQRUQsIG9yCi0gICAgICogICAgICAgICBLZXlFdmVudC5LRVlfUkVMRUFTRUQuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGludCBnZXRLZXlFdmVudFR5cGUoKSB7Ci0gICAgICAgIGlmIChrZXlDb2RlID09IEtleUV2ZW50LlZLX1VOREVGSU5FRCkgewotICAgICAgICAgICAgcmV0dXJuIEtleUV2ZW50LktFWV9UWVBFRDsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gKG9uS2V5UmVsZWFzZSA/IEtleUV2ZW50LktFWV9SRUxFQVNFRCA6IEtleUV2ZW50LktFWV9QUkVTU0VEKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIGtleSBldmVudCBpcyBhc3NvY2lhdGVkIHdpdGggdGhlIEFXVEtleVN0cm9rZSBpcwotICAgICAqIEtFWV9SRUxFQVNFRCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgaWYgdGhlIGtleSBldmVudCBhc3NvY2lhdGVkIHdpdGggdGhlIEFXVEtleVN0cm9rZSBpcwotICAgICAqICAgICAgICAgS0VZX1JFTEVBU0VELCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGJvb2xlYW4gaXNPbktleVJlbGVhc2UoKSB7Ci0gICAgICAgIHJldHVybiBvbktleVJlbGVhc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVhZCByZXNvbHZlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG9iamVjdC4KLSAgICAgKiBAdGhyb3dzIE9iamVjdFN0cmVhbUV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIHRoZSBvYmplY3Qgc3RyZWFtIGV4Y2VwdGlvbi4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgT2JqZWN0IHJlYWRSZXNvbHZlKCkgdGhyb3dzIE9iamVjdFN0cmVhbUV4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBnZXRBV1RLZXlTdHJva2UodGhpcy5rZXlDaGFyLCB0aGlzLmtleUNvZGUsIHRoaXMubW9kaWZpZXJzLCB0aGlzLm9uS2V5UmVsZWFzZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVnaXN0ZXIgc3ViY2xhc3MuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN1YmNsYXNzCi0gICAgICogICAgICAgICAgICB0aGUgc3ViY2xhc3MuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHN0YXRpYyB2b2lkIHJlZ2lzdGVyU3ViY2xhc3MoQ2xhc3M8Pz4gc3ViY2xhc3MpIHsKLSAgICAgICAgLy8gPz8/QVdUCi0gICAgICAgIC8qCi0gICAgICAgICAqIGlmIChzdWJjbGFzcyA9PSBudWxsKSB7IC8vIGF3dC4wMT0nezB9JyBwYXJhbWV0ZXIgaXMgbnVsbCB0aHJvdyBuZXcKLSAgICAgICAgICogSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjAxIiwgInN1YmNsYXNzIikpOwotICAgICAgICAgKiAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgfSBpZiAoIQotICAgICAgICAgKiBBV1RLZXlTdHJva2UuY2xhc3MuaXNBc3NpZ25hYmxlRnJvbShzdWJjbGFzcykpIHsgLy8gYXd0LjY3PXN1YmNsYXNzCi0gICAgICAgICAqIGlzIG5vdCBkZXJpdmVkIGZyb20gQVdUS2V5U3Ryb2tlIHRocm93IG5ldwotICAgICAgICAgKiBDbGFzc0Nhc3RFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjciKSk7IC8vJE5PTi1OTFMtMSQgfSB0cnkKLSAgICAgICAgICogeyBzdWJDb25zdHJ1Y3RvciA9IHN1YmNsYXNzLmdldERlY2xhcmVkQ29uc3RydWN0b3IoKTsKLSAgICAgICAgICogc3ViQ29uc3RydWN0b3Iuc2V0QWNjZXNzaWJsZSh0cnVlKTsgfSBjYXRjaCAoU2VjdXJpdHlFeGNlcHRpb24gZSkgewotICAgICAgICAgKiB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbihlKTsgfSBjYXRjaCAoTm9TdWNoTWV0aG9kRXhjZXB0aW9uIGUpIHsgLy8KLSAgICAgICAgICogYXd0LjY4PXN1YmNsYXNzIGNvdWxkIG5vdCBiZSBpbnN0YW50aWF0ZWQgdGhyb3cgbmV3Ci0gICAgICAgICAqIElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42OCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgKiB9IGNhY2hlLmNsZWFyKCk7IC8vZmx1c2ggdGhlIGNhY2hlCi0gICAgICAgICAqLwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBhcnNlcyB0aGUgbW9kaWZpZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0ck1vZAotICAgICAqICAgICAgICAgICAgdGhlIHN0ciBtb2QuCi0gICAgICogQHJldHVybiB0aGUgbG9uZy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBsb25nIHBhcnNlTW9kaWZpZXIoU3RyaW5nIHN0ck1vZCkgewotICAgICAgICBsb25nIG1vZGlmaWVycyA9IDBsOwotICAgICAgICBpZiAoc3RyTW9kLmVxdWFscygic2hpZnQiKSkgeyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBtb2RpZmllcnMgfD0gSW5wdXRFdmVudC5TSElGVF9ET1dOX01BU0s7Ci0gICAgICAgIH0gZWxzZSBpZiAoc3RyTW9kLmVxdWFscygiY29udHJvbCIpIHx8IHN0ck1vZC5lcXVhbHMoImN0cmwiKSkgeyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgICAgIG1vZGlmaWVycyB8PSBJbnB1dEV2ZW50LkNUUkxfRE9XTl9NQVNLOwotICAgICAgICB9IGVsc2UgaWYgKHN0ck1vZC5lcXVhbHMoIm1ldGEiKSkgeyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBtb2RpZmllcnMgfD0gSW5wdXRFdmVudC5NRVRBX0RPV05fTUFTSzsKLSAgICAgICAgfSBlbHNlIGlmIChzdHJNb2QuZXF1YWxzKCJhbHQiKSkgeyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBtb2RpZmllcnMgfD0gSW5wdXRFdmVudC5BTFRfRE9XTl9NQVNLOwotICAgICAgICB9IGVsc2UgaWYgKHN0ck1vZC5lcXVhbHMoImFsdEdyYXBoIikpIHsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgbW9kaWZpZXJzIHw9IElucHV0RXZlbnQuQUxUX0dSQVBIX0RPV05fTUFTSzsKLSAgICAgICAgfSBlbHNlIGlmIChzdHJNb2QuZXF1YWxzKCJidXR0b24xIikpIHsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgbW9kaWZpZXJzIHw9IElucHV0RXZlbnQuQlVUVE9OMV9ET1dOX01BU0s7Ci0gICAgICAgIH0gZWxzZSBpZiAoc3RyTW9kLmVxdWFscygiYnV0dG9uMiIpKSB7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIG1vZGlmaWVycyB8PSBJbnB1dEV2ZW50LkJVVFRPTjJfRE9XTl9NQVNLOwotICAgICAgICB9IGVsc2UgaWYgKHN0ck1vZC5lcXVhbHMoImJ1dHRvbjMiKSkgeyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBtb2RpZmllcnMgfD0gSW5wdXRFdmVudC5CVVRUT04zX0RPV05fTUFTSzsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbW9kaWZpZXJzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBhcnNlcyB0aGUgdHlwZWQgaWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0clR5cGVkCi0gICAgICogICAgICAgICAgICB0aGUgc3RyIHR5cGVkLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgc3VjY2Vzc2Z1bC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBib29sZWFuIHBhcnNlVHlwZWRJRChTdHJpbmcgc3RyVHlwZWQpIHsKLSAgICAgICAgaWYgKHN0clR5cGVkLmVxdWFscygidHlwZWQiKSkgeyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQYXJzZXMgdGhlIHR5cGVkIGtleS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3RyQ2hhcgotICAgICAqICAgICAgICAgICAgdGhlIHN0ciBjaGFyLgotICAgICAqIEByZXR1cm4gdGhlIGNoYXIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgY2hhciBwYXJzZVR5cGVkS2V5KFN0cmluZyBzdHJDaGFyKSB7Ci0gICAgICAgIGNoYXIga2V5Q2hhciA9IEtleUV2ZW50LkNIQVJfVU5ERUZJTkVEOwotCi0gICAgICAgIGlmIChzdHJDaGFyLmxlbmd0aCgpICE9IDEpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Nj1JbnZhbGlkIGZvcm1hdAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42NiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGtleUNoYXIgPSBzdHJDaGFyLmNoYXJBdCgwKTsKLSAgICAgICAgcmV0dXJuIGtleUNoYXI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUGFyc2VzIHRoZSBwcmVzc2VkIHJlbGVhc2VkIGlkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdHIuCi0gICAgICogQHJldHVybiB0aGUgYm9vbGVhbi4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBCb29sZWFuIHBhcnNlUHJlc3NlZFJlbGVhc2VkSUQoU3RyaW5nIHN0cikgewotCi0gICAgICAgIGlmIChzdHIuZXF1YWxzKCJwcmVzc2VkIikpIHsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgcmV0dXJuIEJvb2xlYW4uRkFMU0U7Ci0gICAgICAgIH0gZWxzZSBpZiAoc3RyLmVxdWFscygicmVsZWFzZWQiKSkgeyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICByZXR1cm4gQm9vbGVhbi5UUlVFOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBhcnNlcyB0aGUga2V5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHJDb2RlCi0gICAgICogICAgICAgICAgICB0aGUgc3RyIGNvZGUuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGludCBwYXJzZUtleShTdHJpbmcgc3RyQ29kZSkgewotICAgICAgICBpbnQga2V5Q29kZSA9IEtleUV2ZW50LlZLX1VOREVGSU5FRDsKLQotICAgICAgICBrZXlDb2RlID0gZ2V0S2V5Q29kZShzdHJDb2RlKTsKLQotICAgICAgICBpZiAoa2V5Q29kZSA9PSBLZXlFdmVudC5WS19VTkRFRklORUQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Nj1JbnZhbGlkIGZvcm1hdAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42NiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBrZXlDb2RlOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9BV1RMaXN0ZW5lckxpc3QuamF2YSBiL2F3dC9qYXZhL2F3dC9BV1RMaXN0ZW5lckxpc3QuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzMyN2Q2My4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvQVdUTGlzdGVuZXJMaXN0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0NyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lkxpc3RlbmVyTGlzdDsKLQotZmluYWwgY2xhc3MgQVdUTGlzdGVuZXJMaXN0PFQgZXh0ZW5kcyBFdmVudExpc3RlbmVyPiBleHRlbmRzIExpc3RlbmVyTGlzdDxUPiB7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gLTI2MjIwNzcxNzE1MzI4NDA5NTNMOwotCi0gICAgcHJpdmF0ZSBmaW5hbCBDb21wb25lbnQgb3duZXI7Ci0gICAgCi0gICAgQVdUTGlzdGVuZXJMaXN0KCkgewotICAgICAgICBzdXBlcigpOwotICAgICAgICB0aGlzLm93bmVyID0gbnVsbDsKLSAgICB9Ci0KLSAgICBBV1RMaXN0ZW5lckxpc3QoQ29tcG9uZW50IG93bmVyKSB7Ci0gICAgICAgIHN1cGVyKCk7Ci0gICAgICAgIHRoaXMub3duZXIgPSBvd25lcjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBhZGRVc2VyTGlzdGVuZXIoVCBsaXN0ZW5lcikgewotICAgICAgICBzdXBlci5hZGRVc2VyTGlzdGVuZXIobGlzdGVuZXIpOwotCi0gICAgICAgIGlmIChvd25lciAhPSBudWxsKSB7Ci0gICAgICAgICAgICBvd25lci5kZXByZWNhdGVkRXZlbnRIYW5kbGVyID0gZmFsc2U7Ci0gICAgICAgIH0KLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvQVdUUGVybWlzc2lvbi5qYXZhIGIvYXd0L2phdmEvYXd0L0FXVFBlcm1pc3Npb24uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNGJkODM1Ny4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvQVdUUGVybWlzc2lvbi5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNjEgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5zZWN1cml0eS5CYXNpY1Blcm1pc3Npb247Ci0KLS8qKgotICogVGhlIEFXVFBlcm1pc3Npb24gc3BlY2lmaWVzIHRoZSBuYW1lIG9mIHRoZSBwZXJtaXNzaW9uIGFuZCB0aGUgY29ycmVzcG9uZGluZwotICogYWN0aW9uIGxpc3QuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgZmluYWwgY2xhc3MgQVdUUGVybWlzc2lvbiBleHRlbmRzIEJhc2ljUGVybWlzc2lvbiB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSA4ODkwMzkyNDAyNTg4ODE0NDY1TDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBBV1RQZXJtaXNzaW9uIHdpdGggZGVmaW5lZCBuYW1lIGFuZCBhY3Rpb25zLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiBhIG5ldyBBV1RQZXJtaXNzaW9uLgotICAgICAqIEBwYXJhbSBhY3Rpb25zCi0gICAgICogICAgICAgICAgICB0aGUgYWN0aW9ucyBvZiBhIG5ldyBBV1RQZXJtaXNzaW9uLgotICAgICAqLwotICAgIHB1YmxpYyBBV1RQZXJtaXNzaW9uKFN0cmluZyBuYW1lLCBTdHJpbmcgYWN0aW9ucykgewotICAgICAgICBzdXBlcihuYW1lLCBhY3Rpb25zKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQVdUIHBlcm1pc3Npb24gd2l0aCB0aGUgZGVmaW5lZCBuYW1lLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiBhIG5ldyBBV1RQZXJtaXNzaW9uLgotICAgICAqLwotICAgIHB1YmxpYyBBV1RQZXJtaXNzaW9uKFN0cmluZyBuYW1lKSB7Ci0gICAgICAgIHN1cGVyKG5hbWUpOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0FjdGl2ZUV2ZW50LmphdmEgYi9hd3QvamF2YS9hd3QvQWN0aXZlRXZlbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzA0NDYyMy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvQWN0aXZlRXZlbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDM5ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotLyoqCi0gKiBUaGlzIGludGVyZmFjZSBkZWZpbmVzIGV2ZW50cyB0aGF0IGtub3cgaG93IHRvIGRpc3BhdGNoIHRoZW1zZWx2ZXMuIFN1Y2gKLSAqIGV2ZW50IGNhbiBiZSBwbGFjZWQgdXBvbiB0aGUgZXZlbnQgcXVldWUgYW5kIGl0cyBkaXNwYXRjaCBtZXRob2Qgd2lsbCBiZQotICogY2FsbGVkIHdoZW4gdGhlIGV2ZW50IGlzIGRpc3BhdGNoZWQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIEFjdGl2ZUV2ZW50IHsKLQotICAgIC8qKgotICAgICAqIERpc3BhdGNoZXMgdGhlIGV2ZW50IHRvIHRoZSBsaXN0ZW5lcnMgb2YgdGhlIGV2ZW50J3Mgc291cmNlLCBvciBkb2VzCi0gICAgICogd2hhdGV2ZXIgaXQgaXMgdGhpcyBldmVudCBpcyBzdXBwb3NlZCB0byBkby4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBkaXNwYXRjaCgpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvQWRqdXN0YWJsZS5qYXZhIGIvYXd0L2phdmEvYXd0L0FkanVzdGFibGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYmFmODBmNy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvQWRqdXN0YWJsZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTY2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEuYXd0LmV2ZW50LkFkanVzdG1lbnRMaXN0ZW5lcjsKLQotLyoqCi0gKiBUaGUgQWRqdXN0YWJsZSBpbnRlcmZhY2UgcmVwcmVzZW50cyBhbiBhZGp1c3RhYmxlIG51bWVyaWMgdmFsdWUgY29udGFpbmVkCi0gKiB3aXRoaW4gYSBib3VuZGVkIHJhbmdlIG9mIHZhbHVlcywgc3VjaCBhcyB0aGUgY3VycmVudCBsb2NhdGlvbiBpbiBzY3JvbGxhYmxlCi0gKiByZWdpb24gb3IgdGhlIHZhbHVlIG9mIGEgZ2F1Z2UuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIEFkanVzdGFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEhPUklaT05UQUwgaW5kaWNhdGVzIHRoYXQgdGhlIEFkanVzdGFibGUncyBvcmllbnRhdGlvbiBpcwotICAgICAqIGhvcml6b250YWwuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSE9SSVpPTlRBTCA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVkVSVElDQUwgaW5kaWNhdGVzIHRoYXQgdGhlIEFkanVzdGFibGUncyBvcmllbnRhdGlvbiBpcwotICAgICAqIHZlcnRpY2FsLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZFUlRJQ0FMID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBOT19PUklFTlRBVElPTiBpbmRpY2F0ZXMgdGhhdCB0aGUgQWRqdXN0YWJsZSBoYXMgbm8KLSAgICAgKiBvcmllbnRhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBOT19PUklFTlRBVElPTiA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB2YWx1ZSBvZiB0aGUgQWRqdXN0YWJsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjdXJyZW50IHZhbHVlIG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VmFsdWUoKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHZhbHVlIHRvIHRoZSBBZGp1c3RhYmxlIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYTAKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgdmFsdWUgb2YgdGhlIEFkanVzdGFibGUgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFZhbHVlKGludCBhMCk7Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBBZGp1c3RtZW50TGlzdGVuZXIgdG8gY3VycmVudCBBZGp1c3RtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIEFkanVzdG1lbnRMaXN0ZW5lciBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkQWRqdXN0bWVudExpc3RlbmVyKEFkanVzdG1lbnRMaXN0ZW5lciBhMCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBibG9jayBpbmNyZW1lbnQgb2YgdGhlIEFkanVzdGFibGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYmxvY2sgaW5jcmVtZW50IG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0QmxvY2tJbmNyZW1lbnQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1heGltdW0gdmFsdWUgb2YgdGhlIEFkanVzdGFibGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWF4aW11bSB2YWx1ZSBvZiB0aGUgQWRqdXN0YWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE1heGltdW0oKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1pbmltdW0gdmFsdWUgb2YgdGhlIEFkanVzdGFibGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWluaW11bSB2YWx1ZSBvZiB0aGUgQWRqdXN0YWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE1pbmltdW0oKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG9yaWVudGF0aW9uIG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG9yaWVudGF0aW9uIG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0T3JpZW50YXRpb24oKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHVuaXQgaW5jcmVtZW50IG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHVuaXQgaW5jcmVtZW50IG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VW5pdEluY3JlbWVudCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdmlzaWJsZSBhbW91bnQgb2YgdGhlIEFkanVzdGFibGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdmlzaWJsZSBhbW91bnQgb2YgdGhlIEFkanVzdGFibGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRWaXNpYmxlQW1vdW50KCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHRoZSBhZGp1c3RtZW50IGxpc3RlbmVyIG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBBZGp1c3RtZW50TGlzdGVuZXIgdG8gYmUgcmVtb3ZlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVBZGp1c3RtZW50TGlzdGVuZXIoQWRqdXN0bWVudExpc3RlbmVyIGEwKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGJsb2NrIGluY3JlbWVudCBmb3IgdGhlIEFkanVzdGFibGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IGJsb2NrIGluY3JlbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRCbG9ja0luY3JlbWVudChpbnQgYTApOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbWF4aW11bSB2YWx1ZSBvZiB0aGUgQWRqdXN0YWJsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYTAKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgbWF4aW11bSBvZiB0aGUgQWRqdXN0YWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRNYXhpbXVtKGludCBhMCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBtaW5pbXVtIHZhbHVlIG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBtaW5pbXVtIG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldE1pbmltdW0oaW50IGEwKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHVuaXQgaW5jcmVtZW50IG9mIHRoZSBBZGp1c3RhYmxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIG5ldyB1bml0IGluY3JlbWVudCBvZiB0aGUgQWRqdXN0YWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRVbml0SW5jcmVtZW50KGludCBhMCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSB2aXNpYmxlIGFtb3VudCBvZiB0aGUgQWRqdXN0YWJsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYTAKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgdmlzaWJsZSBhbW91bnQgb2YgdGhlIEFkanVzdGFibGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0VmlzaWJsZUFtb3VudChpbnQgYTApOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvQWxwaGFDb21wb3NpdGUuamF2YSBiL2F3dC9qYXZhL2F3dC9BbHBoYUNvbXBvc2l0ZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4Mzg5ZWI0Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9BbHBoYUNvbXBvc2l0ZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzUyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuQ29tcG9zaXRlOwotaW1wb3J0IGphdmEuYXd0LkNvbXBvc2l0ZUNvbnRleHQ7Ci1pbXBvcnQgamF2YS5hd3QuUmVuZGVyaW5nSGludHM7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuSUNvbXBvc2l0ZUNvbnRleHQ7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIEFscGhhQ29tcG9zaXRlIGNsYXNzIGRlZmluZXMgYSBiYXNpYyBhbHBoYSBjb21wb3NpdGluZyBydWxlcyBmb3IKLSAqIGNvbWJpbmluZyBzb3VyY2UgYW5kIGRlc3RpbmF0aW9uIGNvbG9ycyB0byBhY2hpZXZlIGJsZW5kaW5nIGFuZCB0cmFuc3BhcmVuY3kKLSAqIGVmZmVjdHMgd2l0aCBncmFwaGljcyBhbmQgaW1hZ2VzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIEFscGhhQ29tcG9zaXRlIGltcGxlbWVudHMgQ29tcG9zaXRlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDTEVBUiBpbmRpY2F0ZXMgdGhhdCBib3RoIHRoZSBjb2xvciBhbmQgdGhlIGFscGhhIG9mIHRoZQotICAgICAqIGRlc3RpbmF0aW9uIGFyZSBjbGVhcmVkIChQb3J0ZXItRHVmZiBDbGVhciBydWxlKS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDTEVBUiA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU1JDIGluZGljYXRlcyB0aGF0IHRoZSBzb3VyY2UgaXMgY29waWVkIHRvIHRoZSBkZXN0aW5hdGlvbgotICAgICAqIChQb3J0ZXItRHVmZiBTb3VyY2UgcnVsZSkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU1JDID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBEU1QgaW5kaWNhdGVzIHRoYXQgdGhlIGRlc3RpbmF0aW9uIGlzIGxlZnQgdW50b3VjaGVkCi0gICAgICogKFBvcnRlci1EdWZmIERlc3RpbmF0aW9uIHJ1bGUpLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IERTVCA9IDk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU1JDX09WRVIgaW5kaWNhdGVzIHRoYXQgdGhlIHNvdXJjZSBpcyBjb21wb3NpdGVkIG92ZXIgdGhlCi0gICAgICogZGVzdGluYXRpb24gKFBvcnRlci1EdWZmIFNvdXJjZSBPdmVyIERlc3RpbmF0aW9uIHJ1bGUpLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNSQ19PVkVSID0gMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBEU1RfT1ZFUiBpbmRpY2F0ZXMgdGhhdCBUaGUgZGVzdGluYXRpb24gaXMgY29tcG9zaXRlZCBvdmVyCi0gICAgICogdGhlIHNvdXJjZSBhbmQgdGhlIHJlc3VsdCByZXBsYWNlcyB0aGUgZGVzdGluYXRpb24gKFBvcnRlci1EdWZmCi0gICAgICogRGVzdGluYXRpb24gT3ZlciBTb3VyY2UgcnVsZSkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRFNUX09WRVIgPSA0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNSQ19JTiBpbmRpY2F0ZXMgdGhhdCB0aGUgcGFydCBvZiB0aGUgc291cmNlIGx5aW5nIGluc2lkZSBvZgotICAgICAqIHRoZSBkZXN0aW5hdGlvbiByZXBsYWNlcyB0aGUgZGVzdGluYXRpb24gKFBvcnRlci1EdWZmIFNvdXJjZSBJbgotICAgICAqIERlc3RpbmF0aW9uIHJ1bGUpLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNSQ19JTiA9IDU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRFNUX0lOIGluZGljYXRlcyB0aGF0IHRoZSBwYXJ0IG9mIHRoZSBkZXN0aW5hdGlvbiBseWluZwotICAgICAqIGluc2lkZSBvZiB0aGUgc291cmNlIHJlcGxhY2VzIHRoZSBkZXN0aW5hdGlvbiAoUG9ydGVyLUR1ZmYgRGVzdGluYXRpb24gSW4KLSAgICAgKiBTb3VyY2UgcnVsZSkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRFNUX0lOID0gNjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTUkNfT1VUIGluZGljYXRlcyB0aGF0IHRoZSBwYXJ0IG9mIHRoZSBzb3VyY2UgbHlpbmcgb3V0c2lkZQotICAgICAqIG9mIHRoZSBkZXN0aW5hdGlvbiByZXBsYWNlcyB0aGUgZGVzdGluYXRpb24gKFBvcnRlci1EdWZmIFNvdXJjZSBIZWxkIE91dAotICAgICAqIEJ5IERlc3RpbmF0aW9uIHJ1bGUpLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNSQ19PVVQgPSA3OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IERTVF9PVVQgaW5kaWNhdGVzIHRoYXQgdGhlIHBhcnQgb2YgdGhlIGRlc3RpbmF0aW9uIGx5aW5nCi0gICAgICogb3V0c2lkZSBvZiB0aGUgc291cmNlIHJlcGxhY2VzIHRoZSBkZXN0aW5hdGlvbiAoUG9ydGVyLUR1ZmYgRGVzdGluYXRpb24KLSAgICAgKiBIZWxkIE91dCBCeSBTb3VyY2UgcnVsZSkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRFNUX09VVCA9IDg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU1JDX0FUT1AgaW5kaWNhdGVzIHRoYXQgdGhlIHBhcnQgb2YgdGhlIHNvdXJjZSBseWluZyBpbnNpZGUKLSAgICAgKiBvZiB0aGUgZGVzdGluYXRpb24gaXMgY29tcG9zaXRlZCBvbnRvIHRoZSBkZXN0aW5hdGlvbiAoUG9ydGVyLUR1ZmYgU291cmNlCi0gICAgICogQXRvcCBEZXN0aW5hdGlvbiBydWxlKS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTUkNfQVRPUCA9IDEwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IERTVF9BVE9QIGluZGljYXRlcyB0aGF0IHRoZSBwYXJ0IG9mIHRoZSBkZXN0aW5hdGlvbiBseWluZwotICAgICAqIGluc2lkZSBvZiB0aGUgc291cmNlIGlzIGNvbXBvc2l0ZWQgb3ZlciB0aGUgc291cmNlIGFuZCByZXBsYWNlcyB0aGUKLSAgICAgKiBkZXN0aW5hdGlvbiAoUG9ydGVyLUR1ZmYgRGVzdGluYXRpb24gQXRvcCBTb3VyY2UgcnVsZSkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRFNUX0FUT1AgPSAxMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBYT1IgaW5kaWNhdGVzIHRoYXQgdGhlIHBhcnQgb2YgdGhlIHNvdXJjZSB0aGF0IGxpZXMgb3V0c2lkZQotICAgICAqIG9mIHRoZSBkZXN0aW5hdGlvbiBpcyBjb21iaW5lZCB3aXRoIHRoZSBwYXJ0IG9mIHRoZSBkZXN0aW5hdGlvbiB0aGF0IGxpZXMKLSAgICAgKiBvdXRzaWRlIG9mIHRoZSBzb3VyY2UgKFBvcnRlci1EdWZmIFNvdXJjZSBYb3IgRGVzdGluYXRpb24gcnVsZSkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgWE9SID0gMTI7Ci0KLSAgICAvKioKLSAgICAgKiBBbHBoYUNvbXBvc2l0ZSBvYmplY3Qgd2l0aCB0aGUgb3BhcXVlIENMRUFSIHJ1bGUgYW5kIGFuIGFscGhhIG9mIDEuMGYuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBBbHBoYUNvbXBvc2l0ZSBDbGVhciA9IG5ldyBBbHBoYUNvbXBvc2l0ZShDTEVBUik7Ci0KLSAgICAvKioKLSAgICAgKiBBbHBoYUNvbXBvc2l0ZSBvYmplY3Qgd2l0aCB0aGUgb3BhcXVlIFNSQyBydWxlIGFuZCBhbiBhbHBoYSBvZiAxLjBmLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQWxwaGFDb21wb3NpdGUgU3JjID0gbmV3IEFscGhhQ29tcG9zaXRlKFNSQyk7Ci0KLSAgICAvKioKLSAgICAgKiBBbHBoYUNvbXBvc2l0ZSBvYmplY3Qgd2l0aCB0aGUgb3BhcXVlIERTVCBydWxlIGFuZCBhbiBhbHBoYSBvZiAxLjBmLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQWxwaGFDb21wb3NpdGUgRHN0ID0gbmV3IEFscGhhQ29tcG9zaXRlKERTVCk7Ci0KLSAgICAvKioKLSAgICAgKiBBbHBoYUNvbXBvc2l0ZSBvYmplY3Qgd2l0aCB0aGUgb3BhcXVlIFNSQ19PVkVSIHJ1bGUgYW5kIGFuIGFscGhhIG9mIDEuMGYuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBBbHBoYUNvbXBvc2l0ZSBTcmNPdmVyID0gbmV3IEFscGhhQ29tcG9zaXRlKFNSQ19PVkVSKTsKLQotICAgIC8qKgotICAgICAqIEFscGhhQ29tcG9zaXRlIG9iamVjdCB3aXRoIHRoZSBvcGFxdWUgRFNUX09WRVIgcnVsZSBhbmQgYW4gYWxwaGEgb2YgMS4wZi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEFscGhhQ29tcG9zaXRlIERzdE92ZXIgPSBuZXcgQWxwaGFDb21wb3NpdGUoRFNUX09WRVIpOwotCi0gICAgLyoqCi0gICAgICogQWxwaGFDb21wb3NpdGUgb2JqZWN0IHdpdGggdGhlIG9wYXF1ZSBTUkNfSU4gcnVsZSBhbmQgYW4gYWxwaGEgb2YgMS4wZi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEFscGhhQ29tcG9zaXRlIFNyY0luID0gbmV3IEFscGhhQ29tcG9zaXRlKFNSQ19JTik7Ci0KLSAgICAvKioKLSAgICAgKiBBbHBoYUNvbXBvc2l0ZSBvYmplY3Qgd2l0aCB0aGUgb3BhcXVlIERTVF9JTiBydWxlIGFuZCBhbiBhbHBoYSBvZiAxLjBmLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQWxwaGFDb21wb3NpdGUgRHN0SW4gPSBuZXcgQWxwaGFDb21wb3NpdGUoRFNUX0lOKTsKLQotICAgIC8qKgotICAgICAqIEFscGhhQ29tcG9zaXRlIG9iamVjdCB3aXRoIHRoZSBvcGFxdWUgU1JDX09VVCBydWxlIGFuZCBhbiBhbHBoYSBvZiAxLjBmLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQWxwaGFDb21wb3NpdGUgU3JjT3V0ID0gbmV3IEFscGhhQ29tcG9zaXRlKFNSQ19PVVQpOwotCi0gICAgLyoqCi0gICAgICogQWxwaGFDb21wb3NpdGUgb2JqZWN0IHdpdGggdGhlIG9wYXF1ZSBEU1RfT1VUIHJ1bGUgYW5kIGFuIGFscGhhIG9mIDEuMGYuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBBbHBoYUNvbXBvc2l0ZSBEc3RPdXQgPSBuZXcgQWxwaGFDb21wb3NpdGUoRFNUX09VVCk7Ci0KLSAgICAvKioKLSAgICAgKiBBbHBoYUNvbXBvc2l0ZSBvYmplY3Qgd2l0aCB0aGUgb3BhcXVlIFNSQ19BVE9QIHJ1bGUgYW5kIGFuIGFscGhhIG9mIDEuMGYuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBBbHBoYUNvbXBvc2l0ZSBTcmNBdG9wID0gbmV3IEFscGhhQ29tcG9zaXRlKFNSQ19BVE9QKTsKLQotICAgIC8qKgotICAgICAqIEFscGhhQ29tcG9zaXRlIG9iamVjdCB3aXRoIHRoZSBvcGFxdWUgRFNUX0FUT1AgcnVsZSBhbmQgYW4gYWxwaGEgb2YgMS4wZi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEFscGhhQ29tcG9zaXRlIERzdEF0b3AgPSBuZXcgQWxwaGFDb21wb3NpdGUoRFNUX0FUT1ApOwotCi0gICAgLyoqCi0gICAgICogQWxwaGFDb21wb3NpdGUgb2JqZWN0IHdpdGggdGhlIG9wYXF1ZSBYT1IgcnVsZSBhbmQgYW4gYWxwaGEgb2YgMS4wZi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEFscGhhQ29tcG9zaXRlIFhvciA9IG5ldyBBbHBoYUNvbXBvc2l0ZShYT1IpOwotCi0gICAgLyoqCi0gICAgICogVGhlIHJ1bGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgcnVsZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBhbHBoYS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGFscGhhOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGFscGhhIGNvbXBvc2l0ZS4gQ3JlYXRlcyBhIGNvbnRleHQgZm9yIHRoZSBjb21wb3NpdGluZwotICAgICAqIG9wZXJhdGlvbi4gVGhlIGNvbnRleHQgY29udGFpbnMgc3RhdGUgdGhhdCBpcyB1c2VkIGluIHBlcmZvcm1pbmcgdGhlCi0gICAgICogY29tcG9zaXRpbmcgb3BlcmF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBydWxlCi0gICAgICogICAgICAgICAgICB0aGUgcnVsZS4KLSAgICAgKiBAcGFyYW0gYWxwaGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbHBoYS4KLSAgICAgKi8KLSAgICBwcml2YXRlIEFscGhhQ29tcG9zaXRlKGludCBydWxlLCBmbG9hdCBhbHBoYSkgewotICAgICAgICBpZiAocnVsZSA8IENMRUFSIHx8IHJ1bGUgPiBYT1IpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xMUQ9VW5rbm93biBydWxlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjExRCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGlmIChhbHBoYSA8IDAuMGYgfHwgYWxwaGEgPiAxLjBmKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTFFPVdyb25nIGFscGhhIHZhbHVlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjExRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5ydWxlID0gcnVsZTsKLSAgICAgICAgdGhpcy5hbHBoYSA9IGFscGhhOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBhbHBoYSBjb21wb3NpdGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJ1bGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBydWxlLgotICAgICAqLwotICAgIHByaXZhdGUgQWxwaGFDb21wb3NpdGUoaW50IHJ1bGUpIHsKLSAgICAgICAgdGhpcyhydWxlLCAxLjBmKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgQ29tcG9zaXRlQ29udGV4dCBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIHNvdXJjZSBDb2xvck1vZGVsLAotICAgICAqIGRlc3RpbmF0aW9uIENvbG9yTW9kZWwgYW5kIFJlbmRlcmluZ0hpbnRzIHBhcmFtZXRlcnMgZm9yIGEgY29tcG9zaW5nCi0gICAgICogb3BlcmF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmNDb2xvck1vZGVsCi0gICAgICogICAgICAgICAgICB0aGUgc291cmNlJ3MgQ29sb3JNb2RlbC4KLSAgICAgKiBAcGFyYW0gZHN0Q29sb3JNb2RlbAotICAgICAqICAgICAgICAgICAgdGhlIGRlc3RpbmF0aW9uJ3MgQ29sb3JNb2RlbC4KLSAgICAgKiBAcGFyYW0gaGludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJpbmdIaW50cyBvYmplY3QuCi0gICAgICogQHJldHVybiB0aGUgQ29tcG9zaXRlQ29udGV4dCBvYmplY3QuCi0gICAgICogQHNlZSBqYXZhLmF3dC5Db21wb3NpdGUjY3JlYXRlQ29udGV4dChqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsLAotICAgICAqICAgICAgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbCwgamF2YS5hd3QuUmVuZGVyaW5nSGludHMpCi0gICAgICovCi0gICAgcHVibGljIENvbXBvc2l0ZUNvbnRleHQgY3JlYXRlQ29udGV4dChDb2xvck1vZGVsIHNyY0NvbG9yTW9kZWwsIENvbG9yTW9kZWwgZHN0Q29sb3JNb2RlbCwKLSAgICAgICAgICAgIFJlbmRlcmluZ0hpbnRzIGhpbnRzKSB7Ci0gICAgICAgIHJldHVybiBuZXcgSUNvbXBvc2l0ZUNvbnRleHQodGhpcywgc3JjQ29sb3JNb2RlbCwgZHN0Q29sb3JNb2RlbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgdGhlIEFscGhhQ29tcG9zaXRlIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgQWxwaGFDb21wb3NpdGUgb2JqZWN0IGlzIGVxdWFsIHRvIHRoZSBzcGVjaWZpZWQKLSAgICAgKiAgICAgICAgIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG9iaikgewotICAgICAgICBpZiAoIShvYmogaW5zdGFuY2VvZiBBbHBoYUNvbXBvc2l0ZSkpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgICAgICBBbHBoYUNvbXBvc2l0ZSBvdGhlciA9IChBbHBoYUNvbXBvc2l0ZSlvYmo7Ci0gICAgICAgIHJldHVybiAodGhpcy5ydWxlID09IG90aGVyLmdldFJ1bGUoKSAmJiB0aGlzLmFscGhhID09IG90aGVyLmdldEFscGhhKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGhhc2ggY29kZSBvZiB0aGUgQWxwaGFDb21wb3NpdGUgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhhc2ggY29kZSBvZiB0aGUgQWxwaGFDb21wb3NpdGUgb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgaGFzaENvZGUoKSB7Ci0gICAgICAgIGludCBoYXNoID0gRmxvYXQuZmxvYXRUb0ludEJpdHMoYWxwaGEpOwotICAgICAgICBpbnQgdG1wID0gaGFzaCA+Pj4gMjQ7Ci0gICAgICAgIGhhc2ggPDw9IDg7Ci0gICAgICAgIGhhc2ggfD0gdG1wOwotICAgICAgICBoYXNoIF49IHJ1bGU7Ci0gICAgICAgIHJldHVybiBoYXNoOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvbXBvc2l0aW5nIHJ1bGUgb2YgdGhpcyBBbHBoYUNvbXBvc2l0ZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY29tcG9zaXRpbmcgcnVsZSBvZiB0aGlzIEFscGhhQ29tcG9zaXRlIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFJ1bGUoKSB7Ci0gICAgICAgIHJldHVybiBydWxlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFscGhhIHZhbHVlIG9mIHRoaXMgQWxwaGFDb21wb3NpdGUgb2JqZWN0OyByZXR1cm5zIDEuMCBpZiB0aGlzCi0gICAgICogQWxwaGFDb21wb3NpdGUgb2JqZWN0IGRvZXNuJ3QgaGF2ZSBhbHBoYSB2YWx1ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhbHBoYSB2YWx1ZSBvZiB0aGlzIEFscGhhQ29tcG9zaXRlIG9iamVjdCBvciAxLjAgaWYgdGhpcwotICAgICAqICAgICAgICAgQWxwaGFDb21wb3NpdGUgb2JqZWN0IGRvZXNuJ3QgaGF2ZSBhbHBoYSB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0QWxwaGEoKSB7Ci0gICAgICAgIHJldHVybiBhbHBoYTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBBbHBoYUNvbXBvc2l0ZSBpbnN0YW5jZSB3aXRoIHRoZSBzcGVjaWZpZWQgcnVsZSBhbmQgYWxwaGEgdmFsdWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJ1bGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb21wb3NpdGluZyBydWxlLgotICAgICAqIEBwYXJhbSBhbHBoYQotICAgICAqICAgICAgICAgICAgdGhlIGFscGhhIHZhbHVlLgotICAgICAqIEByZXR1cm4gdGhlIEFscGhhQ29tcG9zaXRlIGluc3RhbmNlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgQWxwaGFDb21wb3NpdGUgZ2V0SW5zdGFuY2UoaW50IHJ1bGUsIGZsb2F0IGFscGhhKSB7Ci0gICAgICAgIGlmIChhbHBoYSA9PSAxLjBmKSB7Ci0gICAgICAgICAgICByZXR1cm4gZ2V0SW5zdGFuY2UocnVsZSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBBbHBoYUNvbXBvc2l0ZShydWxlLCBhbHBoYSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQWxwaGFDb21wb3NpdGUgaW5zdGFuY2Ugd2l0aCB0aGUgc3BlY2lmaWVkIHJ1bGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJ1bGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb21wb3NpdGluZyBydWxlLgotICAgICAqIEByZXR1cm4gdGhlIEFscGhhQ29tcG9zaXRlIGluc3RhbmNlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgQWxwaGFDb21wb3NpdGUgZ2V0SW5zdGFuY2UoaW50IHJ1bGUpIHsKLSAgICAgICAgc3dpdGNoIChydWxlKSB7Ci0gICAgICAgICAgICBjYXNlIENMRUFSOgotICAgICAgICAgICAgICAgIHJldHVybiBDbGVhcjsKLSAgICAgICAgICAgIGNhc2UgU1JDOgotICAgICAgICAgICAgICAgIHJldHVybiBTcmM7Ci0gICAgICAgICAgICBjYXNlIERTVDoKLSAgICAgICAgICAgICAgICByZXR1cm4gRHN0OwotICAgICAgICAgICAgY2FzZSBTUkNfT1ZFUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gU3JjT3ZlcjsKLSAgICAgICAgICAgIGNhc2UgRFNUX09WRVI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIERzdE92ZXI7Ci0gICAgICAgICAgICBjYXNlIFNSQ19JTjoKLSAgICAgICAgICAgICAgICByZXR1cm4gU3JjSW47Ci0gICAgICAgICAgICBjYXNlIERTVF9JTjoKLSAgICAgICAgICAgICAgICByZXR1cm4gRHN0SW47Ci0gICAgICAgICAgICBjYXNlIFNSQ19PVVQ6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIFNyY091dDsKLSAgICAgICAgICAgIGNhc2UgRFNUX09VVDoKLSAgICAgICAgICAgICAgICByZXR1cm4gRHN0T3V0OwotICAgICAgICAgICAgY2FzZSBTUkNfQVRPUDoKLSAgICAgICAgICAgICAgICByZXR1cm4gU3JjQXRvcDsKLSAgICAgICAgICAgIGNhc2UgRFNUX0FUT1A6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIERzdEF0b3A7Ci0gICAgICAgICAgICBjYXNlIFhPUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gWG9yOwotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAvLyBhd3QuMTFEPVVua25vd24gcnVsZQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTFEIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9CYXNpY1N0cm9rZS5qYXZhIGIvYXd0L2phdmEvYXd0L0Jhc2ljU3Ryb2tlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI0NTc4MTUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0Jhc2ljU3Ryb2tlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNDQzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuZ2VvbS5HZW5lcmFsUGF0aDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBhdGhJdGVyYXRvcjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5taXNjLkhhc2hDb2RlOwotCi0vKioKLSAqIFRoZSBCYXNpY1N0cm9rZSBjbGFzcyBzcGVjaWZpZXMgYSBzZXQgb2YgcmVuZGVyaW5nIGF0dHJpYnV0ZXMgZm9yIHRoZQotICogb3V0bGluZXMgb2YgZ3JhcGhpY3MgcHJpbWl0aXZlcy4gVGhlIEJhc2ljU3Ryb2tlIGF0dHJpYnV0ZXMgZGVzY3JpYmUgdGhlCi0gKiBzaGFwZSBvZiB0aGUgcGVuIHdoaWNoIGRyYXdzIHRoZSBvdXRsaW5lIG9mIGEgU2hhcGUgYW5kIHRoZSBkZWNvcmF0aW9ucwotICogYXBwbGllZCBhdCB0aGUgZW5kcyBhbmQgam9pbnMgb2YgcGF0aCBzZWdtZW50cyBvZiB0aGUgU2hhcGUuIFRoZSBCYXNpY1N0cm9rZQotICogaGFzIHRoZSBmb2xsb3dpbmcgcmVuZGVyaW5nIGF0dHJpYnV0ZXM6Ci0gKiA8cD4KLSAqIDx1bD4KLSAqIDxsaT5saW5lIHdpZHRoIC10aGUgcGVuIHdpZHRoIHdoaWNoIGRyYXdzIHRoZSBvdXRsaW5lcy48L2xpPgotICogPGxpPmVuZCBjYXBzIC0gaW5kaWNhdGVzIHRoZSBkZWNvcmF0aW9uIGFwcGxpZWQgdG8gdGhlIGVuZHMgb2YgdW5jbG9zZWQKLSAqIHN1YnBhdGhzIGFuZCBkYXNoIHNlZ21lbnRzLiBUaGUgQmFzaWNTdHJva2UgZGVmaW5lcyB0aHJlZSBkaWZmZXJlbnQKLSAqIGRlY29yYXRpb25zOiBDQVBfQlVUVCwgQ0FQX1JPVU5ELCBhbmQgQ0FQX1NRVUFSRS48L2xpPgotICogPGxpPmxpbmUgam9pbnMgLSBpbmRpY2F0ZXMgdGhlIGRlY29yYXRpb24gYXBwbGllZCBhdCB0aGUgaW50ZXJzZWN0aW9uIG9mIHR3bwotICogcGF0aCBzZWdtZW50cyBhbmQgYXQgdGhlIGludGVyc2VjdGlvbiBvZiB0aGUgZW5kcG9pbnRzIG9mIGEgc3VicGF0aC4gVGhlCi0gKiBCYXNpY1N0cm9rZSBkZWZpbmVzIHRocmVlIGRlY29yYXRpb25zOiBKT0lOX0JFVkVMLCBKT0lOX01JVEVSLCBhbmQKLSAqIEpPSU5fUk9VTkQuPC9saT4KLSAqIDxsaT5taXRlciBsaW1pdCAtIHRoZSBsaW1pdCB0byB0cmltIGEgbGluZSBqb2luIHRoYXQgaGFzIGEgSk9JTl9NSVRFUgotICogZGVjb3JhdGlvbi48L2xpPgotICogPGxpPmRhc2ggYXR0cmlidXRlcyAtIHRoZSBkZWZpbml0aW9uIG9mIGhvdyB0byBtYWtlIGEgZGFzaCBwYXR0ZXJuIGJ5Ci0gKiBhbHRlcm5hdGluZyBiZXR3ZWVuIG9wYXF1ZSBhbmQgdHJhbnNwYXJlbnQgc2VjdGlvbnM8L2xpPgotICogPC91bD4KLSAqIDwvcD4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBCYXNpY1N0cm9rZSBpbXBsZW1lbnRzIFN0cm9rZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQ0FQX0JVVFQgaW5kaWNhdGVzIHRoZSBlbmRzIG9mIHVuY2xvc2VkIHN1YnBhdGhzIGFuZCBkYXNoCi0gICAgICogc2VnbWVudHMgaGF2ZSBubyBhZGRlZCBkZWNvcmF0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENBUF9CVVRUID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDQVBfUk9VTkQgaW5kaWNhdGVzIHRoZSBlbmRzIG9mIHVuY2xvc2VkIHN1YnBhdGhzIGFuZCBkYXNoCi0gICAgICogc2VnbWVudHMgaGF2ZSBhIHJvdW5kIGRlY29yYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ0FQX1JPVU5EID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDQVBfU1FVQVJFIGluZGljYXRlcyB0aGUgZW5kcyBvZiB1bmNsb3NlZCBzdWJwYXRocyBhbmQgZGFzaAotICAgICAqIHNlZ21lbnRzIGhhdmUgYSBzcXVhcmUgcHJvamVjdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDQVBfU1FVQVJFID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBKT0lOX01JVEVSIGluZGljYXRlcyB0aGF0IHBhdGggc2VnbWVudHMgYXJlIGpvaW5lZCBieQotICAgICAqIGV4dGVuZGluZyB0aGVpciBvdXRzaWRlIGVkZ2VzIHVudGlsIHRoZXkgbWVldC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBKT0lOX01JVEVSID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBKT0lOX1JPVU5EIGluZGljYXRlcyB0aGF0IHBhdGggc2VnbWVudHMgYXJlIGpvaW5lZCBieQotICAgICAqIHJvdW5kaW5nIG9mZiB0aGUgY29ybmVyIGF0IGEgcmFkaXVzIG9mIGhhbGYgdGhlIGxpbmUgd2lkdGguCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSk9JTl9ST1VORCA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgSk9JTl9CRVZFTCBpbmRpY2F0ZXMgdGhhdCBwYXRoIHNlZ21lbnRzIGFyZSBqb2luZWQgYnkKLSAgICAgKiBjb25uZWN0aW5nIHRoZSBvdXRlciBjb3JuZXJzIG9mIHRoZWlyIHdpZGUgb3V0bGluZXMgd2l0aCBhIHN0cmFpZ2h0Ci0gICAgICogc2VnbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBKT0lOX0JFVkVMID0gMjsKLQotICAgIC8qKgotICAgICAqIENvbnN0YW50cyBmb3IgY2FsY3VsYXRpbmcuCi0gICAgICovCi0gICAgc3RhdGljIGZpbmFsIGludCBNQVhfTEVWRUwgPSAyMDsgLy8gTWF4aW1hbCBkZWVwbmVzcyBvZiBjdXJ2ZSBzdWJkaXZpc2lvbgotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENVUlZFX0RFTFRBLgotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBkb3VibGUgQ1VSVkVfREVMVEEgPSAyLjA7IC8vIFdpZHRoIHRvbGVyYW5jZQotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENPUk5FUl9BTkdMRS4KLSAgICAgKi8KLSAgICBzdGF0aWMgZmluYWwgZG91YmxlIENPUk5FUl9BTkdMRSA9IDQuMDsgLy8gTWluaW11bSBjb3JuZXIgYW5nbGUKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDT1JORVJfWkVSTy4KLSAgICAgKi8KLSAgICBzdGF0aWMgZmluYWwgZG91YmxlIENPUk5FUl9aRVJPID0gMC4wMTsgLy8gWmVybyBhbmdsZQotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENVQklDX0FSQy4KLSAgICAgKi8KLSAgICBzdGF0aWMgZmluYWwgZG91YmxlIENVQklDX0FSQyA9IDQuMCAvIDMuMCAqIChNYXRoLnNxcnQoMi4wKSAtIDEpOwotCi0gICAgLyoqCi0gICAgICogU3Ryb2tlIHdpZHRoLgotICAgICAqLwotICAgIGZsb2F0IHdpZHRoOwotCi0gICAgLyoqCi0gICAgICogU3Ryb2tlIGNhcCB0eXBlLgotICAgICAqLwotICAgIGludCBjYXA7Ci0KLSAgICAvKioKLSAgICAgKiBTdHJva2Ugam9pbiB0eXBlLgotICAgICAqLwotICAgIGludCBqb2luOwotCi0gICAgLyoqCi0gICAgICogU3Ryb2tlIG1pdGVyIGxpbWl0LgotICAgICAqLwotICAgIGZsb2F0IG1pdGVyTGltaXQ7Ci0KLSAgICAvKioKLSAgICAgKiBTdHJva2UgZGFzaGVzIGFycmF5LgotICAgICAqLwotICAgIGZsb2F0IGRhc2hbXTsKLQotICAgIC8qKgotICAgICAqIFN0cm9rZSBkYXNoIHBoYXNlLgotICAgICAqLwotICAgIGZsb2F0IGRhc2hQaGFzZTsKLQotICAgIC8qKgotICAgICAqIFRoZSB0ZW1wb3JhcnkgcHJlLWNhbGN1bGF0ZWQgdmFsdWVzLgotICAgICAqLwotICAgIGRvdWJsZSBjdXJ2ZURlbHRhOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvcm5lciBkZWx0YS4KLSAgICAgKi8KLSAgICBkb3VibGUgY29ybmVyRGVsdGE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgemVybyBkZWx0YS4KLSAgICAgKi8KLSAgICBkb3VibGUgemVyb0RlbHRhOwotCi0gICAgLyoqCi0gICAgICogVGhlIHcyLgotICAgICAqLwotICAgIGRvdWJsZSB3MjsKLQotICAgIC8qKgotICAgICAqIFRoZSBmbXkuCi0gICAgICovCi0gICAgZG91YmxlIGZteCwgZm15OwotCi0gICAgLyoqCi0gICAgICogVGhlIHNteS4KLSAgICAgKi8KLSAgICBkb3VibGUgc2N4LCBzY3ksIHNteCwgc215OwotCi0gICAgLyoqCi0gICAgICogVGhlIGN5LgotICAgICAqLwotICAgIGRvdWJsZSBteCwgbXksIGN4LCBjeTsKLQotICAgIC8qKgotICAgICAqIFRoZSB0ZW1wb3JhcnkgaW5kaWNhdG9ycy4KLSAgICAgKi8KLSAgICBib29sZWFuIGlzTW92ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBpcyBmaXJzdC4KLSAgICAgKi8KLSAgICBib29sZWFuIGlzRmlyc3Q7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY2hlY2sgbW92ZS4KLSAgICAgKi8KLSAgICBib29sZWFuIGNoZWNrTW92ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSB0ZW1wb3JhcnkgYW5kIGRlc3RpbmF0aW9uIHdvcmsgcGF0aHMuCi0gICAgICovCi0gICAgQnVmZmVyZWRQYXRoIGRzdCwgbHAsIHJwLCBzcDsKLQotICAgIC8qKgotICAgICAqIFN0cm9rZSBkYXNoZXIgY2xhc3MuCi0gICAgICovCi0gICAgRGFzaGVyIGRhc2hlcjsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBCYXNpY1N0cm9rZSB3aXRoIGRlZmF1bHQgd2lkdGgsIGNhcCwgam9pbiwgbGltaXQsIGRhc2gKLSAgICAgKiBhdHRyaWJ1dGVzIHBhcmFtZXRlcnMuIFRoZSBkZWZhdWx0IHBhcmFtZXRlcnMgYXJlIGEgc29saWQgbGluZSBvZiB3aWR0aAotICAgICAqIDEuMCwgQ0FQX1NRVUFSRSwgSk9JTl9NSVRFUiwgYSBtaXRlciBsaW1pdCBvZiAxMC4wLCBudWxsIGRhc2ggYXR0cmlidXRlcywKLSAgICAgKiBhbmQgYSBkYXNoIHBoYXNlIG9mIDAuMGYuCi0gICAgICovCi0gICAgcHVibGljIEJhc2ljU3Ryb2tlKCkgewotICAgICAgICB0aGlzKDEuMGYsIENBUF9TUVVBUkUsIEpPSU5fTUlURVIsIDEwLjBmLCBudWxsLCAwLjBmKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQmFzaWNTdHJva2Ugd2l0aCB0aGUgc3BlY2lmaWVkIHdpZHRoLCBjYXBzLCBqb2lucywKLSAgICAgKiBsaW1pdCwgZGFzaCBhdHRyaWJ1dGVzLCBkYXNoIHBoYXNlIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgQmFzaWtTdHJva2UuCi0gICAgICogQHBhcmFtIGNhcAotICAgICAqICAgICAgICAgICAgdGhlIGVuZCBkZWNvcmF0aW9uIG9mIEJhc2lrU3Ryb2tlLgotICAgICAqIEBwYXJhbSBqb2luCi0gICAgICogICAgICAgICAgICB0aGUgam9pbiBzZWdtZW50cyBkZWNvcmF0aW9uLgotICAgICAqIEBwYXJhbSBtaXRlckxpbWl0Ci0gICAgICogICAgICAgICAgICB0aGUgbGltaXQgdG8gdHJpbSB0aGUgbWl0ZXIgam9pbi4KLSAgICAgKiBAcGFyYW0gZGFzaAotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IHdpdGggdGhlIGRhc2hpbmcgcGF0dGVybi4KLSAgICAgKiBAcGFyYW0gZGFzaFBoYXNlCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IHRvIHN0YXJ0IHRoZSBkYXNoaW5nIHBhdHRlcm4uCi0gICAgICovCi0gICAgcHVibGljIEJhc2ljU3Ryb2tlKGZsb2F0IHdpZHRoLCBpbnQgY2FwLCBpbnQgam9pbiwgZmxvYXQgbWl0ZXJMaW1pdCwgZmxvYXRbXSBkYXNoLAotICAgICAgICAgICAgZmxvYXQgZGFzaFBoYXNlKSB7Ci0gICAgICAgIGlmICh3aWR0aCA8IDAuMGYpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xMzM9TmVnYXRpdmUgd2lkdGgKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTMzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKGNhcCAhPSBDQVBfQlVUVCAmJiBjYXAgIT0gQ0FQX1JPVU5EICYmIGNhcCAhPSBDQVBfU1FVQVJFKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTM0PUlsbGVnYWwgY2FwCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEzNCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGlmIChqb2luICE9IEpPSU5fTUlURVIgJiYgam9pbiAhPSBKT0lOX1JPVU5EICYmIGpvaW4gIT0gSk9JTl9CRVZFTCkgewotICAgICAgICAgICAgLy8gYXd0LjEzNT1JbGxlZ2FsIGpvaW4KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTM1IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKGpvaW4gPT0gSk9JTl9NSVRFUiAmJiBtaXRlckxpbWl0IDwgMS4wZikgewotICAgICAgICAgICAgLy8gYXd0LjEzNj1taXRlckxpbWl0IGxlc3MgdGhhbiAxLjBmCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEzNiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGlmIChkYXNoICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChkYXNoUGhhc2UgPCAwLjBmKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjEzNz1OZWdhdGl2ZSBkYXNoUGhhc2UKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEzNyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGRhc2gubGVuZ3RoID09IDApIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMTM4PVplcm8gZGFzaCBsZW5ndGgKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEzOCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgWkVSTzogewotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgZGFzaC5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBpZiAoZGFzaFtpXSA8IDAuMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjEzOT1OZWdhdGl2ZSBkYXNoW3swfV0KLSAgICAgICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTM5IiwgaSkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWYgKGRhc2hbaV0gPiAwLjApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrIFpFUk87Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjEzQT1BbGwgZGFzaCBsZW5ndGhzIHplcm8KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEzQSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHRoaXMud2lkdGggPSB3aWR0aDsKLSAgICAgICAgdGhpcy5jYXAgPSBjYXA7Ci0gICAgICAgIHRoaXMuam9pbiA9IGpvaW47Ci0gICAgICAgIHRoaXMubWl0ZXJMaW1pdCA9IG1pdGVyTGltaXQ7Ci0gICAgICAgIHRoaXMuZGFzaCA9IGRhc2g7Ci0gICAgICAgIHRoaXMuZGFzaFBoYXNlID0gZGFzaFBoYXNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBCYXNpY1N0cm9rZSB3aXRoIHNwZWNpZmllZCB3aWR0aCwgY2FwLCBqb2luLCBsaW1pdCBhbmQKLSAgICAgKiBkZWZhdWx0IGRhc2ggYXR0cmlidXRlcyBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIEJhc2lrU3Ryb2tlLgotICAgICAqIEBwYXJhbSBjYXAKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbmQgZGVjb3JhdGlvbiBvZiBCYXNpa1N0cm9rZS4KLSAgICAgKiBAcGFyYW0gam9pbgotICAgICAqICAgICAgICAgICAgdGhlIGpvaW4gc2VnbWVudHMgZGVjb3JhdGlvbi4KLSAgICAgKiBAcGFyYW0gbWl0ZXJMaW1pdAotICAgICAqICAgICAgICAgICAgdGhlIGxpbWl0IHRvIHRyaW0gdGhlIG1pdGVyIGpvaW4uCi0gICAgICovCi0gICAgcHVibGljIEJhc2ljU3Ryb2tlKGZsb2F0IHdpZHRoLCBpbnQgY2FwLCBpbnQgam9pbiwgZmxvYXQgbWl0ZXJMaW1pdCkgewotICAgICAgICB0aGlzKHdpZHRoLCBjYXAsIGpvaW4sIG1pdGVyTGltaXQsIG51bGwsIDAuMGYpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBCYXNpY1N0cm9rZSB3aXRoIHNwZWNpZmllZCB3aWR0aCwgY2FwLCBqb2luIGFuZAotICAgICAqIGRlZmF1bHQgbGltaXQgYW5kIGRhc2ggYXR0cmlidXRlcyBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIEJhc2lrU3Ryb2tlLgotICAgICAqIEBwYXJhbSBjYXAKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbmQgZGVjb3JhdGlvbiBvZiBCYXNpa1N0cm9rZS4KLSAgICAgKiBAcGFyYW0gam9pbgotICAgICAqICAgICAgICAgICAgdGhlIGpvaW4gc2VnbWVudHMgZGVjb3JhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgQmFzaWNTdHJva2UoZmxvYXQgd2lkdGgsIGludCBjYXAsIGludCBqb2luKSB7Ci0gICAgICAgIHRoaXMod2lkdGgsIGNhcCwgam9pbiwgMTAuMGYsIG51bGwsIDAuMGYpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBCYXNpY1N0cm9rZSB3aXRoIHNwZWNpZmllZCB3aWR0aCBhbmQgZGVmYXVsdCBjYXAsCi0gICAgICogam9pbiwgbGltaXQsIGRhc2ggYXR0cmlidXRlcyBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIEJhc2ljU3Ryb2tlLgotICAgICAqLwotICAgIHB1YmxpYyBCYXNpY1N0cm9rZShmbG9hdCB3aWR0aCkgewotICAgICAgICB0aGlzKHdpZHRoLCBDQVBfU1FVQVJFLCBKT0lOX01JVEVSLCAxMC4wZiwgbnVsbCwgMC4wZik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbGluZSB3aWR0aCBvZiB0aGUgQmFzaWNTdHJva2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbGluZSB3aWR0aCBvZiB0aGUgQmFzaWNTdHJva2UuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldExpbmVXaWR0aCgpIHsKLSAgICAgICAgcmV0dXJuIHdpZHRoOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGVuZCBjYXAgc3R5bGUgb2YgdGhlIEJhc2ljU3Ryb2tlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGVuZCBjYXAgc3R5bGUgb2YgdGhlIEJhc2ljU3Ryb2tlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0RW5kQ2FwKCkgewotICAgICAgICByZXR1cm4gY2FwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxpbmUgam9pbiBzdHlsZSBvZiB0aGUgQmFzaWNTdHJva2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbGluZSBqb2luIHN0eWxlIG9mIHRoZSBCYXNpY1N0cm9rZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldExpbmVKb2luKCkgewotICAgICAgICByZXR1cm4gam9pbjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtaXRlciBsaW1pdCBvZiB0aGUgQmFzaWNTdHJva2UgKHRoZSBsaW1pdCB0byB0cmltIHRoZSBtaXRlcgotICAgICAqIGpvaW4pLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG1pdGVyIGxpbWl0IG9mIHRoZSBCYXNpY1N0cm9rZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0TWl0ZXJMaW1pdCgpIHsKLSAgICAgICAgcmV0dXJuIG1pdGVyTGltaXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGFzaCBhdHRyaWJ1dGVzIGFycmF5IG9mIHRoZSBCYXNpY1N0cm9rZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkYXNoIGF0dHJpYnV0ZXMgYXJyYXkgb2YgdGhlIEJhc2ljU3Ryb2tlLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldERhc2hBcnJheSgpIHsKLSAgICAgICAgcmV0dXJuIGRhc2g7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGFzaCBwaGFzZSBvZiB0aGUgQmFzaWNTdHJva2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGFzaCBwaGFzZSBvZiB0aGUgQmFzaWNTdHJva2UuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldERhc2hQaGFzZSgpIHsKLSAgICAgICAgcmV0dXJuIGRhc2hQaGFzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGhhc2ggY29kZSBvZiB0aGlzIEJhc2ljU3Ryb2tlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhhc2ggY29kZSBvZiB0aGlzIEJhc2ljU3Ryb2tlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgaGFzaENvZGUoKSB7Ci0gICAgICAgIEhhc2hDb2RlIGhhc2ggPSBuZXcgSGFzaENvZGUoKTsKLSAgICAgICAgaGFzaC5hcHBlbmQod2lkdGgpOwotICAgICAgICBoYXNoLmFwcGVuZChjYXApOwotICAgICAgICBoYXNoLmFwcGVuZChqb2luKTsKLSAgICAgICAgaGFzaC5hcHBlbmQobWl0ZXJMaW1pdCk7Ci0gICAgICAgIGlmIChkYXNoICE9IG51bGwpIHsKLSAgICAgICAgICAgIGhhc2guYXBwZW5kKGRhc2hQaGFzZSk7Ci0gICAgICAgICAgICBmb3IgKGZsb2F0IGVsZW1lbnQgOiBkYXNoKSB7Ci0gICAgICAgICAgICAgICAgaGFzaC5hcHBlbmQoZWxlbWVudCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGhhc2guaGFzaENvZGUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wYXJlcyB0aGlzIEJhc2ljU3Ryb2tlIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgT2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgT2JqZWN0IGlzIGEgQmFzaWNTdHJva2Ugd2l0aCB0aGUgc2FtZSBkYXRhIHZhbHVlcyBhcwotICAgICAqICAgICAgICAgdGhpcyBCYXNpY1N0cm9rZTsgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3Qgb2JqKSB7Ci0gICAgICAgIGlmIChvYmogPT0gdGhpcykgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG9iaiBpbnN0YW5jZW9mIEJhc2ljU3Ryb2tlKSB7Ci0gICAgICAgICAgICBCYXNpY1N0cm9rZSBicyA9IChCYXNpY1N0cm9rZSlvYmo7Ci0gICAgICAgICAgICByZXR1cm4gYnMud2lkdGggPT0gd2lkdGggJiYgYnMuY2FwID09IGNhcCAmJiBicy5qb2luID09IGpvaW4KLSAgICAgICAgICAgICAgICAgICAgJiYgYnMubWl0ZXJMaW1pdCA9PSBtaXRlckxpbWl0ICYmIGJzLmRhc2hQaGFzZSA9PSBkYXNoUGhhc2UKLSAgICAgICAgICAgICAgICAgICAgJiYgamF2YS51dGlsLkFycmF5cy5lcXVhbHMoYnMuZGFzaCwgZGFzaCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENhbGN1bGF0ZXMgYWxsb3dhYmxlIGN1cnZlIGRlcml2YXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGguCi0gICAgICogQHJldHVybiB0aGUgY3VydmUgZGVsdGEuCi0gICAgICovCi0gICAgZG91YmxlIGdldEN1cnZlRGVsdGEoZG91YmxlIHdpZHRoKSB7Ci0gICAgICAgIGRvdWJsZSBhID0gd2lkdGggKyBDVVJWRV9ERUxUQTsKLSAgICAgICAgZG91YmxlIGNvcyA9IDEuMCAtIDIuMCAqIHdpZHRoICogd2lkdGggLyAoYSAqIGEpOwotICAgICAgICBkb3VibGUgc2luID0gTWF0aC5zcXJ0KDEuMCAtIGNvcyAqIGNvcyk7Ci0gICAgICAgIHJldHVybiBNYXRoLmFicyhzaW4gLyBjb3MpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENhbGN1bGF0ZXMgdGhlIHZhbHVlIHRvIGRldGVjdCBhIHNtYWxsIGFuZ2xlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoLgotICAgICAqIEByZXR1cm4gdGhlIGNvcm5lciBkZWx0YS4KLSAgICAgKi8KLSAgICBkb3VibGUgZ2V0Q29ybmVyRGVsdGEoZG91YmxlIHdpZHRoKSB7Ci0gICAgICAgIHJldHVybiB3aWR0aCAqIHdpZHRoICogTWF0aC5zaW4oTWF0aC5QSSAqIENPUk5FUl9BTkdMRSAvIDE4MC4wKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDYWxjdWxhdGVzIHZhbHVlIHRvIGRldGVjdCBhIHplcm8gYW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGguCi0gICAgICogQHJldHVybiB0aGUgemVybyBkZWx0YS4KLSAgICAgKi8KLSAgICBkb3VibGUgZ2V0WmVyb0RlbHRhKGRvdWJsZSB3aWR0aCkgewotICAgICAgICByZXR1cm4gd2lkdGggKiB3aWR0aCAqIE1hdGguc2luKE1hdGguUEkgKiBDT1JORVJfWkVSTyAvIDE4MC4wKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgU2hhcGUgZnJvbSB0aGUgb3V0bGluZSBvZiB0aGUgc3BlY2lmaWVkIHNoYXBlIGRyYXduIHdpdGggdGhpcwotICAgICAqIEJhc2ljU3Ryb2tlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIFNoYXBlIHRvIGJlIHN0cm9rZWQuCi0gICAgICogQHJldHVybiB0aGUgU2hhcGUgb2YgdGhlIHN0cm9rZWQgb3V0bGluZS4KLSAgICAgKiBAc2VlIGphdmEuYXd0LlN0cm9rZSNjcmVhdGVTdHJva2VkU2hhcGUoamF2YS5hd3QuU2hhcGUpCi0gICAgICovCi0gICAgcHVibGljIFNoYXBlIGNyZWF0ZVN0cm9rZWRTaGFwZShTaGFwZSBzKSB7Ci0gICAgICAgIHcyID0gd2lkdGggLyAyLjA7Ci0gICAgICAgIGN1cnZlRGVsdGEgPSBnZXRDdXJ2ZURlbHRhKHcyKTsKLSAgICAgICAgY29ybmVyRGVsdGEgPSBnZXRDb3JuZXJEZWx0YSh3Mik7Ci0gICAgICAgIHplcm9EZWx0YSA9IGdldFplcm9EZWx0YSh3Mik7Ci0KLSAgICAgICAgZHN0ID0gbmV3IEJ1ZmZlcmVkUGF0aCgpOwotICAgICAgICBscCA9IG5ldyBCdWZmZXJlZFBhdGgoKTsKLSAgICAgICAgcnAgPSBuZXcgQnVmZmVyZWRQYXRoKCk7Ci0KLSAgICAgICAgaWYgKGRhc2ggPT0gbnVsbCkgewotICAgICAgICAgICAgY3JlYXRlU29saWRTaGFwZShzLmdldFBhdGhJdGVyYXRvcihudWxsKSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBjcmVhdGVEYXNoZWRTaGFwZShzLmdldFBhdGhJdGVyYXRvcihudWxsKSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZHN0LmNyZWF0ZUdlbmVyYWxQYXRoKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2VuZXJhdGVzIGEgc2hhcGUgd2l0aCBhIHNvbGlkIChub3QgZGFzaGVkKSBvdXRsaW5lLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgUGF0aEl0ZXJhdG9yIG9mIHNvdXJjZSBzaGFwZS4KLSAgICAgKi8KLSAgICB2b2lkIGNyZWF0ZVNvbGlkU2hhcGUoUGF0aEl0ZXJhdG9yIHApIHsKLSAgICAgICAgZG91YmxlIGNvb3Jkc1tdID0gbmV3IGRvdWJsZVs2XTsKLSAgICAgICAgbXggPSBteSA9IGN4ID0gY3kgPSAwLjA7Ci0gICAgICAgIGlzTW92ZSA9IGZhbHNlOwotICAgICAgICBpc0ZpcnN0ID0gZmFsc2U7Ci0gICAgICAgIGNoZWNrTW92ZSA9IHRydWU7Ci0gICAgICAgIGJvb2xlYW4gaXNDbG9zZWQgPSB0cnVlOwotCi0gICAgICAgIHdoaWxlICghcC5pc0RvbmUoKSkgewotICAgICAgICAgICAgc3dpdGNoIChwLmN1cnJlbnRTZWdtZW50KGNvb3JkcykpIHsKLSAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfTU9WRVRPOgotICAgICAgICAgICAgICAgICAgICBpZiAoIWlzQ2xvc2VkKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjbG9zZVNvbGlkU2hhcGUoKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBycC5jbGVhbigpOwotICAgICAgICAgICAgICAgICAgICBteCA9IGN4ID0gY29vcmRzWzBdOwotICAgICAgICAgICAgICAgICAgICBteSA9IGN5ID0gY29vcmRzWzFdOwotICAgICAgICAgICAgICAgICAgICBpc01vdmUgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICBpc0Nsb3NlZCA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfTElORVRPOgotICAgICAgICAgICAgICAgICAgICBhZGRMaW5lKGN4LCBjeSwgY3ggPSBjb29yZHNbMF0sIGN5ID0gY29vcmRzWzFdLCB0cnVlKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX1FVQURUTzoKLSAgICAgICAgICAgICAgICAgICAgYWRkUXVhZChjeCwgY3ksIGNvb3Jkc1swXSwgY29vcmRzWzFdLCBjeCA9IGNvb3Jkc1syXSwgY3kgPSBjb29yZHNbM10pOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfQ1VCSUNUTzoKLSAgICAgICAgICAgICAgICAgICAgYWRkQ3ViaWMoY3gsIGN5LCBjb29yZHNbMF0sIGNvb3Jkc1sxXSwgY29vcmRzWzJdLCBjb29yZHNbM10sIGN4ID0gY29vcmRzWzRdLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGN5ID0gY29vcmRzWzVdKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0NMT1NFOgotICAgICAgICAgICAgICAgICAgICBhZGRMaW5lKGN4LCBjeSwgbXgsIG15LCBmYWxzZSk7Ci0gICAgICAgICAgICAgICAgICAgIGFkZEpvaW4obHAsIG14LCBteSwgbHAueE1vdmUsIGxwLnlNb3ZlLCB0cnVlKTsKLSAgICAgICAgICAgICAgICAgICAgYWRkSm9pbihycCwgbXgsIG15LCBycC54TW92ZSwgcnAueU1vdmUsIGZhbHNlKTsKLSAgICAgICAgICAgICAgICAgICAgbHAuY2xvc2VQYXRoKCk7Ci0gICAgICAgICAgICAgICAgICAgIHJwLmNsb3NlUGF0aCgpOwotICAgICAgICAgICAgICAgICAgICBscC5hcHBlbmRSZXZlcnNlKHJwKTsKLSAgICAgICAgICAgICAgICAgICAgaXNDbG9zZWQgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHAubmV4dCgpOwotICAgICAgICB9Ci0gICAgICAgIGlmICghaXNDbG9zZWQpIHsKLSAgICAgICAgICAgIGNsb3NlU29saWRTaGFwZSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgZHN0ID0gbHA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2xvc2VzIHNvbGlkIHNoYXBlIHBhdGguCi0gICAgICovCi0gICAgdm9pZCBjbG9zZVNvbGlkU2hhcGUoKSB7Ci0gICAgICAgIGFkZENhcChscCwgY3gsIGN5LCBycC54TGFzdCwgcnAueUxhc3QpOwotICAgICAgICBscC5jb21iaW5lKHJwKTsKLSAgICAgICAgYWRkQ2FwKGxwLCBteCwgbXksIGxwLnhNb3ZlLCBscC55TW92ZSk7Ci0gICAgICAgIGxwLmNsb3NlUGF0aCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdlbmVyYXRlcyBkYXNoZWQgc3Ryb2tlZCBzaGFwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIFBhdGhJdGVyYXRvciBvZiBzb3VyY2Ugc2hhcGUuCi0gICAgICovCi0gICAgdm9pZCBjcmVhdGVEYXNoZWRTaGFwZShQYXRoSXRlcmF0b3IgcCkgewotICAgICAgICBkb3VibGUgY29vcmRzW10gPSBuZXcgZG91YmxlWzZdOwotICAgICAgICBteCA9IG15ID0gY3ggPSBjeSA9IDAuMDsKLSAgICAgICAgc214ID0gc215ID0gc2N4ID0gc2N5ID0gMC4wOwotICAgICAgICBpc01vdmUgPSBmYWxzZTsKLSAgICAgICAgY2hlY2tNb3ZlID0gZmFsc2U7Ci0gICAgICAgIGJvb2xlYW4gaXNDbG9zZWQgPSB0cnVlOwotCi0gICAgICAgIHdoaWxlICghcC5pc0RvbmUoKSkgewotICAgICAgICAgICAgc3dpdGNoIChwLmN1cnJlbnRTZWdtZW50KGNvb3JkcykpIHsKLSAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfTU9WRVRPOgotCi0gICAgICAgICAgICAgICAgICAgIGlmICghaXNDbG9zZWQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNsb3NlRGFzaGVkU2hhcGUoKTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGRhc2hlciA9IG5ldyBEYXNoZXIoZGFzaCwgZGFzaFBoYXNlKTsKLSAgICAgICAgICAgICAgICAgICAgbHAuY2xlYW4oKTsKLSAgICAgICAgICAgICAgICAgICAgcnAuY2xlYW4oKTsKLSAgICAgICAgICAgICAgICAgICAgc3AgPSBudWxsOwotICAgICAgICAgICAgICAgICAgICBpc0ZpcnN0ID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgaXNNb3ZlID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgaXNDbG9zZWQgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICAgICAgbXggPSBjeCA9IGNvb3Jkc1swXTsKLSAgICAgICAgICAgICAgICAgICAgbXkgPSBjeSA9IGNvb3Jkc1sxXTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0xJTkVUTzoKLSAgICAgICAgICAgICAgICAgICAgYWRkRGFzaExpbmUoY3gsIGN5LCBjeCA9IGNvb3Jkc1swXSwgY3kgPSBjb29yZHNbMV0pOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfUVVBRFRPOgotICAgICAgICAgICAgICAgICAgICBhZGREYXNoUXVhZChjeCwgY3ksIGNvb3Jkc1swXSwgY29vcmRzWzFdLCBjeCA9IGNvb3Jkc1syXSwgY3kgPSBjb29yZHNbM10pOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfQ1VCSUNUTzoKLSAgICAgICAgICAgICAgICAgICAgYWRkRGFzaEN1YmljKGN4LCBjeSwgY29vcmRzWzBdLCBjb29yZHNbMV0sIGNvb3Jkc1syXSwgY29vcmRzWzNdLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGN4ID0gY29vcmRzWzRdLCBjeSA9IGNvb3Jkc1s1XSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19DTE9TRToKLSAgICAgICAgICAgICAgICAgICAgYWRkRGFzaExpbmUoY3gsIGN5LCBjeCA9IG14LCBjeSA9IG15KTsKLQotICAgICAgICAgICAgICAgICAgICBpZiAoZGFzaGVyLmlzQ29ubmVjdGVkKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIENvbm5lY3QgY3VycmVudCBhbmQgaGVhZCBzZWdtZW50cwotICAgICAgICAgICAgICAgICAgICAgICAgYWRkSm9pbihscCwgZm14LCBmbXksIHNwLnhNb3ZlLCBzcC55TW92ZSwgdHJ1ZSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBscC5qb2luKHNwKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGFkZEpvaW4obHAsIGZteCwgZm15LCBycC54TGFzdCwgcnAueUxhc3QsIHRydWUpOwotICAgICAgICAgICAgICAgICAgICAgICAgbHAuY29tYmluZShycCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBhZGRDYXAobHAsIHNteCwgc215LCBscC54TW92ZSwgbHAueU1vdmUpOwotICAgICAgICAgICAgICAgICAgICAgICAgbHAuY2xvc2VQYXRoKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkc3QuYXBwZW5kKGxwKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHNwID0gbnVsbDsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNsb3NlRGFzaGVkU2hhcGUoKTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGlzQ2xvc2VkID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwLm5leHQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICghaXNDbG9zZWQpIHsKLSAgICAgICAgICAgIGNsb3NlRGFzaGVkU2hhcGUoKTsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2xvc2VzIGRhc2hlZCBzaGFwZSBwYXRoLgotICAgICAqLwotICAgIHZvaWQgY2xvc2VEYXNoZWRTaGFwZSgpIHsKLSAgICAgICAgLy8gQWRkIGhlYWQgc2VnbWVudAotICAgICAgICBpZiAoc3AgIT0gbnVsbCkgewotICAgICAgICAgICAgYWRkQ2FwKHNwLCBmbXgsIGZteSwgc3AueE1vdmUsIHNwLnlNb3ZlKTsKLSAgICAgICAgICAgIHNwLmNsb3NlUGF0aCgpOwotICAgICAgICAgICAgZHN0LmFwcGVuZChzcCk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGxwLnR5cGVTaXplID4gMCkgewotICAgICAgICAgICAgLy8gQ2xvc2UgY3VycmVudCBzZWdtZW50Ci0gICAgICAgICAgICBpZiAoIWRhc2hlci5pc0Nsb3NlZCgpKSB7Ci0gICAgICAgICAgICAgICAgYWRkQ2FwKGxwLCBzY3gsIHNjeSwgcnAueExhc3QsIHJwLnlMYXN0KTsKLSAgICAgICAgICAgICAgICBscC5jb21iaW5lKHJwKTsKLSAgICAgICAgICAgICAgICBhZGRDYXAobHAsIHNteCwgc215LCBscC54TW92ZSwgbHAueU1vdmUpOwotICAgICAgICAgICAgICAgIGxwLmNsb3NlUGF0aCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZHN0LmFwcGVuZChscCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGNhcCB0byB0aGUgd29yayBwYXRoLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgQnVmZmVyZWRQYXRoIG9iamVjdCBvZiB3b3JrIHBhdGguCi0gICAgICogQHBhcmFtIHgwCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzb3VyY2UgcGF0aC4KLSAgICAgKiBAcGFyYW0geTAKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb24gdGhlIHNvdXJjZSBwYXRoLgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgbmV4dCBwb2ludCBvbiB0aGUgd29yayBwYXRoLgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgbmV4dCBwb2ludCBvbiB0aGUgd29yayBwYXRoLgotICAgICAqLwotICAgIHZvaWQgYWRkQ2FwKEJ1ZmZlcmVkUGF0aCBwLCBkb3VibGUgeDAsIGRvdWJsZSB5MCwgZG91YmxlIHgyLCBkb3VibGUgeTIpIHsKLSAgICAgICAgZG91YmxlIHgxID0gcC54TGFzdDsKLSAgICAgICAgZG91YmxlIHkxID0gcC55TGFzdDsKLSAgICAgICAgZG91YmxlIHgxMCA9IHgxIC0geDA7Ci0gICAgICAgIGRvdWJsZSB5MTAgPSB5MSAtIHkwOwotICAgICAgICBkb3VibGUgeDIwID0geDIgLSB4MDsKLSAgICAgICAgZG91YmxlIHkyMCA9IHkyIC0geTA7Ci0KLSAgICAgICAgc3dpdGNoIChjYXApIHsKLSAgICAgICAgICAgIGNhc2UgQ0FQX0JVVFQ6Ci0gICAgICAgICAgICAgICAgcC5saW5lVG8oeDIsIHkyKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgQ0FQX1JPVU5EOgotICAgICAgICAgICAgICAgIGRvdWJsZSBteCA9IHgxMCAqIENVQklDX0FSQzsKLSAgICAgICAgICAgICAgICBkb3VibGUgbXkgPSB5MTAgKiBDVUJJQ19BUkM7Ci0KLSAgICAgICAgICAgICAgICBkb3VibGUgeDMgPSB4MCArIHkxMDsKLSAgICAgICAgICAgICAgICBkb3VibGUgeTMgPSB5MCAtIHgxMDsKLQotICAgICAgICAgICAgICAgIHgxMCAqPSBDVUJJQ19BUkM7Ci0gICAgICAgICAgICAgICAgeTEwICo9IENVQklDX0FSQzsKLSAgICAgICAgICAgICAgICB4MjAgKj0gQ1VCSUNfQVJDOwotICAgICAgICAgICAgICAgIHkyMCAqPSBDVUJJQ19BUkM7Ci0KLSAgICAgICAgICAgICAgICBwLmN1YmljVG8oeDEgKyB5MTAsIHkxIC0geDEwLCB4MyArIG14LCB5MyArIG15LCB4MywgeTMpOwotICAgICAgICAgICAgICAgIHAuY3ViaWNUbyh4MyAtIG14LCB5MyAtIG15LCB4MiAtIHkyMCwgeTIgKyB4MjAsIHgyLCB5Mik7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIENBUF9TUVVBUkU6Ci0gICAgICAgICAgICAgICAgcC5saW5lVG8oeDEgKyB5MTAsIHkxIC0geDEwKTsKLSAgICAgICAgICAgICAgICBwLmxpbmVUbyh4MiAtIHkyMCwgeTIgKyB4MjApOwotICAgICAgICAgICAgICAgIHAubGluZVRvKHgyLCB5Mik7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGJldmVsIGFuZCBtaXRlciBqb2luIHRvIHRoZSB3b3JrIHBhdGguCi0gICAgICogCi0gICAgICogQHBhcmFtIHAKLSAgICAgKiAgICAgICAgICAgIHRoZSBCdWZmZXJlZFBhdGggb2JqZWN0IG9mIHdvcmsgcGF0aC4KLSAgICAgKiBAcGFyYW0geDAKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHNvdXJjZSBwYXRoLgotICAgICAqIEBwYXJhbSB5MAotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvbiB0aGUgc291cmNlIHBhdGguCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBuZXh0IHBvaW50IG9uIHRoZSB3b3JrIHBhdGguCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBuZXh0IHBvaW50IG9uIHRoZSB3b3JrIHBhdGguCi0gICAgICogQHBhcmFtIGlzTGVmdAotICAgICAqICAgICAgICAgICAgdGhlIG9yaWVudGF0aW9uIG9mIHdvcmsgcGF0aCwgdHJ1ZSBpZiB3b3JrIHBhdGggbGllcyB0byB0aGUKLSAgICAgKiAgICAgICAgICAgIGxlZnQgZnJvbSBzb3VyY2UgcGF0aCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHZvaWQgYWRkSm9pbihCdWZmZXJlZFBhdGggcCwgZG91YmxlIHgwLCBkb3VibGUgeTAsIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBib29sZWFuIGlzTGVmdCkgewotICAgICAgICBkb3VibGUgeDEgPSBwLnhMYXN0OwotICAgICAgICBkb3VibGUgeTEgPSBwLnlMYXN0OwotICAgICAgICBkb3VibGUgeDEwID0geDEgLSB4MDsKLSAgICAgICAgZG91YmxlIHkxMCA9IHkxIC0geTA7Ci0gICAgICAgIGRvdWJsZSB4MjAgPSB4MiAtIHgwOwotICAgICAgICBkb3VibGUgeTIwID0geTIgLSB5MDsKLSAgICAgICAgZG91YmxlIHNpbjAgPSB4MTAgKiB5MjAgLSB5MTAgKiB4MjA7Ci0KLSAgICAgICAgLy8gU21hbGwgY29ybmVyCi0gICAgICAgIGlmICgtY29ybmVyRGVsdGEgPCBzaW4wICYmIHNpbjAgPCBjb3JuZXJEZWx0YSkgewotICAgICAgICAgICAgZG91YmxlIGNvczAgPSB4MTAgKiB4MjAgKyB5MTAgKiB5MjA7Ci0gICAgICAgICAgICBpZiAoY29zMCA+IDAuMCkgewotICAgICAgICAgICAgICAgIC8vIGlmIHplcm8gY29ybmVyIGRvIG5vdGhpbmcKLSAgICAgICAgICAgICAgICBpZiAoLXplcm9EZWx0YSA+IHNpbjAgfHwgc2luMCA+IHplcm9EZWx0YSkgewotICAgICAgICAgICAgICAgICAgICBkb3VibGUgeDMgPSB4MCArIHcyICogdzIgKiAoeTIwIC0geTEwKSAvIHNpbjA7Ci0gICAgICAgICAgICAgICAgICAgIGRvdWJsZSB5MyA9IHkwICsgdzIgKiB3MiAqICh4MTAgLSB4MjApIC8gc2luMDsKLSAgICAgICAgICAgICAgICAgICAgcC5zZXRMYXN0KHgzLCB5Myk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIC8vIFplcm8gY29ybmVyCi0gICAgICAgICAgICBpZiAoLXplcm9EZWx0YSA8IHNpbjAgJiYgc2luMCA8IHplcm9EZWx0YSkgewotICAgICAgICAgICAgICAgIHAubGluZVRvKHgyLCB5Mik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoaXNMZWZ0IF4gKHNpbjAgPCAwLjApKSB7Ci0gICAgICAgICAgICAvLyBUd2lzdGVkIGNvcm5lcgotICAgICAgICAgICAgcC5saW5lVG8oeDAsIHkwKTsKLSAgICAgICAgICAgIHAubGluZVRvKHgyLCB5Mik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBzd2l0Y2ggKGpvaW4pIHsKLSAgICAgICAgICAgICAgICBjYXNlIEpPSU5fQkVWRUw6Ci0gICAgICAgICAgICAgICAgICAgIHAubGluZVRvKHgyLCB5Mik7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgSk9JTl9NSVRFUjoKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHMxID0geDEgKiB4MTAgKyB5MSAqIHkxMDsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHMyID0geDIgKiB4MjAgKyB5MiAqIHkyMDsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHgzID0gKHMxICogeTIwIC0gczIgKiB5MTApIC8gc2luMDsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHkzID0gKHMyICogeDEwIC0gczEgKiB4MjApIC8gc2luMDsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHgzMCA9IHgzIC0geDA7Ci0gICAgICAgICAgICAgICAgICAgIGRvdWJsZSB5MzAgPSB5MyAtIHkwOwotICAgICAgICAgICAgICAgICAgICBkb3VibGUgbWl0ZXJMZW5ndGggPSBNYXRoLnNxcnQoeDMwICogeDMwICsgeTMwICogeTMwKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKG1pdGVyTGVuZ3RoIDwgbWl0ZXJMaW1pdCAqIHcyKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwLmxpbmVUbyh4MywgeTMpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHAubGluZVRvKHgyLCB5Mik7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgSk9JTl9ST1VORDoKLSAgICAgICAgICAgICAgICAgICAgYWRkUm91bmRKb2luKHAsIHgwLCB5MCwgeDIsIHkyLCBpc0xlZnQpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgcm91bmQgam9pbiB0byB0aGUgd29yayBwYXRoLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgQnVmZmVyZWRQYXRoIG9iamVjdCBvZiB3b3JrIHBhdGguCi0gICAgICogQHBhcmFtIHgwCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzb3VyY2UgcGF0aC4KLSAgICAgKiBAcGFyYW0geTAKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb24gdGhlIHNvdXJjZSBwYXRoLgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgbmV4dCBwb2ludCBvbiB0aGUgd29yayBwYXRoLgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgbmV4dCBwb2ludCBvbiB0aGUgd29yayBwYXRoLgotICAgICAqIEBwYXJhbSBpc0xlZnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvcmllbnRhdGlvbiBvZiB3b3JrIHBhdGgsIHRydWUgaWYgd29yayBwYXRoIGxpZXMgdG8gdGhlCi0gICAgICogICAgICAgICAgICBsZWZ0IGZyb20gc291cmNlIHBhdGgsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICB2b2lkIGFkZFJvdW5kSm9pbihCdWZmZXJlZFBhdGggcCwgZG91YmxlIHgwLCBkb3VibGUgeTAsIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBib29sZWFuIGlzTGVmdCkgewotICAgICAgICBkb3VibGUgeDEgPSBwLnhMYXN0OwotICAgICAgICBkb3VibGUgeTEgPSBwLnlMYXN0OwotICAgICAgICBkb3VibGUgeDEwID0geDEgLSB4MDsKLSAgICAgICAgZG91YmxlIHkxMCA9IHkxIC0geTA7Ci0gICAgICAgIGRvdWJsZSB4MjAgPSB4MiAtIHgwOwotICAgICAgICBkb3VibGUgeTIwID0geTIgLSB5MDsKLQotICAgICAgICBkb3VibGUgeDMwID0geDEwICsgeDIwOwotICAgICAgICBkb3VibGUgeTMwID0geTEwICsgeTIwOwotCi0gICAgICAgIGRvdWJsZSBsMzAgPSBNYXRoLnNxcnQoeDMwICogeDMwICsgeTMwICogeTMwKTsKLQotICAgICAgICBpZiAobDMwIDwgMUUtNSkgewotICAgICAgICAgICAgcC5saW5lVG8oeDIsIHkyKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGRvdWJsZSB3ID0gdzIgLyBsMzA7Ci0KLSAgICAgICAgeDMwICo9IHc7Ci0gICAgICAgIHkzMCAqPSB3OwotCi0gICAgICAgIGRvdWJsZSB4MyA9IHgwICsgeDMwOwotICAgICAgICBkb3VibGUgeTMgPSB5MCArIHkzMDsKLQotICAgICAgICBkb3VibGUgY29zID0geDEwICogeDIwICsgeTEwICogeTIwOwotICAgICAgICBkb3VibGUgYSA9IE1hdGguYWNvcyhjb3MgLyAodzIgKiB3MikpOwotICAgICAgICBpZiAoY29zID49IDAuMCkgewotICAgICAgICAgICAgZG91YmxlIGsgPSA0LjAgLyAzLjAgKiBNYXRoLnRhbihhIC8gNC4wKTsKLSAgICAgICAgICAgIGlmIChpc0xlZnQpIHsKLSAgICAgICAgICAgICAgICBrID0gLWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHgxMCAqPSBrOwotICAgICAgICAgICAgeTEwICo9IGs7Ci0gICAgICAgICAgICB4MjAgKj0gazsKLSAgICAgICAgICAgIHkyMCAqPSBrOwotCi0gICAgICAgICAgICBwLmN1YmljVG8oeDEgLSB5MTAsIHkxICsgeDEwLCB4MiArIHkyMCwgeTIgLSB4MjAsIHgyLCB5Mik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBkb3VibGUgayA9IDQuMCAvIDMuMCAqIE1hdGgudGFuKGEgLyA4LjApOwotICAgICAgICAgICAgaWYgKGlzTGVmdCkgewotICAgICAgICAgICAgICAgIGsgPSAtazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgeDEwICo9IGs7Ci0gICAgICAgICAgICB5MTAgKj0gazsKLSAgICAgICAgICAgIHgyMCAqPSBrOwotICAgICAgICAgICAgeTIwICo9IGs7Ci0gICAgICAgICAgICB4MzAgKj0gazsKLSAgICAgICAgICAgIHkzMCAqPSBrOwotCi0gICAgICAgICAgICBwLmN1YmljVG8oeDEgLSB5MTAsIHkxICsgeDEwLCB4MyArIHkzMCwgeTMgLSB4MzAsIHgzLCB5Myk7Ci0gICAgICAgICAgICBwLmN1YmljVG8oeDMgLSB5MzAsIHkzICsgeDMwLCB4MiArIHkyMCwgeTIgLSB4MjAsIHgyLCB5Mik7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgc29saWQgbGluZSBzZWdtZW50IHRvIHRoZSB3b3JrIHBhdGguCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydCBsaW5lIHBvaW50LgotICAgICAqIEBwYXJhbSB5MQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnQgbGluZSBwb2ludC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBsaW5lIHBvaW50LgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIGxpbmUgcG9pbnQuCi0gICAgICogQHBhcmFtIHplcm8KLSAgICAgKiAgICAgICAgICAgIGlmIHRydWUgaXQncyBhbGxvd2FibGUgdG8gYWRkIHplcm8gbGVuZ3RoIGxpbmUgc2VnbWVudC4KLSAgICAgKi8KLSAgICB2b2lkIGFkZExpbmUoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBib29sZWFuIHplcm8pIHsKLSAgICAgICAgZG91YmxlIGR4ID0geDIgLSB4MTsKLSAgICAgICAgZG91YmxlIGR5ID0geTIgLSB5MTsKLQotICAgICAgICBpZiAoZHggPT0gMC4wICYmIGR5ID09IDAuMCkgewotICAgICAgICAgICAgaWYgKCF6ZXJvKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZHggPSB3MjsKLSAgICAgICAgICAgIGR5ID0gMDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGRvdWJsZSB3ID0gdzIgLyBNYXRoLnNxcnQoZHggKiBkeCArIGR5ICogZHkpOwotICAgICAgICAgICAgZHggKj0gdzsKLSAgICAgICAgICAgIGR5ICo9IHc7Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgbHgxID0geDEgLSBkeTsKLSAgICAgICAgZG91YmxlIGx5MSA9IHkxICsgZHg7Ci0gICAgICAgIGRvdWJsZSByeDEgPSB4MSArIGR5OwotICAgICAgICBkb3VibGUgcnkxID0geTEgLSBkeDsKLQotICAgICAgICBpZiAoY2hlY2tNb3ZlKSB7Ci0gICAgICAgICAgICBpZiAoaXNNb3ZlKSB7Ci0gICAgICAgICAgICAgICAgaXNNb3ZlID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgbHAubW92ZVRvKGx4MSwgbHkxKTsKLSAgICAgICAgICAgICAgICBycC5tb3ZlVG8ocngxLCByeTEpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBhZGRKb2luKGxwLCB4MSwgeTEsIGx4MSwgbHkxLCB0cnVlKTsKLSAgICAgICAgICAgICAgICBhZGRKb2luKHJwLCB4MSwgeTEsIHJ4MSwgcnkxLCBmYWxzZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBscC5saW5lVG8oeDIgLSBkeSwgeTIgKyBkeCk7Ci0gICAgICAgIHJwLmxpbmVUbyh4MiArIGR5LCB5MiAtIGR4KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHNvbGlkIHF1YWQgc2VnbWVudCB0byB0aGUgd29yayBwYXRoLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB4MwotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdGhpcmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTMKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHRoaXJkIGNvbnRyb2wgcG9pbnQuCi0gICAgICovCi0gICAgdm9pZCBhZGRRdWFkKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MiwgZG91YmxlIHgzLCBkb3VibGUgeTMpIHsKLSAgICAgICAgZG91YmxlIHgyMSA9IHgyIC0geDE7Ci0gICAgICAgIGRvdWJsZSB5MjEgPSB5MiAtIHkxOwotICAgICAgICBkb3VibGUgeDIzID0geDIgLSB4MzsKLSAgICAgICAgZG91YmxlIHkyMyA9IHkyIC0geTM7Ci0KLSAgICAgICAgZG91YmxlIGwyMSA9IE1hdGguc3FydCh4MjEgKiB4MjEgKyB5MjEgKiB5MjEpOwotICAgICAgICBkb3VibGUgbDIzID0gTWF0aC5zcXJ0KHgyMyAqIHgyMyArIHkyMyAqIHkyMyk7Ci0KLSAgICAgICAgaWYgKGwyMSA9PSAwLjAgJiYgbDIzID09IDAuMCkgewotICAgICAgICAgICAgYWRkTGluZSh4MSwgeTEsIHgzLCB5MywgZmFsc2UpOwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGwyMSA9PSAwLjApIHsKLSAgICAgICAgICAgIGFkZExpbmUoeDIsIHkyLCB4MywgeTMsIGZhbHNlKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChsMjMgPT0gMC4wKSB7Ci0gICAgICAgICAgICBhZGRMaW5lKHgxLCB5MSwgeDIsIHkyLCBmYWxzZSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgdzsKLSAgICAgICAgdyA9IHcyIC8gbDIxOwotICAgICAgICBkb3VibGUgbXgxID0gLXkyMSAqIHc7Ci0gICAgICAgIGRvdWJsZSBteTEgPSB4MjEgKiB3OwotICAgICAgICB3ID0gdzIgLyBsMjM7Ci0gICAgICAgIGRvdWJsZSBteDMgPSB5MjMgKiB3OwotICAgICAgICBkb3VibGUgbXkzID0gLXgyMyAqIHc7Ci0KLSAgICAgICAgZG91YmxlIGx4MSA9IHgxICsgbXgxOwotICAgICAgICBkb3VibGUgbHkxID0geTEgKyBteTE7Ci0gICAgICAgIGRvdWJsZSByeDEgPSB4MSAtIG14MTsKLSAgICAgICAgZG91YmxlIHJ5MSA9IHkxIC0gbXkxOwotCi0gICAgICAgIGlmIChjaGVja01vdmUpIHsKLSAgICAgICAgICAgIGlmIChpc01vdmUpIHsKLSAgICAgICAgICAgICAgICBpc01vdmUgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICBscC5tb3ZlVG8obHgxLCBseTEpOwotICAgICAgICAgICAgICAgIHJwLm1vdmVUbyhyeDEsIHJ5MSk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGFkZEpvaW4obHAsIHgxLCB5MSwgbHgxLCBseTEsIHRydWUpOwotICAgICAgICAgICAgICAgIGFkZEpvaW4ocnAsIHgxLCB5MSwgcngxLCByeTEsIGZhbHNlKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmICh4MjEgKiB5MjMgLSB5MjEgKiB4MjMgPT0gMC4wKSB7Ci0gICAgICAgICAgICAvLyBPbiBsaW5lIGN1cnZlCi0gICAgICAgICAgICBpZiAoeDIxICogeDIzICsgeTIxICogeTIzID4gMC4wKSB7Ci0gICAgICAgICAgICAgICAgLy8gVHdpc3RlZCBjdXJ2ZQotICAgICAgICAgICAgICAgIGlmIChsMjEgPT0gbDIzKSB7Ci0gICAgICAgICAgICAgICAgICAgIGRvdWJsZSBweCA9IHgxICsgKHgyMSArIHgyMykgLyA0LjA7Ci0gICAgICAgICAgICAgICAgICAgIGRvdWJsZSBweSA9IHkxICsgKHkyMSArIHkyMykgLyA0LjA7Ci0gICAgICAgICAgICAgICAgICAgIGxwLmxpbmVUbyhweCArIG14MSwgcHkgKyBteTEpOwotICAgICAgICAgICAgICAgICAgICBycC5saW5lVG8ocHggLSBteDEsIHB5IC0gbXkxKTsKLSAgICAgICAgICAgICAgICAgICAgbHAubGluZVRvKHB4IC0gbXgxLCBweSAtIG15MSk7Ci0gICAgICAgICAgICAgICAgICAgIHJwLmxpbmVUbyhweCArIG14MSwgcHkgKyBteTEpOwotICAgICAgICAgICAgICAgICAgICBscC5saW5lVG8oeDMgLSBteDEsIHkzIC0gbXkxKTsKLSAgICAgICAgICAgICAgICAgICAgcnAubGluZVRvKHgzICsgbXgxLCB5MyArIG15MSk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHB4MSwgcHkxOwotICAgICAgICAgICAgICAgICAgICBkb3VibGUgayA9IGwyMSAvIChsMjEgKyBsMjMpOwotICAgICAgICAgICAgICAgICAgICBkb3VibGUgcHggPSB4MSArICh4MjEgKyB4MjMpICogayAqIGs7Ci0gICAgICAgICAgICAgICAgICAgIGRvdWJsZSBweSA9IHkxICsgKHkyMSArIHkyMykgKiBrICogazsKLSAgICAgICAgICAgICAgICAgICAgcHgxID0gKHgxICsgcHgpIC8gMi4wOwotICAgICAgICAgICAgICAgICAgICBweTEgPSAoeTEgKyBweSkgLyAyLjA7Ci0gICAgICAgICAgICAgICAgICAgIGxwLnF1YWRUbyhweDEgKyBteDEsIHB5MSArIG15MSwgcHggKyBteDEsIHB5ICsgbXkxKTsKLSAgICAgICAgICAgICAgICAgICAgcnAucXVhZFRvKHB4MSAtIG14MSwgcHkxIC0gbXkxLCBweCAtIG14MSwgcHkgLSBteTEpOwotICAgICAgICAgICAgICAgICAgICBscC5saW5lVG8ocHggLSBteDEsIHB5IC0gbXkxKTsKLSAgICAgICAgICAgICAgICAgICAgcnAubGluZVRvKHB4ICsgbXgxLCBweSArIG15MSk7Ci0gICAgICAgICAgICAgICAgICAgIHB4MSA9ICh4MyArIHB4KSAvIDIuMDsKLSAgICAgICAgICAgICAgICAgICAgcHkxID0gKHkzICsgcHkpIC8gMi4wOwotICAgICAgICAgICAgICAgICAgICBscC5xdWFkVG8ocHgxIC0gbXgxLCBweTEgLSBteTEsIHgzIC0gbXgxLCB5MyAtIG15MSk7Ci0gICAgICAgICAgICAgICAgICAgIHJwLnF1YWRUbyhweDEgKyBteDEsIHB5MSArIG15MSwgeDMgKyBteDEsIHkzICsgbXkxKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIC8vIFNpbXBsZSBjdXJ2ZQotICAgICAgICAgICAgICAgIGxwLnF1YWRUbyh4MiArIG14MSwgeTIgKyBteTEsIHgzICsgbXgzLCB5MyArIG15Myk7Ci0gICAgICAgICAgICAgICAgcnAucXVhZFRvKHgyIC0gbXgxLCB5MiAtIG15MSwgeDMgLSBteDMsIHkzIC0gbXkzKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGFkZFN1YlF1YWQoeDEsIHkxLCB4MiwgeTIsIHgzLCB5MywgMCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTdWJkaXZpZGVzIHNvbGlkIHF1YWQgY3VydmUgdG8gbWFrZSBvdXRsaW5lIGZvciBzb3VyY2UgcXVhZCBzZWdtZW50IGFuZAotICAgICAqIGFkZHMgaXQgdG8gd29yayBwYXRoLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB4MwotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdGhpcmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTMKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHRoaXJkIGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIGxldmVsCi0gICAgICogICAgICAgICAgICB0aGUgbWF4aW11bSBsZXZlbCBvZiBzdWJkaXZpc2lvbiBkZWVwbmVzcy4KLSAgICAgKi8KLSAgICB2b2lkIGFkZFN1YlF1YWQoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBkb3VibGUgeDMsIGRvdWJsZSB5MywgaW50IGxldmVsKSB7Ci0gICAgICAgIGRvdWJsZSB4MjEgPSB4MiAtIHgxOwotICAgICAgICBkb3VibGUgeTIxID0geTIgLSB5MTsKLSAgICAgICAgZG91YmxlIHgyMyA9IHgyIC0geDM7Ci0gICAgICAgIGRvdWJsZSB5MjMgPSB5MiAtIHkzOwotCi0gICAgICAgIGRvdWJsZSBjb3MgPSB4MjEgKiB4MjMgKyB5MjEgKiB5MjM7Ci0gICAgICAgIGRvdWJsZSBzaW4gPSB4MjEgKiB5MjMgLSB5MjEgKiB4MjM7Ci0KLSAgICAgICAgaWYgKGxldmVsIDwgTUFYX0xFVkVMICYmIChjb3MgPj0gMC4wIHx8IChNYXRoLmFicyhzaW4gLyBjb3MpID4gY3VydmVEZWx0YSkpKSB7Ci0gICAgICAgICAgICBkb3VibGUgYzF4ID0gKHgyICsgeDEpIC8gMi4wOwotICAgICAgICAgICAgZG91YmxlIGMxeSA9ICh5MiArIHkxKSAvIDIuMDsKLSAgICAgICAgICAgIGRvdWJsZSBjMnggPSAoeDIgKyB4MykgLyAyLjA7Ci0gICAgICAgICAgICBkb3VibGUgYzJ5ID0gKHkyICsgeTMpIC8gMi4wOwotICAgICAgICAgICAgZG91YmxlIGMzeCA9IChjMXggKyBjMngpIC8gMi4wOwotICAgICAgICAgICAgZG91YmxlIGMzeSA9IChjMXkgKyBjMnkpIC8gMi4wOwotICAgICAgICAgICAgYWRkU3ViUXVhZCh4MSwgeTEsIGMxeCwgYzF5LCBjM3gsIGMzeSwgbGV2ZWwgKyAxKTsKLSAgICAgICAgICAgIGFkZFN1YlF1YWQoYzN4LCBjM3ksIGMyeCwgYzJ5LCB4MywgeTMsIGxldmVsICsgMSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBkb3VibGUgdzsKLSAgICAgICAgICAgIGRvdWJsZSBsMjEgPSBNYXRoLnNxcnQoeDIxICogeDIxICsgeTIxICogeTIxKTsKLSAgICAgICAgICAgIGRvdWJsZSBsMjMgPSBNYXRoLnNxcnQoeDIzICogeDIzICsgeTIzICogeTIzKTsKLSAgICAgICAgICAgIHcgPSB3MiAvIHNpbjsKLSAgICAgICAgICAgIGRvdWJsZSBteDIgPSAoeDIxICogbDIzICsgeDIzICogbDIxKSAqIHc7Ci0gICAgICAgICAgICBkb3VibGUgbXkyID0gKHkyMSAqIGwyMyArIHkyMyAqIGwyMSkgKiB3OwotICAgICAgICAgICAgdyA9IHcyIC8gbDIzOwotICAgICAgICAgICAgZG91YmxlIG14MyA9IHkyMyAqIHc7Ci0gICAgICAgICAgICBkb3VibGUgbXkzID0gLXgyMyAqIHc7Ci0gICAgICAgICAgICBscC5xdWFkVG8oeDIgKyBteDIsIHkyICsgbXkyLCB4MyArIG14MywgeTMgKyBteTMpOwotICAgICAgICAgICAgcnAucXVhZFRvKHgyIC0gbXgyLCB5MiAtIG15MiwgeDMgLSBteDMsIHkzIC0gbXkzKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgc29saWQgY3ViaWMgc2VnbWVudCB0byB0aGUgd29yayBwYXRoLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB4MwotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdGhpcmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTMKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHRoaXJkIGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHg0Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBmb3VycyBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB5NAotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZm91cnMgY29udHJvbCBwb2ludC4KLSAgICAgKi8KLSAgICB2b2lkIGFkZEN1YmljKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MiwgZG91YmxlIHgzLCBkb3VibGUgeTMsIGRvdWJsZSB4NCwKLSAgICAgICAgICAgIGRvdWJsZSB5NCkgewotICAgICAgICBkb3VibGUgeDEyID0geDEgLSB4MjsKLSAgICAgICAgZG91YmxlIHkxMiA9IHkxIC0geTI7Ci0gICAgICAgIGRvdWJsZSB4MjMgPSB4MiAtIHgzOwotICAgICAgICBkb3VibGUgeTIzID0geTIgLSB5MzsKLSAgICAgICAgZG91YmxlIHgzNCA9IHgzIC0geDQ7Ci0gICAgICAgIGRvdWJsZSB5MzQgPSB5MyAtIHk0OwotCi0gICAgICAgIGRvdWJsZSBsMTIgPSBNYXRoLnNxcnQoeDEyICogeDEyICsgeTEyICogeTEyKTsKLSAgICAgICAgZG91YmxlIGwyMyA9IE1hdGguc3FydCh4MjMgKiB4MjMgKyB5MjMgKiB5MjMpOwotICAgICAgICBkb3VibGUgbDM0ID0gTWF0aC5zcXJ0KHgzNCAqIHgzNCArIHkzNCAqIHkzNCk7Ci0KLSAgICAgICAgLy8gQWxsIGVkZ2VzIGFyZSB6ZXJvCi0gICAgICAgIGlmIChsMTIgPT0gMC4wICYmIGwyMyA9PSAwLjAgJiYgbDM0ID09IDAuMCkgewotICAgICAgICAgICAgYWRkTGluZSh4MSwgeTEsIHg0LCB5NCwgZmFsc2UpOwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gT25lIHplcm8gZWRnZQotICAgICAgICBpZiAobDEyID09IDAuMCAmJiBsMjMgPT0gMC4wKSB7Ci0gICAgICAgICAgICBhZGRMaW5lKHgzLCB5MywgeDQsIHk0LCBmYWxzZSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAobDIzID09IDAuMCAmJiBsMzQgPT0gMC4wKSB7Ci0gICAgICAgICAgICBhZGRMaW5lKHgxLCB5MSwgeDIsIHkyLCBmYWxzZSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAobDEyID09IDAuMCAmJiBsMzQgPT0gMC4wKSB7Ci0gICAgICAgICAgICBhZGRMaW5lKHgyLCB5MiwgeDMsIHkzLCBmYWxzZSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgdywgbXgxLCBteTEsIG14NCwgbXk0OwotICAgICAgICBib29sZWFuIG9uTGluZTsKLQotICAgICAgICBpZiAobDEyID09IDAuMCkgewotICAgICAgICAgICAgdyA9IHcyIC8gbDIzOwotICAgICAgICAgICAgbXgxID0geTIzICogdzsKLSAgICAgICAgICAgIG15MSA9IC14MjMgKiB3OwotICAgICAgICAgICAgdyA9IHcyIC8gbDM0OwotICAgICAgICAgICAgbXg0ID0geTM0ICogdzsKLSAgICAgICAgICAgIG15NCA9IC14MzQgKiB3OwotICAgICAgICAgICAgb25MaW5lID0gLXgyMyAqIHkzNCArIHkyMyAqIHgzNCA9PSAwLjA7IC8vIHNpbjMKLSAgICAgICAgfSBlbHNlIGlmIChsMzQgPT0gMC4wKSB7Ci0gICAgICAgICAgICB3ID0gdzIgLyBsMTI7Ci0gICAgICAgICAgICBteDEgPSB5MTIgKiB3OwotICAgICAgICAgICAgbXkxID0gLXgxMiAqIHc7Ci0gICAgICAgICAgICB3ID0gdzIgLyBsMjM7Ci0gICAgICAgICAgICBteDQgPSB5MjMgKiB3OwotICAgICAgICAgICAgbXk0ID0gLXgyMyAqIHc7Ci0gICAgICAgICAgICBvbkxpbmUgPSAteDEyICogeTIzICsgeTEyICogeDIzID09IDAuMDsgLy8gc2luMgotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdyA9IHcyIC8gbDEyOwotICAgICAgICAgICAgbXgxID0geTEyICogdzsKLSAgICAgICAgICAgIG15MSA9IC14MTIgKiB3OwotICAgICAgICAgICAgdyA9IHcyIC8gbDM0OwotICAgICAgICAgICAgbXg0ID0geTM0ICogdzsKLSAgICAgICAgICAgIG15NCA9IC14MzQgKiB3OwotICAgICAgICAgICAgaWYgKGwyMyA9PSAwLjApIHsKLSAgICAgICAgICAgICAgICBvbkxpbmUgPSAteDEyICogeTM0ICsgeTEyICogeDM0ID09IDAuMDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgb25MaW5lID0gLXgxMiAqIHkzNCArIHkxMiAqIHgzNCA9PSAwLjAgJiYgLXgxMiAqIHkyMyArIHkxMiAqIHgyMyA9PSAwLjAgJiYgLy8gc2luMgotICAgICAgICAgICAgICAgICAgICAgICAgLXgyMyAqIHkzNCArIHkyMyAqIHgzNCA9PSAwLjA7IC8vIHNpbjMKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGRvdWJsZSBseDEgPSB4MSArIG14MTsKLSAgICAgICAgZG91YmxlIGx5MSA9IHkxICsgbXkxOwotICAgICAgICBkb3VibGUgcngxID0geDEgLSBteDE7Ci0gICAgICAgIGRvdWJsZSByeTEgPSB5MSAtIG15MTsKLQotICAgICAgICBpZiAoY2hlY2tNb3ZlKSB7Ci0gICAgICAgICAgICBpZiAoaXNNb3ZlKSB7Ci0gICAgICAgICAgICAgICAgaXNNb3ZlID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgbHAubW92ZVRvKGx4MSwgbHkxKTsKLSAgICAgICAgICAgICAgICBycC5tb3ZlVG8ocngxLCByeTEpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBhZGRKb2luKGxwLCB4MSwgeTEsIGx4MSwgbHkxLCB0cnVlKTsKLSAgICAgICAgICAgICAgICBhZGRKb2luKHJwLCB4MSwgeTEsIHJ4MSwgcnkxLCBmYWxzZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAob25MaW5lKSB7Ci0gICAgICAgICAgICBpZiAoKHgxID09IHgyICYmIHkxIDwgeTIpIHx8IHgxIDwgeDIpIHsKLSAgICAgICAgICAgICAgICBsMTIgPSAtbDEyOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKCh4MiA9PSB4MyAmJiB5MiA8IHkzKSB8fCB4MiA8IHgzKSB7Ci0gICAgICAgICAgICAgICAgbDIzID0gLWwyMzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICgoeDMgPT0geDQgJiYgeTMgPCB5NCkgfHwgeDMgPCB4NCkgewotICAgICAgICAgICAgICAgIGwzNCA9IC1sMzQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkb3VibGUgZCA9IGwyMyAqIGwyMyAtIGwxMiAqIGwzNDsKLSAgICAgICAgICAgIGRvdWJsZSByb290c1tdID0gbmV3IGRvdWJsZVszXTsKLSAgICAgICAgICAgIGludCByYyA9IDA7Ci0gICAgICAgICAgICBpZiAoZCA9PSAwLjApIHsKLSAgICAgICAgICAgICAgICBkb3VibGUgdCA9IChsMTIgLSBsMjMpIC8gKGwxMiArIGwzNCAtIGwyMyAtIGwyMyk7Ci0gICAgICAgICAgICAgICAgaWYgKDAuMCA8IHQgJiYgdCA8IDEuMCkgewotICAgICAgICAgICAgICAgICAgICByb290c1tyYysrXSA9IHQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIGlmIChkID4gMC4wKSB7Ci0gICAgICAgICAgICAgICAgZCA9IE1hdGguc3FydChkKTsKLSAgICAgICAgICAgICAgICBkb3VibGUgeiA9IGwxMiArIGwzNCAtIGwyMyAtIGwyMzsKLSAgICAgICAgICAgICAgICBkb3VibGUgdDsKLSAgICAgICAgICAgICAgICB0ID0gKGwxMiAtIGwyMyArIGQpIC8gejsKLSAgICAgICAgICAgICAgICBpZiAoMC4wIDwgdCAmJiB0IDwgMS4wKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJvb3RzW3JjKytdID0gdDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgdCA9IChsMTIgLSBsMjMgLSBkKSAvIHo7Ci0gICAgICAgICAgICAgICAgaWYgKDAuMCA8IHQgJiYgdCA8IDEuMCkgewotICAgICAgICAgICAgICAgICAgICByb290c1tyYysrXSA9IHQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAocmMgPiAwKSB7Ci0gICAgICAgICAgICAgICAgLy8gU29ydCByb290cwotICAgICAgICAgICAgICAgIGlmIChyYyA9PSAyICYmIHJvb3RzWzBdID4gcm9vdHNbMV0pIHsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHRtcCA9IHJvb3RzWzBdOwotICAgICAgICAgICAgICAgICAgICByb290c1swXSA9IHJvb3RzWzFdOwotICAgICAgICAgICAgICAgICAgICByb290c1sxXSA9IHRtcDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcm9vdHNbcmMrK10gPSAxLjA7Ci0KLSAgICAgICAgICAgICAgICBkb3VibGUgYXggPSAteDM0IC0geDEyICsgeDIzICsgeDIzOwotICAgICAgICAgICAgICAgIGRvdWJsZSBheSA9IC15MzQgLSB5MTIgKyB5MjMgKyB5MjM7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGJ4ID0gMy4wICogKC14MjMgKyB4MTIpOwotICAgICAgICAgICAgICAgIGRvdWJsZSBieSA9IDMuMCAqICgteTIzICsgeTEyKTsKLSAgICAgICAgICAgICAgICBkb3VibGUgY3ggPSAzLjAgKiAoLXgxMik7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGN5ID0gMy4wICogKC15MTIpOwotICAgICAgICAgICAgICAgIGRvdWJsZSB4UHJldiA9IHgxOwotICAgICAgICAgICAgICAgIGRvdWJsZSB5UHJldiA9IHkxOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcmM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBkb3VibGUgdCA9IHJvb3RzW2ldOwotICAgICAgICAgICAgICAgICAgICBkb3VibGUgcHggPSB0ICogKHQgKiAodCAqIGF4ICsgYngpICsgY3gpICsgeDE7Ci0gICAgICAgICAgICAgICAgICAgIGRvdWJsZSBweSA9IHQgKiAodCAqICh0ICogYXkgKyBieSkgKyBjeSkgKyB5MTsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHB4MSA9ICh4UHJldiArIHB4KSAvIDIuMDsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHB5MSA9ICh5UHJldiArIHB5KSAvIDIuMDsKLSAgICAgICAgICAgICAgICAgICAgbHAuY3ViaWNUbyhweDEgKyBteDEsIHB5MSArIG15MSwgcHgxICsgbXgxLCBweTEgKyBteTEsIHB4ICsgbXgxLCBweSArIG15MSk7Ci0gICAgICAgICAgICAgICAgICAgIHJwLmN1YmljVG8ocHgxIC0gbXgxLCBweTEgLSBteTEsIHB4MSAtIG14MSwgcHkxIC0gbXkxLCBweCAtIG14MSwgcHkgLSBteTEpOwotICAgICAgICAgICAgICAgICAgICBpZiAoaSA8IHJjIC0gMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgbHAubGluZVRvKHB4IC0gbXgxLCBweSAtIG15MSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBycC5saW5lVG8ocHggKyBteDEsIHB5ICsgbXkxKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB4UHJldiA9IHB4OwotICAgICAgICAgICAgICAgICAgICB5UHJldiA9IHB5OwotICAgICAgICAgICAgICAgICAgICBteDEgPSAtbXgxOwotICAgICAgICAgICAgICAgICAgICBteTEgPSAtbXkxOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgbHAuY3ViaWNUbyh4MiArIG14MSwgeTIgKyBteTEsIHgzICsgbXg0LCB5MyArIG15NCwgeDQgKyBteDQsIHk0ICsgbXk0KTsKLSAgICAgICAgICAgICAgICBycC5jdWJpY1RvKHgyIC0gbXgxLCB5MiAtIG15MSwgeDMgLSBteDQsIHkzIC0gbXk0LCB4NCAtIG14NCwgeTQgLSBteTQpOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgYWRkU3ViQ3ViaWMoeDEsIHkxLCB4MiwgeTIsIHgzLCB5MywgeDQsIHk0LCAwKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFN1YmRpdmlkZXMgc29saWQgY3ViaWMgY3VydmUgdG8gbWFrZSBvdXRsaW5lIGZvciBzb3VyY2UgcXVhZCBzZWdtZW50IGFuZAotICAgICAqIGFkZHMgaXQgdG8gd29yayBwYXRoLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB4MwotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdGhpcmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTMKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHRoaXJkIGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHg0Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBmb3VycyBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB5NAotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZm91cnMgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0gbGV2ZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBtYXhpbXVtIGxldmVsIG9mIHN1YmRpdmlzaW9uIGRlZXBuZXNzLgotICAgICAqLwotICAgIHZvaWQgYWRkU3ViQ3ViaWMoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBkb3VibGUgeDMsIGRvdWJsZSB5MywgZG91YmxlIHg0LAotICAgICAgICAgICAgZG91YmxlIHk0LCBpbnQgbGV2ZWwpIHsKLSAgICAgICAgZG91YmxlIHgxMiA9IHgxIC0geDI7Ci0gICAgICAgIGRvdWJsZSB5MTIgPSB5MSAtIHkyOwotICAgICAgICBkb3VibGUgeDIzID0geDIgLSB4MzsKLSAgICAgICAgZG91YmxlIHkyMyA9IHkyIC0geTM7Ci0gICAgICAgIGRvdWJsZSB4MzQgPSB4MyAtIHg0OwotICAgICAgICBkb3VibGUgeTM0ID0geTMgLSB5NDsKLQotICAgICAgICBkb3VibGUgY29zMiA9IC14MTIgKiB4MjMgLSB5MTIgKiB5MjM7Ci0gICAgICAgIGRvdWJsZSBjb3MzID0gLXgyMyAqIHgzNCAtIHkyMyAqIHkzNDsKLSAgICAgICAgZG91YmxlIHNpbjIgPSAteDEyICogeTIzICsgeTEyICogeDIzOwotICAgICAgICBkb3VibGUgc2luMyA9IC14MjMgKiB5MzQgKyB5MjMgKiB4MzQ7Ci0gICAgICAgIGRvdWJsZSBzaW4wID0gLXgxMiAqIHkzNCArIHkxMiAqIHgzNDsKLSAgICAgICAgZG91YmxlIGNvczAgPSAteDEyICogeDM0IC0geTEyICogeTM0OwotCi0gICAgICAgIGlmIChsZXZlbCA8IE1BWF9MRVZFTAotICAgICAgICAgICAgICAgICYmIChzaW4yICE9IDAuMCB8fCBzaW4zICE9IDAuMCB8fCBzaW4wICE9IDAuMCkKLSAgICAgICAgICAgICAgICAmJiAoY29zMiA+PSAwLjAgfHwgY29zMyA+PSAwLjAgfHwgY29zMCA+PSAwLjAKLSAgICAgICAgICAgICAgICAgICAgICAgIHx8IChNYXRoLmFicyhzaW4yIC8gY29zMikgPiBjdXJ2ZURlbHRhKQotICAgICAgICAgICAgICAgICAgICAgICAgfHwgKE1hdGguYWJzKHNpbjMgLyBjb3MzKSA+IGN1cnZlRGVsdGEpIHx8IChNYXRoLmFicyhzaW4wIC8gY29zMCkgPiBjdXJ2ZURlbHRhKSkpIHsKLSAgICAgICAgICAgIGRvdWJsZSBjeCA9ICh4MiArIHgzKSAvIDIuMDsKLSAgICAgICAgICAgIGRvdWJsZSBjeSA9ICh5MiArIHkzKSAvIDIuMDsKLSAgICAgICAgICAgIGRvdWJsZSBseDIgPSAoeDIgKyB4MSkgLyAyLjA7Ci0gICAgICAgICAgICBkb3VibGUgbHkyID0gKHkyICsgeTEpIC8gMi4wOwotICAgICAgICAgICAgZG91YmxlIHJ4MyA9ICh4MyArIHg0KSAvIDIuMDsKLSAgICAgICAgICAgIGRvdWJsZSByeTMgPSAoeTMgKyB5NCkgLyAyLjA7Ci0gICAgICAgICAgICBkb3VibGUgbHgzID0gKGN4ICsgbHgyKSAvIDIuMDsKLSAgICAgICAgICAgIGRvdWJsZSBseTMgPSAoY3kgKyBseTIpIC8gMi4wOwotICAgICAgICAgICAgZG91YmxlIHJ4MiA9IChjeCArIHJ4MykgLyAyLjA7Ci0gICAgICAgICAgICBkb3VibGUgcnkyID0gKGN5ICsgcnkzKSAvIDIuMDsKLSAgICAgICAgICAgIGN4ID0gKGx4MyArIHJ4MikgLyAyLjA7Ci0gICAgICAgICAgICBjeSA9IChseTMgKyByeTIpIC8gMi4wOwotICAgICAgICAgICAgYWRkU3ViQ3ViaWMoeDEsIHkxLCBseDIsIGx5MiwgbHgzLCBseTMsIGN4LCBjeSwgbGV2ZWwgKyAxKTsKLSAgICAgICAgICAgIGFkZFN1YkN1YmljKGN4LCBjeSwgcngyLCByeTIsIHJ4MywgcnkzLCB4NCwgeTQsIGxldmVsICsgMSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBkb3VibGUgdywgbXgxLCBteTEsIG14MiwgbXkyLCBteDMsIG15MywgbXg0LCBteTQ7Ci0gICAgICAgICAgICBkb3VibGUgbDEyID0gTWF0aC5zcXJ0KHgxMiAqIHgxMiArIHkxMiAqIHkxMik7Ci0gICAgICAgICAgICBkb3VibGUgbDIzID0gTWF0aC5zcXJ0KHgyMyAqIHgyMyArIHkyMyAqIHkyMyk7Ci0gICAgICAgICAgICBkb3VibGUgbDM0ID0gTWF0aC5zcXJ0KHgzNCAqIHgzNCArIHkzNCAqIHkzNCk7Ci0KLSAgICAgICAgICAgIGlmIChsMTIgPT0gMC4wKSB7Ci0gICAgICAgICAgICAgICAgdyA9IHcyIC8gbDIzOwotICAgICAgICAgICAgICAgIG14MSA9IHkyMyAqIHc7Ci0gICAgICAgICAgICAgICAgbXkxID0gLXgyMyAqIHc7Ci0gICAgICAgICAgICAgICAgdyA9IHcyIC8gbDM0OwotICAgICAgICAgICAgICAgIG14NCA9IHkzNCAqIHc7Ci0gICAgICAgICAgICAgICAgbXk0ID0gLXgzNCAqIHc7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGwzNCA9PSAwLjApIHsKLSAgICAgICAgICAgICAgICB3ID0gdzIgLyBsMTI7Ci0gICAgICAgICAgICAgICAgbXgxID0geTEyICogdzsKLSAgICAgICAgICAgICAgICBteTEgPSAteDEyICogdzsKLSAgICAgICAgICAgICAgICB3ID0gdzIgLyBsMjM7Ci0gICAgICAgICAgICAgICAgbXg0ID0geTIzICogdzsKLSAgICAgICAgICAgICAgICBteTQgPSAteDIzICogdzsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgLy8gQ29tbW9uIGNhc2UKLSAgICAgICAgICAgICAgICB3ID0gdzIgLyBsMTI7Ci0gICAgICAgICAgICAgICAgbXgxID0geTEyICogdzsKLSAgICAgICAgICAgICAgICBteTEgPSAteDEyICogdzsKLSAgICAgICAgICAgICAgICB3ID0gdzIgLyBsMzQ7Ci0gICAgICAgICAgICAgICAgbXg0ID0geTM0ICogdzsKLSAgICAgICAgICAgICAgICBteTQgPSAteDM0ICogdzsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKHNpbjIgPT0gMC4wKSB7Ci0gICAgICAgICAgICAgICAgbXgyID0gbXgxOwotICAgICAgICAgICAgICAgIG15MiA9IG15MTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgdyA9IHcyIC8gc2luMjsKLSAgICAgICAgICAgICAgICBteDIgPSAtKHgxMiAqIGwyMyAtIHgyMyAqIGwxMikgKiB3OwotICAgICAgICAgICAgICAgIG15MiA9IC0oeTEyICogbDIzIC0geTIzICogbDEyKSAqIHc7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoc2luMyA9PSAwLjApIHsKLSAgICAgICAgICAgICAgICBteDMgPSBteDQ7Ci0gICAgICAgICAgICAgICAgbXkzID0gbXk0OwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICB3ID0gdzIgLyBzaW4zOwotICAgICAgICAgICAgICAgIG14MyA9IC0oeDIzICogbDM0IC0geDM0ICogbDIzKSAqIHc7Ci0gICAgICAgICAgICAgICAgbXkzID0gLSh5MjMgKiBsMzQgLSB5MzQgKiBsMjMpICogdzsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgbHAuY3ViaWNUbyh4MiArIG14MiwgeTIgKyBteTIsIHgzICsgbXgzLCB5MyArIG15MywgeDQgKyBteDQsIHk0ICsgbXk0KTsKLSAgICAgICAgICAgIHJwLmN1YmljVG8oeDIgLSBteDIsIHkyIC0gbXkyLCB4MyAtIG14MywgeTMgLSBteTMsIHg0IC0gbXg0LCB5NCAtIG15NCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGRhc2hlZCBsaW5lIHNlZ21lbnQgdG8gdGhlIHdvcmsgcGF0aC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0IGxpbmUgcG9pbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydCBsaW5lIHBvaW50LgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIGxpbmUgcG9pbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgbGluZSBwb2ludC4KLSAgICAgKi8KLSAgICB2b2lkIGFkZERhc2hMaW5lKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MikgewotICAgICAgICBkb3VibGUgeDIxID0geDIgLSB4MTsKLSAgICAgICAgZG91YmxlIHkyMSA9IHkyIC0geTE7Ci0KLSAgICAgICAgZG91YmxlIGwyMSA9IE1hdGguc3FydCh4MjEgKiB4MjEgKyB5MjEgKiB5MjEpOwotCi0gICAgICAgIGlmIChsMjEgPT0gMC4wKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgcHgxLCBweTE7Ci0gICAgICAgIHB4MSA9IHB5MSA9IDAuMDsKLSAgICAgICAgZG91YmxlIHcgPSB3MiAvIGwyMTsKLSAgICAgICAgZG91YmxlIG14ID0gLXkyMSAqIHc7Ci0gICAgICAgIGRvdWJsZSBteSA9IHgyMSAqIHc7Ci0KLSAgICAgICAgZGFzaGVyLmluaXQobmV3IERhc2hJdGVyYXRvci5MaW5lKGwyMSkpOwotCi0gICAgICAgIHdoaWxlICghZGFzaGVyLmVvZigpKSB7Ci0gICAgICAgICAgICBkb3VibGUgdCA9IGRhc2hlci5nZXRWYWx1ZSgpOwotICAgICAgICAgICAgc2N4ID0geDEgKyB0ICogeDIxOwotICAgICAgICAgICAgc2N5ID0geTEgKyB0ICogeTIxOwotCi0gICAgICAgICAgICBpZiAoZGFzaGVyLmlzT3BlbigpKSB7Ci0gICAgICAgICAgICAgICAgcHgxID0gc2N4OwotICAgICAgICAgICAgICAgIHB5MSA9IHNjeTsKLSAgICAgICAgICAgICAgICBkb3VibGUgbHgxID0gcHgxICsgbXg7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGx5MSA9IHB5MSArIG15OwotICAgICAgICAgICAgICAgIGRvdWJsZSByeDEgPSBweDEgLSBteDsKLSAgICAgICAgICAgICAgICBkb3VibGUgcnkxID0gcHkxIC0gbXk7Ci0gICAgICAgICAgICAgICAgaWYgKGlzTW92ZSkgewotICAgICAgICAgICAgICAgICAgICBpc01vdmUgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICAgICAgc214ID0gcHgxOwotICAgICAgICAgICAgICAgICAgICBzbXkgPSBweTE7Ci0gICAgICAgICAgICAgICAgICAgIHJwLmNsZWFuKCk7Ci0gICAgICAgICAgICAgICAgICAgIGxwLm1vdmVUbyhseDEsIGx5MSk7Ci0gICAgICAgICAgICAgICAgICAgIHJwLm1vdmVUbyhyeDEsIHJ5MSk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgYWRkSm9pbihscCwgeDEsIHkxLCBseDEsIGx5MSwgdHJ1ZSk7Ci0gICAgICAgICAgICAgICAgICAgIGFkZEpvaW4ocnAsIHgxLCB5MSwgcngxLCByeTEsIGZhbHNlKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGRhc2hlci5pc0NvbnRpbnVlKCkpIHsKLSAgICAgICAgICAgICAgICBkb3VibGUgcHgyID0gc2N4OwotICAgICAgICAgICAgICAgIGRvdWJsZSBweTIgPSBzY3k7Ci0gICAgICAgICAgICAgICAgbHAubGluZVRvKHB4MiArIG14LCBweTIgKyBteSk7Ci0gICAgICAgICAgICAgICAgcnAubGluZVRvKHB4MiAtIG14LCBweTIgLSBteSk7Ci0gICAgICAgICAgICAgICAgaWYgKGRhc2hlci5jbG9zZSkgewotICAgICAgICAgICAgICAgICAgICBhZGRDYXAobHAsIHB4MiwgcHkyLCBycC54TGFzdCwgcnAueUxhc3QpOwotICAgICAgICAgICAgICAgICAgICBscC5jb21iaW5lKHJwKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGlzRmlyc3QpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlzRmlyc3QgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZteCA9IHNteDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZteSA9IHNteTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHNwID0gbHA7Ci0gICAgICAgICAgICAgICAgICAgICAgICBscCA9IG5ldyBCdWZmZXJlZFBhdGgoKTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGFkZENhcChscCwgc214LCBzbXksIGxwLnhNb3ZlLCBscC55TW92ZSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBscC5jbG9zZVBhdGgoKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpc01vdmUgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZGFzaGVyLm5leHQoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgZGFzaGVkIHF1YWQgc2VnbWVudCB0byB0aGUgd29yayBwYXRoLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB4MwotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdGhpcmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTMKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHRoaXJkIGNvbnRyb2wgcG9pbnQuCi0gICAgICovCi0gICAgdm9pZCBhZGREYXNoUXVhZChkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIHgyLCBkb3VibGUgeTIsIGRvdWJsZSB4MywgZG91YmxlIHkzKSB7Ci0KLSAgICAgICAgZG91YmxlIHgyMSA9IHgyIC0geDE7Ci0gICAgICAgIGRvdWJsZSB5MjEgPSB5MiAtIHkxOwotICAgICAgICBkb3VibGUgeDIzID0geDIgLSB4MzsKLSAgICAgICAgZG91YmxlIHkyMyA9IHkyIC0geTM7Ci0KLSAgICAgICAgZG91YmxlIGwyMSA9IE1hdGguc3FydCh4MjEgKiB4MjEgKyB5MjEgKiB5MjEpOwotICAgICAgICBkb3VibGUgbDIzID0gTWF0aC5zcXJ0KHgyMyAqIHgyMyArIHkyMyAqIHkyMyk7Ci0KLSAgICAgICAgaWYgKGwyMSA9PSAwLjAgJiYgbDIzID09IDAuMCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGwyMSA9PSAwLjApIHsKLSAgICAgICAgICAgIGFkZERhc2hMaW5lKHgyLCB5MiwgeDMsIHkzKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChsMjMgPT0gMC4wKSB7Ci0gICAgICAgICAgICBhZGREYXNoTGluZSh4MSwgeTEsIHgyLCB5Mik7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgYXggPSB4MSArIHgzIC0geDIgLSB4MjsKLSAgICAgICAgZG91YmxlIGF5ID0geTEgKyB5MyAtIHkyIC0geTI7Ci0gICAgICAgIGRvdWJsZSBieCA9IHgyIC0geDE7Ci0gICAgICAgIGRvdWJsZSBieSA9IHkyIC0geTE7Ci0gICAgICAgIGRvdWJsZSBjeCA9IHgxOwotICAgICAgICBkb3VibGUgY3kgPSB5MTsKLQotICAgICAgICBkb3VibGUgcHgxLCBweTEsIGR4MSwgZHkxOwotICAgICAgICBweDEgPSBweTEgPSBkeDEgPSBkeTEgPSAwLjA7Ci0gICAgICAgIGRvdWJsZSBwcmV2ID0gMC4wOwotCi0gICAgICAgIGRhc2hlci5pbml0KG5ldyBEYXNoSXRlcmF0b3IuUXVhZCh4MSwgeTEsIHgyLCB5MiwgeDMsIHkzKSk7Ci0KLSAgICAgICAgd2hpbGUgKCFkYXNoZXIuZW9mKCkpIHsKLSAgICAgICAgICAgIGRvdWJsZSB0ID0gZGFzaGVyLmdldFZhbHVlKCk7Ci0gICAgICAgICAgICBkb3VibGUgZHggPSB0ICogYXggKyBieDsKLSAgICAgICAgICAgIGRvdWJsZSBkeSA9IHQgKiBheSArIGJ5OwotICAgICAgICAgICAgc2N4ID0gdCAqIChkeCArIGJ4KSArIGN4OyAvLyB0XjIgKiBheCArIDIuMCAqIHQgKiBieCArIGN4Ci0gICAgICAgICAgICBzY3kgPSB0ICogKGR5ICsgYnkpICsgY3k7IC8vIHReMiAqIGF5ICsgMi4wICogdCAqIGJ5ICsgY3kKLSAgICAgICAgICAgIGlmIChkYXNoZXIuaXNPcGVuKCkpIHsKLSAgICAgICAgICAgICAgICBweDEgPSBzY3g7Ci0gICAgICAgICAgICAgICAgcHkxID0gc2N5OwotICAgICAgICAgICAgICAgIGR4MSA9IGR4OwotICAgICAgICAgICAgICAgIGR5MSA9IGR5OwotICAgICAgICAgICAgICAgIGRvdWJsZSB3ID0gdzIgLyBNYXRoLnNxcnQoZHgxICogZHgxICsgZHkxICogZHkxKTsKLSAgICAgICAgICAgICAgICBkb3VibGUgbXgxID0gLWR5MSAqIHc7Ci0gICAgICAgICAgICAgICAgZG91YmxlIG15MSA9IGR4MSAqIHc7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGx4MSA9IHB4MSArIG14MTsKLSAgICAgICAgICAgICAgICBkb3VibGUgbHkxID0gcHkxICsgbXkxOwotICAgICAgICAgICAgICAgIGRvdWJsZSByeDEgPSBweDEgLSBteDE7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHJ5MSA9IHB5MSAtIG15MTsKLSAgICAgICAgICAgICAgICBpZiAoaXNNb3ZlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlzTW92ZSA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICBzbXggPSBweDE7Ci0gICAgICAgICAgICAgICAgICAgIHNteSA9IHB5MTsKLSAgICAgICAgICAgICAgICAgICAgcnAuY2xlYW4oKTsKLSAgICAgICAgICAgICAgICAgICAgbHAubW92ZVRvKGx4MSwgbHkxKTsKLSAgICAgICAgICAgICAgICAgICAgcnAubW92ZVRvKHJ4MSwgcnkxKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBhZGRKb2luKGxwLCB4MSwgeTEsIGx4MSwgbHkxLCB0cnVlKTsKLSAgICAgICAgICAgICAgICAgICAgYWRkSm9pbihycCwgeDEsIHkxLCByeDEsIHJ5MSwgZmFsc2UpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSBpZiAoZGFzaGVyLmlzQ29udGludWUoKSkgewotICAgICAgICAgICAgICAgIGRvdWJsZSBweDMgPSBzY3g7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHB5MyA9IHNjeTsKLSAgICAgICAgICAgICAgICBkb3VibGUgc3ggPSB4MiAtIHgyMyAqIHByZXY7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHN5ID0geTIgLSB5MjMgKiBwcmV2OwotICAgICAgICAgICAgICAgIGRvdWJsZSB0MiA9ICh0IC0gcHJldikgLyAoMSAtIHByZXYpOwotICAgICAgICAgICAgICAgIGRvdWJsZSBweDIgPSBweDEgKyAoc3ggLSBweDEpICogdDI7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHB5MiA9IHB5MSArIChzeSAtIHB5MSkgKiB0MjsKLQotICAgICAgICAgICAgICAgIGFkZFF1YWQocHgxLCBweTEsIHB4MiwgcHkyLCBweDMsIHB5Myk7Ci0gICAgICAgICAgICAgICAgaWYgKGRhc2hlci5pc0Nsb3NlZCgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGFkZENhcChscCwgcHgzLCBweTMsIHJwLnhMYXN0LCBycC55TGFzdCk7Ci0gICAgICAgICAgICAgICAgICAgIGxwLmNvbWJpbmUocnApOwotICAgICAgICAgICAgICAgICAgICBpZiAoaXNGaXJzdCkgewotICAgICAgICAgICAgICAgICAgICAgICAgaXNGaXJzdCA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICAgICAgZm14ID0gc214OwotICAgICAgICAgICAgICAgICAgICAgICAgZm15ID0gc215OwotICAgICAgICAgICAgICAgICAgICAgICAgc3AgPSBscDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGxwID0gbmV3IEJ1ZmZlcmVkUGF0aCgpOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgYWRkQ2FwKGxwLCBzbXgsIHNteSwgbHAueE1vdmUsIGxwLnlNb3ZlKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGxwLmNsb3NlUGF0aCgpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGlzTW92ZSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBwcmV2ID0gdDsKLSAgICAgICAgICAgIGRhc2hlci5uZXh0KCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGRhc2hlZCBjdWJpYyBzZWdtZW50IHRvIHRoZSB3b3JrIHBhdGguCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB5MQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHgzCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSB0aGlyZCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB5MwotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgdGhpcmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geDQKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGZvdXJzIGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHk0Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBmb3VycyBjb250cm9sIHBvaW50LgotICAgICAqLwotICAgIHZvaWQgYWRkRGFzaEN1YmljKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MiwgZG91YmxlIHgzLCBkb3VibGUgeTMsIGRvdWJsZSB4NCwKLSAgICAgICAgICAgIGRvdWJsZSB5NCkgewotCi0gICAgICAgIGRvdWJsZSB4MTIgPSB4MSAtIHgyOwotICAgICAgICBkb3VibGUgeTEyID0geTEgLSB5MjsKLSAgICAgICAgZG91YmxlIHgyMyA9IHgyIC0geDM7Ci0gICAgICAgIGRvdWJsZSB5MjMgPSB5MiAtIHkzOwotICAgICAgICBkb3VibGUgeDM0ID0geDMgLSB4NDsKLSAgICAgICAgZG91YmxlIHkzNCA9IHkzIC0geTQ7Ci0KLSAgICAgICAgZG91YmxlIGwxMiA9IE1hdGguc3FydCh4MTIgKiB4MTIgKyB5MTIgKiB5MTIpOwotICAgICAgICBkb3VibGUgbDIzID0gTWF0aC5zcXJ0KHgyMyAqIHgyMyArIHkyMyAqIHkyMyk7Ci0gICAgICAgIGRvdWJsZSBsMzQgPSBNYXRoLnNxcnQoeDM0ICogeDM0ICsgeTM0ICogeTM0KTsKLQotICAgICAgICAvLyBBbGwgZWRnZXMgYXJlIHplcm8KLSAgICAgICAgaWYgKGwxMiA9PSAwLjAgJiYgbDIzID09IDAuMCAmJiBsMzQgPT0gMC4wKSB7Ci0gICAgICAgICAgICAvLyBOT1RISU5HCi0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBPbmUgemVybyBlZGdlCi0gICAgICAgIGlmIChsMTIgPT0gMC4wICYmIGwyMyA9PSAwLjApIHsKLSAgICAgICAgICAgIGFkZERhc2hMaW5lKHgzLCB5MywgeDQsIHk0KTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChsMjMgPT0gMC4wICYmIGwzNCA9PSAwLjApIHsKLSAgICAgICAgICAgIGFkZERhc2hMaW5lKHgxLCB5MSwgeDIsIHkyKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChsMTIgPT0gMC4wICYmIGwzNCA9PSAwLjApIHsKLSAgICAgICAgICAgIGFkZERhc2hMaW5lKHgyLCB5MiwgeDMsIHkzKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGRvdWJsZSBheCA9IHg0IC0geDEgKyAzLjAgKiAoeDIgLSB4Myk7Ci0gICAgICAgIGRvdWJsZSBheSA9IHk0IC0geTEgKyAzLjAgKiAoeTIgLSB5Myk7Ci0gICAgICAgIGRvdWJsZSBieCA9IDMuMCAqICh4MSArIHgzIC0geDIgLSB4Mik7Ci0gICAgICAgIGRvdWJsZSBieSA9IDMuMCAqICh5MSArIHkzIC0geTIgLSB5Mik7Ci0gICAgICAgIGRvdWJsZSBjeCA9IDMuMCAqICh4MiAtIHgxKTsKLSAgICAgICAgZG91YmxlIGN5ID0gMy4wICogKHkyIC0geTEpOwotICAgICAgICBkb3VibGUgZHggPSB4MTsKLSAgICAgICAgZG91YmxlIGR5ID0geTE7Ci0KLSAgICAgICAgZG91YmxlIHB4MSA9IDAuMDsKLSAgICAgICAgZG91YmxlIHB5MSA9IDAuMDsKLSAgICAgICAgZG91YmxlIHByZXYgPSAwLjA7Ci0KLSAgICAgICAgZGFzaGVyLmluaXQobmV3IERhc2hJdGVyYXRvci5DdWJpYyh4MSwgeTEsIHgyLCB5MiwgeDMsIHkzLCB4NCwgeTQpKTsKLQotICAgICAgICB3aGlsZSAoIWRhc2hlci5lb2YoKSkgewotCi0gICAgICAgICAgICBkb3VibGUgdCA9IGRhc2hlci5nZXRWYWx1ZSgpOwotICAgICAgICAgICAgc2N4ID0gdCAqICh0ICogKHQgKiBheCArIGJ4KSArIGN4KSArIGR4OwotICAgICAgICAgICAgc2N5ID0gdCAqICh0ICogKHQgKiBheSArIGJ5KSArIGN5KSArIGR5OwotICAgICAgICAgICAgaWYgKGRhc2hlci5pc09wZW4oKSkgewotICAgICAgICAgICAgICAgIHB4MSA9IHNjeDsKLSAgICAgICAgICAgICAgICBweTEgPSBzY3k7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGR4MSA9IHQgKiAodCAqIChheCArIGF4ICsgYXgpICsgYnggKyBieCkgKyBjeDsKLSAgICAgICAgICAgICAgICBkb3VibGUgZHkxID0gdCAqICh0ICogKGF5ICsgYXkgKyBheSkgKyBieSArIGJ5KSArIGN5OwotICAgICAgICAgICAgICAgIGRvdWJsZSB3ID0gdzIgLyBNYXRoLnNxcnQoZHgxICogZHgxICsgZHkxICogZHkxKTsKLSAgICAgICAgICAgICAgICBkb3VibGUgbXgxID0gLWR5MSAqIHc7Ci0gICAgICAgICAgICAgICAgZG91YmxlIG15MSA9IGR4MSAqIHc7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGx4MSA9IHB4MSArIG14MTsKLSAgICAgICAgICAgICAgICBkb3VibGUgbHkxID0gcHkxICsgbXkxOwotICAgICAgICAgICAgICAgIGRvdWJsZSByeDEgPSBweDEgLSBteDE7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHJ5MSA9IHB5MSAtIG15MTsKLSAgICAgICAgICAgICAgICBpZiAoaXNNb3ZlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlzTW92ZSA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICBzbXggPSBweDE7Ci0gICAgICAgICAgICAgICAgICAgIHNteSA9IHB5MTsKLSAgICAgICAgICAgICAgICAgICAgcnAuY2xlYW4oKTsKLSAgICAgICAgICAgICAgICAgICAgbHAubW92ZVRvKGx4MSwgbHkxKTsKLSAgICAgICAgICAgICAgICAgICAgcnAubW92ZVRvKHJ4MSwgcnkxKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBhZGRKb2luKGxwLCB4MSwgeTEsIGx4MSwgbHkxLCB0cnVlKTsKLSAgICAgICAgICAgICAgICAgICAgYWRkSm9pbihycCwgeDEsIHkxLCByeDEsIHJ5MSwgZmFsc2UpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSBpZiAoZGFzaGVyLmlzQ29udGludWUoKSkgewotICAgICAgICAgICAgICAgIGRvdWJsZSBzeDEgPSB4MiAtIHgyMyAqIHByZXY7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHN5MSA9IHkyIC0geTIzICogcHJldjsKLSAgICAgICAgICAgICAgICBkb3VibGUgc3gyID0geDMgLSB4MzQgKiBwcmV2OwotICAgICAgICAgICAgICAgIGRvdWJsZSBzeTIgPSB5MyAtIHkzNCAqIHByZXY7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHN4MyA9IHN4MSArIChzeDIgLSBzeDEpICogcHJldjsKLSAgICAgICAgICAgICAgICBkb3VibGUgc3kzID0gc3kxICsgKHN5MiAtIHN5MSkgKiBwcmV2OwotICAgICAgICAgICAgICAgIGRvdWJsZSB0MiA9ICh0IC0gcHJldikgLyAoMSAtIHByZXYpOwotICAgICAgICAgICAgICAgIGRvdWJsZSBzeDQgPSBzeDMgKyAoc3gyIC0gc3gzKSAqIHQyOwotICAgICAgICAgICAgICAgIGRvdWJsZSBzeTQgPSBzeTMgKyAoc3kyIC0gc3kzKSAqIHQyOwotCi0gICAgICAgICAgICAgICAgZG91YmxlIHB4NCA9IHNjeDsKLSAgICAgICAgICAgICAgICBkb3VibGUgcHk0ID0gc2N5OwotICAgICAgICAgICAgICAgIGRvdWJsZSBweDIgPSBweDEgKyAoc3gzIC0gcHgxKSAqIHQyOwotICAgICAgICAgICAgICAgIGRvdWJsZSBweTIgPSBweTEgKyAoc3kzIC0gcHkxKSAqIHQyOwotICAgICAgICAgICAgICAgIGRvdWJsZSBweDMgPSBweDIgKyAoc3g0IC0gcHgyKSAqIHQyOwotICAgICAgICAgICAgICAgIGRvdWJsZSBweTMgPSBweTIgKyAoc3k0IC0gcHkyKSAqIHQyOwotCi0gICAgICAgICAgICAgICAgYWRkQ3ViaWMocHgxLCBweTEsIHB4MiwgcHkyLCBweDMsIHB5MywgcHg0LCBweTQpOwotICAgICAgICAgICAgICAgIGlmIChkYXNoZXIuaXNDbG9zZWQoKSkgewotICAgICAgICAgICAgICAgICAgICBhZGRDYXAobHAsIHB4NCwgcHk0LCBycC54TGFzdCwgcnAueUxhc3QpOwotICAgICAgICAgICAgICAgICAgICBscC5jb21iaW5lKHJwKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGlzRmlyc3QpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlzRmlyc3QgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZteCA9IHNteDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZteSA9IHNteTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHNwID0gbHA7Ci0gICAgICAgICAgICAgICAgICAgICAgICBscCA9IG5ldyBCdWZmZXJlZFBhdGgoKTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGFkZENhcChscCwgc214LCBzbXksIGxwLnhNb3ZlLCBscC55TW92ZSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBscC5jbG9zZVBhdGgoKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpc01vdmUgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcHJldiA9IHQ7Ci0gICAgICAgICAgICBkYXNoZXIubmV4dCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGFzaGVyIGNsYXNzIHByb3ZpZGVzIGRhc2hpbmcgZm9yIHBhcnRpY3VsYXIgZGFzaCBzdHlsZS4KLSAgICAgKi8KLSAgICBjbGFzcyBEYXNoZXIgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcG9zLgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIHBvczsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGZpcnN0LgotICAgICAgICAgKi8KLSAgICAgICAgYm9vbGVhbiBjbG9zZSwgdmlzaWJsZSwgZmlyc3Q7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBkYXNoLgotICAgICAgICAgKi8KLSAgICAgICAgZmxvYXQgZGFzaFtdOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcGhhc2UuCi0gICAgICAgICAqLwotICAgICAgICBmbG9hdCBwaGFzZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGluZGV4LgotICAgICAgICAgKi8KLSAgICAgICAgaW50IGluZGV4OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgaXRlci4KLSAgICAgICAgICovCi0gICAgICAgIERhc2hJdGVyYXRvciBpdGVyOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGFzaGVyLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGRhc2gKLSAgICAgICAgICogICAgICAgICAgICB0aGUgZGFzaC4KLSAgICAgICAgICogQHBhcmFtIHBoYXNlCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHBoYXNlLgotICAgICAgICAgKi8KLSAgICAgICAgRGFzaGVyKGZsb2F0IGRhc2hbXSwgZmxvYXQgcGhhc2UpIHsKLSAgICAgICAgICAgIHRoaXMuZGFzaCA9IGRhc2g7Ci0gICAgICAgICAgICB0aGlzLnBoYXNlID0gcGhhc2U7Ci0gICAgICAgICAgICBpbmRleCA9IDA7Ci0gICAgICAgICAgICBwb3MgPSBwaGFzZTsKLSAgICAgICAgICAgIHZpc2libGUgPSB0cnVlOwotICAgICAgICAgICAgd2hpbGUgKHBvcyA+PSBkYXNoW2luZGV4XSkgewotICAgICAgICAgICAgICAgIHZpc2libGUgPSAhdmlzaWJsZTsKLSAgICAgICAgICAgICAgICBwb3MgLT0gZGFzaFtpbmRleF07Ci0gICAgICAgICAgICAgICAgaW5kZXggPSAoaW5kZXggKyAxKSAlIGRhc2gubGVuZ3RoOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcG9zID0gLXBvczsKLSAgICAgICAgICAgIGZpcnN0ID0gdmlzaWJsZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbml0cyB0aGUuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gaXRlcgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBpdGVyLgotICAgICAgICAgKi8KLSAgICAgICAgdm9pZCBpbml0KERhc2hJdGVyYXRvciBpdGVyKSB7Ci0gICAgICAgICAgICB0aGlzLml0ZXIgPSBpdGVyOwotICAgICAgICAgICAgY2xvc2UgPSB0cnVlOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENoZWNrcyBpZiBpcyBvcGVuLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0cnVlLCBpZiBpcyBvcGVuLgotICAgICAgICAgKi8KLSAgICAgICAgYm9vbGVhbiBpc09wZW4oKSB7Ci0gICAgICAgICAgICByZXR1cm4gdmlzaWJsZSAmJiBwb3MgPCBpdGVyLmxlbmd0aDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDaGVja3MgaWYgaXMgY29udGludWUuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGlzIGNvbnRpbnVlLgotICAgICAgICAgKi8KLSAgICAgICAgYm9vbGVhbiBpc0NvbnRpbnVlKCkgewotICAgICAgICAgICAgcmV0dXJuICF2aXNpYmxlICYmIHBvcyA+IDA7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ2hlY2tzIGlmIGlzIGNsb3NlZC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgaXMgY2xvc2VkLgotICAgICAgICAgKi8KLSAgICAgICAgYm9vbGVhbiBpc0Nsb3NlZCgpIHsKLSAgICAgICAgICAgIHJldHVybiBjbG9zZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDaGVja3MgaWYgaXMgY29ubmVjdGVkLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0cnVlLCBpZiBpcyBjb25uZWN0ZWQuCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIGlzQ29ubmVjdGVkKCkgewotICAgICAgICAgICAgcmV0dXJuIGZpcnN0ICYmICFjbG9zZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBFb2YuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN1Y2Nlc3NmdWwuCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIGVvZigpIHsKLSAgICAgICAgICAgIGlmICghY2xvc2UpIHsKLSAgICAgICAgICAgICAgICBwb3MgLT0gaXRlci5sZW5ndGg7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAocG9zID49IGl0ZXIubGVuZ3RoKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHZpc2libGUpIHsKLSAgICAgICAgICAgICAgICAgICAgcG9zIC09IGl0ZXIubGVuZ3RoOwotICAgICAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgY2xvc2UgPSBwb3MgPT0gaXRlci5sZW5ndGg7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogTmV4dC4KLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgbmV4dCgpIHsKLSAgICAgICAgICAgIGlmIChjbG9zZSkgewotICAgICAgICAgICAgICAgIHBvcyArPSBkYXNoW2luZGV4XTsKLSAgICAgICAgICAgICAgICBpbmRleCA9IChpbmRleCArIDEpICUgZGFzaC5sZW5ndGg7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIC8vIEdvIGJhY2sKLSAgICAgICAgICAgICAgICBpbmRleCA9IChpbmRleCArIGRhc2gubGVuZ3RoIC0gMSkgJSBkYXNoLmxlbmd0aDsKLSAgICAgICAgICAgICAgICBwb3MgLT0gZGFzaFtpbmRleF07Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB2aXNpYmxlID0gIXZpc2libGU7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogR2V0cyB0aGUgdmFsdWUuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSB2YWx1ZS4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSBnZXRWYWx1ZSgpIHsKLSAgICAgICAgICAgIGRvdWJsZSB0ID0gaXRlci5nZXROZXh0KHBvcyk7Ci0gICAgICAgICAgICByZXR1cm4gdCA8IDAgPyAwIDogKHQgPiAxID8gMSA6IHQpOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEYXNoSXRlcmF0b3IgY2xhc3MgcHJvdmlkZXMgZGFzaGluZyBmb3IgcGFydGljdWxhciBzZWdtZW50IHR5cGUuCi0gICAgICovCi0gICAgc3RhdGljIGFic3RyYWN0IGNsYXNzIERhc2hJdGVyYXRvciB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBDb25zdGFudCBGTEFUTkVTUy4KLSAgICAgICAgICovCi0gICAgICAgIHN0YXRpYyBmaW5hbCBkb3VibGUgRkxBVE5FU1MgPSAxLjA7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBDbGFzcyBMaW5lLgotICAgICAgICAgKi8KLSAgICAgICAgc3RhdGljIGNsYXNzIExpbmUgZXh0ZW5kcyBEYXNoSXRlcmF0b3IgewotCi0gICAgICAgICAgICAvKioKLSAgICAgICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBsaW5lLgotICAgICAgICAgICAgICogCi0gICAgICAgICAgICAgKiBAcGFyYW0gbGVuCi0gICAgICAgICAgICAgKiAgICAgICAgICAgIHRoZSBsZW4uCi0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIExpbmUoZG91YmxlIGxlbikgewotICAgICAgICAgICAgICAgIGxlbmd0aCA9IGxlbjsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgICAgICBkb3VibGUgZ2V0TmV4dChkb3VibGUgZGFzaFBvcykgewotICAgICAgICAgICAgICAgIHJldHVybiBkYXNoUG9zIC8gbGVuZ3RoOwotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIENsYXNzIFF1YWQuCi0gICAgICAgICAqLwotICAgICAgICBzdGF0aWMgY2xhc3MgUXVhZCBleHRlbmRzIERhc2hJdGVyYXRvciB7Ci0KLSAgICAgICAgICAgIC8qKgotICAgICAgICAgICAgICogVGhlIHZhbCBzaXplLgotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBpbnQgdmFsU2l6ZTsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBUaGUgdmFsIHBvcy4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgaW50IHZhbFBvczsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBUaGUgY3VyIGxlbi4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgZG91YmxlIGN1ckxlbjsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBUaGUgcHJldiBsZW4uCi0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIGRvdWJsZSBwcmV2TGVuOwotCi0gICAgICAgICAgICAvKioKLSAgICAgICAgICAgICAqIFRoZSBsYXN0IGxlbi4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgZG91YmxlIGxhc3RMZW47Ci0KLSAgICAgICAgICAgIC8qKgotICAgICAgICAgICAgICogVGhlIHZhbHVlcy4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgZG91YmxlW10gdmFsdWVzOwotCi0gICAgICAgICAgICAvKioKLSAgICAgICAgICAgICAqIFRoZSBzdGVwLgotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBkb3VibGUgc3RlcDsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgcXVhZC4KLSAgICAgICAgICAgICAqIAotICAgICAgICAgICAgICogQHBhcmFtIHgxCi0gICAgICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4MS4KLSAgICAgICAgICAgICAqIEBwYXJhbSB5MQotICAgICAgICAgICAgICogICAgICAgICAgICB0aGUgeTEuCi0gICAgICAgICAgICAgKiBAcGFyYW0geDIKLSAgICAgICAgICAgICAqICAgICAgICAgICAgdGhlIHgyLgotICAgICAgICAgICAgICogQHBhcmFtIHkyCi0gICAgICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5Mi4KLSAgICAgICAgICAgICAqIEBwYXJhbSB4MwotICAgICAgICAgICAgICogICAgICAgICAgICB0aGUgeDMuCi0gICAgICAgICAgICAgKiBAcGFyYW0geTMKLSAgICAgICAgICAgICAqICAgICAgICAgICAgdGhlIHkzLgotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBRdWFkKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MiwgZG91YmxlIHgzLCBkb3VibGUgeTMpIHsKLQotICAgICAgICAgICAgICAgIGRvdWJsZSBueCA9IHgxICsgeDMgLSB4MiAtIHgyOwotICAgICAgICAgICAgICAgIGRvdWJsZSBueSA9IHkxICsgeTMgLSB5MiAtIHkyOwotCi0gICAgICAgICAgICAgICAgaW50IG4gPSAoaW50KSgxICsgTWF0aC5zcXJ0KDAuNzUgKiAoTWF0aC5hYnMobngpICsgTWF0aC5hYnMobnkpKSAqIEZMQVRORVNTKSk7Ci0gICAgICAgICAgICAgICAgc3RlcCA9IDEuMCAvIG47Ci0KLSAgICAgICAgICAgICAgICBkb3VibGUgYXggPSB4MSArIHgzIC0geDIgLSB4MjsKLSAgICAgICAgICAgICAgICBkb3VibGUgYXkgPSB5MSArIHkzIC0geTIgLSB5MjsKLSAgICAgICAgICAgICAgICBkb3VibGUgYnggPSAyLjAgKiAoeDIgLSB4MSk7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGJ5ID0gMi4wICogKHkyIC0geTEpOwotCi0gICAgICAgICAgICAgICAgZG91YmxlIGR4MSA9IHN0ZXAgKiAoc3RlcCAqIGF4ICsgYngpOwotICAgICAgICAgICAgICAgIGRvdWJsZSBkeTEgPSBzdGVwICogKHN0ZXAgKiBheSArIGJ5KTsKLSAgICAgICAgICAgICAgICBkb3VibGUgZHgyID0gc3RlcCAqIChzdGVwICogYXggKiAyLjApOwotICAgICAgICAgICAgICAgIGRvdWJsZSBkeTIgPSBzdGVwICogKHN0ZXAgKiBheSAqIDIuMCk7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHZ4ID0geDE7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHZ5ID0geTE7Ci0KLSAgICAgICAgICAgICAgICB2YWxTaXplID0gbjsKLSAgICAgICAgICAgICAgICB2YWx1ZXMgPSBuZXcgZG91YmxlW3ZhbFNpemVdOwotICAgICAgICAgICAgICAgIGRvdWJsZSBwdnggPSB2eDsKLSAgICAgICAgICAgICAgICBkb3VibGUgcHZ5ID0gdnk7Ci0gICAgICAgICAgICAgICAgbGVuZ3RoID0gMC4wOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbjsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHZ4ICs9IGR4MTsKLSAgICAgICAgICAgICAgICAgICAgdnkgKz0gZHkxOwotICAgICAgICAgICAgICAgICAgICBkeDEgKz0gZHgyOwotICAgICAgICAgICAgICAgICAgICBkeTEgKz0gZHkyOwotICAgICAgICAgICAgICAgICAgICBkb3VibGUgbHggPSB2eCAtIHB2eDsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIGx5ID0gdnkgLSBwdnk7Ci0gICAgICAgICAgICAgICAgICAgIHZhbHVlc1tpXSA9IE1hdGguc3FydChseCAqIGx4ICsgbHkgKiBseSk7Ci0gICAgICAgICAgICAgICAgICAgIGxlbmd0aCArPSB2YWx1ZXNbaV07Ci0gICAgICAgICAgICAgICAgICAgIHB2eCA9IHZ4OwotICAgICAgICAgICAgICAgICAgICBwdnkgPSB2eTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICB2YWxQb3MgPSAwOwotICAgICAgICAgICAgICAgIGN1ckxlbiA9IDAuMDsKLSAgICAgICAgICAgICAgICBwcmV2TGVuID0gMC4wOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgICAgIGRvdWJsZSBnZXROZXh0KGRvdWJsZSBkYXNoUG9zKSB7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHQgPSAyLjA7Ci0gICAgICAgICAgICAgICAgd2hpbGUgKGN1ckxlbiA8PSBkYXNoUG9zICYmIHZhbFBvcyA8IHZhbFNpemUpIHsKLSAgICAgICAgICAgICAgICAgICAgcHJldkxlbiA9IGN1ckxlbjsKLSAgICAgICAgICAgICAgICAgICAgY3VyTGVuICs9IGxhc3RMZW4gPSB2YWx1ZXNbdmFsUG9zKytdOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAoY3VyTGVuID4gZGFzaFBvcykgewotICAgICAgICAgICAgICAgICAgICB0ID0gKHZhbFBvcyAtIDEgKyAoZGFzaFBvcyAtIHByZXZMZW4pIC8gbGFzdExlbikgKiBzdGVwOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gdDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBDbGFzcyBDdWJpYy4KLSAgICAgICAgICovCi0gICAgICAgIHN0YXRpYyBjbGFzcyBDdWJpYyBleHRlbmRzIERhc2hJdGVyYXRvciB7Ci0KLSAgICAgICAgICAgIC8qKgotICAgICAgICAgICAgICogVGhlIHZhbCBzaXplLgotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBpbnQgdmFsU2l6ZTsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBUaGUgdmFsIHBvcy4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgaW50IHZhbFBvczsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBUaGUgY3VyIGxlbi4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgZG91YmxlIGN1ckxlbjsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBUaGUgcHJldiBsZW4uCi0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIGRvdWJsZSBwcmV2TGVuOwotCi0gICAgICAgICAgICAvKioKLSAgICAgICAgICAgICAqIFRoZSBsYXN0IGxlbi4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgZG91YmxlIGxhc3RMZW47Ci0KLSAgICAgICAgICAgIC8qKgotICAgICAgICAgICAgICogVGhlIHZhbHVlcy4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgZG91YmxlW10gdmFsdWVzOwotCi0gICAgICAgICAgICAvKioKLSAgICAgICAgICAgICAqIFRoZSBzdGVwLgotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBkb3VibGUgc3RlcDsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgY3ViaWMuCi0gICAgICAgICAgICAgKiAKLSAgICAgICAgICAgICAqIEBwYXJhbSB4MQotICAgICAgICAgICAgICogICAgICAgICAgICB0aGUgeDEuCi0gICAgICAgICAgICAgKiBAcGFyYW0geTEKLSAgICAgICAgICAgICAqICAgICAgICAgICAgdGhlIHkxLgotICAgICAgICAgICAgICogQHBhcmFtIHgyCi0gICAgICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4Mi4KLSAgICAgICAgICAgICAqIEBwYXJhbSB5MgotICAgICAgICAgICAgICogICAgICAgICAgICB0aGUgeTIuCi0gICAgICAgICAgICAgKiBAcGFyYW0geDMKLSAgICAgICAgICAgICAqICAgICAgICAgICAgdGhlIHgzLgotICAgICAgICAgICAgICogQHBhcmFtIHkzCi0gICAgICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5My4KLSAgICAgICAgICAgICAqIEBwYXJhbSB4NAotICAgICAgICAgICAgICogICAgICAgICAgICB0aGUgeDQuCi0gICAgICAgICAgICAgKiBAcGFyYW0geTQKLSAgICAgICAgICAgICAqICAgICAgICAgICAgdGhlIHk0LgotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBDdWJpYyhkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIHgyLCBkb3VibGUgeTIsIGRvdWJsZSB4MywgZG91YmxlIHkzLCBkb3VibGUgeDQsCi0gICAgICAgICAgICAgICAgICAgIGRvdWJsZSB5NCkgewotCi0gICAgICAgICAgICAgICAgZG91YmxlIG54MSA9IHgxICsgeDMgLSB4MiAtIHgyOwotICAgICAgICAgICAgICAgIGRvdWJsZSBueTEgPSB5MSArIHkzIC0geTIgLSB5MjsKLSAgICAgICAgICAgICAgICBkb3VibGUgbngyID0geDIgKyB4NCAtIHgzIC0geDM7Ci0gICAgICAgICAgICAgICAgZG91YmxlIG55MiA9IHkyICsgeTQgLSB5MyAtIHkzOwotCi0gICAgICAgICAgICAgICAgZG91YmxlIG1heCA9IE1hdGgubWF4KE1hdGguYWJzKG54MSkgKyBNYXRoLmFicyhueTEpLCBNYXRoLmFicyhueDIpICsgTWF0aC5hYnMobnkyKSk7Ci0gICAgICAgICAgICAgICAgaW50IG4gPSAoaW50KSgxICsgTWF0aC5zcXJ0KDAuNzUgKiBtYXgpICogRkxBVE5FU1MpOwotICAgICAgICAgICAgICAgIHN0ZXAgPSAxLjAgLyBuOwotCi0gICAgICAgICAgICAgICAgZG91YmxlIGF4ID0geDQgLSB4MSArIDMuMCAqICh4MiAtIHgzKTsKLSAgICAgICAgICAgICAgICBkb3VibGUgYXkgPSB5NCAtIHkxICsgMy4wICogKHkyIC0geTMpOwotICAgICAgICAgICAgICAgIGRvdWJsZSBieCA9IDMuMCAqICh4MSArIHgzIC0geDIgLSB4Mik7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGJ5ID0gMy4wICogKHkxICsgeTMgLSB5MiAtIHkyKTsKLSAgICAgICAgICAgICAgICBkb3VibGUgY3ggPSAzLjAgKiAoeDIgLSB4MSk7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGN5ID0gMy4wICogKHkyIC0geTEpOwotCi0gICAgICAgICAgICAgICAgZG91YmxlIGR4MSA9IHN0ZXAgKiAoc3RlcCAqIChzdGVwICogYXggKyBieCkgKyBjeCk7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGR5MSA9IHN0ZXAgKiAoc3RlcCAqIChzdGVwICogYXkgKyBieSkgKyBjeSk7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGR4MiA9IHN0ZXAgKiAoc3RlcCAqIChzdGVwICogYXggKiA2LjAgKyBieCAqIDIuMCkpOwotICAgICAgICAgICAgICAgIGRvdWJsZSBkeTIgPSBzdGVwICogKHN0ZXAgKiAoc3RlcCAqIGF5ICogNi4wICsgYnkgKiAyLjApKTsKLSAgICAgICAgICAgICAgICBkb3VibGUgZHgzID0gc3RlcCAqIChzdGVwICogKHN0ZXAgKiBheCAqIDYuMCkpOwotICAgICAgICAgICAgICAgIGRvdWJsZSBkeTMgPSBzdGVwICogKHN0ZXAgKiAoc3RlcCAqIGF5ICogNi4wKSk7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHZ4ID0geDE7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHZ5ID0geTE7Ci0KLSAgICAgICAgICAgICAgICB2YWxTaXplID0gbjsKLSAgICAgICAgICAgICAgICB2YWx1ZXMgPSBuZXcgZG91YmxlW3ZhbFNpemVdOwotICAgICAgICAgICAgICAgIGRvdWJsZSBwdnggPSB2eDsKLSAgICAgICAgICAgICAgICBkb3VibGUgcHZ5ID0gdnk7Ci0gICAgICAgICAgICAgICAgbGVuZ3RoID0gMC4wOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbjsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHZ4ICs9IGR4MTsKLSAgICAgICAgICAgICAgICAgICAgdnkgKz0gZHkxOwotICAgICAgICAgICAgICAgICAgICBkeDEgKz0gZHgyOwotICAgICAgICAgICAgICAgICAgICBkeTEgKz0gZHkyOwotICAgICAgICAgICAgICAgICAgICBkeDIgKz0gZHgzOwotICAgICAgICAgICAgICAgICAgICBkeTIgKz0gZHkzOwotICAgICAgICAgICAgICAgICAgICBkb3VibGUgbHggPSB2eCAtIHB2eDsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIGx5ID0gdnkgLSBwdnk7Ci0gICAgICAgICAgICAgICAgICAgIHZhbHVlc1tpXSA9IE1hdGguc3FydChseCAqIGx4ICsgbHkgKiBseSk7Ci0gICAgICAgICAgICAgICAgICAgIGxlbmd0aCArPSB2YWx1ZXNbaV07Ci0gICAgICAgICAgICAgICAgICAgIHB2eCA9IHZ4OwotICAgICAgICAgICAgICAgICAgICBwdnkgPSB2eTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICB2YWxQb3MgPSAwOwotICAgICAgICAgICAgICAgIGN1ckxlbiA9IDAuMDsKLSAgICAgICAgICAgICAgICBwcmV2TGVuID0gMC4wOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgICAgIGRvdWJsZSBnZXROZXh0KGRvdWJsZSBkYXNoUG9zKSB7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHQgPSAyLjA7Ci0gICAgICAgICAgICAgICAgd2hpbGUgKGN1ckxlbiA8PSBkYXNoUG9zICYmIHZhbFBvcyA8IHZhbFNpemUpIHsKLSAgICAgICAgICAgICAgICAgICAgcHJldkxlbiA9IGN1ckxlbjsKLSAgICAgICAgICAgICAgICAgICAgY3VyTGVuICs9IGxhc3RMZW4gPSB2YWx1ZXNbdmFsUG9zKytdOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAoY3VyTGVuID4gZGFzaFBvcykgewotICAgICAgICAgICAgICAgICAgICB0ID0gKHZhbFBvcyAtIDEgKyAoZGFzaFBvcyAtIHByZXZMZW4pIC8gbGFzdExlbikgKiBzdGVwOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gdDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBsZW5ndGguCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgbGVuZ3RoOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBHZXRzIHRoZSBuZXh0LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGRhc2hQb3MKLSAgICAgICAgICogICAgICAgICAgICB0aGUgZGFzaCBwb3MuCi0gICAgICAgICAqIEByZXR1cm4gdGhlIG5leHQuCi0gICAgICAgICAqLwotICAgICAgICBhYnN0cmFjdCBkb3VibGUgZ2V0TmV4dChkb3VibGUgZGFzaFBvcyk7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBCdWZmZXJlZFBhdGggY2xhc3MgcHJvdmlkZXMgd29yayBwYXRoIHN0b3JpbmcgYW5kIHByb2Nlc3NpbmcuCi0gICAgICovCi0gICAgc3RhdGljIGNsYXNzIEJ1ZmZlcmVkUGF0aCB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBDb25zdGFudCBidWZDYXBhY2l0eS4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBidWZDYXBhY2l0eSA9IDEwOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcG9pbnQgc2hpZnQuCi0gICAgICAgICAqLwotICAgICAgICBzdGF0aWMgaW50IHBvaW50U2hpZnRbXSA9IHsKLSAgICAgICAgICAgICAgICAyLCAvLyBNT1ZFVE8KLSAgICAgICAgICAgICAgICAyLCAvLyBMSU5FVE8KLSAgICAgICAgICAgICAgICA0LCAvLyBRVUFEVE8KLSAgICAgICAgICAgICAgICA2LCAvLyBDVUJJQ1RPCi0gICAgICAgICAgICAgICAgMAotICAgICAgICB9OyAvLyBDTE9TRQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgdHlwZXMuCi0gICAgICAgICAqLwotICAgICAgICBieXRlW10gdHlwZXM7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBwb2ludHMuCi0gICAgICAgICAqLwotICAgICAgICBmbG9hdFtdIHBvaW50czsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHR5cGUgc2l6ZS4KLSAgICAgICAgICovCi0gICAgICAgIGludCB0eXBlU2l6ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHBvaW50IHNpemUuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgcG9pbnRTaXplOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBsYXN0LgotICAgICAgICAgKi8KLSAgICAgICAgZmxvYXQgeExhc3Q7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGxhc3QuCi0gICAgICAgICAqLwotICAgICAgICBmbG9hdCB5TGFzdDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggbW92ZS4KLSAgICAgICAgICovCi0gICAgICAgIGZsb2F0IHhNb3ZlOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBtb3ZlLgotICAgICAgICAgKi8KLSAgICAgICAgZmxvYXQgeU1vdmU7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBidWZmZXJlZCBwYXRoLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEJ1ZmZlcmVkUGF0aCgpIHsKLSAgICAgICAgICAgIHR5cGVzID0gbmV3IGJ5dGVbYnVmQ2FwYWNpdHldOwotICAgICAgICAgICAgcG9pbnRzID0gbmV3IGZsb2F0W2J1ZkNhcGFjaXR5ICogMl07Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ2hlY2sgYnVmLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHR5cGVDb3VudAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB0eXBlIGNvdW50LgotICAgICAgICAgKiBAcGFyYW0gcG9pbnRDb3VudAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBwb2ludCBjb3VudC4KLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgY2hlY2tCdWYoaW50IHR5cGVDb3VudCwgaW50IHBvaW50Q291bnQpIHsKLSAgICAgICAgICAgIGlmICh0eXBlU2l6ZSArIHR5cGVDb3VudCA+IHR5cGVzLmxlbmd0aCkgewotICAgICAgICAgICAgICAgIGJ5dGUgdG1wW10gPSBuZXcgYnl0ZVt0eXBlU2l6ZSArIE1hdGgubWF4KGJ1ZkNhcGFjaXR5LCB0eXBlQ291bnQpXTsKLSAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHR5cGVzLCAwLCB0bXAsIDAsIHR5cGVTaXplKTsKLSAgICAgICAgICAgICAgICB0eXBlcyA9IHRtcDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChwb2ludFNpemUgKyBwb2ludENvdW50ID4gcG9pbnRzLmxlbmd0aCkgewotICAgICAgICAgICAgICAgIGZsb2F0IHRtcFtdID0gbmV3IGZsb2F0W3BvaW50U2l6ZSArIE1hdGgubWF4KGJ1ZkNhcGFjaXR5ICogMiwgcG9pbnRDb3VudCldOwotICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkocG9pbnRzLCAwLCB0bXAsIDAsIHBvaW50U2l6ZSk7Ci0gICAgICAgICAgICAgICAgcG9pbnRzID0gdG1wOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENoZWNrcyBpZiBpcyBlbXB0eS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgaXMgZW1wdHkuCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIGlzRW1wdHkoKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHlwZVNpemUgPT0gMDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDbGVhbi4KLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgY2xlYW4oKSB7Ci0gICAgICAgICAgICB0eXBlU2l6ZSA9IDA7Ci0gICAgICAgICAgICBwb2ludFNpemUgPSAwOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIE1vdmUgdG8uCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4LgotICAgICAgICAgKiBAcGFyYW0geQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5LgotICAgICAgICAgKi8KLSAgICAgICAgdm9pZCBtb3ZlVG8oZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0gICAgICAgICAgICBjaGVja0J1ZigxLCAyKTsKLSAgICAgICAgICAgIHR5cGVzW3R5cGVTaXplKytdID0gUGF0aEl0ZXJhdG9yLlNFR19NT1ZFVE87Ci0gICAgICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0geE1vdmUgPSAoZmxvYXQpeDsKLSAgICAgICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSB5TW92ZSA9IChmbG9hdCl5OwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIExpbmUgdG8uCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4LgotICAgICAgICAgKiBAcGFyYW0geQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5LgotICAgICAgICAgKi8KLSAgICAgICAgdm9pZCBsaW5lVG8oZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0gICAgICAgICAgICBjaGVja0J1ZigxLCAyKTsKLSAgICAgICAgICAgIHR5cGVzW3R5cGVTaXplKytdID0gUGF0aEl0ZXJhdG9yLlNFR19MSU5FVE87Ci0gICAgICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0geExhc3QgPSAoZmxvYXQpeDsKLSAgICAgICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSB5TGFzdCA9IChmbG9hdCl5OwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFF1YWQgdG8uCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geDEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeDEuCi0gICAgICAgICAqIEBwYXJhbSB5MQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5MS4KLSAgICAgICAgICogQHBhcmFtIHgyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHgyLgotICAgICAgICAgKiBAcGFyYW0geTIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeTIuCi0gICAgICAgICAqLwotICAgICAgICB2b2lkIHF1YWRUbyhkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIHgyLCBkb3VibGUgeTIpIHsKLSAgICAgICAgICAgIGNoZWNrQnVmKDEsIDQpOwotICAgICAgICAgICAgdHlwZXNbdHlwZVNpemUrK10gPSBQYXRoSXRlcmF0b3IuU0VHX1FVQURUTzsKLSAgICAgICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSAoZmxvYXQpeDE7Ci0gICAgICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0gKGZsb2F0KXkxOwotICAgICAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSsrXSA9IHhMYXN0ID0gKGZsb2F0KXgyOwotICAgICAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSsrXSA9IHlMYXN0ID0gKGZsb2F0KXkyOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEN1YmljIHRvLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHgxLgotICAgICAgICAgKiBAcGFyYW0geTEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeTEuCi0gICAgICAgICAqIEBwYXJhbSB4MgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4Mi4KLSAgICAgICAgICogQHBhcmFtIHkyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkyLgotICAgICAgICAgKiBAcGFyYW0geDMKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeDMuCi0gICAgICAgICAqIEBwYXJhbSB5MwotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5My4KLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgY3ViaWNUbyhkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIHgyLCBkb3VibGUgeTIsIGRvdWJsZSB4MywgZG91YmxlIHkzKSB7Ci0gICAgICAgICAgICBjaGVja0J1ZigxLCA2KTsKLSAgICAgICAgICAgIHR5cGVzW3R5cGVTaXplKytdID0gUGF0aEl0ZXJhdG9yLlNFR19DVUJJQ1RPOwotICAgICAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSsrXSA9IChmbG9hdCl4MTsKLSAgICAgICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSAoZmxvYXQpeTE7Ci0gICAgICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0gKGZsb2F0KXgyOwotICAgICAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSsrXSA9IChmbG9hdCl5MjsKLSAgICAgICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSB4TGFzdCA9IChmbG9hdCl4MzsKLSAgICAgICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSB5TGFzdCA9IChmbG9hdCl5MzsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDbG9zZSBwYXRoLgotICAgICAgICAgKi8KLSAgICAgICAgdm9pZCBjbG9zZVBhdGgoKSB7Ci0gICAgICAgICAgICBjaGVja0J1ZigxLCAwKTsKLSAgICAgICAgICAgIHR5cGVzW3R5cGVTaXplKytdID0gUGF0aEl0ZXJhdG9yLlNFR19DTE9TRTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBTZXRzIHRoZSBsYXN0LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeC4KLSAgICAgICAgICogQHBhcmFtIHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeS4KLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgc2V0TGFzdChkb3VibGUgeCwgZG91YmxlIHkpIHsKLSAgICAgICAgICAgIHBvaW50c1twb2ludFNpemUgLSAyXSA9IHhMYXN0ID0gKGZsb2F0KXg7Ci0gICAgICAgICAgICBwb2ludHNbcG9pbnRTaXplIC0gMV0gPSB5TGFzdCA9IChmbG9hdCl5OwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEFwcGVuZC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBwCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHAuCi0gICAgICAgICAqLwotICAgICAgICB2b2lkIGFwcGVuZChCdWZmZXJlZFBhdGggcCkgewotICAgICAgICAgICAgY2hlY2tCdWYocC50eXBlU2l6ZSwgcC5wb2ludFNpemUpOwotICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShwLnBvaW50cywgMCwgcG9pbnRzLCBwb2ludFNpemUsIHAucG9pbnRTaXplKTsKLSAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkocC50eXBlcywgMCwgdHlwZXMsIHR5cGVTaXplLCBwLnR5cGVTaXplKTsKLSAgICAgICAgICAgIHBvaW50U2l6ZSArPSBwLnBvaW50U2l6ZTsKLSAgICAgICAgICAgIHR5cGVTaXplICs9IHAudHlwZVNpemU7Ci0gICAgICAgICAgICB4TGFzdCA9IHBvaW50c1twb2ludFNpemUgLSAyXTsKLSAgICAgICAgICAgIHlMYXN0ID0gcG9pbnRzW3BvaW50U2l6ZSAtIDFdOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEFwcGVuZCByZXZlcnNlLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHAKLSAgICAgICAgICogICAgICAgICAgICB0aGUgcC4KLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgYXBwZW5kUmV2ZXJzZShCdWZmZXJlZFBhdGggcCkgewotICAgICAgICAgICAgY2hlY2tCdWYocC50eXBlU2l6ZSwgcC5wb2ludFNpemUpOwotICAgICAgICAgICAgLy8gU2tpcCBsYXN0IHBvaW50LCBiZWFjYXVzZSBpdCdzIHRoZSBmaXJzdCBwb2ludCBvZiB0aGUgc2Vjb25kIHBhdGgKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSBwLnBvaW50U2l6ZSAtIDI7IGkgPj0gMDsgaSAtPSAyKSB7Ci0gICAgICAgICAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSsrXSA9IHAucG9pbnRzW2kgKyAwXTsKLSAgICAgICAgICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0gcC5wb2ludHNbaSArIDFdOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgLy8gU2tpcCBmaXJzdCB0eXBlLCBiZWFjdXNlIGl0J3MgYWx3YXlzIE1PVkVUTwotICAgICAgICAgICAgaW50IGNsb3NlSW5kZXggPSAwOwotICAgICAgICAgICAgZm9yIChpbnQgaSA9IHAudHlwZVNpemUgLSAxOyBpID49IDA7IGktLSkgewotICAgICAgICAgICAgICAgIGJ5dGUgdHlwZSA9IHAudHlwZXNbaV07Ci0gICAgICAgICAgICAgICAgaWYgKHR5cGUgPT0gUGF0aEl0ZXJhdG9yLlNFR19NT1ZFVE8pIHsKLSAgICAgICAgICAgICAgICAgICAgdHlwZXNbY2xvc2VJbmRleF0gPSBQYXRoSXRlcmF0b3IuU0VHX01PVkVUTzsKLSAgICAgICAgICAgICAgICAgICAgdHlwZXNbdHlwZVNpemUrK10gPSBQYXRoSXRlcmF0b3IuU0VHX0NMT1NFOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmICh0eXBlID09IFBhdGhJdGVyYXRvci5TRUdfQ0xPU0UpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNsb3NlSW5kZXggPSB0eXBlU2l6ZTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB0eXBlc1t0eXBlU2l6ZSsrXSA9IHR5cGU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgeExhc3QgPSBwb2ludHNbcG9pbnRTaXplIC0gMl07Ci0gICAgICAgICAgICB5TGFzdCA9IHBvaW50c1twb2ludFNpemUgLSAxXTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBKb2luLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHAKLSAgICAgICAgICogICAgICAgICAgICB0aGUgcC4KLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgam9pbihCdWZmZXJlZFBhdGggcCkgewotICAgICAgICAgICAgLy8gU2tpcCBNT1ZFVE8KLSAgICAgICAgICAgIGNoZWNrQnVmKHAudHlwZVNpemUgLSAxLCBwLnBvaW50U2l6ZSAtIDIpOwotICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShwLnBvaW50cywgMiwgcG9pbnRzLCBwb2ludFNpemUsIHAucG9pbnRTaXplIC0gMik7Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHAudHlwZXMsIDEsIHR5cGVzLCB0eXBlU2l6ZSwgcC50eXBlU2l6ZSAtIDEpOwotICAgICAgICAgICAgcG9pbnRTaXplICs9IHAucG9pbnRTaXplIC0gMjsKLSAgICAgICAgICAgIHR5cGVTaXplICs9IHAudHlwZVNpemUgLSAxOwotICAgICAgICAgICAgeExhc3QgPSBwb2ludHNbcG9pbnRTaXplIC0gMl07Ci0gICAgICAgICAgICB5TGFzdCA9IHBvaW50c1twb2ludFNpemUgLSAxXTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDb21iaW5lLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHAKLSAgICAgICAgICogICAgICAgICAgICB0aGUgcC4KLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgY29tYmluZShCdWZmZXJlZFBhdGggcCkgewotICAgICAgICAgICAgY2hlY2tCdWYocC50eXBlU2l6ZSAtIDEsIHAucG9pbnRTaXplIC0gMik7Ci0gICAgICAgICAgICAvLyBTa2lwIGxhc3QgcG9pbnQsIGJlYWNhdXNlIGl0J3MgdGhlIGZpcnN0IHBvaW50IG9mIHRoZSBzZWNvbmQgcGF0aAotICAgICAgICAgICAgZm9yIChpbnQgaSA9IHAucG9pbnRTaXplIC0gNDsgaSA+PSAwOyBpIC09IDIpIHsKLSAgICAgICAgICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0gcC5wb2ludHNbaSArIDBdOwotICAgICAgICAgICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSBwLnBvaW50c1tpICsgMV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAvLyBTa2lwIGZpcnN0IHR5cGUsIGJlYWN1c2UgaXQncyBhbHdheXMgTU9WRVRPCi0gICAgICAgICAgICBmb3IgKGludCBpID0gcC50eXBlU2l6ZSAtIDE7IGkgPj0gMTsgaS0tKSB7Ci0gICAgICAgICAgICAgICAgdHlwZXNbdHlwZVNpemUrK10gPSBwLnR5cGVzW2ldOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgeExhc3QgPSBwb2ludHNbcG9pbnRTaXplIC0gMl07Ci0gICAgICAgICAgICB5TGFzdCA9IHBvaW50c1twb2ludFNpemUgLSAxXTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDcmVhdGVzIHRoZSBnZW5lcmFsIHBhdGguCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBnZW5lcmFsIHBhdGguCi0gICAgICAgICAqLwotICAgICAgICBHZW5lcmFsUGF0aCBjcmVhdGVHZW5lcmFsUGF0aCgpIHsKLSAgICAgICAgICAgIEdlbmVyYWxQYXRoIHAgPSBuZXcgR2VuZXJhbFBhdGgoKTsKLSAgICAgICAgICAgIGludCBqID0gMDsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgdHlwZVNpemU7IGkrKykgewotICAgICAgICAgICAgICAgIGludCB0eXBlID0gdHlwZXNbaV07Ci0gICAgICAgICAgICAgICAgc3dpdGNoICh0eXBlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19NT1ZFVE86Ci0gICAgICAgICAgICAgICAgICAgICAgICBwLm1vdmVUbyhwb2ludHNbal0sIHBvaW50c1tqICsgMV0pOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19MSU5FVE86Ci0gICAgICAgICAgICAgICAgICAgICAgICBwLmxpbmVUbyhwb2ludHNbal0sIHBvaW50c1tqICsgMV0pOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19RVUFEVE86Ci0gICAgICAgICAgICAgICAgICAgICAgICBwLnF1YWRUbyhwb2ludHNbal0sIHBvaW50c1tqICsgMV0sIHBvaW50c1tqICsgMl0sIHBvaW50c1tqICsgM10pOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19DVUJJQ1RPOgotICAgICAgICAgICAgICAgICAgICAgICAgcC5jdXJ2ZVRvKHBvaW50c1tqXSwgcG9pbnRzW2ogKyAxXSwgcG9pbnRzW2ogKyAyXSwgcG9pbnRzW2ogKyAzXSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcG9pbnRzW2ogKyA0XSwgcG9pbnRzW2ogKyA1XSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0NMT1NFOgotICAgICAgICAgICAgICAgICAgICAgICAgcC5jbG9zZVBhdGgoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBqICs9IHBvaW50U2hpZnRbdHlwZV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gcDsKLSAgICAgICAgfQotCi0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvQnVmZmVyQ2FwYWJpbGl0aWVzLmphdmEgYi9hd3QvamF2YS9hd3QvQnVmZmVyQ2FwYWJpbGl0aWVzLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGNkNWZlN2IuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0J1ZmZlckNhcGFiaWxpdGllcy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTk1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgQWxleGV5IEEuIFBldHJlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotLyoqCi0gKiBUaGUgQnVmZmVyQ2FwYWJpbGl0aWVzIGNsYXNzIHJlcHJlc2VudHMgdGhlIGNhcGFiaWxpdGllcyBhbmQgb3RoZXIgcHJvcGVydGllcwotICogb2YgdGhlIGltYWdlIGJ1ZmZlcnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgQnVmZmVyQ2FwYWJpbGl0aWVzIGltcGxlbWVudHMgQ2xvbmVhYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBmcm9udCBidWZmZXIgY2FwYWJpbGl0aWVzLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgSW1hZ2VDYXBhYmlsaXRpZXMgZnJvbnRCdWZmZXJDYXBhYmlsaXRpZXM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYmFjayBidWZmZXIgY2FwYWJpbGl0aWVzLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgSW1hZ2VDYXBhYmlsaXRpZXMgYmFja0J1ZmZlckNhcGFiaWxpdGllczsKLQotICAgIC8qKgotICAgICAqIFRoZSBmbGlwIGNvbnRlbnRzLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgRmxpcENvbnRlbnRzIGZsaXBDb250ZW50czsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBCdWZmZXJDYXBhYmlsaXRpZXMgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmcm9udEJ1ZmZlckNhcGFiaWxpdGllcwotICAgICAqICAgICAgICAgICAgdGhlIGZyb250IGJ1ZmZlciBjYXBhYmlsaXRpZXMsIGNhbiBub3QgYmUgbnVsbC4KLSAgICAgKiBAcGFyYW0gYmFja0J1ZmZlckNhcGFiaWxpdGllcwotICAgICAqICAgICAgICAgICAgdGhlIHRoZSBiYWNrIGFuZCBpbnRlcm1lZGlhdGUgYnVmZmVycyBjYXBhYmlsaXRpZXMsIGNhbiBub3QgYmUKLSAgICAgKiAgICAgICAgICAgIG51bGwuCi0gICAgICogQHBhcmFtIGZsaXBDb250ZW50cwotICAgICAqICAgICAgICAgICAgdGhlIGJhY2sgYnVmZmVyIGNvbnRlbnRzIGFmdGVyIHBhZ2UgZmxpcHBpbmcsIG51bGwgaWYgcGFnZQotICAgICAqICAgICAgICAgICAgZmxpcHBpbmcgaXMgbm90IHVzZWQuCi0gICAgICovCi0gICAgcHVibGljIEJ1ZmZlckNhcGFiaWxpdGllcyhJbWFnZUNhcGFiaWxpdGllcyBmcm9udEJ1ZmZlckNhcGFiaWxpdGllcywKLSAgICAgICAgICAgIEltYWdlQ2FwYWJpbGl0aWVzIGJhY2tCdWZmZXJDYXBhYmlsaXRpZXMsIEZsaXBDb250ZW50cyBmbGlwQ29udGVudHMpIHsKLSAgICAgICAgaWYgKGZyb250QnVmZmVyQ2FwYWJpbGl0aWVzID09IG51bGwgfHwgYmFja0J1ZmZlckNhcGFiaWxpdGllcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCk7Ci0gICAgICAgIH0KLQotICAgICAgICB0aGlzLmZyb250QnVmZmVyQ2FwYWJpbGl0aWVzID0gZnJvbnRCdWZmZXJDYXBhYmlsaXRpZXM7Ci0gICAgICAgIHRoaXMuYmFja0J1ZmZlckNhcGFiaWxpdGllcyA9IGJhY2tCdWZmZXJDYXBhYmlsaXRpZXM7Ci0gICAgICAgIHRoaXMuZmxpcENvbnRlbnRzID0gZmxpcENvbnRlbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBjb3B5IG9mIHRoZSBCdWZmZXJDYXBhYmlsaXRpZXMgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBjb3B5IG9mIHRoZSBCdWZmZXJDYXBhYmlsaXRpZXMgb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgY2xvbmUoKSB7Ci0gICAgICAgIHJldHVybiBuZXcgQnVmZmVyQ2FwYWJpbGl0aWVzKGZyb250QnVmZmVyQ2FwYWJpbGl0aWVzLCBiYWNrQnVmZmVyQ2FwYWJpbGl0aWVzLCBmbGlwQ29udGVudHMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGltYWdlIGNhcGFiaWxpdGllcyBvZiB0aGUgZnJvbnQgYnVmZmVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEltYWdlQ2FwYWJpbGl0aWVzIG9iamVjdCByZXByZXNlbnRlZCBjYXBhYmlsaXRpZXMgb2YgdGhlCi0gICAgICogICAgICAgICBmcm9udCBidWZmZXIuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlQ2FwYWJpbGl0aWVzIGdldEZyb250QnVmZmVyQ2FwYWJpbGl0aWVzKCkgewotICAgICAgICByZXR1cm4gZnJvbnRCdWZmZXJDYXBhYmlsaXRpZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW1hZ2UgY2FwYWJpbGl0aWVzIG9mIHRoZSBiYWNrIGJ1ZmZlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZUNhcGFiaWxpdGllcyBvYmplY3QgcmVwcmVzZW50ZWQgY2FwYWJpbGl0aWVzIG9mIHRoZSBiYWNrCi0gICAgICogICAgICAgICBidWZmZXIuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlQ2FwYWJpbGl0aWVzIGdldEJhY2tCdWZmZXJDYXBhYmlsaXRpZXMoKSB7Ci0gICAgICAgIHJldHVybiBiYWNrQnVmZmVyQ2FwYWJpbGl0aWVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZsaXAgY29udGVudHMgb2YgdGhlIGJhY2sgYnVmZmVyIGFmdGVyIHBhZ2UtZmxpcHBpbmcuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgRmxpcENvbnRlbnRzIG9mIHRoZSBiYWNrIGJ1ZmZlciBhZnRlciBwYWdlLWZsaXBwaW5nLgotICAgICAqLwotICAgIHB1YmxpYyBGbGlwQ29udGVudHMgZ2V0RmxpcENvbnRlbnRzKCkgewotICAgICAgICByZXR1cm4gZmxpcENvbnRlbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGUgYnVmZmVyIHN0cmF0ZWd5IHVzZXMgcGFnZSBmbGlwcGluZy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBidWZmZXIgc3RyYXRlZ3kgdXNlcyBwYWdlIGZsaXBwaW5nLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNQYWdlRmxpcHBpbmcoKSB7Ci0gICAgICAgIHJldHVybiBmbGlwQ29udGVudHMgIT0gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgcGFnZSBmbGlwcGluZyBpcyBvbmx5IGF2YWlsYWJsZSBpbiBmdWxsLXNjcmVlbiBtb2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgcGFnZSBmbGlwcGluZyBpcyBvbmx5IGF2YWlsYWJsZSBpbiBmdWxsLXNjcmVlbiBtb2RlLAotICAgICAqICAgICAgICAgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzRnVsbFNjcmVlblJlcXVpcmVkKCkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHBhZ2UgZmxpcHBpbmcgY2FuIGJlIHBlcmZvcm1lZCB1c2luZyBtb3JlIHRoYW4gdHdvIGJ1ZmZlcnMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBwYWdlIGZsaXBwaW5nIGNhbiBiZSBwZXJmb3JtZWQgdXNpbmcgbW9yZSB0aGFuIHR3bwotICAgICAqICAgICAgICAgYnVmZmVycywgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzTXVsdGlCdWZmZXJBdmFpbGFibGUoKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgRmxpcENvbnRlbnRzIGNsYXNzIHJlcHJlc2VudHMgYSBzZXQgb2YgcG9zc2libGUgYmFjayBidWZmZXIgY29udGVudHMKLSAgICAgKiBhZnRlciBwYWdlLWZsaXBwaW5nLgotICAgICAqIAotICAgICAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgY2xhc3MgRmxpcENvbnRlbnRzIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGJhY2sgYnVmZmVyZWQgY29udGVudHMgYXJlIGNsZWFyZWQgd2l0aCB0aGUgYmFja2dyb3VuZCBjb2xvcgotICAgICAgICAgKiBhZnRlciBmbGlwcGluZy4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgRmxpcENvbnRlbnRzIEJBQ0tHUk9VTkQgPSBuZXcgRmxpcENvbnRlbnRzKCk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBiYWNrIGJ1ZmZlcmVkIGNvbnRlbnRzIGFyZSBjb3BpZWQgdG8gdGhlIGZyb250IGJ1ZmZlciBiZWZvcmUKLSAgICAgICAgICogZmxpcHBpbmcuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEZsaXBDb250ZW50cyBDT1BJRUQgPSBuZXcgRmxpcENvbnRlbnRzKCk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBiYWNrIGJ1ZmZlciBjb250ZW50cyBhcmUgdGhlIHByaW9yIGNvbnRlbnRzIG9mIHRoZSBmcm9udCBidWZmZXIuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEZsaXBDb250ZW50cyBQUklPUiA9IG5ldyBGbGlwQ29udGVudHMoKTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGJhY2sgYnVmZmVyIGNvbnRlbnRzIGFyZSB1bmRlZmluZWQgYWZ0ZXIgZmxpcHBpbmcKLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgRmxpcENvbnRlbnRzIFVOREVGSU5FRCA9IG5ldyBGbGlwQ29udGVudHMoKTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGZsaXAgY29udGVudHMuCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIEZsaXBDb250ZW50cygpIHsKLQotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFJldHVybnMgdGhlIGhhc2ggY29kZSBvZiB0aGUgRmxpcENvbnRlbnRzIG9iamVjdC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdGhlIGhhc2ggY29kZSBvZiB0aGUgRmxpcENvbnRlbnRzIG9iamVjdC4KLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgaW50IGhhc2hDb2RlKCkgewotICAgICAgICAgICAgcmV0dXJuIHN1cGVyLmhhc2hDb2RlKCk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyB0aGUgU3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBGbGlwQ29udGVudHMgb2JqZWN0LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgc3RyaW5nCi0gICAgICAgICAqLwotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsKLSAgICAgICAgICAgIHJldHVybiBzdXBlci50b1N0cmluZygpOwotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0NvbG9yLmphdmEgYi9hd3QvamF2YS9hd3QvQ29sb3IuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTNjNTMyZC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvQ29sb3IuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDk5MCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuY29sb3IuQ29sb3JTcGFjZTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGF0YUJ1ZmZlckludDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5SYXN0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuV3JpdGFibGVSYXN0ZXI7Ci1pbXBvcnQgamF2YS5pby5TZXJpYWxpemFibGU7Ci1pbXBvcnQgamF2YS51dGlsLkFycmF5czsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBDb2xvciBjbGFzcyBkZWZpbmVzIGNvbG9ycyBpbiB0aGUgZGVmYXVsdCBzUkdCIGNvbG9yIHNwYWNlIG9yIGluIHRoZQotICogc3BlY2lmaWVkIENvbG9yU3BhY2UuIEV2ZXJ5IENvbG9yIGNvbnRhaW5zIGFscGhhIHZhbHVlLiBUaGUgYWxwaGEgdmFsdWUKLSAqIGRlZmluZXMgdGhlIHRyYW5zcGFyZW5jeSBvZiBhIGNvbG9yIGFuZCBjYW4gYmUgcmVwcmVzZW50ZWQgYnkgYSBmbG9hdCB2YWx1ZQotICogaW4gdGhlIHJhbmdlIDAuMCAtIDEuMCBvciAwIC0gMjU1LgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIENvbG9yIGltcGxlbWVudHMgUGFpbnQsIFNlcmlhbGl6YWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAxMTg1MjY4MTY4ODExNjEwNzdMOwotCi0gICAgLyoKLSAgICAgKiBUaGUgdmFsdWVzIG9mIHRoZSBmb2xsb3dpbmcgY29sb3JzIGFyZSBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvcgotICAgICAqIHdoaWNoIGNhbiBiZSByZXZlYWxlZCB1c2luZyB0aGUgZm9sbG93aW5nIG9yIHNpbWlsYXIgY29kZTogQ29sb3IgYyA9Ci0gICAgICogQ29sb3Iud2hpdGU7IFN5c3RlbS5vdXQucHJpbnRsbihjKTsKLSAgICAgKi8KLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciB3aGl0ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIHdoaXRlID0gbmV3IENvbG9yKDI1NSwgMjU1LCAyNTUpOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIHdoaXRlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQ29sb3IgV0hJVEUgPSB3aGl0ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciBsaWdodCBncmF5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQ29sb3IgbGlnaHRHcmF5ID0gbmV3IENvbG9yKDE5MiwgMTkyLCAxOTIpOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIGxpZ2h0IGdyYXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDb2xvciBMSUdIVF9HUkFZID0gbGlnaHRHcmF5OwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIGdyYXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDb2xvciBncmF5ID0gbmV3IENvbG9yKDEyOCwgMTI4LCAxMjgpOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIGdyYXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDb2xvciBHUkFZID0gZ3JheTsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciBkYXJrIGdyYXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDb2xvciBkYXJrR3JheSA9IG5ldyBDb2xvcig2NCwgNjQsIDY0KTsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciBkYXJrIGdyYXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDb2xvciBEQVJLX0dSQVkgPSBkYXJrR3JheTsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciBibGFjay4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIGJsYWNrID0gbmV3IENvbG9yKDAsIDAsIDApOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIGJsYWNrLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQ29sb3IgQkxBQ0sgPSBibGFjazsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciByZWQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDb2xvciByZWQgPSBuZXcgQ29sb3IoMjU1LCAwLCAwKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciByZWQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDb2xvciBSRUQgPSByZWQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29sb3IgcGluay4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIHBpbmsgPSBuZXcgQ29sb3IoMjU1LCAxNzUsIDE3NSk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29sb3IgcGluay4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIFBJTksgPSBwaW5rOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIG9yYW5nZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIG9yYW5nZSA9IG5ldyBDb2xvcigyNTUsIDIwMCwgMCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29sb3Igb3JhbmdlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQ29sb3IgT1JBTkdFID0gb3JhbmdlOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIHllbGxvdy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIHllbGxvdyA9IG5ldyBDb2xvcigyNTUsIDI1NSwgMCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29sb3IgeWVsbG93LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQ29sb3IgWUVMTE9XID0geWVsbG93OwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIGdyZWVuLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQ29sb3IgZ3JlZW4gPSBuZXcgQ29sb3IoMCwgMjU1LCAwKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciBncmVlbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIEdSRUVOID0gZ3JlZW47Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29sb3IgbWFnZW50YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIG1hZ2VudGEgPSBuZXcgQ29sb3IoMjU1LCAwLCAyNTUpOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIG1hZ2VudGEuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDb2xvciBNQUdFTlRBID0gbWFnZW50YTsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciBjeWFuLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQ29sb3IgY3lhbiA9IG5ldyBDb2xvcigwLCAyNTUsIDI1NSk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29sb3IgY3lhbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIENZQU4gPSBjeWFuOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbG9yIGJsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDb2xvciBibHVlID0gbmV3IENvbG9yKDAsIDAsIDI1NSk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29sb3IgYmx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbG9yIEJMVUUgPSBibHVlOwotCi0gICAgLyoqCi0gICAgICogaW50ZWdlciBSR0IgdmFsdWUuCi0gICAgICovCi0gICAgaW50IHZhbHVlOwotCi0gICAgLyoqCi0gICAgICogRmxvYXQgc1JHQiB2YWx1ZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0W10gZnJnYnZhbHVlOwotCi0gICAgLyoqCi0gICAgICogQ29sb3IgaW4gYW4gYXJiaXRyYXJ5IGNvbG9yIHNwYWNlIHdpdGggPGNvZGU+ZmxvYXQ8L2NvZGU+IGNvbXBvbmVudHMuIElmCi0gICAgICogbnVsbCwgb3RoZXIgdmFsdWUgc2hvdWxkIGJlIHVzZWQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmbG9hdCBmdmFsdWVbXTsKLQotICAgIC8qKgotICAgICAqIEZsb2F0IGFscGhhIHZhbHVlLiBJZiBmcmdidmFsdWUgaXMgbnVsbCwgdGhpcyBpcyBub3QgdmFsaWQgZGF0YS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGZhbHBoYTsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvcidzIGNvbG9yIHNwYWNlIGlmIGFwcGxpY2FibGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBDb2xvclNwYWNlIGNzOwotCi0gICAgLyoKLSAgICAgKiBUaGUgdmFsdWUgb2YgdGhlIFNDQUxFX0ZBQ1RPUiBpcyBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvciB3aGljaCBjYW4KLSAgICAgKiBiZSByZXZlYWxlZCB1c2luZyB0aGUgZm9sbG93aW5nIGNvZGU6IENvbG9yIGMgPSBuZXcgQ29sb3IoMTAwLCAxMDAsIDEwMCk7Ci0gICAgICogQ29sb3IgYmMgPSBjLmJyaWdodGVyKCk7IFN5c3RlbS5vdXQucHJpbnRsbigiQnJpZ2h0ZXIgZmFjdG9yOiAiICsKLSAgICAgKiAoKGZsb2F0KWMuZ2V0UmVkKCkpLygoZmxvYXQpYmMuZ2V0UmVkKCkpKTsgQ29sb3IgZGMgPSBjLmRhcmtlcigpOwotICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbigiRGFya2VyIGZhY3RvcjogIiArCi0gICAgICogKChmbG9hdClkYy5nZXRSZWQoKSkvKChmbG9hdCljLmdldFJlZCgpKSk7IFRoZSByZXN1bHQgaXMgdGhlIHNhbWUgZm9yCi0gICAgICogYnJpZ2h0ZXIgYW5kIGRhcmtlciBtZXRob2RzLCBzbyB3ZSBuZWVkIG9ubHkgb25lIHNjYWxlIGZhY3RvciBmb3IgYm90aC4KLSAgICAgKi8KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU0NBTEVfRkFDVE9SLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGRvdWJsZSBTQ0FMRV9GQUNUT1IgPSAwLjc7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTUlOX1NDQUxBQkxFLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBNSU5fU0NBTEFCTEUgPSAzOyAvLyBzaG91bGQgaW5jcmVhc2Ugd2hlbgotCi0gICAgLy8gbXVsdGlwbGllZCBieSBTQ0FMRV9GQUNUT1IKLQotICAgIC8qKgotICAgICAqIFRoZSBjdXJyZW50IHBhaW50IGNvbnRleHQuCi0gICAgICovCi0gICAgdHJhbnNpZW50IHByaXZhdGUgUGFpbnRDb250ZXh0IGN1cnJlbnRQYWludENvbnRleHQ7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgY29sb3IgaW4gdGhlIHNwZWNpZmllZCBDb2xvclNwYWNlLCB0aGUgc3BlY2lmaWVkIGNvbG9yCi0gICAgICogY29tcG9uZW50cyBhbmQgdGhlIHNwZWNpZmllZCBhbHBoYS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY3NwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgQ29sb3JTcGFjZSB0byBiZSB1c2VkIHRvIGRlZmluZSB0aGUgY29tcG9uZW50cy4KLSAgICAgKiBAcGFyYW0gY29tcG9uZW50cwotICAgICAqICAgICAgICAgICAgdGhlIGNvbXBvbmVudHMuCi0gICAgICogQHBhcmFtIGFscGhhCi0gICAgICogICAgICAgICAgICB0aGUgYWxwaGEuCi0gICAgICovCi0gICAgcHVibGljIENvbG9yKENvbG9yU3BhY2UgY3NwYWNlLCBmbG9hdFtdIGNvbXBvbmVudHMsIGZsb2F0IGFscGhhKSB7Ci0gICAgICAgIGludCBuQ29tcHMgPSBjc3BhY2UuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICBmbG9hdCBjb21wOwotICAgICAgICBmdmFsdWUgPSBuZXcgZmxvYXRbbkNvbXBzXTsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG5Db21wczsgaSsrKSB7Ci0gICAgICAgICAgICBjb21wID0gY29tcG9uZW50c1tpXTsKLSAgICAgICAgICAgIGlmIChjb21wIDwgMC4wZiB8fCBjb21wID4gMS4wZikgewotICAgICAgICAgICAgICAgIC8vIGF3dC4xMDc9Q29sb3IgcGFyYW1ldGVyIG91dHNpZGUgb2YgZXhwZWN0ZWQgcmFuZ2U6IGNvbXBvbmVudAotICAgICAgICAgICAgICAgIC8vIHswfS4KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEwNyIsIGkpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgZnZhbHVlW2ldID0gY29tcG9uZW50c1tpXTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChhbHBoYSA8IDAuMGYgfHwgYWxwaGEgPiAxLjBmKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTA4PUFscGhhIHZhbHVlIG91dHNpZGUgb2YgZXhwZWN0ZWQgcmFuZ2UuCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEwOCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGZhbHBoYSA9IGFscGhhOwotCi0gICAgICAgIGNzID0gY3NwYWNlOwotCi0gICAgICAgIGZyZ2J2YWx1ZSA9IGNzLnRvUkdCKGZ2YWx1ZSk7Ci0KLSAgICAgICAgdmFsdWUgPSAoKGludCkoZnJnYnZhbHVlWzJdICogMjU1ICsgMC41KSkgfCAoKChpbnQpKGZyZ2J2YWx1ZVsxXSAqIDI1NSArIDAuNSkpIDw8IDgpCi0gICAgICAgICAgICAgICAgfCAoKChpbnQpKGZyZ2J2YWx1ZVswXSAqIDI1NSArIDAuNSkpIDw8IDE2KSB8ICgoKGludCkoZmFscGhhICogMjU1ICsgMC41KSkgPDwgMjQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBzUkdCIGNvbG9yIHdpdGggdGhlIHNwZWNpZmllZCBjb21iaW5lZCBSR0JBIHZhbHVlCi0gICAgICogY29uc2lzdGluZyBvZiB0aGUgYWxwaGEgY29tcG9uZW50IGluIGJpdHMgMjQtMzEsIHRoZSByZWQgY29tcG9uZW50IGluCi0gICAgICogYml0cyAxNi0yMywgdGhlIGdyZWVuIGNvbXBvbmVudCBpbiBiaXRzIDgtMTUsIGFuZCB0aGUgYmx1ZSBjb21wb25lbnQgaW4KLSAgICAgKiBiaXRzIDAtNy4gSWYgdGhlIGhhc2FscGhhIGFyZ3VtZW50IGlzIGZhbHNlLCB0aGUgYWxwaGEgaGFzIGRlZmF1bHQgdmFsdWUKLSAgICAgKiAtIDI1NS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmdiYQotICAgICAqICAgICAgICAgICAgdGhlIFJHQkEgY29tcG9uZW50cy4KLSAgICAgKiBAcGFyYW0gaGFzQWxwaGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbHBoYSBwYXJhbWV0ZXIgaXMgdHJ1ZSBpZiBhbHBoYSBiaXRzIGFyZSB2YWxpZCwgZmFsc2UKLSAgICAgKiAgICAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3IoaW50IHJnYmEsIGJvb2xlYW4gaGFzQWxwaGEpIHsKLSAgICAgICAgaWYgKCFoYXNBbHBoYSkgewotICAgICAgICAgICAgdmFsdWUgPSByZ2JhIHwgMHhGRjAwMDAwMDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHZhbHVlID0gcmdiYTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBjb2xvciB3aXRoIHRoZSBzcGVjaWZpZWQgcmVkLCBncmVlbiwgYmx1ZSBhbmQgYWxwaGEKLSAgICAgKiBjb21wb25lbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSByCi0gICAgICogICAgICAgICAgICB0aGUgcmVkIGNvbXBvbmVudC4KLSAgICAgKiBAcGFyYW0gZwotICAgICAqICAgICAgICAgICAgdGhlIGdyZWVuIGNvbXBvbmVudC4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJsdWUgY29tcG9uZW50LgotICAgICAqIEBwYXJhbSBhCi0gICAgICogICAgICAgICAgICB0aGUgYWxwaGEgY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBDb2xvcihpbnQgciwgaW50IGcsIGludCBiLCBpbnQgYSkgewotICAgICAgICBpZiAoKHIgJiAweEZGKSAhPSByIHx8IChnICYgMHhGRikgIT0gZyB8fCAoYiAmIDB4RkYpICE9IGIgfHwgKGEgJiAweEZGKSAhPSBhKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTA5PUNvbG9yIHBhcmFtZXRlciBvdXRzaWRlIG9mIGV4cGVjdGVkIHJhbmdlLgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xMDkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB2YWx1ZSA9IGIgfCAoZyA8PCA4KSB8IChyIDw8IDE2KSB8IChhIDw8IDI0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgb3BhcXVlIHNSR0IgY29sb3Igd2l0aCB0aGUgc3BlY2lmaWVkIHJlZCwgZ3JlZW4sIGFuZAotICAgICAqIGJsdWUgdmFsdWVzLiBUaGUgQWxwaGEgY29tcG9uZW50IGlzIHNldCB0byB0aGUgZGVmYXVsdCAtIDEuMC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcgotICAgICAqICAgICAgICAgICAgdGhlIHJlZCBjb21wb25lbnQuCi0gICAgICogQHBhcmFtIGcKLSAgICAgKiAgICAgICAgICAgIHRoZSBncmVlbiBjb21wb25lbnQuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBibHVlIGNvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3IoaW50IHIsIGludCBnLCBpbnQgYikgewotICAgICAgICBpZiAoKHIgJiAweEZGKSAhPSByIHx8IChnICYgMHhGRikgIT0gZyB8fCAoYiAmIDB4RkYpICE9IGIpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xMDk9Q29sb3IgcGFyYW1ldGVyIG91dHNpZGUgb2YgZXhwZWN0ZWQgcmFuZ2UuCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEwOSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIC8vIDB4RkYgZm9yIGFscGhhIGNoYW5uZWwKLSAgICAgICAgdmFsdWUgPSBiIHwgKGcgPDwgOCkgfCAociA8PCAxNikgfCAweEZGMDAwMDAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBzUkdCIGNvbG9yIHdpdGggdGhlIHNwZWNpZmllZCBSR0IgdmFsdWUgY29uc2lzdGluZyBvZgotICAgICAqIHRoZSByZWQgY29tcG9uZW50IGluIGJpdHMgMTYtMjMsIHRoZSBncmVlbiBjb21wb25lbnQgaW4gYml0cyA4LTE1LCBhbmQKLSAgICAgKiB0aGUgYmx1ZSBjb21wb25lbnQgaW4gYml0cyAwLTcuIEFscGhhIGhhcyBkZWZhdWx0IHZhbHVlIC0gMjU1LgotICAgICAqIAotICAgICAqIEBwYXJhbSByZ2IKLSAgICAgKiAgICAgICAgICAgIHRoZSBSR0IgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3IoaW50IHJnYikgewotICAgICAgICB2YWx1ZSA9IHJnYiB8IDB4RkYwMDAwMDA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGNvbG9yIHdpdGggdGhlIHNwZWNpZmllZCByZWQsIGdyZWVuLCBibHVlIGFuZCBhbHBoYQotICAgICAqIGNvbXBvbmVudHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSByZWQgY29tcG9uZW50LgotICAgICAqIEBwYXJhbSBnCi0gICAgICogICAgICAgICAgICB0aGUgZ3JlZW4gY29tcG9uZW50LgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgYmx1ZSBjb21wb25lbnQuCi0gICAgICogQHBhcmFtIGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbHBoYSBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIENvbG9yKGZsb2F0IHIsIGZsb2F0IGcsIGZsb2F0IGIsIGZsb2F0IGEpIHsKLSAgICAgICAgdGhpcygoaW50KShyICogMjU1ICsgMC41KSwgKGludCkoZyAqIDI1NSArIDAuNSksIChpbnQpKGIgKiAyNTUgKyAwLjUpLCAoaW50KShhICogMjU1ICsgMC41KSk7Ci0gICAgICAgIGZhbHBoYSA9IGE7Ci0gICAgICAgIGZ2YWx1ZSA9IG5ldyBmbG9hdFszXTsKLSAgICAgICAgZnZhbHVlWzBdID0gcjsKLSAgICAgICAgZnZhbHVlWzFdID0gZzsKLSAgICAgICAgZnZhbHVlWzJdID0gYjsKLSAgICAgICAgZnJnYnZhbHVlID0gZnZhbHVlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBjb2xvciB3aXRoIHRoZSBzcGVjaWZpZWQgcmVkLCBncmVlbiwgYW5kIGJsdWUKLSAgICAgKiBjb21wb25lbnRzIGFuZCBkZWZhdWx0IGFscGhhIHZhbHVlIC0gMS4wLgotICAgICAqIAotICAgICAqIEBwYXJhbSByCi0gICAgICogICAgICAgICAgICB0aGUgcmVkIGNvbXBvbmVudC4KLSAgICAgKiBAcGFyYW0gZwotICAgICAqICAgICAgICAgICAgdGhlIGdyZWVuIGNvbXBvbmVudC4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJsdWUgY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBDb2xvcihmbG9hdCByLCBmbG9hdCBnLCBmbG9hdCBiKSB7Ci0gICAgICAgIHRoaXMociwgZywgYiwgMS4wZik7Ci0gICAgfQotCi0gICAgcHVibGljIFBhaW50Q29udGV4dCBjcmVhdGVDb250ZXh0KENvbG9yTW9kZWwgY20sIFJlY3RhbmdsZSByLCBSZWN0YW5nbGUyRCByMmQsCi0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0sIFJlbmRlcmluZ0hpbnRzIHJocykgewotICAgICAgICBpZiAoY3VycmVudFBhaW50Q29udGV4dCAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gY3VycmVudFBhaW50Q29udGV4dDsKLSAgICAgICAgfQotICAgICAgICBjdXJyZW50UGFpbnRDb250ZXh0ID0gbmV3IENvbG9yLkNvbG9yUGFpbnRDb250ZXh0KHZhbHVlKTsKLSAgICAgICAgcmV0dXJuIGN1cnJlbnRQYWludENvbnRleHQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgQ29sb3Igb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgQ29sb3Igb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgIC8qCi0gICAgICAgICAqIFRoZSBmb3JtYXQgb2YgdGhlIHN0cmluZyBpcyBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvciB3aGljaCBjYW4KLSAgICAgICAgICogYmUgcmV2ZWFsZWQgdXNpbmcgdGhlIGZvbGxvd2luZyBjb2RlOiBDb2xvciBjID0gbmV3IENvbG9yKDEsIDIsIDMpOwotICAgICAgICAgKiBTeXN0ZW0ub3V0LnByaW50bG4oYyk7Ci0gICAgICAgICAqLwotCi0gICAgICAgIHJldHVybiBnZXRDbGFzcygpLmdldE5hbWUoKSArICJbcj0iICsgZ2V0UmVkKCkgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgIixnPSIgKyBnZXRHcmVlbigpICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICIsYj0iICsgZ2V0Qmx1ZSgpICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICJdIjsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIHRoZSBzcGVjaWZpZWQgT2JqZWN0IHRvIHRoZSBDb2xvci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb2JqCi0gICAgICogICAgICAgICAgICB0aGUgT2JqZWN0IHRvIGJlIGNvbXBhcmVkLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCBPYmplY3QgaXMgYSBDb2xvciB3aG9zZSB2YWx1ZSBpcyBlcXVhbCB0bwotICAgICAqICAgICAgICAgdGhpcyBDb2xvciwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3Qgb2JqKSB7Ci0gICAgICAgIGlmIChvYmogaW5zdGFuY2VvZiBDb2xvcikgewotICAgICAgICAgICAgcmV0dXJuICgoQ29sb3Ipb2JqKS52YWx1ZSA9PSB0aGlzLnZhbHVlOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgZmxvYXQgYXJyYXkgY29udGFpbmluZyB0aGUgY29sb3IgYW5kIGFscGhhIGNvbXBvbmVudHMgb2YgdGhlCi0gICAgICogQ29sb3IgaW4gdGhlIHNwZWNpZmllZCBDb2xvclNwYWNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvclNwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIENvbG9yU3BhY2UuCi0gICAgICogQHBhcmFtIGNvbXBvbmVudHMKLSAgICAgKiAgICAgICAgICAgIHRoZSByZXN1bHRzIG9mIHRoaXMgbWV0aG9kIHdpbGwgYmUgd3JpdHRlbiB0byB0aGlzIGZsb2F0Ci0gICAgICogICAgICAgICAgICBhcnJheS4gSWYgbnVsbCwgYSBmbG9hdCBhcnJheSB3aWxsIGJlIGNyZWF0ZWQuCi0gICAgICogQHJldHVybiB0aGUgY29sb3IgYW5kIGFscGhhIGNvbXBvbmVudHMgaW4gYSBmbG9hdCBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXRbXSBnZXRDb21wb25lbnRzKENvbG9yU3BhY2UgY29sb3JTcGFjZSwgZmxvYXRbXSBjb21wb25lbnRzKSB7Ci0gICAgICAgIGludCBuQ29tcHMgPSBjb2xvclNwYWNlLmdldE51bUNvbXBvbmVudHMoKTsKLSAgICAgICAgaWYgKGNvbXBvbmVudHMgPT0gbnVsbCkgewotICAgICAgICAgICAgY29tcG9uZW50cyA9IG5ldyBmbG9hdFtuQ29tcHMgKyAxXTsKLSAgICAgICAgfQotCi0gICAgICAgIGdldENvbG9yQ29tcG9uZW50cyhjb2xvclNwYWNlLCBjb21wb25lbnRzKTsKLQotICAgICAgICBpZiAoZnJnYnZhbHVlICE9IG51bGwpIHsKLSAgICAgICAgICAgIGNvbXBvbmVudHNbbkNvbXBzXSA9IGZhbHBoYTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbXBvbmVudHNbbkNvbXBzXSA9IGdldEFscGhhKCkgLyAyNTVmOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGNvbXBvbmVudHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIGZsb2F0IGFycmF5IGNvbnRhaW5pbmcgdGhlIGNvbG9yIGNvbXBvbmVudHMgb2YgdGhlIENvbG9yIGluIHRoZQotICAgICAqIHNwZWNpZmllZCBDb2xvclNwYWNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvclNwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIENvbG9yU3BhY2UuCi0gICAgICogQHBhcmFtIGNvbXBvbmVudHMKLSAgICAgKiAgICAgICAgICAgIHRoZSByZXN1bHRzIG9mIHRoaXMgbWV0aG9kIHdpbGwgYmUgd3JpdHRlbiB0byB0aGlzIGZsb2F0Ci0gICAgICogICAgICAgICAgICBhcnJheS4gSWYgbnVsbCwgYSBmbG9hdCBhcnJheSB3aWxsIGJlIGNyZWF0ZWQuCi0gICAgICogQHJldHVybiB0aGUgY29sb3IgY29tcG9uZW50cyBpbiBhIGZsb2F0IGFycmF5LgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldENvbG9yQ29tcG9uZW50cyhDb2xvclNwYWNlIGNvbG9yU3BhY2UsIGZsb2F0W10gY29tcG9uZW50cykgewotICAgICAgICBmbG9hdFtdIGNpZVhZWkNvbXBvbmVudHMgPSBnZXRDb2xvclNwYWNlKCkudG9DSUVYWVooZ2V0Q29sb3JDb21wb25lbnRzKG51bGwpKTsKLSAgICAgICAgZmxvYXRbXSBjc0NvbXBvbmVudHMgPSBjb2xvclNwYWNlLmZyb21DSUVYWVooY2llWFlaQ29tcG9uZW50cyk7Ci0KLSAgICAgICAgaWYgKGNvbXBvbmVudHMgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIGNzQ29tcG9uZW50czsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY3NDb21wb25lbnRzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBjb21wb25lbnRzW2ldID0gY3NDb21wb25lbnRzW2ldOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGNvbXBvbmVudHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQ29sb3JTcGFjZSBvZiB0aGlzIENvbG9yLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIENvbG9yU3BhY2Ugb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBDb2xvclNwYWNlIGdldENvbG9yU3BhY2UoKSB7Ci0gICAgICAgIGlmIChjcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICBjcyA9IENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19zUkdCKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBjczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgbmV3IENvbG9yIHdoaWNoIGlzIGEgZGFya2VyIHRoYW4gdGhpcyBDb2xvciBhY2NvcmRpbmcgdG8gYQotICAgICAqIGZpeGVkIHNjYWxlIGZhY3Rvci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkYXJrZXIgQ29sb3IuCi0gICAgICovCi0gICAgcHVibGljIENvbG9yIGRhcmtlcigpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBDb2xvcigoaW50KShnZXRSZWQoKSAqIFNDQUxFX0ZBQ1RPUiksIChpbnQpKGdldEdyZWVuKCkgKiBTQ0FMRV9GQUNUT1IpLAotICAgICAgICAgICAgICAgIChpbnQpKGdldEJsdWUoKSAqIFNDQUxFX0ZBQ1RPUikpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBuZXcgQ29sb3Igd2hpY2ggaXMgYSBicmlnaHRlciB0aGFuIHRoaXMgQ29sb3IuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYnJpZ2h0ZXIgQ29sb3IuCi0gICAgICovCi0gICAgcHVibGljIENvbG9yIGJyaWdodGVyKCkgewotCi0gICAgICAgIGludCByID0gZ2V0UmVkKCk7Ci0gICAgICAgIGludCBiID0gZ2V0Qmx1ZSgpOwotICAgICAgICBpbnQgZyA9IGdldEdyZWVuKCk7Ci0KLSAgICAgICAgaWYgKHIgPT0gMCAmJiBiID09IDAgJiYgZyA9PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IENvbG9yKE1JTl9TQ0FMQUJMRSwgTUlOX1NDQUxBQkxFLCBNSU5fU0NBTEFCTEUpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHIgPCBNSU5fU0NBTEFCTEUgJiYgciAhPSAwKSB7Ci0gICAgICAgICAgICByID0gTUlOX1NDQUxBQkxFOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgciA9IChpbnQpKHIgLyBTQ0FMRV9GQUNUT1IpOwotICAgICAgICAgICAgciA9IChyID4gMjU1KSA/IDI1NSA6IHI7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoYiA8IE1JTl9TQ0FMQUJMRSAmJiBiICE9IDApIHsKLSAgICAgICAgICAgIGIgPSBNSU5fU0NBTEFCTEU7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBiID0gKGludCkoYiAvIFNDQUxFX0ZBQ1RPUik7Ci0gICAgICAgICAgICBiID0gKGIgPiAyNTUpID8gMjU1IDogYjsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChnIDwgTUlOX1NDQUxBQkxFICYmIGcgIT0gMCkgewotICAgICAgICAgICAgZyA9IE1JTl9TQ0FMQUJMRTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGcgPSAoaW50KShnIC8gU0NBTEVfRkFDVE9SKTsKLSAgICAgICAgICAgIGcgPSAoZyA+IDI1NSkgPyAyNTUgOiBnOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBDb2xvcihyLCBnLCBiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgZmxvYXQgYXJyYXkgY29udGFpbmluZyB0aGUgY29sb3IgYW5kIGFscGhhIGNvbXBvbmVudHMgb2YgdGhlCi0gICAgICogQ29sb3IgaW4gdGhlIGRlZmF1bHQgc1JHQiBjb2xvciBzcGFjZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29tcG9uZW50cwotICAgICAqICAgICAgICAgICAgdGhlIHJlc3VsdHMgb2YgdGhpcyBtZXRob2Qgd2lsbCBiZSB3cml0dGVuIHRvIHRoaXMgZmxvYXQKLSAgICAgKiAgICAgICAgICAgIGFycmF5LiBBIG5ldyBmbG9hdCBhcnJheSB3aWxsIGJlIGNyZWF0ZWQgaWYgdGhpcyBhcmd1bWVudCBpcwotICAgICAqICAgICAgICAgICAgbnVsbC4KLSAgICAgKiBAcmV0dXJuIHRoZSBSR0IgY29sb3IgYW5kIGFscGhhIGNvbXBvbmVudHMgaW4gYSBmbG9hdCBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXRbXSBnZXRSR0JDb21wb25lbnRzKGZsb2F0W10gY29tcG9uZW50cykgewotICAgICAgICBpZiAoY29tcG9uZW50cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICBjb21wb25lbnRzID0gbmV3IGZsb2F0WzRdOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGZyZ2J2YWx1ZSAhPSBudWxsKSB7Ci0gICAgICAgICAgICBjb21wb25lbnRzWzNdID0gZmFscGhhOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgY29tcG9uZW50c1szXSA9IGdldEFscGhhKCkgLyAyNTVmOwotICAgICAgICB9Ci0KLSAgICAgICAgZ2V0UkdCQ29sb3JDb21wb25lbnRzKGNvbXBvbmVudHMpOwotCi0gICAgICAgIHJldHVybiBjb21wb25lbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBmbG9hdCBhcnJheSBjb250YWluaW5nIHRoZSBjb2xvciBjb21wb25lbnRzIG9mIHRoZSBDb2xvciBpbiB0aGUKLSAgICAgKiBkZWZhdWx0IHNSR0IgY29sb3Igc3BhY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbXBvbmVudHMKLSAgICAgKiAgICAgICAgICAgIHRoZSByZXN1bHRzIG9mIHRoaXMgbWV0aG9kIHdpbGwgYmUgd3JpdHRlbiB0byB0aGlzIGZsb2F0Ci0gICAgICogICAgICAgICAgICBhcnJheS4gQSBuZXcgZmxvYXQgYXJyYXkgd2lsbCBiZSBjcmVhdGVkIGlmIHRoaXMgYXJndW1lbnQgaXMKLSAgICAgKiAgICAgICAgICAgIG51bGwuCi0gICAgICogQHJldHVybiB0aGUgUkdCIGNvbG9yIGNvbXBvbmVudHMgaW4gYSBmbG9hdCBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXRbXSBnZXRSR0JDb2xvckNvbXBvbmVudHMoZmxvYXRbXSBjb21wb25lbnRzKSB7Ci0gICAgICAgIGlmIChjb21wb25lbnRzID09IG51bGwpIHsKLSAgICAgICAgICAgIGNvbXBvbmVudHMgPSBuZXcgZmxvYXRbM107Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZnJnYnZhbHVlICE9IG51bGwpIHsKLSAgICAgICAgICAgIGNvbXBvbmVudHNbMl0gPSBmcmdidmFsdWVbMl07Ci0gICAgICAgICAgICBjb21wb25lbnRzWzFdID0gZnJnYnZhbHVlWzFdOwotICAgICAgICAgICAgY29tcG9uZW50c1swXSA9IGZyZ2J2YWx1ZVswXTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbXBvbmVudHNbMl0gPSBnZXRCbHVlKCkgLyAyNTVmOwotICAgICAgICAgICAgY29tcG9uZW50c1sxXSA9IGdldEdyZWVuKCkgLyAyNTVmOwotICAgICAgICAgICAgY29tcG9uZW50c1swXSA9IGdldFJlZCgpIC8gMjU1ZjsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBjb21wb25lbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBmbG9hdCBhcnJheSB3aGljaCBjb250YWlucyB0aGUgY29sb3IgYW5kIGFscGhhIGNvbXBvbmVudHMgb2YKLSAgICAgKiB0aGUgQ29sb3IgaW4gdGhlIENvbG9yU3BhY2Ugb2YgdGhlIENvbG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb21wb25lbnRzCi0gICAgICogICAgICAgICAgICB0aGUgcmVzdWx0cyBvZiB0aGlzIG1ldGhvZCB3aWxsIGJlIHdyaXR0ZW4gdG8gdGhpcyBmbG9hdAotICAgICAqICAgICAgICAgICAgYXJyYXkuIEEgbmV3IGZsb2F0IGFycmF5IHdpbGwgYmUgY3JlYXRlZCBpZiB0aGlzIGFyZ3VtZW50IGlzCi0gICAgICogICAgICAgICAgICBudWxsLgotICAgICAqIEByZXR1cm4gdGhlIGNvbG9yIGFuZCBhbHBoYSBjb21wb25lbnRzIGluIGEgZmxvYXQgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0W10gZ2V0Q29tcG9uZW50cyhmbG9hdFtdIGNvbXBvbmVudHMpIHsKLSAgICAgICAgaWYgKGZ2YWx1ZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gZ2V0UkdCQ29tcG9uZW50cyhjb21wb25lbnRzKTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBuQ29sb3JDb21wcyA9IGZ2YWx1ZS5sZW5ndGg7Ci0KLSAgICAgICAgaWYgKGNvbXBvbmVudHMgPT0gbnVsbCkgewotICAgICAgICAgICAgY29tcG9uZW50cyA9IG5ldyBmbG9hdFtuQ29sb3JDb21wcyArIDFdOwotICAgICAgICB9Ci0KLSAgICAgICAgZ2V0Q29sb3JDb21wb25lbnRzKGNvbXBvbmVudHMpOwotCi0gICAgICAgIGNvbXBvbmVudHNbbkNvbG9yQ29tcHNdID0gZmFscGhhOwotCi0gICAgICAgIHJldHVybiBjb21wb25lbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBmbG9hdCBhcnJheSB3aGljaCBjb250YWlucyB0aGUgY29sb3IgY29tcG9uZW50cyBvZiB0aGUgQ29sb3IgaW4KLSAgICAgKiB0aGUgQ29sb3JTcGFjZSBvZiB0aGUgQ29sb3IuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbXBvbmVudHMKLSAgICAgKiAgICAgICAgICAgIHRoZSByZXN1bHRzIG9mIHRoaXMgbWV0aG9kIHdpbGwgYmUgd3JpdHRlbiB0byB0aGlzIGZsb2F0Ci0gICAgICogICAgICAgICAgICBhcnJheS4gQSBuZXcgZmxvYXQgYXJyYXkgd2lsbCBiZSBjcmVhdGVkIGlmIHRoaXMgYXJndW1lbnQgaXMKLSAgICAgKiAgICAgICAgICAgIG51bGwuCi0gICAgICogQHJldHVybiB0aGUgY29sb3IgY29tcG9uZW50cyBpbiBhIGZsb2F0IGFycmF5LgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldENvbG9yQ29tcG9uZW50cyhmbG9hdFtdIGNvbXBvbmVudHMpIHsKLSAgICAgICAgaWYgKGZ2YWx1ZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gZ2V0UkdCQ29sb3JDb21wb25lbnRzKGNvbXBvbmVudHMpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGNvbXBvbmVudHMgPT0gbnVsbCkgewotICAgICAgICAgICAgY29tcG9uZW50cyA9IG5ldyBmbG9hdFtmdmFsdWUubGVuZ3RoXTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgZnZhbHVlLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBjb21wb25lbnRzW2ldID0gZnZhbHVlW2ldOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGNvbXBvbmVudHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIGhhc2ggY29kZSBvZiB0aGlzIENvbG9yIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgaGFzaCBjb2RlIG9mIHRoaXMgQ29sb3Igb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgaGFzaENvZGUoKSB7Ci0gICAgICAgIHJldHVybiB2YWx1ZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFRyYW5zcGFyZW5jeSgpIHsKLSAgICAgICAgc3dpdGNoIChnZXRBbHBoYSgpKSB7Ci0gICAgICAgICAgICBjYXNlIDB4ZmY6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIFRyYW5zcGFyZW5jeS5PUEFRVUU7Ci0gICAgICAgICAgICBjYXNlIDA6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIFRyYW5zcGFyZW5jeS5CSVRNQVNLOwotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICByZXR1cm4gVHJhbnNwYXJlbmN5LlRSQU5TTFVDRU5UOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcmVkIGNvbXBvbmVudCBvZiB0aGUgQ29sb3IgaW4gdGhlIHJhbmdlIDAtMjU1LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHJlZCBjb21wb25lbnQgb2YgdGhlIENvbG9yLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0UmVkKCkgewotICAgICAgICByZXR1cm4gKHZhbHVlID4+IDE2KSAmIDB4RkY7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgUkdCIHZhbHVlIHRoYXQgcmVwcmVzZW50cyB0aGUgY29sb3IgaW4gdGhlIGRlZmF1bHQgc1JHQgotICAgICAqIENvbG9yTW9kZWwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgUkdCIGNvbG9yIHZhbHVlIGluIHRoZSBkZWZhdWx0IHNSR0IgQ29sb3JNb2RlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFJHQigpIHsKLSAgICAgICAgcmV0dXJuIHZhbHVlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGdyZWVuIGNvbXBvbmVudCBvZiB0aGUgQ29sb3IgaW4gdGhlIHJhbmdlIDAtMjU1LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGdyZWVuIGNvbXBvbmVudCBvZiB0aGUgQ29sb3IuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRHcmVlbigpIHsKLSAgICAgICAgcmV0dXJuICh2YWx1ZSA+PiA4KSAmIDB4RkY7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYmx1ZSBjb21wb25lbnQgb2YgdGhlIENvbG9yIGluIHRoZSByYW5nZSAwLTI1NS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBibHVlIGNvbXBvbmVudCBvZiB0aGUgQ29sb3IuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRCbHVlKCkgewotICAgICAgICByZXR1cm4gdmFsdWUgJiAweEZGOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFscGhhIGNvbXBvbmVudCBvZiB0aGUgQ29sb3IgaW4gdGhlIHJhbmdlIDAtMjU1LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFscGhhIGNvbXBvbmVudCBvZiB0aGUgQ29sb3IuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRBbHBoYSgpIHsKLSAgICAgICAgcmV0dXJuICh2YWx1ZSA+PiAyNCkgJiAweEZGOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIENvbG9yIGZyb20gdGhlIHNwZWNpZmllZCBzdHJpbmcsIG9yIHJldHVybnMgdGhlIENvbG9yIHNwZWNpZmllZAotICAgICAqIGJ5IHRoZSBzZWNvbmQgcGFyYW1ldGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBubQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBzdHJpbmcuCi0gICAgICogQHBhcmFtIGRlZgotICAgICAqICAgICAgICAgICAgdGhlIGRlZmF1bHQgQ29sb3IuCi0gICAgICogQHJldHVybiB0aGUgY29sb3IgZnJvbSB0aGUgc3BlY2lmaWVkIHN0cmluZywgb3IgdGhlIENvbG9yIHNwZWNpZmllZCBieQotICAgICAqICAgICAgICAgdGhlIHNlY29uZCBwYXJhbWV0ZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBDb2xvciBnZXRDb2xvcihTdHJpbmcgbm0sIENvbG9yIGRlZikgewotICAgICAgICBJbnRlZ2VyIGludGVnZXIgPSBJbnRlZ2VyLmdldEludGVnZXIobm0pOwotCi0gICAgICAgIGlmIChpbnRlZ2VyID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBkZWY7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbmV3IENvbG9yKGludGVnZXIuaW50VmFsdWUoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQ29sb3IgZnJvbSB0aGUgc3BlY2lmaWVkIHN0cmluZywgb3IgcmV0dXJucyB0aGUgQ29sb3IgY29udmVydGVkCi0gICAgICogZnJvbSB0aGUgc2Vjb25kIHBhcmFtZXRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbm0KLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgc3RyaW5nLgotICAgICAqIEBwYXJhbSBkZWYKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZWZhdWx0IENvbG9yLgotICAgICAqIEByZXR1cm4gdGhlIGNvbG9yIGZyb20gdGhlIHNwZWNpZmllZCBzdHJpbmcsIG9yIHRoZSBDb2xvciBjb252ZXJ0ZWQgZnJvbQotICAgICAqICAgICAgICAgdGhlIHNlY29uZCBwYXJhbWV0ZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBDb2xvciBnZXRDb2xvcihTdHJpbmcgbm0sIGludCBkZWYpIHsKLSAgICAgICAgSW50ZWdlciBpbnRlZ2VyID0gSW50ZWdlci5nZXRJbnRlZ2VyKG5tKTsKLQotICAgICAgICBpZiAoaW50ZWdlciA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IENvbG9yKGRlZik7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbmV3IENvbG9yKGludGVnZXIuaW50VmFsdWUoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQ29sb3IgZnJvbSB0aGUgc3BlY2lmaWVkIFN0cmluZy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbm0KLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgc3RyaW5nLgotICAgICAqIEByZXR1cm4gdGhlIENvbG9yIG9iamVjdCwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIENvbG9yIGdldENvbG9yKFN0cmluZyBubSkgewotICAgICAgICBJbnRlZ2VyIGludGVnZXIgPSBJbnRlZ2VyLmdldEludGVnZXIobm0pOwotCi0gICAgICAgIGlmIChpbnRlZ2VyID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBDb2xvcihpbnRlZ2VyLmludFZhbHVlKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERlY29kZXMgYSBTdHJpbmcgdG8gYW4gaW50ZWdlciBhbmQgcmV0dXJucyB0aGUgc3BlY2lmaWVkIG9wYXF1ZSBDb2xvci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbm0KLSAgICAgKiAgICAgICAgICAgIHRoZSBTdHJpbmcgd2hpY2ggcmVwcmVzZW50cyBhbiBvcGFxdWUgY29sb3IgYXMgYSAyNC1iaXQKLSAgICAgKiAgICAgICAgICAgIGludGVnZXIuCi0gICAgICogQHJldHVybiB0aGUgQ29sb3Igb2JqZWN0IGZyb20gdGhlIGdpdmVuIFN0cmluZy4KLSAgICAgKiBAdGhyb3dzIE51bWJlckZvcm1hdEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBzcGVjaWZpZWQgc3RyaW5nIGNhbiBub3QgYmUgY29udmVydGVkIHRvIGFuIGludGVnZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBDb2xvciBkZWNvZGUoU3RyaW5nIG5tKSB0aHJvd3MgTnVtYmVyRm9ybWF0RXhjZXB0aW9uIHsKLSAgICAgICAgSW50ZWdlciBpbnRlZ2VyID0gSW50ZWdlci5kZWNvZGUobm0pOwotICAgICAgICByZXR1cm4gbmV3IENvbG9yKGludGVnZXIuaW50VmFsdWUoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIENvbG9yIG9iamVjdCB1c2luZyB0aGUgc3BlY2lmaWVkIHZhbHVlcyBvZiB0aGUgSFNCIGNvbG9yIG1vZGVsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaHVlIGNvbXBvbmVudCBvZiB0aGUgQ29sb3IuCi0gICAgICogQHBhcmFtIHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzYXR1cmF0aW9uIG9mIHRoZSBDb2xvci4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJyaWdodG5lc3Mgb2YgdGhlIENvbG9yLgotICAgICAqIEByZXR1cm4gYSBjb2xvciBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIGh1ZSwgc2F0dXJhdGlvbiBhbmQgYnJpZ2h0bmVzcwotICAgICAqICAgICAgICAgdmFsdWVzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgQ29sb3IgZ2V0SFNCQ29sb3IoZmxvYXQgaCwgZmxvYXQgcywgZmxvYXQgYikgewotICAgICAgICByZXR1cm4gbmV3IENvbG9yKEhTQnRvUkdCKGgsIHMsIGIpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb252ZXJ0cyB0aGUgQ29sb3Igc3BlY2lmaWVkIGJ5IHRoZSBSR0IgbW9kZWwgdG8gYW4gZXF1aXZhbGVudCBjb2xvciBpbgotICAgICAqIHRoZSBIU0IgbW9kZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSByZWQgY29tcG9uZW50LgotICAgICAqIEBwYXJhbSBnCi0gICAgICogICAgICAgICAgICB0aGUgZ3JlZW4gY29tcG9uZW50LgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgYmx1ZSBjb21wb25lbnQuCi0gICAgICogQHBhcmFtIGhzYnZhbHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiByZXN1bHQgaHVlLCBzYXR1cmF0aW9uLCBicmlnaHRuZXNzIHZhbHVlcyBvcgotICAgICAqICAgICAgICAgICAgbnVsbC4KLSAgICAgKiBAcmV0dXJuIHRoZSBmbG9hdCBhcnJheSBvZiBodWUsIHNhdHVyYXRpb24sIGJyaWdodG5lc3MgdmFsdWVzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmxvYXRbXSBSR0J0b0hTQihpbnQgciwgaW50IGcsIGludCBiLCBmbG9hdFtdIGhzYnZhbHMpIHsKLSAgICAgICAgaWYgKGhzYnZhbHMgPT0gbnVsbCkgewotICAgICAgICAgICAgaHNidmFscyA9IG5ldyBmbG9hdFszXTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBWID0gTWF0aC5tYXgoYiwgTWF0aC5tYXgociwgZykpOwotICAgICAgICBpbnQgdGVtcCA9IE1hdGgubWluKGIsIE1hdGgubWluKHIsIGcpKTsKLQotICAgICAgICBmbG9hdCBILCBTLCBCOwotCi0gICAgICAgIEIgPSBWIC8gMjU1LmY7Ci0KLSAgICAgICAgaWYgKFYgPT0gdGVtcCkgewotICAgICAgICAgICAgSCA9IFMgPSAwOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgUyA9IChWIC0gdGVtcCkgLyAoKGZsb2F0KVYpOwotCi0gICAgICAgICAgICBmbG9hdCBDciA9IChWIC0gcikgLyAoZmxvYXQpKFYgLSB0ZW1wKTsKLSAgICAgICAgICAgIGZsb2F0IENnID0gKFYgLSBnKSAvIChmbG9hdCkoViAtIHRlbXApOwotICAgICAgICAgICAgZmxvYXQgQ2IgPSAoViAtIGIpIC8gKGZsb2F0KShWIC0gdGVtcCk7Ci0KLSAgICAgICAgICAgIGlmIChyID09IFYpIHsKLSAgICAgICAgICAgICAgICBIID0gQ2IgLSBDZzsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoZyA9PSBWKSB7Ci0gICAgICAgICAgICAgICAgSCA9IDIgKyBDciAtIENiOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBIID0gNCArIENnIC0gQ3I7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIEggLz0gNi5mOwotICAgICAgICAgICAgaWYgKEggPCAwKSB7Ci0gICAgICAgICAgICAgICAgSCsrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaHNidmFsc1swXSA9IEg7Ci0gICAgICAgIGhzYnZhbHNbMV0gPSBTOwotICAgICAgICBoc2J2YWxzWzJdID0gQjsKLQotICAgICAgICByZXR1cm4gaHNidmFsczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb252ZXJ0cyB0aGUgQ29sb3Igc3BlY2lmaWVkIGJ5IHRoZSBIU0IgbW9kZWwgdG8gYW4gZXF1aXZhbGVudCBjb2xvciBpbgotICAgICAqIHRoZSBkZWZhdWx0IFJHQiBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaHVlCi0gICAgICogICAgICAgICAgICB0aGUgaHVlIGNvbXBvbmVudCBvZiB0aGUgQ29sb3IuCi0gICAgICogQHBhcmFtIHNhdHVyYXRpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBzYXR1cmF0aW9uIG9mIHRoZSBDb2xvci4KLSAgICAgKiBAcGFyYW0gYnJpZ2h0bmVzcwotICAgICAqICAgICAgICAgICAgdGhlIGJyaWdodG5lc3Mgb2YgdGhlIENvbG9yLgotICAgICAqIEByZXR1cm4gdGhlIFJHQiB2YWx1ZSBvZiB0aGUgY29sb3Igd2l0aCB0aGUgc3BlY2lmaWVkIGh1ZSwgc2F0dXJhdGlvbiBhbmQKLSAgICAgKiAgICAgICAgIGJyaWdodG5lc3MuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnQgSFNCdG9SR0IoZmxvYXQgaHVlLCBmbG9hdCBzYXR1cmF0aW9uLCBmbG9hdCBicmlnaHRuZXNzKSB7Ci0gICAgICAgIGZsb2F0IGZyLCBmZywgZmI7Ci0KLSAgICAgICAgaWYgKHNhdHVyYXRpb24gPT0gMCkgewotICAgICAgICAgICAgZnIgPSBmZyA9IGZiID0gYnJpZ2h0bmVzczsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZsb2F0IEggPSAoaHVlIC0gKGZsb2F0KU1hdGguZmxvb3IoaHVlKSkgKiA2OwotICAgICAgICAgICAgaW50IEkgPSAoaW50KU1hdGguZmxvb3IoSCk7Ci0gICAgICAgICAgICBmbG9hdCBGID0gSCAtIEk7Ci0gICAgICAgICAgICBmbG9hdCBNID0gYnJpZ2h0bmVzcyAqICgxIC0gc2F0dXJhdGlvbik7Ci0gICAgICAgICAgICBmbG9hdCBOID0gYnJpZ2h0bmVzcyAqICgxIC0gc2F0dXJhdGlvbiAqIEYpOwotICAgICAgICAgICAgZmxvYXQgSyA9IGJyaWdodG5lc3MgKiAoMSAtIHNhdHVyYXRpb24gKiAoMSAtIEYpKTsKLQotICAgICAgICAgICAgc3dpdGNoIChJKSB7Ci0gICAgICAgICAgICAgICAgY2FzZSAwOgotICAgICAgICAgICAgICAgICAgICBmciA9IGJyaWdodG5lc3M7Ci0gICAgICAgICAgICAgICAgICAgIGZnID0gSzsKLSAgICAgICAgICAgICAgICAgICAgZmIgPSBNOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIDE6Ci0gICAgICAgICAgICAgICAgICAgIGZyID0gTjsKLSAgICAgICAgICAgICAgICAgICAgZmcgPSBicmlnaHRuZXNzOwotICAgICAgICAgICAgICAgICAgICBmYiA9IE07Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgMjoKLSAgICAgICAgICAgICAgICAgICAgZnIgPSBNOwotICAgICAgICAgICAgICAgICAgICBmZyA9IGJyaWdodG5lc3M7Ci0gICAgICAgICAgICAgICAgICAgIGZiID0gSzsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSAzOgotICAgICAgICAgICAgICAgICAgICBmciA9IE07Ci0gICAgICAgICAgICAgICAgICAgIGZnID0gTjsKLSAgICAgICAgICAgICAgICAgICAgZmIgPSBicmlnaHRuZXNzOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIDQ6Ci0gICAgICAgICAgICAgICAgICAgIGZyID0gSzsKLSAgICAgICAgICAgICAgICAgICAgZmcgPSBNOwotICAgICAgICAgICAgICAgICAgICBmYiA9IGJyaWdodG5lc3M7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgNToKLSAgICAgICAgICAgICAgICAgICAgZnIgPSBicmlnaHRuZXNzOwotICAgICAgICAgICAgICAgICAgICBmZyA9IE07Ci0gICAgICAgICAgICAgICAgICAgIGZiID0gTjsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAgICAgZnIgPSBmYiA9IGZnID0gMDsgLy8gaW1wb3NzaWJsZSwgdG8gc3VwcmVzcyBjb21waWxlciBlcnJvcgotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHIgPSAoaW50KShmciAqIDI1NS4gKyAwLjUpOwotICAgICAgICBpbnQgZyA9IChpbnQpKGZnICogMjU1LiArIDAuNSk7Ci0gICAgICAgIGludCBiID0gKGludCkoZmIgKiAyNTUuICsgMC41KTsKLQotICAgICAgICByZXR1cm4gKHIgPDwgMTYpIHwgKGcgPDwgOCkgfCBiIHwgMHhGRjAwMDAwMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgQ29sb3JQYWludENvbnRleHQuCi0gICAgICovCi0gICAgY2xhc3MgQ29sb3JQYWludENvbnRleHQgaW1wbGVtZW50cyBQYWludENvbnRleHQgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgUkdCIHZhbHVlLgotICAgICAgICAgKi8KLSAgICAgICAgaW50IHJnYlZhbHVlOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgc2F2ZWQgcmFzdGVyLgotICAgICAgICAgKi8KLSAgICAgICAgV3JpdGFibGVSYXN0ZXIgc2F2ZWRSYXN0ZXIgPSBudWxsOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgY29sb3IgcGFpbnQgY29udGV4dC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSByZ2IKLSAgICAgICAgICogICAgICAgICAgICB0aGUgUkdCIHZhbHVlLgotICAgICAgICAgKi8KLSAgICAgICAgcHJvdGVjdGVkIENvbG9yUGFpbnRDb250ZXh0KGludCByZ2IpIHsKLSAgICAgICAgICAgIHJnYlZhbHVlID0gcmdiOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIHZvaWQgZGlzcG9zZSgpIHsKLSAgICAgICAgICAgIHNhdmVkUmFzdGVyID0gbnVsbDsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoKSB7Ci0gICAgICAgICAgICByZXR1cm4gQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCk7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgUmFzdGVyIGdldFJhc3RlcihpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCkgewotICAgICAgICAgICAgaWYgKHNhdmVkUmFzdGVyID09IG51bGwgfHwgdyAhPSBzYXZlZFJhc3Rlci5nZXRXaWR0aCgpIHx8IGggIT0gc2F2ZWRSYXN0ZXIuZ2V0SGVpZ2h0KCkpIHsKLSAgICAgICAgICAgICAgICBzYXZlZFJhc3RlciA9IGdldENvbG9yTW9kZWwoKS5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIodywgaCk7Ci0KLSAgICAgICAgICAgICAgICAvLyBTdXBwb3NlIHdlIGhhdmUgaGVyZSBzaW1wbGUgSU5UL1JHQiBjb2xvci9zYW1wbGUgbW9kZWwKLSAgICAgICAgICAgICAgICBEYXRhQnVmZmVySW50IGludEJ1ZmZlciA9IChEYXRhQnVmZmVySW50KXNhdmVkUmFzdGVyLmdldERhdGFCdWZmZXIoKTsKLSAgICAgICAgICAgICAgICBpbnQgcmdiVmFsdWVzW10gPSBpbnRCdWZmZXIuZ2V0RGF0YSgpOwotICAgICAgICAgICAgICAgIGludCByZ2JGaWxsVmFsdWUgPSByZ2JWYWx1ZTsKLSAgICAgICAgICAgICAgICBBcnJheXMuZmlsbChyZ2JWYWx1ZXMsIHJnYkZpbGxWYWx1ZSk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBzYXZlZFJhc3RlcjsKLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9Db21wb25lbnQuamF2YSBiL2F3dC9qYXZhL2F3dC9Db21wb25lbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzUyYTlmNC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvQ29tcG9uZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2MDIwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLS8vaW1wb3J0IGphdmEuYXd0LmRuZC5Ecm9wVGFyZ2V0OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LkNvbXBvbmVudEV2ZW50OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LkNvbXBvbmVudExpc3RlbmVyOwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LkZvY3VzRXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuRm9jdXNMaXN0ZW5lcjsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5IaWVyYXJjaHlCb3VuZHNMaXN0ZW5lcjsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5IaWVyYXJjaHlFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5IaWVyYXJjaHlMaXN0ZW5lcjsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5JbnB1dE1ldGhvZEV2ZW50OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LklucHV0TWV0aG9kTGlzdGVuZXI7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuSW52b2NhdGlvbkV2ZW50OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LktleUV2ZW50OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LktleUxpc3RlbmVyOwotaW1wb3J0IGphdmEuYXd0LmV2ZW50Lk1vdXNlRXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuTW91c2VMaXN0ZW5lcjsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5Nb3VzZU1vdGlvbkxpc3RlbmVyOwotaW1wb3J0IGphdmEuYXd0LmV2ZW50Lk1vdXNlV2hlZWxFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5Nb3VzZVdoZWVsTGlzdGVuZXI7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuUGFpbnRFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5XaW5kb3dFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5pbS5JbnB1dENvbnRleHQ7Ci1pbXBvcnQgamF2YS5hd3QuaW0uSW5wdXRNZXRob2RSZXF1ZXN0czsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJTdHJhdGVneTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW1hZ2VPYnNlcnZlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbWFnZVByb2R1Y2VyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlZvbGF0aWxlSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuV3JpdGFibGVSYXN0ZXI7Ci1pbXBvcnQgamF2YS5hd3QucGVlci5Db21wb25lbnRQZWVyOwotaW1wb3J0IGphdmEuYmVhbnMuUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcjsKLWltcG9ydCBqYXZhLmJlYW5zLlByb3BlcnR5Q2hhbmdlU3VwcG9ydDsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uT2JqZWN0SW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5QcmludFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLlByaW50V3JpdGVyOwotaW1wb3J0IGphdmEuaW8uU2VyaWFsaXphYmxlOwotaW1wb3J0IGphdmEubGFuZy5yZWZsZWN0LkFycmF5OwotaW1wb3J0IGphdmEubGFuZy5yZWZsZWN0Lk1ldGhvZDsKLWltcG9ydCBqYXZhLnNlY3VyaXR5LkFjY2Vzc0NvbnRyb2xsZXI7Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5Qcml2aWxlZ2VkQWN0aW9uOwotaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLkNvbGxlY3Rpb247Ci1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2hNYXA7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2hTZXQ7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2h0YWJsZTsKLWltcG9ydCBqYXZhLnV0aWwuSXRlcmF0b3I7Ci1pbXBvcnQgamF2YS51dGlsLkxpbmtlZExpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLWltcG9ydCBqYXZhLnV0aWwuTWFwOwotaW1wb3J0IGphdmEudXRpbC5TZXQ7Ci0KLS8vPz8/QVdUCi0vL2ltcG9ydCBqYXZheC5hY2Nlc3NpYmlsaXR5LkFjY2Vzc2libGU7Ci0vL2ltcG9ydCBqYXZheC5hY2Nlc3NpYmlsaXR5LkFjY2Vzc2libGVDb21wb25lbnQ7Ci0vL2ltcG9ydCBqYXZheC5hY2Nlc3NpYmlsaXR5LkFjY2Vzc2libGVDb250ZXh0OwotLy9pbXBvcnQgamF2YXguYWNjZXNzaWJpbGl0eS5BY2Nlc3NpYmxlUm9sZTsKLS8vaW1wb3J0IGphdmF4LmFjY2Vzc2liaWxpdHkuQWNjZXNzaWJsZVN0YXRlOwotLy9pbXBvcnQgamF2YXguYWNjZXNzaWJpbGl0eS5BY2Nlc3NpYmxlU3RhdGVTZXQ7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LkNsaXBSZWdpb247IC8vaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuRmllbGRzQWNjZXNzb3I7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5NdWx0aVJlY3RBcmVhOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Quc3RhdGUuU3RhdGU7IC8vaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QudGV4dC5UZXh0RmllbGRLaXQ7Ci0vL2ltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LnRleHQuVGV4dEtpdDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5OYXRpdmVXaW5kb3c7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Ob3RJbXBsZW1lbnRlZEV4Y2VwdGlvbjsKLQotLyoqCi0gKiBUaGUgYWJzdHJhY3QgQ29tcG9uZW50IGNsYXNzIHNwZWNpZmllcyBhbiBvYmplY3Qgd2l0aCBhIGdyYXBoaWNhbAotICogcmVwcmVzZW50YXRpb24gdGhhdCBjYW4gYmUgZGlzcGxheWVkIG9uIHRoZSBzY3JlZW4gYW5kIHRoYXQgY2FuIGludGVyYWN0IHdpdGgKLSAqIHRoZSB1c2VyIChmb3IgZXhhbXBsZTogc2Nyb2xsYmFycywgYnV0dG9ucywgY2hlY2tib3hlcykuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgQ29tcG9uZW50IGltcGxlbWVudHMgSW1hZ2VPYnNlcnZlciwgTWVudUNvbnRhaW5lciwgU2VyaWFsaXphYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC03NjQ0MTE0NTEyNzE0NjE5NzUwTDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUT1BfQUxJR05NRU5UIGluZGljYXRlcyB0aGUgdG9wIGFsaWdubWVudCBvZiB0aGUgY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgZmxvYXQgVE9QX0FMSUdOTUVOVCA9IDAuMGY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQ0VOVEVSX0FMSUdOTUVOVCBpbmRpY2F0ZXMgdGhlIGNlbnRlciBhbGlnbm1lbnQgb2YgdGhlCi0gICAgICogY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgZmxvYXQgQ0VOVEVSX0FMSUdOTUVOVCA9IDAuNWY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQk9UVE9NX0FMSUdOTUVOVCBpbmRpY2F0ZXMgdGhlIGJvdHRvbSBhbGlnbm1lbnQgb2YgdGhlCi0gICAgICogY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgZmxvYXQgQk9UVE9NX0FMSUdOTUVOVCA9IDEuMGY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTEVGVF9BTElHTk1FTlQgaW5kaWNhdGVzIHRoZSBsZWZ0IGFsaWdubWVudCBvZiB0aGUKLSAgICAgKiBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBmbG9hdCBMRUZUX0FMSUdOTUVOVCA9IDAuMGY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUklHSFRfQUxJR05NRU5UIGluZGljYXRlcyB0aGUgcmlnaHQgYWxpZ25tZW50IG9mIHRoZQotICAgICAqIGNvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGZsb2F0IFJJR0hUX0FMSUdOTUVOVCA9IDEuMGY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgY2hpbGRDbGFzc2VzRmxhZ3MuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgSGFzaHRhYmxlPENsYXNzPD8+LCBCb29sZWFuPiBjaGlsZENsYXNzZXNGbGFncyA9IG5ldyBIYXNodGFibGU8Q2xhc3M8Pz4sIEJvb2xlYW4+KCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgcGVlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBDb21wb25lbnRQZWVyIHBlZXIgPSBuZXcgQ29tcG9uZW50UGVlcigpIHsKLSAgICB9OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGluY3JlbWVudGFsSW1hZ2VVcGRhdGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgYm9vbGVhbiBpbmNyZW1lbnRhbEltYWdlVXBkYXRlOwotCi0gICAgLyoqCi0gICAgICogVGhlIHRvb2xraXQuCi0gICAgICovCi0gICAgZmluYWwgdHJhbnNpZW50IFRvb2xraXQgdG9vbGtpdCA9IFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKTsKLQotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogcHJvdGVjdGVkIGFic3RyYWN0IGNsYXNzIEFjY2Vzc2libGVBV1RDb21wb25lbnQgZXh0ZW5kcyBBY2Nlc3NpYmxlQ29udGV4dAotICAgICAqIGltcGxlbWVudHMgU2VyaWFsaXphYmxlLCBBY2Nlc3NpYmxlQ29tcG9uZW50IHsgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZwotICAgICAqIHNlcmlhbFZlcnNpb25VSUQgPSA2NDIzMjE2NTU3NTc4MDAxOTFMOyBwcm90ZWN0ZWQgY2xhc3MKLSAgICAgKiBBY2Nlc3NpYmxlQVdUQ29tcG9uZW50SGFuZGxlciBpbXBsZW1lbnRzIENvbXBvbmVudExpc3RlbmVyIHsgcHJvdGVjdGVkCi0gICAgICogQWNjZXNzaWJsZUFXVENvbXBvbmVudEhhbmRsZXIoKSB7IH0gcHVibGljIHZvaWQKLSAgICAgKiBjb21wb25lbnRIaWRkZW4oQ29tcG9uZW50RXZlbnQgZSkgeyBpZiAoYmVoYXZpb3VyLmlzTGlnaHR3ZWlnaHQoKSkgewotICAgICAqIHJldHVybjsgfSBmaXJlUHJvcGVydHlDaGFuZ2UoQWNjZXNzaWJsZUNvbnRleHQuQUNDRVNTSUJMRV9TVEFURV9QUk9QRVJUWSwKLSAgICAgKiBBY2Nlc3NpYmxlU3RhdGUuVklTSUJMRSwgbnVsbCk7IH0gcHVibGljIHZvaWQKLSAgICAgKiBjb21wb25lbnRNb3ZlZChDb21wb25lbnRFdmVudCBlKSB7IH0gcHVibGljIHZvaWQKLSAgICAgKiBjb21wb25lbnRSZXNpemVkKENvbXBvbmVudEV2ZW50IGUpIHsgfSBwdWJsaWMgdm9pZAotICAgICAqIGNvbXBvbmVudFNob3duKENvbXBvbmVudEV2ZW50IGUpIHsgaWYgKGJlaGF2aW91ci5pc0xpZ2h0d2VpZ2h0KCkpIHsKLSAgICAgKiByZXR1cm47IH0gZmlyZVByb3BlcnR5Q2hhbmdlKEFjY2Vzc2libGVDb250ZXh0LkFDQ0VTU0lCTEVfU1RBVEVfUFJPUEVSVFksCi0gICAgICogbnVsbCwgQWNjZXNzaWJsZVN0YXRlLlZJU0lCTEUpOyB9IH0gcHJvdGVjdGVkIGNsYXNzCi0gICAgICogQWNjZXNzaWJsZUFXVEZvY3VzSGFuZGxlciBpbXBsZW1lbnRzIEZvY3VzTGlzdGVuZXIgeyBwdWJsaWMgdm9pZAotICAgICAqIGZvY3VzR2FpbmVkKEZvY3VzRXZlbnQgZSkgeyBpZiAoYmVoYXZpb3VyLmlzTGlnaHR3ZWlnaHQoKSkgeyByZXR1cm47IH0KLSAgICAgKiBmaXJlUHJvcGVydHlDaGFuZ2UoQWNjZXNzaWJsZUNvbnRleHQuQUNDRVNTSUJMRV9TVEFURV9QUk9QRVJUWSwgbnVsbCwKLSAgICAgKiBBY2Nlc3NpYmxlU3RhdGUuRk9DVVNFRCk7IH0gcHVibGljIHZvaWQgZm9jdXNMb3N0KEZvY3VzRXZlbnQgZSkgeyBpZgotICAgICAqIChiZWhhdmlvdXIuaXNMaWdodHdlaWdodCgpKSB7IHJldHVybjsgfQotICAgICAqIGZpcmVQcm9wZXJ0eUNoYW5nZShBY2Nlc3NpYmxlQ29udGV4dC5BQ0NFU1NJQkxFX1NUQVRFX1BST1BFUlRZLAotICAgICAqIEFjY2Vzc2libGVTdGF0ZS5GT0NVU0VELCBudWxsKTsgfSB9IHByb3RlY3RlZCBDb21wb25lbnRMaXN0ZW5lcgotICAgICAqIGFjY2Vzc2libGVBV1RDb21wb25lbnRIYW5kbGVyOyBwcm90ZWN0ZWQgRm9jdXNMaXN0ZW5lcgotICAgICAqIGFjY2Vzc2libGVBV1RGb2N1c0hhbmRsZXI7Ci0gICAgICovCi0gICAgLyoKLSAgICAgKiBOdW1iZXIgb2YgcmVnaXN0ZXJlZCBwcm9wZXJ0eSBjaGFuZ2UgbGlzdGVuZXJzLgotICAgICAqLwotICAgIC8qCi0gICAgICogaW50IGxpc3RlbmVyc0NvdW50OyBwdWJsaWMgdm9pZCBhZGRGb2N1c0xpc3RlbmVyKEZvY3VzTGlzdGVuZXIgbCkgewotICAgICAqIENvbXBvbmVudC50aGlzLmFkZEZvY3VzTGlzdGVuZXIobCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHVibGljIHZvaWQgYWRkUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcihQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyCi0gICAgICogbGlzdGVuZXIpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7Ci0gICAgICogc3VwZXIuYWRkUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcihsaXN0ZW5lcik7IGxpc3RlbmVyc0NvdW50Kys7IGlmCi0gICAgICogKGFjY2Vzc2libGVBV1RDb21wb25lbnRIYW5kbGVyID09IG51bGwpIHsgYWNjZXNzaWJsZUFXVENvbXBvbmVudEhhbmRsZXIgPQotICAgICAqIG5ldyBBY2Nlc3NpYmxlQVdUQ29tcG9uZW50SGFuZGxlcigpOwotICAgICAqIENvbXBvbmVudC50aGlzLmFkZENvbXBvbmVudExpc3RlbmVyKGFjY2Vzc2libGVBV1RDb21wb25lbnRIYW5kbGVyKTsgfSBpZgotICAgICAqIChhY2Nlc3NpYmxlQVdURm9jdXNIYW5kbGVyID09IG51bGwpIHsgYWNjZXNzaWJsZUFXVEZvY3VzSGFuZGxlciA9IG5ldwotICAgICAqIEFjY2Vzc2libGVBV1RGb2N1c0hhbmRsZXIoKTsKLSAgICAgKiBDb21wb25lbnQudGhpcy5hZGRGb2N1c0xpc3RlbmVyKGFjY2Vzc2libGVBV1RGb2N1c0hhbmRsZXIpOyB9IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFBvaW50IHApIHsKLSAgICAgKiB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgcmV0dXJuIENvbXBvbmVudC50aGlzLmNvbnRhaW5zKHApOyB9IGZpbmFsbHkgewotICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfSBwdWJsaWMgQWNjZXNzaWJsZSBnZXRBY2Nlc3NpYmxlQXQoUG9pbnQgYXJnMCkgewotICAgICAqIHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4gbnVsbDsgfSBmaW5hbGx5IHsgdG9vbGtpdC51bmxvY2tBV1QoKTsgfQotICAgICAqIH0gcHVibGljIENvbG9yIGdldEJhY2tncm91bmQoKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4KLSAgICAgKiBDb21wb25lbnQudGhpcy5nZXRCYWNrZ3JvdW5kKCk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Qm91bmRzKCkgeyB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgcmV0dXJuCi0gICAgICogQ29tcG9uZW50LnRoaXMuZ2V0Qm91bmRzKCk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfSBwdWJsaWMKLSAgICAgKiBDdXJzb3IgZ2V0Q3Vyc29yKCkgeyB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgcmV0dXJuCi0gICAgICogQ29tcG9uZW50LnRoaXMuZ2V0Q3Vyc29yKCk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfSBwdWJsaWMKLSAgICAgKiBGb250IGdldEZvbnQoKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4KLSAgICAgKiBDb21wb25lbnQudGhpcy5nZXRGb250KCk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfSBwdWJsaWMKLSAgICAgKiBGb250TWV0cmljcyBnZXRGb250TWV0cmljcyhGb250IGYpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7IHJldHVybgotICAgICAqIENvbXBvbmVudC50aGlzLmdldEZvbnRNZXRyaWNzKGYpOyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKiBwdWJsaWMgQ29sb3IgZ2V0Rm9yZWdyb3VuZCgpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7IHJldHVybgotICAgICAqIENvbXBvbmVudC50aGlzLmdldEZvcmVncm91bmQoKTsgfSBmaW5hbGx5IHsgdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9Ci0gICAgICogcHVibGljIFBvaW50IGdldExvY2F0aW9uKCkgeyB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgcmV0dXJuCi0gICAgICogQ29tcG9uZW50LnRoaXMuZ2V0TG9jYXRpb24oKTsgfSBmaW5hbGx5IHsgdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IHB1YmxpYwotICAgICAqIFBvaW50IGdldExvY2F0aW9uT25TY3JlZW4oKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4KLSAgICAgKiBDb21wb25lbnQudGhpcy5nZXRMb2NhdGlvbk9uU2NyZWVuKCk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0KLSAgICAgKiB9IHB1YmxpYyBEaW1lbnNpb24gZ2V0U2l6ZSgpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7IHJldHVybgotICAgICAqIENvbXBvbmVudC50aGlzLmdldFNpemUoKTsgfSBmaW5hbGx5IHsgdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IHB1YmxpYwotICAgICAqIGJvb2xlYW4gaXNFbmFibGVkKCkgeyB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgcmV0dXJuCi0gICAgICogQ29tcG9uZW50LnRoaXMuaXNFbmFibGVkKCk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfSBwdWJsaWMKLSAgICAgKiBib29sZWFuIGlzRm9jdXNUcmF2ZXJzYWJsZSgpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7IHJldHVybgotICAgICAqIENvbXBvbmVudC50aGlzLmlzRm9jdXNUcmF2ZXJzYWJsZSgpOyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKiBwdWJsaWMgYm9vbGVhbiBpc1Nob3dpbmcoKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4KLSAgICAgKiBDb21wb25lbnQudGhpcy5pc1Nob3dpbmcoKTsgfSBmaW5hbGx5IHsgdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IHB1YmxpYwotICAgICAqIGJvb2xlYW4gaXNWaXNpYmxlKCkgeyB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgcmV0dXJuCi0gICAgICogQ29tcG9uZW50LnRoaXMuaXNWaXNpYmxlKCk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfSBwdWJsaWMKLSAgICAgKiB2b2lkIHJlbW92ZUZvY3VzTGlzdGVuZXIoRm9jdXNMaXN0ZW5lciBsKSB7Ci0gICAgICogQ29tcG9uZW50LnRoaXMucmVtb3ZlRm9jdXNMaXN0ZW5lcihsKTsgfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgdm9pZCByZW1vdmVQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyKFByb3BlcnR5Q2hhbmdlTGlzdGVuZXIKLSAgICAgKiBsaXN0ZW5lcikgeyB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsKLSAgICAgKiBzdXBlci5yZW1vdmVQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyKGxpc3RlbmVyKTsgbGlzdGVuZXJzQ291bnQtLTsgaWYKLSAgICAgKiAobGlzdGVuZXJzQ291bnQgPiAwKSB7IHJldHVybjsgfSAvLyBpZiB0aGVyZSBhcmUgbm8gbW9yZSBsaXN0ZW5lcnMsCi0gICAgICogcmVtb3ZlIGhhbmRsZXJzOgotICAgICAqIENvbXBvbmVudC50aGlzLnJlbW92ZUZvY3VzTGlzdGVuZXIoYWNjZXNzaWJsZUFXVEZvY3VzSGFuZGxlcik7Ci0gICAgICogQ29tcG9uZW50LnRoaXMucmVtb3ZlQ29tcG9uZW50TGlzdGVuZXIoYWNjZXNzaWJsZUFXVENvbXBvbmVudEhhbmRsZXIpOwotICAgICAqIGFjY2Vzc2libGVBV1RDb21wb25lbnRIYW5kbGVyID0gbnVsbDsgYWNjZXNzaWJsZUFXVEZvY3VzSGFuZGxlciA9IG51bGw7IH0KLSAgICAgKiBmaW5hbGx5IHsgdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IHB1YmxpYyB2b2lkIHJlcXVlc3RGb2N1cygpIHsKLSAgICAgKiB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgQ29tcG9uZW50LnRoaXMucmVxdWVzdEZvY3VzKCk7IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IHB1YmxpYyB2b2lkIHNldEJhY2tncm91bmQoQ29sb3IgY29sb3IpIHsKLSAgICAgKiB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgQ29tcG9uZW50LnRoaXMuc2V0QmFja2dyb3VuZChjb2xvcik7IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IHB1YmxpYyB2b2lkIHNldEJvdW5kcyhSZWN0YW5nbGUgcikgewotICAgICAqIHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyBDb21wb25lbnQudGhpcy5zZXRCb3VuZHMocik7IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IHB1YmxpYyB2b2lkIHNldEN1cnNvcihDdXJzb3IgY3Vyc29yKSB7Ci0gICAgICogdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7IENvbXBvbmVudC50aGlzLnNldEN1cnNvcihjdXJzb3IpOyB9IGZpbmFsbHkgewotICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfSBwdWJsaWMgdm9pZCBzZXRFbmFibGVkKGJvb2xlYW4gZW5hYmxlZCkgewotICAgICAqIHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyBDb21wb25lbnQudGhpcy5zZXRFbmFibGVkKGVuYWJsZWQpOyB9IGZpbmFsbHkgewotICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfSBwdWJsaWMgdm9pZCBzZXRGb250KEZvbnQgZikgeyB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgKiB0cnkgeyBDb21wb25lbnQudGhpcy5zZXRGb250KGYpOyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKiBwdWJsaWMgdm9pZCBzZXRGb3JlZ3JvdW5kKENvbG9yIGNvbG9yKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgewotICAgICAqIENvbXBvbmVudC50aGlzLnNldEZvcmVncm91bmQoY29sb3IpOyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKiBwdWJsaWMgdm9pZCBzZXRMb2NhdGlvbihQb2ludCBwKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgewotICAgICAqIENvbXBvbmVudC50aGlzLnNldExvY2F0aW9uKHApOyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKiBwdWJsaWMgdm9pZCBzZXRTaXplKERpbWVuc2lvbiBzaXplKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgewotICAgICAqIENvbXBvbmVudC50aGlzLnNldFNpemUoc2l6ZSk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfSBwdWJsaWMKLSAgICAgKiB2b2lkIHNldFZpc2libGUoYm9vbGVhbiB2aXNpYmxlKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgewotICAgICAqIENvbXBvbmVudC50aGlzLnNldFZpc2libGUodmlzaWJsZSk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgQWNjZXNzaWJsZSBnZXRBY2Nlc3NpYmxlUGFyZW50KCkgeyB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgKiB0cnkgeyBBY2Nlc3NpYmxlIGFQYXJlbnQgPSBzdXBlci5nZXRBY2Nlc3NpYmxlUGFyZW50KCk7IGlmIChhUGFyZW50ICE9Ci0gICAgICogbnVsbCkgeyByZXR1cm4gYVBhcmVudDsgfSBDb250YWluZXIgcGFyZW50ID0gZ2V0UGFyZW50KCk7IHJldHVybiAocGFyZW50Ci0gICAgICogaW5zdGFuY2VvZiBBY2Nlc3NpYmxlID8gKEFjY2Vzc2libGUpIHBhcmVudCA6IG51bGwpOyB9IGZpbmFsbHkgewotICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgQWNjZXNzaWJsZSBnZXRBY2Nlc3NpYmxlQ2hpbGQoaW50IGkpIHsKLSAgICAgKiB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgcmV0dXJuIG51bGw7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0KLSAgICAgKiB9Ci0gICAgICogQE92ZXJyaWRlIHB1YmxpYyBpbnQgZ2V0QWNjZXNzaWJsZUNoaWxkcmVuQ291bnQoKSB7IHRvb2xraXQubG9ja0FXVCgpOwotICAgICAqIHRyeSB7IHJldHVybiAwOyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHVibGljIEFjY2Vzc2libGVDb21wb25lbnQgZ2V0QWNjZXNzaWJsZUNvbXBvbmVudCgpIHsgcmV0dXJuCi0gICAgICogdGhpczsgfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgU3RyaW5nIGdldEFjY2Vzc2libGVEZXNjcmlwdGlvbigpIHsgcmV0dXJuCi0gICAgICogc3VwZXIuZ2V0QWNjZXNzaWJsZURlc2NyaXB0aW9uKCk7IC8vIHdoeSBvdmVycmlkZT8gfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgaW50IGdldEFjY2Vzc2libGVJbmRleEluUGFyZW50KCkgeyB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgKiB0cnkgeyBpZiAoZ2V0QWNjZXNzaWJsZVBhcmVudCgpID09IG51bGwpIHsgcmV0dXJuIC0xOyB9IGludCBjb3VudCA9IDA7Ci0gICAgICogQ29udGFpbmVyIHBhcmVudCA9IGdldFBhcmVudCgpOyBmb3IgKGludCBpID0gMDsgaSA8Ci0gICAgICogcGFyZW50LmdldENvbXBvbmVudENvdW50KCk7IGkrKykgeyBDb21wb25lbnQgYUNvbXAgPQotICAgICAqIHBhcmVudC5nZXRDb21wb25lbnQoaSk7IGlmIChhQ29tcCBpbnN0YW5jZW9mIEFjY2Vzc2libGUpIHsgaWYgKGFDb21wID09Ci0gICAgICogQ29tcG9uZW50LnRoaXMpIHsgcmV0dXJuIGNvdW50OyB9ICsrY291bnQ7IH0gfSByZXR1cm4gLTE7IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9Ci0gICAgICogQE92ZXJyaWRlIHB1YmxpYyBBY2Nlc3NpYmxlUm9sZSBnZXRBY2Nlc3NpYmxlUm9sZSgpIHsgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICogdHJ5IHsgcmV0dXJuIEFjY2Vzc2libGVSb2xlLkFXVF9DT01QT05FTlQ7IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9Ci0gICAgICogQE92ZXJyaWRlIHB1YmxpYyBBY2Nlc3NpYmxlU3RhdGVTZXQgZ2V0QWNjZXNzaWJsZVN0YXRlU2V0KCkgewotICAgICAqIHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyBBY2Nlc3NpYmxlU3RhdGVTZXQgc2V0ID0gbmV3Ci0gICAgICogQWNjZXNzaWJsZVN0YXRlU2V0KCk7IGlmIChpc0VuYWJsZWQoKSkgewotICAgICAqIHNldC5hZGQoQWNjZXNzaWJsZVN0YXRlLkVOQUJMRUQpOyB9IGlmIChpc0ZvY3VzYWJsZSgpKSB7Ci0gICAgICogc2V0LmFkZChBY2Nlc3NpYmxlU3RhdGUuRk9DVVNBQkxFKTsgfSBpZiAoaGFzRm9jdXMoKSkgewotICAgICAqIHNldC5hZGQoQWNjZXNzaWJsZVN0YXRlLkZPQ1VTRUQpOyB9IGlmIChpc09wYXF1ZSgpKSB7Ci0gICAgICogc2V0LmFkZChBY2Nlc3NpYmxlU3RhdGUuT1BBUVVFKTsgfSBpZiAoaXNTaG93aW5nKCkpIHsKLSAgICAgKiBzZXQuYWRkKEFjY2Vzc2libGVTdGF0ZS5TSE9XSU5HKTsgfSBpZiAoaXNWaXNpYmxlKCkpIHsKLSAgICAgKiBzZXQuYWRkKEFjY2Vzc2libGVTdGF0ZS5WSVNJQkxFKTsgfSByZXR1cm4gc2V0OyB9IGZpbmFsbHkgewotICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgTG9jYWxlIGdldExvY2FsZSgpIHRocm93cyBJbGxlZ2FsQ29tcG9uZW50U3RhdGVFeGNlcHRpb24KLSAgICAgKiB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4gQ29tcG9uZW50LnRoaXMuZ2V0TG9jYWxlKCk7IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IH0KLSAgICAgKi8KLSAgICAvKioKLSAgICAgKiBUaGUgQmx0QnVmZmVyU3RyYXRlZ3kgY2xhc3MgcHJvdmlkZXMgb3Bwb3J0dW5pdHkgb2YgYmxpdHRpbmcgb2Zmc2NyZWVuCi0gICAgICogc3VyZmFjZXMgdG8gYSBjb21wb25lbnQuIEZvciBtb3JlIGluZm9ybWF0aW9uIG9uIGJsaXR0aW5nLCBzZWUgPGEKLSAgICAgKiBocmVmPSJodHRwOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0JpdF9ibGl0Ij5CaXQgYmxpdDwvYT4uCi0gICAgICogCi0gICAgICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGNsYXNzIEJsdEJ1ZmZlclN0cmF0ZWd5IGV4dGVuZHMgQnVmZmVyU3RyYXRlZ3kgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgYmFjayBidWZmZXJzLgotICAgICAgICAgKi8KLSAgICAgICAgcHJvdGVjdGVkIFZvbGF0aWxlSW1hZ2VbXSBiYWNrQnVmZmVyczsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGNhcHMuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgQnVmZmVyQ2FwYWJpbGl0aWVzIGNhcHM7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB3aWR0aC4KLSAgICAgICAgICovCi0gICAgICAgIHByb3RlY3RlZCBpbnQgd2lkdGg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBoZWlnaHQuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgaW50IGhlaWdodDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHZhbGlkYXRlZCBjb250ZW50cy4KLSAgICAgICAgICovCi0gICAgICAgIHByb3RlY3RlZCBib29sZWFuIHZhbGlkYXRlZENvbnRlbnRzOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQmx0QnVmZmVyU3RyYXRlZ3kgYnVmZmVyIHN0cmF0ZWd5LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIG51bUJ1ZmZlcnMKLSAgICAgICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGJ1ZmZlcnMuCi0gICAgICAgICAqIEBwYXJhbSBjYXBzCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIEJ1ZmZlckNhcGFiaWxpdGllcy4KLSAgICAgICAgICogQHRocm93cyBOb3RJbXBsZW1lbnRlZEV4Y2VwdGlvbgotICAgICAgICAgKiAgICAgICAgICAgICB0aGUgbm90IGltcGxlbWVudGVkIGV4Y2VwdGlvbi4KLSAgICAgICAgICovCi0gICAgICAgIHByb3RlY3RlZCBCbHRCdWZmZXJTdHJhdGVneShpbnQgbnVtQnVmZmVycywgQnVmZmVyQ2FwYWJpbGl0aWVzIGNhcHMpCi0gICAgICAgICAgICAgICAgdGhyb3dzIG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuTm90SW1wbGVtZW50ZWRFeGNlcHRpb24gewotICAgICAgICAgICAgaWYgKHRydWUpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyB0cnVlIGlmIHRoZSBkcmF3aW5nIGJ1ZmZlciBoYXMgYmVlbiBsb3N0IHNpbmNlIHRoZSBsYXN0IGNhbGwKLSAgICAgICAgICogdG8gZ2V0RHJhd0dyYXBoaWNzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0cnVlIGlmIHRoZSBkcmF3aW5nIGJ1ZmZlciBoYXMgYmVlbiBsb3N0IHNpbmNlIHRoZSBsYXN0IGNhbGwKLSAgICAgICAgICogICAgICAgICB0byBnZXREcmF3R3JhcGhpY3MsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgICAgICogQHNlZSBqYXZhLmF3dC5pbWFnZS5CdWZmZXJTdHJhdGVneSNjb250ZW50c0xvc3QoKQotICAgICAgICAgKi8KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGNvbnRlbnRzTG9zdCgpIHsKLSAgICAgICAgICAgIGlmICh0cnVlKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgZHJhd2luZyBidWZmZXIgaGFzIGJlZW4gcmVzdG9yZWQgZnJvbSBhIGxvc3QKLSAgICAgICAgICogc3RhdGUgYW5kIHJlaW5pdGlhbGl6ZWQgdG8gdGhlIGRlZmF1bHQgYmFja2dyb3VuZCBjb2xvci4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgZHJhd2luZyBidWZmZXIgaGFzIGJlZW4gcmVzdG9yZWQgZnJvbSBhIGxvc3QKLSAgICAgICAgICogICAgICAgICBzdGF0ZSBhbmQgcmVpbml0aWFsaXplZCB0byB0aGUgZGVmYXVsdCBiYWNrZ3JvdW5kIGNvbG9yLAotICAgICAgICAgKiAgICAgICAgIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgICAgICogQHNlZSBqYXZhLmF3dC5pbWFnZS5CdWZmZXJTdHJhdGVneSNjb250ZW50c1Jlc3RvcmVkKCkKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBjb250ZW50c1Jlc3RvcmVkKCkgewotICAgICAgICAgICAgaWYgKHRydWUpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ3JlYXRlcyB0aGUgYmFjayBidWZmZXJzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIG51bUJ1ZmZlcnMKLSAgICAgICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGJ1ZmZlcnMuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgdm9pZCBjcmVhdGVCYWNrQnVmZmVycyhpbnQgbnVtQnVmZmVycykgewotICAgICAgICAgICAgaWYgKHRydWUpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyB0aGUgQnVmZmVyQ2FwYWJpbGl0aWVzIG9mIHRoZSBidWZmZXIgc3RyYXRlZ3kuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBCdWZmZXJDYXBhYmlsaXRpZXMuCi0gICAgICAgICAqIEBzZWUgamF2YS5hd3QuaW1hZ2UuQnVmZmVyU3RyYXRlZ3kjZ2V0Q2FwYWJpbGl0aWVzKCkKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgQnVmZmVyQ2FwYWJpbGl0aWVzIGdldENhcGFiaWxpdGllcygpIHsKLSAgICAgICAgICAgIHJldHVybiAoQnVmZmVyQ2FwYWJpbGl0aWVzKWNhcHMuY2xvbmUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBHZXRzIEdyYXBoaWNzIG9mIGN1cnJlbnQgYnVmZmVyIHN0cmF0ZWd5LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgR3JhcGhpY3Mgb2YgY3VycmVudCBidWZmZXIgc3RyYXRlZ3kuCi0gICAgICAgICAqIEBzZWUgamF2YS5hd3QuaW1hZ2UuQnVmZmVyU3RyYXRlZ3kjZ2V0RHJhd0dyYXBoaWNzKCkKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgR3JhcGhpY3MgZ2V0RHJhd0dyYXBoaWNzKCkgewotICAgICAgICAgICAgaWYgKHRydWUpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBSZXZhbGlkYXRlcyB0aGUgbG9zdCBkcmF3aW5nIGJ1ZmZlci4KLSAgICAgICAgICovCi0gICAgICAgIHByb3RlY3RlZCB2b2lkIHJldmFsaWRhdGUoKSB7Ci0gICAgICAgICAgICBpZiAodHJ1ZSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJNZXRob2QgaXMgbm90IGltcGxlbWVudGVkIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBTaG93cyB0aGUgbmV4dCBhdmFpbGFibGUgYnVmZmVyLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHNlZSBqYXZhLmF3dC5pbWFnZS5CdWZmZXJTdHJhdGVneSNzaG93KCkKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzaG93KCkgewotICAgICAgICAgICAgaWYgKHRydWUpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgRmxpcEJ1ZmZlclN0cmF0ZWd5IGNsYXNzIGlzIGZvciBmbGlwcGluZyBidWZmZXJzIG9uIGEgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICAgICAqLwotICAgIHByb3RlY3RlZCBjbGFzcyBGbGlwQnVmZmVyU3RyYXRlZ3kgZXh0ZW5kcyBCdWZmZXJTdHJhdGVneSB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBCdWZmZXIgQ2FwYWJpbGl0aWVzLgotICAgICAgICAgKi8KLSAgICAgICAgcHJvdGVjdGVkIEJ1ZmZlckNhcGFiaWxpdGllcyBjYXBzOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgZHJhd2luZyBidWZmZXIuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgSW1hZ2UgZHJhd0J1ZmZlcjsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGRyYXdpbmcgVm9sYXRpbGVJbWFnZSBidWZmZXIuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgVm9sYXRpbGVJbWFnZSBkcmF3VkJ1ZmZlcjsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG51bWJlciBvZiBidWZmZXJzLgotICAgICAgICAgKi8KLSAgICAgICAgcHJvdGVjdGVkIGludCBudW1CdWZmZXJzOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgdmFsaWRhdGVkIGNvbnRlbnRzIGluZGljYXRlcyBpZiB0aGUgZHJhd2luZyBidWZmZXIgaXMgcmVzdG9yZWQKLSAgICAgICAgICogZnJvbSBsb3N0IHN0YXRlLgotICAgICAgICAgKi8KLSAgICAgICAgcHJvdGVjdGVkIGJvb2xlYW4gdmFsaWRhdGVkQ29udGVudHM7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBmbGlwIGJ1ZmZlciBzdHJhdGVneS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBudW1CdWZmZXJzCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBidWZmZXJzLgotICAgICAgICAgKiBAcGFyYW0gY2FwcwotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBCdWZmZXJDYXBhYmlsaXRpZXMuCi0gICAgICAgICAqIEB0aHJvd3MgQVdURXhjZXB0aW9uCi0gICAgICAgICAqICAgICAgICAgICAgIGlmIHRoZSBjYXBhYmlsaXRpZXMgc3VwcGxpZWQgY291bGQgbm90IGJlIHN1cHBvcnRlZCBvcgotICAgICAgICAgKiAgICAgICAgICAgICBtZXQuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgRmxpcEJ1ZmZlclN0cmF0ZWd5KGludCBudW1CdWZmZXJzLCBCdWZmZXJDYXBhYmlsaXRpZXMgY2FwcykgdGhyb3dzIEFXVEV4Y2VwdGlvbiB7Ci0gICAgICAgICAgICAvLyA/Pz9BV1QKLSAgICAgICAgICAgIC8qCi0gICAgICAgICAgICAgKiBpZiAoIShDb21wb25lbnQudGhpcyBpbnN0YW5jZW9mIFdpbmRvdykgJiYgIShDb21wb25lbnQudGhpcwotICAgICAgICAgICAgICogaW5zdGFuY2VvZiBDYW52YXMpKSB7IC8vIGF3dC4xNEI9T25seSBDYW52YXMgb3IgV2luZG93IGlzIGFsbG93ZWQKLSAgICAgICAgICAgICAqIHRocm93IG5ldyBDbGFzc0Nhc3RFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTRCIikpOwotICAgICAgICAgICAgICogLy8kTk9OLU5MUy0xJCB9Ci0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIC8vIFRPRE86IHRocm93IG5ldyBBV1RFeGNlcHRpb24oIkNhcGFiaWxpdGllcyBhcmUgbm90IHN1cHBvcnRlZCIpOwotICAgICAgICAgICAgdGhpcy5udW1CdWZmZXJzID0gbnVtQnVmZmVyczsKLSAgICAgICAgICAgIHRoaXMuY2FwcyA9IChCdWZmZXJDYXBhYmlsaXRpZXMpY2Fwcy5jbG9uZSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgZHJhd2luZyBidWZmZXIgaGFzIGJlZW4gbG9zdCBzaW5jZSB0aGUgbGFzdCBjYWxsCi0gICAgICAgICAqIHRvIGdldERyYXdHcmFwaGljcy4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgZHJhd2luZyBidWZmZXIgaGFzIGJlZW4gbG9zdCBzaW5jZSB0aGUgbGFzdCBjYWxsCi0gICAgICAgICAqICAgICAgICAgdG8gZ2V0RHJhd0dyYXBoaWNzLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICAgICAqIEBzZWUgamF2YS5hd3QuaW1hZ2UuQnVmZmVyU3RyYXRlZ3kjY29udGVudHNMb3N0KCkKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBjb250ZW50c0xvc3QoKSB7Ci0gICAgICAgICAgICBpZiAodHJ1ZSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJNZXRob2QgaXMgbm90IGltcGxlbWVudGVkIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIGRyYXdpbmcgYnVmZmVyIGhhcyBiZWVuIHJlc3RvcmVkIGZyb20gYSBsb3N0Ci0gICAgICAgICAqIHN0YXRlIGFuZCByZWluaXRpYWxpemVkIHRvIHRoZSBkZWZhdWx0IGJhY2tncm91bmQgY29sb3IuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRydWUgaWYgdGhlIGRyYXdpbmcgYnVmZmVyIGhhcyBiZWVuIHJlc3RvcmVkIGZyb20gYSBsb3N0Ci0gICAgICAgICAqICAgICAgICAgc3RhdGUgYW5kIHJlaW5pdGlhbGl6ZWQgdG8gdGhlIGRlZmF1bHQgYmFja2dyb3VuZCBjb2xvciwKLSAgICAgICAgICogICAgICAgICBmYWxzZSBvdGhlcndpc2UuCi0gICAgICAgICAqIEBzZWUgamF2YS5hd3QuaW1hZ2UuQnVmZmVyU3RyYXRlZ3kjY29udGVudHNSZXN0b3JlZCgpCi0gICAgICAgICAqLwotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGJvb2xlYW4gY29udGVudHNSZXN0b3JlZCgpIHsKLSAgICAgICAgICAgIGlmICh0cnVlKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENyZWF0ZXMgZmxpcHBpbmcgYnVmZmVycyB3aXRoIHRoZSBzcGVjaWZpZWQgYnVmZmVyIGNhcGFiaWxpdGllcy4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBudW1CdWZmZXJzCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBidWZmZXJzLgotICAgICAgICAgKiBAcGFyYW0gY2FwcwotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBCdWZmZXJDYXBhYmlsaXRpZXMuCi0gICAgICAgICAqIEB0aHJvd3MgQVdURXhjZXB0aW9uCi0gICAgICAgICAqICAgICAgICAgICAgIGlmIHRoZSBjYXBhYmlsaXRpZXMgY291bGQgbm90IGJlIHN1cHBvcnRlZCBvciBtZXQuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgdm9pZCBjcmVhdGVCdWZmZXJzKGludCBudW1CdWZmZXJzLCBCdWZmZXJDYXBhYmlsaXRpZXMgY2FwcykgdGhyb3dzIEFXVEV4Y2VwdGlvbiB7Ci0gICAgICAgICAgICBpZiAobnVtQnVmZmVycyA8IDIpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMTRDPU51bWJlciBvZiBidWZmZXJzIG11c3QgYmUgZ3JlYXRlciB0aGFuIG9uZQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTRDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoIWNhcHMuaXNQYWdlRmxpcHBpbmcoKSkgewotICAgICAgICAgICAgICAgIC8vIGF3dC4xNEQ9QnVmZmVyIGNhcGFiaWxpdGllcyBzaG91bGQgc3VwcG9ydCBmbGlwcGluZwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTREIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoIUNvbXBvbmVudC50aGlzLmJlaGF2aW91ci5pc0Rpc3BsYXlhYmxlKCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMTRFPUNvbXBvbmVudCBzaG91bGQgYmUgZGlzcGxheWFibGUKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE0RSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgLy8gVE9ETzogdGhyb3cgbmV3IEFXVEV4Y2VwdGlvbigiQ2FwYWJpbGl0aWVzIGFyZSBub3Qgc3VwcG9ydGVkIik7Ci0gICAgICAgICAgICBpZiAodHJ1ZSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJNZXRob2QgaXMgbm90IGltcGxlbWVudGVkIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBEZXN0cm95IGJ1ZmZlcnMuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgdm9pZCBkZXN0cm95QnVmZmVycygpIHsKLSAgICAgICAgICAgIGlmICh0cnVlKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEZsaXBzIHRoZSBjb250ZW50cyBvZiB0aGUgYmFjayBidWZmZXIgdG8gdGhlIGZyb250IGJ1ZmZlci4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBmbGlwQWN0aW9uCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGZsaXAgYWN0aW9uLgotICAgICAgICAgKi8KLSAgICAgICAgcHJvdGVjdGVkIHZvaWQgZmxpcChCdWZmZXJDYXBhYmlsaXRpZXMuRmxpcENvbnRlbnRzIGZsaXBBY3Rpb24pIHsKLSAgICAgICAgICAgIGlmICh0cnVlKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEdldHMgdGhlIGJhY2sgYnVmZmVyIGFzIEltYWdlLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgYmFjayBidWZmZXIgYXMgSW1hZ2UuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgSW1hZ2UgZ2V0QmFja0J1ZmZlcigpIHsKLSAgICAgICAgICAgIGlmICh0cnVlKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyB0aGUgQnVmZmVyQ2FwYWJpbGl0aWVzIG9mIHRoZSBidWZmZXIgc3RyYXRlZ3kuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBCdWZmZXJDYXBhYmlsaXRpZXMuCi0gICAgICAgICAqIEBzZWUgamF2YS5hd3QuaW1hZ2UuQnVmZmVyU3RyYXRlZ3kjZ2V0Q2FwYWJpbGl0aWVzKCkKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgQnVmZmVyQ2FwYWJpbGl0aWVzIGdldENhcGFiaWxpdGllcygpIHsKLSAgICAgICAgICAgIHJldHVybiAoQnVmZmVyQ2FwYWJpbGl0aWVzKWNhcHMuY2xvbmUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBHZXRzIEdyYXBoaWNzIG9mIGN1cnJlbnQgYnVmZmVyIHN0cmF0ZWd5LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgR3JhcGhpY3Mgb2YgY3VycmVudCBidWZmZXIgc3RyYXRlZ3kuCi0gICAgICAgICAqIEBzZWUgamF2YS5hd3QuaW1hZ2UuQnVmZmVyU3RyYXRlZ3kjZ2V0RHJhd0dyYXBoaWNzKCkKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgR3JhcGhpY3MgZ2V0RHJhd0dyYXBoaWNzKCkgewotICAgICAgICAgICAgaWYgKHRydWUpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBSZXZhbGlkYXRlcyB0aGUgbG9zdCBkcmF3aW5nIGJ1ZmZlci4KLSAgICAgICAgICovCi0gICAgICAgIHByb3RlY3RlZCB2b2lkIHJldmFsaWRhdGUoKSB7Ci0gICAgICAgICAgICBpZiAodHJ1ZSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJNZXRob2QgaXMgbm90IGltcGxlbWVudGVkIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBTaG93cyB0aGUgbmV4dCBhdmFpbGFibGUgYnVmZmVyLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHNlZSBqYXZhLmF3dC5pbWFnZS5CdWZmZXJTdHJhdGVneSNzaG93KCkKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzaG93KCkgewotICAgICAgICAgICAgaWYgKHRydWUpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaW50ZXJuYWwgY29tcG9uZW50J3Mgc3RhdGUgdXRpbGl6ZWQgYnkgdGhlIHZpc3VhbCB0aGVtZS4KLSAgICAgKi8KLSAgICBjbGFzcyBDb21wb25lbnRTdGF0ZSBpbXBsZW1lbnRzIFN0YXRlIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGRlZmF1bHQgbWluaW11bSBzaXplLgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBEaW1lbnNpb24gZGVmYXVsdE1pbmltdW1TaXplID0gbmV3IERpbWVuc2lvbigpOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDaGVja3MgaWYgdGhlIGNvbXBvbmVudCBpcyBlbmFibGVkLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgY29tcG9uZW50IGlzIGVuYWJsZWQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgYm9vbGVhbiBpc0VuYWJsZWQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gZW5hYmxlZDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDaGVja3MgaWYgdGhlIGNvbXBvbmVudCBpcyB2aXNpYmxlLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgY29tcG9uZW50IGlzIHZpc2libGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgYm9vbGVhbiBpc1Zpc2libGUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gdmlzaWJsZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDaGVja3MgaWYgaXMgZm9jdXNlZC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgaXMgZm9jdXNlZC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzRm9jdXNlZCgpIHsKLSAgICAgICAgICAgIC8vID8/P0FXVDogcmV0dXJuIGlzRm9jdXNPd25lcigpOwotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEdldHMgdGhlIGZvbnQuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBmb250LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEZvbnQgZ2V0Rm9udCgpIHsKLSAgICAgICAgICAgIHJldHVybiBDb21wb25lbnQudGhpcy5nZXRGb250KCk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ2hlY2tzIGlmIHRoZSBmb250IGhhcyBiZWVuIHNldC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIGZvbnQgaGFzIGJlZW4gc2V0LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGJvb2xlYW4gaXNGb250U2V0KCkgewotICAgICAgICAgICAgcmV0dXJuIGZvbnQgIT0gbnVsbDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBHZXRzIHRoZSBiYWNrZ3JvdW5kIGNvbG9yLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgYmFja2dyb3VuZCBjb2xvci4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBDb2xvciBnZXRCYWNrZ3JvdW5kKCkgewotICAgICAgICAgICAgQ29sb3IgYyA9IENvbXBvbmVudC50aGlzLmdldEJhY2tncm91bmQoKTsKLSAgICAgICAgICAgIHJldHVybiAoYyAhPSBudWxsKSA/IGMgOiBnZXREZWZhdWx0QmFja2dyb3VuZCgpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENoZWNrcyBpZiB0aGUgYmFja2dyb3VuZCBpcyBzZXQuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBiYWNrZ3JvdW5kIGlzIHNldC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzQmFja2dyb3VuZFNldCgpIHsKLSAgICAgICAgICAgIHJldHVybiBiYWNrQ29sb3IgIT0gbnVsbDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBHZXRzIHRoZSB0ZXh0IGNvbG9yLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgdGV4dCBjb2xvci4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBDb2xvciBnZXRUZXh0Q29sb3IoKSB7Ci0gICAgICAgICAgICBDb2xvciBjID0gZ2V0Rm9yZWdyb3VuZCgpOwotICAgICAgICAgICAgcmV0dXJuIChjICE9IG51bGwpID8gYyA6IGdldERlZmF1bHRGb3JlZ3JvdW5kKCk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ2hlY2tzIGlmIHRoZSB0ZXh0IGNvbG9yIGlzIHNldC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHRleHQgY29sb3IgaXMgc2V0LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGJvb2xlYW4gaXNUZXh0Q29sb3JTZXQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gZm9yZUNvbG9yICE9IG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogR2V0cyB0aGUgZm9udCBtZXRyaWNzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgZm9udCBtZXRyaWNzLgotICAgICAgICAgKi8KLSAgICAgICAgQFN1cHByZXNzV2FybmluZ3MoImRlcHJlY2F0aW9uIikKLSAgICAgICAgcHVibGljIEZvbnRNZXRyaWNzIGdldEZvbnRNZXRyaWNzKCkgewotICAgICAgICAgICAgcmV0dXJuIHRvb2xraXQuZ2V0Rm9udE1ldHJpY3MoQ29tcG9uZW50LnRoaXMuZ2V0Rm9udCgpKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBHZXRzIHRoZSBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldEJvdW5kcygpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlKHgsIHksIHcsIGgpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEdldHMgdGhlIHNpemUgb2YgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdGhlIHNpemUgb2YgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBEaW1lbnNpb24gZ2V0U2l6ZSgpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgRGltZW5zaW9uKHcsIGgpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEdldHMgdGhlIHdpbmRvdyBpZC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdGhlIHdpbmRvdyBpZC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBsb25nIGdldFdpbmRvd0lkKCkgewotICAgICAgICAgICAgTmF0aXZlV2luZG93IHdpbiA9IGdldE5hdGl2ZVdpbmRvdygpOwotICAgICAgICAgICAgcmV0dXJuICh3aW4gIT0gbnVsbCkgPyB3aW4uZ2V0SWQoKSA6IDA7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogR2V0cyB0aGUgZGVmYXVsdCBtaW5pbXVtIHNpemUuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBkZWZhdWx0IG1pbmltdW0gc2l6ZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBEaW1lbnNpb24gZ2V0RGVmYXVsdE1pbmltdW1TaXplKCkgewotICAgICAgICAgICAgaWYgKGRlZmF1bHRNaW5pbXVtU2l6ZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgY2FsY3VsYXRlKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gZGVmYXVsdE1pbmltdW1TaXplOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFNldHMgdGhlIGRlZmF1bHQgbWluaW11bSBzaXplLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHNpemUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgbmV3IGRlZmF1bHQgbWluaW11bSBzaXplLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIHZvaWQgc2V0RGVmYXVsdE1pbmltdW1TaXplKERpbWVuc2lvbiBzaXplKSB7Ci0gICAgICAgICAgICBkZWZhdWx0TWluaW11bVNpemUgPSBzaXplOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFJlc2V0IHRoZSBkZWZhdWx0IG1pbmltdW0gc2l6ZSB0byBudWxsLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIHZvaWQgcmVzZXQoKSB7Ci0gICAgICAgICAgICBkZWZhdWx0TWluaW11bVNpemUgPSBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENhbGN1bGF0ZSB0aGUgZGVmYXVsdCBtaW5pbXVtIHNpemU6IHRvIGJlIG92ZXJyaWRkZW4uCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgdm9pZCBjYWxjdWxhdGUoKSB7Ci0gICAgICAgICAgICAvLyB0byBiZSBvdmVycmlkZGVuCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyA/Pz9BV1Q6IHByaXZhdGUgdHJhbnNpZW50IEFjY2Vzc2libGVDb250ZXh0IGFjY2Vzc2libGVDb250ZXh0OwotCi0gICAgLyoqCi0gICAgICogVGhlIGJlaGF2aW91ci4KLSAgICAgKi8KLSAgICBmaW5hbCB0cmFuc2llbnQgQ29tcG9uZW50QmVoYXZpb3IgYmVoYXZpb3VyOwotCi0gICAgLy8gPz8/QVdUOiBDb250YWluZXIgcGFyZW50OwotCi0gICAgLyoqCi0gICAgICogVGhlIG5hbWUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBTdHJpbmcgbmFtZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBhdXRvIG5hbWUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGF1dG9OYW1lID0gdHJ1ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBmb250LgotICAgICAqLwotICAgIHByaXZhdGUgRm9udCBmb250OwotCi0gICAgLyoqCi0gICAgICogVGhlIGJhY2sgY29sb3IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBDb2xvciBiYWNrQ29sb3I7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZm9yZSBjb2xvci4KLSAgICAgKi8KLSAgICBwcml2YXRlIENvbG9yIGZvcmVDb2xvcjsKLQotICAgIC8qKgotICAgICAqIFRoZSBkZXByZWNhdGVkIGV2ZW50IGhhbmRsZXIuCi0gICAgICovCi0gICAgYm9vbGVhbiBkZXByZWNhdGVkRXZlbnRIYW5kbGVyID0gdHJ1ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBlbmFibGVkIGV2ZW50cy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGxvbmcgZW5hYmxlZEV2ZW50czsKLQotICAgIC8qKgotICAgICAqIFRoZSBlbmFibGVkIEFXVCBldmVudHMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBsb25nIGVuYWJsZWRBV1RFdmVudHM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29tcG9uZW50IGxpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIEFXVExpc3RlbmVyTGlzdDxDb21wb25lbnRMaXN0ZW5lcj4gY29tcG9uZW50TGlzdGVuZXJzID0gbmV3IEFXVExpc3RlbmVyTGlzdDxDb21wb25lbnRMaXN0ZW5lcj4oCi0gICAgICAgICAgICB0aGlzKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBmb2N1cyBsaXN0ZW5lcnMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBBV1RMaXN0ZW5lckxpc3Q8Rm9jdXNMaXN0ZW5lcj4gZm9jdXNMaXN0ZW5lcnMgPSBuZXcgQVdUTGlzdGVuZXJMaXN0PEZvY3VzTGlzdGVuZXI+KAotICAgICAgICAgICAgdGhpcyk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGllcmFyY2h5IGxpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIEFXVExpc3RlbmVyTGlzdDxIaWVyYXJjaHlMaXN0ZW5lcj4gaGllcmFyY2h5TGlzdGVuZXJzID0gbmV3IEFXVExpc3RlbmVyTGlzdDxIaWVyYXJjaHlMaXN0ZW5lcj4oCi0gICAgICAgICAgICB0aGlzKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBoaWVyYXJjaHkgYm91bmRzIGxpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIEFXVExpc3RlbmVyTGlzdDxIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lcj4gaGllcmFyY2h5Qm91bmRzTGlzdGVuZXJzID0gbmV3IEFXVExpc3RlbmVyTGlzdDxIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lcj4oCi0gICAgICAgICAgICB0aGlzKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBrZXkgbGlzdGVuZXJzLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgQVdUTGlzdGVuZXJMaXN0PEtleUxpc3RlbmVyPiBrZXlMaXN0ZW5lcnMgPSBuZXcgQVdUTGlzdGVuZXJMaXN0PEtleUxpc3RlbmVyPih0aGlzKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBtb3VzZSBsaXN0ZW5lcnMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBBV1RMaXN0ZW5lckxpc3Q8TW91c2VMaXN0ZW5lcj4gbW91c2VMaXN0ZW5lcnMgPSBuZXcgQVdUTGlzdGVuZXJMaXN0PE1vdXNlTGlzdGVuZXI+KAotICAgICAgICAgICAgdGhpcyk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbW91c2UgbW90aW9uIGxpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIEFXVExpc3RlbmVyTGlzdDxNb3VzZU1vdGlvbkxpc3RlbmVyPiBtb3VzZU1vdGlvbkxpc3RlbmVycyA9IG5ldyBBV1RMaXN0ZW5lckxpc3Q8TW91c2VNb3Rpb25MaXN0ZW5lcj4oCi0gICAgICAgICAgICB0aGlzKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBtb3VzZSB3aGVlbCBsaXN0ZW5lcnMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBBV1RMaXN0ZW5lckxpc3Q8TW91c2VXaGVlbExpc3RlbmVyPiBtb3VzZVdoZWVsTGlzdGVuZXJzID0gbmV3IEFXVExpc3RlbmVyTGlzdDxNb3VzZVdoZWVsTGlzdGVuZXI+KAotICAgICAgICAgICAgdGhpcyk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaW5wdXQgbWV0aG9kIGxpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIEFXVExpc3RlbmVyTGlzdDxJbnB1dE1ldGhvZExpc3RlbmVyPiBpbnB1dE1ldGhvZExpc3RlbmVycyA9IG5ldyBBV1RMaXN0ZW5lckxpc3Q8SW5wdXRNZXRob2RMaXN0ZW5lcj4oCi0gICAgICAgICAgICB0aGlzKTsKLQotICAgIC8qKgotICAgICAqIFRoZSB4LgotICAgICAqLwotICAgIGludCB4OwotCi0gICAgLyoqCi0gICAgICogVGhlIHkuCi0gICAgICovCi0gICAgaW50IHk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdy4KLSAgICAgKi8KLSAgICBpbnQgdzsKLQotICAgIC8qKgotICAgICAqIFRoZSBoLgotICAgICAqLwotICAgIGludCBoOwotCi0gICAgLyoqCi0gICAgICogVGhlIG1heGltdW0gc2l6ZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIERpbWVuc2lvbiBtYXhpbXVtU2l6ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBtaW5pbXVtIHNpemUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBEaW1lbnNpb24gbWluaW11bVNpemU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcHJlZmVycmVkIHNpemUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBEaW1lbnNpb24gcHJlZmVycmVkU2l6ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBib3VuZHMgbWFzayBwYXJhbS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBib3VuZHNNYXNrUGFyYW07Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaWdub3JlIHJlcGFpbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGlnbm9yZVJlcGFpbnQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZW5hYmxlZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gZW5hYmxlZCA9IHRydWU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaW5wdXQgbWV0aG9kcyBlbmFibGVkLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBpbnB1dE1ldGhvZHNFbmFibGVkID0gdHJ1ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBkaXNwYXRjaCB0byBpbS4KLSAgICAgKi8KLSAgICB0cmFuc2llbnQgYm9vbGVhbiBkaXNwYXRjaFRvSU0gPSB0cnVlOwotCi0gICAgLyoqCi0gICAgICogVGhlIGZvY3VzYWJsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gZm9jdXNhYmxlID0gdHJ1ZTsgLy8gQnkgZGVmYXVsdCwgYWxsIENvbXBvbmVudHMgcmV0dXJuCi0KLSAgICAvLyB0cnVlIGZyb20gaXNGb2N1c2FibGUoKSBtZXRob2QKLSAgICAvKioKLSAgICAgKiBUaGUgdmlzaWJsZS4KLSAgICAgKi8KLSAgICBib29sZWFuIHZpc2libGUgPSB0cnVlOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNhbGxlZCBzZXQgZm9jdXNhYmxlLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBjYWxsZWRTZXRGb2N1c2FibGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgb3ZlcnJpZGRlbiBpcyBmb2N1c2FibGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIG92ZXJyaWRlbklzRm9jdXNhYmxlID0gdHJ1ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBmb2N1cyB0cmF2ZXJzYWwga2V5cyBlbmFibGVkLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBmb2N1c1RyYXZlcnNhbEtleXNFbmFibGVkID0gdHJ1ZTsKLQotICAgIC8qKgotICAgICAqIFBvc3NpYmxlIGtleXMgYXJlOiBGT1JXQVJEX1RSQVZFUlNBTF9LRVlTLCBCQUNLV0FSRF9UUkFWRVJTQUxfS0VZUywKLSAgICAgKiBVUF9DWUNMRV9UUkFWRVJTQUxfS0VZUy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIE1hcDxJbnRlZ2VyLCBTZXQ8PyBleHRlbmRzIEFXVEtleVN0cm9rZT4+IHRyYXZlcnNhbEtleXMgPSBuZXcgSGFzaE1hcDxJbnRlZ2VyLCBTZXQ8PyBleHRlbmRzIEFXVEtleVN0cm9rZT4+KCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdHJhdmVyc2FsIGkgZHMuCi0gICAgICovCi0gICAgaW50W10gdHJhdmVyc2FsSURzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGxvY2FsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIExvY2FsZSBsb2NhbGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgb3JpZW50YXRpb24uCi0gICAgICovCi0gICAgcHJpdmF0ZSBDb21wb25lbnRPcmllbnRhdGlvbiBvcmllbnRhdGlvbjsKLQotICAgIC8qKgotICAgICAqIFRoZSBwcm9wZXJ0eSBjaGFuZ2Ugc3VwcG9ydC4KLSAgICAgKi8KLSAgICBwcml2YXRlIFByb3BlcnR5Q2hhbmdlU3VwcG9ydCBwcm9wZXJ0eUNoYW5nZVN1cHBvcnQ7Ci0KLSAgICAvLyA/Pz9BV1Q6IHByaXZhdGUgQXJyYXlMaXN0PFBvcHVwTWVudT4gcG9wdXBzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvYWxlc2Nlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gY29hbGVzY2VyOwotCi0gICAgLyoqCi0gICAgICogVGhlIGV2ZW50cyB0YWJsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIEhhc2h0YWJsZTxJbnRlZ2VyLCBMaW5rZWRMaXN0PEFXVEV2ZW50Pj4gZXZlbnRzVGFibGU7Ci0KLSAgICAvKioKLSAgICAgKiBDYXNoZWQgcmVmZXJlbmNlIHVzZWQgZHVyaW5nIEV2ZW50UXVldWUucG9zdEV2ZW50KCkKLSAgICAgKi8KLSAgICBwcml2YXRlIExpbmtlZExpc3Q8QVdURXZlbnQ+IGV2ZW50c0xpc3Q7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGllcmFyY2h5IGNoYW5naW5nIGNvdW50ZXIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgaGllcmFyY2h5Q2hhbmdpbmdDb3VudGVyOwotCi0gICAgLyoqCi0gICAgICogVGhlIHdhcyBzaG93aW5nLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiB3YXNTaG93aW5nOwotCi0gICAgLyoqCi0gICAgICogVGhlIHdhcyBkaXNwbGF5YWJsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gd2FzRGlzcGxheWFibGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY3Vyc29yLgotICAgICAqLwotICAgIEN1cnNvciBjdXJzb3I7Ci0KLSAgICAvLyA/Pz9BV1Q6IERyb3BUYXJnZXQgZHJvcFRhcmdldDsKLQotICAgIC8qKgotICAgICAqIFRoZSBtb3VzZSBleGl0ZWQgZXhwZWN0ZWQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIG1vdXNlRXhpdGVkRXhwZWN0ZWQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcmVwYWludCByZWdpb24uCi0gICAgICovCi0gICAgdHJhbnNpZW50IE11bHRpUmVjdEFyZWEgcmVwYWludFJlZ2lvbjsKLQotICAgIC8vID8/P0FXVDogdHJhbnNpZW50IFJlZHJhd01hbmFnZXIgcmVkcmF3TWFuYWdlcjsKLSAgICAvKioKLSAgICAgKiBUaGUgcmVkcmF3IG1hbmFnZXIuCi0gICAgICovCi0gICAgdHJhbnNpZW50IE9iamVjdCByZWRyYXdNYW5hZ2VyOwotCi0gICAgLyoqCi0gICAgICogVGhlIHZhbGlkLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiB2YWxpZDsKLQotICAgIC8qKgotICAgICAqIFRoZSB1cGRhdGVkIGltYWdlcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIEhhc2hNYXA8SW1hZ2UsIEltYWdlUGFyYW1ldGVycz4gdXBkYXRlZEltYWdlczsKLQotICAgIC8qKgotICAgICAqIFRoZSBsb2NrIG9iamVjdCBmb3IgcHJpdmF0ZSBjb21wb25lbnQncyBkYXRhIHdoaWNoIGRvbid0IGFmZmVjdCB0aGUKLSAgICAgKiBjb21wb25lbnQgaGllcmFyY2h5LgotICAgICAqLwotICAgIHByaXZhdGUgY2xhc3MgQ29tcG9uZW50TG9jayB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbXBvbmVudCBsb2NrLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgdHJhbnNpZW50IE9iamVjdCBjb21wb25lbnRMb2NrID0gbmV3IENvbXBvbmVudExvY2soKTsKLSAgICBzdGF0aWMgewotICAgICAgICBQcml2aWxlZ2VkQWN0aW9uPFN0cmluZ1tdPiBhY3Rpb24gPSBuZXcgUHJpdmlsZWdlZEFjdGlvbjxTdHJpbmdbXT4oKSB7Ci0gICAgICAgICAgICBwdWJsaWMgU3RyaW5nW10gcnVuKCkgewotICAgICAgICAgICAgICAgIFN0cmluZyBwcm9wZXJ0aWVzW10gPSBuZXcgU3RyaW5nWzJdOwotICAgICAgICAgICAgICAgIHByb3BlcnRpZXNbMF0gPSBTeXN0ZW0uZ2V0UHJvcGVydHkoImF3dC5pbWFnZS5yZWRyYXdyYXRlIiwgIjEwMCIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgICAgICAgICBwcm9wZXJ0aWVzWzFdID0gU3lzdGVtLmdldFByb3BlcnR5KCJhd3QuaW1hZ2UuaW5jcmVtZW50YWxkcmF3IiwgInRydWUiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgcmV0dXJuIHByb3BlcnRpZXM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH07Ci0gICAgICAgIFN0cmluZyBwcm9wZXJ0aWVzW10gPSBBY2Nlc3NDb250cm9sbGVyLmRvUHJpdmlsZWdlZChhY3Rpb24pOwotICAgICAgICAvLyBGSVhNRTogcmF0ZSBpcyBuZXZlciB1c2VkLCBjYW4gdGhpcyBjb2RlIGFuZCB0aGUgZ2V0IHByb3BlcnR5IGFib3ZlCi0gICAgICAgIC8vIGJlIHJlbW92ZWQ/Ci0gICAgICAgIC8vIGludCByYXRlOwotICAgICAgICAvLwotICAgICAgICAvLyB0cnkgewotICAgICAgICAvLyByYXRlID0gSW50ZWdlci5kZWNvZGUocHJvcGVydGllc1swXSkuaW50VmFsdWUoKTsKLSAgICAgICAgLy8gfSBjYXRjaCAoTnVtYmVyRm9ybWF0RXhjZXB0aW9uIGUpIHsKLSAgICAgICAgLy8gcmF0ZSA9IDEwMDsKLSAgICAgICAgLy8gfQotICAgICAgICBpbmNyZW1lbnRhbEltYWdlVXBkYXRlID0gcHJvcGVydGllc1sxXS5lcXVhbHMoInRydWUiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIENvbXBvbmVudCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBvcmllbnRhdGlvbiA9IENvbXBvbmVudE9yaWVudGF0aW9uLlVOS05PV047Ci0gICAgICAgICAgICByZWRyYXdNYW5hZ2VyID0gbnVsbDsKLSAgICAgICAgICAgIC8vID8/P0FXVAotICAgICAgICAgICAgLyoKLSAgICAgICAgICAgICAqIHRyYXZlcnNhbElEcyA9IHRoaXMgaW5zdGFuY2VvZiBDb250YWluZXIgPwotICAgICAgICAgICAgICogS2V5Ym9hcmRGb2N1c01hbmFnZXIuY29udFRyYXZlcnNhbElEcyA6Ci0gICAgICAgICAgICAgKiBLZXlib2FyZEZvY3VzTWFuYWdlci5jb21wVHJhdmVyc2FsSURzOyBmb3IgKGludCBlbGVtZW50IDoKLSAgICAgICAgICAgICAqIHRyYXZlcnNhbElEcykgeyB0cmF2ZXJzYWxLZXlzLnB1dChuZXcgSW50ZWdlcihlbGVtZW50KSwgbnVsbCk7IH0KLSAgICAgICAgICAgICAqIGJlaGF2aW91ciA9IGNyZWF0ZUJlaGF2aW9yKCk7Ci0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIGJlaGF2aW91ciA9IG51bGw7Ci0KLSAgICAgICAgICAgIGRlcml2ZUNvYWxlc2NlckZsYWcoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXRlcm1pbmUgdGhhdCB0aGUgY2xhc3MgaW5oZXJpdGVkIGZyb20gQ29tcG9uZW50IGRlY2xhcmVzIHRoZSBtZXRob2QKLSAgICAgKiBjb2FsZXNjZUV2ZW50cygpLCBhbmQgcHV0IHRoZSByZXN1bHRzIHRvIHRoZSBjaGlsZENsYXNzZXNGbGFncyBtYXAuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIGRlcml2ZUNvYWxlc2NlckZsYWcoKSB7Ci0gICAgICAgIENsYXNzPD8+IHRoaXNDbGFzcyA9IGdldENsYXNzKCk7Ci0gICAgICAgIGJvb2xlYW4gZmxhZyA9IHRydWU7Ci0gICAgICAgIHN5bmNocm9uaXplZCAoY2hpbGRDbGFzc2VzRmxhZ3MpIHsKLSAgICAgICAgICAgIEJvb2xlYW4gZmxhZ1dyYXBwZXIgPSBjaGlsZENsYXNzZXNGbGFncy5nZXQodGhpc0NsYXNzKTsKLSAgICAgICAgICAgIGlmIChmbGFnV3JhcHBlciA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgTWV0aG9kIGNvYWxlc2NlTWV0aG9kID0gbnVsbDsKLSAgICAgICAgICAgICAgICBmb3IgKENsYXNzPD8+IGMgPSB0aGlzQ2xhc3M7IGMgIT0gQ29tcG9uZW50LmNsYXNzOyBjID0gYy5nZXRTdXBlcmNsYXNzKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvYWxlc2NlTWV0aG9kID0gYy5nZXREZWNsYXJlZE1ldGhvZCgiY29hbGVzY2VFdmVudHMiLCBuZXcgQ2xhc3NbXSB7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBDbGFzcy5mb3JOYW1lKCJqYXZhLmF3dC5BV1RFdmVudCIpLCAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ2xhc3MuZm9yTmFtZSgiamF2YS5hd3QuQVdURXZlbnQiKX0pOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWYgKGNvYWxlc2NlTWV0aG9kICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGZsYWcgPSAoY29hbGVzY2VNZXRob2QgIT0gbnVsbCk7Ci0gICAgICAgICAgICAgICAgY2hpbGRDbGFzc2VzRmxhZ3MucHV0KHRoaXNDbGFzcywgQm9vbGVhbi52YWx1ZU9mKGZsYWcpKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZmxhZyA9IGZsYWdXcmFwcGVyLmJvb2xlYW5WYWx1ZSgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGNvYWxlc2NlciA9IGZsYWc7Ci0gICAgICAgIGlmIChmbGFnKSB7Ci0gICAgICAgICAgICBldmVudHNUYWJsZSA9IG5ldyBIYXNodGFibGU8SW50ZWdlciwgTGlua2VkTGlzdDxBV1RFdmVudD4+KCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBldmVudHNUYWJsZSA9IG51bGw7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBuYW1lIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgbmFtZSBvZiB0aGUgQ29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldE5hbWUoU3RyaW5nIG5hbWUpIHsKLSAgICAgICAgU3RyaW5nIG9sZE5hbWU7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgYXV0b05hbWUgPSBmYWxzZTsKLSAgICAgICAgICAgIG9sZE5hbWUgPSB0aGlzLm5hbWU7Ci0gICAgICAgICAgICB0aGlzLm5hbWUgPSBuYW1lOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgICAgICBmaXJlUHJvcGVydHlDaGFuZ2UoIm5hbWUiLCBvbGROYW1lLCBuYW1lKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG5hbWUgb2YgdGhpcyBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbmFtZSBvZiB0aGlzIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldE5hbWUoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKChuYW1lID09IG51bGwpICYmIGF1dG9OYW1lKSB7Ci0gICAgICAgICAgICAgICAgbmFtZSA9IGF1dG9OYW1lKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gbmFtZTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBdXRvIG5hbWUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc3RyaW5nLgotICAgICAqLwotICAgIFN0cmluZyBhdXRvTmFtZSgpIHsKLSAgICAgICAgU3RyaW5nIG5hbWUgPSBnZXRDbGFzcygpLmdldE5hbWUoKTsKLSAgICAgICAgaWYgKG5hbWUuaW5kZXhPZigiJCIpICE9IC0xKSB7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgICAgIC8vID8/P0FXVAotICAgICAgICAvLyBpbnQgbnVtYmVyID0gdG9vbGtpdC5hdXRvTnVtYmVyLm5leHRDb21wb25lbnQrKzsKLSAgICAgICAgaW50IG51bWJlciA9IDA7Ci0gICAgICAgIG5hbWUgPSBuYW1lLnN1YnN0cmluZyhuYW1lLmxhc3RJbmRleE9mKCIuIikgKyAxKSArIEludGVnZXIudG9TdHJpbmcobnVtYmVyKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICByZXR1cm4gbmFtZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICAvKgotICAgICAgICAgKiBUaGUgZm9ybWF0IGlzIGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW9yIHdoaWNoIGNhbiBiZSByZXZlYWxlZCBieQotICAgICAgICAgKiB0aGUgZm9sbG93aW5nIGNvZGU6IENvbXBvbmVudCBjID0gbmV3IENvbXBvbmVudCgpe307Ci0gICAgICAgICAqIGMuc2V0VmlzaWJsZShmYWxzZSk7IFN5c3RlbS5vdXQucHJpbnRsbihjKTsKLSAgICAgICAgICovCi0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGdldENsYXNzKCkuZ2V0TmFtZSgpICsgIlsiICsgcGFyYW1TdHJpbmcoKSArICJdIjsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBwdWJsaWMgdm9pZCBhZGQoUG9wdXBNZW51IHBvcHVwKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyBpZgotICAgICAqIChwb3B1cC5nZXRQYXJlbnQoKSA9PSB0aGlzKSB7IHJldHVybjsgfSBpZiAocG9wdXBzID09IG51bGwpIHsgcG9wdXBzID0KLSAgICAgKiBuZXcgQXJyYXlMaXN0PFBvcHVwTWVudT4oKTsgfSBwb3B1cC5zZXRQYXJlbnQodGhpcyk7IHBvcHVwcy5hZGQocG9wdXApOyB9Ci0gICAgICogZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlLCBpZiB0aGUgY29tcG9uZW50IGNvbnRhaW5zIHRoZSBzcGVjaWZpZWQgUG9pbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHAKLSAgICAgKiAgICAgICAgICAgIHRoZSBQb2ludC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBjb21wb25lbnQgY29udGFpbnMgdGhlIHNwZWNpZmllZCBQb2ludCwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhQb2ludCBwKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGNvbnRhaW5zKHAueCwgcC55KTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUsIGlmIHRoZSBjb21wb25lbnQgY29udGFpbnMgdGhlIHBvaW50IHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIGNvb3JkaW5hdGVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIGNvbXBvbmVudCBjb250YWlucyB0aGUgcG9pbnQgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBjb29yZGluYXRlcywgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGludCB4LCBpbnQgeSkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBpbnNpZGUoeCwgeSk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGVwcmVjYXRlZDogcmVwbGFjZWQgYnkgcmVwbGFjZWQgYnkgZ2V0U2l6ZSgpIG1ldGhvZC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkaW1lbnNpb24uCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgZ2V0U2l6ZSgpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBEaW1lbnNpb24gc2l6ZSgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IERpbWVuc2lvbih3LCBoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIHB1YmxpYyBDb250YWluZXIgZ2V0UGFyZW50KCkgeyB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgcmV0dXJuIHBhcmVudDsgfQotICAgICAqIGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKi8KLQotICAgIC8qKgotICAgICAqIExpc3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIG91dAotICAgICAqICAgICAgICAgICAgdGhlIG91dC4KLSAgICAgKiBAcGFyYW0gaW5kZW50Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZW50Ci0gICAgICogQHJldHVybiB0aGUgbmVhcmVzdCBoZWF2eXdlaWdodCBhbmNlc3RvciBpbiBoaWVyYXJjaHkgb3IKLSAgICAgKiAgICAgICAgIDxjb2RlPm51bGw8L2NvZGU+IGlmIG5vdCBmb3VuZC4KLSAgICAgKi8KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIENvbXBvbmVudCBnZXRIV0FuY2VzdG9yKCkgeyByZXR1cm4gKHBhcmVudCAhPSBudWxsID8KLSAgICAgKiBwYXJlbnQuZ2V0SFdTdXJmYWNlKCkgOiBudWxsKTsgfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogQHJldHVybiBoZWF2eXdlaWdodCBjb21wb25lbnQgdGhhdCBpcyBlcXVhbCB0byBvciBpcyBhIG5lYXJlc3QKLSAgICAgKiAgICAgICAgIGhlYXZ5d2VpZ2h0IGNvbnRhaW5lciBvZiB0aGUgY3VycmVudCBjb21wb25lbnQsIG9yCi0gICAgICogICAgICAgICA8Y29kZT5udWxsPC9jb2RlPiBpZiBub3QgZm91bmQuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBDb21wb25lbnQgZ2V0SFdTdXJmYWNlKCkgeyBDb21wb25lbnQgcGFyZW50OyBmb3IgKHBhcmVudCA9IHRoaXM7IChwYXJlbnQKLSAgICAgKiAhPSBudWxsKSAmJiAocGFyZW50LmlzTGlnaHR3ZWlnaHQoKSk7IHBhcmVudCA9IHBhcmVudCAuZ2V0UGFyZW50KCkpIHsgOyB9Ci0gICAgICogcmV0dXJuIHBhcmVudDsgfSBXaW5kb3cgZ2V0V2luZG93QW5jZXN0b3IoKSB7IENvbXBvbmVudCBwYXI7IGZvciAocGFyID0KLSAgICAgKiB0aGlzOyBwYXIgIT0gbnVsbCAmJiAhKHBhciBpbnN0YW5jZW9mIFdpbmRvdyk7IHBhciA9IHBhci5nZXRQYXJlbnQoKSkgeyA7Ci0gICAgICogfSByZXR1cm4gKFdpbmRvdykgcGFyOyB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBUbyBiZSBjYWxsZWQgYnkgY29udGFpbmVyCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiB2b2lkIHNldFBhcmVudChDb250YWluZXIgcGFyZW50KSB7IHRoaXMucGFyZW50ID0gcGFyZW50OwotICAgICAqIHNldFJlZHJhd01hbmFnZXIoKTsgfSB2b2lkIHNldFJlZHJhd01hbmFnZXIoKSB7IHJlZHJhd01hbmFnZXIgPQotICAgICAqIGdldFJlZHJhd01hbmFnZXIoKTsgfSBwdWJsaWMgdm9pZCByZW1vdmUoTWVudUNvbXBvbmVudCBtZW51KSB7Ci0gICAgICogdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7IGlmIChtZW51LmdldFBhcmVudCgpID09IHRoaXMpIHsKLSAgICAgKiBtZW51LnNldFBhcmVudChudWxsKTsgcG9wdXBzLnJlbW92ZShtZW51KTsgfSB9IGZpbmFsbHkgewotICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqLwotICAgIC8qKgotICAgICAqIFByaW50cyBhIGxpc3Qgb2YgdGhpcyBjb21wb25lbnQgd2l0aCB0aGUgc3BlY2lmaWVkIG51bWJlciBvZiBsZWFkaW5nCi0gICAgICogd2hpdGVzcGFjZSBjaGFyYWN0ZXJzIHRvIHRoZSBzcGVjaWZpZWQgUHJpbnRTdHJlYW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIG91dAotICAgICAqICAgICAgICAgICAgdGhlIG91dHB1dCBQcmludFN0cmVhbSBvYmplY3QuCi0gICAgICogQHBhcmFtIGluZGVudAotICAgICAqICAgICAgICAgICAgaG93IG1hbnkgbGVhZGluZyB3aGl0ZXNwYWNlIGNoYXJhY3RlcnMgdG8gcHJlcGVuZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBsaXN0KFByaW50U3RyZWFtIG91dCwgaW50IGluZGVudCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIG91dC5wcmludGxuKGdldEluZGVudFN0cihpbmRlbnQpICsgdGhpcyk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJpbnRzIGEgbGlzdCBvZiB0aGlzIGNvbXBvbmVudCB0byB0aGUgc3BlY2lmaWVkIFByaW50V3JpdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvdXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvdXRwdXQgUHJpbnRXcml0ZXIgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGxpc3QoUHJpbnRXcml0ZXIgb3V0KSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgbGlzdChvdXQsIDEpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByaW50cyBhIGxpc3Qgb2YgdGhpcyBjb21wb25lbnQgd2l0aCB0aGUgc3BlY2lmaWVkIG51bWJlciBvZiBsZWFkaW5nCi0gICAgICogd2hpdGVzcGFjZSBjaGFyYWN0ZXJzIHRvIHRoZSBzcGVjaWZpZWQgUHJpbnRXcml0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIG91dAotICAgICAqICAgICAgICAgICAgdGhlIG91dHB1dCBQcmludFdyaXRlciBvYmplY3QuCi0gICAgICogQHBhcmFtIGluZGVudAotICAgICAqICAgICAgICAgICAgaG93IG1hbnkgbGVhZGluZyB3aGl0ZXNwYWNlIGNoYXJhY3RlcnMgdG8gcHJlcGVuZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBsaXN0KFByaW50V3JpdGVyIG91dCwgaW50IGluZGVudCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIG91dC5wcmludGxuKGdldEluZGVudFN0cihpbmRlbnQpICsgdGhpcyk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIHN0cmluZyBjb21wb3NlZCBvZiB0aGUgZGVzaXJlZCBudW1iZXIgb2Ygd2hpdGVzcGFjZSBjaGFyYWN0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggb2YgdGhlIFN0cmluZyB0byByZXR1cm4uCi0gICAgICogQHJldHVybiB0aGUgc3RyaW5nIGNvbXBvc2VkIG9mIHRoZSBkZXNpcmVkIG51bWJlciBvZiB3aGl0ZXNwYWNlCi0gICAgICogICAgICAgICBjaGFyYWN0ZXJzLgotICAgICAqLwotICAgIFN0cmluZyBnZXRJbmRlbnRTdHIoaW50IGluZGVudCkgewotICAgICAgICBjaGFyW10gaW5kID0gbmV3IGNoYXJbaW5kZW50XTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBpbmRlbnQ7IGluZFtpKytdID0gJyAnKSB7Ci0gICAgICAgICAgICA7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBTdHJpbmcoaW5kKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcmludHMgYSBsaXN0IG9mIHRoaXMgY29tcG9uZW50IHRvIHRoZSBzcGVjaWZpZWQgUHJpbnRTdHJlYW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIG91dAotICAgICAqICAgICAgICAgICAgdGhlIG91dHB1dCBQcmludFN0cmVhbSBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgbGlzdChQcmludFN0cmVhbSBvdXQpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAvLyBkZWZhdWx0IGluZGVudCA9IDEKLSAgICAgICAgICAgIGxpc3Qob3V0LCAxKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcmludHMgYSBsaXN0IG9mIHRoaXMgY29tcG9uZW50IHRvIHRoZSBzdGFuZGFyZCBzeXN0ZW0gb3V0cHV0IHN0cmVhbS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBsaXN0KCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGxpc3QoU3lzdGVtLm91dCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJpbnRzIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBnCi0gICAgICogICAgICAgICAgICB0aGUgR3JhcGhpY3MgdG8gYmUgdXNlZCBmb3IgcGFpbnRpbmcuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcHJpbnQoR3JhcGhpY3MgZykgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHBhaW50KGcpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByaW50cyB0aGUgY29tcG9uZW50IGFuZCBhbGwgb2YgaXRzIHN1YmNvbXBvbmVudHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGcKLSAgICAgKiAgICAgICAgICAgIHRoZSBHcmFwaGljcyB0byBiZSB1c2VkIGZvciBwYWludGluZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBwcmludEFsbChHcmFwaGljcyBnKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcGFpbnRBbGwoZyk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2l6ZSBvZiB0aGUgQ29tcG9uZW50IHNwZWNpZmllZCBieSB3aWR0aCBhbmQgaGVpZ2h0IHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0U2l6ZShpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXNpemUod2lkdGgsIGhlaWdodCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2l6ZSBvZiB0aGUgQ29tcG9uZW50IHNwZWNpZmllZCBieSBEaW1lbnNpb24gb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBkCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHNpemUgb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRTaXplKERpbWVuc2lvbiBkKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmVzaXplKGQpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIERlcHJlY2F0ZWQ6IHJlcGxhY2VkIGJ5IHNldFNpemUoaW50LCBpbnQpIG1ldGhvZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aC4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0LgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IHNldFNpemUoaW50LCBpbnQpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyB2b2lkIHJlc2l6ZShpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBib3VuZHNNYXNrUGFyYW0gPSBOYXRpdmVXaW5kb3cuQk9VTkRTX05PTU9WRTsKLSAgICAgICAgICAgIHNldEJvdW5kcyh4LCB5LCB3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXByZWNhdGVkOiByZXBsYWNlZCBieSBzZXRTaXplKGludCwgaW50KSBtZXRob2QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzaXplLgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IHNldFNpemUoaW50LCBpbnQpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyB2b2lkIHJlc2l6ZShEaW1lbnNpb24gc2l6ZSkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHNldFNpemUoc2l6ZS53aWR0aCwgc2l6ZS5oZWlnaHQpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGlzIGNvbXBvbmVudCBpcyBjb21wbGV0ZWx5IG9wYXF1ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgY29tcG9uZW50IGlzIGNvbXBsZXRlbHkgb3BhcXVlLCBmYWxzZSBieSBkZWZhdWx0LgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzT3BhcXVlKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBiZWhhdmlvdXIuaXNPcGFxdWUoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEaXNhYmxlcy4KLSAgICAgKiAKLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSBzZXRFbmFibGVkKGJvb2xlYW4pIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyB2b2lkIGRpc2FibGUoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgc2V0RW5hYmxlZEltcGwoZmFsc2UpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgICAgICAvLyA/Pz9BV1Q6IGZpcmVBY2Nlc3NpYmxlU3RhdGVDaGFuZ2UoQWNjZXNzaWJsZVN0YXRlLkVOQUJMRUQsIGZhbHNlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBFbmFibGVzIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IHNldEVuYWJsZWQoYm9vbGVhbikgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIHZvaWQgZW5hYmxlKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHNldEVuYWJsZWRJbXBsKHRydWUpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgICAgICAvLyA/Pz9BV1Q6IGZpcmVBY2Nlc3NpYmxlU3RhdGVDaGFuZ2UoQWNjZXNzaWJsZVN0YXRlLkVOQUJMRUQsIHRydWUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEVuYWJsZXMgb3IgZGlzYWJsZSB0aGlzIGNvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJvb2xlYW4gcGFyYW1ldGVyLgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IHNldEVuYWJsZWQoYm9vbGVhbikgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIHZvaWQgZW5hYmxlKGJvb2xlYW4gYikgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGlmIChiKSB7Ci0gICAgICAgICAgICAgICAgZW5hYmxlKCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGRpc2FibGUoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTdG9yZXMgdGhlIGxvY2F0aW9uIG9mIHRoaXMgY29tcG9uZW50IHRvIHRoZSBzcGVjaWZpZWQgUG9pbnQgb2JqZWN0OwotICAgICAqIHJldHVybnMgdGhlIHBvaW50IG9mIHRoZSBjb21wb25lbnQncyB0b3AtbGVmdCBjb3JuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJ2Ci0gICAgICogICAgICAgICAgICB0aGUgUG9pbnQgb2JqZWN0IHdoZXJlIHRoZSBjb21wb25lbnQncyB0b3AtbGVmdCBjb3JuZXIKLSAgICAgKiAgICAgICAgICAgIHBvc2l0aW9uIHdpbGwgYmUgc3RvcmVkLgotICAgICAqIEByZXR1cm4gdGhlIFBvaW50IHdoaWNoIHNwZWNpZmllcyB0aGUgY29tcG9uZW50J3MgdG9wLWxlZnQgY29ybmVyLgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludCBnZXRMb2NhdGlvbihQb2ludCBydikgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGlmIChydiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcnYgPSBuZXcgUG9pbnQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJ2LnNldExvY2F0aW9uKGdldFgoKSwgZ2V0WSgpKTsKLSAgICAgICAgICAgIHJldHVybiBydjsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBsb2NhdGlvbiBvZiB0aGlzIGNvbXBvbmVudCBvbiB0aGUgZm9ybTsgcmV0dXJucyB0aGUgcG9pbnQgb2YgdGhlCi0gICAgICogY29tcG9uZW50J3MgdG9wLWxlZnQgY29ybmVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFBvaW50IHdoaWNoIHNwZWNpZmllcyB0aGUgY29tcG9uZW50J3MgdG9wLWxlZnQgY29ybmVyLgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludCBnZXRMb2NhdGlvbigpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gbG9jYXRpb24oKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzaXplIG9mIHRoaXMgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNpemUgb2YgdGhpcyBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIERpbWVuc2lvbiBnZXRTaXplKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBzaXplKCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU3RvcmVzIHRoZSBzaXplIG9mIHRoaXMgQ29tcG9uZW50IHRvIHRoZSBzcGVjaWZpZWQgRGltZW5zaW9uIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcnYKLSAgICAgKiAgICAgICAgICAgIHRoZSBEaW1lbnNpb24gb2JqZWN0IHdoZXJlIHRoZSBzaXplIG9mIHRoZSBDb21wb25lbnQgd2lsbCBiZQotICAgICAqICAgICAgICAgICAgc3RvcmVkLgotICAgICAqIEByZXR1cm4gdGhlIERpbWVuc2lvbiBvZiB0aGlzIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGltZW5zaW9uIGdldFNpemUoRGltZW5zaW9uIHJ2KSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKHJ2ID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBydiA9IG5ldyBEaW1lbnNpb24oKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJ2LnNldFNpemUoZ2V0V2lkdGgoKSwgZ2V0SGVpZ2h0KCkpOwotICAgICAgICAgICAgcmV0dXJuIHJ2OwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGlzIENvbXBvbmVudCBpcyB2YWxpZC4gQSBjb21wb25lbnQgaXMgdmFsaWQgaWYgaXQKLSAgICAgKiBpcyBjb3JyZWN0bHkgc2l6ZWQgYW5kIHBvc2l0aW9uZWQgd2l0aGluIGl0cyBwYXJlbnQgY29udGFpbmVyIGFuZCBhbGwgaXRzCi0gICAgICogY2hpbGRyZW4gYXJlIGFsc28gdmFsaWQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgQ29tcG9uZW50IGlzIHZhbGlkLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNWYWxpZCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gdmFsaWQgJiYgYmVoYXZpb3VyLmlzRGlzcGxheWFibGUoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXByZWNhdGVkOiByZXBsYWNlZCBieSBnZXRDb21wb25lbnRBdChpbnQsIGludCkgbWV0aG9kLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFBvaW50LgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IGdldENvbXBvbmVudEF0KGludCwgaW50KSBtZXRob2QuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgUG9pbnQgbG9jYXRpb24oKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBQb2ludCh4LCB5KTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb25uZWN0cyB0aGlzIENvbXBvbmVudCB0byBhIG5hdGl2ZSBzY3JlZW4gcmVzb3VyY2UgYW5kIG1ha2VzIGl0Ci0gICAgICogZGlzcGxheWFibGUuIFRoaXMgbWV0aG9kIG5vdCBiZSBjYWxsZWQgZGlyZWN0bHkgYnkgdXNlciBhcHBsaWNhdGlvbnMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkTm90aWZ5KCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHByZXBhcmU0SGllcmFyY2h5Q2hhbmdlKCk7Ci0gICAgICAgICAgICBiZWhhdmlvdXIuYWRkTm90aWZ5KCk7Ci0gICAgICAgICAgICAvLyA/Pz9BV1QKLSAgICAgICAgICAgIC8vIGZpbmlzaEhpZXJhcmNoeUNoYW5nZSh0aGlzLCBwYXJlbnQsIDApOwotICAgICAgICAgICAgLy8gaWYgKGRyb3BUYXJnZXQgIT0gbnVsbCkgewotICAgICAgICAgICAgLy8gZHJvcFRhcmdldC5hZGROb3RpZnkocGVlcik7Ci0gICAgICAgICAgICAvLyB9Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogTWFwIHRvIGRpc3BsYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBiLgotICAgICAqLwotICAgIHZvaWQgbWFwVG9EaXNwbGF5KGJvb2xlYW4gYikgewotICAgICAgICAvLyA/Pz9BV1QKLSAgICAgICAgLyoKLSAgICAgICAgICogaWYgKGIgJiYgIWlzRGlzcGxheWFibGUoKSkgeyBpZiAoKHRoaXMgaW5zdGFuY2VvZiBXaW5kb3cpIHx8ICgocGFyZW50Ci0gICAgICAgICAqICE9IG51bGwpICYmIHBhcmVudC5pc0Rpc3BsYXlhYmxlKCkpKSB7IGFkZE5vdGlmeSgpOyB9IH0gZWxzZSBpZiAoIWIKLSAgICAgICAgICogJiYgaXNEaXNwbGF5YWJsZSgpKSB7IHJlbW92ZU5vdGlmeSgpOyB9Ci0gICAgICAgICAqLwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRvb2xraXQuCi0gICAgICogCi0gICAgICogQHJldHVybiBhY2Nlc3NpYmxlIGNvbnRleHQgc3BlY2lmaWMgZm9yIHBhcnRpY3VsYXIgY29tcG9uZW50LgotICAgICAqLwotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogQWNjZXNzaWJsZUNvbnRleHQgY3JlYXRlQWNjZXNzaWJsZUNvbnRleHQoKSB7IHJldHVybiBudWxsOyB9IHB1YmxpYwotICAgICAqIEFjY2Vzc2libGVDb250ZXh0IGdldEFjY2Vzc2libGVDb250ZXh0KCkgeyB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgaWYKLSAgICAgKiAoYWNjZXNzaWJsZUNvbnRleHQgPT0gbnVsbCkgeyBhY2Nlc3NpYmxlQ29udGV4dCA9Ci0gICAgICogY3JlYXRlQWNjZXNzaWJsZUNvbnRleHQoKTsgfSByZXR1cm4gYWNjZXNzaWJsZUNvbnRleHQ7IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBHZXRzIFRvb2xraXQgZm9yIHRoZSBjdXJyZW50IENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBUb29sa2l0IG9mIHRoaXMgQ29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBUb29sa2l0IGdldFRvb2xraXQoKSB7Ci0gICAgICAgIHJldHVybiB0b29sa2l0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhpcyBjb21wb25lbnQncyBsb2NraW5nIG9iamVjdCBmb3IgQVdUIGNvbXBvbmVudCB0cmVlIGFuZCBsYXlvdXQKLSAgICAgKiBvcGVyYXRpb25zLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHRyZWUgbG9ja2luZyBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIE9iamVjdCBnZXRUcmVlTG9jaygpIHsKLSAgICAgICAgcmV0dXJuIHRvb2xraXQuYXd0VHJlZUxvY2s7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSGFuZGxlcyB0aGUgZXZlbnQuIFVzZSBBY3Rpb25MaXN0ZW5lciBpbnN0ZWFkIG9mIHRoaXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2dAotICAgICAqICAgICAgICAgICAgdGhlIEV2ZW50LgotICAgICAqIEBwYXJhbSB3aGF0Ci0gICAgICogICAgICAgICAgICB0aGUgZXZlbnQncyBrZXkuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLgotICAgICAqIEBkZXByZWNhdGVkIFVzZSBBY3Rpb25MaXN0ZW5lciBjbGFzcyBmb3IgcmVnaXN0ZXJpbmcgZXZlbnQgbGlzdGVuZXIuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgYm9vbGVhbiBhY3Rpb24oRXZlbnQgZXZ0LCBPYmplY3Qgd2hhdCkgewotICAgICAgICAvLyB0byBiZSBvdmVycmlkZGVuOiBkbyBub3RoaW5nLAotICAgICAgICAvLyBqdXN0IHJldHVybiBmYWxzZSB0byBwcm9wYWdhdGUgZXZlbnQgdXAgdG8gdGhlIHBhcmVudCBjb250YWluZXIKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHByb3BlcnR5IGNoYW5nZSBzdXBwb3J0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHByb3BlcnR5IGNoYW5nZSBzdXBwb3J0LgotICAgICAqLwotICAgIHByaXZhdGUgUHJvcGVydHlDaGFuZ2VTdXBwb3J0IGdldFByb3BlcnR5Q2hhbmdlU3VwcG9ydCgpIHsKLSAgICAgICAgc3luY2hyb25pemVkIChjb21wb25lbnRMb2NrKSB7Ci0gICAgICAgICAgICBpZiAocHJvcGVydHlDaGFuZ2VTdXBwb3J0ID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBwcm9wZXJ0eUNoYW5nZVN1cHBvcnQgPSBuZXcgUHJvcGVydHlDaGFuZ2VTdXBwb3J0KHRoaXMpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIHByb3BlcnR5Q2hhbmdlU3VwcG9ydDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogcHVibGljIHZvaWQgYWRkUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcihQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyIGxpc3RlbmVyKSB7Ci0gICAgICogZ2V0UHJvcGVydHlDaGFuZ2VTdXBwb3J0KCkuYWRkUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcihsaXN0ZW5lcik7IH0gcHVibGljCi0gICAgICogdm9pZCBhZGRQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyKFN0cmluZyBwcm9wZXJ0eU5hbWUsCi0gICAgICogUHJvcGVydHlDaGFuZ2VMaXN0ZW5lciBsaXN0ZW5lcikgewotICAgICAqIGdldFByb3BlcnR5Q2hhbmdlU3VwcG9ydCgpLmFkZFByb3BlcnR5Q2hhbmdlTGlzdGVuZXIocHJvcGVydHlOYW1lLAotICAgICAqIGxpc3RlbmVyKTsgfSBwdWJsaWMgdm9pZCBhcHBseUNvbXBvbmVudE9yaWVudGF0aW9uKENvbXBvbmVudE9yaWVudGF0aW9uCi0gICAgICogb3JpZW50YXRpb24pIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7Ci0gICAgICogc2V0Q29tcG9uZW50T3JpZW50YXRpb24ob3JpZW50YXRpb24pOyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9Ci0gICAgICogfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoZSBzZXQgb2YgZm9jdXMgdHJhdmVyc2FsIGtleXMgZm9yIHRoZSBnaXZlbiBmb2N1cwotICAgICAqIHRyYXZlcnNhbCBvcGVyYXRpb24gaGFzIGJlZW4gZXhwbGljaXRseSBkZWZpbmVkIGZvciB0aGlzIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBJRCBvZiB0cmF2ZXJzYWwga2V5LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNldCBvZiBmb2N1cyB0cmF2ZXJzYWwga2V5cyBmb3IgdGhlIGdpdmVuIGZvY3VzLgotICAgICAqICAgICAgICAgdHJhdmVyc2FsIG9wZXJhdGlvbiBoYXMgYmVlbiBleHBsaWNpdGx5IGRlZmluZWQgZm9yIHRoaXMKLSAgICAgKiAgICAgICAgIENvbXBvbmVudCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGFyZUZvY3VzVHJhdmVyc2FsS2V5c1NldChpbnQgaWQpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBJbnRlZ2VyIElkID0gbmV3IEludGVnZXIoaWQpOwotICAgICAgICAgICAgaWYgKHRyYXZlcnNhbEtleXMuY29udGFpbnNLZXkoSWQpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRyYXZlcnNhbEtleXMuZ2V0KElkKSAhPSBudWxsOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgLy8gYXd0LjE0Rj1pbnZhbGlkIGZvY3VzIHRyYXZlcnNhbCBrZXkgaWRlbnRpZmllcgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNEYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZHMgb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSByZWN0YW5nbGUgYm91bmRzIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICogQGRlcHJlY2F0ZWQgVXNlIGdldEJvdW5kcygpIG1ldGhvb2QuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGJvdW5kcygpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZSh4LCB5LCB3LCBoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBjb25zdHJ1Y3Rpb24gc3RhdHVzIG9mIGEgc3BlY2lmaWVkIGltYWdlIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIHdpZHRoIGFuZCBoZWlnaHQgdGhhdCBpcyBiZWluZyBjcmVhdGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHRvIGJlIGNoZWNrZWQuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2Ygc2NhbGVkIGltYWdlIHdoaWNoIHN0YXR1cyBpcyBiZWluZyBjaGVja2VkLCBvcgotICAgICAqICAgICAgICAgICAgLTEuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBzY2FsZWQgaW1hZ2Ugd2hpY2ggc3RhdHVzIGlzIGJlaW5nIGNoZWNrZWQsIG9yCi0gICAgICogICAgICAgICAgICAtMS4KLSAgICAgKiBAcGFyYW0gb2JzZXJ2ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZU9ic2VydmVyIG9iamVjdCB0byBiZSBub3RpZmllZCB3aGlsZSB0aGUgaW1hZ2UgaXMKLSAgICAgKiAgICAgICAgICAgIGJlaW5nIHByZXBhcmVkLgotICAgICAqIEByZXR1cm4gdGhlIEltYWdlT2JzZXJ2ZXIgZmxhZ3Mgb2YgdGhlIGN1cnJlbnQgc3RhdGUgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIGludCBjaGVja0ltYWdlKEltYWdlIGltYWdlLCBpbnQgd2lkdGgsIGludCBoZWlnaHQsIEltYWdlT2JzZXJ2ZXIgb2JzZXJ2ZXIpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gdG9vbGtpdC5jaGVja0ltYWdlKGltYWdlLCB3aWR0aCwgaGVpZ2h0LCBvYnNlcnZlcik7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgY29uc3RydWN0aW9uIHN0YXR1cyBvZiBhIHNwZWNpZmllZCBpbWFnZSB0aGF0IGlzIGJlaW5nCi0gICAgICogY3JlYXRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSB0byBiZSBjaGVja2VkLgotICAgICAqIEBwYXJhbSBvYnNlcnZlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlT2JzZXJ2ZXIgb2JqZWN0IHRvIGJlIG5vdGlmaWVkIHdoaWxlIHRoZSBpbWFnZSBpcwotICAgICAqICAgICAgICAgICAgYmVpbmcgcHJlcGFyZWQuCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VPYnNlcnZlciBmbGFncyBvZiB0aGUgY3VycmVudCBzdGF0ZSBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGNoZWNrSW1hZ2UoSW1hZ2UgaW1hZ2UsIEltYWdlT2JzZXJ2ZXIgb2JzZXJ2ZXIpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gdG9vbGtpdC5jaGVja0ltYWdlKGltYWdlLCAtMSwgLTEsIG9ic2VydmVyKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb2FsZXNjZXMgdGhlIGV4aXN0ZWQgZXZlbnQgd2l0aCBuZXcgZXZlbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV4aXN0aW5nRXZlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBleGlzdGluZyBldmVudCBpbiB0aGUgRXZlbnRRdWV1ZS4KLSAgICAgKiBAcGFyYW0gbmV3RXZlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgZXZlbnQgdG8gYmUgcG9zdGVkIHRvIHRoZSBFdmVudFF1ZXVlLgotICAgICAqIEByZXR1cm4gdGhlIGNvYWxlc2NlZCBBV1RFdmVudCwgb3IgbnVsbCBpZiB0aGVyZSBpcyBubyBjb2FsZXNjaW5nIGRvbmUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEFXVEV2ZW50IGNvYWxlc2NlRXZlbnRzKEFXVEV2ZW50IGV4aXN0aW5nRXZlbnQsIEFXVEV2ZW50IG5ld0V2ZW50KSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgLy8gTm90aGluZyB0byBkbzoKLSAgICAgICAgICAgIC8vIDEuIE1vdXNlIGV2ZW50cyBjb2FsZXNjZWQgYXQgV1RLIGxldmVsCi0gICAgICAgICAgICAvLyAyLiBQYWludCBldmVudHMgaGFuZGxlZCBieSBSZWRyYXdNYW5hZ2VyCi0gICAgICAgICAgICAvLyBUaGlzIG1ldGhvZCBpcyBmb3Igb3ZlcnJpZGluZyBvbmx5Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyBDb21wb25lbnQgaXMgYSBjb2FsZXNjZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBpcyBjb2FsZXNjZXIuCi0gICAgICovCi0gICAgYm9vbGVhbiBpc0NvYWxlc2NlcigpIHsKLSAgICAgICAgcmV0dXJuIGNvYWxlc2NlcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSByZWxhdGl2ZSBldmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBpZC4KLSAgICAgKiBAcmV0dXJuIHRoZSByZWxhdGl2ZSBldmVudC4KLSAgICAgKi8KLSAgICBBV1RFdmVudCBnZXRSZWxhdGl2ZUV2ZW50KGludCBpZCkgewotICAgICAgICBJbnRlZ2VyIGlkV3JhcHBlciA9IG5ldyBJbnRlZ2VyKGlkKTsKLSAgICAgICAgZXZlbnRzTGlzdCA9IGV2ZW50c1RhYmxlLmdldChpZFdyYXBwZXIpOwotICAgICAgICBpZiAoZXZlbnRzTGlzdCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBldmVudHNMaXN0ID0gbmV3IExpbmtlZExpc3Q8QVdURXZlbnQ+KCk7Ci0gICAgICAgICAgICBldmVudHNUYWJsZS5wdXQoaWRXcmFwcGVyLCBldmVudHNMaXN0KTsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgICAgIGlmIChldmVudHNMaXN0LmlzRW1wdHkoKSkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGV2ZW50c0xpc3QuZ2V0TGFzdCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgdGhlIG5ldyBldmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXZlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBldmVudC4KLSAgICAgKi8KLSAgICB2b2lkIGFkZE5ld0V2ZW50KEFXVEV2ZW50IGV2ZW50KSB7Ci0gICAgICAgIGV2ZW50c0xpc3QuYWRkTGFzdChldmVudCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgcmVsYXRpdmUgZXZlbnQuCi0gICAgICovCi0gICAgdm9pZCByZW1vdmVSZWxhdGl2ZUV2ZW50KCkgewotICAgICAgICBldmVudHNMaXN0LnJlbW92ZUxhc3QoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHRoZSBuZXh0IGV2ZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpZAotICAgICAqICAgICAgICAgICAgdGhlIGlkLgotICAgICAqLwotICAgIHZvaWQgcmVtb3ZlTmV4dEV2ZW50KGludCBpZCkgewotICAgICAgICBldmVudHNUYWJsZS5nZXQobmV3IEludGVnZXIoaWQpKS5yZW1vdmVGaXJzdCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGltYWdlIHdpdGggdGhlIHNwZWNpZmllZCBJbWFnZVByb2R1Y2VyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm9kdWNlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUHJvZHVjZXIgdG8gYmUgdXNlZCBmb3IgaW1hZ2UgY3JlYXRpb24uCi0gICAgICogQHJldHVybiB0aGUgaW1hZ2Ugd2l0aCB0aGUgc3BlY2lmaWVkIEltYWdlUHJvZHVjZXIuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlIGNyZWF0ZUltYWdlKEltYWdlUHJvZHVjZXIgcHJvZHVjZXIpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gdG9vbGtpdC5jcmVhdGVJbWFnZShwcm9kdWNlcik7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhbiBvZmYtc2NyZWVuIGRyYXdhYmxlIGltYWdlIHRvIGJlIHVzZWQgZm9yIGRvdWJsZSBidWZmZXJpbmcuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGltYWdlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlLgotICAgICAqIEByZXR1cm4gdGhlIG9mZi1zY3JlZW4gZHJhd2FibGUgaW1hZ2Ugb3IgbnVsbCBpZiB0aGUgY29tcG9uZW50IGlzIG5vdAotICAgICAqICAgICAgICAgZGlzcGxheWFibGUgb3IgR3JhcGhpY3NFbnZpcm9ubWVudC5pc0hlYWRsZXNzKCkgbWV0aG9kIHJldHVybnMKLSAgICAgKiAgICAgICAgIHRydWUuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlIGNyZWF0ZUltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGlmICghaXNEaXNwbGF5YWJsZSgpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBHcmFwaGljc0NvbmZpZ3VyYXRpb24gZ2MgPSBnZXRHcmFwaGljc0NvbmZpZ3VyYXRpb24oKTsKLSAgICAgICAgICAgIGlmIChnYyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBDb2xvck1vZGVsIGNtID0gZ2MuZ2V0Q29sb3JNb2RlbChUcmFuc3BhcmVuY3kuT1BBUVVFKTsKLSAgICAgICAgICAgIFdyaXRhYmxlUmFzdGVyIHdyID0gY20uY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKHdpZHRoLCBoZWlnaHQpOwotICAgICAgICAgICAgSW1hZ2UgaW1hZ2UgPSBuZXcgQnVmZmVyZWRJbWFnZShjbSwgd3IsIGNtLmlzQWxwaGFQcmVtdWx0aXBsaWVkKCksIG51bGwpOwotICAgICAgICAgICAgZmlsbEltYWdlQmFja2dyb3VuZChpbWFnZSwgd2lkdGgsIGhlaWdodCk7Ci0gICAgICAgICAgICByZXR1cm4gaW1hZ2U7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhbiBvZmYtc2NyZWVuIGRyYXdhYmxlIGltYWdlIHdpdGggdGhlIHNwZWNpZmllZCB3aWR0aCwgaGVpZ2h0IGFuZAotICAgICAqIEltYWdlQ2FwYWJpbGl0aWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQuCi0gICAgICogQHBhcmFtIGNhcHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZUNhcGFiaWxpdGllcy4KLSAgICAgKiBAcmV0dXJuIHRoZSB2b2xhdGlsZSBpbWFnZS4KLSAgICAgKiBAdGhyb3dzIEFXVEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIGltYWdlIHdpdGggdGhlIHNwZWNpZmllZCBjYXBhYmlsaXRpZXMgY2Fubm90IGJlCi0gICAgICogICAgICAgICAgICAgY3JlYXRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgVm9sYXRpbGVJbWFnZSBjcmVhdGVWb2xhdGlsZUltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCwgSW1hZ2VDYXBhYmlsaXRpZXMgY2FwcykKLSAgICAgICAgICAgIHRocm93cyBBV1RFeGNlcHRpb24gewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGlmICghaXNEaXNwbGF5YWJsZSgpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBHcmFwaGljc0NvbmZpZ3VyYXRpb24gZ2MgPSBnZXRHcmFwaGljc0NvbmZpZ3VyYXRpb24oKTsKLSAgICAgICAgICAgIGlmIChnYyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBWb2xhdGlsZUltYWdlIGltYWdlID0gZ2MuY3JlYXRlQ29tcGF0aWJsZVZvbGF0aWxlSW1hZ2Uod2lkdGgsIGhlaWdodCwgY2Fwcyk7Ci0gICAgICAgICAgICBmaWxsSW1hZ2VCYWNrZ3JvdW5kKGltYWdlLCB3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgICAgIHJldHVybiBpbWFnZTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgdm9sYXRpbGUgb2ZmLXNjcmVlbiBkcmF3YWJsZSBpbWFnZSB3aGljaCBpcyB1c2VkIGZvciBkb3VibGUKLSAgICAgKiBidWZmZXJpbmcuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgaW1hZ2UuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBpbWFnZS4KLSAgICAgKiBAcmV0dXJuIHRoZSB2b2xhdGlsZSBpbWFnZSBhIHZvbGF0aWxlIG9mZi1zY3JlZW4gZHJhd2FibGUgaW1hZ2Ugd2hpY2ggaXMKLSAgICAgKiAgICAgICAgIHVzZWQgZm9yIGRvdWJsZSBidWZmZXJpbmcgb3IgbnVsbCBpZiB0aGUgY29tcG9uZW50IGlzIG5vdAotICAgICAqICAgICAgICAgZGlzcGxheWFibGUsIG9yIEdyYXBoaWNzRW52aXJvbm1lbnQuaXNIZWFkbGVzcygpIG1ldGhvZCByZXR1cm5zCi0gICAgICogICAgICAgICB0cnVlLgotICAgICAqLwotICAgIHB1YmxpYyBWb2xhdGlsZUltYWdlIGNyZWF0ZVZvbGF0aWxlSW1hZ2UoaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKCFpc0Rpc3BsYXlhYmxlKCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBnYyA9IGdldEdyYXBoaWNzQ29uZmlndXJhdGlvbigpOwotICAgICAgICAgICAgaWYgKGdjID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIFZvbGF0aWxlSW1hZ2UgaW1hZ2UgPSBnYy5jcmVhdGVDb21wYXRpYmxlVm9sYXRpbGVJbWFnZSh3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgICAgIGZpbGxJbWFnZUJhY2tncm91bmQoaW1hZ2UsIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICAgICAgcmV0dXJuIGltYWdlOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbGwgdGhlIGltYWdlIGJlaW5nIGNyZWF0ZWQgYnkgY3JlYXRlSW1hZ2UoKSBvciBjcmVhdGVWb2xhdGlsZUltYWdlKCkKLSAgICAgKiB3aXRoIHRoZSBjb21wb25lbnQncyBiYWNrZ3JvdW5kIGNvbG9yIHRvIHByZXBhcmUgaXQgZm9yIGRvdWJsZS1idWZmZXJlZAotICAgICAqIHBhaW50aW5nLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIGZpbGxJbWFnZUJhY2tncm91bmQoSW1hZ2UgaW1hZ2UsIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICBHcmFwaGljcyBnciA9IGltYWdlLmdldEdyYXBoaWNzKCk7Ci0gICAgICAgIGdyLnNldENvbG9yKGdldEJhY2tncm91bmQoKSk7Ci0gICAgICAgIGdyLmZpbGxSZWN0KDAsIDAsIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICBnci5kaXNwb3NlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGVsaXZlcnMgZXZlbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2dAotICAgICAqICAgICAgICAgICAgdGhlIGV2ZW50LgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IGRpc3BhdGNoRXZlbnQoQVdURXZlbnQgZSkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIHZvaWQgZGVsaXZlckV2ZW50KEV2ZW50IGV2dCkgewotICAgICAgICBwb3N0RXZlbnQoZXZ0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9tcHRzIHRoZSBsYXlvdXQgbWFuYWdlciB0byBsYXkgb3V0IHRoaXMgY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGRvTGF5b3V0KCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGxheW91dCgpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgICAgICAvLyBJbXBsZW1lbnRlZCBpbiBDb250YWluZXIKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaXJlIHByb3BlcnR5IGNoYW5nZSBpbXBsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm9wZXJ0eU5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSBuYW1lLgotICAgICAqIEBwYXJhbSBvbGRWYWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIG9sZCB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gbmV3VmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgdmFsdWUuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIGZpcmVQcm9wZXJ0eUNoYW5nZUltcGwoU3RyaW5nIHByb3BlcnR5TmFtZSwgT2JqZWN0IG9sZFZhbHVlLCBPYmplY3QgbmV3VmFsdWUpIHsKLSAgICAgICAgUHJvcGVydHlDaGFuZ2VTdXBwb3J0IHBjczsKLSAgICAgICAgc3luY2hyb25pemVkIChjb21wb25lbnRMb2NrKSB7Ci0gICAgICAgICAgICBpZiAocHJvcGVydHlDaGFuZ2VTdXBwb3J0ID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwY3MgPSBwcm9wZXJ0eUNoYW5nZVN1cHBvcnQ7Ci0gICAgICAgIH0KLSAgICAgICAgcGNzLmZpcmVQcm9wZXJ0eUNoYW5nZShwcm9wZXJ0eU5hbWUsIG9sZFZhbHVlLCBuZXdWYWx1ZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwb3J0cyBhIGJvdW5kIHByb3BlcnR5IGNoYW5nZXMgZm9yIGludCBwcm9wZXJ0aWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm9wZXJ0eU5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSBuYW1lLgotICAgICAqIEBwYXJhbSBvbGRWYWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIG9sZCBwcm9wZXJ0eSdzIHZhbHVlLgotICAgICAqIEBwYXJhbSBuZXdWYWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBwcm9wZXJ0eSdzIHZhbHVlLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIGZpcmVQcm9wZXJ0eUNoYW5nZShTdHJpbmcgcHJvcGVydHlOYW1lLCBpbnQgb2xkVmFsdWUsIGludCBuZXdWYWx1ZSkgewotICAgICAgICBmaXJlUHJvcGVydHlDaGFuZ2VJbXBsKHByb3BlcnR5TmFtZSwgbmV3IEludGVnZXIob2xkVmFsdWUpLCBuZXcgSW50ZWdlcihuZXdWYWx1ZSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcG9ydCBhIGJvdW5kIHByb3BlcnR5IGNoYW5nZSBmb3IgYSBib29sZWFuLXZhbHVlZCBwcm9wZXJ0eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcGVydHlOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydHkgbmFtZS4KLSAgICAgKiBAcGFyYW0gb2xkVmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSdzIG9sZCB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gbmV3VmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSdzIG5ldyB2YWx1ZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBmaXJlUHJvcGVydHlDaGFuZ2UoU3RyaW5nIHByb3BlcnR5TmFtZSwgYm9vbGVhbiBvbGRWYWx1ZSwgYm9vbGVhbiBuZXdWYWx1ZSkgewotICAgICAgICBmaXJlUHJvcGVydHlDaGFuZ2VJbXBsKHByb3BlcnR5TmFtZSwgQm9vbGVhbi52YWx1ZU9mKG9sZFZhbHVlKSwgQm9vbGVhbi52YWx1ZU9mKG5ld1ZhbHVlKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwb3J0cyBhIGJvdW5kIHByb3BlcnR5IGNoYW5nZSBmb3IgYW4gT2JqZWN0LXZhbHVlZCBwcm9wZXJ0eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcGVydHlOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydHkgbmFtZS4KLSAgICAgKiBAcGFyYW0gb2xkVmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSdzIG9sZCB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gbmV3VmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSdzIG5ldyB2YWx1ZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBmaXJlUHJvcGVydHlDaGFuZ2UoZmluYWwgU3RyaW5nIHByb3BlcnR5TmFtZSwgZmluYWwgT2JqZWN0IG9sZFZhbHVlLAotICAgICAgICAgICAgZmluYWwgT2JqZWN0IG5ld1ZhbHVlKSB7Ci0gICAgICAgIGZpcmVQcm9wZXJ0eUNoYW5nZUltcGwocHJvcGVydHlOYW1lLCBvbGRWYWx1ZSwgbmV3VmFsdWUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcG9ydCBhIGJvdW5kIHByb3BlcnR5IGNoYW5nZSBmb3IgYSBieXRlLXZhbHVlZCBwcm9wZXJ0eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcGVydHlOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydHkgbmFtZS4KLSAgICAgKiBAcGFyYW0gb2xkVmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSdzIG9sZCB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gbmV3VmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSdzIG5ldyB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBmaXJlUHJvcGVydHlDaGFuZ2UoU3RyaW5nIHByb3BlcnR5TmFtZSwgYnl0ZSBvbGRWYWx1ZSwgYnl0ZSBuZXdWYWx1ZSkgewotICAgICAgICBmaXJlUHJvcGVydHlDaGFuZ2VJbXBsKHByb3BlcnR5TmFtZSwgbmV3IEJ5dGUob2xkVmFsdWUpLCBuZXcgQnl0ZShuZXdWYWx1ZSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcG9ydCBhIGJvdW5kIHByb3BlcnR5IGNoYW5nZSBmb3IgYSBjaGFyLXZhbHVlZCBwcm9wZXJ0eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcGVydHlOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydHkgbmFtZS4KLSAgICAgKiBAcGFyYW0gb2xkVmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBvbGQgcHJvcGVydHkncyB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gbmV3VmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgcHJvcGVydHkncyB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBmaXJlUHJvcGVydHlDaGFuZ2UoU3RyaW5nIHByb3BlcnR5TmFtZSwgY2hhciBvbGRWYWx1ZSwgY2hhciBuZXdWYWx1ZSkgewotICAgICAgICBmaXJlUHJvcGVydHlDaGFuZ2VJbXBsKHByb3BlcnR5TmFtZSwgbmV3IENoYXJhY3RlcihvbGRWYWx1ZSksIG5ldyBDaGFyYWN0ZXIobmV3VmFsdWUpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXBvcnQgYSBib3VuZCBwcm9wZXJ0eSBjaGFuZ2UgZm9yIGEgc2hvcnQtdmFsdWVkIHByb3BlcnR5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm9wZXJ0eU5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSBuYW1lLgotICAgICAqIEBwYXJhbSBvbGRWYWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIG9sZCBwcm9wZXJ0eSdzIHZhbHVlLgotICAgICAqIEBwYXJhbSBuZXdWYWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBwcm9wZXJ0eSdzIHZhbHVlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGZpcmVQcm9wZXJ0eUNoYW5nZShTdHJpbmcgcHJvcGVydHlOYW1lLCBzaG9ydCBvbGRWYWx1ZSwgc2hvcnQgbmV3VmFsdWUpIHsKLSAgICAgICAgZmlyZVByb3BlcnR5Q2hhbmdlSW1wbChwcm9wZXJ0eU5hbWUsIG5ldyBTaG9ydChvbGRWYWx1ZSksIG5ldyBTaG9ydChuZXdWYWx1ZSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcG9ydCBhIGJvdW5kIHByb3BlcnR5IGNoYW5nZSBmb3IgYSBsb25nLXZhbHVlZCBwcm9wZXJ0eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcGVydHlOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydHkgbmFtZS4KLSAgICAgKiBAcGFyYW0gb2xkVmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBvbGQgcHJvcGVydHkncyB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gbmV3VmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgcHJvcGVydHkncyB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBmaXJlUHJvcGVydHlDaGFuZ2UoU3RyaW5nIHByb3BlcnR5TmFtZSwgbG9uZyBvbGRWYWx1ZSwgbG9uZyBuZXdWYWx1ZSkgewotICAgICAgICBmaXJlUHJvcGVydHlDaGFuZ2VJbXBsKHByb3BlcnR5TmFtZSwgbmV3IExvbmcob2xkVmFsdWUpLCBuZXcgTG9uZyhuZXdWYWx1ZSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcG9ydCBhIGJvdW5kIHByb3BlcnR5IGNoYW5nZSBmb3IgYSBmbG9hdC12YWx1ZWQgcHJvcGVydHkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb3BlcnR5TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIHByb3BlcnR5IG5hbWUuCi0gICAgICogQHBhcmFtIG9sZFZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgb2xkIHByb3BlcnR5J3MgdmFsdWUuCi0gICAgICogQHBhcmFtIG5ld1ZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHByb3BlcnR5J3MgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZmlyZVByb3BlcnR5Q2hhbmdlKFN0cmluZyBwcm9wZXJ0eU5hbWUsIGZsb2F0IG9sZFZhbHVlLCBmbG9hdCBuZXdWYWx1ZSkgewotICAgICAgICBmaXJlUHJvcGVydHlDaGFuZ2VJbXBsKHByb3BlcnR5TmFtZSwgbmV3IEZsb2F0KG9sZFZhbHVlKSwgbmV3IEZsb2F0KG5ld1ZhbHVlKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwb3J0IGEgYm91bmQgcHJvcGVydHkgY2hhbmdlIGZvciBhIGRvdWJsZS12YWx1ZWQgcHJvcGVydHkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb3BlcnR5TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIHByb3BlcnR5IG5hbWUuCi0gICAgICogQHBhcmFtIG9sZFZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgb2xkIHByb3BlcnR5J3MgdmFsdWUuCi0gICAgICogQHBhcmFtIG5ld1ZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHByb3BlcnR5J3MgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZmlyZVByb3BlcnR5Q2hhbmdlKFN0cmluZyBwcm9wZXJ0eU5hbWUsIGRvdWJsZSBvbGRWYWx1ZSwgZG91YmxlIG5ld1ZhbHVlKSB7Ci0gICAgICAgIGZpcmVQcm9wZXJ0eUNoYW5nZUltcGwocHJvcGVydHlOYW1lLCBuZXcgRG91YmxlKG9sZFZhbHVlKSwgbmV3IERvdWJsZShuZXdWYWx1ZSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFsaWdubWVudCBhbG9uZyB0aGUgeCBheGlzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFsaWdubWVudCBhbG9uZyB0aGUgeCBheGlzLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRBbGlnbm1lbnRYKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBDRU5URVJfQUxJR05NRU5UOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFsaWdubWVudCBhbG9uZyB0aGUgeSBheGlzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFsaWdubWVudCBhbG9uZyB5IGF4aXMuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldEFsaWdubWVudFkoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIENFTlRFUl9BTElHTk1FTlQ7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYmFja2dyb3VuZCBjb2xvciBmb3IgdGhpcyBjb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYmFja2dyb3VuZCBjb2xvciBmb3IgdGhpcyBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIENvbG9yIGdldEJhY2tncm91bmQoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgLy8gPz8/QVdUCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgICogaWYgKChiYWNrQ29sb3IgPT0gbnVsbCkgJiYgKHBhcmVudCAhPSBudWxsKSkgeyByZXR1cm4KLSAgICAgICAgICAgICAqIHBhcmVudC5nZXRCYWNrZ3JvdW5kKCk7IH0KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgcmV0dXJuIGJhY2tDb2xvcjsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZGluZyByZWN0YW5nbGUgb2YgdGhpcyBjb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlIG9mIHRoaXMgY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Qm91bmRzKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBib3VuZHMoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcml0ZXMgdGhlIGRhdGEgb2YgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZSB0byB0aGUgc3BlY2lmaWVkIFJlY3RhbmdsZQotICAgICAqIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcnYKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZWN0YW5nbGUgb2JqZWN0IHdoZXJlIHRoZSBib3VuZGluZyByZWN0YW5nbGUncyBkYXRhIGlzCi0gICAgICogICAgICAgICAgICBzdG9yZWQuCi0gICAgICogQHJldHVybiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Qm91bmRzKFJlY3RhbmdsZSBydikgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGlmIChydiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcnYgPSBuZXcgUmVjdGFuZ2xlKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBydi5zZXRCb3VuZHMoeCwgeSwgdywgaCk7Ci0gICAgICAgICAgICByZXR1cm4gcnY7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29sb3IgbW9kZWwgb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjb2xvciBtb2RlbCBvZiB0aGUgQ29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGdldFRvb2xraXQoKS5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQ29tcG9uZW50IHdoaWNoIGNvbnRhaW5zIHRoZSBzcGVjaWZpZWQgUG9pbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHAKLSAgICAgKiAgICAgICAgICAgIHRoZSBQb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSBDb21wb25lbnQgd2hpY2ggY29udGFpbnMgdGhlIHNwZWNpZmllZCBQb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29tcG9uZW50IGdldENvbXBvbmVudEF0KFBvaW50IHApIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gZ2V0Q29tcG9uZW50QXQocC54LCBwLnkpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIENvbXBvbmVudCB3aGljaCBjb250YWlucyB0aGUgcG9pbnQgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICogY29vcmRpbmF0ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHBvaW50LgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSBDb21wb25lbnQgd2hpY2ggY29udGFpbnMgdGhlIHBvaW50IHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgY29vcmRpbmF0ZXMuCi0gICAgICovCi0gICAgcHVibGljIENvbXBvbmVudCBnZXRDb21wb25lbnRBdChpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gbG9jYXRlKHgsIHkpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvbXBvbmVudCdzIG9yaWVudGF0aW9uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGNvbXBvbmVudCdzIG9yaWVudGF0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBDb21wb25lbnRPcmllbnRhdGlvbiBnZXRDb21wb25lbnRPcmllbnRhdGlvbigpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gb3JpZW50YXRpb247Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY3Vyc29yIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgQ3Vyc29yLgotICAgICAqLwotICAgIHB1YmxpYyBDdXJzb3IgZ2V0Q3Vyc29yKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGlmIChjdXJzb3IgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHJldHVybiBjdXJzb3I7Ci0gICAgICAgICAgICAgICAgLy8gPz8/QVdUCi0gICAgICAgICAgICAgICAgLyoKLSAgICAgICAgICAgICAgICAgKiB9IGVsc2UgaWYgKHBhcmVudCAhPSBudWxsKSB7IHJldHVybiBwYXJlbnQuZ2V0Q3Vyc29yKCk7Ci0gICAgICAgICAgICAgICAgICovCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gQ3Vyc29yLmdldERlZmF1bHRDdXJzb3IoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIHB1YmxpYyBEcm9wVGFyZ2V0IGdldERyb3BUYXJnZXQoKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4KLSAgICAgKiBkcm9wVGFyZ2V0OyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0gcHVibGljIENvbnRhaW5lcgotICAgICAqIGdldEZvY3VzQ3ljbGVSb290QW5jZXN0b3IoKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyBmb3IgKENvbnRhaW5lciBjID0KLSAgICAgKiBwYXJlbnQ7IGMgIT0gbnVsbDsgYyA9IGMuZ2V0UGFyZW50KCkpIHsgaWYgKGMuaXNGb2N1c0N5Y2xlUm9vdCgpKSB7Ci0gICAgICogcmV0dXJuIGM7IH0gfSByZXR1cm4gbnVsbDsgfSBmaW5hbGx5IHsgdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9Ci0gICAgICogQFN1cHByZXNzV2FybmluZ3MoInVuY2hlY2tlZCIpIHB1YmxpYyBTZXQ8QVdUS2V5U3Ryb2tlPgotICAgICAqIGdldEZvY3VzVHJhdmVyc2FsS2V5cyhpbnQgaWQpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7IEludGVnZXIga0lkID0KLSAgICAgKiBuZXcgSW50ZWdlcihpZCk7IEtleWJvYXJkRm9jdXNNYW5hZ2VyLmNoZWNrVHJhdmVyc2FsS2V5c0lEKHRyYXZlcnNhbEtleXMsCi0gICAgICoga0lkKTsgU2V0PD8gZXh0ZW5kcyBBV1RLZXlTdHJva2U+IGtleXMgPSB0cmF2ZXJzYWxLZXlzLmdldChrSWQpOyBpZiAoa2V5cwotICAgICAqID09IG51bGwgJiYgcGFyZW50ICE9IG51bGwpIHsga2V5cyA9IHBhcmVudC5nZXRGb2N1c1RyYXZlcnNhbEtleXMoaWQpOyB9Ci0gICAgICogaWYgKGtleXMgPT0gbnVsbCkgeyBrZXlzID0KLSAgICAgKiBLZXlib2FyZEZvY3VzTWFuYWdlci5nZXRDdXJyZW50S2V5Ym9hcmRGb2N1c01hbmFnZXIoKQotICAgICAqIC5nZXREZWZhdWx0Rm9jdXNUcmF2ZXJzYWxLZXlzKGlkKTsgfSByZXR1cm4gKFNldDxBV1RLZXlTdHJva2U+KSBrZXlzOyB9Ci0gICAgICogZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoZSB0aGUgZm9jdXMgdHJhdmVyc2FsIGtleXMgYXJlIGVuYWJsZWQgZm9yIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHRoZSBmb2N1cyB0cmF2ZXJzYWwga2V5cyBhcmUgZW5hYmxlZCBmb3IgdGhpcwotICAgICAqICAgICAgICAgY29tcG9uZW50LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gZ2V0Rm9jdXNUcmF2ZXJzYWxLZXlzRW5hYmxlZCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gZm9jdXNUcmF2ZXJzYWxLZXlzRW5hYmxlZDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmb250IG1ldHJpY3Mgb2YgdGhlIHNwZWNpZmllZCBGb250LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmCi0gICAgICogICAgICAgICAgICB0aGUgRm9udC4KLSAgICAgKiBAcmV0dXJuIHRoZSBGb250TWV0cmljcyBvZiB0aGUgc3BlY2lmaWVkIEZvbnQuCi0gICAgICovCi0gICAgQFN1cHByZXNzV2FybmluZ3MoImRlcHJlY2F0aW9uIikKLSAgICBwdWJsaWMgRm9udE1ldHJpY3MgZ2V0Rm9udE1ldHJpY3MoRm9udCBmKSB7Ci0gICAgICAgIHJldHVybiB0b29sa2l0LmdldEZvbnRNZXRyaWNzKGYpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZvcmVncm91bmQgY29sb3Igb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBmb3JlZ3JvdW5kIGNvbG9yIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIENvbG9yIGdldEZvcmVncm91bmQoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgLy8gPz8/QVdUCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgICogaWYgKGZvcmVDb2xvciA9PSBudWxsICYmIHBhcmVudCAhPSBudWxsKSB7IHJldHVybgotICAgICAgICAgICAgICogcGFyZW50LmdldEZvcmVncm91bmQoKTsgfQotICAgICAgICAgICAgICovCi0gICAgICAgICAgICByZXR1cm4gZm9yZUNvbG9yOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEdyYXBoaWNzIG9mIHRoZSBDb21wb25lbnQgb3IgbnVsbCBpZiB0aGlzIENvbXBvbmVudCBpcyBub3QKLSAgICAgKiBkaXNwbGF5YWJsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBHcmFwaGljcyBvZiB0aGUgQ29tcG9uZW50IG9yIG51bGwgaWYgdGhpcyBDb21wb25lbnQgaXMgbm90Ci0gICAgICogICAgICAgICBkaXNwbGF5YWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgR3JhcGhpY3MgZ2V0R3JhcGhpY3MoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKCFpc0Rpc3BsYXlhYmxlKCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIEdyYXBoaWNzIGcgPSBiZWhhdmlvdXIuZ2V0R3JhcGhpY3MoMCwgMCwgdywgaCk7Ci0gICAgICAgICAgICBnLnNldENvbG9yKGZvcmVDb2xvcik7Ci0gICAgICAgICAgICBnLnNldEZvbnQoZm9udCk7Ci0gICAgICAgICAgICByZXR1cm4gZzsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBHcmFwaGljc0NvbmZpZ3VyYXRpb24gYXNzb2NpYXRlZCB3aXRoIHRoaXMgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBhc3NvY2lhdGVkIHdpdGggdGhpcyBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBnZXRHcmFwaGljc0NvbmZpZ3VyYXRpb24oKSB7Ci0gICAgICAgIC8vID8/P0FXVAotICAgICAgICAvKgotICAgICAgICAgKiB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgV2luZG93IHdpbiA9IGdldFdpbmRvd0FuY2VzdG9yKCk7IGlmICh3aW4gPT0KLSAgICAgICAgICogbnVsbCkgeyByZXR1cm4gbnVsbDsgfSByZXR1cm4gd2luLmdldEdyYXBoaWNzQ29uZmlndXJhdGlvbigpOyB9Ci0gICAgICAgICAqIGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9Ci0gICAgICAgICAqLwotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBoZWlnaHQgb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBoZWlnaHQgb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEhlaWdodCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gaDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgcGFpbnQgbWVzc2FnZXMgcmVjZWl2ZWQgZnJvbSB0aGUgb3BlcmF0aW5nIHN5c3RlbSBzaG91bGQKLSAgICAgKiBiZSBpZ25vcmVkLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiBwYWludCBtZXNzYWdlcyByZWNlaXZlZCBmcm9tIHRoZSBvcGVyYXRpbmcgc3lzdGVtIHNob3VsZAotICAgICAqICAgICAgICAgYmUgaWdub3JlZCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGdldElnbm9yZVJlcGFpbnQoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGlnbm9yZVJlcGFpbnQ7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW5wdXQgY29udGV4dCBvZiB0aGlzIGNvbXBvbmVudCBmb3IgaGFuZGxpbmcgdGhlIGNvbW11bmljYXRpb24KLSAgICAgKiB3aXRoIGlucHV0IG1ldGhvZHMgd2hlbiB0ZXh0IGlzIGVudGVyZWQgaW4gdGhpcyBjb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgSW5wdXRDb250ZXh0IHVzZWQgYnkgdGhpcyBDb21wb25lbnQgb3IgbnVsbCBpZiBubyBjb250ZXh0IGlzCi0gICAgICogICAgICAgICBzcGVjaWZpbmVkLgotICAgICAqLwotICAgIHB1YmxpYyBJbnB1dENvbnRleHQgZ2V0SW5wdXRDb250ZXh0KCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIC8vID8/P0FXVAotICAgICAgICAgICAgLyoKLSAgICAgICAgICAgICAqIENvbnRhaW5lciBwYXJlbnQgPSBnZXRQYXJlbnQoKTsgaWYgKHBhcmVudCAhPSBudWxsKSB7IHJldHVybgotICAgICAgICAgICAgICogcGFyZW50LmdldElucHV0Q29udGV4dCgpOyB9Ci0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGlucHV0IG1ldGhvZCByZXF1ZXN0IGhhbmRsZXIgd2hpY2ggc3VwcG9ydHMgcmVxdWVzdHMgZnJvbSBpbnB1dAotICAgICAqIG1ldGhvZHMgZm9yIHRoaXMgY29tcG9uZW50LCBvciBudWxsIGZvciBkZWZhdWx0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGlucHV0IG1ldGhvZCByZXF1ZXN0IGhhbmRsZXIgd2hpY2ggc3VwcG9ydHMgcmVxdWVzdHMgZnJvbQotICAgICAqICAgICAgICAgaW5wdXQgbWV0aG9kcyBmb3IgdGhpcyBjb21wb25lbnQsIG9yIG51bGwgZm9yIGRlZmF1bHQuCi0gICAgICovCi0gICAgcHVibGljIElucHV0TWV0aG9kUmVxdWVzdHMgZ2V0SW5wdXRNZXRob2RSZXF1ZXN0cygpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9jYWxlIG9mIHRoaXMgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGxvY2FsZSBvZiB0aGlzIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgTG9jYWxlIGdldExvY2FsZSgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAvLyA/Pz9BV1QKLSAgICAgICAgICAgIC8qCi0gICAgICAgICAgICAgKiBpZiAobG9jYWxlID09IG51bGwpIHsgaWYgKHBhcmVudCA9PSBudWxsKSB7IGlmICh0aGlzIGluc3RhbmNlb2YKLSAgICAgICAgICAgICAqIFdpbmRvdykgeyByZXR1cm4gTG9jYWxlLmdldERlZmF1bHQoKTsgfSAvLyBhd3QuMTUwPW5vIHBhcmVudAotICAgICAgICAgICAgICogdGhyb3cgbmV3Ci0gICAgICAgICAgICAgKiBJbGxlZ2FsQ29tcG9uZW50U3RhdGVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTUwIikpOwotICAgICAgICAgICAgICogLy8kTk9OLU5MUy0xJCB9IHJldHVybiBnZXRQYXJlbnQoKS5nZXRMb2NhbGUoKTsgfQotICAgICAgICAgICAgICovCi0gICAgICAgICAgICByZXR1cm4gbG9jYWxlOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvY2F0aW9uIG9mIHRoaXMgY29tcG9uZW50IGluIHRoZSBmb3JtIG9mIGEgcG9pbnQgc3BlY2lmeWluZyB0aGUKLSAgICAgKiBjb21wb25lbnQncyB0b3AtbGVmdCBjb3JuZXIgaW4gdGhlIHNjcmVlbidzIGNvb3JkaW5hdGUgc3BhY2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgUG9pbnQgZ2l2aW5nIHRoZSBjb21wb25lbnQncyBsb2NhdGlvbiBpbiB0aGUgc2NyZWVuJ3MKLSAgICAgKiAgICAgICAgIGNvb3JkaW5hdGUgc3BhY2UuCi0gICAgICogQHRocm93cyBJbGxlZ2FsQ29tcG9uZW50U3RhdGVFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgY29tcG9uZW50IGlzIG5vdCBzaG93biBvbiB0aGUgc2NyZWVuLgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludCBnZXRMb2NhdGlvbk9uU2NyZWVuKCkgdGhyb3dzIElsbGVnYWxDb21wb25lbnRTdGF0ZUV4Y2VwdGlvbiB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgUG9pbnQgcCA9IG5ldyBQb2ludCgpOwotICAgICAgICAgICAgaWYgKGlzU2hvd2luZygpKSB7Ci0gICAgICAgICAgICAgICAgLy8gPz8/QVdUCi0gICAgICAgICAgICAgICAgLyoKLSAgICAgICAgICAgICAgICAgKiBDb21wb25lbnQgY29tcDsgZm9yIChjb21wID0gdGhpczsgY29tcCAhPSBudWxsICYmICEoY29tcAotICAgICAgICAgICAgICAgICAqIGluc3RhbmNlb2YgV2luZG93KTsgY29tcCA9IGNvbXAgLmdldFBhcmVudCgpKSB7Ci0gICAgICAgICAgICAgICAgICogcC50cmFuc2xhdGUoY29tcC5nZXRYKCksIGNvbXAuZ2V0WSgpKTsgfSBpZiAoY29tcCBpbnN0YW5jZW9mCi0gICAgICAgICAgICAgICAgICogV2luZG93KSB7IHAudHJhbnNsYXRlKGNvbXAuZ2V0WCgpLCBjb21wLmdldFkoKSk7IH0KLSAgICAgICAgICAgICAgICAgKi8KLSAgICAgICAgICAgICAgICByZXR1cm4gcDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIC8vIGF3dC4xNTE9Y29tcG9uZW50IG11c3QgYmUgc2hvd2luZyBvbiB0aGUgc2NyZWVuIHRvIGRldGVybWluZSBpdHMKLSAgICAgICAgICAgIC8vIGxvY2F0aW9uCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbENvbXBvbmVudFN0YXRlRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE1MSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHBlZXIuIFRoaXMgbWV0aG9kIHNob3VsZCBub3QgYmUgY2FsbGVkIGRpcmVjdGx5IGJ5IHVzZXIKLSAgICAgKiBhcHBsaWNhdGlvbnMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgQ29tcG9uZW50UGVlci4KLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSBpc0Rpc3BsYXlhYmxlKCkuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgQ29tcG9uZW50UGVlciBnZXRQZWVyKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGlmIChiZWhhdmlvdXIuaXNEaXNwbGF5YWJsZSgpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHBlZXI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIHRoZSBwcm9wZXJ0eSBjaGFuZ2UgbGlzdGVuZXJzIHJlZ2lzdGVyZWQgdG8gdGhpcwotICAgICAqIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIHRoZSBQcm9wZXJ0eUNoYW5nZUxpc3RlbmVycyByZWdpc3RlcmVkIHRvIHRoaXMKLSAgICAgKiAgICAgICAgIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcltdIGdldFByb3BlcnR5Q2hhbmdlTGlzdGVuZXJzKCkgewotICAgICAgICByZXR1cm4gZ2V0UHJvcGVydHlDaGFuZ2VTdXBwb3J0KCkuZ2V0UHJvcGVydHlDaGFuZ2VMaXN0ZW5lcnMoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIFByb3BlcnR5Q2hhbmdlTGlzdGVuZXIgb2JqZWN0cyByZWdpc3RlcmVkIHRvIHRoaXMKLSAgICAgKiBDb21wb25lbnQgZm9yIHRoZSBzcGVjaWZpZWQgcHJvcGVydHkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb3BlcnR5TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIHByb3BlcnR5IG5hbWUuCi0gICAgICogQHJldHVybiBhbiBhcnJheSBvZiBQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyIG9iamVjdHMgcmVnaXN0ZXJlZCB0byB0aGlzCi0gICAgICogICAgICAgICBDb21wb25lbnQgZm9yIHRoZSBzcGVjaWZpZWQgcHJvcGVydHkuCi0gICAgICovCi0gICAgcHVibGljIFByb3BlcnR5Q2hhbmdlTGlzdGVuZXJbXSBnZXRQcm9wZXJ0eUNoYW5nZUxpc3RlbmVycyhTdHJpbmcgcHJvcGVydHlOYW1lKSB7Ci0gICAgICAgIHJldHVybiBnZXRQcm9wZXJ0eUNoYW5nZVN1cHBvcnQoKS5nZXRQcm9wZXJ0eUNoYW5nZUxpc3RlbmVycyhwcm9wZXJ0eU5hbWUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHdpZHRoIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgd2lkdGggb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFdpZHRoKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiB3OwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgY29tcG9uZW50J3MgdG9wLWxlZnQgY29ybmVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgY29tcG9uZW50J3MgdG9wLWxlZnQgY29ybmVyLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0WCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4geDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGNvbXBvbmVudCdzIHRvcC1sZWZ0IGNvcm5lci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGNvbXBvbmVudCdzIHRvcC1sZWZ0IGNvcm5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFkoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIHk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR290IHRoZSBmb2N1cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXZ0Ci0gICAgICogICAgICAgICAgICB0aGUgRXZlbnQuCi0gICAgICogQHBhcmFtIHdoYXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IHByb2Nlc3NGb2N1c0V2ZW50KEZvY3VzRXZlbnQpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBib29sZWFuIGdvdEZvY3VzKEV2ZW50IGV2dCwgT2JqZWN0IHdoYXQpIHsKLSAgICAgICAgLy8gdG8gYmUgb3ZlcnJpZGRlbjogZG8gbm90aGluZywKLSAgICAgICAgLy8ganVzdCByZXR1cm4gZmFsc2UgdG8gcHJvcGFnYXRlIGV2ZW50IHVwIHRvIHRoZSBwYXJlbnQgY29udGFpbmVyCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBIYW5kbGVzIGV2ZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBldnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBFdmVudC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN1Y2Nlc3NmdWwuCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgcHJvY2Vzc0V2ZW50KEFXVEV2ZW50KSBtZXRob2QuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgYm9vbGVhbiBoYW5kbGVFdmVudChFdmVudCBldnQpIHsKLSAgICAgICAgc3dpdGNoIChldnQuaWQpIHsKLSAgICAgICAgICAgIGNhc2UgRXZlbnQuQUNUSU9OX0VWRU5UOgotICAgICAgICAgICAgICAgIHJldHVybiBhY3Rpb24oZXZ0LCBldnQuYXJnKTsKLSAgICAgICAgICAgIGNhc2UgRXZlbnQuR09UX0ZPQ1VTOgotICAgICAgICAgICAgICAgIHJldHVybiBnb3RGb2N1cyhldnQsIG51bGwpOwotICAgICAgICAgICAgY2FzZSBFdmVudC5MT1NUX0ZPQ1VTOgotICAgICAgICAgICAgICAgIHJldHVybiBsb3N0Rm9jdXMoZXZ0LCBudWxsKTsKLSAgICAgICAgICAgIGNhc2UgRXZlbnQuTU9VU0VfRE9XTjoKLSAgICAgICAgICAgICAgICByZXR1cm4gbW91c2VEb3duKGV2dCwgZXZ0LngsIGV2dC55KTsKLSAgICAgICAgICAgIGNhc2UgRXZlbnQuTU9VU0VfRFJBRzoKLSAgICAgICAgICAgICAgICByZXR1cm4gbW91c2VEcmFnKGV2dCwgZXZ0LngsIGV2dC55KTsKLSAgICAgICAgICAgIGNhc2UgRXZlbnQuTU9VU0VfRU5URVI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG1vdXNlRW50ZXIoZXZ0LCBldnQueCwgZXZ0LnkpOwotICAgICAgICAgICAgY2FzZSBFdmVudC5NT1VTRV9FWElUOgotICAgICAgICAgICAgICAgIHJldHVybiBtb3VzZUV4aXQoZXZ0LCBldnQueCwgZXZ0LnkpOwotICAgICAgICAgICAgY2FzZSBFdmVudC5NT1VTRV9NT1ZFOgotICAgICAgICAgICAgICAgIHJldHVybiBtb3VzZU1vdmUoZXZ0LCBldnQueCwgZXZ0LnkpOwotICAgICAgICAgICAgY2FzZSBFdmVudC5NT1VTRV9VUDoKLSAgICAgICAgICAgICAgICByZXR1cm4gbW91c2VVcChldnQsIGV2dC54LCBldnQueSk7Ci0gICAgICAgICAgICBjYXNlIEV2ZW50LktFWV9BQ1RJT046Ci0gICAgICAgICAgICBjYXNlIEV2ZW50LktFWV9QUkVTUzoKLSAgICAgICAgICAgICAgICByZXR1cm4ga2V5RG93bihldnQsIGV2dC5rZXkpOwotICAgICAgICAgICAgY2FzZSBFdmVudC5LRVlfQUNUSU9OX1JFTEVBU0U6Ci0gICAgICAgICAgICBjYXNlIEV2ZW50LktFWV9SRUxFQVNFOgotICAgICAgICAgICAgICAgIHJldHVybiBrZXlVcChldnQsIGV2dC5rZXkpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsvLyBldmVudCBub3QgaGFuZGxlZAotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIHRoZSBDb21wb25lbnQgaXMgdGhlIGZvY3VzIG93bmVyIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBDb21wb25lbnQgaXMgdGhlIGZvY3VzIG93bmVyLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaGFzRm9jdXMoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgLy8gPz8/QVdUOiByZXR1cm4gaXNGb2N1c093bmVyKCk7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSGlkZXMgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSBzZXRWaXNpYmxlKGJvb2xlYW4pIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyB2b2lkIGhpZGUoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKCF2aXNpYmxlKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcHJlcGFyZTRIaWVyYXJjaHlDaGFuZ2UoKTsKLSAgICAgICAgICAgIHZpc2libGUgPSBmYWxzZTsKLSAgICAgICAgICAgIG1vdmVGb2N1c09uSGlkZSgpOwotICAgICAgICAgICAgYmVoYXZpb3VyLnNldFZpc2libGUoZmFsc2UpOwotICAgICAgICAgICAgcG9zdEV2ZW50KG5ldyBDb21wb25lbnRFdmVudCh0aGlzLCBDb21wb25lbnRFdmVudC5DT01QT05FTlRfSElEREVOKSk7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGZpbmlzaEhpZXJhcmNoeUNoYW5nZSh0aGlzLCBwYXJlbnQsIDApOwotICAgICAgICAgICAgbm90aWZ5SW5wdXRNZXRob2QobnVsbCk7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGludmFsaWRhdGVSZWFsUGFyZW50KCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IHRoZSBwb2ludCB3aXRoIHRoZSBzcGVjaWZpZWQgY29vcmRpbmF0ZXMgYmVsb25ncyB0bwotICAgICAqIHRoZSBDb21tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBQb2ludC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgUG9pbnQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgcG9pbnQgd2l0aCB0aGUgc3BlY2lmaWVkIGNvb3JkaW5hdGVzIGJlbG9uZ3MgdG8gdGhlCi0gICAgICogICAgICAgICBDb21tcG9uZW50LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgY29udGFpbnMoaW50LCBpbnQpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBib29sZWFuIGluc2lkZShpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4geCA+PSAwICYmIHggPCBnZXRXaWR0aCgpICYmIHkgPj0gMCAmJiB5IDwgZ2V0SGVpZ2h0KCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW52YWxpZGF0ZXMgdGhlIGNvbXBvbmVudCwgdGhpcyBjb21wb25lbnQgYW5kIGFsbCBwYXJlbnRzIGFib3ZlIGl0IGFyZQotICAgICAqIG1hcmtlZCBhcyBuZWVkaW5nIHRvIGJlIGxhaWQgb3V0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGludmFsaWRhdGUoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgdmFsaWQgPSBmYWxzZTsKLSAgICAgICAgICAgIHJlc2V0RGVmYXVsdFNpemUoKTsKLSAgICAgICAgICAgIC8vID8/P0FXVDogaW52YWxpZGF0ZVJlYWxQYXJlbnQoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIGJhY2tncm91bmQgY29sb3IgaXMgc2V0IHRvIHRoaXMgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIGJhY2tncm91bmQgY29sb3IgaXMgc2V0IHRvIHRoaXMgQ29tcG9uZW50LCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzQmFja2dyb3VuZFNldCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gYmFja0NvbG9yICE9IG51bGw7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IGEgY3Vyc29yIGlzIHNldCBmb3IgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGEgY3Vyc29yIGlzIHNldCBmb3IgdGhlIENvbXBvbmVudCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzQ3Vyc29yU2V0KCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBjdXJzb3IgIT0gbnVsbDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhpcyBDb21wb25lbnQgaXMgZGlzcGxheWFibGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIENvbXBvbmVudCBpcyBkaXNwbGF5YWJsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzRGlzcGxheWFibGUoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGJlaGF2aW91ci5pc0Rpc3BsYXlhYmxlKCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IHRoaXMgY29tcG9uZW50IGlzIHBhaW50ZWQgdG8gYW4gYnVmZmVyIHdoaWNoIGlzCi0gICAgICogY29waWVkIHRvIHRoZSBzY3JlZW4gbGF0ZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIGNvbXBvbmVudCBpcyBwYWludGVkIHRvIGFuIGJ1ZmZlciB3aGljaCBpcyBjb3BpZWQKLSAgICAgKiAgICAgICAgIHRvIHRoZSBzY3JlZW4gbGF0ZXIsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0RvdWJsZUJ1ZmZlcmVkKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIC8vIGZhbHNlIGJ5IGRlZmF1bHQKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhpcyBDb21wb25lbnQgaXMgZW5hYmxlZC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgQ29tcG9uZW50IGlzIGVuYWJsZWQsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0VuYWJsZWQoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGVuYWJsZWQ7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogIlJlY3Vyc2l2ZSIgaXNFbmFibGVkKCkuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlIGlmIG5vdCBvbmx5IGNvbXBvbmVudCBpdHNlbGYgaXMgZW5hYmxlZCBidXQgaXRzIGhlYXZ5d2VpZ2h0Ci0gICAgICogICAgICAgICBwYXJlbnQgaXMgYWxzbyAiaW5kaXJlY3RseSIgZW5hYmxlZC4KLSAgICAgKi8KLSAgICBib29sZWFuIGlzSW5kaXJlY3RseUVuYWJsZWQoKSB7Ci0gICAgICAgIENvbXBvbmVudCBjb21wID0gdGhpczsKLSAgICAgICAgd2hpbGUgKGNvbXAgIT0gbnVsbCkgewotICAgICAgICAgICAgaWYgKCFjb21wLmlzTGlnaHR3ZWlnaHQoKSAmJiAhY29tcC5pc0VuYWJsZWQoKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIC8vID8/P0FXVDogY29tcCA9IGNvbXAuZ2V0UmVhbFBhcmVudCgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGUgY29tcG9uZW50IGlzIGtleSBlbmFibGVkLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIGNvbXBvbmVudCBpcyBlbmFibGVkIGFuZCBpbmRpcmVjdGx5IGVuYWJsZWQuCi0gICAgICovCi0gICAgYm9vbGVhbiBpc0tleUVuYWJsZWQoKSB7Ci0gICAgICAgIGlmICghaXNFbmFibGVkKCkpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gaXNJbmRpcmVjdGx5RW5hYmxlZCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgb25seSBwYXJlbnQgb2YgYSBjaGlsZCBjb21wb25lbnQsIGJ1dCBub3Qgb3duZXIgb2YgYSB3aW5kb3cuCi0gICAgICogCi0gICAgICogQHJldHVybiBwYXJlbnQgb2YgY2hpbGQgY29tcG9uZW50LCBudWxsIGlmIGNvbXBvbmVudCBpcyBhIHRvcC1sZXZlbAotICAgICAqICAgICAgICAgKFdpbmRvdyBpbnN0YW5jZSkuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBDb250YWluZXIgZ2V0UmVhbFBhcmVudCgpIHsgcmV0dXJuICghKHRoaXMgaW5zdGFuY2VvZiBXaW5kb3cpID8KLSAgICAgKiBnZXRQYXJlbnQoKSA6IG51bGwpOyB9IHB1YmxpYyBib29sZWFuIGlzRm9jdXNDeWNsZVJvb3QoQ29udGFpbmVyCi0gICAgICogY29udGFpbmVyKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4gZ2V0Rm9jdXNDeWNsZVJvb3RBbmNlc3RvcigpCi0gICAgICogPT0gY29udGFpbmVyOyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0gcHVibGljIGJvb2xlYW4KLSAgICAgKiBpc0ZvY3VzT3duZXIoKSB7IHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4KLSAgICAgKiBLZXlib2FyZEZvY3VzTWFuYWdlci5nZXRDdXJyZW50S2V5Ym9hcmRGb2N1c01hbmFnZXIoKS5nZXRGb2N1c093bmVyKCkgPT0KLSAgICAgKiB0aGlzOyB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKi8KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGlzIENvbXBvbmVudCBjYW4gYmUgZm9jdXNhYmxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBDb21wb25lbnQgY2FuIGJlIGZvY3VzYWJsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IGlzRm9jdXNhYmxlKCkuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgYm9vbGVhbiBpc0ZvY3VzVHJhdmVyc2FibGUoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgb3ZlcnJpZGVuSXNGb2N1c2FibGUgPSBmYWxzZTsKLSAgICAgICAgICAgIHJldHVybiBmb2N1c2FibGU7IC8vIGEgQ29tcG9uZW50IG11c3QgZWl0aGVyIGJlIGJvdGggZm9jdXNhYmxlIGFuZAotICAgICAgICAgICAgLy8gZm9jdXMgdHJhdmVyc2FibGUsIG9yIG5laXRoZXIKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyBDb21wb25lbnQgY2FuIGJlIGZvY3VzYWJsZSBvciBub3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIENvbXBvbmVudCBjYW4gYmUgZm9jdXNhYmxlLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNGb2N1c2FibGUoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGlzRm9jdXNUcmF2ZXJzYWJsZSgpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGUgRm9udCBpcyBzZXQgZm9yIHRoaXMgQ29tcG9uZW50IG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBGb250IGlzIHNldCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzRm9udFNldCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gZm9udCAhPSBudWxsOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiBmb3JlZ3JvdW5kIGNvbG9yIGlzIHNldCBmb3IgdGhlIENvbXBvbmVudCBvciBub3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBpcyBmb3JlZ3JvdW5kIGNvbG9yIGlzIHNldCBmb3IgdGhlIENvbXBvbmVudCwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0ZvcmVncm91bmRTZXQoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGZvcmVDb2xvciAhPSBudWxsOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGlzIGNvbXBvbmVudCBoYXMgYSBsaWdodHdlaWdodCBwZWVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBjb21wb25lbnQgaGFzIGEgbGlnaHR3ZWlnaHQgcGVlciwgZmFsc2UgaWYgaXQgaGFzIGEKLSAgICAgKiAgICAgICAgIG5hdGl2ZSBwZWVyIG9yIG5vIHBlZXIuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNMaWdodHdlaWdodCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gYmVoYXZpb3VyLmlzTGlnaHR3ZWlnaHQoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhpcyBDb21wb25lbnQgaXMgc2hvd24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIENvbXBvbmVudCBpcyBzaG93biwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzU2hvd2luZygpIHsKLSAgICAgICAgLy8gPz8/QVdUCi0gICAgICAgIC8qCi0gICAgICAgICAqIHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4gKGlzVmlzaWJsZSgpICYmIGlzRGlzcGxheWFibGUoKSAmJgotICAgICAgICAgKiAocGFyZW50ICE9IG51bGwpICYmIHBhcmVudC5pc1Nob3dpbmcoKSk7IH0gZmluYWxseSB7Ci0gICAgICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0KLSAgICAgICAgICovCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhpcyBDb21wb25lbnQgaXMgdmlzaWJsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBDb21wb25lbnQgaXMgdmlzaWJsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzVmlzaWJsZSgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gdmlzaWJsZTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXByZWNhdGVkOiByZXBsYWNlZCBieSBwcm9jZXNzS2V5RXZlbnQoS2V5RXZlbnQpIG1ldGhvZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXZ0Ci0gICAgICogICAgICAgICAgICB0aGUgRXZlbnQuCi0gICAgICogQHBhcmFtIGtleQotICAgICAqICAgICAgICAgICAgdGhlIGtleSBjb2RlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgc3VjY2Vzc2Z1bC4KLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSByZXBsYWNlZCBieSBwcm9jZXNzS2V5RXZlbnQoS2V5RXZlbnQpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBib29sZWFuIGtleURvd24oRXZlbnQgZXZ0LCBpbnQga2V5KSB7Ci0gICAgICAgIC8vIHRvIGJlIG92ZXJyaWRkZW46IGRvIG5vdGhpbmcsCi0gICAgICAgIC8vIGp1c3QgcmV0dXJuIGZhbHNlIHRvIHByb3BhZ2F0ZSBldmVudCB1cCB0byB0aGUgcGFyZW50IGNvbnRhaW5lcgotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGVwcmVjYXRlZDogcmVwbGFjZWQgYnkgcHJvY2Vzc0tleUV2ZW50KEtleUV2ZW50KSBtZXRob2QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2dAotICAgICAqICAgICAgICAgICAgdGhlIEV2ZW50LgotICAgICAqIEBwYXJhbSBrZXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBrZXkgY29kZS4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN1Y2Nlc3NmdWwuCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgcHJvY2Vzc0tleUV2ZW50KEtleUV2ZW50KSBtZXRob2QuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgYm9vbGVhbiBrZXlVcChFdmVudCBldnQsIGludCBrZXkpIHsKLSAgICAgICAgLy8gdG8gYmUgb3ZlcnJpZGRlbjogZG8gbm90aGluZywKLSAgICAgICAgLy8ganVzdCByZXR1cm4gZmFsc2UgdG8gcHJvcGFnYXRlIGV2ZW50IHVwIHRvIHRoZSBwYXJlbnQgY29udGFpbmVyCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXByZWNhdGVkOiBSZXBsYWNlZCBieSBkb0xheW91dCgpIG1ldGhvZC4KLSAgICAgKiAKLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSBkb0xheW91dCgpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyB2b2lkIGxheW91dCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAvLyBJbXBsZW1lbnRlZCBpbiBDb250YWluZXIKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXByZWNhdGVkOiByZXBsYWNlZCBieSBnZXRDb21wb25lbnRBdChpbnQsIGludCkgbWV0aG9kLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlLgotICAgICAqIEByZXR1cm4gVGhlIGNvbXBvbmVudC4KLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSBnZXRDb21wb25lbnRBdChpbnQsIGludCkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIENvbXBvbmVudCBsb2NhdGUoaW50IHgsIGludCB5KSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKGNvbnRhaW5zKHgsIHkpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRoaXM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXByZWNhdGVkOiByZXBsYWNlZCBieSBwcm9jZXNzRm9jdXNFdmVudChGb2N1c0V2ZW50KS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXZ0Ci0gICAgICogICAgICAgICAgICB0aGUgRXZlbnQuCi0gICAgICogQHBhcmFtIHdoYXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IHByb2Nlc3NGb2N1c0V2ZW50KEZvY3VzRXZlbnQpLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIGJvb2xlYW4gbG9zdEZvY3VzKEV2ZW50IGV2dCwgT2JqZWN0IHdoYXQpIHsKLSAgICAgICAgLy8gdG8gYmUgb3ZlcnJpZGRlbjogZG8gbm90aGluZywKLSAgICAgICAgLy8ganVzdCByZXR1cm4gZmFsc2UgdG8gcHJvcGFnYXRlIGV2ZW50IHVwIHRvIHRoZSBwYXJlbnQgY29udGFpbmVyCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXByZWNhdGVkOiByZXBsYWNlZCBieSBwcm9jZXNzTW91c2VFdmVudChNb3VzZUV2ZW50KSBtZXRob2QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2dAotICAgICAqICAgICAgICAgICAgdGhlIE1vdXNlRXZlbnQuCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IHByb2Nlc3NNb3VzZUV2ZW50KE1vdXNlRXZlbnQpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBib29sZWFuIG1vdXNlRG93bihFdmVudCBldnQsIGludCB4LCBpbnQgeSkgewotICAgICAgICAvLyB0byBiZSBvdmVycmlkZGVuOiBkbyBub3RoaW5nLAotICAgICAgICAvLyBqdXN0IHJldHVybiBmYWxzZSB0byBwcm9wYWdhdGUgZXZlbnQgdXAgdG8gdGhlIHBhcmVudCBjb250YWluZXIKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERlcHJlY2F0ZWQ6IHJlcGxhY2VkIGJ5IGdldE1pbmltdW1TaXplKCkgbWV0aG9kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBldnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBFdmVudC4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZS4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN1Y2Nlc3NmdWwuCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgZ2V0TWluaW11bVNpemUoKSBtZXRob2QuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgYm9vbGVhbiBtb3VzZURyYWcoRXZlbnQgZXZ0LCBpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgLy8gdG8gYmUgb3ZlcnJpZGRlbjogZG8gbm90aGluZywKLSAgICAgICAgLy8ganVzdCByZXR1cm4gZmFsc2UgdG8gcHJvcGFnYXRlIGV2ZW50IHVwIHRvIHRoZSBwYXJlbnQgY29udGFpbmVyCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXBsYWNlZCBieSBwcm9jZXNzTW91c2VFdmVudChNb3VzZUV2ZW50KSBtZXRob2QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2dAotICAgICAqICAgICAgICAgICAgdGhlIEV2ZW50LgotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgc3VjY2Vzc2Z1bC4KLSAgICAgKiBAZGVwcmVjYXRlZCByZXBsYWNlZCBieSBwcm9jZXNzTW91c2VFdmVudChNb3VzZUV2ZW50KSBtZXRob2QuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgYm9vbGVhbiBtb3VzZUVudGVyKEV2ZW50IGV2dCwgaW50IHgsIGludCB5KSB7Ci0gICAgICAgIC8vIHRvIGJlIG92ZXJyaWRkZW46IGRvIG5vdGhpbmcsCi0gICAgICAgIC8vIGp1c3QgcmV0dXJuIGZhbHNlIHRvIHByb3BhZ2F0ZSBldmVudCB1cCB0byB0aGUgcGFyZW50IGNvbnRhaW5lcgotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwbGFjZWQgYnkgcHJvY2Vzc01vdXNlRXZlbnQoTW91c2VFdmVudCkgbWV0aG9kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBldnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBFdmVudC4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZS4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN1Y2Nlc3NmdWwuCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgcHJvY2Vzc01vdXNlRXZlbnQoTW91c2VFdmVudCkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIGJvb2xlYW4gbW91c2VFeGl0KEV2ZW50IGV2dCwgaW50IHgsIGludCB5KSB7Ci0gICAgICAgIC8vIHRvIGJlIG92ZXJyaWRkZW46IGRvIG5vdGhpbmcsCi0gICAgICAgIC8vIGp1c3QgcmV0dXJuIGZhbHNlIHRvIHByb3BhZ2F0ZSBldmVudCB1cCB0byB0aGUgcGFyZW50IGNvbnRhaW5lcgotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwbGFjZWQgYnkgcHJvY2Vzc01vdXNlRXZlbnQoTW91c2VFdmVudCkgbWV0aG9kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBldnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBFdmVudC4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZS4KLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSBwcm9jZXNzTW91c2VFdmVudChNb3VzZUV2ZW50KSBtZXRob2QuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIGJvb2xlYW4gbW91c2VNb3ZlKEV2ZW50IGV2dCwgaW50IHgsIGludCB5KSB7Ci0gICAgICAgIC8vIHRvIGJlIG92ZXJyaWRkZW46IGRvIG5vdGhpbmcsCi0gICAgICAgIC8vIGp1c3QgcmV0dXJuIGZhbHNlIHRvIHByb3BhZ2F0ZSBldmVudCB1cCB0byB0aGUgcGFyZW50IGNvbnRhaW5lcgotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwbGFjZWQgYnkgcHJvY2Vzc01vdXNlRXZlbnQoTW91c2VFdmVudCkgbWV0aG9kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBldnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBFdmVudC4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZS4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN1Y2Nlc3NmdWwuCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgcHJvY2Vzc01vdXNlRXZlbnQoTW91c2VFdmVudCkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIGJvb2xlYW4gbW91c2VVcChFdmVudCBldnQsIGludCB4LCBpbnQgeSkgewotICAgICAgICAvLyB0byBiZSBvdmVycmlkZGVuOiBkbyBub3RoaW5nLAotICAgICAgICAvLyBqdXN0IHJldHVybiBmYWxzZSB0byBwcm9wYWdhdGUgZXZlbnQgdXAgdG8gdGhlIHBhcmVudCBjb250YWluZXIKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERlcHJlY2F0ZWQ6IHJlcGxhY2VkIGJ5IHNldExvY2F0aW9uKGludCwgaW50KSBtZXRob2QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGVzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlcy4KLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSBzZXRMb2NhdGlvbihpbnQsIGludCkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIHZvaWQgbW92ZShpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBib3VuZHNNYXNrUGFyYW0gPSBOYXRpdmVXaW5kb3cuQk9VTkRTX05PU0laRTsKLSAgICAgICAgICAgIHNldEJvdW5kcyh4LCB5LCB3LCBoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIEBEZXByZWNhdGVkIHB1YmxpYyB2b2lkIG5leHRGb2N1cygpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7Ci0gICAgICogdHJhbnNmZXJGb2N1cyhLZXlib2FyZEZvY3VzTWFuYWdlci5GT1JXQVJEX1RSQVZFUlNBTF9LRVlTKTsgfSBmaW5hbGx5IHsKLSAgICAgKiB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKi8KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIGNvbXBvbmVudCdzIHN0YXRlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgY29tcG9uZW50J3Mgc3RhdGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZyBwYXJhbVN0cmluZygpIHsKLSAgICAgICAgLyoKLSAgICAgICAgICogVGhlIGZvcm1hdCBpcyBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvciB3aGljaCBjYW4gYmUgcmV2ZWFsZWQgYnkKLSAgICAgICAgICogdGhlIGZvbGxvd2luZyBjb2RlOiBDb21wb25lbnQgYyA9IG5ldyBDb21wb25lbnQoKXt9OwotICAgICAgICAgKiBjLnNldFZpc2libGUoZmFsc2UpOyBTeXN0ZW0ub3V0LnByaW50bG4oYyk7Ci0gICAgICAgICAqLwotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBnZXROYW1lKCkgKyAiLCIgKyBnZXRYKCkgKyAiLCIgKyBnZXRZKCkgKyAiLCIgKyBnZXRXaWR0aCgpICsgIngiIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQKLSAgICAgICAgICAgICAgICAgICAgKyBnZXRIZWlnaHQoKSArICghaXNWaXNpYmxlKCkgPyAiLGhpZGRlbiIgOiAiIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBEZXByZWNhdGVkCi0gICAgQFN1cHByZXNzV2FybmluZ3MoImRlcHJlY2F0aW9uIikKLSAgICBwdWJsaWMgYm9vbGVhbiBwb3N0RXZlbnQoRXZlbnQgZXZ0KSB7Ci0gICAgICAgIGJvb2xlYW4gaGFuZGxlZCA9IGhhbmRsZUV2ZW50KGV2dCk7Ci0gICAgICAgIGlmIChoYW5kbGVkKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgICAgICAvLyA/Pz9BV1QKLSAgICAgICAgLyoKLSAgICAgICAgICogLy8gcHJvcGFnYXRlIG5vbi1oYW5kbGVkIGV2ZW50cyB1cCB0byBwYXJlbnQgQ29tcG9uZW50IHBhciA9IHBhcmVudDsKLSAgICAgICAgICogLy8gdHJ5IHRvIGNhbGwgcG9zdEV2ZW50IG9ubHkgb24gY29tcG9uZW50cyB3aGljaCAvLyBvdmVycmlkZSBhbnkgb2YKLSAgICAgICAgICogZGVwcmVjYXRlZCBtZXRob2QgaGFuZGxlcnMgLy8gd2hpbGUgKHBhciAhPSBudWxsICYmCi0gICAgICAgICAqICFwYXIuZGVwcmVjYXRlZEV2ZW50SGFuZGxlcikgeyAvLyBwYXIgPSBwYXIucGFyZW50OyAvLyB9IC8vIHRyYW5zbGF0ZQotICAgICAgICAgKiBldmVudCBjb29yZGluYXRlcyBiZWZvcmUgcG9zdGluZyBpdCB0byBwYXJlbnQgaWYgKHBhciAhPSBudWxsKSB7Ci0gICAgICAgICAqIGV2dC50cmFuc2xhdGUoeCwgeSk7IHBhci5wb3N0RXZlbnQoZXZ0KTsgfQotICAgICAgICAgKi8KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByZXBhcmVzIGFuIGltYWdlIGZvciByZW5kZXJpbmcgb24gdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZSB0byBiZSBwcmVwYXJlZC4KLSAgICAgKiBAcGFyYW0gb2JzZXJ2ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZU9ic2VydmVyIG9iamVjdCB0byBiZSBub3RpZmllZCBhcyBzb29uIGFzIHRoZSBpbWFnZQotICAgICAqICAgICAgICAgICAgaXMgcHJlcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlIGlmIHRoZSBpbWFnZSBoYXMgYmVlbiBmdWxseSBwcmVwYXJlZCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIHByZXBhcmVJbWFnZShJbWFnZSBpbWFnZSwgSW1hZ2VPYnNlcnZlciBvYnNlcnZlcikgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiB0b29sa2l0LnByZXBhcmVJbWFnZShpbWFnZSwgLTEsIC0xLCBvYnNlcnZlcik7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJlcGFyZXMgYW4gaW1hZ2UgZm9yIHJlbmRlcmluZyBvbiB0aGUgQ29tcG9uZW50IHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIHdpZHRoLCBoZWlnaHQsIGFuZCBJbWFnZU9ic2VydmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlIHRvIGJlIHByZXBhcmVkLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHNjYWxlZCBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHNjYWxlZCBoZWlnaHQuCi0gICAgICogQHBhcmFtIG9ic2VydmVyCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VPYnNlcnZlciBvYmplY3QgdG8gYmUgbm90aWZpZWQgYXMgc29vbiBhcyB0aGUgaW1hZ2UKLSAgICAgKiAgICAgICAgICAgIGlzIHByZXBhcmVkLgotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgaW1hZ2UgaXMgYmVlbiBmdWxseSBwcmVwYXJlZCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIHByZXBhcmVJbWFnZShJbWFnZSBpbWFnZSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBJbWFnZU9ic2VydmVyIG9ic2VydmVyKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIHRvb2xraXQucHJlcGFyZUltYWdlKGltYWdlLCB3aWR0aCwgaGVpZ2h0LCBvYnNlcnZlcik7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogTWFrZXMgdGhpcyBDb21wb25lbnQgdW5kaXNwbGF5YWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVOb3RpZnkoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgLy8gPz8/QVdUCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgICogaWYgKGRyb3BUYXJnZXQgIT0gbnVsbCkgeyBkcm9wVGFyZ2V0LnJlbW92ZU5vdGlmeShwZWVyKTsgfQotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBwcmVwYXJlNEhpZXJhcmNoeUNoYW5nZSgpOwotICAgICAgICAgICAgLy8gLz8/P0FXVDogbW92ZUZvY3VzKCk7Ci0gICAgICAgICAgICBiZWhhdmlvdXIucmVtb3ZlTm90aWZ5KCk7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGZpbmlzaEhpZXJhcmNoeUNoYW5nZSh0aGlzLCBwYXJlbnQsIDApOwotICAgICAgICAgICAgcmVtb3ZlTm90aWZ5SW5wdXRDb250ZXh0KCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2FsbHMgSW5wdXRDb250ZXh0LnJlbW92ZU5vdGlmeS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgcmVtb3ZlTm90aWZ5SW5wdXRDb250ZXh0KCkgewotICAgICAgICBpZiAoIWlucHV0TWV0aG9kc0VuYWJsZWQpIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBJbnB1dENvbnRleHQgaWMgPSBnZXRJbnB1dENvbnRleHQoKTsKLSAgICAgICAgaWYgKGljICE9IG51bGwpIHsKLSAgICAgICAgICAgIC8vID8/P0FXVDogaWMucmVtb3ZlTm90aWZ5KHRoaXMpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhpcyBtZXRob2QgaXMgY2FsbGVkIHdoZW4gc29tZSBwcm9wZXJ0eSBvZiBhIGNvbXBvbmVudCBjaGFuZ2VzLCBtYWtpbmcKLSAgICAgKiBpdCB1bmZvY3VzYWJsZSwgZS4gZy4gaGlkZSgpLCByZW1vdmVOb3RpZnkoKSwgc2V0RW5hYmxlZChmYWxzZSksCi0gICAgICogc2V0Rm9jdXNhYmxlKGZhbHNlKSBpcyBjYWxsZWQsIGFuZCB0aGVyZWZvcmUgYXV0b21hdGljIGZvcndhcmQgZm9jdXMKLSAgICAgKiB0cmF2ZXJzYWwgaXMgbmVjZXNzYXJ5Ci0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiB2b2lkIG1vdmVGb2N1cygpIHsgLy8gZG9uJ3QgdXNlIHRyYW5zZmVyRm9jdXMoKSwgYnV0IHF1ZXJ5IGZvY3VzCi0gICAgICogdHJhdmVyc2FsIHBvbGljeSBkaXJlY3RseSAvLyBhbmQgaWYgaXQgcmV0dXJucyBudWxsLCB0cmFuc2ZlciBmb2N1cyB1cAotICAgICAqIGN5Y2xlIC8vIGFuZCBmaW5kIG5leHQgZm9jdXNhYmxlIGNvbXBvbmVudCB0aGVyZSBLZXlib2FyZEZvY3VzTWFuYWdlciBrZm0KLSAgICAgKiA9IEtleWJvYXJkRm9jdXNNYW5hZ2VyLmdldEN1cnJlbnRLZXlib2FyZEZvY3VzTWFuYWdlcigpOyBDb250YWluZXIgcm9vdCA9Ci0gICAgICoga2ZtLmdldEN1cnJlbnRGb2N1c0N5Y2xlUm9vdCgpOyBDb21wb25lbnQgbmV4dENvbXAgPSB0aGlzOyBib29sZWFuCi0gICAgICogc3VjY2VzcyA9ICFpc0ZvY3VzT3duZXIoKTsgd2hpbGUgKCFzdWNjZXNzKSB7IGlmIChyb290ICE9Ci0gICAgICogbmV4dENvbXAuZ2V0Rm9jdXNDeWNsZVJvb3RBbmNlc3RvcigpKSB7IC8vIGNvbXBvbmVudCB3YXMgcHJvYmFibHkgcmVtb3ZlZAotICAgICAqIGZyb20gY29udGFpbmVyIC8vIHNvIGZvY3VzIHdpbGwgYmUgbG9zdCBpbiBzb21lIHRpbWUgcmV0dXJuOyB9IG5leHRDb21wID0KLSAgICAgKiByb290LmdldEZvY3VzVHJhdmVyc2FsUG9saWN5KCkuZ2V0Q29tcG9uZW50QWZ0ZXIocm9vdCwgbmV4dENvbXApOyBpZgotICAgICAqIChuZXh0Q29tcCA9PSB0aGlzKSB7IG5leHRDb21wID0gbnVsbDsgLy8gYXZvaWQgbG9vcGluZyB9IGlmIChuZXh0Q29tcCAhPQotICAgICAqIG51bGwpIHsgc3VjY2VzcyA9IG5leHRDb21wLnJlcXVlc3RGb2N1c0luV2luZG93KCk7IH0gZWxzZSB7IG5leHRDb21wID0KLSAgICAgKiByb290OyByb290ID0gcm9vdC5nZXRGb2N1c0N5Y2xlUm9vdEFuY2VzdG9yKCk7IC8vIGlmIG5vIGFjY2VwdGFibGUKLSAgICAgKiBjb21wb25lbnQgaXMgZm91bmQgYXQgYWxsIC0gY2xlYXIgZ2xvYmFsIC8vIGZvY3VzIG93bmVyIGlmIChyb290ID09IG51bGwpCi0gICAgICogeyBpZiAobmV4dENvbXAgaW5zdGFuY2VvZiBXaW5kb3cpIHsgV2luZG93IHduZCA9IChXaW5kb3cpIG5leHRDb21wOwotICAgICAqIHduZC5zZXRGb2N1c093bmVyKG51bGwpOyB3bmQuc2V0UmVxdWVzdGVkRm9jdXMobnVsbCk7IH0KLSAgICAgKiBrZm0uY2xlYXJHbG9iYWxGb2N1c093bmVyKCk7IHJldHVybjsgfSB9IH0gfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogRm9yIENvbnRhaW5lciB0aGVyZSdzIGEgZGlmZmVyZW5jZSBiZXR3ZWVuIG1vdmluZyBmb2N1cyB3aGVuIGJlaW5nIG1hZGUKLSAgICAgKiBpbnZpc2libGUgb3IgbWFkZSB1bmZvY3VzYWJsZSBpbiBzb21lIG90aGVyIHdheSwgYmVjYXVzZSB3aGVuIGNvbnRhaW5lcgotICAgICAqIGlzIG1hZGUgaW52aXNpYmxlLCBjb21wb25lbnQgc3RpbGwgcmVtYWlucyB2aXNpYmxlLCBpLiBlLiBpdHMgaGlkZSgpIG9yCi0gICAgICogc2V0VmlzaWJsZSgpIGlzIG5vdCBjYWxsZWQuCi0gICAgICovCi0gICAgdm9pZCBtb3ZlRm9jdXNPbkhpZGUoKSB7Ci0gICAgICAgIC8vID8/P0FXVDogbW92ZUZvY3VzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgcHJvcGVydHkgY2hhbmdlIGxpc3RlbmVyIHJlZ2lzdGVyZWQgZm9yIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsaXN0ZW5lcgotICAgICAqICAgICAgICAgICAgdGhlIFByb3BlcnR5Q2hhbmdlTGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcihQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyIGxpc3RlbmVyKSB7Ci0gICAgICAgIGdldFByb3BlcnR5Q2hhbmdlU3VwcG9ydCgpLnJlbW92ZVByb3BlcnR5Q2hhbmdlTGlzdGVuZXIobGlzdGVuZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIHByb3BlcnR5IGNoYW5nZSBsaXN0ZW5lciByZWdpc3RlcmVkIGZvdCB0aGlzIGNvbXBvbmVudCBmb3IKLSAgICAgKiB0aGUgc3BlY2lmaWVkIHByb3BlcnR5eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcGVydHlOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydHkgbmFtZS4KLSAgICAgKiBAcGFyYW0gbGlzdGVuZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZVByb3BlcnR5Q2hhbmdlTGlzdGVuZXIoU3RyaW5nIHByb3BlcnR5TmFtZSwgUHJvcGVydHlDaGFuZ2VMaXN0ZW5lciBsaXN0ZW5lcikgewotICAgICAgICBnZXRQcm9wZXJ0eUNoYW5nZVN1cHBvcnQoKS5yZW1vdmVQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyKHByb3BlcnR5TmFtZSwgbGlzdGVuZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcGFpbnRzIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ2xlIG9mIHRoaXMgY29tcG9uZW50IHdpdGhpbiB0bQotICAgICAqIG1pbGxpc2Vjb25kcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdG0KLSAgICAgKiAgICAgICAgICAgIHRoZSB0aW1lIGluIG1pbGxpc2Vjb25kcyBiZWZvcmUgdXBkYXRpbmcuCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgUmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIFJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiBSZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBSZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVwYWludChsb25nIHRtLCBpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICAvLyA/Pz9BV1QKLSAgICAgICAgLyoKLSAgICAgICAgICogdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7IGlmICh3aWR0aCA8PSAwIHx8IGhlaWdodCA8PSAwIHx8Ci0gICAgICAgICAqIChyZWRyYXdNYW5hZ2VyID09IG51bGwpIHx8ICFpc1Nob3dpbmcoKSkgeyByZXR1cm47IH0gaWYgKGJlaGF2aW91cgotICAgICAgICAgKiBpbnN0YW5jZW9mIExXQmVoYXZpb3IpIHsgaWYgKHBhcmVudCA9PSBudWxsIHx8ICFwYXJlbnQudmlzaWJsZSB8fAotICAgICAgICAgKiAhcGFyZW50LmJlaGF2aW91ci5pc0Rpc3BsYXlhYmxlKCkpIHsgcmV0dXJuOyB9IGlmIChyZXBhaW50UmVnaW9uID09Ci0gICAgICAgICAqIG51bGwpIHsgcmVwYWludFJlZ2lvbiA9IG5ldyBNdWx0aVJlY3RBcmVhKG5ldyBSZWN0YW5nbGUoeCwgeSwgd2lkdGgsCi0gICAgICAgICAqIGhlaWdodCkpOyB9IHJlcGFpbnRSZWdpb24uaW50ZXJzZWN0KG5ldyBSZWN0YW5nbGUoMCwgMCwgdGhpcy53LAotICAgICAgICAgKiB0aGlzLmgpKTsgcmVwYWludFJlZ2lvbi50cmFuc2xhdGUodGhpcy54LCB0aGlzLnkpOwotICAgICAgICAgKiBwYXJlbnQucmVwYWludFJlZ2lvbiA9IHJlcGFpbnRSZWdpb247IHJlcGFpbnRSZWdpb24gPSBudWxsOwotICAgICAgICAgKiBwYXJlbnQucmVwYWludCh0bSwgeCArIHRoaXMueCwgeSArIHRoaXMueSwgd2lkdGgsIGhlaWdodCk7IH0gZWxzZSB7Ci0gICAgICAgICAqIGlmIChyZXBhaW50UmVnaW9uICE9IG51bGwpIHsgcmVkcmF3TWFuYWdlci5hZGRVcGRhdGVSZWdpb24odGhpcywKLSAgICAgICAgICogcmVwYWludFJlZ2lvbik7IHJlcGFpbnRSZWdpb24gPSBudWxsOyB9IGVsc2UgewotICAgICAgICAgKiByZWRyYXdNYW5hZ2VyLmFkZFVwZGF0ZVJlZ2lvbih0aGlzLCBuZXcgUmVjdGFuZ2xlKHgsIHksIHdpZHRoLAotICAgICAgICAgKiBoZWlnaHQpKTsgfQotICAgICAgICAgKiB0b29sa2l0LmdldFN5c3RlbUV2ZW50UXVldWVDb3JlKCkubm90aWZ5RXZlbnRNb25pdG9yKHRvb2xraXQpOyB9IH0KLSAgICAgICAgICogZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0KLSAgICAgICAgICovCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogUG9zdCBldmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIGUuCi0gICAgICovCi0gICAgdm9pZCBwb3N0RXZlbnQoQVdURXZlbnQgZSkgewotICAgICAgICBnZXRUb29sa2l0KCkuZ2V0U3lzdGVtRXZlbnRRdWV1ZUltcGwoKS5wb3N0RXZlbnQoZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwYWludHMgdGhlIHNwZWNpZmllZCBSZWN0YW5nbGUgb2YgdGhpcyBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgUmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIFJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiBSZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBSZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVwYWludChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJlcGFpbnQoMCwgeCwgeSwgd2lkdGgsIGhlaWdodCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwYWludHMgdGhpcyBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVwYWludCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpZiAodyA+IDAgJiYgaCA+IDApIHsKLSAgICAgICAgICAgICAgICByZXBhaW50KDAsIDAsIDAsIHcsIGgpOwotICAgICAgICAgICAgfQotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcGFpbnRzIHRoZSBjb21wb25lbnQgd2l0aGluIHRtIG1pbGxpc2Vjb25kcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdG0KLSAgICAgKiAgICAgICAgICAgIHRoZSB0aW1lIGluIG1pbGxpc2Vjb25kcyBiZWZvcmUgdXBkYXRpbmcuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVwYWludChsb25nIHRtKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmVwYWludCh0bSwgMCwgMCwgdywgaCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVxdWVzdHMgdGhhdCB0aGlzIENvbXBvbmVudCBnZXQgdGhlIGlucHV0IGZvY3VzIHRlbXBvcmFyaWx5LiBUaGlzCi0gICAgICogY29tcG9uZW50IG11c3QgYmUgZGlzcGxheWFibGUsIHZpc2libGUsIGFuZCBmb2N1c2FibGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRlbXBvcmFyeQotICAgICAqICAgICAgICAgICAgdGhpcyBwYXJhbWV0ZXIgaXMgdHJ1ZSBpZiB0aGUgZm9jdXMgY2hhbmdlIGlzIHRlbXBvcmFyeSwgd2hlbgotICAgICAqICAgICAgICAgICAgdGhlIHdpbmRvdyBsb3NlcyB0aGUgZm9jdXMuCi0gICAgICogQHJldHVybiB0cnVlIGlmIHRoZSBmb2N1cyBjaGFuZ2UgcmVxdWVzdCBpcyBzdWNjZWVkZWQsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgYm9vbGVhbiByZXF1ZXN0Rm9jdXMoYm9vbGVhbiB0ZW1wb3JhcnkpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IHJldHVybiByZXF1ZXN0Rm9jdXNJbXBsKHRlbXBvcmFyeSwgdHJ1ZSwgZmFsc2UpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgICAgICAvLyA/Pz9BV1QKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcXVlc3RzIHRoYXQgdGhpcyBDb21wb25lbnQgZ2V0IHRoZSBpbnB1dCBmb2N1cy4gVGhpcyBjb21wb25lbnQgbXVzdCBiZQotICAgICAqIGRpc3BsYXlhYmxlLCB2aXNpYmxlLCBhbmQgZm9jdXNhYmxlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlcXVlc3RGb2N1cygpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXF1ZXN0Rm9jdXMoZmFsc2UpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogcHJvdGVjdGVkIGJvb2xlYW4gcmVxdWVzdEZvY3VzSW5XaW5kb3coYm9vbGVhbiB0ZW1wb3JhcnkpIHsKLSAgICAgKiB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgV2luZG93IHduZCA9IGdldFdpbmRvd0FuY2VzdG9yKCk7IGlmICgod25kID09Ci0gICAgICogbnVsbCkgfHwgIXduZC5pc0ZvY3VzZWQoKSkgeyByZXR1cm4gZmFsc2U7IH0gcmV0dXJuCi0gICAgICogcmVxdWVzdEZvY3VzSW1wbCh0ZW1wb3JhcnksIGZhbHNlLCBmYWxzZSk7IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IGJvb2xlYW4gcmVxdWVzdEZvY3VzSW1wbChib29sZWFuIHRlbXBvcmFyeSwKLSAgICAgKiBib29sZWFuIGNyb3NzV2luZG93LCBib29sZWFuIHJlamVjdGlvblJlY292ZXJ5KSB7IGlmICghcmVqZWN0aW9uUmVjb3ZlcnkKLSAgICAgKiAmJiBpc0ZvY3VzT3duZXIoKSkgeyByZXR1cm4gdHJ1ZTsgfSBXaW5kb3cgd25kID0gZ2V0V2luZG93QW5jZXN0b3IoKTsKLSAgICAgKiBDb250YWluZXIgcGFyID0gZ2V0UmVhbFBhcmVudCgpOyBpZiAoKHBhciAhPSBudWxsKSAmJiBwYXIuaXNSZW1vdmVkKSB7Ci0gICAgICogcmV0dXJuIGZhbHNlOyB9IGlmICghaXNTaG93aW5nKCkgfHwgIWlzRm9jdXNhYmxlKCkgfHwKLSAgICAgKiAhd25kLmlzRm9jdXNhYmxlV2luZG93KCkpIHsgcmV0dXJuIGZhbHNlOyB9IHJldHVybgotICAgICAqIEtleWJvYXJkRm9jdXNNYW5hZ2VyLmdldEN1cnJlbnRLZXlib2FyZEZvY3VzTWFuYWdlcigpLnJlcXVlc3RGb2N1cyh0aGlzLAotICAgICAqIHRlbXBvcmFyeSwgY3Jvc3NXaW5kb3csIHRydWUpOyB9IHB1YmxpYyBib29sZWFuIHJlcXVlc3RGb2N1c0luV2luZG93KCkgewotICAgICAqIHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyByZXR1cm4gcmVxdWVzdEZvY3VzSW5XaW5kb3coZmFsc2UpOyB9IGZpbmFsbHkgewotICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogRGVwcmVjYXRlZDogcmVwbGFjZWQgYnkgc2V0Qm91bmRzKGludCwgaW50LCBpbnQsIGludCkgbWV0aG9kLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGguCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQuCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgc2V0Qm91bmRzKGludCwgaW50LCBpbnQsIGludCkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIHZvaWQgcmVzaGFwZShpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHNldEJvdW5kcyh4LCB5LCB3LCBoLCBib3VuZHNNYXNrUGFyYW0sIHRydWUpOwotICAgICAgICAgICAgYm91bmRzTWFza1BhcmFtID0gMDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHJlY3RhbmdsZSBmb3IgdGhpcyBDb21wb25lbnQgdG8gYmUgdGhlIHJlY3RhbmdsZSB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiB4LHkgY29vcmRpbmF0ZXMgb2YgdGhlIHRvcC1sZWZ0IGNvcm5lciBhbmQgdGhlIHdpZHRoIGFuZCBoZWlnaHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHRvcC1sZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdG9wLWxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRCb3VuZHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXNoYXBlKHgsIHksIHcsIGgpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgcmVjdGFuZ2xlIGZvciB0aGlzIENvbXBvbmVudCB0byBiZSB0aGUgcmVjdGFuZ2xlIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIHgseSBjb29yZGluYXRlcyBvZiB0aGUgdG9wLWxlZnQgY29ybmVyIGFuZCB0aGUgd2lkdGggYW5kIGhlaWdodCBhbmQgcG9zdHMKLSAgICAgKiB0aGUgYXBwcm9wcmlhdGUgZXZlbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB0b3AtbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHRvcC1sZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGJNYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYml0bWFzayBvZiBib3VuZHMgb3B0aW9ucy4KLSAgICAgKiBAcGFyYW0gdXBkYXRlQmVoYXZpb3IKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aGV0aGVyIHRvIHVwZGF0ZSB0aGUgYmVoYXZvaXIncyBib3VuZHMgYXMgd2VsbC4KLSAgICAgKi8KLSAgICB2b2lkIHNldEJvdW5kcyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGJNYXNrLCBib29sZWFuIHVwZGF0ZUJlaGF2aW9yKSB7Ci0gICAgICAgIGludCBvbGRYID0gdGhpcy54OwotICAgICAgICBpbnQgb2xkWSA9IHRoaXMueTsKLSAgICAgICAgaW50IG9sZFcgPSB0aGlzLnc7Ci0gICAgICAgIGludCBvbGRIID0gdGhpcy5oOwotICAgICAgICBzZXRCb3VuZHNGaWVsZHMoeCwgeSwgdywgaCwgYk1hc2spOwotICAgICAgICAvLyBNb3ZlZAotICAgICAgICBpZiAoKG9sZFggIT0gdGhpcy54KSB8fCAob2xkWSAhPSB0aGlzLnkpKSB7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGludmFsaWRhdGVSZWFsUGFyZW50KCk7Ci0gICAgICAgICAgICBwb3N0RXZlbnQobmV3IENvbXBvbmVudEV2ZW50KHRoaXMsIENvbXBvbmVudEV2ZW50LkNPTVBPTkVOVF9NT1ZFRCkpOwotICAgICAgICAgICAgc3ByZWFkSGllcmFyY2h5Qm91bmRzRXZlbnRzKHRoaXMsIEhpZXJhcmNoeUV2ZW50LkFOQ0VTVE9SX01PVkVEKTsKLSAgICAgICAgfQotICAgICAgICAvLyBSZXNpemVkCi0gICAgICAgIGlmICgob2xkVyAhPSB0aGlzLncpIHx8IChvbGRIICE9IHRoaXMuaCkpIHsKLSAgICAgICAgICAgIGludmFsaWRhdGUoKTsKLSAgICAgICAgICAgIHBvc3RFdmVudChuZXcgQ29tcG9uZW50RXZlbnQodGhpcywgQ29tcG9uZW50RXZlbnQuQ09NUE9ORU5UX1JFU0laRUQpKTsKLSAgICAgICAgICAgIHNwcmVhZEhpZXJhcmNoeUJvdW5kc0V2ZW50cyh0aGlzLCBIaWVyYXJjaHlFdmVudC5BTkNFU1RPUl9SRVNJWkVEKTsKLSAgICAgICAgfQotICAgICAgICBpZiAodXBkYXRlQmVoYXZpb3IpIHsKLSAgICAgICAgICAgIGJlaGF2aW91ci5zZXRCb3VuZHModGhpcy54LCB0aGlzLnksIHRoaXMudywgdGhpcy5oLCBiTWFzayk7Ci0gICAgICAgIH0KLSAgICAgICAgbm90aWZ5SW5wdXRNZXRob2QobmV3IFJlY3RhbmdsZSh4LCB5LCB3LCBoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2FsbHMgSW5wdXRDb250ZXh0SW1wbC5ub3RpZnlDbGllbnRXaW5kb3dDaGFuZ2VkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBib3VuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBib3VuZHMuCi0gICAgICovCi0gICAgdm9pZCBub3RpZnlJbnB1dE1ldGhvZChSZWN0YW5nbGUgYm91bmRzKSB7Ci0gICAgICAgIC8vIG9ubHkgV2luZG93IGFjdHVhbGx5IG5vdGlmaWVzIElNIG9mIGJvdW5kcyBjaGFuZ2UKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBib3VuZHMgZmllbGRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3LgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaC4KLSAgICAgKiBAcGFyYW0gYk1hc2sKLSAgICAgKiAgICAgICAgICAgIHRoZSBiIG1hc2suCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHNldEJvdW5kc0ZpZWxkcyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGJNYXNrKSB7Ci0gICAgICAgIGlmICgoYk1hc2sgJiBOYXRpdmVXaW5kb3cuQk9VTkRTX05PU0laRSkgPT0gMCkgewotICAgICAgICAgICAgdGhpcy53ID0gdzsKLSAgICAgICAgICAgIHRoaXMuaCA9IGg7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChiTWFzayAmIE5hdGl2ZVdpbmRvdy5CT1VORFNfTk9NT1ZFKSA9PSAwKSB7Ci0gICAgICAgICAgICB0aGlzLnggPSB4OwotICAgICAgICAgICAgdGhpcy55ID0geTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG5hdGl2ZSBpbnNldHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbmF0aXZlIGluc2V0cy4KLSAgICAgKi8KLSAgICBJbnNldHMgZ2V0TmF0aXZlSW5zZXRzKCkgewotICAgICAgICByZXR1cm4gbmV3IEluc2V0cygwLCAwLCAwLCAwKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBpbnNldHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaW5zZXRzLgotICAgICAqLwotICAgIEluc2V0cyBnZXRJbnNldHMoKSB7Ci0gICAgICAgIHJldHVybiBuZXcgSW5zZXRzKDAsIDAsIDAsIDApOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiBpcyBtb3VzZSBleGl0ZWQgZXhwZWN0ZWQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBpcyBtb3VzZSBleGl0ZWQgZXhwZWN0ZWQuCi0gICAgICovCi0gICAgYm9vbGVhbiBpc01vdXNlRXhpdGVkRXhwZWN0ZWQoKSB7Ci0gICAgICAgIHJldHVybiBtb3VzZUV4aXRlZEV4cGVjdGVkOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIG1vdXNlIGV4aXRlZCBleHBlY3RlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXhwZWN0ZWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgbW91c2UgZXhpdGVkIGV4cGVjdGVkLgotICAgICAqLwotICAgIHZvaWQgc2V0TW91c2VFeGl0ZWRFeHBlY3RlZChib29sZWFuIGV4cGVjdGVkKSB7Ci0gICAgICAgIG1vdXNlRXhpdGVkRXhwZWN0ZWQgPSBleHBlY3RlZDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBuZXcgYm91bmRpbmcgcmVjdGFuZ2xlIGZvciB0aGlzIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcgotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Qm91bmRzKFJlY3RhbmdsZSByKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgc2V0Qm91bmRzKHIueCwgci55LCByLndpZHRoLCByLmhlaWdodCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgY29tcG9uZW50IG9yaWVudGF0aW9uIHdoaWNoIGFmZmVjdHMgdGhlIGNvbXBvbmVudCdzIGVsZW1lbnRzIGFuZAotICAgICAqIHRleHQgd2l0aGluIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvCi0gICAgICogICAgICAgICAgICB0aGUgQ29tcG9uZW50T3JpZW50YXRpb24gb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldENvbXBvbmVudE9yaWVudGF0aW9uKENvbXBvbmVudE9yaWVudGF0aW9uIG8pIHsKLSAgICAgICAgQ29tcG9uZW50T3JpZW50YXRpb24gb2xkT3JpZW50YXRpb247Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgb2xkT3JpZW50YXRpb24gPSBvcmllbnRhdGlvbjsKLSAgICAgICAgICAgIG9yaWVudGF0aW9uID0gbzsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICAgICAgZmlyZVByb3BlcnR5Q2hhbmdlKCJjb21wb25lbnRPcmllbnRhdGlvbiIsIG9sZE9yaWVudGF0aW9uLCBvcmllbnRhdGlvbik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgaW52YWxpZGF0ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHNwZWNpZmllZCBjdXJzb3IgZm9yIHRoaXMgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjdXJzb3IKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgQ3Vyc29yLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEN1cnNvcihDdXJzb3IgY3Vyc29yKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgdGhpcy5jdXJzb3IgPSBjdXJzb3I7Ci0gICAgICAgICAgICBzZXRDdXJzb3IoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXQgY3VycmVudCBjdXJzb3Igc2hhcGUgdG8gQ29tcG9uZW50J3MgQ3Vyc29yLgotICAgICAqLwotICAgIHZvaWQgc2V0Q3Vyc29yKCkgewotICAgICAgICBpZiAoaXNEaXNwbGF5YWJsZSgpICYmIGlzU2hvd2luZygpKSB7Ci0gICAgICAgICAgICBSZWN0YW5nbGUgYWJzUmVjdCA9IG5ldyBSZWN0YW5nbGUoZ2V0TG9jYXRpb25PblNjcmVlbigpLCBnZXRTaXplKCkpOwotICAgICAgICAgICAgUG9pbnQgYWJzUG9pbnRlclBvcyA9IHRvb2xraXQuZGlzcGF0Y2hlci5tb3VzZURpc3BhdGNoZXIuZ2V0UG9pbnRlclBvcygpOwotICAgICAgICAgICAgLy8gPz8/QVdUCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgICogaWYgKGFic1JlY3QuY29udGFpbnMoYWJzUG9pbnRlclBvcykpIHsgLy8gc2V0IEN1cnNvciBvbmx5IG9uCi0gICAgICAgICAgICAgKiB0b3AtbGV2ZWwgV2luZG93cyhvbiBYMTEpIFdpbmRvdyB0b3BMZXZlbFduZCA9Ci0gICAgICAgICAgICAgKiBnZXRXaW5kb3dBbmNlc3RvcigpOyBpZiAodG9wTGV2ZWxXbmQgIT0gbnVsbCkgeyBQb2ludCBwb2ludGVyUG9zCi0gICAgICAgICAgICAgKiA9IE1vdXNlRGlzcGF0Y2hlci5jb252ZXJ0UG9pbnQobnVsbCwgYWJzUG9pbnRlclBvcywgdG9wTGV2ZWxXbmQpOwotICAgICAgICAgICAgICogQ29tcG9uZW50IGNvbXBVbmRlckN1cnNvciA9Ci0gICAgICAgICAgICAgKiB0b3BMZXZlbFduZC5maW5kQ29tcG9uZW50QXQocG9pbnRlclBvcyk7IC8vIGlmIChjb21wVW5kZXJDdXJzb3IKLSAgICAgICAgICAgICAqID09IHRoaXMgfHwgLy8gY29tcFVuZGVyQ3Vyc29yLmdldEN1cnNvckFuY2VzdG9yKCkgPT0gdGhpcykgewotICAgICAgICAgICAgICogTmF0aXZlV2luZG93IHduZCA9IHRvcExldmVsV25kLmdldE5hdGl2ZVdpbmRvdygpOyBpZgotICAgICAgICAgICAgICogKGNvbXBVbmRlckN1cnNvciAhPSBudWxsICYmIHduZCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgKiBjb21wVW5kZXJDdXJzb3IuZ2V0UmVhbEN1cnNvcigpLmdldE5hdGl2ZUN1cnNvcigpCi0gICAgICAgICAgICAgKiAuc2V0Q3Vyc29yKHduZC5nZXRJZCgpKTsgfSAvLyB9IH0gfQotICAgICAgICAgICAgICovCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhbmNlc3RvciBDdXJzb3IgaWYgQ29tcG9uZW50IGlzIGRpc2FibGVkIChkaXJlY3RseSBvciB2aWEgYW4KLSAgICAgKiBhbmNlc3RvcikgZXZlbiBpZiBDdXJzb3IgaXMgZXhwbGljaXRseSBzZXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgdmFsdWUuCi0gICAgICogQHJldHVybiB0aGUgYWN0dWFsIEN1cnNvciB0byBiZSBkaXNwbGF5ZWQuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBDdXJzb3IgZ2V0UmVhbEN1cnNvcigpIHsgQ29tcG9uZW50IGN1cnNvckFuY2VzdG9yID0gZ2V0Q3Vyc29yQW5jZXN0b3IoKTsKLSAgICAgKiByZXR1cm4gY3Vyc29yQW5jZXN0b3IgIT0gbnVsbCA/IGN1cnNvckFuY2VzdG9yLmdldEN1cnNvcigpIDoKLSAgICAgKiBDdXJzb3IuZ2V0RGVmYXVsdEN1cnNvcigpOyB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhbmNlc3RvcihvciBjb21wb25lbnQgaXRzZWxmKSB3aG9zZSBjdXJzb3IgaXMgc2V0IHdoZW4gcG9pbnRlcgotICAgICAqIGlzIGluc2lkZSBjb21wb25lbnQKLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhY3R1YWwgQ3Vyc29yIHRvIGJlIGRpc3BsYXllZC4KLSAgICAgKi8KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIENvbXBvbmVudCBnZXRDdXJzb3JBbmNlc3RvcigpIHsgQ29tcG9uZW50IGNvbXA7IGZvciAoY29tcCA9IHRoaXM7IGNvbXAgIT0KLSAgICAgKiBudWxsOyBjb21wID0gY29tcC5nZXRQYXJlbnQoKSkgeyBpZiAoY29tcCBpbnN0YW5jZW9mIFdpbmRvdyB8fAotICAgICAqIGNvbXAuaXNDdXJzb3JTZXQoKSAmJiBjb21wLmlzS2V5RW5hYmxlZCgpKSB7IHJldHVybiBjb21wOyB9IH0gcmV0dXJuCi0gICAgICogbnVsbDsgfSBwdWJsaWMgdm9pZCBzZXREcm9wVGFyZ2V0KERyb3BUYXJnZXQgZHQpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeQotICAgICAqIHsgaWYgKGRyb3BUYXJnZXQgPT0gZHQpIHsgcmV0dXJuOyB9IERyb3BUYXJnZXQgb2xkRHJvcFRhcmdldCA9Ci0gICAgICogZHJvcFRhcmdldDsgZHJvcFRhcmdldCA9IGR0OyBpZiAob2xkRHJvcFRhcmdldCAhPSBudWxsKSB7IGlmCi0gICAgICogKGJlaGF2aW91ci5pc0Rpc3BsYXlhYmxlKCkpIHsgb2xkRHJvcFRhcmdldC5yZW1vdmVOb3RpZnkocGVlcik7IH0KLSAgICAgKiBvbGREcm9wVGFyZ2V0LnNldENvbXBvbmVudChudWxsKTsgfSBpZiAoZHQgIT0gbnVsbCkgewotICAgICAqIGR0LnNldENvbXBvbmVudCh0aGlzKTsgaWYgKGJlaGF2aW91ci5pc0Rpc3BsYXlhYmxlKCkpIHsKLSAgICAgKiBkdC5hZGROb3RpZnkocGVlcik7IH0gfSB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKi8KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhpcyBjb21wb25lbnQgdG8gdGhlICJlbmFibGVkIiBvciAiZGlzYWJsZWQiIHN0YXRlIGRlcGVuZGluZyBvbiB0aGUKLSAgICAgKiBzcGVjaWZpZWQgYm9vbGVhbiBwYXJhbWV0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHZhbHVlCi0gICAgICogICAgICAgICAgICB0cnVlIGlmIHRoaXMgY29tcG9uZW50IHNob3VsZCBiZSBlbmFibGVkOyBmYWxzZSBpZiB0aGlzCi0gICAgICogICAgICAgICAgICBjb21wb25lbnQgc2hvdWxkIGJlIGRpc2FibGVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEVuYWJsZWQoYm9vbGVhbiB2YWx1ZSkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGVuYWJsZSh2YWx1ZSk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZW5hYmxlZCBpbXBsLgotICAgICAqIAotICAgICAqIEBwYXJhbSB2YWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBlbmFibGVkIGltcGwuCi0gICAgICovCi0gICAgdm9pZCBzZXRFbmFibGVkSW1wbChib29sZWFuIHZhbHVlKSB7Ci0gICAgICAgIGlmIChlbmFibGVkICE9IHZhbHVlKSB7Ci0gICAgICAgICAgICBlbmFibGVkID0gdmFsdWU7Ci0gICAgICAgICAgICBzZXRDdXJzb3IoKTsKLSAgICAgICAgICAgIGlmICghZW5hYmxlZCkgewotICAgICAgICAgICAgICAgIG1vdmVGb2N1c09uSGlkZSgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgYmVoYXZpb3VyLnNldEVuYWJsZWQodmFsdWUpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBwcml2YXRlIHZvaWQgZmlyZUFjY2Vzc2libGVTdGF0ZUNoYW5nZShBY2Nlc3NpYmxlU3RhdGUgc3RhdGUsIGJvb2xlYW4KLSAgICAgKiB2YWx1ZSkgeyBpZiAoYmVoYXZpb3VyLmlzTGlnaHR3ZWlnaHQoKSkgeyByZXR1cm47IH0gQWNjZXNzaWJsZUNvbnRleHQgYWMKLSAgICAgKiA9IGdldEFjY2Vzc2libGVDb250ZXh0KCk7IGlmIChhYyAhPSBudWxsKSB7IEFjY2Vzc2libGVTdGF0ZSBvbGRWYWx1ZSA9Ci0gICAgICogbnVsbDsgQWNjZXNzaWJsZVN0YXRlIG5ld1ZhbHVlID0gbnVsbDsgaWYgKHZhbHVlKSB7IG5ld1ZhbHVlID0gc3RhdGU7IH0KLSAgICAgKiBlbHNlIHsgb2xkVmFsdWUgPSBzdGF0ZTsgfQotICAgICAqIGFjLmZpcmVQcm9wZXJ0eUNoYW5nZShBY2Nlc3NpYmxlQ29udGV4dC5BQ0NFU1NJQkxFX1NUQVRFX1BST1BFUlRZLAotICAgICAqIG9sZFZhbHVlLCBuZXdWYWx1ZSk7IH0gfQotICAgICAqLwotCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBwdWJsaWMgdm9pZCBzZXRGb2N1c1RyYXZlcnNhbEtleXMoaW50IGlkLCBTZXQ8PyBleHRlbmRzIEFXVEtleVN0cm9rZT4KLSAgICAgKiBrZXlzdHJva2VzKSB7IFNldDw/IGV4dGVuZHMgQVdUS2V5U3Ryb2tlPiBvbGRUcmF2ZXJzYWxLZXlzOyBTdHJpbmcKLSAgICAgKiBwcm9wTmFtZSA9ICJGb2N1c1RyYXZlcnNhbEtleXMiOyAvLyROT04tTkxTLTEkIHRvb2xraXQubG9ja0FXVCgpOyB0cnkgewotICAgICAqIEludGVnZXIga0lkID0gbmV3IEludGVnZXIoaWQpOwotICAgICAqIEtleWJvYXJkRm9jdXNNYW5hZ2VyLmNoZWNrVHJhdmVyc2FsS2V5c0lEKHRyYXZlcnNhbEtleXMsIGtJZCk7Ci0gICAgICogTWFwPEludGVnZXIsIFNldDw/IGV4dGVuZHMgQVdUS2V5U3Ryb2tlPj4ga2V5cyA9IG5ldyBIYXNoTWFwPEludGVnZXIsCi0gICAgICogU2V0PD8gZXh0ZW5kcyBBV1RLZXlTdHJva2U+PigpOyBmb3IgKGludCBraWQgOiB0cmF2ZXJzYWxJRHMpIHsgSW50ZWdlcgotICAgICAqIGtleSA9IG5ldyBJbnRlZ2VyKGtpZCk7IGtleXMucHV0KGtleSwgZ2V0Rm9jdXNUcmF2ZXJzYWxLZXlzKGtpZCkpOyB9Ci0gICAgICogS2V5Ym9hcmRGb2N1c01hbmFnZXIuY2hlY2tLZXlTdHJva2VzKHRyYXZlcnNhbElEcywga2V5cywga0lkLAotICAgICAqIGtleXN0cm9rZXMpOyBvbGRUcmF2ZXJzYWxLZXlzID0gdHJhdmVyc2FsS2V5cy5nZXQobmV3IEludGVnZXIoaWQpKTsgLy8KLSAgICAgKiBwdXQgYSBjb3B5IG9mIGtleXN0cm9rZXMgb2JqZWN0IGludG8gbWFwOiBTZXQ8PyBleHRlbmRzIEFXVEtleVN0cm9rZT4KLSAgICAgKiBuZXdLZXlzID0ga2V5c3Ryb2tlczsgaWYgKGtleXN0cm9rZXMgIT0gbnVsbCkgeyBuZXdLZXlzID0gbmV3Ci0gICAgICogSGFzaFNldDxBV1RLZXlTdHJva2U+KGtleXN0cm9rZXMpOyB9IHRyYXZlcnNhbEtleXMucHV0KGtJZCwgbmV3S2V5cyk7Ci0gICAgICogU3RyaW5nIGRpcmVjdGlvbiA9ICIiOyAvLyROT04tTkxTLTEkIHN3aXRjaCAoaWQpIHsgY2FzZQotICAgICAqIEtleWJvYXJkRm9jdXNNYW5hZ2VyLkZPUldBUkRfVFJBVkVSU0FMX0tFWVM6IGRpcmVjdGlvbiA9ICJmb3J3YXJkIjsKLSAgICAgKiAvLyROT04tTkxTLTEkIGJyZWFrOyBjYXNlIEtleWJvYXJkRm9jdXNNYW5hZ2VyLkJBQ0tXQVJEX1RSQVZFUlNBTF9LRVlTOgotICAgICAqIGRpcmVjdGlvbiA9ICJiYWNrd2FyZCI7IC8vJE5PTi1OTFMtMSQgYnJlYWs7IGNhc2UKLSAgICAgKiBLZXlib2FyZEZvY3VzTWFuYWdlci5VUF9DWUNMRV9UUkFWRVJTQUxfS0VZUzogZGlyZWN0aW9uID0gInVwQ3ljbGUiOwotICAgICAqIC8vJE5PTi1OTFMtMSQgYnJlYWs7IGNhc2UgS2V5Ym9hcmRGb2N1c01hbmFnZXIuRE9XTl9DWUNMRV9UUkFWRVJTQUxfS0VZUzoKLSAgICAgKiBkaXJlY3Rpb24gPSAiZG93bkN5Y2xlIjsgLy8kTk9OLU5MUy0xJCBicmVhazsgfSBwcm9wTmFtZSA9IGRpcmVjdGlvbiArCi0gICAgICogcHJvcE5hbWU7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0gZmlyZVByb3BlcnR5Q2hhbmdlKHByb3BOYW1lLAotICAgICAqIG9sZFRyYXZlcnNhbEtleXMsIGtleXN0cm9rZXMpOyB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBmb2N1cyB0cmF2ZXJzYWwga2V5cyBzdGF0ZSBmb3IgdGhpcyBjb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHZhbHVlCi0gICAgICogICAgICAgICAgICB0cnVlIGlmIHRoZSBmb2N1cyB0cmF2ZXJzYWwga2V5cyBzdGF0ZSBpcyBlbmFibGVkLCBmYWxzZSBpZgotICAgICAqICAgICAgICAgICAgdGhlIGZvY3VzIHRyYXZlcnNhbCBrZXlzIHN0YXRlIGlzIGRpc2FibGVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEZvY3VzVHJhdmVyc2FsS2V5c0VuYWJsZWQoYm9vbGVhbiB2YWx1ZSkgewotICAgICAgICBib29sZWFuIG9sZEZvY3VzVHJhdmVyc2FsS2V5c0VuYWJsZWQ7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgb2xkRm9jdXNUcmF2ZXJzYWxLZXlzRW5hYmxlZCA9IGZvY3VzVHJhdmVyc2FsS2V5c0VuYWJsZWQ7Ci0gICAgICAgICAgICBmb2N1c1RyYXZlcnNhbEtleXNFbmFibGVkID0gdmFsdWU7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgICAgIGZpcmVQcm9wZXJ0eUNoYW5nZSgiZm9jdXNUcmF2ZXJzYWxLZXlzRW5hYmxlZCIsIG9sZEZvY3VzVHJhdmVyc2FsS2V5c0VuYWJsZWQsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICBmb2N1c1RyYXZlcnNhbEtleXNFbmFibGVkKTsKLSAgICB9Ci0KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIHB1YmxpYyB2b2lkIHNldEZvY3VzYWJsZShib29sZWFuIGZvY3VzYWJsZSkgeyBib29sZWFuIG9sZEZvY3VzYWJsZTsKLSAgICAgKiB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgY2FsbGVkU2V0Rm9jdXNhYmxlID0gdHJ1ZTsgb2xkRm9jdXNhYmxlID0KLSAgICAgKiB0aGlzLmZvY3VzYWJsZTsgdGhpcy5mb2N1c2FibGUgPSBmb2N1c2FibGU7IGlmICghZm9jdXNhYmxlKSB7Ci0gICAgICogbW92ZUZvY3VzKCk7IH0gfSBmaW5hbGx5IHsgdG9vbGtpdC51bmxvY2tBV1QoKTsgfQotICAgICAqIGZpcmVQcm9wZXJ0eUNoYW5nZSgiZm9jdXNhYmxlIiwgb2xkRm9jdXNhYmxlLCBmb2N1c2FibGUpOyAvLyROT04tTkxTLTEkIH0KLSAgICAgKiBwdWJsaWMgRm9udCBnZXRGb250KCkgeyB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgcmV0dXJuIChmb250ID09IG51bGwpICYmCi0gICAgICogKHBhcmVudCAhPSBudWxsKSA/IHBhcmVudC5nZXRGb250KCkgOiBmb250OyB9IGZpbmFsbHkgewotICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZm9udCBmb3IgdGhpcyBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGYKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgZm9udCBvZiB0aGUgQ29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEZvbnQoRm9udCBmKSB7Ci0gICAgICAgIEZvbnQgb2xkRm9udDsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBvbGRGb250ID0gZm9udDsKLSAgICAgICAgICAgIHNldEZvbnRJbXBsKGYpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgICAgICBmaXJlUHJvcGVydHlDaGFuZ2UoImZvbnQiLCBvbGRGb250LCBmb250KTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGZvbnQgaW1wbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZgotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBmb250IGltcGwuCi0gICAgICovCi0gICAgdm9pZCBzZXRGb250SW1wbChGb250IGYpIHsKLSAgICAgICAgZm9udCA9IGY7Ci0gICAgICAgIGludmFsaWRhdGUoKTsKLSAgICAgICAgaWYgKGlzU2hvd2luZygpKSB7Ci0gICAgICAgICAgICByZXBhaW50KCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnZhbGlkYXRlIHRoZSBjb21wb25lbnQgaWYgaXQgaW5oZXJpdHMgdGhlIGZvbnQgZnJvbSB0aGUgcGFyZW50LiBUaGlzCi0gICAgICogbWV0aG9kIGlzIG92ZXJyaWRkZW4gaW4gQ29udGFpbmVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgY29tcG9uZW50IHdhcyBpbnZhbGlkYXRlZCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIGJvb2xlYW4gcHJvcGFnYXRlRm9udCgpIHsKLSAgICAgICAgaWYgKGZvbnQgPT0gbnVsbCkgewotICAgICAgICAgICAgaW52YWxpZGF0ZSgpOwotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGZvcmVncm91bmQgY29sb3IgZm9yIHRoaXMgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IGZvcmVncm91bmQgY29sb3IuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Rm9yZWdyb3VuZChDb2xvciBjKSB7Ci0gICAgICAgIENvbG9yIG9sZEZnQ29sb3I7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgb2xkRmdDb2xvciA9IGZvcmVDb2xvcjsKLSAgICAgICAgICAgIGZvcmVDb2xvciA9IGM7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgICAgIGZpcmVQcm9wZXJ0eUNoYW5nZSgiZm9yZWdyb3VuZCIsIG9sZEZnQ29sb3IsIGZvcmVDb2xvcik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgcmVwYWludCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGJhY2tncm91bmQgY29sb3IgZm9yIHRoZSBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGMKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgYmFja2dyb3VuZCBjb2xvciBmb3IgdGhpcyBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0QmFja2dyb3VuZChDb2xvciBjKSB7Ci0gICAgICAgIENvbG9yIG9sZEJrQ29sb3I7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgb2xkQmtDb2xvciA9IGJhY2tDb2xvcjsKLSAgICAgICAgICAgIGJhY2tDb2xvciA9IGM7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgICAgIGZpcmVQcm9wZXJ0eUNoYW5nZSgiYmFja2dyb3VuZCIsIG9sZEJrQ29sb3IsIGJhY2tDb2xvcik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgcmVwYWludCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGZsYWcgZm9yIHdoZXRoZXIgcGFpbnQgbWVzc2FnZXMgcmVjZWl2ZWQgZnJvbSB0aGUgb3BlcmF0aW5nCi0gICAgICogc3lzdGVtIHNob3VsZCBiZSBpZ25vcmVkIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRydWUgaWYgcGFpbnQgbWVzc2FnZXMgcmVjZWl2ZWQgZnJvbSB0aGUgb3BlcmF0aW5nIHN5c3RlbQotICAgICAqICAgICAgICAgICAgc2hvdWxkIGJlIGlnbm9yZWQsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRJZ25vcmVSZXBhaW50KGJvb2xlYW4gdmFsdWUpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpZ25vcmVSZXBhaW50ID0gdmFsdWU7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbG9jYWxlIG9mIHRoZSBjb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxvY2FsZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBMb2NhbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0TG9jYWxlKExvY2FsZSBsb2NhbGUpIHsKLSAgICAgICAgTG9jYWxlIG9sZExvY2FsZTsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBvbGRMb2NhbGUgPSB0aGlzLmxvY2FsZTsKLSAgICAgICAgICAgIHRoaXMubG9jYWxlID0gbG9jYWxlOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgICAgICBmaXJlUHJvcGVydHlDaGFuZ2UoImxvY2FsZSIsIG9sZExvY2FsZSwgbG9jYWxlKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGxvY2F0aW9uIG9mIHRoZSBDb21wb25lbnQgdG8gdGhlIHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBsb2NhdGlvbiBvZiB0aGUgQ29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldExvY2F0aW9uKFBvaW50IHApIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBzZXRMb2NhdGlvbihwLngsIHAueSk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbG9jYXRpb24gb2YgdGhlIENvbXBvbmVudCB0byB0aGUgc3BlY2lmaWVkIHgsIHkgY29vcmRpbmF0ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0TG9jYXRpb24oaW50IHgsIGludCB5KSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgbW92ZSh4LCB5KTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSB2aXNpYmlsaXR5IHN0YXRlIG9mIHRoZSBjb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRydWUgaWYgdGhlIGNvbXBvbmVudCBpcyB2aXNpYmxlLCBmYWxzZSBpZiB0aGUgY29tcG9uZW50IGlzCi0gICAgICogICAgICAgICAgICBub3Qgc2hvd24uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0VmlzaWJsZShib29sZWFuIGIpIHsKLSAgICAgICAgLy8gc2hvdygpICYgaGlkZSgpIGFyZSBub3QgZGVwcmVjYXRlZCBmb3IgV2luZG93LAotICAgICAgICAvLyBzbyBoYXZlIHRvIGNhbGwgdGhlbSBmcm9tIHNldFZpc2libGUoKQotICAgICAgICBzaG93KGIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERlcHJlY2F0ZWQ6IHJlcGxhY2VkIGJ5IHNldFZpc2libGUoYm9vbGVhbikgbWV0aG9kLgotICAgICAqIAotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IHNldFZpc2libGUoYm9vbGVhbikgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIHZvaWQgc2hvdygpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpZiAodmlzaWJsZSkgewotICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHByZXBhcmU0SGllcmFyY2h5Q2hhbmdlKCk7Ci0gICAgICAgICAgICBtYXBUb0Rpc3BsYXkodHJ1ZSk7Ci0gICAgICAgICAgICB2YWxpZGF0ZSgpOwotICAgICAgICAgICAgdmlzaWJsZSA9IHRydWU7Ci0gICAgICAgICAgICBiZWhhdmlvdXIuc2V0VmlzaWJsZSh0cnVlKTsKLSAgICAgICAgICAgIHBvc3RFdmVudChuZXcgQ29tcG9uZW50RXZlbnQodGhpcywgQ29tcG9uZW50RXZlbnQuQ09NUE9ORU5UX1NIT1dOKSk7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGZpbmlzaEhpZXJhcmNoeUNoYW5nZSh0aGlzLCBwYXJlbnQsIDApOwotICAgICAgICAgICAgbm90aWZ5SW5wdXRNZXRob2QobmV3IFJlY3RhbmdsZSh4LCB5LCB3LCBoKSk7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGludmFsaWRhdGVSZWFsUGFyZW50KCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGVwcmVjYXRlZDogcmVwbGFjZWQgYnkgc2V0VmlzaWJsZShib29sZWFuKSBtZXRob2QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSB2aXNpYmlsaXR5J3Mgc3RhdGUuCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgc2V0VmlzaWJsZShib29sZWFuKSBtZXRob2QuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgdm9pZCBzaG93KGJvb2xlYW4gYikgewotICAgICAgICBpZiAoYikgewotICAgICAgICAgICAgc2hvdygpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaGlkZSgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiB2b2lkIHRyYW5zZmVyRm9jdXMoaW50IGRpcikgeyBDb250YWluZXIgcm9vdCA9IG51bGw7IGlmICh0aGlzIGluc3RhbmNlb2YKLSAgICAgKiBDb250YWluZXIpIHsgQ29udGFpbmVyIGNvbnQgPSAoQ29udGFpbmVyKSB0aGlzOyBpZgotICAgICAqIChjb250LmlzRm9jdXNDeWNsZVJvb3QoKSkgeyByb290ID0gY29udC5nZXRGb2N1c1RyYXZlcnNhbFJvb3QoKTsgfSB9IGlmCi0gICAgICogKHJvb3QgPT0gbnVsbCkgeyByb290ID0gZ2V0Rm9jdXNDeWNsZVJvb3RBbmNlc3RvcigpOyB9IC8vIHRyYW5zZmVyIGZvY3VzCi0gICAgICogdXAgY3ljbGUgaWYgcm9vdCBpcyB1bnJlYWNoYWJsZSBDb21wb25lbnQgY29tcCA9IHRoaXM7IHdoaWxlICgocm9vdCAhPQotICAgICAqIG51bGwpICYmICEocm9vdC5pc0ZvY3VzQ3ljbGVSb290KCkgJiYgcm9vdC5pc1Nob3dpbmcoKSAmJgotICAgICAqIHJvb3QuaXNFbmFibGVkKCkgJiYgcm9vdCAuaXNGb2N1c2FibGUoKSkpIHsgY29tcCA9IHJvb3Q7IHJvb3QgPQotICAgICAqIHJvb3QuZ2V0Rm9jdXNDeWNsZVJvb3RBbmNlc3RvcigpOyB9IGlmIChyb290ID09IG51bGwpIHsgcmV0dXJuOyB9Ci0gICAgICogRm9jdXNUcmF2ZXJzYWxQb2xpY3kgcG9saWN5ID0gcm9vdC5nZXRGb2N1c1RyYXZlcnNhbFBvbGljeSgpOyBDb21wb25lbnQKLSAgICAgKiBuZXh0Q29tcCA9IG51bGw7IHN3aXRjaCAoZGlyKSB7IGNhc2UKLSAgICAgKiBLZXlib2FyZEZvY3VzTWFuYWdlci5GT1JXQVJEX1RSQVZFUlNBTF9LRVlTOiBuZXh0Q29tcCA9Ci0gICAgICogcG9saWN5LmdldENvbXBvbmVudEFmdGVyKHJvb3QsIGNvbXApOyBicmVhazsgY2FzZQotICAgICAqIEtleWJvYXJkRm9jdXNNYW5hZ2VyLkJBQ0tXQVJEX1RSQVZFUlNBTF9LRVlTOiBuZXh0Q29tcCA9Ci0gICAgICogcG9saWN5LmdldENvbXBvbmVudEJlZm9yZShyb290LCBjb21wKTsgYnJlYWs7IH0gaWYgKG5leHRDb21wICE9IG51bGwpIHsKLSAgICAgKiBuZXh0Q29tcC5yZXF1ZXN0Rm9jdXMoZmFsc2UpOyB9IH0gcHVibGljIHZvaWQgdHJhbnNmZXJGb2N1cygpIHsKLSAgICAgKiB0b29sa2l0LmxvY2tBV1QoKTsgdHJ5IHsgbmV4dEZvY3VzKCk7IH0gZmluYWxseSB7IHRvb2xraXQudW5sb2NrQVdUKCk7IH0KLSAgICAgKiB9IHB1YmxpYyB2b2lkIHRyYW5zZmVyRm9jdXNCYWNrd2FyZCgpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeSB7Ci0gICAgICogdHJhbnNmZXJGb2N1cyhLZXlib2FyZEZvY3VzTWFuYWdlci5CQUNLV0FSRF9UUkFWRVJTQUxfS0VZUyk7IH0gZmluYWxseSB7Ci0gICAgICogdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9IHB1YmxpYyB2b2lkIHRyYW5zZmVyRm9jdXNVcEN5Y2xlKCkgewotICAgICAqIHRvb2xraXQubG9ja0FXVCgpOyB0cnkgeyBLZXlib2FyZEZvY3VzTWFuYWdlciBrZm0gPQotICAgICAqIEtleWJvYXJkRm9jdXNNYW5hZ2VyLmdldEN1cnJlbnRLZXlib2FyZEZvY3VzTWFuYWdlcigpOyBDb250YWluZXIgcm9vdCA9Ci0gICAgICoga2ZtLmdldEN1cnJlbnRGb2N1c0N5Y2xlUm9vdCgpOyBpZihyb290ID09IG51bGwpIHsgcmV0dXJuOyB9IGJvb2xlYW4KLSAgICAgKiBzdWNjZXNzID0gZmFsc2U7IENvbXBvbmVudCBuZXh0Q29tcCA9IG51bGw7IENvbnRhaW5lciBuZXdSb290ID0gcm9vdDsgZG8KLSAgICAgKiB7IG5leHRDb21wID0gbmV3Um9vdCBpbnN0YW5jZW9mIFdpbmRvdyA/Ci0gICAgICogbmV3Um9vdC5nZXRGb2N1c1RyYXZlcnNhbFBvbGljeSgpIC5nZXREZWZhdWx0Q29tcG9uZW50KG5ld1Jvb3QpIDoKLSAgICAgKiBuZXdSb290OyBuZXdSb290ID0gbmV3Um9vdC5nZXRGb2N1c0N5Y2xlUm9vdEFuY2VzdG9yKCk7IGlmIChuZXh0Q29tcCA9PQotICAgICAqIG51bGwpIHsgYnJlYWs7IH0gc3VjY2VzcyA9IG5leHRDb21wLnJlcXVlc3RGb2N1c0luV2luZG93KCk7IGlmIChuZXdSb290Ci0gICAgICogPT0gbnVsbCkgeyBicmVhazsgfSBrZm0uc2V0R2xvYmFsQ3VycmVudEZvY3VzQ3ljbGVSb290KG5ld1Jvb3QpOyB9IHdoaWxlCi0gICAgICogKCFzdWNjZXNzKTsgaWYgKCFzdWNjZXNzICYmIHJvb3QgIT0gbmV3Um9vdCkgewotICAgICAqIGtmbS5zZXRHbG9iYWxDdXJyZW50Rm9jdXNDeWNsZVJvb3Qocm9vdCk7IH0gfSBmaW5hbGx5IHsKLSAgICAgKiB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKi8KLQotICAgIC8qKgotICAgICAqIFZhbGlkYXRlcyB0aGF0IHRoaXMgY29tcG9uZW50IGhhcyBhIHZhbGlkIGxheW91dC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB2YWxpZGF0ZSgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpZiAoIWJlaGF2aW91ci5pc0Rpc3BsYXlhYmxlKCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB2YWxpZGF0ZUltcGwoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBWYWxpZGF0ZSBpbXBsLgotICAgICAqLwotICAgIHZvaWQgdmFsaWRhdGVJbXBsKCkgewotICAgICAgICB2YWxpZCA9IHRydWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbmF0aXZlIHdpbmRvdy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBuYXRpdmUgd2luZG93LgotICAgICAqLwotICAgIE5hdGl2ZVdpbmRvdyBnZXROYXRpdmVXaW5kb3coKSB7Ci0gICAgICAgIHJldHVybiBiZWhhdmlvdXIuZ2V0TmF0aXZlV2luZG93KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IGEgbWF4aW11bSBzaXplIGlzIHNldCBmb3IgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBtYXhpbXVtIHNpemUgaXMgc2V0IGZvciB0aGUgQ29tcG9uZW50LCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzTWF4aW11bVNpemVTZXQoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIG1heGltdW1TaXplICE9IG51bGw7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IHRoZSBtaW5pbXVtIHNpemUgaXMgc2V0IGZvciB0aGUgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIG1pbmltdW0gc2l6ZSBpcyBzZXQgZm9yIHRoZSBjb21wb25lbnQsIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNNaW5pbXVtU2l6ZVNldCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gbWluaW11bVNpemUgIT0gbnVsbDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIHByZWZlcnJlZCBzaXplIGlzIHNldCBmb3IgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBwcmVmZXJyZWQgc2l6ZSBpcyBzZXQgZm9yIHRoZSBDb21wb25lbnQsIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNQcmVmZXJyZWRTaXplU2V0KCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBwcmVmZXJyZWRTaXplICE9IG51bGw7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWF4aW11bSBzaXplIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWF4aW11bSBzaXplIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIERpbWVuc2lvbiBnZXRNYXhpbXVtU2l6ZSgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gaXNNYXhpbXVtU2l6ZVNldCgpID8gbmV3IERpbWVuc2lvbihtYXhpbXVtU2l6ZSkgOiBuZXcgRGltZW5zaW9uKFNob3J0Lk1BWF9WQUxVRSwKLSAgICAgICAgICAgICAgICAgICAgU2hvcnQuTUFYX1ZBTFVFKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtaW5pbXVtIHNpemUgb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtaW5pbXVtIHNpemUgb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGltZW5zaW9uIGdldE1pbmltdW1TaXplKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBtaW5pbXVtU2l6ZSgpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIERlcHJlY2F0ZWQ6IHJlcGxhY2VkIGJ5IGdldE1pbmltdW1TaXplKCkgbWV0aG9kLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIERpbWVuc2lvbi4KLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSBnZXRNaW5pbXVtU2l6ZSgpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBEaW1lbnNpb24gbWluaW11bVNpemUoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKGlzTWluaW11bVNpemVTZXQoKSkgewotICAgICAgICAgICAgICAgIHJldHVybiAoRGltZW5zaW9uKW1pbmltdW1TaXplLmNsb25lKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBEaW1lbnNpb24gZGVmU2l6ZSA9IGdldERlZmF1bHRNaW5pbXVtU2l6ZSgpOwotICAgICAgICAgICAgaWYgKGRlZlNpemUgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHJldHVybiAoRGltZW5zaW9uKWRlZlNpemUuY2xvbmUoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBpc0Rpc3BsYXlhYmxlKCkgPyBuZXcgRGltZW5zaW9uKDEsIDEpIDogbmV3IERpbWVuc2lvbih3LCBoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBwcmVmZXJyZWQgc2l6ZSBvZiB0aGUgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHByZWZlcnJlZCBzaXplIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIERpbWVuc2lvbiBnZXRQcmVmZXJyZWRTaXplKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBwcmVmZXJyZWRTaXplKCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGVwcmVjYXRlZDogcmVwbGFjZWQgYnkgZ2V0UHJlZmVycmVkU2l6ZSgpIG1ldGhvZC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBEaW1lbnNpb24uCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgYnkgZ2V0UHJlZmVycmVkU2l6ZSgpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBEaW1lbnNpb24gcHJlZmVycmVkU2l6ZSgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpZiAoaXNQcmVmZXJyZWRTaXplU2V0KCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbmV3IERpbWVuc2lvbihwcmVmZXJyZWRTaXplKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIERpbWVuc2lvbiBkZWZTaXplID0gZ2V0RGVmYXVsdFByZWZlcnJlZFNpemUoKTsKLSAgICAgICAgICAgIGlmIChkZWZTaXplICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbmV3IERpbWVuc2lvbihkZWZTaXplKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBuZXcgRGltZW5zaW9uKGdldE1pbmltdW1TaXplKCkpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIG1heGltdW0gc2l6ZSBvZiB0aGUgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBtYXhpbXVtU2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBtYXhpbXVtIHNpemUgb2YgdGhlIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRNYXhpbXVtU2l6ZShEaW1lbnNpb24gbWF4aW11bVNpemUpIHsKLSAgICAgICAgRGltZW5zaW9uIG9sZE1heGltdW1TaXplOwotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIG9sZE1heGltdW1TaXplID0gdGhpcy5tYXhpbXVtU2l6ZTsKLSAgICAgICAgICAgIGlmIChvbGRNYXhpbXVtU2l6ZSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgb2xkTWF4aW11bVNpemUgPSBvbGRNYXhpbXVtU2l6ZS5nZXRTaXplKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAodGhpcy5tYXhpbXVtU2l6ZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaWYgKG1heGltdW1TaXplICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgdGhpcy5tYXhpbXVtU2l6ZSA9IG5ldyBEaW1lbnNpb24obWF4aW11bVNpemUpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgaWYgKG1heGltdW1TaXplICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgdGhpcy5tYXhpbXVtU2l6ZS5zZXRTaXplKG1heGltdW1TaXplKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICB0aGlzLm1heGltdW1TaXplID0gbnVsbDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgICAgIGZpcmVQcm9wZXJ0eUNoYW5nZSgibWF4aW11bVNpemUiLCBvbGRNYXhpbXVtU2l6ZSwgdGhpcy5tYXhpbXVtU2l6ZSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGludmFsaWRhdGVSZWFsUGFyZW50KCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbWluaW11bSBzaXplIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1pbmltdW1TaXplCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IG1pbmltdW0gc2l6ZSBvZiB0aGUgQ29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldE1pbmltdW1TaXplKERpbWVuc2lvbiBtaW5pbXVtU2l6ZSkgewotICAgICAgICBEaW1lbnNpb24gb2xkTWluaW11bVNpemU7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgb2xkTWluaW11bVNpemUgPSB0aGlzLm1pbmltdW1TaXplOwotICAgICAgICAgICAgaWYgKG9sZE1pbmltdW1TaXplICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBvbGRNaW5pbXVtU2l6ZSA9IG9sZE1pbmltdW1TaXplLmdldFNpemUoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh0aGlzLm1pbmltdW1TaXplID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBpZiAobWluaW11bVNpemUgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICB0aGlzLm1pbmltdW1TaXplID0gbmV3IERpbWVuc2lvbihtaW5pbXVtU2l6ZSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBpZiAobWluaW11bVNpemUgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICB0aGlzLm1pbmltdW1TaXplLnNldFNpemUobWluaW11bVNpemUpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMubWluaW11bVNpemUgPSBudWxsOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICAgICAgZmlyZVByb3BlcnR5Q2hhbmdlKCJtaW5pbXVtU2l6ZSIsIG9sZE1pbmltdW1TaXplLCB0aGlzLm1pbmltdW1TaXplKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIC8vID8/P0FXVDogaW52YWxpZGF0ZVJlYWxQYXJlbnQoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBwcmVmZXJyZWQgc2l6ZSBvZiB0aGUgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcmVmZXJyZWRTaXplCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHByZWZlcnJlZCBzaXplIG9mIHRoZSBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0UHJlZmVycmVkU2l6ZShEaW1lbnNpb24gcHJlZmVycmVkU2l6ZSkgewotICAgICAgICBEaW1lbnNpb24gb2xkUHJlZmVycmVkU2l6ZTsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBvbGRQcmVmZXJyZWRTaXplID0gdGhpcy5wcmVmZXJyZWRTaXplOwotICAgICAgICAgICAgaWYgKG9sZFByZWZlcnJlZFNpemUgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIG9sZFByZWZlcnJlZFNpemUgPSBvbGRQcmVmZXJyZWRTaXplLmdldFNpemUoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh0aGlzLnByZWZlcnJlZFNpemUgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIGlmIChwcmVmZXJyZWRTaXplICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgdGhpcy5wcmVmZXJyZWRTaXplID0gbmV3IERpbWVuc2lvbihwcmVmZXJyZWRTaXplKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGlmIChwcmVmZXJyZWRTaXplICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgdGhpcy5wcmVmZXJyZWRTaXplLnNldFNpemUocHJlZmVycmVkU2l6ZSk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgdGhpcy5wcmVmZXJyZWRTaXplID0gbnVsbDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgICAgIGZpcmVQcm9wZXJ0eUNoYW5nZSgicHJlZmVycmVkU2l6ZSIsIG9sZFByZWZlcnJlZFNpemUsIHRoaXMucHJlZmVycmVkU2l6ZSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGludmFsaWRhdGVSZWFsUGFyZW50KCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBSZWRyYXdNYW5hZ2VyIGdldFJlZHJhd01hbmFnZXIoKSB7IGlmIChwYXJlbnQgPT0gbnVsbCkgeyByZXR1cm4gbnVsbDsgfQotICAgICAqIHJldHVybiBwYXJlbnQuZ2V0UmVkcmF3TWFuYWdlcigpOyB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgaXMgZm9jdXNhYmlsaXR5IGV4cGxpY2l0bHkgc2V0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiBjb21wb25lbnQgaGFzIGEgZm9jdXNhYmxlIHBlZXIuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBib29sZWFuIGlzUGVlckZvY3VzYWJsZSgpIHsgLy8gVGhlIHJlY29tbWVuZGF0aW9ucyBmb3IgV2luZG93cyBhbmQgVW5peAotICAgICAqIGFyZSB0aGF0IC8vIENhbnZhc2VzLCBMYWJlbHMsIFBhbmVscywgU2Nyb2xsYmFycywgU2Nyb2xsUGFuZXMsIFdpbmRvd3MsCi0gICAgICogLy8gYW5kIGxpZ2h0d2VpZ2h0IENvbXBvbmVudHMgaGF2ZSBub24tZm9jdXNhYmxlIHBlZXJzLCAvLyBhbmQgYWxsIG90aGVyCi0gICAgICogQ29tcG9uZW50cyBoYXZlIGZvY3VzYWJsZSBwZWVycy4gaWYgKHRoaXMgaW5zdGFuY2VvZiBDYW52YXMgfHwgdGhpcwotICAgICAqIGluc3RhbmNlb2YgTGFiZWwgfHwgdGhpcyBpbnN0YW5jZW9mIFBhbmVsIHx8IHRoaXMgaW5zdGFuY2VvZiBTY3JvbGxiYXIgfHwKLSAgICAgKiB0aGlzIGluc3RhbmNlb2YgU2Nyb2xsUGFuZSB8fCB0aGlzIGluc3RhbmNlb2YgV2luZG93IHx8IGlzTGlnaHR3ZWlnaHQoKSkKLSAgICAgKiB7IHJldHVybiBmYWxzZTsgfSByZXR1cm4gdHJ1ZTsgfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogQHJldHVybiB0cnVlIGlmIGZvY3VzYWJpbGl0eSB3YXMgZXhwbGljaXRseSBzZXQgdmlhIGEgY2FsbCB0bwotICAgICAqICAgICAgICAgc2V0Rm9jdXNhYmxlKCkgb3IgdmlhIG92ZXJyaWRpbmcgaXNGb2N1c2FibGUoKSBvcgotICAgICAqICAgICAgICAgaXNGb2N1c1RyYXZlcnNhYmxlKCkuCi0gICAgICovCi0gICAgYm9vbGVhbiBpc0ZvY3VzYWJpbGl0eUV4cGxpY2l0bHlTZXQoKSB7Ci0gICAgICAgIHJldHVybiBjYWxsZWRTZXRGb2N1c2FibGUgfHwgb3ZlcnJpZGVuSXNGb2N1c2FibGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUGFpbnRzIHRoZSBjb21wb25lbnQgYW5kIGFsbCBvZiBpdHMgc3ViY29tcG9uZW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZwotICAgICAqICAgICAgICAgICAgdGhlIEdyYXBoaWNzIHRvIGJlIHVzZWQgZm9yIHBhaW50aW5nLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHBhaW50QWxsKEdyYXBoaWNzIGcpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBwYWludChnKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBVcGRhdGVzIHRoaXMgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBnCi0gICAgICogICAgICAgICAgICB0aGUgR3JhcGhpY3MgdG8gYmUgdXNlZCBmb3IgdXBkYXRpbmcuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgdXBkYXRlKEdyYXBoaWNzIGcpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpZiAoIWlzTGlnaHR3ZWlnaHQoKSAmJiAhaXNQcmVwYWludGVyKCkpIHsKLSAgICAgICAgICAgICAgICBnLnNldENvbG9yKGdldEJhY2tncm91bmQoKSk7Ci0gICAgICAgICAgICAgICAgZy5maWxsUmVjdCgwLCAwLCB3LCBoKTsKLSAgICAgICAgICAgICAgICBnLnNldENvbG9yKGdldEZvcmVncm91bmQoKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwYWludChnKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQYWludHMgdGhpcyBjb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGcKLSAgICAgKiAgICAgICAgICAgIHRoZSBHcmFwaGljcyB0byBiZSB1c2VkIGZvciBwYWludGluZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBwYWludChHcmFwaGljcyBnKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgLy8gSnVzdCB0byBub3RoaW5nCi0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJlcGFyZXMgdGhlIGNvbXBvbmVudCB0byBiZSBwYWludGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnCi0gICAgICogICAgICAgICAgICB0aGUgR3JhcGhpY3MgdG8gYmUgdXNlZCBmb3IgcGFpbnRpbmcuCi0gICAgICovCi0gICAgdm9pZCBwcmVwYWludChHcmFwaGljcyBnKSB7Ci0gICAgICAgIC8vIEp1c3QgdG8gbm90aGluZy4gRm9yIG92ZXJyaWRpbmcuCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIGlzIHByZXBhaW50ZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBpcyBwcmVwYWludGVyLgotICAgICAqLwotICAgIGJvb2xlYW4gaXNQcmVwYWludGVyKCkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJlcGFyZTQgaGllcmFyY2h5IGNoYW5nZS4KLSAgICAgKi8KLSAgICB2b2lkIHByZXBhcmU0SGllcmFyY2h5Q2hhbmdlKCkgewotICAgICAgICBpZiAoaGllcmFyY2h5Q2hhbmdpbmdDb3VudGVyKysgPT0gMCkgewotICAgICAgICAgICAgd2FzU2hvd2luZyA9IGlzU2hvd2luZygpOwotICAgICAgICAgICAgd2FzRGlzcGxheWFibGUgPSBpc0Rpc3BsYXlhYmxlKCk7Ci0gICAgICAgICAgICBwcmVwYXJlQ2hpbGRyZW40SGllcmFyY2h5Q2hhbmdlKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcmVwYXJlIGNoaWxkcmVuNCBoaWVyYXJjaHkgY2hhbmdlLgotICAgICAqLwotICAgIHZvaWQgcHJlcGFyZUNoaWxkcmVuNEhpZXJhcmNoeUNoYW5nZSgpIHsKLSAgICAgICAgLy8gVG8gYmUgaW5oZXJpdGVkIGJ5IENvbnRhaW5lcgotICAgIH0KLQotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogdm9pZCBmaW5pc2hIaWVyYXJjaHlDaGFuZ2UoQ29tcG9uZW50IGNoYW5nZWQsIENvbnRhaW5lciBjaGFuZ2VkUGFyZW50LAotICAgICAqIGludCBhbmNlc3RvckZsYWdzKSB7IGlmICgtLWhpZXJhcmNoeUNoYW5naW5nQ291bnRlciA9PSAwKSB7IGludAotICAgICAqIGNoYW5nZUZsYWdzID0gYW5jZXN0b3JGbGFnczsgaWYgKHdhc1Nob3dpbmcgIT0gaXNTaG93aW5nKCkpIHsgY2hhbmdlRmxhZ3MKLSAgICAgKiB8PSBIaWVyYXJjaHlFdmVudC5TSE9XSU5HX0NIQU5HRUQ7IH0gaWYgKHdhc0Rpc3BsYXlhYmxlICE9Ci0gICAgICogaXNEaXNwbGF5YWJsZSgpKSB7IGNoYW5nZUZsYWdzIHw9IEhpZXJhcmNoeUV2ZW50LkRJU1BMQVlBQklMSVRZX0NIQU5HRUQ7Ci0gICAgICogfSBpZiAoY2hhbmdlRmxhZ3MgPiAwKSB7IHBvc3RFdmVudChuZXcgSGllcmFyY2h5RXZlbnQodGhpcywKLSAgICAgKiBIaWVyYXJjaHlFdmVudC5ISUVSQVJDSFlfQ0hBTkdFRCwgY2hhbmdlZCwgY2hhbmdlZFBhcmVudCwgY2hhbmdlRmxhZ3MpKTsKLSAgICAgKiB9IGZpbmlzaENoaWxkcmVuSGllcmFyY2h5Q2hhbmdlKGNoYW5nZWQsIGNoYW5nZWRQYXJlbnQsIGFuY2VzdG9yRmxhZ3MpOyB9Ci0gICAgICogfSB2b2lkIGZpbmlzaENoaWxkcmVuSGllcmFyY2h5Q2hhbmdlKENvbXBvbmVudCBjaGFuZ2VkLCBDb250YWluZXIKLSAgICAgKiBjaGFuZ2VkUGFyZW50LCBpbnQgYW5jZXN0b3JGbGFncykgeyAvLyBUbyBiZSBpbmhlcml0ZWQgYnkgQ29udGFpbmVyIH0KLSAgICAgKiB2b2lkIHBvc3RIaWVyYXJjaHlCb3VuZHNFdmVudHMoQ29tcG9uZW50IGNoYW5nZWQsIGludCBpZCkgeyBwb3N0RXZlbnQobmV3Ci0gICAgICogSGllcmFyY2h5RXZlbnQodGhpcywgaWQsIGNoYW5nZWQsIG51bGwsIDApKTsgfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogU3ByZWFkIGhpZXJhcmNoeSBib3VuZHMgZXZlbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaGFuZ2VkCi0gICAgICogICAgICAgICAgICB0aGUgY2hhbmdlZC4KLSAgICAgKiBAcGFyYW0gaWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBpZC4KLSAgICAgKi8KLSAgICB2b2lkIHNwcmVhZEhpZXJhcmNoeUJvdW5kc0V2ZW50cyhDb21wb25lbnQgY2hhbmdlZCwgaW50IGlkKSB7Ci0gICAgICAgIC8vIFRvIGJlIGluaGVyaXRlZCBieSBDb250YWluZXIKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEaXNwYXRjaGVzIGFuIGV2ZW50IHRvIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgRXZlbnQuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIHZvaWQgZGlzcGF0Y2hFdmVudChBV1RFdmVudCBlKSB7Ci0gICAgICAgIC8vID8/P0FXVAotICAgICAgICAvKgotICAgICAgICAgKiBpZiAoZS5pc0NvbnN1bWVkKCkpIHsgcmV0dXJuOyB9IGlmIChlIGluc3RhbmNlb2YgUGFpbnRFdmVudCkgewotICAgICAgICAgKiB0b29sa2l0LmRpc3BhdGNoQVdURXZlbnQoZSk7IHByb2Nlc3NQYWludEV2ZW50KChQYWludEV2ZW50KSBlKTsKLSAgICAgICAgICogcmV0dXJuOyB9IEtleWJvYXJkRm9jdXNNYW5hZ2VyIGtmbSA9Ci0gICAgICAgICAqIEtleWJvYXJkRm9jdXNNYW5hZ2VyLmdldEN1cnJlbnRLZXlib2FyZEZvY3VzTWFuYWdlcigpOyBpZgotICAgICAgICAgKiAoIWUuZGlzcGF0Y2hlZEJ5S0ZNICYmIGtmbS5kaXNwYXRjaEV2ZW50KGUpKSB7IHJldHVybjsgfSBpZiAoZQotICAgICAgICAgKiBpbnN0YW5jZW9mIEtleUV2ZW50KSB7IEtleUV2ZW50IGtlID0gKEtleUV2ZW50KSBlOyAvLyBjb25zdW1lcwotICAgICAgICAgKiBLZXlFdmVudCB3aGljaCByZXByZXNlbnRzIGEgZm9jdXMgdHJhdmVyc2FsIGtleSBpZgotICAgICAgICAgKiAoZ2V0Rm9jdXNUcmF2ZXJzYWxLZXlzRW5hYmxlZCgpKSB7IGtmbS5wcm9jZXNzS2V5RXZlbnQodGhpcywga2UpOyBpZgotICAgICAgICAgKiAoa2UuaXNDb25zdW1lZCgpKSB7IHJldHVybjsgfSB9IH0gaWYgKGlucHV0TWV0aG9kc0VuYWJsZWQgJiYKLSAgICAgICAgICogZGlzcGF0Y2hUb0lNICYmIGUuaXNQb3N0ZWQgJiYgZGlzcGF0Y2hFdmVudFRvSU0oZSkpIHsgcmV0dXJuOyB9IGlmCi0gICAgICAgICAqIChlLmdldElEKCkgPT0gV2luZG93RXZlbnQuV0lORE9XX0lDT05JRklFRCkgewotICAgICAgICAgKiBub3RpZnlJbnB1dE1ldGhvZChudWxsKTsgfSBBV1RFdmVudC5FdmVudERlc2NyaXB0b3IgZGVzY3JpcHRvciA9Ci0gICAgICAgICAqIHRvb2xraXQuZXZlbnRUeXBlTG9va3VwLmdldEV2ZW50RGVzY3JpcHRvcihlKTsKLSAgICAgICAgICogdG9vbGtpdC5kaXNwYXRjaEFXVEV2ZW50KGUpOyBpZiAoZGVzY3JpcHRvciAhPSBudWxsKSB7IGlmCi0gICAgICAgICAqIChpc0V2ZW50RW5hYmxlZChkZXNjcmlwdG9yLmV2ZW50TWFzaykgfHwKLSAgICAgICAgICogKGdldExpc3RlbmVycyhkZXNjcmlwdG9yLmxpc3RlbmVyVHlwZSkubGVuZ3RoID4gMCkpIHsKLSAgICAgICAgICogcHJvY2Vzc0V2ZW50KGUpOyB9IC8vIGlucHV0IGV2ZW50cyBjYW4gYmUgY29uc3VtZWQgYnkgdXNlciBsaXN0ZW5lcnM6Ci0gICAgICAgICAqIGlmICghZS5pc0NvbnN1bWVkKCkgJiYgKChlbmFibGVkQVdURXZlbnRzICYgZGVzY3JpcHRvci5ldmVudE1hc2spICE9Ci0gICAgICAgICAqIDApKSB7IHBvc3Rwcm9jZXNzRXZlbnQoZSwgZGVzY3JpcHRvci5ldmVudE1hc2spOyB9IH0KLSAgICAgICAgICogcG9zdERlcHJlY2F0ZWRFdmVudChlKTsKLSAgICAgICAgICovCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogUG9zdCBkZXByZWNhdGVkIGV2ZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgcG9zdERlcHJlY2F0ZWRFdmVudChBV1RFdmVudCBlKSB7Ci0gICAgICAgIGlmIChkZXByZWNhdGVkRXZlbnRIYW5kbGVyKSB7Ci0gICAgICAgICAgICBFdmVudCBldnQgPSBlLmdldEV2ZW50KCk7Ci0gICAgICAgICAgICBpZiAoZXZ0ICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBwb3N0RXZlbnQoZXZ0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBvc3Rwcm9jZXNzIGV2ZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgZS4KLSAgICAgKiBAcGFyYW0gZXZlbnRNYXNrCi0gICAgICogICAgICAgICAgICB0aGUgZXZlbnQgbWFzay4KLSAgICAgKi8KLSAgICB2b2lkIHBvc3Rwcm9jZXNzRXZlbnQoQVdURXZlbnQgZSwgbG9uZyBldmVudE1hc2spIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAvLyBjYWxsIHN5c3RlbSBsaXN0ZW5lcnMgdW5kZXIgQVdUIGxvY2sKLSAgICAgICAgICAgIGlmIChldmVudE1hc2sgPT0gQVdURXZlbnQuRk9DVVNfRVZFTlRfTUFTSykgewotICAgICAgICAgICAgICAgIHByZXByb2Nlc3NGb2N1c0V2ZW50KChGb2N1c0V2ZW50KWUpOwotICAgICAgICAgICAgfSBlbHNlIGlmIChldmVudE1hc2sgPT0gQVdURXZlbnQuS0VZX0VWRU5UX01BU0spIHsKLSAgICAgICAgICAgICAgICBwcmVwcm9jZXNzS2V5RXZlbnQoKEtleUV2ZW50KWUpOwotICAgICAgICAgICAgfSBlbHNlIGlmIChldmVudE1hc2sgPT0gQVdURXZlbnQuTU9VU0VfRVZFTlRfTUFTSykgewotICAgICAgICAgICAgICAgIHByZXByb2Nlc3NNb3VzZUV2ZW50KChNb3VzZUV2ZW50KWUpOwotICAgICAgICAgICAgfSBlbHNlIGlmIChldmVudE1hc2sgPT0gQVdURXZlbnQuTU9VU0VfTU9USU9OX0VWRU5UX01BU0spIHsKLSAgICAgICAgICAgICAgICBwcmVwcm9jZXNzTW91c2VNb3Rpb25FdmVudCgoTW91c2VFdmVudCllKTsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoZXZlbnRNYXNrID09IEFXVEV2ZW50LkNPTVBPTkVOVF9FVkVOVF9NQVNLKSB7Ci0gICAgICAgICAgICAgICAgcHJlcHJvY2Vzc0NvbXBvbmVudEV2ZW50KChDb21wb25lbnRFdmVudCllKTsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoZXZlbnRNYXNrID09IEFXVEV2ZW50Lk1PVVNFX1dIRUVMX0VWRU5UX01BU0spIHsKLSAgICAgICAgICAgICAgICBwcmVwcm9jZXNzTW91c2VXaGVlbEV2ZW50KChNb3VzZVdoZWVsRXZlbnQpZSk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGV2ZW50TWFzayA9PSBBV1RFdmVudC5JTlBVVF9NRVRIT0RfRVZFTlRfTUFTSykgewotICAgICAgICAgICAgICAgIHByZXByb2Nlc3NJbnB1dE1ldGhvZEV2ZW50KChJbnB1dE1ldGhvZEV2ZW50KWUpOwotICAgICAgICAgICAgfQotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByZXByb2Nlc3MgaW5wdXQgbWV0aG9kIGV2ZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgcHJlcHJvY2Vzc0lucHV0TWV0aG9kRXZlbnQoSW5wdXRNZXRob2RFdmVudCBlKSB7Ci0gICAgICAgIHByb2Nlc3NJbnB1dE1ldGhvZEV2ZW50SW1wbChlLCBpbnB1dE1ldGhvZExpc3RlbmVycy5nZXRTeXN0ZW1MaXN0ZW5lcnMoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJlcHJvY2VzcyBtb3VzZSB3aGVlbCBldmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHByZXByb2Nlc3NNb3VzZVdoZWVsRXZlbnQoTW91c2VXaGVlbEV2ZW50IGUpIHsKLSAgICAgICAgcHJvY2Vzc01vdXNlV2hlZWxFdmVudEltcGwoZSwgbW91c2VXaGVlbExpc3RlbmVycy5nZXRTeXN0ZW1MaXN0ZW5lcnMoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2VzcyBtb3VzZSB3aGVlbCBldmVudCBpbXBsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgZS4KLSAgICAgKiBAcGFyYW0gYwotICAgICAqICAgICAgICAgICAgdGhlIGMuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHByb2Nlc3NNb3VzZVdoZWVsRXZlbnRJbXBsKE1vdXNlV2hlZWxFdmVudCBlLCBDb2xsZWN0aW9uPE1vdXNlV2hlZWxMaXN0ZW5lcj4gYykgewotICAgICAgICBmb3IgKE1vdXNlV2hlZWxMaXN0ZW5lciBsaXN0ZW5lciA6IGMpIHsKLSAgICAgICAgICAgIHN3aXRjaCAoZS5nZXRJRCgpKSB7Ci0gICAgICAgICAgICAgICAgY2FzZSBNb3VzZUV2ZW50Lk1PVVNFX1dIRUVMOgotICAgICAgICAgICAgICAgICAgICBsaXN0ZW5lci5tb3VzZVdoZWVsTW92ZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJlcHJvY2VzcyBjb21wb25lbnQgZXZlbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBwcmVwcm9jZXNzQ29tcG9uZW50RXZlbnQoQ29tcG9uZW50RXZlbnQgZSkgewotICAgICAgICBwcm9jZXNzQ29tcG9uZW50RXZlbnRJbXBsKGUsIGNvbXBvbmVudExpc3RlbmVycy5nZXRTeXN0ZW1MaXN0ZW5lcnMoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJlcHJvY2VzcyBtb3VzZSBtb3Rpb24gZXZlbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlLgotICAgICAqLwotICAgIHZvaWQgcHJlcHJvY2Vzc01vdXNlTW90aW9uRXZlbnQoTW91c2VFdmVudCBlKSB7Ci0gICAgICAgIHByb2Nlc3NNb3VzZU1vdGlvbkV2ZW50SW1wbChlLCBtb3VzZU1vdGlvbkxpc3RlbmVycy5nZXRTeXN0ZW1MaXN0ZW5lcnMoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJlcHJvY2VzcyBtb3VzZSBldmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIGUKLSAgICAgKi8KLSAgICB2b2lkIHByZXByb2Nlc3NNb3VzZUV2ZW50KE1vdXNlRXZlbnQgZSkgewotICAgICAgICBwcm9jZXNzTW91c2VFdmVudEltcGwoZSwgbW91c2VMaXN0ZW5lcnMuZ2V0U3lzdGVtTGlzdGVuZXJzKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByZXByb2Nlc3Mga2V5IGV2ZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgZS4KLSAgICAgKi8KLSAgICB2b2lkIHByZXByb2Nlc3NLZXlFdmVudChLZXlFdmVudCBlKSB7Ci0gICAgICAgIHByb2Nlc3NLZXlFdmVudEltcGwoZSwga2V5TGlzdGVuZXJzLmdldFN5c3RlbUxpc3RlbmVycygpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcmVwcm9jZXNzIGZvY3VzIGV2ZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgZS4KLSAgICAgKi8KLSAgICB2b2lkIHByZXByb2Nlc3NGb2N1c0V2ZW50KEZvY3VzRXZlbnQgZSkgewotICAgICAgICBwcm9jZXNzRm9jdXNFdmVudEltcGwoZSwgZm9jdXNMaXN0ZW5lcnMuZ2V0U3lzdGVtTGlzdGVuZXJzKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyBBV1RFdmVudCBvY2N1cnJlZCBvbiB0aGlzIGNvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIEFXVEV2ZW50LgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NFdmVudChBV1RFdmVudCBlKSB7Ci0gICAgICAgIGxvbmcgZXZlbnRNYXNrID0gdG9vbGtpdC5ldmVudFR5cGVMb29rdXAuZ2V0RXZlbnRNYXNrKGUpOwotICAgICAgICBpZiAoZXZlbnRNYXNrID09IEFXVEV2ZW50LkNPTVBPTkVOVF9FVkVOVF9NQVNLKSB7Ci0gICAgICAgICAgICBwcm9jZXNzQ29tcG9uZW50RXZlbnQoKENvbXBvbmVudEV2ZW50KWUpOwotICAgICAgICB9IGVsc2UgaWYgKGV2ZW50TWFzayA9PSBBV1RFdmVudC5GT0NVU19FVkVOVF9NQVNLKSB7Ci0gICAgICAgICAgICBwcm9jZXNzRm9jdXNFdmVudCgoRm9jdXNFdmVudCllKTsKLSAgICAgICAgfSBlbHNlIGlmIChldmVudE1hc2sgPT0gQVdURXZlbnQuS0VZX0VWRU5UX01BU0spIHsKLSAgICAgICAgICAgIHByb2Nlc3NLZXlFdmVudCgoS2V5RXZlbnQpZSk7Ci0gICAgICAgIH0gZWxzZSBpZiAoZXZlbnRNYXNrID09IEFXVEV2ZW50Lk1PVVNFX0VWRU5UX01BU0spIHsKLSAgICAgICAgICAgIHByb2Nlc3NNb3VzZUV2ZW50KChNb3VzZUV2ZW50KWUpOwotICAgICAgICB9IGVsc2UgaWYgKGV2ZW50TWFzayA9PSBBV1RFdmVudC5NT1VTRV9XSEVFTF9FVkVOVF9NQVNLKSB7Ci0gICAgICAgICAgICBwcm9jZXNzTW91c2VXaGVlbEV2ZW50KChNb3VzZVdoZWVsRXZlbnQpZSk7Ci0gICAgICAgIH0gZWxzZSBpZiAoZXZlbnRNYXNrID09IEFXVEV2ZW50Lk1PVVNFX01PVElPTl9FVkVOVF9NQVNLKSB7Ci0gICAgICAgICAgICBwcm9jZXNzTW91c2VNb3Rpb25FdmVudCgoTW91c2VFdmVudCllKTsKLSAgICAgICAgfSBlbHNlIGlmIChldmVudE1hc2sgPT0gQVdURXZlbnQuSElFUkFSQ0hZX0VWRU5UX01BU0spIHsKLSAgICAgICAgICAgIHByb2Nlc3NIaWVyYXJjaHlFdmVudCgoSGllcmFyY2h5RXZlbnQpZSk7Ci0gICAgICAgIH0gZWxzZSBpZiAoZXZlbnRNYXNrID09IEFXVEV2ZW50LkhJRVJBUkNIWV9CT1VORFNfRVZFTlRfTUFTSykgewotICAgICAgICAgICAgcHJvY2Vzc0hpZXJhcmNoeUJvdW5kc0V2ZW50KChIaWVyYXJjaHlFdmVudCllKTsKLSAgICAgICAgfSBlbHNlIGlmIChldmVudE1hc2sgPT0gQVdURXZlbnQuSU5QVVRfTUVUSE9EX0VWRU5UX01BU0spIHsKLSAgICAgICAgICAgIHByb2Nlc3NJbnB1dE1ldGhvZEV2ZW50KChJbnB1dE1ldGhvZEV2ZW50KWUpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiBhbGwgbGlzdGVuZXIncyBvYmplY3RzIGJhc2VkIG9uIHRoZSBzcGVjaWZpZWQgbGlzdGVuZXIKLSAgICAgKiB0eXBlIGFuZCByZWdpc3RlcmVkIHRvIHRoaXMgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsaXN0ZW5lclR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsaXN0ZW5lciB0eXBlLgotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgb2YgYWxsIGxpc3RlbmVyJ3Mgb2JqZWN0cyBiYXNlZCBvbiB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBsaXN0ZW5lciB0eXBlIGFuZCByZWdpc3RlcmVkIHRvIHRoaXMgQ29tcG9uZW50LgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bmNoZWNrZWQiKQotICAgIHB1YmxpYyA8VCBleHRlbmRzIEV2ZW50TGlzdGVuZXI+IFRbXSBnZXRMaXN0ZW5lcnMoQ2xhc3M8VD4gbGlzdGVuZXJUeXBlKSB7Ci0gICAgICAgIGlmIChDb21wb25lbnRMaXN0ZW5lci5jbGFzcy5pc0Fzc2lnbmFibGVGcm9tKGxpc3RlbmVyVHlwZSkpIHsKLSAgICAgICAgICAgIHJldHVybiAoVFtdKWdldENvbXBvbmVudExpc3RlbmVycygpOwotICAgICAgICB9IGVsc2UgaWYgKEZvY3VzTGlzdGVuZXIuY2xhc3MuaXNBc3NpZ25hYmxlRnJvbShsaXN0ZW5lclR5cGUpKSB7Ci0gICAgICAgICAgICByZXR1cm4gKFRbXSlnZXRGb2N1c0xpc3RlbmVycygpOwotICAgICAgICB9IGVsc2UgaWYgKEhpZXJhcmNoeUJvdW5kc0xpc3RlbmVyLmNsYXNzLmlzQXNzaWduYWJsZUZyb20obGlzdGVuZXJUeXBlKSkgewotICAgICAgICAgICAgcmV0dXJuIChUW10pZ2V0SGllcmFyY2h5Qm91bmRzTGlzdGVuZXJzKCk7Ci0gICAgICAgIH0gZWxzZSBpZiAoSGllcmFyY2h5TGlzdGVuZXIuY2xhc3MuaXNBc3NpZ25hYmxlRnJvbShsaXN0ZW5lclR5cGUpKSB7Ci0gICAgICAgICAgICByZXR1cm4gKFRbXSlnZXRIaWVyYXJjaHlMaXN0ZW5lcnMoKTsKLSAgICAgICAgfSBlbHNlIGlmIChJbnB1dE1ldGhvZExpc3RlbmVyLmNsYXNzLmlzQXNzaWduYWJsZUZyb20obGlzdGVuZXJUeXBlKSkgewotICAgICAgICAgICAgcmV0dXJuIChUW10pZ2V0SW5wdXRNZXRob2RMaXN0ZW5lcnMoKTsKLSAgICAgICAgfSBlbHNlIGlmIChLZXlMaXN0ZW5lci5jbGFzcy5pc0Fzc2lnbmFibGVGcm9tKGxpc3RlbmVyVHlwZSkpIHsKLSAgICAgICAgICAgIHJldHVybiAoVFtdKWdldEtleUxpc3RlbmVycygpOwotICAgICAgICB9IGVsc2UgaWYgKE1vdXNlV2hlZWxMaXN0ZW5lci5jbGFzcy5pc0Fzc2lnbmFibGVGcm9tKGxpc3RlbmVyVHlwZSkpIHsKLSAgICAgICAgICAgIHJldHVybiAoVFtdKWdldE1vdXNlV2hlZWxMaXN0ZW5lcnMoKTsKLSAgICAgICAgfSBlbHNlIGlmIChNb3VzZU1vdGlvbkxpc3RlbmVyLmNsYXNzLmlzQXNzaWduYWJsZUZyb20obGlzdGVuZXJUeXBlKSkgewotICAgICAgICAgICAgcmV0dXJuIChUW10pZ2V0TW91c2VNb3Rpb25MaXN0ZW5lcnMoKTsKLSAgICAgICAgfSBlbHNlIGlmIChNb3VzZUxpc3RlbmVyLmNsYXNzLmlzQXNzaWduYWJsZUZyb20obGlzdGVuZXJUeXBlKSkgewotICAgICAgICAgICAgcmV0dXJuIChUW10pZ2V0TW91c2VMaXN0ZW5lcnMoKTsKLSAgICAgICAgfSBlbHNlIGlmIChQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyLmNsYXNzLmlzQXNzaWduYWJsZUZyb20obGlzdGVuZXJUeXBlKSkgewotICAgICAgICAgICAgcmV0dXJuIChUW10pZ2V0UHJvcGVydHlDaGFuZ2VMaXN0ZW5lcnMoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gKFRbXSlBcnJheS5uZXdJbnN0YW5jZShsaXN0ZW5lclR5cGUsIDApOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3MgcGFpbnQgZXZlbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2ZW50Ci0gICAgICogICAgICAgICAgICB0aGUgZXZlbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHByb2Nlc3NQYWludEV2ZW50KFBhaW50RXZlbnQgZXZlbnQpIHsKLSAgICAgICAgaWYgKHJlZHJhd01hbmFnZXIgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIFJlY3RhbmdsZSBjbGlwUmVjdCA9IGV2ZW50LmdldFVwZGF0ZVJlY3QoKTsKLSAgICAgICAgaWYgKChjbGlwUmVjdC53aWR0aCA8PSAwKSB8fCAoY2xpcFJlY3QuaGVpZ2h0IDw9IDApKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgR3JhcGhpY3MgZyA9IGdldEdyYXBoaWNzKCk7Ci0gICAgICAgIGlmIChnID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBpbml0R3JhcGhpY3MoZywgZXZlbnQpOwotICAgICAgICBpZiAoIWdldElnbm9yZVJlcGFpbnQoKSkgewotICAgICAgICAgICAgaWYgKGV2ZW50LmdldElEKCkgPT0gUGFpbnRFdmVudC5QQUlOVCkgewotICAgICAgICAgICAgICAgIHBhaW50KGcpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICB1cGRhdGUoZyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgZy5kaXNwb3NlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5pdHMgdGhlIGdyYXBoaWNzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnCi0gICAgICogICAgICAgICAgICB0aGUgZy4KLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIGUuCi0gICAgICovCi0gICAgdm9pZCBpbml0R3JhcGhpY3MoR3JhcGhpY3MgZywgUGFpbnRFdmVudCBlKSB7Ci0gICAgICAgIFJlY3RhbmdsZSBjbGlwID0gZS5nZXRVcGRhdGVSZWN0KCk7Ci0gICAgICAgIGlmIChjbGlwIGluc3RhbmNlb2YgQ2xpcFJlZ2lvbikgewotICAgICAgICAgICAgZy5zZXRDbGlwKCgoQ2xpcFJlZ2lvbiljbGlwKS5nZXRDbGlwKCkpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZy5zZXRDbGlwKGNsaXApOwotICAgICAgICB9Ci0gICAgICAgIGlmIChpc1ByZXBhaW50ZXIoKSkgewotICAgICAgICAgICAgcHJlcGFpbnQoZyk7Ci0gICAgICAgIH0gZWxzZSBpZiAoIWlzTGlnaHR3ZWlnaHQoKSAmJiAoZS5nZXRJRCgpID09IFBhaW50RXZlbnQuUEFJTlQpKSB7Ci0gICAgICAgICAgICBnLnNldENvbG9yKGdldEJhY2tncm91bmQoKSk7Ci0gICAgICAgICAgICBnLmZpbGxSZWN0KDAsIDAsIHcsIGgpOwotICAgICAgICB9Ci0gICAgICAgIGcuc2V0Rm9udChnZXRGb250KCkpOwotICAgICAgICBnLnNldENvbG9yKGdldEZvcmVncm91bmQoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRW5hYmxlcyB0aGUgZXZlbnRzIHdpdGggdGhlIHNwZWNpZmllZCBldmVudCBtYXNrIHRvIGJlIGRlbGl2ZXJlZCB0byB0aGlzCi0gICAgICogY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBldmVudHNUb0VuYWJsZQotICAgICAqICAgICAgICAgICAgdGhlIGV2ZW50cyBtYXNrIHdoaWNoIHNwZWNpZmllcyB0aGUgdHlwZXMgb2YgZXZlbnRzIHRvIGVuYWJsZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgZmluYWwgdm9pZCBlbmFibGVFdmVudHMobG9uZyBldmVudHNUb0VuYWJsZSkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGVuYWJsZWRFdmVudHMgfD0gZXZlbnRzVG9FbmFibGU7Ci0gICAgICAgICAgICBkZXByZWNhdGVkRXZlbnRIYW5kbGVyID0gZmFsc2U7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRW5hYmxlIGF3dCBldmVudHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2ZW50c1RvRW5hYmxlCi0gICAgICogICAgICAgICAgICB0aGUgZXZlbnRzIHRvIGVuYWJsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgZW5hYmxlQVdURXZlbnRzKGxvbmcgZXZlbnRzVG9FbmFibGUpIHsKLSAgICAgICAgZW5hYmxlZEFXVEV2ZW50cyB8PSBldmVudHNUb0VuYWJsZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEaXNhYmxlcyB0aGUgZXZlbnRzIHdpdGggdHlwZXMgc3BlY2lmaWVkIGJ5IHRoZSBzcGVjaWZpZWQgZXZlbnQgbWFzayBmcm9tCi0gICAgICogYmVpbmcgZGVsaXZlcmVkIHRvIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBldmVudHNUb0Rpc2FibGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBldmVudCBtYXNrIHNwZWNpZnlpbmcgdGhlIGV2ZW50IHR5cGVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBmaW5hbCB2b2lkIGRpc2FibGVFdmVudHMobG9uZyBldmVudHNUb0Rpc2FibGUpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBlbmFibGVkRXZlbnRzICY9IH5ldmVudHNUb0Rpc2FibGU7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBGb3IgdXNlIGluIE1vdXNlRGlzcGF0Y2hlciBvbmx5LiBSZWFsbHkgaXQgY2hlY2tzIG5vdCBvbmx5IG1vdXNlIGV2ZW50cy4KLSAgICAgKi8KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgaXMgbW91c2UgZXZlbnQgZW5hYmxlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXZlbnRNYXNrCi0gICAgICogICAgICAgICAgICB0aGUgZXZlbnQgbWFzay4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGlzIG1vdXNlIGV2ZW50IGVuYWJsZWQuCi0gICAgICovCi0gICAgYm9vbGVhbiBpc01vdXNlRXZlbnRFbmFibGVkKGxvbmcgZXZlbnRNYXNrKSB7Ci0gICAgICAgIHJldHVybiAoaXNFdmVudEVuYWJsZWQoZXZlbnRNYXNrKSB8fCAoZW5hYmxlZEFXVEV2ZW50cyAmIGV2ZW50TWFzaykgIT0gMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIGlzIGV2ZW50IGVuYWJsZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2ZW50TWFzawotICAgICAqICAgICAgICAgICAgdGhlIGV2ZW50IG1hc2suCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBpcyBldmVudCBlbmFibGVkLgotICAgICAqLwotICAgIGJvb2xlYW4gaXNFdmVudEVuYWJsZWQobG9uZyBldmVudE1hc2spIHsKLSAgICAgICAgcmV0dXJuICgoZW5hYmxlZEV2ZW50cyAmIGV2ZW50TWFzaykgIT0gMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRW5hYmxlcyBvciBkaXNhYmxlcyBpbnB1dCBtZXRob2Qgc3VwcG9ydCBmb3IgdGhpcyBjb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVuYWJsZQotICAgICAqICAgICAgICAgICAgdHJ1ZSB0byBlbmFibGUgaW5wdXQgbWV0aG9kIHN1cHBvcnQsIGZhbHNlIHRvIGRpc2FibGUgaXQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZW5hYmxlSW5wdXRNZXRob2RzKGJvb2xlYW4gZW5hYmxlKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKCFlbmFibGUpIHsKLSAgICAgICAgICAgICAgICByZW1vdmVOb3RpZnlJbnB1dENvbnRleHQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlucHV0TWV0aG9kc0VuYWJsZWQgPSBlbmFibGU7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiBhbGwgY29tcG9uZW50J3MgbGlzdGVuZXJzIHJlZ2lzdGVyZWQgZm9yIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgb2YgYWxsIGNvbXBvbmVudCdzIGxpc3RlbmVycyByZWdpc3RlcmVkIGZvciB0aGlzCi0gICAgICogICAgICAgICBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIENvbXBvbmVudExpc3RlbmVyW10gZ2V0Q29tcG9uZW50TGlzdGVuZXJzKCkgewotICAgICAgICByZXR1cm4gY29tcG9uZW50TGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMobmV3IENvbXBvbmVudExpc3RlbmVyWzBdKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBzcGVjaWZpZWQgY29tcG9uZW50IGxpc3RlbmVyIHRvIHRoZSBDb21wb25lbnQgZm9yIHJlY2VpdmluZwotICAgICAqIGNvbXBvbmVudCdzIGV2ZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgQ29tcG9uZW50TGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkQ29tcG9uZW50TGlzdGVuZXIoQ29tcG9uZW50TGlzdGVuZXIgbCkgewotICAgICAgICBjb21wb25lbnRMaXN0ZW5lcnMuYWRkVXNlckxpc3RlbmVyKGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIGNvbXBvbmVudCBsaXN0ZW5lciByZWdpc3RlcmVkIGZvciB0aGlzIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbAotICAgICAqICAgICAgICAgICAgdGhlIENvbXBvbmVudExpc3RlbmVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZUNvbXBvbmVudExpc3RlbmVyKENvbXBvbmVudExpc3RlbmVyIGwpIHsKLSAgICAgICAgY29tcG9uZW50TGlzdGVuZXJzLnJlbW92ZVVzZXJMaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgYSBjb21wb25lbnQgZXZlbnQgdGhhdCBoYXMgb2NjdXJyZWQgb24gdGhpcyBjb21wb25lbnQgYnkKLSAgICAgKiBkaXNwYXRjaGluZyB0aGVtIHRvIGFueSByZWdpc3RlcmVkIENvbXBvbmVudExpc3RlbmVyIG9iamVjdHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBDb21wb25lbnRFdmVudC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzQ29tcG9uZW50RXZlbnQoQ29tcG9uZW50RXZlbnQgZSkgewotICAgICAgICBwcm9jZXNzQ29tcG9uZW50RXZlbnRJbXBsKGUsIGNvbXBvbmVudExpc3RlbmVycy5nZXRVc2VyTGlzdGVuZXJzKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3MgY29tcG9uZW50IGV2ZW50IGltcGwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlLgotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgYy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgcHJvY2Vzc0NvbXBvbmVudEV2ZW50SW1wbChDb21wb25lbnRFdmVudCBlLCBDb2xsZWN0aW9uPENvbXBvbmVudExpc3RlbmVyPiBjKSB7Ci0gICAgICAgIGZvciAoQ29tcG9uZW50TGlzdGVuZXIgbGlzdGVuZXIgOiBjKSB7Ci0gICAgICAgICAgICBzd2l0Y2ggKGUuZ2V0SUQoKSkgewotICAgICAgICAgICAgICAgIGNhc2UgQ29tcG9uZW50RXZlbnQuQ09NUE9ORU5UX0hJRERFTjoKLSAgICAgICAgICAgICAgICAgICAgbGlzdGVuZXIuY29tcG9uZW50SGlkZGVuKGUpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIENvbXBvbmVudEV2ZW50LkNPTVBPTkVOVF9NT1ZFRDoKLSAgICAgICAgICAgICAgICAgICAgbGlzdGVuZXIuY29tcG9uZW50TW92ZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgQ29tcG9uZW50RXZlbnQuQ09NUE9ORU5UX1JFU0laRUQ6Ci0gICAgICAgICAgICAgICAgICAgIGxpc3RlbmVyLmNvbXBvbmVudFJlc2l6ZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgQ29tcG9uZW50RXZlbnQuQ09NUE9ORU5UX1NIT1dOOgotICAgICAgICAgICAgICAgICAgICBsaXN0ZW5lci5jb21wb25lbnRTaG93bihlKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIGZvY3VzIGxpc3RlbmVycyByZWdpc3RlcmVkIGZvciB0aGlzIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBmb2N1cyBsaXN0ZW5lcnMgcmVnaXN0ZXJlZCBmb3IgdGhpcyBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIEZvY3VzTGlzdGVuZXJbXSBnZXRGb2N1c0xpc3RlbmVycygpIHsKLSAgICAgICAgcmV0dXJuIGZvY3VzTGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMobmV3IEZvY3VzTGlzdGVuZXJbMF0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgdGhlIHNwZWNpZmllZCBmb2N1cyBsaXN0ZW5lciB0byB0aGUgQ29tcG9uZW50IGZvciByZWNlaXZpbmcgZm9jdXMKLSAgICAgKiBldmVudHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb2N1c0xpc3RlbmVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGFkZEZvY3VzTGlzdGVuZXIoRm9jdXNMaXN0ZW5lciBsKSB7Ci0gICAgICAgIGZvY3VzTGlzdGVuZXJzLmFkZFVzZXJMaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBhd3QgZm9jdXMgbGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBsLgotICAgICAqLwotICAgIHZvaWQgYWRkQVdURm9jdXNMaXN0ZW5lcihGb2N1c0xpc3RlbmVyIGwpIHsKLSAgICAgICAgZW5hYmxlQVdURXZlbnRzKEFXVEV2ZW50LkZPQ1VTX0VWRU5UX01BU0spOwotICAgICAgICBmb2N1c0xpc3RlbmVycy5hZGRTeXN0ZW1MaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHRoZSBmb2N1cyBsaXN0ZW5lciByZWdpc3RlcmVkIGZvciB0aGlzIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbAotICAgICAqICAgICAgICAgICAgdGhlIEZvY3VzTGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlRm9jdXNMaXN0ZW5lcihGb2N1c0xpc3RlbmVyIGwpIHsKLSAgICAgICAgZm9jdXNMaXN0ZW5lcnMucmVtb3ZlVXNlckxpc3RlbmVyKGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyBhIEZvY3VzRXZlbnQgdGhhdCBoYXMgb2NjdXJyZWQgb24gdGhpcyBjb21wb25lbnQgYnkgZGlzcGF0Y2hpbmcKLSAgICAgKiBpdCB0byB0aGUgcmVnaXN0ZXJlZCBsaXN0ZW5lcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb2N1c0V2ZW50LgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NGb2N1c0V2ZW50KEZvY3VzRXZlbnQgZSkgewotICAgICAgICBwcm9jZXNzRm9jdXNFdmVudEltcGwoZSwgZm9jdXNMaXN0ZW5lcnMuZ2V0VXNlckxpc3RlbmVycygpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzIGZvY3VzIGV2ZW50IGltcGwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlLgotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgYy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgcHJvY2Vzc0ZvY3VzRXZlbnRJbXBsKEZvY3VzRXZlbnQgZSwgQ29sbGVjdGlvbjxGb2N1c0xpc3RlbmVyPiBjKSB7Ci0gICAgICAgIGZvciAoRm9jdXNMaXN0ZW5lciBsaXN0ZW5lciA6IGMpIHsKLSAgICAgICAgICAgIHN3aXRjaCAoZS5nZXRJRCgpKSB7Ci0gICAgICAgICAgICAgICAgY2FzZSBGb2N1c0V2ZW50LkZPQ1VTX0dBSU5FRDoKLSAgICAgICAgICAgICAgICAgICAgbGlzdGVuZXIuZm9jdXNHYWluZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgRm9jdXNFdmVudC5GT0NVU19MT1NUOgotICAgICAgICAgICAgICAgICAgICBsaXN0ZW5lci5mb2N1c0xvc3QoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiByZWdpc3RlcmVkIEhpZXJhcmNoeUxpc3RlbmVycyBmb3IgdGhpcyBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiBhbiBhcnJheSBvZiByZWdpc3RlcmVkIEhpZXJhcmNoeUxpc3RlbmVycyBmb3IgdGhpcyBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIEhpZXJhcmNoeUxpc3RlbmVyW10gZ2V0SGllcmFyY2h5TGlzdGVuZXJzKCkgewotICAgICAgICByZXR1cm4gaGllcmFyY2h5TGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMobmV3IEhpZXJhcmNoeUxpc3RlbmVyWzBdKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBzcGVjaWZpZWQgaGllcmFyY2h5IGxpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgSGllcmFyY2h5TGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkSGllcmFyY2h5TGlzdGVuZXIoSGllcmFyY2h5TGlzdGVuZXIgbCkgewotICAgICAgICBoaWVyYXJjaHlMaXN0ZW5lcnMuYWRkVXNlckxpc3RlbmVyKGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIGhpZXJhcmNoeSBsaXN0ZW5lciByZWdpc3RlcmVkIGZvciB0aGlzIGNvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbAotICAgICAqICAgICAgICAgICAgdGhlIEhpZXJhcmNoeUxpc3RlbmVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZUhpZXJhcmNoeUxpc3RlbmVyKEhpZXJhcmNoeUxpc3RlbmVyIGwpIHsKLSAgICAgICAgaGllcmFyY2h5TGlzdGVuZXJzLnJlbW92ZVVzZXJMaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgYSBoaWVyYXJjaHkgZXZlbnQgdGhhdCBoYXMgb2NjdXJyZWQgb24gdGhpcyBjb21wb25lbnQgYnkKLSAgICAgKiBkaXNwYXRjaGluZyBpdCB0byB0aGUgcmVnaXN0ZXJlZCBsaXN0ZW5lcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBIaWVyYXJjaHlFdmVudC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzSGllcmFyY2h5RXZlbnQoSGllcmFyY2h5RXZlbnQgZSkgewotICAgICAgICBmb3IgKEhpZXJhcmNoeUxpc3RlbmVyIGxpc3RlbmVyIDogaGllcmFyY2h5TGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMoKSkgewotICAgICAgICAgICAgc3dpdGNoIChlLmdldElEKCkpIHsKLSAgICAgICAgICAgICAgICBjYXNlIEhpZXJhcmNoeUV2ZW50LkhJRVJBUkNIWV9DSEFOR0VEOgotICAgICAgICAgICAgICAgICAgICBsaXN0ZW5lci5oaWVyYXJjaHlDaGFuZ2VkKGUpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgSGllcmFyY2h5Qm91bmRzTGlzdGVuZXIgb2JqZWN0cyByZWdpc3RlcmVkIHRvIHRoaXMKLSAgICAgKiBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiBhbiBhcnJheSBvZiBIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lciBvYmplY3RzLgotICAgICAqLwotICAgIHB1YmxpYyBIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lcltdIGdldEhpZXJhcmNoeUJvdW5kc0xpc3RlbmVycygpIHsKLSAgICAgICAgcmV0dXJuIGhpZXJhcmNoeUJvdW5kc0xpc3RlbmVycy5nZXRVc2VyTGlzdGVuZXJzKG5ldyBIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lclswXSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgc3BlY2lmaWVkIGhpZXJhcmNoeSBib3VuZHMgbGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lcihIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lciBsKSB7Ci0gICAgICAgIGhpZXJhcmNoeUJvdW5kc0xpc3RlbmVycy5hZGRVc2VyTGlzdGVuZXIobCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgaGllcmFyY2h5IGJvdW5kcyBsaXN0ZW5lciByZWdpc3RlcmVkIGZvciB0aGlzIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbAotICAgICAqICAgICAgICAgICAgdGhlIEhpZXJhcmNoeUJvdW5kc0xpc3RlbmVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZUhpZXJhcmNoeUJvdW5kc0xpc3RlbmVyKEhpZXJhcmNoeUJvdW5kc0xpc3RlbmVyIGwpIHsKLSAgICAgICAgaGllcmFyY2h5Qm91bmRzTGlzdGVuZXJzLnJlbW92ZVVzZXJMaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgYSBoaWVyYXJjaHkgYm91bmRzIGV2ZW50IHRoYXQgaGFzIG9jY3VycmVkIG9uIHRoaXMgY29tcG9uZW50IGJ5Ci0gICAgICogZGlzcGF0Y2hpbmcgaXQgdG8gdGhlIHJlZ2lzdGVyZWQgbGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgSGllcmFyY2h5Qm91bmRzRXZlbnQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgcHJvY2Vzc0hpZXJhcmNoeUJvdW5kc0V2ZW50KEhpZXJhcmNoeUV2ZW50IGUpIHsKLSAgICAgICAgZm9yIChIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lciBsaXN0ZW5lciA6IGhpZXJhcmNoeUJvdW5kc0xpc3RlbmVycy5nZXRVc2VyTGlzdGVuZXJzKCkpIHsKLSAgICAgICAgICAgIHN3aXRjaCAoZS5nZXRJRCgpKSB7Ci0gICAgICAgICAgICAgICAgY2FzZSBIaWVyYXJjaHlFdmVudC5BTkNFU1RPUl9NT1ZFRDoKLSAgICAgICAgICAgICAgICAgICAgbGlzdGVuZXIuYW5jZXN0b3JNb3ZlZChlKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBIaWVyYXJjaHlFdmVudC5BTkNFU1RPUl9SRVNJWkVEOgotICAgICAgICAgICAgICAgICAgICBsaXN0ZW5lci5hbmNlc3RvclJlc2l6ZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiB0aGUga2V5IGxpc3RlbmVycyByZWdpc3RlcmVkIHRvIHRoZSBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiBhbiBhcnJheSBvZiB0aGUga2V5IGxpc3RlbmVycyByZWdpc3RlcmVkIHRvIHRoZSBDb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIEtleUxpc3RlbmVyW10gZ2V0S2V5TGlzdGVuZXJzKCkgewotICAgICAgICByZXR1cm4ga2V5TGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMobmV3IEtleUxpc3RlbmVyWzBdKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBzcGVjaWZpZWQga2V5IGxpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgS2V5TGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkS2V5TGlzdGVuZXIoS2V5TGlzdGVuZXIgbCkgewotICAgICAgICBrZXlMaXN0ZW5lcnMuYWRkVXNlckxpc3RlbmVyKGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgdGhlIGF3dCBrZXkgbGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBsLgotICAgICAqLwotICAgIHZvaWQgYWRkQVdUS2V5TGlzdGVuZXIoS2V5TGlzdGVuZXIgbCkgewotICAgICAgICBlbmFibGVBV1RFdmVudHMoQVdURXZlbnQuS0VZX0VWRU5UX01BU0spOwotICAgICAgICBrZXlMaXN0ZW5lcnMuYWRkU3lzdGVtTGlzdGVuZXIobCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUga2V5IGxpc3RlbmVyIHJlZ2lzdGVyZWQgZm9yIHRoaXMgQ29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgS2V5TGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlS2V5TGlzdGVuZXIoS2V5TGlzdGVuZXIgbCkgewotICAgICAgICBrZXlMaXN0ZW5lcnMucmVtb3ZlVXNlckxpc3RlbmVyKGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyBhIGtleSBldmVudCB0aGF0IGhhcyBvY2N1cnJlZCBvbiB0aGlzIGNvbXBvbmVudCBieSBkaXNwYXRjaGluZwotICAgICAqIGl0IHRvIHRoZSByZWdpc3RlcmVkIGxpc3RlbmVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIEtleUV2ZW50LgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NLZXlFdmVudChLZXlFdmVudCBlKSB7Ci0gICAgICAgIHByb2Nlc3NLZXlFdmVudEltcGwoZSwga2V5TGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2VzcyBrZXkgZXZlbnQgaW1wbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIGUuCi0gICAgICogQHBhcmFtIGMKLSAgICAgKiAgICAgICAgICAgIHRoZSBjLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBwcm9jZXNzS2V5RXZlbnRJbXBsKEtleUV2ZW50IGUsIENvbGxlY3Rpb248S2V5TGlzdGVuZXI+IGMpIHsKLSAgICAgICAgZm9yIChLZXlMaXN0ZW5lciBsaXN0ZW5lciA6IGMpIHsKLSAgICAgICAgICAgIHN3aXRjaCAoZS5nZXRJRCgpKSB7Ci0gICAgICAgICAgICAgICAgY2FzZSBLZXlFdmVudC5LRVlfUFJFU1NFRDoKLSAgICAgICAgICAgICAgICAgICAgbGlzdGVuZXIua2V5UHJlc3NlZChlKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBLZXlFdmVudC5LRVlfUkVMRUFTRUQ6Ci0gICAgICAgICAgICAgICAgICAgIGxpc3RlbmVyLmtleVJlbGVhc2VkKGUpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIEtleUV2ZW50LktFWV9UWVBFRDoKLSAgICAgICAgICAgICAgICAgICAgbGlzdGVuZXIua2V5VHlwZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiB0aGUgbW91c2UgbGlzdGVuZXJzIHJlZ2lzdGVyZWQgdG8gdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIHRoZSBtb3VzZSBsaXN0ZW5lcnMgcmVnaXN0ZXJlZCB0byB0aGUgQ29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBNb3VzZUxpc3RlbmVyW10gZ2V0TW91c2VMaXN0ZW5lcnMoKSB7Ci0gICAgICAgIHJldHVybiBtb3VzZUxpc3RlbmVycy5nZXRVc2VyTGlzdGVuZXJzKG5ldyBNb3VzZUxpc3RlbmVyWzBdKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBzcGVjaWZpZWQgbW91c2UgbGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBNb3VzZUxpc3RlbmVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGFkZE1vdXNlTGlzdGVuZXIoTW91c2VMaXN0ZW5lciBsKSB7Ci0gICAgICAgIG1vdXNlTGlzdGVuZXJzLmFkZFVzZXJMaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBhd3QgbW91c2UgbGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBsLgotICAgICAqLwotICAgIHZvaWQgYWRkQVdUTW91c2VMaXN0ZW5lcihNb3VzZUxpc3RlbmVyIGwpIHsKLSAgICAgICAgZW5hYmxlQVdURXZlbnRzKEFXVEV2ZW50Lk1PVVNFX0VWRU5UX01BU0spOwotICAgICAgICBtb3VzZUxpc3RlbmVycy5hZGRTeXN0ZW1MaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBhd3QgbW91c2UgbW90aW9uIGxpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgbC4KLSAgICAgKi8KLSAgICB2b2lkIGFkZEFXVE1vdXNlTW90aW9uTGlzdGVuZXIoTW91c2VNb3Rpb25MaXN0ZW5lciBsKSB7Ci0gICAgICAgIGVuYWJsZUFXVEV2ZW50cyhBV1RFdmVudC5NT1VTRV9NT1RJT05fRVZFTlRfTUFTSyk7Ci0gICAgICAgIG1vdXNlTW90aW9uTGlzdGVuZXJzLmFkZFN5c3RlbUxpc3RlbmVyKGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgdGhlIGF3dCBjb21wb25lbnQgbGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBsLgotICAgICAqLwotICAgIHZvaWQgYWRkQVdUQ29tcG9uZW50TGlzdGVuZXIoQ29tcG9uZW50TGlzdGVuZXIgbCkgewotICAgICAgICBlbmFibGVBV1RFdmVudHMoQVdURXZlbnQuQ09NUE9ORU5UX0VWRU5UX01BU0spOwotICAgICAgICBjb21wb25lbnRMaXN0ZW5lcnMuYWRkU3lzdGVtTGlzdGVuZXIobCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgYXd0IGlucHV0IG1ldGhvZCBsaXN0ZW5lci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbAotICAgICAqICAgICAgICAgICAgdGhlIGwuCi0gICAgICovCi0gICAgdm9pZCBhZGRBV1RJbnB1dE1ldGhvZExpc3RlbmVyKElucHV0TWV0aG9kTGlzdGVuZXIgbCkgewotICAgICAgICBlbmFibGVBV1RFdmVudHMoQVdURXZlbnQuSU5QVVRfTUVUSE9EX0VWRU5UX01BU0spOwotICAgICAgICBpbnB1dE1ldGhvZExpc3RlbmVycy5hZGRTeXN0ZW1MaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBhd3QgbW91c2Ugd2hlZWwgbGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBsLgotICAgICAqLwotICAgIHZvaWQgYWRkQVdUTW91c2VXaGVlbExpc3RlbmVyKE1vdXNlV2hlZWxMaXN0ZW5lciBsKSB7Ci0gICAgICAgIGVuYWJsZUFXVEV2ZW50cyhBV1RFdmVudC5NT1VTRV9XSEVFTF9FVkVOVF9NQVNLKTsKLSAgICAgICAgbW91c2VXaGVlbExpc3RlbmVycy5hZGRTeXN0ZW1MaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHRoZSBtb3VzZSBsaXN0ZW5lciByZWdpc3RlcmVkIGZvciB0aGlzIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbAotICAgICAqICAgICAgICAgICAgdGhlIE1vdXNlTGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlTW91c2VMaXN0ZW5lcihNb3VzZUxpc3RlbmVyIGwpIHsKLSAgICAgICAgbW91c2VMaXN0ZW5lcnMucmVtb3ZlVXNlckxpc3RlbmVyKGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyBhIG1vdXNlIGV2ZW50IHRoYXQgaGFzIG9jY3VycmVkIG9uIHRoaXMgY29tcG9uZW50IGJ5Ci0gICAgICogZGlzcGF0Y2hpbmcgaXQgdG8gdGhlIHJlZ2lzdGVyZWQgbGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgTW91c2VFdmVudC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzTW91c2VFdmVudChNb3VzZUV2ZW50IGUpIHsKLSAgICAgICAgcHJvY2Vzc01vdXNlRXZlbnRJbXBsKGUsIG1vdXNlTGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2VzcyBtb3VzZSBldmVudCBpbXBsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgZS4KLSAgICAgKiBAcGFyYW0gYwotICAgICAqICAgICAgICAgICAgdGhlIGMuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHByb2Nlc3NNb3VzZUV2ZW50SW1wbChNb3VzZUV2ZW50IGUsIENvbGxlY3Rpb248TW91c2VMaXN0ZW5lcj4gYykgewotICAgICAgICBmb3IgKE1vdXNlTGlzdGVuZXIgbGlzdGVuZXIgOiBjKSB7Ci0gICAgICAgICAgICBzd2l0Y2ggKGUuZ2V0SUQoKSkgewotICAgICAgICAgICAgICAgIGNhc2UgTW91c2VFdmVudC5NT1VTRV9DTElDS0VEOgotICAgICAgICAgICAgICAgICAgICBsaXN0ZW5lci5tb3VzZUNsaWNrZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgTW91c2VFdmVudC5NT1VTRV9FTlRFUkVEOgotICAgICAgICAgICAgICAgICAgICBsaXN0ZW5lci5tb3VzZUVudGVyZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgTW91c2VFdmVudC5NT1VTRV9FWElURUQ6Ci0gICAgICAgICAgICAgICAgICAgIGxpc3RlbmVyLm1vdXNlRXhpdGVkKGUpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIE1vdXNlRXZlbnQuTU9VU0VfUFJFU1NFRDoKLSAgICAgICAgICAgICAgICAgICAgbGlzdGVuZXIubW91c2VQcmVzc2VkKGUpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIE1vdXNlRXZlbnQuTU9VU0VfUkVMRUFTRUQ6Ci0gICAgICAgICAgICAgICAgICAgIGxpc3RlbmVyLm1vdXNlUmVsZWFzZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2VzcyBtb3VzZSBtb3Rpb24gZXZlbnQgaW1wbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIGUuCi0gICAgICogQHBhcmFtIGMKLSAgICAgKiAgICAgICAgICAgIHRoZSBjLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBwcm9jZXNzTW91c2VNb3Rpb25FdmVudEltcGwoTW91c2VFdmVudCBlLCBDb2xsZWN0aW9uPE1vdXNlTW90aW9uTGlzdGVuZXI+IGMpIHsKLSAgICAgICAgZm9yIChNb3VzZU1vdGlvbkxpc3RlbmVyIGxpc3RlbmVyIDogYykgewotICAgICAgICAgICAgc3dpdGNoIChlLmdldElEKCkpIHsKLSAgICAgICAgICAgICAgICBjYXNlIE1vdXNlRXZlbnQuTU9VU0VfRFJBR0dFRDoKLSAgICAgICAgICAgICAgICAgICAgbGlzdGVuZXIubW91c2VEcmFnZ2VkKGUpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIE1vdXNlRXZlbnQuTU9VU0VfTU9WRUQ6Ci0gICAgICAgICAgICAgICAgICAgIGxpc3RlbmVyLm1vdXNlTW92ZWQoZSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiB0aGUgbW91c2UgbW90aW9uIGxpc3RlbmVycyByZWdpc3RlcmVkIHRvIHRoZSBDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiBhbiBhcnJheSBvZiB0aGUgTW91c2VNb3Rpb25MaXN0ZW5lcnMgcmVnaXN0ZXJlZCB0byB0aGUgQ29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBNb3VzZU1vdGlvbkxpc3RlbmVyW10gZ2V0TW91c2VNb3Rpb25MaXN0ZW5lcnMoKSB7Ci0gICAgICAgIHJldHVybiBtb3VzZU1vdGlvbkxpc3RlbmVycy5nZXRVc2VyTGlzdGVuZXJzKG5ldyBNb3VzZU1vdGlvbkxpc3RlbmVyWzBdKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBzcGVjaWZpZWQgbW91c2UgbW90aW9uIGxpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgTW91c2VNb3Rpb25MaXN0ZW5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRNb3VzZU1vdGlvbkxpc3RlbmVyKE1vdXNlTW90aW9uTGlzdGVuZXIgbCkgewotICAgICAgICBtb3VzZU1vdGlvbkxpc3RlbmVycy5hZGRVc2VyTGlzdGVuZXIobCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgbW91c2UgbW90aW9uIGxpc3RlbmVyIHJlZ2lzdGVyZWQgZm9yIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgTW91c2VNb3Rpb25MaXN0ZW5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVNb3VzZU1vdGlvbkxpc3RlbmVyKE1vdXNlTW90aW9uTGlzdGVuZXIgbCkgewotICAgICAgICBtb3VzZU1vdGlvbkxpc3RlbmVycy5yZW1vdmVVc2VyTGlzdGVuZXIobCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIGEgbW91c2UgbW90aW9uIGV2ZW50IHRoYXQgaGFzIG9jY3VycmVkIG9uIHRoaXMgY29tcG9uZW50IGJ5Ci0gICAgICogZGlzcGF0Y2hpbmcgaXQgdG8gdGhlIHJlZ2lzdGVyZWQgbGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlCi0gICAgICogICAgICAgICAgICB0aGUgTW91c2VFdmVudC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzTW91c2VNb3Rpb25FdmVudChNb3VzZUV2ZW50IGUpIHsKLSAgICAgICAgcHJvY2Vzc01vdXNlTW90aW9uRXZlbnRJbXBsKGUsIG1vdXNlTW90aW9uTGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiB0aGUgbW91c2Ugd2hlZWwgbGlzdGVuZXJzIHJlZ2lzdGVyZWQgdG8gdGhlIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIHRoZSBNb3VzZVdoZWVsTGlzdGVuZXJzIHJlZ2lzdGVyZWQgdG8gdGhlIENvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgTW91c2VXaGVlbExpc3RlbmVyW10gZ2V0TW91c2VXaGVlbExpc3RlbmVycygpIHsKLSAgICAgICAgcmV0dXJuIG1vdXNlV2hlZWxMaXN0ZW5lcnMuZ2V0VXNlckxpc3RlbmVycyhuZXcgTW91c2VXaGVlbExpc3RlbmVyWzBdKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBzcGVjaWZpZWQgbW91c2Ugd2hlZWwgbGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBNb3VzZVdoZWVsTGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkTW91c2VXaGVlbExpc3RlbmVyKE1vdXNlV2hlZWxMaXN0ZW5lciBsKSB7Ci0gICAgICAgIG1vdXNlV2hlZWxMaXN0ZW5lcnMuYWRkVXNlckxpc3RlbmVyKGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIG1vdXNlIHdoZWVsIGxpc3RlbmVyIHJlZ2lzdGVyZWQgZm9yIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgTW91c2VXaGVlbExpc3RlbmVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZU1vdXNlV2hlZWxMaXN0ZW5lcihNb3VzZVdoZWVsTGlzdGVuZXIgbCkgewotICAgICAgICBtb3VzZVdoZWVsTGlzdGVuZXJzLnJlbW92ZVVzZXJMaXN0ZW5lcihsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgYSBtb3VzZSB3aGVlbCBldmVudCB0aGF0IGhhcyBvY2N1cnJlZCBvbiB0aGlzIGNvbXBvbmVudCBieQotICAgICAqIGRpc3BhdGNoaW5nIGl0IHRvIHRoZSByZWdpc3RlcmVkIGxpc3RlbmVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIE1vdXNlV2hlZWxFdmVudC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzTW91c2VXaGVlbEV2ZW50KE1vdXNlV2hlZWxFdmVudCBlKSB7Ci0gICAgICAgIHByb2Nlc3NNb3VzZVdoZWVsRXZlbnRJbXBsKGUsIG1vdXNlV2hlZWxMaXN0ZW5lcnMuZ2V0VXNlckxpc3RlbmVycygpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIHRoZSBJbnB1dE1ldGhvZExpc3RlbmVyIGxpc3RlbmVycyByZWdpc3RlcmVkIHRvIHRoZQotICAgICAqIENvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIHRoZSBJbnB1dE1ldGhvZExpc3RlbmVyIGxpc3RlbmVycyByZWdpc3RlcmVkIHRvIHRoZQotICAgICAqICAgICAgICAgQ29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBJbnB1dE1ldGhvZExpc3RlbmVyW10gZ2V0SW5wdXRNZXRob2RMaXN0ZW5lcnMoKSB7Ci0gICAgICAgIHJldHVybiBpbnB1dE1ldGhvZExpc3RlbmVycy5nZXRVc2VyTGlzdGVuZXJzKG5ldyBJbnB1dE1ldGhvZExpc3RlbmVyWzBdKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBzcGVjaWZpZWQgaW5wdXQgbWV0aG9kIGxpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgSW5wdXRNZXRob2RMaXN0ZW5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRJbnB1dE1ldGhvZExpc3RlbmVyKElucHV0TWV0aG9kTGlzdGVuZXIgbCkgewotICAgICAgICBpbnB1dE1ldGhvZExpc3RlbmVycy5hZGRVc2VyTGlzdGVuZXIobCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgaW5wdXQgbWV0aG9kIGxpc3RlbmVyIHJlZ2lzdGVyZWQgZm9yIHRoaXMgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgSW5wdXRNZXRob2RMaXN0ZW5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVJbnB1dE1ldGhvZExpc3RlbmVyKElucHV0TWV0aG9kTGlzdGVuZXIgbCkgewotICAgICAgICBpbnB1dE1ldGhvZExpc3RlbmVycy5yZW1vdmVVc2VyTGlzdGVuZXIobCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIGFuIGlucHV0IG1ldGhvZCBldmVudCB0aGF0IGhhcyBvY2N1cnJlZCBvbiB0aGlzIGNvbXBvbmVudCBieQotICAgICAqIGRpc3BhdGNoaW5nIGl0IHRvIHRoZSByZWdpc3RlcmVkIGxpc3RlbmVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIElucHV0TWV0aG9kRXZlbnQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgcHJvY2Vzc0lucHV0TWV0aG9kRXZlbnQoSW5wdXRNZXRob2RFdmVudCBlKSB7Ci0gICAgICAgIHByb2Nlc3NJbnB1dE1ldGhvZEV2ZW50SW1wbChlLCBpbnB1dE1ldGhvZExpc3RlbmVycy5nZXRVc2VyTGlzdGVuZXJzKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3MgaW5wdXQgbWV0aG9kIGV2ZW50IGltcGwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlLgotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgYy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgcHJvY2Vzc0lucHV0TWV0aG9kRXZlbnRJbXBsKElucHV0TWV0aG9kRXZlbnQgZSwgQ29sbGVjdGlvbjxJbnB1dE1ldGhvZExpc3RlbmVyPiBjKSB7Ci0gICAgICAgIGZvciAoSW5wdXRNZXRob2RMaXN0ZW5lciBsaXN0ZW5lciA6IGMpIHsKLSAgICAgICAgICAgIHN3aXRjaCAoZS5nZXRJRCgpKSB7Ci0gICAgICAgICAgICAgICAgY2FzZSBJbnB1dE1ldGhvZEV2ZW50LkNBUkVUX1BPU0lUSU9OX0NIQU5HRUQ6Ci0gICAgICAgICAgICAgICAgICAgIGxpc3RlbmVyLmNhcmV0UG9zaXRpb25DaGFuZ2VkKGUpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIElucHV0TWV0aG9kRXZlbnQuSU5QVVRfTUVUSE9EX1RFWFRfQ0hBTkdFRDoKLSAgICAgICAgICAgICAgICAgICAgbGlzdGVuZXIuaW5wdXRNZXRob2RUZXh0Q2hhbmdlZChlKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIHB1YmxpYyBQb2ludCBnZXRNb3VzZVBvc2l0aW9uKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsgUG9pbnQKLSAgICAgKiBhYnNQb2ludGVyUG9zID0gTW91c2VJbmZvLmdldFBvaW50ZXJJbmZvKCkuZ2V0TG9jYXRpb24oKTsgV2luZG93Ci0gICAgICogd2luVW5kZXJQdHIgPQotICAgICAqIHRvb2xraXQuZGlzcGF0Y2hlci5tb3VzZURpc3BhdGNoZXIuZmluZFdpbmRvd0F0KGFic1BvaW50ZXJQb3MpOyBQb2ludAotICAgICAqIHBvaW50ZXJQb3MgPSBNb3VzZURpc3BhdGNoZXIuY29udmVydFBvaW50KG51bGwsIGFic1BvaW50ZXJQb3MsCi0gICAgICogd2luVW5kZXJQdHIpOyBib29sZWFuIGlzVW5kZXJQb2ludGVyID0gZmFsc2U7IGlmICh3aW5VbmRlclB0ciA9PSBudWxsKSB7Ci0gICAgICogcmV0dXJuIG51bGw7IH0gaXNVbmRlclBvaW50ZXIgPSB3aW5VbmRlclB0ci5pc0NvbXBvbmVudEF0KHRoaXMsCi0gICAgICogcG9pbnRlclBvcyk7IGlmIChpc1VuZGVyUG9pbnRlcikgeyByZXR1cm4KLSAgICAgKiBNb3VzZURpc3BhdGNoZXIuY29udmVydFBvaW50KG51bGwsIGFic1BvaW50ZXJQb3MsIHRoaXMpOyB9IHJldHVybiBudWxsOyB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBTZXQgbmF0aXZlIGNhcmV0IGF0IHRoZSBnaXZlbiBwb3NpdGlvbiA8YnI+Ci0gICAgICogTm90ZTogdGhpcyBtZXRob2QgdGFrZXMgQVdUIGxvY2sgaW5zaWRlIGJlY2F1c2UgaXQgd2Fsa3MgdGhyb3VnaCB0aGUKLSAgICAgKiBjb21wb25lbnQgaGllcmFyY2h5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkuCi0gICAgICovCi0gICAgdm9pZCBzZXRDYXJldFBvcyhmaW5hbCBpbnQgeCwgZmluYWwgaW50IHkpIHsKLSAgICAgICAgUnVubmFibGUgciA9IG5ldyBSdW5uYWJsZSgpIHsKLSAgICAgICAgICAgIHB1YmxpYyB2b2lkIHJ1bigpIHsKLSAgICAgICAgICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgICAgICBzZXRDYXJldFBvc0ltcGwoeCwgeSk7Ci0gICAgICAgICAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH07Ci0gICAgICAgIGlmIChUaHJlYWQuY3VycmVudFRocmVhZCgpIGluc3RhbmNlb2YgRXZlbnREaXNwYXRjaFRocmVhZCkgewotICAgICAgICAgICAgci5ydW4oKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHRvb2xraXQuZ2V0U3lzdGVtRXZlbnRRdWV1ZUltcGwoKS5wb3N0RXZlbnQobmV3IEludm9jYXRpb25FdmVudCh0aGlzLCByKSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCBzaG91bGQgYmUgY2FsbGVkIG9ubHkgYXQgZXZlbnQgZGlzcGF0Y2ggdGhyZWFkLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkuCi0gICAgICovCi0gICAgdm9pZCBzZXRDYXJldFBvc0ltcGwoaW50IHgsIGludCB5KSB7Ci0gICAgICAgIENvbXBvbmVudCBjID0gdGhpczsKLSAgICAgICAgd2hpbGUgKChjICE9IG51bGwpICYmIGMuYmVoYXZpb3VyLmlzTGlnaHR3ZWlnaHQoKSkgewotICAgICAgICAgICAgeCArPSBjLng7Ci0gICAgICAgICAgICB5ICs9IGMueTsKLSAgICAgICAgICAgIC8vID8/P0FXVDogYyA9IGMuZ2V0UGFyZW50KCk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGMgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIC8vID8/P0FXVAotICAgICAgICAvKgotICAgICAgICAgKiBpZiAoYyBpbnN0YW5jZW9mIFdpbmRvdykgeyBJbnNldHMgaW5zZXRzID0gYy5nZXROYXRpdmVJbnNldHMoKTsgeCAtPQotICAgICAgICAgKiBpbnNldHMubGVmdDsgeSAtPSBpbnNldHMudG9wOyB9Ci0gICAgICAgICAqIHRvb2xraXQuZ2V0V2luZG93RmFjdG9yeSgpLnNldENhcmV0UG9zaXRpb24oeCwgeSk7Ci0gICAgICAgICAqLwotICAgIH0KLQotICAgIC8vIHRvIGJlIG92ZXJyaWRkZW4gaW4gc3RhbmRhcmQgY29tcG9uZW50cyBzdWNoIGFzIEJ1dHRvbiBhbmQgTGlzdAotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRlZmF1bHQgbWluaW11bSBzaXplLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRlZmF1bHQgbWluaW11bSBzaXplLgotICAgICAqLwotICAgIERpbWVuc2lvbiBnZXREZWZhdWx0TWluaW11bVNpemUoKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8vIHRvIGJlIG92ZXJyaWRkZW4gaW4gc3RhbmRhcmQgY29tcG9uZW50cyBzdWNoIGFzIEJ1dHRvbiBhbmQgTGlzdAotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRlZmF1bHQgcHJlZmVycmVkIHNpemUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGVmYXVsdCBwcmVmZXJyZWQgc2l6ZS4KLSAgICAgKi8KLSAgICBEaW1lbnNpb24gZ2V0RGVmYXVsdFByZWZlcnJlZFNpemUoKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8vIHRvIGJlIG92ZXJyaWRkZW4gaW4gc3RhbmRhcmQgY29tcG9uZW50cyBzdWNoIGFzIEJ1dHRvbiBhbmQgTGlzdAotICAgIC8qKgotICAgICAqIFJlc2V0IGRlZmF1bHQgc2l6ZS4KLSAgICAgKi8KLSAgICB2b2lkIHJlc2V0RGVmYXVsdFNpemUoKSB7Ci0gICAgfQotCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBDb21wb25lbnRCZWhhdmlvciBjcmVhdGVCZWhhdmlvcigpIHsgcmV0dXJuIG5ldyBMV0JlaGF2aW9yKHRoaXMpOyB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkZWZhdWx0IGJhY2tncm91bmQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGVmYXVsdCBiYWNrZ3JvdW5kLgotICAgICAqLwotICAgIENvbG9yIGdldERlZmF1bHRCYWNrZ3JvdW5kKCkgewotICAgICAgICAvLyA/Pz9BV1Q6IHJldHVybiBnZXRXaW5kb3dBbmNlc3RvcigpLmdldERlZmF1bHRCYWNrZ3JvdW5kKCk7Ci0gICAgICAgIHJldHVybiBnZXRCYWNrZ3JvdW5kKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGVmYXVsdCBmb3JlZ3JvdW5kLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRlZmF1bHQgZm9yZWdyb3VuZC4KLSAgICAgKi8KLSAgICBDb2xvciBnZXREZWZhdWx0Rm9yZWdyb3VuZCgpIHsKLSAgICAgICAgLy8gPz8/QVdUIHJldHVybiBnZXRXaW5kb3dBbmNlc3RvcigpLmdldERlZmF1bHRGb3JlZ3JvdW5kKCk7Ci0gICAgICAgIHJldHVybiBnZXRGb3JlZ3JvdW5kKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2FsbGVkIHdoZW4gbmF0aXZlIHJlc291cmNlIGZvciB0aGlzIGNvbXBvbmVudCBpcyBjcmVhdGVkIChmb3IKLSAgICAgKiBoZWF2eXdlaWdodHMgb25seSkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpbgotICAgICAqICAgICAgICAgICAgdGhlIHdpbi4KLSAgICAgKi8KLSAgICB2b2lkIG5hdGl2ZVdpbmRvd0NyZWF0ZWQoTmF0aXZlV2luZG93IHdpbikgewotICAgICAgICAvLyB0byBiZSBvdmVycmlkZGVuCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGV0ZXJtaW5lIHRoZSBjb21wb25lbnQncyBhcmVhIGhpZGRlbiBiZWhpbmQgdGhlIHdpbmRvd3MgdGhhdCBoYXZlIGhpZ2hlcgotICAgICAqIFotb3JkZXIsIGluY2x1ZGluZyB3aW5kb3dzIG9mIG90aGVyIGFwcGxpY2F0aW9ucy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gZGVzdExvY2F0aW9uCi0gICAgICogICAgICAgICAgICB0aGUgZGVzdCBsb2NhdGlvbi4KLSAgICAgKiBAcGFyYW0gZGVzdFNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXN0IHNpemUuCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBjYWxjdWxhdGVkIHJlZ2lvbiwgb3IgbnVsbCBpZiBpdCBjYW5ub3QgYmUgZGV0ZXJtaW5lZC4KLSAgICAgKi8KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIE11bHRpUmVjdEFyZWEgZ2V0T2JzY3VyZWRSZWdpb24oUmVjdGFuZ2xlIHBhcnQpIHsgaWYgKCF2aXNpYmxlIHx8IHBhcmVudAotICAgICAqID09IG51bGwgfHwgIXBhcmVudC52aXNpYmxlKSB7IHJldHVybiBudWxsOyB9IFJlY3RhbmdsZSByID0gbmV3Ci0gICAgICogUmVjdGFuZ2xlKDAsIDAsIHcsIGgpOyBpZiAocGFydCAhPSBudWxsKSB7IHIgPSByLmludGVyc2VjdGlvbihwYXJ0KTsgfSBpZgotICAgICAqIChyLmlzRW1wdHkoKSkgeyByZXR1cm4gbnVsbDsgfSByLnRyYW5zbGF0ZSh4LCB5KTsgTXVsdGlSZWN0QXJlYSByZXQgPQotICAgICAqIHBhcmVudC5nZXRPYnNjdXJlZFJlZ2lvbihyKTsgaWYgKHJldCAhPSBudWxsKSB7Ci0gICAgICogcGFyZW50LmFkZE9ic2N1cmVkUmVnaW9ucyhyZXQsIHRoaXMpOyByZXQudHJhbnNsYXRlKC14LCAteSk7Ci0gICAgICogcmV0LmludGVyc2VjdChuZXcgUmVjdGFuZ2xlKDAsIDAsIHcsIGgpKTsgfSByZXR1cm4gcmV0OyB9Ci0gICAgICovCi0KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIHByaXZhdGUgdm9pZCByZWFkT2JqZWN0KE9iamVjdElucHV0U3RyZWFtIHN0cmVhbSkgdGhyb3dzIElPRXhjZXB0aW9uLAotICAgICAqIENsYXNzTm90Rm91bmRFeGNlcHRpb24geyBzdHJlYW0uZGVmYXVsdFJlYWRPYmplY3QoKTsgRmllbGRzQWNjZXNzb3IKLSAgICAgKiBhY2Nlc3NvciA9IG5ldyBGaWVsZHNBY2Nlc3NvcihDb21wb25lbnQuY2xhc3MsIHRoaXMpOwotICAgICAqIGFjY2Vzc29yLnNldCgidG9vbGtpdCIsIFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgKiBhY2Nlc3Nvci5zZXQoImJlaGF2aW91ciIsIGNyZWF0ZUJlaGF2aW9yKCkpOyAvLyROT04tTkxTLTEkCi0gICAgICogYWNjZXNzb3Iuc2V0KCJjb21wb25lbnRMb2NrIiwgbmV3IE9iamVjdCgpKTsgLy8gJE5PTi1MT0NLLTEkCi0gICAgICogLy8kTk9OLU5MUy0xJCB9Ci0gICAgICovCi0KLSAgICBmaW5hbCB2b2lkIG9uRHJhd0ltYWdlKEltYWdlIGltYWdlLCBQb2ludCBkZXN0TG9jYXRpb24sIERpbWVuc2lvbiBkZXN0U2l6ZSwgUmVjdGFuZ2xlIHNvdXJjZSkgewotICAgICAgICBJbWFnZVBhcmFtZXRlcnMgaW1hZ2VQYXJhbXM7Ci0gICAgICAgIGlmICh1cGRhdGVkSW1hZ2VzID09IG51bGwpIHsKLSAgICAgICAgICAgIHVwZGF0ZWRJbWFnZXMgPSBuZXcgSGFzaE1hcDxJbWFnZSwgSW1hZ2VQYXJhbWV0ZXJzPigpOwotICAgICAgICB9Ci0gICAgICAgIGltYWdlUGFyYW1zID0gdXBkYXRlZEltYWdlcy5nZXQoaW1hZ2UpOwotICAgICAgICBpZiAoaW1hZ2VQYXJhbXMgPT0gbnVsbCkgewotICAgICAgICAgICAgaW1hZ2VQYXJhbXMgPSBuZXcgSW1hZ2VQYXJhbWV0ZXJzKCk7Ci0gICAgICAgICAgICB1cGRhdGVkSW1hZ2VzLnB1dChpbWFnZSwgaW1hZ2VQYXJhbXMpOwotICAgICAgICB9Ci0gICAgICAgIGltYWdlUGFyYW1zLmFkZERyYXdpbmcoZGVzdExvY2F0aW9uLCBkZXN0U2l6ZSwgc291cmNlKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpbWFnZVVwZGF0ZShJbWFnZSBpbWcsIGludCBpbmZvZmxhZ3MsIGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgYm9vbGVhbiBkb25lID0gZmFsc2U7Ci0gICAgICAgICAgICBpZiAoKGluZm9mbGFncyAmIChBTExCSVRTIHwgRlJBTUVCSVRTKSkgIT0gMCkgewotICAgICAgICAgICAgICAgIGRvbmUgPSB0cnVlOwotICAgICAgICAgICAgfSBlbHNlIGlmICgoaW5mb2ZsYWdzICYgU09NRUJJVFMpICE9IDAgJiYgaW5jcmVtZW50YWxJbWFnZVVwZGF0ZSkgewotICAgICAgICAgICAgICAgIGRvbmUgPSB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGRvbmUpIHsKLSAgICAgICAgICAgICAgICByZXBhaW50KCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gKGluZm9mbGFncyAmIChBQk9SVCB8IEFMTEJJVFMpKSA9PSAwOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogcHJpdmF0ZSB2b2lkIGludmFsaWRhdGVSZWFsUGFyZW50KCkgeyBDb250YWluZXIgcmVhbFBhcmVudCA9Ci0gICAgICogZ2V0UmVhbFBhcmVudCgpOyBpZiAoKHJlYWxQYXJlbnQgIT0gbnVsbCkgJiYgcmVhbFBhcmVudC5pc1ZhbGlkKCkpIHsKLSAgICAgKiByZWFsUGFyZW50LmludmFsaWRhdGUoKTsgfSB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgSW1hZ2VQYXJhbWV0ZXJzLgotICAgICAqLwotICAgIHByaXZhdGUgY2xhc3MgSW1hZ2VQYXJhbWV0ZXJzIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGRyYXdpbmcgcGFyYW1zLgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBmaW5hbCBMaW5rZWRMaXN0PERyYXdpbmdQYXJhbWV0ZXJzPiBkcmF3aW5nUGFyYW1zID0gbmV3IExpbmtlZExpc3Q8RHJhd2luZ1BhcmFtZXRlcnM+KCk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzaXplLgotICAgICAgICAgKi8KLSAgICAgICAgRGltZW5zaW9uIHNpemUgPSBuZXcgRGltZW5zaW9uKENvbXBvbmVudC50aGlzLncsIENvbXBvbmVudC50aGlzLmgpOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBBZGRzIHRoZSBkcmF3aW5nLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGRlc3RMb2NhdGlvbgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBkZXN0IGxvY2F0aW9uLgotICAgICAgICAgKiBAcGFyYW0gZGVzdFNpemUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgZGVzdCBzaXplLgotICAgICAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZS4KLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgYWRkRHJhd2luZyhQb2ludCBkZXN0TG9jYXRpb24sIERpbWVuc2lvbiBkZXN0U2l6ZSwgUmVjdGFuZ2xlIHNvdXJjZSkgewotICAgICAgICAgICAgZHJhd2luZ1BhcmFtcy5hZGQobmV3IERyYXdpbmdQYXJhbWV0ZXJzKGRlc3RMb2NhdGlvbiwgZGVzdFNpemUsIHNvdXJjZSkpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIERyYXdpbmcgcGFyYW1ldGVycyBpdGVyYXRvci4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdGhlIGl0ZXJhdG9yPCBkcmF3aW5nIHBhcmFtZXRlcnM+LgotICAgICAgICAgKi8KLSAgICAgICAgSXRlcmF0b3I8RHJhd2luZ1BhcmFtZXRlcnM+IGRyYXdpbmdQYXJhbWV0ZXJzSXRlcmF0b3IoKSB7Ci0gICAgICAgICAgICByZXR1cm4gZHJhd2luZ1BhcmFtcy5pdGVyYXRvcigpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBDbGFzcyBEcmF3aW5nUGFyYW1ldGVycy4KLSAgICAgICAgICovCi0gICAgICAgIGNsYXNzIERyYXdpbmdQYXJhbWV0ZXJzIHsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBUaGUgZGVzdCBsb2NhdGlvbi4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgUG9pbnQgZGVzdExvY2F0aW9uOwotCi0gICAgICAgICAgICAvKioKLSAgICAgICAgICAgICAqIFRoZSBkZXN0IHNpemUuCi0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIERpbWVuc2lvbiBkZXN0U2l6ZTsKLQotICAgICAgICAgICAgLyoqCi0gICAgICAgICAgICAgKiBUaGUgc291cmNlLgotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBSZWN0YW5nbGUgc291cmNlOwotCi0gICAgICAgICAgICAvKioKLSAgICAgICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkcmF3aW5nIHBhcmFtZXRlcnMuCi0gICAgICAgICAgICAgKiAKLSAgICAgICAgICAgICAqIEBwYXJhbSBkZXN0TG9jYXRpb24KLSAgICAgICAgICAgICAqICAgICAgICAgICAgdGhlIGRlc3QgbG9jYXRpb24uCi0gICAgICAgICAgICAgKiBAcGFyYW0gZGVzdFNpemUKLSAgICAgICAgICAgICAqICAgICAgICAgICAgdGhlIGRlc3Qgc2l6ZS4KLSAgICAgICAgICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgICAgICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZS4KLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgRHJhd2luZ1BhcmFtZXRlcnMoUG9pbnQgZGVzdExvY2F0aW9uLCBEaW1lbnNpb24gZGVzdFNpemUsIFJlY3RhbmdsZSBzb3VyY2UpIHsKLSAgICAgICAgICAgICAgICB0aGlzLmRlc3RMb2NhdGlvbiA9IG5ldyBQb2ludChkZXN0TG9jYXRpb24pOwotICAgICAgICAgICAgICAgIGlmIChkZXN0U2l6ZSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMuZGVzdFNpemUgPSBuZXcgRGltZW5zaW9uKGRlc3RTaXplKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICB0aGlzLmRlc3RTaXplID0gbnVsbDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKHNvdXJjZSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMuc291cmNlID0gbmV3IFJlY3RhbmdsZShzb3VyY2UpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMuc291cmNlID0gbnVsbDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUZXh0Q29tcG9uZW50IHN1cHBvcnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgZGlzcGF0Y2ggZXZlbnQgdG8gaW0uCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBwcml2YXRlIFRleHRLaXQgdGV4dEtpdCA9IG51bGw7IFRleHRLaXQgZ2V0VGV4dEtpdCgpIHsgcmV0dXJuIHRleHRLaXQ7IH0KLSAgICAgKiB2b2lkIHNldFRleHRLaXQoVGV4dEtpdCBraXQpIHsgdGV4dEtpdCA9IGtpdDsgfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogVGV4dEZpZWxkIHN1cHBvcnQuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBwcml2YXRlIFRleHRGaWVsZEtpdCB0ZXh0RmllbGRLaXQgPSBudWxsOyBUZXh0RmllbGRLaXQgZ2V0VGV4dEZpZWxkS2l0KCkKLSAgICAgKiB7IHJldHVybiB0ZXh0RmllbGRLaXQ7IH0gdm9pZCBzZXRUZXh0RmllbGRLaXQoVGV4dEZpZWxkS2l0IGtpdCkgewotICAgICAqIHRleHRGaWVsZEtpdCA9IGtpdDsgfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogRGlzcGF0Y2hlcyBpbnB1dCAmIGZvY3VzIGV2ZW50cyB0byBpbnB1dCBtZXRob2QgY29udGV4dC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgZXZlbnQgdG8gcGFzcyB0byBJbnB1dENvbnRleHQuZGlzcGF0Y2hFdmVudCgpLgotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiBldmVudCB3YXMgY29uc3VtZWQgYnkgSU0sIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gZGlzcGF0Y2hFdmVudFRvSU0oQVdURXZlbnQgZSkgewotICAgICAgICBJbnB1dENvbnRleHQgaWMgPSBnZXRJbnB1dENvbnRleHQoKTsKLSAgICAgICAgaWYgKGljID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgICAgICBpbnQgaWQgPSBlLmdldElEKCk7Ci0gICAgICAgIGJvb2xlYW4gaXNJbnB1dEV2ZW50ID0gKChpZCA+PSBLZXlFdmVudC5LRVlfRklSU1QpICYmIChpZCA8PSBLZXlFdmVudC5LRVlfTEFTVCkpCi0gICAgICAgICAgICAgICAgfHwgKChpZCA+PSBNb3VzZUV2ZW50Lk1PVVNFX0ZJUlNUKSAmJiAoaWQgPD0gTW91c2VFdmVudC5NT1VTRV9MQVNUKSk7Ci0gICAgICAgIGlmICgoKGlkID49IEZvY3VzRXZlbnQuRk9DVVNfRklSU1QpICYmIChpZCA8PSBGb2N1c0V2ZW50LkZPQ1VTX0xBU1QpKSB8fCBpc0lucHV0RXZlbnQpIHsKLSAgICAgICAgICAgIGljLmRpc3BhdGNoRXZlbnQoZSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGUuaXNDb25zdW1lZCgpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9Db21wb25lbnRCZWhhdmlvci5qYXZhIGIvYXd0L2phdmEvYXd0L0NvbXBvbmVudEJlaGF2aW9yLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGY0ZThmZmIuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0NvbXBvbmVudEJlaGF2aW9yLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERtaXRyeSBBLiBEdXJuZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLk5hdGl2ZVdpbmRvdzsKLQotLyoqCi0gKiBUaGUgaW50ZXJmYWNlIG9mIHRoZSBoZWxwZXIgb2JqZWN0IHRoYXQgZW5jYXBzdWxhdGVzIHRoZSBkaWZmZXJlbmNlCi0gKiBiZXR3ZWVuIGxpZ2h0d2VpZ2h0IGFuZCBoZWF2eXdlaWdodCBjb21wb25lbnRzLgotICovCi1pbnRlcmZhY2UgQ29tcG9uZW50QmVoYXZpb3IgewotCi0gICAgdm9pZCBhZGROb3RpZnkoKTsKLQotICAgIHZvaWQgc2V0Qm91bmRzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnQgYk1hc2spOwotCi0gICAgdm9pZCBzZXRWaXNpYmxlKGJvb2xlYW4gYik7Ci0KLSAgICBHcmFwaGljcyBnZXRHcmFwaGljcyhpbnQgdHJhbnNsYXRpb25YLCBpbnQgdHJhbnNsYXRpb25ZLCBpbnQgd2lkdGgsIGludCBoZWlnaHQpOwotCi0gICAgTmF0aXZlV2luZG93IGdldE5hdGl2ZVdpbmRvdygpOwotCi0gICAgYm9vbGVhbiBpc0xpZ2h0d2VpZ2h0KCk7Ci0KLSAgICB2b2lkIG9uTW92ZShpbnQgeCwgaW50IHkpOwotCi0gICAgYm9vbGVhbiBpc09wYXF1ZSgpOwotCi0gICAgYm9vbGVhbiBpc0Rpc3BsYXlhYmxlKCk7Ci0KLSAgICB2b2lkIHNldEVuYWJsZWQoYm9vbGVhbiB2YWx1ZSk7Ci0KLSAgICB2b2lkIHJlbW92ZU5vdGlmeSgpOwotCi0gICAgdm9pZCBzZXRaT3JkZXIoaW50IG5ld0luZGV4LCBpbnQgb2xkSW5kZXgpOwotCi0gICAgYm9vbGVhbiBzZXRGb2N1cyhib29sZWFuIGZvY3VzLCBDb21wb25lbnQgb3Bwb3NpdGUpOwotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0NvbXBvbmVudE9yaWVudGF0aW9uLmphdmEgYi9hd3QvamF2YS9hd3QvQ29tcG9uZW50T3JpZW50YXRpb24uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNWFjYzExYS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvQ29tcG9uZW50T3JpZW50YXRpb24uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE1NCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdiwgRG1pdHJ5IEEuIER1cm5ldgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmlvLlNlcmlhbGl6YWJsZTsKLWltcG9ydCBqYXZhLnV0aWwuKjsKLQotLyoqCi0gKiBUaGUgQ29tcG9uZW50T3JpZW50YXRpb24gY2xhc3Mgc3BlY2lmaWVzIHRoZSBsYW5ndWFnZS1zZW5zaXRpdmUgb3JpZW50YXRpb24KLSAqIG9mIGNvbXBvbmVudCdzIGVsZW1lbnRzIG9yIHRleHQuIEl0IGlzIHVzZWQgdG8gcmVmbGVjdCB0aGUgZGlmZmVyZW5jZXMgaW4KLSAqIHRoaXMgb3JkZXJpbmcgYmV0d2VlbiBkaWZmZXJlbnQgd3JpdGluZyBzeXN0ZW1zLiBUaGUgQ29tcG9uZW50T3JpZW50YXRpb24KLSAqIGNsYXNzIGluZGljYXRlcyB0aGUgb3JpZW50YXRpb24gb2YgdGhlIGVsZW1lbnRzL3RleHQgaW4gdGhlIGhvcml6b250YWwKLSAqIGRpcmVjdGlvbiAoImxlZnQgdG8gcmlnaHQiIG9yICJyaWdodCB0byBsZWZ0IikgYW5kIGluIHRoZSB2ZXJ0aWNhbCBkaXJlY3Rpb24KLSAqICgidG9wIHRvIGJvdHRvbSIgb3IgImJvdHRvbSB0byB0b3AiKS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBmaW5hbCBjbGFzcyBDb21wb25lbnRPcmllbnRhdGlvbiBpbXBsZW1lbnRzIFNlcmlhbGl6YWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtNDExMzI5MTM5MjE0MzU2MzgyOEw7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTEVGVF9UT19SSUdIVCBpbmRpY2F0ZXMgdGhhdCBpdGVtcyBydW4gbGVmdCB0byByaWdodC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbXBvbmVudE9yaWVudGF0aW9uIExFRlRfVE9fUklHSFQgPSBuZXcgQ29tcG9uZW50T3JpZW50YXRpb24odHJ1ZSwgdHJ1ZSk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUklHSFRfVE9fTEVGVCBpbmRpY2F0ZXMgdGhhdCBpdGVtcyBydW4gcmlnaHQgdG8gbGVmdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENvbXBvbmVudE9yaWVudGF0aW9uIFJJR0hUX1RPX0xFRlQgPSBuZXcgQ29tcG9uZW50T3JpZW50YXRpb24odHJ1ZSwgZmFsc2UpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFVOS05PV04gaW5kaWNhdGVzIHRoYXQgYSBjb21wb25lbnQncyBvcmllbnRhdGlvbiBpcyBub3Qgc2V0LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQ29tcG9uZW50T3JpZW50YXRpb24gVU5LTk9XTiA9IG5ldyBDb21wb25lbnRPcmllbnRhdGlvbih0cnVlLCB0cnVlKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBybExhbmdzLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFNldDxTdHJpbmc+IHJsTGFuZ3MgPSBuZXcgSGFzaFNldDxTdHJpbmc+KCk7IC8vIFJJR0hUX1RPX0xFRlQKLQotICAgIC8vIGxhbmd1YWdlcwotCi0gICAgLyoqCi0gICAgICogVGhlIGhvcml6b250YWwuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBib29sZWFuIGhvcml6b250YWw7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbGVmdDJyaWdodC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIGJvb2xlYW4gbGVmdDJyaWdodDsKLQotICAgIHN0YXRpYyB7Ci0gICAgICAgIHJsTGFuZ3MuYWRkKCJhciIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIHJsTGFuZ3MuYWRkKCJmYSIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIHJsTGFuZ3MuYWRkKCJpdyIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIHJsTGFuZ3MuYWRkKCJ1ciIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgb3JpZW50YXRpb24gZm9yIHRoZSBnaXZlbiBSZXNvdXJjZUJ1bmRsZSdzIGxvY2FsaXphdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmRsCi0gICAgICogICAgICAgICAgICB0aGUgUmVzb3VyY2VCdW5kbGUuCi0gICAgICogQHJldHVybiB0aGUgQ29tcG9uZW50T3JpZW50YXRpb24uCi0gICAgICogQGRlcHJlY2F0ZWQgVXNlIGdldE9yaWVudGF0aW9uKGphdmEudXRpbC5Mb2NhbGUpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBzdGF0aWMgQ29tcG9uZW50T3JpZW50YXRpb24gZ2V0T3JpZW50YXRpb24oUmVzb3VyY2VCdW5kbGUgYmRsKSB7Ci0gICAgICAgIE9iamVjdCBvYmogPSBudWxsOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgb2JqID0gYmRsLmdldE9iamVjdCgiT3JpZW50YXRpb24iKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9IGNhdGNoIChNaXNzaW5nUmVzb3VyY2VFeGNlcHRpb24gbXJlKSB7Ci0gICAgICAgICAgICBvYmogPSBudWxsOwotICAgICAgICB9Ci0gICAgICAgIGlmIChvYmogaW5zdGFuY2VvZiBDb21wb25lbnRPcmllbnRhdGlvbikgewotICAgICAgICAgICAgcmV0dXJuIChDb21wb25lbnRPcmllbnRhdGlvbilvYmo7Ci0gICAgICAgIH0KLSAgICAgICAgTG9jYWxlIGxvY2FsZSA9IGJkbC5nZXRMb2NhbGUoKTsKLSAgICAgICAgaWYgKGxvY2FsZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBsb2NhbGUgPSBMb2NhbGUuZ2V0RGVmYXVsdCgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBnZXRPcmllbnRhdGlvbihsb2NhbGUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG9yaWVudGF0aW9uIGZvciB0aGUgc3BlY2lmaWVkIGxvY2FsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbG9jYWxlCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIExvY2FsZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBDb21wb25lbnRPcmllbnRhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIENvbXBvbmVudE9yaWVudGF0aW9uIGdldE9yaWVudGF0aW9uKExvY2FsZSBsb2NhbGUpIHsKLSAgICAgICAgU3RyaW5nIGxhbmcgPSBsb2NhbGUuZ2V0TGFuZ3VhZ2UoKTsKLSAgICAgICAgcmV0dXJuIHJsTGFuZ3MuY29udGFpbnMobGFuZykgPyBSSUdIVF9UT19MRUZUIDogTEVGVF9UT19SSUdIVDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgY29tcG9uZW50IG9yaWVudGF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBob3IKLSAgICAgKiAgICAgICAgICAgIHdoZXRoZXIgdGhlIGl0ZW1zIHNob3VsZCBiZSBhcnJhbmdlZCBob3Jpem9udGFsbHkuCi0gICAgICogQHBhcmFtIGwycgotICAgICAqICAgICAgICAgICAgd2hldGhlciB0aGlzIG9yaWVudGF0aW9uIHNwZWNpZmllcyBhIGxlZnQtdG8tcmlnaHQgZmxvdy4KLSAgICAgKi8KLSAgICBwcml2YXRlIENvbXBvbmVudE9yaWVudGF0aW9uKGJvb2xlYW4gaG9yLCBib29sZWFuIGwycikgewotICAgICAgICBob3Jpem9udGFsID0gaG9yOwotICAgICAgICBsZWZ0MnJpZ2h0ID0gbDJyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgdGV4dCBvZiB0aGUgb2Ygd3JpdGluZyBzeXN0ZW1zIGFycmFuZ2VkIGhvcml6b250YWxseS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSB0ZXh0IGlzIHdyaXR0ZW4gaG9yaXpvbnRhbGx5LCBmYWxzZSBmb3IgYSB2ZXJ0aWNhbAotICAgICAqICAgICAgICAgYXJyYW5nZW1lbnQuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNIb3Jpem9udGFsKCkgewotICAgICAgICByZXR1cm4gaG9yaXpvbnRhbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIHRleHQgaXMgYXJyYW5nZWQgZnJvbSBsZWZ0IHRvIHJpZ2h0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgZm9yIHdyaXRpbmcgc3lzdGVtcyB3cml0dGVuIGZyb20gbGVmdCB0byByaWdodDsgZmFsc2UgZm9yCi0gICAgICogICAgICAgICByaWdodC10by1sZWZ0LgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzTGVmdFRvUmlnaHQoKSB7Ci0gICAgICAgIHJldHVybiBsZWZ0MnJpZ2h0OwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0NvbXBvc2l0ZS5qYXZhIGIvYXd0L2phdmEvYXd0L0NvbXBvc2l0ZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkMTczMGZlLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9Db21wb3NpdGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDUxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLQotLyoqCi0gKiBUaGUgQ29tcG9zaXRlIGludGVyZmFjZSBhbGxvd3MgdGhlIG1ldGhvZHMgdG8gY29tcG9zZSBhIGRyYXcgcHJpbWl0aXZlIG9uIHRoZQotICogZ3JhcGhpY3MgYXJlYS4gVGhlIGNsYXNzZXMgaW1wbGVtZW50aW5nIHRoaXMgaW50ZXJmYWNlIHByb3ZpZGVzIHRoZSBydWxlcyBhbmQKLSAqIGEgbWV0aG9kIHRvIGNyZWF0ZSB0aGUgY29udGV4dCBmb3IgYSBwYXJ0aWN1bGFyIG9wZXJhdGlvbi4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgQ29tcG9zaXRlIHsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBDb21wb3NpdGVDb250ZXh0IHdoaWNoIGRlZmluZXMgdGhlIGVuY2Fwc3VsYXRlZCBhbmQgb3B0aW1pemVkCi0gICAgICogZW52aXJvbm1lbnQgZm9yIGEgY29tcG9zaXRpbmcgb3BlcmF0aW9uLiBTZXZlcmFsIGNvbnRleHRzIGNhbiBleGlzdCBmb3IgYQotICAgICAqIHNpbmdsZSBDb21wb3NpdGUgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmNDb2xvck1vZGVsCi0gICAgICogICAgICAgICAgICB0aGUgc291cmNlJ3MgQ29sb3JNb2RlbC4KLSAgICAgKiBAcGFyYW0gZHN0Q29sb3JNb2RlbAotICAgICAqICAgICAgICAgICAgdGhlIGRlc3RpbmF0aW9uJ3MgQ29sb3JNb2RlbC4KLSAgICAgKiBAcGFyYW0gaGludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJpbmdIaW50cy4KLSAgICAgKiBAcmV0dXJuIHRoZSBDb21wb3NpdGVDb250ZXh0IG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29tcG9zaXRlQ29udGV4dCBjcmVhdGVDb250ZXh0KENvbG9yTW9kZWwgc3JjQ29sb3JNb2RlbCwgQ29sb3JNb2RlbCBkc3RDb2xvck1vZGVsLAotICAgICAgICAgICAgUmVuZGVyaW5nSGludHMgaGludHMpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvQ29tcG9zaXRlQ29udGV4dC5qYXZhIGIvYXd0L2phdmEvYXd0L0NvbXBvc2l0ZUNvbnRleHQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzk1NjQwZC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvQ29tcG9zaXRlQ29udGV4dC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5SYXN0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuV3JpdGFibGVSYXN0ZXI7Ci0KLS8qKgotICogVGhlIENvbXBvc2l0ZUNvbnRleHQgaW50ZXJmYWNlIHNwZWNpZmllcyB0aGUgZW5jYXBzdWxhdGVkIGFuZCBvcHRpbWl6ZWQKLSAqIGVudmlyb25tZW50IGZvciBhIGNvbXBvc2l0aW5nIG9wZXJhdGlvbi4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgQ29tcG9zaXRlQ29udGV4dCB7Ci0KLSAgICAvKioKLSAgICAgKiBDb21wb3NlcyB0aGUgdHdvIHNvdXJjZSBSYXN0ZXIgb2JqZWN0cyBhbmQgcGxhY2VzIHRoZSByZXN1bHQgaW4gdGhlCi0gICAgICogZGVzdGluYXRpb24gV3JpdGFibGVSYXN0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBSYXN0ZXIuCi0gICAgICogQHBhcmFtIGRzdEluCi0gICAgICogICAgICAgICAgICB0aGUgZGVzdGluYXRpb24gUmFzdGVyLgotICAgICAqIEBwYXJhbSBkc3RPdXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBXcml0YWJsZVJhc3RlciBvYmplY3Qgd2hlcmUgdGhlIHJlc3VsdCBvZiBjb21wb3NpbmcKLSAgICAgKiAgICAgICAgICAgIG9wZXJhdGlvbiBpcyBzdG9yZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgY29tcG9zZShSYXN0ZXIgc3JjLCBSYXN0ZXIgZHN0SW4sIFdyaXRhYmxlUmFzdGVyIGRzdE91dCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZWxlYXNlcyByZXNvdXJjZXMgYWxsb2NhdGVkIGZvciBhIGNvbnRleHQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZGlzcG9zZSgpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvQ3Vyc29yLmphdmEgYi9hd3QvamF2YS9hd3QvQ3Vyc29yLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBhMGNjODQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0N1cnNvci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDI3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRG1pdHJ5IEEuIER1cm5ldgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmlvLkZpbGU7Ci1pbXBvcnQgamF2YS5pby5GaWxlSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5GaWxlTm90Rm91bmRFeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLQotaW1wb3J0IGphdmEuaW8uU2VyaWFsaXphYmxlOwotaW1wb3J0IGphdmEudXRpbC5IYXNoTWFwOwotaW1wb3J0IGphdmEudXRpbC5NYXA7Ci1pbXBvcnQgamF2YS51dGlsLlByb3BlcnRpZXM7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5OYXRpdmVDdXJzb3I7Ci0KLS8qKgotICogVGhlIEN1cnNvciBjbGFzcyByZXByZXNlbnRzIHRoZSBiaXRtYXAgb2YgdGhlIG1vdXNlIGN1cnNvci4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBDdXJzb3IgaW1wbGVtZW50cyBTZXJpYWxpemFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IHNlcmlhbFZlcnNpb25VSUQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gODAyODIzNzQ5NzU2ODk4NTUwNEw7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgREVGQVVMVF9DVVJTT1IgaW5kaWNhdGVzIHRoZSBkZWZhdWx0IGN1cnNvciB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IERFRkFVTFRfQ1VSU09SID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDUk9TU0hBSVJfQ1VSU09SIGN1cnNvciB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENST1NTSEFJUl9DVVJTT1IgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRFWFRfQ1VSU09SIGN1cnNvciB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRFWFRfQ1VSU09SID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBXQUlUX0NVUlNPUiBjdXJzb3IgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXQUlUX0NVUlNPUiA9IDM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU1dfUkVTSVpFX0NVUlNPUiBjdXJzb3IgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTV19SRVNJWkVfQ1VSU09SID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTRV9SRVNJWkVfQ1VSU09SIGN1cnNvciB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNFX1JFU0laRV9DVVJTT1IgPSA1OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE5XX1JFU0laRV9DVVJTT1IgY3Vyc29yIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTldfUkVTSVpFX0NVUlNPUiA9IDY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTkVfUkVTSVpFX0NVUlNPUiBjdXJzb3IgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBORV9SRVNJWkVfQ1VSU09SID0gNzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBOX1JFU0laRV9DVVJTT1IgY3Vyc29yIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTl9SRVNJWkVfQ1VSU09SID0gODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTX1JFU0laRV9DVVJTT1IgY3Vyc29yIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU19SRVNJWkVfQ1VSU09SID0gOTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBXX1JFU0laRV9DVVJTT1IgY3Vyc29yIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV19SRVNJWkVfQ1VSU09SID0gMTA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRV9SRVNJWkVfQ1VSU09SIGN1cnNvciB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEVfUkVTSVpFX0NVUlNPUiA9IDExOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEhBTkRfQ1VSU09SIGN1cnNvciB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEhBTkRfQ1VSU09SID0gMTI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTU9WRV9DVVJTT1IgY3Vyc29yIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTU9WRV9DVVJTT1IgPSAxMzsKLQotICAgIC8qKgotICAgICAqIEEgbWFwcGluZyBmcm9tIG5hbWVzIHRvIHN5c3RlbSBjdXN0b20gY3Vyc29ycy4KLSAgICAgKi8KLSAgICBzdGF0aWMgTWFwPFN0cmluZywgQ3Vyc29yPiBzeXN0ZW1DdXN0b21DdXJzb3JzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGN1cnNvciBwcm9wcy4KLSAgICAgKi8KLSAgICBzdGF0aWMgUHJvcGVydGllcyBjdXJzb3JQcm9wczsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBwcmVkZWZpbmVkTmFtZXMuCi0gICAgICovCi0gICAgc3RhdGljIGZpbmFsIFN0cmluZ1tdIHByZWRlZmluZWROYW1lcyA9IHsKLSAgICAgICAgICAgICJEZWZhdWx0IiwgIkNyb3NzaGFpciIsICJUZXh0IiwgIldhaXQiLCAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJCAvLyROT04tTkxTLTQkCi0gICAgICAgICAgICAiU291dGh3ZXN0IFJlc2l6ZSIsICJTb3V0aGVhc3QgUmVzaXplIiwgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAiTm9ydGh3ZXN0IFJlc2l6ZSIsICJOb3J0aGVhc3QgUmVzaXplIiwgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAiTm9ydGggUmVzaXplIiwgIlNvdXRoIFJlc2l6ZSIsICJXZXN0IFJlc2l6ZSIsICJFYXN0IFJlc2l6ZSIsIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQKLSAgICAgICAgICAgICJIYW5kIiwgIk1vdmUiIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotCi0gICAgfTsKLQotICAgIC8qKgotICAgICAqIFRoZSBwcmVkZWZpbmVkIHNldCBvZiBjdXJzb3JzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBzdGF0aWMgQ3Vyc29yW10gcHJlZGVmaW5lZCA9IHsKLSAgICAgICAgICAgIG5ldyBDdXJzb3IoREVGQVVMVF9DVVJTT1IpLCBudWxsLCBudWxsLCBudWxsLCBudWxsLCBudWxsLCBudWxsLCBudWxsLCBudWxsLCBudWxsLCBudWxsLAotICAgICAgICAgICAgbnVsbCwgbnVsbCwgbnVsbAotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQ1VTVE9NX0NVUlNPUiBpcyBhc3NvY2lhdGVkIHdpdGggYWxsIGN1c3RvbSBjdXJzb3IgdHlwZXMuCi0gICAgICogKFRob3NlIHdoaWNoIGFyZSBub3QgcHJlZGVmaW5lZCkKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDVVNUT01fQ1VSU09SID0gLTE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbmFtZSBvZiB0aGUgY3Vyc29yLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmcgbmFtZTsKLQotICAgIC8qKgotICAgICAqIFRoZSB0eXBlIG9mIHRoZSBjdXJzb3IsIGNob3NlbiBmcm9tIHRoZSBsaXN0IG9mIGN1cnNvciB0eXBlIGNvbnN0YW50cy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIGludCB0eXBlOwotCi0gICAgLyoqCi0gICAgICogVGhlIG5hdGl2ZSBjdXJzb3IuCi0gICAgICovCi0gICAgcHJpdmF0ZSB0cmFuc2llbnQgTmF0aXZlQ3Vyc29yIG5hdGl2ZUN1cnNvcjsKLQotICAgIC8qKgotICAgICAqIFRoZSBleGFjdCBwb2ludCBvbiB0aGUgY3Vyc29yIGltYWdlIHRoYXQgaW5kaWNhdGVzIHdoaWNoIHBvaW50IHRoZSBjdXJzb3IKLSAgICAgKiBpcyBzZWxlY3RpbmcgKHBvaW50aW5nIHRvKS4gVGhlIGNvb3JkaW5hdGVzIGFyZSBnaXZlbiB3aXRoIHJlc3BlY3QgdGhlCi0gICAgICogb3JpZ2luIG9mIHRoZSBJbWFnZSAoaXRzIHVwcGVyIGxlZnQgY29ybmVyKS4KLSAgICAgKi8KLSAgICBwcml2YXRlIFBvaW50IGhvdFNwb3Q7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaW1hZ2UgdG8gZHJhdyBvbiB0aGUgc2NyZWVuIHJlcHJlc2VudGluZyB0aGUgY3Vyc29yLgotICAgICAqLwotICAgIHByaXZhdGUgSW1hZ2UgaW1hZ2U7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgY3Vyc29yIHdpdGggdGhlIHNwZWNpZmllZCBuYW1lLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiBjdXJzb3IuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEN1cnNvcihTdHJpbmcgbmFtZSkgewotICAgICAgICB0aGlzKG5hbWUsIG51bGwsIG5ldyBQb2ludCgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgY3Vyc29yIG9mIHRoZSBzcGVjaWZpZWQgdHlwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHR5cGUgb2YgY3Vyc29yLgotICAgICAqLwotICAgIHB1YmxpYyBDdXJzb3IoaW50IHR5cGUpIHsKLSAgICAgICAgY2hlY2tUeXBlKHR5cGUpOwotICAgICAgICB0aGlzLnR5cGUgPSB0eXBlOwotICAgICAgICBpZiAoKHR5cGUgPj0gMCkgJiYgKHR5cGUgPCBwcmVkZWZpbmVkTmFtZXMubGVuZ3RoKSkgewotICAgICAgICAgICAgbmFtZSA9IHByZWRlZmluZWROYW1lc1t0eXBlXSArICIgQ3Vyc29yIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGN1cnNvci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWUuCi0gICAgICogQHBhcmFtIGltZwotICAgICAqICAgICAgICAgICAgdGhlIGltZy4KLSAgICAgKiBAcGFyYW0gaG90U3BvdAotICAgICAqICAgICAgICAgICAgdGhlIGhvdCBzcG90LgotICAgICAqLwotICAgIEN1cnNvcihTdHJpbmcgbmFtZSwgSW1hZ2UgaW1nLCBQb2ludCBob3RTcG90KSB7Ci0gICAgICAgIHRoaXMubmFtZSA9IG5hbWU7Ci0gICAgICAgIHR5cGUgPSBDVVNUT01fQ1VSU09SOwotICAgICAgICB0aGlzLmhvdFNwb3QgPSBob3RTcG90OwotICAgICAgICBpbWFnZSA9IGltZzsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaW5hbGl6ZSBtZXRob2Qgb3ZlcnJpZGVzIHRoZSBmaW5hbGl6ZSBtZXRob2QgZnJvbSBPYmplY3QgY2xhc3MuCi0gICAgICogCi0gICAgICogQHRocm93cyBUaHJvd2FibGUKLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgbmF0aXZlIGN1cnNvciBpcyBub3QgbnVsbCBhbmQgdGhyb3dzIGEgVGhyb3dhYmxlIHdoZW4KLSAgICAgKiAgICAgICAgICAgICBkZXN0cm95ZWQuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHJvdGVjdGVkIHZvaWQgZmluYWxpemUoKSB0aHJvd3MgVGhyb3dhYmxlIHsKLSAgICAgICAgaWYgKG5hdGl2ZUN1cnNvciAhPSBudWxsKSB7Ci0gICAgICAgICAgICBuYXRpdmVDdXJzb3IuZGVzdHJveUN1cnNvcigpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbmFtZSBvZiB0aGUgY3Vyc29yLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG5hbWUgb2YgdGhlIGN1cnNvci4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldE5hbWUoKSB7Ci0gICAgICAgIHJldHVybiBuYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIFN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgY3Vyc29yLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgY3Vyc29yLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgIHJldHVybiBnZXRDbGFzcygpLmdldE5hbWUoKSArICJbIiArIG5hbWUgKyAiXSI7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGN1cnNvciB0eXBlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGN1cnNvciB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIHR5cGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcHJlZGVmaW5lZCBjdXJzb3Igd2l0aCB0aGUgc3BlY2lmaWVkIHR5cGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSB0eXBlIG9mIGN1cnNvci4KLSAgICAgKiBAcmV0dXJuIHRoZSBwcmVkZWZpbmVkIGN1cnNvciB3aXRoIHRoZSBzcGVjaWZpZWQgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEN1cnNvciBnZXRQcmVkZWZpbmVkQ3Vyc29yKGludCB0eXBlKSB7Ci0gICAgICAgIGNoZWNrVHlwZSh0eXBlKTsKLSAgICAgICAgQ3Vyc29yIGN1cnNvciA9IHByZWRlZmluZWRbdHlwZV07Ci0gICAgICAgIGlmIChjdXJzb3IgPT0gbnVsbCkgewotICAgICAgICAgICAgY3Vyc29yID0gbmV3IEN1cnNvcih0eXBlKTsKLSAgICAgICAgICAgIHByZWRlZmluZWRbdHlwZV0gPSBjdXJzb3I7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGN1cnNvcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkZWZhdWx0IGN1cnNvci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkZWZhdWx0IGN1cnNvci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEN1cnNvciBnZXREZWZhdWx0Q3Vyc29yKCkgewotICAgICAgICByZXR1cm4gZ2V0UHJlZGVmaW5lZEN1cnNvcihERUZBVUxUX0NVUlNPUik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc3BlY2lmaWVkIHN5c3RlbSBjdXN0b20gY3Vyc29yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiB0aGUgZGVzaXJlZCBzeXN0ZW0gY3Vyc29yLgotICAgICAqIEByZXR1cm4gdGhlIHNwZWNpZmljIHN5c3RlbSBjdXJzb3Igd2l0aCB0aGUgc3BlY2lmaWVkIG5hbWUuCi0gICAgICogQHRocm93cyBBV1RFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgZGVzaXJlZCBjdXJzb3IgaGFzIG1hbGZvcm1lZCBkYXRhIHN1Y2ggYXMgYW4KLSAgICAgKiAgICAgICAgICAgICBpbmNvcnJlY3RseSBkZWZpbmVkIGhvdCBzcG90LgotICAgICAqIEB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgaXNIZWFkbGVzcyBtZXRob2Qgb2YgdGhlIEdyYXBoaWNzRW52aXJvbm1lbnQgcmV0dXJucwotICAgICAqICAgICAgICAgICAgIHRydWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBDdXJzb3IgZ2V0U3lzdGVtQ3VzdG9tQ3Vyc29yKFN0cmluZyBuYW1lKSB0aHJvd3MgQVdURXhjZXB0aW9uLCBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIFRvb2xraXQuY2hlY2tIZWFkbGVzcygpOwotICAgICAgICByZXR1cm4gZ2V0U3lzdGVtQ3VzdG9tQ3Vyc29yRnJvbU1hcChuYW1lKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzcGVjaWZpZWQgc3lzdGVtIGN1c3RvbSBjdXJzb3IgZnJvbSB0aGUgbWFwIG9mIHN5c3RlbSBjdXN0b20KLSAgICAgKiBjdXJzb3JzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiB0aGUgZGVzaXJlZCBjdXJzb3IuCi0gICAgICogQHJldHVybiB0aGUgZGVzaXJlZCBzeXN0ZW0gY3VzdG9tIGN1cnNvciBmcm9tIHRoZSBtYXAgb2Ygc3lzdGVtIGN1c3RvbQotICAgICAqICAgICAgICAgY3Vyc29ycy4KLSAgICAgKiBAdGhyb3dzIEFXVEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIHRoZSBBV1QgZXhjZXB0aW9uLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIEN1cnNvciBnZXRTeXN0ZW1DdXN0b21DdXJzb3JGcm9tTWFwKFN0cmluZyBuYW1lKSB0aHJvd3MgQVdURXhjZXB0aW9uIHsKLSAgICAgICAgbG9hZEN1cnNvclByb3BzKCk7Ci0gICAgICAgIGlmIChzeXN0ZW1DdXN0b21DdXJzb3JzID09IG51bGwpIHsKLSAgICAgICAgICAgIHN5c3RlbUN1c3RvbUN1cnNvcnMgPSBuZXcgSGFzaE1hcDxTdHJpbmcsIEN1cnNvcj4oKTsKLSAgICAgICAgfQotICAgICAgICBDdXJzb3IgY3Vyc29yID0gc3lzdGVtQ3VzdG9tQ3Vyc29ycy5nZXQobmFtZSk7Ci0gICAgICAgIGlmIChjdXJzb3IgIT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIGN1cnNvcjsKLSAgICAgICAgfQotICAgICAgICAvLyBhd3QuMTQxPWZhaWxlZCB0byBwYXJzZSBob3RzcG90IHByb3BlcnR5IGZvciBjdXJzb3I6Ci0gICAgICAgIFN0cmluZyBleE1zZyA9IE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE0MSIpICsgbmFtZTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBTdHJpbmcgbm0gPSAiQ3Vyc29yLiIgKyBuYW1lOyAvLyROT04tTkxTLTEkCi0gICAgICAgIFN0cmluZyBuYW1lU3RyID0gY3Vyc29yUHJvcHMuZ2V0UHJvcGVydHkobm0gKyAiLk5hbWUiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBTdHJpbmcgaG90U3BvdFN0ciA9IGN1cnNvclByb3BzLmdldFByb3BlcnR5KG5tICsgIi5Ib3RTcG90Iik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgU3RyaW5nIGZpbGVTdHIgPSBjdXJzb3JQcm9wcy5nZXRQcm9wZXJ0eShubSArICIuRmlsZSIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIGludCBpZHggPSBob3RTcG90U3RyLmluZGV4T2YoJywnKTsKLSAgICAgICAgaWYgKGlkeCA8IDApIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBBV1RFeGNlcHRpb24oZXhNc2cpOwotICAgICAgICB9Ci0gICAgICAgIGludCB4LCB5OwotICAgICAgICB0cnkgewotICAgICAgICAgICAgeCA9IG5ldyBJbnRlZ2VyKGhvdFNwb3RTdHIuc3Vic3RyaW5nKDAsIGlkeCkpLmludFZhbHVlKCk7Ci0gICAgICAgICAgICB5ID0gbmV3IEludGVnZXIoaG90U3BvdFN0ci5zdWJzdHJpbmcoaWR4ICsgMSwgaG90U3BvdFN0ci5sZW5ndGgoKSkpLmludFZhbHVlKCk7Ci0gICAgICAgIH0gY2F0Y2ggKE51bWJlckZvcm1hdEV4Y2VwdGlvbiBuZmUpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBBV1RFeGNlcHRpb24oZXhNc2cpOwotICAgICAgICB9Ci0gICAgICAgIEltYWdlIGltZyA9IFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKS5jcmVhdGVJbWFnZShmaWxlU3RyKTsKLSAgICAgICAgY3Vyc29yID0gbmV3IEN1cnNvcihuYW1lU3RyLCBpbWcsIG5ldyBQb2ludCh4LCB5KSk7Ci0gICAgICAgIHN5c3RlbUN1c3RvbUN1cnNvcnMucHV0KG5hbWUsIGN1cnNvcik7Ci0KLSAgICAgICAgcmV0dXJuIGN1cnNvcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBMb2FkIGN1cnNvciBwcm9wcy4KLSAgICAgKiAKLSAgICAgKiBAdGhyb3dzIEFXVEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIHRoZSBBV1QgZXhjZXB0aW9uLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIHZvaWQgbG9hZEN1cnNvclByb3BzKCkgdGhyb3dzIEFXVEV4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChjdXJzb3JQcm9wcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgU3RyaW5nIHNlcCA9IEZpbGUuc2VwYXJhdG9yOwotICAgICAgICBTdHJpbmcgY3Vyc29yc0RpciA9ICJsaWIiICsgc2VwICsgImltYWdlcyIgKyBzZXAgKyAiY3Vyc29ycyI7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkCi0gICAgICAgIFN0cmluZyBjdXJzb3JzQWJzRGlyID0gU3lzdGVtLmdldFByb3BlcnR5KCJqYXZhLmhvbWUiKSArIHNlcCArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICBjdXJzb3JzRGlyOwotICAgICAgICBTdHJpbmcgY3Vyc29yUHJvcHNGaWxlTmFtZSA9ICJjdXJzb3JzLnByb3BlcnRpZXMiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIFN0cmluZyBjdXJzb3JQcm9wc0Z1bGxGaWxlTmFtZSA9IChjdXJzb3JzQWJzRGlyICsgc2VwICsgY3Vyc29yUHJvcHNGaWxlTmFtZSk7Ci0gICAgICAgIGN1cnNvclByb3BzID0gbmV3IFByb3BlcnRpZXMoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGN1cnNvclByb3BzLmxvYWQobmV3IEZpbGVJbnB1dFN0cmVhbShuZXcgRmlsZShjdXJzb3JQcm9wc0Z1bGxGaWxlTmFtZSkpKTsKLSAgICAgICAgfSBjYXRjaCAoRmlsZU5vdEZvdW5kRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xNDI9RXhjZXB0aW9uOiBjbGFzcyB7MH0gezF9IG9jY3VycmVkIHdoaWxlIGxvYWRpbmc6IHsyfQotICAgICAgICAgICAgdGhyb3cgbmV3IEFXVEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNDIiLC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgbmV3IE9iamVjdFtdIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBlLmdldENsYXNzKCksIGUuZ2V0TWVzc2FnZSgpLCBjdXJzb3JQcm9wc0Z1bGxGaWxlTmFtZQotICAgICAgICAgICAgICAgICAgICB9KSk7Ci0gICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBBV1RFeGNlcHRpb24oZS5nZXRNZXNzYWdlKCkpOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVjayB0eXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0eXBlCi0gICAgICogICAgICAgICAgICB0aGUgdHlwZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgdm9pZCBjaGVja1R5cGUoaW50IHR5cGUpIHsKLSAgICAgICAgLy8gY2FuJ3QgdXNlIHByZWRlZmluZWQgYXJyYXkgaGVyZSBiZWNhdXNlIGl0IG1heSBub3QgaGF2ZSBiZWVuCi0gICAgICAgIC8vIGluaXRpYWxpemVkIHlldAotICAgICAgICBpZiAoKHR5cGUgPCAwKSB8fCAodHlwZSA+PSBwcmVkZWZpbmVkTmFtZXMubGVuZ3RoKSkgewotICAgICAgICAgICAgLy8gYXd0LjE0Mz1pbGxlZ2FsIGN1cnNvciB0eXBlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE0MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gImxhemlseSIgY3JlYXRlIG5hdGl2ZSBjdXJzb3JzOgotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG5hdGl2ZSBjdXJzb3IuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbmF0aXZlIGN1cnNvci4KLSAgICAgKi8KLSAgICBOYXRpdmVDdXJzb3IgZ2V0TmF0aXZlQ3Vyc29yKCkgewotICAgICAgICBpZiAobmF0aXZlQ3Vyc29yICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBuYXRpdmVDdXJzb3I7Ci0gICAgICAgIH0KLSAgICAgICAgVG9vbGtpdCB0b29sa2l0ID0gVG9vbGtpdC5nZXREZWZhdWx0VG9vbGtpdCgpOwotICAgICAgICBpZiAodHlwZSAhPSBDVVNUT01fQ1VSU09SKSB7Ci0gICAgICAgICAgICBuYXRpdmVDdXJzb3IgPSB0b29sa2l0LmNyZWF0ZU5hdGl2ZUN1cnNvcih0eXBlKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG5hdGl2ZUN1cnNvciA9IHRvb2xraXQuY3JlYXRlQ3VzdG9tTmF0aXZlQ3Vyc29yKGltYWdlLCBob3RTcG90LCBuYW1lKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbmF0aXZlQ3Vyc29yOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIG5hdGl2ZSBjdXJzb3IuCi0gICAgICogCi0gICAgICogQHBhcmFtIG5hdGl2ZUN1cnNvcgotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBuYXRpdmUgY3Vyc29yLgotICAgICAqLwotICAgIHZvaWQgc2V0TmF0aXZlQ3Vyc29yKE5hdGl2ZUN1cnNvciBuYXRpdmVDdXJzb3IpIHsKLSAgICAgICAgdGhpcy5uYXRpdmVDdXJzb3IgPSBuYXRpdmVDdXJzb3I7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0RpbWVuc2lvbi5qYXZhIGIvYXd0L2phdmEvYXd0L0RpbWVuc2lvbi5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2Nzc3OTYyLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9EaW1lbnNpb24uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDIwMSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERlbmlzIE0uIEtpc2hlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEuYXd0Lmdlb20uRGltZW5zaW9uMkQ7Ci1pbXBvcnQgamF2YS5pby5TZXJpYWxpemFibGU7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkubWlzYy5IYXNoQ29kZTsKLQotLyoqCi0gKiBUaGUgRGltZW5zaW9uIHJlcHJlc2VudHMgdGhlIHNpemUgKHdpZHRoIGFuZCBoZWlnaHQpIG9mIGEgY29tcG9uZW50LiBUaGUKLSAqIHdpZHRoIGFuZCBoZWlnaHQgdmFsdWVzIGNhbiBiZSBuZWdhdGl2ZSwgYnV0IGluIHRoYXQgY2FzZSB0aGUgYmVoYXZpb3Igb2YKLSAqIHNvbWUgbWV0aG9kcyBpcyB1bmV4cGVjdGVkLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIERpbWVuc2lvbiBleHRlbmRzIERpbWVuc2lvbjJEIGltcGxlbWVudHMgU2VyaWFsaXphYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IDQ3MjM5NTI1Nzk0OTEzNDk1MjRMOwotCi0gICAgLyoqCi0gICAgICogVGhlIHdpZHRoIGRpbWVuc2lvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHdpZHRoOwotCi0gICAgLyoqCi0gICAgICogVGhlIGhlaWdodCBkaW1lbnNpb24uCi0gICAgICovCi0gICAgcHVibGljIGludCBoZWlnaHQ7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRGltZW5zaW9uIHdpdGggdGhlIHNhbWUgZGF0YSBhcyB0aGUgc3BlY2lmaWVkCi0gICAgICogRGltZW5zaW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkCi0gICAgICogICAgICAgICAgICB0aGUgRGltZW5zaW9uIHRvIGNvcHkgdGhlIGRhdGEgZnJvbSB3aGVuIGNyZWF0aW5nIHRoZSBuZXcKLSAgICAgKiAgICAgICAgICAgIERpbWVuc2lvbiBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIERpbWVuc2lvbihEaW1lbnNpb24gZCkgewotICAgICAgICB0aGlzKGQud2lkdGgsIGQuaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRGltZW5zaW9uIHdpdGggemVybyB3aWR0aCBhbmQgaGVpZ2h0LgotICAgICAqLwotICAgIHB1YmxpYyBEaW1lbnNpb24oKSB7Ci0gICAgICAgIHRoaXMoMCwgMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IERpbWVuc2lvbiB3aXRoIHRoZSBzcGVjaWZpZWQgd2lkdGggYW5kIGhlaWdodC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgbmV3IERpbWVuc2lvbi4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBuZXcgRGltZW5zaW9uLgotICAgICAqLwotICAgIHB1YmxpYyBEaW1lbnNpb24oaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIHNldFNpemUod2lkdGgsIGhlaWdodCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgaGFzaCBjb2RlIG9mIHRoZSBEaW1lbnNpb24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaGFzaCBjb2RlIG9mIHRoZSBEaW1lbnNpb24uCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgSGFzaENvZGUgaGFzaCA9IG5ldyBIYXNoQ29kZSgpOwotICAgICAgICBoYXNoLmFwcGVuZCh3aWR0aCk7Ci0gICAgICAgIGhhc2guYXBwZW5kKGhlaWdodCk7Ci0gICAgICAgIHJldHVybiBoYXNoLmhhc2hDb2RlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgdGhpcyBEaW1lbnNpb24gb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9iagotICAgICAqICAgICAgICAgICAgdGhlIE9iamVjdCB0byBiZSBjb21wYXJlZC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgT2JqZWN0IGlzIGEgRGltZW5zaW9uIHdpdGggdGhlIHNhbWUgd2lkdGgKLSAgICAgKiAgICAgICAgIGFuZCBoZWlnaHQgZGF0YSBhcyB0aGlzIERpbWVuc2lvbi4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG9iaikgewotICAgICAgICBpZiAob2JqID09IHRoaXMpIHsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChvYmogaW5zdGFuY2VvZiBEaW1lbnNpb24pIHsKLSAgICAgICAgICAgIERpbWVuc2lvbiBkID0gKERpbWVuc2lvbilvYmo7Ci0gICAgICAgICAgICByZXR1cm4gKGQud2lkdGggPT0gd2lkdGggJiYgZC5oZWlnaHQgPT0gaGVpZ2h0KTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgU3RyaW5nIGFzc29jaWF0ZWQgdG8gdGhpcyBEaW1lbnNpb24gb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFN0cmluZyBhc3NvY2lhdGVkIHRvIHRoaXMgRGltZW5zaW9uIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICAvLyBUaGUgb3V0cHV0IGZvcm1hdCBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvdXIuIEl0IGNvdWxkIGJlCi0gICAgICAgIC8vIG9idGFpbmVkIGluIHRoZSBmb2xsb3dpbmcgd2F5Ci0gICAgICAgIC8vIFN5c3RlbS5vdXQucHJpbnRsbihuZXcgRGltZW5zaW9uKCkudG9TdHJpbmcoKSkKLSAgICAgICAgcmV0dXJuIGdldENsYXNzKCkuZ2V0TmFtZSgpICsgIlt3aWR0aD0iICsgd2lkdGggKyAiLGhlaWdodD0iICsgaGVpZ2h0ICsgIl0iOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHNpemUgb2YgdGhpcyBEaW1lbnNpb24gb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCB3aWR0aCBhbmQKLSAgICAgKiBoZWlnaHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIERpbWVuc2lvbi4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBEaW1lbnNpb24uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0U2l6ZShpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgdGhpcy53aWR0aCA9IHdpZHRoOwotICAgICAgICB0aGlzLmhlaWdodCA9IGhlaWdodDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBzaXplIG9mIHRoaXMgRGltZW5zaW9uIG9iamVjdCBieSBjb3B5aW5nIHRoZSBkYXRhIGZyb20gdGhlCi0gICAgICogc3BlY2lmaWVkIERpbWVuc2lvbiBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGQKLSAgICAgKiAgICAgICAgICAgIHRoZSBEaW1lbnNpb24gdGhhdCBnaXZlcyB0aGUgbmV3IHNpemUgdmFsdWVzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNpemUoRGltZW5zaW9uIGQpIHsKLSAgICAgICAgc2V0U2l6ZShkLndpZHRoLCBkLmhlaWdodCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2l6ZSBvZiB0aGlzIERpbWVuc2lvbiBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIGRvdWJsZSB3aWR0aAotICAgICAqIGFuZCBoZWlnaHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIERpbWVuc2lvbi4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBEaW1lbnNpb24uCi0gICAgICogQHNlZSBqYXZhLmF3dC5nZW9tLkRpbWVuc2lvbjJEI3NldFNpemUoZG91YmxlLCBkb3VibGUpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0U2l6ZShkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQpIHsKLSAgICAgICAgc2V0U2l6ZSgoaW50KU1hdGguY2VpbCh3aWR0aCksIChpbnQpTWF0aC5jZWlsKGhlaWdodCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNpemUgb2YgdGhlIERpbWVuc2lvbi4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzaXplIG9mIHRoZSBEaW1lbnNpb24uCi0gICAgICovCi0gICAgcHVibGljIERpbWVuc2lvbiBnZXRTaXplKCkgewotICAgICAgICByZXR1cm4gbmV3IERpbWVuc2lvbih3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBoZWlnaHQgb2YgdGhlIERpbWVuc2lvbi4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBoZWlnaHQgb2YgdGhlIERpbWVuc2lvbi4KLSAgICAgKiBAc2VlIGphdmEuYXd0Lmdlb20uRGltZW5zaW9uMkQjZ2V0SGVpZ2h0KCkKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZG91YmxlIGdldEhlaWdodCgpIHsKLSAgICAgICAgcmV0dXJuIGhlaWdodDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB3aWR0aCBvZiB0aGUgRGltZW5zaW9uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHdpZHRoIG9mIHRoZSBEaW1lbnNpb24uCi0gICAgICogQHNlZSBqYXZhLmF3dC5nZW9tLkRpbWVuc2lvbjJEI2dldFdpZHRoKCkKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZG91YmxlIGdldFdpZHRoKCkgewotICAgICAgICByZXR1cm4gd2lkdGg7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvRGlzcGF0Y2hlci5qYXZhIGIvYXd0L2phdmEvYXd0L0Rpc3BhdGNoZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDQ1N2FmNC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvRGlzcGF0Y2hlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNzIzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92LCBEbWl0cnkgQS4gRHVybmV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmF3dC5ldmVudC5Db21wb25lbnRFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5Gb2N1c0V2ZW50OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuS2V5RXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuTW91c2VFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5QYWludEV2ZW50OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LldpbmRvd0V2ZW50OwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuTmF0aXZlRXZlbnQ7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuTmF0aXZlV2luZG93OwotCi0KLS8qKgotICogSGVscGVyIHBhY2thZ2UtcHJpdmF0ZSBjbGFzcyBmb3IgbWFuYWdpbmcgbGlnaHR3ZWlnaHQgY29tcG9uZW50cyAmCi0gKiBkaXNwYXRjaGluZyBldmVudHMgZnJvbSBoZWF2eXdlaWdodCBzb3VyY2UKLSAqLwotY2xhc3MgRGlzcGF0Y2hlciB7Ci0KLSAgICAvLz8/P0FXVDogZmluYWwgUG9wdXBEaXNwYXRjaGVyIHBvcHVwRGlzcGF0Y2hlciA9IG5ldyBQb3B1cERpc3BhdGNoZXIoKTsKLQotICAgIC8vPz8/QVdUOiBmaW5hbCBGb2N1c0Rpc3BhdGNoZXIgZm9jdXNEaXNwYXRjaGVyOwotCi0gICAgZmluYWwgTW91c2VHcmFiTWFuYWdlciBtb3VzZUdyYWJNYW5hZ2VyID0gbmV3IE1vdXNlR3JhYk1hbmFnZXIoKTsKLQotICAgIGZpbmFsIE1vdXNlRGlzcGF0Y2hlciBtb3VzZURpc3BhdGNoZXI7Ci0KLSAgICBwcml2YXRlIGZpbmFsIENvbXBvbmVudERpc3BhdGNoZXIgY29tcG9uZW50RGlzcGF0Y2hlciA9IG5ldyBDb21wb25lbnREaXNwYXRjaGVyKCk7Ci0KLSAgICBwcml2YXRlIGZpbmFsIEtleURpc3BhdGNoZXIga2V5RGlzcGF0Y2hlciA9IG5ldyBLZXlEaXNwYXRjaGVyKCk7Ci0KLSAgICBwcml2YXRlIGZpbmFsIFRvb2xraXQgdG9vbGtpdDsKLQotICAgIGludCBjbGlja0ludGVydmFsID0gMjUwOwotCi0gICAgLyoqCi0gICAgICogQHBhcmFtIHRvb2xraXQgLSBBV1QgdG9vbGtpdAotICAgICAqLwotICAgIERpc3BhdGNoZXIoVG9vbGtpdCB0b29sa2l0KSB7Ci0gICAgICAgIHRoaXMudG9vbGtpdCA9IHRvb2xraXQ7Ci0KLSAgICAgICAgLy8/Pz9BV1Q6IGZvY3VzRGlzcGF0Y2hlciA9IG5ldyBGb2N1c0Rpc3BhdGNoZXIodG9vbGtpdCk7Ci0gICAgICAgIG1vdXNlRGlzcGF0Y2hlciA9IG5ldyBNb3VzZURpc3BhdGNoZXIobW91c2VHcmFiTWFuYWdlciwgdG9vbGtpdCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGlzcGF0Y2ggbmF0aXZlIGV2ZW50OiBwcm9kdWNlIGFwcHJvcHJpYXRlIEFXVCBldmVudHMsIAotICAgICAqIHVwZGF0ZSBjb21wb25lbnQncyBmaWVsZHMgd2hlbiBuZWVkZWQKLSAgICAgKiBAcGFyYW0gZXZlbnQgLSBuYXRpdmUgZXZlbnQgdG8gZGlzcGF0Y2gKLSAgICAgKiBAcmV0dXJuIC0gdHJ1ZSBtZWFucyBkZWZhdWx0IHByb2Nlc3NpbmcgYnkgT1MgaXMgbm90IG5lZWRlZAotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIG9uRXZlbnQoTmF0aXZlRXZlbnQgZXZlbnQpIHsKLSAgICAgICAgaW50IGV2ZW50SWQgPSBldmVudC5nZXRFdmVudElkKCk7Ci0KLSAgICAgICAgaWYgKGV2ZW50SWQgPT0gTmF0aXZlRXZlbnQuSURfQ1JFQVRFRCkgewotICAgICAgICAgICAgcmV0dXJuIHRvb2xraXQub25XaW5kb3dDcmVhdGVkKGV2ZW50LmdldFdpbmRvd0lkKCkpOwotICAgICAgICB9IGVsc2UgaWYgKGV2ZW50SWQgPT0gTmF0aXZlRXZlbnQuSURfTU9VU0VfR1JBQl9DQU5DRUxFRCkgewotICAgICAgICAgICAgcmV0dXJuIG1vdXNlR3JhYk1hbmFnZXIub25HcmFiQ2FuY2VsZWQoKTsKLSAgICAgICAgLy8/Pz9BV1QKLS8vICAgICAgICB9IGVsc2UgaWYgKHBvcHVwRGlzcGF0Y2hlci5vbkV2ZW50KGV2ZW50KSkgewotLy8gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBDb21wb25lbnQgc3JjID0gdG9vbGtpdC5nZXRDb21wb25lbnRCeUlkKGV2ZW50LmdldFdpbmRvd0lkKCkpOwotCi0gICAgICAgICAgICBpZiAoc3JjICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBpZiAoKChldmVudElkID49IENvbXBvbmVudEV2ZW50LkNPTVBPTkVOVF9GSVJTVCkgJiYgKGV2ZW50SWQgPD0gQ29tcG9uZW50RXZlbnQuQ09NUE9ORU5UX0xBU1QpKQotICAgICAgICAgICAgICAgICAgICAgICAgfHwgKChldmVudElkID49IFdpbmRvd0V2ZW50LldJTkRPV19GSVJTVCkgJiYgKGV2ZW50SWQgPD0gV2luZG93RXZlbnQuV0lORE9XX0xBU1QpKQotICAgICAgICAgICAgICAgICAgICAgICAgfHwgKGV2ZW50SWQgPT0gTmF0aXZlRXZlbnQuSURfSU5TRVRTX0NIQU5HRUQpCi0gICAgICAgICAgICAgICAgICAgICAgICB8fCAoZXZlbnRJZCA9PSBOYXRpdmVFdmVudC5JRF9CT1VORFNfQ0hBTkdFRCkKLSAgICAgICAgICAgICAgICAgICAgICAgIHx8IChldmVudElkID09IE5hdGl2ZUV2ZW50LklEX1RIRU1FX0NIQU5HRUQpKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBjb21wb25lbnREaXNwYXRjaGVyLmRpc3BhdGNoKHNyYywgZXZlbnQpOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoKGV2ZW50SWQgPj0gTW91c2VFdmVudC5NT1VTRV9GSVJTVCkKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIChldmVudElkIDw9IE1vdXNlRXZlbnQuTU9VU0VfTEFTVCkpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIG1vdXNlRGlzcGF0Y2hlci5kaXNwYXRjaChzcmMsIGV2ZW50KTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGV2ZW50SWQgPT0gUGFpbnRFdmVudC5QQUlOVCkgewotICAgICAgICAgICAgICAgICAgICAvLz8/P0FXVDogc3JjLnJlZHJhd01hbmFnZXIuYWRkUGFpbnRSZWdpb24oc3JjLCBldmVudC5nZXRDbGlwUmVjdHMoKSk7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICgoZXZlbnRJZCA+PSBGb2N1c0V2ZW50LkZPQ1VTX0ZJUlNUKQotICAgICAgICAgICAgICAgICAgICAmJiAoZXZlbnRJZCA8PSBGb2N1c0V2ZW50LkZPQ1VTX0xBU1QpKSB7Ci0KLSAgICAgICAgICAgICAgICAvLz8/P0FXVDogcmV0dXJuIGZvY3VzRGlzcGF0Y2hlci5kaXNwYXRjaChzcmMsIGV2ZW50KTsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKChldmVudElkID49IEtleUV2ZW50LktFWV9GSVJTVCkKLSAgICAgICAgICAgICAgICAgICAgJiYgKGV2ZW50SWQgPD0gS2V5RXZlbnQuS0VZX0xBU1QpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGtleURpc3BhdGNoZXIuZGlzcGF0Y2goc3JjLCBldmVudCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIGRpc3BhdGNoZXIgb2YgbmF0aXZlIGV2ZW50cyB0aGF0IGFmZmVjdCAKLSAgICAgKiBjb21wb25lbnQncyBzdGF0ZSBvciBib3VuZHMKLSAgICAgKi8KLSAgICBmaW5hbCBjbGFzcyBDb21wb25lbnREaXNwYXRjaGVyIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSGFuZGxlIG5hdGl2ZSBldmVudCB0aGF0IGFmZmVjdHMgY29tcG9uZW50J3Mgc3RhdGUgb3IgYm91bmRzCi0gICAgICAgICAqIEBwYXJhbSBzcmMgLSB0aGUgY29tcG9uZW50IHVwZGF0ZWQgYnkgdGhlIGV2ZW50Ci0gICAgICAgICAqIEBwYXJhbSBldmVudCAtIHRoZSBuYXRpdmUgZXZlbnQKLSAgICAgICAgICogQHJldHVybiAtIGFzIGluIERpc3BhdGNoZXIub25FdmVudCgpCi0gICAgICAgICAqIEBzZWUgRGlzcGF0Y2hlciNvbkV2ZW50KE5hdGl2ZUV2ZW50KQotICAgICAgICAgKi8KLSAgICAgICAgYm9vbGVhbiBkaXNwYXRjaChDb21wb25lbnQgc3JjLCBOYXRpdmVFdmVudCBldmVudCkgewotICAgICAgICAgICAgaW50IGlkID0gZXZlbnQuZ2V0RXZlbnRJZCgpOwotCi0gICAgICAgICAgICBpZiAoKGlkID09IE5hdGl2ZUV2ZW50LklEX0lOU0VUU19DSEFOR0VEKQotICAgICAgICAgICAgICAgICAgICB8fCAoaWQgPT0gTmF0aXZlRXZlbnQuSURfVEhFTUVfQ0hBTkdFRCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZGlzcGF0Y2hJbnNldHMoZXZlbnQsIHNyYyk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKChpZCA+PSBXaW5kb3dFdmVudC5XSU5ET1dfRklSU1QpCi0gICAgICAgICAgICAgICAgICAgICYmIChpZCA8PSBXaW5kb3dFdmVudC5XSU5ET1dfTEFTVCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZGlzcGF0Y2hXaW5kb3coZXZlbnQsIHNyYyk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJldHVybiBkaXNwYXRjaFB1cmVDb21wb25lbnQoZXZlbnQsIHNyYyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSGFuZGxlIHRoZSBjaGFuZ2Ugb2YgdG9wLWxldmVsIHdpbmRvdydzIG5hdGl2ZSBkZWNvcmF0aW9ucyAKLSAgICAgICAgICogQHBhcmFtIGV2ZW50IC0gdGhlIG5hdGl2ZSBldmVudAotICAgICAgICAgKiBAcGFyYW0gc3JjIC0gdGhlIGNvbXBvbmVudCB1cGRhdGVkIGJ5IHRoZSBldmVudAotICAgICAgICAgKiBAcmV0dXJuIC0gYXMgaW4gRGlzcGF0Y2hlci5vbkV2ZW50KCkKLSAgICAgICAgICogQHNlZSBEaXNwYXRjaGVyI29uRXZlbnQoTmF0aXZlRXZlbnQpCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIGRpc3BhdGNoSW5zZXRzKE5hdGl2ZUV2ZW50IGV2ZW50LCBDb21wb25lbnQgc3JjKSB7Ci0gICAgICAgICAgICAvLz8/P0FXVAotICAgICAgICAgICAgLyoKLSAgICAgICAgICAgIGlmIChzcmMgaW5zdGFuY2VvZiBXaW5kb3cpIHsKLSAgICAgICAgICAgICAgICAoKFdpbmRvdykgc3JjKS5zZXROYXRpdmVJbnNldHMoZXZlbnQuZ2V0SW5zZXRzKCkpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBIYW5kbGUgdGhlIGNoYW5nZSBvZiB0b3AtbGV2ZWwgd2luZG93J3Mgc3RhdGUKLSAgICAgICAgICogQHBhcmFtIGV2ZW50IC0gdGhlIG5hdGl2ZSBldmVudAotICAgICAgICAgKiBAcGFyYW0gc3JjIC0gdGhlIGNvbXBvbmVudCB1cGRhdGVkIGJ5IHRoZSBldmVudAotICAgICAgICAgKiBAcmV0dXJuIC0gYXMgaW4gRGlzcGF0Y2hlci5vbkV2ZW50KCkKLSAgICAgICAgICogQHNlZSBEaXNwYXRjaGVyI29uRXZlbnQoTmF0aXZlRXZlbnQpCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIGRpc3BhdGNoV2luZG93KE5hdGl2ZUV2ZW50IGV2ZW50LCBDb21wb25lbnQgc3JjKSB7Ci0gICAgICAgICAgICAvLz8/P0FXVAotICAgICAgICAgICAgLyoKLSAgICAgICAgICAgIFdpbmRvdyB3aW5kb3cgPSAoV2luZG93KSBzcmM7Ci0gICAgICAgICAgICBpbnQgaWQgPSBldmVudC5nZXRFdmVudElkKCk7Ci0KLSAgICAgICAgICAgIGlmIChpZCA9PSBXaW5kb3dFdmVudC5XSU5ET1dfQ0xPU0lORykgewotICAgICAgICAgICAgICAgIHRvb2xraXQuZ2V0U3lzdGVtRXZlbnRRdWV1ZUltcGwoKS5wb3N0RXZlbnQoCi0gICAgICAgICAgICAgICAgICAgICAgICAgIG5ldyBXaW5kb3dFdmVudCh3aW5kb3csIFdpbmRvd0V2ZW50LldJTkRPV19DTE9TSU5HKSk7Ci0KLSAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoaWQgPT0gV2luZG93RXZlbnQuV0lORE9XX1NUQVRFX0NIQU5HRUQpIHsKLSAgICAgICAgICAgICAgICBpZiAod2luZG93IGluc3RhbmNlb2YgRnJhbWUpIHsKLSAgICAgICAgICAgICAgICAgICAgKChGcmFtZSkgd2luZG93KQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIC51cGRhdGVFeHRlbmRlZFN0YXRlKGV2ZW50LmdldFdpbmRvd1N0YXRlKCkpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgICovCi0KLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBIYW5kbGUgdGhlIGNoYW5nZSBvZiBjb21wb25lbnQncyBzaXplIGFuZC9vciBwb3NpdGlvbgotICAgICAgICAgKiBAcGFyYW0gZXZlbnQgLSB0aGUgbmF0aXZlIGV2ZW50Ci0gICAgICAgICAqIEBwYXJhbSBzcmMgLSB0aGUgY29tcG9uZW50IHVwZGF0ZWQgYnkgdGhlIGV2ZW50Ci0gICAgICAgICAqIEByZXR1cm4gLSBhcyBpbiBEaXNwYXRjaGVyLm9uRXZlbnQoKQotICAgICAgICAgKiBAc2VlIERpc3BhdGNoZXIjb25FdmVudChOYXRpdmVFdmVudCkKLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgYm9vbGVhbiBkaXNwYXRjaFB1cmVDb21wb25lbnQoTmF0aXZlRXZlbnQgZXZlbnQsIENvbXBvbmVudCBzcmMpIHsKLSAgICAgICAgICAgIFJlY3RhbmdsZSByZWN0ID0gZXZlbnQuZ2V0V2luZG93UmVjdCgpOwotICAgICAgICAgICAgUG9pbnQgbG9jID0gcmVjdC5nZXRMb2NhdGlvbigpOwotICAgICAgICAgICAgaW50IG1hc2s7Ci0KLSAgICAgICAgICAgIHN3aXRjaCAoZXZlbnQuZ2V0RXZlbnRJZCgpKSB7Ci0gICAgICAgICAgICBjYXNlIE5hdGl2ZUV2ZW50LklEX0JPVU5EU19DSEFOR0VEOgotICAgICAgICAgICAgICAgIG1hc2sgPSAwOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBDb21wb25lbnRFdmVudC5DT01QT05FTlRfTU9WRUQ6Ci0gICAgICAgICAgICAgICAgbWFzayA9IE5hdGl2ZVdpbmRvdy5CT1VORFNfTk9TSVpFOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBDb21wb25lbnRFdmVudC5DT01QT05FTlRfUkVTSVpFRDoKLSAgICAgICAgICAgICAgICBtYXNrID0gTmF0aXZlV2luZG93LkJPVU5EU19OT01PVkU7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4xMkU9VW5rbm93biBjb21wb25lbnQgZXZlbnQgaWQuCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTJFIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vPz8/QVdUCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgaWYgKCEoc3JjIGluc3RhbmNlb2YgV2luZG93KSkgewotICAgICAgICAgICAgICAgIENvbXBvbmVudCBjb21wVG8gPSBzcmMuZ2V0UGFyZW50KCk7Ci0gICAgICAgICAgICAgICAgQ29tcG9uZW50IGNvbXBGcm9tID0gc3JjLmdldEhXQW5jZXN0b3IoKTsKLQotICAgICAgICAgICAgICAgIGlmICgoY29tcFRvICE9IG51bGwpICYmIChjb21wRnJvbSAhPSBudWxsKSkgewotICAgICAgICAgICAgICAgICAgICBsb2MgPSBNb3VzZURpc3BhdGNoZXIuY29udmVydFBvaW50KGNvbXBGcm9tLCBsb2MsIGNvbXBUbyk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBpbnQgd2luZG93U3RhdGUgPSBldmVudC5nZXRXaW5kb3dTdGF0ZSgpOwotCi0gICAgICAgICAgICAgICAgaWYgKCh3aW5kb3dTdGF0ZSA+PSAwKSAmJiAoc3JjIGluc3RhbmNlb2YgRnJhbWUpKSB7Ci0gICAgICAgICAgICAgICAgICAgICgoRnJhbWUpIHNyYykudXBkYXRlRXh0ZW5kZWRTdGF0ZSh3aW5kb3dTdGF0ZSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgc3JjLnNldEJvdW5kcyhsb2MueCwgbG9jLnksIHJlY3Qud2lkdGgsIHJlY3QuaGVpZ2h0LCBtYXNrLCBmYWxzZSk7Ci0gICAgICAgICAgICAqLwotICAgICAgICAgICAgCi0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoZSBkaXNwYXRjaGVyIG9mIHRoZSBrZXlib2FyZCBldmVudHMKLSAgICAgKi8KLSAgICBmaW5hbCBjbGFzcyBLZXlEaXNwYXRjaGVyIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSGFuZGxlIHRoZSBrZXlib2FyZCBldmVudCB1c2luZyB0aGUgS2V5Ym9hcmRGb2N1c01hbmFnZXIKLSAgICAgICAgICogQHBhcmFtIHNyYyAtIHRoZSBjb21wb25lbnQgcmVjZWl2aW5nIHRoZSBldmVudAotICAgICAgICAgKiBAcGFyYW0gZXZlbnQgLSB0aGUgbmF0aXZlIGV2ZW50Ci0gICAgICAgICAqIEByZXR1cm4gLSBhcyBpbiBEaXNwYXRjaGVyLm9uRXZlbnQoKQotICAgICAgICAgKiBAc2VlIERpc3BhdGNoZXIjb25FdmVudChOYXRpdmVFdmVudCkKLSAgICAgICAgICovCi0gICAgICAgIGJvb2xlYW4gZGlzcGF0Y2goQ29tcG9uZW50IHNyYywgTmF0aXZlRXZlbnQgZXZlbnQpIHsKLSAgICAgICAgICAgIGludCBpZCA9IGV2ZW50LmdldEV2ZW50SWQoKTsKLSAgICAgICAgICAgIGludCBtb2RpZmllcnMgPSBldmVudC5nZXRJbnB1dE1vZGlmaWVycygpOwotICAgICAgICAgICAgaW50IGxvY2F0aW9uID0gZXZlbnQuZ2V0S2V5TG9jYXRpb24oKTsKLSAgICAgICAgICAgIGludCBjb2RlID0gZXZlbnQuZ2V0VktleSgpOwotICAgICAgICAgICAgU3RyaW5nQnVmZmVyIGNoYXJzID0gZXZlbnQuZ2V0S2V5Q2hhcnMoKTsKLSAgICAgICAgICAgIGludCBjaGFyc0xlbmd0aCA9IGNoYXJzLmxlbmd0aCgpOwotICAgICAgICAgICAgbG9uZyB0aW1lID0gZXZlbnQuZ2V0VGltZSgpOwotICAgICAgICAgICAgY2hhciBrZXlDaGFyID0gZXZlbnQuZ2V0TGFzdENoYXIoKTsKLQotICAgICAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgICAgIC8qCi0gICAgICAgICAgICBpZiAoc3JjID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAvL3JldGFyZ2V0IGZvY3VzIHByb3h5IGtleSBldmVudHMgdG8gZm9jdXNPd25lcjoKLSAgICAgICAgICAgICAgICBXaW5kb3cgZm9jdXNQcm94eU93bmVyID0gdG9vbGtpdC5nZXRGb2N1c1Byb3h5T3duZXJCeUlkKGV2ZW50Ci0gICAgICAgICAgICAgICAgICAgICAgICAuZ2V0V2luZG93SWQoKSk7Ci0gICAgICAgICAgICAgICAgaWYgKGZvY3VzUHJveHlPd25lciA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgc3JjID0gS2V5Ym9hcmRGb2N1c01hbmFnZXIuYWN0dWFsRm9jdXNPd25lcjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgICovCi0KLSAgICAgICAgICAgIEV2ZW50UXVldWUgZXZlbnRRdWV1ZSA9IHRvb2xraXQuZ2V0U3lzdGVtRXZlbnRRdWV1ZUltcGwoKTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgaWYgKHNyYyAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgZXZlbnRRdWV1ZS5wb3N0RXZlbnQobmV3IEtleUV2ZW50KHNyYywgaWQsIHRpbWUsIG1vZGlmaWVycywKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvZGUsIGtleUNoYXIsIGxvY2F0aW9uKSk7Ci0gICAgICAgICAgICAgICAgLy8gS0VZX1RZUEVEIGdvZXMgYWZ0ZXIgS0VZX1BSRVNTRUQKLSAgICAgICAgICAgICAgICBpZiAoaWQgPT0gS2V5RXZlbnQuS0VZX1BSRVNTRUQpIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBjaGFyc0xlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBrZXlDaGFyID0gY2hhcnMuY2hhckF0KGkpOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGtleUNoYXIgIT0gS2V5RXZlbnQuQ0hBUl9VTkRFRklORUQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBldmVudFF1ZXVlLnBvc3RFdmVudChuZXcgS2V5RXZlbnQoc3JjLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgS2V5RXZlbnQuS0VZX1RZUEVELCB0aW1lLCBtb2RpZmllcnMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBLZXlFdmVudC5WS19VTkRFRklORUQsIGtleUNoYXIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBLZXlFdmVudC5LRVlfTE9DQVRJT05fVU5LTk9XTikpOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldGFyZ2V0cyB0aGUgbW91c2UgZXZlbnRzIHRvIHRoZSBncmFiIG93bmVyIHdoZW4gbW91c2UgaXMgZ3JhYmJlZCwKLSAgICAgKiBncmFiIGFuZCB1bmdyYWIgbW91c2Ugd2hlbiBtb3VzZSBidXR0b25zIGFyZSBwcmVzc2VkIGFuZCByZWxlYXNlZAotICAgICAqLwotCi0gICAgc3RhdGljIGZpbmFsIGNsYXNzIE1vdXNlR3JhYk1hbmFnZXIgewotCi0gICAgICAgIC8qKiAKLSAgICAgICAgICogVGhlIHRvcC1sZXZlbCB3aW5kb3cgaG9sZGluZyB0aGUgbW91c2UgZ3JhYiAKLSAgICAgICAgICogdGhhdCB3YXMgZXhwbGljaXRseSBzdGFydGVkIGJ5IHN0YXJ0R3JhYigpIG1ldGhvZAotICAgICAgICAgKi8KLSAgICAgICAgLy8/Pz9BV1Q6IHByaXZhdGUgV2luZG93IG5hdGl2ZUdyYWJPd25lciA9IG51bGw7Ci0gICAgICAgIC8qKiAKLSAgICAgICAgICogVGhlIGNvbXBvbmVudCB0aGF0IG93bnMgdGhlIHN5bnRoZXRpYyAKLSAgICAgICAgICogbW91c2UgZ3JhYiB3aGlsZSBhdCBsZWFzdCBvbmUgb2YgdGhlCi0gICAgICAgICAqIG1vdXNlIGJ1dHRvbnMgaXMgcHJlc3NlZAotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBDb21wb25lbnQgc3ludGhldGljR3JhYk93bmVyID0gbnVsbDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogUHJldmlvdXMgdmFsdWUgb2Ygc3ludGhldGljR3JhYk93bmVyCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIENvbXBvbmVudCBsYXN0U3ludGhldGljR3JhYk93bmVyID0gbnVsbDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogTnVtYmVyIG9mIG1vdXNlIGJ1dHRvbnMgY3VycmVudGx5IHByZXNzZWQKLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgaW50IHN5bnRoZXRpY0dyYWJEZXB0aCA9IDA7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjYWxsYmFjayB0byBiZSBjYWxsZWQgd2hlbiB0aGUgZXhwbGljaXQgbW91c2UgZ3JhYiBlbmRzCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIFJ1bm5hYmxlIHdoZW5DYW5jZWxlZDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogRXhwbGljaXRseSBzdGFydCB0aGUgbW91c2UgZ3JhYgotICAgICAgICAgKiBAcGFyYW0gZ3JhYldpbmRvdyAtIHRoZSB3aW5kb3cgdGhhdCB3aWxsIG93biB0aGUgZ3JhYgotICAgICAgICAgKiBAcGFyYW0gd2hlbkNhbmNlbGVkIC0gdGhlIGNhbGxiYWNrIHRvIGNhbGwgd2hlbiB0aGUgZ3JhYiBlbmRzLiAKLSAgICAgICAgICogVGhpcyBwYXJhbWV0ZXIgY2FuIGJlIG51bGwKLSAgICAgICAgICovCi0gICAgICAgIC8vPz8/QVdUCi0gICAgICAgIC8qCi0gICAgICAgIHZvaWQgc3RhcnRHcmFiKFdpbmRvdyBncmFiV2luZG93LCBSdW5uYWJsZSB3aGVuQ2FuY2VsZWQpIHsKLQotICAgICAgICAgICAgaWYgKG5hdGl2ZUdyYWJPd25lciAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjEyRj1BdHRlbXB0IHRvIHN0YXJ0IG5lc3RlZCBtb3VzZSBncmFiCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTJGIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIE5hdGl2ZVdpbmRvdyB3aW4gPSBncmFiV2luZG93LmdldE5hdGl2ZVdpbmRvdygpOwotICAgICAgICAgICAgaWYgKHdpbiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjEzMD1BdHRlbXB0IHRvIGdyYWIgbW91c2UgaW4gbm90IGRpc3BsYXlhYmxlIHdpbmRvdwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEzMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBuYXRpdmVHcmFiT3duZXIgPSBncmFiV2luZG93OwotICAgICAgICAgICAgdGhpcy53aGVuQ2FuY2VsZWQgPSB3aGVuQ2FuY2VsZWQ7Ci0gICAgICAgICAgICB3aW4uZ3JhYk1vdXNlKCk7Ci0gICAgICAgIH0KLSAgICAgICAgKi8KLQotICAgICAgICAvKioKLSAgICAgICAgICogRW5kcyB0aGUgZXhwbGljaXQgbW91c2UgZ3JhYi4gSWYgdGhlIG5vbi1udWxsIGNhbGxiYWNrIHdhcyBwcm92aWRlZAotICAgICAgICAgKiBpbiB0aGUgc3RhcnRHcmFiKCkgbWV0aG9kLCB0aGlzIGNhbGxiYWNrIGlzIGNhbGxlZCAKLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgZW5kR3JhYigpIHsKLSAgICAgICAgICAgIC8vPz8/QVdUCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgaWYgKG5hdGl2ZUdyYWJPd25lciA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBXaW5kb3cgZ3JhYldpbmRvdyA9IG5hdGl2ZUdyYWJPd25lcjsKLSAgICAgICAgICAgIG5hdGl2ZUdyYWJPd25lciA9IG51bGw7Ci0gICAgICAgICAgICBOYXRpdmVXaW5kb3cgd2luID0gZ3JhYldpbmRvdy5nZXROYXRpdmVXaW5kb3coKTsKLQotICAgICAgICAgICAgaWYgKHdpbiAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgd2luLnVuZ3JhYk1vdXNlKCk7Ci0gICAgICAgICAgICAgICAgaWYgKHdoZW5DYW5jZWxlZCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHdoZW5DYW5jZWxlZC5ydW4oKTsKLSAgICAgICAgICAgICAgICAgICAgd2hlbkNhbmNlbGVkID0gbnVsbDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAqLwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEVuZHMgYm90aCBleHBsaWNpdCBhbmQgc3ludGhldGljIGdyYW5zIAotICAgICAgICAgKiBAcmV0dXJuIC0gYWx3YXlzIHJldHVybnMgZmFsc2UKLSAgICAgICAgICovCi0gICAgICAgIGJvb2xlYW4gb25HcmFiQ2FuY2VsZWQoKSB7Ci0gICAgICAgICAgICBlbmRHcmFiKCk7Ci0gICAgICAgICAgICByZXNldFN5bnRoZXRpY0dyYWIoKTsKLQotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFN0YXJ0cyB0aGUgc3ludGhldGljIG1vdXNlIGdyYWIsIGluY3JlYXNlcyB0aGUgY291bnRlciAKLSAgICAgICAgICogb2YgY3VycmVudGx5IHByZXNzZWQgbW91c2UgYnV0dG9ucwotICAgICAgICAgKiBAcGFyYW0gc291cmNlIC0gdGhlIGNvbXBvbmVudCB3aGVyZSBtb3VzZSBwcmVzcyBldmVudCBvY2N1cmVkCi0gICAgICAgICAqIEByZXR1cm4gLSB0aGUgY29tcG9uZW50IHRoYXQgb3ducyB0aGUgc3ludGhldGljIGdyYWIKLSAgICAgICAgICovCi0gICAgICAgIENvbXBvbmVudCBvbk1vdXNlUHJlc3NlZChDb21wb25lbnQgc291cmNlKSB7Ci0gICAgICAgICAgICBpZiAoc3ludGhldGljR3JhYkRlcHRoID09IDApIHsKLSAgICAgICAgICAgICAgICBzeW50aGV0aWNHcmFiT3duZXIgPSBzb3VyY2U7Ci0gICAgICAgICAgICAgICAgbGFzdFN5bnRoZXRpY0dyYWJPd25lciA9IHNvdXJjZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHN5bnRoZXRpY0dyYWJEZXB0aCsrOwotCi0gICAgICAgICAgICByZXR1cm4gc3ludGhldGljR3JhYk93bmVyOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIERlY3JlYXNlcyB0aGUgY291bnRlciBvZiBjdXJyZW50bHkgcHJlc3NlZCBtb3VzZSBidXR0b25zLAotICAgICAgICAgKiBlbmRzIHRoZSBzeW50aGV0aWMgbW91c2UgZ3JhYiwgd2hlbiB0aGlzIGNvdW50ZXIgYmVjb21lcyB6ZXJvCi0gICAgICAgICAqIEBwYXJhbSBzb3VyY2UgLSB0aGUgY29tcG9uZW50IHdoZXJlIG1vdXNlIHByZXNzIGV2ZW50IG9jY3VyZWQKLSAgICAgICAgICogQHJldHVybiAtIHRoZSBjb21wb25lbnQgdGhhdCBvd25zIHRoZSBzeW50aGV0aWMgZ3JhYiwgCi0gICAgICAgICAqIG9yIHNvdXJjZSBwYXJhbWV0ZXIgaWYgbW91c2UgZ3JhYiB3YXMgcmVsZWFzZWQKLSAgICAgICAgICovCi0gICAgICAgIENvbXBvbmVudCBvbk1vdXNlUmVsZWFzZWQoQ29tcG9uZW50IHNvdXJjZSkgewotICAgICAgICAgICAgQ29tcG9uZW50IHJldCA9IHNvdXJjZTsKLQotICAgICAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgICAgIC8qCi0gICAgICAgICAgICBpZiAoc3ludGhldGljR3JhYk93bmVyICE9IG51bGwgJiYgbmF0aXZlR3JhYk93bmVyID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICByZXQgPSBzeW50aGV0aWNHcmFiT3duZXI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAqLwotICAgICAgICAgICAgc3ludGhldGljR3JhYkRlcHRoLS07Ci0gICAgICAgICAgICBpZiAoc3ludGhldGljR3JhYkRlcHRoIDw9IDApIHsKLSAgICAgICAgICAgICAgICByZXNldFN5bnRoZXRpY0dyYWIoKTsKLSAgICAgICAgICAgICAgICBsYXN0U3ludGhldGljR3JhYk93bmVyID0gbnVsbDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIHJldDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBVcGRhdGUgdGhlIHN0YXRlIG9mIHN5bnRoZXRpYyBvdXNlIGdyYW0gCi0gICAgICAgICAqIHdoZW4gdGhlIG1vdXNlIGlzIG1vdmVkL2RyYWdnZWQKLSAgICAgICAgICogQHBhcmFtIGV2ZW50IC0gdGhlIG5hdGl2ZSBldmVudAotICAgICAgICAgKi8KLSAgICAgICAgdm9pZCBwcmVwcm9jZXNzRXZlbnQoTmF0aXZlRXZlbnQgZXZlbnQpIHsKLSAgICAgICAgICAgIGludCBpZCA9IGV2ZW50LmdldEV2ZW50SWQoKTsKLSAgICAgICAgICAgIHN3aXRjaCAoaWQpIHsKLSAgICAgICAgICAgIGNhc2UgTW91c2VFdmVudC5NT1VTRV9NT1ZFRDoKLSAgICAgICAgICAgICAgICBpZiAoc3ludGhldGljR3JhYk93bmVyICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgc3ludGhldGljR3JhYk93bmVyID0gbnVsbDsKLSAgICAgICAgICAgICAgICAgICAgc3ludGhldGljR3JhYkRlcHRoID0gMDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKGxhc3RTeW50aGV0aWNHcmFiT3duZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBsYXN0U3ludGhldGljR3JhYk93bmVyID0gbnVsbDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICBjYXNlIE1vdXNlRXZlbnQuTU9VU0VfRFJBR0dFRDoKLSAgICAgICAgICAgICAgICBpZiAoc3ludGhldGljR3JhYk93bmVyID09IG51bGwKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIGxhc3RTeW50aGV0aWNHcmFiT3duZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBzeW50aGV0aWNHcmFiT3duZXIgPSBsYXN0U3ludGhldGljR3JhYk93bmVyOwotICAgICAgICAgICAgICAgICAgICBzeW50aGV0aWNHcmFiRGVwdGggPSAwOwotICAgICAgICAgICAgICAgICAgICBpbnQgbWFzayA9IGV2ZW50LmdldElucHV0TW9kaWZpZXJzKCk7Ci0gICAgICAgICAgICAgICAgICAgIHN5bnRoZXRpY0dyYWJEZXB0aCArPSAobWFzayAmIElucHV0RXZlbnQuQlVUVE9OMV9ET1dOX01BU0spICE9IDAgPyAxCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgOiAwOwotICAgICAgICAgICAgICAgICAgICBzeW50aGV0aWNHcmFiRGVwdGggKz0gKG1hc2sgJiBJbnB1dEV2ZW50LkJVVFRPTjJfRE9XTl9NQVNLKSAhPSAwID8gMQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIDogMDsKLSAgICAgICAgICAgICAgICAgICAgc3ludGhldGljR3JhYkRlcHRoICs9IChtYXNrICYgSW5wdXRFdmVudC5CVVRUT04zX0RPV05fTUFTSykgIT0gMCA/IDEKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICA6IDA7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEByZXR1cm4gdGhlIGNvbXBvbmVudCB0aGF0IGN1cnJlbnRseSBvd25zIHRoZSBzeW50aGV0aWMgZ3JhYiAKLSAgICAgICAgICovCi0gICAgICAgIENvbXBvbmVudCBnZXRTeW50aGV0aWNHcmFiT3duZXIoKSB7Ci0gICAgICAgICAgICByZXR1cm4gc3ludGhldGljR3JhYk93bmVyOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIGVuZHMgc3ludGhldGljIGdyYWIKLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgdm9pZCByZXNldFN5bnRoZXRpY0dyYWIoKSB7Ci0gICAgICAgICAgICBzeW50aGV0aWNHcmFiT3duZXIgPSBudWxsOwotICAgICAgICAgICAgc3ludGhldGljR3JhYkRlcHRoID0gMDsKLSAgICAgICAgfQotCi0gICAgfQotICAgIAotICAgIC8qKgotICAgICAqIERpc3BhdGNoZXMgbmF0aXZlIGV2ZW50cyByZWxhdGVkIHRvIHRoZSBwb3AtdXAgYm94ZXMgCi0gICAgICogKHRoZSBub24tY29tcG9uZW50IHdpbmRvd3Mgc3VjaCBhcyBtZW51cyBhbmQgZHJvcCBsaXN0cykKLSAgICAgKi8KLS8vICAgIGZpbmFsIGNsYXNzIFBvcHVwRGlzcGF0Y2hlciB7Ci0vLwotLy8gICAgICAgIHByaXZhdGUgUG9wdXBCb3ggYWN0aXZlUG9wdXA7Ci0vLwotLy8gICAgICAgIHByaXZhdGUgUG9wdXBCb3ggdW5kZXJDdXJzb3I7Ci0vLwotLy8gICAgICAgIHByaXZhdGUgZmluYWwgTW91c2VHcmFiIGdyYWIgPSBuZXcgTW91c2VHcmFiKCk7Ci0vLwotLy8gICAgICAgIC8qKgotLy8gICAgICAgICAqIEhhbmRsZXMgdGhlIG1vdXNlIGdyYWIgZm9yIHBvcC11cCBib3hlcwotLy8gICAgICAgICAqLwotLy8gICAgICAgIHByaXZhdGUgZmluYWwgY2xhc3MgTW91c2VHcmFiIHsKLS8vICAgICAgICAgICAgcHJpdmF0ZSBpbnQgZGVwdGg7Ci0vLwotLy8gICAgICAgICAgICBwcml2YXRlIFBvcHVwQm94IG93bmVyOwotLy8KLS8vICAgICAgICAgICAgcHJpdmF0ZSBmaW5hbCBQb2ludCBzdGFydCA9IG5ldyBQb2ludCgpOwotLy8KLS8vICAgICAgICAgICAgLyoqCi0vLyAgICAgICAgICAgICAqIFN0YXJ0cyB0aGUgZ3JhYiB3aGVuIG1vdXNlIGlzIHByZXNzZWQKLS8vICAgICAgICAgICAgICogQHBhcmFtIHNyYyAtIHRoZSBwb3AtdXAgYm94IHdoZXJlIG1vdXNlIGV2ZW50IGhhcyBvY2N1cmVkCi0vLyAgICAgICAgICAgICAqIEBwYXJhbSB3aGVyZSAtIHRoZSBtb3VzZSBwb2ludGVyIGxvY2F0aW9uCi0vLyAgICAgICAgICAgICAqIEByZXR1cm4gLSB0aGUgZ3JhYiBvd25lcgotLy8gICAgICAgICAgICAgKi8KLS8vICAgICAgICAgICAgUG9wdXBCb3ggbW91c2VQcmVzc2VkKFBvcHVwQm94IHNyYywgUG9pbnQgd2hlcmUpIHsKLS8vICAgICAgICAgICAgICAgIGlmIChkZXB0aCA9PSAwKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgb3duZXIgPSBzcmM7Ci0vLyAgICAgICAgICAgICAgICAgICAgc3RhcnQuc2V0TG9jYXRpb24od2hlcmUpOwotLy8gICAgICAgICAgICAgICAgfQotLy8gICAgICAgICAgICAgICAgZGVwdGgrKzsKLS8vICAgICAgICAgICAgICAgIHJldHVybiBvd25lcjsKLS8vICAgICAgICAgICAgfQotLy8KLS8vICAgICAgICAgICAgLyoqCi0vLyAgICAgICAgICAgICAqIEVuZHMgdGhlIGdyYWIgd2hlbiBhbGwgbW91c2VidXR0b25zIGFyZSByZWxlYXNlZAotLy8gICAgICAgICAgICAgKiBAcGFyYW0gc3JjIC0gdGhlIHBvcC11cCBib3ggd2hlcmUgbW91c2UgZXZlbnQgaGFzIG9jY3VyZWQKLS8vICAgICAgICAgICAgICogQHBhcmFtIHdoZXJlIC0gdGhlIG1vdXNlIHBvaW50ZXIgbG9jYXRpb24KLS8vICAgICAgICAgICAgICogQHJldHVybiAtIHRoZSBncmFiIG93bmVyLCBvciBzcmMgcGFyYW1ldGVyIGlmIHRoZSBncmFiIGhhcyBlbmRlZAotLy8gICAgICAgICAgICAgKi8KLS8vICAgICAgICAgICAgUG9wdXBCb3ggbW91c2VSZWxlYXNlZChQb3B1cEJveCBzcmMsIFBvaW50IHdoZXJlKSB7Ci0vLyAgICAgICAgICAgICAgICBQb3B1cEJveCByZXQgPSAob3duZXIgIT0gbnVsbCkgPyBvd25lciA6IHNyYzsKLS8vICAgICAgICAgICAgICAgIGlmIChkZXB0aCA9PSAwKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHJldDsKLS8vICAgICAgICAgICAgICAgIH0KLS8vICAgICAgICAgICAgICAgIGRlcHRoLS07Ci0vLyAgICAgICAgICAgICAgICBpZiAoZGVwdGggPT0gMCkgewotLy8gICAgICAgICAgICAgICAgICAgIFBvcHVwQm94IHRndCA9IG93bmVyOwotLy8gICAgICAgICAgICAgICAgICAgIG93bmVyID0gbnVsbDsKLS8vICAgICAgICAgICAgICAgICAgICBpZiAodGd0ICE9IG51bGwgJiYgc3JjID09IG51bGwpIHsKLS8vICAgICAgICAgICAgICAgICAgICAgICAgUG9pbnQgYSA9IG5ldyBQb2ludChzdGFydCk7Ci0vLyAgICAgICAgICAgICAgICAgICAgICAgIFBvaW50IGIgPSBuZXcgUG9pbnQod2hlcmUpOwotLy8gICAgICAgICAgICAgICAgICAgICAgICBQb2ludCBwb3MgPSB0Z3QuZ2V0U2NyZWVuTG9jYXRpb24oKTsKLS8vICAgICAgICAgICAgICAgICAgICAgICAgYS50cmFuc2xhdGUoLXBvcy54LCAtcG9zLnkpOwotLy8gICAgICAgICAgICAgICAgICAgICAgICBiLnRyYW5zbGF0ZSgtcG9zLngsIC1wb3MueSk7Ci0vLyAgICAgICAgICAgICAgICAgICAgICAgIGlmICh0Z3QuY2xvc2VPblVuZ3JhYihhLCBiKSkgewotLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0vLyAgICAgICAgICAgICAgICAgICAgICAgIH0KLS8vICAgICAgICAgICAgICAgICAgICB9Ci0vLyAgICAgICAgICAgICAgICB9Ci0vLyAgICAgICAgICAgICAgICByZXR1cm4gcmV0OwotLy8gICAgICAgICAgICB9Ci0vLwotLy8gICAgICAgICAgICAvKioKLS8vICAgICAgICAgICAgICogU2V0IHRoZSBncmFiIG93bmVyIHRvIG51bGwKLS8vICAgICAgICAgICAgICovCi0vLyAgICAgICAgICAgIHZvaWQgcmVzZXQoKSB7Ci0vLyAgICAgICAgICAgICAgICBkZXB0aCA9IDA7Ci0vLyAgICAgICAgICAgICAgICBvd25lciA9IG51bGw7Ci0vLyAgICAgICAgICAgICAgICBzdGFydC5zZXRMb2NhdGlvbigwLCAwKTsKLS8vICAgICAgICAgICAgfQotLy8KLS8vICAgICAgICAgICAgLyoqCi0vLyAgICAgICAgICAgICAqIEByZXR1cm4gLSB0aGUgcG9wLXVwIGJveCBjdXJyZW50bHkgb3duaW5nIHRoZSBncmFiCi0vLyAgICAgICAgICAgICAqLwotLy8gICAgICAgICAgICBwdWJsaWMgUG9wdXBCb3ggZ2V0T3duZXIoKSB7Ci0vLyAgICAgICAgICAgICAgICByZXR1cm4gb3duZXI7Ci0vLyAgICAgICAgICAgIH0KLS8vICAgICAgICB9Ci0vLwotLy8gICAgICAgIC8qKgotLy8gICAgICAgICAqIENhbGwgdGhlIG1vdXNlIGV2ZW50IGhhbmRsZXIgb2YgdGhlIHBvcC11cCBib3gKLS8vICAgICAgICAgKiBAcGFyYW0gc3JjIC0gdGhlIHBvcC11cCBib3ggd2hlcmUgdGhlIG1vdXNlIGV2ZW50IG9jY3VyZWQKLS8vICAgICAgICAgKiBAcGFyYW0gZXZlbnRJZCAtIHRoZSBldmVudCBJRCwgb25lIG9mIE1vdXNlRXZlbnQuTU9VU0VfKiBjb25zdGFudHMKLS8vICAgICAgICAgKiBAcGFyYW0gd2hlcmUgLSB0aGUgbW91c2UgcG9pbnRlciBsb2NhdGlvbgotLy8gICAgICAgICAqIEBwYXJhbSBldmVudCAtIG5hdGl2ZSBldmVudAotLy8gICAgICAgICAqLwotLy8gICAgICAgIHByaXZhdGUgdm9pZCBtb3VzZUV2ZW50KFBvcHVwQm94IHNyYywgaW50IGV2ZW50SWQsIFBvaW50IHdoZXJlLAotLy8gICAgICAgICAgICAgICAgTmF0aXZlRXZlbnQgZXZlbnQpIHsKLS8vICAgICAgICAgICAgUG9pbnQgcG9zID0gc3JjLmdldFNjcmVlbkxvY2F0aW9uKCk7Ci0vLyAgICAgICAgICAgIHBvcy5zZXRMb2NhdGlvbih3aGVyZS54IC0gcG9zLngsIHdoZXJlLnkgLSBwb3MueSk7Ci0vLwotLy8gICAgICAgICAgICBzcmMub25Nb3VzZUV2ZW50KGV2ZW50SWQsIHBvcywgZXZlbnQuZ2V0TW91c2VCdXR0b24oKSwgZXZlbnQKLS8vICAgICAgICAgICAgICAgICAgICAuZ2V0VGltZSgpLCBldmVudC5nZXRJbnB1dE1vZGlmaWVycygpLCBldmVudAotLy8gICAgICAgICAgICAgICAgICAgIC5nZXRXaGVlbFJvdGF0aW9uKCkpOwotLy8gICAgICAgIH0KLS8vCi0vLyAgICAgICAgLyoqCi0vLyAgICAgICAgICogSGFuZGxlIHRoZSBuYXRpdmUgZXZlbnQgdGFyZ2V0ZWQgYnkgYSBwb3AtdXAgYm94LiBUaGlzIGNvdWxkIGJlIAotLy8gICAgICAgICAqIHBhaW50IGV2ZW50LCBtb3VzZSBvciBrZXlib2FyZCBldmVudC4KLS8vICAgICAgICAgKiBAcGFyYW0gZXZlbnQgLSB0aGUgbmF0aXZlIGV2ZW50Ci0vLyAgICAgICAgICogQHJldHVybiAtIGZhbHNlIGlmIHRoZSBldmVudCB3YXMgaGFuZGxlZCBhbmQgZG9lc24ndCAKLS8vICAgICAgICAgKiBuZWVkIHRoZSBmdXJ0aGVyIHByb2Nlc3Npbmc7IHRydWUgd2hlbiB0aGUgZnVydGhlciAKLS8vICAgICAgICAgKiBwcm9jZXNzaW5nIGlzIG5lZWRlZAotLy8gICAgICAgICAqLwotLy8gICAgICAgIGJvb2xlYW4gb25FdmVudChOYXRpdmVFdmVudCBldmVudCkgewotLy8gICAgICAgICAgICBQb3B1cEJveCBzcmMgPSB0b29sa2l0LmdldFBvcHVwQm94QnlJZChldmVudC5nZXRXaW5kb3dJZCgpKTsKLS8vICAgICAgICAgICAgaW50IGlkID0gZXZlbnQuZ2V0RXZlbnRJZCgpOwotLy8KLS8vICAgICAgICAgICAgaWYgKChpZCA9PSBQYWludEV2ZW50LlBBSU5UKSkgewotLy8gICAgICAgICAgICAgICAgaWYgKHNyYyAhPSBudWxsKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgc3JjLnBhaW50KGV2ZW50LmdldENsaXBSZWN0cygpKTsKLS8vICAgICAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLS8vICAgICAgICAgICAgICAgIH0KLS8vICAgICAgICAgICAgICAgIENvbXBvbmVudCBjID0gdG9vbGtpdC5nZXRDb21wb25lbnRCeUlkKGV2ZW50LmdldFdpbmRvd0lkKCkpOwotLy8gICAgICAgICAgICAgICAgaWYgKChjICE9IG51bGwpICYmIChjIGluc3RhbmNlb2YgRnJhbWUpKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgKChGcmFtZSkgYykucGFpbnRNZW51QmFyKGV2ZW50LmdldENsaXBSZWN0cygpKTsKLS8vICAgICAgICAgICAgICAgIH0KLS8vICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLS8vICAgICAgICAgICAgfQotLy8KLS8vICAgICAgICAgICAgaWYgKChpZCA+PSBNb3VzZUV2ZW50Lk1PVVNFX0ZJUlNUKSAmJiAoaWQgPD0gTW91c2VFdmVudC5NT1VTRV9MQVNUKSkgewotLy8gICAgICAgICAgICAgICAgUG9pbnQgd2hlcmUgPSBldmVudC5nZXRTY3JlZW5Qb3MoKTsKLS8vCi0vLyAgICAgICAgICAgICAgICBpZiAoc3JjICE9IHVuZGVyQ3Vyc29yKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgaWYgKHVuZGVyQ3Vyc29yICE9IG51bGwpIHsKLS8vICAgICAgICAgICAgICAgICAgICAgICAgbW91c2VFdmVudCh1bmRlckN1cnNvciwgTW91c2VFdmVudC5NT1VTRV9FWElURUQsIHdoZXJlLAotLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV2ZW50KTsKLS8vICAgICAgICAgICAgICAgICAgICB9Ci0vLyAgICAgICAgICAgICAgICAgICAgdW5kZXJDdXJzb3IgPSBzcmM7Ci0vLyAgICAgICAgICAgICAgICAgICAgaWYgKHVuZGVyQ3Vyc29yICE9IG51bGwpIHsKLS8vICAgICAgICAgICAgICAgICAgICAgICAgbW91c2VFdmVudCh1bmRlckN1cnNvciwgTW91c2VFdmVudC5NT1VTRV9FTlRFUkVELAotLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHdoZXJlLCBldmVudCk7Ci0vLyAgICAgICAgICAgICAgICAgICAgICAgIHVuZGVyQ3Vyc29yLnNldERlZmF1bHRDdXJzb3IoKTsKLS8vICAgICAgICAgICAgICAgICAgICB9Ci0vLyAgICAgICAgICAgICAgICB9Ci0vLyAgICAgICAgICAgICAgICBpZiAoaWQgPT0gTW91c2VFdmVudC5NT1VTRV9FWElURUQpIHsKLS8vICAgICAgICAgICAgICAgICAgICB1bmRlckN1cnNvciA9IG51bGw7Ci0vLyAgICAgICAgICAgICAgICB9Ci0vLwotLy8gICAgICAgICAgICAgICAgaWYgKChhY3RpdmVQb3B1cCA9PSBudWxsKSAmJiAoc3JjID09IG51bGwgfHwgIXNyYy5pc01lbnVCYXIoKSkpIHsKLS8vICAgICAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0vLyAgICAgICAgICAgICAgICB9Ci0vLwotLy8gICAgICAgICAgICAgICAgaWYgKGlkID09IE1vdXNlRXZlbnQuTU9VU0VfUFJFU1NFRCkgewotLy8gICAgICAgICAgICAgICAgICAgIHNyYyA9IGdyYWIubW91c2VQcmVzc2VkKHNyYywgd2hlcmUpOwotLy8gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChpZCA9PSBNb3VzZUV2ZW50Lk1PVVNFX1JFTEVBU0VEKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgc3JjID0gZ3JhYi5tb3VzZVJlbGVhc2VkKHNyYywgd2hlcmUpOwotLy8gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChzcmMgPT0gbnVsbCkgewotLy8gICAgICAgICAgICAgICAgICAgIHNyYyA9IGdyYWIuZ2V0T3duZXIoKTsKLS8vICAgICAgICAgICAgICAgIH0KLS8vCi0vLyAgICAgICAgICAgICAgICBQb3B1cEJveCB3YXNBY3RpdmUgPSBhY3RpdmVQb3B1cDsKLS8vCi0vLyAgICAgICAgICAgICAgICBpZiAoc3JjICE9IG51bGwpIHsKLS8vICAgICAgICAgICAgICAgICAgICBtb3VzZUV2ZW50KHNyYywgaWQsIHdoZXJlLCBldmVudCk7Ci0vLyAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNyYy5pc01lbnUoKSB8fCBzcmMuY29udGFpbnMod2hlcmUpOwotLy8gICAgICAgICAgICAgICAgfQotLy8KLS8vICAgICAgICAgICAgICAgIGlmICh3YXNBY3RpdmUgIT0gbnVsbCAmJiBhY3RpdmVQb3B1cCA9PSBudWxsKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHdhc0FjdGl2ZS5pc01lbnUoKTsKLS8vICAgICAgICAgICAgICAgIH0KLS8vCi0vLyAgICAgICAgICAgICAgICBpZiAoKGlkID09IE1vdXNlRXZlbnQuTU9VU0VfUFJFU1NFRCkKLS8vICAgICAgICAgICAgICAgICAgICAgICAgfHwgKGlkID09IE1vdXNlRXZlbnQuTU9VU0VfUkVMRUFTRUQpKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgYm9vbGVhbiBpc01lbnUgPSBhY3RpdmVQb3B1cC5pc01lbnUoKTsKLS8vICAgICAgICAgICAgICAgICAgICBkZWFjdGl2YXRlQWxsKCk7Ci0vLyAgICAgICAgICAgICAgICAgICAgcmV0dXJuICFpc01lbnU7Ci0vLyAgICAgICAgICAgICAgICB9Ci0vLyAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLS8vICAgICAgICAgICAgfQotLy8KLS8vICAgICAgICAgICAgaWYgKGFjdGl2ZVBvcHVwID09IG51bGwpIHsKLS8vICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLS8vICAgICAgICAgICAgfQotLy8KLS8vICAgICAgICAgICAgaWYgKChpZCA+PSBLZXlFdmVudC5LRVlfRklSU1QpICYmIChpZCA8PSBLZXlFdmVudC5LRVlfTEFTVCkpIHsKLS8vICAgICAgICAgICAgICAgIGJvb2xlYW4gaXNNZW51ID0gYWN0aXZlUG9wdXAuaXNNZW51KCk7Ci0vLyAgICAgICAgICAgICAgICBhY3RpdmVQb3B1cC5kaXNwYXRjaEtleUV2ZW50KGlkLCBldmVudC5nZXRWS2V5KCksIGV2ZW50Ci0vLyAgICAgICAgICAgICAgICAgICAgICAgIC5nZXRUaW1lKCksIGV2ZW50LmdldElucHV0TW9kaWZpZXJzKCkpOwotLy8KLS8vICAgICAgICAgICAgICAgIHJldHVybiBpc01lbnU7Ci0vLyAgICAgICAgICAgIH0KLS8vCi0vLyAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLS8vICAgICAgICB9Ci0vLwotLy8gICAgICAgIC8qKgotLy8gICAgICAgICAqIFJlbWVtYmVyIHRoZSBwb3AtdXAgYXMgYWN0aXZlIGFuZCBncmFiIHRoZSBtb3VzZSBvbiBpdAotLy8gICAgICAgICAqIEBwYXJhbSBwb3B1cCAtIHRoZSBwb3AtdXAgYm94IHRvIGFjdGl2YXRlCi0vLyAgICAgICAgICovCi0vLyAgICAgICAgdm9pZCBhY3RpdmF0ZShmaW5hbCBQb3B1cEJveCBwb3B1cCkgewotLy8gICAgICAgICAgICBpZiAoYWN0aXZlUG9wdXAgPT0gbnVsbCkgewotLy8KLS8vICAgICAgICAgICAgICAgIGFjdGl2ZVBvcHVwID0gcG9wdXA7Ci0vLyAgICAgICAgICAgICAgICBtb3VzZUdyYWJNYW5hZ2VyLnN0YXJ0R3JhYihwb3B1cC5nZXRPd25lcigpLCBuZXcgUnVubmFibGUoKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgcHVibGljIHZvaWQgcnVuKCkgewotLy8gICAgICAgICAgICAgICAgICAgICAgICBkZWFjdGl2YXRlKHBvcHVwKTsKLS8vICAgICAgICAgICAgICAgICAgICB9Ci0vLyAgICAgICAgICAgICAgICB9KTsKLS8vICAgICAgICAgICAgfQotLy8gICAgICAgIH0KLS8vCi0vLyAgICAgICAgLyoqCi0vLyAgICAgICAgICogRGVhY3RpdmF0ZSB0aGUgY3VycmVudGx5IGFjdGl2ZSBwb3AtdXAgYm94Ci0vLyAgICAgICAgICovCi0vLyAgICAgICAgdm9pZCBkZWFjdGl2YXRlQWxsKCkgewotLy8gICAgICAgICAgICBkZWFjdGl2YXRlKGFjdGl2ZVBvcHVwKTsKLS8vICAgICAgICB9Ci0vLwotLy8gICAgICAgIC8qKgotLy8gICAgICAgICAqIERlYWN0aXZhdGUgdGhlIHBvcC11cCBib3gsIGVuZCB0aGUgbW91c2UgZ3JhYgotLy8gICAgICAgICAqLwotLy8gICAgICAgIHZvaWQgZGVhY3RpdmF0ZShQb3B1cEJveCBwb3B1cCkgewotLy8gICAgICAgICAgICBncmFiLnJlc2V0KCk7Ci0vLwotLy8gICAgICAgICAgICBpZiAoYWN0aXZlUG9wdXAgIT0gbnVsbCAmJiBhY3RpdmVQb3B1cCA9PSBwb3B1cCkgewotLy8gICAgICAgICAgICAgICAgYWN0aXZlUG9wdXAgPSBudWxsOwotLy8gICAgICAgICAgICAgICAgbW91c2VHcmFiTWFuYWdlci5lbmRHcmFiKCk7Ci0vLyAgICAgICAgICAgICAgICBwb3B1cC5oaWRlKCk7Ci0vLyAgICAgICAgICAgICAgICB1bmRlckN1cnNvciA9IG51bGw7Ci0vLyAgICAgICAgICAgIH0KLS8vICAgICAgICB9Ci0vLwotLy8gICAgICAgIC8qKgotLy8gICAgICAgICAqIENoZWNrIHRoYXQgdGhlIHBvcC11cCBib3ggaXMgY3VycmVudGx5IGFjdGl2ZQotLy8gICAgICAgICAqIEBwYXJhbSBwb3B1cCAtIHRoZSBwb3AtdXAgYm94IHRvIGNoZWNrCi0vLyAgICAgICAgICogQHJldHVybiAtIHRydWUgaWYgYWN0aXZlCi0vLyAgICAgICAgICovCi0vLyAgICAgICAgYm9vbGVhbiBpc0FjdGl2ZShQb3B1cEJveCBwb3B1cCkgewotLy8gICAgICAgICAgICByZXR1cm4gKHBvcHVwID09IGFjdGl2ZVBvcHVwKSAmJiAocG9wdXAgIT0gbnVsbCk7Ci0vLyAgICAgICAgfQotLy8gICAgfQotCi19ClwgTm8gbmV3bGluZSBhdCBlbmQgb2YgZmlsZQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0Rpc3BsYXlNb2RlLmphdmEgYi9hd3QvamF2YS9hd3QvRGlzcGxheU1vZGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggODAyMTAxMC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvRGlzcGxheU1vZGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE2NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLS8qKgotICogVGhlIERpc3BsYXlNb2RlIGNsYXNzIGNvbnRhaW5zIHRoZSBiaXQgZGVwdGgsIGhlaWdodCwgd2lkdGggYW5kIHJlZnJlc2ggcmF0ZQotICogb2YgYSBHcmFwaGljc0RldmljZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBmaW5hbCBjbGFzcyBEaXNwbGF5TW9kZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgd2lkdGguCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBpbnQgd2lkdGg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGVpZ2h0LgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgaW50IGhlaWdodDsKLQotICAgIC8qKgotICAgICAqIFRoZSBiaXQgZGVwdGguCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBpbnQgYml0RGVwdGg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcmVmcmVzaCByYXRlLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgaW50IHJlZnJlc2hSYXRlOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZhbHVlIEJJVF9ERVBUSF9NVUxUSSBpbmRpY2F0ZXMgdGhlIGJpdCBkZXB0aAotICAgICAqLwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQklUX0RFUFRIX01VTFRJID0gLTE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUkVGUkVTSF9SQVRFX1VOS05PV04gaW5kaWNhdGVzIHRoZSByZWZyZXNoIHJhdGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgUkVGUkVTSF9SQVRFX1VOS05PV04gPSAwOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIG5ldyBEaXNwbGF5TW9kZSBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGRpc3BsYXkuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgZGlzcGxheS4KLSAgICAgKiBAcGFyYW0gYml0RGVwdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBiaXQgZGVwdGggb2YgdGhlIGRpc3BsYXkuCi0gICAgICogQHBhcmFtIHJlZnJlc2hSYXRlCi0gICAgICogICAgICAgICAgICB0aGUgcmVmcmVzaCByYXRlIG9mIHRoZSBkaXNwbGF5LgotICAgICAqLwotCi0gICAgcHVibGljIERpc3BsYXlNb2RlKGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGJpdERlcHRoLCBpbnQgcmVmcmVzaFJhdGUpIHsKLSAgICAgICAgdGhpcy53aWR0aCA9IHdpZHRoOwotICAgICAgICB0aGlzLmhlaWdodCA9IGhlaWdodDsKLSAgICAgICAgdGhpcy5iaXREZXB0aCA9IGJpdERlcHRoOwotICAgICAgICB0aGlzLnJlZnJlc2hSYXRlID0gcmVmcmVzaFJhdGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgaWYgdGhpcyBEaXNwbGF5TW9kZSBpcyBlcXVhbCB0byB0aGUgc3BlY2lmaWVkIG9iamVjdCBvciBub3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRtCi0gICAgICogICAgICAgICAgICB0aGUgT2JqZWN0IHRvIGJlIGNvbXBhcmVkLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCBvYmplY3QgaXMgYSBEaXNwbGF5TW9kZSB3aXRoIHRoZSBzYW1lIGRhdGEKLSAgICAgKiAgICAgICAgIHZhbHVlcyBhcyB0aGlzIERpc3BsYXlNb2RlLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IGRtKSB7Ci0gICAgICAgIGlmIChkbSBpbnN0YW5jZW9mIERpc3BsYXlNb2RlKSB7Ci0gICAgICAgICAgICByZXR1cm4gZXF1YWxzKChEaXNwbGF5TW9kZSlkbSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIGlmIHRoaXMgRGlzcGxheU1vZGUgaXMgZXF1YWwgdG8gdGhlIHNwZWNpZmllZCBEaXNwbGF5TW9kZSBvYmplY3QKLSAgICAgKiBvciBub3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRtCi0gICAgICogICAgICAgICAgICB0aGUgRGlzcGxheU1vZGUgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBhbGwgb2YgdGhlIGRhdGEgdmFsdWVzIG9mIHRoaXMgRGlzcGxheU1vZGUgYXJlIGVxdWFsIHRvCi0gICAgICogICAgICAgICB0aGUgdmFsdWVzIG9mIHRoZSBzcGVjaWZpZWQgRGlzcGxheU1vZGUgb2JqZWN0LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKERpc3BsYXlNb2RlIGRtKSB7Ci0gICAgICAgIGlmIChkbSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGRtLmJpdERlcHRoICE9IGJpdERlcHRoKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGRtLnJlZnJlc2hSYXRlICE9IHJlZnJlc2hSYXRlKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGRtLndpZHRoICE9IHdpZHRoKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGRtLmhlaWdodCAhPSBoZWlnaHQpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBiaXQgZGVwdGggb2YgdGhlIERpc3BsYXlNb2RlLCByZXR1cm5zIEJJVF9ERVBUSF9NVUxUSSB2YWx1ZSBpZgotICAgICAqIG11bHRpcGxlIGJpdCBkZXB0aHMgYXJlIHN1cHBvcnRlZCBpbiB0aGlzIGRpc3BsYXkgbW9kZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBiaXQgZGVwdGggb2YgdGhlIERpc3BsYXlNb2RlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0Qml0RGVwdGgoKSB7Ci0gICAgICAgIHJldHVybiBiaXREZXB0aDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBoZWlnaHQgb2YgdGhlIERpc3BsYXlNb2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhlaWdodCBvZiB0aGUgRGlzcGxheU1vZGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRIZWlnaHQoKSB7Ci0gICAgICAgIHJldHVybiBoZWlnaHQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcmVmcmVzaCByYXRlIG9mIHRoZSBEaXNwbGF5TW9kZSwgcmV0dXJucyBSRUZSRVNIX1JBVEVfVU5LTk9XTgotICAgICAqIHZhbHVlIGlmIHRoZSBpbmZvcm1hdGlvbiBpcyBub3QgYXZhaWxhYmxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHJlZnJlc2ggcmF0ZSBvZiB0aGUgRGlzcGxheU1vZGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRSZWZyZXNoUmF0ZSgpIHsKLSAgICAgICAgcmV0dXJuIHJlZnJlc2hSYXRlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHdpZHRoIG9mIHRoZSBEaXNwbGF5TW9kZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB3aWR0aCBvZiB0aGUgRGlzcGxheU1vZGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRXaWR0aCgpIHsKLSAgICAgICAgcmV0dXJuIHdpZHRoOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9FdmVudC5qYXZhIGIvYXd0L2phdmEvYXd0L0V2ZW50LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDIyNmE2MWYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0V2ZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1OTYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEbWl0cnkgQS4gRHVybmV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEuaW8uU2VyaWFsaXphYmxlOwotCi0vKioKLSAqIFRoZSBFdmVudCBjbGFzcyBpcyBvYnNvbGV0ZSBhbmQgaGFzIGJlZW4gcmVwbGFjZWQgYnkgQVdURXZlbnQgY2xhc3MuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgRXZlbnQgaW1wbGVtZW50cyBTZXJpYWxpemFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IHNlcmlhbFZlcnNpb25VSUQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gNTQ4ODkyMjUwOTQwMDUwNDcwM0w7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU0hJRlRfTUFTSyBpbmRpY2F0ZXMgdGhhdCB0aGUgU2hpZnQga2V5IGlzIGRvd24gd2hlbiB0aGUKLSAgICAgKiBldmVudCBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTSElGVF9NQVNLID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDVFJMX01BU0sgaW5kaWNhdGVzIHRoYXQgdGhlIENvbnRyb2wga2V5IGlzIGRvd24gd2hlbiB0aGUKLSAgICAgKiBldmVudCBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDVFJMX01BU0sgPSAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE1FVEFfTUFTSyBpbmRpY2F0ZXMgdGhhdCB0aGUgTWV0YSBrZXkgaXMgZG93biB3aGVuIHQgaGUKLSAgICAgKiBldmVudCBvY2N1cnJlZCAob3IgdGhlIHJpZ2h0IG1vdXNlIGJ1dHRvbikuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTUVUQV9NQVNLID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBBTFRfTUFTSyBpbmRpY2F0ZXMgdGhhdCB0aGUgQWx0IGtleSBpcyBkb3duIHdoZW4gdGhlIGV2ZW50Ci0gICAgICogb2NjdXJyZWQgKG9yIHRoZSBtaWRkbGUgbW91c2UgYnV0dG9uKS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBBTFRfTUFTSyA9IDg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgSE9NRSBpbmRpY2F0ZXMgSG9tZSBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSE9NRSA9IDEwMDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRU5EIGluZGljYXRlcyBFbmQga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEVORCA9IDEwMDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUEdVUCBpbmRpY2F0ZXMgUGFnZSBVcCBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgUEdVUCA9IDEwMDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUEdETiBpbmRpY2F0ZXMgUGFnZSBEb3duIGtleS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBQR0ROID0gMTAwMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBVUCBpbmRpY2F0ZXMgVXAga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFVQID0gMTAwNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBET1dOIGluZGljYXRlcyBEb3duIGtleS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBET1dOID0gMTAwNTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBMRUZUIGluZGljYXRlcyBMZWZ0IGtleS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBMRUZUID0gMTAwNjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBSSUdIVCBpbmRpY2F0ZXMgUmlnaHQga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFJJR0hUID0gMTAwNzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGMSBpbmRpY2F0ZXMgRjEga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEYxID0gMTAwODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGMiBpbmRpY2F0ZXMgRjIga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEYyID0gMTAwOTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGMyBpbmRpY2F0ZXMgRjMga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEYzID0gMTAxMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGNCBpbmRpY2F0ZXMgRjQga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEY0ID0gMTAxMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGNSBpbmRpY2F0ZXMgRjUga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEY1ID0gMTAxMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGNiBpbmRpY2F0ZXMgRjYga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEY2ID0gMTAxMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGNyBpbmRpY2F0ZXMgRjcga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEY3ID0gMTAxNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGOCBpbmRpY2F0ZXMgRjgga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEY4ID0gMTAxNTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGOSBpbmRpY2F0ZXMgRjkga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEY5ID0gMTAxNjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGMTAgaW5kaWNhdGVzIEYxMCBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRjEwID0gMTAxNzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGMTEgaW5kaWNhdGVzIEYxMSBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRjExID0gMTAxODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGMTIgaW5kaWNhdGVzIEYxMiBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRjEyID0gMTAxOTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBQUklOVF9TQ1JFRU4gaW5kaWNhdGVzIFByaW50IFNjcmVlbiBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgUFJJTlRfU0NSRUVOID0gMTAyMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTQ1JPTExfTE9DSyBpbmRpY2F0ZXMgU2Nyb2xsIExvY2sga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNDUk9MTF9MT0NLID0gMTAyMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDQVBTX0xPQ0sgaW5kaWNhdGVzIENhcHMgTG9jayBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ0FQU19MT0NLID0gMTAyMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBOVU1fTE9DSyBpbmRpY2F0ZXMgTnVtIExvY2sga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IE5VTV9MT0NLID0gMTAyMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBQQVVTRSBpbmRpY2F0ZXMgUGF1c2Uga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFBBVVNFID0gMTAyNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBJTlNFUlQgaW5kaWNhdGVzIEluc2VydCBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSU5TRVJUID0gMTAyNTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBFTlRFUiBpbmRpY2F0ZXMgRW50ZXIga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEVOVEVSID0gMTA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQkFDS19TUEFDRSBpbmRpY2F0ZXMgQmFjayBTcGFjZSBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQkFDS19TUEFDRSA9IDg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFCIGluZGljYXRlcyBUQWIga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBQiA9IDk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRVNDQVBFIGluZGljYXRlcyBFc2NhcGUga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEVTQ0FQRSA9IDI3OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IERFTEVURSBpbmRpY2F0ZXMgRGVsZXRlIGtleS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBERUxFVEUgPSAxMjc7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgV0lORE9XX0RFU1RST1kgaW5kaWNhdGVzIGFuIGV2ZW50IHdoZW4gdGhlIHVzZXIgaGFzIGFza2VkCi0gICAgICogdGhlIHdpbmRvdyBtYW5hZ2VyIHRvIGtpbGwgdGhlIHdpbmRvdy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXSU5ET1dfREVTVFJPWSA9IDIwMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBXSU5ET1dfRVhQT1NFIGluZGljYXRlcyBhbiBldmVudCB3aGVuIHRoZSB1c2VyIGhhcyBhc2tlZCB0aGUKLSAgICAgKiB3aW5kb3cgbWFuYWdlciB0byBleHBvc2UgdGhlIHdpbmRvdy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXSU5ET1dfRVhQT1NFID0gMjAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFdJTkRPV19JQ09OSUZZIGluZGljYXRlcyBhbiBldmVudCB3aGVuIHRoZSB1c2VyIGhhcyBhc2tlZAotICAgICAqIHRoZSB3aW5kb3cgbWFuYWdlciB0byBpY29uaWZ5IHRoZSB3aW5kb3cuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lORE9XX0lDT05JRlkgPSAyMDM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgV0lORE9XX0RFSUNPTklGWSBpbmRpY2F0ZXMgYW4gZXZlbnQgd2hlbiB0aGUgdXNlciBoYXMgYXNrZWQKLSAgICAgKiB0aGUgd2luZG93IG1hbmFnZXIgdG8gZGVpY29uaWZ5IHRoZSB3aW5kb3cuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lORE9XX0RFSUNPTklGWSA9IDIwNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBXSU5ET1dfTU9WRUQgaW5kaWNhdGVzIGFuIGV2ZW50IHdoZW4gdGhlIHVzZXIgaGFzIGFza2VkIHRoZQotICAgICAqIHdpbmRvdyBtYW5hZ2VyIHRvIG1vdmUgdGhlIHdpbmRvdy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXSU5ET1dfTU9WRUQgPSAyMDU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgS0VZX1BSRVNTIGluZGljYXRlcyBhbiBldmVudCB3aGVuIHRoZSB1c2VyIHByZXNzZXMgYSBub3JtYWwKLSAgICAgKiBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgS0VZX1BSRVNTID0gNDAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEtFWV9SRUxFQVNFIGluZGljYXRlcyBhbiBldmVudCB3aGVuIHRoZSB1c2VyIHJlbGVhc2VzIGEKLSAgICAgKiBub3JtYWwga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEtFWV9SRUxFQVNFID0gNDAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEtFWV9BQ1RJT04gaW5kaWNhdGVzIGFuIGV2ZW50IHdoZW4gdGhlIHVzZXIgcHJlc3NlZCBhCi0gICAgICogbm9uLUFTQ0lJIGFjdGlvbiBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgS0VZX0FDVElPTiA9IDQwMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBLRVlfQUNUSU9OX1JFTEVBU0UgaW5kaWNhdGVzIGFuIGV2ZW50IHdoZW4gdGhlIHVzZXIgcmVsZWFzZWQKLSAgICAgKiBhIG5vbi1BU0NJSSBhY3Rpb24ga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEtFWV9BQ1RJT05fUkVMRUFTRSA9IDQwNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBNT1VTRV9ET1dOIGluZGljYXRlcyBhbiBldmVudCB3aGVuIHRoZSB1c2VyIGhhcyBwcmVzc2VkIHRoZQotICAgICAqIG1vdXNlIGJ1dHRvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNT1VTRV9ET1dOID0gNTAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE1PVVNFX1VQIGluZGljYXRlcyBhbiBldmVudCB3aGVuIHRoZSB1c2VyIGhhcyByZWxlYXNlZCB0aGUKLSAgICAgKiBtb3VzZSBidXR0b24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTU9VU0VfVVAgPSA1MDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTU9VU0VfTU9WRSBpbmRpY2F0ZXMgYW4gZXZlbnQgd2hlbiB0aGUgdXNlciBoYXMgbW92ZWQgdGhlCi0gICAgICogbW91c2Ugd2l0aCBubyBidXR0b24gcHJlc3NlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNT1VTRV9NT1ZFID0gNTAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE1PVVNFX0VOVEVSIGluZGljYXRlcyBhbiBldmVudCB3aGVuIHRoZSBtb3VzZSBoYXMgZW50ZXJlZCBhCi0gICAgICogY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IE1PVVNFX0VOVEVSID0gNTA0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE1PVVNFX0VYSVQgaW5kaWNhdGVzIGFuIGV2ZW50IHdoZW4gdGhlIG1vdXNlIGhhcyBleGl0ZWQgYQotICAgICAqIGNvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNT1VTRV9FWElUID0gNTA1OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE1PVVNFX0RSQUcgaW5kaWNhdGVzIGFuIGV2ZW50IHdoZW4gdGhlIHVzZXIgaGFzIG1vdmVkIGEKLSAgICAgKiBtb3VzZSB3aXRoIHRoZSBwcmVzc2VkIGJ1dHRvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNT1VTRV9EUkFHID0gNTA2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNDUk9MTF9MSU5FX1VQIGluZGljYXRlcyBhbiBldmVudCB3aGVuIHRoZSB1c2VyIGhhcwotICAgICAqIGFjdGl2YXRlZCBsaW5lLXVwIGFyZWEgb2Ygc2Nyb2xsYmFyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNDUk9MTF9MSU5FX1VQID0gNjAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNDUk9MTF9MSU5FX0RPV04gaW5kaWNhdGVzIGFuIGV2ZW50IHdoZW4gdGhlIHVzZXIgaGFzCi0gICAgICogYWN0aXZhdGVkIGxpbmUtZG93biBhcmVhIG9mIHNjcm9sbGJhci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTQ1JPTExfTElORV9ET1dOID0gNjAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNDUk9MTF9QQUdFX1VQIGluZGljYXRlcyBhbiBldmVudCB3aGVuIHRoZSB1c2VyIGhhcwotICAgICAqIGFjdGl2YXRlZCBwYWdlIHVwIGFyZWEgb2Ygc2Nyb2xsYmFyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNDUk9MTF9QQUdFX1VQID0gNjAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNDUk9MTF9QQUdFX0RPV04gaW5kaWNhdGVzIGFuIGV2ZW50IHdoZW4gdGhlIHVzZXIgaGFzCi0gICAgICogYWN0aXZhdGVkIHBhZ2UgZG93biBhcmVhIG9mIHNjcm9sbGJhci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTQ1JPTExfUEFHRV9ET1dOID0gNjA0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNDUk9MTF9BQlNPTFVURSBpbmRpY2F0ZXMgYW4gZXZlbnQgd2hlbiB0aGUgdXNlciBoYXMgbW92ZWQKLSAgICAgKiB0aGUgYnViYmxlIGluIGEgc2Nyb2xsIGJhci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTQ1JPTExfQUJTT0xVVEUgPSA2MDU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU0NST0xMX0JFR0lOIGluZGljYXRlcyBhIHNjcm9sbCBiZWdpbiBldmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTQ1JPTExfQkVHSU4gPSA2MDY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU0NST0xMX0VORCBpbmRpY2F0ZXMgYSBzY3JvbGwgZW5kIGV2ZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNDUk9MTF9FTkQgPSA2MDc7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTElTVF9TRUxFQ1QgaW5kaWNhdGVzIHRoYXQgYW4gaXRlbSBpbiBhIGxpc3QgaGFzIGJlZW4KLSAgICAgKiBzZWxlY3RlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBMSVNUX1NFTEVDVCA9IDcwMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBMSVNUX0RFU0VMRUNUIGluZGljYXRlcyB0aGF0IGFuIGl0ZW0gaW4gYSBsaXN0IGhhcyBiZWVuCi0gICAgICogdW5zZWxlY3RlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBMSVNUX0RFU0VMRUNUID0gNzAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEFDVElPTl9FVkVOVCBpbmRpY2F0ZXMgdGhhdCB0aGUgdXNlciB3YW50cyBzb21lIGFjdGlvbiB0bwotICAgICAqIG9jY3VyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEFDVElPTl9FVkVOVCA9IDEwMDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTE9BRF9GSUxFIGluZGljYXRlcyBhIGZpbGUgbG9hZGluZyBldmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBMT0FEX0ZJTEUgPSAxMDAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNBVkVfRklMRSBpbmRpY2F0ZXMgYSBmaWxlIHNhdmluZyBldmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTQVZFX0ZJTEUgPSAxMDAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEdPVF9GT0NVUyBpbmRpY2F0ZXMgdGhhdCBhIGNvbXBvbmVudCBnb3QgdGhlIGZvY3VzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEdPVF9GT0NVUyA9IDEwMDQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTE9TVF9GT0NVUyBpbmRpY2F0ZXMgdGhhdCB0aGUgY29tcG9uZW50IGxvc3QgdGhlIGZvY3VzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IExPU1RfRk9DVVMgPSAxMDA1OwotCi0gICAgLyoqCi0gICAgICogVGhlIHRhcmdldCBpcyB0aGUgY29tcG9uZW50IHdpdGggd2hpY2ggdGhlIGV2ZW50IGlzIGFzc29jaWF0ZWQuCi0gICAgICovCi0gICAgcHVibGljIE9iamVjdCB0YXJnZXQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgd2hlbiBpcyB0aW1lc3RhbXAgd2hlbiBldmVudCBoYXMgb2NjdXJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgbG9uZyB3aGVuOwotCi0gICAgLyoqCi0gICAgICogVGhlIGlkIGluZGljYXRlcyB0aGUgdHlwZSBvZiB0aGUgZXZlbnQuCi0gICAgICovCi0gICAgcHVibGljIGludCBpZDsKLQotICAgIC8qKgotICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgZXZlbnQuCi0gICAgICovCi0gICAgcHVibGljIGludCB4OwotCi0gICAgLyoqCi0gICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiBldmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUga2V5IGNvZGUgb2Yga2V5IGV2ZW50LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQga2V5OwotCi0gICAgLyoqCi0gICAgICogVGhlIHN0YXRlIG9mIHRoZSBtb2RpZmllciBrZXlzIChnaXZlbiBieSBhIGJpdG1hc2spLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgbW9kaWZpZXJzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNsaWNrIGNvdW50IGluZGljYXRlcyB0aGUgbnVtYmVyIG9mIGNvbnNlY3V0aXZlIGNsaWNrcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGNsaWNrQ291bnQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYXJndW1lbnQgb2YgdGhlIGV2ZW50LgotICAgICAqLwotICAgIHB1YmxpYyBPYmplY3QgYXJnOwotCi0gICAgLyoqCi0gICAgICogVGhlIG5leHQgZXZlbnQuCi0gICAgICovCi0gICAgcHVibGljIEV2ZW50IGV2dDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBldmVudCB3aXRoIHRoZSBzcGVjaWZpZWQgdGFyZ2V0IGNvbXBvbmVudCwgZXZlbnQgdHlwZSwKLSAgICAgKiBhbmQgYXJndW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRhcmdldAotICAgICAqICAgICAgICAgICAgdGhlIHRhcmdldCBjb21wb25lbnQuCi0gICAgICogQHBhcmFtIGlkCi0gICAgICogICAgICAgICAgICB0aGUgZXZlbnQgdHlwZS4KLSAgICAgKiBAcGFyYW0gYXJnCi0gICAgICogICAgICAgICAgICB0aGUgYXJndW1lbnQuCi0gICAgICovCi0gICAgcHVibGljIEV2ZW50KE9iamVjdCB0YXJnZXQsIGludCBpZCwgT2JqZWN0IGFyZykgewotICAgICAgICB0aGlzKHRhcmdldCwgMGwsIGlkLCAwLCAwLCAwLCAwLCBhcmcpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBldmVudCB3aXRoIHRoZSBzcGVjaWZpZWQgdGFyZ2V0IGNvbXBvbmVudCwgdGltZSBzdGFtcCwKLSAgICAgKiBldmVudCB0eXBlLCB4IGFuZCB5IGNvb3JkaW5hdGVzLCBrZXlib2FyZCBrZXksIHN0YXRlIG9mIHRoZSBtb2RpZmllcgotICAgICAqIGtleXMsIGFuZCBhbiBhcmd1bWVudCBzZXQgdG8gbnVsbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdGFyZ2V0Ci0gICAgICogICAgICAgICAgICB0aGUgdGFyZ2V0IGNvbXBvbmVudC4KLSAgICAgKiBAcGFyYW0gd2hlbgotICAgICAqICAgICAgICAgICAgdGhlIHRpbWUgc3RhbXAuCi0gICAgICogQHBhcmFtIGlkCi0gICAgICogICAgICAgICAgICB0aGUgZXZlbnQgdHlwZS4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0ga2V5Ci0gICAgICogICAgICAgICAgICB0aGUga2V5LgotICAgICAqIEBwYXJhbSBtb2RpZmllcnMKLSAgICAgKiAgICAgICAgICAgIHRoZSBtb2RpZmllciBrZXlzIHN0YXRlLgotICAgICAqLwotICAgIHB1YmxpYyBFdmVudChPYmplY3QgdGFyZ2V0LCBsb25nIHdoZW4sIGludCBpZCwgaW50IHgsIGludCB5LCBpbnQga2V5LCBpbnQgbW9kaWZpZXJzKSB7Ci0gICAgICAgIHRoaXModGFyZ2V0LCB3aGVuLCBpZCwgeCwgeSwga2V5LCBtb2RpZmllcnMsIG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBldmVudCB3aXRoIHRoZSBzcGVjaWZpZWQgdGFyZ2V0IGNvbXBvbmVudCwgdGltZSBzdGFtcCwKLSAgICAgKiBldmVudCB0eXBlLCB4IGFuZCB5IGNvb3JkaW5hdGVzLCBrZXlib2FyZCBrZXksIHN0YXRlIG9mIHRoZSBtb2RpZmllcgotICAgICAqIGtleXMsIGFuZCBhbiBhcmd1bWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdGFyZ2V0Ci0gICAgICogICAgICAgICAgICB0aGUgdGFyZ2V0IGNvbXBvbmVudC4KLSAgICAgKiBAcGFyYW0gd2hlbgotICAgICAqICAgICAgICAgICAgdGhlIHRpbWUgc3RhbXAuCi0gICAgICogQHBhcmFtIGlkCi0gICAgICogICAgICAgICAgICB0aGUgZXZlbnQgdHlwZS4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0ga2V5Ci0gICAgICogICAgICAgICAgICB0aGUga2V5LgotICAgICAqIEBwYXJhbSBtb2RpZmllcnMKLSAgICAgKiAgICAgICAgICAgIHRoZSBtb2RpZmllciBrZXlzIHN0YXRlLgotICAgICAqIEBwYXJhbSBhcmcKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYXJndW1lbnQuCi0gICAgICovCi0gICAgcHVibGljIEV2ZW50KE9iamVjdCB0YXJnZXQsIGxvbmcgd2hlbiwgaW50IGlkLCBpbnQgeCwgaW50IHksIGludCBrZXksIGludCBtb2RpZmllcnMsIE9iamVjdCBhcmcpIHsKLSAgICAgICAgdGhpcy50YXJnZXQgPSB0YXJnZXQ7Ci0gICAgICAgIHRoaXMud2hlbiA9IHdoZW47Ci0gICAgICAgIHRoaXMuaWQgPSBpZDsKLSAgICAgICAgdGhpcy54ID0geDsKLSAgICAgICAgdGhpcy55ID0geTsKLSAgICAgICAgdGhpcy5rZXkgPSBrZXk7Ci0gICAgICAgIHRoaXMubW9kaWZpZXJzID0gbW9kaWZpZXJzOwotICAgICAgICB0aGlzLmFyZyA9IGFyZzsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoaXMgRXZlbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiBhIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGlzIEV2ZW50LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgIC8qCi0gICAgICAgICAqIFRoZSBmb3JtYXQgaXMgYmFzZWQgb24gMS41IHJlbGVhc2UgYmVoYXZpb3Igd2hpY2ggY2FuIGJlIHJldmVhbGVkIGJ5Ci0gICAgICAgICAqIHRoZSBmb2xsb3dpbmcgY29kZTogRXZlbnQgZSA9IG5ldyBFdmVudChuZXcgQnV0dG9uKCksIDBsLAotICAgICAgICAgKiBFdmVudC5LRVlfUFJFU1MsIDAsIDAsIEV2ZW50LlRBQiwgRXZlbnQuU0hJRlRfTUFTSywgImFyZyIpOwotICAgICAgICAgKiBTeXN0ZW0ub3V0LnByaW50bG4oZSk7Ci0gICAgICAgICAqLwotCi0gICAgICAgIHJldHVybiBnZXRDbGFzcygpLmdldE5hbWUoKSArICJbIiArIHBhcmFtU3RyaW5nKCkgKyAiXSI7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBzdHJpbmcgcmVwcmVzZW50aW5nIHRoZSBzdGF0ZSBvZiB0aGlzIEV2ZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBzdHJpbmcgcmVwcmVzZW50aW5nIHRoZSBzdGF0ZSBvZiB0aGlzIEV2ZW50LgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmcgcGFyYW1TdHJpbmcoKSB7Ci0gICAgICAgIHJldHVybiAiaWQ9IiArIGlkICsgIix4PSIgKyB4ICsgIix5PSIgKyB5ICsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQKLSAgICAgICAgICAgICAgICAoa2V5ICE9IDAgPyAiLGtleT0iICsga2V5ICsgZ2V0TW9kaWZpZXJzU3RyaW5nKCkgOiAiIikgKyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgICAgICAgICAiLHRhcmdldD0iICsgdGFyZ2V0ICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgIChhcmcgIT0gbnVsbCA/ICIsYXJnPSIgKyBhcmcgOiAiIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIG1vZGlmaWVycy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBtb2RpZmllcnMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBTdHJpbmcgZ2V0TW9kaWZpZXJzU3RyaW5nKCkgewotICAgICAgICBTdHJpbmcgc3RyTW9kID0gIiI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgaWYgKHNoaWZ0RG93bigpKSB7Ci0gICAgICAgICAgICBzdHJNb2QgKz0gIixzaGlmdCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZiAoY29udHJvbERvd24oKSkgewotICAgICAgICAgICAgc3RyTW9kICs9ICIsY29udHJvbCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZiAobWV0YURvd24oKSkgewotICAgICAgICAgICAgc3RyTW9kICs9ICIsbWV0YSI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gc3RyTW9kOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRyYW5zbGF0ZXMgeCBhbmQgeSBjb29yZGluYXRlcyBvZiBoaXMgZXZlbnQgdG8gdGhlIHgrZHggYW5kIHgrZHkKLSAgICAgKiBjb29yZGluYXRlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSBieSB3aGljaCB0aGUgZXZlbnQncyB4IGNvb3JkaW5hdGUgaXMgaW5jcmVhc2VkLgotICAgICAqIEBwYXJhbSBkeQotICAgICAqICAgICAgICAgICAgdGhlIGRpc3RhbmNlIGJ5IHdoaWNoIHRoZSBldmVudCdzIHkgY29vcmRpbmF0ZSBpcyBpbmNyZWFzZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgdHJhbnNsYXRlKGludCBkeCwgaW50IGR5KSB7Ci0gICAgICAgIHggKz0gZHg7Ci0gICAgICAgIHkgKz0gZHk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIENvbnRyb2wga2V5IGlzIGRvd24gb3Igbm90LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgQ29udHJvbCBrZXkgaXMgZG93bjsgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRyb2xEb3duKCkgewotICAgICAgICByZXR1cm4gKG1vZGlmaWVycyAmIENUUkxfTUFTSykgIT0gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgTWV0YSBrZXkgaXMgZG93biBvciBub3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBNZXRhIGtleSBpcyBkb3duOyBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gbWV0YURvd24oKSB7Ci0gICAgICAgIHJldHVybiAobW9kaWZpZXJzICYgTUVUQV9NQVNLKSAhPSAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiBTaGlmdCBrZXkgaXMgZG93biBvciBub3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBTaGlmdCBrZXkgaXMgZG93bjsgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIHNoaWZ0RG93bigpIHsKLSAgICAgICAgcmV0dXJuIChtb2RpZmllcnMgJiBTSElGVF9NQVNLKSAhPSAwOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0V2ZW50RGlzcGF0Y2hUaHJlYWQuamF2YSBiL2F3dC9qYXZhL2F3dC9FdmVudERpc3BhdGNoVGhyZWFkLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQ0MmM4YTIuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0V2ZW50RGlzcGF0Y2hUaHJlYWQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDExOCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdiwgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5OYXRpdmVFdmVudDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5OYXRpdmVFdmVudFF1ZXVlOwotCi1jbGFzcyBFdmVudERpc3BhdGNoVGhyZWFkIGV4dGVuZHMgVGhyZWFkICB7Ci0gICAgCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgY2xhc3MgTWFya2VyRXZlbnQgZXh0ZW5kcyBBV1RFdmVudCB7Ci0gICAgICAgIE1hcmtlckV2ZW50KE9iamVjdCBzb3VyY2UsIGludCBpZCkgewotICAgICAgICAgICAgc3VwZXIoc291cmNlLCBpZCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBmaW5hbCBEaXNwYXRjaGVyIGRpc3BhdGNoZXI7Ci0gICAgZmluYWwgVG9vbGtpdCB0b29sa2l0OwotICAgIHByaXZhdGUgTmF0aXZlRXZlbnRRdWV1ZSBuYXRpdmVRdWV1ZTsKLQotICAgIHByb3RlY3RlZCB2b2xhdGlsZSBib29sZWFuIHNodXRkb3duUGVuZGluZyA9IGZhbHNlOwotCi0gICAgLyoqCi0gICAgICogSW5pdGlhbGlzZSBhbmQgcnVuIHRoZSBtYWluIGV2ZW50IGxvb3AKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBydW4oKSB7Ci0gICAgICAgIG5hdGl2ZVF1ZXVlID0gdG9vbGtpdC5nZXROYXRpdmVFdmVudFF1ZXVlKCk7Ci0KLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJ1bk1vZGFsTG9vcChudWxsKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQuc2h1dGRvd25XYXRjaGRvZy5mb3JjZVNodXRkb3duKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICB2b2lkIHJ1bk1vZGFsTG9vcChNb2RhbENvbnRleHQgY29udGV4dCkgewotICAgICAgICBsb25nIGxhc3RQYWludFRpbWUgPSBTeXN0ZW0uY3VycmVudFRpbWVNaWxsaXMoKTsKLSAgICAgICAgd2hpbGUgKCFzaHV0ZG93blBlbmRpbmcgJiYgKGNvbnRleHQgPT0gbnVsbCB8fCBjb250ZXh0LmlzTW9kYWxMb29wUnVubmluZygpKSkgewotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIEV2ZW50UXVldWUgZXZlbnRRdWV1ZSA9IHRvb2xraXQuZ2V0U3lzdGVtRXZlbnRRdWV1ZUltcGwoKTsKLQotICAgICAgICAgICAgTmF0aXZlRXZlbnQgbmUgPSBuYXRpdmVRdWV1ZS5nZXROZXh0RXZlbnQoKTsKLSAgICAgICAgICAgIGlmIChuZSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgZGlzcGF0Y2hlci5vbkV2ZW50KG5lKTsKLSAgICAgICAgICAgICAgICBNYXJrZXJFdmVudCBtYXJrZXIgPSBuZXcgTWFya2VyRXZlbnQodGhpcywgMCk7Ci0gICAgICAgICAgICAgICAgZXZlbnRRdWV1ZS5wb3N0RXZlbnQobWFya2VyKTsKLSAgICAgICAgICAgICAgICBmb3IgKEFXVEV2ZW50IGFlID0gZXZlbnRRdWV1ZS5nZXROZXh0RXZlbnROb1dhaXQoKTsgCi0gICAgICAgICAgICAgICAgICAgICAgICAoYWUgIT0gbnVsbCkgJiYgKGFlICE9IG1hcmtlcik7IAotICAgICAgICAgICAgICAgICAgICAgICAgYWUgPSBldmVudFF1ZXVlLmdldE5leHRFdmVudE5vV2FpdCgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGV2ZW50UXVldWUuZGlzcGF0Y2hFdmVudChhZSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICB0b29sa2l0LnNodXRkb3duV2F0Y2hkb2cuc2V0TmF0aXZlUXVldWVFbXB0eSh0cnVlKTsKLSAgICAgICAgICAgICAgICBBV1RFdmVudCBhZSA9IGV2ZW50UXVldWUuZ2V0TmV4dEV2ZW50Tm9XYWl0KCk7Ci0gICAgICAgICAgICAgICAgaWYgKGFlICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgZXZlbnRRdWV1ZS5kaXNwYXRjaEV2ZW50KGFlKTsKLSAgICAgICAgICAgICAgICAgICAgbG9uZyBjdXJUaW1lID0gU3lzdGVtLmN1cnJlbnRUaW1lTWlsbGlzKCk7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChjdXJUaW1lIC0gbGFzdFBhaW50VGltZSA+IDEwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB0b29sa2l0Lm9uUXVldWVFbXB0eSgpOwotICAgICAgICAgICAgICAgICAgICAgICAgbGFzdFBhaW50VGltZSA9IFN5c3RlbS5jdXJyZW50VGltZU1pbGxpcygpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgdG9vbGtpdC5zaHV0ZG93bldhdGNoZG9nLnNldEF3dFF1ZXVlRW1wdHkodHJ1ZSk7Ci0gICAgICAgICAgICAgICAgICAgIHRvb2xraXQub25RdWV1ZUVtcHR5KCk7Ci0gICAgICAgICAgICAgICAgICAgIGxhc3RQYWludFRpbWUgPSBTeXN0ZW0uY3VycmVudFRpbWVNaWxsaXMoKTsKLSAgICAgICAgICAgICAgICAgICAgd2FpdEZvckFueUV2ZW50KCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBjYXRjaCAoVGhyb3dhYmxlIHQpIHsKLSAgICAgICAgICAgICAgICAvLyBUT0RPOiBFeGNlcHRpb24gaGFuZGxlciBzaG91bGQgYmUgaW1wbGVtZW50ZWQKLSAgICAgICAgICAgICAgICAvLyB0LnByaW50U3RhY2tUcmFjZSgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIHByaXZhdGUgdm9pZCB3YWl0Rm9yQW55RXZlbnQoKSB7Ci0gICAgICAgIEV2ZW50UXVldWUgZXZlbnRRdWV1ZSA9IHRvb2xraXQuZ2V0U3lzdGVtRXZlbnRRdWV1ZUltcGwoKTsKLSAgICAgICAgaWYgKCFldmVudFF1ZXVlLmlzRW1wdHkoKSB8fCAhbmF0aXZlUXVldWUuaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgT2JqZWN0IGV2ZW50TW9uaXRvciA9IG5hdGl2ZVF1ZXVlLmdldEV2ZW50TW9uaXRvcigpOwotICAgICAgICBzeW5jaHJvbml6ZWQoZXZlbnRNb25pdG9yKSB7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIGV2ZW50TW9uaXRvci53YWl0KCk7Ci0gICAgICAgICAgICB9IGNhdGNoIChJbnRlcnJ1cHRlZEV4Y2VwdGlvbiBlKSB7fQotICAgICAgICB9Ci0gICAgfQotCi0gICAgdm9pZCBzaHV0ZG93bigpIHsKLSAgICAgICAgc2h1dGRvd25QZW5kaW5nID0gdHJ1ZTsKLSAgICB9Ci0KLSAgICBFdmVudERpc3BhdGNoVGhyZWFkKFRvb2xraXQgdG9vbGtpdCwgRGlzcGF0Y2hlciBkaXNwYXRjaGVyICkgewotICAgICAgICB0aGlzLnRvb2xraXQgPSB0b29sa2l0OwotICAgICAgICB0aGlzLmRpc3BhdGNoZXIgPSBkaXNwYXRjaGVyOwotICAgICAgICBzZXROYW1lKCJBV1QtRXZlbnREaXNwYXRjaFRocmVhZCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIHNldERhZW1vbih0cnVlKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9FdmVudFF1ZXVlLmphdmEgYi9hd3QvamF2YS9hd3QvRXZlbnRRdWV1ZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxMjZhNTkzLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9FdmVudFF1ZXVlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzMjAgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YsIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmF3dC5ldmVudC5JbnZvY2F0aW9uRXZlbnQ7Ci1pbXBvcnQgamF2YS5sYW5nLnJlZmxlY3QuSW52b2NhdGlvblRhcmdldEV4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuRW1wdHlTdGFja0V4Y2VwdGlvbjsKLQotLyoqCi0gKiBUaGUgRXZlbnRRdWV1ZSBjbGFzcyBtYW5hZ2VzIGV2ZW50cy4gSXQgaXMgYSBwbGF0Zm9ybS1pbmRlcGVuZGVudCBjbGFzcyB0aGF0Ci0gKiBxdWV1ZXMgZXZlbnRzIGJvdGggZnJvbSB0aGUgdW5kZXJseWluZyBwZWVyIGNsYXNzZXMgYW5kIGZyb20gdHJ1c3RlZAotICogYXBwbGljYXRpb24gY2xhc3Nlcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBFdmVudFF1ZXVlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb3JlIHJlZi4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIEV2ZW50UXVldWVDb3JlQXRvbWljUmVmZXJlbmNlIGNvcmVSZWYgPSBuZXcgRXZlbnRRdWV1ZUNvcmVBdG9taWNSZWZlcmVuY2UoKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBFdmVudFF1ZXVlQ29yZUF0b21pY1JlZmVyZW5jZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBjbGFzcyBFdmVudFF1ZXVlQ29yZUF0b21pY1JlZmVyZW5jZSB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjb3JlLgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBFdmVudFF1ZXVlQ29yZSBjb3JlOwotCi0gICAgICAgIC8qIHN5bmNocm9uaXplZCAqLwotICAgICAgICAvKioKLSAgICAgICAgICogR2V0cyB0aGUuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBldmVudCBxdWV1ZSBjb3JlLgotICAgICAgICAgKi8KLSAgICAgICAgRXZlbnRRdWV1ZUNvcmUgZ2V0KCkgewotICAgICAgICAgICAgcmV0dXJuIGNvcmU7Ci0gICAgICAgIH0KLQotICAgICAgICAvKiBzeW5jaHJvbml6ZWQgKi8KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFNldHMgdGhlLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIG5ld0NvcmUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgbmV3IGNvcmUuCi0gICAgICAgICAqLwotICAgICAgICB2b2lkIHNldChFdmVudFF1ZXVlQ29yZSBuZXdDb3JlKSB7Ci0gICAgICAgICAgICBjb3JlID0gbmV3Q29yZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgY2FsbGluZyB0aHJlYWQgaXMgdGhlIGN1cnJlbnQgQVdUIEV2ZW50UXVldWUncwotICAgICAqIGRpc3BhdGNoIHRocmVhZC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBjYWxsaW5nIHRocmVhZCBpcyB0aGUgY3VycmVudCBBV1QgRXZlbnRRdWV1ZSdzCi0gICAgICogICAgICAgICBkaXNwYXRjaCB0aHJlYWQ7IGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGJvb2xlYW4gaXNEaXNwYXRjaFRocmVhZCgpIHsKLSAgICAgICAgcmV0dXJuIFRocmVhZC5jdXJyZW50VGhyZWFkKCkgaW5zdGFuY2VvZiBFdmVudERpc3BhdGNoVGhyZWFkOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBvc3RzIGFuIEludm9jYXRpb25FdmVudCB3aGljaCBleGVjdXRlcyB0aGUgcnVuKCkgbWV0aG9kIG9uIGEgUnVubmFibGUKLSAgICAgKiB3aGVuIGRpc3BhdGNoZWQgYnkgdGhlIEFXVCBldmVudCBkaXNwYXRjaGVyIHRocmVhZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcnVubmFibGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBSdW5uYWJsZSB3aG9zZSBydW4gbWV0aG9kIHNob3VsZCBiZSBleGVjdXRlZCBzeW5jaHJvbm91c2x5Ci0gICAgICogICAgICAgICAgICBvbiB0aGUgRXZlbnRRdWV1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgaW52b2tlTGF0ZXIoUnVubmFibGUgcnVubmFibGUpIHsKLSAgICAgICAgVG9vbGtpdCB0b29sa2l0ID0gVG9vbGtpdC5nZXREZWZhdWx0VG9vbGtpdCgpOwotICAgICAgICBJbnZvY2F0aW9uRXZlbnQgZXZlbnQgPSBuZXcgSW52b2NhdGlvbkV2ZW50KHRvb2xraXQsIHJ1bm5hYmxlKTsKLSAgICAgICAgdG9vbGtpdC5nZXRTeXN0ZW1FdmVudFF1ZXVlSW1wbCgpLnBvc3RFdmVudChldmVudCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUG9zdHMgYW4gSW52b2NhdGlvbkV2ZW50IHdoaWNoIGV4ZWN1dGVzIHRoZSBydW4oKSBtZXRob2Qgb24gYSBSdW5uYWJsZQotICAgICAqIHdoZW4gZGlzcGF0Y2hlZCBieSB0aGUgQVdUIGV2ZW50IGRpc3BhdGNoZXIgdGhyZWFkIGFuZCB0aGUgbm90aWZ5QWxsCi0gICAgICogbWV0aG9kIGlzIGNhbGxlZCBvbiBpdCBpbW1lZGlhdGVseSBhZnRlciBydW4gcmV0dXJucy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcnVubmFibGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBSdW5uYWJsZSB3aG9zZSBydW4gbWV0aG9kIHNob3VsZCBiZSBleGVjdXRlZCBzeW5jaHJvbm91c2x5Ci0gICAgICogICAgICAgICAgICBvbiB0aGUgRXZlbnRRdWV1ZS4KLSAgICAgKiBAdGhyb3dzIEludGVycnVwdGVkRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW5vdGhlciB0aHJlYWQgaGFzIGludGVycnVwdGVkIHRoaXMgdGhyZWFkLgotICAgICAqIEB0aHJvd3MgSW52b2NhdGlvblRhcmdldEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIGVycm9yIG9jY3VycmVkIHdoaWxlIHJ1bm5pbmcgdGhlIHJ1bm5hYmxlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgdm9pZCBpbnZva2VBbmRXYWl0KFJ1bm5hYmxlIHJ1bm5hYmxlKSB0aHJvd3MgSW50ZXJydXB0ZWRFeGNlcHRpb24sCi0gICAgICAgICAgICBJbnZvY2F0aW9uVGFyZ2V0RXhjZXB0aW9uIHsKLQotICAgICAgICBpZiAoaXNEaXNwYXRjaFRocmVhZCgpKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGZpbmFsIFRvb2xraXQgdG9vbGtpdCA9IFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKTsKLSAgICAgICAgZmluYWwgT2JqZWN0IG5vdGlmaWVyID0gbmV3IE9iamVjdCgpOyAvLyAkTk9OLUxPQ0stMSQKLSAgICAgICAgSW52b2NhdGlvbkV2ZW50IGV2ZW50ID0gbmV3IEludm9jYXRpb25FdmVudCh0b29sa2l0LCBydW5uYWJsZSwgbm90aWZpZXIsIHRydWUpOwotCi0gICAgICAgIHN5bmNocm9uaXplZCAobm90aWZpZXIpIHsKLSAgICAgICAgICAgIHRvb2xraXQuZ2V0U3lzdGVtRXZlbnRRdWV1ZUltcGwoKS5wb3N0RXZlbnQoZXZlbnQpOwotICAgICAgICAgICAgbm90aWZpZXIud2FpdCgpOwotICAgICAgICB9Ci0KLSAgICAgICAgRXhjZXB0aW9uIGV4Y2VwdGlvbiA9IGV2ZW50LmdldEV4Y2VwdGlvbigpOwotCi0gICAgICAgIGlmIChleGNlcHRpb24gIT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IEludm9jYXRpb25UYXJnZXRFeGNlcHRpb24oZXhjZXB0aW9uKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN5c3RlbSBldmVudCBxdWV1ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzeXN0ZW0gZXZlbnQgcXVldWUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgRXZlbnRRdWV1ZSBnZXRTeXN0ZW1FdmVudFF1ZXVlKCkgewotICAgICAgICBUaHJlYWQgdGggPSBUaHJlYWQuY3VycmVudFRocmVhZCgpOwotICAgICAgICBpZiAodGggaW5zdGFuY2VvZiBFdmVudERpc3BhdGNoVGhyZWFkKSB7Ci0gICAgICAgICAgICByZXR1cm4gKChFdmVudERpc3BhdGNoVGhyZWFkKXRoKS50b29sa2l0LmdldFN5c3RlbUV2ZW50UXVldWVJbXBsKCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbW9zdCByZWNlbnQgZXZlbnQncyB0aW1lc3RhbXAuIFRoaXMgZXZlbnQgd2FzIGRpc3BhdGNoZWQgZnJvbQotICAgICAqIHRoZSBFdmVudFF1ZXVlIGFzc29jaWF0ZWQgd2l0aCB0aGUgY2FsbGluZyB0aHJlYWQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdGltZXN0YW1wIG9mIHRoZSBsYXN0IEV2ZW50IHRvIGJlIGRpc3BhdGNoZWQsIG9yCi0gICAgICogICAgICAgICBTeXN0ZW0uY3VycmVudFRpbWVNaWxsaXMoKSBpZiB0aGlzIG1ldGhvZCBpcyBpbnZva2VkIGZyb20gYQotICAgICAqICAgICAgICAgdGhyZWFkIG90aGVyIHRoYW4gYW4gZXZlbnQtZGlzcGF0Y2hpbmcgdGhyZWFkLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgbG9uZyBnZXRNb3N0UmVjZW50RXZlbnRUaW1lKCkgewotICAgICAgICBFdmVudFF1ZXVlIGVxID0gZ2V0U3lzdGVtRXZlbnRRdWV1ZSgpOwotICAgICAgICByZXR1cm4gKGVxICE9IG51bGwpID8gZXEuZ2V0TW9zdFJlY2VudEV2ZW50VGltZUltcGwoKSA6IFN5c3RlbS5jdXJyZW50VGltZU1pbGxpcygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1vc3QgcmVjZW50IGV2ZW50IHRpbWUgaW1wbC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtb3N0IHJlY2VudCBldmVudCB0aW1lIGltcGwuCi0gICAgICovCi0gICAgcHJpdmF0ZSBsb25nIGdldE1vc3RSZWNlbnRFdmVudFRpbWVJbXBsKCkgewotICAgICAgICByZXR1cm4gZ2V0Q29yZSgpLmdldE1vc3RSZWNlbnRFdmVudFRpbWUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSB0aGUgY3VycmVudGx5IGRpc3BhdGNoZWQgZXZlbnQgYnkgdGhlIEV2ZW50UXVldWUgYXNzb2NpYXRlZAotICAgICAqIHdpdGggdGhlIGNhbGxpbmcgdGhyZWFkLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGN1cnJlbnRseSBkaXNwYXRjaGVkIGV2ZW50IG9yIG51bGwgaWYgdGhpcyBtZXRob2QgaXMgaW52b2tlZAotICAgICAqICAgICAgICAgZnJvbSBhIHRocmVhZCBvdGhlciB0aGFuIGFuIGV2ZW50LWRpc3BhdGNoaW5nIHRocmVhZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEFXVEV2ZW50IGdldEN1cnJlbnRFdmVudCgpIHsKLSAgICAgICAgRXZlbnRRdWV1ZSBlcSA9IGdldFN5c3RlbUV2ZW50UXVldWUoKTsKLSAgICAgICAgcmV0dXJuIChlcSAhPSBudWxsKSA/IGVxLmdldEN1cnJlbnRFdmVudEltcGwoKSA6IG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY3VycmVudCBldmVudCBpbXBsLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGN1cnJlbnQgZXZlbnQgaW1wbC4KLSAgICAgKi8KLSAgICBwcml2YXRlIEFXVEV2ZW50IGdldEN1cnJlbnRFdmVudEltcGwoKSB7Ci0gICAgICAgIHJldHVybiBnZXRDb3JlKCkuZ2V0Q3VycmVudEV2ZW50KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGV2ZW50IHF1ZXVlLgotICAgICAqLwotICAgIHB1YmxpYyBFdmVudFF1ZXVlKCkgewotICAgICAgICBzZXRDb3JlKG5ldyBFdmVudFF1ZXVlQ29yZSh0aGlzKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGV2ZW50IHF1ZXVlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0Ci0gICAgICogICAgICAgICAgICB0aGUgdC4KLSAgICAgKi8KLSAgICBFdmVudFF1ZXVlKFRvb2xraXQgdCkgewotICAgICAgICBzZXRDb3JlKG5ldyBFdmVudFF1ZXVlQ29yZSh0aGlzLCB0KSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUG9zdHMgYSBldmVudCB0byB0aGUgRXZlbnRRdWV1ZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXZlbnQKLSAgICAgKiAgICAgICAgICAgIEFXVEV2ZW50LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHBvc3RFdmVudChBV1RFdmVudCBldmVudCkgewotICAgICAgICBldmVudC5pc1Bvc3RlZCA9IHRydWU7Ci0gICAgICAgIGdldENvcmUoKS5wb3N0RXZlbnQoZXZlbnQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gZXZlbnQgZnJvbSB0aGUgRXZlbnRRdWV1ZSBhbmQgcmVtb3ZlcyBpdCBmcm9tIHRoaXMgcXVldWUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbmV4dCBBV1RFdmVudC4KLSAgICAgKiBAdGhyb3dzIEludGVycnVwdGVkRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaXMgdGhyb3duIGlmIGFub3RoZXIgdGhyZWFkIGludGVycnVwdHMgdGhpcyB0aHJlYWQuCi0gICAgICovCi0gICAgcHVibGljIEFXVEV2ZW50IGdldE5leHRFdmVudCgpIHRocm93cyBJbnRlcnJ1cHRlZEV4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBnZXRDb3JlKCkuZ2V0TmV4dEV2ZW50KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbmV4dCBldmVudCBubyB3YWl0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG5leHQgZXZlbnQgbm8gd2FpdC4KLSAgICAgKi8KLSAgICBBV1RFdmVudCBnZXROZXh0RXZlbnROb1dhaXQoKSB7Ci0gICAgICAgIHJldHVybiBnZXRDb3JlKCkuZ2V0TmV4dEV2ZW50Tm9XYWl0KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgZmlyc3QgZXZlbnQgb2YgdGhlIEV2ZW50UXVldWUgKHdpdGhvdXQgcmVtb3ZpbmcgaXQgZnJvbSB0aGUKLSAgICAgKiBxdWV1ZSkuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdGhlIGZpcnN0IEFXVCBldmVudCBvZiB0aGUgRXZlbnRRdWV1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQVdURXZlbnQgcGVla0V2ZW50KCkgewotICAgICAgICByZXR1cm4gZ2V0Q29yZSgpLnBlZWtFdmVudCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGZpcnN0IGV2ZW50IG9mIHRoZSBFdmVudFF1ZXVlIHdpdGggdGhlIHNwZWNpZmllZCBJRCAod2l0aG91dAotICAgICAqIHJlbW92aW5nIGl0IGZyb20gdGhlIHF1ZXVlKS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWQKLSAgICAgKiAgICAgICAgICAgIHRoZSB0eXBlIElEIG9mIGV2ZW50LgotICAgICAqIEByZXR1cm4gdGhlIGZpcnN0IGV2ZW50IG9mIHRoZSBFdmVudFF1ZXVlIHdpdGggdGhlIHNwZWNpZmllZCBJRC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQVdURXZlbnQgcGVla0V2ZW50KGludCBpZCkgewotICAgICAgICByZXR1cm4gZ2V0Q29yZSgpLnBlZWtFdmVudChpZCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwbGFjZXMgdGhlIGV4aXN0aW5nIEV2ZW50UXVldWUgd2l0aCB0aGUgc3BlY2lmaWVkIEV2ZW50UXVldWUuIEFueQotICAgICAqIHBlbmRpbmcgZXZlbnRzIGFyZSB0cmFuc2ZlcnJlZCB0byB0aGUgbmV3IEV2ZW50UXVldWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIG5ld0V2ZW50UXVldWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgZXZlbnQgcXVldWUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcHVzaChFdmVudFF1ZXVlIG5ld0V2ZW50UXVldWUpIHsKLSAgICAgICAgZ2V0Q29yZSgpLnB1c2gobmV3RXZlbnRRdWV1ZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU3RvcHMgZGlzcGF0Y2hpbmcgZXZlbnRzIHVzaW5nIHRoaXMgRXZlbnRRdWV1ZS4gQW55IHBlbmRpbmcgZXZlbnRzIGFyZQotICAgICAqIHRyYW5zZmVycmVkIHRvIHRoZSBwcmV2aW91cyBFdmVudFF1ZXVlLgotICAgICAqIAotICAgICAqIEB0aHJvd3MgRW1wdHlTdGFja0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlzIHRocm93biBpZiBubyBwcmV2aW91cyBwdXNoIHdhcyBtYWRlIG9uIHRoaXMgRXZlbnRRdWV1ZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwb3AoKSB0aHJvd3MgRW1wdHlTdGFja0V4Y2VwdGlvbiB7Ci0gICAgICAgIGdldENvcmUoKS5wb3AoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEaXNwYXRjaGVzIHRoZSBzcGVjaWZpZWQgZXZlbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2ZW50Ci0gICAgICogICAgICAgICAgICB0aGUgQVdURXZlbnQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgZGlzcGF0Y2hFdmVudChBV1RFdmVudCBldmVudCkgewotICAgICAgICBnZXRDb3JlKCkuZGlzcGF0Y2hFdmVudEltcGwoZXZlbnQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGUgcXVldWUgaXMgZW1wdHkuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBpcyBlbXB0eS4KLSAgICAgKi8KLSAgICBib29sZWFuIGlzRW1wdHkoKSB7Ci0gICAgICAgIHJldHVybiBnZXRDb3JlKCkuaXNFbXB0eSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvcmUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY29yZS4KLSAgICAgKi8KLSAgICBFdmVudFF1ZXVlQ29yZSBnZXRDb3JlKCkgewotICAgICAgICByZXR1cm4gY29yZVJlZi5nZXQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBjb3JlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuZXdDb3JlCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IGNvcmUuCi0gICAgICovCi0gICAgdm9pZCBzZXRDb3JlKEV2ZW50UXVldWVDb3JlIG5ld0NvcmUpIHsKLSAgICAgICAgY29yZVJlZi5zZXQoKG5ld0NvcmUgIT0gbnVsbCkgPyBuZXdDb3JlIDogbmV3IEV2ZW50UXVldWVDb3JlKHRoaXMpKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvRXZlbnRRdWV1ZUNvcmUuamF2YSBiL2F3dC9qYXZhL2F3dC9FdmVudFF1ZXVlQ29yZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmZmM3YzQ2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9FdmVudFF1ZXVlQ29yZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjUzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKiogCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuZXZlbnQuQWN0aW9uRXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuSW5wdXRFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5JbnB1dE1ldGhvZEV2ZW50OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50Lkludm9jYXRpb25FdmVudDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5Nb3VzZUV2ZW50OwotaW1wb3J0IGphdmEudXRpbC5MaW5rZWRMaXN0OwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIGV2ZW50cyBzdG9yYWdlIGZvciBFdmVudFF1ZXVlCi0gKi8KLWZpbmFsIGNsYXNzIEV2ZW50UXVldWVDb3JlIHsKLSAgICAKLSAgICBwcml2YXRlIGZpbmFsIExpbmtlZExpc3Q8RXZlbnRRdWV1ZT4gcXVldWVTdGFjayA9IG5ldyBMaW5rZWRMaXN0PEV2ZW50UXVldWU+KCk7Ci0gICAgcHJpdmF0ZSBmaW5hbCBMaW5rZWRMaXN0PEFXVEV2ZW50PiBldmVudHMgPSBuZXcgTGlua2VkTGlzdDxBV1RFdmVudD4oKTsKLSAgICAKLSAgICBwcml2YXRlIFRvb2xraXQgdG9vbGtpdDsKLSAgICBwcml2YXRlIEV2ZW50UXVldWUgYWN0aXZlUXVldWU7Ci0gICAgcHJpdmF0ZSBUaHJlYWQgZGlzcGF0Y2hUaHJlYWQ7Ci0gICAgCi0gICAgQVdURXZlbnQgY3VycmVudEV2ZW50OwotICAgIGxvbmcgbW9zdFJlY2VudEV2ZW50VGltZSA9IFN5c3RlbS5jdXJyZW50VGltZU1pbGxpcygpOwotICAgIAotICAgIEV2ZW50UXVldWVDb3JlKEV2ZW50UXVldWUgZXEpIHsKLSAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7Ci0gICAgICAgICAgICBxdWV1ZVN0YWNrLmFkZExhc3QoZXEpOwotICAgICAgICAgICAgYWN0aXZlUXVldWUgPSBlcTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEV2ZW50UXVldWVDb3JlKEV2ZW50UXVldWUgZXEsIFRvb2xraXQgdCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIHF1ZXVlU3RhY2suYWRkTGFzdChlcSk7Ci0gICAgICAgICAgICBhY3RpdmVRdWV1ZSA9IGVxOwotICAgICAgICAgICAgc2V0VG9vbGtpdCh0KTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHN5bmNocm9uaXplZCBsb25nIGdldE1vc3RSZWNlbnRFdmVudFRpbWUoKSB7Ci0gICAgICAgIHJldHVybiBtb3N0UmVjZW50RXZlbnRUaW1lOwotICAgIH0KLSAgICAKLSAgICBzeW5jaHJvbml6ZWQgQVdURXZlbnQgZ2V0Q3VycmVudEV2ZW50KCkgewotICAgICAgICByZXR1cm4gY3VycmVudEV2ZW50OwotICAgIH0KLSAgICAKLSAgICBzeW5jaHJvbml6ZWQgYm9vbGVhbiBpc1N5c3RlbUV2ZW50UXVldWUoKSB7Ci0gICAgICAgIHJldHVybiB0b29sa2l0ICE9IG51bGw7Ci0gICAgfQotICAgIAotICAgIHByaXZhdGUgdm9pZCBzZXRUb29sa2l0KFRvb2xraXQgdCkgewotICAgICAgICB0b29sa2l0ID0gdDsKLSAgICAgICAgaWYgKHRvb2xraXQgIT0gbnVsbCkgewotICAgICAgICAgICAgdG9vbGtpdC5zZXRTeXN0ZW1FdmVudFF1ZXVlQ29yZSh0aGlzKTsKLSAgICAgICAgICAgIGRpc3BhdGNoVGhyZWFkID0gdG9vbGtpdC5kaXNwYXRjaFRocmVhZDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHN5bmNocm9uaXplZCB2b2lkIHBvc3RFdmVudChBV1RFdmVudCBldmVudCkgewotICAgICAgICAvLz8/P0FXVAotICAgICAgICAvKgotICAgICAgICBldmVudHMuYWRkTGFzdChldmVudCk7Ci0gICAgICAgIGlmICgodG9vbGtpdCA9PSBudWxsKSAmJiAoZGlzcGF0Y2hUaHJlYWQgPT0gbnVsbCkpIHsKLSAgICAgICAgICAgIGRpc3BhdGNoVGhyZWFkID0gbmV3IEV2ZW50UXVldWVUaHJlYWQodGhpcyk7Ci0gICAgICAgICAgICBkaXNwYXRjaFRocmVhZC5zdGFydCgpOwotICAgICAgICB9Ci0gICAgICAgIC8vIFRPRE86IGFkZCBldmVudCBjb2FsZXNjaW5nCi0gICAgICAgIGlmICh0b29sa2l0ICE9IG51bGwpIHsKLSAgICAgICAgICAgIHRvb2xraXQuc2h1dGRvd25XYXRjaGRvZy5zZXRBd3RRdWV1ZUVtcHR5KGZhbHNlKTsKLSAgICAgICAgICAgIGlmICghR3JhcGhpY3NFbnZpcm9ubWVudC5nZXRMb2NhbEdyYXBoaWNzRW52aXJvbm1lbnQoKS5pc0hlYWRsZXNzSW5zdGFuY2UoKSkgewotICAgICAgICAgICAgICAgIG5vdGlmeUV2ZW50TW9uaXRvcih0b29sa2l0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBub3RpZnlBbGwoKTsKLSAgICAgICAgKi8KLSAgICB9Ci0gICAgCi0gICAgdm9pZCBub3RpZnlFdmVudE1vbml0b3IoVG9vbGtpdCB0KSB7Ci0gICAgICAgIE9iamVjdCBlbSA9IHQuZ2V0TmF0aXZlRXZlbnRRdWV1ZSgpLmdldEV2ZW50TW9uaXRvcigpOwotICAgICAgICBzeW5jaHJvbml6ZWQgKGVtKSB7Ci0gICAgICAgICAgICBlbS5ub3RpZnlBbGwoKTsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBzeW5jaHJvbml6ZWQgQVdURXZlbnQgZ2V0TmV4dEV2ZW50KCkgdGhyb3dzIEludGVycnVwdGVkRXhjZXB0aW9uIHsKLSAgICAgICAgd2hpbGUgKGV2ZW50cy5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgIHdhaXQoKTsKLSAgICAgICAgfQotICAgICAgICBBV1RFdmVudCBldmVudCA9IGV2ZW50cy5yZW1vdmVGaXJzdCgpOwotICAgICAgICAvLyBUT0RPOiBhZGQgZXZlbnQgY29hbGVzY2luZwotICAgICAgICByZXR1cm4gZXZlbnQ7Ci0gICAgfSAgICAKLSAgICAKLSAgICBzeW5jaHJvbml6ZWQgQVdURXZlbnQgcGVla0V2ZW50KCkgewotICAgICAgICByZXR1cm4gZXZlbnRzLmlzRW1wdHkoKSA/IG51bGwgOiBldmVudHMuZ2V0Rmlyc3QoKTsKLSAgICB9Ci0gICAgCi0gICAgc3luY2hyb25pemVkIEFXVEV2ZW50IHBlZWtFdmVudChpbnQgaWQpIHsKLSAgICAgICAgZm9yIChBV1RFdmVudCBldmVudCA6IGV2ZW50cykgewotICAgICAgICAgICAgaWYgKGV2ZW50LmdldElEKCkgPT0gaWQpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZXZlbnQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotICAgIAotICAgIHN5bmNocm9uaXplZCB2b2lkIGRpc3BhdGNoRXZlbnQoQVdURXZlbnQgZXZlbnQpIHsKLSAgICAgICAgdXBkYXRlQ3VycmVudEV2ZW50QW5kVGltZShldmVudCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBhY3RpdmVRdWV1ZS5kaXNwYXRjaEV2ZW50KGV2ZW50KTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIGN1cnJlbnRFdmVudCA9IG51bGw7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgdm9pZCBkaXNwYXRjaEV2ZW50SW1wbChBV1RFdmVudCBldmVudCkgewotICAgICAgICBpZiAoZXZlbnQgaW5zdGFuY2VvZiBBY3RpdmVFdmVudCkgewotICAgICAgICAgICAgdXBkYXRlQ3VycmVudEV2ZW50QW5kVGltZShldmVudCk7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgICgoQWN0aXZlRXZlbnQpIGV2ZW50KS5kaXNwYXRjaCgpOwotICAgICAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgICAgICBjdXJyZW50RXZlbnQgPSBudWxsOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgT2JqZWN0IHNyYyA9IGV2ZW50LmdldFNvdXJjZSgpOwotCi0gICAgICAgIGlmIChzcmMgaW5zdGFuY2VvZiBDb21wb25lbnQpIHsKLSAgICAgICAgICAgIGlmIChwcmVwcm9jZXNzQ29tcG9uZW50RXZlbnQoZXZlbnQpKSB7Ci0gICAgICAgICAgICAgICAgKChDb21wb25lbnQpIHNyYykuZGlzcGF0Y2hFdmVudChldmVudCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpZiAodG9vbGtpdCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdG9vbGtpdC5kaXNwYXRjaEFXVEV2ZW50KGV2ZW50KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChzcmMgaW5zdGFuY2VvZiBNZW51Q29tcG9uZW50KSB7Ci0gICAgICAgICAgICAgICAgKChNZW51Q29tcG9uZW50KSBzcmMpLmRpc3BhdGNoRXZlbnQoZXZlbnQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBmaW5hbCBib29sZWFuIHByZXByb2Nlc3NDb21wb25lbnRFdmVudChBV1RFdmVudCBldmVudCkgewotICAgICAgaWYgKGV2ZW50IGluc3RhbmNlb2YgTW91c2VFdmVudCkgewotICAgICAgICAgIHJldHVybiBwcmVwcm9jZXNzTW91c2VFdmVudCgoTW91c2VFdmVudClldmVudCk7Ci0gICAgICB9Ci0gICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBwcml2YXRlIGZpbmFsIGJvb2xlYW4gcHJlcHJvY2Vzc01vdXNlRXZlbnQoTW91c2VFdmVudCBldmVudCkgewotICAgICAgICAvLz8/P0FXVAotICAgICAgICAvKgotICAgICAgaWYgKHRvb2xraXQgIT0gbnVsbCAmJiB0b29sa2l0Lm1vdXNlRXZlbnRQcmVwcm9jZXNzb3IgIT0gbnVsbCkgewotICAgICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgIHJldHVybiB0b29sa2l0Lm1vdXNlRXZlbnRQcmVwcm9jZXNzb3IucHJlcHJvY2VzcyhldmVudCk7Ci0gICAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgICB9Ci0gICAgICB9Ci0gICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgKi8KLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotICAgIAotICAgIHByaXZhdGUgdm9pZCB1cGRhdGVDdXJyZW50RXZlbnRBbmRUaW1lKEFXVEV2ZW50IGV2ZW50KSB7Ci0gICAgICAgIGN1cnJlbnRFdmVudCA9IGV2ZW50OwotICAgICAgICBsb25nIHdoZW4gPSAwOwotICAgICAgICBpZiAoZXZlbnQgaW5zdGFuY2VvZiBBY3Rpb25FdmVudCkgewotICAgICAgICAgICAgd2hlbiA9ICgoQWN0aW9uRXZlbnQpIGV2ZW50KS5nZXRXaGVuKCk7Ci0gICAgICAgIH0gZWxzZSBpZiAoZXZlbnQgaW5zdGFuY2VvZiBJbnB1dEV2ZW50KSB7Ci0gICAgICAgICAgICB3aGVuID0gKChJbnB1dEV2ZW50KSBldmVudCkuZ2V0V2hlbigpOwotICAgICAgICB9IGVsc2UgaWYgKGV2ZW50IGluc3RhbmNlb2YgSW5wdXRNZXRob2RFdmVudCkgewotICAgICAgICAgICAgd2hlbiA9ICgoSW5wdXRNZXRob2RFdmVudCkgZXZlbnQpLmdldFdoZW4oKTsKLSAgICAgICAgfSBlbHNlIGlmIChldmVudCBpbnN0YW5jZW9mIEludm9jYXRpb25FdmVudCkgewotICAgICAgICAgICAgd2hlbiA9ICgoSW52b2NhdGlvbkV2ZW50KSBldmVudCkuZ2V0V2hlbigpOwotICAgICAgICB9Ci0gICAgICAgIGlmICh3aGVuICE9IDApIHsKLSAgICAgICAgICAgIG1vc3RSZWNlbnRFdmVudFRpbWUgPSB3aGVuOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIHN5bmNocm9uaXplZCB2b2lkIHB1c2goRXZlbnRRdWV1ZSBuZXdFdmVudFF1ZXVlKSB7Ci0gICAgICAgIC8vIFRPRE86IGhhbmRsZSBpbmNvcnJlY3Qgc2l0dWF0aW9ucwotICAgICAgICBpZiAocXVldWVTdGFjay5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Qj1RdWV1ZSBzdGFjayBpcyBlbXB0eQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxTdGF0ZUV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42QiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBxdWV1ZVN0YWNrLmFkZExhc3QobmV3RXZlbnRRdWV1ZSk7Ci0gICAgICAgIGFjdGl2ZVF1ZXVlID0gbmV3RXZlbnRRdWV1ZTsKLSAgICAgICAgYWN0aXZlUXVldWUuc2V0Q29yZSh0aGlzKTsKLSAgICB9Ci0gICAgCi0gICAgc3luY2hyb25pemVkIHZvaWQgcG9wKCkgewotICAgICAgICBFdmVudFF1ZXVlIHJlbW92ZWQgPSBxdWV1ZVN0YWNrLnJlbW92ZUxhc3QoKTsKLSAgICAgICAgaWYgKHJlbW92ZWQgIT0gYWN0aXZlUXVldWUpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Qz1FdmVudCBxdWV1ZSBzdGFjayBpcyBicm9rZW4KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNkMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBhY3RpdmVRdWV1ZSA9IHF1ZXVlU3RhY2suZ2V0TGFzdCgpOwotICAgICAgICByZW1vdmVkLnNldENvcmUobnVsbCk7Ci0gICAgfQotCi0gICAgc3luY2hyb25pemVkIEFXVEV2ZW50IGdldE5leHRFdmVudE5vV2FpdCgpIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBldmVudHMuaXNFbXB0eSgpID8gbnVsbCA6IGFjdGl2ZVF1ZXVlLmdldE5leHRFdmVudCgpOwotICAgICAgICB9IGNhdGNoIChJbnRlcnJ1cHRlZEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHN5bmNocm9uaXplZCBib29sZWFuIGlzRW1wdHkoKSB7Ci0gICAgICAgIHJldHVybiAoY3VycmVudEV2ZW50ID09IG51bGwpICYmIGV2ZW50cy5pc0VtcHR5KCk7Ci0gICAgfQotICAgIAotICAgIHN5bmNocm9uaXplZCBib29sZWFuIGlzRW1wdHkobG9uZyB0aW1lb3V0KSB7Ci0gICAgICAgIGlmICghaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHdhaXQodGltZW91dCk7Ci0gICAgICAgIH0gY2F0Y2ggKEludGVycnVwdGVkRXhjZXB0aW9uIGUpIHt9Ci0gICAgICAgIHJldHVybiBpc0VtcHR5KCk7Ci0gICAgfQotICAgIAotICAgIHN5bmNocm9uaXplZCBFdmVudFF1ZXVlIGdldEFjdGl2ZUV2ZW50UXVldWUoKSB7Ci0gICAgICAgIHJldHVybiBhY3RpdmVRdWV1ZTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvRm9udC5qYXZhIGIvYXd0L2phdmEvYXd0L0ZvbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNGVkOTM0My4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvRm9udC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTU0MSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgY29tLmFuZHJvaWQuaW50ZXJuYWwuYXd0LkFuZHJvaWRHcmFwaGljczJEOwotCi1pbXBvcnQgamF2YS5hd3QuZm9udC5Gb250UmVuZGVyQ29udGV4dDsKLWltcG9ydCBqYXZhLmF3dC5mb250LkdseXBoVmVjdG9yOwotaW1wb3J0IGphdmEuYXd0LmZvbnQuTGluZU1ldHJpY3M7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5UZXh0QXR0cmlidXRlOwotaW1wb3J0IGphdmEuYXd0LmZvbnQuVHJhbnNmb3JtQXR0cmlidXRlOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS5pby5GaWxlOwotaW1wb3J0IGphdmEuaW8uRmlsZUlucHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uQnVmZmVyZWRJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLkZpbGVPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLklucHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uU2VyaWFsaXphYmxlOwotaW1wb3J0IGphdmEudGV4dC5DaGFyYWN0ZXJJdGVyYXRvcjsKLWltcG9ydCBqYXZhLnRleHQuQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yLkF0dHJpYnV0ZTsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaHRhYmxlOwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci1pbXBvcnQgamF2YS51dGlsLk1hcDsKLWltcG9ydCBqYXZhLnV0aWwuU3RyaW5nVG9rZW5pemVyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250LkFuZHJvaWRHbHlwaFZlY3RvcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuQ29tbW9uR2x5cGhWZWN0b3I7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250LkZvbnRQZWVySW1wbDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuRm9udE1ldHJpY3NJbXBsOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5MaW5lTWV0cmljc0ltcGw7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Ob3RJbXBsZW1lbnRlZEV4Y2VwdGlvbjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkubWlzYy5IYXNoQ29kZTsKLQotLyoqCi0gKiBUaGUgRm9udCBjbGFzcyByZXByZXNlbnRzIGZvbnRzIGZvciByZW5kZXJpbmcgdGV4dC4gVGhpcyBjbGFzcyBhbGxvdyB0byBtYXAKLSAqIGNoYXJhY3RlcnMgdG8gZ2x5cGhzLgotICogPHA+Ci0gKiBBIGdseXBoIGlzIGEgc2hhcGUgdXNlZCB0byByZW5kZXIgYSBjaGFyYWN0ZXIgb3IgYSBzZXF1ZW5jZSBvZiBjaGFyYWN0ZXJzLgotICogRm9yIGV4YW1wbGUgb25lIGNoYXJhY3RlciBvZiBMYXRpbiB3cml0aW5nIHN5c3RlbSByZXByZXNlbnRlZCBieSBvbmUgZ2x5cGgsCi0gKiBidXQgaW4gY29tcGxleCB3cml0aW5nIHN5c3RlbSBzdWNoIGFzIFNvdXRoIGFuZCBTb3V0aC1FYXN0IEFzaWFuIHRoZXJlIGlzCi0gKiBtb3JlIGNvbXBsaWNhdGVkIGNvcnJlc3BvbmRlbmNlIGJldHdlZW4gY2hhcmFjdGVycyBhbmQgZ2x5cGhzLgotICogPHA+Ci0gKiBUaGUgRm9udCBvYmplY3QgaXMgaWRlbnRpZmllZCBieSB0d28gdHlwZXMgb2YgbmFtZXMuIFRoZSBsb2dpY2FsIGZvbnQgbmFtZSBpcwotICogdGhlIG5hbWUgdGhhdCBpcyB1c2VkIHRvIGNvbnN0cnVjdCB0aGUgZm9udC4gVGhlIGZvbnQgbmFtZSBpcyB0aGUgbmFtZSBvZiBhCi0gKiBwYXJ0aWN1bGFyIGZvbnQgZmFjZSAoZm9yIGV4YW1wbGUsIEFyaWFsIEJvbGQpLiBUaGUgZmFtaWx5IG5hbWUgaXMgdGhlIGZvbnQncwotICogZmFtaWx5IG5hbWUgdGhhdCBzcGVjaWZpZXMgdGhlIHR5cG9ncmFwaGljIGRlc2lnbiBhY3Jvc3Mgc2V2ZXJhbCBmYWNlcyAoZm9yCi0gKiBleGFtcGxlLCBBcmlhbCkuIEluIGFsbCB0aGUgRm9udCBpcyBpZGVudGlmaWVkIGJ5IHRocmVlIGF0dHJpYnV0ZXM6IHRoZQotICogZmFtaWx5IG5hbWUsIHRoZSBzdHlsZSAoc3VjaCBhcyBib2xkIG9yIGl0YWxpYyksIGFuZCB0aGUgc2l6ZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBGb250IGltcGxlbWVudHMgU2VyaWFsaXphYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC00MjA2MDIxMzExNTkxNDU5MjEzTDsKLQotICAgIC8vIElkZW50aXR5IFRyYW5zZm9ybSBhdHRyaWJ1dGUKLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgSURFTlRJVFlfVFJBTlNGT1JNLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFRyYW5zZm9ybUF0dHJpYnV0ZSBJREVOVElUWV9UUkFOU0ZPUk0gPSBuZXcgVHJhbnNmb3JtQXR0cmlidXRlKAotICAgICAgICAgICAgbmV3IEFmZmluZVRyYW5zZm9ybSgpKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBQTEFJTiBpbmRpY2F0ZXMgZm9udCdzIHBsYWluIHN0eWxlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFBMQUlOID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBCT0xEIGluZGljYXRlcyBmb250J3MgYm9sZCBzdHlsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBCT0xEID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBJVEFMSUMgaW5kaWNhdGVzIGZvbnQncyBpdGFsaWMgc3R5bGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSVRBTElDID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBST01BTl9CQVNFTElORSBpbmRpY2F0ZWQgcm9tYW4gYmFzZWxpbmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgUk9NQU5fQkFTRUxJTkUgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENFTlRFUl9CQVNFTElORSBpbmRpY2F0ZXMgY2VudGVyIGJhc2VsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENFTlRFUl9CQVNFTElORSA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgSEFOR0lOR19CQVNFTElORSBpbmRpY2F0ZXMgaGFuZ2luZyBiYXNlbGluZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBIQU5HSU5HX0JBU0VMSU5FID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUUlVFVFlQRV9GT05UIGluZGljYXRlcyBhIGZvbnQgcmVzb3VyY2Ugb2YgdHlwZSBUUlVFVFlQRS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUUlVFVFlQRV9GT05UID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFMV9GT05UIGluZGljYXRlcyBhIGZvbnQgcmVzb3VyY2Ugb2YgdHlwZSBUWVBFMS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFMV9GT05UID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBMQVlPVVRfTEVGVF9UT19SSUdIVCBpbmRpY2F0ZXMgdGhhdCB0ZXh0IGlzIGxlZnQgdG8gcmlnaHQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTEFZT1VUX0xFRlRfVE9fUklHSFQgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IExBWU9VVF9SSUdIVF9UT19MRUZUIGluZGljYXRlcyB0aGF0IHRleHQgaXMgcmlnaHQgdG8gbGVmdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBMQVlPVVRfUklHSFRfVE9fTEVGVCA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTEFZT1VUX05PX1NUQVJUX0NPTlRFWFQgaW5kaWNhdGVzIHRoYXQgdGhlIHRleHQgaW4gdGhlIGNoYXIKLSAgICAgKiBhcnJheSBiZWZvcmUgdGhlIGluZGljYXRlZCBzdGFydCBzaG91bGQgbm90IGJlIGV4YW1pbmVkLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IExBWU9VVF9OT19TVEFSVF9DT05URVhUID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBMQVlPVVRfTk9fTElNSVRfQ09OVEVYVCBpbmRpY2F0ZXMgdGhhdCB0ZXh0IGluIHRoZSBjaGFyCi0gICAgICogYXJyYXkgYWZ0ZXIgdGhlIGluZGljYXRlZCBsaW1pdCBzaG91bGQgbm90IGJlIGV4YW1pbmVkLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IExBWU9VVF9OT19MSU1JVF9DT05URVhUID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBERUZBVUxUX0ZPTlQuCi0gICAgICovCi0gICAgc3RhdGljIGZpbmFsIEZvbnQgREVGQVVMVF9GT05UID0gbmV3IEZvbnQoIkRpYWxvZyIsIEZvbnQuUExBSU4sIDEyKTsgLy8kTk9OLU5MUy0xJAotCi0gICAgLyoqCi0gICAgICogVGhlIG5hbWUgb2YgdGhlIEZvbnQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZyBuYW1lOwotCi0gICAgLyoqCi0gICAgICogVGhlIHN0eWxlIG9mIHRoZSBGb250LgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgc3R5bGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc2l6ZSBvZiB0aGUgRm9udC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IHNpemU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcG9pbnQgc2l6ZSBvZiB0aGUgRm9udC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgZmxvYXQgcG9pbnRTaXplOwotCi0gICAgLy8gRmxhZyBpZiB0aGUgRm9udCBvYmplY3QgdHJhbnNmb3JtZWQKLSAgICAvKioKLSAgICAgKiBUaGUgdHJhbnNmb3JtZWQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIHRyYW5zZm9ybWVkOwotCi0gICAgLy8gU2V0IG9mIGZvbnQgYXR0cmlidXRlcwotICAgIC8qKgotICAgICAqIFRoZSByZXF1ZXN0ZWQgYXR0cmlidXRlcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIEhhc2h0YWJsZTxBdHRyaWJ1dGUsIE9iamVjdD4gZlJlcXVlc3RlZEF0dHJpYnV0ZXM7Ci0KLSAgICAvLyBmb250IHBlZXIgb2JqZWN0IGNvcnJlc3BvbmRpbmcgdG8gdGhpcyBGb250Ci0gICAgLyoqCi0gICAgICogVGhlIGZvbnQgcGVlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIHRyYW5zaWVudCBGb250UGVlckltcGwgZm9udFBlZXI7Ci0KLSAgICAvLyBudW1iZXIgb2YgZ2x5cGhzIGluIHRoaXMgRm9udAotICAgIC8qKgotICAgICAqIFRoZSBudW0gZ2x5cGhzLgotICAgICAqLwotICAgIHByaXZhdGUgdHJhbnNpZW50IGludCBudW1HbHlwaHMgPSAtMTsKLQotICAgIC8vIGNvZGUgZm9yIG1pc3NpbmcgZ2x5cGggZm9yIHRoaXMgRm9udAotICAgIC8qKgotICAgICAqIFRoZSBtaXNzaW5nIGdseXBoIGNvZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSB0cmFuc2llbnQgaW50IG1pc3NpbmdHbHlwaENvZGUgPSAtMTsKLQotICAgIC8qKgotICAgICAqIFdyaXRlcyBvYmplY3QgdG8gT2JqZWN0T3V0cHV0U3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvdXQKLSAgICAgKiAgICAgICAgICAgIE9iamVjdE91dHB1dFN0cmVhbS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgU2lnbmFscyB0aGF0IGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCB3cml0ZU9iamVjdChqYXZhLmlvLk9iamVjdE91dHB1dFN0cmVhbSBvdXQpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIG91dC5kZWZhdWx0V3JpdGVPYmplY3QoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyBvYmplY3QgZnJvbSBPYmplY3RJbnB1dFN0cmVhbSBvYmplY3QgYW5kIHNldCBuYXRpdmUgcGxhdGZvcm0KLSAgICAgKiBkZXBlbmRlbnQgZmllbGRzIHRvIGRlZmF1bHQgdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbgotICAgICAqICAgICAgICAgICAgT2JqZWN0SW5wdXRTdHJlYW0gb2JqZWN0LgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBTaWduYWxzIHRoYXQgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICogQHRocm93cyBDbGFzc05vdEZvdW5kRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgdGhlIGNsYXNzIG5vdCBmb3VuZCBleGNlcHRpb24uCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHJlYWRPYmplY3QoamF2YS5pby5PYmplY3RJbnB1dFN0cmVhbSBpbikgdGhyb3dzIElPRXhjZXB0aW9uLAotICAgICAgICAgICAgQ2xhc3NOb3RGb3VuZEV4Y2VwdGlvbiB7Ci0gICAgICAgIGluLmRlZmF1bHRSZWFkT2JqZWN0KCk7Ci0KLSAgICAgICAgbnVtR2x5cGhzID0gLTE7Ci0gICAgICAgIG1pc3NpbmdHbHlwaENvZGUgPSAtMTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBGb250IHdpdGggdGhlIHNwZWNpZmllZCBhdHRyaWJ1dGVzLiBUaGUgRm9udCB3aWxsIGJlCi0gICAgICogY3JlYXRlZCB3aXRoIGRlZmF1bHQgYXR0cmlidXRlcyBpZiB0aGUgYXR0cmlidXRlJ3MgcGFyYW1ldGVyIGlzIG51bGwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGF0dHJpYnV0ZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhdHRyaWJ1dGVzIHRvIGJlIGFzc2lnbmVkIHRvIHRoZSBuZXcgRm9udCwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgRm9udChNYXA8PyBleHRlbmRzIEF0dHJpYnV0ZSwgPz4gYXR0cmlidXRlcykgewotICAgICAgICBPYmplY3QgY3VyckF0dHI7Ci0KLSAgICAgICAgLy8gRGVmYXVsdCB2YWx1ZXMgYXJlIHRha2VuIGZyb20gdGhlIGRvY3VtZW50YXRpb24gb2YgdGhlIEZvbnQgY2xhc3MuCi0gICAgICAgIC8vIFNlZSBGb250IGNvbnN0cnVjdG9yLCBkZWNvZGUgYW5kIGdldEZvbnQgc2VjdGlvbnMuCi0KLSAgICAgICAgdGhpcy5uYW1lID0gImRlZmF1bHQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIHRoaXMuc2l6ZSA9IDEyOwotICAgICAgICB0aGlzLnBvaW50U2l6ZSA9IDEyOwotICAgICAgICB0aGlzLnN0eWxlID0gRm9udC5QTEFJTjsKLQotICAgICAgICBpZiAoYXR0cmlidXRlcyAhPSBudWxsKSB7Ci0KLSAgICAgICAgICAgIGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzID0gbmV3IEhhc2h0YWJsZTxBdHRyaWJ1dGUsIE9iamVjdD4oYXR0cmlidXRlcyk7Ci0KLSAgICAgICAgICAgIGN1cnJBdHRyID0gYXR0cmlidXRlcy5nZXQoVGV4dEF0dHJpYnV0ZS5TSVpFKTsKLSAgICAgICAgICAgIGlmIChjdXJyQXR0ciAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdGhpcy5wb2ludFNpemUgPSAoKEZsb2F0KWN1cnJBdHRyKS5mbG9hdFZhbHVlKCk7Ci0gICAgICAgICAgICAgICAgdGhpcy5zaXplID0gKGludClNYXRoLmNlaWwodGhpcy5wb2ludFNpemUpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjdXJyQXR0ciA9IGF0dHJpYnV0ZXMuZ2V0KFRleHRBdHRyaWJ1dGUuUE9TVFVSRSk7Ci0gICAgICAgICAgICBpZiAoY3VyckF0dHIgIT0gbnVsbCAmJiBjdXJyQXR0ci5lcXVhbHMoVGV4dEF0dHJpYnV0ZS5QT1NUVVJFX09CTElRVUUpKSB7Ci0gICAgICAgICAgICAgICAgdGhpcy5zdHlsZSB8PSBGb250LklUQUxJQzsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY3VyckF0dHIgPSBhdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLldFSUdIVCk7Ci0gICAgICAgICAgICBpZiAoKGN1cnJBdHRyICE9IG51bGwpCi0gICAgICAgICAgICAgICAgICAgICYmICgoKEZsb2F0KWN1cnJBdHRyKS5mbG9hdFZhbHVlKCkgPj0gKFRleHRBdHRyaWJ1dGUuV0VJR0hUX0JPTEQpLmZsb2F0VmFsdWUoKSkpIHsKLSAgICAgICAgICAgICAgICB0aGlzLnN0eWxlIHw9IEZvbnQuQk9MRDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY3VyckF0dHIgPSBhdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLkZBTUlMWSk7Ci0gICAgICAgICAgICBpZiAoY3VyckF0dHIgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHRoaXMubmFtZSA9IChTdHJpbmcpY3VyckF0dHI7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGN1cnJBdHRyID0gYXR0cmlidXRlcy5nZXQoVGV4dEF0dHJpYnV0ZS5UUkFOU0ZPUk0pOwotICAgICAgICAgICAgaWYgKGN1cnJBdHRyICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBpZiAoY3VyckF0dHIgaW5zdGFuY2VvZiBUcmFuc2Zvcm1BdHRyaWJ1dGUpIHsKLSAgICAgICAgICAgICAgICAgICAgdGhpcy50cmFuc2Zvcm1lZCA9ICEoKFRyYW5zZm9ybUF0dHJpYnV0ZSljdXJyQXR0cikuZ2V0VHJhbnNmb3JtKCkuaXNJZGVudGl0eSgpOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoY3VyckF0dHIgaW5zdGFuY2VvZiBBZmZpbmVUcmFuc2Zvcm0pIHsKLSAgICAgICAgICAgICAgICAgICAgdGhpcy50cmFuc2Zvcm1lZCA9ICEoKEFmZmluZVRyYW5zZm9ybSljdXJyQXR0cikuaXNJZGVudGl0eSgpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZlJlcXVlc3RlZEF0dHJpYnV0ZXMgPSBuZXcgSGFzaHRhYmxlPEF0dHJpYnV0ZSwgT2JqZWN0Pig1KTsKLSAgICAgICAgICAgIGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnB1dChUZXh0QXR0cmlidXRlLlRSQU5TRk9STSwgSURFTlRJVFlfVFJBTlNGT1JNKTsKLQotICAgICAgICAgICAgdGhpcy50cmFuc2Zvcm1lZCA9IGZhbHNlOwotCi0gICAgICAgICAgICBmUmVxdWVzdGVkQXR0cmlidXRlcy5wdXQoVGV4dEF0dHJpYnV0ZS5GQU1JTFksIG5hbWUpOwotCi0gICAgICAgICAgICBmUmVxdWVzdGVkQXR0cmlidXRlcy5wdXQoVGV4dEF0dHJpYnV0ZS5TSVpFLCBuZXcgRmxvYXQodGhpcy5zaXplKSk7Ci0KLSAgICAgICAgICAgIGlmICgodGhpcy5zdHlsZSAmIEZvbnQuQk9MRCkgIT0gMCkgewotICAgICAgICAgICAgICAgIGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnB1dChUZXh0QXR0cmlidXRlLldFSUdIVCwgVGV4dEF0dHJpYnV0ZS5XRUlHSFRfQk9MRCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnB1dChUZXh0QXR0cmlidXRlLldFSUdIVCwgVGV4dEF0dHJpYnV0ZS5XRUlHSFRfUkVHVUxBUik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoKHRoaXMuc3R5bGUgJiBGb250LklUQUxJQykgIT0gMCkgewotICAgICAgICAgICAgICAgIGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnB1dChUZXh0QXR0cmlidXRlLlBPU1RVUkUsIFRleHRBdHRyaWJ1dGUuUE9TVFVSRV9PQkxJUVVFKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZlJlcXVlc3RlZEF0dHJpYnV0ZXMucHV0KFRleHRBdHRyaWJ1dGUuUE9TVFVSRSwgVGV4dEF0dHJpYnV0ZS5QT1NUVVJFX1JFR1VMQVIpOwotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRm9udCB3aXRoIHRoZSBzcGVjaWZpZWQgbmFtZSwgc3R5bGUgYW5kIHNpemUuCi0gICAgICogCi0gICAgICogQHBhcmFtIG5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYW1lIG9mIGZvbnQuCi0gICAgICogQHBhcmFtIHN0eWxlCi0gICAgICogICAgICAgICAgICB0aGUgc3R5bGUgb2YgZm9udC4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIHNpemUgb2YgZm9udC4KLSAgICAgKi8KLSAgICBwdWJsaWMgRm9udChTdHJpbmcgbmFtZSwgaW50IHN0eWxlLCBpbnQgc2l6ZSkgewotCi0gICAgICAgIHRoaXMubmFtZSA9IChuYW1lICE9IG51bGwpID8gbmFtZSA6ICJEZWZhdWx0IjsgLy8kTk9OLU5MUy0xJAotICAgICAgICB0aGlzLnNpemUgPSAoc2l6ZSA+PSAwKSA/IHNpemUgOiAwOwotICAgICAgICB0aGlzLnN0eWxlID0gKHN0eWxlICYgfjB4MDMpID09IDAgPyBzdHlsZSA6IEZvbnQuUExBSU47Ci0gICAgICAgIHRoaXMucG9pbnRTaXplID0gdGhpcy5zaXplOwotCi0gICAgICAgIGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzID0gbmV3IEhhc2h0YWJsZTxBdHRyaWJ1dGUsIE9iamVjdD4oNSk7Ci0KLSAgICAgICAgZlJlcXVlc3RlZEF0dHJpYnV0ZXMucHV0KFRleHRBdHRyaWJ1dGUuVFJBTlNGT1JNLCBJREVOVElUWV9UUkFOU0ZPUk0pOwotCi0gICAgICAgIHRoaXMudHJhbnNmb3JtZWQgPSBmYWxzZTsKLQotICAgICAgICBmUmVxdWVzdGVkQXR0cmlidXRlcy5wdXQoVGV4dEF0dHJpYnV0ZS5GQU1JTFksIHRoaXMubmFtZSk7Ci0gICAgICAgIGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnB1dChUZXh0QXR0cmlidXRlLlNJWkUsIG5ldyBGbG9hdCh0aGlzLnNpemUpKTsKLQotICAgICAgICBpZiAoKHRoaXMuc3R5bGUgJiBGb250LkJPTEQpICE9IDApIHsKLSAgICAgICAgICAgIGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnB1dChUZXh0QXR0cmlidXRlLldFSUdIVCwgVGV4dEF0dHJpYnV0ZS5XRUlHSFRfQk9MRCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmUmVxdWVzdGVkQXR0cmlidXRlcy5wdXQoVGV4dEF0dHJpYnV0ZS5XRUlHSFQsIFRleHRBdHRyaWJ1dGUuV0VJR0hUX1JFR1VMQVIpOwotICAgICAgICB9Ci0gICAgICAgIGlmICgodGhpcy5zdHlsZSAmIEZvbnQuSVRBTElDKSAhPSAwKSB7Ci0gICAgICAgICAgICBmUmVxdWVzdGVkQXR0cmlidXRlcy5wdXQoVGV4dEF0dHJpYnV0ZS5QT1NUVVJFLCBUZXh0QXR0cmlidXRlLlBPU1RVUkVfT0JMSVFVRSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmUmVxdWVzdGVkQXR0cmlidXRlcy5wdXQoVGV4dEF0dHJpYnV0ZS5QT1NUVVJFLCBUZXh0QXR0cmlidXRlLlBPU1RVUkVfUkVHVUxBUik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhpcyBGb250IGhhcyBhIGdseXBoIGZvciB0aGUgc3BlY2lmaWVkIGNoYXJhY3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYwotICAgICAqICAgICAgICAgICAgdGhlIGNoYXJhY3Rlci4KLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgdGhpcyBGb250IGhhcyBhIGdseXBoIGZvciB0aGUgc3BlY2lmaWVkIGNoYXJhY3RlciwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjYW5EaXNwbGF5KGNoYXIgYykgewotICAgICAgICBGb250UGVlckltcGwgcGVlciA9IChGb250UGVlckltcGwpdGhpcy5nZXRQZWVyKCk7Ci0gICAgICAgIHJldHVybiBwZWVyLmNhbkRpc3BsYXkoYyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoZSBGb250IGNhbiBkaXNwbGF5IHRoZSBjaGFyYWN0ZXJzIG9mIHRoZSB0aGUgc3BlY2lmaWVkCi0gICAgICogdGV4dCBmcm9tIHRoZSBzcGVjaWZpZWQgc3RhcnQgcG9zaXRpb24gdG8gdGhlIHNwZWNpZmllZCBsaW1pdCBwb3NpdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdGV4dAotICAgICAqICAgICAgICAgICAgdGhlIHRleHQuCi0gICAgICogQHBhcmFtIHN0YXJ0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgb2Zmc2V0IChpbiB0aGUgY2hhcmFjdGVyIGFycmF5KS4KLSAgICAgKiBAcGFyYW0gbGltaXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBsaW1pdCBvZmZzZXQgKGluIHRoZSBjaGFyYWN0ZXIgYXJyYXkpLgotICAgICAqIEByZXR1cm4gdGhlIGEgY2hhcmFjdGVyJ3MgcG9zaXRpb24gaW4gdGhlIHRleHQgdGhhdCB0aGlzIEZvbnQgY2FuIG5vdAotICAgICAqICAgICAgICAgZGlzcGxheSwgb3IgLTEgaWYgdGhpcyBGb250IGNhbiBkaXNwbGF5IGFsbCBjaGFyYWN0ZXJzIGluIHRoaXMKLSAgICAgKiAgICAgICAgIHRleHQuCi0gICAgICovCi0gICAgcHVibGljIGludCBjYW5EaXNwbGF5VXBUbyhjaGFyW10gdGV4dCwgaW50IHN0YXJ0LCBpbnQgbGltaXQpIHsKLSAgICAgICAgaW50IHN0ID0gc3RhcnQ7Ci0gICAgICAgIGludCByZXN1bHQ7Ci0gICAgICAgIHdoaWxlICgoc3QgPCBsaW1pdCkgJiYgY2FuRGlzcGxheSh0ZXh0W3N0XSkpIHsKLSAgICAgICAgICAgIHN0Kys7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoc3QgPT0gbGltaXQpIHsKLSAgICAgICAgICAgIHJlc3VsdCA9IC0xOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcmVzdWx0ID0gc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcmVzdWx0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgRm9udCBjYW4gZGlzcGxheSB0aGUgY2hhcmFjdGVycyBvZiB0aGUgdGhlIHNwZWNpZmllZAotICAgICAqIENoYXJhY3Rlckl0ZXJhdG9yIGZyb20gdGhlIHNwZWNpZmllZCBzdGFydCBwb3NpdGlvbiBhbmQgdGhlIHNwZWNpZmllZAotICAgICAqIGxpbWl0IHBvc2l0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpdGVyCi0gICAgICogICAgICAgICAgICB0aGUgQ2hhcmFjdGVySXRlcmF0b3IuCi0gICAgICogQHBhcmFtIHN0YXJ0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBsaW1pdAotICAgICAqICAgICAgICAgICAgdGhlIGxpbWl0IG9mZnNldC4KLSAgICAgKiBAcmV0dXJuIHRoZSBhIGNoYXJhY3RlcidzIHBvc2l0aW9uIGluIHRoZSBDaGFyYWN0ZXJJdGVyYXRvciB0aGF0IHRoaXMKLSAgICAgKiAgICAgICAgIEZvbnQgY2FuIG5vdCBkaXNwbGF5LCBvciAtMSBpZiB0aGlzIEZvbnQgY2FuIGRpc3BsYXkgYWxsCi0gICAgICogICAgICAgICBjaGFyYWN0ZXJzIGluIHRoaXMgdGV4dC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGNhbkRpc3BsYXlVcFRvKENoYXJhY3Rlckl0ZXJhdG9yIGl0ZXIsIGludCBzdGFydCwgaW50IGxpbWl0KSB7Ci0gICAgICAgIGludCBzdCA9IHN0YXJ0OwotICAgICAgICBjaGFyIGMgPSBpdGVyLnNldEluZGV4KHN0YXJ0KTsKLSAgICAgICAgaW50IHJlc3VsdDsKLQotICAgICAgICB3aGlsZSAoKHN0IDwgbGltaXQpICYmIChjYW5EaXNwbGF5KGMpKSkgewotICAgICAgICAgICAgc3QrKzsKLSAgICAgICAgICAgIGMgPSBpdGVyLm5leHQoKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoc3QgPT0gbGltaXQpIHsKLSAgICAgICAgICAgIHJlc3VsdCA9IC0xOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcmVzdWx0ID0gc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcmVzdWx0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGlzIEZvbnQgY2FuIGRpc3BsYXkgYSBzcGVjaWZpZWQgU3RyaW5nLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBTdHJpbmcuCi0gICAgICogQHJldHVybiB0aGUgYSBjaGFyYWN0ZXIncyBwb3NpdGlvbiBpbiB0aGUgU3RyaW5nIHRoYXQgdGhpcyBGb250IGNhbiBub3QKLSAgICAgKiAgICAgICAgIGRpc3BsYXksIG9yIC0xIGlmIHRoaXMgRm9udCBjYW4gZGlzcGxheSBhbGwgY2hhcmFjdGVycyBpbiB0aGlzCi0gICAgICogICAgICAgICB0ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgY2FuRGlzcGxheVVwVG8oU3RyaW5nIHN0cikgewotICAgICAgICBjaGFyW10gY2hhcnMgPSBzdHIudG9DaGFyQXJyYXkoKTsKLSAgICAgICAgcmV0dXJuIGNhbkRpc3BsYXlVcFRvKGNoYXJzLCAwLCBjaGFycy5sZW5ndGgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBHbHlwaFZlY3RvciBvZiBhc3NvY2lhdGluZyBjaGFyYWN0ZXJzIHRvIGdseXBocyBiYXNlZCBvbiB0aGUKLSAgICAgKiBVbmljb2RlIG1hcCBvZiB0aGlzIEZvbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZyYwotICAgICAqICAgICAgICAgICAgdGhlIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqIEBwYXJhbSBjaGFycwotICAgICAqICAgICAgICAgICAgdGhlIGNoYXJhY3RlcnMgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgR2x5cGhWZWN0b3Igb2YgYXNzb2NpYXRpbmcgY2hhcmFjdGVycyB0byBnbHlwaHMgYmFzZWQgb24gdGhlCi0gICAgICogICAgICAgICBVbmljb2RlIG1hcCBvZiB0aGlzIEZvbnQuCi0gICAgICovCi0gICAgcHVibGljIEdseXBoVmVjdG9yIGNyZWF0ZUdseXBoVmVjdG9yKEZvbnRSZW5kZXJDb250ZXh0IGZyYywgY2hhcltdIGNoYXJzKSB7Ci0gICAgICAgIHJldHVybiBuZXcgQW5kcm9pZEdseXBoVmVjdG9yKGNoYXJzLCBmcmMsIHRoaXMsIDApOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBHbHlwaFZlY3RvciBvZiBhc3NvY2lhdGluZyBjaGFyYWN0ZXJzIGNvbnRhaW5lZCBpbiB0aGUKLSAgICAgKiBzcGVjaWZpZWQgQ2hhcmFjdGVySXRlcmF0b3IgdG8gZ2x5cGhzIGJhc2VkIG9uIHRoZSBVbmljb2RlIG1hcCBvZiB0aGlzCi0gICAgICogRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZnJjCi0gICAgICogICAgICAgICAgICB0aGUgRm9udFJlbmRlckNvbnRleHQuCi0gICAgICogQHBhcmFtIGl0ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBDaGFyYWN0ZXJJdGVyYXRvci4KLSAgICAgKiBAcmV0dXJuIHRoZSBHbHlwaFZlY3RvciBvZiBhc3NvY2lhdGluZyBjaGFyYWN0ZXJzIGNvbnRhaW5lZCBpbiB0aGUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZCBDaGFyYWN0ZXJJdGVyYXRvciB0byBnbHlwaHMgYmFzZWQgb24gdGhlIFVuaWNvZGUgbWFwIG9mCi0gICAgICogICAgICAgICB0aGlzIEZvbnQuCi0gICAgICovCi0gICAgcHVibGljIEdseXBoVmVjdG9yIGNyZWF0ZUdseXBoVmVjdG9yKEZvbnRSZW5kZXJDb250ZXh0IGZyYywgQ2hhcmFjdGVySXRlcmF0b3IgaXRlcikgewotICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOyAvLyROT04tTkxTLTEkICAgIAotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBHbHlwaFZlY3RvciBvZiBhc3NvY2lhdGluZyBjaGFyYWN0ZXJzIHRvIGdseXBocyBiYXNlZCBvbiB0aGUKLSAgICAgKiBVbmljb2RlIG1hcCBvZiB0aGlzIEZvbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZyYwotICAgICAqICAgICAgICAgICAgdGhlIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqIEBwYXJhbSBnbHlwaENvZGVzCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGludGVnZXIgYXJyYXkgb2YgZ2x5cGggY29kZXMuCi0gICAgICogQHJldHVybiB0aGUgR2x5cGhWZWN0b3Igb2YgYXNzb2NpYXRpbmcgY2hhcmFjdGVycyB0byBnbHlwaHMgYmFzZWQgb24gdGhlCi0gICAgICogICAgICAgICBVbmljb2RlIG1hcCBvZiB0aGlzIEZvbnQuCi0gICAgICogQHRocm93cyBOb3RJbXBsZW1lbnRlZEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoaXMgbWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCBieSBhIHN1YmNsYXNzLgotICAgICAqLwotICAgIHB1YmxpYyBHbHlwaFZlY3RvciBjcmVhdGVHbHlwaFZlY3RvcihGb250UmVuZGVyQ29udGV4dCBmcmMsIGludFtdIGdseXBoQ29kZXMpCi0gICAgICAgICAgICB0aHJvd3Mgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Ob3RJbXBsZW1lbnRlZEV4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQhIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgR2x5cGhWZWN0b3Igb2YgYXNzb2NpYXRpbmcgY2hhcmFjdGVycyB0byBnbHlwaHMgYmFzZWQgb24gdGhlCi0gICAgICogVW5pY29kZSBtYXAgb2YgdGhpcyBGb250LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb250UmVuZGVyQ29udGV4dC4KLSAgICAgKiBAcGFyYW0gc3RyCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIFN0cmluZy4KLSAgICAgKiBAcmV0dXJuIHRoZSBHbHlwaFZlY3RvciBvZiBhc3NvY2lhdGluZyBjaGFyYWN0ZXJzIHRvIGdseXBocyBiYXNlZCBvbiB0aGUKLSAgICAgKiAgICAgICAgIFVuaWNvZGUgbWFwIG9mIHRoaXMgRm9udC4KLSAgICAgKi8KLSAgICBwdWJsaWMgR2x5cGhWZWN0b3IgY3JlYXRlR2x5cGhWZWN0b3IoRm9udFJlbmRlckNvbnRleHQgZnJjLCBTdHJpbmcgc3RyKSB7Ci0gICAgICAgIHJldHVybiBuZXcgQW5kcm9pZEdseXBoVmVjdG9yKHN0ci50b0NoYXJBcnJheSgpLCBmcmMsIHRoaXMsIDApOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgZm9udCBzdHlsZSBjb25zdGFudCB2YWx1ZSBjb3JyZXNwb25kaW5nIHRvIG9uZSBvZiB0aGUgZm9udAotICAgICAqIHN0eWxlIG5hbWVzICgiQk9MRCIsICJJVEFMSUMiLCAiQk9MRElUQUxJQyIpLiBUaGlzIG1ldGhvZCByZXR1cm5zCi0gICAgICogRm9udC5QTEFJTiBpZiB0aGUgYXJndW1lbnQgaXMgbm90IG9uZSBvZiB0aGUgcHJlZGVmaW5lZCBzdHlsZSBuYW1lcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZm9udFN0eWxlTmFtZQotICAgICAqICAgICAgICAgICAgZm9udCBzdHlsZSBuYW1lLgotICAgICAqIEByZXR1cm4gZm9udCBzdHlsZSBjb25zdGFudCB2YWx1ZSBjb3JyZXNwb25kaW5nIHRvIHRoZSBmb250IHN0eWxlIG5hbWUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBpbnQgZ2V0Rm9udFN0eWxlKFN0cmluZyBmb250U3R5bGVOYW1lKSB7Ci0gICAgICAgIGludCByZXN1bHQgPSBGb250LlBMQUlOOwotCi0gICAgICAgIGlmIChmb250U3R5bGVOYW1lLnRvVXBwZXJDYXNlKCkuZXF1YWxzKCJCT0xESVRBTElDIikpIHsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgcmVzdWx0ID0gRm9udC5CT0xEIHwgRm9udC5JVEFMSUM7Ci0gICAgICAgIH0gZWxzZSBpZiAoZm9udFN0eWxlTmFtZS50b1VwcGVyQ2FzZSgpLmVxdWFscygiQk9MRCIpKSB7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIHJlc3VsdCA9IEZvbnQuQk9MRDsKLSAgICAgICAgfSBlbHNlIGlmIChmb250U3R5bGVOYW1lLnRvVXBwZXJDYXNlKCkuZXF1YWxzKCJJVEFMSUMiKSkgeyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICByZXN1bHQgPSBGb250LklUQUxJQzsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiByZXN1bHQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGVjb2RlcyB0aGUgc3BlY2lmaWVkIHN0cmluZyB3aGljaCBkZXNjcmliZWQgdGhlIEZvbnQuIFRoZSBzdHJpbmcgc2hvdWxkCi0gICAgICogaGF2ZSB0aGUgZm9sbG93aW5nIGZvcm1hdDogZm9udG5hbWUtc3R5bGUtcG9pbnRzaXplLiBUaGUgc3R5bGUgY2FuIGJlCi0gICAgICogUExBSU4sIEJPTEQsIEJPTERJVEFMSUMsIG9yIElUQUxJQy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3RyCi0gICAgICogICAgICAgICAgICB0aGUgc3RyaW5nIHdoaWNoIGRlc2NyaWJlcyB0aGUgZm9udC4KLSAgICAgKiBAcmV0dXJuIHRoZSBGb250IGZyb20gdGhlIHNwZWNpZmllZCBzdHJpbmcuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBGb250IGRlY29kZShTdHJpbmcgc3RyKSB7Ci0gICAgICAgIC8vIFhYWDogRG9jdW1lbnRhdGlvbiBkb2Vzbid0IGRlc2NyaWJlIGFsbCBjYXNlcywgZS5nLiBmb250cyBmYWNlIG5hbWVzCi0gICAgICAgIC8vIHdpdGgKLSAgICAgICAgLy8gc3ltYm9scyB0aGF0IGFyZSBzdWdnZXN0ZWQgYXMgZGVsaW1pdGVycyBpbiB0aGUgZG9jdW1lbnRhdGlvbi4KLSAgICAgICAgLy8gSW4gdGhpcyBkZWNvZGUgaW1wbGVtZW50YXRpb24gb25seSAqKiotKioqLSoqKiBmb3JtYXQgaXMgdXNlZCB3aXRoCi0gICAgICAgIC8vICctJwotICAgICAgICAvLyBhcyB0aGUgZGVsaW1pdGVyIHRvIGF2b2lkIHVuZXhwZWN0ZWQgcGFyc2UgcmVzdWx0cyBvZiBmb250IGZhY2UgbmFtZXMKLSAgICAgICAgLy8gd2l0aCBzcGFjZXMuCi0KLSAgICAgICAgaWYgKHN0ciA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gREVGQVVMVF9GT05UOwotICAgICAgICB9Ci0KLSAgICAgICAgU3RyaW5nVG9rZW5pemVyIHN0clRva2VuczsKLSAgICAgICAgU3RyaW5nIGRlbGltID0gIi0iOyAvLyROT04tTkxTLTEkCi0gICAgICAgIFN0cmluZyBzdWJzdHI7Ci0KLSAgICAgICAgaW50IGZvbnRTaXplID0gREVGQVVMVF9GT05ULnNpemU7Ci0gICAgICAgIGludCBmb250U3R5bGUgPSBERUZBVUxUX0ZPTlQuc3R5bGU7Ci0gICAgICAgIFN0cmluZyBmb250TmFtZSA9IERFRkFVTFRfRk9OVC5uYW1lOwotCi0gICAgICAgIHN0clRva2VucyA9IG5ldyBTdHJpbmdUb2tlbml6ZXIoc3RyLnRyaW0oKSwgZGVsaW0pOwotCi0gICAgICAgIC8vIEZvbnQgTmFtZQotICAgICAgICBpZiAoc3RyVG9rZW5zLmhhc01vcmVUb2tlbnMoKSkgewotICAgICAgICAgICAgZm9udE5hbWUgPSBzdHJUb2tlbnMubmV4dFRva2VuKCk7IC8vIGZpcnN0IHRva2VuIGlzIHRoZSBmb250IG5hbWUKLSAgICAgICAgfQotCi0gICAgICAgIC8vIEZvbnQgU3R5bGUgb3IgU2l6ZSAoaWYgdGhlIHN0eWxlIGlzIHVuZGVmaW5lZCkKLSAgICAgICAgaWYgKHN0clRva2Vucy5oYXNNb3JlVG9rZW5zKCkpIHsKLSAgICAgICAgICAgIHN1YnN0ciA9IHN0clRva2Vucy5uZXh0VG9rZW4oKTsKLQotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICAvLyBpZiBzZWNvbmQgdG9rZW4gaXMgdGhlIGZvbnQgc2l6ZQotICAgICAgICAgICAgICAgIGZvbnRTaXplID0gSW50ZWdlci5wYXJzZUludChzdWJzdHIpOwotICAgICAgICAgICAgfSBjYXRjaCAoTnVtYmVyRm9ybWF0RXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICAvLyB0aGVuIHNlY29uZCB0b2tlbiBpcyB0aGUgZm9udCBzdHlsZQotICAgICAgICAgICAgICAgIGZvbnRTdHlsZSA9IGdldEZvbnRTdHlsZShzdWJzdHIpOwotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLQotICAgICAgICAvLyBGb250IFNpemUKLSAgICAgICAgaWYgKHN0clRva2Vucy5oYXNNb3JlVG9rZW5zKCkpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgZm9udFNpemUgPSBJbnRlZ2VyLnBhcnNlSW50KHN0clRva2Vucy5uZXh0VG9rZW4oKSk7Ci0gICAgICAgICAgICB9IGNhdGNoIChOdW1iZXJGb3JtYXRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBGb250KGZvbnROYW1lLCBmb250U3R5bGUsIGZvbnRTaXplKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQZXJmb3JtcyB0aGUgc3BlY2lmaWVkIGFmZmluZSB0cmFuc2Zvcm0gdG8gdGhlIEZvbnQgYW5kIHJldHVybnMgYSBuZXcKLSAgICAgKiBGb250LgotICAgICAqIAotICAgICAqIEBwYXJhbSB0cmFucwotICAgICAqICAgICAgICAgICAgdGhlIEFmZmluZVRyYW5zZm9ybS4KLSAgICAgKiBAcmV0dXJuIHRoZSBGb250IG9iamVjdC4KLSAgICAgKiBAdGhyb3dzIElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFmZmluZSB0cmFuc2Zvcm0gcGFyYW1ldGVyIGlzIG51bGwuCi0gICAgICovCi0gICAgQFN1cHByZXNzV2FybmluZ3MoInVuY2hlY2tlZCIpCi0gICAgcHVibGljIEZvbnQgZGVyaXZlRm9udChBZmZpbmVUcmFuc2Zvcm0gdHJhbnMpIHsKLQotICAgICAgICBpZiAodHJhbnMgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0Ljk0PXRyYW5zZm9ybSBjYW4gbm90IGJlIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuOTQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIEhhc2h0YWJsZTxBdHRyaWJ1dGUsIE9iamVjdD4gZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMgPSAoSGFzaHRhYmxlPEF0dHJpYnV0ZSwgT2JqZWN0PilmUmVxdWVzdGVkQXR0cmlidXRlcwotICAgICAgICAgICAgICAgIC5jbG9uZSgpOwotCi0gICAgICAgIGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnB1dChUZXh0QXR0cmlidXRlLlRSQU5TRk9STSwgbmV3IFRyYW5zZm9ybUF0dHJpYnV0ZSh0cmFucykpOwotCi0gICAgICAgIHJldHVybiBuZXcgRm9udChkZXJpdmVmUmVxdWVzdGVkQXR0cmlidXRlcyk7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgbmV3IEZvbnQgdGhhdCBpcyBhIGNvcHkgb2YgdGhlIGN1cnJlbnQgRm9udCBtb2RpZmllZCBzbyB0aGF0Ci0gICAgICogdGhlIHNpemUgaXMgdGhlIHNwZWNpZmllZCBzaXplLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgc2l6ZSBvZiBmb250LgotICAgICAqIEByZXR1cm4gdGhlIEZvbnQgb2JqZWN0LgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bmNoZWNrZWQiKQotICAgIHB1YmxpYyBGb250IGRlcml2ZUZvbnQoZmxvYXQgc2l6ZSkgewotICAgICAgICBIYXNodGFibGU8QXR0cmlidXRlLCBPYmplY3Q+IGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzID0gKEhhc2h0YWJsZTxBdHRyaWJ1dGUsIE9iamVjdD4pZlJlcXVlc3RlZEF0dHJpYnV0ZXMKLSAgICAgICAgICAgICAgICAuY2xvbmUoKTsKLSAgICAgICAgZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMucHV0KFRleHRBdHRyaWJ1dGUuU0laRSwgbmV3IEZsb2F0KHNpemUpKTsKLSAgICAgICAgcmV0dXJuIG5ldyBGb250KGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgbmV3IEZvbnQgdGhhdCBpcyBhIGNvcHkgb2YgdGhlIGN1cnJlbnQgRm9udCBtb2RpZmllZCBzbyB0aGF0Ci0gICAgICogdGhlIHN0eWxlIGlzIHRoZSBzcGVjaWZpZWQgc3R5bGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0eWxlCi0gICAgICogICAgICAgICAgICB0aGUgc3R5bGUgb2YgZm9udC4KLSAgICAgKiBAcmV0dXJuIHRoZSBGb250IG9iamVjdC4KLSAgICAgKi8KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygidW5jaGVja2VkIikKLSAgICBwdWJsaWMgRm9udCBkZXJpdmVGb250KGludCBzdHlsZSkgewotICAgICAgICBIYXNodGFibGU8QXR0cmlidXRlLCBPYmplY3Q+IGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzID0gKEhhc2h0YWJsZTxBdHRyaWJ1dGUsIE9iamVjdD4pZlJlcXVlc3RlZEF0dHJpYnV0ZXMKLSAgICAgICAgICAgICAgICAuY2xvbmUoKTsKLQotICAgICAgICBpZiAoKHN0eWxlICYgRm9udC5CT0xEKSAhPSAwKSB7Ci0gICAgICAgICAgICBkZXJpdmVmUmVxdWVzdGVkQXR0cmlidXRlcy5wdXQoVGV4dEF0dHJpYnV0ZS5XRUlHSFQsIFRleHRBdHRyaWJ1dGUuV0VJR0hUX0JPTEQpOwotICAgICAgICB9IGVsc2UgaWYgKGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLldFSUdIVCkgIT0gbnVsbCkgewotICAgICAgICAgICAgZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMucmVtb3ZlKFRleHRBdHRyaWJ1dGUuV0VJR0hUKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICgoc3R5bGUgJiBGb250LklUQUxJQykgIT0gMCkgewotICAgICAgICAgICAgZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMucHV0KFRleHRBdHRyaWJ1dGUuUE9TVFVSRSwgVGV4dEF0dHJpYnV0ZS5QT1NUVVJFX09CTElRVUUpOwotICAgICAgICB9IGVsc2UgaWYgKGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLlBPU1RVUkUpICE9IG51bGwpIHsKLSAgICAgICAgICAgIGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnJlbW92ZShUZXh0QXR0cmlidXRlLlBPU1RVUkUpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBGb250KGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgbmV3IEZvbnQgdGhhdCBpcyBhIGNvcHkgb2YgdGhlIGN1cnJlbnQgRm9udCBtb2RpZmllZCB0byBtYXRjaAotICAgICAqIHRoZSBzcGVjaWZpZWQgc3R5bGUgYW5kIHdpdGggdGhlIHNwZWNpZmllZCBhZmZpbmUgdHJhbnNmb3JtIGFwcGxpZWQgdG8KLSAgICAgKiBpdHMgZ2x5cGhzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHlsZQotICAgICAqICAgICAgICAgICAgdGhlIHN0eWxlIG9mIGZvbnQuCi0gICAgICogQHBhcmFtIHRyYW5zCi0gICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtLgotICAgICAqIEByZXR1cm4gdGhlIEZvbnQgb2JqZWN0LgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bmNoZWNrZWQiKQotICAgIHB1YmxpYyBGb250IGRlcml2ZUZvbnQoaW50IHN0eWxlLCBBZmZpbmVUcmFuc2Zvcm0gdHJhbnMpIHsKLQotICAgICAgICBpZiAodHJhbnMgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0Ljk0PXRyYW5zZm9ybSBjYW4gbm90IGJlIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuOTQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBIYXNodGFibGU8QXR0cmlidXRlLCBPYmplY3Q+IGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzID0gKEhhc2h0YWJsZTxBdHRyaWJ1dGUsIE9iamVjdD4pZlJlcXVlc3RlZEF0dHJpYnV0ZXMKLSAgICAgICAgICAgICAgICAuY2xvbmUoKTsKLQotICAgICAgICBpZiAoKHN0eWxlICYgQk9MRCkgIT0gMCkgewotICAgICAgICAgICAgZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMucHV0KFRleHRBdHRyaWJ1dGUuV0VJR0hULCBUZXh0QXR0cmlidXRlLldFSUdIVF9CT0xEKTsKLSAgICAgICAgfSBlbHNlIGlmIChkZXJpdmVmUmVxdWVzdGVkQXR0cmlidXRlcy5nZXQoVGV4dEF0dHJpYnV0ZS5XRUlHSFQpICE9IG51bGwpIHsKLSAgICAgICAgICAgIGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnJlbW92ZShUZXh0QXR0cmlidXRlLldFSUdIVCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoKHN0eWxlICYgSVRBTElDKSAhPSAwKSB7Ci0gICAgICAgICAgICBkZXJpdmVmUmVxdWVzdGVkQXR0cmlidXRlcy5wdXQoVGV4dEF0dHJpYnV0ZS5QT1NUVVJFLCBUZXh0QXR0cmlidXRlLlBPU1RVUkVfT0JMSVFVRSk7Ci0gICAgICAgIH0gZWxzZSBpZiAoZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMuZ2V0KFRleHRBdHRyaWJ1dGUuUE9TVFVSRSkgIT0gbnVsbCkgewotICAgICAgICAgICAgZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMucmVtb3ZlKFRleHRBdHRyaWJ1dGUuUE9TVFVSRSk7Ci0gICAgICAgIH0KLSAgICAgICAgZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMucHV0KFRleHRBdHRyaWJ1dGUuVFJBTlNGT1JNLCBuZXcgVHJhbnNmb3JtQXR0cmlidXRlKHRyYW5zKSk7Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBGb250KGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgbmV3IEZvbnQgdGhhdCBpcyBhIGNvcHkgb2YgdGhlIGN1cnJlbnQgRm9udCBtb2RpZmllZCBzbyB0aGF0Ci0gICAgICogdGhlIHNpemUgYW5kIHN0eWxlIGFyZSB0aGUgc3BlY2lmaWVkIHNpemUgYW5kIHN0eWxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHlsZQotICAgICAqICAgICAgICAgICAgdGhlIHN0eWxlIG9mIGZvbnQuCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzaXplIG9mIGZvbnQuCi0gICAgICogQHJldHVybiB0aGUgRm9udCBvYmplY3QuCi0gICAgICovCi0gICAgQFN1cHByZXNzV2FybmluZ3MoInVuY2hlY2tlZCIpCi0gICAgcHVibGljIEZvbnQgZGVyaXZlRm9udChpbnQgc3R5bGUsIGZsb2F0IHNpemUpIHsKLSAgICAgICAgSGFzaHRhYmxlPEF0dHJpYnV0ZSwgT2JqZWN0PiBkZXJpdmVmUmVxdWVzdGVkQXR0cmlidXRlcyA9IChIYXNodGFibGU8QXR0cmlidXRlLCBPYmplY3Q+KWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzCi0gICAgICAgICAgICAgICAgLmNsb25lKCk7Ci0KLSAgICAgICAgaWYgKChzdHlsZSAmIEJPTEQpICE9IDApIHsKLSAgICAgICAgICAgIGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnB1dChUZXh0QXR0cmlidXRlLldFSUdIVCwgVGV4dEF0dHJpYnV0ZS5XRUlHSFRfQk9MRCk7Ci0gICAgICAgIH0gZWxzZSBpZiAoZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMuZ2V0KFRleHRBdHRyaWJ1dGUuV0VJR0hUKSAhPSBudWxsKSB7Ci0gICAgICAgICAgICBkZXJpdmVmUmVxdWVzdGVkQXR0cmlidXRlcy5yZW1vdmUoVGV4dEF0dHJpYnV0ZS5XRUlHSFQpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChzdHlsZSAmIElUQUxJQykgIT0gMCkgewotICAgICAgICAgICAgZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMucHV0KFRleHRBdHRyaWJ1dGUuUE9TVFVSRSwgVGV4dEF0dHJpYnV0ZS5QT1NUVVJFX09CTElRVUUpOwotICAgICAgICB9IGVsc2UgaWYgKGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLlBPU1RVUkUpICE9IG51bGwpIHsKLSAgICAgICAgICAgIGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLnJlbW92ZShUZXh0QXR0cmlidXRlLlBPU1RVUkUpOwotICAgICAgICB9Ci0KLSAgICAgICAgZGVyaXZlZlJlcXVlc3RlZEF0dHJpYnV0ZXMucHV0KFRleHRBdHRyaWJ1dGUuU0laRSwgbmV3IEZsb2F0KHNpemUpKTsKLSAgICAgICAgcmV0dXJuIG5ldyBGb250KGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzKTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBuZXcgRm9udCBvYmplY3Qgd2l0aCBhIG5ldyBzZXQgb2YgZm9udCBhdHRyaWJ1dGVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhdHRyaWJ1dGVzCi0gICAgICogICAgICAgICAgICB0aGUgbWFwIG9mIGF0dHJpYnV0ZXMuCi0gICAgICogQHJldHVybiB0aGUgRm9udC4KLSAgICAgKi8KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygidW5jaGVja2VkIikKLSAgICBwdWJsaWMgRm9udCBkZXJpdmVGb250KE1hcDw/IGV4dGVuZHMgQXR0cmlidXRlLCA/PiBhdHRyaWJ1dGVzKSB7Ci0gICAgICAgIEF0dHJpYnV0ZVtdIGF2YWxBdHRyaWJ1dGVzID0gdGhpcy5nZXRBdmFpbGFibGVBdHRyaWJ1dGVzKCk7Ci0KLSAgICAgICAgSGFzaHRhYmxlPEF0dHJpYnV0ZSwgT2JqZWN0PiBkZXJpdmVmUmVxdWVzdGVkQXR0cmlidXRlcyA9IChIYXNodGFibGU8QXR0cmlidXRlLCBPYmplY3Q+KWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzCi0gICAgICAgICAgICAgICAgLmNsb25lKCk7Ci0gICAgICAgIE9iamVjdCBjdXJyQXR0cmlidXRlOwotICAgICAgICBmb3IgKEF0dHJpYnV0ZSBlbGVtZW50IDogYXZhbEF0dHJpYnV0ZXMpIHsKLSAgICAgICAgICAgIGN1cnJBdHRyaWJ1dGUgPSBhdHRyaWJ1dGVzLmdldChlbGVtZW50KTsKLSAgICAgICAgICAgIGlmIChjdXJyQXR0cmlidXRlICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBkZXJpdmVmUmVxdWVzdGVkQXR0cmlidXRlcy5wdXQoZWxlbWVudCwgY3VyckF0dHJpYnV0ZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBGb250KGRlcml2ZWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wYXJlcyB0aGUgc3BlY2lmaWVkIE9iamVjdCB3aXRoIHRoZSBjdXJyZW50IEZvbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9iagotICAgICAqICAgICAgICAgICAgdGhlIE9iamVjdCB0byBiZSBjb21wYXJlZC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgT2JqZWN0IGlzIGFuIGluc3RhbmNlIG9mIEZvbnQgd2l0aCB0aGUKLSAgICAgKiAgICAgICAgIHNhbWUgZmFtaWx5LCBzaXplLCBhbmQgc3R5bGUgYXMgdGhpcyBGb250LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvYmopIHsKLSAgICAgICAgaWYgKG9iaiA9PSB0aGlzKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChvYmogIT0gbnVsbCkgewotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICBGb250IGZvbnQgPSAoRm9udClvYmo7Ci0KLSAgICAgICAgICAgICAgICByZXR1cm4gKCh0aGlzLnN0eWxlID09IGZvbnQuc3R5bGUpICYmICh0aGlzLnNpemUgPT0gZm9udC5zaXplKQotICAgICAgICAgICAgICAgICAgICAgICAgJiYgdGhpcy5uYW1lLmVxdWFscyhmb250Lm5hbWUpICYmICh0aGlzLnBvaW50U2l6ZSA9PSBmb250LnBvaW50U2l6ZSkgJiYgKHRoaXMKLSAgICAgICAgICAgICAgICAgICAgICAgIC5nZXRUcmFuc2Zvcm0oKSkuZXF1YWxzKGZvbnQuZ2V0VHJhbnNmb3JtKCkpKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKENsYXNzQ2FzdEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWFwIG9mIGZvbnQncyBhdHRyaWJ1dGVzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG1hcCBvZiBmb250J3MgYXR0cmlidXRlcy4KLSAgICAgKi8KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygidW5jaGVja2VkIikKLSAgICBwdWJsaWMgTWFwPFRleHRBdHRyaWJ1dGUsID8+IGdldEF0dHJpYnV0ZXMoKSB7Ci0gICAgICAgIHJldHVybiAoTWFwPFRleHRBdHRyaWJ1dGUsID8+KWZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUga2V5cyBvZiBhbGwgYXZhaWxhYmxlIGF0dHJpYnV0ZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUga2V5cyBhcnJheSBvZiBhbGwgYXZhaWxhYmxlIGF0dHJpYnV0ZXMuCi0gICAgICovCi0gICAgcHVibGljIEF0dHJpYnV0ZVtdIGdldEF2YWlsYWJsZUF0dHJpYnV0ZXMoKSB7Ci0gICAgICAgIEF0dHJpYnV0ZVtdIGF0dHJzID0gewotICAgICAgICAgICAgICAgIFRleHRBdHRyaWJ1dGUuRkFNSUxZLCBUZXh0QXR0cmlidXRlLlBPU1RVUkUsIFRleHRBdHRyaWJ1dGUuU0laRSwKLSAgICAgICAgICAgICAgICBUZXh0QXR0cmlidXRlLlRSQU5TRk9STSwgVGV4dEF0dHJpYnV0ZS5XRUlHSFQsIFRleHRBdHRyaWJ1dGUuU1VQRVJTQ1JJUFQsCi0gICAgICAgICAgICAgICAgVGV4dEF0dHJpYnV0ZS5XSURUSAotICAgICAgICB9OwotICAgICAgICByZXR1cm4gYXR0cnM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYmFzZWxpbmUgZm9yIHRoaXMgY2hhcmFjdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgY2hhcmFjdGVyLgotICAgICAqIEByZXR1cm4gdGhlIGJhc2VsaW5lIGZvciB0aGlzIGNoYXJhY3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYnl0ZSBnZXRCYXNlbGluZUZvcihjaGFyIGMpIHsKLSAgICAgICAgLy8gVE9ETzogaW1wbGVtZW50IHVzaW5nIFRUIEJBU0UgdGFibGUgZGF0YQotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmYW1pbHkgbmFtZSBvZiB0aGUgRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBmYW1pbHkgbmFtZSBvZiB0aGUgRm9udC4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldEZhbWlseSgpIHsKLSAgICAgICAgaWYgKGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzICE9IG51bGwpIHsKLSAgICAgICAgICAgIGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLkZBTUlMWSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgZmFtaWx5IG5hbWUgb2YgdGhpcyBGb250IGFzc29jaWF0ZWQgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICogbG9jYWxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgbG9jYWxlLgotICAgICAqIEByZXR1cm4gdGhlIGZhbWlseSBuYW1lIG9mIHRoaXMgRm9udCBhc3NvY2lhdGVkIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgbG9jYWxlLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RmFtaWx5KExvY2FsZSBsKSB7Ci0gICAgICAgIGlmIChsID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4wMT0nezB9JyBwYXJhbWV0ZXIgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjAxIiwgIkxvY2FsZSIpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBnZXRGYW1pbHkoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGEgRm9udCB3aXRoIHRoZSBzcGVjaWZpZWQgYXR0cmlidXRlIHNldC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYXR0cmlidXRlcwotICAgICAqICAgICAgICAgICAgdGhlIGF0dHJpYnV0ZXMgdG8gYmUgYXNzaWduZWQgdG8gdGhlIG5ldyBGb250LgotICAgICAqIEByZXR1cm4gdGhlIEZvbnQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBGb250IGdldEZvbnQoTWFwPD8gZXh0ZW5kcyBBdHRyaWJ1dGUsID8+IGF0dHJpYnV0ZXMpIHsKLSAgICAgICAgRm9udCBmbnQgPSAoRm9udClhdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLkZPTlQpOwotICAgICAgICBpZiAoZm50ICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBmbnQ7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBGb250KGF0dHJpYnV0ZXMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBGb250IG9iamVjdCBmcm9tIHRoZSBzeXN0ZW0gcHJvcGVydGllcyBsaXN0IHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIG5hbWUgb3IgcmV0dXJucyB0aGUgc3BlY2lmaWVkIEZvbnQgaWYgdGhlcmUgaXMgbm8gc3VjaCBwcm9wZXJ0eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3AKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgcHJvcGVydHkgbmFtZS4KLSAgICAgKiBAcGFyYW0gZgotICAgICAqICAgICAgICAgICAgdGhlIEZvbnQuCi0gICAgICogQHJldHVybiB0aGUgRm9udCBvYmplY3QgZnJvbSB0aGUgc3lzdGVtIHByb3BlcnRpZXMgbGlzdCB3aXRoIHRoZQotICAgICAqICAgICAgICAgc3BlY2lmaWVkIG5hbWUgb3IgdGhlIHNwZWNpZmllZCBGb250IGlmIHRoZXJlIGlzIG5vIHN1Y2gKLSAgICAgKiAgICAgICAgIHByb3BlcnR5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgRm9udCBnZXRGb250KFN0cmluZyBzcCwgRm9udCBmKSB7Ci0gICAgICAgIFN0cmluZyBwciA9IFN5c3RlbS5nZXRQcm9wZXJ0eShzcCk7Ci0gICAgICAgIGlmIChwciA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gZjsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZGVjb2RlKHByKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGEgRm9udCBvYmplY3QgZnJvbSB0aGUgc3lzdGVtIHByb3BlcnRpZXMgbGlzdCB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBuYW1lLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcAotICAgICAqICAgICAgICAgICAgdGhlIHN5c3RlbSBwcm9wZXJ0eSBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIEZvbnQsIG9yIG51bGwgaWYgdGhlcmUgaXMgbm8gc3VjaCBwcm9wZXJ0eSB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiAgICAgICAgIG5hbWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBGb250IGdldEZvbnQoU3RyaW5nIHNwKSB7Ci0gICAgICAgIHJldHVybiBnZXRGb250KHNwLCBudWxsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmb250IG5hbWUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZm9udCBuYW1lLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0Rm9udE5hbWUoKSB7Ci0gICAgICAgIGlmIChmUmVxdWVzdGVkQXR0cmlidXRlcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICBmUmVxdWVzdGVkQXR0cmlidXRlcy5nZXQoVGV4dEF0dHJpYnV0ZS5GQU1JTFkpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGZvbnQgbmFtZSBhc3NvY2lhdGVkIHdpdGggdGhlIHNwZWNpZmllZCBsb2NhbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBsb2NhbGUuCi0gICAgICogQHJldHVybiB0aGUgZm9udCBuYW1lIGFzc29jaWF0ZWQgd2l0aCB0aGUgc3BlY2lmaWVkIGxvY2FsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldEZvbnROYW1lKExvY2FsZSBsKSB7Ci0gICAgICAgIHJldHVybiBnZXRGYW1pbHkoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgTGluZU1ldHJpY3Mgb2JqZWN0IGNyZWF0ZWQgd2l0aCB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNoYXJzCi0gICAgICogICAgICAgICAgICB0aGUgY2hhcnMgYXJyYXkuCi0gICAgICogQHBhcmFtIHN0YXJ0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBlbmQKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbmQgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBmcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb250UmVuZGVyQ29udGV4dC4KLSAgICAgKiBAcmV0dXJuIHRoZSBMaW5lTWV0cmljcyBmb3IgdGhlIHNwZWNpZmllZCBwYXJhbWV0ZXJzLgotICAgICAqLwotICAgIHB1YmxpYyBMaW5lTWV0cmljcyBnZXRMaW5lTWV0cmljcyhjaGFyW10gY2hhcnMsIGludCBzdGFydCwgaW50IGVuZCwgRm9udFJlbmRlckNvbnRleHQgZnJjKSB7Ci0gICAgICAgIGlmIChmcmMgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjAwPUZvbnRSZW5kZXJDb250ZXh0IGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4wMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgLy8gRm9udE1ldHJpY3MgZm0gPSBBbmRyb2lkR3JhcGhpY3MyRC5nZXRJbnN0YW5jZSgpLmdldEZvbnRNZXRyaWNzKCk7Ci0gICAgICAgIEZvbnRNZXRyaWNzIGZtID0gbmV3IEZvbnRNZXRyaWNzSW1wbCh0aGlzKTsKLSAgICAgICAgZmxvYXRbXSBmbWV0ID0gewotICAgICAgICAgICAgICAgIGZtLmdldEFzY2VudCgpLCBmbS5nZXREZXNjZW50KCksIGZtLmdldExlYWRpbmcoKQotICAgICAgICB9OwotICAgICAgICByZXR1cm4gbmV3IExpbmVNZXRyaWNzSW1wbChjaGFycy5sZW5ndGgsIGZtZXQsIG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBMaW5lTWV0cmljcyBvYmplY3QgY3JlYXRlZCB3aXRoIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaXRlcgotICAgICAqICAgICAgICAgICAgdGhlIENoYXJhY3Rlckl0ZXJhdG9yLgotICAgICAqIEBwYXJhbSBzdGFydAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IG9mZnNldC4KLSAgICAgKiBAcGFyYW0gZW5kCi0gICAgICogICAgICAgICAgICB0aGUgZW5kIG9mZnNldC4KLSAgICAgKiBAcGFyYW0gZnJjCi0gICAgICogICAgICAgICAgICB0aGUgRm9udFJlbmRlckNvbnRleHQuCi0gICAgICogQHJldHVybiB0aGUgTGluZU1ldHJpY3MgZm9yIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgTGluZU1ldHJpY3MgZ2V0TGluZU1ldHJpY3MoQ2hhcmFjdGVySXRlcmF0b3IgaXRlciwgaW50IHN0YXJ0LCBpbnQgZW5kLAotICAgICAgICAgICAgRm9udFJlbmRlckNvbnRleHQgZnJjKSB7Ci0KLSAgICAgICAgaWYgKGZyYyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMDA9Rm9udFJlbmRlckNvbnRleHQgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjAwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBTdHJpbmcgcmVzdWx0U3RyaW5nOwotICAgICAgICBpbnQgaXRlckNvdW50OwotCi0gICAgICAgIGl0ZXJDb3VudCA9IGVuZCAtIHN0YXJ0OwotICAgICAgICBpZiAoaXRlckNvdW50IDwgMCkgewotICAgICAgICAgICAgcmVzdWx0U3RyaW5nID0gIiI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNoYXJbXSBjaGFycyA9IG5ldyBjaGFyW2l0ZXJDb3VudF07Ci0gICAgICAgICAgICBpbnQgaSA9IDA7Ci0gICAgICAgICAgICBmb3IgKGNoYXIgYyA9IGl0ZXIuc2V0SW5kZXgoc3RhcnQpOyBjICE9IENoYXJhY3Rlckl0ZXJhdG9yLkRPTkUgJiYgKGkgPCBpdGVyQ291bnQpOyBjID0gaXRlcgotICAgICAgICAgICAgICAgICAgICAubmV4dCgpKSB7Ci0gICAgICAgICAgICAgICAgY2hhcnNbaV0gPSBjOwotICAgICAgICAgICAgICAgIGkrKzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJlc3VsdFN0cmluZyA9IG5ldyBTdHJpbmcoY2hhcnMpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB0aGlzLmdldExpbmVNZXRyaWNzKHJlc3VsdFN0cmluZywgZnJjKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgTGluZU1ldHJpY3Mgb2JqZWN0IGNyZWF0ZWQgd2l0aCB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cgotICAgICAqICAgICAgICAgICAgdGhlIFN0cmluZy4KLSAgICAgKiBAcGFyYW0gZnJjCi0gICAgICogICAgICAgICAgICB0aGUgRm9udFJlbmRlckNvbnRleHQuCi0gICAgICogQHJldHVybiB0aGUgTGluZU1ldHJpY3MgZm9yIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgTGluZU1ldHJpY3MgZ2V0TGluZU1ldHJpY3MoU3RyaW5nIHN0ciwgRm9udFJlbmRlckNvbnRleHQgZnJjKSB7Ci0gICAgICAgIC8vIEZvbnRNZXRyaWNzIGZtID0gQW5kcm9pZEdyYXBoaWNzMkQuZ2V0SW5zdGFuY2UoKS5nZXRGb250TWV0cmljcygpOwotICAgICAgICBGb250TWV0cmljcyBmbSA9IG5ldyBGb250TWV0cmljc0ltcGwodGhpcyk7Ci0gICAgICAgIGZsb2F0W10gZm1ldCA9IHsKLSAgICAgICAgICAgICAgICBmbS5nZXRBc2NlbnQoKSwgZm0uZ2V0RGVzY2VudCgpLCBmbS5nZXRMZWFkaW5nKCkKLSAgICAgICAgfTsKLSAgICAgICAgLy8gTG9nLmkoIkZPTlQgRk1FVCIsIGZtZXQudG9TdHJpbmcoKSk7Ci0gICAgICAgIHJldHVybiBuZXcgTGluZU1ldHJpY3NJbXBsKHN0ci5sZW5ndGgoKSwgZm1ldCwgbnVsbCk7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgTGluZU1ldHJpY3Mgb2JqZWN0IGNyZWF0ZWQgd2l0aCB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cgotICAgICAqICAgICAgICAgICAgdGhlIFN0cmluZy4KLSAgICAgKiBAcGFyYW0gc3RhcnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdGFydCBvZmZzZXQuCi0gICAgICogQHBhcmFtIGVuZAotICAgICAqICAgICAgICAgICAgdGhlIGVuZCBvZmZzZXQuCi0gICAgICogQHBhcmFtIGZyYwotICAgICAqICAgICAgICAgICAgdGhlIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqIEByZXR1cm4gdGhlIExpbmVNZXRyaWNzIGZvciB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMuCi0gICAgICovCi0gICAgcHVibGljIExpbmVNZXRyaWNzIGdldExpbmVNZXRyaWNzKFN0cmluZyBzdHIsIGludCBzdGFydCwgaW50IGVuZCwgRm9udFJlbmRlckNvbnRleHQgZnJjKSB7Ci0gICAgICAgIHJldHVybiB0aGlzLmdldExpbmVNZXRyaWNzKHN0ci5zdWJzdHJpbmcoc3RhcnQsIGVuZCksIGZyYyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9naWNhbCBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBTdHJpbmcgaW4gdGhlIHNwZWNpZmllZAotICAgICAqIEZvbnRSZW5kZXJDb250ZXh0LiBUaGUgbG9naWNhbCBib3VuZHMgY29udGFpbnMgdGhlIG9yaWdpbiwgYXNjZW50LAotICAgICAqIGFkdmFuY2UsIGFuZCBoZWlnaHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNpCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIENoYXJhY3Rlckl0ZXJhdG9yLgotICAgICAqIEBwYXJhbSBzdGFydAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IG9mZnNldC4KLSAgICAgKiBAcGFyYW0gZW5kCi0gICAgICogICAgICAgICAgICB0aGUgZW5kIG9mZnNldC4KLSAgICAgKiBAcGFyYW0gZnJjCi0gICAgICogICAgICAgICAgICB0aGUgRm9udFJlbmRlckNvbnRleHQuCi0gICAgICogQHJldHVybiBhIFJlY3RhbmdsZTJEIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0U3RyaW5nQm91bmRzKENoYXJhY3Rlckl0ZXJhdG9yIGNpLCBpbnQgc3RhcnQsIGludCBlbmQsCi0gICAgICAgICAgICBGb250UmVuZGVyQ29udGV4dCBmcmMpIHsKLSAgICAgICAgaW50IGZpcnN0ID0gY2kuZ2V0QmVnaW5JbmRleCgpOwotICAgICAgICBpbnQgZmluaXNoID0gY2kuZ2V0RW5kSW5kZXgoKTsKLSAgICAgICAgY2hhcltdIGNoYXJzOwotCi0gICAgICAgIGlmIChzdGFydCA8IGZpcnN0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuOTU9V3Jvbmcgc3RhcnQgaW5kZXg6IHswfQotICAgICAgICAgICAgdGhyb3cgbmV3IEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuOTUiLCBzdGFydCkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKGVuZCA+IGZpbmlzaCkgewotICAgICAgICAgICAgLy8gYXd0Ljk2PVdyb25nIGZpbmlzaCBpbmRleDogezB9Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC45NiIsIGVuZCkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKHN0YXJ0ID4gZW5kKSB7Ci0gICAgICAgICAgICAvLyBhd3QuOTc9V3JvbmcgcmFuZ2UgbGVuZ3RoOiB7MH0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0Ljk3IiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICAoZW5kIC0gc3RhcnQpKSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZnJjID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4wMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgY2hhcnMgPSBuZXcgY2hhcltlbmQgLSBzdGFydF07Ci0KLSAgICAgICAgY2kuc2V0SW5kZXgoc3RhcnQpOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGNoYXJzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBjaGFyc1tpXSA9IGNpLmN1cnJlbnQoKTsKLSAgICAgICAgICAgIGNpLm5leHQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiB0aGlzLmdldFN0cmluZ0JvdW5kcyhjaGFycywgMCwgY2hhcnMubGVuZ3RoLCBmcmMpOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9naWNhbCBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBTdHJpbmcgaW4gdGhlIHNwZWNpZmllZAotICAgICAqIEZvbnRSZW5kZXJDb250ZXh0LiBUaGUgbG9naWNhbCBib3VuZHMgY29udGFpbnMgdGhlIG9yaWdpbiwgYXNjZW50LAotICAgICAqIGFkdmFuY2UsIGFuZCBoZWlnaHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBTdHJpbmcuCi0gICAgICogQHBhcmFtIGZyYwotICAgICAqICAgICAgICAgICAgdGhlIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqIEByZXR1cm4gYSBSZWN0YW5nbGUyRCBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGdldFN0cmluZ0JvdW5kcyhTdHJpbmcgc3RyLCBGb250UmVuZGVyQ29udGV4dCBmcmMpIHsKLSAgICAgICAgY2hhcltdIGNoYXJzID0gc3RyLnRvQ2hhckFycmF5KCk7Ci0gICAgICAgIHJldHVybiB0aGlzLmdldFN0cmluZ0JvdW5kcyhjaGFycywgMCwgY2hhcnMubGVuZ3RoLCBmcmMpOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9naWNhbCBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBTdHJpbmcgaW4gdGhlIHNwZWNpZmllZAotICAgICAqIEZvbnRSZW5kZXJDb250ZXh0LiBUaGUgbG9naWNhbCBib3VuZHMgY29udGFpbnMgdGhlIG9yaWdpbiwgYXNjZW50LAotICAgICAqIGFkdmFuY2UsIGFuZCBoZWlnaHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBTdHJpbmcuCi0gICAgICogQHBhcmFtIHN0YXJ0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBlbmQKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbmQgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBmcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb250UmVuZGVyQ29udGV4dC4KLSAgICAgKiBAcmV0dXJuIGEgUmVjdGFuZ2xlMkQgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRTdHJpbmdCb3VuZHMoU3RyaW5nIHN0ciwgaW50IHN0YXJ0LCBpbnQgZW5kLCBGb250UmVuZGVyQ29udGV4dCBmcmMpIHsKLQotICAgICAgICByZXR1cm4gdGhpcy5nZXRTdHJpbmdCb3VuZHMoKHN0ci5zdWJzdHJpbmcoc3RhcnQsIGVuZCkpLCBmcmMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvZ2ljYWwgYm91bmRzIG9mIHRoZSBzcGVjaWZpZWQgU3RyaW5nIGluIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBGb250UmVuZGVyQ29udGV4dC4gVGhlIGxvZ2ljYWwgYm91bmRzIGNvbnRhaW5zIHRoZSBvcmlnaW4sIGFzY2VudCwKLSAgICAgKiBhZHZhbmNlLCBhbmQgaGVpZ2h0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaGFycwotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBjaGFyYWN0ZXIgYXJyYXkuCi0gICAgICogQHBhcmFtIHN0YXJ0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBlbmQKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbmQgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBmcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb250UmVuZGVyQ29udGV4dC4KLSAgICAgKiBAcmV0dXJuIGEgUmVjdGFuZ2xlMkQgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRTdHJpbmdCb3VuZHMoY2hhcltdIGNoYXJzLCBpbnQgc3RhcnQsIGludCBlbmQsIEZvbnRSZW5kZXJDb250ZXh0IGZyYykgewotICAgICAgICBpZiAoc3RhcnQgPCAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuOTU9V3Jvbmcgc3RhcnQgaW5kZXg6IHswfQotICAgICAgICAgICAgdGhyb3cgbmV3IEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuOTUiLCBzdGFydCkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKGVuZCA+IGNoYXJzLmxlbmd0aCkgewotICAgICAgICAgICAgLy8gYXd0Ljk2PVdyb25nIGZpbmlzaCBpbmRleDogezB9Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC45NiIsIGVuZCkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKHN0YXJ0ID4gZW5kKSB7Ci0gICAgICAgICAgICAvLyBhd3QuOTc9V3JvbmcgcmFuZ2UgbGVuZ3RoOiB7MH0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0Ljk3IiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICAoZW5kIC0gc3RhcnQpKSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZnJjID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4wMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgRm9udFBlZXJJbXBsIHBlZXIgPSAoRm9udFBlZXJJbXBsKXRoaXMuZ2V0UGVlcigpOwotCi0gICAgICAgIGZpbmFsIGludCBUUkFOU0ZPUk1fTUFTSyA9IEFmZmluZVRyYW5zZm9ybS5UWVBFX0dFTkVSQUxfUk9UQVRJT04KLSAgICAgICAgICAgICAgICB8IEFmZmluZVRyYW5zZm9ybS5UWVBFX0dFTkVSQUxfVFJBTlNGT1JNOwotICAgICAgICBSZWN0YW5nbGUyRCBib3VuZHM7Ci0KLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHRyYW5zZm9ybSA9IGdldFRyYW5zZm9ybSgpOwotCi0gICAgICAgIC8vIFhYWDogZm9yIHRyYW5zZm9ybXMgd2hlcmUgYW4gYW5nbGUgYmV0d2VlbiBiYXNpcyB2ZWN0b3JzIGlzIG5vdCA5MAotICAgICAgICAvLyBkZWdyZWVzIFJlY3RhbmxnZTJEIGNsYXNzIGRvZXNuJ3QgZml0IGFzIExvZ2ljYWwgYm91bmRzLgotICAgICAgICBpZiAoKHRyYW5zZm9ybS5nZXRUeXBlKCkgJiBUUkFOU0ZPUk1fTUFTSykgPT0gMCkgewotICAgICAgICAgICAgaW50IHdpZHRoID0gMDsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSBzdGFydDsgaSA8IGVuZDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgd2lkdGggKz0gcGVlci5jaGFyV2lkdGgoY2hhcnNbaV0pOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgLy8gTGluZU1ldHJpY3MgbmxtID0gcGVlci5nZXRMaW5lTWV0cmljcygpOwotCi0gICAgICAgICAgICBMaW5lTWV0cmljcyBubG0gPSBnZXRMaW5lTWV0cmljcyhjaGFycywgc3RhcnQsIGVuZCwgZnJjKTsKLQotICAgICAgICAgICAgYm91bmRzID0gdHJhbnNmb3JtLmNyZWF0ZVRyYW5zZm9ybWVkU2hhcGUoCi0gICAgICAgICAgICAgICAgICAgIG5ldyBSZWN0YW5nbGUyRC5GbG9hdCgwLCAtbmxtLmdldEFzY2VudCgpLCB3aWR0aCwgbmxtLmdldEhlaWdodCgpKSkKLSAgICAgICAgICAgICAgICAgICAgLmdldEJvdW5kczJEKCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpbnQgbGVuID0gZW5kIC0gc3RhcnQ7Ci0gICAgICAgICAgICBjaGFyW10gc3ViQ2hhcnMgPSBuZXcgY2hhcltsZW5dOwotICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShjaGFycywgc3RhcnQsIHN1YkNoYXJzLCAwLCBsZW4pOwotICAgICAgICAgICAgYm91bmRzID0gY3JlYXRlR2x5cGhWZWN0b3IoZnJjLCBzdWJDaGFycykuZ2V0TG9naWNhbEJvdW5kcygpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBib3VuZHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY2hhcmFjdGVyJ3MgbWF4aW11bSBib3VuZHMgYXMgZGVmaW5lZCBpbiB0aGUgc3BlY2lmaWVkCi0gICAgICogRm9udFJlbmRlckNvbnRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZyYwotICAgICAqICAgICAgICAgICAgdGhlIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqIEByZXR1cm4gdGhlIGNoYXJhY3RlcidzIG1heGltdW0gYm91bmRzLgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRNYXhDaGFyQm91bmRzKEZvbnRSZW5kZXJDb250ZXh0IGZyYykgewotICAgICAgICBpZiAoZnJjID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4wMD1Gb250UmVuZGVyQ29udGV4dCBpcyBudWxsCi0gICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMDAiKSk7IC8vJE5PTi1OTFMtMSQgCi0gICAgICAgIH0KLQotICAgICAgICBGb250UGVlckltcGwgcGVlciA9IChGb250UGVlckltcGwpdGhpcy5nZXRQZWVyKCk7Ci0KLSAgICAgICAgUmVjdGFuZ2xlMkQgYm91bmRzID0gcGVlci5nZXRNYXhDaGFyQm91bmRzKGZyYyk7Ci0gICAgICAgIEFmZmluZVRyYW5zZm9ybSB0cmFuc2Zvcm0gPSBnZXRUcmFuc2Zvcm0oKTsKLSAgICAgICAgLy8gISEgRG9jdW1lbnRhdGlvbiBkb2Vzbid0IGRlc2NyaWJlIG1lYW5pbmcgb2YgbWF4IGNoYXIgYm91bmRzCi0gICAgICAgIC8vIGZvciB0aGUgZm9udHMgdGhhdCBoYXZlIHJvdGF0ZSB0cmFuc2Zvcm1zLiBGb3IgYWxsIHRyYW5zZm9ybXMKLSAgICAgICAgLy8gcmV0dXJuZWQgYm91bmRzIGFyZSB0aGUgYm91bmRzIG9mIHRyYW5zZm9ybWVkIG1heENoYXJCb3VuZHMKLSAgICAgICAgLy8gUmVjdGFuZ2xlMkQgdGhhdCBjb3JyZXNwb25kcyB0byB0aGUgZm9udCB3aXRoIGlkZW50aXR5IHRyYW5zZm9ybS4KLSAgICAgICAgLy8gVE9ETzogcmVzb2x2ZSB0aGlzIGlzc3VlIHRvIHJldHVybiBjb3JyZWN0IGJvdW5kcwotICAgICAgICBib3VuZHMgPSB0cmFuc2Zvcm0uY3JlYXRlVHJhbnNmb3JtZWRTaGFwZShib3VuZHMpLmdldEJvdW5kczJEKCk7Ci0KLSAgICAgICAgcmV0dXJuIGJvdW5kczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgbmV3IEdseXBoVmVjdG9yIG9iamVjdCBwZXJmb3JtaW5nIGZ1bGwgbGF5b3V0IG9mIHRoZSB0ZXh0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb250UmVuZGVyQ29udGV4dC4KLSAgICAgKiBAcGFyYW0gY2hhcnMKLSAgICAgKiAgICAgICAgICAgIHRoZSBjaGFyYWN0ZXIgYXJyYXkgdG8gYmUgbGF5b3V0LgotICAgICAqIEBwYXJhbSBzdGFydAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IG9mZnNldCBvZiB0aGUgdGV4dCB0byB1c2UgZm9yIHRoZSBHbHlwaFZlY3Rvci4KLSAgICAgKiBAcGFyYW0gY291bnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb3VudCBvZiBjaGFyYWN0ZXJzIHRvIHVzZSBmb3IgdGhlIEdseXBoVmVjdG9yLgotICAgICAqIEBwYXJhbSBmbGFncwotICAgICAqICAgICAgICAgICAgdGhlIGZsYWcgaW5kaWNhdGluZyB0ZXh0IGRpcmVjdGlvbjogTEFZT1VUX1JJR0hUX1RPX0xFRlQsCi0gICAgICogICAgICAgICAgICBMQVlPVVRfTEVGVF9UT19SSUdIVC4KLSAgICAgKiBAcmV0dXJuIHRoZSBHbHlwaFZlY3Rvci4KLSAgICAgKi8KLSAgICBwdWJsaWMgR2x5cGhWZWN0b3IgbGF5b3V0R2x5cGhWZWN0b3IoRm9udFJlbmRlckNvbnRleHQgZnJjLCBjaGFyW10gY2hhcnMsIGludCBzdGFydCwgaW50IGNvdW50LAotICAgICAgICAgICAgaW50IGZsYWdzKSB7Ci0gICAgICAgIC8vIFRPRE86IGltcGxlbWVudCBtZXRob2QgZm9yIGJpZGlyZWN0aW9uYWwgdGV4dC4KLSAgICAgICAgLy8gQXQgdGhlIG1vbWVudCBvbmx5IExUUiBhbmQgUlRMIHRleHRzIHN1cHBvcnRlZC4KLSAgICAgICAgaWYgKHN0YXJ0IDwgMCkgewotICAgICAgICAgICAgLy8gYXd0Ljk1PVdyb25nIHN0YXJ0IGluZGV4OiB7MH0KLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuOTUiLCAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgIHN0YXJ0KSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoY291bnQgPCAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuOTg9V3JvbmcgY291bnQgdmFsdWUsIGNhbiBub3QgYmUgbmVnYXRpdmU6IHswfQotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC45OCIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgY291bnQpKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChzdGFydCArIGNvdW50ID4gY2hhcnMubGVuZ3RoKSB7Ci0gICAgICAgICAgICAvLyBhd3QuOTk9V3JvbmcgW3N0YXJ0ICsgY291bnRdIGlzIG91dCBvZiByYW5nZTogezB9Ci0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0Ljk5IiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICAoc3RhcnQgKyBjb3VudCkpKTsKLSAgICAgICAgfQotCi0gICAgICAgIGNoYXJbXSBvdXQgPSBuZXcgY2hhcltjb3VudF07Ci0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkoY2hhcnMsIHN0YXJ0LCBvdXQsIDAsIGNvdW50KTsKLQotICAgICAgICByZXR1cm4gbmV3IENvbW1vbkdseXBoVmVjdG9yKG91dCwgZnJjLCB0aGlzLCBmbGFncyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgU3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoaXMgRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBTdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBGb250LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgIFN0cmluZyBzdGwgPSAicGxhaW4iOyAvLyROT04tTkxTLTEkCi0gICAgICAgIFN0cmluZyByZXN1bHQ7Ci0KLSAgICAgICAgaWYgKHRoaXMuaXNCb2xkKCkgJiYgdGhpcy5pc0l0YWxpYygpKSB7Ci0gICAgICAgICAgICBzdGwgPSAiYm9sZGl0YWxpYyI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZiAodGhpcy5pc0JvbGQoKSAmJiAhdGhpcy5pc0l0YWxpYygpKSB7Ci0gICAgICAgICAgICBzdGwgPSAiYm9sZCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmICghdGhpcy5pc0JvbGQoKSAmJiB0aGlzLmlzSXRhbGljKCkpIHsKLSAgICAgICAgICAgIHN0bCA9ICJpdGFsaWMiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXN1bHQgPSB0aGlzLmdldENsYXNzKCkuZ2V0TmFtZSgpICsgIltmYW1pbHk9IiArIHRoaXMuZ2V0RmFtaWx5KCkgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgIixuYW1lPSIgKyB0aGlzLm5hbWUgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgIixzdHlsZT0iICsgc3RsICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICIsc2l6ZT0iICsgdGhpcy5zaXplICsgIl0iOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBwb3N0c2NyaXB0IG5hbWUgb2YgdGhpcyBGb250LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHBvc3RzY3JpcHQgbmFtZSBvZiB0aGlzIEZvbnQuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZyBnZXRQU05hbWUoKSB7Ci0gICAgICAgIEZvbnRQZWVySW1wbCBwZWVyID0gKEZvbnRQZWVySW1wbCl0aGlzLmdldFBlZXIoKTsKLSAgICAgICAgcmV0dXJuIHBlZXIuZ2V0UFNOYW1lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9naWNhbCBuYW1lIG9mIHRoaXMgRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBsb2dpY2FsIG5hbWUgb2YgdGhpcyBGb250LgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0TmFtZSgpIHsKLSAgICAgICAgcmV0dXJuICh0aGlzLm5hbWUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHBlZXIgb2YgdGhpcyBGb250LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHBlZXIgb2YgdGhpcyBGb250LgotICAgICAqIEBkZXByZWNhdGVkIEZvbnQgcmVuZGVyaW5nIGlzIHBsYXRmb3JtIGluZGVwZW5kZW50IG5vdy4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBqYXZhLmF3dC5wZWVyLkZvbnRQZWVyIGdldFBlZXIoKSB7Ci0gICAgICAgIGlmIChmb250UGVlciA9PSBudWxsKSB7Ci0gICAgICAgICAgICBmb250UGVlciA9IChGb250UGVlckltcGwpVG9vbGtpdC5nZXREZWZhdWx0VG9vbGtpdCgpLmdldEdyYXBoaWNzRmFjdG9yeSgpLmdldEZvbnRQZWVyKAotICAgICAgICAgICAgICAgICAgICB0aGlzKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZm9udFBlZXI7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0cmFuc2Zvcm0gYWN0aW5nIG9uIHRoaXMgRm9udCAoZnJvbSB0aGUgRm9udCdzIGF0dHJpYnV0ZXMpLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHRyYW5zZm9ybWF0aW9uIG9mIHRoaXMgRm9udC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQWZmaW5lVHJhbnNmb3JtIGdldFRyYW5zZm9ybSgpIHsKLSAgICAgICAgT2JqZWN0IHRyYW5zZm9ybSA9IGZSZXF1ZXN0ZWRBdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLlRSQU5TRk9STSk7Ci0KLSAgICAgICAgaWYgKHRyYW5zZm9ybSAhPSBudWxsKSB7Ci0gICAgICAgICAgICBpZiAodHJhbnNmb3JtIGluc3RhbmNlb2YgVHJhbnNmb3JtQXR0cmlidXRlKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuICgoVHJhbnNmb3JtQXR0cmlidXRlKXRyYW5zZm9ybSkuZ2V0VHJhbnNmb3JtKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAodHJhbnNmb3JtIGluc3RhbmNlb2YgQWZmaW5lVHJhbnNmb3JtKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG5ldyBBZmZpbmVUcmFuc2Zvcm0oKEFmZmluZVRyYW5zZm9ybSl0cmFuc2Zvcm0pOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdHJhbnNmb3JtID0gbmV3IEFmZmluZVRyYW5zZm9ybSgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiAoQWZmaW5lVHJhbnNmb3JtKXRyYW5zZm9ybTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGlzIGZvbnQgaXMgdHJhbnNmb3JtZWQgb3Igbm90LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBmb250IGlzIHRyYW5zZm9ybWVkLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNUcmFuc2Zvcm1lZCgpIHsKLSAgICAgICAgcmV0dXJuIHRoaXMudHJhbnNmb3JtZWQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoaXMgZm9udCBoYXMgcGxhaW4gc3R5bGUgb3Igbm90LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBmb250IGhhcyBwbGFpbiBzdHlsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzUGxhaW4oKSB7Ci0gICAgICAgIHJldHVybiAodGhpcy5zdHlsZSA9PSBQTEFJTik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoaXMgZm9udCBoYXMgaXRhbGljIHN0eWxlIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgZm9udCBoYXMgaXRhbGljIHN0eWxlLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNJdGFsaWMoKSB7Ci0gICAgICAgIHJldHVybiAodGhpcy5zdHlsZSAmIElUQUxJQykgIT0gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyBmb250IGhhcyBib2xkIHN0eWxlIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgZm9udCBoYXMgYm9sZCBzdHlsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzQm9sZCgpIHsKLSAgICAgICAgcmV0dXJuICh0aGlzLnN0eWxlICYgQk9MRCkgIT0gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhpcyBGb250IGhhcyB1bmlmb3JtIGxpbmUgbWV0cmljcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgdGhpcyBGb250IGhhcyB1bmlmb3JtIGxpbmUgbWV0cmljcywgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGhhc1VuaWZvcm1MaW5lTWV0cmljcygpIHsKLSAgICAgICAgRm9udFBlZXJJbXBsIHBlZXIgPSAoRm9udFBlZXJJbXBsKXRoaXMuZ2V0UGVlcigpOwotICAgICAgICByZXR1cm4gcGVlci5oYXNVbmlmb3JtTGluZU1ldHJpY3MoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGhhc2ggY29kZSBvZiB0aGlzIEZvbnQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhhc2ggY29kZSBvZiB0aGlzIEZvbnQgb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgaGFzaENvZGUoKSB7Ci0gICAgICAgIEhhc2hDb2RlIGhhc2ggPSBuZXcgSGFzaENvZGUoKTsKLQotICAgICAgICBoYXNoLmFwcGVuZCh0aGlzLm5hbWUpOwotICAgICAgICBoYXNoLmFwcGVuZCh0aGlzLnN0eWxlKTsKLSAgICAgICAgaGFzaC5hcHBlbmQodGhpcy5zaXplKTsKLQotICAgICAgICByZXR1cm4gaGFzaC5oYXNoQ29kZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN0eWxlIG9mIHRoaXMgRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHlsZSBvZiB0aGlzIEZvbnQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRTdHlsZSgpIHsKLSAgICAgICAgcmV0dXJuIHRoaXMuc3R5bGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2l6ZSBvZiB0aGlzIEZvbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc2l6ZSBvZiB0aGlzIEZvbnQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRTaXplKCkgewotICAgICAgICByZXR1cm4gdGhpcy5zaXplOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG51bWJlciBvZiBnbHlwaHMgZm9yIHRoaXMgRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgZ2x5cGhzIGZvciB0aGlzIEZvbnQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXROdW1HbHlwaHMoKSB7Ci0gICAgICAgIGlmIChudW1HbHlwaHMgPT0gLTEpIHsKLSAgICAgICAgICAgIEZvbnRQZWVySW1wbCBwZWVyID0gKEZvbnRQZWVySW1wbCl0aGlzLmdldFBlZXIoKTsKLSAgICAgICAgICAgIHRoaXMubnVtR2x5cGhzID0gcGVlci5nZXROdW1HbHlwaHMoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdGhpcy5udW1HbHlwaHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZ2x5cGhDb2RlIHdoaWNoIGlzIHVzZWQgYXMgZGVmYXVsdCBnbHlwaCB3aGVuIHRoaXMgRm9udCBkb2VzIG5vdAotICAgICAqIGhhdmUgYSBnbHlwaCBmb3IgYSBzcGVjaWZpZWQgVW5pY29kZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtaXNzaW5nIGdseXBoIGNvZGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRNaXNzaW5nR2x5cGhDb2RlKCkgewotICAgICAgICBpZiAobWlzc2luZ0dseXBoQ29kZSA9PSAtMSkgewotICAgICAgICAgICAgRm9udFBlZXJJbXBsIHBlZXIgPSAoRm9udFBlZXJJbXBsKXRoaXMuZ2V0UGVlcigpOwotICAgICAgICAgICAgdGhpcy5taXNzaW5nR2x5cGhDb2RlID0gcGVlci5nZXRNaXNzaW5nR2x5cGhDb2RlKCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHRoaXMubWlzc2luZ0dseXBoQ29kZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmbG9hdCB2YWx1ZSBvZiBmb250J3Mgc2l6ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBmbG9hdCB2YWx1ZSBvZiBmb250J3Mgc2l6ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0U2l6ZTJEKCkgewotICAgICAgICByZXR1cm4gdGhpcy5wb2ludFNpemU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaXRhbGljIGFuZ2xlIG9mIHRoaXMgRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBpdGFsaWMgYW5nbGUgb2YgdGhpcyBGb250LgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRJdGFsaWNBbmdsZSgpIHsKLSAgICAgICAgRm9udFBlZXJJbXBsIHBlZXIgPSAoRm9udFBlZXJJbXBsKXRoaXMuZ2V0UGVlcigpOwotICAgICAgICByZXR1cm4gcGVlci5nZXRJdGFsaWNBbmdsZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGZvbnQgd2l0aCB0aGUgc3BlY2lmaWVkIGZvbnQgZm9ybWF0IGFuZCBmb250IGZpbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZvbnRGb3JtYXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBmb250IGZvcm1hdC4KLSAgICAgKiBAcGFyYW0gZm9udEZpbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBmaWxlIG9iamVjdCByZXByZXNlbnRlZCB0aGUgaW5wdXQgZGF0YSBmb3IgdGhlIGZvbnQuCi0gICAgICogQHJldHVybiB0aGUgRm9udC4KLSAgICAgKiBAdGhyb3dzIEZvbnRGb3JtYXRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpcyB0aHJvd24gaWYgZm9udEZpbGUgZG9lcyBub3QgY29udGFpbiB0aGUgcmVxdWlyZWQgZm9udAotICAgICAqICAgICAgICAgICAgIHRhYmxlcyBmb3IgdGhlIHNwZWNpZmllZCBmb3JtYXQuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIHNpZ25hbHMgdGhhdCBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEZvbnQgY3JlYXRlRm9udChpbnQgZm9udEZvcm1hdCwgRmlsZSBmb250RmlsZSkgdGhyb3dzIEZvbnRGb3JtYXRFeGNlcHRpb24sCi0gICAgICAgICAgICBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vID8/P0FXVCBub3Qgc3VwcG9ydGVkCi0gICAgICAgIElucHV0U3RyZWFtIGlzID0gbmV3IEZpbGVJbnB1dFN0cmVhbShmb250RmlsZSk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gY3JlYXRlRm9udChmb250Rm9ybWF0LCBpcyk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICBpcy5jbG9zZSgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgZm9udCB3aXRoIHRoZSBzcGVjaWZpZWQgZm9udCBmb3JtYXQgYW5kIGlucHV0IHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZm9udEZvcm1hdAotICAgICAqICAgICAgICAgICAgdGhlIGZvbnQgZm9ybWF0LgotICAgICAqIEBwYXJhbSBmb250U3RyZWFtCi0gICAgICogICAgICAgICAgICB0aGUgaW5wdXQgc3RyZWFtIHJlcHJlc2VudGVkIGlucHV0IGRhdGEgZm9yIHRoZSBmb250LgotICAgICAqIEByZXR1cm4gdGhlIEZvbnQuCi0gICAgICogQHRocm93cyBGb250Rm9ybWF0RXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaXMgdGhyb3duIGlmIGZvbnRGaWxlIGRvZXMgbm90IGNvbnRhaW4gdGhlIHJlcXVpcmVkIGZvbnQKLSAgICAgKiAgICAgICAgICAgICB0YWJsZXMgZm9yIHRoZSBzcGVjaWZpZWQgZm9ybWF0LgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBzaWduYWxzIHRoYXQgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBGb250IGNyZWF0ZUZvbnQoaW50IGZvbnRGb3JtYXQsIElucHV0U3RyZWFtIGZvbnRTdHJlYW0pCi0gICAgICAgICAgICB0aHJvd3MgRm9udEZvcm1hdEV4Y2VwdGlvbiwgSU9FeGNlcHRpb24gewotCi0gICAgICAgIC8vID8/P0FXVCBub3Qgc3VwcG9ydGVkCi0KLSAgICAgICAgQnVmZmVyZWRJbnB1dFN0cmVhbSBidWZmU3RyZWFtOwotICAgICAgICBpbnQgYlJlYWQgPSAwOwotICAgICAgICBpbnQgc2l6ZSA9IDgxOTI7Ci0gICAgICAgIC8vIG1lbW9yeSBwYWdlIHNpemUsIGZvciB0aGUgZmFzdGVyIHJlYWRpbmcKLSAgICAgICAgYnl0ZSBidWZbXSA9IG5ldyBieXRlW3NpemVdOwotCi0gICAgICAgIGlmIChmb250Rm9ybWF0ICE9IFRSVUVUWVBFX0ZPTlQpIHsgLy8gYXd0LjlBPVVuc3VwcG9ydGVkIGZvbnQgZm9ybWF0Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjlBIikpOyAvLyROT04tTkxTLTEkIAotICAgICAgICB9Ci0KLSAgICAgICAgLyogR2V0IGZvbnQgZmlsZSBpbiBzeXN0ZW0tc3BlY2lmaWMgZGlyZWN0b3J5ICovCi0KLSAgICAgICAgRmlsZSBmb250RmlsZSA9IFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKS5nZXRHcmFwaGljc0ZhY3RvcnkoKS5nZXRGb250TWFuYWdlcigpCi0gICAgICAgICAgICAgICAgLmdldFRlbXBGb250RmlsZSgpOwotCi0gICAgICAgIC8vIEJFR0lOIGFuZHJvaWQtbW9kaWZpZWQKLSAgICAgICAgYnVmZlN0cmVhbSA9IG5ldyBCdWZmZXJlZElucHV0U3RyZWFtKGZvbnRTdHJlYW0sIDgxOTIpOwotICAgICAgICAvLyBFTkQgYW5kcm9pZC1tb2RpZmllZAotICAgICAgICBGaWxlT3V0cHV0U3RyZWFtIGZPdXRTdHJlYW0gPSBuZXcgRmlsZU91dHB1dFN0cmVhbShmb250RmlsZSk7Ci0KLSAgICAgICAgYlJlYWQgPSBidWZmU3RyZWFtLnJlYWQoYnVmLCAwLCBzaXplKTsKLQotICAgICAgICB3aGlsZSAoYlJlYWQgIT0gLTEpIHsKLSAgICAgICAgICAgIGZPdXRTdHJlYW0ud3JpdGUoYnVmLCAwLCBiUmVhZCk7Ci0gICAgICAgICAgICBiUmVhZCA9IGJ1ZmZTdHJlYW0ucmVhZChidWYsIDAsIHNpemUpOwotICAgICAgICB9Ci0KLSAgICAgICAgYnVmZlN0cmVhbS5jbG9zZSgpOwotICAgICAgICBmT3V0U3RyZWFtLmNsb3NlKCk7Ci0KLSAgICAgICAgRm9udCBmb250ID0gbnVsbDsKLQotICAgICAgICBmb250ID0gVG9vbGtpdC5nZXREZWZhdWx0VG9vbGtpdCgpLmdldEdyYXBoaWNzRmFjdG9yeSgpLmVtYmVkRm9udCgKLSAgICAgICAgICAgICAgICBmb250RmlsZS5nZXRBYnNvbHV0ZVBhdGgoKSk7Ci0gICAgICAgIGlmIChmb250ID09IG51bGwpIHsgLy8gYXd0LjlCPUNhbid0IGNyZWF0ZSBmb250IC0gYmFkIGZvbnQgZGF0YQotICAgICAgICAgICAgdGhyb3cgbmV3IEZvbnRGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuOUIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZm9udDsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9Gb250Rm9ybWF0RXhjZXB0aW9uLmphdmEgYi9hd3QvamF2YS9hd3QvRm9udEZvcm1hdEV4Y2VwdGlvbi5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4MDY3MTFhLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9Gb250Rm9ybWF0RXhjZXB0aW9uLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0NyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotLyoqCi0gKiBUaGUgRm9udEZvcm1hdEV4Y2VwdGlvbiBjbGFzcyBpcyB1c2VkIHRvIHByb3ZpZGUgbm90aWZpY2F0aW9uIGFuZCBpbmZvcm1hdGlvbgotICogdGhhdCBmb250IGNhbid0IGJlIGNyZWF0ZWQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgRm9udEZvcm1hdEV4Y2VwdGlvbiBleHRlbmRzIEV4Y2VwdGlvbiB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtNDQ4MTI5MDE0NzgxMTM2MTI3Mkw7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZm9udCBmb3JtYXQgZXhjZXB0aW9uIHdpdGggZGV0YWlsZWQgbWVzc2FnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmVhc29uCi0gICAgICogICAgICAgICAgICB0aGUgZGV0YWlsZWQgbWVzc2FnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgRm9udEZvcm1hdEV4Y2VwdGlvbihTdHJpbmcgcmVhc29uKSB7Ci0gICAgICAgIHN1cGVyKHJlYXNvbik7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvRm9udE1ldHJpY3MuamF2YSBiL2F3dC9qYXZhL2F3dC9Gb250TWV0cmljcy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5MDgyNjI2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9Gb250TWV0cmljcy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDY2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuZm9udC5Gb250UmVuZGVyQ29udGV4dDsKLWltcG9ydCBqYXZhLmF3dC5mb250LkxpbmVNZXRyaWNzOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS5pby5TZXJpYWxpemFibGU7Ci1pbXBvcnQgamF2YS50ZXh0LkNoYXJhY3Rlckl0ZXJhdG9yOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIEZvbnRNZXRyaWNzIGNsYXNzIGNvbnRhaW5zIGluZm9ybWF0aW9uIGFib3V0IHRoZSByZW5kZXJpbmcgb2YgYQotICogcGFydGljdWxhciBmb250IG9uIGEgcGFydGljdWxhciBzY3JlZW4uCi0gKiA8cD4KLSAqIEVhY2ggY2hhcmFjdGVyIGluIHRoZSBGb250IGhhcyB0aHJlZSB2YWx1ZXMgdGhhdCBoZWxwIGRlZmluZSB3aGVyZSB0byBwbGFjZQotICogaXQ6IGFuIGFzY2VudCwgYSBkZXNjZW50LCBhbmQgYW4gYWR2YW5jZS4gVGhlIGFzY2VudCBpcyB0aGUgZGlzdGFuY2UgdGhlCi0gKiBjaGFyYWN0ZXIgZXh0ZW5kcyBhYm92ZSB0aGUgYmFzZWxpbmUuIFRoZSBkZXNjZW50IGlzIHRoZSBkaXN0YW5jZSB0aGUKLSAqIGNoYXJhY3RlciBleHRlbmRzIGJlbG93IHRoZSBiYXNlbGluZS4gVGhlIGFkdmFuY2Ugd2lkdGggZGVmaW5lcyB0aGUgcG9zaXRpb24KLSAqIGF0IHdoaWNoIHRoZSBuZXh0IGNoYXJhY3RlciBzaG91bGQgYmUgcGxhY2VkLgotICogPHA+Ci0gKiBBbiBhcnJheSBvZiBjaGFyYWN0ZXJzIG9yIGEgc3RyaW5nIGhhcyBhbiBhc2NlbnQsIGEgZGVzY2VudCwgYW5kIGFuIGFkdmFuY2UKLSAqIHdpZHRoIHRvby4gVGhlIGFzY2VudCBvciBkZXNjZW50IG9mIHRoZSBhcnJheSBpcyBzcGVjaWZpZWQgYnkgdGhlIG1heGltdW0KLSAqIGFzY2VudCBvciBkZXNjZW50IG9mIHRoZSBjaGFyYWN0ZXJzIGluIHRoZSBhcnJheS4gVGhlIGFkdmFuY2Ugd2lkdGggaXMgdGhlCi0gKiBzdW0gb2YgdGhlIGFkdmFuY2Ugd2lkdGhzIG9mIGVhY2ggb2YgdGhlIGNoYXJhY3RlcnMgaW4gdGhlIGNoYXJhY3RlciBhcnJheS4KLSAqIDwvcD4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBGb250TWV0cmljcyBpbXBsZW1lbnRzIFNlcmlhbGl6YWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAxNjgxMTI2MjI1MjA1MDUwMTQ3TDsKLQotICAgIC8qKgotICAgICAqIFRoZSBmb250IGZyb20gd2hpY2ggdGhlIEZvbnRNZXRyaWNzIGlzIGNyZWF0ZWQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEZvbnQgZm9udDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBmb250IG1ldHJpY3MgZnJvbSB0aGUgc3BlY2lmaWVkIEZvbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZudAotICAgICAqICAgICAgICAgICAgdGhlIEZvbnQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEZvbnRNZXRyaWNzKEZvbnQgZm50KSB7Ci0gICAgICAgIHRoaXMuZm9udCA9IGZudDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBTdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBGb250TWV0cmljcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsKLSAgICAgICAgcmV0dXJuIHRoaXMuZ2V0Q2xhc3MoKS5nZXROYW1lKCkgKyAiW2ZvbnQ9IiArIHRoaXMuZ2V0Rm9udCgpICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICJhc2NlbnQ9IiArIHRoaXMuZ2V0QXNjZW50KCkgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgIiwgZGVzY2VudD0iICsgdGhpcy5nZXREZXNjZW50KCkgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgIiwgaGVpZ2h0PSIgKyB0aGlzLmdldEhlaWdodCgpICsgIl0iOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmb250IGFzc29jaWF0ZWQgd2l0aCB0aGlzIEZvbnRNZXRyaWNzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGZvbnQgYXNzb2NpYXRlZCB3aXRoIHRoaXMgRm9udE1ldHJpY3MuCi0gICAgICovCi0gICAgcHVibGljIEZvbnQgZ2V0Rm9udCgpIHsKLSAgICAgICAgcmV0dXJuIGZvbnQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaGVpZ2h0IG9mIHRoZSB0ZXh0IGxpbmUgaW4gdGhpcyBGb250LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhlaWdodCBvZiB0aGUgdGV4dCBsaW5lIGluIHRoaXMgRm9udC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEhlaWdodCgpIHsKLSAgICAgICAgcmV0dXJuIHRoaXMuZ2V0QXNjZW50KCkgKyB0aGlzLmdldERlc2NlbnQoKSArIHRoaXMuZ2V0TGVhZGluZygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZvbnQgYXNjZW50IG9mIHRoZSBGb250IGFzc29jaWF0ZWQgd2l0aCB0aGlzIEZvbnRNZXRyaWNzLiBUaGUKLSAgICAgKiBmb250IGFzY2VudCBpcyB0aGUgZGlzdGFuY2UgZnJvbSB0aGUgZm9udCdzIGJhc2VsaW5lIHRvIHRoZSB0b3Agb2YgbW9zdAotICAgICAqIGFscGhhbnVtZXJpYyBjaGFyYWN0ZXJzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFzY2VudCBvZiB0aGUgRm9udCBhc3NvY2lhdGVkIHdpdGggdGhpcyBGb250TWV0cmljcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEFzY2VudCgpIHsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZm9udCBkZXNjZW50IG9mIHRoZSBGb250IGFzc29jaWF0ZWQgd2l0aCB0aGlzIEZvbnRNZXRyaWNzLiBUaGUKLSAgICAgKiBmb250IGRlc2NlbnQgaXMgdGhlIGRpc3RhbmNlIGZyb20gdGhlIGZvbnQncyBiYXNlbGluZSB0byB0aGUgYm90dG9tIG9mCi0gICAgICogbW9zdCBhbHBoYW51bWVyaWMgY2hhcmFjdGVycyB3aXRoIGRlc2NlbmRlcnMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGVzY2VudCBvZiB0aGUgRm9udCBhc3NvY2lhdGVkIHdpdGggdGhpcyBGb250TWV0cmljcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldERlc2NlbnQoKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxlYWRpbmcgb2YgdGhlIEZvbnQgYXNzb2NpYXRlZCB3aXRoIHRoaXMgRm9udE1ldHJpY3MuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbGVhZGluZyBvZiB0aGUgRm9udCBhc3NvY2lhdGVkIHdpdGggdGhpcyBGb250TWV0cmljcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldExlYWRpbmcoKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIExpbmVNZXRyaWNzIG9iamVjdCBmb3IgdGhlIHNwZWNpZmllZCBDaGFyYWN0ZXJJdGVyYXRvciBpbiB0aGUKLSAgICAgKiBzcGVjaWZpZWQgR3JhcGhpY3MuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNpCi0gICAgICogICAgICAgICAgICB0aGUgQ2hhcmFjdGVySXRlcmF0b3IuCi0gICAgICogQHBhcmFtIGJlZ2luSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQuCi0gICAgICogQHBhcmFtIGxpbWl0Ci0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMgdG8gYmUgdXNlZC4KLSAgICAgKiBAcGFyYW0gY29udGV4dAotICAgICAqICAgICAgICAgICAgdGhlIEdyYXBoaWNzLgotICAgICAqIEByZXR1cm4gdGhlIExpbmVNZXRyaWNzIG9iamVjdCBmb3IgdGhlIHNwZWNpZmllZCBDaGFyYWN0ZXJJdGVyYXRvciBpbiB0aGUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZCBHcmFwaGljcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgTGluZU1ldHJpY3MgZ2V0TGluZU1ldHJpY3MoQ2hhcmFjdGVySXRlcmF0b3IgY2ksIGludCBiZWdpbkluZGV4LCBpbnQgbGltaXQsCi0gICAgICAgICAgICBHcmFwaGljcyBjb250ZXh0KSB7Ci0gICAgICAgIHJldHVybiBmb250LmdldExpbmVNZXRyaWNzKGNpLCBiZWdpbkluZGV4LCBsaW1pdCwgdGhpcy5nZXRGUkNGcm9tR3JhcGhpY3MoY29udGV4dCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIExpbmVNZXRyaWNzIG9iamVjdCBmb3IgdGhlIHNwZWNpZmllZCBTdHJpbmcgaW4gdGhlIHNwZWNpZmllZAotICAgICAqIEdyYXBoaWNzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBTdHJpbmcuCi0gICAgICogQHBhcmFtIGNvbnRleHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBHcmFwaGljcy4KLSAgICAgKiBAcmV0dXJuIHRoZSBMaW5lTWV0cmljcyBvYmplY3QgZm9yIHRoZSBzcGVjaWZpZWQgU3RyaW5nIGluIHRoZSBzcGVjaWZpZWQKLSAgICAgKiAgICAgICAgIEdyYXBoaWNzLgotICAgICAqLwotICAgIHB1YmxpYyBMaW5lTWV0cmljcyBnZXRMaW5lTWV0cmljcyhTdHJpbmcgc3RyLCBHcmFwaGljcyBjb250ZXh0KSB7Ci0gICAgICAgIHJldHVybiBmb250LmdldExpbmVNZXRyaWNzKHN0ciwgdGhpcy5nZXRGUkNGcm9tR3JhcGhpY3MoY29udGV4dCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIExpbmVNZXRyaWNzIG9iamVjdCBmb3IgdGhlIHNwZWNpZmllZCBjaGFyYWN0ZXIgYXJyYXkgaW4gdGhlCi0gICAgICogc3BlY2lmaWVkIEdyYXBoaWNzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaGFycwotICAgICAqICAgICAgICAgICAgdGhlIGNoYXJhY3RlciBhcnJheS4KLSAgICAgKiBAcGFyYW0gYmVnaW5JbmRleAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBvZiBhcnJheS4KLSAgICAgKiBAcGFyYW0gbGltaXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgY2hhcmFjdGVycyB0byBiZSB1c2VkLgotICAgICAqIEBwYXJhbSBjb250ZXh0Ci0gICAgICogICAgICAgICAgICB0aGUgR3JhcGhpY3MuCi0gICAgICogQHJldHVybiB0aGUgTGluZU1ldHJpY3Mgb2JqZWN0IGZvciB0aGUgc3BlY2lmaWVkIGNoYXJhY3RlciBhcnJheSBpbiB0aGUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZCBHcmFwaGljcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgTGluZU1ldHJpY3MgZ2V0TGluZU1ldHJpY3MoY2hhcltdIGNoYXJzLCBpbnQgYmVnaW5JbmRleCwgaW50IGxpbWl0LCBHcmFwaGljcyBjb250ZXh0KSB7Ci0gICAgICAgIHJldHVybiBmb250LmdldExpbmVNZXRyaWNzKGNoYXJzLCBiZWdpbkluZGV4LCBsaW1pdCwgdGhpcy5nZXRGUkNGcm9tR3JhcGhpY3MoY29udGV4dCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIExpbmVNZXRyaWNzIG9iamVjdCBmb3IgdGhlIHNwZWNpZmllZCBTdHJpbmcgaW4gdGhlIHNwZWNpZmllZAotICAgICAqIEdyYXBoaWNzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBTdHJpbmcuCi0gICAgICogQHBhcmFtIGJlZ2luSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQuCi0gICAgICogQHBhcmFtIGxpbWl0Ci0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMgdG8gYmUgdXNlZC4KLSAgICAgKiBAcGFyYW0gY29udGV4dAotICAgICAqICAgICAgICAgICAgdGhlIEdyYXBoaWNzLgotICAgICAqIEByZXR1cm4gdGhlIExpbmVNZXRyaWNzIG9iamVjdCBmb3IgdGhlIHNwZWNpZmllZCBTdHJpbmcgaW4gdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgR3JhcGhpY3MuCi0gICAgICovCi0gICAgcHVibGljIExpbmVNZXRyaWNzIGdldExpbmVNZXRyaWNzKFN0cmluZyBzdHIsIGludCBiZWdpbkluZGV4LCBpbnQgbGltaXQsIEdyYXBoaWNzIGNvbnRleHQpIHsKLSAgICAgICAgcmV0dXJuIGZvbnQuZ2V0TGluZU1ldHJpY3Moc3RyLCBiZWdpbkluZGV4LCBsaW1pdCwgdGhpcy5nZXRGUkNGcm9tR3JhcGhpY3MoY29udGV4dCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGNoYXJhY3RlcidzIG1heGltdW0gYm91bmRzIGluIHRoZSBzcGVjaWZpZWQgR3JhcGhpY3MgY29udGV4dC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29udGV4dAotICAgICAqICAgICAgICAgICAgdGhlIEdyYXBoaWNzIGNvbnRleHQuCi0gICAgICogQHJldHVybiB0aGUgY2hhcmFjdGVyJ3MgbWF4aW11bSBib3VuZHMgaW4gdGhlIHNwZWNpZmllZCBHcmFwaGljcyBjb250ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRNYXhDaGFyQm91bmRzKEdyYXBoaWNzIGNvbnRleHQpIHsKLSAgICAgICAgcmV0dXJuIHRoaXMuZm9udC5nZXRNYXhDaGFyQm91bmRzKHRoaXMuZ2V0RlJDRnJvbUdyYXBoaWNzKGNvbnRleHQpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBDaGFyYWN0ZXJJdGVyYXRvciBpbiB0aGUgc3BlY2lmaWVkCi0gICAgICogR3JhcGhpY3MgY29udGV4dC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2kKLSAgICAgKiAgICAgICAgICAgIHRoZSBDaGFyYWN0ZXJJdGVyYXRvci4KLSAgICAgKiBAcGFyYW0gYmVnaW5JbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGJlZ2luIG9mZnNldCBvZiB0aGUgYXJyYXkuCi0gICAgICogQHBhcmFtIGxpbWl0Ci0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMuCi0gICAgICogQHBhcmFtIGNvbnRleHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBHcmFwaGljcy4KLSAgICAgKiBAcmV0dXJuIHRoZSBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBDaGFyYWN0ZXJJdGVyYXRvciBpbiB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBHcmFwaGljcyBjb250ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRTdHJpbmdCb3VuZHMoQ2hhcmFjdGVySXRlcmF0b3IgY2ksIGludCBiZWdpbkluZGV4LCBpbnQgbGltaXQsCi0gICAgICAgICAgICBHcmFwaGljcyBjb250ZXh0KSB7Ci0gICAgICAgIHJldHVybiBmb250LmdldFN0cmluZ0JvdW5kcyhjaSwgYmVnaW5JbmRleCwgbGltaXQsIHRoaXMuZ2V0RlJDRnJvbUdyYXBoaWNzKGNvbnRleHQpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBTdHJpbmcgaW4gdGhlIHNwZWNpZmllZCBHcmFwaGljcwotICAgICAqIGNvbnRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cgotICAgICAqICAgICAgICAgICAgdGhlIFN0cmluZy4KLSAgICAgKiBAcGFyYW0gYmVnaW5JbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGJlZ2luIG9mZnNldCBvZiB0aGUgYXJyYXkuCi0gICAgICogQHBhcmFtIGxpbWl0Ci0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMuCi0gICAgICogQHBhcmFtIGNvbnRleHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBHcmFwaGljcy4KLSAgICAgKiBAcmV0dXJuIHRoZSBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBTdHJpbmcgaW4gdGhlIHNwZWNpZmllZCBHcmFwaGljcwotICAgICAqICAgICAgICAgY29udGV4dC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0U3RyaW5nQm91bmRzKFN0cmluZyBzdHIsIGludCBiZWdpbkluZGV4LCBpbnQgbGltaXQsIEdyYXBoaWNzIGNvbnRleHQpIHsKLSAgICAgICAgcmV0dXJuIGZvbnQuZ2V0U3RyaW5nQm91bmRzKHN0ciwgYmVnaW5JbmRleCwgbGltaXQsIHRoaXMuZ2V0RlJDRnJvbUdyYXBoaWNzKGNvbnRleHQpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBjaGFyYWN0ZXJzIGFycmF5IGluIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBHcmFwaGljcyBjb250ZXh0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaGFycwotICAgICAqICAgICAgICAgICAgdGhlIGNoYXJhY3RlcnMgYXJyYXkuCi0gICAgICogQHBhcmFtIGJlZ2luSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBiZWdpbiBvZmZzZXQgb2YgdGhlIGFycmF5LgotICAgICAqIEBwYXJhbSBsaW1pdAotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBjaGFyYWN0ZXJzLgotICAgICAqIEBwYXJhbSBjb250ZXh0Ci0gICAgICogICAgICAgICAgICB0aGUgR3JhcGhpY3MuCi0gICAgICogQHJldHVybiB0aGUgYm91bmRzIG9mIHRoZSBzcGVjaWZpZWQgY2hhcmFjdGVycyBhcnJheSBpbiB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBHcmFwaGljcyBjb250ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRTdHJpbmdCb3VuZHMoY2hhcltdIGNoYXJzLCBpbnQgYmVnaW5JbmRleCwgaW50IGxpbWl0LCBHcmFwaGljcyBjb250ZXh0KSB7Ci0gICAgICAgIHJldHVybiBmb250LmdldFN0cmluZ0JvdW5kcyhjaGFycywgYmVnaW5JbmRleCwgbGltaXQsIHRoaXMuZ2V0RlJDRnJvbUdyYXBoaWNzKGNvbnRleHQpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBTdHJpbmcgaW4gdGhlIHNwZWNpZmllZCBHcmFwaGljcwotICAgICAqIGNvbnRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cgotICAgICAqICAgICAgICAgICAgdGhlIFN0cmluZy4KLSAgICAgKiBAcGFyYW0gY29udGV4dAotICAgICAqICAgICAgICAgICAgdGhlIEdyYXBoaWNzLgotICAgICAqIEByZXR1cm4gdGhlIGJvdW5kcyBvZiB0aGUgc3BlY2lmaWVkIFN0cmluZyBpbiB0aGUgc3BlY2lmaWVkIEdyYXBoaWNzCi0gICAgICogICAgICAgICBjb250ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRTdHJpbmdCb3VuZHMoU3RyaW5nIHN0ciwgR3JhcGhpY3MgY29udGV4dCkgewotICAgICAgICByZXR1cm4gZm9udC5nZXRTdHJpbmdCb3VuZHMoc3RyLCB0aGlzLmdldEZSQ0Zyb21HcmFwaGljcyhjb250ZXh0KSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoZSBGb250IGhhcyB1bmlmb3JtIGxpbmUgbWV0cmljcyBvciBub3QuIFRoZSBGb250IGNhbiBjb250YWluCi0gICAgICogY2hhcmFjdGVycyBvZiBvdGhlciBmb250cyBmb3IgY292ZXJpbmcgY2hhcmFjdGVyIHNldC4gSW4gdGhpcyBjYXNlIHRoZQotICAgICAqIEZvbnQgaXNuJ3QgdW5pZm9ybS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBGb250IGhhcyB1bmlmb3JtIGxpbmUgbWV0cmljcywgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGhhc1VuaWZvcm1MaW5lTWV0cmljcygpIHsKLSAgICAgICAgcmV0dXJuIHRoaXMuZm9udC5oYXNVbmlmb3JtTGluZU1ldHJpY3MoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBkaXN0YW5jZSBmcm9tIHRoZSBsZWZ0bW9zdCBwb2ludCB0byB0aGUgcmlnaHRtb3N0IHBvaW50IG9uCi0gICAgICogdGhlIHN0cmluZydzIGJhc2VsaW5lIHNob3dpbmcgdGhlIHNwZWNpZmllZCBhcnJheSBvZiBieXRlcyBpbiB0aGlzIEZvbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBieXRlcyB0byBiZSBtZWFzdXJlZC4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBsZW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgYnl0ZXMgdG8gYmUgbWVhc3VyZWQuCi0gICAgICogQHJldHVybiB0aGUgYWR2YW5jZSB3aWR0aCBvZiB0aGUgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIGludCBieXRlc1dpZHRoKGJ5dGVbXSBkYXRhLCBpbnQgb2ZmLCBpbnQgbGVuKSB7Ci0gICAgICAgIGludCB3aWR0aCA9IDA7Ci0gICAgICAgIGlmICgob2ZmID49IGRhdGEubGVuZ3RoKSB8fCAob2ZmIDwgMCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xM0I9b2Zmc2V0IG9mZiBpcyBvdXQgb2YgcmFuZ2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTNCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoKG9mZiArIGxlbiA+IGRhdGEubGVuZ3RoKSkgewotICAgICAgICAgICAgLy8gYXd0LjEzQz1udW1iZXIgb2YgZWxlbWV0cyBsZW4gaXMgb3V0IG9mIHJhbmdlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEzQyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IG9mZjsgaSA8IG9mZiArIGxlbjsgaSsrKSB7Ci0gICAgICAgICAgICB3aWR0aCArPSBjaGFyV2lkdGgoZGF0YVtpXSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gd2lkdGg7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgZGlzdGFuY2UgZnJvbSB0aGUgbGVmdG1vc3QgcG9pbnQgdG8gdGhlIHJpZ2h0bW9zdCBwb2ludCBvbgotICAgICAqIHRoZSBzdHJpbmcncyBiYXNlbGluZSBzaG93aW5nIHRoZSBzcGVjaWZpZWQgYXJyYXkgb2YgY2hhcmFjdGVycyBpbiB0aGlzCi0gICAgICogRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGNoYXJhY3RlcnMgdG8gYmUgbWVhc3VyZWQuCi0gICAgICogQHBhcmFtIG9mZgotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IG9mZnNldC4KLSAgICAgKiBAcGFyYW0gbGVuCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGJ5dGVzIHRvIGJlIG1lYXN1cmVkLgotICAgICAqIEByZXR1cm4gdGhlIGFkdmFuY2Ugd2lkdGggb2YgdGhlIGFycmF5LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgY2hhcnNXaWR0aChjaGFyW10gZGF0YSwgaW50IG9mZiwgaW50IGxlbikgewotICAgICAgICBpbnQgd2lkdGggPSAwOwotICAgICAgICBpZiAoKG9mZiA+PSBkYXRhLmxlbmd0aCkgfHwgKG9mZiA8IDApKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTNCPW9mZnNldCBvZmYgaXMgb3V0IG9mIHJhbmdlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjEzQiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChvZmYgKyBsZW4gPiBkYXRhLmxlbmd0aCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xM0M9bnVtYmVyIG9mIGVsZW1ldHMgbGVuIGlzIG91dCBvZiByYW5nZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xM0MiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSBvZmY7IGkgPCBvZmYgKyBsZW47IGkrKykgewotICAgICAgICAgICAgd2lkdGggKz0gY2hhcldpZHRoKGRhdGFbaV0pOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHdpZHRoOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGRpc3RhbmNlIGZyb20gdGhlIGxlZnRtb3N0IHBvaW50IHRvIHRoZSByaWdodG1vc3QgcG9pbnQgb2YKLSAgICAgKiB0aGUgc3BlY2lmaWVkIGNoYXJhY3RlciBpbiB0aGlzIEZvbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNoCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIFVuaWNvZGUgcG9pbnQgY29kZSBvZiBjaGFyYWN0ZXIgdG8gYmUgbWVhc3VyZWQuCi0gICAgICogQHJldHVybiB0aGUgYWR2YW5jZSB3aWR0aCBvZiB0aGUgY2hhcmFjdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgY2hhcldpZHRoKGludCBjaCkgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBkaXN0YW5jZSBmcm9tIHRoZSBsZWZ0bW9zdCBwb2ludCB0byB0aGUgcmlnaHRtb3N0IHBvaW50IG9mCi0gICAgICogdGhlIHNwZWNpZmllZCBjaGFyYWN0ZXIgaW4gdGhpcyBGb250LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBjaGFyYWN0ZXIgdG8gYmUgbWVhc3VyZWQuCi0gICAgICogQHJldHVybiB0aGUgYWR2YW5jZSB3aWR0aCBvZiB0aGUgY2hhcmFjdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgY2hhcldpZHRoKGNoYXIgY2gpIHsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWF4aW11bSBhZHZhbmNlIHdpZHRoIG9mIGNoYXJhY3RlciBpbiB0aGlzIEZvbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWF4aW11bSBhZHZhbmNlIHdpZHRoIG9mIGNoYXJhY3RlciBpbiB0aGlzIEZvbnQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRNYXhBZHZhbmNlKCkgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtYXhpbXVtIGZvbnQgYXNjZW50IG9mIHRoZSBGb250IGFzc29jaWF0ZWQgd2l0aCB0aGlzCi0gICAgICogRm9udE1ldHJpY3MuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWF4aW11bSBmb250IGFzY2VudCBvZiB0aGUgRm9udCBhc3NvY2lhdGVkIHdpdGggdGhpcwotICAgICAqICAgICAgICAgRm9udE1ldHJpY3MuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRNYXhBc2NlbnQoKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1heGltdW0gZm9udCBkZXNjZW50IG9mIGNoYXJhY3RlciBpbiB0aGlzIEZvbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWF4aW11bSBmb250IGRlc2NlbnQgb2YgY2hhcmFjdGVyIGluIHRoaXMgRm9udC4KLSAgICAgKiBAZGVwcmVjYXRlZCBSZXBsYWNlZCBieSBnZXRNYXhEZXNjZW50KCkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIGludCBnZXRNYXhEZWNlbnQoKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1heGltdW0gZm9udCBkZXNjZW50IG9mIGNoYXJhY3RlciBpbiB0aGlzIEZvbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWF4aW11bSBmb250IGRlc2NlbnQgb2YgY2hhcmFjdGVyIGluIHRoaXMgRm9udC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE1heERlc2NlbnQoKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFkdmFuY2Ugd2lkdGhzIG9mIHRoZSBjaGFyYWN0ZXJzIGluIHRoZSBGb250LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFkdmFuY2Ugd2lkdGhzIG9mIHRoZSBjaGFyYWN0ZXJzIGluIHRoZSBGb250LgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXRXaWR0aHMoKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGFkdmFuY2Ugd2lkdGggZm9yIHRoZSBzcGVjaWZpZWQgU3RyaW5nIGluIHRoaXMgRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3RyCi0gICAgICogICAgICAgICAgICBTdHJpbmcgdG8gYmUgbWVhc3VyZWQuCi0gICAgICogQHJldHVybiB0aGUgdGhlIGFkdmFuY2Ugd2lkdGggZm9yIHRoZSBzcGVjaWZpZWQgU3RyaW5nIGluIHRoaXMgRm9udC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHN0cmluZ1dpZHRoKFN0cmluZyBzdHIpIHsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIEZvbnRSZW5kZXJDb250ZXh0IGluc3RhbmNlIG9mIHRoZSBHcmFwaGljcyBjb250ZXh0IHNwZWNpZmllZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29udGV4dAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBHcmFwaGljcyBjb250ZXh0LgotICAgICAqIEByZXR1cm4gYSBGb250UmVuZGVyQ29udGV4dCBvZiB0aGUgc3BlY2lmaWVkIEdyYXBoaWNzIGNvbnRleHQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBGb250UmVuZGVyQ29udGV4dCBnZXRGUkNGcm9tR3JhcGhpY3MoR3JhcGhpY3MgY29udGV4dCkgewotICAgICAgICBGb250UmVuZGVyQ29udGV4dCBmcmM7Ci0gICAgICAgIGlmIChjb250ZXh0IGluc3RhbmNlb2YgR3JhcGhpY3MyRCkgewotICAgICAgICAgICAgZnJjID0gKChHcmFwaGljczJEKWNvbnRleHQpLmdldEZvbnRSZW5kZXJDb250ZXh0KCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmcmMgPSBuZXcgRm9udFJlbmRlckNvbnRleHQobnVsbCwgZmFsc2UsIGZhbHNlKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBmcmM7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0dyYWRpZW50UGFpbnQuamF2YSBiL2F3dC9qYXZhL2F3dC9HcmFkaWVudFBhaW50LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDNiMzJlZjUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0dyYWRpZW50UGFpbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuZ2VvbS5BZmZpbmVUcmFuc2Zvcm07Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5Qb2ludDJEOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBHcmFkaWVudFBhaW50IGNsYXNzIGRlZmluZXMgYSB3YXkgdG8gZmlsbCBhIFNoYXBlIHdpdGggYSBsaW5lYXIgY29sb3IKLSAqIGdyYWRpZW50IHBhdHRlcm4uCi0gKiA8cD4KLSAqIFRoZSBHcmFkaWVudFBhaW50J3MgZmlsbCBwYXR0ZXJuIGlzIGRldGVybWluZWQgYnkgdHdvIHBvaW50cyBhbmQgdHdvIGNvbG9ycywKLSAqIHBsdXMgdGhlIGN5Y2xpYyBtb2RlIG9wdGlvbi4gRWFjaCBvZiB0aGUgdHdvIHBvaW50cyBpcyBwYWludGVkIHdpdGggaXRzCi0gKiBjb3JyZXNwb25kaW5nIGNvbG9yLCBhbmQgb24gdGhlIGxpbmUgc2VnbWVudCBjb25uZWN0aW5nIHRoZSB0d28gcG9pbnRzLCB0aGUKLSAqIGNvbG9yIGlzIHByb3BvcnRpb25hbGx5IGNoYW5nZWQgYmV0d2VlbiB0aGUgdHdvIGNvbG9ycy4gRm9yIHBvaW50cyBvbiB0aGUKLSAqIHNhbWUgbGluZSB3aGljaCBhcmUgbm90IGJldHdlZW4gdGhlIHR3byBzcGVjaWZpZWQgcG9pbnRzIChvdXRzaWRlIG9mIHRoZQotICogY29ubmVjdGluZyBzZWdtZW50KSB0aGVpciBjb2xvciBpcyBkZXRlcm1pbmVkIGJ5IHRoZSBjeWNsaWMgbW9kZSBvcHRpb24uIElmCi0gKiB0aGUgbW9kZSBpcyBjeWNsaWMsIHRoZW4gdGhlIHJlc3Qgb2YgdGhlIGxpbmUgcmVwZWF0cyB0aGUgY29sb3IgcGF0dGVybiBvZgotICogdGhlIGNvbm5lY3Rpbmcgc2VnbWVudCwgY3ljbGluZyBiYWNrIGFuZCBmb3J0aCBiZXR3ZWVuIHRoZSB0d28gY29sb3JzLiBJZgotICogbm90LCB0aGUgbW9kZSBpcyBhY3ljbGljIHdoaWNoIG1lYW5zIHRoYXQgYWxsIHBvaW50cyBvbiB0aGUgbGluZSBvdXRzaWRlIHRoZQotICogY29ubmVjdGluZyBsaW5lIHNlZ21lbnQgYXJlIGdpdmVuIHRoZSBzYW1lIGNvbG9yIGFzIHRoZSBjbG9zZXN0IG9mIHRoZSB0d28KLSAqIHNwZWNpZmllZCBwb2ludHMuCi0gKiA8cD4KLSAqIFRoZSBjb2xvciBvZiBwb2ludHMgdGhhdCBhcmUgbm90IG9uIHRoZSBsaW5lIGNvbm5lY3RpbmcgdGhlIHR3byBzcGVjaWZpZWQKLSAqIHBvaW50cyBhcmUgZ2l2ZW4gYnkgcGVycGVuZGljdWxhciBwcm9qZWN0aW9uOiBieSB0YWtpbmcgdGhlIHNldCBvZiBsaW5lcwotICogcGVycGVuZGljdWxhciB0byB0aGUgY29ubmVjdGluZyBsaW5lIGFuZCBmb3IgZWFjaCBvbmUsIHRoZSB3aG9sZSBsaW5lIGlzCi0gKiBjb2xvcmVkIHdpdGggdGhlIHNhbWUgY29sb3IuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgR3JhZGllbnRQYWludCBpbXBsZW1lbnRzIFBhaW50IHsKLQotICAgIC8qKgotICAgICAqIFRoZSBzdGFydCBwb2ludCBjb2xvci4KLSAgICAgKi8KLSAgICBDb2xvciBjb2xvcjE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZW5kIGNvbG9yIHBvaW50LgotICAgICAqLwotICAgIENvbG9yIGNvbG9yMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBsb2NhdGlvbiBvZiB0aGUgc3RhcnQgcG9pbnQuCi0gICAgICovCi0gICAgUG9pbnQyRCBwb2ludDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbG9jYXRpb24gb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgKi8KLSAgICBQb2ludDJEIHBvaW50MjsKLQotICAgIC8qKgotICAgICAqIFRoZSBpbmRpY2F0b3Igb2YgY3ljbGUgZmlsbGluZy4gSWYgVFJVRSBmaWxsaW5nIHJlcGVhdGVkIG91dHNpZGUgcG9pbnRzCi0gICAgICogc3RyaXBlLCBpZiBGQUxTRSBzb2xpZCBjb2xvciBmaWxsaW5nIG91dHNpZGUuCi0gICAgICovCi0gICAgYm9vbGVhbiBjeWNsaWM7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgR3JhZGllbnRQYWludCB3aXRoIGN5Y2xpYyBvciBhY3ljbGljIG1vZGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBvaW50MQotICAgICAqICAgICAgICAgICAgdGhlIGZpcnN0IHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY29sb3IxCi0gICAgICogICAgICAgICAgICB0aGUgQ29sb3Igb2YgdGhlIGZpcnN0IHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiBAcGFyYW0gcG9pbnQyCi0gICAgICogICAgICAgICAgICB0aGUgc2Vjb25kIHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY29sb3IyCi0gICAgICogICAgICAgICAgICB0aGUgQ29sb3Igb2YgdGhlIHNlY29uZCBzcGVjaWZpZWQgcG9pbnQuCi0gICAgICogQHBhcmFtIGN5Y2xpYwotICAgICAqICAgICAgICAgICAgdGhlIGN5Y2xpYyBtb2RlIC0gdHJ1ZSBpZiB0aGUgZ3JhZGllbnQgcGF0dGVybiBzaG91bGQgY3ljbGUKLSAgICAgKiAgICAgICAgICAgIHJlcGVhdGVkbHkgYmV0d2VlbiB0aGUgdHdvIGNvbG9yczsgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBHcmFkaWVudFBhaW50KFBvaW50MkQgcG9pbnQxLCBDb2xvciBjb2xvcjEsIFBvaW50MkQgcG9pbnQyLCBDb2xvciBjb2xvcjIsIGJvb2xlYW4gY3ljbGljKSB7Ci0gICAgICAgIGlmIChwb2ludDEgPT0gbnVsbCB8fCBwb2ludDIgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjZEPVBvaW50IGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42RCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGlmIChjb2xvcjEgPT0gbnVsbCB8fCBjb2xvcjIgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjZFPUNvbG9yIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42RSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5wb2ludDEgPSBwb2ludDE7Ci0gICAgICAgIHRoaXMucG9pbnQyID0gcG9pbnQyOwotICAgICAgICB0aGlzLmNvbG9yMSA9IGNvbG9yMTsKLSAgICAgICAgdGhpcy5jb2xvcjIgPSBjb2xvcjI7Ci0gICAgICAgIHRoaXMuY3ljbGljID0gY3ljbGljOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBHcmFkaWVudFBhaW50IHdpdGggY3ljbGljIG9yIGFjeWNsaWMgbW9kZTsgcG9pbnRzIGFyZQotICAgICAqIHNwZWNpZmllZCBieSBjb29yZGluYXRlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IHBvaW50LgotICAgICAqIEBwYXJhbSB5MQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgcG9pbnQuCi0gICAgICogQHBhcmFtIGNvbG9yMQotICAgICAqICAgICAgICAgICAgdGhlIGNvbG9yIG9mIHRoZSBmaXJzdCBwb2ludC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY29sb3IyCi0gICAgICogICAgICAgICAgICB0aGUgY29sb3Igb2YgdGhlIHNlY29uZCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY3ljbGljCi0gICAgICogICAgICAgICAgICB0aGUgY3ljbGljIG1vZGUgLSB0cnVlIGlmIHRoZSBncmFkaWVudCBwYXR0ZXJuIHNob3VsZCBjeWNsZQotICAgICAqICAgICAgICAgICAgcmVwZWF0ZWRseSBiZXR3ZWVuIHRoZSB0d28gY29sb3JzOyBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIEdyYWRpZW50UGFpbnQoZmxvYXQgeDEsIGZsb2F0IHkxLCBDb2xvciBjb2xvcjEsIGZsb2F0IHgyLCBmbG9hdCB5MiwgQ29sb3IgY29sb3IyLAotICAgICAgICAgICAgYm9vbGVhbiBjeWNsaWMpIHsKLSAgICAgICAgdGhpcyhuZXcgUG9pbnQyRC5GbG9hdCh4MSwgeTEpLCBjb2xvcjEsIG5ldyBQb2ludDJELkZsb2F0KHgyLCB5MiksIGNvbG9yMiwgY3ljbGljKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgYWN5Y2xpYyBHcmFkaWVudFBhaW50OyBwb2ludHMgYXJlIHNwZWNpZmllZCBieQotICAgICAqIGNvb3JkaW5hdGVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgcG9pbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY29sb3IxCi0gICAgICogICAgICAgICAgICB0aGUgY29sb3Igb2YgdGhlIGZpcnN0IHBvaW50LgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIHBvaW50LgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIHBvaW50LgotICAgICAqIEBwYXJhbSBjb2xvcjIKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb2xvciBvZiB0aGUgc2Vjb25kIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBHcmFkaWVudFBhaW50KGZsb2F0IHgxLCBmbG9hdCB5MSwgQ29sb3IgY29sb3IxLCBmbG9hdCB4MiwgZmxvYXQgeTIsIENvbG9yIGNvbG9yMikgewotICAgICAgICB0aGlzKHgxLCB5MSwgY29sb3IxLCB4MiwgeTIsIGNvbG9yMiwgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBhY3ljbGljIEdyYWRpZW50UGFpbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBvaW50MQotICAgICAqICAgICAgICAgICAgdGhlIGZpcnN0IHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY29sb3IxCi0gICAgICogICAgICAgICAgICB0aGUgQ29sb3Igb2YgdGhlIGZpcnN0IHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiBAcGFyYW0gcG9pbnQyCi0gICAgICogICAgICAgICAgICB0aGUgc2Vjb25kIHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY29sb3IyCi0gICAgICogICAgICAgICAgICB0aGUgQ29sb3Igb2YgdGhlIHNlY29uZCBzcGVjaWZpZWQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIEdyYWRpZW50UGFpbnQoUG9pbnQyRCBwb2ludDEsIENvbG9yIGNvbG9yMSwgUG9pbnQyRCBwb2ludDIsIENvbG9yIGNvbG9yMikgewotICAgICAgICB0aGlzKHBvaW50MSwgY29sb3IxLCBwb2ludDIsIGNvbG9yMiwgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgUGFpbnRDb250ZXh0IGZvciBhIGNvbG9yIHBhdHRlcm4gZ2VuZXJhdGluZy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY20KLSAgICAgKiAgICAgICAgICAgIHRoZSBDb2xvck1vZGVsIG9mIHRoZSBQYWludCBkYXRhLgotICAgICAqIEBwYXJhbSBkZXZpY2VCb3VuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBib3VuZGluZyBSZWN0YW5nbGUgb2YgZ3JhcGhpY3MgcHJpbWl0aXZlcyBiZWluZyByZW5kZXJlZAotICAgICAqICAgICAgICAgICAgaW4gdGhlIGRldmljZSBzcGFjZS4KLSAgICAgKiBAcGFyYW0gdXNlckJvdW5kcwotICAgICAqICAgICAgICAgICAgdGhlIGJvdW5kaW5nIFJlY3RhbmdsZSBvZiBncmFwaGljcyBwcmltaXRpdmVzIGJlaW5nIHJlbmRlcmVkCi0gICAgICogICAgICAgICAgICBpbiB0aGUgdXNlciBzcGFjZS4KLSAgICAgKiBAcGFyYW0gdAotICAgICAqICAgICAgICAgICAgdGhlIEFmZmluZVRyYW5zZm9ybSBmcm9tIHVzZXIgc3BhY2UgaW50byBkZXZpY2Ugc3BhY2UuCi0gICAgICogQHBhcmFtIGhpbnRzCi0gICAgICogICAgICAgICAgICB0aGUgUnJlbmRlcmluZ0hpbnRzIG9iamVjdC4KLSAgICAgKiBAcmV0dXJuIHRoZSBQYWludENvbnRleHQgZm9yIGNvbG9yIHBhdHRlcm4gZ2VuZXJhdGluZy4KLSAgICAgKiBAc2VlIGphdmEuYXd0LlBhaW50I2NyZWF0ZUNvbnRleHQoamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbCwKLSAgICAgKiAgICAgIGphdmEuYXd0LlJlY3RhbmdsZSwgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRCwKLSAgICAgKiAgICAgIGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtLCBqYXZhLmF3dC5SZW5kZXJpbmdIaW50cykKLSAgICAgKi8KLSAgICBwdWJsaWMgUGFpbnRDb250ZXh0IGNyZWF0ZUNvbnRleHQoQ29sb3JNb2RlbCBjbSwgUmVjdGFuZ2xlIGRldmljZUJvdW5kcywKLSAgICAgICAgICAgIFJlY3RhbmdsZTJEIHVzZXJCb3VuZHMsIEFmZmluZVRyYW5zZm9ybSB0LCBSZW5kZXJpbmdIaW50cyBoaW50cykgewotICAgICAgICByZXR1cm4gbmV3IEdyYWRpZW50UGFpbnRDb250ZXh0KGNtLCB0LCBwb2ludDEsIGNvbG9yMSwgcG9pbnQyLCBjb2xvcjIsIGN5Y2xpYyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29sb3Igb2YgdGhlIGZpcnN0IHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGNvbG9yIG9mIHRoZSBmaXJzdCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3IgZ2V0Q29sb3IxKCkgewotICAgICAgICByZXR1cm4gY29sb3IxOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvbG9yIG9mIHRoZSBzZWNvbmQgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY29sb3Igb2YgdGhlIHNlY29uZCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3IgZ2V0Q29sb3IyKCkgewotICAgICAgICByZXR1cm4gY29sb3IyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZpcnN0IHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFBvaW50IG9iamVjdCAtIHRoZSBmaXJzdCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUG9pbnQyRCBnZXRQb2ludDEoKSB7Ci0gICAgICAgIHJldHVybiBwb2ludDE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2Vjb25kIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFBvaW50IG9iamVjdCAtIHRoZSBzZWNvbmQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIFBvaW50MkQgZ2V0UG9pbnQyKCkgewotICAgICAgICByZXR1cm4gcG9pbnQyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRyYW5zcGFyZW5jeSBtb2RlIGZvciB0aGUgR3JhZGllbnRQYWludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB0cmFuc3BhcmVuY3kgbW9kZSBmb3IgdGhlIEdyYWRpZW50UGFpbnQuCi0gICAgICogQHNlZSBqYXZhLmF3dC5UcmFuc3BhcmVuY3kjZ2V0VHJhbnNwYXJlbmN5KCkKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFRyYW5zcGFyZW5jeSgpIHsKLSAgICAgICAgaW50IGExID0gY29sb3IxLmdldEFscGhhKCk7Ci0gICAgICAgIGludCBhMiA9IGNvbG9yMi5nZXRBbHBoYSgpOwotICAgICAgICByZXR1cm4gKGExID09IDB4RkYgJiYgYTIgPT0gMHhGRikgPyBPUEFRVUUgOiBUUkFOU0xVQ0VOVDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBHcmFkaWVudFBhaW50IG1vZGU6IHRydWUgZm9yIGN5Y2xpYyBtb2RlLCBmYWxzZSBmb3IgYWN5Y2xpYwotICAgICAqIG1vZGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlIGlmIHRoZSBncmFkaWVudCBjeWNsZXMgcmVwZWF0ZWRseSBiZXR3ZWVuIHRoZSB0d28gY29sb3JzOwotICAgICAqICAgICAgICAgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzQ3ljbGljKCkgewotICAgICAgICByZXR1cm4gY3ljbGljOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9HcmFkaWVudFBhaW50Q29udGV4dC5qYXZhIGIvYXd0L2phdmEvYXd0L0dyYWRpZW50UGFpbnRDb250ZXh0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDc0NTc1ZjUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0dyYWRpZW50UGFpbnRDb250ZXh0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMDQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuZ2VvbS5BZmZpbmVUcmFuc2Zvcm07Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5Qb2ludDJEOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGF0YUJ1ZmZlckludDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5SYXN0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuV3JpdGFibGVSYXN0ZXI7Ci0KLWNsYXNzIEdyYWRpZW50UGFpbnRDb250ZXh0IGltcGxlbWVudHMgUGFpbnRDb250ZXh0IHsKLQotICAgIC8qKgotICAgICAqIFRoZSBzaXplIG9mIG5vbmN5Y2xpYyBwYXJ0IG9mIGNvbG9yIGxvb2t1cCB0YWJsZQotICAgICAqLwotICAgIHN0YXRpYyBpbnQgTE9PS1VQX1NJWkUgPSAyNTY7Ci0gICAgCi0gICAgLyoqCi0gICAgICogVGhlIGluZGV4IG1hc2sgdG8gbG9va3VwIGNvbG9yIGluIHRoZSB0YWJsZQotICAgICAqLwotICAgIHN0YXRpYyBpbnQgTE9PS1VQX01BU0sgPSAweDFGRjsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgbWluIHZhbHVlIGVxdWl2YWxlbnQgdG8gemVyby4gSWYgYWJzb2x1dGUgdmFsdWUgbGVzcyB0aGVuIFpFUk8gaXQgY29uc2lkZXJlZCBhcyB6ZXJvLiAgCi0gICAgICovCi0gICAgc3RhdGljIGRvdWJsZSBaRVJPID0gMUUtMTA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29sb3JNb2RlbCB1c2VyIGRlZmluZWQgZm9yIFBhaW50Q29udGV4dAotICAgICAqLwotICAgIENvbG9yTW9kZWwgY207Ci0gICAgCi0gICAgLyoqCi0gICAgICogVGhlIGluZGljYXRvciBvZiBjeWNsZSBmaWxsaW5nLgotICAgICAqLwotICAgIGJvb2xlYW4gY3ljbGljOwotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBpbnRlZ2VyIGNvbG9yIHZhbHVlIG9mIHRoZSBzdGFydCBwb2ludAotICAgICAqLwotICAgIGludCBjMTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgaW50ZWdlciBjb2xvciB2YWx1ZSBvZiB0aGUgZW5kIHBvaW50Ci0gICAgICovCi0gICAgaW50IGMyOwotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBsb29rdXAgZ3JhZGllbnQgY29sb3IgdGFibGUgCi0gICAgICovCi0gICAgaW50W10gdGFibGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdGVtcG9wYXJ5IHByZS1jYWxjdWxhdGVkIHZhbHVlIHRvIGV2YWx1dGFlIGNvbG9yIGluZGV4IAotICAgICAqLwotICAgIGludCBkeDsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgdGVtcG9wYXJ5IHByZS1jYWxjdWxhdGVkIHZhbHVlIHRvIGV2YWx1dGFlIGNvbG9yIGluZGV4IAotICAgICAqLwotICAgIGludCBkeTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgdGVtcG9wYXJ5IHByZS1jYWxjdWxhdGVkIHZhbHVlIHRvIGV2YWx1dGFlIGNvbG9yIGluZGV4IAotICAgICAqLwotICAgIGludCBkZWx0YTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBDb25zdHJ1Y3RzIGEgbmV3IEdyYWRpZW50UGFpbnRjb250ZXh0Ci0gICAgICogQHBhcmFtIGNtIC0gbm90IHVzZWQKLSAgICAgKiBAcGFyYW0gdCAtIHRoZSBmaWxsIHRyYW5zZm9ybWF0aW9uCi0gICAgICogQHBhcmFtIHBvaW50MSAtIHRoZSBzdGFydCBmaWxsIHBvaW50Ci0gICAgICogQHBhcmFtIGNvbG9yMSAtIGNvbG9yIG9mIHRoZSBzdGFydCBwb2ludCAKLSAgICAgKiBAcGFyYW0gcG9pbnQyIC0gdGhlIGVuZCBmaWxsIHBvaW50Ci0gICAgICogQHBhcmFtIGNvbG9yMiAtIGNvbG9yIG9mIHRoZSBlbmQgcG9pbnQKLSAgICAgKiBAcGFyYW0gY3ljbGljIC0gdGhlIGluZGljYXRvciBvZiBjeWNsZSBmaWxsaW5nCi0gICAgICovCi0gICAgR3JhZGllbnRQYWludENvbnRleHQoQ29sb3JNb2RlbCBjbSwgQWZmaW5lVHJhbnNmb3JtIHQsIFBvaW50MkQgcG9pbnQxLCBDb2xvciBjb2xvcjEsIFBvaW50MkQgcG9pbnQyLCBDb2xvciBjb2xvcjIsIGJvb2xlYW4gY3ljbGljKSB7Ci0gICAgICAgIHRoaXMuY3ljbGljID0gY3ljbGljOwotICAgICAgICB0aGlzLmNtID0gQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCk7Ci0KLSAgICAgICAgYzEgPSBjb2xvcjEuZ2V0UkdCKCk7Ci0gICAgICAgIGMyID0gY29sb3IyLmdldFJHQigpOwotCi0gICAgICAgIGRvdWJsZSBweCA9IHBvaW50Mi5nZXRYKCkgLSBwb2ludDEuZ2V0WCgpOwotICAgICAgICBkb3VibGUgcHkgPSBwb2ludDIuZ2V0WSgpIC0gcG9pbnQxLmdldFkoKTsKLQotICAgICAgICBQb2ludDJEIHAgPSB0LnRyYW5zZm9ybShwb2ludDEsIG51bGwpOwotICAgICAgICBQb2ludDJEIGJ4ID0gbmV3IFBvaW50MkQuRG91YmxlKHB4LCBweSk7Ci0gICAgICAgIFBvaW50MkQgYnkgPSBuZXcgUG9pbnQyRC5Eb3VibGUocHksIC1weCk7Ci0KLSAgICAgICAgdC5kZWx0YVRyYW5zZm9ybShieCwgYngpOwotICAgICAgICB0LmRlbHRhVHJhbnNmb3JtKGJ5LCBieSk7Ci0KLSAgICAgICAgZG91YmxlIHZlYyA9IGJ4LmdldFgoKSAqIGJ5LmdldFkoKSAtIGJ4LmdldFkoKSAqIGJ5LmdldFgoKTsKLQotICAgICAgICBpZiAoTWF0aC5hYnModmVjKSA8IFpFUk8pIHsKLSAgICAgICAgICAgIGR4ID0gZHkgPSBkZWx0YSA9IDA7Ci0gICAgICAgICAgICB0YWJsZSA9IG5ldyBpbnRbMV07Ci0gICAgICAgICAgICB0YWJsZVswXSA9IGMxOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZG91YmxlIG11bHQgPSBMT09LVVBfU0laRSAqIDI1NiAvIHZlYzsKLSAgICAgICAgICAgIGR4ID0gKGludCkoYnkuZ2V0WCgpICogbXVsdCk7Ci0gICAgICAgICAgICBkeSA9IChpbnQpKGJ5LmdldFkoKSAqIG11bHQpOwotICAgICAgICAgICAgZGVsdGEgPSAoaW50KSgocC5nZXRYKCkgKiBieS5nZXRZKCkgLSBwLmdldFkoKSAqIGJ5LmdldFgoKSkgKiBtdWx0KTsKLSAgICAgICAgICAgIGNyZWF0ZVRhYmxlKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGUgY29sb3IgaW5kZXggbG9va3VwIHRhYmxlLiBDYWxjdWxhdGUgMjU2IHN0ZXAgdHJhc2Zvcm1hdGlvbiBmcm9tIAotICAgICAqIHRoZSBzdGFydCBwb2ludCBjb2xvciB0byB0aGUgZW5kIHBvaW50IGNvbG9yLiBDb2xvcnMgbXVsdGlwbGllZCBieSAyNTYgdG8gZG8gaW50ZWdlciBjYWxjdWxhdGlvbnMuIAotICAgICAqLwotICAgIHZvaWQgY3JlYXRlVGFibGUoKSB7Ci0gICAgICAgIGRvdWJsZSBjYSA9IChjMSA+PiAyNCkgJiAweEZGOwotICAgICAgICBkb3VibGUgY3IgPSAoYzEgPj4gMTYpICYgMHhGRjsKLSAgICAgICAgZG91YmxlIGNnID0gKGMxID4+IDgpICYgMHhGRjsKLSAgICAgICAgZG91YmxlIGNiID0gYzEgJiAweEZGOwotCi0gICAgICAgIGRvdWJsZSBrID0gMS4wIC8gTE9PS1VQX1NJWkU7Ci0gICAgICAgIGRvdWJsZSBkYSA9ICgoKGMyID4+IDI0KSAmIDB4RkYpIC0gY2EpICogazsKLSAgICAgICAgZG91YmxlIGRyID0gKCgoYzIgPj4gMTYpICYgMHhGRikgLSBjcikgKiBrOwotICAgICAgICBkb3VibGUgZGcgPSAoKChjMiA+PiA4KSAmIDB4RkYpIC0gY2cpICogazsKLSAgICAgICAgZG91YmxlIGRiID0gKChjMiAmIDB4RkYpIC0gY2IpICogazsKLQotICAgICAgICB0YWJsZSA9IG5ldyBpbnRbY3ljbGljID8gTE9PS1VQX1NJWkUgKyBMT09LVVBfU0laRSA6IExPT0tVUF9TSVpFXTsKLSAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IExPT0tVUF9TSVpFOyBpKyspIHsKLSAgICAgICAgICAgIHRhYmxlW2ldID0KLSAgICAgICAgICAgICAgICAoaW50KWNhIDw8IDI0IHwKLSAgICAgICAgICAgICAgICAoaW50KWNyIDw8IDE2IHwKLSAgICAgICAgICAgICAgICAoaW50KWNnIDw8IDggfAotICAgICAgICAgICAgICAgIChpbnQpY2I7Ci0gICAgICAgICAgICBjYSArPSBkYTsKLSAgICAgICAgICAgIGNyICs9IGRyOwotICAgICAgICAgICAgY2cgKz0gZGc7Ci0gICAgICAgICAgICBjYiArPSBkYjsKLSAgICAgICAgfQotICAgICAgICBpZiAoY3ljbGljKSB7Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgTE9PS1VQX1NJWkU7IGkrKykgewotICAgICAgICAgICAgICAgIHRhYmxlW0xPT0tVUF9TSVpFICsgTE9PS1VQX1NJWkUgLSAxIC0gaV0gPSB0YWJsZVtpXTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoKSB7Ci0gICAgICAgIHJldHVybiBjbTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBkaXNwb3NlKCkgewotICAgIH0KLQotICAgIHB1YmxpYyBSYXN0ZXIgZ2V0UmFzdGVyKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoKSB7Ci0gICAgICAgIFdyaXRhYmxlUmFzdGVyIHJhc3QgPSBjbS5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIodywgaCk7Ci0KLSAgICAgICAgaW50W10gYnVmID0gKChEYXRhQnVmZmVySW50KXJhc3QuZ2V0RGF0YUJ1ZmZlcigpKS5nZXREYXRhKCk7Ci0KLSAgICAgICAgaW50IGMgPSB4ICogZHkgLSB5ICogZHggLSBkZWx0YTsKLSAgICAgICAgaW50IGN4ID0gZHk7Ci0gICAgICAgIGludCBjeSA9IC0gdyAqIGR5IC0gZHg7Ci0gICAgICAgIGludCBrID0gMDsKLQotICAgICAgICBpZiAoY3ljbGljKSB7Ci0gICAgICAgICAgICBmb3IoaW50IGogPSAwOyBqIDwgaDsgaisrKSB7Ci0gICAgICAgICAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IHc7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBidWZbaysrXSA9IHRhYmxlWyhjID4+IDgpICYgTE9PS1VQX01BU0tdOwotICAgICAgICAgICAgICAgICAgICBjICs9IGN4OwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBjICs9IGN5OwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZm9yKGludCBqID0gMDsgaiA8IGg7IGorKykgewotICAgICAgICAgICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCB3OyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IGluZGV4ID0gYyA+PiA4OwotICAgICAgICAgICAgICAgICAgICBidWZbaysrXSA9IGluZGV4IDwgMCA/IGMxIDogaW5kZXggPj0gTE9PS1VQX1NJWkUgPyBjMiA6IHRhYmxlW2luZGV4XTsKLSAgICAgICAgICAgICAgICAgICAgYyArPSBjeDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYyArPSBjeTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiByYXN0OwotICAgIH0KLQotfQotCmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvR3JhcGhpY3MuamF2YSBiL2F3dC9qYXZhL2F3dC9HcmFwaGljcy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyZDZlNzlmLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9HcmFwaGljcy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsOTI0ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgQWxleGV5IEEuIFBldHJlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEuYXd0LmltYWdlLkltYWdlT2JzZXJ2ZXI7Ci1pbXBvcnQgamF2YS50ZXh0LkF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvcjsKLQotLyoqCi0gKiBUaGUgYWJzdHJhY3QgR3JhcGhpY3MgY2xhc3MgYWxsb3dzIGFwcGxpY2F0aW9ucyB0byBkcmF3IG9uIGEgc2NyZWVuIG9yIG90aGVyCi0gKiByZW5kZXJpbmcgdGFyZ2V0LiBUaGVyZSBhcmUgc2V2ZXJhbCBwcm9wZXJ0aWVzIHdoaWNoIGRlZmluZSByZW5kZXJpbmcKLSAqIG9wdGlvbnM6IG9yaWdpbiBwb2ludCwgY2xpcHBpbmcgYXJlYSwgY29sb3IsIGZvbnQuIDxicj4KLSAqIDxicj4KLSAqIFRoZSBvcmlnaW4gcG9pbnQgc3BlY2lmaWVzIHRoZSBiZWdpbm5pbmcgb2YgdGhlIGNsaXBwaW5nIGFyZWEgY29vcmRpbmF0ZQotICogc3lzdGVtLiBBbGwgY29vcmRpbmF0ZXMgdXNlZCBpbiByZW5kZXJpbmcgb3BlcmF0aW9ucyBhcmUgY29tcHV0ZWQgd2l0aAotICogcmVzcGVjdCB0byB0aGlzIHBvaW50LiBUaGUgY2xpcHBpbmcgYXJlYSBkZWZpbmVzIHRoZSBib3VuZGFyaWVzIHdoZXJlCi0gKiByZW5kZXJpbmcgb3BlcmF0aW9ucyBjYW4gYmUgcGVyZm9ybWVkLiBSZW5kZXJpbmcgb3BlcmF0aW9ucyBjYW4ndCBtb2RpZnkKLSAqIHBpeGVscyBvdXRzaWRlIG9mIHRoZSBjbGlwcGluZyBhcmVhLiA8YnI+Ci0gKiA8YnI+Ci0gKiBUaGUgZHJhdyBhbmQgZmlsbCBtZXRob2RzIGFsbG93IGFwcGxpY2F0aW9ucyB0byBkcmF3aW5nIHNoYXBlcywgdGV4dCwgaW1hZ2VzCi0gKiB3aXRoIHNwZWNpZmllZCBmb250IGFuZCBjb2xvciBvcHRpb25zIGluIHRoZSBzcGVjaWZpZWQgcGFydCBvZiB0aGUgc2NyZWVuLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEdyYXBoaWNzIHsKLQotICAgIC8vIENvbnN0cnVjdG9ycwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEdyYXBoaWNzLiBUaGlzIGNvbnN0cnVjdG9yIGlzIGRlZmF1bHQgZm9yIEdyYXBoaWNzIGFuZAotICAgICAqIGNhbiBub3QgYmUgY2FsbGVkIGRpcmVjdGx5LgotICAgICAqLwotICAgIHByb3RlY3RlZCBHcmFwaGljcygpIHsKLSAgICB9Ci0KLSAgICAvLyBQdWJsaWMgbWV0aG9kcwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIGNvcHkgb2YgdGhlIEdyYXBoaWNzIG9iamVjdCB3aXRoIGEgbmV3IG9yaWdpbiBhbmQgYSBuZXcKLSAgICAgKiBzcGVjaWZpZWQgY2xpcCBhcmVhLiBUaGUgbmV3IGNsaXAgYXJlYSBpcyB0aGUgcmVjdGFuZ2xlIGRlZmluZWQgYnkgdGhlCi0gICAgICogb3JpZ2luIHBvaW50IHdpdGggY29vcmRpbmF0ZXMgWCxZIGFuZCB0aGUgZ2l2ZW4gd2lkdGggYW5kIGhlaWdodC4gVGhlCi0gICAgICogY29vcmRpbmF0ZXMgb2YgYWxsIHN1YnNlcXVlbnQgcmVuZGVyaW5nIG9wZXJhdGlvbnMgd2lsbCBiZSBjb21wdXRlZCB3aXRoCi0gICAgICogcmVzcGVjdCB0byB0aGUgbmV3IG9yaWdpbiBhbmQgY2FuIGJlIHBlcmZvcm1lZCBvbmx5IHdpdGhpbiB0aGUgcmFuZ2Ugb2YKLSAgICAgKiB0aGUgY2xpcHBpbmcgYXJlYSBkaW1lbnNpb25zLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBvcmlnaW5hbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgb3JpZ2luYWwgcG9pbnQuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgY2xpcHBpbmcgYXJlYS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIGNsaXBwaW5nIGFyZWEuCi0gICAgICogQHJldHVybiB0aGUgR3JhcGhpY3Mgb2JqZWN0IHdpdGggbmV3IG9yaWdpbiBwb2ludCBhbmQgY2xpcHBpbmcgYXJlYS4KLSAgICAgKi8KLSAgICBwdWJsaWMgR3JhcGhpY3MgY3JlYXRlKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIEdyYXBoaWNzIHJlcyA9IGNyZWF0ZSgpOwotICAgICAgICByZXMudHJhbnNsYXRlKHgsIHkpOwotICAgICAgICByZXMuY2xpcFJlY3QoMCwgMCwgd2lkdGgsIGhlaWdodCk7Ci0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRHJhd3MgdGhlIGhpZ2hsaWdodGVkIG91dGxpbmUgb2YgYSByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHRvcCBsZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdG9wIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gcmFpc2VkCi0gICAgICogICAgICAgICAgICBhIGJvb2xlYW4gdmFsdWUgdGhhdCBkZXRlcm1pbmVzIHdoZXRoZXIgdGhlIHJlY3RhbmdsZSBpcyBkcmF3bgotICAgICAqICAgICAgICAgICAgYXMgcmFpc2VkIG9yIGluZGVudGVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGRyYXczRFJlY3QoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQsIGJvb2xlYW4gcmFpc2VkKSB7Ci0gICAgICAgIC8vIE5vdGU6IGxpZ2h0ZXIvZGFya2VyIGNvbG9ycyBzaG91bGQgYmUgdXNlZCB0byBkcmF3IDNkIHJlY3QuCi0gICAgICAgIC8vIFRoZSByZXN1bHRpbmcgcmVjdCBpcyAod2lkdGgrMSl4KGhlaWdodCsxKS4gU3Ryb2tlIGFuZCBwYWludAotICAgICAgICAvLyBhdHRyaWJ1dGVzIG9mCi0gICAgICAgIC8vIHRoZSBHcmFwaGljczJEIHNob3VsZCBiZSByZXNldCB0byB0aGUgZGVmYXVsdCB2YWx1ZXMuCi0gICAgICAgIC8vIGZpbGxSZWN0IGlzIHVzZWQgaW5zdGVhZCBvZiBkcmF3TGluZSB0byBieXBhc3Mgc3Ryb2tlCi0gICAgICAgIC8vIHJlc2V0L3NldCBhbmQgcmFzdGVyaXphdGlvbi4KLQotICAgICAgICBDb2xvciBjb2xvciA9IGdldENvbG9yKCk7Ci0gICAgICAgIENvbG9yIGNvbG9yVXAsIGNvbG9yRG93bjsKLSAgICAgICAgaWYgKHJhaXNlZCkgewotICAgICAgICAgICAgY29sb3JVcCA9IGNvbG9yLmJyaWdodGVyKCk7Ci0gICAgICAgICAgICBjb2xvckRvd24gPSBjb2xvci5kYXJrZXIoKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbG9yVXAgPSBjb2xvci5kYXJrZXIoKTsKLSAgICAgICAgICAgIGNvbG9yRG93biA9IGNvbG9yLmJyaWdodGVyKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBzZXRDb2xvcihjb2xvclVwKTsKLSAgICAgICAgZmlsbFJlY3QoeCwgeSwgd2lkdGgsIDEpOwotICAgICAgICBmaWxsUmVjdCh4LCB5ICsgMSwgMSwgaGVpZ2h0KTsKLQotICAgICAgICBzZXRDb2xvcihjb2xvckRvd24pOwotICAgICAgICBmaWxsUmVjdCh4ICsgd2lkdGgsIHksIDEsIGhlaWdodCk7Ci0gICAgICAgIGZpbGxSZWN0KHggKyAxLCB5ICsgaGVpZ2h0LCB3aWR0aCwgMSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRHJhd3MgdGhlIHRleHQgcmVwcmVzZW50ZWQgYnkgYnl0ZSBhcnJheS4gVGhpcyBtZXRob2QgdXNlcyB0aGUgY3VycmVudAotICAgICAqIGZvbnQgYW5kIGNvbG9yIGZvciByZW5kZXJpbmcuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJ5dGVzCi0gICAgICogICAgICAgICAgICB0aGUgYnl0ZSBhcnJheSB3aGljaCBjb250YWlucyB0aGUgdGV4dCB0byBiZSBkcmF3bi4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IHdpdGhpbiB0aGUgYnl0ZSBhcnJheSBvZiB0aGUgdGV4dCB0byBiZSBkcmF3bi4KLSAgICAgKiBAcGFyYW0gbGVuCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGJ5dGVzIG9mIHRleHQgdG8gZHJhdy4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSB3aGVyZSB0aGUgdGV4dCBpcyB0byBiZSBkcmF3bi4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSB3aGVyZSB0aGUgdGV4dCBpcyB0byBiZSBkcmF3bi4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBkcmF3Qnl0ZXMoYnl0ZVtdIGJ5dGVzLCBpbnQgb2ZmLCBpbnQgbGVuLCBpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgZHJhd1N0cmluZyhuZXcgU3RyaW5nKGJ5dGVzLCBvZmYsIGxlbiksIHgsIHkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERyYXdzIHRoZSB0ZXh0IHJlcHJlc2VudGVkIGJ5IGNoYXJhY3RlciBhcnJheS4gVGhpcyBtZXRob2QgdXNlcyB0aGUKLSAgICAgKiBjdXJyZW50IGZvbnQgYW5kIGNvbG9yIGZvciByZW5kZXJpbmcuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNoYXJzCi0gICAgICogICAgICAgICAgICB0aGUgY2hhcmFjdGVyIGFycmF5LgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgd2l0aGluIHRoZSBjaGFyYWN0ZXIgYXJyYXkgb2YgdGhlIHRleHQgdG8gYmUgZHJhd24uCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBjaGFyYWN0ZXJzIHdoaWNoIHdpbGwgYmUgZHJhd24uCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgd2hlcmUgdGhlIHRleHQgaXMgdG8gYmUgZHJhd24uCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgd2hlcmUgdGhlIHRleHQgaXMgdG8gYmUgZHJhd24uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZHJhd0NoYXJzKGNoYXJbXSBjaGFycywgaW50IG9mZiwgaW50IGxlbiwgaW50IHgsIGludCB5KSB7Ci0gICAgICAgIGRyYXdTdHJpbmcobmV3IFN0cmluZyhjaGFycywgb2ZmLCBsZW4pLCB4LCB5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyB0aGUgb3V0bGluZSBvZiBhIHBvbHlnb24gd2hpY2ggaXMgZGVmaW5lZCBieSBQb2x5Z29uIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIFBvbHlnb24gb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGRyYXdQb2x5Z29uKFBvbHlnb24gcCkgewotICAgICAgICBkcmF3UG9seWdvbihwLnhwb2ludHMsIHAueXBvaW50cywgcC5ucG9pbnRzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyB0aGUgcmVjdGFuZ2xlIHdpdGggdGhlIHNwZWNpZmllZCB3aWR0aCBhbmQgbGVuZ3RoIGFuZCB0b3AgbGVmdAotICAgICAqIGNvcm5lciBjb29yZGluYXRlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdG9wIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB0b3AgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZHJhd1JlY3QoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgaW50W10geHBvaW50cyA9IHsKLSAgICAgICAgICAgICAgICB4LCB4LCB4ICsgd2lkdGgsIHggKyB3aWR0aAotICAgICAgICB9OwotICAgICAgICBpbnRbXSB5cG9pbnRzID0gewotICAgICAgICAgICAgICAgIHksIHkgKyBoZWlnaHQsIHkgKyBoZWlnaHQsIHkKLSAgICAgICAgfTsKLQotICAgICAgICBkcmF3UG9seWdvbih4cG9pbnRzLCB5cG9pbnRzLCA0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaWxscyB0aGUgaGlnaGxpZ2h0ZWQgb3V0bGluZSBvZiBhIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdG9wIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB0b3AgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHJhaXNlZAotICAgICAqICAgICAgICAgICAgYSBib29sZWFuIHZhbHVlIHRoYXQgZGV0ZXJtaW5lcyB3aGV0aGVyIHRoZSByZWN0YW5nbGUgaXMgZHJhd24KLSAgICAgKiAgICAgICAgICAgIGFzIHJhaXNlZCBvciBpbmRlbnRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBmaWxsM0RSZWN0KGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBib29sZWFuIHJhaXNlZCkgewotICAgICAgICAvLyBOb3RlOiBsaWdodGVyL2RhcmtlciBjb2xvcnMgc2hvdWxkIGJlIHVzZWQgdG8gZHJhdyAzZCByZWN0LgotICAgICAgICAvLyBUaGUgcmVzdWx0aW5nIHJlY3QgaXMgKHdpZHRoKXgoaGVpZ2h0KSwgc2FtZSBhcyBmaWxsUmVjdC4KLSAgICAgICAgLy8gU3Ryb2tlIGFuZCBwYWludCBhdHRyaWJ1dGVzIG9mIHRoZSBHcmFwaGljczJEIHNob3VsZCBiZSByZXNldAotICAgICAgICAvLyB0byB0aGUgZGVmYXVsdCB2YWx1ZXMuIGZpbGxSZWN0IGlzIHVzZWQgaW5zdGVhZCBvZiBkcmF3TGluZSB0bwotICAgICAgICAvLyBieXBhc3Mgc3Ryb2tlIHJlc2V0L3NldCBhbmQgbGluZSByYXN0ZXJpemF0aW9uLgotCi0gICAgICAgIENvbG9yIGNvbG9yID0gZ2V0Q29sb3IoKTsKLSAgICAgICAgQ29sb3IgY29sb3JVcCwgY29sb3JEb3duOwotICAgICAgICBpZiAocmFpc2VkKSB7Ci0gICAgICAgICAgICBjb2xvclVwID0gY29sb3IuYnJpZ2h0ZXIoKTsKLSAgICAgICAgICAgIGNvbG9yRG93biA9IGNvbG9yLmRhcmtlcigpOwotICAgICAgICAgICAgc2V0Q29sb3IoY29sb3IpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgY29sb3JVcCA9IGNvbG9yLmRhcmtlcigpOwotICAgICAgICAgICAgY29sb3JEb3duID0gY29sb3IuYnJpZ2h0ZXIoKTsKLSAgICAgICAgICAgIHNldENvbG9yKGNvbG9yVXApOwotICAgICAgICB9Ci0KLSAgICAgICAgd2lkdGgtLTsKLSAgICAgICAgaGVpZ2h0LS07Ci0gICAgICAgIGZpbGxSZWN0KHggKyAxLCB5ICsgMSwgd2lkdGggLSAxLCBoZWlnaHQgLSAxKTsKLQotICAgICAgICBzZXRDb2xvcihjb2xvclVwKTsKLSAgICAgICAgZmlsbFJlY3QoeCwgeSwgd2lkdGgsIDEpOwotICAgICAgICBmaWxsUmVjdCh4LCB5ICsgMSwgMSwgaGVpZ2h0KTsKLQotICAgICAgICBzZXRDb2xvcihjb2xvckRvd24pOwotICAgICAgICBmaWxsUmVjdCh4ICsgd2lkdGgsIHksIDEsIGhlaWdodCk7Ci0gICAgICAgIGZpbGxSZWN0KHggKyAxLCB5ICsgaGVpZ2h0LCB3aWR0aCwgMSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRmlsbHMgdGhlIHBvbHlnb24gd2l0aCB0aGUgY3VycmVudCBjb2xvci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIFBvbHlnb24gb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGZpbGxQb2x5Z29uKFBvbHlnb24gcCkgewotICAgICAgICBmaWxsUG9seWdvbihwLnhwb2ludHMsIHAueXBvaW50cywgcC5ucG9pbnRzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEaXNwb3NlcyBvZiB0aGUgR3JhcGhpY3MuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZmluYWxpemUoKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYm91bmRzIG9mIHRoZSBjdXJyZW50IGNsaXBwaW5nIGFyZWEgYXMgYSByZWN0YW5nbGUgYW5kIGNvcGllcyBpdAotICAgICAqIHRvIGFuIGV4aXN0aW5nIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcgotICAgICAqICAgICAgICAgICAgYSBSZWN0YW5nbGUgb2JqZWN0IHdoZXJlIHRoZSBjdXJyZW50IGNsaXBwaW5nIGFyZWEgYm91bmRzIGFyZQotICAgICAqICAgICAgICAgICAgdG8gYmUgY29waWVkLgotICAgICAqIEByZXR1cm4gdGhlIGJvdW5kcyBvZiB0aGUgY3VycmVudCBjbGlwcGluZyBhcmVhLgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Q2xpcEJvdW5kcyhSZWN0YW5nbGUgcikgewotICAgICAgICBTaGFwZSBjbGlwID0gZ2V0Q2xpcCgpOwotCi0gICAgICAgIGlmIChjbGlwICE9IG51bGwpIHsKLSAgICAgICAgICAgIC8vIFRPRE86IENhbiB3ZSBnZXQgc2hhcGUgYm91bmRzIHdpdGhvdXQgY3JlYXRpbmcgUmVjdGFuZ2xlIG9iamVjdD8KLSAgICAgICAgICAgIFJlY3RhbmdsZSBiID0gY2xpcC5nZXRCb3VuZHMoKTsKLSAgICAgICAgICAgIHIueCA9IGIueDsKLSAgICAgICAgICAgIHIueSA9IGIueTsKLSAgICAgICAgICAgIHIud2lkdGggPSBiLndpZHRoOwotICAgICAgICAgICAgci5oZWlnaHQgPSBiLmhlaWdodDsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiByOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJvdW5kcyBvZiB0aGUgY3VycmVudCBjbGlwcGluZyBhcmVhIGFzIGEgcmVjdGFuZ2xlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBSZWN0YW5nbGUgb2JqZWN0LgotICAgICAqIEBkZXByZWNhdGVkIFVzZSB7QGxpbmsgI2dldENsaXBCb3VuZHMoKX0KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Q2xpcFJlY3QoKSB7Ci0gICAgICAgIHJldHVybiBnZXRDbGlwQm91bmRzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZm9udCBtZXRyaWNzIG9mIHRoZSBjdXJyZW50IGZvbnQuIFRoZSBmb250IG1ldHJpY3Mgb2JqZWN0Ci0gICAgICogY29udGFpbnMgaW5mb3JtYXRpb24gYWJvdXQgdGhlIHJlbmRlcmluZyBvZiBhIHBhcnRpY3VsYXIgZm9udC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBmb250IG1ldHJpY3Mgb2YgY3VycmVudCBmb250LgotICAgICAqLwotICAgIHB1YmxpYyBGb250TWV0cmljcyBnZXRGb250TWV0cmljcygpIHsKLSAgICAgICAgcmV0dXJuIGdldEZvbnRNZXRyaWNzKGdldEZvbnQoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGV0ZXJtaW5lcyB3aGV0aGVyIG9yIG5vdCB0aGUgc3BlY2lmaWVkIHJlY3RhbmdsZSBpbnRlcnNlY3RzIHRoZSBjdXJyZW50Ci0gICAgICogY2xpcHBpbmcgYXJlYS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIHJlY3RhbmdsZSBpbnRlcnNlY3RzIHRoZSBjdXJyZW50IGNsaXBwaW5nCi0gICAgICogICAgICAgICBhcmVhLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaGl0Q2xpcChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICAvLyBUT0RPOiBDcmVhdGUgcGFja2FnZSBwcml2YXRlIG1ldGhvZCBSZWN0YW5nbGUuaW50ZXJzZWN0cyhpbnQsIGludCwKLSAgICAgICAgLy8gaW50LCBpbnQpOwotICAgICAgICByZXR1cm4gZ2V0Q2xpcEJvdW5kcygpLmludGVyc2VjdHMobmV3IFJlY3RhbmdsZSh4LCB5LCB3aWR0aCwgaGVpZ2h0KSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBzdHJpbmcgd2hpY2ggcmVwcmVzZW50cyB0aGlzIEdyYXBoaWNzIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgd2hpY2ggcmVwcmVzZW50cyB0aGlzIEdyYXBoaWNzIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICAvLyBUT0RPOiBUaGluayBhYm91dCBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgR3JhcGhpY3MuCi0gICAgICAgIHJldHVybiAiR3JhcGhpY3MiOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLy8gQWJzdHJhY3QgbWV0aG9kcwotCi0gICAgLyoqCi0gICAgICogQ2xlYXJzIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ2xlLiBUaGlzIG1ldGhvZCBmaWxscyBzcGVjaWZpZWQgcmVjdGFuZ2xlCi0gICAgICogd2l0aCBiYWNrZ3JvdW5kIGNvbG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBjbGVhclJlY3QoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQpOwotCi0gICAgLyoqCi0gICAgICogSW50ZXJzZWN0cyB0aGUgY3VycmVudCBjbGlwcGluZyBhcmVhIHdpdGggYSBuZXcgcmVjdGFuZ2xlLiBJZiB0aGUgY3VycmVudAotICAgICAqIGNsaXBwaW5nIGFyZWEgaXMgbm90IGRlZmluZWQsIHRoZSByZWN0YW5nbGUgYmVjb21lcyBhIG5ldyBjbGlwcGluZyBhcmVhLgotICAgICAqIFJlbmRlcmluZyBvcGVyYXRpb25zIGFyZSBvbmx5IGFsbG93ZWQgd2l0aGluIHRoZSBuZXcgdGhlIGNsaXBwaW5nIGFyZWEuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSBmb3IgaW50ZXJzZWN0aW9uLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgZm9yIGludGVyc2VjdGlvbi4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIGZvciBpbnRlcnNlY3Rpb24uCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlIGZvciBpbnRlcnNlY3Rpb24uCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgY2xpcFJlY3QoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQpOwotCi0gICAgLyoqCi0gICAgICogQ29waWVzIHRoZSByZWN0YW5nbGUgYXJlYSB0byBhbm90aGVyIGFyZWEgc3BlY2lmaWVkIGJ5IGEgZGlzdGFuY2UgKGR4LAotICAgICAqIGR5KSBmcm9tIHRoZSBvcmlnaW5hbCByZWN0YW5nbGUncyBsb2NhdGlvbi4gUG9zaXRpdmUgZHggYW5kIGR5IHZhbHVlcwotICAgICAqIGdpdmUgYSBuZXcgbG9jYXRpb24gZGVmaW5lZCBieSB0cmFuc2xhdGlvbiB0byB0aGUgcmlnaHQgYW5kIGRvd24gZnJvbSB0aGUKLSAgICAgKiBvcmlnaW5hbCBsb2NhdGlvbiwgbmVnYXRpdmUgZHggYW5kIGR5IHZhbHVlcyAtIHRvIHRoZSBsZWZ0IGFuZCB1cC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3gKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSB3aGljaCB3aWxsIGJlIGNvcGllZC4KLSAgICAgKiBAcGFyYW0gc3kKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSB3aGljaCB3aWxsIGJlIGNvcGllZC4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIHdoaWNoIHdpbGwgYmUgY29waWVkLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZSB3aGljaCB3aWxsIGJlIGNvcGllZC4KLSAgICAgKiBAcGFyYW0gZHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBob3Jpem9udGFsIGRpc3RhbmNlIGZyb20gdGhlIHNvdXJjZSByZWN0YW5nbGUncyBsb2NhdGlvbgotICAgICAqICAgICAgICAgICAgdG8gdGhlIGNvcHkncyBsb2NhdGlvbi4KLSAgICAgKiBAcGFyYW0gZHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZXJ0aWNhbCBkaXN0YW5jZSBmcm9tIHRoZSBzb3VyY2UgcmVjdGFuZ2xlJ3MgbG9jYXRpb24gdG8KLSAgICAgKiAgICAgICAgICAgIHRoZSBjb3B5J3MgbG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgY29weUFyZWEoaW50IHN4LCBpbnQgc3ksIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGR4LCBpbnQgZHkpOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIG5ldyBjb3B5IG9mIHRoaXMgR3JhcGhpY3MuCi0gICAgICogCi0gICAgICogQHJldHVybiBhIG5ldyBHcmFwaGljcyBjb250ZXh0IHdoaWNoIGlzIGEgY29weSBvZiB0aGlzIEdyYXBoaWNzLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBHcmFwaGljcyBjcmVhdGUoKTsKLQotICAgIC8qKgotICAgICAqIERpc3Bvc2VzIG9mIHRoZSBHcmFwaGljcy4gVGhpcyBHcmFwaGljcyBvYmplY3QgY2FuIG5vdCBiZSB1c2VkIGFmdGVyCi0gICAgICogY2FsbGluZyB0aGlzIG1ldGhvZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBkaXNwb3NlKCk7Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyB0aGUgYXJjIGNvdmVyaW5nIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ2xlIGFuZCB1c2luZyB0aGUgY3VycmVudAotICAgICAqIGNvbG9yLiBUaGUgcmVjdGFuZ2xlIGlzIGRlZmluZWQgYnkgdGhlIG9yaWdpbiBwb2ludCAoWCwgWSkgYW5kIGRpbWVuc2lvbnMKLSAgICAgKiAod2lkdGggYW5kIGhlaWdodCkuIFRoZSBhcmMgY2VudGVyIGlzIHRoZSB0aGUgY2VudGVyIG9mIHNwZWNpZmllZAotICAgICAqIHJlY3RhbmdsZS4gVGhlIGFuZ2xlIG9yaWdpbiBpcyAzIG8nY2xvY2sgcG9zaXRpb24sIHRoZSBwb3NpdGl2ZSBhbmdsZSBpcwotICAgICAqIGNvdW50ZWQgYXMgYSBjb3VudGVyLWNsb2Nrd2lzZSByb3RhdGlvbiwgdGhlIG5lZ2F0aXZlIGFuZ2xlIGlzIGNvdW50ZWQgYXMKLSAgICAgKiBjbG9ja3dpc2Ugcm90YXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIG9yaWdpbiBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgd2hpY2ggc2NhbGVzIHRoZSBhcmMuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIG9yaWdpbiBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgd2hpY2ggc2NhbGVzIHRoZSBhcmMuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZSB3aGljaCBzY2FsZXMgdGhlIGFyYy4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUgd2hpY2ggc2NhbGVzIHRoZSBhcmMuCi0gICAgICogQHBhcmFtIHNhCi0gICAgICogICAgICAgICAgICBzdGFydCBhbmdsZSAtIHRoZSBvcmlnaW4gYW5nbGUgb2YgYXJjLgotICAgICAqIEBwYXJhbSBlYQotICAgICAqICAgICAgICAgICAgYXJjIGFuZ2xlIC0gdGhlIGFuZ3VsYXIgYXJjIHZhbHVlIHJlbGF0aXZlIHRvIHRoZSBzdGFydCBhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBkcmF3QXJjKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgc2EsIGludCBlYSk7Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyB0aGUgc3BlY2lmaWVkIGltYWdlIHdpdGggdGhlIGRlZmluZWQgYmFja2dyb3VuZCBjb2xvci4gVGhlIHRvcCBsZWZ0Ci0gICAgICogY29ybmVyIG9mIGltYWdlIHdpbGwgYmUgZHJhd24gYXQgcG9pbnQgKHgsIHkpIGluIGN1cnJlbnQgY29vcmRpbmF0ZQotICAgICAqIHN5c3RlbS4gVGhlIGltYWdlIGxvYWRpbmcgcHJvY2VzcyBub3RpZmllcyB0aGUgc3BlY2lmaWVkIEltYWdlIE9ic2VydmVyLgotICAgICAqIFRoaXMgbWV0aG9kIHJldHVybnMgdHJ1ZSBpZiB0aGUgaW1hZ2UgaGFzIGxvYWRlZCwgb3RoZXJ3aXNlIGl0IHJldHVybnMKLSAgICAgKiBmYWxzZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1nCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2Ugd2hpY2ggd2lsbCBiZSBkcmF3bi4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgaW1hZ2UgdG9wIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBpbWFnZSB0b3AgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIGJnY29sb3IKLSAgICAgKiAgICAgICAgICAgIHRoZSBiYWNrZ3JvdW5kIGNvbG9yLgotICAgICAqIEBwYXJhbSBvYnNlcnZlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlT2JzZXJ2ZXIgb2JqZWN0IHdoaWNoIHNob3VsZCBiZSBub3RpZmllZCBhYm91dCBpbWFnZQotICAgICAqICAgICAgICAgICAgbG9hZGluZyBwcm9jZXNzLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgbG9hZGluZyBpbWFnZSBpcyBzdWNjZXNzZnVsIG9yIGltYWdlIGlzIG51bGwsIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGJvb2xlYW4gZHJhd0ltYWdlKEltYWdlIGltZywgaW50IHgsIGludCB5LCBDb2xvciBiZ2NvbG9yLCBJbWFnZU9ic2VydmVyIG9ic2VydmVyKTsKLQotICAgIC8qKgotICAgICAqIERyYXdzIHRoZSBzcGVjaWZpZWQgaW1hZ2UuIFRoZSB0b3AgbGVmdCBjb3JuZXIgb2YgaW1hZ2Ugd2lsbCBiZSBkcmF3biBhdAotICAgICAqIHBvaW50ICh4LCB5KSBpbiBjdXJyZW50IGNvb3JkaW5hdGUgc3lzdGVtLiBUaGUgaW1hZ2UgbG9hZGluZyBwcm9jZXNzCi0gICAgICogbm90aWZpZXMgdGhlIHNwZWNpZmllZCBJbWFnZSBPYnNlcnZlci4gVGhpcyBtZXRob2QgcmV0dXJucyB0cnVlIGlmIHRoZQotICAgICAqIGltYWdlIGhhcyBsb2FkZWQsIG90aGVyd2lzZSBpdCByZXR1cm5zIGZhbHNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWcKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSB3aGljaCB3aWxsIGJlIGRyYXduLgotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBpbWFnZSB0b3AgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIGltYWdlIHRvcCBsZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0gb2JzZXJ2ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZU9ic2VydmVyIG9iamVjdCB3aGljaCBzaG91bGQgYmUgbm90aWZpZWQgYWJvdXQgaW1hZ2UKLSAgICAgKiAgICAgICAgICAgIGxvYWRpbmcgcHJvY2Vzcy4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGxvYWRpbmcgaW1hZ2UgaXMgc3VjY2Vzc2Z1bCBvciBpbWFnZSBpcyBudWxsLCBvdGhlcndpc2UKLSAgICAgKiAgICAgICAgIGZhbHNlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBib29sZWFuIGRyYXdJbWFnZShJbWFnZSBpbWcsIGludCB4LCBpbnQgeSwgSW1hZ2VPYnNlcnZlciBvYnNlcnZlcik7Ci0KLSAgICAvKioKLSAgICAgKiBTY2FsZXMgdGhlIHNwZWNpZmllZCBpbWFnZSB0byBmaXQgaW4gdGhlIHNwZWNpZmllZCByZWN0YW5nbGUgYW5kIGRyYXdzIGl0Ci0gICAgICogd2l0aCB0aGUgZGVmaW5lZCBiYWNrZ3JvdW5kIGNvbG9yLiBUaGUgdG9wIGxlZnQgY29ybmVyIG9mIHRoZSBpbWFnZSB3aWxsCi0gICAgICogYmUgZHJhd24gYXQgdGhlIHBvaW50ICh4LCB5KSBpbiBjdXJyZW50IGNvb3JkaW5hdGUgc3lzdGVtLiBUaGUgbm9uLW9wYXF1ZQotICAgICAqIHBpeGVscyB3aWxsIGJlIGRyYXduIGluIHRoZSBiYWNrZ3JvdW5kIGNvbG9yLiBUaGUgaW1hZ2UgbG9hZGluZyBwcm9jZXNzCi0gICAgICogbm90aWZpZXMgdGhlIHNwZWNpZmllZCBJbWFnZSBPYnNlcnZlci4gVGhpcyBtZXRob2QgcmV0dXJucyB0cnVlIGlmIHRoZQotICAgICAqIGltYWdlIGhhcyBsb2FkZWQsIG90aGVyd2lzZSBpdCByZXR1cm5zIGZhbHNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWcKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSB3aGljaCB3aWxsIGJlIGRyYXduLgotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBpbWFnZSdzIHRvcCBsZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgaW1hZ2UncyB0b3AgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgcmVjdGFuZ2xlIHdoaWNoIHNjYWxlcyB0aGUgaW1hZ2UuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiByZWN0YW5nbGUgd2hpY2ggc2NhbGVzIHRoZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gYmdjb2xvcgotICAgICAqICAgICAgICAgICAgdGhlIGJhY2tncm91bmQgY29sb3IuCi0gICAgICogQHBhcmFtIG9ic2VydmVyCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VPYnNlcnZlciBvYmplY3Qgd2hpY2ggc2hvdWxkIGJlIG5vdGlmaWVkIGFib3V0IGltYWdlCi0gICAgICogICAgICAgICAgICBsb2FkaW5nIHByb2Nlc3MuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBsb2FkaW5nIGltYWdlIGlzIHN1Y2Nlc3NmdWwgb3IgaW1hZ2UgaXMgbnVsbCwgb3RoZXJ3aXNlCi0gICAgICogICAgICAgICBmYWxzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1nLCBpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCwKLSAgICAgICAgICAgIENvbG9yIGJnY29sb3IsIEltYWdlT2JzZXJ2ZXIgb2JzZXJ2ZXIpOwotCi0gICAgLyoqCi0gICAgICogU2NhbGVzIHRoZSBzcGVjaWZpZWQgaW1hZ2UgdG8gZml0IGluIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ2xlIGFuZCBkcmF3cwotICAgICAqIGl0LiBUaGUgdG9wIGxlZnQgY29ybmVyIG9mIHRoZSBpbWFnZSB3aWxsIGJlIGRyYXduIGF0IHRoZSBwb2ludCAoeCwgeSkgaW4KLSAgICAgKiBjdXJyZW50IGNvb3JkaW5hdGUgc3lzdGVtLiBUaGUgaW1hZ2UgbG9hZGluZyBwcm9jZXNzIG5vdGlmaWVzIHRoZQotICAgICAqIHNwZWNpZmllZCBJbWFnZSBPYnNlcnZlci4gVGhpcyBtZXRob2QgcmV0dXJucyB0cnVlIGlmIHRoZSBpbWFnZSBoYXMKLSAgICAgKiBsb2FkZWQsIG90aGVyd2lzZSBpdCByZXR1cm5zIGZhbHNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWcKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSB3aGljaCB3aWxsIGJlIGRyYXduLgotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBpbWFnZSB0b3AgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIGltYWdlIHRvcCBsZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiByZWN0YW5nbGUgd2hpY2ggc2NhbGVzIHRoZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHJlY3RhbmdsZSB3aGljaCBzY2FsZXMgdGhlIGltYWdlLgotICAgICAqIEBwYXJhbSBvYnNlcnZlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlT2JzZXJ2ZXIgb2JqZWN0IHdoaWNoIHNob3VsZCBiZSBub3RpZmllZCBhYm91dCBpbWFnZQotICAgICAqICAgICAgICAgICAgbG9hZGluZyBwcm9jZXNzLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgbG9hZGluZyBpbWFnZSBpcyBzdWNjZXNzZnVsIG9yIGltYWdlIGlzIG51bGwsIG90aGVyd2lzZQotICAgICAqICAgICAgICAgZmFsc2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGJvb2xlYW4gZHJhd0ltYWdlKEltYWdlIGltZywgaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQsCi0gICAgICAgICAgICBJbWFnZU9ic2VydmVyIG9ic2VydmVyKTsKLQotICAgIC8qKgotICAgICAqIFNjYWxlcyB0aGUgc3BlY2lmaWVkIGFyZWEgb2YgdGhlIHNwZWNpZmllZCBpbWFnZSB0byBmaXQgaW4gdGhlIHJlY3RhbmdsZQotICAgICAqIGFyZWEgZGVmaW5lZCBieSBpdHMgY29ybmVycyBjb29yZGluYXRlcyBhbmQgZHJhd3MgdGhlIHN1Yi1pbWFnZSB3aXRoIHRoZQotICAgICAqIHNwZWNpZmllZCBiYWNrZ3JvdW5kIGNvbG9yLiBUaGUgc3ViLWltYWdlIHRvIGJlIGRyYXduIGlzIGRlZmluZWQgYnkgaXRzCi0gICAgICogdG9wIGxlZnQgY29ybmVyIGNvb3JkaW5hdGVzIChzeDEsIHN5MSkgYW5kIGJvdHRvbSByaWdodCBjb3JuZXIKLSAgICAgKiBjb29yZGluYXRlcyAoc3gyLCBzeTIpIGNvbXB1dGVkIHdpdGggcmVzcGVjdCB0byB0aGUgb3JpZ2luICh0b3AgbGVmdAotICAgICAqIGNvcm5lcikgb2YgdGhlIHNvdXJjZSBpbWFnZS4gVGhlIG5vbiBvcGFxdWUgcGl4ZWxzIHdpbGwgYmUgZHJhd24gaW4gdGhlCi0gICAgICogYmFja2dyb3VuZCBjb2xvci4gVGhlIGltYWdlIGxvYWRpbmcgcHJvY2VzcyBub3RpZmllcyBzcGVjaWZpZWQgSW1hZ2UKLSAgICAgKiBPYnNlcnZlci4gVGhpcyBtZXRob2QgcmV0dXJucyB0cnVlIGlmIHRoZSBpbWFnZSBoYXMgbG9hZGVkLCBvdGhlcndpc2UgaXQKLSAgICAgKiByZXR1cm5zIGZhbHNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWcKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSB3aGljaCB3aWxsIGJlIGRyYXduLgotICAgICAqIEBwYXJhbSBkeDEKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIHRvcCBsZWZ0IGNvcm5lciBjb29yZGluYXRlIG9mIHRoZSBkZXN0aW5hdGlvbiByZWN0YW5nbGUKLSAgICAgKiAgICAgICAgICAgIGFyZWEuCi0gICAgICogQHBhcmFtIGR5MQotICAgICAqICAgICAgICAgICAgdGhlIFkgdG9wIGxlZnQgY29ybmVyIGNvb3JkaW5hdGUgb2YgdGhlIGRlc3RpbmF0aW9uIHJlY3RhbmdsZQotICAgICAqICAgICAgICAgICAgYXJlYS4KLSAgICAgKiBAcGFyYW0gZHgyCi0gICAgICogICAgICAgICAgICB0aGUgWCBib3R0b20gcmlnaHQgY29ybmVyIGNvb3JkaW5hdGUgb2YgdGhlIGRlc3RpbmF0aW9uCi0gICAgICogICAgICAgICAgICByZWN0YW5nbGUgYXJlYS4KLSAgICAgKiBAcGFyYW0gZHkyCi0gICAgICogICAgICAgICAgICB0aGUgWSBib3R0b20gcmlnaHQgY29ybmVyIGNvb3JkaW5hdGUgb2YgdGhlIGRlc3RpbmF0aW9uCi0gICAgICogICAgICAgICAgICByZWN0YW5nbGUgYXJlYS4KLSAgICAgKiBAcGFyYW0gc3gxCi0gICAgICogICAgICAgICAgICB0aGUgWCB0b3AgbGVmdCBjb3JuZXIgY29vcmRpbmF0ZSBvZiB0aGUgYXJlYSB0byBiZSBkcmF3bgotICAgICAqICAgICAgICAgICAgd2l0aGluIHRoZSBzb3VyY2UgaW1hZ2UuCi0gICAgICogQHBhcmFtIHN5MQotICAgICAqICAgICAgICAgICAgdGhlIFkgdG9wIGxlZnQgY29ybmVyIGNvb3JkaW5hdGUgb2YgdGhlIGFyZWEgdG8gYmUgZHJhd24KLSAgICAgKiAgICAgICAgICAgIHdpdGhpbiB0aGUgc291cmNlIGltYWdlLgotICAgICAqIEBwYXJhbSBzeDIKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGJvdHRvbSByaWdodCBjb3JuZXIgY29vcmRpbmF0ZSBvZiB0aGUgYXJlYSB0byBiZSBkcmF3bgotICAgICAqICAgICAgICAgICAgd2l0aGluIHRoZSBzb3VyY2UgaW1hZ2UuCi0gICAgICogQHBhcmFtIHN5MgotICAgICAqICAgICAgICAgICAgdGhlIFkgYm90dG9tIHJpZ2h0IGNvcm5lciBjb29yZGluYXRlIG9mIHRoZSBhcmVhIHRvIGJlIGRyYXduCi0gICAgICogICAgICAgICAgICB3aXRoaW4gdGhlIHNvdXJjZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gYmdjb2xvcgotICAgICAqICAgICAgICAgICAgdGhlIGJhY2tncm91bmQgY29sb3IuCi0gICAgICogQHBhcmFtIG9ic2VydmVyCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VPYnNlcnZlciBvYmplY3Qgd2hpY2ggc2hvdWxkIGJlIG5vdGlmaWVkIGFib3V0IGltYWdlCi0gICAgICogICAgICAgICAgICBsb2FkaW5nIHByb2Nlc3MuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBsb2FkaW5nIGltYWdlIGlzIHN1Y2Nlc3NmdWwgb3IgaW1hZ2UgaXMgbnVsbCwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1nLCBpbnQgZHgxLCBpbnQgZHkxLCBpbnQgZHgyLCBpbnQgZHkyLCBpbnQgc3gxLAotICAgICAgICAgICAgaW50IHN5MSwgaW50IHN4MiwgaW50IHN5MiwgQ29sb3IgYmdjb2xvciwgSW1hZ2VPYnNlcnZlciBvYnNlcnZlcik7Ci0KLSAgICAvKioKLSAgICAgKiBTY2FsZXMgdGhlIHNwZWNpZmllZCBhcmVhIG9mIHRoZSBzcGVjaWZpZWQgaW1hZ2UgdG8gZml0IGluIHRoZSByZWN0YW5nbGUKLSAgICAgKiBhcmVhIGRlZmluZWQgYnkgaXRzIGNvcm5lcnMgY29vcmRpbmF0ZXMgYW5kIGRyYXdzIHRoZSBzdWItaW1hZ2UuIFRoZQotICAgICAqIHN1Yi1pbWFnZSB0byBiZSBkcmF3biBpcyBkZWZpbmVkIGJ5IGl0cyB0b3AgbGVmdCBjb3JuZXIgY29vcmRpbmF0ZXMgKHN4MSwKLSAgICAgKiBzeTEpIGFuZCBib3R0b20gcmlnaHQgY29ybmVyIGNvb3JkaW5hdGVzIChzeDIsIHN5MikgY29tcHV0ZWQgd2l0aCByZXNwZWN0Ci0gICAgICogdG8gdGhlIG9yaWdpbiAodG9wIGxlZnQgY29ybmVyKSBvZiB0aGUgc291cmNlIGltYWdlLiBUaGUgaW1hZ2UgbG9hZGluZwotICAgICAqIHByb2Nlc3Mgbm90aWZpZXMgc3BlY2lmaWVkIEltYWdlIE9ic2VydmVyLiBUaGlzIG1ldGhvZCByZXR1cm5zIHRydWUgaWYKLSAgICAgKiB0aGUgaW1hZ2UgaGFzIGxvYWRlZCwgb3RoZXJ3aXNlIGl0IHJldHVybnMgZmFsc2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltZwotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHdoaWNoIHdpbGwgYmUgZHJhd24uCi0gICAgICogQHBhcmFtIGR4MQotICAgICAqICAgICAgICAgICAgdGhlIFggdG9wIGxlZnQgY29ybmVyIGNvb3JkaW5hdGUgb2YgdGhlIGRlc3RpbmF0aW9uIHJlY3RhbmdsZQotICAgICAqICAgICAgICAgICAgYXJlYS4KLSAgICAgKiBAcGFyYW0gZHkxCi0gICAgICogICAgICAgICAgICB0aGUgWSB0b3AgbGVmdCBjb3JuZXIgY29vcmRpbmF0ZSBvZiB0aGUgZGVzdGluYXRpb24gcmVjdGFuZ2xlCi0gICAgICogICAgICAgICAgICBhcmVhLgotICAgICAqIEBwYXJhbSBkeDIKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGJvdHRvbSByaWdodCBjb3JuZXIgY29vcmRpbmF0ZSBvZiB0aGUgZGVzdGluYXRpb24KLSAgICAgKiAgICAgICAgICAgIHJlY3RhbmdsZSBhcmVhLgotICAgICAqIEBwYXJhbSBkeTIKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGJvdHRvbSByaWdodCBjb3JuZXIgY29vcmRpbmF0ZSBvZiB0aGUgZGVzdGluYXRpb24KLSAgICAgKiAgICAgICAgICAgIHJlY3RhbmdsZSBhcmVhLgotICAgICAqIEBwYXJhbSBzeDEKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIHRvcCBsZWZ0IGNvcm5lciBjb29yZGluYXRlIG9mIHRoZSBhcmVhIHRvIGJlIGRyYXduCi0gICAgICogICAgICAgICAgICB3aXRoaW4gdGhlIHNvdXJjZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gc3kxCi0gICAgICogICAgICAgICAgICB0aGUgWSB0b3AgbGVmdCBjb3JuZXIgY29vcmRpbmF0ZSBvZiB0aGUgYXJlYSB0byBiZSBkcmF3bgotICAgICAqICAgICAgICAgICAgd2l0aGluIHRoZSBzb3VyY2UgaW1hZ2UuCi0gICAgICogQHBhcmFtIHN4MgotICAgICAqICAgICAgICAgICAgdGhlIFggYm90dG9tIHJpZ2h0IGNvcm5lciBjb29yZGluYXRlIG9mIHRoZSBhcmVhIHRvIGJlIGRyYXduCi0gICAgICogICAgICAgICAgICB3aXRoaW4gdGhlIHNvdXJjZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gc3kyCi0gICAgICogICAgICAgICAgICB0aGUgWSBib3R0b20gcmlnaHQgY29ybmVyIGNvb3JkaW5hdGUgb2YgdGhlIGFyZWEgdG8gYmUgZHJhd24KLSAgICAgKiAgICAgICAgICAgIHdpdGhpbiB0aGUgc291cmNlIGltYWdlLgotICAgICAqIEBwYXJhbSBvYnNlcnZlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlT2JzZXJ2ZXIgb2JqZWN0IHdoaWNoIHNob3VsZCBiZSBub3RpZmllZCBhYm91dCBpbWFnZQotICAgICAqICAgICAgICAgICAgbG9hZGluZyBwcm9jZXNzLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgbG9hZGluZyBpbWFnZSBpcyBzdWNjZXNzZnVsIG9yIGltYWdlIGlzIG51bGwsIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGJvb2xlYW4gZHJhd0ltYWdlKEltYWdlIGltZywgaW50IGR4MSwgaW50IGR5MSwgaW50IGR4MiwgaW50IGR5MiwgaW50IHN4MSwKLSAgICAgICAgICAgIGludCBzeTEsIGludCBzeDIsIGludCBzeTIsIEltYWdlT2JzZXJ2ZXIgb2JzZXJ2ZXIpOwotCi0gICAgLyoqCi0gICAgICogRHJhd3MgYSBsaW5lIGZyb20gdGhlIHBvaW50ICh4MSwgeTEpIHRvIHRoZSBwb2ludCAoeDIsIHkyKS4gVGhpcyBtZXRob2QKLSAgICAgKiBkcmF3cyB0aGUgbGluZSB3aXRoIGN1cnJlbnQgY29sb3Igd2hpY2ggY2FuIGJlIGNoYW5nZWQgYnkgc2V0Q29sb3IoQ29sb3IKLSAgICAgKiBjKSBtZXRob2QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IHBvaW50LgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIHBvaW50LgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGRyYXdMaW5lKGludCB4MSwgaW50IHkxLCBpbnQgeDIsIGludCB5Mik7Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyB0aGUgb3V0bGluZSBvZiBhbiBvdmFsIHRvIGZpdCBpbiB0aGUgcmVjdGFuZ2xlIGRlZmluZWQgYnkgdGhlIGdpdmVuCi0gICAgICogd2lkdGgsIGhlaWdodCwgYW5kIHRvcCBsZWZ0IGNvcm5lci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggdG9wIGxlZnQgY29ybmVyIG92YWwgY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgdG9wIGxlZnQgY29ybmVyIG92YWwgY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBvdmFsIHdpZHRoLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvdmFsIGhlaWdodC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBkcmF3T3ZhbChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCk7Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyB0aGUgb3V0bGluZSBvZiBhIHBvbHlnb24uIFRoZSBwb2x5Z29uIHZlcnRpY2VzIGFyZSBkZWZpbmVkIGJ5Ci0gICAgICogcG9pbnRzIHdpdGggeHBvaW50c1tpXSwgeXBvaW50c1tpXSBhcyBjb29yZGluYXRlcy4gVGhlIHBvbHlnb24gZWRnZXMgYXJlCi0gICAgICogdGhlIGxpbmVzIGZyb20gdGhlIHBvaW50cyB3aXRoICh4cG9pbnRzW2ktMV0sIHlwb2ludHNbaS0xXSkgY29vcmRpbmF0ZXMKLSAgICAgKiB0byB0aGUgcG9pbnRzIHdpdGggKHhwb2ludHNbaV0sIHlwb2ludHNbaV0pIGNvb3JkaW5hdGVzLCBmb3IgMCA8IGkgPAotICAgICAqIG5wb2ludHMgKzEuCi0gICAgICogCi0gICAgICogQHBhcmFtIHhwb2ludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBYIGNvb3JkaW5hdGVzIG9mIHRoZSBwb2x5Z29uIHZlcnRpY2VzLgotICAgICAqIEBwYXJhbSB5cG9pbnRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgWSBjb29yZGluYXRlcyBvZiB0aGUgcG9seWdvbiB2ZXJ0aWNlcy4KLSAgICAgKiBAcGFyYW0gbnBvaW50cwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBwb2x5Z29uIHZlcnRpY2VzL3BvaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBkcmF3UG9seWdvbihpbnRbXSB4cG9pbnRzLCBpbnRbXSB5cG9pbnRzLCBpbnQgbnBvaW50cyk7Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyBhIHNldCBvZiBjb25uZWN0ZWQgbGluZXMgd2hpY2ggYXJlIGRlZmluZWQgYnkgdGhlIHggYW5kIHkKLSAgICAgKiBjb29yZGluYXRlIGFycmF5cy4gVGhlIHBvbHlsaW5lIGlzIGNsb3NlZCBpZiBjb29yZGluYXRlcyBvZiB0aGUgZmlyc3QKLSAgICAgKiBwb2ludCBhcmUgdGhlIHNhbWUgYXMgY29vcmRpbmF0ZXMgb2YgdGhlIGxhc3QgcG9pbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHhwb2ludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBYIHBvaW50IGNvb3JkaW5hdGVzLgotICAgICAqIEBwYXJhbSB5cG9pbnRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgWSBwb2ludCBjb29yZGluYXRlcy4KLSAgICAgKiBAcGFyYW0gbnBvaW50cwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBwb2ludHMuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd1BvbHlsaW5lKGludFtdIHhwb2ludHMsIGludFtdIHlwb2ludHMsIGludCBucG9pbnRzKTsKLQotICAgIC8qKgotICAgICAqIERyYXdzIHRoZSBvdXRsaW5lIG9mIGEgcmVjdGFuZ2xlIHdpdGggcm91bmQgY29ybmVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdG9wIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB0b3AgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGFyY1dpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgYXJjIHdpZHRoIGZvciB0aGUgY29ybmVycy4KLSAgICAgKiBAcGFyYW0gYXJjSGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgYXJjIGhlaWdodCBmb3IgdGhlIGNvcm5lcnMuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd1JvdW5kUmVjdChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGFyY1dpZHRoLAotICAgICAgICAgICAgaW50IGFyY0hlaWdodCk7Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyBhIHRleHQgZGVmaW5lZCBieSBhbiBpdGVyYXRvci4gVGhlIGl0ZXJhdG9yIHNob3VsZCBzcGVjaWZ5IHRoZSBmb250Ci0gICAgICogZm9yIGV2ZXJ5IGNoYXJhY3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaXRlcmF0b3IKLSAgICAgKiAgICAgICAgICAgIHRoZSBpdGVyYXRvci4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY2hhcmFjdGVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjaGFyYWN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd1N0cmluZyhBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgaXRlcmF0b3IsIGludCB4LCBpbnQgeSk7Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyBhIHRleHQgZGVmaW5lZCBieSBhIHN0cmluZy4gVGhpcyBtZXRob2QgZHJhd3MgdGhlIHRleHQgd2l0aCBjdXJyZW50Ci0gICAgICogZm9udCBhbmQgY29sb3IuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cgotICAgICAqICAgICAgICAgICAgdGhlIHN0cmluZy4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY2hhcmFjdGVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjaGFyYWN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd1N0cmluZyhTdHJpbmcgc3RyLCBpbnQgeCwgaW50IHkpOwotCi0gICAgLyoqCi0gICAgICogRmlsbHMgdGhlIGFyYyBjb3ZlcmluZyB0aGUgcmVjdGFuZ2xlIGFuZCB1c2luZyB0aGUgY3VycmVudCBjb2xvci4gVGhlCi0gICAgICogcmVjdGFuZ2xlIGlzIGRlZmluZWQgYnkgdGhlIG9yaWdpbiBwb2ludCAoWCwgWSkgYW5kIGRpbWVuc2lvbnMgKHdpZHRoIGFuZAotICAgICAqIGhlaWdodCkuIFRoZSBhcmMgY2VudGVyIGlzIHRoZSB0aGUgY2VudGVyIG9mIHNwZWNpZmllZCByZWN0YW5nbGUuIFRoZQotICAgICAqIGFuZ2xlIG9yaWdpbiBpcyBhdCB0aGUgMyBvJ2Nsb2NrIHBvc2l0aW9uLCBhbmQgYSBwb3NpdGl2ZSBhbmdsZSBnaXZlcwotICAgICAqIGNvdW50ZXItY2xvY2t3aXNlIHJvdGF0aW9uLCBhIG5lZ2F0aXZlIGFuZ2xlIGdpdmVzIGNsb2Nrd2lzZSByb3RhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggb3JpZ2luIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSB3aGljaCBzY2FsZXMgdGhlIGFyYy4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgb3JpZ2luIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSB3aGljaCBzY2FsZXMgdGhlIGFyYy4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIHdoaWNoIHNjYWxlcyB0aGUgYXJjLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZSB3aGljaCBzY2FsZXMgdGhlIGFyYy4KLSAgICAgKiBAcGFyYW0gc2EKLSAgICAgKiAgICAgICAgICAgIHN0YXJ0IGFuZ2xlIC0gdGhlIG9yaWdpbiBhbmdsZSBvZiBhcmMuCi0gICAgICogQHBhcmFtIGVhCi0gICAgICogICAgICAgICAgICBhcmMgYW5nbGUgLSB0aGUgYW5ndWxhciBhcmMgdmFsdWUgcmVsYXRpdmUgdG8gdGhlIHN0YXJ0IGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGZpbGxBcmMoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQsIGludCBzYSwgaW50IGVhKTsKLQotICAgIC8qKgotICAgICAqIEZpbGxzIGFuIG92YWwgd2l0aCB0aGUgY3VycmVudCBjb2xvciB3aGVyZSB0aGUgb3ZhbCBpcyBkZWZpbmVkIGJ5IHRoZQotICAgICAqIGJvdW5kaW5nIHJlY3RhbmdsZSB3aXRoIHRoZSBnaXZlbiB3aWR0aCwgaGVpZ2h0LCBhbmQgdG9wIGxlZnQgY29ybmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCB0b3AgbGVmdCBjb3JuZXIgb3ZhbCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSB0b3AgbGVmdCBjb3JuZXIgb3ZhbCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIG92YWwgd2lkdGguCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIG92YWwgaGVpZ2h0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGZpbGxPdmFsKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KTsKLQotICAgIC8qKgotICAgICAqIEZpbGxzIGEgcG9seWdvbiB3aXRoIHRoZSBjdXJyZW50IGNvbG9yLiBUaGUgcG9seWdvbiB2ZXJ0aWNlcyBhcmUgZGVmaW5lZAotICAgICAqIGJ5IHRoZSBwb2ludHMgd2l0aCB4cG9pbnRzW2ldLCB5cG9pbnRzW2ldIGFzIGNvb3JkaW5hdGVzLiBUaGUgcG9seWdvbgotICAgICAqIGVkZ2VzIGFyZSB0aGUgbGluZXMgZnJvbSB0aGUgcG9pbnRzIHdpdGggKHhwb2ludHNbaS0xXSwgeXBvaW50c1tpLTFdKQotICAgICAqIGNvb3JkaW5hdGVzIHRvIHRoZSBwb2ludHMgd2l0aCAoeHBvaW50c1tpXSwgeXBvaW50c1tpXSkgY29vcmRpbmF0ZXMsIGZvcgotICAgICAqIDAgPCBpIDwgbnBvaW50cyArMS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geHBvaW50cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIFggY29vcmRpbmF0ZXMgb2YgdGhlIHBvbHlnb24gdmVydGljZXMuCi0gICAgICogQHBhcmFtIHlwb2ludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBZIGNvb3JkaW5hdGVzIG9mIHRoZSBwb2x5Z29uIHZlcnRpY2VzLgotICAgICAqIEBwYXJhbSBucG9pbnRzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIHBvbHlnb24gdmVydGljZXMvcG9pbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGZpbGxQb2x5Z29uKGludFtdIHhwb2ludHMsIGludFtdIHlwb2ludHMsIGludCBucG9pbnRzKTsKLQotICAgIC8qKgotICAgICAqIEZpbGxzIGEgcmVjdGFuZ2xlIHdpdGggdGhlIGN1cnJlbnQgY29sb3IuIFRoZSByZWN0YW5nbGUgaXMgZGVmaW5lZCBieSBpdHMKLSAgICAgKiB3aWR0aCBhbmQgbGVuZ3RoIGFuZCB0b3AgbGVmdCBjb3JuZXIgY29vcmRpbmF0ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHRvcCBsZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdG9wIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBmaWxsUmVjdChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCk7Ci0KLSAgICAvKioKLSAgICAgKiBGaWxscyBhIHJvdW5kIGNvcm5lcmVkIHJlY3RhbmdsZSB3aXRoIHRoZSBjdXJyZW50IGNvbG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSB0b3AgbGVmdCBjb3JuZXIgb2YgdGhlIGJvdW5kaW5nCi0gICAgICogICAgICAgICAgICByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHRvcCBsZWZ0IGNvcm5lciBvZiB0aGUgYm91bmRpbmcKLSAgICAgKiAgICAgICAgICAgIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gYXJjV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcmMgd2lkdGggYXQgdGhlIGNvcm5lcnMuCi0gICAgICogQHBhcmFtIGFyY0hlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGFyYyBoZWlnaHQgYXQgdGhlIGNvcm5lcnMuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZmlsbFJvdW5kUmVjdChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGFyY1dpZHRoLAotICAgICAgICAgICAgaW50IGFyY0hlaWdodCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBjbGlwcGluZyBhcmVhLiA8YnI+Ci0gICAgICogPGJyPgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBTaGFwZSBvYmplY3Qgb2YgdGhlIGNsaXBwaW5nIGFyZWEgb3IgbnVsbCBpZiBpdCBpcyBub3Qgc2V0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBTaGFwZSBnZXRDbGlwKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZHMgb2YgdGhlIGN1cnJlbnQgY2xpcHBpbmcgYXJlYSBhcyBhIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgUmVjdGFuZ2xlIG9iamVjdCB3aGljaCByZXByZXNlbnRzIHRoZSBib3VuZHMgb2YgdGhlIGN1cnJlbnQKLSAgICAgKiAgICAgICAgIGNsaXBwaW5nIGFyZWEuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFJlY3RhbmdsZSBnZXRDbGlwQm91bmRzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBjdXJyZW50IGNvbG9yIG9mIEdyYXBoaWNzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGN1cnJlbnQgY29sb3IuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IENvbG9yIGdldENvbG9yKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBjdXJyZW50IGZvbnQgb2YgR3JhcGhpY3MuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY3VycmVudCBmb250LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBGb250IGdldEZvbnQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZvbnQgbWV0cmljcyBvZiB0aGUgc3BlY2lmaWVkIGZvbnQuIFRoZSBmb250IG1ldHJpY3Mgb2JqZWN0Ci0gICAgICogY29udGFpbnMgaW5mb3JtYXRpb24gYWJvdXQgdGhlIHJlbmRlcmluZyBvZiBhIHBhcnRpY3VsYXIgZm9udC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZm9udAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBmb250LgotICAgICAqIEByZXR1cm4gdGhlIGZvbnQgbWV0cmljcyBmb3IgdGhlIHNwZWNpZmllZCBmb250LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBGb250TWV0cmljcyBnZXRGb250TWV0cmljcyhGb250IGZvbnQpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbmV3IGNsaXBwaW5nIGFyZWEgc3BlY2lmaWVkIGJ5IHJlY3RhbmdsZS4gVGhlIG5ldyBjbGlwcGluZyBhcmVhCi0gICAgICogZG9lc24ndCBkZXBlbmQgb24gdGhlIHdpbmRvdydzIHZpc2liaWxpdHkuIFJlbmRlcmluZyBvcGVyYXRpb25zIGNhbid0IGJlCi0gICAgICogcGVyZm9ybWVkIG91dHNpZGUgbmV3IGNsaXBwaW5nIGFyZWEuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIG5ldyBjbGlwcGluZyByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIG5ldyBjbGlwcGluZyByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIG5ldyBjbGlwcGluZyByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgbmV3IGNsaXBwaW5nIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzZXRDbGlwKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIG5ldyBjbGlwcGluZyBhcmVhIHRvIGJlIHRoZSBhcmVhIHNwZWNpZmllZCBieSBTaGFwZSBvYmplY3QuIFRoZQotICAgICAqIG5ldyBjbGlwcGluZyBhcmVhIGRvZXNuJ3QgZGVwZW5kIG9uIHRoZSB3aW5kb3cncyB2aXNpYmlsaXR5LiBSZW5kZXJpbmcKLSAgICAgKiBvcGVyYXRpb25zIGNhbid0IGJlIHBlcmZvcm1lZCBvdXRzaWRlIG5ldyBjbGlwcGluZyBhcmVhLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjbGlwCi0gICAgICogICAgICAgICAgICB0aGUgU2hhcGUgb2JqZWN0IHdoaWNoIHJlcHJlc2VudHMgbmV3IGNsaXBwaW5nIGFyZWEuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0Q2xpcChTaGFwZSBjbGlwKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGN1cnJlbnQgR3JhcGhpY3MgY29sb3IuIEFsbCByZW5kZXJpbmcgb3BlcmF0aW9ucyB3aXRoIHRoaXMKLSAgICAgKiBHcmFwaGljcyB3aWxsIHVzZSB0aGlzIGNvbG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IGNvbG9yLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldENvbG9yKENvbG9yIGMpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgY3VycmVudCBHcmFwaGljcyBmb250LiBBbGwgcmVuZGVyaW5nIG9wZXJhdGlvbnMgd2l0aCB0aGlzCi0gICAgICogR3JhcGhpY3Mgd2lsbCB1c2UgdGhpcyBmb250LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmb250Ci0gICAgICogICAgICAgICAgICB0aGUgbmV3IGZvbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0Rm9udChGb250IGZvbnQpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgcGFpbnQgbW9kZSBmb3IgdGhlIEdyYXBoaWNzIHdoaWNoIG92ZXJ3cml0ZXMgYWxsIHJlbmRlcmluZwotICAgICAqIG9wZXJhdGlvbnMgd2l0aCB0aGUgY3VycmVudCBjb2xvci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzZXRQYWludE1vZGUoKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIFhPUiBtb2RlIGZvciB0aGUgR3JhcGhpY3Mgd2hpY2ggY2hhbmdlcyBhIHBpeGVsIGZyb20gdGhlIGN1cnJlbnQKLSAgICAgKiBjb2xvciB0byB0aGUgc3BlY2lmaWVkIFhPUiBjb2xvci4gPGJyPgotICAgICAqIDxicj4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29sb3IKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgWE9SIG1vZGUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0WE9STW9kZShDb2xvciBjb2xvcik7Ci0KLSAgICAvKioKLSAgICAgKiBUcmFuc2xhdGVzIHRoZSBvcmlnaW4gb2YgR3JhcGhpY3MgY3VycmVudCBjb29yZGluYXRlIHN5c3RlbSB0byB0aGUgcG9pbnQKLSAgICAgKiB3aXRoIFgsIFkgY29vcmRpbmF0ZXMgaW4gdGhlIGN1cnJlbnQgY29vcmRpbmF0ZSBzeXN0ZW0uIEFsbCByZW5kZXJpbmcKLSAgICAgKiBvcGVyYXRpb24gaW4gdGhpcyBHcmFwaGljcyB3aWxsIGJlIHJlbGF0ZWQgdG8gdGhlIG5ldyBvcmlnaW4uCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIG9yaWdpbi4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgb3JpZ2luLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHRyYW5zbGF0ZShpbnQgeCwgaW50IHkpOwotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0dyYXBoaWNzMkQuamF2YSBiL2F3dC9qYXZhL2F3dC9HcmFwaGljczJELmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDA0YTczMTkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0dyYXBoaWNzMkQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDUxMyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuZm9udC5HbHlwaFZlY3RvcjsKLWltcG9ydCBqYXZhLmF3dC5mb250LkZvbnRSZW5kZXJDb250ZXh0OwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZU9wOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkltYWdlT2JzZXJ2ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmVuZGVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5yZW5kZXJhYmxlLlJlbmRlcmFibGVJbWFnZTsKLWltcG9ydCBqYXZhLnRleHQuQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yOwotaW1wb3J0IGphdmEudXRpbC5NYXA7Ci0KLS8qKgotICogVGhlIEdyYXBoaWNzMkQgY2xhc3MgZXh0ZW5kcyBHcmFwaGljcyBjbGFzcyBhbmQgcHJvdmlkZXMgbW9yZSBjYXBhYmlsaXRpZXMKLSAqIGZvciByZW5kZXJpbmcgdGV4dCwgaW1hZ2VzLCBzaGFwZXMuIFRoaXMgcHJvdmlkZXMgbWV0aG9kcyB0byBwZXJmb3JtCi0gKiB0cmFuc2Zvcm1hdGlvbiBvZiBjb29yZGluYXRlIHN5c3RlbSwgY29sb3IgbWFuYWdlbWVudCwgYW5kIHRleHQgbGF5b3V0LiBUaGUKLSAqIGZvbGxvd2luZyBhdHRyaWJ1dGVzIGV4aXN0IGZvciByZW5kZXJpbmc6Ci0gKiA8dWw+Ci0gKiA8bGk+Q29sb3IgLSBjdXJyZW50IEdyYXBoaWNzMkQgY29sb3I7PC9saT4KLSAqIDxsaT5Gb250IC0gY3VycmVudCBHcmFwaGljczJEIGZvbnQ7PC9saT4KLSAqIDxsaT5TdHJva2UgLSBwZW4gd2l0aCBhIHdpZHRoIG9mIDEgcGl4ZWw7PC9saT4KLSAqIDxsaT5UcmFuc2Zvcm0gLSBjdXJyZW50IEdyYXBoaWNzMkQgVHJhbnNmb3JtYXRpb247PC9saT4KLSAqIDxsaT5Db21wb3NpdGUgLSBhbHBoYSBjb21wb3NpdGluZyBydWxlcyBmb3IgY29tYmluaW5nIHNvdXJjZSBhbmQgZGVzdGluYXRpb24KLSAqIGNvbG9ycy48L2xpPgotICogPC91bD4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBHcmFwaGljczJEIGV4dGVuZHMgR3JhcGhpY3MgewotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEdyYXBoaWNzMkQgb2JqZWN0LiBUaGlzIGNvbnN0cnVjdG9yIHNob3VsZCBuZXZlciBiZQotICAgICAqIGNhbGxlZCBkaXJlY3RseS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgR3JhcGhpY3MyRCgpIHsKLSAgICAgICAgc3VwZXIoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHByZWZlcmVuY2VzIGZvciB0aGUgcmVuZGVyaW5nIGFsZ29yaXRobXMuIFRoZSBwcmVmZXJlbmNlcyBhcmUKLSAgICAgKiBhcmJpdHJhcnkgYW5kIHNwZWNpZmllZCBieSBNYXAgb2JqZWN0cy4gQWxsIHNwZWNpZmllZCBieSBNYXAgb2JqZWN0Ci0gICAgICogcHJlZmVyZW5jZXMgY2FuIGJlIG1vZGlmaWVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBoaW50cwotICAgICAqICAgICAgICAgICAgdGhlIHJlbmRlcmluZyBoaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBhZGRSZW5kZXJpbmdIaW50cyhNYXA8PywgPz4gaGludHMpOwotCi0gICAgLyoqCi0gICAgICogSW50ZXJzZWN0cyB0aGUgY3VycmVudCBjbGlwcGluZyBhcmVhIHdpdGggdGhlIHNwZWNpZmllZCBTaGFwZSBhbmQgdGhlCi0gICAgICogcmVzdWx0IGJlY29tZXMgYSBuZXcgY2xpcHBpbmcgYXJlYS4gSWYgY3VycmVudCBjbGlwcGluZyBhcmVhIGlzIG5vdAotICAgICAqIGRlZmluZWQsIHRoZSBTaGFwZSBiZWNvbWVzIHRoZSBuZXcgY2xpcHBpbmcgYXJlYS4gTm8gcmVuZGVyaW5nIG9wZXJhdGlvbnMKLSAgICAgKiBhcmUgYWxsb3dlZCBvdXRzaWRlIHRoZSBjbGlwcGluZyBhcmVhLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIFNoYXBlIG9iamVjdCB3aGljaCB3aWxsIGJlIGludGVyc2VjdGVkIHdpdGgKLSAgICAgKiAgICAgICAgICAgIGN1cnJlbnQgY2xpcHBpbmcgYXJlYS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBjbGlwKFNoYXBlIHMpOwotCi0gICAgLyoqCi0gICAgICogRHJhd3MgdGhlIG91dGxpbmUgb2YgdGhlIHNwZWNpZmllZCBTaGFwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIFNoYXBlIHdoaWNoIG91dGxpbmUgaXMgZHJhd24uCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhdyhTaGFwZSBzKTsKLQotICAgIC8qKgotICAgICAqIERyYXdzIHRoZSBzcGVjaWZpZWQgR2x5cGhWZWN0b3Igb2JqZWN0J3MgdGV4dCBhdCB0aGUgcG9pbnQgeCwgeS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZwotICAgICAqICAgICAgICAgICAgdGhlIEdseXBoVmVjdG9yIG9iamVjdCB0byBiZSBkcmF3bi4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggcG9zaXRpb24gd2hlcmUgdGhlIEdseXBoVmVjdG9yJ3MgdGV4dCBzaG91bGQgYmUKLSAgICAgKiAgICAgICAgICAgIHJlbmRlcmVkLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBwb3NpdGlvbiB3aGVyZSB0aGUgR2x5cGhWZWN0b3IncyB0ZXh0IHNob3VsZCBiZQotICAgICAqICAgICAgICAgICAgcmVuZGVyZWQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd0dseXBoVmVjdG9yKEdseXBoVmVjdG9yIGcsIGZsb2F0IHgsIGZsb2F0IHkpOwotCi0gICAgLyoqCi0gICAgICogRHJhd3MgdGhlIEJ1ZmZlcmVkSW1hZ2UgLS0gbW9kaWZpZWQgYWNjb3JkaW5nIHRvIHRoZSBvcGVyYXRpb24KLSAgICAgKiBCdWZmZXJlZEltYWdlT3AgLS0gYXQgdGhlIHBvaW50IHgsIHkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltZwotICAgICAqICAgICAgICAgICAgdGhlIEJ1ZmZlcmVkSW1hZ2UgdG8gYmUgcmVuZGVyZWQuCi0gICAgICogQHBhcmFtIG9wCi0gICAgICogICAgICAgICAgICB0aGUgZmlsdGVyIHRvIGJlIGFwcGxpZWQgdG8gdGhlIGltYWdlIGJlZm9yZSByZW5kZXJpbmcuCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHBvaW50IHdoZXJlIHRoZSBpbWFnZSdzIHVwcGVyIGxlZnQKLSAgICAgKiAgICAgICAgICAgIGNvcm5lciB3aWxsIGJlIHBsYWNlZC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcG9pbnQgd2hlcmUgdGhlIGltYWdlJ3MgdXBwZXIgbGVmdAotICAgICAqICAgICAgICAgICAgY29ybmVyIHdpbGwgYmUgcGxhY2VkLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGRyYXdJbWFnZShCdWZmZXJlZEltYWdlIGltZywgQnVmZmVyZWRJbWFnZU9wIG9wLCBpbnQgeCwgaW50IHkpOwotCi0gICAgLyoqCi0gICAgICogRHJhd3MgQnVmZmVyZWRJbWFnZSB0cmFuc2Zvcm1lZCBmcm9tIGltYWdlIHNwYWNlIGludG8gdXNlciBzcGFjZQotICAgICAqIGFjY29yZGluZyB0byB0aGUgQWZmaW5lVHJhbnNmb3JtIHhmb3JtIGFuZCBub3RpZmllcyB0aGUgSW1hZ2VPYnNlcnZlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1nCi0gICAgICogICAgICAgICAgICB0aGUgQnVmZmVyZWRJbWFnZSB0byBiZSByZW5kZXJlZC4KLSAgICAgKiBAcGFyYW0geGZvcm0KLSAgICAgKiAgICAgICAgICAgIHRoZSBhZmZpbmUgdHJhbnNmb3JtYXRpb24gZnJvbSB0aGUgaW1hZ2UgdG8gdGhlIHVzZXIgc3BhY2UuCi0gICAgICogQHBhcmFtIG9icwotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlT2JzZXJ2ZXIgdG8gYmUgbm90aWZpZWQgYWJvdXQgdGhlIGltYWdlIGNvbnZlcnNpb24uCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgaW1hZ2UgaXMgc3VjY2Vzc2Z1bGx5IGxvYWRlZCBhbmQgcmVuZGVyZWQsIG9yIGl0J3MKLSAgICAgKiAgICAgICAgIG51bGwsIG90aGVyd2lzZSBmYWxzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1nLCBBZmZpbmVUcmFuc2Zvcm0geGZvcm0sIEltYWdlT2JzZXJ2ZXIgb2JzKTsKLQotICAgIC8qKgotICAgICAqIERyYXdzIGEgUmVuZGVyYWJsZUltYWdlIHdoaWNoIGlzIHRyYW5zZm9ybWVkIGZyb20gaW1hZ2Ugc3BhY2UgaW50byB1c2VyCi0gICAgICogYWNjb3JkaW5nIHRvIHRoZSBBZmZpbmVUcmFuc2Zvcm0geGZvcm0uCi0gICAgICogCi0gICAgICogQHBhcmFtIGltZwotICAgICAqICAgICAgICAgICAgdGhlIFJlbmRlcmFibGVJbWFnZSB0byBiZSByZW5kZXJlZC4KLSAgICAgKiBAcGFyYW0geGZvcm0KLSAgICAgKiAgICAgICAgICAgIHRoZSBhZmZpbmUgdHJhbnNmb3JtYXRpb24gZnJvbSBpbWFnZSB0byB1c2VyIHNwYWNlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGRyYXdSZW5kZXJhYmxlSW1hZ2UoUmVuZGVyYWJsZUltYWdlIGltZywgQWZmaW5lVHJhbnNmb3JtIHhmb3JtKTsKLQotICAgIC8qKgotICAgICAqIERyYXdzIGEgUmVuZGVyZWRJbWFnZSB3aGljaCBpcyB0cmFuc2Zvcm1lZCBmcm9tIGltYWdlIHNwYWNlIGludG8gdXNlcgotICAgICAqIGFjY29yZGluZyB0byB0aGUgQWZmaW5lVHJhbnNmb3JtIHhmb3JtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWcKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJlZEltYWdlIHRvIGJlIHJlbmRlcmVkLgotICAgICAqIEBwYXJhbSB4Zm9ybQotICAgICAqICAgICAgICAgICAgdGhlIGFmZmluZSB0cmFuc2Zvcm1hdGlvbiBmcm9tIGltYWdlIHRvIHVzZXIgc3BhY2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd1JlbmRlcmVkSW1hZ2UoUmVuZGVyZWRJbWFnZSBpbWcsIEFmZmluZVRyYW5zZm9ybSB4Zm9ybSk7Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyB0aGUgc3RyaW5nIHNwZWNpZmllZCBieSB0aGUgQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yLiBUaGUgZmlyc3QKLSAgICAgKiBjaGFyYWN0ZXIncyBwb3NpdGlvbiBpcyBzcGVjaWZpZWQgYnkgdGhlIFgsIFkgcGFyYW1ldGVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaXRlcmF0b3IKLSAgICAgKiAgICAgICAgICAgIHdob3NlIHRleHQgaXMgZHJhd24uCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIHBvc2l0aW9uIHdoZXJlIHRoZSBmaXJzdCBjaGFyYWN0ZXIgaXMgZHJhd24uCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIHBvc2l0aW9uIHdoZXJlIHRoZSBmaXJzdCBjaGFyYWN0ZXIgaXMgZHJhd24uCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd1N0cmluZyhBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgaXRlcmF0b3IsIGZsb2F0IHgsIGZsb2F0IHkpOwotCi0gICAgLyoqCi0gICAgICogRHJhd3MgdGhlIHN0cmluZyBzcGVjaWZpZWQgYnkgdGhlIEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvci4gVGhlIGZpcnN0Ci0gICAgICogY2hhcmFjdGVyJ3MgcG9zaXRpb24gaXMgc3BlY2lmaWVkIGJ5IHRoZSBYLCBZIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGl0ZXJhdG9yCi0gICAgICogICAgICAgICAgICB3aG9zZSB0ZXh0IGlzIGRyYXduLgotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBwb3NpdGlvbiB3aGVyZSB0aGUgZmlyc3QgY2hhcmFjdGVyIGlzIGRyYXduLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBwb3NpdGlvbiB3aGVyZSB0aGUgZmlyc3QgY2hhcmFjdGVyIGlzIGRyYXduLgotICAgICAqIEBzZWUgamF2YS5hd3QuR3JhcGhpY3MjZHJhd1N0cmluZyhBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IsIGludCwgaW50KQotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGRyYXdTdHJpbmcoQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIGl0ZXJhdG9yLCBpbnQgeCwgaW50IHkpOwotCi0gICAgLyoqCi0gICAgICogRHJhd3MgdGhlIFN0cmluZyB3aG9zZSB0aGUgZmlyc3QgY2hhcmFjdGVyIHBvc2l0aW9uIGlzIHNwZWNpZmllZCBieSB0aGUKLSAgICAgKiBwYXJhbWV0ZXJzIFgsIFkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBTdHJpbmcgdG8gYmUgZHJhd24uCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIHBvc2l0aW9uIG9mIHRoZSBmaXJzdCBjaGFyYWN0ZXIuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIHBvc2l0aW9uIG9mIHRoZSBmaXJzdCBjaGFyYWN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd1N0cmluZyhTdHJpbmcgcywgZmxvYXQgeCwgZmxvYXQgeSk7Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyB0aGUgU3RyaW5nIHdob3NlIHRoZSBmaXJzdCBjaGFyYWN0ZXIgY29vcmRpbmF0ZXMgYXJlIHNwZWNpZmllZCBieQotICAgICAqIHRoZSBwYXJhbWV0ZXJzIFgsIFkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cgotICAgICAqICAgICAgICAgICAgdGhlIFN0cmluZyB0byBiZSBkcmF3bi4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY2hhcmFjdGVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjaGFyYWN0ZXIuCi0gICAgICogQHNlZSBqYXZhLmF3dC5HcmFwaGljcyNkcmF3U3RyaW5nKFN0cmluZywgaW50LCBpbnQpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd1N0cmluZyhTdHJpbmcgc3RyLCBpbnQgeCwgaW50IHkpOwotCi0gICAgLyoqCi0gICAgICogRmlsbHMgdGhlIGludGVyaW9yIG9mIHRoZSBzcGVjaWZpZWQgU2hhcGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBTaGFwZSB0byBiZSBmaWxsZWQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZmlsbChTaGFwZSBzKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJhY2tncm91bmQgY29sb3IuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY3VycmVudCBiYWNrZ3JvdW5kIGNvbG9yLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBDb2xvciBnZXRCYWNrZ3JvdW5kKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBjdXJyZW50IGNvbXBvc2l0ZSBvZiB0aGUgR3JhcGhpY3MyRC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjdXJyZW50IGNvbXBvc2l0ZSB3aGljaCBzcGVjaWZpZXMgdGhlIGNvbXBvc2l0aW5nIHN0eWxlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBDb21wb3NpdGUgZ2V0Q29tcG9zaXRlKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkZXZpY2UgY29uZmlndXJhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkZXZpY2UgY29uZmlndXJhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgR3JhcGhpY3NDb25maWd1cmF0aW9uIGdldERldmljZUNvbmZpZ3VyYXRpb24oKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHJlbmRlcmluZyBjb250ZXh0IG9mIHRoZSBGb250LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBGb250UmVuZGVyQ29udGV4dCBnZXRGb250UmVuZGVyQ29udGV4dCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY3VycmVudCBQYWludCBvZiBHcmFwaGljczJELgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGN1cnJlbnQgUGFpbnQgb2YgR3JhcGhpY3MyRC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgUGFpbnQgZ2V0UGFpbnQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHZhbHVlIG9mIHNpbmdsZSBwcmVmZXJlbmNlIGZvciBzcGVjaWZpZWQga2V5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBrZXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQga2V5IG9mIHRoZSByZW5kZXJpbmcgaGludC4KLSAgICAgKiBAcmV0dXJuIHRoZSB2YWx1ZSBvZiByZW5kZXJpbmcgaGludCBmb3Igc3BlY2lmaWVkIGtleS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgT2JqZWN0IGdldFJlbmRlcmluZ0hpbnQoUmVuZGVyaW5nSGludHMuS2V5IGtleSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzZXQgb2YgdGhlIHJlbmRlcmluZyBwcmVmZXJlbmNlcyBhcyBhIGNvbGxlY3Rpb24gb2Yga2V5L3ZhbHVlCi0gICAgICogcGFpcnMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgUmVuZGVyaW5nSGludHMgd2hpY2ggY29udGFpbnMgdGhlIHJlbmRlcmluZyBwcmVmZXJlbmNlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgUmVuZGVyaW5nSGludHMgZ2V0UmVuZGVyaW5nSGludHMoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgY3VycmVudCBzdHJva2Ugb2YgdGhlIEdyYXBoaWNzMkQuCi0gICAgICogCi0gICAgICogQHJldHVybiBjdXJyZW50IHN0cm9rZSBvZiB0aGUgR3JhcGhpY3MyRC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgU3Ryb2tlIGdldFN0cm9rZSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyBjdXJyZW50IGFmZmluZSB0cmFuc2Zvcm0gb2YgdGhlIEdyYXBoaWNzMkQuCi0gICAgICogCi0gICAgICogQHJldHVybiBjdXJyZW50IEFmZmluZVRyYW5zZm9ybSBvZiB0aGUgR3JhcGhpY3MyRC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgQWZmaW5lVHJhbnNmb3JtIGdldFRyYW5zZm9ybSgpOwotCi0gICAgLyoqCi0gICAgICogRGV0ZXJtaW5lcyB3aGV0aGVyIG9yIG5vdCB0aGUgc3BlY2lmaWVkIFNoYXBlIGludGVyc2VjdHMgdGhlIHNwZWNpZmllZAotICAgICAqIFJlY3RhbmdsZS4gSWYgdGhlIG9uU3Ryb2tlIHBhcmFtZXRlciBpcyB0cnVlLCB0aGlzIG1ldGhvZCBjaGVja3Mgd2hldGhlcgotICAgICAqIG9yIG5vdCB0aGUgc3BlY2lmaWVkIFNoYXBlIG91dGxpbmUgaW50ZXJzZWN0cyB0aGUgc3BlY2lmaWVkIFJlY3RhbmdsZSwKLSAgICAgKiBvdGhlcndpc2UgdGhpcyBtZXRob2QgY2hlY2tzIHdoZXRoZXIgb3Igbm90IHRoZSBzcGVjaWZpZWQgU2hhcGUncwotICAgICAqIGludGVyaW9yIGludGVyc2VjdHMgdGhlIHNwZWNpZmllZCBSZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJlY3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgUmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgU2hhcGUgdG8gY2hlY2sgZm9yIGludGVyc2VjdGlvbi4KLSAgICAgKiBAcGFyYW0gb25TdHJva2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBwYXJhbWV0ZXIgZGV0ZXJtaW5lcyB3aGV0aGVyIG9yIG5vdCB0aGlzIG1ldGhvZCBjaGVja3MgZm9yCi0gICAgICogICAgICAgICAgICBpbnRlcnNlY3Rpb24gb2YgdGhlIFNoYXBlIG91dGxpbmUgb3Igb2YgdGhlIFNoYXBlIGludGVyaW9yCi0gICAgICogICAgICAgICAgICB3aXRoIHRoZSBSZWN0YW5nbGUuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGVyZSBpcyBhIGhpdCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBib29sZWFuIGhpdChSZWN0YW5nbGUgcmVjdCwgU2hhcGUgcywgYm9vbGVhbiBvblN0cm9rZSk7Ci0KLSAgICAvKioKLSAgICAgKiBQZXJmb3JtcyBhIHJvdGF0aW9uIHRyYW5zZm9ybSByZWxhdGl2ZSB0byBjdXJyZW50IEdyYXBoaWNzMkQgVHJhbnNmb3JtLgotICAgICAqIFRoZSBjb29yZGluYXRlIHN5c3RlbSBpcyByb3RhdGVkIGJ5IHRoZSBzcGVjaWZpZWQgYW5nbGUgaW4gcmFkaWFucwotICAgICAqIHJlbGF0aXZlIHRvIGN1cnJlbnQgb3JpZ2luLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0aGV0YQotICAgICAqICAgICAgICAgICAgdGhlIGFuZ2xlIG9mIHJvdGF0aW9uIGluIHJhZGlhbnMuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgcm90YXRlKGRvdWJsZSB0aGV0YSk7Ci0KLSAgICAvKioKLSAgICAgKiBQZXJmb3JtcyBhIHRyYW5zbGF0ZWQgcm90YXRpb24gdHJhbnNmb3JtIHJlbGF0aXZlIHRvIGN1cnJlbnQgR3JhcGhpY3MyRAotICAgICAqIFRyYW5zZm9ybS4gVGhlIGNvb3JkaW5hdGUgc3lzdGVtIGlzIHJvdGF0ZWQgYnkgdGhlIHNwZWNpZmllZCBhbmdsZSBpbgotICAgICAqIHJhZGlhbnMgcmVsYXRpdmUgdG8gY3VycmVudCBvcmlnaW4gYW5kIHRoZW4gbW92ZWQgdG8gcG9pbnQgKHgsIHkpLiBJcwotICAgICAqIHRoaXMgcmlnaHQ/Ci0gICAgICogCi0gICAgICogQHBhcmFtIHRoZXRhCi0gICAgICogICAgICAgICAgICB0aGUgYW5nbGUgb2Ygcm90YXRpb24gaW4gcmFkaWFucy4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCByb3RhdGUoZG91YmxlIHRoZXRhLCBkb3VibGUgeCwgZG91YmxlIHkpOwotCi0gICAgLyoqCi0gICAgICogUGVyZm9ybXMgYSBsaW5lYXIgc2NhbGUgdHJhbnNmb3JtIHJlbGF0aXZlIHRvIGN1cnJlbnQgR3JhcGhpY3MyRAotICAgICAqIFRyYW5zZm9ybS4gVGhlIGNvb3JkaW5hdGUgc3lzdGVtIGlzIHJlc2NhbGVkIHZlcnRpY2FsbHkgYW5kIGhvcml6b250YWxseQotICAgICAqIGJ5IHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3gKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FsaW5nIGZhY3RvciBieSB3aGljaCB0aGUgWCBjb29yZGluYXRlIGlzIG11bHRpcGxpZWQuCi0gICAgICogQHBhcmFtIHN5Ci0gICAgICogICAgICAgICAgICB0aGUgc2NhbGluZyBmYWN0b3IgYnkgd2hpY2ggdGhlIFkgY29vcmRpbmF0ZSBpcyBtdWx0aXBsaWVkLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNjYWxlKGRvdWJsZSBzeCwgZG91YmxlIHN5KTsKLQotICAgIC8qKgotICAgICAqIFNldHMgYSBuZXcgYmFja2dyb3VuZCBjb2xvciBmb3IgY2xlYXJpbmcgcmVjdGFuZ3VsYXIgYXJlYXMuIFRoZSBjbGVhclJlY3QKLSAgICAgKiBtZXRob2QgdXNlcyB0aGUgY3VycmVudCBiYWNrZ3JvdW5kIGNvbG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvcgotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBiYWNrZ3JvdW5kIGNvbG9yLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldEJhY2tncm91bmQoQ29sb3IgY29sb3IpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgY3VycmVudCBjb21wb3NpdGUgZm9yIEdyYXBoaWNzMkQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbXAKLSAgICAgKiAgICAgICAgICAgIHRoZSBDb21wb3NpdGUgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldENvbXBvc2l0ZShDb21wb3NpdGUgY29tcCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBwYWludCBmb3IgR3JhcGhpY3MyRC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcGFpbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBQYWludCBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0UGFpbnQoUGFpbnQgcGFpbnQpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyBhIGtleS12YWx1ZSBwYWlyIGluIHRoZSBjdXJyZW50IFJlbmRlcmluZ0hpbnRzIG1hcC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0ga2V5Ci0gICAgICogICAgICAgICAgICB0aGUga2V5IG9mIHRoZSByZW5kZXJpbmcgaGludCB0byBzZXQuCi0gICAgICogQHBhcmFtIHZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgdmFsdWUgdG8gc2V0IGZvciB0aGUgcmVuZGVyaW5nIGhpbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0UmVuZGVyaW5nSGludChSZW5kZXJpbmdIaW50cy5LZXkga2V5LCBPYmplY3QgdmFsdWUpOwotCi0gICAgLyoqCi0gICAgICogUmVwbGFjZXMgdGhlIGN1cnJlbnQgcmVuZGVyaW5nIGhpbnRzIHdpdGggdGhlIHNwZWNpZmllZCByZW5kZXJpbmcKLSAgICAgKiBwcmVmZXJlbmNlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaGludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgTWFwIG9mIHJlbmRlcmluZyBoaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzZXRSZW5kZXJpbmdIaW50cyhNYXA8PywgPz4gaGludHMpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc3Ryb2tlIGZvciB0aGUgR3JhcGhpY3MyRC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIFN0cm9rZSBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0U3Ryb2tlKFN0cm9rZSBzKTsKLQotICAgIC8qKgotICAgICAqIE92ZXJ3cml0ZSB0aGUgY3VycmVudCBUcmFuc2Zvcm0gb2YgdGhlIEdyYXBoaWNzMkQuIFRoZSBzcGVjaWZpZWQKLSAgICAgKiBUcmFuc2Zvcm0gc2hvdWxkIGJlIHJlY2VpdmVkIGZyb20gdGhlIGdldFRyYW5zZm9ybSgpIG1ldGhvZCBhbmQgc2hvdWxkIGJlCi0gICAgICogdXNlZCBvbmx5IGZvciByZXN0b3JpbmcgdGhlIG9yaWdpbmFsIEdyYXBoaWNzMkQgdHJhbnNmb3JtIGFmdGVyIGNhbGxpbmcKLSAgICAgKiBkcmF3IG9yIGZpbGwgbWV0aG9kcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gVHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgVHJhbnNmb3JtLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldFRyYW5zZm9ybShBZmZpbmVUcmFuc2Zvcm0gVHgpOwotCi0gICAgLyoqCi0gICAgICogUGVyZm9ybXMgYSBzaGVhciB0cmFuc2Zvcm0gcmVsYXRpdmUgdG8gY3VycmVudCBHcmFwaGljczJEIFRyYW5zZm9ybS4gVGhlCi0gICAgICogY29vcmRpbmF0ZSBzeXN0ZW0gaXMgc2hpZnRlZCBieSB0aGUgc3BlY2lmaWVkIG11bHRpcGxpZXJzIHJlbGF0aXZlIHRvCi0gICAgICogY3VycmVudCBwb3NpdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2h4Ci0gICAgICogICAgICAgICAgICB0aGUgbXVsdGlwbGllciBieSB3aGljaCB0aGUgWCBjb29yZGluYXRlcyBzaGlmdCBwb3NpdGlvbiBhbG9uZwotICAgICAqICAgICAgICAgICAgWCBheGlzIGFzIGEgZnVuY3Rpb24gb2YgWSBjb29yZGluYXRlcy4KLSAgICAgKiBAcGFyYW0gc2h5Ci0gICAgICogICAgICAgICAgICB0aGUgbXVsdGlwbGllciBieSB3aGljaCB0aGUgWSBjb29yZGluYXRlcyBzaGlmdCBwb3NpdGlvbiBhbG9uZwotICAgICAqICAgICAgICAgICAgWSBheGlzIGFzIGEgZnVuY3Rpb24gb2YgWCBjb29yZGluYXRlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzaGVhcihkb3VibGUgc2h4LCBkb3VibGUgc2h5KTsKLQotICAgIC8qKgotICAgICAqIENvbmNhdGVuYXRlcyB0aGUgQWZmaW5lVHJhbnNmb3JtIG9iamVjdCB3aXRoIGN1cnJlbnQgVHJhbnNmb3JtIG9mIHRoaXMKLSAgICAgKiBHcmFwaGljczJELiBUaGUgdHJhbnNmb3JtcyBhcmUgYXBwbGllZCBpbiByZXZlcnNlIG9yZGVyIHdpdGggdGhlIGxhc3QKLSAgICAgKiBzcGVjaWZpZWQgdHJhbnNmb3JtIGFwcGxpZWQgZmlyc3QgYW5kIHRoZSBuZXh0IHRyYW5zZm9ybWF0aW9uIGFwcGxpZWQgdG8KLSAgICAgKiB0aGUgcmVzdWx0IG9mIHByZXZpb3VzIHRyYW5zZm9ybWF0aW9uLiBNb3JlIHByZWNpc2VseSwgaWYgQ3ggaXMgdGhlCi0gICAgICogY3VycmVudCBHcmFwaGljczJEIHRyYW5zZm9ybSwgdGhlIHRyYW5zZm9ybSBtZXRob2QncyByZXN1bHQgd2l0aCBUeCBhcwotICAgICAqIHRoZSBwYXJhbWV0ZXIgaXMgdGhlIHRyYW5zZm9ybWF0aW9uIFJ4LCB3aGVyZSBSeChwKSA9IEN4KFR4KHApKSwgZm9yIHAgLQotICAgICAqIGEgcG9pbnQgaW4gY3VycmVudCBjb29yZGluYXRlIHN5c3RlbS4gUnggYmVjb21lcyB0aGUgY3VycmVudCBUcmFuc2Zvcm0KLSAgICAgKiBmb3IgdGhpcyBHcmFwaGljczJELgotICAgICAqIAotICAgICAqIEBwYXJhbSBUeAotICAgICAqICAgICAgICAgICAgdGhlIEFmZmluZVRyYW5zZm9ybSBvYmplY3QgdG8gYmUgY29uY2F0ZW5hdGVkIHdpdGggY3VycmVudAotICAgICAqICAgICAgICAgICAgVHJhbnNmb3JtLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHRyYW5zZm9ybShBZmZpbmVUcmFuc2Zvcm0gVHgpOwotCi0gICAgLyoqCi0gICAgICogUGVyZm9ybXMgYSB0cmFuc2xhdGUgdHJhbnNmb3JtIHJlbGF0aXZlIHRvIGN1cnJlbnQgR3JhcGhpY3MyRCBUcmFuc2Zvcm0uCi0gICAgICogVGhlIGNvb3JkaW5hdGUgc3lzdGVtIGlzIG1vdmVkIGJ5IHRoZSBzcGVjaWZpZWQgZGlzdGFuY2UgcmVsYXRpdmUgdG8KLSAgICAgKiBjdXJyZW50IHBvc2l0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0eAotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zbGF0aW9uIGRpc3RhbmNlIGFsb25nIHRoZSBYIGF4aXMuCi0gICAgICogQHBhcmFtIHR5Ci0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNsYXRpb24gZGlzdGFuY2UgYWxvbmcgdGhlIFkgYXhpcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCB0cmFuc2xhdGUoZG91YmxlIHR4LCBkb3VibGUgdHkpOwotCi0gICAgLyoqCi0gICAgICogTW92ZXMgdGhlIG9yaWdpbiBHcmFwaGljczJEIFRyYW5zZm9ybSB0byB0aGUgcG9pbnQgd2l0aCB4LCB5IGNvb3JkaW5hdGVzCi0gICAgICogaW4gY3VycmVudCBjb29yZGluYXRlIHN5c3RlbS4gVGhlIG5ldyBvcmlnaW4gb2YgY29vcmRpbmF0ZSBzeXN0ZW0gaXMKLSAgICAgKiBtb3ZlZCB0byB0aGUgKHgsIHkpIHBvaW50IGFjY29yZGluZ2x5LiBBbGwgcmVuZGVyaW5nIGFuZCB0cmFuc2Zvcm0KLSAgICAgKiBvcGVyYXRpb25zIGFyZSBwZXJmb3JtZWQgcmVsYXRpdmUgdG8gdGhpcyBuZXcgb3JpZ2luLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlLgotICAgICAqIEBzZWUgamF2YS5hd3QuR3JhcGhpY3MjdHJhbnNsYXRlKGludCwgaW50KQotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHRyYW5zbGF0ZShpbnQgeCwgaW50IHkpOwotCi0gICAgLyoqCi0gICAgICogRmlsbHMgYSAzRCByZWN0YW5nbGUgd2l0aCB0aGUgY3VycmVudCBjb2xvci4gVGhlIHJlY3RhbmdsZSBpcyBzcGVjaWZpZWQKLSAgICAgKiBieSBpdHMgd2lkdGgsIGhlaWdodCwgYW5kIHRvcCBsZWZ0IGNvcm5lciBjb29yZGluYXRlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdG9wIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB0b3AgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSByYWlzZWQKLSAgICAgKiAgICAgICAgICAgIGEgYm9vbGVhbiB2YWx1ZSB0aGF0IGRldGVybWluZXMgd2hldGhlciB0aGUgcmVjdGFuZ2xlIGlzIGRyYXduCi0gICAgICogICAgICAgICAgICBhcyByYWlzZWQgb3IgaW5kZW50ZWQuCi0gICAgICogQHNlZSBqYXZhLmF3dC5HcmFwaGljcyNmaWxsM0RSZWN0KGludCwgaW50LCBpbnQsIGludCwgYm9vbGVhbikKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmaWxsM0RSZWN0KGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBib29sZWFuIHJhaXNlZCkgewotICAgICAgICAvLyBBY2NvcmRpbmcgdG8gdGhlIHNwZWMsIGNvbG9yIHNob3VsZCBiZSB1c2VkIGluc3RlYWQgb2YgcGFpbnQsCi0gICAgICAgIC8vIHNvIEdyYXBoaWNzLmZpbGwzRFJlY3QgcmVzZXRzIHBhaW50IGFuZAotICAgICAgICAvLyBpdCBzaG91bGQgYmUgcmVzdG9yZWQgYWZ0ZXIgdGhlIGNhbGwKLSAgICAgICAgUGFpbnQgc2F2ZWRQYWludCA9IGdldFBhaW50KCk7Ci0gICAgICAgIHN1cGVyLmZpbGwzRFJlY3QoeCwgeSwgd2lkdGgsIGhlaWdodCwgcmFpc2VkKTsKLSAgICAgICAgc2V0UGFpbnQoc2F2ZWRQYWludCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRHJhd3MgdGhlIGhpZ2hsaWdodGVkIG91dGxpbmUgb2YgYSByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHRvcCBsZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdG9wIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gcmFpc2VkCi0gICAgICogICAgICAgICAgICBhIGJvb2xlYW4gdmFsdWUgdGhhdCBkZXRlcm1pbmVzIHdoZXRoZXIgdGhlIHJlY3RhbmdsZSBpcyBkcmF3bgotICAgICAqICAgICAgICAgICAgYXMgcmFpc2VkIG9yIGluZGVudGVkLgotICAgICAqIEBzZWUgamF2YS5hd3QuR3JhcGhpY3MjZHJhdzNEUmVjdChpbnQsIGludCwgaW50LCBpbnQsIGJvb2xlYW4pCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhdzNEUmVjdChpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCwgYm9vbGVhbiByYWlzZWQpIHsKLSAgICAgICAgLy8gQWNjb3JkaW5nIHRvIHRoZSBzcGVjLCBjb2xvciBzaG91bGQgYmUgdXNlZCBpbnN0ZWFkIG9mIHBhaW50LAotICAgICAgICAvLyBzbyBHcmFwaGljcy5kcmF3M0RSZWN0IHJlc2V0cyBwYWludCBhbmQKLSAgICAgICAgLy8gaXQgc2hvdWxkIGJlIHJlc3RvcmVkIGFmdGVyIHRoZSBjYWxsCi0gICAgICAgIFBhaW50IHNhdmVkUGFpbnQgPSBnZXRQYWludCgpOwotICAgICAgICBzdXBlci5kcmF3M0RSZWN0KHgsIHksIHdpZHRoLCBoZWlnaHQsIHJhaXNlZCk7Ci0gICAgICAgIHNldFBhaW50KHNhdmVkUGFpbnQpOwotICAgIH0KLX0KXCBObyBuZXdsaW5lIGF0IGVuZCBvZiBmaWxlCmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvR3JhcGhpY3NDb25maWd1cmF0aW9uLmphdmEgYi9hd3QvamF2YS9hd3QvR3JhcGhpY3NDb25maWd1cmF0aW9uLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGQ1OWU4OTYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0dyYXBoaWNzQ29uZmlndXJhdGlvbi5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjI2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgQWxleGV5IEEuIFBldHJlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Wb2xhdGlsZUltYWdlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBjbGFzcyBjb250YWlucyB0aGUgY2hhcmFjdGVyaXN0aWNzIG9mIGdyYXBoaWNzCi0gKiBkZXZpY2VzIHN1Y2ggYXMgYSBwcmludGVyIG9yIG1vbml0b3IsIGFuZCByZXByZXNlbnRzIGRldmljZSdzIGNhcGFiaWxpdGllcwotICogYW5kIG1vZGVzLiBNYW55IEdyYXBoaWNzQ29uZmlndXJhdGlvbiBvYmplY3RzIGNhbiBiZSBhc3NvY2lhdGVkIHdpdGggc2luZ2xlCi0gKiBncmFwaGljcyBkZXZpY2UuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgR3JhcGhpY3NDb25maWd1cmF0aW9uIHsKLQotICAgIC8qKgotICAgICAqIENvbnN0cnVjdG9yIGNvdWxkIG5vdCBiZSB1c2VkIGRpcmVjdGx5IGFuZCBzaG91bGQgYmUgb2J0YWluZWQgaW4gZXh0ZW5kZWQKLSAgICAgKiBjbGFzc2VzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBHcmFwaGljc0NvbmZpZ3VyYXRpb24oKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBCdWZmZXJlZEltYWdlIGltYWdlIG9iamVjdCB3aXRoIGEgZGF0YSBsYXlvdXQgYW5kIGNvbG9yIG1vZGVsCi0gICAgICogY29tcGF0aWJsZSB3aXRoIHRoaXMgR3JhcGhpY3NDb25maWd1cmF0aW9uIHdpdGggc3BlY2lmaWVkIHdpZHRoIGFuZAotICAgICAqIGhlaWdodCBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBCdWZmZXJlZEltYWdlLgotICAgICAqIEByZXR1cm4gdGhlIEJ1ZmZlcmVkSW1hZ2Ugb2JqZWN0IHdpdGggc3BlY2lmaWVkIHdpZHRoIGFuZCBoZWlnaHQKLSAgICAgKiAgICAgICAgIHBhcmFtZXRlcnMuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEJ1ZmZlcmVkSW1hZ2UgY3JlYXRlQ29tcGF0aWJsZUltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCk7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgQnVmZmVyZWRJbWFnZSB0aGF0IGhhcyB0aGUgc3BlY2lmaWVkIHdpZHRoLCBoZWlnaHQsCi0gICAgICogdHJhbnNwYXJlbmN5IGFuZCBoYXMgYSBkYXRhIGxheW91dCBhbmQgY29sb3IgbW9kZWwgY29tcGF0aWJsZSB3aXRoIHRoaXMKLSAgICAgKiBHcmFwaGljc0NvbmZpZ3VyYXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgaW1hZ2UuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBpbWFnZS4KLSAgICAgKiBAcGFyYW0gdHJhbnNwYXJlbmN5Ci0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNwYXJlbmN5IG1vZGUuCi0gICAgICogQHJldHVybiB0aGUgQnVmZmVyZWRJbWFnZSBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEJ1ZmZlcmVkSW1hZ2UgY3JlYXRlQ29tcGF0aWJsZUltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IHRyYW5zcGFyZW5jeSk7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgVm9sYXRpbGVJbWFnZSB0aGF0IGhhcyB0aGUgc3BlY2lmaWVkIHdpZHRoIGFuZCBoZWlnaHQgYW5kIGhhcyBhCi0gICAgICogZGF0YSBsYXlvdXQgYW5kIGNvbG9yIG1vZGVsIGNvbXBhdGlibGUgd2l0aCB0aGlzIEdyYXBoaWNzQ29uZmlndXJhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIGltYWdlLgotICAgICAqIEByZXR1cm4gdGhlIFZvbGF0aWxlSW1hZ2Ugb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBWb2xhdGlsZUltYWdlIGNyZWF0ZUNvbXBhdGlibGVWb2xhdGlsZUltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCk7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgVm9sYXRpbGVJbWFnZSB0aGF0IHN1cHBvcnRzIHRoZSBzcGVjaWZpZWQgd2lkdGgsIGhlaWdodCwKLSAgICAgKiB0cmFuc3BhcmVuY3kgYW5kIGhhcyBhIGRhdGEgbGF5b3V0IGFuZCBjb2xvciBtb2RlbCBjb21wYXRpYmxlIHdpdGggdGhpcwotICAgICAqIEdyYXBoaWNzQ29uZmlndXJhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIGltYWdlLgotICAgICAqIEBwYXJhbSB0cmFuc3BhcmVuY3kKLSAgICAgKiAgICAgICAgICAgIHRoZSB0cmFuc3BhcmVuY3kgbW9kZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBWb2xhdGlsZUltYWdlIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgVm9sYXRpbGVJbWFnZSBjcmVhdGVDb21wYXRpYmxlVm9sYXRpbGVJbWFnZShpbnQgd2lkdGgsIGludCBoZWlnaHQsCi0gICAgICAgICAgICBpbnQgdHJhbnNwYXJlbmN5KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJvdW5kcyBvZiBhcmVhIGNvdmVyZWQgYnkgdGhlIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBpbiB0aGUKLSAgICAgKiBkZXZpY2UgY29vcmRpbmF0ZXMgc3BhY2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgUmVjdGFuZ2xlIG9mIEdyYXBoaWNzQ29uZmlndXJhdGlvbidzIGJvdW5kcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgUmVjdGFuZ2xlIGdldEJvdW5kcygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQ29sb3JNb2RlbCBvZiB0aGUgR3JhcGhpY3NDb25maWd1cmF0aW9uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIENvbG9yTW9kZWwgb2JqZWN0IG9mIHRoZSBHcmFwaGljc0NvbmZpZ3VyYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IENvbG9yTW9kZWwgZ2V0Q29sb3JNb2RlbCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQ29sb3JNb2RlbCBvZiB0aGUgR3JhcGhpY3NDb25maWd1cmF0aW9uIHdoaWNoIHN1cHBvcnRzIHNwZWNpZmllZAotICAgICAqIFRyYW5zcGFyZW5jeS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdHJhbnNwYXJlbmN5Ci0gICAgICogICAgICAgICAgICB0aGUgVHJhbnNwYXJlbmN5IG1vZGU6IE9QQVFVRSwgQklUTUFTSywgb3IgVFJBTlNMVUNFTlQuCi0gICAgICogQHJldHVybiB0aGUgQ29sb3JNb2RlbCBvZiB0aGUgR3JhcGhpY3NDb25maWd1cmF0aW9uIHdoaWNoIHN1cHBvcnRzCi0gICAgICogICAgICAgICBzcGVjaWZpZWQgVHJhbnNwYXJlbmN5LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoaW50IHRyYW5zcGFyZW5jeSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkZWZhdWx0IEFmZmluZVRyYW5zZm9ybSBvZiB0aGUgR3JhcGhpY3NDb25maWd1cmF0aW9uLiBUaGlzCi0gICAgICogbWV0aG9kIHRyYW5zbGF0ZXMgdXNlciBjb29yZGluYXRlcyB0byBkZXZpY2UgY29vcmRpbmF0ZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGVmYXVsdCBBZmZpbmVUcmFuc2Zvcm0gb2YgdGhlIEdyYXBoaWNzQ29uZmlndXJhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgQWZmaW5lVHJhbnNmb3JtIGdldERlZmF1bHRUcmFuc2Zvcm0oKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEdyYXBoaWNzRGV2aWNlIG9mIHRoZSBHcmFwaGljc0NvbmZpZ3VyYXRpb24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgR3JhcGhpY3NEZXZpY2Ugb2YgdGhlIEdyYXBoaWNzQ29uZmlndXJhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgR3JhcGhpY3NEZXZpY2UgZ2V0RGV2aWNlKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBub3JtYWxpemluZyBBZmZpbmVUcmFuc2Zvcm0gb2YgdGhlIEdyYXBoaWNzQ29uZmlndXJhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBub3JtYWxpemluZyBBZmZpbmVUcmFuc2Zvcm0gb2YgdGhlIEdyYXBoaWNzQ29uZmlndXJhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgQWZmaW5lVHJhbnNmb3JtIGdldE5vcm1hbGl6aW5nVHJhbnNmb3JtKCk7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIFZvbGF0aWxlSW1hZ2Ugd2l0aCBzcGVjaWZpZWQgd2lkdGgsIGhlaWdodCwgSW1hZ2VDYXBhYmlsaXRpZXM7IGEKLSAgICAgKiBkYXRhIGxheW91dCBhbmQgY29sb3IgbW9kZWwgY29tcGF0aWJsZSB3aXRoIHRoaXMgR3JhcGhpY3NDb25maWd1cmF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIGltYWdlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgaW1hZ2UuCi0gICAgICogQHBhcmFtIGNhcHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZUNhcGFiaWxpdGllcyBvYmplY3QuCi0gICAgICogQHJldHVybiB0aGUgVm9sYXRpbGVJbWFnZSB3aGljaCBkYXRhIGxheW91dCBhbmQgY29sb3IgbW9kZWwgY29tcGF0aWJsZQotICAgICAqICAgICAgICAgd2l0aCB0aGlzIEdyYXBoaWNzQ29uZmlndXJhdGlvbi4KLSAgICAgKiBAdGhyb3dzIEFXVEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIEltYWdlQ2FwYWJpbGl0aWVzIGlzIG5vdCBzdXBwb3J0ZWQgYnkgdGhlCi0gICAgICogICAgICAgICAgICAgR3JhcGhpY3NDb25maWd1cmF0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBWb2xhdGlsZUltYWdlIGNyZWF0ZUNvbXBhdGlibGVWb2xhdGlsZUltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCwgSW1hZ2VDYXBhYmlsaXRpZXMgY2FwcykKLSAgICAgICAgICAgIHRocm93cyBBV1RFeGNlcHRpb24gewotICAgICAgICBWb2xhdGlsZUltYWdlIHJlcyA9IGNyZWF0ZUNvbXBhdGlibGVWb2xhdGlsZUltYWdlKHdpZHRoLCBoZWlnaHQpOwotICAgICAgICBpZiAoIXJlcy5nZXRDYXBhYmlsaXRpZXMoKS5lcXVhbHMoY2FwcykpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xNEE9Q2FuIG5vdCBjcmVhdGUgVm9sYXRpbGVJbWFnZSB3aXRoIHNwZWNpZmllZCBjYXBhYmlsaXRpZXMKLSAgICAgICAgICAgIHRocm93IG5ldyBBV1RFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTRBIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgVm9sYXRpbGVJbWFnZSB3aXRoIHNwZWNpZmllZCB3aWR0aCwgaGVpZ2h0LCB0cmFuc3BhcmVuY3kgYW5kCi0gICAgICogSW1hZ2VDYXBhYmlsaXRpZXM7IGEgZGF0YSBsYXlvdXQgYW5kIGNvbG9yIG1vZGVsIGNvbXBhdGlibGUgd2l0aCB0aGlzCi0gICAgICogR3JhcGhpY3NDb25maWd1cmF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIGltYWdlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgaW1hZ2UuCi0gICAgICogQHBhcmFtIGNhcHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZUNhcGFiaWxpdGllcyBvYmplY3QuCi0gICAgICogQHBhcmFtIHRyYW5zcGFyZW5jeQotICAgICAqICAgICAgICAgICAgdGhlIFRyYW5zcGFyZW5jeSBtb2RlOiBPUEFRVUUsIEJJVE1BU0ssIG9yIFRSQU5TTFVDRU5ULgotICAgICAqIEByZXR1cm4gdGhlIFZvbGF0aWxlSW1hZ2Ugd2hpY2ggZGF0YSBsYXlvdXQgYW5kIGNvbG9yIG1vZGVsIGNvbXBhdGlibGUKLSAgICAgKiAgICAgICAgIHdpdGggdGhpcyBHcmFwaGljc0NvbmZpZ3VyYXRpb24uCi0gICAgICogQHRocm93cyBBV1RFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBJbWFnZUNhcGFiaWxpdGllcyBpcyBub3Qgc3VwcG9ydGVkIGJ5IHRoZQotICAgICAqICAgICAgICAgICAgIEdyYXBoaWNzQ29uZmlndXJhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgVm9sYXRpbGVJbWFnZSBjcmVhdGVDb21wYXRpYmxlVm9sYXRpbGVJbWFnZShpbnQgd2lkdGgsIGludCBoZWlnaHQsCi0gICAgICAgICAgICBJbWFnZUNhcGFiaWxpdGllcyBjYXBzLCBpbnQgdHJhbnNwYXJlbmN5KSB0aHJvd3MgQVdURXhjZXB0aW9uIHsKLSAgICAgICAgVm9sYXRpbGVJbWFnZSByZXMgPSBjcmVhdGVDb21wYXRpYmxlVm9sYXRpbGVJbWFnZSh3aWR0aCwgaGVpZ2h0LCB0cmFuc3BhcmVuY3kpOwotICAgICAgICBpZiAoIXJlcy5nZXRDYXBhYmlsaXRpZXMoKS5lcXVhbHMoY2FwcykpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xNEE9Q2FuIG5vdCBjcmVhdGUgVm9sYXRpbGVJbWFnZSB3aXRoIHNwZWNpZmllZCBjYXBhYmlsaXRpZXMKLSAgICAgICAgICAgIHRocm93IG5ldyBBV1RFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTRBIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBidWZmZXJpbmcgY2FwYWJpbGl0aWVzIG9mIHRoZSBHcmFwaGljc0NvbmZpZ3VyYXRpb24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgQnVmZmVyQ2FwYWJpbGl0aWVzIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQnVmZmVyQ2FwYWJpbGl0aWVzIGdldEJ1ZmZlckNhcGFiaWxpdGllcygpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBCdWZmZXJDYXBhYmlsaXRpZXMobmV3IEltYWdlQ2FwYWJpbGl0aWVzKGZhbHNlKSwgbmV3IEltYWdlQ2FwYWJpbGl0aWVzKGZhbHNlKSwKLSAgICAgICAgICAgICAgICBCdWZmZXJDYXBhYmlsaXRpZXMuRmxpcENvbnRlbnRzLlVOREVGSU5FRCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW1hZ2UgY2FwYWJpbGl0aWVzIG9mIHRoZSBHcmFwaGljc0NvbmZpZ3VyYXRpb24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VDYXBhYmlsaXRpZXMgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBJbWFnZUNhcGFiaWxpdGllcyBnZXRJbWFnZUNhcGFiaWxpdGllcygpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJbWFnZUNhcGFiaWxpdGllcyhmYWxzZSk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0dyYXBoaWNzRGV2aWNlLmphdmEgYi9hd3QvamF2YS9hd3QvR3JhcGhpY3NEZXZpY2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOWVkYTRlMC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvR3JhcGhpY3NEZXZpY2UuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE5NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgR3JhcGhpY3NEZXZpY2UgY2xhc3MgZGVzY3JpYmVzIHRoZSBncmFwaGljcyBkZXZpY2VzIChzdWNoIGFzIHNjcmVlbnMgb3IKLSAqIHByaW50ZXJzKSB3aGljaCBhcmUgYXZhaWxhYmxlIGluIGEgcGFydGljdWxhciBncmFwaGljcyBlbnZpcm9ubWVudC4gTWFueQotICogR3JhcGhpY3NEZXZpY2UgaW5zdGFuY2VzIGNhbiBiZSBhc3NvY2lhdGVkIHdpdGggYSBzaW5nbGUgR3JhcGhpY3NFbnZpcm9ubWVudC4KLSAqIEVhY2ggR3JhcGhpY3NEZXZpY2UgaGFzIG9uZSBvciBtb3JlIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBvYmplY3RzIHdoaWNoCi0gKiBzcGVjaWZ5IHRoZSBkaWZmZXJlbnQgY29uZmlndXJhdGlvbnMgYW5kIG1vZGVzIG9mIEdyYXBoaWNzRGV2aWNlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEdyYXBoaWNzRGV2aWNlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBkaXNwbGF5IG1vZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBEaXNwbGF5TW9kZSBkaXNwbGF5TW9kZTsKLQotICAgIC8vID8/P0FXVAotICAgIC8vIHByaXZhdGUgV2luZG93IGZ1bGxTY3JlZW5XaW5kb3cgPSBudWxsOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfSU1BR0VfQlVGRkVSIGluZGljYXRlcyBhIGltYWdlIGJ1ZmZlciBkZXZpY2UuCi0gICAgICovCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0lNQUdFX0JVRkZFUiA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9QUklOVEVSIGluZGljYXRlcyBhIHByaW50ZXIgZGV2aWNlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfUFJJTlRFUiA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9SQVNURVJfU0NSRUVOIGluZGljYXRlcyBhIHJhc3RlciBzY3JlZW4gZGV2aWNlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfUkFTVEVSX1NDUkVFTiA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBDb25zdHJ1Y3RvciBpcyBub3QgdG8gYmUgdXNlZCBkaXJlY3RseSBhcyB0aGlzIGNsYXNzIGlzIGFic3RyYWN0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBHcmFwaGljc0RldmljZSgpIHsKLSAgICAgICAgZGlzcGxheU1vZGUgPSBuZXcgRGlzcGxheU1vZGUoMCwgMCwgRGlzcGxheU1vZGUuQklUX0RFUFRIX01VTFRJLAotICAgICAgICAgICAgICAgIERpc3BsYXlNb2RlLlJFRlJFU0hfUkFURV9VTktOT1dOKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGFuIGFycmF5IG9mIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBvYmplY3RzIGFzc29jaWF0ZWQgd2l0aCB0aGUKLSAgICAgKiBHcmFwaGljc0RldmljZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBvYmplY3RzIGFzc29jaWF0ZWQgd2l0aCB0aGUKLSAgICAgKiAgICAgICAgIEdyYXBoaWNzRGV2aWNlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBHcmFwaGljc0NvbmZpZ3VyYXRpb25bXSBnZXRDb25maWd1cmF0aW9ucygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGVmYXVsdCBjb25maWd1cmF0aW9uIGZvciB0aGUgR3JhcGhpY3NEZXZpY2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGVmYXVsdCBHcmFwaGljc0NvbmZpZ3VyYXRpb24gb2JqZWN0IGZvciB0aGUgR3JhcGhpY3NEZXZpY2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEdyYXBoaWNzQ29uZmlndXJhdGlvbiBnZXREZWZhdWx0Q29uZmlndXJhdGlvbigpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgU3RyaW5nIGlkZW50aWZpZXIgd2hpY2ggYXNzb2NpYXRlZCB3aXRoIHRoZSBHcmFwaGljc0RldmljZSBpbgotICAgICAqIHRoZSBHcmFwaGljc0Vudmlyb25tZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFN0cmluZyBpZGVudGlmaWVyIG9mIHRoZSBHcmFwaGljc0RldmljZSBpbiB0aGUKLSAgICAgKiAgICAgICAgIEdyYXBoaWNzRW52aXJvbm1lbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFN0cmluZyBnZXRJRHN0cmluZygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdHlwZSBvZiB0aGlzIEdyYXBoaWNzRGV2aWNlOiBUWVBFX0lNQUdFX0JVRkZFUiwgVFlQRV9QUklOVEVSIG9yCi0gICAgICogVFlQRV9SQVNURVJfU0NSRUVOLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHR5cGUgb2YgdGhpcyBHcmFwaGljc0RldmljZTogVFlQRV9JTUFHRV9CVUZGRVIsIFRZUEVfUFJJTlRFUgotICAgICAqICAgICAgICAgb3IgVFlQRV9SQVNURVJfU0NSRUVOLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0VHlwZSgpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbnVtYmVyIG9mIGJ5dGVzIGF2YWlsYWJsZSBpbiBhY2NlbGVyYXRlZCBtZW1vcnkgb24gdGhpcwotICAgICAqIGRldmljZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgYnl0ZXMgYXZhaWxhYmxlIGFjY2VsZXJhdGVkIG1lbW9yeS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEF2YWlsYWJsZUFjY2VsZXJhdGVkTWVtb3J5KCkgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqID8/P0FXVCBwdWJsaWMgR3JhcGhpY3NDb25maWd1cmF0aW9uCi0gICAgICogZ2V0QmVzdENvbmZpZ3VyYXRpb24oR3JhcGhpY3NDb25maWdUZW1wbGF0ZSBnY3QpIHsgcmV0dXJuCi0gICAgICogZ2N0LmdldEJlc3RDb25maWd1cmF0aW9uKGdldENvbmZpZ3VyYXRpb25zKCkpOyB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBjdXJyZW50IGRpc3BsYXkgbW9kZSBvZiB0aGUgR3JhcGhpY3NEZXZpY2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY3VycmVudCBkaXNwbGF5IG1vZGUgb2YgdGhlIEdyYXBoaWNzRGV2aWNlLgotICAgICAqLwotICAgIHB1YmxpYyBEaXNwbGF5TW9kZSBnZXREaXNwbGF5TW9kZSgpIHsKLSAgICAgICAgcmV0dXJuIGRpc3BsYXlNb2RlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgZGlzcGxheSBtb2RlcyBhdmFpbGFibGUgaW4gdGhpcyBHcmFwaGljc0RldmljZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIGRpc3BsYXkgbW9kZXMgYXZhaWxhYmxlIGluIHRoaXMgR3JhcGhpY3NEZXZpY2UuCi0gICAgICovCi0gICAgcHVibGljIERpc3BsYXlNb2RlW10gZ2V0RGlzcGxheU1vZGVzKCkgewotICAgICAgICBEaXNwbGF5TW9kZVtdIGRtcyA9IHsKLSAgICAgICAgICAgIGRpc3BsYXlNb2RlCi0gICAgICAgIH07Ci0gICAgICAgIHJldHVybiBkbXM7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiA/Pz9BV1QgcHVibGljIFdpbmRvdyBnZXRGdWxsU2NyZWVuV2luZG93KCkgeyByZXR1cm4gZnVsbFNjcmVlbldpbmRvdzsgfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoaXMgR3JhcGhpY3NEZXZpY2Ugc3VwcG9ydHMgbG93LWxldmVsIGRpc3BsYXkgY2hhbmdlcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgR3JhcGhpY3NEZXZpY2Ugc3VwcG9ydHMgbG93LWxldmVsIGRpc3BsYXkgY2hhbmdlczsKLSAgICAgKiAgICAgICAgIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0Rpc3BsYXlDaGFuZ2VTdXBwb3J0ZWQoKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhpcyBHcmFwaGljc0RldmljZSBzdXBwb3J0cyBmdWxsIHNjcmVlbiBtb2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBHcmFwaGljc0RldmljZSBzdXBwb3J0cyBmdWxsIHNjcmVlbiBtb2RlLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzRnVsbFNjcmVlblN1cHBvcnRlZCgpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8vIGFuIGFycmF5IG9mIGRpc3BsYXkgbW9kZXMgYXZhaWxhYmxlIGluIHRoaXMgR3JhcGhpY3NEZXZpY2UuCi0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBkaXNwbGF5IG1vZGUgb2YgdGhpcyBHcmFwaGljc0RldmljZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZG0KLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgZGlzcGxheSBtb2RlIG9mIHRoaXMgR3JhcGhpY3NEZXZpY2UuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RGlzcGxheU1vZGUoRGlzcGxheU1vZGUgZG0pIHsKLSAgICAgICAgaWYgKCFpc0Rpc3BsYXlDaGFuZ2VTdXBwb3J0ZWQoKSkgewotICAgICAgICAgICAgLy8gYXd0LjEyMj1Eb2VzIG5vdCBzdXBwb3J0IGRpc3BsYXkgbW9kZSBjaGFuZ2VzCi0gICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTIyIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBEaXNwbGF5TW9kZVtdIGRtcyA9IGdldERpc3BsYXlNb2RlcygpOwotICAgICAgICBmb3IgKERpc3BsYXlNb2RlIGVsZW1lbnQgOiBkbXMpIHsKLSAgICAgICAgICAgIGlmIChlbGVtZW50LmVxdWFscyhkbSkpIHsKLSAgICAgICAgICAgICAgICBkaXNwbGF5TW9kZSA9IGRtOwotICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICAvLyBhd3QuMTIzPVVuc3VwcG9ydGVkIGRpc3BsYXkgbW9kZTogezB9Ci0gICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTIzIiwgZG0pKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qCi0gICAgICogPz8/QVdUIHB1YmxpYyB2b2lkIHNldEZ1bGxTY3JlZW5XaW5kb3coV2luZG93IHcpIHsgaWYgKHcgPT0gbnVsbCkgewotICAgICAqIGZ1bGxTY3JlZW5XaW5kb3cgPSBudWxsOyByZXR1cm47IH0gZnVsbFNjcmVlbldpbmRvdyA9IHc7IGlmCi0gICAgICogKGlzRnVsbFNjcmVlblN1cHBvcnRlZCgpKSB7IHcuZW5hYmxlSW5wdXRNZXRob2RzKGZhbHNlKTsgfSBlbHNlIHsKLSAgICAgKiB3LnNldFNpemUoZGlzcGxheU1vZGUuZ2V0V2lkdGgoKSwgZGlzcGxheU1vZGUuZ2V0SGVpZ2h0KCkpOwotICAgICAqIHcuc2V0TG9jYXRpb24oMCwgMCk7IH0gdy5zZXRWaXNpYmxlKHRydWUpOyB3LnNldEFsd2F5c09uVG9wKHRydWUpOyB9Ci0gICAgICovCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvR3JhcGhpY3NFbnZpcm9ubWVudC5qYXZhIGIvYXd0L2phdmEvYXd0L0dyYXBoaWNzRW52aXJvbm1lbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDUyNzQxNy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvR3JhcGhpY3NFbnZpcm9ubWVudC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjEyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LkNvbnRleHRTdG9yYWdlOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuQ29tbW9uR3JhcGhpY3MyREZhY3Rvcnk7Ci0KLS8qKgotICogVGhlIEdyYXBoaWNzRW52aXJvbm1lbnQgY2xhc3MgZGVmaW5lcyBhIGNvbGxlY3Rpb24gb2YgR3JhcGhpY3NEZXZpY2Ugb2JqZWN0cwotICogYW5kIEZvbnQgb2JqZWN0cyB3aGljaCBhcmUgYXZhaWxhYmxlIGZvciBKYXZhIGFwcGxpY2F0aW9uIG9uIGN1cnJlbnQKLSAqIHBsYXRmb3JtLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEdyYXBoaWNzRW52aXJvbm1lbnQgewotCi0gICAgLyoqCi0gICAgICogQ29uc3RydWN0b3IgY291bGQgbm90IGJlIHVzZWQgZGlyZWN0bHkgYW5kIHNob3VsZCBiZSBvYnRhaW5lZCBpbiBleHRlbmRlZAotICAgICAqIGNsYXNzZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEdyYXBoaWNzRW52aXJvbm1lbnQoKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9jYWwgR3JhcGhpY3NFbnZpcm9ubWVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBsb2NhbCBHcmFwaGljc0Vudmlyb25tZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgR3JhcGhpY3NFbnZpcm9ubWVudCBnZXRMb2NhbEdyYXBoaWNzRW52aXJvbm1lbnQoKSB7Ci0gICAgICAgIHN5bmNocm9uaXplZCAoQ29udGV4dFN0b3JhZ2UuZ2V0Q29udGV4dExvY2soKSkgewotICAgICAgICAgICAgaWYgKENvbnRleHRTdG9yYWdlLmdldEdyYXBoaWNzRW52aXJvbm1lbnQoKSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGlzSGVhZGxlc3MoKSkgewotICAgICAgICAgICAgICAgICAgICBDb250ZXh0U3RvcmFnZS5zZXRHcmFwaGljc0Vudmlyb25tZW50KG5ldyBIZWFkbGVzc0dyYXBoaWNzRW52aXJvbm1lbnQoKSk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgQ29tbW9uR3JhcGhpY3MyREZhY3RvcnkgZzJkZiA9IChDb21tb25HcmFwaGljczJERmFjdG9yeSlUb29sa2l0Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLmdldERlZmF1bHRUb29sa2l0KCkuZ2V0R3JhcGhpY3NGYWN0b3J5KCk7Ci0KLSAgICAgICAgICAgICAgICAgICAgQ29udGV4dFN0b3JhZ2Uuc2V0R3JhcGhpY3NFbnZpcm9ubWVudChnMmRmCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLmNyZWF0ZUdyYXBoaWNzRW52aXJvbm1lbnQoQ29udGV4dFN0b3JhZ2UuZ2V0V2luZG93RmFjdG9yeSgpKSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZXR1cm4gQ29udGV4dFN0b3JhZ2UuZ2V0R3JhcGhpY3NFbnZpcm9ubWVudCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB3aGV0aGVyIG9yIG5vdCBhIGRpc3BsYXksIGtleWJvYXJkLCBhbmQgbW91c2UgYXJlIHN1cHBvcnRlZCBpbgotICAgICAqIHRoaXMgZ3JhcGhpY3MgZW52aXJvbm1lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBIZWFkbGVzc0V4Y2VwdGlvbiB3aWxsIGJlIHRocm93biBmcm9tIGFyZWFzIG9mIHRoZQotICAgICAqICAgICAgICAgZ3JhcGhpY3MgZW52aXJvbm1lbnQgdGhhdCBhcmUgZGVwZW5kZW50IG9uIGEgZGlzcGxheSwga2V5Ym9hcmQsCi0gICAgICogICAgICAgICBvciBtb3VzZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzSGVhZGxlc3NJbnN0YW5jZSgpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCBhIGRpc3BsYXksIGtleWJvYXJkLCBhbmQgbW91c2UgYXJlIHN1cHBvcnRlZCBpbgotICAgICAqIHRoaXMgZW52aXJvbm1lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBhIEhlYWRsZXNzRXhjZXB0aW9uIGlzIHRocm93biBmcm9tIGFyZWFzIG9mIHRoZSBUb29sa2l0Ci0gICAgICogICAgICAgICBhbmQgR3JhcGhpY3NFbnZpcm9ubWVudCB0aGF0IGFyZSBkZXBlbmRlbnQgb24gYSBkaXNwbGF5LAotICAgICAqICAgICAgICAga2V5Ym9hcmQsIG9yIG1vdXNlLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBib29sZWFuIGlzSGVhZGxlc3MoKSB7Ci0gICAgICAgIHJldHVybiAidHJ1ZSIuZXF1YWxzKFN5c3RlbS5nZXRQcm9wZXJ0eSgiamF2YS5hd3QuaGVhZGxlc3MiKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWF4aW11bSBib3VuZHMgb2Ygc3lzdGVtIGNlbnRlcmVkIHdpbmRvd3MuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWF4aW11bSBib3VuZHMgb2Ygc3lzdGVtIGNlbnRlcmVkIHdpbmRvd3MuCi0gICAgICogQHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGlzSGVhZGxlc3MoKSBtZXRob2QgcmV0dXJucyB0cnVlLgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0TWF4aW11bVdpbmRvd0JvdW5kcygpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBnZXREZWZhdWx0U2NyZWVuRGV2aWNlKCkuZ2V0RGVmYXVsdENvbmZpZ3VyYXRpb24oKS5nZXRCb3VuZHMoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBQb2ludCB3aGljaCBzaG91bGQgZGVmaW5lcyB0aGUgY2VudGVyIG9mIHN5c3RlbSB3aW5kb3cuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgUG9pbnQgd2hlcmUgdGhlIHN5c3RlbSB3aW5kb3cgc2hvdWxkIGJlIGNlbnRlcmVkLgotICAgICAqIEB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBpc0hlYWRsZXNzKCkgbWV0aG9kIHJldHVybnMgdHJ1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUG9pbnQgZ2V0Q2VudGVyUG9pbnQoKSB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24gewotICAgICAgICBSZWN0YW5nbGUgbXdiID0gZ2V0TWF4aW11bVdpbmRvd0JvdW5kcygpOwotICAgICAgICByZXR1cm4gbmV3IFBvaW50KG13Yi53aWR0aCA+PiAxLCBtd2IuaGVpZ2h0ID4+IDEpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluZGljYXRlcyB0aGF0IHRoZSBwcmltYXJ5IGZvbnQgc2hvdWxkIGJlIHVzZWQuIFByaW1hcnkgZm9udCBpcyBzcGVjaWZpZWQKLSAgICAgKiBieSBpbml0aWFsIHN5c3RlbSBsb2NhbGUgb3IgZGVmYXVsdCBlbmNvZGluZykuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcHJlZmVyTG9jYWxlRm9udHMoKSB7Ci0gICAgICAgIC8vIE5vdGU6IEFQSSBzcGVjaWZpY2F0aW9uIHNheXMgZm9sbG93aW5nOgotICAgICAgICAvLyAiVGhlIGFjdHVhbCBjaGFuZ2UgaW4gZm9udCByZW5kZXJpbmcgYmVoYXZpb3IgcmVzdWx0aW5nCi0gICAgICAgIC8vIGZyb20gYSBjYWxsIHRvIHRoaXMgbWV0aG9kIGlzIGltcGxlbWVudGF0aW9uIGRlcGVuZGVudDsKLSAgICAgICAgLy8gaXQgbWF5IGhhdmUgbm8gZWZmZWN0IGF0IGFsbC4iIFNvLCBkb2luZyBub3RoaW5nIGlzIGFuCi0gICAgICAgIC8vIGFjY2VwdGFibGUgYmVoYXZpb3IgZm9yIHRoaXMgbWV0aG9kLgotCi0gICAgICAgIC8vIEZvciBub3cgRm9udE1hbmFnZXIgdXNlcyAxLjQgZm9udC5wcm9wZXJ0aWVzIHNjaGVtZSBmb3IgZm9udCBtYXBwaW5nLAotICAgICAgICAvLyBzbwotICAgICAgICAvLyB0aGlzIG1ldGhvZCBkb2Vzbid0IG1ha2UgYW55IHNlbnNlLiBUaGUgaW1wbGVtZW50YXRpb24gb2YgdGhpcyBtZXRob2QKLSAgICAgICAgLy8gd2hpY2ggd2lsbCBpbmZsdWVuY2UgZm9udCBtYXBwaW5nIGlzIHBvc3Rwb25lZCB1bnRpbAotICAgICAgICAvLyAxLjUgbWFwcGluZyBzY2hlbWUgbm90IGltcGxlbWVudGVkLgotCi0gICAgICAgIC8vIHRvZG8gLSBJbXBsZW1lbnQgbm9uLWRlZmF1bHQgYmVoYXZpb3Igd2l0aCAxLjUgZm9udCBtYXBwaW5nIHNjaGVtZQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluZGljYXRlcyB0aGF0IGEgcHJvcG9ydGlvbmFsIHByZWZlcmVuY2Ugb2YgdGhlIGZvbnQgc2hvdWxkIGJlIHVzZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcHJlZmVyUHJvcG9ydGlvbmFsRm9udHMoKSB7Ci0gICAgICAgIC8vIE5vdGU6IEFQSSBzcGVjaWZpY2F0aW9uIHNheXMgZm9sbG93aW5nOgotICAgICAgICAvLyAiVGhlIGFjdHVhbCBjaGFuZ2UgaW4gZm9udCByZW5kZXJpbmcgYmVoYXZpb3IgcmVzdWx0aW5nCi0gICAgICAgIC8vIGZyb20gYSBjYWxsIHRvIHRoaXMgbWV0aG9kIGlzIGltcGxlbWVudGF0aW9uIGRlcGVuZGVudDsKLSAgICAgICAgLy8gaXQgbWF5IGhhdmUgbm8gZWZmZWN0IGF0IGFsbC4iIFNvLCBkb2luZyBub3RoaW5nIGlzIGFuCi0gICAgICAgIC8vIGFjY2VwdGFibGUgYmVoYXZpb3IgZm9yIHRoaXMgbWV0aG9kLgotCi0gICAgICAgIC8vIEZvciBub3cgRm9udE1hbmFnZXIgdXNlcyAxLjQgZm9udC5wcm9wZXJ0aWVzIHNjaGVtZSBmb3IgZm9udCBtYXBwaW5nLAotICAgICAgICAvLyBzbwotICAgICAgICAvLyB0aGlzIG1ldGhvZCBkb2Vzbid0IG1ha2UgYW55IHNlbnNlLiBUaGUgaW1wbGVtZW50YXRpb24gb2YgdGhpcyBtZXRob2QKLSAgICAgICAgLy8gd2hpY2ggd2lsbCBpbmZsdWVuY2UgZm9udCBtYXBwaW5nIGlzIHBvc3Rwb25lZCB1bnRpbAotICAgICAgICAvLyAxLjUgbWFwcGluZyBzY2hlbWUgbm90IGltcGxlbWVudGVkLgotCi0gICAgICAgIC8vIHRvZG8gLSBJbXBsZW1lbnQgbm9uLWRlZmF1bHQgYmVoYXZpb3Igd2l0aCAxLjUgZm9udCBtYXBwaW5nIHNjaGVtZQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIEdyYXBoaWNzMkQgb2JqZWN0IGZvciByZW5kZXJpbmcgdG8gdGhlIHNwZWNpZmllZAotICAgICAqIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJ1ZmZlcmVkSW1hZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBCdWZmZXJlZEltYWdlIG9iamVjdC4KLSAgICAgKiBAcmV0dXJuIHRoZSBHcmFwaGljczJEIG9iamVjdCB3aGljaCBhbGxvd3MgdG8gcmVuZGVyIHRvIHRoZSBzcGVjaWZpZWQKLSAgICAgKiAgICAgICAgIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEdyYXBoaWNzMkQgY3JlYXRlR3JhcGhpY3MoQnVmZmVyZWRJbWFnZSBidWZmZXJlZEltYWdlKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFycmF5IG9mIGFsbCBhdmFpbGFibGUgZm9udHMgaW5zdGFuY2VzIGluIHRoaXMKLSAgICAgKiBHcmFwaGljc0Vudmlyb21lbnRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGFsbCBhdmFpbGFibGUgZm9udHMgaW5zdGFuY2VzIGluIHRoaXMKLSAgICAgKiAgICAgICAgIEdyYXBoaWNzRW52aXJvbWVudHMuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEZvbnRbXSBnZXRBbGxGb250cygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYXJyYXkgb2YgYWxsIGF2YWlsYWJsZSBmb250IGZhbWlseSBuYW1lcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBhbGwgYXZhaWxhYmxlIGZvbnQgZmFtaWx5IG5hbWVzLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBTdHJpbmdbXSBnZXRBdmFpbGFibGVGb250RmFtaWx5TmFtZXMoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFycmF5IG9mIGFsbCBhdmFpbGFibGUgZm9udCBmYW1pbHkgbmFtZXMgZm9yIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBsb2NhbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxvY2FsZQotICAgICAqICAgICAgICAgICAgdGhlIExvY2FsZSBvYmplY3Qgd2hpY2ggcmVwcmVzZW50cyBnZW9ncmFwaGljYWwgcmVnaW9uLiBUaGUKLSAgICAgKiAgICAgICAgICAgIGRlZmF1bHQgbG9jYWxlIGlzIHVzZWQgaWYgbG9jYWxlIGlzIG51bGwuCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgYXZhaWxhYmxlIGZvbnQgZmFtaWx5IG5hbWVzIGZvciB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBsb2NhbGUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFN0cmluZ1tdIGdldEF2YWlsYWJsZUZvbnRGYW1pbHlOYW1lcyhMb2NhbGUgbG9jYWxlKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRlZmF1bHQgc2NyZWVuIGRldmljZSBhcyBHcmFwaGljRGV2aWNlIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBHcmFwaGljRGV2aWNlIG9iamVjdCB3aGljaCByZXByZXNlbnRzIGRlZmF1bHQgc2NyZWVuIGRldmljZS4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgaXNIZWFkbGVzcygpIHJldHVybnMgdHJ1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgR3JhcGhpY3NEZXZpY2UgZ2V0RGVmYXVsdFNjcmVlbkRldmljZSgpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgYWxsIGF2YWlsYWJsZSBzY3JlZW4gZGV2aWNlcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBHcmFwaGljc0RldmljZSBvYmplY3RzIHdoaWNoIHJlcHJlc2VudHMgYWxsCi0gICAgICogICAgICAgICBhdmFpbGFibGUgc2NyZWVuIGRldmljZXMuCi0gICAgICogQHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGlzSGVhZGxlc3MoKSByZXR1cm5zIHRydWUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEdyYXBoaWNzRGV2aWNlW10gZ2V0U2NyZWVuRGV2aWNlcygpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbjsKLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9IZWFkbGVzc0V4Y2VwdGlvbi5qYXZhIGIvYXd0L2phdmEvYXd0L0hlYWRsZXNzRXhjZXB0aW9uLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGVjMTExZjEuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0hlYWRsZXNzRXhjZXB0aW9uLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLS8qKgotICogVGhlIEhlYWRsZXNzRXhjZXB0aW9uIGNsYXNzIHByb3ZpZGVzIG5vdGlmaWNhdGlvbnMgYW5kIGVycm9yIG1lc3NhZ2VzIHdoZW4KLSAqIGNvZGUgdGhhdCBpcyBkZXBlbmRlbnQgb24gYSBrZXlib2FyZCwgZGlzcGxheSwgb3IgbW91c2UgaXMgY2FsbGVkIGluIGFuCi0gKiBlbnZpcm9ubWVudCB0aGF0IGRvZXMgbm90IHN1cHBvcnQgYSBrZXlib2FyZCwgZGlzcGxheSwgb3IgbW91c2UuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgSGVhZGxlc3NFeGNlcHRpb24gZXh0ZW5kcyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbiB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAxNjcxODM2NDQ5NDQzNTg1NjNMOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGhlYWRsZXNzIGV4Y2VwdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgSGVhZGxlc3NFeGNlcHRpb24oKSB7Ci0gICAgICAgIHN1cGVyKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGhlYWRsZXNzIGV4Y2VwdGlvbiB3aXRoIHRoZSBzcGVjaWZpZWQgbWVzc2FnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXNnCi0gICAgICogICAgICAgICAgICB0aGUgU3RyaW5nIHdoaWNoIHJlcHJlc2VudHMgZXJyb3IgbWVzc2FnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgSGVhZGxlc3NFeGNlcHRpb24oU3RyaW5nIG1zZykgewotICAgICAgICBzdXBlcihtc2cpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9IZWFkbGVzc0dyYXBoaWNzRW52aXJvbm1lbnQuamF2YSBiL2F3dC9qYXZhL2F3dC9IZWFkbGVzc0dyYXBoaWNzRW52aXJvbm1lbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzA2MzkzZi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvSGVhZGxlc3NHcmFwaGljc0Vudmlyb25tZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw3MiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3NEZXZpY2U7Ci1pbXBvcnQgamF2YS5hd3QuSGVhZGxlc3NFeGNlcHRpb247Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLkNvbW1vbkdyYXBoaWNzRW52aXJvbm1lbnQ7Ci0KLS8qKgotICogVGhlIEhlYWRsZXNzR3JhcGhpY3NFbnZpcm9ubWVudCBjbGFzcyBpcyB0aGUgQ29tbW9uR3JhcGhpY3NFbnZpcm9ubWVudAotICogaW1wbGVtZW50YXRpb24gdG8gdXNlIGluIHRoZSBjYXNlIHdoZXJlIHRoZSBlbnZpcm9ubWVudCBsYWNrcyBkaXNwbGF5LAotICoga2V5Ym9hcmQsIGFuZCBtb3VzZSBzdXBwb3J0LgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEhlYWRsZXNzR3JhcGhpY3NFbnZpcm9ubWVudCBleHRlbmRzIENvbW1vbkdyYXBoaWNzRW52aXJvbm1lbnQgewotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB3aGV0aGVyIG9yIG5vdCBhIGRpc3BsYXksIGtleWJvYXJkLCBhbmQgbW91c2UgYXJlIHN1cHBvcnRlZCBpbgotICAgICAqIHRoaXMgZ3JhcGhpY3MgZW52aXJvbm1lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBIZWFkbGVzc0V4Y2VwdGlvbiB3aWxsIGJlIHRocm93biBmcm9tIGFyZWFzIG9mIHRoZQotICAgICAqICAgICAgICAgZ3JhcGhpY3MgZW52aXJvbm1lbnQgdGhhdCBhcmUgZGVwZW5kZW50IG9uIGEgZGlzcGxheSwga2V5Ym9hcmQsCi0gICAgICogICAgICAgICBvciBtb3VzZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGlzSGVhZGxlc3NJbnN0YW5jZSgpIHsKLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGVmYXVsdCBzY3JlZW4gZGV2aWNlIGFzIEdyYXBoaWNEZXZpY2Ugb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEdyYXBoaWNEZXZpY2Ugb2JqZWN0IHdoaWNoIHJlcHJlc2VudHMgZGVmYXVsdCBzY3JlZW4gZGV2aWNlLgotICAgICAqIEB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBpc0hlYWRsZXNzKCkgcmV0dXJucyB0cnVlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBHcmFwaGljc0RldmljZSBnZXREZWZhdWx0U2NyZWVuRGV2aWNlKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiBhbGwgYXZhaWxhYmxlIHNjcmVlbiBkZXZpY2VzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIEdyYXBoaWNzRGV2aWNlIG9iamVjdHMgd2hpY2ggcmVwcmVzZW50cyBhbGwKLSAgICAgKiAgICAgICAgIGF2YWlsYWJsZSBzY3JlZW4gZGV2aWNlcy4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgaXNIZWFkbGVzcygpIHJldHVybnMgdHJ1ZS4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR3JhcGhpY3NEZXZpY2VbXSBnZXRTY3JlZW5EZXZpY2VzKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvSGVhZGxlc3NUb29sa2l0LmphdmEgYi9hd3QvamF2YS9hd3QvSGVhZGxlc3NUb29sa2l0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM2NGE4NWEuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L0hlYWRsZXNzVG9vbGtpdC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjI2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLS8vPz8/QVdUCi0vL2ltcG9ydCBqYXZhLmF3dC5kYXRhdHJhbnNmZXIuQ2xpcGJvYXJkOwotLy9pbXBvcnQgamF2YS5hd3QuZG5kLkRyYWdHZXN0dXJlRXZlbnQ7Ci0vL2ltcG9ydCBqYXZhLmF3dC5kbmQuRHJhZ0dlc3R1cmVMaXN0ZW5lcjsKLS8vaW1wb3J0IGphdmEuYXd0LmRuZC5EcmFnR2VzdHVyZVJlY29nbml6ZXI7Ci0vL2ltcG9ydCBqYXZhLmF3dC5kbmQuRHJhZ1NvdXJjZTsKLS8vaW1wb3J0IGphdmEuYXd0LmRuZC5JbnZhbGlkRG5ET3BlcmF0aW9uRXhjZXB0aW9uOwotLy9pbXBvcnQgamF2YS5hd3QuZG5kLnBlZXIuRHJhZ1NvdXJjZUNvbnRleHRQZWVyOwotaW1wb3J0IGphdmEuYXd0LmltLklucHV0TWV0aG9kSGlnaGxpZ2h0OwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7IC8vaW1wb3J0IGphdmEuYXd0LnBlZXIuKjsKLS8vaW1wb3J0IGphdmEuYmVhbnMuUHJvcGVydHlDaGFuZ2VTdXBwb3J0OwotaW1wb3J0IGphdmEudXRpbC5IYXNoTWFwOwotaW1wb3J0IGphdmEudXRpbC5NYXA7Ci1pbXBvcnQgamF2YS51dGlsLlByb3BlcnRpZXM7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LkNvbXBvbmVudEludGVybmFsczsgLy9pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5kYXRhdHJhbnNmZXIuRFRLOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLkdyYXBoaWNzRmFjdG9yeTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5OYXRpdmVFdmVudFF1ZXVlOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLldpbmRvd0ZhY3Rvcnk7Ci0KLS8qKgotICogVGhlIEhlYWRsZXNzVG9vbGtpdCBjbGFzcyBpcyBhIHN1YmNsYXNzIG9mIFRvb2xraXRJbXBsIHRvIGJlIHVzZWQgZm9yCi0gKiBncmFwaGljYWwgZW52aXJvbm1lbnRzIHRoYXQgbGFjayBrZXlib2FyZCBhbmQgbW91c2UgY2FwYWJpbGl0aWVzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIEhlYWRsZXNzVG9vbGtpdCBleHRlbmRzIFRvb2xraXRJbXBsIHsKLQotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogQE92ZXJyaWRlIHByb3RlY3RlZCBCdXR0b25QZWVyIGNyZWF0ZUJ1dHRvbihCdXR0b24gYTApIHRocm93cwotICAgICAqIEhlYWRsZXNzRXhjZXB0aW9uIHsgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHJvdGVjdGVkIENoZWNrYm94UGVlciBjcmVhdGVDaGVja2JveChDaGVja2JveCBhMCkgdGhyb3dzCi0gICAgICogSGVhZGxlc3NFeGNlcHRpb24geyB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsgfQotICAgICAqIEBPdmVycmlkZSBwcm90ZWN0ZWQgQ2hlY2tib3hNZW51SXRlbVBlZXIKLSAgICAgKiBjcmVhdGVDaGVja2JveE1lbnVJdGVtKENoZWNrYm94TWVudUl0ZW0gYTApIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICogdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHJvdGVjdGVkIENob2ljZVBlZXIgY3JlYXRlQ2hvaWNlKENob2ljZSBhMCkgdGhyb3dzCi0gICAgICogSGVhZGxlc3NFeGNlcHRpb24geyB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsgfSBwdWJsaWMgQ3Vyc29yCi0gICAgICogY3JlYXRlQ3VzdG9tQ3Vyc29yKEltYWdlIGltZywgUG9pbnQgaG90U3BvdCwgU3RyaW5nIG5hbWUpIHRocm93cwotICAgICAqIEhlYWRsZXNzRXhjZXB0aW9uIHsgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHJvdGVjdGVkIERpYWxvZ1BlZXIgY3JlYXRlRGlhbG9nKERpYWxvZyBhMCkgdGhyb3dzCi0gICAgICogSGVhZGxlc3NFeGNlcHRpb24geyB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsgfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgPFQgZXh0ZW5kcyBEcmFnR2VzdHVyZVJlY29nbml6ZXI+IFQKLSAgICAgKiBjcmVhdGVEcmFnR2VzdHVyZVJlY29nbml6ZXIoIENsYXNzPFQ+IHJlY29nbml6ZXJBYnN0cmFjdENsYXNzLCBEcmFnU291cmNlCi0gICAgICogZHMsIENvbXBvbmVudCBjLCBpbnQgc3JjQWN0aW9ucywgRHJhZ0dlc3R1cmVMaXN0ZW5lciBkZ2wpIHsgcmV0dXJuIG51bGw7Ci0gICAgICogfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgRHJhZ1NvdXJjZUNvbnRleHRQZWVyCi0gICAgICogY3JlYXRlRHJhZ1NvdXJjZUNvbnRleHRQZWVyKERyYWdHZXN0dXJlRXZlbnQgZGdlKSB0aHJvd3MKLSAgICAgKiBJbnZhbGlkRG5ET3BlcmF0aW9uRXhjZXB0aW9uIHsgdGhyb3cgbmV3IEludmFsaWREbkRPcGVyYXRpb25FeGNlcHRpb24oKTsKLSAgICAgKiB9Ci0gICAgICogQE92ZXJyaWRlIHByb3RlY3RlZCBGaWxlRGlhbG9nUGVlciBjcmVhdGVGaWxlRGlhbG9nKEZpbGVEaWFsb2cgYTApIHRocm93cwotICAgICAqIEhlYWRsZXNzRXhjZXB0aW9uIHsgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHJvdGVjdGVkIEZyYW1lUGVlciBjcmVhdGVGcmFtZShGcmFtZSBhMCkgdGhyb3dzCi0gICAgICogSGVhZGxlc3NFeGNlcHRpb24geyB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsgfQotICAgICAqIEBPdmVycmlkZSBwcm90ZWN0ZWQgTGFiZWxQZWVyIGNyZWF0ZUxhYmVsKExhYmVsIGEwKSB0aHJvd3MKLSAgICAgKiBIZWFkbGVzc0V4Y2VwdGlvbiB7IHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOyB9Ci0gICAgICogQE92ZXJyaWRlIHByb3RlY3RlZCBMaXN0UGVlciBjcmVhdGVMaXN0KExpc3QgYTApIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqIHsgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHJvdGVjdGVkIE1lbnVQZWVyIGNyZWF0ZU1lbnUoTWVudSBhMCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogeyB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsgfQotICAgICAqIEBPdmVycmlkZSBwcm90ZWN0ZWQgTWVudUJhclBlZXIgY3JlYXRlTWVudUJhcihNZW51QmFyIGEwKSB0aHJvd3MKLSAgICAgKiBIZWFkbGVzc0V4Y2VwdGlvbiB7IHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOyB9Ci0gICAgICogQE92ZXJyaWRlIHByb3RlY3RlZCBNZW51SXRlbVBlZXIgY3JlYXRlTWVudUl0ZW0oTWVudUl0ZW0gYTApIHRocm93cwotICAgICAqIEhlYWRsZXNzRXhjZXB0aW9uIHsgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHJvdGVjdGVkIFBvcHVwTWVudVBlZXIgY3JlYXRlUG9wdXBNZW51KFBvcHVwTWVudSBhMCkgdGhyb3dzCi0gICAgICogSGVhZGxlc3NFeGNlcHRpb24geyB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsgfQotICAgICAqIEBPdmVycmlkZSBwcm90ZWN0ZWQgU2Nyb2xsYmFyUGVlciBjcmVhdGVTY3JvbGxiYXIoU2Nyb2xsYmFyIGEwKSB0aHJvd3MKLSAgICAgKiBIZWFkbGVzc0V4Y2VwdGlvbiB7IHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOyB9Ci0gICAgICogQE92ZXJyaWRlIHByb3RlY3RlZCBTY3JvbGxQYW5lUGVlciBjcmVhdGVTY3JvbGxQYW5lKFNjcm9sbFBhbmUgYTApIHRocm93cwotICAgICAqIEhlYWRsZXNzRXhjZXB0aW9uIHsgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHJvdGVjdGVkIFRleHRBcmVhUGVlciBjcmVhdGVUZXh0QXJlYShUZXh0QXJlYSBhMCkgdGhyb3dzCi0gICAgICogSGVhZGxlc3NFeGNlcHRpb24geyB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsgfQotICAgICAqIEBPdmVycmlkZSBwcm90ZWN0ZWQgVGV4dEZpZWxkUGVlciBjcmVhdGVUZXh0RmllbGQoVGV4dEZpZWxkIGEwKSB0aHJvd3MKLSAgICAgKiBIZWFkbGVzc0V4Y2VwdGlvbiB7IHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOyB9Ci0gICAgICogQE92ZXJyaWRlIHByb3RlY3RlZCBXaW5kb3dQZWVyIGNyZWF0ZVdpbmRvdyhXaW5kb3cgYTApIHRocm93cwotICAgICAqIEhlYWRsZXNzRXhjZXB0aW9uIHsgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKi8KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBEaW1lbnNpb24gZ2V0QmVzdEN1cnNvclNpemUoaW50IHByZWZXaWR0aCwgaW50IHByZWZIZWlnaHQpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoKSB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR3JhcGhpY3NGYWN0b3J5IGdldEdyYXBoaWNzRmFjdG9yeSgpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGdldExvY2tpbmdLZXlTdGF0ZShpbnQga2V5Q29kZSkgdGhyb3dzIFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRNYXhpbXVtQ3Vyc29yQ29sb3JzKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRNZW51U2hvcnRjdXRLZXlNYXNrKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7Ci0gICAgfQotCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBAT3ZlcnJpZGUgTmF0aXZlRXZlbnRRdWV1ZSBnZXROYXRpdmVFdmVudFF1ZXVlKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogeyB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsgfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgUHJpbnRKb2IgZ2V0UHJpbnRKb2IoRnJhbWUgZnJhbWUsIFN0cmluZyBqb2J0aXRsZSwKLSAgICAgKiBKb2JBdHRyaWJ1dGVzIGpvYkF0dHJpYnV0ZXMsIFBhZ2VBdHRyaWJ1dGVzIHBhZ2VBdHRyaWJ1dGVzKSB0aHJvd3MKLSAgICAgKiBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24geyB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHVibGljIFByaW50Sm9iIGdldFByaW50Sm9iKEZyYW1lIGZyYW1lLCBTdHJpbmcgam9idGl0bGUsCi0gICAgICogUHJvcGVydGllcyBwcm9wcykgdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uIHsgdGhyb3cgbmV3Ci0gICAgICogTnVsbFBvaW50ZXJFeGNlcHRpb24oKTsgfQotICAgICAqLwotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEluc2V0cyBnZXRTY3JlZW5JbnNldHMoR3JhcGhpY3NDb25maWd1cmF0aW9uIGdjKSB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldFNjcmVlblJlc29sdXRpb24oKSB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgRGltZW5zaW9uIGdldFNjcmVlblNpemUoKSB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsKLSAgICB9Ci0KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgQ2xpcGJvYXJkIGdldFN5c3RlbUNsaXBib2FyZCgpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqIHsgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHVibGljIENsaXBib2FyZCBnZXRTeXN0ZW1TZWxlY3Rpb24oKSB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24KLSAgICAgKiB7IHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOyB9Ci0gICAgICogQE92ZXJyaWRlIFdpbmRvd0ZhY3RvcnkgZ2V0V2luZG93RmFjdG9yeSgpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICogdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7IH0KLSAgICAgKi8KLQotICAgIEBPdmVycmlkZQotICAgIHByb3RlY3RlZCB2b2lkIGluaXQoKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIENvbXBvbmVudEludGVybmFscy5zZXRDb21wb25lbnRJbnRlcm5hbHMobmV3IENvbXBvbmVudEludGVybmFsc0ltcGwoKSk7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IG5ldyBFdmVudFF1ZXVlKHRoaXMpOyAvLyBjcmVhdGUgdGhlIHN5c3RlbSBFdmVudFF1ZXVlCi0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGRpc3BhdGNoZXIgPSBuZXcgRGlzcGF0Y2hlcih0aGlzKTsKLSAgICAgICAgICAgIGRlc2t0b3BQcm9wZXJ0aWVzID0gbmV3IEhhc2hNYXA8U3RyaW5nLCBPYmplY3Q+KCk7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IGRlc2t0b3BQcm9wc1N1cHBvcnQgPSBuZXcgUHJvcGVydHlDaGFuZ2VTdXBwb3J0KHRoaXMpOwotICAgICAgICAgICAgLy8gPz8/QVdUOiBhd3RFdmVudHNNYW5hZ2VyID0gbmV3IEFXVEV2ZW50c01hbmFnZXIoKTsKLSAgICAgICAgICAgIC8vID8/P0FXVDogZGlzcGF0Y2hUaHJlYWQgPSBuZXcgSGVhZGxlc3NFdmVudERpc3BhdGNoVGhyZWFkKHRoaXMsCi0gICAgICAgICAgICAvLyBkaXNwYXRjaGVyKTsKLSAgICAgICAgICAgIC8vID8/P0FXVDogZHRrID0gRFRLLmdldERUSygpOwotICAgICAgICAgICAgZGlzcGF0Y2hUaHJlYWQuc3RhcnQoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNEeW5hbWljTGF5b3V0QWN0aXZlKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHJvdGVjdGVkIGJvb2xlYW4gaXNEeW5hbWljTGF5b3V0U2V0KCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNGcmFtZVN0YXRlU3VwcG9ydGVkKGludCBzdGF0ZSkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHJvdGVjdGVkIHZvaWQgbG9hZFN5c3RlbUNvbG9ycyhpbnRbXSBzeXN0ZW1Db2xvcnMpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBNYXA8amF2YS5hd3QuZm9udC5UZXh0QXR0cmlidXRlLCA/PiBtYXBJbnB1dE1ldGhvZEhpZ2hsaWdodCgKLSAgICAgICAgICAgIElucHV0TWV0aG9kSGlnaGxpZ2h0IGhpZ2hsaWdodCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IEhlYWRsZXNzRXhjZXB0aW9uKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgTWFwPGphdmEuYXd0LmZvbnQuVGV4dEF0dHJpYnV0ZSwgPz4gbWFwSW5wdXRNZXRob2RIaWdobGlnaHRJbXBsKElucHV0TWV0aG9kSGlnaGxpZ2h0IGhpZ2hsaWdodCkKLSAgICAgICAgICAgIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldER5bmFtaWNMYXlvdXQoYm9vbGVhbiBkeW5hbWljKSB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRMb2NraW5nS2V5U3RhdGUoaW50IGtleUNvZGUsIGJvb2xlYW4gb24pIHRocm93cyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBIZWFkbGVzc0V4Y2VwdGlvbigpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9JbGxlZ2FsQ29tcG9uZW50U3RhdGVFeGNlcHRpb24uamF2YSBiL2F3dC9qYXZhL2F3dC9JbGxlZ2FsQ29tcG9uZW50U3RhdGVFeGNlcHRpb24uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYmVkMTcyOS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvSWxsZWdhbENvbXBvbmVudFN0YXRlRXhjZXB0aW9uLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLS8qKgotICogVGhlIElsbGVnYWxDb21wb25lbnRTdGF0ZUV4Y2VwdGlvbiBjbGFzcyBpcyB1c2VkIHRvIHByb3ZpZGUgbm90aWZpY2F0aW9uIHRoYXQKLSAqIEFXVCBjb21wb25lbnQgaXMgbm90IGluIGFuIGFwcHJvcHJpYXRlIHN0YXRlIGZvciB0aGUgcmVxdWVzdGVkIG9wZXJhdGlvbi4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJbGxlZ2FsQ29tcG9uZW50U3RhdGVFeGNlcHRpb24gZXh0ZW5kcyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24gewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IHNlcmlhbFZlcnNpb25VSUQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gLTE4ODkzMzk1ODcyMDgxNDQyMzhMOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IElsbGVnYWxDb21wb25lbnRTdGF0ZUV4Y2VwdGlvbiB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBtZXNzYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgU3RyaW5nIG1lc3NhZ2Ugd2hpY2ggZGVzY3JpYmVzIHRoZSBleGNlcHRpb24uCi0gICAgICovCi0gICAgcHVibGljIElsbGVnYWxDb21wb25lbnRTdGF0ZUV4Y2VwdGlvbihTdHJpbmcgcykgewotICAgICAgICBzdXBlcihzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSWxsZWdhbENvbXBvbmVudFN0YXRlRXhjZXB0aW9uIHdpdGhvdXQgZGV0YWlsZWQKLSAgICAgKiBtZXNzYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBJbGxlZ2FsQ29tcG9uZW50U3RhdGVFeGNlcHRpb24oKSB7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvSW1hZ2UuamF2YSBiL2F3dC9qYXZhL2F3dC9JbWFnZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3YWUzZWQ4Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9JbWFnZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjA1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQXJlYUF2ZXJhZ2luZ1NjYWxlRmlsdGVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkZpbHRlcmVkSW1hZ2VTb3VyY2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW1hZ2VGaWx0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW1hZ2VPYnNlcnZlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbWFnZVByb2R1Y2VyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJlcGxpY2F0ZVNjYWxlRmlsdGVyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIEltYWdlIGFic3RyYWN0IGNsYXNzIHJlcHJlc2VudHMgdGhlIGdyYXBoaWMgaW1hZ2VzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEltYWdlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBVbmRlZmluZWRQcm9wZXJ0eSBvYmplY3Qgc2hvdWxkIGJlIHJldHVybmVkIGlmIHByb3BlcnR5IGlzIG5vdAotICAgICAqIGRlZmluZWQgZm9yIGEgcGFydGljdWxhciBpbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBVbmRlZmluZWRQcm9wZXJ0eSA9IG5ldyBPYmplY3QoKTsgLy8gJE5PTi1MT0NLLTEkCi0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU0NBTEVfREVGQVVMVCBpbmRpY2F0ZXMgdGhlIGRlZmF1bHQgaW1hZ2Ugc2NhbGluZyBhbGdvcml0aG0uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU0NBTEVfREVGQVVMVCA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU0NBTEVfRkFTVCBpbmRpY2F0ZXMgYW4gaW1hZ2Ugc2NhbGluZyBhbGdvcml0aG0gd2hpY2ggcGxhY2VzCi0gICAgICogYSBoaWdoZXIgcHJpb3JpdHkgb24gc2NhbGluZyBzcGVlZCB0aGFuIG9uIHRoZSBpbWFnZSdzIHNtb290aG5lc3MuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU0NBTEVfRkFTVCA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU0NBTEVfU01PT1RIIGluZGljYXRlcyBhbiBpbWFnZSBzY2FsaW5nIGFsZ29yaXRobSB3aGljaAotICAgICAqIHBsYWNlcyBhIGhpZ2hlciBwcmlvcml0eSBvbiBpbWFnZSBzbW9vdGhuZXNzIHRoYW4gb24gc2NhbGluZyBzcGVlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTQ0FMRV9TTU9PVEggPSA0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNDQUxFX1JFUExJQ0FURSBpbmRpY2F0ZXMgdGhlIGltYWdlIHNjYWxpbmcgYWxnb3JpdGhtIGluIHRoZQotICAgICAqIFJlcGxpY2F0ZVNjYWxlRmlsdGVyIGNsYXNzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNDQUxFX1JFUExJQ0FURSA9IDg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU0NBTEVfQVJFQV9BVkVSQUdJTkcgaW5kaWNhdGVzIHRoZSBhcmVhIGF2ZXJhZ2luZyBpbWFnZQotICAgICAqIHNjYWxpbmcgYWxnb3JpdGhtLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNDQUxFX0FSRUFfQVZFUkFHSU5HID0gMTY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYWNjZWxlcmF0aW9uIHByaW9yaXR5IGluZGljYXRlcyBpbWFnZSBhY2NlbGVyYXRpb24uCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGZsb2F0IGFjY2VsZXJhdGlvblByaW9yaXR5ID0gMC41ZjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBjYXBhYmlsaXRpZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgSW1hZ2VDYXBhYmlsaXRpZXMgY2FwYWJpbGl0aWVzID0gbmV3IEltYWdlQ2FwYWJpbGl0aWVzKGZhbHNlKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGltYWdlIHByb3BlcnR5IHdpdGggdGhlIHNwZWNpZmllZCBuYW1lLiBUaGUgVW5kZWZpbmVkUHJvcGVydHkKLSAgICAgKiBvYmplY3Qgc2hvdWxkIGJlIHJldHVybiBpZiB0aGUgcHJvcGVydHkgaXMgbm90IHNwZWNpZmllZCBmb3IgdGhpcyBpbWFnZS4KLSAgICAgKiBUaGUgcmV0dXJuIHZhbHVlIHNob3VsZCBiZSBudWxsIGlmIHRoZSBwcm9wZXJ0eSBpcyBjdXJyZW50bHkgdW5rbm93biB5ZXQKLSAgICAgKiBhbmQgdGhlIHNwZWNpZmllZCBJbWFnZU9ic2VydmVyIGlzIHRvIGJlIG5vdGlmaWVkIGxhdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiBpbWFnZSdzIHByb3BlcnR5LgotICAgICAqIEBwYXJhbSBvYnNlcnZlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlT2JzZXJ2ZXIuCi0gICAgICogQHJldHVybiB0aGUgT2JqZWN0IHdoaWNoIHJlcHJlc2VudHMgdmFsdWUgb2YgdGhlIHNwZWNpZmllZCBwcm9wZXJ0eS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgT2JqZWN0IGdldFByb3BlcnR5KFN0cmluZyBuYW1lLCBJbWFnZU9ic2VydmVyIG9ic2VydmVyKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEltYWdlUHJvZHVjZXIgb2JqZWN0IHdoaWNoIHJlcHJlc2VudHMgZGF0YSBvZiB0aGlzIEltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEltYWdlUHJvZHVjZXIgb2JqZWN0IHdoaWNoIHJlcHJlc2VudHMgZGF0YSBvZiB0aGlzIEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJbWFnZVByb2R1Y2VyIGdldFNvdXJjZSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgd2lkdGggb2YgdGhpcyBpbWFnZS4gVGhlIHNwZWNpZmllZCBJbWFnZU9ic2VydmVyIG9iamVjdCBpcwotICAgICAqIG5vdGlmaWVkIHdoZW4gdGhlIHdpZHRoIG9mIHRoaXMgaW1hZ2UgaXMgYXZhaWxhYmxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvYnNlcnZlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlT2JzZXJ2ZXIgb2JqZWN0IHdoaWNoIGlzIGlzIG5vdGlmaWVkIHdoZW4gdGhlIHdpZHRoCi0gICAgICogICAgICAgICAgICBvZiB0aGlzIGltYWdlIGlzIGF2YWlsYWJsZS4KLSAgICAgKiBAcmV0dXJuIHRoZSB3aWR0aCBvZiBpbWFnZSwgb3IgLTEgaWYgdGhlIHdpZHRoIG9mIHRoaXMgaW1hZ2UgaXMgbm90Ci0gICAgICogICAgICAgICBhdmFpbGFibGUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXRXaWR0aChJbWFnZU9ic2VydmVyIG9ic2VydmVyKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGhlaWdodCBvZiB0aGlzIGltYWdlLiBUaGUgc3BlY2lmaWVkIEltYWdlT2JzZXJ2ZXIgb2JqZWN0IGlzCi0gICAgICogbm90aWZpZWQgd2hlbiB0aGUgaGVpZ2h0IG9mIHRoaXMgaW1hZ2UgaXMgYXZhaWxhYmxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvYnNlcnZlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlT2JzZXJ2ZXIgb2JqZWN0IHdoaWNoIGlzIGlzIG5vdGlmaWVkIHdoZW4gdGhlIGhlaWdodAotICAgICAqICAgICAgICAgICAgb2YgdGhpcyBpbWFnZSBpcyBhdmFpbGFibGUuCi0gICAgICogQHJldHVybiB0aGUgaGVpZ2h0IG9mIGltYWdlLCBvciAtMSBpZiB0aGUgaGVpZ2h0IG9mIHRoaXMgaW1hZ2UgaXMgbm90Ci0gICAgICogICAgICAgICBhdmFpbGFibGUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXRIZWlnaHQoSW1hZ2VPYnNlcnZlciBvYnNlcnZlcik7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzY2FsZWQgaW5zdGFuY2Ugb2YgdGhpcyBJbWFnZS4gVGhpcyBtZXRob2QgcmV0dXJucyBhbiBJbWFnZQotICAgICAqIG9iamVjdCBjb25zdHJ1Y3RlZCBmcm9tIHRoZSBzb3VyY2Ugb2YgdGhpcyBpbWFnZSB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiB3aWR0aCwgaGVpZ2h0LCBhbmQgYXBwbGllZCBzY2FsaW5nIGFsZ29yaXRobS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiBzY2FsZWQgSW1hZ2UuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBzY2FsZWQgSW1hZ2UuCi0gICAgICogQHBhcmFtIGhpbnRzCi0gICAgICogICAgICAgICAgICB0aGUgY29uc3RhbnQgd2hpY2ggaW5kaWNhdGVzIHNjYWxpbmcgYWxnb3JpdGhtLgotICAgICAqIEByZXR1cm4gdGhlIHNjYWxlZCBJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2UgZ2V0U2NhbGVkSW5zdGFuY2UoaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgaGludHMpIHsKLSAgICAgICAgSW1hZ2VGaWx0ZXIgZmlsdGVyOwotICAgICAgICBpZiAoKGhpbnRzICYgKFNDQUxFX1NNT09USCB8IFNDQUxFX0FSRUFfQVZFUkFHSU5HKSkgIT0gMCkgewotICAgICAgICAgICAgZmlsdGVyID0gbmV3IEFyZWFBdmVyYWdpbmdTY2FsZUZpbHRlcih3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZpbHRlciA9IG5ldyBSZXBsaWNhdGVTY2FsZUZpbHRlcih3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgfQotICAgICAgICBJbWFnZVByb2R1Y2VyIHByb2R1Y2VyID0gbmV3IEZpbHRlcmVkSW1hZ2VTb3VyY2UoZ2V0U291cmNlKCksIGZpbHRlcik7Ci0gICAgICAgIHJldHVybiBUb29sa2l0LmdldERlZmF1bHRUb29sa2l0KCkuY3JlYXRlSW1hZ2UocHJvZHVjZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBHcmFwaGljcyBvYmplY3QgZm9yIHJlbmRlcmluZyB0aGlzIGltYWdlLiBUaGlzIG1ldGhvZCBjYW4gYmUgdXNlZAotICAgICAqIGZvciBvZmYtc2NyZWVuIGltYWdlcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgR3JhcGhpY3Mgb2JqZWN0IGZvciByZW5kZXJpbmcgdG8gdGhpcyBpbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgR3JhcGhpY3MgZ2V0R3JhcGhpY3MoKTsKLQotICAgIC8qKgotICAgICAqIEZsdXNoZXMgcmVzb3VyY2VzIHdoaWNoIGFyZSB1c2VkIGJ5IHRoaXMgSW1hZ2Ugb2JqZWN0LiBUaGlzIG1ldGhvZCByZXNldHMKLSAgICAgKiB0aGUgaW1hZ2UgdG8gdGhlIHJlY29uc3RydWN0ZWQgc3RhdGUgZnJvbSB0aGUgaW1hZ2UncyBzb3VyY2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZmx1c2goKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFjY2VsZXJhdGlvbiBwcmlvcml0eSBvZiB0aGlzIGltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFjY2VsZXJhdGlvbiBwcmlvcml0eSBvZiB0aGlzIGltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRBY2NlbGVyYXRpb25Qcmlvcml0eSgpIHsKLSAgICAgICAgcmV0dXJuIGFjY2VsZXJhdGlvblByaW9yaXR5OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGFjY2VsZXJhdGlvbiBwcmlvcml0eSBmb3IgdGhpcyBpbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJpb3JpdHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgYWNjZWxlcmF0aW9uIHByaW9yaXR5ICh2YWx1ZSBpbiB0aGUgcmFuZ2UgMC0xKS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRBY2NlbGVyYXRpb25Qcmlvcml0eShmbG9hdCBwcmlvcml0eSkgewotICAgICAgICBpZiAocHJpb3JpdHkgPCAwIHx8IHByaW9yaXR5ID4gMSkgewotICAgICAgICAgICAgLy8gYXd0LjEwQT1Qcmlvcml0eSBtdXN0IGJlIGEgdmFsdWUgYmV0d2VlbiAwIGFuZCAxLCBpbmNsdXNpdmUKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTBBIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgYWNjZWxlcmF0aW9uUHJpb3JpdHkgPSBwcmlvcml0eTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIEltYWdlQ2FwYWJpbGl0aWVzIG9iamVjdCBvZiB0aGlzIEltYWdlIG9iamVjdCBmb3IgdGhlIHNwZWNpZmllZAotICAgICAqIEdyYXBoaWNzQ29uZmlndXJhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2MKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgR3JhcGhpY3NDb25maWd1cmF0aW9uIG9iamVjdCAobnVsbCB2YWx1ZSBtZWFucwotICAgICAqICAgICAgICAgICAgZGVmYXVsdCBHcmFwaGljc0NvbmZpZ3VyYXRpb24pLgotICAgICAqIEByZXR1cm4gYW4gSW1hZ2VDYXBhYmlsaXRpZXMgb2JqZWN0IG9mIHRoaXMgSW1hZ2Ugb2JqZWN0IGZvciB0aGUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZCBHcmFwaGljc0NvbmZpZ3VyYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIEltYWdlQ2FwYWJpbGl0aWVzIGdldENhcGFiaWxpdGllcyhHcmFwaGljc0NvbmZpZ3VyYXRpb24gZ2MpIHsKLSAgICAgICAgLy8gTm90ZTogY29tbW9uIGltYWdlIGlzIG5vdCBhY2NlbGVyYXRlZC4KLSAgICAgICAgcmV0dXJuIGNhcGFiaWxpdGllczsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvSW1hZ2VDYXBhYmlsaXRpZXMuamF2YSBiL2F3dC9qYXZhL2F3dC9JbWFnZUNhcGFiaWxpdGllcy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjNmQ1OTQ2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9JbWFnZUNhcGFiaWxpdGllcy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNzggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBBbGV4ZXkgQS4gUGV0cmVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi0vKioKLSAqIFRoZSBJbWFnZUNhcGFiaWxpdGllcyBjbGFzcyBnaXZlcyBpbmZvcm1hdGlvbiBhYm91dCBhbiBpbWFnZSdzIGNhcGFiaWxpdGllcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJbWFnZUNhcGFiaWxpdGllcyBpbXBsZW1lbnRzIENsb25lYWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYWNjZWxlcmF0ZWQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBib29sZWFuIGFjY2VsZXJhdGVkOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEltYWdlQ2FwYWJpbGl0aWVzIHdpdGggdGhlIHNwZWNpZmllZCBhY2NlbGVyYXRpb24gZmxhZwotICAgICAqIHdoaWNoIGluZGljYXRlcyB3aGV0aGVyIGFjY2VsZXJhdGlvbiBpcyBkZXNpcmVkIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYWNjZWxlcmF0ZWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBhY2NlbGVyYXRlZCBmbGFnLgotICAgICAqLwotICAgIHB1YmxpYyBJbWFnZUNhcGFiaWxpdGllcyhib29sZWFuIGFjY2VsZXJhdGVkKSB7Ci0gICAgICAgIHRoaXMuYWNjZWxlcmF0ZWQgPSBhY2NlbGVyYXRlZDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgY29weSBvZiB0aGlzIEltYWdlQ2FwYWJpbGl0aWVzIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjb3B5IG9mIHRoaXMgSW1hZ2VDYXBhYmlsaXRpZXMgb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgY2xvbmUoKSB7Ci0gICAgICAgIHJldHVybiBuZXcgSW1hZ2VDYXBhYmlsaXRpZXMoYWNjZWxlcmF0ZWQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgSW1hZ2Ugb2YgdGhpcyBJbWFnZUNhcGFiaWxpdGllcyBpcyBvciBjYW4gYmUKLSAgICAgKiBhY2NlbGVyYXRlZC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBJbWFnZSBvZiB0aGlzIEltYWdlQ2FwYWJpbGl0aWVzIGlzIG9yIGNhbiBiZQotICAgICAqICAgICAgICAgYWNjZWxlcmF0ZWQsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0FjY2VsZXJhdGVkKCkgewotICAgICAgICByZXR1cm4gYWNjZWxlcmF0ZWQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoaXMgSW1hZ2VDYXBhYmlsaXRpZXMgYXBwbGllcyB0byB0aGUgVm9sYXRpbGVJbWFnZSB3aGljaAotICAgICAqIGNhbiBsb3NlIGl0cyBzdXJmYWNlcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgdGhpcyBJbWFnZUNhcGFiaWxpdGllcyBhcHBsaWVzIHRvIHRoZSBWb2xhdGlsZUltYWdlIHdoaWNoCi0gICAgICogICAgICAgICBjYW4gbG9zZSBpdHMgc3VyZmFjZXMsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1RydWVWb2xhdGlsZSgpIHsKLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L0luc2V0cy5qYXZhIGIvYXd0L2phdmEvYXd0L0luc2V0cy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwNGYxOThjLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9JbnNldHMuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE3OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERtaXRyeSBBLiBEdXJuZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5pby5TZXJpYWxpemFibGU7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkubWlzYy5IYXNoQ29kZTsKLQotLyoqCi0gKiBUaGUgSW5zZXRzIGNsYXNzIHJlcHJlc2VudHMgdGhlIGJvcmRlcnMgb2YgYSBjb250YWluZXIuIFRoaXMgY2xhc3MgZGVzY3JpYmVzCi0gKiB0aGUgc3BhY2UgdGhhdCBhIGNvbnRhaW5lciBzaG91bGQgbGVhdmUgYXQgZWFjaCBlZGdlOiB0aGUgdG9wLCB0aGUgYm90dG9tLAotICogdGhlIHJpZ2h0IHNpZGUsIGFuZCB0aGUgbGVmdCBzaWRlLiBUaGUgc3BhY2UgY2FuIGJlIGZpbGxlZCB3aXRoIGEgYm9yZGVyLCBhCi0gKiBibGFuayBzcGFjZSwgb3IgYSB0aXRsZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJbnNldHMgaW1wbGVtZW50cyBDbG9uZWFibGUsIFNlcmlhbGl6YWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtMjI3MjU3MjYzNzY5NTQ2Njc0OUw7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdG9wIGluc2V0IGluZGljYXRlcyB0aGUgc2l6ZSBvZiB0aGUgc3BhY2UgYWRkZWQgdG8gdGhlIHRvcCBvZiB0aGUKLSAgICAgKiByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIGludCB0b3A7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbGVmdCBpbnNldCBpbmRpY2F0ZXMgdGhlIHNpemUgb2YgdGhlIHNwYWNlIGFkZGVkIHRvIHRoZSBsZWZ0IHNpZGUgb2YKLSAgICAgKiB0aGUgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgbGVmdDsKLQotICAgIC8qKgotICAgICAqIFRoZSBib3R0b20gaW5zZXQgaW5kaWNhdGVzIHRoZSBzaXplIG9mIHRoZSBzcGFjZSBzdWJ0cmFjdGVkIGZyb20gdGhlCi0gICAgICogYm90dG9tIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBib3R0b207Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcmlnaHQgaW5zZXQgaW5kaWNhdGVzIHRoZSBzaXplIG9mIHRoZSBzcGFjZSBzdWJ0cmFjdGVkIGZyb20gdGhlIHJpZ2h0Ci0gICAgICogc2lkZSBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgcmlnaHQ7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW5zZXQgb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCB0b3AsIGxlZnQsIGJvdHRvbSwKLSAgICAgKiByaWdodCBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0b3AKLSAgICAgKiAgICAgICAgICAgIHRoZSB0b3AgaW5zZXQuCi0gICAgICogQHBhcmFtIGxlZnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZWZ0IGluc2V0LgotICAgICAqIEBwYXJhbSBib3R0b20KLSAgICAgKiAgICAgICAgICAgIHRoZSBib3R0b20gaW5zZXQuCi0gICAgICogQHBhcmFtIHJpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgcmlnaHQgaW5zZXQuCi0gICAgICovCi0gICAgcHVibGljIEluc2V0cyhpbnQgdG9wLCBpbnQgbGVmdCwgaW50IGJvdHRvbSwgaW50IHJpZ2h0KSB7Ci0gICAgICAgIHNldFZhbHVlcyh0b3AsIGxlZnQsIGJvdHRvbSwgcmlnaHQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBoYXNoIGNvZGUgb2YgdGhlIEluc2V0cyBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiBhIGhhc2ggY29kZSBvZiB0aGUgSW5zZXRzIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGhhc2hDb2RlKCkgewotICAgICAgICBpbnQgaGFzaENvZGUgPSBIYXNoQ29kZS5FTVBUWV9IQVNIX0NPREU7Ci0gICAgICAgIGhhc2hDb2RlID0gSGFzaENvZGUuY29tYmluZShoYXNoQ29kZSwgdG9wKTsKLSAgICAgICAgaGFzaENvZGUgPSBIYXNoQ29kZS5jb21iaW5lKGhhc2hDb2RlLCBsZWZ0KTsKLSAgICAgICAgaGFzaENvZGUgPSBIYXNoQ29kZS5jb21iaW5lKGhhc2hDb2RlLCBib3R0b20pOwotICAgICAgICBoYXNoQ29kZSA9IEhhc2hDb2RlLmNvbWJpbmUoaGFzaENvZGUsIHJpZ2h0KTsKLSAgICAgICAgcmV0dXJuIGhhc2hDb2RlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBjb3B5IG9mIHRoaXMgSW5zZXRzIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgY29weSBvZiB0aGlzIEluc2V0cyBvYmplY3QuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBjbG9uZSgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJbnNldHModG9wLCBsZWZ0LCBib3R0b20sIHJpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyBJbnNldHMgb2JqZWN0IGlzIGVxdWFsIHRvIHRoZSBzcGVjaWZpZWQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvCi0gICAgICogICAgICAgICAgICB0aGUgT2JqZWN0IHRvIGJlIGNvbXBhcmVkLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIG9iamVjdCBpcyBhbiBJbnNldHMgb2JqZWN0IHdob3NlIGRhdGEgdmFsdWVzIGFyZQotICAgICAqICAgICAgICAgZXF1YWwgdG8gdGhvc2Ugb2YgdGhpcyBvYmplY3QsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG8pIHsKLSAgICAgICAgaWYgKG8gPT0gdGhpcykgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG8gaW5zdGFuY2VvZiBJbnNldHMpIHsKLSAgICAgICAgICAgIEluc2V0cyBpID0gKEluc2V0cylvOwotICAgICAgICAgICAgcmV0dXJuICgoaS5sZWZ0ID09IGxlZnQpICYmIChpLmJvdHRvbSA9PSBib3R0b20pICYmIChpLnJpZ2h0ID09IHJpZ2h0KSAmJiAoaS50b3AgPT0gdG9wKSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBTdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBJbnNldHMgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBTdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBJbnNldHMgb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgIC8qCi0gICAgICAgICAqIFRoZSBmb3JtYXQgaXMgYmFzZWQgb24gMS41IHJlbGVhc2UgYmVoYXZpb3Igd2hpY2ggY2FuIGJlIHJldmVhbGVkIGJ5Ci0gICAgICAgICAqIHRoZSBmb2xsb3dpbmcgY29kZTogU3lzdGVtLm91dC5wcmludGxuKG5ldyBJbnNldHMoMSwgMiwgMywgNCkpOwotICAgICAgICAgKi8KLQotICAgICAgICByZXR1cm4gKGdldENsYXNzKCkuZ2V0TmFtZSgpICsgIltsZWZ0PSIgKyBsZWZ0ICsgIix0b3A9IiArIHRvcCArIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgICAgICIscmlnaHQ9IiArIHJpZ2h0ICsgIixib3R0b209IiArIGJvdHRvbSArICJdIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0b3AsIGxlZnQsIGJvdHRvbSwgYW5kIHJpZ2h0IGluc2V0cyB0byB0aGUgc3BlY2lmaWVkIHZhbHVlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdG9wCi0gICAgICogICAgICAgICAgICB0aGUgdG9wIGluc2V0LgotICAgICAqIEBwYXJhbSBsZWZ0Ci0gICAgICogICAgICAgICAgICB0aGUgbGVmdCBpbnNldC4KLSAgICAgKiBAcGFyYW0gYm90dG9tCi0gICAgICogICAgICAgICAgICB0aGUgYm90dG9tIGluc2V0LgotICAgICAqIEBwYXJhbSByaWdodAotICAgICAqICAgICAgICAgICAgdGhlIHJpZ2h0IGluc2V0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldChpbnQgdG9wLCBpbnQgbGVmdCwgaW50IGJvdHRvbSwgaW50IHJpZ2h0KSB7Ci0gICAgICAgIHNldFZhbHVlcyh0b3AsIGxlZnQsIGJvdHRvbSwgcmlnaHQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHZhbHVlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdG9wCi0gICAgICogICAgICAgICAgICB0aGUgdG9wLgotICAgICAqIEBwYXJhbSBsZWZ0Ci0gICAgICogICAgICAgICAgICB0aGUgbGVmdC4KLSAgICAgKiBAcGFyYW0gYm90dG9tCi0gICAgICogICAgICAgICAgICB0aGUgYm90dG9tLgotICAgICAqIEBwYXJhbSByaWdodAotICAgICAqICAgICAgICAgICAgdGhlIHJpZ2h0LgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBzZXRWYWx1ZXMoaW50IHRvcCwgaW50IGxlZnQsIGludCBib3R0b20sIGludCByaWdodCkgewotICAgICAgICB0aGlzLnRvcCA9IHRvcDsKLSAgICAgICAgdGhpcy5sZWZ0ID0gbGVmdDsKLSAgICAgICAgdGhpcy5ib3R0b20gPSBib3R0b207Ci0gICAgICAgIHRoaXMucmlnaHQgPSByaWdodDsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvSXRlbVNlbGVjdGFibGUuamF2YSBiL2F3dC9qYXZhL2F3dC9JdGVtU2VsZWN0YWJsZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyMTJjZjcwLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9JdGVtU2VsZWN0YWJsZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuZXZlbnQuSXRlbUxpc3RlbmVyOwotCi0vKioKLSAqIFRoZSBJdGVtU2VsZWN0YWJsZSBpbnRlcmZhY2UgcmVwcmVzZW50cyBhIHNldCBvZiBpdGVtcyB3aGljaCBjYW4gYmUgc2VsZWN0ZWQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIEl0ZW1TZWxlY3RhYmxlIHsKLQotICAgIC8qKgotICAgICAqIEFkZHMgYW4gSXRlbUxpc3RlbmVyIGZvciByZWNlaXZpbmcgaXRlbSBldmVudHMgd2hlbiB0aGUgc3RhdGUgb2YgYW4gaXRlbQotICAgICAqIGlzIGNoYW5nZWQgYnkgdGhlIHVzZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBJdGVtTGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkSXRlbUxpc3RlbmVyKEl0ZW1MaXN0ZW5lciBsKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgdGhlIHNlbGVjdGVkIG9iamVjdHMgb3IgbnVsbCBpZiB0aGVyZSBpcyBubyBzZWxlY3RlZAotICAgICAqIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIHRoZSBzZWxlY3RlZCBvYmplY3RzIG9yIG51bGwgaWYgdGhlcmUgaXMgbm8gc2VsZWN0ZWQKLSAgICAgKiAgICAgICAgIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0W10gZ2V0U2VsZWN0ZWRPYmplY3RzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHRoZSBzcGVjaWZpZWQgSXRlbUxpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgSXRlbUxpc3RlbmVyIHdoaWNoIHdpbGwgYmUgcmVtb3ZlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVJdGVtTGlzdGVuZXIoSXRlbUxpc3RlbmVyIGwpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvTWVudUNvbXBvbmVudC5qYXZhIGIvYXd0L2phdmEvYXd0L01lbnVDb21wb25lbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOWMxYjEyMC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvTWVudUNvbXBvbmVudC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNzgzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmF3dC5ldmVudC5Gb2N1c0xpc3RlbmVyOwotaW1wb3J0IGphdmEuYXd0LmV2ZW50Lk1vdXNlRXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QucGVlci5NZW51Q29tcG9uZW50UGVlcjsKLWltcG9ydCBqYXZhLmlvLlNlcmlhbGl6YWJsZTsKLWltcG9ydCBqYXZhLnV0aWwuTG9jYWxlOyAvL2ltcG9ydCBqYXZheC5hY2Nlc3NpYmlsaXR5LkFjY2Vzc2libGU7Ci0vL2ltcG9ydCBqYXZheC5hY2Nlc3NpYmlsaXR5LkFjY2Vzc2libGVDb21wb25lbnQ7Ci0vL2ltcG9ydCBqYXZheC5hY2Nlc3NpYmlsaXR5LkFjY2Vzc2libGVDb250ZXh0OwotLy9pbXBvcnQgamF2YXguYWNjZXNzaWJpbGl0eS5BY2Nlc3NpYmxlUm9sZTsKLS8vaW1wb3J0IGphdmF4LmFjY2Vzc2liaWxpdHkuQWNjZXNzaWJsZVNlbGVjdGlvbjsKLS8vaW1wb3J0IGphdmF4LmFjY2Vzc2liaWxpdHkuQWNjZXNzaWJsZVN0YXRlU2V0OwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuTXVsdGlSZWN0QXJlYTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LnN0YXRlLk1lbnVJdGVtU3RhdGU7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5zdGF0ZS5NZW51U3RhdGU7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Ob3RJbXBsZW1lbnRlZEV4Y2VwdGlvbjsKLQotLyoqCi0gKiBUaGUgTWVudUNvbXBvbmVudCBhYnN0cmFjdCBjbGFzcyBpcyB0aGUgc3VwZXJjbGFzcyBmb3IgbWVudSBjb21wb25lbnRzLiBNZW51Ci0gKiBjb21wb25lbnRzIHJlY2VpdmUgYW5kIHByb2Nlc3MgQVdUIGV2ZW50cy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBNZW51Q29tcG9uZW50IGltcGxlbWVudHMgU2VyaWFsaXphYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC00NTM2OTAyMzU2MjIzODk0Mzc5TDsKLQotICAgIC8qKgotICAgICAqIFRoZSBuYW1lLgotICAgICAqLwotICAgIHByaXZhdGUgU3RyaW5nIG5hbWU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZm9udC4KLSAgICAgKi8KLSAgICBwcml2YXRlIEZvbnQgZm9udDsKLQotICAgIC8qKgotICAgICAqIFRoZSBwYXJlbnQuCi0gICAgICovCi0gICAgTWVudUNvbnRhaW5lciBwYXJlbnQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGVwcmVjYXRlZCBldmVudCBoYW5kbGVyLgotICAgICAqLwotICAgIGJvb2xlYW4gZGVwcmVjYXRlZEV2ZW50SGFuZGxlciA9IHRydWU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc2VsZWN0ZWQgaXRlbSBpbmRleC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBzZWxlY3RlZEl0ZW1JbmRleDsKLQotICAgIC8vID8/P0FXVDogcHJpdmF0ZSBBY2Nlc3NpYmxlQ29udGV4dCBhY2Nlc3NpYmxlQ29udGV4dDsKLQotICAgIC8qKgotICAgICAqIFRoZSB0b29sa2l0LgotICAgICAqLwotICAgIGZpbmFsIFRvb2xraXQgdG9vbGtpdCA9IFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKTsKLQotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogcHJvdGVjdGVkIGFic3RyYWN0IGNsYXNzIEFjY2Vzc2libGVBV1RNZW51Q29tcG9uZW50IGV4dGVuZHMKLSAgICAgKiBBY2Nlc3NpYmxlQ29udGV4dCBpbXBsZW1lbnRzIFNlcmlhbGl6YWJsZSwgQWNjZXNzaWJsZUNvbXBvbmVudCwKLSAgICAgKiBBY2Nlc3NpYmxlU2VsZWN0aW9uIHsgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0KLSAgICAgKiAtNDI2OTUzMzQxNjIyMzc5ODY5OEw7IHB1YmxpYyB2b2lkIGFkZEZvY3VzTGlzdGVuZXIoRm9jdXNMaXN0ZW5lcgotICAgICAqIGxpc3RlbmVyKSB7IH0gcHVibGljIGJvb2xlYW4gY29udGFpbnMoUG9pbnQgcHQpIHsgcmV0dXJuIGZhbHNlOyB9IHB1YmxpYwotICAgICAqIEFjY2Vzc2libGUgZ2V0QWNjZXNzaWJsZUF0KFBvaW50IHB0KSB7IHJldHVybiBudWxsOyB9IHB1YmxpYyBDb2xvcgotICAgICAqIGdldEJhY2tncm91bmQoKSB7IHJldHVybiBudWxsOyB9IHB1YmxpYyBSZWN0YW5nbGUgZ2V0Qm91bmRzKCkgeyByZXR1cm4KLSAgICAgKiBudWxsOyB9IHB1YmxpYyBDdXJzb3IgZ2V0Q3Vyc29yKCkgeyByZXR1cm4gbnVsbDsgfSBwdWJsaWMgRm9udCBnZXRGb250KCkKLSAgICAgKiB7IHJldHVybiBNZW51Q29tcG9uZW50LnRoaXMuZ2V0Rm9udCgpOyB9IHB1YmxpYyBGb250TWV0cmljcwotICAgICAqIGdldEZvbnRNZXRyaWNzKEZvbnQgZm9udCkgeyByZXR1cm4gbnVsbDsgfSBwdWJsaWMgQ29sb3IgZ2V0Rm9yZWdyb3VuZCgpIHsKLSAgICAgKiByZXR1cm4gbnVsbDsgfSBwdWJsaWMgUG9pbnQgZ2V0TG9jYXRpb24oKSB7IHJldHVybiBudWxsOyB9IHB1YmxpYyBQb2ludAotICAgICAqIGdldExvY2F0aW9uT25TY3JlZW4oKSB7IHJldHVybiBudWxsOyB9IHB1YmxpYyBEaW1lbnNpb24gZ2V0U2l6ZSgpIHsKLSAgICAgKiByZXR1cm4gbnVsbDsgfSBwdWJsaWMgYm9vbGVhbiBpc0VuYWJsZWQoKSB7IHJldHVybiB0cnVlOyAvLyBhbHdheXMKLSAgICAgKiBlbmFibGVkIH0gcHVibGljIGJvb2xlYW4gaXNGb2N1c1RyYXZlcnNhYmxlKCkgeyByZXR1cm4gdHJ1ZTsgLy8gYWx3YXlzCi0gICAgICogZm9jdXMgdHJhdmVyc2FibGUgfSBwdWJsaWMgYm9vbGVhbiBpc1Nob3dpbmcoKSB7IHJldHVybiB0cnVlOy8vIGFsd2F5cwotICAgICAqIHNob3dpbmcgfSBwdWJsaWMgYm9vbGVhbiBpc1Zpc2libGUoKSB7IHJldHVybiB0cnVlOyAvLyBhbHdheXMgdmlzaWJsZSB9Ci0gICAgICogcHVibGljIHZvaWQgcmVtb3ZlRm9jdXNMaXN0ZW5lcihGb2N1c0xpc3RlbmVyIGxpc3RlbmVyKSB7IH0gcHVibGljIHZvaWQKLSAgICAgKiByZXF1ZXN0Rm9jdXMoKSB7IH0gcHVibGljIHZvaWQgc2V0QmFja2dyb3VuZChDb2xvciBjb2xvcikgeyB9IHB1YmxpYyB2b2lkCi0gICAgICogc2V0Qm91bmRzKFJlY3RhbmdsZSByZWN0KSB7IH0gcHVibGljIHZvaWQgc2V0Q3Vyc29yKEN1cnNvciBjdXJzb3IpIHsgfQotICAgICAqIHB1YmxpYyB2b2lkIHNldEVuYWJsZWQoYm9vbGVhbiBlbmFibGVkKSB7IH0gcHVibGljIHZvaWQgc2V0Rm9udChGb250Ci0gICAgICogZm9udCkgeyBNZW51Q29tcG9uZW50LnRoaXMuc2V0Rm9udChmb250KTsgfSBwdWJsaWMgdm9pZAotICAgICAqIHNldEZvcmVncm91bmQoQ29sb3IgY29sb3IpIHsgfSBwdWJsaWMgdm9pZCBzZXRMb2NhdGlvbihQb2ludCBwdCkgeyB9Ci0gICAgICogcHVibGljIHZvaWQgc2V0U2l6ZShEaW1lbnNpb24gcHQpIHsgfSBwdWJsaWMgdm9pZCBzZXRWaXNpYmxlKGJvb2xlYW4KLSAgICAgKiB2aXNpYmxlKSB7IH0gcHVibGljIHZvaWQgYWRkQWNjZXNzaWJsZVNlbGVjdGlvbihpbnQgaW5kZXgpIHsgfSBwdWJsaWMKLSAgICAgKiB2b2lkIGNsZWFyQWNjZXNzaWJsZVNlbGVjdGlvbigpIHsgfSBwdWJsaWMgQWNjZXNzaWJsZQotICAgICAqIGdldEFjY2Vzc2libGVTZWxlY3Rpb24oaW50IGluZGV4KSB7IHJldHVybiBudWxsOyB9IHB1YmxpYyBpbnQKLSAgICAgKiBnZXRBY2Nlc3NpYmxlU2VsZWN0aW9uQ291bnQoKSB7IHJldHVybiAwOyB9IHB1YmxpYyBib29sZWFuCi0gICAgICogaXNBY2Nlc3NpYmxlQ2hpbGRTZWxlY3RlZChpbnQgaW5kZXgpIHsgcmV0dXJuIGZhbHNlOyB9IHB1YmxpYyB2b2lkCi0gICAgICogcmVtb3ZlQWNjZXNzaWJsZVNlbGVjdGlvbihpbnQgaW5kZXgpIHsgfSBwdWJsaWMgdm9pZAotICAgICAqIHNlbGVjdEFsbEFjY2Vzc2libGVTZWxlY3Rpb24oKSB7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHVibGljIEFjY2Vzc2libGUgZ2V0QWNjZXNzaWJsZUNoaWxkKGludCBpbmRleCkgeyByZXR1cm4gbnVsbDsKLSAgICAgKiB9Ci0gICAgICogQE92ZXJyaWRlIHB1YmxpYyBpbnQgZ2V0QWNjZXNzaWJsZUNoaWxkcmVuQ291bnQoKSB7IHJldHVybiAwOyB9Ci0gICAgICogQE92ZXJyaWRlIHB1YmxpYyBBY2Nlc3NpYmxlQ29tcG9uZW50IGdldEFjY2Vzc2libGVDb21wb25lbnQoKSB7IHJldHVybgotICAgICAqIHRoaXM7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHVibGljIFN0cmluZyBnZXRBY2Nlc3NpYmxlRGVzY3JpcHRpb24oKSB7IHJldHVybgotICAgICAqIHN1cGVyLmdldEFjY2Vzc2libGVEZXNjcmlwdGlvbigpOyB9Ci0gICAgICogQE92ZXJyaWRlIHB1YmxpYyBpbnQgZ2V0QWNjZXNzaWJsZUluZGV4SW5QYXJlbnQoKSB7IHRvb2xraXQubG9ja0FXVCgpOwotICAgICAqIHRyeSB7IEFjY2Vzc2libGUgYVBhcmVudCA9IGdldEFjY2Vzc2libGVQYXJlbnQoKTsgaW50IGFJbmRleCA9IC0xOyBpZgotICAgICAqIChhUGFyZW50IGluc3RhbmNlb2YgTWVudUNvbXBvbmVudCkgeyBNZW51Q29tcG9uZW50IHBhcmVudCA9Ci0gICAgICogKE1lbnVDb21wb25lbnQpIGFQYXJlbnQ7IGludCBjb3VudCA9IHBhcmVudC5nZXRJdGVtQ291bnQoKTsgZm9yIChpbnQgaSA9Ci0gICAgICogMDsgaSA8IGNvdW50OyBpKyspIHsgTWVudUNvbXBvbmVudCBjb21wID0gcGFyZW50LmdldEl0ZW0oaSk7IGlmIChjb21wCi0gICAgICogaW5zdGFuY2VvZiBBY2Nlc3NpYmxlKSB7IGFJbmRleCsrOyBpZiAoY29tcCA9PSBNZW51Q29tcG9uZW50LnRoaXMpIHsKLSAgICAgKiByZXR1cm4gYUluZGV4OyB9IH0gfSB9IHJldHVybiAtMTsgfSBmaW5hbGx5IHsgdG9vbGtpdC51bmxvY2tBV1QoKTsgfSB9Ci0gICAgICogQE92ZXJyaWRlIHB1YmxpYyBTdHJpbmcgZ2V0QWNjZXNzaWJsZU5hbWUoKSB7IHJldHVybgotICAgICAqIHN1cGVyLmdldEFjY2Vzc2libGVOYW1lKCk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHVibGljIEFjY2Vzc2libGUgZ2V0QWNjZXNzaWJsZVBhcmVudCgpIHsgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICogdHJ5IHsgQWNjZXNzaWJsZSBhUGFyZW50ID0gc3VwZXIuZ2V0QWNjZXNzaWJsZVBhcmVudCgpOyBpZiAoYVBhcmVudCAhPQotICAgICAqIG51bGwpIHsgcmV0dXJuIGFQYXJlbnQ7IH0gTWVudUNvbnRhaW5lciBwYXJlbnQgPSBnZXRQYXJlbnQoKTsgaWYgKHBhcmVudAotICAgICAqIGluc3RhbmNlb2YgQWNjZXNzaWJsZSkgeyBhUGFyZW50ID0gKEFjY2Vzc2libGUpIHBhcmVudDsgfSByZXR1cm4gYVBhcmVudDsKLSAgICAgKiB9IGZpbmFsbHkgeyB0b29sa2l0LnVubG9ja0FXVCgpOyB9IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHVibGljIEFjY2Vzc2libGVSb2xlIGdldEFjY2Vzc2libGVSb2xlKCkgeyByZXR1cm4KLSAgICAgKiBBY2Nlc3NpYmxlUm9sZS5BV1RfQ09NUE9ORU5UOyB9Ci0gICAgICogQE92ZXJyaWRlIHB1YmxpYyBBY2Nlc3NpYmxlU2VsZWN0aW9uIGdldEFjY2Vzc2libGVTZWxlY3Rpb24oKSB7IHJldHVybgotICAgICAqIHRoaXM7IH0KLSAgICAgKiBAT3ZlcnJpZGUgcHVibGljIEFjY2Vzc2libGVTdGF0ZVNldCBnZXRBY2Nlc3NpYmxlU3RhdGVTZXQoKSB7IHJldHVybiBuZXcKLSAgICAgKiBBY2Nlc3NpYmxlU3RhdGVTZXQoKTsgfQotICAgICAqIEBPdmVycmlkZSBwdWJsaWMgTG9jYWxlIGdldExvY2FsZSgpIHsgcmV0dXJuIExvY2FsZS5nZXREZWZhdWx0KCk7IH0gfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogVGhlIGFjY2Vzc29yIHRvIE1lbnVDb21wb25lbnQgaW50ZXJuYWwgc3RhdGUsIHV0aWxpemVkIGJ5IHRoZSB2aXN1YWwKLSAgICAgKiB0aGVtZS4KLSAgICAgKiAKLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgdGhlIGhlYWRsZXNzIGV4Y2VwdGlvbi4KLSAgICAgKi8KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIGNsYXNzIFN0YXRlIGltcGxlbWVudHMgTWVudVN0YXRlIHsgRGltZW5zaW9uIHNpemU7IERpbWVuc2lvbiBnZXRTaXplKCkgewotICAgICAqIGlmIChzaXplID09IG51bGwpIHsgY2FsY3VsYXRlKCk7IH0gcmV0dXJuIHNpemU7IH0gcHVibGljIGludCBnZXRXaWR0aCgpIHsKLSAgICAgKiByZXR1cm4gZ2V0U2l6ZSgpLndpZHRoOyB9IHB1YmxpYyBpbnQgZ2V0SGVpZ2h0KCkgeyByZXR1cm4KLSAgICAgKiBnZXRTaXplKCkuaGVpZ2h0OyB9IHB1YmxpYyBGb250IGdldEZvbnQoKSB7IHJldHVybgotICAgICAqIE1lbnVDb21wb25lbnQudGhpcy5nZXRGb250KCk7IH0gcHVibGljIGludCBnZXRJdGVtQ291bnQoKSB7IHJldHVybgotICAgICAqIE1lbnVDb21wb25lbnQudGhpcy5nZXRJdGVtQ291bnQoKTsgfSBwdWJsaWMgaW50IGdldFNlbGVjdGVkSXRlbUluZGV4KCkgewotICAgICAqIHJldHVybiBNZW51Q29tcG9uZW50LnRoaXMuZ2V0U2VsZWN0ZWRJdGVtSW5kZXgoKTsgfSBwdWJsaWMgYm9vbGVhbgotICAgICAqIGlzRm9udFNldCgpIHsgcmV0dXJuIE1lbnVDb21wb25lbnQudGhpcy5pc0ZvbnRTZXQoKTsgfQotICAgICAqIEBTdXBwcmVzc1dhcm5pbmdzKCJkZXByZWNhdGlvbiIpIHB1YmxpYyBGb250TWV0cmljcyBnZXRGb250TWV0cmljcyhGb250Ci0gICAgICogZikgeyByZXR1cm4gTWVudUNvbXBvbmVudC50aGlzLnRvb2xraXQuZ2V0Rm9udE1ldHJpY3MoZik7IH0gcHVibGljIFBvaW50Ci0gICAgICogZ2V0TG9jYXRpb24oKSB7IHJldHVybiBNZW51Q29tcG9uZW50LnRoaXMuZ2V0TG9jYXRpb24oKTsgfSBwdWJsaWMKLSAgICAgKiBNZW51SXRlbVN0YXRlIGdldEl0ZW0oaW50IGluZGV4KSB7IE1lbnVJdGVtIGl0ZW0gPQotICAgICAqIE1lbnVDb21wb25lbnQudGhpcy5nZXRJdGVtKGluZGV4KTsgcmV0dXJuIGl0ZW0uaXRlbVN0YXRlOyB9IHB1YmxpYyB2b2lkCi0gICAgICogc2V0U2l6ZShpbnQgdywgaW50IGgpIHsgdGhpcy5zaXplID0gbmV3IERpbWVuc2lvbih3LCBoKTsgfSB2b2lkCi0gICAgICogY2FsY3VsYXRlKCkgeyBzaXplID0gbmV3IERpbWVuc2lvbigpOwotICAgICAqIHNpemUuc2V0U2l6ZSh0b29sa2l0LnRoZW1lLmNhbGN1bGF0ZU1lbnVTaXplKHRoaXMpKTsgfSB2b2lkIHJlc2V0KCkgeyBmb3IKLSAgICAgKiAoaW50IGkgPSAwOyBpIDwgZ2V0SXRlbUNvdW50KCk7IGkrKykgeyAoKE1lbnVJdGVtLlN0YXRlKQotICAgICAqIGdldEl0ZW0oaSkpLnJlc2V0KCk7IH0gfSB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBQb3AtdXAgYm94IGZvciBtZW51LiBJdCB0cmFuc2ZlcnMgdGhlIHBhaW50IGV2ZW50cywga2V5Ym9hcmQgYW5kIG1vdXNlCi0gICAgICogZXZlbnRzIHRvIHRoZSBtZW51IGNvbXBvbmVudCBpdHNlbGYuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBjbGFzcyBNZW51UG9wdXBCb3ggZXh0ZW5kcyBQb3B1cEJveCB7IHByaXZhdGUgZmluYWwgUG9pbnQgbGFzdE1vdXNlUG9zID0KLSAgICAgKiBuZXcgUG9pbnQoKTsKLSAgICAgKiBAT3ZlcnJpZGUgYm9vbGVhbiBpc01lbnUoKSB7IHJldHVybiB0cnVlOyB9Ci0gICAgICogQE92ZXJyaWRlIHZvaWQgcGFpbnQoR3JhcGhpY3MgZ3IpIHsgTWVudUNvbXBvbmVudC50aGlzLnBhaW50KGdyKTsgfQotICAgICAqIEBPdmVycmlkZSB2b2lkIG9uS2V5RXZlbnQoaW50IGV2ZW50SWQsIGludCB2S2V5LCBsb25nIHdoZW4sIGludAotICAgICAqIG1vZGlmaWVycykgeyBNZW51Q29tcG9uZW50LnRoaXMub25LZXlFdmVudChldmVudElkLCB2S2V5LCB3aGVuLAotICAgICAqIG1vZGlmaWVycyk7IH0KLSAgICAgKiBAT3ZlcnJpZGUgdm9pZCBvbk1vdXNlRXZlbnQoaW50IGV2ZW50SWQsIFBvaW50IHdoZXJlLCBpbnQgbW91c2VCdXR0b24sCi0gICAgICogbG9uZyB3aGVuLCBpbnQgbW9kaWZpZXJzLCBpbnQgd2hlZWxSb3RhdGlvbikgeyAvLyBwcmV2ZW50IGNvbmZsaWN0IG9mCi0gICAgICogbW91c2UgYW5kIGtleWJvYXJkIC8vIHdoZW4gc3ViLW1lbnUgZHJvcHMgZG93biBkdWUgdG8ga2V5Ym9hcmQgbmF2aWdhdGlvbgotICAgICAqIGlmIChsYXN0TW91c2VQb3MuZXF1YWxzKHdoZXJlKSAmJiAoZXZlbnRJZCA9PSBNb3VzZUV2ZW50Lk1PVVNFX01PVkVEIHx8Ci0gICAgICogZXZlbnRJZCA9PSBNb3VzZUV2ZW50Lk1PVVNFX0VOVEVSRUQpKSB7IHJldHVybjsgfQotICAgICAqIGxhc3RNb3VzZVBvcy5zZXRMb2NhdGlvbih3aGVyZSk7IE1lbnVDb21wb25lbnQudGhpcy5vbk1vdXNlRXZlbnQoZXZlbnRJZCwKLSAgICAgKiB3aGVyZSwgbW91c2VCdXR0b24sIHdoZW4sIG1vZGlmaWVycyk7IH0gfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IE1lbnVDb21wb25lbnQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgZ3JhcGhpY2FsIGludGVyZmFjZSBlbnZpcm9ubWVudCBjYW4ndCBzdXBwb3J0Ci0gICAgICogICAgICAgICAgICAgTWVudUNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHVibGljIE1lbnVDb21wb25lbnQoKSB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24gewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIFRvb2xraXQuY2hlY2tIZWFkbGVzcygpOwotICAgICAgICAgICAgbmFtZSA9IGF1dG9OYW1lKCk7Ci0gICAgICAgICAgICBzZWxlY3RlZEl0ZW1JbmRleCA9IC0xOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG5hbWUgb2YgdGhlIE1lbnVDb21wb25lbnQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG5hbWUgb2YgdGhlIE1lbnVDb21wb25lbnQgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0TmFtZSgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gbmFtZTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgU3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBNZW51Q29tcG9uZW50IG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgU3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBNZW51Q29tcG9uZW50IG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBnZXRDbGFzcygpLmdldE5hbWUoKSArICJbIiArIHBhcmFtU3RyaW5nKCkgKyAiXSI7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdG9vbGtpdC51bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHBhcmVudCBtZW51IGNvbnRhaW5lci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBwYXJlbnQuCi0gICAgICovCi0gICAgcHVibGljIE1lbnVDb250YWluZXIgZ2V0UGFyZW50KCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBwYXJlbnQ7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbmFtZSBvZiB0aGUgTWVudUNvbXBvbmVudCB0byB0aGUgc3BlY2lmaWVkIHN0cmluZy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBuYW1lIG9mIHRoZSBNZW51Q29tcG9uZW50IG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXROYW1lKFN0cmluZyBuYW1lKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgdGhpcy5uYW1lID0gbmFtZTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEaXNwYXRjaGVzIEFXVCBldmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXZlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBBV1RFdmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgdm9pZCBkaXNwYXRjaEV2ZW50KEFXVEV2ZW50IGV2ZW50KSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcHJvY2Vzc0V2ZW50KGV2ZW50KTsKLSAgICAgICAgICAgIGlmIChkZXByZWNhdGVkRXZlbnRIYW5kbGVyKSB7Ci0gICAgICAgICAgICAgICAgcG9zdERlcHJlY2F0ZWRFdmVudChldmVudCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUG9zdCBkZXByZWNhdGVkIGV2ZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBldmVudAotICAgICAqICAgICAgICAgICAgdGhlIGV2ZW50LgotICAgICAqLwotICAgIHZvaWQgcG9zdERlcHJlY2F0ZWRFdmVudChBV1RFdmVudCBldmVudCkgewotICAgICAgICBFdmVudCBldnQgPSBldmVudC5nZXRFdmVudCgpOwotICAgICAgICBpZiAoZXZ0ICE9IG51bGwpIHsKLSAgICAgICAgICAgIHBvc3RFdmVudChldnQpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcGVlciBvZiB0aGUgTWVudUNvbXBvbmVudDsgYW4gYXBwbGljYXRpb24gbXVzdCBub3QgdXNlIHRoaXMKLSAgICAgKiBtZXRob2QgZGlyZWN0bHkuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgTWVudUNvbXBvbmVudFBlZXIgb2JqZWN0LgotICAgICAqIEB0aHJvd3MgTm90SW1wbGVtZW50ZWRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGlzIG1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQgYnkgYSBzdWJjbGFzcy4KLSAgICAgKiBAZGVwcmVjYXRlZCBhbiBhcHBsaWNhdGlvbiBtdXN0IG5vdCB1c2UgdGhpcyBtZXRob2QgZGlyZWN0bHkuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgTWVudUNvbXBvbmVudFBlZXIgZ2V0UGVlcigpIHRocm93cyBvcmcuYXBhY2hlLmhhcm1vbnkubHVuaS51dGlsLk5vdEltcGxlbWVudGVkRXhjZXB0aW9uIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgICAgIGlmICh0cnVlKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCIpOyAvL1RPRE86IGltcGxlbWVudCAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9ja2luZyBvYmplY3Qgb2YgdGhpcyBNZW51Q29tcG9uZW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGxvY2tpbmcgb2JqZWN0IG9mIHRoaXMgTWVudUNvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgZmluYWwgT2JqZWN0IGdldFRyZWVMb2NrKCkgewotICAgICAgICByZXR1cm4gdG9vbGtpdC5hd3RUcmVlTG9jazsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQb3N0cyB0aGUgRXZlbnQgdG8gdGhlIE1lbnVDb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBFdmVudC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBldmVudCBpcyBwb3N0ZWQgc3VjY2Vzc2Z1bGx5LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICogQGRlcHJlY2F0ZWQgUmVwbGFjZWQgZGlzcGF0Y2hFdmVudCBtZXRob2QuCi0gICAgICovCi0gICAgQFN1cHByZXNzV2FybmluZ3MoImRlcHJlY2F0aW9uIikKLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyBib29sZWFuIHBvc3RFdmVudChFdmVudCBlKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKHBhcmVudCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHBhcmVudC5wb3N0RXZlbnQoZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBNZW51Q29tcG9uZW50IHN0YXRlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gcmV0dXJucyB0aGUgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBNZW51Q29tcG9uZW50IHN0YXRlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmcgcGFyYW1TdHJpbmcoKSB7Ci0gICAgICAgIHRvb2xraXQubG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGdldE5hbWUoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIHB1YmxpYyBBY2Nlc3NpYmxlQ29udGV4dCBnZXRBY2Nlc3NpYmxlQ29udGV4dCgpIHsgdG9vbGtpdC5sb2NrQVdUKCk7IHRyeQotICAgICAqIHsgaWYgKGFjY2Vzc2libGVDb250ZXh0ID09IG51bGwpIHsgYWNjZXNzaWJsZUNvbnRleHQgPQotICAgICAqIGNyZWF0ZUFjY2Vzc2libGVDb250ZXh0KCk7IH0gcmV0dXJuIGFjY2Vzc2libGVDb250ZXh0OyB9IGZpbmFsbHkgewotICAgICAqIHRvb2xraXQudW5sb2NrQVdUKCk7IH0gfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZm9udCBvZiB0aGUgTWVudUNvbXBvbmVudCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgRm9udCBvZiB0aGUgTWVudUNvbXBvbmVudCBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIEZvbnQgZ2V0Rm9udCgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpZiAoZm9udCA9PSBudWxsICYmIGhhc0RlZmF1bHRGb250KCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gdG9vbGtpdC5nZXREZWZhdWx0Rm9udCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGZvbnQgPT0gbnVsbCAmJiBwYXJlbnQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHJldHVybiBwYXJlbnQuZ2V0Rm9udCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGZvbnQ7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIGlzIGZvbnQgc2V0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgaXMgZm9udCBzZXQKLSAgICAgKi8KLSAgICBib29sZWFuIGlzRm9udFNldCgpIHsKLSAgICAgICAgcmV0dXJuIGZvbnQgIT0gbnVsbAotICAgICAgICAgICAgICAgIHx8ICgocGFyZW50IGluc3RhbmNlb2YgTWVudUNvbXBvbmVudCkgJiYgKChNZW51Q29tcG9uZW50KXBhcmVudCkuaXNGb250U2V0KCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBmb3IgZGVmYXVsdCBmb250LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgc3VjY2Vzc2Z1bC4KLSAgICAgKi8KLSAgICBib29sZWFuIGhhc0RlZmF1bHRGb250KCkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIGFuIEFXVEVldmVudCBvbiB0aGlzIG1lbnUgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBldmVudAotICAgICAqICAgICAgICAgICAgdGhlIEFXVEV2ZW50LgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NFdmVudChBV1RFdmVudCBldmVudCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIC8vIGRvIG5vdGhpbmcKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHRvb2xraXQudW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHRoZSBwZWVyIG9mIHRoZSBNZW51Q29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZU5vdGlmeSgpIHsKLSAgICAgICAgdG9vbGtpdC5sb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgRm9udCBmb3IgdGhpcyBNZW51Q29tcG9uZW50IG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZm9udAotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBGb250IHRvIGJlIHVzZWQgZm9yIHRoaXMgTWVudUNvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRGb250KEZvbnQgZm9udCkgewotICAgICAgICB0b29sa2l0LmxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHRoaXMuZm9udCA9IGZvbnQ7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB0b29sa2l0LnVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgcGFyZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXJlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgcGFyZW50LgotICAgICAqLwotICAgIHZvaWQgc2V0UGFyZW50KE1lbnVDb250YWluZXIgcGFyZW50KSB7Ci0gICAgICAgIHRoaXMucGFyZW50ID0gcGFyZW50OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvY2F0aW9uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGxvY2F0aW9uLgotICAgICAqLwotICAgIFBvaW50IGdldExvY2F0aW9uKCkgewotICAgICAgICAvLyB0byBiZSBvdmVycmlkZGVuCi0gICAgICAgIHJldHVybiBuZXcgUG9pbnQoMCwgMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgd2lkdGguCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgd2lkdGguCi0gICAgICovCi0gICAgaW50IGdldFdpZHRoKCkgewotICAgICAgICAvLyB0byBiZSBvdmVycmlkZGVuCi0gICAgICAgIHJldHVybiAxOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGhlaWdodC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBoZWlnaHQuCi0gICAgICovCi0gICAgaW50IGdldEhlaWdodCgpIHsKLSAgICAgICAgLy8gdG8gYmUgb3ZlcnJpZGRlbgotICAgICAgICByZXR1cm4gMTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZWN1cnNpdmVseSBmaW5kIHRoZSBtZW51IGl0ZW0gZm9yIGEgbWVudSBzaG9ydGN1dC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ3IKLSAgICAgKiAgICAgICAgICAgIHRoZSBnci4KLSAgICAgKiBAcmV0dXJuIHRoZSBtZW51IGl0ZW07IG9yIG51bGwgaWYgdGhlIGl0ZW0gaXMgbm90IGF2YWlsYWJsZSBmb3IgdGhpcwotICAgICAqICAgICAgICAgc2hvcnRjdXQuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBNZW51SXRlbSBnZXRTaG9ydGN1dE1lbnVJdGVtSW1wbChNZW51U2hvcnRjdXQgbXMpIHsgaWYgKG1zID09IG51bGwpIHsKLSAgICAgKiByZXR1cm4gbnVsbDsgfSBmb3IgKGludCBpID0gMDsgaSA8IGdldEl0ZW1Db3VudCgpOyBpKyspIHsgTWVudUl0ZW0gbWkgPQotICAgICAqIGdldEl0ZW0oaSk7IGlmIChtaSBpbnN0YW5jZW9mIE1lbnUpIHsgbWkgPSAoKE1lbnUpCi0gICAgICogbWkpLmdldFNob3J0Y3V0TWVudUl0ZW1JbXBsKG1zKTsgaWYgKG1pICE9IG51bGwpIHsgcmV0dXJuIG1pOyB9IH0gZWxzZSBpZgotICAgICAqIChtcy5lcXVhbHMobWkuZ2V0U2hvcnRjdXQoKSkpIHsgcmV0dXJuIG1pOyB9IH0gcmV0dXJuIG51bGw7IH0KLSAgICAgKi8KLQotICAgIHZvaWQgcGFpbnQoR3JhcGhpY3MgZ3IpIHsKLSAgICAgICAgZ3Iuc2V0Q29sb3IoQ29sb3IuTElHSFRfR1JBWSk7Ci0gICAgICAgIGdyLmZpbGxSZWN0KDAsIDAsIGdldFdpZHRoKCksIGdldEhlaWdodCgpKTsKLSAgICAgICAgZ3Iuc2V0Q29sb3IoQ29sb3IuQkxBQ0spOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIE1vdXNlIGV2ZW50cyBoYW5kbGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBldmVudElkCi0gICAgICogICAgICAgICAgICBvbmUgb2YgdGhlIE1vdXNlRXZlbnQuTU9VU0VfKiBjb25zdGFudHMuCi0gICAgICogQHBhcmFtIHdoZXJlCi0gICAgICogICAgICAgICAgICBtb3VzZSBsb2NhdGlvbi4KLSAgICAgKiBAcGFyYW0gbW91c2VCdXR0b24KLSAgICAgKiAgICAgICAgICAgIG1vdXNlIGJ1dHRvbiB0aGF0IHdhcyBwcmVzc2VkIG9yIHJlbGVhc2VkLgotICAgICAqIEBwYXJhbSB3aGVuCi0gICAgICogICAgICAgICAgICBldmVudCB0aW1lLgotICAgICAqIEBwYXJhbSBtb2RpZmllcnMKLSAgICAgKiAgICAgICAgICAgIGlucHV0IGV2ZW50IG1vZGlmaWVycy4KLSAgICAgKi8KLSAgICB2b2lkIG9uTW91c2VFdmVudChpbnQgZXZlbnRJZCwgUG9pbnQgd2hlcmUsIGludCBtb3VzZUJ1dHRvbiwgbG9uZyB3aGVuLCBpbnQgbW9kaWZpZXJzKSB7Ci0gICAgICAgIC8vIHRvIGJlIG92ZXJyaWRkZW4KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBLZXlib2FyZCBldmVudCBoYW5kbGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBldmVudElkCi0gICAgICogICAgICAgICAgICBvbmUgb2YgdGhlIEtleUV2ZW50LktFWV8qIGNvbnN0YW50cy4KLSAgICAgKiBAcGFyYW0gdktleQotICAgICAqICAgICAgICAgICAgdGhlIGtleSBjb2RlLgotICAgICAqIEBwYXJhbSB3aGVuCi0gICAgICogICAgICAgICAgICBldmVudCB0aW1lLgotICAgICAqIEBwYXJhbSBtb2RpZmllcnMKLSAgICAgKiAgICAgICAgICAgIGlucHV0IGV2ZW50IG1vZGlmaWVycy4KLSAgICAgKi8KLSAgICB2b2lkIG9uS2V5RXZlbnQoaW50IGV2ZW50SWQsIGludCB2S2V5LCBsb25nIHdoZW4sIGludCBtb2RpZmllcnMpIHsKLSAgICAgICAgLy8gdG8gYmUgb3ZlcnJpZGRlbgotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBvc3QgdGhlIEFjdGlvbkV2ZW50IG9yIEl0ZW1FdmVudCwgZGVwZW5kaW5nIG9uIHR5cGUgb2YgdGhlIG1lbnUgaXRlbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSBpdGVtIHJlY3QuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiB2b2lkIGZpcmVJdGVtQWN0aW9uKGludCBpdGVtLCBsb25nIHdoZW4sIGludCBtb2RpZmllcnMpIHsgTWVudUl0ZW0gbWkgPQotICAgICAqIGdldEl0ZW0oaXRlbSk7IG1pLml0ZW1TZWxlY3RlZCh3aGVuLCBtb2RpZmllcnMpOyB9IE1lbnVJdGVtIGdldEl0ZW0oaW50Ci0gICAgICogaW5kZXgpIHsgLy8gdG8gYmUgb3ZlcnJpZGRlbiByZXR1cm4gbnVsbDsgfSBpbnQgZ2V0SXRlbUNvdW50KCkgeyByZXR1cm4KLSAgICAgKiAwOyB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBAcmV0dXJuIFRoZSBzdWItbWVudSBvZiBjdXJyZW50bHkgc2VsZWNldGQgaXRlbSwgb3IgbnVsbCBpZiBzdWNoIGEKLSAgICAgKiAgICAgICAgIHN1Yi1tZW51IGlzIG5vdCBhdmFpbGFibGUuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBNZW51IGdldFNlbGVjdGVkU3VibWVudSgpIHsgaWYgKHNlbGVjdGVkSXRlbUluZGV4IDwgMCkgeyByZXR1cm4gbnVsbDsgfQotICAgICAqIE1lbnVJdGVtIGl0ZW0gPSBnZXRJdGVtKHNlbGVjdGVkSXRlbUluZGV4KTsgcmV0dXJuIChpdGVtIGluc3RhbmNlb2YgTWVudSkKLSAgICAgKiA/IChNZW51KSBpdGVtIDogbnVsbDsgfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogQ29udmVuaWVuY2UgbWV0aG9kIGZvciBzZWxlY3RJdGVtKGluZGV4LCB0cnVlKS4KLSAgICAgKi8KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIHZvaWQgc2VsZWN0SXRlbShpbnQgaW5kZXgpIHsgc2VsZWN0SXRlbShpbmRleCwgdHJ1ZSk7IH0KLSAgICAgKi8KLQotICAgIC8qKgotICAgICAqIENoYW5nZSB0aGUgc2VsZWN0aW9uIGluIHRoZSBtZW51LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgbmV3IHNlbGVjZXRkIGl0ZW0ncyBpbmRleC4KLSAgICAgKiBAcGFyYW0gc2hvd1N1Yk1lbnUKLSAgICAgKiAgICAgICAgICAgIGlmIG5ldyBzZWxlY3RlZCBpdGVtIGhhcyBhIHN1Yi1tZW51LCBzaG91bGQgdGhhdCBzdWItbWVudSBiZQotICAgICAqICAgICAgICAgICAgZGlzcGxheWVkLgotICAgICAqLwotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogdm9pZCBzZWxlY3RJdGVtKGludCBpbmRleCwgYm9vbGVhbiBzaG93U3ViTWVudSkgeyBpZiAoc2VsZWN0ZWRJdGVtSW5kZXgKLSAgICAgKiA9PSBpbmRleCkgeyByZXR1cm47IH0gaWYgKHNlbGVjdGVkSXRlbUluZGV4ID49IDAgJiYKLSAgICAgKiBnZXRJdGVtKHNlbGVjdGVkSXRlbUluZGV4KSBpbnN0YW5jZW9mIE1lbnUpIHsgKChNZW51KQotICAgICAqIGdldEl0ZW0oc2VsZWN0ZWRJdGVtSW5kZXgpKS5oaWRlKCk7IH0gTXVsdGlSZWN0QXJlYSBjbGlwID0KLSAgICAgKiBnZXRVcGRhdGVDbGlwKGluZGV4LCBzZWxlY3RlZEl0ZW1JbmRleCk7IHNlbGVjdGVkSXRlbUluZGV4ID0gaW5kZXg7Ci0gICAgICogR3JhcGhpY3MgZ3IgPSBnZXRHcmFwaGljcyhjbGlwKTsgaWYgKGdyICE9IG51bGwpIHsgcGFpbnQoZ3IpOyB9IGlmCi0gICAgICogKHNob3dTdWJNZW51KSB7IHNob3dTdWJNZW51KHNlbGVjdGVkSXRlbUluZGV4KTsgfSB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBDaGFuZ2UgdGhlIHNlbGVjdGVkIGl0ZW0gdG8gdGhlIG5leHQgb25lIGluIHRoZSByZXF1ZXN0ZWQgZGlyZWN0aW9uCi0gICAgICogbW92aW5nIGN5Y2xpY2FsbHksIHNraXBwaW5nIHNlcGFyYXRvcnMKLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZm9yd2FyZAotICAgICAqICAgICAgICAgICAgdGhlIGRpcmVjdGlvbiB0byBtb3ZlIHRoZSBzZWxlY3Rpb24uCi0gICAgICogQHBhcmFtIHNob3dTdWJNZW51Ci0gICAgICogICAgICAgICAgICBpZiBuZXcgc2VsZWN0ZWQgaXRlbSBoYXMgYSBzdWItbWVudSwgc2hvdWxkIHRoYXQgc3ViLW1lbnUgYmUKLSAgICAgKiAgICAgICAgICAgIGRpc3BsYXllZC4KLSAgICAgKi8KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIHZvaWQgc2VsZWN0TmV4dEl0ZW0oYm9vbGVhbiBmb3J3YXJkLCBib29sZWFuIHNob3dTdWJNZW51KSB7IGludCBzZWxlY3RlZAotICAgICAqID0gZ2V0U2VsZWN0ZWRJdGVtSW5kZXgoKTsgaW50IGNvdW50ID0gZ2V0SXRlbUNvdW50KCk7IGlmIChjb3VudCA9PSAwKSB7Ci0gICAgICogcmV0dXJuOyB9IGlmIChzZWxlY3RlZCA8IDApIHsgc2VsZWN0ZWQgPSAoZm9yd2FyZCA/IGNvdW50IC0gMSA6IDApOyB9IGludAotICAgICAqIGkgPSBzZWxlY3RlZDsgZG8geyBpID0gKGZvcndhcmQgPyAoaSArIDEpIDogKGkgKyBjb3VudCAtIDEpKSAlIGNvdW50OyBpCi0gICAgICogJT0gY291bnQ7IE1lbnVJdGVtIGl0ZW0gPSBnZXRJdGVtKGkpOyBpZiAoISItIi5lcXVhbHMoaXRlbS5nZXRMYWJlbCgpKSkgewotICAgICAqIC8vJE5PTi1OTFMtMSQgc2VsZWN0SXRlbShpLCBzaG93U3ViTWVudSk7IHJldHVybjsgfSB9IHdoaWxlIChpICE9Ci0gICAgICogc2VsZWN0ZWQpOyB9IHZvaWQgc2hvd1N1Yk1lbnUoaW50IGluZGV4KSB7IGlmICgoaW5kZXggPCAwKSB8fAotICAgICAqICFpc0FjdGl2ZSgpKSB7IHJldHVybjsgfSBNZW51SXRlbSBpdGVtID0gZ2V0SXRlbShpbmRleCk7IGlmIChpdGVtCi0gICAgICogaW5zdGFuY2VvZiBNZW51KSB7IE1lbnUgbWVudSA9ICgoTWVudSkgZ2V0SXRlbShpbmRleCkpOyBpZgotICAgICAqIChtZW51LmdldEl0ZW1Db3VudCgpID09IDApIHsgcmV0dXJuOyB9IFBvaW50IGxvY2F0aW9uID0KLSAgICAgKiBnZXRTdWJtZW51TG9jYXRpb24oaW5kZXgpOyBtZW51LnNob3cobG9jYXRpb24ueCwgbG9jYXRpb24ueSwgZmFsc2UpOyB9IH0KLSAgICAgKi8KLQotICAgIC8qKgotICAgICAqIEByZXR1cm4gdGhlIG1lbnUgYmFyIHdoaWNoIGlzIHRoZSByb290IG9mIGN1cnJlbnQgbWVudSdzIGhpZXJhcmNoeTsgb3IKLSAgICAgKiAgICAgICAgIG51bGwgaWYgdGhlIGhpZXJhcmNoeSByb290IGlzIG5vdCBhIG1lbnUgYmFyLgotICAgICAqLwotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogTWVudUJhciBnZXRNZW51QmFyKCkgeyBpZiAocGFyZW50IGluc3RhbmNlb2YgTWVudUJhcikgeyByZXR1cm4gKE1lbnVCYXIpCi0gICAgICogcGFyZW50OyB9IGlmIChwYXJlbnQgaW5zdGFuY2VvZiBNZW51Q29tcG9uZW50KSB7IHJldHVybiAoKE1lbnVDb21wb25lbnQpCi0gICAgICogcGFyZW50KS5nZXRNZW51QmFyKCk7IH0gcmV0dXJuIG51bGw7IH0gUG9wdXBCb3ggZ2V0UG9wdXBCb3goKSB7IHJldHVybgotICAgICAqIG51bGw7IH0KLSAgICAgKi8KLQotICAgIFJlY3RhbmdsZSBnZXRJdGVtUmVjdChpbnQgaW5kZXgpIHsKLSAgICAgICAgLy8gdG8gYmUgb3ZlcnJpZGRlbgotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXRlcm1pbmUgdGhlIGNsaXAgcmVnaW9uIHdoZW4gbWVudSBzZWxlY3Rpb24gaXMgY2hhbmdlZCBmcm9tIGluZGV4MSB0bwotICAgICAqIGluZGV4Mi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5kZXgxCi0gICAgICogICAgICAgICAgICBvbGQgc2VsZWN0ZWQgaXRlbS4KLSAgICAgKiBAcGFyYW0gaW5kZXgyCi0gICAgICogICAgICAgICAgICBuZXcgc2VsZWN0ZWQgaXRlbS4KLSAgICAgKiBAcmV0dXJuIHRoZSByZWdpb24gdG8gcmVwYWludC4KLSAgICAgKi8KLSAgICBmaW5hbCBNdWx0aVJlY3RBcmVhIGdldFVwZGF0ZUNsaXAoaW50IGluZGV4MSwgaW50IGluZGV4MikgewotICAgICAgICBNdWx0aVJlY3RBcmVhIGNsaXAgPSBuZXcgTXVsdGlSZWN0QXJlYSgpOwotICAgICAgICBpZiAoaW5kZXgxID49IDApIHsKLSAgICAgICAgICAgIGNsaXAuYWRkKGdldEl0ZW1SZWN0KGluZGV4MSkpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChpbmRleDIgPj0gMCkgewotICAgICAgICAgICAgY2xpcC5hZGQoZ2V0SXRlbVJlY3QoaW5kZXgyKSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGNsaXA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc3VibWVudSBsb2NhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSBzdWJtZW51IGxvY2F0aW9uLgotICAgICAqLwotICAgIFBvaW50IGdldFN1Ym1lbnVMb2NhdGlvbihpbnQgaW5kZXgpIHsKLSAgICAgICAgLy8gdG8gYmUgb3ZlcnJpZGRlbgotICAgICAgICByZXR1cm4gbmV3IFBvaW50KDAsIDApOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNlbGVjdGVkIGl0ZW0gaW5kZXguCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc2VsZWN0ZWQgaXRlbSBpbmRleC4KLSAgICAgKi8KLSAgICBpbnQgZ2V0U2VsZWN0ZWRJdGVtSW5kZXgoKSB7Ci0gICAgICAgIHJldHVybiBzZWxlY3RlZEl0ZW1JbmRleDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBIaWRlLgotICAgICAqLwotICAgIHZvaWQgaGlkZSgpIHsKLSAgICAgICAgc2VsZWN0ZWRJdGVtSW5kZXggPSAtMTsKLSAgICAgICAgaWYgKHBhcmVudCBpbnN0YW5jZW9mIE1lbnVDb21wb25lbnQpIHsKLSAgICAgICAgICAgICgoTWVudUNvbXBvbmVudClwYXJlbnQpLml0ZW1IaWRkZW4odGhpcyk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJdGVtIGhpZGRlbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbWMKLSAgICAgKiAgICAgICAgICAgIHRoZSBtYy4KLSAgICAgKi8KLSAgICB2b2lkIGl0ZW1IaWRkZW4oTWVudUNvbXBvbmVudCBtYykgewotICAgICAgICAvLyB0byBiZSBvdmVycmlkZGVuCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIGlzIHZpc2libGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBpcyB2aXNpYmxlLgotICAgICAqLwotICAgIGJvb2xlYW4gaXNWaXNpYmxlKCkgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgaXMgYWN0aXZlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgaXMgYWN0aXZlLgotICAgICAqLwotICAgIGJvb2xlYW4gaXNBY3RpdmUoKSB7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEhpZGUgYWxsIG1lbnUgaGllcmFyY2h5LgotICAgICAqLwotICAgIHZvaWQgZW5kTWVudSgpIHsKLSAgICAgICAgLy8gPz8/QVdUOiB0b29sa2l0LmRpc3BhdGNoZXIucG9wdXBEaXNwYXRjaGVyLmRlYWN0aXZhdGVBbGwoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBIYW5kbGUgdGhlIG1vdXNlIGNsaWNrIG9yIEVudGVyIGtleSBldmVudCBvbiBhIG1lbnUncyBpdGVtLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aGVuCi0gICAgICogICAgICAgICAgICB0aGUgZXZlbnQgdGltZS4KLSAgICAgKiBAcGFyYW0gbW9kaWZpZXJzCi0gICAgICogICAgICAgICAgICBpbnB1dCBldmVudCBtb2RpZmllcnMuCi0gICAgICovCi0gICAgdm9pZCBpdGVtU2VsZWN0ZWQobG9uZyB3aGVuLCBpbnQgbW9kaWZpZXJzKSB7Ci0gICAgICAgIGVuZE1lbnUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBdXRvIG5hbWUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc3RyaW5nLgotICAgICAqLwotICAgIFN0cmluZyBhdXRvTmFtZSgpIHsKLSAgICAgICAgU3RyaW5nIG5hbWUgPSBnZXRDbGFzcygpLmdldE5hbWUoKTsKLSAgICAgICAgaWYgKG5hbWUuaW5kZXhPZigiJCIpICE9IC0xKSB7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgICAgIC8vID8/P0FXVDogaW50IG51bWJlciA9IHRvb2xraXQuYXV0b051bWJlci5uZXh0TWVudUNvbXBvbmVudCsrOwotICAgICAgICBpbnQgbnVtYmVyID0gMDsKLSAgICAgICAgbmFtZSA9IG5hbWUuc3Vic3RyaW5nKG5hbWUubGFzdEluZGV4T2YoIi4iKSArIDEpICsgSW50ZWdlci50b1N0cmluZyhudW1iZXIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIHJldHVybiBuYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIEdyYXBoaWNzIG9iamVjdCBmb3IgdGhlIHBvcC11cCBib3ggb2YgdGhpcyBtZW51IGNvbXBvbmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2xpcAotICAgICAqICAgICAgICAgICAgdGhlIGNsaXAgdG8gc2V0IG9uIHRoaXMgR3JhcGhpY3MuCi0gICAgICogQHJldHVybiB0aGUgY3JlYXRlZCBHcmFwaGljcyBvYmplY3QsIG9yIG51bGwgaWYgc3VjaCBvYmplY3QgaXMgbm90Ci0gICAgICogICAgICAgICBhdmFpbGFibGUuCi0gICAgICovCi0gICAgR3JhcGhpY3MgZ2V0R3JhcGhpY3MoTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0gICAgICAgIC8vIHRvIGJlIG92ZXJyaWRkZW4KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQHJldHVybiBhY2Nlc3NpYmxlIGNvbnRleHQgc3BlY2lmaWMgZm9yIHBhcnRpY3VsYXIgbWVudSBjb21wb25lbnQuCi0gICAgICovCi0gICAgLy8gPz8/QVdUCi0gICAgLyoKLSAgICAgKiBBY2Nlc3NpYmxlQ29udGV4dCBjcmVhdGVBY2Nlc3NpYmxlQ29udGV4dCgpIHsgcmV0dXJuIG51bGw7IH0KLSAgICAgKi8KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9NZW51Q29udGFpbmVyLmphdmEgYi9hd3QvamF2YS9hd3QvTWVudUNvbnRhaW5lci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlNTA5YTFiLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9NZW51Q29udGFpbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLS8qKgotICogVGhlIE1lbnVDb250YWluZXIgaW50ZXJmYWNlIHJlcHJlc2VudHMgYWxsIG1lbnUgY29udGFpbmVycy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgTWVudUNvbnRhaW5lciB7Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHRoZSBzcGVjaWZpZWQgTWVudUNvbXBvbmVudCBmcm9tIHRoZSBNZW51Q29udGFpbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgTWVudUNvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmUoTWVudUNvbXBvbmVudCBjKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEZvbnQgb2YgdGhlIE1lbnVDb250YWluZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZm9udCBvZiB0aGUgTWVudUNvbnRhaW5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgRm9udCBnZXRGb250KCk7Ci0KLSAgICAvKioKLSAgICAgKiBQb3N0cyBhbiBFdmVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZQotICAgICAqICAgICAgICAgICAgdGhlIEV2ZW50LgotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgZXZlbnQgaXMgcG9zdGVkIHN1Y2Nlc3NmdWxseSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEBkZXByZWNhdGVkIFJlcGxhY2VkIGJ5IGRpc3BhdGNoRXZlbnQgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIGJvb2xlYW4gcG9zdEV2ZW50KEV2ZW50IGUpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvTW9kYWxDb250ZXh0LmphdmEgYi9hd3QvamF2YS9hd3QvTW9kYWxDb250ZXh0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDMyYTU5MTIuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L01vZGFsQ29udGV4dC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNjQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dDsKLQotLyoqCi0gKgotICogVGhlIGNvbnRleHQgZm9yIG5lc3RlZCBldmVudCBsb29wLiBJdCBjYW4gYmUgZGlhbG9nLCBwb3B1cCBtZW51IGV0Yy4KLSAqLwotY2xhc3MgTW9kYWxDb250ZXh0IHsKLQotICAgIHByaXZhdGUgYm9vbGVhbiBydW5uaW5nID0gZmFsc2U7Ci0KLSAgICBwcml2YXRlIGZpbmFsIFRvb2xraXQgdG9vbGtpdDsKLQotICAgIE1vZGFsQ29udGV4dCgpIHsKLSAgICAgICAgdG9vbGtpdCA9IFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXQgdXAgYW5kIHJ1biBtb2RhbCBsb29wIGluIHRoaXMgY29udGV4dAotICAgICAqCi0gICAgICovCi0gICAgdm9pZCBydW5Nb2RhbExvb3AoKSB7Ci0gICAgICAgIHJ1bm5pbmcgPSB0cnVlOwotICAgICAgICB0b29sa2l0LmRpc3BhdGNoVGhyZWFkLnJ1bk1vZGFsTG9vcCh0aGlzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBMZWF2ZSB0aGUgbW9kYWwgbG9vcCBydW5uaW5nIGluIHRoaXMgY29udGV4dAotICAgICAqIFRoaXMgbWV0aG9kIGRvZXNuJ3Qgc3RvcHMgdGhlIGxvb3AgaW1tZWRpYXRlbHksCi0gICAgICogaXQganVzdCBzZXRzIHRoZSBmbGFnIHRoYXQgc2F5cyB0aGUgbW9kYWwgbG9vcCB0byBzdG9wCi0gICAgICoKLSAgICAgKi8KLSAgICB2b2lkIGVuZE1vZGFsTG9vcCgpIHsKLSAgICAgICAgcnVubmluZyA9IGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqCi0gICAgICogQHJldHVybiBtb2RhbCBsb29wIGlzIGN1cnJlbnRseSBydW5uaW5nIGluIHRoaXMgY29udGV4dAotICAgICAqLwotICAgIGJvb2xlYW4gaXNNb2RhbExvb3BSdW5uaW5nKCkgewotICAgICAgICByZXR1cm4gcnVubmluZzsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9Nb3VzZURpc3BhdGNoZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9Nb3VzZURpc3BhdGNoZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGY0OGY5ZC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvTW91c2VEaXNwYXRjaGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0MTggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEbWl0cnkgQS4gRHVybmV2LCBNaWNoYWVsIERhbmlsb3YsIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuZXZlbnQuTW91c2VFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5Nb3VzZUxpc3RlbmVyOwotaW1wb3J0IGphdmEuYXd0LmV2ZW50Lk1vdXNlTW90aW9uTGlzdGVuZXI7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuTW91c2VXaGVlbEV2ZW50OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50Lk1vdXNlV2hlZWxMaXN0ZW5lcjsKLWltcG9ydCBqYXZhLmF3dC5EaXNwYXRjaGVyLk1vdXNlR3JhYk1hbmFnZXI7Ci1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5OYXRpdmVFdmVudDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5OYXRpdmVXaW5kb3c7Ci0KLQotY2xhc3MgTW91c2VEaXNwYXRjaGVyIHsKLQotICAgIC8vIEZpZWxkcyBmb3Igc3ludGhldGljIG1vdXNlIGNsaWNrIGV2ZW50cyBnZW5lcmF0aW9uCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IGNsaWNrRGVsdGEgPSA1OwotICAgIHByaXZhdGUgZmluYWwgbG9uZ1tdIGxhc3RQcmVzc1RpbWUgPSBuZXcgbG9uZ1tdIHswbCwgMGwsIDBsfTsKLSAgICBwcml2YXRlIGZpbmFsIFBvaW50W10gbGFzdFByZXNzUG9zID0gbmV3IFBvaW50W10ge251bGwsIG51bGwsIG51bGx9OwotICAgIHByaXZhdGUgZmluYWwgYm9vbGVhbltdIGJ1dHRvblByZXNzZWQgPSBuZXcgYm9vbGVhbltdIHtmYWxzZSwgZmFsc2UsIGZhbHNlfTsKLSAgICBwcml2YXRlIGZpbmFsIGludFtdIGNsaWNrQ291bnQgPSBuZXcgaW50W10gezAsIDAsIDB9OwotCi0gICAgLy8gRmllbGRzIGZvciBtb3VzZSBlbnRlcmVkL2V4aXRlZCBzdXBwb3J0Ci0gICAgcHJpdmF0ZSBDb21wb25lbnQgbGFzdFVuZGVyUG9pbnRlciA9IG51bGw7Ci0gICAgcHJpdmF0ZSBmaW5hbCBQb2ludCBsYXN0U2NyZWVuUG9zID0gbmV3IFBvaW50KC0xLCAtMSk7Ci0KLSAgICAvLyBGaWVsZHMgZm9yIHJlZHVuZGFudCBtb3VzZSBtb3ZlZC9kcmFnZ2VkIGZpbHRlcmluZwotICAgIHByaXZhdGUgQ29tcG9uZW50IGxhc3RVbmRlck1vdGlvbiA9IG51bGw7Ci0gICAgcHJpdmF0ZSBQb2ludCBsYXN0TG9jYWxQb3MgPSBuZXcgUG9pbnQoLTEsIC0xKTsKLQotICAgIHByaXZhdGUgZmluYWwgTW91c2VHcmFiTWFuYWdlciBtb3VzZUdyYWJNYW5hZ2VyOwotICAgIHByaXZhdGUgZmluYWwgVG9vbGtpdCB0b29sa2l0OwotCi0gICAgc3RhdGljIFBvaW50IGNvbnZlcnRQb2ludChDb21wb25lbnQgc3JjLCBpbnQgeCwgaW50IHksIENvbXBvbmVudCBkZXN0KSB7Ci0gICAgICAgIFBvaW50IHNyY1BvaW50ID0gZ2V0QWJzTG9jYXRpb24oc3JjKTsKLSAgICAgICAgUG9pbnQgZGVzdFBvaW50ID0gZ2V0QWJzTG9jYXRpb24oZGVzdCk7Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBQb2ludCh4ICsgKHNyY1BvaW50LnggLSBkZXN0UG9pbnQueCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgeSArIChzcmNQb2ludC55IC0gZGVzdFBvaW50LnkpKTsKLSAgICB9Ci0KLSAgICBzdGF0aWMgUG9pbnQgY29udmVydFBvaW50KENvbXBvbmVudCBzcmMsIFBvaW50IHAsIENvbXBvbmVudCBkc3QpIHsKLSAgICAgICAgcmV0dXJuIGNvbnZlcnRQb2ludChzcmMsIHAueCwgcC55LCBkc3QpOwotICAgIH0KLQotICAgIHByaXZhdGUgc3RhdGljIFBvaW50IGdldEFic0xvY2F0aW9uKENvbXBvbmVudCBjb21wKSB7Ci0gICAgICAgIFBvaW50IGxvY2F0aW9uID0gbmV3IFBvaW50KDAsIDApOwotLy8gQkVHSU4gYW5kcm9pZC1jaGFuZ2VkOiBBV1QgY29tcG9uZW50cyBub3Qgc3VwcG9ydGVkCi0vLyAgICAgICAgZm9yIChDb21wb25lbnQgcGFyZW50ID0gY29tcDsgcGFyZW50ICE9IG51bGw7IHBhcmVudCA9IHBhcmVudC5wYXJlbnQpIHsKLS8vICAgICAgICAgICAgUG9pbnQgcGFyZW50UG9zID0gKHBhcmVudCBpbnN0YW5jZW9mIEVtYmVkZGVkV2luZG93ID8KLS8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBhcmVudC5nZXROYXRpdmVXaW5kb3coKS5nZXRTY3JlZW5Qb3MoKSA6Ci0vLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwYXJlbnQuZ2V0TG9jYXRpb24oKSk7Ci0vLwotLy8gICAgICAgICAgICBsb2NhdGlvbi50cmFuc2xhdGUocGFyZW50UG9zLngsIHBhcmVudFBvcy55KTsKLS8vCi0vLyAgICAgICAgICAgIGlmIChwYXJlbnQgaW5zdGFuY2VvZiBXaW5kb3cpIHsKLS8vICAgICAgICAgICAgICAgIGJyZWFrOwotLy8gICAgICAgICAgICB9Ci0vLyAgICAgICAgfQotLy8gRU5EIGFuZHJvaWQtY2hhbmdlZAotCi0gICAgICAgIHJldHVybiBsb2NhdGlvbjsKLSAgICB9Ci0KLSAgICBNb3VzZURpc3BhdGNoZXIoTW91c2VHcmFiTWFuYWdlciBtb3VzZUdyYWJNYW5hZ2VyLAotICAgICAgICAgICAgICAgICAgICBUb29sa2l0IHRvb2xraXQpIHsKLSAgICAgICAgdGhpcy5tb3VzZUdyYWJNYW5hZ2VyID0gbW91c2VHcmFiTWFuYWdlcjsKLSAgICAgICAgdGhpcy50b29sa2l0ID0gdG9vbGtpdDsKLSAgICB9Ci0KLSAgICBQb2ludCBnZXRQb2ludGVyUG9zKCkgewotICAgICAgICByZXR1cm4gbGFzdFNjcmVlblBvczsKLSAgICB9Ci0KLSAgICBib29sZWFuIGRpc3BhdGNoKENvbXBvbmVudCBzcmMsIE5hdGl2ZUV2ZW50IGV2ZW50KSB7Ci0gICAgICAgIGludCBpZCA9IGV2ZW50LmdldEV2ZW50SWQoKTsKLQotICAgICAgICBsYXN0U2NyZWVuUG9zLnNldExvY2F0aW9uKGV2ZW50LmdldFNjcmVlblBvcygpKTsKLSAgICAgICAgY2hlY2tNb3VzZUVudGVyRXhpdChldmVudC5nZXRJbnB1dE1vZGlmaWVycygpLCBldmVudC5nZXRUaW1lKCkpOwotCi0gICAgICAgIGlmIChpZCA9PSBNb3VzZUV2ZW50Lk1PVVNFX1dIRUVMKSB7Ci0vLyBCRUdJTiBhbmRyb2lkLWNoYW5nZWQ6IEFXVCBjb21wb25lbnRzIG5vdCBzdXBwb3J0ZWQKLS8vICAgICAgICAgICAgZGlzcGF0Y2hXaGVlbEV2ZW50KHNyYywgZXZlbnQpOwotLy8gRU5EIGFuZHJvaWQtY2hhbmdlZAotICAgICAgICB9IGVsc2UgaWYgKChpZCAhPSBNb3VzZUV2ZW50Lk1PVVNFX0VOVEVSRUQpICYmCi0gICAgICAgICAgICAgICAgICAgKGlkICE9IE1vdXNlRXZlbnQuTU9VU0VfRVhJVEVEKSkgewotICAgICAgICAgICAgUG9pbnRlckluZm8gaW5mbyA9IG5ldyBQb2ludGVySW5mbyhzcmMsIGV2ZW50LmdldExvY2FsUG9zKCkpOwotCi0gICAgICAgICAgICBtb3VzZUdyYWJNYW5hZ2VyLnByZXByb2Nlc3NFdmVudChldmVudCk7Ci0gICAgICAgICAgICBmaW5kRXZlbnRTb3VyY2UoaW5mbyk7Ci0gICAgICAgICAgICBpZiAoKGlkID09IE1vdXNlRXZlbnQuTU9VU0VfUFJFU1NFRCkgfHwKLSAgICAgICAgICAgICAgICAoaWQgPT0gTW91c2VFdmVudC5NT1VTRV9SRUxFQVNFRCkpIHsKLQotICAgICAgICAgICAgICAgIGRpc3BhdGNoQnV0dG9uRXZlbnQoaW5mbywgZXZlbnQpOwotICAgICAgICAgICAgfSBlbHNlIGlmICgoaWQgPT0gTW91c2VFdmVudC5NT1VTRV9NT1ZFRCkgfHwKLSAgICAgICAgICAgICAgICAgICAgICAgKGlkID09IE1vdXNlRXZlbnQuTU9VU0VfRFJBR0dFRCkpIHsKLQotICAgICAgICAgICAgICAgIGRpc3BhdGNoTW90aW9uRXZlbnQoaW5mbywgZXZlbnQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIHByaXZhdGUgdm9pZCBjaGVja01vdXNlRW50ZXJFeGl0KGludCBtb2RpZmllcnMsIGxvbmcgd2hlbikgewotLy8gQkVHSU4gYW5kcm9pZC1jaGFuZ2VkOiBBV1QgY29tcG9uZW50cyBub3Qgc3VwcG9ydGVkCi0vLyAgICAgICAgUG9pbnRlckluZm8gaW5mbyA9IGZpbmRDb21wb25lbnRVbmRlclBvaW50ZXIoKTsKLS8vICAgICAgICBDb21wb25lbnQgY3VyVW5kZXJQb2ludGVyID0KLS8vICAgICAgICAgICAgICAgIHByb3BhZ2F0ZUV2ZW50KGluZm8sIEFXVEV2ZW50Lk1PVVNFX0VWRU5UX01BU0ssCi0vLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBNb3VzZUxpc3RlbmVyLmNsYXNzLCBmYWxzZSkuc3JjOwotLy8KLS8vICAgICAgICBpZiAoY3VyVW5kZXJQb2ludGVyICE9IGxhc3RVbmRlclBvaW50ZXIpIHsKLS8vICAgICAgICAgICAgUG9pbnQgcG9zID0gaW5mby5wb3NpdGlvbjsKLS8vICAgICAgICAgICAgaWYgKChsYXN0VW5kZXJQb2ludGVyICE9IG51bGwpICYmCi0vLyAgICAgICAgICAgICAgICAgbGFzdFVuZGVyUG9pbnRlci5pc01vdXNlRXhpdGVkRXhwZWN0ZWQoKSkgewotLy8KLS8vICAgICAgICAgICAgICAgIFBvaW50IGV4aXRQb3MgPSBjb252ZXJ0UG9pbnQobnVsbCwgbGFzdFNjcmVlblBvcy54LAotLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsYXN0U2NyZWVuUG9zLnksIGxhc3RVbmRlclBvaW50ZXIpOwotLy8KLS8vICAgICAgICAgICAgICAgIHBvc3RNb3VzZUVudGVyRXhpdChNb3VzZUV2ZW50Lk1PVVNFX0VYSVRFRCwgbW9kaWZpZXJzLCB3aGVuLAotLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV4aXRQb3MueCwgZXhpdFBvcy55LCBsYXN0VW5kZXJQb2ludGVyKTsKLS8vICAgICAgICAgICAgfQotLy8gICAgICAgICAgICBzZXRDdXJzb3IoY3VyVW5kZXJQb2ludGVyKTsKLS8vICAgICAgICAgICAgaWYgKGN1clVuZGVyUG9pbnRlciAhPSBudWxsKSB7Ci0vLyAgICAgICAgICAgICAgICBwb3N0TW91c2VFbnRlckV4aXQoTW91c2VFdmVudC5NT1VTRV9FTlRFUkVELCBtb2RpZmllcnMsIHdoZW4sCi0vLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcG9zLngsIHBvcy55LCBjdXJVbmRlclBvaW50ZXIpOwotLy8gICAgICAgICAgICB9Ci0vLyAgICAgICAgICAgIGxhc3RVbmRlclBvaW50ZXIgPSBjdXJVbmRlclBvaW50ZXI7Ci0vLyAgICAgICAgfQotLy8gRU5EIGFuZHJvaWQtY2hhbmdlZAotICAgIH0KLQotICAgIHByaXZhdGUgdm9pZCBzZXRDdXJzb3IoQ29tcG9uZW50IGNvbXApIHsKLSAgICAgICAgaWYgKGNvbXAgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIENvbXBvbmVudCBncmFiT3duZXIgPSBtb3VzZUdyYWJNYW5hZ2VyLmdldFN5bnRoZXRpY0dyYWJPd25lcigpOwotICAgICAgICBDb21wb25lbnQgY3Vyc29yQ29tcCA9ICgoZ3JhYk93bmVyICE9IG51bGwpICYmCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBncmFiT3duZXIuaXNTaG93aW5nKCkgPyBncmFiT3duZXIgOiBjb21wKTsKLSAgICAgICAgY3Vyc29yQ29tcC5zZXRDdXJzb3IoKTsKLSAgICB9Ci0KLSAgICBwcml2YXRlIHZvaWQgcG9zdE1vdXNlRW50ZXJFeGl0KGludCBpZCwgaW50IG1vZCwgbG9uZyB3aGVuLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IHgsIGludCB5LCBDb21wb25lbnQgY29tcCkgewotICAgICAgICBpZiAoY29tcC5pc0luZGlyZWN0bHlFbmFibGVkKCkpIHsKLSAgICAgICAgICAgIHRvb2xraXQuZ2V0U3lzdGVtRXZlbnRRdWV1ZUltcGwoKS5wb3N0RXZlbnQoCi0gICAgICAgICAgICAgICAgICAgIG5ldyBNb3VzZUV2ZW50KGNvbXAsIGlkLCB3aGVuLCBtb2QsIHgsIHksIDAsIGZhbHNlKSk7Ci0gICAgICAgICAgICBjb21wLnNldE1vdXNlRXhpdGVkRXhwZWN0ZWQoaWQgPT0gTW91c2VFdmVudC5NT1VTRV9FTlRFUkVEKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbXAuc2V0TW91c2VFeGl0ZWRFeHBlY3RlZChmYWxzZSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAvLyBCRUdJTiBhbmRyb2lkLWNoYW5nZWQ6IEFXVCBjb21wb25lbnRzIG5vdCBzdXBwb3J0ZWQKLS8vICAgIHByaXZhdGUgUG9pbnRlckluZm8gZmluZENvbXBvbmVudFVuZGVyUG9pbnRlcigpIHsKLS8vICAgICAgICBOYXRpdmVXaW5kb3cgbmF0aXZlV2luZG93ID0gdG9vbGtpdC5nZXRXaW5kb3dGYWN0b3J5KCkuCi0vLyAgICAgICAgZ2V0V2luZG93RnJvbVBvaW50KGxhc3RTY3JlZW5Qb3MpOwotLy8KLS8vICAgICAgICBpZiAobmF0aXZlV2luZG93ICE9IG51bGwpIHsKLS8vICAgICAgICAgICAgQ29tcG9uZW50IGNvbXAgPSB0b29sa2l0LmdldENvbXBvbmVudEJ5SWQobmF0aXZlV2luZG93LmdldElkKCkpOwotLy8KLS8vICAgICAgICAgICAgaWYgKGNvbXAgIT0gbnVsbCkgewotLy8gICAgICAgICAgICAgICAgV2luZG93IHdpbmRvdyA9IGNvbXAuZ2V0V2luZG93QW5jZXN0b3IoKTsKLS8vICAgICAgICAgICAgICAgIFBvaW50IHBvaW50ZXJQb3MgPSBjb252ZXJ0UG9pbnQobnVsbCwgbGFzdFNjcmVlblBvcy54LAotLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsYXN0U2NyZWVuUG9zLnksIHdpbmRvdyk7Ci0vLwotLy8gICAgICAgICAgICAgICAgaWYgKHdpbmRvdy5nZXRDbGllbnQoKS5jb250YWlucyhwb2ludGVyUG9zKSkgewotLy8gICAgICAgICAgICAgICAgICAgIFBvaW50ZXJJbmZvIGluZm8gPSBuZXcgUG9pbnRlckluZm8od2luZG93LCBwb2ludGVyUG9zKTsKLS8vCi0vLyAgICAgICAgICAgICAgICAgICAgZmFsbDJDaGlsZChpbmZvKTsKLS8vCi0vLyAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGluZm87Ci0vLyAgICAgICAgICAgICAgICB9Ci0vLyAgICAgICAgICAgIH0KLS8vICAgICAgICB9Ci0vLwotLy8gICAgICAgIHJldHVybiBuZXcgUG9pbnRlckluZm8obnVsbCwgbnVsbCk7Ci0vLyAgICB9Ci0vLyBFTkQgYW5kcm9pZC1jaGFuZ2VkCi0gICAgCi0gICAgcHJpdmF0ZSB2b2lkIGZpbmRFdmVudFNvdXJjZShQb2ludGVySW5mbyBpbmZvKSB7Ci0gICAgICAgIENvbXBvbmVudCBncmFiT3duZXIgPSBtb3VzZUdyYWJNYW5hZ2VyLmdldFN5bnRoZXRpY0dyYWJPd25lcigpOwotCi0gICAgICAgIGlmIChncmFiT3duZXIgIT0gbnVsbCAmJiBncmFiT3duZXIuaXNTaG93aW5nKCkpIHsKLSAgICAgICAgICAgIGluZm8ucG9zaXRpb24gPSBjb252ZXJ0UG9pbnQoaW5mby5zcmMsIGluZm8ucG9zaXRpb24sIGdyYWJPd25lcik7Ci0gICAgICAgICAgICBpbmZvLnNyYyA9IGdyYWJPd25lcjsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vPz8/QVdUOiByaXNlMlRvcExldmVsKGluZm8pOwotICAgICAgICAgICAgLy8/Pz9BV1Q6IGZhbGwyQ2hpbGQoaW5mbyk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAvLyBCRUdJTiBhbmRyb2lkLWNoYW5nZWQ6IEFXVCBjb21wb25lbnRzIG5vdCBzdXBwb3J0ZWQKLS8vICAgIHByaXZhdGUgdm9pZCByaXNlMlRvcExldmVsKFBvaW50ZXJJbmZvIGluZm8pIHsKLS8vICAgICAgICB3aGlsZSAoIShpbmZvLnNyYyBpbnN0YW5jZW9mIFdpbmRvdykpIHsKLS8vICAgICAgICAgICAgaW5mby5wb3NpdGlvbi50cmFuc2xhdGUoaW5mby5zcmMueCwgaW5mby5zcmMueSk7Ci0vLyAgICAgICAgICAgIGluZm8uc3JjID0gaW5mby5zcmMucGFyZW50OwotLy8gICAgICAgIH0KLS8vICAgIH0KLS8vCi0vLyAgICBwcml2YXRlIHZvaWQgZmFsbDJDaGlsZChQb2ludGVySW5mbyBpbmZvKSB7Ci0vLyAgICAgICAgSW5zZXRzIGluc2V0cyA9IGluZm8uc3JjLmdldEluc2V0cygpOwotLy8KLS8vICAgICAgICBmaW5hbCBQb2ludCBwb3MgPSBpbmZvLnBvc2l0aW9uOwotLy8gICAgICAgIGZpbmFsIGludCB4ID0gcG9zLng7Ci0vLyAgICAgICAgZmluYWwgaW50IHkgPSBwb3MueTsKLS8vICAgICAgICBpZiAoKHggPj0gaW5zZXRzLmxlZnQpICYmICh5ID49IGluc2V0cy50b3ApICYmCi0vLyAgICAgICAgICAgICAgICAoeCA8IChpbmZvLnNyYy53IC0gaW5zZXRzLnJpZ2h0KSkgJiYKLS8vICAgICAgICAgICAgICAgICh5IDwgKGluZm8uc3JjLmggLSBpbnNldHMuYm90dG9tKSkpCi0vLyAgICAgICAgewotLy8gICAgICAgICAgICBDb21wb25lbnRbXSBjaGlsZHJlbiA9ICgoQ29udGFpbmVyKSBpbmZvLnNyYykuZ2V0Q29tcG9uZW50cygpOwotLy8KLS8vICAgICAgICAgICAgZm9yIChDb21wb25lbnQgY2hpbGQgOiBjaGlsZHJlbikgewotLy8gICAgICAgICAgICAgICAgaWYgKGNoaWxkLmlzU2hvd2luZygpKSB7Ci0vLyAgICAgICAgICAgICAgICAgICAgaWYgKGNoaWxkLmNvbnRhaW5zKHggLSBjaGlsZC5nZXRYKCksCi0vLyAgICAgICAgICAgICAgICAgICAgICAgICAgICB5IC0gY2hpbGQuZ2V0WSgpKSkKLS8vICAgICAgICAgICAgICAgICAgICB7Ci0vLyAgICAgICAgICAgICAgICAgICAgICAgIGluZm8uc3JjID0gY2hpbGQ7Ci0vLyAgICAgICAgICAgICAgICAgICAgICAgIHBvcy50cmFuc2xhdGUoLWNoaWxkLngsIC1jaGlsZC55KTsKLS8vCi0vLyAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjaGlsZCBpbnN0YW5jZW9mIENvbnRhaW5lcikgewotLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgZmFsbDJDaGlsZChpbmZvKTsKLS8vICAgICAgICAgICAgICAgICAgICAgICAgfQotLy8KLS8vICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuOwotLy8gICAgICAgICAgICAgICAgICAgIH0KLS8vICAgICAgICAgICAgICAgIH0KLS8vICAgICAgICAgICAgfQotLy8gICAgICAgIH0KLS8vICAgIH0KLS8vIEVORCBhbmRyb2lkLWNoYW5nZWQKLQotICAgIHByaXZhdGUgdm9pZCBkaXNwYXRjaEJ1dHRvbkV2ZW50KFBvaW50ZXJJbmZvIGluZm8sIE5hdGl2ZUV2ZW50IGV2ZW50KSB7Ci0gICAgICAgIGludCBidXR0b24gPSBldmVudC5nZXRNb3VzZUJ1dHRvbigpOwotICAgICAgICBsb25nIHRpbWUgPSBldmVudC5nZXRUaW1lKCk7Ci0gICAgICAgIGludCBpZCA9IGV2ZW50LmdldEV2ZW50SWQoKTsKLSAgICAgICAgaW50IGluZGV4ID0gYnV0dG9uIC0gMTsKLSAgICAgICAgYm9vbGVhbiBjbGlja1JlcXVpcmVkID0gZmFsc2U7Ci0KLSAgICAgICAgcHJvcGFnYXRlRXZlbnQoaW5mbywgQVdURXZlbnQuTU9VU0VfRVZFTlRfTUFTSywKLSAgICAgICAgICAgICAgICAgICAgICAgTW91c2VMaXN0ZW5lci5jbGFzcywgZmFsc2UpOwotICAgICAgICBpZiAoaWQgPT0gTW91c2VFdmVudC5NT1VTRV9QUkVTU0VEKSB7Ci0gICAgICAgICAgICBpbnQgY2xpY2tJbnRlcnZhbCA9IHRvb2xraXQuZGlzcGF0Y2hlci5jbGlja0ludGVydmFsOwotICAgICAgICAgICAgbW91c2VHcmFiTWFuYWdlci5vbk1vdXNlUHJlc3NlZChpbmZvLnNyYyk7Ci0gICAgICAgICAgICBidXR0b25QcmVzc2VkW2luZGV4XSA9IHRydWU7Ci0gICAgICAgICAgICBjbGlja0NvdW50W2luZGV4XSA9ICghZGVsdGFFeGNlZWRlZChpbmRleCwgaW5mbykgJiYKLSAgICAgICAgICAgICAgICAgICAgKCh0aW1lIC0gbGFzdFByZXNzVGltZVtpbmRleF0pIDw9IGNsaWNrSW50ZXJ2YWwpKSA/Ci0gICAgICAgICAgICAgICAgICAgIGNsaWNrQ291bnRbaW5kZXhdICsgMSA6IDE7Ci0gICAgICAgICAgICBsYXN0UHJlc3NUaW1lW2luZGV4XSA9IHRpbWU7Ci0gICAgICAgICAgICBsYXN0UHJlc3NQb3NbaW5kZXhdID0gaW5mby5wb3NpdGlvbjsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG1vdXNlR3JhYk1hbmFnZXIub25Nb3VzZVJlbGVhc2VkKGluZm8uc3JjKTsKLSAgICAgICAgICAgIC8vIHNldCBjdXJzb3IgYmFjayBvbiBzeW50aGV0aWMgbW91c2UgZ3JhYiBlbmQ6Ci0vLyBCRUdJTiBhbmRyb2lkLWNoYW5nZWQ6IEFXVCBjb21wb25lbnRzIG5vdCBzdXBwb3J0ZWQKLS8vICAgICAgICAgICAgc2V0Q3Vyc29yKGZpbmRDb21wb25lbnRVbmRlclBvaW50ZXIoKS5zcmMpOwotLy8gRU5EIGFuZHJvaWQtY2hhbmdlZAotICAgICAgICAgICAgaWYgKGJ1dHRvblByZXNzZWRbaW5kZXhdKSB7Ci0gICAgICAgICAgICAgICAgYnV0dG9uUHJlc3NlZFtpbmRleF0gPSBmYWxzZTsKLSAgICAgICAgICAgICAgICBjbGlja1JlcXVpcmVkID0gIWRlbHRhRXhjZWVkZWQoaW5kZXgsIGluZm8pOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBjbGlja0NvdW50W2luZGV4XSA9IDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGluZm8uc3JjLmlzSW5kaXJlY3RseUVuYWJsZWQoKSkgewotICAgICAgICAgICAgZmluYWwgUG9pbnQgcG9zID0gaW5mby5wb3NpdGlvbjsKLSAgICAgICAgICAgIGZpbmFsIGludCBtb2QgPSBldmVudC5nZXRJbnB1dE1vZGlmaWVycygpOwotICAgICAgICAgICAgdG9vbGtpdC5nZXRTeXN0ZW1FdmVudFF1ZXVlSW1wbCgpLnBvc3RFdmVudCgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXcgTW91c2VFdmVudChpbmZvLnNyYywgaWQsIHRpbWUsIG1vZCwgcG9zLngsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcG9zLnksIGNsaWNrQ291bnRbaW5kZXhdLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV2ZW50LmdldFRyaWdnZXIoKSwgYnV0dG9uKSk7Ci0gICAgICAgICAgICBpZiAoY2xpY2tSZXF1aXJlZCkgewotICAgICAgICAgICAgICAgIHRvb2xraXQuZ2V0U3lzdGVtRXZlbnRRdWV1ZUltcGwoKS5wb3N0RXZlbnQoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgbmV3IE1vdXNlRXZlbnQoaW5mby5zcmMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgTW91c2VFdmVudC5NT1VTRV9DTElDS0VELAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRpbWUsIG1vZCwgcG9zLngsIHBvcy55LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNsaWNrQ291bnRbaW5kZXhdLCBmYWxzZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBidXR0b24pKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHByaXZhdGUgYm9vbGVhbiBkZWx0YUV4Y2VlZGVkKGludCBpbmRleCwgUG9pbnRlckluZm8gaW5mbykgewotICAgICAgICBmaW5hbCBQb2ludCBsYXN0UG9zID0gbGFzdFByZXNzUG9zW2luZGV4XTsKLSAgICAgICAgaWYgKGxhc3RQb3MgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuICgoTWF0aC5hYnMobGFzdFBvcy54IC0gaW5mby5wb3NpdGlvbi54KSA+IGNsaWNrRGVsdGEpIHx8Ci0gICAgICAgICAgICAgICAgKE1hdGguYWJzKGxhc3RQb3MueSAtIGluZm8ucG9zaXRpb24ueSkgPiBjbGlja0RlbHRhKSk7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIGRpc3BhdGNoTW90aW9uRXZlbnQoUG9pbnRlckluZm8gaW5mbywgTmF0aXZlRXZlbnQgZXZlbnQpIHsKLSAgICAgICAgcHJvcGFnYXRlRXZlbnQoaW5mbywgQVdURXZlbnQuTU9VU0VfTU9USU9OX0VWRU5UX01BU0ssCi0gICAgICAgICAgICAgICAgICAgICAgIE1vdXNlTW90aW9uTGlzdGVuZXIuY2xhc3MsIGZhbHNlKTsKLSAgICAgICAgZmluYWwgUG9pbnQgcG9zID0gaW5mby5wb3NpdGlvbjsKLSAgICAgICAgaWYgKChsYXN0VW5kZXJNb3Rpb24gIT0gaW5mby5zcmMpIHx8Ci0gICAgICAgICAgICAhbGFzdExvY2FsUG9zLmVxdWFscyhwb3MpKSB7Ci0KLSAgICAgICAgICAgIGxhc3RVbmRlck1vdGlvbiA9IGluZm8uc3JjOwotICAgICAgICAgICAgbGFzdExvY2FsUG9zID0gcG9zOwotCi0gICAgICAgICAgICBpZiAoaW5mby5zcmMuaXNJbmRpcmVjdGx5RW5hYmxlZCgpKSB7Ci0gICAgICAgICAgICAgICAgdG9vbGtpdC5nZXRTeXN0ZW1FdmVudFF1ZXVlSW1wbCgpLnBvc3RFdmVudCgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXcgTW91c2VFdmVudChpbmZvLnNyYywgZXZlbnQuZ2V0RXZlbnRJZCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV2ZW50LmdldFRpbWUoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBldmVudC5nZXRJbnB1dE1vZGlmaWVycygpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBvcy54LCBwb3MueSwgMCwgZmFsc2UpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIE1vdXNlV2hlZWxFdmVudCBjcmVhdGVXaGVlbEV2ZW50KENvbXBvbmVudCBzcmMsIE5hdGl2ZUV2ZW50IGV2ZW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBvaW50IHdoZXJlKSB7Ci0KLSAgICAgICAgSW50ZWdlciBzY3JvbGxBbW91bnRQcm9wZXJ0eSA9Ci0gICAgICAgICAgICAoSW50ZWdlcil0b29sa2l0LmdldERlc2t0b3BQcm9wZXJ0eSgiYXd0LndoZWVsU2Nyb2xsaW5nU2l6ZSIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIGludCBhbW91bnQgPSAxOwotICAgICAgICBpbnQgdHlwZSA9IE1vdXNlV2hlZWxFdmVudC5XSEVFTF9VTklUX1NDUk9MTDsKLQotICAgICAgICBpZiAoc2Nyb2xsQW1vdW50UHJvcGVydHkgIT0gbnVsbCkgewotICAgICAgICAgICAgYW1vdW50ID0gc2Nyb2xsQW1vdW50UHJvcGVydHkuaW50VmFsdWUoKTsKLSAgICAgICAgICAgIGlmIChhbW91bnQgPT0gLTEpIHsKLSAgICAgICAgICAgICAgICB0eXBlID0gTW91c2VXaGVlbEV2ZW50LldIRUVMX0JMT0NLX1NDUk9MTDsKLSAgICAgICAgICAgICAgICBhbW91bnQgPSAxOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBuZXcgTW91c2VXaGVlbEV2ZW50KHNyYywgZXZlbnQuZ2V0RXZlbnRJZCgpLAotICAgICAgICAgICAgICAgIGV2ZW50LmdldFRpbWUoKSwgZXZlbnQuZ2V0SW5wdXRNb2RpZmllcnMoKSwKLSAgICAgICAgICAgICAgICB3aGVyZS54LCB3aGVyZS55LCAwLCBmYWxzZSwgdHlwZSwgYW1vdW50LAotICAgICAgICAgICAgICAgIGV2ZW50LmdldFdoZWVsUm90YXRpb24oKSk7Ci0gICAgfQotCi0vLyBCRUdJTiBhbmRyb2lkLWNoYW5nZWQ6IEFXVCBjb21wb25lbnRzIG5vdCBzdXBwb3J0ZWQKLS8vICAgIHByaXZhdGUgdm9pZCBkaXNwYXRjaFdoZWVsRXZlbnQoQ29tcG9uZW50IHNyYywgTmF0aXZlRXZlbnQgZXZlbnQpIHsKLS8vICAgICAgICBQb2ludGVySW5mbyBpbmZvID0gZmluZENvbXBvbmVudFVuZGVyUG9pbnRlcigpOwotLy8KLS8vICAgICAgICBpZiAoaW5mby5zcmMgPT0gbnVsbCkgewotLy8gICAgICAgICAgICBpbmZvLnNyYyA9IHNyYzsKLS8vICAgICAgICAgICAgaW5mby5wb3NpdGlvbiA9IGV2ZW50LmdldExvY2FsUG9zKCk7Ci0vLyAgICAgICAgfQotLy8KLS8vICAgICAgICBwcm9wYWdhdGVFdmVudChpbmZvLCBBV1RFdmVudC5NT1VTRV9XSEVFTF9FVkVOVF9NQVNLLAotLy8gICAgICAgICAgICAgICAgICAgICAgIE1vdXNlV2hlZWxMaXN0ZW5lci5jbGFzcywgdHJ1ZSk7Ci0vLyAgICAgICAgaWYgKChpbmZvLnNyYyAhPSBudWxsKSAmJiBpbmZvLnNyYy5pc0luZGlyZWN0bHlFbmFibGVkKCkpIHsKLS8vICAgICAgICAgICAgdG9vbGtpdC5nZXRTeXN0ZW1FdmVudFF1ZXVlSW1wbCgpLnBvc3RFdmVudCgKLS8vICAgICAgICAgICAgICAgICAgICBjcmVhdGVXaGVlbEV2ZW50KGluZm8uc3JjLCBldmVudCwgaW5mby5wb3NpdGlvbikpOwotLy8gICAgICAgIH0KLS8vICAgIH0KLS8vIEVORCBhbmRyb2lkLWNoYW5nZWQKLQotICAgIHByaXZhdGUgUG9pbnRlckluZm8gcHJvcGFnYXRlRXZlbnQoUG9pbnRlckluZm8gaW5mbywgbG9uZyBtYXNrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ2xhc3M8PyBleHRlbmRzIEV2ZW50TGlzdGVuZXI+IHR5cGUsIGJvb2xlYW4gcGllcmNlSFcpIHsKLSAgICAgICAgQ29tcG9uZW50IHNyYyA9IGluZm8uc3JjOwotICAgICAgICB3aGlsZSAoKHNyYyAhPSBudWxsKSAmJgotICAgICAgICAgICAgICAgKHNyYy5pc0xpZ2h0d2VpZ2h0KCkgfHwgcGllcmNlSFcpICYmCi0gICAgICAgICAgICAgICEoc3JjLmlzTW91c2VFdmVudEVuYWJsZWQobWFzaykgfHwKLSAgICAgICAgICAgICAgIChzcmMuZ2V0TGlzdGVuZXJzKHR5cGUpLmxlbmd0aCA+IDApKSkgewotCi0gICAgICAgICAgICBpbmZvLnBvc2l0aW9uLnRyYW5zbGF0ZShzcmMueCwgc3JjLnkpOwotLy8gQkVHSU4gYW5kcm9pZC1jaGFuZ2VkOiBBV1QgY29tcG9uZW50cyBub3Qgc3VwcG9ydGVkCi0vLyAgICAgICAgICAgIHNyYyA9IHNyYy5wYXJlbnQ7Ci0vLyBFTkQgYW5kcm9pZC1jaGFuZ2VkCi0gICAgICAgICAgICBpbmZvLnNyYyA9IHNyYzsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBpbmZvOwotICAgIH0KLQotLy8gQkVHSU4gYW5kcm9pZC1jaGFuZ2VkOiBBV1QgY29tcG9uZW50cyBub3Qgc3VwcG9ydGVkCi0vLyAgICBXaW5kb3cgZmluZFdpbmRvd0F0KFBvaW50IHApIHsKLS8vICAgICAgICBOYXRpdmVXaW5kb3cgbmF0aXZlV2luZG93ID0KLS8vICAgICAgICAgICAgdG9vbGtpdC5nZXRXaW5kb3dGYWN0b3J5KCkuZ2V0V2luZG93RnJvbVBvaW50KHApOwotLy8KLS8vICAgICAgICBXaW5kb3cgd2luZG93ID0gbnVsbDsKLS8vICAgICAgICBpZiAobmF0aXZlV2luZG93ICE9IG51bGwpIHsKLS8vICAgICAgICAgICAgQ29tcG9uZW50IGNvbXAgPSB0b29sa2l0LmdldENvbXBvbmVudEJ5SWQobmF0aXZlV2luZG93LmdldElkKCkpOwotLy8KLS8vICAgICAgICAgICAgaWYgKGNvbXAgIT0gbnVsbCkgewotLy8gICAgICAgICAgICAgICAgd2luZG93ID0gY29tcC5nZXRXaW5kb3dBbmNlc3RvcigpOwotLy8gICAgICAgICAgICB9Ci0vLyAgICAgICAgfQotLy8gICAgICAgIHJldHVybiB3aW5kb3c7Ci0vLyAgICB9Ci0vLyBFTkQgYW5kcm9pZC1jaGFuZ2VkCi0KLSAgICBwcml2YXRlIGNsYXNzIFBvaW50ZXJJbmZvIHsKLQotICAgICAgICBDb21wb25lbnQgc3JjOwotICAgICAgICBQb2ludCBwb3NpdGlvbjsKLQotICAgICAgICBQb2ludGVySW5mbyhDb21wb25lbnQgc3JjLCBQb2ludCBwb3NpdGlvbikgewotICAgICAgICAgICAgdGhpcy5zcmMgPSBzcmM7Ci0gICAgICAgICAgICB0aGlzLnBvc2l0aW9uID0gcG9zaXRpb247Ci0gICAgICAgIH0KLQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L1BhaW50LmphdmEgYi9hd3QvamF2YS9hd3QvUGFpbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGZlYTNhNy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvUGFpbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDU3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgQWxleGV5IEEuIFBldHJlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLQotLyoqCi0gKiBUaGUgUGFpbnQgaW50ZXJmYWNlIHByb3ZpZGVzIHBvc3NpYmlsaXR5IG9mIGdlbmVyYXRpbmcgY29sb3IgcGF0dGVybnMgaW4KLSAqIGRldmljZSBzcGFjZSBmb3IgZmlsbCwgZHJhdywgb3Igc3Ryb2tlIG9wZXJhdGlvbnMgaW4gYSBHcmFwaGljczJELgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBQYWludCBleHRlbmRzIFRyYW5zcGFyZW5jeSB7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBQYWludENvbnRleHQgd2hpY2ggaXMgdXNlZCB0byBnZW5lcmF0ZSBjb2xvciBwYXR0ZXJucyBmb3IKLSAgICAgKiByZW5kZXJpbmcgb3BlcmF0aW9ucyBvZiBHcmFwaGljczJELgotICAgICAqIAotICAgICAqIEBwYXJhbSBjbQotICAgICAqICAgICAgICAgICAgdGhlIENvbG9yTW9kZWwgb2JqZWN0LCBvciBudWxsLgotICAgICAqIEBwYXJhbSBkZXZpY2VCb3VuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZWN0YW5nbGUgcmVwcmVzZW50cyB0aGUgYm91bmRpbmcgYm94IG9mIGRldmljZSBzcGFjZSBmb3IKLSAgICAgKiAgICAgICAgICAgIHRoZSBncmFwaGljcyByZW5kZXJpbmcgb3BlcmF0aW9ucy4KLSAgICAgKiBAcGFyYW0gdXNlckJvdW5kcwotICAgICAqICAgICAgICAgICAgdGhlIFJlY3RhbmdsZSByZXByZXNlbnRzIGJvdW5kaW5nIGJveCBvZiB1c2VyIHNwYWNlIGZvciB0aGUKLSAgICAgKiAgICAgICAgICAgIGdyYXBoaWNzIHJlbmRlcmluZyBvcGVyYXRpb25zLgotICAgICAqIEBwYXJhbSB4Zm9ybQotICAgICAqICAgICAgICAgICAgdGhlIEFmZmluZVRyYW5zZm9ybSBmb3IgdHJhbnNsYXRpb24gZnJvbSB1c2VyIHNwYWNlIHRvIGRldmljZQotICAgICAqICAgICAgICAgICAgc3BhY2UuCi0gICAgICogQHBhcmFtIGhpbnRzCi0gICAgICogICAgICAgICAgICB0aGUgUmVuZGVyaW5nSGludHMgcHJlZmVyZW5jZXMuCi0gICAgICogQHJldHVybiB0aGUgUGFpbnRDb250ZXh0IGZvciBnZW5lcmF0aW5nIGNvbG9yIHBhdHRlcm5zLgotICAgICAqLwotICAgIFBhaW50Q29udGV4dCBjcmVhdGVDb250ZXh0KENvbG9yTW9kZWwgY20sIFJlY3RhbmdsZSBkZXZpY2VCb3VuZHMsIFJlY3RhbmdsZTJEIHVzZXJCb3VuZHMsCi0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0sIFJlbmRlcmluZ0hpbnRzIGhpbnRzKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9QYWludENvbnRleHQuamF2YSBiL2F3dC9qYXZhL2F3dC9QYWludENvbnRleHQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTY2YjZjYS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvUGFpbnRDb250ZXh0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJhc3RlcjsKLQotLyoqCi0gKiBUaGUgUGFpbnRDb250ZXh0IGludGVyZmFjZSBkZXRlcm1pbmVzIHRoZSBzcGVjaWZpYyBlbnZpcm9ubWVudCBmb3IgZ2VuZXJhdGluZwotICogY29sb3IgcGF0dGVybnMgaW4gZGV2aWNlIHNwYWNlIGZvciBmaWxsLCBkcmF3LCBvciBzdHJva2UgcmVuZGVyaW5nIG9wZXJhdGlvbnMKLSAqIHVzaW5nIEdyYXBoaWNzMkQuIFRoaXMgaW50ZXJmYWNlIHByb3ZpZGVzIGNvbG9ycyB0aHJvdWdoIHRoZSBSYXN0ZXIgb2JqZWN0Ci0gKiBhc3NvY2lhdGVkIHdpdGggdGhlIHNwZWNpZmljIENvbG9yTW9kZWwgZm9yIEdyYXBoaWNzMkQgcmVuZGVyaW5nIG9wZXJhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIFBhaW50Q29udGV4dCB7Ci0KLSAgICAvKioKLSAgICAgKiBSZWxlYXNlcyB0aGUgcmVzb3VyY2VzIGFsbG9jYXRlZCBmb3IgdGhlIG9wZXJhdGlvbi4KLSAgICAgKi8KLSAgICB2b2lkIGRpc3Bvc2UoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvbG9yIG1vZGVsLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIENvbG9yTW9kZWwgb2JqZWN0LgotICAgICAqLwotICAgIENvbG9yTW9kZWwgZ2V0Q29sb3JNb2RlbCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgUmFzdGVyIHdoaWNoIGRlZmluZXMgdGhlIGNvbG9ycyBvZiB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyCi0gICAgICogYXJlYSBmb3IgR3JhcGhpY3MyRCByZW5kZXJpbmcgb3BlcmF0aW9ucy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgZGV2aWNlIHNwYWNlIGFyZWEgZm9yIHdoaWNoIGNvbG9ycyBhcmUKLSAgICAgKiAgICAgICAgICAgIGdlbmVyYXRlZC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgZGV2aWNlIHNwYWNlIGFyZWEgZm9yIHdoaWNoIGNvbG9ycyBhcmUKLSAgICAgKiAgICAgICAgICAgIGdlbmVyYXRlZC4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBkZXZpY2Ugc3BhY2UgYXJlYSBmb3Igd2hpY2ggY29sb3JzIGFyZQotICAgICAqICAgICAgICAgICAgZ2VuZXJhdGVkLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBkZXZpY2Ugc3BhY2UgYXJlYSBmb3Igd2hpY2ggY29sb3JzIGFyZQotICAgICAqICAgICAgICAgICAgZ2VuZXJhdGVkLgotICAgICAqIEByZXR1cm4gdGhlIFJhc3RlciBvYmplY3Qgd2hpY2ggY29udGFpbnMgdGhlIGNvbG9ycyBvZiB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICByZWN0YW5ndWxhciBhcmVhIGZvciBHcmFwaGljczJEIHJlbmRlcmluZyBvcGVyYXRpb25zLgotICAgICAqLwotICAgIFJhc3RlciBnZXRSYXN0ZXIoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgpOwotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L1BvaW50LmphdmEgYi9hd3QvamF2YS9hd3QvUG9pbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOGVjNDI0MS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvUG9pbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDIxMSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERlbmlzIE0uIEtpc2hlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEuYXd0Lmdlb20uUG9pbnQyRDsKLWltcG9ydCBqYXZhLmlvLlNlcmlhbGl6YWJsZTsKLQotLyoqCi0gKiBUaGUgUG9pbnQgY2xhc3MgcmVwcmVzZW50cyBhIHBvaW50IGxvY2F0aW9uIHdpdGggY29vcmRpbmF0ZXMgWCwgWSBpbiBjdXJyZW50Ci0gKiBjb29yZGluYXRlIHN5c3RlbS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBQb2ludCBleHRlbmRzIFBvaW50MkQgaW1wbGVtZW50cyBTZXJpYWxpemFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IHNlcmlhbFZlcnNpb25VSUQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gLTUyNzY5NDA2NDAyNTk3NDk4NTBMOwotCi0gICAgLyoqCi0gICAgICogVGhlIFggY29vcmRpbmF0ZSBvZiBQb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgWSBjb29yZGluYXRlIG9mIFBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgeTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBwb2ludCB3aXRoICgwLCBPKSBjb29yZGluYXRlcywgdGhlIG9yaWdpbiBvZgotICAgICAqIGNvb3JkaW5hdGUgc3lzdGVtLgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludCgpIHsKLSAgICAgICAgc2V0TG9jYXRpb24oMCwgMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHBvaW50IHdpdGggKHgsIHkpIGNvb3JkaW5hdGVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIFBvaW50LgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIFBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludChpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgc2V0TG9jYXRpb24oeCwgeSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHBvaW50LCBnaXZpbmcgaXQgdGhlIHNhbWUgbG9jYXRpb24gYXMgdGhlIHBhcmFtZXRlciBwLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgUG9pbnQgb2JqZWN0IGdpdmluZyB0aGUgY29vcmRpbmF0ZXMgb2YgdGhlIG5ldyBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUG9pbnQoUG9pbnQgcCkgewotICAgICAgICBzZXRMb2NhdGlvbihwLngsIHAueSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgY3VycmVudCBQb2ludCB3aXRoIHRoZSBzcGVjaWZpZWQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgT2JqZWN0IGJlaW5nIGNvbXBhcmVkIGlzIGEgUG9pbnQgd2hvc2UgY29vcmRpbmF0ZXMKLSAgICAgKiAgICAgICAgIGFyZSBlcXVhbCB0byB0aGUgY29vcmRpbmF0ZXMgb2YgdGhpcyBQb2ludCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEBzZWUgamF2YS5hd3QuZ2VvbS5Qb2ludDJEI2VxdWFscyhPYmplY3QpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvYmopIHsKLSAgICAgICAgaWYgKG9iaiA9PSB0aGlzKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgICAgICBpZiAob2JqIGluc3RhbmNlb2YgUG9pbnQpIHsKLSAgICAgICAgICAgIFBvaW50IHAgPSAoUG9pbnQpb2JqOwotICAgICAgICAgICAgcmV0dXJuIHggPT0gcC54ICYmIHkgPT0gcC55OwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgY3VycmVudCBQb2ludCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiBhIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgY3VycmVudCBQb2ludCBvYmplY3QuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsKLSAgICAgICAgcmV0dXJuIGdldENsYXNzKCkuZ2V0TmFtZSgpICsgIlt4PSIgKyB4ICsgIix5PSIgKyB5ICsgIl0iOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgWCBjb29yZGluYXRlIG9mIFBvaW50IGFzIGEgZG91YmxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gWCBjb29yZGluYXRlIG9mIHRoZSBwb2ludCBhcyBhIGRvdWJsZS4KLSAgICAgKiBAc2VlIGphdmEuYXd0Lmdlb20uUG9pbnQyRCNnZXRYKCkKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZG91YmxlIGdldFgoKSB7Ci0gICAgICAgIHJldHVybiB4OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgWSBjb29yZGluYXRlIG9mIFBvaW50IGFzIGEgZG91YmxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gWSBjb29yZGluYXRlIG9mIHRoZSBwb2ludCBhcyBhIGRvdWJsZS4KLSAgICAgKiBAc2VlIGphdmEuYXd0Lmdlb20uUG9pbnQyRCNnZXRZKCkKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZG91YmxlIGdldFkoKSB7Ci0gICAgICAgIHJldHVybiB5OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvY2F0aW9uIG9mIHRoZSBQb2ludCBhcyBhIG5ldyBQb2ludCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiBhIGNvcHkgb2YgdGhlIFBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludCBnZXRMb2NhdGlvbigpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBQb2ludCh4LCB5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBsb2NhdGlvbiBvZiB0aGUgUG9pbnQgdG8gdGhlIHNhbWUgY29vcmRpbmF0ZXMgYXMgcC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIFBvaW50IHRoYXQgZ2l2ZXMgdGhlIG5ldyBsb2NhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRMb2NhdGlvbihQb2ludCBwKSB7Ci0gICAgICAgIHNldExvY2F0aW9uKHAueCwgcC55KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBsb2NhdGlvbiBvZiB0aGUgUG9pbnQgdG8gdGhlIGNvb3JkaW5hdGVzIFgsIFkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIFBvaW50J3MgbmV3IGxvY2F0aW9uLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBQb2ludCdzIG5ldyBsb2NhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRMb2NhdGlvbihpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgdGhpcy54ID0geDsKLSAgICAgICAgdGhpcy55ID0geTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBsb2NhdGlvbiBvZiBQb2ludCB0byB0aGUgc3BlY2lmaWVkIGRvdWJsZSBjb29yZGluYXRlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggdGhlIFBvaW50J3MgbmV3IGxvY2F0aW9uLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSB0aGUgUG9pbnQncyBuZXcgbG9jYXRpb24uCi0gICAgICogQHNlZSBqYXZhLmF3dC5nZW9tLlBvaW50MkQjc2V0TG9jYXRpb24oZG91YmxlLCBkb3VibGUpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0TG9jYXRpb24oZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0gICAgICAgIHggPSB4IDwgSW50ZWdlci5NSU5fVkFMVUUgPyBJbnRlZ2VyLk1JTl9WQUxVRSA6IHggPiBJbnRlZ2VyLk1BWF9WQUxVRSA/IEludGVnZXIuTUFYX1ZBTFVFCi0gICAgICAgICAgICAgICAgOiB4OwotICAgICAgICB5ID0geSA8IEludGVnZXIuTUlOX1ZBTFVFID8gSW50ZWdlci5NSU5fVkFMVUUgOiB5ID4gSW50ZWdlci5NQVhfVkFMVUUgPyBJbnRlZ2VyLk1BWF9WQUxVRQotICAgICAgICAgICAgICAgIDogeTsKLSAgICAgICAgc2V0TG9jYXRpb24oKGludClNYXRoLnJvdW5kKHgpLCAoaW50KU1hdGgucm91bmQoeSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIE1vdmVzIHRoZSBQb2ludCB0byB0aGUgc3BlY2lmaWVkICh4LCB5KSBsb2NhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgbmV3IGxvY2F0aW9uLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBuZXcgbG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgbW92ZShpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgc2V0TG9jYXRpb24oeCwgeSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVHJhbnNsYXRlcyBjdXJyZW50IFBvaW50IG1vdmluZyBpdCBmcm9tIHRoZSBwb3NpdGlvbiAoeCwgeSkgdG8gdGhlIG5ldwotICAgICAqIHBvc2l0aW9uIGdpdmVuIGJ5ICh4K2R4LCB4K2R5KSBjb29yZGluYXRlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBob3Jpem9udGFsIGRlbHRhIC0gdGhlIFBvaW50IGlzIG1vdmVkIHRvIHRoaXMgZGlzdGFuY2UKLSAgICAgKiAgICAgICAgICAgIGFsb25nIFggYXhpcy4KLSAgICAgKiBAcGFyYW0gZHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZXJ0aWNhbCBkZWx0YSAtIHRoZSBQb2ludCBpcyBtb3ZlZCB0byB0aGlzIGRpc3RhbmNlIGFsb25nCi0gICAgICogICAgICAgICAgICBZIGF4aXMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgdHJhbnNsYXRlKGludCBkeCwgaW50IGR5KSB7Ci0gICAgICAgIHggKz0gZHg7Ci0gICAgICAgIHkgKz0gZHk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvUG9seWdvbi5qYXZhIGIvYXd0L2phdmEvYXd0L1BvbHlnb24uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGUzMWViOS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvUG9seWdvbi5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTE1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuUG9pbnQ7Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LlNoYXBlOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUGF0aEl0ZXJhdG9yOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUG9pbnQyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEuaW8uU2VyaWFsaXphYmxlOwotaW1wb3J0IGphdmEudXRpbC5Ob1N1Y2hFbGVtZW50RXhjZXB0aW9uOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC4qOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBQb2x5Z29uIGNsYXNzIGRlZmluZXMgYW4gY2xvc2VkIGFyZWEgc3BlY2lmaWVkIGJ5IG4gdmVydGljZXMgYW5kIG4gZWRnZXMuCi0gKiBUaGUgY29vcmRpbmF0ZXMgb2YgdGhlIHZlcnRpY2VzIGFyZSBzcGVjaWZpZWQgYnkgeCwgeSBhcnJheXMuIFRoZSBlZGdlcyBhcmUKLSAqIHRoZSBsaW5lIHNlZ21lbnRzIGZyb20gdGhlIHBvaW50ICh4W2ldLCB5W2ldKSB0byB0aGUgcG9pbnQgKHhbaSsxXSwgeVtpKzFdKSwKLSAqIGZvciAtMSA8IGkgPCAobi0xKSBwbHVzIHRoZSBsaW5lIHNlZ21lbnQgZnJvbSB0aGUgcG9pbnQgKHhbbi0xXSwgeVtuLTFdKSB0bwotICogdGhlIHBvaW50ICh4WzBdLCB5WzBdKSBwb2ludC4gVGhlIFBvbHlnb24gaXMgZW1wdHkgaWYgdGhlIG51bWJlciBvZiB2ZXJ0aWNlcwotICogaXMgemVyby4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBQb2x5Z29uIGltcGxlbWVudHMgU2hhcGUsIFNlcmlhbGl6YWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtNjQ2MDA2MTQzNzkwMDA2OTk2OUw7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcG9pbnRzIGJ1ZmZlciBjYXBhY2l0eS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgQlVGRkVSX0NBUEFDSVRZID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBudW1iZXIgb2YgUG9seWdvbiB2ZXJ0aWNlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IG5wb2ludHM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYXJyYXkgb2YgWCBjb29yZGluYXRlcyBvZiB0aGUgdmVydGljZXMuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIHhwb2ludHM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYXJyYXkgb2YgWSBjb29yZGluYXRlcyBvZiB0aGUgdmVydGljZXMuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIHlwb2ludHM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc21hbGxlc3QgUmVjdGFuZ2xlIHRoYXQgY29tcGxldGVseSBjb250YWlucyB0aGlzIFBvbHlnb24uCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFJlY3RhbmdsZSBib3VuZHM7Ci0KLSAgICAvKgotICAgICAqIFBvbHlnb24gcGF0aCBpdGVyYXRvcgotICAgICAqLwotICAgIC8qKgotICAgICAqIFRoZSBpbnRlcm5hbCBDbGFzcyBJdGVyYXRvci4KLSAgICAgKi8KLSAgICBjbGFzcyBJdGVyYXRvciBpbXBsZW1lbnRzIFBhdGhJdGVyYXRvciB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzb3VyY2UgUG9seWdvbiBvYmplY3QuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgUG9seWdvbiBwOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcGF0aCBpdGVyYXRvciB0cmFuc2Zvcm1hdGlvbi4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0gdDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGN1cnJlbnQgc2VnbWVudCBpbmRleC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBpbnQgaW5kZXg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENvbnN0cnVjdHMgYSBuZXcgUG9seWdvbi5JdGVyYXRvciBmb3IgdGhlIGdpdmVuIHBvbHlnb24gYW5kCi0gICAgICAgICAqIHRyYW5zZm9ybWF0aW9uCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gYXQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtIG9iamVjdCB0byBhcHBseSByZWN0YW5nbGUgcGF0aC4KLSAgICAgICAgICogQHBhcmFtIHAKLSAgICAgICAgICogICAgICAgICAgICB0aGUgcC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gYXQsIFBvbHlnb24gcCkgewotICAgICAgICAgICAgdGhpcy5wID0gcDsKLSAgICAgICAgICAgIHRoaXMudCA9IGF0OwotICAgICAgICAgICAgaWYgKHAubnBvaW50cyA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgaW5kZXggPSAxOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGludCBnZXRXaW5kaW5nUnVsZSgpIHsKLSAgICAgICAgICAgIHJldHVybiBXSU5EX0VWRU5fT0REOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGJvb2xlYW4gaXNEb25lKCkgewotICAgICAgICAgICAgcmV0dXJuIGluZGV4ID4gcC5ucG9pbnRzOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIHZvaWQgbmV4dCgpIHsKLSAgICAgICAgICAgIGluZGV4Kys7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGRvdWJsZVtdIGNvb3JkcykgewotICAgICAgICAgICAgaWYgKGlzRG9uZSgpKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjExMD1JdGVyYXRvciBvdXQgb2YgYm91bmRzCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IE5vU3VjaEVsZW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTEwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoaW5kZXggPT0gcC5ucG9pbnRzKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIFNFR19DTE9TRTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNvb3Jkc1swXSA9IHAueHBvaW50c1tpbmRleF07Ci0gICAgICAgICAgICBjb29yZHNbMV0gPSBwLnlwb2ludHNbaW5kZXhdOwotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCAxKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBpbmRleCA9PSAwID8gU0VHX01PVkVUTyA6IFNFR19MSU5FVE87Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGZsb2F0W10gY29vcmRzKSB7Ci0gICAgICAgICAgICBpZiAoaXNEb25lKCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMTEwPUl0ZXJhdG9yIG91dCBvZiBib3VuZHMKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgTm9TdWNoRWxlbWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xMTAiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChpbmRleCA9PSBwLm5wb2ludHMpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gU0VHX0NMT1NFOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY29vcmRzWzBdID0gcC54cG9pbnRzW2luZGV4XTsKLSAgICAgICAgICAgIGNvb3Jkc1sxXSA9IHAueXBvaW50c1tpbmRleF07Ci0gICAgICAgICAgICBpZiAodCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdC50cmFuc2Zvcm0oY29vcmRzLCAwLCBjb29yZHMsIDAsIDEpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGluZGV4ID09IDAgPyBTRUdfTU9WRVRPIDogU0VHX0xJTkVUTzsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBlbXB0eSBwb2x5Z29uLgotICAgICAqLwotICAgIHB1YmxpYyBQb2x5Z29uKCkgewotICAgICAgICB4cG9pbnRzID0gbmV3IGludFtCVUZGRVJfQ0FQQUNJVFldOwotICAgICAgICB5cG9pbnRzID0gbmV3IGludFtCVUZGRVJfQ0FQQUNJVFldOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBwb2x5Z29uIHdpdGggdGhlIHNwZWNpZmllZCBudW1iZXIgb2YgdmVydGljZXMsIGFuZCB0aGUKLSAgICAgKiBnaXZlbiBhcnJheXMgb2YgeCwgeSB2ZXJ0ZXggY29vcmRpbmF0ZXMuIFRoZSBsZW5ndGggb2YgZWFjaCBjb29yZGluYXRlCi0gICAgICogYXJyYXkgbWF5IG5vdCBiZSBsZXNzIHRoYW4gdGhlIHNwZWNpZmllZCBudW1iZXIgb2YgdmVydGljZXMgYnV0IG1heSBiZQotICAgICAqIGdyZWF0ZXIuIE9ubHkgdGhlIGZpcnN0IG4gZWxlbWVudHMgYXJlIHVzZWQgZnJvbSBlYWNoIGNvb3JkaW5hdGUgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHhwb2ludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBYIHZlcnRleCBjb29yZGluYXRlcy4KLSAgICAgKiBAcGFyYW0geXBvaW50cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIFkgdmVydGV4IGNvb3JkaW5hdGVzLgotICAgICAqIEBwYXJhbSBucG9pbnRzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIHZlcnRpY2VzIG9mIHRoZSBwb2x5Z29uLgotICAgICAqIEB0aHJvd3MgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBsZW5ndGggb2YgeHBvaW50cyBvciB5cG9pbnRzIGlzIGxlc3MgdGhhbiBuLgotICAgICAqIEB0aHJvd3MgTmVnYXRpdmVBcnJheVNpemVFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBuIGlzIG5lZ2F0aXZlLgotICAgICAqLwotICAgIHB1YmxpYyBQb2x5Z29uKGludFtdIHhwb2ludHMsIGludFtdIHlwb2ludHMsIGludCBucG9pbnRzKSB7Ci0gICAgICAgIGlmIChucG9pbnRzID4geHBvaW50cy5sZW5ndGggfHwgbnBvaW50cyA+IHlwb2ludHMubGVuZ3RoKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTExPVBhcmFtZXRlciBucG9pbnRzIGlzIGdyZWF0ZXIgdGhhbiBhcnJheSBsZW5ndGgKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjExMSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGlmIChucG9pbnRzIDwgMCkgewotICAgICAgICAgICAgLy8gYXd0LjExMj1OZWdhdGl2ZSBudW1iZXIgb2YgcG9pbnRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgTmVnYXRpdmVBcnJheVNpemVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTEyIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy5ucG9pbnRzID0gbnBvaW50czsKLSAgICAgICAgdGhpcy54cG9pbnRzID0gbmV3IGludFtucG9pbnRzXTsKLSAgICAgICAgdGhpcy55cG9pbnRzID0gbmV3IGludFtucG9pbnRzXTsKLSAgICAgICAgU3lzdGVtLmFycmF5Y29weSh4cG9pbnRzLCAwLCB0aGlzLnhwb2ludHMsIDAsIG5wb2ludHMpOwotICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHlwb2ludHMsIDAsIHRoaXMueXBvaW50cywgMCwgbnBvaW50cyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVzZXRzIHRoZSBjdXJyZW50IFBvbHlnb24gdG8gYW4gZW1wdHkgUG9seWdvbi4gTW9yZSBwcmVjaXNlbHksIHRoZQotICAgICAqIG51bWJlciBvZiBQb2x5Z29uIHZlcnRpY2VzIGlzIHNldCB0byB6ZXJvLCBidXQgeCwgeSBjb29yZGluYXRlcyBhcnJheXMKLSAgICAgKiBhcmUgbm90IGFmZmVjdGVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlc2V0KCkgewotICAgICAgICBucG9pbnRzID0gMDsKLSAgICAgICAgYm91bmRzID0gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnZhbGlkYXRlcyB0aGUgZGF0YSB0aGF0IGRlcGVuZHMgb24gdGhlIHZlcnRleCBjb29yZGluYXRlcy4gVGhpcyBtZXRob2QKLSAgICAgKiBzaG91bGQgYmUgY2FsbGVkIGFmdGVyIGRpcmVjdCBtYW5pcHVsYXRpb25zIG9mIHRoZSB4LCB5IHZlcnRleAotICAgICAqIGNvb3JkaW5hdGVzIGFycmF5cyB0byBhdm9pZCB1bnByZWRpY3RhYmxlIHJlc3VsdHMgb2YgbWV0aG9kcyB3aGljaCByZWx5Ci0gICAgICogb24gdGhlIGJvdW5kaW5nIGJveC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBpbnZhbGlkYXRlKCkgewotICAgICAgICBib3VuZHMgPSBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgdGhlIHBvaW50IHRvIHRoZSBQb2x5Z29uIGFuZCB1cGRhdGVzIHRoZSBib3VuZGluZyBib3ggYWNjb3JkaW5nbHkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBhZGRlZCB2ZXJ0ZXguCi0gICAgICogQHBhcmFtIHB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBhZGRlZCB2ZXJ0ZXguCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkUG9pbnQoaW50IHB4LCBpbnQgcHkpIHsKLSAgICAgICAgaWYgKG5wb2ludHMgPT0geHBvaW50cy5sZW5ndGgpIHsKLSAgICAgICAgICAgIGludFtdIHRtcDsKLQotICAgICAgICAgICAgdG1wID0gbmV3IGludFt4cG9pbnRzLmxlbmd0aCArIEJVRkZFUl9DQVBBQ0lUWV07Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHhwb2ludHMsIDAsIHRtcCwgMCwgeHBvaW50cy5sZW5ndGgpOwotICAgICAgICAgICAgeHBvaW50cyA9IHRtcDsKLQotICAgICAgICAgICAgdG1wID0gbmV3IGludFt5cG9pbnRzLmxlbmd0aCArIEJVRkZFUl9DQVBBQ0lUWV07Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHlwb2ludHMsIDAsIHRtcCwgMCwgeXBvaW50cy5sZW5ndGgpOwotICAgICAgICAgICAgeXBvaW50cyA9IHRtcDsKLSAgICAgICAgfQotCi0gICAgICAgIHhwb2ludHNbbnBvaW50c10gPSBweDsKLSAgICAgICAgeXBvaW50c1tucG9pbnRzXSA9IHB5OwotICAgICAgICBucG9pbnRzKys7Ci0KLSAgICAgICAgaWYgKGJvdW5kcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICBib3VuZHMuc2V0RnJhbWVGcm9tRGlhZ29uYWwoTWF0aC5taW4oYm91bmRzLmdldE1pblgoKSwgcHgpLCBNYXRoLm1pbihib3VuZHMuZ2V0TWluWSgpLAotICAgICAgICAgICAgICAgICAgICBweSksIE1hdGgubWF4KGJvdW5kcy5nZXRNYXhYKCksIHB4KSwgTWF0aC5tYXgoYm91bmRzLmdldE1heFkoKSwgcHkpKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZSBvZiB0aGUgUG9seWdvbi4gVGhlIGJvdW5kaW5nIHJlY3RhbmdsZSBpcyB0aGUKLSAgICAgKiBzbWFsbGVzdCByZWN0YW5nbGUgd2hpY2ggY29udGFpbnMgdGhlIFBvbHlnb24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlIG9mIHRoZSBQb2x5Z29uLgotICAgICAqIEBzZWUgamF2YS5hd3QuU2hhcGUjZ2V0Qm91bmRzKCkKLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldEJvdW5kcygpIHsKLSAgICAgICAgaWYgKGJvdW5kcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gYm91bmRzOwotICAgICAgICB9Ci0gICAgICAgIGlmIChucG9pbnRzID09IDApIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgYngxID0geHBvaW50c1swXTsKLSAgICAgICAgaW50IGJ5MSA9IHlwb2ludHNbMF07Ci0gICAgICAgIGludCBieDIgPSBieDE7Ci0gICAgICAgIGludCBieTIgPSBieTE7Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDE7IGkgPCBucG9pbnRzOyBpKyspIHsKLSAgICAgICAgICAgIGludCB4ID0geHBvaW50c1tpXTsKLSAgICAgICAgICAgIGludCB5ID0geXBvaW50c1tpXTsKLSAgICAgICAgICAgIGlmICh4IDwgYngxKSB7Ci0gICAgICAgICAgICAgICAgYngxID0geDsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoeCA+IGJ4MikgewotICAgICAgICAgICAgICAgIGJ4MiA9IHg7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoeSA8IGJ5MSkgewotICAgICAgICAgICAgICAgIGJ5MSA9IHk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKHkgPiBieTIpIHsKLSAgICAgICAgICAgICAgICBieTIgPSB5OwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGJvdW5kcyA9IG5ldyBSZWN0YW5nbGUoYngxLCBieTEsIGJ4MiAtIGJ4MSwgYnkyIC0gYnkxKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZGluZyByZWN0YW5nbGUgb2YgdGhlIFBvbHlnb24uIFRoZSBib3VuZGluZyByZWN0YW5nbGUgaXMgdGhlCi0gICAgICogc21hbGxlc3QgcmVjdGFuZ2xlIHdoaWNoIGNvbnRhaW5zIHRoZSBQb2x5Z29uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJvdW5kaW5nIHJlY3RhbmdsZSBvZiB0aGUgUG9seWdvbi4KLSAgICAgKiBAZGVwcmVjYXRlZCBVc2UgZ2V0Qm91bmRzKCkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIFJlY3RhbmdsZSBnZXRCb3VuZGluZ0JveCgpIHsKLSAgICAgICAgcmV0dXJuIGdldEJvdW5kcygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFJlY3RhbmdsZTJEIHdoaWNoIHJlcHJlc2VudHMgUG9seWdvbiBib3VuZHMuIFRoZSBib3VuZGluZwotICAgICAqIHJlY3RhbmdsZSBpcyB0aGUgc21hbGxlc3QgcmVjdGFuZ2xlIHdoaWNoIGNvbnRhaW5zIHRoZSBQb2x5Z29uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJvdW5kaW5nIHJlY3RhbmdsZSBvZiB0aGUgUG9seWdvbi4KLSAgICAgKiBAc2VlIGphdmEuYXd0LlNoYXBlI2dldEJvdW5kczJEKCkKLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoKSB7Ci0gICAgICAgIHJldHVybiBnZXRCb3VuZHMoKS5nZXRCb3VuZHMyRCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRyYW5zbGF0ZXMgYWxsIHZlcnRpY2VzIG9mIFBvbHlnb24gdGhlIHNwZWNpZmllZCBkaXN0YW5jZXMgYWxvbmcgWCwgWQotICAgICAqIGF4aXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIG14Ci0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgdG8gdHJhbnNsYXRlIGhvcml6b250YWxseS4KLSAgICAgKiBAcGFyYW0gbXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSB0byB0cmFuc2xhdGUgdmVydGljYWxseS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB0cmFuc2xhdGUoaW50IG14LCBpbnQgbXkpIHsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBucG9pbnRzOyBpKyspIHsKLSAgICAgICAgICAgIHhwb2ludHNbaV0gKz0gbXg7Ci0gICAgICAgICAgICB5cG9pbnRzW2ldICs9IG15OwotICAgICAgICB9Ci0gICAgICAgIGlmIChib3VuZHMgIT0gbnVsbCkgewotICAgICAgICAgICAgYm91bmRzLnRyYW5zbGF0ZShteCwgbXkpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IHRoZSBwb2ludCBnaXZlbiBieSB0aGUgY29vcmRpbmF0ZXMgeCwgeSBsaWVzIGluc2lkZQotICAgICAqIHRoZSBQb2x5Z29uLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBwb2ludCB0byBjaGVjay4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcG9pbnQgdG8gY2hlY2suCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIHBvaW50IGxpZXMgaW5zaWRlIHRoZSBQb2x5Z29uLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqIEBkZXByZWNhdGVkIFVzZSBjb250YWlucyhpbnQsIGludCkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIGJvb2xlYW4gaW5zaWRlKGludCB4LCBpbnQgeSkgewotICAgICAgICByZXR1cm4gY29udGFpbnMoKGRvdWJsZSl4LCAoZG91YmxlKXkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGUgcG9pbnQgZ2l2ZW4gYnkgdGhlIGNvb3JkaW5hdGVzIHgsIHkgbGllcyBpbnNpZGUKLSAgICAgKiB0aGUgUG9seWdvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcG9pbnQgdG8gY2hlY2suCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHBvaW50IHRvIGNoZWNrLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCBwb2ludCBsaWVzIGluc2lkZSB0aGUgUG9seWdvbiwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgcmV0dXJuIGNvbnRhaW5zKChkb3VibGUpeCwgKGRvdWJsZSl5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIHBvaW50IHdpdGggc3BlY2lmaWVkIGRvdWJsZSBjb29yZGluYXRlcyBsaWVzCi0gICAgICogaW5zaWRlIHRoZSBQb2x5Z29uLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBwb2ludCB0byBjaGVjay4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcG9pbnQgdG8gY2hlY2suCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgcG9pbnQgZ2l2ZW4gYnkgdGhlIGRvdWJsZSBjb29yZGluYXRlcyBsaWVzIGluc2lkZQotICAgICAqICAgICAgICAgdGhlIFBvbHlnb24sIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKiBAc2VlIGphdmEuYXd0LlNoYXBlI2NvbnRhaW5zKGRvdWJsZSwgZG91YmxlKQotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSB4LCBkb3VibGUgeSkgewotICAgICAgICByZXR1cm4gQ3Jvc3NpbmcuaXNJbnNpZGVFdmVuT2RkKENyb3NzaW5nLmNyb3NzU2hhcGUodGhpcywgeCwgeSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGUgcmVjdGFuZ2xlIGRldGVybWluZWQgYnkgdGhlIHBhcmFtZXRlcnMgW3gsIHksCi0gICAgICogd2lkdGgsIGhlaWdodF0gbGllcyBpbnNpZGUgdGhlIFBvbHlnb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZXMncyBsZWZ0IHVwcGVyIGNvcm5lciBhcyBhCi0gICAgICogICAgICAgICAgICBkb3VibGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZXMncyBsZWZ0IHVwcGVyIGNvcm5lciBhcyBhCi0gICAgICogICAgICAgICAgICBkb3VibGUuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgcmVjdGFuZ2xlIGFzIGEgZG91YmxlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ2xlIGFzIGEgZG91YmxlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCByZWN0YW5nbGUgbGllcyBpbnNpZGUgdGhlIFBvbHlnb24sIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICogQHNlZSBqYXZhLmF3dC5TaGFwZSNjb250YWlucyhkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUpCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQpIHsKLSAgICAgICAgaW50IGNyb3NzID0gQ3Jvc3NpbmcuaW50ZXJzZWN0U2hhcGUodGhpcywgeCwgeSwgd2lkdGgsIGhlaWdodCk7Ci0gICAgICAgIHJldHVybiBjcm9zcyAhPSBDcm9zc2luZy5DUk9TU0lORyAmJiBDcm9zc2luZy5pc0luc2lkZUV2ZW5PZGQoY3Jvc3MpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGUgcmVjdGFuZ2xlIGRldGVybWluZWQgYnkgdGhlIHBhcmFtZXRlcnMgW3gsIHksCi0gICAgICogd2lkdGgsIGhlaWdodF0gaW50ZXJzZWN0cyB0aGUgaW50ZXJpb3Igb2YgdGhlIFBvbHlnb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZXMncyBsZWZ0IHVwcGVyIGNvcm5lciBhcyBhCi0gICAgICogICAgICAgICAgICBkb3VibGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZXMncyBsZWZ0IHVwcGVyIGNvcm5lciBhcyBhCi0gICAgICogICAgICAgICAgICBkb3VibGUuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgcmVjdGFuZ2xlIGFzIGEgZG91YmxlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ2xlIGFzIGEgZG91YmxlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCByZWN0YW5nbGUgaW50ZXJzZWN0cyB0aGUgaW50ZXJpb3Igb2YgdGhlCi0gICAgICogICAgICAgICBQb2x5Z29uLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICogQHNlZSBqYXZhLmF3dC5TaGFwZSNpbnRlcnNlY3RzKGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSkKLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpbnRlcnNlY3RzKGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0KSB7Ci0gICAgICAgIGludCBjcm9zcyA9IENyb3NzaW5nLmludGVyc2VjdFNoYXBlKHRoaXMsIHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICByZXR1cm4gY3Jvc3MgPT0gQ3Jvc3NpbmcuQ1JPU1NJTkcgfHwgQ3Jvc3NpbmcuaXNJbnNpZGVFdmVuT2RkKGNyb3NzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIHNwZWNpZmllZCByZWN0YW5nbGUgbGllcyBpbnNpZGUgdGhlIFBvbHlnb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHJlY3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZWN0YW5nbGUyRCBvYmplY3QuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIHJlY3RhbmdsZSBsaWVzIGluc2lkZSB0aGUgUG9seWdvbiwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKiBAc2VlIGphdmEuYXd0LlNoYXBlI2NvbnRhaW5zKGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQpCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoUmVjdGFuZ2xlMkQgcmVjdCkgewotICAgICAgICByZXR1cm4gY29udGFpbnMocmVjdC5nZXRYKCksIHJlY3QuZ2V0WSgpLCByZWN0LmdldFdpZHRoKCksIHJlY3QuZ2V0SGVpZ2h0KCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGUgc3BlY2lmaWVkIFBvaW50IGxpZXMgaW5zaWRlIHRoZSBQb2x5Z29uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwb2ludAotICAgICAqICAgICAgICAgICAgdGhlIFBvaW50IG9iamVjdC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgUG9pbnQgbGllcyBpbnNpZGUgdGhlIFBvbHlnb24sIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoUG9pbnQgcG9pbnQpIHsKLSAgICAgICAgcmV0dXJuIGNvbnRhaW5zKHBvaW50LmdldFgoKSwgcG9pbnQuZ2V0WSgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIHNwZWNpZmllZCBQb2ludDJEIGxpZXMgaW5zaWRlIHRoZSBQb2x5Z29uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwb2ludAotICAgICAqICAgICAgICAgICAgdGhlIFBvaW50MkQgb2JqZWN0LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCBQb2ludDJEIGxpZXMgaW5zaWRlIHRoZSBQb2x5Z29uLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqIEBzZWUgamF2YS5hd3QuU2hhcGUjY29udGFpbnMoamF2YS5hd3QuZ2VvbS5Qb2ludDJEKQotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFBvaW50MkQgcG9pbnQpIHsKLSAgICAgICAgcmV0dXJuIGNvbnRhaW5zKHBvaW50LmdldFgoKSwgcG9pbnQuZ2V0WSgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIGludGVyaW9yIG9mIHJlY3RhbmdsZSBzcGVjaWZpZWQgYnkgdGhlCi0gICAgICogUmVjdGFuZ2xlMkQgb2JqZWN0IGludGVyc2VjdHMgdGhlIGludGVyaW9yIG9mIHRoZSBQb2x5Z29uLgotICAgICAqIAotICAgICAqIEBwYXJhbSByZWN0Ci0gICAgICogICAgICAgICAgICB0aGUgUmVjdGFuZ2xlMkQgb2JqZWN0LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIFJlY3RhbmdsZTJEIGludGVyc2VjdHMgdGhlIGludGVyaW9yIG9mIHRoZSBQb2x5Z29uLAotICAgICAqICAgICAgICAgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEBzZWUgamF2YS5hd3QuU2hhcGUjaW50ZXJzZWN0cyhqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEKQotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHMoUmVjdGFuZ2xlMkQgcmVjdCkgewotICAgICAgICByZXR1cm4gaW50ZXJzZWN0cyhyZWN0LmdldFgoKSwgcmVjdC5nZXRZKCksIHJlY3QuZ2V0V2lkdGgoKSwgcmVjdC5nZXRIZWlnaHQoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgUGF0aEl0ZXJhdG9yIG9iamVjdCB3aGljaCBnaXZlcyB0aGUgY29vcmRpbmF0ZXMgb2YgdGhlIHBvbHlnb24sCi0gICAgICogdHJhbnNmb3JtZWQgYWNjb3JkaW5nIHRvIHRoZSBzcGVjaWZpZWQgQWZmaW5lVHJhbnNmb3JtLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0Ci0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIEFmZmluZVRyYW5zZm9ybSBvYmplY3Qgb3IgbnVsbC4KLSAgICAgKiBAcmV0dXJuIFBhdGhJdGVyYXRvciBvYmplY3QgZm9yIHRoZSBQb2x5Z29uLgotICAgICAqIEBzZWUgamF2YS5hd3QuU2hhcGUjZ2V0UGF0aEl0ZXJhdG9yKGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtKQotICAgICAqLwotICAgIHB1YmxpYyBQYXRoSXRlcmF0b3IgZ2V0UGF0aEl0ZXJhdG9yKEFmZmluZVRyYW5zZm9ybSB0KSB7Ci0gICAgICAgIHJldHVybiBuZXcgSXRlcmF0b3IodCwgdGhpcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgUGF0aEl0ZXJhdG9yIG9iamVjdCB3aGljaCBnaXZlcyB0aGUgY29vcmRpbmF0ZXMgb2YgdGhlIHBvbHlnb24sCi0gICAgICogdHJhbnNmb3JtZWQgYWNjb3JkaW5nIHRvIHRoZSBzcGVjaWZpZWQgQWZmaW5lVHJhbnNmb3JtLiBUaGUgZmxhdG5lc3MKLSAgICAgKiBwYXJhbWV0ZXIgaXMgaWdub3JlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBBZmZpbmVUcmFuc2Zvcm0gb2JqZWN0IG9yIG51bGwuCi0gICAgICogQHBhcmFtIGZsYXRuZXNzCi0gICAgICogICAgICAgICAgICB0aGUgbWF4aW11bSBudW1iZXIgb2YgdGhlIGNvbnRyb2wgcG9pbnRzIGZvciBhIGdpdmVuIGN1cnZlCi0gICAgICogICAgICAgICAgICB3aGljaCB2YXJpZXMgZnJvbSBjb2xpbmVhciBiZWZvcmUgYSBzdWJkaXZpZGVkIGN1cnZlIGlzCi0gICAgICogICAgICAgICAgICByZXBsYWNlZCBieSBhIHN0cmFpZ2h0IGxpbmUgY29ubmVjdGluZyB0aGUgZW5kcG9pbnRzLiBUaGlzCi0gICAgICogICAgICAgICAgICBwYXJhbWV0ZXIgaXMgaWdub3JlZCBmb3IgdGhlIFBvbHlnb24gY2xhc3MuCi0gICAgICogQHJldHVybiBQYXRoSXRlcmF0b3Igb2JqZWN0IGZvciB0aGUgUG9seWdvbi4KLSAgICAgKiBAc2VlIGphdmEuYXd0LlNoYXBlI2dldFBhdGhJdGVyYXRvcihqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybSwKLSAgICAgKiAgICAgIGRvdWJsZSkKLSAgICAgKi8KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gdCwgZG91YmxlIGZsYXRuZXNzKSB7Ci0gICAgICAgIHJldHVybiBuZXcgSXRlcmF0b3IodCwgdGhpcyk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvUmVjdGFuZ2xlLmphdmEgYi9hd3QvamF2YS9hd3QvUmVjdGFuZ2xlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGQ4ZWJiM2EuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L1JlY3RhbmdsZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNzIzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0OwotCi1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLWltcG9ydCBqYXZhLmlvLlNlcmlhbGl6YWJsZTsKLQotLyoqCi0gKiBUaGUgUmVjdGFuZ2xlIGNsYXNzIGRlZmluZXMgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgaW4gdGVybXMgb2YgaXRzIHVwcGVyIGxlZnQKLSAqIGNvcm5lciBjb29yZGluYXRlcyBbeCx5XSwgaXRzIHdpZHRoLCBhbmQgaXRzIGhlaWdodC4gQSBSZWN0YW5nbGUgc3BlY2lmaWVkIGJ5Ci0gKiBbeCwgeSwgd2lkdGgsIGhlaWdodF0gcGFyYW1ldGVycyBoYXMgYW4gb3V0bGluZSBwYXRoIHdpdGggY29ybmVycyBhdCBbeCwgeV0sCi0gKiBbeCArIHdpZHRoLHldLCBbeCArIHdpZHRoLHkgKyBoZWlnaHRdLCBhbmQgW3gsIHkgKyBoZWlnaHRdLiA8YnI+Ci0gKiA8YnI+Ci0gKiBUaGUgcmVjdGFuZ2xlIGlzIGVtcHR5IGlmIHRoZSB3aWR0aCBvciBoZWlnaHQgaXMgbmVnYXRpdmUgb3IgemVyby4gSW4gdGhpcwotICogY2FzZSB0aGUgaXNFbXB0eSBtZXRob2QgcmV0dXJucyB0cnVlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIFJlY3RhbmdsZSBleHRlbmRzIFJlY3RhbmdsZTJEIGltcGxlbWVudHMgU2hhcGUsIFNlcmlhbGl6YWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtNDM0NTg1NzA3MDI1NTY3NDc2NEw7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgWCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyBsZWZ0IHVwcGVyIGNvcm5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyBsZWZ0IHVwcGVyIGNvcm5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgd2lkdGggb2YgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgd2lkdGg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGVpZ2h0IG9mIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGhlaWdodDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyByZWN0YW5nbGUgd2l0aCBbMCwgMF0gdXBwZXIgbGVmdCBjb3JuZXIgY29vcmRpbmF0ZXMsCi0gICAgICogdGhlIHdpZHRoIGFuZCB0aGUgaGVpZ2h0IGFyZSB6ZXJvLgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUoKSB7Ci0gICAgICAgIHNldEJvdW5kcygwLCAwLCAwLCAwKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgcmVjdGFuZ2xlIHdob3NlIHVwcGVyIGxlZnQgY29ybmVyIGNvb3JkaW5hdGVzIGFyZQotICAgICAqIGdpdmVuIGJ5IHRoZSBQb2ludCBvYmplY3QgKHAuWCBhbmQgcC5ZKSwgYW5kIHRoZSB3aWR0aCBhbmQgdGhlIGhlaWdodCBhcmUKLSAgICAgKiB6ZXJvLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgUG9pbnQgc3BlY2lmaWVzIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBjb29yZGluYXRlcyBvZiB0aGUKLSAgICAgKiAgICAgICAgICAgIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlKFBvaW50IHApIHsKLSAgICAgICAgc2V0Qm91bmRzKHAueCwgcC55LCAwLCAwKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgcmVjdGFuZ2xlIHdob3NlIHVwcGVyIGxlZnQgY29ybmVyIGNvb3JkaW5hdGVzIGFyZQotICAgICAqIGdpdmVuIGJ5IHRoZSBQb2ludCBvYmplY3QgKHAuWCBhbmQgcC5ZKSwgYW5kIHRoZSB3aWR0aCBhbmQgdGhlIGhlaWdodCBhcmUKLSAgICAgKiBnaXZlbiBieSBEaW1lbnNpb24gb2JqZWN0IChkLndpZHRoIGFuZCBkLmhlaWdodCkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHAKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb2ludCBzcGVjaWZpZXMgdGhlIHVwcGVyIGxlZnQgY29ybmVyIGNvb3JkaW5hdGVzIG9mIHRoZQotICAgICAqICAgICAgICAgICAgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBkCi0gICAgICogICAgICAgICAgICB0aGUgZGltZW5zaW9uIHNwZWNpZmllcyB0aGUgd2lkdGggYW5kIHRoZSBoZWlnaHQgb2YgdGhlCi0gICAgICogICAgICAgICAgICByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZShQb2ludCBwLCBEaW1lbnNpb24gZCkgewotICAgICAgICBzZXRCb3VuZHMocC54LCBwLnksIGQud2lkdGgsIGQuaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgcmVjdGFuZ2xlIGRldGVybWluZWQgYnkgdGhlIHVwcGVyIGxlZnQgY29ybmVyCi0gICAgICogY29vcmRpbmF0ZXMgKHgsIHkpLCB3aWR0aCBhbmQgaGVpZ2h0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCB1cHBlciBsZWZ0IGNvcm5lciBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIHVwcGVyIGxlZnQgY29ybmVyIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZShpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICBzZXRCb3VuZHMoeCwgeSwgd2lkdGgsIGhlaWdodCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHJlY3RhbmdsZSB3aXRoIFswLCAwXSBhcyBpdHMgdXBwZXIgbGVmdCBjb3JuZXIKLSAgICAgKiBjb29yZGluYXRlcyBhbmQgdGhlIHNwZWNpZmllZCB3aWR0aCBhbmQgaGVpZ2h0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlKGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICBzZXRCb3VuZHMoMCwgMCwgd2lkdGgsIGhlaWdodCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHJlY3RhbmdsZSB3aXRoIHRoZSBzYW1lIGNvb3JkaW5hdGVzIGFzIHRoZSBnaXZlbgotICAgICAqIHNvdXJjZSByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZWN0YW5nbGUgb2JqZWN0IHdoaWNoIHBhcmFtZXRlcnMgd2lsbCBiZSB1c2VkIGZvcgotICAgICAqICAgICAgICAgICAgaW5zdGFudGlhdGluZyBhIG5ldyBSZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZShSZWN0YW5nbGUgcikgewotICAgICAgICBzZXRCb3VuZHMoci54LCByLnksIHIud2lkdGgsIHIuaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIHB1YmxpYyBSZWN0YW5nbGUoRGltZW5zaW9uIGQpIHsgc2V0Qm91bmRzKDAsIDAsIGQud2lkdGgsIGQuaGVpZ2h0KTsgfQotICAgICAqLwotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFggY29vcmRpbmF0ZSBvZiBib3VuZCBhcyBhIGRvdWJsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBYIGNvb3JkaW5hdGUgb2YgYm91bmQgYXMgYSBkb3VibGUuCi0gICAgICogQHNlZSBqYXZhLmF3dC5nZW9tLlJlY3Rhbmd1bGFyU2hhcGUjZ2V0WCgpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGRvdWJsZSBnZXRYKCkgewotICAgICAgICByZXR1cm4geDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBZIGNvb3JkaW5hdGUgb2YgYm91bmQgYXMgYSBkb3VibGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgWSBjb29yZGluYXRlIG9mIGJvdW5kIGFzIGEgZG91YmxlLgotICAgICAqIEBzZWUgamF2YS5hd3QuZ2VvbS5SZWN0YW5ndWxhclNoYXBlI2dldFkoKQotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBkb3VibGUgZ2V0WSgpIHsKLSAgICAgICAgcmV0dXJuIHk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUgYXMgYSBkb3VibGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUgYXMgYSBkb3VibGUuCi0gICAgICogQHNlZSBqYXZhLmF3dC5nZW9tLlJlY3Rhbmd1bGFyU2hhcGUjZ2V0SGVpZ2h0KCkKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZG91YmxlIGdldEhlaWdodCgpIHsKLSAgICAgICAgcmV0dXJuIGhlaWdodDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIGFzIGEgZG91YmxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUgYXMgYSBkb3VibGUuCi0gICAgICogQHNlZSBqYXZhLmF3dC5nZW9tLlJlY3Rhbmd1bGFyU2hhcGUjZ2V0V2lkdGgoKQotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBkb3VibGUgZ2V0V2lkdGgoKSB7Ci0gICAgICAgIHJldHVybiB3aWR0aDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXRlcm1pbmVzIHdoZXRoZXIgb3Igbm90IHRoZSByZWN0YW5nbGUgaXMgZW1wdHkuIFRoZSByZWN0YW5nbGUgaXMgZW1wdHkKLSAgICAgKiBpZiBpdHMgd2lkdGggb3IgaGVpZ2h0IGlzIG5lZ2F0aXZlIG9yIHplcm8uCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgcmVjdGFuZ2xlIGlzIGVtcHR5LCBvdGhlcndpc2UgZmFsc2UuCi0gICAgICogQHNlZSBqYXZhLmF3dC5nZW9tLlJlY3Rhbmd1bGFyU2hhcGUjaXNFbXB0eSgpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNFbXB0eSgpIHsKLSAgICAgICAgcmV0dXJuIHdpZHRoIDw9IDAgfHwgaGVpZ2h0IDw9IDA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2l6ZSBvZiBhIFJlY3RhbmdsZSBhcyBEaW1lbnNpb24gb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBEaW1lbnNpb24gb2JqZWN0IHdoaWNoIHJlcHJlc2VudHMgc2l6ZSBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBEaW1lbnNpb24gZ2V0U2l6ZSgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBEaW1lbnNpb24od2lkdGgsIGhlaWdodCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2l6ZSBvZiB0aGUgUmVjdGFuZ2xlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIG5ldyB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0U2l6ZShpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgdGhpcy53aWR0aCA9IHdpZHRoOwotICAgICAgICB0aGlzLmhlaWdodCA9IGhlaWdodDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBzaXplIG9mIGEgUmVjdGFuZ2xlIHNwZWNpZmllZCBhcyBEaW1lbnNpb24gb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBkCi0gICAgICogICAgICAgICAgICBhIERpbWVuc2lvbiBvYmplY3Qgd2hpY2ggcmVwcmVzZW50cyBuZXcgc2l6ZSBvZiBhIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRTaXplKERpbWVuc2lvbiBkKSB7Ci0gICAgICAgIHNldFNpemUoZC53aWR0aCwgZC5oZWlnaHQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvY2F0aW9uIG9mIGEgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIgYXMgYSBQb2ludCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgUG9pbnQgb2JqZWN0IHdpdGggY29vcmRpbmF0ZXMgZXF1YWwgdG8gdGhlIHVwcGVyIGxlZnQgY29ybmVyCi0gICAgICogICAgICAgICBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludCBnZXRMb2NhdGlvbigpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBQb2ludCh4LCB5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBsb2NhdGlvbiBvZiB0aGUgcmVjdGFuZ2xlIGluIHRlcm1zIG9mIGl0cyB1cHBlciBsZWZ0IGNvcm5lcgotICAgICAqIGNvb3JkaW5hdGVzIFggYW5kIFkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRMb2NhdGlvbihpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgdGhpcy54ID0geDsKLSAgICAgICAgdGhpcy55ID0geTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBsb2NhdGlvbiBvZiBhIHJlY3RhbmdsZSB1c2luZyBhIFBvaW50IG9iamVjdCB0byBnaXZlIHRoZQotICAgICAqIGNvb3JkaW5hdGVzIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIFBvaW50IG9iamVjdCB3aGljaCByZXByZXNlbnRzIHRoZSBuZXcgdXBwZXIgbGVmdCBjb3JuZXIKLSAgICAgKiAgICAgICAgICAgIGNvb3JkaW5hdGVzIG9mIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRMb2NhdGlvbihQb2ludCBwKSB7Ci0gICAgICAgIHNldExvY2F0aW9uKHAueCwgcC55KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBNb3ZlcyBhIHJlY3RhbmdsZSB0byB0aGUgbmV3IGxvY2F0aW9uIGJ5IG1vdmluZyBpdHMgdXBwZXIgbGVmdCBjb3JuZXIgdG8KLSAgICAgKiB0aGUgcG9pbnQgd2l0aCBjb29yZGluYXRlcyBYIGFuZCBZLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgbmV3IFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgKiBAZGVwcmVjYXRlZCBVc2Ugc2V0TG9jYXRpb24oaW50LCBpbnQpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyB2b2lkIG1vdmUoaW50IHgsIGludCB5KSB7Ci0gICAgICAgIHNldExvY2F0aW9uKHgsIHkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHJlY3RhbmdsZSB0byBiZSB0aGUgbmVhcmVzdCByZWN0YW5nbGUgd2l0aCBpbnRlZ2VyIGNvb3JkaW5hdGVzCi0gICAgICogYm91bmRpbmcgdGhlIHJlY3RhbmdsZSBkZWZpbmVkIGJ5IHRoZSBkb3VibGUtdmFsdWVkIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSBkb3VibGUtdmFsdWVkCi0gICAgICogICAgICAgICAgICByZWN0YW5nbGUgdG8gYmUgYm91bmRlZC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIGRvdWJsZS12YWx1ZWQKLSAgICAgKiAgICAgICAgICAgIHJlY3RhbmdsZSB0byBiZSBib3VuZGVkLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUgdG8gYmUgYm91bmRlZC4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUgdG8gYmUgYm91bmRlZC4KLSAgICAgKiBAc2VlIGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQjc2V0UmVjdChkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UmVjdChkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3aWR0aCwgZG91YmxlIGhlaWdodCkgewotICAgICAgICBpbnQgeDEgPSAoaW50KU1hdGguZmxvb3IoeCk7Ci0gICAgICAgIGludCB5MSA9IChpbnQpTWF0aC5mbG9vcih5KTsKLSAgICAgICAgaW50IHgyID0gKGludClNYXRoLmNlaWwoeCArIHdpZHRoKTsKLSAgICAgICAgaW50IHkyID0gKGludClNYXRoLmNlaWwoeSArIGhlaWdodCk7Ci0gICAgICAgIHNldEJvdW5kcyh4MSwgeTEsIHgyIC0geDEsIHkyIC0geTEpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgYSBuZXcgc2l6ZSBmb3IgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSByZWN0YW5nbGUncyBuZXcgd2lkdGguCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIHJlY3RhbmdsZSdzIG5ldyBoZWlnaHQuCi0gICAgICogQGRlcHJlY2F0ZWQgdXNlIHRoZSBzZXRTaXplKGludCwgaW50KSBtZXRob2QuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgdm9pZCByZXNpemUoaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIHNldEJvdW5kcyh4LCB5LCB3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXNldHMgdGhlIGJvdW5kcyBvZiBhIHJlY3RhbmdsZSB0byB0aGUgc3BlY2lmaWVkIHgsIHksIHdpZHRoIGFuZCBoZWlnaHQKLSAgICAgKiBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgbmV3IFggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgWSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgd2lkdGggb2YgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgaGVpZ2h0IG9mIHJlY3RhbmdsZS4KLSAgICAgKiBAZGVwcmVjYXRlZCB1c2Ugc2V0Qm91bmRzKGludCwgaW50LCBpbnQsIGludCkgbWV0aG9kCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgdm9pZCByZXNoYXBlKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIHNldEJvdW5kcyh4LCB5LCB3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGJvdW5kcyBvZiB0aGUgcmVjdGFuZ2xlIGFzIGEgbmV3IFJlY3RhbmdsZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgUmVjdGFuZ2xlIG9iamVjdCB3aXRoIHRoZSBzYW1lIGJvdW5kcyBhcyB0aGUgb3JpZ2luYWwKLSAgICAgKiAgICAgICAgIHJlY3RhbmdsZS4KLSAgICAgKiBAc2VlIGphdmEuYXd0Lmdlb20uUmVjdGFuZ3VsYXJTaGFwZSNnZXRCb3VuZHMoKQotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Qm91bmRzKCkgewotICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZSh4LCB5LCB3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZHMgb2YgdGhlIG9yaWdpbmFsIHJlY3RhbmdsZSBhcyBhIFJlY3RhbmdsZTJEIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBSZWN0YW5nbGUyRCBvYmplY3Qgd2hpY2ggcmVwcmVzZW50cyB0aGUgYm91bmRzIG9mIHRoZQotICAgICAqICAgICAgICAgb3JpZ2luYWwgcmVjdGFuZ2xlLgotICAgICAqIEBzZWUgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRCNnZXRCb3VuZHMyRCgpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKCkgewotICAgICAgICByZXR1cm4gZ2V0Qm91bmRzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgYm91bmRzIG9mIGEgcmVjdGFuZ2xlIHRvIHRoZSBzcGVjaWZpZWQgeCwgeSwgd2lkdGgsIGFuZCBoZWlnaHQKLSAgICAgKiBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEJvdW5kcyhpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICB0aGlzLnggPSB4OwotICAgICAgICB0aGlzLnkgPSB5OwotICAgICAgICB0aGlzLmhlaWdodCA9IGhlaWdodDsKLSAgICAgICAgdGhpcy53aWR0aCA9IHdpZHRoOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGJvdW5kcyBvZiB0aGUgcmVjdGFuZ2xlIHRvIG1hdGNoIHRoZSBib3VuZHMgb2YgdGhlIFJlY3RhbmdsZQotICAgICAqIG9iamVjdCBzZW50IGFzIGEgcGFyYW1ldGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSByCi0gICAgICogICAgICAgICAgICB0aGUgUmVjdGFuZ2xlIG9iamVjdCB3aGljaCBzcGVjaWZpZXMgdGhlIG5ldyBib3VuZHMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Qm91bmRzKFJlY3RhbmdsZSByKSB7Ci0gICAgICAgIHNldEJvdW5kcyhyLngsIHIueSwgci53aWR0aCwgci5oZWlnaHQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEVubGFyZ2VzIHRoZSByZWN0YW5nbGUgYnkgbW92aW5nIGVhY2ggY29ybmVyIG91dHdhcmQgZnJvbSB0aGUgY2VudGVyIGJ5IGEKLSAgICAgKiBkaXN0YW5jZSBvZiBkeCBob3Jpem9uYWxseSBhbmQgYSBkaXN0YW5jZSBvZiBkeSB2ZXJ0aWNhbGx5LiBTcGVjaWZpY2FsbHksCi0gICAgICogY2hhbmdlcyBhIHJlY3RhbmdsZSB3aXRoIFt4LCB5LCB3aWR0aCwgaGVpZ2h0XSBwYXJhbWV0ZXJzIHRvIGEgcmVjdGFuZ2xlCi0gICAgICogd2l0aCBbeC1keCwgeS1keSwgd2lkdGgrMipkeCwgaGVpZ2h0KzIqZHldIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGR4Ci0gICAgICogICAgICAgICAgICB0aGUgaG9yaXpvbnRhbCBkaXN0YW5jZSB0byBtb3ZlIGVhY2ggY29ybmVyIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIGR5Ci0gICAgICogICAgICAgICAgICB0aGUgdmVydGljYWwgZGlzdGFuY2UgdG8gbW92ZSBlYWNoIGNvcm5lciBjb29yZGluYXRlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGdyb3coaW50IGR4LCBpbnQgZHkpIHsKLSAgICAgICAgeCAtPSBkeDsKLSAgICAgICAgeSAtPSBkeTsKLSAgICAgICAgd2lkdGggKz0gZHggKyBkeDsKLSAgICAgICAgaGVpZ2h0ICs9IGR5ICsgZHk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogTW92ZXMgYSByZWN0YW5nbGUgYSBkaXN0YW5jZSBvZiBteCBhbG9uZyB0aGUgeCBjb29yZGluYXRlIGF4aXMgYW5kIGEKLSAgICAgKiBkaXN0YW5jZSBvZiBteSBhbG9uZyB5IGNvb3JkaW5hdGUgYXhpcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBob3Jpem9udGFsIHRyYW5zbGF0aW9uIGluY3JlbWVudC4KLSAgICAgKiBAcGFyYW0gbXkKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZXJ0aWNhbCB0cmFuc2xhdGlvbiBpbmNyZW1lbnQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgdHJhbnNsYXRlKGludCBteCwgaW50IG15KSB7Ci0gICAgICAgIHggKz0gbXg7Ci0gICAgICAgIHkgKz0gbXk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRW5sYXJnZXMgdGhlIHJlY3RhbmdsZSB0byBjb3ZlciB0aGUgc3BlY2lmaWVkIHBvaW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBweAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgbmV3IHBvaW50IHRvIGJlIGNvdmVyZWQgYnkgdGhlCi0gICAgICogICAgICAgICAgICByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBuZXcgcG9pbnQgdG8gYmUgY292ZXJlZCBieSB0aGUKLSAgICAgKiAgICAgICAgICAgIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGQoaW50IHB4LCBpbnQgcHkpIHsKLSAgICAgICAgaW50IHgxID0gTWF0aC5taW4oeCwgcHgpOwotICAgICAgICBpbnQgeDIgPSBNYXRoLm1heCh4ICsgd2lkdGgsIHB4KTsKLSAgICAgICAgaW50IHkxID0gTWF0aC5taW4oeSwgcHkpOwotICAgICAgICBpbnQgeTIgPSBNYXRoLm1heCh5ICsgaGVpZ2h0LCBweSk7Ci0gICAgICAgIHNldEJvdW5kcyh4MSwgeTEsIHgyIC0geDEsIHkyIC0geTEpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEVubGFyZ2VzIHRoZSByZWN0YW5nbGUgdG8gY292ZXIgdGhlIHNwZWNpZmllZCBwb2ludCB3aXRoIHRoZSBuZXcgcG9pbnQKLSAgICAgKiBnaXZlbiBhcyBhIFBvaW50IG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIFBvaW50IG9iamVjdCB0aGF0IHNwZWNpZmllcyB0aGUgbmV3IHBvaW50IHRvIGJlIGNvdmVyZWQgYnkKLSAgICAgKiAgICAgICAgICAgIHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkKFBvaW50IHApIHsKLSAgICAgICAgYWRkKHAueCwgcC55KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGEgbmV3IHJlY3RhbmdsZSB0byB0aGUgb3JpZ2luYWwgcmVjdGFuZ2xlLCB0aGUgcmVzdWx0IGlzIGFuIHVuaW9uIG9mCi0gICAgICogdGhlIHNwZWNpZmllZCBzcGVjaWZpZWQgcmVjdGFuZ2xlIGFuZCBvcmlnaW5hbCByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZWN0YW5nbGUgd2hpY2ggaXMgYWRkZWQgdG8gdGhlIG9yaWdpbmFsIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGQoUmVjdGFuZ2xlIHIpIHsKLSAgICAgICAgaW50IHgxID0gTWF0aC5taW4oeCwgci54KTsKLSAgICAgICAgaW50IHgyID0gTWF0aC5tYXgoeCArIHdpZHRoLCByLnggKyByLndpZHRoKTsKLSAgICAgICAgaW50IHkxID0gTWF0aC5taW4oeSwgci55KTsKLSAgICAgICAgaW50IHkyID0gTWF0aC5tYXgoeSArIGhlaWdodCwgci55ICsgci5oZWlnaHQpOwotICAgICAgICBzZXRCb3VuZHMoeDEsIHkxLCB4MiAtIHgxLCB5MiAtIHkxKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXRlcm1pbmVzIHdoZXRoZXIgb3Igbm90IHRoZSBwb2ludCB3aXRoIHNwZWNpZmllZCBjb29yZGluYXRlcyBbcHgsIHB5XQotICAgICAqIGlzIHdpdGhpbiB0aGUgYm91bmRzIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHBvaW50LgotICAgICAqIEBwYXJhbSBweQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBwb2ludCB3aXRoIHNwZWNpZmllZCBjb29yZGluYXRlcyBbcHgsIHB5XSBpcyB3aXRoaW4KLSAgICAgKiAgICAgICAgIHRoZSBib3VuZHMgb2YgdGhlIHJlY3RhbmdsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGludCBweCwgaW50IHB5KSB7Ci0gICAgICAgIGlmIChpc0VtcHR5KCkpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgICAgICBpZiAocHggPCB4IHx8IHB5IDwgeSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIHB4IC09IHg7Ci0gICAgICAgIHB5IC09IHk7Ci0gICAgICAgIHJldHVybiBweCA8IHdpZHRoICYmIHB5IDwgaGVpZ2h0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERldGVybWluZXMgd2hldGhlciBvciBub3QgdGhlIHBvaW50IGdpdmVuIGFzIGEgUG9pbnQgb2JqZWN0IGlzIHdpdGhpbiB0aGUKLSAgICAgKiBib3VuZHMgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIFBvaW50IG9iamVjdAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHBvaW50IHAgaXMgd2l0aGluIHRoZSBib3VuZHMgb2YgdGhlIHJlY3RhbmdsZSwKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZSBmYWxzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhQb2ludCBwKSB7Ci0gICAgICAgIHJldHVybiBjb250YWlucyhwLngsIHAueSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGV0ZXJtaW5lcyB3aGV0aGVyIG9yIG5vdCB0aGUgcmVjdGFuZ2xlIHNwZWNpZmllZCBieSBbcngsIHJ5LCBydywgcmhdCi0gICAgICogcGFyYW1ldGVycyBpcyBsb2NhdGVkIGluc2lkZSB0aGUgb3JpZ2luYWwgcmVjdGFuZ2xlLgotICAgICAqIAotICAgICAqIEBwYXJhbSByeAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlIHRvIGNvbXBhcmUuCi0gICAgICogQHBhcmFtIHJ5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgdG8gY29tcGFyZS4KLSAgICAgKiBAcGFyYW0gcncKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIHRvIGNvbXBhcmUuCi0gICAgICogQHBhcmFtIHJoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUgdG8gY29tcGFyZS4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGEgcmVjdGFuZ2xlIHdpdGggW3J4LCByeSwgcncsIHJoXSBwYXJhbWV0ZXJzIGlzIGVudGlyZWx5Ci0gICAgICogICAgICAgICBjb250YWluZWQgaW4gdGhlIG9yaWdpbmFsIHJlY3RhbmdsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGludCByeCwgaW50IHJ5LCBpbnQgcncsIGludCByaCkgewotICAgICAgICByZXR1cm4gY29udGFpbnMocngsIHJ5KSAmJiBjb250YWlucyhyeCArIHJ3IC0gMSwgcnkgKyByaCAtIDEpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIHdoZXRoZXIgb3Igbm90IHRoZSByZWN0YW5nbGUgc3BlY2lmaWVkIGJ5IHRoZSBSZWN0YW5nbGUgb2JqZWN0Ci0gICAgICogaXMgbG9jYXRlZCBpbnNpZGUgdGhlIG9yaWdpbmFsIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcgotICAgICAqICAgICAgICAgICAgdGhlIFJlY3RhbmdsZSBvYmplY3QuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgcmVjdGFuZ2xlIHNwZWNpZmllZCBieSBSZWN0YW5nbGUgb2JqZWN0IGlzIGVudGlyZWx5Ci0gICAgICogICAgICAgICBjb250YWluZWQgaW4gdGhlIG9yaWdpbmFsIHJlY3RhbmdsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFJlY3RhbmdsZSByKSB7Ci0gICAgICAgIHJldHVybiBjb250YWlucyhyLngsIHIueSwgci53aWR0aCwgci5oZWlnaHQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIHdoZXRoZXIgb3Igbm90IGEgcG9pbnQgd2l0aCBzcGVjaWZpZWQgY29vcmRpbmF0ZXMgW3B4LCBweV0KLSAgICAgKiBiZWxvbmdzIHRvIGEgcmVjdGFuZ2xlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBweAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiBhIHBvaW50LgotICAgICAqIEBwYXJhbSBweQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiBhIHBvaW50LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgYSBwb2ludCB3aXRoIHNwZWNpZmllZCBjb29yZGluYXRlcyBbcHgsIHB5XSBiZWxvbmdzIHRvIGEKLSAgICAgKiAgICAgICAgIHJlY3RhbmdsZSwgb3RoZXJ3aXNlIGZhbHNlLgotICAgICAqIEBkZXByZWNhdGVkIHVzZSBjb250YWlucyhpbnQsIGludCkgbWV0aG9kLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIGJvb2xlYW4gaW5zaWRlKGludCBweCwgaW50IHB5KSB7Ci0gICAgICAgIHJldHVybiBjb250YWlucyhweCwgcHkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGludGVyc2VjdGlvbiBvZiB0aGUgb3JpZ2luYWwgcmVjdGFuZ2xlIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIFJlY3RhbmdsZTJELgotICAgICAqIAotICAgICAqIEBwYXJhbSByCi0gICAgICogICAgICAgICAgICB0aGUgUmVjdGFuZ2xlMkQgb2JqZWN0LgotICAgICAqIEByZXR1cm4gdGhlIFJlY3RhbmdsZTJEIG9iamVjdCB0aGF0IGlzIHRoZSByZXN1bHQgb2YgaW50ZXJzZWN0aW5nIHRoZQotICAgICAqICAgICAgICAgb3JpZ2luYWwgcmVjdGFuZ2xlIHdpdGggdGhlIHNwZWNpZmllZCBSZWN0YW5nbGUyRC4KLSAgICAgKiBAc2VlIGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQjY3JlYXRlSW50ZXJzZWN0aW9uKGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGNyZWF0ZUludGVyc2VjdGlvbihSZWN0YW5nbGUyRCByKSB7Ci0gICAgICAgIGlmIChyIGluc3RhbmNlb2YgUmVjdGFuZ2xlKSB7Ci0gICAgICAgICAgICByZXR1cm4gaW50ZXJzZWN0aW9uKChSZWN0YW5nbGUpcik7Ci0gICAgICAgIH0KLSAgICAgICAgUmVjdGFuZ2xlMkQgZHN0ID0gbmV3IFJlY3RhbmdsZTJELkRvdWJsZSgpOwotICAgICAgICBSZWN0YW5nbGUyRC5pbnRlcnNlY3QodGhpcywgciwgZHN0KTsKLSAgICAgICAgcmV0dXJuIGRzdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBpbnRlcnNlY3Rpb24gb2YgdGhlIG9yaWdpbmFsIHJlY3RhbmdsZSB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiByZWN0YW5nbGUuIEFuIGVtcHR5IHJlY3RhbmdsZSBpcyByZXR1cm5lZCBpZiB0aGVyZSBpcyBubyBpbnRlcnNlY3Rpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZWN0YW5nbGUgb2JqZWN0LgotICAgICAqIEByZXR1cm4gdGhlIFJlY3RhbmdsZSBvYmplY3QgaXMgcmVzdWx0IG9mIHRoZSBvcmlnaW5hbCByZWN0YW5nbGUgd2l0aCB0aGUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZCByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZSBpbnRlcnNlY3Rpb24oUmVjdGFuZ2xlIHIpIHsKLSAgICAgICAgaW50IHgxID0gTWF0aC5tYXgoeCwgci54KTsKLSAgICAgICAgaW50IHkxID0gTWF0aC5tYXgoeSwgci55KTsKLSAgICAgICAgaW50IHgyID0gTWF0aC5taW4oeCArIHdpZHRoLCByLnggKyByLndpZHRoKTsKLSAgICAgICAgaW50IHkyID0gTWF0aC5taW4oeSArIGhlaWdodCwgci55ICsgci5oZWlnaHQpOwotICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZSh4MSwgeTEsIHgyIC0geDEsIHkyIC0geTEpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERldGVybWluZXMgd2hldGhlciBvciBub3QgdGhlIG9yaWdpbmFsIHJlY3RhbmdsZSBpbnRlcnNlY3RzIHRoZSBzcGVjaWZpZWQKLSAgICAgKiByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZWN0YW5nbGUgb2JqZWN0LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHR3byByZWN0YW5nbGVzIG92ZXJsYXAsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpbnRlcnNlY3RzKFJlY3RhbmdsZSByKSB7Ci0gICAgICAgIHJldHVybiAhaW50ZXJzZWN0aW9uKHIpLmlzRW1wdHkoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXRlcm1pbmVzIHdoZXJlIHRoZSBzcGVjaWZpZWQgUG9pbnQgaXMgbG9jYXRlZCB3aXRoIHJlc3BlY3QgdG8gdGhlCi0gICAgICogcmVjdGFuZ2xlLiBUaGlzIG1ldGhvZCBjb21wdXRlcyB3aGV0aGVyIHRoZSBwb2ludCBpcyB0byB0aGUgcmlnaHQgb3IgdG8KLSAgICAgKiB0aGUgbGVmdCBvZiB0aGUgcmVjdGFuZ2xlIGFuZCB3aGV0aGVyIGl0IGlzIGFib3ZlIG9yIGJlbG93IHRoZSByZWN0YW5nbGUsCi0gICAgICogYW5kIHBhY2tzIHRoZSByZXN1bHQgaW50byBhbiBpbnRlZ2VyIGJ5IHVzaW5nIGEgYmluYXJ5IE9SIG9wZXJhdGlvbiB3aXRoCi0gICAgICogdGhlIGZvbGxvd2luZyBtYXNrczoKLSAgICAgKiA8dWw+Ci0gICAgICo8bGk+UmVjdGFuZ2xlMkQuT1VUX0xFRlQ8L2xpPgotICAgICAqPGxpPlJlY3RhbmdsZTJELk9VVF9UT1A8L2xpPgotICAgICAqPGxpPlJlY3RhbmdsZTJELk9VVF9SSUdIVDwvbGk+Ci0gICAgICo8bGk+UmVjdGFuZ2xlMkQuT1VUX0JPVFRPTTwvbGk+Ci0gICAgICo8L3VsPgotICAgICAqIElmIHRoZSByZWN0YW5nbGUgaXMgZW1wdHksIGFsbCBtYXNrcyBhcmUgc2V0LCBhbmQgaWYgdGhlIHBvaW50IGlzIGluc2lkZQotICAgICAqIHRoZSByZWN0YW5nbGUsIG5vbmUgYXJlIHNldC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiBAcGFyYW0gcHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSBsb2NhdGlvbiBvZiB0aGUgUG9pbnQgcmVsYXRpdmUgdG8gdGhlIHJlY3RhbmdsZSBhcyB0aGUgcmVzdWx0Ci0gICAgICogICAgICAgICBvZiBsb2dpY2FsIE9SIG9wZXJhdGlvbiB3aXRoIGFsbCBvdXQgbWFza3MuCi0gICAgICogQHNlZSBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEI291dGNvZGUoZG91YmxlLCBkb3VibGUpCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBvdXRjb2RlKGRvdWJsZSBweCwgZG91YmxlIHB5KSB7Ci0gICAgICAgIGludCBjb2RlID0gMDsKLQotICAgICAgICBpZiAod2lkdGggPD0gMCkgewotICAgICAgICAgICAgY29kZSB8PSBPVVRfTEVGVCB8IE9VVF9SSUdIVDsKLSAgICAgICAgfSBlbHNlIGlmIChweCA8IHgpIHsKLSAgICAgICAgICAgIGNvZGUgfD0gT1VUX0xFRlQ7Ci0gICAgICAgIH0gZWxzZSBpZiAocHggPiB4ICsgd2lkdGgpIHsKLSAgICAgICAgICAgIGNvZGUgfD0gT1VUX1JJR0hUOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGhlaWdodCA8PSAwKSB7Ci0gICAgICAgICAgICBjb2RlIHw9IE9VVF9UT1AgfCBPVVRfQk9UVE9NOwotICAgICAgICB9IGVsc2UgaWYgKHB5IDwgeSkgewotICAgICAgICAgICAgY29kZSB8PSBPVVRfVE9QOwotICAgICAgICB9IGVsc2UgaWYgKHB5ID4geSArIGhlaWdodCkgewotICAgICAgICAgICAgY29kZSB8PSBPVVRfQk9UVE9NOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGNvZGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRW5sYXJnZXMgdGhlIHJlY3RhbmdsZSB0byBjb3ZlciB0aGUgc3BlY2lmaWVkIFJlY3RhbmdsZTJELgotICAgICAqIAotICAgICAqIEBwYXJhbSByCi0gICAgICogICAgICAgICAgICB0aGUgUmVjdGFuZ2xlMkQgb2JqZWN0LgotICAgICAqIEByZXR1cm4gdGhlIHVuaW9uIG9mIHRoZSBvcmlnaW5hbCBhbmQgdGhlIHNwZWNpZmllZCBSZWN0YW5nbGUyRC4KLSAgICAgKiBAc2VlIGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQjY3JlYXRlVW5pb24oamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRCkKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgY3JlYXRlVW5pb24oUmVjdGFuZ2xlMkQgcikgewotICAgICAgICBpZiAociBpbnN0YW5jZW9mIFJlY3RhbmdsZSkgewotICAgICAgICAgICAgcmV0dXJuIHVuaW9uKChSZWN0YW5nbGUpcik7Ci0gICAgICAgIH0KLSAgICAgICAgUmVjdGFuZ2xlMkQgZHN0ID0gbmV3IFJlY3RhbmdsZTJELkRvdWJsZSgpOwotICAgICAgICBSZWN0YW5nbGUyRC51bmlvbih0aGlzLCByLCBkc3QpOwotICAgICAgICByZXR1cm4gZHN0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEVubGFyZ2VzIHRoZSByZWN0YW5nbGUgdG8gY292ZXIgdGhlIHNwZWNpZmllZCByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZWN0YW5nbGUuCi0gICAgICogQHJldHVybiB0aGUgdW5pb24gb2YgdGhlIG9yaWdpbmFsIGFuZCB0aGUgc3BlY2lmaWVkIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlIHVuaW9uKFJlY3RhbmdsZSByKSB7Ci0gICAgICAgIFJlY3RhbmdsZSBkc3QgPSBuZXcgUmVjdGFuZ2xlKHRoaXMpOwotICAgICAgICBkc3QuYWRkKHIpOwotICAgICAgICByZXR1cm4gZHN0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIHRoZSBvcmlnaW5hbCBSZWN0YW5nbGUgd2l0aCB0aGUgc3BlY2lmaWVkIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb2JqCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIE9iamVjdCBmb3IgY29tcGFyaXNvbi4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgT2JqZWN0IGlzIGEgcmVjdGFuZ2xlIHdpdGggdGhlIHNhbWUKLSAgICAgKiAgICAgICAgIGRpbWVuc2lvbnMgYXMgdGhlIG9yaWdpbmFsIHJlY3RhbmdsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEBzZWUgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRCNlcXVhbHMoT2JqZWN0KQotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3Qgb2JqKSB7Ci0gICAgICAgIGlmIChvYmogPT0gdGhpcykgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG9iaiBpbnN0YW5jZW9mIFJlY3RhbmdsZSkgewotICAgICAgICAgICAgUmVjdGFuZ2xlIHIgPSAoUmVjdGFuZ2xlKW9iajsKLSAgICAgICAgICAgIHJldHVybiByLnggPT0geCAmJiByLnkgPT0geSAmJiByLndpZHRoID09IHdpZHRoICYmIHIuaGVpZ2h0ID09IGhlaWdodDsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgcmVjdGFuZ2xlOyB0aGUgc3RyaW5nIGNvbnRhaW5zIFt4LAotICAgICAqIHksIHdpZHRoLCBoZWlnaHRdIHBhcmFtZXRlcnMgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICAvLyBUaGUgb3V0cHV0IGZvcm1hdCBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvdXIuIEl0IGNvdWxkIGJlCi0gICAgICAgIC8vIG9idGFpbmVkIGluIHRoZSBmb2xsb3dpbmcgd2F5Ci0gICAgICAgIC8vIFN5c3RlbS5vdXQucHJpbnRsbihuZXcgUmVjdGFuZ2xlKCkudG9TdHJpbmcoKSkKLSAgICAgICAgcmV0dXJuIGdldENsYXNzKCkuZ2V0TmFtZSgpICsgIlt4PSIgKyB4ICsgIix5PSIgKyB5ICsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgIix3aWR0aD0iICsgd2lkdGggKyAiLGhlaWdodD0iICsgaGVpZ2h0ICsgIl0iOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L1JlbmRlcmluZ0hpbnRzLmphdmEgYi9hd3QvamF2YS9hd3QvUmVuZGVyaW5nSGludHMuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYWNmNmZhMS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvUmVuZGVyaW5nSGludHMuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDYwNiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLnV0aWwuQ29sbGVjdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaE1hcDsKLWltcG9ydCBqYXZhLnV0aWwuSXRlcmF0b3I7Ci1pbXBvcnQgamF2YS51dGlsLk1hcDsKLWltcG9ydCBqYXZhLnV0aWwuU2V0OwotCi0vKioKLSAqIFRoZSBSZW5kZXJpbmdIaW50cyBjbGFzcyByZXByZXNlbnRzIHByZWZlcmVuY2VzIGZvciB0aGUgcmVuZGVyaW5nIGFsZ29yaXRobXMuCi0gKiBUaGUgcHJlZmVyZW5jZXMgYXJlIGFyYml0cmFyeSBhbmQgY2FuIGJlIHNwZWNpZmllZCBieSBNYXAgb2JqZWN0cyBvciBieQotICoga2V5LXZhbHVlIHBhaXJzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIFJlbmRlcmluZ0hpbnRzIGltcGxlbWVudHMgTWFwPE9iamVjdCwgT2JqZWN0PiwgQ2xvbmVhYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBLRVlfQUxQSEFfSU5URVJQT0xBVElPTiAtIGFscGhhIGludGVycG9sYXRpb24gcmVuZGVyaW5nIGhpbnQKLSAgICAgKiBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBLZXkgS0VZX0FMUEhBX0lOVEVSUE9MQVRJT04gPSBuZXcgS2V5SW1wbCgxKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9BTFBIQV9JTlRFUlBPTEFUSU9OX0RFRkFVTFQgLSBhbHBoYSBpbnRlcnBvbGF0aW9uCi0gICAgICogcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfQUxQSEFfSU5URVJQT0xBVElPTl9ERUZBVUxUID0gbmV3IEtleVZhbHVlKAotICAgICAgICAgICAgS0VZX0FMUEhBX0lOVEVSUE9MQVRJT04pOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZBTFVFX0FMUEhBX0lOVEVSUE9MQVRJT05fU1BFRUQgLSBhbHBoYSBpbnRlcnBvbGF0aW9uCi0gICAgICogcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfQUxQSEFfSU5URVJQT0xBVElPTl9TUEVFRCA9IG5ldyBLZXlWYWx1ZSgKLSAgICAgICAgICAgIEtFWV9BTFBIQV9JTlRFUlBPTEFUSU9OKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9BTFBIQV9JTlRFUlBPTEFUSU9OX1FVQUxJVFkgLSBhbHBoYSBpbnRlcnBvbGF0aW9uCi0gICAgICogcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfQUxQSEFfSU5URVJQT0xBVElPTl9RVUFMSVRZID0gbmV3IEtleVZhbHVlKAotICAgICAgICAgICAgS0VZX0FMUEhBX0lOVEVSUE9MQVRJT04pOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEtFWV9BTlRJQUxJQVNJTkcgLSBhbnRpYWxpYXNpbmcgcmVuZGVyaW5nIGhpbnQga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgS2V5IEtFWV9BTlRJQUxJQVNJTkcgPSBuZXcgS2V5SW1wbCgyKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9BTlRJQUxJQVNfREVGQVVMVCAtIGFudGlhbGlhc2luZyByZW5kZXJpbmcgaGludCB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9BTlRJQUxJQVNfREVGQVVMVCA9IG5ldyBLZXlWYWx1ZShLRVlfQU5USUFMSUFTSU5HKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9BTlRJQUxJQVNfT04gLSBhbnRpYWxpYXNpbmcgcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfQU5USUFMSUFTX09OID0gbmV3IEtleVZhbHVlKEtFWV9BTlRJQUxJQVNJTkcpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZBTFVFX0FOVElBTElBU19PRkYgLSBhbnRpYWxpYXNpbmcgcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfQU5USUFMSUFTX09GRiA9IG5ldyBLZXlWYWx1ZShLRVlfQU5USUFMSUFTSU5HKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBLRVlfQ09MT1JfUkVOREVSSU5HIC0gY29sb3IgcmVuZGVyaW5nIGhpbnQga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgS2V5IEtFWV9DT0xPUl9SRU5ERVJJTkcgPSBuZXcgS2V5SW1wbCgzKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9DT0xPUl9SRU5ERVJfREVGQVVMVCAtIGNvbG9yIHJlbmRlcmluZyBoaW50IHZhbHVlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgT2JqZWN0IFZBTFVFX0NPTE9SX1JFTkRFUl9ERUZBVUxUID0gbmV3IEtleVZhbHVlKEtFWV9DT0xPUl9SRU5ERVJJTkcpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZBTFVFX0NPTE9SX1JFTkRFUl9TUEVFRCAtIGNvbG9yIHJlbmRlcmluZyBoaW50IHZhbHVlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgT2JqZWN0IFZBTFVFX0NPTE9SX1JFTkRFUl9TUEVFRCA9IG5ldyBLZXlWYWx1ZShLRVlfQ09MT1JfUkVOREVSSU5HKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9DT0xPUl9SRU5ERVJfUVVBTElUWSAtIGNvbG9yIHJlbmRlcmluZyBoaW50IHZhbHVlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgT2JqZWN0IFZBTFVFX0NPTE9SX1JFTkRFUl9RVUFMSVRZID0gbmV3IEtleVZhbHVlKEtFWV9DT0xPUl9SRU5ERVJJTkcpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEtFWV9ESVRIRVJJTkcgLSBkaXRoZXJpbmcgcmVuZGVyaW5nIGhpbnQga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgS2V5IEtFWV9ESVRIRVJJTkcgPSBuZXcgS2V5SW1wbCg0KTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9ESVRIRVJfREVGQVVMVCAtIGRpdGhlcmluZyByZW5kZXJpbmcgaGludCB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9ESVRIRVJfREVGQVVMVCA9IG5ldyBLZXlWYWx1ZShLRVlfRElUSEVSSU5HKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9ESVRIRVJfRElTQUJMRSAtIGRpdGhlcmluZyByZW5kZXJpbmcgaGludCB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9ESVRIRVJfRElTQUJMRSA9IG5ldyBLZXlWYWx1ZShLRVlfRElUSEVSSU5HKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9ESVRIRVJfRElTQUJMRSAtIGRpdGhlcmluZyByZW5kZXJpbmcgaGludCB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9ESVRIRVJfRU5BQkxFID0gbmV3IEtleVZhbHVlKEtFWV9ESVRIRVJJTkcpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEtFWV9GUkFDVElPTkFMTUVUUklDUyAtIGZyYWN0aW9uYWwgbWV0cmljcyByZW5kZXJpbmcgaGludAotICAgICAqIGtleS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEtleSBLRVlfRlJBQ1RJT05BTE1FVFJJQ1MgPSBuZXcgS2V5SW1wbCg1KTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9GUkFDVElPTkFMTUVUUklDU19ERUZBVUxUIC0gZnJhY3Rpb25hbCBtZXRyaWNzCi0gICAgICogcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfRlJBQ1RJT05BTE1FVFJJQ1NfREVGQVVMVCA9IG5ldyBLZXlWYWx1ZShLRVlfRlJBQ1RJT05BTE1FVFJJQ1MpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZBTFVFX0ZSQUNUSU9OQUxNRVRSSUNTX09OIC0gZnJhY3Rpb25hbCBtZXRyaWNzIHJlbmRlcmluZwotICAgICAqIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfRlJBQ1RJT05BTE1FVFJJQ1NfT04gPSBuZXcgS2V5VmFsdWUoS0VZX0ZSQUNUSU9OQUxNRVRSSUNTKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9GUkFDVElPTkFMTUVUUklDU19PRkYgLSBmcmFjdGlvbmFsIG1ldHJpY3MgcmVuZGVyaW5nCi0gICAgICogaGludCB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9GUkFDVElPTkFMTUVUUklDU19PRkYgPSBuZXcgS2V5VmFsdWUoS0VZX0ZSQUNUSU9OQUxNRVRSSUNTKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBLRVlfSU5URVJQT0xBVElPTiAtIGludGVycG9sYXRpb24gcmVuZGVyaW5nIGhpbnQga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgS2V5IEtFWV9JTlRFUlBPTEFUSU9OID0gbmV3IEtleUltcGwoNik7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVkFMVUVfSU5URVJQT0xBVElPTl9CSUNVQklDIC0gaW50ZXJwb2xhdGlvbiByZW5kZXJpbmcgaGludAotICAgICAqIHZhbHVlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgT2JqZWN0IFZBTFVFX0lOVEVSUE9MQVRJT05fQklDVUJJQyA9IG5ldyBLZXlWYWx1ZShLRVlfSU5URVJQT0xBVElPTik7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVkFMVUVfSU5URVJQT0xBVElPTl9CSUxJTkVBUiAtIGludGVycG9sYXRpb24gcmVuZGVyaW5nIGhpbnQKLSAgICAgKiB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9JTlRFUlBPTEFUSU9OX0JJTElORUFSID0gbmV3IEtleVZhbHVlKEtFWV9JTlRFUlBPTEFUSU9OKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBWQUxVRV9JTlRFUlBPTEFUSU9OX05FQVJFU1RfTkVJR0hCT1IgLSBpbnRlcnBvbGF0aW9uCi0gICAgICogcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfSU5URVJQT0xBVElPTl9ORUFSRVNUX05FSUdIQk9SID0gbmV3IEtleVZhbHVlKAotICAgICAgICAgICAgS0VZX0lOVEVSUE9MQVRJT04pOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEtFWV9SRU5ERVJJTkcgLSByZW5kZXJpbmcgaGludCBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBLZXkgS0VZX1JFTkRFUklORyA9IG5ldyBLZXlJbXBsKDcpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZBTFVFX1JFTkRFUl9ERUZBVUxUIC0gcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfUkVOREVSX0RFRkFVTFQgPSBuZXcgS2V5VmFsdWUoS0VZX1JFTkRFUklORyk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVkFMVUVfUkVOREVSX1NQRUVEIC0gcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfUkVOREVSX1NQRUVEID0gbmV3IEtleVZhbHVlKEtFWV9SRU5ERVJJTkcpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZBTFVFX1JFTkRFUl9RVUFMSVRZIC0gcmVuZGVyaW5nIGhpbnQgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBPYmplY3QgVkFMVUVfUkVOREVSX1FVQUxJVFkgPSBuZXcgS2V5VmFsdWUoS0VZX1JFTkRFUklORyk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgS0VZX1NUUk9LRV9DT05UUk9MIC0gc3Ryb2tlIGNvbnRyb2wgaGludCBrZXkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBLZXkgS0VZX1NUUk9LRV9DT05UUk9MID0gbmV3IEtleUltcGwoOCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVkFMVUVfU1RST0tFX0RFRkFVTFQgLSBzdHJva2UgaGludCB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9TVFJPS0VfREVGQVVMVCA9IG5ldyBLZXlWYWx1ZShLRVlfU1RST0tFX0NPTlRST0wpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZBTFVFX1NUUk9LRV9OT1JNQUxJWkUgLSBzdHJva2UgaGludCB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9TVFJPS0VfTk9STUFMSVpFID0gbmV3IEtleVZhbHVlKEtFWV9TVFJPS0VfQ09OVFJPTCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVkFMVUVfU1RST0tFX1BVUkUgLSBzdHJva2UgaGludCB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9TVFJPS0VfUFVSRSA9IG5ldyBLZXlWYWx1ZShLRVlfU1RST0tFX0NPTlRST0wpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEtFWV9URVhUX0FOVElBTElBU0lORyAtIHRleHQgYW50aWFsaWFzaW5nIGhpbnQga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgS2V5IEtFWV9URVhUX0FOVElBTElBU0lORyA9IG5ldyBLZXlJbXBsKDkpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZBTFVFX1RFWFRfQU5USUFMSUFTX0RFRkFVTFQgLSB0ZXh0IGFudGlhbGlhc2luZyBoaW50IGtleS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9URVhUX0FOVElBTElBU19ERUZBVUxUID0gbmV3IEtleVZhbHVlKEtFWV9URVhUX0FOVElBTElBU0lORyk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVkFMVUVfVEVYVF9BTlRJQUxJQVNfT04gLSB0ZXh0IGFudGlhbGlhc2luZyBoaW50IGtleS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIE9iamVjdCBWQUxVRV9URVhUX0FOVElBTElBU19PTiA9IG5ldyBLZXlWYWx1ZShLRVlfVEVYVF9BTlRJQUxJQVNJTkcpOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFZBTFVFX1RFWFRfQU5USUFMSUFTX09GRiAtIHRleHQgYW50aWFsaWFzaW5nIGhpbnQga2V5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgT2JqZWN0IFZBTFVFX1RFWFRfQU5USUFMSUFTX09GRiA9IG5ldyBLZXlWYWx1ZShLRVlfVEVYVF9BTlRJQUxJQVNJTkcpOwotCi0gICAgLyoqIFRoZSBtYXAuICovCi0gICAgcHJpdmF0ZSBIYXNoTWFwPE9iamVjdCwgT2JqZWN0PiBtYXAgPSBuZXcgSGFzaE1hcDxPYmplY3QsIE9iamVjdD4oKTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyByZW5kZXJpbmcgaGludHMgb2JqZWN0IGZyb20gc3BlY2lmaWVkIE1hcCBvYmplY3Qgd2l0aAotICAgICAqIGRlZmluZWQga2V5L3ZhbHVlIHBhaXJzIG9yIG51bGwgZm9yIGVtcHR5IFJlbmRlcmluZ0hpbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBtYXAKLSAgICAgKiAgICAgICAgICAgIHRoZSBNYXAgb2JqZWN0IHdpdGggZGVmaW5lZCBrZXkvdmFsdWUgcGFpcnMgb3IgbnVsbCBmb3IgZW1wdHkKLSAgICAgKiAgICAgICAgICAgIFJlbmRlcmluZ0hpbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBSZW5kZXJpbmdIaW50cyhNYXA8S2V5LCA/PiBtYXApIHsKLSAgICAgICAgc3VwZXIoKTsKLSAgICAgICAgaWYgKG1hcCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBwdXRBbGwobWFwKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyByZW5kZXJpbmcgaGludHMgb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCBrZXkvdmFsdWUKLSAgICAgKiBwYWlyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBrZXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBrZXkgb2YgaGludCBwcm9wZXJ0eS4KLSAgICAgKiBAcGFyYW0gdmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSB2YWx1ZSBvZiBoaW50IHByb3BlcnR5LgotICAgICAqLwotICAgIHB1YmxpYyBSZW5kZXJpbmdIaW50cyhLZXkga2V5LCBPYmplY3QgdmFsdWUpIHsKLSAgICAgICAgc3VwZXIoKTsKLSAgICAgICAgcHV0KGtleSwgdmFsdWUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgdGhlIHByb3BlcnRpZXMgcmVwcmVzZW50ZWQgYnkga2V5L3ZhbHVlIHBhaXJzIGZyb20gdGhlIHNwZWNpZmllZAotICAgICAqIFJlbmRlcmluZ0hpbnRzIG9iamVjdCB0byBjdXJyZW50IG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaGludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJpbmdIaW50cyB0byBiZSBhZGRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGQoUmVuZGVyaW5nSGludHMgaGludHMpIHsKLSAgICAgICAgbWFwLnB1dEFsbChoaW50cy5tYXApOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFB1dHMgdGhlIHNwZWNpZmllZCB2YWx1ZSB0byB0aGUgc3BlY2lmaWVkIGtleS4gTmVpdGhlciB0aGUga2V5IG5vciB0aGUKLSAgICAgKiB2YWx1ZSBjYW4gYmUgbnVsbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0ga2V5Ci0gICAgICogICAgICAgICAgICB0aGUgcmVuZGVyaW5nIGhpbnQga2V5LgotICAgICAqIEBwYXJhbSB2YWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIHJlbmRlcmluZyBoaW50IHZhbHVlLgotICAgICAqIEByZXR1cm4gdGhlIHByZXZpb3VzIHJlbmRlcmluZyBoaW50IHZhbHVlIGFzc2lnbmVkIHRvIHRoZSBrZXkgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IHB1dChPYmplY3Qga2V5LCBPYmplY3QgdmFsdWUpIHsKLSAgICAgICAgaWYgKCEoKEtleSlrZXkpLmlzQ29tcGF0aWJsZVZhbHVlKHZhbHVlKSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG1hcC5wdXQoa2V5LCB2YWx1ZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgc3BlY2lmaWVkIGtleSBhbmQgY29ycmVzcG9uZGluZyB2YWx1ZSBmcm9tIHRoZSBSZW5kZXJpbmdIaW50cwotICAgICAqIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0ga2V5Ci0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGhpbnQga2V5IHRvIGJlIHJlbW92ZWQuCi0gICAgICogQHJldHVybiB0aGUgb2JqZWN0IG9mIHByZXZpb3VzIHJlbmRlcmluZyBoaW50IHZhbHVlIHdoaWNoIGlzIGFzc2lnbmVkIHRvCi0gICAgICogICAgICAgICB0aGUgc3BlY2lmaWVkIGtleSwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IHJlbW92ZShPYmplY3Qga2V5KSB7Ci0gICAgICAgIHJldHVybiBtYXAucmVtb3ZlKGtleSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdmFsdWUgYXNzaWduZWQgdG8gdGhlIHNwZWNpZmllZCBrZXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGtleQotICAgICAqICAgICAgICAgICAgdGhlIHJlbmRlcmluZyBoaW50IGtleS4KLSAgICAgKiBAcmV0dXJuIHRoZSBvYmplY3QgYXNzaWduZWQgdG8gdGhlIHNwZWNpZmllZCBrZXkuCi0gICAgICovCi0gICAgcHVibGljIE9iamVjdCBnZXQoT2JqZWN0IGtleSkgewotICAgICAgICByZXR1cm4gbWFwLmdldChrZXkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBzZXQgb2YgcmVuZGVyaW5nIGhpbnRzIGtleXMgZm9yIGN1cnJlbnQgUmVuZGVyaW5nSGludHMgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNldCBvZiByZW5kZXJpbmcgaGludHMga2V5cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgU2V0PE9iamVjdD4ga2V5U2V0KCkgewotICAgICAgICByZXR1cm4gbWFwLmtleVNldCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBzZXQgb2YgTWFwLkVudHJ5IG9iamVjdHMgd2hpY2ggY29udGFpbiBjdXJyZW50IFJlbmRlcmluZ0hpbnQKLSAgICAgKiBrZXkvdmFsdWUgcGFpcnMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYSBzZXQgb2YgbWFwcGVkIFJlbmRlcmluZ0hpbnQga2V5L3ZhbHVlIHBhaXJzLgotICAgICAqLwotICAgIHB1YmxpYyBTZXQ8TWFwLkVudHJ5PE9iamVjdCwgT2JqZWN0Pj4gZW50cnlTZXQoKSB7Ci0gICAgICAgIHJldHVybiBtYXAuZW50cnlTZXQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQdXRzIGFsbCBvZiB0aGUgcHJlZmVyZW5jZXMgZnJvbSB0aGUgc3BlY2lmaWVkIE1hcCBpbnRvIHRoZSBjdXJyZW50Ci0gICAgICogUmVuZGVyaW5nSGludHMgb2JqZWN0LiBUaGVzZSBtYXBwaW5ncyByZXBsYWNlIGFsbCBleGlzdGluZyBwcmVmZXJlbmNlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBNYXAgb2YgcHJlZmVyZW5jZXMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcHV0QWxsKE1hcDw/LCA/PiBtKSB7Ci0gICAgICAgIGlmIChtIGluc3RhbmNlb2YgUmVuZGVyaW5nSGludHMpIHsKLSAgICAgICAgICAgIG1hcC5wdXRBbGwoKChSZW5kZXJpbmdIaW50cyltKS5tYXApOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgU2V0PD8+IGVudHJpZXMgPSBtLmVudHJ5U2V0KCk7Ci0KLSAgICAgICAgICAgIGlmIChlbnRyaWVzICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBJdGVyYXRvcjw/PiBpdCA9IGVudHJpZXMuaXRlcmF0b3IoKTsKLSAgICAgICAgICAgICAgICB3aGlsZSAoaXQuaGFzTmV4dCgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIE1hcC5FbnRyeTw/LCA/PiBlbnRyeSA9IChNYXAuRW50cnk8PywgPz4paXQubmV4dCgpOwotICAgICAgICAgICAgICAgICAgICBLZXkga2V5ID0gKEtleSllbnRyeS5nZXRLZXkoKTsKLSAgICAgICAgICAgICAgICAgICAgT2JqZWN0IHZhbCA9IGVudHJ5LmdldFZhbHVlKCk7Ci0gICAgICAgICAgICAgICAgICAgIHB1dChrZXksIHZhbCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIENvbGxlY3Rpb24gb2YgdmFsdWVzIGNvbnRhaW5lZCBpbiBjdXJyZW50IFJlbmRlcmluZ0hpbnRzCi0gICAgICogb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIENvbGxlY3Rpb24gb2YgUmVuZGVyaW5nSGludHMncyB2YWx1ZXMuCi0gICAgICovCi0gICAgcHVibGljIENvbGxlY3Rpb248T2JqZWN0PiB2YWx1ZXMoKSB7Ci0gICAgICAgIHJldHVybiBtYXAudmFsdWVzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IGN1cnJlbnQgUmVuZGVyaW5nSGludHMgb2JqZWN0IGNvbnRhaW5zIGF0IGxlYXN0IG9uZQotICAgICAqIHRoZSB2YWx1ZSB3aGljaCBpcyBlcXVhbCB0byB0aGUgc3BlY2lmaWVkIE9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgT2JqZWN0LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCBvYmplY3QgaXMgYXNzaWduZWQgdG8gYXQgbGVhc3Qgb25lCi0gICAgICogICAgICAgICBSZW5kZXJpbmdIaW50J3Mga2V5LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnNWYWx1ZShPYmplY3QgdmFsdWUpIHsKLSAgICAgICAgcmV0dXJuIG1hcC5jb250YWluc1ZhbHVlKHZhbHVlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgY3VycmVudCBSZW5kZXJpbmdIaW50cyBvYmplY3QgY29udGFpbnMgdGhlIGtleQotICAgICAqIHdoaWNoIGlzIGVxdWFsIHRvIHRoZSBzcGVjaWZpZWQgT2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBrZXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgT2JqZWN0LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIFJlbmRlcmluZ0hpbnRzIG9iamVjdCBjb250YWlucyB0aGUgc3BlY2lmaWVkIE9iamVjdAotICAgICAqICAgICAgICAgYXMgYSBrZXksIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWluc0tleShPYmplY3Qga2V5KSB7Ci0gICAgICAgIGlmIChrZXkgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbWFwLmNvbnRhaW5zS2V5KGtleSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IHRoZSBSZW5kZXJpbmdIaW50cyBvYmplY3QgY29udGFpbnMgYW55IGtleS92YWx1ZQotICAgICAqIHBhaXJzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIFJlbmRlcmluZ0hpbnRzIG9iamVjdCBpcyBlbXB0eSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzRW1wdHkoKSB7Ci0gICAgICAgIHJldHVybiBtYXAuaXNFbXB0eSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENsZWFycyB0aGUgUmVuZGVyaW5nSGludHMgb2YgYWxsIGtleS92YWx1ZSBwYWlycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBjbGVhcigpIHsKLSAgICAgICAgbWFwLmNsZWFyKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbnVtYmVyIG9mIGtleS92YWx1ZSBwYWlycyBpbiB0aGUgUmVuZGVyaW5nSGludHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIGtleS92YWx1ZSBwYWlycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHNpemUoKSB7Ci0gICAgICAgIHJldHVybiBtYXAuc2l6ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIHRoZSBSZW5kZXJpbmdIaW50cyBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbwotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBPYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgT2JqZWN0IGlzIGEgTWFwIHdob3NlIGtleS92YWx1ZSBwYWlycyBtYXRjaCB0aGlzCi0gICAgICogICAgICAgICBSZW5kZXJpbmdIaW50cycga2V5L3ZhbHVlIHBhaXJzLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvKSB7Ci0gICAgICAgIGlmICghKG8gaW5zdGFuY2VvZiBNYXApKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBNYXA8PywgPz4gbSA9IChNYXA8PywgPz4pbzsKLSAgICAgICAgU2V0PD8+IGtleXMgPSBrZXlTZXQoKTsKLSAgICAgICAgaWYgKCFrZXlzLmVxdWFscyhtLmtleVNldCgpKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgSXRlcmF0b3I8Pz4gaXQgPSBrZXlzLml0ZXJhdG9yKCk7Ci0gICAgICAgIHdoaWxlIChpdC5oYXNOZXh0KCkpIHsKLSAgICAgICAgICAgIEtleSBrZXkgPSAoS2V5KWl0Lm5leHQoKTsKLSAgICAgICAgICAgIE9iamVjdCB2MSA9IGdldChrZXkpOwotICAgICAgICAgICAgT2JqZWN0IHYyID0gbS5nZXQoa2V5KTsKLSAgICAgICAgICAgIGlmICghKHYxID09IG51bGwgPyB2MiA9PSBudWxsIDogdjEuZXF1YWxzKHYyKSkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgaGFzaCBjb2RlIGZvciB0aGlzIFJlbmRlcmluZ0hpbnRzIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBoYXNoIGNvZGUgZm9yIHRoaXMgUmVuZGVyaW5nSGludHMgb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgaGFzaENvZGUoKSB7Ci0gICAgICAgIHJldHVybiBtYXAuaGFzaENvZGUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBjbG9uZSBvZiB0aGUgUmVuZGVyaW5nSGludHMgb2JqZWN0IHdpdGggdGhlIHNhbWUgY29udGVudHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY2xvbmUgb2YgdGhlIFJlbmRlcmluZ0hpbnRzIGluc3RhbmNlLgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bmNoZWNrZWQiKQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgY2xvbmUoKSB7Ci0gICAgICAgIFJlbmRlcmluZ0hpbnRzIGNsb25lID0gbmV3IFJlbmRlcmluZ0hpbnRzKG51bGwpOwotICAgICAgICBjbG9uZS5tYXAgPSAoSGFzaE1hcDxPYmplY3QsIE9iamVjdD4pdGhpcy5tYXAuY2xvbmUoKTsKLSAgICAgICAgcmV0dXJuIGNsb25lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGUgUmVuZGVyaW5nSGludHMgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFN0cmluZyBvYmplY3Qgd2hpY2ggcmVwcmVzZW50cyBSZW5kZXJpbmdIaW50cyBvYmplY3QuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsKLSAgICAgICAgcmV0dXJuICJSZW5kZXJpbmdIaW50c1siICsgbWFwLnRvU3RyaW5nKCkgKyAiXSI7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoZSBSZW5kZXJpbmdIaW50cy5LZXkgY2xhc3MgaXMgYWJzdHJhY3QgYW5kIGRlZmluZXMgYSBiYXNlIHR5cGUgZm9yIGFsbAotICAgICAqIFJlbmRlcmluZ0hpbnRzIGtleXMuCi0gICAgICogCi0gICAgICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHN0YXRpYyBjbGFzcyBLZXkgewotCi0gICAgICAgIC8qKiBUaGUga2V5LiAqLwotICAgICAgICBwcml2YXRlIGZpbmFsIGludCBrZXk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBrZXkgd2l0aCB1bmlxdWUgaW50ZWdlciBpZGVudGlmaWVyLiBObyB0d28gb2JqZWN0cwotICAgICAgICAgKiBvZiB0aGUgc2FtZSBzdWJjbGFzcyB3aXRoIHRoZSBzYW1lIGludGVnZXIga2V5IGNhbiBiZSBpbnN0YW50aWF0ZWQuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0ga2V5Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHVuaXF1ZSBrZXkuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgS2V5KGludCBrZXkpIHsKLSAgICAgICAgICAgIHRoaXMua2V5ID0ga2V5OwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENvbXBhcmVzIHRoZSBLZXkgb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCBvYmplY3QuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gbwotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgT2JqZWN0IHRvIGJlIGNvbXBhcmVkLgotICAgICAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBLZXkgaXMgZXF1YWwgdG8gdGhlIHNwZWNpZmllZCBvYmplY3QsIGZhbHNlCi0gICAgICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAgICAgKi8KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBmaW5hbCBib29sZWFuIGVxdWFscyhPYmplY3QgbykgewotICAgICAgICAgICAgcmV0dXJuIHRoaXMgPT0gbzsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBSZXR1cm5zIHRoZSBoYXNoIGNvZGUgZm9yIHRoaXMgS2V5IG9iamVjdC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdGhlIGhhc2ggY29kZSBmb3IgdGhpcyBLZXkgb2JqZWN0LgotICAgICAgICAgKi8KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBmaW5hbCBpbnQgaGFzaENvZGUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gU3lzdGVtLmlkZW50aXR5SGFzaENvZGUodGhpcyk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyBpbnRlZ2VyIHVuaXF1ZSBrZXkgd2l0aCB3aGljaCB0aGlzIEtleSBvYmplY3QgaGFzIGJlZW4KLSAgICAgICAgICogaW5zdGFudGlhdGVkLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgaW50ZWdlciB1bmlxdWUga2V5IHdpdGggd2hpY2ggdGhpcyBLZXkgb2JqZWN0IGhhcyBiZWVuCi0gICAgICAgICAqICAgICAgICAgaW5zdGFudGlhdGVkLgotICAgICAgICAgKi8KLSAgICAgICAgcHJvdGVjdGVkIGZpbmFsIGludCBpbnRLZXkoKSB7Ci0gICAgICAgICAgICByZXR1cm4ga2V5OwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCBzcGVjaWZpZWQgdmFsdWUgaXMgY29tcGF0aWJsZSB3aXRoIHRoZSBLZXkuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gdmFsCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIE9iamVjdC4KLSAgICAgICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIHZhbHVlIGlzIGNvbXBhdGlibGUgd2l0aCB0aGUgS2V5LAotICAgICAgICAgKiAgICAgICAgIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBhYnN0cmFjdCBib29sZWFuIGlzQ29tcGF0aWJsZVZhbHVlKE9iamVjdCB2YWwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByaXZhdGUgaW1wbGVtZW50YXRpb24gb2YgS2V5IGNsYXNzLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGNsYXNzIEtleUltcGwgZXh0ZW5kcyBLZXkgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcga2V5IGltcGxlbWVudGF0aW9uLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGtleQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBrZXkuCi0gICAgICAgICAqLwotICAgICAgICBwcm90ZWN0ZWQgS2V5SW1wbChpbnQga2V5KSB7Ci0gICAgICAgICAgICBzdXBlcihrZXkpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzQ29tcGF0aWJsZVZhbHVlKE9iamVjdCB2YWwpIHsKLSAgICAgICAgICAgIGlmICghKHZhbCBpbnN0YW5jZW9mIEtleVZhbHVlKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuICgoS2V5VmFsdWUpdmFsKS5rZXkgPT0gdGhpczsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByaXZhdGUgY2xhc3MgS2V5VmFsdWUgaXMgdXNlZCBhcyB2YWx1ZSBmb3IgS2V5IGNsYXNzIGluc3RhbmNlLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGNsYXNzIEtleVZhbHVlIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGtleS4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgZmluYWwgS2V5IGtleTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGtleSB2YWx1ZS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBrZXkKLSAgICAgICAgICogICAgICAgICAgICB0aGUga2V5LgotICAgICAgICAgKi8KLSAgICAgICAgcHJvdGVjdGVkIEtleVZhbHVlKEtleSBrZXkpIHsKLSAgICAgICAgICAgIHRoaXMua2V5ID0ga2V5OwotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L1NoYXBlLmphdmEgYi9hd3QvamF2YS9hd3QvU2hhcGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNTliYzYyMy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvU2hhcGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE2MiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBhdGhJdGVyYXRvcjsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBvaW50MkQ7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLQotLyoqCi0gKiBUaGUgU2hhcGUgaW50ZXJmYWNlIGRlZmluZXMgYSBnZW9tZXRyaWMgc2hhcGUgZGVmaW5lZCBieSBhIGJvdW5kYXJ5IChvdXRsaW5lKQotICogcGF0aC4gVGhlIHBhdGggb3V0bGluZSBjYW4gYmUgYWNjZXNzZWQgdGhyb3VnaCBhIFBhdGhJdGVyYXRvciBvYmplY3QuIFRoZQotICogU2hhcGUgaW50ZXJmYWNlIHByb3ZpZGVzIG1ldGhvZHMgZm9yIG9idGFpbmluZyB0aGUgYm91bmRpbmcgYm94ICh3aGljaCBpcyB0aGUKLSAqIHNtYWxsZXN0IHJlY3RhbmdsZSBjb250YWluaW5nIHRoZSBzaGFwZSBhbmQgZm9yIG9idGFpbmluZyBhIFBhdGhJdGVyYXRvcgotICogb2JqZWN0IGZvciBjdXJyZW50IFNoYXBlLCBhcyB3ZWxsIGFzIHV0aWxpdHkgbWV0aG9kcyB3aGljaCBkZXRlcm1pbmUgaWYgdGhlCi0gKiBTaGFwZSBjb250YWlucyBvciBpbnRlcnNlY3RzIGEgUmVjdGFuZ2xlIG9yIGNvbnRhaW5zIGEgUG9pbnQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIFNoYXBlIHsKLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGUgcG9pbnQgd2l0aCBzcGVjaWZpZWQgY29vcmRpbmF0ZXMgbGllcyBpbnNpZGUKLSAgICAgKiB0aGUgU2hhcGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIGNvb3JkaW5hdGVzIGxpZSBpbnNpZGUgdGhlIFNoYXBlLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSB4LCBkb3VibGUgeSk7Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIHJlY3RhbmdsZSB3aXRoIHNwZWNpZmllZCBbeCwgeSwgd2lkdGgsIGhlaWdodF0KLSAgICAgKiBwYXJhbWV0ZXJzIGxpZXMgaW5zaWRlIHRoZSBTaGFwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggZG91YmxlIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBkb3VibGUgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ2xlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCByZWN0YW5nbGUgbGllcyBpbnNpZGUgdGhlIFNoYXBlLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHcsIGRvdWJsZSBoKTsKLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGUgc3BlY2lmaWVkIFBvaW50MkQgbGllcyBpbnNpZGUgdGhlIFNoYXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwb2ludAotICAgICAqICAgICAgICAgICAgdGhlIFBvaW50MkQgb2JqZWN0LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCBQb2ludDJEIGxpZXMgaW5zaWRlIHRoZSBTaGFwZSwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhQb2ludDJEIHBvaW50KTsKLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGUgc3BlY2lmaWVkIHJlY3RhbmdsZSBsaWVzIGluc2lkZSB0aGUgU2hhcGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZWN0YW5nbGUyRCBvYmplY3QuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIHJlY3RhbmdsZSBsaWVzIGluc2lkZSB0aGUgU2hhcGUsIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoUmVjdGFuZ2xlMkQgcik7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZGluZyByZWN0YW5nbGUgb2YgdGhlIFNoYXBlLiBUaGUgYm91bmRpbmcgcmVjdGFuZ2xlIGlzIHRoZQotICAgICAqIHNtYWxsZXN0IHJlY3RhbmdsZSB3aGljaCBjb250YWlucyB0aGUgU2hhcGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlIG9mIHRoZSBTaGFwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldEJvdW5kcygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgUmVjdGFuZ2xlMkQgd2hpY2ggcmVwcmVzZW50cyBTaGFwZSBib3VuZHMuIFRoZSBib3VuZGluZwotICAgICAqIHJlY3RhbmdsZSBpcyB0aGUgc21hbGxlc3QgcmVjdGFuZ2xlIHdoaWNoIGNvbnRhaW5zIHRoZSBTaGFwZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBib3VuZGluZyByZWN0YW5nbGUgb2YgdGhlIFNoYXBlLgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgUGF0aEl0ZXJhdG9yIG9iamVjdCBvZiB0aGUgU2hhcGUgd2hpY2ggcHJvdmlkZXMgYWNjZXNzIHRvIHRoZQotICAgICAqIHNoYXBlJ3MgYm91bmRhcnkgbW9kaWZpZWQgYnkgdGhlIHNwZWNpZmllZCBBZmZpbmVUcmFuc2Zvcm0uCi0gICAgICogCi0gICAgICogQHBhcmFtIGF0Ci0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIEFmZmluZVRyYW5zZm9ybSBvYmplY3Qgb3IgbnVsbC4KLSAgICAgKiBAcmV0dXJuIFBhdGhJdGVyYXRvciBvYmplY3QgZm9yIHRoZSBTaGFwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gYXQpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgUGF0aEl0ZXJhdG9yIG9iamVjdCBvZiB0aGUgU2hhcGUgd2hpY2ggcHJvdmlkZXMgYWNjZXNzIHRvIHRoZQotICAgICAqIGNvb3JkaW5hdGVzIG9mIHRoZSBzaGFwZXMgYm91bmRhcnkgbW9kaWZpZWQgYnkgdGhlIHNwZWNpZmllZAotICAgICAqIEFmZmluZVRyYW5zZm9ybS4gVGhlIGZsYXRuZXNzIHBhcmFtZXRlciBkZWZpbmVzIHRoZSBhbW91bnQgb2Ygc3ViZGl2aXNpb24KLSAgICAgKiBvZiB0aGUgY3VydmVkIHNlZ21lbnRzIGFuZCBzcGVjaWZpZXMgdGhlIG1heGltdW0gZGlzdGFuY2Ugd2hpY2ggZXZlcnkKLSAgICAgKiBwb2ludCBvbiB0aGUgdW5mbGF0dGVuZWQgdHJhbnNmb3JtZWQgY3VydmUgY2FuIGRldmlhdGUgZnJvbSB0aGUgcmV0dXJuZWQKLSAgICAgKiBmbGF0dGVuZWQgcGF0aCBzZWdtZW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgQWZmaW5lVHJhbnNmb3JtIG9iamVjdCBvciBudWxsLgotICAgICAqIEBwYXJhbSBmbGF0bmVzcwotICAgICAqICAgICAgICAgICAgdGhlIG1heGltdW0gbnVtYmVyIG9mIHRoZSBjb250cm9sIHBvaW50cyBmb3IgYSBnaXZlbiBjdXJ2ZQotICAgICAqICAgICAgICAgICAgd2hpY2ggdmFyaWVzIGZyb20gY29saW5lYXIgYmVmb3JlIGEgc3ViZGl2aWRlZCBjdXJ2ZSBpcwotICAgICAqICAgICAgICAgICAgcmVwbGFjZWQgYnkgYSBzdHJhaWdodCBsaW5lIGNvbm5lY3RpbmcgdGhlIGVuZHBvaW50cy4KLSAgICAgKiBAcmV0dXJuIFBhdGhJdGVyYXRvciBvYmplY3QgZm9yIHRoZSBTaGFwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gYXQsIGRvdWJsZSBmbGF0bmVzcyk7Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIGludGVyaW9yIG9mIHJlY3Rhbmd1bGFyIHNwZWNpZmllZCBieSBbeCwgeSwKLSAgICAgKiB3aWR0aCwgaGVpZ2h0XSBwYXJhbWV0ZXJzIGludGVyc2VjdHMgdGhlIGludGVyaW9yIG9mIHRoZSBTaGFwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggZG91YmxlIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBkb3VibGUgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ2xlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHJlY3RhbmdsZSBzcGVjaWZpZWQgYnkgW3gsIHksIHdpZHRoLCBoZWlnaHRdCi0gICAgICogICAgICAgICBwYXJhbWV0ZXJzIGludGVyc2VjdHMgdGhlIGludGVyaW9yIG9mIHRoZSBTaGFwZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHMoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgdywgZG91YmxlIGgpOwotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IHRoZSBpbnRlcmlvciBvZiByZWN0YW5nbGUgc3BlY2lmaWVkIGJ5IFJlY3RhbmdsZTJECi0gICAgICogb2JqZWN0IGludGVyc2VjdHMgdGhlIGludGVyaW9yIG9mIHRoZSBTaGFwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcgotICAgICAqICAgICAgICAgICAgdGhlIFJlY3RhbmdsZTJEIG9iamVjdC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBSZWN0YW5nbGUyRCBpbnRlcnNlY3RzIHRoZSBpbnRlcmlvciBvZiB0aGUgU2hhcGUsCi0gICAgICogICAgICAgICBvdGhlcndpc2UgZmFsc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0cyhSZWN0YW5nbGUyRCByKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9TdHJva2UuamF2YSBiL2F3dC9qYXZhL2F3dC9TdHJva2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNmQxN2EyMy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvU3Ryb2tlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1MCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLS8qKgotICogVGhlIFN0cm9rZSBpbnRlcmZhY2UgZ2l2ZXMgYSBwZW4gc3R5bGUgdG8gYmUgdXNlZCBieSB0aGUgR3JhcGhpY3MyRAotICogaW50ZXJmYWNlLiBJdCBwcm92aWRlcyBhIG1lYW5zIGZvciBnZXR0aW5nIGEgc3Ryb2tlZCB2ZXJzaW9uIG9mIGEgc2hhcGUsCi0gKiB3aGljaCBpcyB0aGUgdmVyc2lvbiB0aGF0IGlzIHN1aXRhYmxlIGZvciBkcmF3aW5nIHZpYSB0aGUgR3JhcGhpY3MyRAotICogaW50ZXJmYWNlLiBTdHJva2luZyBhIHNoYXBlIGdpdmVzIHRoZSBzaGFwZSdzIG91dGxpbmUgYSB3aWR0aCBvciBkcmF3aW5nCi0gKiBzdHlsZS4KLSAqIDxwPgotICogVGhlIERyYXcgbWV0aG9kcyBmcm9tIEdyYXBoaWNzMkQgaW50ZXJmYWNlIHNob3VsZCB1c2UgdGhlIFN0cm9rZSBvYmplY3QgZm9yCi0gKiByZW5kZXJpbmcgdGhlIHNoYXBlJ3Mgb3V0bGluZS4gVGhlIHN0cm9rZSBzaG91bGQgYmUgc2V0IGJ5Ci0gKiBzZXRTdHJva2UoamF2YS5hd3QuU3Ryb2tlKSBtZXRob2Qgb2YgdGhlIEdyYXBoaWNzMkQgaW50ZXJmYWNlLgotICogCi0gKiBAc2VlIGphdmEuYXd0LkdyYXBoaWNzMkQjc2V0U3Ryb2tlKGphdmEuYXd0LlN0cm9rZSkKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIFN0cm9rZSB7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBzdHJva2VkIHNoYXBlLCB3aGljaCBpcyB0aGUgdmVyc2lvbiB0aGF0IGlzIHN1aXRhYmxlIGZvcgotICAgICAqIGRyYXdpbmcgdmlhIHRoZSBHcmFwaGljczJEIGludGVyZmFjZS4gU3Ryb2tpbmcgYSBzaGFwZSBnaXZlcyB0aGUgc2hhcGUncwotICAgICAqIG91dGxpbmUgYSB3aWR0aCBvciBkcmF3aW5nIHN0eWxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgb3JpZ2luYWwgc2hhcGUuCi0gICAgICogQHJldHVybiB0aGUgc3Ryb2tlZCBzaGFwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgU2hhcGUgY3JlYXRlU3Ryb2tlZFNoYXBlKFNoYXBlIHApOwotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L1Rvb2xraXQuamF2YSBiL2F3dC9qYXZhL2F3dC9Ub29sa2l0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGUzOGQ1MjQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L1Rvb2xraXQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE0NDQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEuYXd0LmV2ZW50LkFXVEV2ZW50TGlzdGVuZXI7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuQVdURXZlbnRMaXN0ZW5lclByb3h5OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LklucHV0RXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuaW0uSW5wdXRNZXRob2RIaWdobGlnaHQ7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbWFnZU9ic2VydmVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkltYWdlUHJvZHVjZXI7Ci1pbXBvcnQgamF2YS5hd3QucGVlci5Gb250UGVlcjsKLWltcG9ydCBqYXZhLmJlYW5zLlByb3BlcnR5Q2hhbmdlTGlzdGVuZXI7Ci1pbXBvcnQgamF2YS5iZWFucy5Qcm9wZXJ0eUNoYW5nZVN1cHBvcnQ7Ci0KLWltcG9ydCBqYXZhLmxhbmcucmVmbGVjdC5JbnZvY2F0aW9uVGFyZ2V0RXhjZXB0aW9uOwotaW1wb3J0IGphdmEubmV0LlVSTDsKLWltcG9ydCBqYXZhLnNlY3VyaXR5LkFjY2Vzc0NvbnRyb2xsZXI7Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5Qcml2aWxlZ2VkQWN0aW9uOwotaW1wb3J0IGphdmEudXRpbC5Db2xsZWN0aW9uczsKLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaE1hcDsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaFNldDsKLWltcG9ydCBqYXZhLnV0aWwuSXRlcmF0b3I7Ci1pbXBvcnQgamF2YS51dGlsLk1hcDsKLWltcG9ydCBqYXZhLnV0aWwuTWlzc2luZ1Jlc291cmNlRXhjZXB0aW9uOwotaW1wb3J0IGphdmEudXRpbC5Qcm9wZXJ0aWVzOwotaW1wb3J0IGphdmEudXRpbC5SZXNvdXJjZUJ1bmRsZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuQ2hvaWNlU3R5bGU7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5Db21wb25lbnRJbnRlcm5hbHM7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5Db250ZXh0U3RvcmFnZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LlJlYWRPbmx5SXRlcmF0b3I7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuQ3JlYXRpb25QYXJhbXM7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuR3JhcGhpY3NGYWN0b3J5OwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLk5hdGl2ZUN1cnNvcjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLk5hdGl2ZUV2ZW50UXVldWU7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuTmF0aXZlRXZlbnRUaHJlYWQ7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuU2h1dGRvd25XYXRjaGRvZzsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5TeW5jaHJvbml6ZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuV1RLOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuTm90SW1wbGVtZW50ZWRFeGNlcHRpb247Ci0KLS8qKgotICogVGhlIFRvb2xraXQgY2xhc3MgaXMgdGhlIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBwbGF0Zm9ybS1zcGVjaWZpYyBBYnN0cmFjdAotICogV2luZG93IFRvb2xraXQgaW1wbGVtZW50YXRpb24uIFRvb2xraXQncyBzdWJjbGFzc2VzIGFyZSB1c2VkIHRvIGJpbmQgdGhlCi0gKiB2YXJpb3VzIGNvbXBvbmVudHMgdG8gcGFydGljdWxhciBuYXRpdmUgdG9vbGtpdCBpbXBsZW1lbnRhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgVG9vbGtpdCB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUkVDT1VSQ0VfUEFUSC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgUkVDT1VSQ0VfUEFUSCA9ICJvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LnJlc291cmNlcy5BV1RQcm9wZXJ0aWVzIjsgLy8kTk9OLU5MUy0xJAotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IHByb3BlcnRpZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgUmVzb3VyY2VCdW5kbGUgcHJvcGVydGllcyA9IGxvYWRSZXNvdXJjZXMoUkVDT1VSQ0VfUEFUSCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGlzcGF0Y2hlci4KLSAgICAgKi8KLSAgICBEaXNwYXRjaGVyIGRpc3BhdGNoZXI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc3lzdGVtIGV2ZW50IHF1ZXVlIGNvcmUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBFdmVudFF1ZXVlQ29yZSBzeXN0ZW1FdmVudFF1ZXVlQ29yZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBkaXNwYXRjaCB0aHJlYWQuCi0gICAgICovCi0gICAgRXZlbnREaXNwYXRjaFRocmVhZCBkaXNwYXRjaFRocmVhZDsKLQotICAgIC8qKgotICAgICAqIFRoZSBuYXRpdmUgdGhyZWFkLgotICAgICAqLwotICAgIE5hdGl2ZUV2ZW50VGhyZWFkIG5hdGl2ZVRocmVhZDsKLQotICAgIC8qKgotICAgICAqIFRoZSBBV1QgZXZlbnRzIG1hbmFnZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEFXVEV2ZW50c01hbmFnZXIgYXd0RXZlbnRzTWFuYWdlcjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBBV1RUcmVlTG9jay4KLSAgICAgKi8KLSAgICBwcml2YXRlIGNsYXNzIEFXVFRyZWVMb2NrIHsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQVdUIHRyZWUgbG9jay4KLSAgICAgKi8KLSAgICBmaW5hbCBPYmplY3QgYXd0VHJlZUxvY2sgPSBuZXcgQVdUVHJlZUxvY2soKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBzeW5jaHJvbml6ZXIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBTeW5jaHJvbml6ZXIgc3luY2hyb25pemVyID0gQ29udGV4dFN0b3JhZ2UuZ2V0U3luY2hyb25pemVyKCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc2h1dGRvd24gd2F0Y2hkb2cuCi0gICAgICovCi0gICAgZmluYWwgU2h1dGRvd25XYXRjaGRvZyBzaHV0ZG93bldhdGNoZG9nID0gbmV3IFNodXRkb3duV2F0Y2hkb2coKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBhdXRvIG51bWJlci4KLSAgICAgKi8KLSAgICBmaW5hbCBBdXRvTnVtYmVyIGF1dG9OdW1iZXIgPSBuZXcgQXV0b051bWJlcigpOwotCi0gICAgLyoqCi0gICAgICogVGhlIGV2ZW50IHR5cGUgbG9va3VwLgotICAgICAqLwotICAgIGZpbmFsIEFXVEV2ZW50LkV2ZW50VHlwZUxvb2t1cCBldmVudFR5cGVMb29rdXAgPSBuZXcgQVdURXZlbnQuRXZlbnRUeXBlTG9va3VwKCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYiBkeW5hbWljIGxheW91dCBzZXQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGJEeW5hbWljTGF5b3V0U2V0ID0gdHJ1ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBzZXQgb2YgZGVza3RvcCBwcm9wZXJ0aWVzIHRoYXQgdXNlciBzZXQgZGlyZWN0bHkuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBIYXNoU2V0PFN0cmluZz4gdXNlclByb3BTZXQgPSBuZXcgSGFzaFNldDxTdHJpbmc+KCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGVza3RvcCBwcm9wZXJ0aWVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBNYXA8U3RyaW5nLCBPYmplY3Q+IGRlc2t0b3BQcm9wZXJ0aWVzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGRlc2t0b3AgcHJvcHMgc3VwcG9ydC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgUHJvcGVydHlDaGFuZ2VTdXBwb3J0IGRlc2t0b3BQcm9wc1N1cHBvcnQ7Ci0KLSAgICAvKioKLSAgICAgKiBGb3IgdGhpcyBjb21wb25lbnQgdGhlIG5hdGl2ZSB3aW5kb3cgaXMgYmVpbmcgY3JlYXRlZCBJdCBpcyB1c2VkIGluIHRoZQotICAgICAqIGNhbGxiYWNrLWRyaXZlbiB3aW5kb3cgY3JlYXRpb24gKGUuZy4gb24gV2luZG93cyBpbiB0aGUgaGFuZGxlciBvZgotICAgICAqIFdNX0NSRUFURSBldmVudCkgdG8gZXN0YWJsaXNoIHRoZSBjb25uZWN0aW9uIGJldHdlZW4gdGhpcyBjb21wb25lbnQgYW5kCi0gICAgICogaXRzIG5hdGl2ZSB3aW5kb3cuCi0gICAgICovCi0gICAgcHJpdmF0ZSBPYmplY3QgcmVjZW50TmF0aXZlV2luZG93Q29tcG9uZW50OwotCi0gICAgLyoqCi0gICAgICogVGhlIHd0ay4KLSAgICAgKi8KLSAgICBwcml2YXRlIFdUSyB3dGs7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgQ29tcG9uZW50SW50ZXJuYWxzSW1wbC4KLSAgICAgKiAKLSAgICAgKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgZmluYWwgY2xhc3MgQ29tcG9uZW50SW50ZXJuYWxzSW1wbCBleHRlbmRzIENvbXBvbmVudEludGVybmFscyB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFNodXRkb3duLgotICAgICAgICAgKi8KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyB2b2lkIHNodXRkb3duKCkgewotICAgICAgICAgICAgZGlzcGF0Y2hUaHJlYWQuc2h1dGRvd24oKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBTZXRzIHRoZSBkZXNrdG9wIHByb3BlcnR5IHRvIHRoZSBzcGVjaWZpZWQgdmFsdWUgYW5kIGZpcmVzIGEgcHJvcGVydHkKLSAgICAgICAgICogY2hhbmdlIGV2ZW50LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIG5hbWUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiBwcm9wZXJ0eS4KLSAgICAgICAgICogQHBhcmFtIHZhbHVlCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIG5ldyB2YWx1ZSBvZiBwcm9wZXJ0eS4KLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXREZXNrdG9wUHJvcGVydHkoU3RyaW5nIG5hbWUsIE9iamVjdCB2YWx1ZSkgewotICAgICAgICAgICAgVG9vbGtpdC50aGlzLnNldERlc2t0b3BQcm9wZXJ0eShuYW1lLCB2YWx1ZSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBIGxvdCBvZiBtZXRob2RzIG11c3QgdGhyb3cgSGVhZGxlc3NFeGNlcHRpb24gaWYKLSAgICAgKiA8Y29kZT5HcmFwaGljc0Vudmlyb25tZW50LmlzSGVhZGxlc3MoKTwvY29kZT4gcmV0dXJucyA8Y29kZT50cnVlPC9jb2RlPi4KLSAgICAgKiAKLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgdGhlIGhlYWRsZXNzIGV4Y2VwdGlvbi4KLSAgICAgKi8KLSAgICBzdGF0aWMgdm9pZCBjaGVja0hlYWRsZXNzKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKEdyYXBoaWNzRW52aXJvbm1lbnQuZ2V0TG9jYWxHcmFwaGljc0Vudmlyb25tZW50KCkuaXNIZWFkbGVzc0luc3RhbmNlKCkpCi0gICAgICAgICAgICB0aHJvdyBuZXcgSGVhZGxlc3NFeGNlcHRpb24oKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBMb2NrIEFXVC4KLSAgICAgKi8KLSAgICBmaW5hbCB2b2lkIGxvY2tBV1QoKSB7Ci0gICAgICAgIHN5bmNocm9uaXplci5sb2NrKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU3RhdGljIGxvY2sgQVdULgotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCB2b2lkIHN0YXRpY0xvY2tBV1QoKSB7Ci0gICAgICAgIENvbnRleHRTdG9yYWdlLmdldFN5bmNocm9uaXplcigpLmxvY2soKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBVbmxvY2sgQVdULgotICAgICAqLwotICAgIGZpbmFsIHZvaWQgdW5sb2NrQVdUKCkgewotICAgICAgICBzeW5jaHJvbml6ZXIudW5sb2NrKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU3RhdGljIHVubG9jayBBV1QuCi0gICAgICovCi0gICAgc3RhdGljIGZpbmFsIHZvaWQgc3RhdGljVW5sb2NrQVdUKCkgewotICAgICAgICBDb250ZXh0U3RvcmFnZS5nZXRTeW5jaHJvbml6ZXIoKS51bmxvY2soKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnZva2VBbmRXYWl0IHVuZGVyIEFXVCBsb2NrLiBXL28gdGhpcyBtZXRob2Qgc3lzdGVtIGNhbiBoYW5nIHVwLiBBZGRlZAotICAgICAqIHRvIHN1cHBvcnQgbW9kYWxpdHkgKERpYWxvZy5zaG93KCkgJiBQb3B1cE1lbnUuc2hvdygpKSBmcm9tIG5vdCBldmVudAotICAgICAqIGRpc3BhdGNoIHRocmVhZC4gVXNlIGluIG90aGVyIGNhc2VzIGlzIG5vdCByZWNvbW1lbmRlZC4gU3RpbGwgY2FuIGJlCi0gICAgICogY2FsbGVkIG9ubHkgZm9yIHdob2xlIEFQSSBtZXRob2RzIHRoYXQgY2Fubm90IGJlIGNhbGxlZCBmcm9tIG90aGVyCi0gICAgICogY2xhc3NlcyBBUEkgbWV0aG9kcy4gRXhhbXBsZXM6IHNob3coKSBmb3IgbW9kYWwgZGlhbG9ncyAtIGNvcnJlY3QsIG9ubHkKLSAgICAgKiB1c2VyIGNhbiBjYWxsIGl0LCBkaXJlY3RseSBvciB0aHJvdWdoIHNldFZpc2libGUodHJ1ZSkgc2V0Qm91bmRzKCkgZm9yCi0gICAgICogY29tcG9uZW50cyAtIGluY29ycmVjdCwgc2V0Qm91bmRzKCkgY2FuIGJlIGNhbGxlZCBmcm9tIGxheW91dENvbnRhaW5lcigpCi0gICAgICogZm9yIGxheW91dCBtYW5hZ2VycwotICAgICAqIAotICAgICAqIEBwYXJhbSBydW5uYWJsZQotICAgICAqICAgICAgICAgICAgdGhlIHJ1bm5hYmxlLgotICAgICAqIEB0aHJvd3MgSW50ZXJydXB0ZWRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICB0aGUgaW50ZXJydXB0ZWQgZXhjZXB0aW9uLgotICAgICAqIEB0aHJvd3MgSW52b2NhdGlvblRhcmdldEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIHRoZSBpbnZvY2F0aW9uIHRhcmdldCBleGNlcHRpb24uCi0gICAgICovCi0gICAgZmluYWwgdm9pZCB1bnNhZmVJbnZva2VBbmRXYWl0KFJ1bm5hYmxlIHJ1bm5hYmxlKSB0aHJvd3MgSW50ZXJydXB0ZWRFeGNlcHRpb24sCi0gICAgICAgICAgICBJbnZvY2F0aW9uVGFyZ2V0RXhjZXB0aW9uIHsKLSAgICAgICAgc3luY2hyb25pemVyLnN0b3JlU3RhdGVBbmRGcmVlKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBFdmVudFF1ZXVlLmludm9rZUFuZFdhaXQocnVubmFibGUpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgc3luY2hyb25pemVyLmxvY2tBbmRSZXN0b3JlU3RhdGUoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN5bmNocm9uaXplci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzeW5jaHJvbml6ZXIuCi0gICAgICovCi0gICAgZmluYWwgU3luY2hyb25pemVyIGdldFN5bmNocm9uaXplcigpIHsKLSAgICAgICAgcmV0dXJuIHN5bmNocm9uaXplcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB3VEsuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgd1RLLgotICAgICAqLwotICAgIGZpbmFsIFdUSyBnZXRXVEsoKSB7Ci0gICAgICAgIHJldHVybiB3dGs7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcHJvcGVydHkgd2l0aCB0aGUgc3BlY2lmaWVkIGtleSBhbmQgZGVmYXVsdCB2YWx1ZS4gVGhpcyBtZXRob2QKLSAgICAgKiByZXR1cm5zIHRoZSBkZWZWYWx1ZSBpZiB0aGUgcHJvcGVydHkgaXMgbm90IGZvdW5kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm9wTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWUgb2YgcHJvcGVydHkuCi0gICAgICogQHBhcmFtIGRlZlZhbAotICAgICAqICAgICAgICAgICAgdGhlIGRlZmF1bHQgdmFsdWUuCi0gICAgICogQHJldHVybiB0aGUgcHJvcGVydHkgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBTdHJpbmcgZ2V0UHJvcGVydHkoU3RyaW5nIHByb3BOYW1lLCBTdHJpbmcgZGVmVmFsKSB7Ci0gICAgICAgIGlmIChwcm9wTmFtZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuN0Q9UHJvcGVydHkgbmFtZSBpcyBudWxsCi0gICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuN0QiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBzdGF0aWNMb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBTdHJpbmcgcmV0VmFsID0gbnVsbDsKLSAgICAgICAgICAgIGlmIChwcm9wZXJ0aWVzICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgICAgICByZXRWYWwgPSBwcm9wZXJ0aWVzLmdldFN0cmluZyhwcm9wTmFtZSk7Ci0gICAgICAgICAgICAgICAgfSBjYXRjaCAoTWlzc2luZ1Jlc291cmNlRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICB9IGNhdGNoIChDbGFzc0Nhc3RFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiAocmV0VmFsID09IG51bGwpID8gZGVmVmFsIDogcmV0VmFsOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgc3RhdGljVW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkZWZhdWx0IFRvb2xraXQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGVmYXVsdCBUb29sa2l0LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgVG9vbGtpdCBnZXREZWZhdWx0VG9vbGtpdCgpIHsKLSAgICAgICAgc3luY2hyb25pemVkIChDb250ZXh0U3RvcmFnZS5nZXRDb250ZXh0TG9jaygpKSB7Ci0gICAgICAgICAgICBpZiAoQ29udGV4dFN0b3JhZ2Uuc2h1dGRvd25QZW5kaW5nKCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIFRvb2xraXQgZGVmVG9vbGtpdCA9IENvbnRleHRTdG9yYWdlLmdldERlZmF1bHRUb29sa2l0KCk7Ci0gICAgICAgICAgICBpZiAoZGVmVG9vbGtpdCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGRlZlRvb2xraXQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBzdGF0aWNMb2NrQVdUKCk7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIGRlZlRvb2xraXQgPSBHcmFwaGljc0Vudmlyb25tZW50LmlzSGVhZGxlc3MoKSA/IG5ldyBIZWFkbGVzc1Rvb2xraXQoKQotICAgICAgICAgICAgICAgICAgICAgICAgOiBuZXcgVG9vbGtpdEltcGwoKTsKLSAgICAgICAgICAgICAgICBDb250ZXh0U3RvcmFnZS5zZXREZWZhdWx0VG9vbGtpdChkZWZUb29sa2l0KTsKLSAgICAgICAgICAgICAgICByZXR1cm4gZGVmVG9vbGtpdDsKLSAgICAgICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICAgICAgc3RhdGljVW5sb2NrQVdUKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAvLyBUT0RPOiByZWFkIHN5c3RlbSBwcm9wZXJ0eSBuYW1lZCBhd3QudG9vbGtpdAotICAgICAgICAgICAgLy8gYW5kIGNyZWF0ZSBhbiBpbnN0YW5jZSBvZiB0aGUgc3BlY2lmaWVkIGNsYXNzLAotICAgICAgICAgICAgLy8gYnkgZGVmYXVsdCB1c2UgVG9vbGtpdEltcGwKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRlZmF1bHQgRm9udC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkZWZhdWx0IEZvbnQgZm9yIFRvb2xraXQuCi0gICAgICovCi0gICAgRm9udCBnZXREZWZhdWx0Rm9udCgpIHsKLSAgICAgICAgcmV0dXJuIHd0ay5nZXRTeXN0ZW1Qcm9wZXJ0aWVzKCkuZ2V0RGVmYXVsdEZvbnQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBMb2FkIHJlc291cmNlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcGF0aAotICAgICAqICAgICAgICAgICAgdGhlIHBhdGguCi0gICAgICogQHJldHVybiB0aGUgcmVzb3VyY2UgYnVuZGxlLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIFJlc291cmNlQnVuZGxlIGxvYWRSZXNvdXJjZXMoU3RyaW5nIHBhdGgpIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBSZXNvdXJjZUJ1bmRsZS5nZXRCdW5kbGUocGF0aCk7Ci0gICAgICAgIH0gY2F0Y2ggKE1pc3NpbmdSZXNvdXJjZUV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHdUSyBjbGFzcyBuYW1lLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHdUSyBjbGFzcyBuYW1lLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIFN0cmluZyBnZXRXVEtDbGFzc05hbWUoKSB7Ci0gICAgICAgIHJldHVybiAiY29tLmFuZHJvaWQuaW50ZXJuYWwuYXd0LkFuZHJvaWRXVEsiOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvbXBvbmVudCBieSBpZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBpZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBjb21wb25lbnQgYnkgaWQuCi0gICAgICovCi0gICAgQ29tcG9uZW50IGdldENvbXBvbmVudEJ5SWQobG9uZyBpZCkgewotICAgICAgICBpZiAoaWQgPT0gMCkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgR3JhcGhpY3NGYWN0b3J5LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEdyYXBoaWNzRmFjdG9yeSBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIEdyYXBoaWNzRmFjdG9yeSBnZXRHcmFwaGljc0ZhY3RvcnkoKSB7Ci0gICAgICAgIHJldHVybiB3dGsuZ2V0R3JhcGhpY3NGYWN0b3J5KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHRvb2xraXQuCi0gICAgICovCi0gICAgcHVibGljIFRvb2xraXQoKSB7Ci0gICAgICAgIGluaXQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbml0aWF0ZXMgQVdULgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIGluaXQoKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIENvbXBvbmVudEludGVybmFscy5zZXRDb21wb25lbnRJbnRlcm5hbHMobmV3IENvbXBvbmVudEludGVybmFsc0ltcGwoKSk7Ci0gICAgICAgICAgICBuZXcgRXZlbnRRdWV1ZSh0aGlzKTsgLy8gY3JlYXRlIHRoZSBzeXN0ZW0gRXZlbnRRdWV1ZQotICAgICAgICAgICAgZGlzcGF0Y2hlciA9IG5ldyBEaXNwYXRjaGVyKHRoaXMpOwotICAgICAgICAgICAgZmluYWwgU3RyaW5nIGNsYXNzTmFtZSA9IGdldFdUS0NsYXNzTmFtZSgpOwotICAgICAgICAgICAgZGVza3RvcFByb3BlcnRpZXMgPSBuZXcgSGFzaE1hcDxTdHJpbmcsIE9iamVjdD4oKTsKLSAgICAgICAgICAgIGRlc2t0b3BQcm9wc1N1cHBvcnQgPSBuZXcgUHJvcGVydHlDaGFuZ2VTdXBwb3J0KHRoaXMpOwotICAgICAgICAgICAgYXd0RXZlbnRzTWFuYWdlciA9IG5ldyBBV1RFdmVudHNNYW5hZ2VyKCk7Ci0gICAgICAgICAgICBkaXNwYXRjaFRocmVhZCA9IG5ldyBFdmVudERpc3BhdGNoVGhyZWFkKHRoaXMsIGRpc3BhdGNoZXIpOwotICAgICAgICAgICAgbmF0aXZlVGhyZWFkID0gbmV3IE5hdGl2ZUV2ZW50VGhyZWFkKCk7Ci0gICAgICAgICAgICBOYXRpdmVFdmVudFRocmVhZC5Jbml0IGluaXQgPSBuZXcgTmF0aXZlRXZlbnRUaHJlYWQuSW5pdCgpIHsKLSAgICAgICAgICAgICAgICBwdWJsaWMgV1RLIGluaXQoKSB7Ci0gICAgICAgICAgICAgICAgICAgIHd0ayA9IGNyZWF0ZVdUSyhjbGFzc05hbWUpOwotICAgICAgICAgICAgICAgICAgICB3dGsuZ2V0TmF0aXZlRXZlbnRRdWV1ZSgpLnNldFNodXRkb3duV2F0Y2hkb2coc2h1dGRvd25XYXRjaGRvZyk7Ci0gICAgICAgICAgICAgICAgICAgIHN5bmNocm9uaXplci5zZXRFbnZpcm9ubWVudCh3dGssIGRpc3BhdGNoVGhyZWFkKTsKLSAgICAgICAgICAgICAgICAgICAgQ29udGV4dFN0b3JhZ2Uuc2V0V1RLKHd0ayk7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB3dGs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfTsKLSAgICAgICAgICAgIG5hdGl2ZVRocmVhZC5zdGFydChpbml0KTsKLSAgICAgICAgICAgIGRpc3BhdGNoVGhyZWFkLnN0YXJ0KCk7Ci0gICAgICAgICAgICB3dGsuZ2V0TmF0aXZlRXZlbnRRdWV1ZSgpLmF3YWtlKCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFN5bmNocm9uaXplcyB0aGlzIHRvb2xraXQncyBncmFwaGljcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzeW5jKCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBjb25zdHJ1Y3Rpb24gc3RhdHVzIG9mIGEgc3BlY2lmaWVkIGltYWdlIHRoYXQgaXMgYmVpbmcKLSAgICAgKiBjcmVhdGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHRvIGJlIGNoZWNrZWQuCi0gICAgICogQHBhcmFtIGExCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2Ygc2NhbGVkIGltYWdlIGZvciB3aGljaCB0aGUgc3RhdHVzIGlzIGJlaW5nCi0gICAgICogICAgICAgICAgICBjaGVja2VkIG9yIC0xLgotICAgICAqIEBwYXJhbSBhMgotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBzY2FsZWQgaW1hZ2UgZm9yIHdoaWNoIHRoZSBzdGF0dXMgaXMgYmVpbmcKLSAgICAgKiAgICAgICAgICAgIGNoZWNrZWQgb3IgLTEuCi0gICAgICogQHBhcmFtIGEzCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VPYnNlcnZlciBvYmplY3QgdG8gYmUgbm90aWZpZWQgd2hpbGUgdGhlIGltYWdlIGlzCi0gICAgICogICAgICAgICAgICBiZWluZyBwcmVwYXJlZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZU9ic2VydmVyIGZsYWdzIHdoaWNoIGdpdmUgdGhlIGN1cnJlbnQgc3RhdGUgb2YgdGhlIGltYWdlCi0gICAgICogICAgICAgICBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgY2hlY2tJbWFnZShJbWFnZSBhMCwgaW50IGExLCBpbnQgYTIsIEltYWdlT2JzZXJ2ZXIgYTMpOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgaW1hZ2Ugd2l0aCB0aGUgc3BlY2lmaWVkIEltYWdlUHJvZHVjZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VQcm9kdWNlciB0byBiZSB1c2VkIGZvciBpbWFnZSBjcmVhdGlvbi4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbWFnZSB3aXRoIHRoZSBzcGVjaWZpZWQgSW1hZ2VQcm9kdWNlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgSW1hZ2UgY3JlYXRlSW1hZ2UoSW1hZ2VQcm9kdWNlciBhMCk7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBpbWFnZSBmcm9tIHRoZSBzcGVjaWZpZWQgYnl0ZSBhcnJheSwgb2Zmc2V0IGFuZCBsZW5ndGguIFRoZQotICAgICAqIGJ5dGUgYXJyYXkgc2hvdWxkIGNvbnRhaW4gZGF0YSB3aXRoIGltYWdlIGZvcm1hdCBzdXBwb3J0ZWQgYnkgVG9vbGtpdAotICAgICAqIHN1Y2ggYXMgSlBFRywgR0lGLCBvciBQTkcuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUgYnl0ZSBhcnJheSB3aXRoIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBhMQotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBvZiB0aGUgYmVnaW5uaW5nIHRoZSBpbWFnZSBkYXRhIGluIHRoZSBieXRlIGFycmF5LgotICAgICAqIEBwYXJhbSBhMgotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCBvZiB0aGUgaW1hZ2UgZGF0YSBpbiB0aGUgYnl0ZSBhcnJheS4KLSAgICAgKiBAcmV0dXJuIHRoZSBjcmVhdGVkIEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJbWFnZSBjcmVhdGVJbWFnZShieXRlW10gYTAsIGludCBhMSwgaW50IGEyKTsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGltYWdlIHVzaW5nIGltYWdlIGRhdGEgZnJvbSB0aGUgc3BlY2lmaWVkIFVSTC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYTAKLSAgICAgKiAgICAgICAgICAgIHRoZSBVUkwgZm9yIGV4dHJhY3RpbmcgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgSW1hZ2UgY3JlYXRlSW1hZ2UoVVJMIGEwKTsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGltYWdlIHVzaW5nIGltYWdlIGRhdGEgZnJvbSB0aGUgc3BlY2lmaWVkIGZpbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUgZmlsZSBuYW1lIHdoaWNoIGNvbnRhaW5zIGltYWdlIGRhdGEgb2Ygc3VwcG9ydGVkIGZvcm1hdC4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgSW1hZ2UgY3JlYXRlSW1hZ2UoU3RyaW5nIGEwKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvbG9yIG1vZGVsLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIENvbG9yTW9kZWwgb2YgVG9vbGtpdCdzIHNjcmVlbi4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIEdyYXBoaWNzRW52aXJvbm1lbnQuaXNIZWFkbGVzcygpIG1ldGhvZCByZXR1cm5zIHRydWUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IENvbG9yTW9kZWwgZ2V0Q29sb3JNb2RlbCgpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNjcmVlbiBkZXZpY2UgbWV0cmljcyBmb3IgdGhlIHNwZWNpZmllZCBmb250LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmb250Ci0gICAgICogICAgICAgICAgICB0aGUgRm9udC4KLSAgICAgKiBAcmV0dXJuIHRoZSBGb250TWV0cmljcyBmb3IgdGhlIHNwZWNpZmllZCBGb250LgotICAgICAqIEBkZXByZWNhdGVkIFVzZSBnZXRMaW5lTWV0cmljcyBtZXRob2QgZnJvbSBGb250IGNsYXNzLgotICAgICAqLwotCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgYWJzdHJhY3QgRm9udE1ldHJpY3MgZ2V0Rm9udE1ldHJpY3MoRm9udCBmb250KTsKLQotICAgIC8qKgotICAgICAqIFByZXBhcmVzIHRoZSBzcGVjaWZpZWQgaW1hZ2UgZm9yIHJlbmRlcmluZyBvbiB0aGUgc2NyZWVuIHdpdGggdGhlCi0gICAgICogc3BlY2lmaWVkIHNpemUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2UgdG8gYmUgcHJlcGFyZWQuCi0gICAgICogQHBhcmFtIGExCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHNjcmVlbiByZXByZXNlbnRhdGlvbiBvciAtMSBmb3IgdGhlIGN1cnJlbnQKLSAgICAgKiAgICAgICAgICAgIHNjcmVlbi4KLSAgICAgKiBAcGFyYW0gYTIKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHNjcmVlbiByZXByZXNlbnRhdGlvbiBvciAtMSBmb3IgdGhlIGN1cnJlbnQKLSAgICAgKiAgICAgICAgICAgIHNjcmVlbi4KLSAgICAgKiBAcGFyYW0gYTMKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZU9ic2VydmVyIG9iamVjdCB0byBiZSBub3RpZmllZCBhcyBzb29uIGFzIHRoZSBpbWFnZQotICAgICAqICAgICAgICAgICAgaXMgcHJlcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBpbWFnZSBpcyBmdWxseSBwcmVwYXJlZCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBib29sZWFuIHByZXBhcmVJbWFnZShJbWFnZSBhMCwgaW50IGExLCBpbnQgYTIsIEltYWdlT2JzZXJ2ZXIgYTMpOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhbiBhdWRpbyBiZWVwLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGJlZXAoKTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGFycmF5IG9mIGZvbnQgbmFtZXMgd2hpY2ggYXJlIGF2YWlsYWJsZSBpbiB0aGlzIFRvb2xraXQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgZm9udCBuYW1lcyB3aGljaCBhcmUgYXZhaWxhYmxlIGluIHRoaXMgVG9vbGtpdC4KLSAgICAgKiBAZGVwcmVjYXRlZCB1c2UgR3JhcGhpY3NFbnZpcm9ubWVudC5nZXRBdmFpbGFibGVGb250RmFtaWx5TmFtZXMoKSBtZXRob2QuCi0gICAgICovCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgYWJzdHJhY3QgU3RyaW5nW10gZ2V0Rm9udExpc3QoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRoZSBGb250IGltcGxlbWVudGF0aW9uIHVzaW5nIHRoZSBzcGVjaWZpZWQgcGVlciBpbnRlcmZhY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUgRm9udCBuYW1lIHRvIGJlIGltcGxlbWVudGVkLgotICAgICAqIEBwYXJhbSBhMQotICAgICAqICAgICAgICAgICAgdGhlIHRoZSBmb250IHN0eWxlOiBQTEFJTiwgQk9MRCwgSVRBTElDLgotICAgICAqIEByZXR1cm4gdGhlIEZvbnRQZWVyIGltcGxlbWVudGF0aW9uIG9mIHRoZSBzcGVjaWZpZWQgRm9udC4KLSAgICAgKiBAZGVwcmVjYXRlZCB1c2UgamF2YS5hd3QuR3JhcGhpY3NFbnZpcm9ubWVudC5nZXRBbGxGb250cyBtZXRob2QuCi0gICAgICovCi0KLSAgICBARGVwcmVjYXRlZAotICAgIHByb3RlY3RlZCBhYnN0cmFjdCBGb250UGVlciBnZXRGb250UGVlcihTdHJpbmcgYTAsIGludCBhMSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBpbWFnZSBmcm9tIHRoZSBzcGVjaWZpZWQgZmlsZSB3aGljaCBjb250YWlucyBpbWFnZSBkYXRhIGluIGEKLSAgICAgKiBzdXBwb3J0ZWQgaW1hZ2UgZm9ybWF0IChzdWNoIGFzIEpQRUcsIEdJRiwgb3IgUE5HKTsgdGhpcyBtZXRob2Qgc2hvdWxkCi0gICAgICogcmV0dXJuIHRoZSBzYW1lIEltYWdlIGZvciBtdWx0aXBsZSBjYWxscyBvZiB0aGlzIG1ldGhvZCB3aXRoIHRoZSBzYW1lCi0gICAgICogaW1hZ2UgZmlsZSBuYW1lLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIGZpbGUgbmFtZSB3aGljaCBjb250YWlucyBpbWFnZSBkYXRhIGluIGEgc3VwcG9ydGVkIGltYWdlCi0gICAgICogICAgICAgICAgICBmb3JtYXQgKHN1Y2ggYXMgSlBFRywgR0lGLCBvciBQTkcpLgotICAgICAqIEByZXR1cm4gdGhlIEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJbWFnZSBnZXRJbWFnZShTdHJpbmcgYTApOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW1hZ2UgZnJvbSB0aGUgc3BlY2lmaWVkIFVSTCB3aGljaCBjb250YWlucyBpbWFnZSBkYXRhIGluIGEKLSAgICAgKiBzdXBwb3J0ZWQgaW1hZ2UgZm9ybWF0IChzdWNoIGFzIEpQRUcsIEdJRiwgb3IgUE5HKTsgdGhpcyBtZXRob2Qgc2hvdWxkCi0gICAgICogcmV0dXJuIHRoZSBzYW1lIEltYWdlIGZvciBtdWx0aXBsZSBjYWxscyBvZiB0aGlzIG1ldGhvZCB3aXRoIHRoZSBzYW1lCi0gICAgICogaW1hZ2UgVVJMLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIFVSTCB3aGljaCBjb250YWlucyBpbWFnZSBkYXRhIGluIGEgc3VwcG9ydGVkIGltYWdlIGZvcm1hdAotICAgICAqICAgICAgICAgICAgKHN1Y2ggYXMgSlBFRywgR0lGLCBvciBQTkcpLgotICAgICAqIEByZXR1cm4gdGhlIEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJbWFnZSBnZXRJbWFnZShVUkwgYTApOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2NyZWVuIHJlc29sdXRpb24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc2NyZWVuIHJlc29sdXRpb24uCi0gICAgICogQHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBHcmFwaGljc0Vudmlyb25tZW50LmlzSGVhZGxlc3MoKSBtZXRob2QgcmV0dXJucyB0cnVlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0U2NyZWVuUmVzb2x1dGlvbigpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNjcmVlbiBzaXplLgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBEaW1lbnNpb24gb2JqZWN0IGNvbnRhaW5pbmcgdGhlIHdpZHRoIGFuZCBoZWlnaHQgb2YgdGhlIHNjcmVlbi4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIEdyYXBoaWNzRW52aXJvbm1lbnQuaXNIZWFkbGVzcygpIG1ldGhvZCByZXR1cm5zIHRydWUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IERpbWVuc2lvbiBnZXRTY3JlZW5TaXplKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgRXZlbnRRdWV1ZSBpbnN0YW5jZSB3aXRob3V0IGNoZWNraW5nIGFjY2Vzcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzeXN0ZW0gRXZlbnRRdWV1ZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgYWJzdHJhY3QgRXZlbnRRdWV1ZSBnZXRTeXN0ZW1FdmVudFF1ZXVlSW1wbCgpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIG1hcCBvZiB0ZXh0IGF0dHJpYnV0ZXMgZm9yIHRoZSBhYnN0cmFjdCBsZXZlbCBkZXNjcmlwdGlvbiBvZgotICAgICAqIHRoZSBzcGVjaWZpZWQgaW5wdXQgbWV0aG9kIGhpZ2hsaWdodCwgb3IgbnVsbCBpZiBubyBtYXBwaW5nIGlzIGZvdW5kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBoaWdobGlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbnB1dE1ldGhvZEhpZ2hsaWdodC4KLSAgICAgKiBAcmV0dXJuIHRoZSBNYXA8amF2YS5hd3QuZm9udC4gdGV4dCBhdHRyaWJ1dGUsPz4uCi0gICAgICogQHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBHcmFwaGljc0Vudmlyb25tZW50LmlzSGVhZGxlc3MoKSBtZXRob2QgcmV0dXJucyB0cnVlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBNYXA8amF2YS5hd3QuZm9udC5UZXh0QXR0cmlidXRlLCA/PiBtYXBJbnB1dE1ldGhvZEhpZ2hsaWdodCgKLSAgICAgICAgICAgIElucHV0TWV0aG9kSGlnaGxpZ2h0IGhpZ2hsaWdodCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogTWFwIGlucHV0IG1ldGhvZCBoaWdobGlnaHQgaW1wbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaGlnaGxpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGlnaGxpZ2h0LgotICAgICAqIEByZXR1cm4gdGhlIG1hcDxqYXZhLmF3dC5mb250LiB0ZXh0IGF0dHJpYnV0ZSw/Pi4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgdGhlIGhlYWRsZXNzIGV4Y2VwdGlvbi4KLSAgICAgKi8KLSAgICBNYXA8amF2YS5hd3QuZm9udC5UZXh0QXR0cmlidXRlLCA/PiBtYXBJbnB1dE1ldGhvZEhpZ2hsaWdodEltcGwoSW5wdXRNZXRob2RIaWdobGlnaHQgaGlnaGxpZ2h0KQotICAgICAgICAgICAgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgSGFzaE1hcDxqYXZhLmF3dC5mb250LlRleHRBdHRyaWJ1dGUsID8+IG1hcCA9IG5ldyBIYXNoTWFwPGphdmEuYXd0LmZvbnQuVGV4dEF0dHJpYnV0ZSwgT2JqZWN0PigpOwotICAgICAgICB3dGsuZ2V0U3lzdGVtUHJvcGVydGllcygpLm1hcElucHV0TWV0aG9kSGlnaGxpZ2h0KGhpZ2hsaWdodCwgbWFwKTsKLSAgICAgICAgcmV0dXJuIENvbGxlY3Rpb25zLjxqYXZhLmF3dC5mb250LlRleHRBdHRyaWJ1dGUsIE9iamVjdD4gdW5tb2RpZmlhYmxlTWFwKG1hcCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgc3BlY2lmaWVkIFByb3BlcnR5Q2hhbmdlTGlzdGVuZXIgbGlzdGVuZXIgZm9yIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBwcm9wZXJ0eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSBuYW1lIGZvciB3aGljaCB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICAgICBQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyIHdpbGwgYmUgYWRkZWQuCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyKFN0cmluZyBwcm9wTmFtZSwgUHJvcGVydHlDaGFuZ2VMaXN0ZW5lciBsKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGlmIChkZXNrdG9wUHJvcGVydGllcy5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgICAgICBpbml0aWFsaXplRGVza3RvcFByb3BlcnRpZXMoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChsICE9IG51bGwpIHsgLy8gdGhlcmUgaXMgbm8gZ3VhcmFudGVlIHRoYXQgbnVsbCBsaXN0ZW5lciB3aWxsIG5vdCBiZQotICAgICAgICAgICAgLy8gYWRkZWQKLSAgICAgICAgICAgIGRlc2t0b3BQcm9wc1N1cHBvcnQuYWRkUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcihwcm9wTmFtZSwgbCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGFuIGFycmF5IG9mIHRoZSBwcm9wZXJ0eSBjaGFuZ2UgbGlzdGVuZXJzIHJlZ2lzdGVyZWQgd2l0aCB0aGlzCi0gICAgICogVG9vbGtpdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIHRoZSBwcm9wZXJ0eSBjaGFuZ2UgbGlzdGVuZXJzIHJlZ2lzdGVyZWQgd2l0aCB0aGlzCi0gICAgICogICAgICAgICBUb29sa2l0LgotICAgICAqLwotICAgIHB1YmxpYyBQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyW10gZ2V0UHJvcGVydHlDaGFuZ2VMaXN0ZW5lcnMoKSB7Ci0gICAgICAgIHJldHVybiBkZXNrdG9wUHJvcHNTdXBwb3J0LmdldFByb3BlcnR5Q2hhbmdlTGlzdGVuZXJzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhbiBhcnJheSBvZiB0aGUgcHJvcGVydHkgY2hhbmdlIGxpc3RlbmVycyByZWdpc3RlcmVkIHdpdGggdGhpcwotICAgICAqIFRvb2xraXQgZm9yIG5vdGlmaWNhdGlvbiByZWdhcmRpbmcgdGhlIHNwZWNpZmllZCBwcm9wZXJ0eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSBuYW1lIGZvciB3aGljaCB0aGUgUHJvcGVydHlDaGFuZ2VMaXN0ZW5lciB3YXMKLSAgICAgKiAgICAgICAgICAgIHJlZ2lzdGVyZWQuCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcnMgcmVnaXN0ZXJlZCBmb3IgdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgcHJvcGVydHkgbmFtZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcltdIGdldFByb3BlcnR5Q2hhbmdlTGlzdGVuZXJzKFN0cmluZyBwcm9wTmFtZSkgewotICAgICAgICByZXR1cm4gZGVza3RvcFByb3BzU3VwcG9ydC5nZXRQcm9wZXJ0eUNoYW5nZUxpc3RlbmVycyhwcm9wTmFtZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgc3BlY2lmaWVkIHByb3BlcnR5IGNoYW5nZSBsaXN0ZW5lciByZWdpc3RlcmVkIGZvciB0aGUKLSAgICAgKiBzcGVjaWZpZWQgcHJvcGVydHkgbmFtZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSBuYW1lLgotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgUHJvcGVydHlDaGFuZ2VMaXN0ZW5lciByZWdpc3RlcmVkIGZvciB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICAgICBwcm9wZXJ0eSBuYW1lLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZVByb3BlcnR5Q2hhbmdlTGlzdGVuZXIoU3RyaW5nIHByb3BOYW1lLCBQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyIGwpIHsKLSAgICAgICAgZGVza3RvcFByb3BzU3VwcG9ydC5yZW1vdmVQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyKHByb3BOYW1lLCBsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgY3VzdG9tIGN1cnNvciB3aXRoIHRoZSBzcGVjaWZpZWQgSW1hZ2UsIGhvdCBzcG90LCBhbmQgY3Vyc29yCi0gICAgICogZGVzY3JpcHRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIGltZwotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIG9mIGFjdGl2YXRlZCBjdXJzb3IuCi0gICAgICogQHBhcmFtIGhvdFNwb3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBQb2ludCBnaXZpbmcgdGhlIGNvb3JkaW5hdGVzIG9mIHRoZSBjdXJzb3IncyBob3Qgc3BvdC4KLSAgICAgKiBAcGFyYW0gbmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGN1cnNvciBkZXNjcmlwdGlvbi4KLSAgICAgKiBAcmV0dXJuIHRoZSBjdXJzb3Igd2l0aCB0aGUgc3BlY2lmaWVkIEltYWdlLCBob3Qgc3BvdCwgYW5kIGN1cnNvcgotICAgICAqICAgICAgICAgZGVzY3JpcHRpb24uCi0gICAgICogQHRocm93cyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIGhvdCBzcG90IHZhbHVlcyBhcmUgb3V0c2lkZSB0aGUgYm91bmRzIG9mIHRoZSBjdXJzb3IuCi0gICAgICogQHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGlzSGVhZGxlc3MoKSBtZXRob2Qgb2YgR3JhcGhpY3NFbnZpcm9ubWVudCBjbGFzcyByZXR1cm5zCi0gICAgICogICAgICAgICAgICAgdHJ1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ3Vyc29yIGNyZWF0ZUN1c3RvbUN1cnNvcihJbWFnZSBpbWcsIFBvaW50IGhvdFNwb3QsIFN0cmluZyBuYW1lKQotICAgICAgICAgICAgdGhyb3dzIEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24sIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaW50IHcgPSBpbWcuZ2V0V2lkdGgobnVsbCksIHggPSBob3RTcG90Lng7Ci0gICAgICAgICAgICBpbnQgaCA9IGltZy5nZXRIZWlnaHQobnVsbCksIHkgPSBob3RTcG90Lnk7Ci0gICAgICAgICAgICBpZiAoeCA8IDAgfHwgeCA+PSB3IHx8IHkgPCAwIHx8IHkgPj0gaCkgewotICAgICAgICAgICAgICAgIC8vIGF3dC43RT1pbnZhbGlkIGhvdFNwb3QKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC43RSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIG5ldyBDdXJzb3IobmFtZSwgaW1nLCBob3RTcG90KTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgc3VwcG9ydGVkIGN1cnNvciBkaW1lbnNpb24gd2hpY2ggaXMgY2xvc2VzdCB0byB0aGUgc3BlY2lmaWVkCi0gICAgICogd2lkdGggYW5kIGhlaWdodC4gSWYgdGhlIFRvb2xraXQgb25seSBzdXBwb3J0cyBhIHNpbmdsZSBjdXJzb3Igc2l6ZSwgdGhpcwotICAgICAqIG1ldGhvZCBzaG91bGQgcmV0dXJuIHRoZSBzdXBwb3J0ZWQgY3Vyc29yIHNpemUuIElmIGN1c3RvbSBjdXJzb3IgaXMgbm90Ci0gICAgICogc3VwcG9ydGVkLCBhIGRpbWVuc2lvbiBvZiAwLCAwIHNob3VsZCBiZSByZXR1cm5lZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJlZldpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgcHJlZmVycmVkIGN1cnNvciB3aWR0aC4KLSAgICAgKiBAcGFyYW0gcHJlZkhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIHByZWZlcnJlZCBjdXJzb3IgaGVpZ2h0LgotICAgICAqIEByZXR1cm4gdGhlIHN1cHBvcnRlZCBjdXJzb3IgZGltZW5zaW9uIHdoaWNoIGlzIGNsb3Nlc3QgdG8gdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgd2lkdGggYW5kIGhlaWdodC4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgR3JhcGhpY3NFbnZpcm9ubWVudC5pc0hlYWRsZXNzKCkgcmV0dXJucyB0cnVlLgotICAgICAqLwotICAgIHB1YmxpYyBEaW1lbnNpb24gZ2V0QmVzdEN1cnNvclNpemUoaW50IHByZWZXaWR0aCwgaW50IHByZWZIZWlnaHQpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiB3dGsuZ2V0Q3Vyc29yRmFjdG9yeSgpLmdldEJlc3RDdXJzb3JTaXplKHByZWZXaWR0aCwgcHJlZkhlaWdodCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHZhbHVlIGZvciB0aGUgc3BlY2lmaWVkIGRlc2t0b3AgcHJvcGVydHkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb3BOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydHkgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBPYmplY3QgdGhhdCBpcyB0aGUgcHJvcGVydHkncyB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgT2JqZWN0IGdldERlc2t0b3BQcm9wZXJ0eShTdHJpbmcgcHJvcE5hbWUpIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKGRlc2t0b3BQcm9wZXJ0aWVzLmlzRW1wdHkoKSkgewotICAgICAgICAgICAgICAgIGluaXRpYWxpemVEZXNrdG9wUHJvcGVydGllcygpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHByb3BOYW1lLmVxdWFscygiYXd0LmR5bmFtaWNMYXlvdXRTdXBwb3J0ZWQiKSkgeyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgLy8gZHluYW1pY0xheW91dFN1cHBvcnRlZCBpcyBzcGVjaWFsIGNhc2UKLSAgICAgICAgICAgICAgICByZXR1cm4gQm9vbGVhbi52YWx1ZU9mKGlzRHluYW1pY0xheW91dEFjdGl2ZSgpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIE9iamVjdCB2YWwgPSBkZXNrdG9wUHJvcGVydGllcy5nZXQocHJvcE5hbWUpOwotICAgICAgICAgICAgaWYgKHZhbCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgLy8gdHJ5IHRvIGxhemlseSBsb2FkIHByb3AgdmFsdWUKLSAgICAgICAgICAgICAgICAvLyBqdXN0IGZvciBjb21wYXRpYmlsaXR5LCBvdXIgbGF6aWx5TG9hZCBpcyBlbXB0eQotICAgICAgICAgICAgICAgIHZhbCA9IGxhemlseUxvYWREZXNrdG9wUHJvcGVydHkocHJvcE5hbWUpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIHZhbDsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbG9ja2luZyBrZXkgc3RhdGUgZm9yIHRoZSBzcGVjaWZpZWQga2V5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIGtleSBjb2RlOiBWS19DQVBTX0xPQ0ssIFZLX05VTV9MT0NLLCBWS19TQ1JPTExfTE9DSywgb3IKLSAgICAgKiAgICAgICAgICAgIFZLX0tBTkFfTE9DSy4KLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgdGhlIHNwZWNpZmllZCBrZXkgY29kZSBpcyBpbiB0aGUgbG9ja2VkIHN0YXRlLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqIEB0aHJvd3MgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgc3RhdGUgb2YgdGhpcyBrZXkgY2FuJ3QgYmUgcmV0cmlldmVkLCBvciBpZiB0aGUKLSAgICAgKiAgICAgICAgICAgICBrZXlib2FyZCBkb2Vzbid0IGhhdmUgdGhpcyBrZXkuCi0gICAgICogQHRocm93cyBOb3RJbXBsZW1lbnRlZEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoaXMgbWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBnZXRMb2NraW5nS2V5U3RhdGUoaW50IGEwKSB0aHJvd3MgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24sCi0gICAgICAgICAgICBvcmcuYXBhY2hlLmhhcm1vbnkubHVuaS51dGlsLk5vdEltcGxlbWVudGVkRXhjZXB0aW9uIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHRydWUpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJNZXRob2QgaXMgbm90IGltcGxlbWVudGVkIik7IC8vVE9ETzogaW1wbGVtZW50IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBtYXhpbXVtIG51bWJlciBvZiBjb2xvcnMgd2hpY2ggdGhlIFRvb2xraXQgc3VwcG9ydHMgZm9yCi0gICAgICogY3VzdG9tIGN1cnNvci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtYXhpbXVtIGN1cnNvciBjb2xvcnMuCi0gICAgICogQHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBHcmFwaGljc0Vudmlyb25tZW50LmlzSGVhZGxlc3MoKSBtZXRob2QgcmV0dXJucyB0cnVlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TWF4aW11bUN1cnNvckNvbG9ycygpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiB3dGsuZ2V0Q3Vyc29yRmFjdG9yeSgpLmdldE1heGltdW1DdXJzb3JDb2xvcnMoKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWVudSBzaG9ydGN1dCBrZXkgbWFzay4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtZW51IHNob3J0Y3V0IGtleSBtYXNrLgotICAgICAqIEB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgR3JhcGhpY3NFbnZpcm9ubWVudC5pc0hlYWRsZXNzKCkgbWV0aG9kIHJldHVybnMgdHJ1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE1lbnVTaG9ydGN1dEtleU1hc2soKSB0aHJvd3MgSGVhZGxlc3NFeGNlcHRpb24gewotICAgICAgICBsb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gSW5wdXRFdmVudC5DVFJMX01BU0s7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNjcmVlbiBpbnNldHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGdjCi0gICAgICogICAgICAgICAgICB0aGUgR3JhcGhpY3NDb25maWd1cmF0aW9uLgotICAgICAqIEByZXR1cm4gdGhlIGluc2V0cyBvZiB0aGlzIHRvb2xraXQuCi0gICAgICogQHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBHcmFwaGljc0Vudmlyb25tZW50LmlzSGVhZGxlc3MoKSBtZXRob2QgcmV0dXJucyB0cnVlLgotICAgICAqLwotICAgIHB1YmxpYyBJbnNldHMgZ2V0U2NyZWVuSW5zZXRzKEdyYXBoaWNzQ29uZmlndXJhdGlvbiBnYykgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKGdjID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbigpOwotICAgICAgICB9Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgSW5zZXRzKDAsIDAsIDAsIDApOyAvLyBUT0RPOiBnZXQgcmVhbCBzY3JlZW4gaW5zZXRzCi0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN5c3RlbSBFdmVudFF1ZXVlIGluc3RhbmNlLiBJZiB0aGUgZGVmYXVsdCBpbXBsZW1lbnRhdGlvbiBvZgotICAgICAqIGNoZWNrQXd0RXZlbnRRdWV1ZUFjY2VzcyBpcyB1c2VkLCB0aGVuIHRoaXMgcmVzdWx0cyBvZiBhIGNhbGwgdG8gdGhlCi0gICAgICogc2VjdXJpdHkgbWFuYWdlcidzIGNoZWNrUGVybWlzc2lvbiBtZXRob2Qgd2l0aCBhbgotICAgICAqIEFXVFBlcm1pc3Npb24oImFjY2Vzc0V2ZW50UXVldWUiKSBwZXJtaXNzaW9uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN5c3RlbSBFdmVudFF1ZXVlIGluc3RhbmNlLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBFdmVudFF1ZXVlIGdldFN5c3RlbUV2ZW50UXVldWUoKSB7Ci0gICAgICAgIFNlY3VyaXR5TWFuYWdlciBzbSA9IFN5c3RlbS5nZXRTZWN1cml0eU1hbmFnZXIoKTsKLSAgICAgICAgaWYgKHNtICE9IG51bGwpIHsKLSAgICAgICAgICAgIHNtLmNoZWNrQXd0RXZlbnRRdWV1ZUFjY2VzcygpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBnZXRTeXN0ZW1FdmVudFF1ZXVlSW1wbCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN5c3RlbSBldmVudCBxdWV1ZSBjb3JlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN5c3RlbSBldmVudCBxdWV1ZSBjb3JlLgotICAgICAqLwotICAgIEV2ZW50UXVldWVDb3JlIGdldFN5c3RlbUV2ZW50UXVldWVDb3JlKCkgewotICAgICAgICByZXR1cm4gc3lzdGVtRXZlbnRRdWV1ZUNvcmU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc3lzdGVtIGV2ZW50IHF1ZXVlIGNvcmUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvcmUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgc3lzdGVtIGV2ZW50IHF1ZXVlIGNvcmUuCi0gICAgICovCi0gICAgdm9pZCBzZXRTeXN0ZW1FdmVudFF1ZXVlQ29yZShFdmVudFF1ZXVlQ29yZSBjb3JlKSB7Ci0gICAgICAgIHN5c3RlbUV2ZW50UXVldWVDb3JlID0gY29yZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbml0aWFsaXplIHRoZSBkZXNrdG9wIHByb3BlcnRpZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgaW5pdGlhbGl6ZURlc2t0b3BQcm9wZXJ0aWVzKCkgewotICAgICAgICBsb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICB3dGsuZ2V0U3lzdGVtUHJvcGVydGllcygpLmluaXQoZGVza3RvcFByb3BlcnRpZXMpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgZHluYW1pYyBsYXlvdXQgb2YgQ29udGFpbmVycyBpcyBhY3RpdmUgb3Igbm90LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgaXMgZHluYW1pYyBsYXlvdXQgb2YgQ29udGFpbmVycyBpcyBhY3RpdmUsIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICogQHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBHcmFwaGljc0Vudmlyb25tZW50LmlzSGVhZGxlc3MoKSBtZXRob2QgcmV0dXJucyB0cnVlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzRHluYW1pY0xheW91dEFjdGl2ZSgpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIC8vIGFsd2F5cyByZXR1cm4gdHJ1ZQotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgaWYgdGhlIGxheW91dCBvZiBDb250YWluZXJzIGlzIGNoZWNrZWQgZHluYW1pY2FsbHkgZHVyaW5nCi0gICAgICogcmVzaXppbmcsIG9yIHN0YXRpY2FsbHkgYWZ0ZXIgcmVzaXppbmcgaXMgY29tcGxldGVkLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgaWYgdGhlIGxheW91dCBvZiBDb250YWluZXJzIGlzIGNoZWNrZWQgZHluYW1pY2FsbHkKLSAgICAgKiAgICAgICAgIGR1cmluZyByZXNpemluZzsgZmFsc2UsIGlmIHRoZSBsYXlvdXQgb2YgQ29udGFpbmVycyBpcyBjaGVja2VkCi0gICAgICogICAgICAgICBzdGF0aWNhbGx5IGFmdGVyIHJlc2l6aW5nIGlzIGNvbXBsZXRlZC4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIEdyYXBoaWNzRW52aXJvbm1lbnQuaXNIZWFkbGVzcygpIG1ldGhvZCByZXR1cm5zIHRydWUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGJvb2xlYW4gaXNEeW5hbWljTGF5b3V0U2V0KCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGJEeW5hbWljTGF5b3V0U2V0OwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIHNwZWNpZmllZCBmcmFtZSBzdGF0ZSBpcyBzdXBwb3J0ZWQgYnkgVG9vbGtpdCBvciBub3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0YXRlCi0gICAgICogICAgICAgICAgICB0aGUgZnJhbWUgc3RhdGUuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBmcmFtZSBzdGF0ZSBpcyBzdXBwb3J0ZWQsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIEdyYXBoaWNzRW52aXJvbm1lbnQuaXNIZWFkbGVzcygpIG1ldGhvZCByZXR1cm5zIHRydWUuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNGcmFtZVN0YXRlU3VwcG9ydGVkKGludCBzdGF0ZSkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIHd0ay5nZXRXaW5kb3dGYWN0b3J5KCkuaXNXaW5kb3dTdGF0ZVN1cHBvcnRlZChzdGF0ZSk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIExvYWRzIHRoZSB2YWx1ZSBvZiB0aGUgZGVza3RvcCBwcm9wZXJ0eSB3aXRoIHRoZSBzcGVjaWZpZWQgcHJvcGVydHkgbmFtZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0eSBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIGRlc2t0b3AgcHJvcGVydHkgdmFsdWVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBPYmplY3QgbGF6aWx5TG9hZERlc2t0b3BQcm9wZXJ0eShTdHJpbmcgcHJvcE5hbWUpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogTG9hZHMgdGhlIGN1cnJlbnQgc3lzdGVtIGNvbG9yIHZhbHVlcyB0byB0aGUgc3BlY2lmaWVkIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvcnMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB3aGVyZSB0aGUgY3VycmVudCBzeXN0ZW0gY29sb3IgdmFsdWVzIGFyZSB3cml0dGVuIGJ5Ci0gICAgICogICAgICAgICAgICB0aGlzIG1ldGhvZC4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIEdyYXBoaWNzRW52aXJvbm1lbnQuaXNIZWFkbGVzcygpIG1ldGhvZCByZXR1cm5zIHRydWUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgbG9hZFN5c3RlbUNvbG9ycyhpbnRbXSBjb2xvcnMpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgdmFsdWUgb2YgdGhlIGRlc2t0b3AgcHJvcGVydHkgd2l0aCB0aGUgc3BlY2lmaWVkIG5hbWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb3BOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydHkncyBuYW1lLgotICAgICAqIEBwYXJhbSB2YWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIHByb3BlcnR5J3MgdmFsdWUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGZpbmFsIHZvaWQgc2V0RGVza3RvcFByb3BlcnR5KFN0cmluZyBwcm9wTmFtZSwgT2JqZWN0IHZhbHVlKSB7Ci0gICAgICAgIE9iamVjdCBvbGRWYWw7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIG9sZFZhbCA9IGdldERlc2t0b3BQcm9wZXJ0eShwcm9wTmFtZSk7Ci0gICAgICAgICAgICB1c2VyUHJvcFNldC5hZGQocHJvcE5hbWUpOwotICAgICAgICAgICAgZGVza3RvcFByb3BlcnRpZXMucHV0KHByb3BOYW1lLCB2YWx1ZSk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgICAgICBkZXNrdG9wUHJvcHNTdXBwb3J0LmZpcmVQcm9wZXJ0eUNoYW5nZShwcm9wTmFtZSwgb2xkVmFsLCB2YWx1ZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbGF5b3V0IHN0YXRlLCB3aGV0aGVyIHRoZSBDb250YWluZXIgbGF5b3V0IGlzIGNoZWNrZWQKLSAgICAgKiBkeW5hbWljYWxseSBkdXJpbmcgcmVzaXppbmcsIG9yIHN0YXRpY2FsbHkgYWZ0ZXIgcmVzaXppbmcgaXMgY29tcGxldGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkeW5hbWljCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IGR5bmFtaWMgbGF5b3V0IHN0YXRlIC0gaWYgdHJ1ZSB0aGUgbGF5b3V0IG9mCi0gICAgICogICAgICAgICAgICBDb250YWluZXJzIGlzIGNoZWNrZWQgZHluYW1pY2FsbHkgZHVyaW5nIHJlc2l6aW5nLCBpZiBmYWxzZSAtCi0gICAgICogICAgICAgICAgICBzdGF0aWNhbGx5IGFmdGVyIHJlc2l6aW5nIGlzIGNvbXBsZXRlZC4KLSAgICAgKiBAdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIEdyYXBoaWNzRW52aXJvbm1lbnQuaXNIZWFkbGVzcygpIG1ldGhvZCByZXR1cm5zIHRydWUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RHluYW1pY0xheW91dChib29sZWFuIGR5bmFtaWMpIHRocm93cyBIZWFkbGVzc0V4Y2VwdGlvbiB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGJEeW5hbWljTGF5b3V0U2V0ID0gZHluYW1pYzsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbG9ja2luZyBrZXkgc3RhdGUgZm9yIHRoZSBzcGVjaWZpZWQga2V5IGNvZGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUga2V5IGNvZGU6IFZLX0NBUFNfTE9DSywgVktfTlVNX0xPQ0ssIFZLX1NDUk9MTF9MT0NLLCBvcgotICAgICAqICAgICAgICAgICAgVktfS0FOQV9MT0NLLgotICAgICAqIEBwYXJhbSBhMQotICAgICAqICAgICAgICAgICAgdGhlIHN0YXRlIC0gdHJ1ZSB0byBzZXQgdGhlIHNwZWNpZmllZCBrZXkgY29kZSB0byB0aGUgbG9ja2VkCi0gICAgICogICAgICAgICAgICBzdGF0ZSwgZmFsc2UgLSB0byB1bmxvY2sgaXQuCi0gICAgICogQHRocm93cyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBzdGF0ZSBvZiB0aGlzIGtleSBjYW4ndCBiZSBzZXQsIG9yIGlmIHRoZSBrZXlib2FyZAotICAgICAqICAgICAgICAgICAgIGRvZXNuJ3QgaGF2ZSB0aGlzIGtleS4KLSAgICAgKiBAdGhyb3dzIE5vdEltcGxlbWVudGVkRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhpcyBtZXRob2QgaXMgbm90IGltcGxlbWVudGVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldExvY2tpbmdLZXlTdGF0ZShpbnQgYTAsIGJvb2xlYW4gYTEpIHRocm93cyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbiwKLSAgICAgICAgICAgIG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuTm90SW1wbGVtZW50ZWRFeGNlcHRpb24gewotICAgICAgICBsb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgICAgICBpZiAodHJ1ZSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQiKTsgLy9UT0RPOiBpbXBsZW1lbnQgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBPbiBxdWV1ZSBlbXB0eS4KLSAgICAgKi8KLSAgICB2b2lkIG9uUXVldWVFbXB0eSgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCEiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSB3dGsuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNsc05hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBjbHMgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSB3VEsuCi0gICAgICovCi0gICAgcHJpdmF0ZSBXVEsgY3JlYXRlV1RLKFN0cmluZyBjbHNOYW1lKSB7Ci0gICAgICAgIFdUSyBuZXdXVEsgPSBudWxsOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgbmV3V1RLID0gKFdUSylDbGFzcy5mb3JOYW1lKGNsc05hbWUpLm5ld0luc3RhbmNlKCk7Ci0gICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbihlKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbmV3V1RLOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbm5lY3QgdGhlIGNvbXBvbmVudCB0byBpdHMgbmF0aXZlIHdpbmRvdwotICAgICAqIAotICAgICAqIEBwYXJhbSB3aW5JZAotICAgICAqICAgICAgICAgICAgdGhlIGlkIG9mIG5hdGl2ZSB3aW5kb3cganVzdCBjcmVhdGVkLgotICAgICAqLwotICAgIGJvb2xlYW4gb25XaW5kb3dDcmVhdGVkKGxvbmcgd2luSWQpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG5hdGl2ZSBldmVudCBxdWV1ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBuYXRpdmUgZXZlbnQgcXVldWUuCi0gICAgICovCi0gICAgTmF0aXZlRXZlbnRRdWV1ZSBnZXROYXRpdmVFdmVudFF1ZXVlKCkgewotICAgICAgICByZXR1cm4gd3RrLmdldE5hdGl2ZUV2ZW50UXVldWUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgc2hhcmVkIGluc3RhbmNlIG9mIGltcGxlbWVudGF0aW9uIG9mCi0gICAgICogb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuTmF0aXZlQ3Vyc29yIGZvciBjdXJyZW50IHBsYXRmb3JtIGZvci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdHlwZQotICAgICAqICAgICAgICAgICAgdGhlIEphdmEgQ3Vyc29yIHR5cGUuCi0gICAgICogQHJldHVybiBuZXcgaW5zdGFuY2Ugb2YgaW1wbGVtZW50YXRpb24gb2YgTmF0aXZlQ3Vyc29yLgotICAgICAqLwotICAgIE5hdGl2ZUN1cnNvciBjcmVhdGVOYXRpdmVDdXJzb3IoaW50IHR5cGUpIHsKLSAgICAgICAgcmV0dXJuIHd0ay5nZXRDdXJzb3JGYWN0b3J5KCkuZ2V0Q3Vyc29yKHR5cGUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBzaGFyZWQgaW5zdGFuY2Ugb2YgaW1wbGVtZW50YXRpb24gb2YKLSAgICAgKiBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5OYXRpdmVDdXJzb3IgZm9yIGN1cnJlbnQgcGxhdGZvcm0gZm9yIGN1c3RvbQotICAgICAqIGN1cnNvcgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWcKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWcuCi0gICAgICogQHBhcmFtIGhvdFNwb3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBob3Qgc3BvdC4KLSAgICAgKiBAcGFyYW0gbmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWUuCi0gICAgICogQHJldHVybiBuZXcgaW5zdGFuY2Ugb2YgaW1wbGVtZW50YXRpb24gb2YgTmF0aXZlQ3Vyc29yLgotICAgICAqLwotICAgIE5hdGl2ZUN1cnNvciBjcmVhdGVDdXN0b21OYXRpdmVDdXJzb3IoSW1hZ2UgaW1nLCBQb2ludCBob3RTcG90LCBTdHJpbmcgbmFtZSkgewotICAgICAgICByZXR1cm4gd3RrLmdldEN1cnNvckZhY3RvcnkoKS5jcmVhdGVDdXN0b21DdXJzb3IoaW1nLCBob3RTcG90LngsIGhvdFNwb3QueSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyBhbiBBV1RFdmVudExpc3RlbmVyIHRvIHRoZSBUb29sa2l0IHRvIGxpc3RlbiBmb3IgZXZlbnRzIG9mIHR5cGVzCi0gICAgICogY29ycmVzcG9uZGluZyB0byBiaXRzIGluIHRoZSBzcGVjaWZpZWQgZXZlbnQgbWFzay4gRXZlbnQgbWFza3MgYXJlCi0gICAgICogZGVmaW5lZCBpbiBBV1RFdmVudCBjbGFzcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbGlzdGVuZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBBV1RFdmVudExpc3RlbmVyLgotICAgICAqIEBwYXJhbSBldmVudE1hc2sKLSAgICAgKiAgICAgICAgICAgIHRoZSBiaXRtYXNrIG9mIGV2ZW50IHR5cGVzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGFkZEFXVEV2ZW50TGlzdGVuZXIoQVdURXZlbnRMaXN0ZW5lciBsaXN0ZW5lciwgbG9uZyBldmVudE1hc2spIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgU2VjdXJpdHlNYW5hZ2VyIHNlY3VyaXR5ID0gU3lzdGVtLmdldFNlY3VyaXR5TWFuYWdlcigpOwotICAgICAgICAgICAgaWYgKHNlY3VyaXR5ICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBzZWN1cml0eS5jaGVja1Blcm1pc3Npb24oYXd0RXZlbnRzTWFuYWdlci5wZXJtaXNzaW9uKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGF3dEV2ZW50c01hbmFnZXIuYWRkQVdURXZlbnRMaXN0ZW5lcihsaXN0ZW5lciwgZXZlbnRNYXNrKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgc3BlY2lmaWVkIEFXVCBldmVudCBsaXN0ZW5lci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbGlzdGVuZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBBV1RFdmVudExpc3RlbmVyIHRvIGJlIHJlbW92ZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlQVdURXZlbnRMaXN0ZW5lcihBV1RFdmVudExpc3RlbmVyIGxpc3RlbmVyKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIFNlY3VyaXR5TWFuYWdlciBzZWN1cml0eSA9IFN5c3RlbS5nZXRTZWN1cml0eU1hbmFnZXIoKTsKLSAgICAgICAgICAgIGlmIChzZWN1cml0eSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgc2VjdXJpdHkuY2hlY2tQZXJtaXNzaW9uKGF3dEV2ZW50c01hbmFnZXIucGVybWlzc2lvbik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBhd3RFdmVudHNNYW5hZ2VyLnJlbW92ZUFXVEV2ZW50TGlzdGVuZXIobGlzdGVuZXIpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhcnJheSBvZiBhbGwgQVdUIGV2ZW50IGxpc3RlbmVycyByZWdpc3RlcmVkIHdpdGggdGhpcyBUb29sa2l0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGFsbCBBV1QgZXZlbnQgbGlzdGVuZXJzIHJlZ2lzdGVyZWQgd2l0aCB0aGlzCi0gICAgICogICAgICAgICBUb29sa2l0LgotICAgICAqLwotICAgIHB1YmxpYyBBV1RFdmVudExpc3RlbmVyW10gZ2V0QVdURXZlbnRMaXN0ZW5lcnMoKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIFNlY3VyaXR5TWFuYWdlciBzZWN1cml0eSA9IFN5c3RlbS5nZXRTZWN1cml0eU1hbmFnZXIoKTsKLSAgICAgICAgICAgIGlmIChzZWN1cml0eSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgc2VjdXJpdHkuY2hlY2tQZXJtaXNzaW9uKGF3dEV2ZW50c01hbmFnZXIucGVybWlzc2lvbik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gYXd0RXZlbnRzTWFuYWdlci5nZXRBV1RFdmVudExpc3RlbmVycygpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBhcnJheSBvZiB0aGUgQVdUIGV2ZW50IGxpc3RlbmVycyByZWdpc3RlcmVkIHdpdGggdGhpcyBUb29sa2l0Ci0gICAgICogZm9yIHRoZSBldmVudCB0eXBlcyBjb3JyZXNwb25kaW5nIHRvIHRoZSBzcGVjaWZpZWQgZXZlbnQgbWFzay4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXZlbnRNYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYml0IG1hc2sgb2YgZXZlbnQgdHlwZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiB0aGUgQVdUIGV2ZW50IGxpc3RlbmVycyByZWdpc3RlcmVkIGluIHRoaXMgVG9vbGtpdAotICAgICAqICAgICAgICAgZm9yIHRoZSBldmVudCB0eXBlcyBjb3JyZXNwb25kaW5nIHRvIHRoZSBzcGVjaWZpZWQgZXZlbnQgbWFzay4KLSAgICAgKi8KLSAgICBwdWJsaWMgQVdURXZlbnRMaXN0ZW5lcltdIGdldEFXVEV2ZW50TGlzdGVuZXJzKGxvbmcgZXZlbnRNYXNrKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIFNlY3VyaXR5TWFuYWdlciBzZWN1cml0eSA9IFN5c3RlbS5nZXRTZWN1cml0eU1hbmFnZXIoKTsKLSAgICAgICAgICAgIGlmIChzZWN1cml0eSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgc2VjdXJpdHkuY2hlY2tQZXJtaXNzaW9uKGF3dEV2ZW50c01hbmFnZXIucGVybWlzc2lvbik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gYXd0RXZlbnRzTWFuYWdlci5nZXRBV1RFdmVudExpc3RlbmVycyhldmVudE1hc2spOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEaXNwYXRjaCBBV1QgZXZlbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGV2ZW50Ci0gICAgICogICAgICAgICAgICB0aGUgZXZlbnQuCi0gICAgICovCi0gICAgdm9pZCBkaXNwYXRjaEFXVEV2ZW50KEFXVEV2ZW50IGV2ZW50KSB7Ci0gICAgICAgIGF3dEV2ZW50c01hbmFnZXIuZGlzcGF0Y2hBV1RFdmVudChldmVudCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIEFXVEV2ZW50c01hbmFnZXIuCi0gICAgICovCi0gICAgZmluYWwgY2xhc3MgQVdURXZlbnRzTWFuYWdlciB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBwZXJtaXNzaW9uLgotICAgICAgICAgKi8KLSAgICAgICAgQVdUUGVybWlzc2lvbiBwZXJtaXNzaW9uID0gbmV3IEFXVFBlcm1pc3Npb24oImxpc3RlblRvQWxsQVdURXZlbnRzIik7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGxpc3RlbmVycy4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgZmluYWwgQVdUTGlzdGVuZXJMaXN0PEFXVEV2ZW50TGlzdGVuZXJQcm94eT4gbGlzdGVuZXJzID0gbmV3IEFXVExpc3RlbmVyTGlzdDxBV1RFdmVudExpc3RlbmVyUHJveHk+KCk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEFkZHMgdGhlIEFXVCBldmVudCBsaXN0ZW5lci4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBsaXN0ZW5lcgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBsaXN0ZW5lci4KLSAgICAgICAgICogQHBhcmFtIGV2ZW50TWFzawotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBldmVudCBtYXNrLgotICAgICAgICAgKi8KLSAgICAgICAgdm9pZCBhZGRBV1RFdmVudExpc3RlbmVyKEFXVEV2ZW50TGlzdGVuZXIgbGlzdGVuZXIsIGxvbmcgZXZlbnRNYXNrKSB7Ci0gICAgICAgICAgICBpZiAobGlzdGVuZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIGxpc3RlbmVycy5hZGRVc2VyTGlzdGVuZXIobmV3IEFXVEV2ZW50TGlzdGVuZXJQcm94eShldmVudE1hc2ssIGxpc3RlbmVyKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmVtb3ZlcyB0aGUgQVdUIGV2ZW50IGxpc3RlbmVyLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGxpc3RlbmVyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGxpc3RlbmVyLgotICAgICAgICAgKi8KLSAgICAgICAgdm9pZCByZW1vdmVBV1RFdmVudExpc3RlbmVyKEFXVEV2ZW50TGlzdGVuZXIgbGlzdGVuZXIpIHsKLSAgICAgICAgICAgIGlmIChsaXN0ZW5lciAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChBV1RFdmVudExpc3RlbmVyUHJveHkgcHJveHkgOiBsaXN0ZW5lcnMuZ2V0VXNlckxpc3RlbmVycygpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChsaXN0ZW5lciA9PSBwcm94eS5nZXRMaXN0ZW5lcigpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBsaXN0ZW5lcnMucmVtb3ZlVXNlckxpc3RlbmVyKHByb3h5KTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBHZXRzIHRoZSBBV1QgZXZlbnQgbGlzdGVuZXJzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgQVdUIGV2ZW50IGxpc3RlbmVycy4KLSAgICAgICAgICovCi0gICAgICAgIEFXVEV2ZW50TGlzdGVuZXJbXSBnZXRBV1RFdmVudExpc3RlbmVycygpIHsKLSAgICAgICAgICAgIEhhc2hTZXQ8RXZlbnRMaXN0ZW5lcj4gbGlzdGVuZXJzU2V0ID0gbmV3IEhhc2hTZXQ8RXZlbnRMaXN0ZW5lcj4oKTsKLSAgICAgICAgICAgIGZvciAoQVdURXZlbnRMaXN0ZW5lclByb3h5IHByb3h5IDogbGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMoKSkgewotICAgICAgICAgICAgICAgIGxpc3RlbmVyc1NldC5hZGQocHJveHkuZ2V0TGlzdGVuZXIoKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gbGlzdGVuZXJzU2V0LnRvQXJyYXkobmV3IEFXVEV2ZW50TGlzdGVuZXJbbGlzdGVuZXJzU2V0LnNpemUoKV0pOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEdldHMgdGhlIEFXVCBldmVudCBsaXN0ZW5lcnMuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gZXZlbnRNYXNrCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGV2ZW50IG1hc2suCi0gICAgICAgICAqIEByZXR1cm4gdGhlIEFXVCBldmVudCBsaXN0ZW5lcnMuCi0gICAgICAgICAqLwotICAgICAgICBBV1RFdmVudExpc3RlbmVyW10gZ2V0QVdURXZlbnRMaXN0ZW5lcnMobG9uZyBldmVudE1hc2spIHsKLSAgICAgICAgICAgIEhhc2hTZXQ8RXZlbnRMaXN0ZW5lcj4gbGlzdGVuZXJzU2V0ID0gbmV3IEhhc2hTZXQ8RXZlbnRMaXN0ZW5lcj4oKTsKLSAgICAgICAgICAgIGZvciAoQVdURXZlbnRMaXN0ZW5lclByb3h5IHByb3h5IDogbGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMoKSkgewotICAgICAgICAgICAgICAgIGlmICgocHJveHkuZ2V0RXZlbnRNYXNrKCkgJiBldmVudE1hc2spID09IGV2ZW50TWFzaykgewotICAgICAgICAgICAgICAgICAgICBsaXN0ZW5lcnNTZXQuYWRkKHByb3h5LmdldExpc3RlbmVyKCkpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBsaXN0ZW5lcnNTZXQudG9BcnJheShuZXcgQVdURXZlbnRMaXN0ZW5lcltsaXN0ZW5lcnNTZXQuc2l6ZSgpXSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogRGlzcGF0Y2ggQVdUIGV2ZW50LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGV2ZW50Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGV2ZW50LgotICAgICAgICAgKi8KLSAgICAgICAgdm9pZCBkaXNwYXRjaEFXVEV2ZW50KEFXVEV2ZW50IGV2ZW50KSB7Ci0gICAgICAgICAgICBBV1RFdmVudC5FdmVudERlc2NyaXB0b3IgZGVzY3JpcHRvciA9IGV2ZW50VHlwZUxvb2t1cC5nZXRFdmVudERlc2NyaXB0b3IoZXZlbnQpOwotICAgICAgICAgICAgaWYgKGRlc2NyaXB0b3IgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGZvciAoQVdURXZlbnRMaXN0ZW5lclByb3h5IHByb3h5IDogbGlzdGVuZXJzLmdldFVzZXJMaXN0ZW5lcnMoKSkgewotICAgICAgICAgICAgICAgIGlmICgocHJveHkuZ2V0RXZlbnRNYXNrKCkgJiBkZXNjcmlwdG9yLmV2ZW50TWFzaykgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICBwcm94eS5ldmVudERpc3BhdGNoZWQoZXZlbnQpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBBdXRvTnVtYmVyLgotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBjbGFzcyBBdXRvTnVtYmVyIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG5leHQgY29tcG9uZW50LgotICAgICAgICAgKi8KLSAgICAgICAgaW50IG5leHRDb21wb25lbnQgPSAwOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbmV4dCBjYW52YXMuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgbmV4dENhbnZhcyA9IDA7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBuZXh0IHBhbmVsLgotICAgICAgICAgKi8KLSAgICAgICAgaW50IG5leHRQYW5lbCA9IDA7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBuZXh0IHdpbmRvdy4KLSAgICAgICAgICovCi0gICAgICAgIGludCBuZXh0V2luZG93ID0gMDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG5leHQgZnJhbWUuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgbmV4dEZyYW1lID0gMDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG5leHQgZGlhbG9nLgotICAgICAgICAgKi8KLSAgICAgICAgaW50IG5leHREaWFsb2cgPSAwOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbmV4dCBidXR0b24uCi0gICAgICAgICAqLwotICAgICAgICBpbnQgbmV4dEJ1dHRvbiA9IDA7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBuZXh0IG1lbnUgY29tcG9uZW50LgotICAgICAgICAgKi8KLSAgICAgICAgaW50IG5leHRNZW51Q29tcG9uZW50ID0gMDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG5leHQgbGFiZWwuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgbmV4dExhYmVsID0gMDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG5leHQgY2hlY2sgYm94LgotICAgICAgICAgKi8KLSAgICAgICAgaW50IG5leHRDaGVja0JveCA9IDA7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBuZXh0IHNjcm9sbGJhci4KLSAgICAgICAgICovCi0gICAgICAgIGludCBuZXh0U2Nyb2xsYmFyID0gMDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG5leHQgc2Nyb2xsIHBhbmUuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgbmV4dFNjcm9sbFBhbmUgPSAwOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbmV4dCBsaXN0LgotICAgICAgICAgKi8KLSAgICAgICAgaW50IG5leHRMaXN0ID0gMDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG5leHQgY2hvaWNlLgotICAgICAgICAgKi8KLSAgICAgICAgaW50IG5leHRDaG9pY2UgPSAwOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbmV4dCBmaWxlIGRpYWxvZy4KLSAgICAgICAgICovCi0gICAgICAgIGludCBuZXh0RmlsZURpYWxvZyA9IDA7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBuZXh0IHRleHQgYXJlYS4KLSAgICAgICAgICovCi0gICAgICAgIGludCBuZXh0VGV4dEFyZWEgPSAwOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbmV4dCB0ZXh0IGZpZWxkLgotICAgICAgICAgKi8KLSAgICAgICAgaW50IG5leHRUZXh0RmllbGQgPSAwOwotICAgIH0KLQotICAgIHByaXZhdGUgY2xhc3MgTG9jayB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIGxvY2suCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBPYmplY3QgbG9jayA9IG5ldyBMb2NrKCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9Ub29sa2l0SW1wbC5qYXZhIGIvYXd0L2phdmEvYXd0L1Rvb2xraXRJbXBsLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDUwMTVhZWYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L1Rvb2xraXRJbXBsLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNTUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dDsKLQotaW1wb3J0IGphdmEuYXd0LmltLklucHV0TWV0aG9kSGlnaGxpZ2h0OwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW1hZ2VPYnNlcnZlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbWFnZVByb2R1Y2VyOwotaW1wb3J0IGphdmEuYXd0LnBlZXIuKjsKLWltcG9ydCBqYXZhLmlvLlNlcmlhbGl6YWJsZTsKLWltcG9ydCBqYXZhLm5ldC5VUkw7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2h0YWJsZTsKLWltcG9ydCBqYXZhLnV0aWwuTWFwOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2UuKjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0ay5HcmFwaGljc0ZhY3Rvcnk7Ci0KLWNsYXNzIFRvb2xraXRJbXBsIGV4dGVuZHMgVG9vbGtpdCB7Ci0JCi0gICAgc3RhdGljIGZpbmFsIEhhc2h0YWJsZTxTZXJpYWxpemFibGUsIEltYWdlPiBpbWFnZUNhY2hlID0gbmV3IEhhc2h0YWJsZTxTZXJpYWxpemFibGUsIEltYWdlPigpOwotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc3luYygpIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGNoZWNrSW1hZ2UoSW1hZ2UgaW1hZ2UsIGludCB3aWR0aCwgaW50IGhlaWdodCwgSW1hZ2VPYnNlcnZlciBvYnNlcnZlcikgewotICAgICAgICBsb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpZiAod2lkdGggPT0gMCB8fCBoZWlnaHQgPT0gMCkgewotICAgICAgICAgICAgICAgIHJldHVybiBJbWFnZU9ic2VydmVyLkFMTEJJVFM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoIShpbWFnZSBpbnN0YW5jZW9mIE9mZnNjcmVlbkltYWdlKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBJbWFnZU9ic2VydmVyLkFMTEJJVFM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBPZmZzY3JlZW5JbWFnZSBvaSA9IChPZmZzY3JlZW5JbWFnZSkgaW1hZ2U7Ci0gICAgICAgICAgICByZXR1cm4gb2kuY2hlY2tJbWFnZShvYnNlcnZlcik7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJbWFnZSBjcmVhdGVJbWFnZShJbWFnZVByb2R1Y2VyIHByb2R1Y2VyKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgT2Zmc2NyZWVuSW1hZ2UocHJvZHVjZXIpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSW1hZ2UgY3JlYXRlSW1hZ2UoYnl0ZVtdIGltYWdlZGF0YSwgaW50IGltYWdlb2Zmc2V0LCBpbnQgaW1hZ2VsZW5ndGgpIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBPZmZzY3JlZW5JbWFnZShuZXcgQnl0ZUFycmF5RGVjb2RpbmdJbWFnZVNvdXJjZShpbWFnZWRhdGEsIGltYWdlb2Zmc2V0LAotICAgICAgICAgICAgICAgICAgICBpbWFnZWxlbmd0aCkpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSW1hZ2UgY3JlYXRlSW1hZ2UoVVJMIHVybCkgewotICAgICAgICBsb2NrQVdUKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IE9mZnNjcmVlbkltYWdlKG5ldyBVUkxEZWNvZGluZ0ltYWdlU291cmNlKHVybCkpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSW1hZ2UgY3JlYXRlSW1hZ2UoU3RyaW5nIGZpbGVuYW1lKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgT2Zmc2NyZWVuSW1hZ2UobmV3IEZpbGVEZWNvZGluZ0ltYWdlU291cmNlKGZpbGVuYW1lKSk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBHcmFwaGljc0Vudmlyb25tZW50LmdldExvY2FsR3JhcGhpY3NFbnZpcm9ubWVudCgpLmdldERlZmF1bHRTY3JlZW5EZXZpY2UoKQotICAgICAgICAgICAgICAgICAgICAuZ2V0RGVmYXVsdENvbmZpZ3VyYXRpb24oKS5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJkZXByZWNhdGlvbiIpCi0gICAgQE92ZXJyaWRlCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgRm9udE1ldHJpY3MgZ2V0Rm9udE1ldHJpY3MoRm9udCBmb250KSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgCUdyYXBoaWNzRmFjdG9yeSBnZiA9IGdldEdyYXBoaWNzRmFjdG9yeSgpOwotICAgICAgICAgICAgcmV0dXJuIGdmLmdldEZvbnRNZXRyaWNzKGZvbnQpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gcHJlcGFyZUltYWdlKEltYWdlIGltYWdlLCBpbnQgd2lkdGgsIGludCBoZWlnaHQsIEltYWdlT2JzZXJ2ZXIgb2JzZXJ2ZXIpIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaWYgKHdpZHRoID09IDAgfHwgaGVpZ2h0ID09IDApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICghKGltYWdlIGluc3RhbmNlb2YgT2Zmc2NyZWVuSW1hZ2UpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBPZmZzY3JlZW5JbWFnZSBvaSA9IChPZmZzY3JlZW5JbWFnZSkgaW1hZ2U7Ci0gICAgICAgICAgICByZXR1cm4gb2kucHJlcGFyZUltYWdlKG9ic2VydmVyKTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgYmVlcCgpIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAJLy8gPz8/QVdUOiBpcyB0aGVyZSBub3RoaW5nIHRvIGJlIGltcGxlbWVudGVkIGhlcmU/Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJkZXByZWNhdGlvbiIpCi0gICAgQE92ZXJyaWRlCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0Rm9udExpc3QoKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIHVubG9ja0FXVCgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJkZXByZWNhdGlvbiIpCi0gICAgQE92ZXJyaWRlCi0gICAgQERlcHJlY2F0ZWQKLSAgICBwcm90ZWN0ZWQgRm9udFBlZXIgZ2V0Rm9udFBlZXIoU3RyaW5nIGEwLCBpbnQgYTEpIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJbWFnZSBnZXRJbWFnZShTdHJpbmcgZmlsZW5hbWUpIHsKLSAgICAgICAgcmV0dXJuIGdldEltYWdlKGZpbGVuYW1lLCB0aGlzKTsKLSAgICB9Ci0KLSAgICBzdGF0aWMgSW1hZ2UgZ2V0SW1hZ2UoU3RyaW5nIGZpbGVuYW1lLCBUb29sa2l0IHRvb2xraXQpIHsKLSAgICAgICAgc3luY2hyb25pemVkIChpbWFnZUNhY2hlKSB7Ci0gICAgICAgICAgICBJbWFnZSBpbSA9IChmaWxlbmFtZSA9PSBudWxsID8gbnVsbCA6IGltYWdlQ2FjaGUuZ2V0KGZpbGVuYW1lKSk7Ci0KLSAgICAgICAgICAgIGlmIChpbSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICAgICAgaW0gPSB0b29sa2l0LmNyZWF0ZUltYWdlKGZpbGVuYW1lKTsKLSAgICAgICAgICAgICAgICAgICAgaW1hZ2VDYWNoZS5wdXQoZmlsZW5hbWUsIGltKTsKLSAgICAgICAgICAgICAgICB9IGNhdGNoIChFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIGltOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEltYWdlIGdldEltYWdlKFVSTCB1cmwpIHsKLSAgICAgICAgcmV0dXJuIGdldEltYWdlKHVybCwgdGhpcyk7Ci0gICAgfQotCi0gICAgc3RhdGljIEltYWdlIGdldEltYWdlKFVSTCB1cmwsIFRvb2xraXQgdG9vbGtpdCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKGltYWdlQ2FjaGUpIHsKLSAgICAgICAgICAgIEltYWdlIGltID0gaW1hZ2VDYWNoZS5nZXQodXJsKTsKLSAgICAgICAgICAgIGlmIChpbSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICAgICAgaW0gPSB0b29sa2l0LmNyZWF0ZUltYWdlKHVybCk7Ci0gICAgICAgICAgICAgICAgICAgIGltYWdlQ2FjaGUucHV0KHVybCwgaW0pOwotICAgICAgICAgICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGltOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRTY3JlZW5SZXNvbHV0aW9uKCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAJcmV0dXJuIDYyOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgRGltZW5zaW9uIGdldFNjcmVlblNpemUoKSB7Ci0gICAgICAgIGxvY2tBV1QoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIERpc3BsYXlNb2RlIGRtID0gR3JhcGhpY3NFbnZpcm9ubWVudC5nZXRMb2NhbEdyYXBoaWNzRW52aXJvbm1lbnQoKQotICAgICAgICAgICAgICAgICAgICAuZ2V0RGVmYXVsdFNjcmVlbkRldmljZSgpLmdldERpc3BsYXlNb2RlKCk7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IERpbWVuc2lvbihkbS5nZXRXaWR0aCgpLCBkbS5nZXRIZWlnaHQoKSk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICB1bmxvY2tBV1QoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBNYXA8amF2YS5hd3QuZm9udC5UZXh0QXR0cmlidXRlLCA/PiBtYXBJbnB1dE1ldGhvZEhpZ2hsaWdodCgKLSAgICAgICAgICAgIElucHV0TWV0aG9kSGlnaGxpZ2h0IGhpZ2hsaWdodCkgdGhyb3dzIEhlYWRsZXNzRXhjZXB0aW9uIHsKLSAgICAgICAgbG9ja0FXVCgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIG1hcElucHV0TWV0aG9kSGlnaGxpZ2h0SW1wbChoaWdobGlnaHQpOwotICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgdW5sb2NrQVdUKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgRXZlbnRRdWV1ZSBnZXRTeXN0ZW1FdmVudFF1ZXVlSW1wbCgpIHsKLSAgICAgICAgcmV0dXJuIGdldFN5c3RlbUV2ZW50UXVldWVDb3JlKCkuZ2V0QWN0aXZlRXZlbnRRdWV1ZSgpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9UcmFuc3BhcmVuY3kuamF2YSBiL2F3dC9qYXZhL2F3dC9UcmFuc3BhcmVuY3kuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDRhMWU3Zi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvVHJhbnNwYXJlbmN5LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3Q7Ci0KLS8qKgotICogVGhlIFRyYW5zcGFyZW5jeSBpbnRlcmZhY2UgZGVmaW5lcyB0cmFuc3BhcmVuY3kncyBnZW5lcmFsIG1vZGVzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBUcmFuc3BhcmVuY3kgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE9QQVFVRSByZXByZXNlbnRzIGNvbXBsZXRlbHkgb3BhcXVlIGRhdGEsIGFsbCBwaXhlbHMgaGF2ZSBhbgotICAgICAqIGFscGhhIHZhbHVlIG9mIDEuMC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBPUEFRVUUgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEJJVE1BU0sgcmVwcmVzZW50cyBkYXRhIHdoaWNoIGNhbiBiZSBlaXRoZXIgY29tcGxldGVseQotICAgICAqIG9wYXF1ZSwgd2l0aCBhbiBhbHBoYSB2YWx1ZSBvZiAxLjAsIG9yIGNvbXBsZXRlbHkgdHJhbnNwYXJlbnQsIHdpdGggYW4KLSAgICAgKiBhbHBoYSB2YWx1ZSBvZiAwLjAuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQklUTUFTSyA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFJBTlNMVUNFTlQgcmVwcmVzZW50cyBkYXRhIHdoaWNoIGFscGhhIHZhbHVlIGNhbiB2YXJ5Ci0gICAgICogYmV0d2VlbiBhbmQgaW5jbHVkaW5nIDAuMCBhbmQgMS4wLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRSQU5TTFVDRU5UID0gMzsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRyYW5zcGFyZW5jeSBtb2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHRyYW5zcGFyZW5jeSBtb2RlOiBPUEFRVUUsIEJJVE1BU0sgb3IgVFJBTlNMVUNFTlQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRUcmFuc3BhcmVuY3koKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2NvbG9yL0NNTUV4Y2VwdGlvbi5qYXZhIGIvYXd0L2phdmEvYXd0L2NvbG9yL0NNTUV4Y2VwdGlvbi5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxOGI5YTdlLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9jb2xvci9DTU1FeGNlcHRpb24uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ0ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmNvbG9yOwotCi0vKioKLSAqIFRoZSBDTU1FeGNlcHRpb24gaXMgdGhyb3duIGFzIHNvb24gYXMgYSBuYXRpdmUgQ01NIGVycm9yIG9jY3Vycy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBDTU1FeGNlcHRpb24gZXh0ZW5kcyBqYXZhLmxhbmcuUnVudGltZUV4Y2VwdGlvbiB7Ci0gICAgCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IHNlcmlhbFZlcnNpb25VSUQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gNTc3NTU1ODA0NDE0Mjk5NDk2NUw7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQ01NIGV4Y2VwdGlvbiB3aXRoIGRldGFpbCBtZXNzYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgZGV0YWlsIG1lc3NhZ2Ugb2YgdGhlIGV4Y2VwdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ01NRXhjZXB0aW9uIChTdHJpbmcgcykgewotICAgICAgICBzdXBlciAocyk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2NvbG9yL0NvbG9yU3BhY2UuamF2YSBiL2F3dC9qYXZhL2F3dC9jb2xvci9Db2xvclNwYWNlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQ0YzQ5MWIuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2NvbG9yL0NvbG9yU3BhY2UuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQxNCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5jb2xvcjsKLQotaW1wb3J0IGphdmEuaW8uU2VyaWFsaXphYmxlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvci5MVVRDb2xvckNvbnZlcnRlcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgQ29sb3JTcGFjZSBjbGFzcyBkZWZpbmVzIGEgY29sb3Igc3BhY2UgdHlwZSBmb3IgYSBDb2xvciBhbmQgcHJvdmlkZXMKLSAqIG1ldGhvZHMgZm9yIGFycmF5cyBvZiBjb2xvciBjb21wb25lbnQgb3BlcmF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBDb2xvclNwYWNlIGltcGxlbWVudHMgU2VyaWFsaXphYmxlIHsKLQotICAgIC8qKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4gKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtNDA5NDUyNzA0MzA4Njg5NzI0TDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX1hZWiBpbmRpY2F0ZXMgWFlaIGNvbG9yIHNwYWNlIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9YWVogPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfTGFiIGluZGljYXRlcyBMYWIgY29sb3Igc3BhY2UgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0xhYiA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9MdXYgaW5kaWNhdGVzIEx1diBjb2xvciBzcGFjZSB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfTHV2ID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX1lDYkNyIGluZGljYXRlcyBZQ2JDciBjb2xvciBzcGFjZSB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfWUNiQ3IgPSAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfWXh5IGluZGljYXRlcyBZeHkgY29sb3Igc3BhY2UgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX1l4eSA9IDQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9SR0IgaW5kaWNhdGVzIFJHQiBjb2xvciBzcGFjZSB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfUkdCID0gNTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0dSQVkgaW5kaWNhdGVzIEdyYXkgY29sb3Igc3BhY2UgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0dSQVkgPSA2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfSFNWIGluZGljYXRlcyBIU1YgY29sb3Igc3BhY2UgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0hTViA9IDc7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9ITFMgaW5kaWNhdGVzIEhMUyBjb2xvciBzcGFjZSB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfSExTID0gODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0NNWUsgaW5kaWNhdGVzIENNWUsgY29sb3Igc3BhY2UgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0NNWUsgPSA5OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfQ01ZIGluZGljYXRlcyBDTVkgY29sb3Igc3BhY2UgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0NNWSA9IDExOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfMkNMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggMiBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfMkNMUiA9IDEyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfM0NMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggMyBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfM0NMUiA9IDEzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfNENMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggNCBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfNENMUiA9IDE0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfNUNMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggNSBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfNUNMUiA9IDE1OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfNkNMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggNiBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfNkNMUiA9IDE2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfN0NMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggNyBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfN0NMUiA9IDE3OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfOENMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggOCBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfOENMUiA9IDE4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfOUNMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggOSBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfOUNMUiA9IDE5OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfQUNMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggMTAgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0FDTFIgPSAyMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0JDTFIgaW5kaWNhdGVzIGNvbG9yIHNwYWNlcyB3aXRoIDExIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9CQ0xSID0gMjE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9DQ0xSIGluZGljYXRlcyBjb2xvciBzcGFjZXMgd2l0aCAxMiBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfQ0NMUiA9IDIyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfRENMUiBpbmRpY2F0ZXMgY29sb3Igc3BhY2VzIHdpdGggMTMgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0RDTFIgPSAyMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0VDTFIgaW5kaWNhdGVzIGNvbG9yIHNwYWNlcyB3aXRoIDE0IGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9FQ0xSID0gMjQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9GQ0xSIGluZGljYXRlcyBjb2xvciBzcGFjZXMgd2l0aCAxNSBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfRkNMUiA9IDI1OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENTX3NSR0IgaW5kaWNhdGVzIHN0YW5kYXJkIFJHQiBjb2xvciBzcGFjZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDU19zUkdCID0gMTAwMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDU19MSU5FQVJfUkdCIGluZGljYXRlcyBsaW5lYXIgUkdCIGNvbG9yIHNwYWNlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENTX0xJTkVBUl9SR0IgPSAxMDA0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENTX0NJRVhZWiBpbmRpY2F0ZXMgQ0lFWFlaIGNvbnZlcnNpb24gY29sb3Igc3BhY2UuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ1NfQ0lFWFlaID0gMTAwMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDU19QWUNDIGluZGljYXRlcyBQaG90byBZQ0MgY29udmVyc2lvbiBjb2xvciBzcGFjZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDU19QWUNDID0gMTAwMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDU19HUkFZIGluZGljYXRlcyBsaW5lYXIgZ3JheSBzY2FsZSBjb2xvciBzcGFjZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDU19HUkFZID0gMTAwMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBjc18gZ3JheS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBDb2xvclNwYWNlIGNzX0dyYXkgPSBudWxsOwotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBjc18gcHljYy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBDb2xvclNwYWNlIGNzX1BZQ0MgPSBudWxsOwotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBjc18gY2lleHl6LgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIENvbG9yU3BhY2UgY3NfQ0lFWFlaID0gbnVsbDsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgY3NfIGxyZ2IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgQ29sb3JTcGFjZSBjc19MUkdCID0gbnVsbDsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgY3NfcyByZ2IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgQ29sb3JTcGFjZSBjc19zUkdCID0gbnVsbDsKLQotICAgIC8qKgotICAgICAqIFRoZSB0eXBlLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IHR5cGU7Ci0gICAgCi0gICAgLyoqCi0gICAgICogVGhlIG51bSBjb21wb25lbnRzLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IG51bUNvbXBvbmVudHM7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBDb2xvclNwYWNlIHdpdGggdGhlIHNwZWNpZmllZCBDb2xvclNwYWNlIHR5cGUgYW5kIG51bWJlcgotICAgICAqIG9mIGNvbXBvbmVudHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSB0eXBlIG9mIGNvbG9yIHNwYWNlLgotICAgICAqIEBwYXJhbSBudW1jb21wb25lbnRzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIENvbG9yU3BhY2UoaW50IHR5cGUsIGludCBudW1jb21wb25lbnRzKSB7Ci0gICAgICAgIHRoaXMubnVtQ29tcG9uZW50cyA9IG51bWNvbXBvbmVudHM7Ci0gICAgICAgIHRoaXMudHlwZSA9IHR5cGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbmFtZSBvZiB0aGUgY29tcG9uZW50IGZvciB0aGUgc3BlY2lmaWVkIGNvbXBvbmVudCBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWR4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggb2YgdGhlIGNvbXBvbmVudC4KLSAgICAgKiBAcmV0dXJuIHRoZSBuYW1lIG9mIHRoZSBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZyBnZXROYW1lKGludCBpZHgpIHsKLSAgICAgICAgaWYgKGlkeCA8IDAgfHwgaWR4ID4gbnVtQ29tcG9uZW50cyAtIDEpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xNkE9SW52YWxpZCBjb21wb25lbnQgaW5kZXg6IHswfQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNkEiLCBpZHgpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgIHJldHVybiAiVW5uYW1lZCBjb2xvciBjb21wb25lbnQgIyIgKyBpZHg7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQZXJmb3JtcyB0aGUgdHJhbnNmb3JtYXRpb24gb2YgYSBjb2xvciBmcm9tIHRoaXMgQ29sb3JTcGFjZSBpbnRvIHRoZSBSR0IKLSAgICAgKiBjb2xvciBzcGFjZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29sb3J2YWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIGNvbG9yIHZhbHVlIGluIHRoaXMgQ29sb3JTcGFjZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBmbG9hdCBhcnJheSB3aXRoIGNvbG9yIGNvbXBvbmVudHMgaW4gdGhlIFJHQiBjb2xvciBzcGFjZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZmxvYXRbXSB0b1JHQihmbG9hdFtdIGNvbG9ydmFsdWUpOwotCi0gICAgLyoqCi0gICAgICogUGVyZm9ybXMgdGhlIHRyYW5zZm9ybWF0aW9uIG9mIGEgY29sb3IgZnJvbSB0aGlzIENvbG9yU3BhY2UgaW50byB0aGUKLSAgICAgKiBDU19DSUVYWVogY29sb3Igc3BhY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbG9ydmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb2xvciB2YWx1ZSBpbiB0aGlzIENvbG9yU3BhY2UuCi0gICAgICogQHJldHVybiB0aGUgZmxvYXQgYXJyYXkgd2l0aCBjb2xvciBjb21wb25lbnRzIGluIHRoZSBDU19DSUVYWVogY29sb3IKLSAgICAgKiAgICAgICAgIHNwYWNlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBmbG9hdFtdIHRvQ0lFWFlaKGZsb2F0W10gY29sb3J2YWx1ZSk7Ci0KLSAgICAvKioKLSAgICAgKiBQZXJmb3JtcyB0aGUgdHJhbnNmb3JtYXRpb24gb2YgYSBjb2xvciBmcm9tIHRoZSBSR0IgY29sb3Igc3BhY2UgaW50byB0aGlzCi0gICAgICogQ29sb3JTcGFjZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmdidmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbG9hdCBhcnJheSByZXByZXNlbnRpbmcgYSBjb2xvciBpbiB0aGUgUkdCIGNvbG9yIHNwYWNlLgotICAgICAqIEByZXR1cm4gdGhlIGZsb2F0IGFycmF5IHdpdGggdGhlIHRyYW5zZm9ybWVkIGNvbG9yIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGZsb2F0W10gZnJvbVJHQihmbG9hdFtdIHJnYnZhbHVlKTsKLQotICAgIC8qKgotICAgICAqIFBlcmZvcm1zIHRoZSB0cmFuc2Zvcm1hdGlvbiBvZiBhIGNvbG9yIGZyb20gdGhlIENTX0NJRVhZWiBjb2xvciBzcGFjZQotICAgICAqIGludG8gdGhpcyBDb2xvclNwYWNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvcnZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgZmxvYXQgYXJyYXkgcmVwcmVzZW50aW5nIGEgY29sb3IgaW4gdGhlIENTX0NJRVhZWiBjb2xvcgotICAgICAqICAgICAgICAgICAgc3BhY2UuCi0gICAgICogQHJldHVybiB0aGUgZmxvYXQgYXJyYXkgd2l0aCB0aGUgdHJhbnNmb3JtZWQgY29sb3IgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZmxvYXRbXSBmcm9tQ0lFWFlaKGZsb2F0W10gY29sb3J2YWx1ZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtaW5pbXVtIG5vcm1hbGl6ZWQgY29sb3IgY29tcG9uZW50IHZhbHVlIGZvciB0aGUgc3BlY2lmaWVkCi0gICAgICogY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb21wb25lbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb21wb25lbnQgdG8gZGV0ZXJtaW5lIHRoZSBtaW5pbXVtIHZhbHVlLgotICAgICAqIEByZXR1cm4gdGhlIG1pbmltdW0gbm9ybWFsaXplZCB2YWx1ZSBvZiB0aGUgY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRNaW5WYWx1ZShpbnQgY29tcG9uZW50KSB7Ci0gICAgICAgIGlmIChjb21wb25lbnQgPCAwIHx8IGNvbXBvbmVudCA+IG51bUNvbXBvbmVudHMgLSAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTZBPUludmFsaWQgY29tcG9uZW50IGluZGV4OiB7MH0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTZBIiwgY29tcG9uZW50KSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtYXhpbXVtIG5vcm1hbGl6ZWQgY29sb3IgY29tcG9uZW50IHZhbHVlIGZvciB0aGUgc3BlY2lmaWVkCi0gICAgICogY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb21wb25lbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb21wb25lbnQgdG8gZGV0ZXJtaW5lIHRoZSBtYXhpbXVtIHZhbHVlLgotICAgICAqIEByZXR1cm4gdGhlIG1heGltdW0gbm9ybWFsaXplZCB2YWx1ZSBvZiB0aGUgY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRNYXhWYWx1ZShpbnQgY29tcG9uZW50KSB7Ci0gICAgICAgIGlmIChjb21wb25lbnQgPCAwIHx8IGNvbXBvbmVudCA+IG51bUNvbXBvbmVudHMgLSAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTZBPUludmFsaWQgY29tcG9uZW50IGluZGV4OiB7MH0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTZBIiwgY29tcG9uZW50KSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gMTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyBDb2xvclNwYWNlIGhhcyBDU19zUkdCIHR5cGUgb3Igbm90LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBDb2xvclNwYWNlIGhhcyBDU19zUkdCIHR5cGUsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NTX3NSR0IoKSB7Ci0gICAgICAgIC8vIElmIG91ciBjb2xvciBzcGFjZSBpcyBzUkdCLCB0aGVuIGNzX3NSR0IKLSAgICAgICAgLy8gaXMgYWxyZWFkeSBpbml0aWFsaXplZAotICAgICAgICByZXR1cm4gKHRoaXMgPT0gY3Nfc1JHQik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdHlwZSBvZiB0aGUgQ29sb3JTcGFjZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB0eXBlIG9mIHRoZSBDb2xvclNwYWNlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIHR5cGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGNvbXBvbmVudHMgZm9yIHRoaXMgQ29sb3JTcGFjZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE51bUNvbXBvbmVudHMoKSB7Ci0gICAgICAgIHJldHVybiBudW1Db21wb25lbnRzOwotICAgIH0KLQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2luZ2xlIGluc3RhbmNlIG9mIENvbG9yU3BhY2Ugd2l0aCB0aGUgc3BlY2lmaWVkIENvbG9yU3BhY2U6Ci0gICAgICogQ1Nfc1JHQiwgQ1NfTElORUFSX1JHQiwgQ1NfQ0lFWFlaLCBDU19HUkFZLCBvciBDU19QWUNDLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvcnNwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgaWRlbnRpZmllciBvZiB0aGUgc3BlY2lmaWVkIENvbG9yc3BhY2UuCi0gICAgICogQHJldHVybiB0aGUgc2luZ2xlIGluc3RhbmNlIG9mIHRoZSBkZXNpcmVkIENvbG9yU3BhY2UuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBDb2xvclNwYWNlIGdldEluc3RhbmNlKGludCBjb2xvcnNwYWNlKSB7Ci0gICAgICAgIHN3aXRjaCAoY29sb3JzcGFjZSkgewotICAgICAgICAgICAgY2FzZSBDU19zUkdCOgotICAgICAgICAgICAgICAgIGlmIChjc19zUkdCID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgY3Nfc1JHQiA9IG5ldyBJQ0NfQ29sb3JTcGFjZSgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXcgSUNDX1Byb2ZpbGVTdHViKENTX3NSR0IpKTsKLSAgICAgICAgICAgICAgICAgICAgTFVUQ29sb3JDb252ZXJ0ZXIuc1JHQl9DUyA9IGNzX3NSR0I7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy9JQ0NfUHJvZmlsZS5nZXRJbnN0YW5jZSAoQ1Nfc1JHQikpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gY3Nfc1JHQjsKLSAgICAgICAgICAgIGNhc2UgQ1NfQ0lFWFlaOgotICAgICAgICAgICAgICAgIGlmIChjc19DSUVYWVogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBjc19DSUVYWVogPSBuZXcgSUNDX0NvbG9yU3BhY2UoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgbmV3IElDQ19Qcm9maWxlU3R1YihDU19DSUVYWVopKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAvL0lDQ19Qcm9maWxlLmdldEluc3RhbmNlIChDU19DSUVYWVopKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGNzX0NJRVhZWjsKLSAgICAgICAgICAgIGNhc2UgQ1NfR1JBWToKLSAgICAgICAgICAgICAgICBpZiAoY3NfR3JheSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGNzX0dyYXkgPSBuZXcgSUNDX0NvbG9yU3BhY2UoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgbmV3IElDQ19Qcm9maWxlU3R1YihDU19HUkFZKSk7Ci0gICAgICAgICAgICAgICAgICAgIExVVENvbG9yQ29udmVydGVyLkxJTkVBUl9HUkFZX0NTID0gY3NfR3JheTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAvL0lDQ19Qcm9maWxlLmdldEluc3RhbmNlIChDU19HUkFZKSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBjc19HcmF5OwotICAgICAgICAgICAgY2FzZSBDU19QWUNDOgotICAgICAgICAgICAgICAgIGlmIChjc19QWUNDID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgY3NfUFlDQyA9IG5ldyBJQ0NfQ29sb3JTcGFjZSgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXcgSUNDX1Byb2ZpbGVTdHViKENTX1BZQ0MpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAvL0lDQ19Qcm9maWxlLmdldEluc3RhbmNlIChDU19QWUNDKSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBjc19QWUNDOwotICAgICAgICAgICAgY2FzZSBDU19MSU5FQVJfUkdCOgotICAgICAgICAgICAgICAgIGlmIChjc19MUkdCID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgY3NfTFJHQiA9IG5ldyBJQ0NfQ29sb3JTcGFjZSgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXcgSUNDX1Byb2ZpbGVTdHViKENTX0xJTkVBUl9SR0IpKTsKLSAgICAgICAgICAgICAgICAgICAgTFVUQ29sb3JDb252ZXJ0ZXIuTElORUFSX0dSQVlfQ1MgPSBjc19HcmF5OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vSUNDX1Byb2ZpbGUuZ2V0SW5zdGFuY2UgKENTX0xJTkVBUl9SR0IpKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGNzX0xSR0I7Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICB9Ci0KLSAgICAgICAgLy8gVW5rbm93biBhcmd1bWVudCBwYXNzZWQKLSAgICAgICAgLy8gYXd0LjE2Qj1Ob3QgYSBwcmVkZWZpbmVkIGNvbG9yc3BhY2UKLSAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoIk5vdCBhIHByZWRlZmluZWQgY29sb3JzcGFjZSIpKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotfQpcIE5vIG5ld2xpbmUgYXQgZW5kIG9mIGZpbGUKZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9jb2xvci9JQ0NfQ29sb3JTcGFjZS5qYXZhIGIvYXd0L2phdmEvYXd0L2NvbG9yL0lDQ19Db2xvclNwYWNlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDViNGQ3ZTkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2NvbG9yL0lDQ19Db2xvclNwYWNlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0NjggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuY29sb3I7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmNvbG9yLkNvbG9yQ29udmVydGVyOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuY29sb3IuQ29sb3JTY2FsZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvci5JQ0NfVHJhbnNmb3JtOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi1pbXBvcnQgamF2YS5pby4qOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaW1wbGVtZW50cyB0aGUgYWJzdHJhY3QgY2xhc3MgQ29sb3JTcGFjZSBhbmQgcmVwcmVzZW50cyBkZXZpY2UKLSAqIGluZGVwZW5kZW50IGFuZCBkZXZpY2UgZGVwZW5kZW50IGNvbG9yIHNwYWNlcy4gVGhpcyBjb2xvciBzcGFjZSBpcyBiYXNlZCBvbgotICogdGhlIEludGVybmF0aW9uYWwgQ29sb3IgQ29uc29ydGl1bSBTcGVjaWZpY2F0aW9uIChJQ0MpIEZpbGUgRm9ybWF0IGZvciBDb2xvcgotICogUHJvZmlsZXM6IDxhIGhyZWY9Imh0dHA6Ly93d3cuY29sb3Iub3JnIj5odHRwOi8vd3d3LmNvbG9yLm9yZzwvYT4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJQ0NfQ29sb3JTcGFjZSBleHRlbmRzIENvbG9yU3BhY2UgewotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IDM0NTU4ODkxMTQwNzA0MzE0ODNMOwotCi0gICAgLy8gTmVlZCB0byBrZWVwIGNvbXBhdGliaWxpdHkgd2l0aCBzZXJpYWxpemVkIGZvcm0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsUGVyc2lzdGVudEZpZWxkcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBPYmplY3RTdHJlYW1GaWVsZFtdCi0gICAgICBzZXJpYWxQZXJzaXN0ZW50RmllbGRzID0gewotICAgICAgICBuZXcgT2JqZWN0U3RyZWFtRmllbGQoInRoaXNQcm9maWxlIiwgSUNDX1Byb2ZpbGUuY2xhc3MpLCAvLyROT04tTkxTLTEkCi0gICAgICAgIG5ldyBPYmplY3RTdHJlYW1GaWVsZCgibWluVmFsIiwgZmxvYXRbXS5jbGFzcyksIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgbmV3IE9iamVjdFN0cmVhbUZpZWxkKCJtYXhWYWwiLCBmbG9hdFtdLmNsYXNzKSwgLy8kTk9OLU5MUy0xJAotICAgICAgICBuZXcgT2JqZWN0U3RyZWFtRmllbGQoImRpZmZNaW5NYXgiLCBmbG9hdFtdLmNsYXNzKSwgLy8kTk9OLU5MUy0xJAotICAgICAgICBuZXcgT2JqZWN0U3RyZWFtRmllbGQoImludkRpZmZNaW5NYXgiLCBmbG9hdFtdLmNsYXNzKSwgLy8kTk9OLU5MUy0xJAotICAgICAgICBuZXcgT2JqZWN0U3RyZWFtRmllbGQoIm5lZWRTY2FsZUluaXQiLCBCb29sZWFuLlRZUEUpIC8vJE5PTi1OTFMtMSQKLSAgICB9OwotCi0KLSAgIC8qKgotICAgICAqIEFjY29yZGluZyB0byBJQ0Mgc3BlY2lmaWNhdGlvbiAoZnJvbSBodHRwOi8vd3d3LmNvbG9yLm9yZykgIkZvciB0aGUKLSAgICAgKiBDSUVYWVogZW5jb2RpbmcsIGVhY2ggY29tcG9uZW50IChYLCBZLCBhbmQgWikgaXMgZW5jb2RlZCBhcyBhCi0gICAgICogdTFGaXhlZDE1TnVtYmVyIi4gVGhpcyBtZWFucyB0aGF0IG1heCB2YWx1ZSBmb3IgdGhpcyBlbmNvZGluZyBpcyAxICsKLSAgICAgKiAoMzI3NjcvMzI3NjgpCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgZmxvYXQgTUFYX1hZWiA9IDFmICsgKDMyNzY3Zi8zMjc2OGYpOwotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBNQVhfU0hPUlQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgZmxvYXQgTUFYX1NIT1JUID0gNjU1MzVmOwotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBJTlZfTUFYX1NIT1JULgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGZsb2F0IElOVl9NQVhfU0hPUlQgPSAxZi9NQVhfU0hPUlQ7Ci0gICAgCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNIT1JUMlhZWl9GQUNUT1IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgZmxvYXQgU0hPUlQyWFlaX0ZBQ1RPUiA9IE1BWF9YWVovTUFYX1NIT1JUOwotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBYWVoyU0hPUlRfRkFDVE9SLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGZsb2F0IFhZWjJTSE9SVF9GQUNUT1IgPSBNQVhfU0hPUlQvTUFYX1hZWjsKLQotICAgIC8qKgotICAgICAqIFRoZSBwcm9maWxlLgotICAgICAqLwotICAgIHByaXZhdGUgSUNDX1Byb2ZpbGUgcHJvZmlsZSA9IG51bGw7Ci0gICAgCi0gICAgLyoqCi0gICAgICogVGhlIG1pbiB2YWx1ZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmbG9hdCBtaW5WYWx1ZXNbXSA9IG51bGw7Ci0gICAgCi0gICAgLyoqCi0gICAgICogVGhlIG1heCB2YWx1ZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmbG9hdCBtYXhWYWx1ZXNbXSA9IG51bGw7Ci0KLSAgICAvLyBjYWNoZSB0cmFuc2Zvcm1zIGhlcmUgLSBwZXJmb3JtYW5jZSBnYWluCi0gICAgLyoqCi0gICAgICogVGhlIHRvIHJnYiB0cmFuc2Zvcm0uCi0gICAgICovCi0gICAgcHJpdmF0ZSBJQ0NfVHJhbnNmb3JtIHRvUkdCVHJhbnNmb3JtID0gbnVsbDsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgZnJvbSByZ2IgdHJhbnNmb3JtLgotICAgICAqLwotICAgIHByaXZhdGUgSUNDX1RyYW5zZm9ybSBmcm9tUkdCVHJhbnNmb3JtID0gbnVsbDsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgdG8geHl6IHRyYW5zZm9ybS4KLSAgICAgKi8KLSAgICBwcml2YXRlIElDQ19UcmFuc2Zvcm0gdG9YWVpUcmFuc2Zvcm0gPSBudWxsOwotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBmcm9tIHh5eiB0cmFuc2Zvcm0uCi0gICAgICovCi0gICAgcHJpdmF0ZSBJQ0NfVHJhbnNmb3JtIGZyb21YWVpUcmFuc2Zvcm0gPSBudWxsOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbnZlcnRlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIENvbG9yQ29udmVydGVyIGNvbnZlcnRlciA9IG5ldyBDb2xvckNvbnZlcnRlcigpOwotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBzY2FsZXIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBDb2xvclNjYWxlciBzY2FsZXIgPSBuZXcgQ29sb3JTY2FsZXIoKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgc2NhbGluZyBkYXRhIGxvYWRlZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gc2NhbGluZ0RhdGFMb2FkZWQgPSBmYWxzZTsKLQotICAgIC8qKgotICAgICAqIFRoZSByZXNvbHZlZCBkZXNlcmlhbGl6ZWQgaW5zdC4KLSAgICAgKi8KLSAgICBwcml2YXRlIElDQ19Db2xvclNwYWNlIHJlc29sdmVkRGVzZXJpYWxpemVkSW5zdDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJQ0MgY29sb3Igc3BhY2UgZnJvbSBhbiBJQ0NfUHJvZmlsZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBmCi0gICAgICogICAgICAgICAgICB0aGUgSUNDX1Byb2ZpbGUgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBJQ0NfQ29sb3JTcGFjZShJQ0NfUHJvZmlsZSBwZikgewotICAgICAgICBzdXBlcihwZi5nZXRDb2xvclNwYWNlVHlwZSgpLCBwZi5nZXROdW1Db21wb25lbnRzKCkpOwotCi0gICAgICAgIGludCBwZkNsYXNzID0gcGYuZ2V0UHJvZmlsZUNsYXNzKCk7Ci0KLSAgICAgICAgc3dpdGNoIChwZkNsYXNzKSB7Ci0gICAgICAgICAgICBjYXNlIElDQ19Qcm9maWxlLkNMQVNTX0NPTE9SU1BBQ0VDT05WRVJTSU9OOgotICAgICAgICAgICAgY2FzZSBJQ0NfUHJvZmlsZS5DTEFTU19ESVNQTEFZOgotICAgICAgICAgICAgY2FzZSBJQ0NfUHJvZmlsZS5DTEFTU19PVVRQVVQ6Ci0gICAgICAgICAgICBjYXNlIElDQ19Qcm9maWxlLkNMQVNTX0lOUFVUOgotICAgICAgICAgICAgICAgIGJyZWFrOyAvLyBPSywgaXQgaXMgY29sb3IgY29udmVyc2lvbiBwcm9maWxlCi0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4xNjg9SW52YWxpZCBwcm9maWxlIGNsYXNzLgotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTY4IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBwcm9maWxlID0gcGY7Ci0gICAgICAgIGZpbGxNaW5NYXhWYWx1ZXMoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBJQ0NfUHJvZmlsZSBmb3IgdGhpcyBJQ0NfQ29sb3JTcGFjZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBJQ0NfUHJvZmlsZSBmb3IgdGhpcyBJQ0NfQ29sb3JTcGFjZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgSUNDX1Byb2ZpbGUgZ2V0UHJvZmlsZSgpIHsKLSAgICAgICAgaWYgKHByb2ZpbGUgaW5zdGFuY2VvZiBJQ0NfUHJvZmlsZVN0dWIpIHsKLSAgICAgICAgICAgIHByb2ZpbGUgPSAoKElDQ19Qcm9maWxlU3R1YikgcHJvZmlsZSkubG9hZFByb2ZpbGUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBwcm9maWxlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBlcmZvcm1zIHRoZSB0cmFuc2Zvcm1hdGlvbiBvZiBhIGNvbG9yIGZyb20gdGhpcyBDb2xvclNwYWNlIGludG8gdGhlIFJHQgotICAgICAqIGNvbG9yIHNwYWNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvcnZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgY29sb3IgdmFsdWUgaW4gdGhpcyBDb2xvclNwYWNlLgotICAgICAqIEByZXR1cm4gdGhlIGZsb2F0IGFycmF5IHdpdGggY29sb3IgY29tcG9uZW50cyBpbiB0aGUgUkdCIGNvbG9yIHNwYWNlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdFtdIHRvUkdCKGZsb2F0W10gY29sb3J2YWx1ZSkgewotICAgICAgICBpZiAodG9SR0JUcmFuc2Zvcm0gPT0gbnVsbCkgewotICAgICAgICAgICAgSUNDX1Byb2ZpbGUgc1JHQlByb2ZpbGUgPQotICAgICAgICAgICAgICAgICgoSUNDX0NvbG9yU3BhY2UpIENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ1Nfc1JHQikpLmdldFByb2ZpbGUoKTsKLSAgICAgICAgICAgIElDQ19Qcm9maWxlW10gcHJvZmlsZXMgPSB7Z2V0UHJvZmlsZSgpLCBzUkdCUHJvZmlsZX07Ci0gICAgICAgICAgICB0b1JHQlRyYW5zZm9ybSA9IG5ldyBJQ0NfVHJhbnNmb3JtKHByb2ZpbGVzKTsKLSAgICAgICAgICAgIGlmICghc2NhbGluZ0RhdGFMb2FkZWQpIHsKLSAgICAgICAgICAgICAgICBzY2FsZXIubG9hZFNjYWxpbmdEYXRhKHRoaXMpOwotICAgICAgICAgICAgICAgIHNjYWxpbmdEYXRhTG9hZGVkID0gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHNob3J0W10gZGF0YSA9IG5ldyBzaG9ydFtnZXROdW1Db21wb25lbnRzKCldOwotCi0gICAgICAgIHNjYWxlci5zY2FsZShjb2xvcnZhbHVlLCBkYXRhLCAwKTsKLQotICAgICAgICBzaG9ydFtdIGNvbnZlcnRlZCA9Ci0gICAgICAgICAgICBjb252ZXJ0ZXIudHJhbnNsYXRlQ29sb3IodG9SR0JUcmFuc2Zvcm0sIGRhdGEsIG51bGwpOwotCi0gICAgICAgIC8vIHVuc2NhbGUgdG8gc1JHQgotICAgICAgICBmbG9hdFtdIHJlcyA9IG5ldyBmbG9hdFszXTsKLQotICAgICAgICByZXNbMF0gPSAoKGNvbnZlcnRlZFswXSAmIDB4RkZGRikpICogSU5WX01BWF9TSE9SVDsKLSAgICAgICAgcmVzWzFdID0gKChjb252ZXJ0ZWRbMV0gJiAweEZGRkYpKSAqIElOVl9NQVhfU0hPUlQ7Ci0gICAgICAgIHJlc1syXSA9ICgoY29udmVydGVkWzJdICYgMHhGRkZGKSkgKiBJTlZfTUFYX1NIT1JUOwotCi0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUGVyZm9ybXMgdGhlIHRyYW5zZm9ybWF0aW9uIG9mIGEgY29sb3IgZnJvbSB0aGlzIENvbG9yU3BhY2UgaW50byB0aGUKLSAgICAgKiBDU19DSUVYWVogY29sb3Igc3BhY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbG9ydmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb2xvciB2YWx1ZSBpbiB0aGlzIENvbG9yU3BhY2UuCi0gICAgICogQHJldHVybiB0aGUgZmxvYXQgYXJyYXkgd2l0aCBjb2xvciBjb21wb25lbnRzIGluIHRoZSBDU19DSUVYWVogY29sb3IKLSAgICAgKiAgICAgICAgIHNwYWNlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdFtdIHRvQ0lFWFlaKGZsb2F0W10gY29sb3J2YWx1ZSkgewotICAgICAgICBpZiAodG9YWVpUcmFuc2Zvcm0gPT0gbnVsbCkgewotICAgICAgICAgICAgSUNDX1Byb2ZpbGUgeHl6UHJvZmlsZSA9Ci0gICAgICAgICAgICAgICAgKChJQ0NfQ29sb3JTcGFjZSkgQ29sb3JTcGFjZS5nZXRJbnN0YW5jZShDU19DSUVYWVopKS5nZXRQcm9maWxlKCk7Ci0gICAgICAgICAgICBJQ0NfUHJvZmlsZVtdIHByb2ZpbGVzID0ge2dldFByb2ZpbGUoKSwgeHl6UHJvZmlsZX07Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIGludFtdIGludGVudHMgPSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBJQ0NfUHJvZmlsZS5pY1JlbGF0aXZlQ29sb3JpbWV0cmljLAotICAgICAgICAgICAgICAgICAgICAgICAgSUNDX1Byb2ZpbGUuaWNQZXJjZXB0dWFsfTsKLSAgICAgICAgICAgICAgICB0b1hZWlRyYW5zZm9ybSA9IG5ldyBJQ0NfVHJhbnNmb3JtKHByb2ZpbGVzLCBpbnRlbnRzKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKENNTUV4Y2VwdGlvbiBlKSB7IC8vIE5vIHN1Y2ggdGFnLCB1c2Ugd2hhdCB3ZSBjYW4KLSAgICAgICAgICAgICAgICB0b1hZWlRyYW5zZm9ybSA9IG5ldyBJQ0NfVHJhbnNmb3JtKHByb2ZpbGVzKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKCFzY2FsaW5nRGF0YUxvYWRlZCkgewotICAgICAgICAgICAgICAgIHNjYWxlci5sb2FkU2NhbGluZ0RhdGEodGhpcyk7Ci0gICAgICAgICAgICAgICAgc2NhbGluZ0RhdGFMb2FkZWQgPSB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgc2hvcnRbXSBkYXRhID0gbmV3IHNob3J0W2dldE51bUNvbXBvbmVudHMoKV07Ci0KLSAgICAgICAgc2NhbGVyLnNjYWxlKGNvbG9ydmFsdWUsIGRhdGEsIDApOwotCi0gICAgICAgIHNob3J0W10gY29udmVydGVkID0KLSAgICAgICAgICAgIGNvbnZlcnRlci50cmFuc2xhdGVDb2xvcih0b1hZWlRyYW5zZm9ybSwgZGF0YSwgbnVsbCk7Ci0KLSAgICAgICAgLy8gdW5zY2FsZSB0byBYWVoKLSAgICAgICAgZmxvYXRbXSByZXMgPSBuZXcgZmxvYXRbM107Ci0KLSAgICAgICAgcmVzWzBdID0gKChjb252ZXJ0ZWRbMF0gJiAweEZGRkYpKSAqIFNIT1JUMlhZWl9GQUNUT1I7Ci0gICAgICAgIHJlc1sxXSA9ICgoY29udmVydGVkWzFdICYgMHhGRkZGKSkgKiBTSE9SVDJYWVpfRkFDVE9SOwotICAgICAgICByZXNbMl0gPSAoKGNvbnZlcnRlZFsyXSAmIDB4RkZGRikpICogU0hPUlQyWFlaX0ZBQ1RPUjsKLQotICAgICAgICByZXR1cm4gcmVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBlcmZvcm1zIHRoZSB0cmFuc2Zvcm1hdGlvbiBvZiBhIGNvbG9yIGZyb20gdGhlIFJHQiBjb2xvciBzcGFjZSBpbnRvIHRoaXMKLSAgICAgKiBDb2xvclNwYWNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSByZ2J2YWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIGZsb2F0IGFycmF5IHJlcHJlc2VudGluZyBhIGNvbG9yIGluIHRoZSBSR0IgY29sb3Igc3BhY2UuCi0gICAgICogQHJldHVybiB0aGUgZmxvYXQgYXJyYXkgd2l0aCB0aGUgdHJhbnNmb3JtZWQgY29sb3IgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXRbXSBmcm9tUkdCKGZsb2F0W10gcmdidmFsdWUpIHsKLSAgICAgICAgaWYgKGZyb21SR0JUcmFuc2Zvcm0gPT0gbnVsbCkgewotICAgICAgICAgICAgSUNDX1Byb2ZpbGUgc1JHQlByb2ZpbGUgPQotICAgICAgICAgICAgICAgICgoSUNDX0NvbG9yU3BhY2UpIENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ1Nfc1JHQikpLmdldFByb2ZpbGUoKTsKLSAgICAgICAgICAgIElDQ19Qcm9maWxlW10gcHJvZmlsZXMgPSB7c1JHQlByb2ZpbGUsIGdldFByb2ZpbGUoKX07Ci0gICAgICAgICAgICBmcm9tUkdCVHJhbnNmb3JtID0gbmV3IElDQ19UcmFuc2Zvcm0ocHJvZmlsZXMpOwotICAgICAgICAgICAgaWYgKCFzY2FsaW5nRGF0YUxvYWRlZCkgewotICAgICAgICAgICAgICAgIHNjYWxlci5sb2FkU2NhbGluZ0RhdGEodGhpcyk7Ci0gICAgICAgICAgICAgICAgc2NhbGluZ0RhdGFMb2FkZWQgPSB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLy8gc2NhbGUgcmdiIHZhbHVlIHRvIHNob3J0Ci0gICAgICAgIHNob3J0W10gc2NhbGVkUkdCVmFsdWUgPSBuZXcgc2hvcnRbM107Ci0gICAgICAgIHNjYWxlZFJHQlZhbHVlWzBdID0gKHNob3J0KShyZ2J2YWx1ZVswXSAqIE1BWF9TSE9SVCArIDAuNWYpOwotICAgICAgICBzY2FsZWRSR0JWYWx1ZVsxXSA9IChzaG9ydCkocmdidmFsdWVbMV0gKiBNQVhfU0hPUlQgKyAwLjVmKTsKLSAgICAgICAgc2NhbGVkUkdCVmFsdWVbMl0gPSAoc2hvcnQpKHJnYnZhbHVlWzJdICogTUFYX1NIT1JUICsgMC41Zik7Ci0KLSAgICAgICAgc2hvcnRbXSBjb252ZXJ0ZWQgPQotICAgICAgICAgICAgY29udmVydGVyLnRyYW5zbGF0ZUNvbG9yKGZyb21SR0JUcmFuc2Zvcm0sIHNjYWxlZFJHQlZhbHVlLCBudWxsKTsKLQotICAgICAgICBmbG9hdFtdIHJlcyA9IG5ldyBmbG9hdFtnZXROdW1Db21wb25lbnRzKCldOwotCi0gICAgICAgIHNjYWxlci51bnNjYWxlKHJlcywgY29udmVydGVkLCAwKTsKLQotICAgICAgICByZXR1cm4gcmVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBlcmZvcm1zIHRoZSB0cmFuc2Zvcm1hdGlvbiBvZiBhIGNvbG9yIGZyb20gdGhlIENTX0NJRVhZWiBjb2xvciBzcGFjZQotICAgICAqIGludG8gdGhpcyBDb2xvclNwYWNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4eXp2YWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIGZsb2F0IGFycmF5IHJlcHJlc2VudGluZyBhIGNvbG9yIGluIHRoZSBDU19DSUVYWVogY29sb3IKLSAgICAgKiAgICAgICAgICAgIHNwYWNlLgotICAgICAqIEByZXR1cm4gdGhlIGZsb2F0IGFycmF5IHdpdGggdGhlIHRyYW5zZm9ybWVkIGNvbG9yIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZsb2F0W10gZnJvbUNJRVhZWihmbG9hdFtdIHh5enZhbHVlKSB7Ci0gICAgICAgIGlmIChmcm9tWFlaVHJhbnNmb3JtID09IG51bGwpIHsKLSAgICAgICAgICAgIElDQ19Qcm9maWxlIHh5elByb2ZpbGUgPQotICAgICAgICAgICAgICAgICgoSUNDX0NvbG9yU3BhY2UpIENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ1NfQ0lFWFlaKSkuZ2V0UHJvZmlsZSgpOwotICAgICAgICAgICAgSUNDX1Byb2ZpbGVbXSBwcm9maWxlcyA9IHt4eXpQcm9maWxlLCBnZXRQcm9maWxlKCl9OwotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICBpbnRbXSBpbnRlbnRzID0gewotICAgICAgICAgICAgICAgICAgICAgICAgSUNDX1Byb2ZpbGUuaWNQZXJjZXB0dWFsLAotICAgICAgICAgICAgICAgICAgICAgICAgSUNDX1Byb2ZpbGUuaWNSZWxhdGl2ZUNvbG9yaW1ldHJpY307Ci0gICAgICAgICAgICAgICAgZnJvbVhZWlRyYW5zZm9ybSA9IG5ldyBJQ0NfVHJhbnNmb3JtKHByb2ZpbGVzLCBpbnRlbnRzKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKENNTUV4Y2VwdGlvbiBlKSB7IC8vIE5vIHN1Y2ggdGFnLCB1c2Ugd2hhdCB3ZSBjYW4KLSAgICAgICAgICAgICAgICBmcm9tWFlaVHJhbnNmb3JtID0gbmV3IElDQ19UcmFuc2Zvcm0ocHJvZmlsZXMpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoIXNjYWxpbmdEYXRhTG9hZGVkKSB7Ci0gICAgICAgICAgICAgICAgc2NhbGVyLmxvYWRTY2FsaW5nRGF0YSh0aGlzKTsKLSAgICAgICAgICAgICAgICBzY2FsaW5nRGF0YUxvYWRlZCA9IHRydWU7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIHNjYWxlIHh5eiB2YWx1ZSB0byBzaG9ydAotICAgICAgICBzaG9ydFtdIHNjYWxlZFhZWlZhbHVlID0gbmV3IHNob3J0WzNdOwotICAgICAgICBzY2FsZWRYWVpWYWx1ZVswXSA9IChzaG9ydCkoeHl6dmFsdWVbMF0gKiBYWVoyU0hPUlRfRkFDVE9SICsgMC41Zik7Ci0gICAgICAgIHNjYWxlZFhZWlZhbHVlWzFdID0gKHNob3J0KSh4eXp2YWx1ZVsxXSAqIFhZWjJTSE9SVF9GQUNUT1IgKyAwLjVmKTsKLSAgICAgICAgc2NhbGVkWFlaVmFsdWVbMl0gPSAoc2hvcnQpKHh5enZhbHVlWzJdICogWFlaMlNIT1JUX0ZBQ1RPUiArIDAuNWYpOwotCi0gICAgICAgIHNob3J0W10gY29udmVydGVkID0KLSAgICAgICAgICAgIGNvbnZlcnRlci50cmFuc2xhdGVDb2xvcihmcm9tWFlaVHJhbnNmb3JtLCBzY2FsZWRYWVpWYWx1ZSwgbnVsbCk7Ci0KLSAgICAgICAgZmxvYXRbXSByZXMgPSBuZXcgZmxvYXRbZ2V0TnVtQ29tcG9uZW50cygpXTsKLQotICAgICAgICBzY2FsZXIudW5zY2FsZShyZXMsIGNvbnZlcnRlZCwgMCk7Ci0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtaW5pbXVtIG5vcm1hbGl6ZWQgY29sb3IgY29tcG9uZW50IHZhbHVlIGZvciB0aGUgc3BlY2lmaWVkCi0gICAgICogY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb21wb25lbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb21wb25lbnQgdG8gZGV0ZXJtaW5lIHRoZSBtaW5pbXVtIHZhbHVlLgotICAgICAqIEByZXR1cm4gdGhlIG1pbmltdW0gbm9ybWFsaXplZCB2YWx1ZSBvZiB0aGUgY29tcG9uZW50LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRNaW5WYWx1ZShpbnQgY29tcG9uZW50KSB7Ci0gICAgICAgIGlmICgoY29tcG9uZW50IDwgMCkgfHwgKGNvbXBvbmVudCA+IHRoaXMuZ2V0TnVtQ29tcG9uZW50cygpIC0gMSkpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xNjk9Q29tcG9uZW50IGluZGV4IG91dCBvZiByYW5nZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNjkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBtaW5WYWx1ZXNbY29tcG9uZW50XTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtYXhpbXVtIG5vcm1hbGl6ZWQgY29sb3IgY29tcG9uZW50IHZhbHVlIGZvciB0aGUgc3BlY2lmaWVkCi0gICAgICogY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb21wb25lbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb21wb25lbnQgdG8gZGV0ZXJtaW5lIHRoZSBtYXhpbXVtIHZhbHVlLgotICAgICAqIEByZXR1cm4gdGhlIG1heGltdW0gbm9ybWFsaXplZCB2YWx1ZSBvZiB0aGUgY29tcG9uZW50LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRNYXhWYWx1ZShpbnQgY29tcG9uZW50KSB7Ci0gICAgICAgIGlmICgoY29tcG9uZW50IDwgMCkgfHwgKGNvbXBvbmVudCA+IHRoaXMuZ2V0TnVtQ29tcG9uZW50cygpIC0gMSkpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xNjk9Q29tcG9uZW50IGluZGV4IG91dCBvZiByYW5nZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNjkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBtYXhWYWx1ZXNbY29tcG9uZW50XTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaWxsIG1pbiBtYXggdmFsdWVzLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBmaWxsTWluTWF4VmFsdWVzKCkgewotICAgICAgICBpbnQgbiA9IGdldE51bUNvbXBvbmVudHMoKTsKLSAgICAgICAgbWF4VmFsdWVzID0gbmV3IGZsb2F0W25dOwotICAgICAgICBtaW5WYWx1ZXMgPSBuZXcgZmxvYXRbbl07Ci0gICAgICAgIHN3aXRjaCAoZ2V0VHlwZSgpKSB7Ci0gICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuVFlQRV9YWVo6Ci0gICAgICAgICAgICAgICAgbWluVmFsdWVzWzBdID0gMDsKLSAgICAgICAgICAgICAgICBtaW5WYWx1ZXNbMV0gPSAwOwotICAgICAgICAgICAgICAgIG1pblZhbHVlc1syXSA9IDA7Ci0gICAgICAgICAgICAgICAgbWF4VmFsdWVzWzBdID0gTUFYX1hZWjsKLSAgICAgICAgICAgICAgICBtYXhWYWx1ZXNbMV0gPSBNQVhfWFlaOwotICAgICAgICAgICAgICAgIG1heFZhbHVlc1syXSA9IE1BWF9YWVo7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuVFlQRV9MYWI6Ci0gICAgICAgICAgICAgICAgbWluVmFsdWVzWzBdID0gMDsKLSAgICAgICAgICAgICAgICBtaW5WYWx1ZXNbMV0gPSAtMTI4OwotICAgICAgICAgICAgICAgIG1pblZhbHVlc1syXSA9IC0xMjg7Ci0gICAgICAgICAgICAgICAgbWF4VmFsdWVzWzBdID0gMTAwOwotICAgICAgICAgICAgICAgIG1heFZhbHVlc1sxXSA9IDEyNzsKLSAgICAgICAgICAgICAgICBtYXhWYWx1ZXNbMl0gPSAxMjc7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIGZvcihpbnQgaT0wOyBpPG47IGkrKykgewotICAgICAgICAgICAgICAgICAgICBtaW5WYWx1ZXNbaV0gPSAwOwotICAgICAgICAgICAgICAgICAgICBtYXhWYWx1ZXNbaV0gPSAxOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFdyaXRlIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb3V0Ci0gICAgICogICAgICAgICAgICB0aGUgb3V0Ci0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIFNpZ25hbHMgdGhhdCBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgd3JpdGVPYmplY3QoT2JqZWN0T3V0cHV0U3RyZWFtIG91dCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgT2JqZWN0T3V0cHV0U3RyZWFtLlB1dEZpZWxkIGZpZWxkcyA9IG91dC5wdXRGaWVsZHMoKTsKLQotICAgICAgICBmaWVsZHMucHV0KCJ0aGlzUHJvZmlsZSIsIHByb2ZpbGUpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIGZpZWxkcy5wdXQoIm1pblZhbCIsIG51bGwpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIGZpZWxkcy5wdXQoIm1heFZhbCIsIG51bGwpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIGZpZWxkcy5wdXQoImRpZmZNaW5NYXgiLCBudWxsKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBmaWVsZHMucHV0KCJpbnZEaWZmTWluTWF4IiwgbnVsbCk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgZmllbGRzLnB1dCgibmVlZFNjYWxlSW5pdCIsIHRydWUpOyAvLyROT04tTkxTLTEkCi0KLSAgICAgICAgb3V0LndyaXRlRmllbGRzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVhZCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGluCi0gICAgICogICAgICAgICAgICB0aGUgaW4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgU2lnbmFscyB0aGF0IGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqIEB0aHJvd3MgQ2xhc3NOb3RGb3VuZEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIHRoZSBjbGFzcyBub3QgZm91bmQgZXhjZXB0aW9uCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHJlYWRPYmplY3QoT2JqZWN0SW5wdXRTdHJlYW0gaW4pIHRocm93cyBJT0V4Y2VwdGlvbiwgQ2xhc3NOb3RGb3VuZEV4Y2VwdGlvbiB7Ci0gICAgICAgIE9iamVjdElucHV0U3RyZWFtLkdldEZpZWxkIGZpZWxkcyA9IGluLnJlYWRGaWVsZHMoKTsKLSAgICAgICAgcmVzb2x2ZWREZXNlcmlhbGl6ZWRJbnN0ID0KLSAgICAgICAgICAgICAgICBuZXcgSUNDX0NvbG9yU3BhY2UoKElDQ19Qcm9maWxlKSBmaWVsZHMuZ2V0KCJ0aGlzUHJvZmlsZSIsIG51bGwpKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlYWQgcmVzb2x2ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBvYmplY3QKLSAgICAgKiBAdGhyb3dzIE9iamVjdFN0cmVhbUV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIHRoZSBvYmplY3Qgc3RyZWFtIGV4Y2VwdGlvbgotICAgICAqLwotICAgIE9iamVjdCByZWFkUmVzb2x2ZSgpIHRocm93cyBPYmplY3RTdHJlYW1FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gcmVzb2x2ZWREZXNlcmlhbGl6ZWRJbnN0OwotICAgIH0KLX0KLQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2NvbG9yL0lDQ19Qcm9maWxlLmphdmEgYi9hd3QvamF2YS9hd3QvY29sb3IvSUNDX1Byb2ZpbGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOGZmZWU2Yy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvY29sb3IvSUNDX1Byb2ZpbGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE0NzcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5jb2xvcjsKLQotaW1wb3J0IGphdmEuaW8uRmlsZTsKLWltcG9ydCBqYXZhLmlvLkZpbGVJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLkZpbGVOb3RGb3VuZEV4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLkZpbGVPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLklucHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uT2JqZWN0SW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5PYmplY3RPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5PYmplY3RTdHJlYW1FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5PdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5TZXJpYWxpemFibGU7Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5BY2Nlc3NDb250cm9sbGVyOwotaW1wb3J0IGphdmEuc2VjdXJpdHkuUHJpdmlsZWdlZEFjdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuU3RyaW5nVG9rZW5pemVyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvci5JQ0NfUHJvZmlsZUhlbHBlcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmNvbG9yLk5hdGl2ZUNNTTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgSUNDX1Byb2ZpbGUgY2xhc3MgcmVwcmVzZW50cyBhIGNvbG9yIHByb2ZpbGUgZGF0YSBmb3IgY29sb3Igc3BhY2VzIGJhc2VkCi0gKiBvbiB0aGUgSW50ZXJuYXRpb25hbCBDb2xvciBDb25zb3J0aXVtIFNwZWNpZmljYXRpb24gSUNDLjE6MjAwMS0xMiwgRmlsZQotICogRm9ybWF0IGZvciBDb2xvciBQcm9maWxlcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJQ0NfUHJvZmlsZSBpbXBsZW1lbnRzIFNlcmlhbGl6YWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtMzkzODUxNTg2MTk5MDkzNjc2Nkw7Ci0KLSAgICAvLyBOT1RFOiBDb25zdGFudCBmaWVsZCB2YWx1ZXMgYXJlIG5vdGVkIGluIDEuNSBzcGVjaWZpY2F0aW9uLgotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENMQVNTX0lOUFVUIGluZGljYXRlcyB0aGF0IHByb2ZpbGUgY2xhc3MgaXMgaW5wdXQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ0xBU1NfSU5QVVQgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENMQVNTX0RJU1BMQVkgaW5kaWNhdGVzIHRoYXQgcHJvZmlsZSBjbGFzcyBpcyBkaXNwbGF5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENMQVNTX0RJU1BMQVkgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENMQVNTX09VVFBVVCBpbmRpY2F0ZXMgdGhhdCBwcm9maWxlIGNsYXNzIGlzIG91dHB1dC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDTEFTU19PVVRQVVQgPSAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENMQVNTX0RFVklDRUxJTksgaW5kaWNhdGVzIHRoYXQgcHJvZmlsZSBjbGFzcyBpcyBkZXZpY2UKLSAgICAgKiBsaW5rLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENMQVNTX0RFVklDRUxJTksgPSAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENMQVNTX0NPTE9SU1BBQ0VDT05WRVJTSU9OIGluZGljYXRlcyB0aGF0IHByb2ZpbGUgY2xhc3MgaXMKLSAgICAgKiBjb2xvciBzcGFjZSBjb252ZXJzaW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENMQVNTX0NPTE9SU1BBQ0VDT05WRVJTSU9OID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDTEFTU19BQlNUUkFDVCBpbmRpY2F0ZXMgdGhhdCBwcm9maWxlIGNsYXNzIGlzIGFic3RyYWN0LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENMQVNTX0FCU1RSQUNUID0gNTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDTEFTU19OQU1FRENPTE9SIGluZGljYXRlcyB0aGF0IHByb2ZpbGUgY2xhc3MgaXMgbmFtZWQKLSAgICAgKiBjb2xvci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDTEFTU19OQU1FRENPTE9SID0gNjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1hZWkRhdGEgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1hZWkRhdGEgPSAxNDgyMjUwNzg0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnTGFiRGF0YSAtIElDQyBQcm9maWxlIENvbG9yIFNwYWNlIFR5cGUgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnTGFiRGF0YSA9IDEyODE0NTA1Mjg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdMdXZEYXRhIC0gSUNDIFByb2ZpbGUgQ29sb3IgU3BhY2UgVHlwZSBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdMdXZEYXRhID0gMTI4Mjc2NjM2ODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1lDYkNyRGF0YSAtIElDQyBQcm9maWxlIENvbG9yIFNwYWNlIFR5cGUgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnWUNiQ3JEYXRhID0gMTQ5NzU4ODMzODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1l4eURhdGEgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1l4eURhdGEgPSAxNTAxMDY3NTUyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnUmdiRGF0YSAtIElDQyBQcm9maWxlIENvbG9yIFNwYWNlIFR5cGUgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnUmdiRGF0YSA9IDEzODA0MDE2OTY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdHcmF5RGF0YSAtIElDQyBQcm9maWxlIENvbG9yIFNwYWNlIFR5cGUgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnR3JheURhdGEgPSAxMTk2NTczMDE3OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnSHN2RGF0YSAtIElDQyBQcm9maWxlIENvbG9yIFNwYWNlIFR5cGUgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnSHN2RGF0YSA9IDEyMTM0MjEwODg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdIbHNEYXRhIC0gSUNDIFByb2ZpbGUgQ29sb3IgU3BhY2UgVHlwZSBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdIbHNEYXRhID0gMTIxMjk2MTU2ODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ0NteWtEYXRhIC0gSUNDIFByb2ZpbGUgQ29sb3IgU3BhY2UgVHlwZSBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdDbXlrRGF0YSA9IDExMjkxNDI2MDM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdDbXlEYXRhIC0gSUNDIFByb2ZpbGUgQ29sb3IgU3BhY2UgVHlwZSBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdDbXlEYXRhID0gMTEyOTE0MjU2MDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1NwYWNlMkNMUiAtIElDQyBQcm9maWxlIENvbG9yIFNwYWNlIFR5cGUgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnU3BhY2UyQ0xSID0gODQzMjcxMjUwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnU3BhY2UzQ0xSIC0gSUNDIFByb2ZpbGUgQ29sb3IgU3BhY2UgVHlwZSBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdTcGFjZTNDTFIgPSA4NjAwNDg0NjY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdTcGFjZTRDTFIgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NwYWNlNENMUiA9IDg3NjgyNTY4MjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1NwYWNlNUNMUiAtIElDQyBQcm9maWxlIENvbG9yIFNwYWNlIFR5cGUgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnU3BhY2U1Q0xSID0gODkzNjAyODk4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnU3BhY2U2Q0xSIC0gSUNDIFByb2ZpbGUgQ29sb3IgU3BhY2UgVHlwZSBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdTcGFjZTZDTFIgPSA5MTAzODAxMTQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdTcGFjZTdDTFIgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NwYWNlN0NMUiA9IDkyNzE1NzMzMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1NwYWNlOENMUiAtIElDQyBQcm9maWxlIENvbG9yIFNwYWNlIFR5cGUgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnU3BhY2U4Q0xSID0gOTQzOTM0NTQ2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnU3BhY2U5Q0xSIC0gSUNDIFByb2ZpbGUgQ29sb3IgU3BhY2UgVHlwZSBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdTcGFjZTlDTFIgPSA5NjA3MTE3NjI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdTcGFjZUFDTFIgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NwYWNlQUNMUiA9IDEwOTQ5Mjk0OTA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdTcGFjZUJDTFIgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NwYWNlQkNMUiA9IDExMTE3MDY3MDY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdTcGFjZUNDTFIgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NwYWNlQ0NMUiA9IDExMjg0ODM5MjI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdTcGFjZURDTFIgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NwYWNlRENMUiA9IDExNDUyNjExMzg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdTcGFjZUVDTFIgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NwYWNlRUNMUiA9IDExNjIwMzgzNTQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdTcGFjZUZDTFIgLSBJQ0MgUHJvZmlsZSBDb2xvciBTcGFjZSBUeXBlIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NwYWNlRkNMUiA9IDExNzg4MTU1NzA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdJbnB1dENsYXNzIC0gSUNDIFByb2ZpbGUgQ2xhc3MgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnSW5wdXRDbGFzcyA9IDE5MzU4OTYxNzg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdEaXNwbGF5Q2xhc3MgLSBJQ0MgUHJvZmlsZSBDbGFzcyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdEaXNwbGF5Q2xhc3MgPSAxODM1OTU1MzE0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnT3V0cHV0Q2xhc3MgLSBJQ0MgUHJvZmlsZSBDbGFzcyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdPdXRwdXRDbGFzcyA9IDE4ODY1NDkxMDY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdMaW5rQ2xhc3MgLSBJQ0MgUHJvZmlsZSBDbGFzcyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdMaW5rQ2xhc3MgPSAxODE4ODQ4ODc1OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnQWJzdHJhY3RDbGFzcyAtIElDQyBQcm9maWxlIENsYXNzIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0Fic3RyYWN0Q2xhc3MgPSAxNjMzODQyMDM2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnQ29sb3JhbnRPcmRlclRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdDb2xvcmFudE9yZGVyVGFnID0gMTY2ODA1MTU2NzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ0NvbG9yYW50VGFibGVUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnQ29sb3JhbnRUYWJsZVRhZyA9IDE2NjgwNTE1NzI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdDb2xvclNwYWNlQ2xhc3MgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnQ29sb3JTcGFjZUNsYXNzID0gMTkzNjc0NDgwMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ05hbWVkQ29sb3JDbGFzcyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdOYW1lZENvbG9yQ2xhc3MgPSAxODUyNjYyNjM2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljUGVyY2VwdHVhbCAtIElDQyBQcm9maWxlIFJlbmRlcmluZyBJbnRlbnQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNQZXJjZXB0dWFsID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1JlbGF0aXZlQ29sb3JpbWV0cmljIC0gSUNDIFByb2ZpbGUgUmVuZGVyaW5nIEludGVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1JlbGF0aXZlQ29sb3JpbWV0cmljID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NhdHVyYXRpb24gLSBJQ0MgUHJvZmlsZSBSZW5kZXJpbmcgSW50ZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2F0dXJhdGlvbiA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNBYnNvbHV0ZUNvbG9yaW1ldHJpYyAtIElDQyBQcm9maWxlIFJlbmRlcmluZyBJbnRlbnQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNBYnNvbHV0ZUNvbG9yaW1ldHJpYyA9IDM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdIZWFkIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0hlYWQgPSAxNzUxNDc0NTMyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnQVRvQjBUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnQVRvQjBUYWcgPSAxMDkzODEyNzg0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnQVRvQjFUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnQVRvQjFUYWcgPSAxMDkzODEyNzg1OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnQVRvQjJUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnQVRvQjJUYWcgPSAxMDkzODEyNzg2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnQmx1ZUNvbG9yYW50VGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0JsdWVDb2xvcmFudFRhZyA9IDE2NDk5NTcyMTA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdCbHVlTWF0cml4Q29sdW1uVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0JsdWVNYXRyaXhDb2x1bW5UYWcgPSAxNjQ5OTU3MjEwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnQmx1ZVRSQ1RhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdCbHVlVFJDVGFnID0gMTY0OTY5MzI1MTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ0JUb0EwVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0JUb0EwVGFnID0gMTExMDU4OTc0NDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ0JUb0ExVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0JUb0ExVGFnID0gMTExMDU4OTc0NTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ0JUb0EyVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0JUb0EyVGFnID0gMTExMDU4OTc0NjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ0NhbGlicmF0aW9uRGF0ZVRpbWVUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnQ2FsaWJyYXRpb25EYXRlVGltZVRhZyA9IDE2NjczMjkxNDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdDaGFyVGFyZ2V0VGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0NoYXJUYXJnZXRUYWcgPSAxOTUyNTQzMzM1OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnQ29weXJpZ2h0VGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0NvcHlyaWdodFRhZyA9IDE2NjgzMTM3MTY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdDcmRJbmZvVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0NyZEluZm9UYWcgPSAxNjY4NDQxMTkzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnRGV2aWNlTWZnRGVzY1RhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdEZXZpY2VNZmdEZXNjVGFnID0gMTY4NDg5MzI4NDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ0RldmljZU1vZGVsRGVzY1RhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdEZXZpY2VNb2RlbERlc2NUYWcgPSAxNjg0ODkwNzI0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnRGV2aWNlU2V0dGluZ3NUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnRGV2aWNlU2V0dGluZ3NUYWcgPSAxNjg0MzcxMDU5OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnR2FtdXRUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnR2FtdXRUYWcgPSAxNzM0NDM4MjYwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnR3JheVRSQ1RhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdHcmF5VFJDVGFnID0gMTgwMDY4ODE5NTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ0dyZWVuQ29sb3JhbnRUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnR3JlZW5Db2xvcmFudFRhZyA9IDE3MzM4NDMyOTA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdHcmVlbk1hdHJpeENvbHVtblRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdHcmVlbk1hdHJpeENvbHVtblRhZyA9IDE3MzM4NDMyOTA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdHcmVlblRSQ1RhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdHcmVlblRSQ1RhZyA9IDE3MzM1NzkzMzE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdMdW1pbmFuY2VUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnTHVtaW5hbmNlVGFnID0gMTgxOTYzNTA0OTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ01lYXN1cmVtZW50VGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ01lYXN1cmVtZW50VGFnID0gMTgzNTM2MDYyNzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ01lZGlhQmxhY2tQb2ludFRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdNZWRpYUJsYWNrUG9pbnRUYWcgPSAxNjUxMjA4MzA4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnTWVkaWFXaGl0ZVBvaW50VGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ01lZGlhV2hpdGVQb2ludFRhZyA9IDIwMDQxMTk2Njg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdOYW1lZENvbG9yMlRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdOYW1lZENvbG9yMlRhZyA9IDE4NTIwMDk1MjI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdPdXRwdXRSZXNwb25zZVRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdPdXRwdXRSZXNwb25zZVRhZyA9IDE5MTkyNTEzMTI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdQcmV2aWV3MFRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdQcmV2aWV3MFRhZyA9IDE4ODY1NDUyMDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdQcmV2aWV3MVRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdQcmV2aWV3MVRhZyA9IDE4ODY1NDUyMDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdQcmV2aWV3MlRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdQcmV2aWV3MlRhZyA9IDE4ODY1NDUyMDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdQcm9maWxlRGVzY3JpcHRpb25UYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnUHJvZmlsZURlc2NyaXB0aW9uVGFnID0gMTY4NDM3MDI3NTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1Byb2ZpbGVTZXF1ZW5jZURlc2NUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnUHJvZmlsZVNlcXVlbmNlRGVzY1RhZyA9IDE4ODY2MTA4MDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdQczJDUkQwVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1BzMkNSRDBUYWcgPSAxODg2NjEwNDgwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnUHMyQ1JEMVRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdQczJDUkQxVGFnID0gMTg4NjYxMDQ4MTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1BzMkNSRDJUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnUHMyQ1JEMlRhZyA9IDE4ODY2MTA0ODI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdQczJDUkQzVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1BzMkNSRDNUYWcgPSAxODg2NjEwNDgzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnUHMyQ1NBVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1BzMkNTQVRhZyA9IDE4ODY1OTc3NDc7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdQczJSZW5kZXJpbmdJbnRlbnRUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnUHMyUmVuZGVyaW5nSW50ZW50VGFnID0gMTg4NjU5NzczNzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1JlZENvbG9yYW50VGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1JlZENvbG9yYW50VGFnID0gMTkxODM5MjY2NjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY1NpZ1JlZE1hdHJpeENvbHVtblRhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdSZWRNYXRyaXhDb2x1bW5UYWcgPSAxOTE4MzkyNjY2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnUmVkVFJDVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1JlZFRSQ1RhZyA9IDE5MTgxMjg3MDc7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdTY3JlZW5pbmdEZXNjVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NjcmVlbmluZ0Rlc2NUYWcgPSAxOTM1ODk3MTg4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnU2NyZWVuaW5nVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1NjcmVlbmluZ1RhZyA9IDE5MzU4OTcxOTg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdUZWNobm9sb2d5VGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1RlY2hub2xvZ3lUYWcgPSAxOTUyODAxNjQwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnVWNyQmdUYWcgLSBJQ0MgUHJvZmlsZSBUYWcgU2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljU2lnVWNyQmdUYWcgPSAxNjUwODc3NDcyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnVmlld2luZ0NvbmREZXNjVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ1ZpZXdpbmdDb25kRGVzY1RhZyA9IDE5ODc0MDUxNTY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdWaWV3aW5nQ29uZGl0aW9uc1RhZyAtIElDQyBQcm9maWxlIFRhZyBTaWduYXR1cmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNTaWdWaWV3aW5nQ29uZGl0aW9uc1RhZyA9IDE5ODY2MTg3NDM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNTaWdDaHJvbWF0aWNBZGFwdGF0aW9uVGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0Nocm9tYXRpY0FkYXB0YXRpb25UYWcgPSAxNjY3Nzg1MDYwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljU2lnQ2hyb21hdGljaXR5VGFnIC0gSUNDIFByb2ZpbGUgVGFnIFNpZ25hdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY1NpZ0Nocm9tYXRpY2l0eVRhZyA9IDE2Njc3ODk0MjE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNIZHJTaXplIC0gSUNDIFByb2ZpbGUgSGVhZGVyIExvY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljSGRyU2l6ZSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNIZHJDbW1JZCAtIElDQyBQcm9maWxlIEhlYWRlciBMb2NhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY0hkckNtbUlkID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY0hkclZlcnNpb24gLSBJQ0MgUHJvZmlsZSBIZWFkZXIgTG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNIZHJWZXJzaW9uID0gODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY0hkckRldmljZUNsYXNzIC0gSUNDIFByb2ZpbGUgSGVhZGVyIExvY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljSGRyRGV2aWNlQ2xhc3MgPSAxMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY0hkckNvbG9yU3BhY2UgLSBJQ0MgUHJvZmlsZSBIZWFkZXIgTG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNIZHJDb2xvclNwYWNlID0gMTY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNIZHJQY3MgLSBJQ0MgUHJvZmlsZSBIZWFkZXIgTG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNIZHJQY3MgPSAyMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY0hkckRhdGUgLSBJQ0MgUHJvZmlsZSBIZWFkZXIgTG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNIZHJEYXRlID0gMjQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNIZHJNYWdpYyAtIElDQyBQcm9maWxlIEhlYWRlciBMb2NhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY0hkck1hZ2ljID0gMzY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNIZHJQbGF0Zm9ybSAtIElDQyBQcm9maWxlIEhlYWRlciBMb2NhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY0hkclBsYXRmb3JtID0gNDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNIZHJQcm9maWxlSUQgLSBJQ0MgUHJvZmlsZSBIZWFkZXIgTG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNIZHJQcm9maWxlSUQgPSA4NDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY0hkckZsYWdzIC0gSUNDIFByb2ZpbGUgSGVhZGVyIExvY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljSGRyRmxhZ3MgPSA0NDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY0hkck1hbnVmYWN0dXJlciAtIElDQyBQcm9maWxlIEhlYWRlciBMb2NhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY0hkck1hbnVmYWN0dXJlciA9IDQ4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljSGRyTW9kZWwgLSBJQ0MgUHJvZmlsZSBIZWFkZXIgTG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNIZHJNb2RlbCA9IDUyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljSGRyQXR0cmlidXRlcyAtIElDQyBQcm9maWxlIEhlYWRlciBMb2NhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY0hkckF0dHJpYnV0ZXMgPSA1NjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY0hkclJlbmRlcmluZ0ludGVudCAtIElDQyBQcm9maWxlIEhlYWRlciBMb2NhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY0hkclJlbmRlcmluZ0ludGVudCA9IDY0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljSGRySWxsdW1pbmFudCAtIElDQyBQcm9maWxlIEhlYWRlciBMb2NhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY0hkcklsbHVtaW5hbnQgPSA2ODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBpY0hkckNyZWF0b3IgLSBJQ0MgUHJvZmlsZSBIZWFkZXIgTG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNIZHJDcmVhdG9yID0gODA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNJQ0NBYnNvbHV0ZUNvbG9yaW1ldHJpYyAtIElDQyBQcm9maWxlIFJlbmRlcmluZyBJbnRlbnQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgaWNJQ0NBYnNvbHV0ZUNvbG9yaW1ldHJpYyA9IDM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNNZWRpYVJlbGF0aXZlQ29sb3JpbWV0cmljIC0gSUNDIFByb2ZpbGUgUmVuZGVyaW5nIEludGVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY01lZGlhUmVsYXRpdmVDb2xvcmltZXRyaWMgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljVGFnVHlwZSAtIElDQyBQcm9maWxlIENvbnN0YW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljVGFnVHlwZSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNUYWdSZXNlcnZlZCAtIElDQyBQcm9maWxlIENvbnN0YW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljVGFnUmVzZXJ2ZWQgPSA0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljQ3VydmVDb3VudCAtIElDQyBQcm9maWxlIENvbnN0YW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljQ3VydmVDb3VudCA9IDg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgaWNDdXJ2ZURhdGEgLSBJQ0MgUHJvZmlsZSBDb25zdGFudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBpY0N1cnZlRGF0YSA9IDEyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGljWFlaTnVtYmVyWCAtIElDQyBQcm9maWxlIENvbnN0YW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IGljWFlaTnVtYmVyWCA9IDg7Ci0KLSAgICAvKioKLSAgICAgKiBTaXplIG9mIGEgcHJvZmlsZSBoZWFkZXIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IGhlYWRlclNpemUgPSAxMjg7Ci0KLSAgICAvKioKLSAgICAgKiBoZWFkZXIgbWFnaWMgbnVtYmVyLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBoZWFkZXJNYWdpY051bWJlciA9IDB4NjE2MzczNzA7Ci0KLSAgICAvLyBDYWNoZSBvZiBwcmVkZWZpbmVkIHByb2ZpbGVzCi0gICAgLyoqCi0gICAgICogVGhlIHMgcmdiIHByb2ZpbGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgSUNDX1Byb2ZpbGUgc1JHQlByb2ZpbGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgeHl6IHByb2ZpbGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgSUNDX1Byb2ZpbGUgeHl6UHJvZmlsZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBncmF5IHByb2ZpbGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgSUNDX1Byb2ZpbGUgZ3JheVByb2ZpbGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcHljYyBwcm9maWxlLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIElDQ19Qcm9maWxlIHB5Y2NQcm9maWxlOwotCi0gICAgLyoqCi0gICAgICogVGhlIGxpbmVhciByZ2IgcHJvZmlsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBJQ0NfUHJvZmlsZSBsaW5lYXJSR0JQcm9maWxlOwotCi0gICAgLyoqCi0gICAgICogSGFuZGxlIHRvIHRoZSBjdXJyZW50IHByb2ZpbGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSB0cmFuc2llbnQgbG9uZyBwcm9maWxlSGFuZGxlID0gMDsKLQotICAgIC8qKgotICAgICAqIElmIGhhbmRsZSBpcyB1c2VkIGJ5IGFub3RoZXIgY2xhc3MgdGhpcyBvYmplY3QgaXMgbm90IHJlc3BvbnNpYmxlIGZvcgotICAgICAqIGNsb3NpbmcgcHJvZmlsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHRyYW5zaWVudCBib29sZWFuIGhhbmRsZVN0b2xlbiA9IGZhbHNlOwotCi0gICAgLyoqCi0gICAgICogQ2FjaGVkIGhlYWRlciBkYXRhLgotICAgICAqLwotICAgIHByaXZhdGUgdHJhbnNpZW50IGJ5dGVbXSBoZWFkZXJEYXRhID0gbnVsbDsKLQotICAgIC8qKgotICAgICAqIFNlcmlhbGl6YXRpb24gc3VwcG9ydC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHRyYW5zaWVudCBJQ0NfUHJvZmlsZSBvcGVuZWRQcm9maWxlT2JqZWN0OwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IElDQyBwcm9maWxlIHdpdGggdGhlIGdpdmVuIGRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhLgotICAgICAqLwotICAgIHByaXZhdGUgSUNDX1Byb2ZpbGUoYnl0ZVtdIGRhdGEpIHsKLSAgICAgICAgcHJvZmlsZUhhbmRsZSA9IE5hdGl2ZUNNTS5jbW1PcGVuUHJvZmlsZShkYXRhKTsKLSAgICAgICAgTmF0aXZlQ01NLmFkZEhhbmRsZSh0aGlzLCBwcm9maWxlSGFuZGxlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBVc2VkIHRvIGluc3RhbnRpYXRlIGR1bW15IElDQ19Qcm9maWxlU3R1YiBvYmplY3RzLgotICAgICAqLwotICAgIElDQ19Qcm9maWxlKCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIFVzZWQgdG8gaW5zdGFudGlhdGUgc3ViY2xhc3NlcyAoSUNDX1Byb2ZpbGVHcmV5IGFuZCBJQ0NfUHJvZmlsZVJHQikuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb2ZpbGVIYW5kbGUKLSAgICAgKiAgICAgICAgICAgIC0gc2hvdWxkIGJlIHZhbGlkIGhhbmRsZSB0byBvcGVuZWQgY29sb3IgcHJvZmlsZQotICAgICAqLwotICAgIElDQ19Qcm9maWxlKGxvbmcgcHJvZmlsZUhhbmRsZSkgewotICAgICAgICB0aGlzLnByb2ZpbGVIYW5kbGUgPSBwcm9maWxlSGFuZGxlOwotICAgICAgICAvLyBBIG5ldyBvYmplY3QgcmVmZXJlbmNlLCBuZWVkIHRvIGFkZCBpdC4KLSAgICAgICAgTmF0aXZlQ01NLmFkZEhhbmRsZSh0aGlzLCBwcm9maWxlSGFuZGxlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcml0ZXMgdGhlIElDQ19Qcm9maWxlIHRvIGEgZmlsZSB3aXRoIHRoZSBzcGVjaWZpZWQgbmFtZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZmlsZU5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBmaWxlIG5hbWUuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkIGR1cmluZyB3cml0aW5nIG9yIG9wZW5pbmcKLSAgICAgKiAgICAgICAgICAgICB0aGUgZmlsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB3cml0ZShTdHJpbmcgZmlsZU5hbWUpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIEZpbGVPdXRwdXRTdHJlYW0gb1N0cmVhbSA9IG5ldyBGaWxlT3V0cHV0U3RyZWFtKGZpbGVOYW1lKTsKLSAgICAgICAgb1N0cmVhbS53cml0ZShnZXREYXRhKCkpOwotICAgICAgICBvU3RyZWFtLmNsb3NlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2VyaWFsaXphYmxlIGltcGxlbWVudGF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgcwotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBTaWduYWxzIHRoYXQgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHdyaXRlT2JqZWN0KE9iamVjdE91dHB1dFN0cmVhbSBzKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBzLmRlZmF1bHRXcml0ZU9iamVjdCgpOwotICAgICAgICBzLndyaXRlT2JqZWN0KG51bGwpOwotICAgICAgICBzLndyaXRlT2JqZWN0KGdldERhdGEoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2VyaWFsaXphYmxlIGltcGxlbWVudGF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgcwotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBTaWduYWxzIHRoYXQgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICogQHRocm93cyBDbGFzc05vdEZvdW5kRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgdGhlIGNsYXNzIG5vdCBmb3VuZCBleGNlcHRpb24KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgcmVhZE9iamVjdChPYmplY3RJbnB1dFN0cmVhbSBzKSB0aHJvd3MgSU9FeGNlcHRpb24sIENsYXNzTm90Rm91bmRFeGNlcHRpb24gewotICAgICAgICBzLmRlZmF1bHRSZWFkT2JqZWN0KCk7Ci0gICAgICAgIFN0cmluZyBjb2xvclNwYWNlU3RyID0gKFN0cmluZylzLnJlYWRPYmplY3QoKTsKLSAgICAgICAgYnl0ZVtdIGRhdGEgPSAoYnl0ZVtdKXMucmVhZE9iamVjdCgpOwotCi0gICAgICAgIGlmIChjb2xvclNwYWNlU3RyICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChjb2xvclNwYWNlU3RyLmVxdWFscygiQ1Nfc1JHQiIpKSB7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICBvcGVuZWRQcm9maWxlT2JqZWN0ID0gZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19zUkdCKTsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoY29sb3JTcGFjZVN0ci5lcXVhbHMoIkNTX0dSQVkiKSkgeyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgb3BlbmVkUHJvZmlsZU9iamVjdCA9IGdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1NfR1JBWSk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGNvbG9yU3BhY2VTdHIuZXF1YWxzKCJDU19MSU5FQVJfUkdCIikpIHsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgIG9wZW5lZFByb2ZpbGVPYmplY3QgPSBnZXRJbnN0YW5jZShDb2xvclNwYWNlLkNTX0xJTkVBUl9SR0IpOwotICAgICAgICAgICAgfSBlbHNlIGlmIChjb2xvclNwYWNlU3RyLmVxdWFscygiQ1NfQ0lFWFlaIikpIHsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgIG9wZW5lZFByb2ZpbGVPYmplY3QgPSBnZXRJbnN0YW5jZShDb2xvclNwYWNlLkNTX0NJRVhZWik7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGNvbG9yU3BhY2VTdHIuZXF1YWxzKCJDU19QWUNDIikpIHsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgIG9wZW5lZFByb2ZpbGVPYmplY3QgPSBnZXRJbnN0YW5jZShDb2xvclNwYWNlLkNTX1BZQ0MpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBvcGVuZWRQcm9maWxlT2JqZWN0ID0gSUNDX1Byb2ZpbGUuZ2V0SW5zdGFuY2UoZGF0YSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBvcGVuZWRQcm9maWxlT2JqZWN0ID0gSUNDX1Byb2ZpbGUuZ2V0SW5zdGFuY2UoZGF0YSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXNvbHZlcyBpbnN0YW5jZXMgYmVpbmcgZGVzZXJpYWxpemVkIGludG8gaW5zdGFuY2VzIHJlZ2lzdGVyZWQgd2l0aCBDTU0uCi0gICAgICogCi0gICAgICogQHJldHVybiBJQ0NfUHJvZmlsZSBvYmplY3QgZm9yIHByb2ZpbGUgcmVnaXN0ZXJlZCB3aXRoIENNTS4KLSAgICAgKiBAdGhyb3dzIE9iamVjdFN0cmVhbUV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZXJlIGlzIGFuIGVycm9yIGluIHRoZSBzZXJpYWxpemVkIGZpbGVzIG9yIGR1cmluZyB0aGUKLSAgICAgKiAgICAgICAgICAgICBwcm9jZXNzIG9mIHJlYWRpbmcgdGhlbS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgT2JqZWN0IHJlYWRSZXNvbHZlKCkgdGhyb3dzIE9iamVjdFN0cmVhbUV4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBvcGVuZWRQcm9maWxlT2JqZWN0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFdyaXRlcyB0aGUgSUNDX1Byb2ZpbGUgdG8gYW4gT3V0cHV0U3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgT3V0cHV0U3RyZWFtLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBzaWduYWxzIHRoYXQgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQgZHVyaW5nIHdyaXRpbmcgb3IKLSAgICAgKiAgICAgICAgICAgICBvcGVuaW5nIE91dHB1dFN0cmVhbS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB3cml0ZShPdXRwdXRTdHJlYW0gcykgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcy53cml0ZShnZXREYXRhKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgYSB0YWdnZWQgZGF0YSBlbGVtZW50IGluIHRoZSBwcm9maWxlIGZyb20gYSBieXRlIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB0YWdTaWduYXR1cmUKLSAgICAgKiAgICAgICAgICAgIHRoZSBJQ0MgdGFnIHNpZ25hdHVyZSBmb3IgdGhlIGRhdGEgZWxlbWVudCB0byBiZSBzZXQuCi0gICAgICogQHBhcmFtIHRhZ0RhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHRvIGJlIHNldCBmb3IgdGhlIHNwZWNpZmllZCB0YWcgc2lnbmF0dXJlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldERhdGEoaW50IHRhZ1NpZ25hdHVyZSwgYnl0ZVtdIHRhZ0RhdGEpIHsKLSAgICAgICAgTmF0aXZlQ01NLmNtbVNldFByb2ZpbGVFbGVtZW50KHByb2ZpbGVIYW5kbGUsIHRhZ1NpZ25hdHVyZSwgdGFnRGF0YSk7Ci0gICAgICAgIC8vIFJlbW92ZSBjYWNoZWQgaGVhZGVyIGRhdGEgaWYgaGVhZGVyIGlzIG1vZGlmaWVkCi0gICAgICAgIGlmICh0YWdTaWduYXR1cmUgPT0gaWNTaWdIZWFkKSB7Ci0gICAgICAgICAgICBoZWFkZXJEYXRhID0gbnVsbDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSB0YWdnZWQgZGF0YSBlbGVtZW50IGZyb20gdGhlIHByb2ZpbGUgYXMgYSBieXRlIGFycmF5LiBFbGVtZW50cyBhcmUKLSAgICAgKiBpZGVudGlmaWVkIGJ5IHRhZyBzaWduYXR1cmVzIGFzIGRlZmluZWQgaW4gdGhlIElDQyBzcGVjaWZpY2F0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0YWdTaWduYXR1cmUKLSAgICAgKiAgICAgICAgICAgIHRoZSBJQ0MgdGFnIHNpZ25hdHVyZSBmb3IgdGhlIGRhdGEgZWxlbWVudCB0byBnZXQuCi0gICAgICogQHJldHVybiBhIGJ5dGUgYXJyYXkgdGhhdCBjb250YWlucyB0aGUgdGFnZ2VkIGRhdGEgZWxlbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYnl0ZVtdIGdldERhdGEoaW50IHRhZ1NpZ25hdHVyZSkgewotICAgICAgICBpbnQgdGFnU2l6ZSA9IDA7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICB0YWdTaXplID0gTmF0aXZlQ01NLmNtbUdldFByb2ZpbGVFbGVtZW50U2l6ZShwcm9maWxlSGFuZGxlLCB0YWdTaWduYXR1cmUpOwotICAgICAgICB9IGNhdGNoIChDTU1FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgLy8gV2UnbGwgZ2V0IHRoaXMgZXhjZXB0aW9uIGlmIHRoZXJlJ3Mgbm8gZWxlbWVudCB3aXRoCi0gICAgICAgICAgICAvLyB0aGUgc3BlY2lmaWVkIHRhZyBzaWduYXR1cmUKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgYnl0ZVtdIGRhdGEgPSBuZXcgYnl0ZVt0YWdTaXplXTsKLSAgICAgICAgTmF0aXZlQ01NLmNtbUdldFByb2ZpbGVFbGVtZW50KHByb2ZpbGVIYW5kbGUsIHRhZ1NpZ25hdHVyZSwgZGF0YSk7Ci0gICAgICAgIHJldHVybiBkYXRhOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBkYXRhIGJ5dGUgYXJyYXkgb2YgdGhpcyBJQ0NfUHJvZmlsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgYnl0ZSBhcnJheSB0aGF0IGNvbnRhaW5zIHRoZSBJQ0MgUHJvZmlsZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBieXRlW10gZ2V0RGF0YSgpIHsKLSAgICAgICAgaW50IHByb2ZpbGVTaXplID0gTmF0aXZlQ01NLmNtbUdldFByb2ZpbGVTaXplKHByb2ZpbGVIYW5kbGUpOwotICAgICAgICBieXRlW10gZGF0YSA9IG5ldyBieXRlW3Byb2ZpbGVTaXplXTsKLSAgICAgICAgTmF0aXZlQ01NLmNtbUdldFByb2ZpbGUocHJvZmlsZUhhbmRsZSwgZGF0YSk7Ci0gICAgICAgIHJldHVybiBkYXRhOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZyZWVzIHRoZSByZXNvdXJjZXMgYXNzb2NpYXRlZCB3aXRoIGFuIElDQ19Qcm9maWxlIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgdm9pZCBmaW5hbGl6ZSgpIHsKLSAgICAgICAgaWYgKHByb2ZpbGVIYW5kbGUgIT0gMCAmJiAhaGFuZGxlU3RvbGVuKSB7Ci0gICAgICAgICAgICBOYXRpdmVDTU0uY21tQ2xvc2VQcm9maWxlKHByb2ZpbGVIYW5kbGUpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gQWx3YXlzIHJlbW92ZSBiZWNhdXNlIGtleSBubyBtb3JlIGV4aXN0Ci0gICAgICAgIC8vIHdoZW4gb2JqZWN0IGlzIGRlc3Ryb3llZAotICAgICAgICBOYXRpdmVDTU0ucmVtb3ZlSGFuZGxlKHRoaXMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHByb2ZpbGUgY2xhc3MuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgcHJvZmlsZSBjbGFzcyBjb25zdGFudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFByb2ZpbGVDbGFzcygpIHsKLSAgICAgICAgaW50IGRldmljZUNsYXNzU2lnbmF0dXJlID0gZ2V0SW50RnJvbUhlYWRlcihpY0hkckRldmljZUNsYXNzKTsKLQotICAgICAgICBzd2l0Y2ggKGRldmljZUNsYXNzU2lnbmF0dXJlKSB7Ci0gICAgICAgICAgICBjYXNlIGljU2lnQ29sb3JTcGFjZUNsYXNzOgotICAgICAgICAgICAgICAgIHJldHVybiBDTEFTU19DT0xPUlNQQUNFQ09OVkVSU0lPTjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdEaXNwbGF5Q2xhc3M6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENMQVNTX0RJU1BMQVk7Ci0gICAgICAgICAgICBjYXNlIGljU2lnT3V0cHV0Q2xhc3M6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENMQVNTX09VVFBVVDsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdJbnB1dENsYXNzOgotICAgICAgICAgICAgICAgIHJldHVybiBDTEFTU19JTlBVVDsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdMaW5rQ2xhc3M6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENMQVNTX0RFVklDRUxJTks7Ci0gICAgICAgICAgICBjYXNlIGljU2lnQWJzdHJhY3RDbGFzczoKLSAgICAgICAgICAgICAgICByZXR1cm4gQ0xBU1NfQUJTVFJBQ1Q7Ci0gICAgICAgICAgICBjYXNlIGljU2lnTmFtZWRDb2xvckNsYXNzOgotICAgICAgICAgICAgICAgIHJldHVybiBDTEFTU19OQU1FRENPTE9SOwotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgfQotCi0gICAgICAgIC8vIE5vdCBhbiBJQ0MgcHJvZmlsZSBjbGFzcwotICAgICAgICAvLyBhd3QuMTVGPVByb2ZpbGUgY2xhc3MgZG9lcyBub3QgY29tcGx5IHdpdGggSUNDIHNwZWNpZmljYXRpb24KLSAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNUYiKSk7IC8vJE5PTi1OTFMtMSQKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvbG9yIHNwYWNlIHR5cGUgb2YgdGhlIFByb2ZpbGUgQ29ubmVjdGlvbiBTcGFjZSAoUENTKS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBQQ1MgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFBDU1R5cGUoKSB7Ci0gICAgICAgIHJldHVybiBjc0Zyb21TaWduYXR1cmUoZ2V0SW50RnJvbUhlYWRlcihpY0hkclBjcykpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG51bWJlciBvZiBjb21wb25lbnRzIG9mIHRoaXMgSUNDIFByb2ZpbGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIGNvbXBvbmVudHMgb2YgdGhpcyBJQ0MgUHJvZmlsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE51bUNvbXBvbmVudHMoKSB7Ci0gICAgICAgIHN3aXRjaCAoZ2V0SW50RnJvbUhlYWRlcihpY0hkckNvbG9yU3BhY2UpKSB7Ci0gICAgICAgICAgICAvLyBUaGUgbW9zdCBjb21tb24gY2FzZXMgZ28gZmlyc3QgdG8gaW5jcmVhc2Ugc3BlZWQKLSAgICAgICAgICAgIGNhc2UgaWNTaWdSZ2JEYXRhOgotICAgICAgICAgICAgY2FzZSBpY1NpZ1hZWkRhdGE6Ci0gICAgICAgICAgICBjYXNlIGljU2lnTGFiRGF0YToKLSAgICAgICAgICAgICAgICByZXR1cm4gMzsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdDbXlrRGF0YToKLSAgICAgICAgICAgICAgICByZXR1cm4gNDsKLSAgICAgICAgICAgICAgICAvLyBUaGVuIGFsbCBvdGhlcgotICAgICAgICAgICAgY2FzZSBpY1NpZ0dyYXlEYXRhOgotICAgICAgICAgICAgICAgIHJldHVybiAxOwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlMkNMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gMjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdZQ2JDckRhdGE6Ci0gICAgICAgICAgICBjYXNlIGljU2lnTHV2RGF0YToKLSAgICAgICAgICAgIGNhc2UgaWNTaWdZeHlEYXRhOgotICAgICAgICAgICAgY2FzZSBpY1NpZ0hsc0RhdGE6Ci0gICAgICAgICAgICBjYXNlIGljU2lnSHN2RGF0YToKLSAgICAgICAgICAgIGNhc2UgaWNTaWdDbXlEYXRhOgotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlM0NMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gMzsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdTcGFjZTRDTFI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIDQ7Ci0gICAgICAgICAgICBjYXNlIGljU2lnU3BhY2U1Q0xSOgotICAgICAgICAgICAgICAgIHJldHVybiA1OwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlNkNMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gNjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdTcGFjZTdDTFI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIDc7Ci0gICAgICAgICAgICBjYXNlIGljU2lnU3BhY2U4Q0xSOgotICAgICAgICAgICAgICAgIHJldHVybiA4OwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlOUNMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gOTsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdTcGFjZUFDTFI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIDEwOwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlQkNMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gMTE7Ci0gICAgICAgICAgICBjYXNlIGljU2lnU3BhY2VDQ0xSOgotICAgICAgICAgICAgICAgIHJldHVybiAxMjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdTcGFjZURDTFI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIDEzOwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlRUNMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gMTQ7Ci0gICAgICAgICAgICBjYXNlIGljU2lnU3BhY2VGQ0xSOgotICAgICAgICAgICAgICAgIHJldHVybiAxNTsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBhd3QuMTYwPUNvbG9yIHNwYWNlIGRvZXNuJ3QgY29tcGx5IHdpdGggSUNDIHNwZWNpZmljYXRpb24KLSAgICAgICAgdGhyb3cgbmV3IFByb2ZpbGVEYXRhRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE2MCIpIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtaW5vciB2ZXJzaW9uIG9mIHRoaXMgSUNDIHByb2ZpbGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWlub3IgdmVyc2lvbiBvZiB0aGlzIElDQyBwcm9maWxlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TWlub3JWZXJzaW9uKCkgewotICAgICAgICByZXR1cm4gZ2V0Qnl0ZUZyb21IZWFkZXIoaWNIZHJWZXJzaW9uICsgMSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWFqb3IgdmVyc2lvbiBvZiB0aGlzIElDQyBwcm9maWxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG1ham9yIHZlcnNpb24gb2YgdGhpcyBJQ0MgcHJvZmlsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE1ham9yVmVyc2lvbigpIHsKLSAgICAgICAgcmV0dXJuIGdldEJ5dGVGcm9tSGVhZGVyKGljSGRyVmVyc2lvbik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29sb3Igc3BhY2UgdHlwZSBvZiB0aGlzIElDQ19Qcm9maWxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGNvbG9yIHNwYWNlIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRDb2xvclNwYWNlVHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIGNzRnJvbVNpZ25hdHVyZShnZXRJbnRGcm9tSGVhZGVyKGljSGRyQ29sb3JTcGFjZSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRyaWVzIHRvIG9wZW4gdGhlIGZpbGUgYXQgdGhlIHNwZWNpZmllZCBwYXRoLiBQYXRoIGVudHJpZXMgY2FuIGJlIGRpdmlkZWQKLSAgICAgKiBieSBhIHNlcGFyYXRvciBjaGFyYWN0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBhdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBwYXRoIHRvIHRoZSBmaWxlLgotICAgICAqIEBwYXJhbSBmaWxlTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGZpbGUgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbnB1dCBzdHJlYW0gdG8gcmVhZCB0aGUgZmlsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBGaWxlSW5wdXRTdHJlYW0gdHJ5UGF0aChTdHJpbmcgcGF0aCwgU3RyaW5nIGZpbGVOYW1lKSB7Ci0gICAgICAgIEZpbGVJbnB1dFN0cmVhbSBmaVN0cmVhbSA9IG51bGw7Ci0KLSAgICAgICAgaWYgKHBhdGggPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICBTdHJpbmdUb2tlbml6ZXIgc3QgPSBuZXcgU3RyaW5nVG9rZW5pemVyKHBhdGgsIEZpbGUucGF0aFNlcGFyYXRvcik7Ci0KLSAgICAgICAgd2hpbGUgKHN0Lmhhc01vcmVUb2tlbnMoKSkgewotICAgICAgICAgICAgU3RyaW5nIHBhdGhFbnRyeSA9IHN0Lm5leHRUb2tlbigpOwotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICBmaVN0cmVhbSA9IG5ldyBGaWxlSW5wdXRTdHJlYW0ocGF0aEVudHJ5ICsgRmlsZS5zZXBhcmF0b3JDaGFyICsgZmlsZU5hbWUpOwotICAgICAgICAgICAgICAgIGlmIChmaVN0cmVhbSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBmaVN0cmVhbTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGNhdGNoIChGaWxlTm90Rm91bmRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGZpU3RyZWFtOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNpbmdsZSBpbnN0YW5jZSBvZiBJQ0NfUHJvZmlsZSBmcm9tIGRhdGEgaW4gdGhlIHNwZWNpZmllZCBmaWxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBmaWxlTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBuYW1lIG9mIGZpbGUgd2l0aCBJQ0MgcHJvZmlsZSBkYXRhLgotICAgICAqIEByZXR1cm4gc2luZ2xlIGluc3RhbmNlIG9mIElDQ19Qcm9maWxlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBzaWduYWxzIHRoYXQgYW4gSS9PIGVycm9yIG9jY3VycmVkIHdoaWxlIHJlYWRpbmcgdGhlIGZpbGUgb3IKLSAgICAgKiAgICAgICAgICAgICB0aGUgZmlsZSBkb2VzIG5vdCBleGlzdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIElDQ19Qcm9maWxlIGdldEluc3RhbmNlKFN0cmluZyBmaWxlTmFtZSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgZmluYWwgU3RyaW5nIGZOYW1lID0gZmlsZU5hbWU7IC8vIHRvIHVzZSBpbiB0aGUgcHJpdmlsZWdlZCBibG9jawotCi0gICAgICAgIEZpbGVJbnB1dFN0cmVhbSBmaVN0cmVhbSA9IChGaWxlSW5wdXRTdHJlYW0pQWNjZXNzQ29udHJvbGxlcgotICAgICAgICAgICAgICAgIC5kb1ByaXZpbGVnZWQobmV3IFByaXZpbGVnZWRBY3Rpb248RmlsZUlucHV0U3RyZWFtPigpIHsKLSAgICAgICAgICAgICAgICAgICAgcHVibGljIEZpbGVJbnB1dFN0cmVhbSBydW4oKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBGaWxlSW5wdXRTdHJlYW0gZmlTdHJlYW0gPSBudWxsOwotCi0gICAgICAgICAgICAgICAgICAgICAgICAvLyBPcGVuIGFic29sdXRlIHBhdGgKLSAgICAgICAgICAgICAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZmlTdHJlYW0gPSBuZXcgRmlsZUlucHV0U3RyZWFtKGZOYW1lKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoZmlTdHJlYW0gIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gZmlTdHJlYW07Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfSBjYXRjaCAoRmlsZU5vdEZvdW5kRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICAgICAgLy8gQ2hlY2sgamF2YS5pY2Nwcm9maWxlLnBhdGggZW50cmllcwotICAgICAgICAgICAgICAgICAgICAgICAgZmlTdHJlYW0gPSB0cnlQYXRoKFN5c3RlbS5nZXRQcm9wZXJ0eSgiamF2YS5pY2Nwcm9maWxlLnBhdGgiKSwgZk5hbWUpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoZmlTdHJlYW0gIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBmaVN0cmVhbTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICAgICAgLy8gQ2hlY2sgamF2YS5jbGFzcy5wYXRoIGVudHJpZXMKLSAgICAgICAgICAgICAgICAgICAgICAgIGZpU3RyZWFtID0gdHJ5UGF0aChTeXN0ZW0uZ2V0UHJvcGVydHkoImphdmEuY2xhc3MucGF0aCIpLCBmTmFtZSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChmaVN0cmVhbSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZpU3RyZWFtOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgICAgICAvLyBDaGVjayBkaXJlY3Rvcnkgd2l0aCBqYXZhIHNhbXBsZSBwcm9maWxlcwotICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nIGhvbWUgPSBTeXN0ZW0uZ2V0UHJvcGVydHkoImphdmEuaG9tZSIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoaG9tZSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZmlTdHJlYW0gPSB0cnlQYXRoKGhvbWUgKyBGaWxlLnNlcGFyYXRvckNoYXIKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICsgImxpYiIgKyBGaWxlLnNlcGFyYXRvckNoYXIgKyAiY21tIiwgZk5hbWUgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZpU3RyZWFtOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSk7Ci0KLSAgICAgICAgaWYgKGZpU3RyZWFtID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xNjE9VW5hYmxlIHRvIG9wZW4gZmlsZSB7MH0KLSAgICAgICAgICAgIHRocm93IG5ldyBJT0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNjEiLCBmaWxlTmFtZSkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBJQ0NfUHJvZmlsZSBwZiA9IGdldEluc3RhbmNlKGZpU3RyZWFtKTsKLSAgICAgICAgZmlTdHJlYW0uY2xvc2UoKTsKLSAgICAgICAgcmV0dXJuIHBmOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNpbmdsZSBpbnN0YW5jZSBvZiBJQ0NfUHJvZmlsZSB3aXRoIGRhdGEgaW4gdGhlIHNwZWNpZmllZAotICAgICAqIElucHV0U3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgSW5wdXRTdHJlYW0gd2l0aCBJQ0MgcHJvZmlsZSBkYXRhLgotICAgICAqIEByZXR1cm4gc2luZ2xlIGluc3RhbmNlIG9mIElDQ19Qcm9maWxlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZCBkdXJpbmcgcmVhZGluZyBmcm9tCi0gICAgICogICAgICAgICAgICAgSW5wdXRTdHJlYW0uCi0gICAgICogQHRocm93cyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgZmlsZSBkb2VzIG5vdCBjb250YWluIHZhbGlkIElDQyBQcm9maWxlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJQ0NfUHJvZmlsZSBnZXRJbnN0YW5jZShJbnB1dFN0cmVhbSBzKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBieXRlW10gaGVhZGVyID0gbmV3IGJ5dGVbaGVhZGVyU2l6ZV07Ci0gICAgICAgIC8vIGF3dC4xNjI9SW52YWxpZCBJQ0MgUHJvZmlsZSBEYXRhCi0gICAgICAgIFN0cmluZyBpbnZhbGlkRGF0YU1lc3NhZ2UgPSBNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNjIiKTsgLy8kTk9OLU5MUy0xJAotCi0gICAgICAgIC8vIEdldCBoZWFkZXIgZnJvbSB0aGUgaW5wdXQgc3RyZWFtCi0gICAgICAgIGlmIChzLnJlYWQoaGVhZGVyKSAhPSBoZWFkZXJTaXplKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKGludmFsaWREYXRhTWVzc2FnZSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBDaGVjayB0aGUgcHJvZmlsZSBkYXRhIGZvciBjb25zaXN0ZW5jeQotICAgICAgICBpZiAoSUNDX1Byb2ZpbGVIZWxwZXIuZ2V0QmlnRW5kaWFuRnJvbUJ5dGVBcnJheShoZWFkZXIsIGljSGRyTWFnaWMpICE9IGhlYWRlck1hZ2ljTnVtYmVyKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKGludmFsaWREYXRhTWVzc2FnZSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBHZXQgcHJvZmlsZSBzaXplIGZyb20gaGVhZGVyLCBjcmVhdGUgYW4gYXJyYXkgZm9yIHByb2ZpbGUgZGF0YQotICAgICAgICBpbnQgcHJvZmlsZVNpemUgPSBJQ0NfUHJvZmlsZUhlbHBlci5nZXRCaWdFbmRpYW5Gcm9tQnl0ZUFycmF5KGhlYWRlciwgaWNIZHJTaXplKTsKLSAgICAgICAgYnl0ZVtdIHByb2ZpbGVEYXRhID0gbmV3IGJ5dGVbcHJvZmlsZVNpemVdOwotCi0gICAgICAgIC8vIENvcHkgaGVhZGVyIGludG8gaXQKLSAgICAgICAgU3lzdGVtLmFycmF5Y29weShoZWFkZXIsIDAsIHByb2ZpbGVEYXRhLCAwLCBoZWFkZXJTaXplKTsKLQotICAgICAgICAvLyBSZWFkIHRoZSBwcm9maWxlIGl0c2VsZgotICAgICAgICBpZiAocy5yZWFkKHByb2ZpbGVEYXRhLCBoZWFkZXJTaXplLCBwcm9maWxlU2l6ZSAtIGhlYWRlclNpemUpICE9IHByb2ZpbGVTaXplIC0gaGVhZGVyU2l6ZSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihpbnZhbGlkRGF0YU1lc3NhZ2UpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGdldEluc3RhbmNlKHByb2ZpbGVEYXRhKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzaW5nbGUgaW5zdGFuY2Ugb2YgSUNDX1Byb2ZpbGUgZnJvbSB0aGUgc3BlY2lmaWVkIGRhdGEgaW4gYSBieXRlCi0gICAgICogYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBieXRlIGFycmF5IG9mIElDQyBwcm9maWxlLgotICAgICAqIEByZXR1cm4gc2luZ2xlIGluc3RhbmNlIG9mIElDQ19Qcm9maWxlIGZyb20gdGhlIHNwZWNpZmllZCBkYXRhIGluIGEgYnl0ZQotICAgICAqICAgICAgICAgYXJyYXkuCi0gICAgICogQHRocm93cyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgZmlsZSBkb2VzIG5vdCBjb250YWluIHZhbGlkIElDQyBQcm9maWxlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJQ0NfUHJvZmlsZSBnZXRJbnN0YW5jZShieXRlW10gZGF0YSkgewotICAgICAgICBJQ0NfUHJvZmlsZSByZXMgPSBudWxsOwotCi0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXMgPSBuZXcgSUNDX1Byb2ZpbGUoZGF0YSk7Ci0gICAgICAgIH0gY2F0Y2ggKENNTUV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTYyPUludmFsaWQgSUNDIFByb2ZpbGUgRGF0YQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNjIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChTeXN0ZW0uZ2V0UHJvcGVydHkoIm9zLm5hbWUiKS50b0xvd2VyQ2FzZSgpLmluZGV4T2YoIndpbmRvd3MiKSA+PSAwKSB7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICBpZiAocmVzLmdldENvbG9yU3BhY2VUeXBlKCkgPT0gQ29sb3JTcGFjZS5UWVBFX1JHQgotICAgICAgICAgICAgICAgICAgICAgICAgJiYgcmVzLmdldERhdGFTaXplKGljU2lnTWVkaWFXaGl0ZVBvaW50VGFnKSA+IDAKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIHJlcy5nZXREYXRhU2l6ZShpY1NpZ1JlZENvbG9yYW50VGFnKSA+IDAKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIHJlcy5nZXREYXRhU2l6ZShpY1NpZ0dyZWVuQ29sb3JhbnRUYWcpID4gMAotICAgICAgICAgICAgICAgICAgICAgICAgJiYgcmVzLmdldERhdGFTaXplKGljU2lnQmx1ZUNvbG9yYW50VGFnKSA+IDAKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIHJlcy5nZXREYXRhU2l6ZShpY1NpZ1JlZFRSQ1RhZykgPiAwCi0gICAgICAgICAgICAgICAgICAgICAgICAmJiByZXMuZ2V0RGF0YVNpemUoaWNTaWdHcmVlblRSQ1RhZykgPiAwCi0gICAgICAgICAgICAgICAgICAgICAgICAmJiByZXMuZ2V0RGF0YVNpemUoaWNTaWdCbHVlVFJDVGFnKSA+IDApIHsKLSAgICAgICAgICAgICAgICAgICAgcmVzID0gbmV3IElDQ19Qcm9maWxlUkdCKHJlcy5nZXRQcm9maWxlSGFuZGxlKCkpOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAocmVzLmdldENvbG9yU3BhY2VUeXBlKCkgPT0gQ29sb3JTcGFjZS5UWVBFX0dSQVkKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIHJlcy5nZXREYXRhU2l6ZShpY1NpZ01lZGlhV2hpdGVQb2ludFRhZykgPiAwCi0gICAgICAgICAgICAgICAgICAgICAgICAmJiByZXMuZ2V0RGF0YVNpemUoaWNTaWdHcmF5VFJDVGFnKSA+IDApIHsKLSAgICAgICAgICAgICAgICAgICAgcmVzID0gbmV3IElDQ19Qcm9maWxlR3JheShyZXMuZ2V0UHJvZmlsZUhhbmRsZSgpKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIH0gY2F0Y2ggKENNTUV4Y2VwdGlvbiBlKSB7IC8qIHJldHVybiByZXMgaW4gdGhpcyBjYXNlICovCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcmVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNpbmdsZSBpbnN0YW5jZSBvZiBJQ0NfUHJvZmlsZSB3aXRoIHRoZSBzcGVjaWZpYyBjb2xvciBzcGFjZQotICAgICAqIGRlZmluZWQgYnkgdGhlIENvbG9yU3BhY2UgY2xhc3M6IENTX3NSR0IsIENTX0xJTkVBUl9SR0IsIENTX0NJRVhZWiwKLSAgICAgKiBDU19QWUNDLCBDU19HUkFZLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjc3BhY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSB0eXBlIG9mIGNvbG9yIHNwYWNlIGRlZmluZWQgaW4gdGhlIENvbG9yU3BhY2UgY2xhc3MuCi0gICAgICogQHJldHVybiBzaW5nbGUgaW5zdGFuY2Ugb2YgSUNDX1Byb2ZpbGUuCi0gICAgICogQHRocm93cyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpcyBub3Qgb25lIG9mIHRoZSBkZWZpbmVkIGNvbG9yIHNwYWNlIHR5cGVzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgSUNDX1Byb2ZpbGUgZ2V0SW5zdGFuY2UoaW50IGNzcGFjZSkgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgc3dpdGNoIChjc3BhY2UpIHsKLQotICAgICAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5DU19zUkdCOgotICAgICAgICAgICAgICAgICAgICBpZiAoc1JHQlByb2ZpbGUgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICAgICAgc1JHQlByb2ZpbGUgPSBnZXRJbnN0YW5jZSgic1JHQi5wZiIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNSR0JQcm9maWxlOwotCi0gICAgICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLkNTX0NJRVhZWjoKLSAgICAgICAgICAgICAgICAgICAgaWYgKHh5elByb2ZpbGUgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICAgICAgeHl6UHJvZmlsZSA9IGdldEluc3RhbmNlKCJDSUVYWVoucGYiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB4eXpQcm9maWxlOwotCi0gICAgICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLkNTX0dSQVk6Ci0gICAgICAgICAgICAgICAgICAgIGlmIChncmF5UHJvZmlsZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBncmF5UHJvZmlsZSA9IGdldEluc3RhbmNlKCJHUkFZLnBmIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICByZXR1cm4gZ3JheVByb2ZpbGU7Ci0KLSAgICAgICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuQ1NfUFlDQzoKLSAgICAgICAgICAgICAgICAgICAgaWYgKHB5Y2NQcm9maWxlID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHB5Y2NQcm9maWxlID0gZ2V0SW5zdGFuY2UoIlBZQ0MucGYiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBweWNjUHJvZmlsZTsKLQotICAgICAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5DU19MSU5FQVJfUkdCOgotICAgICAgICAgICAgICAgICAgICBpZiAobGluZWFyUkdCUHJvZmlsZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBsaW5lYXJSR0JQcm9maWxlID0gZ2V0SW5zdGFuY2UoIkxJTkVBUl9SR0IucGYiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBsaW5lYXJSR0JQcm9maWxlOwotICAgICAgICAgICAgfQotCi0gICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xNjM9Q2FuJ3Qgb3BlbiBjb2xvciBwcm9maWxlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiQ2FuJ3Qgb3BlbiBjb2xvciBwcm9maWxlIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICAvLyBhd3QuMTY0PU5vdCBhIHByZWRlZmluZWQgY29sb3Igc3BhY2UKLSAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoIk5vdCBhIHByZWRlZmluZWQgY29sb3Igc3BhY2UiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyBhbiBpbnRlZ2VyIGZyb20gdGhlIHByb2ZpbGUgaGVhZGVyIGF0IHRoZSBzcGVjaWZpZWQgcG9zaXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIGlkeAotICAgICAqICAgICAgICAgICAgLSBvZmZzZXQgaW4gYnl0ZXMgZnJvbSB0aGUgYmVnaW5uaW5nIG9mIHRoZSBoZWFkZXIKLSAgICAgKiBAcmV0dXJuIHRoZSBpbnRlZ2VyIHZhbHVlIGZyb20gaGVhZGVyCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgZ2V0SW50RnJvbUhlYWRlcihpbnQgaWR4KSB7Ci0gICAgICAgIGlmIChoZWFkZXJEYXRhID09IG51bGwpIHsKLSAgICAgICAgICAgIGhlYWRlckRhdGEgPSBnZXREYXRhKGljU2lnSGVhZCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gKChoZWFkZXJEYXRhW2lkeF0gJiAweEZGKSA8PCAyNCkgfCAoKGhlYWRlckRhdGFbaWR4ICsgMV0gJiAweEZGKSA8PCAxNikKLSAgICAgICAgICAgICAgICB8ICgoaGVhZGVyRGF0YVtpZHggKyAyXSAmIDB4RkYpIDw8IDgpIHwgKChoZWFkZXJEYXRhW2lkeCArIDNdICYgMHhGRikpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlYWRzIGJ5dGUgZnJvbSB0aGUgcHJvZmlsZSBoZWFkZXIgYXQgdGhlIHNwZWNpZmllZCBwb3NpdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWR4Ci0gICAgICogICAgICAgICAgICAtIG9mZnNldCBpbiBieXRlcyBmcm9tIHRoZSBiZWdpbm5pbmcgb2YgdGhlIGhlYWRlcgotICAgICAqIEByZXR1cm4gdGhlIGJ5dGUgZnJvbSBoZWFkZXIKLSAgICAgKi8KLSAgICBwcml2YXRlIGJ5dGUgZ2V0Qnl0ZUZyb21IZWFkZXIoaW50IGlkeCkgewotICAgICAgICBpZiAoaGVhZGVyRGF0YSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBoZWFkZXJEYXRhID0gZ2V0RGF0YShpY1NpZ0hlYWQpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGhlYWRlckRhdGFbaWR4XTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb252ZXJ0cyBJQ0MgY29sb3Igc3BhY2Ugc2lnbmF0dXJlIHRvIHRoZSBqYXZhIHByZWRlZmluZWQgY29sb3Igc3BhY2UKLSAgICAgKiB0eXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzaWduYXR1cmUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzaWduYXR1cmUKLSAgICAgKiBAcmV0dXJuIHRoZSBpbnQKLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBjc0Zyb21TaWduYXR1cmUoaW50IHNpZ25hdHVyZSkgewotICAgICAgICBzd2l0Y2ggKHNpZ25hdHVyZSkgewotICAgICAgICAgICAgY2FzZSBpY1NpZ1JnYkRhdGE6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENvbG9yU3BhY2UuVFlQRV9SR0I7Ci0gICAgICAgICAgICBjYXNlIGljU2lnWFlaRGF0YToKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX1hZWjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdDbXlrRGF0YToKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX0NNWUs7Ci0gICAgICAgICAgICBjYXNlIGljU2lnTGFiRGF0YToKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX0xhYjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdHcmF5RGF0YToKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX0dSQVk7Ci0gICAgICAgICAgICBjYXNlIGljU2lnSGxzRGF0YToKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX0hMUzsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdMdXZEYXRhOgotICAgICAgICAgICAgICAgIHJldHVybiBDb2xvclNwYWNlLlRZUEVfTHV2OwotICAgICAgICAgICAgY2FzZSBpY1NpZ1lDYkNyRGF0YToKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX1lDYkNyOwotICAgICAgICAgICAgY2FzZSBpY1NpZ1l4eURhdGE6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENvbG9yU3BhY2UuVFlQRV9ZeHk7Ci0gICAgICAgICAgICBjYXNlIGljU2lnSHN2RGF0YToKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX0hTVjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdDbXlEYXRhOgotICAgICAgICAgICAgICAgIHJldHVybiBDb2xvclNwYWNlLlRZUEVfQ01ZOwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlMkNMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFXzJDTFI7Ci0gICAgICAgICAgICBjYXNlIGljU2lnU3BhY2UzQ0xSOgotICAgICAgICAgICAgICAgIHJldHVybiBDb2xvclNwYWNlLlRZUEVfM0NMUjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdTcGFjZTRDTFI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENvbG9yU3BhY2UuVFlQRV80Q0xSOwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlNUNMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFXzVDTFI7Ci0gICAgICAgICAgICBjYXNlIGljU2lnU3BhY2U2Q0xSOgotICAgICAgICAgICAgICAgIHJldHVybiBDb2xvclNwYWNlLlRZUEVfNkNMUjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdTcGFjZTdDTFI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENvbG9yU3BhY2UuVFlQRV83Q0xSOwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlOENMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFXzhDTFI7Ci0gICAgICAgICAgICBjYXNlIGljU2lnU3BhY2U5Q0xSOgotICAgICAgICAgICAgICAgIHJldHVybiBDb2xvclNwYWNlLlRZUEVfOUNMUjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdTcGFjZUFDTFI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENvbG9yU3BhY2UuVFlQRV9BQ0xSOwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlQkNMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX0JDTFI7Ci0gICAgICAgICAgICBjYXNlIGljU2lnU3BhY2VDQ0xSOgotICAgICAgICAgICAgICAgIHJldHVybiBDb2xvclNwYWNlLlRZUEVfQ0NMUjsKLSAgICAgICAgICAgIGNhc2UgaWNTaWdTcGFjZURDTFI6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENvbG9yU3BhY2UuVFlQRV9EQ0xSOwotICAgICAgICAgICAgY2FzZSBpY1NpZ1NwYWNlRUNMUjoKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX0VDTFI7Ci0gICAgICAgICAgICBjYXNlIGljU2lnU3BhY2VGQ0xSOgotICAgICAgICAgICAgICAgIHJldHVybiBDb2xvclNwYWNlLlRZUEVfRkNMUjsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBhd3QuMTY1PUNvbG9yIHNwYWNlIGRvZXNuJ3QgY29tcGx5IHdpdGggSUNDIHNwZWNpZmljYXRpb24KLSAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNjUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBwcm9maWxlIGhhbmRsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBwcm9maWxlIGhhbmRsZQotICAgICAqLwotICAgIHByaXZhdGUgbG9uZyBnZXRQcm9maWxlSGFuZGxlKCkgewotICAgICAgICBoYW5kbGVTdG9sZW4gPSB0cnVlOwotICAgICAgICByZXR1cm4gcHJvZmlsZUhhbmRsZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkYXRhIHNpemUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRhZ1NpZ25hdHVyZQotICAgICAqICAgICAgICAgICAgdGhlIHRhZyBzaWduYXR1cmUKLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhIHNpemUKLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBnZXREYXRhU2l6ZShpbnQgdGFnU2lnbmF0dXJlKSB7Ci0gICAgICAgIHJldHVybiBOYXRpdmVDTU0uY21tR2V0UHJvZmlsZUVsZW1lbnRTaXplKHByb2ZpbGVIYW5kbGUsIHRhZ1NpZ25hdHVyZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVhZHMgWFlaIHZhbHVlIGZyb20gdGhlIHRhZyBkYXRhLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0YWdTaWduYXR1cmUKLSAgICAgKiAgICAgICAgICAgIHRoZSB0YWcgc2lnbmF0dXJlCi0gICAgICogQHJldHVybiB0aGUgWFlaIHZhbHVlCi0gICAgICovCi0gICAgZmxvYXRbXSBnZXRYWVpWYWx1ZShpbnQgdGFnU2lnbmF0dXJlKSB7Ci0gICAgICAgIGZsb2F0W10gcmVzID0gbmV3IGZsb2F0WzNdOwotICAgICAgICBieXRlW10gZGF0YSA9IGdldERhdGEodGFnU2lnbmF0dXJlKTsKLQotICAgICAgICAvLyBDb252ZXJ0IGZyb20gSUNDIHMxNUZpeGVkMTZOdW1iZXIgdHlwZQotICAgICAgICAvLyAxIChmbG9hdCkgPSAweDEwMDAwIChzMTVGaXhlZDE2TnVtYmVyKSwKLSAgICAgICAgLy8gaGVuY2UgZGl2aWRpbmcgYnkgMHgxMDAwMAotICAgICAgICByZXNbMF0gPSBJQ0NfUHJvZmlsZUhlbHBlci5nZXRJbnRGcm9tQnl0ZUFycmF5KGRhdGEsIDApIC8gNjU1MzYuZjsKLSAgICAgICAgcmVzWzFdID0gSUNDX1Byb2ZpbGVIZWxwZXIuZ2V0SW50RnJvbUJ5dGVBcnJheShkYXRhLCA0KSAvIDY1NTM2LmY7Ci0gICAgICAgIHJlc1syXSA9IElDQ19Qcm9maWxlSGVscGVyLmdldEludEZyb21CeXRlQXJyYXkoZGF0YSwgOCkgLyA2NTUzNi5mOwotCi0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWVkaWEgd2hpdGUgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWVkaWEgd2hpdGUgcG9pbnQuCi0gICAgICovCi0gICAgZmxvYXRbXSBnZXRNZWRpYVdoaXRlUG9pbnQoKSB7Ci0gICAgICAgIHJldHVybiBnZXRYWVpWYWx1ZShpY1NpZ01lZGlhV2hpdGVQb2ludFRhZyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSWYgVFJDIGlzIG5vdCBhIHRhYmxlIHJldHVybnMgZ2FtbWEgdmlhIHJldHVybiB2YWx1ZSBhbmQgc2V0cyBkYXRhVFJDIHRvCi0gICAgICogbnVsbC4gSWYgVFJDIGlzIGEgdGFibGUgcmV0dXJucyAwIGFuZCBmaWxscyBkYXRhVFJDIHdpdGggdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0YWdTaWduYXR1cmUKLSAgICAgKiAgICAgICAgICAgIHRoZSB0YWcgc2lnbmF0dXJlCi0gICAgICogQHBhcmFtIGRhdGFUUkMKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHRyYwotICAgICAqIEByZXR1cm4gLSBnYW1tYSBvciB6ZXJvIGlmIFRSQyBpcyBhIHRhYmxlCi0gICAgICovCi0gICAgcHJpdmF0ZSBmbG9hdCBnZXRHYW1tYU9yVFJDKGludCB0YWdTaWduYXR1cmUsIHNob3J0W10gZGF0YVRSQykgewotICAgICAgICBieXRlW10gZGF0YSA9IGdldERhdGEodGFnU2lnbmF0dXJlKTsKLSAgICAgICAgaW50IHRyY1NpemUgPSBJQ0NfUHJvZmlsZUhlbHBlci5nZXRJbnRGcm9tQnl0ZUFycmF5KGRhdGEsIGljQ3VydmVDb3VudCk7Ci0KLSAgICAgICAgZGF0YVRSQyA9IG51bGw7Ci0KLSAgICAgICAgaWYgKHRyY1NpemUgPT0gMCkgewotICAgICAgICAgICAgcmV0dXJuIDEuMGY7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAodHJjU2l6ZSA9PSAxKSB7Ci0gICAgICAgICAgICAvLyBDYXN0IGZyb20gSUNDIHU4Rml4ZWQ4TnVtYmVyIHRvIGZsb2F0Ci0gICAgICAgICAgICByZXR1cm4gSUNDX1Byb2ZpbGVIZWxwZXIuZ2V0U2hvcnRGcm9tQnl0ZUFycmF5KGRhdGEsIGljQ3VydmVEYXRhKSAvIDI1Ni5mOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gVFJDIGlzIGEgdGFibGUKLSAgICAgICAgZGF0YVRSQyA9IG5ldyBzaG9ydFt0cmNTaXplXTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDAsIHBvcyA9IGljQ3VydmVEYXRhOyBpIDwgdHJjU2l6ZTsgaSsrLCBwb3MgKz0gMikgewotICAgICAgICAgICAgZGF0YVRSQ1tpXSA9IElDQ19Qcm9maWxlSGVscGVyLmdldFNob3J0RnJvbUJ5dGVBcnJheShkYXRhLCBwb3MpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGdhbW1hLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0YWdTaWduYXR1cmUKLSAgICAgKiAgICAgICAgICAgIHRoZSB0YWcgc2lnbmF0dXJlCi0gICAgICogQHJldHVybiB0aGUgZ2FtbWEKLSAgICAgKi8KLSAgICBmbG9hdCBnZXRHYW1tYShpbnQgdGFnU2lnbmF0dXJlKSB7Ci0gICAgICAgIHNob3J0W10gZGF0YVRSQyA9IG51bGw7Ci0gICAgICAgIGZsb2F0IGdhbW1hID0gZ2V0R2FtbWFPclRSQyh0YWdTaWduYXR1cmUsIGRhdGFUUkMpOwotCi0gICAgICAgIGlmIChkYXRhVFJDID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBnYW1tYTsKLSAgICAgICAgfQotICAgICAgICAvLyBhd3QuMTY2PVRSQyBpcyBub3QgYSBzaW1wbGUgZ2FtbWEgdmFsdWUuCi0gICAgICAgIHRocm93IG5ldyBQcm9maWxlRGF0YUV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNjYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBUUkMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRhZ1NpZ25hdHVyZQotICAgICAqICAgICAgICAgICAgdGhlIHRhZyBzaWduYXR1cmUKLSAgICAgKiBAcmV0dXJuIHRoZSB0UkMKLSAgICAgKi8KLSAgICBzaG9ydFtdIGdldFRSQyhpbnQgdGFnU2lnbmF0dXJlKSB7Ci0gICAgICAgIHNob3J0W10gZGF0YVRSQyA9IG51bGw7Ci0gICAgICAgIGdldEdhbW1hT3JUUkModGFnU2lnbmF0dXJlLCBkYXRhVFJDKTsKLQotICAgICAgICBpZiAoZGF0YVRSQyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTY3PVRSQyBpcyBhIGdhbW1hIHZhbHVlLCBub3QgYSB0YWJsZS4KLSAgICAgICAgICAgIHRocm93IG5ldyBQcm9maWxlRGF0YUV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xNjciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZGF0YVRSQzsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvY29sb3IvSUNDX1Byb2ZpbGVHcmF5LmphdmEgYi9hd3QvamF2YS9hd3QvY29sb3IvSUNDX1Byb2ZpbGVHcmF5LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGY3NDgxMDEuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2NvbG9yL0lDQ19Qcm9maWxlR3JheS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNzggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuY29sb3I7Ci0KLS8qKgotICogVGhlIElDQ19Qcm9maWxlR3JheSBjbGFzcyByZXByZXNlbnQgcHJvZmlsZXMgd2l0aCBUWVBFX0dSQVkgY29sb3Igc3BhY2UgdHlwZSwKLSAqIGFuZCBpbmNsdWRlcyB0aGUgZ3JheVRSQ1RhZyBhbmQgbWVkaWFXaGl0ZVBvaW50VGFnIHRhZ3MuIFRoZSBncmF5IGNvbXBvbmVudAotICogY2FuIGJlIHRyYW5zZm9ybWVkIGZyb20gYSBHUkFZIGRldmljZSBwcm9maWxlIGNvbG9yIHNwYWNlIHRvIHRoZSBDSUVYWVoKLSAqIFByb2ZpbGUgdGhyb3VnaCB0aGUgdG9uZSByZXByb2R1Y3Rpb24gY3VydmUgKFRSQyk6Ci0gKiA8cD4KLSAqIFBDU1kgPSBncmF5VFJDW2RldmljZUdyYXldCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgSUNDX1Byb2ZpbGVHcmF5IGV4dGVuZHMgSUNDX1Byb2ZpbGUgewotICAgIAotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC0xMTI0NzIxMjkwNzMyMDAyNjQ5TDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBpQyBjXyBwcm9maWxlIGdyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb2ZpbGVIYW5kbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9maWxlIGhhbmRsZQotICAgICAqLwotICAgIElDQ19Qcm9maWxlR3JheShsb25nIHByb2ZpbGVIYW5kbGUpIHsKLSAgICAgICAgc3VwZXIocHJvZmlsZUhhbmRsZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgVFJDIGFzIGFuIGFycmF5IG9mIHNob3J0cy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzaG9ydCBhcnJheSBvZiB0aGUgVFJDLgotICAgICAqLwotICAgIHB1YmxpYyBzaG9ydFtdIGdldFRSQygpIHsKLSAgICAgICAgcmV0dXJuIHN1cGVyLmdldFRSQyhpY1NpZ0dyYXlUUkNUYWcpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1lZGlhIHdoaXRlIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG1lZGlhIHdoaXRlIHBvaW50Ci0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZsb2F0W10gZ2V0TWVkaWFXaGl0ZVBvaW50KCkgewotICAgICAgICByZXR1cm4gc3VwZXIuZ2V0TWVkaWFXaGl0ZVBvaW50KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIGdhbW1hIHZhbHVlIHJlcHJlc2VudGluZyB0aGUgdG9uZSByZXByb2R1Y3Rpb24gY3VydmUgKFRSQykuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZ2FtbWEgdmFsdWUgcmVwcmVzZW50aW5nIHRoZSB0b25lIHJlcHJvZHVjdGlvbiBjdXJ2ZSAoVFJDKS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0R2FtbWEoKSB7Ci0gICAgICAgIHJldHVybiBzdXBlci5nZXRHYW1tYShpY1NpZ0dyYXlUUkNUYWcpOwotICAgIH0KLX0KLQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2NvbG9yL0lDQ19Qcm9maWxlUkdCLmphdmEgYi9hd3QvamF2YS9hd3QvY29sb3IvSUNDX1Byb2ZpbGVSR0IuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOWM2MDEwZi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvY29sb3IvSUNDX1Byb2ZpbGVSR0IuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE1NCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5jb2xvcjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBJQ0NfUHJvZmlsZVJHQiBjbGFzcyByZXByZXNlbnRzIHByb2ZpbGVzIHdpdGggUkdCIGNvbG9yIHNwYWNlIHR5cGUgYW5kCi0gKiBjb250YWlucyB0aGUgcmVkQ29sb3JhbnRUYWcsIGdyZWVuQ29sb3JhbnRUYWcsIGJsdWVDb2xvcmFudFRhZywgcmVkVFJDVGFnLAotICogZ3JlZW5UUkNUYWcsIGJsdWVUUkNUYWcsIGFuZCBtZWRpYVdoaXRlUG9pbnRUYWcgdGFncy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJQ0NfUHJvZmlsZVJHQiBleHRlbmRzIElDQ19Qcm9maWxlIHsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSA4NTA1MDY3Mzg1MTUyNTc5MzM0TDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBSR0IgSUNDX1Byb2ZpbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb2ZpbGVIYW5kbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9maWxlIGhhbmRsZQotICAgICAqLwotICAgIElDQ19Qcm9maWxlUkdCKGxvbmcgcHJvZmlsZUhhbmRsZSkgewotICAgICAgICBzdXBlcihwcm9maWxlSGFuZGxlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUkVEQ09NUE9ORU5UIGluZGljYXRlcyB0aGUgcmVkIGNvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBSRURDT01QT05FTlQgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEdSRUVOQ09NUE9ORU5UIGluZGljYXRlcyB0aGUgZ3JlZW4gY29tcG9uZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEdSRUVOQ09NUE9ORU5UID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBCTFVFQ09NUE9ORU5UIGluZGljYXRlcyB0aGUgYmx1ZSBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQkxVRUNPTVBPTkVOVCA9IDI7Ci0KLSAgICAvLyBhd3QuMTVFPVVua25vd24gY29tcG9uZW50LiBNdXN0IGJlIFJFRENPTVBPTkVOVCwgR1JFRU5DT01QT05FTlQgb3IgQkxVRUNPTVBPTkVOVC4KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVU5LTk9XTl9DT01QT05FTlRfTVNHLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBVTktOT1dOX0NPTVBPTkVOVF9NU0cgPSBNZXNzYWdlcwotICAgICAgICAgICAgLmdldFN0cmluZygiYXd0LjE1RSIpOyAvLyROT04tTkxTLTEkCi0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBUUkMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbXBvbmVudAotICAgICAqICAgICAgICAgICAgdGhlIHRhZyBzaWduYXR1cmUuCi0gICAgICogQHJldHVybiB0aGUgVFJDIHZhbHVlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBzaG9ydFtdIGdldFRSQyhpbnQgY29tcG9uZW50KSB7Ci0gICAgICAgIHN3aXRjaCAoY29tcG9uZW50KSB7Ci0gICAgICAgICAgICBjYXNlIFJFRENPTVBPTkVOVDoKLSAgICAgICAgICAgICAgICByZXR1cm4gc3VwZXIuZ2V0VFJDKGljU2lnUmVkVFJDVGFnKTsKLSAgICAgICAgICAgIGNhc2UgR1JFRU5DT01QT05FTlQ6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHN1cGVyLmdldFRSQyhpY1NpZ0dyZWVuVFJDVGFnKTsKLSAgICAgICAgICAgIGNhc2UgQkxVRUNPTVBPTkVOVDoKLSAgICAgICAgICAgICAgICByZXR1cm4gc3VwZXIuZ2V0VFJDKGljU2lnQmx1ZVRSQ1RhZyk7Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICB9Ci0KLSAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihVTktOT1dOX0NPTVBPTkVOVF9NU0cpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGdhbW1hLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb21wb25lbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSB0YWcgc2lnbmF0dXJlLgotICAgICAqIEByZXR1cm4gdGhlIGdhbW1hIHZhbHVlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRHYW1tYShpbnQgY29tcG9uZW50KSB7Ci0gICAgICAgIHN3aXRjaCAoY29tcG9uZW50KSB7Ci0gICAgICAgICAgICBjYXNlIFJFRENPTVBPTkVOVDoKLSAgICAgICAgICAgICAgICByZXR1cm4gc3VwZXIuZ2V0R2FtbWEoaWNTaWdSZWRUUkNUYWcpOwotICAgICAgICAgICAgY2FzZSBHUkVFTkNPTVBPTkVOVDoKLSAgICAgICAgICAgICAgICByZXR1cm4gc3VwZXIuZ2V0R2FtbWEoaWNTaWdHcmVlblRSQ1RhZyk7Ci0gICAgICAgICAgICBjYXNlIEJMVUVDT01QT05FTlQ6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHN1cGVyLmdldEdhbW1hKGljU2lnQmx1ZVRSQ1RhZyk7Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICB9Ci0KLSAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihVTktOT1dOX0NPTVBPTkVOVF9NU0cpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBmbG9hdCBtYXRyaXggd2hpY2ggY29udGFpbnMgdGhlIFgsIFksIGFuZCBaIGNvbXBvbmVudHMgb2YgdGhlCi0gICAgICogcHJvZmlsZSdzIHJlZENvbG9yYW50VGFnLCBncmVlbkNvbG9yYW50VGFnLCBhbmQgYmx1ZUNvbG9yYW50VGFnLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGZsb2F0IG1hdHJpeCB3aGljaCBjb250YWlucyB0aGUgWCwgWSwgYW5kIFogY29tcG9uZW50cyBvZiB0aGUKLSAgICAgKiAgICAgICAgIHByb2ZpbGUncyByZWRDb2xvcmFudFRhZywgZ3JlZW5Db2xvcmFudFRhZywgYW5kIGJsdWVDb2xvcmFudFRhZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXRbXVtdIGdldE1hdHJpeCgpIHsKLSAgICAgICAgZmxvYXQgW11bXSBtID0gbmV3IGZsb2F0WzNdWzNdOyAvLyBUaGUgbWF0cml4Ci0KLSAgICAgICAgZmxvYXRbXSByZWRYWVogPSBnZXRYWVpWYWx1ZShpY1NpZ1JlZENvbG9yYW50VGFnKTsKLSAgICAgICAgZmxvYXRbXSBncmVlblhZWiA9IGdldFhZWlZhbHVlKGljU2lnR3JlZW5Db2xvcmFudFRhZyk7Ci0gICAgICAgIGZsb2F0W10gYmx1ZVhZWiA9IGdldFhZWlZhbHVlKGljU2lnQmx1ZUNvbG9yYW50VGFnKTsKLQotICAgICAgICBtWzBdWzBdID0gcmVkWFlaWzBdOwotICAgICAgICBtWzFdWzBdID0gcmVkWFlaWzFdOwotICAgICAgICBtWzJdWzBdID0gcmVkWFlaWzJdOwotCi0gICAgICAgIG1bMF1bMV0gPSBncmVlblhZWlswXTsKLSAgICAgICAgbVsxXVsxXSA9IGdyZWVuWFlaWzFdOwotICAgICAgICBtWzJdWzFdID0gZ3JlZW5YWVpbMl07Ci0KLSAgICAgICAgbVswXVsyXSA9IGJsdWVYWVpbMF07Ci0gICAgICAgIG1bMV1bMl0gPSBibHVlWFlaWzFdOwotICAgICAgICBtWzJdWzJdID0gYmx1ZVhZWlsyXTsKLQotICAgICAgICByZXR1cm4gbTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtZWRpYSB3aGl0ZSBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtZWRpYSB3aGl0ZSBwb2ludC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXRbXSBnZXRNZWRpYVdoaXRlUG9pbnQoKSB7Ci0gICAgICAgIHJldHVybiBzdXBlci5nZXRNZWRpYVdoaXRlUG9pbnQoKTsKLSAgICB9Ci19Ci0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9jb2xvci9JQ0NfUHJvZmlsZVN0dWIuamF2YSBiL2F3dC9qYXZhL2F3dC9jb2xvci9JQ0NfUHJvZmlsZVN0dWIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYmMwNGM4YS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvY29sb3IvSUNDX1Byb2ZpbGVTdHViLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNzMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5jb2xvcjsKLQotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLk9iamVjdFN0cmVhbUV4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLk91dHB1dFN0cmVhbTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi1maW5hbCBjbGFzcyBJQ0NfUHJvZmlsZVN0dWIgZXh0ZW5kcyBJQ0NfUHJvZmlsZSB7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gNTAxMzg5NzYwODc1MjUzNTA3TDsKLQotICAgIHRyYW5zaWVudCBpbnQgY29sb3JzcGFjZTsKLQotICAgIHB1YmxpYyBJQ0NfUHJvZmlsZVN0dWIoaW50IGNzU3BlY2lmaWVyKSB7Ci0gICAgICAgIHN3aXRjaCAoY3NTcGVjaWZpZXIpIHsKLSAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5DU19zUkdCOgotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLkNTX0NJRVhZWjoKLSAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5DU19MSU5FQVJfUkdCOgotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLkNTX1BZQ0M6Ci0gICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuQ1NfR1JBWToKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjE1RD1JbnZhbGlkIGNvbG9yc3BhY2UKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE1RCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGNvbG9yc3BhY2UgPSBjc1NwZWNpZmllcjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB3cml0ZShTdHJpbmcgZmlsZU5hbWUpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiU3R1YiBjYW5ub3QgcGVyZm9ybSB0aGlzIG9wZXJhdGlvbiIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2VyaWFsaXphYmxlIGltcGxlbWVudGF0aW9uCi0gICAgICoKLSAgICAgKiBAdGhyb3dzIE9iamVjdFN0cmVhbUV4Y2VwdGlvbgotICAgICAqLwotICAgIHByaXZhdGUgT2JqZWN0IHdyaXRlUmVwbGFjZSgpIHRocm93cyBPYmplY3RTdHJlYW1FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gbG9hZFByb2ZpbGUoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB3cml0ZShPdXRwdXRTdHJlYW0gcykgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJTdHViIGNhbm5vdCBwZXJmb3JtIHRoaXMgb3BlcmF0aW9uIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXREYXRhKGludCB0YWdTaWduYXR1cmUsIGJ5dGVbXSB0YWdEYXRhKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiU3R1YiBjYW5ub3QgcGVyZm9ybSB0aGlzIG9wZXJhdGlvbiIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJ5dGVbXSBnZXREYXRhKGludCB0YWdTaWduYXR1cmUpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJTdHViIGNhbm5vdCBwZXJmb3JtIHRoaXMgb3BlcmF0aW9uIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYnl0ZVtdIGdldERhdGEoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiU3R1YiBjYW5ub3QgcGVyZm9ybSB0aGlzIG9wZXJhdGlvbiIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHJvdGVjdGVkIHZvaWQgZmluYWxpemUoKSB7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRQcm9maWxlQ2xhc3MoKSB7Ci0gICAgICAgIHJldHVybiBDTEFTU19DT0xPUlNQQUNFQ09OVkVSU0lPTjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldFBDU1R5cGUoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiU3R1YiBjYW5ub3QgcGVyZm9ybSB0aGlzIG9wZXJhdGlvbiIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXROdW1Db21wb25lbnRzKCkgewotICAgICAgICBzd2l0Y2ggKGNvbG9yc3BhY2UpIHsKLSAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5DU19zUkdCOgotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLkNTX0NJRVhZWjoKLSAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5DU19MSU5FQVJfUkdCOgotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLkNTX1BZQ0M6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIDM7Ci0gICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuQ1NfR1JBWToKLSAgICAgICAgICAgICAgICByZXR1cm4gMTsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJTdHViIGNhbm5vdCBwZXJmb3JtIHRoaXMgb3BlcmF0aW9uIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0TWlub3JWZXJzaW9uKCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIlN0dWIgY2Fubm90IHBlcmZvcm0gdGhpcyBvcGVyYXRpb24iKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0TWFqb3JWZXJzaW9uKCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIlN0dWIgY2Fubm90IHBlcmZvcm0gdGhpcyBvcGVyYXRpb24iKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0Q29sb3JTcGFjZVR5cGUoKSB7Ci0gICAgICAgIHN3aXRjaCAoY29sb3JzcGFjZSkgewotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLkNTX3NSR0I6Ci0gICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuQ1NfTElORUFSX1JHQjoKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX1JHQjsKLSAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5DU19DSUVYWVo6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENvbG9yU3BhY2UuVFlQRV9YWVo7Ci0gICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuQ1NfUFlDQzoKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFXzNDTFI7Ci0gICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuQ1NfR1JBWToKLSAgICAgICAgICAgICAgICByZXR1cm4gQ29sb3JTcGFjZS5UWVBFX0dSQVk7Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiU3R1YiBjYW5ub3QgcGVyZm9ybSB0aGlzIG9wZXJhdGlvbiIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIElDQ19Qcm9maWxlIGdldEluc3RhbmNlKFN0cmluZyBmaWxlTmFtZSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJTdHViIGNhbm5vdCBwZXJmb3JtIHRoaXMgb3BlcmF0aW9uIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIElDQ19Qcm9maWxlIGdldEluc3RhbmNlKElucHV0U3RyZWFtIHMpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiU3R1YiBjYW5ub3QgcGVyZm9ybSB0aGlzIG9wZXJhdGlvbiIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyBJQ0NfUHJvZmlsZSBnZXRJbnN0YW5jZShieXRlW10gZGF0YSkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIlN0dWIgY2Fubm90IHBlcmZvcm0gdGhpcyBvcGVyYXRpb24iKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgSUNDX1Byb2ZpbGUgZ2V0SW5zdGFuY2UoaW50IGNzcGFjZSkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIlN0dWIgY2Fubm90IHBlcmZvcm0gdGhpcyBvcGVyYXRpb24iKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIHB1YmxpYyBJQ0NfUHJvZmlsZSBsb2FkUHJvZmlsZSgpIHsKLSAgICAgICAgc3dpdGNoIChjb2xvcnNwYWNlKSB7Ci0gICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuQ1Nfc1JHQjoKLSAgICAgICAgICAgICAgICByZXR1cm4gSUNDX1Byb2ZpbGUuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19zUkdCKTsKLSAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5DU19HUkFZOgotICAgICAgICAgICAgICAgIHJldHVybiBJQ0NfUHJvZmlsZS5nZXRJbnN0YW5jZShDb2xvclNwYWNlLkNTX0dSQVkpOwotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLkNTX0NJRVhZWjoKLSAgICAgICAgICAgICAgICByZXR1cm4gSUNDX1Byb2ZpbGUuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19DSUVYWVopOwotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLkNTX0xJTkVBUl9SR0I6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIElDQ19Qcm9maWxlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1NfTElORUFSX1JHQik7Ci0gICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuQ1NfUFlDQzoKLSAgICAgICAgICAgICAgICByZXR1cm4gSUNDX1Byb2ZpbGUuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19QWUNDKTsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJTdHViIGNhbm5vdCBwZXJmb3JtIHRoaXMgb3BlcmF0aW9uIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgIH0KLX0KXCBObyBuZXdsaW5lIGF0IGVuZCBvZiBmaWxlCmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvY29sb3IvUHJvZmlsZURhdGFFeGNlcHRpb24uamF2YSBiL2F3dC9qYXZhL2F3dC9jb2xvci9Qcm9maWxlRGF0YUV4Y2VwdGlvbi5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzMzVmMzE0Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9jb2xvci9Qcm9maWxlRGF0YUV4Y2VwdGlvbi5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuY29sb3I7Ci0KLS8qKgotICogVGhlIFByb2ZpbGVEYXRhRXhjZXB0aW9uIGNsYXNzIHJlcHJlc2VudHMgYW4gZXJyb3Igd2hpY2ggb2NjdXJzIHdoaWxlCi0gKiBhY2Nlc3Npbmcgb3IgcHJvY2Vzc2luZyBhbiBJQ0NfUHJvZmlsZSBvYmplY3QuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgUHJvZmlsZURhdGFFeGNlcHRpb24gZXh0ZW5kcyBSdW50aW1lRXhjZXB0aW9uIHsKLSAgICAKLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSA3Mjg2MTQwODg4MjQwMzIyNDk4TDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBwcm9maWxlIGRhdGEgZXhjZXB0aW9uIHdpdGggZGV0YWlsZWQgbWVzc2FnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIGRldGFpbGVkIG1lc3NhZ2Ugb2YgdGhlIGV4Y2VwdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgUHJvZmlsZURhdGFFeGNlcHRpb24oU3RyaW5nIHMpIHsKLSAgICAgICAgc3VwZXIocyk7Ci0gICAgfQotCi19Ci0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9jb2xvci9wYWNrYWdlLmh0bWwgYi9hd3QvamF2YS9hd3QvY29sb3IvcGFja2FnZS5odG1sCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2MDlkOTYzLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9jb2xvci9wYWNrYWdlLmh0bWwKKysrIC9kZXYvbnVsbApAQCAtMSw4ICswLDAgQEAKLTxodG1sPgotICA8Ym9keT4KLSAgICA8cD4KLSAgICAgIFRoaXMgcGFja2FnZSBjb250YWlucyBjbGFzc2VzIHJlcHJlc2VudGluZyBjb2xvciBzcGFjZXMgYW5kIHByb2ZpbGVzIGJhc2VkIG9uIHRoZSBJbnRlcm5hdGlvbmFsIENvbG9yIENvbnNvcnRpdW0gKElDQykgUHJvZmlsZSBGb3JtYXQgU3BlY2lmaWNhdGlvbi4gCi0gICAgPC9wPgotICAgIEBzaW5jZSBBbmRyb2lkIDEuMAotICA8L2JvZHk+Ci08L2h0bWw+CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvQVdURXZlbnRMaXN0ZW5lci5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L0FXVEV2ZW50TGlzdGVuZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzYyOTNiMy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvQVdURXZlbnRMaXN0ZW5lci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotaW1wb3J0IGphdmEuYXd0LkFXVEV2ZW50OwotaW1wb3J0IGphdmEudXRpbC5FdmVudExpc3RlbmVyOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgbm90IHN1cHBvcnRlZCBpbiBBbmRyb2lkIDEuMC4gSXQgaXMgbWVyZWx5IHByb3ZpZGVkIHRvIG1haW50YWluCi0gKiBpbnRlcmZhY2UgY29tcGF0aWJpbGl0eSB3aXRoIGRlc2t0b3AgSmF2YSBpbXBsZW1lbnRhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIEFXVEV2ZW50TGlzdGVuZXIgZXh0ZW5kcyBFdmVudExpc3RlbmVyIHsKLQotICAgIHB1YmxpYyB2b2lkIGV2ZW50RGlzcGF0Y2hlZChBV1RFdmVudCBldmVudCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9BV1RFdmVudExpc3RlbmVyUHJveHkuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9BV1RFdmVudExpc3RlbmVyUHJveHkuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggM2VkYzQxZi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvQVdURXZlbnRMaXN0ZW5lclByb3h5LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1OCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS5hd3QuQVdURXZlbnQ7Ci0KLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lclByb3h5OwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBBV1RFdmVudExpc3RlbmVyUHJveHkgZXh0ZW5kcyBFdmVudExpc3RlbmVyUHJveHkgaW1wbGVtZW50cyBBV1RFdmVudExpc3RlbmVyIHsKLQotICAgIHByaXZhdGUgQVdURXZlbnRMaXN0ZW5lciBsaXN0ZW5lcjsKLSAgICBwcml2YXRlIGxvbmcgZXZlbnRNYXNrOwotCi0gICAgcHVibGljIEFXVEV2ZW50TGlzdGVuZXJQcm94eShsb25nIGV2ZW50TWFzaywgQVdURXZlbnRMaXN0ZW5lciBsaXN0ZW5lcikgewotICAgICAgICBzdXBlcihsaXN0ZW5lcik7Ci0KLSAgICAgICAgLy8gYXd0LjE5Mz1MaXN0ZW5lciBjYW4ndCBiZSB6ZXJvCi0gICAgICAgIGFzc2VydCBsaXN0ZW5lciAhPSBudWxsIDogTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTkzIik7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICB0aGlzLmxpc3RlbmVyID0gbGlzdGVuZXI7Ci0gICAgICAgIHRoaXMuZXZlbnRNYXNrID0gZXZlbnRNYXNrOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGV2ZW50RGlzcGF0Y2hlZChBV1RFdmVudCBldnQpIHsKLSAgICAgICAgbGlzdGVuZXIuZXZlbnREaXNwYXRjaGVkKGV2dCk7Ci0gICAgfQotCi0gICAgcHVibGljIGxvbmcgZ2V0RXZlbnRNYXNrKCkgewotICAgICAgICByZXR1cm4gZXZlbnRNYXNrOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0FjdGlvbkV2ZW50LmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvQWN0aW9uRXZlbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZTg4MmUwZC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvQWN0aW9uRXZlbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDExNCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS5hd3QuQVdURXZlbnQ7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBBY3Rpb25FdmVudCBleHRlbmRzIEFXVEV2ZW50IHsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC03NjcxMDc4Nzk2MjczODMyMTQ5TDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNISUZUX01BU0sgPSAxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ1RSTF9NQVNLID0gMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IE1FVEFfTUFTSyA9IDQ7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBBTFRfTUFTSyA9IDg7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBBQ1RJT05fRklSU1QgPSAxMDAxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQUNUSU9OX0xBU1QgPSAxMDAxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQUNUSU9OX1BFUkZPUk1FRCA9IDEwMDE7Ci0KLSAgICBwcml2YXRlIGxvbmcgd2hlbjsKLSAgICBwcml2YXRlIGludCBtb2RpZmllcnM7Ci0gICAgcHJpdmF0ZSBTdHJpbmcgY29tbWFuZDsKLQotICAgIHB1YmxpYyBBY3Rpb25FdmVudChPYmplY3Qgc291cmNlLCBpbnQgaWQsIFN0cmluZyBjb21tYW5kKSB7Ci0gICAgICAgIHRoaXMoc291cmNlLCBpZCwgY29tbWFuZCwgMCk7Ci0gICAgfQotCi0gICAgcHVibGljIEFjdGlvbkV2ZW50KE9iamVjdCBzb3VyY2UsIGludCBpZCwgU3RyaW5nIGNvbW1hbmQsIGludCBtb2RpZmllcnMpIHsKLSAgICAgICAgdGhpcyhzb3VyY2UsIGlkLCBjb21tYW5kLCAwbCwgbW9kaWZpZXJzKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQWN0aW9uRXZlbnQoT2JqZWN0IHNvdXJjZSwgaW50IGlkLCBTdHJpbmcgY29tbWFuZCwgbG9uZyB3aGVuLCBpbnQgbW9kaWZpZXJzKSB7Ci0gICAgICAgIHN1cGVyKHNvdXJjZSwgaWQpOwotCi0gICAgICAgIHRoaXMuY29tbWFuZCA9IGNvbW1hbmQ7Ci0gICAgICAgIHRoaXMud2hlbiA9IHdoZW47Ci0gICAgICAgIHRoaXMubW9kaWZpZXJzID0gbW9kaWZpZXJzOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0TW9kaWZpZXJzKCkgewotICAgICAgICByZXR1cm4gbW9kaWZpZXJzOwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0QWN0aW9uQ29tbWFuZCgpIHsKLSAgICAgICAgcmV0dXJuIGNvbW1hbmQ7Ci0gICAgfQotCi0gICAgcHVibGljIGxvbmcgZ2V0V2hlbigpIHsKLSAgICAgICAgcmV0dXJuIHdoZW47Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyBwYXJhbVN0cmluZygpIHsKLSAgICAgICAgLyogVGhlIGZvcm1hdCBpcyBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvciAKLSAgICAgICAgICogd2hpY2ggY2FuIGJlIHJldmVhbGVkIGJ5IHRoZSBmb2xsb3dpbmcgY29kZToKLSAgICAgICAgICogCi0gICAgICAgICAqIEFjdGlvbkV2ZW50IGUgPSBuZXcgQWN0aW9uRXZlbnQobmV3IENvbXBvbmVudCgpe30sCi0gICAgICAgICAqICAgICAgIEFjdGlvbkV2ZW50LkFDVElPTl9QRVJGT1JNRUQsICJDb21tYW5kIiwKLSAgICAgICAgICogICAgICAgQWN0aW9uRXZlbnQuU0hJRlRfTUFTS3xBY3Rpb25FdmVudC5DVFJMX01BU0t8Ci0gICAgICAgICAqICAgICAgIEFjdGlvbkV2ZW50Lk1FVEFfTUFTS3xBY3Rpb25FdmVudC5BTFRfTUFTSyk7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihlKTsKLSAgICAgICAgICovCi0KLSAgICAgICAgU3RyaW5nIGlkU3RyaW5nID0gKGlkID09IEFDVElPTl9QRVJGT1JNRUQpID8gCi0gICAgICAgICAgICAgICAgICAgICAgICAgICJBQ1RJT05fUEVSRk9STUVEIiA6ICJ1bmtub3duIHR5cGUiOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgU3RyaW5nIG1vZGlmaWVyc1N0cmluZyA9ICIiOyAvLyROT04tTkxTLTEkCi0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBTSElGVF9NQVNLKSA+IDApIHsKLSAgICAgICAgICAgIG1vZGlmaWVyc1N0cmluZyArPSAiU2hpZnQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBDVFJMX01BU0spID4gMCkgewotICAgICAgICAgICAgbW9kaWZpZXJzU3RyaW5nICs9IG1vZGlmaWVyc1N0cmluZy5sZW5ndGgoKSA9PSAwID8gIkN0cmwiIDogIitDdHJsIjsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBNRVRBX01BU0spID4gMCkgewotICAgICAgICAgICAgbW9kaWZpZXJzU3RyaW5nICs9IG1vZGlmaWVyc1N0cmluZy5sZW5ndGgoKSA9PSAwID8gIk1ldGEiIDogIitNZXRhIjsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBBTFRfTUFTSykgPiAwKSB7Ci0gICAgICAgICAgICBtb2RpZmllcnNTdHJpbmcgKz0gbW9kaWZpZXJzU3RyaW5nLmxlbmd0aCgpID09IDAgPyAiQWx0IiA6ICIrQWx0IjsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gKGlkU3RyaW5nICsgIixjbWQ9IiArIGNvbW1hbmQgKyAiLHdoZW49IiArIHdoZW4gKyAgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgIixtb2RpZmllcnM9IiArIG1vZGlmaWVyc1N0cmluZyk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9BY3Rpb25MaXN0ZW5lci5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L0FjdGlvbkxpc3RlbmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGE2ZWVlN2EuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L0FjdGlvbkxpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgQWN0aW9uTGlzdGVuZXIgZXh0ZW5kcyBFdmVudExpc3RlbmVyIHsKLQotICAgIHB1YmxpYyB2b2lkIGFjdGlvblBlcmZvcm1lZChBY3Rpb25FdmVudCBlKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0FkanVzdG1lbnRFdmVudC5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L0FkanVzdG1lbnRFdmVudC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiZTJkNmM0Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9BZGp1c3RtZW50RXZlbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEyMyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS5hd3QuQVdURXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuQWRqdXN0YWJsZTsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEFkanVzdG1lbnRFdmVudCBleHRlbmRzIEFXVEV2ZW50IHsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IDU3MDAyOTA2NDUyMDUyNzk5MjFMOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQURKVVNUTUVOVF9GSVJTVCA9IDYwMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEFESlVTVE1FTlRfTEFTVCA9IDYwMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEFESlVTVE1FTlRfVkFMVUVfQ0hBTkdFRCA9IDYwMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFVOSVRfSU5DUkVNRU5UID0gMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFVOSVRfREVDUkVNRU5UID0gMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEJMT0NLX0RFQ1JFTUVOVCA9IDM7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBCTE9DS19JTkNSRU1FTlQgPSA0OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFJBQ0sgPSA1OwotCi0gICAgcHJpdmF0ZSBpbnQgdHlwZTsKLSAgICBwcml2YXRlIGludCB2YWx1ZTsKLSAgICBwcml2YXRlIGJvb2xlYW4gaXNBZGp1c3Rpbmc7Ci0KLSAgICBwdWJsaWMgQWRqdXN0bWVudEV2ZW50KEFkanVzdGFibGUgc291cmNlLCBpbnQgaWQsIGludCB0eXBlLCBpbnQgdmFsdWUpIHsKLSAgICAgICAgdGhpcyhzb3VyY2UsIGlkLCB0eXBlLCB2YWx1ZSwgZmFsc2UpOwotICAgIH0KLQotICAgIHB1YmxpYyBBZGp1c3RtZW50RXZlbnQoQWRqdXN0YWJsZSBzb3VyY2UsIGludCBpZCwgaW50IHR5cGUsIGludCB2YWx1ZSwgCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBib29sZWFuIGlzQWRqdXN0aW5nKSB7Ci0gICAgICAgIHN1cGVyKHNvdXJjZSwgaWQpOwotICAgICAgICB0aGlzLnR5cGUgPSB0eXBlOwotICAgICAgICB0aGlzLnZhbHVlID0gdmFsdWU7Ci0gICAgICAgIHRoaXMuaXNBZGp1c3RpbmcgPSBpc0FkanVzdGluZzsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFZhbHVlKCkgewotICAgICAgICByZXR1cm4gdmFsdWU7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRBZGp1c3RtZW50VHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIHR5cGU7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gZ2V0VmFsdWVJc0FkanVzdGluZygpIHsKLSAgICAgICAgcmV0dXJuIGlzQWRqdXN0aW5nOwotICAgIH0KLQotICAgIHB1YmxpYyBBZGp1c3RhYmxlIGdldEFkanVzdGFibGUoKSB7Ci0gICAgICAgIHJldHVybiAoQWRqdXN0YWJsZSkgc291cmNlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgcGFyYW1TdHJpbmcoKSB7Ci0gICAgICAgIC8qIFRoZSBmb3JtYXQgaXMgYmFzZWQgb24gMS41IHJlbGVhc2UgYmVoYXZpb3IgCi0gICAgICAgICAqIHdoaWNoIGNhbiBiZSByZXZlYWxlZCBieSB0aGUgZm9sbG93aW5nIGNvZGU6Ci0gICAgICAgICAqIAotICAgICAgICAgKiBBZGp1c3RtZW50RXZlbnQgZSA9IG5ldyBBZGp1c3RtZW50RXZlbnQobmV3IFNjcm9sbGJhcigpLCAKLSAgICAgICAgICogICAgICAgQWRqdXN0bWVudEV2ZW50LkFESlVTVE1FTlRfVkFMVUVfQ0hBTkdFRCwgCi0gICAgICAgICAqICAgICAgIEFkanVzdG1lbnRFdmVudC5VTklUX0lOQ1JFTUVOVCwgMSk7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihlKTsKLSAgICAgICAgICovCi0KLSAgICAgICAgU3RyaW5nIGlkU3RyaW5nID0gKGlkID09IEFESlVTVE1FTlRfVkFMVUVfQ0hBTkdFRCA/Ci0gICAgICAgICAgICAgICAgIkFESlVTVE1FTlRfVkFMVUVfQ0hBTkdFRCIgOiAidW5rbm93biB0eXBlIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICBTdHJpbmcgYWRqVHlwZSA9IG51bGw7Ci0KLSAgICAgICAgc3dpdGNoICh0eXBlKSB7Ci0gICAgICAgIGNhc2UgVU5JVF9JTkNSRU1FTlQ6Ci0gICAgICAgICAgICBhZGpUeXBlID0gIlVOSVRfSU5DUkVNRU5UIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgVU5JVF9ERUNSRU1FTlQ6Ci0gICAgICAgICAgICBhZGpUeXBlID0gIlVOSVRfREVDUkVNRU5UIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgQkxPQ0tfSU5DUkVNRU5UOgotICAgICAgICAgICAgYWRqVHlwZSA9ICJCTE9DS19JTkNSRU1FTlQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBCTE9DS19ERUNSRU1FTlQ6Ci0gICAgICAgICAgICBhZGpUeXBlID0gIkJMT0NLX0RFQ1JFTUVOVCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFRSQUNLOgotICAgICAgICAgICAgYWRqVHlwZSA9ICJUUkFDSyI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgYWRqVHlwZSA9ICJ1bmtub3duIHR5cGUiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gKGlkU3RyaW5nICsgIixhZGpUeXBlPSIgKyBhZGpUeXBlICsgIix2YWx1ZT0iICsgdmFsdWUgKyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgICAgICAgICAiLGlzQWRqdXN0aW5nPSIgKyBpc0FkanVzdGluZyk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9BZGp1c3RtZW50TGlzdGVuZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9BZGp1c3RtZW50TGlzdGVuZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNWY2YTcyNC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvQWRqdXN0bWVudExpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgQWRqdXN0bWVudExpc3RlbmVyIGV4dGVuZHMgRXZlbnRMaXN0ZW5lciB7Ci0KLSAgICBwdWJsaWMgdm9pZCBhZGp1c3RtZW50VmFsdWVDaGFuZ2VkKEFkanVzdG1lbnRFdmVudCBlKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0NvbXBvbmVudEFkYXB0ZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9Db21wb25lbnRBZGFwdGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM0MjIzNWYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L0NvbXBvbmVudEFkYXB0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBDb21wb25lbnRBZGFwdGVyIGltcGxlbWVudHMgQ29tcG9uZW50TGlzdGVuZXIgewotCi0gICAgcHVibGljIENvbXBvbmVudEFkYXB0ZXIoKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgY29tcG9uZW50SGlkZGVuKENvbXBvbmVudEV2ZW50IGUpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBjb21wb25lbnRNb3ZlZChDb21wb25lbnRFdmVudCBlKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgY29tcG9uZW50UmVzaXplZChDb21wb25lbnRFdmVudCBlKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgY29tcG9uZW50U2hvd24oQ29tcG9uZW50RXZlbnQgZSkgewotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0NvbXBvbmVudEV2ZW50LmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvQ29tcG9uZW50RXZlbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzYwZDNhYi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvQ29tcG9uZW50RXZlbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDg4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5BV1RFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5Db21wb25lbnQ7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBDb21wb25lbnRFdmVudCBleHRlbmRzIEFXVEV2ZW50IHsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IDgxMDE0MDY4MjM5MDI5OTI5NjVMOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ09NUE9ORU5UX0ZJUlNUID0gMTAwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ09NUE9ORU5UX0xBU1QgPSAxMDM7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDT01QT05FTlRfTU9WRUQgPSAxMDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDT01QT05FTlRfUkVTSVpFRCA9IDEwMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENPTVBPTkVOVF9TSE9XTiA9IDEwMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENPTVBPTkVOVF9ISURERU4gPSAxMDM7Ci0KLSAgICBwdWJsaWMgQ29tcG9uZW50RXZlbnQoQ29tcG9uZW50IHNvdXJjZSwgaW50IGlkKSB7Ci0gICAgICAgIHN1cGVyKHNvdXJjZSwgaWQpOwotICAgIH0KLQotICAgIHB1YmxpYyBDb21wb25lbnQgZ2V0Q29tcG9uZW50KCkgewotICAgICAgICByZXR1cm4gKENvbXBvbmVudCkgc291cmNlOwotICAgIH0KLSAgICAKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHBhcmFtU3RyaW5nKCkgewotICAgICAgICAvKiBUaGUgZm9ybWF0IGlzIGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW9yIAotICAgICAgICAgKiB3aGljaCBjYW4gYmUgcmV2ZWFsZWQgYnkgdGhlIGZvbGxvd2luZyBjb2RlOgotICAgICAgICAgKiAKLSAgICAgICAgICogQ29tcG9uZW50RXZlbnQgZSA9IG5ldyBDb21wb25lbnRFdmVudChuZXcgQnV0dG9uKCJCdXR0b24iKSwgCi0gICAgICAgICAqICAgICAgICAgIENvbXBvbmVudEV2ZW50LkNPTVBPTkVOVF9TSE9XTik7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihlKTsKLSAgICAgICAgICovCi0KLSAgICAgICAgU3RyaW5nIGlkU3RyaW5nID0gbnVsbDsKLSAgICAgICAgQ29tcG9uZW50IGMgPSBnZXRDb21wb25lbnQoKTsKLQotICAgICAgICBzd2l0Y2ggKGlkKSB7Ci0gICAgICAgIGNhc2UgQ09NUE9ORU5UX01PVkVEOgotICAgICAgICAgICAgaWRTdHJpbmcgPSAiQ09NUE9ORU5UX01PVkVEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgQ09NUE9ORU5UX1JFU0laRUQ6Ci0gICAgICAgICAgICBpZFN0cmluZyA9ICJDT01QT05FTlRfUkVTSVpFRCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIENPTVBPTkVOVF9TSE9XTjoKLSAgICAgICAgICAgIHJldHVybiAiQ09NUE9ORU5UX1NIT1dOIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICBjYXNlIENPTVBPTkVOVF9ISURERU46Ci0gICAgICAgICAgICByZXR1cm4gIkNPTVBPTkVOVF9ISURERU4iOyAvLyROT04tTkxTLTEkCi0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gInVua25vd24gdHlwZSI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiAoaWRTdHJpbmcgKyAiICgiICsgYy5nZXRYKCkgKyAiLCIgKyBjLmdldFkoKSArICAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgICAgICAgICAiICIgKyBjLmdldFdpZHRoKCkrICJ4IiArIGMuZ2V0SGVpZ2h0KCkgKyAiKSIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0NvbXBvbmVudExpc3RlbmVyLmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvQ29tcG9uZW50TGlzdGVuZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTVhZGJhMi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvQ29tcG9uZW50TGlzdGVuZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBDb21wb25lbnRMaXN0ZW5lciBleHRlbmRzIEV2ZW50TGlzdGVuZXIgewotCi0gICAgcHVibGljIHZvaWQgY29tcG9uZW50SGlkZGVuKENvbXBvbmVudEV2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQgY29tcG9uZW50TW92ZWQoQ29tcG9uZW50RXZlbnQgZSk7Ci0KLSAgICBwdWJsaWMgdm9pZCBjb21wb25lbnRSZXNpemVkKENvbXBvbmVudEV2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQgY29tcG9uZW50U2hvd24oQ29tcG9uZW50RXZlbnQgZSk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9Db250YWluZXJBZGFwdGVyLmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvQ29udGFpbmVyQWRhcHRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0NDk4M2M3Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9Db250YWluZXJBZGFwdGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0MCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgbm90IHN1cHBvcnRlZCBpbiBBbmRyb2lkIDEuMC4gSXQgaXMgbWVyZWx5IHByb3ZpZGVkIHRvIG1haW50YWluCi0gKiBpbnRlcmZhY2UgY29tcGF0aWJpbGl0eSB3aXRoIGRlc2t0b3AgSmF2YSBpbXBsZW1lbnRhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgQ29udGFpbmVyQWRhcHRlciBpbXBsZW1lbnRzIENvbnRhaW5lckxpc3RlbmVyIHsKLQotICAgIHB1YmxpYyBDb250YWluZXJBZGFwdGVyKCkgewotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGNvbXBvbmVudEFkZGVkKENvbnRhaW5lckV2ZW50IGUpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBjb21wb25lbnRSZW1vdmVkKENvbnRhaW5lckV2ZW50IGUpIHsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9Db250YWluZXJFdmVudC5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L0NvbnRhaW5lckV2ZW50LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDM3MmM5ZTQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L0NvbnRhaW5lckV2ZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS5hd3QuQ29tcG9uZW50OwotLy8/Pz9BV1Q6IGltcG9ydCBqYXZhLmF3dC5Db250YWluZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBDb250YWluZXJFdmVudCBleHRlbmRzIENvbXBvbmVudEV2ZW50IHsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC00MTE0OTQyMjUwNTM5NzcyMDQxTDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENPTlRBSU5FUl9GSVJTVCA9IDMwMDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENPTlRBSU5FUl9MQVNUID0gMzAxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ09NUE9ORU5UX0FEREVEID0gMzAwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ09NUE9ORU5UX1JFTU9WRUQgPSAzMDE7Ci0KLSAgICBwcml2YXRlIENvbXBvbmVudCBjaGlsZDsKLQotICAgIHB1YmxpYyBDb250YWluZXJFdmVudChDb21wb25lbnQgc3JjLCBpbnQgaWQsIENvbXBvbmVudCBjaGlsZCkgewotICAgICAgICBzdXBlcihzcmMsIGlkKTsKLSAgICAgICAgdGhpcy5jaGlsZCA9IGNoaWxkOwotICAgIH0KLQotICAgIHB1YmxpYyBDb21wb25lbnQgZ2V0Q2hpbGQoKSB7Ci0gICAgICAgIHJldHVybiBjaGlsZDsKLSAgICB9Ci0KLSAgICAvLz8/P0FXVAotICAgIC8qCi0gICAgcHVibGljIENvbnRhaW5lciBnZXRDb250YWluZXIoKSB7Ci0gICAgICAgIHJldHVybiAoQ29udGFpbmVyKSBzb3VyY2U7Ci0gICAgfQotICAgICovCi0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHBhcmFtU3RyaW5nKCkgewotICAgICAgICAvKiBUaGUgZm9ybWF0IGlzIGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW9yIAotICAgICAgICAgKiB3aGljaCBjYW4gYmUgcmV2ZWFsZWQgYnkgdGhlIGZvbGxvd2luZyBjb2RlOgotICAgICAgICAgKiAKLSAgICAgICAgICogQ29udGFpbmVyRXZlbnQgZSA9IG5ldyBDb250YWluZXJFdmVudChuZXcgUGFuZWwoKSwKLSAgICAgICAgICogICAgICAgICAgQ29udGFpbmVyRXZlbnQuQ09NUE9ORU5UX0FEREVELAotICAgICAgICAgKiAgICAgICAgICBuZXcgQnV0dG9uKCJCdXR0b24iKSk7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihlKTsKLSAgICAgICAgICovCi0KLSAgICAgICAgU3RyaW5nIGlkU3RyaW5nID0gbnVsbDsKLQotICAgICAgICBzd2l0Y2ggKGlkKSB7Ci0gICAgICAgIGNhc2UgQ09NUE9ORU5UX0FEREVEOgotICAgICAgICAgICAgaWRTdHJpbmcgPSAiQ09NUE9ORU5UX0FEREVEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgQ09NUE9ORU5UX1JFTU9WRUQ6Ci0gICAgICAgICAgICBpZFN0cmluZyA9ICJDT01QT05FTlRfUkVNT1ZFRCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgaWRTdHJpbmcgPSAidW5rbm93biB0eXBlIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIChpZFN0cmluZyArICIsY2hpbGQ9IiArIGNoaWxkLmdldE5hbWUoKSk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9Db250YWluZXJMaXN0ZW5lci5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L0NvbnRhaW5lckxpc3RlbmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDUxNzg1OWUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L0NvbnRhaW5lckxpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgQ29udGFpbmVyTGlzdGVuZXIgZXh0ZW5kcyBFdmVudExpc3RlbmVyIHsKLQotICAgIHB1YmxpYyB2b2lkIGNvbXBvbmVudEFkZGVkKENvbnRhaW5lckV2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQgY29tcG9uZW50UmVtb3ZlZChDb250YWluZXJFdmVudCBlKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0ZvY3VzQWRhcHRlci5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L0ZvY3VzQWRhcHRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzYTNlMzdmLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9Gb2N1c0FkYXB0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBGb2N1c0FkYXB0ZXIgaW1wbGVtZW50cyBGb2N1c0xpc3RlbmVyIHsKLQotICAgIHB1YmxpYyBGb2N1c0FkYXB0ZXIoKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZm9jdXNHYWluZWQoRm9jdXNFdmVudCBlKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZm9jdXNMb3N0KEZvY3VzRXZlbnQgZSkgewotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0ZvY3VzRXZlbnQuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9Gb2N1c0V2ZW50LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDRhMTg2ODkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L0ZvY3VzRXZlbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDk2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5Db21wb25lbnQ7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBGb2N1c0V2ZW50IGV4dGVuZHMgQ29tcG9uZW50RXZlbnQgewotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gNTIzNzUzNzg2NDU3NDE2Mzk2TDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEZPQ1VTX0ZJUlNUID0gMTAwNDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEZPQ1VTX0xBU1QgPSAxMDA1OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRk9DVVNfR0FJTkVEID0gMTAwNDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEZPQ1VTX0xPU1QgPSAxMDA1OwotCi0gICAgcHJpdmF0ZSBib29sZWFuIHRlbXBvcmFyeTsKLSAgICBwcml2YXRlIENvbXBvbmVudCBvcHBvc2l0ZTsKLQotICAgIHB1YmxpYyBGb2N1c0V2ZW50KENvbXBvbmVudCBzb3VyY2UsIGludCBpZCkgewotICAgICAgICB0aGlzKHNvdXJjZSwgaWQsIGZhbHNlKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgRm9jdXNFdmVudChDb21wb25lbnQgc291cmNlLCBpbnQgaWQsIGJvb2xlYW4gdGVtcG9yYXJ5KSB7Ci0gICAgICAgIHRoaXMoc291cmNlLCBpZCwgdGVtcG9yYXJ5LCBudWxsKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgRm9jdXNFdmVudChDb21wb25lbnQgc291cmNlLCBpbnQgaWQsIGJvb2xlYW4gdGVtcG9yYXJ5LCBDb21wb25lbnQgb3Bwb3NpdGUpIHsKLSAgICAgICAgc3VwZXIoc291cmNlLCBpZCk7Ci0gICAgICAgIHRoaXMudGVtcG9yYXJ5ID0gdGVtcG9yYXJ5OwotICAgICAgICB0aGlzLm9wcG9zaXRlID0gb3Bwb3NpdGU7Ci0gICAgfQotCi0gICAgcHVibGljIENvbXBvbmVudCBnZXRPcHBvc2l0ZUNvbXBvbmVudCgpIHsKLSAgICAgICAgcmV0dXJuIG9wcG9zaXRlOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGlzVGVtcG9yYXJ5KCkgewotICAgICAgICByZXR1cm4gdGVtcG9yYXJ5OwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgcGFyYW1TdHJpbmcoKSB7Ci0gICAgICAgIC8qIFRoZSBmb3JtYXQgaXMgYmFzZWQgb24gMS41IHJlbGVhc2UgYmVoYXZpb3IgCi0gICAgICAgICAqIHdoaWNoIGNhbiBiZSByZXZlYWxlZCBieSB0aGUgZm9sbG93aW5nIGNvZGU6Ci0gICAgICAgICAqIAotICAgICAgICAgKiBGb2N1c0V2ZW50IGUgPSBuZXcgRm9jdXNFdmVudChuZXcgQnV0dG9uKCJCdXR0b24wIiksCi0gICAgICAgICAqICAgICAgIEZvY3VzRXZlbnQuRk9DVVNfR0FJTkVELCBmYWxzZSwgbmV3IEJ1dHRvbigiQnV0dG9uMSIpKTsKLSAgICAgICAgICogU3lzdGVtLm91dC5wcmludGxuKGUpOwotICAgICAgICAgKi8KLQotICAgICAgICBTdHJpbmcgaWRTdHJpbmcgPSBudWxsOwotCi0gICAgICAgIHN3aXRjaCAoaWQpIHsKLSAgICAgICAgY2FzZSBGT0NVU19HQUlORUQ6Ci0gICAgICAgICAgICBpZFN0cmluZyA9ICJGT0NVU19HQUlORUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBGT0NVU19MT1NUOgotICAgICAgICAgICAgaWRTdHJpbmcgPSAiRk9DVVNfTE9TVCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgaWRTdHJpbmcgPSAidW5rbm93biB0eXBlIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIChpZFN0cmluZyArCi0gICAgICAgICAgICAgICAgKHRlbXBvcmFyeSA/ICIsdGVtcG9yYXJ5IiA6ICIscGVybWFuZW50IikgKyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgICAgICAgICAiLG9wcG9zaXRlPSIgKyBvcHBvc2l0ZSk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9Gb2N1c0xpc3RlbmVyLmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvRm9jdXNMaXN0ZW5lci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2YmJiZDAwLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9Gb2N1c0xpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgRm9jdXNMaXN0ZW5lciBleHRlbmRzIEV2ZW50TGlzdGVuZXIgewotCi0gICAgcHVibGljIHZvaWQgZm9jdXNHYWluZWQoRm9jdXNFdmVudCBlKTsKLQotICAgIHB1YmxpYyB2b2lkIGZvY3VzTG9zdChGb2N1c0V2ZW50IGUpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvSGllcmFyY2h5Qm91bmRzQWRhcHRlci5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L0hpZXJhcmNoeUJvdW5kc0FkYXB0ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYmJmZThmZi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvSGllcmFyY2h5Qm91bmRzQWRhcHRlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDAgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEhpZXJhcmNoeUJvdW5kc0FkYXB0ZXIgaW1wbGVtZW50cyBIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lciB7Ci0KLSAgICBwdWJsaWMgSGllcmFyY2h5Qm91bmRzQWRhcHRlcigpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBhbmNlc3Rvck1vdmVkKEhpZXJhcmNoeUV2ZW50IGUpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBhbmNlc3RvclJlc2l6ZWQoSGllcmFyY2h5RXZlbnQgZSkgewotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0hpZXJhcmNoeUJvdW5kc0xpc3RlbmVyLmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvSGllcmFyY2h5Qm91bmRzTGlzdGVuZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggM2U4ZjJlNy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvSGllcmFyY2h5Qm91bmRzTGlzdGVuZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDM3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBIaWVyYXJjaHlCb3VuZHNMaXN0ZW5lciBleHRlbmRzIEV2ZW50TGlzdGVuZXIgewotCi0gICAgcHVibGljIHZvaWQgYW5jZXN0b3JNb3ZlZChIaWVyYXJjaHlFdmVudCBlKTsKLQotICAgIHB1YmxpYyB2b2lkIGFuY2VzdG9yUmVzaXplZChIaWVyYXJjaHlFdmVudCBlKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0hpZXJhcmNoeUV2ZW50LmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvSGllcmFyY2h5RXZlbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzFkMjJmNC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvSGllcmFyY2h5RXZlbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE1NCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS5hd3QuQVdURXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuQ29tcG9uZW50OwotLy8/Pz9BV1Q6IGltcG9ydCBqYXZhLmF3dC5Db250YWluZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBIaWVyYXJjaHlFdmVudCBleHRlbmRzIEFXVEV2ZW50IHsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC01MzM3NTc2OTcwMDM4MDQzOTkwTDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEhJRVJBUkNIWV9GSVJTVCA9IDE0MDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBISUVSQVJDSFlfQ0hBTkdFRCA9IDE0MDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBBTkNFU1RPUl9NT1ZFRCA9IDE0MDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBBTkNFU1RPUl9SRVNJWkVEID0gMTQwMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEhJRVJBUkNIWV9MQVNUID0gMTQwMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFBBUkVOVF9DSEFOR0VEID0gMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IERJU1BMQVlBQklMSVRZX0NIQU5HRUQgPSAyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU0hPV0lOR19DSEFOR0VEID0gNDsKLQotICAgIC8vPz8/QVdUOiBwcml2YXRlIENvbnRhaW5lciBjaGFuZ2VkUGFyZW50OwotICAgIHByaXZhdGUgQ29tcG9uZW50IGNoYW5nZWQ7Ci0gICAgcHJpdmF0ZSBsb25nIGNoYW5nZUZsYWc7Ci0KLSAgICAvLz8/P0FXVAotICAgIC8qCi0gICAgcHVibGljIEhpZXJhcmNoeUV2ZW50KENvbXBvbmVudCBzb3VyY2UsIGludCBpZCwgQ29tcG9uZW50IGNoYW5nZWQsIAotICAgICAgICAgICAgICAgICAgICAgICAgICBDb250YWluZXIgY2hhbmdlZFBhcmVudCkgewotICAgICAgICB0aGlzKHNvdXJjZSwgaWQsIGNoYW5nZWQsIGNoYW5nZWRQYXJlbnQsIDBsKTsKLSAgICB9Ci0gICAgKi8KLQotICAgIC8vPz8/QVdUCi0gICAgLyoKLSAgICBwdWJsaWMgSGllcmFyY2h5RXZlbnQoQ29tcG9uZW50IHNvdXJjZSwgaW50IGlkLCBDb21wb25lbnQgY2hhbmdlZCwKLSAgICAgICAgICAgIENvbnRhaW5lciBjaGFuZ2VkUGFyZW50LCBsb25nIGNoYW5nZUZsYWdzKSB7Ci0gICAgICAgIHN1cGVyKHNvdXJjZSwgaWQpOwotCi0gICAgICAgIHRoaXMuY2hhbmdlZCA9IGNoYW5nZWQ7Ci0gICAgICAgIHRoaXMuY2hhbmdlZFBhcmVudCA9IGNoYW5nZWRQYXJlbnQ7Ci0gICAgICAgIHRoaXMuY2hhbmdlRmxhZyA9IGNoYW5nZUZsYWdzOwotICAgIH0KLSAgICAqLwotICAgIC8vPz8/QVdUOiBGYWtlIGNvbnN0cnVjdG9yLCBzaG91bGQgYmUgYXMgYWJvdmUuCi0gICAgcHVibGljIEhpZXJhcmNoeUV2ZW50KENvbXBvbmVudCBzb3VyY2UsIGludCBpZCwgQ29tcG9uZW50IGNoYW5nZWQsCi0gICAgICAgICAgICBPYmplY3QgY2hhbmdlZFBhcmVudCwgbG9uZyBjaGFuZ2VGbGFncykgewotICAgICAgICBzdXBlcihzb3VyY2UsIGlkKTsKLQotLy8gICAgICAgIHRoaXMuY2hhbmdlZCA9IGNoYW5nZWQ7Ci0vLyAgICAgICAgdGhpcy5jaGFuZ2VkUGFyZW50ID0gY2hhbmdlZFBhcmVudDsKLS8vICAgICAgICB0aGlzLmNoYW5nZUZsYWcgPSBjaGFuZ2VGbGFnczsKLSAgICB9Ci0gICAgCi0gICAgcHVibGljIENvbXBvbmVudCBnZXRDb21wb25lbnQoKSB7Ci0gICAgICAgIHJldHVybiAoQ29tcG9uZW50KSBzb3VyY2U7Ci0gICAgfQotCi0gICAgcHVibGljIGxvbmcgZ2V0Q2hhbmdlRmxhZ3MoKSB7Ci0gICAgICAgIHJldHVybiBjaGFuZ2VGbGFnOwotICAgIH0KLQotICAgIHB1YmxpYyBDb21wb25lbnQgZ2V0Q2hhbmdlZCgpIHsKLSAgICAgICAgcmV0dXJuIGNoYW5nZWQ7Ci0gICAgfQotCi0gICAgLy8/Pz9BV1QKLSAgICAvKgotICAgIHB1YmxpYyBDb250YWluZXIgZ2V0Q2hhbmdlZFBhcmVudCgpIHsKLSAgICAgICAgcmV0dXJuIGNoYW5nZWRQYXJlbnQ7Ci0KLSAgICB9Ci0gICAgKi8KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgcGFyYW1TdHJpbmcoKSB7Ci0gICAgICAgIC8qIFRoZSBmb3JtYXQgaXMgYmFzZWQgb24gMS41IHJlbGVhc2UgYmVoYXZpb3IgCi0gICAgICAgICAqIHdoaWNoIGNhbiBiZSByZXZlYWxlZCBieSB0aGUgZm9sbG93aW5nIGNvZGU6Ci0gICAgICAgICAqIAotICAgICAgICAgKiBIaWVyYXJjaHlFdmVudCBlID0gbmV3IEhpZXJhcmNoeUV2ZW50KG5ldyBCdXR0b24oIkJ1dHRvbiIpLAotICAgICAgICAgKiAgICAgICAgICBIaWVyYXJjaHlFdmVudC5ISUVSQVJDSFlfQ0hBTkdFRCwKLSAgICAgICAgICogICAgICAgICAgbmV3IFBhbmVsKCksIG5ldyBDb250YWluZXIoKSk7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihlKTsKLSAgICAgICAgICovCi0gICAgICAgIFN0cmluZyBwYXJhbVN0cmluZyA9IG51bGw7Ci0KLSAgICAgICAgc3dpdGNoIChpZCkgewotICAgICAgICBjYXNlIEhJRVJBUkNIWV9DSEFOR0VEOgotICAgICAgICAgICAgcGFyYW1TdHJpbmcgPSAiSElFUkFSQ0hZX0NIQU5HRUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBBTkNFU1RPUl9NT1ZFRDoKLSAgICAgICAgICAgIHBhcmFtU3RyaW5nID0gIkFOQ0VTVE9SX01PVkVEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgQU5DRVNUT1JfUkVTSVpFRDoKLSAgICAgICAgICAgIHBhcmFtU3RyaW5nID0gIkFOQ0VTVE9SX1JFU0laRUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHBhcmFtU3RyaW5nID0gInVua25vd24gdHlwZSI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHBhcmFtU3RyaW5nICs9ICIgKCI7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICBpZiAoaWQgPT0gSElFUkFSQ0hZX0NIQU5HRUQpIHsKLSAgICAgICAgICAgIGlmICgoY2hhbmdlRmxhZyAmIFBBUkVOVF9DSEFOR0VEKSA+IDApIHsKLSAgICAgICAgICAgICAgICBwYXJhbVN0cmluZyArPSAiUEFSRU5UX0NIQU5HRUQsIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKChjaGFuZ2VGbGFnICYgRElTUExBWUFCSUxJVFlfQ0hBTkdFRCkgPiAwKSB7Ci0gICAgICAgICAgICAgICAgcGFyYW1TdHJpbmcgKz0gIkRJU1BMQVlBQklMSVRZX0NIQU5HRUQsIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKChjaGFuZ2VGbGFnICYgU0hPV0lOR19DSEFOR0VEKSA+IDApIHsKLSAgICAgICAgICAgICAgICBwYXJhbVN0cmluZyArPSAiU0hPV0lOR19DSEFOR0VELCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vPz8/QVdUCi0gICAgICAgIC8qCi0gICAgICAgIHJldHVybiBwYXJhbVN0cmluZyArICJjaGFuZ2VkPSIgKyBjaGFuZ2VkICsgIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAiLGNoYW5nZWRQYXJlbnQ9IiArIGNoYW5nZWRQYXJlbnQgKyAiKSI7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAqLwotICAgICAgICByZXR1cm4gcGFyYW1TdHJpbmc7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvSGllcmFyY2h5TGlzdGVuZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9IaWVyYXJjaHlMaXN0ZW5lci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmZjNkOWJjLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9IaWVyYXJjaHlMaXN0ZW5lci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotaW1wb3J0IGphdmEudXRpbC5FdmVudExpc3RlbmVyOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgbm90IHN1cHBvcnRlZCBpbiBBbmRyb2lkIDEuMC4gSXQgaXMgbWVyZWx5IHByb3ZpZGVkIHRvIG1haW50YWluCi0gKiBpbnRlcmZhY2UgY29tcGF0aWJpbGl0eSB3aXRoIGRlc2t0b3AgSmF2YSBpbXBsZW1lbnRhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIEhpZXJhcmNoeUxpc3RlbmVyIGV4dGVuZHMgRXZlbnRMaXN0ZW5lciB7Ci0KLSAgICBwdWJsaWMgdm9pZCBoaWVyYXJjaHlDaGFuZ2VkKEhpZXJhcmNoeUV2ZW50IGUpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvSW5wdXRFdmVudC5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L0lucHV0RXZlbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzQzYjdhMy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvSW5wdXRFdmVudC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTkwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5Db21wb25lbnQ7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBJbnB1dEV2ZW50IGV4dGVuZHMgQ29tcG9uZW50RXZlbnQgewotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gLTI0ODI1MjU5ODE2OTgzMDk3ODZMOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU0hJRlRfTUFTSyA9IDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDVFJMX01BU0sgPSAyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTUVUQV9NQVNLID0gNDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEFMVF9NQVNLID0gODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEFMVF9HUkFQSF9NQVNLID0gMzI7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBCVVRUT04xX01BU0sgPSAxNjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEJVVFRPTjJfTUFTSyA9IDg7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBCVVRUT04zX01BU0sgPSA0OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU0hJRlRfRE9XTl9NQVNLID0gNjQ7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDVFJMX0RPV05fTUFTSyA9IDEyODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IE1FVEFfRE9XTl9NQVNLID0gMjU2OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQUxUX0RPV05fTUFTSyA9IDUxMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEJVVFRPTjFfRE9XTl9NQVNLID0gMTAyNDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEJVVFRPTjJfRE9XTl9NQVNLID0gMjA0ODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEJVVFRPTjNfRE9XTl9NQVNLID0gNDA5NjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEFMVF9HUkFQSF9ET1dOX01BU0sgPSA4MTkyOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IERPV05fTUFTS1MgPSBTSElGVF9ET1dOX01BU0sgfCBDVFJMX0RPV05fTUFTSyB8Ci0gICAgICAgICAgICBNRVRBX0RPV05fTUFTSyB8IEFMVF9ET1dOX01BU0sgfCBCVVRUT04xX0RPV05fTUFTSyB8Ci0gICAgICAgICAgICBCVVRUT04yX0RPV05fTUFTSyB8IEJVVFRPTjNfRE9XTl9NQVNLIHwgQUxUX0dSQVBIX0RPV05fTUFTSzsKLQotICAgIHByaXZhdGUgbG9uZyB3aGVuOwotICAgIHByaXZhdGUgaW50IG1vZGlmaWVyc0V4OwotCi0gICAgcHVibGljIHN0YXRpYyBTdHJpbmcgZ2V0TW9kaWZpZXJzRXhUZXh0KGludCBtb2RpZmllcnMvKkV4Ki8pIHsKLSAgICAgICAgcmV0dXJuIE1vdXNlRXZlbnQuYWRkTW91c2VNb2RpZmllcnNFeFRleHQoCi0gICAgICAgICAgICAgICAgS2V5RXZlbnQuZ2V0S2V5TW9kaWZpZXJzRXhUZXh0KG1vZGlmaWVycyksIG1vZGlmaWVycyk7Ci0gICAgfQotCi0gICAgc3RhdGljIGludCBleHRyYWN0RXhGbGFncyhpbnQgbW9kaWZpZXJzKSB7Ci0gICAgICAgIGludCBleEZsYWdzID0gbW9kaWZpZXJzICYgRE9XTl9NQVNLUzsKLQotICAgICAgICBpZiAoKG1vZGlmaWVycyAmIFNISUZUX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIGV4RmxhZ3MgfD0gU0hJRlRfRE9XTl9NQVNLOwotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzICYgQ1RSTF9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICBleEZsYWdzIHw9IENUUkxfRE9XTl9NQVNLOwotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzICYgTUVUQV9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICBleEZsYWdzIHw9IE1FVEFfRE9XTl9NQVNLOwotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzICYgQUxUX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIGV4RmxhZ3MgfD0gQUxUX0RPV05fTUFTSzsKLSAgICAgICAgfQotICAgICAgICBpZiAoKG1vZGlmaWVycyAmIEFMVF9HUkFQSF9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICBleEZsYWdzIHw9IEFMVF9HUkFQSF9ET1dOX01BU0s7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBCVVRUT04xX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIGV4RmxhZ3MgfD0gQlVUVE9OMV9ET1dOX01BU0s7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBCVVRUT04yX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIGV4RmxhZ3MgfD0gQlVUVE9OMl9ET1dOX01BU0s7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBCVVRUT04zX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIGV4RmxhZ3MgfD0gQlVUVE9OM19ET1dOX01BU0s7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZXhGbGFnczsKLSAgICB9Ci0KLSAgICBJbnB1dEV2ZW50KENvbXBvbmVudCBzb3VyY2UsIGludCBpZCwgbG9uZyB3aGVuLCBpbnQgbW9kaWZpZXJzKSB7Ci0gICAgICAgIHN1cGVyKHNvdXJjZSwgaWQpOwotCi0gICAgICAgIHRoaXMud2hlbiA9IHdoZW47Ci0gICAgICAgIG1vZGlmaWVyc0V4ID0gZXh0cmFjdEV4RmxhZ3MobW9kaWZpZXJzKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldE1vZGlmaWVycygpIHsKLSAgICAgICAgaW50IG1vZGlmaWVycyA9IDA7Ci0KLSAgICAgICAgaWYgKChtb2RpZmllcnNFeCAmIFNISUZUX0RPV05fTUFTSykgIT0gMCkgewotICAgICAgICAgICAgbW9kaWZpZXJzIHw9IFNISUZUX01BU0s7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnNFeCAmIENUUkxfRE9XTl9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICBtb2RpZmllcnMgfD0gQ1RSTF9NQVNLOwotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzRXggJiBNRVRBX0RPV05fTUFTSykgIT0gMCkgewotICAgICAgICAgICAgbW9kaWZpZXJzIHw9IE1FVEFfTUFTSzsKLSAgICAgICAgfQotICAgICAgICBpZiAoKG1vZGlmaWVyc0V4ICYgQUxUX0RPV05fTUFTSykgIT0gMCkgewotICAgICAgICAgICAgbW9kaWZpZXJzIHw9IEFMVF9NQVNLOwotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzRXggJiBBTFRfR1JBUEhfRE9XTl9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICBtb2RpZmllcnMgfD0gQUxUX0dSQVBIX01BU0s7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnNFeCAmIEJVVFRPTjFfRE9XTl9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICBtb2RpZmllcnMgfD0gQlVUVE9OMV9NQVNLOwotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzRXggJiBCVVRUT04yX0RPV05fTUFTSykgIT0gMCkgewotICAgICAgICAgICAgbW9kaWZpZXJzIHw9IEJVVFRPTjJfTUFTSzsKLSAgICAgICAgfQotICAgICAgICBpZiAoKG1vZGlmaWVyc0V4ICYgQlVUVE9OM19ET1dOX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIG1vZGlmaWVycyB8PSBCVVRUT04zX01BU0s7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbW9kaWZpZXJzOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0TW9kaWZpZXJzRXgoKSB7Ci0gICAgICAgIHJldHVybiBtb2RpZmllcnNFeDsKLSAgICB9Ci0KLSAgICB2b2lkIHNldE1vZGlmaWVycyhpbnQgbW9kaWZpZXJzKSB7Ci0gICAgICAgIG1vZGlmaWVyc0V4ID0gZXh0cmFjdEV4RmxhZ3MobW9kaWZpZXJzKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0FsdERvd24oKSB7Ci0gICAgICAgIHJldHVybiAoKG1vZGlmaWVyc0V4ICYgQUxUX0RPV05fTUFTSykgIT0gMCk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNBbHRHcmFwaERvd24oKSB7Ci0gICAgICAgIHJldHVybiAoKG1vZGlmaWVyc0V4ICYgQUxUX0dSQVBIX0RPV05fTUFTSykgIT0gMCk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb250cm9sRG93bigpIHsKLSAgICAgICAgcmV0dXJuICgobW9kaWZpZXJzRXggJiBDVFJMX0RPV05fTUFTSykgIT0gMCk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNNZXRhRG93bigpIHsKLSAgICAgICAgcmV0dXJuICgobW9kaWZpZXJzRXggJiBNRVRBX0RPV05fTUFTSykgIT0gMCk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNTaGlmdERvd24oKSB7Ci0gICAgICAgIHJldHVybiAoKG1vZGlmaWVyc0V4ICYgU0hJRlRfRE9XTl9NQVNLKSAhPSAwKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgbG9uZyBnZXRXaGVuKCkgewotICAgICAgICByZXR1cm4gd2hlbjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBjb25zdW1lKCkgewotICAgICAgICBzdXBlci5jb25zdW1lKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb25zdW1lZCgpIHsKLSAgICAgICAgcmV0dXJuIHN1cGVyLmlzQ29uc3VtZWQoKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9JbnB1dE1ldGhvZEV2ZW50LmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvSW5wdXRNZXRob2RFdmVudC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiZTAwMWE1Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9JbnB1dE1ldGhvZEV2ZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNTYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotaW1wb3J0IGphdmEuYXd0LkFXVEV2ZW50OwotaW1wb3J0IGphdmEuYXd0LkNvbXBvbmVudDsKLWltcG9ydCBqYXZhLmF3dC5mb250LlRleHRIaXRJbmZvOwotaW1wb3J0IGphdmEudGV4dC5BdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3I7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIElucHV0TWV0aG9kRXZlbnQgZXh0ZW5kcyBBV1RFdmVudCB7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSA0NzI3MTkwODc0Nzc4OTIyNjYxTDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IElOUFVUX01FVEhPRF9GSVJTVCA9IDExMDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJTlBVVF9NRVRIT0RfVEVYVF9DSEFOR0VEID0gMTEwMDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENBUkVUX1BPU0lUSU9OX0NIQU5HRUQgPSAxMTAxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSU5QVVRfTUVUSE9EX0xBU1QgPSAxMTAxOwotCi0gICAgcHJpdmF0ZSBBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgdGV4dDsKLSAgICBwcml2YXRlIFRleHRIaXRJbmZvIHZpc2libGVQb3NpdGlvbjsKLSAgICBwcml2YXRlIFRleHRIaXRJbmZvIGNhcmV0OwotICAgIHByaXZhdGUgaW50IGNvbW1pdHRlZENoYXJhY3RlckNvdW50OwotICAgIHByaXZhdGUgbG9uZyB3aGVuOwotCi0gICAgcHVibGljIElucHV0TWV0aG9kRXZlbnQoQ29tcG9uZW50IHNyYywgaW50IGlkLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRleHRIaXRJbmZvIGNhcmV0LCAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBUZXh0SGl0SW5mbyB2aXNpYmxlUG9zKSB7Ci0gICAgICAgIHRoaXMoc3JjLCBpZCwgbnVsbCwgMCwgY2FyZXQsIHZpc2libGVQb3MpOwotICAgIH0KLQotICAgIHB1YmxpYyBJbnB1dE1ldGhvZEV2ZW50KENvbXBvbmVudCBzcmMsIGludCBpZCwgCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIHRleHQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGNvbW1pdGVkQ2hhckNvdW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRleHRIaXRJbmZvIGNhcmV0LCAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBUZXh0SGl0SW5mbyB2aXNpYmxlUG9zKSB7Ci0gICAgICAgIHRoaXMoc3JjLCBpZCwgMGwsIHRleHQsIGNvbW1pdGVkQ2hhckNvdW50LCBjYXJldCwgdmlzaWJsZVBvcyk7Ci0gICAgfQotCi0gICAgcHVibGljIElucHV0TWV0aG9kRXZlbnQoQ29tcG9uZW50IHNyYywgaW50IGlkLCBsb25nIHdoZW4sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIHRleHQsIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBjb21taXR0ZWRDaGFyYWN0ZXJDb3VudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBUZXh0SGl0SW5mbyBjYXJldCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBUZXh0SGl0SW5mbyB2aXNpYmxlUG9zKSB7Ci0gICAgICAgIHN1cGVyKHNyYywgaWQpOwotCi0gICAgICAgIGlmICgoaWQgPCBJTlBVVF9NRVRIT0RfRklSU1QpIHx8IChpZCA+IElOUFVUX01FVEhPRF9MQVNUKSkgewotICAgICAgICAgICAgLy8gYXd0LjE4RT1Xcm9uZyBldmVudCBpZAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOEUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZiAoKGlkID09IENBUkVUX1BPU0lUSU9OX0NIQU5HRUQpICYmICh0ZXh0ICE9IG51bGwpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMThGPVRleHQgbXVzdCBiZSBudWxsIGZvciBDQVJFVF9QT1NJVElPTl9DSEFOR0VECi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE4RiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGlmICgodGV4dCAhPSBudWxsKSAmJgotICAgICAgICAgICAgICAgICgoY29tbWl0dGVkQ2hhcmFjdGVyQ291bnQgPCAwKSB8fAotICAgICAgICAgICAgICAgICAoY29tbWl0dGVkQ2hhcmFjdGVyQ291bnQgPiAKLSAgICAgICAgICAgICAgICAgICAgICAgICh0ZXh0LmdldEVuZEluZGV4KCkgLSB0ZXh0LmdldEJlZ2luSW5kZXgoKSkpKSkgewotICAgICAgICAgICAgLy8gYXd0LjE5MD1Xcm9uZyBjb21taXR0ZWRDaGFyYWN0ZXJDb3VudAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOTAiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMud2hlbiA9IHdoZW47Ci0gICAgICAgIHRoaXMudGV4dCA9IHRleHQ7Ci0gICAgICAgIHRoaXMuY2FyZXQgPSBjYXJldDsKLSAgICAgICAgdGhpcy52aXNpYmxlUG9zaXRpb24gPSB2aXNpYmxlUG9zOwotICAgICAgICB0aGlzLmNvbW1pdHRlZENoYXJhY3RlckNvdW50ID0gY29tbWl0dGVkQ2hhcmFjdGVyQ291bnQ7Ci0gICAgfQotCi0gICAgcHVibGljIFRleHRIaXRJbmZvIGdldENhcmV0KCkgewotICAgICAgICByZXR1cm4gY2FyZXQ7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRDb21taXR0ZWRDaGFyYWN0ZXJDb3VudCgpIHsKLSAgICAgICAgcmV0dXJuIGNvbW1pdHRlZENoYXJhY3RlckNvdW50OwotICAgIH0KLQotICAgIHB1YmxpYyBBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgZ2V0VGV4dCgpIHsKLSAgICAgICAgcmV0dXJuIHRleHQ7Ci0gICAgfQotCi0gICAgcHVibGljIFRleHRIaXRJbmZvIGdldFZpc2libGVQb3NpdGlvbigpIHsKLSAgICAgICAgcmV0dXJuIHZpc2libGVQb3NpdGlvbjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgbG9uZyBnZXRXaGVuKCkgewotICAgICAgICByZXR1cm4gd2hlbjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBjb25zdW1lKCkgewotICAgICAgICBzdXBlci5jb25zdW1lKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb25zdW1lZCgpIHsKLSAgICAgICAgcmV0dXJuIHN1cGVyLmlzQ29uc3VtZWQoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHBhcmFtU3RyaW5nKCkgewotICAgICAgICAvKiBUaGUgZm9ybWF0IGlzIGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW9yIAotICAgICAgICAgKiB3aGljaCBjYW4gYmUgcmV2ZWFsZWQgYnkgdGhlIGZvbGxvd2luZyBjb2RlOgotICAgICAgICAgKiAKLSAgICAgICAgICogSW5wdXRNZXRob2RFdmVudCBlID0gbmV3IElucHV0TWV0aG9kRXZlbnQobmV3IENvbXBvbmVudCgpe30sCi0gICAgICAgICAqICAgICAgICAgIElucHV0TWV0aG9kRXZlbnQuSU5QVVRfTUVUSE9EX1RFWFRfQ0hBTkdFRCwKLSAgICAgICAgICogICAgICAgICAgVGV4dEhpdEluZm8ubGVhZGluZygxKSwgVGV4dEhpdEluZm8udHJhaWxpbmcoMikpOwotICAgICAgICAgKiBTeXN0ZW0ub3V0LnByaW50bG4oZSk7Ci0gICAgICAgICAqLwotICAgICAgICBTdHJpbmcgdHlwZVN0cmluZyA9IG51bGw7Ci0KLSAgICAgICAgc3dpdGNoIChpZCkgewotICAgICAgICBjYXNlIElOUFVUX01FVEhPRF9URVhUX0NIQU5HRUQ6Ci0gICAgICAgICAgICB0eXBlU3RyaW5nID0gIklOUFVUX01FVEhPRF9URVhUX0NIQU5HRUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBDQVJFVF9QT1NJVElPTl9DSEFOR0VEOgotICAgICAgICAgICAgdHlwZVN0cmluZyA9ICJDQVJFVF9QT1NJVElPTl9DSEFOR0VEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICB0eXBlU3RyaW5nID0gInVua25vd24gdHlwZSI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiB0eXBlU3RyaW5nICsgIix0ZXh0PSIgKyB0ZXh0ICsgIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAiLGNvbW1pdGVkQ2hhckNvdW50PSIgKyBjb21taXR0ZWRDaGFyYWN0ZXJDb3VudCArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAiLGNhcmV0PSIgKyBjYXJldCArICIsdmlzaWJsZVBvc2l0aW9uPSIgKyB2aXNpYmxlUG9zaXRpb247IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0lucHV0TWV0aG9kTGlzdGVuZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9JbnB1dE1ldGhvZExpc3RlbmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDg1ZWFhN2UuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L0lucHV0TWV0aG9kTGlzdGVuZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDM3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJbnB1dE1ldGhvZExpc3RlbmVyIGV4dGVuZHMgRXZlbnRMaXN0ZW5lciB7Ci0KLSAgICBwdWJsaWMgdm9pZCBjYXJldFBvc2l0aW9uQ2hhbmdlZChJbnB1dE1ldGhvZEV2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQgaW5wdXRNZXRob2RUZXh0Q2hhbmdlZChJbnB1dE1ldGhvZEV2ZW50IGUpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvSW52b2NhdGlvbkV2ZW50LmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvSW52b2NhdGlvbkV2ZW50LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDU4ZTNiNzIuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L0ludm9jYXRpb25FdmVudC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTM4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5BV1RFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5BY3RpdmVFdmVudDsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgbm90IHN1cHBvcnRlZCBpbiBBbmRyb2lkIDEuMC4gSXQgaXMgbWVyZWx5IHByb3ZpZGVkIHRvIG1haW50YWluCi0gKiBpbnRlcmZhY2UgY29tcGF0aWJpbGl0eSB3aXRoIGRlc2t0b3AgSmF2YSBpbXBsZW1lbnRhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgSW52b2NhdGlvbkV2ZW50IGV4dGVuZHMgQVdURXZlbnQgaW1wbGVtZW50cyBBY3RpdmVFdmVudCB7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSA0MzYwNTYzNDQ5MDk0NTk0NTBMOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSU5WT0NBVElPTl9GSVJTVCA9IDEyMDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJTlZPQ0FUSU9OX0RFRkFVTFQgPSAxMjAwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSU5WT0NBVElPTl9MQVNUID0gMTIwMDsKLQotICAgIHByb3RlY3RlZCBSdW5uYWJsZSBydW5uYWJsZTsKLQotICAgIHByb3RlY3RlZCBPYmplY3Qgbm90aWZpZXI7Ci0KLSAgICBwcm90ZWN0ZWQgYm9vbGVhbiBjYXRjaEV4Y2VwdGlvbnM7Ci0KLSAgICBwcml2YXRlIGxvbmcgd2hlbjsKLSAgICBwcml2YXRlIFRocm93YWJsZSB0aHJvd2FibGU7Ci0KLSAgICBwdWJsaWMgSW52b2NhdGlvbkV2ZW50KE9iamVjdCBzb3VyY2UsIFJ1bm5hYmxlIHJ1bm5hYmxlKSB7Ci0gICAgICAgIHRoaXMoc291cmNlLCBydW5uYWJsZSwgbnVsbCwgZmFsc2UpOwotICAgIH0KLQotICAgIHB1YmxpYyBJbnZvY2F0aW9uRXZlbnQoT2JqZWN0IHNvdXJjZSwgUnVubmFibGUgcnVubmFibGUsIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgT2JqZWN0IG5vdGlmaWVyLCBib29sZWFuIGNhdGNoRXhjZXB0aW9ucykgewotICAgICAgICB0aGlzKHNvdXJjZSwgSU5WT0NBVElPTl9ERUZBVUxULCBydW5uYWJsZSwgbm90aWZpZXIsIGNhdGNoRXhjZXB0aW9ucyk7Ci0gICAgfQotCi0gICAgcHJvdGVjdGVkIEludm9jYXRpb25FdmVudChPYmplY3Qgc291cmNlLCBpbnQgaWQsIFJ1bm5hYmxlIHJ1bm5hYmxlLAotICAgICAgICAgICAgT2JqZWN0IG5vdGlmaWVyLCBib29sZWFuIGNhdGNoRXhjZXB0aW9ucykKLSAgICB7Ci0gICAgICAgIHN1cGVyKHNvdXJjZSwgaWQpOwotCi0gICAgICAgIC8vIGF3dC4xOEM9Q2Fubm90IGludm9rZSBudWxsIHJ1bm5hYmxlCi0gICAgICAgIGFzc2VydCBydW5uYWJsZSAhPSBudWxsIDogTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMThDIik7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICBpZiAoc291cmNlID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4xOEQ9U291cmNlIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMThEIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy5ydW5uYWJsZSA9IHJ1bm5hYmxlOwotICAgICAgICB0aGlzLm5vdGlmaWVyID0gbm90aWZpZXI7Ci0gICAgICAgIHRoaXMuY2F0Y2hFeGNlcHRpb25zID0gY2F0Y2hFeGNlcHRpb25zOwotCi0gICAgICAgIHRocm93YWJsZSA9IG51bGw7Ci0gICAgICAgIHdoZW4gPSBTeXN0ZW0uY3VycmVudFRpbWVNaWxsaXMoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBkaXNwYXRjaCgpIHsKLSAgICAgICAgaWYgKCFjYXRjaEV4Y2VwdGlvbnMpIHsKLSAgICAgICAgICAgIHJ1bkFuZE5vdGlmeSgpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICBydW5BbmROb3RpZnkoKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKFRocm93YWJsZSB0KSB7Ci0gICAgICAgICAgICAgICAgdGhyb3dhYmxlID0gdDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHByaXZhdGUgdm9pZCBydW5BbmROb3RpZnkoKSB7Ci0gICAgICAgIGlmIChub3RpZmllciAhPSBudWxsKSB7Ci0gICAgICAgICAgICBzeW5jaHJvbml6ZWQobm90aWZpZXIpIHsKLSAgICAgICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgICAgICBydW5uYWJsZS5ydW4oKTsKLSAgICAgICAgICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgICAgICAgICBub3RpZmllci5ub3RpZnlBbGwoKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBydW5uYWJsZS5ydW4oKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBFeGNlcHRpb24gZ2V0RXhjZXB0aW9uKCkgewotICAgICAgICByZXR1cm4gKHRocm93YWJsZSAhPSBudWxsICYmIHRocm93YWJsZSBpbnN0YW5jZW9mIEV4Y2VwdGlvbikgPwotICAgICAgICAgICAgICAgIChFeGNlcHRpb24pdGhyb3dhYmxlIDogbnVsbDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgVGhyb3dhYmxlIGdldFRocm93YWJsZSgpIHsKLSAgICAgICAgcmV0dXJuIHRocm93YWJsZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgbG9uZyBnZXRXaGVuKCkgewotICAgICAgICByZXR1cm4gd2hlbjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHBhcmFtU3RyaW5nKCkgewotICAgICAgICAvKiBUaGUgZm9ybWF0IGlzIGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW9yIAotICAgICAgICAgKiB3aGljaCBjYW4gYmUgcmV2ZWFsZWQgYnkgdGhlIGZvbGxvd2luZyBjb2RlOgotICAgICAgICAgKiAKLSAgICAgICAgICogSW52b2NhdGlvbkV2ZW50IGUgPSBuZXcgSW52b2NhdGlvbkV2ZW50KG5ldyBDb21wb25lbnQoKXt9LAotICAgICAgICAgKiAgICAgICBuZXcgUnVubmFibGUoKSB7IHB1YmxpYyB2b2lkIHJ1bigpe30gfSk7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihlKTsKLSAgICAgICAgICovCi0KLSAgICAgICAgcmV0dXJuICgoaWQgPT0gSU5WT0NBVElPTl9ERUZBVUxUID8gIklOVk9DQVRJT05fREVGQVVMVCIgOiAidW5rbm93biB0eXBlIikgKyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgICAgICAgICAiLHJ1bm5hYmxlPSIgKyBydW5uYWJsZSArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAiLG5vdGlmaWVyPSIgKyBub3RpZmllciArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAiLGNhdGNoRXhjZXB0aW9ucz0iICsgY2F0Y2hFeGNlcHRpb25zICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICIsd2hlbj0iICsgd2hlbik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9JdGVtRXZlbnQuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9JdGVtRXZlbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMDk5MDhmMi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvSXRlbUV2ZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw5NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS5hd3QuQVdURXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuSXRlbVNlbGVjdGFibGU7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJdGVtRXZlbnQgZXh0ZW5kcyBBV1RFdmVudCB7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtNjA4NzA4MTMyNDQ3MjA2OTMzTDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IElURU1fRklSU1QgPSA3MDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJVEVNX0xBU1QgPSA3MDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJVEVNX1NUQVRFX0NIQU5HRUQgPSA3MDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTRUxFQ1RFRCA9IDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBERVNFTEVDVEVEID0gMjsKLQotICAgIHByaXZhdGUgT2JqZWN0IGl0ZW07Ci0gICAgcHJpdmF0ZSBpbnQgc3RhdGVDaGFuZ2U7Ci0KLSAgICBwdWJsaWMgSXRlbUV2ZW50KEl0ZW1TZWxlY3RhYmxlIHNvdXJjZSwgaW50IGlkLCBPYmplY3QgaXRlbSwgaW50IHN0YXRlQ2hhbmdlKSB7Ci0gICAgICAgIHN1cGVyKHNvdXJjZSwgaWQpOwotCi0gICAgICAgIHRoaXMuaXRlbSA9IGl0ZW07Ci0gICAgICAgIHRoaXMuc3RhdGVDaGFuZ2UgPSBzdGF0ZUNoYW5nZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgT2JqZWN0IGdldEl0ZW0oKSB7Ci0gICAgICAgIHJldHVybiBpdGVtOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0U3RhdGVDaGFuZ2UoKSB7Ci0gICAgICAgIHJldHVybiBzdGF0ZUNoYW5nZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgSXRlbVNlbGVjdGFibGUgZ2V0SXRlbVNlbGVjdGFibGUoKSB7Ci0gICAgICAgIHJldHVybiAoSXRlbVNlbGVjdGFibGUpIHNvdXJjZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHBhcmFtU3RyaW5nKCkgewotICAgICAgICAvKiBUaGUgZm9ybWF0IGlzIGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW9yIAotICAgICAgICAgKiB3aGljaCBjYW4gYmUgcmV2ZWFsZWQgYnkgdGhlIGZvbGxvd2luZyBjb2RlOgotICAgICAgICAgKiAKLSAgICAgICAgICogQ2hlY2tib3ggYyA9IG5ldyBDaGVja2JveCgiQ2hlY2tib3giLCB0cnVlKTsKLSAgICAgICAgICogSXRlbUV2ZW50IGUgPSBuZXcgSXRlbUV2ZW50KGMsIEl0ZW1FdmVudC5JVEVNX1NUQVRFX0NIQU5HRUQsIAotICAgICAgICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYywgSXRlbUV2ZW50LlNFTEVDVEVEKTsKLSAgICAgICAgICogU3lzdGVtLm91dC5wcmludGxuKGUpOwotICAgICAgICAgKi8KLQotICAgICAgICBTdHJpbmcgc3RhdGVTdHJpbmcgPSBudWxsOwotCi0gICAgICAgIHN3aXRjaCAoc3RhdGVDaGFuZ2UpIHsKLSAgICAgICAgY2FzZSBTRUxFQ1RFRDoKLSAgICAgICAgICAgIHN0YXRlU3RyaW5nID0gIlNFTEVDVEVEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgREVTRUxFQ1RFRDoKLSAgICAgICAgICAgIHN0YXRlU3RyaW5nID0gIkRFU0VMRUNURUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHN0YXRlU3RyaW5nID0gInVua25vd24gdHlwZSI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiAoKGlkID09IElURU1fU1RBVEVfQ0hBTkdFRCA/ICJJVEVNX1NUQVRFX0NIQU5HRUQiIDogInVua25vd24gdHlwZSIpICsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgIixpdGVtPSIgKyBpdGVtICsgIixzdGF0ZUNoYW5nZT0iICsgc3RhdGVTdHJpbmcpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9JdGVtTGlzdGVuZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9JdGVtTGlzdGVuZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOGRlYzY3My4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvSXRlbUxpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgSXRlbUxpc3RlbmVyIGV4dGVuZHMgRXZlbnRMaXN0ZW5lciB7Ci0KLSAgICBwdWJsaWMgdm9pZCBpdGVtU3RhdGVDaGFuZ2VkKEl0ZW1FdmVudCBlKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L0tleUFkYXB0ZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9LZXlBZGFwdGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGE5NmNjYTguLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L0tleUFkYXB0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBLZXlBZGFwdGVyIGltcGxlbWVudHMgS2V5TGlzdGVuZXIgewotCi0gICAgcHVibGljIEtleUFkYXB0ZXIoKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQga2V5UHJlc3NlZChLZXlFdmVudCBlKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQga2V5UmVsZWFzZWQoS2V5RXZlbnQgZSkgewotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGtleVR5cGVkKEtleUV2ZW50IGUpIHsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9LZXlFdmVudC5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L0tleUV2ZW50LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDg2MjdmNzAuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L0tleUV2ZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2ODcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotaW1wb3J0IGphdmEuYXd0LkNvbXBvbmVudDsKLWltcG9ydCBqYXZhLmF3dC5Ub29sa2l0OwotaW1wb3J0IGphdmEubGFuZy5yZWZsZWN0LkZpZWxkOwotaW1wb3J0IGphdmEubGFuZy5yZWZsZWN0Lk1vZGlmaWVyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBLZXlFdmVudCBleHRlbmRzIElucHV0RXZlbnQgewotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gLTIzNTIxMzA5NTMwMjgxMjY5NTRMOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgS0VZX0ZJUlNUID0gNDAwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgS0VZX0xBU1QgPSA0MDI7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBLRVlfVFlQRUQgPSA0MDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBLRVlfUFJFU1NFRCA9IDQwMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEtFWV9SRUxFQVNFRCA9IDQwMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0VOVEVSID0gMTA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19CQUNLX1NQQUNFID0gODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1RBQiA9IDk7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19DQU5DRUwgPSAzOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQ0xFQVIgPSAxMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1NISUZUID0gMTY7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19DT05UUk9MID0gMTc7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19BTFQgPSAxODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1BBVVNFID0gMTk7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19DQVBTX0xPQ0sgPSAyMDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0VTQ0FQRSA9IDI3OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfU1BBQ0UgPSAzMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1BBR0VfVVAgPSAzMzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1BBR0VfRE9XTiA9IDM0OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRU5EID0gMzU7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19IT01FID0gMzY7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19MRUZUID0gMzc7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19VUCA9IDM4OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfUklHSFQgPSAzOTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0RPV04gPSA0MDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0NPTU1BID0gNDQ7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19NSU5VUyA9IDQ1OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfUEVSSU9EID0gNDY7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19TTEFTSCA9IDQ3OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfMCA9IDQ4OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfMSA9IDQ5OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfMiA9IDUwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfMyA9IDUxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfNCA9IDUyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfNSA9IDUzOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfNiA9IDU0OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfNyA9IDU1OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfOCA9IDU2OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfOSA9IDU3OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfU0VNSUNPTE9OID0gNTk7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19FUVVBTFMgPSA2MTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0EgPSA2NTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0IgPSA2NjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0MgPSA2NzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0QgPSA2ODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0UgPSA2OTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0YgPSA3MDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0cgPSA3MTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0ggPSA3MjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0kgPSA3MzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0ogPSA3NDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0sgPSA3NTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0wgPSA3NjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX00gPSA3NzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX04gPSA3ODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX08gPSA3OTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1AgPSA4MDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1EgPSA4MTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1IgPSA4MjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1MgPSA4MzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1QgPSA4NDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1UgPSA4NTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1YgPSA4NjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1cgPSA4NzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1ggPSA4ODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1kgPSA4OTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1ogPSA5MDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX09QRU5fQlJBQ0tFVCA9IDkxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQkFDS19TTEFTSCA9IDkyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQ0xPU0VfQlJBQ0tFVCA9IDkzOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTlVNUEFEMCA9IDk2OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTlVNUEFEMSA9IDk3OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTlVNUEFEMiA9IDk4OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTlVNUEFEMyA9IDk5OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTlVNUEFENCA9IDEwMDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX05VTVBBRDUgPSAxMDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19OVU1QQUQ2ID0gMTAyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTlVNUEFENyA9IDEwMzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX05VTVBBRDggPSAxMDQ7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19OVU1QQUQ5ID0gMTA1OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTVVMVElQTFkgPSAxMDY7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19BREQgPSAxMDc7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19TRVBBUkFURVIgPSAxMDg7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19TRVBBUkFUT1IgPSAxMDg7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19TVUJUUkFDVCA9IDEwOTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0RFQ0lNQUwgPSAxMTA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19ESVZJREUgPSAxMTE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19ERUxFVEUgPSAxMjc7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19OVU1fTE9DSyA9IDE0NDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1NDUk9MTF9MT0NLID0gMTQ1OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjEgPSAxMTI7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19GMiA9IDExMzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0YzID0gMTE0OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjQgPSAxMTU7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19GNSA9IDExNjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0Y2ID0gMTE3OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjcgPSAxMTg7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19GOCA9IDExOTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0Y5ID0gMTIwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjEwID0gMTIxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjExID0gMTIyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjEyID0gMTIzOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjEzID0gNjE0NDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19GMTQgPSA2MTQ0MTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0YxNSA9IDYxNDQyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjE2ID0gNjE0NDM7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19GMTcgPSA2MTQ0NDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0YxOCA9IDYxNDQ1OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjE5ID0gNjE0NDY7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19GMjAgPSA2MTQ0NzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0YyMSA9IDYxNDQ4OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRjIyID0gNjE0NDk7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19GMjMgPSA2MTQ1MDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0YyNCA9IDYxNDUxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfUFJJTlRTQ1JFRU4gPSAxNTQ7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19JTlNFUlQgPSAxNTU7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19IRUxQID0gMTU2OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTUVUQSA9IDE1NzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0JBQ0tfUVVPVEUgPSAxOTI7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19RVU9URSA9IDIyMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0tQX1VQID0gMjI0OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfS1BfRE9XTiA9IDIyNTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0tQX0xFRlQgPSAyMjY7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19LUF9SSUdIVCA9IDIyNzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0RFQURfR1JBVkUgPSAxMjg7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19ERUFEX0FDVVRFID0gMTI5OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfREVBRF9DSVJDVU1GTEVYID0gMTMwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfREVBRF9USUxERSA9IDEzMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0RFQURfTUFDUk9OID0gMTMyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfREVBRF9CUkVWRSA9IDEzMzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0RFQURfQUJPVkVET1QgPSAxMzQ7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19ERUFEX0RJQUVSRVNJUyA9IDEzNTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0RFQURfQUJPVkVSSU5HID0gMTM2OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfREVBRF9ET1VCTEVBQ1VURSA9IDEzNzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0RFQURfQ0FST04gPSAxMzg7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19ERUFEX0NFRElMTEEgPSAxMzk7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19ERUFEX09HT05FSyA9IDE0MDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0RFQURfSU9UQSA9IDE0MTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0RFQURfVk9JQ0VEX1NPVU5EID0gMTQyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfREVBRF9TRU1JVk9JQ0VEX1NPVU5EID0gMTQzOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQU1QRVJTQU5EID0gMTUwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQVNURVJJU0sgPSAxNTE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19RVU9URURCTCA9IDE1MjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0xFU1MgPSAxNTM7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19HUkVBVEVSID0gMTYwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQlJBQ0VMRUZUID0gMTYxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQlJBQ0VSSUdIVCA9IDE2MjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0FUID0gNTEyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQ09MT04gPSA1MTM7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19DSVJDVU1GTEVYID0gNTE0OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRE9MTEFSID0gNTE1OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRVVST19TSUdOID0gNTE2OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRVhDTEFNQVRJT05fTUFSSyA9IDUxNzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0lOVkVSVEVEX0VYQ0xBTUFUSU9OX01BUksgPSA1MTg7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19MRUZUX1BBUkVOVEhFU0lTID0gNTE5OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTlVNQkVSX1NJR04gPSA1MjA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19QTFVTID0gNTIxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfUklHSFRfUEFSRU5USEVTSVMgPSA1MjI7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19VTkRFUlNDT1JFID0gNTIzOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRklOQUwgPSAyNDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1dJTkRPV1MgPSA1MjQ7IAotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQ09OVEVYVF9NRU5VID0gNTI1OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQ09OVkVSVCA9IDI4OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfTk9OQ09OVkVSVCA9IDI5OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQUNDRVBUID0gMzA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19NT0RFQ0hBTkdFID0gMzE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19LQU5BID0gMjE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19LQU5KSSA9IDI1OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQUxQSEFOVU1FUklDID0gMjQwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfS0FUQUtBTkEgPSAyNDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19ISVJBR0FOQSA9IDI0MjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0ZVTExfV0lEVEggPSAyNDM7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19IQUxGX1dJRFRIID0gMjQ0OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfUk9NQU5fQ0hBUkFDVEVSUyA9IDI0NTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0FMTF9DQU5ESURBVEVTID0gMjU2OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfUFJFVklPVVNfQ0FORElEQVRFID0gMjU3OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQ09ERV9JTlBVVCA9IDI1ODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0pBUEFORVNFX0tBVEFLQU5BID0gMjU5OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfSkFQQU5FU0VfSElSQUdBTkEgPSAyNjA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19KQVBBTkVTRV9ST01BTiA9IDI2MTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0tBTkFfTE9DSyA9IDI2MjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0lOUFVUX01FVEhPRF9PTl9PRkYgPSAyNjM7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19DVVQgPSA2NTQ4OTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0NPUFkgPSA2NTQ4NTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1BBU1RFID0gNjU0ODc7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19VTkRPID0gNjU0ODM7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBWS19BR0FJTiA9IDY1NDgxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfRklORCA9IDY1NDg4OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfUFJPUFMgPSA2NTQ4MjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1NUT1AgPSA2NTQ4MDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0NPTVBPU0UgPSA2NTMxMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX0FMVF9HUkFQSCA9IDY1NDA2OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVktfQkVHSU4gPSA2NTM2ODsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFZLX1VOREVGSU5FRCA9IDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGNoYXIgQ0hBUl9VTkRFRklORUQgPSAoY2hhcikoLTEpOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgS0VZX0xPQ0FUSU9OX1VOS05PV04gPSAwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgS0VZX0xPQ0FUSU9OX1NUQU5EQVJEID0gMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEtFWV9MT0NBVElPTl9MRUZUID0gMjsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEtFWV9MT0NBVElPTl9SSUdIVCA9IDM7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBLRVlfTE9DQVRJT05fTlVNUEFEID0gNDsKLQotICAgIHByaXZhdGUgaW50IGtleUNvZGU7Ci0gICAgcHJpdmF0ZSBjaGFyIGtleUNoYXI7Ci0gICAgcHJpdmF0ZSBpbnQga2V5TG9jYXRpb247Ci0KLSAgICBwdWJsaWMgc3RhdGljIFN0cmluZyBnZXRLZXlNb2RpZmllcnNUZXh0KGludCBtb2RpZmllcnMpIHsKLSAgICAgICAgcmV0dXJuIGdldEtleU1vZGlmaWVyc0V4VGV4dChleHRyYWN0RXhGbGFncyhtb2RpZmllcnMpKTsKLSAgICB9Ci0KLSAgICBzdGF0aWMgU3RyaW5nIGdldEtleU1vZGlmaWVyc0V4VGV4dChpbnQgbW9kaWZpZXJzRXgpIHsKLSAgICAgICAgU3RyaW5nIHRleHQgPSAiIjsgLy8kTk9OLU5MUy0xJAotCi0gICAgICAgIGlmICgobW9kaWZpZXJzRXggJiBJbnB1dEV2ZW50Lk1FVEFfRE9XTl9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICB0ZXh0ICs9IFRvb2xraXQuZ2V0UHJvcGVydHkoIkFXVC5tZXRhIiwgIk1ldGEiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnNFeCAmIElucHV0RXZlbnQuQ1RSTF9ET1dOX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIHRleHQgKz0gKCh0ZXh0Lmxlbmd0aCgpID4gMCkgPyAiKyIgOiAiIikgKyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgICAgICAgICAgICAgVG9vbGtpdC5nZXRQcm9wZXJ0eSgiQVdULmNvbnRyb2wiLCAiQ3RybCIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgfQotICAgICAgICBpZiAoKG1vZGlmaWVyc0V4ICYgSW5wdXRFdmVudC5BTFRfRE9XTl9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICB0ZXh0ICs9ICgodGV4dC5sZW5ndGgoKSA+IDApID8gIisiIDogIiIpICsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgICAgIFRvb2xraXQuZ2V0UHJvcGVydHkoIkFXVC5hbHQiLCAiQWx0Iik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzRXggJiBJbnB1dEV2ZW50LlNISUZUX0RPV05fTUFTSykgIT0gMCkgewotICAgICAgICAgICAgdGV4dCArPSAoKHRleHQubGVuZ3RoKCkgPiAwKSA/ICIrIiA6ICIiKSArIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgICAgICAgICBUb29sa2l0LmdldFByb3BlcnR5KCJBV1Quc2hpZnQiLCAiU2hpZnQiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnNFeCAmIElucHV0RXZlbnQuQUxUX0dSQVBIX0RPV05fTUFTSykgIT0gMCkgewotICAgICAgICAgICAgdGV4dCArPSAoKHRleHQubGVuZ3RoKCkgPiAwKSA/ICIrIiA6ICIiKSArIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgICAgICAgICBUb29sa2l0LmdldFByb3BlcnR5KCJBV1QuYWx0R3JhcGgiLCAiQWx0IEdyYXBoIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHRleHQ7Ci0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyBTdHJpbmcgZ2V0S2V5VGV4dChpbnQga2V5Q29kZSkgewotICAgICAgICBTdHJpbmdbXSByYXdOYW1lID0gZ2V0UHVibGljU3RhdGljRmluYWxJbnRGaWVsZE5hbWUoa2V5Q29kZSk7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICBpZiAoKHJhd05hbWUgPT0gbnVsbCkgfHwgKHJhd05hbWUubGVuZ3RoID09IDApKSB7Ci0gICAgICAgICAgICByZXR1cm4gKCJVbmtub3duIGtleUNvZGU6ICIgKyAoa2V5Q29kZSA+PSAwID8gIjB4IiA6ICItMHgiKSArIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkCi0gICAgICAgICAgICAgICAgICAgIEludGVnZXIudG9IZXhTdHJpbmcoTWF0aC5hYnMoa2V5Q29kZSkpKTsKLSAgICAgICAgfQotCi0gICAgICAgIFN0cmluZyBwcm9wZXJ0eU5hbWUgPSBnZXRQcm9wZXJ0eU5hbWUocmF3TmFtZSk7Ci0gICAgICAgIFN0cmluZyBkZWZhdWx0TmFtZSA9IGdldERlZmF1bHROYW1lKHJhd05hbWUpOwotCi0gICAgICAgIHJldHVybiBUb29sa2l0LmdldFByb3BlcnR5KHByb3BlcnR5TmFtZSwgZGVmYXVsdE5hbWUpOwotICAgIH0KLQotICAgIHByaXZhdGUgc3RhdGljIFN0cmluZyBnZXREZWZhdWx0TmFtZShTdHJpbmdbXSByYXdOYW1lKSB7Ci0gICAgICAgIFN0cmluZyBuYW1lID0gIiI7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgdHJ1ZTsgaSsrKSB7Ci0gICAgICAgICAgICBTdHJpbmcgcGFydCA9IHJhd05hbWVbaV07Ci0KLSAgICAgICAgICAgIG5hbWUgKz0gbmV3IFN0cmluZyhuZXcgY2hhcltdIHtwYXJ0LmNoYXJBdCgwKX0pLnRvVXBwZXJDYXNlKCkgKwotICAgICAgICAgICAgICAgICAgICBwYXJ0LnN1YnN0cmluZygxKS50b0xvd2VyQ2FzZSgpOwotCi0gICAgICAgICAgICBpZiAoaSA9PSAocmF3TmFtZS5sZW5ndGggLSAxKSkgewotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgbmFtZSArPSAiICI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBuYW1lOwotICAgIH0KLQotICAgIHByaXZhdGUgc3RhdGljIFN0cmluZyBnZXRQcm9wZXJ0eU5hbWUoU3RyaW5nW10gcmF3TmFtZSkgewotICAgICAgICBTdHJpbmcgbmFtZSA9IHJhd05hbWVbMF0udG9Mb3dlckNhc2UoKTsKLQotICAgICAgICBmb3IgKGludCBpID0gMTsgaSA8IHJhd05hbWUubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIFN0cmluZyBwYXJ0ID0gcmF3TmFtZVtpXTsKLQotICAgICAgICAgICAgbmFtZSArPSBuZXcgU3RyaW5nKG5ldyBjaGFyW10ge3BhcnQuY2hhckF0KDApfSkudG9VcHBlckNhc2UoKSArCi0gICAgICAgICAgICAgICAgICAgIHBhcnQuc3Vic3RyaW5nKDEpLnRvTG93ZXJDYXNlKCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gKCJBV1QuIiArIG5hbWUpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgcHJpdmF0ZSBzdGF0aWMgU3RyaW5nW10gZ2V0UHVibGljU3RhdGljRmluYWxJbnRGaWVsZE5hbWUoaW50IHZhbHVlKSB7Ci0gICAgICAgIEZpZWxkW10gYWxsRmllbGRzID0gS2V5RXZlbnQuY2xhc3MuZ2V0RGVjbGFyZWRGaWVsZHMoKTsKLQotICAgICAgICB0cnkgewotICAgICAgICAgICAgZm9yIChGaWVsZCBmaWVsZCA6IGFsbEZpZWxkcykgewotICAgICAgICAgICAgICAgIENsYXNzPD8+IHNzYWxjID0gZmllbGQuZ2V0VHlwZSgpOwotICAgICAgICAgICAgICAgIGludCBtb2RpZmllcnMgPSBmaWVsZC5nZXRNb2RpZmllcnMoKTsKLQotICAgICAgICAgICAgICAgIGlmIChzc2FsYy5pc1ByaW1pdGl2ZSgpICYmIHNzYWxjLmdldE5hbWUoKS5lcXVhbHMoImludCIpICYmIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgICAgIE1vZGlmaWVyLmlzRmluYWwobW9kaWZpZXJzKSAmJiBNb2RpZmllci5pc1B1YmxpYyhtb2RpZmllcnMpICYmCi0gICAgICAgICAgICAgICAgICAgICAgICBNb2RpZmllci5pc1N0YXRpYyhtb2RpZmllcnMpKQotICAgICAgICAgICAgICAgIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGZpZWxkLmdldEludChudWxsKSA9PSB2YWx1ZSl7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmaW5hbCBTdHJpbmcgbmFtZSA9IGZpZWxkLmdldE5hbWUoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZpbmFsIGludCBwcmVmaXhMZW5ndGggPSBuYW1lLmluZGV4T2YoIl8iKSArIDE7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gbmFtZS5zdWJzdHJpbmcocHJlZml4TGVuZ3RoKS5zcGxpdCgiXyIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbihlKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIEtleUV2ZW50KENvbXBvbmVudCBzcmMsIGludCBpZCwKLSAgICAgICAgICAgICAgICAgICAgbG9uZyB3aGVuLCBpbnQgbW9kaWZpZXJzLAotICAgICAgICAgICAgICAgICAgICBpbnQga2V5Q29kZSkgewotICAgICAgICB0aGlzKHNyYywgaWQsIHdoZW4sIG1vZGlmaWVycywga2V5Q29kZSwKLSAgICAgICAgICAgICAgICAoa2V5Q29kZSA+ICgyIDw8IDcpIC0gMSkgPyBDSEFSX1VOREVGSU5FRCA6IChjaGFyKSBrZXlDb2RlKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgS2V5RXZlbnQoQ29tcG9uZW50IHNyYywgaW50IGlkLAotICAgICAgICAgICAgICAgICAgICBsb25nIHdoZW4sIGludCBtb2RpZmllcnMsCi0gICAgICAgICAgICAgICAgICAgIGludCBrZXlDb2RlLCBjaGFyIGtleUNoYXIpIHsKLSAgICAgICAgdGhpcyhzcmMsIGlkLCB3aGVuLCBtb2RpZmllcnMsIGtleUNvZGUsIGtleUNoYXIsIEtFWV9MT0NBVElPTl9VTktOT1dOKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgS2V5RXZlbnQoQ29tcG9uZW50IHNyYywgaW50IGlkLAotICAgICAgICAgICAgICAgICAgICBsb25nIHdoZW4sIGludCBtb2RpZmllcnMsCi0gICAgICAgICAgICAgICAgICAgIGludCBrZXlDb2RlLCBjaGFyIGtleUNoYXIsCi0gICAgICAgICAgICAgICAgICAgIGludCBrZXlMb2NhdGlvbikgewotICAgICAgICBzdXBlcihzcmMsIGlkLCB3aGVuLCBtb2RpZmllcnMpOwotCi0gICAgICAgIGlmIChpZCA9PSBLRVlfVFlQRUQpIHsKLSAgICAgICAgICAgIGlmIChrZXlDb2RlICE9IFZLX1VOREVGSU5FRCkgewotICAgICAgICAgICAgICAgIC8vIGF3dC4xOTE9SW52YWxpZCBrZXlDb2RlIGZvciBLRVlfVFlQRUQgZXZlbnQsIG11c3QgYmUgVktfVU5ERUZJTkVECi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOTEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChrZXlDaGFyID09IENIQVJfVU5ERUZJTkVEKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjE5Mj1JbnZhbGlkIGtleUNoYXIgZm9yIEtFWV9UWVBFRCBldmVudCwgY2FuJ3QgYmUgQ0hBUl9VTkRFRklORUQKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE5MiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBpZiAoKGtleUxvY2F0aW9uIDwgS0VZX0xPQ0FUSU9OX1VOS05PV04pCi0gICAgICAgICAgICAgICAgfHwgKGtleUxvY2F0aW9uID4gS0VZX0xPQ0FUSU9OX05VTVBBRCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yOTc9SW52YWxpZCBrZXlMb2NhdGlvbgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOTciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMua2V5Q2hhciA9IGtleUNoYXI7Ci0gICAgICAgIHRoaXMua2V5TG9jYXRpb24gPSBrZXlMb2NhdGlvbjsKLSAgICAgICAgdGhpcy5rZXlDb2RlID0ga2V5Q29kZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldEtleUNvZGUoKSB7Ci0gICAgICAgIHJldHVybiBrZXlDb2RlOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldEtleUNvZGUoaW50IGtleUNvZGUpIHsKLSAgICAgICAgdGhpcy5rZXlDb2RlID0ga2V5Q29kZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgY2hhciBnZXRLZXlDaGFyKCkgewotICAgICAgICByZXR1cm4ga2V5Q2hhcjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRLZXlDaGFyKGNoYXIga2V5Q2hhcikgewotICAgICAgICB0aGlzLmtleUNoYXIgPSBrZXlDaGFyOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0S2V5TG9jYXRpb24oKSB7Ci0gICAgICAgIHJldHVybiBrZXlMb2NhdGlvbjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBARGVwcmVjYXRlZAotICAgIHB1YmxpYyB2b2lkIHNldE1vZGlmaWVycyhpbnQgbW9kaWZpZXJzKSB7Ci0gICAgICAgIHN1cGVyLnNldE1vZGlmaWVycyhtb2RpZmllcnMpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGlzQWN0aW9uS2V5KCkgewotICAgICAgICByZXR1cm4gKChrZXlDaGFyID09IENIQVJfVU5ERUZJTkVEKSAmJiAoa2V5Q29kZSAhPSBWS19VTkRFRklORUQpICYmCi0gICAgICAgICAgICAgICAgISgoa2V5Q29kZSA9PSBWS19BTFQpIHx8IChrZXlDb2RlID09IFZLX0FMVF9HUkFQSCkgfHwKLSAgICAgICAgICAgICAgICAgICAgKGtleUNvZGUgPT0gVktfQ09OVFJPTCkgfHwgKGtleUNvZGUgPT0gVktfTUVUQSkgfHwgKGtleUNvZGUgPT0gVktfU0hJRlQpKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyBwYXJhbVN0cmluZygpIHsKLSAgICAgICAgLyoKLSAgICAgICAgICogVGhlIGZvcm1hdCBpcyBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvcgotICAgICAgICAgKiB3aGljaCBjYW4gYmUgcmV2ZWFsZWQgYnkgdGhlIGZvbGxvd2luZyBjb2RlOgotICAgICAgICAgKgotICAgICAgICAgKiBLZXlFdmVudCBlID0gbmV3IEtleUV2ZW50KG5ldyBDb21wb25lbnQoKSB7fSwgCi0gICAgICAgICAqICAgICAgIEtleUV2ZW50LktFWV9QUkVTU0VELCAwLCAKLSAgICAgICAgICogICAgICAgS2V5RXZlbnQuQ1RSTF9ET1dOX01BU0t8S2V5RXZlbnQuU0hJRlRfRE9XTl9NQVNLLCAKLSAgICAgICAgICogICAgICAgS2V5RXZlbnQuVktfQSwgJ0EnLCBLZXlFdmVudC5LRVlfTE9DQVRJT05fU1RBTkRBUkQpOwotICAgICAgICAgKiBTeXN0ZW0ub3V0LnByaW50bG4oZSk7Ci0gICAgICAgICAqLwotCi0gICAgICAgIFN0cmluZyBpZFN0cmluZyA9IG51bGw7Ci0gICAgICAgIFN0cmluZyBsb2NTdHJpbmcgPSBudWxsOwotICAgICAgICBTdHJpbmcgcGFyYW1TdHJpbmcgPSBudWxsOwotICAgICAgICBTdHJpbmcga2V5Q2hhclN0cmluZyA9IChrZXlDaGFyID09ICdcbicpID8KLSAgICAgICAgICAgICAgICBrZXlDaGFyU3RyaW5nID0gZ2V0S2V5VGV4dChWS19FTlRFUikgOiAiJyIgKyBrZXlDaGFyICsgIiciOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLQotICAgICAgICBzd2l0Y2ggKGlkKSB7Ci0gICAgICAgIGNhc2UgS0VZX1BSRVNTRUQ6Ci0gICAgICAgICAgICBpZFN0cmluZyA9ICJLRVlfUFJFU1NFRCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEtFWV9SRUxFQVNFRDoKLSAgICAgICAgICAgIGlkU3RyaW5nID0gIktFWV9SRUxFQVNFRCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEtFWV9UWVBFRDoKLSAgICAgICAgICAgIGlkU3RyaW5nID0gIktFWV9UWVBFRCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgaWRTdHJpbmcgPSAidW5rbm93biB0eXBlIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgc3dpdGNoKGtleUxvY2F0aW9uKXsKLSAgICAgICAgY2FzZSBLRVlfTE9DQVRJT05fU1RBTkRBUkQ6Ci0gICAgICAgICAgICBsb2NTdHJpbmcgPSAiS0VZX0xPQ0FUSU9OX1NUQU5EQVJEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgS0VZX0xPQ0FUSU9OX0xFRlQ6Ci0gICAgICAgICAgICBsb2NTdHJpbmcgPSAiS0VZX0xPQ0FUSU9OX0xFRlQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBLRVlfTE9DQVRJT05fUklHSFQ6Ci0gICAgICAgICAgICBsb2NTdHJpbmcgPSAiS0VZX0xPQ0FUSU9OX1JJR0hUIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgS0VZX0xPQ0FUSU9OX05VTVBBRDoKLSAgICAgICAgICAgIGxvY1N0cmluZyA9ICJLRVlfTE9DQVRJT05fTlVNUEFEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgS0VZX0xPQ0FUSU9OX1VOS05PV046Ci0gICAgICAgICAgICBsb2NTdHJpbmcgPSAiS0VZX0xPQ0FUSU9OX1VOS05PV04iOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIGxvY1N0cmluZyA9ICJ1bmtub3duIHR5cGUiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBwYXJhbVN0cmluZyA9IGlkU3RyaW5nICsgIixrZXlDb2RlPSIgKyBrZXlDb2RlOyAvLyROT04tTkxTLTEkCi0gICAgICAgIGlmIChpc0FjdGlvbktleSgpKSB7Ci0gICAgICAgICAgICBwYXJhbVN0cmluZyArPSAiLCIgKyBnZXRLZXlUZXh0KGtleUNvZGUpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBwYXJhbVN0cmluZyArPSAiLGtleUNoYXI9IiArIGtleUNoYXJTdHJpbmc7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZiAoZ2V0TW9kaWZpZXJzRXgoKSA+IDApIHsKLSAgICAgICAgICAgIHBhcmFtU3RyaW5nICs9ICIsbW9kaWZpZXJzPSIgKyBnZXRNb2RpZmllcnNFeFRleHQoZ2V0TW9kaWZpZXJzRXgoKSkgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgICIsZXh0TW9kaWZpZXJzPSIgKyBnZXRNb2RpZmllcnNFeFRleHQoZ2V0TW9kaWZpZXJzRXgoKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBwYXJhbVN0cmluZyArPSAiLGtleUxvY2F0aW9uPSIgKyBsb2NTdHJpbmc7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICByZXR1cm4gcGFyYW1TdHJpbmc7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvS2V5TGlzdGVuZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9LZXlMaXN0ZW5lci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlYzE0NGRmLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9LZXlMaXN0ZW5lci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotaW1wb3J0IGphdmEudXRpbC5FdmVudExpc3RlbmVyOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgbm90IHN1cHBvcnRlZCBpbiBBbmRyb2lkIDEuMC4gSXQgaXMgbWVyZWx5IHByb3ZpZGVkIHRvIG1haW50YWluCi0gKiBpbnRlcmZhY2UgY29tcGF0aWJpbGl0eSB3aXRoIGRlc2t0b3AgSmF2YSBpbXBsZW1lbnRhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIEtleUxpc3RlbmVyIGV4dGVuZHMgRXZlbnRMaXN0ZW5lciB7Ci0KLSAgICBwdWJsaWMgdm9pZCBrZXlQcmVzc2VkKEtleUV2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQga2V5UmVsZWFzZWQoS2V5RXZlbnQgZSk7Ci0KLSAgICBwdWJsaWMgdm9pZCBrZXlUeXBlZChLZXlFdmVudCBlKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L01vdXNlQWRhcHRlci5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L01vdXNlQWRhcHRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkYzE5MTczLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZUFkYXB0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ5ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBNb3VzZUFkYXB0ZXIgaW1wbGVtZW50cyBNb3VzZUxpc3RlbmVyIHsKLQotICAgIHB1YmxpYyBNb3VzZUFkYXB0ZXIoKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgbW91c2VDbGlja2VkKE1vdXNlRXZlbnQgZSkgewotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIG1vdXNlRW50ZXJlZChNb3VzZUV2ZW50IGUpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBtb3VzZUV4aXRlZChNb3VzZUV2ZW50IGUpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBtb3VzZVByZXNzZWQoTW91c2VFdmVudCBlKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgbW91c2VSZWxlYXNlZChNb3VzZUV2ZW50IGUpIHsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZUV2ZW50LmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvTW91c2VFdmVudC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyYjFmYThiLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZUV2ZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMzIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotaW1wb3J0IGphdmEuYXd0LkNvbXBvbmVudDsKLWltcG9ydCBqYXZhLmF3dC5Qb2ludDsKLWltcG9ydCBqYXZhLmF3dC5Ub29sa2l0OwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBNb3VzZUV2ZW50IGV4dGVuZHMgSW5wdXRFdmVudCB7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtOTkxMjE0MTUzNDk0ODQyODQ4TDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IE1PVVNFX0ZJUlNUID0gNTAwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTU9VU0VfTEFTVCA9IDUwNzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IE1PVVNFX0NMSUNLRUQgPSA1MDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNT1VTRV9QUkVTU0VEID0gNTAxOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTU9VU0VfUkVMRUFTRUQgPSA1MDI7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNT1VTRV9NT1ZFRCA9IDUwMzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IE1PVVNFX0VOVEVSRUQgPSA1MDQ7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNT1VTRV9FWElURUQgPSA1MDU7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNT1VTRV9EUkFHR0VEID0gNTA2OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTU9VU0VfV0hFRUwgPSA1MDc7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBOT0JVVFRPTiA9IDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBCVVRUT04xID0gMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEJVVFRPTjIgPSAyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQlVUVE9OMyA9IDM7Ci0KLSAgICBwcml2YXRlIGJvb2xlYW4gcG9wdXBUcmlnZ2VyOwotICAgIHByaXZhdGUgaW50IGNsaWNrQ291bnQ7Ci0gICAgcHJpdmF0ZSBpbnQgYnV0dG9uOwotICAgIHByaXZhdGUgaW50IHg7Ci0gICAgcHJpdmF0ZSBpbnQgeTsKLQotICAgIHB1YmxpYyBzdGF0aWMgU3RyaW5nIGdldE1vdXNlTW9kaWZpZXJzVGV4dChpbnQgbW9kaWZpZXJzKSB7Ci0gICAgICAgIGZpbmFsIFN0cmluZ0J1ZmZlciB0ZXh0ID0gbmV3IFN0cmluZ0J1ZmZlcigpOwotCi0gICAgICAgIGlmICgobW9kaWZpZXJzICYgTUVUQV9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICB0ZXh0LmFwcGVuZChUb29sa2l0LmdldFByb3BlcnR5KCJBV1QubWV0YSIsICJNZXRhIikpLmFwcGVuZCgiKyIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzICYgU0hJRlRfTUFTSykgIT0gMCkgewotICAgICAgICAgICAgdGV4dC5hcHBlbmQoVG9vbGtpdC5nZXRQcm9wZXJ0eSgiQVdULnNoaWZ0IiwgIlNoaWZ0IikpLmFwcGVuZCgiKyIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzICYgQ1RSTF9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICB0ZXh0LmFwcGVuZChUb29sa2l0LmdldFByb3BlcnR5KCJBV1QuY29udHJvbCIsICJDdHJsIikpLmFwcGVuZCgiKyIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzICYgQUxUX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIHRleHQuYXBwZW5kKFRvb2xraXQuZ2V0UHJvcGVydHkoIkFXVC5hbHQiLCAiQWx0IikpLmFwcGVuZCgiKyIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzICYgQUxUX0dSQVBIX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIHRleHQuYXBwZW5kKFRvb2xraXQuZ2V0UHJvcGVydHkoIkFXVC5hbHRHcmFwaCIsICJBbHQgR3JhcGgiKSkuYXBwZW5kKCIrIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBCVVRUT04xX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIHRleHQuYXBwZW5kKFRvb2xraXQuZ2V0UHJvcGVydHkoIkFXVC5idXR0b24xIiwgIkJ1dHRvbjEiKSkuYXBwZW5kKCIrIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBCVVRUT04yX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIHRleHQuYXBwZW5kKFRvb2xraXQuZ2V0UHJvcGVydHkoIkFXVC5idXR0b24yIiwgIkJ1dHRvbjIiKSkuYXBwZW5kKCIrIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnMgJiBCVVRUT04zX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIHRleHQuYXBwZW5kKFRvb2xraXQuZ2V0UHJvcGVydHkoIkFXVC5idXR0b24zIiwgIkJ1dHRvbjMiKSkuYXBwZW5kKCIrIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gdGV4dC5sZW5ndGgoKSA9PSAwID8gdGV4dC50b1N0cmluZygpIDogdGV4dC5zdWJzdHJpbmcoMCwgdGV4dAotICAgICAgICAgICAgICAgIC5sZW5ndGgoKSAtIDEpOwotICAgIH0KLQotICAgIHN0YXRpYyBTdHJpbmcgYWRkTW91c2VNb2RpZmllcnNFeFRleHQoU3RyaW5nIHRleHQsIGludCBtb2RpZmllcnNFeCkgewotICAgICAgICBpZiAoKG1vZGlmaWVyc0V4ICYgSW5wdXRFdmVudC5CVVRUT04xX0RPV05fTUFTSykgIT0gMCkgewotICAgICAgICAgICAgdGV4dCArPSAoKHRleHQubGVuZ3RoKCkgPiAwKSA/ICIrIiA6ICIiKSArIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgICAgICAgICBUb29sa2l0LmdldFByb3BlcnR5KCJBV1QuYnV0dG9uMSIsICJCdXR0b24xIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICB9Ci0gICAgICAgIGlmICgobW9kaWZpZXJzRXggJiBJbnB1dEV2ZW50LkJVVFRPTjJfRE9XTl9NQVNLKSAhPSAwKSB7Ci0gICAgICAgICAgICB0ZXh0ICs9ICgodGV4dC5sZW5ndGgoKSA+IDApID8gIisiIDogIiIpICsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgICAgIFRvb2xraXQuZ2V0UHJvcGVydHkoIkFXVC5idXR0b24yIiwgIkJ1dHRvbjIiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKChtb2RpZmllcnNFeCAmIElucHV0RXZlbnQuQlVUVE9OM19ET1dOX01BU0spICE9IDApIHsKLSAgICAgICAgICAgIHRleHQgKz0gKCh0ZXh0Lmxlbmd0aCgpID4gMCkgPyAiKyIgOiAiIikgKyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgICAgICAgICAgICAgVG9vbGtpdC5nZXRQcm9wZXJ0eSgiQVdULmJ1dHRvbjMiLCAiQnV0dG9uMyIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiB0ZXh0OwotICAgIH0KLQotICAgIHB1YmxpYyBNb3VzZUV2ZW50KENvbXBvbmVudCBzb3VyY2UsIGludCBpZCwgbG9uZyB3aGVuLAotICAgICAgICAgICAgICAgICAgICAgIGludCBtb2RpZmllcnMsIGludCB4LCBpbnQgeSwKLSAgICAgICAgICAgICAgICAgICAgICBpbnQgY2xpY2tDb3VudCwgYm9vbGVhbiBwb3B1cFRyaWdnZXIpIHsKLSAgICAgICAgdGhpcyhzb3VyY2UsIGlkLCB3aGVuLCBtb2RpZmllcnMsIHgsIHksCi0gICAgICAgICAgICAgY2xpY2tDb3VudCwgcG9wdXBUcmlnZ2VyLCBOT0JVVFRPTik7Ci0gICAgfQotCi0gICAgcHVibGljIE1vdXNlRXZlbnQoQ29tcG9uZW50IHNvdXJjZSwgaW50IGlkLCBsb25nIHdoZW4sCi0gICAgICAgICAgICAgICAgICAgICAgaW50IG1vZGlmaWVycywgaW50IHgsIGludCB5LAotICAgICAgICAgICAgICAgICAgICAgIGludCBjbGlja0NvdW50LCBib29sZWFuIHBvcHVwVHJpZ2dlciwgaW50IGJ1dHRvbikgewotICAgICAgICBzdXBlcihzb3VyY2UsIGlkLCB3aGVuLCBtb2RpZmllcnMpOwotCi0gICAgICAgIGlmICgoYnV0dG9uICE9IE5PQlVUVE9OKSAmJiAoYnV0dG9uICE9IEJVVFRPTjEpICYmCi0gICAgICAgICAgICAgICAgKGJ1dHRvbiAhPSBCVVRUT04yKSAmJiAoYnV0dG9uICE9IEJVVFRPTjMpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMThCPUludmFsaWQgYnV0dG9uIHZhbHVlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE4QiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5wb3B1cFRyaWdnZXIgPSBwb3B1cFRyaWdnZXI7Ci0gICAgICAgIHRoaXMuY2xpY2tDb3VudCA9IGNsaWNrQ291bnQ7Ci0gICAgICAgIHRoaXMuYnV0dG9uID0gYnV0dG9uOwotICAgICAgICB0aGlzLnggPSB4OwotICAgICAgICB0aGlzLnkgPSB5OwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0QnV0dG9uKCkgewotICAgICAgICByZXR1cm4gYnV0dG9uOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0Q2xpY2tDb3VudCgpIHsKLSAgICAgICAgcmV0dXJuIGNsaWNrQ291bnQ7Ci0gICAgfQotCi0gICAgcHVibGljIFBvaW50IGdldFBvaW50KCkgewotICAgICAgICByZXR1cm4gbmV3IFBvaW50KHgsIHkpOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0WCgpIHsKLSAgICAgICAgcmV0dXJuIHg7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRZKCkgewotICAgICAgICByZXR1cm4geTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1BvcHVwVHJpZ2dlcigpIHsKLSAgICAgICAgcmV0dXJuIHBvcHVwVHJpZ2dlcjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB0cmFuc2xhdGVQb2ludChpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgdGhpcy54ICs9IHg7Ci0gICAgICAgIHRoaXMueSArPSB5OwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgcGFyYW1TdHJpbmcoKSB7Ci0gICAgICAgIC8qIFRoZSBmb3JtYXQgaXMgYmFzZWQgb24gMS41IHJlbGVhc2UgYmVoYXZpb3IgCi0gICAgICAgICAqIHdoaWNoIGNhbiBiZSByZXZlYWxlZCBieSB0aGUgZm9sbG93aW5nIGNvZGU6Ci0gICAgICAgICAqIAotICAgICAgICAgKiBNb3VzZUV2ZW50IGUgPSBuZXcgTW91c2VFdmVudChuZXcgQ29tcG9uZW50KCl7fSwgCi0gICAgICAgICAqICAgICAgICAgIE1vdXNlRXZlbnQuTU9VU0VfUFJFU1NFRCwgMCwgCi0gICAgICAgICAqICAgICAgICAgIE1vdXNlRXZlbnQuQlVUVE9OMV9ET1dOX01BU0t8TW91c2VFdmVudC5DVFJMX0RPV05fTUFTSywKLSAgICAgICAgICogICAgICAgICAgMTAsIDIwLCAxLCBmYWxzZSwgTW91c2VFdmVudC5CVVRUT04xKTsKLSAgICAgICAgICogU3lzdGVtLm91dC5wcmludGxuKGUpOwotICAgICAgICAgKi8KLQotICAgICAgICBTdHJpbmcgaWRTdHJpbmcgPSBudWxsOwotICAgICAgICBTdHJpbmcgcGFyYW1TdHJpbmcgPSBudWxsOwotCi0gICAgICAgIHN3aXRjaCAoaWQpIHsKLSAgICAgICAgY2FzZSBNT1VTRV9NT1ZFRDoKLSAgICAgICAgICAgIGlkU3RyaW5nID0gIk1PVVNFX01PVkVEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgTU9VU0VfQ0xJQ0tFRDoKLSAgICAgICAgICAgIGlkU3RyaW5nID0gIk1PVVNFX0NMSUNLRUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBNT1VTRV9QUkVTU0VEOgotICAgICAgICAgICAgaWRTdHJpbmcgPSAiTU9VU0VfUFJFU1NFRCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIE1PVVNFX1JFTEVBU0VEOgotICAgICAgICAgICAgaWRTdHJpbmcgPSAiTU9VU0VfUkVMRUFTRUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBNT1VTRV9EUkFHR0VEOgotICAgICAgICAgICAgaWRTdHJpbmcgPSAiTU9VU0VfRFJBR0dFRCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIE1PVVNFX0VOVEVSRUQ6Ci0gICAgICAgICAgICBpZFN0cmluZyA9ICJNT1VTRV9FTlRFUkVEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgTU9VU0VfRVhJVEVEOgotICAgICAgICAgICAgaWRTdHJpbmcgPSAiTU9VU0VfRVhJVEVEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgTU9VU0VfV0hFRUw6Ci0gICAgICAgICAgICBpZFN0cmluZyA9ICJNT1VTRV9XSEVFTCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgaWRTdHJpbmcgPSAidW5rbm93biB0eXBlIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcGFyYW1TdHJpbmcgPSBpZFN0cmluZyArICIsKCIgKyBnZXRYKCkgKyAiLCIgKyBnZXRZKCkgKyAiKSIgKyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgICAgICAgICAgICAgICIsYnV0dG9uPSIgKyBidXR0b247IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgaWYgKGdldE1vZGlmaWVyc0V4KCkgPiAwKSB7Ci0gICAgICAgICAgICBwYXJhbVN0cmluZyArPSAKLSAgICAgICAgICAgICAgICAgICAgIixtb2RpZmllcnM9IiArIGdldE1vZGlmaWVyc0V4VGV4dChnZXRNb2RpZmllcnNFeCgpKSArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgIixleHRNb2RpZmllcnM9IiArIGdldE1vZGlmaWVyc0V4VGV4dChnZXRNb2RpZmllcnNFeCgpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHBhcmFtU3RyaW5nICs9ICIsY2xpY2tDb3VudD0iICsgZ2V0Q2xpY2tDb3VudCgpOyAvLyROT04tTkxTLTEkCi0KLSAgICAgICAgcmV0dXJuIHBhcmFtU3RyaW5nOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L01vdXNlTGlzdGVuZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZUxpc3RlbmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDk1ODc5YjkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L01vdXNlTGlzdGVuZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBNb3VzZUxpc3RlbmVyIGV4dGVuZHMgRXZlbnRMaXN0ZW5lciB7Ci0KLSAgICBwdWJsaWMgdm9pZCBtb3VzZUNsaWNrZWQoTW91c2VFdmVudCBlKTsKLQotICAgIHB1YmxpYyB2b2lkIG1vdXNlRW50ZXJlZChNb3VzZUV2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQgbW91c2VFeGl0ZWQoTW91c2VFdmVudCBlKTsKLQotICAgIHB1YmxpYyB2b2lkIG1vdXNlUHJlc3NlZChNb3VzZUV2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQgbW91c2VSZWxlYXNlZChNb3VzZUV2ZW50IGUpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvTW91c2VNb3Rpb25BZGFwdGVyLmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvTW91c2VNb3Rpb25BZGFwdGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDFlY2QwZDUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L01vdXNlTW90aW9uQWRhcHRlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDAgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIE1vdXNlTW90aW9uQWRhcHRlciBpbXBsZW1lbnRzIE1vdXNlTW90aW9uTGlzdGVuZXIgewotCi0gICAgcHVibGljIE1vdXNlTW90aW9uQWRhcHRlcigpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBtb3VzZURyYWdnZWQoTW91c2VFdmVudCBlKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgbW91c2VNb3ZlZChNb3VzZUV2ZW50IGUpIHsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZU1vdGlvbkxpc3RlbmVyLmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvTW91c2VNb3Rpb25MaXN0ZW5lci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlMTMxM2MzLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZU1vdGlvbkxpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgTW91c2VNb3Rpb25MaXN0ZW5lciBleHRlbmRzIEV2ZW50TGlzdGVuZXIgewotCi0gICAgcHVibGljIHZvaWQgbW91c2VEcmFnZ2VkKE1vdXNlRXZlbnQgZSk7Ci0KLSAgICBwdWJsaWMgdm9pZCBtb3VzZU1vdmVkKE1vdXNlRXZlbnQgZSk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZVdoZWVsRXZlbnQuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZVdoZWVsRXZlbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTNlZDQyNC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvTW91c2VXaGVlbEV2ZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMDMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotaW1wb3J0IGphdmEuYXd0LkNvbXBvbmVudDsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIE1vdXNlV2hlZWxFdmVudCBleHRlbmRzIE1vdXNlRXZlbnQgewotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gLTkxODc0MTM1ODE5OTM1NjM5MjlMOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0hFRUxfVU5JVF9TQ1JPTEwgPSAwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0hFRUxfQkxPQ0tfU0NST0xMID0gMTsKLQotICAgIHByaXZhdGUgaW50IHdoZWVsUm90YXRpb247Ci0gICAgcHJpdmF0ZSBpbnQgc2Nyb2xsQW1vdW50OwotICAgIHByaXZhdGUgaW50IHNjcm9sbFR5cGU7Ci0KLSAgICBwdWJsaWMgTW91c2VXaGVlbEV2ZW50KENvbXBvbmVudCBzb3VyY2UsIGludCBpZCwgbG9uZyB3aGVuLCBpbnQgbW9kaWZpZXJzLAotICAgICAgICAgICAgaW50IHgsIGludCB5LCBpbnQgY2xpY2tDb3VudCwgYm9vbGVhbiBwb3B1cFRyaWdnZXIsIGludCBzY3JvbGxUeXBlLAotICAgICAgICAgICAgaW50IHNjcm9sbEFtb3VudCwgaW50IHdoZWVsUm90YXRpb24pIHsKLSAgICAgICAgc3VwZXIoc291cmNlLCBpZCwgd2hlbiwgbW9kaWZpZXJzLCB4LCB5LCBjbGlja0NvdW50LCBwb3B1cFRyaWdnZXIpOwotCi0gICAgICAgIHRoaXMuc2Nyb2xsVHlwZSA9IHNjcm9sbFR5cGU7Ci0gICAgICAgIHRoaXMuc2Nyb2xsQW1vdW50ID0gc2Nyb2xsQW1vdW50OwotICAgICAgICB0aGlzLndoZWVsUm90YXRpb24gPSB3aGVlbFJvdGF0aW9uOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0U2Nyb2xsQW1vdW50KCkgewotICAgICAgICByZXR1cm4gc2Nyb2xsQW1vdW50OwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0U2Nyb2xsVHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIHNjcm9sbFR5cGU7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRXaGVlbFJvdGF0aW9uKCkgewotICAgICAgICByZXR1cm4gd2hlZWxSb3RhdGlvbjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFVuaXRzVG9TY3JvbGwoKSB7Ci0gICAgICAgIHJldHVybiAoc2Nyb2xsQW1vdW50ICogd2hlZWxSb3RhdGlvbik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyBwYXJhbVN0cmluZygpIHsKLSAgICAgICAgLyogVGhlIGZvcm1hdCBpcyBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvciAKLSAgICAgICAgICogd2hpY2ggY2FuIGJlIHJldmVhbGVkIGJ5IHRoZSBmb2xsb3dpbmcgY29kZToKLSAgICAgICAgICogCi0gICAgICAgICAqIE1vdXNlV2hlZWxFdmVudCBlID0gbmV3IE1vdXNlV2hlZWxFdmVudChuZXcgQ29tcG9uZW50KCl7fSwgCi0gICAgICAgICAqICAgICAgICAgIE1vdXNlV2hlZWxFdmVudC5NT1VTRV9XSEVFTCwgMCwgCi0gICAgICAgICAqICAgICAgICAgIE1vdXNlRXZlbnQuQlVUVE9OMV9ET1dOX01BU0t8TW91c2VFdmVudC5DVFJMX0RPV05fTUFTSywKLSAgICAgICAgICogICAgICAgICAgMTAsIDIwLCAxLCBmYWxzZSwgTW91c2VXaGVlbEV2ZW50LldIRUVMX1VOSVRfU0NST0xMLAotICAgICAgICAgKiAgICAgICAgICAxLCAzKTsKLSAgICAgICAgICogU3lzdGVtLm91dC5wcmludGxuKGUpOwotICAgICAgICAgKi8KLQotICAgICAgICBTdHJpbmcgcGFyYW1TdHJpbmcgPSBzdXBlci5wYXJhbVN0cmluZygpOwotICAgICAgICBTdHJpbmcgdHlwZVN0cmluZyA9IG51bGw7Ci0KLSAgICAgICAgc3dpdGNoIChzY3JvbGxUeXBlKSB7Ci0gICAgICAgIGNhc2UgV0hFRUxfVU5JVF9TQ1JPTEw6Ci0gICAgICAgICAgICB0eXBlU3RyaW5nID0gIldIRUVMX1VOSVRfU0NST0xMIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgV0hFRUxfQkxPQ0tfU0NST0xMOgotICAgICAgICAgICAgdHlwZVN0cmluZyA9ICJXSEVFTF9CTE9DS19TQ1JPTEwiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHR5cGVTdHJpbmcgPSAidW5rbm93biB0eXBlIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcGFyYW1TdHJpbmcgKz0gIixzY3JvbGxUeXBlPSIgKyB0eXBlU3RyaW5nICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICIsc2Nyb2xsQW1vdW50PSIgKyBzY3JvbGxBbW91bnQgKyAgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICIsd2hlZWxSb3RhdGlvbj0iICsgd2hlZWxSb3RhdGlvbjsgLy8kTk9OLU5MUy0xJAotCi0gICAgICAgIHJldHVybiBwYXJhbVN0cmluZzsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZVdoZWVsTGlzdGVuZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9Nb3VzZVdoZWVsTGlzdGVuZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMmQ2YTk4Mi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvTW91c2VXaGVlbExpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgTW91c2VXaGVlbExpc3RlbmVyIGV4dGVuZHMgRXZlbnRMaXN0ZW5lciB7Ci0KLSAgICBwdWJsaWMgdm9pZCBtb3VzZVdoZWVsTW92ZWQoTW91c2VXaGVlbEV2ZW50IGUpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvUGFpbnRFdmVudC5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L1BhaW50RXZlbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjJhYzA5MC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvUGFpbnRFdmVudC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsODYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotaW1wb3J0IGphdmEuYXd0LkNvbXBvbmVudDsKLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBQYWludEV2ZW50IGV4dGVuZHMgQ29tcG9uZW50RXZlbnQgewotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gMTI2NzQ5MjAyNjQzMzMzNzU5M0w7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBQQUlOVF9GSVJTVCA9IDgwMDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFBBSU5UX0xBU1QgPSA4MDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBQQUlOVCA9IDgwMDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFVQREFURSA9IDgwMTsKLQotICAgIHByaXZhdGUgUmVjdGFuZ2xlIHVwZGF0ZVJlY3Q7Ci0KLSAgICBwdWJsaWMgUGFpbnRFdmVudChDb21wb25lbnQgc291cmNlLCBpbnQgaWQsIFJlY3RhbmdsZSB1cGRhdGVSZWN0KSB7Ci0gICAgICAgIHN1cGVyKHNvdXJjZSwgaWQpOwotCi0gICAgICAgIHRoaXMudXBkYXRlUmVjdCA9IHVwZGF0ZVJlY3Q7Ci0gICAgfQotCi0gICAgcHVibGljIFJlY3RhbmdsZSBnZXRVcGRhdGVSZWN0KCkgewotICAgICAgICByZXR1cm4gdXBkYXRlUmVjdDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRVcGRhdGVSZWN0KFJlY3RhbmdsZSB1cGRhdGVSZWN0KSB7Ci0gICAgICAgIHRoaXMudXBkYXRlUmVjdCA9IHVwZGF0ZVJlY3Q7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyBwYXJhbVN0cmluZygpIHsKLSAgICAgICAgLyogVGhlIGZvcm1hdCBpcyBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvciAKLSAgICAgICAgICogd2hpY2ggY2FuIGJlIHJldmVhbGVkIGJ5IHRoZSBmb2xsb3dpbmcgY29kZToKLSAgICAgICAgICogCi0gICAgICAgICAqIFBhaW50RXZlbnQgZSA9IG5ldyBQYWludEV2ZW50KG5ldyBDb21wb25lbnQoKXt9LCAKLSAgICAgICAgICogICAgICAgICAgUGFpbnRFdmVudC5QQUlOVCwgbmV3IFJlY3RhbmdsZSgwLCAwLCAxMCwgMjApKTsgCi0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihlKTsKLSAgICAgICAgICovCi0KLSAgICAgICAgU3RyaW5nIHR5cGVTdHJpbmcgPSBudWxsOwotCi0gICAgICAgIHN3aXRjaCAoaWQpIHsKLSAgICAgICAgY2FzZSBQQUlOVDoKLSAgICAgICAgICAgIHR5cGVTdHJpbmcgPSAiUEFJTlQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBVUERBVEU6Ci0gICAgICAgICAgICB0eXBlU3RyaW5nID0gIlVQREFURSI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgdHlwZVN0cmluZyA9ICJ1bmtub3duIHR5cGUiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gdHlwZVN0cmluZyArICIsdXBkYXRlUmVjdD0iICsgdXBkYXRlUmVjdDsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L1RleHRFdmVudC5qYXZhIGIvYXd0L2phdmEvYXd0L2V2ZW50L1RleHRFdmVudC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyYTY5MGFkLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9UZXh0RXZlbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDU5ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5BV1RFdmVudDsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIFRleHRFdmVudCBleHRlbmRzIEFXVEV2ZW50IHsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IDYyNjk5MDIyOTEyNTA5NDExNzlMOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEVYVF9GSVJTVCA9IDkwMDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRFWFRfTEFTVCA9IDkwMDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRFWFRfVkFMVUVfQ0hBTkdFRCA9IDkwMDsKLQotICAgIHB1YmxpYyBUZXh0RXZlbnQoT2JqZWN0IHNyYywgaW50IGlkKSB7Ci0gICAgICAgIHN1cGVyKHNyYywgaWQpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgcGFyYW1TdHJpbmcoKSB7Ci0gICAgICAgIC8qIFRoZSBmb3JtYXQgaXMgYmFzZWQgb24gMS41IHJlbGVhc2UgYmVoYXZpb3IgCi0gICAgICAgICAqIHdoaWNoIGNhbiBiZSByZXZlYWxlZCBieSB0aGUgZm9sbG93aW5nIGNvZGU6Ci0gICAgICAgICAqIAotICAgICAgICAgKiBUZXh0RXZlbnQgZSA9IG5ldyBUZXh0RXZlbnQobmV3IENvbXBvbmVudCgpe30sIAotICAgICAgICAgKiAgICAgICAgICBUZXh0RXZlbnQuVEVYVF9WQUxVRV9DSEFOR0VEKTsgCi0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihlKTsKLSAgICAgICAgICovCi0KLSAgICAgICAgcmV0dXJuIChpZCA9PSBURVhUX1ZBTFVFX0NIQU5HRUQpID8gCi0gICAgICAgICAgICAgICAgIlRFWFRfVkFMVUVfQ0hBTkdFRCIgOiAidW5rbm93biB0eXBlIjsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvVGV4dExpc3RlbmVyLmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvVGV4dExpc3RlbmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDA1NzU3YzQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L1RleHRMaXN0ZW5lci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWNoYWVsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5ldmVudDsKLQotaW1wb3J0IGphdmEudXRpbC5FdmVudExpc3RlbmVyOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgbm90IHN1cHBvcnRlZCBpbiBBbmRyb2lkIDEuMC4gSXQgaXMgbWVyZWx5IHByb3ZpZGVkIHRvIG1haW50YWluCi0gKiBpbnRlcmZhY2UgY29tcGF0aWJpbGl0eSB3aXRoIGRlc2t0b3AgSmF2YSBpbXBsZW1lbnRhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIFRleHRMaXN0ZW5lciBleHRlbmRzIEV2ZW50TGlzdGVuZXIgewotCi0gICAgcHVibGljIHZvaWQgdGV4dFZhbHVlQ2hhbmdlZChUZXh0RXZlbnQgZSk7Ci0KLX0KLQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L1dpbmRvd0FkYXB0ZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9XaW5kb3dBZGFwdGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDk3MGFhOGQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L1dpbmRvd0FkYXB0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDY0ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBXaW5kb3dBZGFwdGVyIGltcGxlbWVudHMgV2luZG93TGlzdGVuZXIsIFdpbmRvd1N0YXRlTGlzdGVuZXIsIFdpbmRvd0ZvY3VzTGlzdGVuZXIgewotCi0gICAgcHVibGljIFdpbmRvd0FkYXB0ZXIoKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgd2luZG93QWN0aXZhdGVkKFdpbmRvd0V2ZW50IGUpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3aW5kb3dDbG9zZWQoV2luZG93RXZlbnQgZSkgewotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdpbmRvd0Nsb3NpbmcoV2luZG93RXZlbnQgZSkgewotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdpbmRvd0RlYWN0aXZhdGVkKFdpbmRvd0V2ZW50IGUpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3aW5kb3dEZWljb25pZmllZChXaW5kb3dFdmVudCBlKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgd2luZG93R2FpbmVkRm9jdXMoV2luZG93RXZlbnQgZSkgewotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdpbmRvd0ljb25pZmllZChXaW5kb3dFdmVudCBlKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgd2luZG93TG9zdEZvY3VzKFdpbmRvd0V2ZW50IGUpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3aW5kb3dPcGVuZWQoV2luZG93RXZlbnQgZSkgewotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdpbmRvd1N0YXRlQ2hhbmdlZChXaW5kb3dFdmVudCBlKSB7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZXZlbnQvV2luZG93RXZlbnQuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9XaW5kb3dFdmVudC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0NzRkMmFjLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9XaW5kb3dFdmVudC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTY4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLQotLy8/Pz9BV1QKLS8vaW1wb3J0IGphdmEuYXd0LldpbmRvdzsKLS8vaW1wb3J0IGphdmEuYXd0LkZyYW1lOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgbm90IHN1cHBvcnRlZCBpbiBBbmRyb2lkIDEuMC4gSXQgaXMgbWVyZWx5IHByb3ZpZGVkIHRvIG1haW50YWluCi0gKiBpbnRlcmZhY2UgY29tcGF0aWJpbGl0eSB3aXRoIGRlc2t0b3AgSmF2YSBpbXBsZW1lbnRhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgV2luZG93RXZlbnQgZXh0ZW5kcyBDb21wb25lbnRFdmVudCB7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtMTU2Nzk1OTEzMzE0NzkxMjEyN0w7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXSU5ET1dfRklSU1QgPSAyMDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXSU5ET1dfT1BFTkVEID0gMjAwOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lORE9XX0NMT1NJTkcgPSAyMDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXSU5ET1dfQ0xPU0VEID0gMjAyOwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lORE9XX0lDT05JRklFRCA9IDIwMzsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFdJTkRPV19ERUlDT05JRklFRCA9IDIwNDsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFdJTkRPV19BQ1RJVkFURUQgPSAyMDU7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXSU5ET1dfREVBQ1RJVkFURUQgPSAyMDY7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXSU5ET1dfR0FJTkVEX0ZPQ1VTID0gMjA3OwotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lORE9XX0xPU1RfRk9DVVMgPSAyMDg7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBXSU5ET1dfU1RBVEVfQ0hBTkdFRCA9IDIwOTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFdJTkRPV19MQVNUID0gMjA5OwotCi0gICAgLy8/Pz9BV1Q6IHByaXZhdGUgV2luZG93IG9wcG9zaXRlV2luZG93OwotICAgIHByaXZhdGUgaW50IG9sZFN0YXRlOwotICAgIHByaXZhdGUgaW50IG5ld1N0YXRlOwotCi0gICAgLy8/Pz9BV1QKLSAgICAvKgotICAgIHB1YmxpYyBXaW5kb3dFdmVudChXaW5kb3cgc291cmNlLCBpbnQgaWQpIHsKLSAgICAgICAgdGhpcyhzb3VyY2UsIGlkLCBudWxsKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgV2luZG93RXZlbnQoV2luZG93IHNvdXJjZSwgaW50IGlkLCBXaW5kb3cgb3Bwb3NpdGUpIHsKLSAgICAgICAgdGhpcyhzb3VyY2UsIGlkLCBvcHBvc2l0ZSwgRnJhbWUuTk9STUFMLCBGcmFtZS5OT1JNQUwpOwotICAgIH0KLQotICAgIHB1YmxpYyBXaW5kb3dFdmVudChXaW5kb3cgc291cmNlLCBpbnQgaWQsIGludCBvbGRTdGF0ZSwgaW50IG5ld1N0YXRlKSB7Ci0gICAgICAgIHRoaXMoc291cmNlLCBpZCwgbnVsbCwgb2xkU3RhdGUsIG5ld1N0YXRlKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgV2luZG93RXZlbnQoV2luZG93IHNvdXJjZSwgaW50IGlkLCBXaW5kb3cgb3Bwb3NpdGUsIAotICAgICAgICAgICAgICAgICAgICAgICBpbnQgb2xkU3RhdGUsIGludCBuZXdTdGF0ZSkgewotICAgICAgICBzdXBlcihzb3VyY2UsIGlkKTsKLQotICAgICAgICBvcHBvc2l0ZVdpbmRvdyA9IG9wcG9zaXRlOwotICAgICAgICB0aGlzLm9sZFN0YXRlID0gb2xkU3RhdGU7Ci0gICAgICAgIHRoaXMubmV3U3RhdGUgPSBuZXdTdGF0ZTsKLSAgICB9Ci0gICAgKi8KLSAgICAvLz8/P0FXVDogRmFrZSBjb25zdHJ1Y3RvcgotICAgIHB1YmxpYyBXaW5kb3dFdmVudCgpIHsKLSAgICAgICAgc3VwZXIobnVsbCwgMCk7Ci0gICAgfQotICAgIAotICAgIHB1YmxpYyBpbnQgZ2V0TmV3U3RhdGUoKSB7Ci0gICAgICAgIHJldHVybiBuZXdTdGF0ZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldE9sZFN0YXRlKCkgewotICAgICAgICByZXR1cm4gb2xkU3RhdGU7Ci0gICAgfQotCi0gICAgLy8/Pz9BV1QKLSAgICAvKgotICAgIHB1YmxpYyBXaW5kb3cgZ2V0T3Bwb3NpdGVXaW5kb3coKSB7Ci0gICAgICAgIHJldHVybiBvcHBvc2l0ZVdpbmRvdzsKLSAgICB9Ci0KLSAgICBwdWJsaWMgV2luZG93IGdldFdpbmRvdygpIHsKLSAgICAgICAgcmV0dXJuIChXaW5kb3cpIHNvdXJjZTsKLSAgICB9Ci0gICAgKi8KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgcGFyYW1TdHJpbmcoKSB7Ci0gICAgICAgIC8qIFRoZSBmb3JtYXQgaXMgYmFzZWQgb24gMS41IHJlbGVhc2UgYmVoYXZpb3IgCi0gICAgICAgICAqIHdoaWNoIGNhbiBiZSByZXZlYWxlZCBieSB0aGUgZm9sbG93aW5nIGNvZGU6Ci0gICAgICAgICAqIAotICAgICAgICAgKiBXaW5kb3dFdmVudCBlID0gbmV3IFdpbmRvd0V2ZW50KG5ldyBGcmFtZSgpLCAKLSAgICAgICAgICogICAgICAgICAgV2luZG93RXZlbnQuV0lORE9XX09QRU5FRCk7IAotICAgICAgICAgKiBTeXN0ZW0ub3V0LnByaW50bG4oZSk7Ci0gICAgICAgICAqLwotCi0gICAgICAgIFN0cmluZyB0eXBlU3RyaW5nID0gbnVsbDsKLQotICAgICAgICBzd2l0Y2ggKGlkKSB7Ci0gICAgICAgIGNhc2UgV0lORE9XX09QRU5FRDoKLSAgICAgICAgICAgIHR5cGVTdHJpbmcgPSAiV0lORE9XX09QRU5FRCI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFdJTkRPV19DTE9TSU5HOgotICAgICAgICAgICAgdHlwZVN0cmluZyA9ICJXSU5ET1dfQ0xPU0lORyI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFdJTkRPV19DTE9TRUQ6Ci0gICAgICAgICAgICB0eXBlU3RyaW5nID0gIldJTkRPV19DTE9TRUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBXSU5ET1dfSUNPTklGSUVEOgotICAgICAgICAgICAgdHlwZVN0cmluZyA9ICJXSU5ET1dfSUNPTklGSUVEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgV0lORE9XX0RFSUNPTklGSUVEOgotICAgICAgICAgICAgdHlwZVN0cmluZyA9ICJXSU5ET1dfREVJQ09OSUZJRUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBXSU5ET1dfQUNUSVZBVEVEOgotICAgICAgICAgICAgdHlwZVN0cmluZyA9ICJXSU5ET1dfQUNUSVZBVEVEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgV0lORE9XX0RFQUNUSVZBVEVEOgotICAgICAgICAgICAgdHlwZVN0cmluZyA9ICJXSU5ET1dfREVBQ1RJVkFURUQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBXSU5ET1dfR0FJTkVEX0ZPQ1VTOgotICAgICAgICAgICAgdHlwZVN0cmluZyA9ICJXSU5ET1dfR0FJTkVEX0ZPQ1VTIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgV0lORE9XX0xPU1RfRk9DVVM6Ci0gICAgICAgICAgICB0eXBlU3RyaW5nID0gIldJTkRPV19MT1NUX0ZPQ1VTIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgV0lORE9XX1NUQVRFX0NIQU5HRUQ6Ci0gICAgICAgICAgICB0eXBlU3RyaW5nID0gIldJTkRPV19TVEFURV9DSEFOR0VEIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICB0eXBlU3RyaW5nID0gInVua25vd24gdHlwZSI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIC8vPz8/QVdUCi0gICAgICAgIC8qCi0gICAgICAgIHJldHVybiB0eXBlU3RyaW5nICsgIixvcHBvc2l0ZT0iICsgb3Bwb3NpdGVXaW5kb3cgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgIixvbGRTdGF0ZT0iICsgb2xkU3RhdGUgKyAiLG5ld1N0YXRlPSIgKyBuZXdTdGF0ZTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICovCi0gICAgICAgIHJldHVybiB0eXBlU3RyaW5nOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L1dpbmRvd0ZvY3VzTGlzdGVuZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9ldmVudC9XaW5kb3dGb2N1c0xpc3RlbmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDUyODQ1OWYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2V2ZW50L1dpbmRvd0ZvY3VzTGlzdGVuZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDM3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBXaW5kb3dGb2N1c0xpc3RlbmVyIGV4dGVuZHMgRXZlbnRMaXN0ZW5lciB7Ci0KLSAgICBwdWJsaWMgdm9pZCB3aW5kb3dHYWluZWRGb2N1cyhXaW5kb3dFdmVudCBlKTsKLQotICAgIHB1YmxpYyB2b2lkIHdpbmRvd0xvc3RGb2N1cyhXaW5kb3dFdmVudCBlKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2V2ZW50L1dpbmRvd0xpc3RlbmVyLmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvV2luZG93TGlzdGVuZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzFiZDU0Ny4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZXZlbnQvV2luZG93TGlzdGVuZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBXaW5kb3dMaXN0ZW5lciBleHRlbmRzIEV2ZW50TGlzdGVuZXIgewotCi0gICAgcHVibGljIHZvaWQgd2luZG93QWN0aXZhdGVkKFdpbmRvd0V2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQgd2luZG93Q2xvc2VkKFdpbmRvd0V2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQgd2luZG93Q2xvc2luZyhXaW5kb3dFdmVudCBlKTsKLQotICAgIHB1YmxpYyB2b2lkIHdpbmRvd0RlYWN0aXZhdGVkKFdpbmRvd0V2ZW50IGUpOwotCi0gICAgcHVibGljIHZvaWQgd2luZG93RGVpY29uaWZpZWQoV2luZG93RXZlbnQgZSk7Ci0KLSAgICBwdWJsaWMgdm9pZCB3aW5kb3dJY29uaWZpZWQoV2luZG93RXZlbnQgZSk7Ci0KLSAgICBwdWJsaWMgdm9pZCB3aW5kb3dPcGVuZWQoV2luZG93RXZlbnQgZSk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9ldmVudC9XaW5kb3dTdGF0ZUxpc3RlbmVyLmphdmEgYi9hd3QvamF2YS9hd3QvZXZlbnQvV2luZG93U3RhdGVMaXN0ZW5lci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiYTE0ZDllLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9ldmVudC9XaW5kb3dTdGF0ZUxpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pY2hhZWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmV2ZW50OwotCi1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgV2luZG93U3RhdGVMaXN0ZW5lciBleHRlbmRzIEV2ZW50TGlzdGVuZXIgewotCi0gICAgcHVibGljIHZvaWQgd2luZG93U3RhdGVDaGFuZ2VkKFdpbmRvd0V2ZW50IGUpOwotCi19Ci0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9mb250L0ZvbnRSZW5kZXJDb250ZXh0LmphdmEgYi9hd3QvamF2YS9hd3QvZm9udC9Gb250UmVuZGVyQ29udGV4dC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkN2RlMDBmLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9mb250L0ZvbnRSZW5kZXJDb250ZXh0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNzggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJbHlhIFMuIE9rb21pbgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmZvbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLQotLyoqCi0gKiBUaGUgRm9udFJlbmRlckNvbnRleHQgY2xhc3MgY29udGFpbnMgdGhlIGluZm9ybWF0aW9uIGFib3V0IHRleHQgbWVhc3VyZW1lbnQuCi0gKiBBbnRpLWFsaWFzaW5nIGFuZCBmcmFjdGlvbmFsLW1ldHJpY3MgbW9kZXMgYXJlIGRlZmluZWQgYnkgYW4gYXBwbGljYXRpb24gYW5kCi0gKiBhZmZlY3QgdGhlIHNpemUgb2YgYSBjaGFyYWN0ZXIuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgRm9udFJlbmRlckNvbnRleHQgewotCi0gICAgLy8gQWZmaW5lIHRyYW5zZm9ybSBvZiB0aGlzIG1vZGUKLSAgICAvKioKLSAgICAgKiBUaGUgdHJhbnNmb3JtLgotICAgICAqLwotICAgIHByaXZhdGUgQWZmaW5lVHJhbnNmb3JtIHRyYW5zZm9ybTsKLQotICAgIC8vIElzIHRoZSBhbnRpLWFsaWFzZWQgbW9kZSB1c2VkCi0gICAgLyoqCi0gICAgICogVGhlIGFudGkgYWxpYXNlZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gZkFudGlBbGlhc2VkOwotCi0gICAgLy8gSXMgdGhlIGZyYWN0aW9uYWwgbWV0cmljcyB1c2VkCi0gICAgLyoqCi0gICAgICogVGhlIGZyYWN0aW9uYWwgbWV0cmljcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gZkZyYWN0aW9uYWxNZXRyaWNzOwotCi0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIEFmZmluZVRyYW5zZm9ybSwgYW50aS1hbGlhc2luZyBhbmQgZnJhY3Rpb25hbCBtZXRyaWNzIGZsYWdzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0cmFucwotICAgICAqICAgICAgICAgICAgdGhlIEFmZmluZVRyYW5zZm9ybS4KLSAgICAgKiBAcGFyYW0gYW50aUFsaWFzZWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbnRpLWFsaWFzaW5nIGZsYWcuCi0gICAgICogQHBhcmFtIHVzZXNGcmFjdGlvbmFsTWV0cmljcwotICAgICAqICAgICAgICAgICAgdGhlIGZyYWN0aW9uYWwgbWV0cmljcyBmbGFnLgotICAgICAqLwotICAgIHB1YmxpYyBGb250UmVuZGVyQ29udGV4dChBZmZpbmVUcmFuc2Zvcm0gdHJhbnMsIGJvb2xlYW4gYW50aUFsaWFzZWQsIAotICAgICAgICAgICAgYm9vbGVhbiB1c2VzRnJhY3Rpb25hbE1ldHJpY3MpIHsKLSAgICAgICAgaWYgKHRyYW5zICE9IG51bGwpewotICAgICAgICAgICAgdHJhbnNmb3JtID0gbmV3IEFmZmluZVRyYW5zZm9ybSh0cmFucyk7Ci0gICAgICAgIH0KLSAgICAgICAgZkFudGlBbGlhc2VkID0gYW50aUFsaWFzZWQ7Ci0gICAgICAgIGZGcmFjdGlvbmFsTWV0cmljcyA9IHVzZXNGcmFjdGlvbmFsTWV0cmljczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBGb250UmVuZGVyQ29udGV4dCgpIHsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wYXJlcyB0aGUgc3BlY2lmaWVkIE9iamVjdCB3aXRoIGN1cnJlbnQgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIE9iamVjdCBpcyBlcXVhbCB0byBjdXJyZW50Ci0gICAgICogICAgICAgICBGb250UmVuZGVyQ29udGV4dCBvYmplY3QuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvYmopIHsKLSAgICAgICAgaWYgKG9iaiA9PSB0aGlzKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChvYmogIT0gbnVsbCkgewotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZXF1YWxzKChGb250UmVuZGVyQ29udGV4dCkgb2JqKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKENsYXNzQ2FzdEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRyYW5zZm9ybSB3aGljaCBpcyB1c2VkIGZvciBzY2FsaW5nIHR5cG9ncmFwaGljYWwgcG9pbnRzIHRvCi0gICAgICogcGl4ZWxzIGluIHRoaXMgRm9udFJlbmRlckNvbnRleHQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgQWZmaW5lVHJhbnNmb3JtIHdoaWNoIGlzIHVzZWQgZm9yIHNjYWxpbmcgdHlwb2dyYXBoaWNhbAotICAgICAqICAgICAgICAgcG9pbnRzIHRvIHBpeGVscyBpbiB0aGlzIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0gZ2V0VHJhbnNmb3JtKCkgewotICAgICAgICBpZiAodHJhbnNmb3JtICE9IG51bGwpewotICAgICAgICAgICAgcmV0dXJuIG5ldyBBZmZpbmVUcmFuc2Zvcm0odHJhbnNmb3JtKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbmV3IEFmZmluZVRyYW5zZm9ybSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIHRoZSBzcGVjaWZpZWQgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0IHdpdGggY3VycmVudAotICAgICAqIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb250UmVuZGVyQ29udGV4dCBvYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIEZvbnRSZW5kZXJDb250ZXh0IG9iamVjdCBpcyBlcXVhbCB0bwotICAgICAqICAgICAgICAgY3VycmVudCBGb250UmVuZGVyQ29udGV4dC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoRm9udFJlbmRlckNvbnRleHQgZnJjKSB7Ci0gICAgICAgIGlmICh0aGlzID09IGZyYyl7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChmcmMgPT0gbnVsbCl7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoIWZyYy5nZXRUcmFuc2Zvcm0oKS5lcXVhbHModGhpcy5nZXRUcmFuc2Zvcm0oKSkgJiYKLSAgICAgICAgICAgICFmcmMuaXNBbnRpQWxpYXNlZCgpID09IHRoaXMuZkFudGlBbGlhc2VkICYmCi0gICAgICAgICAgICAhZnJjLnVzZXNGcmFjdGlvbmFsTWV0cmljcygpID09IHRoaXMuZkZyYWN0aW9uYWxNZXRyaWNzKXsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIHRleHQgZnJhY3Rpb25hbCBtZXRyaWNzIGFyZSB1c2VkIGluIHRoaXMKLSAgICAgKiBGb250UmVuZGVyQ29udGV4dC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSB0ZXh0IGZyYWN0aW9uYWwgbWV0cmljcyBhcmUgdXNlZCBpbiB0aGlzCi0gICAgICogICAgICAgICBGb250UmVuZGVyQ29udGV4dCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIHVzZXNGcmFjdGlvbmFsTWV0cmljcygpIHsKLSAgICAgICAgcmV0dXJuIHRoaXMuZkZyYWN0aW9uYWxNZXRyaWNzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiBhbnRpLWFsaWFzaW5nIGlzIHVzZWQgaW4gdGhpcyBGb250UmVuZGVyQ29udGV4dC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGlzIGFudGktYWxpYXNpbmcgaXMgdXNlZCBpbiB0aGlzIEZvbnRSZW5kZXJDb250ZXh0LAotICAgICAqICAgICAgICAgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzQW50aUFsaWFzZWQoKSB7Ci0gICAgICAgIHJldHVybiB0aGlzLmZBbnRpQWxpYXNlZDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGhhc2ggY29kZSBvZiB0aGUgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhhc2ggY29kZSBvZiB0aGUgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgaGFzaENvZGUoKSB7Ci0gICAgICAgIHJldHVybiB0aGlzLmdldFRyYW5zZm9ybSgpLmhhc2hDb2RlKCkgXgotICAgICAgICAgICAgICAgIG5ldyBCb29sZWFuKHRoaXMuZkZyYWN0aW9uYWxNZXRyaWNzKS5oYXNoQ29kZSgpIF4KLSAgICAgICAgICAgICAgICBuZXcgQm9vbGVhbih0aGlzLmZBbnRpQWxpYXNlZCkuaGFzaENvZGUoKTsKLSAgICB9Ci0KLX0KLQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ZvbnQvR2x5cGhKdXN0aWZpY2F0aW9uSW5mby5qYXZhIGIvYXd0L2phdmEvYXd0L2ZvbnQvR2x5cGhKdXN0aWZpY2F0aW9uSW5mby5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiMDNkZTBhLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9mb250L0dseXBoSnVzdGlmaWNhdGlvbkluZm8uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE5NyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5mb250OwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIEdseXBoSnVzdGlmaWNhdGlvbkluZm8gY2xhc3MgcHJvdmlkZXMgaW5mb3JtYXRpb24gYWJvdXQgdGhlIGdseXBoJ3MKLSAqIGp1c3RpZmljYXRpb24gcHJvcGVydGllcy4gVGhlcmUgYXJlIGZvdXIganVzdGlmaWNhdGlvbiBwcm9wZXJ0aWVzOiB3ZWlnaHQsCi0gKiBwcmlvcml0eSwgYWJzb3JiLCBhbmQgbGltaXQuCi0gKiA8cD4KLSAqIFRoZXJlIGFyZSB0d28gc2V0cyBvZiBtZXRyaWNzOiBncm93aW5nIGFuZCBzaHJpbmtpbmcuIEdyb3dpbmcgbWV0cmljcyBhcmUKLSAqIHVzZWQgd2hlbiB0aGUgZ2x5cGhzIGFyZSB0byBiZSBzcHJlYWQgYXBhcnQgdG8gZml0IGEgbGFyZ2VyIHdpZHRoLiBTaHJpbmtpbmcKLSAqIG1ldHJpY3MgYXJlIHVzZWQgd2hlbiB0aGUgZ2x5cGhzIGFyZSB0byBiZSBtb3ZlZCB0b2dldGhlciB0byBmaXQgYSBzbWFsbGVyCi0gKiB3aWR0aC4KLSAqIDwvcD4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBmaW5hbCBjbGFzcyBHbHlwaEp1c3RpZmljYXRpb25JbmZvIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBQUklPUklUWV9LQVNISURBIGluZGljYXRlcyB0aGUgaGlnaGVzdCBqdXN0aWZpY2F0aW9uCi0gICAgICogcHJpb3JpdHkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgUFJJT1JJVFlfS0FTSElEQSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUFJJT1JJVFlfV0hJVEVTUEFDRSBpbmRpY2F0ZXMgdGhlIHNlY29uZCBoaWdoZXN0Ci0gICAgICoganVzdGlmaWNhdGlvbiBwcmlvcml0eS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBQUklPUklUWV9XSElURVNQQUNFID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBQUklPUklUWV9JTlRFUkNIQVIgaW5kaWNhdGVzIHRoZSBzZWNvbmQgbG93ZXN0IGp1c3RpZmljYXRpb24KLSAgICAgKiBwcmlvcml0eS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBQUklPUklUWV9JTlRFUkNIQVIgPSAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFBSSU9SSVRZX05PTkUgaW5kaWNhdGVzIHRoZSBsb3dlc3QganVzdGlmaWNhdGlvbiBwcmlvcml0eS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBQUklPUklUWV9OT05FID0gMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBncm93IGFic29yYiBmbGFnIGluZGljYXRlcyBpZiB0aGlzIGdseXBoIGFic29yYnMgYWxsIGV4dHJhIHNwYWNlIGF0Ci0gICAgICogdGhpcyBhbmQgbG93ZXIgcHJpb3JpdHkgbGV2ZWxzIHdoZW4gaXQgZ3Jvd3MuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGJvb2xlYW4gZ3Jvd0Fic29yYjsKLQotICAgIC8qKgotICAgICAqIFRoZSBncm93IGxlZnQgbGltaXQgdmFsdWUgcmVwcmVzZW50cyB0aGUgbWF4aW11bSB2YWx1ZSBieSB3aGljaCB0aGUgbGVmdAotICAgICAqIHNpZGUgb2YgdGhpcyBnbHlwaCBncm93cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgZmxvYXQgZ3Jvd0xlZnRMaW1pdDsKLQotICAgIC8qKgotICAgICAqIFRoZSBncm93IHJpZ2h0IGxpbWl0IHZhbHVlIHJlcGVzZW50cyB0aGUgbWF4aW11bSB2YWx1ZSBieSB3aGljaCB0aGUgcmlnaHQKLSAgICAgKiBzaWRlIG9mIHRoaXMgZ2x5cGggZ3Jvd3MuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGZsb2F0IGdyb3dSaWdodExpbWl0OwotCi0gICAgLyoqCi0gICAgICogVGhlIGdyb3cgcHJpb3JpdHkgdmFsdWUgcmVwcmVzZW50cyB0aGUgcHJpb3JpdHkgbGV2ZWwgb2YgdGhpcyBnbHlwaCBhcyBpdAotICAgICAqIGlzIGdyb3dpbmcuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGludCBncm93UHJpb3JpdHk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc2hyaW5rIGFic29yYiBmbGVnIGluZGljYXRlcyB0aGlzIGdseXBoIGFic29yYnMgYWxsIHJlbWFpbmluZwotICAgICAqIHNocmlua2FnZSBhdCB0aGlzIGFuZCBsb3dlciBwcmlvcml0eSBsZXZlbHMgYXMgaXQgc2hyaW5rcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgYm9vbGVhbiBzaHJpbmtBYnNvcmI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc2hyaW5rIGxlZnQgbGltaXQgdmFsdWUgcmVwcmVzZW50cyB0aGUgbWF4aW11bSB2YWx1ZSBieSB3aGljaCB0aGUKLSAgICAgKiBsZWZ0IHNpZGUgb2YgdGhpcyBnbHlwaCBzaHJpbmtzLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBmbG9hdCBzaHJpbmtMZWZ0TGltaXQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc2hyaW5rIHJpZ2h0IGxpbWl0IHZhbHVlIHJlcHJlc2VudHMgdGhlIG1heGltdW0gdmFsdWUgYnkgd2hpY2ggdGhlCi0gICAgICogcmlnaHQgc2lkZSBvZiB0aGlzIGdseXBoIHNocmlua3MuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGZsb2F0IHNocmlua1JpZ2h0TGltaXQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc2hyaW5rIHByaW9yaXR5IHJlcHJlc2VudHMgdGhlIGdseXRoJ3MgcHJpb3JpdHkgbGV2ZWwgYXMgaXQgaXMKLSAgICAgKiBzaHJpbmtpbmcuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGludCBzaHJpbmtQcmlvcml0eTsKLQotICAgIC8qKgotICAgICAqIFRoZSB3ZWlnaHQgb2YgdGhlIGdseXBoLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBmbG9hdCB3ZWlnaHQ7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBvYmplY3Qgd2hpY2ggY29udGFpbnMgZ2x5cGgncwotICAgICAqIGp1c3RpZmljYXRpb24gcHJvcGVydGllcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2VpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgd2VpZ2h0IG9mIGdseXBoLgotICAgICAqIEBwYXJhbSBncm93QWJzb3JiCi0gICAgICogICAgICAgICAgICBpbmRpY2F0ZXMgaWYgdGhpcyBnbHlwaCBjb250YWlzIGFsbCBzcGFjZSBhdCB0aGlzIHByaW9yaXR5IGFuZAotICAgICAqICAgICAgICAgICAgbG93ZXIgcHJpb3JpdHkgbGV2ZWxzIHdoZW4gaXQgZ3Jvd3MuCi0gICAgICogQHBhcmFtIGdyb3dQcmlvcml0eQotICAgICAqICAgICAgICAgICAgaW5kaWNhdGVzIHRoZSBwcmlvcml0eSBsZXZlbCBvZiB0aGlzIGdseXBoIHdoZW4gaXQgZ3Jvd3MuCi0gICAgICogQHBhcmFtIGdyb3dMZWZ0TGltaXQKLSAgICAgKiAgICAgICAgICAgIGluZGljYXRlcyB0aGUgbWF4aW11bSB2YWx1ZSBvZiB3aGljaCB0aGUgbGVmdCBzaWRlIG9mIHRoaXMKLSAgICAgKiAgICAgICAgICAgIGdseXBoIGNhbiBncm93LgotICAgICAqIEBwYXJhbSBncm93UmlnaHRMaW1pdAotICAgICAqICAgICAgICAgICAgdGhlIG1heGltdW0gdmFsdWUgb2Ygd2hpY2ggdGhlIHJpZ2h0IHNpZGUgb2YgdGhpcyBnbHlwaCBjYW4KLSAgICAgKiAgICAgICAgICAgIGdyb3cuCi0gICAgICogQHBhcmFtIHNocmlua0Fic29yYgotICAgICAqICAgICAgICAgICAgaW5kaWNhdGVzIGlmIHRoaXMgZ2x5cGggY29udGFpbnMgYWxsIHJlbWFpbmluZyBzaHJpbmthZ2UgYXQKLSAgICAgKiAgICAgICAgICAgIHRoaXMgYW5kIGxvd2VyIHByaW9yaXR5IGxldmVscyB3aGVuIGl0IHNocmlua3MuCi0gICAgICogQHBhcmFtIHNocmlua1ByaW9yaXR5Ci0gICAgICogICAgICAgICAgICBpbmRpY2F0ZXMgdGhlIGdseXBoJ3MgcHJpb3JpdHkgbGV2ZWwgd2hlbiBpdCBzaHJpbmtzLgotICAgICAqIEBwYXJhbSBzaHJpbmtMZWZ0TGltaXQKLSAgICAgKiAgICAgICAgICAgIGluZGljYXRlcyB0aGUgbWF4aW11bSB2YWx1ZSBvZiB3aGljaCB0aGUgbGVmdCBzaWRlIG9mIHRoaXMKLSAgICAgKiAgICAgICAgICAgIGdseXBoIGNhbiBzaHJpbmsuCi0gICAgICogQHBhcmFtIHNocmlua1JpZ2h0TGltaXQKLSAgICAgKiAgICAgICAgICAgIGluZGljYXRlcyB0aGUgbWF4aW11bSBhbW91bnQgYnkgd2hpY2ggdGhlIHJpZ2h0IHNpZGUgb2YgdGhpcwotICAgICAqICAgICAgICAgICAgZ2x5cGggY2FuIHNocmluay4KLSAgICAgKi8KLSAgICBwdWJsaWMgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyhmbG9hdCB3ZWlnaHQsIGJvb2xlYW4gZ3Jvd0Fic29yYiwgaW50IGdyb3dQcmlvcml0eSwKLSAgICAgICAgICAgIGZsb2F0IGdyb3dMZWZ0TGltaXQsIGZsb2F0IGdyb3dSaWdodExpbWl0LCBib29sZWFuIHNocmlua0Fic29yYiwgaW50IHNocmlua1ByaW9yaXR5LAotICAgICAgICAgICAgZmxvYXQgc2hyaW5rTGVmdExpbWl0LCBmbG9hdCBzaHJpbmtSaWdodExpbWl0KSB7Ci0KLSAgICAgICAgaWYgKHdlaWdodCA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4xOUM9d2VpZ2h0IG11c3QgYmUgYSBwb3NpdGl2ZSBudW1iZXIKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTlDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy53ZWlnaHQgPSB3ZWlnaHQ7Ci0KLSAgICAgICAgaWYgKGdyb3dMZWZ0TGltaXQgPCAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTlEPWdyb3dMZWZ0TGltaXQgbXVzdCBiZSBhIHBvc2l0aXZlIG51bWJlcgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOUQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB0aGlzLmdyb3dMZWZ0TGltaXQgPSBncm93TGVmdExpbWl0OwotCi0gICAgICAgIGlmIChncm93UmlnaHRMaW1pdCA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4xOUU9Z3Jvd1JpZ2h0TGltaXQgbXVzdCBiZSBhIHBvc2l0aXZlIG51bWJlcgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOUUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB0aGlzLmdyb3dSaWdodExpbWl0ID0gZ3Jvd1JpZ2h0TGltaXQ7Ci0KLSAgICAgICAgaWYgKChzaHJpbmtQcmlvcml0eSA8IDApIHx8IChzaHJpbmtQcmlvcml0eSA+IFBSSU9SSVRZX05PTkUpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTlGPWluY29ycmVjdCB2YWx1ZSBmb3Igc2hyaW5rUHJpb3JpdHksIG1vcmUgdGhhbgotICAgICAgICAgICAgLy8gUFJJT1JJVFlfTk9ORSBvciBsZXNzIHRoYW4gUFJJT1JJVFlfS0FTSElEQSB2YWx1ZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOUYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB0aGlzLnNocmlua1ByaW9yaXR5ID0gc2hyaW5rUHJpb3JpdHk7Ci0KLSAgICAgICAgaWYgKChncm93UHJpb3JpdHkgPCAwKSB8fCAoZ3Jvd1ByaW9yaXR5ID4gUFJJT1JJVFlfTk9ORSkpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMDA9aW5jb3JyZWN0IHZhbHVlIGZvciBncm93UHJpb3JpdHksIG1vcmUgdGhhbiBQUklPUklUWV9OT05FCi0gICAgICAgICAgICAvLyBvciBsZXNzIHRoYW4gUFJJT1JJVFlfS0FTSElEQSB2YWx1ZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMDAiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB0aGlzLmdyb3dQcmlvcml0eSA9IGdyb3dQcmlvcml0eTsKLQotICAgICAgICBpZiAoc2hyaW5rTGVmdExpbWl0IDwgMCkgewotICAgICAgICAgICAgLy8gYXd0LjIwMT1zaHJpbmtMZWZ0TGltaXQgbXVzdCBiZSBhIHBvc2l0aXZlIG51bWJlcgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMDEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB0aGlzLnNocmlua0xlZnRMaW1pdCA9IHNocmlua0xlZnRMaW1pdDsKLQotICAgICAgICBpZiAoc2hyaW5rUmlnaHRMaW1pdCA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMDI9c2hyaW5rUmlnaHRMaW1pdCBtdXN0IGJlIGEgcG9zaXRpdmUgbnVtYmVyCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIwMiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHRoaXMuc2hyaW5rUmlnaHRMaW1pdCA9IHNocmlua1JpZ2h0TGltaXQ7Ci0KLSAgICAgICAgdGhpcy5zaHJpbmtBYnNvcmIgPSBzaHJpbmtBYnNvcmI7Ci0gICAgICAgIHRoaXMuZ3Jvd0Fic29yYiA9IGdyb3dBYnNvcmI7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ZvbnQvR2x5cGhNZXRyaWNzLmphdmEgYi9hd3QvamF2YS9hd3QvZm9udC9HbHlwaE1ldHJpY3MuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjg3MTcyMi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZm9udC9HbHlwaE1ldHJpY3MuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI2NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLQotLyoqCi0gKiBUaGUgR2x5cGhNZXRyaWNzIGNsYXNzIHByb3ZpZGVzIGluZm9ybWF0aW9uIGFib3V0IHRoZSBzaXplIGFuZCBzaGFwZSBvZiBhCi0gKiBzaW5nbGUgZ2x5cGguIEVhY2ggZ2x5cGggaGFzIGluZm9ybWF0aW9uIHRvIHNwZWNpZnkgd2hldGhlciBpdHMgYmFzZWxpbmUgaXMKLSAqIGhvcml6b250YWwgb3IgdmVydGljYWwgYXMgd2VsbCBhcyBpbmZvcm1hdGlvbiBvbiBob3cgaXQgaW50ZXJhY3RzIHdpdGggb3RoZXIKLSAqIGNoYXJhY3RlcnMgaW4gYSB0ZXh0LCBnaXZlbiBhcyBvbmUgb2YgdGhlIGZvbGxvd2luZyB0eXBlczogU1RBTkRBUkQsCi0gKiBMSUdBVFVSRSwgQ09NQklOSU5HLCBvciBDT01QT05FTlQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgZmluYWwgY2xhc3MgR2x5cGhNZXRyaWNzIHsKLQotICAgIC8vIGFkdmFuY2Ugd2lkdGggb2YgdGhlIGdseXBoIGNoYXJhY3RlciBjZWxsCi0gICAgLyoqCi0gICAgICogVGhlIGFkdmFuY2UgeC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGFkdmFuY2VYOwotCi0gICAgLy8gYWR2YW5jZSBoZWlnaHQgb2YgdGhlIGdseXBoIGNoYXJhY3RlciBjZWxsCi0gICAgLyoqCi0gICAgICogVGhlIGFkdmFuY2UgeS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGFkdmFuY2VZOwotCi0gICAgLy8gZmxhZyBpZiB0aGUgZ2x5cGggaG9yaXpvbnRhbAotICAgIC8qKgotICAgICAqIFRoZSBob3Jpem9udGFsLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBob3Jpem9udGFsOwotCi0gICAgLy8gZ2x5cGggdHlwZSBjb2RlCi0gICAgLyoqCi0gICAgICogVGhlIGdseXBoIHR5cGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBieXRlIGdseXBoVHlwZTsKLQotICAgIC8vIGJvdW5kaW5nIGJveCBmb3Igb3V0bGluZSBvZiB0aGUgZ2x5cGgKLSAgICAvKioKLSAgICAgKiBUaGUgYm91bmRzLgotICAgICAqLwotICAgIHByaXZhdGUgUmVjdGFuZ2xlMkQuRmxvYXQgYm91bmRzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNUQU5EQVJEIGluZGljYXRlcyBhIGdseXBoIHRoYXQgcmVwcmVzZW50cyBhIHNpbmdsZQotICAgICAqIGNoYXJhY3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGJ5dGUgU1RBTkRBUkQgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IExJR0FUVVJFIGluZGljYXRlcyBhIGdseXBoIHRoYXQgcmVwcmVzZW50cyBtdWx0aXBsZQotICAgICAqIGNoYXJhY3RlcnMgYXMgYSBsaWdhdHVyZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGJ5dGUgTElHQVRVUkUgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENPTUJJTklORyBpbmRpY2F0ZXMgYSBnbHlwaCB3aGljaCBoYXMgbm8gY2FyZXQgcG9zaXRpb24KLSAgICAgKiBiZXR3ZWVuIGdseXBocyAoZm9yIGV4YW1wbGUgdW1sYXV0KS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGJ5dGUgQ09NQklOSU5HID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDT01QT05FTlQgaW5kaWNhdGVzIGEgZ2x5cGggd2l0aCBubyBjb3JyZXNwb25kaW5nIGNoYXJhY3RlcgotICAgICAqIGluIHRoZSBiYWNraW5nIHN0b3JlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgYnl0ZSBDT01QT05FTlQgPSAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFdISVRFU1BBQ0UgaW5kaWNhdGVzIGEgZ2x5cGggd2l0aG91dCB2aXN1YWwgcmVwcmVzZW50YXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBieXRlIFdISVRFU1BBQ0UgPSA0OwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEdseXBoTWV0cmljcyBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGhvcml6b250YWwKLSAgICAgKiAgICAgICAgICAgIHNwZWNpZmllcyBpZiBtZXRyaWNzIGFyZSBmb3IgYSBob3Jpem9udGFsIGJhc2VsaW5lICh0cnVlCi0gICAgICogICAgICAgICAgICB2YWx1ZSksIG9yIGEgdmVydGljYWwgYmFzZWxpbmUgKGZhbHNlIHZhbHVlKS4KLSAgICAgKiBAcGFyYW0gYWR2YW5jZVgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvbXBvbmVudCBvZiB0aGUgZ2x5cGgncyBhZHZhbmNlLgotICAgICAqIEBwYXJhbSBhZHZhbmNlWQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29tcG9uZW50IG9mIHRoZSBnbHlwaCdzIGFkdmFuY2UuCi0gICAgICogQHBhcmFtIGJvdW5kcwotICAgICAqICAgICAgICAgICAgdGhlIGdseXBoJ3MgYm91bmRzLgotICAgICAqIEBwYXJhbSBnbHlwaFR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBnbHlwaCdzIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIEdseXBoTWV0cmljcyhib29sZWFuIGhvcml6b250YWwsIGZsb2F0IGFkdmFuY2VYLCBmbG9hdCBhZHZhbmNlWSwgUmVjdGFuZ2xlMkQgYm91bmRzLAotICAgICAgICAgICAgYnl0ZSBnbHlwaFR5cGUpIHsKLSAgICAgICAgdGhpcy5ob3Jpem9udGFsID0gaG9yaXpvbnRhbDsKLSAgICAgICAgdGhpcy5hZHZhbmNlWCA9IGFkdmFuY2VYOwotICAgICAgICB0aGlzLmFkdmFuY2VZID0gYWR2YW5jZVk7Ci0KLSAgICAgICAgdGhpcy5ib3VuZHMgPSBuZXcgUmVjdGFuZ2xlMkQuRmxvYXQoKTsKLSAgICAgICAgdGhpcy5ib3VuZHMuc2V0UmVjdChib3VuZHMpOwotCi0gICAgICAgIHRoaXMuZ2x5cGhUeXBlID0gZ2x5cGhUeXBlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBob3Jpem9udGFsIEdseXBoTWV0cmljcyB3aXRoIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYWR2YW5jZVgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvbXBvbmVudCBvZiB0aGUgZ2x5cGgncyBhZHZhbmNlLgotICAgICAqIEBwYXJhbSBib3VuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBnbHlwaCdzIGJvdW5kcy4KLSAgICAgKiBAcGFyYW0gZ2x5cGhUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZ2x5cGgncyB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBHbHlwaE1ldHJpY3MoZmxvYXQgYWR2YW5jZVgsIFJlY3RhbmdsZTJEIGJvdW5kcywgYnl0ZSBnbHlwaFR5cGUpIHsKLSAgICAgICAgdGhpcy5hZHZhbmNlWCA9IGFkdmFuY2VYOwotICAgICAgICB0aGlzLmFkdmFuY2VZID0gMDsKLQotICAgICAgICB0aGlzLmhvcml6b250YWwgPSB0cnVlOwotCi0gICAgICAgIHRoaXMuYm91bmRzID0gbmV3IFJlY3RhbmdsZTJELkZsb2F0KCk7Ci0gICAgICAgIHRoaXMuYm91bmRzLnNldFJlY3QoYm91bmRzKTsKLQotICAgICAgICB0aGlzLmdseXBoVHlwZSA9IGdseXBoVHlwZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBnbHlwaCdzIGJvdW5kcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGdseXBoJ3MgYm91bmRzLgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRCgpIHsKLSAgICAgICAgcmV0dXJuIChSZWN0YW5nbGUyRC5GbG9hdCl0aGlzLmJvdW5kcy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGlzIGdseXBoIGlzIHdoaXRlc3BhY2Ugb3Igbm90LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBnbHlwaCBpcyB3aGl0ZXNwYWNlLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNXaGl0ZXNwYWNlKCkgewotICAgICAgICByZXR1cm4gKCh0aGlzLmdseXBoVHlwZSAmIDQpID09IFdISVRFU1BBQ0UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGlzIGdseXBoIGlzIHN0YW5kYXJkIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgZ2x5cGggaXMgc3RhbmRhcmQsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1N0YW5kYXJkKCkgewotICAgICAgICByZXR1cm4gKCh0aGlzLmdseXBoVHlwZSAmIDMpID09IFNUQU5EQVJEKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyBnbHlwaCBpcyBsaWdhdHVyZSBvciBub3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIGdseXBoIGlzIGxpZ2F0dXJlLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNMaWdhdHVyZSgpIHsKLSAgICAgICAgcmV0dXJuICgodGhpcy5nbHlwaFR5cGUgJiAzKSA9PSBMSUdBVFVSRSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoaXMgZ2x5cGggaXMgY29tcG9uZW50IG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgZ2x5cGggaXMgY29tcG9uZW50LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb21wb25lbnQoKSB7Ci0gICAgICAgIHJldHVybiAoKHRoaXMuZ2x5cGhUeXBlICYgMykgPT0gQ09NUE9ORU5UKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyBnbHlwaCBpcyBjb21iaW5pbmcgb3Igbm90LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBnbHlwaCBpcyBjb21iaW5pbmcsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NvbWJpbmluZygpIHsKLSAgICAgICAgcmV0dXJuICgodGhpcy5nbHlwaFR5cGUgJiAzKSA9PSBDT01CSU5JTkcpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGdseXBoJ3MgdHlwZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBnbHlwaCdzIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRUeXBlKCkgewotICAgICAgICByZXR1cm4gdGhpcy5nbHlwaFR5cGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGlzdGFuY2UgZnJvbSB0aGUgcmlnaHQgKGZvciBob3Jpem9udGFsKSBvciBib3R0b20gKGZvcgotICAgICAqIHZlcnRpY2FsKSBvZiB0aGUgZ2x5cGggYm91bmRzIHRvIHRoZSBhZHZhbmNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRpc3RhbmNlIGZyb20gdGhlIHJpZ2h0IChmb3IgaG9yaXpvbnRhbCkgb3IgYm90dG9tIChmb3IKLSAgICAgKiAgICAgICAgIHZlcnRpY2FsKSBvZiB0aGUgZ2x5cGggYm91bmRzIHRvIHRoZSBhZHZhbmNlLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRSU0IoKSB7Ci0gICAgICAgIGlmICh0aGlzLmhvcml6b250YWwpIHsKLSAgICAgICAgICAgIHJldHVybiB0aGlzLmFkdmFuY2VYIC0gdGhpcy5ib3VuZHMueCAtIChmbG9hdCl0aGlzLmJvdW5kcy5nZXRXaWR0aCgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB0aGlzLmFkdmFuY2VZIC0gdGhpcy5ib3VuZHMueSAtIChmbG9hdCl0aGlzLmJvdW5kcy5nZXRIZWlnaHQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkaXN0YW5jZSBmcm9tIDAsIDAgdG8gdGhlIGxlZnQgKGZvciBob3Jpem9udGFsKSBvciB0b3AgKGZvcgotICAgICAqIHZlcnRpY2FsKSBvZiB0aGUgZ2x5cGggYm91bmRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRpc3RhbmNlIGZyb20gMCwgMCB0byB0aGUgbGVmdCAoZm9yIGhvcml6b250YWwpIG9yIHRvcCAoZm9yCi0gICAgICogICAgICAgICB2ZXJ0aWNhbCkgb2YgdGhlIGdseXBoIGJvdW5kcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0TFNCKCkgewotICAgICAgICBpZiAodGhpcy5ob3Jpem9udGFsKSB7Ci0gICAgICAgICAgICByZXR1cm4gdGhpcy5ib3VuZHMueDsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdGhpcy5ib3VuZHMueTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBZIGNvbXBvbmVudCBvZiB0aGUgZ2x5cGgncyBhZHZhbmNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFkgY29tcG9uZW50IG9mIHRoZSBnbHlwaCdzIGFkdmFuY2UuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldEFkdmFuY2VZKCkgewotICAgICAgICByZXR1cm4gdGhpcy5hZHZhbmNlWTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBYIGNvbXBvbmVudCBvZiB0aGUgZ2x5cGgncyBhZHZhbmNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFggY29tcG9uZW50IG9mIHRoZSBnbHlwaCdzIGFkdmFuY2UuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldEFkdmFuY2VYKCkgewotICAgICAgICByZXR1cm4gdGhpcy5hZHZhbmNlWDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBnbHlwaCdzIGFkdmFuY2UgYWxvbmcgdGhlIGJhc2VsaW5lLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGdseXBoJ3MgYWR2YW5jZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0QWR2YW5jZSgpIHsKLSAgICAgICAgaWYgKHRoaXMuaG9yaXpvbnRhbCkgewotICAgICAgICAgICAgcmV0dXJuIHRoaXMuYWR2YW5jZVg7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHRoaXMuYWR2YW5jZVk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZm9udC9HbHlwaFZlY3Rvci5qYXZhIGIvYXd0L2phdmEvYXd0L2ZvbnQvR2x5cGhWZWN0b3IuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTcyYjc3NC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZm9udC9HbHlwaFZlY3Rvci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDAzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmZvbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5Gb250OwotaW1wb3J0IGphdmEuYXd0LmZvbnQuRm9udFJlbmRlckNvbnRleHQ7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5HbHlwaEp1c3RpZmljYXRpb25JbmZvOwotaW1wb3J0IGphdmEuYXd0LmZvbnQuR2x5cGhNZXRyaWNzOwotCi1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LlNoYXBlOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUG9pbnQyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotCi0vKioKLSAqIFRoZSBHbHlwaFZlY3RvciBjbGFzcyBjb250YWlucyBhIGNvbGxlY3Rpb24gb2YgZ2x5cGhzIHdpdGggZ2VvbWV0cmljCi0gKiBpbmZvcm1hdGlvbiBhbmQgZWFjaCBnbHlwaCdzIGxvY2F0aW9uLiBFYWNoIEdseXBoVmVjdG9yIGNhbiBiZSBhc3NvY2lhdGVkCi0gKiB3aXRoIG9ubHkgb25lIEZvbnQuIEdseXBoVmVjdG9yIGNvbnRhaW5zIHRoZSBmb2xsb3dpbmcgcHJvcGVydGllcyBmb3IgZWFjaAotICogZ2x5cGg6Ci0gKiA8dWw+Ci0gKiA8bGk+dGhlIGdseXBoIHBvc2l0aW9uOzwvbGk+Ci0gKiA8bGk+dGhlIHRyYW5zZm9ybSBvZiB0aGUgZ2x5cGg7PC9saT4KLSAqIDxsaT50aGUgbWV0cmljcyBvZiB0aGUgZ2x5cGggaW4gdGhlIGNvbnRleHQgb2YgdGhlIEdseXBoVmVjdG9yLjwvbGk+Ci0gKiA8L3VsPgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEdseXBoVmVjdG9yIGltcGxlbWVudHMgQ2xvbmVhYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGTEFHX0hBU19UUkFOU0ZPUk1TIGluZGljYXRlcyB0aGF0IHRoaXMgR2x5cGhWZWN0b3IgaGFzCi0gICAgICogcGVyLWdseXBoIHRyYW5zZm9ybXMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRkxBR19IQVNfVFJBTlNGT1JNUyA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRkxBR19IQVNfUE9TSVRJT05fQURKVVNUTUVOVFMgaW5kaWNhdGVzIHRoYXQgdGhlIEdseXBoVmVjdG9yCi0gICAgICogaGFzIHBlci1nbHlwaCBwb3NpdGlvbiBhZGp1c3RtZW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBGTEFHX0hBU19QT1NJVElPTl9BREpVU1RNRU5UUyA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRkxBR19SVU5fUlRMIGluZGljYXRlcyB0aGF0IHRoaXMgR2x5cGhWZWN0b3IgaGFzIGEgcmlnaHQgdG8KLSAgICAgKiBsZWZ0IHJ1biBkaXJlY3Rpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRkxBR19SVU5fUlRMID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGTEFHX0NPTVBMRVhfR0xZUEhTIGluZGljYXRlcyB0aGF0IHRoaXMgR2x5cGhWZWN0b3IgaGFzIGEKLSAgICAgKiBjb21wbGV4IGdseXBoIHRvIGNoYXIgbWFwcGluZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBGTEFHX0NPTVBMRVhfR0xZUEhTID0gODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBGTEFHX01BU0sgaW5kaWNhdGVzIGEgbWFzayBmb3Igc3VwcG9ydGVkIGZsYWdzIGZyb20KLSAgICAgKiBnZXRMYXlvdXRGbGFncy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBGTEFHX01BU0sgPSAxNTsgLy8gKHwpIG1hc2sgb2Ygb3RoZXIgZmxhZ3MKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBHbHlwaFZlY3Rvci4KLSAgICAgKi8KLSAgICBwdWJsaWMgR2x5cGhWZWN0b3IoKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcGl4ZWwgYm91bmRzIG9mIHRoZSBHbHlwaFZlY3RvciB3aGVuIHJlbmRlcmVkIGF0IHRoZSBzcGVjaWZpZWQKLSAgICAgKiBsb2NhdGlvbiB3aXRoIHRoZSBzcGVjaWZpZWQgRm9udFJlbmRlckNvbnRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZyYwotICAgICAqICAgICAgICAgICAgdGhlIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBHbHlwaFZlY3RvcidzIGxvY2F0aW9uLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBHbHlwaFZlY3RvcidzIGxvY2F0aW9uLgotICAgICAqIEByZXR1cm4gdGhlIHBpeGVsIGJvdW5kcwotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0UGl4ZWxCb3VuZHMoRm9udFJlbmRlckNvbnRleHQgZnJjLCBmbG9hdCB4LCBmbG9hdCB5KSB7Ci0gICAgICAgIC8vIGRlZmF1bHQgaW1wbGVtZW50YXRpb24gLSBpbnRlZ2VyIFJlY3RhbmdsZSwgdGhhdCBlbmNsb3NlcyB2aXN1YWwKLSAgICAgICAgLy8gYm91bmRzIHJlY3RhbmdsZQotICAgICAgICBSZWN0YW5nbGUyRCB2aXN1YWxSZWN0ID0gZ2V0VmlzdWFsQm91bmRzKCk7Ci0KLSAgICAgICAgaW50IG1pblggPSAoaW50KU1hdGguZmxvb3IodmlzdWFsUmVjdC5nZXRNaW5YKCkgKyB4KTsKLSAgICAgICAgaW50IG1pblkgPSAoaW50KU1hdGguZmxvb3IodmlzdWFsUmVjdC5nZXRNaW5ZKCkgKyB5KTsKLSAgICAgICAgaW50IHdpZHRoID0gKGludClNYXRoLmNlaWwodmlzdWFsUmVjdC5nZXRNYXhYKCkgKyB4KSAtIG1pblg7Ci0gICAgICAgIGludCBoZWlnaHQgPSAoaW50KU1hdGguY2VpbCh2aXN1YWxSZWN0LmdldE1heFkoKSArIHkpIC0gbWluWTsKLQotICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZShtaW5YLCBtaW5ZLCB3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBwaXhlbCBib3VuZHMgb2YgdGhlIGdseXBoIHdpdGggdGhlIHNwZWNpZmllZCBpbmRleCBpbiB0aGlzCi0gICAgICogR2x5cGhWZWN0b3Igd2hpY2ggaXMgcmVuZGVyZWQgd2l0aCB0aGUgc3BlY2lmaWVkIEZvbnRSZW5kZXJDb250ZXh0IGF0IHRoZQotICAgICAqIHNwZWNpZmllZCBsb2NhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBnbHlwaCBpbmRleCBpbiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqIEBwYXJhbSBmcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb250UmVuZGVyQ29udGV4dC4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgR2x5cGhWZWN0b3IncyBsb2NhdGlvbi4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgR2x5cGhWZWN0b3IncyBsb2NhdGlvbi4KLSAgICAgKiBAcmV0dXJuIGEgUmVjdGFuZ2xlIGJvdW5kcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldEdseXBoUGl4ZWxCb3VuZHMoaW50IGluZGV4LCBGb250UmVuZGVyQ29udGV4dCBmcmMsIGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgUmVjdGFuZ2xlMkQgdmlzdWFsUmVjdCA9IGdldEdseXBoVmlzdWFsQm91bmRzKGluZGV4KS5nZXRCb3VuZHMyRCgpOwotCi0gICAgICAgIGludCBtaW5YID0gKGludClNYXRoLmZsb29yKHZpc3VhbFJlY3QuZ2V0TWluWCgpICsgeCk7Ci0gICAgICAgIGludCBtaW5ZID0gKGludClNYXRoLmZsb29yKHZpc3VhbFJlY3QuZ2V0TWluWSgpICsgeSk7Ci0gICAgICAgIGludCB3aWR0aCA9IChpbnQpTWF0aC5jZWlsKHZpc3VhbFJlY3QuZ2V0TWF4WCgpICsgeCkgLSBtaW5YOwotICAgICAgICBpbnQgaGVpZ2h0ID0gKGludClNYXRoLmNlaWwodmlzdWFsUmVjdC5nZXRNYXhZKCkgKyB5KSAtIG1pblk7Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBSZWN0YW5nbGUobWluWCwgbWluWSwgd2lkdGgsIGhlaWdodCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdmlzdWFsIGJvdW5kcyBvZiB0aGUgR2x5cGhWZWN0b3IuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdmlzdWFsIGJvdW5kcyBvZiB0aGUgR2x5cGhWZWN0b3IuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFJlY3RhbmdsZTJEIGdldFZpc3VhbEJvdW5kcygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9naWNhbCBib3VuZHMgb2YgdGhlIEdseXBoVmVjdG9yLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGxvZ2ljYWwgYm91bmRzIG9mIHRoZSBHbHlwaFZlY3Rvci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgUmVjdGFuZ2xlMkQgZ2V0TG9naWNhbEJvdW5kcygpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgcG9zaXRpb24gb2YgdGhlIHNwZWNpZmllZCBnbHlwaCBpbiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgZ2x5cGggaW5kZXggaW4gdGhpcyBHbHlwaFZlY3Rvci4KLSAgICAgKiBAcGFyYW0gbmV3UG9zCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHBvc2l0aW9uIG9mIHRoZSBnbHlwaCBhdCB0aGUgc3BlY2lmaWVkIGdseXBoSW5kZXguCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0R2x5cGhQb3NpdGlvbihpbnQgZ2x5cGhJbmRleCwgUG9pbnQyRCBuZXdQb3MpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcG9zaXRpb24gb2YgdGhlIHNwZWNpZmllZCBnbHlwaCBpbiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgZ2x5cGggaW5kZXggaW4gdGhpcyBHbHlwaFZlY3Rvci4KLSAgICAgKiBAcmV0dXJuIHRoZSBwb3NpdGlvbiBvZiB0aGUgc3BlY2lmaWVkIGdseXBoIGluIHRoaXMgR2x5cGhWZWN0b3IuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFBvaW50MkQgZ2V0R2x5cGhQb3NpdGlvbihpbnQgZ2x5cGhJbmRleCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBhZmZpbmUgdHJhbnNmb3JtIHRvIGEgZ2x5cGggd2l0aCB0aGUgc3BlY2lmaWVkIGluZGV4IGluIHRoaXMKLSAgICAgKiBHbHlwaFZlY3Rvci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2x5cGhJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGdseXRoIGluZGV4IGluIHRoaXMgR2x5cGhWZWN0b3IuCi0gICAgICogQHBhcmFtIHRyYW5zCi0gICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtIHRvIGJlIGFzc2lnbmVkIHRvIHRoZSBzcGVjaWZpZWQgZ2x5cGguCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0R2x5cGhUcmFuc2Zvcm0oaW50IGdseXBoSW5kZXgsIEFmZmluZVRyYW5zZm9ybSB0cmFucyk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0cmFuc2Zvcm0gb2YgdGhlIHNwZWNpZmllZCBnbHlwaCBpbiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgZ2x5cGggaW5kZXggaW4gdGhpcyBHbHlwaFZlY3Rvci4KLSAgICAgKiBAcmV0dXJuIHRoZSBuZXcgdHJhbnNmb3JtIG9mIHRoZSBnbHlwaC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgQWZmaW5lVHJhbnNmb3JtIGdldEdseXBoVHJhbnNmb3JtKGludCBnbHlwaEluZGV4KTsKLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIHRoaXMgR2x5cGhWZWN0b3Igd2l0aCB0aGUgc3BlY2lmaWVkIEdseXBoVmVjdG9yIG9iamVjdHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGdseXBoVmVjdG9yCi0gICAgICogICAgICAgICAgICB0aGUgR2x5cGhWZWN0b3Igb2JqZWN0IHRvIGJlIGNvbXBhcmVkLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBHbHlwaFZlY3RvciBpcyBlcXVhbCB0byB0aGUgc3BlY2lmaWVkIEdseXBoVmVjdG9yCi0gICAgICogICAgICAgICBvYmplY3QsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgYm9vbGVhbiBlcXVhbHMoR2x5cGhWZWN0b3IgZ2x5cGhWZWN0b3IpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWV0cmljcyBvZiB0aGUgZ2x5cGggd2l0aCB0aGUgc3BlY2lmaWVkIGluZGV4IGluIHRoaXMKLSAgICAgKiBHbHlwaFZlY3Rvci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2x5cGhJbmRleAotICAgICAqICAgICAgICAgICAgaW5kZXggaW4gdGhpcyBHbHlwaFZlY3Rvci4KLSAgICAgKiBAcmV0dXJuIHRoZSBtZXRyaWNzIG9mIHRoZSBnbHlwaCB3aXRoIHRoZSBzcGVjaWZpZWQgaW5kZXggaW4gdGhpcwotICAgICAqICAgICAgICAgR2x5cGhWZWN0b3IuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEdseXBoTWV0cmljcyBnZXRHbHlwaE1ldHJpY3MoaW50IGdseXBoSW5kZXgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUganVzdGlmaWNhdGlvbiBpbmZvcm1hdGlvbiBvZiB0aGUgZ2x5cGggd2hvc2UgaW5kZXggaXMgc3BlY2lmaWVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgZ2x5cGggaW5kZXguCi0gICAgICogQHJldHVybiB0aGUgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBmb3IgdGhlIHNwZWNpZmllZCBnbHlwaC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBnZXRHbHlwaEp1c3RpZmljYXRpb25JbmZvKGludCBnbHlwaEluZGV4KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEZvbnRSZW5kZXJDb250ZXh0IG9mIHRoaXMgR2x5cGhWZWN0b3IuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgRm9udFJlbmRlckNvbnRleHQgb2YgdGhpcyBHbHlwaFZlY3Rvci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgRm9udFJlbmRlckNvbnRleHQgZ2V0Rm9udFJlbmRlckNvbnRleHQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgYSBTaGFwZSBvYmplY3Qgd2hpY2ggZGVmaW5lcyB0aGUgdmlzdWFsIHJlcHJlc2VudGF0aW9uIG9mIHRoZQotICAgICAqIHNwZWNpZmllZCBnbHlwaCBpbiB0aGlzIEdseXBoVmVjdG9yLCB0cmFuc2xhdGVkIGEgZGlzdGFuY2Ugb2YgeCBpbiB0aGUgWAotICAgICAqIGRpcmVjdGlvbiBhbmQgeSBpbiB0aGUgWSBkaXJlY3Rpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIGdseXBoSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBnbHl0aCBpbmRleCBpbiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgaW4gdGhlIFggZGlyZWN0aW9uIHRvIHRyYW5zbGF0ZSB0aGUgc2hhcGUgb2JqZWN0Ci0gICAgICogICAgICAgICAgICBiZWZvcmUgcmV0dXJuaW5nIGl0LgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgaW4gdGhlIFkgZGlyZWN0aW9uIHRvIHRyYW5zbGF0ZSB0aGUgc2hhcGUgb2JqZWN0Ci0gICAgICogICAgICAgICAgICBiZWZvcmUgcmV0dXJuaW5nIGl0LgotICAgICAqIEByZXR1cm4gYSBTaGFwZSBvYmplY3Qgd2hpY2ggcmVwcmVzZW50cyB0aGUgdmlzdWFsIHJlcHJlc2VudGF0aW9uIG9mIHRoZQotICAgICAqICAgICAgICAgc3BlY2lmaWVkIGdseXBoIGluIHRoaXMgR2x5cGhWZWN0b3IgLSBnbHlwaCBvdXRsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZSBnZXRHbHlwaE91dGxpbmUoaW50IGdseXBoSW5kZXgsIGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgU2hhcGUgaW5pdGlhbFNoYXBlID0gZ2V0R2x5cGhPdXRsaW5lKGdseXBoSW5kZXgpOwotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gdHJhbnMgPSBBZmZpbmVUcmFuc2Zvcm0uZ2V0VHJhbnNsYXRlSW5zdGFuY2UoeCwgeSk7Ci0gICAgICAgIHJldHVybiB0cmFucy5jcmVhdGVUcmFuc2Zvcm1lZFNoYXBlKGluaXRpYWxTaGFwZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdmlzdWFsIGJvdW5kcyBvZiB0aGUgc3BlY2lmaWVkIGdseXBoIGluIHRoZSBHbHlwaFZlY3Rvci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2x5cGhJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGdseXBoIGluZGV4IGluIHRoaXMgR2x5cGhWZWN0b3IuCi0gICAgICogQHJldHVybiB0aGUgZ2x5cGggdmlzdWFsIGJvdW5kcyBvZiB0aGUgZ2x5cGggd2l0aCB0aGUgc3BlY2lmaWVkIGluZGV4IGluCi0gICAgICogICAgICAgICB0aGUgR2x5cGhWZWN0b3IuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFNoYXBlIGdldEdseXBoVmlzdWFsQm91bmRzKGludCBnbHlwaEluZGV4KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgYSBTaGFwZSBvYmplY3Qgd2hpY2ggZGVmaW5lcyB0aGUgdmlzdWFsIHJlcHJlc2VudGF0aW9uIG9mIHRoZQotICAgICAqIHNwZWNpZmllZCBnbHlwaCBpbiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgZ2x5dGggaW5kZXggaW4gdGhpcyBHbHlwaFZlY3Rvci4KLSAgICAgKiBAcmV0dXJuIGEgU2hhcGUgb2JqZWN0IHdoaWNoIHJlcHJlc2VudHMgdGhlIHZpc3VhbCByZXByZXNlbnRhdGlvbiBvZiB0aGUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZCBnbHlwaCBpbiB0aGlzIEdseXBoVmVjdG9yIC0gZ2x5cGggb3V0bGluZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgU2hhcGUgZ2V0R2x5cGhPdXRsaW5lKGludCBnbHlwaEluZGV4KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvZ2ljYWwgYm91bmRzIG9mIHRoZSBzcGVjaWZpZWQgZ2x5cGggaW4gdGhlIEdseXBoVmVjdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggaW4gdGhpcyBHbHlwaFZlY3RvciBvZiB0aGUgZ2x5cGggZnJvbSB3aGljaCB0bwotICAgICAqICAgICAgICAgICAgcmV0cmlldmUgaXRzIGxvZ2ljYWwgYm91bmRzCi0gICAgICogQHJldHVybiB0aGUgbG9naWNhbCBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBnbHlwaCBpbiB0aGUgR2x5cGhWZWN0b3IuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFNoYXBlIGdldEdseXBoTG9naWNhbEJvdW5kcyhpbnQgZ2x5cGhJbmRleCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB2aXN1YWwgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBHbHlwaFZlY3RvciByZW5kZXJlZCBpbiB4LCB5Ci0gICAgICogbG9jYXRpb24gYXMgYSBTaGFwZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIEdseXBoVmVjdG9yLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBHbHlwaFZlY3Rvci4KLSAgICAgKiBAcmV0dXJuIHRoZSB2aXN1YWwgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBHbHlwaFZlY3RvciBhcyBhIFNoYXBlIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgU2hhcGUgZ2V0T3V0bGluZShmbG9hdCB4LCBmbG9hdCB5KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHZpc3VhbCByZXByZXNlbnRhdGlvbiBvZiB0aGlzIEdseXBoVmVjdG9yIGFzIGEgU2hhcGUgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHZpc3VhbCByZXByZXNlbnRhdGlvbiBvZiB0aGlzIEdseXBoVmVjdG9yIGFzIGEgU2hhcGUgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBTaGFwZSBnZXRPdXRsaW5lKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmb250IG9mIHRoaXMgR2x5cGhWZWN0b3IuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZm9udCBvZiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBGb250IGdldEZvbnQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgdGhlIGdseXBoIGNvZGVzIG9mIHRoZSBzcGVjaWZpZWQgZ2x5cGhzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiZWdpbkdseXBoSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBpbnRvIHRoaXMgR2x5cGhWZWN0b3IgYXQgd2hpY2ggdG8gc3RhcnQgcmV0cmlldmluZwotICAgICAqICAgICAgICAgICAgZ2x5cGggY29kZXMuCi0gICAgICogQHBhcmFtIG51bUVudHJpZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgZ2x5cGggY29kZXMuCi0gICAgICogQHBhcmFtIGNvZGVSZXR1cm4KLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBpbnRvIHdoaWNoIHRoZSByZXN1bHRpbmcgZ2x5cGhjb2RlcyB3aWxsIGJlIHdyaXR0ZW4uCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgdGhlIGdseXBoIGNvZGVzLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnRbXSBnZXRHbHlwaENvZGVzKGludCBiZWdpbkdseXBoSW5kZXgsIGludCBudW1FbnRyaWVzLCBpbnRbXSBjb2RlUmV0dXJuKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgdGhlIGNoYXJhY3RlciBpbmRpY2VzIG9mIHRoZSBzcGVjaWZpZWQgZ2x5cGhzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiZWdpbkdseXBoSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgZmlyc3QgZ2x5cGggdG8gcmV0dXJuIGluZm9ybWF0aW9uIGZvci4KLSAgICAgKiBAcGFyYW0gbnVtRW50cmllcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBnbHlwaCBpbmRpY2VzIHRvIHJldHVybi4KLSAgICAgKiBAcGFyYW0gY29kZVJldHVybgotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IGludG8gd2hpY2ggdGhlIHJlc3VsdGluZyBjaGFyYWN0ZXIgaW5kaWNlcyB3aWxsIGJlCi0gICAgICogICAgICAgICAgICB3cml0dGVuLgotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgb2YgY2hhcmFjdGVyIGluZGljZXMgZm9yIHRoZSBzcGVjaWZpZXMgZ2x5cGhzLgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXRHbHlwaENoYXJJbmRpY2VzKGludCBiZWdpbkdseXBoSW5kZXgsIGludCBudW1FbnRyaWVzLCBpbnRbXSBjb2RlUmV0dXJuKSB7Ci0gICAgICAgIGlmIChjb2RlUmV0dXJuID09IG51bGwpIHsKLSAgICAgICAgICAgIGNvZGVSZXR1cm4gPSBuZXcgaW50W251bUVudHJpZXNdOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1FbnRyaWVzOyBpKyspIHsKLSAgICAgICAgICAgIGNvZGVSZXR1cm5baV0gPSBnZXRHbHlwaENoYXJJbmRleChpICsgYmVnaW5HbHlwaEluZGV4KTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gY29kZVJldHVybjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIHRoZSBwb3NpdGlvbnMgb2YgdGhlIHNwZWNpZmllZCBnbHlwaHMgaW4gdGhpcwotICAgICAqIEdseXBoVmVjdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiZWdpbkdseXBoSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgZmlyc3QgZ2x5cGggdG8gcmV0dXJuIGluZm9ybWF0aW9uIGZvci4KLSAgICAgKiBAcGFyYW0gbnVtRW50cmllcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBnbHlwaHMgdG8gcmV0dXJuIGluZm9ybWF0aW9uIGZvci4KLSAgICAgKiBAcGFyYW0gcG9zaXRpb25SZXR1cm4KLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB3aGVyZSB0aGUgcmVzdWx0IHdpbGwgYmUgc3RvcmVkLgotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgb2YgZ2x5cGggcG9zaXRpb25zLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBmbG9hdFtdIGdldEdseXBoUG9zaXRpb25zKGludCBiZWdpbkdseXBoSW5kZXgsIGludCBudW1FbnRyaWVzLAotICAgICAgICAgICAgZmxvYXRbXSBwb3NpdGlvblJldHVybik7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBnbHlwaCBjb2RlIG9mIHRoZSBzcGVjaWZpZWQgZ2x5cGguCi0gICAgICogCi0gICAgICogQHBhcmFtIGdseXBoSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBpbiB0aGlzIEdseXBoVmVjdG9yIHdoaWNoIGNvcnJlc3BvbmRzIHRvIHRoZSBnbHlwaAotICAgICAqICAgICAgICAgICAgZnJvbSB3aGljaCB0byByZXRyaWV2ZSB0aGUgZ2x5cGhjb2RlLgotICAgICAqIEByZXR1cm4gdGhlIGdseXBoY29kZSBvZiB0aGUgc3BlY2lmaWVkIGdseXBoLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0R2x5cGhDb2RlKGludCBnbHlwaEluZGV4KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZpcnN0IGxvZ2ljYWwgY2hhcmFjdGVyJ3MgaW5kZXggb2YgdGhlIHNwZWNpZmllZCBnbHlwaC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2x5cGhJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGdseXBoIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIHRoZSBmaXJzdCBsb2dpY2FsIGNoYXJhY3RlcidzIGluZGV4LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0R2x5cGhDaGFySW5kZXgoaW50IGdseXBoSW5kZXgpIHsKLSAgICAgICAgLy8gZGVmYXVsdCBpbXBsZW1ldGF0aW9uIG9uZS10by1vbmUKLSAgICAgICAgcmV0dXJuIGdseXBoSW5kZXg7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBkZWZhdWx0IGxheW91dCB0byB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHBlcmZvcm1EZWZhdWx0TGF5b3V0KCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBudW1iZXIgb2YgZ2x5cGhzIGluIHRoZSBHbHlwaFZlY3Rvci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgZ2x5cGhzIGluIHRoZSBHbHlwaFZlY3Rvci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgaW50IGdldE51bUdseXBocygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyBmbGFncyB3aGljaCBkZXNjcmliZSB0aGUgZ2xvYmFsIHN0YXRlIG9mIHRoZSBHbHlwaFZlY3Rvci4gVGhlCi0gICAgICogZGVmYXVsdCBpbXBsZW1lbnRhdGlvbiByZXR1cm5zIDAuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbGF5b3V0IGZsYWdzCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRMYXlvdXRGbGFncygpIHsKLSAgICAgICAgLy8gZGVmYXVsdCBpbXBsZW1lbnRhdGlvbiAtIHJldHVybmVkIHZhbHVlIGlzIDAKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZm9udC9HcmFwaGljQXR0cmlidXRlLmphdmEgYi9hd3QvamF2YS9hd3QvZm9udC9HcmFwaGljQXR0cmlidXRlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDg0ODBlMGYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ZvbnQvR3JhcGhpY0F0dHJpYnV0ZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTc5ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmZvbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljczJEOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgR3JhcGhpY0F0dHJpYnV0ZSBhYnN0cmFjdCBjbGFzcyBwcm92aWRlcyBhbiBvcHBvcnR1bml0eSB0byBpbnNlcnQKLSAqIGdyYXBoaWNhbCBlbGVtZW50cyBpbiBwcmludGVkIHRleHQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgR3JhcGhpY0F0dHJpYnV0ZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVE9QX0FMSUdOTUVOVCBpbmRpY2F0ZXMgdXNpbmcgdGhlIHRvcCBsaW5lIHRvIGNhbGN1bGF0ZQotICAgICAqIHBsYWNlbWVudCBvZiBncmFwaGljcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUT1BfQUxJR05NRU5UID0gLTE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQk9UVE9NX0FMSUdOTUVOVCBpbmRpY2F0ZXMgdXNpbmcgdGhlIGJvdHRvbSBsaW5lIHRvCi0gICAgICogY2FsY3VsYXRlIHBsYWNlbWVudCBvZiBncmFwaGljcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBCT1RUT01fQUxJR05NRU5UID0gLTI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUk9NQU5fQkFTRUxJTkUgaW5kaWNhdGVzIHRoZSBwbGFjZW1lbnQgb2YgdGhlIHJvbWFuIGJhc2VsaW5lCi0gICAgICogd2l0aCByZXNwZWN0IHRvIHRoZSBncmFwaGljcyBvcmlnaW4uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgUk9NQU5fQkFTRUxJTkUgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENFTlRFUl9CQVNFTElORSBpbmRpY2F0ZXMgdGhlIHBsYWNlbWVudCBvZiB0aGUgY2VudGVyCi0gICAgICogYmFzZWxpbmUgd2l0aCByZXNwZWN0IHRvIHRoZSBncmFwaGljcyBvcmlnaW4uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ0VOVEVSX0JBU0VMSU5FID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBIQU5HSU5HX0JBU0VMSU5FIGluZGljYXRlcyB0aGUgcGxhY2VtZW50IG9mIHRoZSBoYW5naW5nCi0gICAgICogYmFzZWxpbmUgd2l0aCByZXNwZWN0IHRvIHRoZSBncmFwaGljcyBvcmlnaW4uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSEFOR0lOR19CQVNFTElORSA9IDI7Ci0KLSAgICAvLyB0aGUgYWxpZ25tZW50IG9mIHRoaXMgR3JhcGhpY0F0dHJpYnV0ZQotICAgIC8qKgotICAgICAqIFRoZSBhbGlnbm1lbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgYWxpZ25tZW50OwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGdyYXBoaWMgYXR0cmlidXRlIHdpdGggdGhlIHNwZWNpZmllZCBhbGlnbm1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGFsaWduCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGFsaWdubWVudC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgR3JhcGhpY0F0dHJpYnV0ZShpbnQgYWxpZ24pIHsKLSAgICAgICAgaWYgKChhbGlnbiA8IEJPVFRPTV9BTElHTk1FTlQpIHx8IChhbGlnbiA+IEhBTkdJTkdfQkFTRUxJTkUpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTk4PUlsbGVnYWwgYWxpZ25tZW50IGFyZ3VtZW50Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE5OCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHRoaXMuYWxpZ25tZW50ID0gYWxpZ247Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRHJhd3MgdGhlIEdyYXBoaWNBdHRyaWJ1dGUgYXQgdGhlIHNwZWNpZmllZCBsb2NhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ3JhcGhpY3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBHcmFwaGljcy4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiBHcmFwaGljQXR0cmlidXRlIGxvY2F0aW9uLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIEdyYXBoaWNBdHRyaWJ1dGUgbG9jYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhdyhHcmFwaGljczJEIGdyYXBoaWNzLCBmbG9hdCB4LCBmbG9hdCB5KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEdyYXBoaWNBdHRyaWJ1dGUncyBhZHZhbmNlLiBJdCdzIHRoZSBkaXN0YW5jZSBmcm9tIHRoZSBwb2ludCBhdAotICAgICAqIHdoaWNoIHRoZSBncmFwaGljIGlzIHJlbmRlcmVkIGFuZCB0aGUgcG9pbnQgd2hlcmUgdGhlIG5leHQgY2hhcmFjdGVyIG9yCi0gICAgICogZ3JhcGhpYyBpcyByZW5kZXJlZC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBHcmFwaGljQXR0cmlidXRlJ3MgYWR2YW5jZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZmxvYXQgZ2V0QWR2YW5jZSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYWxpZ25tZW50IG9mIHRoaXMgR3JhcGhpY0F0dHJpYnV0ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhbGlnbm1lbnQgb2YgdGhpcyBHcmFwaGljQXR0cmlidXRlLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0QWxpZ25tZW50KCkgewotICAgICAgICByZXR1cm4gdGhpcy5hbGlnbm1lbnQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYXNjZW50IG9mIHRoaXMgR3JhcGhpY0F0dHJpYnV0ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhc2NlbnQgb2YgdGhpcyBHcmFwaGljQXR0cmlidXRlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBmbG9hdCBnZXRBc2NlbnQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJvdW5kcyBvZiB0aGlzIEdyYXBoaWNBdHRyaWJ1dGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYm91bmRzIG9mIHRoaXMgR3JhcGhpY0F0dHJpYnV0ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzKCkgewotICAgICAgICBmbG9hdCBhc2NlbnQgPSBnZXRBc2NlbnQoKTsKLSAgICAgICAgZmxvYXQgYWR2YW5jZSA9IGdldEFkdmFuY2UoKTsKLSAgICAgICAgZmxvYXQgZGVzY2VudCA9IGdldERlc2NlbnQoKTsKLQotICAgICAgICAvLyBEZWZhdWx0IGltcGxlbWVudGF0aW9uIC0gc2VlIEFQSSBkb2N1bWVudGF0aW9uLgotICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZTJELkZsb2F0KDAsIC1hc2NlbnQsIGFkdmFuY2UsIGFzY2VudCArIGRlc2NlbnQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRlc2NlbnQgb2YgdGhpcyBHcmFwaGljQXR0cmlidXRlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRlc2NlbnQgb2YgdGhpcyBHcmFwaGljQXR0cmlidXRlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBmbG9hdCBnZXREZXNjZW50KCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBHbHlwaEp1c3RpZmljYXRpb25JbmZvIG9mIHRoaXMgR3JhcGhpY0F0dHJpYnV0ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBHbHlwaEp1c3RpZmljYXRpb25JbmZvIG9mIHRoaXMgR3JhcGhpY0F0dHJpYnV0ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBnZXRKdXN0aWZpY2F0aW9uSW5mbygpIHsKLQotICAgICAgICAvKgotICAgICAgICAgKiBEZWZhdWx0IGltcGxlbWVudGF0aW9uLiBTaW5jZSBkb2N1bWVudGF0aW9uIGRvZXNuJ3QgZGVzY3JpYmUgZGVmYXVsdAotICAgICAgICAgKiB2YWx1ZXMsIHRoZXkgd2VyZSBjYWxjdWxhdGVkIGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW9yIGFuZCBjYW4gYmUKLSAgICAgICAgICogb2J0YWluZWQgdXNpbmcgbmV4dCB0ZXN0IHNhbXBsZTogLy8gQ3JlYXRlIEdyYXBoaWNBdHRyaWJ1dGUgY2xhc3MKLSAgICAgICAgICogaW1wbGVtZW50YXRpb24gcHVibGljIGNsYXNzIE15R3JhcGhpY0F0dHJpYnV0ZSBleHRlbmRzCi0gICAgICAgICAqIEdyYXBoaWNBdHRyaWJ1dGUgeyBwcm90ZWN0ZWQgTXlHcmFwaGljQXR0cmlidXRlKGludCBhbGlnbikgewotICAgICAgICAgKiBzdXBlcihhbGlnbik7IH0gcHVibGljIGZsb2F0IGdldERlc2NlbnQoKSB7IHJldHVybiAwOyB9IHB1YmxpYyBmbG9hdAotICAgICAgICAgKiBnZXRBZHZhbmNlKCkgeyByZXR1cm4gMTsgfSBwdWJsaWMgdm9pZCBkcmF3KEdyYXBoaWNzMkQgZzIsIGZsb2F0IHgsCi0gICAgICAgICAqIGZsb2F0IHkpIHsgfSBwdWJsaWMgZmxvYXQgZ2V0QXNjZW50KCkgeyByZXR1cm4gMDsgfSB9Ci0gICAgICAgICAqIE15R3JhcGhpY0F0dHJpYnV0ZSBteUdBID0gZ2F0Lm5ldyBNeUdyYXBoaWNBdHRyaWJ1dGUoMCk7IC8vIHByaW50Ci0gICAgICAgICAqIGp1c3RpZmljYXRpb24gcGFyYW1ldGVycwotICAgICAgICAgKiBTeXN0ZW0ub3V0LnByaW50bG4obXlHQS5nZXRKdXN0aWZpY2F0aW9uSW5mbygpLmdyb3dBYnNvcmIpOwotICAgICAgICAgKiBTeXN0ZW0ub3V0LnByaW50bG4obXlHQS5nZXRKdXN0aWZpY2F0aW9uSW5mbygpLnNocmlua0Fic29yYik7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihteUdBLmdldEp1c3RpZmljYXRpb25JbmZvKCkuZ3Jvd0xlZnRMaW1pdCk7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihteUdBLmdldEp1c3RpZmljYXRpb25JbmZvKCkuZ3Jvd1ByaW9yaXR5KTsKLSAgICAgICAgICogU3lzdGVtLm91dC5wcmludGxuKG15R0EuZ2V0SnVzdGlmaWNhdGlvbkluZm8oKS5ncm93UmlnaHRMaW1pdCk7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihteUdBLmdldEp1c3RpZmljYXRpb25JbmZvKCkuc2hyaW5rTGVmdExpbWl0KTsKLSAgICAgICAgICogU3lzdGVtLm91dC5wcmludGxuKG15R0EuZ2V0SnVzdGlmaWNhdGlvbkluZm8oKS5zaHJpbmtQcmlvcml0eSk7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihteUdBLmdldEp1c3RpZmljYXRpb25JbmZvKCkuc2hyaW5rUmlnaHRMaW1pdCk7Ci0gICAgICAgICAqIFN5c3RlbS5vdXQucHJpbnRsbihteUdBLmdldEp1c3RpZmljYXRpb25JbmZvKCkud2VpZ2h0KTsKLSAgICAgICAgICovCi0gICAgICAgIGZsb2F0IGFkdmFuY2UgPSBnZXRBZHZhbmNlKCk7Ci0gICAgICAgIHJldHVybiBuZXcgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyhhZHZhbmNlLCBmYWxzZSwKLSAgICAgICAgICAgICAgICBHbHlwaEp1c3RpZmljYXRpb25JbmZvLlBSSU9SSVRZX0lOVEVSQ0hBUiwgYWR2YW5jZSAvIDMsIGFkdmFuY2UgLyAzLCBmYWxzZSwKLSAgICAgICAgICAgICAgICBHbHlwaEp1c3RpZmljYXRpb25JbmZvLlBSSU9SSVRZX1dISVRFU1BBQ0UsIDAsIDApOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ZvbnQvSW1hZ2VHcmFwaGljQXR0cmlidXRlLmphdmEgYi9hd3QvamF2YS9hd3QvZm9udC9JbWFnZUdyYXBoaWNBdHRyaWJ1dGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDZkNDc1OC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZm9udC9JbWFnZUdyYXBoaWNBdHRyaWJ1dGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE4NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3MyRDsKLWltcG9ydCBqYXZhLmF3dC5JbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lm1pc2MuSGFzaENvZGU7Ci0KLS8qKgotICogVGhlIEltYWdlR3JhcGhpY0F0dHJpYnV0ZSBjbGFzcyBwcm92aWRlcyBhbiBvcHBvcnR1bml0eSB0byBpbnNlcnQgaW1hZ2VzIHRvIGEKLSAqIHRleHQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgZmluYWwgY2xhc3MgSW1hZ2VHcmFwaGljQXR0cmlidXRlIGV4dGVuZHMgR3JhcGhpY0F0dHJpYnV0ZSB7Ci0KLSAgICAvLyBJbWFnZSBvYmplY3QgcmVuZGVyZWQgYnkgdGhpcyBJbWFnZUdyYXBoaWNBdHRyaWJ1dGUKLSAgICAvKioKLSAgICAgKiBUaGUgaW1hZ2UuCi0gICAgICovCi0gICAgcHJpdmF0ZSBJbWFnZSBmSW1hZ2U7Ci0KLSAgICAvLyBYIGNvb3JkaW5hdGUgb2YgdGhlIG9yaWdpbiBwb2ludAotICAgIC8qKgotICAgICAqIFRoZSBvcmlnaW4geC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGZPcmlnaW5YOwotCi0gICAgLy8gWSBjb29yZGluYXRlIG9mIHRoZSBvcmlnaW4gcG9pbnQKLSAgICAvKioKLSAgICAgKiBUaGUgb3JpZ2luIHkuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmbG9hdCBmT3JpZ2luWTsKLQotICAgIC8vIHRoZSB3aWR0aCBvZiB0aGUgaW1hZ2Ugb2JqZWN0Ci0gICAgLyoqCi0gICAgICogVGhlIGltZyB3aWR0aC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGZJbWdXaWR0aDsKLQotICAgIC8vIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlIG9iamVjdAotICAgIC8qKgotICAgICAqIFRoZSBpbWcgaGVpZ2h0LgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgZkltZ0hlaWdodDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJbWFnZUdyYXBoaWNBdHRyaWJ1dGUgd2l0aCB0aGUgc3BlY2lmaWVkIGltYWdlLAotICAgICAqIGFsaWdubWVudCBhbmQgb3JpZ2lucy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZSB0byBiZSByZW5kZXJlZCBieSBJbWFnZUdyYXBoaWNBdHRyaWJ1dGUuCi0gICAgICogQHBhcmFtIGFsaWdubWVudAotICAgICAqICAgICAgICAgICAgdGhlIGFsaWdubWVudCBvZiB0aGUgSW1hZ2VHcmFwaGljQXR0cmlidXRlLgotICAgICAqIEBwYXJhbSBvcmlnaW5YCi0gICAgICogICAgICAgICAgICB0aGUgb3JpZ2luIFggY29vcmRpbmF0ZSBpbiB0aGUgaW1hZ2Ugb2YgSW1hZ2VHcmFwaGljQXR0cmlidXRlLgotICAgICAqIEBwYXJhbSBvcmlnaW5ZCi0gICAgICogICAgICAgICAgICB0aGUgb3JpZ2luIFkgY29vcmRpbmF0ZSBpbiB0aGUgaW1hZ2Ugb2YgSW1hZ2VHcmFwaGljQXR0cmlidXRlLgotICAgICAqLwotICAgIHB1YmxpYyBJbWFnZUdyYXBoaWNBdHRyaWJ1dGUoSW1hZ2UgaW1hZ2UsIGludCBhbGlnbm1lbnQsIGZsb2F0IG9yaWdpblgsIGZsb2F0IG9yaWdpblkpIHsKLSAgICAgICAgc3VwZXIoYWxpZ25tZW50KTsKLQotICAgICAgICB0aGlzLmZJbWFnZSA9IGltYWdlOwotICAgICAgICB0aGlzLmZPcmlnaW5YID0gb3JpZ2luWDsKLSAgICAgICAgdGhpcy5mT3JpZ2luWSA9IG9yaWdpblk7Ci0KLSAgICAgICAgdGhpcy5mSW1nV2lkdGggPSBmSW1hZ2UuZ2V0V2lkdGgobnVsbCk7Ci0gICAgICAgIHRoaXMuZkltZ0hlaWdodCA9IGZJbWFnZS5nZXRIZWlnaHQobnVsbCk7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VHcmFwaGljQXR0cmlidXRlIHdpdGggdGhlIHNwZWNpZmllZCBpbWFnZSBhbmQKLSAgICAgKiBhbGlnbm1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2UgdG8gYmUgcmVuZGVyZWQgYnkgSW1hZ2VHcmFwaGljQXR0cmlidXRlLgotICAgICAqIEBwYXJhbSBhbGlnbm1lbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbGlnbm1lbnQgb2YgdGhlIEltYWdlR3JhcGhpY0F0dHJpYnV0ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VHcmFwaGljQXR0cmlidXRlKEltYWdlIGltYWdlLCBpbnQgYWxpZ25tZW50KSB7Ci0gICAgICAgIHRoaXMoaW1hZ2UsIGFsaWdubWVudCwgMCwgMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIGhhc2ggY29kZSBvZiB0aGlzIEltYWdlR3JhcGhpY0F0dHJpYnV0ZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaGFzaCBjb2RlIG9mIHRoaXMgSW1hZ2VHcmFwaGljQXR0cmlidXRlIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGhhc2hDb2RlKCkgewotICAgICAgICBIYXNoQ29kZSBoYXNoID0gbmV3IEhhc2hDb2RlKCk7Ci0KLSAgICAgICAgaGFzaC5hcHBlbmQoZkltYWdlLmhhc2hDb2RlKCkpOwotICAgICAgICBoYXNoLmFwcGVuZChnZXRBbGlnbm1lbnQoKSk7Ci0gICAgICAgIHJldHVybiBoYXNoLmhhc2hDb2RlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgdGhlIHNwZWNpZmllZCBJbWFnZUdyYXBoaWNBdHRyaWJ1dGUgb2JqZWN0IHdpdGggdGhpcwotICAgICAqIEltYWdlR3JhcGhpY0F0dHJpYnV0ZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGlnYQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlR3JhcGhpY0F0dHJpYnV0ZSBvYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIEltYWdlR3JhcGhpY0F0dHJpYnV0ZSBvYmplY3QgaXMgZXF1YWwgdG8KLSAgICAgKiAgICAgICAgIHRoaXMgSW1hZ2VHcmFwaGljQXR0cmlidXRlIG9iamVjdCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhJbWFnZUdyYXBoaWNBdHRyaWJ1dGUgaWdhKSB7Ci0gICAgICAgIGlmIChpZ2EgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGlnYSA9PSB0aGlzKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiAoZk9yaWdpblggPT0gaWdhLmZPcmlnaW5YICYmIGZPcmlnaW5ZID09IGlnYS5mT3JpZ2luWQotICAgICAgICAgICAgICAgICYmIGdldEFsaWdubWVudCgpID09IGlnYS5nZXRBbGlnbm1lbnQoKSAmJiBmSW1hZ2UuZXF1YWxzKGlnYS5mSW1hZ2UpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wYXJlcyB0aGUgc3BlY2lmaWVkIE9iamVjdCB3aXRoIHRoaXMgSW1hZ2VHcmFwaGljQXR0cmlidXRlIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb2JqCi0gICAgICogICAgICAgICAgICB0aGUgT2JqZWN0IHRvIGJlIGNvbXBhcmVkLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCBPYmplY3QgaXMgZXF1YWwgdG8gdGhpcwotICAgICAqICAgICAgICAgSW1hZ2VHcmFwaGljQXR0cmlidXRlIG9iamVjdCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3Qgb2JqKSB7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gZXF1YWxzKChJbWFnZUdyYXBoaWNBdHRyaWJ1dGUpb2JqKTsKLSAgICAgICAgfSBjYXRjaCAoQ2xhc3NDYXN0RXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhdyhHcmFwaGljczJEIGcyLCBmbG9hdCB4LCBmbG9hdCB5KSB7Ci0gICAgICAgIGcyLmRyYXdJbWFnZShmSW1hZ2UsIChpbnQpKHggLSBmT3JpZ2luWCksIChpbnQpKHkgLSBmT3JpZ2luWSksIG51bGwpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRBZHZhbmNlKCkgewotICAgICAgICByZXR1cm4gTWF0aC5tYXgoMCwgZkltZ1dpZHRoIC0gZk9yaWdpblgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRBc2NlbnQoKSB7Ci0gICAgICAgIHJldHVybiBNYXRoLm1heCgwLCBmT3JpZ2luWSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kcygpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBSZWN0YW5nbGUyRC5GbG9hdCgtZk9yaWdpblgsIC1mT3JpZ2luWSwgZkltZ1dpZHRoLCBmSW1nSGVpZ2h0KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXQgZ2V0RGVzY2VudCgpIHsKLSAgICAgICAgcmV0dXJuIE1hdGgubWF4KDAsIGZJbWdIZWlnaHQgLSBmT3JpZ2luWSk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZm9udC9MaW5lQnJlYWtNZWFzdXJlci5qYXZhIGIvYXd0L2phdmEvYXd0L2ZvbnQvTGluZUJyZWFrTWVhc3VyZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDgwMDA5My4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZm9udC9MaW5lQnJlYWtNZWFzdXJlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjM4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5mb250OwotCi1pbXBvcnQgamF2YS50ZXh0LkF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvcjsgLy8/Pz9BV1Q6IGltcG9ydCBqYXZhLnRleHQuQnJlYWtJdGVyYXRvcjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBjbGFzcyBMaW5lQnJlYWtNZWFzdXJlciBwcm92aWRlcyBtZXRob2RzIHRvIG1lYXN1cmUgdGhlIGdyYXBoaWNhbAotICogcmVwcmVzZW50YXRpb24gb2YgYSB0ZXh0IGluIG9yZGVyIHRvIGRldGVybWluZSB3aGVyZSB0byBhZGQgbGluZSBicmVha3Mgc28KLSAqIHRoZSByZXN1bHRpbmcgbGluZSBvZiB0ZXh0IGZpdHMgaXRzIHdyYXBwaW5nIHdpZHRoLiBUaGUgd3JhcHBpbmcgd2lkdGgKLSAqIGRlZmluZXMgdGhlIHZpc3VhbCB3aWR0aCBvZiB0aGUgcGFyYWdyYXBoLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIExpbmVCcmVha01lYXN1cmVyIHsKLQotICAgIC8qKgotICAgICAqIFRoZSB0bS4KLSAgICAgKi8KLSAgICBwcml2YXRlIFRleHRNZWFzdXJlciB0bSA9IG51bGw7Ci0KLSAgICAvLyA/Pz9BV1QgcHJpdmF0ZSBCcmVha0l0ZXJhdG9yIGJpID0gbnVsbDsKLSAgICAvKioKLSAgICAgKiBUaGUgcG9zaXRpb24uCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgcG9zaXRpb24gPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIG1heHBvcy4KLSAgICAgKi8KLSAgICBpbnQgbWF4cG9zID0gMDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBMaW5lQnJlYWtNZWFzdXJlciBvYmplY3QgZm9yIHRoZSBzcGVjaWZpZWQgdGV4dC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdGV4dAotICAgICAqICAgICAgICAgICAgdGhlIEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvciBvYmplY3Qgd2hpY2ggY29udGFpbnMgdGV4dAotICAgICAqICAgICAgICAgICAgd2l0aCBhdCBsZWFzdCBvbmUgY2hhcmFjdGVyLgotICAgICAqIEBwYXJhbSBmcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBGb250UmVuZGVyQ29udGV4dCByZXByZXNlbnRlZCBpbmZvcm1hdGlvbiBhYm91dCBncmFwaGljCi0gICAgICogICAgICAgICAgICBkZXZpY2UuCi0gICAgICovCi0gICAgcHVibGljIExpbmVCcmVha01lYXN1cmVyKEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvciB0ZXh0LCBGb250UmVuZGVyQ29udGV4dCBmcmMpIHsKLSAgICAgICAgLy8gPz8/QVdUOiB0aGlzKHRleHQsIEJyZWFrSXRlcmF0b3IuZ2V0TGluZUluc3RhbmNlKCksIGZyYyk7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiA/Pz9BV1QgcHVibGljIExpbmVCcmVha01lYXN1cmVyKCBBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgdGV4dCwKLSAgICAgKiBCcmVha0l0ZXJhdG9yIGJpLCBGb250UmVuZGVyQ29udGV4dCBmcmMgKSB7IHRtID0gbmV3IFRleHRNZWFzdXJlcih0ZXh0LAotICAgICAqIGZyYyk7IHRoaXMuYmkgPSBiaTsgdGhpcy5iaS5zZXRUZXh0KHRleHQpOyBwb3NpdGlvbiA9Ci0gICAgICogdGV4dC5nZXRCZWdpbkluZGV4KCk7IG1heHBvcyA9IHRtLmFjaS5nZXRFbmRJbmRleCgpOyB9Ci0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBEZWxldGVzIGEgY2hhcmFjdGVyIGZyb20gdGhlIHNwZWNpZmllZCBwb3NpdGlvbiBvZiB0aGUgdGV4dCwgdXBkYXRlcyB0aGlzCi0gICAgICogTGluZUJyZWFrTWVhc3VyZXIgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBuZXdUZXh0Ci0gICAgICogICAgICAgICAgICB0aGUgbmV3IHRleHQuCi0gICAgICogQHBhcmFtIHBvcwotICAgICAqICAgICAgICAgICAgdGhlIHBvc2l0aW9uIG9mIHRoZSBjaGFyYWN0ZXIgd2hpY2ggaXMgZGVsZXRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBkZWxldGVDaGFyKEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvciBuZXdUZXh0LCBpbnQgcG9zKSB7Ci0gICAgICAgIHRtLmRlbGV0ZUNoYXIobmV3VGV4dCwgcG9zKTsKLSAgICAgICAgLy8gPz8/QVdUOiBiaS5zZXRUZXh0KG5ld1RleHQpOwotCi0gICAgICAgIHBvc2l0aW9uID0gbmV3VGV4dC5nZXRCZWdpbkluZGV4KCk7Ci0KLSAgICAgICAgbWF4cG9zLS07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBjdXJyZW50IHBvc2l0aW9uIG9mIHRoaXMgTGluZUJyZWFrTWVhc3VyZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY3VycmVudCBwb3NpdGlvbiBvZiB0aGlzIExpbmVCcmVha01lYXN1cmVyCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRQb3NpdGlvbigpIHsKLSAgICAgICAgcmV0dXJuIHBvc2l0aW9uOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc2VydHMgYSBjaGFyYWN0ZXIgYXQgdGhlIHNwZWNpZmllZCBwb3NpdGlvbiBpbiB0aGUgdGV4dCwgdXBkYXRlcyB0aGlzCi0gICAgICogTGluZUJyZWFrTWVhc3VyZXIgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBuZXdUZXh0Ci0gICAgICogICAgICAgICAgICB0aGUgbmV3IHRleHQuCi0gICAgICogQHBhcmFtIHBvcwotICAgICAqICAgICAgICAgICAgdGhlIHBvc2l0aW9uIG9mIHRoZSBjaGFyYWN0ZXIgd2hpY2ggaXMgaW5zZXJ0ZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgaW5zZXJ0Q2hhcihBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgbmV3VGV4dCwgaW50IHBvcykgewotICAgICAgICB0bS5pbnNlcnRDaGFyKG5ld1RleHQsIHBvcyk7Ci0gICAgICAgIC8vID8/P0FXVDogYmkuc2V0VGV4dChuZXdUZXh0KTsKLQotICAgICAgICBwb3NpdGlvbiA9IG5ld1RleHQuZ2V0QmVnaW5JbmRleCgpOwotCi0gICAgICAgIG1heHBvcysrOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIG5leHQgbGluZSBvZiB0ZXh0LCB1cGRhdGVzIGN1cnJlbnQgcG9zaXRpb24gaW4gdGhpcwotICAgICAqIExpbmVCcmVha01lYXN1cmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3cmFwcGluZ1dpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgbWF4aW11bSB2aXNpYmxlIGxpbmUgd2lkdGguCi0gICAgICogQHBhcmFtIG9mZnNldExpbWl0Ci0gICAgICogICAgICAgICAgICB0aGUgbGltaXQgcG9pbnQgd2l0aGluIHRoZSB0ZXh0IGluZGljYXRpbmcgdGhhdCBubyBmdXJ0aGVyCi0gICAgICogICAgICAgICAgICB0ZXh0IHNob3VsZCBiZSBpbmNsdWRlZCBvbiB0aGUgbGluZTsgdGhlIHBhcmFncmFwaCBicmVhay4KLSAgICAgKiBAcGFyYW0gcmVxdWlyZU5leHRXb3JkCi0gICAgICogICAgICAgICAgICBpZiB0cnVlLCBudWxsIGlzIHJldHVybmVkICh0aGUgZW50aXJlIHdvcmQgYXQgdGhlIGN1cnJlbnQKLSAgICAgKiAgICAgICAgICAgIHBvc2l0aW9uIGRvZXMgbm90IGZpdCB3aXRoaW4gdGhlIHdyYXBwaW5nIHdpZHRoKTsgaWYgZmFsc2UsIGEKLSAgICAgKiAgICAgICAgICAgIHZhbGlkIGxheW91dCBpcyByZXR1cm5lZCB0aGF0IGluY2x1ZGVzIGF0IGxlYXN0IHRoZSBjaGFyYWN0ZXIKLSAgICAgKiAgICAgICAgICAgIGF0IHRoZSBjdXJyZW50IHBvc2l0aW9uLgotICAgICAqIEByZXR1cm4gdGhlIG5leHQgVGV4dExheW91dCB3aGljaCBiZWdpbnMgYXQgdGhlIGN1cnJlbnQgcG9zaXRpb24gYW5kCi0gICAgICogICAgICAgICByZXByZXNlbnRzIHRoZSBuZXh0IGxpbmUgb2YgdGV4dCB3aXRoIHdpZHRoIHdyYXBwaW5nV2lkdGgsIG51bGwKLSAgICAgKiAgICAgICAgIGlzIHJldHVybmVkIGlmIHRoZSBlbnRpcmUgd29yZCBhdCB0aGUgY3VycmVudCBwb3NpdGlvbiBkb2VzIG5vdAotICAgICAqICAgICAgICAgZml0IHdpdGhpbiB0aGUgd3JhcHBpbmcgd2lkdGguCi0gICAgICovCi0gICAgcHVibGljIFRleHRMYXlvdXQgbmV4dExheW91dChmbG9hdCB3cmFwcGluZ1dpZHRoLCBpbnQgb2Zmc2V0TGltaXQsIGJvb2xlYW4gcmVxdWlyZU5leHRXb3JkKSB7Ci0gICAgICAgIGlmIChwb3NpdGlvbiA9PSBtYXhwb3MpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IG5leHRQb3NpdGlvbiA9IG5leHRPZmZzZXQod3JhcHBpbmdXaWR0aCwgb2Zmc2V0TGltaXQsIHJlcXVpcmVOZXh0V29yZCk7Ci0KLSAgICAgICAgaWYgKG5leHRQb3NpdGlvbiA9PSBwb3NpdGlvbikgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLSAgICAgICAgVGV4dExheW91dCBsYXlvdXQgPSB0bS5nZXRMYXlvdXQocG9zaXRpb24sIG5leHRQb3NpdGlvbik7Ci0gICAgICAgIHBvc2l0aW9uID0gbmV4dFBvc2l0aW9uOwotICAgICAgICByZXR1cm4gbGF5b3V0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIG5leHQgbGluZSBvZiB0ZXh0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB3cmFwcGluZ1dpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgbWF4aW11bSB2aXNpYmxlIGxpbmUgd2lkdGguCi0gICAgICogQHJldHVybiB0aGUgbmV4dCBsaW5lIG9mIHRleHQuCi0gICAgICovCi0gICAgcHVibGljIFRleHRMYXlvdXQgbmV4dExheW91dChmbG9hdCB3cmFwcGluZ1dpZHRoKSB7Ci0gICAgICAgIHJldHVybiBuZXh0TGF5b3V0KHdyYXBwaW5nV2lkdGgsIG1heHBvcywgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGVuZCBwb3NpdGlvbiBvZiB0aGUgbmV4dCBsaW5lIG9mIHRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdyYXBwaW5nV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBtYXhpbXVtIHZpc2libGUgbGluZSB3aWR0aC4KLSAgICAgKiBAcmV0dXJuIHRoZSBlbmQgcG9zaXRpb24gb2YgdGhlIG5leHQgbGluZSBvZiB0ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgbmV4dE9mZnNldChmbG9hdCB3cmFwcGluZ1dpZHRoKSB7Ci0gICAgICAgIHJldHVybiBuZXh0T2Zmc2V0KHdyYXBwaW5nV2lkdGgsIG1heHBvcywgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGVuZCBwb3NpdGlvbiBvZiB0aGUgbmV4dCBsaW5lIG9mIHRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdyYXBwaW5nV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBtYXhpbXVtIHZpc2libGUgbGluZSB3aWR0aC4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0TGltaXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBsaW1pdCBwb2ludCB3aXRoaW5nIHRoZSB0ZXh0IGluZGljYXRpbmcgdGhhdCBubyBmdXJ0aGVyCi0gICAgICogICAgICAgICAgICB0ZXh0IHNob3VsZCBiZSBpbmNsdWRlZCBvbiB0aGUgbGluZTsgdGhlIHBhcmFncmFwaCBicmVhay4KLSAgICAgKiBAcGFyYW0gcmVxdWlyZU5leHRXb3JkCi0gICAgICogICAgICAgICAgICBpZiB0cnVlLCB0aGUgY3VycmVudCBwb3NpdGlvbiBpcyByZXR1cm5lZCBpZiB0aGUgZW50aXJlIG5leHQKLSAgICAgKiAgICAgICAgICAgIHdvcmQgZG9lcyBub3QgZml0IHdpdGhpbiB3cmFwcGluZ1dpZHRoOyBpZiBmYWxzZSwgdGhlIG9mZnNldAotICAgICAqICAgICAgICAgICAgcmV0dXJuZWQgaXMgYXQgbGVhc3Qgb25lIGdyZWF0ZXIgdGhhbiB0aGUgY3VycmVudCBwb3NpdGlvbi4KLSAgICAgKiBAcmV0dXJuIHRoZSBlbmQgcG9zaXRpb24gb2YgdGhlIG5leHQgbGluZSBvZiB0ZXh0LgotICAgICAqIEB0aHJvd3MgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIG9mZnNldExpbWl0IGlzIGxlc3MgdGhhbiB0aGUgY3VycmVudCBwb3NpdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IG5leHRPZmZzZXQoZmxvYXQgd3JhcHBpbmdXaWR0aCwgaW50IG9mZnNldExpbWl0LCBib29sZWFuIHJlcXVpcmVOZXh0V29yZCkgewotICAgICAgICBpZiAob2Zmc2V0TGltaXQgPD0gcG9zaXRpb24pIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMDM9T2Zmc2V0IGxpbWl0IHNob3VsZCBiZSBncmVhdGVyIHRoYW4gY3VycmVudCBwb3NpdGlvbi4KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjAzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAocG9zaXRpb24gPT0gbWF4cG9zKSB7Ci0gICAgICAgICAgICByZXR1cm4gcG9zaXRpb247Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgYnJlYWtQb3MgPSB0bS5nZXRMaW5lQnJlYWtJbmRleChwb3NpdGlvbiwgd3JhcHBpbmdXaWR0aCk7Ci0gICAgICAgIGludCBjb3JyZWN0ZWRQb3MgPSBicmVha1BvczsKLQotICAgICAgICAvLyBUaGlzIGNoZWNrIGlzIHJlcXVpcmVkIGJlY2F1c2UgYmkucHJlY2VkaW5nKG1heHBvcykgdGhyb3dzIGFuCi0gICAgICAgIC8vIGV4Y2VwdGlvbgotICAgICAgICAvKgotICAgICAgICAgKiA/Pz9BV1QgaWYgKGJyZWFrUG9zID09IG1heHBvcykgeyBjb3JyZWN0ZWRQb3MgPSBtYXhwb3M7IH0gZWxzZSBpZgotICAgICAgICAgKiAoQ2hhcmFjdGVyLmlzV2hpdGVzcGFjZShiaS5nZXRUZXh0KCkuc2V0SW5kZXgoYnJlYWtQb3MpKSkgewotICAgICAgICAgKiBjb3JyZWN0ZWRQb3MgPSBiaS5mb2xsb3dpbmcoYnJlYWtQb3MpOyB9IGVsc2UgeyBjb3JyZWN0ZWRQb3MgPQotICAgICAgICAgKiBiaS5wcmVjZWRpbmcoYnJlYWtQb3MpOyB9Ci0gICAgICAgICAqLwotCi0gICAgICAgIGlmIChwb3NpdGlvbiA+PSBjb3JyZWN0ZWRQb3MpIHsKLSAgICAgICAgICAgIGlmIChyZXF1aXJlTmV4dFdvcmQpIHsKLSAgICAgICAgICAgICAgICBjb3JyZWN0ZWRQb3MgPSBwb3NpdGlvbjsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgY29ycmVjdGVkUG9zID0gTWF0aC5tYXgocG9zaXRpb24gKyAxLCBicmVha1Bvcyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gTWF0aC5taW4oY29ycmVjdGVkUG9zLCBvZmZzZXRMaW1pdCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbmV3IHBvc2l0aW9uIG9mIHRoaXMgTGluZUJyZWFrTWVhc3VyZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBvcwotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBwb3NpdGlvbiBvZiB0aGlzIExpbmVCcmVha01lYXN1cmVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFBvc2l0aW9uKGludCBwb3MpIHsKLSAgICAgICAgaWYgKHRtLmFjaS5nZXRCZWdpbkluZGV4KCkgPiBwb3MgfHwgbWF4cG9zIDwgcG9zKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMzM9aW5kZXggaXMgb3V0IG9mIHJhbmdlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjMzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgcG9zaXRpb24gPSBwb3M7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ZvbnQvTGluZU1ldHJpY3MuamF2YSBiL2F3dC9qYXZhL2F3dC9mb250L0xpbmVNZXRyaWNzLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDRiMDNlNWQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ZvbnQvTGluZU1ldHJpY3MuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDExNiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5mb250OwotCi0vKioKLSAqIFRoZSBMaW5lTWV0cmljcyBjbGFzcyBwcm92aWRlcyBpbmZvcm1hdGlvbiBzdWNoIGFzIGNvbmNlcm5pbmcgaG93IHRoZSB0ZXh0IGlzCi0gKiBwb3NpdGlvbmVkIHdpdGggcmVzcGVjdCB0byB0aGUgYmFzZSBsaW5lLCBzdWNoIGFzIGFzY2VudCwgZGVzY2VudCwgYW5kCi0gKiBsZWFkaW5nLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIExpbmVNZXRyaWNzIHsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJhc2VsaW5lIG9mZnNldHMgb2YgdGhlIHRleHQgYWNjb3JkaW5nIHRvIHRoZSB0aGUgYmFzZWxpbmUgb2YKLSAgICAgKiB0aGlzIHRleHQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYmFzZWxpbmUgb2Zmc2V0cyBvZiB0aGUgdGV4dCBhY2NvcmRpbmcgdG8gdGhlIHRoZSBiYXNlbGluZSBvZgotICAgICAqICAgICAgICAgdGhpcyB0ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBmbG9hdFtdIGdldEJhc2VsaW5lT2Zmc2V0cygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMgb2YgdGhlIHRleHQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMgb2YgdGhlIHRleHQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXROdW1DaGFycygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYmFzZWxpbmUgaW5kZXgsIHJldHVybnMgb25lIG9mIHRoZSBmb2xsb3dpbmcgaW5kZXg6Ci0gICAgICogUk9NQU5fQkFTRUxJTkUsIENFTlRFUl9CQVNFTElORSwgSEFOR0lOR19CQVNFTElORS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBiYXNlbGluZSBpbmRleDogUk9NQU5fQkFTRUxJTkUsIENFTlRFUl9CQVNFTElORSBvcgotICAgICAqICAgICAgICAgSEFOR0lOR19CQVNFTElORS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgaW50IGdldEJhc2VsaW5lSW5kZXgoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRoaWNrbmVzcyBvZiB0aGUgdW5kZXJsaW5lLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHRoaWNrbmVzcyBvZiB0aGUgdW5kZXJsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBmbG9hdCBnZXRVbmRlcmxpbmVUaGlja25lc3MoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG9mZnNldCBvZiB0aGUgdW5kZXJsaW5lLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG9mZnNldCBvZiB0aGUgdW5kZXJsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBmbG9hdCBnZXRVbmRlcmxpbmVPZmZzZXQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRoaWNrbmVzcyBvZiBzdHJpa2UgdGhyb3VnaCBsaW5lLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHRoaWNrbmVzcyBvZiBzdHJpa2UgdGhyb3VnaCBsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBmbG9hdCBnZXRTdHJpa2V0aHJvdWdoVGhpY2tuZXNzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBvZmZzZXQgb2YgdGhlIHN0cmlrZSB0aHJvdWdoIGxpbmUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgb2Zmc2V0IG9mIHRoZSBzdHJpa2UgdGhyb3VnaCBsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBmbG9hdCBnZXRTdHJpa2V0aHJvdWdoT2Zmc2V0KCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBsZWFkaW5nIG9mIHRoZSB0ZXh0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGxlYWRpbmcgb2YgdGhlIHRleHQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGZsb2F0IGdldExlYWRpbmcoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGhlaWdodCBvZiB0aGUgdGV4dCBhcyBhIHN1bSBvZiB0aGUgYXNjZW50LCB0aGUgZGVzY2VudCBhbmQgdGhlCi0gICAgICogbGVhZGluZy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBoZWlnaHQgb2YgdGhlIHRleHQgYXMgYSBzdW0gb2YgdGhlIGFzY2VudCwgdGhlIGRlc2NlbnQgYW5kCi0gICAgICogICAgICAgICB0aGUgbGVhZGluZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZmxvYXQgZ2V0SGVpZ2h0KCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkZXNjZW50IG9mIHRoZSB0ZXh0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRlc2NlbnQgb2YgdGhlIHRleHQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGZsb2F0IGdldERlc2NlbnQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFzY2VudCBvZiB0aGUgdGV4dC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhc2NlbnQgb2YgdGhlIHRleHQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGZsb2F0IGdldEFzY2VudCgpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZm9udC9NdWx0aXBsZU1hc3Rlci5qYXZhIGIvYXd0L2phdmEvYXd0L2ZvbnQvTXVsdGlwbGVNYXN0ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDI2NGYyNC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZm9udC9NdWx0aXBsZU1hc3Rlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsOTEgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJbHlhIFMuIE9rb21pbgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZm9udDsKLQotaW1wb3J0IGphdmEuYXd0LkZvbnQ7Ci0KLS8qKgotICogVGhlIE11bHRpcGxlTWFzdGVyIGludGVyZmFjZSBwcm92aWRlcyBtZXRob2RzIHRvIG1hbmlwdWxhdGUgTXVsdGlwbGVNYXN0ZXIKLSAqIHR5cGUgZm9udHMgYW5kIHJldHJpZXZlIGdyYXBoaWNhbCBhbmQgZGVzaWduIGRhdGEgZnJvbSB0aGVtLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBNdWx0aXBsZU1hc3RlciB7Ci0KLSAgICAvKioKLSAgICAgKiBEZXJpdmVzIGEgbmV3IG11bHRpcGxlIG1hc3RlciBmb250IGJhc2VkIG9uIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2x5cGhXaWR0aHMKLSAgICAgKiAgICAgICAgICAgIGZsb2F0IGFycmF5IHdoaWNoIHJlcHJlc2VudHMgd2lkdGggb2YgZWFjaCBnbHlwaCBpbiBmb250Ci0gICAgICogICAgICAgICAgICBzcGFjZS4KLSAgICAgKiBAcGFyYW0gYXZnU3RlbVdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgYXZlcmFnZSBzdGVtIHdpZHRoIGluIGZvbnQgc3BhY2UuCi0gICAgICogQHBhcmFtIHR5cGljYWxDYXBIZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSB0eXBpY2FsIHVwcGVyIGNhc2UgY2hhciBoZWlnaHQuCi0gICAgICogQHBhcmFtIHR5cGljYWxYSGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgdHlwaWNhbCBsb3dlciBjYXNlIGNoYXIgaGVpZ2h0LgotICAgICAqIEBwYXJhbSBpdGFsaWNBbmdsZQotICAgICAqICAgICAgICAgICAgdGhlIHNsb3BlIGFuZ2xlIGZvciBpdGFsaWNzLgotICAgICAqIEByZXR1cm4gYSBNdWx0aXBsZU1hc3RlciBmb250LgotICAgICAqLwotICAgIHB1YmxpYyBGb250IGRlcml2ZU1NRm9udChmbG9hdFtdIGdseXBoV2lkdGhzLCBmbG9hdCBhdmdTdGVtV2lkdGgsIGZsb2F0IHR5cGljYWxDYXBIZWlnaHQsCi0gICAgICAgICAgICBmbG9hdCB0eXBpY2FsWEhlaWdodCwgZmxvYXQgaXRhbGljQW5nbGUpOwotCi0gICAgLyoqCi0gICAgICogRGVyaXZlcyBhIG5ldyBtdWx0aXBsZSBtYXN0ZXIgZm9udCBiYXNlZCBvbiB0aGUgZGVzaWduIGF4aXMgdmFsdWVzCi0gICAgICogY29udGFpbmVkIGluIHRoZSBzcGVjaWZpZWQgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGF4ZXMKLSAgICAgKiAgICAgICAgICAgIGFuIGZsb2F0IGFycmF5IHdoaWNoIGNvbnRhaW5zIGF4aXMgdmFsdWVzLgotICAgICAqIEByZXR1cm4gYSBNdWx0aXBsZU1hc3RlciBmb250LgotICAgICAqLwotICAgIHB1YmxpYyBGb250IGRlcml2ZU1NRm9udChmbG9hdFtdIGF4ZXMpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyBkZWZhdWx0IGRlc2lnbiB2YWx1ZXMgZm9yIHRoZSBheGVzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRlZmF1bHQgZGVzaWduIHZhbHVlcyBmb3IgdGhlIGF4ZXMuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0W10gZ2V0RGVzaWduQXhpc0RlZmF1bHRzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhcnJheSBvZiBkZXNpZ24gYXhpcyBuYW1lcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBkZXNpZ24gYXhpcyBuYW1lcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0RGVzaWduQXhpc05hbWVzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhcnJheSBvZiBkZXNpZ24gYXhpcyByYW5nZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgZGVzaWduIGF4aXMgcmFuZ2VzLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldERlc2lnbkF4aXNSYW5nZXMoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG51bWJlciBvZiBtdWx0aXBsZSBtYXN0ZXIgZGVzaWduIGNvbnRyb2xzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiBtdWx0aXBsZSBtYXN0ZXIgZGVzaWduIGNvbnRyb2xzLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TnVtRGVzaWduQXhlcygpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZm9udC9PcGVuVHlwZS5qYXZhIGIvYXd0L2phdmEvYXd0L2ZvbnQvT3BlblR5cGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGI2NjkxMS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZm9udC9PcGVuVHlwZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDE4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmZvbnQ7Ci0KLS8qKgotICogVGhlIE9wZW5UeXBlIGludGVyZmFjZSBwcm92aWRlcyBjb25zdGFudHMgYW5kIG1ldGhvZHMgZm9yIGdldHRpbmcgaW5zdGFuY2UKLSAqIGRhdGEgZm9yIGZvbnRzIG9mIHR5cGUgT3BlblR5cGUgYW5kIFRydWVUeXBlLiBGb3IgbW9yZSBpbmZvcm1hdGlvbiwgc2VlIHRoZQotICogPGEKLSAqIGhyZWY9Imh0dHA6Ly9wYXJ0bmVycy5hZG9iZS5jb20vcHVibGljL2RldmVsb3Blci9vcGVudHlwZS9pbmRleF9zcGVjLmh0bWwiPgotICogT3BlblR5cGUgc3BlY2lmaWNhdGlvbjwvYT4uCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIE9wZW5UeXBlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfQUNOVCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0FDTlQgPSAxNjMzOTA2MjkyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19BVkFSIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfQVZBUiA9IDE2MzUxNDgxNDY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0JBU0UgaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19CQVNFID0gMTExMTU3NzQxMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfQkRBVCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0JEQVQgPSAxNjUwNzQ1NzE2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19CTE9DIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfQkxPQyA9IDE2NTEyNzM1NzE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0JTTE4gaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19CU0xOID0gMTY1MTczMTU2NjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfQ0ZGIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfQ0ZGID0gMTEyODY3ODk0NDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfQ01BUCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0NNQVAgPSAxNjY4MTEyNzUyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19DVkFSIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfQ1ZBUiA9IDE2Njg3MDI1Nzg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0NWVCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0NWVCA9IDE2Njg3MDczNjA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0RTSUcgaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19EU0lHID0gMTE0NjMwODkzNTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfRUJEVCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0VCRFQgPSAxMTYxOTcwNzcyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19FQkxDIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfRUJMQyA9IDExNjE5NzI4MDM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0VCU0MgaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19FQlNDID0gMTE2MTk3NDU5NTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfRkRTQyBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0ZEU0MgPSAxNzE3ODU5MTcxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19GRUFUIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfRkVBVCA9IDE3MTc5MjAxMTY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0ZNVFggaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19GTVRYID0gMTcxODQ0OTI3MjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfRlBHTSBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0ZQR00gPSAxNzE4NjQyNTQxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19GVkFSIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfRlZBUiA9IDE3MTkwMzQyMjY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0dBU1AgaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19HQVNQID0gMTczNDQzOTc5MjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfR0RFRiBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0dERUYgPSAxMTk1NjU2NTE4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19HTFlGIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfR0xZRiA9IDE3MzUxNjIyMTQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0dQT1MgaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19HUE9TID0gMTE5NjQ0NTUyMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfR1NVQiBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0dTVUIgPSAxMTk2NjQzNjUwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19HVkFSIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfR1ZBUiA9IDE3MzU4MTE0NDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0hETVggaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19IRE1YID0gMTc1MTQxMjA4ODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfSEVBRCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0hFQUQgPSAxNzUxNDc0NTMyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19ISEVBIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfSEhFQSA9IDE3NTE2NzIxNjE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0hNVFggaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19ITVRYID0gMTc1MjAwMzcwNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfSlNURiBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0pTVEYgPSAxMjQ2OTc1MDQ2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19KVVNUIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfSlVTVCA9IDE3ODYwODIxNjQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0tFUk4gaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19LRVJOID0gMTgwMTgxMDU0MjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfTENBUiBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX0xDQVIgPSAxODE4NDUyMzM4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19MT0NBIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfTE9DQSA9IDE4MTkyMzkyNjU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX0xUU0ggaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19MVFNIID0gMTI4MDU5NDc2MDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfTUFYUCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX01BWFAgPSAxODM1MTA0MzY4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19NTUZYIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfTU1GWCA9IDEyOTY5MDk5MTI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX01NU0QgaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19NTVNEID0gMTI5NjkxMzIyMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfTU9SVCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX01PUlQgPSAxODM2MDIwMzQwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19OQU1FIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfTkFNRSA9IDE4NTE4Nzg3NTc7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX09QQkQgaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19PUEJEID0gMTgzNjAyMDM0MDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfT1MyIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfT1MyID0gMTMzMDg1MTYzNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfUENMVCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX1BDTFQgPSAxMzQ2NTg3NzMyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19QT1NUIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfUE9TVCA9IDE4ODYzNTIyNDQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX1BSRVAgaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19QUkVQID0gMTg4NjU0NTI2NDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfUFJPUCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX1BST1AgPSAxODg2NTQ3ODI0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19UUkFLIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfVFJBSyA9IDE5NTM2NTMwOTk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX1RZUDEgaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19UWVAxID0gMTk1NDExNTYzMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUQUdfVkRNWCBpbmRpY2F0ZXMgY29ycmVzcG9uZGluZyB0YWJsZSB0YWcgaW4gdGhlIE9wZW4gVHlwZQotICAgICAqIFNwZWNpZmljYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVEFHX1ZETVggPSAxNDQ3MzE2ODI0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRBR19WSEVBIGluZGljYXRlcyBjb3JyZXNwb25kaW5nIHRhYmxlIHRhZyBpbiB0aGUgT3BlbiBUeXBlCi0gICAgICogU3BlY2lmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUQUdfVkhFQSA9IDE5ODY1NTMxODU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVEFHX1ZNVFggaW5kaWNhdGVzIGNvcnJlc3BvbmRpbmcgdGFibGUgdGFnIGluIHRoZSBPcGVuIFR5cGUKLSAgICAgKiBTcGVjaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRBR19WTVRYID0gMTk4Njg4NDcyODsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIE9wZW5UeXBlIGZvbnQgdmVyc2lvbi4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB0aGUgT3BlblR5cGUgZm9udCB2ZXJzaW9uLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VmVyc2lvbigpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdGFibGUgZm9yIGEgc3BlY2lmaWVkIHRhZy4gU2ZudCB0YWJsZXMgaW5jbHVkZSBjbWFwLCBuYW1lIGFuZAotICAgICAqIGhlYWQgaXRlbXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNmbnRUYWcKLSAgICAgKiAgICAgICAgICAgIHRoZSBzZm50IHRhZy4KLSAgICAgKiBAcmV0dXJuIGEgYnl0ZSBhcnJheSBjb250YWlucyB0aGUgZm9udCBkYXRhIGNvcnJlc3BvbmRpbmcgdG8gdGhlCi0gICAgICogICAgICAgICBzcGVjaWZpZWQgdGFnLgotICAgICAqLwotICAgIHB1YmxpYyBieXRlW10gZ2V0Rm9udFRhYmxlKGludCBzZm50VGFnKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRhYmxlIGZvciBhIHNwZWNpZmllZCB0YWcuIFNmbnQgdGFibGVzIGluY2x1ZGUgY21hcCwgbmFtZSBhbmQKLSAgICAgKiBoZWFkIGl0ZW1zLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzZm50VGFnCi0gICAgICogICAgICAgICAgICB0aGUgc2ZudCB0YWcuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBvZiB0aGUgcmV0dXJuZWQgdGFibGUuCi0gICAgICogQHBhcmFtIGNvdW50Ci0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIHJldHVybmVkIHRhYmxlLgotICAgICAqIEByZXR1cm4gdGhlIHRhYmxlIGNvcnJlc3BvbmRpbmcgdG8gc2ZudFRhZyBhbmQgY29udGFpbmluZyB0aGUgYnl0ZXMKLSAgICAgKiAgICAgICAgIHN0YXJ0aW5nIGF0IG9mZnNldCBieXRlIGFuZCBpbmNsdWRpbmcgY291bnQgYnl0ZXMuCi0gICAgICovCi0gICAgcHVibGljIGJ5dGVbXSBnZXRGb250VGFibGUoaW50IHNmbnRUYWcsIGludCBvZmZzZXQsIGludCBjb3VudCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0YWJsZSBmb3IgYSBzcGVjaWZpZWQgdGFnLiBTZm50IHRhYmxlcyBpbmNsdWRlIGNtYXAsIG5hbWUgYW5kCi0gICAgICogaGVhZCBpdGVtcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3RyU2ZudFRhZwotICAgICAqICAgICAgICAgICAgdGhlIHN0ciBzZm50IHRhZyBhcyBhIFN0cmluZy4KLSAgICAgKiBAcmV0dXJuIGEgYnl0ZSBhcnJheSBjb250YWlucyB0aGUgZm9udCBkYXRhIGNvcnJlc3BvbmRpbmcgdG8gdGhlCi0gICAgICogICAgICAgICBzcGVjaWZpZWQgdGFnLgotICAgICAqLwotICAgIHB1YmxpYyBieXRlW10gZ2V0Rm9udFRhYmxlKFN0cmluZyBzdHJTZm50VGFnKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRhYmxlIGZvciBhIHNwZWNpZmllZCB0YWcuIFNmbnQgdGFibGVzIGluY2x1ZGUgY21hcCwgbmFtZSBhbmQKLSAgICAgKiBoZWFkIGl0ZW1zLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHJTZm50VGFnCi0gICAgICogICAgICAgICAgICB0aGUgc2ZudCB0YWcgYXMgYSBTdHJpbmcuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBvZiB0aGUgcmV0dXJuZWQgdGFibGUuCi0gICAgICogQHBhcmFtIGNvdW50Ci0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIHJldHVybmVkIHRhYmxlLgotICAgICAqIEByZXR1cm4gdGhlIHRhYmxlIGNvcnJlc3BvbmRpbmcgdG8gc2ZudFRhZyBhbmQgY29udGFpbmluZyB0aGUgYnl0ZXMKLSAgICAgKiAgICAgICAgIHN0YXJ0aW5nIGF0IG9mZnNldCBieXRlIGFuZCBpbmNsdWRpbmcgY291bnQgYnl0ZXMuCi0gICAgICovCi0gICAgcHVibGljIGJ5dGVbXSBnZXRGb250VGFibGUoU3RyaW5nIHN0clNmbnRUYWcsIGludCBvZmZzZXQsIGludCBjb3VudCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0YWJsZSBzaXplIGZvciBhIHNwZWNpZmllZCB0YWcuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0clNmbnRUYWcKLSAgICAgKiAgICAgICAgICAgIHRoZSBzZm50IHRhZyBhcyBhIFN0cmluZy4KLSAgICAgKiBAcmV0dXJuIHRoZSB0YWJsZSBzaXplIGZvciBhIHNwZWNpZmllZCB0YWcuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRGb250VGFibGVTaXplKFN0cmluZyBzdHJTZm50VGFnKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRhYmxlIHNpemUgZm9yIGEgc3BlY2lmaWVkIHRhZy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2ZudFRhZwotICAgICAqICAgICAgICAgICAgdGhlIHNmbnQgdGFnLgotICAgICAqIEByZXR1cm4gdGhlIHRhYmxlIHNpemUgZm9yIGEgc3BlY2lmaWVkIHRhZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEZvbnRUYWJsZVNpemUoaW50IHNmbnRUYWcpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZm9udC9TaGFwZUdyYXBoaWNBdHRyaWJ1dGUuamF2YSBiL2F3dC9qYXZhL2F3dC9mb250L1NoYXBlR3JhcGhpY0F0dHJpYnV0ZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxODJiZmZkLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9mb250L1NoYXBlR3JhcGhpY0F0dHJpYnV0ZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjA2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmZvbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5CYXNpY1N0cm9rZTsKLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljczJEOwotaW1wb3J0IGphdmEuYXd0LlNoYXBlOwotaW1wb3J0IGphdmEuYXd0LlN0cm9rZTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lm1pc2MuSGFzaENvZGU7Ci0KLS8qKgotICogVGhlIFNoYXBlR3JhcGhpY0F0dHJpYnV0ZSBjbGFzcyBwcm92aWRlcyBhbiBvcHBvcnR1bml0eSB0byBpbnNlcnQgc2hhcGVzIHRvIGEKLSAqIHRleHQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgZmluYWwgY2xhc3MgU2hhcGVHcmFwaGljQXR0cmlidXRlIGV4dGVuZHMgR3JhcGhpY0F0dHJpYnV0ZSB7Ci0KLSAgICAvLyBzaGFwZSB0byByZW5kZXIKLSAgICAvKioKLSAgICAgKiBUaGUgc2hhcGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBTaGFwZSBmU2hhcGU7Ci0KLSAgICAvLyBmbGFnLCBpZiB0aGUgc2hhcGUgc2hvdWxkIGJlIHN0cm9rZWQgKHRydWUpIG9yIGZpbGxlZCAoZmFsc2UpCi0gICAgLyoqCi0gICAgICogVGhlIHN0cm9rZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gZlN0cm9rZTsKLQotICAgIC8vIGJvdW5kcyBvZiB0aGUgc2hhcGUKLSAgICAvKioKLSAgICAgKiBUaGUgYm91bmRzLgotICAgICAqLwotICAgIHByaXZhdGUgUmVjdGFuZ2xlMkQgZkJvdW5kczsKLQotICAgIC8vIFggY29vcmRpbmF0ZSBvZiB0aGUgb3JpZ2luIHBvaW50Ci0gICAgLyoqCi0gICAgICogVGhlIG9yaWdpbiB4LgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgZk9yaWdpblg7Ci0KLSAgICAvLyBZIGNvb3JkaW5hdGUgb2YgdGhlIG9yaWdpbiBwb2ludAotICAgIC8qKgotICAgICAqIFRoZSBvcmlnaW4geS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGZPcmlnaW5ZOwotCi0gICAgLy8gd2lkdGggb2YgdGhlIHNoYXBlCi0gICAgLyoqCi0gICAgICogVGhlIHNoYXBlIHdpZHRoLgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgZlNoYXBlV2lkdGg7Ci0KLSAgICAvLyBoZWlnaHQgb2YgdGhlIHNoYXBlCi0gICAgLyoqCi0gICAgICogVGhlIHNoYXBlIGhlaWdodC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGZTaGFwZUhlaWdodDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTVFJPS0UgaW5kaWNhdGVzIHdoZXRoZXIgdGhlIFNoYXBlIGlzIHN0cm9rZWQgb3Igbm90LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgYm9vbGVhbiBTVFJPS0UgPSB0cnVlOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEZJTEwgaW5kaWNhdGVzIHdoZXRoZXIgdGhlIFNoYXBlIGlzIGZpbGxlZCBvciBub3QuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBib29sZWFuIEZJTEwgPSBmYWxzZTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBTaGFwZUdyYXBoaWNBdHRyaWJ1dGUgb2JqZWN0IGZvciB0aGUgc3BlY2lmaWVkIFNoYXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzaGFwZQotICAgICAqICAgICAgICAgICAgdGhlIHNoYXBlIHRvIGJlIHJlbmRlcmVkIGJ5IHRoaXMgU2hhcGVHcmFwaGljQXR0cmlidXRlLgotICAgICAqIEBwYXJhbSBhbGlnbm1lbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbGlnbm1lbnQgb2YgdGhpcyBTaGFwZUdyYXBoaWNBdHRyaWJ1dGUuCi0gICAgICogQHBhcmFtIHN0cm9rZQotICAgICAqICAgICAgICAgICAgdHJ1ZSBpZiB0aGUgU2hhcGUgaXMgc3Ryb2tlZCwgZmFsc2UgaWYgdGhlIFNoYXBlIGlzIGZpbGxlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgU2hhcGVHcmFwaGljQXR0cmlidXRlKFNoYXBlIHNoYXBlLCBpbnQgYWxpZ25tZW50LCBib29sZWFuIHN0cm9rZSkgewotICAgICAgICBzdXBlcihhbGlnbm1lbnQpOwotCi0gICAgICAgIHRoaXMuZlNoYXBlID0gc2hhcGU7Ci0gICAgICAgIHRoaXMuZlN0cm9rZSA9IHN0cm9rZTsKLQotICAgICAgICB0aGlzLmZCb3VuZHMgPSBmU2hhcGUuZ2V0Qm91bmRzMkQoKTsKLQotICAgICAgICB0aGlzLmZPcmlnaW5YID0gKGZsb2F0KWZCb3VuZHMuZ2V0TWluWCgpOwotICAgICAgICB0aGlzLmZPcmlnaW5ZID0gKGZsb2F0KWZCb3VuZHMuZ2V0TWluWSgpOwotCi0gICAgICAgIHRoaXMuZlNoYXBlV2lkdGggPSAoZmxvYXQpZkJvdW5kcy5nZXRXaWR0aCgpOwotICAgICAgICB0aGlzLmZTaGFwZUhlaWdodCA9IChmbG9hdClmQm91bmRzLmdldEhlaWdodCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBoYXNoIGNvZGUgb2YgdGhpcyBTaGFwZUdyYXBoaWNBdHRyaWJ1dGUgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBoYXNoIGNvZGUgb2YgdGhpcyBTaGFwZUdyYXBoaWNBdHRyaWJ1dGUgb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgaGFzaENvZGUoKSB7Ci0gICAgICAgIEhhc2hDb2RlIGhhc2ggPSBuZXcgSGFzaENvZGUoKTsKLQotICAgICAgICBoYXNoLmFwcGVuZChmU2hhcGUuaGFzaENvZGUoKSk7Ci0gICAgICAgIGhhc2guYXBwZW5kKGdldEFsaWdubWVudCgpKTsKLSAgICAgICAgcmV0dXJuIGhhc2guaGFzaENvZGUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wYXJlcyB0aGlzIFNoYXBlR3JhcGhpY0F0dHJpYnV0ZSBvYmplY3QgdG8gdGhlIHNwZWNpZmllZAotICAgICAqIFNoYXBlR3JhcGhpY0F0dHJpYnV0ZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNnYQotICAgICAqICAgICAgICAgICAgdGhlIFNoYXBlR3JhcGhpY0F0dHJpYnV0ZSBvYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIFNoYXBlR3JhcGhpY0F0dHJpYnV0ZSBvYmplY3QgaXMgZXF1YWwgdG8gdGhlCi0gICAgICogICAgICAgICBzcGVjaWZpZWQgU2hhcGVHcmFwaGljQXR0cmlidXRlIG9iamVjdCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhTaGFwZUdyYXBoaWNBdHRyaWJ1dGUgc2dhKSB7Ci0gICAgICAgIGlmIChzZ2EgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHNnYSA9PSB0aGlzKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiAoZlN0cm9rZSA9PSBzZ2EuZlN0cm9rZSAmJiBnZXRBbGlnbm1lbnQoKSA9PSBzZ2EuZ2V0QWxpZ25tZW50KCkgJiYgZlNoYXBlCi0gICAgICAgICAgICAgICAgLmVxdWFscyhzZ2EuZlNoYXBlKSk7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wYXJlcyB0aGlzIFNoYXBlR3JhcGhpY0F0dHJpYnV0ZSBvYmplY3QgdG8gdGhlIHNwZWNpZmllZCBPYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9iagotICAgICAqICAgICAgICAgICAgdGhlIE9iamVjdCB0byBiZSBjb21wYXJlZC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgU2hhcGVHcmFwaGljQXR0cmlidXRlIG9iamVjdCBpcyBlcXVhbCB0byB0aGUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZCBPYmplY3QsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG9iaikgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIGVxdWFscygoU2hhcGVHcmFwaGljQXR0cmlidXRlKW9iaik7Ci0gICAgICAgIH0gY2F0Y2ggKENsYXNzQ2FzdEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3KEdyYXBoaWNzMkQgZzIsIGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIGF0ID0gQWZmaW5lVHJhbnNmb3JtLmdldFRyYW5zbGF0ZUluc3RhbmNlKHgsIHkpOwotICAgICAgICBpZiAoZlN0cm9rZSA9PSBTVFJPS0UpIHsKLSAgICAgICAgICAgIFN0cm9rZSBvbGRTdHJva2UgPSBnMi5nZXRTdHJva2UoKTsKLSAgICAgICAgICAgIGcyLnNldFN0cm9rZShuZXcgQmFzaWNTdHJva2UoKSk7Ci0gICAgICAgICAgICBnMi5kcmF3KGF0LmNyZWF0ZVRyYW5zZm9ybWVkU2hhcGUoZlNoYXBlKSk7Ci0gICAgICAgICAgICBnMi5zZXRTdHJva2Uob2xkU3Ryb2tlKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGcyLmZpbGwoYXQuY3JlYXRlVHJhbnNmb3JtZWRTaGFwZShmU2hhcGUpKTsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZsb2F0IGdldEFkdmFuY2UoKSB7Ci0gICAgICAgIHJldHVybiBNYXRoLm1heCgwLCBmU2hhcGVXaWR0aCArIGZPcmlnaW5YKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXQgZ2V0QXNjZW50KCkgewotICAgICAgICByZXR1cm4gTWF0aC5tYXgoMCwgLWZPcmlnaW5ZKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzKCkgewotICAgICAgICByZXR1cm4gKFJlY3RhbmdsZTJEKWZCb3VuZHMuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXQgZ2V0RGVzY2VudCgpIHsKLSAgICAgICAgcmV0dXJuIE1hdGgubWF4KDAsIGZTaGFwZUhlaWdodCArIGZPcmlnaW5ZKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9mb250L1RleHRIaXRJbmZvLmphdmEgYi9hd3QvamF2YS9hd3QvZm9udC9UZXh0SGl0SW5mby5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2NDYwZWJhLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9mb250L1RleHRIaXRJbmZvLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMTUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmZvbnQ7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkubWlzYy5IYXNoQ29kZTsKLQotLyoqCi0gKiBUaGUgVGV4dEhpdEluZm8gY2xhc3MgcHJvdmlkZXMgaW5mb3JtYXRpb24gYWJvdXQgYSBjYXJldCBwb3NpdGlvbiBpbiBhIHRleHQKLSAqIG1vZGVsIGZvciBpbnNlcnRpb24gb3IgZGVsZXRpb24gb2YgYSBjaGFyYWN0ZXIgaW4gYSB0ZXh0LiBUaGUgVGV4dEhpdEluZm8KLSAqIGRlZmluZXMgdHdvIGJpYXNlcyBvZiB0aGUgY2hhcmFjdGVyOiBsZWFkaW5nIG9yIHRyYWlsaW5nLiBMZWFkaW5nIHBvc2l0aW9uCi0gKiBtZWFucyB0aGUgbGVmdCBlZGdlIG9mIHRoZSBzcGVjaWZpZWQgY2hhcmFjdGVyIChUZXh0SGl0SW5mby5sZWFkaW5nKDIpIG1ldGhvZAotICogZm9yICJ0ZXh0IiByZXR1cm5zIHRoZSBsZWZ0IHNpZGUgb2YgIngiKS4gVHJhaWxpbmcgcG9zaXRpb24gbWVhbnMgdGhlIHJpZ2h0Ci0gKiBlZGdlIG9mIHRoZSBzcGVjaWZpZWQgY2hhcmFjdGVyIChUZXh0SGl0SW5mby50cmFpbGluZygyKSBtZXRob2QgZm9yICJ0ZXh0IgotICogcmV0dXJucyB0aGUgcmlnaHQgc2lkZSBvZiAieCIpLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIFRleHRIaXRJbmZvIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBjaGFyIGlkeC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBjaGFySWR4OyAvLyBSZXByZXNlbnRzIGNoYXJhY3RlciBpbmRleCBpbiB0aGUgbGluZQotCi0gICAgLyoqCi0gICAgICogVGhlIGlzIHRyYWlsaW5nLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBpc1RyYWlsaW5nOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHRleHQgaGl0IGluZm8uCi0gICAgICogCi0gICAgICogQHBhcmFtIGlkeAotICAgICAqICAgICAgICAgICAgdGhlIGlkeC4KLSAgICAgKiBAcGFyYW0gaXNUcmFpbGluZwotICAgICAqICAgICAgICAgICAgdGhlIGlzIHRyYWlsaW5nLgotICAgICAqLwotICAgIHByaXZhdGUgVGV4dEhpdEluZm8oaW50IGlkeCwgYm9vbGVhbiBpc1RyYWlsaW5nKSB7Ci0gICAgICAgIGNoYXJJZHggPSBpZHg7Ci0gICAgICAgIHRoaXMuaXNUcmFpbGluZyA9IGlzVHJhaWxpbmc7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgdGV4dHVhbCBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBUZXh0SGl0SW5mbyBpbnN0YW5jZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24uCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBTdHJpbmcoIlRleHRIaXRJbmZvWyIgKyBjaGFySWR4ICsgIiwgIiArIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgICAgIChpc1RyYWlsaW5nID8gIlRyYWlsaW5nIiA6ICJMZWFkaW5nIikgKyAiXSIgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQKLSAgICAgICAgKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wYXJlcyB0aGlzIFRleHRIaXRJbmZvIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIG9iamVjdCBpcyBhIFRleHRIaXRJbmZvIG9iamVjdCB3aXRoIHRoZQotICAgICAqICAgICAgICAgc2FtZSBkYXRhIHZhbHVlcyBhcyB0aGlzIFRleHRIaXRJbmZvLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvYmopIHsKLSAgICAgICAgaWYgKG9iaiBpbnN0YW5jZW9mIFRleHRIaXRJbmZvKSB7Ci0gICAgICAgICAgICByZXR1cm4gZXF1YWxzKChUZXh0SGl0SW5mbylvYmopOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wYXJlcyB0aGlzIFRleHRIaXRJbmZvIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgVGV4dEhpdEluZm8gb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB0aGkKLSAgICAgKiAgICAgICAgICAgIHRoZSBUZXh0SGl0SW5mbyBvYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIFRleHRIaXRJbmZvIG9iamVjdCBoYXMgdGhlIHNhbWUgZGF0YSB2YWx1ZXMgYXMgdGhlCi0gICAgICogICAgICAgICBzcGVjaWZpZWQgVGV4dEhpdEluZm8gb2JqZWN0LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKFRleHRIaXRJbmZvIHRoaSkgewotICAgICAgICByZXR1cm4gdGhpICE9IG51bGwgJiYgdGhpLmNoYXJJZHggPT0gY2hhcklkeCAmJiB0aGkuaXNUcmFpbGluZyA9PSBpc1RyYWlsaW5nOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBUZXh0SGl0SW5mbyBvYmplY3Qgd2l0aCBpdHMgY2hhcmFjdGVyIGluZGV4IGF0IHRoZSBzcGVjaWZpZWQKLSAgICAgKiBvZmZzZXQgZnJvbSB0aGUgY2hhcmFjdGVyIGluZGV4IG9mIHRoaXMgVGV4dEhpdEluZm8uCi0gICAgICogCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldC4KLSAgICAgKiBAcmV0dXJuIHRoZSBUZXh0SGl0SW5mby4KLSAgICAgKi8KLSAgICBwdWJsaWMgVGV4dEhpdEluZm8gZ2V0T2Zmc2V0SGl0KGludCBvZmZzZXQpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBUZXh0SGl0SW5mbyhjaGFySWR4ICsgb2Zmc2V0LCBpc1RyYWlsaW5nKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGEgVGV4dEhpdEluZm8gYXNzb2NpYXRlZCB3aXRoIHRoZSBvdGhlciBzaWRlIG9mIHRoZSBpbnNlcnRpb24gcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgb3RoZXIgaGl0LgotICAgICAqLwotICAgIHB1YmxpYyBUZXh0SGl0SW5mbyBnZXRPdGhlckhpdCgpIHsKLSAgICAgICAgcmV0dXJuIGlzVHJhaWxpbmcgPyBuZXcgVGV4dEhpdEluZm8oY2hhcklkeCArIDEsIGZhbHNlKQotICAgICAgICAgICAgICAgIDogbmV3IFRleHRIaXRJbmZvKGNoYXJJZHggLSAxLCB0cnVlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIGxlYWRpbmcgZWRnZSBvZiB0aGUgY2hhcmFjdGVyIGlzIGhpdCwgZmFsc2UgaWYgdGhlCi0gICAgICogdHJhaWxpbmcgZWRnZSBvZiB0aGUgY2hhcmFjdGVyIGlzIGhpdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgdGhlIGxlYWRpbmcgZWRnZSBvZiB0aGUgY2hhcmFjdGVyIGlzIGhpdCwgZmFsc2UgaWYgdGhlCi0gICAgICogICAgICAgICB0cmFpbGluZyBlZGdlIG9mIHRoZSBjaGFyYWN0ZXIgaXMgaGl0LgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzTGVhZGluZ0VkZ2UoKSB7Ci0gICAgICAgIHJldHVybiAhaXNUcmFpbGluZzsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBoYXNoIGNvZGUgdmFsdWUgb2YgdGhpcyBUZXh0SGl0SW5mbyBpbnN0YW5jZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBoYXNoIGNvZGUgdmFsdWUuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgcmV0dXJuIEhhc2hDb2RlLmNvbWJpbmUoY2hhcklkeCwgaXNUcmFpbGluZyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW5zZXJ0aW9uIGluZGV4LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGluc2VydGlvbiBpbmRleDogY2hhcmFjdGVyIGluZGV4IGlmIHRoZSBsZWFkaW5nIGVkZ2UgaXMgaGl0LAotICAgICAqICAgICAgICAgb3IgY2hhcmFjdGVyIGluZGV4ICsgMSBpZiB0aGUgdHJhaWxpbmcgZWRnZSBpcyBoaXQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRJbnNlcnRpb25JbmRleCgpIHsKLSAgICAgICAgcmV0dXJuIGlzVHJhaWxpbmcgPyBjaGFySWR4ICsgMSA6IGNoYXJJZHg7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW5kZXggb2YgdGhlIGNoYXJhY3RlciBoaXQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY2hhcmFjdGVyIGhpdCdzIGluZGV4LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0Q2hhckluZGV4KCkgewotICAgICAgICByZXR1cm4gY2hhcklkeDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgVGV4dEhpdEluZm8gYXNzb2NpYXRlZCB3aXRoIHRoZSB0cmFpbGluZyBlZGdlIG9mIHRoZSBjaGFyYWN0ZXIKLSAgICAgKiBhdCB0aGUgc3BlY2lmaWVkIGNoYXIgaW5kZXguCi0gICAgICogCi0gICAgICogQHBhcmFtIGNoYXJJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGNoYXIgaW5kZXguCi0gICAgICogQHJldHVybiB0aGUgVGV4dEhpdEluZm8gYXNzb2NpYXRlZCB3aXRoIHRoZSB0cmFpbGluZyBlZGdlIG9mIHRoZQotICAgICAqICAgICAgICAgY2hhcmFjdGVyIGF0IHRoZSBzcGVjaWZpZWQgY2hhciBpbmRleC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIFRleHRIaXRJbmZvIHRyYWlsaW5nKGludCBjaGFySW5kZXgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBUZXh0SGl0SW5mbyhjaGFySW5kZXgsIHRydWUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBUZXh0SGl0SW5mbyBvYmplY3QgYXNzb2NpYXRlZCB3aXRoIHRoZSBsZWFkaW5nIGVkZ2Ugb2YgdGhlCi0gICAgICogY2hhcmFjdGVyIGF0IHRoZSBzcGVjaWZpZWQgY2hhciBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2hhckluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgY2hhciBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSBUZXh0SGl0SW5mbyBvYmplY3QgYXNzb2NpYXRlZCB3aXRoIHRoZSBsZWFkaW5nIGVkZ2Ugb2YgdGhlCi0gICAgICogICAgICAgICBjaGFyYWN0ZXIgYXQgdGhlIHNwZWNpZmllZCBjaGFyIGluZGV4LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgVGV4dEhpdEluZm8gbGVhZGluZyhpbnQgY2hhckluZGV4KSB7Ci0gICAgICAgIHJldHVybiBuZXcgVGV4dEhpdEluZm8oY2hhckluZGV4LCBmYWxzZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhICh0cmFpbGluZykgVGV4dEhpdEluZm8gb2JqZWN0IGFzc29jaWF0ZWQgd2l0aCB0aGUgY2hhcmFjdGVyCi0gICAgICogYmVmb3JlIHRoZSBzcGVjaWZpZWQgb2Zmc2V0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQuCi0gICAgICogQHJldHVybiB0aGUgVGV4dEhpdEluZm8gb2JqZWN0IGFzc29jaWF0ZWQgd2l0aCB0aGUgY2hhcmFjdGVyIGJlZm9yZSB0aGUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZCBvZmZzZXQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBUZXh0SGl0SW5mbyBiZWZvcmVPZmZzZXQoaW50IG9mZnNldCkgewotICAgICAgICByZXR1cm4gbmV3IFRleHRIaXRJbmZvKG9mZnNldCAtIDEsIHRydWUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSAobGVhZGluZykgVGV4dEhpdEluZm8gb2JqZWN0IGFzc29jaWF0ZWQgd2l0aCB0aGUgY2hhcmFjdGVyCi0gICAgICogYWZ0ZXIgdGhlIHNwZWNpZmllZCBvZmZzZXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldC4KLSAgICAgKiBAcmV0dXJuIHRoZSBUZXh0SGl0SW5mbyBvYmplY3QgYXNzb2NpYXRlZCB3aXRoIHRoZSBjaGFyYWN0ZXIgYWZ0ZXIgdGhlCi0gICAgICogICAgICAgICBzcGVjaWZpZWQgb2Zmc2V0LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgVGV4dEhpdEluZm8gYWZ0ZXJPZmZzZXQoaW50IG9mZnNldCkgewotICAgICAgICByZXR1cm4gbmV3IFRleHRIaXRJbmZvKG9mZnNldCwgZmFsc2UpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9mb250L1RleHRMYXlvdXQuamF2YSBiL2F3dC9qYXZhL2F3dC9mb250L1RleHRMYXlvdXQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggY2M2ZjBiYS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZm9udC9UZXh0TGF5b3V0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw5MjcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuRm9udDsKLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljczJEOwotaW1wb3J0IGphdmEuYXd0LlNoYXBlOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5HZW5lcmFsUGF0aDsKLWltcG9ydCBqYXZhLnRleHQuQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yOwotaW1wb3J0IGphdmEudGV4dC5BdHRyaWJ1dGVkU3RyaW5nOwotaW1wb3J0IGphdmEudXRpbC5NYXA7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuQmFzaWNNZXRyaWNzOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5DYXJldE1hbmFnZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250LlRleHRNZXRyaWNzQ2FsY3VsYXRvcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuVGV4dFJ1bkJyZWFrZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIFRleHRMYXlvdXQgY2xhc3MgZGVmaW5lcyB0aGUgZ3JhcGhpY2FsIHJlcHJlc2VudGF0aW9uIG9mIGNoYXJhY3RlciBkYXRhLgotICogVGhpcyBjbGFzcyBwcm92aWRlcyBtZXRob2QgZm9yIG9idGFpbmluZyBpbmZvcm1hdGlvbiBhYm91dCBjdXJzb3IgcG9zaXRpb25pbmcKLSAqIGFuZCBtb3ZlbWVudCwgc3BsaXQgY3Vyc29ycyBmb3IgdGV4dCB3aXRoIGRpZmZlcmVudCBkaXJlY3Rpb25zLCBsb2dpY2FsIGFuZAotICogdmlzdWFsIGhpZ2hsaWdodGluZywgbXVsdGlwbGUgYmFzZWxpbmVzLCBoaXRzLCBqdXN0aWZpY2F0aW9uLCBhc2NlbnQsCi0gKiBkZXNjZW50LCBhbmQgYWR2YW5jZSwgYW5kIHJlbmRlcmluZy4gQSBUZXh0TGF5b3V0IG9iamVjdCBjYW4gYmUgcmVuZGVyZWQKLSAqIHVzaW5nIEdyYXBoaWNzIGNvbnRleHQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgZmluYWwgY2xhc3MgVGV4dExheW91dCBpbXBsZW1lbnRzIENsb25lYWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2FyZXRQb2xpY3kgY2xhc3MgcHJvdmlkZXMgYSBwb2xpY3kgZm9yIG9idGFpbmluZyB0aGUgY2FyZXQgbG9jYXRpb24uCi0gICAgICogVGhlIHNpbmdsZSBnZXRTdHJvbmdDYXJldCBtZXRob2Qgc3BlY2lmaWVzIHRoZSBwb2xpY3kuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBjbGFzcyBDYXJldFBvbGljeSB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBDYXJldFBvbGljeS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBDYXJldFBvbGljeSgpIHsKLSAgICAgICAgICAgIC8vIE5vdGhpbmcgdG8gZG8KLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBSZXR1cm5zIHdoaWNoZXZlciBvZiB0aGUgdHdvIHNwZWNpZmllZCBUZXh0SGl0SW5mbyBvYmplY3RzIGhhcyB0aGUKLSAgICAgICAgICogc3Ryb25nZXIgY2FyZXQgKGhpZ2hlciBjaGFyYWN0ZXIgbGV2ZWwpIGluIHRoZSBzcGVjaWZpZWQgVGV4dExheW91dC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBoaXQxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGZpcnN0IFRleHRIaXRJbmZvIG9mIHRoZSBzcGVjaWZpZWQgVGV4dExheW91dC4KLSAgICAgICAgICogQHBhcmFtIGhpdDIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgc2Vjb25kIFRleHRIaXRJbmZvIG9mIHRoZSBzcGVjaWZpZWQgVGV4dExheW91dC4KLSAgICAgICAgICogQHBhcmFtIGxheW91dAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBUZXh0TGF5b3V0LgotICAgICAgICAgKiBAcmV0dXJuIHRoZSBUZXh0SGl0SW5mbyB3aXRoIHRoZSBzdHJvbmdlciBjYXJldC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBUZXh0SGl0SW5mbyBnZXRTdHJvbmdDYXJldChUZXh0SGl0SW5mbyBoaXQxLCBUZXh0SGl0SW5mbyBoaXQyLCBUZXh0TGF5b3V0IGxheW91dCkgewotICAgICAgICAgICAgLy8gU3Ryb25nZXIgaGl0IGlzIHRoZSBvbmUgd2l0aCBncmVhdGVyIGxldmVsLgotICAgICAgICAgICAgLy8gSWYgdGhlIGxldmVsIGlzIHNhbWUsIGxlYWRpbmcgZWRnZSBpcyBzdHJvbmdlci4KLQotICAgICAgICAgICAgaW50IGxldmVsMSA9IGxheW91dC5nZXRDaGFyYWN0ZXJMZXZlbChoaXQxLmdldENoYXJJbmRleCgpKTsKLSAgICAgICAgICAgIGludCBsZXZlbDIgPSBsYXlvdXQuZ2V0Q2hhcmFjdGVyTGV2ZWwoaGl0Mi5nZXRDaGFySW5kZXgoKSk7Ci0KLSAgICAgICAgICAgIGlmIChsZXZlbDEgPT0gbGV2ZWwyKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIChoaXQyLmlzTGVhZGluZ0VkZ2UoKSAmJiAoIWhpdDEuaXNMZWFkaW5nRWRnZSgpKSkgPyBoaXQyIDogaGl0MTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBsZXZlbDEgPiBsZXZlbDIgPyBoaXQxIDogaGl0MjsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IERFRkFVTFRfQ0FSRVRfUE9MSUNZIGluZGljYXRlcyB0aGUgZGVmYXVsdCBjYXJldCBwb2xpY3kuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBUZXh0TGF5b3V0LkNhcmV0UG9saWN5IERFRkFVTFRfQ0FSRVRfUE9MSUNZID0gbmV3IENhcmV0UG9saWN5KCk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYnJlYWtlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIFRleHRSdW5CcmVha2VyIGJyZWFrZXI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbWV0cmljcyB2YWxpZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gbWV0cmljc1ZhbGlkID0gZmFsc2U7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdG1jLgotICAgICAqLwotICAgIHByaXZhdGUgVGV4dE1ldHJpY3NDYWxjdWxhdG9yIHRtYzsKLQotICAgIC8qKgotICAgICAqIFRoZSBtZXRyaWNzLgotICAgICAqLwotICAgIHByaXZhdGUgQmFzaWNNZXRyaWNzIG1ldHJpY3M7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY2FyZXQgbWFuYWdlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIENhcmV0TWFuYWdlciBjYXJldE1hbmFnZXI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUganVzdGlmaWNhdGlvbiB3aWR0aC4KLSAgICAgKi8KLSAgICBmbG9hdCBqdXN0aWZpY2F0aW9uV2lkdGggPSAtMTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBUZXh0TGF5b3V0IG9iamVjdCBmcm9tIHRoZSBzcGVjaWZpZWQgc3RyaW5nIGFuZCBGb250LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHJpbmcKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdHJpbmcgdG8gYmUgZGlzcGxheWVkLgotICAgICAqIEBwYXJhbSBmb250Ci0gICAgICogICAgICAgICAgICB0aGUgZm9udCBvZiB0aGUgdGV4dC4KLSAgICAgKiBAcGFyYW0gZnJjCi0gICAgICogICAgICAgICAgICB0aGUgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0IGZvciBvYnRhaW5pbmcgaW5mb3JtYXRpb24gYWJvdXQgYQotICAgICAqICAgICAgICAgICAgZ3JhcGhpY3MgZGV2aWNlLgotICAgICAqLwotICAgIHB1YmxpYyBUZXh0TGF5b3V0KFN0cmluZyBzdHJpbmcsIEZvbnQgZm9udCwgRm9udFJlbmRlckNvbnRleHQgZnJjKSB7Ci0gICAgICAgIGlmIChzdHJpbmcgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjAxPSd7MH0nIHBhcmFtZXRlciBpcyBudWxsCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjAxIiwgInN0cmluZyIpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoZm9udCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMDE9J3swfScgcGFyYW1ldGVyIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMDEiLCAiZm9udCIpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoc3RyaW5nLmxlbmd0aCgpID09IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4wMj0nezB9JyBwYXJhbWV0ZXIgaGFzIHplcm8gbGVuZ3RoCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjAyIiwgInN0cmluZyIpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICBBdHRyaWJ1dGVkU3RyaW5nIGFzID0gbmV3IEF0dHJpYnV0ZWRTdHJpbmcoc3RyaW5nKTsKLSAgICAgICAgYXMuYWRkQXR0cmlidXRlKFRleHRBdHRyaWJ1dGUuRk9OVCwgZm9udCk7Ci0gICAgICAgIHRoaXMuYnJlYWtlciA9IG5ldyBUZXh0UnVuQnJlYWtlcihhcy5nZXRJdGVyYXRvcigpLCBmcmMpOwotICAgICAgICBjYXJldE1hbmFnZXIgPSBuZXcgQ2FyZXRNYW5hZ2VyKGJyZWFrZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBUZXh0TGF5b3V0IGZyb20gdGhlIHNwZWNpZmllZCB0ZXh0IGFuZCBhIG1hcCBvZgotICAgICAqIGF0dHJpYnV0ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cmluZwotICAgICAqICAgICAgICAgICAgdGhlIHN0cmluZyB0byBiZSBkaXNwbGF5ZWQuCi0gICAgICogQHBhcmFtIGF0dHJpYnV0ZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhdHRyaWJ1dGVzIHRvIGJlIHVzZWQgZm9yIG9idGFpbmluZyB0aGUgdGV4dCBzdHlsZS4KLSAgICAgKiBAcGFyYW0gZnJjCi0gICAgICogICAgICAgICAgICB0aGUgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0IGZvciBvYnRhaW5pbmcgaW5mb3JtYXRpb24gYWJvdXQgYQotICAgICAqICAgICAgICAgICAgZ3JhcGhpY3MgZGV2aWNlLgotICAgICAqLwotICAgIHB1YmxpYyBUZXh0TGF5b3V0KFN0cmluZyBzdHJpbmcsCi0gICAgICAgICAgICBNYXA8PyBleHRlbmRzIGphdmEudGV4dC5BdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IuQXR0cmlidXRlLCA/PiBhdHRyaWJ1dGVzLAotICAgICAgICAgICAgRm9udFJlbmRlckNvbnRleHQgZnJjKSB7Ci0gICAgICAgIGlmIChzdHJpbmcgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjAxPSd7MH0nIHBhcmFtZXRlciBpcyBudWxsCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjAxIiwgInN0cmluZyIpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoYXR0cmlidXRlcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMDE9J3swfScgcGFyYW1ldGVyIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMDEiLCAiYXR0cmlidXRlcyIpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoc3RyaW5nLmxlbmd0aCgpID09IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4wMj0nezB9JyBwYXJhbWV0ZXIgaGFzIHplcm8gbGVuZ3RoCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjAyIiwgInN0cmluZyIpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICBBdHRyaWJ1dGVkU3RyaW5nIGFzID0gbmV3IEF0dHJpYnV0ZWRTdHJpbmcoc3RyaW5nKTsKLSAgICAgICAgYXMuYWRkQXR0cmlidXRlcyhhdHRyaWJ1dGVzLCAwLCBzdHJpbmcubGVuZ3RoKCkpOwotICAgICAgICB0aGlzLmJyZWFrZXIgPSBuZXcgVGV4dFJ1bkJyZWFrZXIoYXMuZ2V0SXRlcmF0b3IoKSwgZnJjKTsKLSAgICAgICAgY2FyZXRNYW5hZ2VyID0gbmV3IENhcmV0TWFuYWdlcihicmVha2VyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgVGV4dExheW91dCBmcm9tIHRoZSBBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRleHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IuCi0gICAgICogQHBhcmFtIGZyYwotICAgICAqICAgICAgICAgICAgdGhlIEZvbnRSZW5kZXJDb250ZXh0IG9iamVjdCBmb3Igb2J0YWluaW5nIGluZm9ybWF0aW9uIGFib3V0IGEKLSAgICAgKiAgICAgICAgICAgIGdyYXBoaWNzIGRldmljZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgVGV4dExheW91dChBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgdGV4dCwgRm9udFJlbmRlckNvbnRleHQgZnJjKSB7Ci0gICAgICAgIGlmICh0ZXh0ID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4wMz0nezB9JyBpdGVyYXRvciBwYXJhbWV0ZXIgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4wMyIsICJ0ZXh0IikpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLSAgICAgICAgfQotCi0gICAgICAgIGlmICh0ZXh0LmdldEJlZ2luSW5kZXgoKSA9PSB0ZXh0LmdldEVuZEluZGV4KCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC4wND0nezB9JyBpdGVyYXRvciBwYXJhbWV0ZXIgaGFzIHplcm8gbGVuZ3RoCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjA0IiwgInRleHQiKSk7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5icmVha2VyID0gbmV3IFRleHRSdW5CcmVha2VyKHRleHQsIGZyYyk7Ci0gICAgICAgIGNhcmV0TWFuYWdlciA9IG5ldyBDYXJldE1hbmFnZXIoYnJlYWtlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHRleHQgbGF5b3V0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBicmVha2VyCi0gICAgICogICAgICAgICAgICB0aGUgYnJlYWtlci4KLSAgICAgKi8KLSAgICBUZXh0TGF5b3V0KFRleHRSdW5CcmVha2VyIGJyZWFrZXIpIHsKLSAgICAgICAgdGhpcy5icmVha2VyID0gYnJlYWtlcjsKLSAgICAgICAgY2FyZXRNYW5hZ2VyID0gbmV3IENhcmV0TWFuYWdlcih0aGlzLmJyZWFrZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBoYXNoIGNvZGUgb2YgdGhpcyBUZXh0TGF5b3V0IG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgaGFzaCBjb2RlIG9mIHRoaXMgVGV4dExheW91dCBvYmplY3QuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgcmV0dXJuIGJyZWFrZXIuaGFzaENvZGUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgY29weSBvZiB0aGlzIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgY29weSBvZiB0aGlzIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgT2JqZWN0IGNsb25lKCkgewotICAgICAgICBUZXh0TGF5b3V0IHJlcyA9IG5ldyBUZXh0TGF5b3V0KChUZXh0UnVuQnJlYWtlcilicmVha2VyLmNsb25lKCkpOwotCi0gICAgICAgIGlmIChqdXN0aWZpY2F0aW9uV2lkdGggPj0gMCkgewotICAgICAgICAgICAgcmVzLmhhbmRsZUp1c3RpZnkoanVzdGlmaWNhdGlvbldpZHRoKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgdGhpcyBUZXh0TGF5b3V0IG9iamVjdCB0byB0aGUgc3BlY2lmaWVkIFRleHRMYXlvdXQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsYXlvdXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBUZXh0TGF5b3V0IG9iamVjdCB0byBiZSBjb21wYXJlZC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgVGV4dExheW91dCBvYmplY3QgaXMgZXF1YWwgdG8gdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgVGV4dExheW91dCBvYmplY3QsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoVGV4dExheW91dCBsYXlvdXQpIHsKLSAgICAgICAgaWYgKGxheW91dCA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHRoaXMuYnJlYWtlci5lcXVhbHMobGF5b3V0LmJyZWFrZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIHRoaXMgVGV4dExheW91dCBvYmplY3QgdG8gdGhlIHNwZWNpZmllZCBPYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9iagotICAgICAqICAgICAgICAgICAgdGhlIE9iamVjdCB0byBiZSBjb21wYXJlZC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgVGV4dExheW91dCBvYmplY3QgaXMgZXF1YWwgdG8gdGhlIHNwZWNpZmllZCBPYmplY3QsCi0gICAgICogICAgICAgICBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvYmopIHsKLSAgICAgICAgcmV0dXJuIG9iaiBpbnN0YW5jZW9mIFRleHRMYXlvdXQgPyBlcXVhbHMoKFRleHRMYXlvdXQpb2JqKSA6IGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN0cmluZyByZXByZXNlbnRhdGlvbiBmb3IgdGhpcyBUZXh0TGF5b3V0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0cmluZyByZXByZXNlbnRhdGlvbiBmb3IgdGhpcyBUZXh0TGF5b3V0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7IC8vIHdoYXQgZm9yPwotICAgICAgICByZXR1cm4gc3VwZXIudG9TdHJpbmcoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyB0aGlzIFRleHRMYXlvdXQgYXQgdGhlIHNwZWNpZmllZCBsb2NhdGlvbiB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBHcmFwaGljczJEIGNvbnRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGcyZAotICAgICAqICAgICAgICAgICAgdGhlIEdyYXBoaWNzMkQgb2JqZWN0IHdoaWNoIHJlbmRlcnMgdGhpcyBUZXh0TGF5b3V0LgotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBUZXh0TGF5b3V0IG9yaWdpbi4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgVGV4dExheW91dCBvcmlnaW4uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZHJhdyhHcmFwaGljczJEIGcyZCwgZmxvYXQgeCwgZmxvYXQgeSkgewotICAgICAgICB1cGRhdGVNZXRyaWNzKCk7Ci0gICAgICAgIGJyZWFrZXIuZHJhd1NlZ21lbnRzKGcyZCwgeCwgeSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVXBkYXRlIG1ldHJpY3MuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHVwZGF0ZU1ldHJpY3MoKSB7Ci0gICAgICAgIGlmICghbWV0cmljc1ZhbGlkKSB7Ci0gICAgICAgICAgICBicmVha2VyLmNyZWF0ZUFsbFNlZ21lbnRzKCk7Ci0gICAgICAgICAgICB0bWMgPSBuZXcgVGV4dE1ldHJpY3NDYWxjdWxhdG9yKGJyZWFrZXIpOwotICAgICAgICAgICAgbWV0cmljcyA9IHRtYy5jcmVhdGVNZXRyaWNzKCk7Ci0gICAgICAgICAgICBtZXRyaWNzVmFsaWQgPSB0cnVlOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYWR2YW5jZSBvZiB0aGlzIFRleHRMYXlvdXQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFkdmFuY2Ugb2YgdGhpcyBUZXh0TGF5b3V0IG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0QWR2YW5jZSgpIHsKLSAgICAgICAgdXBkYXRlTWV0cmljcygpOwotICAgICAgICByZXR1cm4gbWV0cmljcy5nZXRBZHZhbmNlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYXNjZW50IG9mIHRoaXMgVGV4dExheW91dCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXNjZW50IG9mIHRoaXMgVGV4dExheW91dCBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldEFzY2VudCgpIHsKLSAgICAgICAgdXBkYXRlTWV0cmljcygpOwotICAgICAgICByZXR1cm4gbWV0cmljcy5nZXRBc2NlbnQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBiYXNlbGluZSBvZiB0aGlzIFRleHRMYXlvdXQgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJhc2VsaW5lIG9mIHRoaXMgVGV4dExheW91dCBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIGJ5dGUgZ2V0QmFzZWxpbmUoKSB7Ci0gICAgICAgIHVwZGF0ZU1ldHJpY3MoKTsKLSAgICAgICAgcmV0dXJuIChieXRlKW1ldHJpY3MuZ2V0QmFzZUxpbmVJbmRleCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZsb2F0IGFycmF5IG9mIG9mZnNldHMgZm9yIHRoZSBiYXNlbGluZXMgd2hpY2ggYXJlIHVzZWQgaW4gdGhpcwotICAgICAqIFRleHRMYXlvdXQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZmxvYXQgYXJyYXkgb2Ygb2Zmc2V0cyBmb3IgdGhlIGJhc2VsaW5lcyB3aGljaCBhcmUgdXNlZCBpbgotICAgICAqICAgICAgICAgdGhpcyBUZXh0TGF5b3V0LgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldEJhc2VsaW5lT2Zmc2V0cygpIHsKLSAgICAgICAgdXBkYXRlTWV0cmljcygpOwotICAgICAgICByZXR1cm4gdG1jLmdldEJhc2VsaW5lT2Zmc2V0cygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJsYWNrIGJveCBib3VuZHMgb2YgdGhlIGNoYXJhY3RlcnMgaW4gdGhlIHNwZWNpZmllZCBhcmVhLiBUaGUKLSAgICAgKiBibGFjayBib3ggYm91bmRzIGlzIGFuIFNoYXBlIHdoaWNoIGNvbnRhaW5zIGFsbCBib3VuZGluZyBib3hlcyBvZiBhbGwgdGhlCi0gICAgICogZ2x5cGhzIG9mIHRoZSBjaGFyYWN0ZXJzIGJldHdlZW4gZmlyc3RFbmRwb2ludCBhbmQgc2Vjb25kRW5kcG9pbnQKLSAgICAgKiBwYXJhbWV0ZXJzIHZhbHVlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZmlyc3RFbmRwb2ludAotICAgICAqICAgICAgICAgICAgdGhlIGZpcnN0IHBvaW50IG9mIHRoZSBhcmVhLgotICAgICAqIEBwYXJhbSBzZWNvbmRFbmRwb2ludAotICAgICAqICAgICAgICAgICAgdGhlIHNlY29uZCBwb2ludCBvZiB0aGUgYXJlYS4KLSAgICAgKiBAcmV0dXJuIHRoZSBTaGFwZSB3aGljaCBjb250YWlucyBibGFjayBib3ggYm91bmRzLgotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZSBnZXRCbGFja0JveEJvdW5kcyhpbnQgZmlyc3RFbmRwb2ludCwgaW50IHNlY29uZEVuZHBvaW50KSB7Ci0gICAgICAgIHVwZGF0ZU1ldHJpY3MoKTsKLSAgICAgICAgaWYgKGZpcnN0RW5kcG9pbnQgPCBzZWNvbmRFbmRwb2ludCkgewotICAgICAgICAgICAgcmV0dXJuIGJyZWFrZXIuZ2V0QmxhY2tCb3hCb3VuZHMoZmlyc3RFbmRwb2ludCwgc2Vjb25kRW5kcG9pbnQpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBicmVha2VyLmdldEJsYWNrQm94Qm91bmRzKHNlY29uZEVuZHBvaW50LCBmaXJzdEVuZHBvaW50KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZHMgb2YgdGhpcyBUZXh0TGF5b3V0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJvdW5kcyBvZiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kcygpIHsKLSAgICAgICAgdXBkYXRlTWV0cmljcygpOwotICAgICAgICByZXR1cm4gYnJlYWtlci5nZXRWaXN1YWxCb3VuZHMoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGluZm9ybWF0aW9uIGFib3V0IHRoZSBjYXJldCBvZiB0aGUgc3BlY2lmaWVkIFRleHRIaXRJbmZvLgotICAgICAqIAotICAgICAqIEBwYXJhbSBoaXRJbmZvCi0gICAgICogICAgICAgICAgICB0aGUgVGV4dEhpdEluZm8uCi0gICAgICogQHJldHVybiB0aGUgaW5mb3JtYXRpb24gYWJvdXQgdGhlIGNhcmV0IG9mIHRoZSBzcGVjaWZpZWQgVGV4dEhpdEluZm8uCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0W10gZ2V0Q2FyZXRJbmZvKFRleHRIaXRJbmZvIGhpdEluZm8pIHsKLSAgICAgICAgdXBkYXRlTWV0cmljcygpOwotICAgICAgICByZXR1cm4gY2FyZXRNYW5hZ2VyLmdldENhcmV0SW5mbyhoaXRJbmZvKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGluZm9ybWF0aW9uIGFib3V0IHRoZSBjYXJldCBvZiB0aGUgc3BlY2lmaWVkIFRleHRIaXRJbmZvIG9mIGEKLSAgICAgKiBjaGFyYWN0ZXIgaW4gdGhpcyBUZXh0TGF5b3V0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBoaXRJbmZvCi0gICAgICogICAgICAgICAgICB0aGUgVGV4dEhpdEluZm8gb2YgYSBjaGFyYWN0ZXIgaW4gdGhpcyBUZXh0TGF5b3V0LgotICAgICAqIEBwYXJhbSBib3VuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBib3VuZHMgdG8gd2hpY2ggdGhlIGNhcmV0IGluZm8gaXMgY29uc3RydWN0ZWQuCi0gICAgICogQHJldHVybiB0aGUgY2FyZXQgb2YgdGhlIHNwZWNpZmllZCBUZXh0SGl0SW5mby4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXRbXSBnZXRDYXJldEluZm8oVGV4dEhpdEluZm8gaGl0SW5mbywgUmVjdGFuZ2xlMkQgYm91bmRzKSB7Ci0gICAgICAgIHVwZGF0ZU1ldHJpY3MoKTsKLSAgICAgICAgcmV0dXJuIGNhcmV0TWFuYWdlci5nZXRDYXJldEluZm8oaGl0SW5mbyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIFNoYXBlIHdoaWNoIHJlcHJlc2VudHMgdGhlIGNhcmV0IG9mIHRoZSBzcGVjaWZpZWQgVGV4dEhpdEluZm8gaW4KLSAgICAgKiB0aGUgYm91bmRzIG9mIHRoaXMgVGV4dExheW91dC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaGl0SW5mbwotICAgICAqICAgICAgICAgICAgdGhlIFRleHRIaXRJbmZvLgotICAgICAqIEBwYXJhbSBib3VuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBib3VuZHMgdG8gd2hpY2ggdGhlIGNhcmV0IGluZm8gaXMgY29uc3RydWN0ZWQuCi0gICAgICogQHJldHVybiB0aGUgU2hhcGUgd2hpY2ggcmVwcmVzZW50cyB0aGUgY2FyZXQuCi0gICAgICovCi0gICAgcHVibGljIFNoYXBlIGdldENhcmV0U2hhcGUoVGV4dEhpdEluZm8gaGl0SW5mbywgUmVjdGFuZ2xlMkQgYm91bmRzKSB7Ci0gICAgICAgIHVwZGF0ZU1ldHJpY3MoKTsKLSAgICAgICAgcmV0dXJuIGNhcmV0TWFuYWdlci5nZXRDYXJldFNoYXBlKGhpdEluZm8sIHRoaXMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBTaGFwZSB3aGljaCByZXByZXNlbnRzIHRoZSBjYXJldCBvZiB0aGUgc3BlY2lmaWVkIFRleHRIaXRJbmZvIGluCi0gICAgICogdGhlIGJvdW5kcyBvZiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGhpdEluZm8KLSAgICAgKiAgICAgICAgICAgIHRoZSBUZXh0SGl0SW5mby4KLSAgICAgKiBAcmV0dXJuIHRoZSBTaGFwZSB3aGljaCByZXByZXNlbnRzIHRoZSBjYXJldC4KLSAgICAgKi8KLSAgICBwdWJsaWMgU2hhcGUgZ2V0Q2FyZXRTaGFwZShUZXh0SGl0SW5mbyBoaXRJbmZvKSB7Ci0gICAgICAgIHVwZGF0ZU1ldHJpY3MoKTsKLSAgICAgICAgcmV0dXJuIGNhcmV0TWFuYWdlci5nZXRDYXJldFNoYXBlKGhpdEluZm8sIHRoaXMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdHdvIFNoYXBlcyBmb3IgdGhlIHN0cm9uZyBhbmQgd2VhayBjYXJldHMgd2l0aCBkZWZhdWx0IGNhcmV0IHBvbGljeQotICAgICAqIGFuZCBudWxsIGJvdW5kczogdGhlIGZpcnN0IGVsZW1lbnQgaXMgdGhlIHN0cm9uZyBjYXJldCwgdGhlIHNlY29uZCBpcyB0aGUKLSAgICAgKiB3ZWFrIGNhcmV0IG9yIG51bGwuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgYW4gb2Zmc2V0IGluIHRoZSBUZXh0TGF5b3V0LgotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgb2YgdHdvIFNoYXBlcyBjb3JyZXNwb25kZWQgdG8gdGhlIHN0cm9uZyBhbmQgd2VhawotICAgICAqICAgICAgICAgY2FyZXRzLgotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZVtdIGdldENhcmV0U2hhcGVzKGludCBvZmZzZXQpIHsKLSAgICAgICAgcmV0dXJuIGdldENhcmV0U2hhcGVzKG9mZnNldCwgbnVsbCwgVGV4dExheW91dC5ERUZBVUxUX0NBUkVUX1BPTElDWSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0d28gU2hhcGVzIGZvciB0aGUgc3Ryb25nIGFuZCB3ZWFrIGNhcmV0cyB3aXRoIHRoZSBkZWZhdWx0IGNhcmV0Ci0gICAgICogcG9saWN5OiB0aGUgZmlyc3QgZWxlbWVudCBpcyB0aGUgc3Ryb25nIGNhcmV0LCB0aGUgc2Vjb25kIGlzIHRoZSB3ZWFrCi0gICAgICogY2FyZXQgb3IgbnVsbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICBhbiBvZmZzZXQgaW4gdGhlIFRleHRMYXlvdXQuCi0gICAgICogQHBhcmFtIGJvdW5kcwotICAgICAqICAgICAgICAgICAgdGhlIGJvdW5kcyB0byB3aGljaCB0byBleHRlbmQgdGhlIGNhcmV0cy4KLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIHR3byBTaGFwZXMgY29ycmVzcG9uZGVkIHRvIHRoZSBzdHJvbmcgYW5kIHdlYWsKLSAgICAgKiAgICAgICAgIGNhcmV0cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgU2hhcGVbXSBnZXRDYXJldFNoYXBlcyhpbnQgb2Zmc2V0LCBSZWN0YW5nbGUyRCBib3VuZHMpIHsKLSAgICAgICAgcmV0dXJuIGdldENhcmV0U2hhcGVzKG9mZnNldCwgYm91bmRzLCBUZXh0TGF5b3V0LkRFRkFVTFRfQ0FSRVRfUE9MSUNZKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHR3byBTaGFwZXMgZm9yIHRoZSBzdHJvbmcgYW5kIHdlYWsgY2FyZXRzOiB0aGUgZmlyc3QgZWxlbWVudCBpcyB0aGUKLSAgICAgKiBzdHJvbmcgY2FyZXQsIHRoZSBzZWNvbmQgaXMgdGhlIHdlYWsgY2FyZXQgb3IgbnVsbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICBhbiBvZmZzZXQgaW4gdGhlIFRleHRMYXlvdXQuCi0gICAgICogQHBhcmFtIGJvdW5kcwotICAgICAqICAgICAgICAgICAgdGhlIGJvdW5kcyB0byB3aGljaCB0byBleHRlbmQgdGhlIGNhcmV0cy4KLSAgICAgKiBAcGFyYW0gcG9saWN5Ci0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIENhcmV0UG9saWN5LgotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgb2YgdHdvIFNoYXBlcyBjb3JyZXNwb25kZWQgdG8gdGhlIHN0cm9uZyBhbmQgd2VhawotICAgICAqICAgICAgICAgY2FyZXRzLgotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZVtdIGdldENhcmV0U2hhcGVzKGludCBvZmZzZXQsIFJlY3RhbmdsZTJEIGJvdW5kcywgVGV4dExheW91dC5DYXJldFBvbGljeSBwb2xpY3kpIHsKLSAgICAgICAgaWYgKG9mZnNldCA8IDAgfHwgb2Zmc2V0ID4gYnJlYWtlci5nZXRDaGFyQ291bnQoKSkgewotICAgICAgICAgICAgLy8gYXd0LjE5NT1PZmZzZXQgaXMgb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOTUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHVwZGF0ZU1ldHJpY3MoKTsKLSAgICAgICAgcmV0dXJuIGNhcmV0TWFuYWdlci5nZXRDYXJldFNoYXBlcyhvZmZzZXQsIGJvdW5kcywgcG9saWN5LCB0aGlzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBudW1iZXIgb2YgY2hhcmFjdGVycyBpbiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMgaW4gdGhpcyBUZXh0TGF5b3V0LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0Q2hhcmFjdGVyQ291bnQoKSB7Ci0gICAgICAgIHJldHVybiBicmVha2VyLmdldENoYXJDb3VudCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxldmVsIG9mIHRoZSBjaGFyYWN0ZXIgd2l0aCB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBpbmRleCBvZiB0aGUgY2hhcmFjdGVyLgotICAgICAqIEByZXR1cm4gdGhlIGxldmVsIG9mIHRoZSBjaGFyYWN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIGJ5dGUgZ2V0Q2hhcmFjdGVyTGV2ZWwoaW50IGluZGV4KSB7Ci0gICAgICAgIGlmIChpbmRleCA9PSAtMSB8fCBpbmRleCA9PSBnZXRDaGFyYWN0ZXJDb3VudCgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gKGJ5dGUpYnJlYWtlci5nZXRCYXNlTGV2ZWwoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gYnJlYWtlci5nZXRMZXZlbChpbmRleCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGVzY2VudCBvZiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGVzY2VudCBvZiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldERlc2NlbnQoKSB7Ci0gICAgICAgIHVwZGF0ZU1ldHJpY3MoKTsKLSAgICAgICAgcmV0dXJuIG1ldHJpY3MuZ2V0RGVzY2VudCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFRleHRMYXlvdXQgd2ljaCBpcyBqdXN0aWZpZWQgd2l0aCB0aGUgc3BlY2lmaWVkIHdpZHRoIHJlbGF0ZWQgdG8KLSAgICAgKiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGp1c3RpZmljYXRpb25XaWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIHdoaWNoIGlzIHVzZWQgZm9yIGp1c3RpZmljYXRpb24uCi0gICAgICogQHJldHVybiBhIFRleHRMYXlvdXQganVzdGlmaWVkIHRvIHRoZSBzcGVjaWZpZWQgd2lkdGguCi0gICAgICogQHRocm93cyBFcnJvcgotICAgICAqICAgICAgICAgICAgIHRoZSBlcnJvciBvY2N1cmVzIGlmIHRoaXMgVGV4dExheW91dCBoYXMgYmVlbiBhbHJlYWR5Ci0gICAgICogICAgICAgICAgICAganVzdGlmaWVkLgotICAgICAqLwotICAgIHB1YmxpYyBUZXh0TGF5b3V0IGdldEp1c3RpZmllZExheW91dChmbG9hdCBqdXN0aWZpY2F0aW9uV2lkdGgpIHRocm93cyBFcnJvciB7Ci0gICAgICAgIGZsb2F0IGp1c3RpZmljYXRpb24gPSBicmVha2VyLmdldEp1c3RpZmljYXRpb24oKTsKLQotICAgICAgICBpZiAoanVzdGlmaWNhdGlvbiA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4xOTY9SnVzdGlmaWNhdGlvbiBpbXBvc3NpYmxlLCBsYXlvdXQgYWxyZWFkeSBqdXN0aWZpZWQKLSAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOTYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfSBlbHNlIGlmIChqdXN0aWZpY2F0aW9uID09IDApIHsKLSAgICAgICAgICAgIHJldHVybiB0aGlzOwotICAgICAgICB9Ci0KLSAgICAgICAgVGV4dExheW91dCBqdXN0aWZpZWRMYXlvdXQgPSBuZXcgVGV4dExheW91dCgoVGV4dFJ1bkJyZWFrZXIpYnJlYWtlci5jbG9uZSgpKTsKLSAgICAgICAganVzdGlmaWVkTGF5b3V0LmhhbmRsZUp1c3RpZnkoanVzdGlmaWNhdGlvbldpZHRoKTsKLSAgICAgICAgcmV0dXJuIGp1c3RpZmllZExheW91dDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBsZWFkaW5nIG9mIHRoaXMgVGV4dExheW91dC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBsZWFkaW5nIG9mIHRoaXMgVGV4dExheW91dC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0TGVhZGluZygpIHsKLSAgICAgICAgdXBkYXRlTWV0cmljcygpOwotICAgICAgICByZXR1cm4gbWV0cmljcy5nZXRMZWFkaW5nKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIFNoYXBlIHJlcHJlc2VudGluZyB0aGUgbG9naWNhbCBzZWxlY3Rpb24gYmV0d2VlZW4gdGhlIHNwZWNpZmllZAotICAgICAqIGVuZHBvaW50cyBhbmQgZXh0ZW5kZWQgdG8gdGhlIG5hdHVyYWwgYm91bmRzIG9mIHRoaXMgVGV4dExheW91dC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZmlyc3RFbmRwb2ludAotICAgICAqICAgICAgICAgICAgdGhlIGZpcnN0IHNlbGVjdGVkIGVuZHBvaW50IHdpdGhpbiB0aGUgYXJlYSBvZiBjaGFyYWN0ZXJzCi0gICAgICogQHBhcmFtIHNlY29uZEVuZHBvaW50Ci0gICAgICogICAgICAgICAgICB0aGUgc2Vjb25kIHNlbGVjdGVkIGVuZHBvaW50IHdpdGhpbiB0aGUgYXJlYSBvZiBjaGFyYWN0ZXJzCi0gICAgICogQHJldHVybiBhIFNoYXBlIHJlcHJlc2VudGVkIHRoZSBsb2dpY2FsIHNlbGVjdGlvbiBiZXR3ZWVlbiB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBlbmRwb2ludHMuCi0gICAgICovCi0gICAgcHVibGljIFNoYXBlIGdldExvZ2ljYWxIaWdobGlnaHRTaGFwZShpbnQgZmlyc3RFbmRwb2ludCwgaW50IHNlY29uZEVuZHBvaW50KSB7Ci0gICAgICAgIHVwZGF0ZU1ldHJpY3MoKTsKLSAgICAgICAgcmV0dXJuIGdldExvZ2ljYWxIaWdobGlnaHRTaGFwZShmaXJzdEVuZHBvaW50LCBzZWNvbmRFbmRwb2ludCwgYnJlYWtlci5nZXRMb2dpY2FsQm91bmRzKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBTaGFwZSByZXByZXNlbnRpbmcgdGhlIGxvZ2ljYWwgc2VsZWN0aW9uIGJldHdlZWVuIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBlbmRwb2ludHMgYW5kIGV4dGVuZGVkIHRvIHRoZSBzcGVjaWZpZWQgYm91bmRzIG9mIHRoaXMgVGV4dExheW91dC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZmlyc3RFbmRwb2ludAotICAgICAqICAgICAgICAgICAgdGhlIGZpcnN0IHNlbGVjdGVkIGVuZHBvaW50IHdpdGhpbiB0aGUgYXJlYSBvZiBjaGFyYWN0ZXJzCi0gICAgICogQHBhcmFtIHNlY29uZEVuZHBvaW50Ci0gICAgICogICAgICAgICAgICB0aGUgc2Vjb25kIHNlbGVjdGVkIGVuZHBvaW50IHdpdGhpbiB0aGUgYXJlYSBvZiBjaGFyYWN0ZXJzCi0gICAgICogQHBhcmFtIGJvdW5kcwotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBib3VuZHMgb2YgdGhpcyBUZXh0TGF5b3V0LgotICAgICAqIEByZXR1cm4gYSBTaGFwZSByZXByZXNlbnRlZCB0aGUgbG9naWNhbCBzZWxlY3Rpb24gYmV0d2VlZW4gdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgZW5kcG9pbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZSBnZXRMb2dpY2FsSGlnaGxpZ2h0U2hhcGUoaW50IGZpcnN0RW5kcG9pbnQsIGludCBzZWNvbmRFbmRwb2ludCwgUmVjdGFuZ2xlMkQgYm91bmRzKSB7Ci0gICAgICAgIHVwZGF0ZU1ldHJpY3MoKTsKLQotICAgICAgICBpZiAoZmlyc3RFbmRwb2ludCA+IHNlY29uZEVuZHBvaW50KSB7Ci0gICAgICAgICAgICBpZiAoc2Vjb25kRW5kcG9pbnQgPCAwIHx8IGZpcnN0RW5kcG9pbnQgPiBicmVha2VyLmdldENoYXJDb3VudCgpKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjE5Nz1FbmRwb2ludHMgYXJlIG91dCBvZiByYW5nZQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMTk3IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gY2FyZXRNYW5hZ2VyLmdldExvZ2ljYWxIaWdobGlnaHRTaGFwZShzZWNvbmRFbmRwb2ludCwgZmlyc3RFbmRwb2ludCwgYm91bmRzLAotICAgICAgICAgICAgICAgICAgICB0aGlzKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoZmlyc3RFbmRwb2ludCA8IDAgfHwgc2Vjb25kRW5kcG9pbnQgPiBicmVha2VyLmdldENoYXJDb3VudCgpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTk3PUVuZHBvaW50cyBhcmUgb3V0IG9mIHJhbmdlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE5NyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBjYXJldE1hbmFnZXIuZ2V0TG9naWNhbEhpZ2hsaWdodFNoYXBlKGZpcnN0RW5kcG9pbnQsIHNlY29uZEVuZHBvaW50LCBib3VuZHMsIHRoaXMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvZ2ljYWwgcmFuZ2VzIG9mIHRleHQgd2hpY2ggY29ycmVzcG9uZHMgdG8gYSB2aXN1YWwgc2VsZWN0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBoaXQxCi0gICAgICogICAgICAgICAgICB0aGUgZmlyc3QgZW5kcG9pbnQgb2YgdGhlIHZpc3VhbCByYW5nZS4KLSAgICAgKiBAcGFyYW0gaGl0MgotICAgICAqICAgICAgICAgICAgdGhlIHNlY29uZCBlbmRwb2ludCBvZiB0aGUgdmlzdWFsIHJhbmdlLgotICAgICAqIEByZXR1cm4gdGhlIGxvZ2ljYWwgcmFuZ2VzIG9mIHRleHQgd2hpY2ggY29ycmVzcG9uZHMgdG8gYSB2aXN1YWwKLSAgICAgKiAgICAgICAgIHNlbGVjdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50W10gZ2V0TG9naWNhbFJhbmdlc0ZvclZpc3VhbFNlbGVjdGlvbihUZXh0SGl0SW5mbyBoaXQxLCBUZXh0SGl0SW5mbyBoaXQyKSB7Ci0gICAgICAgIHJldHVybiBjYXJldE1hbmFnZXIuZ2V0TG9naWNhbFJhbmdlc0ZvclZpc3VhbFNlbGVjdGlvbihoaXQxLCBoaXQyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBUZXh0SGl0SW5mbyBmb3IgdGhlIG5leHQgY2FyZXQgdG8gdGhlIGxlZnQgKG9yIHVwIGF0IHRoZSBlbmQgb2YKLSAgICAgKiB0aGUgbGluZSkgb2YgdGhlIHNwZWNpZmllZCBvZmZzZXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICogQHJldHVybiB0aGUgVGV4dEhpdEluZm8gZm9yIHRoZSBuZXh0IGNhcmV0IHRvIHRoZSBsZWZ0IChvciB1cCBhdCB0aGUgZW5kCi0gICAgICogICAgICAgICBvZiB0aGUgbGluZSkgb2YgdGhlIHNwZWNpZmllZCBoaXQsIG9yIG51bGwgaWYgdGhlcmUgaXMgbm8gaGl0LgotICAgICAqLwotICAgIHB1YmxpYyBUZXh0SGl0SW5mbyBnZXROZXh0TGVmdEhpdChpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIHJldHVybiBnZXROZXh0TGVmdEhpdChvZmZzZXQsIERFRkFVTFRfQ0FSRVRfUE9MSUNZKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBUZXh0SGl0SW5mbyBmb3IgdGhlIG5leHQgY2FyZXQgdG8gdGhlIGxlZnQgKG9yIHVwIGF0IHRoZSBlbmQgb2YKLSAgICAgKiB0aGUgbGluZSkgb2YgdGhlIHNwZWNpZmllZCBoaXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGhpdEluZm8KLSAgICAgKiAgICAgICAgICAgIHRoZSBpbml0aWFsIGhpdC4KLSAgICAgKiBAcmV0dXJuIHRoZSBUZXh0SGl0SW5mbyBmb3IgdGhlIG5leHQgY2FyZXQgdG8gdGhlIGxlZnQgKG9yIHVwIGF0IHRoZSBlbmQKLSAgICAgKiAgICAgICAgIG9mIHRoZSBsaW5lKSBvZiB0aGUgc3BlY2lmaWVkIGhpdCwgb3IgbnVsbCBpZiB0aGVyZSBpcyBubyBoaXQuCi0gICAgICovCi0gICAgcHVibGljIFRleHRIaXRJbmZvIGdldE5leHRMZWZ0SGl0KFRleHRIaXRJbmZvIGhpdEluZm8pIHsKLSAgICAgICAgYnJlYWtlci5jcmVhdGVBbGxTZWdtZW50cygpOwotICAgICAgICByZXR1cm4gY2FyZXRNYW5hZ2VyLmdldE5leHRMZWZ0SGl0KGhpdEluZm8pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFRleHRIaXRJbmZvIGZvciB0aGUgbmV4dCBjYXJldCB0byB0aGUgbGVmdCAob3IgdXAgYXQgdGhlIGVuZCBvZgotICAgICAqIHRoZSBsaW5lKSBvZiB0aGUgc3BlY2lmaWVkIG9mZnNldCwgZ2l2ZW4gdGhlIHNwZWNpZmllZCBjYXJldCBwb2xpY3kuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICogQHBhcmFtIHBvbGljeQotICAgICAqICAgICAgICAgICAgdGhlIHBvbGljeSB0byBiZSB1c2VkIGZvciBvYnRhaW5pbmcgdGhlIHN0cm9uZyBjYXJldC4KLSAgICAgKiBAcmV0dXJuIHRoZSBUZXh0SGl0SW5mbyBmb3IgdGhlIG5leHQgY2FyZXQgdG8gdGhlIGxlZnQgb2YgdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgb2Zmc2V0LCBvciBudWxsIGlmIHRoZXJlIGlzIG5vIGhpdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgVGV4dEhpdEluZm8gZ2V0TmV4dExlZnRIaXQoaW50IG9mZnNldCwgVGV4dExheW91dC5DYXJldFBvbGljeSBwb2xpY3kpIHsKLSAgICAgICAgaWYgKG9mZnNldCA8IDAgfHwgb2Zmc2V0ID4gYnJlYWtlci5nZXRDaGFyQ291bnQoKSkgewotICAgICAgICAgICAgLy8gYXd0LjE5NT1PZmZzZXQgaXMgb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOTUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIFRleHRIaXRJbmZvIGhpdCA9IFRleHRIaXRJbmZvLmFmdGVyT2Zmc2V0KG9mZnNldCk7Ci0gICAgICAgIFRleHRIaXRJbmZvIHN0cm9uZ0hpdCA9IHBvbGljeS5nZXRTdHJvbmdDYXJldChoaXQsIGhpdC5nZXRPdGhlckhpdCgpLCB0aGlzKTsKLSAgICAgICAgVGV4dEhpdEluZm8gbmV4dExlZnRIaXQgPSBnZXROZXh0TGVmdEhpdChzdHJvbmdIaXQpOwotCi0gICAgICAgIGlmIChuZXh0TGVmdEhpdCAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gcG9saWN5LmdldFN0cm9uZ0NhcmV0KGdldFZpc3VhbE90aGVySGl0KG5leHRMZWZ0SGl0KSwgbmV4dExlZnRIaXQsIHRoaXMpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFRleHRIaXRJbmZvIGZvciB0aGUgbmV4dCBjYXJldCB0byB0aGUgcmlnaHQgKG9yIGRvd24gYXQgdGhlIGVuZAotICAgICAqIG9mIHRoZSBsaW5lKSBvZiB0aGUgc3BlY2lmaWVkIGhpdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaGl0SW5mbwotICAgICAqICAgICAgICAgICAgdGhlIGluaXRpYWwgaGl0LgotICAgICAqIEByZXR1cm4gdGhlIFRleHRIaXRJbmZvIGZvciB0aGUgbmV4dCBjYXJldCB0byB0aGUgcmlnaHQgKG9yIGRvd24gYXQgdGhlCi0gICAgICogICAgICAgICBlbmQgb2YgdGhlIGxpbmUpIG9mIHRoZSBzcGVjaWZpZWQgaGl0LCBvciBudWxsIGlmIHRoZXJlIGlzIG5vCi0gICAgICogICAgICAgICBoaXQuCi0gICAgICovCi0gICAgcHVibGljIFRleHRIaXRJbmZvIGdldE5leHRSaWdodEhpdChUZXh0SGl0SW5mbyBoaXRJbmZvKSB7Ci0gICAgICAgIGJyZWFrZXIuY3JlYXRlQWxsU2VnbWVudHMoKTsKLSAgICAgICAgcmV0dXJuIGNhcmV0TWFuYWdlci5nZXROZXh0UmlnaHRIaXQoaGl0SW5mbyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgVGV4dEhpdEluZm8gZm9yIHRoZSBuZXh0IGNhcmV0IHRvIHRoZSByaWdodCAob3IgZG93biBhdCB0aGUgZW5kCi0gICAgICogb2YgdGhlIGxpbmUpIG9mIHRoZSBzcGVjaWZpZWQgb2Zmc2V0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhpcyBUZXh0TGF5b3V0LgotICAgICAqIEByZXR1cm4gdGhlIFRleHRIaXRJbmZvIGZvciB0aGUgbmV4dCBjYXJldCB0byB0aGUgcmlnaHQgb2YgdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgb2Zmc2V0LCBvciBudWxsIGlmIHRoZXJlIGlzIG5vIGhpdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgVGV4dEhpdEluZm8gZ2V0TmV4dFJpZ2h0SGl0KGludCBvZmZzZXQpIHsKLSAgICAgICAgcmV0dXJuIGdldE5leHRSaWdodEhpdChvZmZzZXQsIERFRkFVTFRfQ0FSRVRfUE9MSUNZKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBUZXh0SGl0SW5mbyBmb3IgdGhlIG5leHQgY2FyZXQgdG8gdGhlIHJpZ2h0IChvciBkb3duIGF0IHRoZSBlbmQKLSAgICAgKiBvZiB0aGUgbGluZSkgb2YgdGhlIHNwZWNpZmllZCBvZmZzZXQsIGdpdmVuIHRoZSBzcGVjaWZpZWQgY2FyZXQgcG9saWN5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhpcyBUZXh0TGF5b3V0LgotICAgICAqIEBwYXJhbSBwb2xpY3kKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb2xpY3kgdG8gYmUgdXNlZCBmb3Igb2J0YWluaW5nIHRoZSBzdHJvbmcgY2FyZXQuCi0gICAgICogQHJldHVybiB0aGUgVGV4dEhpdEluZm8gZm9yIHRoZSBuZXh0IGNhcmV0IHRvIHRoZSByaWdodCBvZiB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBvZmZzZXQsIG9yIG51bGwgaWYgdGhlcmUgaXMgbm8gaGl0LgotICAgICAqLwotICAgIHB1YmxpYyBUZXh0SGl0SW5mbyBnZXROZXh0UmlnaHRIaXQoaW50IG9mZnNldCwgVGV4dExheW91dC5DYXJldFBvbGljeSBwb2xpY3kpIHsKLSAgICAgICAgaWYgKG9mZnNldCA8IDAgfHwgb2Zmc2V0ID4gYnJlYWtlci5nZXRDaGFyQ291bnQoKSkgewotICAgICAgICAgICAgLy8gYXd0LjE5NT1PZmZzZXQgaXMgb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4xOTUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIFRleHRIaXRJbmZvIGhpdCA9IFRleHRIaXRJbmZvLmFmdGVyT2Zmc2V0KG9mZnNldCk7Ci0gICAgICAgIFRleHRIaXRJbmZvIHN0cm9uZ0hpdCA9IHBvbGljeS5nZXRTdHJvbmdDYXJldChoaXQsIGhpdC5nZXRPdGhlckhpdCgpLCB0aGlzKTsKLSAgICAgICAgVGV4dEhpdEluZm8gbmV4dFJpZ2h0SGl0ID0gZ2V0TmV4dFJpZ2h0SGl0KHN0cm9uZ0hpdCk7Ci0KLSAgICAgICAgaWYgKG5leHRSaWdodEhpdCAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gcG9saWN5LmdldFN0cm9uZ0NhcmV0KGdldFZpc3VhbE90aGVySGl0KG5leHRSaWdodEhpdCksIG5leHRSaWdodEhpdCwgdGhpcyk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgb3V0bGluZSBvZiB0aGlzIFRleHRMYXlvdXQgYXMgYSBTaGFwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geGZvcm0KLSAgICAgKiAgICAgICAgICAgIHRoZSBBZmZpbmVUcmFuc2Zvcm0gdG8gYmUgdXNlZCB0byB0cmFuc2Zvcm0gdGhlIG91dGxpbmUgYmVmb3JlCi0gICAgICogICAgICAgICAgICByZXR1cm5pbmcgaXQsIG9yIG51bGwgaWYgbm8gdHJhbnNmb3JtYXRpb24gaXMgZGVzaXJlZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBvdXRsaW5lIG9mIHRoaXMgVGV4dExheW91dCBhcyBhIFNoYXBlLgotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZSBnZXRPdXRsaW5lKEFmZmluZVRyYW5zZm9ybSB4Zm9ybSkgewotICAgICAgICBicmVha2VyLmNyZWF0ZUFsbFNlZ21lbnRzKCk7Ci0KLSAgICAgICAgR2VuZXJhbFBhdGggb3V0bGluZSA9IGJyZWFrZXIuZ2V0T3V0bGluZSgpOwotCi0gICAgICAgIGlmIChvdXRsaW5lICE9IG51bGwgJiYgeGZvcm0gIT0gbnVsbCkgewotICAgICAgICAgICAgb3V0bGluZS50cmFuc2Zvcm0oeGZvcm0pOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG91dGxpbmU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdmlzaWJsZSBhZHZhbmNlIG9mIHRoaXMgVGV4dExheW91dCB3aGljaCBpcyBkZWZpbmVkIGFzIGRpZmZlbmNlCi0gICAgICogYmV0d2VlbiBsZWFkaW5nIChhZHZhbmNlKSBhbmQgdHJhaWxpbmcgd2hpdGVzcGFjZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB2aXNpYmxlIGFkdmFuY2Ugb2YgdGhpcyBUZXh0TGF5b3V0LgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRWaXNpYmxlQWR2YW5jZSgpIHsKLSAgICAgICAgdXBkYXRlTWV0cmljcygpOwotCi0gICAgICAgIC8vIFRyYWlsaW5nIHdoaXRlc3BhY2UgX1NIT1VMRF8gYmUgcmVvcmRlcmVkIChVbmljb2RlIHNwZWMpIHRvCi0gICAgICAgIC8vIGJhc2UgZGlyZWN0aW9uLCBzbyBpdCBpcyBhbHNvIHRyYWlsaW5nCi0gICAgICAgIC8vIGluIGxvZ2ljYWwgcmVwcmVzZW50YXRpb24uIFdlIHVzZSB0aGlzIGZhY3QuCi0gICAgICAgIGludCBsYXN0Tm9uV2hpdGVzcGFjZSA9IGJyZWFrZXIuZ2V0TGFzdE5vbldoaXRlc3BhY2UoKTsKLQotICAgICAgICBpZiAobGFzdE5vbldoaXRlc3BhY2UgPCAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gMDsKLSAgICAgICAgfSBlbHNlIGlmIChsYXN0Tm9uV2hpdGVzcGFjZSA9PSBnZXRDaGFyYWN0ZXJDb3VudCgpIC0gMSkgewotICAgICAgICAgICAgcmV0dXJuIGdldEFkdmFuY2UoKTsKLSAgICAgICAgfSBlbHNlIGlmIChqdXN0aWZpY2F0aW9uV2lkdGggPj0gMCkgeyAvLyBMYXlvdXQgaXMganVzdGlmaWVkCi0gICAgICAgICAgICByZXR1cm4ganVzdGlmaWNhdGlvbldpZHRoOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgYnJlYWtlci5wdXNoU2VnbWVudHMoYnJlYWtlci5nZXRBQ0koKS5nZXRCZWdpbkluZGV4KCksIGxhc3ROb25XaGl0ZXNwYWNlCi0gICAgICAgICAgICAgICAgICAgICsgYnJlYWtlci5nZXRBQ0koKS5nZXRCZWdpbkluZGV4KCkgKyAxKTsKLQotICAgICAgICAgICAgYnJlYWtlci5jcmVhdGVBbGxTZWdtZW50cygpOwotCi0gICAgICAgICAgICBmbG9hdCB2aXNBZHZhbmNlID0gdG1jLmNyZWF0ZU1ldHJpY3MoKS5nZXRBZHZhbmNlKCk7Ci0KLSAgICAgICAgICAgIGJyZWFrZXIucG9wU2VnbWVudHMoKTsKLSAgICAgICAgICAgIHJldHVybiB2aXNBZHZhbmNlOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIFNoYXBlIHdoaWNoIGNvcnJlc3BvbmRzIHRvIHRoZSBoaWdobGlnaHRlZCAoc2VsZWN0ZWQpIGFyZWEgYmFzZWQKLSAgICAgKiBvbiB0d28gaGl0IGxvY2F0aW9ucyB3aXRoaW4gdGhlIHRleHQgYW5kIGV4dGVuZHMgdG8gdGhlIGJvdW5kcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaGl0MQotICAgICAqICAgICAgICAgICAgdGhlIGZpcnN0IHRleHQgaGl0IGxvY2F0aW9uLgotICAgICAqIEBwYXJhbSBoaXQyCi0gICAgICogICAgICAgICAgICB0aGUgc2Vjb25kIHRleHQgaGl0IGxvY2F0aW9uLgotICAgICAqIEBwYXJhbSBib3VuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSByZWN0YW5nbGUgdGhhdCB0aGUgaGlnaGxpZ2h0ZWQgYXJlYSBzaG91bGQgYmUgZXh0ZW5kZWQgb3IKLSAgICAgKiAgICAgICAgICAgIHJlc3RyaWN0ZWQgdG8uCi0gICAgICogQHJldHVybiBhIFNoYXBlIHdoaWNoIGNvcnJlc3BvbmRzIHRvIHRoZSBoaWdobGlnaHRlZCAoc2VsZWN0ZWQpIGFyZWEuCi0gICAgICovCi0gICAgcHVibGljIFNoYXBlIGdldFZpc3VhbEhpZ2hsaWdodFNoYXBlKFRleHRIaXRJbmZvIGhpdDEsIFRleHRIaXRJbmZvIGhpdDIsIFJlY3RhbmdsZTJEIGJvdW5kcykgewotICAgICAgICByZXR1cm4gY2FyZXRNYW5hZ2VyLmdldFZpc3VhbEhpZ2hsaWdodFNoYXBlKGhpdDEsIGhpdDIsIGJvdW5kcywgdGhpcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIFNoYXBlIHdoaWNoIGNvcnJlc3BvbmRzIHRvIHRoZSBoaWdobGlnaHRlZCAoc2VsZWN0ZWQpIGFyZWEgYmFzZWQKLSAgICAgKiBvbiB0d28gaGl0IGxvY2F0aW9ucyB3aXRoaW4gdGhlIHRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGhpdDEKLSAgICAgKiAgICAgICAgICAgIHRoZSBmaXJzdCB0ZXh0IGhpdCBsb2NhdGlvbi4KLSAgICAgKiBAcGFyYW0gaGl0MgotICAgICAqICAgICAgICAgICAgdGhlIHNlY29uZCB0ZXh0IGhpdCBsb2NhdGlvbi4KLSAgICAgKiBAcmV0dXJuIGEgU2hhcGUgd2hpY2ggY29ycmVzcG9uZHMgdG8gdGhlIGhpZ2hsaWdodGVkIChzZWxlY3RlZCkgYXJlYS4KLSAgICAgKi8KLSAgICBwdWJsaWMgU2hhcGUgZ2V0VmlzdWFsSGlnaGxpZ2h0U2hhcGUoVGV4dEhpdEluZm8gaGl0MSwgVGV4dEhpdEluZm8gaGl0MikgewotICAgICAgICBicmVha2VyLmNyZWF0ZUFsbFNlZ21lbnRzKCk7Ci0gICAgICAgIHJldHVybiBjYXJldE1hbmFnZXIuZ2V0VmlzdWFsSGlnaGxpZ2h0U2hhcGUoaGl0MSwgaGl0MiwgYnJlYWtlci5nZXRMb2dpY2FsQm91bmRzKCksIHRoaXMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFRleHRIaXRJbmZvIGZvciBhIGhpdCBvbiB0aGUgb3Bwb3NpdGUgc2lkZSBvZiB0aGUgc3BlY2lmaWVkCi0gICAgICogaGl0J3MgY2FyZXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGhpdEluZm8KLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgVGV4dEhpdEluZm8uCi0gICAgICogQHJldHVybiB0aGUgVGV4dEhpdEluZm8gZm9yIGEgaGl0IG9uIHRoZSBvcHBvc2l0ZSBzaWRlIG9mIHRoZSBzcGVjaWZpZWQKLSAgICAgKiAgICAgICAgIGhpdCdzIGNhcmV0LgotICAgICAqLwotICAgIHB1YmxpYyBUZXh0SGl0SW5mbyBnZXRWaXN1YWxPdGhlckhpdChUZXh0SGl0SW5mbyBoaXRJbmZvKSB7Ci0gICAgICAgIHJldHVybiBjYXJldE1hbmFnZXIuZ2V0VmlzdWFsT3RoZXJIaXQoaGl0SW5mbyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSnVzdGlmaWVzIHRoZSB0ZXh0OyB0aGlzIG1ldGhvZCBzaG91bGQgYmUgb3ZlcnJpZGRlbiBieSBzdWJjbGFzc2VzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBqdXN0aWZpY2F0aW9uV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBmb3IganVzdGlmaWNhdGlvbi4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBoYW5kbGVKdXN0aWZ5KGZsb2F0IGp1c3RpZmljYXRpb25XaWR0aCkgewotICAgICAgICBmbG9hdCBqdXN0aWZpY2F0aW9uID0gYnJlYWtlci5nZXRKdXN0aWZpY2F0aW9uKCk7Ci0KLSAgICAgICAgaWYgKGp1c3RpZmljYXRpb24gPCAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMTk2PUp1c3RpZmljYXRpb24gaW1wb3NzaWJsZSwgbGF5b3V0IGFscmVhZHkganVzdGlmaWVkCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjE5NiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9IGVsc2UgaWYgKGp1c3RpZmljYXRpb24gPT0gMCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgZmxvYXQgZ2FwID0gKGp1c3RpZmljYXRpb25XaWR0aCAtIGdldFZpc2libGVBZHZhbmNlKCkpICoganVzdGlmaWNhdGlvbjsKLSAgICAgICAgYnJlYWtlci5qdXN0aWZ5KGdhcCk7Ci0gICAgICAgIHRoaXMuanVzdGlmaWNhdGlvbldpZHRoID0ganVzdGlmaWNhdGlvbldpZHRoOwotCi0gICAgICAgIC8vIENvcnJlY3QgbWV0cmljcwotICAgICAgICB0bWMgPSBuZXcgVGV4dE1ldHJpY3NDYWxjdWxhdG9yKGJyZWFrZXIpOwotICAgICAgICB0bWMuY29ycmVjdEFkdmFuY2UobWV0cmljcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIFRleHRIaXRJbmZvIG9iamVjdCB0aGF0IGdpdmVzIGluZm9ybWF0aW9uIG9uIHdoaWNoIGRpdmlzaW9uCi0gICAgICogcG9pbnQgKGJldHdlZW4gdHdvIGNoYXJhY3RlcnMpIGlzIGNvcnJlc3BvbmRzIHRvIGEgaGl0IChzdWNoIGFzIGEgbW91c2UKLSAgICAgKiBjbGljaykgYXQgdGhlIHNwZWNpZmllZCBjb29yZGluYXRlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBpbiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgaW4gdGhpcyBUZXh0TGF5b3V0LiBUZXh0SGl0SW5mbyBvYmplY3QKLSAgICAgKiAgICAgICAgICAgIGNvcnJlc3BvbmRpbmcgdG8gdGhlIGdpdmVuIGNvb3JkaW5hdGVzIHdpdGhpbiB0aGUgdGV4dC4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbmZvcm1hdGlvbiBhYm91dCB0aGUgY2hhcmFjdGVyIGF0IHRoZSBzcGVjaWZpZWQgcG9zaXRpb24uCi0gICAgICovCi0gICAgcHVibGljIFRleHRIaXRJbmZvIGhpdFRlc3RDaGFyKGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgcmV0dXJuIGhpdFRlc3RDaGFyKHgsIHksIGdldEJvdW5kcygpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgVGV4dEhpdEluZm8gb2JqZWN0IHRoYXQgZ2l2ZXMgaW5mb3JtYXRpb24gb24gd2hpY2ggZGl2aXNpb24KLSAgICAgKiBwb2ludCAoYmV0d2VlbiB0d28gY2hhcmFjdGVycykgaXMgY29ycmVzcG9uZHMgdG8gYSBoaXQgKHN1Y2ggYXMgYSBtb3VzZQotICAgICAqIGNsaWNrKSBhdCB0aGUgc3BlY2lmaWVkIGNvb3JkaW5hdGVzIHdpdGhpbiB0aGUgc3BlY2lmaWVkIHRleHQgcmVjdGFuZ2xlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIGluIHRoaXMgVGV4dExheW91dC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBpbiB0aGlzIFRleHRMYXlvdXQuCi0gICAgICogQHBhcmFtIGJvdW5kcwotICAgICAqICAgICAgICAgICAgdGhlIGJvdW5kcyBvZiB0aGUgdGV4dCBhcmVhLiBUZXh0SGl0SW5mbyBvYmplY3QgY29ycmVzcG9uZGluZwotICAgICAqICAgICAgICAgICAgdG8gdGhlIGdpdmVuIGNvb3JkaW5hdGVzIHdpdGhpbiB0aGUgdGV4dC4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbmZvcm1hdGlvbiBhYm91dCB0aGUgY2hhcmFjdGVyIGF0IHRoZSBzcGVjaWZpZWQgcG9zaXRpb24uCi0gICAgICovCi0gICAgcHVibGljIFRleHRIaXRJbmZvIGhpdFRlc3RDaGFyKGZsb2F0IHgsIGZsb2F0IHksIFJlY3RhbmdsZTJEIGJvdW5kcykgewotICAgICAgICBpZiAoeCA+IGJvdW5kcy5nZXRNYXhYKCkpIHsKLSAgICAgICAgICAgIHJldHVybiBicmVha2VyLmlzTFRSKCkgPyBUZXh0SGl0SW5mby50cmFpbGluZyhicmVha2VyLmdldENoYXJDb3VudCgpIC0gMSkgOiBUZXh0SGl0SW5mbwotICAgICAgICAgICAgICAgICAgICAubGVhZGluZygwKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICh4IDwgYm91bmRzLmdldE1pblgoKSkgewotICAgICAgICAgICAgcmV0dXJuIGJyZWFrZXIuaXNMVFIoKSA/IFRleHRIaXRJbmZvLmxlYWRpbmcoMCkgOiBUZXh0SGl0SW5mby50cmFpbGluZyhicmVha2VyCi0gICAgICAgICAgICAgICAgICAgIC5nZXRDaGFyQ291bnQoKSAtIDEpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGJyZWFrZXIuaGl0VGVzdCh4LCB5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhpcyBUZXh0TGF5b3V0IGhhcyBhICJsZWZ0IHRvIHJpZ2h0IiBkaXJlY3Rpb24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlIGlmIHRoaXMgVGV4dExheW91dCBoYXMgYSAibGVmdCB0byByaWdodCIgZGlyZWN0aW9uLCBmYWxzZSBpZgotICAgICAqICAgICAgICAgdGhpcyBUZXh0TGF5b3V0IGhhcyBhICJyaWdodCB0byBsZWZ0IiBkaXJlY3Rpb24uCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNMZWZ0VG9SaWdodCgpIHsKLSAgICAgICAgcmV0dXJuIGJyZWFrZXIuaXNMVFIoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhpcyBUZXh0TGF5b3V0IGlzIHZlcnRpY2FsLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlIGlmIHRoaXMgVGV4dExheW91dCBpcyB2ZXJ0aWNhbCwgZmFsc2UgaWYgaG9yaXpvbnRhbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1ZlcnRpY2FsKCkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ZvbnQvVGV4dE1lYXN1cmVyLmphdmEgYi9hd3QvamF2YS9hd3QvZm9udC9UZXh0TWVhc3VyZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTc0MWY1OS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZm9udC9UZXh0TWVhc3VyZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE4MiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZm9udDsKLQotaW1wb3J0IGphdmEudGV4dC5BdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3I7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuVGV4dE1ldHJpY3NDYWxjdWxhdG9yOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5UZXh0UnVuQnJlYWtlcjsKLQotLyoqCi0gKiBUaGUgVGV4dE1lYXN1cmVyIGNsYXNzIHByb3ZpZGVzIHV0aWxpdGllcyBmb3IgbGluZSBicmVhayBvcGVyYXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIFRleHRNZWFzdXJlciBpbXBsZW1lbnRzIENsb25lYWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYWNpLgotICAgICAqLwotICAgIEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvciBhY2k7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZnJjLgotICAgICAqLwotICAgIEZvbnRSZW5kZXJDb250ZXh0IGZyYzsKLQotICAgIC8qKgotICAgICAqIFRoZSBicmVha2VyLgotICAgICAqLwotICAgIFRleHRSdW5CcmVha2VyIGJyZWFrZXIgPSBudWxsOwotCi0gICAgLyoqCi0gICAgICogVGhlIHRtYy4KLSAgICAgKi8KLSAgICBUZXh0TWV0cmljc0NhbGN1bGF0b3IgdG1jID0gbnVsbDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyB0ZXh0IG1lYXN1cmVyIGZyb20gdGhlIHNwZWNpZmllZCB0ZXh0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB0ZXh0Ci0gICAgICogICAgICAgICAgICB0aGUgc291cmNlIHRleHQuCi0gICAgICogQHBhcmFtIGZyYwotICAgICAqICAgICAgICAgICAgdGhlIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBUZXh0TWVhc3VyZXIoQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIHRleHQsIEZvbnRSZW5kZXJDb250ZXh0IGZyYykgewotICAgICAgICB0aGlzLmFjaSA9IHRleHQ7Ci0gICAgICAgIHRoaXMuZnJjID0gZnJjOwotICAgICAgICBicmVha2VyID0gbmV3IFRleHRSdW5CcmVha2VyKGFjaSwgdGhpcy5mcmMpOwotICAgICAgICB0bWMgPSBuZXcgVGV4dE1ldHJpY3NDYWxjdWxhdG9yKGJyZWFrZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcGxhY2VzIHRoZSBjdXJyZW50IHRleHQgd2l0aCB0aGUgbmV3IHRleHQsIGluc2VydGluZyBhIGJyZWFrIGNoYXJhY3RlcgotICAgICAqIGF0IHRoZSBzcGVjaWZpZWQgaW5zZXJ0IHBvc2l0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuZXdQYXJhZ3JhcGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgcGFyYWdyYXBoIHRleHQuCi0gICAgICogQHBhcmFtIGluc2VydFBvcwotICAgICAqICAgICAgICAgICAgdGhlIHBvc2l0aW9uIGluIHRoZSB0ZXh0IHdoZXJlIHRoZSBjaGFyYWN0ZXIgaXMgaW5zZXJ0ZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgaW5zZXJ0Q2hhcihBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgbmV3UGFyYWdyYXBoLCBpbnQgaW5zZXJ0UG9zKSB7Ci0gICAgICAgIEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvciBvbGRBY2kgPSBhY2k7Ci0gICAgICAgIGFjaSA9IG5ld1BhcmFncmFwaDsKLSAgICAgICAgaWYgKChvbGRBY2kuZ2V0RW5kSW5kZXgoKSAtIG9sZEFjaS5nZXRCZWdpbkluZGV4KCkpCi0gICAgICAgICAgICAgICAgLSAoYWNpLmdldEVuZEluZGV4KCkgLSBhY2kuZ2V0QmVnaW5JbmRleCgpKSAhPSAtMSkgewotICAgICAgICAgICAgYnJlYWtlciA9IG5ldyBUZXh0UnVuQnJlYWtlcihhY2ksIHRoaXMuZnJjKTsKLSAgICAgICAgICAgIHRtYyA9IG5ldyBUZXh0TWV0cmljc0NhbGN1bGF0b3IoYnJlYWtlcik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBicmVha2VyLmluc2VydENoYXIobmV3UGFyYWdyYXBoLCBpbnNlcnRQb3MpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwbGFjZXMgdGhlIGN1cnJlbnQgdGV4dCB3aXRoIHRoZSBuZXcgdGV4dCBhbmQgZGVsZXRlcyBhIGNoYXJhY3RlciBhdAotICAgICAqIHRoZSBzcGVjaWZpZWQgcG9zaXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIG5ld1BhcmFncmFwaAotICAgICAqICAgICAgICAgICAgdGhlIHBhcmFncmFwaCB0ZXh0IGFmdGVyIGRlbGV0aW9uLgotICAgICAqIEBwYXJhbSBkZWxldGVQb3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb3NpdGlvbiBpbiB0aGUgdGV4dCB3aGVyZSB0aGUgY2hhcmFjdGVyIGlzIHJlbW92ZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZGVsZXRlQ2hhcihBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgbmV3UGFyYWdyYXBoLCBpbnQgZGVsZXRlUG9zKSB7Ci0gICAgICAgIEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvciBvbGRBY2kgPSBhY2k7Ci0gICAgICAgIGFjaSA9IG5ld1BhcmFncmFwaDsKLSAgICAgICAgaWYgKChvbGRBY2kuZ2V0RW5kSW5kZXgoKSAtIG9sZEFjaS5nZXRCZWdpbkluZGV4KCkpCi0gICAgICAgICAgICAgICAgLSAoYWNpLmdldEVuZEluZGV4KCkgLSBhY2kuZ2V0QmVnaW5JbmRleCgpKSAhPSAxKSB7Ci0gICAgICAgICAgICBicmVha2VyID0gbmV3IFRleHRSdW5CcmVha2VyKGFjaSwgdGhpcy5mcmMpOwotICAgICAgICAgICAgdG1jID0gbmV3IFRleHRNZXRyaWNzQ2FsY3VsYXRvcihicmVha2VyKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGJyZWFrZXIuZGVsZXRlQ2hhcihuZXdQYXJhZ3JhcGgsIGRlbGV0ZVBvcyk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgY29weSBvZiB0aGlzIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgY29weSBvZiB0aGlzIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgT2JqZWN0IGNsb25lKCkgewotICAgICAgICByZXR1cm4gbmV3IFRleHRNZWFzdXJlcigoQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yKWFjaS5jbG9uZSgpLCBmcmMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBUZXh0TGF5b3V0IG9mIHRoZSBzcGVjaWZpZWQgY2hhcmFjdGVyIHJhbmdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdGFydAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBmaXJzdCBjaGFyYWN0ZXIuCi0gICAgICogQHBhcmFtIGxpbWl0Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggYWZ0ZXIgdGhlIGxhc3QgY2hhcmFjdGVyLgotICAgICAqIEByZXR1cm4gYSBUZXh0TGF5b3V0IGZvciB0aGUgY2hhcmFjdGVycyBiZWdpbm5pbmcgYXQgInN0YXJ0IiB1cCB0byAiZW5kIi4KLSAgICAgKi8KLSAgICBwdWJsaWMgVGV4dExheW91dCBnZXRMYXlvdXQoaW50IHN0YXJ0LCBpbnQgbGltaXQpIHsKLSAgICAgICAgYnJlYWtlci5wdXNoU2VnbWVudHMoc3RhcnQgLSBhY2kuZ2V0QmVnaW5JbmRleCgpLCBsaW1pdCAtIGFjaS5nZXRCZWdpbkluZGV4KCkpOwotCi0gICAgICAgIGJyZWFrZXIuY3JlYXRlQWxsU2VnbWVudHMoKTsKLSAgICAgICAgVGV4dExheW91dCBsYXlvdXQgPSBuZXcgVGV4dExheW91dCgoVGV4dFJ1bkJyZWFrZXIpYnJlYWtlci5jbG9uZSgpKTsKLQotICAgICAgICBicmVha2VyLnBvcFNlZ21lbnRzKCk7Ci0gICAgICAgIHJldHVybiBsYXlvdXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgZ3JhcGhpY2FsIHdpZHRoIG9mIGEgbGluZSBiZWdpbm5pbmcgYXQgInN0YXJ0IiBwYXJhbWV0ZXIgYW5kCi0gICAgICogaW5jbHVkaW5nIGNoYXJhY3RlcnMgdXAgdG8gImVuZCIgcGFyYW1ldGVyLiAic3RhcnQiIGFuZCAiZW5kIiBhcmUKLSAgICAgKiBhYnNvbHV0ZSBpbmRpY2VzLCBub3QgcmVsYXRpdmUgdG8gdGhlICJzdGFydCIgb2YgdGhlIHBhcmFncmFwaC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3RhcnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBjaGFyYWN0ZXIgaW5kZXggYXQgd2hpY2ggdG8gc3RhcnQgbWVhc3VyaW5nLgotICAgICAqIEBwYXJhbSBlbmQKLSAgICAgKiAgICAgICAgICAgIHRoZSBjaGFyYWN0ZXIgaW5kZXggYXQgd2hpY2ggdG8gc3RvcCBtZWFzdXJpbmcuCi0gICAgICogQHJldHVybiB0aGUgZ3JhcGhpY2FsIHdpZHRoIG9mIGEgbGluZSBiZWdpbm5pbmcgYXQgInN0YXJ0IiBhbmQgaW5jbHVkaW5nCi0gICAgICogICAgICAgICBjaGFyYWN0ZXJzIHVwIHRvICJlbmQiLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRBZHZhbmNlQmV0d2VlbihpbnQgc3RhcnQsIGludCBlbmQpIHsKLSAgICAgICAgYnJlYWtlci5wdXNoU2VnbWVudHMoc3RhcnQgLSBhY2kuZ2V0QmVnaW5JbmRleCgpLCBlbmQgLSBhY2kuZ2V0QmVnaW5JbmRleCgpKTsKLQotICAgICAgICBicmVha2VyLmNyZWF0ZUFsbFNlZ21lbnRzKCk7Ci0gICAgICAgIGZsb2F0IHJldHZhbCA9IHRtYy5jcmVhdGVNZXRyaWNzKCkuZ2V0QWR2YW5jZSgpOwotCi0gICAgICAgIGJyZWFrZXIucG9wU2VnbWVudHMoKTsKLSAgICAgICAgcmV0dXJuIHJldHZhbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBpbmRleCBvZiB0aGUgZmlyc3QgY2hhcmFjdGVyIHdoaWNoIGlzIG5vdCBmaXQgb24gYSBsaW5lCi0gICAgICogYmVnaW5uaW5nIGF0IHN0YXJ0IGFuZCBwb3NzaWJsZSBtZWFzdXJpbmcgdXAgdG8gbWF4QWR2YW5jZSBpbiBncmFwaGljYWwKLSAgICAgKiB3aWR0aC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3RhcnQKLSAgICAgKiAgICAgICAgICAgIGhlIGNoYXJhY3RlciBpbmRleCBhdCB3aGljaCB0byBzdGFydCBtZWFzdXJpbmcuCi0gICAgICogQHBhcmFtIG1heEFkdmFuY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBncmFwaGljYWwgd2lkdGggaW4gd2hpY2ggdGhlIGxpbmUgbXVzdCBmaXQuCi0gICAgICogQHJldHVybiB0aGUgaW5kZXggYWZ0ZXIgdGhlIGxhc3QgY2hhcmFjdGVyIHRoYXQgaXMgZml0IG9uIGEgbGluZQotICAgICAqICAgICAgICAgYmVnaW5uaW5nIGF0IHN0YXJ0LCB3aGljaCBpcyBub3QgbG9uZ2VyIHRoYW4gbWF4QWR2YW5jZSBpbgotICAgICAqICAgICAgICAgZ3JhcGhpY2FsIHdpZHRoLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TGluZUJyZWFrSW5kZXgoaW50IHN0YXJ0LCBmbG9hdCBtYXhBZHZhbmNlKSB7Ci0gICAgICAgIGJyZWFrZXIuY3JlYXRlQWxsU2VnbWVudHMoKTsKLSAgICAgICAgcmV0dXJuIGJyZWFrZXIuZ2V0TGluZUJyZWFrSW5kZXgoc3RhcnQgLSBhY2kuZ2V0QmVnaW5JbmRleCgpLCBtYXhBZHZhbmNlKQotICAgICAgICAgICAgICAgICsgYWNpLmdldEJlZ2luSW5kZXgoKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZm9udC9UcmFuc2Zvcm1BdHRyaWJ1dGUuamF2YSBiL2F3dC9qYXZhL2F3dC9mb250L1RyYW5zZm9ybUF0dHJpYnV0ZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmZjJjYWEyLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9mb250L1RyYW5zZm9ybUF0dHJpYnV0ZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsODYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJbHlhIFMuIE9rb21pbgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZm9udDsKLQotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuaW8uU2VyaWFsaXphYmxlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIFRyYW5zZm9ybUF0dHJpYnV0ZSBjbGFzcyBpcyBhIHdyYXBwZXIgZm9yIHRoZSBBZmZpbmVUcmFuc2Zvcm0gY2xhc3MgaW4KLSAqIG9yZGVyIHRvIHVzZSBpdCBhcyBhdHRyaWJ1dGUuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgZmluYWwgY2xhc3MgVHJhbnNmb3JtQXR0cmlidXRlIGltcGxlbWVudHMgU2VyaWFsaXphYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IDMzNTYyNDczNTc4Mjc3MDk1MzBMOwotCi0gICAgLy8gYWZmaW5lIHRyYW5zZm9ybSBvZiB0aGlzIFRyYW5zZm9ybUF0dHJpYnV0ZSBpbnN0YW5jZQotICAgIC8qKgotICAgICAqIFRoZSB0cmFuc2Zvcm0uCi0gICAgICovCi0gICAgcHJpdmF0ZSBBZmZpbmVUcmFuc2Zvcm0gZlRyYW5zZm9ybTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBUcmFuc2Zvcm1BdHRyaWJ1dGUgZnJvbSB0aGUgc3BlY2lmaWVkIEFmZmluZVRyYW5zZm9ybS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdHJhbnNmb3JtCi0gICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtIHRvIGJlIHdyYXBwZWQuCi0gICAgICovCi0gICAgcHVibGljIFRyYW5zZm9ybUF0dHJpYnV0ZShBZmZpbmVUcmFuc2Zvcm0gdHJhbnNmb3JtKSB7Ci0gICAgICAgIGlmICh0cmFuc2Zvcm0gPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0Ljk0PXRyYW5zZm9ybSBjYW4gbm90IGJlIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuOTQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZiAoIXRyYW5zZm9ybS5pc0lkZW50aXR5KCkpIHsKLSAgICAgICAgICAgIHRoaXMuZlRyYW5zZm9ybSA9IG5ldyBBZmZpbmVUcmFuc2Zvcm0odHJhbnNmb3JtKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGluaXRpYWwgQWZmaW5lVHJhbnNmb3JtIHdoaWNoIGlzIHdyYXBwZWQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaW5pdGlhbCBBZmZpbmVUcmFuc2Zvcm0gd2hpY2ggaXMgd3JhcHBlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQWZmaW5lVHJhbnNmb3JtIGdldFRyYW5zZm9ybSgpIHsKLSAgICAgICAgaWYgKGZUcmFuc2Zvcm0gIT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBBZmZpbmVUcmFuc2Zvcm0oZlRyYW5zZm9ybSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBBZmZpbmVUcmFuc2Zvcm0oKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyB0cmFuc2Zvcm0gaXMgYW4gaWRlbnRpdHkgdHJhbnNmb3JtLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyB0cmFuc2Zvcm0gaXMgYW4gaWRlbnRpdHkgdHJhbnNmb3JtLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzSWRlbnRpdHkoKSB7Ci0gICAgICAgIHJldHVybiAoZlRyYW5zZm9ybSA9PSBudWxsKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9mb250L3BhY2thZ2UuaHRtbCBiL2F3dC9qYXZhL2F3dC9mb250L3BhY2thZ2UuaHRtbApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzg4ZGNjMC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZm9udC9wYWNrYWdlLmh0bWwKKysrIC9kZXYvbnVsbApAQCAtMSw4ICswLDAgQEAKLTxodG1sPgotICA8Ym9keT4KLSAgICA8cD4KLSAgICAgIFRoaXMgcGFja2FnZSBjb250YWlucyBjbGFzc2VzIHRvIHN1cHBvcnQgdGhlIHJlcHJlc2VudGF0aW9uIG9mIGRpZmZlcmVudCB0eXBlcyBvZiBmb250cyBmb3IgZXhhbXBsZSBUcnVlVHlwZSBmb250cy4KLSAgICA8L3A+Ci0gICAgQHNpbmNlIEFuZHJvaWQgMS4wCi0gIDwvYm9keT4KLTwvaHRtbD4KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9nZW9tL0FmZmluZVRyYW5zZm9ybS5qYXZhIGIvYXd0L2phdmEvYXd0L2dlb20vQWZmaW5lVHJhbnNmb3JtLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDhhNjkzOGMuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2dlb20vQWZmaW5lVHJhbnNmb3JtLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMjY3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0Lmdlb207Ci0KLWltcG9ydCBqYXZhLmF3dC5TaGFwZTsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uU2VyaWFsaXphYmxlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lm1pc2MuSGFzaENvZGU7Ci0KLS8qKgotICogVGhlIENsYXNzIEFmZmluZVRyYW5zZm9ybSByZXByZXNlbnRzIGEgbGluZWFyIHRyYW5zZm9ybWF0aW9uIChyb3RhdGlvbiwKLSAqIHNjYWxpbmcsIG9yIHNoZWFyKSBmb2xsb3dlZCBieSBhIHRyYW5zbGF0aW9uIHRoYXQgYWN0cyBvbiBhIGNvb3JkaW5hdGUgc3BhY2UuCi0gKiBJdCBwcmVzZXJ2ZXMgY29sbGluZWFyaXR5IG9mIHBvaW50cyBhbmQgcmF0aW9zIG9mIGRpc3RhbmNlcyBiZXR3ZWVuIGNvbGxpbmVhcgotICogcG9pbnRzOiBzbyBpZiBBLCBCLCBhbmQgQyBhcmUgb24gYSBsaW5lLCB0aGVuIGFmdGVyIHRoZSBzcGFjZSBoYXMgYmVlbgotICogdHJhbnNmb3JtZWQgdmlhIHRoZSBhZmZpbmUgdHJhbnNmb3JtLCB0aGUgaW1hZ2VzIG9mIHRoZSB0aHJlZSBwb2ludHMgd2lsbAotICogc3RpbGwgYmUgb24gYSBsaW5lLCBhbmQgdGhlIHJhdGlvIG9mIHRoZSBkaXN0YW5jZSBmcm9tIEEgdG8gQiB3aXRoIHRoZQotICogZGlzdGFuY2UgZnJvbSBCIHRvIEMgd2lsbCBiZSB0aGUgc2FtZSBhcyB0aGUgY29ycmVzcG9uZGluZyByYXRpbyBpbiB0aGUgaW1hZ2UKLSAqIHNwYWNlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEFmZmluZVRyYW5zZm9ybSBpbXBsZW1lbnRzIENsb25lYWJsZSwgU2VyaWFsaXphYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IDEzMzA5NzMyMTA1MjM4NjA4MzRMOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfSURFTlRJVFkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9JREVOVElUWSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9UUkFOU0xBVElPTi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX1RSQU5TTEFUSU9OID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX1VOSUZPUk1fU0NBTEUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9VTklGT1JNX1NDQUxFID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0dFTkVSQUxfU0NBTEUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9HRU5FUkFMX1NDQUxFID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX1FVQURSQU5UX1JPVEFUSU9OLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfUVVBRFJBTlRfUk9UQVRJT04gPSA4OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfR0VORVJBTF9ST1RBVElPTi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0dFTkVSQUxfUk9UQVRJT04gPSAxNjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0dFTkVSQUxfVFJBTlNGT1JNLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfR0VORVJBTF9UUkFOU0ZPUk0gPSAzMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0ZMSVAuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9GTElQID0gNjQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9NQVNLX1NDQUxFLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfTUFTS19TQ0FMRSA9IFRZUEVfVU5JRk9STV9TQ0FMRSB8IFRZUEVfR0VORVJBTF9TQ0FMRTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX01BU0tfUk9UQVRJT04uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9NQVNLX1JPVEFUSU9OID0gVFlQRV9RVUFEUkFOVF9ST1RBVElPTiB8IFRZUEVfR0VORVJBTF9ST1RBVElPTjsKLQotICAgIC8qKgotICAgICAqIFRoZSA8Y29kZT5UWVBFX1VOS05PV048L2NvZGU+IGlzIGFuIGluaXRpYWwgdHlwZSB2YWx1ZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgZmluYWwgaW50IFRZUEVfVU5LTk9XTiA9IC0xOwotCi0gICAgLyoqCi0gICAgICogVGhlIG1pbiB2YWx1ZSBlcXVpdmFsZW50IHRvIHplcm8uIElmIGFic29sdXRlIHZhbHVlIGxlc3MgdGhlbiBaRVJPIGl0Ci0gICAgICogY29uc2lkZXJlZCBhcyB6ZXJvLgotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBkb3VibGUgWkVSTyA9IDFFLTEwOwotCi0gICAgLyoqCi0gICAgICogVGhlIHZhbHVlcyBvZiB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICovCi0gICAgZG91YmxlIG0wMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBtMTAuCi0gICAgICovCi0gICAgZG91YmxlIG0xMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBtMDEuCi0gICAgICovCi0gICAgZG91YmxlIG0wMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBtMTEuCi0gICAgICovCi0gICAgZG91YmxlIG0xMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBtMDIuCi0gICAgICovCi0gICAgZG91YmxlIG0wMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBtMTIuCi0gICAgICovCi0gICAgZG91YmxlIG0xMjsKLQotICAgIC8qKgotICAgICAqIFRoZSB0cmFuc2Zvcm1hdGlvbiA8Y29kZT50eXBlPC9jb2RlPi4KLSAgICAgKi8KLSAgICB0cmFuc2llbnQgaW50IHR5cGU7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgYWZmaW5lIHRyYW5zZm9ybSBvZiB0eXBlIDxjb2RlPlRZUEVfSURFTlRJVFk8L2NvZGU+Ci0gICAgICogKHdoaWNoIGxlYXZlcyBjb29yZGluYXRlcyB1bmNoYW5nZWQpLgotICAgICAqLwotICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0oKSB7Ci0gICAgICAgIHR5cGUgPSBUWVBFX0lERU5USVRZOwotICAgICAgICBtMDAgPSBtMTEgPSAxLjA7Ci0gICAgICAgIG0xMCA9IG0wMSA9IG0wMiA9IG0xMiA9IDAuMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgYWZmaW5lIHRyYW5zZm9ybSB0aGF0IGhhcyB0aGUgc2FtZSBkYXRhIGFzIHRoZSBnaXZlbgotICAgICAqIEFmZmluZVRyYW5zZm9ybS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdAotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zZm9ybSB0byBjb3B5LgotICAgICAqLwotICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0oQWZmaW5lVHJhbnNmb3JtIHQpIHsKLSAgICAgICAgdGhpcy50eXBlID0gdC50eXBlOwotICAgICAgICB0aGlzLm0wMCA9IHQubTAwOwotICAgICAgICB0aGlzLm0xMCA9IHQubTEwOwotICAgICAgICB0aGlzLm0wMSA9IHQubTAxOwotICAgICAgICB0aGlzLm0xMSA9IHQubTExOwotICAgICAgICB0aGlzLm0wMiA9IHQubTAyOwotICAgICAgICB0aGlzLm0xMiA9IHQubTEyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBhZmZpbmUgdHJhbnNmb3JtIGJ5IHNwZWNpZnlpbmcgdGhlIHZhbHVlcyBvZiB0aGUgMngzCi0gICAgICogdHJhbnNmb3JtYXRpb24gbWF0cml4IGFzIGZsb2F0cy4gVGhlIHR5cGUgaXMgc2V0IHRvIHRoZSBkZWZhdWx0IHR5cGU6Ci0gICAgICogPGNvZGU+VFlQRV9VTktOT1dOPC9jb2RlPgotICAgICAqIAotICAgICAqIEBwYXJhbSBtMDAKLSAgICAgKiAgICAgICAgICAgIHRoZSBtMDAgZW50cnkgaW4gdGhlIHRyYW5zZm9ybWF0aW9uIG1hdHJpeC4KLSAgICAgKiBAcGFyYW0gbTEwCi0gICAgICogICAgICAgICAgICB0aGUgbTEwIGVudHJ5IGluIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogQHBhcmFtIG0wMQotICAgICAqICAgICAgICAgICAgdGhlIG0wMSBlbnRyeSBpbiB0aGUgdHJhbnNmb3JtYXRpb24gbWF0cml4LgotICAgICAqIEBwYXJhbSBtMTEKLSAgICAgKiAgICAgICAgICAgIHRoZSBtMTEgZW50cnkgaW4gdGhlIHRyYW5zZm9ybWF0aW9uIG1hdHJpeC4KLSAgICAgKiBAcGFyYW0gbTAyCi0gICAgICogICAgICAgICAgICB0aGUgbTAyIGVudHJ5IGluIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogQHBhcmFtIG0xMgotICAgICAqICAgICAgICAgICAgdGhlIG0xMiBlbnRyeSBpbiB0aGUgdHJhbnNmb3JtYXRpb24gbWF0cml4LgotICAgICAqLwotICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0oZmxvYXQgbTAwLCBmbG9hdCBtMTAsIGZsb2F0IG0wMSwgZmxvYXQgbTExLCBmbG9hdCBtMDIsIGZsb2F0IG0xMikgewotICAgICAgICB0aGlzLnR5cGUgPSBUWVBFX1VOS05PV047Ci0gICAgICAgIHRoaXMubTAwID0gbTAwOwotICAgICAgICB0aGlzLm0xMCA9IG0xMDsKLSAgICAgICAgdGhpcy5tMDEgPSBtMDE7Ci0gICAgICAgIHRoaXMubTExID0gbTExOwotICAgICAgICB0aGlzLm0wMiA9IG0wMjsKLSAgICAgICAgdGhpcy5tMTIgPSBtMTI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGFmZmluZSB0cmFuc2Zvcm0gYnkgc3BlY2lmeWluZyB0aGUgdmFsdWVzIG9mIHRoZSAyeDMKLSAgICAgKiB0cmFuc2Zvcm1hdGlvbiBtYXRyaXggYXMgZG91Ymxlcy4gVGhlIHR5cGUgaXMgc2V0IHRvIHRoZSBkZWZhdWx0IHR5cGU6Ci0gICAgICogPGNvZGU+VFlQRV9VTktOT1dOPC9jb2RlPgotICAgICAqIAotICAgICAqIEBwYXJhbSBtMDAKLSAgICAgKiAgICAgICAgICAgIHRoZSBtMDAgZW50cnkgaW4gdGhlIHRyYW5zZm9ybWF0aW9uIG1hdHJpeC4KLSAgICAgKiBAcGFyYW0gbTEwCi0gICAgICogICAgICAgICAgICB0aGUgbTEwIGVudHJ5IGluIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogQHBhcmFtIG0wMQotICAgICAqICAgICAgICAgICAgdGhlIG0wMSBlbnRyeSBpbiB0aGUgdHJhbnNmb3JtYXRpb24gbWF0cml4LgotICAgICAqIEBwYXJhbSBtMTEKLSAgICAgKiAgICAgICAgICAgIHRoZSBtMTEgZW50cnkgaW4gdGhlIHRyYW5zZm9ybWF0aW9uIG1hdHJpeC4KLSAgICAgKiBAcGFyYW0gbTAyCi0gICAgICogICAgICAgICAgICB0aGUgbTAyIGVudHJ5IGluIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogQHBhcmFtIG0xMgotICAgICAqICAgICAgICAgICAgdGhlIG0xMiBlbnRyeSBpbiB0aGUgdHJhbnNmb3JtYXRpb24gbWF0cml4LgotICAgICAqLwotICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0oZG91YmxlIG0wMCwgZG91YmxlIG0xMCwgZG91YmxlIG0wMSwgZG91YmxlIG0xMSwgZG91YmxlIG0wMiwgZG91YmxlIG0xMikgewotICAgICAgICB0aGlzLnR5cGUgPSBUWVBFX1VOS05PV047Ci0gICAgICAgIHRoaXMubTAwID0gbTAwOwotICAgICAgICB0aGlzLm0xMCA9IG0xMDsKLSAgICAgICAgdGhpcy5tMDEgPSBtMDE7Ci0gICAgICAgIHRoaXMubTExID0gbTExOwotICAgICAgICB0aGlzLm0wMiA9IG0wMjsKLSAgICAgICAgdGhpcy5tMTIgPSBtMTI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGFmZmluZSB0cmFuc2Zvcm0gYnkgcmVhZGluZyB0aGUgdmFsdWVzIG9mIHRoZQotICAgICAqIHRyYW5zZm9ybWF0aW9uIG1hdHJpeCBmcm9tIGFuIGFycmF5IG9mIGZsb2F0cy4gVGhlIG1hcHBpbmcgZnJvbSB0aGUgYXJyYXkKLSAgICAgKiB0byB0aGUgbWF0cml4IHN0YXJ0cyB3aXRoIDxjb2RlPm1hdHJpeFswXTwvY29kZT4gZ2l2aW5nIHRoZSB0b3AtbGVmdAotICAgICAqIGVudHJ5IG9mIHRoZSBtYXRyaXggYW5kIHByb2NlZWRzIHdpdGggdGhlIHVzdWFsIGxlZnQtdG8tcmlnaHQgYW5kCi0gICAgICogdG9wLWRvd24gb3JkZXJpbmcuCi0gICAgICogPHA+Ci0gICAgICogSWYgdGhlIGFycmF5IGhhcyBvbmx5IGZvdXIgZW50cmllcywgdGhlbiB0aGUgdHdvIGVudHJpZXMgb2YgdGhlIGxhc3Qgcm93Ci0gICAgICogb2YgdGhlIHRyYW5zZm9ybWF0aW9uIG1hdHJpeCBkZWZhdWx0IHRvIHplcm8uCi0gICAgICogCi0gICAgICogQHBhcmFtIG1hdHJpeAotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGZvdXIgb3Igc2l4IGZsb2F0cyBnaXZpbmcgdGhlIHZhbHVlcyBvZiB0aGUKLSAgICAgKiAgICAgICAgICAgIG1hdHJpeC4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBzaXplIG9mIHRoZSBhcnJheSBpcyAwLCAxLCAyLCAzLCBvciA1LgotICAgICAqLwotICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0oZmxvYXRbXSBtYXRyaXgpIHsKLSAgICAgICAgdGhpcy50eXBlID0gVFlQRV9VTktOT1dOOwotICAgICAgICBtMDAgPSBtYXRyaXhbMF07Ci0gICAgICAgIG0xMCA9IG1hdHJpeFsxXTsKLSAgICAgICAgbTAxID0gbWF0cml4WzJdOwotICAgICAgICBtMTEgPSBtYXRyaXhbM107Ci0gICAgICAgIGlmIChtYXRyaXgubGVuZ3RoID4gNCkgewotICAgICAgICAgICAgbTAyID0gbWF0cml4WzRdOwotICAgICAgICAgICAgbTEyID0gbWF0cml4WzVdOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGFmZmluZSB0cmFuc2Zvcm0gYnkgcmVhZGluZyB0aGUgdmFsdWVzIG9mIHRoZQotICAgICAqIHRyYW5zZm9ybWF0aW9uIG1hdHJpeCBmcm9tIGFuIGFycmF5IG9mIGRvdWJsZXMuIFRoZSBtYXBwaW5nIGZyb20gdGhlCi0gICAgICogYXJyYXkgdG8gdGhlIG1hdHJpeCBzdGFydHMgd2l0aCA8Y29kZT5tYXRyaXhbMF08L2NvZGU+IGdpdmluZyB0aGUKLSAgICAgKiB0b3AtbGVmdCBlbnRyeSBvZiB0aGUgbWF0cml4IGFuZCBwcm9jZWVkcyB3aXRoIHRoZSB1c3VhbCBsZWZ0LXRvLXJpZ2h0Ci0gICAgICogYW5kIHRvcC1kb3duIG9yZGVyaW5nLgotICAgICAqIDxwPgotICAgICAqIElmIHRoZSBhcnJheSBoYXMgb25seSBmb3VyIGVudHJpZXMsIHRoZW4gdGhlIHR3byBlbnRyaWVzIG9mIHRoZSBsYXN0IHJvdwotICAgICAqIG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXggZGVmYXVsdCB0byB6ZXJvLgotICAgICAqIAotICAgICAqIEBwYXJhbSBtYXRyaXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBmb3VyIG9yIHNpeCBkb3VibGVzIGdpdmluZyB0aGUgdmFsdWVzIG9mIHRoZQotICAgICAqICAgICAgICAgICAgbWF0cml4LgotICAgICAqIEB0aHJvd3MgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHNpemUgb2YgdGhlIGFycmF5IGlzIDAsIDEsIDIsIDMsIG9yIDUuCi0gICAgICovCi0gICAgcHVibGljIEFmZmluZVRyYW5zZm9ybShkb3VibGVbXSBtYXRyaXgpIHsKLSAgICAgICAgdGhpcy50eXBlID0gVFlQRV9VTktOT1dOOwotICAgICAgICBtMDAgPSBtYXRyaXhbMF07Ci0gICAgICAgIG0xMCA9IG1hdHJpeFsxXTsKLSAgICAgICAgbTAxID0gbWF0cml4WzJdOwotICAgICAgICBtMTEgPSBtYXRyaXhbM107Ci0gICAgICAgIGlmIChtYXRyaXgubGVuZ3RoID4gNCkgewotICAgICAgICAgICAgbTAyID0gbWF0cml4WzRdOwotICAgICAgICAgICAgbTEyID0gbWF0cml4WzVdOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0eXBlIG9mIHRoZSBhZmZpbmUgdHJhbnNmb3JtYXRpb24uCi0gICAgICogPHA+Ci0gICAgICogVGhlIHR5cGUgaXMgY29tcHV0ZWQgYXMgZm9sbG93czogTGFiZWwgdGhlIGVudHJpZXMgb2YgdGhlIHRyYW5zZm9ybWF0aW9uCi0gICAgICogbWF0cml4IGFzIHRocmVlIHJvd3MgKG0wMCwgbTAxKSwgKG0xMCwgbTExKSwgYW5kIChtMDIsIG0xMikuIFRoZW4gaWYgdGhlCi0gICAgICogb3JpZ2luYWwgYmFzaXMgdmVjdG9ycyBhcmUgKDEsIDApIGFuZCAoMCwgMSksIHRoZSBuZXcgYmFzaXMgdmVjdG9ycyBhZnRlcgotICAgICAqIHRyYW5zZm9ybWF0aW9uIGFyZSBnaXZlbiBieSAobTAwLCBtMDEpIGFuZCAobTEwLCBtMTEpLCBhbmQgdGhlCi0gICAgICogdHJhbnNsYXRpb24gdmVjdG9yIGlzIChtMDIsIG0xMikuCi0gICAgICogPHA+Ci0gICAgICogVGhlIHR5cGVzIGFyZSBjbGFzc2lmaWVkIGFzIGZvbGxvd3M6IDxici8+IFRZUEVfSURFTlRJVFkgLSBubyBjaGFuZ2U8YnIvPgotICAgICAqIFRZUEVfVFJBTlNMQVRJT04gLSBUaGUgdHJhbnNsYXRpb24gdmVjdG9yIGlzbid0IHplcm88YnIvPgotICAgICAqIFRZUEVfVU5JRk9STV9TQ0FMRSAtIFRoZSBuZXcgYmFzaXMgdmVjdG9ycyBoYXZlIGVxdWFsIGxlbmd0aDxici8+Ci0gICAgICogVFlQRV9HRU5FUkFMX1NDQUxFIC0gVGhlIG5ldyBiYXNpcyB2ZWN0b3JzIGRvbnQnIGhhdmUgZXF1YWwgbGVuZ3RoPGJyLz4KLSAgICAgKiBUWVBFX0ZMSVAgLSBUaGUgbmV3IGJhc2lzIHZlY3RvciBvcmllbnRhdGlvbiBkaWZmZXJzIGZyb20gdGhlIG9yaWdpbmFsCi0gICAgICogb25lPGJyLz4gVFlQRV9RVUFEUkFOVF9ST1RBVElPTiAtIFRoZSBuZXcgYmFzaXMgaXMgYSByb3RhdGlvbiBvZiB0aGUKLSAgICAgKiBvcmlnaW5hbCBieSA5MCwgMTgwLCAyNzAsIG9yIDM2MCBkZWdyZWVzPGJyLz4gVFlQRV9HRU5FUkFMX1JPVEFUSU9OIC0gVGhlCi0gICAgICogbmV3IGJhc2lzIGlzIGEgcm90YXRpb24gb2YgdGhlIG9yaWdpbmFsIGJ5IGFuIGFyYml0cmFyeSBhbmdsZTxici8+Ci0gICAgICogVFlQRV9HRU5FUkFMX1RSQU5TRk9STSAtIFRoZSB0cmFuc2Zvcm1hdGlvbiBjYW4ndCBiZSBpbnZlcnRlZC48YnIvPgotICAgICAqIDxwPgotICAgICAqIE5vdGUgdGhhdCBtdWx0aXBsZSB0eXBlcyBhcmUgcG9zc2libGUsIHRodXMgdGhlIHR5cGVzIGNhbiBiZSBjb21iaW5lZAotICAgICAqIHVzaW5nIGJpdHdpc2UgY29tYmluYXRpb25zLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHR5cGUgb2YgdGhlIEFmZmluZSBUcmFuc2Zvcm0uCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRUeXBlKCkgewotICAgICAgICBpZiAodHlwZSAhPSBUWVBFX1VOS05PV04pIHsKLSAgICAgICAgICAgIHJldHVybiB0eXBlOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHR5cGUgPSAwOwotCi0gICAgICAgIGlmIChtMDAgKiBtMDEgKyBtMTAgKiBtMTEgIT0gMC4wKSB7Ci0gICAgICAgICAgICB0eXBlIHw9IFRZUEVfR0VORVJBTF9UUkFOU0ZPUk07Ci0gICAgICAgICAgICByZXR1cm4gdHlwZTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChtMDIgIT0gMC4wIHx8IG0xMiAhPSAwLjApIHsKLSAgICAgICAgICAgIHR5cGUgfD0gVFlQRV9UUkFOU0xBVElPTjsKLSAgICAgICAgfSBlbHNlIGlmIChtMDAgPT0gMS4wICYmIG0xMSA9PSAxLjAgJiYgbTAxID09IDAuMCAmJiBtMTAgPT0gMC4wKSB7Ci0gICAgICAgICAgICB0eXBlID0gVFlQRV9JREVOVElUWTsKLSAgICAgICAgICAgIHJldHVybiB0eXBlOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG0wMCAqIG0xMSAtIG0wMSAqIG0xMCA8IDAuMCkgewotICAgICAgICAgICAgdHlwZSB8PSBUWVBFX0ZMSVA7Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgZHggPSBtMDAgKiBtMDAgKyBtMTAgKiBtMTA7Ci0gICAgICAgIGRvdWJsZSBkeSA9IG0wMSAqIG0wMSArIG0xMSAqIG0xMTsKLSAgICAgICAgaWYgKGR4ICE9IGR5KSB7Ci0gICAgICAgICAgICB0eXBlIHw9IFRZUEVfR0VORVJBTF9TQ0FMRTsKLSAgICAgICAgfSBlbHNlIGlmIChkeCAhPSAxLjApIHsKLSAgICAgICAgICAgIHR5cGUgfD0gVFlQRV9VTklGT1JNX1NDQUxFOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChtMDAgPT0gMC4wICYmIG0xMSA9PSAwLjApIHx8IChtMTAgPT0gMC4wICYmIG0wMSA9PSAwLjAgJiYgKG0wMCA8IDAuMCB8fCBtMTEgPCAwLjApKSkgewotICAgICAgICAgICAgdHlwZSB8PSBUWVBFX1FVQURSQU5UX1JPVEFUSU9OOwotICAgICAgICB9IGVsc2UgaWYgKG0wMSAhPSAwLjAgfHwgbTEwICE9IDAuMCkgewotICAgICAgICAgICAgdHlwZSB8PSBUWVBFX0dFTkVSQUxfUk9UQVRJT047Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gdHlwZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzY2FsZSB4IGVudHJ5IG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXggKHRoZSB1cHBlciBsZWZ0Ci0gICAgICogbWF0cml4IGVudHJ5KS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzY2FsZSB4IHZhbHVlLgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgZ2V0U2NhbGVYKCkgewotICAgICAgICByZXR1cm4gbTAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNjYWxlIHkgZW50cnkgb2YgdGhlIHRyYW5zZm9ybWF0aW9uIG1hdHJpeCAodGhlIGxvd2VyIHJpZ2h0Ci0gICAgICogZW50cnkgb2YgdGhlIGxpbmVhciB0cmFuc2Zvcm1hdGlvbikuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc2NhbGUgeSB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlIGdldFNjYWxlWSgpIHsKLSAgICAgICAgcmV0dXJuIG0xMTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzaGVhciB4IGVudHJ5IG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXggKHRoZSB1cHBlciByaWdodAotICAgICAqIGVudHJ5IG9mIHRoZSBsaW5lYXIgdHJhbnNmb3JtYXRpb24pLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNoZWFyIHggdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRTaGVhclgoKSB7Ci0gICAgICAgIHJldHVybiBtMDE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2hlYXIgeSBlbnRyeSBvZiB0aGUgdHJhbnNmb3JtYXRpb24gbWF0cml4ICh0aGUgbG93ZXIgbGVmdCBlbnRyeQotICAgICAqIG9mIHRoZSBsaW5lYXIgdHJhbnNmb3JtYXRpb24pLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNoZWFyIHkgdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRTaGVhclkoKSB7Ci0gICAgICAgIHJldHVybiBtMTA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSB0cmFuc2xhdGlvbiB2ZWN0b3IuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSB0cmFuc2xhdGlvbiB2ZWN0b3IuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRUcmFuc2xhdGVYKCkgewotICAgICAgICByZXR1cm4gbTAyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgdHJhbnNsYXRpb24gdmVjdG9yLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgdHJhbnNsYXRpb24gdmVjdG9yLgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgZ2V0VHJhbnNsYXRlWSgpIHsKLSAgICAgICAgcmV0dXJuIG0xMjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIEFmZmluZVRyYW5zZm9ybWF0aW9uIGlzIHRoZSBpZGVudGl0eS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBBZmZpbmVUcmFuc2Zvcm1hdGlvbiBpcyB0aGUgaWRlbnRpdHkuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNJZGVudGl0eSgpIHsKLSAgICAgICAgcmV0dXJuIGdldFR5cGUoKSA9PSBUWVBFX0lERU5USVRZOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFdyaXRlcyB0aGUgdmFsdWVzIG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXggaW50byB0aGUgZ2l2ZW4gYXJyYXkgb2YKLSAgICAgKiBkb3VibGVzLiBJZiB0aGUgYXJyYXkgaGFzIGxlbmd0aCA0LCBvbmx5IHRoZSBsaW5lYXIgdHJhbnNmb3JtYXRpb24gcGFydAotICAgICAqIHdpbGwgYmUgd3JpdHRlbiBpbnRvIGl0LiBJZiBpdCBoYXMgbGVuZ3RoIGdyZWF0ZXIgdGhhbiA0LCB0aGUgdHJhbnNsYXRpb24KLSAgICAgKiB2ZWN0b3Igd2lsbCBiZSBpbmNsdWRlZCBhcyB3ZWxsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBtYXRyaXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB0byBmaWxsIHdpdGggdGhlIHZhbHVlcyBvZiB0aGUgbWF0cml4LgotICAgICAqIEB0aHJvd3MgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHNpemUgb2YgdGhlIGFycmF5IGlzIDAsIDEsIDIsIDMsIG9yIDUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZ2V0TWF0cml4KGRvdWJsZVtdIG1hdHJpeCkgewotICAgICAgICBtYXRyaXhbMF0gPSBtMDA7Ci0gICAgICAgIG1hdHJpeFsxXSA9IG0xMDsKLSAgICAgICAgbWF0cml4WzJdID0gbTAxOwotICAgICAgICBtYXRyaXhbM10gPSBtMTE7Ci0gICAgICAgIGlmIChtYXRyaXgubGVuZ3RoID4gNCkgewotICAgICAgICAgICAgbWF0cml4WzRdID0gbTAyOwotICAgICAgICAgICAgbWF0cml4WzVdID0gbTEyOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGV0ZXJtaW5hbnQgb2YgdGhlIGxpbmVhciB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGV0ZXJtaW5hbnQgb2YgdGhlIGxpbmVhciB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXREZXRlcm1pbmFudCgpIHsKLSAgICAgICAgcmV0dXJuIG0wMCAqIG0xMSAtIG0wMSAqIG0xMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSB0cmFuc2Zvcm0gaW4gdGVybXMgb2YgYSBsaXN0IG9mIGRvdWJsZSB2YWx1ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIG0wMAotICAgICAqICAgICAgICAgICAgdGhlIG0wMCBjb29yZGluYXRlIG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogQHBhcmFtIG0xMAotICAgICAqICAgICAgICAgICAgdGhlIG0xMCBjb29yZGluYXRlIG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogQHBhcmFtIG0wMQotICAgICAqICAgICAgICAgICAgdGhlIG0wMSBjb29yZGluYXRlIG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogQHBhcmFtIG0xMQotICAgICAqICAgICAgICAgICAgdGhlIG0xMSBjb29yZGluYXRlIG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogQHBhcmFtIG0wMgotICAgICAqICAgICAgICAgICAgdGhlIG0wMiBjb29yZGluYXRlIG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICogQHBhcmFtIG0xMgotICAgICAqICAgICAgICAgICAgdGhlIG0xMiBjb29yZGluYXRlIG9mIHRoZSB0cmFuc2Zvcm1hdGlvbiBtYXRyaXguCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0VHJhbnNmb3JtKGRvdWJsZSBtMDAsIGRvdWJsZSBtMTAsIGRvdWJsZSBtMDEsIGRvdWJsZSBtMTEsIGRvdWJsZSBtMDIsIGRvdWJsZSBtMTIpIHsKLSAgICAgICAgdGhpcy50eXBlID0gVFlQRV9VTktOT1dOOwotICAgICAgICB0aGlzLm0wMCA9IG0wMDsKLSAgICAgICAgdGhpcy5tMTAgPSBtMTA7Ci0gICAgICAgIHRoaXMubTAxID0gbTAxOwotICAgICAgICB0aGlzLm0xMSA9IG0xMTsKLSAgICAgICAgdGhpcy5tMDIgPSBtMDI7Ci0gICAgICAgIHRoaXMubTEyID0gbTEyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHRyYW5zZm9ybSdzIGRhdGEgdG8gbWF0Y2ggdGhlIGRhdGEgb2YgdGhlIHRyYW5zZm9ybSBzZW50IGFzIGEKLSAgICAgKiBwYXJhbWV0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHQKLSAgICAgKiAgICAgICAgICAgIHRoZSB0cmFuc2Zvcm0gdGhhdCBnaXZlcyB0aGUgbmV3IHZhbHVlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRUcmFuc2Zvcm0oQWZmaW5lVHJhbnNmb3JtIHQpIHsKLSAgICAgICAgdHlwZSA9IHQudHlwZTsKLSAgICAgICAgc2V0VHJhbnNmb3JtKHQubTAwLCB0Lm0xMCwgdC5tMDEsIHQubTExLCB0Lm0wMiwgdC5tMTIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHRyYW5zZm9ybSB0byB0aGUgaWRlbnRpdHkgdHJhbnNmb3JtLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFRvSWRlbnRpdHkoKSB7Ci0gICAgICAgIHR5cGUgPSBUWVBFX0lERU5USVRZOwotICAgICAgICBtMDAgPSBtMTEgPSAxLjA7Ci0gICAgICAgIG0xMCA9IG0wMSA9IG0wMiA9IG0xMiA9IDAuMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSB0cmFuc2Zvcm1hdGlvbiB0byBhIHRyYW5zbGF0aW9uIGFsb25lLiBTZXRzIHRoZSBsaW5lYXIgcGFydCBvZgotICAgICAqIHRoZSB0cmFuc2Zvcm1hdGlvbiB0byBpZGVudGl0eSBhbmQgdGhlIHRyYW5zbGF0aW9uIHZlY3RvciB0byB0aGUgdmFsdWVzCi0gICAgICogc2VudCBhcyBwYXJhbWV0ZXJzLiBTZXRzIHRoZSB0eXBlIHRvIDxjb2RlPlRZUEVfSURFTlRJVFk8L2NvZGU+IGlmIHRoZQotICAgICAqIHJlc3VsdGluZyBBZmZpbmVUcmFuc2Zvcm1hdGlvbiBpcyB0aGUgaWRlbnRpdHkgdHJhbnNmb3JtYXRpb24sIG90aGVyd2lzZQotICAgICAqIHNldHMgaXQgdG8gPGNvZGU+VFlQRV9UUkFOU0xBVElPTjwvY29kZT4uCi0gICAgICogCi0gICAgICogQHBhcmFtIG14Ci0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgdG8gdHJhbnNsYXRlIGluIHRoZSB4IGRpcmVjdGlvbi4KLSAgICAgKiBAcGFyYW0gbXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSB0byB0cmFuc2xhdGUgaW4gdGhlIHkgZGlyZWN0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFRvVHJhbnNsYXRpb24oZG91YmxlIG14LCBkb3VibGUgbXkpIHsKLSAgICAgICAgbTAwID0gbTExID0gMS4wOwotICAgICAgICBtMDEgPSBtMTAgPSAwLjA7Ci0gICAgICAgIG0wMiA9IG14OwotICAgICAgICBtMTIgPSBteTsKLSAgICAgICAgaWYgKG14ID09IDAuMCAmJiBteSA9PSAwLjApIHsKLSAgICAgICAgICAgIHR5cGUgPSBUWVBFX0lERU5USVRZOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdHlwZSA9IFRZUEVfVFJBTlNMQVRJT047Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSB0cmFuc2Zvcm1hdGlvbiB0byBiZWluZyBhIHNjYWxlIGFsb25lLCBlbGltaW5hdGluZyByb3RhdGlvbiwKLSAgICAgKiBzaGVhciwgYW5kIHRyYW5zbGF0aW9uIGVsZW1lbnRzLiBTZXRzIHRoZSB0eXBlIHRvCi0gICAgICogPGNvZGU+VFlQRV9JREVOVElUWTwvY29kZT4gaWYgdGhlIHJlc3VsdGluZyBBZmZpbmVUcmFuc2Zvcm1hdGlvbiBpcyB0aGUKLSAgICAgKiBpZGVudGl0eSB0cmFuc2Zvcm1hdGlvbiwgb3RoZXJ3aXNlIHNldHMgaXQgdG8gPGNvZGU+VFlQRV9VTktOT1dOPC9jb2RlPi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2N4Ci0gICAgICogICAgICAgICAgICB0aGUgc2NhbGluZyBmYWN0b3IgaW4gdGhlIHggZGlyZWN0aW9uLgotICAgICAqIEBwYXJhbSBzY3kKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FsaW5nIGZhY3RvciBpbiB0aGUgeSBkaXJlY3Rpb24uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0VG9TY2FsZShkb3VibGUgc2N4LCBkb3VibGUgc2N5KSB7Ci0gICAgICAgIG0wMCA9IHNjeDsKLSAgICAgICAgbTExID0gc2N5OwotICAgICAgICBtMTAgPSBtMDEgPSBtMDIgPSBtMTIgPSAwLjA7Ci0gICAgICAgIGlmIChzY3ggIT0gMS4wIHx8IHNjeSAhPSAxLjApIHsKLSAgICAgICAgICAgIHR5cGUgPSBUWVBFX1VOS05PV047Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICB0eXBlID0gVFlQRV9JREVOVElUWTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHRyYW5zZm9ybWF0aW9uIHRvIGJlaW5nIGEgc2hlYXIgYWxvbmUsIGVsaW1pbmF0aW5nIHJvdGF0aW9uLAotICAgICAqIHNjYWxpbmcsIGFuZCB0cmFuc2xhdGlvbiBlbGVtZW50cy4gU2V0cyB0aGUgdHlwZSB0bwotICAgICAqIDxjb2RlPlRZUEVfSURFTlRJVFk8L2NvZGU+IGlmIHRoZSByZXN1bHRpbmcgQWZmaW5lVHJhbnNmb3JtYXRpb24gaXMgdGhlCi0gICAgICogaWRlbnRpdHkgdHJhbnNmb3JtYXRpb24sIG90aGVyd2lzZSBzZXRzIGl0IHRvIDxjb2RlPlRZUEVfVU5LTk9XTjwvY29kZT4uCi0gICAgICogCi0gICAgICogQHBhcmFtIHNoeAotICAgICAqICAgICAgICAgICAgdGhlIHNoZWFyaW5nIGZhY3RvciBpbiB0aGUgeCBkaXJlY3Rpb24uCi0gICAgICogQHBhcmFtIHNoeQotICAgICAqICAgICAgICAgICAgdGhlIHNoZWFyaW5nIGZhY3RvciBpbiB0aGUgeSBkaXJlY3Rpb24uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0VG9TaGVhcihkb3VibGUgc2h4LCBkb3VibGUgc2h5KSB7Ci0gICAgICAgIG0wMCA9IG0xMSA9IDEuMDsKLSAgICAgICAgbTAyID0gbTEyID0gMC4wOwotICAgICAgICBtMDEgPSBzaHg7Ci0gICAgICAgIG0xMCA9IHNoeTsKLSAgICAgICAgaWYgKHNoeCAhPSAwLjAgfHwgc2h5ICE9IDAuMCkgewotICAgICAgICAgICAgdHlwZSA9IFRZUEVfVU5LTk9XTjsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHR5cGUgPSBUWVBFX0lERU5USVRZOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgdHJhbnNmb3JtYXRpb24gdG8gYmVpbmcgYSByb3RhdGlvbiBhbG9uZSwgZWxpbWluYXRpbmcgc2hlYXJpbmcsCi0gICAgICogc2NhbGluZywgYW5kIHRyYW5zbGF0aW9uIGVsZW1lbnRzLiBTZXRzIHRoZSB0eXBlIHRvCi0gICAgICogPGNvZGU+VFlQRV9JREVOVElUWTwvY29kZT4gaWYgdGhlIHJlc3VsdGluZyBBZmZpbmVUcmFuc2Zvcm1hdGlvbiBpcyB0aGUKLSAgICAgKiBpZGVudGl0eSB0cmFuc2Zvcm1hdGlvbiwgb3RoZXJ3aXNlIHNldHMgaXQgdG8gPGNvZGU+VFlQRV9VTktOT1dOPC9jb2RlPi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYW5nbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbmdsZSBvZiByb3RhdGlvbiBpbiByYWRpYW5zLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFRvUm90YXRpb24oZG91YmxlIGFuZ2xlKSB7Ci0gICAgICAgIGRvdWJsZSBzaW4gPSBNYXRoLnNpbihhbmdsZSk7Ci0gICAgICAgIGRvdWJsZSBjb3MgPSBNYXRoLmNvcyhhbmdsZSk7Ci0gICAgICAgIGlmIChNYXRoLmFicyhjb3MpIDwgWkVSTykgewotICAgICAgICAgICAgY29zID0gMC4wOwotICAgICAgICAgICAgc2luID0gc2luID4gMC4wID8gMS4wIDogLTEuMDsKLSAgICAgICAgfSBlbHNlIGlmIChNYXRoLmFicyhzaW4pIDwgWkVSTykgewotICAgICAgICAgICAgc2luID0gMC4wOwotICAgICAgICAgICAgY29zID0gY29zID4gMC4wID8gMS4wIDogLTEuMDsKLSAgICAgICAgfQotICAgICAgICBtMDAgPSBtMTEgPSBjb3M7Ci0gICAgICAgIG0wMSA9IC1zaW47Ci0gICAgICAgIG0xMCA9IHNpbjsKLSAgICAgICAgbTAyID0gbTEyID0gMC4wOwotICAgICAgICB0eXBlID0gVFlQRV9VTktOT1dOOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHRyYW5zZm9ybWF0aW9uIHRvIGJlaW5nIGEgcm90YXRpb24gZm9sbG93ZWQgYnkgYSB0cmFuc2xhdGlvbi4KLSAgICAgKiBTZXRzIHRoZSB0eXBlIHRvIDxjb2RlPlRZUEVfVU5LTk9XTjwvY29kZT4uCi0gICAgICogCi0gICAgICogQHBhcmFtIGFuZ2xlCi0gICAgICogICAgICAgICAgICB0aGUgYW5nbGUgb2Ygcm90YXRpb24gaW4gcmFkaWFucy4KLSAgICAgKiBAcGFyYW0gcHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSB0byB0cmFuc2xhdGUgaW4gdGhlIHggZGlyZWN0aW9uLgotICAgICAqIEBwYXJhbSBweQotICAgICAqICAgICAgICAgICAgdGhlIGRpc3RhbmNlIHRvIHRyYW5zbGF0ZSBpbiB0aGUgeSBkaXJlY3Rpb24uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0VG9Sb3RhdGlvbihkb3VibGUgYW5nbGUsIGRvdWJsZSBweCwgZG91YmxlIHB5KSB7Ci0gICAgICAgIHNldFRvUm90YXRpb24oYW5nbGUpOwotICAgICAgICBtMDIgPSBweCAqICgxLjAgLSBtMDApICsgcHkgKiBtMTA7Ci0gICAgICAgIG0xMiA9IHB5ICogKDEuMCAtIG0wMCkgLSBweCAqIG0xMDsKLSAgICAgICAgdHlwZSA9IFRZUEVfVU5LTk9XTjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgbmV3IEFmZmluZVRyYW5zZm9ybWF0aW9uIHRoYXQgaXMgYSB0cmFuc2xhdGlvbiBhbG9uZSB3aXRoIHRoZQotICAgICAqIHRyYW5zbGF0aW9uIHZlY3RvciBnaXZlbiBieSB0aGUgdmFsdWVzIHNlbnQgYXMgcGFyYW1ldGVycy4gVGhlIG5ldwotICAgICAqIHRyYW5zZm9ybWF0aW9uJ3MgdHlwZSBpcyA8Y29kZT5UWVBFX0lERU5USVRZPC9jb2RlPiBpZiB0aGUKLSAgICAgKiBBZmZpbmVUcmFuc2Zvcm1hdGlvbiBpcyB0aGUgaWRlbnRpdHkgdHJhbnNmb3JtYXRpb24sIG90aGVyd2lzZSBpdCdzCi0gICAgICogPGNvZGU+VFlQRV9UUkFOU0xBVElPTjwvY29kZT4uCi0gICAgICogCi0gICAgICogQHBhcmFtIG14Ci0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgdG8gdHJhbnNsYXRlIGluIHRoZSB4IGRpcmVjdGlvbi4KLSAgICAgKiBAcGFyYW0gbXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSB0byB0cmFuc2xhdGUgaW4gdGhlIHkgZGlyZWN0aW9uLgotICAgICAqIEByZXR1cm4gdGhlIG5ldyBBZmZpbmVUcmFuc2Zvcm1hdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEFmZmluZVRyYW5zZm9ybSBnZXRUcmFuc2xhdGVJbnN0YW5jZShkb3VibGUgbXgsIGRvdWJsZSBteSkgewotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gdCA9IG5ldyBBZmZpbmVUcmFuc2Zvcm0oKTsKLSAgICAgICAgdC5zZXRUb1RyYW5zbGF0aW9uKG14LCBteSk7Ci0gICAgICAgIHJldHVybiB0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBuZXcgQWZmaW5lVHJhbnNmb3JtYXRpb24gdGhhdCBpcyBhIHNjYWxlIGFsb25lLiBUaGUgbmV3Ci0gICAgICogdHJhbnNmb3JtYXRpb24ncyB0eXBlIGlzIDxjb2RlPlRZUEVfSURFTlRJVFk8L2NvZGU+IGlmIHRoZQotICAgICAqIEFmZmluZVRyYW5zZm9ybWF0aW9uIGlzIHRoZSBpZGVudGl0eSB0cmFuc2Zvcm1hdGlvbiwgb3RoZXJ3aXNlIGl0J3MKLSAgICAgKiA8Y29kZT5UWVBFX1VOS05PV048L2NvZGU+LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzY3gKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FsaW5nIGZhY3RvciBpbiB0aGUgeCBkaXJlY3Rpb24uCi0gICAgICogQHBhcmFtIHNjWQotICAgICAqICAgICAgICAgICAgdGhlIHNjYWxpbmcgZmFjdG9yIGluIHRoZSB5IGRpcmVjdGlvbi4KLSAgICAgKiBAcmV0dXJuIHRoZSBuZXcgQWZmaW5lVHJhbnNmb3JtYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBBZmZpbmVUcmFuc2Zvcm0gZ2V0U2NhbGVJbnN0YW5jZShkb3VibGUgc2N4LCBkb3VibGUgc2NZKSB7Ci0gICAgICAgIEFmZmluZVRyYW5zZm9ybSB0ID0gbmV3IEFmZmluZVRyYW5zZm9ybSgpOwotICAgICAgICB0LnNldFRvU2NhbGUoc2N4LCBzY1kpOwotICAgICAgICByZXR1cm4gdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgbmV3IEFmZmluZVRyYW5zZm9ybWF0aW9uIHRoYXQgaXMgYSBzaGVhciBhbG9uZS4gVGhlIG5ldwotICAgICAqIHRyYW5zZm9ybWF0aW9uJ3MgdHlwZSBpcyA8Y29kZT5UWVBFX0lERU5USVRZPC9jb2RlPiBpZiB0aGUKLSAgICAgKiBBZmZpbmVUcmFuc2Zvcm1hdGlvbiBpcyB0aGUgaWRlbnRpdHkgdHJhbnNmb3JtYXRpb24sIG90aGVyd2lzZSBpdCdzCi0gICAgICogPGNvZGU+VFlQRV9VTktOT1dOPC9jb2RlPi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2h4Ci0gICAgICogICAgICAgICAgICB0aGUgc2hlYXJpbmcgZmFjdG9yIGluIHRoZSB4IGRpcmVjdGlvbi4KLSAgICAgKiBAcGFyYW0gc2h5Ci0gICAgICogICAgICAgICAgICB0aGUgc2hlYXJpbmcgZmFjdG9yIGluIHRoZSB5IGRpcmVjdGlvbi4KLSAgICAgKiBAcmV0dXJuIHRoZSBuZXcgQWZmaW5lVHJhbnNmb3JtYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBBZmZpbmVUcmFuc2Zvcm0gZ2V0U2hlYXJJbnN0YW5jZShkb3VibGUgc2h4LCBkb3VibGUgc2h5KSB7Ci0gICAgICAgIEFmZmluZVRyYW5zZm9ybSBtID0gbmV3IEFmZmluZVRyYW5zZm9ybSgpOwotICAgICAgICBtLnNldFRvU2hlYXIoc2h4LCBzaHkpOwotICAgICAgICByZXR1cm4gbTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgbmV3IEFmZmluZVRyYW5zZm9ybWF0aW9uIHRoYXQgaXMgYSByb3RhdGlvbiBhbG9uZS4gVGhlIG5ldwotICAgICAqIHRyYW5zZm9ybWF0aW9uJ3MgdHlwZSBpcyA8Y29kZT5UWVBFX0lERU5USVRZPC9jb2RlPiBpZiB0aGUKLSAgICAgKiBBZmZpbmVUcmFuc2Zvcm1hdGlvbiBpcyB0aGUgaWRlbnRpdHkgdHJhbnNmb3JtYXRpb24sIG90aGVyd2lzZSBpdCdzCi0gICAgICogPGNvZGU+VFlQRV9VTktOT1dOPC9jb2RlPi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYW5nbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbmdsZSBvZiByb3RhdGlvbiBpbiByYWRpYW5zLgotICAgICAqIEByZXR1cm4gdGhlIG5ldyBBZmZpbmVUcmFuc2Zvcm1hdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEFmZmluZVRyYW5zZm9ybSBnZXRSb3RhdGVJbnN0YW5jZShkb3VibGUgYW5nbGUpIHsKLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHQgPSBuZXcgQWZmaW5lVHJhbnNmb3JtKCk7Ci0gICAgICAgIHQuc2V0VG9Sb3RhdGlvbihhbmdsZSk7Ci0gICAgICAgIHJldHVybiB0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBuZXcgQWZmaW5lVHJhbnNmb3JtYXRpb24gdGhhdCBpcyBhIHJvdGF0aW9uIGZvbGxvd2VkIGJ5IGEKLSAgICAgKiB0cmFuc2xhdGlvbi4gU2V0cyB0aGUgdHlwZSB0byA8Y29kZT5UWVBFX1VOS05PV048L2NvZGU+LgotICAgICAqIAotICAgICAqIEBwYXJhbSBhbmdsZQotICAgICAqICAgICAgICAgICAgdGhlIGFuZ2xlIG9mIHJvdGF0aW9uIGluIHJhZGlhbnMuCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSB0byB0cmFuc2xhdGUgaW4gdGhlIHggZGlyZWN0aW9uLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgdG8gdHJhbnNsYXRlIGluIHRoZSB5IGRpcmVjdGlvbi4KLSAgICAgKiBAcmV0dXJuIHRoZSBuZXcgQWZmaW5lVHJhbnNmb3JtYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBBZmZpbmVUcmFuc2Zvcm0gZ2V0Um90YXRlSW5zdGFuY2UoZG91YmxlIGFuZ2xlLCBkb3VibGUgeCwgZG91YmxlIHkpIHsKLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHQgPSBuZXcgQWZmaW5lVHJhbnNmb3JtKCk7Ci0gICAgICAgIHQuc2V0VG9Sb3RhdGlvbihhbmdsZSwgeCwgeSk7Ci0gICAgICAgIHJldHVybiB0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFwcGxpZXMgYSB0cmFuc2xhdGlvbiB0byB0aGlzIEFmZmluZVRyYW5zZm9ybWF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBteAotICAgICAqICAgICAgICAgICAgdGhlIGRpc3RhbmNlIHRvIHRyYW5zbGF0ZSBpbiB0aGUgeCBkaXJlY3Rpb24uCi0gICAgICogQHBhcmFtIG15Ci0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgdG8gdHJhbnNsYXRlIGluIHRoZSB5IGRpcmVjdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB0cmFuc2xhdGUoZG91YmxlIG14LCBkb3VibGUgbXkpIHsKLSAgICAgICAgY29uY2F0ZW5hdGUoQWZmaW5lVHJhbnNmb3JtLmdldFRyYW5zbGF0ZUluc3RhbmNlKG14LCBteSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFwcGxpZXMgYSBzY2FsaW5nIHRyYW5zZm9ybWF0aW9uIHRvIHRoaXMgQWZmaW5lVHJhbnNmb3JtYXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHNjeAotICAgICAqICAgICAgICAgICAgdGhlIHNjYWxpbmcgZmFjdG9yIGluIHRoZSB4IGRpcmVjdGlvbi4KLSAgICAgKiBAcGFyYW0gc2N5Ci0gICAgICogICAgICAgICAgICB0aGUgc2NhbGluZyBmYWN0b3IgaW4gdGhlIHkgZGlyZWN0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNjYWxlKGRvdWJsZSBzY3gsIGRvdWJsZSBzY3kpIHsKLSAgICAgICAgY29uY2F0ZW5hdGUoQWZmaW5lVHJhbnNmb3JtLmdldFNjYWxlSW5zdGFuY2Uoc2N4LCBzY3kpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBcHBsaWVzIGEgc2hlYXJpbmcgdHJhbnNmb3JtYXRpb24gdG8gdGhpcyBBZmZpbmVUcmFuc2Zvcm1hdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2h4Ci0gICAgICogICAgICAgICAgICB0aGUgc2hlYXJpbmcgZmFjdG9yIGluIHRoZSB4IGRpcmVjdGlvbi4KLSAgICAgKiBAcGFyYW0gc2h5Ci0gICAgICogICAgICAgICAgICB0aGUgc2hlYXJpbmcgZmFjdG9yIGluIHRoZSB5IGRpcmVjdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzaGVhcihkb3VibGUgc2h4LCBkb3VibGUgc2h5KSB7Ci0gICAgICAgIGNvbmNhdGVuYXRlKEFmZmluZVRyYW5zZm9ybS5nZXRTaGVhckluc3RhbmNlKHNoeCwgc2h5KSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXBwbGllcyBhIHJvdGF0aW9uIHRyYW5zZm9ybWF0aW9uIHRvIHRoaXMgQWZmaW5lVHJhbnNmb3JtYXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIGFuZ2xlCi0gICAgICogICAgICAgICAgICB0aGUgYW5nbGUgb2Ygcm90YXRpb24gaW4gcmFkaWFucy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByb3RhdGUoZG91YmxlIGFuZ2xlKSB7Ci0gICAgICAgIGNvbmNhdGVuYXRlKEFmZmluZVRyYW5zZm9ybS5nZXRSb3RhdGVJbnN0YW5jZShhbmdsZSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFwcGxpZXMgYSByb3RhdGlvbiBhbmQgdHJhbnNsYXRpb24gdHJhbnNmb3JtYXRpb24gdG8gdGhpcwotICAgICAqIEFmZmluZVRyYW5zZm9ybWF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhbmdsZQotICAgICAqICAgICAgICAgICAgdGhlIGFuZ2xlIG9mIHJvdGF0aW9uIGluIHJhZGlhbnMuCi0gICAgICogQHBhcmFtIHB4Ci0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgdG8gdHJhbnNsYXRlIGluIHRoZSB4IGRpcmVjdGlvbi4KLSAgICAgKiBAcGFyYW0gcHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSB0byB0cmFuc2xhdGUgaW4gdGhlIHkgZGlyZWN0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJvdGF0ZShkb3VibGUgYW5nbGUsIGRvdWJsZSBweCwgZG91YmxlIHB5KSB7Ci0gICAgICAgIGNvbmNhdGVuYXRlKEFmZmluZVRyYW5zZm9ybS5nZXRSb3RhdGVJbnN0YW5jZShhbmdsZSwgcHgsIHB5KSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogTXVsdGlwbGllcyB0aGUgbWF0cml4IHJlcHJlc2VudGF0aW9ucyBvZiB0d28gQWZmaW5lVHJhbnNmb3JtIG9iamVjdHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHQxCi0gICAgICogICAgICAgICAgICAtIHRoZSBBZmZpbmVUcmFuc2Zvcm0gb2JqZWN0IGlzIGEgbXVsdGlwbGljYW5kCi0gICAgICogQHBhcmFtIHQyCi0gICAgICogICAgICAgICAgICAtIHRoZSBBZmZpbmVUcmFuc2Zvcm0gb2JqZWN0IGlzIGEgbXVsdGlwbGllcgotICAgICAqIEByZXR1cm4gYW4gQWZmaW5lVHJhbnNmb3JtIG9iamVjdCB0aGF0IGlzIHRoZSByZXN1bHQgb2YgdDEgbXVsdGlwbGllZCBieQotICAgICAqICAgICAgICAgdGhlIG1hdHJpeCB0Mi4KLSAgICAgKi8KLSAgICBBZmZpbmVUcmFuc2Zvcm0gbXVsdGlwbHkoQWZmaW5lVHJhbnNmb3JtIHQxLCBBZmZpbmVUcmFuc2Zvcm0gdDIpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBBZmZpbmVUcmFuc2Zvcm0odDEubTAwICogdDIubTAwICsgdDEubTEwICogdDIubTAxLCAvLyBtMDAKLSAgICAgICAgICAgICAgICB0MS5tMDAgKiB0Mi5tMTAgKyB0MS5tMTAgKiB0Mi5tMTEsIC8vIG0wMQotICAgICAgICAgICAgICAgIHQxLm0wMSAqIHQyLm0wMCArIHQxLm0xMSAqIHQyLm0wMSwgLy8gbTEwCi0gICAgICAgICAgICAgICAgdDEubTAxICogdDIubTEwICsgdDEubTExICogdDIubTExLCAvLyBtMTEKLSAgICAgICAgICAgICAgICB0MS5tMDIgKiB0Mi5tMDAgKyB0MS5tMTIgKiB0Mi5tMDEgKyB0Mi5tMDIsIC8vIG0wMgotICAgICAgICAgICAgICAgIHQxLm0wMiAqIHQyLm0xMCArIHQxLm0xMiAqIHQyLm0xMSArIHQyLm0xMik7Ly8gbTEyCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXBwbGllcyB0aGUgZ2l2ZW4gQWZmaW5lVHJhbnNmb3JtIHRvIHRoaXMgQWZmaW5lVHJhbnNmb3JtIHZpYSBtYXRyaXgKLSAgICAgKiBtdWx0aXBsaWNhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdAotICAgICAqICAgICAgICAgICAgdGhlIEFmZmluZVRyYW5zZm9ybSB0byBhcHBseSB0byB0aGlzIEFmZmluZVRyYW5zZm9ybS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBjb25jYXRlbmF0ZShBZmZpbmVUcmFuc2Zvcm0gdCkgewotICAgICAgICBzZXRUcmFuc2Zvcm0obXVsdGlwbHkodCwgdGhpcykpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoYW5nZXMgdGhlIGN1cnJlbnQgQWZmaW5lVHJhbnNmb3JtIHRoZSBvbmUgb2J0YWluZWQgYnkgdGFraW5nIHRoZQotICAgICAqIHRyYW5zZm9ybSB0IGFuZCBhcHBseWluZyB0aGlzIEFmZmluZVRyYW5zZm9ybSB0byBpdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdAotICAgICAqICAgICAgICAgICAgdGhlIEFmZmluZVRyYW5zZm9ybSB0aGF0IHRoaXMgQWZmaW5lVHJhbnNmb3JtIGlzIG11bHRpcGxpZWQKLSAgICAgKiAgICAgICAgICAgIGJ5LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHByZUNvbmNhdGVuYXRlKEFmZmluZVRyYW5zZm9ybSB0KSB7Ci0gICAgICAgIHNldFRyYW5zZm9ybShtdWx0aXBseSh0aGlzLCB0KSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhbiBBZmZpbmVUcmFuc2Zvcm0gdGhhdCBpcyB0aGUgaW52ZXJzZSBvZiB0aGlzIHRyYW5zZm9ybS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhZmZpbmUgdHJhbnNmb3JtIHRoYXQgaXMgdGhlIGludmVyc2Ugb2YgdGhpcyBBZmZpbmVUcmFuc2Zvcm0uCi0gICAgICogQHRocm93cyBOb25pbnZlcnRpYmxlVHJhbnNmb3JtRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhpcyBBZmZpbmVUcmFuc2Zvcm0gY2Fubm90IGJlIGludmVydGVkICh0aGUgZGV0ZXJtaW5hbnQKLSAgICAgKiAgICAgICAgICAgICBvZiB0aGUgbGluZWFyIHRyYW5zZm9ybWF0aW9uIHBhcnQgaXMgemVybykuCi0gICAgICovCi0gICAgcHVibGljIEFmZmluZVRyYW5zZm9ybSBjcmVhdGVJbnZlcnNlKCkgdGhyb3dzIE5vbmludmVydGlibGVUcmFuc2Zvcm1FeGNlcHRpb24gewotICAgICAgICBkb3VibGUgZGV0ID0gZ2V0RGV0ZXJtaW5hbnQoKTsKLSAgICAgICAgaWYgKE1hdGguYWJzKGRldCkgPCBaRVJPKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjA0PURldGVybWluYW50IGlzIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBOb25pbnZlcnRpYmxlVHJhbnNmb3JtRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIwNCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBuZXcgQWZmaW5lVHJhbnNmb3JtKG0xMSAvIGRldCwgLy8gbTAwCi0gICAgICAgICAgICAgICAgLW0xMCAvIGRldCwgLy8gbTEwCi0gICAgICAgICAgICAgICAgLW0wMSAvIGRldCwgLy8gbTAxCi0gICAgICAgICAgICAgICAgbTAwIC8gZGV0LCAvLyBtMTEKLSAgICAgICAgICAgICAgICAobTAxICogbTEyIC0gbTExICogbTAyKSAvIGRldCwgLy8gbTAyCi0gICAgICAgICAgICAgICAgKG0xMCAqIG0wMiAtIG0wMCAqIG0xMikgLyBkZXQgLy8gbTEyCi0gICAgICAgICk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXBwbHkgdGhlIGN1cnJlbnQgQWZmaW5lVHJhbnNmb3JtIHRvIHRoZSBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgb3JpZ2luYWwgcG9pbnQuCi0gICAgICogQHBhcmFtIGRzdAotICAgICAqICAgICAgICAgICAgUG9pbnQyRCBvYmplY3QgdG8gYmUgZmlsbGVkIHdpdGggdGhlIGRlc3RpbmF0aW9uIGNvb3JkaW5hdGVzCi0gICAgICogICAgICAgICAgICAod2hlcmUgdGhlIG9yaWdpbmFsIHBvaW50IGlzIHNlbnQgYnkgdGhpcyBBZmZpbmVUcmFuc2Zvcm0pLgotICAgICAqICAgICAgICAgICAgTWF5IGJlIG51bGwuCi0gICAgICogQHJldHVybiB0aGUgcG9pbnQgaW4gdGhlIEFmZmluZVRyYW5zZm9ybSdzIGltYWdlIHNwYWNlIHdoZXJlIHRoZSBvcmlnaW5hbAotICAgICAqICAgICAgICAgcG9pbnQgaXMgc2VudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUG9pbnQyRCB0cmFuc2Zvcm0oUG9pbnQyRCBzcmMsIFBvaW50MkQgZHN0KSB7Ci0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgaWYgKHNyYyBpbnN0YW5jZW9mIFBvaW50MkQuRG91YmxlKSB7Ci0gICAgICAgICAgICAgICAgZHN0ID0gbmV3IFBvaW50MkQuRG91YmxlKCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGRzdCA9IG5ldyBQb2ludDJELkZsb2F0KCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgeCA9IHNyYy5nZXRYKCk7Ci0gICAgICAgIGRvdWJsZSB5ID0gc3JjLmdldFkoKTsKLQotICAgICAgICBkc3Quc2V0TG9jYXRpb24oeCAqIG0wMCArIHkgKiBtMDEgKyBtMDIsIHggKiBtMTAgKyB5ICogbTExICsgbTEyKTsKLSAgICAgICAgcmV0dXJuIGRzdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBcHBsaWVzIHRoaXMgQWZmaW5lVHJhbnNmb3JtIHRvIGFuIGFycmF5IG9mIHBvaW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgcG9pbnRzIHRvIGJlIHRyYW5zZm9ybWVkLgotICAgICAqIEBwYXJhbSBzcmNPZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIHNvdXJjZSBwb2ludCBhcnJheSBvZiB0aGUgZmlyc3QgcG9pbnQgdG8gYmUKLSAgICAgKiAgICAgICAgICAgIHRyYW5zZm9ybWVkLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb2ludCBhcnJheSB3aGVyZSB0aGUgaW1hZ2VzIG9mIHRoZSBwb2ludHMgKGFmdGVyIGFwcGx5aW5nCi0gICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtYXRpb24pIHNob3VsZCBiZSBwbGFjZWQuCi0gICAgICogQHBhcmFtIGRzdE9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgZGVzdGluYXRpb24gYXJyYXkgd2hlcmUgdGhlIG5ldyB2YWx1ZXMKLSAgICAgKiAgICAgICAgICAgIHNob3VsZCBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSBsZW5ndGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgcG9pbnRzIHRvIHRyYW5zZm9ybS4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIDxjb2RlPnNyY09mZiArIGxlbmd0aCA+IHNyYy5sZW5ndGg8L2NvZGU+IG9yCi0gICAgICogICAgICAgICAgICAgPGNvZGU+ZHN0T2ZmICsgbGVuZ3RoID4gZHN0Lmxlbmd0aDwvY29kZT4uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgdHJhbnNmb3JtKFBvaW50MkRbXSBzcmMsIGludCBzcmNPZmYsIFBvaW50MkRbXSBkc3QsIGludCBkc3RPZmYsIGludCBsZW5ndGgpIHsKLSAgICAgICAgd2hpbGUgKC0tbGVuZ3RoID49IDApIHsKLSAgICAgICAgICAgIFBvaW50MkQgc3JjUG9pbnQgPSBzcmNbc3JjT2ZmKytdOwotICAgICAgICAgICAgZG91YmxlIHggPSBzcmNQb2ludC5nZXRYKCk7Ci0gICAgICAgICAgICBkb3VibGUgeSA9IHNyY1BvaW50LmdldFkoKTsKLSAgICAgICAgICAgIFBvaW50MkQgZHN0UG9pbnQgPSBkc3RbZHN0T2ZmXTsKLSAgICAgICAgICAgIGlmIChkc3RQb2ludCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHNyY1BvaW50IGluc3RhbmNlb2YgUG9pbnQyRC5Eb3VibGUpIHsKLSAgICAgICAgICAgICAgICAgICAgZHN0UG9pbnQgPSBuZXcgUG9pbnQyRC5Eb3VibGUoKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBkc3RQb2ludCA9IG5ldyBQb2ludDJELkZsb2F0KCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgZHN0UG9pbnQuc2V0TG9jYXRpb24oeCAqIG0wMCArIHkgKiBtMDEgKyBtMDIsIHggKiBtMTAgKyB5ICogbTExICsgbTEyKTsKLSAgICAgICAgICAgIGRzdFtkc3RPZmYrK10gPSBkc3RQb2ludDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFwcGxpZXMgdGhpcyBBZmZpbmVUcmFuc2Zvcm0gdG8gYSBzZXQgb2YgcG9pbnRzIGdpdmVuIGFzIGFuIGFycmF5IG9mCi0gICAgICogZG91YmxlIHZhbHVlcyB3aGVyZSBldmVyeSB0d28gdmFsdWVzIGluIHRoZSBhcnJheSBnaXZlIHRoZSBjb29yZGluYXRlcyBvZgotICAgICAqIGEgcG9pbnQ7IHRoZSBldmVuLWluZGV4ZWQgdmFsdWVzIGdpdmluZyB0aGUgeCBjb29yZGluYXRlcyBhbmQgdGhlCi0gICAgICogb2RkLWluZGV4ZWQgdmFsdWVzIGdpdmluZyB0aGUgeSBjb29yZGluYXRlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgcG9pbnRzIHRvIGJlIHRyYW5zZm9ybWVkLgotICAgICAqIEBwYXJhbSBzcmNPZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIHNvdXJjZSBwb2ludCBhcnJheSBvZiB0aGUgZmlyc3QgcG9pbnQgdG8gYmUKLSAgICAgKiAgICAgICAgICAgIHRyYW5zZm9ybWVkLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb2ludCBhcnJheSB3aGVyZSB0aGUgaW1hZ2VzIG9mIHRoZSBwb2ludHMgKGFmdGVyIGFwcGx5aW5nCi0gICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtYXRpb24pIHNob3VsZCBiZSBwbGFjZWQuCi0gICAgICogQHBhcmFtIGRzdE9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgZGVzdGluYXRpb24gYXJyYXkgd2hlcmUgdGhlIG5ldyB2YWx1ZXMKLSAgICAgKiAgICAgICAgICAgIHNob3VsZCBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSBsZW5ndGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgcG9pbnRzIHRvIHRyYW5zZm9ybS4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIDxjb2RlPnNyY09mZiArIGxlbmd0aCoyID4gc3JjLmxlbmd0aDwvY29kZT4gb3IKLSAgICAgKiAgICAgICAgICAgICA8Y29kZT5kc3RPZmYgKyBsZW5ndGgqMiA+IGRzdC5sZW5ndGg8L2NvZGU+LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHRyYW5zZm9ybShkb3VibGVbXSBzcmMsIGludCBzcmNPZmYsIGRvdWJsZVtdIGRzdCwgaW50IGRzdE9mZiwgaW50IGxlbmd0aCkgewotICAgICAgICBpbnQgc3RlcCA9IDI7Ci0gICAgICAgIGlmIChzcmMgPT0gZHN0ICYmIHNyY09mZiA8IGRzdE9mZiAmJiBkc3RPZmYgPCBzcmNPZmYgKyBsZW5ndGggKiAyKSB7Ci0gICAgICAgICAgICBzcmNPZmYgPSBzcmNPZmYgKyBsZW5ndGggKiAyIC0gMjsKLSAgICAgICAgICAgIGRzdE9mZiA9IGRzdE9mZiArIGxlbmd0aCAqIDIgLSAyOwotICAgICAgICAgICAgc3RlcCA9IC0yOwotICAgICAgICB9Ci0gICAgICAgIHdoaWxlICgtLWxlbmd0aCA+PSAwKSB7Ci0gICAgICAgICAgICBkb3VibGUgeCA9IHNyY1tzcmNPZmYgKyAwXTsKLSAgICAgICAgICAgIGRvdWJsZSB5ID0gc3JjW3NyY09mZiArIDFdOwotICAgICAgICAgICAgZHN0W2RzdE9mZiArIDBdID0geCAqIG0wMCArIHkgKiBtMDEgKyBtMDI7Ci0gICAgICAgICAgICBkc3RbZHN0T2ZmICsgMV0gPSB4ICogbTEwICsgeSAqIG0xMSArIG0xMjsKLSAgICAgICAgICAgIHNyY09mZiArPSBzdGVwOwotICAgICAgICAgICAgZHN0T2ZmICs9IHN0ZXA7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBcHBsaWVzIHRoaXMgQWZmaW5lVHJhbnNmb3JtIHRvIGEgc2V0IG9mIHBvaW50cyBnaXZlbiBhcyBhbiBhcnJheSBvZgotICAgICAqIGZsb2F0IHZhbHVlcyB3aGVyZSBldmVyeSB0d28gdmFsdWVzIGluIHRoZSBhcnJheSBnaXZlIHRoZSBjb29yZGluYXRlcyBvZgotICAgICAqIGEgcG9pbnQ7IHRoZSBldmVuLWluZGV4ZWQgdmFsdWVzIGdpdmluZyB0aGUgeCBjb29yZGluYXRlcyBhbmQgdGhlCi0gICAgICogb2RkLWluZGV4ZWQgdmFsdWVzIGdpdmluZyB0aGUgeSBjb29yZGluYXRlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgcG9pbnRzIHRvIGJlIHRyYW5zZm9ybWVkLgotICAgICAqIEBwYXJhbSBzcmNPZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIHNvdXJjZSBwb2ludCBhcnJheSBvZiB0aGUgZmlyc3QgcG9pbnQgdG8gYmUKLSAgICAgKiAgICAgICAgICAgIHRyYW5zZm9ybWVkLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb2ludCBhcnJheSB3aGVyZSB0aGUgaW1hZ2VzIG9mIHRoZSBwb2ludHMgKGFmdGVyIGFwcGx5aW5nCi0gICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtYXRpb24pIHNob3VsZCBiZSBwbGFjZWQuCi0gICAgICogQHBhcmFtIGRzdE9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgZGVzdGluYXRpb24gYXJyYXkgd2hlcmUgdGhlIG5ldyB2YWx1ZXMKLSAgICAgKiAgICAgICAgICAgIHNob3VsZCBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSBsZW5ndGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgcG9pbnRzIHRvIHRyYW5zZm9ybS4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIDxjb2RlPnNyY09mZiArIGxlbmd0aCoyID4gc3JjLmxlbmd0aDwvY29kZT4gb3IKLSAgICAgKiAgICAgICAgICAgICA8Y29kZT5kc3RPZmYgKyBsZW5ndGgqMiA+IGRzdC5sZW5ndGg8L2NvZGU+LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHRyYW5zZm9ybShmbG9hdFtdIHNyYywgaW50IHNyY09mZiwgZmxvYXRbXSBkc3QsIGludCBkc3RPZmYsIGludCBsZW5ndGgpIHsKLSAgICAgICAgaW50IHN0ZXAgPSAyOwotICAgICAgICBpZiAoc3JjID09IGRzdCAmJiBzcmNPZmYgPCBkc3RPZmYgJiYgZHN0T2ZmIDwgc3JjT2ZmICsgbGVuZ3RoICogMikgewotICAgICAgICAgICAgc3JjT2ZmID0gc3JjT2ZmICsgbGVuZ3RoICogMiAtIDI7Ci0gICAgICAgICAgICBkc3RPZmYgPSBkc3RPZmYgKyBsZW5ndGggKiAyIC0gMjsKLSAgICAgICAgICAgIHN0ZXAgPSAtMjsKLSAgICAgICAgfQotICAgICAgICB3aGlsZSAoLS1sZW5ndGggPj0gMCkgewotICAgICAgICAgICAgZmxvYXQgeCA9IHNyY1tzcmNPZmYgKyAwXTsKLSAgICAgICAgICAgIGZsb2F0IHkgPSBzcmNbc3JjT2ZmICsgMV07Ci0gICAgICAgICAgICBkc3RbZHN0T2ZmICsgMF0gPSAoZmxvYXQpKHggKiBtMDAgKyB5ICogbTAxICsgbTAyKTsKLSAgICAgICAgICAgIGRzdFtkc3RPZmYgKyAxXSA9IChmbG9hdCkoeCAqIG0xMCArIHkgKiBtMTEgKyBtMTIpOwotICAgICAgICAgICAgc3JjT2ZmICs9IHN0ZXA7Ci0gICAgICAgICAgICBkc3RPZmYgKz0gc3RlcDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFwcGxpZXMgdGhpcyBBZmZpbmVUcmFuc2Zvcm0gdG8gYSBzZXQgb2YgcG9pbnRzIGdpdmVuIGFzIGFuIGFycmF5IG9mCi0gICAgICogZmxvYXQgdmFsdWVzIHdoZXJlIGV2ZXJ5IHR3byB2YWx1ZXMgaW4gdGhlIGFycmF5IGdpdmUgdGhlIGNvb3JkaW5hdGVzIG9mCi0gICAgICogYSBwb2ludDsgdGhlIGV2ZW4taW5kZXhlZCB2YWx1ZXMgZ2l2aW5nIHRoZSB4IGNvb3JkaW5hdGVzIGFuZCB0aGUKLSAgICAgKiBvZGQtaW5kZXhlZCB2YWx1ZXMgZ2l2aW5nIHRoZSB5IGNvb3JkaW5hdGVzLiBUaGUgZGVzdGluYXRpb24gY29vcmRpbmF0ZXMKLSAgICAgKiBhcmUgZ2l2ZW4gYXMgdmFsdWVzIG9mIHR5cGUgPGNvZGU+ZG91YmxlPC9jb2RlPi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgcG9pbnRzIHRvIGJlIHRyYW5zZm9ybWVkLgotICAgICAqIEBwYXJhbSBzcmNPZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIHNvdXJjZSBwb2ludCBhcnJheSBvZiB0aGUgZmlyc3QgcG9pbnQgdG8gYmUKLSAgICAgKiAgICAgICAgICAgIHRyYW5zZm9ybWVkLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb2ludCBhcnJheSB3aGVyZSB0aGUgaW1hZ2VzIG9mIHRoZSBwb2ludHMgKGFmdGVyIGFwcGx5aW5nCi0gICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtYXRpb24pIHNob3VsZCBiZSBwbGFjZWQuCi0gICAgICogQHBhcmFtIGRzdE9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgZGVzdGluYXRpb24gYXJyYXkgd2hlcmUgdGhlIG5ldyB2YWx1ZXMKLSAgICAgKiAgICAgICAgICAgIHNob3VsZCBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSBsZW5ndGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgcG9pbnRzIHRvIHRyYW5zZm9ybS4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIDxjb2RlPnNyY09mZiArIGxlbmd0aCoyID4gc3JjLmxlbmd0aDwvY29kZT4gb3IKLSAgICAgKiAgICAgICAgICAgICA8Y29kZT5kc3RPZmYgKyBsZW5ndGgqMiA+IGRzdC5sZW5ndGg8L2NvZGU+LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHRyYW5zZm9ybShmbG9hdFtdIHNyYywgaW50IHNyY09mZiwgZG91YmxlW10gZHN0LCBpbnQgZHN0T2ZmLCBpbnQgbGVuZ3RoKSB7Ci0gICAgICAgIHdoaWxlICgtLWxlbmd0aCA+PSAwKSB7Ci0gICAgICAgICAgICBmbG9hdCB4ID0gc3JjW3NyY09mZisrXTsKLSAgICAgICAgICAgIGZsb2F0IHkgPSBzcmNbc3JjT2ZmKytdOwotICAgICAgICAgICAgZHN0W2RzdE9mZisrXSA9IHggKiBtMDAgKyB5ICogbTAxICsgbTAyOwotICAgICAgICAgICAgZHN0W2RzdE9mZisrXSA9IHggKiBtMTAgKyB5ICogbTExICsgbTEyOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXBwbGllcyB0aGlzIEFmZmluZVRyYW5zZm9ybSB0byBhIHNldCBvZiBwb2ludHMgZ2l2ZW4gYXMgYW4gYXJyYXkgb2YKLSAgICAgKiBkb3VibGUgdmFsdWVzIHdoZXJlIGV2ZXJ5IHR3byB2YWx1ZXMgaW4gdGhlIGFycmF5IGdpdmUgdGhlIGNvb3JkaW5hdGVzIG9mCi0gICAgICogYSBwb2ludDsgdGhlIGV2ZW4taW5kZXhlZCB2YWx1ZXMgZ2l2aW5nIHRoZSB4IGNvb3JkaW5hdGVzIGFuZCB0aGUKLSAgICAgKiBvZGQtaW5kZXhlZCB2YWx1ZXMgZ2l2aW5nIHRoZSB5IGNvb3JkaW5hdGVzLiBUaGUgZGVzdGluYXRpb24gY29vcmRpbmF0ZXMKLSAgICAgKiBhcmUgZ2l2ZW4gYXMgdmFsdWVzIG9mIHR5cGUgPGNvZGU+ZmxvYXQ8L2NvZGU+LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBwb2ludHMgdG8gYmUgdHJhbnNmb3JtZWQuCi0gICAgICogQHBhcmFtIHNyY09mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgc291cmNlIHBvaW50IGFycmF5IG9mIHRoZSBmaXJzdCBwb2ludCB0byBiZQotICAgICAqICAgICAgICAgICAgdHJhbnNmb3JtZWQuCi0gICAgICogQHBhcmFtIGRzdAotICAgICAqICAgICAgICAgICAgdGhlIHBvaW50IGFycmF5IHdoZXJlIHRoZSBpbWFnZXMgb2YgdGhlIHBvaW50cyAoYWZ0ZXIgYXBwbHlpbmcKLSAgICAgKiAgICAgICAgICAgIHRoZSBBZmZpbmVUcmFuc2Zvcm1hdGlvbikgc2hvdWxkIGJlIHBsYWNlZC4KLSAgICAgKiBAcGFyYW0gZHN0T2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBkZXN0aW5hdGlvbiBhcnJheSB3aGVyZSB0aGUgbmV3IHZhbHVlcwotICAgICAqICAgICAgICAgICAgc2hvdWxkIGJlIHdyaXR0ZW4uCi0gICAgICogQHBhcmFtIGxlbmd0aAotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBwb2ludHMgdG8gdHJhbnNmb3JtLgotICAgICAqIEB0aHJvd3MgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgPGNvZGU+c3JjT2ZmICsgbGVuZ3RoKjIgPiBzcmMubGVuZ3RoPC9jb2RlPiBvcgotICAgICAqICAgICAgICAgICAgIDxjb2RlPmRzdE9mZiArIGxlbmd0aCoyID4gZHN0Lmxlbmd0aDwvY29kZT4uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgdHJhbnNmb3JtKGRvdWJsZVtdIHNyYywgaW50IHNyY09mZiwgZmxvYXRbXSBkc3QsIGludCBkc3RPZmYsIGludCBsZW5ndGgpIHsKLSAgICAgICAgd2hpbGUgKC0tbGVuZ3RoID49IDApIHsKLSAgICAgICAgICAgIGRvdWJsZSB4ID0gc3JjW3NyY09mZisrXTsKLSAgICAgICAgICAgIGRvdWJsZSB5ID0gc3JjW3NyY09mZisrXTsKLSAgICAgICAgICAgIGRzdFtkc3RPZmYrK10gPSAoZmxvYXQpKHggKiBtMDAgKyB5ICogbTAxICsgbTAyKTsKLSAgICAgICAgICAgIGRzdFtkc3RPZmYrK10gPSAoZmxvYXQpKHggKiBtMTAgKyB5ICogbTExICsgbTEyKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRyYW5zZm9ybXMgdGhlIHBvaW50IGFjY29yZGluZyB0byB0aGUgbGluZWFyIHRyYW5zZm9ybWF0aW9uIHBhcnQgb2YgdGhpcwotICAgICAqIEFmZmluZVRyYW5zZm9ybWF0aW9uICh3aXRob3V0IGFwcGx5aW5nIHRoZSB0cmFuc2xhdGlvbikuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIG9yaWdpbmFsIHBvaW50LgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb2ludCBvYmplY3Qgd2hlcmUgdGhlIHJlc3VsdCBvZiB0aGUgZGVsdGEgdHJhbnNmb3JtIGlzCi0gICAgICogICAgICAgICAgICB3cml0dGVuLgotICAgICAqIEByZXR1cm4gdGhlIHJlc3VsdCBvZiBhcHBseWluZyB0aGUgZGVsdGEgdHJhbnNmb3JtIChsaW5lYXIgcGFydCBvbmx5KSB0bwotICAgICAqICAgICAgICAgdGhlIG9yaWdpbmFsIHBvaW50LgotICAgICAqLwotICAgIC8vIFRPRE86IGlzIHRoaXMgcmlnaHQ/IGlmIGRzdCBpcyBudWxsLCB3ZSBjaGVjayB3aGF0IGl0J3MgYW4KLSAgICAvLyBpbnN0YW5jZSBvZj8gU2hvdWxkbid0IGl0IGJlIHNyYyBpbnN0YW5jZW9mIFBvaW50MkQuRG91YmxlPwotICAgIHB1YmxpYyBQb2ludDJEIGRlbHRhVHJhbnNmb3JtKFBvaW50MkQgc3JjLCBQb2ludDJEIGRzdCkgewotICAgICAgICBpZiAoZHN0ID09IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChkc3QgaW5zdGFuY2VvZiBQb2ludDJELkRvdWJsZSkgewotICAgICAgICAgICAgICAgIGRzdCA9IG5ldyBQb2ludDJELkRvdWJsZSgpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBkc3QgPSBuZXcgUG9pbnQyRC5GbG9hdCgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIHggPSBzcmMuZ2V0WCgpOwotICAgICAgICBkb3VibGUgeSA9IHNyYy5nZXRZKCk7Ci0KLSAgICAgICAgZHN0LnNldExvY2F0aW9uKHggKiBtMDAgKyB5ICogbTAxLCB4ICogbTEwICsgeSAqIG0xMSk7Ci0gICAgICAgIHJldHVybiBkc3Q7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXBwbGllcyB0aGUgbGluZWFyIHRyYW5zZm9ybWF0aW9uIHBhcnQgb2YgdGhpcyBBZmZpbmVUcmFuc2Zvcm0gKGlnbm9yaW5nCi0gICAgICogdGhlIHRyYW5zbGF0aW9uIHBhcnQpIHRvIGEgc2V0IG9mIHBvaW50cyBnaXZlbiBhcyBhbiBhcnJheSBvZiBkb3VibGUKLSAgICAgKiB2YWx1ZXMgd2hlcmUgZXZlcnkgdHdvIHZhbHVlcyBpbiB0aGUgYXJyYXkgZ2l2ZSB0aGUgY29vcmRpbmF0ZXMgb2YgYQotICAgICAqIHBvaW50OyB0aGUgZXZlbi1pbmRleGVkIHZhbHVlcyBnaXZpbmcgdGhlIHggY29vcmRpbmF0ZXMgYW5kIHRoZQotICAgICAqIG9kZC1pbmRleGVkIHZhbHVlcyBnaXZpbmcgdGhlIHkgY29vcmRpbmF0ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHBvaW50cyB0byBiZSB0cmFuc2Zvcm1lZC4KLSAgICAgKiBAcGFyYW0gc3JjT2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBzb3VyY2UgcG9pbnQgYXJyYXkgb2YgdGhlIGZpcnN0IHBvaW50IHRvIGJlCi0gICAgICogICAgICAgICAgICB0cmFuc2Zvcm1lZC4KLSAgICAgKiBAcGFyYW0gZHN0Ci0gICAgICogICAgICAgICAgICB0aGUgcG9pbnQgYXJyYXkgd2hlcmUgdGhlIGltYWdlcyBvZiB0aGUgcG9pbnRzIChhZnRlciBhcHBseWluZwotICAgICAqICAgICAgICAgICAgdGhlIGRlbHRhIHRyYW5zZm9ybWF0aW9uKSBzaG91bGQgYmUgcGxhY2VkLgotICAgICAqIEBwYXJhbSBkc3RPZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIGRlc3RpbmF0aW9uIGFycmF5IHdoZXJlIHRoZSBuZXcgdmFsdWVzCi0gICAgICogICAgICAgICAgICBzaG91bGQgYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gbGVuZ3RoCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIHBvaW50cyB0byB0cmFuc2Zvcm0uCi0gICAgICogQHRocm93cyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiA8Y29kZT5zcmNPZmYgKyBsZW5ndGgqMiA+IHNyYy5sZW5ndGg8L2NvZGU+IG9yCi0gICAgICogICAgICAgICAgICAgPGNvZGU+ZHN0T2ZmICsgbGVuZ3RoKjIgPiBkc3QubGVuZ3RoPC9jb2RlPi4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBkZWx0YVRyYW5zZm9ybShkb3VibGVbXSBzcmMsIGludCBzcmNPZmYsIGRvdWJsZVtdIGRzdCwgaW50IGRzdE9mZiwgaW50IGxlbmd0aCkgewotICAgICAgICB3aGlsZSAoLS1sZW5ndGggPj0gMCkgewotICAgICAgICAgICAgZG91YmxlIHggPSBzcmNbc3JjT2ZmKytdOwotICAgICAgICAgICAgZG91YmxlIHkgPSBzcmNbc3JjT2ZmKytdOwotICAgICAgICAgICAgZHN0W2RzdE9mZisrXSA9IHggKiBtMDAgKyB5ICogbTAxOwotICAgICAgICAgICAgZHN0W2RzdE9mZisrXSA9IHggKiBtMTAgKyB5ICogbTExOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVHJhbnNmb3JtcyB0aGUgcG9pbnQgYWNjb3JkaW5nIHRvIHRoZSBpbnZlcnNlIG9mIHRoaXMKLSAgICAgKiBBZmZpbmVUcmFuc2Zvcm1hdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgb3JpZ2luYWwgcG9pbnQuCi0gICAgICogQHBhcmFtIGRzdAotICAgICAqICAgICAgICAgICAgdGhlIHBvaW50IG9iamVjdCB3aGVyZSB0aGUgcmVzdWx0IG9mIHRoZSBpbnZlcnNlIHRyYW5zZm9ybSBpcwotICAgICAqICAgICAgICAgICAgd3JpdHRlbiAobWF5IGJlIG51bGwpLgotICAgICAqIEByZXR1cm4gdGhlIHJlc3VsdCBvZiBhcHBseWluZyB0aGUgaW52ZXJzZSB0cmFuc2Zvcm0uIEludmVyc2UgdHJhbnNmb3JtLgotICAgICAqIEB0aHJvd3MgTm9uaW52ZXJ0aWJsZVRyYW5zZm9ybUV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoaXMgQWZmaW5lVHJhbnNmb3JtIGNhbm5vdCBiZSBpbnZlcnRlZCAodGhlIGRldGVybWluYW50Ci0gICAgICogICAgICAgICAgICAgb2YgdGhlIGxpbmVhciB0cmFuc2Zvcm1hdGlvbiBwYXJ0IGlzIHplcm8pLgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludDJEIGludmVyc2VUcmFuc2Zvcm0oUG9pbnQyRCBzcmMsIFBvaW50MkQgZHN0KQotICAgICAgICAgICAgdGhyb3dzIE5vbmludmVydGlibGVUcmFuc2Zvcm1FeGNlcHRpb24gewotICAgICAgICBkb3VibGUgZGV0ID0gZ2V0RGV0ZXJtaW5hbnQoKTsKLSAgICAgICAgaWYgKE1hdGguYWJzKGRldCkgPCBaRVJPKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjA0PURldGVybWluYW50IGlzIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBOb25pbnZlcnRpYmxlVHJhbnNmb3JtRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIwNCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRzdCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBpZiAoc3JjIGluc3RhbmNlb2YgUG9pbnQyRC5Eb3VibGUpIHsKLSAgICAgICAgICAgICAgICBkc3QgPSBuZXcgUG9pbnQyRC5Eb3VibGUoKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZHN0ID0gbmV3IFBvaW50MkQuRmxvYXQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGRvdWJsZSB4ID0gc3JjLmdldFgoKSAtIG0wMjsKLSAgICAgICAgZG91YmxlIHkgPSBzcmMuZ2V0WSgpIC0gbTEyOwotCi0gICAgICAgIGRzdC5zZXRMb2NhdGlvbigoeCAqIG0xMSAtIHkgKiBtMDEpIC8gZGV0LCAoeSAqIG0wMCAtIHggKiBtMTApIC8gZGV0KTsKLSAgICAgICAgcmV0dXJuIGRzdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBcHBsaWVzIHRoZSBpbnZlcnNlIG9mIHRoaXMgQWZmaW5lVHJhbnNmb3JtIHRvIGEgc2V0IG9mIHBvaW50cyBnaXZlbiBhcwotICAgICAqIGFuIGFycmF5IG9mIGRvdWJsZSB2YWx1ZXMgd2hlcmUgZXZlcnkgdHdvIHZhbHVlcyBpbiB0aGUgYXJyYXkgZ2l2ZSB0aGUKLSAgICAgKiBjb29yZGluYXRlcyBvZiBhIHBvaW50OyB0aGUgZXZlbi1pbmRleGVkIHZhbHVlcyBnaXZpbmcgdGhlIHggY29vcmRpbmF0ZXMKLSAgICAgKiBhbmQgdGhlIG9kZC1pbmRleGVkIHZhbHVlcyBnaXZpbmcgdGhlIHkgY29vcmRpbmF0ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHBvaW50cyB0byBiZSB0cmFuc2Zvcm1lZC4KLSAgICAgKiBAcGFyYW0gc3JjT2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBzb3VyY2UgcG9pbnQgYXJyYXkgb2YgdGhlIGZpcnN0IHBvaW50IHRvIGJlCi0gICAgICogICAgICAgICAgICB0cmFuc2Zvcm1lZC4KLSAgICAgKiBAcGFyYW0gZHN0Ci0gICAgICogICAgICAgICAgICB0aGUgcG9pbnQgYXJyYXkgd2hlcmUgdGhlIGltYWdlcyBvZiB0aGUgcG9pbnRzIChhZnRlciBhcHBseWluZwotICAgICAqICAgICAgICAgICAgdGhlIGludmVyc2Ugb2YgdGhlIEFmZmluZVRyYW5zZm9ybWF0aW9uKSBzaG91bGQgYmUgcGxhY2VkLgotICAgICAqIEBwYXJhbSBkc3RPZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIGRlc3RpbmF0aW9uIGFycmF5IHdoZXJlIHRoZSBuZXcgdmFsdWVzCi0gICAgICogICAgICAgICAgICBzaG91bGQgYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gbGVuZ3RoCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIHBvaW50cyB0byB0cmFuc2Zvcm0uCi0gICAgICogQHRocm93cyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiA8Y29kZT5zcmNPZmYgKyBsZW5ndGgqMiA+IHNyYy5sZW5ndGg8L2NvZGU+IG9yCi0gICAgICogICAgICAgICAgICAgPGNvZGU+ZHN0T2ZmICsgbGVuZ3RoKjIgPiBkc3QubGVuZ3RoPC9jb2RlPi4KLSAgICAgKiBAdGhyb3dzIE5vbmludmVydGlibGVUcmFuc2Zvcm1FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGlzIEFmZmluZVRyYW5zZm9ybSBjYW5ub3QgYmUgaW52ZXJ0ZWQgKHRoZSBkZXRlcm1pbmFudAotICAgICAqICAgICAgICAgICAgIG9mIHRoZSBsaW5lYXIgdHJhbnNmb3JtYXRpb24gcGFydCBpcyB6ZXJvKS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBpbnZlcnNlVHJhbnNmb3JtKGRvdWJsZVtdIHNyYywgaW50IHNyY09mZiwgZG91YmxlW10gZHN0LCBpbnQgZHN0T2ZmLCBpbnQgbGVuZ3RoKQotICAgICAgICAgICAgdGhyb3dzIE5vbmludmVydGlibGVUcmFuc2Zvcm1FeGNlcHRpb24gewotICAgICAgICBkb3VibGUgZGV0ID0gZ2V0RGV0ZXJtaW5hbnQoKTsKLSAgICAgICAgaWYgKE1hdGguYWJzKGRldCkgPCBaRVJPKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjA0PURldGVybWluYW50IGlzIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBOb25pbnZlcnRpYmxlVHJhbnNmb3JtRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIwNCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgd2hpbGUgKC0tbGVuZ3RoID49IDApIHsKLSAgICAgICAgICAgIGRvdWJsZSB4ID0gc3JjW3NyY09mZisrXSAtIG0wMjsKLSAgICAgICAgICAgIGRvdWJsZSB5ID0gc3JjW3NyY09mZisrXSAtIG0xMjsKLSAgICAgICAgICAgIGRzdFtkc3RPZmYrK10gPSAoeCAqIG0xMSAtIHkgKiBtMDEpIC8gZGV0OwotICAgICAgICAgICAgZHN0W2RzdE9mZisrXSA9ICh5ICogbTAwIC0geCAqIG0xMCkgLyBkZXQ7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgbmV3IHNoYXBlIHdob3NlIGRhdGEgaXMgZ2l2ZW4gYnkgYXBwbHlpbmcgdGhpcyBBZmZpbmVUcmFuc2Zvcm0KLSAgICAgKiB0byB0aGUgc3BlY2lmaWVkIHNoYXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBvcmlnaW5hbCBzaGFwZSB3aG9zZSBkYXRhIGlzIHRvIGJlIHRyYW5zZm9ybWVkLgotICAgICAqIEByZXR1cm4gdGhlIG5ldyBzaGFwZSBmb3VuZCBieSBhcHBseWluZyB0aGlzIEFmZmluZVRyYW5zZm9ybSB0byB0aGUKLSAgICAgKiAgICAgICAgIG9yaWdpbmFsIHNoYXBlLgotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZSBjcmVhdGVUcmFuc2Zvcm1lZFNoYXBlKFNoYXBlIHNyYykgewotICAgICAgICBpZiAoc3JjID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgICAgIGlmIChzcmMgaW5zdGFuY2VvZiBHZW5lcmFsUGF0aCkgewotICAgICAgICAgICAgcmV0dXJuICgoR2VuZXJhbFBhdGgpc3JjKS5jcmVhdGVUcmFuc2Zvcm1lZFNoYXBlKHRoaXMpOwotICAgICAgICB9Ci0gICAgICAgIFBhdGhJdGVyYXRvciBwYXRoID0gc3JjLmdldFBhdGhJdGVyYXRvcih0aGlzKTsKLSAgICAgICAgR2VuZXJhbFBhdGggZHN0ID0gbmV3IEdlbmVyYWxQYXRoKHBhdGguZ2V0V2luZGluZ1J1bGUoKSk7Ci0gICAgICAgIGRzdC5hcHBlbmQocGF0aCwgZmFsc2UpOwotICAgICAgICByZXR1cm4gZHN0OwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgIHJldHVybiBnZXRDbGFzcygpLmdldE5hbWUoKSArICJbWyIgKyBtMDAgKyAiLCAiICsgbTAxICsgIiwgIiArIG0wMiArICJdLCBbIiAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJCAvLyROT04tTkxTLTQkCi0gICAgICAgICAgICAgICAgKyBtMTAgKyAiLCAiICsgbTExICsgIiwgIiArIG0xMiArICJdXSI7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBjbG9uZSgpIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBzdXBlci5jbG9uZSgpOwotICAgICAgICB9IGNhdGNoIChDbG9uZU5vdFN1cHBvcnRlZEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcigpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgSGFzaENvZGUgaGFzaCA9IG5ldyBIYXNoQ29kZSgpOwotICAgICAgICBoYXNoLmFwcGVuZChtMDApOwotICAgICAgICBoYXNoLmFwcGVuZChtMDEpOwotICAgICAgICBoYXNoLmFwcGVuZChtMDIpOwotICAgICAgICBoYXNoLmFwcGVuZChtMTApOwotICAgICAgICBoYXNoLmFwcGVuZChtMTEpOwotICAgICAgICBoYXNoLmFwcGVuZChtMTIpOwotICAgICAgICByZXR1cm4gaGFzaC5oYXNoQ29kZSgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3Qgb2JqKSB7Ci0gICAgICAgIGlmIChvYmogPT0gdGhpcykgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG9iaiBpbnN0YW5jZW9mIEFmZmluZVRyYW5zZm9ybSkgewotICAgICAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHQgPSAoQWZmaW5lVHJhbnNmb3JtKW9iajsKLSAgICAgICAgICAgIHJldHVybiBtMDAgPT0gdC5tMDAgJiYgbTAxID09IHQubTAxICYmIG0wMiA9PSB0Lm0wMiAmJiBtMTAgPT0gdC5tMTAgJiYgbTExID09IHQubTExCi0gICAgICAgICAgICAgICAgICAgICYmIG0xMiA9PSB0Lm0xMjsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIHRoZSBBZmZpbmVUcmFzc2Zvcm0gb2JqZWN0IHRvIHRoZSBvdXRwdXQgc3RlYW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cmVhbQotICAgICAqICAgICAgICAgICAgLSB0aGUgb3V0cHV0IHN0cmVhbS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgLSBpZiB0aGVyZSBhcmUgSS9PIGVycm9ycyB3aGlsZSB3cml0aW5nIHRvIHRoZSBvdXRwdXQgc3RyZWFtLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCB3cml0ZU9iamVjdChqYXZhLmlvLk9iamVjdE91dHB1dFN0cmVhbSBzdHJlYW0pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHN0cmVhbS5kZWZhdWx0V3JpdGVPYmplY3QoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkIHRoZSBBZmZpbmVUcmFuc2Zvcm0gb2JqZWN0IGZyb20gdGhlIGlucHV0IHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3RyZWFtCi0gICAgICogICAgICAgICAgICAtIHRoZSBpbnB1dCBzdHJlYW0uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIC0gaWYgdGhlcmUgYXJlIEkvTyBlcnJvcnMgd2hpbGUgcmVhZGluZyBmcm9tIHRoZSBpbnB1dAotICAgICAqICAgICAgICAgICAgIHN0cmVhbS4KLSAgICAgKiBAdGhyb3dzIENsYXNzTm90Rm91bmRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICAtIGlmIGNsYXNzIGNvdWxkIG5vdCBiZSBmb3VuZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgcmVhZE9iamVjdChqYXZhLmlvLk9iamVjdElucHV0U3RyZWFtIHN0cmVhbSkgdGhyb3dzIElPRXhjZXB0aW9uLAotICAgICAgICAgICAgQ2xhc3NOb3RGb3VuZEV4Y2VwdGlvbiB7Ci0gICAgICAgIHN0cmVhbS5kZWZhdWx0UmVhZE9iamVjdCgpOwotICAgICAgICB0eXBlID0gVFlQRV9VTktOT1dOOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2dlb20vQXJjMkQuamF2YSBiL2F3dC9qYXZhL2F3dC9nZW9tL0FyYzJELmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDU2ZjVjZDMuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2dlb20vQXJjMkQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDExNTcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZ2VvbTsKLQotaW1wb3J0IGphdmEudXRpbC5Ob1N1Y2hFbGVtZW50RXhjZXB0aW9uOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIENsYXNzIEFyYzJEIHJlcHJlc2VudHMgYSBzZWdtZW50IG9mIGEgY3VydmUgaW5zY3JpYmVkIGluIGEgcmVjdGFuZ2xlLiBUaGUKLSAqIGN1cnZlIGlzIGRlZmluZWQgYnkgYSBzdGFydCBhbmdsZSBhbmQgYW4gZXh0ZW50IGFuZ2xlICh0aGUgZW5kIGFuZ2xlIG1pbnVzCi0gKiB0aGUgc3RhcnQgYW5nbGUpIGFzIGEgcGllIHdlZGdlIHdob3NlIHBvaW50IGlzIGluIHRoZSBjZW50ZXIgb2YgdGhlCi0gKiByZWN0YW5nbGUuIFRoZSBBcmMyRCBhcyBhIHNoYXBlIG1heSBiZSBlaXRoZXIgT1BFTiAoaW5jbHVkaW5nIG5vdGhpbmcgYnV0IHRoZQotICogY3VydmVkIGFyYyBzZWdtZW50IGl0c2VsZiksIENIT1JEICh0aGUgY3VydmVkIGFyYyBzZWdtZW50IGNsb3NlZCBieSBhCi0gKiBjb25uZWN0aW5nIHNlZ21lbnQgZnJvbSB0aGUgZW5kIHRvIHRoZSBiZWdpbm5pbmcgb2YgdGhlIGFyYywgb3IgUElFICh0aGUKLSAqIHNlZ21lbnRzIGZyb20gdGhlIGVuZCBvZiB0aGUgYXJjIHRvIHRoZSBjZW50ZXIgb2YgdGhlIHJlY3RhbmdsZSBhbmQgZnJvbSB0aGUKLSAqIGNlbnRlciBvZiB0aGUgcmVjdGFuZ2xlIGJhY2sgdG8gdGhlIGFyYydzIHN0YXJ0IHBvaW50IGFyZSBpbmNsdWRlZCkuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgQXJjMkQgZXh0ZW5kcyBSZWN0YW5ndWxhclNoYXBlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBhcmMgdHlwZSBPUEVOIGluZGljYXRlcyB0aGF0IHRoZSBzaGFwZSBpbmNsdWRlcyBvbmx5IHRoZSBjdXJ2ZWQgYXJjCi0gICAgICogc2VnbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgc3RhdGljIGludCBPUEVOID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBhcmMgdHlwZSBDSE9SRCBpbmRpY2F0ZXMgdGhhdCBhcyBhIHNoYXBlIHRoZSBjb25uZWN0aW5nIHNlZ21lbnQgZnJvbQotICAgICAqIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBhcmMgdG8gdGhlIGJlZ2lubmluZyBwb2ludCBpcyBpbmNsdWRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgc3RhdGljIGludCBDSE9SRCA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYXJjIHR5cGUgUElFIGluZGljYXRlcyB0aGF0IGFzIGEgc2hhcGUgdGhlIHR3byBzZWdtZW50cyBmcm9tIHRoZQotICAgICAqIGFyYydzIGVuZHBvaW50IHRvIHRoZSBjZW50ZXIgb2YgdGhlIHJlY3RhbmdsZSBhbmQgZnJvbSB0aGUgY2VudGVyIG9mIHRoZQotICAgICAqIHJlY3RhbmdsZSB0byB0aGUgYXJjJ3MgZW5kcG9pbnQgYXJlIGluY2x1ZGVkLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBzdGF0aWMgaW50IFBJRSA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgRmxvYXQgaXMgYSBzdWJjbGFzcyBvZiBBcmMyRCBpbiB3aGljaCBhbGwgb2YgdGhlIGRhdGEgdmFsdWVzCi0gICAgICogYXJlIGdpdmVuIGFzIGZsb2F0cy4KLSAgICAgKiAKLSAgICAgKiBAc2VlIEFyYzJELkRvdWJsZQotICAgICAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgY2xhc3MgRmxvYXQgZXh0ZW5kcyBBcmMyRCB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUgdGhhdAotICAgICAgICAgKiBjb250YWlucyB0aGUgYXJjLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUgdGhhdAotICAgICAgICAgKiBjb250YWlucyB0aGUgYXJjLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIHRoYXQgY29udGFpbnMgdGhlIGFyYy4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCB3aWR0aDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlIHRoYXQgY29udGFpbnMgdGhlIGFyYy4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCBoZWlnaHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzdGFydCBhbmdsZSBvZiB0aGUgYXJjIGluIGRlZ3JlZXMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgc3RhcnQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB3aWR0aCBhbmdsZSBvZiB0aGUgYXJjIGluIGRlZ3JlZXMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgZXh0ZW50OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQXJjMkQgb2YgdHlwZSBPUEVOIHdpdGggZmxvYXQgdmFsdWVzLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEZsb2F0KCkgewotICAgICAgICAgICAgc3VwZXIoT1BFTik7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEFyYzJEIG9mIHRoZSBzcGVjaWZpZWQgdHlwZSB3aXRoIGZsb2F0IHZhbHVlcy4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSB0eXBlCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHR5cGUgb2YgdGhlIG5ldyBBcmMyRCwgZWl0aGVyIHtAbGluayBBcmMyRCNPUEVOfSwKLSAgICAgICAgICogICAgICAgICAgICB7QGxpbmsgQXJjMkQjQ0hPUkR9LCBvciB7QGxpbmsgQXJjMkQjUElFfS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBGbG9hdChpbnQgdHlwZSkgewotICAgICAgICAgICAgc3VwZXIodHlwZSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgQXJjMkQgd2l0aCB0aGUgc3BlY2lmaWVkIGZsb2F0LXZhbHVlZCBkYXRhLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgcmVjdGFuZ2xlCi0gICAgICAgICAqICAgICAgICAgICAgdGhhdCBjb250YWlucyB0aGUgYXJjLgotICAgICAgICAgKiBAcGFyYW0geQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUKLSAgICAgICAgICogICAgICAgICAgICB0aGF0IGNvbnRhaW5zIHRoZSBhcmMuCi0gICAgICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIHRoYXQgY29udGFpbnMgdGhlIGFyYy4KLSAgICAgICAgICogQHBhcmFtIGhlaWdodAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZSB0aGF0IGNvbnRhaW5zIHRoZSBhcmMuCi0gICAgICAgICAqIEBwYXJhbSBzdGFydAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBzdGFydCBhbmdsZSBvZiB0aGUgYXJjIGluIGRlZ3JlZXMuCi0gICAgICAgICAqIEBwYXJhbSBleHRlbnQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgd2lkdGggYW5nbGUgb2YgdGhlIGFyYyBpbiBkZWdyZWVzLgotICAgICAgICAgKiBAcGFyYW0gdHlwZQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB0eXBlIG9mIHRoZSBuZXcgQXJjMkQsIGVpdGhlciB7QGxpbmsgQXJjMkQjT1BFTn0sCi0gICAgICAgICAqICAgICAgICAgICAge0BsaW5rIEFyYzJEI0NIT1JEfSwgb3Ige0BsaW5rIEFyYzJEI1BJRX0uCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRmxvYXQoZmxvYXQgeCwgZmxvYXQgeSwgZmxvYXQgd2lkdGgsIGZsb2F0IGhlaWdodCwgZmxvYXQgc3RhcnQsIGZsb2F0IGV4dGVudCwKLSAgICAgICAgICAgICAgICBpbnQgdHlwZSkgewotICAgICAgICAgICAgc3VwZXIodHlwZSk7Ci0gICAgICAgICAgICB0aGlzLnggPSB4OwotICAgICAgICAgICAgdGhpcy55ID0geTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSB3aWR0aDsKLSAgICAgICAgICAgIHRoaXMuaGVpZ2h0ID0gaGVpZ2h0OwotICAgICAgICAgICAgdGhpcy5zdGFydCA9IHN0YXJ0OwotICAgICAgICAgICAgdGhpcy5leHRlbnQgPSBleHRlbnQ7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEFuZ2xlMkQgd2l0aCB0aGUgc3BlY2lmaWVkIGZsb2F0LXZhbHVlZCBkYXRhIGFuZAotICAgICAgICAgKiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlIGdpdmVuIGJ5IHRoZSBwYXJhbWV0ZXIgYm91bmRzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGJvdW5kcwotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBib3VuZGluZyByZWN0YW5nbGUgb2YgdGhlIEFuZ2xlMkQuCi0gICAgICAgICAqIEBwYXJhbSBzdGFydAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBzdGFydCBhbmdsZSBvZiB0aGUgYXJjIGluIGRlZ3JlZXMuCi0gICAgICAgICAqIEBwYXJhbSBleHRlbnQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgd2lkdGggYW5nbGUgb2YgdGhlIGFyYyBpbiBkZWdyZWVzLgotICAgICAgICAgKiBAcGFyYW0gdHlwZQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB0eXBlIG9mIHRoZSBuZXcgQXJjMkQsIGVpdGhlciB7QGxpbmsgQXJjMkQjT1BFTn0sCi0gICAgICAgICAqICAgICAgICAgICAge0BsaW5rIEFyYzJEI0NIT1JEfSwgb3Ige0BsaW5rIEFyYzJEI1BJRX0uCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRmxvYXQoUmVjdGFuZ2xlMkQgYm91bmRzLCBmbG9hdCBzdGFydCwgZmxvYXQgZXh0ZW50LCBpbnQgdHlwZSkgewotICAgICAgICAgICAgc3VwZXIodHlwZSk7Ci0gICAgICAgICAgICB0aGlzLnggPSAoZmxvYXQpYm91bmRzLmdldFgoKTsKLSAgICAgICAgICAgIHRoaXMueSA9IChmbG9hdClib3VuZHMuZ2V0WSgpOwotICAgICAgICAgICAgdGhpcy53aWR0aCA9IChmbG9hdClib3VuZHMuZ2V0V2lkdGgoKTsKLSAgICAgICAgICAgIHRoaXMuaGVpZ2h0ID0gKGZsb2F0KWJvdW5kcy5nZXRIZWlnaHQoKTsKLSAgICAgICAgICAgIHRoaXMuc3RhcnQgPSBzdGFydDsKLSAgICAgICAgICAgIHRoaXMuZXh0ZW50ID0gZXh0ZW50OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WCgpIHsKLSAgICAgICAgICAgIHJldHVybiB4OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WSgpIHsKLSAgICAgICAgICAgIHJldHVybiB5OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0V2lkdGgoKSB7Ci0gICAgICAgICAgICByZXR1cm4gd2lkdGg7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRIZWlnaHQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gaGVpZ2h0OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0QW5nbGVTdGFydCgpIHsKLSAgICAgICAgICAgIHJldHVybiBzdGFydDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldEFuZ2xlRXh0ZW50KCkgewotICAgICAgICAgICAgcmV0dXJuIGV4dGVudDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBpc0VtcHR5KCkgewotICAgICAgICAgICAgcmV0dXJuIHdpZHRoIDw9IDAuMGYgfHwgaGVpZ2h0IDw9IDAuMGY7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgc2V0QXJjKGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0LCBkb3VibGUgc3RhcnQsCi0gICAgICAgICAgICAgICAgZG91YmxlIGV4dGVudCwgaW50IHR5cGUpIHsKLSAgICAgICAgICAgIHRoaXMuc2V0QXJjVHlwZSh0eXBlKTsKLSAgICAgICAgICAgIHRoaXMueCA9IChmbG9hdCl4OwotICAgICAgICAgICAgdGhpcy55ID0gKGZsb2F0KXk7Ci0gICAgICAgICAgICB0aGlzLndpZHRoID0gKGZsb2F0KXdpZHRoOwotICAgICAgICAgICAgdGhpcy5oZWlnaHQgPSAoZmxvYXQpaGVpZ2h0OwotICAgICAgICAgICAgdGhpcy5zdGFydCA9IChmbG9hdClzdGFydDsKLSAgICAgICAgICAgIHRoaXMuZXh0ZW50ID0gKGZsb2F0KWV4dGVudDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRBbmdsZVN0YXJ0KGRvdWJsZSBzdGFydCkgewotICAgICAgICAgICAgdGhpcy5zdGFydCA9IChmbG9hdClzdGFydDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRBbmdsZUV4dGVudChkb3VibGUgZXh0ZW50KSB7Ci0gICAgICAgICAgICB0aGlzLmV4dGVudCA9IChmbG9hdClleHRlbnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHJvdGVjdGVkIFJlY3RhbmdsZTJEIG1ha2VCb3VuZHMoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlMkQuRmxvYXQoKGZsb2F0KXgsIChmbG9hdCl5LCAoZmxvYXQpd2lkdGgsIChmbG9hdCloZWlnaHQpOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgRG91YmxlIGlzIGEgc3ViY2xhc3Mgb2YgQXJjMkQgaW4gd2hpY2ggYWxsIG9mIHRoZSBkYXRhIHZhbHVlcwotICAgICAqIGFyZSBnaXZlbiBhcyBkb3VibGVzLgotICAgICAqIAotICAgICAqIEBzZWUgQXJjMkQuRmxvYXQKLSAgICAgKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGNsYXNzIERvdWJsZSBleHRlbmRzIEFyYzJEIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIHJlY3RhbmdsZSB0aGF0Ci0gICAgICAgICAqIGNvbnRhaW5zIHRoZSBhcmMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIHg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUgdGhhdAotICAgICAgICAgKiBjb250YWlucyB0aGUgYXJjLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB5OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZSB0aGF0IGNvbnRhaW5zIHRoZSBhcmMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIHdpZHRoOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUgdGhhdCBjb250YWlucyB0aGUgYXJjLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSBoZWlnaHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzdGFydCBhbmdsZSBvZiB0aGUgYXJjIGluIGRlZ3JlZXMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIHN0YXJ0OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgd2lkdGggYW5nbGUgb2YgdGhlIGFyYyBpbiBkZWdyZWVzLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSBleHRlbnQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBBcmMyRCBvZiB0eXBlIE9QRU4gd2l0aCBkb3VibGUgdmFsdWVzLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIERvdWJsZSgpIHsKLSAgICAgICAgICAgIHN1cGVyKE9QRU4pOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBBcmMyRCBvZiB0aGUgc3BlY2lmaWVkIHR5cGUgd2l0aCBkb3VibGUgdmFsdWVzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHR5cGUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgdHlwZSBvZiB0aGUgbmV3IEFyYzJELCBlaXRoZXIge0BsaW5rIEFyYzJEI09QRU59LAotICAgICAgICAgKiAgICAgICAgICAgIHtAbGluayBBcmMyRCNDSE9SRH0sIG9yIHtAbGluayBBcmMyRCNQSUV9LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIERvdWJsZShpbnQgdHlwZSkgewotICAgICAgICAgICAgc3VwZXIodHlwZSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgQXJjMkQgd2l0aCB0aGUgc3BlY2lmaWVkIGRvdWJsZS12YWx1ZWQgZGF0YS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSB4Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIHJlY3RhbmdsZQotICAgICAgICAgKiAgICAgICAgICAgIHRoYXQgY29udGFpbnMgdGhlIGFyYy4KLSAgICAgICAgICogQHBhcmFtIHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgcmVjdGFuZ2xlCi0gICAgICAgICAqICAgICAgICAgICAgdGhhdCBjb250YWlucyB0aGUgYXJjLgotICAgICAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZSB0aGF0IGNvbnRhaW5zIHRoZSBhcmMuCi0gICAgICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUgdGhhdCBjb250YWlucyB0aGUgYXJjLgotICAgICAgICAgKiBAcGFyYW0gc3RhcnQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgc3RhcnQgYW5nbGUgb2YgdGhlIGFyYyBpbiBkZWdyZWVzLgotICAgICAgICAgKiBAcGFyYW0gZXh0ZW50Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIGFuZ2xlIG9mIHRoZSBhcmMgaW4gZGVncmVlcy4KLSAgICAgICAgICogQHBhcmFtIHR5cGUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgdHlwZSBvZiB0aGUgbmV3IEFyYzJELCBlaXRoZXIge0BsaW5rIEFyYzJEI09QRU59LAotICAgICAgICAgKiAgICAgICAgICAgIHtAbGluayBBcmMyRCNDSE9SRH0sIG9yIHtAbGluayBBcmMyRCNQSUV9LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIERvdWJsZShkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3aWR0aCwgZG91YmxlIGhlaWdodCwgZG91YmxlIHN0YXJ0LCBkb3VibGUgZXh0ZW50LAotICAgICAgICAgICAgICAgIGludCB0eXBlKSB7Ci0gICAgICAgICAgICBzdXBlcih0eXBlKTsKLSAgICAgICAgICAgIHRoaXMueCA9IHg7Ci0gICAgICAgICAgICB0aGlzLnkgPSB5OwotICAgICAgICAgICAgdGhpcy53aWR0aCA9IHdpZHRoOwotICAgICAgICAgICAgdGhpcy5oZWlnaHQgPSBoZWlnaHQ7Ci0gICAgICAgICAgICB0aGlzLnN0YXJ0ID0gc3RhcnQ7Ci0gICAgICAgICAgICB0aGlzLmV4dGVudCA9IGV4dGVudDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQW5nbGUyRCB3aXRoIHRoZSBzcGVjaWZpZWQgZmxvYXQtdmFsdWVkIGRhdGEgYW5kCi0gICAgICAgICAqIHRoZSBib3VuZGluZyByZWN0YW5nbGUgZ2l2ZW4gYnkgdGhlIHBhcmFtZXRlciBib3VuZHMuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gYm91bmRzCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZSBvZiB0aGUgQW5nbGUyRC4KLSAgICAgICAgICogQHBhcmFtIHN0YXJ0Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IGFuZ2xlIG9mIHRoZSBhcmMgaW4gZGVncmVlcy4KLSAgICAgICAgICogQHBhcmFtIGV4dGVudAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBhbmdsZSBvZiB0aGUgYXJjIGluIGRlZ3JlZXMuCi0gICAgICAgICAqIEBwYXJhbSB0eXBlCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHR5cGUgb2YgdGhlIG5ldyBBcmMyRCwgZWl0aGVyIHtAbGluayBBcmMyRCNPUEVOfSwKLSAgICAgICAgICogICAgICAgICAgICB7QGxpbmsgQXJjMkQjQ0hPUkR9LCBvciB7QGxpbmsgQXJjMkQjUElFfS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBEb3VibGUoUmVjdGFuZ2xlMkQgYm91bmRzLCBkb3VibGUgc3RhcnQsIGRvdWJsZSBleHRlbnQsIGludCB0eXBlKSB7Ci0gICAgICAgICAgICBzdXBlcih0eXBlKTsKLSAgICAgICAgICAgIHRoaXMueCA9IGJvdW5kcy5nZXRYKCk7Ci0gICAgICAgICAgICB0aGlzLnkgPSBib3VuZHMuZ2V0WSgpOwotICAgICAgICAgICAgdGhpcy53aWR0aCA9IGJvdW5kcy5nZXRXaWR0aCgpOwotICAgICAgICAgICAgdGhpcy5oZWlnaHQgPSBib3VuZHMuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICB0aGlzLnN0YXJ0ID0gc3RhcnQ7Ci0gICAgICAgICAgICB0aGlzLmV4dGVudCA9IGV4dGVudDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFgoKSB7Ci0gICAgICAgICAgICByZXR1cm4geDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFkoKSB7Ci0gICAgICAgICAgICByZXR1cm4geTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFdpZHRoKCkgewotICAgICAgICAgICAgcmV0dXJuIHdpZHRoOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0SGVpZ2h0KCkgewotICAgICAgICAgICAgcmV0dXJuIGhlaWdodDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldEFuZ2xlU3RhcnQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gc3RhcnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRBbmdsZUV4dGVudCgpIHsKLSAgICAgICAgICAgIHJldHVybiBleHRlbnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGJvb2xlYW4gaXNFbXB0eSgpIHsKLSAgICAgICAgICAgIHJldHVybiB3aWR0aCA8PSAwLjAgfHwgaGVpZ2h0IDw9IDAuMDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRBcmMoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQsIGRvdWJsZSBzdGFydCwKLSAgICAgICAgICAgICAgICBkb3VibGUgZXh0ZW50LCBpbnQgdHlwZSkgewotICAgICAgICAgICAgdGhpcy5zZXRBcmNUeXBlKHR5cGUpOwotICAgICAgICAgICAgdGhpcy54ID0geDsKLSAgICAgICAgICAgIHRoaXMueSA9IHk7Ci0gICAgICAgICAgICB0aGlzLndpZHRoID0gd2lkdGg7Ci0gICAgICAgICAgICB0aGlzLmhlaWdodCA9IGhlaWdodDsKLSAgICAgICAgICAgIHRoaXMuc3RhcnQgPSBzdGFydDsKLSAgICAgICAgICAgIHRoaXMuZXh0ZW50ID0gZXh0ZW50OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyB2b2lkIHNldEFuZ2xlU3RhcnQoZG91YmxlIHN0YXJ0KSB7Ci0gICAgICAgICAgICB0aGlzLnN0YXJ0ID0gc3RhcnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgc2V0QW5nbGVFeHRlbnQoZG91YmxlIGV4dGVudCkgewotICAgICAgICAgICAgdGhpcy5leHRlbnQgPSBleHRlbnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHJvdGVjdGVkIFJlY3RhbmdsZTJEIG1ha2VCb3VuZHMoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgSXRlcmF0b3IgaXMgdGhlIHN1YmNsYXNzIG9mIFBhdGhJdGVyYXRvciB0aGF0IGlzIHVzZWQgdG8KLSAgICAgKiB0cmF2ZXJzZSB0aGUgYm91bmRhcnkgb2YgYSBzaGFwZSBvZiB0eXBlIEFyYzJELgotICAgICAqLwotICAgIGNsYXNzIEl0ZXJhdG9yIGltcGxlbWVudHMgUGF0aEl0ZXJhdG9yIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgY2VudGVyIG9mIHRoZSBhcmMncyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgeDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgY2VudGVyIG9mIHRoZSBhcmMncyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgeTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSGFsZiBvZiB0aGUgd2lkdGggb2YgdGhlIGFyYydzIGJvdW5kaW5nIHJlY3RhbmdsZSAodGhlIHJhZGl1cyBpbiB0aGUKLSAgICAgICAgICogY2FzZSBvZiBhIGNpcmN1bGFyIGFyYykuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgd2lkdGg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEhhbGYgb2YgdGhlIGhlaWdodCBvZiB0aGUgYXJjJ3MgYm91bmRpbmcgcmVjdGFuZ2xlICh0aGUgcmFkaXVzIGluIHRoZQotICAgICAgICAgKiBjYXNlIG9mIGEgY2lyY3VsYXIgYXJjKS4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSBoZWlnaHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzdGFydCBhbmdsZSBvZiB0aGUgYXJjIGluIGRlZ3JlZXMuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgYW5nbGU7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBhbmdsZSBleHRlbnQgaW4gZGVncmVlcy4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSBleHRlbnQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjbG9zdXJlIHR5cGUgb2YgdGhlIGFyYy4KLSAgICAgICAgICovCi0gICAgICAgIGludCB0eXBlOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcGF0aCBpdGVyYXRvciB0cmFuc2Zvcm1hdGlvbi4KLSAgICAgICAgICovCi0gICAgICAgIEFmZmluZVRyYW5zZm9ybSB0OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgY3VycmVudCBzZWdtZW50IGluZGV4LgotICAgICAgICAgKi8KLSAgICAgICAgaW50IGluZGV4OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbnVtYmVyIG9mIGFyYyBzZWdtZW50cyB0aGUgc291cmNlIGFyYyBzdWJkaXZpZGVkIHRvIGJlCi0gICAgICAgICAqIGFwcHJveGltYXRlZCBieSBCZXppZXIgY3VydmVzLiBEZXBlbmRzIG9uIGV4dGVudCB2YWx1ZS4KLSAgICAgICAgICovCi0gICAgICAgIGludCBhcmNDb3VudDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG51bWJlciBvZiBsaW5lIHNlZ21lbnRzLiBEZXBlbmRzIG9uIGNsb3N1cmUgdHlwZS4KLSAgICAgICAgICovCi0gICAgICAgIGludCBsaW5lQ291bnQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzdGVwIHRvIGNhbGN1bGF0ZSBuZXh0IGFyYyBzdWJkaXZpc2lvbiBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSBzdGVwOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgdGVtcG9yYXJ5IHZhbHVlIG9mIGNvc2ludXMgb2YgdGhlIGN1cnJlbnQgYW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgY29zOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgdGVtcG9yYXJ5IHZhbHVlIG9mIHNpbnVzIG9mIHRoZSBjdXJyZW50IGFuZ2xlLgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIHNpbjsKLQotICAgICAgICAvKiogVGhlIGNvZWZmaWNpZW50IHRvIGNhbGN1bGF0ZSBjb250cm9sIHBvaW50cyBvZiBCZXppZXIgY3VydmVzLiAqLwotICAgICAgICBkb3VibGUgazsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHRlbXBvcmFyeSB2YWx1ZSBvZiB4IGNvb3JkaW5hdGUgb2YgdGhlIEJlemllciBjdXJ2ZSBjb250cm9sCi0gICAgICAgICAqIHZlY3Rvci4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSBreDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHRlbXBvcmFyeSB2YWx1ZSBvZiB5IGNvb3JkaW5hdGUgb2YgdGhlIEJlemllciBjdXJ2ZSBjb250cm9sCi0gICAgICAgICAqIHZlY3Rvci4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSBreTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgcGF0aCBwb2ludCAoTU9WRV9UTykuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgbXg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IHBhdGggcG9pbnQgKE1PVkVfVE8pLgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIG15OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDb25zdHJ1Y3RzIGEgbmV3IEFyYzJELkl0ZXJhdG9yIGZvciBnaXZlbiBsaW5lIGFuZCB0cmFuc2Zvcm1hdGlvbgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgc291cmNlIEFyYzJEIG9iamVjdC4KLSAgICAgICAgICogQHBhcmFtIHQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtYXRpb24uCi0gICAgICAgICAqLwotICAgICAgICBJdGVyYXRvcihBcmMyRCBhLCBBZmZpbmVUcmFuc2Zvcm0gdCkgewotICAgICAgICAgICAgaWYgKHdpZHRoIDwgMCB8fCBoZWlnaHQgPCAwKSB7Ci0gICAgICAgICAgICAgICAgYXJjQ291bnQgPSAwOwotICAgICAgICAgICAgICAgIGxpbmVDb3VudCA9IDA7Ci0gICAgICAgICAgICAgICAgaW5kZXggPSAxOwotICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgdGhpcy53aWR0aCA9IGEuZ2V0V2lkdGgoKSAvIDIuMDsKLSAgICAgICAgICAgIHRoaXMuaGVpZ2h0ID0gYS5nZXRIZWlnaHQoKSAvIDIuMDsKLSAgICAgICAgICAgIHRoaXMueCA9IGEuZ2V0WCgpICsgd2lkdGg7Ci0gICAgICAgICAgICB0aGlzLnkgPSBhLmdldFkoKSArIGhlaWdodDsKLSAgICAgICAgICAgIHRoaXMuYW5nbGUgPSAtTWF0aC50b1JhZGlhbnMoYS5nZXRBbmdsZVN0YXJ0KCkpOwotICAgICAgICAgICAgdGhpcy5leHRlbnQgPSAtYS5nZXRBbmdsZUV4dGVudCgpOwotICAgICAgICAgICAgdGhpcy50eXBlID0gYS5nZXRBcmNUeXBlKCk7Ci0gICAgICAgICAgICB0aGlzLnQgPSB0OwotCi0gICAgICAgICAgICBpZiAoTWF0aC5hYnMoZXh0ZW50KSA+PSAzNjAuMCkgewotICAgICAgICAgICAgICAgIGFyY0NvdW50ID0gNDsKLSAgICAgICAgICAgICAgICBrID0gNC4wIC8gMy4wICogKE1hdGguc3FydCgyLjApIC0gMS4wKTsKLSAgICAgICAgICAgICAgICBzdGVwID0gTWF0aC5QSSAvIDIuMDsKLSAgICAgICAgICAgICAgICBpZiAoZXh0ZW50IDwgMC4wKSB7Ci0gICAgICAgICAgICAgICAgICAgIHN0ZXAgPSAtc3RlcDsKLSAgICAgICAgICAgICAgICAgICAgayA9IC1rOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgYXJjQ291bnQgPSAoaW50KU1hdGgucmludChNYXRoLmFicyhleHRlbnQpIC8gOTAuMCk7Ci0gICAgICAgICAgICAgICAgc3RlcCA9IE1hdGgudG9SYWRpYW5zKGV4dGVudCAvIGFyY0NvdW50KTsKLSAgICAgICAgICAgICAgICBrID0gNC4wIC8gMy4wICogKDEuMCAtIE1hdGguY29zKHN0ZXAgLyAyLjApKSAvIE1hdGguc2luKHN0ZXAgLyAyLjApOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBsaW5lQ291bnQgPSAwOwotICAgICAgICAgICAgaWYgKHR5cGUgPT0gQXJjMkQuQ0hPUkQpIHsKLSAgICAgICAgICAgICAgICBsaW5lQ291bnQrKzsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAodHlwZSA9PSBBcmMyRC5QSUUpIHsKLSAgICAgICAgICAgICAgICBsaW5lQ291bnQgKz0gMjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgZ2V0V2luZGluZ1J1bGUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gV0lORF9OT05fWkVSTzsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzRG9uZSgpIHsKLSAgICAgICAgICAgIHJldHVybiBpbmRleCA+IGFyY0NvdW50ICsgbGluZUNvdW50OwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIHZvaWQgbmV4dCgpIHsKLSAgICAgICAgICAgIGluZGV4Kys7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGRvdWJsZVtdIGNvb3JkcykgewotICAgICAgICAgICAgaWYgKGlzRG9uZSgpKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjRCPUl0ZXJhdG9yIG91dCBvZiBib3VuZHMKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgTm9TdWNoRWxlbWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40QiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgaW50IHR5cGU7Ci0gICAgICAgICAgICBpbnQgY291bnQ7Ci0gICAgICAgICAgICBpZiAoaW5kZXggPT0gMCkgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTU9WRVRPOwotICAgICAgICAgICAgICAgIGNvdW50ID0gMTsKLSAgICAgICAgICAgICAgICBjb3MgPSBNYXRoLmNvcyhhbmdsZSk7Ci0gICAgICAgICAgICAgICAgc2luID0gTWF0aC5zaW4oYW5nbGUpOwotICAgICAgICAgICAgICAgIGt4ID0gayAqIHdpZHRoICogc2luOwotICAgICAgICAgICAgICAgIGt5ID0gayAqIGhlaWdodCAqIGNvczsKLSAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSBteCA9IHggKyBjb3MgKiB3aWR0aDsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSBteSA9IHkgKyBzaW4gKiBoZWlnaHQ7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGluZGV4IDw9IGFyY0NvdW50KSB7Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19DVUJJQ1RPOwotICAgICAgICAgICAgICAgIGNvdW50ID0gMzsKLSAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSBteCAtIGt4OwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IG15ICsga3k7Ci0gICAgICAgICAgICAgICAgYW5nbGUgKz0gc3RlcDsKLSAgICAgICAgICAgICAgICBjb3MgPSBNYXRoLmNvcyhhbmdsZSk7Ci0gICAgICAgICAgICAgICAgc2luID0gTWF0aC5zaW4oYW5nbGUpOwotICAgICAgICAgICAgICAgIGt4ID0gayAqIHdpZHRoICogc2luOwotICAgICAgICAgICAgICAgIGt5ID0gayAqIGhlaWdodCAqIGNvczsKLSAgICAgICAgICAgICAgICBjb29yZHNbNF0gPSBteCA9IHggKyBjb3MgKiB3aWR0aDsKLSAgICAgICAgICAgICAgICBjb29yZHNbNV0gPSBteSA9IHkgKyBzaW4gKiBoZWlnaHQ7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzJdID0gbXggKyBreDsKLSAgICAgICAgICAgICAgICBjb29yZHNbM10gPSBteSAtIGt5OwotICAgICAgICAgICAgfSBlbHNlIGlmIChpbmRleCA9PSBhcmNDb3VudCArIGxpbmVDb3VudCkgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfQ0xPU0U7Ci0gICAgICAgICAgICAgICAgY291bnQgPSAwOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX0xJTkVUTzsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDE7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzBdID0geDsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSB5OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCBjb3VudCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gdHlwZTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgY3VycmVudFNlZ21lbnQoZmxvYXRbXSBjb29yZHMpIHsKLSAgICAgICAgICAgIGlmIChpc0RvbmUoKSkgewotICAgICAgICAgICAgICAgIC8vIGF3dC40Qj1JdGVyYXRvciBvdXQgb2YgYm91bmRzCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IE5vU3VjaEVsZW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNEIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGludCB0eXBlOwotICAgICAgICAgICAgaW50IGNvdW50OwotICAgICAgICAgICAgaWYgKGluZGV4ID09IDApIHsKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX01PVkVUTzsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDE7Ci0gICAgICAgICAgICAgICAgY29zID0gTWF0aC5jb3MoYW5nbGUpOwotICAgICAgICAgICAgICAgIHNpbiA9IE1hdGguc2luKGFuZ2xlKTsKLSAgICAgICAgICAgICAgICBreCA9IGsgKiB3aWR0aCAqIHNpbjsKLSAgICAgICAgICAgICAgICBreSA9IGsgKiBoZWlnaHQgKiBjb3M7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzBdID0gKGZsb2F0KShteCA9IHggKyBjb3MgKiB3aWR0aCk7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzFdID0gKGZsb2F0KShteSA9IHkgKyBzaW4gKiBoZWlnaHQpOwotICAgICAgICAgICAgfSBlbHNlIGlmIChpbmRleCA8PSBhcmNDb3VudCkgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfQ1VCSUNUTzsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDM7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzBdID0gKGZsb2F0KShteCAtIGt4KTsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSAoZmxvYXQpKG15ICsga3kpOwotICAgICAgICAgICAgICAgIGFuZ2xlICs9IHN0ZXA7Ci0gICAgICAgICAgICAgICAgY29zID0gTWF0aC5jb3MoYW5nbGUpOwotICAgICAgICAgICAgICAgIHNpbiA9IE1hdGguc2luKGFuZ2xlKTsKLSAgICAgICAgICAgICAgICBreCA9IGsgKiB3aWR0aCAqIHNpbjsKLSAgICAgICAgICAgICAgICBreSA9IGsgKiBoZWlnaHQgKiBjb3M7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzRdID0gKGZsb2F0KShteCA9IHggKyBjb3MgKiB3aWR0aCk7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzVdID0gKGZsb2F0KShteSA9IHkgKyBzaW4gKiBoZWlnaHQpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1syXSA9IChmbG9hdCkobXggKyBreCk7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzNdID0gKGZsb2F0KShteSAtIGt5KTsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoaW5kZXggPT0gYXJjQ291bnQgKyBsaW5lQ291bnQpIHsKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX0NMT1NFOwotICAgICAgICAgICAgICAgIGNvdW50ID0gMDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19MSU5FVE87Ci0gICAgICAgICAgICAgICAgY291bnQgPSAxOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IChmbG9hdCl4OwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IChmbG9hdCl5OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCBjb3VudCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gdHlwZTsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIGNsb3N1cmUgdHlwZSBvZiB0aGUgYXJjLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IHR5cGU7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgYXJjMkQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBjbG9zdXJlIHR5cGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEFyYzJEKGludCB0eXBlKSB7Ci0gICAgICAgIHNldEFyY1R5cGUodHlwZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGFrZXMgdGhlIGRvdWJsZS12YWx1ZWQgZGF0YSBhbmQgY3JlYXRlcyB0aGUgY29ycmVzcG9uZGluZyBSZWN0YW5nbGUyRAotICAgICAqIG9iamVjdCB3aXRoIHZhbHVlcyBlaXRoZXIgb2YgdHlwZSBmbG9hdCBvciBvZiB0eXBlIGRvdWJsZSBkZXBlbmRpbmcgb24KLSAgICAgKiB3aGV0aGVyIHRoaXMgQXJjMkQgaW5zdGFuY2UgaXMgb2YgdHlwZSBGbG9hdCBvciBEb3VibGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSBib3VuZGluZwotICAgICAqICAgICAgICAgICAgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgYm91bmRpbmcKLSAgICAgKiAgICAgICAgICAgIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBjb3JyZXNwb25kaW5nIFJlY3RhbmdsZTJEIG9iamVjdC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgYWJzdHJhY3QgUmVjdGFuZ2xlMkQgbWFrZUJvdW5kcyhkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3aWR0aCwgZG91YmxlIGhlaWdodCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzdGFydCBhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdGFydCBhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldEFuZ2xlU3RhcnQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHdpZHRoIGFuZ2xlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHdpZHRoIGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0QW5nbGVFeHRlbnQoKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHN0YXJ0IGFuZ2xlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdGFydAotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBzdGFydCBhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzZXRBbmdsZVN0YXJ0KGRvdWJsZSBzdGFydCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSB3aWR0aCBhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXh0ZW50Ci0gICAgICogICAgICAgICAgICB0aGUgbmV3IHdpZHRoIGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldEFuZ2xlRXh0ZW50KGRvdWJsZSBleHRlbnQpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGF0YSB2YWx1ZXMgdGhhdCBkZWZpbmUgdGhlIGFyYy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIHJlY3RhbmdsZQotICAgICAqICAgICAgICAgICAgdGhhdCBjb250YWlucyB0aGUgYXJjLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgcmVjdGFuZ2xlCi0gICAgICogICAgICAgICAgICB0aGF0IGNvbnRhaW5zIHRoZSBhcmMuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZSB0aGF0IGNvbnRhaW5zIHRoZSBhcmMuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlIHRoYXQgY29udGFpbnMgdGhlIGFyYy4KLSAgICAgKiBAcGFyYW0gc3RhcnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdGFydCBhbmdsZSBvZiB0aGUgYXJjIGluIGRlZ3JlZXMuCi0gICAgICogQHBhcmFtIGV4dGVudAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIGFuZ2xlIG9mIHRoZSBhcmMgaW4gZGVncmVlcy4KLSAgICAgKiBAcGFyYW0gdHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHR5cGUgb2YgdGhlIG5ldyBBcmMyRCwgZWl0aGVyIHtAbGluayBBcmMyRCNPUEVOfSwKLSAgICAgKiAgICAgICAgICAgIHtAbGluayBBcmMyRCNDSE9SRH0sIG9yIHtAbGluayBBcmMyRCNQSUV9LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldEFyYyhkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3aWR0aCwgZG91YmxlIGhlaWdodCwgZG91YmxlIHN0YXJ0LAotICAgICAgICAgICAgZG91YmxlIGV4dGVudCwgaW50IHR5cGUpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYXJjIHR5cGUsIGVpdGhlciB7QGxpbmsgQXJjMkQjT1BFTn0sIHtAbGluayBBcmMyRCNDSE9SRH0sIG9yCi0gICAgICoge0BsaW5rIEFyYzJEI1BJRX0uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJjIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRBcmNUeXBlKCkgewotICAgICAgICByZXR1cm4gdHlwZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBhcmMgdHlwZSwgZWl0aGVyIHtAbGluayBBcmMyRCNPUEVOfSwge0BsaW5rIEFyYzJEI0NIT1JEfSwgb3IKLSAgICAgKiB7QGxpbmsgQXJjMkQjUElFfS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdHlwZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBhcmMgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRBcmNUeXBlKGludCB0eXBlKSB7Ci0gICAgICAgIGlmICh0eXBlICE9IE9QRU4gJiYgdHlwZSAhPSBDSE9SRCAmJiB0eXBlICE9IFBJRSkgewotICAgICAgICAgICAgLy8gYXd0LjIwNT1JbnZhbGlkIHR5cGUgb2YgQXJjOiB7MH0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjA1IiwgdHlwZSkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy50eXBlID0gdHlwZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzdGFydCBwb2ludCBvZiB0aGUgYXJjIGFzIGEgUG9pbnQyRC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdGFydCBwb2ludCBvZiB0aGUgY3VydmVkIGFyYyBzZWdtZW50LgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludDJEIGdldFN0YXJ0UG9pbnQoKSB7Ci0gICAgICAgIGRvdWJsZSBhID0gTWF0aC50b1JhZGlhbnMoZ2V0QW5nbGVTdGFydCgpKTsKLSAgICAgICAgcmV0dXJuIG5ldyBQb2ludDJELkRvdWJsZShnZXRYKCkgKyAoMS4wICsgTWF0aC5jb3MoYSkpICogZ2V0V2lkdGgoKSAvIDIuMCwgZ2V0WSgpCi0gICAgICAgICAgICAgICAgKyAoMS4wIC0gTWF0aC5zaW4oYSkpICogZ2V0SGVpZ2h0KCkgLyAyLjApOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGVuZCBwb2ludCBvZiB0aGUgYXJjIGFzIGEgUG9pbnQyRC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBhcmMgc2VnbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUG9pbnQyRCBnZXRFbmRQb2ludCgpIHsKLSAgICAgICAgZG91YmxlIGEgPSBNYXRoLnRvUmFkaWFucyhnZXRBbmdsZVN0YXJ0KCkgKyBnZXRBbmdsZUV4dGVudCgpKTsKLSAgICAgICAgcmV0dXJuIG5ldyBQb2ludDJELkRvdWJsZShnZXRYKCkgKyAoMS4wICsgTWF0aC5jb3MoYSkpICogZ2V0V2lkdGgoKSAvIDIuMCwgZ2V0WSgpCi0gICAgICAgICAgICAgICAgKyAoMS4wIC0gTWF0aC5zaW4oYSkpICogZ2V0SGVpZ2h0KCkgLyAyLjApOwotICAgIH0KLQotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRCgpIHsKLSAgICAgICAgaWYgKGlzRW1wdHkoKSkgewotICAgICAgICAgICAgcmV0dXJuIG1ha2VCb3VuZHMoZ2V0WCgpLCBnZXRZKCksIGdldFdpZHRoKCksIGdldEhlaWdodCgpKTsKLSAgICAgICAgfQotICAgICAgICBkb3VibGUgcngxID0gZ2V0WCgpOwotICAgICAgICBkb3VibGUgcnkxID0gZ2V0WSgpOwotICAgICAgICBkb3VibGUgcngyID0gcngxICsgZ2V0V2lkdGgoKTsKLSAgICAgICAgZG91YmxlIHJ5MiA9IHJ5MSArIGdldEhlaWdodCgpOwotCi0gICAgICAgIFBvaW50MkQgcDEgPSBnZXRTdGFydFBvaW50KCk7Ci0gICAgICAgIFBvaW50MkQgcDIgPSBnZXRFbmRQb2ludCgpOwotCi0gICAgICAgIGRvdWJsZSBieDEgPSBjb250YWluc0FuZ2xlKDE4MC4wKSA/IHJ4MSA6IE1hdGgubWluKHAxLmdldFgoKSwgcDIuZ2V0WCgpKTsKLSAgICAgICAgZG91YmxlIGJ5MSA9IGNvbnRhaW5zQW5nbGUoOTAuMCkgPyByeTEgOiBNYXRoLm1pbihwMS5nZXRZKCksIHAyLmdldFkoKSk7Ci0gICAgICAgIGRvdWJsZSBieDIgPSBjb250YWluc0FuZ2xlKDAuMCkgPyByeDIgOiBNYXRoLm1heChwMS5nZXRYKCksIHAyLmdldFgoKSk7Ci0gICAgICAgIGRvdWJsZSBieTIgPSBjb250YWluc0FuZ2xlKDI3MC4wKSA/IHJ5MiA6IE1hdGgubWF4KHAxLmdldFkoKSwgcDIuZ2V0WSgpKTsKLQotICAgICAgICBpZiAodHlwZSA9PSBQSUUpIHsKLSAgICAgICAgICAgIGRvdWJsZSBjeCA9IGdldENlbnRlclgoKTsKLSAgICAgICAgICAgIGRvdWJsZSBjeSA9IGdldENlbnRlclkoKTsKLSAgICAgICAgICAgIGJ4MSA9IE1hdGgubWluKGJ4MSwgY3gpOwotICAgICAgICAgICAgYnkxID0gTWF0aC5taW4oYnkxLCBjeSk7Ci0gICAgICAgICAgICBieDIgPSBNYXRoLm1heChieDIsIGN4KTsKLSAgICAgICAgICAgIGJ5MiA9IE1hdGgubWF4KGJ5MiwgY3kpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBtYWtlQm91bmRzKGJ4MSwgYnkxLCBieDIgLSBieDEsIGJ5MiAtIGJ5MSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0RnJhbWUoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQpIHsKLSAgICAgICAgc2V0QXJjKHgsIHksIHdpZHRoLCBoZWlnaHQsIGdldEFuZ2xlU3RhcnQoKSwgZ2V0QW5nbGVFeHRlbnQoKSwgdHlwZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGF0YSB0aGF0IGRlZmluZXMgdGhlIGFyYy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcG9pbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgc2l6ZSBvZiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBzdGFydAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IGFuZ2xlIG9mIHRoZSBhcmMgaW4gZGVncmVlcy4KLSAgICAgKiBAcGFyYW0gZXh0ZW50Ci0gICAgICogICAgICAgICAgICB0aGUgYW5nbGUgd2lkdGggb2YgdGhlIGFyYyBpbiBkZWdyZWVzLgotICAgICAqIEBwYXJhbSB0eXBlCi0gICAgICogICAgICAgICAgICB0aGUgY2xvc3VyZSB0eXBlLCBlaXRoZXIge0BsaW5rIEFyYzJEI09QRU59LAotICAgICAqICAgICAgICAgICAge0BsaW5rIEFyYzJEI0NIT1JEfSwgb3Ige0BsaW5rIEFyYzJEI1BJRX0uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0QXJjKFBvaW50MkQgcG9pbnQsIERpbWVuc2lvbjJEIHNpemUsIGRvdWJsZSBzdGFydCwgZG91YmxlIGV4dGVudCwgaW50IHR5cGUpIHsKLSAgICAgICAgc2V0QXJjKHBvaW50LmdldFgoKSwgcG9pbnQuZ2V0WSgpLCBzaXplLmdldFdpZHRoKCksIHNpemUuZ2V0SGVpZ2h0KCksIHN0YXJ0LCBleHRlbnQsIHR5cGUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgdGhhdCBkZWZpbmVzIHRoZSBhcmMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJlY3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcmMncyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHN0YXJ0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgYW5nbGUgb2YgdGhlIGFyYyBpbiBkZWdyZWVzLgotICAgICAqIEBwYXJhbSBleHRlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbmdsZSB3aWR0aCBvZiB0aGUgYXJjIGluIGRlZ3JlZXMuCi0gICAgICogQHBhcmFtIHR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBjbG9zdXJlIHR5cGUsIGVpdGhlciB7QGxpbmsgQXJjMkQjT1BFTn0sCi0gICAgICogICAgICAgICAgICB7QGxpbmsgQXJjMkQjQ0hPUkR9LCBvciB7QGxpbmsgQXJjMkQjUElFfS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRBcmMoUmVjdGFuZ2xlMkQgcmVjdCwgZG91YmxlIHN0YXJ0LCBkb3VibGUgZXh0ZW50LCBpbnQgdHlwZSkgewotICAgICAgICBzZXRBcmMocmVjdC5nZXRYKCksIHJlY3QuZ2V0WSgpLCByZWN0LmdldFdpZHRoKCksIHJlY3QuZ2V0SGVpZ2h0KCksIHN0YXJ0LCBleHRlbnQsIHR5cGUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgdGhhdCBkZWZpbmVzIHRoZSBhcmMgYnkgY29weWluZyBpdCBmcm9tIGFub3RoZXIgQXJjMkQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGFyYwotICAgICAqICAgICAgICAgICAgdGhlIGFyYyB3aG9zZSBkYXRhIGlzIGNvcGllZCBpbnRvIHRoaXMgYXJjLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEFyYyhBcmMyRCBhcmMpIHsKLSAgICAgICAgc2V0QXJjKGFyYy5nZXRYKCksIGFyYy5nZXRZKCksIGFyYy5nZXRXaWR0aCgpLCBhcmMuZ2V0SGVpZ2h0KCksIGFyYy5nZXRBbmdsZVN0YXJ0KCksIGFyYwotICAgICAgICAgICAgICAgIC5nZXRBbmdsZUV4dGVudCgpLCBhcmMuZ2V0QXJjVHlwZSgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBkYXRhIGZvciBhIGNpcmN1bGFyIGFyYyBieSBnaXZpbmcgaXRzIGNlbnRlciBhbmQgcmFkaXVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBjZW50ZXIgb2YgdGhlIGNpcmNsZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgY2VudGVyIG9mIHRoZSBjaXJjbGUuCi0gICAgICogQHBhcmFtIHJhZGl1cwotICAgICAqICAgICAgICAgICAgdGhlIHJhZGl1cyBvZiB0aGUgY2lyY2xlLgotICAgICAqIEBwYXJhbSBzdGFydAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IGFuZ2xlIG9mIHRoZSBhcmMgaW4gZGVncmVlcy4KLSAgICAgKiBAcGFyYW0gZXh0ZW50Ci0gICAgICogICAgICAgICAgICB0aGUgYW5nbGUgd2lkdGggb2YgdGhlIGFyYyBpbiBkZWdyZWVzLgotICAgICAqIEBwYXJhbSB0eXBlCi0gICAgICogICAgICAgICAgICB0aGUgY2xvc3VyZSB0eXBlLCBlaXRoZXIge0BsaW5rIEFyYzJEI09QRU59LAotICAgICAqICAgICAgICAgICAge0BsaW5rIEFyYzJEI0NIT1JEfSwgb3Ige0BsaW5rIEFyYzJEI1BJRX0uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0QXJjQnlDZW50ZXIoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgcmFkaXVzLCBkb3VibGUgc3RhcnQsIGRvdWJsZSBleHRlbnQsCi0gICAgICAgICAgICBpbnQgdHlwZSkgewotICAgICAgICBzZXRBcmMoeCAtIHJhZGl1cywgeSAtIHJhZGl1cywgcmFkaXVzICogMi4wLCByYWRpdXMgKiAyLjAsIHN0YXJ0LCBleHRlbnQsIHR5cGUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGFyYyBkYXRhIGZvciBhIGNpcmN1bGFyIGFyYyBiYXNlZCBvbiB0d28gdGFuZ2VudCBsaW5lcyBhbmQgdGhlCi0gICAgICogcmFkaXVzLiBUaGUgdHdvIHRhbmdlbnQgbGluZXMgYXJlIHRoZSBsaW5lcyBmcm9tIHAxIHRvIHAyIGFuZCBmcm9tIHAyIHRvCi0gICAgICogcDMsIHdoaWNoIGRldGVybWluZSBhIHVuaXF1ZSBjaXJjbGUgd2l0aCB0aGUgZ2l2ZW4gcmFkaXVzLiBUaGUgc3RhcnQgYW5kCi0gICAgICogZW5kIHBvaW50cyBvZiB0aGUgYXJjIGFyZSB0aGUgcG9pbnRzIHdoZXJlIHRoZSBjaXJjbGUgdG91Y2hlcyB0aGUgdHdvCi0gICAgICogbGluZXMsIGFuZCB0aGUgYXJjIGl0c2VsZiBpcyB0aGUgc2hvcnRlciBvZiB0aGUgdHdvIGNpcmNsZSBzZWdtZW50cwotICAgICAqIGRldGVybWluZWQgYnkgdGhlIHR3byBwb2ludHMgKGluIG90aGVyIHdvcmRzLCBpdCBpcyB0aGUgcGllY2Ugb2YgdGhlCi0gICAgICogY2lyY2xlIHRoYXQgaXMgY2xvc2VyIHRvIHRoZSBsaW5lcycgaW50ZXJzZWN0aW9uIHBvaW50IHAyIGFuZCBmb3JtcyBhCi0gICAgICogY29uY2F2ZSBzaGFwZSB3aXRoIHRoZSBzZWdtZW50cyBmcm9tIHAxIHRvIHAyIGFuZCBmcm9tIHAyIHRvIHAzKS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcDEKLSAgICAgKiAgICAgICAgICAgIGEgcG9pbnQgd2hpY2ggZGV0ZXJtaW5lcyBvbmUgb2YgdGhlIHR3byB0YW5nZW50IGxpbmVzICh3aXRoCi0gICAgICogICAgICAgICAgICBwMikuCi0gICAgICogQHBhcmFtIHAyCi0gICAgICogICAgICAgICAgICB0aGUgcG9pbnQgb2YgaW50ZXJzZWN0aW9uIG9mIHRoZSB0d28gdGFuZ2VudCBsaW5lcy4KLSAgICAgKiBAcGFyYW0gcDMKLSAgICAgKiAgICAgICAgICAgIGEgcG9pbnQgd2hpY2ggZGV0ZXJtaW5lcyBvbmUgb2YgdGhlIHR3byB0YW5nZW50IGxpbmVzICh3aXRoCi0gICAgICogICAgICAgICAgICBwMikuCi0gICAgICogQHBhcmFtIHJhZGl1cwotICAgICAqICAgICAgICAgICAgdGhlIHJhZGl1cyBvZiB0aGUgY2lyY3VsYXIgYXJjLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEFyY0J5VGFuZ2VudChQb2ludDJEIHAxLCBQb2ludDJEIHAyLCBQb2ludDJEIHAzLCBkb3VibGUgcmFkaXVzKSB7Ci0gICAgICAgIC8vIFVzZWQgc2ltcGxlIGdlb21ldHJpYyBjYWxjdWxhdGlvbnMgb2YgYXJjIGNlbnRlciwgcmFkaXVzIGFuZCBhbmdsZXMKLSAgICAgICAgLy8gYnkgdGFuZ2VudHMKLSAgICAgICAgZG91YmxlIGExID0gLU1hdGguYXRhbjIocDEuZ2V0WSgpIC0gcDIuZ2V0WSgpLCBwMS5nZXRYKCkgLSBwMi5nZXRYKCkpOwotICAgICAgICBkb3VibGUgYTIgPSAtTWF0aC5hdGFuMihwMy5nZXRZKCkgLSBwMi5nZXRZKCksIHAzLmdldFgoKSAtIHAyLmdldFgoKSk7Ci0gICAgICAgIGRvdWJsZSBhbSA9IChhMSArIGEyKSAvIDIuMDsKLSAgICAgICAgZG91YmxlIGFoID0gYTEgLSBhbTsKLSAgICAgICAgZG91YmxlIGQgPSByYWRpdXMgLyBNYXRoLmFicyhNYXRoLnNpbihhaCkpOwotICAgICAgICBkb3VibGUgeCA9IHAyLmdldFgoKSArIGQgKiBNYXRoLmNvcyhhbSk7Ci0gICAgICAgIGRvdWJsZSB5ID0gcDIuZ2V0WSgpIC0gZCAqIE1hdGguc2luKGFtKTsKLSAgICAgICAgYWggPSBhaCA+PSAwLjAgPyBNYXRoLlBJICogMS41IC0gYWggOiBNYXRoLlBJICogMC41IC0gYWg7Ci0gICAgICAgIGExID0gZ2V0Tm9ybUFuZ2xlKE1hdGgudG9EZWdyZWVzKGFtIC0gYWgpKTsKLSAgICAgICAgYTIgPSBnZXROb3JtQW5nbGUoTWF0aC50b0RlZ3JlZXMoYW0gKyBhaCkpOwotICAgICAgICBkb3VibGUgZGVsdGEgPSBhMiAtIGExOwotICAgICAgICBpZiAoZGVsdGEgPD0gMC4wKSB7Ci0gICAgICAgICAgICBkZWx0YSArPSAzNjAuMDsKLSAgICAgICAgfQotICAgICAgICBzZXRBcmNCeUNlbnRlcih4LCB5LCByYWRpdXMsIGExLCBkZWx0YSwgdHlwZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhIG5ldyBzdGFydCBhbmdsZSB0byBiZSB0aGUgYW5nbGUgZ2l2ZW4gYnkgdGhlIHRoZSB2ZWN0b3IgZnJvbSB0aGUKLSAgICAgKiBjdXJyZW50IGNlbnRlciBwb2ludCB0byB0aGUgc3BlY2lmaWVkIHBvaW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwb2ludAotICAgICAqICAgICAgICAgICAgdGhlIHBvaW50IHRoYXQgZGV0ZXJtaW5lcyB0aGUgbmV3IHN0YXJ0IGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEFuZ2xlU3RhcnQoUG9pbnQyRCBwb2ludCkgewotICAgICAgICBkb3VibGUgYW5nbGUgPSBNYXRoLmF0YW4yKHBvaW50LmdldFkoKSAtIGdldENlbnRlclkoKSwgcG9pbnQuZ2V0WCgpIC0gZ2V0Q2VudGVyWCgpKTsKLSAgICAgICAgc2V0QW5nbGVTdGFydChnZXROb3JtQW5nbGUoLU1hdGgudG9EZWdyZWVzKGFuZ2xlKSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGFuZ2xlcyBpbiB0ZXJtcyBvZiB2ZWN0b3JzIGZyb20gdGhlIGN1cnJlbnQgYXJjIGNlbnRlciB0byB0aGUKLSAgICAgKiBwb2ludHMgKHgxLCB5MSkgYW5kICh4MiwgeTIpLiBUaGUgc3RhcnQgYW5nbGUgaXMgZ2l2ZW4gYnkgdGhlIHZlY3RvciBmcm9tCi0gICAgICogdGhlIGN1cnJlbnQgY2VudGVyIHRvIHRoZSBwb2ludCAoeDEsIHkxKSBhbmQgdGhlIGVuZCBhbmdsZSBpcyBnaXZlbiBieQotICAgICAqIHRoZSB2ZWN0b3IgZnJvbSB0aGUgY2VudGVyIHRvIHRoZSBwb2ludCAoeDIsIHkyKS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHBvaW50IHdob3NlIHZlY3RvciBmcm9tIHRoZSBjZW50ZXIKLSAgICAgKiAgICAgICAgICAgIHBvaW50IGRldGVybWluZXMgdGhlIG5ldyBzdGFydCBhbmdsZSBvZiB0aGUgYXJjLgotICAgICAqIEBwYXJhbSB5MQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgcG9pbnQgd2hvc2UgdmVjdG9yIGZyb20gdGhlIGNlbnRlcgotICAgICAqICAgICAgICAgICAgcG9pbnQgZGV0ZXJtaW5lcyB0aGUgbmV3IHN0YXJ0IGFuZ2xlIG9mIHRoZSBhcmMuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBwb2ludCB3aG9zZSB2ZWN0b3IgZnJvbSB0aGUgY2VudGVyCi0gICAgICogICAgICAgICAgICBwb2ludCBkZXRlcm1pbmVzIHRoZSBuZXcgZW5kIGFuZ2xlIG9mIHRoZSBhcmMuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBwb2ludCB3aG9zZSB2ZWN0b3IgZnJvbSB0aGUgY2VudGVyCi0gICAgICogICAgICAgICAgICBwb2ludCBkZXRlcm1pbmVzIHRoZSBuZXcgZW5kIGFuZ2xlIG9mIHRoZSBhcmMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0QW5nbGVzKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MikgewotICAgICAgICBkb3VibGUgY3ggPSBnZXRDZW50ZXJYKCk7Ci0gICAgICAgIGRvdWJsZSBjeSA9IGdldENlbnRlclkoKTsKLSAgICAgICAgZG91YmxlIGExID0gZ2V0Tm9ybUFuZ2xlKC1NYXRoLnRvRGVncmVlcyhNYXRoLmF0YW4yKHkxIC0gY3ksIHgxIC0gY3gpKSk7Ci0gICAgICAgIGRvdWJsZSBhMiA9IGdldE5vcm1BbmdsZSgtTWF0aC50b0RlZ3JlZXMoTWF0aC5hdGFuMih5MiAtIGN5LCB4MiAtIGN4KSkpOwotICAgICAgICBhMiAtPSBhMTsKLSAgICAgICAgaWYgKGEyIDw9IDAuMCkgewotICAgICAgICAgICAgYTIgKz0gMzYwLjA7Ci0gICAgICAgIH0KLSAgICAgICAgc2V0QW5nbGVTdGFydChhMSk7Ci0gICAgICAgIHNldEFuZ2xlRXh0ZW50KGEyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBhbmdsZXMgaW4gdGVybXMgb2YgdmVjdG9ycyBmcm9tIHRoZSBjdXJyZW50IGFyYyBjZW50ZXIgdG8gdGhlCi0gICAgICogcG9pbnRzIHAxIGFuZCBwMi4gVGhlIHN0YXJ0IGFuZ2xlIGlzIGdpdmVuIGJ5IHRoZSB2ZWN0b3IgZnJvbSB0aGUgY3VycmVudAotICAgICAqIGNlbnRlciB0byB0aGUgcG9pbnQgcDEgYW5kIHRoZSBlbmQgYW5nbGUgaXMgZ2l2ZW4gYnkgdGhlIHZlY3RvciBmcm9tIHRoZQotICAgICAqIGNlbnRlciB0byB0aGUgcG9pbnQgcDIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHAxCi0gICAgICogICAgICAgICAgICB0aGUgcG9pbnQgd2hvc2UgdmVjdG9yIGZyb20gdGhlIGNlbnRlciBwb2ludCBkZXRlcm1pbmVzIHRoZQotICAgICAqICAgICAgICAgICAgbmV3IHN0YXJ0IGFuZ2xlIG9mIHRoZSBhcmMuCi0gICAgICogQHBhcmFtIHAyCi0gICAgICogICAgICAgICAgICB0aGUgcG9pbnQgd2hvc2UgdmVjdG9yIGZyb20gdGhlIGNlbnRlciBwb2ludCBkZXRlcm1pbmVzIHRoZQotICAgICAqICAgICAgICAgICAgbmV3IGVuZCBhbmdsZSBvZiB0aGUgYXJjLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEFuZ2xlcyhQb2ludDJEIHAxLCBQb2ludDJEIHAyKSB7Ci0gICAgICAgIHNldEFuZ2xlcyhwMS5nZXRYKCksIHAxLmdldFkoKSwgcDIuZ2V0WCgpLCBwMi5nZXRZKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIE5vcm1hbGl6ZXMgdGhlIGFuZ2xlIGJ5IHJlbW92aW5nIGV4dHJhIHdpbmRpbmcgKHBhc3QgMzYwIGRlZ3JlZXMpIGFuZAotICAgICAqIHBsYWNpbmcgaXQgaW4gdGhlIHBvc2l0aXZlIGRlZ3JlZSByYW5nZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYW5nbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgYW5nbGUgaW4gZGVncmVlcy4KLSAgICAgKiBAcmV0dXJuIGFuIGFuZ2xlIGJldHdlZW4gMCBhbmQgMzYwIGRlZ3JlZXMgd2hpY2ggY29ycmVzcG9uZHMgdG8gdGhlIHNhbWUKLSAgICAgKiAgICAgICAgIGRpcmVjdGlvbiB2ZWN0b3IgYXMgdGhlIHNvdXJjZSBhbmdsZS4KLSAgICAgKi8KLSAgICBkb3VibGUgZ2V0Tm9ybUFuZ2xlKGRvdWJsZSBhbmdsZSkgewotICAgICAgICBkb3VibGUgbiA9IE1hdGguZmxvb3IoYW5nbGUgLyAzNjAuMCk7Ci0gICAgICAgIHJldHVybiBhbmdsZSAtIG4gKiAzNjAuMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXRlcm1pbmVzIHdoZXRoZXIgdGhlIGdpdmVuIGFuZ2xlIGlzIGNvbnRhaW5lZCBpbiB0aGUgc3BhbiBvZiB0aGUgYXJjLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhbmdsZQotICAgICAqICAgICAgICAgICAgdGhlIGFuZ2xlIHRvIHRlc3QgaW4gZGVncmVlcy4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBnaXZlbiBhbmdsZSBpcyBiZXR3ZWVuIHRoZSBzdGFydCBhbmdsZSBhbmQgdGhlIGVuZAotICAgICAqICAgICAgICAgYW5nbGUgb2YgdGhlIGFyYy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWluc0FuZ2xlKGRvdWJsZSBhbmdsZSkgewotICAgICAgICBkb3VibGUgZXh0ZW50ID0gZ2V0QW5nbGVFeHRlbnQoKTsKLSAgICAgICAgaWYgKGV4dGVudCA+PSAzNjAuMCkgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgYW5nbGUgPSBnZXROb3JtQW5nbGUoYW5nbGUpOwotICAgICAgICBkb3VibGUgYTEgPSBnZXROb3JtQW5nbGUoZ2V0QW5nbGVTdGFydCgpKTsKLSAgICAgICAgZG91YmxlIGEyID0gYTEgKyBleHRlbnQ7Ci0gICAgICAgIGlmIChhMiA+IDM2MC4wKSB7Ci0gICAgICAgICAgICByZXR1cm4gYW5nbGUgPj0gYTEgfHwgYW5nbGUgPD0gYTIgLSAzNjAuMDsKLSAgICAgICAgfQotICAgICAgICBpZiAoYTIgPCAwLjApIHsKLSAgICAgICAgICAgIHJldHVybiBhbmdsZSA+PSBhMiArIDM2MC4wIHx8IGFuZ2xlIDw9IGExOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBleHRlbnQgPiAwLjAgPyBhMSA8PSBhbmdsZSAmJiBhbmdsZSA8PSBhMiA6IGEyIDw9IGFuZ2xlICYmIGFuZ2xlIDw9IGExOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSBweCwgZG91YmxlIHB5KSB7Ci0gICAgICAgIC8vIE5vcm1hbGl6ZSBwb2ludAotICAgICAgICBkb3VibGUgbnggPSAocHggLSBnZXRYKCkpIC8gZ2V0V2lkdGgoKSAtIDAuNTsKLSAgICAgICAgZG91YmxlIG55ID0gKHB5IC0gZ2V0WSgpKSAvIGdldEhlaWdodCgpIC0gMC41OwotCi0gICAgICAgIGlmICgobnggKiBueCArIG55ICogbnkpID4gMC4yNSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIGV4dGVudCA9IGdldEFuZ2xlRXh0ZW50KCk7Ci0gICAgICAgIGRvdWJsZSBhYnNFeHRlbnQgPSBNYXRoLmFicyhleHRlbnQpOwotICAgICAgICBpZiAoYWJzRXh0ZW50ID49IDM2MC4wKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGJvb2xlYW4gY29udGFpbnNBbmdsZSA9IGNvbnRhaW5zQW5nbGUoTWF0aC50b0RlZ3JlZXMoLU1hdGguYXRhbjIobnksIG54KSkpOwotICAgICAgICBpZiAodHlwZSA9PSBQSUUpIHsKLSAgICAgICAgICAgIHJldHVybiBjb250YWluc0FuZ2xlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChhYnNFeHRlbnQgPD0gMTgwLjAgJiYgIWNvbnRhaW5zQW5nbGUpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIExpbmUyRCBsID0gbmV3IExpbmUyRC5Eb3VibGUoZ2V0U3RhcnRQb2ludCgpLCBnZXRFbmRQb2ludCgpKTsKLSAgICAgICAgaW50IGNjdzEgPSBsLnJlbGF0aXZlQ0NXKHB4LCBweSk7Ci0gICAgICAgIGludCBjY3cyID0gbC5yZWxhdGl2ZUNDVyhnZXRDZW50ZXJYKCksIGdldENlbnRlclkoKSk7Ci0gICAgICAgIHJldHVybiBjY3cxID09IDAgfHwgY2N3MiA9PSAwIHx8ICgoY2N3MSArIGNjdzIpID09IDAgXiBhYnNFeHRlbnQgPiAxODAuMCk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoZG91YmxlIHJ4LCBkb3VibGUgcnksIGRvdWJsZSBydywgZG91YmxlIHJoKSB7Ci0KLSAgICAgICAgaWYgKCEoY29udGFpbnMocngsIHJ5KSAmJiBjb250YWlucyhyeCArIHJ3LCByeSkgJiYgY29udGFpbnMocnggKyBydywgcnkgKyByaCkgJiYgY29udGFpbnMoCi0gICAgICAgICAgICAgICAgcngsIHJ5ICsgcmgpKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIGFic0V4dGVudCA9IE1hdGguYWJzKGdldEFuZ2xlRXh0ZW50KCkpOwotICAgICAgICBpZiAodHlwZSAhPSBQSUUgfHwgYWJzRXh0ZW50IDw9IDE4MC4wIHx8IGFic0V4dGVudCA+PSAzNjAuMCkgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICBSZWN0YW5nbGUyRCByID0gbmV3IFJlY3RhbmdsZTJELkRvdWJsZShyeCwgcnksIHJ3LCByaCk7Ci0KLSAgICAgICAgZG91YmxlIGN4ID0gZ2V0Q2VudGVyWCgpOwotICAgICAgICBkb3VibGUgY3kgPSBnZXRDZW50ZXJZKCk7Ci0gICAgICAgIGlmIChyLmNvbnRhaW5zKGN4LCBjeSkpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIFBvaW50MkQgcDEgPSBnZXRTdGFydFBvaW50KCk7Ci0gICAgICAgIFBvaW50MkQgcDIgPSBnZXRFbmRQb2ludCgpOwotCi0gICAgICAgIHJldHVybiAhci5pbnRlcnNlY3RzTGluZShjeCwgY3ksIHAxLmdldFgoKSwgcDEuZ2V0WSgpKQotICAgICAgICAgICAgICAgICYmICFyLmludGVyc2VjdHNMaW5lKGN4LCBjeSwgcDIuZ2V0WCgpLCBwMi5nZXRZKCkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFJlY3RhbmdsZTJEIHJlY3QpIHsKLSAgICAgICAgcmV0dXJuIGNvbnRhaW5zKHJlY3QuZ2V0WCgpLCByZWN0LmdldFkoKSwgcmVjdC5nZXRXaWR0aCgpLCByZWN0LmdldEhlaWdodCgpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpbnRlcnNlY3RzKGRvdWJsZSByeCwgZG91YmxlIHJ5LCBkb3VibGUgcncsIGRvdWJsZSByaCkgewotCi0gICAgICAgIGlmIChpc0VtcHR5KCkgfHwgcncgPD0gMC4wIHx8IHJoIDw9IDAuMCkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gQ2hlY2s6IERvZXMgYXJjIGNvbnRhaW4gcmVjdGFuZ2xlJ3MgcG9pbnRzCi0gICAgICAgIGlmIChjb250YWlucyhyeCwgcnkpIHx8IGNvbnRhaW5zKHJ4ICsgcncsIHJ5KSB8fCBjb250YWlucyhyeCwgcnkgKyByaCkKLSAgICAgICAgICAgICAgICB8fCBjb250YWlucyhyeCArIHJ3LCByeSArIHJoKSkgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgY3ggPSBnZXRDZW50ZXJYKCk7Ci0gICAgICAgIGRvdWJsZSBjeSA9IGdldENlbnRlclkoKTsKLSAgICAgICAgUG9pbnQyRCBwMSA9IGdldFN0YXJ0UG9pbnQoKTsKLSAgICAgICAgUG9pbnQyRCBwMiA9IGdldEVuZFBvaW50KCk7Ci0gICAgICAgIFJlY3RhbmdsZTJEIHIgPSBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKHJ4LCByeSwgcncsIHJoKTsKLQotICAgICAgICAvLyBDaGVjazogRG9lcyByZWN0YW5nbGUgY29udGFpbiBhcmMncyBwb2ludHMKLSAgICAgICAgaWYgKHIuY29udGFpbnMocDEpIHx8IHIuY29udGFpbnMocDIpIHx8ICh0eXBlID09IFBJRSAmJiByLmNvbnRhaW5zKGN4LCBjeSkpKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICh0eXBlID09IFBJRSkgewotICAgICAgICAgICAgaWYgKHIuaW50ZXJzZWN0c0xpbmUocDEuZ2V0WCgpLCBwMS5nZXRZKCksIGN4LCBjeSkKLSAgICAgICAgICAgICAgICAgICAgfHwgci5pbnRlcnNlY3RzTGluZShwMi5nZXRYKCksIHAyLmdldFkoKSwgY3gsIGN5KSkgewotICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaWYgKHIuaW50ZXJzZWN0c0xpbmUocDEuZ2V0WCgpLCBwMS5nZXRZKCksIHAyLmdldFgoKSwgcDIuZ2V0WSgpKSkgewotICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLy8gTmVhcmVzdCByZWN0YW5nbGUgcG9pbnQKLSAgICAgICAgZG91YmxlIG54ID0gY3ggPCByeCA/IHJ4IDogKGN4ID4gcnggKyBydyA/IHJ4ICsgcncgOiBjeCk7Ci0gICAgICAgIGRvdWJsZSBueSA9IGN5IDwgcnkgPyByeSA6IChjeSA+IHJ5ICsgcmggPyByeSArIHJoIDogY3kpOwotICAgICAgICByZXR1cm4gY29udGFpbnMobngsIG55KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gYXQpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJdGVyYXRvcih0aGlzLCBhdCk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZ2VvbS9BcmVhLmphdmEgYi9hd3QvamF2YS9hd3QvZ2VvbS9BcmVhLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGU2NjE5ZTMuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2dlb20vQXJlYS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzMwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0Lmdlb207Ci0KLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci1pbXBvcnQgamF2YS5hd3QuU2hhcGU7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5QYXRoSXRlcmF0b3I7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLWltcG9ydCBqYXZhLnV0aWwuTm9TdWNoRWxlbWVudEV4Y2VwdGlvbjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuTm90SW1wbGVtZW50ZWRFeGNlcHRpb247Ci0KLS8qKgotICogVGhlIENsYXNzIEFyZWEgcHJvdmlkZXMgYSBtaW5pbWFsIGltcGxlbWVudGF0aW9uIGZvciBhIGdlbmVyaWMgc2hhcGUuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgQXJlYSBpbXBsZW1lbnRzIFNoYXBlLCBDbG9uZWFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIHNvdXJjZSBTaGFwZSBvYmplY3QuCi0gICAgICovCi0gICAgU2hhcGUgczsKLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBOdWxsSXRlcmF0b3IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgY2xhc3MgTnVsbEl0ZXJhdG9yIGltcGxlbWVudHMgUGF0aEl0ZXJhdG9yIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IG51bGwgaXRlcmF0b3IuCi0gICAgICAgICAqLwotICAgICAgICBOdWxsSXRlcmF0b3IoKSB7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGdldFdpbmRpbmdSdWxlKCkgewotICAgICAgICAgICAgcmV0dXJuIFdJTkRfTk9OX1pFUk87Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBpc0RvbmUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyB2b2lkIG5leHQoKSB7Ci0gICAgICAgICAgICAvLyBub3RoaW5nCi0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGRvdWJsZVtdIGNvb3JkcykgewotICAgICAgICAgICAgLy8gYXd0LjRCPUl0ZXJhdG9yIG91dCBvZiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBOb1N1Y2hFbGVtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGZsb2F0W10gY29vcmRzKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNEI9SXRlcmF0b3Igb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IE5vU3VjaEVsZW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNEIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGFyZWEgd2l0aCBubyBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBBcmVhKCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBhcmVhIHdpdGggZGF0YSBnaXZlbiBieSB0aGUgc3BlY2lmaWVkIHNoYXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgc2hhcGUgdGhhdCBnaXZlcyB0aGUgZGF0YSBmb3IgdGhpcyBBcmVhLgotICAgICAqLwotICAgIHB1YmxpYyBBcmVhKFNoYXBlIHMpIHsKLSAgICAgICAgaWYgKHMgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCk7Ci0gICAgICAgIH0KLSAgICAgICAgdGhpcy5zID0gczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhkb3VibGUgeCwgZG91YmxlIHkpIHsKLSAgICAgICAgcmV0dXJuIHMgPT0gbnVsbCA/IGZhbHNlIDogcy5jb250YWlucyh4LCB5KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3aWR0aCwgZG91YmxlIGhlaWdodCkgewotICAgICAgICByZXR1cm4gcyA9PSBudWxsID8gZmFsc2UgOiBzLmNvbnRhaW5zKHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFBvaW50MkQgcCkgewotICAgICAgICBpZiAocCA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcyA9PSBudWxsID8gZmFsc2UgOiBzLmNvbnRhaW5zKHApOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgaWYgKHIgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHMgPT0gbnVsbCA/IGZhbHNlIDogcy5jb250YWlucyhyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUZXN0cyB3aGV0aGVyIHRoZSBvYmplY3QgaXMgZXF1YWwgdG8gdGhpcyBBcmVhLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBvYmplY3QgdG8gY29tcGFyZS4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN1Y2Nlc3NmdWwuCi0gICAgICogQHRocm93cyBOb3RJbXBsZW1lbnRlZEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoaXMgbWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoQXJlYSBvYmopIHRocm93cyBvcmcuYXBhY2hlLmhhcm1vbnkubHVuaS51dGlsLk5vdEltcGxlbWVudGVkRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0cyhkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3aWR0aCwgZG91YmxlIGhlaWdodCkgewotICAgICAgICByZXR1cm4gcyA9PSBudWxsID8gZmFsc2UgOiBzLmludGVyc2VjdHMoeCwgeSwgd2lkdGgsIGhlaWdodCk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0cyhSZWN0YW5nbGUyRCByKSB7Ci0gICAgICAgIGlmIChyID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbigpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBzID09IG51bGwgPyBmYWxzZSA6IHMuaW50ZXJzZWN0cyhyKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldEJvdW5kcygpIHsKLSAgICAgICAgcmV0dXJuIHMgPT0gbnVsbCA/IG5ldyBSZWN0YW5nbGUoKSA6IHMuZ2V0Qm91bmRzKCk7Ci0gICAgfQotCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKCkgewotICAgICAgICByZXR1cm4gcyA9PSBudWxsID8gbmV3IFJlY3RhbmdsZTJELkRvdWJsZSgpIDogcy5nZXRCb3VuZHMyRCgpOwotICAgIH0KLQotICAgIHB1YmxpYyBQYXRoSXRlcmF0b3IgZ2V0UGF0aEl0ZXJhdG9yKEFmZmluZVRyYW5zZm9ybSB0KSB7Ci0gICAgICAgIHJldHVybiBzID09IG51bGwgPyBuZXcgTnVsbEl0ZXJhdG9yKCkgOiBzLmdldFBhdGhJdGVyYXRvcih0KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gdCwgZG91YmxlIGZsYXRuZXNzKSB7Ci0gICAgICAgIHJldHVybiBzID09IG51bGwgPyBuZXcgTnVsbEl0ZXJhdG9yKCkgOiBzLmdldFBhdGhJdGVyYXRvcih0LCBmbGF0bmVzcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgc3BlY2lmaWVkIGFyZWEgdG8gdGhpcyBhcmVhLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhcmVhCi0gICAgICogICAgICAgICAgICB0aGUgYXJlYSB0byBhZGQgdG8gdGhpcyBhcmVhLgotICAgICAqIEB0aHJvd3MgTm90SW1wbGVtZW50ZWRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGlzIG1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkKEFyZWEgYXJlYSkgdGhyb3dzIG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuTm90SW1wbGVtZW50ZWRFeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQZXJmb3JtcyBhbiBleGNsdXNpdmUgb3Igb3BlcmF0aW9uIGJldHdlZW4gdGhpcyBzaGFwZSBhbmQgdGhlIHNwZWNpZmllZAotICAgICAqIHNoYXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhcmVhCi0gICAgICogICAgICAgICAgICB0aGUgYXJlYSB0byBYT1IgYWdhaW5zdCB0aGlzIGFyZWEuCi0gICAgICogQHRocm93cyBOb3RJbXBsZW1lbnRlZEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoaXMgbWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBleGNsdXNpdmVPcihBcmVhIGFyZWEpIHRocm93cyBvcmcuYXBhY2hlLmhhcm1vbnkubHVuaS51dGlsLk5vdEltcGxlbWVudGVkRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogRXh0cmFjdHMgYSBSZWN0YW5nbGUyRCBmcm9tIHRoZSBzb3VyY2Ugc2hhcGUgaWYgdGhlIHVuZGVybHlpbmcgc2hhcGUgZGF0YQotICAgICAqIGRlc2NyaWJlcyBhIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgUmVjdGFuZ2xlMkQgb2JqZWN0IGlmIHRoZSBzb3VyY2Ugc2hhcGUgaXMgcmVjdGFuZ2xlLCBvciBudWxsIGlmCi0gICAgICogICAgICAgICBzaGFwZSBpcyBlbXB0eSBvciBub3QgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIFJlY3RhbmdsZTJEIGV4dHJhY3RSZWN0YW5nbGUoKSB7Ci0gICAgICAgIGlmIChzID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgICAgIGZsb2F0W10gcG9pbnRzID0gbmV3IGZsb2F0WzEyXTsKLSAgICAgICAgaW50IGNvdW50ID0gMDsKLSAgICAgICAgUGF0aEl0ZXJhdG9yIHAgPSBzLmdldFBhdGhJdGVyYXRvcihudWxsKTsKLSAgICAgICAgZmxvYXRbXSBjb29yZHMgPSBuZXcgZmxvYXRbNl07Ci0gICAgICAgIHdoaWxlICghcC5pc0RvbmUoKSkgewotICAgICAgICAgICAgaW50IHR5cGUgPSBwLmN1cnJlbnRTZWdtZW50KGNvb3Jkcyk7Ci0gICAgICAgICAgICBpZiAoY291bnQgPiAxMiB8fCB0eXBlID09IFBhdGhJdGVyYXRvci5TRUdfUVVBRFRPIHx8IHR5cGUgPT0gUGF0aEl0ZXJhdG9yLlNFR19DVUJJQ1RPKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwb2ludHNbY291bnQrK10gPSBjb29yZHNbMF07Ci0gICAgICAgICAgICBwb2ludHNbY291bnQrK10gPSBjb29yZHNbMV07Ci0gICAgICAgICAgICBwLm5leHQoKTsKLSAgICAgICAgfQotICAgICAgICBpZiAocG9pbnRzWzBdID09IHBvaW50c1s2XSAmJiBwb2ludHNbNl0gPT0gcG9pbnRzWzhdICYmIHBvaW50c1syXSA9PSBwb2ludHNbNF0KLSAgICAgICAgICAgICAgICAmJiBwb2ludHNbMV0gPT0gcG9pbnRzWzNdICYmIHBvaW50c1szXSA9PSBwb2ludHNbOV0gJiYgcG9pbnRzWzVdID09IHBvaW50c1s3XSkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBSZWN0YW5nbGUyRC5GbG9hdChwb2ludHNbMF0sIHBvaW50c1sxXSwgcG9pbnRzWzJdIC0gcG9pbnRzWzBdLCBwb2ludHNbN10KLSAgICAgICAgICAgICAgICAgICAgLSBwb2ludHNbMV0pOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlZHVjZXMgdGhlIHNpemUgb2YgdGhpcyBBcmVhIGJ5IGludGVyc2VjdGluZyBpdCB3aXRoIHRoZSBzcGVjaWZpZWQgQXJlYQotICAgICAqIGlmIHRoZXkgYXJlIGJvdGggcmVjdGFuZ2xlcy4KLSAgICAgKiAKLSAgICAgKiBAc2VlIGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQjaW50ZXJzZWN0KFJlY3RhbmdsZTJELCBSZWN0YW5nbGUyRCwKLSAgICAgKiAgICAgIFJlY3RhbmdsZTJEKQotICAgICAqIEBwYXJhbSBhcmVhCi0gICAgICogICAgICAgICAgICB0aGUgYXJlYS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBpbnRlcnNlY3QoQXJlYSBhcmVhKSB7Ci0gICAgICAgIFJlY3RhbmdsZTJEIHNyYzEgPSBleHRyYWN0UmVjdGFuZ2xlKCk7Ci0gICAgICAgIFJlY3RhbmdsZTJEIHNyYzIgPSBhcmVhLmV4dHJhY3RSZWN0YW5nbGUoKTsKLSAgICAgICAgaWYgKHNyYzEgIT0gbnVsbCAmJiBzcmMyICE9IG51bGwpIHsKLSAgICAgICAgICAgIFJlY3RhbmdsZTJELmludGVyc2VjdChzcmMxLCBzcmMyLCAoUmVjdGFuZ2xlMkQpcyk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTdWJ0cmFjdCB0aGUgc3BlY2lmaWVkIGFyZWEgZnJvbSB0aGlzIGFyZWEuCi0gICAgICogCi0gICAgICogQHBhcmFtIGFyZWEKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcmVhIHRvIHN1YnRyYWN0LgotICAgICAqIEB0aHJvd3MgTm90SW1wbGVtZW50ZWRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGlzIG1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc3VidHJhY3QoQXJlYSBhcmVhKSB0aHJvd3Mgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Ob3RJbXBsZW1lbnRlZEV4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGlzIEFyZWEgaXMgZW1wdHkuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIEFyZWEgaXMgZW1wdHkuCi0gICAgICogQHRocm93cyBOb3RJbXBsZW1lbnRlZEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoaXMgbWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0VtcHR5KCkgdGhyb3dzIG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuTm90SW1wbGVtZW50ZWRFeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyBBcmVhIGlzIHBvbHlnb25hbC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgQXJlYSBpcyBwb2x5Z29uYWwuCi0gICAgICogQHRocm93cyBOb3RJbXBsZW1lbnRlZEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoaXMgbWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1BvbHlnb25hbCgpIHRocm93cyBvcmcuYXBhY2hlLmhhcm1vbnkubHVuaS51dGlsLk5vdEltcGxlbWVudGVkRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoaXMgQXJlYSBpcyByZWN0YW5ndWxhci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgQXJlYSBpcyByZWN0YW5ndWxhci4KLSAgICAgKiBAdGhyb3dzIE5vdEltcGxlbWVudGVkRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhpcyBtZXRob2QgaXMgbm90IGltcGxlbWVudGVkLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzUmVjdGFuZ3VsYXIoKSB0aHJvd3Mgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Ob3RJbXBsZW1lbnRlZEV4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGlzIEFyZWEgaXMgc2luZ3VsYXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIEFyZWEgaXMgc2luZ3VsYXIuCi0gICAgICogQHRocm93cyBOb3RJbXBsZW1lbnRlZEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoaXMgbWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1Npbmd1bGFyKCkgdGhyb3dzIG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuTm90SW1wbGVtZW50ZWRFeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXNldHMgdGhlIGRhdGEgb2YgdGhpcyBBcmVhLgotICAgICAqIAotICAgICAqIEB0aHJvd3MgTm90SW1wbGVtZW50ZWRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGlzIG1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVzZXQoKSB0aHJvd3Mgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Ob3RJbXBsZW1lbnRlZEV4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRyYW5zZm9ybXMgdGhlIGRhdGEgb2YgdGhpcyBBcmVhIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkCi0gICAgICogQWZmaW5lVHJhbnNmb3JtLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0Ci0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNmb3JtIHRvIHVzZSB0byB0cmFuc2Zvcm0gdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgdHJhbnNmb3JtKEFmZmluZVRyYW5zZm9ybSB0KSB7Ci0gICAgICAgIHMgPSB0LmNyZWF0ZVRyYW5zZm9ybWVkU2hhcGUocyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIG5ldyBBcmVhIHRoYXQgaXMgdGhlIHJlc3VsdCBvZiB0cmFuc2Zvcm1pbmcgdGhlIGRhdGEgb2YgdGhpcwotICAgICAqIEFyZWEgYWNjb3JkaW5nIHRvIHRoZSBzcGVjaWZpZWQgQWZmaW5lVHJhbnNmb3JtLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0Ci0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNmb3JtIHRvIHVzZSB0byB0cmFuc2Zvcm0gdGhlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgbmV3IEFyZWEgdGhhdCBpcyB0aGUgcmVzdWx0IG9mIHRyYW5zZm9ybWluZyB0aGUgZGF0YSBvZiB0aGlzCi0gICAgICogICAgICAgICBBcmVhIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkIEFmZmluZVRyYW5zZm9ybS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQXJlYSBjcmVhdGVUcmFuc2Zvcm1lZEFyZWEoQWZmaW5lVHJhbnNmb3JtIHQpIHsKLSAgICAgICAgcmV0dXJuIHMgPT0gbnVsbCA/IG5ldyBBcmVhKCkgOiBuZXcgQXJlYSh0LmNyZWF0ZVRyYW5zZm9ybWVkU2hhcGUocykpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgY2xvbmUoKSB7Ci0gICAgICAgIHJldHVybiBuZXcgQXJlYSh0aGlzKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9nZW9tL0N1YmljQ3VydmUyRC5qYXZhIGIvYXd0L2phdmEvYXd0L2dlb20vQ3ViaWNDdXJ2ZTJELmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDFkZGVkZjMuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2dlb20vQ3ViaWNDdXJ2ZTJELmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMDQ3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0Lmdlb207Ci0KLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci1pbXBvcnQgamF2YS5hd3QuU2hhcGU7Ci1pbXBvcnQgamF2YS51dGlsLk5vU3VjaEVsZW1lbnRFeGNlcHRpb247Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLkNyb3NzaW5nOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBDbGFzcyBDdWJpY0N1cnZlMkQgaXMgYSBTaGFwZSB0aGF0IHJlcHJlc2VudHMgYSBzZWdtZW50IG9mIGEgcXVhZHJhdGljCi0gKiAoQmV6aWVyKSBjdXJ2ZS4gVGhlIGN1cnZlZCBzZWdtZW50IGlzIGRldGVybWluZWQgYnkgZm91ciBwb2ludHM6IGEgc3RhcnQKLSAqIHBvaW50LCBhbiBlbmQgcG9pbnQsIGFuZCB0d28gY29udHJvbCBwb2ludHMuIFRoZSBjb250cm9sIHBvaW50cyBnaXZlCi0gKiBpbmZvcm1hdGlvbiBhYm91dCB0aGUgdGFuZ2VudCBhbmQgbmV4dCBkZXJpdmF0aXZlIGF0IHRoZSBlbmRwb2ludHMgYWNjb3JkaW5nCi0gKiB0byB0aGUgc3RhbmRhcmQgdGhlb3J5IG9mIEJlemllciBjdXJ2ZXMuIEZvciBtb3JlIGluZm9ybWF0aW9uIG9uIEJlemllcgotICogY3VydmVzLCBzZWUgPGEgaHJlZj0iaHR0cDovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9CJUMzJUE5emllcl9jdXJ2ZSI+dGhpcwotICogYXJ0aWNsZTwvYT4uCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgQ3ViaWNDdXJ2ZTJEIGltcGxlbWVudHMgU2hhcGUsIENsb25lYWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgRmxvYXQgaXMgdGhlIHN1YmNsYXNzIG9mIEN1YmljQ3VydmUyRCB0aGF0IGhhcyBhbGwgb2YgaXRzIGRhdGEKLSAgICAgKiB2YWx1ZXMgc3RvcmVkIHdpdGggZmxvYXQtbGV2ZWwgcHJlY2lzaW9uLgotICAgICAqIAotICAgICAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgY2xhc3MgRmxvYXQgZXh0ZW5kcyBDdWJpY0N1cnZlMkQgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCB4MTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgeTE7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgY3RybHgxOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IGN0cmx5MTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgY3RybHgyOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCBjdHJseTI7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCB4MjsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHkyOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZmxvYXQtdmFsdWVkIEN1YmljQ3VydmUyRCB3aXRoIGFsbCBjb29yZGluYXRlCi0gICAgICAgICAqIHZhbHVlcyBzZXQgdG8gemVyby4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBGbG9hdCgpIHsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZmxvYXQtdmFsdWVkIEN1YmljQ3VydmUyRCB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgICAgICogY29vcmRpbmF0ZSB2YWx1ZXMuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geDEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIHkxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSBjdHJseDEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHkxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIGN0cmx4MgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHkyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSB4MgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIHkyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEZsb2F0KGZsb2F0IHgxLCBmbG9hdCB5MSwgZmxvYXQgY3RybHgxLCBmbG9hdCBjdHJseTEsIGZsb2F0IGN0cmx4MiwgZmxvYXQgY3RybHkyLAotICAgICAgICAgICAgICAgIGZsb2F0IHgyLCBmbG9hdCB5MikgewotICAgICAgICAgICAgc2V0Q3VydmUoeDEsIHkxLCBjdHJseDEsIGN0cmx5MSwgY3RybHgyLCBjdHJseTIsIHgyLCB5Mik7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRYMSgpIHsKLSAgICAgICAgICAgIHJldHVybiB4MTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFkxKCkgewotICAgICAgICAgICAgcmV0dXJuIHkxOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0Q3RybFgxKCkgewotICAgICAgICAgICAgcmV0dXJuIGN0cmx4MTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldEN0cmxZMSgpIHsKLSAgICAgICAgICAgIHJldHVybiBjdHJseTE7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRDdHJsWDIoKSB7Ci0gICAgICAgICAgICByZXR1cm4gY3RybHgyOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0Q3RybFkyKCkgewotICAgICAgICAgICAgcmV0dXJuIGN0cmx5MjsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFgyKCkgewotICAgICAgICAgICAgcmV0dXJuIHgyOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WTIoKSB7Ci0gICAgICAgICAgICByZXR1cm4geTI7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFBvaW50MkQgZ2V0UDEoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFBvaW50MkQuRmxvYXQoeDEsIHkxKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUG9pbnQyRCBnZXRDdHJsUDEoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFBvaW50MkQuRmxvYXQoY3RybHgxLCBjdHJseTEpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBQb2ludDJEIGdldEN0cmxQMigpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUG9pbnQyRC5GbG9hdChjdHJseDIsIGN0cmx5Mik7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFBvaW50MkQgZ2V0UDIoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFBvaW50MkQuRmxvYXQoeDIsIHkyKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRDdXJ2ZShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIGN0cmx4MSwgZG91YmxlIGN0cmx5MSwgZG91YmxlIGN0cmx4MiwKLSAgICAgICAgICAgICAgICBkb3VibGUgY3RybHkyLCBkb3VibGUgeDIsIGRvdWJsZSB5MikgewotICAgICAgICAgICAgdGhpcy54MSA9IChmbG9hdCl4MTsKLSAgICAgICAgICAgIHRoaXMueTEgPSAoZmxvYXQpeTE7Ci0gICAgICAgICAgICB0aGlzLmN0cmx4MSA9IChmbG9hdCljdHJseDE7Ci0gICAgICAgICAgICB0aGlzLmN0cmx5MSA9IChmbG9hdCljdHJseTE7Ci0gICAgICAgICAgICB0aGlzLmN0cmx4MiA9IChmbG9hdCljdHJseDI7Ci0gICAgICAgICAgICB0aGlzLmN0cmx5MiA9IChmbG9hdCljdHJseTI7Ci0gICAgICAgICAgICB0aGlzLngyID0gKGZsb2F0KXgyOwotICAgICAgICAgICAgdGhpcy55MiA9IChmbG9hdCl5MjsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBTZXRzIHRoZSBkYXRhIHZhbHVlcyBvZiB0aGUgY3VydmUuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geDEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIHkxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSBjdHJseDEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHkxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIGN0cmx4MgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHkyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSB4MgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIHkyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIHZvaWQgc2V0Q3VydmUoZmxvYXQgeDEsIGZsb2F0IHkxLCBmbG9hdCBjdHJseDEsIGZsb2F0IGN0cmx5MSwgZmxvYXQgY3RybHgyLAotICAgICAgICAgICAgICAgIGZsb2F0IGN0cmx5MiwgZmxvYXQgeDIsIGZsb2F0IHkyKSB7Ci0gICAgICAgICAgICB0aGlzLngxID0geDE7Ci0gICAgICAgICAgICB0aGlzLnkxID0geTE7Ci0gICAgICAgICAgICB0aGlzLmN0cmx4MSA9IGN0cmx4MTsKLSAgICAgICAgICAgIHRoaXMuY3RybHkxID0gY3RybHkxOwotICAgICAgICAgICAgdGhpcy5jdHJseDIgPSBjdHJseDI7Ci0gICAgICAgICAgICB0aGlzLmN0cmx5MiA9IGN0cmx5MjsKLSAgICAgICAgICAgIHRoaXMueDIgPSB4MjsKLSAgICAgICAgICAgIHRoaXMueTIgPSB5MjsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRCgpIHsKLSAgICAgICAgICAgIGZsb2F0IHJ4MSA9IE1hdGgubWluKE1hdGgubWluKHgxLCB4MiksIE1hdGgubWluKGN0cmx4MSwgY3RybHgyKSk7Ci0gICAgICAgICAgICBmbG9hdCByeTEgPSBNYXRoLm1pbihNYXRoLm1pbih5MSwgeTIpLCBNYXRoLm1pbihjdHJseTEsIGN0cmx5MikpOwotICAgICAgICAgICAgZmxvYXQgcngyID0gTWF0aC5tYXgoTWF0aC5tYXgoeDEsIHgyKSwgTWF0aC5tYXgoY3RybHgxLCBjdHJseDIpKTsKLSAgICAgICAgICAgIGZsb2F0IHJ5MiA9IE1hdGgubWF4KE1hdGgubWF4KHkxLCB5MiksIE1hdGgubWF4KGN0cmx5MSwgY3RybHkyKSk7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZTJELkZsb2F0KHJ4MSwgcnkxLCByeDIgLSByeDEsIHJ5MiAtIHJ5MSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgRG91YmxlIGlzIHRoZSBzdWJjbGFzcyBvZiBDdWJpY0N1cnZlMkQgdGhhdCBoYXMgYWxsIG9mIGl0cyBkYXRhCi0gICAgICogdmFsdWVzIHN0b3JlZCB3aXRoIGRvdWJsZS1sZXZlbCBwcmVjaXNpb24uCi0gICAgICogCi0gICAgICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBjbGFzcyBEb3VibGUgZXh0ZW5kcyBDdWJpY0N1cnZlMkQgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgeDE7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB5MTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgY3RybHgxOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSBjdHJseTE7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSBjdHJseDI7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSBjdHJseTI7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgeDI7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgeTI7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkb3VibGUtdmFsdWVkIEN1YmljQ3VydmUyRCB3aXRoIGFsbCBjb29yZGluYXRlCi0gICAgICAgICAqIHZhbHVlcyBzZXQgdG8gemVyby4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBEb3VibGUoKSB7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRvdWJsZS12YWx1ZWQgQ3ViaWNDdXJ2ZTJEIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAgICAgKiBjb29yZGluYXRlIHZhbHVlcy4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSB4MQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0geTEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIGN0cmx4MQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSBjdHJseTEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHgyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSBjdHJseTIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIHgyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0geTIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRG91YmxlKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgY3RybHgxLCBkb3VibGUgY3RybHkxLCBkb3VibGUgY3RybHgyLAotICAgICAgICAgICAgICAgIGRvdWJsZSBjdHJseTIsIGRvdWJsZSB4MiwgZG91YmxlIHkyKSB7Ci0gICAgICAgICAgICBzZXRDdXJ2ZSh4MSwgeTEsIGN0cmx4MSwgY3RybHkxLCBjdHJseDIsIGN0cmx5MiwgeDIsIHkyKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFgxKCkgewotICAgICAgICAgICAgcmV0dXJuIHgxOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WTEoKSB7Ci0gICAgICAgICAgICByZXR1cm4geTE7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRDdHJsWDEoKSB7Ci0gICAgICAgICAgICByZXR1cm4gY3RybHgxOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0Q3RybFkxKCkgewotICAgICAgICAgICAgcmV0dXJuIGN0cmx5MTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldEN0cmxYMigpIHsKLSAgICAgICAgICAgIHJldHVybiBjdHJseDI7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRDdHJsWTIoKSB7Ci0gICAgICAgICAgICByZXR1cm4gY3RybHkyOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WDIoKSB7Ci0gICAgICAgICAgICByZXR1cm4geDI7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRZMigpIHsKLSAgICAgICAgICAgIHJldHVybiB5MjsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUG9pbnQyRCBnZXRQMSgpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUG9pbnQyRC5Eb3VibGUoeDEsIHkxKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUG9pbnQyRCBnZXRDdHJsUDEoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFBvaW50MkQuRG91YmxlKGN0cmx4MSwgY3RybHkxKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUG9pbnQyRCBnZXRDdHJsUDIoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFBvaW50MkQuRG91YmxlKGN0cmx4MiwgY3RybHkyKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUG9pbnQyRCBnZXRQMigpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUG9pbnQyRC5Eb3VibGUoeDIsIHkyKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRDdXJ2ZShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIGN0cmx4MSwgZG91YmxlIGN0cmx5MSwgZG91YmxlIGN0cmx4MiwKLSAgICAgICAgICAgICAgICBkb3VibGUgY3RybHkyLCBkb3VibGUgeDIsIGRvdWJsZSB5MikgewotICAgICAgICAgICAgdGhpcy54MSA9IHgxOwotICAgICAgICAgICAgdGhpcy55MSA9IHkxOwotICAgICAgICAgICAgdGhpcy5jdHJseDEgPSBjdHJseDE7Ci0gICAgICAgICAgICB0aGlzLmN0cmx5MSA9IGN0cmx5MTsKLSAgICAgICAgICAgIHRoaXMuY3RybHgyID0gY3RybHgyOwotICAgICAgICAgICAgdGhpcy5jdHJseTIgPSBjdHJseTI7Ci0gICAgICAgICAgICB0aGlzLngyID0geDI7Ci0gICAgICAgICAgICB0aGlzLnkyID0geTI7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoKSB7Ci0gICAgICAgICAgICBkb3VibGUgcngxID0gTWF0aC5taW4oTWF0aC5taW4oeDEsIHgyKSwgTWF0aC5taW4oY3RybHgxLCBjdHJseDIpKTsKLSAgICAgICAgICAgIGRvdWJsZSByeTEgPSBNYXRoLm1pbihNYXRoLm1pbih5MSwgeTIpLCBNYXRoLm1pbihjdHJseTEsIGN0cmx5MikpOwotICAgICAgICAgICAgZG91YmxlIHJ4MiA9IE1hdGgubWF4KE1hdGgubWF4KHgxLCB4MiksIE1hdGgubWF4KGN0cmx4MSwgY3RybHgyKSk7Ci0gICAgICAgICAgICBkb3VibGUgcnkyID0gTWF0aC5tYXgoTWF0aC5tYXgoeTEsIHkyKSwgTWF0aC5tYXgoY3RybHkxLCBjdHJseTIpKTsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKHJ4MSwgcnkxLCByeDIgLSByeDEsIHJ5MiAtIHJ5MSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKgotICAgICAqIEN1YmljQ3VydmUyRCBwYXRoIGl0ZXJhdG9yCi0gICAgICovCi0gICAgLyoqCi0gICAgICogVGhlIEl0ZXJhdG9yIGNsYXNzIGZvciB0aGUgU2hhcGUgQ3ViaWNDdXJ2ZTJELgotICAgICAqLwotICAgIGNsYXNzIEl0ZXJhdG9yIGltcGxlbWVudHMgUGF0aEl0ZXJhdG9yIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHNvdXJjZSBDdWJpY0N1cnZlMkQgb2JqZWN0LgotICAgICAgICAgKi8KLSAgICAgICAgQ3ViaWNDdXJ2ZTJEIGM7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBwYXRoIGl0ZXJhdG9yIHRyYW5zZm9ybWF0aW9uLgotICAgICAgICAgKi8KLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjdXJyZW50IHNlZ21lbnQgaW5kZXguCi0gICAgICAgICAqLwotICAgICAgICBpbnQgaW5kZXg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENvbnN0cnVjdHMgYSBuZXcgQ3ViaWNDdXJ2ZTJELkl0ZXJhdG9yIGZvciBnaXZlbiBsaW5lIGFuZAotICAgICAgICAgKiB0cmFuc2Zvcm1hdGlvbgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGMKLSAgICAgICAgICogICAgICAgICAgICB0aGUgc291cmNlIEN1YmljQ3VydmUyRCBvYmplY3QuCi0gICAgICAgICAqIEBwYXJhbSB0Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGFmZmluZSB0cmFuc2Zvcm1hdGlvbiBvYmplY3QuCi0gICAgICAgICAqLwotICAgICAgICBJdGVyYXRvcihDdWJpY0N1cnZlMkQgYywgQWZmaW5lVHJhbnNmb3JtIHQpIHsKLSAgICAgICAgICAgIHRoaXMuYyA9IGM7Ci0gICAgICAgICAgICB0aGlzLnQgPSB0OwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGludCBnZXRXaW5kaW5nUnVsZSgpIHsKLSAgICAgICAgICAgIHJldHVybiBXSU5EX05PTl9aRVJPOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGJvb2xlYW4gaXNEb25lKCkgewotICAgICAgICAgICAgcmV0dXJuIGluZGV4ID4gMTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyB2b2lkIG5leHQoKSB7Ci0gICAgICAgICAgICBpbmRleCsrOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGludCBjdXJyZW50U2VnbWVudChkb3VibGVbXSBjb29yZHMpIHsKLSAgICAgICAgICAgIGlmIChpc0RvbmUoKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBOb1N1Y2hFbGVtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbnQgdHlwZTsKLSAgICAgICAgICAgIGludCBjb3VudDsKLSAgICAgICAgICAgIGlmIChpbmRleCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19NT1ZFVE87Ci0gICAgICAgICAgICAgICAgY29vcmRzWzBdID0gYy5nZXRYMSgpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IGMuZ2V0WTEoKTsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDE7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfQ1VCSUNUTzsKLSAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSBjLmdldEN0cmxYMSgpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IGMuZ2V0Q3RybFkxKCk7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzJdID0gYy5nZXRDdHJsWDIoKTsKLSAgICAgICAgICAgICAgICBjb29yZHNbM10gPSBjLmdldEN0cmxZMigpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1s0XSA9IGMuZ2V0WDIoKTsKLSAgICAgICAgICAgICAgICBjb29yZHNbNV0gPSBjLmdldFkyKCk7Ci0gICAgICAgICAgICAgICAgY291bnQgPSAzOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCBjb3VudCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gdHlwZTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgY3VycmVudFNlZ21lbnQoZmxvYXRbXSBjb29yZHMpIHsKLSAgICAgICAgICAgIGlmIChpc0RvbmUoKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBOb1N1Y2hFbGVtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbnQgdHlwZTsKLSAgICAgICAgICAgIGludCBjb3VudDsKLSAgICAgICAgICAgIGlmIChpbmRleCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19NT1ZFVE87Ci0gICAgICAgICAgICAgICAgY29vcmRzWzBdID0gKGZsb2F0KWMuZ2V0WDEoKTsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSAoZmxvYXQpYy5nZXRZMSgpOwotICAgICAgICAgICAgICAgIGNvdW50ID0gMTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19DVUJJQ1RPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IChmbG9hdCljLmdldEN0cmxYMSgpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IChmbG9hdCljLmdldEN0cmxZMSgpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1syXSA9IChmbG9hdCljLmdldEN0cmxYMigpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1szXSA9IChmbG9hdCljLmdldEN0cmxZMigpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1s0XSA9IChmbG9hdCljLmdldFgyKCk7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzVdID0gKGZsb2F0KWMuZ2V0WTIoKTsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAodCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdC50cmFuc2Zvcm0oY29vcmRzLCAwLCBjb29yZHMsIDAsIGNvdW50KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB0eXBlOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgMi1EIGN1YmljIGN1cnZlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBDdWJpY0N1cnZlMkQoKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0WDEoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldFkxKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgUG9pbnQyRCBnZXRQMSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldEN0cmxYMSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldEN0cmxZMSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFBvaW50MkQgZ2V0Q3RybFAxKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldEN0cmxYMigpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50Ci0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZSBnZXRDdHJsWTIoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBQb2ludDJEIGdldEN0cmxQMigpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZSBnZXRYMigpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZSBnZXRZMigpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZW5kIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGVuZCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgUG9pbnQyRCBnZXRQMigpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGF0YSBvZiB0aGUgY3VydmUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAqIEBwYXJhbSBjdHJseDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIGN0cmx5MQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY3RybHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY3RybHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzZXRDdXJ2ZShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIGN0cmx4MSwgZG91YmxlIGN0cmx5MSwKLSAgICAgICAgICAgIGRvdWJsZSBjdHJseDIsIGRvdWJsZSBjdHJseTIsIGRvdWJsZSB4MiwgZG91YmxlIHkyKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgb2YgdGhlIGN1cnZlIGFzIHBvaW50IG9iamVjdHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHAxCi0gICAgICogICAgICAgICAgICB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICogQHBhcmFtIGNwMQotICAgICAqICAgICAgICAgICAgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIGNwMgotICAgICAqICAgICAgICAgICAgdGhlIHNlY29uZCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSBwMgotICAgICAqICAgICAgICAgICAgdGhlIGVuZCBwb2ludC4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW55IG9mIHRoZSBwb2ludHMgaXMgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRDdXJ2ZShQb2ludDJEIHAxLCBQb2ludDJEIGNwMSwgUG9pbnQyRCBjcDIsIFBvaW50MkQgcDIpIHsKLSAgICAgICAgc2V0Q3VydmUocDEuZ2V0WCgpLCBwMS5nZXRZKCksIGNwMS5nZXRYKCksIGNwMS5nZXRZKCksIGNwMi5nZXRYKCksIGNwMi5nZXRZKCksIHAyLmdldFgoKSwKLSAgICAgICAgICAgICAgICBwMi5nZXRZKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgb2YgdGhlIGN1cnZlIGJ5IHJlYWRpbmcgdGhlIGRhdGEgZnJvbSBhbiBhcnJheSBvZiB2YWx1ZXMuCi0gICAgICogVGhlIHZhbHVlcyBhcmUgcmVhZCBpbiB0aGUgc2FtZSBvcmRlciBhcyB0aGUgYXJndW1lbnRzIG9mIHRoZSBtZXRob2QKLSAgICAgKiB7QGxpbmsgQ3ViaWNDdXJ2ZTJEI3NldEN1cnZlKGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSwgZG91YmxlKX0KLSAgICAgKiAuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvb3JkcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHZhbHVlcyBjb250YWluaW5nIHRoZSBuZXcgY29vcmRpbmF0ZXMuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBvZiB0aGUgZGF0YSB0byByZWFkIHdpdGhpbiB0aGUgYXJyYXkuCi0gICAgICogQHRocm93cyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB7QGNvZGUgY29vcmRzLmxlbmd0aH0gPCBvZmZzZXQgKyA4LgotICAgICAqIEB0aHJvd3MgTnVsbFBvaW50ZXJFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgY29vcmRpbmF0ZSBhcnJheSBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEN1cnZlKGRvdWJsZVtdIGNvb3JkcywgaW50IG9mZnNldCkgewotICAgICAgICBzZXRDdXJ2ZShjb29yZHNbb2Zmc2V0ICsgMF0sIGNvb3Jkc1tvZmZzZXQgKyAxXSwgY29vcmRzW29mZnNldCArIDJdLCBjb29yZHNbb2Zmc2V0ICsgM10sCi0gICAgICAgICAgICAgICAgY29vcmRzW29mZnNldCArIDRdLCBjb29yZHNbb2Zmc2V0ICsgNV0sIGNvb3Jkc1tvZmZzZXQgKyA2XSwgY29vcmRzW29mZnNldCArIDddKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBkYXRhIG9mIHRoZSBjdXJ2ZSBieSByZWFkaW5nIHRoZSBkYXRhIGZyb20gYW4gYXJyYXkgb2YgcG9pbnRzLgotICAgICAqIFRoZSB2YWx1ZXMgYXJlIHJlYWQgaW4gdGhlIHNhbWUgb3JkZXIgYXMgdGhlIGFyZ3VtZW50cyBvZiB0aGUgbWV0aG9kCi0gICAgICoge0BsaW5rIEN1YmljQ3VydmUyRCNzZXRDdXJ2ZShQb2ludDJELCBQb2ludDJELCBQb2ludDJELCBQb2ludDJEKX0KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcG9pbnRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgcG9pbnRzIGNvbnRhaW5pbmcgdGhlIG5ldyBjb29yZGluYXRlcy4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IG9mIHRoZSBkYXRhIHRvIHJlYWQgd2l0aGluIHRoZSBhcnJheS4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHtAY29kZSBwb2ludHMubGVuZ3RofSA8IG9mZnNldCArIC4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHBvaW50IGFycmF5IGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Q3VydmUoUG9pbnQyRFtdIHBvaW50cywgaW50IG9mZnNldCkgewotICAgICAgICBzZXRDdXJ2ZShwb2ludHNbb2Zmc2V0ICsgMF0uZ2V0WCgpLCBwb2ludHNbb2Zmc2V0ICsgMF0uZ2V0WSgpLCBwb2ludHNbb2Zmc2V0ICsgMV0uZ2V0WCgpLAotICAgICAgICAgICAgICAgIHBvaW50c1tvZmZzZXQgKyAxXS5nZXRZKCksIHBvaW50c1tvZmZzZXQgKyAyXS5nZXRYKCksIHBvaW50c1tvZmZzZXQgKyAyXS5nZXRZKCksCi0gICAgICAgICAgICAgICAgcG9pbnRzW29mZnNldCArIDNdLmdldFgoKSwgcG9pbnRzW29mZnNldCArIDNdLmdldFkoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGF0YSBvZiB0aGUgY3VydmUgYnkgY29weWluZyBpdCBmcm9tIGFub3RoZXIgQ3ViaWNDdXJ2ZTJELgotICAgICAqIAotICAgICAqIEBwYXJhbSBjdXJ2ZQotICAgICAqICAgICAgICAgICAgdGhlIGN1cnZlIHRvIGNvcHkgdGhlIGRhdGEgcG9pbnRzIGZyb20uCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBjdXJ2ZSBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEN1cnZlKEN1YmljQ3VydmUyRCBjdXJ2ZSkgewotICAgICAgICBzZXRDdXJ2ZShjdXJ2ZS5nZXRYMSgpLCBjdXJ2ZS5nZXRZMSgpLCBjdXJ2ZS5nZXRDdHJsWDEoKSwgY3VydmUuZ2V0Q3RybFkxKCksIGN1cnZlCi0gICAgICAgICAgICAgICAgLmdldEN0cmxYMigpLCBjdXJ2ZS5nZXRDdHJsWTIoKSwgY3VydmUuZ2V0WDIoKSwgY3VydmUuZ2V0WTIoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc3F1YXJlIG9mIHRoZSBmbGF0bmVzcyBvZiB0aGlzIGN1cnZlLCB3aGVyZSB0aGUgZmxhdG5lc3MgaXMgdGhlCi0gICAgICogbWF4aW11bSBkaXN0YW5jZSBmcm9tIHRoZSBjdXJ2ZXMgY29udHJvbCBwb2ludHMgdG8gdGhlIGxpbmUgc2VnbWVudAotICAgICAqIGNvbm5lY3RpbmcgdGhlIHR3byBwb2ludHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc3F1YXJlIG9mIHRoZSBmbGF0bmVzcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlIGdldEZsYXRuZXNzU3EoKSB7Ci0gICAgICAgIHJldHVybiBnZXRGbGF0bmVzc1NxKGdldFgxKCksIGdldFkxKCksIGdldEN0cmxYMSgpLCBnZXRDdHJsWTEoKSwgZ2V0Q3RybFgyKCksIGdldEN0cmxZMigpLAotICAgICAgICAgICAgICAgIGdldFgyKCksIGdldFkyKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNxdWFyZSBvZiB0aGUgZmxhdG5lc3Mgb2YgdGhlIGN1YmljIGN1cnZlIHNlZ21lbnQgZGVmaW5lZCBieSB0aGUKLSAgICAgKiBzcGVjaWZpZWQgdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKiBAcGFyYW0gY3RybHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSBjdHJseTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIGN0cmx4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIGN0cmx5MgotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICogQHJldHVybiB0aGUgc3F1YXJlIG9mIHRoZSBmbGF0bmVzcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGRvdWJsZSBnZXRGbGF0bmVzc1NxKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgY3RybHgxLCBkb3VibGUgY3RybHkxLAotICAgICAgICAgICAgZG91YmxlIGN0cmx4MiwgZG91YmxlIGN0cmx5MiwgZG91YmxlIHgyLCBkb3VibGUgeTIpIHsKLSAgICAgICAgcmV0dXJuIE1hdGgubWF4KExpbmUyRC5wdFNlZ0Rpc3RTcSh4MSwgeTEsIHgyLCB5MiwgY3RybHgxLCBjdHJseTEpLCBMaW5lMkQucHRTZWdEaXN0U3EoeDEsCi0gICAgICAgICAgICAgICAgeTEsIHgyLCB5MiwgY3RybHgyLCBjdHJseTIpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzcXVhcmUgb2YgdGhlIGZsYXRuZXNzIG9mIHRoZSBjdWJpYyBjdXJ2ZSBzZWdtZW50IGRlZmluZWQgYnkgdGhlCi0gICAgICogc3BlY2lmaWVkIHZhbHVlcy4gVGhlIHZhbHVlcyBhcmUgcmVhZCBpbiB0aGUgc2FtZSBvcmRlciBhcyB0aGUgYXJndW1lbnRzCi0gICAgICogb2YgdGhlIG1ldGhvZAotICAgICAqIHtAbGluayBDdWJpY0N1cnZlMkQjZ2V0RmxhdG5lc3NTcShkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSl9Ci0gICAgICogLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb29yZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBwb2ludHMgY29udGFpbmluZyB0aGUgbmV3IGNvb3JkaW5hdGVzLgotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgb2YgdGhlIGRhdGEgdG8gcmVhZCB3aXRoaW4gdGhlIGFycmF5LgotICAgICAqIEByZXR1cm4gdGhlIHNxdWFyZSBvZiB0aGUgZmxhdG5lc3MuCi0gICAgICogQHRocm93cyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBwb2ludHMubGVuZ3RoIDwgb2Zmc2V0ICsgLgotICAgICAqIEB0aHJvd3MgTnVsbFBvaW50ZXJFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgcG9pbnQgYXJyYXkgaXMgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGRvdWJsZSBnZXRGbGF0bmVzc1NxKGRvdWJsZSBjb29yZHNbXSwgaW50IG9mZnNldCkgewotICAgICAgICByZXR1cm4gZ2V0RmxhdG5lc3NTcShjb29yZHNbb2Zmc2V0ICsgMF0sIGNvb3Jkc1tvZmZzZXQgKyAxXSwgY29vcmRzW29mZnNldCArIDJdLAotICAgICAgICAgICAgICAgIGNvb3Jkc1tvZmZzZXQgKyAzXSwgY29vcmRzW29mZnNldCArIDRdLCBjb29yZHNbb2Zmc2V0ICsgNV0sIGNvb3Jkc1tvZmZzZXQgKyA2XSwKLSAgICAgICAgICAgICAgICBjb29yZHNbb2Zmc2V0ICsgN10pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZsYXRuZXNzIG9mIHRoaXMgY3VydmUsIHdoZXJlIHRoZSBmbGF0bmVzcyBpcyB0aGUgbWF4aW11bQotICAgICAqIGRpc3RhbmNlIGZyb20gdGhlIGN1cnZlcyBjb250cm9sIHBvaW50cyB0byB0aGUgbGluZSBzZWdtZW50IGNvbm5lY3RpbmcKLSAgICAgKiB0aGUgdHdvIHBvaW50cy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBmbGF0bmVzcyBvZiB0aGlzIGN1cnZlLgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgZ2V0RmxhdG5lc3MoKSB7Ci0gICAgICAgIHJldHVybiBnZXRGbGF0bmVzcyhnZXRYMSgpLCBnZXRZMSgpLCBnZXRDdHJsWDEoKSwgZ2V0Q3RybFkxKCksIGdldEN0cmxYMigpLCBnZXRDdHJsWTIoKSwKLSAgICAgICAgICAgICAgICBnZXRYMigpLCBnZXRZMigpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmbGF0bmVzcyBvZiB0aGUgY3ViaWMgY3VydmUgc2VnbWVudCBkZWZpbmVkIGJ5IHRoZSBzcGVjaWZpZWQKLSAgICAgKiB2YWx1ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAqIEBwYXJhbSBjdHJseDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIGN0cmx5MQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY3RybHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0gY3RybHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzZWNvbmQgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSBmbGF0bmVzcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGRvdWJsZSBnZXRGbGF0bmVzcyhkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIGN0cmx4MSwgZG91YmxlIGN0cmx5MSwKLSAgICAgICAgICAgIGRvdWJsZSBjdHJseDIsIGRvdWJsZSBjdHJseTIsIGRvdWJsZSB4MiwgZG91YmxlIHkyKSB7Ci0gICAgICAgIHJldHVybiBNYXRoLnNxcnQoZ2V0RmxhdG5lc3NTcSh4MSwgeTEsIGN0cmx4MSwgY3RybHkxLCBjdHJseDIsIGN0cmx5MiwgeDIsIHkyKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZmxhdG5lc3Mgb2YgdGhlIGN1YmljIGN1cnZlIHNlZ21lbnQgZGVmaW5lZCBieSB0aGUgc3BlY2lmaWVkCi0gICAgICogdmFsdWVzLiBUaGUgdmFsdWVzIGFyZSByZWFkIGluIHRoZSBzYW1lIG9yZGVyIGFzIHRoZSBhcmd1bWVudHMgb2YgdGhlCi0gICAgICogbWV0aG9kCi0gICAgICoge0BsaW5rIEN1YmljQ3VydmUyRCNnZXRGbGF0bmVzcyhkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSl9Ci0gICAgICogLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb29yZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBwb2ludHMgY29udGFpbmluZyB0aGUgbmV3IGNvb3JkaW5hdGVzLgotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgb2YgdGhlIGRhdGEgdG8gcmVhZCB3aXRoaW4gdGhlIGFycmF5LgotICAgICAqIEByZXR1cm4gdGhlIGZsYXRuZXNzLgotICAgICAqIEB0aHJvd3MgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgcG9pbnRzLmxlbmd0aCA8IG9mZnNldCArIC4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHBvaW50IGFycmF5IGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBkb3VibGUgZ2V0RmxhdG5lc3MoZG91YmxlIGNvb3Jkc1tdLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIHJldHVybiBnZXRGbGF0bmVzcyhjb29yZHNbb2Zmc2V0ICsgMF0sIGNvb3Jkc1tvZmZzZXQgKyAxXSwgY29vcmRzW29mZnNldCArIDJdLAotICAgICAgICAgICAgICAgIGNvb3Jkc1tvZmZzZXQgKyAzXSwgY29vcmRzW29mZnNldCArIDRdLCBjb29yZHNbb2Zmc2V0ICsgNV0sIGNvb3Jkc1tvZmZzZXQgKyA2XSwKLSAgICAgICAgICAgICAgICBjb29yZHNbb2Zmc2V0ICsgN10pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGRhdGEgZm9yIHR3byBjdWJpYyBjdXJ2ZXMgYnkgZGl2aWRpbmcgdGhpcyBjdXJ2ZSBpbiB0d28uIFRoZQotICAgICAqIGRpdmlzaW9uIHBvaW50IGlzIHRoZSBwb2ludCBvbiB0aGUgY3VydmUgdGhhdCBpcyBjbG9zZXN0IHRvIHRoZSBhdmVyYWdlCi0gICAgICogb2YgY3VydmUncyB0d28gY29udHJvbCBwb2ludHMuIFRoZSB0d28gbmV3IGNvbnRyb2wgcG9pbnRzIChuZWFyZXN0IHRoZQotICAgICAqIG5ldyBlbmRwb2ludCkgYXJlIGNvbXB1dGVkIGJ5IGF2ZXJhZ2luZyB0aGUgb3JpZ2luYWwgY29udHJvbCBwb2ludHMgd2l0aAotICAgICAqIHRoZSBuZXcgZW5kcG9pbnQuIFRoZSBkYXRhIG9mIHRoaXMgY3VydmUgaXMgbGVmdCB1bmNoYW5nZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxlZnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBDdWJpY0N1cnZlMkQgd2hlcmUgdGhlIGxlZnQgKHN0YXJ0KSBzZWdtZW50J3MgZGF0YSBpcwotICAgICAqICAgICAgICAgICAgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gcmlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBDdWJpY0N1cnZlMkQgd2hlcmUgdGhlIHJpZ2h0IChlbmQpIHNlZ21lbnQncyBkYXRhIGlzCi0gICAgICogICAgICAgICAgICB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgTnVsbFBvaW50ZXJFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBlaXRoZXIgY3VydmUgaXMgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzdWJkaXZpZGUoQ3ViaWNDdXJ2ZTJEIGxlZnQsIEN1YmljQ3VydmUyRCByaWdodCkgewotICAgICAgICBzdWJkaXZpZGUodGhpcywgbGVmdCwgcmlnaHQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGRhdGEgZm9yIHR3byBjdWJpYyBjdXJ2ZXMgYnkgZGl2aWRpbmcgdGhlIHNwZWNpZmllZCBjdXJ2ZSBpbgotICAgICAqIHR3by4gVGhlIGRpdmlzaW9uIHBvaW50IGlzIHRoZSBwb2ludCBvbiB0aGUgY3VydmUgdGhhdCBpcyBjbG9zZXN0IHRvIHRoZQotICAgICAqIGF2ZXJhZ2Ugb2YgY3VydmUncyB0d28gY29udHJvbCBwb2ludHMuIFRoZSB0d28gbmV3IGNvbnRyb2wgcG9pbnRzCi0gICAgICogKG5lYXJlc3QgdGhlIG5ldyBlbmRwb2ludCkgYXJlIGNvbXB1dGVkIGJ5IGF2ZXJhZ2luZyB0aGUgb3JpZ2luYWwgY29udHJvbAotICAgICAqIHBvaW50cyB3aXRoIHRoZSBuZXcgZW5kcG9pbnQuIFRoZSBkYXRhIG9mIHRoZSBzb3VyY2UgY3VydmUgaXMgbGVmdAotICAgICAqIHVuY2hhbmdlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgb3JpZ2luYWwgY3VydmUgdG8gYmUgZGl2aWRlZCBpbiB0d28uCi0gICAgICogQHBhcmFtIGxlZnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBDdWJpY0N1cnZlMkQgd2hlcmUgdGhlIGxlZnQgKHN0YXJ0KSBzZWdtZW50J3MgZGF0YSBpcwotICAgICAqICAgICAgICAgICAgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gcmlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBDdWJpY0N1cnZlMkQgd2hlcmUgdGhlIHJpZ2h0IChlbmQpIHNlZ21lbnQncyBkYXRhIGlzCi0gICAgICogICAgICAgICAgICB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgTnVsbFBvaW50ZXJFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBlaXRoZXIgY3VydmUgaXMgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgc3ViZGl2aWRlKEN1YmljQ3VydmUyRCBzcmMsIEN1YmljQ3VydmUyRCBsZWZ0LCBDdWJpY0N1cnZlMkQgcmlnaHQpIHsKLSAgICAgICAgZG91YmxlIHgxID0gc3JjLmdldFgxKCk7Ci0gICAgICAgIGRvdWJsZSB5MSA9IHNyYy5nZXRZMSgpOwotICAgICAgICBkb3VibGUgY3gxID0gc3JjLmdldEN0cmxYMSgpOwotICAgICAgICBkb3VibGUgY3kxID0gc3JjLmdldEN0cmxZMSgpOwotICAgICAgICBkb3VibGUgY3gyID0gc3JjLmdldEN0cmxYMigpOwotICAgICAgICBkb3VibGUgY3kyID0gc3JjLmdldEN0cmxZMigpOwotICAgICAgICBkb3VibGUgeDIgPSBzcmMuZ2V0WDIoKTsKLSAgICAgICAgZG91YmxlIHkyID0gc3JjLmdldFkyKCk7Ci0gICAgICAgIGRvdWJsZSBjeCA9IChjeDEgKyBjeDIpIC8gMi4wOwotICAgICAgICBkb3VibGUgY3kgPSAoY3kxICsgY3kyKSAvIDIuMDsKLSAgICAgICAgY3gxID0gKHgxICsgY3gxKSAvIDIuMDsKLSAgICAgICAgY3kxID0gKHkxICsgY3kxKSAvIDIuMDsKLSAgICAgICAgY3gyID0gKHgyICsgY3gyKSAvIDIuMDsKLSAgICAgICAgY3kyID0gKHkyICsgY3kyKSAvIDIuMDsKLSAgICAgICAgZG91YmxlIGF4ID0gKGN4MSArIGN4KSAvIDIuMDsKLSAgICAgICAgZG91YmxlIGF5ID0gKGN5MSArIGN5KSAvIDIuMDsKLSAgICAgICAgZG91YmxlIGJ4ID0gKGN4MiArIGN4KSAvIDIuMDsKLSAgICAgICAgZG91YmxlIGJ5ID0gKGN5MiArIGN5KSAvIDIuMDsKLSAgICAgICAgY3ggPSAoYXggKyBieCkgLyAyLjA7Ci0gICAgICAgIGN5ID0gKGF5ICsgYnkpIC8gMi4wOwotICAgICAgICBpZiAobGVmdCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBsZWZ0LnNldEN1cnZlKHgxLCB5MSwgY3gxLCBjeTEsIGF4LCBheSwgY3gsIGN5KTsKLSAgICAgICAgfQotICAgICAgICBpZiAocmlnaHQgIT0gbnVsbCkgewotICAgICAgICAgICAgcmlnaHQuc2V0Q3VydmUoY3gsIGN5LCBieCwgYnksIGN4MiwgY3kyLCB4MiwgeTIpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgZGF0YSBmb3IgdHdvIGN1YmljIGN1cnZlcyBieSBkaXZpZGluZyB0aGUgc3BlY2lmaWVkIGN1cnZlIGluCi0gICAgICogdHdvLiBUaGUgZGl2aXNpb24gcG9pbnQgaXMgdGhlIHBvaW50IG9uIHRoZSBjdXJ2ZSB0aGF0IGlzIGNsb3Nlc3QgdG8gdGhlCi0gICAgICogYXZlcmFnZSBvZiBjdXJ2ZSdzIHR3byBjb250cm9sIHBvaW50cy4gVGhlIHR3byBuZXcgY29udHJvbCBwb2ludHMKLSAgICAgKiAobmVhcmVzdCB0aGUgbmV3IGVuZHBvaW50KSBhcmUgY29tcHV0ZWQgYnkgYXZlcmFnaW5nIHRoZSBvcmlnaW5hbCBjb250cm9sCi0gICAgICogcG9pbnRzIHdpdGggdGhlIG5ldyBlbmRwb2ludC4gVGhlIGRhdGEgb2YgdGhlIHNvdXJjZSBjdXJ2ZSBpcyBsZWZ0Ci0gICAgICogdW5jaGFuZ2VkLiBUaGUgZGF0YSBmb3IgdGhlIHRocmVlIGN1cnZlcyBpcyByZWFkL3dyaXR0ZW4gaW4gdGhlIHVzdWFsCi0gICAgICogb3JkZXI6IHsgeDEsIHkxLCBjdHJseDEsIGN0cmx5MSwgY3RybHgyLCBjcnRyeTIsIHgyLCB5MyB9Ci0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IHRoYXQgZ2l2ZXMgdGhlIGRhdGEgdmFsdWVzIGZvciB0aGUgc291cmNlIGN1cnZlLgotICAgICAqIEBwYXJhbSBzcmNPZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIHNyYyBhcnJheSB0byByZWFkIHRoZSB2YWx1ZXMgZnJvbS4KLSAgICAgKiBAcGFyYW0gbGVmdAotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IHdoZXJlIHRoZSBjb29yZGluYXRlcyBvZiB0aGUgc3RhcnQgY3VydmUgc2hvdWxkIGJlCi0gICAgICogICAgICAgICAgICB3cml0dGVuLgotICAgICAqIEBwYXJhbSBsZWZ0T2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBsZWZ0IGFycmF5IHRvIHN0YXJ0IHdyaXRpbmcgdGhlIHZhbHVlcy4KLSAgICAgKiBAcGFyYW0gcmlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB3aGVyZSB0aGUgY29vcmRpbmF0ZXMgb2YgdGhlIGVuZCBjdXJ2ZSBzaG91bGQgYmUKLSAgICAgKiAgICAgICAgICAgIHdyaXR0ZW4uCi0gICAgICogQHBhcmFtIHJpZ2h0T2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSByaWdodCBhcnJheSB0byBzdGFydCB3cml0aW5nIHRoZSB2YWx1ZXMuCi0gICAgICogQHRocm93cyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBzcmMubGVuZ3RoIDwgc3Jjb2ZmICsgOCBvciBpZiBsZWZ0Lmxlbmd0aCA8IGxlZnRPZmYgKyA4IG9yCi0gICAgICogICAgICAgICAgICAgaWYgcmlnaHQubGVuZ3RoIDwgcmlnaHRPZmYgKyA4LgotICAgICAqIEB0aHJvd3MgTnVsbFBvaW50ZXJFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBvbmUgb2YgdGhlIGFycmF5cyBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgdm9pZCBzdWJkaXZpZGUoZG91YmxlIHNyY1tdLCBpbnQgc3JjT2ZmLCBkb3VibGUgbGVmdFtdLCBpbnQgbGVmdE9mZiwKLSAgICAgICAgICAgIGRvdWJsZSByaWdodFtdLCBpbnQgcmlnaHRPZmYpIHsKLSAgICAgICAgZG91YmxlIHgxID0gc3JjW3NyY09mZiArIDBdOwotICAgICAgICBkb3VibGUgeTEgPSBzcmNbc3JjT2ZmICsgMV07Ci0gICAgICAgIGRvdWJsZSBjeDEgPSBzcmNbc3JjT2ZmICsgMl07Ci0gICAgICAgIGRvdWJsZSBjeTEgPSBzcmNbc3JjT2ZmICsgM107Ci0gICAgICAgIGRvdWJsZSBjeDIgPSBzcmNbc3JjT2ZmICsgNF07Ci0gICAgICAgIGRvdWJsZSBjeTIgPSBzcmNbc3JjT2ZmICsgNV07Ci0gICAgICAgIGRvdWJsZSB4MiA9IHNyY1tzcmNPZmYgKyA2XTsKLSAgICAgICAgZG91YmxlIHkyID0gc3JjW3NyY09mZiArIDddOwotICAgICAgICBkb3VibGUgY3ggPSAoY3gxICsgY3gyKSAvIDIuMDsKLSAgICAgICAgZG91YmxlIGN5ID0gKGN5MSArIGN5MikgLyAyLjA7Ci0gICAgICAgIGN4MSA9ICh4MSArIGN4MSkgLyAyLjA7Ci0gICAgICAgIGN5MSA9ICh5MSArIGN5MSkgLyAyLjA7Ci0gICAgICAgIGN4MiA9ICh4MiArIGN4MikgLyAyLjA7Ci0gICAgICAgIGN5MiA9ICh5MiArIGN5MikgLyAyLjA7Ci0gICAgICAgIGRvdWJsZSBheCA9IChjeDEgKyBjeCkgLyAyLjA7Ci0gICAgICAgIGRvdWJsZSBheSA9IChjeTEgKyBjeSkgLyAyLjA7Ci0gICAgICAgIGRvdWJsZSBieCA9IChjeDIgKyBjeCkgLyAyLjA7Ci0gICAgICAgIGRvdWJsZSBieSA9IChjeTIgKyBjeSkgLyAyLjA7Ci0gICAgICAgIGN4ID0gKGF4ICsgYngpIC8gMi4wOwotICAgICAgICBjeSA9IChheSArIGJ5KSAvIDIuMDsKLSAgICAgICAgaWYgKGxlZnQgIT0gbnVsbCkgewotICAgICAgICAgICAgbGVmdFtsZWZ0T2ZmICsgMF0gPSB4MTsKLSAgICAgICAgICAgIGxlZnRbbGVmdE9mZiArIDFdID0geTE7Ci0gICAgICAgICAgICBsZWZ0W2xlZnRPZmYgKyAyXSA9IGN4MTsKLSAgICAgICAgICAgIGxlZnRbbGVmdE9mZiArIDNdID0gY3kxOwotICAgICAgICAgICAgbGVmdFtsZWZ0T2ZmICsgNF0gPSBheDsKLSAgICAgICAgICAgIGxlZnRbbGVmdE9mZiArIDVdID0gYXk7Ci0gICAgICAgICAgICBsZWZ0W2xlZnRPZmYgKyA2XSA9IGN4OwotICAgICAgICAgICAgbGVmdFtsZWZ0T2ZmICsgN10gPSBjeTsKLSAgICAgICAgfQotICAgICAgICBpZiAocmlnaHQgIT0gbnVsbCkgewotICAgICAgICAgICAgcmlnaHRbcmlnaHRPZmYgKyAwXSA9IGN4OwotICAgICAgICAgICAgcmlnaHRbcmlnaHRPZmYgKyAxXSA9IGN5OwotICAgICAgICAgICAgcmlnaHRbcmlnaHRPZmYgKyAyXSA9IGJ4OwotICAgICAgICAgICAgcmlnaHRbcmlnaHRPZmYgKyAzXSA9IGJ5OwotICAgICAgICAgICAgcmlnaHRbcmlnaHRPZmYgKyA0XSA9IGN4MjsKLSAgICAgICAgICAgIHJpZ2h0W3JpZ2h0T2ZmICsgNV0gPSBjeTI7Ci0gICAgICAgICAgICByaWdodFtyaWdodE9mZiArIDZdID0geDI7Ci0gICAgICAgICAgICByaWdodFtyaWdodE9mZiArIDddID0geTI7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaW5kcyB0aGUgcm9vdHMgb2YgdGhlIGN1YmljIHBvbHlub21pYWwuIFRoaXMgaXMgYWNjb21wbGlzaGVkIGJ5IGZpbmRpbmcKLSAgICAgKiB0aGUgKHJlYWwpIHZhbHVlcyBvZiB4IHRoYXQgc29sdmUgdGhlIGZvbGxvd2luZyBlcXVhdGlvbjogZXFuWzNdKngqeCp4ICsKLSAgICAgKiBlcW5bMl0qeCp4ICsgZXFuWzFdKnggKyBlcW5bMF0gPSAwLiBUaGUgc29sdXRpb25zIGFyZSB3cml0dGVuIGJhY2sgaW50bwotICAgICAqIHRoZSBhcnJheSBlcW4gc3RhcnRpbmcgZnJvbSB0aGUgaW5kZXggMCBpbiB0aGUgYXJyYXkuIFRoZSByZXR1cm4gdmFsdWUKLSAgICAgKiB0ZWxscyBob3cgbWFueSBhcnJheSBlbGVtZW50cyBoYXZlIGJlZW4gY2hhbmdlZCBieSB0aGlzIG1ldGhvZCBjYWxsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlcW4KLSAgICAgKiAgICAgICAgICAgIGFuIGFycmF5IGNvbnRhaW5pbmcgdGhlIGNvZWZmaWNpZW50cyBvZiB0aGUgY3ViaWMgcG9seW5vbWlhbAotICAgICAqICAgICAgICAgICAgdG8gc29sdmUuCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIHJvb3RzIG9mIHRoZSBjdWJpYyBwb2x5bm9taWFsLgotICAgICAqIEB0aHJvd3MgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgZXFuLmxlbmd0aCA8IDQuCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBhcnJheSBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgaW50IHNvbHZlQ3ViaWMoZG91YmxlIGVxbltdKSB7Ci0gICAgICAgIHJldHVybiBzb2x2ZUN1YmljKGVxbiwgZXFuKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaW5kcyB0aGUgcm9vdHMgb2YgdGhlIGN1YmljIHBvbHlub21pYWwuIFRoaXMgaXMgYWNjb21wbGlzaGVkIGJ5IGZpbmRpbmcKLSAgICAgKiB0aGUgKHJlYWwpIHZhbHVlcyBvZiB4IHRoYXQgc29sdmUgdGhlIGZvbGxvd2luZyBlcXVhdGlvbjogZXFuWzNdKngqeCp4ICsKLSAgICAgKiBlcW5bMl0qeCp4ICsgZXFuWzFdKnggKyBlcW5bMF0gPSAwLiBUaGUgc29sdXRpb25zIGFyZSB3cml0dGVuIGludG8gdGhlCi0gICAgICogYXJyYXkgcmVzIHN0YXJ0aW5nIGZyb20gdGhlIGluZGV4IDAgaW4gdGhlIGFycmF5LiBUaGUgcmV0dXJuIHZhbHVlIHRlbGxzCi0gICAgICogaG93IG1hbnkgYXJyYXkgZWxlbWVudHMgaGF2ZSBiZWVuIGNoYW5nZWQgYnkgdGhpcyBtZXRob2QgY2FsbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXFuCi0gICAgICogICAgICAgICAgICBhbiBhcnJheSBjb250YWluaW5nIHRoZSBjb2VmZmljaWVudHMgb2YgdGhlIGN1YmljIHBvbHlub21pYWwKLSAgICAgKiAgICAgICAgICAgIHRvIHNvbHZlLgotICAgICAqIEBwYXJhbSByZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB0aGF0IHRoaXMgbWV0aG9kIHdyaXRlcyB0aGUgcmVzdWx0cyBpbnRvLgotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiByb290cyBvZiB0aGUgY3ViaWMgcG9seW5vbWlhbC4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGVxbi5sZW5ndGggPCA0IG9yIGlmIHJlcy5sZW5ndGggaXMgbGVzcyB0aGFuIHRoZSBudW1iZXIgb2YKLSAgICAgKiAgICAgICAgICAgICByb290cy4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgZWl0aGVyIGFycmF5IGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnQgc29sdmVDdWJpYyhkb3VibGUgZXFuW10sIGRvdWJsZSByZXNbXSkgewotICAgICAgICByZXR1cm4gQ3Jvc3Npbmcuc29sdmVDdWJpYyhlcW4sIHJlcyk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoZG91YmxlIHB4LCBkb3VibGUgcHkpIHsKLSAgICAgICAgcmV0dXJuIENyb3NzaW5nLmlzSW5zaWRlRXZlbk9kZChDcm9zc2luZy5jcm9zc1NoYXBlKHRoaXMsIHB4LCBweSkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSByeCwgZG91YmxlIHJ5LCBkb3VibGUgcncsIGRvdWJsZSByaCkgewotICAgICAgICBpbnQgY3Jvc3MgPSBDcm9zc2luZy5pbnRlcnNlY3RTaGFwZSh0aGlzLCByeCwgcnksIHJ3LCByaCk7Ci0gICAgICAgIHJldHVybiBjcm9zcyAhPSBDcm9zc2luZy5DUk9TU0lORyAmJiBDcm9zc2luZy5pc0luc2lkZUV2ZW5PZGQoY3Jvc3MpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHMoZG91YmxlIHJ4LCBkb3VibGUgcnksIGRvdWJsZSBydywgZG91YmxlIHJoKSB7Ci0gICAgICAgIGludCBjcm9zcyA9IENyb3NzaW5nLmludGVyc2VjdFNoYXBlKHRoaXMsIHJ4LCByeSwgcncsIHJoKTsKLSAgICAgICAgcmV0dXJuIGNyb3NzID09IENyb3NzaW5nLkNST1NTSU5HIHx8IENyb3NzaW5nLmlzSW5zaWRlRXZlbk9kZChjcm9zcyk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoUG9pbnQyRCBwKSB7Ci0gICAgICAgIHJldHVybiBjb250YWlucyhwLmdldFgoKSwgcC5nZXRZKCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHMoUmVjdGFuZ2xlMkQgcikgewotICAgICAgICByZXR1cm4gaW50ZXJzZWN0cyhyLmdldFgoKSwgci5nZXRZKCksIHIuZ2V0V2lkdGgoKSwgci5nZXRIZWlnaHQoKSk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoUmVjdGFuZ2xlMkQgcikgewotICAgICAgICByZXR1cm4gY29udGFpbnMoci5nZXRYKCksIHIuZ2V0WSgpLCByLmdldFdpZHRoKCksIHIuZ2V0SGVpZ2h0KCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Qm91bmRzKCkgewotICAgICAgICByZXR1cm4gZ2V0Qm91bmRzMkQoKS5nZXRCb3VuZHMoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gdCkgewotICAgICAgICByZXR1cm4gbmV3IEl0ZXJhdG9yKHRoaXMsIHQpOwotICAgIH0KLQotICAgIHB1YmxpYyBQYXRoSXRlcmF0b3IgZ2V0UGF0aEl0ZXJhdG9yKEFmZmluZVRyYW5zZm9ybSBhdCwgZG91YmxlIGZsYXRuZXNzKSB7Ci0gICAgICAgIHJldHVybiBuZXcgRmxhdHRlbmluZ1BhdGhJdGVyYXRvcihnZXRQYXRoSXRlcmF0b3IoYXQpLCBmbGF0bmVzcyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBjbG9uZSgpIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBzdXBlci5jbG9uZSgpOwotICAgICAgICB9IGNhdGNoIChDbG9uZU5vdFN1cHBvcnRlZEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcigpOwotICAgICAgICB9Ci0gICAgfQotfQpcIE5vIG5ld2xpbmUgYXQgZW5kIG9mIGZpbGUKZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9nZW9tL0RpbWVuc2lvbjJELmphdmEgYi9hd3QvamF2YS9hd3QvZ2VvbS9EaW1lbnNpb24yRC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlYTA4MWM1Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9nZW9tL0RpbWVuc2lvbjJELmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4MyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERlbmlzIE0uIEtpc2hlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5nZW9tOwotCi0vKioKLSAqIFRoZSBDbGFzcyBEaW1lbnNpb24yRCByZXByZXNlbnRzIGEgc2l6ZSAod2lkdGggYW5kIGhlaWdodCkgb2YgYSBnZW9tZXRyaWMKLSAqIG9iamVjdC4gSXQgc3RvcmVzIGRvdWJsZS12YWx1ZWQgZGF0YSBpbiBvcmRlciB0byBiZSBjb21wYXRpYmxlIHdpdGgKLSAqIGhpZ2gtcHJlY2lzaW9uIGdlb21ldHJpYyBvcGVyYXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIERpbWVuc2lvbjJEIGltcGxlbWVudHMgQ2xvbmVhYmxlIHsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkaW1lbnNpb24gMmQgd2l0aCBubyBkYXRhLgotICAgICAqLwotICAgIHByb3RlY3RlZCBEaW1lbnNpb24yRCgpIHsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB3aWR0aC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB3aWR0aC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldFdpZHRoKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBoZWlnaHQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaGVpZ2h0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0SGVpZ2h0KCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSB3aWR0aCBhbmQgaGVpZ2h0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0U2l6ZShkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgd2lkdGggYW5kIGhlaWdodCBiYXNlZCBvbiB0aGUgZGF0YSBvZiBhbm90aGVyIERpbWVuc2lvbjJECi0gICAgICogb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBkCi0gICAgICogICAgICAgICAgICB0aGUgRGltZW5zaW9uMkQgb2JqZWN0IHByb3ZpZGluZyB0aGUgZGF0YSB0byBjb3B5IGludG8gdGhpcwotICAgICAqICAgICAgICAgICAgRGltZW5zaW9uMkQgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNpemUoRGltZW5zaW9uMkQgZCkgewotICAgICAgICBzZXRTaXplKGQuZ2V0V2lkdGgoKSwgZC5nZXRIZWlnaHQoKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBjbG9uZSgpIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBzdXBlci5jbG9uZSgpOwotICAgICAgICB9IGNhdGNoIChDbG9uZU5vdFN1cHBvcnRlZEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcigpOwotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2dlb20vRWxsaXBzZTJELmphdmEgYi9hd3QvamF2YS9hd3QvZ2VvbS9FbGxpcHNlMkQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggODlmZDBkMC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZ2VvbS9FbGxpcHNlMkQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ1OCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERlbmlzIE0uIEtpc2hlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5nZW9tOwotCi1pbXBvcnQgamF2YS51dGlsLk5vU3VjaEVsZW1lbnRFeGNlcHRpb247Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgRWxsaXBzZTJEIGRlc2NyaWJlcyBhbiBlbGxpcHNlIGRlZmluZWQgYnkgYSByZWN0YW5ndWxhciBhcmVhIGluCi0gKiB3aGljaCBpdCBpcyBpbnNjcmliZWQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgRWxsaXBzZTJEIGV4dGVuZHMgUmVjdGFuZ3VsYXJTaGFwZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgRmxvYXQgaXMgdGhlIHN1YmNsYXNzIG9mIEVsbGlwc2UyRCB0aGF0IGhhcyBhbGwgb2YgaXRzIGRhdGEKLSAgICAgKiB2YWx1ZXMgc3RvcmVkIHdpdGggZmxvYXQtbGV2ZWwgcHJlY2lzaW9uLgotICAgICAqIAotICAgICAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgY2xhc3MgRmxvYXQgZXh0ZW5kcyBFbGxpcHNlMkQgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgZWxsaXBzZSdzIGJvdW5kaW5nCi0gICAgICAgICAqIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCB4OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgZWxsaXBzZSdzIGJvdW5kaW5nCi0gICAgICAgICAqIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCB5OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgd2lkdGggb2YgdGhlIGVsbGlwc2UncyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgd2lkdGg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBoZWlnaHQgb2YgdGhlIGVsbGlwc2UncyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgaGVpZ2h0OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZmxvYXQtdmFsdWVkIEVsbGlwc2UyRC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBGbG9hdCgpIHsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZmxvYXQtdmFsdWVkIEVsbGlwc2UyRCB3aXRoIHRoZSBzcGVjaWZpZWQgZGF0YS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSB4Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIGVsbGlwc2UncwotICAgICAgICAgKiAgICAgICAgICAgIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgICAgICogQHBhcmFtIHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgZWxsaXBzZSdzCi0gICAgICAgICAqICAgICAgICAgICAgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGVsbGlwc2UncyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBlbGxpcHNlJ3MgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEZsb2F0KGZsb2F0IHgsIGZsb2F0IHksIGZsb2F0IHdpZHRoLCBmbG9hdCBoZWlnaHQpIHsKLSAgICAgICAgICAgIHNldEZyYW1lKHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WCgpIHsKLSAgICAgICAgICAgIHJldHVybiB4OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WSgpIHsKLSAgICAgICAgICAgIHJldHVybiB5OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0V2lkdGgoKSB7Ci0gICAgICAgICAgICByZXR1cm4gd2lkdGg7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRIZWlnaHQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gaGVpZ2h0OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzRW1wdHkoKSB7Ci0gICAgICAgICAgICByZXR1cm4gd2lkdGggPD0gMC4wIHx8IGhlaWdodCA8PSAwLjA7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogU2V0cyB0aGUgZGF0YSBvZiB0aGUgZWxsaXBzZSdzIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSB4Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIGVsbGlwc2UncwotICAgICAgICAgKiAgICAgICAgICAgIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgICAgICogQHBhcmFtIHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgZWxsaXBzZSdzCi0gICAgICAgICAqICAgICAgICAgICAgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGVsbGlwc2UncyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBlbGxpcHNlJ3MgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIHZvaWQgc2V0RnJhbWUoZmxvYXQgeCwgZmxvYXQgeSwgZmxvYXQgd2lkdGgsIGZsb2F0IGhlaWdodCkgewotICAgICAgICAgICAgdGhpcy54ID0geDsKLSAgICAgICAgICAgIHRoaXMueSA9IHk7Ci0gICAgICAgICAgICB0aGlzLndpZHRoID0gd2lkdGg7Ci0gICAgICAgICAgICB0aGlzLmhlaWdodCA9IGhlaWdodDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRGcmFtZShkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3aWR0aCwgZG91YmxlIGhlaWdodCkgewotICAgICAgICAgICAgdGhpcy54ID0gKGZsb2F0KXg7Ci0gICAgICAgICAgICB0aGlzLnkgPSAoZmxvYXQpeTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSAoZmxvYXQpd2lkdGg7Ci0gICAgICAgICAgICB0aGlzLmhlaWdodCA9IChmbG9hdCloZWlnaHQ7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZTJELkZsb2F0KHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIERvdWJsZSBpcyB0aGUgc3ViY2xhc3Mgb2YgRWxsaXBzZTJEIHRoYXQgaGFzIGFsbCBvZiBpdHMgZGF0YQotICAgICAqIHZhbHVlcyBzdG9yZWQgd2l0aCBkb3VibGUtbGV2ZWwgcHJlY2lzaW9uLgotICAgICAqIAotICAgICAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgY2xhc3MgRG91YmxlIGV4dGVuZHMgRWxsaXBzZTJEIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIGVsbGlwc2UncyBib3VuZGluZwotICAgICAgICAgKiByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIHg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSBlbGxpcHNlJ3MgYm91bmRpbmcKLSAgICAgICAgICogcmVjdGFuZ2xlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB5OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgd2lkdGggb2YgdGhlIGVsbGlwc2UncyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIHdpZHRoOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgaGVpZ2h0IG9mIHRoZSBlbGxpcHNlJ3MgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSBoZWlnaHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkb3VibGUtdmFsdWVkIEVsbGlwc2UyRC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBEb3VibGUoKSB7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRvdWJsZS12YWx1ZWQgRWxsaXBzZTJEIHdpdGggdGhlIHNwZWNpZmllZCBkYXRhLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgZWxsaXBzZSdzCi0gICAgICAgICAqICAgICAgICAgICAgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAgICAgKiBAcGFyYW0geQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSBlbGxpcHNlJ3MKLSAgICAgICAgICogICAgICAgICAgICBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgZWxsaXBzZSdzIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgICAgICogQHBhcmFtIGhlaWdodAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGVsbGlwc2UncyBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRG91YmxlKGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0KSB7Ci0gICAgICAgICAgICBzZXRGcmFtZSh4LCB5LCB3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFgoKSB7Ci0gICAgICAgICAgICByZXR1cm4geDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFkoKSB7Ci0gICAgICAgICAgICByZXR1cm4geTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFdpZHRoKCkgewotICAgICAgICAgICAgcmV0dXJuIHdpZHRoOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0SGVpZ2h0KCkgewotICAgICAgICAgICAgcmV0dXJuIGhlaWdodDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBpc0VtcHR5KCkgewotICAgICAgICAgICAgcmV0dXJuIHdpZHRoIDw9IDAuMCB8fCBoZWlnaHQgPD0gMC4wOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyB2b2lkIHNldEZyYW1lKGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0KSB7Ci0gICAgICAgICAgICB0aGlzLnggPSB4OwotICAgICAgICAgICAgdGhpcy55ID0geTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSB3aWR0aDsKLSAgICAgICAgICAgIHRoaXMuaGVpZ2h0ID0gaGVpZ2h0OwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBSZWN0YW5nbGUyRC5Eb3VibGUoeCwgeSwgd2lkdGgsIGhlaWdodCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKgotICAgICAqIEVsbGlwc2UyRCBwYXRoIGl0ZXJhdG9yCi0gICAgICovCi0gICAgLyoqCi0gICAgICogVGhlIHN1YmNsYXNzIG9mIFBhdGhJdGVyYXRvciB0byB0cmF2ZXJzZSBhbiBFbGxpcHNlMkQuCi0gICAgICovCi0gICAgY2xhc3MgSXRlcmF0b3IgaW1wbGVtZW50cyBQYXRoSXRlcmF0b3IgewotCi0gICAgICAgIC8qCi0gICAgICAgICAqIEVsbGlwc2UgaXMgc3ViZGl2aWRlZCBpbnRvIGZvdXIgcXVhcnRlcnMgYnkgeCBhbmQgeSBheGlzLiBFYWNoIHBhcnQKLSAgICAgICAgICogYXBwcm94aW1hdGVkIGJ5IGN1YmljIEJlemllciBjdXJ2ZS4gQXJjIGluIGZpcnN0IHF1YXJ0ZXIgaXMgc3RhcnRlZAotICAgICAgICAgKiBpbiAoYSwgMCkgYW5kIGZpbmlzaGVkIGluICgwLCBiKSBwb2ludHMuIENvbnRyb2wgcG9pbnRzIGZvciBjdWJpYwotICAgICAgICAgKiBjdXJ2ZSB3aWlsIGJlIChhLCAwKSwgKGEsIG0pLCAobiwgYikgYW5kICgwLCBiKSB3aGVyZSBuIGFuZCBtIGFyZQotICAgICAgICAgKiBjYWxjdWxhdGVkIGJhc2VkIG9uIHJlcXVpcmVtZW50IEJlemllciBjdXJ2ZSBpbiBwb2ludCAwLjUgc2hvdWxkIGxheQotICAgICAgICAgKiBvbiB0aGUgYXJjLgotICAgICAgICAgKi8KLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGNvZWZmaWNpZW50IHRvIGNhbGN1bGF0ZSBjb250cm9sIHBvaW50cyBvZiBCZXppZXIgY3VydmVzLgotICAgICAgICAgKi8KLSAgICAgICAgZmluYWwgZG91YmxlIHUgPSAyLjAgLyAzLjAgKiAoTWF0aC5zcXJ0KDIuMCkgLSAxLjApOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcG9pbnRzIGNvb3JkaW5hdGVzIGNhbGN1bGF0aW9uIHRhYmxlLgotICAgICAgICAgKi8KLSAgICAgICAgZmluYWwgZG91YmxlIHBvaW50c1tdW10gPSB7Ci0gICAgICAgICAgICAgICAgewotICAgICAgICAgICAgICAgICAgICAgICAgMS4wLCAwLjUgKyB1LCAwLjUgKyB1LCAxLjAsIDAuNSwgMS4wCi0gICAgICAgICAgICAgICAgfSwgewotICAgICAgICAgICAgICAgICAgICAgICAgMC41IC0gdSwgMS4wLCAwLjAsIDAuNSArIHUsIDAuMCwgMC41Ci0gICAgICAgICAgICAgICAgfSwgewotICAgICAgICAgICAgICAgICAgICAgICAgMC4wLCAwLjUgLSB1LCAwLjUgLSB1LCAwLjAsIDAuNSwgMC4wCi0gICAgICAgICAgICAgICAgfSwgewotICAgICAgICAgICAgICAgICAgICAgICAgMC41ICsgdSwgMC4wLCAxLjAsIDAuNSAtIHUsIDEuMCwgMC41Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICB9OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIGxlZnQtdXBwZXIgY29ybmVyIG9mIHRoZSBlbGxpcHNlIGJvdW5kcy4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSB4OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIGxlZnQtdXBwZXIgY29ybmVyIG9mIHRoZSBlbGxpcHNlIGJvdW5kcy4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSB5OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgd2lkdGggb2YgdGhlIGVsbGlwc2UgYm91bmRzLgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIHdpZHRoOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgaGVpZ2h0IG9mIHRoZSBlbGxpcHNlIGJvdW5kcy4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSBoZWlnaHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBwYXRoIGl0ZXJhdG9yIHRyYW5zZm9ybWF0aW9uLgotICAgICAgICAgKi8KLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjdXJyZW50IHNlZ21lbnQgaW5kZXguCi0gICAgICAgICAqLwotICAgICAgICBpbnQgaW5kZXg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENvbnN0cnVjdHMgYSBuZXcgRWxsaXBzZTJELkl0ZXJhdG9yIGZvciBnaXZlbiBlbGxpcHNlIGFuZAotICAgICAgICAgKiB0cmFuc2Zvcm1hdGlvbgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgc291cmNlIEVsbGlwc2UyRCBvYmplY3QuCi0gICAgICAgICAqIEBwYXJhbSB0Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGFmZmluZSB0cmFuc2Zvcm1hdGlvbiBvYmplY3QuCi0gICAgICAgICAqLwotICAgICAgICBJdGVyYXRvcihFbGxpcHNlMkQgZSwgQWZmaW5lVHJhbnNmb3JtIHQpIHsKLSAgICAgICAgICAgIHRoaXMueCA9IGUuZ2V0WCgpOwotICAgICAgICAgICAgdGhpcy55ID0gZS5nZXRZKCk7Ci0gICAgICAgICAgICB0aGlzLndpZHRoID0gZS5nZXRXaWR0aCgpOwotICAgICAgICAgICAgdGhpcy5oZWlnaHQgPSBlLmdldEhlaWdodCgpOwotICAgICAgICAgICAgdGhpcy50ID0gdDsKLSAgICAgICAgICAgIGlmICh3aWR0aCA8IDAuMCB8fCBoZWlnaHQgPCAwLjApIHsKLSAgICAgICAgICAgICAgICBpbmRleCA9IDY7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGdldFdpbmRpbmdSdWxlKCkgewotICAgICAgICAgICAgcmV0dXJuIFdJTkRfTk9OX1pFUk87Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBpc0RvbmUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gaW5kZXggPiA1OwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIHZvaWQgbmV4dCgpIHsKLSAgICAgICAgICAgIGluZGV4Kys7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGRvdWJsZVtdIGNvb3JkcykgewotICAgICAgICAgICAgaWYgKGlzRG9uZSgpKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjRCPUl0ZXJhdG9yIG91dCBvZiBib3VuZHMKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgTm9TdWNoRWxlbWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40QiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGluZGV4ID09IDUpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gU0VHX0NMT1NFOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaW50IHR5cGU7Ci0gICAgICAgICAgICBpbnQgY291bnQ7Ci0gICAgICAgICAgICBpZiAoaW5kZXggPT0gMCkgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTU9WRVRPOwotICAgICAgICAgICAgICAgIGNvdW50ID0gMTsKLSAgICAgICAgICAgICAgICBkb3VibGUgcFtdID0gcG9pbnRzWzNdOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IHggKyBwWzRdICogd2lkdGg7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzFdID0geSArIHBbNV0gKiBoZWlnaHQ7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfQ1VCSUNUTzsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDM7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHBbXSA9IHBvaW50c1tpbmRleCAtIDFdOwotICAgICAgICAgICAgICAgIGludCBqID0gMDsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IDM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBjb29yZHNbal0gPSB4ICsgcFtqKytdICogd2lkdGg7Ci0gICAgICAgICAgICAgICAgICAgIGNvb3Jkc1tqXSA9IHkgKyBwW2orK10gKiBoZWlnaHQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCBjb3VudCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gdHlwZTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgY3VycmVudFNlZ21lbnQoZmxvYXRbXSBjb29yZHMpIHsKLSAgICAgICAgICAgIGlmIChpc0RvbmUoKSkgewotICAgICAgICAgICAgICAgIC8vIGF3dC40Qj1JdGVyYXRvciBvdXQgb2YgYm91bmRzCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IE5vU3VjaEVsZW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNEIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChpbmRleCA9PSA1KSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIFNFR19DTE9TRTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGludCB0eXBlOwotICAgICAgICAgICAgaW50IGNvdW50OwotICAgICAgICAgICAgaWYgKGluZGV4ID09IDApIHsKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX01PVkVUTzsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDE7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHBbXSA9IHBvaW50c1szXTsKLSAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSAoZmxvYXQpKHggKyBwWzRdICogd2lkdGgpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IChmbG9hdCkoeSArIHBbNV0gKiBoZWlnaHQpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX0NVQklDVE87Ci0gICAgICAgICAgICAgICAgY291bnQgPSAzOwotICAgICAgICAgICAgICAgIGludCBqID0gMDsKLSAgICAgICAgICAgICAgICBkb3VibGUgcFtdID0gcG9pbnRzW2luZGV4IC0gMV07Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCAzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgY29vcmRzW2pdID0gKGZsb2F0KSh4ICsgcFtqKytdICogd2lkdGgpOwotICAgICAgICAgICAgICAgICAgICBjb29yZHNbal0gPSAoZmxvYXQpKHkgKyBwW2orK10gKiBoZWlnaHQpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh0ICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICB0LnRyYW5zZm9ybShjb29yZHMsIDAsIGNvb3JkcywgMCwgY291bnQpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIHR5cGU7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBFbGxpcHNlMkQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEVsbGlwc2UyRCgpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhkb3VibGUgcHgsIGRvdWJsZSBweSkgewotICAgICAgICBpZiAoaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgYSA9IChweCAtIGdldFgoKSkgLyBnZXRXaWR0aCgpIC0gMC41OwotICAgICAgICBkb3VibGUgYiA9IChweSAtIGdldFkoKSkgLyBnZXRIZWlnaHQoKSAtIDAuNTsKLQotICAgICAgICByZXR1cm4gYSAqIGEgKyBiICogYiA8IDAuMjU7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0cyhkb3VibGUgcngsIGRvdWJsZSByeSwgZG91YmxlIHJ3LCBkb3VibGUgcmgpIHsKLSAgICAgICAgaWYgKGlzRW1wdHkoKSB8fCBydyA8PSAwLjAgfHwgcmggPD0gMC4wKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgY3ggPSBnZXRYKCkgKyBnZXRXaWR0aCgpIC8gMi4wOwotICAgICAgICBkb3VibGUgY3kgPSBnZXRZKCkgKyBnZXRIZWlnaHQoKSAvIDIuMDsKLQotICAgICAgICBkb3VibGUgcngxID0gcng7Ci0gICAgICAgIGRvdWJsZSByeTEgPSByeTsKLSAgICAgICAgZG91YmxlIHJ4MiA9IHJ4ICsgcnc7Ci0gICAgICAgIGRvdWJsZSByeTIgPSByeSArIHJoOwotCi0gICAgICAgIGRvdWJsZSBueCA9IGN4IDwgcngxID8gcngxIDogKGN4ID4gcngyID8gcngyIDogY3gpOwotICAgICAgICBkb3VibGUgbnkgPSBjeSA8IHJ5MSA/IHJ5MSA6IChjeSA+IHJ5MiA/IHJ5MiA6IGN5KTsKLQotICAgICAgICByZXR1cm4gY29udGFpbnMobngsIG55KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhkb3VibGUgcngsIGRvdWJsZSByeSwgZG91YmxlIHJ3LCBkb3VibGUgcmgpIHsKLSAgICAgICAgaWYgKGlzRW1wdHkoKSB8fCBydyA8PSAwLjAgfHwgcmggPD0gMC4wKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgcngxID0gcng7Ci0gICAgICAgIGRvdWJsZSByeTEgPSByeTsKLSAgICAgICAgZG91YmxlIHJ4MiA9IHJ4ICsgcnc7Ci0gICAgICAgIGRvdWJsZSByeTIgPSByeSArIHJoOwotCi0gICAgICAgIHJldHVybiBjb250YWlucyhyeDEsIHJ5MSkgJiYgY29udGFpbnMocngyLCByeTEpICYmIGNvbnRhaW5zKHJ4MiwgcnkyKSAmJiBjb250YWlucyhyeDEsIHJ5Mik7Ci0gICAgfQotCi0gICAgcHVibGljIFBhdGhJdGVyYXRvciBnZXRQYXRoSXRlcmF0b3IoQWZmaW5lVHJhbnNmb3JtIGF0KSB7Ci0gICAgICAgIHJldHVybiBuZXcgSXRlcmF0b3IodGhpcywgYXQpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9nZW9tL0ZsYXR0ZW5pbmdQYXRoSXRlcmF0b3IuamF2YSBiL2F3dC9qYXZhL2F3dC9nZW9tL0ZsYXR0ZW5pbmdQYXRoSXRlcmF0b3IuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggODIwOGYzOS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZ2VvbS9GbGF0dGVuaW5nUGF0aEl0ZXJhdG9yLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNTggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZ2VvbTsKLQotaW1wb3J0IGphdmEudXRpbC5Ob1N1Y2hFbGVtZW50RXhjZXB0aW9uOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIENsYXNzIEZsYXR0ZW5pbmdQYXRoSXRlcmF0b3IgdGFrZXMgYSBQYXRoSXRlcmF0b3IgZm9yIHRyYXZlcnNpbmcgYSBjdXJ2ZWQKLSAqIHNoYXBlIGFuZCBmbGF0dGVucyBpdCBieSBlc3RpbWF0aW5nIHRoZSBjdXJ2ZSBhcyBhIHNlcmllcyBvZiBsaW5lIHNlZ21lbnRzLgotICogVGhlIGZsYXR0ZW5pbmcgZmFjdG9yIGluZGljYXRlcyBob3cgZmFyIHRoZSBlc3RpbWF0aW5nIGxpbmUgc2VnbWVudHMgYXJlCi0gKiBhbGxvd2VkIHRvIGJlIGZyb20gdGhlIGFjdHVhbCBjdXJ2ZTogdGhlIEZsYXR0ZW5pbmdQYXRoSXRlcmF0b3Igd2lsbCBrZWVwCi0gKiBkaXZpZGluZyBlYWNoIGN1cnZlZCBzZWdtZW50IGludG8gc21hbGxlciBhbmQgc21hbGxlciBmbGF0IHNlZ21lbnRzIHVudGlsCi0gKiBlaXRoZXIgdGhlIHNlZ21lbnRzIGFyZSB3aXRoaW4gdGhlIGZsYXR0ZW5pbmcgZmFjdG9yIG9mIHRoZSBjdXJ2ZSBvciB1bnRpbAotICogdGhlIGJ1ZmZlciBsaW1pdCBpcyByZWFjaGVkLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEZsYXR0ZW5pbmdQYXRoSXRlcmF0b3IgaW1wbGVtZW50cyBQYXRoSXRlcmF0b3IgewotCi0gICAgLyoqCi0gICAgICogVGhlIGRlZmF1bHQgcG9pbnRzIGJ1ZmZlciBzaXplLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBCVUZGRVJfU0laRSA9IDE2OwotCi0gICAgLyoqCi0gICAgICogVGhlIGRlZmF1bHQgY3VydmUgc3ViZGl2aXNpb24gbGltaXQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJVRkZFUl9MSU1JVCA9IDE2OwotCi0gICAgLyoqCi0gICAgICogVGhlIHBvaW50cyBidWZmZXIgY2FwYWNpdHkuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJVRkZFUl9DQVBBQ0lUWSA9IDE2OwotCi0gICAgLyoqCi0gICAgICogVGhlIHR5cGUgb2YgY3VycmVudCBzZWdtZW50IHRvIGJlIGZsYXQuCi0gICAgICovCi0gICAgaW50IGJ1ZlR5cGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY3VydmUgc3ViZGl2aXNpb24gbGltaXQuCi0gICAgICovCi0gICAgaW50IGJ1ZkxpbWl0OwotCi0gICAgLyoqCi0gICAgICogVGhlIGN1cnJlbnQgcG9pbnRzIGJ1ZmZlciBzaXplLgotICAgICAqLwotICAgIGludCBidWZTaXplOwotCi0gICAgLyoqCi0gICAgICogVGhlIGlubmVyIGN1cnNvciBwb3NpdGlvbiBpbiBwb2ludHMgYnVmZmVyLgotICAgICAqLwotICAgIGludCBidWZJbmRleDsKLQotICAgIC8qKgotICAgICAqIFRoZSBjdXJyZW50IHN1YmRpdmlzaW9uIGNvdW50LgotICAgICAqLwotICAgIGludCBidWZTdWJkaXY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcG9pbnRzIGJ1ZmZlci4KLSAgICAgKi8KLSAgICBkb3VibGUgYnVmW107Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaW5kaWNhdG9yIG9mIGVtcHR5IHBvaW50cyBidWZmZXIuCi0gICAgICovCi0gICAgYm9vbGVhbiBidWZFbXB0eSA9IHRydWU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc291cmNlIFBhdGhJdGVyYXRvci4KLSAgICAgKi8KLSAgICBQYXRoSXRlcmF0b3IgcDsKLQotICAgIC8qKgotICAgICAqIFRoZSBmbGF0bmVzcyBvZiBuZXcgcGF0aC4KLSAgICAgKi8KLSAgICBkb3VibGUgZmxhdG5lc3M7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc3F1YXJlIG9mIGZsYXRuZXNzLgotICAgICAqLwotICAgIGRvdWJsZSBmbGF0bmVzczI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHByZXZpb3VzIHBhdGggc2VnbWVudC4KLSAgICAgKi8KLSAgICBkb3VibGUgcHg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIHByZXZpb3VzIHBhdGggc2VnbWVudC4KLSAgICAgKi8KLSAgICBkb3VibGUgcHk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdGVtcG9yYXJ5IGJ1ZmZlciBmb3IgZ2V0dGluZyBwb2ludHMgZnJvbSBQYXRoSXRlcmF0b3IuCi0gICAgICovCi0gICAgZG91YmxlIGNvb3Jkc1tdID0gbmV3IGRvdWJsZVs2XTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBmbGF0dGVuaW5nIHBhdGggaXRlcmF0b3IgZ2l2ZW4gdGhlIHBhdGggaXRlcmF0b3IgZm9yIGEKLSAgICAgKiAocG9zc2libHkpIGN1cnZlZCBwYXRoIGFuZCBhIGZsYXR0ZW5pbmcgZmFjdG9yIHdoaWNoIGluZGljYXRlcyBob3cgY2xvc2UKLSAgICAgKiB0b2dldGhlciB0aGUgcG9pbnRzIG9uIHRoZSBjdXJ2ZSBzaG91bGQgYmUgY2hvc2VuLiBUaGUgYnVmZmVyIGxpbWl0Ci0gICAgICogZGVmYXVsdHMgdG8gMTYgd2hpY2ggbWVhbnMgdGhhdCBlYWNoIGN1cnZlIHdpbGwgYmUgZGl2aWRlZCBpbnRvIG5vIG1vcmUKLSAgICAgKiB0aGFuIDE2IHNlZ21lbnRzIHJlZ2FyZGxlc3Mgb2YgdGhlIGZsYXR0ZW5pbmcgZmFjdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXRoCi0gICAgICogICAgICAgICAgICB0aGUgcGF0aCBpdGVyYXRvciBvZiB0aGUgb3JpZ2luYWwgY3VydmUuCi0gICAgICogQHBhcmFtIGZsYXRuZXNzCi0gICAgICogICAgICAgICAgICB0aGUgZmxhdHRlbmluZyBmYWN0b3IgdGhhdCBpbmRpY2F0ZXMgaG93IGZhciB0aGUgZmxhdCBwYXRoIGlzCi0gICAgICogICAgICAgICAgICBhbGxvd2VkIHRvIGJlIGZyb20gdGhlIGFjdHVhbCBjdXJ2ZSBpbiBvcmRlciB0byBkZWNpZGUgd2hlbiB0bwotICAgICAqICAgICAgICAgICAgc3RvcCBkaXZpZGluZyB0aGUgcGF0aCBpbnRvIHNtYWxsZXIgYW5kIHNtYWxsZXIgc2VnbWVudHMuCi0gICAgICogQHRocm93cyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgZmxhdG5lc3MgaXMgbGVzcyB0aGFuIHplcm8uCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBwYXRoIGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIEZsYXR0ZW5pbmdQYXRoSXRlcmF0b3IoUGF0aEl0ZXJhdG9yIHBhdGgsIGRvdWJsZSBmbGF0bmVzcykgewotICAgICAgICB0aGlzKHBhdGgsIGZsYXRuZXNzLCBCVUZGRVJfTElNSVQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBmbGF0dGVuaW5nIHBhdGggaXRlcmF0b3IgZ2l2ZW4gdGhlIHBhdGggaXRlcmF0b3IgZm9yIGEKLSAgICAgKiAocG9zc2libHkpIGN1cnZlZCBwYXRoIGFuZCBhIGZsYXR0ZW5pbmcgZmFjdG9yIGFuZCBhIGJ1ZmZlciBsaW1pdC4gVGhlCi0gICAgICogRmxhdHRlbmluZ1BhdGhJdGVyYXRvciB3aWxsIGtlZXAgZGl2aWRpbmcgZWFjaCBjdXJ2ZWQgc2VnbWVudCBpbnRvCi0gICAgICogc21hbGxlciBhbmQgc21hbGxlciBmbGF0IHNlZ21lbnRzIHVudGlsIGVpdGhlciB0aGUgc2VnbWVudHMgYXJlIHdpdGhpbgotICAgICAqIHRoZSBmbGF0dGVuaW5nIGZhY3RvciBvZiB0aGUgY3VydmUgb3IgdW50aWwgdGhlIGJ1ZmZlciBsaW1pdCBpcyByZWFjaGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXRoCi0gICAgICogICAgICAgICAgICB0aGUgcGF0aCBpdGVyYXRvciBvZiB0aGUgb3JpZ2luYWwgY3VydmUuCi0gICAgICogQHBhcmFtIGZsYXRuZXNzCi0gICAgICogICAgICAgICAgICB0aGUgZmxhdHRlbmluZyBmYWN0b3IgdGhhdCBpbmRpY2F0ZXMgaG93IGZhciB0aGUgZmxhdCBwYXRoIGlzCi0gICAgICogICAgICAgICAgICBhbGxvd2VkIHRvIGJlIGZyb20gdGhlIGFjdHVhbCBjdXJ2ZSBpbiBvcmRlciB0byBkZWNpZGUgd2hlbiB0bwotICAgICAqICAgICAgICAgICAgc3RvcCBkaXZpZGluZyB0aGUgcGF0aCBpbnRvIHNtYWxsZXIgYW5kIHNtYWxsZXIgc2VnbWVudHMuCi0gICAgICogQHBhcmFtIGxpbWl0Ci0gICAgICogICAgICAgICAgICB0aGUgbWF4aW11bSBudW1iZXIgb2YgZmxhdCBzZWdtZW50cyB0byBkaXZpZGUgZWFjaCBjdXJ2ZSBpbnRvLgotICAgICAqIEB0aHJvd3MgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIGZsYXRuZXNzIG9yIGxpbWl0IGlzIGxlc3MgdGhhbiB6ZXJvLgotICAgICAqIEB0aHJvd3MgTnVsbFBvaW50ZXJFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgcGF0aCBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBGbGF0dGVuaW5nUGF0aEl0ZXJhdG9yKFBhdGhJdGVyYXRvciBwYXRoLCBkb3VibGUgZmxhdG5lc3MsIGludCBsaW1pdCkgewotICAgICAgICBpZiAoZmxhdG5lc3MgPCAwLjApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMDY9RmxhdG5lc3MgaXMgbGVzcyB0aGVuIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjA2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKGxpbWl0IDwgMCkgewotICAgICAgICAgICAgLy8gYXd0LjIwNz1MaW1pdCBpcyBsZXNzIHRoZW4gemVybwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMDciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZiAocGF0aCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjA4PVBhdGggaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIwOCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHRoaXMucCA9IHBhdGg7Ci0gICAgICAgIHRoaXMuZmxhdG5lc3MgPSBmbGF0bmVzczsKLSAgICAgICAgdGhpcy5mbGF0bmVzczIgPSBmbGF0bmVzcyAqIGZsYXRuZXNzOwotICAgICAgICB0aGlzLmJ1ZkxpbWl0ID0gbGltaXQ7Ci0gICAgICAgIHRoaXMuYnVmU2l6ZSA9IE1hdGgubWluKGJ1ZkxpbWl0LCBCVUZGRVJfU0laRSk7Ci0gICAgICAgIHRoaXMuYnVmID0gbmV3IGRvdWJsZVtidWZTaXplXTsKLSAgICAgICAgdGhpcy5idWZJbmRleCA9IGJ1ZlNpemU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZmxhdHRlbmluZyBmYWN0b3IuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZmxhdHRlbmluZyBmYWN0b3IuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRGbGF0bmVzcygpIHsKLSAgICAgICAgcmV0dXJuIGZsYXRuZXNzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1heGltdW0gbnVtYmVyIG9mIHN1YmRpdmlzaW9ucyBwZXIgY3VydmVkIHNlZ21lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWF4aW11bSBudW1iZXIgb2Ygc3ViZGl2aXNpb25zIHBlciBjdXJ2ZWQgc2VnbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFJlY3Vyc2lvbkxpbWl0KCkgewotICAgICAgICByZXR1cm4gYnVmTGltaXQ7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRXaW5kaW5nUnVsZSgpIHsKLSAgICAgICAgcmV0dXJuIHAuZ2V0V2luZGluZ1J1bGUoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0RvbmUoKSB7Ci0gICAgICAgIHJldHVybiBidWZFbXB0eSAmJiBwLmlzRG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENhbGN1bGF0ZXMgZmxhdCBwYXRoIHBvaW50cyBmb3IgY3VycmVudCBzZWdtZW50IG9mIHRoZSBzb3VyY2Ugc2hhcGUuIExpbmUKLSAgICAgKiBzZWdtZW50IGlzIGZsYXQgYnkgaXRzZWxmLiBGbGF0bmVzcyBvZiBxdWFkIGFuZCBjdWJpYyBjdXJ2ZXMgZXZhbHVhdGVkIGJ5Ci0gICAgICogZ2V0RmxhdG5lc3NTcSgpIG1ldGhvZC4gQ3VydmVzIHN1YmRpdmlkZWQgdW50aWwgY3VycmVudCBmbGF0bmVzcyBpcwotICAgICAqIGJpZ2dlciB0aGFuIHVzZXIgZGVmaW5lZCBhbmQgc3ViZGl2aXNpb24gbGltaXQgaXNuJ3QgZXhoYXVzdGVkLiBTaW5nbGUKLSAgICAgKiBzb3VyY2Ugc2VnbWVudCB0cmFuc2xhdGVkIHRvIHNlcmllcyBvZiBidWZmZXIgcG9pbnRzLiBUaGUgbGVzcyBmbGF0bmVzcwotICAgICAqIHRoZSBiaWdnZXIgc2VyaWVzLiBFdmVyeSBjdXJyZW50U2VnbWVudCgpIGNhbGwgZXh0cmFjdCBvbmUgcG9pbnQgZnJvbSB0aGUKLSAgICAgKiBidWZmZXIuIFdoZW4gc2VyaWVzIGNvbXBsZXRlZCBldmFsdWF0ZSgpIHRha2VzIG5leHQgc291cmNlIHNoYXBlIHNlZ21lbnQuCi0gICAgICovCi0gICAgdm9pZCBldmFsdWF0ZSgpIHsKLSAgICAgICAgaWYgKGJ1ZkVtcHR5KSB7Ci0gICAgICAgICAgICBidWZUeXBlID0gcC5jdXJyZW50U2VnbWVudChjb29yZHMpOwotICAgICAgICB9Ci0KLSAgICAgICAgc3dpdGNoIChidWZUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIFNFR19NT1ZFVE86Ci0gICAgICAgICAgICBjYXNlIFNFR19MSU5FVE86Ci0gICAgICAgICAgICAgICAgcHggPSBjb29yZHNbMF07Ci0gICAgICAgICAgICAgICAgcHkgPSBjb29yZHNbMV07Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIFNFR19RVUFEVE86Ci0gICAgICAgICAgICAgICAgaWYgKGJ1ZkVtcHR5KSB7Ci0gICAgICAgICAgICAgICAgICAgIGJ1ZkluZGV4IC09IDY7Ci0gICAgICAgICAgICAgICAgICAgIGJ1ZltidWZJbmRleCArIDBdID0gcHg7Ci0gICAgICAgICAgICAgICAgICAgIGJ1ZltidWZJbmRleCArIDFdID0gcHk7Ci0gICAgICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkoY29vcmRzLCAwLCBidWYsIGJ1ZkluZGV4ICsgMiwgNCk7Ci0gICAgICAgICAgICAgICAgICAgIGJ1ZlN1YmRpdiA9IDA7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgd2hpbGUgKGJ1ZlN1YmRpdiA8IGJ1ZkxpbWl0KSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChRdWFkQ3VydmUyRC5nZXRGbGF0bmVzc1NxKGJ1ZiwgYnVmSW5kZXgpIDwgZmxhdG5lc3MyKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIC8vIFJlYWxsb2MgYnVmZmVyCi0gICAgICAgICAgICAgICAgICAgIGlmIChidWZJbmRleCA8PSA0KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkb3VibGUgdG1wW10gPSBuZXcgZG91YmxlW2J1ZlNpemUgKyBCVUZGRVJfQ0FQQUNJVFldOwotICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShidWYsIGJ1ZkluZGV4LCB0bXAsIGJ1ZkluZGV4ICsgQlVGRkVSX0NBUEFDSVRZLCBidWZTaXplCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC0gYnVmSW5kZXgpOwotICAgICAgICAgICAgICAgICAgICAgICAgYnVmID0gdG1wOwotICAgICAgICAgICAgICAgICAgICAgICAgYnVmU2l6ZSArPSBCVUZGRVJfQ0FQQUNJVFk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBidWZJbmRleCArPSBCVUZGRVJfQ0FQQUNJVFk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBRdWFkQ3VydmUyRC5zdWJkaXZpZGUoYnVmLCBidWZJbmRleCwgYnVmLCBidWZJbmRleCAtIDQsIGJ1ZiwgYnVmSW5kZXgpOwotCi0gICAgICAgICAgICAgICAgICAgIGJ1ZkluZGV4IC09IDQ7Ci0gICAgICAgICAgICAgICAgICAgIGJ1ZlN1YmRpdisrOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGJ1ZkluZGV4ICs9IDQ7Ci0gICAgICAgICAgICAgICAgcHggPSBidWZbYnVmSW5kZXhdOwotICAgICAgICAgICAgICAgIHB5ID0gYnVmW2J1ZkluZGV4ICsgMV07Ci0KLSAgICAgICAgICAgICAgICBidWZFbXB0eSA9IChidWZJbmRleCA9PSBidWZTaXplIC0gMik7Ci0gICAgICAgICAgICAgICAgaWYgKGJ1ZkVtcHR5KSB7Ci0gICAgICAgICAgICAgICAgICAgIGJ1ZkluZGV4ID0gYnVmU2l6ZTsKLSAgICAgICAgICAgICAgICAgICAgYnVmVHlwZSA9IFNFR19MSU5FVE87Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgYnVmU3ViZGl2LS07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBTRUdfQ1VCSUNUTzoKLSAgICAgICAgICAgICAgICBpZiAoYnVmRW1wdHkpIHsKLSAgICAgICAgICAgICAgICAgICAgYnVmSW5kZXggLT0gODsKLSAgICAgICAgICAgICAgICAgICAgYnVmW2J1ZkluZGV4ICsgMF0gPSBweDsKLSAgICAgICAgICAgICAgICAgICAgYnVmW2J1ZkluZGV4ICsgMV0gPSBweTsKLSAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShjb29yZHMsIDAsIGJ1ZiwgYnVmSW5kZXggKyAyLCA2KTsKLSAgICAgICAgICAgICAgICAgICAgYnVmU3ViZGl2ID0gMDsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICB3aGlsZSAoYnVmU3ViZGl2IDwgYnVmTGltaXQpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKEN1YmljQ3VydmUyRC5nZXRGbGF0bmVzc1NxKGJ1ZiwgYnVmSW5kZXgpIDwgZmxhdG5lc3MyKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIC8vIFJlYWxsb2MgYnVmZmVyCi0gICAgICAgICAgICAgICAgICAgIGlmIChidWZJbmRleCA8PSA2KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkb3VibGUgdG1wW10gPSBuZXcgZG91YmxlW2J1ZlNpemUgKyBCVUZGRVJfQ0FQQUNJVFldOwotICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShidWYsIGJ1ZkluZGV4LCB0bXAsIGJ1ZkluZGV4ICsgQlVGRkVSX0NBUEFDSVRZLCBidWZTaXplCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC0gYnVmSW5kZXgpOwotICAgICAgICAgICAgICAgICAgICAgICAgYnVmID0gdG1wOwotICAgICAgICAgICAgICAgICAgICAgICAgYnVmU2l6ZSArPSBCVUZGRVJfQ0FQQUNJVFk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBidWZJbmRleCArPSBCVUZGRVJfQ0FQQUNJVFk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBDdWJpY0N1cnZlMkQuc3ViZGl2aWRlKGJ1ZiwgYnVmSW5kZXgsIGJ1ZiwgYnVmSW5kZXggLSA2LCBidWYsIGJ1ZkluZGV4KTsKLQotICAgICAgICAgICAgICAgICAgICBidWZJbmRleCAtPSA2OwotICAgICAgICAgICAgICAgICAgICBidWZTdWJkaXYrKzsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBidWZJbmRleCArPSA2OwotICAgICAgICAgICAgICAgIHB4ID0gYnVmW2J1ZkluZGV4XTsKLSAgICAgICAgICAgICAgICBweSA9IGJ1ZltidWZJbmRleCArIDFdOwotCi0gICAgICAgICAgICAgICAgYnVmRW1wdHkgPSAoYnVmSW5kZXggPT0gYnVmU2l6ZSAtIDIpOwotICAgICAgICAgICAgICAgIGlmIChidWZFbXB0eSkgewotICAgICAgICAgICAgICAgICAgICBidWZJbmRleCA9IGJ1ZlNpemU7Ci0gICAgICAgICAgICAgICAgICAgIGJ1ZlR5cGUgPSBTRUdfTElORVRPOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGJ1ZlN1YmRpdi0tOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgcHVibGljIHZvaWQgbmV4dCgpIHsKLSAgICAgICAgaWYgKGJ1ZkVtcHR5KSB7Ci0gICAgICAgICAgICBwLm5leHQoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgY3VycmVudFNlZ21lbnQoZmxvYXRbXSBjb29yZHMpIHsKLSAgICAgICAgaWYgKGlzRG9uZSgpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNEI9SXRlcmF0b3Igb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IE5vU3VjaEVsZW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNEJ4IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgZXZhbHVhdGUoKTsKLSAgICAgICAgaW50IHR5cGUgPSBidWZUeXBlOwotICAgICAgICBpZiAodHlwZSAhPSBTRUdfQ0xPU0UpIHsKLSAgICAgICAgICAgIGNvb3Jkc1swXSA9IChmbG9hdClweDsKLSAgICAgICAgICAgIGNvb3Jkc1sxXSA9IChmbG9hdClweTsKLSAgICAgICAgICAgIGlmICh0eXBlICE9IFNFR19NT1ZFVE8pIHsKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX0xJTkVUTzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdHlwZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGRvdWJsZVtdIGNvb3JkcykgewotICAgICAgICBpZiAoaXNEb25lKCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC40Qj1JdGVyYXRvciBvdXQgb2YgYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgTm9TdWNoRWxlbWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40QiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGV2YWx1YXRlKCk7Ci0gICAgICAgIGludCB0eXBlID0gYnVmVHlwZTsKLSAgICAgICAgaWYgKHR5cGUgIT0gU0VHX0NMT1NFKSB7Ci0gICAgICAgICAgICBjb29yZHNbMF0gPSBweDsKLSAgICAgICAgICAgIGNvb3Jkc1sxXSA9IHB5OwotICAgICAgICAgICAgaWYgKHR5cGUgIT0gU0VHX01PVkVUTykgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTElORVRPOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB0eXBlOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9nZW9tL0dlbmVyYWxQYXRoLmphdmEgYi9hd3QvamF2YS9hd3QvZ2VvbS9HZW5lcmFsUGF0aC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwNjY5YmM3Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9nZW9tL0dlbmVyYWxQYXRoLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2MjQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZ2VvbTsKLQotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLWltcG9ydCBqYXZhLmF3dC5TaGFwZTsKLWltcG9ydCBqYXZhLnV0aWwuTm9TdWNoRWxlbWVudEV4Y2VwdGlvbjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuQ3Jvc3Npbmc7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIGNsYXNzIEdlbmVyYWxQYXRoIHJlcHJlc2VudHMgYSBzaGFwZSB3aG9zZSBvdXRsaW5lIGlzIGdpdmVuIGJ5IGRpZmZlcmVudAotICogdHlwZXMgb2YgY3VydmVkIGFuZCBzdHJhaWdodCBzZWdtZW50cy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBmaW5hbCBjbGFzcyBHZW5lcmFsUGF0aCBpbXBsZW1lbnRzIFNoYXBlLCBDbG9uZWFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFdJTkRfRVZFTl9PREQgc2VlIHtAbGluayBQYXRoSXRlcmF0b3IjV0lORF9FVkVOX09ERH0uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lORF9FVkVOX09ERCA9IFBhdGhJdGVyYXRvci5XSU5EX0VWRU5fT0REOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFdJTkRfTk9OX1pFUk8gc2VlIHtAbGluayBQYXRoSXRlcmF0b3IjV0lORF9OT05fWkVST30uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lORF9OT05fWkVSTyA9IFBhdGhJdGVyYXRvci5XSU5EX05PTl9aRVJPOwotCi0gICAgLyoqCi0gICAgICogVGhlIGJ1ZmZlcnMgc2l6ZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgQlVGRkVSX1NJWkUgPSAxMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBidWZmZXJzIGNhcGFjaXR5LgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBCVUZGRVJfQ0FQQUNJVFkgPSAxMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBwb2ludCdzIHR5cGVzIGJ1ZmZlci4KLSAgICAgKi8KLSAgICBieXRlW10gdHlwZXM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcG9pbnRzIGJ1ZmZlci4KLSAgICAgKi8KLSAgICBmbG9hdFtdIHBvaW50czsKLQotICAgIC8qKgotICAgICAqIFRoZSBwb2ludCdzIHR5cGUgYnVmZmVyIHNpemUuCi0gICAgICovCi0gICAgaW50IHR5cGVTaXplOwotCi0gICAgLyoqCi0gICAgICogVGhlIHBvaW50cyBidWZmZXIgc2l6ZS4KLSAgICAgKi8KLSAgICBpbnQgcG9pbnRTaXplOwotCi0gICAgLyoqCi0gICAgICogVGhlIHBhdGggcnVsZS4KLSAgICAgKi8KLSAgICBpbnQgcnVsZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBzcGFjZSBhbW91bnQgaW4gcG9pbnRzIGJ1ZmZlciBmb3IgZGlmZmVyZW50IHNlZ21lbmV0J3MgdHlwZXMuCi0gICAgICovCi0gICAgc3RhdGljIGludCBwb2ludFNoaWZ0W10gPSB7Ci0gICAgICAgICAgICAyLCAvLyBNT1ZFVE8KLSAgICAgICAgICAgIDIsIC8vIExJTkVUTwotICAgICAgICAgICAgNCwgLy8gUVVBRFRPCi0gICAgICAgICAgICA2LCAvLyBDVUJJQ1RPCi0gICAgICAgICAgICAwCi0gICAgfTsgLy8gQ0xPU0UKLQotICAgIC8qCi0gICAgICogR2VuZXJhbFBhdGggcGF0aCBpdGVyYXRvcgotICAgICAqLwotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBJdGVyYXRvciBpcyB0aGUgc3ViY2xhc3Mgb2YgSXRlcmF0b3IgZm9yIHRyYXZlcnNpbmcgdGhlIG91dGxpbmUKLSAgICAgKiBvZiBhIEdlbmVyYWxQYXRoLgotICAgICAqLwotICAgIGNsYXNzIEl0ZXJhdG9yIGltcGxlbWVudHMgUGF0aEl0ZXJhdG9yIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGN1cnJlbnQgY3Vyc29yIHBvc2l0aW9uIGluIHR5cGVzIGJ1ZmZlci4KLSAgICAgICAgICovCi0gICAgICAgIGludCB0eXBlSW5kZXg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjdXJyZW50IGN1cnNvciBwb3NpdGlvbiBpbiBwb2ludHMgYnVmZmVyLgotICAgICAgICAgKi8KLSAgICAgICAgaW50IHBvaW50SW5kZXg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzb3VyY2UgR2VuZXJhbFBhdGggb2JqZWN0LgotICAgICAgICAgKi8KLSAgICAgICAgR2VuZXJhbFBhdGggcDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHBhdGggaXRlcmF0b3IgdHJhbnNmb3JtYXRpb24uCi0gICAgICAgICAqLwotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gdDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogQ29uc3RydWN0cyBhIG5ldyBHZW5lcmFsUGF0aC5JdGVyYXRvciBmb3IgZ2l2ZW4gZ2VuZXJhbCBwYXRoLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHBhdGgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgc291cmNlIEdlbmVyYWxQYXRoIG9iamVjdC4KLSAgICAgICAgICovCi0gICAgICAgIEl0ZXJhdG9yKEdlbmVyYWxQYXRoIHBhdGgpIHsKLSAgICAgICAgICAgIHRoaXMocGF0aCwgbnVsbCk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ29uc3RydWN0cyBhIG5ldyBHZW5lcmFsUGF0aC5JdGVyYXRvciBmb3IgZ2l2ZW4gZ2VuZXJhbCBwYXRoIGFuZAotICAgICAgICAgKiB0cmFuc2Zvcm1hdGlvbi4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBwYXRoCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBHZW5lcmFsUGF0aCBvYmplY3QuCi0gICAgICAgICAqIEBwYXJhbSBhdAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBBZmZpbmVUcmFuc2Zvcm0gb2JqZWN0IHRvIGFwcGx5IHJlY3RhbmdsZSBwYXRoLgotICAgICAgICAgKi8KLSAgICAgICAgSXRlcmF0b3IoR2VuZXJhbFBhdGggcGF0aCwgQWZmaW5lVHJhbnNmb3JtIGF0KSB7Ci0gICAgICAgICAgICB0aGlzLnAgPSBwYXRoOwotICAgICAgICAgICAgdGhpcy50ID0gYXQ7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGdldFdpbmRpbmdSdWxlKCkgewotICAgICAgICAgICAgcmV0dXJuIHAuZ2V0V2luZGluZ1J1bGUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzRG9uZSgpIHsKLSAgICAgICAgICAgIHJldHVybiB0eXBlSW5kZXggPj0gcC50eXBlU2l6ZTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyB2b2lkIG5leHQoKSB7Ci0gICAgICAgICAgICB0eXBlSW5kZXgrKzsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgY3VycmVudFNlZ21lbnQoZG91YmxlW10gY29vcmRzKSB7Ci0gICAgICAgICAgICBpZiAoaXNEb25lKCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuNEI9SXRlcmF0b3Igb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBOb1N1Y2hFbGVtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbnQgdHlwZSA9IHAudHlwZXNbdHlwZUluZGV4XTsKLSAgICAgICAgICAgIGludCBjb3VudCA9IEdlbmVyYWxQYXRoLnBvaW50U2hpZnRbdHlwZV07Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGNvdW50OyBpKyspIHsKLSAgICAgICAgICAgICAgICBjb29yZHNbaV0gPSBwLnBvaW50c1twb2ludEluZGV4ICsgaV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAodCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdC50cmFuc2Zvcm0oY29vcmRzLCAwLCBjb29yZHMsIDAsIGNvdW50IC8gMik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwb2ludEluZGV4ICs9IGNvdW50OwotICAgICAgICAgICAgcmV0dXJuIHR5cGU7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGZsb2F0W10gY29vcmRzKSB7Ci0gICAgICAgICAgICBpZiAoaXNEb25lKCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuNEI9SXRlcmF0b3Igb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBOb1N1Y2hFbGVtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbnQgdHlwZSA9IHAudHlwZXNbdHlwZUluZGV4XTsKLSAgICAgICAgICAgIGludCBjb3VudCA9IEdlbmVyYWxQYXRoLnBvaW50U2hpZnRbdHlwZV07Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHAucG9pbnRzLCBwb2ludEluZGV4LCBjb29yZHMsIDAsIGNvdW50KTsKLSAgICAgICAgICAgIGlmICh0ICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICB0LnRyYW5zZm9ybShjb29yZHMsIDAsIGNvb3JkcywgMCwgY291bnQgLyAyKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHBvaW50SW5kZXggKz0gY291bnQ7Ci0gICAgICAgICAgICByZXR1cm4gdHlwZTsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGdlbmVyYWwgcGF0aCB3aXRoIHRoZSB3aW5kaW5nIHJ1bGUgc2V0IHRvCi0gICAgICoge0BsaW5rIFBhdGhJdGVyYXRvciNXSU5EX05PTl9aRVJPfSBhbmQgdGhlIGluaXRpYWwgY2FwYWNpdHkgKG51bWJlciBvZgotICAgICAqIHNlZ21lbnRzKSBzZXQgdG8gdGhlIGRlZmF1bHQgdmFsdWUgMTAuCi0gICAgICovCi0gICAgcHVibGljIEdlbmVyYWxQYXRoKCkgewotICAgICAgICB0aGlzKFdJTkRfTk9OX1pFUk8sIEJVRkZFUl9TSVpFKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZ2VuZXJhbCBwYXRoIHdpdGggdGhlIGdpdmVuIHdpbmRpbmcgcnVsZSBhbmQgdGhlCi0gICAgICogaW5pdGlhbCBjYXBhY2l0eSAobnVtYmVyIG9mIHNlZ21lbnRzKSBzZXQgdG8gdGhlIGRlZmF1bHQgdmFsdWUgMTAuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJ1bGUKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aW5kaW5nIHJ1bGUsIGVpdGhlciB7QGxpbmsgUGF0aEl0ZXJhdG9yI1dJTkRfRVZFTl9PRER9IG9yCi0gICAgICogICAgICAgICAgICB7QGxpbmsgUGF0aEl0ZXJhdG9yI1dJTkRfTk9OX1pFUk99LgotICAgICAqLwotICAgIHB1YmxpYyBHZW5lcmFsUGF0aChpbnQgcnVsZSkgewotICAgICAgICB0aGlzKHJ1bGUsIEJVRkZFUl9TSVpFKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZ2VuZXJhbCBwYXRoIHdpdGggdGhlIGdpdmVuIHdpbmRpbmcgcnVsZSBhbmQgaW5pdGlhbAotICAgICAqIGNhcGFjaXR5IChudW1iZXIgb2Ygc2VnbWVudHMpLgotICAgICAqIAotICAgICAqIEBwYXJhbSBydWxlCi0gICAgICogICAgICAgICAgICB0aGUgd2luZGluZyBydWxlLCBlaXRoZXIge0BsaW5rIFBhdGhJdGVyYXRvciNXSU5EX0VWRU5fT0REfSBvcgotICAgICAqICAgICAgICAgICAge0BsaW5rIFBhdGhJdGVyYXRvciNXSU5EX05PTl9aRVJPfS4KLSAgICAgKiBAcGFyYW0gaW5pdGlhbENhcGFjaXR5Ci0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIHNlZ21lbnRzIHRoZSBwYXRoIGlzIHNldCB0byBob2xkLgotICAgICAqLwotICAgIHB1YmxpYyBHZW5lcmFsUGF0aChpbnQgcnVsZSwgaW50IGluaXRpYWxDYXBhY2l0eSkgewotICAgICAgICBzZXRXaW5kaW5nUnVsZShydWxlKTsKLSAgICAgICAgdHlwZXMgPSBuZXcgYnl0ZVtpbml0aWFsQ2FwYWNpdHldOwotICAgICAgICBwb2ludHMgPSBuZXcgZmxvYXRbaW5pdGlhbENhcGFjaXR5ICogMl07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIG5ldyBHZW5lcmFsUGF0aCBmcm9tIHRoZSBvdXRsaW5lIG9mIHRoZSBnaXZlbiBzaGFwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2hhcGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzaGFwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgR2VuZXJhbFBhdGgoU2hhcGUgc2hhcGUpIHsKLSAgICAgICAgdGhpcyhXSU5EX05PTl9aRVJPLCBCVUZGRVJfU0laRSk7Ci0gICAgICAgIFBhdGhJdGVyYXRvciBwID0gc2hhcGUuZ2V0UGF0aEl0ZXJhdG9yKG51bGwpOwotICAgICAgICBzZXRXaW5kaW5nUnVsZShwLmdldFdpbmRpbmdSdWxlKCkpOwotICAgICAgICBhcHBlbmQocCwgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHdpbmRpbmcgcnVsZSwgd2hpY2ggZGV0ZXJtaW5lcyBob3cgdG8gZGVjaWRlIHdoZXRoZXIgYSBwb2ludAotICAgICAqIHRoYXQgaXNuJ3Qgb24gdGhlIHBhdGggaXRzZWxmIGlzIGluc2lkZSBvciBvdXRzaWRlIG9mIHRoZSBzaGFwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcnVsZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyB3aW5kaW5nIHJ1bGUuCi0gICAgICogQHRocm93cyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgd2luZGluZyBydWxlIGlzIG5laXRoZXIKLSAgICAgKiAgICAgICAgICAgICB7QGxpbmsgUGF0aEl0ZXJhdG9yI1dJTkRfRVZFTl9PRER9IG5vcgotICAgICAqICAgICAgICAgICAgIHtAbGluayBQYXRoSXRlcmF0b3IjV0lORF9OT05fWkVST30uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0V2luZGluZ1J1bGUoaW50IHJ1bGUpIHsKLSAgICAgICAgaWYgKHJ1bGUgIT0gV0lORF9FVkVOX09ERCAmJiBydWxlICE9IFdJTkRfTk9OX1pFUk8pIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMDk9SW52YWxpZCB3aW5kaW5nIHJ1bGUgdmFsdWUKLSAgICAgICAgICAgIHRocm93IG5ldyBqYXZhLmxhbmcuSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIwOSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHRoaXMucnVsZSA9IHJ1bGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgd2luZGluZyBydWxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHdpbmRpbmcgcnVsZSwgZWl0aGVyIHtAbGluayBQYXRoSXRlcmF0b3IjV0lORF9FVkVOX09ERH0gb3IKLSAgICAgKiAgICAgICAgIHtAbGluayBQYXRoSXRlcmF0b3IjV0lORF9OT05fWkVST30uCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRXaW5kaW5nUnVsZSgpIHsKLSAgICAgICAgcmV0dXJuIHJ1bGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHRoZSBwb2ludCBkYXRhIGJ1ZmZlciBzaXplcyB0byBzZWUgd2hldGhlciBwb2ludENvdW50IGFkZGl0aW9uYWwKLSAgICAgKiBwb2ludC1kYXRhIGVsZW1lbnRzIGNhbiBmaXQuIChOb3RlIHRoYXQgdGhlIG51bWJlciBvZiBwb2ludCBkYXRhIGVsZW1lbnRzCi0gICAgICogdG8gYWRkIGlzIG1vcmUgdGhhbiBvbmUgcGVyIHBvaW50IC0tIGl0IGRlcGVuZHMgb24gdGhlIHR5cGUgb2YgcG9pbnQKLSAgICAgKiBiZWluZyBhZGRlZC4pIFJlYWxsb2NhdGVzIHRoZSBidWZmZXJzIHRvIGVubGFyZ2UgdGhlIHNpemUgaWYgbmVjZXNzYXJ5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwb2ludENvdW50Ci0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIHBvaW50IGRhdGEgZWxlbWVudHMgdG8gYmUgYWRkZWQuCi0gICAgICogQHBhcmFtIGNoZWNrTW92ZQotICAgICAqICAgICAgICAgICAgd2hldGhlciB0byBjaGVjayBmb3IgZXhpc3RpbmcgcG9pbnRzLgotICAgICAqIEB0aHJvd3MgSWxsZWdhbFBhdGhTdGF0ZUV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGNoZWNrTW92ZSBpcyB0cnVlIGFuZCB0aGUgcGF0aCBpcyBjdXJyZW50bHkgZW1wdHkuCi0gICAgICovCi0gICAgdm9pZCBjaGVja0J1ZihpbnQgcG9pbnRDb3VudCwgYm9vbGVhbiBjaGVja01vdmUpIHsKLSAgICAgICAgaWYgKGNoZWNrTW92ZSAmJiB0eXBlU2l6ZSA9PSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjBBPUZpcnN0IHNlZ21lbnQgc2hvdWxkIGJlIFNFR19NT1ZFVE8gdHlwZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxQYXRoU3RhdGVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjBBIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKHR5cGVTaXplID09IHR5cGVzLmxlbmd0aCkgewotICAgICAgICAgICAgYnl0ZSB0bXBbXSA9IG5ldyBieXRlW3R5cGVTaXplICsgQlVGRkVSX0NBUEFDSVRZXTsKLSAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkodHlwZXMsIDAsIHRtcCwgMCwgdHlwZVNpemUpOwotICAgICAgICAgICAgdHlwZXMgPSB0bXA7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHBvaW50U2l6ZSArIHBvaW50Q291bnQgPiBwb2ludHMubGVuZ3RoKSB7Ci0gICAgICAgICAgICBmbG9hdCB0bXBbXSA9IG5ldyBmbG9hdFtwb2ludFNpemUgKyBNYXRoLm1heChCVUZGRVJfQ0FQQUNJVFkgKiAyLCBwb2ludENvdW50KV07Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHBvaW50cywgMCwgdG1wLCAwLCBwb2ludFNpemUpOwotICAgICAgICAgICAgcG9pbnRzID0gdG1wOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXBwZW5kcyBhIG5ldyBwb2ludCB0byB0aGUgZW5kIG9mIHRoaXMgZ2VuZXJhbCBwYXRoLCBkaXNjb25uZWN0ZWQgZnJvbQotICAgICAqIHRoZSBleGlzdGluZyBwYXRoLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBuZXh0IHBvaW50IHRvIGFwcGVuZC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgbmV4dCBwb2ludCB0byBhcHBlbmQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgbW92ZVRvKGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgaWYgKHR5cGVTaXplID4gMCAmJiB0eXBlc1t0eXBlU2l6ZSAtIDFdID09IFBhdGhJdGVyYXRvci5TRUdfTU9WRVRPKSB7Ci0gICAgICAgICAgICBwb2ludHNbcG9pbnRTaXplIC0gMl0gPSB4OwotICAgICAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSAtIDFdID0geTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNoZWNrQnVmKDIsIGZhbHNlKTsKLSAgICAgICAgICAgIHR5cGVzW3R5cGVTaXplKytdID0gUGF0aEl0ZXJhdG9yLlNFR19NT1ZFVE87Ci0gICAgICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0geDsKLSAgICAgICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSB5OwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXBwZW5kcyBhIG5ldyBzZWdtZW50IHRvIHRoZSBlbmQgb2YgdGhpcyBnZW5lcmFsIHBhdGggYnkgbWFraW5nIGEKLSAgICAgKiBzdHJhaWdodCBsaW5lIHNlZ21lbnQgZnJvbSB0aGUgY3VycmVudCBlbmRwb2ludCB0byB0aGUgZ2l2ZW4gbmV3IHBvaW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBuZXh0IHBvaW50IHRvIGFwcGVuZC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgbmV4dCBwb2ludCB0byBhcHBlbmQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgbGluZVRvKGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgY2hlY2tCdWYoMiwgdHJ1ZSk7Ci0gICAgICAgIHR5cGVzW3R5cGVTaXplKytdID0gUGF0aEl0ZXJhdG9yLlNFR19MSU5FVE87Ci0gICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSB4OwotICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0geTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBcHBlbmRzIGEgbmV3IHNlZ21lbnQgdG8gdGhlIGVuZCBvZiB0aGlzIGdlbmVyYWwgcGF0aCBieSBtYWtpbmcgYQotICAgICAqIHF1YWRyYXRpYyBjdXJ2ZSBmcm9tIHRoZSBjdXJyZW50IGVuZHBvaW50IHRvIHRoZSBwb2ludCAoeDIsIHkyKSB1c2luZyB0aGUKLSAgICAgKiBwb2ludCAoeDEsIHkxKSBhcyB0aGUgcXVhZHJhdGljIGN1cnZlJ3MgY29udHJvbCBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHF1YWRyYXRpYyBjdXJ2ZSdzIGNvbnRyb2wgcG9pbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBxdWFkcmF0aWMgY3VydmUncyBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgcXVhZHJhdGljIGN1cnZlJ3MgZW5kIHBvaW50LgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgcXVhZHJhdGljIGN1cnZlJ3MgZW5kIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHF1YWRUbyhmbG9hdCB4MSwgZmxvYXQgeTEsIGZsb2F0IHgyLCBmbG9hdCB5MikgewotICAgICAgICBjaGVja0J1Zig0LCB0cnVlKTsKLSAgICAgICAgdHlwZXNbdHlwZVNpemUrK10gPSBQYXRoSXRlcmF0b3IuU0VHX1FVQURUTzsKLSAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSsrXSA9IHgxOwotICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0geTE7Ci0gICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSB4MjsKLSAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSsrXSA9IHkyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFwcGVuZHMgYSBuZXcgc2VnbWVudCB0byB0aGUgZW5kIG9mIHRoaXMgZ2VuZXJhbCBwYXRoIGJ5IG1ha2luZyBhIGN1YmljCi0gICAgICogY3VydmUgZnJvbSB0aGUgY3VycmVudCBlbmRwb2ludCB0byB0aGUgcG9pbnQgKHgzLCB5MykgdXNpbmcgKHgxLCB5MSkgYW5kCi0gICAgICogKHgyLCB5MikgYXMgY29udHJvbCBwb2ludHMuCi0gICAgICogCi0gICAgICogQHNlZSBqYXZhLmF3dC5nZW9tLkN1YmljQ3VydmUyRAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgbmV3IGN1YmljIHNlZ21lbnQncyBmaXJzdCBjb250cm9sCi0gICAgICogICAgICAgICAgICBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIG5ldyBjdWJpYyBzZWdtZW50J3MgZmlyc3QgY29udHJvbAotICAgICAqICAgICAgICAgICAgcG9pbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBuZXcgY3ViaWMgc2VnbWVudCdzIHNlY29uZCBjb250cm9sCi0gICAgICogICAgICAgICAgICBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIG5ldyBjdWJpYyBzZWdtZW50J3Mgc2Vjb25kIGNvbnRyb2wKLSAgICAgKiAgICAgICAgICAgIHBvaW50LgotICAgICAqIEBwYXJhbSB4MwotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgbmV3IGN1YmljIHNlZ21lbnQncyBlbmQgcG9pbnQuCi0gICAgICogQHBhcmFtIHkzCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBuZXcgY3ViaWMgc2VnbWVudCdzIGVuZCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBjdXJ2ZVRvKGZsb2F0IHgxLCBmbG9hdCB5MSwgZmxvYXQgeDIsIGZsb2F0IHkyLCBmbG9hdCB4MywgZmxvYXQgeTMpIHsKLSAgICAgICAgY2hlY2tCdWYoNiwgdHJ1ZSk7Ci0gICAgICAgIHR5cGVzW3R5cGVTaXplKytdID0gUGF0aEl0ZXJhdG9yLlNFR19DVUJJQ1RPOwotICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0geDE7Ci0gICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSB5MTsKLSAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSsrXSA9IHgyOwotICAgICAgICBwb2ludHNbcG9pbnRTaXplKytdID0geTI7Ci0gICAgICAgIHBvaW50c1twb2ludFNpemUrK10gPSB4MzsKLSAgICAgICAgcG9pbnRzW3BvaW50U2l6ZSsrXSA9IHkzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFwcGVuZHMgdGhlIHR5cGUgaW5mb3JtYXRpb24gdG8gZGVjbGFyZSB0aGF0IHRoZSBjdXJyZW50IGVuZHBvaW50IGNsb3NlcwotICAgICAqIHRoZSBjdXJ2ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBjbG9zZVBhdGgoKSB7Ci0gICAgICAgIGlmICh0eXBlU2l6ZSA9PSAwIHx8IHR5cGVzW3R5cGVTaXplIC0gMV0gIT0gUGF0aEl0ZXJhdG9yLlNFR19DTE9TRSkgewotICAgICAgICAgICAgY2hlY2tCdWYoMCwgdHJ1ZSk7Ci0gICAgICAgICAgICB0eXBlc1t0eXBlU2l6ZSsrXSA9IFBhdGhJdGVyYXRvci5TRUdfQ0xPU0U7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBcHBlbmRzIHRoZSBvdXRsaW5lIG9mIHRoZSBzcGVjaWZpZWQgc2hhcGUgb250byB0aGUgZW5kIG9mIHRoaXMKLSAgICAgKiBHZW5lcmFsUGF0aC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2hhcGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzaGFwZSB3aG9zZSBvdXRsaW5lIGlzIHRvIGJlIGFwcGVuZGVkLgotICAgICAqIEBwYXJhbSBjb25uZWN0Ci0gICAgICogICAgICAgICAgICB0cnVlIHRvIGNvbm5lY3QgdGhpcyBwYXRoJ3MgY3VycmVudCBlbmRwb2ludCB0byB0aGUgZmlyc3QKLSAgICAgKiAgICAgICAgICAgIHBvaW50IG9mIHRoZSBzaGFwZSdzIG91dGxpbmUgb3IgZmFsc2UgdG8gYXBwZW5kIHRoZSBzaGFwZSdzCi0gICAgICogICAgICAgICAgICBvdXRsaW5lIHdpdGhvdXQgY29ubmVjdGluZyBpdC4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHNoYXBlIHBhcmFtZXRlciBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGFwcGVuZChTaGFwZSBzaGFwZSwgYm9vbGVhbiBjb25uZWN0KSB7Ci0gICAgICAgIFBhdGhJdGVyYXRvciBwID0gc2hhcGUuZ2V0UGF0aEl0ZXJhdG9yKG51bGwpOwotICAgICAgICBhcHBlbmQocCwgY29ubmVjdCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXBwZW5kcyB0aGUgcGF0aCBkZWZpbmVkIGJ5IHRoZSBzcGVjaWZpZWQgUGF0aEl0ZXJhdG9yIG9udG8gdGhlIGVuZCBvZgotICAgICAqIHRoaXMgR2VuZXJhbFBhdGguCi0gICAgICogCi0gICAgICogQHBhcmFtIHBhdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBQYXRoSXRlcmF0b3IgdGhhdCBkZWZpbmVzIHRoZSBuZXcgcGF0aCB0byBhcHBlbmQuCi0gICAgICogQHBhcmFtIGNvbm5lY3QKLSAgICAgKiAgICAgICAgICAgIHRydWUgdG8gY29ubmVjdCB0aGlzIHBhdGgncyBjdXJyZW50IGVuZHBvaW50IHRvIHRoZSBmaXJzdAotICAgICAqICAgICAgICAgICAgcG9pbnQgb2YgdGhlIHNoYXBlJ3Mgb3V0bGluZSBvciBmYWxzZSB0byBhcHBlbmQgdGhlIHNoYXBlJ3MKLSAgICAgKiAgICAgICAgICAgIG91dGxpbmUgd2l0aG91dCBjb25uZWN0aW5nIGl0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGFwcGVuZChQYXRoSXRlcmF0b3IgcGF0aCwgYm9vbGVhbiBjb25uZWN0KSB7Ci0gICAgICAgIHdoaWxlICghcGF0aC5pc0RvbmUoKSkgewotICAgICAgICAgICAgZmxvYXQgY29vcmRzW10gPSBuZXcgZmxvYXRbNl07Ci0gICAgICAgICAgICBzd2l0Y2ggKHBhdGguY3VycmVudFNlZ21lbnQoY29vcmRzKSkgewotICAgICAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19NT1ZFVE86Ci0gICAgICAgICAgICAgICAgICAgIGlmICghY29ubmVjdCB8fCB0eXBlU2l6ZSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBtb3ZlVG8oY29vcmRzWzBdLCBjb29yZHNbMV0pOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWYgKHR5cGVzW3R5cGVTaXplIC0gMV0gIT0gUGF0aEl0ZXJhdG9yLlNFR19DTE9TRQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICYmIHBvaW50c1twb2ludFNpemUgLSAyXSA9PSBjb29yZHNbMF0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAmJiBwb2ludHNbcG9pbnRTaXplIC0gMV0gPT0gY29vcmRzWzFdKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAvLyBOTyBCUkVBSzsKLSAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfTElORVRPOgotICAgICAgICAgICAgICAgICAgICBsaW5lVG8oY29vcmRzWzBdLCBjb29yZHNbMV0pOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfUVVBRFRPOgotICAgICAgICAgICAgICAgICAgICBxdWFkVG8oY29vcmRzWzBdLCBjb29yZHNbMV0sIGNvb3Jkc1syXSwgY29vcmRzWzNdKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0NVQklDVE86Ci0gICAgICAgICAgICAgICAgICAgIGN1cnZlVG8oY29vcmRzWzBdLCBjb29yZHNbMV0sIGNvb3Jkc1syXSwgY29vcmRzWzNdLCBjb29yZHNbNF0sIGNvb3Jkc1s1XSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19DTE9TRToKLSAgICAgICAgICAgICAgICAgICAgY2xvc2VQYXRoKCk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcGF0aC5uZXh0KCk7Ci0gICAgICAgICAgICBjb25uZWN0ID0gZmFsc2U7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBjdXJyZW50IGVuZCBwb2ludCBvZiB0aGUgcGF0aC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjdXJyZW50IGVuZCBwb2ludCBvZiB0aGUgcGF0aC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUG9pbnQyRCBnZXRDdXJyZW50UG9pbnQoKSB7Ci0gICAgICAgIGlmICh0eXBlU2l6ZSA9PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotICAgICAgICBpbnQgaiA9IHBvaW50U2l6ZSAtIDI7Ci0gICAgICAgIGlmICh0eXBlc1t0eXBlU2l6ZSAtIDFdID09IFBhdGhJdGVyYXRvci5TRUdfQ0xPU0UpIHsKLQotICAgICAgICAgICAgZm9yIChpbnQgaSA9IHR5cGVTaXplIC0gMjsgaSA+IDA7IGktLSkgewotICAgICAgICAgICAgICAgIGludCB0eXBlID0gdHlwZXNbaV07Ci0gICAgICAgICAgICAgICAgaWYgKHR5cGUgPT0gUGF0aEl0ZXJhdG9yLlNFR19NT1ZFVE8pIHsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGogLT0gcG9pbnRTaGlmdFt0eXBlXTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbmV3IFBvaW50MkQuRmxvYXQocG9pbnRzW2pdLCBwb2ludHNbaiArIDFdKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXNldHMgdGhlIEdlbmVyYWxQYXRoIHRvIGJlaW5nIGFuIGVtcHR5IHBhdGguIFRoZSB1bmRlcmx5aW5nIHBvaW50IGFuZAotICAgICAqIHNlZ21lbnQgZGF0YSBpcyBub3QgZGVsZXRlZCBidXQgcmF0aGVyIHRoZSBlbmQgaW5kaWNlcyBvZiB0aGUgZGF0YSBhcnJheXMKLSAgICAgKiBhcmUgc2V0IHRvIHplcm8uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVzZXQoKSB7Ci0gICAgICAgIHR5cGVTaXplID0gMDsKLSAgICAgICAgcG9pbnRTaXplID0gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUcmFuc2Zvcm0gYWxsIG9mIHRoZSBjb29yZGluYXRlcyBvZiB0aGlzIHBhdGggYWNjb3JkaW5nIHRvIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBBZmZpbmVUcmFuc2Zvcm0uCi0gICAgICogCi0gICAgICogQHBhcmFtIHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBBZmZpbmVUcmFuc2Zvcm0uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgdHJhbnNmb3JtKEFmZmluZVRyYW5zZm9ybSB0KSB7Ci0gICAgICAgIHQudHJhbnNmb3JtKHBvaW50cywgMCwgcG9pbnRzLCAwLCBwb2ludFNpemUgLyAyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgbmV3IEdlbmVyYWxQYXRoIHdob3NlIGRhdGEgaXMgZ2l2ZW4gYnkgdGhpcyBwYXRoJ3MgZGF0YQotICAgICAqIHRyYW5zZm9ybWVkIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkIEFmZmluZVRyYW5zZm9ybS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdAotICAgICAqICAgICAgICAgICAgdGhlIEFmZmluZVRyYW5zZm9ybS4KLSAgICAgKiBAcmV0dXJuIHRoZSBuZXcgR2VuZXJhbFBhdGggd2hvc2UgZGF0YSBpcyBnaXZlbiBieSB0aGlzIHBhdGgncyBkYXRhCi0gICAgICogICAgICAgICB0cmFuc2Zvcm1lZCBhY2NvcmRpbmcgdG8gdGhlIHNwZWNpZmllZCBBZmZpbmVUcmFuc2Zvcm0uCi0gICAgICovCi0gICAgcHVibGljIFNoYXBlIGNyZWF0ZVRyYW5zZm9ybWVkU2hhcGUoQWZmaW5lVHJhbnNmb3JtIHQpIHsKLSAgICAgICAgR2VuZXJhbFBhdGggcCA9IChHZW5lcmFsUGF0aCljbG9uZSgpOwotICAgICAgICBpZiAodCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBwLnRyYW5zZm9ybSh0KTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoKSB7Ci0gICAgICAgIGZsb2F0IHJ4MSwgcnkxLCByeDIsIHJ5MjsKLSAgICAgICAgaWYgKHBvaW50U2l6ZSA9PSAwKSB7Ci0gICAgICAgICAgICByeDEgPSByeTEgPSByeDIgPSByeTIgPSAwLjBmOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaW50IGkgPSBwb2ludFNpemUgLSAxOwotICAgICAgICAgICAgcnkxID0gcnkyID0gcG9pbnRzW2ktLV07Ci0gICAgICAgICAgICByeDEgPSByeDIgPSBwb2ludHNbaS0tXTsKLSAgICAgICAgICAgIHdoaWxlIChpID4gMCkgewotICAgICAgICAgICAgICAgIGZsb2F0IHkgPSBwb2ludHNbaS0tXTsKLSAgICAgICAgICAgICAgICBmbG9hdCB4ID0gcG9pbnRzW2ktLV07Ci0gICAgICAgICAgICAgICAgaWYgKHggPCByeDEpIHsKLSAgICAgICAgICAgICAgICAgICAgcngxID0geDsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKHggPiByeDIpIHsKLSAgICAgICAgICAgICAgICAgICAgcngyID0geDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKHkgPCByeTEpIHsKLSAgICAgICAgICAgICAgICAgICAgcnkxID0geTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKHkgPiByeTIpIHsKLSAgICAgICAgICAgICAgICAgICAgcnkyID0geTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBSZWN0YW5nbGUyRC5GbG9hdChyeDEsIHJ5MSwgcngyIC0gcngxLCByeTIgLSByeTEpOwotICAgIH0KLQotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Qm91bmRzKCkgewotICAgICAgICByZXR1cm4gZ2V0Qm91bmRzMkQoKS5nZXRCb3VuZHMoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgdGhlIGNyb3NzIGNvdW50IChudW1iZXIgb2YgdGltZXMgYSByYXkgZnJvbSB0aGUgcG9pbnQgY3Jvc3NlcyB0aGUKLSAgICAgKiBzaGFwZSdzIGJvdW5kYXJ5KSB0byBkZXRlcm1pbmUgd2hldGhlciB0aGUgbnVtYmVyIG9mIGNyb3NzaW5ncwotICAgICAqIGNvcnJlc3BvbmRzIHRvIGEgcG9pbnQgaW5zaWRlIHRoZSBzaGFwZSBvciBub3QgKGFjY29yZGluZyB0byB0aGUgc2hhcGUncwotICAgICAqIHBhdGggcnVsZSkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNyb3NzCi0gICAgICogICAgICAgICAgICB0aGUgcG9pbnQncyBjcm9zcyBjb3VudC4KLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgdGhlIHBvaW50IGlzIGluc2lkZSB0aGUgcGF0aCwgb3IgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIGJvb2xlYW4gaXNJbnNpZGUoaW50IGNyb3NzKSB7Ci0gICAgICAgIGlmIChydWxlID09IFdJTkRfTk9OX1pFUk8pIHsKLSAgICAgICAgICAgIHJldHVybiBDcm9zc2luZy5pc0luc2lkZU5vblplcm8oY3Jvc3MpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBDcm9zc2luZy5pc0luc2lkZUV2ZW5PZGQoY3Jvc3MpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSBweCwgZG91YmxlIHB5KSB7Ci0gICAgICAgIHJldHVybiBpc0luc2lkZShDcm9zc2luZy5jcm9zc1NoYXBlKHRoaXMsIHB4LCBweSkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSByeCwgZG91YmxlIHJ5LCBkb3VibGUgcncsIGRvdWJsZSByaCkgewotICAgICAgICBpbnQgY3Jvc3MgPSBDcm9zc2luZy5pbnRlcnNlY3RTaGFwZSh0aGlzLCByeCwgcnksIHJ3LCByaCk7Ci0gICAgICAgIHJldHVybiBjcm9zcyAhPSBDcm9zc2luZy5DUk9TU0lORyAmJiBpc0luc2lkZShjcm9zcyk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0cyhkb3VibGUgcngsIGRvdWJsZSByeSwgZG91YmxlIHJ3LCBkb3VibGUgcmgpIHsKLSAgICAgICAgaW50IGNyb3NzID0gQ3Jvc3NpbmcuaW50ZXJzZWN0U2hhcGUodGhpcywgcngsIHJ5LCBydywgcmgpOwotICAgICAgICByZXR1cm4gY3Jvc3MgPT0gQ3Jvc3NpbmcuQ1JPU1NJTkcgfHwgaXNJbnNpZGUoY3Jvc3MpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFBvaW50MkQgcCkgewotICAgICAgICByZXR1cm4gY29udGFpbnMocC5nZXRYKCksIHAuZ2V0WSgpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhSZWN0YW5nbGUyRCByKSB7Ci0gICAgICAgIHJldHVybiBjb250YWlucyhyLmdldFgoKSwgci5nZXRZKCksIHIuZ2V0V2lkdGgoKSwgci5nZXRIZWlnaHQoKSk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0cyhSZWN0YW5nbGUyRCByKSB7Ci0gICAgICAgIHJldHVybiBpbnRlcnNlY3RzKHIuZ2V0WCgpLCByLmdldFkoKSwgci5nZXRXaWR0aCgpLCByLmdldEhlaWdodCgpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gdCkgewotICAgICAgICByZXR1cm4gbmV3IEl0ZXJhdG9yKHRoaXMsIHQpOwotICAgIH0KLQotICAgIHB1YmxpYyBQYXRoSXRlcmF0b3IgZ2V0UGF0aEl0ZXJhdG9yKEFmZmluZVRyYW5zZm9ybSB0LCBkb3VibGUgZmxhdG5lc3MpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBGbGF0dGVuaW5nUGF0aEl0ZXJhdG9yKGdldFBhdGhJdGVyYXRvcih0KSwgZmxhdG5lc3MpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgY2xvbmUoKSB7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBHZW5lcmFsUGF0aCBwID0gKEdlbmVyYWxQYXRoKXN1cGVyLmNsb25lKCk7Ci0gICAgICAgICAgICBwLnR5cGVzID0gdHlwZXMuY2xvbmUoKTsKLSAgICAgICAgICAgIHAucG9pbnRzID0gcG9pbnRzLmNsb25lKCk7Ci0gICAgICAgICAgICByZXR1cm4gcDsKLSAgICAgICAgfSBjYXRjaCAoQ2xvbmVOb3RTdXBwb3J0ZWRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoKTsKLSAgICAgICAgfQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2dlb20vSWxsZWdhbFBhdGhTdGF0ZUV4Y2VwdGlvbi5qYXZhIGIvYXd0L2phdmEvYXd0L2dlb20vSWxsZWdhbFBhdGhTdGF0ZUV4Y2VwdGlvbi5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3NTBiYTI5Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9nZW9tL0lsbGVnYWxQYXRoU3RhdGVFeGNlcHRpb24uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDU1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0Lmdlb207Ci0KLS8qKgotICogVGhlIENsYXNzIElsbGVnYWxQYXRoU3RhdGVFeGNlcHRpb24gaW5kaWNhdGVzIGVycm9ycyB3aGVyZSB0aGUgY3VycmVudCBzdGF0ZQotICogb2YgYSBwYXRoIG9iamVjdCBpcyBpbmNvbXBhdGlibGUgd2l0aCB0aGUgZGVzaXJlZCBhY3Rpb24sIHN1Y2ggYXMgcGVyZm9ybWluZwotICogbm9uLXRyaXZpYWwgYWN0aW9ucyBvbiBhbiBlbXB0eSBwYXRoLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIElsbGVnYWxQYXRoU3RhdGVFeGNlcHRpb24gZXh0ZW5kcyBSdW50aW1lRXhjZXB0aW9uIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC01MTU4MDg0MjA1MjIwNDgxMDk0TDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBpbGxlZ2FsIHBhdGggc3RhdGUgZXhjZXB0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBJbGxlZ2FsUGF0aFN0YXRlRXhjZXB0aW9uKCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBpbGxlZ2FsIHBhdGggc3RhdGUgZXhjZXB0aW9uIHdpdGggdGhlIHNwZWNpZmllZCBkZXRhaWwKLSAgICAgKiBtZXNzYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgZGV0YWlscyBvZiB0aGUgZXJyb3IuCi0gICAgICovCi0gICAgcHVibGljIElsbGVnYWxQYXRoU3RhdGVFeGNlcHRpb24oU3RyaW5nIHMpIHsKLSAgICAgICAgc3VwZXIocyk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZ2VvbS9MaW5lMkQuamF2YSBiL2F3dC9qYXZhL2F3dC9nZW9tL0xpbmUyRC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmY2Q1MWI2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9nZW9tL0xpbmUyRC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsOTQ4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0Lmdlb207Ci0KLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci1pbXBvcnQgamF2YS5hd3QuU2hhcGU7Ci1pbXBvcnQgamF2YS51dGlsLk5vU3VjaEVsZW1lbnRFeGNlcHRpb247Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgTGluZTJEIHJlcHJlc2VudHMgYSBsaW5lIHdob3NlIGRhdGEgaXMgZ2l2ZW4gaW4gaGlnaC1wcmVjaXNpb24KLSAqIHZhbHVlcyBhcHByb3ByaWF0ZSBmb3IgZ3JhcGhpY2FsIG9wZXJhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgTGluZTJEIGltcGxlbWVudHMgU2hhcGUsIENsb25lYWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgRmxvYXQgaXMgdGhlIHN1YmNsYXNzIG9mIExpbmUyRCB0aGF0IGhhcyBhbGwgb2YgaXRzIGRhdGEgdmFsdWVzCi0gICAgICogc3RvcmVkIHdpdGggZmxvYXQtbGV2ZWwgcHJlY2lzaW9uLgotICAgICAqIAotICAgICAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgY2xhc3MgRmxvYXQgZXh0ZW5kcyBMaW5lMkQgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCB4MTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgeTE7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCB4MjsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHkyOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZmxvYXQtdmFsdWVkIExpbmUyRCB3aXRoIGl0cyBkYXRhIHZhbHVlcyBzZXQgdG8KLSAgICAgICAgICogemVyby4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBGbG9hdCgpIHsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZmxvYXQtdmFsdWVkIExpbmUyRCB3aXRoIHRoZSBzcGVjaWZpZWQgZW5kcG9pbnRzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSB5MQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0geDIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSB5MgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBGbG9hdChmbG9hdCB4MSwgZmxvYXQgeTEsIGZsb2F0IHgyLCBmbG9hdCB5MikgewotICAgICAgICAgICAgc2V0TGluZSh4MSwgeTEsIHgyLCB5Mik7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGZsb2F0LXZhbHVlZCBMaW5lMkQgd2l0aCB0aGUgc3BlY2lmaWVkIGVuZHBvaW50cy4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBwMQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIHAyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGVuZCBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBGbG9hdChQb2ludDJEIHAxLCBQb2ludDJEIHAyKSB7Ci0gICAgICAgICAgICBzZXRMaW5lKHAxLCBwMik7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRYMSgpIHsKLSAgICAgICAgICAgIHJldHVybiB4MTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFkxKCkgewotICAgICAgICAgICAgcmV0dXJuIHkxOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WDIoKSB7Ci0gICAgICAgICAgICByZXR1cm4geDI7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRZMigpIHsKLSAgICAgICAgICAgIHJldHVybiB5MjsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUG9pbnQyRCBnZXRQMSgpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUG9pbnQyRC5GbG9hdCh4MSwgeTEpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBQb2ludDJEIGdldFAyKCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBQb2ludDJELkZsb2F0KHgyLCB5Mik7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgc2V0TGluZShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIHgyLCBkb3VibGUgeTIpIHsKLSAgICAgICAgICAgIHRoaXMueDEgPSAoZmxvYXQpeDE7Ci0gICAgICAgICAgICB0aGlzLnkxID0gKGZsb2F0KXkxOwotICAgICAgICAgICAgdGhpcy54MiA9IChmbG9hdCl4MjsKLSAgICAgICAgICAgIHRoaXMueTIgPSAoZmxvYXQpeTI7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogU2V0cyB0aGUgZGF0YSB2YWx1ZXMgdGhhdCBkZWZpbmUgdGhlIGxpbmUuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geDEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIHkxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSB4MgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIHkyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIHZvaWQgc2V0TGluZShmbG9hdCB4MSwgZmxvYXQgeTEsIGZsb2F0IHgyLCBmbG9hdCB5MikgewotICAgICAgICAgICAgdGhpcy54MSA9IHgxOwotICAgICAgICAgICAgdGhpcy55MSA9IHkxOwotICAgICAgICAgICAgdGhpcy54MiA9IHgyOwotICAgICAgICAgICAgdGhpcy55MiA9IHkyOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKCkgewotICAgICAgICAgICAgZmxvYXQgcngsIHJ5LCBydywgcmg7Ci0gICAgICAgICAgICBpZiAoeDEgPCB4MikgewotICAgICAgICAgICAgICAgIHJ4ID0geDE7Ci0gICAgICAgICAgICAgICAgcncgPSB4MiAtIHgxOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICByeCA9IHgyOwotICAgICAgICAgICAgICAgIHJ3ID0geDEgLSB4MjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh5MSA8IHkyKSB7Ci0gICAgICAgICAgICAgICAgcnkgPSB5MTsKLSAgICAgICAgICAgICAgICByaCA9IHkyIC0geTE7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJ5ID0geTI7Ci0gICAgICAgICAgICAgICAgcmggPSB5MSAtIHkyOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIG5ldyBSZWN0YW5nbGUyRC5GbG9hdChyeCwgcnksIHJ3LCByaCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgRG91YmxlIGlzIHRoZSBzdWJjbGFzcyBvZiBMaW5lMkQgdGhhdCBoYXMgYWxsIG9mIGl0cyBkYXRhCi0gICAgICogdmFsdWVzIHN0b3JlZCB3aXRoIGRvdWJsZS1sZXZlbCBwcmVjaXNpb24uCi0gICAgICogCi0gICAgICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBjbGFzcyBEb3VibGUgZXh0ZW5kcyBMaW5lMkQgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgeDE7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB5MTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB4MjsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB5MjsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRvdWJsZS12YWx1ZWQgTGluZTJEIHdpdGggaXRzIGRhdGEgdmFsdWVzIHNldCB0bwotICAgICAgICAgKiB6ZXJvLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIERvdWJsZSgpIHsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZG91YmxlLXZhbHVlZCBMaW5lMkQgd2l0aCB0aGUgc3BlY2lmaWVkIGVuZHBvaW50cy4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSB4MQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0geTEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgICAgICogQHBhcmFtIHgyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0geTIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRG91YmxlKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MikgewotICAgICAgICAgICAgc2V0TGluZSh4MSwgeTEsIHgyLCB5Mik7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRvdWJsZS12YWx1ZWQgTGluZTJEIHdpdGggdGhlIHNwZWNpZmllZCBlbmRwb2ludHMuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gcDEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICAgICAqIEBwYXJhbSBwMgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBlbmQgcG9pbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRG91YmxlKFBvaW50MkQgcDEsIFBvaW50MkQgcDIpIHsKLSAgICAgICAgICAgIHNldExpbmUocDEsIHAyKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFgxKCkgewotICAgICAgICAgICAgcmV0dXJuIHgxOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WTEoKSB7Ci0gICAgICAgICAgICByZXR1cm4geTE7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRYMigpIHsKLSAgICAgICAgICAgIHJldHVybiB4MjsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFkyKCkgewotICAgICAgICAgICAgcmV0dXJuIHkyOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBQb2ludDJEIGdldFAxKCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBQb2ludDJELkRvdWJsZSh4MSwgeTEpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBQb2ludDJEIGdldFAyKCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBQb2ludDJELkRvdWJsZSh4MiwgeTIpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyB2b2lkIHNldExpbmUoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyKSB7Ci0gICAgICAgICAgICB0aGlzLngxID0geDE7Ci0gICAgICAgICAgICB0aGlzLnkxID0geTE7Ci0gICAgICAgICAgICB0aGlzLngyID0geDI7Ci0gICAgICAgICAgICB0aGlzLnkyID0geTI7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoKSB7Ci0gICAgICAgICAgICBkb3VibGUgcngsIHJ5LCBydywgcmg7Ci0gICAgICAgICAgICBpZiAoeDEgPCB4MikgewotICAgICAgICAgICAgICAgIHJ4ID0geDE7Ci0gICAgICAgICAgICAgICAgcncgPSB4MiAtIHgxOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICByeCA9IHgyOwotICAgICAgICAgICAgICAgIHJ3ID0geDEgLSB4MjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh5MSA8IHkyKSB7Ci0gICAgICAgICAgICAgICAgcnkgPSB5MTsKLSAgICAgICAgICAgICAgICByaCA9IHkyIC0geTE7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJ5ID0geTI7Ci0gICAgICAgICAgICAgICAgcmggPSB5MSAtIHkyOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIG5ldyBSZWN0YW5nbGUyRC5Eb3VibGUocngsIHJ5LCBydywgcmgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBMaW5lMkQgcGF0aCBpdGVyYXRvcgotICAgICAqLwotICAgIC8qKgotICAgICAqIFRoZSBzdWJjbGFzcyBvZiBQYXRoSXRlcmF0b3IgdG8gdHJhdmVyc2UgYSBMaW5lMkQuCi0gICAgICovCi0gICAgY2xhc3MgSXRlcmF0b3IgaW1wbGVtZW50cyBQYXRoSXRlcmF0b3IgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydCBsaW5lIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIHgxOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydCBsaW5lIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIHkxOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgbGluZSBwb2ludC4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSB4MjsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIGxpbmUgcG9pbnQuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgeTI7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBwYXRoIGl0ZXJhdG9yIHRyYW5zZm9ybWF0aW9uLgotICAgICAgICAgKi8KLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjdXJyZW50IHNlZ21lbnQgaW5kZXguCi0gICAgICAgICAqLwotICAgICAgICBpbnQgaW5kZXg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENvbnN0cnVjdHMgYSBuZXcgTGluZTJELkl0ZXJhdG9yIGZvciBnaXZlbiBsaW5lIGFuZCB0cmFuc2Zvcm1hdGlvbi4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBsCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBMaW5lMkQgb2JqZWN0LgotICAgICAgICAgKiBAcGFyYW0gYXQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtIG9iamVjdCB0byBhcHBseSByZWN0YW5nbGUgcGF0aC4KLSAgICAgICAgICovCi0gICAgICAgIEl0ZXJhdG9yKExpbmUyRCBsLCBBZmZpbmVUcmFuc2Zvcm0gYXQpIHsKLSAgICAgICAgICAgIHRoaXMueDEgPSBsLmdldFgxKCk7Ci0gICAgICAgICAgICB0aGlzLnkxID0gbC5nZXRZMSgpOwotICAgICAgICAgICAgdGhpcy54MiA9IGwuZ2V0WDIoKTsKLSAgICAgICAgICAgIHRoaXMueTIgPSBsLmdldFkyKCk7Ci0gICAgICAgICAgICB0aGlzLnQgPSBhdDsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgZ2V0V2luZGluZ1J1bGUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gV0lORF9OT05fWkVSTzsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzRG9uZSgpIHsKLSAgICAgICAgICAgIHJldHVybiBpbmRleCA+IDE7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgdm9pZCBuZXh0KCkgewotICAgICAgICAgICAgaW5kZXgrKzsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgY3VycmVudFNlZ21lbnQoZG91YmxlW10gY29vcmRzKSB7Ci0gICAgICAgICAgICBpZiAoaXNEb25lKCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuNEI9SXRlcmF0b3Igb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBOb1N1Y2hFbGVtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbnQgdHlwZTsKLSAgICAgICAgICAgIGlmIChpbmRleCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19NT1ZFVE87Ci0gICAgICAgICAgICAgICAgY29vcmRzWzBdID0geDE7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzFdID0geTE7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTElORVRPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IHgyOwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IHkyOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCAxKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB0eXBlOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGludCBjdXJyZW50U2VnbWVudChmbG9hdFtdIGNvb3JkcykgewotICAgICAgICAgICAgaWYgKGlzRG9uZSgpKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjRCPUl0ZXJhdG9yIG91dCBvZiBib3VuZHMKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgTm9TdWNoRWxlbWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40QiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgaW50IHR5cGU7Ci0gICAgICAgICAgICBpZiAoaW5kZXggPT0gMCkgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTU9WRVRPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IChmbG9hdCl4MTsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSAoZmxvYXQpeTE7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTElORVRPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IChmbG9hdCl4MjsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSAoZmxvYXQpeTI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAodCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdC50cmFuc2Zvcm0oY29vcmRzLCAwLCBjb29yZHMsIDAsIDEpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIHR5cGU7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBMaW5lMkQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIExpbmUyRCgpIHsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZSBnZXRYMSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0WTEoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHgyLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0WDIoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0WTIoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHAgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHAgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBQb2ludDJEIGdldFAxKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBwIGVuZCBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBwIGVuZCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgUG9pbnQyRCBnZXRQMigpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbGluZSdzIGVuZHBvaW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAqIEBwYXJhbSB5MQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0TGluZShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIHgyLCBkb3VibGUgeTIpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbGluZSdzIGVuZHBvaW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcDEKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKiBAcGFyYW0gcDIKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbmQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0TGluZShQb2ludDJEIHAxLCBQb2ludDJEIHAyKSB7Ci0gICAgICAgIHNldExpbmUocDEuZ2V0WCgpLCBwMS5nZXRZKCksIHAyLmdldFgoKSwgcDIuZ2V0WSgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBsaW5lJ3MgZW5kcG9pbnRzIGJ5IGNvcHlpbmcgdGhlIGRhdGEgZnJvbSBhbm90aGVyIExpbmUyRC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbGluZQotICAgICAqICAgICAgICAgICAgdGhlIExpbmUyRCB0byBjb3B5IHRoZSBlbmRwb2ludCBkYXRhIGZyb20uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0TGluZShMaW5lMkQgbGluZSkgewotICAgICAgICBzZXRMaW5lKGxpbmUuZ2V0WDEoKSwgbGluZS5nZXRZMSgpLCBsaW5lLmdldFgyKCksIGxpbmUuZ2V0WTIoKSk7Ci0gICAgfQotCi0gICAgcHVibGljIFJlY3RhbmdsZSBnZXRCb3VuZHMoKSB7Ci0gICAgICAgIHJldHVybiBnZXRCb3VuZHMyRCgpLmdldEJvdW5kcygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRlbGxzIHdoZXJlIHRoZSBwb2ludCBpcyB3aXRoIHJlc3BlY3QgdG8gdGhlIGxpbmUgc2VnbWVudCwgZ2l2ZW4gdGhlCi0gICAgICogb3JpZW50YXRpb24gb2YgdGhlIGxpbmUgc2VnbWVudC4gSWYgdGhlIHJheSBmb3VuZCBieSBleHRlbmRpbmcgdGhlIGxpbmUKLSAgICAgKiBzZWdtZW50IGZyb20gaXRzIHN0YXJ0aW5nIHBvaW50IGlzIHJvdGF0ZWQsIHRoaXMgbWV0aG9kIHRlbGxzIHdoZXRoZXIgdGhlCi0gICAgICogcmF5IHNob3VsZCByb3RhdGUgaW4gYSBjbG9ja3dpc2UgZGlyZWN0aW9uIG9yIGEgY291bnRlci1jbG9ja3dpc2UKLSAgICAgKiBkaXJlY3Rpb24gdG8gaGl0IHRoZSBwb2ludCBmaXJzdC4gVGhlIHJldHVybiB2YWx1ZSBpcyAwIGlmIHRoZSBwb2ludCBpcwotICAgICAqIG9uIHRoZSBsaW5lIHNlZ21lbnQsIGl0J3MgMSBpZiB0aGUgcG9pbnQgaXMgb24gdGhlIHJheSBvciBpZiB0aGUgcmF5Ci0gICAgICogc2hvdWxkIHJvdGF0ZSBpbiBhIGNvdW50ZXItY2xvY2t3aXNlIGRpcmVjdGlvbiB0byBnZXQgdG8gdGhlIHBvaW50LCBhbmQKLSAgICAgKiBpdCdzIC0xIGlmIHRoZSByYXkgc2hvdWxkIHJvdGF0ZSBpbiBhIGNsb2Nrd2lzZSBkaXJlY3Rpb24gdG8gZ2V0IHRvIHRoZQotICAgICAqIHBvaW50IG9yIGlmIHRoZSBwb2ludCBpcyBvbiB0aGUgbGluZSBkZXRlcm1pbmVkIGJ5IHRoZSBsaW5lIHNlZ21lbnQgYnV0Ci0gICAgICogbm90IG9uIHRoZSByYXkgZnJvbSB0aGUgc2VnbWVudCdzIHN0YXJ0aW5nIHBvaW50IGFuZCB0aHJvdWdoIGl0cyBlbmQKLSAgICAgKiBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBsaW5lIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgbGluZSBzZWdtZW50LgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50IG9mIHRoZSBsaW5lIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGxpbmUgc2VnbWVudC4KLSAgICAgKiBAcGFyYW0gcHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHBhcmFtIHB5Ci0gICAgICogICAgICAgICAgICB0aGUgcCBjb29yZGluYXRlIG9mIHRoZSB0ZXN0IHBvaW50LgotICAgICAqIEByZXR1cm4gdGhlIHZhbHVlIHRoYXQgZGVzY3JpYmVzIHdoZXJlIHRoZSBwb2ludCBpcyB3aXRoIHJlc3BlY3QgdG8gdGhlCi0gICAgICogICAgICAgICBsaW5lIHNlZ21lbnQsIGdpdmVuIHRoZSBvcmllbnRhdGlvbiBvZiB0aGUgbGluZSBzZWdtZW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgaW50IHJlbGF0aXZlQ0NXKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MiwgZG91YmxlIHB4LCBkb3VibGUgcHkpIHsKLSAgICAgICAgLyoKLSAgICAgICAgICogQSA9ICh4Mi14MSwgeTIteTEpIFAgPSAocHgteDEsIHB5LXkxKQotICAgICAgICAgKi8KLSAgICAgICAgeDIgLT0geDE7Ci0gICAgICAgIHkyIC09IHkxOwotICAgICAgICBweCAtPSB4MTsKLSAgICAgICAgcHkgLT0geTE7Ci0gICAgICAgIGRvdWJsZSB0ID0gcHggKiB5MiAtIHB5ICogeDI7IC8vIFB4QQotICAgICAgICBpZiAodCA9PSAwLjApIHsKLSAgICAgICAgICAgIHQgPSBweCAqIHgyICsgcHkgKiB5MjsgLy8gUCpBCi0gICAgICAgICAgICBpZiAodCA+IDAuMCkgewotICAgICAgICAgICAgICAgIHB4IC09IHgyOyAvLyBCLUEKLSAgICAgICAgICAgICAgICBweSAtPSB5MjsKLSAgICAgICAgICAgICAgICB0ID0gcHggKiB4MiArIHB5ICogeTI7IC8vIChQLUEpKkEKLSAgICAgICAgICAgICAgICBpZiAodCA8IDAuMCkgewotICAgICAgICAgICAgICAgICAgICB0ID0gMC4wOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiB0IDwgMC4wID8gLTEgOiAodCA+IDAuMCA/IDEgOiAwKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUZWxscyB3aGVyZSB0aGUgcG9pbnQgaXMgd2l0aCByZXNwZWN0IHRvIHRoaXMgbGluZSBzZWdtZW50LCBnaXZlbiB0aGUKLSAgICAgKiBvcmllbnRhdGlvbiBvZiB0aGlzIGxpbmUgc2VnbWVudC4gSWYgdGhlIHJheSBmb3VuZCBieSBleHRlbmRpbmcgdGhlIGxpbmUKLSAgICAgKiBzZWdtZW50IGZyb20gaXRzIHN0YXJ0aW5nIHBvaW50IGlzIHJvdGF0ZWQsIHRoaXMgbWV0aG9kIHRlbGxzIHdoZXRoZXIgdGhlCi0gICAgICogcmF5IHNob3VsZCByb3RhdGUgaW4gYSBjbG9ja3dpc2UgZGlyZWN0aW9uIG9yIGEgY291bnRlci1jbG9ja3dpc2UKLSAgICAgKiBkaXJlY3Rpb24gdG8gaGl0IHRoZSBwb2ludCBmaXJzdC4gVGhlIHJldHVybiB2YWx1ZSBpcyAwIGlmIHRoZSBwb2ludCBpcwotICAgICAqIG9uIHRoZSBsaW5lIHNlZ21lbnQsIGl0J3MgMSBpZiB0aGUgcG9pbnQgaXMgb24gdGhlIHJheSBvciBpZiB0aGUgcmF5Ci0gICAgICogc2hvdWxkIHJvdGF0ZSBpbiBhIGNvdW50ZXItY2xvY2t3aXNlIGRpcmVjdGlvbiB0byBnZXQgdG8gdGhlIHBvaW50LCBhbmQKLSAgICAgKiBpdCdzIC0xIGlmIHRoZSByYXkgc2hvdWxkIHJvdGF0ZSBpbiBhIGNsb2Nrd2lzZSBkaXJlY3Rpb24gdG8gZ2V0IHRvIHRoZQotICAgICAqIHBvaW50IG9yIGlmIHRoZSBwb2ludCBpcyBvbiB0aGUgbGluZSBkZXRlcm1pbmVkIGJ5IHRoZSBsaW5lIHNlZ21lbnQgYnV0Ci0gICAgICogbm90IG9uIHRoZSByYXkgZnJvbSB0aGUgc2VnbWVudCdzIHN0YXJ0aW5nIHBvaW50IGFuZCB0aHJvdWdoIGl0cyBlbmQKLSAgICAgKiBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHBhcmFtIHB5Ci0gICAgICogICAgICAgICAgICB0aGUgcCBjb29yZGluYXRlIG9mIHRoZSB0ZXN0IHBvaW50LgotICAgICAqIEByZXR1cm4gdGhlIHZhbHVlIHRoYXQgZGVzY3JpYmVzIHdoZXJlIHRoZSBwb2ludCBpcyB3aXRoIHJlc3BlY3QgdG8gdGhpcwotICAgICAqICAgICAgICAgbGluZSBzZWdtZW50LCBnaXZlbiB0aGUgb3JpZW50YXRpb24gb2YgdGhpcyBsaW5lIHNlZ21lbnQuCi0gICAgICovCi0gICAgcHVibGljIGludCByZWxhdGl2ZUNDVyhkb3VibGUgcHgsIGRvdWJsZSBweSkgewotICAgICAgICByZXR1cm4gcmVsYXRpdmVDQ1coZ2V0WDEoKSwgZ2V0WTEoKSwgZ2V0WDIoKSwgZ2V0WTIoKSwgcHgsIHB5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUZWxscyB3aGVyZSB0aGUgcG9pbnQgaXMgd2l0aCByZXNwZWN0IHRvIHRoaXMgbGluZSBzZWdtZW50LCBnaXZlbiB0aGUKLSAgICAgKiBvcmllbnRhdGlvbiBvZiB0aGlzIGxpbmUgc2VnbWVudC4gSWYgdGhlIHJheSBmb3VuZCBieSBleHRlbmRpbmcgdGhlIGxpbmUKLSAgICAgKiBzZWdtZW50IGZyb20gaXRzIHN0YXJ0aW5nIHBvaW50IGlzIHJvdGF0ZWQsIHRoaXMgbWV0aG9kIHRlbGxzIHdoZXRoZXIgdGhlCi0gICAgICogcmF5IHNob3VsZCByb3RhdGUgaW4gYSBjbG9ja3dpc2UgZGlyZWN0aW9uIG9yIGEgY291bnRlci1jbG9ja3dpc2UKLSAgICAgKiBkaXJlY3Rpb24gdG8gaGl0IHRoZSBwb2ludCBmaXJzdC4gVGhlIHJldHVybiB2YWx1ZSBpcyAwIGlmIHRoZSBwb2ludCBpcwotICAgICAqIG9uIHRoZSBsaW5lIHNlZ21lbnQsIGl0J3MgMSBpZiB0aGUgcG9pbnQgaXMgb24gdGhlIHJheSBvciBpZiB0aGUgcmF5Ci0gICAgICogc2hvdWxkIHJvdGF0ZSBpbiBhIGNvdW50ZXItY2xvY2t3aXNlIGRpcmVjdGlvbiB0byBnZXQgdG8gdGhlIHBvaW50LCBhbmQKLSAgICAgKiBpdCdzIC0xIGlmIHRoZSByYXkgc2hvdWxkIHJvdGF0ZSBpbiBhIGNsb2Nrd2lzZSBkaXJlY3Rpb24gdG8gZ2V0IHRvIHRoZQotICAgICAqIHBvaW50IG9yIGlmIHRoZSBwb2ludCBpcyBvbiB0aGUgbGluZSBkZXRlcm1pbmVkIGJ5IHRoZSBsaW5lIHNlZ21lbnQgYnV0Ci0gICAgICogbm90IG9uIHRoZSByYXkgZnJvbSB0aGUgc2VnbWVudCdzIHN0YXJ0aW5nIHBvaW50IGFuZCB0aHJvdWdoIGl0cyBlbmQKLSAgICAgKiBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHJldHVybiB0aGUgdmFsdWUgdGhhdCBkZXNjcmliZXMgd2hlcmUgdGhlIHBvaW50IGlzIHdpdGggcmVzcGVjdCB0byB0aGlzCi0gICAgICogICAgICAgICBsaW5lIHNlZ21lbnQsIGdpdmVuIHRoZSBvcmllbnRhdGlvbiBvZiB0aGlzIGxpbmUgc2VnbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHJlbGF0aXZlQ0NXKFBvaW50MkQgcCkgewotICAgICAgICByZXR1cm4gcmVsYXRpdmVDQ1coZ2V0WDEoKSwgZ2V0WTEoKSwgZ2V0WDIoKSwgZ2V0WTIoKSwgcC5nZXRYKCksIHAuZ2V0WSgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUZWxscyB3aGV0aGVyIHRoZSB0d28gbGluZSBzZWdtZW50cyBjcm9zcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBmaXJzdCBzZWdtZW50LgotICAgICAqIEBwYXJhbSB5MQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQgb2YgdGhlIGZpcnN0IHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGZpcnN0IHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGZpcnN0IHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHgzCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgc2Vjb25kIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkzCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgc2Vjb25kIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHg0Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIHNlY29uZCBzZWdtZW50LgotICAgICAqIEBwYXJhbSB5NAotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50IG9mIHRoZSBzZWNvbmQgc2VnbWVudC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSB0d28gbGluZSBzZWdtZW50cyBjcm9zcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGJvb2xlYW4gbGluZXNJbnRlcnNlY3QoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBkb3VibGUgeDMsCi0gICAgICAgICAgICBkb3VibGUgeTMsIGRvdWJsZSB4NCwgZG91YmxlIHk0KSB7Ci0gICAgICAgIC8qCi0gICAgICAgICAqIEEgPSAoeDIteDEsIHkyLXkxKSBCID0gKHgzLXgxLCB5My15MSkgQyA9ICh4NC14MSwgeTQteTEpIEQgPSAoeDQteDMsCi0gICAgICAgICAqIHk0LXkzKSA9IEMtQiBFID0gKHgxLXgzLCB5MS15MykgPSAtQiBGID0gKHgyLXgzLCB5Mi15MykgPSBBLUIgUmVzdWx0Ci0gICAgICAgICAqIGlzICgoQXhCKSAoQXhDKSA8PTApIGFuZCAoKER4RSkgKER4RikgPD0gMCkgRHhFID0gKEMtQil4KC1CKSA9Ci0gICAgICAgICAqIEJ4Qi1DeEIgPSBCeEMgRHhGID0gKEMtQil4KEEtQikgPSBDeEEtQ3hCLUJ4QStCeEIgPSBBeEIrQnhDLUF4QwotICAgICAgICAgKi8KLQotICAgICAgICB4MiAtPSB4MTsgLy8gQQotICAgICAgICB5MiAtPSB5MTsKLSAgICAgICAgeDMgLT0geDE7IC8vIEIKLSAgICAgICAgeTMgLT0geTE7Ci0gICAgICAgIHg0IC09IHgxOyAvLyBDCi0gICAgICAgIHk0IC09IHkxOwotCi0gICAgICAgIGRvdWJsZSBBdkIgPSB4MiAqIHkzIC0geDMgKiB5MjsKLSAgICAgICAgZG91YmxlIEF2QyA9IHgyICogeTQgLSB4NCAqIHkyOwotCi0gICAgICAgIC8vIE9ubGluZQotICAgICAgICBpZiAoQXZCID09IDAuMCAmJiBBdkMgPT0gMC4wKSB7Ci0gICAgICAgICAgICBpZiAoeDIgIT0gMC4wKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuICh4NCAqIHgzIDw9IDAuMCkKLSAgICAgICAgICAgICAgICAgICAgICAgIHx8ICgoeDMgKiB4MiA+PSAwLjApICYmICh4MiA+IDAuMCA/IHgzIDw9IHgyIHx8IHg0IDw9IHgyIDogeDMgPj0geDIKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgeDQgPj0geDIpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh5MiAhPSAwLjApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gKHk0ICogeTMgPD0gMC4wKQotICAgICAgICAgICAgICAgICAgICAgICAgfHwgKCh5MyAqIHkyID49IDAuMCkgJiYgKHkyID4gMC4wID8geTMgPD0geTIgfHwgeTQgPD0geTIgOiB5MyA+PSB5MgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCB5NCA+PSB5MikpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIEJ2QyA9IHgzICogeTQgLSB4NCAqIHkzOwotCi0gICAgICAgIHJldHVybiAoQXZCICogQXZDIDw9IDAuMCkgJiYgKEJ2QyAqIChBdkIgKyBCdkMgLSBBdkMpIDw9IDAuMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGVsbHMgd2hldGhlciB0aGUgc3BlY2lmaWVkIGxpbmUgc2VnbWVudHMgY3Jvc3NlcyB0aGlzIGxpbmUgc2VnbWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSB0ZXN0IHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgdGVzdCBzZWdtZW50LgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50IG9mIHRoZSB0ZXN0IHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIHRlc3Qgc2VnbWVudC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgbGluZSBzZWdtZW50cyBjcm9zc2VzIHRoaXMgbGluZSBzZWdtZW50LgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHNMaW5lKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MikgewotICAgICAgICByZXR1cm4gbGluZXNJbnRlcnNlY3QoeDEsIHkxLCB4MiwgeTIsIGdldFgxKCksIGdldFkxKCksIGdldFgyKCksIGdldFkyKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRlbGxzIHdoZXRoZXIgdGhlIHNwZWNpZmllZCBsaW5lIHNlZ21lbnRzIGNyb3NzZXMgdGhpcyBsaW5lIHNlZ21lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSB0ZXN0IHNlZ21lbnQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIGxpbmUgc2VnbWVudHMgY3Jvc3NlcyB0aGlzIGxpbmUgc2VnbWVudC4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgbCBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHNMaW5lKExpbmUyRCBsKSB7Ci0gICAgICAgIHJldHVybiBsaW5lc0ludGVyc2VjdChsLmdldFgxKCksIGwuZ2V0WTEoKSwgbC5nZXRYMigpLCBsLmdldFkyKCksIGdldFgxKCksIGdldFkxKCksCi0gICAgICAgICAgICAgICAgZ2V0WDIoKSwgZ2V0WTIoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2l2ZXMgdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgcG9pbnQgYW5kIHRoZSBsaW5lIHNlZ21lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgbGluZSBzZWdtZW50LgotICAgICAqIEBwYXJhbSB5MQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQgb2YgdGhlIGxpbmUgc2VnbWVudC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludCBvZiB0aGUgbGluZSBzZWdtZW50LgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50IG9mIHRoZSBsaW5lIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSB0ZXN0IHBvaW50LgotICAgICAqIEBwYXJhbSBweQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgdGVzdCBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSB0aGUgc3F1YXJlIG9mIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhlIGxpbmUKLSAgICAgKiAgICAgICAgIHNlZ21lbnQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBkb3VibGUgcHRTZWdEaXN0U3EoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBkb3VibGUgcHgsCi0gICAgICAgICAgICBkb3VibGUgcHkpIHsKLSAgICAgICAgLyoKLSAgICAgICAgICogQSA9ICh4MiAtIHgxLCB5MiAtIHkxKSBQID0gKHB4IC0geDEsIHB5IC0geTEpCi0gICAgICAgICAqLwotICAgICAgICB4MiAtPSB4MTsgLy8gQSA9ICh4MiwgeTIpCi0gICAgICAgIHkyIC09IHkxOwotICAgICAgICBweCAtPSB4MTsgLy8gUCA9IChweCwgcHkpCi0gICAgICAgIHB5IC09IHkxOwotICAgICAgICBkb3VibGUgZGlzdDsKLSAgICAgICAgaWYgKHB4ICogeDIgKyBweSAqIHkyIDw9IDAuMCkgeyAvLyBQKkEKLSAgICAgICAgICAgIGRpc3QgPSBweCAqIHB4ICsgcHkgKiBweTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHB4ID0geDIgLSBweDsgLy8gUCA9IEEgLSBQID0gKHgyIC0gcHgsIHkyIC0gcHkpCi0gICAgICAgICAgICBweSA9IHkyIC0gcHk7Ci0gICAgICAgICAgICBpZiAocHggKiB4MiArIHB5ICogeTIgPD0gMC4wKSB7IC8vIFAqQQotICAgICAgICAgICAgICAgIGRpc3QgPSBweCAqIHB4ICsgcHkgKiBweTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZGlzdCA9IHB4ICogeTIgLSBweSAqIHgyOwotICAgICAgICAgICAgICAgIGRpc3QgPSBkaXN0ICogZGlzdCAvICh4MiAqIHgyICsgeTIgKiB5Mik7IC8vIHB4QS98QXwKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBpZiAoZGlzdCA8IDApIHsKLSAgICAgICAgICAgIGRpc3QgPSAwOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBkaXN0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdpdmVzIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhlIGxpbmUgc2VnbWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBsaW5lIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgbGluZSBzZWdtZW50LgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50IG9mIHRoZSBsaW5lIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGxpbmUgc2VnbWVudC4KLSAgICAgKiBAcGFyYW0gcHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHBhcmFtIHB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB0ZXN0IHBvaW50LgotICAgICAqIEByZXR1cm4gdGhlIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhlIGxpbmUgc2VnbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGRvdWJsZSBwdFNlZ0Rpc3QoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBkb3VibGUgcHgsIGRvdWJsZSBweSkgewotICAgICAgICByZXR1cm4gTWF0aC5zcXJ0KHB0U2VnRGlzdFNxKHgxLCB5MSwgeDIsIHkyLCBweCwgcHkpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHaXZlcyB0aGUgc3F1YXJlIG9mIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhpcyBsaW5lIHNlZ21lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSB0ZXN0IHBvaW50LgotICAgICAqIEBwYXJhbSBweQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgdGVzdCBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSB0aGUgc3F1YXJlIG9mIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhpcyBsaW5lCi0gICAgICogICAgICAgICBzZWdtZW50LgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgcHRTZWdEaXN0U3EoZG91YmxlIHB4LCBkb3VibGUgcHkpIHsKLSAgICAgICAgcmV0dXJuIHB0U2VnRGlzdFNxKGdldFgxKCksIGdldFkxKCksIGdldFgyKCksIGdldFkyKCksIHB4LCBweSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2l2ZXMgdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgcG9pbnQgYW5kIHRoaXMgbGluZSBzZWdtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgdGVzdCBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSBzcXVhcmUgb2YgdGhlIGRpc3RhbmNlIGJldHdlZW4gdGhlIHBvaW50IGFuZCB0aGlzIGxpbmUKLSAgICAgKiAgICAgICAgIHNlZ21lbnQuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBwdFNlZ0Rpc3RTcShQb2ludDJEIHApIHsKLSAgICAgICAgcmV0dXJuIHB0U2VnRGlzdFNxKGdldFgxKCksIGdldFkxKCksIGdldFgyKCksIGdldFkyKCksIHAuZ2V0WCgpLCBwLmdldFkoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2l2ZXMgdGhlIGRpc3RhbmNlIGJldHdlZW4gdGhlIHBvaW50IGFuZCB0aGlzIGxpbmUgc2VnbWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHBhcmFtIHB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB0ZXN0IHBvaW50LgotICAgICAqIEByZXR1cm4gdGhlIGRpc3RhbmNlIGJldHdlZW4gdGhlIHBvaW50IGFuZCB0aGlzIGxpbmUgc2VnbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlIHB0U2VnRGlzdChkb3VibGUgcHgsIGRvdWJsZSBweSkgewotICAgICAgICByZXR1cm4gcHRTZWdEaXN0KGdldFgxKCksIGdldFkxKCksIGdldFgyKCksIGdldFkyKCksIHB4LCBweSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2l2ZXMgdGhlIGRpc3RhbmNlIGJldHdlZW4gdGhlIHBvaW50IGFuZCB0aGlzIGxpbmUgc2VnbWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHJldHVybiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgcG9pbnQgYW5kIHRoaXMgbGluZSBzZWdtZW50LgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgcHRTZWdEaXN0KFBvaW50MkQgcCkgewotICAgICAgICByZXR1cm4gcHRTZWdEaXN0KGdldFgxKCksIGdldFkxKCksIGdldFgyKCksIGdldFkyKCksIHAuZ2V0WCgpLCBwLmdldFkoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2l2ZXMgdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgcG9pbnQgYW5kIHRoZSBsaW5lLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQgb2YgdGhlIGxpbmUgc2VnbWVudC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBsaW5lIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGxpbmUgc2VnbWVudC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludCBvZiB0aGUgbGluZSBzZWdtZW50LgotICAgICAqIEBwYXJhbSBweAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdGVzdCBwb2ludC4KLSAgICAgKiBAcGFyYW0gcHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHJldHVybiB0aGUgc3F1YXJlIG9mIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhlIGxpbmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBkb3VibGUgcHRMaW5lRGlzdFNxKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MiwgZG91YmxlIHB4LAotICAgICAgICAgICAgZG91YmxlIHB5KSB7Ci0gICAgICAgIHgyIC09IHgxOwotICAgICAgICB5MiAtPSB5MTsKLSAgICAgICAgcHggLT0geDE7Ci0gICAgICAgIHB5IC09IHkxOwotICAgICAgICBkb3VibGUgcyA9IHB4ICogeTIgLSBweSAqIHgyOwotICAgICAgICByZXR1cm4gcyAqIHMgLyAoeDIgKiB4MiArIHkyICogeTIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdpdmVzIHRoZSBzcXVhcmUgb2YgdGhlIGRpc3RhbmNlIGJldHdlZW4gdGhlIHBvaW50IGFuZCB0aGUgbGluZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBsaW5lIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgbGluZSBzZWdtZW50LgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50IG9mIHRoZSBsaW5lIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGxpbmUgc2VnbWVudC4KLSAgICAgKiBAcGFyYW0gcHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHBhcmFtIHB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB0ZXN0IHBvaW50LgotICAgICAqIEByZXR1cm4gdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgcG9pbnQgYW5kIHRoZSBsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZG91YmxlIHB0TGluZURpc3QoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBkb3VibGUgcHgsIGRvdWJsZSBweSkgewotICAgICAgICByZXR1cm4gTWF0aC5zcXJ0KHB0TGluZURpc3RTcSh4MSwgeTEsIHgyLCB5MiwgcHgsIHB5KSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2l2ZXMgdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgcG9pbnQgYW5kIHRoZSBsaW5lCi0gICAgICogZGV0ZXJtaW5lZCBieSB0aGlzIExpbmUyRC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHBhcmFtIHB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB0ZXN0IHBvaW50LgotICAgICAqIEByZXR1cm4gdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgcG9pbnQgYW5kIHRoZSBsaW5lCi0gICAgICogICAgICAgICBkZXRlcm1pbmVkIGJ5IHRoaXMgTGluZTJELgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgcHRMaW5lRGlzdFNxKGRvdWJsZSBweCwgZG91YmxlIHB5KSB7Ci0gICAgICAgIHJldHVybiBwdExpbmVEaXN0U3EoZ2V0WDEoKSwgZ2V0WTEoKSwgZ2V0WDIoKSwgZ2V0WTIoKSwgcHgsIHB5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHaXZlcyB0aGUgc3F1YXJlIG9mIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhlIGxpbmUKLSAgICAgKiBkZXRlcm1pbmVkIGJ5IHRoaXMgTGluZTJELgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgdGVzdCBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSBzcXVhcmUgb2YgdGhlIGRpc3RhbmNlIGJldHdlZW4gdGhlIHBvaW50IGFuZCB0aGUgbGluZQotICAgICAqICAgICAgICAgZGV0ZXJtaW5lZCBieSB0aGlzIExpbmUyRC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlIHB0TGluZURpc3RTcShQb2ludDJEIHApIHsKLSAgICAgICAgcmV0dXJuIHB0TGluZURpc3RTcShnZXRYMSgpLCBnZXRZMSgpLCBnZXRYMigpLCBnZXRZMigpLCBwLmdldFgoKSwgcC5nZXRZKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdpdmVzIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhlIGxpbmUgZGV0ZXJtaW5lZCBieSB0aGlzCi0gICAgICogTGluZTJELgotICAgICAqIAotICAgICAqIEBwYXJhbSBweAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdGVzdCBwb2ludC4KLSAgICAgKiBAcGFyYW0gcHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHRlc3QgcG9pbnQuCi0gICAgICogQHJldHVybiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgcG9pbnQgYW5kIHRoZSBsaW5lIGRldGVybWluZWQgYnkgdGhpcwotICAgICAqICAgICAgICAgTGluZTJELgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgcHRMaW5lRGlzdChkb3VibGUgcHgsIGRvdWJsZSBweSkgewotICAgICAgICByZXR1cm4gcHRMaW5lRGlzdChnZXRYMSgpLCBnZXRZMSgpLCBnZXRYMigpLCBnZXRZMigpLCBweCwgcHkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdpdmVzIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhlIGxpbmUgZGV0ZXJtaW5lZCBieSB0aGlzCi0gICAgICogTGluZTJELgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgdGVzdCBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSBwb2ludCBhbmQgdGhlIGxpbmUgZGV0ZXJtaW5lZCBieSB0aGlzCi0gICAgICogICAgICAgICBMaW5lMkQuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBwdExpbmVEaXN0KFBvaW50MkQgcCkgewotICAgICAgICByZXR1cm4gcHRMaW5lRGlzdChnZXRYMSgpLCBnZXRZMSgpLCBnZXRYMigpLCBnZXRZMigpLCBwLmdldFgoKSwgcC5nZXRZKCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSBweCwgZG91YmxlIHB5KSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhQb2ludDJEIHApIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSByeCwgZG91YmxlIHJ5LCBkb3VibGUgcncsIGRvdWJsZSByaCkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0cyhkb3VibGUgcngsIGRvdWJsZSByeSwgZG91YmxlIHJ3LCBkb3VibGUgcmgpIHsKLSAgICAgICAgcmV0dXJuIGludGVyc2VjdHMobmV3IFJlY3RhbmdsZTJELkRvdWJsZShyeCwgcnksIHJ3LCByaCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHMoUmVjdGFuZ2xlMkQgcikgewotICAgICAgICByZXR1cm4gci5pbnRlcnNlY3RzTGluZShnZXRYMSgpLCBnZXRZMSgpLCBnZXRYMigpLCBnZXRZMigpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gYXQpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJdGVyYXRvcih0aGlzLCBhdCk7Ci0gICAgfQotCi0gICAgcHVibGljIFBhdGhJdGVyYXRvciBnZXRQYXRoSXRlcmF0b3IoQWZmaW5lVHJhbnNmb3JtIGF0LCBkb3VibGUgZmxhdG5lc3MpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJdGVyYXRvcih0aGlzLCBhdCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBjbG9uZSgpIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiBzdXBlci5jbG9uZSgpOwotICAgICAgICB9IGNhdGNoIChDbG9uZU5vdFN1cHBvcnRlZEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcigpOwotICAgICAgICB9Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZ2VvbS9Ob25pbnZlcnRpYmxlVHJhbnNmb3JtRXhjZXB0aW9uLmphdmEgYi9hd3QvamF2YS9hd3QvZ2VvbS9Ob25pbnZlcnRpYmxlVHJhbnNmb3JtRXhjZXB0aW9uLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGE0ZTZmMGYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2dlb20vTm9uaW52ZXJ0aWJsZVRyYW5zZm9ybUV4Y2VwdGlvbi5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZ2VvbTsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgTm9uaW52ZXJ0aWJsZVRyYW5zZm9ybUV4Y2VwdGlvbiBpcyB0aGUgZXhjZXB0aW9uIHRoYXQgaXMgdGhyb3duCi0gKiB3aGVuIGFuIGFjdGlvbiByZXF1aXJlcyBpbnZlcnRpbmcgYW4ge0BsaW5rIEFmZmluZVRyYW5zZm9ybX0gdGhhdCBpcyBub3QKLSAqIGludmVydGlibGUgKGhhcyBkZXRlcm1pbmFudCAwKS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBOb25pbnZlcnRpYmxlVHJhbnNmb3JtRXhjZXB0aW9uIGV4dGVuZHMgamF2YS5sYW5nLkV4Y2VwdGlvbiB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSA2MTM3MjI1MjQwNTAzOTkwNDY2TDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBub24taW52ZXJ0aWJsZSB0cmFuc2Zvcm0gZXhjZXB0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgZXJyb3IgbWVzc2FnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgTm9uaW52ZXJ0aWJsZVRyYW5zZm9ybUV4Y2VwdGlvbihTdHJpbmcgcykgewotICAgICAgICBzdXBlcihzKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9nZW9tL1BhdGhJdGVyYXRvci5qYXZhIGIvYXd0L2phdmEvYXd0L2dlb20vUGF0aEl0ZXJhdG9yLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJkMWMwZmYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2dlb20vUGF0aEl0ZXJhdG9yLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNDYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZ2VvbTsKLQotLyoqCi0gKiBUaGUgSW50ZXJmYWNlIFBhdGhJdGVyYXRvciByZXByZXNlbnRzIGFuIGl0ZXJhdG9yIG9iamVjdCB0aGF0IGNhbiBiZSB1c2VkIHRvCi0gKiB0cmF2ZXJzZSB0aGUgb3V0bGluZSBvZiBhIHtAbGluayBqYXZhLmF3dC5TaGFwZX0uIEl0IHJldHVybnMgcG9pbnRzIGFsb25nIHRoZQotICogYm91bmRhcnkgb2YgdGhlIFNoYXBlIHdoaWNoIG1heSBiZSBhY3R1YWwgdmVydGljZXMgKGluIHRoZSBjYXNlIG9mIGEgc2hhcGUKLSAqIG1hZGUgb2YgbGluZSBzZWdtZW50cykgb3IgbWF5IGJlIHBvaW50cyBvbiBhIGN1cnZlZCBzZWdtZW50IHdpdGggdGhlIGRpc3RhbmNlCi0gKiBiZXR3ZWVuIHRoZSBwb2ludHMgZGV0ZXJtaW5lZCBieSBhIGNob3NlbiBmbGF0dGVuaW5nIGZhY3Rvci4KLSAqIDxwPgotICogSWYgdGhlIHNoYXBlIGlzIGNsb3NlZCwgdGhlIG91dGxpbmUgaXMgdHJhdmVyc2VkIGluIHRoZSBjb3VudGVyLWNsb2Nrd2lzZQotICogZGlyZWN0aW9uLiBUaGF0IG1lYW5zIHRoYXQgbW92aW5nIGZvcndhcmQgYWxvbmcgdGhlIGJvdW5kYXJ5IGlzIHRvIHRyYXZlbCBpbgotICogc3VjaCBhIHdheSB0aGF0IHRoZSBpbnRlcmlvciBvZiB0aGUgc2hhcGUgaXMgdG8gdGhlIGxlZnQgb2YgdGhlIG91dGxpbmUgcGF0aAotICogYW5kIHRoZSBleHRlcmlvciBvZiB0aGUgc2hhcGUgaXMgdG8gdGhlIHJpZ2h0IG9mIHRoZSBvdXRsaW5lIHBhdGguIFRoZQotICogaW50ZXJpb3IgYW5kIGV4dGVyaW9yIG9mIHRoZSBzaGFwZSBhcmUgZGV0ZXJtaW5lZCBieSBhIHdpbmRpbmcgcnVsZS4KLSAqIDwvcD4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgUGF0aEl0ZXJhdG9yIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBXSU5EX0VWRU5fT0REIGluZGljYXRlcyB0aGUgd2luZGluZyBydWxlIHRoYXQgc2F5cyB0aGF0IGEKLSAgICAgKiBwb2ludCBpcyBvdXRzaWRlIHRoZSBzaGFwZSBpZiBhbnkgaW5maW5pdGUgcmF5IGZyb20gdGhlIHBvaW50IGNyb3NzZXMgdGhlCi0gICAgICogb3V0bGluZSBvZiB0aGUgc2hhcGUgYW4gZXZlbiBudW1iZXIgb2YgdGltZXMsIG90aGVyd2lzZSBpdCBpcyBpbnNpZGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lORF9FVkVOX09ERCA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgV0lORF9OT05fWkVSTyBpbmRpY2F0ZXMgdGhlIHdpbmRpbmcgcnVsZSB0aGF0IHNheXMgdGhhdCBhCi0gICAgICogcG9pbnQgaXMgaW5zaWRlIHRoZSBzaGFwZSBpZiBldmVyeSBpbmZpbml0ZSByYXkgc3RhcnRpbmcgZnJvbSB0aGF0IHBvaW50Ci0gICAgICogY3Jvc3NlcyB0aGUgb3V0bGluZSBvZiB0aGUgc2hhcGUgYSBub24temVybyBudW1iZXIgb2YgdGltZXMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lORF9OT05fWkVSTyA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgU0VHX01PVkVUTyBpbmRpY2F0ZXMgdGhhdCB0byBmb2xsb3cgdGhlIHNoYXBlJ3Mgb3V0bGluZSBmcm9tCi0gICAgICogdGhlIHByZXZpb3VzIHBvaW50IHRvIHRoZSBjdXJyZW50IHBvaW50LCB0aGUgY3Vyc29yICh0cmF2ZXJzYWwgcG9pbnQpCi0gICAgICogc2hvdWxkIGJlIHBsYWNlZCBkaXJlY3RseSBvbiB0aGUgY3VycmVudCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTRUdfTU9WRVRPID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTRUdfTElORVRPIGluZGljYXRlcyB0aGF0IHRvIGZvbGxvdyB0aGUgc2hhcGUncyBvdXRsaW5lIGZyb20KLSAgICAgKiB0aGUgcHJldmlvdXMgcG9pbnQgdG8gdGhlIGN1cnJlbnQgcG9pbnQsIHRoZSBjdXJzb3IgKHRyYXZlcnNhbCBwb2ludCkKLSAgICAgKiBzaG91bGQgZm9sbG93IGEgc3RyYWlnaHQgbGluZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTRUdfTElORVRPID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTRUdfUVVBRFRPIGluZGljYXRlcyB0aGF0IHRvIGZvbGxvdyB0aGUgc2hhcGUncyBvdXRsaW5lIGZyb20KLSAgICAgKiB0aGUgcHJldmlvdXMgcG9pbnQgdG8gdGhlIGN1cnJlbnQgcG9pbnQsIHRoZSBjdXJzb3IgKHRyYXZlcnNhbCBwb2ludCkKLSAgICAgKiBzaG91bGQgZm9sbG93IGEgcXVhZHJhdGljIGN1cnZlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNFR19RVUFEVE8gPSAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNFR19DVUJJQ1RPIGluZGljYXRlcyB0aGF0IHRvIGZvbGxvdyB0aGUgc2hhcGUncyBvdXRsaW5lCi0gICAgICogZnJvbSB0aGUgcHJldmlvdXMgcG9pbnQgdG8gdGhlIGN1cnJlbnQgcG9pbnQsIHRoZSBjdXJzb3IgKHRyYXZlcnNhbAotICAgICAqIHBvaW50KSBzaG91bGQgZm9sbG93IGEgY3ViaWMgY3VydmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU0VHX0NVQklDVE8gPSAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNFR19DTE9TRSBpbmRpY2F0ZXMgdGhhdCB0aGUgcHJldmlvdXMgcG9pbnQgd2FzIHRoZSBlbmQgb2YKLSAgICAgKiB0aGUgc2hhcGUncyBvdXRsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFNFR19DTE9TRSA9IDQ7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB3aW5kaW5nIHJ1bGUsIGVpdGhlciB7QGxpbmsgUGF0aEl0ZXJhdG9yI1dJTkRfRVZFTl9PRER9IG9yCi0gICAgICoge0BsaW5rIFBhdGhJdGVyYXRvciNXSU5EX05PTl9aRVJPfS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB3aW5kaW5nIHJ1bGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRXaW5kaW5nUnVsZSgpOwotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoaXMgUGF0aEl0ZXJhdG9yIGhhcyBiZWVuIGNvbXBsZXRlbHkgdHJhdmVyc2VkLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBQYXRoSXRlcmF0b3IgaGFzIGJlZW4gY29tcGxldGVseSB0cmF2ZXJzZWQuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNEb25lKCk7Ci0KLSAgICAvKioKLSAgICAgKiBUZWxscyB0aGlzIFBhdGhJdGVyYXRvciB0byBza2lwIHRvIHRoZSBuZXh0IHNlZ21lbnQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgbmV4dCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29vcmRpbmF0ZXMgb2YgdGhlIG5leHQgdmVydGV4IHBvaW50IGFsb25nIHRoZSBzaGFwZSdzIG91dGxpbmUKLSAgICAgKiBhbmQgYSBmbGFnIHRoYXQgaW5kaWNhdGVzIHdoYXQga2luZCBvZiBzZWdtZW50IHRvIHVzZSBpbiBvcmRlciB0byBjb25uZWN0Ci0gICAgICogdGhlIHByZXZpb3VzIHZlcnRleCBwb2ludCB0byB0aGUgY3VycmVudCB2ZXJ0ZXggcG9pbnQgdG8gZm9ybSB0aGUgY3VycmVudAotICAgICAqIHNlZ21lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvb3JkcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IHRoYXQgdGhlIGNvb3JkaW5hdGVzIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnJlbnQKLSAgICAgKiAgICAgICAgICAgIHNlZ21lbnQgYXJlIHdyaXR0ZW4gaW50by4KLSAgICAgKiBAcmV0dXJuIHRoZSBmbGFnIHRoYXQgaW5kaWNhdGVzIGhvdyB0byBmb2xsb3cgdGhlIHNoYXBlJ3Mgb3V0bGluZSBmcm9tCi0gICAgICogICAgICAgICB0aGUgcHJldmlvdXMgcG9pbnQgdG8gdGhlIGN1cnJlbnQgb25lLCBjaG9zZW4gZnJvbSB0aGUgZm9sbG93aW5nCi0gICAgICogICAgICAgICBjb25zdGFudHM6IHtAbGluayBQYXRoSXRlcmF0b3IjU0VHX01PVkVUT30sCi0gICAgICogICAgICAgICB7QGxpbmsgUGF0aEl0ZXJhdG9yI1NFR19MSU5FVE99LCB7QGxpbmsgUGF0aEl0ZXJhdG9yI1NFR19RVUFEVE99LAotICAgICAqICAgICAgICAge0BsaW5rIFBhdGhJdGVyYXRvciNTRUdfQ1VCSUNUT30sIG9yCi0gICAgICogICAgICAgICB7QGxpbmsgUGF0aEl0ZXJhdG9yI1NFR19DTE9TRX0uCi0gICAgICovCi0gICAgcHVibGljIGludCBjdXJyZW50U2VnbWVudChmbG9hdFtdIGNvb3Jkcyk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBjb29yZGluYXRlcyBvZiB0aGUgbmV4dCB2ZXJ0ZXggcG9pbnQgYWxvbmcgdGhlIHNoYXBlJ3Mgb3V0bGluZQotICAgICAqIGFuZCBhIGZsYWcgdGhhdCBpbmRpY2F0ZXMgd2hhdCBraW5kIG9mIHNlZ21lbnQgdG8gdXNlIGluIG9yZGVyIHRvIGNvbm5lY3QKLSAgICAgKiB0aGUgcHJldmlvdXMgdmVydGV4IHBvaW50IHRvIHRoZSBjdXJyZW50IHZlcnRleCBwb2ludCB0byBmb3JtIHRoZSBjdXJyZW50Ci0gICAgICogc2VnbWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29vcmRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgdGhhdCB0aGUgY29vcmRpbmF0ZXMgb2YgdGhlIGVuZCBwb2ludCBvZiB0aGUgY3VycmVudAotICAgICAqICAgICAgICAgICAgc2VnbWVudCBhcmUgd3JpdHRlbiBpbnRvLgotICAgICAqIEByZXR1cm4gdGhlIGZsYWcgdGhhdCBpbmRpY2F0ZXMgaG93IHRvIGZvbGxvdyB0aGUgc2hhcGUncyBvdXRsaW5lIGZyb20KLSAgICAgKiAgICAgICAgIHRoZSBwcmV2aW91cyBwb2ludCB0byB0aGUgY3VycmVudCBvbmUsIGNob3NlbiBmcm9tIHRoZSBmb2xsb3dpbmcKLSAgICAgKiAgICAgICAgIGNvbnN0YW50czoge0BsaW5rIFBhdGhJdGVyYXRvciNTRUdfTU9WRVRPfSwKLSAgICAgKiAgICAgICAgIHtAbGluayBQYXRoSXRlcmF0b3IjU0VHX0xJTkVUT30sIHtAbGluayBQYXRoSXRlcmF0b3IjU0VHX1FVQURUT30sCi0gICAgICogICAgICAgICB7QGxpbmsgUGF0aEl0ZXJhdG9yI1NFR19DVUJJQ1RPfSwgb3IKLSAgICAgKiAgICAgICAgIHtAbGluayBQYXRoSXRlcmF0b3IjU0VHX0NMT1NFfS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGRvdWJsZVtdIGNvb3Jkcyk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9nZW9tL1BvaW50MkQuamF2YSBiL2F3dC9qYXZhL2F3dC9nZW9tL1BvaW50MkQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZjcwMjZjOC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvZ2VvbS9Qb2ludDJELmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzMjMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZ2VvbTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5taXNjLkhhc2hDb2RlOwotCi0vKioKLSAqIFRoZSBDbGFzcyBQb2ludDJEIHJlcHJlc2VudHMgYSBwb2ludCB3aG9zZSBkYXRhIGlzIGdpdmVuIGluIGhpZ2gtcHJlY2lzaW9uCi0gKiB2YWx1ZXMgYXBwcm9wcmlhdGUgZm9yIGdyYXBoaWNhbCBvcGVyYXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIFBvaW50MkQgaW1wbGVtZW50cyBDbG9uZWFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIEZsb2F0IGlzIHRoZSBzdWJjbGFzcyBvZiBQb2ludDJEIHRoYXQgaGFzIGFsbCBvZiBpdHMgZGF0YQotICAgICAqIHZhbHVlcyBzdG9yZWQgd2l0aCBmbG9hdC1sZXZlbCBwcmVjaXNpb24uCi0gICAgICogCi0gICAgICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBjbGFzcyBGbG9hdCBleHRlbmRzIFBvaW50MkQgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgeTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGZsb2F0LXZhbHVlZCBQb2ludDJEIHdpdGggaXRzIGRhdGEgc2V0IHRvIHplcm8uCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRmxvYXQoKSB7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGZsb2F0LXZhbHVlZCBQb2ludDJEIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAgICAgKiBjb29yZGluYXRlcy4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSB4Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZS4KLSAgICAgICAgICogQHBhcmFtIHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEZsb2F0KGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgICAgIHRoaXMueCA9IHg7Ci0gICAgICAgICAgICB0aGlzLnkgPSB5OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WCgpIHsKLSAgICAgICAgICAgIHJldHVybiB4OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WSgpIHsKLSAgICAgICAgICAgIHJldHVybiB5OwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFNldHMgdGhlIHBvaW50J3MgY29vcmRpbmF0ZXMuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUuCi0gICAgICAgICAqIEBwYXJhbSB5Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyB2b2lkIHNldExvY2F0aW9uKGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgICAgIHRoaXMueCA9IHg7Ci0gICAgICAgICAgICB0aGlzLnkgPSB5OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyB2b2lkIHNldExvY2F0aW9uKGRvdWJsZSB4LCBkb3VibGUgeSkgewotICAgICAgICAgICAgdGhpcy54ID0gKGZsb2F0KXg7Ci0gICAgICAgICAgICB0aGlzLnkgPSAoZmxvYXQpeTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICAgICAgcmV0dXJuIGdldENsYXNzKCkuZ2V0TmFtZSgpICsgIlt4PSIgKyB4ICsgIix5PSIgKyB5ICsgIl0iOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIERvdWJsZSBpcyB0aGUgc3ViY2xhc3Mgb2YgUG9pbnQyRCB0aGF0IGhhcyBhbGwgb2YgaXRzIGRhdGEKLSAgICAgKiB2YWx1ZXMgc3RvcmVkIHdpdGggZG91YmxlLWxldmVsIHByZWNpc2lvbi4KLSAgICAgKiAKLSAgICAgKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGNsYXNzIERvdWJsZSBleHRlbmRzIFBvaW50MkQgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB4OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB5OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZG91YmxlLXZhbHVlZCBQb2ludDJEIHdpdGggaXRzIGRhdGEgc2V0IHRvIHplcm8uCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRG91YmxlKCkgewotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkb3VibGUtdmFsdWVkIFBvaW50MkQgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICAgICAqIGNvb3JkaW5hdGVzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlLgotICAgICAgICAgKiBAcGFyYW0geQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRG91YmxlKGRvdWJsZSB4LCBkb3VibGUgeSkgewotICAgICAgICAgICAgdGhpcy54ID0geDsKLSAgICAgICAgICAgIHRoaXMueSA9IHk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRYKCkgewotICAgICAgICAgICAgcmV0dXJuIHg7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRZKCkgewotICAgICAgICAgICAgcmV0dXJuIHk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgc2V0TG9jYXRpb24oZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0gICAgICAgICAgICB0aGlzLnggPSB4OwotICAgICAgICAgICAgdGhpcy55ID0geTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICAgICAgcmV0dXJuIGdldENsYXNzKCkuZ2V0TmFtZSgpICsgIlt4PSIgKyB4ICsgIix5PSIgKyB5ICsgIl0iOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IFBvaW50MkQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFBvaW50MkQoKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeCBjb29yZGluYXRlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHggY29vcmRpbmF0ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldFgoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHkgY29vcmRpbmF0ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB5IGNvb3JkaW5hdGUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZSBnZXRZKCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBwb2ludCdzIGNvb3JkaW5hdGVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldExvY2F0aW9uKGRvdWJsZSB4LCBkb3VibGUgeSk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBwb2ludCdzIGNvb3JkaW5hdGVzIGJ5IGNvcHlpbmcgdGhlbSBmcm9tIGFub3RoZXIgcG9pbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHAKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb2ludCB0byBjb3B5IHRoZSBkYXRhIGZyb20uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0TG9jYXRpb24oUG9pbnQyRCBwKSB7Ci0gICAgICAgIHNldExvY2F0aW9uKHAuZ2V0WCgpLCBwLmdldFkoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRmluZHMgdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgdHdvIHNwZWNpZmllZCBwb2ludHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGZpcnN0IHBvaW50LgotICAgICAqIEBwYXJhbSB4MgotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIHBvaW50LgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIHBvaW50LgotICAgICAqIEByZXR1cm4gdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGUgdHdvIHNwZWNpZmllZCBwb2ludHMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBkb3VibGUgZGlzdGFuY2VTcShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIHgyLCBkb3VibGUgeTIpIHsKLSAgICAgICAgeDIgLT0geDE7Ci0gICAgICAgIHkyIC09IHkxOwotICAgICAgICByZXR1cm4geDIgKiB4MiArIHkyICogeTI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRmluZHMgdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGlzIHBvaW50IGFuZCB0aGUgc3BlY2lmaWVkCi0gICAgICogcG9pbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBwb2ludC4KLSAgICAgKiBAcGFyYW0gcHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHBvaW50LgotICAgICAqIEByZXR1cm4gdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGlzIHBvaW50IGFuZCB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlIGRpc3RhbmNlU3EoZG91YmxlIHB4LCBkb3VibGUgcHkpIHsKLSAgICAgICAgcmV0dXJuIFBvaW50MkQuZGlzdGFuY2VTcShnZXRYKCksIGdldFkoKSwgcHgsIHB5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaW5kcyB0aGUgc3F1YXJlIG9mIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoaXMgcG9pbnQgYW5kIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIG90aGVyIHBvaW50LgotICAgICAqIEByZXR1cm4gdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGlzIHBvaW50IGFuZCB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlIGRpc3RhbmNlU3EoUG9pbnQyRCBwKSB7Ci0gICAgICAgIHJldHVybiBQb2ludDJELmRpc3RhbmNlU3EoZ2V0WCgpLCBnZXRZKCksIHAuZ2V0WCgpLCBwLmdldFkoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRmluZHMgdGhlIGRpc3RhbmNlIGJldHdlZW4gdGhlIHR3byBzcGVjaWZpZWQgcG9pbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgcG9pbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBwb2ludC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBwb2ludC4KLSAgICAgKiBAcGFyYW0geTIKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoZSB0d28gc3BlY2lmaWVkIHBvaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGRvdWJsZSBkaXN0YW5jZShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIHgyLCBkb3VibGUgeTIpIHsKLSAgICAgICAgcmV0dXJuIE1hdGguc3FydChkaXN0YW5jZVNxKHgxLCB5MSwgeDIsIHkyKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRmluZHMgdGhlIGRpc3RhbmNlIGJldHdlZW4gdGhpcyBwb2ludCBhbmQgdGhlIHNwZWNpZmllZCBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHBvaW50LgotICAgICAqIEBwYXJhbSBweQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgcG9pbnQuCi0gICAgICogQHJldHVybiB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0aGlzIHBvaW50IGFuZCB0aGUgc3BlY2lmaWVkIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgZGlzdGFuY2UoZG91YmxlIHB4LCBkb3VibGUgcHkpIHsKLSAgICAgICAgcmV0dXJuIE1hdGguc3FydChkaXN0YW5jZVNxKHB4LCBweSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbmRzIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoaXMgcG9pbnQgYW5kIHRoZSBzcGVjaWZpZWQgcG9pbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHAKLSAgICAgKiAgICAgICAgICAgIHRoZSBvdGhlciBwb2ludC4KLSAgICAgKiBAcmV0dXJuIHRoZSBkaXN0YW5jZSBiZXR3ZWVuIHRoaXMgcG9pbnQgYW5kIHRoZSBzcGVjaWZpZWQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBkaXN0YW5jZShQb2ludDJEIHApIHsKLSAgICAgICAgcmV0dXJuIE1hdGguc3FydChkaXN0YW5jZVNxKHApKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgT2JqZWN0IGNsb25lKCkgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIHN1cGVyLmNsb25lKCk7Ci0gICAgICAgIH0gY2F0Y2ggKENsb25lTm90U3VwcG9ydGVkRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGhhc2hDb2RlKCkgewotICAgICAgICBIYXNoQ29kZSBoYXNoID0gbmV3IEhhc2hDb2RlKCk7Ci0gICAgICAgIGhhc2guYXBwZW5kKGdldFgoKSk7Ci0gICAgICAgIGhhc2guYXBwZW5kKGdldFkoKSk7Ci0gICAgICAgIHJldHVybiBoYXNoLmhhc2hDb2RlKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvYmopIHsKLSAgICAgICAgaWYgKG9iaiA9PSB0aGlzKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgICAgICBpZiAob2JqIGluc3RhbmNlb2YgUG9pbnQyRCkgewotICAgICAgICAgICAgUG9pbnQyRCBwID0gKFBvaW50MkQpb2JqOwotICAgICAgICAgICAgcmV0dXJuIGdldFgoKSA9PSBwLmdldFgoKSAmJiBnZXRZKCkgPT0gcC5nZXRZKCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9nZW9tL1F1YWRDdXJ2ZTJELmphdmEgYi9hd3QvamF2YS9hd3QvZ2VvbS9RdWFkQ3VydmUyRC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3YTg2YTQ4Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9nZW9tL1F1YWRDdXJ2ZTJELmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw5MTggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZ2VvbTsKLQotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLWltcG9ydCBqYXZhLmF3dC5TaGFwZTsKLWltcG9ydCBqYXZhLnV0aWwuTm9TdWNoRWxlbWVudEV4Y2VwdGlvbjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuQ3Jvc3Npbmc7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIENsYXNzIFF1YWRDdXJ2ZTJEIGlzIGEgU2hhcGUgdGhhdCByZXByZXNlbnRzIGEgc2VnbWVudCBvZiBhIHF1YWRyYXRpYwotICogKEJlemllcikgY3VydmUuIFRoZSBjdXJ2ZWQgc2VnbWVudCBpcyBkZXRlcm1pbmVkIGJ5IHRocmVlIHBvaW50czogYSBzdGFydAotICogcG9pbnQsIGFuIGVuZCBwb2ludCwgYW5kIGEgY29udHJvbCBwb2ludC4gVGhlIGxpbmUgZnJvbSB0aGUgY29udHJvbCBwb2ludCB0bwotICogdGhlIHN0YXJ0aW5nIHBvaW50IGdpdmVzIHRoZSB0YW5nZW50IHRvIHRoZSBjdXJ2ZSBhdCB0aGUgc3RhcnRpbmcgcG9pbnQsIGFuZAotICogdGhlIGxpbmUgZnJvbSB0aGUgY29udHJvbCBwb2ludCB0byB0aGUgZW5kIHBvaW50IGdpdmVzIHRoZSB0YW5nZW50IHRvIHRoZQotICogY3VydmUgYXQgdGhlIGVuZCBwb2ludC4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBRdWFkQ3VydmUyRCBpbXBsZW1lbnRzIFNoYXBlLCBDbG9uZWFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIEZsb2F0IGlzIHRoZSBzdWJjbGFzcyBvZiBRdWFkQ3VydmUyRCB0aGF0IGhhcyBhbGwgb2YgaXRzIGRhdGEKLSAgICAgKiB2YWx1ZXMgc3RvcmVkIHdpdGggZmxvYXQtbGV2ZWwgcHJlY2lzaW9uLgotICAgICAqIAotICAgICAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgY2xhc3MgRmxvYXQgZXh0ZW5kcyBRdWFkQ3VydmUyRCB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBjdXJ2ZWQgc2VnbWVudC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCB4MTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHkxOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IGN0cmx4OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IGN0cmx5OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHgyOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHkyOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZmxvYXQtdmFsdWVkIFF1YWRDdXJ2ZTJEIHdpdGggYWxsIGNvb3JkaW5hdGUKLSAgICAgICAgICogdmFsdWVzIHNldCB0byB6ZXJvLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEZsb2F0KCkgewotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBmbG9hdC12YWx1ZWQgUXVhZEN1cnZlMkQgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICAgICAqIGNvb3JkaW5hdGUgdmFsdWVzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQgb2YgdGhlIGN1cnZlZAotICAgICAgICAgKiAgICAgICAgICAgIHNlZ21lbnQuCi0gICAgICAgICAqIEBwYXJhbSB5MQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBjdXJ2ZWQKLSAgICAgICAgICogICAgICAgICAgICBzZWdtZW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0geDIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAgICAgKiBAcGFyYW0geTIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEZsb2F0KGZsb2F0IHgxLCBmbG9hdCB5MSwgZmxvYXQgY3RybHgsIGZsb2F0IGN0cmx5LCBmbG9hdCB4MiwgZmxvYXQgeTIpIHsKLSAgICAgICAgICAgIHNldEN1cnZlKHgxLCB5MSwgY3RybHgsIGN0cmx5LCB4MiwgeTIpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WDEoKSB7Ci0gICAgICAgICAgICByZXR1cm4geDE7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRZMSgpIHsKLSAgICAgICAgICAgIHJldHVybiB5MTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldEN0cmxYKCkgewotICAgICAgICAgICAgcmV0dXJuIGN0cmx4OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0Q3RybFkoKSB7Ci0gICAgICAgICAgICByZXR1cm4gY3RybHk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRYMigpIHsKLSAgICAgICAgICAgIHJldHVybiB4MjsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFkyKCkgewotICAgICAgICAgICAgcmV0dXJuIHkyOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBQb2ludDJEIGdldFAxKCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBQb2ludDJELkZsb2F0KHgxLCB5MSk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFBvaW50MkQgZ2V0Q3RybFB0KCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBQb2ludDJELkZsb2F0KGN0cmx4LCBjdHJseSk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFBvaW50MkQgZ2V0UDIoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFBvaW50MkQuRmxvYXQoeDIsIHkyKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRDdXJ2ZShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIGN0cmx4LCBkb3VibGUgY3RybHksIGRvdWJsZSB4MiwgZG91YmxlIHkyKSB7Ci0gICAgICAgICAgICB0aGlzLngxID0gKGZsb2F0KXgxOwotICAgICAgICAgICAgdGhpcy55MSA9IChmbG9hdCl5MTsKLSAgICAgICAgICAgIHRoaXMuY3RybHggPSAoZmxvYXQpY3RybHg7Ci0gICAgICAgICAgICB0aGlzLmN0cmx5ID0gKGZsb2F0KWN0cmx5OwotICAgICAgICAgICAgdGhpcy54MiA9IChmbG9hdCl4MjsKLSAgICAgICAgICAgIHRoaXMueTIgPSAoZmxvYXQpeTI7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogU2V0cyB0aGUgZGF0YSB2YWx1ZXMgb2YgdGhlIGN1cnZlLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQgb2YgdGhlIGN1cnZlZAotICAgICAgICAgKiAgICAgICAgICAgIHNlZ21lbnQuCi0gICAgICAgICAqIEBwYXJhbSB5MQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBjdXJ2ZWQKLSAgICAgICAgICogICAgICAgICAgICBzZWdtZW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0geDIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAgICAgKiBAcGFyYW0geTIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIHZvaWQgc2V0Q3VydmUoZmxvYXQgeDEsIGZsb2F0IHkxLCBmbG9hdCBjdHJseCwgZmxvYXQgY3RybHksIGZsb2F0IHgyLCBmbG9hdCB5MikgewotICAgICAgICAgICAgdGhpcy54MSA9IHgxOwotICAgICAgICAgICAgdGhpcy55MSA9IHkxOwotICAgICAgICAgICAgdGhpcy5jdHJseCA9IGN0cmx4OwotICAgICAgICAgICAgdGhpcy5jdHJseSA9IGN0cmx5OwotICAgICAgICAgICAgdGhpcy54MiA9IHgyOwotICAgICAgICAgICAgdGhpcy55MiA9IHkyOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKCkgewotICAgICAgICAgICAgZmxvYXQgcngwID0gTWF0aC5taW4oTWF0aC5taW4oeDEsIHgyKSwgY3RybHgpOwotICAgICAgICAgICAgZmxvYXQgcnkwID0gTWF0aC5taW4oTWF0aC5taW4oeTEsIHkyKSwgY3RybHkpOwotICAgICAgICAgICAgZmxvYXQgcngxID0gTWF0aC5tYXgoTWF0aC5tYXgoeDEsIHgyKSwgY3RybHgpOwotICAgICAgICAgICAgZmxvYXQgcnkxID0gTWF0aC5tYXgoTWF0aC5tYXgoeTEsIHkyKSwgY3RybHkpOwotICAgICAgICAgICAgcmV0dXJuIG5ldyBSZWN0YW5nbGUyRC5GbG9hdChyeDAsIHJ5MCwgcngxIC0gcngwLCByeTEgLSByeTApOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIERvdWJsZSBpcyB0aGUgc3ViY2xhc3Mgb2YgUXVhZEN1cnZlMkQgdGhhdCBoYXMgYWxsIG9mIGl0cyBkYXRhCi0gICAgICogdmFsdWVzIHN0b3JlZCB3aXRoIGRvdWJsZS1sZXZlbCBwcmVjaXNpb24uCi0gICAgICogCi0gICAgICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBjbGFzcyBEb3VibGUgZXh0ZW5kcyBRdWFkQ3VydmUyRCB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBjdXJ2ZWQgc2VnbWVudC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgeDE7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBjdXJ2ZWQgc2VnbWVudC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgeTE7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGNvbnRyb2wgcG9pbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIGN0cmx4OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeSBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSBjdHJseTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgZW5kIHBvaW50IG9mIHRoZSBjdXJ2ZWQgc2VnbWVudC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgeDI7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIHkyOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZG91YmxlLXZhbHVlZCBRdWFkQ3VydmUyRCB3aXRoIGFsbCBjb29yZGluYXRlCi0gICAgICAgICAqIHZhbHVlcyBzZXQgdG8gemVyby4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBEb3VibGUoKSB7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRvdWJsZS12YWx1ZWQgUXVhZEN1cnZlMkQgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICAgICAqIGNvb3JkaW5hdGUgdmFsdWVzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgxCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQgb2YgdGhlIGN1cnZlZAotICAgICAgICAgKiAgICAgICAgICAgIHNlZ21lbnQuCi0gICAgICAgICAqIEBwYXJhbSB5MQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBjdXJ2ZWQKLSAgICAgICAgICogICAgICAgICAgICBzZWdtZW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0gY3RybHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAgICAgKiBAcGFyYW0geDIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAgICAgKiBAcGFyYW0geTIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIERvdWJsZShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIGN0cmx4LCBkb3VibGUgY3RybHksIGRvdWJsZSB4MiwgZG91YmxlIHkyKSB7Ci0gICAgICAgICAgICBzZXRDdXJ2ZSh4MSwgeTEsIGN0cmx4LCBjdHJseSwgeDIsIHkyKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFgxKCkgewotICAgICAgICAgICAgcmV0dXJuIHgxOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WTEoKSB7Ci0gICAgICAgICAgICByZXR1cm4geTE7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRDdHJsWCgpIHsKLSAgICAgICAgICAgIHJldHVybiBjdHJseDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldEN0cmxZKCkgewotICAgICAgICAgICAgcmV0dXJuIGN0cmx5OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WDIoKSB7Ci0gICAgICAgICAgICByZXR1cm4geDI7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRZMigpIHsKLSAgICAgICAgICAgIHJldHVybiB5MjsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUG9pbnQyRCBnZXRQMSgpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUG9pbnQyRC5Eb3VibGUoeDEsIHkxKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUG9pbnQyRCBnZXRDdHJsUHQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFBvaW50MkQuRG91YmxlKGN0cmx4LCBjdHJseSk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFBvaW50MkQgZ2V0UDIoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFBvaW50MkQuRG91YmxlKHgyLCB5Mik7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgc2V0Q3VydmUoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSBjdHJseCwgZG91YmxlIGN0cmx5LCBkb3VibGUgeDIsIGRvdWJsZSB5MikgewotICAgICAgICAgICAgdGhpcy54MSA9IHgxOwotICAgICAgICAgICAgdGhpcy55MSA9IHkxOwotICAgICAgICAgICAgdGhpcy5jdHJseCA9IGN0cmx4OwotICAgICAgICAgICAgdGhpcy5jdHJseSA9IGN0cmx5OwotICAgICAgICAgICAgdGhpcy54MiA9IHgyOwotICAgICAgICAgICAgdGhpcy55MiA9IHkyOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKCkgewotICAgICAgICAgICAgZG91YmxlIHJ4MCA9IE1hdGgubWluKE1hdGgubWluKHgxLCB4MiksIGN0cmx4KTsKLSAgICAgICAgICAgIGRvdWJsZSByeTAgPSBNYXRoLm1pbihNYXRoLm1pbih5MSwgeTIpLCBjdHJseSk7Ci0gICAgICAgICAgICBkb3VibGUgcngxID0gTWF0aC5tYXgoTWF0aC5tYXgoeDEsIHgyKSwgY3RybHgpOwotICAgICAgICAgICAgZG91YmxlIHJ5MSA9IE1hdGgubWF4KE1hdGgubWF4KHkxLCB5MiksIGN0cmx5KTsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKHJ4MCwgcnkwLCByeDEgLSByeDAsIHJ5MSAtIHJ5MCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFF1YWRDdXJ2ZTJEIHBhdGggaXRlcmF0b3IKLSAgICAgKi8KLSAgICAvKioKLSAgICAgKiBUaGUgUGF0aEl0ZXJhdG9yIGZvciBhIFF1YWQyRCBjdXJ2ZS4KLSAgICAgKi8KLSAgICBjbGFzcyBJdGVyYXRvciBpbXBsZW1lbnRzIFBhdGhJdGVyYXRvciB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzb3VyY2UgUXVhZEN1cnZlMkQgb2JqZWN0LgotICAgICAgICAgKi8KLSAgICAgICAgUXVhZEN1cnZlMkQgYzsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHBhdGggaXRlcmF0b3IgdHJhbnNmb3JtYXRpb24uCi0gICAgICAgICAqLwotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gdDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGN1cnJlbnQgc2VnbWVudCBpbmRleC4KLSAgICAgICAgICovCi0gICAgICAgIGludCBpbmRleDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogQ29uc3RydWN0cyBhIG5ldyBRdWFkQ3VydmUyRC5JdGVyYXRvciBmb3IgZ2l2ZW4gY3VydmUgYW5kCi0gICAgICAgICAqIHRyYW5zZm9ybWF0aW9uCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gcQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgUXVhZEN1cnZlMkQgb2JqZWN0LgotICAgICAgICAgKiBAcGFyYW0gdAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBBZmZpbmVUcmFuc2Zvcm0gdGhhdCBhY3RzIG9uIHRoZSBjb29yZGluYXRlcyBiZWZvcmUKLSAgICAgICAgICogICAgICAgICAgICByZXR1cm5pbmcgdGhlbSAob3IgbnVsbCkuCi0gICAgICAgICAqLwotICAgICAgICBJdGVyYXRvcihRdWFkQ3VydmUyRCBxLCBBZmZpbmVUcmFuc2Zvcm0gdCkgewotICAgICAgICAgICAgdGhpcy5jID0gcTsKLSAgICAgICAgICAgIHRoaXMudCA9IHQ7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGdldFdpbmRpbmdSdWxlKCkgewotICAgICAgICAgICAgcmV0dXJuIFdJTkRfTk9OX1pFUk87Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBpc0RvbmUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gKGluZGV4ID4gMSk7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgdm9pZCBuZXh0KCkgewotICAgICAgICAgICAgaW5kZXgrKzsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgY3VycmVudFNlZ21lbnQoZG91YmxlW10gY29vcmRzKSB7Ci0gICAgICAgICAgICBpZiAoaXNEb25lKCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuNEI9SXRlcmF0b3Igb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBOb1N1Y2hFbGVtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbnQgdHlwZTsKLSAgICAgICAgICAgIGludCBjb3VudDsKLSAgICAgICAgICAgIGlmIChpbmRleCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19NT1ZFVE87Ci0gICAgICAgICAgICAgICAgY29vcmRzWzBdID0gYy5nZXRYMSgpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IGMuZ2V0WTEoKTsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDE7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfUVVBRFRPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IGMuZ2V0Q3RybFgoKTsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSBjLmdldEN0cmxZKCk7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzJdID0gYy5nZXRYMigpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1szXSA9IGMuZ2V0WTIoKTsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAodCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdC50cmFuc2Zvcm0oY29vcmRzLCAwLCBjb29yZHMsIDAsIGNvdW50KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB0eXBlOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGludCBjdXJyZW50U2VnbWVudChmbG9hdFtdIGNvb3JkcykgewotICAgICAgICAgICAgaWYgKGlzRG9uZSgpKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjRCPUl0ZXJhdG9yIG91dCBvZiBib3VuZHMKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgTm9TdWNoRWxlbWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40QiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgaW50IHR5cGU7Ci0gICAgICAgICAgICBpbnQgY291bnQ7Ci0gICAgICAgICAgICBpZiAoaW5kZXggPT0gMCkgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTU9WRVRPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IChmbG9hdCljLmdldFgxKCk7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzFdID0gKGZsb2F0KWMuZ2V0WTEoKTsKLSAgICAgICAgICAgICAgICBjb3VudCA9IDE7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfUVVBRFRPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IChmbG9hdCljLmdldEN0cmxYKCk7Ci0gICAgICAgICAgICAgICAgY29vcmRzWzFdID0gKGZsb2F0KWMuZ2V0Q3RybFkoKTsKLSAgICAgICAgICAgICAgICBjb29yZHNbMl0gPSAoZmxvYXQpYy5nZXRYMigpOwotICAgICAgICAgICAgICAgIGNvb3Jkc1szXSA9IChmbG9hdCljLmdldFkyKCk7Ci0gICAgICAgICAgICAgICAgY291bnQgPSAyOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCBjb3VudCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gdHlwZTsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHF1YWRyYXRpYyBjdXJ2ZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgUXVhZEN1cnZlMkQoKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHN0YXJ0aW5nIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0WDEoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc3RhcnRpbmcgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldFkxKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdGFydGluZyBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgUG9pbnQyRCBnZXRQMSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgY29udHJvbCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldEN0cmxYKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGNvbnRyb2wgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB5IGNvb3JkaW5hdGUgb2YgdGhlIGNvbnRyb2wgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZSBnZXRDdHJsWSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29udHJvbCBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjb250cm9sIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBQb2ludDJEIGdldEN0cmxQdCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZSBnZXRYMigpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZSBnZXRZMigpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZW5kIHBvaW50LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGVuZCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgUG9pbnQyRCBnZXRQMigpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGF0YSBvZiB0aGUgY3VydmUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIGN0cmx4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSBjdHJseQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldEN1cnZlKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgY3RybHgsIGRvdWJsZSBjdHJseSwgZG91YmxlIHgyLAotICAgICAgICAgICAgZG91YmxlIHkyKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgb2YgdGhlIGN1cnZlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwMQotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0aW5nIHBvaW50IG9mIHRoZSBjdXJ2ZWQgc2VnbWVudC4KLSAgICAgKiBAcGFyYW0gY3AKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSBwMgotICAgICAqICAgICAgICAgICAgdGhlIGVuZCBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFueSBvZiB0aGUgdGhyZWUgcG9pbnRzIGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Q3VydmUoUG9pbnQyRCBwMSwgUG9pbnQyRCBjcCwgUG9pbnQyRCBwMikgewotICAgICAgICBzZXRDdXJ2ZShwMS5nZXRYKCksIHAxLmdldFkoKSwgY3AuZ2V0WCgpLCBjcC5nZXRZKCksIHAyLmdldFgoKSwgcDIuZ2V0WSgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBkYXRhIG9mIHRoZSBjdXJ2ZSBieSByZWFkaW5nIHRoZSBkYXRhIGZyb20gYW4gYXJyYXkgb2YgdmFsdWVzLgotICAgICAqIFRoZSB2YWx1ZXMgYXJlIHJlYWQgaW4gdGhlIHNhbWUgb3JkZXIgYXMgdGhlIGFyZ3VtZW50cyBvZiB0aGUgbWV0aG9kCi0gICAgICoge0BsaW5rIFF1YWRDdXJ2ZTJEI3NldEN1cnZlKGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUpfQotICAgICAqIC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29vcmRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgdmFsdWVzIGNvbnRhaW5pbmcgdGhlIG5ldyBjb29yZGluYXRlcy4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IG9mIHRoZSBkYXRhIHRvIHJlYWQgd2l0aGluIHRoZSBhcnJheS4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHtAY29kZSBjb29yZHMubGVuZ3RofSA8IG9mZnNldCArIDYuCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBjb29yZGluYXRlIGFycmF5IGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Q3VydmUoZG91YmxlW10gY29vcmRzLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIHNldEN1cnZlKGNvb3Jkc1tvZmZzZXQgKyAwXSwgY29vcmRzW29mZnNldCArIDFdLCBjb29yZHNbb2Zmc2V0ICsgMl0sIGNvb3Jkc1tvZmZzZXQgKyAzXSwKLSAgICAgICAgICAgICAgICBjb29yZHNbb2Zmc2V0ICsgNF0sIGNvb3Jkc1tvZmZzZXQgKyA1XSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGF0YSBvZiB0aGUgY3VydmUgYnkgcmVhZGluZyB0aGUgZGF0YSBmcm9tIGFuIGFycmF5IG9mIHBvaW50cy4KLSAgICAgKiBUaGUgdmFsdWVzIGFyZSByZWFkIGluIHRoZSBzYW1lIG9yZGVyIGFzIHRoZSBhcmd1bWVudHMgb2YgdGhlIG1ldGhvZAotICAgICAqIHtAbGluayBRdWFkQ3VydmUyRCNzZXRDdXJ2ZShQb2ludDJELCBQb2ludDJELCBQb2ludDJEKX0uCi0gICAgICogCi0gICAgICogQHBhcmFtIHBvaW50cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHBvaW50cyBjb250YWluaW5nIHRoZSBuZXcgY29vcmRpbmF0ZXMuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBvZiB0aGUgZGF0YSB0byByZWFkIHdpdGhpbiB0aGUgYXJyYXkuCi0gICAgICogQHRocm93cyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBwb2ludHMubGVuZ3RoIDwgb2Zmc2V0ICsgMy4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHBvaW50IGFycmF5IGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Q3VydmUoUG9pbnQyRFtdIHBvaW50cywgaW50IG9mZnNldCkgewotICAgICAgICBzZXRDdXJ2ZShwb2ludHNbb2Zmc2V0ICsgMF0uZ2V0WCgpLCBwb2ludHNbb2Zmc2V0ICsgMF0uZ2V0WSgpLCBwb2ludHNbb2Zmc2V0ICsgMV0uZ2V0WCgpLAotICAgICAgICAgICAgICAgIHBvaW50c1tvZmZzZXQgKyAxXS5nZXRZKCksIHBvaW50c1tvZmZzZXQgKyAyXS5nZXRYKCksIHBvaW50c1tvZmZzZXQgKyAyXS5nZXRZKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgb2YgdGhlIGN1cnZlIGJ5IGNvcHlpbmcgaXQgZnJvbSBhbm90aGVyIFF1YWRDdXJ2ZTJELgotICAgICAqIAotICAgICAqIEBwYXJhbSBjdXJ2ZQotICAgICAqICAgICAgICAgICAgdGhlIGN1cnZlIHRvIGNvcHkgdGhlIGRhdGEgcG9pbnRzIGZyb20uCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBjdXJ2ZSBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEN1cnZlKFF1YWRDdXJ2ZTJEIGN1cnZlKSB7Ci0gICAgICAgIHNldEN1cnZlKGN1cnZlLmdldFgxKCksIGN1cnZlLmdldFkxKCksIGN1cnZlLmdldEN0cmxYKCksIGN1cnZlLmdldEN0cmxZKCksIGN1cnZlLmdldFgyKCksCi0gICAgICAgICAgICAgICAgY3VydmUuZ2V0WTIoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc3F1YXJlIG9mIHRoZSBkaXN0YW5jZSBmcm9tIHRoZSBjb250cm9sIHBvaW50IHRvIHRoZSBzdHJhaWdodAotICAgICAqIGxpbmUgc2VnbWVudCBjb25uZWN0aW5nIHRoZSBzdGFydCBwb2ludCBhbmQgdGhlIGVuZCBwb2ludCBmb3IgdGhpcyBjdXJ2ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzcXVhcmUgb2YgdGhlIGRpc3RhbmNlIGZyb20gdGhlIGNvbnRyb2wgcG9pbnQgdG8gdGhlIHN0cmFpZ2h0Ci0gICAgICogICAgICAgICBsaW5lIHNlZ21lbnQgY29ubmVjdGluZyB0aGUgc3RhcnQgcG9pbnQgYW5kIHRoZSBlbmQgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRGbGF0bmVzc1NxKCkgewotICAgICAgICByZXR1cm4gTGluZTJELnB0U2VnRGlzdFNxKGdldFgxKCksIGdldFkxKCksIGdldFgyKCksIGdldFkyKCksIGdldEN0cmxYKCksIGdldEN0cmxZKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgZnJvbSB0aGUgY29udHJvbCBwb2ludCB0byB0aGUgc3RyYWlnaHQKLSAgICAgKiBsaW5lIHNlZ21lbnQgY29ubmVjdGluZyB0aGUgc3RhcnQgcG9pbnQgYW5kIHRoZSBlbmQgcG9pbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIGN0cmx4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSBjdHJseQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAqIEByZXR1cm4gdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgZnJvbSB0aGUgY29udHJvbCBwb2ludCB0byB0aGUgc3RyYWlnaHQKLSAgICAgKiAgICAgICAgIGxpbmUgc2VnbWVudCBjb25uZWN0aW5nIHRoZSBzdGFydCBwb2ludCBhbmQgdGhlIGVuZCBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGRvdWJsZSBnZXRGbGF0bmVzc1NxKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgY3RybHgsIGRvdWJsZSBjdHJseSwgZG91YmxlIHgyLAotICAgICAgICAgICAgZG91YmxlIHkyKSB7Ci0gICAgICAgIHJldHVybiBMaW5lMkQucHRTZWdEaXN0U3EoeDEsIHkxLCB4MiwgeTIsIGN0cmx4LCBjdHJseSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc3F1YXJlIG9mIHRoZSBkaXN0YW5jZSBmcm9tIHRoZSBjb250cm9sIHBvaW50IHRvIHRoZSBzdHJhaWdodAotICAgICAqIGxpbmUgc2VnbWVudCBjb25uZWN0aW5nIHRoZSBzdGFydCBwb2ludCBhbmQgdGhlIGVuZCBwb2ludCBieSByZWFkaW5nIHRoZQotICAgICAqIGNvb3JkaW5hdGVzIG9mIHRoZSBwb2ludHMgZnJvbSBhbiBhcnJheSBvZiB2YWx1ZXMuIFRoZSB2YWx1ZXMgYXJlIHJlYWQgaW4KLSAgICAgKiB0aGUgc2FtZSBvcmRlciBhcyB0aGUgYXJndW1lbnRzIG9mIHRoZSBtZXRob2QKLSAgICAgKiB7QGxpbmsgUXVhZEN1cnZlMkQjZ2V0RmxhdG5lc3NTcShkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSwgZG91YmxlKX0KLSAgICAgKiAuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvb3JkcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHBvaW50cyBjb250YWluaW5nIHRoZSBjb29yZGluYXRlcyB0byB1c2UgZm9yIHRoZQotICAgICAqICAgICAgICAgICAgY2FsY3VsYXRpb24KLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IG9mIHRoZSBkYXRhIHRvIHJlYWQgd2l0aGluIHRoZSBhcnJheQotICAgICAqIEByZXR1cm4gdGhlIHNxdWFyZSBvZiB0aGUgZGlzdGFuY2UgZnJvbSB0aGUgY29udHJvbCBwb2ludCB0byB0aGUgc3RyYWlnaHQKLSAgICAgKiAgICAgICAgIGxpbmUgc2VnbWVudCBjb25uZWN0aW5nIHRoZSBzdGFydCBwb2ludCBhbmQgdGhlIGVuZCBwb2ludC4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHtAY29kZSBjb29yZHMubGVuZ3RofSA8IG9mZnNldCArIDYuCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBjb29yZGluYXRlIGFycmF5IGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBkb3VibGUgZ2V0RmxhdG5lc3NTcShkb3VibGUgY29vcmRzW10sIGludCBvZmZzZXQpIHsKLSAgICAgICAgcmV0dXJuIExpbmUyRC5wdFNlZ0Rpc3RTcShjb29yZHNbb2Zmc2V0ICsgMF0sIGNvb3Jkc1tvZmZzZXQgKyAxXSwgY29vcmRzW29mZnNldCArIDRdLAotICAgICAgICAgICAgICAgIGNvb3Jkc1tvZmZzZXQgKyA1XSwgY29vcmRzW29mZnNldCArIDJdLCBjb29yZHNbb2Zmc2V0ICsgM10pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRpc3RhbmNlIGZyb20gdGhlIGNvbnRyb2wgcG9pbnQgdG8gdGhlIHN0cmFpZ2h0IGxpbmUgc2VnbWVudAotICAgICAqIGNvbm5lY3RpbmcgdGhlIHN0YXJ0IHBvaW50IGFuZCB0aGUgZW5kIHBvaW50IG9mIHRoaXMgUXVhZEN1cnZlMkQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdGhlIGRpc3RhbmNlIGZyb20gdGhlIGNvbnRyb2wgcG9pbnQgdG8gdGhlIHN0cmFpZ2h0IGxpbmUKLSAgICAgKiAgICAgICAgIHNlZ21lbnQgY29ubmVjdGluZyB0aGUgc3RhcnQgcG9pbnQgYW5kIHRoZSBlbmQgcG9pbnQgb2YgdGhpcwotICAgICAqICAgICAgICAgUXVhZEN1cnZlMkQuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRGbGF0bmVzcygpIHsKLSAgICAgICAgcmV0dXJuIExpbmUyRC5wdFNlZ0Rpc3QoZ2V0WDEoKSwgZ2V0WTEoKSwgZ2V0WDIoKSwgZ2V0WTIoKSwgZ2V0Q3RybFgoKSwgZ2V0Q3RybFkoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGlzdGFuY2UgZnJvbSB0aGUgY29udHJvbCBwb2ludCB0byB0aGUgc3RyYWlnaHQgbGluZSBzZWdtZW50Ci0gICAgICogY29ubmVjdGluZyB0aGUgc3RhcnQgcG9pbnQgYW5kIHRoZSBlbmQgcG9pbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgxCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkxCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBzdGFydGluZyBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIGN0cmx4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBjb250cm9sIHBvaW50LgotICAgICAqIEBwYXJhbSBjdHJseQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgY29udHJvbCBwb2ludC4KLSAgICAgKiBAcGFyYW0geDIKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGVuZCBwb2ludCBvZiB0aGUgY3VydmVkIHNlZ21lbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBlbmQgcG9pbnQgb2YgdGhlIGN1cnZlZCBzZWdtZW50LgotICAgICAqIEByZXR1cm4gdGhlIHRoZSBkaXN0YW5jZSBmcm9tIHRoZSBjb250cm9sIHBvaW50IHRvIHRoZSBzdHJhaWdodCBsaW5lCi0gICAgICogICAgICAgICBzZWdtZW50IGNvbm5lY3RpbmcgdGhlIHN0YXJ0IHBvaW50IGFuZCB0aGUgZW5kIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZG91YmxlIGdldEZsYXRuZXNzKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgY3RybHgsIGRvdWJsZSBjdHJseSwgZG91YmxlIHgyLAotICAgICAgICAgICAgZG91YmxlIHkyKSB7Ci0gICAgICAgIHJldHVybiBMaW5lMkQucHRTZWdEaXN0KHgxLCB5MSwgeDIsIHkyLCBjdHJseCwgY3RybHkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRoZSBkaXN0YW5jZSBmcm9tIHRoZSBjb250cm9sIHBvaW50IHRvIHRoZSBzdHJhaWdodCBsaW5lIHNlZ21lbnQKLSAgICAgKiBjb25uZWN0aW5nIHRoZSBzdGFydCBwb2ludCBhbmQgdGhlIGVuZCBwb2ludC4gVGhlIHZhbHVlcyBhcmUgcmVhZCBpbiB0aGUKLSAgICAgKiBzYW1lIG9yZGVyIGFzIHRoZSBhcmd1bWVudHMgb2YgdGhlIG1ldGhvZAotICAgICAqIHtAbGluayBRdWFkQ3VydmUyRCNnZXRGbGF0bmVzcyhkb3VibGUsIGRvdWJsZSwgZG91YmxlLCBkb3VibGUsIGRvdWJsZSwgZG91YmxlKX0KLSAgICAgKiAuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvb3JkcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHBvaW50cyBjb250YWluaW5nIHRoZSBjb29yZGluYXRlcyB0byB1c2UgZm9yIHRoZQotICAgICAqICAgICAgICAgICAgY2FsY3VsYXRpb24uCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBvZiB0aGUgZGF0YSB0byByZWFkIHdpdGhpbiB0aGUgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgdGhlIGRpc3RhbmNlIGZyb20gdGhlIGNvbnRyb2wgcG9pbnQgdG8gdGhlIHN0cmFpZ2h0IGxpbmUKLSAgICAgKiAgICAgICAgIHNlZ21lbnQgY29ubmVjdGluZyB0aGUgc3RhcnQgcG9pbnQgYW5kIHRoZSBlbmQgcG9pbnQuCi0gICAgICogQHRocm93cyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB7Y29kZSBjb29yZHMubGVuZ3RofSA8IG9mZnNldCArIDYuCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBjb29yZGluYXRlIGFycmF5IGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBkb3VibGUgZ2V0RmxhdG5lc3MoZG91YmxlIGNvb3Jkc1tdLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIHJldHVybiBMaW5lMkQucHRTZWdEaXN0KGNvb3Jkc1tvZmZzZXQgKyAwXSwgY29vcmRzW29mZnNldCArIDFdLCBjb29yZHNbb2Zmc2V0ICsgNF0sCi0gICAgICAgICAgICAgICAgY29vcmRzW29mZnNldCArIDVdLCBjb29yZHNbb2Zmc2V0ICsgMl0sIGNvb3Jkc1tvZmZzZXQgKyAzXSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgZGF0YSBmb3IgdHdvIHF1YWRyYXRpYyBjdXJ2ZXMgYnkgZGl2aWRpbmcgdGhpcyBjdXJ2ZSBpbiB0d28uCi0gICAgICogVGhlIGRpdmlzaW9uIHBvaW50IGlzIHRoZSBwb2ludCBvbiB0aGUgY3VydmUgdGhhdCBpcyBjbG9zZXN0IHRvIHRoaXMKLSAgICAgKiBjdXJ2ZSdzIGNvbnRyb2wgcG9pbnQuIFRoZSBkYXRhIG9mIHRoaXMgY3VydmUgaXMgbGVmdCB1bmNoYW5nZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxlZnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBRdWFkQ3VydmUyRCB3aGVyZSB0aGUgbGVmdCAoc3RhcnQpIHNlZ21lbnQncyBkYXRhIGlzCi0gICAgICogICAgICAgICAgICB3cml0dGVuLgotICAgICAqIEBwYXJhbSByaWdodAotICAgICAqICAgICAgICAgICAgdGhlIFF1YWRDdXJ2ZTJEIHdoZXJlIHRoZSByaWdodCAoZW5kKSBzZWdtZW50J3MgZGF0YSBpcwotICAgICAqICAgICAgICAgICAgd3JpdHRlbi4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgZWl0aGVyIGN1cnZlIGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc3ViZGl2aWRlKFF1YWRDdXJ2ZTJEIGxlZnQsIFF1YWRDdXJ2ZTJEIHJpZ2h0KSB7Ci0gICAgICAgIHN1YmRpdmlkZSh0aGlzLCBsZWZ0LCByaWdodCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgZGF0YSBmb3IgdHdvIHF1YWRyYXRpYyBjdXJ2ZXMgYnkgZGl2aWRpbmcgYSBzb3VyY2UgY3VydmUgaW4KLSAgICAgKiB0d28uIFRoZSBkaXZpc2lvbiBwb2ludCBpcyB0aGUgcG9pbnQgb24gdGhlIGN1cnZlIHRoYXQgaXMgY2xvc2VzdCB0byB0aGUKLSAgICAgKiBzb3VyY2UgY3VydmUncyBjb250cm9sIHBvaW50LiBUaGUgZGF0YSBvZiB0aGUgc291cmNlIGN1cnZlIGlzIGxlZnQKLSAgICAgKiB1bmNoYW5nZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIGN1cnZlIHRoYXQgcHJvdmlkZXMgdGhlIGluaXRpYWwgZGF0YS4KLSAgICAgKiBAcGFyYW0gbGVmdAotICAgICAqICAgICAgICAgICAgdGhlIFF1YWRDdXJ2ZTJEIHdoZXJlIHRoZSBsZWZ0IChzdGFydCkgc2VnbWVudCdzIGRhdGEgaXMKLSAgICAgKiAgICAgICAgICAgIHdyaXR0ZW4uCi0gICAgICogQHBhcmFtIHJpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgUXVhZEN1cnZlMkQgd2hlcmUgdGhlIHJpZ2h0IChlbmQpIHNlZ21lbnQncyBkYXRhIGlzCi0gICAgICogICAgICAgICAgICB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgTnVsbFBvaW50ZXJFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBvbmUgb2YgdGhlIGN1cnZlcyBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgdm9pZCBzdWJkaXZpZGUoUXVhZEN1cnZlMkQgc3JjLCBRdWFkQ3VydmUyRCBsZWZ0LCBRdWFkQ3VydmUyRCByaWdodCkgewotICAgICAgICBkb3VibGUgeDEgPSBzcmMuZ2V0WDEoKTsKLSAgICAgICAgZG91YmxlIHkxID0gc3JjLmdldFkxKCk7Ci0gICAgICAgIGRvdWJsZSBjeCA9IHNyYy5nZXRDdHJsWCgpOwotICAgICAgICBkb3VibGUgY3kgPSBzcmMuZ2V0Q3RybFkoKTsKLSAgICAgICAgZG91YmxlIHgyID0gc3JjLmdldFgyKCk7Ci0gICAgICAgIGRvdWJsZSB5MiA9IHNyYy5nZXRZMigpOwotICAgICAgICBkb3VibGUgY3gxID0gKHgxICsgY3gpIC8gMi4wOwotICAgICAgICBkb3VibGUgY3kxID0gKHkxICsgY3kpIC8gMi4wOwotICAgICAgICBkb3VibGUgY3gyID0gKHgyICsgY3gpIC8gMi4wOwotICAgICAgICBkb3VibGUgY3kyID0gKHkyICsgY3kpIC8gMi4wOwotICAgICAgICBjeCA9IChjeDEgKyBjeDIpIC8gMi4wOwotICAgICAgICBjeSA9IChjeTEgKyBjeTIpIC8gMi4wOwotICAgICAgICBpZiAobGVmdCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBsZWZ0LnNldEN1cnZlKHgxLCB5MSwgY3gxLCBjeTEsIGN4LCBjeSk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHJpZ2h0ICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJpZ2h0LnNldEN1cnZlKGN4LCBjeSwgY3gyLCBjeTIsIHgyLCB5Mik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBkYXRhIGZvciB0d28gcXVhZHJhdGljIGN1cnZlcyBieSBkaXZpZGluZyBhIHNvdXJjZSBjdXJ2ZSBpbgotICAgICAqIHR3by4gVGhlIGRpdmlzaW9uIHBvaW50IGlzIHRoZSBwb2ludCBvbiB0aGUgY3VydmUgdGhhdCBpcyBjbG9zZXN0IHRvIHRoZQotICAgICAqIHNvdXJjZSBjdXJ2ZSdzIGNvbnRyb2wgcG9pbnQuIFRoZSBkYXRhIGZvciB0aGUgdGhyZWUgY3VydmVzIGlzIHJlYWQgYW5kCi0gICAgICogd3JpdHRlbiBmcm9tIGFycmF5cyBvZiB2YWx1ZXMgaW4gdGhlIHVzdWFsIG9yZGVyOiB4MSwgeTEsIGN4LCBjeSwgeDIsIHkyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB0aGF0IGdpdmVzIHRoZSBkYXRhIHZhbHVlcyBmb3IgdGhlIHNvdXJjZSBjdXJ2ZS4KLSAgICAgKiBAcGFyYW0gc3Jjb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBzcmMgYXJyYXkgdG8gcmVhZCB0aGUgdmFsdWVzIGZyb20uCi0gICAgICogQHBhcmFtIGxlZnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB3aGVyZSB0aGUgY29vcmRpbmF0ZXMgb2YgdGhlIHN0YXJ0IGN1cnZlIHNob3VsZCBiZQotICAgICAqICAgICAgICAgICAgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gbGVmdE9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgbGVmdCBhcnJheSB0byBzdGFydCB3cml0aW5nIHRoZSB2YWx1ZXMuCi0gICAgICogQHBhcmFtIHJpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgd2hlcmUgdGhlIGNvb3JkaW5hdGVzIG9mIHRoZSBlbmQgY3VydmUgc2hvdWxkIGJlCi0gICAgICogICAgICAgICAgICB3cml0dGVuLgotICAgICAqIEBwYXJhbSByaWdodE9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgcmlnaHQgYXJyYXkgdG8gc3RhcnQgd3JpdGluZyB0aGUgdmFsdWVzLgotICAgICAqIEB0aHJvd3MgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYge0Bjb2RlIHNyYy5sZW5ndGh9IDwgc3Jjb2ZmICsgNiBvciBpZiB7QGNvZGUgbGVmdC5sZW5ndGh9Ci0gICAgICogICAgICAgICAgICAgPCBsZWZ0T2ZmICsgNiBvciBpZiB7QGNvZGUgcmlnaHQubGVuZ3RofSA8IHJpZ2h0T2ZmICsgNi4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgb25lIG9mIHRoZSBhcnJheXMgaXMgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgc3ViZGl2aWRlKGRvdWJsZSBzcmNbXSwgaW50IHNyY29mZiwgZG91YmxlIGxlZnRbXSwgaW50IGxlZnRPZmYsCi0gICAgICAgICAgICBkb3VibGUgcmlnaHRbXSwgaW50IHJpZ2h0T2ZmKSB7Ci0gICAgICAgIGRvdWJsZSB4MSA9IHNyY1tzcmNvZmYgKyAwXTsKLSAgICAgICAgZG91YmxlIHkxID0gc3JjW3NyY29mZiArIDFdOwotICAgICAgICBkb3VibGUgY3ggPSBzcmNbc3Jjb2ZmICsgMl07Ci0gICAgICAgIGRvdWJsZSBjeSA9IHNyY1tzcmNvZmYgKyAzXTsKLSAgICAgICAgZG91YmxlIHgyID0gc3JjW3NyY29mZiArIDRdOwotICAgICAgICBkb3VibGUgeTIgPSBzcmNbc3Jjb2ZmICsgNV07Ci0gICAgICAgIGRvdWJsZSBjeDEgPSAoeDEgKyBjeCkgLyAyLjA7Ci0gICAgICAgIGRvdWJsZSBjeTEgPSAoeTEgKyBjeSkgLyAyLjA7Ci0gICAgICAgIGRvdWJsZSBjeDIgPSAoeDIgKyBjeCkgLyAyLjA7Ci0gICAgICAgIGRvdWJsZSBjeTIgPSAoeTIgKyBjeSkgLyAyLjA7Ci0gICAgICAgIGN4ID0gKGN4MSArIGN4MikgLyAyLjA7Ci0gICAgICAgIGN5ID0gKGN5MSArIGN5MikgLyAyLjA7Ci0gICAgICAgIGlmIChsZWZ0ICE9IG51bGwpIHsKLSAgICAgICAgICAgIGxlZnRbbGVmdE9mZiArIDBdID0geDE7Ci0gICAgICAgICAgICBsZWZ0W2xlZnRPZmYgKyAxXSA9IHkxOwotICAgICAgICAgICAgbGVmdFtsZWZ0T2ZmICsgMl0gPSBjeDE7Ci0gICAgICAgICAgICBsZWZ0W2xlZnRPZmYgKyAzXSA9IGN5MTsKLSAgICAgICAgICAgIGxlZnRbbGVmdE9mZiArIDRdID0gY3g7Ci0gICAgICAgICAgICBsZWZ0W2xlZnRPZmYgKyA1XSA9IGN5OwotICAgICAgICB9Ci0gICAgICAgIGlmIChyaWdodCAhPSBudWxsKSB7Ci0gICAgICAgICAgICByaWdodFtyaWdodE9mZiArIDBdID0gY3g7Ci0gICAgICAgICAgICByaWdodFtyaWdodE9mZiArIDFdID0gY3k7Ci0gICAgICAgICAgICByaWdodFtyaWdodE9mZiArIDJdID0gY3gyOwotICAgICAgICAgICAgcmlnaHRbcmlnaHRPZmYgKyAzXSA9IGN5MjsKLSAgICAgICAgICAgIHJpZ2h0W3JpZ2h0T2ZmICsgNF0gPSB4MjsKLSAgICAgICAgICAgIHJpZ2h0W3JpZ2h0T2ZmICsgNV0gPSB5MjsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbmRzIHRoZSByb290cyBvZiB0aGUgcXVhZHJhdGljIHBvbHlub21pYWwuIFRoaXMgaXMgYWNjb21wbGlzaGVkIGJ5Ci0gICAgICogZmluZGluZyB0aGUgKHJlYWwpIHZhbHVlcyBvZiB4IHRoYXQgc29sdmUgdGhlIGZvbGxvd2luZyBlcXVhdGlvbjoKLSAgICAgKiBlcW5bMl0qeCp4ICsgZXFuWzFdKnggKyBlcW5bMF0gPSAwLiBUaGUgc29sdXRpb25zIGFyZSB3cml0dGVuIGJhY2sgaW50bwotICAgICAqIHRoZSBhcnJheSBlcW4gc3RhcnRpbmcgZnJvbSB0aGUgaW5kZXggMCBpbiB0aGUgYXJyYXkuIFRoZSByZXR1cm4gdmFsdWUKLSAgICAgKiB0ZWxscyBob3cgbWFueSBhcnJheSBlbGVtZW50cyBoYXZlIGJlZW4gY2hhbmdlZCBieSB0aGlzIG1ldGhvZCBjYWxsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlcW4KLSAgICAgKiAgICAgICAgICAgIGFuIGFycmF5IGNvbnRhaW5pbmcgdGhlIGNvZWZmaWNpZW50cyBvZiB0aGUgcXVhZHJhdGljCi0gICAgICogICAgICAgICAgICBwb2x5bm9taWFsIHRvIHNvbHZlLgotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiByb290cyBvZiB0aGUgcXVhZHJhdGljIHBvbHlub21pYWwuCi0gICAgICogQHRocm93cyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB7QGNvZGUgZXFuLmxlbmd0aH0gPCAzLgotICAgICAqIEB0aHJvd3MgTnVsbFBvaW50ZXJFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgYXJyYXkgaXMgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBzb2x2ZVF1YWRyYXRpYyhkb3VibGUgZXFuW10pIHsKLSAgICAgICAgcmV0dXJuIHNvbHZlUXVhZHJhdGljKGVxbiwgZXFuKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaW5kcyB0aGUgcm9vdHMgb2YgdGhlIHF1YWRyYXRpYyBwb2x5bm9taWFsLiBUaGlzIGlzIGFjY29tcGxpc2hlZCBieQotICAgICAqIGZpbmRpbmcgdGhlIChyZWFsKSB2YWx1ZXMgb2YgeCB0aGF0IHNvbHZlIHRoZSBmb2xsb3dpbmcgZXF1YXRpb246Ci0gICAgICogZXFuWzJdKngqeCArIGVxblsxXSp4ICsgZXFuWzBdID0gMC4gVGhlIHNvbHV0aW9ucyBhcmUgd3JpdHRlbiBpbnRvIHRoZQotICAgICAqIGFycmF5IHJlcyBzdGFydGluZyBmcm9tIHRoZSBpbmRleCAwIGluIHRoZSBhcnJheS4gVGhlIHJldHVybiB2YWx1ZSB0ZWxscwotICAgICAqIGhvdyBtYW55IGFycmF5IGVsZW1lbnRzIGhhdmUgYmVlbiB3cml0dGVuIGJ5IHRoaXMgbWV0aG9kIGNhbGwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVxbgotICAgICAqICAgICAgICAgICAgYW4gYXJyYXkgY29udGFpbmluZyB0aGUgY29lZmZpY2llbnRzIG9mIHRoZSBxdWFkcmF0aWMKLSAgICAgKiAgICAgICAgICAgIHBvbHlub21pYWwgdG8gc29sdmUuCi0gICAgICogQHBhcmFtIHJlcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IHRoYXQgdGhpcyBtZXRob2Qgd3JpdGVzIHRoZSByZXN1bHRzIGludG8uCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIHJvb3RzIG9mIHRoZSBxdWFkcmF0aWMgcG9seW5vbWlhbC4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHtAY29kZSBlcW4ubGVuZ3RofSA8IDMgb3IgaWYge0Bjb2RlIHJlcy5sZW5ndGh9IGlzIGxlc3MKLSAgICAgKiAgICAgICAgICAgICB0aGFuIHRoZSBudW1iZXIgb2Ygcm9vdHMuCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGVpdGhlciBhcnJheSBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgaW50IHNvbHZlUXVhZHJhdGljKGRvdWJsZSBlcW5bXSwgZG91YmxlIHJlc1tdKSB7Ci0gICAgICAgIHJldHVybiBDcm9zc2luZy5zb2x2ZVF1YWQoZXFuLCByZXMpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSBweCwgZG91YmxlIHB5KSB7Ci0gICAgICAgIHJldHVybiBDcm9zc2luZy5pc0luc2lkZUV2ZW5PZGQoQ3Jvc3NpbmcuY3Jvc3NTaGFwZSh0aGlzLCBweCwgcHkpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhkb3VibGUgcngsIGRvdWJsZSByeSwgZG91YmxlIHJ3LCBkb3VibGUgcmgpIHsKLSAgICAgICAgaW50IGNyb3NzID0gQ3Jvc3NpbmcuaW50ZXJzZWN0U2hhcGUodGhpcywgcngsIHJ5LCBydywgcmgpOwotICAgICAgICByZXR1cm4gY3Jvc3MgIT0gQ3Jvc3NpbmcuQ1JPU1NJTkcgJiYgQ3Jvc3NpbmcuaXNJbnNpZGVFdmVuT2RkKGNyb3NzKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpbnRlcnNlY3RzKGRvdWJsZSByeCwgZG91YmxlIHJ5LCBkb3VibGUgcncsIGRvdWJsZSByaCkgewotICAgICAgICBpbnQgY3Jvc3MgPSBDcm9zc2luZy5pbnRlcnNlY3RTaGFwZSh0aGlzLCByeCwgcnksIHJ3LCByaCk7Ci0gICAgICAgIHJldHVybiBjcm9zcyA9PSBDcm9zc2luZy5DUk9TU0lORyB8fCBDcm9zc2luZy5pc0luc2lkZUV2ZW5PZGQoY3Jvc3MpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFBvaW50MkQgcCkgewotICAgICAgICByZXR1cm4gY29udGFpbnMocC5nZXRYKCksIHAuZ2V0WSgpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpbnRlcnNlY3RzKFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgcmV0dXJuIGludGVyc2VjdHMoci5nZXRYKCksIHIuZ2V0WSgpLCByLmdldFdpZHRoKCksIHIuZ2V0SGVpZ2h0KCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgcmV0dXJuIGNvbnRhaW5zKHIuZ2V0WCgpLCByLmdldFkoKSwgci5nZXRXaWR0aCgpLCByLmdldEhlaWdodCgpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldEJvdW5kcygpIHsKLSAgICAgICAgcmV0dXJuIGdldEJvdW5kczJEKCkuZ2V0Qm91bmRzKCk7Ci0gICAgfQotCi0gICAgcHVibGljIFBhdGhJdGVyYXRvciBnZXRQYXRoSXRlcmF0b3IoQWZmaW5lVHJhbnNmb3JtIHQpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJdGVyYXRvcih0aGlzLCB0KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gdCwgZG91YmxlIGZsYXRuZXNzKSB7Ci0gICAgICAgIHJldHVybiBuZXcgRmxhdHRlbmluZ1BhdGhJdGVyYXRvcihnZXRQYXRoSXRlcmF0b3IodCksIGZsYXRuZXNzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgT2JqZWN0IGNsb25lKCkgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIHN1cGVyLmNsb25lKCk7Ci0gICAgICAgIH0gY2F0Y2ggKENsb25lTm90U3VwcG9ydGVkRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9nZW9tL1JlY3RhbmdsZTJELmphdmEgYi9hd3QvamF2YS9hd3QvZ2VvbS9SZWN0YW5nbGUyRC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4MTY2MTM0Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9nZW9tL1JlY3RhbmdsZTJELmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4MjQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuZ2VvbTsKLQotaW1wb3J0IGphdmEudXRpbC5Ob1N1Y2hFbGVtZW50RXhjZXB0aW9uOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lm1pc2MuSGFzaENvZGU7Ci0KLS8qKgotICogVGhlIENsYXNzIFJlY3RhbmdsZTJEIHJlcHJlc2VudHMgYSByZWN0YW5nbGUgd2hvc2UgY29vcmRpbmF0ZXMgYXJlIGdpdmVuIHdpdGgKLSAqIHRoZSBjb3JyZWN0IHByZWNpc2lvbiB0byBiZSB1c2VkIHdpdGggdGhlIEdyYXBoaWNzMkQgY2xhc3Nlcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBSZWN0YW5nbGUyRCBleHRlbmRzIFJlY3Rhbmd1bGFyU2hhcGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE9VVF9MRUZUIGlzIGEgbWFzayB0aGF0IGlzIHVzZWQgdG8gaW5kaWNhdGUgdGhhdCBhIGdpdmVuCi0gICAgICogcG9pbnQgaXMgb3V0c2lkZSB0aGUgcmVjdGFuZ2xlIGFuZCB0byBpdHMgbGVmdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBPVVRfTEVGVCA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgT1VUX1RPUCBpcyBhIG1hc2sgdGhhdCBpcyB1c2VkIHRvIGluZGljYXRlIHRoYXQgYSBnaXZlbgotICAgICAqIHBvaW50IGlzIG91dHNpZGUgdGhlIHJlY3RhbmdsZSBhbmQgYWJvdmUgaXQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgT1VUX1RPUCA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgT1VUX1JJR0hUIGlzIGEgbWFzayB0aGF0IGlzIHVzZWQgdG8gaW5kaWNhdGUgdGhhdCBhIGdpdmVuCi0gICAgICogcG9pbnQgaXMgb3V0c2lkZSB0aGUgcmVjdGFuZ2xlIGFuZCB0byBpdHMgcmlnaHQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgT1VUX1JJR0hUID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBPVVRfQk9UVE9NIGlzIGEgbWFzayB0aGF0IGlzIHVzZWQgdG8gaW5kaWNhdGUgdGhhdCBhIGdpdmVuCi0gICAgICogcG9pbnQgaXMgb3V0c2lkZSB0aGUgcmVjdGFuZ2xlIGFuZCBhYm92ZSBpdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBPVVRfQk9UVE9NID0gODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBGbG9hdCBpcyB0aGUgc3ViY2xhc3Mgb2YgUmVjdGFuZ2xlMkQgdGhhdCByZXByZXNlbnRzIGEKLSAgICAgKiByZWN0YW5nbGUgd2hvc2UgZGF0YSB2YWx1ZXMgYXJlIGdpdmVuIGFzIGZsb2F0cyAod2l0aCBmbG9hdC1sZXZlbAotICAgICAqIHByZWNpc2lvbikuCi0gICAgICogCi0gICAgICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBjbGFzcyBGbG9hdCBleHRlbmRzIFJlY3RhbmdsZTJEIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgeDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgeTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgd2lkdGg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBmbG9hdCBoZWlnaHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBlbXB0eSByZWN0YW5nbGUgd2l0aCBmbG9hdC1wcmVjaXNpb24gZGF0YSBmaWVsZHMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRmxvYXQoKSB7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHJlY3RhbmdsZSB3aXRoIHRoZSBzcGVjaWZpZWQgZmxvYXQtcHJlY2lzaW9uIGRhdGEuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAgICAgKiBAcGFyYW0geQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICogQHBhcmFtIGhlaWdodAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBGbG9hdChmbG9hdCB4LCBmbG9hdCB5LCBmbG9hdCB3aWR0aCwgZmxvYXQgaGVpZ2h0KSB7Ci0gICAgICAgICAgICBzZXRSZWN0KHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WCgpIHsKLSAgICAgICAgICAgIHJldHVybiB4OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WSgpIHsKLSAgICAgICAgICAgIHJldHVybiB5OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0V2lkdGgoKSB7Ci0gICAgICAgICAgICByZXR1cm4gd2lkdGg7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRIZWlnaHQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gaGVpZ2h0OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzRW1wdHkoKSB7Ci0gICAgICAgICAgICByZXR1cm4gd2lkdGggPD0gMC4wZiB8fCBoZWlnaHQgPD0gMC4wZjsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBTZXRzIHRoZSByZWN0YW5nbGUncyBkYXRhIHRvIHRoZSBnaXZlbiB2YWx1ZXMuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAgICAgKiBAcGFyYW0geQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICogQHBhcmFtIGhlaWdodAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyB2b2lkIHNldFJlY3QoZmxvYXQgeCwgZmxvYXQgeSwgZmxvYXQgd2lkdGgsIGZsb2F0IGhlaWdodCkgewotICAgICAgICAgICAgdGhpcy54ID0geDsKLSAgICAgICAgICAgIHRoaXMueSA9IHk7Ci0gICAgICAgICAgICB0aGlzLndpZHRoID0gd2lkdGg7Ci0gICAgICAgICAgICB0aGlzLmhlaWdodCA9IGhlaWdodDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRSZWN0KGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0KSB7Ci0gICAgICAgICAgICB0aGlzLnggPSAoZmxvYXQpeDsKLSAgICAgICAgICAgIHRoaXMueSA9IChmbG9hdCl5OwotICAgICAgICAgICAgdGhpcy53aWR0aCA9IChmbG9hdCl3aWR0aDsKLSAgICAgICAgICAgIHRoaXMuaGVpZ2h0ID0gKGZsb2F0KWhlaWdodDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRSZWN0KFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgICAgIHRoaXMueCA9IChmbG9hdClyLmdldFgoKTsKLSAgICAgICAgICAgIHRoaXMueSA9IChmbG9hdClyLmdldFkoKTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSAoZmxvYXQpci5nZXRXaWR0aCgpOwotICAgICAgICAgICAgdGhpcy5oZWlnaHQgPSAoZmxvYXQpci5nZXRIZWlnaHQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgaW50IG91dGNvZGUoZG91YmxlIHB4LCBkb3VibGUgcHkpIHsKLSAgICAgICAgICAgIGludCBjb2RlID0gMDsKLQotICAgICAgICAgICAgaWYgKHdpZHRoIDw9IDAuMGYpIHsKLSAgICAgICAgICAgICAgICBjb2RlIHw9IE9VVF9MRUZUIHwgT1VUX1JJR0hUOwotICAgICAgICAgICAgfSBlbHNlIGlmIChweCA8IHgpIHsKLSAgICAgICAgICAgICAgICBjb2RlIHw9IE9VVF9MRUZUOwotICAgICAgICAgICAgfSBlbHNlIGlmIChweCA+IHggKyB3aWR0aCkgewotICAgICAgICAgICAgICAgIGNvZGUgfD0gT1VUX1JJR0hUOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoaGVpZ2h0IDw9IDAuMGYpIHsKLSAgICAgICAgICAgICAgICBjb2RlIHw9IE9VVF9UT1AgfCBPVVRfQk9UVE9NOwotICAgICAgICAgICAgfSBlbHNlIGlmIChweSA8IHkpIHsKLSAgICAgICAgICAgICAgICBjb2RlIHw9IE9VVF9UT1A7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKHB5ID4geSArIGhlaWdodCkgewotICAgICAgICAgICAgICAgIGNvZGUgfD0gT1VUX0JPVFRPTTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIGNvZGU7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBGbG9hdCh4LCB5LCB3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgY3JlYXRlSW50ZXJzZWN0aW9uKFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgICAgIFJlY3RhbmdsZTJEIGRzdDsKLSAgICAgICAgICAgIGlmIChyIGluc3RhbmNlb2YgRG91YmxlKSB7Ci0gICAgICAgICAgICAgICAgZHN0ID0gbmV3IFJlY3RhbmdsZTJELkRvdWJsZSgpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBkc3QgPSBuZXcgUmVjdGFuZ2xlMkQuRmxvYXQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIFJlY3RhbmdsZTJELmludGVyc2VjdCh0aGlzLCByLCBkc3QpOwotICAgICAgICAgICAgcmV0dXJuIGRzdDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgY3JlYXRlVW5pb24oUmVjdGFuZ2xlMkQgcikgewotICAgICAgICAgICAgUmVjdGFuZ2xlMkQgZHN0OwotICAgICAgICAgICAgaWYgKHIgaW5zdGFuY2VvZiBEb3VibGUpIHsKLSAgICAgICAgICAgICAgICBkc3QgPSBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGRzdCA9IG5ldyBSZWN0YW5nbGUyRC5GbG9hdCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgUmVjdGFuZ2xlMkQudW5pb24odGhpcywgciwgZHN0KTsKLSAgICAgICAgICAgIHJldHVybiBkc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsKLSAgICAgICAgICAgIC8vIFRoZSBvdXRwdXQgZm9ybWF0IGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW91ci4gSXQgY291bGQgYmUKLSAgICAgICAgICAgIC8vIG9idGFpbmVkIGluIHRoZSBmb2xsb3dpbmcgd2F5Ci0gICAgICAgICAgICAvLyBTeXN0ZW0ub3V0LnByaW50bG4obmV3IFJlY3RhbmdsZTJELkZsb2F0KCkudG9TdHJpbmcoKSkKLSAgICAgICAgICAgIHJldHVybiBnZXRDbGFzcygpLmdldE5hbWUoKQotICAgICAgICAgICAgICAgICAgICArICJbeD0iICsgeCArICIseT0iICsgeSArICIsd2lkdGg9IiArIHdpZHRoICsgIixoZWlnaHQ9IiArIGhlaWdodCArICJdIjsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQgLy8kTk9OLU5MUy00JCAvLyROT04tTkxTLTUkCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgRG91YmxlIGlzIHRoZSBzdWJjbGFzcyBvZiBSZWN0YW5nbGUyRCB0aGF0IHJlcHJlc2VudHMgYQotICAgICAqIHJlY3RhbmdsZSB3aG9zZSBkYXRhIHZhbHVlcyBhcmUgZ2l2ZW4gYXMgZG91YmxlcyAod2l0aAotICAgICAqIGRvdWJsZS1wcmVjaXNpb24tbGV2ZWwgcHJlY2lzaW9uKS4KLSAgICAgKiAKLSAgICAgKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGNsYXNzIERvdWJsZSBleHRlbmRzIFJlY3RhbmdsZTJEIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIHg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB5OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgd2lkdGg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgaGVpZ2h0OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZW1wdHkgcmVjdGFuZ2xlIHdpdGggZG91YmxlLXByZWNpc2lvbiBkYXRhIGZpZWxkcy4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBEb3VibGUoKSB7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHJlY3RhbmdsZSB3aXRoIHRoZSBnaXZlbiBkb3VibGUgdmFsdWVzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgICAgICogQHBhcmFtIHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgICAgICogQHBhcmFtIHdpZHRoCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRG91YmxlKGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0KSB7Ci0gICAgICAgICAgICBzZXRSZWN0KHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WCgpIHsKLSAgICAgICAgICAgIHJldHVybiB4OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0WSgpIHsKLSAgICAgICAgICAgIHJldHVybiB5OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0V2lkdGgoKSB7Ci0gICAgICAgICAgICByZXR1cm4gd2lkdGg7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRIZWlnaHQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gaGVpZ2h0OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzRW1wdHkoKSB7Ci0gICAgICAgICAgICByZXR1cm4gd2lkdGggPD0gMC4wIHx8IGhlaWdodCA8PSAwLjA7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgc2V0UmVjdChkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3aWR0aCwgZG91YmxlIGhlaWdodCkgewotICAgICAgICAgICAgdGhpcy54ID0geDsKLSAgICAgICAgICAgIHRoaXMueSA9IHk7Ci0gICAgICAgICAgICB0aGlzLndpZHRoID0gd2lkdGg7Ci0gICAgICAgICAgICB0aGlzLmhlaWdodCA9IGhlaWdodDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRSZWN0KFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgICAgIHRoaXMueCA9IHIuZ2V0WCgpOwotICAgICAgICAgICAgdGhpcy55ID0gci5nZXRZKCk7Ci0gICAgICAgICAgICB0aGlzLndpZHRoID0gci5nZXRXaWR0aCgpOwotICAgICAgICAgICAgdGhpcy5oZWlnaHQgPSByLmdldEhlaWdodCgpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBpbnQgb3V0Y29kZShkb3VibGUgcHgsIGRvdWJsZSBweSkgewotICAgICAgICAgICAgaW50IGNvZGUgPSAwOwotCi0gICAgICAgICAgICBpZiAod2lkdGggPD0gMC4wKSB7Ci0gICAgICAgICAgICAgICAgY29kZSB8PSBPVVRfTEVGVCB8IE9VVF9SSUdIVDsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAocHggPCB4KSB7Ci0gICAgICAgICAgICAgICAgY29kZSB8PSBPVVRfTEVGVDsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAocHggPiB4ICsgd2lkdGgpIHsKLSAgICAgICAgICAgICAgICBjb2RlIHw9IE9VVF9SSUdIVDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKGhlaWdodCA8PSAwLjApIHsKLSAgICAgICAgICAgICAgICBjb2RlIHw9IE9VVF9UT1AgfCBPVVRfQk9UVE9NOwotICAgICAgICAgICAgfSBlbHNlIGlmIChweSA8IHkpIHsKLSAgICAgICAgICAgICAgICBjb2RlIHw9IE9VVF9UT1A7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKHB5ID4geSArIGhlaWdodCkgewotICAgICAgICAgICAgICAgIGNvZGUgfD0gT1VUX0JPVFRPTTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIGNvZGU7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBEb3VibGUoeCwgeSwgd2lkdGgsIGhlaWdodCk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFJlY3RhbmdsZTJEIGNyZWF0ZUludGVyc2VjdGlvbihSZWN0YW5nbGUyRCByKSB7Ci0gICAgICAgICAgICBSZWN0YW5nbGUyRCBkc3QgPSBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKCk7Ci0gICAgICAgICAgICBSZWN0YW5nbGUyRC5pbnRlcnNlY3QodGhpcywgciwgZHN0KTsKLSAgICAgICAgICAgIHJldHVybiBkc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIFJlY3RhbmdsZTJEIGNyZWF0ZVVuaW9uKFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgICAgIFJlY3RhbmdsZTJEIGRlc3QgPSBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKCk7Ci0gICAgICAgICAgICBSZWN0YW5nbGUyRC51bmlvbih0aGlzLCByLCBkZXN0KTsKLSAgICAgICAgICAgIHJldHVybiBkZXN0OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgICAgICAvLyBUaGUgb3V0cHV0IGZvcm1hdCBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvdXIuIEl0IGNvdWxkIGJlCi0gICAgICAgICAgICAvLyBvYnRhaW5lZCBpbiB0aGUgZm9sbG93aW5nIHdheQotICAgICAgICAgICAgLy8gU3lzdGVtLm91dC5wcmludGxuKG5ldyBSZWN0YW5nbGUyRC5Eb3VibGUoKS50b1N0cmluZygpKQotICAgICAgICAgICAgcmV0dXJuIGdldENsYXNzKCkuZ2V0TmFtZSgpCi0gICAgICAgICAgICAgICAgICAgICsgIlt4PSIgKyB4ICsgIix5PSIgKyB5ICsgIix3aWR0aD0iICsgd2lkdGggKyAiLGhlaWdodD0iICsgaGVpZ2h0ICsgIl0iOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJCAvLyROT04tTkxTLTQkIC8vJE5PTi1OTFMtNSQKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBJdGVyYXRvciBwcm92aWRlcyBhY2Nlc3MgdG8gdGhlIGNvb3JkaW5hdGVzIG9mIHRoZQotICAgICAqIFJlY3RhbmdsZTJEJ3MgYm91bmRhcnkgbW9kaWZpZWQgYnkgYW4gQWZmaW5lVHJhbnNmb3JtLgotICAgICAqLwotICAgIGNsYXNzIEl0ZXJhdG9yIGltcGxlbWVudHMgUGF0aEl0ZXJhdG9yIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgeDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgeTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgd2lkdGg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSBoZWlnaHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBBZmZpbmVUcmFuc2Zvcm0gdGhhdCBpcyB1c2VkIHRvIG1vZGlmeSB0aGUgY29vcmRpbmF0ZXMgdGhhdCBhcmUKLSAgICAgICAgICogcmV0dXJuZWQgYnkgdGhlIHBhdGggaXRlcmF0b3IuCi0gICAgICAgICAqLwotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gdDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGN1cnJlbnQgc2VnbWVudCBpbmRleC4KLSAgICAgICAgICovCi0gICAgICAgIGludCBpbmRleDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogQ29uc3RydWN0cyBhIG5ldyBSZWN0YW5nbGUyRC5JdGVyYXRvciBmb3IgZ2l2ZW4gcmVjdGFuZ2xlIGFuZAotICAgICAgICAgKiB0cmFuc2Zvcm1hdGlvbi4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSByCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBSZWN0YW5nbGUyRCBvYmplY3QuCi0gICAgICAgICAqIEBwYXJhbSBhdAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBBZmZpbmVUcmFuc2Zvcm0gb2JqZWN0IHRvIGFwcGx5IHRvIHRoZSBjb29yZGluYXRlcwotICAgICAgICAgKiAgICAgICAgICAgIGJlZm9yZSByZXR1cm5pbmcgdGhlbS4KLSAgICAgICAgICovCi0gICAgICAgIEl0ZXJhdG9yKFJlY3RhbmdsZTJEIHIsIEFmZmluZVRyYW5zZm9ybSBhdCkgewotICAgICAgICAgICAgdGhpcy54ID0gci5nZXRYKCk7Ci0gICAgICAgICAgICB0aGlzLnkgPSByLmdldFkoKTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSByLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICB0aGlzLmhlaWdodCA9IHIuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICB0aGlzLnQgPSBhdDsKLSAgICAgICAgICAgIGlmICh3aWR0aCA8IDAuMCB8fCBoZWlnaHQgPCAwLjApIHsKLSAgICAgICAgICAgICAgICBpbmRleCA9IDY7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGdldFdpbmRpbmdSdWxlKCkgewotICAgICAgICAgICAgcmV0dXJuIFdJTkRfTk9OX1pFUk87Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBpc0RvbmUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gaW5kZXggPiA1OwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIHZvaWQgbmV4dCgpIHsKLSAgICAgICAgICAgIGluZGV4Kys7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgaW50IGN1cnJlbnRTZWdtZW50KGRvdWJsZVtdIGNvb3JkcykgewotICAgICAgICAgICAgaWYgKGlzRG9uZSgpKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IE5vU3VjaEVsZW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNEIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChpbmRleCA9PSA1KSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIFNFR19DTE9TRTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGludCB0eXBlOwotICAgICAgICAgICAgaWYgKGluZGV4ID09IDApIHsKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX01PVkVUTzsKLSAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSB4OwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IHk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTElORVRPOwotICAgICAgICAgICAgICAgIHN3aXRjaCAoaW5kZXgpIHsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSAxOgotICAgICAgICAgICAgICAgICAgICAgICAgY29vcmRzWzBdID0geCArIHdpZHRoOwotICAgICAgICAgICAgICAgICAgICAgICAgY29vcmRzWzFdID0geTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlIDI6Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSB4ICsgd2lkdGg7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSB5ICsgaGVpZ2h0OwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgMzoKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IHg7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSB5ICsgaGVpZ2h0OwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgNDoKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IHg7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSB5OwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCAxKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB0eXBlOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGludCBjdXJyZW50U2VnbWVudChmbG9hdFtdIGNvb3JkcykgewotICAgICAgICAgICAgaWYgKGlzRG9uZSgpKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IE5vU3VjaEVsZW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNEIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChpbmRleCA9PSA1KSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIFNFR19DTE9TRTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGludCB0eXBlOwotICAgICAgICAgICAgaWYgKGluZGV4ID09IDApIHsKLSAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSAoZmxvYXQpeDsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSAoZmxvYXQpeTsKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX01PVkVUTzsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19MSU5FVE87Ci0gICAgICAgICAgICAgICAgc3dpdGNoIChpbmRleCkgewotICAgICAgICAgICAgICAgICAgICBjYXNlIDE6Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSAoZmxvYXQpKHggKyB3aWR0aCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSAoZmxvYXQpeTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlIDI6Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSAoZmxvYXQpKHggKyB3aWR0aCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSAoZmxvYXQpKHkgKyBoZWlnaHQpOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgMzoKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IChmbG9hdCl4OwotICAgICAgICAgICAgICAgICAgICAgICAgY29vcmRzWzFdID0gKGZsb2F0KSh5ICsgaGVpZ2h0KTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlIDQ6Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSAoZmxvYXQpeDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IChmbG9hdCl5OwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCAxKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB0eXBlOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgUmVjdGFuZ2xlMkQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFJlY3RhbmdsZTJEKCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHJlY3RhbmdsZSdzIGxvY2F0aW9uIGFuZCBkaW1lbnNpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzZXRSZWN0KGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvY2F0aW9uIG9mIHRoZSBwb2ludCB3aXRoIHJlc3BlY3QgdG8gdGhlIHJlY3RhbmdsZSBhbmQgcGFja3MKLSAgICAgKiB0aGUgaW5mb3JtYXRpb24gaW50byBhIHNpbmdsZSBpbnRlZ2VyIHVzaW5nIHRoZSBiaXRtYXNrcwotICAgICAqIHtAbGluayBSZWN0YW5nbGUyRCNPVVRfTEVGVH0sIHtAbGluayBSZWN0YW5nbGUyRCNPVVRfUklHSFR9LAotICAgICAqIHtAbGluayBSZWN0YW5nbGUyRCNPVVRfVE9QfSwgYW5kIHtAbGluayBSZWN0YW5nbGUyRCNPVVRfQk9UVE9NfS4gSWYgdGhlCi0gICAgICogcmVjdGFuZ2xlIGhhcyB6ZXJvIG9yIG5lZ2F0aXZlIHdpZHRoLCB0aGVuIGV2ZXJ5IHBvaW50IGlzIHJlZ2FyZGVkIGFzCi0gICAgICogYmVpbmcgYm90aCB0byB0aGUgbGVmdCBhbmQgdG8gdGhlIHJpZ2h0IG9mIHRoZSByZWN0YW5nbGUuIFNpbWlsYXJseSwgaWYKLSAgICAgKiB0aGUgaGVpZ2h0IGlzIHplcm8gb3IgbmVnYXRpdmUgdGhlbiBhbGwgcG9pbnRzIGFyZSBjb25zaWRlcmVkIHRvIGJlIGJvdGgKLSAgICAgKiBib3RoIGFib3ZlIGFuZCBiZWxvdyBpdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgcG9pbnQgdG8gY2hlY2suCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHBvaW50IHRvIGNoZWNrLgotICAgICAqIEByZXR1cm4gdGhlIHBvaW50J3MgbG9jYXRpb24gd2l0aCByZXNwZWN0IHRvIHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBvdXRjb2RlKGRvdWJsZSB4LCBkb3VibGUgeSk7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGFuIG5ldyByZWN0YW5nbGUgdGhhdCBpcyB0aGUgaW50ZXJzZWN0aW9uIG9mIHRoaXMgcmVjdGFuZ2xlIHdpdGgKLSAgICAgKiB0aGUgZ2l2ZW4gcmVjdGFuZ2xlLiBUaGUgcmVzdWx0aW5nIHJlY3RhbmdsZSBtYXkgYmUgZW1wdHkuIFRoZSBkYXRhIG9mCi0gICAgICogdGhpcyByZWN0YW5nbGUgaXMgbGVmdCB1bmNoYW5nZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSByZWN0YW5nbGUgdG8gaW50ZXJzZWN0IHdpdGggdGhpcyByZWN0YW5nbGUuCi0gICAgICogQHJldHVybiB0aGUgbmV3IHJlY3RhbmdsZSBnaXZlbiBieSBpbnRlcnNlY3Rpb24uCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFJlY3RhbmdsZTJEIGNyZWF0ZUludGVyc2VjdGlvbihSZWN0YW5nbGUyRCByKTsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYW4gbmV3IHJlY3RhbmdsZSB0aGF0IGlzIHRoZSB1bmlvbiBvZiB0aGlzIHJlY3RhbmdsZSB3aXRoIHRoZQotICAgICAqIGdpdmVuIHJlY3RhbmdsZS4gVGhlIG5ldyByZWN0YW5nbGUgaXMgdGhlIHNtYWxsZXN0IHJlY3RhbmdsZSB3aGljaAotICAgICAqIGNvbnRhaW5zIGJvdGggdGhpcyByZWN0YW5nbGUgYW5kIHRoZSByZWN0YW5nbGUgc3BlY2lmaWVkIGFzIGEgcGFyYW1ldGVyLgotICAgICAqIFRoZSBkYXRhIG9mIHRoaXMgcmVjdGFuZ2xlIGlzIGxlZnQgdW5jaGFuZ2VkLgotICAgICAqIAotICAgICAqIEBwYXJhbSByCi0gICAgICogICAgICAgICAgICB0aGUgcmVjdGFuZ2xlIHRvIGNvbWJpbmUgd2l0aCB0aGlzIHJlY3RhbmdsZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBuZXcgcmVjdGFuZ2xlIGdpdmVuIGJ5IHVuaW9uLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBSZWN0YW5nbGUyRCBjcmVhdGVVbmlvbihSZWN0YW5nbGUyRCByKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgb2YgdGhpcyByZWN0YW5nbGUgdG8gbWF0Y2ggdGhlIGRhdGEgb2YgdGhlIGdpdmVuIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcgotICAgICAqICAgICAgICAgICAgdGhlIHJlY3RhbmdsZSB3aG9zZSBkYXRhIGlzIHRvIGJlIGNvcGllZCBpbnRvIHRoaXMgcmVjdGFuZ2xlJ3MKLSAgICAgKiAgICAgICAgICAgIGZpZWxkcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRSZWN0KFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgc2V0UmVjdChyLmdldFgoKSwgci5nZXRZKCksIHIuZ2V0V2lkdGgoKSwgci5nZXRIZWlnaHQoKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0RnJhbWUoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQpIHsKLSAgICAgICAgc2V0UmVjdCh4LCB5LCB3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoKSB7Ci0gICAgICAgIHJldHVybiAoUmVjdGFuZ2xlMkQpY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXRlcm1pbmVzIHdoZXRoZXIgYW55IHBhcnQgb2YgdGhlIGxpbmUgc2VnbWVudCBiZXR3ZWVuIChhbmQgaW5jbHVkaW5nKQotICAgICAqIHRoZSB0d28gZ2l2ZW4gcG9pbnRzIHRvdWNoZXMgYW55IHBhcnQgb2YgdGhlIHJlY3RhbmdsZSwgaW5jbHVkaW5nIGl0cwotICAgICAqIGJvdW5kYXJ5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4MQotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiBvbmUgb2YgdGhlIHBvaW50cyB0aGF0IGRldGVybWluZXMgdGhlIGxpbmUKLSAgICAgKiAgICAgICAgICAgIHNlZ21lbnQgdG8gdGVzdC4KLSAgICAgKiBAcGFyYW0geTEKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2Ygb25lIG9mIHRoZSBwb2ludHMgdGhhdCBkZXRlcm1pbmVzIHRoZSBsaW5lCi0gICAgICogICAgICAgICAgICBzZWdtZW50IHRvIHRlc3QuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIG9uZSBvZiB0aGUgcG9pbnRzIHRoYXQgZGV0ZXJtaW5lcyB0aGUgbGluZQotICAgICAqICAgICAgICAgICAgc2VnbWVudCB0byB0ZXN0LgotICAgICAqIEBwYXJhbSB5MgotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiBvbmUgb2YgdGhlIHBvaW50cyB0aGF0IGRldGVybWluZXMgdGhlIGxpbmUKLSAgICAgKiAgICAgICAgICAgIHNlZ21lbnQgdG8gdGVzdC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGF0IGxlYXN0IG9uZSBwb2ludCBvZiB0aGUgbGluZSBzZWdtZW50IGJldHdlZW4gdGhlIHR3bwotICAgICAqICAgICAgICAgcG9pbnRzIG1hdGNoZXMgYW55IHBvaW50IG9mIHRoZSBpbnRlcmlvciBvZiB0aGUgcmVjdGFuZ2xlIG9yIHRoZQotICAgICAqICAgICAgICAgcmVjdGFuZ2xlJ3MgYm91bmRhcnkuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0c0xpbmUoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyKSB7Ci0gICAgICAgIGRvdWJsZSByeDEgPSBnZXRYKCk7Ci0gICAgICAgIGRvdWJsZSByeTEgPSBnZXRZKCk7Ci0gICAgICAgIGRvdWJsZSByeDIgPSByeDEgKyBnZXRXaWR0aCgpOwotICAgICAgICBkb3VibGUgcnkyID0gcnkxICsgZ2V0SGVpZ2h0KCk7Ci0gICAgICAgIHJldHVybiAocngxIDw9IHgxICYmIHgxIDw9IHJ4MiAmJiByeTEgPD0geTEgJiYgeTEgPD0gcnkyKQotICAgICAgICAgICAgICAgIHx8IChyeDEgPD0geDIgJiYgeDIgPD0gcngyICYmIHJ5MSA8PSB5MiAmJiB5MiA8PSByeTIpCi0gICAgICAgICAgICAgICAgfHwgTGluZTJELmxpbmVzSW50ZXJzZWN0KHJ4MSwgcnkxLCByeDIsIHJ5MiwgeDEsIHkxLCB4MiwgeTIpCi0gICAgICAgICAgICAgICAgfHwgTGluZTJELmxpbmVzSW50ZXJzZWN0KHJ4MiwgcnkxLCByeDEsIHJ5MiwgeDEsIHkxLCB4MiwgeTIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERldGVybWluZXMgd2hldGhlciBhbnkgcGFydCBvZiB0aGUgc3BlY2lmaWVkIGxpbmUgc2VnbWVudCB0b3VjaGVzIGFueQotICAgICAqIHBhcnQgb2YgdGhlIHJlY3RhbmdsZSwgaW5jbHVkaW5nIGl0cyBib3VuZGFyeS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbAotICAgICAqICAgICAgICAgICAgdGhlIGxpbmUgc2VnbWVudCB0byB0ZXN0LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgYXQgbGVhc3Qgb25lIHBvaW50IG9mIHRoZSBnaXZlbiBsaW5lIHNlZ21lbnQgbWF0Y2hlcyBhbnkKLSAgICAgKiAgICAgICAgIHBvaW50IG9mIHRoZSBpbnRlcmlvciBvZiB0aGUgcmVjdGFuZ2xlIG9yIHRoZSByZWN0YW5nbGUncwotICAgICAqICAgICAgICAgYm91bmRhcnkuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0c0xpbmUoTGluZTJEIGwpIHsKLSAgICAgICAgcmV0dXJuIGludGVyc2VjdHNMaW5lKGwuZ2V0WDEoKSwgbC5nZXRZMSgpLCBsLmdldFgyKCksIGwuZ2V0WTIoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9jYXRpb24gb2YgdGhlIHBvaW50IHdpdGggcmVzcGVjdCB0byB0aGUgcmVjdGFuZ2xlIGFuZCBwYWNrcwotICAgICAqIHRoZSBpbmZvcm1hdGlvbiBpbnRvIGEgc2luZ2xlIGludGVnZXIgdXNpbmcgdGhlIGJpdG1hc2tzCi0gICAgICoge0BsaW5rIFJlY3RhbmdsZTJEI09VVF9MRUZUfSwge0BsaW5rIFJlY3RhbmdsZTJEI09VVF9SSUdIVH0sCi0gICAgICoge0BsaW5rIFJlY3RhbmdsZTJEI09VVF9UT1B9LCBhbmQge0BsaW5rIFJlY3RhbmdsZTJEI09VVF9CT1RUT019LiBJZiB0aGUKLSAgICAgKiByZWN0YW5nbGUgaGFzIHplcm8gb3IgbmVnYXRpdmUgd2lkdGgsIHRoZW4gZXZlcnkgcG9pbnQgaXMgcmVnYXJkZWQgYXMKLSAgICAgKiBiZWluZyBib3RoIHRvIHRoZSBsZWZ0IGFuZCB0byB0aGUgcmlnaHQgb2YgdGhlIHJlY3RhbmdsZS4gU2ltaWxhcmx5LCBpZgotICAgICAqIHRoZSBoZWlnaHQgaXMgemVybyBvciBuZWdhdGl2ZSB0aGVuIGFsbCBwb2ludHMgYXJlIGNvbnNpZGVyZWQgdG8gYmUgYm90aAotICAgICAqIGJvdGggYWJvdmUgYW5kIGJlbG93IGl0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwCi0gICAgICogICAgICAgICAgICB0aGUgcG9pbnQgdG8gY2hlY2suCi0gICAgICogQHJldHVybiB0aGUgcG9pbnQncyBsb2NhdGlvbiB3aXRoIHJlc3BlY3QgdG8gdGhlIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IG91dGNvZGUoUG9pbnQyRCBwKSB7Ci0gICAgICAgIHJldHVybiBvdXRjb2RlKHAuZ2V0WCgpLCBwLmdldFkoKSk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0gICAgICAgIGlmIChpc0VtcHR5KCkpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIGRvdWJsZSB4MSA9IGdldFgoKTsKLSAgICAgICAgZG91YmxlIHkxID0gZ2V0WSgpOwotICAgICAgICBkb3VibGUgeDIgPSB4MSArIGdldFdpZHRoKCk7Ci0gICAgICAgIGRvdWJsZSB5MiA9IHkxICsgZ2V0SGVpZ2h0KCk7Ci0KLSAgICAgICAgcmV0dXJuIHgxIDw9IHggJiYgeCA8IHgyICYmIHkxIDw9IHkgJiYgeSA8IHkyOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHMoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQpIHsKLSAgICAgICAgaWYgKGlzRW1wdHkoKSB8fCB3aWR0aCA8PSAwLjAgfHwgaGVpZ2h0IDw9IDAuMCkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIHgxID0gZ2V0WCgpOwotICAgICAgICBkb3VibGUgeTEgPSBnZXRZKCk7Ci0gICAgICAgIGRvdWJsZSB4MiA9IHgxICsgZ2V0V2lkdGgoKTsKLSAgICAgICAgZG91YmxlIHkyID0geTEgKyBnZXRIZWlnaHQoKTsKLQotICAgICAgICByZXR1cm4geCArIHdpZHRoID4geDEgJiYgeCA8IHgyICYmIHkgKyBoZWlnaHQgPiB5MSAmJiB5IDwgeTI7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQpIHsKLSAgICAgICAgaWYgKGlzRW1wdHkoKSB8fCB3aWR0aCA8PSAwLjAgfHwgaGVpZ2h0IDw9IDAuMCkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIHgxID0gZ2V0WCgpOwotICAgICAgICBkb3VibGUgeTEgPSBnZXRZKCk7Ci0gICAgICAgIGRvdWJsZSB4MiA9IHgxICsgZ2V0V2lkdGgoKTsKLSAgICAgICAgZG91YmxlIHkyID0geTEgKyBnZXRIZWlnaHQoKTsKLQotICAgICAgICByZXR1cm4geDEgPD0geCAmJiB4ICsgd2lkdGggPD0geDIgJiYgeTEgPD0geSAmJiB5ICsgaGVpZ2h0IDw9IHkyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoYW5nZXMgdGhlIGRhdGEgdmFsdWVzIG9mIHRoZSBkZXN0aW5hdGlvbiByZWN0YW5nbGUgdG8gbWF0Y2ggdGhlCi0gICAgICogaW50ZXJzZWN0aW9uIG9mIHRoZSB0d28gc291cmNlIHJlY3RhbmdsZXMsIGxlYXZpbmcgdGhlIHR3byBzb3VyY2UKLSAgICAgKiByZWN0YW5nbGVzIHVuY2hhbmdlZC4gVGhlIHJlc3VsdGluZyByZWN0YW5nbGUgbWF5IGJlIGVtcHR5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMxCi0gICAgICogICAgICAgICAgICBvbmUgb2YgdGhlIHR3byBzb3VyY2UgcmVjdGFuZ2xlcyBnaXZpbmcgdGhlIGRhdGEgdG8gaW50ZXJzZWN0LgotICAgICAqIEBwYXJhbSBzcmMyCi0gICAgICogICAgICAgICAgICBvbmUgb2YgdGhlIHR3byBzb3VyY2UgcmVjdGFuZ2xlcyBnaXZpbmcgdGhlIGRhdGEgdG8gaW50ZXJzZWN0LgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXN0aW5hdGlvbiBvYmplY3Qgd2hlcmUgdGhlIGRhdGEgb2YgdGhlIGludGVyc2VjdGlvbiBpcwotICAgICAqICAgICAgICAgICAgd3JpdHRlbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgaW50ZXJzZWN0KFJlY3RhbmdsZTJEIHNyYzEsIFJlY3RhbmdsZTJEIHNyYzIsIFJlY3RhbmdsZTJEIGRzdCkgewotICAgICAgICBkb3VibGUgeDEgPSBNYXRoLm1heChzcmMxLmdldE1pblgoKSwgc3JjMi5nZXRNaW5YKCkpOwotICAgICAgICBkb3VibGUgeTEgPSBNYXRoLm1heChzcmMxLmdldE1pblkoKSwgc3JjMi5nZXRNaW5ZKCkpOwotICAgICAgICBkb3VibGUgeDIgPSBNYXRoLm1pbihzcmMxLmdldE1heFgoKSwgc3JjMi5nZXRNYXhYKCkpOwotICAgICAgICBkb3VibGUgeTIgPSBNYXRoLm1pbihzcmMxLmdldE1heFkoKSwgc3JjMi5nZXRNYXhZKCkpOwotICAgICAgICBkc3Quc2V0RnJhbWUoeDEsIHkxLCB4MiAtIHgxLCB5MiAtIHkxKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGFuZ2VzIHRoZSBkYXRhIHZhbHVlcyBvZiB0aGUgZGVzdGluYXRpb24gcmVjdGFuZ2xlIHRvIG1hdGNoIHRoZSB1bmlvbgotICAgICAqIG9mIHRoZSB0d28gc291cmNlIHJlY3RhbmdsZXMsIGxlYXZpbmcgdGhlIHR3byBzb3VyY2UgcmVjdGFuZ2xlcwotICAgICAqIHVuY2hhbmdlZC4gVGhlIHVuaW9uIGlzIHRoZSBzbWFsbGVzdCByZWN0YW5nbGUgdGhhdCBjb21wbGV0ZWx5IGNvdmVycyB0aGUKLSAgICAgKiB0d28gc291cmNlIHJlY3RhbmdsZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYzEKLSAgICAgKiAgICAgICAgICAgIG9uZSBvZiB0aGUgdHdvIHNvdXJjZSByZWN0YW5nbGVzIGdpdmluZyB0aGUgZGF0YS4KLSAgICAgKiBAcGFyYW0gc3JjMgotICAgICAqICAgICAgICAgICAgb25lIG9mIHRoZSB0d28gc291cmNlIHJlY3RhbmdsZXMgZ2l2aW5nIHRoZSBkYXRhLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXN0aW5hdGlvbiBvYmplY3Qgd2hlcmUgdGhlIGRhdGEgb2YgdGhlIHVuaW9uIGlzIHdyaXR0ZW4uCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyB2b2lkIHVuaW9uKFJlY3RhbmdsZTJEIHNyYzEsIFJlY3RhbmdsZTJEIHNyYzIsIFJlY3RhbmdsZTJEIGRzdCkgewotICAgICAgICBkb3VibGUgeDEgPSBNYXRoLm1pbihzcmMxLmdldE1pblgoKSwgc3JjMi5nZXRNaW5YKCkpOwotICAgICAgICBkb3VibGUgeTEgPSBNYXRoLm1pbihzcmMxLmdldE1pblkoKSwgc3JjMi5nZXRNaW5ZKCkpOwotICAgICAgICBkb3VibGUgeDIgPSBNYXRoLm1heChzcmMxLmdldE1heFgoKSwgc3JjMi5nZXRNYXhYKCkpOwotICAgICAgICBkb3VibGUgeTIgPSBNYXRoLm1heChzcmMxLmdldE1heFkoKSwgc3JjMi5nZXRNYXhZKCkpOwotICAgICAgICBkc3Quc2V0RnJhbWUoeDEsIHkxLCB4MiAtIHgxLCB5MiAtIHkxKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBFbmxhcmdlcyB0aGUgcmVjdGFuZ2xlIHNvIHRoYXQgaXQgaW5jbHVkZXMgdGhlIGdpdmVuIHBvaW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBuZXcgcG9pbnQgdG8gYmUgY292ZXJlZCBieSB0aGUKLSAgICAgKiAgICAgICAgICAgIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgbmV3IHBvaW50IHRvIGJlIGNvdmVyZWQgYnkgdGhlCi0gICAgICogICAgICAgICAgICByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkKGRvdWJsZSB4LCBkb3VibGUgeSkgewotICAgICAgICBkb3VibGUgeDEgPSBNYXRoLm1pbihnZXRNaW5YKCksIHgpOwotICAgICAgICBkb3VibGUgeTEgPSBNYXRoLm1pbihnZXRNaW5ZKCksIHkpOwotICAgICAgICBkb3VibGUgeDIgPSBNYXRoLm1heChnZXRNYXhYKCksIHgpOwotICAgICAgICBkb3VibGUgeTIgPSBNYXRoLm1heChnZXRNYXhZKCksIHkpOwotICAgICAgICBzZXRSZWN0KHgxLCB5MSwgeDIgLSB4MSwgeTIgLSB5MSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRW5sYXJnZXMgdGhlIHJlY3RhbmdsZSBzbyB0aGF0IGl0IGluY2x1ZGVzIHRoZSBnaXZlbiBwb2ludC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcAotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBwb2ludCB0byBiZSBjb3ZlcmVkIGJ5IHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkKFBvaW50MkQgcCkgewotICAgICAgICBhZGQocC5nZXRYKCksIHAuZ2V0WSgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBFbmxhcmdlcyB0aGUgcmVjdGFuZ2xlIHNvIHRoYXQgaXQgY292ZXJzIHRoZSBnaXZlbiByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgcmVjdGFuZ2xlIHRvIGJlIGNvdmVyZWQgYnkgdGhpcyByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkKFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgdW5pb24odGhpcywgciwgdGhpcyk7Ci0gICAgfQotCi0gICAgcHVibGljIFBhdGhJdGVyYXRvciBnZXRQYXRoSXRlcmF0b3IoQWZmaW5lVHJhbnNmb3JtIHQpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJdGVyYXRvcih0aGlzLCB0KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gdCwgZG91YmxlIGZsYXRuZXNzKSB7Ci0gICAgICAgIHJldHVybiBuZXcgSXRlcmF0b3IodGhpcywgdCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgSGFzaENvZGUgaGFzaCA9IG5ldyBIYXNoQ29kZSgpOwotICAgICAgICBoYXNoLmFwcGVuZChnZXRYKCkpOwotICAgICAgICBoYXNoLmFwcGVuZChnZXRZKCkpOwotICAgICAgICBoYXNoLmFwcGVuZChnZXRXaWR0aCgpKTsKLSAgICAgICAgaGFzaC5hcHBlbmQoZ2V0SGVpZ2h0KCkpOwotICAgICAgICByZXR1cm4gaGFzaC5oYXNoQ29kZSgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3Qgb2JqKSB7Ci0gICAgICAgIGlmIChvYmogPT0gdGhpcykgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG9iaiBpbnN0YW5jZW9mIFJlY3RhbmdsZTJEKSB7Ci0gICAgICAgICAgICBSZWN0YW5nbGUyRCByID0gKFJlY3RhbmdsZTJEKW9iajsKLSAgICAgICAgICAgIHJldHVybiBnZXRYKCkgPT0gci5nZXRYKCkgJiYgZ2V0WSgpID09IHIuZ2V0WSgpICYmIGdldFdpZHRoKCkgPT0gci5nZXRXaWR0aCgpCi0gICAgICAgICAgICAgICAgICAgICYmIGdldEhlaWdodCgpID09IHIuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2dlb20vUmVjdGFuZ3VsYXJTaGFwZS5qYXZhIGIvYXd0L2phdmEvYXd0L2dlb20vUmVjdGFuZ3VsYXJTaGFwZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwYjBkMDVjLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9nZW9tL1JlY3Rhbmd1bGFyU2hhcGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI5NyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERlbmlzIE0uIEtpc2hlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5nZW9tOwotCi1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LlNoYXBlOwotCi0vKioKLSAqIFRoZSBDbGFzcyBSZWN0YW5ndWxhclNoYXBlIHJlcHJlc2VudHMgYSBTaGFwZSB3aG9zZSBkYXRhIGlzIChhdCBsZWFzdAotICogcGFydGlhbGx5KSBkZXNjcmliZWQgYnkgYSByZWN0YW5ndWxhciBmcmFtZS4gVGhpcyBpbmNsdWRlcyBzaGFwZXMgd2hpY2ggYXJlCi0gKiBvYnZpb3VzbHkgcmVjdGFuZ3VsYXIgKHN1Y2ggYXMgUmVjdGFuZ2xlMkQpIGFzIHdlbGwgYXMgc2hhcGVzIGxpa2UgQXJjMkQKLSAqIHdoaWNoIGFyZSBsYXJnZWx5IGRldGVybWluZWQgYnkgdGhlIHJlY3RhbmdsZSB0aGV5IGZpdCBpbnNpZGUuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgUmVjdGFuZ3VsYXJTaGFwZSBpbXBsZW1lbnRzIFNoYXBlLCBDbG9uZWFibGUgewotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHJlY3Rhbmd1bGFyIHNoYXBlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBSZWN0YW5ndWxhclNoYXBlKCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZSBnZXRYKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0WSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0V2lkdGgoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0SGVpZ2h0KCk7Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhpcyBpcyBhbiBlbXB0eSByZWN0YW5nbGU6IG9uZSB3aXRoIHplcm8gYXMgaXRzIHdpZHRoIG9yCi0gICAgICogaGVpZ2h0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHdpZHRoIG9yIGhlaWdodCBpcyBlbXB0eS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgYm9vbGVhbiBpc0VtcHR5KCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBkYXRhIGZvciB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlIGluIHRlcm1zIG9mIGRvdWJsZSB2YWx1ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0RnJhbWUoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgdywgZG91YmxlIGgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWluaW11bSB4IHZhbHVlIG9mIHRoZSBib3VuZGluZyByZWN0YW5nbGUgKHRoZSB4IGNvb3JkaW5hdGUgb2YKLSAgICAgKiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIHJlY3RhbmdsZSkuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWluaW11bSB4IHZhbHVlIG9mIHRoZSBib3VuZGluZyByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRNaW5YKCkgewotICAgICAgICByZXR1cm4gZ2V0WCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1pbmltdW0geSB2YWx1ZSBvZiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlICh0aGUgeSBjb29yZGluYXRlIG9mCi0gICAgICogdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUpLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG1pbmltdW0geSB2YWx1ZSBvZiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlLgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgZ2V0TWluWSgpIHsKLSAgICAgICAgcmV0dXJuIGdldFkoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtYXhpbXVtIHggdmFsdWUgb2YgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZSAodGhlIHggY29vcmRpbmF0ZSBvZgotICAgICAqIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgcmVjdGFuZ2xlIHBsdXMgdGhlIHJlY3RhbmdsZSdzIHdpZHRoKS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtYXhpbXVtIHggdmFsdWUgb2YgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlIGdldE1heFgoKSB7Ci0gICAgICAgIHJldHVybiBnZXRYKCkgKyBnZXRXaWR0aCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1heGltdW0geSB2YWx1ZSBvZiB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlICh0aGUgeSBjb29yZGluYXRlIG9mCi0gICAgICogdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSByZWN0YW5nbGUgcGx1cyB0aGUgcmVjdGFuZ2xlJ3MgaGVpZ2h0KS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtYXhpbXVtIHkgdmFsdWUgb2YgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlIGdldE1heFkoKSB7Ci0gICAgICAgIHJldHVybiBnZXRZKCkgKyBnZXRIZWlnaHQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGNlbnRlciBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgY2VudGVyIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRDZW50ZXJYKCkgewotICAgICAgICByZXR1cm4gZ2V0WCgpICsgZ2V0V2lkdGgoKSAvIDIuMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIGNlbnRlciBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgY2VudGVyIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRDZW50ZXJZKCkgewotICAgICAgICByZXR1cm4gZ2V0WSgpICsgZ2V0SGVpZ2h0KCkgLyAyLjA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUGxhY2VzIHRoZSByZWN0YW5nbGUncyBzaXplIGFuZCBsb2NhdGlvbiBkYXRhIGluIGEgbmV3IFJlY3RhbmdsZTJEIG9iamVjdAotICAgICAqIGFuZCByZXR1cm5zIGl0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJvdW5kaW5nIHJlY3RhbmdsZSBhcyBhIG5ldyBSZWN0YW5nbGUyRCBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGdldEZyYW1lKCkgewotICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZTJELkRvdWJsZShnZXRYKCksIGdldFkoKSwgZ2V0V2lkdGgoKSwgZ2V0SGVpZ2h0KCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZSBpbiB0ZXJtcyBvZiBhIFBvaW50MkQgd2hpY2ggZ2l2ZXMgaXRzIHVwcGVyCi0gICAgICogbGVmdCBjb3JuZXIgYW5kIGEgRGltZW5zaW9uMkQgb2JqZWN0IGdpdmluZyBpdHMgd2lkdGggYW5kIGhlaWdodC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbG9jCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHVwcGVyIGxlZnQgY29ybmVyIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgc2l6ZSBkaW1lbnNpb25zLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEZyYW1lKFBvaW50MkQgbG9jLCBEaW1lbnNpb24yRCBzaXplKSB7Ci0gICAgICAgIHNldEZyYW1lKGxvYy5nZXRYKCksIGxvYy5nZXRZKCksIHNpemUuZ2V0V2lkdGgoKSwgc2l6ZS5nZXRIZWlnaHQoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgYm91bmRpbmcgcmVjdGFuZ2xlIHRvIG1hdGNoIHRoZSBkYXRhIGNvbnRhaW5lZCBpbiB0aGUgc3BlY2lmaWVkCi0gICAgICogUmVjdGFuZ2xlMkQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSByZWN0YW5nbGUgdGhhdCBnaXZlcyB0aGUgbmV3IGZyYW1lIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RnJhbWUoUmVjdGFuZ2xlMkQgcikgewotICAgICAgICBzZXRGcmFtZShyLmdldFgoKSwgci5nZXRZKCksIHIuZ2V0V2lkdGgoKSwgci5nZXRIZWlnaHQoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZnJhbWluZyByZWN0YW5nbGUgZ2l2ZW4gdHdvIG9wcG9zaXRlIGNvcm5lcnMuIEFueSB0d28gY29ybmVycwotICAgICAqIG1heSBiZSB1c2VkIGluIGFueSBvcmRlciBhcyBsb25nIGFzIHRoZXkgYXJlIGRpYWdvbmFsbHkgb3Bwb3NpdGUgb25lCi0gICAgICogYW5vdGhlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geDEKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2Ygb25lIG9mIHRoZSBjb3JuZXIgcG9pbnRzLgotICAgICAqIEBwYXJhbSB5MQotICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiBvbmUgb2YgdGhlIGNvcm5lciBwb2ludHMuCi0gICAgICogQHBhcmFtIHgyCi0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBvdGhlciBjb3JuZXIgcG9pbnQuCi0gICAgICogQHBhcmFtIHkyCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBvdGhlciBjb3JuZXIgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RnJhbWVGcm9tRGlhZ29uYWwoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSB4MiwgZG91YmxlIHkyKSB7Ci0gICAgICAgIGRvdWJsZSByeCwgcnksIHJ3LCByaDsKLSAgICAgICAgaWYgKHgxIDwgeDIpIHsKLSAgICAgICAgICAgIHJ4ID0geDE7Ci0gICAgICAgICAgICBydyA9IHgyIC0geDE7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICByeCA9IHgyOwotICAgICAgICAgICAgcncgPSB4MSAtIHgyOwotICAgICAgICB9Ci0gICAgICAgIGlmICh5MSA8IHkyKSB7Ci0gICAgICAgICAgICByeSA9IHkxOwotICAgICAgICAgICAgcmggPSB5MiAtIHkxOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcnkgPSB5MjsKLSAgICAgICAgICAgIHJoID0geTEgLSB5MjsKLSAgICAgICAgfQotICAgICAgICBzZXRGcmFtZShyeCwgcnksIHJ3LCByaCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZnJhbWluZyByZWN0YW5nbGUgZ2l2ZW4gdHdvIG9wcG9zaXRlIGNvcm5lcnMuIEFueSB0d28gY29ybmVycwotICAgICAqIG1heSBiZSB1c2VkIGluIGFueSBvcmRlciBhcyBsb25nIGFzIHRoZXkgYXJlIGRpYWdvbmFsbHkgb3Bwb3NpdGUgb25lCi0gICAgICogYW5vdGhlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcDEKLSAgICAgKiAgICAgICAgICAgIG9uZSBvZiB0aGUgY29ybmVyIHBvaW50cy4KLSAgICAgKiBAcGFyYW0gcDIKLSAgICAgKiAgICAgICAgICAgIHRoZSBvdGhlciBjb3JuZXIgcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RnJhbWVGcm9tRGlhZ29uYWwoUG9pbnQyRCBwMSwgUG9pbnQyRCBwMikgewotICAgICAgICBzZXRGcmFtZUZyb21EaWFnb25hbChwMS5nZXRYKCksIHAxLmdldFkoKSwgcDIuZ2V0WCgpLCBwMi5nZXRZKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGZyYW1pbmcgcmVjdGFuZ2xlIGdpdmVuIHRoZSBjZW50ZXIgcG9pbnQgYW5kIG9uZSBjb3JuZXIuIEFueQotICAgICAqIGNvcm5lciBtYXkgYmUgdXNlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2VudGVyWAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgY2VudGVyIHBvaW50LgotICAgICAqIEBwYXJhbSBjZW50ZXJZCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSBjZW50ZXIgcG9pbnQuCi0gICAgICogQHBhcmFtIGNvcm5lclgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2Ygb25lIG9mIHRoZSBjb3JuZXIgcG9pbnRzLgotICAgICAqIEBwYXJhbSBjb3JuZXJZCi0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIG9uZSBvZiB0aGUgY29ybmVyIHBvaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRGcmFtZUZyb21DZW50ZXIoZG91YmxlIGNlbnRlclgsIGRvdWJsZSBjZW50ZXJZLCBkb3VibGUgY29ybmVyWCwgZG91YmxlIGNvcm5lclkpIHsKLSAgICAgICAgZG91YmxlIHdpZHRoID0gTWF0aC5hYnMoY29ybmVyWCAtIGNlbnRlclgpOwotICAgICAgICBkb3VibGUgaGVpZ2h0ID0gTWF0aC5hYnMoY29ybmVyWSAtIGNlbnRlclkpOwotICAgICAgICBzZXRGcmFtZShjZW50ZXJYIC0gd2lkdGgsIGNlbnRlclkgLSBoZWlnaHQsIHdpZHRoICogMi4wLCBoZWlnaHQgKiAyLjApOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGZyYW1pbmcgcmVjdGFuZ2xlIGdpdmVuIHRoZSBjZW50ZXIgcG9pbnQgYW5kIG9uZSBjb3JuZXIuIEFueQotICAgICAqIGNvcm5lciBtYXkgYmUgdXNlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2VudGVyCi0gICAgICogICAgICAgICAgICB0aGUgY2VudGVyIHBvaW50LgotICAgICAqIEBwYXJhbSBjb3JuZXIKLSAgICAgKiAgICAgICAgICAgIGEgY29ybmVyIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEZyYW1lRnJvbUNlbnRlcihQb2ludDJEIGNlbnRlciwgUG9pbnQyRCBjb3JuZXIpIHsKLSAgICAgICAgc2V0RnJhbWVGcm9tQ2VudGVyKGNlbnRlci5nZXRYKCksIGNlbnRlci5nZXRZKCksIGNvcm5lci5nZXRYKCksIGNvcm5lci5nZXRZKCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFBvaW50MkQgcG9pbnQpIHsKLSAgICAgICAgcmV0dXJuIGNvbnRhaW5zKHBvaW50LmdldFgoKSwgcG9pbnQuZ2V0WSgpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpbnRlcnNlY3RzKFJlY3RhbmdsZTJEIHJlY3QpIHsKLSAgICAgICAgcmV0dXJuIGludGVyc2VjdHMocmVjdC5nZXRYKCksIHJlY3QuZ2V0WSgpLCByZWN0LmdldFdpZHRoKCksIHJlY3QuZ2V0SGVpZ2h0KCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFJlY3RhbmdsZTJEIHJlY3QpIHsKLSAgICAgICAgcmV0dXJuIGNvbnRhaW5zKHJlY3QuZ2V0WCgpLCByZWN0LmdldFkoKSwgcmVjdC5nZXRXaWR0aCgpLCByZWN0LmdldEhlaWdodCgpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldEJvdW5kcygpIHsKLSAgICAgICAgaW50IHgxID0gKGludClNYXRoLmZsb29yKGdldE1pblgoKSk7Ci0gICAgICAgIGludCB5MSA9IChpbnQpTWF0aC5mbG9vcihnZXRNaW5ZKCkpOwotICAgICAgICBpbnQgeDIgPSAoaW50KU1hdGguY2VpbChnZXRNYXhYKCkpOwotICAgICAgICBpbnQgeTIgPSAoaW50KU1hdGguY2VpbChnZXRNYXhZKCkpOwotICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZSh4MSwgeTEsIHgyIC0geDEsIHkyIC0geTEpOwotICAgIH0KLQotICAgIHB1YmxpYyBQYXRoSXRlcmF0b3IgZ2V0UGF0aEl0ZXJhdG9yKEFmZmluZVRyYW5zZm9ybSB0LCBkb3VibGUgZmxhdG5lc3MpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBGbGF0dGVuaW5nUGF0aEl0ZXJhdG9yKGdldFBhdGhJdGVyYXRvcih0KSwgZmxhdG5lc3MpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgY2xvbmUoKSB7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gc3VwZXIuY2xvbmUoKTsKLSAgICAgICAgfSBjYXRjaCAoQ2xvbmVOb3RTdXBwb3J0ZWRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoKTsKLSAgICAgICAgfQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2dlb20vUm91bmRSZWN0YW5nbGUyRC5qYXZhIGIvYXd0L2phdmEvYXd0L2dlb20vUm91bmRSZWN0YW5nbGUyRC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4ZmJkZGQ2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9nZW9tL1JvdW5kUmVjdGFuZ2xlMkQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDYzNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERlbmlzIE0uIEtpc2hlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5nZW9tOwotCi1pbXBvcnQgamF2YS51dGlsLk5vU3VjaEVsZW1lbnRFeGNlcHRpb247Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgUm91bmRSZWN0YW5nbGUyRCBkZXNjcmliZXMgYSByZWN0YW5nbGUgd2l0aCByb3VuZGVkIGNvcm5lcnMgd2l0aAotICogaGlnaC1wcmVjaXNpb24gZGF0YSB0aGF0IGlzIGFwcHJvcHJpYXRlIGZvciBnZW9tZXRyaWMgb3BlcmF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBSb3VuZFJlY3RhbmdsZTJEIGV4dGVuZHMgUmVjdGFuZ3VsYXJTaGFwZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgRmxvYXQgaXMgdGhlIHN1YmNsYXNzIG9mIFJvdW5kUmVjdGFuZ2xlMkQgdGhhdCBoYXMgYWxsIG9mIGl0cwotICAgICAqIGRhdGEgdmFsdWVzIHN0b3JlZCB3aXRoIGZsb2F0LWxldmVsIHByZWNpc2lvbi4KLSAgICAgKiAKLSAgICAgKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGNsYXNzIEZsb2F0IGV4dGVuZHMgUm91bmRSZWN0YW5nbGUyRCB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGZsb2F0IHdpZHRoOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgaGVpZ2h0OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgYXJjIHdpZHRoIG9mIHRoZSByb3VuZGVkIGNvcm5lcnMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgYXJjd2lkdGg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBhcmMgaGVpZ2h0IG9mIHRoZSByb3VuZGVkIGNvcm5lcnMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZmxvYXQgYXJjaGVpZ2h0OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZmxvYXQtdmFsdWVkIFJvdW5kUmVjdGFuZ2xlMkQgd2l0aCBpdHMgZGF0YS12YWx1ZXMKLSAgICAgICAgICogc2V0IHRvIHplcm8uCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRmxvYXQoKSB7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGZsb2F0LXZhbHVlZCBSb3VuZFJlY3RhbmdsZTJEIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAgICAgKiBkYXRhIHZhbHVlcy4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSB4Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICAgICAqIEBwYXJhbSB5Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAgICAgKiBAcGFyYW0gYXJjd2lkdGgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgYXJjIHdpZHRoIG9mIHRoZSByb3VuZGVkIGNvcm5lcnMuCi0gICAgICAgICAqIEBwYXJhbSBhcmNoZWlnaHQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgYXJjIGhlaWdodCBvZiB0aGUgcm91bmRlZCBjb3JuZXJzLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEZsb2F0KGZsb2F0IHgsIGZsb2F0IHksIGZsb2F0IHdpZHRoLCBmbG9hdCBoZWlnaHQsIGZsb2F0IGFyY3dpZHRoLCBmbG9hdCBhcmNoZWlnaHQpIHsKLSAgICAgICAgICAgIHNldFJvdW5kUmVjdCh4LCB5LCB3aWR0aCwgaGVpZ2h0LCBhcmN3aWR0aCwgYXJjaGVpZ2h0KTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFgoKSB7Ci0gICAgICAgICAgICByZXR1cm4geDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFkoKSB7Ci0gICAgICAgICAgICByZXR1cm4geTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldFdpZHRoKCkgewotICAgICAgICAgICAgcmV0dXJuIHdpZHRoOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0SGVpZ2h0KCkgewotICAgICAgICAgICAgcmV0dXJuIGhlaWdodDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldEFyY1dpZHRoKCkgewotICAgICAgICAgICAgcmV0dXJuIGFyY3dpZHRoOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBkb3VibGUgZ2V0QXJjSGVpZ2h0KCkgewotICAgICAgICAgICAgcmV0dXJuIGFyY2hlaWdodDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBpc0VtcHR5KCkgewotICAgICAgICAgICAgcmV0dXJuIHdpZHRoIDw9IDAuMGYgfHwgaGVpZ2h0IDw9IDAuMGY7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogU2V0cyB0aGUgZGF0YSBvZiB0aGUgcm91bmQgcmVjdGFuZ2xlLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgICAgICogQHBhcmFtIHkKLSAgICAgICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgICAgICogQHBhcmFtIHdpZHRoCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICAgICAqIEBwYXJhbSBhcmN3aWR0aAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBhcmMgd2lkdGggb2YgdGhlIHJvdW5kZWQgY29ybmVycy4KLSAgICAgICAgICogQHBhcmFtIGFyY2hlaWdodAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBhcmMgaGVpZ2h0IG9mIHRoZSByb3VuZGVkIGNvcm5lcnMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgdm9pZCBzZXRSb3VuZFJlY3QoZmxvYXQgeCwgZmxvYXQgeSwgZmxvYXQgd2lkdGgsIGZsb2F0IGhlaWdodCwgZmxvYXQgYXJjd2lkdGgsCi0gICAgICAgICAgICAgICAgZmxvYXQgYXJjaGVpZ2h0KSB7Ci0gICAgICAgICAgICB0aGlzLnggPSB4OwotICAgICAgICAgICAgdGhpcy55ID0geTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSB3aWR0aDsKLSAgICAgICAgICAgIHRoaXMuaGVpZ2h0ID0gaGVpZ2h0OwotICAgICAgICAgICAgdGhpcy5hcmN3aWR0aCA9IGFyY3dpZHRoOwotICAgICAgICAgICAgdGhpcy5hcmNoZWlnaHQgPSBhcmNoZWlnaHQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgc2V0Um91bmRSZWN0KGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0LCBkb3VibGUgYXJjd2lkdGgsCi0gICAgICAgICAgICAgICAgZG91YmxlIGFyY2hlaWdodCkgewotICAgICAgICAgICAgdGhpcy54ID0gKGZsb2F0KXg7Ci0gICAgICAgICAgICB0aGlzLnkgPSAoZmxvYXQpeTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSAoZmxvYXQpd2lkdGg7Ci0gICAgICAgICAgICB0aGlzLmhlaWdodCA9IChmbG9hdCloZWlnaHQ7Ci0gICAgICAgICAgICB0aGlzLmFyY3dpZHRoID0gKGZsb2F0KWFyY3dpZHRoOwotICAgICAgICAgICAgdGhpcy5hcmNoZWlnaHQgPSAoZmxvYXQpYXJjaGVpZ2h0OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyB2b2lkIHNldFJvdW5kUmVjdChSb3VuZFJlY3RhbmdsZTJEIHJyKSB7Ci0gICAgICAgICAgICB0aGlzLnggPSAoZmxvYXQpcnIuZ2V0WCgpOwotICAgICAgICAgICAgdGhpcy55ID0gKGZsb2F0KXJyLmdldFkoKTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSAoZmxvYXQpcnIuZ2V0V2lkdGgoKTsKLSAgICAgICAgICAgIHRoaXMuaGVpZ2h0ID0gKGZsb2F0KXJyLmdldEhlaWdodCgpOwotICAgICAgICAgICAgdGhpcy5hcmN3aWR0aCA9IChmbG9hdClyci5nZXRBcmNXaWR0aCgpOwotICAgICAgICAgICAgdGhpcy5hcmNoZWlnaHQgPSAoZmxvYXQpcnIuZ2V0QXJjSGVpZ2h0KCk7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZTJELkZsb2F0KHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIERvdWJsZSBpcyB0aGUgc3ViY2xhc3Mgb2YgUm91bmRSZWN0YW5nbGUyRCB0aGF0IGhhcyBhbGwgb2YgaXRzCi0gICAgICogZGF0YSB2YWx1ZXMgc3RvcmVkIHdpdGggZG91YmxlLWxldmVsIHByZWNpc2lvbi4KLSAgICAgKiAKLSAgICAgKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGNsYXNzIERvdWJsZSBleHRlbmRzIFJvdW5kUmVjdGFuZ2xlMkQgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUncyB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgeDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIHk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSB3aWR0aDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGRvdWJsZSBoZWlnaHQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBhcmMgd2lkdGggb2YgdGhlIHJvdW5kZWQgY29ybmVycy4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBkb3VibGUgYXJjd2lkdGg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBhcmMgaGVpZ2h0IG9mIHRoZSByb3VuZGVkIGNvcm5lcnMuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgZG91YmxlIGFyY2hlaWdodDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRvdWJsZS12YWx1ZWQgUm91bmRSZWN0YW5nbGUyRCB3aXRoIGl0cwotICAgICAgICAgKiBkYXRhLXZhbHVlcyBzZXQgdG8gemVyby4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBEb3VibGUoKSB7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRvdWJsZS12YWx1ZWQgUm91bmRSZWN0YW5nbGUyRCB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgICAgICogZGF0YSB2YWx1ZXMuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0geAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAgICAgKiBAcGFyYW0geQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICogQHBhcmFtIGhlaWdodAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgICAgICogQHBhcmFtIGFyY3dpZHRoCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGFyYyB3aWR0aCBvZiB0aGUgcm91bmRlZCBjb3JuZXJzLgotICAgICAgICAgKiBAcGFyYW0gYXJjaGVpZ2h0Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGFyYyBoZWlnaHQgb2YgdGhlIHJvdW5kZWQgY29ybmVycy4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBEb3VibGUoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQsIGRvdWJsZSBhcmN3aWR0aCwKLSAgICAgICAgICAgICAgICBkb3VibGUgYXJjaGVpZ2h0KSB7Ci0gICAgICAgICAgICBzZXRSb3VuZFJlY3QoeCwgeSwgd2lkdGgsIGhlaWdodCwgYXJjd2lkdGgsIGFyY2hlaWdodCk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRYKCkgewotICAgICAgICAgICAgcmV0dXJuIHg7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRZKCkgewotICAgICAgICAgICAgcmV0dXJuIHk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRXaWR0aCgpIHsKLSAgICAgICAgICAgIHJldHVybiB3aWR0aDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldEhlaWdodCgpIHsKLSAgICAgICAgICAgIHJldHVybiBoZWlnaHQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGRvdWJsZSBnZXRBcmNXaWR0aCgpIHsKLSAgICAgICAgICAgIHJldHVybiBhcmN3aWR0aDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgZG91YmxlIGdldEFyY0hlaWdodCgpIHsKLSAgICAgICAgICAgIHJldHVybiBhcmNoZWlnaHQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGJvb2xlYW4gaXNFbXB0eSgpIHsKLSAgICAgICAgICAgIHJldHVybiB3aWR0aCA8PSAwLjAgfHwgaGVpZ2h0IDw9IDAuMDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgdm9pZCBzZXRSb3VuZFJlY3QoZG91YmxlIHgsIGRvdWJsZSB5LCBkb3VibGUgd2lkdGgsIGRvdWJsZSBoZWlnaHQsIGRvdWJsZSBhcmN3aWR0aCwKLSAgICAgICAgICAgICAgICBkb3VibGUgYXJjaGVpZ2h0KSB7Ci0gICAgICAgICAgICB0aGlzLnggPSB4OwotICAgICAgICAgICAgdGhpcy55ID0geTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSB3aWR0aDsKLSAgICAgICAgICAgIHRoaXMuaGVpZ2h0ID0gaGVpZ2h0OwotICAgICAgICAgICAgdGhpcy5hcmN3aWR0aCA9IGFyY3dpZHRoOwotICAgICAgICAgICAgdGhpcy5hcmNoZWlnaHQgPSBhcmNoZWlnaHQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgc2V0Um91bmRSZWN0KFJvdW5kUmVjdGFuZ2xlMkQgcnIpIHsKLSAgICAgICAgICAgIHRoaXMueCA9IHJyLmdldFgoKTsKLSAgICAgICAgICAgIHRoaXMueSA9IHJyLmdldFkoKTsKLSAgICAgICAgICAgIHRoaXMud2lkdGggPSByci5nZXRXaWR0aCgpOwotICAgICAgICAgICAgdGhpcy5oZWlnaHQgPSByci5nZXRIZWlnaHQoKTsKLSAgICAgICAgICAgIHRoaXMuYXJjd2lkdGggPSByci5nZXRBcmNXaWR0aCgpOwotICAgICAgICAgICAgdGhpcy5hcmNoZWlnaHQgPSByci5nZXRBcmNIZWlnaHQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRCgpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBSb3VuZFJlY3RhbmdsZTJEIHBhdGggaXRlcmF0b3IKLSAgICAgKi8KLSAgICAvKioKLSAgICAgKiBUaGUgc3ViY2xhc3Mgb2YgUGF0aEl0ZXJhdG9yIHRvIHRyYXZlcnNlIGEgUm91bmRSZWN0YW5nbGUyRC4KLSAgICAgKi8KLSAgICBjbGFzcyBJdGVyYXRvciBpbXBsZW1lbnRzIFBhdGhJdGVyYXRvciB7Ci0KLSAgICAgICAgLyoKLSAgICAgICAgICogUGF0aCBmb3Igcm91bmQgY29ybmVycyBnZW5lcmF0ZWQgdGhlIHNhbWUgd2F5IGFzIEVsbGlwc2UyRAotICAgICAgICAgKi8KLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGNvZWZmaWNpZW50IHRvIGNhbGN1bGF0ZSBjb250cm9sIHBvaW50cyBvZiBCZXppZXIgY3VydmVzLgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIHUgPSAwLjUgLSAyLjAgLyAzLjAgKiAoTWF0aC5zcXJ0KDIuMCkgLSAxLjApOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcG9pbnRzIGNvb3JkaW5hdGVzIGNhbGN1bGF0aW9uIHRhYmxlLgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIHBvaW50c1tdW10gPSB7Ci0gICAgICAgICAgICAgICAgewotICAgICAgICAgICAgICAgICAgICAgICAgMC4wLCAwLjUsIDAuMCwgMC4wCi0gICAgICAgICAgICAgICAgfSwgLy8gTU9WRVRPCi0gICAgICAgICAgICAgICAgewotICAgICAgICAgICAgICAgICAgICAgICAgMS4wLCAtMC41LCAwLjAsIDAuMAotICAgICAgICAgICAgICAgIH0sIC8vIExJTkVUTwotICAgICAgICAgICAgICAgIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIDEuMCwgLXUsIDAuMCwgMC4wLCAvLyBDVUJJQ1RPCi0gICAgICAgICAgICAgICAgICAgICAgICAxLjAsIDAuMCwgMC4wLCB1LCAxLjAsIDAuMCwgMC4wLCAwLjUKLSAgICAgICAgICAgICAgICB9LCB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAxLjAsIDAuMCwgMS4wLCAtMC41Ci0gICAgICAgICAgICAgICAgfSwgLy8gTElORVRPCi0gICAgICAgICAgICAgICAgewotICAgICAgICAgICAgICAgICAgICAgICAgMS4wLCAwLjAsIDEuMCwgLXUsIC8vIENVQklDVE8KLSAgICAgICAgICAgICAgICAgICAgICAgIDEuMCwgLXUsIDEuMCwgMC4wLCAxLjAsIC0wLjUsIDEuMCwgMC4wCi0gICAgICAgICAgICAgICAgfSwgewotICAgICAgICAgICAgICAgICAgICAgICAgMC4wLCAwLjUsIDEuMCwgMC4wCi0gICAgICAgICAgICAgICAgfSwgLy8gTElORVRPCi0gICAgICAgICAgICAgICAgewotICAgICAgICAgICAgICAgICAgICAgICAgMC4wLCB1LCAxLjAsIDAuMCwgLy8gQ1VCSUNUTwotICAgICAgICAgICAgICAgICAgICAgICAgMC4wLCAwLjAsIDEuMCwgLXUsIDAuMCwgMC4wLCAxLjAsIC0wLjUKLSAgICAgICAgICAgICAgICB9LCB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAwLjAsIDAuMCwgMC4wLCAwLjUKLSAgICAgICAgICAgICAgICB9LCAvLyBMSU5FVE8KLSAgICAgICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAwLjAsIDAuMCwgMC4wLCB1LCAvLyBDVUJJQ1RPCi0gICAgICAgICAgICAgICAgICAgICAgICAwLjAsIHUsIDAuMCwgMC4wLCAwLjAsIDAuNSwgMC4wLCAwLjAKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgIH07Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzZWdtZW50IHR5cGVzIGNvcnJlc3BvbmQgdG8gcG9pbnRzIGFycmF5LgotICAgICAgICAgKi8KLSAgICAgICAgaW50IHR5cGVzW10gPSB7Ci0gICAgICAgICAgICAgICAgU0VHX01PVkVUTywgU0VHX0xJTkVUTywgU0VHX0NVQklDVE8sIFNFR19MSU5FVE8sIFNFR19DVUJJQ1RPLCBTRUdfTElORVRPLAotICAgICAgICAgICAgICAgIFNFR19DVUJJQ1RPLCBTRUdfTElORVRPLCBTRUdfQ1VCSUNUTwotICAgICAgICB9OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgeCBjb29yZGluYXRlIG9mIGxlZnQtdXBwZXIgY29ybmVyIG9mIHRoZSByb3VuZCByZWN0YW5nbGUgYm91bmRzLgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIHg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSB5IGNvb3JkaW5hdGUgb2YgbGVmdC11cHBlciBjb3JuZXIgb2YgdGhlIHJvdW5kIHJlY3RhbmdsZSBib3VuZHMuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgeTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHdpZHRoIG9mIHRoZSByb3VuZCByZWN0YW5nbGUgYm91bmRzLgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIHdpZHRoOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgaGVpZ2h0IG9mIHRoZSByb3VuZCByZWN0YW5nbGUgYm91bmRzLgotICAgICAgICAgKi8KLSAgICAgICAgZG91YmxlIGhlaWdodDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHdpZHRoIG9mIGFyYyBjb3JuZXJzIG9mIHRoZSByb3VuZCByZWN0YW5nbGUuCi0gICAgICAgICAqLwotICAgICAgICBkb3VibGUgYXc7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBoZWlnaHQgb2YgYXJjIGNvcm5lcnMgb2YgdGhlIHJvdW5kIHJlY3RhbmdsZS4KLSAgICAgICAgICovCi0gICAgICAgIGRvdWJsZSBhaDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHBhdGggaXRlcmF0b3IgdHJhbnNmb3JtYXRpb24uCi0gICAgICAgICAqLwotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gdDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGN1cnJlbnQgc2VnbWVudCBpbmRleC4KLSAgICAgICAgICovCi0gICAgICAgIGludCBpbmRleDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogQ29uc3RydWN0cyBhIG5ldyBSb3VuZFJlY3RhbmdsZTJELkl0ZXJhdG9yIGZvciBnaXZlbiByb3VuZCByZWN0YW5nbGUKLSAgICAgICAgICogYW5kIHRyYW5zZm9ybWF0aW9uLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHJyCi0gICAgICAgICAqICAgICAgICAgICAgLSB0aGUgc291cmNlIFJvdW5kUmVjdGFuZ2xlMkQgb2JqZWN0Ci0gICAgICAgICAqIEBwYXJhbSBhdAotICAgICAgICAgKiAgICAgICAgICAgIC0gdGhlIEFmZmluZVRyYW5zZm9ybSBvYmplY3QgdG8gYXBwbHkgcmVjdGFuZ2xlIHBhdGgKLSAgICAgICAgICovCi0gICAgICAgIEl0ZXJhdG9yKFJvdW5kUmVjdGFuZ2xlMkQgcnIsIEFmZmluZVRyYW5zZm9ybSBhdCkgewotICAgICAgICAgICAgdGhpcy54ID0gcnIuZ2V0WCgpOwotICAgICAgICAgICAgdGhpcy55ID0gcnIuZ2V0WSgpOwotICAgICAgICAgICAgdGhpcy53aWR0aCA9IHJyLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICB0aGlzLmhlaWdodCA9IHJyLmdldEhlaWdodCgpOwotICAgICAgICAgICAgdGhpcy5hdyA9IE1hdGgubWluKHdpZHRoLCByci5nZXRBcmNXaWR0aCgpKTsKLSAgICAgICAgICAgIHRoaXMuYWggPSBNYXRoLm1pbihoZWlnaHQsIHJyLmdldEFyY0hlaWdodCgpKTsKLSAgICAgICAgICAgIHRoaXMudCA9IGF0OwotICAgICAgICAgICAgaWYgKHdpZHRoIDwgMC4wIHx8IGhlaWdodCA8IDAuMCB8fCBhdyA8IDAuMCB8fCBhaCA8IDAuMCkgewotICAgICAgICAgICAgICAgIGluZGV4ID0gcG9pbnRzLmxlbmd0aDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgZ2V0V2luZGluZ1J1bGUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gV0lORF9OT05fWkVSTzsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzRG9uZSgpIHsKLSAgICAgICAgICAgIHJldHVybiBpbmRleCA+IHBvaW50cy5sZW5ndGg7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgdm9pZCBuZXh0KCkgewotICAgICAgICAgICAgaW5kZXgrKzsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgY3VycmVudFNlZ21lbnQoZG91YmxlW10gY29vcmRzKSB7Ci0gICAgICAgICAgICBpZiAoaXNEb25lKCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuNEI9SXRlcmF0b3Igb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBOb1N1Y2hFbGVtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoaW5kZXggPT0gcG9pbnRzLmxlbmd0aCkgewotICAgICAgICAgICAgICAgIHJldHVybiBTRUdfQ0xPU0U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbnQgaiA9IDA7Ci0gICAgICAgICAgICBkb3VibGUgcFtdID0gcG9pbnRzW2luZGV4XTsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcC5sZW5ndGg7IGkgKz0gNCkgewotICAgICAgICAgICAgICAgIGNvb3Jkc1tqKytdID0geCArIHBbaSArIDBdICogd2lkdGggKyBwW2kgKyAxXSAqIGF3OwotICAgICAgICAgICAgICAgIGNvb3Jkc1tqKytdID0geSArIHBbaSArIDJdICogaGVpZ2h0ICsgcFtpICsgM10gKiBhaDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh0ICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICB0LnRyYW5zZm9ybShjb29yZHMsIDAsIGNvb3JkcywgMCwgaiAvIDIpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIHR5cGVzW2luZGV4XTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgY3VycmVudFNlZ21lbnQoZmxvYXRbXSBjb29yZHMpIHsKLSAgICAgICAgICAgIGlmIChpc0RvbmUoKSkgewotICAgICAgICAgICAgICAgIC8vIGF3dC40Qj1JdGVyYXRvciBvdXQgb2YgYm91bmRzCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IE5vU3VjaEVsZW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNEIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChpbmRleCA9PSBwb2ludHMubGVuZ3RoKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIFNFR19DTE9TRTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGludCBqID0gMDsKLSAgICAgICAgICAgIGRvdWJsZSBwW10gPSBwb2ludHNbaW5kZXhdOwotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBwLmxlbmd0aDsgaSArPSA0KSB7Ci0gICAgICAgICAgICAgICAgY29vcmRzW2orK10gPSAoZmxvYXQpKHggKyBwW2kgKyAwXSAqIHdpZHRoICsgcFtpICsgMV0gKiBhdyk7Ci0gICAgICAgICAgICAgICAgY29vcmRzW2orK10gPSAoZmxvYXQpKHkgKyBwW2kgKyAyXSAqIGhlaWdodCArIHBbaSArIDNdICogYWgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCBqIC8gMik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gdHlwZXNbaW5kZXhdOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgUm91bmRSZWN0YW5nbGUyRC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgUm91bmRSZWN0YW5nbGUyRCgpIHsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhcmMgd2lkdGguCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJjIHdpZHRoLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBkb3VibGUgZ2V0QXJjV2lkdGgoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFyYyBoZWlnaHQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJjIGhlaWdodC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgZG91YmxlIGdldEFyY0hlaWdodCgpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGF0YSBvZiB0aGUgUm91bmRSZWN0YW5nbGUyRC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlJ3MgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSdzIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBhcmNXaWR0aAotICAgICAqICAgICAgICAgICAgdGhlIGFyYyB3aWR0aCBvZiB0aGUgcm91bmRlZCBjb3JuZXJzLgotICAgICAqIEBwYXJhbSBhcmNIZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcmMgaGVpZ2h0IG9mIHRoZSByb3VuZGVkIGNvcm5lcnMuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0Um91bmRSZWN0KGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0LAotICAgICAgICAgICAgZG91YmxlIGFyY1dpZHRoLCBkb3VibGUgYXJjSGVpZ2h0KTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgb2YgdGhlIFJvdW5kUmVjdGFuZ2xlMkQgYnkgY29weWluZyB0aGUgdmFsdWVzIGZyb20gYW4KLSAgICAgKiBleGlzdGluZyBSb3VuZFJlY3RhbmdsZTJELgotICAgICAqIAotICAgICAqIEBwYXJhbSBycgotICAgICAqICAgICAgICAgICAgdGhlIHJvdW5kIHJlY3RhbmdsZSB0byBjb3B5IHRoZSBkYXRhIGZyb20uCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHJyIGlzIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Um91bmRSZWN0KFJvdW5kUmVjdGFuZ2xlMkQgcnIpIHsKLSAgICAgICAgc2V0Um91bmRSZWN0KHJyLmdldFgoKSwgcnIuZ2V0WSgpLCByci5nZXRXaWR0aCgpLCByci5nZXRIZWlnaHQoKSwgcnIuZ2V0QXJjV2lkdGgoKSwgcnIKLSAgICAgICAgICAgICAgICAuZ2V0QXJjSGVpZ2h0KCkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEZyYW1lKGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHdpZHRoLCBkb3VibGUgaGVpZ2h0KSB7Ci0gICAgICAgIHNldFJvdW5kUmVjdCh4LCB5LCB3aWR0aCwgaGVpZ2h0LCBnZXRBcmNXaWR0aCgpLCBnZXRBcmNIZWlnaHQoKSk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gY29udGFpbnMoZG91YmxlIHB4LCBkb3VibGUgcHkpIHsKLSAgICAgICAgaWYgKGlzRW1wdHkoKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIHJ4MSA9IGdldFgoKTsKLSAgICAgICAgZG91YmxlIHJ5MSA9IGdldFkoKTsKLSAgICAgICAgZG91YmxlIHJ4MiA9IHJ4MSArIGdldFdpZHRoKCk7Ci0gICAgICAgIGRvdWJsZSByeTIgPSByeTEgKyBnZXRIZWlnaHQoKTsKLQotICAgICAgICBpZiAocHggPCByeDEgfHwgcHggPj0gcngyIHx8IHB5IDwgcnkxIHx8IHB5ID49IHJ5MikgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIGF3ID0gZ2V0QXJjV2lkdGgoKSAvIDIuMDsKLSAgICAgICAgZG91YmxlIGFoID0gZ2V0QXJjSGVpZ2h0KCkgLyAyLjA7Ci0KLSAgICAgICAgZG91YmxlIGN4LCBjeTsKLQotICAgICAgICBpZiAocHggPCByeDEgKyBhdykgewotICAgICAgICAgICAgY3ggPSByeDEgKyBhdzsKLSAgICAgICAgfSBlbHNlIGlmIChweCA+IHJ4MiAtIGF3KSB7Ci0gICAgICAgICAgICBjeCA9IHJ4MiAtIGF3OwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAocHkgPCByeTEgKyBhaCkgewotICAgICAgICAgICAgY3kgPSByeTEgKyBhaDsKLSAgICAgICAgfSBlbHNlIGlmIChweSA+IHJ5MiAtIGFoKSB7Ci0gICAgICAgICAgICBjeSA9IHJ5MiAtIGFoOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICBweCA9IChweCAtIGN4KSAvIGF3OwotICAgICAgICBweSA9IChweSAtIGN5KSAvIGFoOwotICAgICAgICByZXR1cm4gcHggKiBweCArIHB5ICogcHkgPD0gMS4wOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHMoZG91YmxlIHJ4LCBkb3VibGUgcnksIGRvdWJsZSBydywgZG91YmxlIHJoKSB7Ci0gICAgICAgIGlmIChpc0VtcHR5KCkgfHwgcncgPD0gMC4wIHx8IHJoIDw9IDAuMCkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIHgxID0gZ2V0WCgpOwotICAgICAgICBkb3VibGUgeTEgPSBnZXRZKCk7Ci0gICAgICAgIGRvdWJsZSB4MiA9IHgxICsgZ2V0V2lkdGgoKTsKLSAgICAgICAgZG91YmxlIHkyID0geTEgKyBnZXRIZWlnaHQoKTsKLQotICAgICAgICBkb3VibGUgcngxID0gcng7Ci0gICAgICAgIGRvdWJsZSByeTEgPSByeTsKLSAgICAgICAgZG91YmxlIHJ4MiA9IHJ4ICsgcnc7Ci0gICAgICAgIGRvdWJsZSByeTIgPSByeSArIHJoOwotCi0gICAgICAgIGlmIChyeDIgPCB4MSB8fCB4MiA8IHJ4MSB8fCByeTIgPCB5MSB8fCB5MiA8IHJ5MSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIGN4ID0gKHgxICsgeDIpIC8gMi4wOwotICAgICAgICBkb3VibGUgY3kgPSAoeTEgKyB5MikgLyAyLjA7Ci0KLSAgICAgICAgZG91YmxlIG54ID0gY3ggPCByeDEgPyByeDEgOiAoY3ggPiByeDIgPyByeDIgOiBjeCk7Ci0gICAgICAgIGRvdWJsZSBueSA9IGN5IDwgcnkxID8gcnkxIDogKGN5ID4gcnkyID8gcnkyIDogY3kpOwotCi0gICAgICAgIHJldHVybiBjb250YWlucyhueCwgbnkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSByeCwgZG91YmxlIHJ5LCBkb3VibGUgcncsIGRvdWJsZSByaCkgewotICAgICAgICBpZiAoaXNFbXB0eSgpIHx8IHJ3IDw9IDAuMCB8fCByaCA8PSAwLjApIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIGRvdWJsZSByeDEgPSByeDsKLSAgICAgICAgZG91YmxlIHJ5MSA9IHJ5OwotICAgICAgICBkb3VibGUgcngyID0gcnggKyBydzsKLSAgICAgICAgZG91YmxlIHJ5MiA9IHJ5ICsgcmg7Ci0KLSAgICAgICAgcmV0dXJuIGNvbnRhaW5zKHJ4MSwgcnkxKSAmJiBjb250YWlucyhyeDIsIHJ5MSkgJiYgY29udGFpbnMocngyLCByeTIpICYmIGNvbnRhaW5zKHJ4MSwgcnkyKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gYXQpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJdGVyYXRvcih0aGlzLCBhdCk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvZ2VvbS9wYWNrYWdlLmh0bWwgYi9hd3QvamF2YS9hd3QvZ2VvbS9wYWNrYWdlLmh0bWwKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGUzYTIzNmUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2dlb20vcGFja2FnZS5odG1sCisrKyAvZGV2L251bGwKQEAgLTEsOCArMCwwIEBACi08aHRtbD4KLSAgPGJvZHk+Ci0gICAgPHA+Ci0gICAgICBUaGlzIHBhY2thZ2UgY29udGFpbnMgY2xhc3NlcyBhbmQgaW50ZXJmYWNlcyByZWxhdGVkIHRvIEphdmEyRCBzaGFwZXMgYW5kIGdlb21ldHJ5LgotICAgIDwvcD4KLSAgICBAc2luY2UgQW5kcm9pZCAxLjAKLSAgPC9ib2R5PgotPC9odG1sPgpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltL0lucHV0Q29udGV4dC5qYXZhIGIvYXd0L2phdmEvYXd0L2ltL0lucHV0Q29udGV4dC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjY2U1MTQ4Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbS9JbnB1dENvbnRleHQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDgzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QuaW07Ci0KLWltcG9ydCBqYXZhLmF3dC5BV1RFdmVudDsKLS8vPz8/QVdUOiBpbXBvcnQgamF2YS5hd3QuQ29tcG9uZW50OwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmltLklucHV0TWV0aG9kQ29udGV4dDsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIElucHV0Q29udGV4dCB7Ci0gICAgcHJvdGVjdGVkIElucHV0Q29udGV4dCgpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIElucHV0Q29udGV4dCBnZXRJbnN0YW5jZSgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJbnB1dE1ldGhvZENvbnRleHQoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBkaXNwYXRjaEV2ZW50KEFXVEV2ZW50IGV2ZW50KSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZGlzcG9zZSgpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBlbmRDb21wb3NpdGlvbigpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgT2JqZWN0IGdldElucHV0TWV0aG9kQ29udHJvbE9iamVjdCgpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgcHVibGljIExvY2FsZSBnZXRMb2NhbGUoKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGlzQ29tcG9zaXRpb25FbmFibGVkKCkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgcmVjb252ZXJ0KCkgewotICAgIH0KLQotICAgIC8vPz8/QVdUCi0gICAgLyoKLSAgICBwdWJsaWMgdm9pZCByZW1vdmVOb3RpZnkoQ29tcG9uZW50IGNsaWVudCkgewotICAgIH0KLSAgICAqLwotCi0gICAgcHVibGljIGJvb2xlYW4gc2VsZWN0SW5wdXRNZXRob2QoTG9jYWxlIGxvY2FsZSkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0Q2hhcmFjdGVyU3Vic2V0cyhDaGFyYWN0ZXIuU3Vic2V0W10gc3Vic2V0cykgewotICAgIH0KLSAgICAKLSAgICBwdWJsaWMgdm9pZCBzZXRDb21wb3NpdGlvbkVuYWJsZWQoYm9vbGVhbiBlbmFibGUpIHsKLSAgICB9Ci19Ci0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbS9JbnB1dE1ldGhvZEhpZ2hsaWdodC5qYXZhIGIvYXd0L2phdmEvYXd0L2ltL0lucHV0TWV0aG9kSGlnaGxpZ2h0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDg2NWQ0N2MuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltL0lucHV0TWV0aG9kSGlnaGxpZ2h0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw5NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERtaXRyeSBBLiBEdXJuZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5pbTsKLQotaW1wb3J0IGphdmEudXRpbC5NYXA7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5UZXh0QXR0cmlidXRlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyBub3Qgc3VwcG9ydGVkIGluIEFuZHJvaWQgMS4wLiBJdCBpcyBtZXJlbHkgcHJvdmlkZWQgdG8gbWFpbnRhaW4KLSAqIGludGVyZmFjZSBjb21wYXRpYmlsaXR5IHdpdGggZGVza3RvcCBKYXZhIGltcGxlbWVudGF0aW9ucy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJbnB1dE1ldGhvZEhpZ2hsaWdodCB7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBSQVdfVEVYVCA9IDA7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDT05WRVJURURfVEVYVCA9IDE7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0TWV0aG9kSGlnaGxpZ2h0Ci0gICAgICAgIFVOU0VMRUNURURfUkFXX1RFWFRfSElHSExJR0hUID0gbmV3IElucHV0TWV0aG9kSGlnaGxpZ2h0KGZhbHNlLCBSQVdfVEVYVCk7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0TWV0aG9kSGlnaGxpZ2h0Ci0gICAgICAgIFNFTEVDVEVEX1JBV19URVhUX0hJR0hMSUdIVCA9IG5ldyBJbnB1dE1ldGhvZEhpZ2hsaWdodCh0cnVlLCBSQVdfVEVYVCk7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0TWV0aG9kSGlnaGxpZ2h0Ci0gICAgICAgIFVOU0VMRUNURURfQ09OVkVSVEVEX1RFWFRfSElHSExJR0hUID0gCi0gICAgICAgICAgICBuZXcgSW5wdXRNZXRob2RIaWdobGlnaHQoZmFsc2UsIENPTlZFUlRFRF9URVhUKTsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgSW5wdXRNZXRob2RIaWdobGlnaHQKLSAgICAgICAgU0VMRUNURURfQ09OVkVSVEVEX1RFWFRfSElHSExJR0hUID0gCi0gICAgICAgICAgICBuZXcgSW5wdXRNZXRob2RIaWdobGlnaHQodHJ1ZSwgQ09OVkVSVEVEX1RFWFQpOwotICAgIAotICAgIHByaXZhdGUgYm9vbGVhbiBzZWxlY3RlZDsKLSAgICBwcml2YXRlIGludCBzdGF0ZTsKLSAgICBwcml2YXRlIGludCB2YXJpYXRpb247Ci0gICAgcHJpdmF0ZSBNYXA8VGV4dEF0dHJpYnV0ZSw/PiBzdHlsZTsKLQotICAgIHB1YmxpYyBJbnB1dE1ldGhvZEhpZ2hsaWdodChib29sZWFuIHNlbGVjdGVkLCBpbnQgc3RhdGUsIGludCB2YXJpYXRpb24pIHsKLSAgICAgICAgdGhpcyhzZWxlY3RlZCwgc3RhdGUsIHZhcmlhdGlvbiwgbnVsbCk7Ci0gICAgfQotCi0gICAgcHVibGljIElucHV0TWV0aG9kSGlnaGxpZ2h0KGJvb2xlYW4gc2VsZWN0ZWQsIGludCBzdGF0ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IHZhcmlhdGlvbiwgTWFwPGphdmEuYXd0LmZvbnQuVGV4dEF0dHJpYnV0ZSwgPz4gc3R5bGUpIHsKLSAgICAgICAgaWYgKChzdGF0ZSAhPSBSQVdfVEVYVCkgJiYgKHN0YXRlICE9IENPTlZFUlRFRF9URVhUKSkgewotICAgICAgICAgICAgLy8gYXd0LjIwQj11bmtub3duIGlucHV0IG1ldGhvZCBoaWdobGlnaHQgc3RhdGUKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjBCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy5zZWxlY3RlZCA9IHNlbGVjdGVkOwotICAgICAgICB0aGlzLnN0YXRlID0gc3RhdGU7Ci0gICAgICAgIHRoaXMudmFyaWF0aW9uID0gdmFyaWF0aW9uOwotICAgICAgICB0aGlzLnN0eWxlID0gc3R5bGU7Ci0gICAgfQotCi0gICAgcHVibGljIElucHV0TWV0aG9kSGlnaGxpZ2h0KGJvb2xlYW4gc2VsZWN0ZWQsIGludCBzdGF0ZSkgewotICAgICAgICB0aGlzKHNlbGVjdGVkLCBzdGF0ZSwgMCwgbnVsbCk7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRTdGF0ZSgpIHsKLSAgICAgICAgcmV0dXJuIHN0YXRlOwotICAgIH0KLQotICAgIHB1YmxpYyBNYXA8amF2YS5hd3QuZm9udC5UZXh0QXR0cmlidXRlLCA/PiBnZXRTdHlsZSgpIHsKLSAgICAgICAgcmV0dXJuIHN0eWxlOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0VmFyaWF0aW9uKCkgewotICAgICAgICByZXR1cm4gdmFyaWF0aW9uOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGlzU2VsZWN0ZWQoKSB7Ci0gICAgICAgIHJldHVybiBzZWxlY3RlZDsKLSAgICB9Ci19Ci0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbS9JbnB1dE1ldGhvZFJlcXVlc3RzLmphdmEgYi9hd3QvamF2YS9hd3QvaW0vSW5wdXRNZXRob2RSZXF1ZXN0cy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiMTJkMzk3Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbS9JbnB1dE1ldGhvZFJlcXVlc3RzLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmltOwotCi1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LmZvbnQuVGV4dEhpdEluZm87Ci1pbXBvcnQgamF2YS50ZXh0LkF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvcjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJbnB1dE1ldGhvZFJlcXVlc3RzIHsKLQotICAgIHB1YmxpYyBBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgY2FuY2VsTGF0ZXN0Q29tbWl0dGVkVGV4dChBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IuQXR0cmlidXRlW10gYXR0cmlidXRlcyk7Ci0KLSAgICBwdWJsaWMgQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIGdldENvbW1pdHRlZFRleHQoaW50IGJlZ2luSW5kZXgsIGludCBlbmRJbmRleCwgQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yLkF0dHJpYnV0ZVtdIGF0dHJpYnV0ZXMpOwotCi0gICAgcHVibGljIGludCBnZXRDb21taXR0ZWRUZXh0TGVuZ3RoKCk7Ci0KLSAgICBwdWJsaWMgaW50IGdldEluc2VydFBvc2l0aW9uT2Zmc2V0KCk7Ci0KLSAgICBwdWJsaWMgVGV4dEhpdEluZm8gZ2V0TG9jYXRpb25PZmZzZXQoaW50IHgsIGludCB5KTsKLQotICAgIHB1YmxpYyBBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgZ2V0U2VsZWN0ZWRUZXh0KEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvci5BdHRyaWJ1dGVbXSBhdHRyaWJ1dGVzKTsKLQotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0VGV4dExvY2F0aW9uKFRleHRIaXRJbmZvIG9mZnNldCk7Ci19Ci0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbS9JbnB1dFN1YnNldC5qYXZhIGIvYXd0L2phdmEvYXd0L2ltL0lucHV0U3Vic2V0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDcwODg4MWUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltL0lucHV0U3Vic2V0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERtaXRyeSBBLiBEdXJuZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5pbTsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIElucHV0U3Vic2V0IGV4dGVuZHMgQ2hhcmFjdGVyLlN1YnNldCB7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0U3Vic2V0IExBVElOID0gbmV3IElucHV0U3Vic2V0KCJMQVRJTiIpOyAvLyROT04tTkxTLTEkCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0U3Vic2V0IAotICAgICAgICBMQVRJTl9ESUdJVFMgPSBuZXcgSW5wdXRTdWJzZXQoIkxBVElOX0RJR0lUUyIpOyAvLyROT04tTkxTLTEkCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0U3Vic2V0IAotICAgICAgICBUUkFESVRJT05BTF9IQU5aSSA9IG5ldyBJbnB1dFN1YnNldCgiVFJBRElUSU9OQUxfSEFOWkkiKTsgLy8kTk9OLU5MUy0xJAotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBJbnB1dFN1YnNldCAKLSAgICAgICAgU0lNUExJRklFRF9IQU5aSSA9IG5ldyBJbnB1dFN1YnNldCgiU0lNUExJRklFRF9IQU5aSSIpOyAvLyROT04tTkxTLTEkCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0U3Vic2V0IEtBTkpJID0gbmV3IElucHV0U3Vic2V0KCJLQU5KSSIpOyAvLyROT04tTkxTLTEkCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0U3Vic2V0IEhBTkpBID0gbmV3IElucHV0U3Vic2V0KCJIQU5KQSIpOyAvLyROT04tTkxTLTEkCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0U3Vic2V0IAotICAgICAgICBIQUxGV0lEVEhfS0FUQUtBTkEgPSBuZXcgSW5wdXRTdWJzZXQoIkhBTEZXSURUSF9LQVRBS0FOQSIpOyAvLyROT04tTkxTLTEkCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0U3Vic2V0IAotICAgICAgICBGVUxMV0lEVEhfTEFUSU4gPSBuZXcgSW5wdXRTdWJzZXQoIkZVTExXSURUSF9MQVRJTiIpOyAvLyROT04tTkxTLTEkCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIElucHV0U3Vic2V0IAotICAgICAgICBGVUxMV0lEVEhfRElHSVRTID0gbmV3IElucHV0U3Vic2V0KCJGVUxMV0lEVEhfRElHSVRTIik7IC8vJE5PTi1OTFMtMSQKLQotICAgIHByaXZhdGUgSW5wdXRTdWJzZXQoU3RyaW5nIG5hbWUpIHsKLSAgICAgICAgc3VwZXIobmFtZSk7Ci0gICAgfQotfQotCmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW0vc3BpL0lucHV0TWV0aG9kLmphdmEgYi9hd3QvamF2YS9hd3QvaW0vc3BpL0lucHV0TWV0aG9kLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDY3YTg4MzQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltL3NwaS9JbnB1dE1ldGhvZC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNjcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5pbS5zcGk7Ci0KLWltcG9ydCBqYXZhLmF3dC5BV1RFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJbnB1dE1ldGhvZCB7Ci0KLSAgICBwdWJsaWMgdm9pZCBhY3RpdmF0ZSgpOwotCi0gICAgcHVibGljIHZvaWQgZGVhY3RpdmF0ZShib29sZWFuIGlzVGVtcG9yYXJ5KTsKLQotICAgIHB1YmxpYyB2b2lkIGRpc3BhdGNoRXZlbnQoQVdURXZlbnQgZXZlbnQpOwotCi0gICAgcHVibGljIHZvaWQgZGlzcG9zZSgpOwotCi0gICAgcHVibGljIHZvaWQgZW5kQ29tcG9zaXRpb24oKTsKLQotICAgIHB1YmxpYyBPYmplY3QgZ2V0Q29udHJvbE9iamVjdCgpOwotCi0gICAgcHVibGljIExvY2FsZSBnZXRMb2NhbGUoKTsKLQotICAgIHB1YmxpYyB2b2lkIGhpZGVXaW5kb3dzKCk7Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NvbXBvc2l0aW9uRW5hYmxlZCgpOwotCi0gICAgcHVibGljIHZvaWQgbm90aWZ5Q2xpZW50V2luZG93Q2hhbmdlKFJlY3RhbmdsZSBib3VuZHMpOwotCi0gICAgcHVibGljIHZvaWQgcmVjb252ZXJ0KCk7Ci0KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVOb3RpZnkoKTsKLQotICAgIHB1YmxpYyB2b2lkIHNldENoYXJhY3RlclN1YnNldHMoQ2hhcmFjdGVyLlN1YnNldFtdIHN1YnNldHMpOwotCi0gICAgcHVibGljIHZvaWQgc2V0Q29tcG9zaXRpb25FbmFibGVkKGJvb2xlYW4gZW5hYmxlKTsKLQotICAgIHB1YmxpYyB2b2lkIHNldElucHV0TWV0aG9kQ29udGV4dChJbnB1dE1ldGhvZENvbnRleHQgY29udGV4dCk7Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBzZXRMb2NhbGUoTG9jYWxlIGxvY2FsZSk7Ci19Ci0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbS9zcGkvSW5wdXRNZXRob2RDb250ZXh0LmphdmEgYi9hd3QvamF2YS9hd3QvaW0vc3BpL0lucHV0TWV0aG9kQ29udGV4dC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiZmM3NzNjLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbS9zcGkvSW5wdXRNZXRob2RDb250ZXh0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LmltLnNwaTsKLQotLy8/Pz9BV1Q6IGltcG9ydCBqYXZhLmF3dC5XaW5kb3c7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5UZXh0SGl0SW5mbzsKLWltcG9ydCBqYXZhLmF3dC5pbS5JbnB1dE1ldGhvZFJlcXVlc3RzOwotaW1wb3J0IGphdmEudGV4dC5BdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3I7Ci0vLz8/P0FXVDogaW1wb3J0IGphdmF4LnN3aW5nLkpGcmFtZTsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJbnB1dE1ldGhvZENvbnRleHQgZXh0ZW5kcyBJbnB1dE1ldGhvZFJlcXVlc3RzIHsKLQotLy8gICAgPz8/QVdUOiBwdWJsaWMgSkZyYW1lIGNyZWF0ZUlucHV0TWV0aG9kSkZyYW1lKFN0cmluZyB0aXRsZSwgYm9vbGVhbiBhdHRhY2hUb0lucHV0Q29udGV4dCk7Ci0KLS8vICAgID8/P0FXVDogcHVibGljIFdpbmRvdyBjcmVhdGVJbnB1dE1ldGhvZFdpbmRvdyhTdHJpbmcgdGl0bGUsIGJvb2xlYW4gYXR0YWNoVG9JbnB1dENvbnRleHQpOwotCi0gICAgcHVibGljIHZvaWQgZGlzcGF0Y2hJbnB1dE1ldGhvZEV2ZW50KGludCBpZCwgQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIHRleHQsIGludCBjb21taXR0ZWRDaGFyYWN0ZXJDb3VudCwgVGV4dEhpdEluZm8gY2FyZXQsIFRleHRIaXRJbmZvIHZpc2libGVQb3NpdGlvbik7Ci0KLSAgICBwdWJsaWMgdm9pZCBlbmFibGVDbGllbnRXaW5kb3dOb3RpZmljYXRpb24oSW5wdXRNZXRob2QgaW5wdXRNZXRob2QsIGJvb2xlYW4gZW5hYmxlKTsKLQotfQotCmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW0vc3BpL0lucHV0TWV0aG9kRGVzY3JpcHRvci5qYXZhIGIvYXd0L2phdmEvYXd0L2ltL3NwaS9JbnB1dE1ldGhvZERlc2NyaXB0b3IuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzgwMGJjMS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW0vc3BpL0lucHV0TWV0aG9kRGVzY3JpcHRvci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5pbS5zcGk7Ci0KLWltcG9ydCBqYXZhLmF3dC5BV1RFeGNlcHRpb247Ci1pbXBvcnQgamF2YS5hd3QuSW1hZ2U7Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIG5vdCBzdXBwb3J0ZWQgaW4gQW5kcm9pZCAxLjAuIEl0IGlzIG1lcmVseSBwcm92aWRlZCB0byBtYWludGFpbgotICogaW50ZXJmYWNlIGNvbXBhdGliaWxpdHkgd2l0aCBkZXNrdG9wIEphdmEgaW1wbGVtZW50YXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJbnB1dE1ldGhvZERlc2NyaXB0b3IgewotCi0gICAgcHVibGljIExvY2FsZVtdIGdldEF2YWlsYWJsZUxvY2FsZXMoKSB0aHJvd3MgQVdURXhjZXB0aW9uOwotCi0gICAgcHVibGljIElucHV0TWV0aG9kIGNyZWF0ZUlucHV0TWV0aG9kKCkgdGhyb3dzIEV4Y2VwdGlvbjsKLQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0SW5wdXRNZXRob2REaXNwbGF5TmFtZShMb2NhbGUgaW5wdXRMb2NhbGUsIExvY2FsZSBkaXNwbGF5TGFuZ3VhZ2UpOwotCi0gICAgcHVibGljIEltYWdlIGdldElucHV0TWV0aG9kSWNvbihMb2NhbGUgaW5wdXRMb2NhbGUpOwotCi0gICAgcHVibGljIGJvb2xlYW4gaGFzRHluYW1pY0xvY2FsZUxpc3QoKTsKLQotfQotCmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvQWZmaW5lVHJhbnNmb3JtT3AuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9BZmZpbmVUcmFuc2Zvcm1PcC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkYjI1ZTFhLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9BZmZpbmVUcmFuc2Zvcm1PcC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNjE4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreSwgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgamF2YS5hd3QuZ2VvbS5BZmZpbmVUcmFuc2Zvcm07Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBvaW50MkQ7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5Ob25pbnZlcnRpYmxlVHJhbnNmb3JtRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuYXd0Lio7Ci1pbXBvcnQgamF2YS51dGlsLkFycmF5czsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBBZmZpbmVUcmFuc2Zvcm0gY2xhc3MgdHJhbnNsYXRlcyBjb29yZGluYXRlcyBmcm9tIDJEIGNvb3JkaW5hdGVzIGluIHRoZQotICogc291cmNlIGltYWdlIG9yIFJhc3RlciB0byAyRCBjb29yZGluYXRlcyBpbiB0aGUgZGVzdGluYXRpb24gaW1hZ2Ugb3IgUmFzdGVyCi0gKiB1c2luZyBhZmZpbmUgdHJhbnNmb3JtYXRpb24uIFRoZSBudW1iZXIgb2YgYmFuZHMgaW4gdGhlIHNvdXJjZSBSYXN0ZXIgc2hvdWxkCi0gKiBlcXVhbCB0byB0aGUgbnVtYmVyIG9mIGJhbmRzIGluIHRoZSBkZXN0aW5hdGlvbiBSYXN0ZXIuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgQWZmaW5lVHJhbnNmb3JtT3AgaW1wbGVtZW50cyBCdWZmZXJlZEltYWdlT3AsIFJhc3Rlck9wIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX05FQVJFU1RfTkVJR0hCT1IgaW5kaWNhdGVzIG5lYXJlc3QtbmVpZ2hib3IKLSAgICAgKiBpbnRlcnBvbGF0aW9uIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9ORUFSRVNUX05FSUdIQk9SID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0JJTElORUFSIGluZGljYXRlcyBiaWxpbmVhciBpbnRlcnBvbGF0aW9uIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9CSUxJTkVBUiA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9CSUNVQklDIGluZGljYXRlcyBiaS1jdWJpYyBpbnRlcnBvbGF0aW9uIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9CSUNVQklDID0gMzsKLQotICAgIC8qKgotICAgICAqIFRoZSBpIHR5cGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgaVR5cGU7IC8vIGludGVycG9sYXRpb24gdHlwZQotCi0gICAgLyoqCi0gICAgICogVGhlIGF0LgotICAgICAqLwotICAgIHByaXZhdGUgQWZmaW5lVHJhbnNmb3JtIGF0OwotCi0gICAgLyoqCi0gICAgICogVGhlIGhpbnRzLgotICAgICAqLwotICAgIHByaXZhdGUgUmVuZGVyaW5nSGludHMgaGludHM7Ci0KLSAgICBzdGF0aWMgewotICAgICAgICAvLyBUT0RPIC0gdW5jb21tZW50Ci0gICAgICAgIC8vIFN5c3RlbS5sb2FkTGlicmFyeSgiaW1hZ2VvcHMiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQWZmaW5lVHJhbnNmb3JtT3Agd2l0aCB0aGUgc3BlY2lmaWVkIEFmZmluZVRyYW5zZm9ybQotICAgICAqIGFuZCBSZW5kZXJpbmdIaW50cyBvYmplY3Qgd2hpY2ggZGVmaW5lcyB0aGUgaW50ZXJwb2xhdGlvbiB0eXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Zm9ybQotICAgICAqICAgICAgICAgICAgdGhlIEFmZmluZVRyYW5zZm9ybS4KLSAgICAgKiBAcGFyYW0gaGludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJpbmdIaW50cyBvYmplY3Qgd2hpY2ggZGVmaW5lcyB0aGUgaW50ZXJwb2xhdGlvbgotICAgICAqICAgICAgICAgICAgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQWZmaW5lVHJhbnNmb3JtT3AoQWZmaW5lVHJhbnNmb3JtIHhmb3JtLCBSZW5kZXJpbmdIaW50cyBoaW50cykgewotICAgICAgICB0aGlzKHhmb3JtLCBUWVBFX05FQVJFU1RfTkVJR0hCT1IpOwotICAgICAgICB0aGlzLmhpbnRzID0gaGludHM7Ci0KLSAgICAgICAgaWYgKGhpbnRzICE9IG51bGwpIHsKLSAgICAgICAgICAgIE9iamVjdCBoaW50ID0gaGludHMuZ2V0KFJlbmRlcmluZ0hpbnRzLktFWV9JTlRFUlBPTEFUSU9OKTsKLSAgICAgICAgICAgIGlmIChoaW50ICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAvLyBOZWFyZXN0IG5laWdoYm9yIGlzIGRlZmF1bHQKLSAgICAgICAgICAgICAgICBpZiAoaGludCA9PSBSZW5kZXJpbmdIaW50cy5WQUxVRV9JTlRFUlBPTEFUSU9OX0JJTElORUFSKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMuaVR5cGUgPSBUWVBFX0JJTElORUFSOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoaGludCA9PSBSZW5kZXJpbmdIaW50cy5WQUxVRV9JTlRFUlBPTEFUSU9OX0JJQ1VCSUMpIHsKLSAgICAgICAgICAgICAgICAgICAgdGhpcy5pVHlwZSA9IFRZUEVfQklDVUJJQzsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGhpbnQgPSBoaW50cy5nZXQoUmVuZGVyaW5nSGludHMuS0VZX1JFTkRFUklORyk7Ci0gICAgICAgICAgICAgICAgLy8gRGV0ZXJtaW5lIGZyb20gcmVuZGVyaW5nIHF1YWxpdHkKLSAgICAgICAgICAgICAgICBpZiAoaGludCA9PSBSZW5kZXJpbmdIaW50cy5WQUxVRV9SRU5ERVJfUVVBTElUWSkgewotICAgICAgICAgICAgICAgICAgICB0aGlzLmlUeXBlID0gVFlQRV9CSUxJTkVBUjsKLSAgICAgICAgICAgICAgICAgICAgLy8gRm9yIHNwZWVkIHVzZSBuZWFyZXN0IG5laWdoYm9yCi0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEFmZmluZVRyYW5zZm9ybU9wIHdpdGggdGhlIHNwZWNpZmllZCBBZmZpbmVUcmFuc2Zvcm0KLSAgICAgKiBhbmQgYSBzcGVjaWZpZWQgaW50ZXJwb2xhdGlvbiB0eXBlIGZyb20gdGhlIGxpc3Qgb2YgcHJlZGVmaW5lZAotICAgICAqIGludGVycG9sYXRpb24gdHlwZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHhmb3JtCi0gICAgICogICAgICAgICAgICB0aGUgQWZmaW5lVHJhbnNmb3JtLgotICAgICAqIEBwYXJhbSBpbnRlcnAKLSAgICAgKiAgICAgICAgICAgIHRoZSBvbmUgb2YgcHJlZGVmaW5lZCBpbnRlcnBvbGF0aW9uIHR5cGVzOgotICAgICAqICAgICAgICAgICAgVFlQRV9ORUFSRVNUX05FSUdIQk9SLCBUWVBFX0JJTElORUFSLCBvciBUWVBFX0JJQ1VCSUMuCi0gICAgICovCi0gICAgcHVibGljIEFmZmluZVRyYW5zZm9ybU9wKEFmZmluZVRyYW5zZm9ybSB4Zm9ybSwgaW50IGludGVycCkgewotICAgICAgICBpZiAoTWF0aC5hYnMoeGZvcm0uZ2V0RGV0ZXJtaW5hbnQoKSkgPD0gRG91YmxlLk1JTl9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI0Rj1VbmFibGUgdG8gaW52ZXJ0IHRyYW5zZm9ybSB7MH0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbWFnaW5nT3BFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjRGIiwgeGZvcm0pKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5hdCA9IChBZmZpbmVUcmFuc2Zvcm0peGZvcm0uY2xvbmUoKTsKLQotICAgICAgICBpZiAoaW50ZXJwICE9IFRZUEVfTkVBUkVTVF9ORUlHSEJPUiAmJiBpbnRlcnAgIT0gVFlQRV9CSUxJTkVBUiAmJiBpbnRlcnAgIT0gVFlQRV9CSUNVQklDKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjUwPVVua25vd24gaW50ZXJwb2xhdGlvbiB0eXBlOiB7MH0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjUwIiwgaW50ZXJwKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMuaVR5cGUgPSBpbnRlcnA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW50ZXJwb2xhdGlvbiB0eXBlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGludGVycG9sYXRpb24gdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldEludGVycG9sYXRpb25UeXBlKCkgewotICAgICAgICByZXR1cm4gaVR5cGU7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIFJlbmRlcmluZ0hpbnRzIGdldFJlbmRlcmluZ0hpbnRzKCkgewotICAgICAgICBpZiAoaGludHMgPT0gbnVsbCkgewotICAgICAgICAgICAgT2JqZWN0IHZhbHVlID0gbnVsbDsKLQotICAgICAgICAgICAgc3dpdGNoIChpVHlwZSkgewotICAgICAgICAgICAgICAgIGNhc2UgVFlQRV9ORUFSRVNUX05FSUdIQk9SOgotICAgICAgICAgICAgICAgICAgICB2YWx1ZSA9IFJlbmRlcmluZ0hpbnRzLlZBTFVFX0lOVEVSUE9MQVRJT05fTkVBUkVTVF9ORUlHSEJPUjsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBUWVBFX0JJTElORUFSOgotICAgICAgICAgICAgICAgICAgICB2YWx1ZSA9IFJlbmRlcmluZ0hpbnRzLlZBTFVFX0lOVEVSUE9MQVRJT05fQklMSU5FQVI7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgVFlQRV9CSUNVQklDOgotICAgICAgICAgICAgICAgICAgICB2YWx1ZSA9IFJlbmRlcmluZ0hpbnRzLlZBTFVFX0lOVEVSUE9MQVRJT05fQklDVUJJQzsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAgICAgdmFsdWUgPSBSZW5kZXJpbmdIaW50cy5WQUxVRV9JTlRFUlBPTEFUSU9OX05FQVJFU1RfTkVJR0hCT1I7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGhpbnRzID0gbmV3IFJlbmRlcmluZ0hpbnRzKFJlbmRlcmluZ0hpbnRzLktFWV9JTlRFUlBPTEFUSU9OLCB2YWx1ZSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gaGludHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYWZmaW5lIHRyYW5zZm9ybSBhc3NvY2lhdGVkIHdpdGggdGhpcyBBZmZpbmVUcmFuc2Zvcm1PcC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBBZmZpbmVUcmFuc2Zvcm0uCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIEFmZmluZVRyYW5zZm9ybSBnZXRUcmFuc2Zvcm0oKSB7Ci0gICAgICAgIHJldHVybiAoQWZmaW5lVHJhbnNmb3JtKWF0LmNsb25lKCk7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIFBvaW50MkQgZ2V0UG9pbnQyRChQb2ludDJEIHNyY1B0LCBQb2ludDJEIGRzdFB0KSB7Ci0gICAgICAgIHJldHVybiBhdC50cmFuc2Zvcm0oc3JjUHQsIGRzdFB0KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoQnVmZmVyZWRJbWFnZSBzcmMpIHsKLSAgICAgICAgcmV0dXJuIGdldEJvdW5kczJEKHNyYy5nZXRSYXN0ZXIoKSk7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKFJhc3RlciBzcmMpIHsKLSAgICAgICAgLy8gV2UgcG9zaXRpb24gc291cmNlIHJhc3RlciB0byAoMCwwKSBldmVuIGlmIGl0IGlzIHRyYW5zbGF0ZWQgY2hpbGQKLSAgICAgICAgLy8gcmFzdGVyLgotICAgICAgICAvLyBUaGlzIG1lYW5zIHRoYXQgd2UgbmVlZCBvbmx5IHdpZHRoIGFuZCBoZWlnaHQgb2YgdGhlIHNyYwotICAgICAgICBpbnQgd2lkdGggPSBzcmMuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IGhlaWdodCA9IHNyYy5nZXRIZWlnaHQoKTsKLQotICAgICAgICBmbG9hdFtdIGNvcm5lcnMgPSB7Ci0gICAgICAgICAgICAgICAgMCwgMCwgd2lkdGgsIDAsIHdpZHRoLCBoZWlnaHQsIDAsIGhlaWdodAotICAgICAgICB9OwotCi0gICAgICAgIGF0LnRyYW5zZm9ybShjb3JuZXJzLCAwLCBjb3JuZXJzLCAwLCA0KTsKLQotICAgICAgICBSZWN0YW5nbGUyRC5GbG9hdCBib3VuZHMgPSBuZXcgUmVjdGFuZ2xlMkQuRmxvYXQoY29ybmVyc1swXSwgY29ybmVyc1sxXSwgMCwgMCk7Ci0gICAgICAgIGJvdW5kcy5hZGQoY29ybmVyc1syXSwgY29ybmVyc1szXSk7Ci0gICAgICAgIGJvdW5kcy5hZGQoY29ybmVyc1s0XSwgY29ybmVyc1s1XSk7Ci0gICAgICAgIGJvdW5kcy5hZGQoY29ybmVyc1s2XSwgY29ybmVyc1s3XSk7Ci0KLSAgICAgICAgcmV0dXJuIGJvdW5kczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQnVmZmVyZWRJbWFnZSBjcmVhdGVDb21wYXRpYmxlRGVzdEltYWdlKEJ1ZmZlcmVkSW1hZ2Ugc3JjLCBDb2xvck1vZGVsIGRlc3RDTSkgewotICAgICAgICBSZWN0YW5nbGUyRCBuZXdCb3VuZHMgPSBnZXRCb3VuZHMyRChzcmMpOwotCi0gICAgICAgIC8vIERlc3RpbmF0aW9uIGltYWdlIHNob3VsZCBpbmNsdWRlICgwLDApICsgcG9zaXRpdmUgcGFydAotICAgICAgICAvLyBvZiB0aGUgYXJlYSBib3VuZGVkIGJ5IG5ld0JvdW5kcyAoaW4gc291cmNlIGNvb3JkaW5hdGUgc3lzdGVtKS4KLSAgICAgICAgZG91YmxlIGRzdFdpZHRoID0gbmV3Qm91bmRzLmdldFgoKSArIG5ld0JvdW5kcy5nZXRXaWR0aCgpOwotICAgICAgICBkb3VibGUgZHN0SGVpZ2h0ID0gbmV3Qm91bmRzLmdldFkoKSArIG5ld0JvdW5kcy5nZXRIZWlnaHQoKTsKLQotICAgICAgICBpZiAoZHN0V2lkdGggPD0gMCB8fCBkc3RIZWlnaHQgPD0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjI1MT1UcmFuc2Zvcm1lZCB3aWR0aCAoezB9KSBhbmQgaGVpZ2h0ICh7MX0pIHNob3VsZCBiZQotICAgICAgICAgICAgLy8gZ3JlYXRlciB0aGFuIDAKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjUxIiwgZHN0V2lkdGgsIGRzdEhlaWdodCkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoZGVzdENNICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgQnVmZmVyZWRJbWFnZShkZXN0Q00sIGRlc3RDTS5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIoKGludClkc3RXaWR0aCwKLSAgICAgICAgICAgICAgICAgICAgKGludClkc3RIZWlnaHQpLCBkZXN0Q00uaXNBbHBoYVByZW11bHRpcGxpZWQoKSwgbnVsbCk7Ci0gICAgICAgIH0KLQotICAgICAgICBDb2xvck1vZGVsIGNtID0gc3JjLmdldENvbG9yTW9kZWwoKTsKLQotICAgICAgICAvLyBJbnRlcnBvbGF0aW9uIG90aGVyIHRoYW4gTk4gZG9lc24ndCBtYWtlIGFueSBzZW5zZSBmb3IgaW5kZXggY29sb3IKLSAgICAgICAgaWYgKGlUeXBlICE9IFRZUEVfTkVBUkVTVF9ORUlHSEJPUiAmJiBjbSBpbnN0YW5jZW9mIEluZGV4Q29sb3JNb2RlbCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBCdWZmZXJlZEltYWdlKChpbnQpZHN0V2lkdGgsIChpbnQpZHN0SGVpZ2h0LCBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0IpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gT0ssIHdlIGNhbiBnZXQgc291cmNlIGNvbG9yIG1vZGVsCi0gICAgICAgIHJldHVybiBuZXcgQnVmZmVyZWRJbWFnZShjbSwgc3JjLmdldFJhc3RlcigpLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3RlcigoaW50KWRzdFdpZHRoLAotICAgICAgICAgICAgICAgIChpbnQpZHN0SGVpZ2h0KSwgY20uaXNBbHBoYVByZW11bHRpcGxpZWQoKSwgbnVsbCk7Ci0gICAgfQotCi0gICAgcHVibGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZUNvbXBhdGlibGVEZXN0UmFzdGVyKFJhc3RlciBzcmMpIHsKLSAgICAgICAgLy8gSGVyZSBhcHByb2FjaCBpcyBvdGhlciB0aGVuIGluIGNyZWF0ZUNvbXBhdGlibGVEZXN0SW1hZ2UgLQotICAgICAgICAvLyBkZXN0aW5hdGlvbiBzaG91bGQgaW5jbHVkZSBvbmx5Ci0gICAgICAgIC8vIHRyYW5zZm9ybWVkIGltYWdlLCBidXQgbm90ICgwLDApIGluIHNvdXJjZSBjb29yZGluYXRlIHN5c3RlbQotCi0gICAgICAgIFJlY3RhbmdsZTJEIG5ld0JvdW5kcyA9IGdldEJvdW5kczJEKHNyYyk7Ci0gICAgICAgIHJldHVybiBzcmMuY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKChpbnQpbmV3Qm91bmRzLmdldFgoKSwgKGludCluZXdCb3VuZHMuZ2V0WSgpLAotICAgICAgICAgICAgICAgIChpbnQpbmV3Qm91bmRzLmdldFdpZHRoKCksIChpbnQpbmV3Qm91bmRzLmdldEhlaWdodCgpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgQnVmZmVyZWRJbWFnZSBmaWx0ZXIoQnVmZmVyZWRJbWFnZSBzcmMsIEJ1ZmZlcmVkSW1hZ2UgZHN0KSB7Ci0gICAgICAgIGlmIChzcmMgPT0gZHN0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjUyPVNvdXJjZSBjYW4ndCBiZSBzYW1lIGFzIHRoZSBkZXN0aW5hdGlvbgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNTIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIENvbG9yTW9kZWwgc3JjQ00gPSBzcmMuZ2V0Q29sb3JNb2RlbCgpOwotICAgICAgICBCdWZmZXJlZEltYWdlIGZpbmFsRHN0ID0gbnVsbDsKLQotICAgICAgICBpZiAoc3JjQ00gaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwKLSAgICAgICAgICAgICAgICAmJiAoaVR5cGUgIT0gVFlQRV9ORUFSRVNUX05FSUdIQk9SIHx8IHNyY0NNLmdldFBpeGVsU2l6ZSgpICUgOCAhPSAwKSkgewotICAgICAgICAgICAgc3JjID0gKChJbmRleENvbG9yTW9kZWwpc3JjQ00pLmNvbnZlcnRUb0ludERpc2NyZXRlKHNyYy5nZXRSYXN0ZXIoKSwgdHJ1ZSk7Ci0gICAgICAgICAgICBzcmNDTSA9IHNyYy5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZHN0ID09IG51bGwpIHsKLSAgICAgICAgICAgIGRzdCA9IGNyZWF0ZUNvbXBhdGlibGVEZXN0SW1hZ2Uoc3JjLCBzcmNDTSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpZiAoIXNyY0NNLmVxdWFscyhkc3QuZ2V0Q29sb3JNb2RlbCgpKSkgewotICAgICAgICAgICAgICAgIC8vIFRyZWF0IEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfUkdCIGFuZAotICAgICAgICAgICAgICAgIC8vIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQiBhcyBzYW1lCi0gICAgICAgICAgICAgICAgaWYgKCEoKHNyYy5nZXRUeXBlKCkgPT0gQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9SR0IgfHwgc3JjLmdldFR5cGUoKSA9PSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0IpICYmIChkc3QKLSAgICAgICAgICAgICAgICAgICAgICAgIC5nZXRUeXBlKCkgPT0gQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9SR0IgfHwgZHN0LmdldFR5cGUoKSA9PSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0IpKSkgewotICAgICAgICAgICAgICAgICAgICBmaW5hbERzdCA9IGRzdDsKLSAgICAgICAgICAgICAgICAgICAgZHN0ID0gY3JlYXRlQ29tcGF0aWJsZURlc3RJbWFnZShzcmMsIHNyY0NNKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBTa2lwIGFscGhhIGNoYW5uZWwgZm9yIFRZUEVfSU5UX1JHQiBpbWFnZXMKLSAgICAgICAgaWYgKHNsb3dGaWx0ZXIoc3JjLmdldFJhc3RlcigpLCBkc3QuZ2V0UmFzdGVyKCkpICE9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMUY9VW5hYmxlIHRvIHRyYW5zZm9ybSBzb3VyY2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbWFnaW5nT3BFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjFGIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAvLyBUT0RPIC0gdW5jb21tZW50Ci0gICAgICAgICAgICAvLyBpZiAoaXBwRmlsdGVyKHNyYy5nZXRSYXN0ZXIoKSwgZHN0LmdldFJhc3RlcigpLCBzcmMuZ2V0VHlwZSgpKSAhPQotICAgICAgICAgICAgLy8gMCkKLSAgICAgICAgICAgIC8vIHRocm93IG5ldyBJbWFnaW5nT3BFeGNlcHRpb24gKCJVbmFibGUgdG8gdHJhbnNmb3JtIHNvdXJjZSIpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGZpbmFsRHN0ICE9IG51bGwpIHsKLSAgICAgICAgICAgIEdyYXBoaWNzMkQgZyA9IGZpbmFsRHN0LmNyZWF0ZUdyYXBoaWNzKCk7Ci0gICAgICAgICAgICBnLnNldENvbXBvc2l0ZShBbHBoYUNvbXBvc2l0ZS5TcmMpOwotICAgICAgICAgICAgZy5kcmF3SW1hZ2UoZHN0LCAwLCAwLCBudWxsKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZpbmFsRHN0ID0gZHN0OwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGZpbmFsRHN0OwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBXcml0YWJsZVJhc3RlciBmaWx0ZXIoUmFzdGVyIHNyYywgV3JpdGFibGVSYXN0ZXIgZHN0KSB7Ci0gICAgICAgIGlmIChzcmMgPT0gZHN0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjUyPVNvdXJjZSBjYW4ndCBiZSBzYW1lIGFzIHRoZSBkZXN0aW5hdGlvbgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNTIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0ID0gY3JlYXRlQ29tcGF0aWJsZURlc3RSYXN0ZXIoc3JjKTsKLSAgICAgICAgfSBlbHNlIGlmIChzcmMuZ2V0TnVtQmFuZHMoKSAhPSBkc3QuZ2V0TnVtQmFuZHMoKSkgewotICAgICAgICAgICAgLy8gYXd0LjI1Mz1EaWZmZXJlbnQgbnVtYmVyIG9mIGJhbmRzIGluIHNvdXJjZSBhbmQgZGVzdGluYXRpb24KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjUzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoc2xvd0ZpbHRlcihzcmMsIGRzdCkgIT0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjIxRj1VbmFibGUgdG8gdHJhbnNmb3JtIHNvdXJjZQotICAgICAgICAgICAgdGhyb3cgbmV3IEltYWdpbmdPcEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMUYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIC8vIFRPRE8gLSB1bmNvbW1lbnQKLSAgICAgICAgICAgIC8vIGlmIChpcHBGaWx0ZXIoc3JjLCBkc3QsIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9DVVNUT00pICE9IDApCi0gICAgICAgICAgICAvLyB0aHJvdyBuZXcgSW1hZ2luZ09wRXhjZXB0aW9uKCJVbmFibGUgdG8gdHJhbnNmb3JtIHNvdXJjZSIpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGRzdDsKLSAgICB9Ci0KLSAgICAvLyBUT0RPIHJlbW92ZSB3aGVuIG1ldGhvZCBpcyB1c2VkCi0gICAgLyoqCi0gICAgICogSXBwIGZpbHRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgc3JjLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3QuCi0gICAgICogQHBhcmFtIGltYWdlVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHR5cGUuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bnVzZWQiKQotICAgIHByaXZhdGUgaW50IGlwcEZpbHRlcihSYXN0ZXIgc3JjLCBXcml0YWJsZVJhc3RlciBkc3QsIGludCBpbWFnZVR5cGUpIHsKLSAgICAgICAgaW50IHNyY1N0cmlkZSwgZHN0U3RyaWRlOwotICAgICAgICBib29sZWFuIHNraXBDaGFubmVsID0gZmFsc2U7Ci0gICAgICAgIGludCBjaGFubmVsczsKLSAgICAgICAgaW50IG9mZnNldHNbXSA9IG51bGw7Ci0KLSAgICAgICAgc3dpdGNoIChpbWFnZVR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9SR0I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQkdSOiB7Ci0gICAgICAgICAgICAgICAgY2hhbm5lbHMgPSA0OwotICAgICAgICAgICAgICAgIHNyY1N0cmlkZSA9IHNyYy5nZXRXaWR0aCgpICogNDsKLSAgICAgICAgICAgICAgICBkc3RTdHJpZGUgPSBkc3QuZ2V0V2lkdGgoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgc2tpcENoYW5uZWwgPSB0cnVlOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQjoKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9BUkdCX1BSRToKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFXzRCWVRFX0FCR1I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV80QllURV9BQkdSX1BSRTogewotICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gNDsKLSAgICAgICAgICAgICAgICBzcmNTdHJpZGUgPSBzcmMuZ2V0V2lkdGgoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCkgKiA0OwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0dSQVk6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0lOREVYRUQ6IHsKLSAgICAgICAgICAgICAgICBjaGFubmVscyA9IDE7Ci0gICAgICAgICAgICAgICAgc3JjU3RyaWRlID0gc3JjLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFXzNCWVRFX0JHUjogewotICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gMzsKLSAgICAgICAgICAgICAgICBzcmNTdHJpZGUgPSBzcmMuZ2V0V2lkdGgoKSAqIDM7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCkgKiAzOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9VU0hPUlRfR1JBWTogLy8gVE9ETyAtIGNvdWxkIGJlIGRvbmUgaW4KLSAgICAgICAgICAgICAgICAvLyBuYXRpdmUgY29kZT8KLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX1VTSE9SVF81NjVfUkdCOgotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfVVNIT1JUXzU1NV9SR0I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0JJTkFSWTogewotICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZGVmYXVsdDogewotICAgICAgICAgICAgICAgIFNhbXBsZU1vZGVsIHNyY1NNID0gc3JjLmdldFNhbXBsZU1vZGVsKCk7Ci0gICAgICAgICAgICAgICAgU2FtcGxlTW9kZWwgZHN0U00gPSBkc3QuZ2V0U2FtcGxlTW9kZWwoKTsKLQotICAgICAgICAgICAgICAgIGlmIChzcmNTTSBpbnN0YW5jZW9mIFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbAotICAgICAgICAgICAgICAgICAgICAgICAgJiYgZHN0U00gaW5zdGFuY2VvZiBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gQ2hlY2sgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsCi0gICAgICAgICAgICAgICAgICAgIGlmIChzcmNTTS5nZXREYXRhVHlwZSgpICE9IERhdGFCdWZmZXIuVFlQRV9CWVRFCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgZHN0U00uZ2V0RGF0YVR5cGUoKSAhPSBEYXRhQnVmZmVyLlRZUEVfQllURSkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QpOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHMgPSBzcmNTTS5nZXROdW1CYW5kcygpOyAvLyBIYXZlIElQUCBmdW5jdGlvbnMgZm9yIDEsCi0gICAgICAgICAgICAgICAgICAgIC8vIDMgYW5kIDQgY2hhbm5lbHMKLSAgICAgICAgICAgICAgICAgICAgaWYgKGNoYW5uZWxzICE9IDEgJiYgY2hhbm5lbHMgIT0gMyAmJiBjaGFubmVscyAhPSA0KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBpbnQgZGF0YVR5cGVTaXplID0gRGF0YUJ1ZmZlci5nZXREYXRhVHlwZVNpemUoc3JjU00uZ2V0RGF0YVR5cGUoKSkgLyA4OwotCi0gICAgICAgICAgICAgICAgICAgIHNyY1N0cmlkZSA9ICgoQ29tcG9uZW50U2FtcGxlTW9kZWwpc3JjU00pLmdldFNjYW5saW5lU3RyaWRlKCkgKiBkYXRhVHlwZVNpemU7Ci0gICAgICAgICAgICAgICAgICAgIGRzdFN0cmlkZSA9ICgoQ29tcG9uZW50U2FtcGxlTW9kZWwpZHN0U00pLmdldFNjYW5saW5lU3RyaWRlKCkgKiBkYXRhVHlwZVNpemU7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChzcmNTTSBpbnN0YW5jZW9mIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIGRzdFNNIGluc3RhbmNlb2YgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkgewotICAgICAgICAgICAgICAgICAgICAvLyBDaGVjayBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsCi0gICAgICAgICAgICAgICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc3Bwc20xID0gKFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpc3JjU007Ci0gICAgICAgICAgICAgICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc3Bwc20yID0gKFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpZHN0U007Ci0KLSAgICAgICAgICAgICAgICAgICAgLy8gTm8gSVBQIGZ1bmN0aW9uIGZvciB0aGlzIHR5cGUKLSAgICAgICAgICAgICAgICAgICAgaWYgKHNwcHNtMS5nZXREYXRhVHlwZSgpID09IERhdGFCdWZmZXIuVFlQRV9VU0hPUlQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gc3Bwc20xLmdldE51bUJhbmRzKCk7Ci0gICAgICAgICAgICAgICAgICAgIC8vIEhhdmUgSVBQIGZ1bmN0aW9ucyBmb3IgMSwgMyBhbmQgNCBjaGFubmVscwotICAgICAgICAgICAgICAgICAgICBpZiAoY2hhbm5lbHMgIT0gMSAmJiBjaGFubmVscyAhPSAzICYmIGNoYW5uZWxzICE9IDQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIC8vIENoZWNrIGNvbXBhdGliaWxpdHkgb2Ygc2FtcGxlIG1vZGVscwotICAgICAgICAgICAgICAgICAgICBpZiAoc3Bwc20xLmdldERhdGFUeXBlKCkgIT0gc3Bwc20yLmdldERhdGFUeXBlKCkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCAhQXJyYXlzLmVxdWFscyhzcHBzbTEuZ2V0Qml0T2Zmc2V0cygpLCBzcHBzbTIuZ2V0Qml0T2Zmc2V0cygpKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8ICFBcnJheXMuZXF1YWxzKHNwcHNtMS5nZXRCaXRNYXNrcygpLCBzcHBzbTIuZ2V0Qml0TWFza3MoKSkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHNwcHNtMS5nZXRTYW1wbGVTaXplKGkpICE9IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBpZiAoY2hhbm5lbHMgPT0gMykgewotICAgICAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHMgPSA0OwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgaW50IGRhdGFUeXBlU2l6ZSA9IERhdGFCdWZmZXIuZ2V0RGF0YVR5cGVTaXplKHNwcHNtMS5nZXREYXRhVHlwZSgpKSAvIDg7Ci0KLSAgICAgICAgICAgICAgICAgICAgc3JjU3RyaWRlID0gc3Bwc20xLmdldFNjYW5saW5lU3RyaWRlKCkgKiBkYXRhVHlwZVNpemU7Ci0gICAgICAgICAgICAgICAgICAgIGRzdFN0cmlkZSA9IHNwcHNtMi5nZXRTY2FubGluZVN0cmlkZSgpICogZGF0YVR5cGVTaXplOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAvLyBGaWxsIG9mZnNldHMgaWYgdGhlcmUncyBhIGNoaWxkIHJhc3RlcgotICAgICAgICAgICAgICAgIGlmIChzcmMuZ2V0UGFyZW50KCkgIT0gbnVsbCB8fCBkc3QuZ2V0UGFyZW50KCkgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBpZiAoc3JjLmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWCgpICE9IDAgfHwgc3JjLmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWSgpICE9IDAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCBkc3QuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVYKCkgIT0gMAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8IGRzdC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXRzID0gbmV3IGludFs0XTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG9mZnNldHNbMF0gPSAtc3JjLmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWCgpICsgc3JjLmdldE1pblgoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG9mZnNldHNbMV0gPSAtc3JjLmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWSgpICsgc3JjLmdldE1pblkoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG9mZnNldHNbMl0gPSAtZHN0LmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWCgpICsgZHN0LmdldE1pblgoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG9mZnNldHNbM10gPSAtZHN0LmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWSgpICsgZHN0LmdldE1pblkoKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGRvdWJsZSBtMDAgPSBhdC5nZXRTY2FsZVgoKTsKLSAgICAgICAgZG91YmxlIG0wMSA9IGF0LmdldFNoZWFyWCgpOwotICAgICAgICBkb3VibGUgbTAyID0gYXQuZ2V0VHJhbnNsYXRlWCgpOwotICAgICAgICBkb3VibGUgbTEwID0gYXQuZ2V0U2hlYXJZKCk7Ci0gICAgICAgIGRvdWJsZSBtMTEgPSBhdC5nZXRTY2FsZVkoKTsKLSAgICAgICAgZG91YmxlIG0xMiA9IGF0LmdldFRyYW5zbGF0ZVkoKTsKLQotICAgICAgICBPYmplY3Qgc3JjRGF0YSwgZHN0RGF0YTsKLSAgICAgICAgQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yIGRiQWNjZXNzID0gQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yLmdldEluc3RhbmNlKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBzcmNEYXRhID0gZGJBY2Nlc3MuZ2V0RGF0YShzcmMuZ2V0RGF0YUJ1ZmZlcigpKTsKLSAgICAgICAgICAgIGRzdERhdGEgPSBkYkFjY2Vzcy5nZXREYXRhKGRzdC5nZXREYXRhQnVmZmVyKCkpOwotICAgICAgICB9IGNhdGNoIChJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgcmV0dXJuIC0xOyAvLyBVbmtub3duIGRhdGEgYnVmZmVyIHR5cGUKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBpcHBBZmZpbmVUcmFuc2Zvcm0obTAwLCBtMDEsIG0wMiwgbTEwLCBtMTEsIG0xMiwgc3JjRGF0YSwgc3JjLmdldFdpZHRoKCksIHNyYwotICAgICAgICAgICAgICAgIC5nZXRIZWlnaHQoKSwgc3JjU3RyaWRlLCBkc3REYXRhLCBkc3QuZ2V0V2lkdGgoKSwgZHN0LmdldEhlaWdodCgpLCBkc3RTdHJpZGUsCi0gICAgICAgICAgICAgICAgaVR5cGUsIGNoYW5uZWxzLCBza2lwQ2hhbm5lbCwgb2Zmc2V0cyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2xvdyBmaWx0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIHNyYy4KLSAgICAgKiBAcGFyYW0gZHN0Ci0gICAgICogICAgICAgICAgICB0aGUgZHN0LgotICAgICAqIEByZXR1cm4gdGhlIGludC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBzbG93RmlsdGVyKFJhc3RlciBzcmMsIFdyaXRhYmxlUmFzdGVyIGRzdCkgewotICAgICAgICAvLyBUT0RPOiBtYWtlIGNvcnJlY3QgaW50ZXJwb2xhdGlvbgotICAgICAgICAvLyBUT0RPOiB3aGF0IGlmIHRoZXJlIGFyZSBkaWZmZXJlbnQgZGF0YSB0eXBlcz8KLQotICAgICAgICBSZWN0YW5nbGUgc3JjQm91bmRzID0gc3JjLmdldEJvdW5kcygpOwotICAgICAgICBSZWN0YW5nbGUgZHN0Qm91bmRzID0gZHN0LmdldEJvdW5kcygpOwotICAgICAgICBSZWN0YW5nbGUgbm9ybURzdEJvdW5kcyA9IG5ldyBSZWN0YW5nbGUoMCwgMCwgZHN0Qm91bmRzLndpZHRoLCBkc3RCb3VuZHMuaGVpZ2h0KTsKLSAgICAgICAgUmVjdGFuZ2xlIGJvdW5kcyA9IGdldEJvdW5kczJEKHNyYykuZ2V0Qm91bmRzKCkuaW50ZXJzZWN0aW9uKG5vcm1Ec3RCb3VuZHMpOwotCi0gICAgICAgIEFmZmluZVRyYW5zZm9ybSBpbnYgPSBudWxsOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaW52ID0gYXQuY3JlYXRlSW52ZXJzZSgpOwotICAgICAgICB9IGNhdGNoIChOb25pbnZlcnRpYmxlVHJhbnNmb3JtRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHJldHVybiAtMTsKLSAgICAgICAgfQotCi0gICAgICAgIGRvdWJsZVtdIG0gPSBuZXcgZG91YmxlWzZdOwotICAgICAgICBpbnYuZ2V0TWF0cml4KG0pOwotCi0gICAgICAgIGludCBtaW5TcmNYID0gc3JjQm91bmRzLng7Ci0gICAgICAgIGludCBtaW5TcmNZID0gc3JjQm91bmRzLnk7Ci0gICAgICAgIGludCBtYXhTcmNYID0gc3JjQm91bmRzLnggKyBzcmNCb3VuZHMud2lkdGg7Ci0gICAgICAgIGludCBtYXhTcmNZID0gc3JjQm91bmRzLnkgKyBzcmNCb3VuZHMuaGVpZ2h0OwotCi0gICAgICAgIGludCBtaW5YID0gYm91bmRzLnggKyBkc3RCb3VuZHMueDsKLSAgICAgICAgaW50IG1pblkgPSBib3VuZHMueSArIGRzdEJvdW5kcy55OwotICAgICAgICBpbnQgbWF4WCA9IG1pblggKyBib3VuZHMud2lkdGg7Ci0gICAgICAgIGludCBtYXhZID0gbWluWSArIGJvdW5kcy5oZWlnaHQ7Ci0KLSAgICAgICAgaW50IGh4ID0gKGludCkobVswXSAqIDI1Nik7Ci0gICAgICAgIGludCBoeSA9IChpbnQpKG1bMV0gKiAyNTYpOwotICAgICAgICBpbnQgdnggPSAoaW50KShtWzJdICogMjU2KTsKLSAgICAgICAgaW50IHZ5ID0gKGludCkobVszXSAqIDI1Nik7Ci0gICAgICAgIGludCBzeCA9IChpbnQpKG1bNF0gKiAyNTYpICsgaHggKiBib3VuZHMueCArIHZ4ICogYm91bmRzLnkgKyAoc3JjQm91bmRzLngpICogMjU2OwotICAgICAgICBpbnQgc3kgPSAoaW50KShtWzVdICogMjU2KSArIGh5ICogYm91bmRzLnggKyB2eSAqIGJvdW5kcy55ICsgKHNyY0JvdW5kcy55KSAqIDI1NjsKLQotICAgICAgICB2eCAtPSBoeCAqIGJvdW5kcy53aWR0aDsKLSAgICAgICAgdnkgLT0gaHkgKiBib3VuZHMud2lkdGg7Ci0KLSAgICAgICAgaWYgKHNyYy5nZXRUcmFuc2ZlclR5cGUoKSA9PSBkc3QuZ2V0VHJhbnNmZXJUeXBlKCkpIHsKLSAgICAgICAgICAgIGZvciAoaW50IHkgPSBtaW5ZOyB5IDwgbWF4WTsgeSsrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgeCA9IG1pblg7IHggPCBtYXhYOyB4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IHB4ID0gc3ggPj4gODsKLSAgICAgICAgICAgICAgICAgICAgaW50IHB5ID0gc3kgPj4gODsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHB4ID49IG1pblNyY1ggJiYgcHkgPj0gbWluU3JjWSAmJiBweCA8IG1heFNyY1ggJiYgcHkgPCBtYXhTcmNZKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBPYmplY3QgdmFsID0gc3JjLmdldERhdGFFbGVtZW50cyhweCwgcHksIG51bGwpOwotICAgICAgICAgICAgICAgICAgICAgICAgZHN0LnNldERhdGFFbGVtZW50cyh4LCB5LCB2YWwpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHN4ICs9IGh4OwotICAgICAgICAgICAgICAgICAgICBzeSArPSBoeTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgc3ggKz0gdng7Ci0gICAgICAgICAgICAgICAgc3kgKz0gdnk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmbG9hdCBwaXhlbFtdID0gbnVsbDsKLSAgICAgICAgICAgIGZvciAoaW50IHkgPSBtaW5ZOyB5IDwgbWF4WTsgeSsrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgeCA9IG1pblg7IHggPCBtYXhYOyB4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IHB4ID0gc3ggPj4gODsKLSAgICAgICAgICAgICAgICAgICAgaW50IHB5ID0gc3kgPj4gODsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHB4ID49IG1pblNyY1ggJiYgcHkgPj0gbWluU3JjWSAmJiBweCA8IG1heFNyY1ggJiYgcHkgPCBtYXhTcmNZKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwaXhlbCA9IHNyYy5nZXRQaXhlbChweCwgcHksIHBpeGVsKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5zZXRQaXhlbCh4LCB5LCBwaXhlbCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgc3ggKz0gaHg7Ci0gICAgICAgICAgICAgICAgICAgIHN5ICs9IGh5OwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBzeCArPSB2eDsKLSAgICAgICAgICAgICAgICBzeSArPSB2eTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIElwcCBhZmZpbmUgdHJhbnNmb3JtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBtMDAKLSAgICAgKiAgICAgICAgICAgIHRoZSBtMDAuCi0gICAgICogQHBhcmFtIG0wMQotICAgICAqICAgICAgICAgICAgdGhlIG0wMS4KLSAgICAgKiBAcGFyYW0gbTAyCi0gICAgICogICAgICAgICAgICB0aGUgbTAyLgotICAgICAqIEBwYXJhbSBtMTAKLSAgICAgKiAgICAgICAgICAgIHRoZSBtMTAuCi0gICAgICogQHBhcmFtIG0xMQotICAgICAqICAgICAgICAgICAgdGhlIG0xMS4KLSAgICAgKiBAcGFyYW0gbTEyCi0gICAgICogICAgICAgICAgICB0aGUgbTEyLgotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMuCi0gICAgICogQHBhcmFtIHNyY1dpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgc3JjIHdpZHRoLgotICAgICAqIEBwYXJhbSBzcmNIZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMgaGVpZ2h0LgotICAgICAqIEBwYXJhbSBzcmNTdHJpZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMgc3RyaWRlLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3QuCi0gICAgICogQHBhcmFtIGRzdFdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgZHN0IHdpZHRoLgotICAgICAqIEBwYXJhbSBkc3RIZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3QgaGVpZ2h0LgotICAgICAqIEBwYXJhbSBkc3RTdHJpZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3Qgc3RyaWRlLgotICAgICAqIEBwYXJhbSBpVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGkgdHlwZS4KLSAgICAgKiBAcGFyYW0gY2hhbm5lbHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBjaGFubmVscy4KLSAgICAgKiBAcGFyYW0gc2tpcENoYW5uZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBza2lwIGNoYW5uZWwuCi0gICAgICogQHBhcmFtIG9mZnNldHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXRzLgotICAgICAqIEByZXR1cm4gdGhlIGludC4KLSAgICAgKi8KLSAgICBwcml2YXRlIG5hdGl2ZSBpbnQgaXBwQWZmaW5lVHJhbnNmb3JtKGRvdWJsZSBtMDAsIGRvdWJsZSBtMDEsIGRvdWJsZSBtMDIsIGRvdWJsZSBtMTAsCi0gICAgICAgICAgICBkb3VibGUgbTExLCBkb3VibGUgbTEyLCBPYmplY3Qgc3JjLCBpbnQgc3JjV2lkdGgsIGludCBzcmNIZWlnaHQsIGludCBzcmNTdHJpZGUsCi0gICAgICAgICAgICBPYmplY3QgZHN0LCBpbnQgZHN0V2lkdGgsIGludCBkc3RIZWlnaHQsIGludCBkc3RTdHJpZGUsIGludCBpVHlwZSwgaW50IGNoYW5uZWxzLAotICAgICAgICAgICAgYm9vbGVhbiBza2lwQ2hhbm5lbCwgaW50IG9mZnNldHNbXSk7Ci19ClwgTm8gbmV3bGluZSBhdCBlbmQgb2YgZmlsZQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0FyZWFBdmVyYWdpbmdTY2FsZUZpbHRlci5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL0FyZWFBdmVyYWdpbmdTY2FsZUZpbHRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3ZmIxMzhlLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9BcmVhQXZlcmFnaW5nU2NhbGVGaWx0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI4OCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEudXRpbC5BcnJheXM7Ci0KLS8qKgotICogVGhlIEFyZWFBdmVyYWdpbmdTY2FsZUZpbHRlciBjbGFzcyBzY2FsZXMgdGhlIHNvdXJjZSBpbWFnZSB1c2luZyBhcmVhCi0gKiBhdmVyYWdpbmcgYWxnb3JpdGhtLiBUaGlzIGFsZ29yaXRobSBwcm92aWRlcyBhIHNvdXJjZSBpbWFnZSB3aXRoIGEgbmV3IGltYWdlCi0gKiBjb250YWluaW5nIHRoZSByZXNhbXBsZWQgaW1hZ2UuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgQXJlYUF2ZXJhZ2luZ1NjYWxlRmlsdGVyIGV4dGVuZHMgUmVwbGljYXRlU2NhbGVGaWx0ZXIgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IHJnYkNNLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIENvbG9yTW9kZWwgcmdiQ00gPSBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBhdmVyYWdpbmdGbGFncy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgYXZlcmFnaW5nRmxhZ3MgPSAoSW1hZ2VDb25zdW1lci5UT1BET1dOTEVGVFJJR0hUIHwgSW1hZ2VDb25zdW1lci5DT01QTEVURVNDQU5MSU5FUyk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcmVzZXQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIHJlc2V0ID0gdHJ1ZTsgLy8gRmxhZyBmb3IgdXNlZCBzdXBlcmNsYXNzIGZpbHRlcgotCi0gICAgLyoqCi0gICAgICogVGhlIGluaXRlZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gaW5pdGVkID0gZmFsc2U7IC8vIEFsbCBkYXRhIGluaXRlZAotCi0gICAgLyoqCi0gICAgICogVGhlIHN1bV9yLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IHN1bV9yW107IC8vIEFycmF5IGZvciBhdmVyYWdlIFJlZCBzYW1wbGVzCi0KLSAgICAvKioKLSAgICAgKiBUaGUgc3VtX2cuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgc3VtX2dbXTsgLy8gQXJyYXkgZm9yIGF2ZXJhZ2UgR3JlZW4gc2FtcGxlcwotCi0gICAgLyoqCi0gICAgICogVGhlIHN1bV9iLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IHN1bV9iW107IC8vIEFycmF5IGZvciBhdmVyYWdlIEJsdWUgc2FtcGxlcwotCi0gICAgLyoqCi0gICAgICogVGhlIHN1bV9hLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IHN1bV9hW107IC8vIEFycmF5IGZvciBhdmVyYWdlIEFscGhhIHNhbXBsZXMKLQotICAgIC8qKgotICAgICAqIFRoZSBidWZmLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IGJ1ZmZbXTsgLy8gU3RyaWRlIGJ1ZmZlcgotCi0gICAgLyoqCi0gICAgICogVGhlIGF2ZyBmYWN0b3IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgYXZnRmFjdG9yOyAvLyBHbG9iYWwgYXZlcmFnaW5nIGZhY3RvcgotCi0gICAgLyoqCi0gICAgICogVGhlIGNhY2hlZCBkeS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBjYWNoZWREWTsgLy8gQ2FjaGVkIG51bWJlciBvZiB0aGUgZGVzdGluYXRpb24gc2NhbmxpbmUKLQotICAgIC8qKgotICAgICAqIFRoZSBjYWNoZWQgZHYgcmVzdC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBjYWNoZWREVlJlc3Q7IC8vIENhY2hlZCB2YWx1ZSBvZiByZXN0IHNyYyBzY2FubGluZXMgZm9yIHN1bQotCi0gICAgLy8gcGl4ZWwgc2FtcGxlcwotICAgIC8vIEJlY2F1c2UgZGF0YSBpZiB0cmFuc2ZlcnJpbmcgYnkgd2hvbGUgc2NhbmxpbmVzCi0gICAgLy8gd2UgYXJlIGNhY2hpbmcgb25seSBZIGNvb3JkaW5hdGUgdmFsdWVzCi0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQXJlYUF2ZXJhZ2luZ1NjYWxlRmlsdGVyIG9iamVjdCB3aGljaCBzY2FsZXMgYSBzb3VyY2UKLSAgICAgKiBpbWFnZSB3aXRoIHRoZSBzcGVjaWZpZWQgd2lkdGggYW5kIGhlaWdodC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FsZWQgd2lkdGggb2YgdGhlIGltYWdlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FsZWQgaGVpZ2h0IG9mIHRoZSBpbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQXJlYUF2ZXJhZ2luZ1NjYWxlRmlsdGVyKGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICBzdXBlcih3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIENvbG9yTW9kZWwgbW9kZWwsIGludFtdIHBpeGVscywgaW50IG9mZiwKLSAgICAgICAgICAgIGludCBzY2Fuc2l6ZSkgewotICAgICAgICBpZiAocmVzZXQpIHsKLSAgICAgICAgICAgIHN1cGVyLnNldFBpeGVscyh4LCB5LCB3LCBoLCBtb2RlbCwgcGl4ZWxzLCBvZmYsIHNjYW5zaXplKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHNldEZpbHRlcmVkUGl4ZWxzKHgsIHksIHcsIGgsIG1vZGVsLCBwaXhlbHMsIG9mZiwgc2NhbnNpemUpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBDb2xvck1vZGVsIG1vZGVsLCBieXRlW10gcGl4ZWxzLCBpbnQgb2ZmLAotICAgICAgICAgICAgaW50IHNjYW5zaXplKSB7Ci0gICAgICAgIGlmIChyZXNldCkgewotICAgICAgICAgICAgc3VwZXIuc2V0UGl4ZWxzKHgsIHksIHcsIGgsIG1vZGVsLCBwaXhlbHMsIG9mZiwgc2NhbnNpemUpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgc2V0RmlsdGVyZWRQaXhlbHMoeCwgeSwgdywgaCwgbW9kZWwsIHBpeGVscywgb2ZmLCBzY2Fuc2l6ZSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRIaW50cyhpbnQgaGludHMpIHsKLSAgICAgICAgc3VwZXIuc2V0SGludHMoaGludHMpOwotICAgICAgICByZXNldCA9ICgoaGludHMgJiBhdmVyYWdpbmdGbGFncykgIT0gYXZlcmFnaW5nRmxhZ3MpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoaXMgbWV0aG9kIGltcGxlbWVudHMgdGhlIEFyZWEgQXZlcmFnaW5nIFNjYWxlIGZpbHRlci4gVGhlIGRlc2NyaXB0aW9uCi0gICAgICogb2YgYWxnb3JpdGhtIGlzIHByZXNlbnRlZCBpbiBKYXZhIEFQSSBTcGVjaWZpY2F0aW9uLiBBcnJheXMgc3VtX3IsIHN1bV9nLAotICAgICAqIHN1bV9iLCBzdW1fYSBoYXZlIGxlbmd0aCBlcXVhbHMgd2lkdGggb2YgZGVzdGluYXRpb24gaW1hZ2UuIEluIGVhY2gKLSAgICAgKiBhcnJheSdzIGVsZW1lbnQgaXMgYWNjdW11bGF0aW5nIHBpeGVsJ3MgY29tcG9uZW50IHZhbHVlcywgcHJvcG9ydGlvbmFsIHRvCi0gICAgICogdGhlIGFyZWEgd2hpY2ggc291cmNlIHBpeGVscyB3aWxsIG9jY3VweSBpbiBkZXN0aW5hdGlvbiBpbWFnZS4gVGhlbiB0aGF0Ci0gICAgICogdmFsdWVzIHdpbGwgZGl2aWRlIGJ5IEdsb2JhbCBhdmVyYWdpbmcgZmFjdG9yIChhcmVhIG9mIHRoZSBkZXN0aW5hdGlvbgotICAgICAqIGltYWdlKSBmb3IgcmVjZWl2aW5nIGF2ZXJhZ2UgdmFsdWVzIG9mIGRlc3RpbmF0aW9uIHBpeGVscy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBwaXhlbHMgWCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgc291cmNlIHBpeGVscyBZIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgYXJlYSBvZiB0aGUgc291cmNlIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgYXJlYSBvZiB0aGUgc291cmNlIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gbW9kZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb2xvciBtb2RlbCBvZiB0aGUgc291cmNlIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gcGl4ZWxzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2Ygc291cmNlIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGludG8gdGhlIHNvdXJjZSBwaXhlbHMgYXJyYXkuCi0gICAgICogQHBhcmFtIHNjYW5zaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIG9mIHNjYW5saW5lIGluIHRoZSBwaXhlbHMgYXJyYXkuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHNldEZpbHRlcmVkUGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBDb2xvck1vZGVsIG1vZGVsLCBPYmplY3QgcGl4ZWxzLAotICAgICAgICAgICAgaW50IG9mZiwgaW50IHNjYW5zaXplKSB7Ci0gICAgICAgIGlmICghaW5pdGVkKSB7Ci0gICAgICAgICAgICBpbml0aWFsaXplKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgc3JjWCwgc3JjWSwgZHgsIGR5OwotICAgICAgICBpbnQgc3ZSZXN0LCBkdlJlc3QsIHNoUmVzdCwgZGhSZXN0LCB2RGlmLCBoRGlmOwotCi0gICAgICAgIGlmICh5ID09IDApIHsKLSAgICAgICAgICAgIGR5ID0gMDsKLSAgICAgICAgICAgIGR2UmVzdCA9IHNyY0hlaWdodDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGR5ID0gY2FjaGVkRFk7Ci0gICAgICAgICAgICBkdlJlc3QgPSBjYWNoZWREVlJlc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICBzcmNZID0geTsKLSAgICAgICAgc3ZSZXN0ID0gZGVzdEhlaWdodDsKLQotICAgICAgICBpbnQgc3JjT2ZmID0gb2ZmOwotICAgICAgICB3aGlsZSAoc3JjWSA8IHkgKyBoKSB7Ci0gICAgICAgICAgICBpZiAoc3ZSZXN0IDwgZHZSZXN0KSB7Ci0gICAgICAgICAgICAgICAgdkRpZiA9IHN2UmVzdDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgdkRpZiA9IGR2UmVzdDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgc3JjWCA9IDA7Ci0gICAgICAgICAgICBkeCA9IDA7Ci0gICAgICAgICAgICBzaFJlc3QgPSBkZXN0V2lkdGg7Ci0gICAgICAgICAgICBkaFJlc3QgPSBzcmNXaWR0aDsKLSAgICAgICAgICAgIHdoaWxlIChzcmNYIDwgdykgewotICAgICAgICAgICAgICAgIGlmIChzaFJlc3QgPCBkaFJlc3QpIHsKLSAgICAgICAgICAgICAgICAgICAgaERpZiA9IHNoUmVzdDsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBoRGlmID0gZGhSZXN0OwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpbnQgYXZnID0gaERpZiAqIHZEaWY7IC8vIGNhbGN1bGF0aW9uIG9mIGNvbnRyaWJ1dGlvbiBmYWN0b3IKLQotICAgICAgICAgICAgICAgIGludCByZ2IsIHBpeDsKLSAgICAgICAgICAgICAgICBpZiAocGl4ZWxzIGluc3RhbmNlb2YgaW50W10pIHsKLSAgICAgICAgICAgICAgICAgICAgcGl4ID0gKChpbnRbXSlwaXhlbHMpW3NyY09mZiArIHNyY1hdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHBpeCA9ICgoYnl0ZVtdKXBpeGVscylbc3JjT2ZmICsgc3JjWF0gJiAweGZmOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIHJnYiA9IG1vZGVsLmdldFJHQihwaXgpOwotICAgICAgICAgICAgICAgIGludCBhID0gcmdiID4+PiAyNDsKLSAgICAgICAgICAgICAgICBpbnQgciA9IChyZ2IgPj4gMTYpICYgMHhmZjsKLSAgICAgICAgICAgICAgICBpbnQgZyA9IChyZ2IgPj4gOCkgJiAweGZmOwotICAgICAgICAgICAgICAgIGludCBiID0gcmdiICYgMHhmZjsKLQotICAgICAgICAgICAgICAgIC8vIGFjY3VtdWxhdGluZyBwaXhlbCdzIGNvbXBvbmVudCB2YWx1ZXMKLSAgICAgICAgICAgICAgICBzdW1fYVtkeF0gKz0gYSAqIGF2ZzsKLSAgICAgICAgICAgICAgICBzdW1fcltkeF0gKz0gciAqIGF2ZzsKLSAgICAgICAgICAgICAgICBzdW1fZ1tkeF0gKz0gZyAqIGF2ZzsKLSAgICAgICAgICAgICAgICBzdW1fYltkeF0gKz0gYiAqIGF2ZzsKLQotICAgICAgICAgICAgICAgIHNoUmVzdCAtPSBoRGlmOwotICAgICAgICAgICAgICAgIGRoUmVzdCAtPSBoRGlmOwotCi0gICAgICAgICAgICAgICAgaWYgKHNoUmVzdCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNyY1grKzsKLSAgICAgICAgICAgICAgICAgICAgc2hSZXN0ID0gZGVzdFdpZHRoOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChkaFJlc3QgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICBkeCsrOwotICAgICAgICAgICAgICAgICAgICBkaFJlc3QgPSBzcmNXaWR0aDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHN2UmVzdCAtPSB2RGlmOwotICAgICAgICAgICAgZHZSZXN0IC09IHZEaWY7Ci0KLSAgICAgICAgICAgIGlmIChzdlJlc3QgPT0gMCkgewotICAgICAgICAgICAgICAgIHN2UmVzdCA9IGRlc3RIZWlnaHQ7Ci0gICAgICAgICAgICAgICAgc3JjWSsrOwotICAgICAgICAgICAgICAgIHNyY09mZiArPSBzY2Fuc2l6ZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKGR2UmVzdCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXZlcmFnaW5nIGRlc3RpbmF0aW9uIHBpeGVsJ3MgdmFsdWVzCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBkZXN0V2lkdGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBpbnQgYSA9IChzdW1fYVtpXSAvIGF2Z0ZhY3RvcikgJiAweGZmOwotICAgICAgICAgICAgICAgICAgICBpbnQgciA9IChzdW1fcltpXSAvIGF2Z0ZhY3RvcikgJiAweGZmOwotICAgICAgICAgICAgICAgICAgICBpbnQgZyA9IChzdW1fZ1tpXSAvIGF2Z0ZhY3RvcikgJiAweGZmOwotICAgICAgICAgICAgICAgICAgICBpbnQgYiA9IChzdW1fYltpXSAvIGF2Z0ZhY3RvcikgJiAweGZmOwotICAgICAgICAgICAgICAgICAgICBpbnQgZnJnYiA9IChhIDw8IDI0KSB8IChyIDw8IDE2KSB8IChnIDw8IDgpIHwgYjsKLSAgICAgICAgICAgICAgICAgICAgYnVmZltpXSA9IGZyZ2I7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGNvbnN1bWVyLnNldFBpeGVscygwLCBkeSwgZGVzdFdpZHRoLCAxLCByZ2JDTSwgYnVmZiwgMCwgZGVzdFdpZHRoKTsKLSAgICAgICAgICAgICAgICBkeSsrOwotICAgICAgICAgICAgICAgIGR2UmVzdCA9IHNyY0hlaWdodDsKLSAgICAgICAgICAgICAgICBBcnJheXMuZmlsbChzdW1fYSwgMCk7Ci0gICAgICAgICAgICAgICAgQXJyYXlzLmZpbGwoc3VtX3IsIDApOwotICAgICAgICAgICAgICAgIEFycmF5cy5maWxsKHN1bV9nLCAwKTsKLSAgICAgICAgICAgICAgICBBcnJheXMuZmlsbChzdW1fYiwgMCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgfQotCi0gICAgICAgIGNhY2hlZERZID0gZHk7Ci0gICAgICAgIGNhY2hlZERWUmVzdCA9IGR2UmVzdDsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluaXRpYWxpemF0aW9uIG9mIHRoZSBhdXhpbGlhcnkgZGF0YS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgaW5pdGlhbGl6ZSgpIHsKLQotICAgICAgICBzdW1fYSA9IG5ldyBpbnRbZGVzdFdpZHRoXTsKLSAgICAgICAgc3VtX3IgPSBuZXcgaW50W2Rlc3RXaWR0aF07Ci0gICAgICAgIHN1bV9nID0gbmV3IGludFtkZXN0V2lkdGhdOwotICAgICAgICBzdW1fYiA9IG5ldyBpbnRbZGVzdFdpZHRoXTsKLQotICAgICAgICBidWZmID0gbmV3IGludFtkZXN0V2lkdGhdOwotICAgICAgICBvdXRwaXhidWYgPSBidWZmOwotICAgICAgICBhdmdGYWN0b3IgPSBzcmNXaWR0aCAqIHNyY0hlaWdodDsKLQotICAgICAgICBpbml0ZWQgPSB0cnVlOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9Bd3RJbWFnZUJhY2tkb29yQWNjZXNzb3JJbXBsLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29ySW1wbC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2ZGZmZWU4Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9Bd3RJbWFnZUJhY2tkb29yQWNjZXNzb3JJbXBsLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNTYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICogQ3JlYXRlZCBvbiAyMy4xMS4yMDA1Ci0gKgotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5JbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EYXRhQnVmZmVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJCeXRlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJEb3VibGU7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGF0YUJ1ZmZlckZsb2F0OwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJJbnQ7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGF0YUJ1ZmZlclNob3J0OwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJVU2hvcnQ7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLkF3dEltYWdlQmFja2Rvb3JBY2Nlc3NvcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLkdMVm9sYXRpbGVJbWFnZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLlN1cmZhY2U7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZS5EYXRhQnVmZmVyTGlzdGVuZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBub3QgcGFydCBvZiBwdWJsaWMgQVBJLiBJdCB1c2VmdWwgZm9yIHJlY2VpdmluZyBwYWNrYWdlIHByaXZhdGUKLSAqIGRhdGEgZnJvbSBvdGhlciBwYWNrYWdlcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLWNsYXNzIEF3dEltYWdlQmFja2Rvb3JBY2Nlc3NvckltcGwgZXh0ZW5kcyBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IgewotCi0gICAgc3RhdGljIHZvaWQgaW5pdCgpIHsKLSAgICAgICAgaW5zdCA9IG5ldyBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3JJbXBsKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN1cmZhY2UgZ2V0SW1hZ2VTdXJmYWNlKEltYWdlIGltYWdlKSB7Ci0gICAgICAgIGlmIChpbWFnZSBpbnN0YW5jZW9mIEJ1ZmZlcmVkSW1hZ2UpIHsKLSAgICAgICAgICAgIHJldHVybiAoKEJ1ZmZlcmVkSW1hZ2UpaW1hZ2UpLmdldEltYWdlU3VyZmFjZSgpOwotICAgICAgICB9IGVsc2UgaWYgKGltYWdlIGluc3RhbmNlb2YgR0xWb2xhdGlsZUltYWdlKSB7Ci0gICAgICAgICAgICByZXR1cm4gKChHTFZvbGF0aWxlSW1hZ2UpaW1hZ2UpLmdldEltYWdlU3VyZmFjZSgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGlzR3JheVBhbGxldGUoSW5kZXhDb2xvck1vZGVsIGljbSkgewotICAgICAgICByZXR1cm4gaWNtLmlzR3JheVBhbGxldGUoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgT2JqZWN0IGdldERhdGEoRGF0YUJ1ZmZlciBkYikgewotICAgICAgICBpZiAoZGIgaW5zdGFuY2VvZiBEYXRhQnVmZmVyQnl0ZSkgewotICAgICAgICAgICAgcmV0dXJuICgoRGF0YUJ1ZmZlckJ5dGUpZGIpLmdldERhdGEoKTsKLSAgICAgICAgfSBlbHNlIGlmIChkYiBpbnN0YW5jZW9mIERhdGFCdWZmZXJVU2hvcnQpIHsKLSAgICAgICAgICAgIHJldHVybiAoKERhdGFCdWZmZXJVU2hvcnQpZGIpLmdldERhdGEoKTsKLSAgICAgICAgfSBlbHNlIGlmIChkYiBpbnN0YW5jZW9mIERhdGFCdWZmZXJTaG9ydCkgewotICAgICAgICAgICAgcmV0dXJuICgoRGF0YUJ1ZmZlclNob3J0KWRiKS5nZXREYXRhKCk7Ci0gICAgICAgIH0gZWxzZSBpZiAoZGIgaW5zdGFuY2VvZiBEYXRhQnVmZmVySW50KSB7Ci0gICAgICAgICAgICByZXR1cm4gKChEYXRhQnVmZmVySW50KWRiKS5nZXREYXRhKCk7Ci0gICAgICAgIH0gZWxzZSBpZiAoZGIgaW5zdGFuY2VvZiBEYXRhQnVmZmVyRmxvYXQpIHsKLSAgICAgICAgICAgIHJldHVybiAoKERhdGFCdWZmZXJGbG9hdClkYikuZ2V0RGF0YSgpOwotICAgICAgICB9IGVsc2UgaWYgKGRiIGluc3RhbmNlb2YgRGF0YUJ1ZmZlckRvdWJsZSkgewotICAgICAgICAgICAgcmV0dXJuICgoRGF0YUJ1ZmZlckRvdWJsZSlkYikuZ2V0RGF0YSgpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLy8gYXd0LjIzNT1Xcm9uZyBEYXRhIEJ1ZmZlciB0eXBlIDogezB9Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzNSIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgZGIuZ2V0Q2xhc3MoKSkpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldERhdGFJbnQoRGF0YUJ1ZmZlciBkYikgewotICAgICAgICBpZiAoZGIgaW5zdGFuY2VvZiBEYXRhQnVmZmVySW50KSB7Ci0gICAgICAgICAgICByZXR1cm4gKChEYXRhQnVmZmVySW50KWRiKS5nZXREYXRhKCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJ5dGVbXSBnZXREYXRhQnl0ZShEYXRhQnVmZmVyIGRiKSB7Ci0gICAgICAgIGlmIChkYiBpbnN0YW5jZW9mIERhdGFCdWZmZXJCeXRlKSB7Ci0gICAgICAgICAgICByZXR1cm4gKChEYXRhQnVmZmVyQnl0ZSlkYikuZ2V0RGF0YSgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBzaG9ydFtdIGdldERhdGFTaG9ydChEYXRhQnVmZmVyIGRiKSB7Ci0gICAgICAgIGlmIChkYiBpbnN0YW5jZW9mIERhdGFCdWZmZXJTaG9ydCkgewotICAgICAgICAgICAgcmV0dXJuICgoRGF0YUJ1ZmZlclNob3J0KWRiKS5nZXREYXRhKCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHNob3J0W10gZ2V0RGF0YVVTaG9ydChEYXRhQnVmZmVyIGRiKSB7Ci0gICAgICAgIGlmIChkYiBpbnN0YW5jZW9mIERhdGFCdWZmZXJVU2hvcnQpIHsKLSAgICAgICAgICAgIHJldHVybiAoKERhdGFCdWZmZXJVU2hvcnQpZGIpLmdldERhdGEoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZG91YmxlW10gZ2V0RGF0YURvdWJsZShEYXRhQnVmZmVyIGRiKSB7Ci0gICAgICAgIGlmIChkYiBpbnN0YW5jZW9mIERhdGFCdWZmZXJEb3VibGUpIHsKLSAgICAgICAgICAgIHJldHVybiAoKERhdGFCdWZmZXJEb3VibGUpZGIpLmdldERhdGEoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXRbXSBnZXREYXRhRmxvYXQoRGF0YUJ1ZmZlciBkYikgewotICAgICAgICBpZiAoZGIgaW5zdGFuY2VvZiBEYXRhQnVmZmVyRmxvYXQpIHsKLSAgICAgICAgICAgIHJldHVybiAoKERhdGFCdWZmZXJGbG9hdClkYikuZ2V0RGF0YSgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGFkZERhdGFCdWZmZXJMaXN0ZW5lcihEYXRhQnVmZmVyIGRiLCBEYXRhQnVmZmVyTGlzdGVuZXIgbGlzdGVuZXIpIHsKLSAgICAgICAgZGIuYWRkRGF0YUJ1ZmZlckxpc3RlbmVyKGxpc3RlbmVyKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCByZW1vdmVEYXRhQnVmZmVyTGlzdGVuZXIoRGF0YUJ1ZmZlciBkYikgewotICAgICAgICBkYi5yZW1vdmVEYXRhQnVmZmVyTGlzdGVuZXIoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB2YWxpZGF0ZShEYXRhQnVmZmVyIGRiKSB7Ci0gICAgICAgIGRiLnZhbGlkYXRlKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgcmVsZWFzZURhdGEoRGF0YUJ1ZmZlciBkYikgewotICAgICAgICBkYi5yZWxlYXNlRGF0YSgpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9CYW5kQ29tYmluZU9wLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvQmFuZENvbWJpbmVPcC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkYTJjYzg5Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9CYW5kQ29tYmluZU9wLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2NTggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKgotICogQGRhdGU6IFNlcCAyMCwgMjAwNQotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC4qOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUG9pbnQyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEudXRpbC5BcnJheXM7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLkF3dEltYWdlQmFja2Rvb3JBY2Nlc3NvcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgQmFuZENvbWJpbmVPcCBjbGFzcyB0cmFuc2xhdGVzIGNvb3JkaW5hdGVzIGZyb20gY29vcmRpbmF0ZXMgaW4gdGhlIHNvdXJjZQotICogUmFzdGVyIHRvIGNvb3JkaW5hdGVzIGluIHRoZSBkZXN0aW5hdGlvbiBSYXN0ZXIgYnkgYW4gYXJiaXRyYXJ5IGxpbmVhcgotICogY29tYmluYXRpb24gb2YgdGhlIGJhbmRzIGluIGEgc291cmNlIFJhc3RlciwgdXNpbmcgYSBzcGVjaWZpZWQgbWF0cml4LiBUaGUKLSAqIG51bWJlciBvZiBiYW5kcyBpbiB0aGUgbWF0cml4IHNob3VsZCBlcXVhbCB0byB0aGUgbnVtYmVyIG9mIGJhbmRzIGluIHRoZQotICogc291cmNlIFJhc3RlciBwbHVzIDEuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgQmFuZENvbWJpbmVPcCBpbXBsZW1lbnRzIFJhc3Rlck9wIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBvZmZzZXRzM2MuCi0gICAgICovCi0gICAgc3RhdGljIGZpbmFsIGludCBvZmZzZXRzM2NbXSA9IHsKLSAgICAgICAgICAgIDE2LCA4LCAwCi0gICAgfTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBvZmZzZXRzNGFjLgotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBpbnQgb2Zmc2V0czRhY1tdID0gewotICAgICAgICAgICAgMTYsIDgsIDAsIDI0Ci0gICAgfTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBtYXNrczNjLgotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBpbnQgbWFza3MzY1tdID0gewotICAgICAgICAgICAgMHhGRjAwMDAsIDB4RkYwMCwgMHhGRgotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgbWFza3M0YWMuCi0gICAgICovCi0gICAgc3RhdGljIGZpbmFsIGludCBtYXNrczRhY1tdID0gewotICAgICAgICAgICAgMHhGRjAwMDAsIDB4RkYwMCwgMHhGRiwgMHhGRjAwMDAwMAotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgcGlPZmZzZXRzLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBwaU9mZnNldHNbXSA9IHsKLSAgICAgICAgICAgIDAsIDEsIDIKLSAgICB9OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IHBpSW52T2Zmc2V0cy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgcGlJbnZPZmZzZXRzW10gPSB7Ci0gICAgICAgICAgICAyLCAxLCAwCi0gICAgfTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0JZVEUzQy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9CWVRFM0MgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfQllURTRBQy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9CWVRFNEFDID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX1VTSE9SVDNDLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBUWVBFX1VTSE9SVDNDID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX1NIT1JUM0MuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFRZUEVfU0hPUlQzQyA9IDM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbXggd2lkdGguCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgbXhXaWR0aDsKLQotICAgIC8qKgotICAgICAqIFRoZSBteCBoZWlnaHQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgbXhIZWlnaHQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbWF0cml4LgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgbWF0cml4W11bXTsKLQotICAgIC8qKgotICAgICAqIFRoZSByIGhpbnRzLgotICAgICAqLwotICAgIHByaXZhdGUgUmVuZGVyaW5nSGludHMgckhpbnRzOwotCi0gICAgc3RhdGljIHsKLSAgICAgICAgLy8gWFhYIC0gdG9kbwotICAgICAgICAvLyBTeXN0ZW0ubG9hZExpYnJhcnkoImltYWdlb3BzIik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEJhbmRDb21iaW5lT3Agb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCBtYXRyaXguCi0gICAgICogCi0gICAgICogQHBhcmFtIG1hdHJpeAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBtYXRyaXggZm9yIGJhbmQgY29tYmluaW5nLgotICAgICAqIEBwYXJhbSBoaW50cwotICAgICAqICAgICAgICAgICAgdGhlIFJlbmRlcmluZ0hpbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBCYW5kQ29tYmluZU9wKGZsb2F0IG1hdHJpeFtdW10sIFJlbmRlcmluZ0hpbnRzIGhpbnRzKSB7Ci0gICAgICAgIHRoaXMubXhIZWlnaHQgPSBtYXRyaXgubGVuZ3RoOwotICAgICAgICB0aGlzLm14V2lkdGggPSBtYXRyaXhbMF0ubGVuZ3RoOwotICAgICAgICB0aGlzLm1hdHJpeCA9IG5ldyBmbG9hdFtteEhlaWdodF1bbXhXaWR0aF07Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBteEhlaWdodDsgaSsrKSB7Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KG1hdHJpeFtpXSwgMCwgdGhpcy5tYXRyaXhbaV0sIDAsIG14V2lkdGgpOwotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5ySGludHMgPSBoaW50czsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUmVuZGVyaW5nSGludHMgZ2V0UmVuZGVyaW5nSGludHMoKSB7Ci0gICAgICAgIHJldHVybiB0aGlzLnJIaW50czsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtYXRyaXggYXNzb2NpYXRlZCB3aXRoIHRoaXMgQmFuZENvbWJpbmVPcCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWF0cml4IGFzc29jaWF0ZWQgd2l0aCB0aGlzIEJhbmRDb21iaW5lT3Agb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBmbG9hdFtdW10gZ2V0TWF0cml4KCkgewotICAgICAgICBmbG9hdCByZXNbXVtdID0gbmV3IGZsb2F0W214SGVpZ2h0XVtteFdpZHRoXTsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG14SGVpZ2h0OyBpKyspIHsKLSAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkobWF0cml4W2ldLCAwLCByZXNbaV0sIDAsIG14V2lkdGgpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUG9pbnQyRCBnZXRQb2ludDJEKFBvaW50MkQgc3JjUG9pbnQsIFBvaW50MkQgZHN0UG9pbnQpIHsKLSAgICAgICAgaWYgKGRzdFBvaW50ID09IG51bGwpIHsKLSAgICAgICAgICAgIGRzdFBvaW50ID0gbmV3IFBvaW50MkQuRmxvYXQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGRzdFBvaW50LnNldExvY2F0aW9uKHNyY1BvaW50KTsKLSAgICAgICAgcmV0dXJuIGRzdFBvaW50OwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRChSYXN0ZXIgc3JjKSB7Ci0gICAgICAgIHJldHVybiBzcmMuZ2V0Qm91bmRzKCk7Ci0gICAgfQotCi0gICAgcHVibGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZUNvbXBhdGlibGVEZXN0UmFzdGVyKFJhc3RlciBzcmMpIHsKLSAgICAgICAgaW50IG51bUJhbmRzID0gc3JjLmdldE51bUJhbmRzKCk7Ci0gICAgICAgIGlmIChteFdpZHRoICE9IG51bUJhbmRzICYmIG14V2lkdGggIT0gKG51bUJhbmRzICsgMSkgfHwgbnVtQmFuZHMgIT0gbXhIZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNTQ9TnVtYmVyIG9mIGJhbmRzIGluIHRoZSBzb3VyY2UgcmFzdGVyICh7MH0pIGlzCi0gICAgICAgICAgICAvLyBpbmNvbXBhdGlibGUgd2l0aCB0aGUgbWF0cml4IFt7MX14ezJ9XQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNTQiLCAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgIG5ldyBPYmplY3RbXSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgbnVtQmFuZHMsIG14V2lkdGgsIG14SGVpZ2h0Ci0gICAgICAgICAgICAgICAgICAgIH0pKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBzcmMuY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKHNyYy5nZXRXaWR0aCgpLCBzcmMuZ2V0SGVpZ2h0KCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBmaWx0ZXIoUmFzdGVyIHNyYywgV3JpdGFibGVSYXN0ZXIgZHN0KSB7Ci0gICAgICAgIGludCBudW1CYW5kcyA9IHNyYy5nZXROdW1CYW5kcygpOwotCi0gICAgICAgIGlmIChteFdpZHRoICE9IG51bUJhbmRzICYmIG14V2lkdGggIT0gKG51bUJhbmRzICsgMSkpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNTQ9TnVtYmVyIG9mIGJhbmRzIGluIHRoZSBzb3VyY2UgcmFzdGVyICh7MH0pIGlzCi0gICAgICAgICAgICAvLyBpbmNvbXBhdGlibGUgd2l0aCB0aGUgbWF0cml4IFt7MX14ezJ9XQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNTQiLCAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgIG5ldyBPYmplY3RbXSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgbnVtQmFuZHMsIG14V2lkdGgsIG14SGVpZ2h0Ci0gICAgICAgICAgICAgICAgICAgIH0pKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0ID0gY3JlYXRlQ29tcGF0aWJsZURlc3RSYXN0ZXIoc3JjKTsKLSAgICAgICAgfSBlbHNlIGlmIChkc3QuZ2V0TnVtQmFuZHMoKSAhPSBteEhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjI1NT1OdW1iZXIgb2YgYmFuZHMgaW4gdGhlIGRlc3RpbmF0aW9uIHJhc3RlciAoezB9KSBpcwotICAgICAgICAgICAgLy8gaW5jb21wYXRpYmxlIHdpdGggdGhlIG1hdHJpeCBbezF9eHsyfV0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjU1IiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICBuZXcgT2JqZWN0W10gewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5nZXROdW1CYW5kcygpLCBteFdpZHRoLCBteEhlaWdodAotICAgICAgICAgICAgICAgICAgICB9KSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBYWFggLSB0b2RvCi0gICAgICAgIC8vIGlmIChpcHBGaWx0ZXIoc3JjLCBkc3QpICE9IDApCi0gICAgICAgIGlmICh2ZXJ5U2xvd0ZpbHRlcihzcmMsIGRzdCkgIT0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjIxRj1VbmFibGUgdG8gdHJhbnNmb3JtIHNvdXJjZQotICAgICAgICAgICAgdGhyb3cgbmV3IEltYWdpbmdPcEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMUYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBkc3Q7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIFNhbXBsZU1vZGVsSW5mby4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBjbGFzcyBTYW1wbGVNb2RlbEluZm8gewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgY2hhbm5lbHMuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgY2hhbm5lbHM7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjaGFubmVscyBvcmRlci4KLSAgICAgICAgICovCi0gICAgICAgIGludCBjaGFubmVsc09yZGVyW107Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBzdHJpZGUuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgc3RyaWRlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrIHNhbXBsZSBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc20KLSAgICAgKiAgICAgICAgICAgIHRoZSBzbS4KLSAgICAgKiBAcmV0dXJuIHRoZSBzYW1wbGUgbW9kZWwgaW5mby4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIFNhbXBsZU1vZGVsSW5mbyBjaGVja1NhbXBsZU1vZGVsKFNhbXBsZU1vZGVsIHNtKSB7Ci0gICAgICAgIFNhbXBsZU1vZGVsSW5mbyByZXQgPSBuZXcgU2FtcGxlTW9kZWxJbmZvKCk7Ci0KLSAgICAgICAgaWYgKHNtIGluc3RhbmNlb2YgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsKSB7Ci0gICAgICAgICAgICAvLyBDaGVjayBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwKLSAgICAgICAgICAgIGlmIChzbS5nZXREYXRhVHlwZSgpICE9IERhdGFCdWZmZXIuVFlQRV9CWVRFKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldC5jaGFubmVscyA9IHNtLmdldE51bUJhbmRzKCk7Ci0gICAgICAgICAgICByZXQuc3RyaWRlID0gKChDb21wb25lbnRTYW1wbGVNb2RlbClzbSkuZ2V0U2NhbmxpbmVTdHJpZGUoKTsKLSAgICAgICAgICAgIHJldC5jaGFubmVsc09yZGVyID0gKChDb21wb25lbnRTYW1wbGVNb2RlbClzbSkuZ2V0QmFuZE9mZnNldHMoKTsKLQotICAgICAgICB9IGVsc2UgaWYgKHNtIGluc3RhbmNlb2YgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkgewotICAgICAgICAgICAgLy8gQ2hlY2sgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbAotICAgICAgICAgICAgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBzcHBzbTEgPSAoU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbClzbTsKLQotICAgICAgICAgICAgcmV0LmNoYW5uZWxzID0gc3Bwc20xLmdldE51bUJhbmRzKCk7Ci0gICAgICAgICAgICBpZiAoc3Bwc20xLmdldERhdGFUeXBlKCkgIT0gRGF0YUJ1ZmZlci5UWVBFX0lOVCkgewotICAgICAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBDaGVjayBzYW1wbGUgbW9kZWxzCi0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHJldC5jaGFubmVsczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHNwcHNtMS5nZXRTYW1wbGVTaXplKGkpICE9IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZXQuY2hhbm5lbHNPcmRlciA9IG5ldyBpbnRbcmV0LmNoYW5uZWxzXTsKLSAgICAgICAgICAgIGludCBiaXRPZmZzZXRzW10gPSBzcHBzbTEuZ2V0Qml0T2Zmc2V0cygpOwotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCByZXQuY2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgICAgIGlmIChiaXRPZmZzZXRzW2ldICUgOCAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIHJldC5jaGFubmVsc09yZGVyW2ldID0gYml0T2Zmc2V0c1tpXSAvIDg7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldC5jaGFubmVscyA9IDQ7Ci0gICAgICAgICAgICByZXQuc3RyaWRlID0gc3Bwc20xLmdldFNjYW5saW5lU3RyaWRlKCkgKiA0OwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcmV0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNsb3cgZmlsdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMuCi0gICAgICogQHBhcmFtIGRzdAotICAgICAqICAgICAgICAgICAgdGhlIGRzdC4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBpbnQgc2xvd0ZpbHRlcihSYXN0ZXIgc3JjLCBXcml0YWJsZVJhc3RlciBkc3QpIHsKLSAgICAgICAgaW50IHJlcyA9IDA7Ci0KLSAgICAgICAgU2FtcGxlTW9kZWxJbmZvIHNyY0luZm8sIGRzdEluZm87Ci0gICAgICAgIGludCBvZmZzZXRzW10gPSBudWxsOwotCi0gICAgICAgIHNyY0luZm8gPSBjaGVja1NhbXBsZU1vZGVsKHNyYy5nZXRTYW1wbGVNb2RlbCgpKTsKLSAgICAgICAgZHN0SW5mbyA9IGNoZWNrU2FtcGxlTW9kZWwoZHN0LmdldFNhbXBsZU1vZGVsKCkpOwotICAgICAgICBpZiAoc3JjSW5mbyA9PSBudWxsIHx8IGRzdEluZm8gPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIHZlcnlTbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIEZpbGwgb2Zmc2V0cyBpZiB0aGVyZSdzIGEgY2hpbGQgcmFzdGVyCi0gICAgICAgIGlmIChzcmMuZ2V0UGFyZW50KCkgIT0gbnVsbCB8fCBkc3QuZ2V0UGFyZW50KCkgIT0gbnVsbCkgewotICAgICAgICAgICAgaWYgKHNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSAhPSAwIHx8IHNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSAhPSAwCi0gICAgICAgICAgICAgICAgICAgIHx8IGRzdC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSAhPSAwIHx8IGRzdC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgb2Zmc2V0cyA9IG5ldyBpbnRbNF07Ci0gICAgICAgICAgICAgICAgb2Zmc2V0c1swXSA9IC1zcmMuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVYKCkgKyBzcmMuZ2V0TWluWCgpOwotICAgICAgICAgICAgICAgIG9mZnNldHNbMV0gPSAtc3JjLmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWSgpICsgc3JjLmdldE1pblkoKTsKLSAgICAgICAgICAgICAgICBvZmZzZXRzWzJdID0gLWRzdC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSArIGRzdC5nZXRNaW5YKCk7Ci0gICAgICAgICAgICAgICAgb2Zmc2V0c1szXSA9IC1kc3QuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVZKCkgKyBkc3QuZ2V0TWluWSgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHJteFdpZHRoID0gKHNyY0luZm8uY2hhbm5lbHMgKyAxKTsgLy8gd2lkdGggb2YgdGhlIHJlb3JkZXJlZCBtYXRyaXgKLSAgICAgICAgZmxvYXQgcmVvcmRlcmVkTWF0cml4W10gPSBuZXcgZmxvYXRbcm14V2lkdGggKiBkc3RJbmZvLmNoYW5uZWxzXTsKLSAgICAgICAgZm9yIChpbnQgaiA9IDA7IGogPCBkc3RJbmZvLmNoYW5uZWxzOyBqKyspIHsKLSAgICAgICAgICAgIGlmIChqID49IGRzdEluZm8uY2hhbm5lbHNPcmRlci5sZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBzcmNJbmZvLmNoYW5uZWxzOyBpKyspIHsKLSAgICAgICAgICAgICAgICBpZiAoaSA+PSBzcmNJbmZvLmNoYW5uZWxzT3JkZXIubGVuZ3RoKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIHJlb3JkZXJlZE1hdHJpeFtkc3RJbmZvLmNoYW5uZWxzT3JkZXJbal0gKiBybXhXaWR0aCArIHNyY0luZm8uY2hhbm5lbHNPcmRlcltpXV0gPSBtYXRyaXhbal1baV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAobXhXaWR0aCA9PSBybXhXaWR0aCkgewotICAgICAgICAgICAgICAgIHJlb3JkZXJlZE1hdHJpeFsoZHN0SW5mby5jaGFubmVsc09yZGVyW2pdICsgMSkgKiBybXhXaWR0aCAtIDFdID0gbWF0cml4W2pdW214V2lkdGggLSAxXTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIE9iamVjdCBzcmNEYXRhLCBkc3REYXRhOwotICAgICAgICBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IgZGJBY2Nlc3MgPSBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IuZ2V0SW5zdGFuY2UoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHNyY0RhdGEgPSBkYkFjY2Vzcy5nZXREYXRhKHNyYy5nZXREYXRhQnVmZmVyKCkpOwotICAgICAgICAgICAgZHN0RGF0YSA9IGRiQWNjZXNzLmdldERhdGEoZHN0LmdldERhdGFCdWZmZXIoKSk7Ci0gICAgICAgIH0gY2F0Y2ggKElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICByZXR1cm4gLTE7IC8vIFVua25vd24gZGF0YSBidWZmZXIgdHlwZQotICAgICAgICB9Ci0KLSAgICAgICAgc2ltcGxlQ29tYmluZUJhbmRzKHNyY0RhdGEsIHNyYy5nZXRXaWR0aCgpLCBzcmMuZ2V0SGVpZ2h0KCksIHNyY0luZm8uc3RyaWRlLAotICAgICAgICAgICAgICAgIHNyY0luZm8uY2hhbm5lbHMsIGRzdERhdGEsIGRzdEluZm8uc3RyaWRlLCBkc3RJbmZvLmNoYW5uZWxzLCByZW9yZGVyZWRNYXRyaXgsCi0gICAgICAgICAgICAgICAgb2Zmc2V0cyk7Ci0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBWZXJ5IHNsb3cgZmlsdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMuCi0gICAgICogQHBhcmFtIGRzdAotICAgICAqICAgICAgICAgICAgdGhlIGRzdC4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgdmVyeVNsb3dGaWx0ZXIoUmFzdGVyIHNyYywgV3JpdGFibGVSYXN0ZXIgZHN0KSB7Ci0gICAgICAgIGludCBudW1CYW5kcyA9IHNyYy5nZXROdW1CYW5kcygpOwotCi0gICAgICAgIGludCBzcmNNaW5YID0gc3JjLmdldE1pblgoKTsKLSAgICAgICAgaW50IHNyY1kgPSBzcmMuZ2V0TWluWSgpOwotCi0gICAgICAgIGludCBkc3RNaW5YID0gZHN0LmdldE1pblgoKTsKLSAgICAgICAgaW50IGRzdFkgPSBkc3QuZ2V0TWluWSgpOwotCi0gICAgICAgIGludCBkWCA9IHNyYy5nZXRXaWR0aCgpOy8vIDwgZHN0LmdldFdpZHRoKCkgPyBzcmMuZ2V0V2lkdGgoKSA6Ci0gICAgICAgIC8vIGRzdC5nZXRXaWR0aCgpOwotICAgICAgICBpbnQgZFkgPSBzcmMuZ2V0SGVpZ2h0KCk7Ly8gPCBkc3QuZ2V0SGVpZ2h0KCkgPyBzcmMuZ2V0SGVpZ2h0KCkgOgotICAgICAgICAvLyBkc3QuZ2V0SGVpZ2h0KCk7Ci0KLSAgICAgICAgZmxvYXQgc2FtcGxlOwotICAgICAgICBpbnQgc3JjUGl4ZWxzW10gPSBuZXcgaW50W251bUJhbmRzICogZFggKiBkWV07Ci0gICAgICAgIGludCBkc3RQaXhlbHNbXSA9IG5ldyBpbnRbbXhIZWlnaHQgKiBkWCAqIGRZXTsKLQotICAgICAgICBzcmNQaXhlbHMgPSBzcmMuZ2V0UGl4ZWxzKHNyY01pblgsIHNyY1ksIGRYLCBkWSwgc3JjUGl4ZWxzKTsKLQotICAgICAgICBpZiAobnVtQmFuZHMgPT0gbXhXaWR0aCkgewotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDAsIGogPSAwOyBpIDwgc3JjUGl4ZWxzLmxlbmd0aDsgaSArPSBudW1CYW5kcykgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGRzdEIgPSAwOyBkc3RCIDwgbXhIZWlnaHQ7IGRzdEIrKykgewotICAgICAgICAgICAgICAgICAgICBzYW1wbGUgPSAwZjsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgc3JjQiA9IDA7IHNyY0IgPCBudW1CYW5kczsgc3JjQisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzYW1wbGUgKz0gbWF0cml4W2RzdEJdW3NyY0JdICogc3JjUGl4ZWxzW2kgKyBzcmNCXTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBkc3RQaXhlbHNbaisrXSA9IChpbnQpc2FtcGxlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBqID0gMDsgaSA8IHNyY1BpeGVscy5sZW5ndGg7IGkgKz0gbnVtQmFuZHMpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBkc3RCID0gMDsgZHN0QiA8IG14SGVpZ2h0OyBkc3RCKyspIHsKLSAgICAgICAgICAgICAgICAgICAgc2FtcGxlID0gMGY7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IHNyY0IgPSAwOyBzcmNCIDwgbnVtQmFuZHM7IHNyY0IrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgc2FtcGxlICs9IG1hdHJpeFtkc3RCXVtzcmNCXSAqIHNyY1BpeGVsc1tpICsgc3JjQl07Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgZHN0UGl4ZWxzW2orK10gPSAoaW50KShzYW1wbGUgKyBtYXRyaXhbZHN0Ql1bbnVtQmFuZHNdKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBkc3Quc2V0UGl4ZWxzKGRzdE1pblgsIGRzdFksIGRYLCBkWSwgZHN0UGl4ZWxzKTsKLQotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvLyBUT0RPIHJlbW92ZSB3aGVuIG1ldGhvZCBpcyB1c2VkCi0gICAgLyoqCi0gICAgICogSXBwIGZpbHRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgc3JjLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3QuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bnVzZWQiKQotICAgIHByaXZhdGUgaW50IGlwcEZpbHRlcihSYXN0ZXIgc3JjLCBXcml0YWJsZVJhc3RlciBkc3QpIHsKLSAgICAgICAgYm9vbGVhbiBpbnZlcnRDaGFubmVsczsKLSAgICAgICAgYm9vbGVhbiBpblBsYWNlID0gKHNyYyA9PSBkc3QpOwotICAgICAgICBpbnQgdHlwZTsKLSAgICAgICAgaW50IHNyY1N0cmlkZSwgZHN0U3RyaWRlOwotICAgICAgICBpbnQgb2Zmc2V0c1tdID0gbnVsbDsKLQotICAgICAgICBpbnQgc3JjQmFuZHMgPSBzcmMuZ2V0TnVtQmFuZHMoKTsKLSAgICAgICAgaW50IGRzdEJhbmRzID0gZHN0LmdldE51bUJhbmRzKCk7Ci0KLSAgICAgICAgaWYgKGRzdEJhbmRzICE9IDMKLSAgICAgICAgICAgICAgICB8fCAoc3JjQmFuZHMgIT0gMyAmJiAhKHNyY0JhbmRzID09IDQgJiYgbWF0cml4WzBdWzNdID09IDAgJiYgbWF0cml4WzFdWzNdID09IDAgJiYgbWF0cml4WzJdWzNdID09IDApKSkgewotICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QpOwotICAgICAgICB9Ci0KLSAgICAgICAgU2FtcGxlTW9kZWwgc3JjU00gPSBzcmMuZ2V0U2FtcGxlTW9kZWwoKTsKLSAgICAgICAgU2FtcGxlTW9kZWwgZHN0U00gPSBkc3QuZ2V0U2FtcGxlTW9kZWwoKTsKLQotICAgICAgICBpZiAoc3JjU00gaW5zdGFuY2VvZiBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsCi0gICAgICAgICAgICAgICAgJiYgZHN0U00gaW5zdGFuY2VvZiBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKSB7Ci0gICAgICAgICAgICAvLyBDaGVjayBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsCi0gICAgICAgICAgICBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIHNwcHNtMSA9IChTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKXNyY1NNOwotICAgICAgICAgICAgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBzcHBzbTIgPSAoU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbClkc3RTTTsKLQotICAgICAgICAgICAgaWYgKHNwcHNtMS5nZXREYXRhVHlwZSgpICE9IERhdGFCdWZmZXIuVFlQRV9JTlQKLSAgICAgICAgICAgICAgICAgICAgfHwgc3Bwc20yLmdldERhdGFUeXBlKCkgIT0gRGF0YUJ1ZmZlci5UWVBFX0lOVCkgewotICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gQ2hlY2sgc2FtcGxlIG1vZGVscwotICAgICAgICAgICAgaWYgKCFBcnJheXMuZXF1YWxzKHNwcHNtMi5nZXRCaXRPZmZzZXRzKCksIG9mZnNldHMzYykKLSAgICAgICAgICAgICAgICAgICAgfHwgIUFycmF5cy5lcXVhbHMoc3Bwc20yLmdldEJpdE1hc2tzKCksIG1hc2tzM2MpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoc3JjQmFuZHMgPT0gMykgewotICAgICAgICAgICAgICAgIGlmICghQXJyYXlzLmVxdWFscyhzcHBzbTEuZ2V0Qml0T2Zmc2V0cygpLCBvZmZzZXRzM2MpCi0gICAgICAgICAgICAgICAgICAgICAgICB8fCAhQXJyYXlzLmVxdWFscyhzcHBzbTEuZ2V0Qml0TWFza3MoKSwgbWFza3MzYykpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSBpZiAoc3JjQmFuZHMgPT0gNCkgewotICAgICAgICAgICAgICAgIGlmICghQXJyYXlzLmVxdWFscyhzcHBzbTEuZ2V0Qml0T2Zmc2V0cygpLCBvZmZzZXRzNGFjKQotICAgICAgICAgICAgICAgICAgICAgICAgfHwgIUFycmF5cy5lcXVhbHMoc3Bwc20xLmdldEJpdE1hc2tzKCksIG1hc2tzNGFjKSkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICB0eXBlID0gVFlQRV9CWVRFNEFDOwotICAgICAgICAgICAgaW52ZXJ0Q2hhbm5lbHMgPSB0cnVlOwotCi0gICAgICAgICAgICBzcmNTdHJpZGUgPSBzcHBzbTEuZ2V0U2NhbmxpbmVTdHJpZGUoKSAqIDQ7Ci0gICAgICAgICAgICBkc3RTdHJpZGUgPSBzcHBzbTIuZ2V0U2NhbmxpbmVTdHJpZGUoKSAqIDQ7Ci0gICAgICAgIH0gZWxzZSBpZiAoc3JjU00gaW5zdGFuY2VvZiBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwKLSAgICAgICAgICAgICAgICAmJiBkc3RTTSBpbnN0YW5jZW9mIFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbCkgewotICAgICAgICAgICAgaWYgKHNyY0JhbmRzICE9IDMpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGludCBzcmNEYXRhVHlwZSA9IHNyY1NNLmdldERhdGFUeXBlKCk7Ci0KLSAgICAgICAgICAgIHN3aXRjaCAoc3JjRGF0YVR5cGUpIHsKLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgICAgICB0eXBlID0gVFlQRV9CWVRFM0M7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgICAgICAgICAgdHlwZSA9IFRZUEVfVVNIT1JUM0M7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1NIT1JUOgotICAgICAgICAgICAgICAgICAgICB0eXBlID0gVFlQRV9TSE9SVDNDOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIENoZWNrIFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbAotICAgICAgICAgICAgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsIHBpc20xID0gKFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbClzcmNTTTsKLSAgICAgICAgICAgIFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbCBwaXNtMiA9IChQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwpZHN0U007Ci0KLSAgICAgICAgICAgIGlmIChzcmNEYXRhVHlwZSAhPSBwaXNtMi5nZXREYXRhVHlwZSgpIHx8IHBpc20xLmdldFBpeGVsU3RyaWRlKCkgIT0gMwotICAgICAgICAgICAgICAgICAgICB8fCBwaXNtMi5nZXRQaXhlbFN0cmlkZSgpICE9IDMKLSAgICAgICAgICAgICAgICAgICAgfHwgIUFycmF5cy5lcXVhbHMocGlzbTEuZ2V0QmFuZE9mZnNldHMoKSwgcGlzbTIuZ2V0QmFuZE9mZnNldHMoKSkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChBcnJheXMuZXF1YWxzKHBpc20xLmdldEJhbmRPZmZzZXRzKCksIHBpSW52T2Zmc2V0cykpIHsKLSAgICAgICAgICAgICAgICBpbnZlcnRDaGFubmVscyA9IHRydWU7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKEFycmF5cy5lcXVhbHMocGlzbTEuZ2V0QmFuZE9mZnNldHMoKSwgcGlPZmZzZXRzKSkgewotICAgICAgICAgICAgICAgIGludmVydENoYW5uZWxzID0gZmFsc2U7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaW50IGRhdGFUeXBlU2l6ZSA9IERhdGFCdWZmZXIuZ2V0RGF0YVR5cGVTaXplKHNyY0RhdGFUeXBlKSAvIDg7Ci0KLSAgICAgICAgICAgIHNyY1N0cmlkZSA9IHBpc20xLmdldFNjYW5saW5lU3RyaWRlKCkgKiBkYXRhVHlwZVNpemU7Ci0gICAgICAgICAgICBkc3RTdHJpZGUgPSBwaXNtMi5nZXRTY2FubGluZVN0cmlkZSgpICogZGF0YVR5cGVTaXplOwotICAgICAgICB9IGVsc2UgeyAvLyBYWFggLSB0b2RvIC0gSVBQIGFsbG93cyBzdXBwb3J0IGZvciBwbGFuYXIgZGF0YSBhbHNvCi0gICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBGaWxsIG9mZnNldHMgaWYgdGhlcmUncyBhIGNoaWxkIHJhc3RlcgotICAgICAgICBpZiAoc3JjLmdldFBhcmVudCgpICE9IG51bGwgfHwgZHN0LmdldFBhcmVudCgpICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChzcmMuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVYKCkgIT0gMCB8fCBzcmMuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVZKCkgIT0gMAotICAgICAgICAgICAgICAgICAgICB8fCBkc3QuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVYKCkgIT0gMCB8fCBkc3QuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVZKCkgIT0gMCkgewotICAgICAgICAgICAgICAgIG9mZnNldHMgPSBuZXcgaW50WzRdOwotICAgICAgICAgICAgICAgIG9mZnNldHNbMF0gPSAtc3JjLmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWCgpICsgc3JjLmdldE1pblgoKTsKLSAgICAgICAgICAgICAgICBvZmZzZXRzWzFdID0gLXNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSArIHNyYy5nZXRNaW5ZKCk7Ci0gICAgICAgICAgICAgICAgb2Zmc2V0c1syXSA9IC1kc3QuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVYKCkgKyBkc3QuZ2V0TWluWCgpOwotICAgICAgICAgICAgICAgIG9mZnNldHNbM10gPSAtZHN0LmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWSgpICsgZHN0LmdldE1pblkoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIE9iamVjdCBzcmNEYXRhLCBkc3REYXRhOwotICAgICAgICBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IgZGJBY2Nlc3MgPSBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IuZ2V0SW5zdGFuY2UoKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHNyY0RhdGEgPSBkYkFjY2Vzcy5nZXREYXRhKHNyYy5nZXREYXRhQnVmZmVyKCkpOwotICAgICAgICAgICAgZHN0RGF0YSA9IGRiQWNjZXNzLmdldERhdGEoZHN0LmdldERhdGFCdWZmZXIoKSk7Ci0gICAgICAgIH0gY2F0Y2ggKElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICByZXR1cm4gLTE7IC8vIFVua25vd24gZGF0YSBidWZmZXIgdHlwZQotICAgICAgICB9Ci0KLSAgICAgICAgZmxvYXQgaXBwTWF0cml4W10gPSBuZXcgZmxvYXRbMTJdOwotCi0gICAgICAgIGlmIChpbnZlcnRDaGFubmVscykgewotICAgICAgICAgICAgLy8gSVBQIHRyZWF0cyBiaWcgZW5kaWFuIGludGVnZXJzIGxpa2UgQkdSLCBzbyB3ZSBoYXZlIHRvCi0gICAgICAgICAgICAvLyBzd2FwIGNvbHVtbnMgMSBhbmQgMyBhbmQgcm93cyAxIGFuZCAzCi0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG14SGVpZ2h0OyBpKyspIHsKLSAgICAgICAgICAgICAgICBpcHBNYXRyaXhbaSAqIDRdID0gbWF0cml4WzIgLSBpXVsyXTsKLSAgICAgICAgICAgICAgICBpcHBNYXRyaXhbaSAqIDQgKyAxXSA9IG1hdHJpeFsyIC0gaV1bMV07Ci0gICAgICAgICAgICAgICAgaXBwTWF0cml4W2kgKiA0ICsgMl0gPSBtYXRyaXhbMiAtIGldWzBdOwotCi0gICAgICAgICAgICAgICAgaWYgKG14V2lkdGggPT0gNCkgewotICAgICAgICAgICAgICAgICAgICBpcHBNYXRyaXhbaSAqIDQgKyAzXSA9IG1hdHJpeFsyIC0gaV1bM107Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChteFdpZHRoID09IDUpIHsKLSAgICAgICAgICAgICAgICAgICAgaXBwTWF0cml4W2kgKiA0ICsgM10gPSBtYXRyaXhbMiAtIGldWzRdOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbXhIZWlnaHQ7IGkrKykgewotICAgICAgICAgICAgICAgIGlwcE1hdHJpeFtpICogNF0gPSBtYXRyaXhbaV1bMF07Ci0gICAgICAgICAgICAgICAgaXBwTWF0cml4W2kgKiA0ICsgMV0gPSBtYXRyaXhbaV1bMV07Ci0gICAgICAgICAgICAgICAgaXBwTWF0cml4W2kgKiA0ICsgMl0gPSBtYXRyaXhbaV1bMl07Ci0KLSAgICAgICAgICAgICAgICBpZiAobXhXaWR0aCA9PSA0KSB7Ci0gICAgICAgICAgICAgICAgICAgIGlwcE1hdHJpeFtpICogNCArIDNdID0gbWF0cml4W2ldWzNdOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAobXhXaWR0aCA9PSA1KSB7Ci0gICAgICAgICAgICAgICAgICAgIGlwcE1hdHJpeFtpICogNCArIDNdID0gbWF0cml4W2ldWzRdOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBpcHBDb2xvclR3aXN0KHNyY0RhdGEsIHNyYy5nZXRXaWR0aCgpLCBzcmMuZ2V0SGVpZ2h0KCksIHNyY1N0cmlkZSwgZHN0RGF0YSwgZHN0Ci0gICAgICAgICAgICAgICAgLmdldFdpZHRoKCksIGRzdC5nZXRIZWlnaHQoKSwgZHN0U3RyaWRlLCBpcHBNYXRyaXgsIHR5cGUsIG9mZnNldHMsIGluUGxhY2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIElwcCBjb2xvciB0d2lzdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjRGF0YQotICAgICAqICAgICAgICAgICAgdGhlIHNyYyBkYXRhLgotICAgICAqIEBwYXJhbSBzcmNXaWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHNyYyB3aWR0aC4KLSAgICAgKiBAcGFyYW0gc3JjSGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgc3JjIGhlaWdodC4KLSAgICAgKiBAcGFyYW0gc3JjU3RyaWRlCi0gICAgICogICAgICAgICAgICB0aGUgc3JjIHN0cmlkZS4KLSAgICAgKiBAcGFyYW0gZHN0RGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGRzdCBkYXRhLgotICAgICAqIEBwYXJhbSBkc3RXaWR0aAotICAgICAqICAgICAgICAgICAgdGhlIGRzdCB3aWR0aC4KLSAgICAgKiBAcGFyYW0gZHN0SGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgZHN0IGhlaWdodC4KLSAgICAgKiBAcGFyYW0gZHN0U3RyaWRlCi0gICAgICogICAgICAgICAgICB0aGUgZHN0IHN0cmlkZS4KLSAgICAgKiBAcGFyYW0gaXBwTWF0cml4Ci0gICAgICogICAgICAgICAgICB0aGUgaXBwIG1hdHJpeC4KLSAgICAgKiBAcGFyYW0gdHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHR5cGUuCi0gICAgICogQHBhcmFtIG9mZnNldHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXRzLgotICAgICAqIEBwYXJhbSBpblBsYWNlCi0gICAgICogICAgICAgICAgICB0aGUgaW4gcGxhY2UuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgbmF0aXZlIGludCBpcHBDb2xvclR3aXN0KE9iamVjdCBzcmNEYXRhLCBpbnQgc3JjV2lkdGgsIGludCBzcmNIZWlnaHQsCi0gICAgICAgICAgICBpbnQgc3JjU3RyaWRlLCBPYmplY3QgZHN0RGF0YSwgaW50IGRzdFdpZHRoLCBpbnQgZHN0SGVpZ2h0LCBpbnQgZHN0U3RyaWRlLAotICAgICAgICAgICAgZmxvYXQgaXBwTWF0cml4W10sIGludCB0eXBlLCBpbnQgb2Zmc2V0c1tdLCBib29sZWFuIGluUGxhY2UpOwotCi0gICAgLyoqCi0gICAgICogU2ltcGxlIGNvbWJpbmUgYmFuZHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyY0RhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMgZGF0YS4KLSAgICAgKiBAcGFyYW0gc3JjV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMgd2lkdGguCi0gICAgICogQHBhcmFtIHNyY0hlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIHNyYyBoZWlnaHQuCi0gICAgICogQHBhcmFtIHNyY1N0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIHNyYyBzdHJpZGUuCi0gICAgICogQHBhcmFtIHNyY0NoYW5uZWxzCi0gICAgICogICAgICAgICAgICB0aGUgc3JjIGNoYW5uZWxzLgotICAgICAqIEBwYXJhbSBkc3REYXRhCi0gICAgICogICAgICAgICAgICB0aGUgZHN0IGRhdGEuCi0gICAgICogQHBhcmFtIGRzdFN0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIGRzdCBzdHJpZGUuCi0gICAgICogQHBhcmFtIGRzdENoYW5uZWxzCi0gICAgICogICAgICAgICAgICB0aGUgZHN0IGNoYW5uZWxzLgotICAgICAqIEBwYXJhbSBtCi0gICAgICogICAgICAgICAgICB0aGUgbS4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldHMuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgbmF0aXZlIGludCBzaW1wbGVDb21iaW5lQmFuZHMoT2JqZWN0IHNyY0RhdGEsIGludCBzcmNXaWR0aCwgaW50IHNyY0hlaWdodCwKLSAgICAgICAgICAgIGludCBzcmNTdHJpZGUsIGludCBzcmNDaGFubmVscywgT2JqZWN0IGRzdERhdGEsIGludCBkc3RTdHJpZGUsIGludCBkc3RDaGFubmVscywKLSAgICAgICAgICAgIGZsb2F0IG1bXSwgaW50IG9mZnNldHNbXSk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvQmFuZGVkU2FtcGxlTW9kZWwuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9CYW5kZWRTYW1wbGVNb2RlbC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwYWEzMGQ3Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9CYW5kZWRTYW1wbGVNb2RlbC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIEJhbmRlZFNhbXBsZU1vZGVsIGNsYXNzIHByb3ZpZGVzIHNhbXBsZXMgb2YgcGl4ZWxzIGluIGFuIGltYWdlIHdoaWNoIGlzCi0gKiBzdG9yZWQgaW4gYSBiYW5kIGludGVybGVhdmVkIG1ldGhvZC4gRWFjaCBwaXhlbCdzIHNhbXBsZSB0YWtlcyBvbmUgZGF0YQotICogZWxlbWVudCBvZiB0aGUgRGF0YUJ1ZmZlci4gVGhlIHBpeGVsIHN0cmlkZSBmb3IgYSBCYW5kZWRTYW1wbGVNb2RlbCBpcyBvbmUuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgZmluYWwgY2xhc3MgQmFuZGVkU2FtcGxlTW9kZWwgZXh0ZW5kcyBDb21wb25lbnRTYW1wbGVNb2RlbCB7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBpbmRpY2VzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBudW1CYW5kcwotICAgICAqICAgICAgICAgICAgdGhlIG51bSBiYW5kcy4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbnRbXS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBpbnRbXSBjcmVhdGVJbmRpY2VzKGludCBudW1CYW5kcykgewotICAgICAgICBpbnQgaW5kaWNlc1tdID0gbmV3IGludFtudW1CYW5kc107Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgaW5kaWNlc1tpXSA9IGk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGluZGljZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgb2Zmc2V0cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbnVtQmFuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW0gYmFuZHMuCi0gICAgICogQHJldHVybiB0aGUgaW50W10uCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgaW50W10gY3JlYXRlT2Zmc2V0cyhpbnQgbnVtQmFuZHMpIHsKLSAgICAgICAgaW50IG9mZnNldHNbXSA9IG5ldyBpbnRbbnVtQmFuZHNdOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgIG9mZnNldHNbaV0gPSAwOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBvZmZzZXRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBCYW5kZWRTYW1wbGVNb2RlbCBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIGRhdGEgdHlwZQotICAgICAqIG9mIHNhbXBsZXMsIHRoZSB3aWR0aCwgaGVpZ2h0IGFuZCBiYW5kcyBudW1iZXIgb2YgaW1hZ2UgZGF0YS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUgb2Ygc2FtcGxlcy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gbnVtQmFuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgYmFuZHMuCi0gICAgICovCi0gICAgcHVibGljIEJhbmRlZFNhbXBsZU1vZGVsKGludCBkYXRhVHlwZSwgaW50IHcsIGludCBoLCBpbnQgbnVtQmFuZHMpIHsKLSAgICAgICAgdGhpcyhkYXRhVHlwZSwgdywgaCwgdywgQmFuZGVkU2FtcGxlTW9kZWwuY3JlYXRlSW5kaWNlcyhudW1CYW5kcyksIEJhbmRlZFNhbXBsZU1vZGVsCi0gICAgICAgICAgICAgICAgLmNyZWF0ZU9mZnNldHMobnVtQmFuZHMpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQmFuZGVkU2FtcGxlTW9kZWwgb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCBkYXRhIHR5cGUKLSAgICAgKiBvZiBzYW1wbGVzLCB0aGUgd2lkdGgsIGhlaWdodCBhbmQgYmFuZHMgbnVtYmVyIG9mIGltYWdlIGRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlIG9mIHNhbXBsZXMuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIHNjYW5saW5lU3RyaWRlCi0gICAgICogICAgICAgICAgICB0aGUgc2NhbmxpbmUgc3RyaWRlIG9mIHRoZSBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gYmFua0luZGljZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiB0aGUgYmFuayBpbmRpY2VzLgotICAgICAqIEBwYXJhbSBiYW5kT2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHRoZSBiYW5kIG9mZnNldHMuCi0gICAgICovCi0gICAgcHVibGljIEJhbmRlZFNhbXBsZU1vZGVsKGludCBkYXRhVHlwZSwgaW50IHcsIGludCBoLCBpbnQgc2NhbmxpbmVTdHJpZGUsIGludCBiYW5rSW5kaWNlc1tdLAotICAgICAgICAgICAgaW50IGJhbmRPZmZzZXRzW10pIHsKLSAgICAgICAgc3VwZXIoZGF0YVR5cGUsIHcsIGgsIDEsIHNjYW5saW5lU3RyaWRlLCBiYW5rSW5kaWNlcywgYmFuZE9mZnNldHMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTYW1wbGVNb2RlbCBjcmVhdGVDb21wYXRpYmxlU2FtcGxlTW9kZWwoaW50IHcsIGludCBoKSB7Ci0gICAgICAgIHJldHVybiBuZXcgQmFuZGVkU2FtcGxlTW9kZWwoZGF0YVR5cGUsIHcsIGgsIHcsIGJhbmtJbmRpY2VzLCBiYW5kT2Zmc2V0cyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIERhdGFCdWZmZXIgY3JlYXRlRGF0YUJ1ZmZlcigpIHsKLSAgICAgICAgRGF0YUJ1ZmZlciBkYXRhID0gbnVsbDsKLSAgICAgICAgaW50IHNpemUgPSBzY2FubGluZVN0cmlkZSAqIGhlaWdodDsKLQotICAgICAgICBzd2l0Y2ggKGRhdGFUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlckJ5dGUoc2l6ZSwgbnVtQmFua3MpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfU0hPUlQ6Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgZGF0YSA9IG5ldyBEYXRhQnVmZmVyU2hvcnQoc2l6ZSwgbnVtQmFua3MpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlckludChzaXplLCBudW1CYW5rcyk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9GTE9BVDoKLSAgICAgICAgICAgICAgICBkYXRhID0gbmV3IERhdGFCdWZmZXJGbG9hdChzaXplLCBudW1CYW5rcyk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9ET1VCTEU6Ci0gICAgICAgICAgICAgICAgZGF0YSA9IG5ldyBEYXRhQnVmZmVyRG91YmxlKHNpemUsIG51bUJhbmtzKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBkYXRhOwotCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFNhbXBsZU1vZGVsIGNyZWF0ZVN1YnNldFNhbXBsZU1vZGVsKGludFtdIGJhbmRzKSB7Ci0gICAgICAgIGlmIChiYW5kcy5sZW5ndGggPiBudW1CYW5kcykgewotICAgICAgICAgICAgLy8gYXd0LjY0PVRoZSBudW1iZXIgb2YgdGhlIGJhbmRzIGluIHRoZSBzdWJzZXQgaXMgZ3JlYXRlciB0aGFuIHRoZQotICAgICAgICAgICAgLy8gbnVtYmVyIG9mIGJhbmRzIGluIHRoZSBzYW1wbGUgbW9kZWwKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGludCBpbmRpY2VzW10gPSBuZXcgaW50W2JhbmRzLmxlbmd0aF07Ci0gICAgICAgIGludCBvZmZzZXRzW10gPSBuZXcgaW50W2JhbmRzLmxlbmd0aF07Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBiYW5kcy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgaW5kaWNlc1tpXSA9IGJhbmtJbmRpY2VzW2JhbmRzW2ldXTsKLSAgICAgICAgICAgIG9mZnNldHNbaV0gPSBiYW5kT2Zmc2V0c1tiYW5kc1tpXV07Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbmV3IEJhbmRlZFNhbXBsZU1vZGVsKGRhdGFUeXBlLCB3aWR0aCwgaGVpZ2h0LCBzY2FubGluZVN0cmlkZSwgaW5kaWNlcywgb2Zmc2V0cyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBnZXREYXRhRWxlbWVudHMoaW50IHgsIGludCB5LCBPYmplY3Qgb2JqLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgc3dpdGNoIChkYXRhVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURTogewotICAgICAgICAgICAgICAgIGJ5dGUgYmRhdGFbXTsKLQotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBiZGF0YSA9IG5ldyBieXRlW251bUJhbmRzXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBiZGF0YSA9IChieXRlW10pb2JqOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBiZGF0YVtpXSA9IChieXRlKWdldFNhbXBsZSh4LCB5LCBpLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBvYmogPSBiZGF0YTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1NIT1JUOgotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOiB7Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2RhdGFbXTsKLQotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBzZGF0YSA9IG5ldyBzaG9ydFtudW1CYW5kc107Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgc2RhdGEgPSAoc2hvcnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1CYW5kczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNkYXRhW2ldID0gKHNob3J0KWdldFNhbXBsZSh4LCB5LCBpLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBvYmogPSBzZGF0YTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDogewotICAgICAgICAgICAgICAgIGludCBpZGF0YVtdOwotCi0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlkYXRhID0gbmV3IGludFtudW1CYW5kc107Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgaWRhdGEgPSAoaW50W10pb2JqOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBpZGF0YVtpXSA9IGdldFNhbXBsZSh4LCB5LCBpLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBvYmogPSBpZGF0YTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FUOiB7Ci0gICAgICAgICAgICAgICAgZmxvYXQgZmRhdGFbXTsKLQotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBmZGF0YSA9IG5ldyBmbG9hdFtudW1CYW5kc107Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgZmRhdGEgPSAoZmxvYXRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1CYW5kczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZkYXRhW2ldID0gZ2V0U2FtcGxlRmxvYXQoeCwgeSwgaSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgb2JqID0gZmRhdGE7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9ET1VCTEU6IHsKLSAgICAgICAgICAgICAgICBkb3VibGUgZGRhdGFbXTsKLQotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBkZGF0YSA9IG5ldyBkb3VibGVbbnVtQmFuZHNdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGRkYXRhID0gKGRvdWJsZVtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgZGRhdGFbaV0gPSBnZXRTYW1wbGVEb3VibGUoeCwgeSwgaSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgb2JqID0gZGRhdGE7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gb2JqOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnRbXSBnZXRQaXhlbChpbnQgeCwgaW50IHksIGludCBpQXJyYXlbXSwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGludCBwaXhlbFtdOwotICAgICAgICBpZiAoaUFycmF5ID09IG51bGwpIHsKLSAgICAgICAgICAgIHBpeGVsID0gbmV3IGludFtudW1CYW5kc107Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBwaXhlbCA9IGlBcnJheTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgcGl4ZWxbaV0gPSBnZXRTYW1wbGUoeCwgeSwgaSwgZGF0YSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcGl4ZWw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID49IHRoaXMud2lkdGggfHwgeSA+PSB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZGF0YS5nZXRFbGVtKGJhbmtJbmRpY2VzW2JdLCB5ICogc2NhbmxpbmVTdHJpZGUgKyB4ICsgYmFuZE9mZnNldHNbYl0pOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBkb3VibGUgZ2V0U2FtcGxlRG91YmxlKGludCB4LCBpbnQgeSwgaW50IGIsIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGRhdGEuZ2V0RWxlbURvdWJsZShiYW5rSW5kaWNlc1tiXSwgeSAqIHNjYW5saW5lU3RyaWRlICsgeCArIGJhbmRPZmZzZXRzW2JdKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXQgZ2V0U2FtcGxlRmxvYXQoaW50IHgsIGludCB5LCBpbnQgYiwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID49IHRoaXMud2lkdGggfHwgeSA+PSB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZGF0YS5nZXRFbGVtRmxvYXQoYmFua0luZGljZXNbYl0sIHkgKiBzY2FubGluZVN0cmlkZSArIHggKyBiYW5kT2Zmc2V0c1tiXSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpbnQgc2FtcGxlc1tdOwotICAgICAgICBpbnQgaWR4ID0gMDsKLQotICAgICAgICBpZiAoaUFycmF5ID09IG51bGwpIHsKLSAgICAgICAgICAgIHNhbXBsZXMgPSBuZXcgaW50W3cgKiBoXTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHNhbXBsZXMgPSBpQXJyYXk7Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIHNhbXBsZXNbaWR4KytdID0gZ2V0U2FtcGxlKGosIGksIGIsIGRhdGEpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHNhbXBsZXM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgaW50IGhhc2ggPSBzdXBlci5oYXNoQ29kZSgpOwotICAgICAgICBpbnQgdG1wID0gaGFzaCA+Pj4gODsKLSAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgaGFzaCB8PSB0bXA7Ci0KLSAgICAgICAgcmV0dXJuIGhhc2ggXiAweDU1OwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldERhdGFFbGVtZW50cyhpbnQgeCwgaW50IHksIE9iamVjdCBvYmosIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBzd2l0Y2ggKGRhdGFUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGJ5dGUgYmRhdGFbXSA9IChieXRlW10pb2JqOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBzZXRTYW1wbGUoeCwgeSwgaSwgYmRhdGFbaV0gJiAweGZmLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1NIT1JUOgotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNkYXRhW10gPSAoc2hvcnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1CYW5kczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNldFNhbXBsZSh4LCB5LCBpLCBzZGF0YVtpXSAmIDB4ZmZmZiwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgaW50IGlkYXRhW10gPSAoaW50W10pb2JqOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBzZXRTYW1wbGUoeCwgeSwgaSwgaWRhdGFbaV0sIGRhdGEpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfRkxPQVQ6Ci0gICAgICAgICAgICAgICAgZmxvYXQgZmRhdGFbXSA9IChmbG9hdFtdKW9iajsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgc2V0U2FtcGxlKHgsIHksIGksIGZkYXRhW2ldLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRToKLSAgICAgICAgICAgICAgICBkb3VibGUgZGRhdGFbXSA9IChkb3VibGVbXSlvYmo7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1CYW5kczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNldFNhbXBsZSh4LCB5LCBpLCBkZGF0YVtpXSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWwoaW50IHgsIGludCB5LCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgIHNldFNhbXBsZSh4LCB5LCBpLCBpQXJyYXlbaV0sIGRhdGEpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpbnQgaWR4ID0gMDsKLQotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtQmFuZHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICBzZXRTYW1wbGUoaiwgaSwgbiwgaUFycmF5W2lkeCsrXSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0U2FtcGxlKGludCB4LCBpbnQgeSwgaW50IGIsIGRvdWJsZSBzLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGRhdGEuc2V0RWxlbURvdWJsZShiYW5rSW5kaWNlc1tiXSwgeSAqIHNjYW5saW5lU3RyaWRlICsgeCArIGJhbmRPZmZzZXRzW2JdLCBzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgZmxvYXQgcywgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID49IHRoaXMud2lkdGggfHwgeSA+PSB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBkYXRhLnNldEVsZW1GbG9hdChiYW5rSW5kaWNlc1tiXSwgeSAqIHNjYW5saW5lU3RyaWRlICsgeCArIGJhbmRPZmZzZXRzW2JdLCBzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgaW50IHMsIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgZGF0YS5zZXRFbGVtKGJhbmtJbmRpY2VzW2JdLCB5ICogc2NhbmxpbmVTdHJpZGUgKyB4ICsgYmFuZE9mZnNldHNbYl0sIHMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpbnQgaWR4ID0gMDsKLQotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIHNldFNhbXBsZShqLCBpLCBiLCBpQXJyYXlbaWR4KytdLCBkYXRhKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvQnVmZmVyU3RyYXRlZ3kuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9CdWZmZXJTdHJhdGVneS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzYzg3NzlkLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9CdWZmZXJTdHJhdGVneS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNzQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5CdWZmZXJDYXBhYmlsaXRpZXM7Ci1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3M7Ci0KLS8qKgotICogVGhlIEJ1ZmZlclN0cmF0ZWd5IGFic3RyYWN0IGNsYXNzIHByb3ZpZGVzIGFuIG9wcG9ydHVuaXR5IHRvIG9yZ2FuaXplIHRoZQotICogYnVmZmVycyBmb3IgYSBDYW52YXMgb3IgV2luZG93LiBUaGUgQnVmZmVyU3RyYXRlZ3kgaW1wbGVtZW50YXRpb24gZGVwZW5kcyBvbgotICogaGFyZHdhcmUgYW5kIHNvZnR3YXJlIGxpbWl0YXRpb25zLiBUaGVzZSBsaW1pdGF0aW9ucyBhcmUgZGV0ZWN0YWJsZSB0aHJvdWdoCi0gKiB0aGUgY2FwYWJpbGl0aWVzIG9iamVjdCB3aGljaCBjYW4gYmUgb2J0YWluZWQgYnkgdGhlIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBvZgotICogdGhlIENhbnZhcyBvciBXaW5kb3cuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgQnVmZmVyU3RyYXRlZ3kgewotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoZSBkcmF3aW5nIGJ1ZmZlciB3YXMgbG9zdCBzaW5jZSB0aGUgbGFzdCBjYWxsIG9mCi0gICAgICogZ2V0RHJhd0dyYXBoaWNzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgZHJhd2luZyBidWZmZXIgd2FzIGxvc3Qgc2luY2UgdGhlIGxhc3QgY2FsbCBvZgotICAgICAqICAgICAgICAgZ2V0RHJhd0dyYXBoaWNzLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGJvb2xlYW4gY29udGVudHNMb3N0KCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIGRyYXdpbmcgYnVmZmVyIGlzIHJlc3RvcmVkIGZyb20gYSBsb3N0IHN0YXRlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgZHJhd2luZyBidWZmZXIgaXMgcmVzdG9yZWQgZnJvbSBhIGxvc3Qgc3RhdGUsIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGJvb2xlYW4gY29udGVudHNSZXN0b3JlZCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQnVmZmVyQ2FwYWJpbGl0aWVzIG9mIEJ1ZmZlclN0cmF0ZWd5LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEJ1ZmZlckNhcGFiaWxpdGllcyBvZiBCdWZmZXJTdHJhdGVneS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgQnVmZmVyQ2FwYWJpbGl0aWVzIGdldENhcGFiaWxpdGllcygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgR3JhcGhpY3Mgb2JqZWN0IHRvIHVzZSB0byBkcmF3IHRvIHRoZSBidWZmZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgR3JhcGhpY3Mgb2JqZWN0IHRvIHVzZSB0byBkcmF3IHRvIHRoZSBidWZmZXIuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEdyYXBoaWNzIGdldERyYXdHcmFwaGljcygpOwotCi0gICAgLyoqCi0gICAgICogU2hvd3MgdGhlIG5leHQgYXZhaWxhYmxlIGJ1ZmZlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzaG93KCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9CdWZmZXJlZEltYWdlLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvQnVmZmVyZWRJbWFnZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjOWQ1OGQ5Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9CdWZmZXJlZEltYWdlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw5NTIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBjb20uYW5kcm9pZC5pbnRlcm5hbC5hd3QuQW5kcm9pZEdyYXBoaWNzMkQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljczsKLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljczJEOwotaW1wb3J0IGphdmEuYXd0LkdyYXBoaWNzRW52aXJvbm1lbnQ7Ci1pbXBvcnQgamF2YS5hd3QuSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuUG9pbnQ7Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LlRyYW5zcGFyZW5jeTsKLWltcG9ydCBqYXZhLmF3dC5jb2xvci5Db2xvclNwYWNlOwotaW1wb3J0IGphdmEudXRpbC5FbnVtZXJhdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaHRhYmxlOwotaW1wb3J0IGphdmEudXRpbC5WZWN0b3I7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLkltYWdlU3VyZmFjZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLlN1cmZhY2U7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZS5CdWZmZXJlZEltYWdlU291cmNlOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBCdWZmZXJlZEltYWdlIGNsYXNzIGRlc2NyaWJlcyBhbiBJbWFnZSB3aGljaCBjb250YWlucyBhIGJ1ZmZlciBvZiBpbWFnZQotICogZGF0YSBhbmQgaW5jbHVkZXMgYSBDb2xvck1vZGVsIGFuZCBhIFJhc3RlciBmb3IgdGhpcyBkYXRhLiBUaGlzIGNsYXNzCi0gKiBwcm92aWRlcyBtZXRob2RzIGZvciBvYnRhaW5pbmcgYW5kIHNldHRpbmcgdGhlIFJhc3RlciBhbmQgZm9yIG1hbmlwdWxhdGluZwotICogdGhlIENvbG9yTW9kZWwgcGFyYW1ldGVycy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBCdWZmZXJlZEltYWdlIGV4dGVuZHMgSW1hZ2UgaW1wbGVtZW50cyBXcml0YWJsZVJlbmRlcmVkSW1hZ2UsIFRyYW5zcGFyZW5jeSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9DVVNUT00gaW5kaWNhdGVzIHRoYXQgSW1hZ2UgdHlwZSBpcyB1bmtub3duLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfQ1VTVE9NID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0lOVF9SR0IgaW5kaWNhdGVzIGFuIGltYWdlIHdpdGggOCBiaXQgUkdCIGNvbG9yCi0gICAgICogY29tcG9uZW50cywgaXQgaGFzIGEgRGlyZWN0Q29sb3JNb2RlbCB3aXRob3V0IGFscGhhLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfSU5UX1JHQiA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9JTlRfQVJHQiBpbmRpY2F0ZXMgYW4gaW1hZ2Ugd2l0aCA4IGJpdCBSR0JBIGNvbG9yCi0gICAgICogY29tcG9uZW50cywgaXQgaGFzIGEgRGlyZWN0Q29sb3JNb2RlbCB3aXRoIGFscGhhLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfSU5UX0FSR0IgPSAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfSU5UX0FSR0JfUFJFIGluZGljYXRlcyBhbiBpbWFnZSB3aXRoIDggYml0IFJHQkEgY29sb3IKLSAgICAgKiBjb21wb25lbnRzLCBpdCBoYXMgYSBEaXJlY3RDb2xvck1vZGVsIHdpdGggYWxwaGEsIGFuZCBpbWFnZSBkYXRhIGlzCi0gICAgICogcHJlLW11bHRpcGxpZWQgYnkgYWxwaGEuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9JTlRfQVJHQl9QUkUgPSAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfSU5UX0JHUiBpbmRpY2F0ZXMgYW4gaW1hZ2Ugd2l0aCA4IGJpdCBSR0IgY29sb3IKLSAgICAgKiBjb21wb25lbnRzLCBCR1IgY29sb3IgbW9kZWwgKHdpdGggdGhlIGNvbG9ycyBCbHVlLCBHcmVlbiwgYW5kIFJlZCkuIFRoZXJlCi0gICAgICogaXMgbm8gYWxwaGEuIFRoZSBpbWFnZSBoYXMgYSBEaXJlY3RDb2xvck1vZGVsLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfSU5UX0JHUiA9IDQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV8zQllURV9CR1IgaW5kaWNhdGVzIGFuIGltYWdlIHdpdGggOCBiaXQgUkdCIGNvbG9yCi0gICAgICogY29tcG9uZW50cywgQkdSIGNvbG9yIG1vZGVsICh3aXRoIHRoZSBjb2xvcnMgQmx1ZSwgR3JlZW4sIGFuZCBSZWQgc3RvcmVkCi0gICAgICogaW4gMyBieXRlcykuIFRoZXJlIGlzIG5vIGFscGhhLiBUaGUgaW1hZ2UgaGFzIGEgQ29tcG9uZW50Q29sb3JNb2RlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFXzNCWVRFX0JHUiA9IDU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV80QllURV9BQkdSIGluZGljYXRlcyBhbiBpbWFnZSB3aXRoIDggYml0IFJHQkEgY29sb3IKLSAgICAgKiBjb21wb25lbnRzIHN0b3JlZCBpbiAzIGJ5dGVzIGFuZCAxIGJ5dGUgb2YgYWxwaGEuIEl0IGhhcyBhCi0gICAgICogQ29tcG9uZW50Q29sb3JNb2RlbCB3aXRoIGFscGhhLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfNEJZVEVfQUJHUiA9IDY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV80QllURV9BQkdSX1BSRSBpbmRpY2F0ZXMgYW4gaW1hZ2Ugd2l0aCA4IGJpdCBSR0JBIGNvbG9yCi0gICAgICogY29tcG9uZW50cyBzdG9yZWQgaW4gMyBieXRlcyBhbmQgMSBieXRlIGZvciBhbHBoYS4gVGhlIGltYWdlIGhhcyBhCi0gICAgICogQ29tcG9uZW50Q29sb3JNb2RlbCB3aXRoIGFscGhhLiBUaGUgY29sb3IgZGF0YSBpcyBwcmUtbXVsdGlwbGllZCB3aXRoCi0gICAgICogYWxwaGEuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV80QllURV9BQkdSX1BSRSA9IDc7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9VU0hPUlRfNTY1X1JHQiBpbmRpY2F0ZXMgYW4gaW1hZ2Ugd2l0aCA1NjUgUkdCIGNvbG9yCi0gICAgICogY29tcG9uZW50cyAoNS1iaXRzIHJlZCwgNi1iaXRzIGdyZWVuLCA1LWJpdHMgYmx1ZSkgd2l0aCBubyBhbHBoYS4gVGhpcwotICAgICAqIGltYWdlIGhhcyBhIERpcmVjdENvbG9yTW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9VU0hPUlRfNTY1X1JHQiA9IDg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9VU0hPUlRfNTU1X1JHQiBpbmRpY2F0ZXMgYW4gaW1hZ2Ugd2l0aCA1NTUgUkdCIGNvbG9yCi0gICAgICogY29tcG9uZW50cyAoNS1iaXRzIHJlZCwgNS1iaXRzIGdyZWVuLCA1LWJpdHMgYmx1ZSkgd2l0aCBubyBhbHBoYS4gVGhpcwotICAgICAqIGltYWdlIGhhcyBhIERpcmVjdENvbG9yTW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9VU0hPUlRfNTU1X1JHQiA9IDk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9CWVRFX0dSQVkgaW5kaWNhdGVzIGEgdW5zaWduZWQgYnl0ZSBpbWFnZS4gVGhpcyBpbWFnZQotICAgICAqIGhhcyBhIENvbXBvbmVudENvbG9yTW9kZWwgd2l0aCBhIENTX0dSQVkgQ29sb3JTcGFjZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBUWVBFX0JZVEVfR1JBWSA9IDEwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfVVNIT1JUX0dSQVkgaW5kaWNhdGVzIGFuIHVuc2lnbmVkIHNob3J0IGltYWdlLiBUaGlzCi0gICAgICogaW1hZ2UgaGFzIGEgQ29tcG9uZW50Q29sb3JNb2RlbCB3aXRoIGEgQ1NfR1JBWSBDb2xvclNwYWNlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfVVNIT1JUX0dSQVkgPSAxMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0JZVEVfQklOQVJZIGluZGljYXRlcyBhbiBvcGFxdWUgYnl0ZS1wYWNrZWQgMSwgMiBvciA0Ci0gICAgICogYml0IGltYWdlLiBUaGUgaW1hZ2UgaGFzIGFuIEluZGV4Q29sb3JNb2RlbCB3aXRob3V0IGFscGhhLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfQllURV9CSU5BUlkgPSAxMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUWVBFX0JZVEVfSU5ERVhFRCBpbmRpY2F0ZXMgYW4gaW5kZXhlZCBieXRlIGltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfQllURV9JTkRFWEVEID0gMTM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQUxQSEFfTUFTSy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgQUxQSEFfTUFTSyA9IDB4ZmYwMDAwMDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUkVEX01BU0suCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFJFRF9NQVNLID0gMHgwMGZmMDAwMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBHUkVFTl9NQVNLLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBHUkVFTl9NQVNLID0gMHgwMDAwZmYwMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBCTFVFX01BU0suCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJMVUVfTUFTSyA9IDB4MDAwMDAwZmY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUkVEX0JHUl9NQVNLLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBSRURfQkdSX01BU0sgPSAweDAwMDAwMGZmOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEdSRUVOX0JHUl9NQVNLLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBHUkVFTl9CR1JfTUFTSyA9IDB4MDAwMGZmMDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQkxVRV9CR1JfTUFTSy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgQkxVRV9CR1JfTUFTSyA9IDB4MDBmZjAwMDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUkVEXzU2NV9NQVNLLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBSRURfNTY1X01BU0sgPSAweGY4MDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgR1JFRU5fNTY1X01BU0suCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEdSRUVOXzU2NV9NQVNLID0gMHgwN2UwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEJMVUVfNTY1X01BU0suCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJMVUVfNTY1X01BU0sgPSAweDAwMWY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUkVEXzU1NV9NQVNLLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBSRURfNTU1X01BU0sgPSAweDdjMDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgR1JFRU5fNTU1X01BU0suCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEdSRUVOXzU1NV9NQVNLID0gMHgwM2UwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEJMVUVfNTU1X01BU0suCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJMVUVfNTU1X01BU0sgPSAweDAwMWY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY20uCi0gICAgICovCi0gICAgcHJpdmF0ZSBDb2xvck1vZGVsIGNtOwotCi0gICAgLyoqCi0gICAgICogVGhlIHJhc3Rlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIFdyaXRhYmxlUmFzdGVyIHJhc3RlcjsKLQotICAgIC8qKgotICAgICAqIFRoZSBpbWFnZSB0eXBlLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgaW50IGltYWdlVHlwZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBwcm9wZXJ0aWVzLgotICAgICAqLwotICAgIHByaXZhdGUgSGFzaHRhYmxlPD8sID8+IHByb3BlcnRpZXM7Ci0KLSAgICAvLyBTdXJmYWNlIG9mIHRoZSBCdWZmZXJlZCBJbWFnZSAtIHVzZWQgZm9yIGJsaXR0aW5nIG9uZSBCdWZmZXJlZCBJbWFnZQotICAgIC8vIG9uIHRoZSBvdGhlciBvbmUgb3Igb24gdGhlIENvbXBvbmVudAotICAgIC8qKgotICAgICAqIFRoZSBpbWFnZSBzdXJmLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgSW1hZ2VTdXJmYWNlIGltYWdlU3VyZjsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBCdWZmZXJlZEltYWdlIHdpdGggdGhlIHNwZWNpZmllZCBDb2xvck1vZGVsLCBhbmQKLSAgICAgKiBXcml0YWJsZVJhc3RlciBvYmplY3RzLiBUaGUgUmFzdGVyIGRhdGEgY2FuIGJlIGJlIGRpdmlkZWQgb3IgbXVsdGlwbGllZAotICAgICAqIGJ5IGFscGhhLiBJdCBkZXBlbmRzIG9uIHRoZSBhbHBoYVByZW11bHRpcGxpZWQgc3RhdGUgaW4gdGhlIENvbG9yTW9kZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNtCi0gICAgICogICAgICAgICAgICB0aGUgQ29sb3JNb2RlbCBvZiB0aGUgbmV3IGltYWdlLgotICAgICAqIEBwYXJhbSByYXN0ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBXcml0YWJsZVJhc3RlciBvZiB0aGUgbmV3IGltYWdlLgotICAgICAqIEBwYXJhbSBpc1Jhc3RlclByZW11bHRpcGxpZWQKLSAgICAgKiAgICAgICAgICAgIGlmIHRydWUgdGhlIGRhdGEgb2YgdGhlIHNwZWNpZmllZCBSYXN0ZXIgaXMgcHJlLW11bHRpcGxpZWQgYnkKLSAgICAgKiAgICAgICAgICAgIGFscGhhLgotICAgICAqIEBwYXJhbSBwcm9wZXJ0aWVzCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydGllcyBvZiBuZXcgSW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UoQ29sb3JNb2RlbCBjbSwgV3JpdGFibGVSYXN0ZXIgcmFzdGVyLCBib29sZWFuIGlzUmFzdGVyUHJlbXVsdGlwbGllZCwKLSAgICAgICAgICAgIEhhc2h0YWJsZTw/LCA/PiBwcm9wZXJ0aWVzKSB7Ci0gICAgICAgIGlmICghY20uaXNDb21wYXRpYmxlUmFzdGVyKHJhc3RlcikpIHsKLSAgICAgICAgICAgIC8vIGF3dC40RD1UaGUgcmFzdGVyIGlzIGluY29tcGF0aWJsZSB3aXRoIHRoaXMgQ29sb3JNb2RlbAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40RCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHJhc3Rlci5nZXRNaW5YKCkgIT0gMCB8fCByYXN0ZXIuZ2V0TWluWSgpICE9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMjg9bWluWCBvciBtaW5ZIG9mIHRoaXMgcmFzdGVyIG5vdCBlcXVhbCB0byB6ZXJvCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIyOCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5jbSA9IGNtOwotICAgICAgICB0aGlzLnJhc3RlciA9IHJhc3RlcjsKLSAgICAgICAgdGhpcy5wcm9wZXJ0aWVzID0gcHJvcGVydGllczsKLQotICAgICAgICBjb2VyY2VEYXRhKGlzUmFzdGVyUHJlbXVsdGlwbGllZCk7Ci0KLSAgICAgICAgaW1hZ2VUeXBlID0gU3VyZmFjZS5nZXRUeXBlKGNtLCByYXN0ZXIpOwotCi0gICAgICAgIGltYWdlU3VyZiA9IGNyZWF0ZUltYWdlU3VyZmFjZShpbWFnZVR5cGUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBCdWZmZXJlZEltYWdlIHdpdGggdGhlIHNwZWNpZmllZCB3aWR0aCwgaGVpZ2h0Ci0gICAgICogcHJlZGVmaW5lZCBpbWFnZSB0eXBlIChUWVBFX0JZVEVfQklOQVJZIG9yIFRZUEVfQllURV9JTkRFWEVEKSBhbmQgdGhlCi0gICAgICogc3BlY2lmaWVkIEluZGV4Q29sb3JNb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiBuZXcgaW1hZ2UuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBuZXcgaW1hZ2UuCi0gICAgICogQHBhcmFtIGltYWdlVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHByZWRlZmluZWQgaW1hZ2UgdHlwZS4KLSAgICAgKiBAcGFyYW0gY20KLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgSW5kZXhDb2xvck1vZGVsLgotICAgICAqLwotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGltYWdlVHlwZSwgSW5kZXhDb2xvck1vZGVsIGNtKSB7Ci0gICAgICAgIHN3aXRjaCAoaW1hZ2VUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIFRZUEVfQllURV9CSU5BUlk6Ci0gICAgICAgICAgICAgICAgaWYgKGNtLmhhc0FscGhhKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjIyNz1UaGlzIGltYWdlIHR5cGUgY2FuJ3QgaGF2ZSBhbHBoYQotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIyNyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpbnQgcGl4ZWxfYml0cyA9IDA7Ci0gICAgICAgICAgICAgICAgaW50IG1hcFNpemUgPSBjbS5nZXRNYXBTaXplKCk7Ci0gICAgICAgICAgICAgICAgaWYgKG1hcFNpemUgPD0gMikgewotICAgICAgICAgICAgICAgICAgICBwaXhlbF9iaXRzID0gMTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKG1hcFNpemUgPD0gNCkgewotICAgICAgICAgICAgICAgICAgICBwaXhlbF9iaXRzID0gMjsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKG1hcFNpemUgPD0gMTYpIHsKLSAgICAgICAgICAgICAgICAgICAgcGl4ZWxfYml0cyA9IDQ7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjIyMT1UaGUgaW1hZ2VUeXBlIGlzIFRZUEVfQllURV9CSU5BUlkgYW5kIHRoZSBjb2xvcgotICAgICAgICAgICAgICAgICAgICAvLyBtYXAgaGFzIG1vcmUgdGhhbiAxNiBlbnRyaWVzCi0gICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjIxIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgcmFzdGVyID0gUmFzdGVyLmNyZWF0ZVBhY2tlZFJhc3RlcihEYXRhQnVmZmVyLlRZUEVfQllURSwgd2lkdGgsIGhlaWdodCwgMSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHBpeGVsX2JpdHMsIG51bGwpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIFRZUEVfQllURV9JTkRFWEVEOgotICAgICAgICAgICAgICAgIHJhc3RlciA9IFJhc3Rlci5jcmVhdGVJbnRlcmxlYXZlZFJhc3RlcihEYXRhQnVmZmVyLlRZUEVfQllURSwgd2lkdGgsIGhlaWdodCwgMSwKLSAgICAgICAgICAgICAgICAgICAgICAgIG51bGwpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4yMjI9VGhlIGltYWdlVHlwZSBpcyBub3QgVFlQRV9CWVRFX0JJTkFSWSBvcgotICAgICAgICAgICAgICAgIC8vIFRZUEVfQllURV9JTkRFWEVECi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMjIiKSk7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKCFjbS5pc0NvbXBhdGlibGVSYXN0ZXIocmFzdGVyKSkgewotICAgICAgICAgICAgLy8gYXd0LjIyMz1UaGUgaW1hZ2VUeXBlIGlzIG5vdCBjb21wYXRpYmxlIHdpdGggQ29sb3JNb2RlbAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMuY20gPSBjbTsKLSAgICAgICAgdGhpcy5pbWFnZVR5cGUgPSBpbWFnZVR5cGU7Ci0gICAgICAgIGltYWdlU3VyZiA9IGNyZWF0ZUltYWdlU3VyZmFjZShpbWFnZVR5cGUpOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEJ1ZmZlcmVkSW1hZ2Ugd2l0aCB0aGUgc3BlY2lmaWVkIHdpZHRoLCBoZWlnaHQgYW5kCi0gICAgICogcHJlZGVmaW5lZCBpbWFnZSB0eXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIG5ldyBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIG5ldyBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaW1hZ2VUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgcHJlZGVmaW5lZCBpbWFnZSB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGltYWdlVHlwZSkgewotCi0gICAgICAgIHN3aXRjaCAoaW1hZ2VUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIFRZUEVfSU5UX1JHQjoKLSAgICAgICAgICAgICAgICBjbSA9IG5ldyBEaXJlY3RDb2xvck1vZGVsKDI0LCBSRURfTUFTSywgR1JFRU5fTUFTSywgQkxVRV9NQVNLKTsKLSAgICAgICAgICAgICAgICByYXN0ZXIgPSBjbS5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIod2lkdGgsIGhlaWdodCk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgVFlQRV9JTlRfQVJHQjoKLSAgICAgICAgICAgICAgICBjbSA9IENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpOwotICAgICAgICAgICAgICAgIHJhc3RlciA9IGNtLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3Rlcih3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBUWVBFX0lOVF9BUkdCX1BSRToKLSAgICAgICAgICAgICAgICBjbSA9IG5ldyBEaXJlY3RDb2xvck1vZGVsKENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19zUkdCKSwgMzIsIFJFRF9NQVNLLAotICAgICAgICAgICAgICAgICAgICAgICAgR1JFRU5fTUFTSywgQkxVRV9NQVNLLCBBTFBIQV9NQVNLLCB0cnVlLCBEYXRhQnVmZmVyLlRZUEVfSU5UKTsKLQotICAgICAgICAgICAgICAgIHJhc3RlciA9IGNtLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3Rlcih3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBUWVBFX0lOVF9CR1I6Ci0gICAgICAgICAgICAgICAgY20gPSBuZXcgRGlyZWN0Q29sb3JNb2RlbCgyNCwgUkVEX0JHUl9NQVNLLCBHUkVFTl9CR1JfTUFTSywgQkxVRV9CR1JfTUFTSyk7Ci0KLSAgICAgICAgICAgICAgICByYXN0ZXIgPSBjbS5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIod2lkdGgsIGhlaWdodCk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgVFlQRV8zQllURV9CR1I6IHsKLSAgICAgICAgICAgICAgICBpbnQgYml0c1tdID0gewotICAgICAgICAgICAgICAgICAgICAgICAgOCwgOCwgOAotICAgICAgICAgICAgICAgIH07Ci0gICAgICAgICAgICAgICAgaW50IGJhbmRPZmZzZXRzW10gPSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAyLCAxLCAwCi0gICAgICAgICAgICAgICAgfTsKLSAgICAgICAgICAgICAgICBjbSA9IG5ldyBDb21wb25lbnRDb2xvck1vZGVsKENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19zUkdCKSwgYml0cywKLSAgICAgICAgICAgICAgICAgICAgICAgIGZhbHNlLCBmYWxzZSwgVHJhbnNwYXJlbmN5Lk9QQVFVRSwgRGF0YUJ1ZmZlci5UWVBFX0JZVEUpOwotCi0gICAgICAgICAgICAgICAgcmFzdGVyID0gUmFzdGVyLmNyZWF0ZUludGVybGVhdmVkUmFzdGVyKERhdGFCdWZmZXIuVFlQRV9CWVRFLCB3aWR0aCwgaGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICAgICAgd2lkdGggKiAzLCAzLCBiYW5kT2Zmc2V0cywgbnVsbCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgVFlQRV80QllURV9BQkdSOiB7Ci0gICAgICAgICAgICAgICAgaW50IGJpdHNbXSA9IHsKLSAgICAgICAgICAgICAgICAgICAgICAgIDgsIDgsIDgsIDgKLSAgICAgICAgICAgICAgICB9OwotICAgICAgICAgICAgICAgIGludCBiYW5kT2Zmc2V0c1tdID0gewotICAgICAgICAgICAgICAgICAgICAgICAgMywgMiwgMSwgMAotICAgICAgICAgICAgICAgIH07Ci0gICAgICAgICAgICAgICAgY20gPSBuZXcgQ29tcG9uZW50Q29sb3JNb2RlbChDb2xvclNwYWNlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1Nfc1JHQiksIGJpdHMsCi0gICAgICAgICAgICAgICAgICAgICAgICB0cnVlLCBmYWxzZSwgVHJhbnNwYXJlbmN5LlRSQU5TTFVDRU5ULCBEYXRhQnVmZmVyLlRZUEVfQllURSk7Ci0KLSAgICAgICAgICAgICAgICByYXN0ZXIgPSBSYXN0ZXIuY3JlYXRlSW50ZXJsZWF2ZWRSYXN0ZXIoRGF0YUJ1ZmZlci5UWVBFX0JZVEUsIHdpZHRoLCBoZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgICAgICB3aWR0aCAqIDQsIDQsIGJhbmRPZmZzZXRzLCBudWxsKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBUWVBFXzRCWVRFX0FCR1JfUFJFOiB7Ci0gICAgICAgICAgICAgICAgaW50IGJpdHNbXSA9IHsKLSAgICAgICAgICAgICAgICAgICAgICAgIDgsIDgsIDgsIDgKLSAgICAgICAgICAgICAgICB9OwotICAgICAgICAgICAgICAgIGludCBiYW5kT2Zmc2V0c1tdID0gewotICAgICAgICAgICAgICAgICAgICAgICAgMywgMiwgMSwgMAotICAgICAgICAgICAgICAgIH07Ci0gICAgICAgICAgICAgICAgY20gPSBuZXcgQ29tcG9uZW50Q29sb3JNb2RlbChDb2xvclNwYWNlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1Nfc1JHQiksIGJpdHMsCi0gICAgICAgICAgICAgICAgICAgICAgICB0cnVlLCB0cnVlLCBUcmFuc3BhcmVuY3kuVFJBTlNMVUNFTlQsIERhdGFCdWZmZXIuVFlQRV9CWVRFKTsKLQotICAgICAgICAgICAgICAgIHJhc3RlciA9IFJhc3Rlci5jcmVhdGVJbnRlcmxlYXZlZFJhc3RlcihEYXRhQnVmZmVyLlRZUEVfQllURSwgd2lkdGgsIGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgICAgIHdpZHRoICogNCwgNCwgYmFuZE9mZnNldHMsIG51bGwpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIFRZUEVfVVNIT1JUXzU2NV9SR0I6Ci0gICAgICAgICAgICAgICAgY20gPSBuZXcgRGlyZWN0Q29sb3JNb2RlbChDb2xvclNwYWNlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1Nfc1JHQiksIDE2LAotICAgICAgICAgICAgICAgICAgICAgICAgUkVEXzU2NV9NQVNLLCBHUkVFTl81NjVfTUFTSywgQkxVRV81NjVfTUFTSywgMCwgZmFsc2UsCi0gICAgICAgICAgICAgICAgICAgICAgICBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUKTsKLQotICAgICAgICAgICAgICAgIHJhc3RlciA9IGNtLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3Rlcih3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBUWVBFX1VTSE9SVF81NTVfUkdCOgotICAgICAgICAgICAgICAgIGNtID0gbmV3IERpcmVjdENvbG9yTW9kZWwoQ29sb3JTcGFjZS5nZXRJbnN0YW5jZShDb2xvclNwYWNlLkNTX3NSR0IpLCAxNSwKLSAgICAgICAgICAgICAgICAgICAgICAgIFJFRF81NTVfTUFTSywgR1JFRU5fNTU1X01BU0ssIEJMVUVfNTU1X01BU0ssIDAsIGZhbHNlLAotICAgICAgICAgICAgICAgICAgICAgICAgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCk7Ci0KLSAgICAgICAgICAgICAgICByYXN0ZXIgPSBjbS5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIod2lkdGgsIGhlaWdodCk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgVFlQRV9CWVRFX0dSQVk6IHsKLSAgICAgICAgICAgICAgICBpbnQgYml0c1tdID0gewotICAgICAgICAgICAgICAgICAgICA4Ci0gICAgICAgICAgICAgICAgfTsKLSAgICAgICAgICAgICAgICBjbSA9IG5ldyBDb21wb25lbnRDb2xvck1vZGVsKENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19HUkFZKSwgYml0cywKLSAgICAgICAgICAgICAgICAgICAgICAgIGZhbHNlLCBmYWxzZSwgVHJhbnNwYXJlbmN5Lk9QQVFVRSwgRGF0YUJ1ZmZlci5UWVBFX0JZVEUpOwotCi0gICAgICAgICAgICAgICAgcmFzdGVyID0gY20uY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKHdpZHRoLCBoZWlnaHQpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIFRZUEVfVVNIT1JUX0dSQVk6IHsKLSAgICAgICAgICAgICAgICBpbnQgYml0c1tdID0gewotICAgICAgICAgICAgICAgICAgICAxNgotICAgICAgICAgICAgICAgIH07Ci0gICAgICAgICAgICAgICAgY20gPSBuZXcgQ29tcG9uZW50Q29sb3JNb2RlbChDb2xvclNwYWNlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1NfR1JBWSksIGJpdHMsCi0gICAgICAgICAgICAgICAgICAgICAgICBmYWxzZSwgZmFsc2UsIFRyYW5zcGFyZW5jeS5PUEFRVUUsIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQpOwotICAgICAgICAgICAgICAgIHJhc3RlciA9IGNtLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3Rlcih3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBUWVBFX0JZVEVfQklOQVJZOiB7Ci0gICAgICAgICAgICAgICAgaW50IGNvbG9yTWFwW10gPSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAwLCAweGZmZmZmZgotICAgICAgICAgICAgICAgIH07Ci0gICAgICAgICAgICAgICAgY20gPSBuZXcgSW5kZXhDb2xvck1vZGVsKDEsIDIsIGNvbG9yTWFwLCAwLCBmYWxzZSwgLTEsIERhdGFCdWZmZXIuVFlQRV9CWVRFKTsKLQotICAgICAgICAgICAgICAgIHJhc3RlciA9IFJhc3Rlci5jcmVhdGVQYWNrZWRSYXN0ZXIoRGF0YUJ1ZmZlci5UWVBFX0JZVEUsIHdpZHRoLCBoZWlnaHQsIDEsIDEsIG51bGwpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIFRZUEVfQllURV9JTkRFWEVEOiB7Ci0gICAgICAgICAgICAgICAgaW50IGNvbG9yTWFwW10gPSBuZXcgaW50WzI1Nl07Ci0gICAgICAgICAgICAgICAgaW50IGkgPSAwOwotICAgICAgICAgICAgICAgIGZvciAoaW50IHIgPSAwOyByIDwgMjU2OyByICs9IDUxKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGcgPSAwOyBnIDwgMjU2OyBnICs9IDUxKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBiID0gMDsgYiA8IDI1NjsgYiArPSA1MSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG9yTWFwW2ldID0gKHIgPDwgMTYpIHwgKGcgPDwgOCkgfCBiOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGkrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGludCBncmF5ID0gMHgxMjsKLSAgICAgICAgICAgICAgICBmb3IgKDsgaSA8IDI1NjsgaSsrLCBncmF5ICs9IDYpIHsKLSAgICAgICAgICAgICAgICAgICAgY29sb3JNYXBbaV0gPSAoZ3JheSA8PCAxNikgfCAoZ3JheSA8PCA4KSB8IGdyYXk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGNtID0gbmV3IEluZGV4Q29sb3JNb2RlbCg4LCAyNTYsIGNvbG9yTWFwLCAwLCBmYWxzZSwgLTEsIERhdGFCdWZmZXIuVFlQRV9CWVRFKTsKLSAgICAgICAgICAgICAgICByYXN0ZXIgPSBSYXN0ZXIuY3JlYXRlSW50ZXJsZWF2ZWRSYXN0ZXIoRGF0YUJ1ZmZlci5UWVBFX0JZVEUsIHdpZHRoLCBoZWlnaHQsIDEsCi0gICAgICAgICAgICAgICAgICAgICAgICBudWxsKTsKLQotICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjI0PVVua25vd24gaW1hZ2UgdHlwZQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjI0IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy5pbWFnZVR5cGUgPSBpbWFnZVR5cGU7Ci0gICAgICAgIGltYWdlU3VyZiA9IGNyZWF0ZUltYWdlU3VyZmFjZShpbWFnZVR5cGUpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgZ2V0UHJvcGVydHkoU3RyaW5nIG5hbWUsIEltYWdlT2JzZXJ2ZXIgb2JzZXJ2ZXIpIHsKLSAgICAgICAgcmV0dXJuIGdldFByb3BlcnR5KG5hbWUpOwotICAgIH0KLQotICAgIHB1YmxpYyBPYmplY3QgZ2V0UHJvcGVydHkoU3RyaW5nIG5hbWUpIHsKLSAgICAgICAgaWYgKG5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjIyNT1Qcm9wZXJ0eSBuYW1lIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMjUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZiAocHJvcGVydGllcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gSW1hZ2UuVW5kZWZpbmVkUHJvcGVydHk7Ci0gICAgICAgIH0KLSAgICAgICAgT2JqZWN0IHByb3BlcnR5ID0gcHJvcGVydGllcy5nZXQobmFtZSk7Ci0gICAgICAgIGlmIChwcm9wZXJ0eSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBwcm9wZXJ0eSA9IEltYWdlLlVuZGVmaW5lZFByb3BlcnR5OwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBwcm9wZXJ0eTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgY29weURhdGEoV3JpdGFibGVSYXN0ZXIgb3V0UmFzdGVyKSB7Ci0gICAgICAgIGlmIChvdXRSYXN0ZXIgPT0gbnVsbCkgewotICAgICAgICAgICAgb3V0UmFzdGVyID0gUmFzdGVyLmNyZWF0ZVdyaXRhYmxlUmFzdGVyKHJhc3Rlci5nZXRTYW1wbGVNb2RlbCgpLCBuZXcgUG9pbnQocmFzdGVyCi0gICAgICAgICAgICAgICAgICAgIC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSwgcmFzdGVyLmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWSgpKSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgdyA9IG91dFJhc3Rlci5nZXRXaWR0aCgpOwotICAgICAgICBpbnQgaCA9IG91dFJhc3Rlci5nZXRIZWlnaHQoKTsKLSAgICAgICAgaW50IG1pblggPSBvdXRSYXN0ZXIuZ2V0TWluWCgpOwotICAgICAgICBpbnQgbWluWSA9IG91dFJhc3Rlci5nZXRNaW5ZKCk7Ci0KLSAgICAgICAgT2JqZWN0IGRhdGEgPSBudWxsOwotCi0gICAgICAgIGRhdGEgPSByYXN0ZXIuZ2V0RGF0YUVsZW1lbnRzKG1pblgsIG1pblksIHcsIGgsIGRhdGEpOwotICAgICAgICBvdXRSYXN0ZXIuc2V0RGF0YUVsZW1lbnRzKG1pblgsIG1pblksIHcsIGgsIGRhdGEpOwotCi0gICAgICAgIHJldHVybiBvdXRSYXN0ZXI7Ci0gICAgfQotCi0gICAgcHVibGljIFJhc3RlciBnZXREYXRhKFJlY3RhbmdsZSByZWN0KSB7Ci0gICAgICAgIGludCBtaW5YID0gcmVjdC54OwotICAgICAgICBpbnQgbWluWSA9IHJlY3QueTsKLSAgICAgICAgaW50IHcgPSByZWN0LndpZHRoOwotICAgICAgICBpbnQgaCA9IHJlY3QuaGVpZ2h0OwotCi0gICAgICAgIFNhbXBsZU1vZGVsIHNtID0gcmFzdGVyLmdldFNhbXBsZU1vZGVsKCk7Ci0gICAgICAgIFNhbXBsZU1vZGVsIG5zbSA9IHNtLmNyZWF0ZUNvbXBhdGlibGVTYW1wbGVNb2RlbCh3LCBoKTsKLSAgICAgICAgV3JpdGFibGVSYXN0ZXIgb3V0ciA9IFJhc3Rlci5jcmVhdGVXcml0YWJsZVJhc3Rlcihuc20sIHJlY3QuZ2V0TG9jYXRpb24oKSk7Ci0gICAgICAgIE9iamVjdCBkYXRhID0gbnVsbDsKLQotICAgICAgICBkYXRhID0gcmFzdGVyLmdldERhdGFFbGVtZW50cyhtaW5YLCBtaW5ZLCB3LCBoLCBkYXRhKTsKLSAgICAgICAgb3V0ci5zZXREYXRhRWxlbWVudHMobWluWCwgbWluWSwgdywgaCwgZGF0YSk7Ci0gICAgICAgIHJldHVybiBvdXRyOwotICAgIH0KLQotICAgIHB1YmxpYyBWZWN0b3I8UmVuZGVyZWRJbWFnZT4gZ2V0U291cmNlcygpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZ1tdIGdldFByb3BlcnR5TmFtZXMoKSB7Ci0gICAgICAgIGlmIChwcm9wZXJ0aWVzID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgICAgIFZlY3RvcjxTdHJpbmc+IHYgPSBuZXcgVmVjdG9yPFN0cmluZz4oKTsKLSAgICAgICAgZm9yIChFbnVtZXJhdGlvbjw/PiBlID0gcHJvcGVydGllcy5rZXlzKCk7IGUuaGFzTW9yZUVsZW1lbnRzKCk7KSB7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIHYuYWRkKChTdHJpbmcpZS5uZXh0RWxlbWVudCgpKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKENsYXNzQ2FzdEV4Y2VwdGlvbiBleCkgewotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGludCBzaXplID0gdi5zaXplKCk7Ci0gICAgICAgIGlmIChzaXplID4gMCkgewotICAgICAgICAgICAgU3RyaW5nIG5hbWVzW10gPSBuZXcgU3RyaW5nW3NpemVdOwotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKLSAgICAgICAgICAgICAgICBuYW1lc1tpXSA9IHYuZWxlbWVudEF0KGkpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIG5hbWVzOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGlzIEJ1ZmZlcmVkSW1hZ2Ugb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGlzIEJ1ZmZlcmVkSW1hZ2Ugb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgIHJldHVybiAiQnVmZmVyZWRJbWFnZUAiICsgSW50ZWdlci50b0hleFN0cmluZyhoYXNoQ29kZSgpKSArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAiOiB0eXBlID0gIiArIGltYWdlVHlwZSArICIgIiArIGNtICsgIiAiICsgcmFzdGVyOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgIH0KLQotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBnZXRXcml0YWJsZVRpbGUoaW50IHRpbGVYLCBpbnQgdGlsZVkpIHsKLSAgICAgICAgcmV0dXJuIHJhc3RlcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBXcml0YWJsZVJhc3RlciBvZiB0aGlzIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgV3JpdGFibGVSYXN0ZXIgb2YgdGhpcyBCdWZmZXJlZEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBnZXRSYXN0ZXIoKSB7Ci0gICAgICAgIHJldHVybiByYXN0ZXI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIFdyaXRhYmxlUmFzdGVyIG9iamVjdCB3aGljaCBjb250YWlucyB0aGUgYWxwaGEgY2hhbm5lbCBvZgotICAgICAqIEJ1ZmZlcmVkSW1hZ2Ugb2JqZWN0IHdpdGggQ29sb3JNb2RlbCBvYmplY3RzIHRoYXQgc3VwcG9ydHMgYSBzZXBhcmF0ZQotICAgICAqIGFscGhhIGNoYW5uZWwgc3VjaCBhcyBDb21wb25lbnRDb2xvck1vZGVsIG9yIERpcmVjdENvbG9yTW9kZWwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgV3JpdGFibGVSYXN0ZXIgb2JqZWN0IHdoaWNoIGNvbnRhaW5zIHRoZSBhbHBoYSBjaGFubmVsIG9mCi0gICAgICogICAgICAgICB0aGlzIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIFdyaXRhYmxlUmFzdGVyIGdldEFscGhhUmFzdGVyKCkgewotICAgICAgICByZXR1cm4gY20uZ2V0QWxwaGFSYXN0ZXIocmFzdGVyKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVUaWxlT2JzZXJ2ZXIoVGlsZU9ic2VydmVyIHRvKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgYWRkVGlsZU9ic2VydmVyKFRpbGVPYnNlcnZlciB0bykgewotICAgIH0KLQotICAgIHB1YmxpYyBTYW1wbGVNb2RlbCBnZXRTYW1wbGVNb2RlbCgpIHsKLSAgICAgICAgcmV0dXJuIHJhc3Rlci5nZXRTYW1wbGVNb2RlbCgpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldERhdGEoUmFzdGVyIHIpIHsKLQotICAgICAgICBSZWN0YW5nbGUgZnJvbSA9IHIuZ2V0Qm91bmRzKCk7Ci0gICAgICAgIFJlY3RhbmdsZSB0byA9IHJhc3Rlci5nZXRCb3VuZHMoKTsKLSAgICAgICAgUmVjdGFuZ2xlIGludGVyc2VjdGlvbiA9IHRvLmludGVyc2VjdGlvbihmcm9tKTsKLQotICAgICAgICBpbnQgbWluWCA9IGludGVyc2VjdGlvbi54OwotICAgICAgICBpbnQgbWluWSA9IGludGVyc2VjdGlvbi55OwotICAgICAgICBpbnQgdyA9IGludGVyc2VjdGlvbi53aWR0aDsKLSAgICAgICAgaW50IGggPSBpbnRlcnNlY3Rpb24uaGVpZ2h0OwotCi0gICAgICAgIE9iamVjdCBkYXRhID0gbnVsbDsKLQotICAgICAgICBkYXRhID0gci5nZXREYXRhRWxlbWVudHMobWluWCwgbWluWSwgdywgaCwgZGF0YSk7Ci0gICAgICAgIHJhc3Rlci5zZXREYXRhRWxlbWVudHMobWluWCwgbWluWSwgdywgaCwgZGF0YSk7Ci0gICAgfQotCi0gICAgcHVibGljIFJhc3RlciBnZXRUaWxlKGludCB0aWxlWCwgaW50IHRpbGVZKSB7Ci0gICAgICAgIGlmICh0aWxlWCA9PSAwICYmIHRpbGVZID09IDApIHsKLSAgICAgICAgICAgIHJldHVybiByYXN0ZXI7Ci0gICAgICAgIH0KLSAgICAgICAgLy8gYXd0LjIyNj1Cb3RoIHRpbGVYIGFuZCB0aWxlWSBhcmUgbm90IGVxdWFsIHRvIDAKLSAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMjYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICBwdWJsaWMgUmFzdGVyIGdldERhdGEoKSB7Ci0gICAgICAgIGludCB3ID0gcmFzdGVyLmdldFdpZHRoKCk7Ci0gICAgICAgIGludCBoID0gcmFzdGVyLmdldEhlaWdodCgpOwotICAgICAgICBpbnQgbWluWCA9IHJhc3Rlci5nZXRNaW5YKCk7Ci0gICAgICAgIGludCBtaW5ZID0gcmFzdGVyLmdldE1pblkoKTsKLQotICAgICAgICBXcml0YWJsZVJhc3RlciBvdXRyID0gUmFzdGVyLmNyZWF0ZVdyaXRhYmxlUmFzdGVyKHJhc3Rlci5nZXRTYW1wbGVNb2RlbCgpLCBuZXcgUG9pbnQocmFzdGVyCi0gICAgICAgICAgICAgICAgLmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWCgpLCByYXN0ZXIuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVZKCkpKTsKLQotICAgICAgICBPYmplY3QgZGF0YSA9IG51bGw7Ci0KLSAgICAgICAgZGF0YSA9IHJhc3Rlci5nZXREYXRhRWxlbWVudHMobWluWCwgbWluWSwgdywgaCwgZGF0YSk7Ci0gICAgICAgIG91dHIuc2V0RGF0YUVsZW1lbnRzKG1pblgsIG1pblksIHcsIGgsIGRhdGEpOwotCi0gICAgICAgIHJldHVybiBvdXRyOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJbWFnZVByb2R1Y2VyIGdldFNvdXJjZSgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBCdWZmZXJlZEltYWdlU291cmNlKHRoaXMsIHByb3BlcnRpZXMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0V2lkdGgoSW1hZ2VPYnNlcnZlciBvYnNlcnZlcikgewotICAgICAgICByZXR1cm4gcmFzdGVyLmdldFdpZHRoKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRIZWlnaHQoSW1hZ2VPYnNlcnZlciBvYnNlcnZlcikgewotICAgICAgICByZXR1cm4gcmFzdGVyLmdldEhlaWdodCgpOwotICAgIH0KLQotICAgIHB1YmxpYyBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoKSB7Ci0gICAgICAgIHJldHVybiBjbTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSByZWN0YW5ndWxhciBhcmVhIG9mIHRoaXMgQnVmZmVyZWRJbWFnZSBhcyBhIHN1YmltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeSBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHN1YmltYWdlLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBzdWJpbWFnZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBCdWZmZXJlZEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlIGdldFN1YmltYWdlKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoKSB7Ci0gICAgICAgIFdyaXRhYmxlUmFzdGVyIHdyID0gcmFzdGVyLmNyZWF0ZVdyaXRhYmxlQ2hpbGQoeCwgeSwgdywgaCwgMCwgMCwgbnVsbCk7Ci0gICAgICAgIHJldHVybiBuZXcgQnVmZmVyZWRJbWFnZShjbSwgd3IsIGNtLmlzQWxwaGFQcmVtdWx0aXBsaWVkKCksIHByb3BlcnRpZXMpOwotICAgIH0KLQotICAgIHB1YmxpYyBQb2ludFtdIGdldFdyaXRhYmxlVGlsZUluZGljZXMoKSB7Ci0gICAgICAgIFBvaW50IHBvaW50c1tdID0gbmV3IFBvaW50WzFdOwotICAgICAgICBwb2ludHNbMF0gPSBuZXcgUG9pbnQoMCwgMCk7Ci0gICAgICAgIHJldHVybiBwb2ludHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgR3JhcGhpY3MyRCBvYmplY3Qgd2hpY2ggYWxsb3dzIHRvIGRyYXcgaW50byB0aGlzCi0gICAgICogQnVmZmVyZWRJbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBncmFwaGljczJEIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgR3JhcGhpY3MyRCBjcmVhdGVHcmFwaGljcygpIHsKLSAgICAgICAgR3JhcGhpY3NFbnZpcm9ubWVudCBnZSA9IEdyYXBoaWNzRW52aXJvbm1lbnQuZ2V0TG9jYWxHcmFwaGljc0Vudmlyb25tZW50KCk7Ci0gICAgICAgIC8vIHJldHVybiBnZS5jcmVhdGVHcmFwaGljcyh0aGlzKTsKLSAgICAgICAgLy8gPz8/QVdUIGhhY2ssIEZJWE1FCi0gICAgICAgIC8vIHJldHVybiBBbmRyb2lkR3JhcGhpY3MyRC5nZXRJbnN0YW5jZSgpOwotICAgICAgICAvLyB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR3JhcGhpY3MgZ2V0R3JhcGhpY3MoKSB7Ci0gICAgICAgIHJldHVybiBjcmVhdGVHcmFwaGljcygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvZXJjZXMgdGhlIGRhdGEgdG8gYWNoaWV2ZSB0aGUgc3RhdGUgd2hpY2ggaXMgc3BlY2lmaWVkIGJ5IHRoZQotICAgICAqIGlzQWxwaGFQcmVtdWx0aXBsaWVkIHZhcmlhYmxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpc0FscGhhUHJlbXVsdGlwbGllZAotICAgICAqICAgICAgICAgICAgdGhlIGlzIGFscGhhIHByZS1tdWx0aXBsaWVkIHN0YXRlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGNvZXJjZURhdGEoYm9vbGVhbiBpc0FscGhhUHJlbXVsdGlwbGllZCkgewotICAgICAgICBpZiAoY20uaGFzQWxwaGEoKSAmJiBjbS5pc0FscGhhUHJlbXVsdGlwbGllZCgpICE9IGlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0gICAgICAgICAgICBjbSA9IGNtLmNvZXJjZURhdGEocmFzdGVyLCBpc0FscGhhUHJlbXVsdGlwbGllZCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIGNvbG9ycyBpbiB0aGUgVFlQRV9JTlRfQVJHQiBjb2xvciBtb2RlbCBhbmQgZGVmYXVsdCBzUkdCCi0gICAgICogY29sb3Igc3BhY2Ugb2YgdGhlIHNwZWNpZmllZCBhcmVhIG9mIHRoaXMgQnVmZmVyZWRJbWFnZS4gVGhlIHJlc3VsdCBhcnJheQotICAgICAqIGlzIGNvbXBvc2VkIGJ5IHRoZSBmb2xsb3dpbmcgYWxnb3JpdGhtOgotICAgICAqIDxwPgotICAgICAqIHBpeGVsID0gcmdiQXJyYXlbb2Zmc2V0ICsgKHktc3RhcnRZKSpzY2Fuc2l6ZSArICh4LXN0YXJ0WCldCi0gICAgICogPC9wPgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdGFydFgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdGFydCBYIGFyZWEgY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0gc3RhcnRZCi0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgWSBhcmVhIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgYXJlYS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgYXJlYS4KLSAgICAgKiBAcGFyYW0gcmdiQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSByZXN1bHQgYXJyYXkgd2lsbCBiZSBzdG9yZWQgdG8gdGhpcyBhcnJheS4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IG9mIHRoZSByZ2JBcnJheSBhcnJheS4KLSAgICAgKiBAcGFyYW0gc2NhbnNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FubGluZSBzdHJpZGUgZm9yIHRoZSByZ2JBcnJheS4KLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIGNvbG9ycyBmb3IgdGhlIHNwZWNpZmllZCBhcmVhLgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXRSR0IoaW50IHN0YXJ0WCwgaW50IHN0YXJ0WSwgaW50IHcsIGludCBoLCBpbnRbXSByZ2JBcnJheSwgaW50IG9mZnNldCwKLSAgICAgICAgICAgIGludCBzY2Fuc2l6ZSkgewotICAgICAgICBpZiAocmdiQXJyYXkgPT0gbnVsbCkgewotICAgICAgICAgICAgcmdiQXJyYXkgPSBuZXcgaW50W29mZnNldCArIGggKiBzY2Fuc2l6ZV07Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgb2ZmID0gb2Zmc2V0OwotICAgICAgICBmb3IgKGludCB5ID0gc3RhcnRZOyB5IDwgc3RhcnRZICsgaDsgeSsrLCBvZmYgKz0gc2NhbnNpemUpIHsKLSAgICAgICAgICAgIGludCBpID0gb2ZmOwotICAgICAgICAgICAgZm9yIChpbnQgeCA9IHN0YXJ0WDsgeCA8IHN0YXJ0WCArIHc7IHgrKywgaSsrKSB7Ci0gICAgICAgICAgICAgICAgcmdiQXJyYXlbaV0gPSBjbS5nZXRSR0IocmFzdGVyLmdldERhdGFFbGVtZW50cyh4LCB5LCBudWxsKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJnYkFycmF5OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgUkdCIHZhbHVlcyBmcm9tIHRoZSBzcGVjaWZpZWQgYXJyYXkgdG8gdGhlIHNwZWNpZmllZCBCdWZmZXJlZEltYWdlCi0gICAgICogYXJlYS4gVGhlIHBpeGVscyBhcmUgaW4gdGhlIGRlZmF1bHQgUkdCIGNvbG9yIG1vZGVsIChUWVBFX0lOVF9BUkdCKSBhbmQKLSAgICAgKiBkZWZhdWx0IHNSR0IgY29sb3Igc3BhY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0YXJ0WAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IFggY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0gc3RhcnRZCi0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgWSBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIEJ1ZmZlcmVkSW1hZ2UgYXJlYS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgQnVmZmVyZWRJbWFnZSBhcmVhLgotICAgICAqIEBwYXJhbSByZ2JBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIFJHQiB2YWx1ZXMuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBvZiB0aGUgcmdiQXJyYXkgYXJyYXkuCi0gICAgICogQHBhcmFtIHNjYW5zaXplCi0gICAgICogICAgICAgICAgICB0aGUgc2NhbmxpbmUgc3RyaWRlIGZvciB0aGUgcmdiQXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0UkdCKGludCBzdGFydFgsIGludCBzdGFydFksIGludCB3LCBpbnQgaCwgaW50W10gcmdiQXJyYXksIGludCBvZmZzZXQsCi0gICAgICAgICAgICBpbnQgc2NhbnNpemUpIHsKLSAgICAgICAgaW50IG9mZiA9IG9mZnNldDsKLSAgICAgICAgZm9yIChpbnQgeSA9IHN0YXJ0WTsgeSA8IHN0YXJ0WSArIGg7IHkrKywgb2ZmICs9IHNjYW5zaXplKSB7Ci0gICAgICAgICAgICBpbnQgaSA9IG9mZjsKLSAgICAgICAgICAgIGZvciAoaW50IHggPSBzdGFydFg7IHggPCBzdGFydFggKyB3OyB4KyssIGkrKykgewotICAgICAgICAgICAgICAgIHJhc3Rlci5zZXREYXRhRWxlbWVudHMoeCwgeSwgY20uZ2V0RGF0YUVsZW1lbnRzKHJnYkFycmF5W2ldLCBudWxsKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIGEgdGhlIHNwZWNpZmllZCBSR0IgdmFsdWUgdG8gdGhlIHNwZWNpZmllZCBwaXhlbCBvZiB0aGlzCi0gICAgICogQnVmZmVyZWRJbWFnZS4gVGhlIHBpeGVsIHNob3VsZCBiZSBpbiB0aGUgZGVmYXVsdCBSR0IgY29sb3IgbW9kZWwKLSAgICAgKiAoVFlQRV9JTlRfQVJHQikgYW5kIGRlZmF1bHQgc1JHQiBjb2xvciBzcGFjZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHBpeGVsLgotICAgICAqIEBwYXJhbSByZ2IKLSAgICAgKiAgICAgICAgICAgIHRoZSBSR0IgdmFsdWUgdG8gYmUgc2V0LgotICAgICAqLwotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCBzZXRSR0IoaW50IHgsIGludCB5LCBpbnQgcmdiKSB7Ci0gICAgICAgIHJhc3Rlci5zZXREYXRhRWxlbWVudHMoeCwgeSwgY20uZ2V0RGF0YUVsZW1lbnRzKHJnYiwgbnVsbCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGlzVGlsZVdyaXRhYmxlKGludCB0aWxlWCwgaW50IHRpbGVZKSB7Ci0gICAgICAgIGlmICh0aWxlWCA9PSAwICYmIHRpbGVZID09IDApIHsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgICAgIC8vIGF3dC4yMjY9Qm90aCB0aWxlWCBhbmQgdGlsZVkgYXJlIG5vdCBlcXVhbCB0byAwCi0gICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjI2IikpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgcHVibGljIHZvaWQgcmVsZWFzZVdyaXRhYmxlVGlsZShpbnQgdGlsZVgsIGludCB0aWxlWSkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBjb2xvciBpbiB0aGUgVFlQRV9JTlRfQVJHQiBjb2xvciBtb2RlbCBhbmQgZGVmYXVsdCBzUkdCIGNvbG9yCi0gICAgICogc3BhY2Ugb2YgdGhlIHNwZWNpZmllZCBwaXhlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHBpeGVsLgotICAgICAqIEByZXR1cm4gdGhlIGNvbG9yIG9mIHRoZSBzcGVjaWZpZWQgcGl4ZWwgaW4gdGhlIFRZUEVfSU5UX0FSR0IgY29sb3IgbW9kZWwKLSAgICAgKiAgICAgICAgIGFuZCBkZWZhdWx0IHNSR0IgY29sb3Igc3BhY2UuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRSR0IoaW50IHgsIGludCB5KSB7Ci0gICAgICAgIHJldHVybiBjbS5nZXRSR0IocmFzdGVyLmdldERhdGFFbGVtZW50cyh4LCB5LCBudWxsKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIGFscGhhIGlzIHByZS1tdWx0aXBsaWVkLCBmYWxzZSBpZiBhbHBoYSBpcyBub3QKLSAgICAgKiBwcmUtbXVsdGlwbGllZCBvciB0aGVyZSBpcyBubyBhbHBoYS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgYWxwaGEgaXMgcHJlLW11bHRpcGxpZWQsIGZhbHNlIGlmIGFscGhhIGlzIG5vdAotICAgICAqICAgICAgICAgcHJlLW11bHRpcGxpZWQgb3IgdGhlcmUgaXMgbm8gYWxwaGEuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNBbHBoYVByZW11bHRpcGxpZWQoKSB7Ci0gICAgICAgIHJldHVybiBjbS5pc0FscGhhUHJlbXVsdGlwbGllZCgpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGhhc1RpbGVXcml0ZXJzKCkgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmbHVzaCgpIHsKLSAgICAgICAgaW1hZ2VTdXJmLmRpc3Bvc2UoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFdpZHRoKCkgewotICAgICAgICByZXR1cm4gcmFzdGVyLmdldFdpZHRoKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW1hZ2UgdHlwZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBpbWFnZSB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIGltYWdlVHlwZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFRpbGVXaWR0aCgpIHsKLSAgICAgICAgcmV0dXJuIHJhc3Rlci5nZXRXaWR0aCgpOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0VGlsZUhlaWdodCgpIHsKLSAgICAgICAgcmV0dXJuIHJhc3Rlci5nZXRIZWlnaHQoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFRpbGVHcmlkWU9mZnNldCgpIHsKLSAgICAgICAgcmV0dXJuIHJhc3Rlci5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFRpbGVHcmlkWE9mZnNldCgpIHsKLSAgICAgICAgcmV0dXJuIHJhc3Rlci5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldE51bVlUaWxlcygpIHsKLSAgICAgICAgcmV0dXJuIDE7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXROdW1YVGlsZXMoKSB7Ci0gICAgICAgIHJldHVybiAxOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0TWluWSgpIHsKLSAgICAgICAgcmV0dXJuIHJhc3Rlci5nZXRNaW5ZKCk7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRNaW5YKCkgewotICAgICAgICByZXR1cm4gcmFzdGVyLmdldE1pblgoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldE1pblRpbGVZKCkgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldE1pblRpbGVYKCkgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldEhlaWdodCgpIHsKLSAgICAgICAgcmV0dXJuIHJhc3Rlci5nZXRIZWlnaHQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBpbWFnZSBzdXJmYWNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0eXBlCi0gICAgICogICAgICAgICAgICB0aGUgdHlwZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbWFnZSBzdXJmYWNlLgotICAgICAqLwotICAgIHByaXZhdGUgSW1hZ2VTdXJmYWNlIGNyZWF0ZUltYWdlU3VyZmFjZShpbnQgdHlwZSkgewotICAgICAgICByZXR1cm4gbmV3IEltYWdlU3VyZmFjZShnZXRDb2xvck1vZGVsKCksIGdldFJhc3RlcigpLCB0eXBlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBpbWFnZSBzdXJmYWNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGltYWdlIHN1cmZhY2UuCi0gICAgICovCi0gICAgSW1hZ2VTdXJmYWNlIGdldEltYWdlU3VyZmFjZSgpIHsKLSAgICAgICAgcmV0dXJuIGltYWdlU3VyZjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFRyYW5zcGFyZW5jeSgpIHsKLSAgICAgICAgcmV0dXJuIGNtLmdldFRyYW5zcGFyZW5jeSgpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9CdWZmZXJlZEltYWdlRmlsdGVyLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvQnVmZmVyZWRJbWFnZUZpbHRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4YjZmY2YwLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9CdWZmZXJlZEltYWdlRmlsdGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzOTcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lCi0gKiAgb3IgbW9yZSBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlCi0gKiAgZGlzdHJpYnV0ZWQgd2l0aCB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24KLSAqICByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4gIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlCi0gKiAgdG8geW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZQotICogICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2UKLSAqICB3aXRoIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywKLSAqICBzb2Z0d2FyZSBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbgotICogICJBUyBJUyIgQkFTSVMsIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWQotICogIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZQotICogIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgbGltaXRhdGlvbnMKLSAqICB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5Bd3RJbWFnZUJhY2tkb29yQWNjZXNzb3I7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIEJ1ZmZlcmVkSW1hZ2VGaWx0ZXIgY2xhc3MgcHJvdmlkZXMgZmlsdGVyaW5nIG9wZXJhdGlvbnMgdG8gdGhlCi0gKiBCdWZmZXJlZEltYWdlIG9iamVjdHMgdXNpbmcgb3BlcmF0b3JzIHdoaWNoIGltcGxlbWVudCBCdWZmZXJlZEltYWdlT3AKLSAqIGludGVyZmFjZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBCdWZmZXJlZEltYWdlRmlsdGVyIGV4dGVuZHMgSW1hZ2VGaWx0ZXIgaW1wbGVtZW50cyBDbG9uZWFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IGFjY2Vzc29yLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIEF3dEltYWdlQmFja2Rvb3JBY2Nlc3NvciBhY2Nlc3NvciA9IEF3dEltYWdlQmFja2Rvb3JBY2Nlc3Nvci5nZXRJbnN0YW5jZSgpOwotCi0gICAgLyoqCi0gICAgICogVGhlIG9wLgotICAgICAqLwotICAgIHByaXZhdGUgQnVmZmVyZWRJbWFnZU9wIG9wOwotCi0gICAgLyoqCi0gICAgICogVGhlIHJhc3Rlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIFdyaXRhYmxlUmFzdGVyIHJhc3RlcjsKLQotICAgIC8qKgotICAgICAqIFRoZSBpIGRhdGEuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgaURhdGFbXTsKLQotICAgIC8qKgotICAgICAqIFRoZSBiIGRhdGEuCi0gICAgICovCi0gICAgcHJpdmF0ZSBieXRlIGJEYXRhW107Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgd2lkdGguCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgd2lkdGg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGVpZ2h0LgotICAgICAqLwotICAgIHByaXZhdGUgaW50IGhlaWdodDsKLQotICAgIC8qKgotICAgICAqIFRoZSBjbS4KLSAgICAgKi8KLSAgICBwcml2YXRlIENvbG9yTW9kZWwgY207Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZm9yY2VkIHJnYi4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gZm9yY2VkUkdCID0gZmFsc2U7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdHJhbnNmZXIgdHlwZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCB0cmFuc2ZlclR5cGUgPSBEYXRhQnVmZmVyLlRZUEVfVU5ERUZJTkVEOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEJ1ZmZlcmVkSW1hZ2VGaWx0ZXIgd2l0aCB0aGUgc3BlY2lmaWVkIEJ1ZmZlcmVkSW1hZ2VPcAotICAgICAqIG9wZXJhdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvcAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBCdWZmZXJlZEltYWdlT3Agb3BlcmF0b3IuCi0gICAgICogQHRocm93cyBOdWxsUG9pbnRlckV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIEJ1ZmZlcmVkSW1hZ2VPcCBpcyBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlRmlsdGVyKEJ1ZmZlcmVkSW1hZ2VPcCBvcCkgewotICAgICAgICBpZiAob3AgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjA1IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy5vcCA9IG9wOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEJ1ZmZlcmVkSW1hZ2VPcCBvcGVyYXRvciBhc3NvY2lhdGVkIHdpdGggdGhpcwotICAgICAqIEJ1ZmZlcmVkSW1hZ2VGaWx0ZXIgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEJ1ZmZlcmVkSW1hZ2VPcCBhc3NvY2lhdGVkIHdpdGggdGhpcyBCdWZmZXJlZEltYWdlRmlsdGVyCi0gICAgICogICAgICAgICBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2VPcCBnZXRCdWZmZXJlZEltYWdlT3AoKSB7Ci0gICAgICAgIHJldHVybiBvcDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXREaW1lbnNpb25zKGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICB0aGlzLndpZHRoID0gd2lkdGg7Ci0gICAgICAgIHRoaXMuaGVpZ2h0ID0gaGVpZ2h0OwotICAgICAgICAvLyBTdG9wIGltYWdlIGNvbnN1bWluZyBpZiBubyBwaXhlbHMgZXhwZWN0ZWQuCi0gICAgICAgIGlmICh3aWR0aCA8PSAwIHx8IGhlaWdodCA8PSAwKSB7Ci0gICAgICAgICAgICBjb25zdW1lci5pbWFnZUNvbXBsZXRlKEltYWdlQ29uc3VtZXIuU1RBVElDSU1BR0VET05FKTsKLSAgICAgICAgICAgIHJlc2V0KCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRDb2xvck1vZGVsKENvbG9yTW9kZWwgbW9kZWwpIHsKLSAgICAgICAgaWYgKHRoaXMuY20gIT0gbnVsbCAmJiB0aGlzLmNtICE9IG1vZGVsICYmIHJhc3RlciAhPSBudWxsKSB7Ci0gICAgICAgICAgICBmb3JjZVJHQigpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdGhpcy5jbSA9IG1vZGVsOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBDb2xvck1vZGVsIG1vZGVsLCBieXRlW10gcGl4ZWxzLCBpbnQgb2ZmLAotICAgICAgICAgICAgaW50IHNjYW5zaXplKSB7Ci0gICAgICAgIHNldFBpeGVscyh4LCB5LCB3LCBoLCBtb2RlbCwgcGl4ZWxzLCBvZmYsIHNjYW5zaXplLCB0cnVlKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIENvbG9yTW9kZWwgbW9kZWwsIGludFtdIHBpeGVscywgaW50IG9mZiwKLSAgICAgICAgICAgIGludCBzY2Fuc2l6ZSkgewotICAgICAgICBzZXRQaXhlbHMoeCwgeSwgdywgaCwgbW9kZWwsIHBpeGVscywgb2ZmLCBzY2Fuc2l6ZSwgZmFsc2UpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGltYWdlQ29tcGxldGUoaW50IHN0YXR1cykgewotICAgICAgICBpZiAoc3RhdHVzID09IFNUQVRJQ0lNQUdFRE9ORSB8fCBzdGF0dXMgPT0gU0lOR0xFRlJBTUVET05FKSB7Ci0gICAgICAgICAgICBCdWZmZXJlZEltYWdlIGJpbSA9IG5ldyBCdWZmZXJlZEltYWdlKGNtLCByYXN0ZXIsIGNtLmlzQWxwaGFQcmVtdWx0aXBsaWVkLCBudWxsKTsKLSAgICAgICAgICAgIGJpbSA9IG9wLmZpbHRlcihiaW0sIG51bGwpOwotICAgICAgICAgICAgRGF0YUJ1ZmZlciBkc3REYiA9IGJpbS5nZXRSYXN0ZXIoKS5nZXREYXRhQnVmZmVyKCk7Ci0gICAgICAgICAgICBDb2xvck1vZGVsIGRzdENtID0gYmltLmdldENvbG9yTW9kZWwoKTsKLSAgICAgICAgICAgIGludCBkc3RXID0gYmltLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgZHN0SCA9IGJpbS5nZXRIZWlnaHQoKTsKLQotICAgICAgICAgICAgY29uc3VtZXIuc2V0RGltZW5zaW9ucyhkc3RXLCBkc3RIKTsKLQotICAgICAgICAgICAgaWYgKGRzdERiLmdldERhdGFUeXBlKCkgPT0gRGF0YUJ1ZmZlci5UWVBFX0lOVCkgewotICAgICAgICAgICAgICAgIGNvbnN1bWVyLnNldENvbG9yTW9kZWwoZHN0Q20pOwotICAgICAgICAgICAgICAgIGNvbnN1bWVyLnNldFBpeGVscygwLCAwLCBkc3RXLCBkc3RILCBkc3RDbSwgYWNjZXNzb3IuZ2V0RGF0YUludChkc3REYiksIDAsIGRzdFcpOwotICAgICAgICAgICAgfSBlbHNlIGlmIChkc3REYi5nZXREYXRhVHlwZSgpID09IERhdGFCdWZmZXIuVFlQRV9CWVRFKSB7Ci0gICAgICAgICAgICAgICAgY29uc3VtZXIuc2V0Q29sb3JNb2RlbChkc3RDbSk7Ci0gICAgICAgICAgICAgICAgY29uc3VtZXIuc2V0UGl4ZWxzKDAsIDAsIGRzdFcsIGRzdEgsIGRzdENtLCBhY2Nlc3Nvci5nZXREYXRhQnl0ZShkc3REYiksIDAsIGRzdFcpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBpbnQgZHN0RGF0YVtdID0gYmltLmdldFJHQigwLCAwLCBkc3RXLCBkc3RILCBudWxsLCAwLCBkc3RXKTsKLSAgICAgICAgICAgICAgICBkc3RDbSA9IENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpOwotICAgICAgICAgICAgICAgIGNvbnN1bWVyLnNldENvbG9yTW9kZWwoZHN0Q20pOwotICAgICAgICAgICAgICAgIGNvbnN1bWVyLnNldFBpeGVscygwLCAwLCBkc3RXLCBkc3RILCBkc3RDbSwgZHN0RGF0YSwgMCwgZHN0Vyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSBpZiAoc3RhdHVzID09IElNQUdFRVJST1IgfHwgc3RhdHVzID09IElNQUdFQUJPUlRFRCkgewotICAgICAgICAgICAgcmVzZXQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGNvbnN1bWVyLmltYWdlQ29tcGxldGUoc3RhdHVzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBwaXhlbHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSB4LgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgeS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHcuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoLgotICAgICAqIEBwYXJhbSBtb2RlbAotICAgICAqICAgICAgICAgICAgdGhlIG1vZGVsLgotICAgICAqIEBwYXJhbSBwaXhlbHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbHMuCi0gICAgICogQHBhcmFtIG9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZi4KLSAgICAgKiBAcGFyYW0gc2NhbnNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2Fuc2l6ZS4KLSAgICAgKiBAcGFyYW0gaXNCeXRlRGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGlzIGJ5dGUgZGF0YS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBDb2xvck1vZGVsIG1vZGVsLCBPYmplY3QgcGl4ZWxzLCBpbnQgb2ZmLAotICAgICAgICAgICAgaW50IHNjYW5zaXplLCBib29sZWFuIGlzQnl0ZURhdGEpIHsKLSAgICAgICAgLy8gQ2hlY2sgYm91bmRzCi0gICAgICAgIC8vIE5lZWQgdG8gY29weSBvbmx5IHRoZSBwaXhlbHMgdGhhdCB3aWxsIGZpdCBpbnRvIHRoZSBkZXN0aW5hdGlvbiBhcmVhCi0gICAgICAgIGlmICh4IDwgMCkgewotICAgICAgICAgICAgdyAtPSB4OwotICAgICAgICAgICAgb2ZmICs9IHg7Ci0gICAgICAgICAgICB4ID0gMDsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICh5IDwgMCkgewotICAgICAgICAgICAgaCAtPSB5OwotICAgICAgICAgICAgb2ZmICs9IHkgKiBzY2Fuc2l6ZTsKLSAgICAgICAgICAgIHkgPSAwOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHggKyB3ID4gd2lkdGgpIHsKLSAgICAgICAgICAgIHcgPSB3aWR0aCAtIHg7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoeSArIGggPiBoZWlnaHQpIHsKLSAgICAgICAgICAgIGggPSBoZWlnaHQgLSB5OwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHcgPD0gMCB8fCBoIDw9IDApIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIENoZWNrIG1vZGVsCi0gICAgICAgIGlmICh0aGlzLmNtID09IG51bGwpIHsKLSAgICAgICAgICAgIHNldENvbG9yTW9kZWwobW9kZWwpOwotICAgICAgICB9IGVsc2UgaWYgKG1vZGVsID09IG51bGwpIHsKLSAgICAgICAgICAgIG1vZGVsID0gdGhpcy5jbTsKLSAgICAgICAgfSBlbHNlIGlmICghbW9kZWwuZXF1YWxzKHRoaXMuY20pKSB7Ci0gICAgICAgICAgICBmb3JjZVJHQigpOwotICAgICAgICB9Ci0KLSAgICAgICAgYm9vbGVhbiBjYW5BcnJheWNvcHk7Ci0gICAgICAgIC8vIFByb2Nlc3MgcGl4ZWxzCi0gICAgICAgIHN3aXRjaCAodHJhbnNmZXJUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VTkRFRklORUQ6IHsKLSAgICAgICAgICAgICAgICBpZiAoaXNCeXRlRGF0YSkgewotICAgICAgICAgICAgICAgICAgICB0cmFuc2ZlclR5cGUgPSBEYXRhQnVmZmVyLlRZUEVfQllURTsKLSAgICAgICAgICAgICAgICAgICAgY3JlYXRlUmFzdGVyKHRyYW5zZmVyVHlwZSk7Ci0gICAgICAgICAgICAgICAgICAgIC8vIGJEYXRhID0gbmV3IGJ5dGVbd2lkdGgqaGVpZ2h0XTsKLSAgICAgICAgICAgICAgICAgICAgY2FuQXJyYXljb3B5ID0gIWZvcmNlZFJHQjsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHRyYW5zZmVyVHlwZSA9IERhdGFCdWZmZXIuVFlQRV9JTlQ7Ci0gICAgICAgICAgICAgICAgY3JlYXRlUmFzdGVyKHRyYW5zZmVyVHlwZSk7Ci0gICAgICAgICAgICAgICAgLy8gaURhdGEgPSBuZXcgaW50W3dpZHRoKmhlaWdodF07Ci0gICAgICAgICAgICAgICAgY2FuQXJyYXljb3B5ID0gIWZvcmNlZFJHQiB8fCBtb2RlbC5lcXVhbHMoQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCkpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfSAvLyBBbmQgcHJvY2VlZCB0byBjb3B5IHRoZSBwaXhlbHMKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDogewotICAgICAgICAgICAgICAgIGlmIChpc0J5dGVEYXRhKSB7IC8vIFRoZXJlIGFyZSBpbnQgZGF0YSBhbHJlYWR5IGJ1dCB0aGUgbmV3IGRhdGEKLSAgICAgICAgICAgICAgICAgICAgLy8gYXJlIGJ5dGVzCi0gICAgICAgICAgICAgICAgICAgIGZvcmNlUkdCKCk7Ci0gICAgICAgICAgICAgICAgICAgIGNhbkFycmF5Y29weSA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKCFmb3JjZWRSR0IgfHwgbW9kZWwuZXF1YWxzKENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpKSkgewotICAgICAgICAgICAgICAgICAgICBjYW5BcnJheWNvcHkgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9IC8vIEVsc2UgZmFsbGJhY2sgdG8gdGhlIFJHQiBjb252ZXJzaW9uCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOiB7Ci0gICAgICAgICAgICAgICAgaWYgKGlzQnl0ZURhdGEgJiYgIWZvcmNlZFJHQikgewotICAgICAgICAgICAgICAgICAgICBjYW5BcnJheWNvcHkgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAvLyBSR0IgY29udmVyc2lvbgotICAgICAgICAgICAgICAgIGNhbkFycmF5Y29weSA9IGZhbHNlOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZGVmYXVsdDogewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMDYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIG9mZiArPSB4OwotICAgICAgICBpbnQgbWF4T2Zmc2V0ID0gb2ZmICsgaCAqIHNjYW5zaXplOwotICAgICAgICBpbnQgZHN0T2Zmc2V0ID0geCArIHkgKiB3aWR0aDsKLQotICAgICAgICBpZiAoY2FuQXJyYXljb3B5KSB7Ci0gICAgICAgICAgICBPYmplY3QgZHN0QXJyYXkgPSBpc0J5dGVEYXRhID8gKE9iamVjdCliRGF0YSA6IChPYmplY3QpaURhdGE7Ci0gICAgICAgICAgICBmb3IgKDsgb2ZmIDwgbWF4T2Zmc2V0OyBvZmYgKz0gc2NhbnNpemUsIGRzdE9mZnNldCArPSB3aWR0aCkgewotICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkocGl4ZWxzLCBvZmYsIGRzdEFycmF5LCBkc3RPZmZzZXQsIHcpOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLy8gUkdCIGNvbnZlcnNpb24KLSAgICAgICAgICAgIGZvciAoOyBvZmYgPCBtYXhPZmZzZXQ7IG9mZiArPSBzY2Fuc2l6ZSwgZHN0T2Zmc2V0ICs9IHdpZHRoKSB7Ci0gICAgICAgICAgICAgICAgaW50IHNyY1BvcyA9IG9mZjsKLSAgICAgICAgICAgICAgICBpbnQgZHN0UG9zID0gZHN0T2Zmc2V0OwotICAgICAgICAgICAgICAgIGludCBtYXhEc3RQb3MgPSBkc3RPZmZzZXQgKyB3OwotICAgICAgICAgICAgICAgIGZvciAoOyBkc3RQb3MgPCBtYXhEc3RQb3M7IGRzdFBvcysrLCBzcmNQb3MrKykgewotICAgICAgICAgICAgICAgICAgICBpRGF0YVtkc3RQb3NdID0gbW9kZWwuZ2V0UkdCKGlzQnl0ZURhdGEgPyAoKGJ5dGVbXSlwaXhlbHMpW3NyY1Bvc10KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICA6ICgoaW50W10pcGl4ZWxzKVtzcmNQb3NdKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGb3JjZSByZ2IuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIGZvcmNlUkdCKCkgewotICAgICAgICBpZiAoIWZvcmNlZFJHQikgewotICAgICAgICAgICAgZm9yY2VkUkdCID0gdHJ1ZTsKLSAgICAgICAgICAgIGludCBzaXplID0gd2lkdGggKiBoZWlnaHQ7Ci0gICAgICAgICAgICBpbnQgcmdiRGF0YVtdID0gbmV3IGludFtzaXplXTsKLQotICAgICAgICAgICAgaWYgKGJEYXRhICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewotICAgICAgICAgICAgICAgICAgICByZ2JEYXRhW2ldID0gY20uZ2V0UkdCKGJEYXRhW2ldKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGlEYXRhICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewotICAgICAgICAgICAgICAgICAgICByZ2JEYXRhW2ldID0gY20uZ2V0UkdCKGlEYXRhW2ldKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNtID0gQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCk7Ci0gICAgICAgICAgICBEYXRhQnVmZmVySW50IGRiID0gbmV3IERhdGFCdWZmZXJJbnQocmdiRGF0YSwgc2l6ZSk7Ci0gICAgICAgICAgICBpbnQgbWFza3NbXSA9IG5ldyBpbnRbXSB7Ci0gICAgICAgICAgICAgICAgICAgIDB4MDBmZjAwMDAsIDB4MDAwMGZmMDAsIDB4MDAwMDAwZmYsIDB4ZmYwMDAwMDAKLSAgICAgICAgICAgIH07Ci0gICAgICAgICAgICByYXN0ZXIgPSBSYXN0ZXIuY3JlYXRlUGFja2VkUmFzdGVyKGRiLCB3aWR0aCwgaGVpZ2h0LCB3aWR0aCwgbWFza3MsIG51bGwpOwotICAgICAgICAgICAgaURhdGEgPSBhY2Nlc3Nvci5nZXREYXRhSW50KGRiKTsKLSAgICAgICAgICAgIGJEYXRhID0gbnVsbDsKLSAgICAgICAgICAgIHRyYW5zZmVyVHlwZSA9IERhdGFCdWZmZXIuVFlQRV9JTlQ7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXNldC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgcmVzZXQoKSB7Ci0gICAgICAgIHdpZHRoID0gMDsKLSAgICAgICAgaGVpZ2h0ID0gMDsKLSAgICAgICAgZm9yY2VkUkdCID0gZmFsc2U7Ci0gICAgICAgIGNtID0gbnVsbDsKLSAgICAgICAgaURhdGEgPSBudWxsOwotICAgICAgICBiRGF0YSA9IG51bGw7Ci0gICAgICAgIHRyYW5zZmVyVHlwZSA9IERhdGFCdWZmZXIuVFlQRV9VTkRFRklORUQ7Ci0gICAgICAgIHJhc3RlciA9IG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgcmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgdHlwZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgY3JlYXRlUmFzdGVyKGludCBkYXRhVHlwZSkgewotICAgICAgICBib29sZWFuIGNyZWF0ZWRWYWxpZEJ1ZmZlciA9IGZhbHNlOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmFzdGVyID0gY20uY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKHdpZHRoLCBoZWlnaHQpOwotICAgICAgICAgICAgaW50IHJhc3RlclR5cGUgPSByYXN0ZXIuZ2V0RGF0YUJ1ZmZlcigpLmdldERhdGFUeXBlKCk7Ci0gICAgICAgICAgICBpZiAocmFzdGVyVHlwZSA9PSBkYXRhVHlwZSkgewotICAgICAgICAgICAgICAgIHN3aXRjaCAocmFzdGVyVHlwZSkgewotICAgICAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6IHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlEYXRhID0gYWNjZXNzb3IuZ2V0RGF0YUludChyYXN0ZXIuZ2V0RGF0YUJ1ZmZlcigpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChpRGF0YSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY3JlYXRlZFZhbGlkQnVmZmVyID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6IHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJEYXRhID0gYWNjZXNzb3IuZ2V0RGF0YUJ5dGUocmFzdGVyLmdldERhdGFCdWZmZXIoKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoYkRhdGEgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZWRWYWxpZEJ1ZmZlciA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgICAgICAgICAgY3JlYXRlZFZhbGlkQnVmZmVyID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgaWYgKGNtID09IENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvcmNlZFJHQiA9IHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBjcmVhdGVkVmFsaWRCdWZmZXIgPSBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIGNyZWF0ZWRWYWxpZEJ1ZmZlciA9IGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGNyZWF0ZWRWYWxpZEJ1ZmZlciA9PSBmYWxzZSkgewotICAgICAgICAgICAgY20gPSBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKTsKLSAgICAgICAgICAgIHJhc3RlciA9IGNtLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3Rlcih3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgICAgIGlEYXRhID0gYWNjZXNzb3IuZ2V0RGF0YUludChyYXN0ZXIuZ2V0RGF0YUJ1ZmZlcigpKTsKLSAgICAgICAgICAgIGJEYXRhID0gbnVsbDsKLSAgICAgICAgICAgIGZvcmNlZFJHQiA9IHRydWU7Ci0gICAgICAgIH0KLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvQnVmZmVyZWRJbWFnZU9wLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvQnVmZmVyZWRJbWFnZU9wLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDg4M2EzOWQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL0J1ZmZlcmVkSW1hZ2VPcC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsOTIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBBbGV4ZXkgQS4gUGV0cmVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgamF2YS5hd3QuUmVuZGVyaW5nSGludHM7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5Qb2ludDJEOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci0KLS8qKgotICogVGhlIEJ1ZmZlcmVkSW1hZ2VPcCBpbnRlcmZhY2UgcHJvdmlkZXMgbWV0aG9kcyBmb3IgcGVyZm9ybWluZyB0cmFuc2Zvcm1hdGlvbnMKLSAqIGZyb20gc291cmNlIGRhdGEgdG8gZGVzdGluYXRpb24gZGF0YSBmb3IgQnVmZmVyZWRJbWFnZSBvYmplY3RzLiBBbiBvYmplY3QKLSAqIGltcGxlbWVudGluZyB0aGlzIGludGVyZmFjZSBjYW4gYmUgcGFzc2VkIGludG8gYSBCdWZmZXJlZEltYWdlRmlsdGVyIHRvCi0gKiBvcGVyYXRlIG9uIGEgQnVmZmVyZWRJbWFnZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgQnVmZmVyZWRJbWFnZU9wIHsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBkZXN0aW5hdGlvbiBpbWFnZSB3aXRoIHRoZSBzcGVjaWZpZWQgQnVmZmVyZWRJbWFnZSBhbmQKLSAgICAgKiBDb2xvck1vZGVsOyB0aGlzIGRlc3RpbmF0aW9uIGltYWdlIGlzIGVtcHR5IGFuZCBoYXMgdGhlIGNvcnJlY3Qgc2l6ZSBhbmQKLSAgICAgKiBudW1iZXIgb2YgYmFuZHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBCdWZmZXJlZEltYWdlLgotICAgICAqIEBwYXJhbSBkZXN0Q00KLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXN0aW5hdGlvbiBDb2xvck1vZGVsLgotICAgICAqIEByZXR1cm4gdGhlIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UgY3JlYXRlQ29tcGF0aWJsZURlc3RJbWFnZShCdWZmZXJlZEltYWdlIHNyYywgQ29sb3JNb2RlbCBkZXN0Q00pOwotCi0gICAgLyoqCi0gICAgICogUGVyZm9ybXMgYSBmaWx0ZXIgb3BlcmF0aW9uIG9uIHRoZSBzb3VyY2UgQnVmZmVyZWRJbWFnZSBhbmQgc3RvcmVzIHRoZQotICAgICAqIHJlc3VsdGluZyBCdWZmZXJlZEltYWdlIHRvIHRoZSBkZXN0aW5hdGlvbiBCdWZmZXJlZEltYWdlLiBJZiB0aGUKLSAgICAgKiBkZXN0aW5hdGlvbiBCdWZmZXJlZEltYWdlIGlzIG51bGwsIGEgQnVmZmVyZWRJbWFnZSB3aXRoIGFuIGFwcHJvcHJpYXRlCi0gICAgICogQ29sb3JNb2RlbCBpcyBjcmVhdGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgQnVmZmVyZWRJbWFnZS4KLSAgICAgKiBAcGFyYW0gZGVzdAotICAgICAqICAgICAgICAgICAgdGhlIGRlc3RpbmF0aW9uIEJ1ZmZlcmVkSW1hZ2UsIHdoZXJlIHRoZSByZXN1bHQgaXMgc3RvcmVkLgotICAgICAqIEByZXR1cm4gdGhlIGZpbHRlcmVkIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UgZmlsdGVyKEJ1ZmZlcmVkSW1hZ2Ugc3JjLCBCdWZmZXJlZEltYWdlIGRlc3QpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYm91bmRzIG9mIGZpbHRlcmVkIGltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgQnVmZmVyZWRJbWFnZSB0byBiZSBmaWx0ZXJlZC4KLSAgICAgKiBAcmV0dXJuIHRoZSByZWN0YW5nbGUgYm91bmRzIG9mIGZpbHRlcmVkIGltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRChCdWZmZXJlZEltYWdlIHNyYyk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBwb2ludCBvZiB0aGUgZGVzdGluYXRpb24gaW1hZ2Ugd2hpY2ggY29ycmVzcG9uZHMgdG8gdGhlCi0gICAgICogc3BlY2lmaWVkIHBvaW50IGluIHRoZSBzb3VyY2UgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyY1B0Ci0gICAgICogICAgICAgICAgICB0aGUgcG9pbnQgb2YgdGhlIHNvdXJjZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gZHN0UHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb2ludCB3aGVyZSB0aGUgcmVzdWx0IHdpbGwgYmUgc3RvcmVkLgotICAgICAqIEByZXR1cm4gdGhlIGRlc3RpbmF0aW9uIHBvaW50LgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludDJEIGdldFBvaW50MkQoUG9pbnQyRCBzcmNQdCwgUG9pbnQyRCBkc3RQdCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBSZW5kZXJpbmdIaW50cyBvZiB0aGUgQnVmZmVyZWRJbWFnZU9wLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFJlbmRlcmluZ0hpbnRzIG9mIHRoZSBCdWZmZXJlZEltYWdlT3AuCi0gICAgICovCi0gICAgcHVibGljIFJlbmRlcmluZ0hpbnRzIGdldFJlbmRlcmluZ0hpbnRzKCk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvQnl0ZUxvb2t1cFRhYmxlLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvQnl0ZUxvb2t1cFRhYmxlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI3Y2VlMzAuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL0J5dGVMb29rdXBUYWJsZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTQwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICoKLSAqIEBkYXRlOiBPY3QgMTQsIDIwMDUKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi0vKioKLSAqIFRoZSBCeXRlTG9va3VwVGFibGUgY2xhc3MgcHJvdmlkZXMgZnVuY3Rpb25hbGl0eSBmb3IgbG9va3VwIG9wZXJhdGlvbnMsIGFuZAotICogaXMgZGVmaW5lZCBieSBhbiBpbnB1dCBieXRlIGFycmF5IGZvciBiYW5kcyBvciBjb21wb25lbnRzIG9mIGltYWdlIGFuZCBhbgotICogb2Zmc2V0IHZhbHVlLiBUaGUgb2Zmc2V0IHZhbHVlIHdpbGwgYmUgc3VidHJhY3RlZCBmcm9tIHRoZSBpbnB1dCB2YWx1ZXMKLSAqIGJlZm9yZSBpbmRleGluZyB0aGUgaW5wdXQgYXJyYXlzLiBUaGUgb3V0cHV0IG9mIGEgbG9va3VwIG9wZXJhdGlvbiBpcwotICogcmVwcmVzZW50ZWQgYXMgYW4gYXJyYXkgb2YgdW5zaWduZWQgYnl0ZXMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgQnl0ZUxvb2t1cFRhYmxlIGV4dGVuZHMgTG9va3VwVGFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIGRhdGEuCi0gICAgICovCi0gICAgcHJpdmF0ZSBieXRlIGRhdGFbXVtdOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEJ5dGVMb29rdXBUYWJsZSB3aXRoIHRoZSBzcGVjaWZpZWQgb2Zmc2V0IHZhbHVlIGFuZAotICAgICAqIHRoZSBzcGVjaWZpZWQgYnl0ZSBhcnJheSB3aGljaCByZXByZXNlbnRzIHRoZSBsb29rdXAgdGFibGUgZm9yIGFsbCBiYW5kcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IHZhbHVlLgotICAgICAqIEBwYXJhbSBkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBhcnJheSBvZiBieXRlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgQnl0ZUxvb2t1cFRhYmxlKGludCBvZmZzZXQsIGJ5dGVbXSBkYXRhKSB7Ci0gICAgICAgIHN1cGVyKG9mZnNldCwgMSk7Ci0gICAgICAgIGlmIChkYXRhLmxlbmd0aCA8IDEpCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJMZW5ndGggb2YgZGF0YSBzaG91bGQgbm90IGJlIGxlc3MgdGhlbiBvbmUiKTsKLSAgICAgICAgdGhpcy5kYXRhID0gbmV3IGJ5dGVbMV1bZGF0YS5sZW5ndGhdOwotICAgICAgICAvLyBUaGUgZGF0YSBhcnJheSBzdG9yZWQgYXMgYSByZWZlcmVuY2UKLSAgICAgICAgdGhpcy5kYXRhWzBdID0gZGF0YTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQnl0ZUxvb2t1cFRhYmxlIHdpdGggdGhlIHNwZWNpZmllZCBvZmZzZXQgdmFsdWUgYW5kCi0gICAgICogdGhlIHNwZWNpZmllZCBieXRlIGFycmF5IG9mIGFycmF5cyB3aGljaCByZXByZXNlbnRzIHRoZSBsb29rdXAgdGFibGUgZm9yCi0gICAgICogZWFjaCBiYW5kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgdmFsdWUuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IG9mIGJ5dGVzIGFycmF5IGZvciBlYWNoIGJhbmQuCi0gICAgICovCi0gICAgcHVibGljIEJ5dGVMb29rdXBUYWJsZShpbnQgb2Zmc2V0LCBieXRlW11bXSBkYXRhKSB7Ci0gICAgICAgIHN1cGVyKG9mZnNldCwgZGF0YS5sZW5ndGgpOwotICAgICAgICB0aGlzLmRhdGEgPSBuZXcgYnl0ZVtkYXRhLmxlbmd0aF1bZGF0YVswXS5sZW5ndGhdOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGRhdGEubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIC8vIFRoZSBkYXRhIGFycmF5IGZvciBlYWNoIGJhbmQgc3RvcmVkIGFzIGEgcmVmZXJlbmNlCi0gICAgICAgICAgICB0aGlzLmRhdGFbaV0gPSBkYXRhW2ldOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9va3VwIHRhYmxlIG9mIHRoaXMgQnl0ZUxvb2t1cFRhYmxlIG9iamVjdC4gSWYgdGhpcwotICAgICAqIEJ5dGVMb29rdXBUYWJsZSBvYmplY3QgaGFzIG9uZSBieXRlIGFycmF5IGZvciBhbGwgYmFuZHMsIHRoZSByZXR1cm5lZAotICAgICAqIGFycmF5IGxlbmd0aCBpcyBvbmUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbG9va3VwIHRhYmxlIG9mIHRoaXMgQnl0ZUxvb2t1cFRhYmxlIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgYnl0ZVtdW10gZ2V0VGFibGUoKSB7Ci0gICAgICAgIC8vIFJldHVybnMgZGF0YSBieSByZWZlcmVuY2UKLSAgICAgICAgcmV0dXJuIGRhdGE7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGxvb2t1cFBpeGVsKGludFtdIHNyYywgaW50W10gZHN0KSB7Ci0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0ID0gbmV3IGludFtzcmMubGVuZ3RoXTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBvZmZzZXQgPSBnZXRPZmZzZXQoKTsKLSAgICAgICAgaWYgKGdldE51bUNvbXBvbmVudHMoKSA9PSAxKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHNyYy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgICAgIGRzdFtpXSA9IGRhdGFbMF1bc3JjW2ldIC0gb2Zmc2V0XTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgZ2V0TnVtQ29tcG9uZW50cygpOyBpKyspIHsKLSAgICAgICAgICAgICAgICBkc3RbaV0gPSBkYXRhW2ldW3NyY1tpXSAtIG9mZnNldF07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZHN0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBieXRlIGFycmF5IHdoaWNoIGNvbnRhaW5zIHNhbXBsZXMgb2YgdGhlIHNwZWNpZmllZCBwaXhlbCB3aGljaAotICAgICAqIGlzIHRyYW5zbGF0ZWQgd2l0aCB0aGUgbG9va3VwIHRhYmxlIG9mIHRoaXMgQnl0ZUxvb2t1cFRhYmxlIG9iamVjdC4gVGhlCi0gICAgICogcmVzdWx0ZWQgYXJyYXkgaXMgc3RvcmVkIHRvIHRoZSBkc3QgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBhcnJheS4KLSAgICAgKiBAcGFyYW0gZHN0Ci0gICAgICogICAgICAgICAgICB0aGUgZGVzdGluYXRpb24gYXJyYXkgd2hlcmUgdGhlIHJlc3VsdCBjYW4gYmUgc3RvcmVkLgotICAgICAqIEByZXR1cm4gdGhlIGJ5dGUgYXJyYXkgb2YgdHJhbnNsYXRlZCBzYW1wbGVzIG9mIGEgcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGJ5dGVbXSBsb29rdXBQaXhlbChieXRlW10gc3JjLCBieXRlW10gZHN0KSB7Ci0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0ID0gbmV3IGJ5dGVbc3JjLmxlbmd0aF07Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgb2Zmc2V0ID0gZ2V0T2Zmc2V0KCk7Ci0gICAgICAgIGlmIChnZXROdW1Db21wb25lbnRzKCkgPT0gMSkgewotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBzcmMubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgICAgICBkc3RbaV0gPSBkYXRhWzBdW3NyY1tpXSAtIG9mZnNldF07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGdldE51bUNvbXBvbmVudHMoKTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgZHN0W2ldID0gZGF0YVtpXVtzcmNbaV0gLSBvZmZzZXRdOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGRzdDsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvQ29sb3JDb252ZXJ0T3AuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9Db2xvckNvbnZlcnRPcC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxYTE3MDBiLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9Db2xvckNvbnZlcnRPcC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNzEwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKiogCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3MyRDsKLWltcG9ydCBqYXZhLmF3dC5Qb2ludDsKLWltcG9ydCBqYXZhLmF3dC5SZW5kZXJpbmdIaW50czsKLWltcG9ydCBqYXZhLmF3dC5jb2xvci5Db2xvclNwYWNlOwotaW1wb3J0IGphdmEuYXd0LmNvbG9yLklDQ19Db2xvclNwYWNlOwotaW1wb3J0IGphdmEuYXd0LmNvbG9yLklDQ19Qcm9maWxlOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUG9pbnQyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmNvbG9yLkNvbG9yQ29udmVydGVyOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuY29sb3IuQ29sb3JTY2FsZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvci5JQ0NfVHJhbnNmb3JtOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBDb2xvckNvbnZlcnRPcCBjbGFzcyBjb252ZXJ0cyB0aGUgcGl4ZWxzIG9mIHRoZSBkYXRhIGluIHRoZSBzb3VyY2UgaW1hZ2UKLSAqIHdpdGggdGhlIHNwZWNpZmllZCBDb2xvclNwYWNlIG9iamVjdHMgb3IgYW4gYXJyYXkgb2YgSUNDX1Byb2ZpbGUgb2JqZWN0cy4gVGhlCi0gKiByZXN1bHQgcGl4ZWxzIGFyZSBzY2FsZWQgdG8gdGhlIHByZWNpc2lvbiBvZiB0aGUgZGVzdGluYXRpb24gaW1hZ2UuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgQ29sb3JDb252ZXJ0T3AgaW1wbGVtZW50cyBCdWZmZXJlZEltYWdlT3AsIFJhc3Rlck9wIHsKLSAgICAvLyBVbnVzZWQgYnV0IHJlcXVpcmVkIGJ5IGludGVyZmFjZXMKLSAgICAvKioKLSAgICAgKiBUaGUgcmVuZGVyaW5nIGhpbnRzLgotICAgICAqLwotICAgIFJlbmRlcmluZ0hpbnRzIHJlbmRlcmluZ0hpbnRzOwotCi0gICAgLy8gU2VxdWVuY2UgY29uc2lzdGluZyBvZiBDb2xvclNwYWNlIGFuZCBJQ0NfUHJvZmlsZSBlbGVtZW50cwotICAgIC8qKgotICAgICAqIFRoZSBjb252ZXJzaW9uIHNlcXVlbmNlLgotICAgICAqLwotICAgIE9iamVjdCBjb252ZXJzaW9uU2VxdWVuY2VbXSA9IG5ldyBJQ0NfUHJvZmlsZVswXTsgLy8gVG8gZWxpbWluYXRlIGNoZWNrcyBmb3IKLQotICAgIC8vIG51bGwKLQotICAgIC8vIE5vdCBudWxsIGlmIENvbG9yQ29udmVydE9wIGlzIGNvbnN0cnVjdGVkIGZyb20gdGhlIGFycmF5IG9mIElDQyBwcm9maWxlcwotICAgIC8qKgotICAgICAqIFRoZSBtaWQgcHJvZmlsZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBJQ0NfUHJvZmlsZSBtaWRQcm9maWxlc1tdOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNjLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgQ29sb3JDb252ZXJ0ZXIgY2MgPSBuZXcgQ29sb3JDb252ZXJ0ZXIoKTsKLQotICAgIC8qKgotICAgICAqIFRoZSB0IGNyZWF0b3IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBJQ0NfVHJhbnNmb21DcmVhdG9yIHRDcmVhdG9yID0gbmV3IElDQ19UcmFuc2ZvbUNyZWF0b3IoKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBpcyBpY2MuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGlzSUNDID0gdHJ1ZTsKLQotICAgIC8vIENhY2hlZCBJQ0NfVHJhbnNmb3JtCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIElDQ19UcmFuc2ZvbUNyZWF0b3IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBjbGFzcyBJQ0NfVHJhbnNmb21DcmVhdG9yIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHRyYW5zZm9ybS4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgSUNDX1RyYW5zZm9ybSB0cmFuc2Zvcm07Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBtYXggY29tcG9uZW50cy4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgaW50IG1heENvbXBvbmVudHM7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEZvciB0aGUgZnVsbCBJQ0MgY2FzZS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBzcmMKLSAgICAgICAgICogICAgICAgICAgICB0aGUgc3JjLgotICAgICAgICAgKiBAcGFyYW0gZHN0Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGRzdC4KLSAgICAgICAgICogQHBhcmFtIGNvbnZTZXEKLSAgICAgICAgICogICAgICAgICAgICB0aGUgY29udiBzZXEuCi0gICAgICAgICAqIEByZXR1cm4gdGhlIHRyYW5zZm9ybS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBJQ0NfVHJhbnNmb3JtIGdldFRyYW5zZm9ybShJQ0NfUHJvZmlsZSBzcmMsIElDQ19Qcm9maWxlIGRzdCwgSUNDX1Byb2ZpbGUgY29udlNlcVtdKSB7Ci0gICAgICAgICAgICBpZiAodHJhbnNmb3JtICE9IG51bGwgJiYgc3JjID09IHRyYW5zZm9ybS5nZXRTcmMoKSAmJiBkc3QgPT0gdHJhbnNmb3JtLmdldERzdCgpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRyYW5zZm9ybTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaW50IGxlbmd0aCA9IGNvbnZTZXEubGVuZ3RoOwotICAgICAgICAgICAgaW50IHNyY0ZsZyA9IDAsIGRzdEZsZyA9IDA7Ci0KLSAgICAgICAgICAgIGlmIChsZW5ndGggPT0gMCB8fCBzcmMgIT0gY29udlNlcVswXSkgewotICAgICAgICAgICAgICAgIGlmIChzcmMgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBzcmNGbGcgPSAxOyAvLyBuZWVkIHNyYyBwcm9maWxlCi0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGxlbmd0aCA9PSAwIHx8IGRzdCAhPSBjb252U2VxW2xlbmd0aCAtIDFdKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGRzdCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGRzdEZsZyA9IDE7IC8vIG5lZWQgZHN0IHByb2ZpbGUKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIElDQ19Qcm9maWxlIHByb2ZpbGVzW107Ci0gICAgICAgICAgICBpbnQgblByb2ZpbGVzID0gbGVuZ3RoICsgc3JjRmxnICsgZHN0RmxnOwotICAgICAgICAgICAgaWYgKG5Qcm9maWxlcyA9PSBsZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBwcm9maWxlcyA9IGNvbnZTZXE7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHByb2ZpbGVzID0gbmV3IElDQ19Qcm9maWxlW25Qcm9maWxlc107Ci0gICAgICAgICAgICAgICAgaW50IHBvcyA9IDA7Ci0gICAgICAgICAgICAgICAgaWYgKHNyY0ZsZyAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHByb2ZpbGVzW3BvcysrXSA9IHNyYzsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBsZW5ndGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBwcm9maWxlc1twb3MrK10gPSBjb252U2VxW2ldOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAoZHN0RmxnICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgcHJvZmlsZXNbcG9zKytdID0gZHN0OwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIHRyYW5zZm9ybSA9IG5ldyBJQ0NfVHJhbnNmb3JtKHByb2ZpbGVzKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBVc2VkIG9ubHkgd2hlbiB0aGVyZSBhcmUgbm9uLUlDQyBjb2xvciBzcGFjZXMuIFJldHVybnMgc2VxdWVuY2Ugb2YKLSAgICAgICAgICogbm9uLUlDQyBjb2xvciBzcGFjZXMgYW5kIElDQyB0cmFuc2Zvcm1zIG1hZGUgZnJvbSBzcmMsIGRzdCBhbmQKLSAgICAgICAgICogY29udmVyc2lvblNlcXVlbmNlLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHNyYwotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBzcmMuCi0gICAgICAgICAqIEBwYXJhbSBkc3QKLSAgICAgICAgICogICAgICAgICAgICB0aGUgZHN0LgotICAgICAgICAgKiBAcmV0dXJuIHRoZSBzZXF1ZW5jZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBPYmplY3RbXSBnZXRTZXF1ZW5jZShPYmplY3Qgc3JjLCBPYmplY3QgZHN0KSB7Ci0gICAgICAgICAgICBBcnJheUxpc3Q8T2JqZWN0PiBwcm9maWxlcyA9IG5ldyBBcnJheUxpc3Q8T2JqZWN0PigxMCk7Ci0gICAgICAgICAgICBBcnJheUxpc3Q8T2JqZWN0PiBzZXF1ZW5jZSA9IG5ldyBBcnJheUxpc3Q8T2JqZWN0PigxMCk7Ci0KLSAgICAgICAgICAgIC8vIFdlIG5lZWQgdGhpcyBwcm9maWxlIGFueXdheQotICAgICAgICAgICAgSUNDX1Byb2ZpbGUgeHl6UHJvZmlsZSA9IElDQ19Qcm9maWxlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1NfQ0lFWFlaKTsKLQotICAgICAgICAgICAgT2JqZWN0IGNvbnZlcnNpb25GaXJzdCA9IG51bGwsIGNvbnZlcnNpb25MYXN0ID0gbnVsbDsKLSAgICAgICAgICAgIGludCBjb252ZXJzaW9uTGVuZ3RoID0gY29udmVyc2lvblNlcXVlbmNlLmxlbmd0aDsKLSAgICAgICAgICAgIGlmIChjb252ZXJzaW9uTGVuZ3RoID4gMCkgewotICAgICAgICAgICAgICAgIGNvbnZlcnNpb25GaXJzdCA9IGNvbnZlcnNpb25TZXF1ZW5jZVswXTsKLSAgICAgICAgICAgICAgICBjb252ZXJzaW9uTGFzdCA9IGNvbnZlcnNpb25TZXF1ZW5jZVtjb252ZXJzaW9uTGVuZ3RoIC0gMV07Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGJvb2xlYW4gaWNjU2VxdWVuY2VTdGFydGVkID0gZmFsc2U7Ci0KLSAgICAgICAgICAgIGlmIChzcmMgIT0gY29udmVyc2lvbkZpcnN0ICYmIHNyYyAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHNyYyBpbnN0YW5jZW9mIElDQ19Qcm9maWxlKSB7Ci0gICAgICAgICAgICAgICAgICAgIHByb2ZpbGVzLmFkZChzcmMpOwotICAgICAgICAgICAgICAgICAgICBpY2NTZXF1ZW5jZVN0YXJ0ZWQgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHByb2ZpbGVzLmFkZCh4eXpQcm9maWxlKTsKLSAgICAgICAgICAgICAgICAgICAgc2VxdWVuY2UuYWRkKHNyYyk7IC8vIEFkZCBub24tSUNDIGNvbG9yIHNwYWNlIHRvIHRoZQotICAgICAgICAgICAgICAgICAgICAvLyBzZXF1ZW5jZQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgcHJvZmlsZXMuYWRkKHh5elByb2ZpbGUpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGNvbnZlcnNpb25MZW5ndGg7IGkrKykgewotICAgICAgICAgICAgICAgIGlmIChjb252ZXJzaW9uU2VxdWVuY2VbaV0gaW5zdGFuY2VvZiBJQ0NfUHJvZmlsZSkgewotICAgICAgICAgICAgICAgICAgICBwcm9maWxlcy5hZGQoY29udmVyc2lvblNlcXVlbmNlW2ldKTsKLSAgICAgICAgICAgICAgICAgICAgaWNjU2VxdWVuY2VTdGFydGVkID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGljY1NlcXVlbmNlU3RhcnRlZCkgewotICAgICAgICAgICAgICAgICAgICBwcm9maWxlcy5hZGQoeHl6UHJvZmlsZSk7Ci0KLSAgICAgICAgICAgICAgICAgICAgLy8gRWxpbWluYXRlIHNhbWUgcHJvZmlsZXMgaWYgdGhlcmUgYXJlIGFueQotICAgICAgICAgICAgICAgICAgICAvLyAoZS5nLiB4eXpQcm9maWxlIG1heSBvY2N1ciBzZXZlcmFsIHRpbWVzKQotICAgICAgICAgICAgICAgICAgICBPYmplY3QgcHJldiA9IHByb2ZpbGVzLmdldCgwKTsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgayA9IDE7IGsgPCBwcm9maWxlcy5zaXplKCk7IGsrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHByZXYgPT0gcHJvZmlsZXMuZ2V0KGspKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgay0tOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHByb2ZpbGVzLnJlbW92ZShrKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIHByZXYgPSBwcm9maWxlcy5nZXQoayk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICAvLyBJZiBvbmx5IG9uZSBwcm9maWxlIGxlZnQgd2Ugc2tpcCB0aGUgdHJhbnNmb3JtIC0KLSAgICAgICAgICAgICAgICAgICAgLy8gaXQgY2FuIGJlIG9ubHkgQ0lFWFlaCi0gICAgICAgICAgICAgICAgICAgIGlmIChwcm9maWxlcy5zaXplKCkgPiAxKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzZXF1ZW5jZS5hZGQobmV3IElDQ19UcmFuc2Zvcm0ocHJvZmlsZXMudG9BcnJheShuZXcgSUNDX1Byb2ZpbGVbMF0pKSk7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIEFkZCBub24tSUNDIGNvbG9yIHNwYWNlIHRvIHRoZSBzZXF1ZW5jZQotICAgICAgICAgICAgICAgICAgICAgICAgc2VxdWVuY2UuYWRkKGNvbnZlcnNpb25TZXF1ZW5jZVtpXSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBwcm9maWxlcy5jbGVhcigpOwotICAgICAgICAgICAgICAgICAgICBwcm9maWxlcy5hZGQoeHl6UHJvZmlsZSk7Ci0gICAgICAgICAgICAgICAgICAgIGljY1NlcXVlbmNlU3RhcnRlZCA9IGZhbHNlOyAvLyBTZXF1ZW5jZSBvZiBJQ0MgcHJvZmlsZXMgaXMKLSAgICAgICAgICAgICAgICAgICAgLy8gcHJvY2Vzc2VkCi0gICAgICAgICAgICAgICAgfSBlbHNlIHsgLy8gQWRkIG5vbi1JQ0MgY29sb3Igc3BhY2UgdG8gdGhlIHNlcXVlbmNlCi0gICAgICAgICAgICAgICAgICAgIHNlcXVlbmNlLmFkZChjb252ZXJzaW9uU2VxdWVuY2VbaV0pOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKGRzdCAhPSBjb252ZXJzaW9uTGFzdCAmJiBkc3QgIT0gbnVsbCkgeyAvLyBBZGQgbGFzdCBwcm9maWxlIGlmCi0gICAgICAgICAgICAgICAgLy8gbmVlZGVkCi0gICAgICAgICAgICAgICAgaWYgKGRzdCBpbnN0YW5jZW9mIElDQ19Qcm9maWxlKSB7Ci0gICAgICAgICAgICAgICAgICAgIHByb2ZpbGVzLmFkZChkc3QpOwotICAgICAgICAgICAgICAgICAgICBpY2NTZXF1ZW5jZVN0YXJ0ZWQgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoaWNjU2VxdWVuY2VTdGFydGVkKSB7Ci0gICAgICAgICAgICAgICAgICAgIHByb2ZpbGVzLmFkZCh4eXpQcm9maWxlKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBzZXF1ZW5jZS5hZGQoZHN0KTsgLy8gQWRkIGxhc3Qgbm9uLUlDQyBjb2xvciBzcGFjZSB0byB0aGUKLSAgICAgICAgICAgICAgICAgICAgLy8gc2VxdWVuY2UKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChpY2NTZXF1ZW5jZVN0YXJ0ZWQpIHsgLy8gTWFrZSBsYXN0IHRyYW5zZm9ybSBpZiBuZWVkZWQKLSAgICAgICAgICAgICAgICBzZXF1ZW5jZS5hZGQobmV3IElDQ19UcmFuc2Zvcm0ocHJvZmlsZXMudG9BcnJheShuZXcgSUNDX1Byb2ZpbGVbMF0pKSk7Ci0gICAgICAgICAgICAgICAgaWYgKGRzdCAhPSBudWxsICYmICEoZHN0IGluc3RhbmNlb2YgSUNDX1Byb2ZpbGUpKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNlcXVlbmNlLmFkZChkc3QpOyAvLyBBZGQgbGFzdCBub24tSUNDIGNvbG9yIHNwYWNlIHRvIHRoZQotICAgICAgICAgICAgICAgICAgICAvLyBzZXF1ZW5jZQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gQ2FsY3VsYXRlIG1heCBudW1iZXIgb2YgY29tcG9uZW50cwotICAgICAgICAgICAgLy8gVGhpcyBudW1iZXIgd2lsbCBiZSB1c2VkIGZvciBtZW1vcnkgYWxsb2NhdGlvbgotICAgICAgICAgICAgbWF4Q29tcG9uZW50cyA9IDA7Ci0gICAgICAgICAgICBPYmplY3QgbzsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBzaXplID0gc2VxdWVuY2Uuc2l6ZSgpOyBpIDwgc2l6ZTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgbyA9IHNlcXVlbmNlLmdldChpKTsKLSAgICAgICAgICAgICAgICBpZiAobyBpbnN0YW5jZW9mIElDQ19UcmFuc2Zvcm0pIHsKLSAgICAgICAgICAgICAgICAgICAgSUNDX1RyYW5zZm9ybSB0ID0gKElDQ19UcmFuc2Zvcm0pbzsKLSAgICAgICAgICAgICAgICAgICAgbWF4Q29tcG9uZW50cyA9IChtYXhDb21wb25lbnRzID4gdC5nZXROdW1JbnB1dENoYW5uZWxzKCkgKyAxKSA/IG1heENvbXBvbmVudHMKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICA6IHQuZ2V0TnVtSW5wdXRDaGFubmVscygpICsgMTsKLSAgICAgICAgICAgICAgICAgICAgbWF4Q29tcG9uZW50cyA9IChtYXhDb21wb25lbnRzID4gdC5nZXROdW1PdXRwdXRDaGFubmVscygpICsgMSkgPyBtYXhDb21wb25lbnRzCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgOiB0LmdldE51bU91dHB1dENoYW5uZWxzKCkgKyAxOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIENvbG9yU3BhY2UgY3MgPSAoQ29sb3JTcGFjZSlvOwotICAgICAgICAgICAgICAgICAgICBtYXhDb21wb25lbnRzID0gKG1heENvbXBvbmVudHMgPiBjcy5nZXROdW1Db21wb25lbnRzKCkgKyAxKSA/IG1heENvbXBvbmVudHMKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICA6IGNzLmdldE51bUNvbXBvbmVudHMoKSArIDE7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZXR1cm4gc2VxdWVuY2UudG9BcnJheSgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IENvbG9yQ29udmVydE9wIG9iamVjdCB1c2luZyB0d28gc3BlY2lmaWVkIENvbG9yU3BhY2UKLSAgICAgKiBvYmplY3RzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmNDUwotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBDb2xvclNwYWNlLgotICAgICAqIEBwYXJhbSBkc3RDUwotICAgICAqICAgICAgICAgICAgdGhlIGRlc3RpbmF0aW9uIENvbG9yU3BhY2UuCi0gICAgICogQHBhcmFtIGhpbnRzCi0gICAgICogICAgICAgICAgICB0aGUgUmVuZGVyaW5nSGludHMgb2JqZWN0IHVzZWQgZm9yIHRoZSBjb2xvciBjb252ZXJzaW9uLCBvcgotICAgICAqICAgICAgICAgICAgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3JDb252ZXJ0T3AoQ29sb3JTcGFjZSBzcmNDUywgQ29sb3JTcGFjZSBkc3RDUywgUmVuZGVyaW5nSGludHMgaGludHMpIHsKLSAgICAgICAgaWYgKHNyY0NTID09IG51bGwgfHwgZHN0Q1MgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI1QiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcmVuZGVyaW5nSGludHMgPSBoaW50czsKLQotICAgICAgICBib29sZWFuIHNyY0lDQyA9IHNyY0NTIGluc3RhbmNlb2YgSUNDX0NvbG9yU3BhY2U7Ci0gICAgICAgIGJvb2xlYW4gZHN0SUNDID0gZHN0Q1MgaW5zdGFuY2VvZiBJQ0NfQ29sb3JTcGFjZTsKLQotICAgICAgICBpZiAoc3JjSUNDICYmIGRzdElDQykgewotICAgICAgICAgICAgY29udmVyc2lvblNlcXVlbmNlID0gbmV3IElDQ19Qcm9maWxlWzJdOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgY29udmVyc2lvblNlcXVlbmNlID0gbmV3IE9iamVjdFsyXTsKLSAgICAgICAgICAgIGlzSUNDID0gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoc3JjSUNDKSB7Ci0gICAgICAgICAgICBjb252ZXJzaW9uU2VxdWVuY2VbMF0gPSAoKElDQ19Db2xvclNwYWNlKXNyY0NTKS5nZXRQcm9maWxlKCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBjb252ZXJzaW9uU2VxdWVuY2VbMF0gPSBzcmNDUzsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkc3RJQ0MpIHsKLSAgICAgICAgICAgIGNvbnZlcnNpb25TZXF1ZW5jZVsxXSA9ICgoSUNDX0NvbG9yU3BhY2UpZHN0Q1MpLmdldFByb2ZpbGUoKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbnZlcnNpb25TZXF1ZW5jZVsxXSA9IGRzdENTOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IENvbG9yQ29udmVydE9wIG9iamVjdCBmcm9tIHRoZSBzcGVjaWZpZWQgSUNDX1Byb2ZpbGUKLSAgICAgKiBvYmplY3RzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm9maWxlcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIElDQ19Qcm9maWxlIG9iamVjdHMuCi0gICAgICogQHBhcmFtIGhpbnRzCi0gICAgICogICAgICAgICAgICB0aGUgUmVuZGVyaW5nSGludHMgb2JqZWN0IHVzZWQgZm9yIHRoZSBjb2xvciBjb252ZXJzaW9uLCBvcgotICAgICAqICAgICAgICAgICAgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3JDb252ZXJ0T3AoSUNDX1Byb2ZpbGUgcHJvZmlsZXNbXSwgUmVuZGVyaW5nSGludHMgaGludHMpIHsKLSAgICAgICAgaWYgKHByb2ZpbGVzID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNUMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJlbmRlcmluZ0hpbnRzID0gaGludHM7Ci0KLSAgICAgICAgLy8gVGhpcyBhcnJheSBpcyBub3QgdXNlZCBpbiB0aGUgcHJvZ3JhbSBsb2dpYywgc28gZG9uJ3QgbmVlZCB0byBjb3B5IGl0Ci0gICAgICAgIC8vIFN0b3JlIGl0IG9ubHkgdG8gcmV0dXJuIGJhY2sKLSAgICAgICAgbWlkUHJvZmlsZXMgPSBwcm9maWxlczsKLQotICAgICAgICBjb252ZXJzaW9uU2VxdWVuY2UgPSBuZXcgSUNDX1Byb2ZpbGVbbWlkUHJvZmlsZXMubGVuZ3RoXTsKLQotICAgICAgICAvLyBBZGQgcHJvZmlsZXMgdG8gdGhlIGNvbnZlcnNpb24gc2VxdWVuY2UKLSAgICAgICAgZm9yIChpbnQgaSA9IDAsIGxlbmd0aCA9IG1pZFByb2ZpbGVzLmxlbmd0aDsgaSA8IGxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBjb252ZXJzaW9uU2VxdWVuY2VbaV0gPSBtaWRQcm9maWxlc1tpXTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBDb2xvckNvbnZlcnRPcCBvYmplY3QgdXNpbmcgdGhlIHNwZWNpZmllZCBDb2xvclNwYWNlCi0gICAgICogb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjcwotICAgICAqICAgICAgICAgICAgdGhlIGRlc3RpbmF0aW9uIENvbG9yU3BhY2Ugb3IgYW4gaW50ZXJtZWRpYXRlIENvbG9yU3BhY2UuCi0gICAgICogQHBhcmFtIGhpbnRzCi0gICAgICogICAgICAgICAgICB0aGUgUmVuZGVyaW5nSGludHMgb2JqZWN0IHVzZWQgZm9yIHRoZSBjb2xvciBjb252ZXJzaW9uLCBvcgotICAgICAqICAgICAgICAgICAgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3JDb252ZXJ0T3AoQ29sb3JTcGFjZSBjcywgUmVuZGVyaW5nSGludHMgaGludHMpIHsKLSAgICAgICAgaWYgKGNzID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNUIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJlbmRlcmluZ0hpbnRzID0gaGludHM7Ci0KLSAgICAgICAgaWYgKGNzIGluc3RhbmNlb2YgSUNDX0NvbG9yU3BhY2UpIHsKLSAgICAgICAgICAgIGNvbnZlcnNpb25TZXF1ZW5jZSA9IG5ldyBJQ0NfUHJvZmlsZVsxXTsKLSAgICAgICAgICAgIGNvbnZlcnNpb25TZXF1ZW5jZVswXSA9ICgoSUNDX0NvbG9yU3BhY2UpY3MpLmdldFByb2ZpbGUoKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbnZlcnNpb25TZXF1ZW5jZSA9IG5ldyBPYmplY3RbMV07Ci0gICAgICAgICAgICBjb252ZXJzaW9uU2VxdWVuY2VbMF0gPSBjczsKLSAgICAgICAgICAgIGlzSUNDID0gZmFsc2U7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQ29sb3JDb252ZXJ0T3Agb2JqZWN0IHdoaWNoIGNvbnZlcnRzIGZyb20gYSBzb3VyY2UKLSAgICAgKiBjb2xvciBzcGFjZSB0byBhIGRlc3RpbmF0aW9uIGNvbG9yIHNwYWNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBoaW50cwotICAgICAqICAgICAgICAgICAgdGhlIFJlbmRlcmluZ0hpbnRzIG9iamVjdCB1c2VkIGZvciB0aGUgY29sb3IgY29udmVyc2lvbiwgb3IKLSAgICAgKiAgICAgICAgICAgIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIENvbG9yQ29udmVydE9wKFJlbmRlcmluZ0hpbnRzIGhpbnRzKSB7Ci0gICAgICAgIHJlbmRlcmluZ0hpbnRzID0gaGludHM7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIFdyaXRhYmxlUmFzdGVyIGZpbHRlcihSYXN0ZXIgc3JjLCBXcml0YWJsZVJhc3RlciBkc3QpIHsKLSAgICAgICAgaWYgKGNvbnZlcnNpb25TZXF1ZW5jZS5sZW5ndGggPCAyKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI1RCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgSUNDX1Byb2ZpbGUgc3JjUGYgPSBudWxsLCBkc3RQZiA9IG51bGw7IC8vIHVudXNlZCBpZiBpc0lDQyBpcyBmYWxzZQotICAgICAgICBpbnQgblNyY0NvbG9yQ29tcHMsIG5Ec3RDb2xvckNvbXBzOwotICAgICAgICBPYmplY3QgZmlyc3QgPSBjb252ZXJzaW9uU2VxdWVuY2VbMF07Ci0gICAgICAgIE9iamVjdCBsYXN0ID0gY29udmVyc2lvblNlcXVlbmNlW2NvbnZlcnNpb25TZXF1ZW5jZS5sZW5ndGggLSAxXTsKLQotICAgICAgICAvLyBHZXQgdGhlIG51bWJlciBvZiBpbnB1dC9vdXRwdXQgY29sb3IgY29tcG9uZW50cwotICAgICAgICBpZiAoaXNJQ0MpIHsKLSAgICAgICAgICAgIHNyY1BmID0gKElDQ19Qcm9maWxlKWZpcnN0OwotICAgICAgICAgICAgZHN0UGYgPSAoSUNDX1Byb2ZpbGUpbGFzdDsKLSAgICAgICAgICAgIG5TcmNDb2xvckNvbXBzID0gc3JjUGYuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICAgICAgbkRzdENvbG9yQ29tcHMgPSBkc3RQZi5nZXROdW1Db21wb25lbnRzKCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpZiAoZmlyc3QgaW5zdGFuY2VvZiBJQ0NfUHJvZmlsZSkgewotICAgICAgICAgICAgICAgIHNyY1BmID0gKElDQ19Qcm9maWxlKWZpcnN0OwotICAgICAgICAgICAgICAgIG5TcmNDb2xvckNvbXBzID0gc3JjUGYuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBuU3JjQ29sb3JDb21wcyA9ICgoQ29sb3JTcGFjZSlmaXJzdCkuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAobGFzdCBpbnN0YW5jZW9mIElDQ19Qcm9maWxlKSB7Ci0gICAgICAgICAgICAgICAgZHN0UGYgPSAoSUNDX1Byb2ZpbGUpbGFzdDsKLSAgICAgICAgICAgICAgICBuRHN0Q29sb3JDb21wcyA9IGRzdFBmLmdldE51bUNvbXBvbmVudHMoKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgbkRzdENvbG9yQ29tcHMgPSAoKENvbG9yU3BhY2UpbGFzdCkuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLy8gQ2hlY2sgdGhhdCBzb3VyY2UgYW5kIGRlc3RpbmF0aW9uIHJhc3RlcnMgYXJlIGNvbXBhdGlibGUgd2l0aAotICAgICAgICAvLyB0cmFuc2Zvcm1zIGFuZCB3aXRoIGVhY2ggb3RoZXIKLSAgICAgICAgaWYgKHNyYy5nZXROdW1CYW5kcygpICE9IG5TcmNDb2xvckNvbXBzKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjVFPUluY29ycmVjdCBudW1iZXIgb2Ygc291cmNlIHJhc3RlciBiYW5kcy4gU2hvdWxkIGJlIGVxdWFsCi0gICAgICAgICAgICAvLyB0byB0aGUgbnVtYmVyIG9mIGNvbG9yIGNvbXBvbmVudHMgb2Ygc291cmNlIGNvbG9yc3BhY2UuCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI1RSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRzdCAhPSBudWxsKSB7IC8vIENoZWNrIGRlc3RpbmF0aW9uIHJhc3RlcgotICAgICAgICAgICAgaWYgKGRzdC5nZXROdW1CYW5kcygpICE9IG5Ec3RDb2xvckNvbXBzKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjI1Rj1JbmNvcnJlY3QgbnVtYmVyIG9mIGRlc3RpbmF0aW9uIHJhc3RlciBiYW5kcy4gU2hvdWxkCi0gICAgICAgICAgICAgICAgLy8gYmUgZXF1YWwgdG8gdGhlIG51bWJlciBvZiBjb2xvciBjb21wb25lbnRzIG9mIGRlc3RpbmF0aW9uCi0gICAgICAgICAgICAgICAgLy8gY29sb3JzcGFjZS4KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI1RiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoc3JjLmdldFdpZHRoKCkgIT0gZHN0LmdldFdpZHRoKCkgfHwgc3JjLmdldEhlaWdodCgpICE9IGRzdC5nZXRIZWlnaHQoKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjYwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGRzdCA9IGNyZWF0ZUNvbXBhdGlibGVEZXN0UmFzdGVyKHNyYyk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoaXNJQ0MpIHsKLSAgICAgICAgICAgIC8vIENyZWF0ZSB0cmFuc2Zvcm0KLSAgICAgICAgICAgIElDQ19UcmFuc2Zvcm0gdCA9IHRDcmVhdG9yCi0gICAgICAgICAgICAgICAgICAgIC5nZXRUcmFuc2Zvcm0oc3JjUGYsIGRzdFBmLCAoSUNDX1Byb2ZpbGVbXSljb252ZXJzaW9uU2VxdWVuY2UpOwotICAgICAgICAgICAgY2MudHJhbnNsYXRlQ29sb3IodCwgc3JjLCBkc3QpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgT2JqZWN0W10gc2VxdWVuY2UgPSB0Q3JlYXRvci5nZXRTZXF1ZW5jZShudWxsLCBudWxsKTsKLQotICAgICAgICAgICAgLy8gR2V0IGRhdGEgZnJvbSB0aGUgc291cmNlIHJhc3RlcgotICAgICAgICAgICAgQ29sb3JTY2FsZXIgc2NhbGVyID0gbmV3IENvbG9yU2NhbGVyKCk7Ci0gICAgICAgICAgICBzY2FsZXIubG9hZFNjYWxpbmdEYXRhKHNyYywgbnVsbCk7Ci0gICAgICAgICAgICBmbG9hdCB0bXBEYXRhW11bXSA9IHNjYWxlci5zY2FsZU5vcm1hbGl6ZShzcmMpOwotCi0gICAgICAgICAgICAvLyBHZXQgc291cmNlIGFuZCBkZXN0aW5hdGlvbiBjb2xvciBzcGFjZXMKLSAgICAgICAgICAgIENvbG9yU3BhY2Ugc3JjQ1MgPSAoc3JjUGYgPT0gbnVsbCkgPyAoQ29sb3JTcGFjZSlmaXJzdCA6IG5ldyBJQ0NfQ29sb3JTcGFjZShzcmNQZik7Ci0gICAgICAgICAgICBDb2xvclNwYWNlIGRzdENTID0gKGRzdFBmID09IG51bGwpID8gKENvbG9yU3BhY2UpbGFzdCA6IG5ldyBJQ0NfQ29sb3JTcGFjZShkc3RQZik7Ci0KLSAgICAgICAgICAgIGFwcGx5U2VxdWVuY2Uoc2VxdWVuY2UsIHRtcERhdGEsIHNyY0NTLCBkc3RDUyk7Ci0KLSAgICAgICAgICAgIHNjYWxlci5sb2FkU2NhbGluZ0RhdGEoZHN0LCBudWxsKTsKLSAgICAgICAgICAgIHNjYWxlci51bnNjYWxlTm9ybWFsaXplZChkc3QsIHRtcERhdGEpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGRzdDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQnVmZmVyZWRJbWFnZSBjcmVhdGVDb21wYXRpYmxlRGVzdEltYWdlKEJ1ZmZlcmVkSW1hZ2Ugc3JjLCBDb2xvck1vZGVsIGRlc3RDTSkgewotICAgICAgICAvLyBJZiBkZXN0aW5hdGlvbiBjb2xvciBtb2RlbCBpcyBwYXNzZWQgb25seSBvbmUgbGluZSBuZWVkZWQKLSAgICAgICAgaWYgKGRlc3RDTSAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IEJ1ZmZlcmVkSW1hZ2UoZGVzdENNLCBkZXN0Q00uY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKHNyYy5nZXRXaWR0aCgpLAotICAgICAgICAgICAgICAgICAgICBzcmMuZ2V0SGVpZ2h0KCkpLCBkZXN0Q00uaXNBbHBoYVByZW11bHRpcGxpZWQoKSwgbnVsbCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgblNwYWNlcyA9IGNvbnZlcnNpb25TZXF1ZW5jZS5sZW5ndGg7Ci0KLSAgICAgICAgaWYgKG5TcGFjZXMgPCAxKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2MSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgLy8gR2V0IGRlc3RpbmF0aW9uIGNvbG9yIHNwYWNlCi0gICAgICAgIE9iamVjdCBkZXN0aW5hdGlvbiA9IGNvbnZlcnNpb25TZXF1ZW5jZVtuU3BhY2VzIC0gMV07Ci0gICAgICAgIENvbG9yU3BhY2UgZHN0Q1MgPSAoZGVzdGluYXRpb24gaW5zdGFuY2VvZiBDb2xvclNwYWNlKSA/IChDb2xvclNwYWNlKWRlc3RpbmF0aW9uCi0gICAgICAgICAgICAgICAgOiBuZXcgSUNDX0NvbG9yU3BhY2UoKElDQ19Qcm9maWxlKWRlc3RpbmF0aW9uKTsKLQotICAgICAgICBDb2xvck1vZGVsIHNyY0NNID0gc3JjLmdldENvbG9yTW9kZWwoKTsKLSAgICAgICAgQ29sb3JNb2RlbCBkc3RDTSA9IG5ldyBDb21wb25lbnRDb2xvck1vZGVsKGRzdENTLCBzcmNDTS5oYXNBbHBoYSgpLCBzcmNDTQotICAgICAgICAgICAgICAgIC5pc0FscGhhUHJlbXVsdGlwbGllZCgpLCBzcmNDTS5nZXRUcmFuc3BhcmVuY3koKSwgc3JjQ00uZ2V0VHJhbnNmZXJUeXBlKCkpOwotCi0gICAgICAgIHJldHVybiBuZXcgQnVmZmVyZWRJbWFnZShkc3RDTSwgZGVzdENNLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3RlcihzcmMuZ2V0V2lkdGgoKSwgc3JjCi0gICAgICAgICAgICAgICAgLmdldEhlaWdodCgpKSwgZGVzdENNLmlzQWxwaGFQcmVtdWx0aXBsaWVkKCksIG51bGwpOwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBCdWZmZXJlZEltYWdlIGZpbHRlcihCdWZmZXJlZEltYWdlIHNyYywgQnVmZmVyZWRJbWFnZSBkc3QpIHsKLSAgICAgICAgaWYgKGRzdCA9PSBudWxsICYmIGNvbnZlcnNpb25TZXF1ZW5jZS5sZW5ndGggPCAxKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2MiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgQ29sb3JNb2RlbCBzcmNDTSA9IHNyYy5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIC8vIEZpcnN0IGhhbmRsZSBpbmRleCBjb2xvciBtb2RlbAotICAgICAgICBpZiAoc3JjQ00gaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwpIHsKLSAgICAgICAgICAgIHNyYyA9ICgoSW5kZXhDb2xvck1vZGVsKXNyY0NNKS5jb252ZXJ0VG9JbnREaXNjcmV0ZShzcmMuZ2V0UmFzdGVyKCksIGZhbHNlKTsKLSAgICAgICAgfQotICAgICAgICBDb2xvclNwYWNlIHNyY0NTID0gc3JjQ00uZ2V0Q29sb3JTcGFjZSgpOwotCi0gICAgICAgIEJ1ZmZlcmVkSW1hZ2UgcmVzOwotICAgICAgICBib29sZWFuIGlzRHN0SW5kZXggPSBmYWxzZTsKLSAgICAgICAgaWYgKGRzdCAhPSBudWxsKSB7Ci0KLSAgICAgICAgICAgIGlmIChzcmMuZ2V0V2lkdGgoKSAhPSBkc3QuZ2V0V2lkdGgoKSB8fCBzcmMuZ2V0SGVpZ2h0KCkgIT0gZHN0LmdldEhlaWdodCgpKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKGRzdC5nZXRDb2xvck1vZGVsKCkgaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwpIHsKLSAgICAgICAgICAgICAgICBpc0RzdEluZGV4ID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICByZXMgPSBjcmVhdGVDb21wYXRpYmxlRGVzdEltYWdlKHNyYywgbnVsbCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJlcyA9IGRzdDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJlcyA9IGNyZWF0ZUNvbXBhdGlibGVEZXN0SW1hZ2Uoc3JjLCBudWxsKTsKLSAgICAgICAgfQotICAgICAgICBDb2xvck1vZGVsIGRzdENNID0gcmVzLmdldENvbG9yTW9kZWwoKTsKLSAgICAgICAgQ29sb3JTcGFjZSBkc3RDUyA9IGRzdENNLmdldENvbG9yU3BhY2UoKTsKLQotICAgICAgICBJQ0NfUHJvZmlsZSBzcmNQZiA9IG51bGwsIGRzdFBmID0gbnVsbDsKLSAgICAgICAgaWYgKHNyY0NTIGluc3RhbmNlb2YgSUNDX0NvbG9yU3BhY2UpIHsKLSAgICAgICAgICAgIHNyY1BmID0gKChJQ0NfQ29sb3JTcGFjZSlzcmNDUykuZ2V0UHJvZmlsZSgpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChkc3RDUyBpbnN0YW5jZW9mIElDQ19Db2xvclNwYWNlKSB7Ci0gICAgICAgICAgICBkc3RQZiA9ICgoSUNDX0NvbG9yU3BhY2UpZHN0Q1MpLmdldFByb2ZpbGUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGJvb2xlYW4gaXNGdWxsSUNDID0gaXNJQ0MgJiYgc3JjUGYgIT0gbnVsbCAmJiBkc3RQZiAhPSBudWxsOwotCi0gICAgICAgIGlmIChpc0Z1bGxJQ0MpIHsKLSAgICAgICAgICAgIElDQ19UcmFuc2Zvcm0gdCA9IHRDcmVhdG9yCi0gICAgICAgICAgICAgICAgICAgIC5nZXRUcmFuc2Zvcm0oc3JjUGYsIGRzdFBmLCAoSUNDX1Byb2ZpbGVbXSljb252ZXJzaW9uU2VxdWVuY2UpOwotICAgICAgICAgICAgY2MudHJhbnNsYXRlQ29sb3IodCwgc3JjLCByZXMpOwotICAgICAgICB9IGVsc2UgeyAvLyBQZXJmb3JtIG5vbi1JQ0MgdHJhbnNmb3JtCi0gICAgICAgICAgICBPYmplY3Qgc2VxdWVuY2VbXSA9IHRDcmVhdG9yLmdldFNlcXVlbmNlKHNyY1BmID09IG51bGwgPyAoT2JqZWN0KXNyY0NTIDogc3JjUGYsCi0gICAgICAgICAgICAgICAgICAgIGRzdFBmID09IG51bGwgPyAoT2JqZWN0KWRzdENTIDogZHN0UGYpOwotCi0gICAgICAgICAgICBpbnQgc3JjVyA9IHNyYy5nZXRXaWR0aCgpOwotICAgICAgICAgICAgaW50IHNyY0ggPSBzcmMuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICBpbnQgbnVtUGl4ZWxzID0gc3JjVyAqIHNyY0g7Ci0KLSAgICAgICAgICAgIC8vIExvYWQgYWxsIHBpeGVsIGRhdGEgaW50byBhcnJheSB0bXBEYXRhCi0gICAgICAgICAgICBmbG9hdCB0bXBEYXRhW11bXSA9IG5ldyBmbG9hdFtudW1QaXhlbHNdW3RDcmVhdG9yLm1heENvbXBvbmVudHNdOwotICAgICAgICAgICAgZm9yIChpbnQgcm93ID0gMCwgZGF0YVBvcyA9IDA7IHJvdyA8IHNyY1c7IHJvdysrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgY29sID0gMDsgY29sIDwgc3JjSDsgY29sKyspIHsKLSAgICAgICAgICAgICAgICAgICAgdG1wRGF0YVtkYXRhUG9zXSA9IHNyY0NNLmdldE5vcm1hbGl6ZWRDb21wb25lbnRzKHNyYy5nZXRSYXN0ZXIoKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIC5nZXREYXRhRWxlbWVudHMocm93LCBjb2wsIG51bGwpLCB0bXBEYXRhW2RhdGFQb3NdLCAwKTsKLSAgICAgICAgICAgICAgICAgICAgZGF0YVBvcysrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gQ29weSBhbHBoYSBjaGFubmVsIGlmIG5lZWRlZAotICAgICAgICAgICAgZmxvYXQgYWxwaGFbXSA9IG51bGw7Ci0gICAgICAgICAgICBpbnQgYWxwaGFJZHggPSBzcmNDTS5udW1Db21wb25lbnRzIC0gMTsKLSAgICAgICAgICAgIGlmIChzcmNDTS5oYXNBbHBoYSgpICYmIGRzdENNLmhhc0FscGhhKCkpIHsKLSAgICAgICAgICAgICAgICBhbHBoYSA9IG5ldyBmbG9hdFtudW1QaXhlbHNdOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtUGl4ZWxzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgYWxwaGFbaV0gPSB0bXBEYXRhW2ldW2FscGhhSWR4XTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIFRyYW5zbGF0ZSBjb2xvcnMKLSAgICAgICAgICAgIGFwcGx5U2VxdWVuY2Uoc2VxdWVuY2UsIHRtcERhdGEsIHNyY0NTLCBkc3RDUyk7Ci0KLSAgICAgICAgICAgIC8vIENvcHkgYWxwaGEgaWYgbmVlZGVkCi0gICAgICAgICAgICBpZiAoZHN0Q00uaGFzQWxwaGEoKSkgewotICAgICAgICAgICAgICAgIGFscGhhSWR4ID0gZHN0Q00ubnVtQ29tcG9uZW50cyAtIDE7Ci0gICAgICAgICAgICAgICAgaWYgKGFscGhhICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1QaXhlbHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgdG1wRGF0YVtpXVthbHBoYUlkeF0gPSBhbHBoYVtpXTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtUGl4ZWxzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHRtcERhdGFbaV1bYWxwaGFJZHhdID0gMWY7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIFN0b3JlIGRhdGEgYmFjayB0byB0aGUgaW1hZ2UKLSAgICAgICAgICAgIGZvciAoaW50IHJvdyA9IDAsIGRhdGFQb3MgPSAwOyByb3cgPCBzcmNXOyByb3crKykgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGNvbCA9IDA7IGNvbCA8IHNyY0g7IGNvbCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJlcy5nZXRSYXN0ZXIoKS5zZXREYXRhRWxlbWVudHMocm93LCBjb2wsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0Q00uZ2V0RGF0YUVsZW1lbnRzKHRtcERhdGFbZGF0YVBvcysrXSwgMCwgbnVsbCkpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChpc0RzdEluZGV4KSB7IC8vIENvbnZlcnQgaW1hZ2UgaW50byBpbmRleGVkIGNvbG9yCi0gICAgICAgICAgICBHcmFwaGljczJEIGcyZCA9IGRzdC5jcmVhdGVHcmFwaGljcygpOwotICAgICAgICAgICAgZzJkLmRyYXdJbWFnZShyZXMsIDAsIDAsIG51bGwpOwotICAgICAgICAgICAgZzJkLmRpc3Bvc2UoKTsKLSAgICAgICAgICAgIHJldHVybiBkc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcmVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFwcGx5IHNlcXVlbmNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzZXF1ZW5jZQotICAgICAqICAgICAgICAgICAgdGhlIHNlcXVlbmNlLgotICAgICAqIEBwYXJhbSB0bXBEYXRhCi0gICAgICogICAgICAgICAgICB0aGUgdG1wIGRhdGEuCi0gICAgICogQHBhcmFtIHNyY0NTCi0gICAgICogICAgICAgICAgICB0aGUgc3JjIGNzLgotICAgICAqIEBwYXJhbSBkc3RDUwotICAgICAqICAgICAgICAgICAgdGhlIGRzdCBjcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgYXBwbHlTZXF1ZW5jZShPYmplY3Qgc2VxdWVuY2VbXSwgZmxvYXQgdG1wRGF0YVtdW10sIENvbG9yU3BhY2Ugc3JjQ1MsCi0gICAgICAgICAgICBDb2xvclNwYWNlIGRzdENTKSB7Ci0gICAgICAgIENvbG9yU3BhY2UgeHl6Q1MgPSBDb2xvclNwYWNlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1NfQ0lFWFlaKTsKLQotICAgICAgICBpbnQgbnVtUGl4ZWxzID0gdG1wRGF0YS5sZW5ndGg7Ci0KLSAgICAgICAgLy8gRmlyc3QgdHJhbnNmb3JtLi4uCi0gICAgICAgIGlmIChzZXF1ZW5jZVswXSBpbnN0YW5jZW9mIElDQ19UcmFuc2Zvcm0pIHsgLy8gSUNDCi0gICAgICAgICAgICBJQ0NfVHJhbnNmb3JtIHQgPSAoSUNDX1RyYW5zZm9ybSlzZXF1ZW5jZVswXTsKLSAgICAgICAgICAgIGNjLnRyYW5zbGF0ZUNvbG9yKHQsIHRtcERhdGEsIHNyY0NTLCB4eXpDUywgbnVtUGl4ZWxzKTsKLSAgICAgICAgfSBlbHNlIHsgLy8gbm9uIElDQwotICAgICAgICAgICAgZm9yIChpbnQgayA9IDA7IGsgPCBudW1QaXhlbHM7IGsrKykgewotICAgICAgICAgICAgICAgIHRtcERhdGFba10gPSBzcmNDUy50b0NJRVhZWih0bXBEYXRhW2tdKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNjLmxvYWRTY2FsaW5nRGF0YSh4eXpDUyk7IC8vIHByZXBhcmUgZm9yIHNjYWxpbmcgWFlaCi0gICAgICAgIH0KLQotICAgICAgICBmb3IgKE9iamVjdCBlbGVtZW50IDogc2VxdWVuY2UpIHsKLSAgICAgICAgICAgIGlmIChlbGVtZW50IGluc3RhbmNlb2YgSUNDX1RyYW5zZm9ybSkgewotICAgICAgICAgICAgICAgIElDQ19UcmFuc2Zvcm0gdCA9IChJQ0NfVHJhbnNmb3JtKWVsZW1lbnQ7Ci0gICAgICAgICAgICAgICAgY2MudHJhbnNsYXRlQ29sb3IodCwgdG1wRGF0YSwgbnVsbCwgbnVsbCwgbnVtUGl4ZWxzKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgQ29sb3JTcGFjZSBjcyA9IChDb2xvclNwYWNlKWVsZW1lbnQ7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgayA9IDA7IGsgPCBudW1QaXhlbHM7IGsrKykgewotICAgICAgICAgICAgICAgICAgICB0bXBEYXRhW2tdID0gY3MuZnJvbUNJRVhZWih0bXBEYXRhW2tdKTsKLSAgICAgICAgICAgICAgICAgICAgdG1wRGF0YVtrXSA9IGNzLnRvQ0lFWFlaKHRtcERhdGFba10pOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIExhc3QgdHJhbnNmb3JtLi4uCi0gICAgICAgIGlmIChzZXF1ZW5jZVtzZXF1ZW5jZS5sZW5ndGggLSAxXSBpbnN0YW5jZW9mIElDQ19UcmFuc2Zvcm0pIHsgLy8gSUNDCi0gICAgICAgICAgICBJQ0NfVHJhbnNmb3JtIHQgPSAoSUNDX1RyYW5zZm9ybSlzZXF1ZW5jZVtzZXF1ZW5jZS5sZW5ndGggLSAxXTsKLSAgICAgICAgICAgIGNjLnRyYW5zbGF0ZUNvbG9yKHQsIHRtcERhdGEsIHh5ekNTLCBkc3RDUywgbnVtUGl4ZWxzKTsKLSAgICAgICAgfSBlbHNlIHsgLy8gbm9uIElDQwotICAgICAgICAgICAgZm9yIChpbnQgayA9IDA7IGsgPCBudW1QaXhlbHM7IGsrKykgewotICAgICAgICAgICAgICAgIHRtcERhdGFba10gPSBkc3RDUy5mcm9tQ0lFWFlaKHRtcERhdGFba10pOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIFBvaW50MkQgZ2V0UG9pbnQyRChQb2ludDJEIHNyY1B0LCBQb2ludDJEIGRzdFB0KSB7Ci0gICAgICAgIGlmIChkc3RQdCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBkc3RQdC5zZXRMb2NhdGlvbihzcmNQdCk7Ci0gICAgICAgICAgICByZXR1cm4gZHN0UHQ7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBQb2ludDJELkZsb2F0KChmbG9hdClzcmNQdC5nZXRYKCksIChmbG9hdClzcmNQdC5nZXRZKCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBjcmVhdGVDb21wYXRpYmxlRGVzdFJhc3RlcihSYXN0ZXIgc3JjKSB7Ci0gICAgICAgIGludCBuQ29tcHMgPSAwOwotICAgICAgICBpbnQgblNwYWNlcyA9IGNvbnZlcnNpb25TZXF1ZW5jZS5sZW5ndGg7Ci0KLSAgICAgICAgaWYgKG5TcGFjZXMgPCAyKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2MSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgT2JqZWN0IGxhc3RDUyA9IGNvbnZlcnNpb25TZXF1ZW5jZVtuU3BhY2VzIC0gMV07Ci0gICAgICAgIGlmIChsYXN0Q1MgaW5zdGFuY2VvZiBDb2xvclNwYWNlKSB7Ci0gICAgICAgICAgICBuQ29tcHMgPSAoKENvbG9yU3BhY2UpbGFzdENTKS5nZXROdW1Db21wb25lbnRzKCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBuQ29tcHMgPSAoKElDQ19Qcm9maWxlKWxhc3RDUykuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gQ2FsY3VsYXRlIGNvcnJlY3QgZGF0YSB0eXBlCi0gICAgICAgIGludCBkc3REYXRhVHlwZSA9IHNyYy5nZXREYXRhQnVmZmVyKCkuZ2V0RGF0YVR5cGUoKTsKLSAgICAgICAgaWYgKGRzdERhdGFUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9CWVRFICYmIGRzdERhdGFUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9TSE9SVCkgewotICAgICAgICAgICAgZHN0RGF0YVR5cGUgPSBEYXRhQnVmZmVyLlRZUEVfU0hPUlQ7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gUmFzdGVyLmNyZWF0ZUludGVybGVhdmVkUmFzdGVyKGRzdERhdGFUeXBlLCBzcmMuZ2V0V2lkdGgoKSwgc3JjLmdldEhlaWdodCgpLCBuQ29tcHMsCi0gICAgICAgICAgICAgICAgbmV3IFBvaW50KHNyYy5nZXRNaW5YKCksIHNyYy5nZXRNaW5ZKCkpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoUmFzdGVyIHNyYykgewotICAgICAgICByZXR1cm4gc3JjLmdldEJvdW5kcygpOwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRChCdWZmZXJlZEltYWdlIHNyYykgewotICAgICAgICByZXR1cm4gc3JjLmdldFJhc3RlcigpLmdldEJvdW5kcygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgSUNDX1Byb2ZpbGVzIG9iamVjdHMgd2hpY2ggY29uc3RydWN0cyB0aGlzCi0gICAgICogQ29sb3JDb252ZXJ0T3Agb2JqZWN0IG9yIHJldHVybnMgbnVsbCBpZiB0aGlzIENvbG9yQ29udmVydE9wIGlzIG5vdAotICAgICAqIGNvbnN0cnVjdGVkIGZyb20gYXJyYXkgb2YgSUNDX1Byb2ZpbGVzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgb2YgSUNDX1Byb2ZpbGVzIG9iamVjdHMgd2hpY2ggY29uc3RydWN0cyB0aGlzCi0gICAgICogICAgICAgICBDb2xvckNvbnZlcnRPcCBvYmplY3Qgb3IgcmV0dXJucyBudWxsIGlmIHRoaXMgQ29sb3JDb252ZXJ0T3AgaXMKLSAgICAgKiAgICAgICAgIG5vdCBjb25zdHJ1Y3RlZCBmcm9tIGFycmF5IG9mIElDQ19Qcm9maWxlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgSUNDX1Byb2ZpbGVbXSBnZXRJQ0NfUHJvZmlsZXMoKSB7Ci0gICAgICAgIGlmIChtaWRQcm9maWxlcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gbWlkUHJvZmlsZXM7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIFJlbmRlcmluZ0hpbnRzIGdldFJlbmRlcmluZ0hpbnRzKCkgewotICAgICAgICByZXR1cm4gcmVuZGVyaW5nSGludHM7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0NvbG9yTW9kZWwuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9Db2xvck1vZGVsLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDFiMDg0ZTEuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL0NvbG9yTW9kZWwuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDk2NCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEuYXd0LlRyYW5zcGFyZW5jeTsKLWltcG9ydCBqYXZhLmF3dC5jb2xvci5Db2xvclNwYWNlOwotaW1wb3J0IGphdmEudXRpbC5BcnJheXM7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgY2xhc3MgQ29sb3JNb2RlbC4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBDb2xvck1vZGVsIGltcGxlbWVudHMgVHJhbnNwYXJlbmN5IHsKLQotICAgIC8qKgotICAgICAqIFRoZSBwaXhlbF9iaXRzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgcGl4ZWxfYml0czsgLy8gUGl4ZWwgbGVuZ3RoIGluIGJpdHMKLQotICAgIC8qKgotICAgICAqIFRoZSB0cmFuc2ZlciB0eXBlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgdHJhbnNmZXJUeXBlOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNzLgotICAgICAqLwotICAgIENvbG9yU3BhY2UgY3M7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGFzIGFscGhhLgotICAgICAqLwotICAgIGJvb2xlYW4gaGFzQWxwaGE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaXMgYWxwaGEgcHJlbXVsdGlwbGllZC4KLSAgICAgKi8KLSAgICBib29sZWFuIGlzQWxwaGFQcmVtdWx0aXBsaWVkOwotCi0gICAgLyoqCi0gICAgICogVGhlIHRyYW5zcGFyZW5jeS4KLSAgICAgKi8KLSAgICBpbnQgdHJhbnNwYXJlbmN5OwotCi0gICAgLyoqCi0gICAgICogVGhlIG51bSBjb2xvciBjb21wb25lbnRzLgotICAgICAqLwotICAgIGludCBudW1Db2xvckNvbXBvbmVudHM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbnVtIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgaW50IG51bUNvbXBvbmVudHM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYml0cy4KLSAgICAgKi8KLSAgICBpbnRbXSBiaXRzOyAvLyBBcnJheSBvZiBjb21wb25lbnRzIG1hc2tzCi0KLSAgICAvKioKLSAgICAgKiBUaGUgbWF4IHZhbHVlcy4KLSAgICAgKi8KLSAgICBpbnRbXSBtYXhWYWx1ZXMgPSBudWxsOyAvLyBNYXggdmFsdWVzIHRoYXQgbWF5IGJlIHJlcHJlc2VudCBieSBjb2xvcgotCi0gICAgLy8gY29tcG9uZW50cwotCi0gICAgLyoqCi0gICAgICogVGhlIG1heCBiaXQgbGVuZ3RoLgotICAgICAqLwotICAgIGludCBtYXhCaXRMZW5ndGg7IC8vIE1heCBsZW5ndGggY29sb3IgY29tcG9uZW50cyBpbiBiaXRzCi0KLSAgICAvKioKLSAgICAgKiBUaGUgUkcgYmRlZmF1bHQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgQ29sb3JNb2RlbCBSR0JkZWZhdWx0OwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGNvbG9yIG1vZGVsIHdpdGggdGhlIHNwZWNpZmllZCB2YWx1ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsX2JpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbCBsZW5ndGggaW4gYml0cy4KLSAgICAgKiBAcGFyYW0gYml0cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGNvbXBvbmVudCBtYXNrcy4KLSAgICAgKiBAcGFyYW0gY3NwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgY29sb3Igc3BhY2UuCi0gICAgICogQHBhcmFtIGhhc0FscGhhCi0gICAgICogICAgICAgICAgICB3aGV0aGVyIHRoZSBjb2xvciBtb2RlbCBoYXMgYWxwaGEuCi0gICAgICogQHBhcmFtIGlzQWxwaGFQcmVtdWx0aXBsaWVkCi0gICAgICogICAgICAgICAgICB3aGV0aGVyIHRoZSBhbHBoYSBpcyBwcmUtbXVsdGlwbGllZC4KLSAgICAgKiBAcGFyYW0gdHJhbnNwYXJlbmN5Ci0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNwYXJlbmN5IHN0cmF0ZWd5LCBAc2VlIGphdmEuYXd0LlRyYW5zcGFyZW5jeS4KLSAgICAgKiBAcGFyYW0gdHJhbnNmZXJUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNmZXIgdHlwZSAocHJpbWl0aXZlIGphdmEgdHlwZSB0byB1c2UgZm9yIHRoZQotICAgICAqICAgICAgICAgICAgY29tcG9uZW50cykuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIENvbG9yTW9kZWwoaW50IHBpeGVsX2JpdHMsIGludFtdIGJpdHMsIENvbG9yU3BhY2UgY3NwYWNlLCBib29sZWFuIGhhc0FscGhhLAotICAgICAgICAgICAgYm9vbGVhbiBpc0FscGhhUHJlbXVsdGlwbGllZCwgaW50IHRyYW5zcGFyZW5jeSwgaW50IHRyYW5zZmVyVHlwZSkgewotCi0gICAgICAgIGlmIChwaXhlbF9iaXRzIDwgMSkgewotICAgICAgICAgICAgLy8gYXd0LjI2Qj1UaGUgbnVtYmVyIG9mIGJpdHMgaW4gdGhlIHBpeGVsIHZhbHVlcyBpcyBsZXNzIHRoYW4gMQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNkIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChiaXRzID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNkM9Yml0cyBpcyBudWxsCi0gICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjZDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpbnQgc3VtID0gMDsKLSAgICAgICAgZm9yIChpbnQgZWxlbWVudCA6IGJpdHMpIHsKLSAgICAgICAgICAgIGlmIChlbGVtZW50IDwgMCkgewotICAgICAgICAgICAgICAgIC8vIGF3dC4yNkQ9VGhlIGVsZW1lbnRzIGluIGJpdHMgaXMgbGVzcyB0aGFuIDAKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2RCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgc3VtICs9IGVsZW1lbnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoc3VtIDwgMSkgewotICAgICAgICAgICAgLy8gYXd0LjI2RT1UaGUgc3VtIG9mIHRoZSBudW1iZXIgb2YgYml0cyBpbiBiaXRzIGlzIGxlc3MgdGhhbiAxCi0gICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjZFIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoY3NwYWNlID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNkY9VGhlIGNzcGFjZSBpcyBudWxsCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2RiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHRyYW5zcGFyZW5jeSA8IFRyYW5zcGFyZW5jeS5PUEFRVUUgfHwgdHJhbnNwYXJlbmN5ID4gVHJhbnNwYXJlbmN5LlRSQU5TTFVDRU5UKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjcwPVRoZSB0cmFuc3BhcmVuY3kgaXMgbm90IGEgdmFsaWQgdmFsdWUKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjcwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICB0aGlzLnBpeGVsX2JpdHMgPSBwaXhlbF9iaXRzOwotICAgICAgICB0aGlzLmJpdHMgPSBiaXRzLmNsb25lKCk7Ci0KLSAgICAgICAgbWF4VmFsdWVzID0gbmV3IGludFtiaXRzLmxlbmd0aF07Ci0gICAgICAgIG1heEJpdExlbmd0aCA9IDA7Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbWF4VmFsdWVzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBtYXhWYWx1ZXNbaV0gPSAoMSA8PCBiaXRzW2ldKSAtIDE7Ci0gICAgICAgICAgICBpZiAoYml0c1tpXSA+IG1heEJpdExlbmd0aCkgewotICAgICAgICAgICAgICAgIG1heEJpdExlbmd0aCA9IGJpdHNbaV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBjcyA9IGNzcGFjZTsKLSAgICAgICAgdGhpcy5oYXNBbHBoYSA9IGhhc0FscGhhOwotICAgICAgICB0aGlzLmlzQWxwaGFQcmVtdWx0aXBsaWVkID0gaXNBbHBoYVByZW11bHRpcGxpZWQ7Ci0gICAgICAgIG51bUNvbG9yQ29tcG9uZW50cyA9IGNzLmdldE51bUNvbXBvbmVudHMoKTsKLQotICAgICAgICBpZiAoaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIG51bUNvbXBvbmVudHMgPSBudW1Db2xvckNvbXBvbmVudHMgKyAxOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbnVtQ29tcG9uZW50cyA9IG51bUNvbG9yQ29tcG9uZW50czsKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMudHJhbnNwYXJlbmN5ID0gdHJhbnNwYXJlbmN5OwotICAgICAgICB0aGlzLnRyYW5zZmVyVHlwZSA9IHRyYW5zZmVyVHlwZTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBjb2xvciBtb2RlbCB3aXRoIHRoZSBzcGVjaWZpZWQgcGl4ZWwgYml0IGRlcHRoLiBUaGUKLSAgICAgKiB0cmFuc2ZlclR5cGUgaXMgY2hvc2VuIGJhc2VkIG9uIHRoZSBwaXhlbCBiaXRzLCBhbmQgdGhlIG90aGVyIGRhdGEgZmllbGRzCi0gICAgICogYXJlIGdpdmVuIGRlZmF1bHQgdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiaXRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgY29tcG9uZW50IG1hc2tzLgotICAgICAqLwotICAgIHB1YmxpYyBDb2xvck1vZGVsKGludCBiaXRzKSB7Ci0KLSAgICAgICAgaWYgKGJpdHMgPCAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjcxPVRoZSBudW1iZXIgb2YgYml0cyBpbiBiaXRzIGlzIGxlc3MgdGhhbiAxCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3MSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcGl4ZWxfYml0cyA9IGJpdHM7Ci0gICAgICAgIHRyYW5zZmVyVHlwZSA9IGdldFRyYW5zZmVyVHlwZShiaXRzKTsKLSAgICAgICAgY3MgPSBDb2xvclNwYWNlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1Nfc1JHQik7Ci0gICAgICAgIGhhc0FscGhhID0gdHJ1ZTsKLSAgICAgICAgaXNBbHBoYVByZW11bHRpcGxpZWQgPSBmYWxzZTsKLSAgICAgICAgdHJhbnNwYXJlbmN5ID0gVHJhbnNwYXJlbmN5LlRSQU5TTFVDRU5UOwotCi0gICAgICAgIG51bUNvbG9yQ29tcG9uZW50cyA9IDM7Ci0gICAgICAgIG51bUNvbXBvbmVudHMgPSA0OwotCi0gICAgICAgIHRoaXMuYml0cyA9IG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBlbGVtZW50cyBmcm9tIHRoZSBzcGVjaWZpZWQgY29tcG9uZW50IGFycmF5LCB0cmFuc2Zvcm1pbmcKLSAgICAgKiB0aGVtIGFjY29yZGluZyB0byBydWxlcyBvZiB0aGUgY29sb3IgbW9kZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbXBvbmVudHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb21wb25lbnRzLgotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIG5vcm1Db21wb25lbnRzIGFycmF5LgotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB0aGF0IHRoZSByZXN1bHQgaXMgd3JpdHRlbiB0bzogYW4gYXJyYXkgb2YgdmFsdWVzCi0gICAgICogICAgICAgICAgICB3aG9zZSBsZW5ndGggbXVzdCBiZSB0aGUgbnVtYmVyIG9mIGNvbXBvbmVudHMgdXNlZCBieSB0aGUKLSAgICAgKiAgICAgICAgICAgIGNvbG9yIG1vZGVsIGFuZCB3aG9zZSB0eXBlIGRlcGVuZHMgb24gdGhlIHRyYW5zZmVyIHR5cGUgKGJhc2VkCi0gICAgICogICAgICAgICAgICBvbiB0aGUgcGl4ZWwgYml0IGRlcHRoKSwgb3IgbnVsbCB0byBoYXZlIHRoZSBhcHByb3ByaWF0ZSBhcnJheQotICAgICAqICAgICAgICAgICAgY3JlYXRlZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBkYXRhIGVsZW1lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBPYmplY3QgZ2V0RGF0YUVsZW1lbnRzKGludFtdIGNvbXBvbmVudHMsIGludCBvZmZzZXQsIE9iamVjdCBvYmopIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJUaGlzIG1ldGhvZCBpcyBub3QgIiArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAic3VwcG9ydGVkIGJ5IHRoaXMgQ29sb3JNb2RlbCIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBlbGVtZW50cyBmcm9tIHRoZSBzcGVjaWZpZWQgYXJyYXkgb2Ygbm9ybWFsaXplZCBjb21wb25lbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBub3JtQ29tcG9uZW50cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG5vcm1hbGl6ZWQgY29tcG9uZW50cy4KLSAgICAgKiBAcGFyYW0gbm9ybU9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgbm9ybUNvbXBvbmVudHMgYXJyYXkuCi0gICAgICogQHBhcmFtIG9iagotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IHRoYXQgdGhlIHJlc3VsdCBpcyB3cml0dGVuIHRvOiBhbiBhcnJheSBvZiB2YWx1ZXMKLSAgICAgKiAgICAgICAgICAgIHdob3NlIGxlbmd0aCBtdXN0IGJlIHRoZSBudW1iZXIgb2YgY29tcG9uZW50cyB1c2VkIGJ5IHRoZQotICAgICAqICAgICAgICAgICAgY29sb3IgbW9kZWwgYW5kIHdob3NlIHR5cGUgZGVwZW5kcyBvbiB0aGUgdHJhbnNmZXIgdHlwZSAoYmFzZWQKLSAgICAgKiAgICAgICAgICAgIG9uIHRoZSBwaXhlbCBiaXQgZGVwdGgpLCBvciBudWxsIHRvIGhhdmUgdGhlIGFwcHJvcHJpYXRlIGFycmF5Ci0gICAgICogICAgICAgICAgICBjcmVhdGVkLgotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGRhdGEgZWxlbWVudHMuCi0gICAgICovCi0gICAgcHVibGljIE9iamVjdCBnZXREYXRhRWxlbWVudHMoZmxvYXRbXSBub3JtQ29tcG9uZW50cywgaW50IG5vcm1PZmZzZXQsIE9iamVjdCBvYmopIHsKLSAgICAgICAgaW50IHVubm9ybUNvbXBvbmVudHNbXSA9IGdldFVubm9ybWFsaXplZENvbXBvbmVudHMobm9ybUNvbXBvbmVudHMsIG5vcm1PZmZzZXQsIG51bGwsIDApOwotICAgICAgICByZXR1cm4gZ2V0RGF0YUVsZW1lbnRzKHVubm9ybUNvbXBvbmVudHMsIDAsIG9iaik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBlbGVtZW50cyBjb3JyZXNwb25kaW5nIHRvIHRoZSBwaXhlbCBkZXRlcm1pbmVkIGJ5IHRoZSBSR0IKLSAgICAgKiBkYXRhLgotICAgICAqIAotICAgICAqIEBwYXJhbSByZ2IKLSAgICAgKiAgICAgICAgICAgIHRoZSBSR0IgaW50ZWdlciB2YWx1ZSB0aGF0IGRlZmluZXMgdGhlIHBpeGVsLgotICAgICAqIEBwYXJhbSBwaXhlbAotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IHRoYXQgdGhlIHJlc3VsdCBpcyB3cml0dGVuIHRvOiBhbiBhcnJheSBvZiB2YWx1ZXMKLSAgICAgKiAgICAgICAgICAgIHdob3NlIGxlbmd0aCBtdXN0IGJlIHRoZSBudW1iZXIgb2YgY29tcG9uZW50cyB1c2VkIGJ5IHRoZQotICAgICAqICAgICAgICAgICAgY29sb3IgbW9kZWwgYW5kIHdob3NlIHR5cGUgZGVwZW5kcyBvbiB0aGUgdHJhbnNmZXIgdHlwZSAoYmFzZWQKLSAgICAgKiAgICAgICAgICAgIG9uIHRoZSBwaXhlbCBiaXQgZGVwdGgpLCBvciBudWxsIHRvIGhhdmUgdGhlIGFwcHJvcHJpYXRlIGFycmF5Ci0gICAgICogICAgICAgICAgICBjcmVhdGVkLgotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGRhdGEgZWxlbWVudHMuCi0gICAgICovCi0gICAgcHVibGljIE9iamVjdCBnZXREYXRhRWxlbWVudHMoaW50IHJnYiwgT2JqZWN0IHBpeGVsKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiVGhpcyBtZXRob2QgaXMgbm90ICIgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgInN1cHBvcnRlZCBieSB0aGlzIENvbG9yTW9kZWwiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNoaWxkIHJhc3RlciBjb3JyZXNwb25kaW5nIHRvIHRoZSBhbHBoYSBjaGFubmVsIG9mIHRoZSBzcGVjaWZpZWQKLSAgICAgKiB3cml0YWJsZSByYXN0ZXIsIG9yIG51bGwgaWYgYWxwaGEgaXMgbm90IHN1cHBvcnRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmFzdGVyCi0gICAgICogICAgICAgICAgICB0aGUgcmFzdGVyLgotICAgICAqIEByZXR1cm4gdGhlIGFscGhhIHJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgZ2V0QWxwaGFSYXN0ZXIoV3JpdGFibGVSYXN0ZXIgcmFzdGVyKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBuZXcgY29sb3IgbW9kZWwgYnkgY29lcmNpbmcgdGhlIGRhdGEgaW4gdGhlIHdyaXRhYmxlIHJhc3RlciBpbgotICAgICAqIGFjY29yZGFuY2Ugd2l0aCB0aGUgYWxwaGEgc3RyYXRlZ3kgb2YgdGhpcyBjb2xvciBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmFzdGVyCi0gICAgICogICAgICAgICAgICB0aGUgcmFzdGVyLgotICAgICAqIEBwYXJhbSBpc0FscGhhUHJlbXVsdGlwbGllZAotICAgICAqICAgICAgICAgICAgd2hldGhlciB0aGUgYWxwaGEgaXMgcHJlLW11bHRpcGxpZWQgaW4gdGhpcyBjb2xvciBtb2RlbAotICAgICAqIEByZXR1cm4gdGhlIG5ldyBjb2xvciBtb2RlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3JNb2RlbCBjb2VyY2VEYXRhKFdyaXRhYmxlUmFzdGVyIHJhc3RlciwgYm9vbGVhbiBpc0FscGhhUHJlbXVsdGlwbGllZCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIlRoaXMgbWV0aG9kIGlzIG5vdCAiICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICJzdXBwb3J0ZWQgYnkgdGhpcyBDb2xvck1vZGVsIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICAvLyBUaGUgb3V0cHV0IGZvcm1hdCBiYXNlZCBvbiAxLjUgcmVsZWFzZSBiZWhhdmlvci4KLSAgICAgICAgLy8gSXQgY291bGQgYmUgcmV2ZWxlZCBzdWNoIHdheToKLSAgICAgICAgLy8gQ29sb3JNb2RlbCBjbSA9IG5ldwotICAgICAgICAvLyBDb21wb25lbnRDb2xvck1vZGVsKENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19zUkdCLAotICAgICAgICAvLyBmYWxzZSwgZmFsc2UsIFRyYW5zcGFyZW5jeS5PUEFRVUUsIERhdGFCdWZmZXIuVFlQRV9CWVRFKTsKLSAgICAgICAgLy8gU3lzdGVtLm91dC5wcmludGxuKGNtLnRvU3RyaW5nKCkpOwotICAgICAgICByZXR1cm4gIkNvbG9yTW9kZWw6IENvbG9yIFNwYWNlID0gIiArIGNzLnRvU3RyaW5nKCkgKyAiOyBoYXMgYWxwaGEgPSAiIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgICAgICsgaGFzQWxwaGEgKyAiOyBpcyBhbHBoYSBwcmVtdWx0aXBpZWQgPSAiIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICArIGlzQWxwaGFQcmVtdWx0aXBsaWVkICsgIjsgdHJhbnNwYXJlbmN5ID0gIiArIHRyYW5zcGFyZW5jeSAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgKyAiOyBudW1iZXIgY29sb3IgY29tcG9uZW50cyA9ICIgKyBudW1Db2xvckNvbXBvbmVudHMgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICsgIjsgcGl4ZWwgYml0cyA9ICIgKyBwaXhlbF9iaXRzICsgIjsgdHJhbnNmZXIgdHlwZSA9ICIgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgKyB0cmFuc2ZlclR5cGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29tcG9uZW50cyBvZiB0aGUgcGl4ZWwgZGV0ZXJtaW5lZCBieSB0aGUgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcGl4ZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRoYXQgZGVmaW5lcyB0aGUgcGl4ZWwgKHdob3NlIHByaW1pdGl2ZSB0eXBlCi0gICAgICogICAgICAgICAgICBjb3JyZXNwb25kcyB0byB0aGUgcGl4ZWwgbGVuZ3RoIGluIGJpdHMuCi0gICAgICogQHNlZSBDb2xvck1vZGVsI2dldFRyYW5zZmVyVHlwZSgpCi0gICAgICogQHBhcmFtIGNvbXBvbmVudHMKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aGUgYXJyYXkgd2hlcmUgdGhlIHJlc3VsdGluZyBjb21wb25lbnRzIGFyZSB3cml0dGVuIChvcgotICAgICAqICAgICAgICAgICAgbnVsbCB0byBwcm9tcHQgdGhlIG1ldGhvZCB0byBjcmVhdGUgdGhlIHJldHVybiBhcnJheSkuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCB0aGF0IHRlbGxzIHdoZXJlIHRoZSByZXN1bHRzIHNob3VsZCBiZSB3cml0dGVuIGluCi0gICAgICogICAgICAgICAgICB0aGUgcmV0dXJuIGFycmF5LgotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldENvbXBvbmVudHMoT2JqZWN0IHBpeGVsLCBpbnRbXSBjb21wb25lbnRzLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiVGhpcyBtZXRob2QgaXMgbm90ICIgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgInN1cHBvcnRlZCBieSB0aGlzIENvbG9yTW9kZWwiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG5vcm1hbGl6ZWQgY29tcG9uZW50cyBvZiB0aGUgcGl4ZWwgZGV0ZXJtaW5lZCBieSB0aGUgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcGl4ZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRoYXQgZGVmaW5lcyB0aGUgcGl4ZWwgKHdob3NlIHByaW1pdGl2ZSB0eXBlCi0gICAgICogICAgICAgICAgICBjb3JyZXNwb25kcyB0byB0aGUgcGl4ZWwgbGVuZ3RoIGluIGJpdHMuCi0gICAgICogQHNlZSBDb2xvck1vZGVsI2dldFRyYW5zZmVyVHlwZSgpCi0gICAgICogQHBhcmFtIG5vcm1Db21wb25lbnRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgd2hlcmUgdGhlIHJlc3VsdGluZyBub3JtYWxpemVkIGNvbXBvbmVudHMgYXJlCi0gICAgICogICAgICAgICAgICB3cml0dGVuIChvciBudWxsIHRvIHByb21wdCB0aGUgbWV0aG9kIHRvIGNyZWF0ZSB0aGUgcmV0dXJuCi0gICAgICogICAgICAgICAgICBhcnJheSkuCi0gICAgICogQHBhcmFtIG5vcm1PZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgdGhhdCB0ZWxscyB3aGVyZSB0aGUgcmVzdWx0cyBzaG91bGQgYmUgd3JpdHRlbiBpbgotICAgICAqICAgICAgICAgICAgdGhlIHJldHVybiBhcnJheS4KLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBub3JtYWxpemVkIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0W10gZ2V0Tm9ybWFsaXplZENvbXBvbmVudHMoT2JqZWN0IHBpeGVsLCBmbG9hdFtdIG5vcm1Db21wb25lbnRzLCBpbnQgbm9ybU9mZnNldCkgewotCi0gICAgICAgIGlmIChwaXhlbCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjk0PXBpeGVsIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOTQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGludCB1bm5vcm1Db21wb25lbnRzW10gPSBnZXRDb21wb25lbnRzKHBpeGVsLCBudWxsLCAwKTsKLSAgICAgICAgcmV0dXJuIGdldE5vcm1hbGl6ZWRDb21wb25lbnRzKHVubm9ybUNvbXBvbmVudHMsIDAsIG5vcm1Db21wb25lbnRzLCBub3JtT2Zmc2V0KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG9iaikgewotICAgICAgICBpZiAoIShvYmogaW5zdGFuY2VvZiBDb2xvck1vZGVsKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIENvbG9yTW9kZWwgY20gPSAoQ29sb3JNb2RlbClvYmo7Ci0KLSAgICAgICAgcmV0dXJuIChwaXhlbF9iaXRzID09IGNtLmdldFBpeGVsU2l6ZSgpICYmIHRyYW5zZmVyVHlwZSA9PSBjbS5nZXRUcmFuc2ZlclR5cGUoKQotICAgICAgICAgICAgICAgICYmIGNzLmdldFR5cGUoKSA9PSBjbS5nZXRDb2xvclNwYWNlKCkuZ2V0VHlwZSgpICYmIGhhc0FscGhhID09IGNtLmhhc0FscGhhKCkKLSAgICAgICAgICAgICAgICAmJiBpc0FscGhhUHJlbXVsdGlwbGllZCA9PSBjbS5pc0FscGhhUHJlbXVsdGlwbGllZCgpCi0gICAgICAgICAgICAgICAgJiYgdHJhbnNwYXJlbmN5ID09IGNtLmdldFRyYW5zcGFyZW5jeSgpCi0gICAgICAgICAgICAgICAgJiYgbnVtQ29sb3JDb21wb25lbnRzID09IGNtLmdldE51bUNvbG9yQ29tcG9uZW50cygpCi0gICAgICAgICAgICAgICAgJiYgbnVtQ29tcG9uZW50cyA9PSBjbS5nZXROdW1Db21wb25lbnRzKCkgJiYgQXJyYXlzLmVxdWFscyhiaXRzLCBjbQotICAgICAgICAgICAgICAgIC5nZXRDb21wb25lbnRTaXplKCkpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSByZWQgY29tcG9uZW50IG9mIHRoZSBwaXhlbCBkZXRlcm1pbmVkIGJ5IHRoZSBkYXRhIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbkRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRoYXQgZGVmaW5lcyB0aGUgcGl4ZWwgKHdob3NlIHByaW1pdGl2ZSB0eXBlCi0gICAgICogICAgICAgICAgICBjb3JyZXNwb25kcyB0byB0aGUgcGl4ZWwgbGVuZ3RoIGluIGJpdHMuCi0gICAgICogQHNlZSBDb2xvck1vZGVsI2dldFRyYW5zZmVyVHlwZSgpCi0gICAgICogQHJldHVybiB0aGUgcmVkLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0UmVkKE9iamVjdCBpbkRhdGEpIHsKLSAgICAgICAgcmV0dXJuIGdldFJlZChjb25zdHJ1Y3RQaXhlbChpbkRhdGEpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBSR0IgaW50ZWdlciB2YWx1ZSBjb3JyZXNwb25kaW5nIHRvIHRoZSBwaXhlbCBkZWZpbmVkIGJ5IHRoZSBkYXRhCi0gICAgICogYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGluRGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXkgdGhhdCBkZWZpbmVzIHRoZSBwaXhlbCAod2hvc2UgcHJpbWl0aXZlIHR5cGUKLSAgICAgKiAgICAgICAgICAgIGNvcnJlc3BvbmRzIHRvIHRoZSBwaXhlbCBsZW5ndGggaW4gYml0cy4KLSAgICAgKiBAc2VlIENvbG9yTW9kZWwjZ2V0VHJhbnNmZXJUeXBlKCkKLSAgICAgKiBAcmV0dXJuIHRoZSBpbnRlZ2VyIHZhbHVlIHRoYXQgZ2l2ZXMgdGhlIHBpeGVsJ3MgY29sb3JzIGluIFJHQiBmb3JtYXQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRSR0IoT2JqZWN0IGluRGF0YSkgewotICAgICAgICByZXR1cm4gKGdldEFscGhhKGluRGF0YSkgPDwgMjQgfCBnZXRSZWQoaW5EYXRhKSA8PCAxNiB8IGdldEdyZWVuKGluRGF0YSkgPDwgOCB8IGdldEJsdWUoaW5EYXRhKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZ3JlZW4gY29tcG9uZW50IG9mIHRoZSBwaXhlbCBkZWZpbmVkIGJ5IHRoZSBkYXRhIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbkRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRoYXQgZGVmaW5lcyB0aGUgcGl4ZWwgKHdob3NlIHByaW1pdGl2ZSB0eXBlCi0gICAgICogICAgICAgICAgICBjb3JyZXNwb25kcyB0byB0aGUgcGl4ZWwgbGVuZ3RoIGluIGJpdHMuCi0gICAgICogQHNlZSBDb2xvck1vZGVsI2dldFRyYW5zZmVyVHlwZSgpCi0gICAgICogQHJldHVybiB0aGUgZ3JlZW4uCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRHcmVlbihPYmplY3QgaW5EYXRhKSB7Ci0gICAgICAgIHJldHVybiBnZXRHcmVlbihjb25zdHJ1Y3RQaXhlbChpbkRhdGEpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBibHVlIGNvbXBvbmVudCBvZiB0aGUgcGl4ZWwgZGVmaW5lZCBieSB0aGUgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5EYXRhCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBhcnJheSB0aGF0IGRlZmluZXMgdGhlIHBpeGVsICh3aG9zZSBwcmltaXRpdmUgdHlwZQotICAgICAqICAgICAgICAgICAgY29ycmVzcG9uZHMgdG8gdGhlIHBpeGVsIGxlbmd0aCBpbiBiaXRzLgotICAgICAqIEBzZWUgQ29sb3JNb2RlbCNnZXRUcmFuc2ZlclR5cGUoKQotICAgICAqIEByZXR1cm4gdGhlIGJsdWUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRCbHVlKE9iamVjdCBpbkRhdGEpIHsKLSAgICAgICAgcmV0dXJuIGdldEJsdWUoY29uc3RydWN0UGl4ZWwoaW5EYXRhKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYWxwaGEgY29tcG9uZW50IG9mIHRoZSBwaXhlbCBkZWZpbmVkIGJ5IHRoZSBkYXRhIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbkRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRoYXQgZGVmaW5lcyB0aGUgcGl4ZWwgKHdob3NlIHByaW1pdGl2ZSB0eXBlCi0gICAgICogICAgICAgICAgICBjb3JyZXNwb25kcyB0byB0aGUgcGl4ZWwgbGVuZ3RoIGluIGJpdHMuCi0gICAgICogQHNlZSBDb2xvck1vZGVsI2dldFRyYW5zZmVyVHlwZSgpCi0gICAgICogQHJldHVybiB0aGUgYWxwaGEuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRBbHBoYShPYmplY3QgaW5EYXRhKSB7Ci0gICAgICAgIHJldHVybiBnZXRBbHBoYShjb25zdHJ1Y3RQaXhlbChpbkRhdGEpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgY29tcGF0aWJsZSB3cml0YWJsZSByYXN0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgZGVzaXJlZCB3cml0YWJsZSByYXN0ZXIuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGRlc2lyZWQgd3JpdGFibGUgcmFzdGVyLgotICAgICAqIEByZXR1cm4gdGhlIHdyaXRhYmxlIHJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKGludCB3LCBpbnQgaCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIlRoaXMgbWV0aG9kIGlzIG5vdCAiICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICJzdXBwb3J0ZWQgYnkgdGhpcyBDb2xvck1vZGVsIik7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIHNhbXBsZSBtb2RlbCBpcyBjb21wYXRpYmxlIHdpdGggdGhpcyBjb2xvciBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc20KLSAgICAgKiAgICAgICAgICAgIHRoZSBzYW1wbGUgbW9kZWwuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc2FtcGxlIG1vZGVsIGlzIGNvbXBhdGlibGUgd2l0aCB0aGlzIGNvbG9yIG1vZGVsLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzQ29tcGF0aWJsZVNhbXBsZU1vZGVsKFNhbXBsZU1vZGVsIHNtKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiVGhpcyBtZXRob2QgaXMgbm90ICIgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgInN1cHBvcnRlZCBieSB0aGlzIENvbG9yTW9kZWwiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGNvbXBhdGlibGUgc2FtcGxlIG1vZGVsLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGRlc2lyZWQgc2FtcGxlIG1vZGVsLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBkZXNpcmVkIHNhbXBsZSBtb2RlbC4KLSAgICAgKiBAcmV0dXJuIHRoZSBzYW1wbGUgbW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIFNhbXBsZU1vZGVsIGNyZWF0ZUNvbXBhdGlibGVTYW1wbGVNb2RlbChpbnQgdywgaW50IGgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJUaGlzIG1ldGhvZCBpcyBub3QgIiArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAic3VwcG9ydGVkIGJ5IHRoaXMgQ29sb3JNb2RlbCIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoZSBzcGVjaWZpZWQgcmFzdGVyIGlzIGNvbXBhdGlibGUgd2l0aCB0aGlzIGNvbG9yIG1vZGVsLgotICAgICAqIAotICAgICAqIEBwYXJhbSByYXN0ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSByYXN0ZXIgdG8gaW5zcGVjdC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSByYXN0ZXIgaXMgY29tcGF0aWJsZSB3aXRoIHRoaXMgY29sb3IgbW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb21wYXRpYmxlUmFzdGVyKFJhc3RlciByYXN0ZXIpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJUaGlzIG1ldGhvZCBpcyBub3QgIiArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAic3VwcG9ydGVkIGJ5IHRoaXMgQ29sb3JNb2RlbCIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29sb3Igc3BhY2Ugb2YgdGhpcyBjb2xvciBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjb2xvciBzcGFjZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgQ29sb3JTcGFjZSBnZXRDb2xvclNwYWNlKCkgewotICAgICAgICByZXR1cm4gY3M7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbm9ybWFsaXplZCBjb21wb25lbnRzIGNvcnJlc3BvbmRpbmcgdG8gdGhlIHNwZWNpZmllZAotICAgICAqIHVubm9ybWFsaXplZCBjb21wb25lbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb21wb25lbnRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgdW5ub3JtYWxpemVkIGNvbXBvbmVudHMuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCB3aGVyZSB0aGUgY29tcG9uZW50cyBzaG91bGQgYmUgcmVhZCBmcm9tIHRoZSBhcnJheQotICAgICAqICAgICAgICAgICAgb2YgdW5ub3JtYWxpemVkIGNvbXBvbmVudHMuCi0gICAgICogQHBhcmFtIG5vcm1Db21wb25lbnRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgd2hlcmUgdGhlIHJlc3VsdGluZyBub3JtYWxpemVkIGNvbXBvbmVudHMgYXJlCi0gICAgICogICAgICAgICAgICB3cml0dGVuIChvciBudWxsIHRvIHByb21wdCB0aGUgbWV0aG9kIHRvIGNyZWF0ZSB0aGUgcmV0dXJuCi0gICAgICogICAgICAgICAgICBhcnJheSkuCi0gICAgICogQHBhcmFtIG5vcm1PZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgdGhhdCB0ZWxscyB3aGVyZSB0aGUgcmVzdWx0cyBzaG91bGQgYmUgd3JpdHRlbiBpbgotICAgICAqICAgICAgICAgICAgdGhlIHJldHVybiBhcnJheS4KLSAgICAgKiBAcmV0dXJuIHRoZSBub3JtYWxpemVkIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0W10gZ2V0Tm9ybWFsaXplZENvbXBvbmVudHMoaW50W10gY29tcG9uZW50cywgaW50IG9mZnNldCwgZmxvYXQgbm9ybUNvbXBvbmVudHNbXSwKLSAgICAgICAgICAgIGludCBub3JtT2Zmc2V0KSB7Ci0gICAgICAgIGlmIChiaXRzID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNkM9Yml0cyBpcyBudWxsCi0gICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjZDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAobm9ybUNvbXBvbmVudHMgPT0gbnVsbCkgewotICAgICAgICAgICAgbm9ybUNvbXBvbmVudHMgPSBuZXcgZmxvYXRbbnVtQ29tcG9uZW50cyArIG5vcm1PZmZzZXRdOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGhhc0FscGhhICYmIGlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0gICAgICAgICAgICBmbG9hdCBub3JtQWxwaGEgPSAoZmxvYXQpY29tcG9uZW50c1tvZmZzZXQgKyBudW1Db2xvckNvbXBvbmVudHNdCi0gICAgICAgICAgICAgICAgICAgIC8gbWF4VmFsdWVzW251bUNvbG9yQ29tcG9uZW50c107Ci0gICAgICAgICAgICBpZiAobm9ybUFscGhhICE9IDAuMGYpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbG9yQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIG5vcm1Db21wb25lbnRzW25vcm1PZmZzZXQgKyBpXSA9IGNvbXBvbmVudHNbb2Zmc2V0ICsgaV0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAvIChub3JtQWxwaGEgKiBtYXhWYWx1ZXNbaV0pOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBub3JtQ29tcG9uZW50c1tub3JtT2Zmc2V0ICsgbnVtQ29sb3JDb21wb25lbnRzXSA9IG5vcm1BbHBoYTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1Db21wb25lbnRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgbm9ybUNvbXBvbmVudHNbbm9ybU9mZnNldCArIGldID0gMC4wZjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgICAgIG5vcm1Db21wb25lbnRzW25vcm1PZmZzZXQgKyBpXSA9IChmbG9hdCljb21wb25lbnRzW29mZnNldCArIGldIC8gbWF4VmFsdWVzW2ldOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5vcm1Db21wb25lbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRhdGEgZWxlbWVudCBjb3JyZXNwb25kaW5nIHRvIHRoZSB1bm5vcm1hbGl6ZWQgY29tcG9uZW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29tcG9uZW50cwotICAgICAqICAgICAgICAgICAgdGhlIGNvbXBvbmVudHMuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCB0byBzdGFydCByZWFkaW5nIHRoZSBjb21wb25lbnRzIGZyb20gdGhlIGFycmF5IG9mCi0gICAgICogICAgICAgICAgICBjb21wb25lbnRzLgotICAgICAqIEByZXR1cm4gdGhlIGRhdGEgZWxlbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldERhdGFFbGVtZW50KGludFtdIGNvbXBvbmVudHMsIGludCBvZmZzZXQpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJUaGlzIG1ldGhvZCBpcyBub3QgIiArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAic3VwcG9ydGVkIGJ5IHRoaXMgQ29sb3JNb2RlbCIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdW5ub3JtYWxpemVkIGNvbXBvbmVudHMgY29ycmVzcG9uZGluZyB0byB0aGUgc3BlY2lmaWVkCi0gICAgICogbm9ybWFsaXplZCBjb21wb25lbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBub3JtQ29tcG9uZW50cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIG5vcm1hbGl6ZWQgY29tcG9uZW50cy4KLSAgICAgKiBAcGFyYW0gbm9ybU9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCB3aGVyZSB0aGUgY29tcG9uZW50cyBzaG91bGQgYmUgcmVhZCBmcm9tIHRoZSBhcnJheQotICAgICAqICAgICAgICAgICAgb2Ygbm9ybWFsaXplZCBjb21wb25lbnRzLgotICAgICAqIEBwYXJhbSBjb21wb25lbnRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgd2hlcmUgdGhlIHJlc3VsdGluZyB1bm5vcm1hbGl6ZWQgY29tcG9uZW50cyBhcmUKLSAgICAgKiAgICAgICAgICAgIHdyaXR0ZW4gKG9yIG51bGwgdG8gcHJvbXB0IHRoZSBtZXRob2QgdG8gY3JlYXRlIHRoZSByZXR1cm4KLSAgICAgKiAgICAgICAgICAgIGFycmF5KS4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IHRoYXQgdGVsbHMgd2hlcmUgdGhlIHJlc3VsdHMgc2hvdWxkIGJlIHdyaXR0ZW4gaW4KLSAgICAgKiAgICAgICAgICAgIHRoZSByZXR1cm4gYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgdW5ub3JtYWxpemVkIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldFVubm9ybWFsaXplZENvbXBvbmVudHMoZmxvYXQgbm9ybUNvbXBvbmVudHNbXSwgaW50IG5vcm1PZmZzZXQsCi0gICAgICAgICAgICBpbnQgY29tcG9uZW50c1tdLCBpbnQgb2Zmc2V0KSB7Ci0KLSAgICAgICAgaWYgKGJpdHMgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjI2Qz1iaXRzIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNkMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChub3JtQ29tcG9uZW50cy5sZW5ndGggLSBub3JtT2Zmc2V0IDwgbnVtQ29tcG9uZW50cykgewotICAgICAgICAgICAgLy8gYXd0LjI3Mz1UaGUgbGVuZ3RoIG9mIG5vcm1Db21wb25lbnRzIG1pbnVzIG5vcm1PZmZzZXQgaXMgbGVzcwotICAgICAgICAgICAgLy8gdGhhbiBudW1Db21wb25lbnRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGNvbXBvbmVudHMgPT0gbnVsbCkgewotICAgICAgICAgICAgY29tcG9uZW50cyA9IG5ldyBpbnRbbnVtQ29tcG9uZW50cyArIG9mZnNldF07Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpZiAoY29tcG9uZW50cy5sZW5ndGggLSBvZmZzZXQgPCBudW1Db21wb25lbnRzKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjI3Mj1UaGUgbGVuZ3RoIG9mIGNvbXBvbmVudHMgbWludXMgb2Zmc2V0IGlzIGxlc3MgdGhhbgotICAgICAgICAgICAgICAgIC8vIG51bUNvbXBvbmVudHMKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3MiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGhhc0FscGhhICYmIGlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0gICAgICAgICAgICBmbG9hdCBhbHBoYSA9IG5vcm1Db21wb25lbnRzW25vcm1PZmZzZXQgKyBudW1Db2xvckNvbXBvbmVudHNdOwotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1Db2xvckNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgICAgIGNvbXBvbmVudHNbb2Zmc2V0ICsgaV0gPSAoaW50KShub3JtQ29tcG9uZW50c1tub3JtT2Zmc2V0ICsgaV0gKiBtYXhWYWx1ZXNbaV0KLSAgICAgICAgICAgICAgICAgICAgICAgICogYWxwaGEgKyAwLjVmKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNvbXBvbmVudHNbb2Zmc2V0ICsgbnVtQ29sb3JDb21wb25lbnRzXSA9IChpbnQpKG5vcm1Db21wb25lbnRzW25vcm1PZmZzZXQKLSAgICAgICAgICAgICAgICAgICAgKyBudW1Db2xvckNvbXBvbmVudHNdCi0gICAgICAgICAgICAgICAgICAgICogbWF4VmFsdWVzW251bUNvbG9yQ29tcG9uZW50c10gKyAwLjVmKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgY29tcG9uZW50c1tvZmZzZXQgKyBpXSA9IChpbnQpKG5vcm1Db21wb25lbnRzW25vcm1PZmZzZXQgKyBpXSAqIG1heFZhbHVlc1tpXSArIDAuNWYpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGNvbXBvbmVudHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBlbGVtZW50IGNvcnJlc3BvbmRpbmcgdG8gdGhlIG5vcm1hbGl6ZWQgY29tcG9uZW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbm9ybUNvbXBvbmVudHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBub3JtYWxpemVkIGNvbXBvbmVudHMuCi0gICAgICogQHBhcmFtIG5vcm1PZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgd2hlcmUgdGhlIG5vcm1hbGl6ZWQgY29tcG9uZW50cyBzaG91bGQgYmUgcmVhZCBmcm9tCi0gICAgICogICAgICAgICAgICB0aGUgbm9ybWFsaXplZCBjb21wb25lbnQgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgZGF0YSBlbGVtZW50LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0RGF0YUVsZW1lbnQoZmxvYXQgbm9ybUNvbXBvbmVudHNbXSwgaW50IG5vcm1PZmZzZXQpIHsKLSAgICAgICAgaW50IHVubm9ybUNvbXBvbmVudHNbXSA9IGdldFVubm9ybWFsaXplZENvbXBvbmVudHMobm9ybUNvbXBvbmVudHMsIG5vcm1PZmZzZXQsIG51bGwsIDApOwotICAgICAgICByZXR1cm4gZ2V0RGF0YUVsZW1lbnQodW5ub3JtQ29tcG9uZW50cywgMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGFrZXMgYSBwaXhlbCB3aG9zZSBkYXRhIGlzIGRlZmluZWQgYnkgYW4gaW50ZWdlciwgYW5kIHdyaXRlcyB0aGUKLSAgICAgKiBjb3JyZXNwb25kaW5nIGNvbXBvbmVudHMgaW50byB0aGUgY29tcG9uZW50cyBhcnJheSwgc3RhcnRpbmcgZnJvbSB0aGUKLSAgICAgKiBpbmRleCBvZmZzZXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsCi0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwgZGF0YS4KLSAgICAgKiBAcGFyYW0gY29tcG9uZW50cwotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXkgdG8gd3JpdGUgdGhlIGNvbXBvbmVudHMgdG8gKG9yIG51bGwgdG8gaGF2ZSB0aGUKLSAgICAgKiAgICAgICAgICAgIG1ldGhvZCBjcmVhdGUgdGhlIHJldHVybiBhcnJheSkuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCB0aGF0IGRldGVybWluZXMgd2hlcmUgdGhlIHJlc3VsdHMgYXJlIHdyaXR0ZW4gaW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBjb21wb25lbnRzIGFycmF5LgotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGNvbXBvbmVudHMgY29ycmVzcG9uZGluZyB0byB0aGUgcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldENvbXBvbmVudHMoaW50IHBpeGVsLCBpbnQgY29tcG9uZW50c1tdLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiVGhpcyBtZXRob2QgaXMgbm90ICIgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgInN1cHBvcnRlZCBieSB0aGlzIENvbG9yTW9kZWwiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHJlZCBjb21wb25lbnQgb2YgdGhlIHBpeGVsIGRldGVybWluZWQgYnkgdGhlIHBpeGVsIGRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsCi0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwuCi0gICAgICogQHJldHVybiB0aGUgcmVkIGNvbXBvbmVudCBvZiB0aGUgZ2l2ZW4gcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXRSZWQoaW50IHBpeGVsKTsKLQotICAgIC8qKgotICAgICAqIFRha2VzIHRoZSBwaXhlbCBkYXRhIGFuZCByZXR1cm5zIHRoZSBpbnRlZ2VyIHZhbHVlIGNvcnJlc3BvbmRpbmcgdG8gdGhlCi0gICAgICogcGl4ZWwncyBjb2xvciBpbiBSR0IgZm9ybWF0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwaXhlbAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgY29ycmVzcG9uZGluZyBSR0IgaW50ZWdlciB2YWx1ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFJHQihpbnQgcGl4ZWwpIHsKLSAgICAgICAgcmV0dXJuIChnZXRBbHBoYShwaXhlbCkgPDwgMjQgfCBnZXRSZWQocGl4ZWwpIDw8IDE2IHwgZ2V0R3JlZW4ocGl4ZWwpIDw8IDggfCBnZXRCbHVlKHBpeGVsKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZ3JlZW4gY29tcG9uZW50IG9mIHRoZSBwaXhlbCBkZXRlcm1pbmVkIGJ5IHRoZSBwaXhlbCBkYXRhLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwaXhlbAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsLgotICAgICAqIEByZXR1cm4gdGhlIGdyZWVuIGNvbXBvbmVudCBvZiB0aGUgZ2l2ZW4gcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXRHcmVlbihpbnQgcGl4ZWwpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2l6ZSBvZiB0aGUgZGVzaXJlZCBjb21wb25lbnQgb2YgdGhpcyBjb2xvciBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29tcG9uZW50SWR4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggdGhhdCBkZXRlcm1pbmVzIHdoaWNoIGNvbXBvbmVudCBzaXplIHRvIGdldC4KLSAgICAgKiBAcmV0dXJuIHRoZSBjb21wb25lbnQgc2l6ZSBjb3JyZXNwb25kaW5nIHRvIHRoZSBpbmRleC4KLSAgICAgKiBAdGhyb3dzIE51bGxQb2ludGVyRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhpcyBjb2xvciBtb2RlbCBkb2Vzbid0IHN1cHBvcnQgYW4gYXJyYXkgb2Ygc2VwYXJhdGUKLSAgICAgKiAgICAgICAgICAgICBjb21wb25lbnRzLgotICAgICAqIEB0aHJvd3MgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIGluZGV4IGlzIG5lZ2F0aXZlIG9yIGdyZWF0ZXIgdGhhbiBvciBlcXVhbCB0byB0aGUKLSAgICAgKiAgICAgICAgICAgICBudW1iZXIgb2YgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldENvbXBvbmVudFNpemUoaW50IGNvbXBvbmVudElkeCkgewotICAgICAgICBpZiAoYml0cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjZDPWJpdHMgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2QyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGNvbXBvbmVudElkeCA8IDAgfHwgY29tcG9uZW50SWR4ID49IGJpdHMubGVuZ3RoKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjc0PWNvbXBvbmVudElkeCBpcyBncmVhdGVyIHRoYW4gdGhlIG51bWJlciBvZiBjb21wb25lbnRzIG9yCi0gICAgICAgICAgICAvLyBsZXNzIHRoYW4gemVybwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNzQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBiaXRzW2NvbXBvbmVudElkeF07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYmx1ZSBjb21wb25lbnQgb2YgdGhlIHBpeGVsIGRldGVybWluZWQgYnkgdGhlIHBpeGVsIGRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsCi0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwuCi0gICAgICogQHJldHVybiB0aGUgYmx1ZSBjb21wb25lbnQgb2YgdGhlIGdpdmVuIHBpeGVsLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0Qmx1ZShpbnQgcGl4ZWwpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYWxwaGEgY29tcG9uZW50IG9mIHRoZSBwaXhlbCBkZXRlcm1pbmVkIGJ5IHRoZSBwaXhlbCBkYXRhLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwaXhlbAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsLgotICAgICAqIEByZXR1cm4gdGhlIGFscGhhIGNvbXBvbmVudCBvZiB0aGUgZ2l2ZW4gcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXRBbHBoYShpbnQgcGl4ZWwpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYXJyYXkgb2Ygc2l6ZXMgb2YgdGhlIGRpZmZlcmVudCBjb21wb25lbnRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIHNpemVzIG9mIHRoZSBkaWZmZXJlbnQgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50W10gZ2V0Q29tcG9uZW50U2l6ZSgpIHsKLSAgICAgICAgaWYgKGJpdHMgIT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIGJpdHMuY2xvbmUoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIGFscGhhIGNvbXBvbmVudCBpcyBwcmUtbXVsdGlwbGllZC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBhbHBoYSBjb21wb25lbnQgaXMgcHJlLW11bHRpcGxpZWQuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGJvb2xlYW4gaXNBbHBoYVByZW11bHRpcGxpZWQoKSB7Ci0gICAgICAgIHJldHVybiBpc0FscGhhUHJlbXVsdGlwbGllZDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciB0aGlzIGNvbG9yIG1vZGVsIHN1cHBvcnRzIGFscGhhLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBjb2xvciBtb2RlbCBoYXMgYWxwaGEuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGJvb2xlYW4gaGFzQWxwaGEoKSB7Ci0gICAgICAgIHJldHVybiBoYXNBbHBoYTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGhhc2hDb2RlKCkgewotICAgICAgICBpbnQgaGFzaCA9IDA7Ci0gICAgICAgIGludCB0bXA7Ci0KLSAgICAgICAgaWYgKGhhc0FscGhhKSB7Ci0gICAgICAgICAgICBoYXNoIF49IDE7Ci0gICAgICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICB9Ci0gICAgICAgIGlmIChpc0FscGhhUHJlbXVsdGlwbGllZCkgewotICAgICAgICAgICAgaGFzaCBePSAxOwotICAgICAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgfQotCi0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIF49IG51bUNvbG9yQ29tcG9uZW50czsKLSAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgaGFzaCB8PSB0bXA7Ci0KLSAgICAgICAgdG1wID0gaGFzaCA+Pj4gMjQ7Ci0gICAgICAgIGhhc2ggXj0gdHJhbnNwYXJlbmN5OwotICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICBoYXNoIHw9IHRtcDsKLQotICAgICAgICB0bXAgPSBoYXNoID4+PiAyNDsKLSAgICAgICAgaGFzaCBePSBjcy5nZXRUeXBlKCk7Ci0gICAgICAgIGhhc2ggPDw9IDg7Ci0gICAgICAgIGhhc2ggfD0gdG1wOwotCi0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIF49IHBpeGVsX2JpdHM7Ci0gICAgICAgIGhhc2ggPDw9IDg7Ci0gICAgICAgIGhhc2ggfD0gdG1wOwotCi0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIF49IHRyYW5zZmVyVHlwZTsKLSAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgaGFzaCB8PSB0bXA7Ci0KLSAgICAgICAgaWYgKGJpdHMgIT0gbnVsbCkgewotCi0gICAgICAgICAgICBmb3IgKGludCBlbGVtZW50IDogYml0cykgewotICAgICAgICAgICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICAgICAgICAgIGhhc2ggXj0gZWxlbWVudDsKLSAgICAgICAgICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICAgICAgICAgIGhhc2ggfD0gdG1wOwotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gaGFzaDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFRyYW5zcGFyZW5jeSgpIHsKLSAgICAgICAgcmV0dXJuIHRyYW5zcGFyZW5jeTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0cmFuc2ZlciB0eXBlLCB3aGljaCBpcyB0aGUgdHlwZSBvZiBKYXZhIHByaW1pdGl2ZSB2YWx1ZSB0aGF0Ci0gICAgICogY29ycmVzcG9uZHMgdG8gdGhlIGJpdCBsZW5ndGggcGVyIHBpeGVsOiBlaXRoZXIKLSAgICAgKiB7QGxpbmsgRGF0YUJ1ZmZlciNUWVBFX0JZVEV9LCB7QGxpbmsgRGF0YUJ1ZmZlciNUWVBFX1VTSE9SVH0sCi0gICAgICoge0BsaW5rIERhdGFCdWZmZXIjVFlQRV9JTlR9LCBvciB7QGxpbmsgRGF0YUJ1ZmZlciNUWVBFX1VOREVGSU5FRH0uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdHJhbnNmZXIgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldFRyYW5zZmVyVHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIHRyYW5zZmVyVHlwZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBwaXhlbCBzaXplIGluIGJpdHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgcGl4ZWwgc2l6ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFBpeGVsU2l6ZSgpIHsKLSAgICAgICAgcmV0dXJuIHBpeGVsX2JpdHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGNvbXBvbmVudHMgb2YgdGhpcyBjb2xvciBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE51bUNvbXBvbmVudHMoKSB7Ci0gICAgICAgIHJldHVybiBudW1Db21wb25lbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG51bWJlciBvZiBjb2xvciBjb21wb25lbnRzIG9mIHRoaXMgY29sb3IgbW9kZWwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIGNvbG9yIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXROdW1Db2xvckNvbXBvbmVudHMoKSB7Ci0gICAgICAgIHJldHVybiBudW1Db2xvckNvbXBvbmVudHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGVmYXVsdCBSR0IgY29sb3IgbW9kZWwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGVmYXVsdCBSR0IgY29sb3IgbW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBDb2xvck1vZGVsIGdldFJHQmRlZmF1bHQoKSB7Ci0gICAgICAgIGlmIChSR0JkZWZhdWx0ID09IG51bGwpIHsKLSAgICAgICAgICAgIFJHQmRlZmF1bHQgPSBuZXcgRGlyZWN0Q29sb3JNb2RlbCgzMiwgMHgwMGZmMDAwMCwgMHgwMDAwZmYwMCwgMHgwMDAwMDBmZiwgMHhmZjAwMDAwMCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIFJHQmRlZmF1bHQ7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBDb25zdHJ1Y3QgSU5UIHBpeGVsIHJlcHJlc2VudGF0aW9uIGZyb20gT2JqZWN0Ci0gICAgICogQHBhcmFtIG9iagotICAgICAqIEByZXR1cm4KLSAgICAgKi8KLSAgICAvKioKLSAgICAgKiBDb25zdHJ1Y3QgcGl4ZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9iagotICAgICAqICAgICAgICAgICAgdGhlIG9iai4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgY29uc3RydWN0UGl4ZWwoT2JqZWN0IG9iaikgewotICAgICAgICBpbnQgcGl4ZWwgPSAwOwotCi0gICAgICAgIHN3aXRjaCAoZ2V0VHJhbnNmZXJUeXBlKCkpIHsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlW10gYlBpeGVsID0gKGJ5dGVbXSlvYmo7Ci0gICAgICAgICAgICAgICAgaWYgKGJQaXhlbC5sZW5ndGggPiAxKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIGF3dC4yNzU9VGhpcyBwaXhlbCByZXByZXNlbnRhdGlvbiBpcyBub3Qgc3V1cG9ydGVkIGJ5IHRpcwotICAgICAgICAgICAgICAgICAgICAvLyBDb2xvciBNb2RlbAotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjc1IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHBpeGVsID0gYlBpeGVsWzBdICYgMHhmZjsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0W10gc1BpeGVsID0gKHNob3J0W10pb2JqOwotICAgICAgICAgICAgICAgIGlmIChzUGl4ZWwubGVuZ3RoID4gMSkgewotICAgICAgICAgICAgICAgICAgICAvLyBhd3QuMjc1PVRoaXMgcGl4ZWwgcmVwcmVzZW50YXRpb24gaXMgbm90IHN1dXBvcnRlZCBieSB0aXMKLSAgICAgICAgICAgICAgICAgICAgLy8gQ29sb3IgTW9kZWwKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3NSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBwaXhlbCA9IHNQaXhlbFswXSAmIDB4ZmZmZjsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGludFtdIGlQaXhlbCA9IChpbnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgaWYgKGlQaXhlbC5sZW5ndGggPiAxKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIGF3dC4yNzU9VGhpcyBwaXhlbCByZXByZXNlbnRhdGlvbiBpcyBub3Qgc3V1cG9ydGVkIGJ5IHRpcwotICAgICAgICAgICAgICAgICAgICAvLyBDb2xvciBNb2RlbAotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjc1IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHBpeGVsID0gaVBpeGVsWzBdOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4yMkQ9VGhpcyB0cmFuc2ZlclR5cGUgKCB7MH0gKSBpcyBub3Qgc3VwcG9ydGVkIGJ5IHRoaXMKLSAgICAgICAgICAgICAgICAvLyBjb2xvciBtb2RlbAotICAgICAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMkQiLCAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgICAgICB0cmFuc2ZlclR5cGUpKTsKLQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBwaXhlbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0cmFuc2ZlciB0eXBlLCB3aGljaCBpcyB0aGUgdHlwZSBvZiBKYXZhIHByaW1pdGl2ZSB2YWx1ZSB0aGF0Ci0gICAgICogY29ycmVzcG9uZHMgdG8gdGhlIGJpdCBsZW5ndGggcGVyIHBpeGVsOiBlaXRoZXIKLSAgICAgKiB7QGxpbmsgRGF0YUJ1ZmZlciNUWVBFX0JZVEV9LCB7QGxpbmsgRGF0YUJ1ZmZlciNUWVBFX1VTSE9SVH0sCi0gICAgICoge0BsaW5rIERhdGFCdWZmZXIjVFlQRV9JTlR9LCBvciB7QGxpbmsgRGF0YUJ1ZmZlciNUWVBFX1VOREVGSU5FRH0uCi0gICAgICogCi0gICAgICogQHBhcmFtIGJpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBjb21wb25lbnQgbWFza3MuCi0gICAgICogQHJldHVybiB0aGUgdHJhbnNmZXIgdHlwZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgaW50IGdldFRyYW5zZmVyVHlwZShpbnQgYml0cykgewotICAgICAgICBpZiAoYml0cyA8PSA4KSB7Ci0gICAgICAgICAgICByZXR1cm4gRGF0YUJ1ZmZlci5UWVBFX0JZVEU7Ci0gICAgICAgIH0gZWxzZSBpZiAoYml0cyA8PSAxNikgewotICAgICAgICAgICAgcmV0dXJuIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ7Ci0gICAgICAgIH0gZWxzZSBpZiAoYml0cyA8PSAzMikgewotICAgICAgICAgICAgcmV0dXJuIERhdGFCdWZmZXIuVFlQRV9JTlQ7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICByZXR1cm4gRGF0YUJ1ZmZlci5UWVBFX1VOREVGSU5FRDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGZpbmFsaXplKCkgewotICAgICAgICAvLyBUaGlzIG1ldGhvZCBpcyBhZGRlZCBmb3IgdGhlIEFQSSBjb21wYXRpYmlsaXR5Ci0gICAgICAgIC8vIERvbid0IG5lZWQgdG8gY2FsbCBzdXBlciBzaW5jZSBPYmplY3QncyBmaW5hbGl6ZSBpcyBhbHdheXMgZW1wdHkKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvQ29tcG9uZW50Q29sb3JNb2RlbC5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL0NvbXBvbmVudENvbG9yTW9kZWwuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDMyOGZkMy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvQ29tcG9uZW50Q29sb3JNb2RlbC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTQ4MiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEuYXd0LmNvbG9yLkNvbG9yU3BhY2U7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmNvbG9yLkxVVENvbG9yQ29udmVydGVyOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBDbGFzcyBDb21wb25lbnRDb2xvck1vZGVsIHJlcHJlc2VudHMgYSBjb2xvciBtb2RlbCB0aGF0IGlzIGRlZmluZWQgaW4KLSAqIHRlcm1zIG9mIGl0cyBjb21wb25lbnRzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIENvbXBvbmVudENvbG9yTW9kZWwgZXh0ZW5kcyBDb2xvck1vZGVsIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBzaWduZWQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIHNpZ25lZDsgLy8gUGl4ZWwgc2FtcGxlcyBhcmUgc2lnbmVkLgotCi0gICAgLy8gU2FtcGxlcyB3aXRoIFRyYW5zZmVyVHlwZSBEYXRhQnVmZmVyLlRZUEVfQllURSwKLSAgICAvLyBEYXRhQnVmZmVyLlRZUEVfVVNIT1JULCBEYXRhQnVmZmVyLlRZUEVfSU5UIC0KLSAgICAvLyB1bnNpZ25lZC4gU2FtcGxlcyB3aXRoIG90aGVycyBUcmFuc2ZlclR5cGUgLQotICAgIC8vIHNpZ25lZC4KLQotICAgIC8qKgotICAgICAqIFRoZSBpbnRlZ3JhbC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gaW50ZWdyYWw7IC8vIFBpeGVsIHNhbXBsZXMgYXJlIGludGVncmFsLgotCi0gICAgLy8gU2FtcGxlcyB3aXRoIFRyYW5zZmVyVHlwZSBEYXRhQnVmZmVyLlRZUEVfQllURSwKLSAgICAvLyBEYXRhQnVmZmVyLlRZUEVfVVNIT1JULCBEYXRhQnVmZmVyLlNob3J0IGFuZAotICAgIC8vIERhdGFCdWZmZXIuVFlQRV9JTlQgLSBpbnRlZ3JhbC4KLQotICAgIC8qKgotICAgICAqIFRoZSBzY2FsZSBmYWN0b3JzLgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgc2NhbGVGYWN0b3JzW107IC8vIEFycmF5IG9mIGZhY3RvcnMgZm9yIHJlZHVjdGlvbiBjb21wb25lbnRzCi0KLSAgICAvLyB2YWx1ZXMgaW50byB0aGUgZm9ybSBzY2FsZWQgZnJvbSAwIHRvIDI1NQotCi0gICAgLyoqCi0gICAgICogVGhlIGRvbm90IHN1cHBvcnQgdW5ub3JtYWxpemVkLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBkb25vdFN1cHBvcnRVbm5vcm1hbGl6ZWQ7IC8vIFRoaXMgQ29sb3IgTW9kZWwgZG9uJ3Qgc3VwcG9ydAotCi0gICAgLy8gdW5ub3Jtb2xpemVkIGZvcm0KLQotICAgIC8qKgotICAgICAqIFRoZSBuZWVkIGFscGhhIGRpdmlkZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gbmVlZEFscGhhRGl2aWRlOyAvLyBoYXNBbHBoYSAmJiBpc0FscGhhUHJlbXVsdGlwbGllZAotCi0gICAgLyoqCi0gICAgICogVGhlIGNhbGMgdmFsdWUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGNhbGNWYWx1ZTsgLy8gVmFsdWUgd2FzIGN1bGN1bGF0ZWQKLQotICAgIC8qKgotICAgICAqIFRoZSBuZWVkIHNjYWxlLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBuZWVkU2NhbGU7IC8vIE5vcm1hbGl6ZWQgdmFsdWUgbmVlZCB0byBzY2FsZQotCi0gICAgLyoqCi0gICAgICogVGhlIG1pbiB2YWxzLgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgbWluVmFsc1tdOyAvLyBBcnJheSBvZiBNaW4gbm9ybWFsaXplZCB2YWx1ZXMKLQotICAgIC8qKgotICAgICAqIFRoZSByYW5nZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmbG9hdCByYW5nZXNbXTsgLy8gQXJyYXkgb2YgcmFuZ2Ugbm9ybWFsaXplZCB2YWx1ZXMKLQotICAgIC8qKgotICAgICAqIFRoZSBhbHBoYSBsdXQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBieXRlIGFscGhhTFVUW107IC8vIExvb2t1cCB0YWJsZSBmb3Igc2NhbGUgYWxwaGEgdmFsdWUKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciBsdSB0cy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJ5dGUgY29sb3JMVVRzW11bXTsgLy8gTG9va3VwIHRhYmxlcyBmb3Igc2NhbGUgY29sb3IgdmFsdWVzCi0KLSAgICAvKioKLSAgICAgKiBUaGUgZnJvbV8gbGluZWEgcl8gcmcgYl8gbHV0LgotICAgICAqLwotICAgIHByaXZhdGUgYnl0ZSBmcm9tX0xJTkVBUl9SR0JfTFVUW107IC8vIExvb2t1cCB0YWJsZSBmb3IgY29udmVyc2lvbiBmcm9tCi0KLSAgICAvLyBMaW5lYXIgUkdCIENvbG9yIFNwYWNlIGludG8gc1JHQgotCi0gICAgLyoqCi0gICAgICogVGhlIHRvXyBsaW5lYSByXzggcmcgYl8gbHV0LgotICAgICAqLwotICAgIHByaXZhdGUgYnl0ZSB0b19MSU5FQVJfOFJHQl9MVVRbXTsgLy8gTG9va3VwIHRhYmxlIGZvciBjb252ZXJzaW9uIGZyb20KLQotICAgIC8vIHNSR0IgQ29sb3IgU3BhY2UgaW50byBMaW5lYXIgUkdCCi0gICAgLy8gOCBiaXQKLQotICAgIC8qKgotICAgICAqIFRoZSB0b18gbGluZWEgcl8xNiByZyBiXyBsdXQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzaG9ydCB0b19MSU5FQVJfMTZSR0JfTFVUW107IC8vIExvb2t1cCB0YWJsZSBmb3IgY29udmVyc2lvbiBmcm9tCi0KLSAgICAvLyBzUkdCIENvbG9yIFNwYWNlIGludG8gTGluZWFyIFJHQgotICAgIC8vIDE2IGJpdAotCi0gICAgLyoqCi0gICAgICogVGhlIExJTkVBIHJfIHJnIGJfIGxlbmd0aC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBMSU5FQVJfUkdCX0xlbmd0aDsgLy8gTGluZWFyIFJHQiBiaXQgbGVuZ3RoCi0KLSAgICAvKioKLSAgICAgKiBUaGUgZmFjdG9yLgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgZkZhY3RvcjsgLy8gU2NhbGUgZmFjdG9yCi0KLSAgICAvKioKLSAgICAgKiBUaGUgaXNfcyByZ2IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGlzX3NSR0I7IC8vIENvbG9yTW9kZWwgaGFzIHNSR0IgQ29sb3JTcGFjZQotCi0gICAgLyoqCi0gICAgICogVGhlIGlzXyBsaW5lYSByXyByZ2IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGlzX0xJTkVBUl9SR0I7IC8vIENvbG9yIE1vZGVsIGhhcyBMaW5lYXIgUkdCIENvbG9yCi0KLSAgICAvLyBTcGFjZQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGNvbXBvbmVudCBjb2xvciBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29sb3JTcGFjZQotICAgICAqICAgICAgICAgICAgdGhlIGNvbG9yIHNwYWNlLgotICAgICAqIEBwYXJhbSBiaXRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgY29tcG9uZW50IG1hc2tzLgotICAgICAqIEBwYXJhbSBoYXNBbHBoYQotICAgICAqICAgICAgICAgICAgd2hldGhlciB0aGUgY29sb3IgbW9kZWwgaGFzIGFscGhhLgotICAgICAqIEBwYXJhbSBpc0FscGhhUHJlbXVsdGlwbGllZAotICAgICAqICAgICAgICAgICAgd2hldGhlciB0aGUgYWxwaGEgaXMgcHJlLW11bHRpcGxpZWQuCi0gICAgICogQHBhcmFtIHRyYW5zcGFyZW5jeQotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zcGFyZW5jeSBzdHJhdGVneSwgQHNlZSBqYXZhLmF3dC5UcmFuc3BhcmVuY3kuCi0gICAgICogQHBhcmFtIHRyYW5zZmVyVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zZmVyIHR5cGUgKHByaW1pdGl2ZSBqYXZhIHR5cGUgdG8gdXNlIGZvciB0aGUKLSAgICAgKiAgICAgICAgICAgIGNvbXBvbmVudHMpLgotICAgICAqLwotICAgIHB1YmxpYyBDb21wb25lbnRDb2xvck1vZGVsKENvbG9yU3BhY2UgY29sb3JTcGFjZSwgaW50IGJpdHNbXSwgYm9vbGVhbiBoYXNBbHBoYSwKLSAgICAgICAgICAgIGJvb2xlYW4gaXNBbHBoYVByZW11bHRpcGxpZWQsIGludCB0cmFuc3BhcmVuY3ksIGludCB0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgc3VwZXIoY3JlYXRlUGl4ZWxCaXRzKGNvbG9yU3BhY2UsIGhhc0FscGhhLCB0cmFuc2ZlclR5cGUpLCB2YWxpZGF0ZUJpdHMoYml0cywgY29sb3JTcGFjZSwKLSAgICAgICAgICAgICAgICBoYXNBbHBoYSwgdHJhbnNmZXJUeXBlKSwgY29sb3JTcGFjZSwgaGFzQWxwaGEsIGlzQWxwaGFQcmVtdWx0aXBsaWVkLCB0cmFuc3BhcmVuY3ksCi0gICAgICAgICAgICAgICAgdHJhbnNmZXJUeXBlKTsKLQotICAgICAgICBuZWVkU2NhbGUgPSBmYWxzZTsKLSAgICAgICAgc3dpdGNoICh0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgc2lnbmVkID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgaW50ZWdyYWwgPSB0cnVlOwotICAgICAgICAgICAgICAgIGRvbm90U3VwcG9ydFVubm9ybWFsaXplZCA9IGZhbHNlOwotICAgICAgICAgICAgICAgIHNjYWxlRmFjdG9ycyA9IG5ldyBmbG9hdFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbG9yQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNjYWxlRmFjdG9yc1tpXSA9IDEuMGYgLyBtYXhWYWx1ZXNbaV07Ci0gICAgICAgICAgICAgICAgICAgIGlmIChjcy5nZXRNaW5WYWx1ZShpKSAhPSAwLjBmIHx8IGNzLmdldE1heFZhbHVlKGkpICE9IDEuMGYpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGRvbm90U3VwcG9ydFVubm9ybWFsaXplZCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKGhhc0FscGhhKSB7Ci0gICAgICAgICAgICAgICAgICAgIG1heFZhbHVlc1tudW1Db2xvckNvbXBvbmVudHNdID0gKDEgPDwgYml0c1tudW1Db2xvckNvbXBvbmVudHNdKSAtIDE7Ci0gICAgICAgICAgICAgICAgICAgIHNjYWxlRmFjdG9yc1tudW1Db2xvckNvbXBvbmVudHNdID0gMS4wZiAvIG1heFZhbHVlc1tudW1Db2xvckNvbXBvbmVudHNdOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1NIT1JUOgotICAgICAgICAgICAgICAgIHNpZ25lZCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgaW50ZWdyYWwgPSB0cnVlOwotICAgICAgICAgICAgICAgIGRvbm90U3VwcG9ydFVubm9ybWFsaXplZCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgc2NhbGVGYWN0b3JzID0gbmV3IGZsb2F0W251bUNvbXBvbmVudHNdOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIG1heFZhbHVlc1tpXSA9IFNob3J0Lk1BWF9WQUxVRTsKLSAgICAgICAgICAgICAgICAgICAgc2NhbGVGYWN0b3JzW2ldID0gMS4wZiAvIG1heFZhbHVlc1tpXTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGNzLmdldE1pblZhbHVlKGkpICE9IDAuMGYgfHwgY3MuZ2V0TWF4VmFsdWUoaSkgIT0gMS4wZikgewotICAgICAgICAgICAgICAgICAgICAgICAgbmVlZFNjYWxlID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAobmVlZFNjYWxlKSB7Ci0gICAgICAgICAgICAgICAgICAgIG1pblZhbHMgPSBuZXcgZmxvYXRbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgcmFuZ2VzID0gbmV3IGZsb2F0W251bUNvbG9yQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQ29sb3JDb21wb25lbnRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIG1pblZhbHNbaV0gPSBjcy5nZXRNaW5WYWx1ZShpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJhbmdlc1tpXSA9IGNzLmdldE1heFZhbHVlKGkpIC0gbWluVmFsc1tpXTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FUOgotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFOgotICAgICAgICAgICAgICAgIHNpZ25lZCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgaW50ZWdyYWwgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICBkb25vdFN1cHBvcnRVbm5vcm1hbGl6ZWQgPSB0cnVlOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjE1PXRyYW5zZmVyVHlwZSBpcyBub3Qgb25lIG9mIERhdGFCdWZmZXIuVFlQRV9CWVRFLAotICAgICAgICAgICAgICAgIC8vIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQsIERhdGFCdWZmZXIuVFlQRV9JTlQsCi0gICAgICAgICAgICAgICAgLy8gRGF0YUJ1ZmZlci5UWVBFX1NIT1JULCBEYXRhQnVmZmVyLlRZUEVfRkxPQVQsIG9yCi0gICAgICAgICAgICAgICAgLy8gRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjE1IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBuZWVkQWxwaGFEaXZpZGUgPSBoYXNBbHBoYSAmJiBpc0FscGhhUHJlbXVsdGlwbGllZDsKLSAgICAgICAgaW5pdExVVHMoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgY29tcG9uZW50IGNvbG9yIG1vZGVsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvclNwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgY29sb3Igc3BhY2UuCi0gICAgICogQHBhcmFtIGhhc0FscGhhCi0gICAgICogICAgICAgICAgICB3aGV0aGVyIHRoZSBjb2xvciBtb2RlbCBoYXMgYWxwaGEuCi0gICAgICogQHBhcmFtIGlzQWxwaGFQcmVtdWx0aXBsaWVkCi0gICAgICogICAgICAgICAgICB3aGV0aGVyIHRoZSBhbHBoYSBpcyBwcmUtbXVsdGlwbGllZC4KLSAgICAgKiBAcGFyYW0gdHJhbnNwYXJlbmN5Ci0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNwYXJlbmN5IHN0cmF0ZWd5LCBAc2VlIGphdmEuYXd0LlRyYW5zcGFyZW5jeS4KLSAgICAgKiBAcGFyYW0gdHJhbnNmZXJUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNmZXIgdHlwZSAocHJpbWl0aXZlIGphdmEgdHlwZSB0byB1c2UgZm9yIHRoZQotICAgICAqICAgICAgICAgICAgY29tcG9uZW50cykuCi0gICAgICovCi0gICAgcHVibGljIENvbXBvbmVudENvbG9yTW9kZWwoQ29sb3JTcGFjZSBjb2xvclNwYWNlLCBib29sZWFuIGhhc0FscGhhLAotICAgICAgICAgICAgYm9vbGVhbiBpc0FscGhhUHJlbXVsdGlwbGllZCwgaW50IHRyYW5zcGFyZW5jeSwgaW50IHRyYW5zZmVyVHlwZSkgewotCi0gICAgICAgIHRoaXMoY29sb3JTcGFjZSwgY3JlYXRlUGl4ZWxCaXRzQXJyYXkoY29sb3JTcGFjZSwgaGFzQWxwaGEsIHRyYW5zZmVyVHlwZSksIGhhc0FscGhhLAotICAgICAgICAgICAgICAgIGlzQWxwaGFQcmVtdWx0aXBsaWVkLCB0cmFuc3BhcmVuY3ksIHRyYW5zZmVyVHlwZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVmFsaWRhdGUgYml0cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYml0cwotICAgICAqICAgICAgICAgICAgdGhlIGJpdHMuCi0gICAgICogQHBhcmFtIGNvbG9yU3BhY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb2xvciBzcGFjZS4KLSAgICAgKiBAcGFyYW0gaGFzQWxwaGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBoYXMgYWxwaGEuCi0gICAgICogQHBhcmFtIHRyYW5zZmVyVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zZmVyIHR5cGUuCi0gICAgICogQHJldHVybiB0aGUgaW50W10uCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgaW50W10gdmFsaWRhdGVCaXRzKGludCBiaXRzW10sIENvbG9yU3BhY2UgY29sb3JTcGFjZSwgYm9vbGVhbiBoYXNBbHBoYSwKLSAgICAgICAgICAgIGludCB0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgaWYgKGJpdHMgIT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIGJpdHM7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgbnVtQ29tcG9uZW50cyA9IGNvbG9yU3BhY2UuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICBpZiAoaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIG51bUNvbXBvbmVudHMrKzsKLSAgICAgICAgfQotICAgICAgICBiaXRzID0gbmV3IGludFtudW1Db21wb25lbnRzXTsKLQotICAgICAgICBpbnQgY29tcG9uZW50TGVuZ3RoID0gRGF0YUJ1ZmZlci5nZXREYXRhVHlwZVNpemUodHJhbnNmZXJUeXBlKTsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgYml0c1tpXSA9IGNvbXBvbmVudExlbmd0aDsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBiaXRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIHBpeGVsIGJpdHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbG9yU3BhY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb2xvciBzcGFjZS4KLSAgICAgKiBAcGFyYW0gaGFzQWxwaGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBoYXMgYWxwaGEuCi0gICAgICogQHBhcmFtIHRyYW5zZmVyVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zZmVyIHR5cGUuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGludCBjcmVhdGVQaXhlbEJpdHMoQ29sb3JTcGFjZSBjb2xvclNwYWNlLCBib29sZWFuIGhhc0FscGhhLCBpbnQgdHJhbnNmZXJUeXBlKSB7Ci0gICAgICAgIGludCBudW1Db21wb25lbnRzID0gY29sb3JTcGFjZS5nZXROdW1Db21wb25lbnRzKCk7Ci0gICAgICAgIGlmIChoYXNBbHBoYSkgewotICAgICAgICAgICAgbnVtQ29tcG9uZW50cysrOwotICAgICAgICB9Ci0gICAgICAgIGludCBjb21wb25lbnRMZW5ndGggPSBEYXRhQnVmZmVyLmdldERhdGFUeXBlU2l6ZSh0cmFuc2ZlclR5cGUpOwotICAgICAgICByZXR1cm4gbnVtQ29tcG9uZW50cyAqIGNvbXBvbmVudExlbmd0aDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBwaXhlbCBiaXRzIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvclNwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgY29sb3Igc3BhY2UuCi0gICAgICogQHBhcmFtIGhhc0FscGhhCi0gICAgICogICAgICAgICAgICB0aGUgaGFzIGFscGhhLgotICAgICAqIEBwYXJhbSB0cmFuc2ZlclR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSB0cmFuc2ZlciB0eXBlLgotICAgICAqIEByZXR1cm4gdGhlIGludFtdLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGludFtdIGNyZWF0ZVBpeGVsQml0c0FycmF5KENvbG9yU3BhY2UgY29sb3JTcGFjZSwgYm9vbGVhbiBoYXNBbHBoYSwKLSAgICAgICAgICAgIGludCB0cmFuc2ZlclR5cGUpIHsKLQotICAgICAgICBpbnQgbnVtQ29tcG9uZW50cyA9IGNvbG9yU3BhY2UuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICBpZiAoaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIG51bUNvbXBvbmVudHMrKzsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBiaXRzW10gPSBuZXcgaW50W251bUNvbXBvbmVudHNdOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgYml0c1tpXSA9IERhdGFCdWZmZXIuZ2V0RGF0YVR5cGVTaXplKHRyYW5zZmVyVHlwZSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGJpdHM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBnZXREYXRhRWxlbWVudHMoaW50IGNvbXBvbmVudHNbXSwgaW50IG9mZnNldCwgT2JqZWN0IG9iaikgewotICAgICAgICBpZiAoZG9ub3RTdXBwb3J0VW5ub3JtYWxpemVkKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjEzPVRoaXMgQ29tcG9uZW50Q29sb3JNb2RlbCBkb2VzIG5vdCBzdXBwb3J0IHRoZQotICAgICAgICAgICAgLy8gdW5ub3JtYWxpemVkIGZvcm0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjEzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAob2Zmc2V0ICsgbnVtQ29tcG9uZW50cyA+IGNvbXBvbmVudHMubGVuZ3RoKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjE2PVRoZSBjb21wb25lbnRzIGFycmF5IGlzIG5vdCBsYXJnZSBlbm91Z2ggdG8gaG9sZCBhbGwgdGhlCi0gICAgICAgICAgICAvLyBjb2xvciBhbmQgYWxwaGEgY29tcG9uZW50cwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHN3aXRjaCAodHJhbnNmZXJUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGJ5dGUgYmFbXTsKLSAgICAgICAgICAgICAgICBpZiAob2JqID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgYmEgPSBuZXcgYnl0ZVtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBiYSA9IChieXRlW10pb2JqOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gb2Zmc2V0OyBpIDwgbnVtQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICBiYVtpXSA9IChieXRlKWNvbXBvbmVudHNbaWR4XTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGJhOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNhID0gbmV3IHNob3J0W251bUNvbXBvbmVudHNdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHNhID0gKHNob3J0W10pb2JqOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gb2Zmc2V0OyBpIDwgbnVtQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICBzYVtpXSA9IChzaG9ydCljb21wb25lbnRzW2lkeF07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBzYTsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaWFbXTsKLSAgICAgICAgICAgICAgICBpZiAob2JqID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgaWEgPSBuZXcgaW50W251bUNvbXBvbmVudHNdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGlhID0gKGludFtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDAsIGlkeCA9IG9mZnNldDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgaWFbaV0gPSBjb21wb25lbnRzW2lkeF07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBpYTsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjIxNz1UaGUgdHJhbnNmZXIgdHlwZSBvZiB0aGlzIENvbXBvbmVudENvbG9yTW9kZWwgaXMgbm90Ci0gICAgICAgICAgICAgICAgLy8gb25lCi0gICAgICAgICAgICAgICAgLy8gb2YgdGhlIGZvbGxvd2luZyB0cmFuc2ZlciB0eXBlczogRGF0YUJ1ZmZlci5UWVBFX0JZVEUsCi0gICAgICAgICAgICAgICAgLy8gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCwgb3IgRGF0YUJ1ZmZlci5UWVBFX0lOVAotICAgICAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgZ2V0RGF0YUVsZW1lbnRzKGZsb2F0IG5vcm1Db21wb25lbnRzW10sIGludCBub3JtT2Zmc2V0LCBPYmplY3Qgb2JqKSB7Ci0gICAgICAgIGlmIChuZWVkU2NhbGUpIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSAwOyBpIDwgbnVtQ29sb3JDb21wb25lbnRzOyBpKyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXBvbmVudHNbaWR4XSA9IChub3JtQ29tcG9uZW50c1tpZHhdIC0gbWluVmFsc1tpXSkgLyByYW5nZXNbaV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJhID0gbmV3IGJ5dGVbbnVtQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgYmEgPSAoYnl0ZVtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBpZiAobmVlZEFscGhhRGl2aWRlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZsb2F0IGFscGhhID0gbm9ybUNvbXBvbmVudHNbbm9ybU9mZnNldCArIG51bUNvbG9yQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSBub3JtT2Zmc2V0OyBpIDwgbnVtQ29sb3JDb21wb25lbnRzOyBpKyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBiYVtpXSA9IChieXRlKShub3JtQ29tcG9uZW50c1tpZHhdICogYWxwaGEgKiBtYXhWYWx1ZXNbaV0gKyAwLjVmKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBiYVtudW1Db2xvckNvbXBvbmVudHNdID0gKGJ5dGUpKG5vcm1Db21wb25lbnRzW25vcm1PZmZzZXQgKyBudW1Db2xvckNvbXBvbmVudHNdCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKiBtYXhWYWx1ZXNbbnVtQ29sb3JDb21wb25lbnRzXSArIDAuNWYpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSBub3JtT2Zmc2V0OyBpIDwgbnVtQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgYmFbaWR4XSA9IChieXRlKShub3JtQ29tcG9uZW50c1tpZHhdICogbWF4VmFsdWVzW2ldICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGJhOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgc2hvcnQgdXNhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHVzYSA9IG5ldyBzaG9ydFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICB1c2EgPSAoc2hvcnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgaWYgKG5lZWRBbHBoYURpdmlkZSkgewotICAgICAgICAgICAgICAgICAgICBmbG9hdCBhbHBoYSA9IG5vcm1Db21wb25lbnRzW25vcm1PZmZzZXQgKyBudW1Db2xvckNvbXBvbmVudHNdOwotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gMDsgaSA8IG51bUNvbG9yQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgdXNhW2ldID0gKHNob3J0KShub3JtQ29tcG9uZW50c1tpZHhdICogYWxwaGEgKiBtYXhWYWx1ZXNbaV0gKyAwLjVmKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB1c2FbbnVtQ29sb3JDb21wb25lbnRzXSA9IChzaG9ydCkoYWxwaGEgKiBtYXhWYWx1ZXNbbnVtQ29sb3JDb21wb25lbnRzXSArIDAuNWYpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSBub3JtT2Zmc2V0OyBpIDwgbnVtQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgdXNhW2ldID0gKHNob3J0KShub3JtQ29tcG9uZW50c1tpZHhdICogbWF4VmFsdWVzW2ldICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHVzYTsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGludCBpYVtdOwotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBpYSA9IG5ldyBpbnRbbnVtQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgaWEgPSAoaW50W10pb2JqOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChuZWVkQWxwaGFEaXZpZGUpIHsKLSAgICAgICAgICAgICAgICAgICAgZmxvYXQgYWxwaGEgPSBub3JtQ29tcG9uZW50c1tub3JtT2Zmc2V0ICsgbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDAsIGlkeCA9IDA7IGkgPCBudW1Db2xvckNvbXBvbmVudHM7IGkrKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlhW2ldID0gKGludCkobm9ybUNvbXBvbmVudHNbaWR4XSAqIGFscGhhICogbWF4VmFsdWVzW2ldICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWFbbnVtQ29sb3JDb21wb25lbnRzXSA9IChpbnQpKGFscGhhICogbWF4VmFsdWVzW251bUNvbG9yQ29tcG9uZW50c10gKyAwLjVmKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gbm9ybU9mZnNldDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlhW2ldID0gKGludCkobm9ybUNvbXBvbmVudHNbaWR4XSAqIG1heFZhbHVlc1tpXSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBpYTsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfU0hPUlQ6Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2FbXTsKLSAgICAgICAgICAgICAgICBpZiAob2JqID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgc2EgPSBuZXcgc2hvcnRbbnVtQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgc2EgPSAoc2hvcnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgaWYgKG5lZWRBbHBoYURpdmlkZSkgewotICAgICAgICAgICAgICAgICAgICBmbG9hdCBhbHBoYSA9IG5vcm1Db21wb25lbnRzW25vcm1PZmZzZXQgKyBudW1Db2xvckNvbXBvbmVudHNdOwotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gMDsgaSA8IG51bUNvbG9yQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgc2FbaV0gPSAoc2hvcnQpKG5vcm1Db21wb25lbnRzW2lkeF0gKiBhbHBoYSAqIG1heFZhbHVlc1tpXSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHNhW251bUNvbG9yQ29tcG9uZW50c10gPSAoc2hvcnQpKGFscGhhICogbWF4VmFsdWVzW251bUNvbG9yQ29tcG9uZW50c10gKyAwLjVmKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gbm9ybU9mZnNldDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHNhW2ldID0gKHNob3J0KShub3JtQ29tcG9uZW50c1tpZHhdICogbWF4VmFsdWVzW2ldICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHNhOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9GTE9BVDoKLSAgICAgICAgICAgICAgICBmbG9hdCBmYVtdOwotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBmYSA9IG5ldyBmbG9hdFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBmYSA9IChmbG9hdFtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBpZiAobmVlZEFscGhhRGl2aWRlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZsb2F0IGFscGhhID0gbm9ybUNvbXBvbmVudHNbbm9ybU9mZnNldCArIG51bUNvbG9yQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSAwOyBpIDwgbnVtQ29sb3JDb21wb25lbnRzOyBpKyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmYVtpXSA9IG5vcm1Db21wb25lbnRzW2lkeF0gKiBhbHBoYTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBmYVtudW1Db2xvckNvbXBvbmVudHNdID0gYWxwaGE7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDAsIGlkeCA9IG5vcm1PZmZzZXQ7IGkgPCBudW1Db21wb25lbnRzOyBpKyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmYVtpXSA9IG5vcm1Db21wb25lbnRzW2lkeF07Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9ET1VCTEU6Ci0gICAgICAgICAgICAgICAgZG91YmxlIGRhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGRhID0gbmV3IGRvdWJsZVtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBkYSA9IChkb3VibGVbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgaWYgKG5lZWRBbHBoYURpdmlkZSkgewotICAgICAgICAgICAgICAgICAgICBkb3VibGUgYWxwaGEgPSBub3JtQ29tcG9uZW50c1tub3JtT2Zmc2V0ICsgbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDAsIGlkeCA9IDA7IGkgPCBudW1Db2xvckNvbXBvbmVudHM7IGkrKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGRhW2ldID0gbm9ybUNvbXBvbmVudHNbaWR4XSAqIGFscGhhOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGRhW251bUNvbG9yQ29tcG9uZW50c10gPSBhbHBoYTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gbm9ybU9mZnNldDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGRhW2ldID0gbm9ybUNvbXBvbmVudHNbaWR4XTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gZGE7Ci0KLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjIxMz1UaGlzIENvbXBvbmVudENvbG9yTW9kZWwgZG9lcyBub3Qgc3VwcG9ydCB0aGUKLSAgICAgICAgICAgICAgICAvLyB1bm5vcm1hbGl6ZWQgZm9ybQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjEzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgT2JqZWN0IGdldERhdGFFbGVtZW50cyhpbnQgcmdiLCBPYmplY3QgcGl4ZWwpIHsKLSAgICAgICAgZmxvYXQgbm9ybUNvbXBbXTsKLSAgICAgICAgZmxvYXQgY29tcFtdOwotCi0gICAgICAgIGludCByZWQgPSAocmdiID4+IDE2KSAmIDB4ZmY7Ci0gICAgICAgIGludCBncmVlbiA9IChyZ2IgPj4gOCkgJiAweGZmOwotICAgICAgICBpbnQgYmx1ZSA9IHJnYiAmIDB4ZmY7Ci0gICAgICAgIGludCBhbHBoYSA9IChyZ2IgPj4gMjQpICYgMHhmZjsKLQotICAgICAgICBjb21wID0gbmV3IGZsb2F0WzNdOwotICAgICAgICBpZiAoaXNfc1JHQiB8fCBpc19MSU5FQVJfUkdCKSB7Ci0gICAgICAgICAgICBpZiAoaXNfTElORUFSX1JHQikgewotICAgICAgICAgICAgICAgIGlmIChMSU5FQVJfUkdCX0xlbmd0aCA9PSA4KSB7Ci0gICAgICAgICAgICAgICAgICAgIHJlZCA9IHRvX0xJTkVBUl84UkdCX0xVVFtyZWRdICYgMHhmZjsKLSAgICAgICAgICAgICAgICAgICAgZ3JlZW4gPSB0b19MSU5FQVJfOFJHQl9MVVRbZ3JlZW5dICYgMHhmZjsKLSAgICAgICAgICAgICAgICAgICAgYmx1ZSA9IHRvX0xJTkVBUl84UkdCX0xVVFtibHVlXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgcmVkID0gdG9fTElORUFSXzE2UkdCX0xVVFtyZWRdICYgMHhmZmZmOwotICAgICAgICAgICAgICAgICAgICBncmVlbiA9IHRvX0xJTkVBUl8xNlJHQl9MVVRbZ3JlZW5dICYgMHhmZmZmOwotICAgICAgICAgICAgICAgICAgICBibHVlID0gdG9fTElORUFSXzE2UkdCX0xVVFtibHVlXSAmIDB4ZmZmZjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjb21wWzBdID0gcmVkIC8gZkZhY3RvcjsKLSAgICAgICAgICAgIGNvbXBbMV0gPSBncmVlbiAvIGZGYWN0b3I7Ci0gICAgICAgICAgICBjb21wWzJdID0gYmx1ZSAvIGZGYWN0b3I7Ci0gICAgICAgICAgICBpZiAoIWhhc0FscGhhKSB7Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXAgPSBjb21wOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBmbG9hdCBub3JtQWxwaGEgPSBhbHBoYSAvIDI1NS4wZjsKLSAgICAgICAgICAgICAgICBub3JtQ29tcCA9IG5ldyBmbG9hdFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbG9yQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIG5vcm1Db21wW2ldID0gY29tcFtpXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXBbbnVtQ29sb3JDb21wb25lbnRzXSA9IG5vcm1BbHBoYTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbXBbMF0gPSByZWQgLyBmRmFjdG9yOwotICAgICAgICAgICAgY29tcFsxXSA9IGdyZWVuIC8gZkZhY3RvcjsKLSAgICAgICAgICAgIGNvbXBbMl0gPSBibHVlIC8gZkZhY3RvcjsKLSAgICAgICAgICAgIGZsb2F0W10gZGVmQ29tcCA9IGNzLmZyb21SR0IoY29tcCk7Ci0gICAgICAgICAgICBpZiAoIWhhc0FscGhhKSB7Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXAgPSBkZWZDb21wOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBmbG9hdCBub3JtQWxwaGEgPSBhbHBoYSAvIDI1NS4wZjsKLSAgICAgICAgICAgICAgICBub3JtQ29tcCA9IG5ldyBmbG9hdFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbG9yQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIG5vcm1Db21wW2ldID0gZGVmQ29tcFtpXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXBbbnVtQ29sb3JDb21wb25lbnRzXSA9IG5vcm1BbHBoYTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBpZiAoaGFzQWxwaGEgJiYgaXNBbHBoYVByZW11bHRpcGxpZWQpIHsKLSAgICAgICAgICAgIG5vcm1Db21wWzBdICo9IG5vcm1Db21wW251bUNvbG9yQ29tcG9uZW50c107Ci0gICAgICAgICAgICBub3JtQ29tcFsxXSAqPSBub3JtQ29tcFtudW1Db2xvckNvbXBvbmVudHNdOwotICAgICAgICAgICAgbm9ybUNvbXBbMl0gKj0gbm9ybUNvbXBbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBnZXREYXRhRWxlbWVudHMobm9ybUNvbXAsIDAsIHBpeGVsKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgZ2V0QWxwaGFSYXN0ZXIoV3JpdGFibGVSYXN0ZXIgcmFzdGVyKSB7Ci0gICAgICAgIGlmICghaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHggPSByYXN0ZXIuZ2V0TWluWCgpOwotICAgICAgICBpbnQgeSA9IHJhc3Rlci5nZXRNaW5ZKCk7Ci0gICAgICAgIGludCBiYW5kTGlzdFtdID0gbmV3IGludFsxXTsKLSAgICAgICAgYmFuZExpc3RbMF0gPSByYXN0ZXIuZ2V0TnVtQmFuZHMoKSAtIDE7Ci0KLSAgICAgICAgcmV0dXJuIHJhc3Rlci5jcmVhdGVXcml0YWJsZUNoaWxkKHgsIHksIHJhc3Rlci5nZXRXaWR0aCgpLCByYXN0ZXIuZ2V0SGVpZ2h0KCksIHgsIHksCi0gICAgICAgICAgICAgICAgYmFuZExpc3QpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBDb2xvck1vZGVsIGNvZXJjZURhdGEoV3JpdGFibGVSYXN0ZXIgcmFzdGVyLCBib29sZWFuIGlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0gICAgICAgIGlmICghaGFzQWxwaGEgfHwgdGhpcy5pc0FscGhhUHJlbXVsdGlwbGllZCA9PSBpc0FscGhhUHJlbXVsdGlwbGllZCkgewotICAgICAgICAgICAgcmV0dXJuIHRoaXM7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgbWluWCA9IHJhc3Rlci5nZXRNaW5YKCk7Ci0gICAgICAgIGludCBtaW5ZID0gcmFzdGVyLmdldE1pblkoKTsKLSAgICAgICAgaW50IHcgPSByYXN0ZXIuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IGggPSByYXN0ZXIuZ2V0SGVpZ2h0KCk7Ci0KLSAgICAgICAgaWYgKGlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0gICAgICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICAgICAgZmxvYXQgYWxwaGFGYWN0b3IgPSBtYXhWYWx1ZXNbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGlDb21wb25lbnRzW10gPSBudWxsOwotICAgICAgICAgICAgICAgICAgICBpbnQgaVRyYW5zcGFyZW50Q29tcG9uZW50c1tdID0gbmV3IGludFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBoOyBpKyssIG1pblkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDAsIHggPSBtaW5YOyBqIDwgdzsgaisrLCB4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpQ29tcG9uZW50cyA9IHJhc3Rlci5nZXRQaXhlbCh4LCBtaW5ZLCBpQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGlDb21wb25lbnRzW251bUNvbG9yQ29tcG9uZW50c10gPT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXN0ZXIuc2V0UGl4ZWwoeCwgbWluWSwgaVRyYW5zcGFyZW50Q29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmxvYXQgYWxwaGEgPSBpQ29tcG9uZW50c1tudW1Db2xvckNvbXBvbmVudHNdIC8gYWxwaGFGYWN0b3I7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtQ29sb3JDb21wb25lbnRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlDb21wb25lbnRzW25dID0gKGludCkoYWxwaGEgKiBpQ29tcG9uZW50c1tuXSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXRQaXhlbCh4LCBtaW5ZLCBpQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9TSE9SVDoKLSAgICAgICAgICAgICAgICAgICAgZmxvYXQgc0FscGhhRmFjdG9yID0gbWF4VmFsdWVzW251bUNvbG9yQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgICAgIHNob3J0IHNDb21wb25lbnRzW10gPSBudWxsOwotICAgICAgICAgICAgICAgICAgICBzaG9ydCBzVHJhbnNwYXJlbnRDb21wb25lbnRzW10gPSBuZXcgc2hvcnRbbnVtQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgaDsgaSsrLCBtaW5ZKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSAwLCB4ID0gbWluWDsgaiA8IHc7IGorKywgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgc0NvbXBvbmVudHMgPSAoc2hvcnRbXSlyYXN0ZXIuZ2V0RGF0YUVsZW1lbnRzKHgsIG1pblksIHNDb21wb25lbnRzKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoc0NvbXBvbmVudHNbbnVtQ29sb3JDb21wb25lbnRzXSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXREYXRhRWxlbWVudHMoeCwgbWluWSwgc1RyYW5zcGFyZW50Q29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmxvYXQgYWxwaGEgPSBzQ29tcG9uZW50c1tudW1Db2xvckNvbXBvbmVudHNdIC8gc0FscGhhRmFjdG9yOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBuID0gMDsgbiA8IG51bUNvbG9yQ29tcG9uZW50czsgbisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzQ29tcG9uZW50c1tuXSA9IChieXRlKShhbHBoYSAqIHNDb21wb25lbnRzW25dICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmFzdGVyLnNldERhdGFFbGVtZW50cyh4LCBtaW5ZLCBzQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9GTE9BVDoKLSAgICAgICAgICAgICAgICAgICAgZmxvYXQgZkNvbXBvbmVudHNbXSA9IG51bGw7Ci0gICAgICAgICAgICAgICAgICAgIGZsb2F0IGZUcmFuc3BhcmVudENvbXBvbmVudHNbXSA9IG5ldyBmbG9hdFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBoOyBpKyssIG1pblkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDAsIHggPSBtaW5YOyBqIDwgdzsgaisrLCB4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBmQ29tcG9uZW50cyA9IHJhc3Rlci5nZXRQaXhlbCh4LCBtaW5ZLCBmQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGZDb21wb25lbnRzW251bUNvbG9yQ29tcG9uZW50c10gPT0gMC4wZikgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXN0ZXIuc2V0RGF0YUVsZW1lbnRzKHgsIG1pblksIGZUcmFuc3BhcmVudENvbXBvbmVudHMpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZsb2F0IGFscGhhID0gZkNvbXBvbmVudHNbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1Db2xvckNvbXBvbmVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZkNvbXBvbmVudHNbbl0gPSBmQ29tcG9uZW50c1tuXSAqIGFscGhhOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXRQaXhlbCh4LCBtaW5ZLCBmQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9ET1VCTEU6Ci0gICAgICAgICAgICAgICAgICAgIGRvdWJsZSBkQ29tcG9uZW50c1tdID0gbnVsbDsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIGRUcmFuc3BhcmVudENvbXBvbmVudHNbXSA9IG5ldyBkb3VibGVbbnVtQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgaDsgaSsrLCBtaW5ZKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSAwLCB4ID0gbWluWDsgaiA8IHc7IGorKywgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZENvbXBvbmVudHMgPSByYXN0ZXIuZ2V0UGl4ZWwoeCwgbWluWSwgZENvbXBvbmVudHMpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChkQ29tcG9uZW50c1tudW1Db2xvckNvbXBvbmVudHNdID09IDAuMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXN0ZXIuc2V0UGl4ZWwoeCwgbWluWSwgZFRyYW5zcGFyZW50Q29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG91YmxlIGFscGhhID0gZENvbXBvbmVudHNbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1Db2xvckNvbXBvbmVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZENvbXBvbmVudHNbbl0gPSBkQ29tcG9uZW50c1tuXSAqIGFscGhhOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXRQaXhlbCh4LCBtaW5ZLCBkQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgICAgICAvLyBhd3QuMjE5PVRoaXMgdHJhbnNmZXJUeXBlIGlzIG5vdCBzdXBwb3J0ZWQgYnkgdGhpcyBjb2xvcgotICAgICAgICAgICAgICAgICAgICAvLyBtb2RlbAotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjE5IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICAgICAgZmxvYXQgYWxwaGFGYWN0b3IgPSBtYXhWYWx1ZXNbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGlDb21wb25lbnRzW10gPSBudWxsOwotICAgICAgICAgICAgICAgICAgICBpbnQgaVRyYW5zcGFyZW50Q29tcG9uZW50c1tdID0gbmV3IGludFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBoOyBpKyssIG1pblkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDAsIHggPSBtaW5YOyBqIDwgdzsgaisrLCB4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpQ29tcG9uZW50cyA9IHJhc3Rlci5nZXRQaXhlbCh4LCBtaW5ZLCBpQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGlDb21wb25lbnRzW251bUNvbG9yQ29tcG9uZW50c10gPT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXN0ZXIuc2V0UGl4ZWwoeCwgbWluWSwgaVRyYW5zcGFyZW50Q29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmxvYXQgYWxwaGEgPSBpQ29tcG9uZW50c1tudW1Db2xvckNvbXBvbmVudHNdIC8gYWxwaGFGYWN0b3I7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtQ29sb3JDb21wb25lbnRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlDb21wb25lbnRzW25dID0gKGludCkoaUNvbXBvbmVudHNbbl0gLyBhbHBoYSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXRQaXhlbCh4LCBtaW5ZLCBpQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9TSE9SVDoKLSAgICAgICAgICAgICAgICAgICAgZmxvYXQgc0FscGhhRmFjdG9yID0gbWF4VmFsdWVzW251bUNvbG9yQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgICAgIHNob3J0IHNDb21wb25lbnRzW10gPSBudWxsOwotICAgICAgICAgICAgICAgICAgICBzaG9ydCBzVHJhbnNwYXJlbnRDb21wb25lbnRzW10gPSBuZXcgc2hvcnRbbnVtQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgaDsgaSsrLCBtaW5ZKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSAwLCB4ID0gbWluWDsgaiA8IHc7IGorKywgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgc0NvbXBvbmVudHMgPSAoc2hvcnRbXSlyYXN0ZXIuZ2V0RGF0YUVsZW1lbnRzKHgsIG1pblksIHNDb21wb25lbnRzKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoc0NvbXBvbmVudHNbbnVtQ29sb3JDb21wb25lbnRzXSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXREYXRhRWxlbWVudHMoeCwgbWluWSwgc1RyYW5zcGFyZW50Q29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmxvYXQgYWxwaGEgPSBzQ29tcG9uZW50c1tudW1Db2xvckNvbXBvbmVudHNdIC8gc0FscGhhRmFjdG9yOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBuID0gMDsgbiA8IG51bUNvbG9yQ29tcG9uZW50czsgbisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzQ29tcG9uZW50c1tuXSA9IChieXRlKShzQ29tcG9uZW50c1tuXSAvIGFscGhhICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmFzdGVyLnNldERhdGFFbGVtZW50cyh4LCBtaW5ZLCBzQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9GTE9BVDoKLSAgICAgICAgICAgICAgICAgICAgZmxvYXQgZkNvbXBvbmVudHNbXSA9IG51bGw7Ci0gICAgICAgICAgICAgICAgICAgIGZsb2F0IGZUcmFuc3BhcmVudENvbXBvbmVudHNbXSA9IG5ldyBmbG9hdFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBoOyBpKyssIG1pblkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDAsIHggPSBtaW5YOyBqIDwgdzsgaisrLCB4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBmQ29tcG9uZW50cyA9IHJhc3Rlci5nZXRQaXhlbCh4LCBtaW5ZLCBmQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGZDb21wb25lbnRzW251bUNvbG9yQ29tcG9uZW50c10gPT0gMC4wZikgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXN0ZXIuc2V0RGF0YUVsZW1lbnRzKHgsIG1pblksIGZUcmFuc3BhcmVudENvbXBvbmVudHMpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZsb2F0IGFscGhhID0gZkNvbXBvbmVudHNbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1Db2xvckNvbXBvbmVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZkNvbXBvbmVudHNbbl0gPSBmQ29tcG9uZW50c1tuXSAvIGFscGhhOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXRQaXhlbCh4LCBtaW5ZLCBmQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9ET1VCTEU6Ci0gICAgICAgICAgICAgICAgICAgIGRvdWJsZSBkQ29tcG9uZW50c1tdID0gbnVsbDsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIGRUcmFuc3BhcmVudENvbXBvbmVudHNbXSA9IG5ldyBkb3VibGVbbnVtQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgaDsgaSsrLCBtaW5ZKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSAwLCB4ID0gbWluWDsgaiA8IHc7IGorKywgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZENvbXBvbmVudHMgPSByYXN0ZXIuZ2V0UGl4ZWwoeCwgbWluWSwgZENvbXBvbmVudHMpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChkQ29tcG9uZW50c1tudW1Db2xvckNvbXBvbmVudHNdID09IDAuMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXN0ZXIuc2V0UGl4ZWwoeCwgbWluWSwgZFRyYW5zcGFyZW50Q29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG91YmxlIGFscGhhID0gZENvbXBvbmVudHNbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1Db2xvckNvbXBvbmVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZENvbXBvbmVudHNbbl0gPSBkQ29tcG9uZW50c1tuXSAvIGFscGhhOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXRQaXhlbCh4LCBtaW5ZLCBkQ29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjIxOT1UaGlzIHRyYW5zZmVyVHlwZSBpcyBub3Qgc3VwcG9ydGVkIGJ5IHRoaXMgY29sb3IKLSAgICAgICAgICAgICAgICAgICAgLy8gbW9kZWwKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxOSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKCFzaWduZWQpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgQ29tcG9uZW50Q29sb3JNb2RlbChjcywgYml0cywgaGFzQWxwaGEsIGlzQWxwaGFQcmVtdWx0aXBsaWVkLCB0cmFuc3BhcmVuY3ksCi0gICAgICAgICAgICAgICAgICAgIHRyYW5zZmVyVHlwZSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbmV3IENvbXBvbmVudENvbG9yTW9kZWwoY3MsIG51bGwsIGhhc0FscGhhLCBpc0FscGhhUHJlbXVsdGlwbGllZCwgdHJhbnNwYXJlbmN5LAotICAgICAgICAgICAgICAgIHRyYW5zZmVyVHlwZSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldENvbXBvbmVudHMoT2JqZWN0IHBpeGVsLCBpbnRbXSBjb21wb25lbnRzLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIGlmIChkb25vdFN1cHBvcnRVbm5vcm1hbGl6ZWQpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMTM9VGhpcyBDb21wb25lbnRDb2xvck1vZGVsIGRvZXMgbm90IHN1cHBvcnQgdGhlCi0gICAgICAgICAgICAvLyB1bm5vcm1hbGl6ZWQgZm9ybQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChjb21wb25lbnRzID09IG51bGwpIHsKLSAgICAgICAgICAgIGNvbXBvbmVudHMgPSBuZXcgaW50W29mZnNldCArIG51bUNvbXBvbmVudHNdOwotICAgICAgICB9IGVsc2UgaWYgKG9mZnNldCArIG51bUNvbXBvbmVudHMgPiBjb21wb25lbnRzLmxlbmd0aCkgewotICAgICAgICAgICAgLy8gYXd0LjIxOD1UaGUgY29tcG9uZW50cyBhcnJheSBpcyBub3QgbGFyZ2UgZW5vdWdoIHRvIGhvbGQgYWxsIHRoZQotICAgICAgICAgICAgLy8gY29sb3IgYW5kIGFscGhhIGNvbXBvbmVudHMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjE4IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJhW10gPSAoYnl0ZVtdKXBpeGVsOwotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDAsIGlkeCA9IG9mZnNldDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgY29tcG9uZW50c1tpZHhdID0gYmFbaV0gJiAweGZmOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gY29tcG9uZW50czsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNhW10gPSAoc2hvcnRbXSlwaXhlbDsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gb2Zmc2V0OyBpIDwgbnVtQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICBjb21wb25lbnRzW2lkeF0gPSBzYVtpXSAmIDB4ZmZmZjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGNvbXBvbmVudHM7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaWFbXSA9IChpbnRbXSlwaXhlbDsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gb2Zmc2V0OyBpIDwgbnVtQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICBjb21wb25lbnRzW2lkeF0gPSBpYVtpXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGNvbXBvbmVudHM7Ci0KLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjIxNz1UaGUgdHJhbnNmZXIgdHlwZSBvZiB0aGlzIENvbXBvbmVudENvbG9yTW9kZWwgaXMgbm90Ci0gICAgICAgICAgICAgICAgLy8gb25lCi0gICAgICAgICAgICAgICAgLy8gb2YgdGhlIGZvbGxvd2luZyB0cmFuc2ZlciB0eXBlczogRGF0YUJ1ZmZlci5UWVBFX0JZVEUsCi0gICAgICAgICAgICAgICAgLy8gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCwgb3IgRGF0YUJ1ZmZlci5UWVBFX0lOVAotICAgICAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZsb2F0W10gZ2V0Tm9ybWFsaXplZENvbXBvbmVudHMoT2JqZWN0IHBpeGVsLCBmbG9hdCBub3JtQ29tcG9uZW50c1tdLCBpbnQgbm9ybU9mZnNldCkgewotCi0gICAgICAgIGlmIChub3JtQ29tcG9uZW50cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICBub3JtQ29tcG9uZW50cyA9IG5ldyBmbG9hdFtudW1Db21wb25lbnRzICsgbm9ybU9mZnNldF07Ci0gICAgICAgIH0KLQotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJhW10gPSAoYnl0ZVtdKXBpeGVsOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSBub3JtT2Zmc2V0OyBpIDwgbnVtQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICBub3JtQ29tcG9uZW50c1tpZHhdID0gKGJhW2ldICYgMHhmZikgKiBzY2FsZUZhY3RvcnNbaV07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgc2hvcnQgdXNhW10gPSAoc2hvcnRbXSlwaXhlbDsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gbm9ybU9mZnNldDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgbm9ybUNvbXBvbmVudHNbaWR4XSA9ICh1c2FbaV0gJiAweGZmZmYpICogc2NhbGVGYWN0b3JzW2ldOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGludCBpYVtdID0gKGludFtdKXBpeGVsOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSBub3JtT2Zmc2V0OyBpIDwgbnVtQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICBub3JtQ29tcG9uZW50c1tpZHhdID0gaWFbaV0gKiBzY2FsZUZhY3RvcnNbaV07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9TSE9SVDoKLSAgICAgICAgICAgICAgICBzaG9ydCBzYVtdID0gKHNob3J0W10pcGl4ZWw7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDAsIGlkeCA9IG5vcm1PZmZzZXQ7IGkgPCBudW1Db21wb25lbnRzOyBpKyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIG5vcm1Db21wb25lbnRzW2lkeF0gPSBzYVtpXSAqIHNjYWxlRmFjdG9yc1tpXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FUOgotICAgICAgICAgICAgICAgIGZsb2F0IGZhW10gPSAoZmxvYXRbXSlwaXhlbDsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMCwgaWR4ID0gbm9ybU9mZnNldDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgbm9ybUNvbXBvbmVudHNbaWR4XSA9IGZhW2ldOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFOgotICAgICAgICAgICAgICAgIGRvdWJsZSBkYVtdID0gKGRvdWJsZVtdKXBpeGVsOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSBub3JtT2Zmc2V0OyBpIDwgbnVtQ29tcG9uZW50czsgaSsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICBub3JtQ29tcG9uZW50c1tpZHhdID0gKGZsb2F0KWRhW2ldOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjFBPVRoaXMgQ29tcG9uZW50Q29sb3JNb2RlbCBkb2VzIG5vdCBzdXBwb3J0IHRoaXMKLSAgICAgICAgICAgICAgICAvLyB0cmFuc2ZlclR5cGUKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxQSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG5lZWRBbHBoYURpdmlkZSkgewotICAgICAgICAgICAgZmxvYXQgYWxwaGEgPSBub3JtQ29tcG9uZW50c1tub3JtT2Zmc2V0ICsgbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSBub3JtT2Zmc2V0OyBpIDwgbnVtQ29sb3JDb21wb25lbnRzOyBpKyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXBvbmVudHNbaWR4XSAvPSBhbHBoYTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChuZWVkU2NhbGUpIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwLCBpZHggPSBub3JtT2Zmc2V0OyBpIDwgbnVtQ29sb3JDb21wb25lbnRzOyBpKyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXBvbmVudHNbaWR4XSA9IG1pblZhbHNbaV0gKyByYW5nZXNbaV0gKiBub3JtQ29tcG9uZW50c1tpZHhdOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBub3JtQ29tcG9uZW50czsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG9iaikgewotICAgICAgICBpZiAoIShvYmogaW5zdGFuY2VvZiBDb21wb25lbnRDb2xvck1vZGVsKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBzdXBlci5lcXVhbHMob2JqKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldFJlZChPYmplY3QgaW5EYXRhKSB7Ci0gICAgICAgIHJldHVybiBnZXRSR0JDb21wb25lbnQoaW5EYXRhLCAwKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldFJHQihPYmplY3QgaW5EYXRhKSB7Ci0gICAgICAgIGludCBhbHBoYSA9IGdldEFscGhhKGluRGF0YSk7Ci0gICAgICAgIGlmIChjcy5nZXRUeXBlKCkgPT0gQ29sb3JTcGFjZS5UWVBFX0dSQVkpIHsKLSAgICAgICAgICAgIGludCBncmF5ID0gZ2V0UmVkKGluRGF0YSk7Ci0gICAgICAgICAgICByZXR1cm4gKGFscGhhIDw8IDI0IHwgZ3JheSA8PCAxNiB8IGdyYXkgPDwgOCB8IGdyYXkpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiAoYWxwaGEgPDwgMjQgfCBnZXRSZWQoaW5EYXRhKSA8PCAxNiB8IGdldEdyZWVuKGluRGF0YSkgPDwgOCB8IGdldEJsdWUoaW5EYXRhKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRHcmVlbihPYmplY3QgaW5EYXRhKSB7Ci0gICAgICAgIHJldHVybiBnZXRSR0JDb21wb25lbnQoaW5EYXRhLCAxKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldEJsdWUoT2JqZWN0IGluRGF0YSkgewotICAgICAgICByZXR1cm4gZ2V0UkdCQ29tcG9uZW50KGluRGF0YSwgMik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRBbHBoYShPYmplY3QgaW5EYXRhKSB7Ci0gICAgICAgIGlmICghaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIHJldHVybiAyNTU7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IGFscGhhID0gMDsKLQotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURTogewotICAgICAgICAgICAgICAgIGJ5dGUgYmFbXSA9IChieXRlW10paW5EYXRhOwotICAgICAgICAgICAgICAgIGFscGhhID0gYmFbbnVtQ29sb3JDb21wb25lbnRzXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgaWYgKGJpdHNbbnVtQ29sb3JDb21wb25lbnRzXSAhPSA4KSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBhbHBoYUxVVFthbHBoYV0gJiAweGZmOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gYWxwaGE7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6IHsKLSAgICAgICAgICAgICAgICBzaG9ydCB1c2FbXSA9IChzaG9ydFtdKWluRGF0YTsKLSAgICAgICAgICAgICAgICBhbHBoYSA9IHVzYVtudW1Db2xvckNvbXBvbmVudHNdICYgMHhmZmZmOwotICAgICAgICAgICAgICAgIGlmIChiaXRzW251bUNvbG9yQ29tcG9uZW50c10gIT0gOCkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gYWxwaGFMVVRbYWxwaGFdICYgMHhmZjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGFscGhhOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOiB7Ci0gICAgICAgICAgICAgICAgaW50IGlhW10gPSAoaW50W10paW5EYXRhOwotICAgICAgICAgICAgICAgIGFscGhhID0gaWFbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICBpZiAoYml0c1tudW1Db2xvckNvbXBvbmVudHNdICE9IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGFscGhhTFVUW2FscGhhXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBhbHBoYTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1NIT1JUOiB7Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2FbXSA9IChzaG9ydFtdKWluRGF0YTsKLSAgICAgICAgICAgICAgICBhbHBoYSA9IHNhW251bUNvbG9yQ29tcG9uZW50c107Ci0gICAgICAgICAgICAgICAgaWYgKGJpdHNbbnVtQ29sb3JDb21wb25lbnRzXSAhPSA4KSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBhbHBoYUxVVFthbHBoYV0gJiAweGZmOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gYWxwaGE7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9GTE9BVDogewotICAgICAgICAgICAgICAgIGZsb2F0IGZhW10gPSAoZmxvYXRbXSlpbkRhdGE7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIChpbnQpKGZhW251bUNvbG9yQ29tcG9uZW50c10gKiAyNTUuMGYgKyAwLjVmKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRTogewotICAgICAgICAgICAgICAgIGRvdWJsZSBkYVtdID0gKGRvdWJsZVtdKWluRGF0YTsKLSAgICAgICAgICAgICAgICByZXR1cm4gKGludCkoZGFbbnVtQ29sb3JDb21wb25lbnRzXSAqIDI1NS4wICsgMC41KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGRlZmF1bHQ6IHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjE0PVRoaXMgQ29sb3IgTW9kZWwgZG9lc24ndCBzdXBwb3J0IHRoaXMgdHJhbnNmZXJUeXBlCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxNCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3RlcihpbnQgdywgaW50IGgpIHsKLSAgICAgICAgU2FtcGxlTW9kZWwgc20gPSBjcmVhdGVDb21wYXRpYmxlU2FtcGxlTW9kZWwodywgaCk7Ci0gICAgICAgIERhdGFCdWZmZXIgZGIgPSBzbS5jcmVhdGVEYXRhQnVmZmVyKCk7Ci0gICAgICAgIHJldHVybiBSYXN0ZXIuY3JlYXRlV3JpdGFibGVSYXN0ZXIoc20sIGRiLCBudWxsKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NvbXBhdGlibGVTYW1wbGVNb2RlbChTYW1wbGVNb2RlbCBzbSkgewotICAgICAgICBpZiAoIShzbSBpbnN0YW5jZW9mIENvbXBvbmVudFNhbXBsZU1vZGVsKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChudW1Db21wb25lbnRzICE9IHNtLmdldE51bUJhbmRzKCkpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgICAgICBpZiAodHJhbnNmZXJUeXBlICE9IHNtLmdldFRyYW5zZmVyVHlwZSgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFNhbXBsZU1vZGVsIGNyZWF0ZUNvbXBhdGlibGVTYW1wbGVNb2RlbChpbnQgdywgaW50IGgpIHsKLSAgICAgICAgaW50IGJhbmRPZmZzZXRzW10gPSBuZXcgaW50W251bUNvbXBvbmVudHNdOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgYmFuZE9mZnNldHNbaV0gPSBpOwotICAgICAgICB9Ci0KLSAgICAgICAgc3dpdGNoICh0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG5ldyBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwodHJhbnNmZXJUeXBlLCB3LCBoLCBudW1Db21wb25lbnRzLCB3Ci0gICAgICAgICAgICAgICAgICAgICAgICAqIG51bUNvbXBvbmVudHMsIGJhbmRPZmZzZXRzKTsKLQotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICByZXR1cm4gbmV3IENvbXBvbmVudFNhbXBsZU1vZGVsKHRyYW5zZmVyVHlwZSwgdywgaCwgbnVtQ29tcG9uZW50cywgdwotICAgICAgICAgICAgICAgICAgICAgICAgKiBudW1Db21wb25lbnRzLCBiYW5kT2Zmc2V0cyk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NvbXBhdGlibGVSYXN0ZXIoUmFzdGVyIHJhc3RlcikgewotICAgICAgICBTYW1wbGVNb2RlbCBzbSA9IHJhc3Rlci5nZXRTYW1wbGVNb2RlbCgpOwotICAgICAgICBpZiAoIShzbSBpbnN0YW5jZW9mIENvbXBvbmVudFNhbXBsZU1vZGVsKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHNtLmdldE51bUJhbmRzKCkgIT0gbnVtQ29tcG9uZW50cykgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChyYXN0ZXIuZ2V0VHJhbnNmZXJUeXBlKCkgIT0gdHJhbnNmZXJUeXBlKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgc2FtcGxlU2l6ZXNbXSA9IHNtLmdldFNhbXBsZVNpemUoKTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1Db21wb25lbnRzOyBpKyspIHsKLSAgICAgICAgICAgIGlmIChiaXRzW2ldICE9IHNhbXBsZVNpemVzW2ldKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdFtdIGdldE5vcm1hbGl6ZWRDb21wb25lbnRzKGludCBjb21wb25lbnRzW10sIGludCBvZmZzZXQsIGZsb2F0IG5vcm1Db21wb25lbnRzW10sCi0gICAgICAgICAgICBpbnQgbm9ybU9mZnNldCkgewotICAgICAgICBpZiAoZG9ub3RTdXBwb3J0VW5ub3JtYWxpemVkKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjEzPVRoaXMgQ29tcG9uZW50Q29sb3JNb2RlbCBkb2VzIG5vdCBzdXBwb3J0IHRoZQotICAgICAgICAgICAgLy8gdW5ub3JtYWxpemVkIGZvcm0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjEzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gc3VwZXIuZ2V0Tm9ybWFsaXplZENvbXBvbmVudHMoY29tcG9uZW50cywgb2Zmc2V0LCBub3JtQ29tcG9uZW50cywgbm9ybU9mZnNldCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXREYXRhRWxlbWVudChpbnRbXSBjb21wb25lbnRzLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIGlmIChudW1Db21wb25lbnRzID4gMSkgewotICAgICAgICAgICAgLy8gYXd0LjIxMj1UaGVyZSBpcyBtb3JlIHRoYW4gb25lIGNvbXBvbmVudCBpbiB0aGlzIENvbG9yTW9kZWwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjEyIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKGRvbm90U3VwcG9ydFVubm9ybWFsaXplZCkgewotICAgICAgICAgICAgLy8gYXd0LjIxMz1UaGlzIENvbXBvbmVudENvbG9yTW9kZWwgZG9lcyBub3Qgc3VwcG9ydCB0aGUKLSAgICAgICAgICAgIC8vIHVubm9ybWFsaXplZCBmb3JtCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxMyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBjb21wb25lbnRzW29mZnNldF07Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldFVubm9ybWFsaXplZENvbXBvbmVudHMoZmxvYXRbXSBub3JtQ29tcG9uZW50cywgaW50IG5vcm1PZmZzZXQsCi0gICAgICAgICAgICBpbnRbXSBjb21wb25lbnRzLCBpbnQgb2Zmc2V0KSB7Ci0KLSAgICAgICAgaWYgKGRvbm90U3VwcG9ydFVubm9ybWFsaXplZCkgewotICAgICAgICAgICAgLy8gYXd0LjIxMz1UaGlzIENvbXBvbmVudENvbG9yTW9kZWwgZG9lcyBub3Qgc3VwcG9ydCB0aGUKLSAgICAgICAgICAgIC8vIHVubm9ybWFsaXplZCBmb3JtCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxMyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG5vcm1Db21wb25lbnRzLmxlbmd0aCAtIG5vcm1PZmZzZXQgPCBudW1Db21wb25lbnRzKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjFCPVRoZSBsZW5ndGggb2Ygbm9ybUNvbXBvbmVudHMgbWludXMgbm9ybU9mZnNldCBpcyBsZXNzCi0gICAgICAgICAgICAvLyB0aGFuIG51bUNvbXBvbmVudHMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjFCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gc3VwZXIuZ2V0VW5ub3JtYWxpemVkQ29tcG9uZW50cyhub3JtQ29tcG9uZW50cywgbm9ybU9mZnNldCwgY29tcG9uZW50cywgb2Zmc2V0KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldERhdGFFbGVtZW50KGZsb2F0IG5vcm1Db21wb25lbnRzW10sIGludCBub3JtT2Zmc2V0KSB7Ci0gICAgICAgIGlmIChudW1Db21wb25lbnRzID4gMSkgewotICAgICAgICAgICAgLy8gYXd0LjIxMj1UaGVyZSBpcyBtb3JlIHRoYW4gb25lIGNvbXBvbmVudCBpbiB0aGlzIENvbG9yTW9kZWwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjEyIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKHNpZ25lZCkgewotICAgICAgICAgICAgLy8gYXd0LjIxMD1UaGUgY29tcG9uZW50IHZhbHVlIGZvciB0aGlzIENvbG9yTW9kZWwgaXMgc2lnbmVkCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgT2JqZWN0IHBpeGVsID0gZ2V0RGF0YUVsZW1lbnRzKG5vcm1Db21wb25lbnRzLCBub3JtT2Zmc2V0LCBudWxsKTsKLQotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJhW10gPSAoYnl0ZVtdKXBpeGVsOwotICAgICAgICAgICAgICAgIHJldHVybiBiYVswXSAmIDB4ZmY7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2FbXSA9IChzaG9ydFtdKXBpeGVsOwotICAgICAgICAgICAgICAgIHJldHVybiBzYVswXSAmIDB4ZmZmZjsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaWFbXSA9IChpbnRbXSlwaXhlbDsKLSAgICAgICAgICAgICAgICByZXR1cm4gaWFbMF07Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4yMTE9UGl4ZWwgdmFsdWVzIGZvciB0aGlzIENvbG9yTW9kZWwgYXJlIG5vdCBjb252ZW5pZW50bHkKLSAgICAgICAgICAgICAgICAvLyByZXByZXNlbnRhYmxlIGFzIGEgc2luZ2xlIGludAotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjExIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50W10gZ2V0Q29tcG9uZW50cyhpbnQgcGl4ZWwsIGludCBjb21wb25lbnRzW10sIGludCBvZmZzZXQpIHsKLSAgICAgICAgaWYgKG51bUNvbXBvbmVudHMgPiAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjEyPVRoZXJlIGlzIG1vcmUgdGhhbiBvbmUgY29tcG9uZW50IGluIHRoaXMgQ29sb3JNb2RlbAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZiAoZG9ub3RTdXBwb3J0VW5ub3JtYWxpemVkKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjEzPVRoaXMgQ29tcG9uZW50Q29sb3JNb2RlbCBkb2VzIG5vdCBzdXBwb3J0IHRoZQotICAgICAgICAgICAgLy8gdW5ub3JtYWxpemVkIGZvcm0KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjEzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoY29tcG9uZW50cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICBjb21wb25lbnRzID0gbmV3IGludFtvZmZzZXQgKyAxXTsKLSAgICAgICAgfQotCi0gICAgICAgIGNvbXBvbmVudHNbb2Zmc2V0XSA9IHBpeGVsICYgbWF4VmFsdWVzWzBdOwotICAgICAgICByZXR1cm4gY29tcG9uZW50czsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldFJlZChpbnQgcGl4ZWwpIHsKLSAgICAgICAgZmxvYXQgcmdiW10gPSB0b1JHQihwaXhlbCk7Ci0gICAgICAgIHJldHVybiAoaW50KShyZ2JbMF0gKiAyNTUuMGYgKyAwLjVmKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldFJHQihpbnQgcGl4ZWwpIHsKLSAgICAgICAgcmV0dXJuIChnZXRBbHBoYShwaXhlbCkgPDwgMjQpIHwgKGdldFJlZChwaXhlbCkgPDwgMTYpIHwgKGdldEdyZWVuKHBpeGVsKSA8PCA4KQotICAgICAgICAgICAgICAgIHwgZ2V0Qmx1ZShwaXhlbCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRHcmVlbihpbnQgcGl4ZWwpIHsKLSAgICAgICAgZmxvYXQgcmdiW10gPSB0b1JHQihwaXhlbCk7Ci0gICAgICAgIHJldHVybiAoaW50KShyZ2JbMV0gKiAyNTUuMGYgKyAwLjVmKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldEJsdWUoaW50IHBpeGVsKSB7Ci0gICAgICAgIGZsb2F0IHJnYltdID0gdG9SR0IocGl4ZWwpOwotICAgICAgICByZXR1cm4gKGludCkocmdiWzJdICogMjU1LjBmICsgMC41Zik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRBbHBoYShpbnQgcGl4ZWwpIHsKLQotICAgICAgICAvLyBUaGlzIG1ldGhvZCB0aHJvdyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24gYWNjb3JkaW5nIHRvCi0gICAgICAgIC8vIEphdmEgQVBJIFNwYWNpZmljYXRpb24KLSAgICAgICAgaWYgKHNpZ25lZCkgewotICAgICAgICAgICAgLy8gYXd0LjIxMD1UaGUgY29tcG9uZW50IHZhbHVlIGZvciB0aGlzIENvbG9yTW9kZWwgaXMgc2lnbmVkCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG51bUNvbXBvbmVudHMgPiAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjEyPVRoZXJlIGlzIG1vcmUgdGhhbiBvbmUgY29tcG9uZW50IGluIHRoaXMgQ29sb3JNb2RlbAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiAyNTU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5pdGlhbGl6YXRpb24gb2YgTG9va3VwIHRhYmxlcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgaW5pdExVVHMoKSB7Ci0gICAgICAgIGlzX3NSR0IgPSBjcy5pc0NTX3NSR0IoKTsKLSAgICAgICAgaXNfTElORUFSX1JHQiA9IChjcyA9PSBMVVRDb2xvckNvbnZlcnRlci5MSU5FQVJfUkdCX0NTKTsKLQotICAgICAgICBpZiAoaGFzQWxwaGEgJiYgYml0c1tudW1Db2xvckNvbXBvbmVudHNdICE9IDggJiYgaW50ZWdyYWwpIHsKLSAgICAgICAgICAgIGFscGhhTFVUID0gbmV3IGJ5dGVbbWF4VmFsdWVzW251bUNvbG9yQ29tcG9uZW50c10gKyAxXTsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDw9IG1heFZhbHVlc1tudW1Db2xvckNvbXBvbmVudHNdOyBpKyspIHsKLSAgICAgICAgICAgICAgICBhbHBoYUxVVFtpXSA9IChieXRlKShzY2FsZUZhY3RvcnNbbnVtQ29sb3JDb21wb25lbnRzXSAqIGkgKyAwLjVmKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChpc19MSU5FQVJfUkdCKSB7Ci0gICAgICAgICAgICBpZiAobWF4Qml0TGVuZ3RoID4gOCkgewotICAgICAgICAgICAgICAgIExJTkVBUl9SR0JfTGVuZ3RoID0gMTY7Ci0gICAgICAgICAgICAgICAgZnJvbV9MSU5FQVJfUkdCX0xVVCA9IExVVENvbG9yQ29udmVydGVyLmdldEZyb20xNmxSR0J0b3NSR0JfTFVUKCk7Ci0gICAgICAgICAgICAgICAgdG9fTElORUFSXzE2UkdCX0xVVCA9IExVVENvbG9yQ29udmVydGVyLmdldEZyb21zUkdCdG8xNmxSR0JfTFVUKCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIExJTkVBUl9SR0JfTGVuZ3RoID0gODsKLSAgICAgICAgICAgICAgICBmcm9tX0xJTkVBUl9SR0JfTFVUID0gTFVUQ29sb3JDb252ZXJ0ZXIuZ2V0RnJvbThsUkdCdG9zUkdCX0xVVCgpOwotICAgICAgICAgICAgICAgIHRvX0xJTkVBUl84UkdCX0xVVCA9IExVVENvbG9yQ29udmVydGVyLmdldEZyb21zUkdCdG84bFJHQl9MVVQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGZGYWN0b3IgPSAoKDEgPDwgTElORUFSX1JHQl9MZW5ndGgpIC0gMSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmRmFjdG9yID0gMjU1LjBmOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKCFpc0FscGhhUHJlbXVsdGlwbGllZCAmJiBpbnRlZ3JhbCkgewotICAgICAgICAgICAgY29sb3JMVVRzID0gbmV3IGJ5dGVbM11bXTsKLQotICAgICAgICAgICAgaWYgKGlzX3NSR0IpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbG9yQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChiaXRzW2ldICE9IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSAwOyBqIDwgaTsgaisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGJpdHNbaV0gPT0gYml0c1tqXSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb2xvckxVVHNbaV0gPSBjb2xvckxVVHNbal07Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbG9yTFVUc1tpXSA9IG5ldyBieXRlW21heFZhbHVlc1tpXSArIDFdOwotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDA7IGogPD0gbWF4VmFsdWVzWzBdOyBqKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb2xvckxVVHNbaV1bal0gPSAoYnl0ZSkoc2NhbGVGYWN0b3JzW2ldICogaiArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoaXNfTElORUFSX1JHQikgewotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1Db2xvckNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBpZiAoYml0c1tpXSAhPSBMSU5FQVJfUkdCX0xlbmd0aCkgewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDA7IGogPCBpOyBqKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoYml0c1tpXSA9PSBiaXRzW2pdKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG9yTFVUc1tpXSA9IGNvbG9yTFVUc1tqXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgY29sb3JMVVRzW2ldID0gbmV3IGJ5dGVbbWF4VmFsdWVzW2ldICsgMV07Ci0gICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBqID0gMDsgaiA8PSBtYXhWYWx1ZXNbMF07IGorKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBpZHg7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKExJTkVBUl9SR0JfTGVuZ3RoID09IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWR4ID0gKGludCkoc2NhbGVGYWN0b3JzW2ldICogaiArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlkeCA9IChpbnQpKHNjYWxlRmFjdG9yc1tpXSAqIGogKiAyNTcuMGYgKyAwLjVmKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sb3JMVVRzW2ldW2pdID0gZnJvbV9MSU5FQVJfUkdCX0xVVFtpZHhdOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUbyByZ2IuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsCi0gICAgICogICAgICAgICAgICB0aGUgaW50ZWdlciByZXByZXNlbnRhdGlvbiBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2Ygbm9ybWFsaXplZCBzUkdCIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmbG9hdFtdIHRvUkdCKGludCBwaXhlbCkgewotCi0gICAgICAgIC8vIFRoaXMgbWV0aG9kIHRocm93IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbiBhY2NvcmRpbmcgdG8KLSAgICAgICAgLy8gSmF2YSBBUEkgU3BhY2lmaWNhdGlvbgotICAgICAgICBpZiAoc2lnbmVkKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjEwPVRoZSBjb21wb25lbnQgdmFsdWUgZm9yIHRoaXMgQ29sb3JNb2RlbCBpcyBzaWduZWQKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjEwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAobnVtQ29tcG9uZW50cyA+IDEpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMTI9VGhlcmUgaXMgbW9yZSB0aGFuIG9uZSBjb21wb25lbnQgaW4gdGhpcyBDb2xvck1vZGVsCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxMiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgT2JqZWN0IG9iaiA9IG51bGw7Ci0KLSAgICAgICAgc3dpdGNoICh0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgYnl0ZSBiYVtdID0gbmV3IGJ5dGVbMV07Ci0gICAgICAgICAgICAgICAgYmFbMF0gPSAoYnl0ZSlwaXhlbDsKLSAgICAgICAgICAgICAgICBvYmogPSBiYTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNhW10gPSBuZXcgc2hvcnRbMV07Ci0gICAgICAgICAgICAgICAgc2FbMF0gPSAoc2hvcnQpcGl4ZWw7Ci0gICAgICAgICAgICAgICAgb2JqID0gc2E7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaWFbXSA9IG5ldyBpbnRbMV07Ci0gICAgICAgICAgICAgICAgaWFbMF0gPSBwaXhlbDsKLSAgICAgICAgICAgICAgICBvYmogPSBpYTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGNzLnRvUkdCKGdldE5vcm1hbGl6ZWRDb21wb25lbnRzKG9iaiwgbnVsbCwgMCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFJHQiBjb21wb25lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsCi0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIGlkeAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIGNvbXBvbmVudC4KLSAgICAgKiBAcmV0dXJuIHRoZSBSR0IgdmFsdWUgZnJvbSAwIHRvIDI1NSBwaXhlbCdzIGNvbXBvbmVudC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBnZXRSR0JDb21wb25lbnQoT2JqZWN0IHBpeGVsLCBpbnQgaWR4KSB7Ci0gICAgICAgIGlmIChpc19zUkdCKSB7Ci0gICAgICAgICAgICBpbnQgY29tcCA9IGdldERlZkNvbXBvbmVudChwaXhlbCwgaWR4KTsKLSAgICAgICAgICAgIGlmIChjYWxjVmFsdWUgfHwgYml0c1tpZHhdID09IDgpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gY29tcDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBjb2xvckxVVHNbaWR4XVtjb21wXSAmIDB4ZmY7Ci0gICAgICAgIH0gZWxzZSBpZiAoaXNfTElORUFSX1JHQikgewotICAgICAgICAgICAgaW50IGNvbXAgPSBnZXREZWZDb21wb25lbnQocGl4ZWwsIGlkeCk7Ci0gICAgICAgICAgICBpZiAoY2FsY1ZhbHVlIHx8IGJpdHNbaWR4XSA9PSBMSU5FQVJfUkdCX0xlbmd0aCkgewotICAgICAgICAgICAgICAgIHJldHVybiBmcm9tX0xJTkVBUl9SR0JfTFVUW2NvbXBdICYgMHhmZjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBjb2xvckxVVHNbaWR4XVtjb21wXSAmIDB4ZmY7Ci0gICAgICAgIH0KLQotICAgICAgICBmbG9hdCBub3JtQ29tcFtdID0gZ2V0Tm9ybWFsaXplZENvbXBvbmVudHMocGl4ZWwsIG51bGwsIDApOwotICAgICAgICBmbG9hdCByZ2JDb21wW10gPSBjcy50b1JHQihub3JtQ29tcCk7Ci0gICAgICAgIHJldHVybiAoaW50KShyZ2JDb21wW2lkeF0gKiAyNTUuMGYgKyAwLjVmKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkZWYgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwaXhlbAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsLgotICAgICAqIEBwYXJhbSBpZHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiBjb21wb25lbnQuCi0gICAgICogQHJldHVybiB0aGUgdGVudGF0aXZlIHZhbHVlIG9mIHRoZSBwaXhlbCBjb21wb25lbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgZ2V0RGVmQ29tcG9uZW50KE9iamVjdCBwaXhlbCwgaW50IGlkeCkgewotICAgICAgICBpbnQgY29tcCA9IDA7Ci0gICAgICAgIGNhbGNWYWx1ZSA9IGZhbHNlOwotCi0gICAgICAgIHN3aXRjaCAodHJhbnNmZXJUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGJ5dGUgYmFbXSA9IChieXRlW10pcGl4ZWw7Ci0gICAgICAgICAgICAgICAgY29tcCA9IGJhW2lkeF0gJiAweGZmOwotICAgICAgICAgICAgICAgIGlmIChuZWVkQWxwaGFEaXZpZGUpIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IGFscGhhID0gYmFbbnVtQ29sb3JDb21wb25lbnRzXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChhbHBoYSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb21wID0gMDsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZsb2F0IG5vcm1BbHBoYSA9IHNjYWxlRmFjdG9yc1tudW1Db2xvckNvbXBvbmVudHNdICogYWxwaGE7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb21wID0gKGludCkoY29tcCAqIGZGYWN0b3IgLyBub3JtQWxwaGEgKyAwLjVmKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBjYWxjVmFsdWUgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gY29tcDsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHVzYVtdID0gKHNob3J0W10pcGl4ZWw7Ci0gICAgICAgICAgICAgICAgY29tcCA9IHVzYVtpZHhdICYgMHhmZmZmOwotICAgICAgICAgICAgICAgIGlmIChuZWVkQWxwaGFEaXZpZGUpIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IGFscGhhID0gdXNhW251bUNvbG9yQ29tcG9uZW50c10gJiAweGZmZmY7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChhbHBoYSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb21wID0gMDsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZsb2F0IG5vcm1BbHBoYSA9IHNjYWxlRmFjdG9yc1tudW1Db2xvckNvbXBvbmVudHNdICogYWxwaGE7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb21wID0gKGludCkoY29tcCAqIGZGYWN0b3IgLyBub3JtQWxwaGEgKyAwLjVmKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBjYWxjVmFsdWUgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gY29tcDsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGludCBpYVtdID0gKGludFtdKXBpeGVsOwotICAgICAgICAgICAgICAgIGNvbXAgPSBpYVtpZHhdOwotICAgICAgICAgICAgICAgIGlmIChuZWVkQWxwaGFEaXZpZGUpIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IGFscGhhID0gaWFbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGFscGhhID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbXAgPSAwOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgZmxvYXQgbm9ybUFscGhhID0gc2NhbGVGYWN0b3JzW251bUNvbG9yQ29tcG9uZW50c10gKiBhbHBoYTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbXAgPSAoaW50KShjb21wICogZkZhY3RvciAvIG5vcm1BbHBoYSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGNhbGNWYWx1ZSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBjb21wOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9TSE9SVDoKLSAgICAgICAgICAgICAgICBzaG9ydCBzYVtdID0gKHNob3J0W10pcGl4ZWw7Ci0gICAgICAgICAgICAgICAgY29tcCA9IHNhW2lkeF07Ci0gICAgICAgICAgICAgICAgaWYgKG5lZWRBbHBoYURpdmlkZSkgewotICAgICAgICAgICAgICAgICAgICBpbnQgYWxwaGEgPSBzYVtudW1Db2xvckNvbXBvbmVudHNdOwotICAgICAgICAgICAgICAgICAgICBpZiAoYWxwaGEgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgY29tcCA9IDA7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmbG9hdCBub3JtQWxwaGEgPSBzY2FsZUZhY3RvcnNbbnVtQ29sb3JDb21wb25lbnRzXSAqIGFscGhhOwotICAgICAgICAgICAgICAgICAgICAgICAgY29tcCA9IChpbnQpKGNvbXAgKiBmRmFjdG9yIC8gbm9ybUFscGhhICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgY2FsY1ZhbHVlID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGNvbXA7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FUOgotICAgICAgICAgICAgICAgIGZsb2F0IGZhW10gPSAoZmxvYXRbXSlwaXhlbDsKLSAgICAgICAgICAgICAgICBpZiAobmVlZEFscGhhRGl2aWRlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZsb2F0IGFscGhhID0gZmFbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGZhW251bUNvbG9yQ29tcG9uZW50c10gPT0gMC4wZikgewotICAgICAgICAgICAgICAgICAgICAgICAgY29tcCA9IDA7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb21wID0gKGludCkoZmFbaWR4XSAqIGZGYWN0b3IgLyBhbHBoYSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgY29tcCA9IChpbnQpKGZhW2lkeF0gKiBmRmFjdG9yICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGNhbGNWYWx1ZSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGNvbXA7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRToKLSAgICAgICAgICAgICAgICBkb3VibGUgZGFbXSA9IChkb3VibGVbXSlwaXhlbDsKLSAgICAgICAgICAgICAgICBpZiAobmVlZEFscGhhRGl2aWRlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChkYVtudW1Db2xvckNvbXBvbmVudHNdID09IDAuMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgY29tcCA9IDA7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb21wID0gKGludCkoZGFbaWR4XSAqIGZGYWN0b3IgLyBkYVtudW1Db2xvckNvbXBvbmVudHNdICsgMC41KTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGNvbXAgPSAoaW50KShkYVtpZHhdICogZkZhY3RvciArIDAuNSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGNhbGNWYWx1ZSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGNvbXA7Ci0KLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjIxND1UaGlzIENvbG9yIE1vZGVsIGRvZXNuJ3Qgc3VwcG9ydCB0aGlzIHRyYW5zZmVyVHlwZQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0NvbXBvbmVudFNhbXBsZU1vZGVsLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvQ29tcG9uZW50U2FtcGxlTW9kZWwuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggN2Y4MTQwOS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvQ29tcG9uZW50U2FtcGxlTW9kZWwuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDcwNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEudXRpbC5BcnJheXM7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgQ29tcG9uZW50U2FtcGxlTW9kZWwgY2xhc3MgcmVwcmVzZW50cyBhIHNldCBvZiBpbWFnZSBkYXRhIHdob3NlIGVhY2gKLSAqIGVsZW1lbnQgLSB0aGUgc2FtcGxlIG9mIGEgcGl4ZWwgLSB0YWtlcyBvbmUgZGF0YSBlbGVtZW50IG9mIHRoZSBEYXRhQnVmZmVyLgotICogPHA+Ci0gKiBUaGUgQmFuayBpbmRpY2VzIGRlbm90ZSB0aGUgY29ycmVzcG9uZGVuY2UgYmV0d2VlbiB0aGUgYmFuayBvZiBkYXRhIGJ1ZmZlcnMKLSAqIGFuZCBhIGJhbmQgb2YgaW1hZ2UgZGF0YS4gVGhlIFBpeGVsIHN0cmlkZSBpcyB0aGUgbnVtYmVyIG9mIGRhdGEgYXJyYXkKLSAqIGVsZW1lbnRzIGJldHdlZW4gdHdvIHNhbXBsZXMgZm9yIHRoZSBzYW1lIGJhbmQgb24gdGhlIHNhbWUgc2NhbmxpbmUuIFRoZQotICogcGl4ZWwgc3RyaWRlIGZvciBhIEJhbmRlZFNhbXBsZU1vZGVsIGlzIG9uZS4gVGhlIHNjYW5saW5lIHN0cmlkZSByZXByZXNlbnRzCi0gKiB0aGUgbnVtYmVyIG9mIGRhdGEgYXJyYXkgZWxlbWVudHMgYmV0d2VlbiBhIHNwZWNpZmllZCBzYW1wbGUgYW5kIHRoZQotICogY29ycmVzcG9uZGluZyBzYW1wbGUgaW4gdGhlIHNhbWUgY29sdW1uIGluIHRoZSBuZXh0IHNjYW5saW5lLiBUaGUgYXJyYXkgb2YKLSAqIGJhbmQgb2Zmc2V0cyBnaXZlcyB0aGUgc3RhcnRpbmcgb2Zmc2V0cyB3aXRoaW4gZWFjaCBkYXRhIGJhbmtzIG9mIHRoZSBpbiB0aGUKLSAqIERhdGFCdWZmZXIuIFRoZSBiYW5rIGluZGljZXMgcmVwcmVzZW50cyB0aGUgaW5kaWNlcyB3aXRoaW4gZWFjaCBiYW5rIG9mIHRoZQotICogRGF0YUJ1ZmZlciBjb3JyZXNwb25kaW5nIHRvIGEgYmFuZCBvZiBpbWFnZSBkYXRhLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIENvbXBvbmVudFNhbXBsZU1vZGVsIGV4dGVuZHMgU2FtcGxlTW9kZWwgewotCi0gICAgLyoqCi0gICAgICogVGhlIGJhbmQgb2Zmc2V0cyBhcnJheSBvZiB0aGlzIENvbXBvbmVudFNhbXBsZU1vZGVsLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgYmFuZE9mZnNldHNbXTsKLQotICAgIC8qKgotICAgICAqIFRoZSBiYW5rIGluZGljZXMgYXJyYXkgb2YgdGhpcyBDb21wb25lbnRTYW1wbGVNb2RlbC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IGJhbmtJbmRpY2VzW107Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbnVtYmVyIG9mIGJhbmRzIGluIHRoaXMgQ29tcG9uZW50U2FtcGxlTW9kZWwuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBudW1CYW5kczsKLQotICAgIC8qKgotICAgICAqIFRoZSBudW1iZXIgYmFua3Mgb2YgdGhpcyBDb21wb25lbnRTYW1wbGVNb2RlbC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IG51bUJhbmtzOwotCi0gICAgLyoqCi0gICAgICogVGhlIHNjYW5saW5lIHN0cmlkZSBvZiB0aGlzIENvbXBvbmVudFNhbXBsZU1vZGVsLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgc2NhbmxpbmVTdHJpZGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcGl4ZWwgc3RyaWRlIG9mIHRoaXMgQ29tcG9uZW50U2FtcGxlTW9kZWwuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBwaXhlbFN0cmlkZTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBDb21wb25lbnRTYW1wbGVNb2RlbCB3aXRoIHRoZSBzcGVjaWZpZWQgcHJvcGVydGllcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUgb2Ygc2FtcGxlcy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBwaXhlbFN0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsIHN0cmlkZSBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gc2NhbmxpbmVTdHJpZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FubGluZSBzdHJpZGUgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGJhbmtJbmRpY2VzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgdGhlIGJhbmsgaW5kaWNlcy4KLSAgICAgKiBAcGFyYW0gYmFuZE9mZnNldHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiB0aGUgYmFuZCBvZmZzZXRzLgotICAgICAqLwotICAgIHB1YmxpYyBDb21wb25lbnRTYW1wbGVNb2RlbChpbnQgZGF0YVR5cGUsIGludCB3LCBpbnQgaCwgaW50IHBpeGVsU3RyaWRlLCBpbnQgc2NhbmxpbmVTdHJpZGUsCi0gICAgICAgICAgICBpbnQgYmFua0luZGljZXNbXSwgaW50IGJhbmRPZmZzZXRzW10pIHsKLQotICAgICAgICBzdXBlcihkYXRhVHlwZSwgdywgaCwgYmFuZE9mZnNldHMubGVuZ3RoKTsKLQotICAgICAgICBpZiAocGl4ZWxTdHJpZGUgPCAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjRCPVBpeGVsIHN0cmlkZSBtdXN0IGJlID49IDAKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoc2NhbmxpbmVTdHJpZGUgPCAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjRDPVNjYW5saW5lIHN0cmlkZSBtdXN0IGJlID49IDAKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjRDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoYmFua0luZGljZXMubGVuZ3RoICE9IGJhbmRPZmZzZXRzLmxlbmd0aCkgewotICAgICAgICAgICAgLy8gYXd0LjI0RD1CYW5rIEluZGljZXMgbGVuZ3RoIG11c3QgYmUgZXF1YWwgQmFuayBPZmZzZXRzIGxlbmd0aAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNEQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMucGl4ZWxTdHJpZGUgPSBwaXhlbFN0cmlkZTsKLSAgICAgICAgdGhpcy5zY2FubGluZVN0cmlkZSA9IHNjYW5saW5lU3RyaWRlOwotICAgICAgICB0aGlzLmJhbmRPZmZzZXRzID0gYmFuZE9mZnNldHMuY2xvbmUoKTsKLSAgICAgICAgdGhpcy5iYW5rSW5kaWNlcyA9IGJhbmtJbmRpY2VzLmNsb25lKCk7Ci0gICAgICAgIHRoaXMubnVtQmFuZHMgPSBiYW5kT2Zmc2V0cy5sZW5ndGg7Ci0KLSAgICAgICAgaW50IG1heEJhbmsgPSAwOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGJhbmtJbmRpY2VzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBpZiAoYmFua0luZGljZXNbaV0gPCAwKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjI0RT1JbmRleCBvZiB7MH0gYmFuayBtdXN0IGJlID49IDAKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI0RSIsIGkpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGJhbmtJbmRpY2VzW2ldID4gbWF4QmFuaykgewotICAgICAgICAgICAgICAgIG1heEJhbmsgPSBiYW5rSW5kaWNlc1tpXTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICB0aGlzLm51bUJhbmtzID0gbWF4QmFuayArIDE7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQ29tcG9uZW50U2FtcGxlTW9kZWwgd2l0aCB0aGUgc3BlY2lmaWVkIHByb3BlcnRpZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlIG9mIHRoZSBzYW1wbGVzLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIHBpeGVsU3RyaWRlCi0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwgc3RyaWRlIG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBzY2FubGluZVN0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIHNjYW5saW5lIHN0cmlkZSBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gYmFuZE9mZnNldHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBiYW5kIG9mZnNldHMuCi0gICAgICovCi0gICAgcHVibGljIENvbXBvbmVudFNhbXBsZU1vZGVsKGludCBkYXRhVHlwZSwgaW50IHcsIGludCBoLCBpbnQgcGl4ZWxTdHJpZGUsIGludCBzY2FubGluZVN0cmlkZSwKLSAgICAgICAgICAgIGludCBiYW5kT2Zmc2V0c1tdKSB7Ci0KLSAgICAgICAgc3VwZXIoZGF0YVR5cGUsIHcsIGgsIGJhbmRPZmZzZXRzLmxlbmd0aCk7Ci0gICAgICAgIGlmIChwaXhlbFN0cmlkZSA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNEI9UGl4ZWwgc3RyaWRlIG11c3QgYmUgPj0gMAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNEIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChzY2FubGluZVN0cmlkZSA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNEM9U2NhbmxpbmUgc3RyaWRlIG11c3QgYmUgPj0gMAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNEMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMucGl4ZWxTdHJpZGUgPSBwaXhlbFN0cmlkZTsKLSAgICAgICAgdGhpcy5zY2FubGluZVN0cmlkZSA9IHNjYW5saW5lU3RyaWRlOwotICAgICAgICB0aGlzLmJhbmRPZmZzZXRzID0gYmFuZE9mZnNldHMuY2xvbmUoKTsKLSAgICAgICAgdGhpcy5udW1CYW5kcyA9IGJhbmRPZmZzZXRzLmxlbmd0aDsKLSAgICAgICAgdGhpcy5udW1CYW5rcyA9IDE7Ci0KLSAgICAgICAgdGhpcy5iYW5rSW5kaWNlcyA9IG5ldyBpbnRbbnVtQmFuZHNdOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgIGJhbmtJbmRpY2VzW2ldID0gMDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgZ2V0RGF0YUVsZW1lbnRzKGludCB4LCBpbnQgeSwgT2JqZWN0IG9iaiwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID49IHRoaXMud2lkdGggfHwgeSA+PSB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgc3dpdGNoIChkYXRhVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJkYXRhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJkYXRhID0gbmV3IGJ5dGVbbnVtQmFuZHNdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGJkYXRhID0gKGJ5dGVbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1CYW5kczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJkYXRhW2ldID0gKGJ5dGUpZ2V0U2FtcGxlKHgsIHksIGksIGRhdGEpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIG9iaiA9IGJkYXRhOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9TSE9SVDoKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgICAgICBzaG9ydCBzZGF0YVtdOwotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBzZGF0YSA9IG5ldyBzaG9ydFtudW1CYW5kc107Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgc2RhdGEgPSAoc2hvcnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1CYW5kczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNkYXRhW2ldID0gKHNob3J0KWdldFNhbXBsZSh4LCB5LCBpLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBvYmogPSBzZGF0YTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGludCBpZGF0YVtdOwotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBpZGF0YSA9IG5ldyBpbnRbbnVtQmFuZHNdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGlkYXRhID0gKGludFtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgaWRhdGFbaV0gPSBnZXRTYW1wbGUoeCwgeSwgaSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgb2JqID0gaWRhdGE7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FUOgotICAgICAgICAgICAgICAgIGZsb2F0IGZkYXRhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZkYXRhID0gbmV3IGZsb2F0W251bUJhbmRzXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBmZGF0YSA9IChmbG9hdFtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgZmRhdGFbaV0gPSBnZXRTYW1wbGVGbG9hdCh4LCB5LCBpLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBvYmogPSBmZGF0YTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFOgotICAgICAgICAgICAgICAgIGRvdWJsZSBkZGF0YVtdOwotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBkZGF0YSA9IG5ldyBkb3VibGVbbnVtQmFuZHNdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGRkYXRhID0gKGRvdWJsZVtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgZGRhdGFbaV0gPSBnZXRTYW1wbGVEb3VibGUoeCwgeSwgaSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgb2JqID0gZGRhdGE7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gb2JqOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldERhdGFFbGVtZW50cyhpbnQgeCwgaW50IHksIE9iamVjdCBvYmosIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHN3aXRjaCAoZGF0YVR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgYnl0ZSBiYXJyW10gPSAoYnl0ZVtdKW9iajsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgc2V0U2FtcGxlKHgsIHksIGksIGJhcnJbaV0gJiAweGZmLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1NIT1JUOgotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNhcnJbXSA9IChzaG9ydFtdKW9iajsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgc2V0U2FtcGxlKHgsIHksIGksIHNhcnJbaV0gJiAweGZmZmYsIGRhdGEpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGludCBpYXJyW10gPSAoaW50W10pb2JqOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBzZXRTYW1wbGUoeCwgeSwgaSwgaWFycltpXSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9GTE9BVDoKLSAgICAgICAgICAgICAgICBmbG9hdCBmYXJyW10gPSAoZmxvYXRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1CYW5kczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNldFNhbXBsZSh4LCB5LCBpLCBmYXJyW2ldLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRToKLSAgICAgICAgICAgICAgICBkb3VibGUgZGFycltdID0gKGRvdWJsZVtdKW9iajsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgc2V0U2FtcGxlKHgsIHksIGksIGRhcnJbaV0sIGRhdGEpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXBhcmVzIHRoaXMgQ29tcG9uZW50U2FtcGxlTW9kZWwgd2l0aCB0aGUgc3BlY2lmaWVkIE9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbwotICAgICAqICAgICAgICAgICAgdGhlIE9iamVjdC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBvYmplY3QgaXMgYSBDb21wb25lbnRTYW1wbGVNb2RlbCB3aXRoIGlkZW50aWNhbCBkYXRhCi0gICAgICogICAgICAgICB2YWx1ZXMgdG8gdGhpcyBDb21wb25lbnRTYW1wbGVNb2RlbCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3QgbykgewotICAgICAgICBpZiAoKG8gPT0gbnVsbCkgfHwgIShvIGluc3RhbmNlb2YgQ29tcG9uZW50U2FtcGxlTW9kZWwpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgQ29tcG9uZW50U2FtcGxlTW9kZWwgbW9kZWwgPSAoQ29tcG9uZW50U2FtcGxlTW9kZWwpbzsKLSAgICAgICAgcmV0dXJuIHRoaXMud2lkdGggPT0gbW9kZWwud2lkdGggJiYgdGhpcy5oZWlnaHQgPT0gbW9kZWwuaGVpZ2h0Ci0gICAgICAgICAgICAgICAgJiYgdGhpcy5udW1CYW5kcyA9PSBtb2RlbC5udW1CYW5kcyAmJiB0aGlzLmRhdGFUeXBlID09IG1vZGVsLmRhdGFUeXBlCi0gICAgICAgICAgICAgICAgJiYgQXJyYXlzLmVxdWFscyh0aGlzLmJhbmRPZmZzZXRzLCBtb2RlbC5iYW5kT2Zmc2V0cykKLSAgICAgICAgICAgICAgICAmJiBBcnJheXMuZXF1YWxzKHRoaXMuYmFua0luZGljZXMsIG1vZGVsLmJhbmtJbmRpY2VzKQotICAgICAgICAgICAgICAgICYmIHRoaXMubnVtQmFuZHMgPT0gbW9kZWwubnVtQmFuZHMgJiYgdGhpcy5udW1CYW5rcyA9PSBtb2RlbC5udW1CYW5rcwotICAgICAgICAgICAgICAgICYmIHRoaXMuc2NhbmxpbmVTdHJpZGUgPT0gbW9kZWwuc2NhbmxpbmVTdHJpZGUKLSAgICAgICAgICAgICAgICAmJiB0aGlzLnBpeGVsU3RyaWRlID09IG1vZGVsLnBpeGVsU3RyaWRlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEBzZWUgamF2YS5hd3QuaW1hZ2UuU2FtcGxlTW9kZWwjY3JlYXRlU3Vic2V0U2FtcGxlTW9kZWwoaW50W10pCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFNhbXBsZU1vZGVsIGNyZWF0ZVN1YnNldFNhbXBsZU1vZGVsKGludCBiYW5kc1tdKSB7Ci0gICAgICAgIGlmIChiYW5kcy5sZW5ndGggPiB0aGlzLm51bUJhbmRzKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjQ9VGhlIG51bWJlciBvZiB0aGUgYmFuZHMgaW4gdGhlIHN1YnNldCBpcyBncmVhdGVyIHRoYW4gdGhlCi0gICAgICAgICAgICAvLyBudW1iZXIgb2YgYmFuZHMgaW4gdGhlIHNhbXBsZSBtb2RlbAotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42NCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGluZGljZXNbXSA9IG5ldyBpbnRbYmFuZHMubGVuZ3RoXTsKLSAgICAgICAgaW50IG9mZnNldHNbXSA9IG5ldyBpbnRbYmFuZHMubGVuZ3RoXTsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGJhbmRzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBpbmRpY2VzW2ldID0gYmFua0luZGljZXNbYmFuZHNbaV1dOwotICAgICAgICAgICAgb2Zmc2V0c1tpXSA9IGJhbmRPZmZzZXRzW2JhbmRzW2ldXTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBuZXcgQ29tcG9uZW50U2FtcGxlTW9kZWwoZGF0YVR5cGUsIHdpZHRoLCBoZWlnaHQsIHBpeGVsU3RyaWRlLCBzY2FubGluZVN0cmlkZSwKLSAgICAgICAgICAgICAgICBpbmRpY2VzLCBvZmZzZXRzKTsKLQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTYW1wbGVNb2RlbCBjcmVhdGVDb21wYXRpYmxlU2FtcGxlTW9kZWwoaW50IHcsIGludCBoKSB7Ci0gICAgICAgIHJldHVybiBuZXcgQ29tcG9uZW50U2FtcGxlTW9kZWwoZGF0YVR5cGUsIHcsIGgsIHBpeGVsU3RyaWRlLCBwaXhlbFN0cmlkZSAqIHcsIGJhbmtJbmRpY2VzLAotICAgICAgICAgICAgICAgIGJhbmRPZmZzZXRzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50W10gZ2V0UGl4ZWwoaW50IHgsIGludCB5LCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpbnQgcGl4ZWxbXTsKLQotICAgICAgICBpZiAoaUFycmF5ID09IG51bGwpIHsKLSAgICAgICAgICAgIHBpeGVsID0gbmV3IGludFtudW1CYW5kc107Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBwaXhlbCA9IGlBcnJheTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgcGl4ZWxbaV0gPSBnZXRTYW1wbGUoeCwgeSwgaSwgZGF0YSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcGl4ZWw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWwoaW50IHgsIGludCB5LCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgIHNldFNhbXBsZSh4LCB5LCBpLCBpQXJyYXlbaV0sIGRhdGEpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID49IHRoaXMud2lkdGggfHwgeSA+PSB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZGF0YS5nZXRFbGVtKGJhbmtJbmRpY2VzW2JdLCB5ICogc2NhbmxpbmVTdHJpZGUgKyB4ICogcGl4ZWxTdHJpZGUgKyBiYW5kT2Zmc2V0c1tiXSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZsb2F0IGdldFNhbXBsZUZsb2F0KGludCB4LCBpbnQgeSwgaW50IGIsIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGRhdGEuZ2V0RWxlbUZsb2F0KGJhbmtJbmRpY2VzW2JdLCB5ICogc2NhbmxpbmVTdHJpZGUgKyB4ICogcGl4ZWxTdHJpZGUKLSAgICAgICAgICAgICAgICArIGJhbmRPZmZzZXRzW2JdKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZG91YmxlIGdldFNhbXBsZURvdWJsZShpbnQgeCwgaW50IHksIGludCBiLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBkYXRhLmdldEVsZW1Eb3VibGUoYmFua0luZGljZXNbYl0sIHkgKiBzY2FubGluZVN0cmlkZSArIHggKiBwaXhlbFN0cmlkZQotICAgICAgICAgICAgICAgICsgYmFuZE9mZnNldHNbYl0pOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnRbXSBnZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBpQXJyYXlbXSwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID4gdGhpcy53aWR0aCB8fCB4ICsgdyA+IHRoaXMud2lkdGggfHwgeSA+IHRoaXMuaGVpZ2h0Ci0gICAgICAgICAgICAgICAgfHwgeSArIGggPiB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaW50IHBpeGVsc1tdID0gbnVsbDsKLSAgICAgICAgaW50IGlkeCA9IDA7Ci0KLSAgICAgICAgaWYgKGlBcnJheSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBwaXhlbHMgPSBuZXcgaW50W3cgKiBoICogbnVtQmFuZHNdOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcGl4ZWxzID0gaUFycmF5OwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBuID0gMDsgbiA8IG51bUJhbmRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgcGl4ZWxzW2lkeCsrXSA9IGdldFNhbXBsZShqLCBpLCBuLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcGl4ZWxzOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGlBcnJheVtdLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggKyB3ID4gdGhpcy53aWR0aCB8fCB5ICsgaCA+IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpbnQgaWR4ID0gMDsKLSAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBuID0gMDsgbiA8IG51bUJhbmRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgc2V0U2FtcGxlKGosIGksIG4sIGlBcnJheVtpZHgrK10sIGRhdGEpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZShpbnQgeCwgaW50IHksIGludCBiLCBpbnQgcywgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID49IHRoaXMud2lkdGggfHwgeSA+PSB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBkYXRhLnNldEVsZW0oYmFua0luZGljZXNbYl0sIHkgKiBzY2FubGluZVN0cmlkZSArIHggKiBwaXhlbFN0cmlkZSArIGJhbmRPZmZzZXRzW2JdLCBzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50W10gZ2V0U2FtcGxlcyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGIsIGludCBpQXJyYXlbXSwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ICsgdyA+IHRoaXMud2lkdGggfHwgeSArIGggPiB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaW50IHNhbXBsZXNbXTsKLSAgICAgICAgaW50IGlkeCA9IDA7Ci0KLSAgICAgICAgaWYgKGlBcnJheSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBzYW1wbGVzID0gbmV3IGludFt3ICogaF07Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBzYW1wbGVzID0gaUFycmF5OwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRhdGEgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjI5NT1kYXRhIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOTUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSB5OyBpIDwgeSArIGg7IGkrKykgewotICAgICAgICAgICAgZm9yIChpbnQgaiA9IHg7IGogPCB4ICsgdzsgaisrKSB7Ci0gICAgICAgICAgICAgICAgc2FtcGxlc1tpZHgrK10gPSBnZXRTYW1wbGUoaiwgaSwgYiwgZGF0YSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gc2FtcGxlczsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGVzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnQgYiwgaW50IGlBcnJheVtdLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggKyB3ID4gdGhpcy53aWR0aCB8fCB5ICsgaCA+IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpbnQgaWR4ID0gMDsKLSAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICBzZXRTYW1wbGUoaiwgaSwgYiwgaUFycmF5W2lkeCsrXSwgZGF0YSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgZmxvYXQgcywgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID49IHRoaXMud2lkdGggfHwgeSA+PSB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBkYXRhLnNldEVsZW1GbG9hdChiYW5rSW5kaWNlc1tiXSwgeSAqIHNjYW5saW5lU3RyaWRlICsgeCAqIHBpeGVsU3RyaWRlICsgYmFuZE9mZnNldHNbYl0sIHMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZShpbnQgeCwgaW50IHksIGludCBiLCBkb3VibGUgcywgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID49IHRoaXMud2lkdGggfHwgeSA+PSB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBkYXRhCi0gICAgICAgICAgICAgICAgLnNldEVsZW1Eb3VibGUoYmFua0luZGljZXNbYl0sIHkgKiBzY2FubGluZVN0cmlkZSArIHggKiBwaXhlbFN0cmlkZQotICAgICAgICAgICAgICAgICAgICAgICAgKyBiYW5kT2Zmc2V0c1tiXSwgcyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIERhdGFCdWZmZXIgY3JlYXRlRGF0YUJ1ZmZlcigpIHsKLSAgICAgICAgRGF0YUJ1ZmZlciBkYXRhID0gbnVsbDsKLQotICAgICAgICBpbnQgbWF4T2Zmc2V0ID0gYmFuZE9mZnNldHNbMF07Ci0gICAgICAgIGZvciAoaW50IGkgPSAxOyBpIDwgYmFuZE9mZnNldHMubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIGlmIChiYW5kT2Zmc2V0c1tpXSA+IG1heE9mZnNldCkgewotICAgICAgICAgICAgICAgIG1heE9mZnNldCA9IGJhbmRPZmZzZXRzW2ldOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGludCBzaXplID0gKGhlaWdodCAtIDEpICogc2NhbmxpbmVTdHJpZGUgKyAod2lkdGggLSAxKSAqIHBpeGVsU3RyaWRlICsgbWF4T2Zmc2V0ICsgMTsKLQotICAgICAgICBzd2l0Y2ggKGRhdGFUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlckJ5dGUoc2l6ZSwgbnVtQmFua3MpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfU0hPUlQ6Ci0gICAgICAgICAgICAgICAgZGF0YSA9IG5ldyBEYXRhQnVmZmVyU2hvcnQoc2l6ZSwgbnVtQmFua3MpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlclVTaG9ydChzaXplLCBudW1CYW5rcyk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgZGF0YSA9IG5ldyBEYXRhQnVmZmVySW50KHNpemUsIG51bUJhbmtzKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FUOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlckZsb2F0KHNpemUsIG51bUJhbmtzKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRToKLSAgICAgICAgICAgICAgICBkYXRhID0gbmV3IERhdGFCdWZmZXJEb3VibGUoc2l6ZSwgbnVtQmFua3MpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGRhdGE7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBvZmZzZXQgb2YgdGhlIHNwZWNpZmllZCBiYW5kIG9mIHRoZSBzcGVjaWZpZWQgcGl4ZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHBpeGVsLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBwaXhlbC4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJhbmQuCi0gICAgICogQHJldHVybiB0aGUgb2Zmc2V0IG9mIHRoZSBzcGVjaWZpZWQgYmFuZCBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0T2Zmc2V0KGludCB4LCBpbnQgeSwgaW50IGIpIHsKLSAgICAgICAgcmV0dXJuIHkgKiBzY2FubGluZVN0cmlkZSArIHggKiBwaXhlbFN0cmlkZSArIGJhbmRPZmZzZXRzW2JdOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG9mZnNldCBvZiB0aGUgZmlyc3QgYmFuZCBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEByZXR1cm4gdGhlIG9mZnNldCBvZiB0aGUgZmlyc3QgYmFuZCBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0T2Zmc2V0KGludCB4LCBpbnQgeSkgewotICAgICAgICByZXR1cm4geSAqIHNjYW5saW5lU3RyaWRlICsgeCAqIHBpeGVsU3RyaWRlICsgYmFuZE9mZnNldHNbMF07Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZpbmFsIGludCBnZXRTYW1wbGVTaXplKGludCBiYW5kKSB7Ci0gICAgICAgIHJldHVybiBEYXRhQnVmZmVyLmdldERhdGFUeXBlU2l6ZShkYXRhVHlwZSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZpbmFsIGludFtdIGdldFNhbXBsZVNpemUoKSB7Ci0gICAgICAgIGludCBzYW1wbGVTaXplc1tdID0gbmV3IGludFtudW1CYW5kc107Ci0gICAgICAgIGludCBzaXplID0gRGF0YUJ1ZmZlci5nZXREYXRhVHlwZVNpemUoZGF0YVR5cGUpOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgc2FtcGxlU2l6ZXNbaV0gPSBzaXplOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBzYW1wbGVTaXplczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIGJhbmsgaW5kaWNlcyBjb3JyZXNwb25kaW5nIHRvIHRoaXMgQ29tcG9uZW50U2FtcGxlTW9kZWwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgYmFuayBpbmRpY2VzLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnRbXSBnZXRCYW5rSW5kaWNlcygpIHsKLSAgICAgICAgcmV0dXJuIGJhbmtJbmRpY2VzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiB0aGUgYmFuZCBvZmZzZXRzIGNvcnJlc3BvbmRpbmcgdG8gdGhpcwotICAgICAqIENvbXBvbmVudFNhbXBsZU1vZGVsLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGJhbmQgb2Zmc2V0cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50W10gZ2V0QmFuZE9mZnNldHMoKSB7Ci0gICAgICAgIHJldHVybiBiYW5kT2Zmc2V0cy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBoYXNoIGNvZGUgb2YgdGhpcyBDb21wb25lbnRTYW1wbGVNb2RlbCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiBhIGhhc2ggY29kZSBvZiB0aGlzIENvbXBvbmVudFNhbXBsZU1vZGVsIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGhhc2hDb2RlKCkgewotICAgICAgICBpbnQgaGFzaCA9IDA7Ci0gICAgICAgIGludCB0bXAgPSAwOwotCi0gICAgICAgIGhhc2ggPSB3aWR0aDsKLSAgICAgICAgdG1wID0gaGFzaCA+Pj4gMjQ7Ci0gICAgICAgIGhhc2ggPDw9IDg7Ci0gICAgICAgIGhhc2ggfD0gdG1wOwotICAgICAgICBoYXNoIF49IGhlaWdodDsKLSAgICAgICAgdG1wID0gaGFzaCA+Pj4gMjQ7Ci0gICAgICAgIGhhc2ggPDw9IDg7Ci0gICAgICAgIGhhc2ggfD0gdG1wOwotICAgICAgICBoYXNoIF49IG51bUJhbmRzOwotICAgICAgICB0bXAgPSBoYXNoID4+PiAyNDsKLSAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgaGFzaCB8PSB0bXA7Ci0gICAgICAgIGhhc2ggXj0gZGF0YVR5cGU7Ci0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICBoYXNoIHw9IHRtcDsKLSAgICAgICAgZm9yIChpbnQgZWxlbWVudCA6IGJhbmRPZmZzZXRzKSB7Ci0gICAgICAgICAgICBoYXNoIF49IGVsZW1lbnQ7Ci0gICAgICAgICAgICB0bXAgPSBoYXNoID4+PiAyNDsKLSAgICAgICAgICAgIGhhc2ggPDw9IDg7Ci0gICAgICAgICAgICBoYXNoIHw9IHRtcDsKLSAgICAgICAgfQotICAgICAgICBmb3IgKGludCBlbGVtZW50IDogYmFua0luZGljZXMpIHsKLSAgICAgICAgICAgIGhhc2ggXj0gZWxlbWVudDsKLSAgICAgICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgICAgIGhhc2ggfD0gdG1wOwotICAgICAgICB9Ci0gICAgICAgIGhhc2ggXj0gcGl4ZWxTdHJpZGU7Ci0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICBoYXNoIHw9IHRtcDsKLQotICAgICAgICBoYXNoIF49IHNjYW5saW5lU3RyaWRlOwotICAgICAgICByZXR1cm4gaGFzaDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzY2FubGluZSBzdHJpZGUgb2YgdGhpcyBDb21wb25lbnRTYW1wbGVNb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzY2FubGluZSBzdHJpZGUgb2YgdGhpcyBDb21wb25lbnRTYW1wbGVNb2RlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldFNjYW5saW5lU3RyaWRlKCkgewotICAgICAgICByZXR1cm4gc2NhbmxpbmVTdHJpZGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcGl4ZWwgc3RyaWRlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHBpeGVsIHN0cmlkZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldFBpeGVsU3RyaWRlKCkgewotICAgICAgICByZXR1cm4gcGl4ZWxTdHJpZGU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZpbmFsIGludCBnZXROdW1EYXRhRWxlbWVudHMoKSB7Ci0gICAgICAgIHJldHVybiBudW1CYW5kczsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9Db252b2x2ZU9wLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvQ29udm9sdmVPcC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2ZWI4OTA5Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9Db252b2x2ZU9wLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NDUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKgotICogQGRhdGU6IFNlcCAyOSwgMjAwNQotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC4qOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUG9pbnQyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEudXRpbC5BcnJheXM7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLkF3dEltYWdlQmFja2Rvb3JBY2Nlc3NvcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgQ29udm9sdmVPcCBjbGFzcyBjb252b2x2ZXMgZnJvbSB0aGUgc291cmNlIGRhdGEgdG8gdGhlIGRlc3RpbmF0aW9uIHVzaW5nCi0gKiBhIGNvbnZvbHV0aW9uIGtlcm5lbC4gRWFjaCBvdXRwdXQgcGl4ZWwgaXMgcmVwcmVzZW50ZWQgYXMgdGhlIHJlc3VsdCBvZgotICogbXVsdGlwbHlpbmcgdGhlIGtlcm5lbCBhbmQgdGhlIHN1cnJvdW5kIG9mIHRoZSBpbnB1dCBwaXhlbC4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBDb252b2x2ZU9wIGltcGxlbWVudHMgQnVmZmVyZWRJbWFnZU9wLCBSYXN0ZXJPcCB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRURHRV9aRVJPX0ZJTEwgaW5kaWNhdGVzIHRoYXQgcGl4ZWxzIGF0IHRoZSBlZGdlIG9mIHRoZQotICAgICAqIGRlc3RpbmF0aW9uIGltYWdlIGFyZSBzZXQgdG8gemVyby4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBFREdFX1pFUk9fRklMTCA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRURHRV9OT19PUCBpbmRpY2F0ZXMgdGhhdCBwaXhlbHMgYXQgdGhlIGVkZ2Ugb2YgdGhlIHNvdXJjZQotICAgICAqIGltYWdlIGFyZSBjb252ZXJ0ZWQgdG8gdGhlIGVkZ2UgcGl4ZWxzIGluIHRoZSBkZXN0aW5hdGlvbiB3aXRob3V0Ci0gICAgICogbW9kaWZpY2F0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEVER0VfTk9fT1AgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIGtlcm5lbC4KLSAgICAgKi8KLSAgICBwcml2YXRlIEtlcm5lbCBrZXJuZWw7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZWRnZSBjb25kLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IGVkZ2VDb25kOwotCi0gICAgLyoqCi0gICAgICogVGhlIHJocy4KLSAgICAgKi8KLSAgICBwcml2YXRlIFJlbmRlcmluZ0hpbnRzIHJocyA9IG51bGw7Ci0KLSAgICBzdGF0aWMgewotICAgICAgICAvLyBUT0RPCi0gICAgICAgIC8vIFN5c3RlbS5sb2FkTGlicmFyeSgiaW1hZ2VvcHMiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQ29udm9sdmVPcCBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIEtlcm5lbCBhbmQKLSAgICAgKiBzcGVjaWZpZWQgZWRnZXMgY29uZGl0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBrZXJuZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgS2VybmVsLgotICAgICAqIEBwYXJhbSBlZGdlQ29uZGl0aW9uCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGVkZ2UgY29uZGl0aW9uLgotICAgICAqIEBwYXJhbSBoaW50cwotICAgICAqICAgICAgICAgICAgdGhlIFJlbmRlcmluZ0hpbnRzIG9iamVjdCwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29udm9sdmVPcChLZXJuZWwga2VybmVsLCBpbnQgZWRnZUNvbmRpdGlvbiwgUmVuZGVyaW5nSGludHMgaGludHMpIHsKLSAgICAgICAgdGhpcy5rZXJuZWwgPSBrZXJuZWw7Ci0gICAgICAgIHRoaXMuZWRnZUNvbmQgPSBlZGdlQ29uZGl0aW9uOwotICAgICAgICB0aGlzLnJocyA9IGhpbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBDb252b2x2ZU9wIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgS2VybmVsIGFuZAotICAgICAqIEVER0VfWkVST19GSUxMIGVkZ2UgY29uZGl0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBrZXJuZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgS2VybmVsLgotICAgICAqLwotICAgIHB1YmxpYyBDb252b2x2ZU9wKEtlcm5lbCBrZXJuZWwpIHsKLSAgICAgICAgdGhpcy5rZXJuZWwgPSBrZXJuZWw7Ci0gICAgICAgIHRoaXMuZWRnZUNvbmQgPSBFREdFX1pFUk9fRklMTDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBLZXJuZWwgb2JqZWN0IG9mIHRoaXMgQ29udm9sdmVPcC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBLZXJuZWwgb2JqZWN0IG9mIHRoaXMgQ29udm9sdmVPcC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgS2VybmVsIGdldEtlcm5lbCgpIHsKLSAgICAgICAgcmV0dXJuIChLZXJuZWwpa2VybmVsLmNsb25lKCk7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIFJlbmRlcmluZ0hpbnRzIGdldFJlbmRlcmluZ0hpbnRzKCkgewotICAgICAgICByZXR1cm4gcmhzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGVkZ2UgY29uZGl0aW9uIG9mIHRoaXMgQ29udm9sdmVPcC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBlZGdlIGNvbmRpdGlvbjogRURHRV9OT19PUCBvciBFREdFX1pFUk9fRklMTC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEVkZ2VDb25kaXRpb24oKSB7Ci0gICAgICAgIHJldHVybiBlZGdlQ29uZDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoUmFzdGVyIHNyYykgewotICAgICAgICByZXR1cm4gc3JjLmdldEJvdW5kcygpOwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRChCdWZmZXJlZEltYWdlIHNyYykgewotICAgICAgICByZXR1cm4gZ2V0Qm91bmRzMkQoc3JjLmdldFJhc3RlcigpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUG9pbnQyRCBnZXRQb2ludDJEKFBvaW50MkQgc3JjUHQsIFBvaW50MkQgZHN0UHQpIHsKLSAgICAgICAgaWYgKGRzdFB0ID09IG51bGwpIHsKLSAgICAgICAgICAgIGRzdFB0ID0gbmV3IFBvaW50MkQuRmxvYXQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGRzdFB0LnNldExvY2F0aW9uKHNyY1B0KTsKLSAgICAgICAgcmV0dXJuIGRzdFB0OwotICAgIH0KLQotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBjcmVhdGVDb21wYXRpYmxlRGVzdFJhc3RlcihSYXN0ZXIgc3JjKSB7Ci0gICAgICAgIHJldHVybiBzcmMuY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKCk7Ci0gICAgfQotCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UgY3JlYXRlQ29tcGF0aWJsZURlc3RJbWFnZShCdWZmZXJlZEltYWdlIHNyYywgQ29sb3JNb2RlbCBkc3RDTSkgewotICAgICAgICBpZiAoZHN0Q00gPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0Q00gPSBzcmMuZ2V0Q29sb3JNb2RlbCgpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRzdENNIGluc3RhbmNlb2YgSW5kZXhDb2xvck1vZGVsKSB7Ci0gICAgICAgICAgICBkc3RDTSA9IENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpOwotICAgICAgICB9Ci0KLSAgICAgICAgV3JpdGFibGVSYXN0ZXIgciA9IGRzdENNLmlzQ29tcGF0aWJsZVNhbXBsZU1vZGVsKHNyYy5nZXRTYW1wbGVNb2RlbCgpKSA/IHNyYy5nZXRSYXN0ZXIoKQotICAgICAgICAgICAgICAgIC5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIoc3JjLmdldFdpZHRoKCksIHNyYy5nZXRIZWlnaHQoKSkgOiBkc3RDTQotICAgICAgICAgICAgICAgIC5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIoc3JjLmdldFdpZHRoKCksIHNyYy5nZXRIZWlnaHQoKSk7Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBCdWZmZXJlZEltYWdlKGRzdENNLCByLCBkc3RDTS5pc0FscGhhUHJlbXVsdGlwbGllZCgpLCBudWxsKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgV3JpdGFibGVSYXN0ZXIgZmlsdGVyKFJhc3RlciBzcmMsIFdyaXRhYmxlUmFzdGVyIGRzdCkgewotICAgICAgICBpZiAoc3JjID09IG51bGwpIHsgLy8gU2hvdWxkIHRocm93IGFjY29yZGluZyB0byBzcGVjCi0gICAgICAgICAgICAvLyBhd3QuMjU2PVNvdXJjZSByYXN0ZXIgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI1NiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHNyYyA9PSBkc3QpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNTc9U291cmNlIHJhc3RlciBpcyBlcXVhbCB0byBkZXN0aW5hdGlvbgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNTciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0ID0gY3JlYXRlQ29tcGF0aWJsZURlc3RSYXN0ZXIoc3JjKTsKLSAgICAgICAgfSBlbHNlIGlmIChzcmMuZ2V0TnVtQmFuZHMoKSAhPSBkc3QuZ2V0TnVtQmFuZHMoKSkgewotICAgICAgICAgICAgLy8gYXd0LjI1OD1OdW1iZXIgb2Ygc291cmNlIGJhbmRzICh7MH0pIGlzIG5vdCBlcXVhbCB0byBudW1iZXIgb2YKLSAgICAgICAgICAgIC8vIGRlc3RpbmF0aW9uIGJhbmRzICh7MX0pCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygKLSAgICAgICAgICAgICAgICAgICAgImF3dC4yNTgiLCBzcmMuZ2V0TnVtQmFuZHMoKSwgZHN0LmdldE51bUJhbmRzKCkpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgLy8gVE9ETwotICAgICAgICAvLyBpZiAoaXBwRmlsdGVyKHNyYywgZHN0LCBCdWZmZXJlZEltYWdlLlRZUEVfQ1VTVE9NKSAhPSAwKQotICAgICAgICBpZiAoc2xvd0ZpbHRlcihzcmMsIGRzdCkgIT0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjIxRj1VbmFibGUgdG8gdHJhbnNmb3JtIHNvdXJjZQotICAgICAgICAgICAgdGhyb3cgbmV3IEltYWdpbmdPcEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMUYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBkc3Q7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2xvdyBmaWx0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIHNyYy4KLSAgICAgKiBAcGFyYW0gZHN0Ci0gICAgICogICAgICAgICAgICB0aGUgZHN0LgotICAgICAqIEByZXR1cm4gdGhlIGludC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBzbG93RmlsdGVyKFJhc3RlciBzcmMsIFdyaXRhYmxlUmFzdGVyIGRzdCkgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgU2FtcGxlTW9kZWwgc20gPSBzcmMuZ2V0U2FtcGxlTW9kZWwoKTsKLQotICAgICAgICAgICAgaW50IG51bUJhbmRzID0gc3JjLmdldE51bUJhbmRzKCk7Ci0gICAgICAgICAgICBpbnQgc3JjSGVpZ2h0ID0gc3JjLmdldEhlaWdodCgpOwotICAgICAgICAgICAgaW50IHNyY1dpZHRoID0gc3JjLmdldFdpZHRoKCk7Ci0KLSAgICAgICAgICAgIGludCB4T3JpZ2luID0ga2VybmVsLmdldFhPcmlnaW4oKTsKLSAgICAgICAgICAgIGludCB5T3JpZ2luID0ga2VybmVsLmdldFlPcmlnaW4oKTsKLSAgICAgICAgICAgIGludCBrV2lkdGggPSBrZXJuZWwuZ2V0V2lkdGgoKTsKLSAgICAgICAgICAgIGludCBrSGVpZ2h0ID0ga2VybmVsLmdldEhlaWdodCgpOwotICAgICAgICAgICAgZmxvYXRbXSBkYXRhID0ga2VybmVsLmdldEtlcm5lbERhdGEobnVsbCk7Ci0KLSAgICAgICAgICAgIGludCBzcmNNaW5YID0gc3JjLmdldE1pblgoKTsKLSAgICAgICAgICAgIGludCBzcmNNaW5ZID0gc3JjLmdldE1pblkoKTsKLSAgICAgICAgICAgIGludCBkc3RNaW5YID0gZHN0LmdldE1pblgoKTsKLSAgICAgICAgICAgIGludCBkc3RNaW5ZID0gZHN0LmdldE1pblkoKTsKLQotICAgICAgICAgICAgaW50IHNyY0NvbnZNYXhYID0gc3JjV2lkdGggLSAoa1dpZHRoIC0geE9yaWdpbiAtIDEpOwotICAgICAgICAgICAgaW50IHNyY0NvbnZNYXhZID0gc3JjSGVpZ2h0IC0gKGtIZWlnaHQgLSB5T3JpZ2luIC0gMSk7Ci0KLSAgICAgICAgICAgIGludFtdIG1heFZhbHVlcyA9IG5ldyBpbnRbbnVtQmFuZHNdOwotICAgICAgICAgICAgaW50W10gbWFza3MgPSBuZXcgaW50W251bUJhbmRzXTsKLSAgICAgICAgICAgIGludFtdIHNhbXBsZVNpemVzID0gc20uZ2V0U2FtcGxlU2l6ZSgpOwotCi0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICBtYXhWYWx1ZXNbaV0gPSAoMSA8PCBzYW1wbGVTaXplc1tpXSkgLSAxOwotICAgICAgICAgICAgICAgIG1hc2tzW2ldID0gfihtYXhWYWx1ZXNbaV0pOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBQcm9jZXNzaW5nIGJvdW5kcwotICAgICAgICAgICAgZmxvYXRbXSBwaXhlbHMgPSBudWxsOwotICAgICAgICAgICAgcGl4ZWxzID0gc3JjLmdldFBpeGVscyhzcmNNaW5YLCBzcmNNaW5ZLCBzcmNXaWR0aCwgc3JjSGVpZ2h0LCBwaXhlbHMpOwotICAgICAgICAgICAgZmxvYXRbXSBuZXdQaXhlbHMgPSBuZXcgZmxvYXRbcGl4ZWxzLmxlbmd0aF07Ci0gICAgICAgICAgICBpbnQgcm93TGVuZ3RoID0gc3JjV2lkdGggKiBudW1CYW5kczsKLSAgICAgICAgICAgIGlmICh0aGlzLmVkZ2VDb25kID09IENvbnZvbHZlT3AuRURHRV9OT19PUCkgewotICAgICAgICAgICAgICAgIC8vIHRvcAotICAgICAgICAgICAgICAgIGludCBzdGFydCA9IDA7Ci0gICAgICAgICAgICAgICAgaW50IGxlbmd0aCA9IHlPcmlnaW4gKiByb3dMZW5ndGg7Ci0gICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShwaXhlbHMsIHN0YXJ0LCBuZXdQaXhlbHMsIHN0YXJ0LCBsZW5ndGgpOwotICAgICAgICAgICAgICAgIC8vIGJvdHRvbQotICAgICAgICAgICAgICAgIHN0YXJ0ID0gKHNyY0hlaWdodCAtIChrSGVpZ2h0IC0geU9yaWdpbiAtIDEpKSAqIHJvd0xlbmd0aDsKLSAgICAgICAgICAgICAgICBsZW5ndGggPSAoa0hlaWdodCAtIHlPcmlnaW4gLSAxKSAqIHJvd0xlbmd0aDsKLSAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHBpeGVscywgc3RhcnQsIG5ld1BpeGVscywgc3RhcnQsIGxlbmd0aCk7Ci0gICAgICAgICAgICAgICAgLy8gbWlkZGxlCi0gICAgICAgICAgICAgICAgbGVuZ3RoID0geE9yaWdpbiAqIG51bUJhbmRzOwotICAgICAgICAgICAgICAgIGludCBsZW5ndGgxID0gKGtXaWR0aCAtIHhPcmlnaW4gLSAxKSAqIG51bUJhbmRzOwotICAgICAgICAgICAgICAgIHN0YXJ0ID0geU9yaWdpbiAqIHJvd0xlbmd0aDsKLSAgICAgICAgICAgICAgICBpbnQgc3RhcnQxID0gKHlPcmlnaW4gKyAxKSAqIHJvd0xlbmd0aCAtIGxlbmd0aDE7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IHlPcmlnaW47IGkgPCAoc3JjSGVpZ2h0IC0gKGtIZWlnaHQgLSB5T3JpZ2luIC0gMSkpOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShwaXhlbHMsIHN0YXJ0LCBuZXdQaXhlbHMsIHN0YXJ0LCBsZW5ndGgpOwotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHBpeGVscywgc3RhcnQxLCBuZXdQaXhlbHMsIHN0YXJ0MSwgbGVuZ3RoMSk7Ci0gICAgICAgICAgICAgICAgICAgIHN0YXJ0ICs9IHJvd0xlbmd0aDsKLSAgICAgICAgICAgICAgICAgICAgc3RhcnQxICs9IHJvd0xlbmd0aDsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gQ3ljbGUgb3ZlciBwaXhlbHMgdG8gYmUgY2FsY3VsYXRlZAotICAgICAgICAgICAgZm9yIChpbnQgaSA9IHlPcmlnaW47IGkgPCBzcmNDb252TWF4WTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IHhPcmlnaW47IGogPCBzcmNDb252TWF4WDsgaisrKSB7Ci0KLSAgICAgICAgICAgICAgICAgICAgLy8gVGFrZSBrZXJuZWwgZGF0YSBpbiBiYWNrd2FyZCBkaXJlY3Rpb24sIGNvbnZvbHV0aW9uCi0gICAgICAgICAgICAgICAgICAgIGludCBrZXJuZWxJZHggPSBkYXRhLmxlbmd0aCAtIDE7Ci0KLSAgICAgICAgICAgICAgICAgICAgaW50IHBpeGVsSW5kZXggPSBpICogcm93TGVuZ3RoICsgaiAqIG51bUJhbmRzOwotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBoSWR4ID0gMCwgcmFzdGVySElkeCA9IGkgLSB5T3JpZ2luOyBoSWR4IDwga0hlaWdodDsgaElkeCsrLCByYXN0ZXJISWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IHdJZHggPSAwLCByYXN0ZXJXSWR4ID0gaiAtIHhPcmlnaW47IHdJZHggPCBrV2lkdGg7IHdJZHgrKywgcmFzdGVyV0lkeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGN1ckluZGV4ID0gcmFzdGVySElkeCAqIHJvd0xlbmd0aCArIHJhc3RlcldJZHggKiBudW1CYW5kczsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBpZHggPSAwOyBpZHggPCBudW1CYW5kczsgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbmV3UGl4ZWxzW3BpeGVsSW5kZXggKyBpZHhdICs9IGRhdGFba2VybmVsSWR4XQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICogcGl4ZWxzW2N1ckluZGV4ICsgaWR4XTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAga2VybmVsSWR4LS07Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICAvLyBDaGVjayBmb3Igb3ZlcmZsb3cgbm93Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGlkeCA9IDA7IGlkeCA8IG51bUJhbmRzOyBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCgoaW50KW5ld1BpeGVsc1twaXhlbEluZGV4ICsgaWR4XSAmIG1hc2tzW2lkeF0pICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAobmV3UGl4ZWxzW3BpeGVsSW5kZXggKyBpZHhdIDwgMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXdQaXhlbHNbcGl4ZWxJbmRleCArIGlkeF0gPSAwOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5ld1BpeGVsc1twaXhlbEluZGV4ICsgaWR4XSA9IG1heFZhbHVlc1tpZHhdOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZHN0LnNldFBpeGVscyhkc3RNaW5YLCBkc3RNaW5ZLCBzcmNXaWR0aCwgc3JjSGVpZ2h0LCBuZXdQaXhlbHMpOwotICAgICAgICB9IGNhdGNoIChFeGNlcHRpb24gZSkgeyAvLyBTb21ldGhpbmcgZ29lcyB3cm9uZywgc2lnbmFsIGVycm9yCi0gICAgICAgICAgICByZXR1cm4gMTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgQnVmZmVyZWRJbWFnZSBmaWx0ZXIoQnVmZmVyZWRJbWFnZSBzcmMsIEJ1ZmZlcmVkSW1hZ2UgZHN0KSB7Ci0gICAgICAgIGlmIChzcmMgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjI1OT1Tb3VyY2UgaW1hZ2UgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI1OSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHNyYyA9PSBkc3QpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNUE9U291cmNlIGVxdWFscyB0byBkZXN0aW5hdGlvbgotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNUEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIENvbG9yTW9kZWwgc3JjQ00gPSBzcmMuZ2V0Q29sb3JNb2RlbCgpOwotICAgICAgICBCdWZmZXJlZEltYWdlIGZpbmFsRHN0ID0gbnVsbDsKLQotICAgICAgICBpZiAoc3JjQ00gaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwpIHsKLSAgICAgICAgICAgIHNyYyA9ICgoSW5kZXhDb2xvck1vZGVsKXNyY0NNKS5jb252ZXJ0VG9JbnREaXNjcmV0ZShzcmMuZ2V0UmFzdGVyKCksIHRydWUpOwotICAgICAgICAgICAgc3JjQ00gPSBzcmMuZ2V0Q29sb3JNb2RlbCgpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRzdCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBkc3QgPSBjcmVhdGVDb21wYXRpYmxlRGVzdEltYWdlKHNyYywgc3JjQ00pOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaWYgKCFzcmNDTS5lcXVhbHMoZHN0LmdldENvbG9yTW9kZWwoKSkpIHsKLSAgICAgICAgICAgICAgICAvLyBUcmVhdCBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX1JHQiBhbmQKLSAgICAgICAgICAgICAgICAvLyBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0IgYXMgc2FtZQotICAgICAgICAgICAgICAgIGlmICghKChzcmMuZ2V0VHlwZSgpID09IEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfUkdCIHx8IHNyYy5nZXRUeXBlKCkgPT0gQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9BUkdCKSAmJiAoZHN0Ci0gICAgICAgICAgICAgICAgICAgICAgICAuZ2V0VHlwZSgpID09IEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfUkdCIHx8IGRzdC5nZXRUeXBlKCkgPT0gQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9BUkdCKSkpIHsKLSAgICAgICAgICAgICAgICAgICAgZmluYWxEc3QgPSBkc3Q7Ci0gICAgICAgICAgICAgICAgICAgIGRzdCA9IGNyZWF0ZUNvbXBhdGlibGVEZXN0SW1hZ2Uoc3JjLCBzcmNDTSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLy8gU2tpcCBhbHBoYSBjaGFubmVsIGZvciBUWVBFX0lOVF9SR0IgaW1hZ2VzCi0gICAgICAgIC8vIFRPRE8KLSAgICAgICAgLy8gaWYgKGlwcEZpbHRlcihzcmMuZ2V0UmFzdGVyKCksIGRzdC5nZXRSYXN0ZXIoKSwgc3JjLmdldFR5cGUoKSkgIT0gMCkKLSAgICAgICAgaWYgKHNsb3dGaWx0ZXIoc3JjLmdldFJhc3RlcigpLCBkc3QuZ2V0UmFzdGVyKCkpICE9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMUY9VW5hYmxlIHRvIHRyYW5zZm9ybSBzb3VyY2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbWFnaW5nT3BFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjFGIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoZmluYWxEc3QgIT0gbnVsbCkgewotICAgICAgICAgICAgR3JhcGhpY3MyRCBnID0gZmluYWxEc3QuY3JlYXRlR3JhcGhpY3MoKTsKLSAgICAgICAgICAgIGcuc2V0Q29tcG9zaXRlKEFscGhhQ29tcG9zaXRlLlNyYyk7Ci0gICAgICAgICAgICBnLmRyYXdJbWFnZShkc3QsIDAsIDAsIG51bGwpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZmluYWxEc3QgPSBkc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZmluYWxEc3Q7Ci0gICAgfQotCi0gICAgLy8gVE9ETyByZW1vdmUgd2hlbiB0aGlzIG1ldGhvZCBpcyB1c2VkCi0gICAgLyoqCi0gICAgICogSXBwIGZpbHRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgc3JjLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3QuCi0gICAgICogQHBhcmFtIGltYWdlVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHR5cGUuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bnVzZWQiKQotICAgIHByaXZhdGUgaW50IGlwcEZpbHRlcihSYXN0ZXIgc3JjLCBXcml0YWJsZVJhc3RlciBkc3QsIGludCBpbWFnZVR5cGUpIHsKLSAgICAgICAgaW50IHNyY1N0cmlkZSwgZHN0U3RyaWRlOwotICAgICAgICBib29sZWFuIHNraXBDaGFubmVsID0gZmFsc2U7Ci0gICAgICAgIGludCBjaGFubmVsczsKLSAgICAgICAgaW50IG9mZnNldHNbXSA9IG51bGw7Ci0KLSAgICAgICAgc3dpdGNoIChpbWFnZVR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9SR0I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQkdSOiB7Ci0gICAgICAgICAgICAgICAgY2hhbm5lbHMgPSA0OwotICAgICAgICAgICAgICAgIHNyY1N0cmlkZSA9IHNyYy5nZXRXaWR0aCgpICogNDsKLSAgICAgICAgICAgICAgICBkc3RTdHJpZGUgPSBkc3QuZ2V0V2lkdGgoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgc2tpcENoYW5uZWwgPSB0cnVlOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQjoKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9BUkdCX1BSRToKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFXzRCWVRFX0FCR1I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV80QllURV9BQkdSX1BSRTogewotICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gNDsKLSAgICAgICAgICAgICAgICBzcmNTdHJpZGUgPSBzcmMuZ2V0V2lkdGgoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCkgKiA0OwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0dSQVk6IHsKLSAgICAgICAgICAgICAgICBjaGFubmVscyA9IDE7Ci0gICAgICAgICAgICAgICAgc3JjU3RyaWRlID0gc3JjLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFXzNCWVRFX0JHUjogewotICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gMzsKLSAgICAgICAgICAgICAgICBzcmNTdHJpZGUgPSBzcmMuZ2V0V2lkdGgoKSAqIDM7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCkgKiAzOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9VU0hPUlRfR1JBWTogLy8gVE9ETyAtIGNvdWxkIGJlIGRvbmUgaW4KLSAgICAgICAgICAgICAgICAvLyBuYXRpdmUgY29kZT8KLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX1VTSE9SVF81NjVfUkdCOgotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfVVNIT1JUXzU1NV9SR0I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0JJTkFSWTogewotICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZGVmYXVsdDogewotICAgICAgICAgICAgICAgIFNhbXBsZU1vZGVsIHNyY1NNID0gc3JjLmdldFNhbXBsZU1vZGVsKCk7Ci0gICAgICAgICAgICAgICAgU2FtcGxlTW9kZWwgZHN0U00gPSBkc3QuZ2V0U2FtcGxlTW9kZWwoKTsKLQotICAgICAgICAgICAgICAgIGlmIChzcmNTTSBpbnN0YW5jZW9mIFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbAotICAgICAgICAgICAgICAgICAgICAgICAgJiYgZHN0U00gaW5zdGFuY2VvZiBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gQ2hlY2sgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsCi0gICAgICAgICAgICAgICAgICAgIGlmIChzcmNTTS5nZXREYXRhVHlwZSgpICE9IERhdGFCdWZmZXIuVFlQRV9CWVRFCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgZHN0U00uZ2V0RGF0YVR5cGUoKSAhPSBEYXRhQnVmZmVyLlRZUEVfQllURSkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QpOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHMgPSBzcmNTTS5nZXROdW1CYW5kcygpOyAvLyBIYXZlIElQUCBmdW5jdGlvbnMgZm9yIDEsCi0gICAgICAgICAgICAgICAgICAgIC8vIDMgYW5kIDQgY2hhbm5lbHMKLSAgICAgICAgICAgICAgICAgICAgaWYgKCEoY2hhbm5lbHMgPT0gMSB8fCBjaGFubmVscyA9PSAzIHx8IGNoYW5uZWxzID09IDQpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBzcmNTdHJpZGUgPSAoKENvbXBvbmVudFNhbXBsZU1vZGVsKXNyY1NNKS5nZXRTY2FubGluZVN0cmlkZSgpOwotICAgICAgICAgICAgICAgICAgICBkc3RTdHJpZGUgPSAoKENvbXBvbmVudFNhbXBsZU1vZGVsKWRzdFNNKS5nZXRTY2FubGluZVN0cmlkZSgpOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoc3JjU00gaW5zdGFuY2VvZiBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsCi0gICAgICAgICAgICAgICAgICAgICAgICAmJiBkc3RTTSBpbnN0YW5jZW9mIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gQ2hlY2sgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbAotICAgICAgICAgICAgICAgICAgICBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIHNwcHNtMSA9IChTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKXNyY1NNOwotICAgICAgICAgICAgICAgICAgICBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIHNwcHNtMiA9IChTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKWRzdFNNOwotCi0gICAgICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gc3Bwc20xLmdldE51bUJhbmRzKCk7Ci0KLSAgICAgICAgICAgICAgICAgICAgLy8gVFlQRV9JTlRfUkdCLCBUWVBFX0lOVF9BUkdCLi4uCi0gICAgICAgICAgICAgICAgICAgIGlmIChzcHBzbTEuZ2V0RGF0YVR5cGUoKSAhPSBEYXRhQnVmZmVyLlRZUEVfSU5UCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgc3Bwc20yLmdldERhdGFUeXBlKCkgIT0gRGF0YUJ1ZmZlci5UWVBFX0lOVAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8ICEoY2hhbm5lbHMgPT0gMyB8fCBjaGFubmVscyA9PSA0KSkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QpOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgLy8gQ2hlY2sgY29tcGF0aWJpbGl0eSBvZiBzYW1wbGUgbW9kZWxzCi0gICAgICAgICAgICAgICAgICAgIGlmICghQXJyYXlzLmVxdWFscyhzcHBzbTEuZ2V0Qml0T2Zmc2V0cygpLCBzcHBzbTIuZ2V0Qml0T2Zmc2V0cygpKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8ICFBcnJheXMuZXF1YWxzKHNwcHNtMS5nZXRCaXRNYXNrcygpLCBzcHBzbTIuZ2V0Qml0TWFza3MoKSkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0KTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHNwcHNtMS5nZXRTYW1wbGVTaXplKGkpICE9IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBpZiAoY2hhbm5lbHMgPT0gMykgeyAvLyBDYW5ub3Qgc2tpcCBjaGFubmVsLCBkb24ndCBrbm93Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyB3aGljaCBpcyBhbHBoYS4uLgotICAgICAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHMgPSA0OwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgc3JjU3RyaWRlID0gc3Bwc20xLmdldFNjYW5saW5lU3RyaWRlKCkgKiA0OwotICAgICAgICAgICAgICAgICAgICBkc3RTdHJpZGUgPSBzcHBzbTIuZ2V0U2NhbmxpbmVTdHJpZGUoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIC8vIEZpbGwgb2Zmc2V0cyBpZiB0aGVyZSdzIGEgY2hpbGQgcmFzdGVyCi0gICAgICAgICAgICAgICAgaWYgKHNyYy5nZXRQYXJlbnQoKSAhPSBudWxsIHx8IGRzdC5nZXRQYXJlbnQoKSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChzcmMuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVYKCkgIT0gMCB8fCBzcmMuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVZKCkgIT0gMAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8IGRzdC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSAhPSAwCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgZHN0LmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWSgpICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIG9mZnNldHMgPSBuZXcgaW50WzRdOwotICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0c1swXSA9IC1zcmMuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVYKCkgKyBzcmMuZ2V0TWluWCgpOwotICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0c1sxXSA9IC1zcmMuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVZKCkgKyBzcmMuZ2V0TWluWSgpOwotICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0c1syXSA9IC1kc3QuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVYKCkgKyBkc3QuZ2V0TWluWCgpOwotICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0c1szXSA9IC1kc3QuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVZKCkgKyBkc3QuZ2V0TWluWSgpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgT2JqZWN0IHNyY0RhdGEsIGRzdERhdGE7Ci0gICAgICAgIEF3dEltYWdlQmFja2Rvb3JBY2Nlc3NvciBkYkFjY2VzcyA9IEF3dEltYWdlQmFja2Rvb3JBY2Nlc3Nvci5nZXRJbnN0YW5jZSgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgc3JjRGF0YSA9IGRiQWNjZXNzLmdldERhdGEoc3JjLmdldERhdGFCdWZmZXIoKSk7Ci0gICAgICAgICAgICBkc3REYXRhID0gZGJBY2Nlc3MuZ2V0RGF0YShkc3QuZ2V0RGF0YUJ1ZmZlcigpKTsKLSAgICAgICAgfSBjYXRjaCAoSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHJldHVybiAtMTsgLy8gVW5rbm93biBkYXRhIGJ1ZmZlciB0eXBlCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gaXBwRmlsdGVyMzJmKGtlcm5lbC5kYXRhLCBrZXJuZWwuZ2V0V2lkdGgoKSwga2VybmVsLmdldEhlaWdodCgpLAotICAgICAgICAgICAgICAgIGtlcm5lbC5nZXRYT3JpZ2luKCksIGtlcm5lbC5nZXRZT3JpZ2luKCksIGVkZ2VDb25kLCBzcmNEYXRhLCBzcmMuZ2V0V2lkdGgoKSwgc3JjCi0gICAgICAgICAgICAgICAgICAgICAgICAuZ2V0SGVpZ2h0KCksIHNyY1N0cmlkZSwgZHN0RGF0YSwgZHN0LmdldFdpZHRoKCksIGRzdC5nZXRIZWlnaHQoKSwKLSAgICAgICAgICAgICAgICBkc3RTdHJpZGUsIGNoYW5uZWxzLCBza2lwQ2hhbm5lbCwgb2Zmc2V0cyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSXBwIGZpbHRlcjMyZi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0ga2VybmVsCi0gICAgICogICAgICAgICAgICB0aGUga2VybmVsLgotICAgICAqIEBwYXJhbSBrV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBrIHdpZHRoLgotICAgICAqIEBwYXJhbSBrSGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgayBoZWlnaHQuCi0gICAgICogQHBhcmFtIGFuY2hvclgKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbmNob3IgeC4KLSAgICAgKiBAcGFyYW0gYW5jaG9yWQotICAgICAqICAgICAgICAgICAgdGhlIGFuY2hvciB5LgotICAgICAqIEBwYXJhbSBib3JkZXJUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgYm9yZGVyIHR5cGUuCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIHNyYy4KLSAgICAgKiBAcGFyYW0gc3JjV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMgd2lkdGguCi0gICAgICogQHBhcmFtIHNyY0hlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIHNyYyBoZWlnaHQuCi0gICAgICogQHBhcmFtIHNyY1N0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIHNyYyBzdHJpZGUuCi0gICAgICogQHBhcmFtIGRzdAotICAgICAqICAgICAgICAgICAgdGhlIGRzdC4KLSAgICAgKiBAcGFyYW0gZHN0V2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3Qgd2lkdGguCi0gICAgICogQHBhcmFtIGRzdEhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGRzdCBoZWlnaHQuCi0gICAgICogQHBhcmFtIGRzdFN0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIGRzdCBzdHJpZGUuCi0gICAgICogQHBhcmFtIGNoYW5uZWxzCi0gICAgICogICAgICAgICAgICB0aGUgY2hhbm5lbHMuCi0gICAgICogQHBhcmFtIHNraXBDaGFubmVsCi0gICAgICogICAgICAgICAgICB0aGUgc2tpcCBjaGFubmVsLgotICAgICAqIEBwYXJhbSBvZmZzZXRzCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0cy4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBuYXRpdmUgaW50IGlwcEZpbHRlcjMyZihmbG9hdCBrZXJuZWxbXSwgaW50IGtXaWR0aCwgaW50IGtIZWlnaHQsIGludCBhbmNob3JYLAotICAgICAgICAgICAgaW50IGFuY2hvclksIGludCBib3JkZXJUeXBlLCBPYmplY3Qgc3JjLCBpbnQgc3JjV2lkdGgsIGludCBzcmNIZWlnaHQsIGludCBzcmNTdHJpZGUsCi0gICAgICAgICAgICBPYmplY3QgZHN0LCBpbnQgZHN0V2lkdGgsIGludCBkc3RIZWlnaHQsIGludCBkc3RTdHJpZGUsIGludCBjaGFubmVscywKLSAgICAgICAgICAgIGJvb2xlYW4gc2tpcENoYW5uZWwsIGludCBvZmZzZXRzW10pOwotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0Nyb3BJbWFnZUZpbHRlci5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL0Nyb3BJbWFnZUZpbHRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyZjRjYTc4Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9Dcm9wSW1hZ2VGaWx0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE5NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEudXRpbC5IYXNodGFibGU7Ci0KLS8qKgotICogVGhlIENyb3BJbWFnZUZpbHRlciBjbGFzcyBjcm9wcyBhIHJlY3Rhbmd1bGFyIHJlZ2lvbiBvZiBhbiBzb3VyY2UgSW1hZ2UgYW5kCi0gKiBwcm92aWRlcyBhIHNvdXJjZSBmb3IgYSBuZXcgaW1hZ2UgY29udGFpbmluZyB0aGUgZXh0cmFjdGVkIHJlZ2lvbi4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBDcm9wSW1hZ2VGaWx0ZXIgZXh0ZW5kcyBJbWFnZUZpbHRlciB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgSEVJR0hULgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgaW50IFgsIFksIFdJRFRILCBIRUlHSFQ7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgQ3JvcEltYWdlRmlsdGVyIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIKLSAgICAgKiBhcmVhLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ3JvcEltYWdlRmlsdGVyKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoKSB7Ci0gICAgICAgIFggPSB4OwotICAgICAgICBZID0geTsKLSAgICAgICAgV0lEVEggPSB3OwotICAgICAgICBIRUlHSFQgPSBoOwotICAgIH0KLQotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bmNoZWNrZWQiKQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFByb3BlcnRpZXMoSGFzaHRhYmxlPD8sID8+IHByb3BzKSB7Ci0gICAgICAgIEhhc2h0YWJsZTxPYmplY3QsIE9iamVjdD4gZnByb3BzOwotICAgICAgICBpZiAocHJvcHMgPT0gbnVsbCkgewotICAgICAgICAgICAgZnByb3BzID0gbmV3IEhhc2h0YWJsZTxPYmplY3QsIE9iamVjdD4oKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZwcm9wcyA9IChIYXNodGFibGU8T2JqZWN0LCBPYmplY3Q+KXByb3BzLmNsb25lKCk7Ci0gICAgICAgIH0KLSAgICAgICAgU3RyaW5nIHByb3BOYW1lID0gIkNyb3AgRmlsdGVycyI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgU3RyaW5nIHByb3AgPSAieD0iICsgWCArICI7IHk9IiArIFkgKyAiOyB3aWR0aD0iICsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQKLSAgICAgICAgICAgICAgICBXSURUSCArICI7IGhlaWdodD0iICsgSEVJR0hUOyAvLyROT04tTkxTLTEkCi0gICAgICAgIE9iamVjdCBvID0gZnByb3BzLmdldChwcm9wTmFtZSk7Ci0gICAgICAgIGlmIChvICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChvIGluc3RhbmNlb2YgU3RyaW5nKSB7Ci0gICAgICAgICAgICAgICAgcHJvcCA9IChTdHJpbmcpbyArICI7ICIgKyBwcm9wOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHByb3AgPSBvLnRvU3RyaW5nKCkgKyAiOyAiICsgcHJvcDsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGZwcm9wcy5wdXQocHJvcE5hbWUsIHByb3ApOwotICAgICAgICBjb25zdW1lci5zZXRQcm9wZXJ0aWVzKGZwcm9wcyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBDb2xvck1vZGVsIG1vZGVsLCBpbnRbXSBwaXhlbHMsIGludCBvZmYsCi0gICAgICAgICAgICBpbnQgc2NhbnNpemUpIHsKLQotICAgICAgICBpZiAoeCArIHcgPCBYIHx8IFggKyBXSURUSCA8IHggfHwgeSArIGggPCBZIHx8IFkgKyBIRUlHSFQgPCB5KSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgZGVzdFgsIGRlc3RZLCBkZXN0V2lkdGgsIGRlc3RIZWlnaHQsIGVuZFgsIGVuZFksIHNyY0VuZFgsIHNyY0VuZFk7Ci0KLSAgICAgICAgaW50IG5ld09mZnNldCA9IG9mZjsKLQotICAgICAgICBlbmRYID0gWCArIFdJRFRIOwotICAgICAgICBlbmRZID0gWSArIEhFSUdIVDsKLQotICAgICAgICBzcmNFbmRYID0geCArIHc7Ci0gICAgICAgIHNyY0VuZFkgPSB5ICsgaDsKLQotICAgICAgICBpZiAoeCA8PSBYKSB7Ci0gICAgICAgICAgICBkZXN0WCA9IDA7Ci0gICAgICAgICAgICBuZXdPZmZzZXQgKz0gWDsKLSAgICAgICAgICAgIGlmIChlbmRYID49IHNyY0VuZFgpIHsKLSAgICAgICAgICAgICAgICBkZXN0V2lkdGggPSBzcmNFbmRYIC0gWDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZGVzdFdpZHRoID0gV0lEVEg7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBkZXN0WCA9IHggLSBYOwotICAgICAgICAgICAgaWYgKGVuZFggPj0gc3JjRW5kWCkgewotICAgICAgICAgICAgICAgIGRlc3RXaWR0aCA9IHc7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGRlc3RXaWR0aCA9IGVuZFggLSB4OwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHkgPD0gWSkgewotICAgICAgICAgICAgbmV3T2Zmc2V0ICs9IHNjYW5zaXplICogKFkgLSB5KTsKLSAgICAgICAgICAgIGRlc3RZID0gMDsKLSAgICAgICAgICAgIGlmIChlbmRZID49IHNyY0VuZFkpIHsKLSAgICAgICAgICAgICAgICBkZXN0SGVpZ2h0ID0gc3JjRW5kWSAtIFk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGRlc3RIZWlnaHQgPSBIRUlHSFQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBkZXN0WSA9IHkgLSBZOwotICAgICAgICAgICAgaWYgKGVuZFkgPj0gc3JjRW5kWSkgewotICAgICAgICAgICAgICAgIGRlc3RIZWlnaHQgPSBoOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBkZXN0SGVpZ2h0ID0gZW5kWSAtIHk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgY29uc3VtZXIuc2V0UGl4ZWxzKGRlc3RYLCBkZXN0WSwgZGVzdFdpZHRoLCBkZXN0SGVpZ2h0LCBtb2RlbCwgcGl4ZWxzLCBuZXdPZmZzZXQsIHNjYW5zaXplKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIENvbG9yTW9kZWwgbW9kZWwsIGJ5dGVbXSBwaXhlbHMsIGludCBvZmYsCi0gICAgICAgICAgICBpbnQgc2NhbnNpemUpIHsKLQotICAgICAgICBpZiAoeCArIHcgPCBYIHx8IFggKyBXSURUSCA8IHggfHwgeSArIGggPCBZIHx8IFkgKyBIRUlHSFQgPCB5KSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgZGVzdFgsIGRlc3RZLCBkZXN0V2lkdGgsIGRlc3RIZWlnaHQsIGVuZFgsIGVuZFksIHNyY0VuZFgsIHNyY0VuZFk7Ci0KLSAgICAgICAgaW50IG5ld09mZnNldCA9IG9mZjsKLQotICAgICAgICBlbmRYID0gWCArIFdJRFRIOwotICAgICAgICBlbmRZID0gWSArIEhFSUdIVDsKLQotICAgICAgICBzcmNFbmRYID0geCArIHc7Ci0gICAgICAgIHNyY0VuZFkgPSB5ICsgaDsKLQotICAgICAgICBpZiAoeCA8PSBYKSB7Ci0gICAgICAgICAgICBkZXN0WCA9IDA7Ci0gICAgICAgICAgICBuZXdPZmZzZXQgKz0gWDsKLSAgICAgICAgICAgIGlmIChlbmRYID49IHNyY0VuZFgpIHsKLSAgICAgICAgICAgICAgICBkZXN0V2lkdGggPSBzcmNFbmRYIC0gWDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZGVzdFdpZHRoID0gV0lEVEg7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBkZXN0WCA9IHggLSBYOwotICAgICAgICAgICAgaWYgKGVuZFggPj0gc3JjRW5kWCkgewotICAgICAgICAgICAgICAgIGRlc3RXaWR0aCA9IHc7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGRlc3RXaWR0aCA9IGVuZFggLSB4OwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHkgPD0gWSkgewotICAgICAgICAgICAgbmV3T2Zmc2V0ICs9IHNjYW5zaXplICogKFkgLSB5KTsKLSAgICAgICAgICAgIGRlc3RZID0gMDsKLSAgICAgICAgICAgIGlmIChlbmRZID49IHNyY0VuZFkpIHsKLSAgICAgICAgICAgICAgICBkZXN0SGVpZ2h0ID0gc3JjRW5kWSAtIFk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGRlc3RIZWlnaHQgPSBIRUlHSFQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBkZXN0WSA9IHkgLSBZOwotICAgICAgICAgICAgaWYgKGVuZFkgPj0gc3JjRW5kWSkgewotICAgICAgICAgICAgICAgIGRlc3RIZWlnaHQgPSBoOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBkZXN0SGVpZ2h0ID0gZW5kWSAtIHk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgY29uc3VtZXIuc2V0UGl4ZWxzKGRlc3RYLCBkZXN0WSwgZGVzdFdpZHRoLCBkZXN0SGVpZ2h0LCBtb2RlbCwgcGl4ZWxzLCBuZXdPZmZzZXQsIHNjYW5zaXplKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXREaW1lbnNpb25zKGludCB3LCBpbnQgaCkgewotICAgICAgICBjb25zdW1lci5zZXREaW1lbnNpb25zKFdJRFRILCBIRUlHSFQpOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9EYXRhQnVmZmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDkyZjkwMGYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ4MSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2UuRGF0YUJ1ZmZlckxpc3RlbmVyOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBDbGFzcyBEYXRhQnVmZmVyIGlzIGEgd3JhcHBlciBjbGFzcyBmb3IgYSBkYXRhIGFycmF5IHRvIGJlIHVzZWQgZm9yIHRoZQotICogc2l0dWF0aW9uIHdoZXJlIGEgc3VpdGUgb2YgZnVuY3Rpb25hbGl0eSBhY3RzIG9uIGEgc2V0IG9mIGRhdGEgaW4gYQotICogY29uc2lzdGVudCB3YXkgZXZlbiB0aG91Z2ggdGhlIHByaW1pdGl2ZSB0eXBlIG9mIHRoZSBkYXRhIG1heSB2YXJ5IGZyb20gb25lCi0gKiB1c2UgdG8gdGhlIG5leHQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgRGF0YUJ1ZmZlciB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9CWVRFLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfQllURSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9VU0hPUlQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9VU0hPUlQgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfU0hPUlQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9TSE9SVCA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9JTlQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9JTlQgPSAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfRkxPQVQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9GTE9BVCA9IDQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgVFlQRV9ET1VCTEUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgVFlQRV9ET1VCTEUgPSA1OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFRZUEVfVU5ERUZJTkVELgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRZUEVfVU5ERUZJTkVEID0gMzI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGF0YSB0eXBlIGluZGljYXRlcyB0aGUgcHJpbWl0aXZlIHR5cGUgb2YgdGhlIGRhdGEgaW4gdGhpcwotICAgICAqIERhdGFCdWZmZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBkYXRhVHlwZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBudW1iZXIgb2YgZGF0YSBhcnJheXMgaW4gdGhpcyBEYXRhQnVmZmVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgYmFua3M7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc3RhcnRpbmcgaW5kZXggZm9yIHJlYWRpbmcgdGhlIGRhdGEgZnJvbSB0aGUgZmlyc3QgKG9yIG9ubHkpIGludGVybmFsCi0gICAgICogZGF0YSBhcnJheS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IG9mZnNldDsKLQotICAgIC8qKgotICAgICAqIFRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgb2YgdGhlIGRhdGEgYXJyYXlzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgc2l6ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBzdGFydGluZyBpbmRpY2VzIGZvciByZWFkaW5nIHRoZSBkYXRhIGZyb20gdGhlIGludGVybmFsIGRhdGEgYXJyYXlzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgb2Zmc2V0c1tdOwotCi0gICAgLyoqCi0gICAgICogVGhlIGRhdGEgY2hhbmdlZC4KLSAgICAgKi8KLSAgICBib29sZWFuIGRhdGFDaGFuZ2VkID0gdHJ1ZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBkYXRhIHRha2VuLgotICAgICAqLwotICAgIGJvb2xlYW4gZGF0YVRha2VuID0gZmFsc2U7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbGlzdGVuZXIuCi0gICAgICovCi0gICAgRGF0YUJ1ZmZlckxpc3RlbmVyIGxpc3RlbmVyOwotCi0gICAgc3RhdGljIHsKLSAgICAgICAgQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29ySW1wbC5pbml0KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRhdGEgYnVmZmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgdHlwZS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSBvZiB0aGUgZGF0YSBhcnJheXMuCi0gICAgICogQHBhcmFtIG51bUJhbmtzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGRhdGEgYXJyYXlzIHRvIGNyZWF0ZS4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0aW5nIGluZGljZXMgZm9yIHJlYWRpbmcgdGhlIGRhdGEgZnJvbSB0aGUgaW50ZXJuYWwKLSAgICAgKiAgICAgICAgICAgIGRhdGEgYXJyYXlzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBEYXRhQnVmZmVyKGludCBkYXRhVHlwZSwgaW50IHNpemUsIGludCBudW1CYW5rcywgaW50W10gb2Zmc2V0cykgewotICAgICAgICB0aGlzLmRhdGFUeXBlID0gZGF0YVR5cGU7Ci0gICAgICAgIHRoaXMuc2l6ZSA9IHNpemU7Ci0gICAgICAgIHRoaXMuYmFua3MgPSBudW1CYW5rczsKLSAgICAgICAgdGhpcy5vZmZzZXRzID0gb2Zmc2V0cy5jbG9uZSgpOwotICAgICAgICB0aGlzLm9mZnNldCA9IG9mZnNldHNbMF07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRhdGEgYnVmZmVyIHdpdGggYWxsIG9mIHRoZSBkYXRhIGFycmF5cyBzdGFydGluZyBhdAotICAgICAqIHRoZSBzYW1lIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgdHlwZS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSBvZiB0aGUgZGF0YSBhcnJheXMuCi0gICAgICogQHBhcmFtIG51bUJhbmtzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGRhdGEgYXJyYXlzIHRvIGNyZWF0ZS4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IHRvIHVzZSBmb3IgYWxsIG9mIHRoZSBkYXRhIGFycmF5cy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgRGF0YUJ1ZmZlcihpbnQgZGF0YVR5cGUsIGludCBzaXplLCBpbnQgbnVtQmFua3MsIGludCBvZmZzZXQpIHsKLSAgICAgICAgdGhpcy5kYXRhVHlwZSA9IGRhdGFUeXBlOwotICAgICAgICB0aGlzLnNpemUgPSBzaXplOwotICAgICAgICB0aGlzLmJhbmtzID0gbnVtQmFua3M7Ci0gICAgICAgIHRoaXMub2Zmc2V0ID0gb2Zmc2V0OwotICAgICAgICB0aGlzLm9mZnNldHMgPSBuZXcgaW50W251bUJhbmtzXTsKLSAgICAgICAgaW50IGkgPSAwOwotICAgICAgICB3aGlsZSAoaSA8IG51bUJhbmtzKSB7Ci0gICAgICAgICAgICBvZmZzZXRzW2krK10gPSBvZmZzZXQ7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgd2l0aCBhbGwgb2YgdGhlIGRhdGEgYXJyYXlzIHJlYWQgZnJvbSB0aGUKLSAgICAgKiBiZWdpbm5pbmcgKGF0IG9mZnNldCB6ZXJvKS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUuCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgb2YgdGhlIGRhdGEgYXJyYXlzLgotICAgICAqIEBwYXJhbSBudW1CYW5rcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBkYXRhIGFycmF5cyB0byBjcmVhdGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIERhdGFCdWZmZXIoaW50IGRhdGFUeXBlLCBpbnQgc2l6ZSwgaW50IG51bUJhbmtzKSB7Ci0gICAgICAgIHRoaXMuZGF0YVR5cGUgPSBkYXRhVHlwZTsKLSAgICAgICAgdGhpcy5zaXplID0gc2l6ZTsKLSAgICAgICAgdGhpcy5iYW5rcyA9IG51bUJhbmtzOwotICAgICAgICB0aGlzLm9mZnNldCA9IDA7Ci0gICAgICAgIHRoaXMub2Zmc2V0cyA9IG5ldyBpbnRbbnVtQmFua3NdOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciB3aXRoIG9uZSBpbnRlcm5hbCBkYXRhIGFycmF5IHJlYWQgZnJvbSB0aGUKLSAgICAgKiBiZWdpbm5pbmcgKGF0IG9mZnNldCB6ZXJvKS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUuCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgb2YgdGhlIGRhdGEgYXJyYXlzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBEYXRhQnVmZmVyKGludCBkYXRhVHlwZSwgaW50IHNpemUpIHsKLSAgICAgICAgdGhpcy5kYXRhVHlwZSA9IGRhdGFUeXBlOwotICAgICAgICB0aGlzLnNpemUgPSBzaXplOwotICAgICAgICB0aGlzLmJhbmtzID0gMTsKLSAgICAgICAgdGhpcy5vZmZzZXQgPSAwOwotICAgICAgICB0aGlzLm9mZnNldHMgPSBuZXcgaW50WzFdOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgdmFsdWUgaW4gdGhlIHNwZWNpZmllZCBhcnJheSBhdCB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBiYW5rCi0gICAgICogICAgICAgICAgICB0aGUgaW50ZXJuYWwgYXJyYXkgdG8gdGhlIGRhdGEgdG8uCi0gICAgICogQHBhcmFtIGkKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCB3aXRoaW4gdGhlIGFycmF5IHdoZXJlIHRoZSBkYXRhIHNob3VsZCBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSB2YWwKLSAgICAgKiAgICAgICAgICAgIHRoZSB2YWx1ZSB0byB3cml0ZSBpbnRvIHRoZSBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzZXRFbGVtKGludCBiYW5rLCBpbnQgaSwgaW50IHZhbCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBmbG9hdCBkYXRhIHZhbHVlIGluIHRoZSBzcGVjaWZpZWQgYXJyYXkgYXQgdGhlIHNwZWNpZmllZCBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmFuawotICAgICAqICAgICAgICAgICAgdGhlIGludGVybmFsIGFycmF5IHRvIHRoZSBkYXRhIHRvLgotICAgICAqIEBwYXJhbSBpCi0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggd2l0aGluIHRoZSBhcnJheSB3aGVyZSB0aGUgZGF0YSBzaG91bGQgYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gdmFsCi0gICAgICogICAgICAgICAgICB0aGUgdmFsdWUgdG8gd3JpdGUgaW50byB0aGUgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RWxlbUZsb2F0KGludCBiYW5rLCBpbnQgaSwgZmxvYXQgdmFsKSB7Ci0gICAgICAgIHNldEVsZW0oYmFuaywgaSwgKGludCl2YWwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRvdWJsZSBkYXRhIHZhbHVlIGluIHRoZSBzcGVjaWZpZWQgYXJyYXkgYXQgdGhlIHNwZWNpZmllZCBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmFuawotICAgICAqICAgICAgICAgICAgdGhlIGludGVybmFsIGFycmF5IHRvIHRoZSBkYXRhIHRvLgotICAgICAqIEBwYXJhbSBpCi0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggd2l0aGluIHRoZSBhcnJheSB3aGVyZSB0aGUgZGF0YSBzaG91bGQgYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gdmFsCi0gICAgICogICAgICAgICAgICB0aGUgdmFsdWUgdG8gd3JpdGUgaW50byB0aGUgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RWxlbURvdWJsZShpbnQgYmFuaywgaW50IGksIGRvdWJsZSB2YWwpIHsKLSAgICAgICAgc2V0RWxlbShiYW5rLCBpLCAoaW50KXZhbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGF0YSB2YWx1ZSBpbiB0aGUgZmlyc3QgYXJyYXkgYXQgdGhlIHNwZWNpZmllZCBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaQotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IHdpdGhpbiB0aGUgYXJyYXkgd2hlcmUgdGhlIGRhdGEgc2hvdWxkIGJlIHdyaXR0ZW4uCi0gICAgICogQHBhcmFtIHZhbAotICAgICAqICAgICAgICAgICAgdGhlIHZhbHVlIHRvIHdyaXRlIGludG8gdGhlIGFycmF5LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEVsZW0oaW50IGksIGludCB2YWwpIHsKLSAgICAgICAgc2V0RWxlbSgwLCBpLCB2YWwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRhdGEgdmFsdWUgZnJvbSB0aGUgc3BlY2lmaWVkIGRhdGEgYXJyYXkgYXQgdGhlIHNwZWNpZmllZCBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmFuawotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXkgdG8gcmVhZCBmcm9tLgotICAgICAqIEBwYXJhbSBpCi0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggd2l0aGluIHRoZSBhcnJheSB3aGVyZSB0aGUgZGF0YSBzaG91bGQgYmUgcmVhZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhIGVsZW1lbnQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXRFbGVtKGludCBiYW5rLCBpbnQgaSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmbG9hdC10eXBlIGRhdGEgdmFsdWUgZnJvbSB0aGUgc3BlY2lmaWVkIGRhdGEgYXJyYXkgYXQgdGhlCi0gICAgICogc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBiYW5rCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBhcnJheSB0byByZWFkIGZyb20uCi0gICAgICogQHBhcmFtIGkKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCB3aXRoaW4gdGhlIGFycmF5IHdoZXJlIHRoZSBkYXRhIHNob3VsZCBiZSByZWFkLgotICAgICAqIEByZXR1cm4gdGhlIGRhdGEgZWxlbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0RWxlbUZsb2F0KGludCBiYW5rLCBpbnQgaSkgewotICAgICAgICByZXR1cm4gZ2V0RWxlbShiYW5rLCBpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkb3VibGUtdHlwZSBkYXRhIHZhbHVlIGZyb20gdGhlIHNwZWNpZmllZCBkYXRhIGFycmF5IGF0IHRoZQotICAgICAqIHNwZWNpZmllZCBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmFuawotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXkgdG8gcmVhZCBmcm9tLgotICAgICAqIEBwYXJhbSBpCi0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggd2l0aGluIHRoZSBhcnJheSB3aGVyZSB0aGUgZGF0YSBzaG91bGQgYmUgcmVhZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhIGVsZW1lbnQuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRFbGVtRG91YmxlKGludCBiYW5rLCBpbnQgaSkgewotICAgICAgICByZXR1cm4gZ2V0RWxlbShiYW5rLCBpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBmbG9hdCBkYXRhIHZhbHVlIGluIHRoZSBmaXJzdCBhcnJheSBhdCB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpCi0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggd2l0aGluIHRoZSBhcnJheSB3aGVyZSB0aGUgZGF0YSBzaG91bGQgYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gdmFsCi0gICAgICogICAgICAgICAgICB0aGUgdmFsdWUgdG8gd3JpdGUgaW50byB0aGUgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RWxlbUZsb2F0KGludCBpLCBmbG9hdCB2YWwpIHsKLSAgICAgICAgc2V0RWxlbUZsb2F0KDAsIGksIHZhbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZG91YmxlIGRhdGEgdmFsdWUgaW4gdGhlIGZpcnN0IGFycmF5IGF0IHRoZSBzcGVjaWZpZWQgaW5kZXguCi0gICAgICogCi0gICAgICogQHBhcmFtIGkKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCB3aXRoaW4gdGhlIGFycmF5IHdoZXJlIHRoZSBkYXRhIHNob3VsZCBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSB2YWwKLSAgICAgKiAgICAgICAgICAgIHRoZSB2YWx1ZSB0byB3cml0ZSBpbnRvIHRoZSBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRFbGVtRG91YmxlKGludCBpLCBkb3VibGUgdmFsKSB7Ci0gICAgICAgIHNldEVsZW1Eb3VibGUoMCwgaSwgdmFsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkYXRhIHZhbHVlIGZyb20gdGhlIGZpcnN0IGRhdGEgYXJyYXkgYXQgdGhlIHNwZWNpZmllZCBpbmRleCBhbmQKLSAgICAgKiByZXR1cm5zIGl0IGFzIGFuIGludGVnZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGkKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCB3aXRoaW4gdGhlIGFycmF5IHdoZXJlIHRoZSBkYXRhIHNob3VsZCBiZSByZWFkLgotICAgICAqIEByZXR1cm4gdGhlIGRhdGEgZWxlbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEVsZW0oaW50IGkpIHsKLSAgICAgICAgcmV0dXJuIGdldEVsZW0oMCwgaSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSB2YWx1ZSBmcm9tIHRoZSBmaXJzdCBkYXRhIGFycmF5IGF0IHRoZSBzcGVjaWZpZWQgaW5kZXggYW5kCi0gICAgICogcmV0dXJucyBpdCBhcyBhIGZsb2F0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpCi0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggd2l0aGluIHRoZSBhcnJheSB3aGVyZSB0aGUgZGF0YSBzaG91bGQgYmUgcmVhZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhIGVsZW1lbnQuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldEVsZW1GbG9hdChpbnQgaSkgewotICAgICAgICByZXR1cm4gZ2V0RWxlbSgwLCBpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkYXRhIHZhbHVlIGZyb20gdGhlIGZpcnN0IGRhdGEgYXJyYXkgYXQgdGhlIHNwZWNpZmllZCBpbmRleCBhbmQKLSAgICAgKiByZXR1cm5zIGl0IGFzIGEgZG91YmxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpCi0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggd2l0aGluIHRoZSBhcnJheSB3aGVyZSB0aGUgZGF0YSBzaG91bGQgYmUgcmVhZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhIGVsZW1lbnQuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRFbGVtRG91YmxlKGludCBpKSB7Ci0gICAgICAgIHJldHVybiBnZXRFbGVtKGkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFycmF5IGdpdmluZyB0aGUgb2Zmc2V0cyBjb3JyZXNwb25kaW5nIHRvIHRoZSBpbnRlcm5hbCBkYXRhCi0gICAgICogYXJyYXlzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIG9mZnNldHMuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldE9mZnNldHMoKSB7Ci0gICAgICAgIHJldHVybiBvZmZzZXRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNpemUgaW4gYml0cyBvZiB0aGUgcHJpbWl0aXZlIGRhdGEgdHlwZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzaXplIGluIGJpdHMgb2YgdGhlIHByaW1pdGl2ZSBkYXRhIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRTaXplKCkgewotICAgICAgICByZXR1cm4gc2l6ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBvZmZzZXQgY29ycmVzcG9uZGluZyB0byB0aGUgZmlyc3QgaW50ZXJuYWwgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBvZmZzZXQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRPZmZzZXQoKSB7Ci0gICAgICAgIHJldHVybiBvZmZzZXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGRhdGEgYXJyYXlzIGluIHRoaXMgRGF0YUJ1ZmZlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgZGF0YSBhcnJheXMuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXROdW1CYW5rcygpIHsKLSAgICAgICAgcmV0dXJuIGJhbmtzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHByaW1pdGl2ZSB0eXBlIG9mIHRoaXMgYnVmZmVyJ3MgZGF0YS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXREYXRhVHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIHRoaXMuZGF0YVR5cGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2l6ZSBpbiBiaXRzIG9mIHRoZSBwcmltaXRpdmUgZGF0YSB0eXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0eXBlCi0gICAgICogICAgICAgICAgICB0aGUgcHJpbWl0aXZlIHR5cGUuCi0gICAgICogQHJldHVybiB0aGUgc2l6ZSBpbiBiaXRzIG9mIHRoZSBwcmltaXRpdmUgZGF0YSB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgaW50IGdldERhdGFUeXBlU2l6ZShpbnQgdHlwZSkgewotICAgICAgICBzd2l0Y2ggKHR5cGUpIHsKLQotICAgICAgICAgICAgY2FzZSBUWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIDg7Ci0KLSAgICAgICAgICAgIGNhc2UgVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICBjYXNlIFRZUEVfU0hPUlQ6Ci0gICAgICAgICAgICAgICAgcmV0dXJuIDE2OwotCi0gICAgICAgICAgICBjYXNlIFRZUEVfSU5UOgotICAgICAgICAgICAgY2FzZSBUWVBFX0ZMT0FUOgotICAgICAgICAgICAgICAgIHJldHVybiAzMjsKLQotICAgICAgICAgICAgY2FzZSBUWVBFX0RPVUJMRToKLSAgICAgICAgICAgICAgICByZXR1cm4gNjQ7Ci0KLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjIyQz1Vbmtub3duIGRhdGEgdHlwZSB7MH0KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIyQyIsIHR5cGUpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogTm90aWZpZXMgdGhlIGxpc3RlbmVyIHRoYXQgdGhlIGRhdGEgaGFzIGNoYW5nZWQuCi0gICAgICovCi0gICAgdm9pZCBub3RpZnlDaGFuZ2VkKCkgewotICAgICAgICBpZiAobGlzdGVuZXIgIT0gbnVsbCAmJiAhZGF0YUNoYW5nZWQpIHsKLSAgICAgICAgICAgIGRhdGFDaGFuZ2VkID0gdHJ1ZTsKLSAgICAgICAgICAgIGxpc3RlbmVyLmRhdGFDaGFuZ2VkKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBOb3RpZmllcyB0aGUgbGlzdGVuZXIgdGhhdCB0aGUgZGF0YSBoYXMgYmVlbiByZWxlYXNlZC4KLSAgICAgKi8KLSAgICB2b2lkIG5vdGlmeVRha2VuKCkgewotICAgICAgICBpZiAobGlzdGVuZXIgIT0gbnVsbCAmJiAhZGF0YVRha2VuKSB7Ci0gICAgICAgICAgICBkYXRhVGFrZW4gPSB0cnVlOwotICAgICAgICAgICAgbGlzdGVuZXIuZGF0YVRha2VuKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZWxlYXNlIHRoZSBkYXRhLgotICAgICAqLwotICAgIHZvaWQgcmVsZWFzZURhdGEoKSB7Ci0gICAgICAgIGlmIChsaXN0ZW5lciAhPSBudWxsICYmIGRhdGFUYWtlbikgewotICAgICAgICAgICAgZGF0YVRha2VuID0gZmFsc2U7Ci0gICAgICAgICAgICBsaXN0ZW5lci5kYXRhUmVsZWFzZWQoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgdGhlIGRhdGEgYnVmZmVyIGxpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsaXN0ZW5lcgotICAgICAqICAgICAgICAgICAgdGhlIGxpc3RlbmVyLgotICAgICAqLwotICAgIHZvaWQgYWRkRGF0YUJ1ZmZlckxpc3RlbmVyKERhdGFCdWZmZXJMaXN0ZW5lciBsaXN0ZW5lcikgewotICAgICAgICB0aGlzLmxpc3RlbmVyID0gbGlzdGVuZXI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgZGF0YSBidWZmZXIgbGlzdGVuZXIuCi0gICAgICovCi0gICAgdm9pZCByZW1vdmVEYXRhQnVmZmVyTGlzdGVuZXIoKSB7Ci0gICAgICAgIGxpc3RlbmVyID0gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBWYWxpZGF0ZS4KLSAgICAgKi8KLSAgICB2b2lkIHZhbGlkYXRlKCkgewotICAgICAgICBkYXRhQ2hhbmdlZCA9IGZhbHNlOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXJCeXRlLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvRGF0YUJ1ZmZlckJ5dGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzQwN2RlOC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvRGF0YUJ1ZmZlckJ5dGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE4MyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgRGF0YUJ1ZmZlckJ5dGUgaXMgdGhlIHN1YmNsYXNzIG9mIERhdGFCdWZmZXIgZm9yIHRoZSBjYXNlIHdoZXJlIHRoZQotICogdW5kZXJseWluZyBkYXRhIGlzIG9mIHR5cGUgYnl0ZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBmaW5hbCBjbGFzcyBEYXRhQnVmZmVyQnl0ZSBleHRlbmRzIERhdGFCdWZmZXIgewotCi0gICAgLyoqCi0gICAgICogVGhlIGRhdGEuCi0gICAgICovCi0gICAgYnl0ZSBkYXRhW11bXTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIHVuc2lnbmVkIHNob3J0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXlzCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBhcnJheXMgdG8gY29weSB0aGUgZGF0YSBmcm9tLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZSBmcm9tIHRoZSBkYXRhIGFycmF5cy4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0aW5nIGluZGljZXMgZm9yIHJlYWRpbmcgdGhlIGRhdGEgZnJvbSB0aGUgaW50ZXJuYWwKLSAgICAgKiAgICAgICAgICAgIGRhdGEgYXJyYXlzLgotICAgICAqLwotICAgIHB1YmxpYyBEYXRhQnVmZmVyQnl0ZShieXRlIGRhdGFBcnJheXNbXVtdLCBpbnQgc2l6ZSwgaW50IG9mZnNldHNbXSkgewotICAgICAgICBzdXBlcihUWVBFX0JZVEUsIHNpemUsIGRhdGFBcnJheXMubGVuZ3RoLCBvZmZzZXRzKTsKLSAgICAgICAgZGF0YSA9IGRhdGFBcnJheXMuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgb2YgdHlwZSB1bnNpZ25lZCBzaG9ydC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YUFycmF5cwotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXlzIHRvIGNvcHkgdGhlIGRhdGEgZnJvbS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UgZnJvbSB0aGUgZGF0YSBhcnJheXMuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJCeXRlKGJ5dGUgZGF0YUFycmF5c1tdW10sIGludCBzaXplKSB7Ci0gICAgICAgIHN1cGVyKFRZUEVfQllURSwgc2l6ZSwgZGF0YUFycmF5cy5sZW5ndGgpOwotICAgICAgICBkYXRhID0gZGF0YUFycmF5cy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIHVuc2lnbmVkIHNob3J0IHdpdGggYSBzaW5nbGUKLSAgICAgKiB1bmRlcmx5aW5nIGFycmF5IG9mIGRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXkgdG8gY29weSB0aGUgZGF0YSBmcm9tLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZS4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnRpbmcgaW5kZXggdG8gdXNlIHdoZW4gcmVhZGluZyB0aGUgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlckJ5dGUoYnl0ZSBkYXRhQXJyYXlbXSwgaW50IHNpemUsIGludCBvZmZzZXQpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9CWVRFLCBzaXplLCAxLCBvZmZzZXQpOwotICAgICAgICBkYXRhID0gbmV3IGJ5dGVbMV1bXTsKLSAgICAgICAgZGF0YVswXSA9IGRhdGFBcnJheTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgb2YgdHlwZSB1bnNpZ25lZCBzaG9ydCB3aXRoIGEgc2luZ2xlCi0gICAgICogdW5kZXJseWluZyBhcnJheSBvZiBkYXRhIHN0YXJ0aW5nIGF0IGluZGV4IDAuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXkgdG8gY29weSB0aGUgZGF0YSBmcm9tLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlckJ5dGUoYnl0ZSBkYXRhQXJyYXlbXSwgaW50IHNpemUpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9CWVRFLCBzaXplKTsKLSAgICAgICAgZGF0YSA9IG5ldyBieXRlWzFdW107Ci0gICAgICAgIGRhdGFbMF0gPSBkYXRhQXJyYXk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGVtcHR5IGRhdGEgYnVmZmVyIG9mIHR5cGUgdW5zaWduZWQgc2hvcnQgd2l0aCBvZmZzZXRzCi0gICAgICogZXF1YWwgdG8gemVyby4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UgZnJvbSB0aGUgZGF0YSBhcnJheXMuCi0gICAgICogQHBhcmFtIG51bUJhbmtzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGRhdGEgYXJyYXlzIHRvIGNyZWF0ZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlckJ5dGUoaW50IHNpemUsIGludCBudW1CYW5rcykgewotICAgICAgICBzdXBlcihUWVBFX0JZVEUsIHNpemUsIG51bUJhbmtzKTsKLSAgICAgICAgZGF0YSA9IG5ldyBieXRlW251bUJhbmtzXVtdOwotICAgICAgICBpbnQgaSA9IDA7Ci0gICAgICAgIHdoaWxlIChpIDwgbnVtQmFua3MpIHsKLSAgICAgICAgICAgIGRhdGFbaSsrXSA9IG5ldyBieXRlW3NpemVdOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGVtcHR5IGRhdGEgYnVmZmVyIG9mIHR5cGUgdW5zaWduZWQgc2hvcnQgd2l0aCBhIHNpbmdsZQotICAgICAqIHVuZGVybHlpbmcgYXJyYXkgb2YgZGF0YSBzdGFydGluZyBhdCBpbmRleCAwLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlckJ5dGUoaW50IHNpemUpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9CWVRFLCBzaXplKTsKLSAgICAgICAgZGF0YSA9IG5ldyBieXRlWzFdW107Ci0gICAgICAgIGRhdGFbMF0gPSBuZXcgYnl0ZVtzaXplXTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRFbGVtKGludCBiYW5rLCBpbnQgaSwgaW50IHZhbCkgewotICAgICAgICBkYXRhW2JhbmtdW29mZnNldHNbYmFua10gKyBpXSA9IChieXRlKXZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEVsZW0oaW50IGksIGludCB2YWwpIHsKLSAgICAgICAgZGF0YVswXVtvZmZzZXQgKyBpXSA9IChieXRlKXZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0RWxlbShpbnQgYmFuaywgaW50IGkpIHsKLSAgICAgICAgcmV0dXJuIChkYXRhW2JhbmtdW29mZnNldHNbYmFua10gKyBpXSkgJiAweGZmOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRhdGEgb2YgdGhlIHNwZWNpZmllZCBpbnRlcm5hbCBkYXRhIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBiYW5rCi0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggb2YgdGhlIGRlc2lyZWQgZGF0YSBhcnJheS4KLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBieXRlW10gZ2V0RGF0YShpbnQgYmFuaykgewotICAgICAgICBub3RpZnlUYWtlbigpOwotICAgICAgICByZXR1cm4gZGF0YVtiYW5rXTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldEVsZW0oaW50IGkpIHsKLSAgICAgICAgcmV0dXJuIChkYXRhWzBdW29mZnNldCArIGldKSAmIDB4ZmY7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYmFuayBkYXRhLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJhbmsgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYnl0ZVtdW10gZ2V0QmFua0RhdGEoKSB7Ci0gICAgICAgIG5vdGlmeVRha2VuKCk7Ci0gICAgICAgIHJldHVybiBkYXRhLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBvZiB0aGUgZmlyc3QgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBieXRlW10gZ2V0RGF0YSgpIHsKLSAgICAgICAgbm90aWZ5VGFrZW4oKTsKLSAgICAgICAgcmV0dXJuIGRhdGFbMF07Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvRGF0YUJ1ZmZlckRvdWJsZS5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXJEb3VibGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNWQwYTUxNi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvRGF0YUJ1ZmZlckRvdWJsZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjI2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi0vKioKLSAqIFRoZSBDbGFzcyBEYXRhQnVmZmVyRG91YmxlIGlzIHRoZSBzdWJjbGFzcyBvZiBEYXRhQnVmZmVyIGZvciB0aGUgY2FzZSB3aGVyZQotICogdGhlIHVuZGVybHlpbmcgZGF0YSBpcyBvZiB0eXBlIGRvdWJsZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBmaW5hbCBjbGFzcyBEYXRhQnVmZmVyRG91YmxlIGV4dGVuZHMgRGF0YUJ1ZmZlciB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGF0YS4KLSAgICAgKi8KLSAgICBkb3VibGUgZGF0YVtdW107Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgb2YgdHlwZSBkb3VibGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFBcnJheXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5cyB0byBjb3B5IHRoZSBkYXRhIGZyb20uCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlIGZyb20gdGhlIGRhdGEgYXJyYXlzLgotICAgICAqIEBwYXJhbSBvZmZzZXRzCi0gICAgICogICAgICAgICAgICB0aGUgc3RhcnRpbmcgaW5kaWNlcyBmb3IgcmVhZGluZyB0aGUgZGF0YSBmcm9tIHRoZSBpbnRlcm5hbAotICAgICAqICAgICAgICAgICAgZGF0YSBhcnJheXMuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJEb3VibGUoZG91YmxlIGRhdGFBcnJheXNbXVtdLCBpbnQgc2l6ZSwgaW50IG9mZnNldHNbXSkgewotICAgICAgICBzdXBlcihUWVBFX0RPVUJMRSwgc2l6ZSwgZGF0YUFycmF5cy5sZW5ndGgsIG9mZnNldHMpOwotICAgICAgICBkYXRhID0gZGF0YUFycmF5cy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIGRvdWJsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YUFycmF5cwotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXlzIHRvIGNvcHkgdGhlIGRhdGEgZnJvbS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UgZnJvbSB0aGUgZGF0YSBhcnJheXMuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJEb3VibGUoZG91YmxlIGRhdGFBcnJheXNbXVtdLCBpbnQgc2l6ZSkgewotICAgICAgICBzdXBlcihUWVBFX0RPVUJMRSwgc2l6ZSwgZGF0YUFycmF5cy5sZW5ndGgpOwotICAgICAgICBkYXRhID0gZGF0YUFycmF5cy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIGRvdWJsZSB3aXRoIGEgc2luZ2xlIHVuZGVybHlpbmcKLSAgICAgKiBhcnJheSBvZiBkYXRhLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRvIGNvcHkgdGhlIGRhdGEgZnJvbS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0aW5nIGluZGV4IHRvIHVzZSB3aGVuIHJlYWRpbmcgdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJEb3VibGUoZG91YmxlIGRhdGFBcnJheVtdLCBpbnQgc2l6ZSwgaW50IG9mZnNldCkgewotICAgICAgICBzdXBlcihUWVBFX0RPVUJMRSwgc2l6ZSwgMSwgb2Zmc2V0KTsKLSAgICAgICAgZGF0YSA9IG5ldyBkb3VibGVbMV1bXTsKLSAgICAgICAgZGF0YVswXSA9IGRhdGFBcnJheTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgb2YgdHlwZSBkb3VibGUgd2l0aCBhIHNpbmdsZSB1bmRlcmx5aW5nCi0gICAgICogYXJyYXkgb2YgZGF0YSBzdGFydGluZyBhdCBpbmRleCAwLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRvIGNvcHkgdGhlIGRhdGEgZnJvbS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJEb3VibGUoZG91YmxlIGRhdGFBcnJheVtdLCBpbnQgc2l6ZSkgewotICAgICAgICBzdXBlcihUWVBFX0RPVUJMRSwgc2l6ZSk7Ci0gICAgICAgIGRhdGEgPSBuZXcgZG91YmxlWzFdW107Ci0gICAgICAgIGRhdGFbMF0gPSBkYXRhQXJyYXk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGVtcHR5IGRhdGEgYnVmZmVyIG9mIHR5cGUgZG91YmxlIHdpdGggb2Zmc2V0cyBlcXVhbCB0bwotICAgICAqIHplcm8uCi0gICAgICogCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlIGZyb20gdGhlIGRhdGEgYXJyYXlzLgotICAgICAqIEBwYXJhbSBudW1CYW5rcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBkYXRhIGFycmF5cyB0byBjcmVhdGUuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJEb3VibGUoaW50IHNpemUsIGludCBudW1CYW5rcykgewotICAgICAgICBzdXBlcihUWVBFX0RPVUJMRSwgc2l6ZSwgbnVtQmFua3MpOwotICAgICAgICBkYXRhID0gbmV3IGRvdWJsZVtudW1CYW5rc11bXTsKLSAgICAgICAgaW50IGkgPSAwOwotICAgICAgICB3aGlsZSAoaSA8IG51bUJhbmtzKSB7Ci0gICAgICAgICAgICBkYXRhW2krK10gPSBuZXcgZG91YmxlW3NpemVdOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGVtcHR5IGRhdGEgYnVmZmVyIG9mIHR5cGUgZG91YmxlIHdpdGggYSBzaW5nbGUKLSAgICAgKiB1bmRlcmx5aW5nIGFycmF5IG9mIGRhdGEgc3RhcnRpbmcgYXQgaW5kZXggMC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJEb3VibGUoaW50IHNpemUpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9ET1VCTEUsIHNpemUpOwotICAgICAgICBkYXRhID0gbmV3IGRvdWJsZVsxXVtdOwotICAgICAgICBkYXRhWzBdID0gbmV3IGRvdWJsZVtzaXplXTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRFbGVtKGludCBiYW5rLCBpbnQgaSwgaW50IHZhbCkgewotICAgICAgICBkYXRhW2JhbmtdW29mZnNldHNbYmFua10gKyBpXSA9IHZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEVsZW1GbG9hdChpbnQgYmFuaywgaW50IGksIGZsb2F0IHZhbCkgewotICAgICAgICBkYXRhW2JhbmtdW29mZnNldHNbYmFua10gKyBpXSA9IHZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEVsZW1Eb3VibGUoaW50IGJhbmssIGludCBpLCBkb3VibGUgdmFsKSB7Ci0gICAgICAgIGRhdGFbYmFua11bb2Zmc2V0c1tiYW5rXSArIGldID0gdmFsOwotICAgICAgICBub3RpZnlDaGFuZ2VkKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0RWxlbShpbnQgaSwgaW50IHZhbCkgewotICAgICAgICBkYXRhWzBdW29mZnNldCArIGldID0gdmFsOwotICAgICAgICBub3RpZnlDaGFuZ2VkKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRFbGVtKGludCBiYW5rLCBpbnQgaSkgewotICAgICAgICByZXR1cm4gKGludCkoZGF0YVtiYW5rXVtvZmZzZXRzW2JhbmtdICsgaV0pOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRFbGVtRmxvYXQoaW50IGJhbmssIGludCBpKSB7Ci0gICAgICAgIHJldHVybiAoZmxvYXQpKGRhdGFbYmFua11bb2Zmc2V0c1tiYW5rXSArIGldKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZG91YmxlIGdldEVsZW1Eb3VibGUoaW50IGJhbmssIGludCBpKSB7Ci0gICAgICAgIHJldHVybiBkYXRhW2JhbmtdW29mZnNldHNbYmFua10gKyBpXTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRFbGVtRmxvYXQoaW50IGksIGZsb2F0IHZhbCkgewotICAgICAgICBkYXRhWzBdW29mZnNldCArIGldID0gdmFsOwotICAgICAgICBub3RpZnlDaGFuZ2VkKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0RWxlbURvdWJsZShpbnQgaSwgZG91YmxlIHZhbCkgewotICAgICAgICBkYXRhWzBdW29mZnNldCArIGldID0gdmFsOwotICAgICAgICBub3RpZnlDaGFuZ2VkKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBvZiB0aGUgc3BlY2lmaWVkIGludGVybmFsIGRhdGEgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJhbmsKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgZGVzaXJlZCBkYXRhIGFycmF5LgotICAgICAqIEByZXR1cm4gdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZVtdIGdldERhdGEoaW50IGJhbmspIHsKLSAgICAgICAgbm90aWZ5VGFrZW4oKTsKLSAgICAgICAgcmV0dXJuIGRhdGFbYmFua107Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRFbGVtKGludCBpKSB7Ci0gICAgICAgIHJldHVybiAoaW50KShkYXRhWzBdW29mZnNldCArIGldKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXQgZ2V0RWxlbUZsb2F0KGludCBpKSB7Ci0gICAgICAgIHJldHVybiAoZmxvYXQpKGRhdGFbMF1bb2Zmc2V0ICsgaV0pOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBkb3VibGUgZ2V0RWxlbURvdWJsZShpbnQgaSkgewotICAgICAgICByZXR1cm4gZGF0YVswXVtvZmZzZXQgKyBpXTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBiYW5rIGRhdGEuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYmFuayBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGVbXVtdIGdldEJhbmtEYXRhKCkgewotICAgICAgICBub3RpZnlUYWtlbigpOwotICAgICAgICByZXR1cm4gZGF0YS5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRhdGEgb2YgdGhlIGZpcnN0IGRhdGEgYXJyYXkuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlW10gZ2V0RGF0YSgpIHsKLSAgICAgICAgbm90aWZ5VGFrZW4oKTsKLSAgICAgICAgcmV0dXJuIGRhdGFbMF07Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXJGbG9hdC5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXJGbG9hdC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5YTRhNmJmLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9EYXRhQnVmZmVyRmxvYXQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDIyNiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgRGF0YUJ1ZmZlckZsb2F0IGlzIHRoZSBzdWJjbGFzcyBvZiBEYXRhQnVmZmVyIGZvciB0aGUgY2FzZSB3aGVyZQotICogdGhlIHVuZGVybHlpbmcgZGF0YSBpcyBmbG9hdC4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBmaW5hbCBjbGFzcyBEYXRhQnVmZmVyRmxvYXQgZXh0ZW5kcyBEYXRhQnVmZmVyIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBkYXRhLgotICAgICAqLwotICAgIGZsb2F0IGRhdGFbXVtdOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRhdGEgYnVmZmVyIG9mIHR5cGUgZmxvYXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFBcnJheXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5cyB0byBjb3B5IHRoZSBkYXRhIGZyb20uCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlIGZyb20gdGhlIGRhdGEgYXJyYXlzLgotICAgICAqIEBwYXJhbSBvZmZzZXRzCi0gICAgICogICAgICAgICAgICB0aGUgc3RhcnRpbmcgaW5kaWNlcyBmb3IgcmVhZGluZyB0aGUgZGF0YSBmcm9tIHRoZSBpbnRlcm5hbAotICAgICAqICAgICAgICAgICAgZGF0YSBhcnJheXMuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJGbG9hdChmbG9hdCBkYXRhQXJyYXlzW11bXSwgaW50IHNpemUsIGludCBvZmZzZXRzW10pIHsKLSAgICAgICAgc3VwZXIoVFlQRV9GTE9BVCwgc2l6ZSwgZGF0YUFycmF5cy5sZW5ndGgsIG9mZnNldHMpOwotICAgICAgICBkYXRhID0gZGF0YUFycmF5cy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIGZsb2F0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXlzCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBhcnJheXMgdG8gY29weSB0aGUgZGF0YSBmcm9tLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZSBmcm9tIHRoZSBkYXRhIGFycmF5cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlckZsb2F0KGZsb2F0IGRhdGFBcnJheXNbXVtdLCBpbnQgc2l6ZSkgewotICAgICAgICBzdXBlcihUWVBFX0ZMT0FULCBzaXplLCBkYXRhQXJyYXlzLmxlbmd0aCk7Ci0gICAgICAgIGRhdGEgPSBkYXRhQXJyYXlzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRhdGEgYnVmZmVyIG9mIHR5cGUgZmxvYXQgd2l0aCBhIHNpbmdsZSB1bmRlcmx5aW5nCi0gICAgICogYXJyYXkgb2YgZGF0YS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YUFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBhcnJheSB0byBjb3B5IHRoZSBkYXRhIGZyb20uCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlLgotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdGFydGluZyBpbmRleCB0byB1c2Ugd2hlbiByZWFkaW5nIHRoZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBEYXRhQnVmZmVyRmxvYXQoZmxvYXQgZGF0YUFycmF5W10sIGludCBzaXplLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIHN1cGVyKFRZUEVfRkxPQVQsIHNpemUsIDEsIG9mZnNldCk7Ci0gICAgICAgIGRhdGEgPSBuZXcgZmxvYXRbMV1bXTsKLSAgICAgICAgZGF0YVswXSA9IGRhdGFBcnJheTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgb2YgdHlwZSBmbG9hdCB3aXRoIGEgc2luZ2xlIHVuZGVybHlpbmcKLSAgICAgKiBhcnJheSBvZiBkYXRhIHN0YXJ0aW5nIGF0IGluZGV4IDAuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXkgdG8gY29weSB0aGUgZGF0YSBmcm9tLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlckZsb2F0KGZsb2F0IGRhdGFBcnJheVtdLCBpbnQgc2l6ZSkgewotICAgICAgICBzdXBlcihUWVBFX0ZMT0FULCBzaXplKTsKLSAgICAgICAgZGF0YSA9IG5ldyBmbG9hdFsxXVtdOwotICAgICAgICBkYXRhWzBdID0gZGF0YUFycmF5OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBlbXB0eSBkYXRhIGJ1ZmZlciBvZiB0eXBlIGZsb2F0IHdpdGggb2Zmc2V0cyBlcXVhbCB0bwotICAgICAqIHplcm8uCi0gICAgICogCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlIGZyb20gdGhlIGRhdGEgYXJyYXlzLgotICAgICAqIEBwYXJhbSBudW1CYW5rcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBkYXRhIGFycmF5cyB0byBjcmVhdGUuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJGbG9hdChpbnQgc2l6ZSwgaW50IG51bUJhbmtzKSB7Ci0gICAgICAgIHN1cGVyKFRZUEVfRkxPQVQsIHNpemUsIG51bUJhbmtzKTsKLSAgICAgICAgZGF0YSA9IG5ldyBmbG9hdFtudW1CYW5rc11bXTsKLSAgICAgICAgaW50IGkgPSAwOwotICAgICAgICB3aGlsZSAoaSA8IG51bUJhbmtzKSB7Ci0gICAgICAgICAgICBkYXRhW2krK10gPSBuZXcgZmxvYXRbc2l6ZV07Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZW1wdHkgZGF0YSBidWZmZXIgb2YgdHlwZSBmbG9hdCB3aXRoIGEgc2luZ2xlCi0gICAgICogdW5kZXJseWluZyBhcnJheSBvZiBkYXRhIHN0YXJ0aW5nIGF0IGluZGV4IDAuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlLgotICAgICAqLwotICAgIHB1YmxpYyBEYXRhQnVmZmVyRmxvYXQoaW50IHNpemUpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9GTE9BVCwgc2l6ZSk7Ci0gICAgICAgIGRhdGEgPSBuZXcgZmxvYXRbMV1bXTsKLSAgICAgICAgZGF0YVswXSA9IG5ldyBmbG9hdFtzaXplXTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRFbGVtKGludCBiYW5rLCBpbnQgaSwgaW50IHZhbCkgewotICAgICAgICBkYXRhW2JhbmtdW29mZnNldHNbYmFua10gKyBpXSA9IHZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEVsZW1GbG9hdChpbnQgYmFuaywgaW50IGksIGZsb2F0IHZhbCkgewotICAgICAgICBkYXRhW2JhbmtdW29mZnNldHNbYmFua10gKyBpXSA9IHZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEVsZW1Eb3VibGUoaW50IGJhbmssIGludCBpLCBkb3VibGUgdmFsKSB7Ci0gICAgICAgIGRhdGFbYmFua11bb2Zmc2V0c1tiYW5rXSArIGldID0gKGZsb2F0KXZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEVsZW0oaW50IGksIGludCB2YWwpIHsKLSAgICAgICAgZGF0YVswXVtvZmZzZXQgKyBpXSA9IHZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0RWxlbShpbnQgYmFuaywgaW50IGkpIHsKLSAgICAgICAgcmV0dXJuIChpbnQpKGRhdGFbYmFua11bb2Zmc2V0c1tiYW5rXSArIGldKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXQgZ2V0RWxlbUZsb2F0KGludCBiYW5rLCBpbnQgaSkgewotICAgICAgICByZXR1cm4gZGF0YVtiYW5rXVtvZmZzZXRzW2JhbmtdICsgaV07Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGRvdWJsZSBnZXRFbGVtRG91YmxlKGludCBiYW5rLCBpbnQgaSkgewotICAgICAgICByZXR1cm4gZGF0YVtiYW5rXVtvZmZzZXRzW2JhbmtdICsgaV07Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0RWxlbUZsb2F0KGludCBpLCBmbG9hdCB2YWwpIHsKLSAgICAgICAgZGF0YVswXVtvZmZzZXQgKyBpXSA9IHZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEVsZW1Eb3VibGUoaW50IGksIGRvdWJsZSB2YWwpIHsKLSAgICAgICAgZGF0YVswXVtvZmZzZXQgKyBpXSA9IChmbG9hdCl2YWw7Ci0gICAgICAgIG5vdGlmeUNoYW5nZWQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkYXRhIG9mIHRoZSBzcGVjaWZpZWQgaW50ZXJuYWwgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmFuawotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBkZXNpcmVkIGFycmF5LgotICAgICAqIEByZXR1cm4gdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0W10gZ2V0RGF0YShpbnQgYmFuaykgewotICAgICAgICBub3RpZnlUYWtlbigpOwotICAgICAgICByZXR1cm4gZGF0YVtiYW5rXTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldEVsZW0oaW50IGkpIHsKLSAgICAgICAgcmV0dXJuIChpbnQpKGRhdGFbMF1bb2Zmc2V0ICsgaV0pOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRFbGVtRmxvYXQoaW50IGkpIHsKLSAgICAgICAgcmV0dXJuIGRhdGFbMF1bb2Zmc2V0ICsgaV07Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGRvdWJsZSBnZXRFbGVtRG91YmxlKGludCBpKSB7Ci0gICAgICAgIHJldHVybiBkYXRhWzBdW29mZnNldCArIGldOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJhbmsgZGF0YS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBiYW5rIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0W11bXSBnZXRCYW5rRGF0YSgpIHsKLSAgICAgICAgbm90aWZ5VGFrZW4oKTsKLSAgICAgICAgcmV0dXJuIGRhdGEuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkYXRhIG9mIHRoZSBmaXJzdCBkYXRhIGFycmF5LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0W10gZ2V0RGF0YSgpIHsKLSAgICAgICAgbm90aWZ5VGFrZW4oKTsKLSAgICAgICAgcmV0dXJuIGRhdGFbMF07Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXJJbnQuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9EYXRhQnVmZmVySW50LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDM4MGExMjcuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXJJbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE4MiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgRGF0YUJ1ZmZlckludCBpcyB0aGUgc3ViY2xhc3Mgb2YgRGF0YUJ1ZmZlciBmb3IgdGhlIGNhc2Ugd2hlcmUgdGhlCi0gKiB1bmRlcmx5aW5nIGRhdGEgaXMgb2YgdHlwZSBpbnRlZ2VyLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIERhdGFCdWZmZXJJbnQgZXh0ZW5kcyBEYXRhQnVmZmVyIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBkYXRhLgotICAgICAqLwotICAgIGludCBkYXRhW11bXTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIGludGVnZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFBcnJheXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5cyB0byBjb3B5IHRoZSBkYXRhIGZyb20uCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlIGZyb20gdGhlIGRhdGEgYXJyYXlzLgotICAgICAqIEBwYXJhbSBvZmZzZXRzCi0gICAgICogICAgICAgICAgICB0aGUgc3RhcnRpbmcgaW5kaWNlcyBmb3IgcmVhZGluZyB0aGUgZGF0YSBmcm9tIHRoZSBpbnRlcm5hbAotICAgICAqICAgICAgICAgICAgZGF0YSBhcnJheXMuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJJbnQoaW50IGRhdGFBcnJheXNbXVtdLCBpbnQgc2l6ZSwgaW50IG9mZnNldHNbXSkgewotICAgICAgICBzdXBlcihUWVBFX0lOVCwgc2l6ZSwgZGF0YUFycmF5cy5sZW5ndGgsIG9mZnNldHMpOwotICAgICAgICBkYXRhID0gZGF0YUFycmF5cy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIGludGVnZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFBcnJheXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5cyB0byBjb3B5IHRoZSBkYXRhIGZyb20uCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlIGZyb20gdGhlIGRhdGEgYXJyYXlzLgotICAgICAqLwotICAgIHB1YmxpYyBEYXRhQnVmZmVySW50KGludCBkYXRhQXJyYXlzW11bXSwgaW50IHNpemUpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9JTlQsIHNpemUsIGRhdGFBcnJheXMubGVuZ3RoKTsKLSAgICAgICAgZGF0YSA9IGRhdGFBcnJheXMuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgb2YgdHlwZSBpbnRlZ2VyIHdpdGggYSBzaW5nbGUgdW5kZXJseWluZwotICAgICAqIGFycmF5IG9mIGRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXkgdG8gY29weSB0aGUgZGF0YSBmcm9tLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZS4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnRpbmcgaW5kZXggdG8gdXNlIHdoZW4gcmVhZGluZyB0aGUgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlckludChpbnQgZGF0YUFycmF5W10sIGludCBzaXplLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIHN1cGVyKFRZUEVfSU5ULCBzaXplLCAxLCBvZmZzZXQpOwotICAgICAgICBkYXRhID0gbmV3IGludFsxXVtdOwotICAgICAgICBkYXRhWzBdID0gZGF0YUFycmF5OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIGludGVnZXIgd2l0aCBhIHNpbmdsZSB1bmRlcmx5aW5nCi0gICAgICogYXJyYXkgb2YgZGF0YSBzdGFydGluZyBhdCBpbmRleCAwLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRvIGNvcHkgdGhlIGRhdGEgZnJvbS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJJbnQoaW50IGRhdGFBcnJheVtdLCBpbnQgc2l6ZSkgewotICAgICAgICBzdXBlcihUWVBFX0lOVCwgc2l6ZSk7Ci0gICAgICAgIGRhdGEgPSBuZXcgaW50WzFdW107Ci0gICAgICAgIGRhdGFbMF0gPSBkYXRhQXJyYXk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGVtcHR5IGRhdGEgYnVmZmVyIG9mIHR5cGUgaW50ZWdlciB3aXRoIG9mZnNldHMgZXF1YWwKLSAgICAgKiB0byB6ZXJvLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZSBmcm9tIHRoZSBkYXRhIGFycmF5cy4KLSAgICAgKiBAcGFyYW0gbnVtQmFua3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgZGF0YSBhcnJheXMgdG8gY3JlYXRlLgotICAgICAqLwotICAgIHB1YmxpYyBEYXRhQnVmZmVySW50KGludCBzaXplLCBpbnQgbnVtQmFua3MpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9JTlQsIHNpemUsIG51bUJhbmtzKTsKLSAgICAgICAgZGF0YSA9IG5ldyBpbnRbbnVtQmFua3NdW107Ci0gICAgICAgIGludCBpID0gMDsKLSAgICAgICAgd2hpbGUgKGkgPCBudW1CYW5rcykgewotICAgICAgICAgICAgZGF0YVtpKytdID0gbmV3IGludFtzaXplXTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBlbXB0eSBkYXRhIGJ1ZmZlciBvZiB0eXBlIGludGVnZXIgd2l0aCBhIHNpbmdsZQotICAgICAqIHVuZGVybHlpbmcgYXJyYXkgb2YgZGF0YSBzdGFydGluZyBhdCBpbmRleCAwLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlckludChpbnQgc2l6ZSkgewotICAgICAgICBzdXBlcihUWVBFX0lOVCwgc2l6ZSk7Ci0gICAgICAgIGRhdGEgPSBuZXcgaW50WzFdW107Ci0gICAgICAgIGRhdGFbMF0gPSBuZXcgaW50W3NpemVdOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEVsZW0oaW50IGJhbmssIGludCBpLCBpbnQgdmFsKSB7Ci0gICAgICAgIGRhdGFbYmFua11bb2Zmc2V0c1tiYW5rXSArIGldID0gdmFsOwotICAgICAgICBub3RpZnlDaGFuZ2VkKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0RWxlbShpbnQgaSwgaW50IHZhbCkgewotICAgICAgICBkYXRhWzBdW29mZnNldCArIGldID0gdmFsOwotICAgICAgICBub3RpZnlDaGFuZ2VkKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRFbGVtKGludCBiYW5rLCBpbnQgaSkgewotICAgICAgICByZXR1cm4gZGF0YVtiYW5rXVtvZmZzZXRzW2JhbmtdICsgaV07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBvZiB0aGUgc3BlY2lmaWVkIGludGVybmFsIGRhdGEgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJhbmsKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgZGVzaXJlZCBkYXRhIGFycmF5LgotICAgICAqIEByZXR1cm4gdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldERhdGEoaW50IGJhbmspIHsKLSAgICAgICAgbm90aWZ5VGFrZW4oKTsKLSAgICAgICAgcmV0dXJuIGRhdGFbYmFua107Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRFbGVtKGludCBpKSB7Ci0gICAgICAgIHJldHVybiBkYXRhWzBdW29mZnNldCArIGldOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJhbmsgZGF0YS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBiYW5rIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIGludFtdW10gZ2V0QmFua0RhdGEoKSB7Ci0gICAgICAgIG5vdGlmeVRha2VuKCk7Ci0gICAgICAgIHJldHVybiBkYXRhLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBvZiB0aGUgZmlyc3QgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXREYXRhKCkgewotICAgICAgICBub3RpZnlUYWtlbigpOwotICAgICAgICByZXR1cm4gZGF0YVswXTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvRGF0YUJ1ZmZlclNob3J0LmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvRGF0YUJ1ZmZlclNob3J0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDFiMTFiMjkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXJTaG9ydC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTgxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi0vKioKLSAqIFRoZSBDbGFzcyBEYXRhQnVmZmVyU2hvcnQgaXMgdGhlIHN1YmNsYXNzIG9mIERhdGFCdWZmZXIgZm9yIHRoZSBjYXNlIHdoZXJlCi0gKiB0aGUgdW5kZXJseWluZyBkYXRhIGlzIHNob3J0LgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIERhdGFCdWZmZXJTaG9ydCBleHRlbmRzIERhdGFCdWZmZXIgewotCi0gICAgLyoqCi0gICAgICogVGhlIGRhdGEuCi0gICAgICovCi0gICAgc2hvcnQgZGF0YVtdW107Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgb2YgdHlwZSBzaG9ydC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YUFycmF5cwotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXlzIHRvIGNvcHkgdGhlIGRhdGEgZnJvbS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UgZnJvbSB0aGUgZGF0YSBhcnJheXMuCi0gICAgICogQHBhcmFtIG9mZnNldHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdGFydGluZyBpbmRpY2VzIGZvciByZWFkaW5nIHRoZSBkYXRhIGZyb20gdGhlIGludGVybmFsCi0gICAgICogICAgICAgICAgICBkYXRhIGFycmF5cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlclNob3J0KHNob3J0IGRhdGFBcnJheXNbXVtdLCBpbnQgc2l6ZSwgaW50IG9mZnNldHNbXSkgewotICAgICAgICBzdXBlcihUWVBFX1NIT1JULCBzaXplLCBkYXRhQXJyYXlzLmxlbmd0aCwgb2Zmc2V0cyk7Ci0gICAgICAgIGRhdGEgPSBkYXRhQXJyYXlzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRhdGEgYnVmZmVyIG9mIHR5cGUgc2hvcnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFBcnJheXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5cyB0byBjb3B5IHRoZSBkYXRhIGZyb20uCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlIGZyb20gdGhlIGRhdGEgYXJyYXlzLgotICAgICAqLwotICAgIHB1YmxpYyBEYXRhQnVmZmVyU2hvcnQoc2hvcnQgZGF0YUFycmF5c1tdW10sIGludCBzaXplKSB7Ci0gICAgICAgIHN1cGVyKFRZUEVfU0hPUlQsIHNpemUsIGRhdGFBcnJheXMubGVuZ3RoKTsKLSAgICAgICAgZGF0YSA9IGRhdGFBcnJheXMuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgb2YgdHlwZSBzaG9ydCB3aXRoIGEgc2luZ2xlIHVuZGVybHlpbmcKLSAgICAgKiBhcnJheSBvZiBkYXRhLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRvIGNvcHkgdGhlIGRhdGEgZnJvbS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0aW5nIGluZGV4IHRvIHVzZSB3aGVuIHJlYWRpbmcgdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJTaG9ydChzaG9ydCBkYXRhQXJyYXlbXSwgaW50IHNpemUsIGludCBvZmZzZXQpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9TSE9SVCwgc2l6ZSwgMSwgb2Zmc2V0KTsKLSAgICAgICAgZGF0YSA9IG5ldyBzaG9ydFsxXVtdOwotICAgICAgICBkYXRhWzBdID0gZGF0YUFycmF5OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIHNob3J0IHdpdGggYSBzaW5nbGUgdW5kZXJseWluZwotICAgICAqIGFycmF5IG9mIGRhdGEgc3RhcnRpbmcgYXQgaW5kZXggMC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YUFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBhcnJheSB0byBjb3B5IHRoZSBkYXRhIGZyb20uCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlLgotICAgICAqLwotICAgIHB1YmxpYyBEYXRhQnVmZmVyU2hvcnQoc2hvcnQgZGF0YUFycmF5W10sIGludCBzaXplKSB7Ci0gICAgICAgIHN1cGVyKFRZUEVfU0hPUlQsIHNpemUpOwotICAgICAgICBkYXRhID0gbmV3IHNob3J0WzFdW107Ci0gICAgICAgIGRhdGFbMF0gPSBkYXRhQXJyYXk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRhdGEgYnVmZmVyIG9mIHR5cGUgc2hvcnQgd2l0aCBvZmZzZXRzIGVxdWFsIHRvIHplcm8uCi0gICAgICogCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlIGZyb20gdGhlIGRhdGEgYXJyYXlzLgotICAgICAqIEBwYXJhbSBudW1CYW5rcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBkYXRhIGFycmF5cyB0byBjcmVhdGUuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJTaG9ydChpbnQgc2l6ZSwgaW50IG51bUJhbmtzKSB7Ci0gICAgICAgIHN1cGVyKFRZUEVfU0hPUlQsIHNpemUsIG51bUJhbmtzKTsKLSAgICAgICAgZGF0YSA9IG5ldyBzaG9ydFtudW1CYW5rc11bXTsKLSAgICAgICAgaW50IGkgPSAwOwotICAgICAgICB3aGlsZSAoaSA8IG51bUJhbmtzKSB7Ci0gICAgICAgICAgICBkYXRhW2krK10gPSBuZXcgc2hvcnRbc2l6ZV07Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZW1wdHkgZGF0YSBidWZmZXIgb2YgdHlwZSBzaG9ydCB3aXRoIGEgc2luZ2xlCi0gICAgICogdW5kZXJseWluZyBhcnJheSBvZiBkYXRhIHN0YXJ0aW5nIGF0IGluZGV4IDAuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlLgotICAgICAqLwotICAgIHB1YmxpYyBEYXRhQnVmZmVyU2hvcnQoaW50IHNpemUpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9TSE9SVCwgc2l6ZSk7Ci0gICAgICAgIGRhdGEgPSBuZXcgc2hvcnRbMV1bXTsKLSAgICAgICAgZGF0YVswXSA9IG5ldyBzaG9ydFtzaXplXTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRFbGVtKGludCBiYW5rLCBpbnQgaSwgaW50IHZhbCkgewotICAgICAgICBkYXRhW2JhbmtdW29mZnNldHNbYmFua10gKyBpXSA9IChzaG9ydCl2YWw7Ci0gICAgICAgIG5vdGlmeUNoYW5nZWQoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRFbGVtKGludCBpLCBpbnQgdmFsKSB7Ci0gICAgICAgIGRhdGFbMF1bb2Zmc2V0ICsgaV0gPSAoc2hvcnQpdmFsOwotICAgICAgICBub3RpZnlDaGFuZ2VkKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRFbGVtKGludCBiYW5rLCBpbnQgaSkgewotICAgICAgICByZXR1cm4gKGRhdGFbYmFua11bb2Zmc2V0c1tiYW5rXSArIGldKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkYXRhIG9mIHRoZSBzcGVjaWZpZWQgaW50ZXJuYWwgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmFuawotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBkZXNpcmVkIGRhdGEgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc2hvcnRbXSBnZXREYXRhKGludCBiYW5rKSB7Ci0gICAgICAgIG5vdGlmeVRha2VuKCk7Ci0gICAgICAgIHJldHVybiBkYXRhW2JhbmtdOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0RWxlbShpbnQgaSkgewotICAgICAgICByZXR1cm4gKGRhdGFbMF1bb2Zmc2V0ICsgaV0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJhbmsgZGF0YS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBiYW5rIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHNob3J0W11bXSBnZXRCYW5rRGF0YSgpIHsKLSAgICAgICAgbm90aWZ5VGFrZW4oKTsKLSAgICAgICAgcmV0dXJuIGRhdGEuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkYXRhIG9mIHRoZSBmaXJzdCBkYXRhIGFycmF5LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHNob3J0W10gZ2V0RGF0YSgpIHsKLSAgICAgICAgbm90aWZ5VGFrZW4oKTsKLSAgICAgICAgcmV0dXJuIGRhdGFbMF07Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXJVU2hvcnQuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9EYXRhQnVmZmVyVVNob3J0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDU4ZDlkODMuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL0RhdGFCdWZmZXJVU2hvcnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE5NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBDbGFzcyBEYXRhQnVmZmVyVVNob3J0IGlzIHRoZSBzdWJjbGFzcyBvZiBEYXRhQnVmZmVyIGZvciB0aGUgY2FzZSB3aGVyZQotICogdGhlIHVuZGVybHlpbmcgZGF0YSBpcyB1bnNpZ25lZCBzaG9ydC4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBmaW5hbCBjbGFzcyBEYXRhQnVmZmVyVVNob3J0IGV4dGVuZHMgRGF0YUJ1ZmZlciB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGF0YS4KLSAgICAgKi8KLSAgICBzaG9ydCBkYXRhW11bXTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIHVuc2lnbmVkIHNob3J0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXlzCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBhcnJheXMgdG8gY29weSB0aGUgZGF0YSBmcm9tLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZSBmcm9tIHRoZSBkYXRhIGFycmF5cy4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0aW5nIGluZGljZXMgZm9yIHJlYWRpbmcgdGhlIGRhdGEgZnJvbSB0aGUgaW50ZXJuYWwKLSAgICAgKiAgICAgICAgICAgIGRhdGEgYXJyYXlzLgotICAgICAqLwotICAgIHB1YmxpYyBEYXRhQnVmZmVyVVNob3J0KHNob3J0IGRhdGFBcnJheXNbXVtdLCBpbnQgc2l6ZSwgaW50IG9mZnNldHNbXSkgewotICAgICAgICBzdXBlcihUWVBFX1VTSE9SVCwgc2l6ZSwgZGF0YUFycmF5cy5sZW5ndGgsIG9mZnNldHMpOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGRhdGFBcnJheXMubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIGlmIChkYXRhQXJyYXlzW2ldLmxlbmd0aCA8IG9mZnNldHNbaV0gKyBzaXplKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjI4ZD1MZW5ndGggb2YgZGF0YUFycmF5W3swfV0gaXMgbGVzcyB0aGFuIHNpemUgKwotICAgICAgICAgICAgICAgIC8vIG9mZnNldFt7MX1dCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOEQiLCBpLCBpKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBkYXRhID0gZGF0YUFycmF5cy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkYXRhIGJ1ZmZlciBvZiB0eXBlIHVuc2lnbmVkIHNob3J0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXlzCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBhcnJheXMgdG8gY29weSB0aGUgZGF0YSBmcm9tLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgbGVuZ3RoIChudW1iZXIgb2YgZWxlbWVudHMpIHRvIHVzZSBmcm9tIHRoZSBkYXRhIGFycmF5cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlclVTaG9ydChzaG9ydCBkYXRhQXJyYXlzW11bXSwgaW50IHNpemUpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9VU0hPUlQsIHNpemUsIGRhdGFBcnJheXMubGVuZ3RoKTsKLSAgICAgICAgZGF0YSA9IGRhdGFBcnJheXMuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGF0YSBidWZmZXIgb2YgdHlwZSB1bnNpZ25lZCBzaG9ydCB3aXRoIGEgc2luZ2xlCi0gICAgICogdW5kZXJseWluZyBhcnJheSBvZiBkYXRhLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRvIGNvcHkgdGhlIGRhdGEgZnJvbS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0aW5nIGluZGV4IHRvIHVzZSB3aGVuIHJlYWRpbmcgdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJVU2hvcnQoc2hvcnQgZGF0YUFycmF5W10sIGludCBzaXplLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIHN1cGVyKFRZUEVfVVNIT1JULCBzaXplLCAxLCBvZmZzZXQpOwotICAgICAgICBpZiAoZGF0YUFycmF5Lmxlbmd0aCA8IHNpemUgKyBvZmZzZXQpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yOEU9TGVuZ3RoIG9mIGRhdGFBcnJheSBpcyBsZXNzIHRoYW4gc2l6ZSArIG9mZnNldAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOEUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBkYXRhID0gbmV3IHNob3J0WzFdW107Ci0gICAgICAgIGRhdGFbMF0gPSBkYXRhQXJyYXk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGRhdGEgYnVmZmVyIG9mIHR5cGUgdW5zaWduZWQgc2hvcnQgd2l0aCBhIHNpbmdsZQotICAgICAqIHVuZGVybHlpbmcgYXJyYXkgb2YgZGF0YSBzdGFydGluZyBhdCBpbmRleCAwLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IHRvIGNvcHkgdGhlIGRhdGEgZnJvbS4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJVU2hvcnQoc2hvcnQgZGF0YUFycmF5W10sIGludCBzaXplKSB7Ci0gICAgICAgIHN1cGVyKFRZUEVfVVNIT1JULCBzaXplKTsKLSAgICAgICAgZGF0YSA9IG5ldyBzaG9ydFsxXVtdOwotICAgICAgICBkYXRhWzBdID0gZGF0YUFycmF5OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBlbXB0eSBkYXRhIGJ1ZmZlciBvZiB0eXBlIHVuc2lnbmVkIHNob3J0IHdpdGggb2Zmc2V0cwotICAgICAqIGVxdWFsIHRvIHplcm8uCi0gICAgICogCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggKG51bWJlciBvZiBlbGVtZW50cykgdG8gdXNlIGZyb20gdGhlIGRhdGEgYXJyYXlzLgotICAgICAqIEBwYXJhbSBudW1CYW5rcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBkYXRhIGFycmF5cyB0byBjcmVhdGUuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJVU2hvcnQoaW50IHNpemUsIGludCBudW1CYW5rcykgewotICAgICAgICBzdXBlcihUWVBFX1VTSE9SVCwgc2l6ZSwgbnVtQmFua3MpOwotICAgICAgICBkYXRhID0gbmV3IHNob3J0W251bUJhbmtzXVtdOwotICAgICAgICBpbnQgaSA9IDA7Ci0gICAgICAgIHdoaWxlIChpIDwgbnVtQmFua3MpIHsKLSAgICAgICAgICAgIGRhdGFbaSsrXSA9IG5ldyBzaG9ydFtzaXplXTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBlbXB0eSBkYXRhIGJ1ZmZlciBvZiB0eXBlIHVuc2lnbmVkIHNob3J0IHdpdGggYSBzaW5nbGUKLSAgICAgKiB1bmRlcmx5aW5nIGFycmF5IG9mIGRhdGEgc3RhcnRpbmcgYXQgaW5kZXggMC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCAobnVtYmVyIG9mIGVsZW1lbnRzKSB0byB1c2UuCi0gICAgICovCi0gICAgcHVibGljIERhdGFCdWZmZXJVU2hvcnQoaW50IHNpemUpIHsKLSAgICAgICAgc3VwZXIoVFlQRV9VU0hPUlQsIHNpemUpOwotICAgICAgICBkYXRhID0gbmV3IHNob3J0WzFdW107Ci0gICAgICAgIGRhdGFbMF0gPSBuZXcgc2hvcnRbc2l6ZV07Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0RWxlbShpbnQgYmFuaywgaW50IGksIGludCB2YWwpIHsKLSAgICAgICAgZGF0YVtiYW5rXVtvZmZzZXRzW2JhbmtdICsgaV0gPSAoc2hvcnQpdmFsOwotICAgICAgICBub3RpZnlDaGFuZ2VkKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0RWxlbShpbnQgaSwgaW50IHZhbCkgewotICAgICAgICBkYXRhWzBdW29mZnNldCArIGldID0gKHNob3J0KXZhbDsKLSAgICAgICAgbm90aWZ5Q2hhbmdlZCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0RWxlbShpbnQgYmFuaywgaW50IGkpIHsKLSAgICAgICAgcmV0dXJuIChkYXRhW2JhbmtdW29mZnNldHNbYmFua10gKyBpXSkgJiAweGZmZmY7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBvZiB0aGUgc3BlY2lmaWVkIGludGVybmFsIGRhdGEgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJhbmsKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgZGVzaXJlZCBkYXRhIGFycmF5LgotICAgICAqIEByZXR1cm4gdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHNob3J0W10gZ2V0RGF0YShpbnQgYmFuaykgewotICAgICAgICBub3RpZnlUYWtlbigpOwotICAgICAgICByZXR1cm4gZGF0YVtiYW5rXTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldEVsZW0oaW50IGkpIHsKLSAgICAgICAgcmV0dXJuIChkYXRhWzBdW29mZnNldCArIGldKSAmIDB4ZmZmZjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBiYW5rIGRhdGEuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYmFuayBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBzaG9ydFtdW10gZ2V0QmFua0RhdGEoKSB7Ci0gICAgICAgIG5vdGlmeVRha2VuKCk7Ci0gICAgICAgIHJldHVybiBkYXRhLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBvZiB0aGUgZmlyc3QgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBzaG9ydFtdIGdldERhdGEoKSB7Ci0gICAgICAgIG5vdGlmeVRha2VuKCk7Ci0gICAgICAgIHJldHVybiBkYXRhWzBdOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9EaXJlY3RDb2xvck1vZGVsLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvRGlyZWN0Q29sb3JNb2RlbC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3MDBlYjdhLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9EaXJlY3RDb2xvck1vZGVsLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4ODkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5jb2xvci5Db2xvclNwYWNlOwotaW1wb3J0IGphdmEuYXd0LlRyYW5zcGFyZW5jeTsKLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlzOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvci5MVVRDb2xvckNvbnZlcnRlcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgRGlyZWN0Q29sb3JNb2RlbCByZXByZXNlbnRzIGEgc3RhbmRhcmQgKHBhY2tlZCkgUkdCIGNvbG9yIG1vZGVsCi0gKiB3aXRoIGFkZGl0aW9uYWwgc3VwcG9ydCBmb3IgY29udmVydGluZyBiZXR3ZWVuIHNSR0IgY29sb3Igc3BhY2UgYW5kIDggb3IgMTYKLSAqIGJpdCBsaW5lYXIgUkdCIGNvbG9yIHNwYWNlIHVzaW5nIGxvb2t1cCB0YWJsZXMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgRGlyZWN0Q29sb3JNb2RlbCBleHRlbmRzIFBhY2tlZENvbG9yTW9kZWwgewotCi0gICAgLyoqCi0gICAgICogVGhlIGZyb21fIGxpbmVhIHJfIHJnIGJfIGx1dC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJ5dGUgZnJvbV9MSU5FQVJfUkdCX0xVVFtdOyAvLyBMb29rdXAgdGFibGUgZm9yIGNvbnZlcnNpb24gZnJvbQotCi0gICAgLy8gTGluZWFyIFJHQiBDb2xvciBTcGFjZSBpbnRvIHNSR0IKLQotICAgIC8qKgotICAgICAqIFRoZSB0b18gbGluZWEgcl84IHJnIGJfIGx1dC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJ5dGUgdG9fTElORUFSXzhSR0JfTFVUW107IC8vIExvb2t1cCB0YWJsZSBmb3IgY29udmVyc2lvbiBmcm9tCi0KLSAgICAvLyBzUkdCIENvbG9yIFNwYWNlIGludG8gTGluZWFyIFJHQgotICAgIC8vIDggYml0Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdG9fIGxpbmVhIHJfMTYgcmcgYl8gbHV0LgotICAgICAqLwotICAgIHByaXZhdGUgc2hvcnQgdG9fTElORUFSXzE2UkdCX0xVVFtdOyAvLyBMb29rdXAgdGFibGUgZm9yIGNvbnZlcnNpb24gZnJvbQotCi0gICAgLy8gc1JHQiBDb2xvciBTcGFjZSBpbnRvIExpbmVhciBSR0IKLSAgICAvLyAxNiBiaXQKLQotICAgIC8qKgotICAgICAqIFRoZSBhbHBoYSBsdXQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBieXRlIGFscGhhTFVUW107IC8vIExvb2t1cCB0YWJsZSBmb3Igc2NhbGUgYWxwaGEgdmFsdWUKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciBsdSB0cy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJ5dGUgY29sb3JMVVRzW11bXTsgLy8gTG9va3VwIHRhYmxlcyBmb3Igc2NhbGUgY29sb3IgdmFsdWVzCi0KLSAgICAvKioKLSAgICAgKiBUaGUgaXNfcyByZ2IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGlzX3NSR0I7IC8vIENvbG9yTW9kZWwgaGFzIHNSR0IgQ29sb3JTcGFjZQotCi0gICAgLyoqCi0gICAgICogVGhlIGlzXyBsaW5lYSByXyByZ2IuCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGlzX0xJTkVBUl9SR0I7IC8vIENvbG9yIE1vZGVsIGhhcyBMaW5lYXIgUkdCIENvbG9yCi0KLSAgICAvLyBTcGFjZQotCi0gICAgLyoqCi0gICAgICogVGhlIExJTkVBIHJfIHJnIGJfIGxlbmd0aC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBMSU5FQVJfUkdCX0xlbmd0aDsgLy8gTGluZWFyIFJHQiBiaXQgbGVuZ3RoCi0KLSAgICAvKioKLSAgICAgKiBUaGUgZmFjdG9yLgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgZkZhY3RvcjsgLy8gU2NhbGUgZmFjdG9yCi0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGlyZWN0IGNvbG9yIG1vZGVsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcGFjZQotICAgICAqICAgICAgICAgICAgdGhlIGNvbG9yIHNwYWNlLgotICAgICAqIEBwYXJhbSBiaXRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgY29tcG9uZW50IG1hc2tzLgotICAgICAqIEBwYXJhbSBybWFzawotICAgICAqICAgICAgICAgICAgdGhlIGJpdG1hc2sgY29ycmVzcG9uZGluZyB0byB0aGUgcmVkIGJhbmQuCi0gICAgICogQHBhcmFtIGdtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYml0bWFzayBjb3JyZXNwb25kaW5nIHRvIHRoZSBncmVlbiBiYW5kLgotICAgICAqIEBwYXJhbSBibWFzawotICAgICAqICAgICAgICAgICAgdGhlIGJpdG1hc2sgY29ycmVzcG9uZGluZyB0byB0aGUgYmx1ZSBiYW5kLgotICAgICAqIEBwYXJhbSBhbWFzawotICAgICAqICAgICAgICAgICAgdGhlIGJpdG1hc2sgY29ycmVzcG9uZGluZyB0byB0aGUgYWxwaGEgYmFuZC4KLSAgICAgKiBAcGFyYW0gaXNBbHBoYVByZW11bHRpcGxpZWQKLSAgICAgKiAgICAgICAgICAgIHdoZXRoZXIgdGhlIGFscGhhIGlzIHByZS1tdWx0aXBsaWVkIGluIHRoaXMgY29sb3IgbW9kZWwuCi0gICAgICogQHBhcmFtIHRyYW5zZmVyVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zZmVyIHR5cGUgKHByaW1pdGl2ZSBqYXZhIHR5cGUgdG8gdXNlIGZvciB0aGUKLSAgICAgKiAgICAgICAgICAgIGNvbXBvbmVudHMpLgotICAgICAqIEB0aHJvd3MgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIG51bWJlciBvZiBiaXRzIGluIHRoZSBjb21iaW5lZCBiaXRtYXNrcyBmb3IgdGhlIGNvbG9yCi0gICAgICogICAgICAgICAgICAgYmFuZHMgaXMgbGVzcyB0aGFuIG9uZSBvciBncmVhdGVyIHRoYW4gMzIuCi0gICAgICovCi0gICAgcHVibGljIERpcmVjdENvbG9yTW9kZWwoQ29sb3JTcGFjZSBzcGFjZSwgaW50IGJpdHMsIGludCBybWFzaywgaW50IGdtYXNrLCBpbnQgYm1hc2ssIGludCBhbWFzaywKLSAgICAgICAgICAgIGJvb2xlYW4gaXNBbHBoYVByZW11bHRpcGxpZWQsIGludCB0cmFuc2ZlclR5cGUpIHsKLQotICAgICAgICBzdXBlcihzcGFjZSwgYml0cywgcm1hc2ssIGdtYXNrLCBibWFzaywgYW1hc2ssIGlzQWxwaGFQcmVtdWx0aXBsaWVkLAotICAgICAgICAgICAgICAgIChhbWFzayA9PSAwID8gVHJhbnNwYXJlbmN5Lk9QQVFVRSA6IFRyYW5zcGFyZW5jeS5UUkFOU0xVQ0VOVCksIHRyYW5zZmVyVHlwZSk7Ci0KLSAgICAgICAgaW5pdExVVHMoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZGlyZWN0IGNvbG9yIG1vZGVsLCBkZXRlcm1pbmluZyB0aGUgdHJhbnNmZXIgdHlwZSBmcm9tCi0gICAgICogdGhlIGJpdHMgYXJyYXksIHRoZSB0cmFuc3BhcmVuY3kgZnJvbSB0aGUgYWxwaGEgbWFzaywgYW5kIHRoZSBkZWZhdWx0Ci0gICAgICogY29sb3Igc3BhY2Uge0BsaW5rIENvbG9yU3BhY2UjQ1Nfc1JHQn0uCi0gICAgICogCi0gICAgICogQHBhcmFtIGJpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBjb21wb25lbnQgbWFza3MuCi0gICAgICogQHBhcmFtIHJtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYml0bWFzayBjb3JyZXNwb25kaW5nIHRvIHRoZSByZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gZ21hc2sKLSAgICAgKiAgICAgICAgICAgIHRoZSBiaXRtYXNrIGNvcnJlc3BvbmRpbmcgdG8gdGhlIGdyZWVuIGJhbmQuCi0gICAgICogQHBhcmFtIGJtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYml0bWFzayBjb3JyZXNwb25kaW5nIHRvIHRoZSBibHVlIGJhbmQuCi0gICAgICogQHBhcmFtIGFtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYml0bWFzayBjb3JyZXNwb25kaW5nIHRvIHRoZSBhbHBoYSBiYW5kLgotICAgICAqLwotICAgIHB1YmxpYyBEaXJlY3RDb2xvck1vZGVsKGludCBiaXRzLCBpbnQgcm1hc2ssIGludCBnbWFzaywgaW50IGJtYXNrLCBpbnQgYW1hc2spIHsKLQotICAgICAgICBzdXBlcihDb2xvclNwYWNlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1Nfc1JHQiksIGJpdHMsIHJtYXNrLCBnbWFzaywgYm1hc2ssIGFtYXNrLCBmYWxzZSwKLSAgICAgICAgICAgICAgICAoYW1hc2sgPT0gMCA/IFRyYW5zcGFyZW5jeS5PUEFRVUUgOiBUcmFuc3BhcmVuY3kuVFJBTlNMVUNFTlQpLCBDb2xvck1vZGVsCi0gICAgICAgICAgICAgICAgICAgICAgICAuZ2V0VHJhbnNmZXJUeXBlKGJpdHMpKTsKLQotICAgICAgICBpbml0TFVUcygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBkaXJlY3QgY29sb3IgbW9kZWwgd2l0aCBubyBhbHBoYSBjaGFubmVsLCBkZXRlcm1pbmluZwotICAgICAqIHRoZSB0cmFuc2ZlciB0eXBlIGZyb20gdGhlIGJpdHMgYXJyYXksIHRoZSBkZWZhdWx0IGNvbG9yIHNwYWNlCi0gICAgICoge0BsaW5rIENvbG9yU3BhY2UjQ1Nfc1JHQn0sIGFuZCB3aXRoIHRoZSB0cmFuc3BhcmVuY3kgc2V0IHRvCi0gICAgICoge0BsaW5rIFRyYW5zcGFyZW5jeSNPUEFRVUV9LgotICAgICAqIAotICAgICAqIEBwYXJhbSBiaXRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgY29tcG9uZW50IG1hc2tzLgotICAgICAqIEBwYXJhbSBybWFzawotICAgICAqICAgICAgICAgICAgdGhlIGJpdG1hc2sgY29ycmVzcG9uZGluZyB0byB0aGUgcmVkIGJhbmQuCi0gICAgICogQHBhcmFtIGdtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYml0bWFzayBjb3JyZXNwb25kaW5nIHRvIHRoZSBncmVlbiBiYW5kLgotICAgICAqIEBwYXJhbSBibWFzawotICAgICAqICAgICAgICAgICAgdGhlIGJpdG1hc2sgY29ycmVzcG9uZGluZyB0byB0aGUgYmx1ZSBiYW5kLgotICAgICAqLwotICAgIHB1YmxpYyBEaXJlY3RDb2xvck1vZGVsKGludCBiaXRzLCBpbnQgcm1hc2ssIGludCBnbWFzaywgaW50IGJtYXNrKSB7Ci0gICAgICAgIHRoaXMoYml0cywgcm1hc2ssIGdtYXNrLCBibWFzaywgMCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBnZXREYXRhRWxlbWVudHMoaW50IGNvbXBvbmVudHNbXSwgaW50IG9mZnNldCwgT2JqZWN0IG9iaikgewotICAgICAgICBpbnQgcGl4ZWwgPSAwOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgcGl4ZWwgfD0gKGNvbXBvbmVudHNbb2Zmc2V0ICsgaV0gPDwgb2Zmc2V0c1tpXSkgJiBjb21wb25lbnRNYXNrc1tpXTsKLSAgICAgICAgfQotCi0gICAgICAgIHN3aXRjaCAodHJhbnNmZXJUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGJ5dGUgYmFbXTsKLSAgICAgICAgICAgICAgICBpZiAob2JqID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgYmEgPSBuZXcgYnl0ZVsxXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBiYSA9IChieXRlW10pb2JqOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBiYVswXSA9IChieXRlKXBpeGVsOwotICAgICAgICAgICAgICAgIG9iaiA9IGJhOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2FbXTsKLSAgICAgICAgICAgICAgICBpZiAob2JqID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgc2EgPSBuZXcgc2hvcnRbMV07Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgc2EgPSAoc2hvcnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHNhWzBdID0gKHNob3J0KXBpeGVsOwotICAgICAgICAgICAgICAgIG9iaiA9IHNhOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgaW50IGlhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlhID0gbmV3IGludFsxXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBpYSA9IChpbnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlhWzBdID0gcGl4ZWw7Ci0gICAgICAgICAgICAgICAgb2JqID0gaWE7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjIxND1UaGlzIENvbG9yIE1vZGVsIGRvZXNuJ3Qgc3VwcG9ydCB0aGlzIHRyYW5zZmVyVHlwZQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBvYmo7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBnZXREYXRhRWxlbWVudHMoaW50IHJnYiwgT2JqZWN0IHBpeGVsKSB7Ci0gICAgICAgIGlmIChlcXVhbHMoQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCkpKSB7Ci0gICAgICAgICAgICBpbnQgaWFbXTsKLSAgICAgICAgICAgIGlmIChwaXhlbCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaWEgPSBuZXcgaW50WzFdOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBpYSA9IChpbnRbXSlwaXhlbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlhWzBdID0gcmdiOwotICAgICAgICAgICAgcmV0dXJuIGlhOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGFscGhhID0gKHJnYiA+PiAyNCkgJiAweGZmOwotICAgICAgICBpbnQgcmVkID0gKHJnYiA+PiAxNikgJiAweGZmOwotICAgICAgICBpbnQgZ3JlZW4gPSAocmdiID4+IDgpICYgMHhmZjsKLSAgICAgICAgaW50IGJsdWUgPSByZ2IgJiAweGZmOwotCi0gICAgICAgIGZsb2F0IGNvbXBbXSA9IG5ldyBmbG9hdFtudW1Db2xvckNvbXBvbmVudHNdOwotICAgICAgICBmbG9hdCBub3JtQ29tcFtdID0gbnVsbDsKLQotICAgICAgICBpZiAoaXNfc1JHQiB8fCBpc19MSU5FQVJfUkdCKSB7Ci0gICAgICAgICAgICBpZiAoaXNfTElORUFSX1JHQikgewotICAgICAgICAgICAgICAgIGlmIChMSU5FQVJfUkdCX0xlbmd0aCA9PSA4KSB7Ci0gICAgICAgICAgICAgICAgICAgIHJlZCA9IHRvX0xJTkVBUl84UkdCX0xVVFtyZWRdICYgMHhmZjsKLSAgICAgICAgICAgICAgICAgICAgZ3JlZW4gPSB0b19MSU5FQVJfOFJHQl9MVVRbZ3JlZW5dICYgMHhmZjsKLSAgICAgICAgICAgICAgICAgICAgYmx1ZSA9IHRvX0xJTkVBUl84UkdCX0xVVFtibHVlXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgcmVkID0gdG9fTElORUFSXzE2UkdCX0xVVFtyZWRdICYgMHhmZmZmOwotICAgICAgICAgICAgICAgICAgICBncmVlbiA9IHRvX0xJTkVBUl8xNlJHQl9MVVRbZ3JlZW5dICYgMHhmZmZmOwotICAgICAgICAgICAgICAgICAgICBibHVlID0gdG9fTElORUFSXzE2UkdCX0xVVFtibHVlXSAmIDB4ZmZmZjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjb21wWzBdID0gcmVkIC8gZkZhY3RvcjsKLSAgICAgICAgICAgIGNvbXBbMV0gPSBncmVlbiAvIGZGYWN0b3I7Ci0gICAgICAgICAgICBjb21wWzJdID0gYmx1ZSAvIGZGYWN0b3I7Ci0gICAgICAgICAgICBpZiAoIWhhc0FscGhhKSB7Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXAgPSBjb21wOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBmbG9hdCBub3JtQWxwaGEgPSBhbHBoYSAvIDI1NS4wZjsKLSAgICAgICAgICAgICAgICBub3JtQ29tcCA9IG5ldyBmbG9hdFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbG9yQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIG5vcm1Db21wW2ldID0gY29tcFtpXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXBbbnVtQ29sb3JDb21wb25lbnRzXSA9IG5vcm1BbHBoYTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbXBbMF0gPSByZWQgLyBmRmFjdG9yOwotICAgICAgICAgICAgY29tcFsxXSA9IGdyZWVuIC8gZkZhY3RvcjsKLSAgICAgICAgICAgIGNvbXBbMl0gPSBibHVlIC8gZkZhY3RvcjsKLSAgICAgICAgICAgIGZsb2F0IHJnYkNvbXBbXSA9IGNzLmZyb21SR0IoY29tcCk7Ci0gICAgICAgICAgICBpZiAoIWhhc0FscGhhKSB7Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXAgPSByZ2JDb21wOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBmbG9hdCBub3JtQWxwaGEgPSBhbHBoYSAvIDI1NS4wZjsKLSAgICAgICAgICAgICAgICBub3JtQ29tcCA9IG5ldyBmbG9hdFtudW1Db21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbG9yQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIG5vcm1Db21wW2ldID0gcmdiQ29tcFtpXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbm9ybUNvbXBbbnVtQ29sb3JDb21wb25lbnRzXSA9IG5vcm1BbHBoYTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGludCBweGwgPSAwOwotICAgICAgICBpZiAoaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIGZsb2F0IG5vcm1BbHBoYSA9IG5vcm1Db21wW251bUNvbG9yQ29tcG9uZW50c107Ci0gICAgICAgICAgICBhbHBoYSA9IChpbnQpKG5vcm1BbHBoYSAqIG1heFZhbHVlc1tudW1Db2xvckNvbXBvbmVudHNdICsgMC41Zik7Ci0gICAgICAgICAgICBpZiAoaXNBbHBoYVByZW11bHRpcGxpZWQpIHsKLSAgICAgICAgICAgICAgICByZWQgPSAoaW50KShub3JtQ29tcFswXSAqIG5vcm1BbHBoYSAqIG1heFZhbHVlc1swXSArIDAuNWYpOwotICAgICAgICAgICAgICAgIGdyZWVuID0gKGludCkobm9ybUNvbXBbMV0gKiBub3JtQWxwaGEgKiBtYXhWYWx1ZXNbMV0gKyAwLjVmKTsKLSAgICAgICAgICAgICAgICBibHVlID0gKGludCkobm9ybUNvbXBbMl0gKiBub3JtQWxwaGEgKiBtYXhWYWx1ZXNbMl0gKyAwLjVmKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgcmVkID0gKGludCkobm9ybUNvbXBbMF0gKiBtYXhWYWx1ZXNbMF0gKyAwLjVmKTsKLSAgICAgICAgICAgICAgICBncmVlbiA9IChpbnQpKG5vcm1Db21wWzFdICogbWF4VmFsdWVzWzFdICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgYmx1ZSA9IChpbnQpKG5vcm1Db21wWzJdICogbWF4VmFsdWVzWzJdICsgMC41Zik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBweGwgPSAoYWxwaGEgPDwgb2Zmc2V0c1szXSkgJiBjb21wb25lbnRNYXNrc1szXTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJlZCA9IChpbnQpKG5vcm1Db21wWzBdICogbWF4VmFsdWVzWzBdICsgMC41Zik7Ci0gICAgICAgICAgICBncmVlbiA9IChpbnQpKG5vcm1Db21wWzFdICogbWF4VmFsdWVzWzFdICsgMC41Zik7Ci0gICAgICAgICAgICBibHVlID0gKGludCkobm9ybUNvbXBbMl0gKiBtYXhWYWx1ZXNbMl0gKyAwLjVmKTsKLSAgICAgICAgfQotCi0gICAgICAgIHB4bCB8PSAoKHJlZCA8PCBvZmZzZXRzWzBdKSAmIGNvbXBvbmVudE1hc2tzWzBdKQotICAgICAgICAgICAgICAgIHwgKChncmVlbiA8PCBvZmZzZXRzWzFdKSAmIGNvbXBvbmVudE1hc2tzWzFdKQotICAgICAgICAgICAgICAgIHwgKChibHVlIDw8IG9mZnNldHNbMl0pICYgY29tcG9uZW50TWFza3NbMl0pOwotCi0gICAgICAgIHN3aXRjaCAodHJhbnNmZXJUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGJ5dGUgYmFbXTsKLSAgICAgICAgICAgICAgICBpZiAocGl4ZWwgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBiYSA9IG5ldyBieXRlWzFdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGJhID0gKGJ5dGVbXSlwaXhlbDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYmFbMF0gPSAoYnl0ZSlweGw7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGJhOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2FbXTsKLSAgICAgICAgICAgICAgICBpZiAocGl4ZWwgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBzYSA9IG5ldyBzaG9ydFsxXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBzYSA9IChzaG9ydFtdKXBpeGVsOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBzYVswXSA9IChzaG9ydClweGw7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHNhOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgaW50IGlhW107Ci0gICAgICAgICAgICAgICAgaWYgKHBpeGVsID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgaWEgPSBuZXcgaW50WzFdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGlhID0gKGludFtdKXBpeGVsOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpYVswXSA9IHB4bDsKLSAgICAgICAgICAgICAgICByZXR1cm4gaWE7Ci0KLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjIxND1UaGlzIENvbG9yIE1vZGVsIGRvZXNuJ3Qgc3VwcG9ydCB0aGlzIHRyYW5zZmVyVHlwZQotICAgICAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmaW5hbCBDb2xvck1vZGVsIGNvZXJjZURhdGEoV3JpdGFibGVSYXN0ZXIgcmFzdGVyLCBib29sZWFuIGlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0KLSAgICAgICAgaWYgKCFoYXNBbHBoYSB8fCB0aGlzLmlzQWxwaGFQcmVtdWx0aXBsaWVkID09IGlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0gICAgICAgICAgICByZXR1cm4gdGhpczsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBtaW5YID0gcmFzdGVyLmdldE1pblgoKTsKLSAgICAgICAgaW50IG1pblkgPSByYXN0ZXIuZ2V0TWluWSgpOwotICAgICAgICBpbnQgdyA9IHJhc3Rlci5nZXRXaWR0aCgpOwotICAgICAgICBpbnQgaCA9IHJhc3Rlci5nZXRIZWlnaHQoKTsKLQotICAgICAgICBpbnQgY29tcG9uZW50c1tdID0gbnVsbDsKLSAgICAgICAgaW50IHRyYW5zcGFyZW50Q29tcG9uZW50c1tdID0gbmV3IGludFtudW1Db21wb25lbnRzXTsKLQotICAgICAgICBmbG9hdCBhbHBoYUZhY3RvciA9IG1heFZhbHVlc1tudW1Db2xvckNvbXBvbmVudHNdOwotCi0gICAgICAgIGlmIChpc0FscGhhUHJlbXVsdGlwbGllZCkgewotICAgICAgICAgICAgc3dpdGNoICh0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgaDsgaSsrLCBtaW5ZKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSAwLCB4ID0gbWluWDsgaiA8IHc7IGorKywgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29tcG9uZW50cyA9IHJhc3Rlci5nZXRQaXhlbCh4LCBtaW5ZLCBjb21wb25lbnRzKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoY29tcG9uZW50c1tudW1Db2xvckNvbXBvbmVudHNdID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmFzdGVyLnNldFBpeGVsKHgsIG1pblksIHRyYW5zcGFyZW50Q29tcG9uZW50cyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmxvYXQgYWxwaGEgPSBjb21wb25lbnRzW251bUNvbG9yQ29tcG9uZW50c10gLyBhbHBoYUZhY3RvcjsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1Db2xvckNvbXBvbmVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29tcG9uZW50c1tuXSA9IChpbnQpKGFscGhhICogY29tcG9uZW50c1tuXSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXRQaXhlbCh4LCBtaW5ZLCBjb21wb25lbnRzKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgICAgIC8vIGF3dC4yMTQ9VGhpcyBDb2xvciBNb2RlbCBkb2Vzbid0IHN1cHBvcnQgdGhpcwotICAgICAgICAgICAgICAgICAgICAvLyB0cmFuc2ZlclR5cGUKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxNCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgc3dpdGNoICh0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgaDsgaSsrLCBtaW5ZKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSAwLCB4ID0gbWluWDsgaiA8IHc7IGorKywgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29tcG9uZW50cyA9IHJhc3Rlci5nZXRQaXhlbCh4LCBtaW5ZLCBjb21wb25lbnRzKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoY29tcG9uZW50c1tudW1Db2xvckNvbXBvbmVudHNdICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmxvYXQgYWxwaGEgPSBhbHBoYUZhY3RvciAvIGNvbXBvbmVudHNbbnVtQ29sb3JDb21wb25lbnRzXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1Db2xvckNvbXBvbmVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29tcG9uZW50c1tuXSA9IChpbnQpKGFscGhhICogY29tcG9uZW50c1tuXSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXRQaXhlbCh4LCBtaW5ZLCBjb21wb25lbnRzKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgICAgIC8vIGF3dC4yMTQ9VGhpcyBDb2xvciBNb2RlbCBkb2Vzbid0IHN1cHBvcnQgdGhpcwotICAgICAgICAgICAgICAgICAgICAvLyB0cmFuc2ZlclR5cGUKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxNCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbmV3IERpcmVjdENvbG9yTW9kZWwoY3MsIHBpeGVsX2JpdHMsIGNvbXBvbmVudE1hc2tzWzBdLCBjb21wb25lbnRNYXNrc1sxXSwKLSAgICAgICAgICAgICAgICBjb21wb25lbnRNYXNrc1syXSwgY29tcG9uZW50TWFza3NbM10sIGlzQWxwaGFQcmVtdWx0aXBsaWVkLCB0cmFuc2ZlclR5cGUpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgIC8vIFRoZSBvdXRwdXQgZm9ybWF0IGJhc2VkIG9uIDEuNSByZWxlYXNlIGJlaGF2aW91ci4KLSAgICAgICAgLy8gSXQgY291bGQgYmUgcmV2ZWxlZCBzdWNoIHdheToKLSAgICAgICAgLy8gQnVmZmVyZWRJbWFnZSBiaSA9IG5ldyBCdWZmZXJlZEltYWdlKDEsIDEsCi0gICAgICAgIC8vIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQik7Ci0gICAgICAgIC8vIENvbG9yTW9kZWwgY20gPSBiaS5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIC8vIFN5c3RlbS5vdXQucHJpbnRsbihjbS50b1N0cmluZygpKTsKLSAgICAgICAgU3RyaW5nIHN0ciA9ICJEaXJlY3RDb2xvck1vZGVsOiIgKyAiIHJtYXNrID0gIiArIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgICAgIEludGVnZXIudG9IZXhTdHJpbmcoY29tcG9uZW50TWFza3NbMF0pICsgIiBnbWFzayA9ICIgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgSW50ZWdlci50b0hleFN0cmluZyhjb21wb25lbnRNYXNrc1sxXSkgKyAiIGJtYXNrID0gIiArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICBJbnRlZ2VyLnRvSGV4U3RyaW5nKGNvbXBvbmVudE1hc2tzWzJdKSArICIgYW1hc2sgPSAiICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICghaGFzQWxwaGEgPyAiMCIgOiBJbnRlZ2VyLnRvSGV4U3RyaW5nKGNvbXBvbmVudE1hc2tzWzNdKSk7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICByZXR1cm4gc3RyOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmaW5hbCBpbnRbXSBnZXRDb21wb25lbnRzKE9iamVjdCBwaXhlbCwgaW50IGNvbXBvbmVudHNbXSwgaW50IG9mZnNldCkgewotCi0gICAgICAgIGlmIChjb21wb25lbnRzID09IG51bGwpIHsKLSAgICAgICAgICAgIGNvbXBvbmVudHMgPSBuZXcgaW50W251bUNvbXBvbmVudHMgKyBvZmZzZXRdOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGludFBpeGVsID0gMDsKLQotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJhW10gPSAoYnl0ZVtdKXBpeGVsOwotICAgICAgICAgICAgICAgIGludFBpeGVsID0gYmFbMF0gJiAweGZmOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2FbXSA9IChzaG9ydFtdKXBpeGVsOwotICAgICAgICAgICAgICAgIGludFBpeGVsID0gc2FbMF0gJiAweGZmZmY7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaWFbXSA9IChpbnRbXSlwaXhlbDsKLSAgICAgICAgICAgICAgICBpbnRQaXhlbCA9IGlhWzBdOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4yMkQ9VGhpcyB0cmFuc2ZlclR5cGUgKCB7MH0gKSBpcyBub3Qgc3VwcG9ydGVkIGJ5IHRoaXMKLSAgICAgICAgICAgICAgICAvLyBjb2xvciBtb2RlbAotICAgICAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMkQiLCAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgICAgICB0cmFuc2ZlclR5cGUpKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBnZXRDb21wb25lbnRzKGludFBpeGVsLCBjb21wb25lbnRzLCBvZmZzZXQpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0UmVkKE9iamVjdCBpbkRhdGEpIHsKLSAgICAgICAgaW50IHBpeGVsID0gMDsKLSAgICAgICAgc3dpdGNoICh0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgYnl0ZSBiYVtdID0gKGJ5dGVbXSlpbkRhdGE7Ci0gICAgICAgICAgICAgICAgcGl4ZWwgPSBiYVswXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgICAgICBzaG9ydCBzYVtdID0gKHNob3J0W10paW5EYXRhOwotICAgICAgICAgICAgICAgIHBpeGVsID0gc2FbMF0gJiAweGZmZmY7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaWFbXSA9IChpbnRbXSlpbkRhdGE7Ci0gICAgICAgICAgICAgICAgcGl4ZWwgPSBpYVswXTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjE0PVRoaXMgQ29sb3IgTW9kZWwgZG9lc24ndCBzdXBwb3J0IHRoaXMgdHJhbnNmZXJUeXBlCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxNCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBnZXRSZWQocGl4ZWwpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0UkdCKE9iamVjdCBpbkRhdGEpIHsKLSAgICAgICAgaW50IHBpeGVsID0gMDsKLSAgICAgICAgc3dpdGNoICh0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgYnl0ZSBiYVtdID0gKGJ5dGVbXSlpbkRhdGE7Ci0gICAgICAgICAgICAgICAgcGl4ZWwgPSBiYVswXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgICAgICBzaG9ydCBzYVtdID0gKHNob3J0W10paW5EYXRhOwotICAgICAgICAgICAgICAgIHBpeGVsID0gc2FbMF0gJiAweGZmZmY7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaWFbXSA9IChpbnRbXSlpbkRhdGE7Ci0gICAgICAgICAgICAgICAgcGl4ZWwgPSBpYVswXTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjE0PVRoaXMgQ29sb3IgTW9kZWwgZG9lc24ndCBzdXBwb3J0IHRoaXMgdHJhbnNmZXJUeXBlCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxNCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBnZXRSR0IocGl4ZWwpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0R3JlZW4oT2JqZWN0IGluRGF0YSkgewotICAgICAgICBpbnQgcGl4ZWwgPSAwOwotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJhW10gPSAoYnl0ZVtdKWluRGF0YTsKLSAgICAgICAgICAgICAgICBwaXhlbCA9IGJhWzBdICYgMHhmZjsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNhW10gPSAoc2hvcnRbXSlpbkRhdGE7Ci0gICAgICAgICAgICAgICAgcGl4ZWwgPSBzYVswXSAmIDB4ZmZmZjsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGludCBpYVtdID0gKGludFtdKWluRGF0YTsKLSAgICAgICAgICAgICAgICBwaXhlbCA9IGlhWzBdOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4yMTQ9VGhpcyBDb2xvciBNb2RlbCBkb2Vzbid0IHN1cHBvcnQgdGhpcyB0cmFuc2ZlclR5cGUKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjE0IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGdldEdyZWVuKHBpeGVsKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldEJsdWUoT2JqZWN0IGluRGF0YSkgewotICAgICAgICBpbnQgcGl4ZWwgPSAwOwotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJhW10gPSAoYnl0ZVtdKWluRGF0YTsKLSAgICAgICAgICAgICAgICBwaXhlbCA9IGJhWzBdICYgMHhmZjsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNhW10gPSAoc2hvcnRbXSlpbkRhdGE7Ci0gICAgICAgICAgICAgICAgcGl4ZWwgPSBzYVswXSAmIDB4ZmZmZjsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGludCBpYVtdID0gKGludFtdKWluRGF0YTsKLSAgICAgICAgICAgICAgICBwaXhlbCA9IGlhWzBdOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4yMTQ9VGhpcyBDb2xvciBNb2RlbCBkb2Vzbid0IHN1cHBvcnQgdGhpcyB0cmFuc2ZlclR5cGUKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjE0IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGdldEJsdWUocGl4ZWwpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0QWxwaGEoT2JqZWN0IGluRGF0YSkgewotICAgICAgICBpbnQgcGl4ZWwgPSAwOwotICAgICAgICBzd2l0Y2ggKHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJhW10gPSAoYnl0ZVtdKWluRGF0YTsKLSAgICAgICAgICAgICAgICBwaXhlbCA9IGJhWzBdICYgMHhmZjsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNhW10gPSAoc2hvcnRbXSlpbkRhdGE7Ci0gICAgICAgICAgICAgICAgcGl4ZWwgPSBzYVswXSAmIDB4ZmZmZjsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGludCBpYVtdID0gKGludFtdKWluRGF0YTsKLSAgICAgICAgICAgICAgICBwaXhlbCA9IGlhWzBdOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4yMTQ9VGhpcyBDb2xvciBNb2RlbCBkb2Vzbid0IHN1cHBvcnQgdGhpcyB0cmFuc2ZlclR5cGUKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjE0IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGdldEFscGhhKHBpeGVsKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmluYWwgV3JpdGFibGVSYXN0ZXIgY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKGludCB3LCBpbnQgaCkgewotICAgICAgICBpZiAodyA8PSAwIHx8IGggPD0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjIyRT13IG9yIGggaXMgbGVzcyB0aGFuIG9yIGVxdWFsIHRvIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjJFIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpbnQgYmFuZE1hc2tzW10gPSBjb21wb25lbnRNYXNrcy5jbG9uZSgpOwotCi0gICAgICAgIGlmIChwaXhlbF9iaXRzID4gMTYpIHsKLSAgICAgICAgICAgIHJldHVybiBSYXN0ZXIuY3JlYXRlUGFja2VkUmFzdGVyKERhdGFCdWZmZXIuVFlQRV9JTlQsIHcsIGgsIGJhbmRNYXNrcywgbnVsbCk7Ci0gICAgICAgIH0gZWxzZSBpZiAocGl4ZWxfYml0cyA+IDgpIHsKLSAgICAgICAgICAgIHJldHVybiBSYXN0ZXIuY3JlYXRlUGFja2VkUmFzdGVyKERhdGFCdWZmZXIuVFlQRV9VU0hPUlQsIHcsIGgsIGJhbmRNYXNrcywgbnVsbCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICByZXR1cm4gUmFzdGVyLmNyZWF0ZVBhY2tlZFJhc3RlcihEYXRhQnVmZmVyLlRZUEVfQllURSwgdywgaCwgYmFuZE1hc2tzLCBudWxsKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGlzQ29tcGF0aWJsZVJhc3RlcihSYXN0ZXIgcmFzdGVyKSB7Ci0gICAgICAgIFNhbXBsZU1vZGVsIHNtID0gcmFzdGVyLmdldFNhbXBsZU1vZGVsKCk7Ci0gICAgICAgIGlmICghKHNtIGluc3RhbmNlb2YgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc3Bwc20gPSAoU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbClzbTsKLQotICAgICAgICBpZiAoc3Bwc20uZ2V0TnVtQmFuZHMoKSAhPSBudW1Db21wb25lbnRzKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHJhc3Rlci5nZXRUcmFuc2ZlclR5cGUoKSAhPSB0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBtYXNrQmFuZHNbXSA9IHNwcHNtLmdldEJpdE1hc2tzKCk7Ci0gICAgICAgIHJldHVybiBBcnJheXMuZXF1YWxzKG1hc2tCYW5kcywgY29tcG9uZW50TWFza3MpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0RGF0YUVsZW1lbnQoaW50IGNvbXBvbmVudHNbXSwgaW50IG9mZnNldCkgewotICAgICAgICBpbnQgcGl4ZWwgPSAwOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgcGl4ZWwgfD0gKGNvbXBvbmVudHNbb2Zmc2V0ICsgaV0gPDwgb2Zmc2V0c1tpXSkgJiBjb21wb25lbnRNYXNrc1tpXTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcGl4ZWw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZpbmFsIGludFtdIGdldENvbXBvbmVudHMoaW50IHBpeGVsLCBpbnQgY29tcG9uZW50c1tdLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIGlmIChjb21wb25lbnRzID09IG51bGwpIHsKLSAgICAgICAgICAgIGNvbXBvbmVudHMgPSBuZXcgaW50W251bUNvbXBvbmVudHMgKyBvZmZzZXRdOwotICAgICAgICB9Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICBjb21wb25lbnRzW29mZnNldCArIGldID0gKHBpeGVsICYgY29tcG9uZW50TWFza3NbaV0pID4+IG9mZnNldHNbaV07Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGNvbXBvbmVudHM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZpbmFsIGludCBnZXRSZWQoaW50IHBpeGVsKSB7Ci0gICAgICAgIGlmIChpc19zUkdCKSB7Ci0gICAgICAgICAgICByZXR1cm4gZ2V0Q29tcG9uZW50RnJvbV9zUkdCKHBpeGVsLCAwKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoaXNfTElORUFSX1JHQikgewotICAgICAgICAgICAgcmV0dXJuIGdldENvbXBvbmVudEZyb21fTElORUFSX1JHQihwaXhlbCwgMCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGdldENvbXBvbmVudEZyb21fUkdCKHBpeGVsLCAwKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmluYWwgaW50IGdldFJHQihpbnQgcGl4ZWwpIHsKLSAgICAgICAgcmV0dXJuIChnZXRBbHBoYShwaXhlbCkgPDwgMjQpIHwgKGdldFJlZChwaXhlbCkgPDwgMTYpIHwgKGdldEdyZWVuKHBpeGVsKSA8PCA4KQotICAgICAgICAgICAgICAgIHwgZ2V0Qmx1ZShwaXhlbCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZpbmFsIGludCBnZXRHcmVlbihpbnQgcGl4ZWwpIHsKLSAgICAgICAgaWYgKGlzX3NSR0IpIHsKLSAgICAgICAgICAgIHJldHVybiBnZXRDb21wb25lbnRGcm9tX3NSR0IocGl4ZWwsIDEpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChpc19MSU5FQVJfUkdCKSB7Ci0gICAgICAgICAgICByZXR1cm4gZ2V0Q29tcG9uZW50RnJvbV9MSU5FQVJfUkdCKHBpeGVsLCAxKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZ2V0Q29tcG9uZW50RnJvbV9SR0IocGl4ZWwsIDEpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0Qmx1ZShpbnQgcGl4ZWwpIHsKLSAgICAgICAgaWYgKGlzX3NSR0IpIHsKLSAgICAgICAgICAgIHJldHVybiBnZXRDb21wb25lbnRGcm9tX3NSR0IocGl4ZWwsIDIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChpc19MSU5FQVJfUkdCKSB7Ci0gICAgICAgICAgICByZXR1cm4gZ2V0Q29tcG9uZW50RnJvbV9MSU5FQVJfUkdCKHBpeGVsLCAyKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZ2V0Q29tcG9uZW50RnJvbV9SR0IocGl4ZWwsIDIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0QWxwaGEoaW50IHBpeGVsKSB7Ci0gICAgICAgIGlmICghaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIHJldHVybiAyNTU7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IGEgPSAocGl4ZWwgJiBjb21wb25lbnRNYXNrc1szXSkgPj4+IG9mZnNldHNbM107Ci0gICAgICAgIGlmIChiaXRzWzNdID09IDgpIHsKLSAgICAgICAgICAgIHJldHVybiBhOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBhbHBoYUxVVFthXSAmIDB4ZmY7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcmVkIG1hc2suCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgcmVkIG1hc2suCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGludCBnZXRSZWRNYXNrKCkgewotICAgICAgICByZXR1cm4gY29tcG9uZW50TWFza3NbMF07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZ3JlZW4gbWFzay4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBncmVlbiBtYXNrLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0R3JlZW5NYXNrKCkgewotICAgICAgICByZXR1cm4gY29tcG9uZW50TWFza3NbMV07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYmx1ZSBtYXNrLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJsdWUgbWFzay4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldEJsdWVNYXNrKCkgewotICAgICAgICByZXR1cm4gY29tcG9uZW50TWFza3NbMl07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYWxwaGEgbWFzay4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhbHBoYSBtYXNrLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0QWxwaGFNYXNrKCkgewotICAgICAgICBpZiAoaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIHJldHVybiBjb21wb25lbnRNYXNrc1szXTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbml0aWFsaXphdGlvbiBvZiBMb29rdXAgdGFibGVzLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBpbml0TFVUcygpIHsKLSAgICAgICAgaXNfc1JHQiA9IGNzLmlzQ1Nfc1JHQigpOwotICAgICAgICBpc19MSU5FQVJfUkdCID0gKGNzID09IExVVENvbG9yQ29udmVydGVyLkxJTkVBUl9SR0JfQ1MpOwotCi0gICAgICAgIGlmIChpc19MSU5FQVJfUkdCKSB7Ci0gICAgICAgICAgICBpZiAobWF4Qml0TGVuZ3RoID4gOCkgewotICAgICAgICAgICAgICAgIExJTkVBUl9SR0JfTGVuZ3RoID0gMTY7Ci0gICAgICAgICAgICAgICAgZnJvbV9MSU5FQVJfUkdCX0xVVCA9IExVVENvbG9yQ29udmVydGVyLmdldEZyb20xNmxSR0J0b3NSR0JfTFVUKCk7Ci0gICAgICAgICAgICAgICAgdG9fTElORUFSXzE2UkdCX0xVVCA9IExVVENvbG9yQ29udmVydGVyLmdldEZyb21zUkdCdG8xNmxSR0JfTFVUKCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIExJTkVBUl9SR0JfTGVuZ3RoID0gODsKLSAgICAgICAgICAgICAgICBmcm9tX0xJTkVBUl9SR0JfTFVUID0gTFVUQ29sb3JDb252ZXJ0ZXIuZ2V0RnJvbThsUkdCdG9zUkdCX0xVVCgpOwotICAgICAgICAgICAgICAgIHRvX0xJTkVBUl84UkdCX0xVVCA9IExVVENvbG9yQ29udmVydGVyLmdldEZyb21zUkdCdG84bFJHQl9MVVQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGZGYWN0b3IgPSAoKDEgPDwgTElORUFSX1JHQl9MZW5ndGgpIC0gMSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmRmFjdG9yID0gMjU1LjBmOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGhhc0FscGhhICYmIGJpdHNbM10gIT0gOCkgewotICAgICAgICAgICAgYWxwaGFMVVQgPSBuZXcgYnl0ZVttYXhWYWx1ZXNbM10gKyAxXTsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDw9IG1heFZhbHVlc1szXTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgYWxwaGFMVVRbaV0gPSAoYnl0ZSkoc2NhbGVzWzNdICogaSArIDAuNWYpOwotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoIWlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0gICAgICAgICAgICBjb2xvckxVVHMgPSBuZXcgYnl0ZVszXVtdOwotCi0gICAgICAgICAgICBpZiAoaXNfc1JHQikgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQ29sb3JDb21wb25lbnRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGJpdHNbaV0gIT0gOCkgewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDA7IGogPCBpOyBqKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoYml0c1tpXSA9PSBiaXRzW2pdKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG9yTFVUc1tpXSA9IGNvbG9yTFVUc1tqXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgY29sb3JMVVRzW2ldID0gbmV3IGJ5dGVbbWF4VmFsdWVzW2ldICsgMV07Ci0gICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBqID0gMDsgaiA8PSBtYXhWYWx1ZXNbaV07IGorKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG9yTFVUc1tpXVtqXSA9IChieXRlKShzY2FsZXNbaV0gKiBqICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChpc19MSU5FQVJfUkdCKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1Db2xvckNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBpZiAoYml0c1tpXSAhPSBMSU5FQVJfUkdCX0xlbmd0aCkgewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDA7IGogPCBpOyBqKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoYml0c1tpXSA9PSBiaXRzW2pdKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG9yTFVUc1tpXSA9IGNvbG9yTFVUc1tqXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgY29sb3JMVVRzW2ldID0gbmV3IGJ5dGVbbWF4VmFsdWVzW2ldICsgMV07Ci0gICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBqID0gMDsgaiA8PSBtYXhWYWx1ZXNbMF07IGorKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBpZHg7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKExJTkVBUl9SR0JfTGVuZ3RoID09IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWR4ID0gKGludCkoc2NhbGVzW2ldICogaiArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlkeCA9IChpbnQpKHNjYWxlc1tpXSAqIGogKiAyNTcuMGYgKyAwLjVmKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sb3JMVVRzW2ldW2pdID0gZnJvbV9MSU5FQVJfUkdCX0xVVFtpZHhdOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCByZXR1cm4gUkdCIGNvbXBvbmVudCB2YWx1ZSBpZiBDb2xvciBNb2RlbCBoYXMgc1JHQgotICAgICAqIENvbG9yU3BhY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsCi0gICAgICogICAgICAgICAgICB0aGUgaW50ZWdlciByZXByZXNlbnRhdGlvbiBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIGlkeAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBwaXhlbCBjb21wb25lbnQuCi0gICAgICogQHJldHVybiB0aGUgdmFsdWUgb2YgdGhlIHBpeGVsIGNvbXBvbmVudCBzY2FsZWQgZnJvbSAwIHRvIDI1NS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBnZXRDb21wb25lbnRGcm9tX3NSR0IoaW50IHBpeGVsLCBpbnQgaWR4KSB7Ci0gICAgICAgIGludCBjb21wID0gKHBpeGVsICYgY29tcG9uZW50TWFza3NbaWR4XSkgPj4gb2Zmc2V0c1tpZHhdOwotICAgICAgICBpZiAoaXNBbHBoYVByZW11bHRpcGxpZWQpIHsKLSAgICAgICAgICAgIGludCBhbHBoYSA9IChwaXhlbCAmIGNvbXBvbmVudE1hc2tzWzNdKSA+Pj4gb2Zmc2V0c1szXTsKLSAgICAgICAgICAgIGNvbXAgPSBhbHBoYSA9PSAwID8gMCA6IChpbnQpKHNjYWxlc1tpZHhdICogY29tcCAqIDI1NS4wZiAvIChzY2FsZXNbM10gKiBhbHBoYSkgKyAwLjVmKTsKLSAgICAgICAgfSBlbHNlIGlmIChiaXRzW2lkeF0gIT0gOCkgewotICAgICAgICAgICAgY29tcCA9IGNvbG9yTFVUc1tpZHhdW2NvbXBdICYgMHhmZjsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gY29tcDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCByZXR1cm4gUkdCIGNvbXBvbmVudCB2YWx1ZSBpZiBDb2xvciBNb2RlbCBoYXMgTGluZWFyIFJHQgotICAgICAqIENvbG9yU3BhY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsCi0gICAgICogICAgICAgICAgICB0aGUgaW50ZWdlciByZXByZXNlbnRhdGlvbiBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIGlkeAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBwaXhlbCBjb21wb25lbnQuCi0gICAgICogQHJldHVybiB0aGUgdmFsdWUgb2YgdGhlIHBpeGVsIGNvbXBvbmVudCBzY2FsZWQgZnJvbSAwIHRvIDI1NS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBnZXRDb21wb25lbnRGcm9tX0xJTkVBUl9SR0IoaW50IHBpeGVsLCBpbnQgaWR4KSB7Ci0gICAgICAgIGludCBjb21wID0gKHBpeGVsICYgY29tcG9uZW50TWFza3NbaWR4XSkgPj4gb2Zmc2V0c1tpZHhdOwotICAgICAgICBpZiAoaXNBbHBoYVByZW11bHRpcGxpZWQpIHsKLSAgICAgICAgICAgIGZsb2F0IGZhY3RvciA9ICgoMSA8PCBMSU5FQVJfUkdCX0xlbmd0aCkgLSAxKTsKLSAgICAgICAgICAgIGludCBhbHBoYSA9IChwaXhlbCAmIGNvbXBvbmVudE1hc2tzWzNdKSA+PiBvZmZzZXRzWzNdOwotICAgICAgICAgICAgY29tcCA9IGFscGhhID09IDAgPyAwIDogKGludCkoc2NhbGVzW2lkeF0gKiBjb21wICogZmFjdG9yIC8gKHNjYWxlc1szXSAqIGFscGhhKSArIDAuNWYpOwotICAgICAgICB9IGVsc2UgaWYgKGJpdHNbaWR4XSAhPSBMSU5FQVJfUkdCX0xlbmd0aCkgewotICAgICAgICAgICAgY29tcCA9IGNvbG9yTFVUc1tpZHhdW2NvbXBdICYgMHhmZjsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbXAgPSBmcm9tX0xJTkVBUl9SR0JfTFVUW2NvbXBdICYgMHhmZjsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gY29tcDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCByZXR1cm4gUkdCIGNvbXBvbmVudCB2YWx1ZSBpZiBDb2xvciBNb2RlbCBoYXMgYXJiaXRyYXJ5IFJHQgotICAgICAqIENvbG9yU2FwY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsCi0gICAgICogICAgICAgICAgICB0aGUgaW50ZWdlciByZXByZXNlbnRhdGlvbiBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIGlkeAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBwaXhlbCBjb21wb25lbnQuCi0gICAgICogQHJldHVybiB0aGUgdmFsdWUgb2YgdGhlIHBpeGVsIGNvbXBvbmVudCBzY2FsZWQgZnJvbSAwIHRvIDI1NS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBnZXRDb21wb25lbnRGcm9tX1JHQihpbnQgcGl4ZWwsIGludCBpZHgpIHsKLSAgICAgICAgaW50IGNvbXBvbmVudHNbXSA9IGdldENvbXBvbmVudHMocGl4ZWwsIG51bGwsIDApOwotICAgICAgICBmbG9hdFtdIG5vcm1Db21wb25lbnRzID0gZ2V0Tm9ybWFsaXplZENvbXBvbmVudHMoY29tcG9uZW50cywgMCwgbnVsbCwgMCk7Ci0gICAgICAgIGZsb2F0W10gc1JHQmNvbXBvbmVudHMgPSBjcy50b1JHQihub3JtQ29tcG9uZW50cyk7Ci0gICAgICAgIHJldHVybiAoaW50KShzUkdCY29tcG9uZW50c1tpZHhdICogMjU1LjBmICsgMC41Zik7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvRmlsdGVyZWRJbWFnZVNvdXJjZS5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL0ZpbHRlcmVkSW1hZ2VTb3VyY2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZWQ4NTU4ZC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvRmlsdGVyZWRJbWFnZVNvdXJjZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsOTggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLnV0aWwuSGFzaHRhYmxlOwotCi0vKioKLSAqIFRoZSBGaWx0ZXJlZEltYWdlU291cmNlIGNsYXNzIGlzIHVzZWQgZm9yIHByb2R1Y2luZyBpbWFnZSBkYXRhIGZvciBhIG5ldwotICogZmlsdGVyZWQgdmVyc2lvbiBvZiB0aGUgb3JpZ2luYWwgaW1hZ2UgdXNpbmcgdGhlIHNwZWNpZmllZCBmaWx0ZXIgb2JqZWN0LgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEZpbHRlcmVkSW1hZ2VTb3VyY2UgaW1wbGVtZW50cyBJbWFnZVByb2R1Y2VyIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBzb3VyY2UuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBJbWFnZVByb2R1Y2VyIHNvdXJjZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBmaWx0ZXIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBJbWFnZUZpbHRlciBmaWx0ZXI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29ucyB0YWJsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIEhhc2h0YWJsZTxJbWFnZUNvbnN1bWVyLCBJbWFnZUNvbnN1bWVyPiBjb25zVGFibGUgPSBuZXcgSGFzaHRhYmxlPEltYWdlQ29uc3VtZXIsIEltYWdlQ29uc3VtZXI+KCk7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRmlsdGVyZWRJbWFnZVNvdXJjZSBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICogSW1hZ2VQcm9kdWNlciBhbmQgdGhlIEltYWdlRmlsdGVyIG9iamVjdHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9yaWcKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgSW1hZ2VQcm9kdWNlci4KLSAgICAgKiBAcGFyYW0gaW1nZgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBJbWFnZUZpbHRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgRmlsdGVyZWRJbWFnZVNvdXJjZShJbWFnZVByb2R1Y2VyIG9yaWcsIEltYWdlRmlsdGVyIGltZ2YpIHsKLSAgICAgICAgc291cmNlID0gb3JpZzsKLSAgICAgICAgZmlsdGVyID0gaW1nZjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIGJvb2xlYW4gaXNDb25zdW1lcihJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIGlmIChpYyAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gY29uc1RhYmxlLmNvbnRhaW5zS2V5KGljKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc3RhcnRQcm9kdWN0aW9uKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgYWRkQ29uc3VtZXIoaWMpOwotICAgICAgICBJbWFnZUNvbnN1bWVyIGZpYyA9IGNvbnNUYWJsZS5nZXQoaWMpOwotICAgICAgICBzb3VyY2Uuc3RhcnRQcm9kdWN0aW9uKGZpYyk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgcmVxdWVzdFRvcERvd25MZWZ0UmlnaHRSZXNlbmQoSW1hZ2VDb25zdW1lciBpYykgewotICAgICAgICBpZiAoaWMgIT0gbnVsbCAmJiBpc0NvbnN1bWVyKGljKSkgewotICAgICAgICAgICAgSW1hZ2VGaWx0ZXIgZmljID0gKEltYWdlRmlsdGVyKWNvbnNUYWJsZS5nZXQoaWMpOwotICAgICAgICAgICAgZmljLnJlc2VuZFRvcERvd25MZWZ0UmlnaHQoc291cmNlKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCByZW1vdmVDb25zdW1lcihJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIGlmIChpYyAhPSBudWxsICYmIGlzQ29uc3VtZXIoaWMpKSB7Ci0gICAgICAgICAgICBJbWFnZUNvbnN1bWVyIGZpYyA9IGNvbnNUYWJsZS5nZXQoaWMpOwotICAgICAgICAgICAgc291cmNlLnJlbW92ZUNvbnN1bWVyKGZpYyk7Ci0gICAgICAgICAgICBjb25zVGFibGUucmVtb3ZlKGljKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCBhZGRDb25zdW1lcihJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIGlmIChpYyAhPSBudWxsICYmICFpc0NvbnN1bWVyKGljKSkgewotICAgICAgICAgICAgSW1hZ2VDb25zdW1lciBmaWMgPSBmaWx0ZXIuZ2V0RmlsdGVySW5zdGFuY2UoaWMpOwotICAgICAgICAgICAgc291cmNlLmFkZENvbnN1bWVyKGZpYyk7Ci0gICAgICAgICAgICBjb25zVGFibGUucHV0KGljLCBmaWMpOwotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0ltYWdlQ29uc3VtZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9JbWFnZUNvbnN1bWVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGNhZjg3ZDEuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL0ltYWdlQ29uc3VtZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE4NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEudXRpbC5IYXNodGFibGU7Ci0KLS8qKgotICogVGhlIEltYWdlQ29uc3VtZXIgaW50ZXJmYWNlIHByb3ZpZGVzIHRoZSBkYXRhIGFib3V0IHRoZSBpbWFnZSBhbmQgYWJvdXQgaG93Ci0gKiBpdHMgZGF0YSBpcyBkZWxpdmVyZWQuIEEgSW1hZ2VQcm9kdWNlciBwcm92aWRlcyBhbGwgb2YgdGhlIGluZm9ybWF0aW9uIGFib3V0Ci0gKiB0aGUgaW1hZ2UgdXNpbmcgdGhlIG1ldGhvZHMgZGVmaW5lZCBpbiB0aGlzIGludGVyZmFjZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgSW1hZ2VDb25zdW1lciB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgUkFORE9NUElYRUxPUkRFUiBpbmRpY2F0ZXMgdGhhdCB0aGUgcGl4ZWxzIGFyZSBkZWxpdmVyZWQgaW4KLSAgICAgKiBhIHJhbmRvbSBvcmRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBSQU5ET01QSVhFTE9SREVSID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBUT1BET1dOTEVGVFJJR0hUIGluZGljYXRlcyB0aGF0IHRoZSBwaXhlbHMgYXJlIGRlbGl2ZXJlZCBpbgotICAgICAqIHRvcC1kb3duLCBsZWZ0LXRvLXJpZ2h0IG9yZGVyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFRPUERPV05MRUZUUklHSFQgPSAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENPTVBMRVRFU0NBTkxJTkVTIGluZGljYXRlcyB0aGF0IHRoZSBwaXhlbHMgYXJlIGRlbGl2ZXJlZCBpbgotICAgICAqIGNvbXBsZXRlIHNjYW5saW5lLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IENPTVBMRVRFU0NBTkxJTkVTID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTSU5HTEVQQVNTIGluZGljYXRlcyB0aGF0IHBpeGVscyBhcmUgZGVsaXZlcmVkIGluIGEgc2luZ2xlCi0gICAgICogcGFzcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTSU5HTEVQQVNTID0gODsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTSU5HTEVGUkFNRSBpbmRpY2F0ZXMgdGhhdCBpbWFnZSBjb25zaXN0cyBvZiBzaW5nbGUgZnJhbWUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU0lOR0xFRlJBTUUgPSAxNjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBJTUFHRUVSUk9SIGluZGljYXRlcyBhbiBpbWFnZSBlcnJvciBkdXJpbmcgaW1hZ2UgcHJvZHVjaW5nLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IElNQUdFRVJST1IgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFNJTkdMRUZSQU1FRE9ORSBpbmRpY2F0ZXMgdGhhdCBvbmx5IG9uZSBvZiB0aGUgaW1hZ2UncwotICAgICAqIGZyYW1lcyBpcyBjb21wbGV0ZWQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU0lOR0xFRlJBTUVET05FID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTVEFUSUNJTUFHRURPTkUgaW5kaWNhdGVzIHRoYXQgdGhlIGltYWdlIGlzIGNvbXBsZXRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTVEFUSUNJTUFHRURPTkUgPSAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IElNQUdFQUJPUlRFRCBpbmRpY2F0ZXMgdGhhdCB0aGUgaW1hZ2UgcHJvZHVjaW5nIHByb2Nlc3MgaXMKLSAgICAgKiBhYm9ydGVkLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IElNQUdFQUJPUlRFRCA9IDQ7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBwcm9wZXJ0aWVzIGZvciB0aGUgaW1hZ2UgYXNzb2NpYXRlZCB3aXRoIHRoaXMgSW1hZ2VDb25zdW1lci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvcHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wZXJ0aWVzIGZvciB0aGUgaW1hZ2UgYXNzb2NpYXRlZCB3aXRoIHRoaXMKLSAgICAgKiAgICAgICAgICAgIEltYWdlQ29uc3VtZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0UHJvcGVydGllcyhIYXNodGFibGU8PywgPz4gcHJvcHMpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgQ29sb3JNb2RlbCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1vZGVsCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IENvbG9yTW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Q29sb3JNb2RlbChDb2xvck1vZGVsIG1vZGVsKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHBpeGVscyBmb3IgdGhlIHNwZWNpZmllZCByZWN0YW5ndWxhciBhcmVhIG9mIHRoZSBpbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiByZWN0YW5ndWxhciBhcmVhLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiByZWN0YW5ndWxhciBhcmVhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIG1vZGVsCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIENvbG9yTW9kZWwgdG8gYmUgdXNlZCBmb3IgcGl4ZWxzIGNvbnZlcnRpbmcuCi0gICAgICogQHBhcmFtIHBpeGVscwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IG9mIHBpeGVscyBhcnJheS4KLSAgICAgKiBAcGFyYW0gc2NhbnNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSBmcm9tIHRoZSBvbmUgcm93IG9mIHBpeGVscyB0byB0aGUgbmV4dCByb3cgaW4gdGhlCi0gICAgICogICAgICAgICAgICBzcGVjaWZpZWQgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBDb2xvck1vZGVsIG1vZGVsLCBpbnRbXSBwaXhlbHMsIGludCBvZmYsCi0gICAgICAgICAgICBpbnQgc2NhbnNpemUpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgcGl4ZWxzIGZvciB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YgdGhlIGltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0gbW9kZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgQ29sb3JNb2RlbCB0byBiZSB1c2VkIGZvciBwaXhlbHMgY29udmVydGluZy4KLSAgICAgKiBAcGFyYW0gcGl4ZWxzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgb2YgcGl4ZWxzIGFycmF5LgotICAgICAqIEBwYXJhbSBzY2Fuc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGRpc3RhbmNlIGZyb20gdGhlIG9uZSByb3cgb2YgcGl4ZWxzIHRvIHRoZSBuZXh0IHJvdyBpbiB0aGUKLSAgICAgKiAgICAgICAgICAgIHNwZWNpZmllZCBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIENvbG9yTW9kZWwgbW9kZWwsIGJ5dGVbXSBwaXhlbHMsIGludCBvZmYsCi0gICAgICAgICAgICBpbnQgc2NhbnNpemUpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGltZW5zaW9ucyBvZiBhIHNvdXJjZSBpbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgaW1hZ2UuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgaW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RGltZW5zaW9ucyhpbnQgd2lkdGgsIGludCBoZWlnaHQpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgaGludCBmbGFncyBvZiBwaXhlbHMgb3JkZXIsIHdoaWNoIGlzIHVzZWQgYnkgdGhlIEltYWdlQ29uc3VtZXIKLSAgICAgKiBmb3Igb2J0YWluaW5nIHBpeGVscyBmcm9tIHRoZSBJbWFnZVByb2R1Y2VyIGZvciB3aGljaCB0aGlzIEltYWdlQ29uc3VtZXIKLSAgICAgKiBpcyBhZGRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaGludGZsYWdzCi0gICAgICogICAgICAgICAgICB0aGUgbWFzayBvZiBoaW50IGZsYWdzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEhpbnRzKGludCBoaW50ZmxhZ3MpOwotCi0gICAgLyoqCi0gICAgICogVEhpcyBtZXRob2QgaXMgY2FsbGVkIGluIHRoZSBvbmUgb2YgdGhlIGZvbGxvd2luZyBjYXNlczoKLSAgICAgKiA8dWw+Ci0gICAgICogPGxpPlRoZSBJbWFnZVByb2R1Y2VyIChmb3Igd2hpY2ggdGhpcyBJbWFnZUNvbnN1bWVyIGlzIGFkZGVkKSBoYXMgYmVlbgotICAgICAqIGRlbGl2ZXJlZCBhbGwgcGl4ZWxzIG9mIHRoZSBzb3VyY2UgaW1hZ2UuPC9saT4KLSAgICAgKiA8bGk+QSBvbmUgZnJhbWUgb2YgYW4gYW5pbWF0aW9uIGhhcyBiZWVuIGNvbXBsZXRlZC48L2xpPgotICAgICAqIDxsaT5BbiBlcnJvciB3aGlsZSBsb2FkaW5nIG9yIHByb2R1Y2luZyBvZiB0aGUgaW1hZ2UgaGFzIG9jY3VycmVkLgotICAgICAqIDwvdWw+Ci0gICAgICogCi0gICAgICogQHBhcmFtIHN0YXR1cwotICAgICAqICAgICAgICAgICAgdGhlIHN0YXR1cyBvZiBpbWFnZSBwcm9kdWNpbmcuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgaW1hZ2VDb21wbGV0ZShpbnQgc3RhdHVzKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0ltYWdlRmlsdGVyLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvSW1hZ2VGaWx0ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDJjOWY1MC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvSW1hZ2VGaWx0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEzNCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEudXRpbC5IYXNodGFibGU7Ci0KLS8qKgotICogVGhlIEltYWdlRmlsdGVyIGNsYXNzIHByb3ZpZGVzIGEgZmlsdGVyIGZvciBkZWxpdmVyaW5nIGltYWdlIGRhdGEgZnJvbSBhbgotICogSW1hZ2VQcm9kdWNlciB0byBhbiBJbWFnZUNvbnN1bWVyLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEltYWdlRmlsdGVyIGltcGxlbWVudHMgSW1hZ2VDb25zdW1lciwgQ2xvbmVhYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb25zdW1lci4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgSW1hZ2VDb25zdW1lciBjb25zdW1lcjsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJbWFnZUZpbHRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VGaWx0ZXIoKSB7Ci0gICAgICAgIHN1cGVyKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBpbnN0YW5jZSBvZiBhbiBJbWFnZUZpbHRlciBvYmplY3Qgd2hpY2ggcGVyZm9ybXMgdGhlIGZpbHRlcmluZwotICAgICAqIGZvciB0aGUgc3BlY2lmaWVkIEltYWdlQ29uc3VtZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGljCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIEltYWdlQ29uc3VtZXIuCi0gICAgICogQHJldHVybiBhbiBJbWFnZUZpbHRlciB1c2VkIHRvIHBlcmZvcm0gdGhlIGZpbHRlcmluZyBmb3IgdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgSW1hZ2VDb25zdW1lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VGaWx0ZXIgZ2V0RmlsdGVySW5zdGFuY2UoSW1hZ2VDb25zdW1lciBpYykgewotICAgICAgICBJbWFnZUZpbHRlciBmaWx0ZXIgPSAoSW1hZ2VGaWx0ZXIpY2xvbmUoKTsKLSAgICAgICAgZmlsdGVyLmNvbnN1bWVyID0gaWM7Ci0gICAgICAgIHJldHVybiBmaWx0ZXI7Ci0gICAgfQotCi0gICAgQFN1cHByZXNzV2FybmluZ3MoInVuY2hlY2tlZCIpCi0gICAgcHVibGljIHZvaWQgc2V0UHJvcGVydGllcyhIYXNodGFibGU8PywgPz4gcHJvcHMpIHsKLSAgICAgICAgSGFzaHRhYmxlPE9iamVjdCwgT2JqZWN0PiBmcHJvcHM7Ci0gICAgICAgIGlmIChwcm9wcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICBmcHJvcHMgPSBuZXcgSGFzaHRhYmxlPE9iamVjdCwgT2JqZWN0PigpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZnByb3BzID0gKEhhc2h0YWJsZTxPYmplY3QsIE9iamVjdD4pcHJvcHMuY2xvbmUoKTsKLSAgICAgICAgfQotICAgICAgICBTdHJpbmcgcHJvcE5hbWUgPSAiRmlsdGVycyI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgU3RyaW5nIHByb3AgPSAiTnVsbCBmaWx0ZXIiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIE9iamVjdCBvID0gZnByb3BzLmdldChwcm9wTmFtZSk7Ci0gICAgICAgIGlmIChvICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChvIGluc3RhbmNlb2YgU3RyaW5nKSB7Ci0gICAgICAgICAgICAgICAgcHJvcCA9IChTdHJpbmcpbyArICI7ICIgKyBwcm9wOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHByb3AgPSBvLnRvU3RyaW5nKCkgKyAiOyAiICsgcHJvcDsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGZwcm9wcy5wdXQocHJvcE5hbWUsIHByb3ApOwotICAgICAgICBjb25zdW1lci5zZXRQcm9wZXJ0aWVzKGZwcm9wcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIGNvcHkgb2YgdGhpcyBJbWFnZUZpbHRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGEgY29weSBvZiB0aGlzIEltYWdlRmlsdGVyLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgY2xvbmUoKSB7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gc3VwZXIuY2xvbmUoKTsKLSAgICAgICAgfSBjYXRjaCAoQ2xvbmVOb3RTdXBwb3J0ZWRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXNwb25kcyB0byBhIHJlcXVlc3QgZm9yIGEgVG9wLURvd24tTGVmdC1SaWdodCBvcmRlcmVkIHJlc2VuZCBvZiB0aGUKLSAgICAgKiBwaXhlbCBkYXRhIGZyb20gYW4gSW1hZ2VDb25zdW1lci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaXAKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVByb2R1Y2VyIHRoYXQgcHJvdmlkZXMgdGhpcyBpbnN0YW5jZSBvZiB0aGUgZmlsdGVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlc2VuZFRvcERvd25MZWZ0UmlnaHQoSW1hZ2VQcm9kdWNlciBpcCkgewotICAgICAgICBpcC5yZXF1ZXN0VG9wRG93bkxlZnRSaWdodFJlc2VuZCh0aGlzKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRDb2xvck1vZGVsKENvbG9yTW9kZWwgbW9kZWwpIHsKLSAgICAgICAgY29uc3VtZXIuc2V0Q29sb3JNb2RlbChtb2RlbCk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBDb2xvck1vZGVsIG1vZGVsLCBpbnRbXSBwaXhlbHMsIGludCBvZmYsCi0gICAgICAgICAgICBpbnQgc2NhbnNpemUpIHsKLSAgICAgICAgY29uc3VtZXIuc2V0UGl4ZWxzKHgsIHksIHcsIGgsIG1vZGVsLCBwaXhlbHMsIG9mZiwgc2NhbnNpemUpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgQ29sb3JNb2RlbCBtb2RlbCwgYnl0ZVtdIHBpeGVscywgaW50IG9mZiwKLSAgICAgICAgICAgIGludCBzY2Fuc2l6ZSkgewotICAgICAgICBjb25zdW1lci5zZXRQaXhlbHMoeCwgeSwgdywgaCwgbW9kZWwsIHBpeGVscywgb2ZmLCBzY2Fuc2l6ZSk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0RGltZW5zaW9ucyhpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgY29uc3VtZXIuc2V0RGltZW5zaW9ucyh3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRIaW50cyhpbnQgaGludHMpIHsKLSAgICAgICAgY29uc3VtZXIuc2V0SGludHMoaGludHMpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGltYWdlQ29tcGxldGUoaW50IHN0YXR1cykgewotICAgICAgICBjb25zdW1lci5pbWFnZUNvbXBsZXRlKHN0YXR1cyk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvSW1hZ2VPYnNlcnZlci5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL0ltYWdlT2JzZXJ2ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjFlYzQxYi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvSW1hZ2VPYnNlcnZlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTAxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgamF2YS5hd3QuSW1hZ2U7Ci0KLS8qKgotICogdGhlIEltYWdlT2JzZXJ2ZXIgaW50ZXJmYWNlIGlzIGFuIGFzeW5jaHJvbm91cyB1cGRhdGUgaW50ZXJmYWNlIGZvciByZWNlaXZpbmcKLSAqIG5vdGlmaWNhdGlvbnMgYWJvdXQgSW1hZ2UgY29uc3RydWN0aW9uIHN0YXR1cy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgSW1hZ2VPYnNlcnZlciB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgV0lEVEggaW5kaWNhdGVzIHRoYXQgdGhlIHdpZHRoIG9mIHRoZSBpbWFnZSBpcyBhdmFpbGFibGUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgV0lEVEggPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEhFSUdIVCBpbmRpY2F0ZXMgdGhhdCB0aGUgd2lkdGggb2YgdGhlIGltYWdlIGlzIGF2YWlsYWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBIRUlHSFQgPSAyOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IFBST1BFUlRJRVMgaW5kaWNhdGVzIHRoYXQgdGhlIHByb3BlcnRpZXMgb2YgdGhlIGltYWdlIGFyZQotICAgICAqIGF2YWlsYWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBQUk9QRVJUSUVTID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTT01FQklUUyBpbmRpY2F0ZXMgdGhhdCBtb3JlIGJpdHMgbmVlZGVkIGZvciBkcmF3aW5nIGEKLSAgICAgKiBzY2FsZWQgdmFyaWF0aW9uIG9mIHRoZSBpbWFnZSBwaXhlbHMgYXJlIGF2YWlsYWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTT01FQklUUyA9IDg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRlJBTUVCSVRTIGluZGljYXRlcyB0aGF0IGNvbXBsZXRlIGZyYW1lIG9mIGEgaW1hZ2Ugd2hpY2ggd2FzCi0gICAgICogcHJldmlvdXNseSBkcmF3biBpcyBub3cgYXZhaWxhYmxlIGZvciBkcmF3aW5nIGFnYWluLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEZSQU1FQklUUyA9IDE2OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEFMTEJJVFMgaW5kaWNhdGVzIHRoYXQgYW4gaW1hZ2Ugd2hpY2ggd2FzIHByZXZpb3VzbHkgZHJhd24KLSAgICAgKiBpcyBub3cgY29tcGxldGUgYW5kIGNhbiBiZSBkcmF3biBhZ2Fpbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBBTExCSVRTID0gMzI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgRVJST1IgaW5kaWNhdGVzIHRoYXQgZXJyb3Igb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRVJST1IgPSA2NDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBBQk9SVCBpbmRpY2F0ZXMgdGhhdCB0aGUgaW1hZ2UgcHJvZHVjaW5nIGlzIGFib3J0ZWQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQUJPUlQgPSAxMjg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCBpcyBjYWxsZWQgd2hlbiBpbmZvcm1hdGlvbiBhYm91dCBhbiBJbWFnZSBpbnRlcmZhY2UgYmVjb21lcwotICAgICAqIGF2YWlsYWJsZS4gVGhpcyBtZXRob2QgcmV0dXJucyB0cnVlIGlmIGZ1cnRoZXIgdXBkYXRlcyBhcmUgbmVlZGVkLCBmYWxzZQotICAgICAqIGlmIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1nCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgdG8gYmUgb2JzZXJ2ZWQuCi0gICAgICogQHBhcmFtIGluZm9mbGFncwotICAgICAqICAgICAgICAgICAgdGhlIGJpdHdpc2UgT1IgY29tYmluYXRpb24gb2YgaW5mb3JtYXRpb24gZmxhZ3M6IEFCT1JULAotICAgICAqICAgICAgICAgICAgQUxMQklUUywgRVJST1IsIEZSQU1FQklUUywgSEVJR0hULCBQUk9QRVJUSUVTLCBTT01FQklUUywKLSAgICAgKiAgICAgICAgICAgIFdJRFRILgotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQuCi0gICAgICogQHJldHVybiB0cnVlIGlmIGZ1cnRoZXIgdXBkYXRlcyBhcmUgbmVlZGVkLCBmYWxzZSBpZiBub3QuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaW1hZ2VVcGRhdGUoSW1hZ2UgaW1nLCBpbnQgaW5mb2ZsYWdzLCBpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9JbWFnZVByb2R1Y2VyLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvSW1hZ2VQcm9kdWNlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5MTM4YmUyLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9JbWFnZVByb2R1Y2VyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw3OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotLyoqCi0gKiBUaGUgSW1hZ2VQcm9kdWNlciBwcm92aWRlcyBhbiBpbnRlcmZhY2UgZm9yIG9iamVjdHMgd2hpY2ggcHJvZHVjZSB0aGUgaW1hZ2UKLSAqIGRhdGEuIEltYWdlUHJvZHVjZXIgaXMgdXNlZCBmb3IgcmVjb25zdHJ1Y3RpbmcgdGhlIGltYWdlLiBFYWNoIGltYWdlIGNvbnRhaW5zCi0gKiBhbiBJbWFnZVByb2R1Y2VyLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJbWFnZVByb2R1Y2VyIHsKLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGUgc3BlY2lmaWVkIEltYWdlQ29uc3VtZXIgaXMgcmVnaXN0ZXJlZCB3aXRoIHRoaXMKLSAgICAgKiBJbWFnZVByb3ZpZGVyIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWMKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZUNvbnN1bWVyIHRvIGJlIGNoZWNrZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIEltYWdlQ29uc3VtZXIgaXMgcmVnaXN0ZXJlZCB3aXRoIHRoaXMKLSAgICAgKiAgICAgICAgIEltYWdlUHJvdmlkZXIsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NvbnN1bWVyKEltYWdlQ29uc3VtZXIgaWMpOwotCi0gICAgLyoqCi0gICAgICogU3RhcnRzIGEgcmVjb25zdHJ1Y3Rpb24gb2YgdGhlIGltYWdlIGRhdGEgd2hpY2ggd2lsbCBiZSBkZWxpdmVyZWQgdG8gdGhpcwotICAgICAqIGNvbnN1bWVyLiBUaGlzIG1ldGhvZCBhZGRzIHRoZSBzcGVjaWZpZWQgSW1hZ2VDb25zdW1lciBiZWZvcmUKLSAgICAgKiByZWNvbnN0cnVjdGluZyB0aGUgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGljCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIEltYWdlQ29uc3VtZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc3RhcnRQcm9kdWN0aW9uKEltYWdlQ29uc3VtZXIgaWMpOwotCi0gICAgLyoqCi0gICAgICogUmVxdWVzdHMgdGhlIEltYWdlUHJvZHVjZXIgdG8gcmVzZW5kIHRoZSBpbWFnZSBkYXRhIGluCi0gICAgICogSW1hZ2VDb25zdW1lci5UT1BET1dOTEVGVFJJR0hUIG9yZGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpYwotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBJbWFnZUNvbnN1bWVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlcXVlc3RUb3BEb3duTGVmdFJpZ2h0UmVzZW5kKEltYWdlQ29uc3VtZXIgaWMpOwotCi0gICAgLyoqCi0gICAgICogRGVyZWdpc3RlcnMgdGhlIHNwZWNpZmllZCBJbWFnZUNvbnN1bWVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpYwotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBJbWFnZUNvbnN1bWVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZUNvbnN1bWVyKEltYWdlQ29uc3VtZXIgaWMpOwotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgc3BlY2lmaWVkIEltYWdlQ29uc3VtZXIgb2JqZWN0IHRvIHRoaXMgSW1hZ2VQcm9kdWNlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgSW1hZ2VDb25zdW1lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRDb25zdW1lcihJbWFnZUNvbnN1bWVyIGljKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0ltYWdpbmdPcEV4Y2VwdGlvbi5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL0ltYWdpbmdPcEV4Y2VwdGlvbi5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlMGMwMTI3Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9JbWFnaW5nT3BFeGNlcHRpb24uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ5ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICoKLSAqIEBkYXRlOiBPY3QgNSwgMjAwNQotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLS8qKgotICogVGhlIEltYWdpbmdPcEV4Y2VwdGlvbiBjbGFzcyBwcm92aWRlcyBlcnJvciBub3RpZmljYXRpb24gd2hlbiB0aGUKLSAqIEJ1ZmZlcmVkSW1hZ2VPcCBvciBSYXN0ZXJPcCBmaWx0ZXIgbWV0aG9kcyBjYW4gbm90IHBlcmZvcm0gdGhlIGRlc2lyZWQgZmlsdGVyCi0gKiBvcGVyYXRpb24uCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgSW1hZ2luZ09wRXhjZXB0aW9uIGV4dGVuZHMgUnVudGltZUV4Y2VwdGlvbiB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSA4MDI2Mjg4NDgxODQ2Mjc2NjU4TDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJbWFnaW5nT3BFeGNlcHRpb24gd2l0aCBhIGRldGFpbCBtZXNzYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgZGV0YWlsIG1lc3NhZ2UuCi0gICAgICovCi0gICAgcHVibGljIEltYWdpbmdPcEV4Y2VwdGlvbihTdHJpbmcgcykgewotICAgICAgICBzdXBlcihzKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvSW5kZXhDb2xvck1vZGVsLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvSW5kZXhDb2xvck1vZGVsLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBiMDZhY2QuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL0luZGV4Q29sb3JNb2RlbC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTA4MCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEuYXd0LlRyYW5zcGFyZW5jeTsKLWltcG9ydCBqYXZhLmF3dC5jb2xvci5Db2xvclNwYWNlOwotaW1wb3J0IGphdmEubWF0aC5CaWdJbnRlZ2VyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIENsYXNzIEluZGV4Q29sb3JNb2RlbCByZXByZXNlbnRzIGEgY29sb3IgbW9kZWwgaW4gd2hpY2ggdGhlIGNvbG9yIHZhbHVlcwotICogb2YgdGhlIHBpeGVscyBhcmUgcmVhZCBmcm9tIGEgcGFsZXR0ZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJbmRleENvbG9yTW9kZWwgZXh0ZW5kcyBDb2xvck1vZGVsIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb2xvciBtYXAuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgY29sb3JNYXBbXTsgLy8gQ29sb3IgTWFwCi0KLSAgICAvKioKLSAgICAgKiBUaGUgbWFwIHNpemUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgbWFwU2l6ZTsgLy8gQ29sb3IgTWFwIHNpemUKLQotICAgIC8qKgotICAgICAqIFRoZSB0cmFuc3BhcmVudCBpbmRleC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCB0cmFuc3BhcmVudEluZGV4OyAvLyBJbmRleCBvZiBmdWxseSB0cmFuc3BhcmVudCBwaXhlbAotCi0gICAgLyoqCi0gICAgICogVGhlIGdyYXkgcGFsZXR0ZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gZ3JheVBhbGV0dGU7IC8vIENvbG9yIE1vZGVsIGhhcyBDb2xvciBNYXAgd2l0aCBHcmF5IFBhbGxldGUKLQotICAgIC8qKgotICAgICAqIFRoZSB2YWxpZCBiaXRzLgotICAgICAqLwotICAgIHByaXZhdGUgQmlnSW50ZWdlciB2YWxpZEJpdHM7IC8vIFNwZWNpZnkgdmFsaWQgQ29sb3IgTWFwIHZhbHVlcwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENBQ0hFU0laRS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgQ0FDSEVTSVpFID0gMjA7IC8vIENhY2hlIHNpemUuIENhY2hlIHVzZWQgZm9yCi0KLSAgICAvLyBpbXByb3ZpbmcgcGVyZm9ybWFjZSBvZiBzZWxlY3Rpb24KLSAgICAvLyBuZWFyZXN0IGNvbG9yIGluIENvbG9yIE1hcAotCi0gICAgLyoqCi0gICAgICogVGhlIGNhY2hldGFibGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBpbnQgY2FjaGV0YWJsZVtdID0gbmV3IGludFtDQUNIRVNJWkUgKiAyXTsgLy8gQ2FjaGUgdGFibGUgLQotCi0gICAgLy8gdXNlZCBmb3IKLQotICAgIC8vIHN0b3JpbmcgUkdCIHZhbHVlcyBhbmQgdGhhdCBhcHByb3ByaWF0ZSBpbmRpY2VzCi0gICAgLy8gaW4gdGhlIENvbG9yIE1hcAotCi0gICAgLyoqCi0gICAgICogVGhlIG5leHQgaW5zZXJ0IGlkeC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBuZXh0SW5zZXJ0SWR4ID0gMDsgLy8gTmV4dCBpbmRleCBmb3IgaW5zZXJ0aW9uIGludG8gQ2FjaGUgdGFibGUKLQotICAgIC8qKgotICAgICAqIFRoZSB0b3RhbCBpbnNlcnRlZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCB0b3RhbEluc2VydGVkID0gMDsgLy8gTnVtYmVyIG9mIGluc2VydGVkIHZhbHVlcyBpbnRvIENhY2hlIHRhYmxlCi0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgaW5kZXggY29sb3IgbW9kZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBjb21wb25lbnQgbWFza3MuCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzaXplIG9mIHRoZSBjb2xvciBtYXAuCi0gICAgICogQHBhcmFtIGNtYXAKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB0aGF0IGdpdmVzIHRoZSBjb2xvciBtYXBwaW5nLgotICAgICAqIEBwYXJhbSBzdGFydAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IGluZGV4IG9mIHRoZSBjb2xvciBtYXBwaW5nIGRhdGEgd2l0aGluIHRoZSBjbWFwCi0gICAgICogICAgICAgICAgICBhcnJheS4KLSAgICAgKiBAcGFyYW0gdHJhbnNmZXJUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNmZXIgdHlwZSAocHJpbWl0aXZlIGphdmEgdHlwZSB0byB1c2UgZm9yIHRoZQotICAgICAqICAgICAgICAgICAgY29tcG9uZW50cykuCi0gICAgICogQHBhcmFtIHZhbGlkQml0cwotICAgICAqICAgICAgICAgICAgYSBsaXN0IG9mIHdoaWNoIGJpdHMgcmVwcmVzZW50IHZhbGlkIGNvbG9ybWFwIHZhbHVlcywgb3IgbnVsbAotICAgICAqICAgICAgICAgICAgaWYgYWxsIGFyZSB2YWxpZC4KLSAgICAgKiBAdGhyb3dzIElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBzaXplIG9mIHRoZSBjb2xvciBtYXAgaXMgbGVzcyB0aGFuIG9uZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW5kZXhDb2xvck1vZGVsKGludCBiaXRzLCBpbnQgc2l6ZSwgaW50IGNtYXBbXSwgaW50IHN0YXJ0LCBpbnQgdHJhbnNmZXJUeXBlLAotICAgICAgICAgICAgQmlnSW50ZWdlciB2YWxpZEJpdHMpIHsKLQotICAgICAgICBzdXBlcihiaXRzLCBJbmRleENvbG9yTW9kZWwuY3JlYXRlQml0cyh0cnVlKSwgQ29sb3JTcGFjZS5nZXRJbnN0YW5jZShDb2xvclNwYWNlLkNTX3NSR0IpLAotICAgICAgICAgICAgICAgIHRydWUsIGZhbHNlLCBUcmFuc3BhcmVuY3kuT1BBUVVFLCB2YWxpZGF0ZVRyYW5zZmVyVHlwZSh0cmFuc2ZlclR5cGUpKTsKLQotICAgICAgICBpZiAoc2l6ZSA8IDEpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNjQ9U2l6ZSBvZiB0aGUgY29sb3IgbWFwIGlzIGxlc3MgdGhhbiAxCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2NCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgbWFwU2l6ZSA9IHNpemU7Ci0gICAgICAgIGNvbG9yTWFwID0gbmV3IGludFttYXBTaXplXTsKLSAgICAgICAgdHJhbnNwYXJlbnRJbmRleCA9IC0xOwotCi0gICAgICAgIGlmICh2YWxpZEJpdHMgIT0gbnVsbCkgewotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBtYXBTaXplOyBpKyspIHsKLSAgICAgICAgICAgICAgICBpZiAoIXZhbGlkQml0cy50ZXN0Qml0KGkpKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMudmFsaWRCaXRzID0gdmFsaWRCaXRzOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHRyYW5zcGFyZW5jeSA9IFRyYW5zcGFyZW5jeS5PUEFRVUU7Ci0gICAgICAgIGludCBhbHBoYU1hc2sgPSAweGZmMDAwMDAwOwotICAgICAgICBpbnQgYWxwaGEgPSAwOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbWFwU2l6ZTsgaSsrLCBzdGFydCsrKSB7Ci0gICAgICAgICAgICBjb2xvck1hcFtpXSA9IGNtYXBbc3RhcnRdOwotICAgICAgICAgICAgYWxwaGEgPSBjbWFwW3N0YXJ0XSAmIGFscGhhTWFzazsKLQotICAgICAgICAgICAgaWYgKGFscGhhID09IGFscGhhTWFzaykgewotICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGFscGhhID09IDApIHsKLSAgICAgICAgICAgICAgICBpZiAodHJhbnNwYXJlbnRJbmRleCA8IDApIHsKLSAgICAgICAgICAgICAgICAgICAgdHJhbnNwYXJlbnRJbmRleCA9IGk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmICh0cmFuc3BhcmVuY3kgPT0gVHJhbnNwYXJlbmN5Lk9QQVFVRSkgewotICAgICAgICAgICAgICAgICAgICB0cmFuc3BhcmVuY3kgPSBUcmFuc3BhcmVuY3kuQklUTUFTSzsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGFscGhhICE9IGFscGhhTWFzayAmJiB0cmFuc3BhcmVuY3kgIT0gVHJhbnNwYXJlbmN5LlRSQU5TTFVDRU5UKSB7Ci0gICAgICAgICAgICAgICAgdHJhbnNwYXJlbmN5ID0gVHJhbnNwYXJlbmN5LlRSQU5TTFVDRU5UOwotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLSAgICAgICAgY2hlY2tQYWxldHRlKCk7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgaW5kZXggY29sb3IgbW9kZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBjb21wb25lbnQgbWFza3MuCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzaXplIG9mIHRoZSBjb2xvciBtYXAuCi0gICAgICogQHBhcmFtIGNtYXAKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSB0aGF0IGdpdmVzIHRoZSBjb2xvciBtYXBwaW5nLgotICAgICAqIEBwYXJhbSBzdGFydAotICAgICAqICAgICAgICAgICAgdGhlIHN0YXJ0IGluZGV4IG9mIHRoZSBjb2xvciBtYXBwaW5nIGRhdGEgd2l0aGluIHRoZSBjbWFwCi0gICAgICogICAgICAgICAgICBhcnJheS4KLSAgICAgKiBAcGFyYW0gaGFzYWxwaGEKLSAgICAgKiAgICAgICAgICAgIHdoZXRoZXIgdGhpcyBjb2xvciBtb2RlbCB1c2VzIGFscGhhLgotICAgICAqIEBwYXJhbSB0cmFucwotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zcGFyZW5jeSBzdXBwb3J0ZWQsIEBzZWUgamF2YS5hd3QuVHJhbnNwYXJlbmN5LgotICAgICAqIEBwYXJhbSB0cmFuc2ZlclR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSB0cmFuc2ZlciB0eXBlIChwcmltaXRpdmUgamF2YSB0eXBlIHRvIHVzZSBmb3IgdGhlCi0gICAgICogICAgICAgICAgICBjb21wb25lbnRzKS4KLSAgICAgKiBAdGhyb3dzIElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBzaXplIG9mIHRoZSBjb2xvciBtYXAgaXMgbGVzcyB0aGFuIG9uZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW5kZXhDb2xvck1vZGVsKGludCBiaXRzLCBpbnQgc2l6ZSwgaW50IGNtYXBbXSwgaW50IHN0YXJ0LCBib29sZWFuIGhhc2FscGhhLCBpbnQgdHJhbnMsCi0gICAgICAgICAgICBpbnQgdHJhbnNmZXJUeXBlKSB7Ci0KLSAgICAgICAgc3VwZXIoYml0cywgSW5kZXhDb2xvck1vZGVsLmNyZWF0ZUJpdHMoaGFzYWxwaGEgfHwgKHRyYW5zID49IDApKSwgQ29sb3JTcGFjZQotICAgICAgICAgICAgICAgIC5nZXRJbnN0YW5jZShDb2xvclNwYWNlLkNTX3NSR0IpLCAoaGFzYWxwaGEgfHwgKHRyYW5zID49IDApKSwgZmFsc2UsCi0gICAgICAgICAgICAgICAgVHJhbnNwYXJlbmN5Lk9QQVFVRSwgdmFsaWRhdGVUcmFuc2ZlclR5cGUodHJhbnNmZXJUeXBlKSk7Ci0KLSAgICAgICAgaWYgKHNpemUgPCAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjY0PVNpemUgb2YgdGhlIGNvbG9yIG1hcCBpcyBsZXNzIHRoYW4gMQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNjQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIG1hcFNpemUgPSBzaXplOwotICAgICAgICBjb2xvck1hcCA9IG5ldyBpbnRbbWFwU2l6ZV07Ci0gICAgICAgIGlmICh0cmFucyA+PSAwICYmIHRyYW5zIDwgbWFwU2l6ZSkgewotICAgICAgICAgICAgdHJhbnNwYXJlbnRJbmRleCA9IHRyYW5zOwotICAgICAgICAgICAgdHJhbnNwYXJlbmN5ID0gVHJhbnNwYXJlbmN5LkJJVE1BU0s7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICB0cmFuc3BhcmVudEluZGV4ID0gLTE7Ci0gICAgICAgICAgICB0cmFuc3BhcmVuY3kgPSBUcmFuc3BhcmVuY3kuT1BBUVVFOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGFscGhhTWFzayA9IDB4ZmYwMDAwMDA7Ci0gICAgICAgIGludCBhbHBoYSA9IDA7Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBtYXBTaXplOyBpKyssIHN0YXJ0KyspIHsKLSAgICAgICAgICAgIGlmICh0cmFuc3BhcmVudEluZGV4ID09IGkpIHsKLSAgICAgICAgICAgICAgICBjb2xvck1hcFtpXSA9IGNtYXBbc3RhcnRdICYgMHgwMGZmZmZmZjsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChoYXNhbHBoYSkgewotICAgICAgICAgICAgICAgIGFscGhhID0gY21hcFtzdGFydF0gJiBhbHBoYU1hc2s7Ci0gICAgICAgICAgICAgICAgY29sb3JNYXBbaV0gPSBjbWFwW3N0YXJ0XTsKLQotICAgICAgICAgICAgICAgIGlmIChhbHBoYSA9PSBhbHBoYU1hc2spIHsKLSAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChhbHBoYSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmICh0cmFucyA8IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHRyYW5zID0gaTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpZiAodHJhbnNwYXJlbmN5ID09IFRyYW5zcGFyZW5jeS5PUEFRVUUpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHRyYW5zcGFyZW5jeSA9IFRyYW5zcGFyZW5jeS5CSVRNQVNLOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChhbHBoYSAhPSAwICYmIHRyYW5zcGFyZW5jeSAhPSBUcmFuc3BhcmVuY3kuVFJBTlNMVUNFTlQpIHsKLSAgICAgICAgICAgICAgICAgICAgdHJhbnNwYXJlbmN5ID0gVHJhbnNwYXJlbmN5LlRSQU5TTFVDRU5UOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgY29sb3JNYXBbaV0gPSBhbHBoYU1hc2sgfCBjbWFwW3N0YXJ0XTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBjaGVja1BhbGV0dGUoKTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBpbmRleCBjb2xvciBtb2RlbCBieSBidWlsZGluZyB0aGUgY29sb3IgbWFwIGZyb20KLSAgICAgKiBhcnJheXMgb2YgcmVkLCBncmVlbiwgYmx1ZSwgYW5kIGFscGhhIHZhbHVlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYml0cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGNvbXBvbmVudCBtYXNrcy4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIHNpemUgb2YgdGhlIGNvbG9yIG1hcC4KLSAgICAgKiBAcGFyYW0gcgotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IGdpdmluZyB0aGUgcmVkIGNvbXBvbmVudHMgb2YgdGhlIGVudHJpZXMgaW4gdGhlCi0gICAgICogICAgICAgICAgICBjb2xvciBtYXAuCi0gICAgICogQHBhcmFtIGcKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBnaXZpbmcgdGhlIGdyZWVuIGNvbXBvbmVudHMgb2YgdGhlIGVudHJpZXMgaW4gdGhlCi0gICAgICogICAgICAgICAgICBjb2xvciBtYXAuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBnaXZpbmcgdGhlIGJsdWUgY29tcG9uZW50cyBvZiB0aGUgZW50cmllcyBpbiB0aGUKLSAgICAgKiAgICAgICAgICAgIGNvbG9yIG1hcC4KLSAgICAgKiBAcGFyYW0gYQotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IGdpdmluZyB0aGUgYWxwaGEgY29tcG9uZW50cyBvZiB0aGUgZW50cmllcyBpbiB0aGUKLSAgICAgKiAgICAgICAgICAgIGNvbG9yIG1hcC4KLSAgICAgKiBAdGhyb3dzIElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBzaXplIG9mIHRoZSBjb2xvciBtYXAgaXMgbGVzcyB0aGFuIG9uZS4KLSAgICAgKiBAdGhyb3dzIEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBzaXplIG9mIG9uZSBvZiB0aGUgY29tcG9uZW50IGFycmF5cyBpcyBsZXNzIHRoYW4gdGhlCi0gICAgICogICAgICAgICAgICAgc2l6ZSBvZiB0aGUgY29sb3IgbWFwLgotICAgICAqLwotICAgIHB1YmxpYyBJbmRleENvbG9yTW9kZWwoaW50IGJpdHMsIGludCBzaXplLCBieXRlIHJbXSwgYnl0ZSBnW10sIGJ5dGUgYltdLCBieXRlIGFbXSkgewotCi0gICAgICAgIHN1cGVyKGJpdHMsIEluZGV4Q29sb3JNb2RlbC5jcmVhdGVCaXRzKHRydWUpLCBDb2xvclNwYWNlLmdldEluc3RhbmNlKENvbG9yU3BhY2UuQ1Nfc1JHQiksCi0gICAgICAgICAgICAgICAgdHJ1ZSwgZmFsc2UsIFRyYW5zcGFyZW5jeS5PUEFRVUUsIHZhbGlkYXRlVHJhbnNmZXJUeXBlKENvbG9yTW9kZWwKLSAgICAgICAgICAgICAgICAgICAgICAgIC5nZXRUcmFuc2ZlclR5cGUoYml0cykpKTsKLQotICAgICAgICBjcmVhdGVDb2xvck1hcChzaXplLCByLCBnLCBiLCBhLCAtMSk7Ci0gICAgICAgIGNoZWNrUGFsZXR0ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBpbmRleCBjb2xvciBtb2RlbCBieSBidWlsZGluZyB0aGUgY29sb3IgbWFwIGZyb20KLSAgICAgKiBhcnJheXMgb2YgcmVkLCBncmVlbiwgYW5kIGJsdWUgdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiaXRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgY29tcG9uZW50IG1hc2tzLgotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgc2l6ZSBvZiB0aGUgY29sb3IgbWFwLgotICAgICAqIEBwYXJhbSByCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgZ2l2aW5nIHRoZSByZWQgY29tcG9uZW50cyBvZiB0aGUgZW50cmllcyBpbiB0aGUKLSAgICAgKiAgICAgICAgICAgIGNvbG9yIG1hcC4KLSAgICAgKiBAcGFyYW0gZwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IGdpdmluZyB0aGUgZ3JlZW4gY29tcG9uZW50cyBvZiB0aGUgZW50cmllcyBpbiB0aGUKLSAgICAgKiAgICAgICAgICAgIGNvbG9yIG1hcC4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IGdpdmluZyB0aGUgYmx1ZSBjb21wb25lbnRzIG9mIHRoZSBlbnRyaWVzIGluIHRoZQotICAgICAqICAgICAgICAgICAgY29sb3IgbWFwLgotICAgICAqIEBwYXJhbSB0cmFucwotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zcGFyZW5jeSBzdXBwb3J0ZWQsIEBzZWUgamF2YS5hd3QuVHJhbnNwYXJlbmN5LgotICAgICAqIEB0aHJvd3MgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHNpemUgb2YgdGhlIGNvbG9yIG1hcCBpcyBsZXNzIHRoYW4gb25lLgotICAgICAqIEB0aHJvd3MgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHNpemUgb2Ygb25lIG9mIHRoZSBjb21wb25lbnQgYXJyYXlzIGlzIGxlc3MgdGhhbiB0aGUKLSAgICAgKiAgICAgICAgICAgICBzaXplIG9mIHRoZSBjb2xvciBtYXAuCi0gICAgICovCi0gICAgcHVibGljIEluZGV4Q29sb3JNb2RlbChpbnQgYml0cywgaW50IHNpemUsIGJ5dGUgcltdLCBieXRlIGdbXSwgYnl0ZSBiW10sIGludCB0cmFucykgewotCi0gICAgICAgIHN1cGVyKGJpdHMsIEluZGV4Q29sb3JNb2RlbC5jcmVhdGVCaXRzKCh0cmFucyA+PSAwKSksIENvbG9yU3BhY2UKLSAgICAgICAgICAgICAgICAuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19zUkdCKSwgKHRyYW5zID49IDApLCBmYWxzZSwgVHJhbnNwYXJlbmN5Lk9QQVFVRSwKLSAgICAgICAgICAgICAgICB2YWxpZGF0ZVRyYW5zZmVyVHlwZShDb2xvck1vZGVsLmdldFRyYW5zZmVyVHlwZShiaXRzKSkpOwotCi0gICAgICAgIGNyZWF0ZUNvbG9yTWFwKHNpemUsIHIsIGcsIGIsIG51bGwsIHRyYW5zKTsKLSAgICAgICAgY2hlY2tQYWxldHRlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGluZGV4IGNvbG9yIG1vZGVsIGJ5IGJ1aWxkaW5nIHRoZSBjb2xvciBtYXAgZnJvbQotICAgICAqIGFycmF5cyBvZiByZWQsIGdyZWVuLCBhbmQgYmx1ZSB2YWx1ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBjb21wb25lbnQgbWFza3MuCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzaXplIG9mIHRoZSBjb2xvciBtYXAuCi0gICAgICogQHBhcmFtIHIKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBnaXZpbmcgdGhlIHJlZCBjb21wb25lbnRzIG9mIHRoZSBlbnRyaWVzIGluIHRoZQotICAgICAqICAgICAgICAgICAgY29sb3IgbWFwLgotICAgICAqIEBwYXJhbSBnCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgZ2l2aW5nIHRoZSBncmVlbiBjb21wb25lbnRzIG9mIHRoZSBlbnRyaWVzIGluIHRoZQotICAgICAqICAgICAgICAgICAgY29sb3IgbWFwLgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgZ2l2aW5nIHRoZSBibHVlIGNvbXBvbmVudHMgb2YgdGhlIGVudHJpZXMgaW4gdGhlCi0gICAgICogICAgICAgICAgICBjb2xvciBtYXAuCi0gICAgICogQHRocm93cyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgc2l6ZSBvZiB0aGUgY29sb3IgbWFwIGlzIGxlc3MgdGhhbiBvbmUuCi0gICAgICogQHRocm93cyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgc2l6ZSBvZiBvbmUgb2YgdGhlIGNvbXBvbmVudCBhcnJheXMgaXMgbGVzcyB0aGFuIHRoZQotICAgICAqICAgICAgICAgICAgIHNpemUgb2YgdGhlIGNvbG9yIG1hcC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW5kZXhDb2xvck1vZGVsKGludCBiaXRzLCBpbnQgc2l6ZSwgYnl0ZSByW10sIGJ5dGUgZ1tdLCBieXRlIGJbXSkgewotICAgICAgICBzdXBlcihiaXRzLCBJbmRleENvbG9yTW9kZWwuY3JlYXRlQml0cyhmYWxzZSksIENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19zUkdCKSwKLSAgICAgICAgICAgICAgICBmYWxzZSwgZmFsc2UsIFRyYW5zcGFyZW5jeS5PUEFRVUUsIHZhbGlkYXRlVHJhbnNmZXJUeXBlKENvbG9yTW9kZWwKLSAgICAgICAgICAgICAgICAgICAgICAgIC5nZXRUcmFuc2ZlclR5cGUoYml0cykpKTsKLQotICAgICAgICBjcmVhdGVDb2xvck1hcChzaXplLCByLCBnLCBiLCBudWxsLCAtMSk7Ci0gICAgICAgIGNoZWNrUGFsZXR0ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBpbmRleCBjb2xvciBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYml0cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGNvbXBvbmVudCBtYXNrcy4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIHNpemUgb2YgdGhlIGNvbG9yIG1hcC4KLSAgICAgKiBAcGFyYW0gY21hcAotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IHRoYXQgZ2l2ZXMgdGhlIGNvbG9yIG1hcHBpbmcuCi0gICAgICogQHBhcmFtIHN0YXJ0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgaW5kZXggb2YgdGhlIGNvbG9yIG1hcHBpbmcgZGF0YSB3aXRoaW4gdGhlIGNtYXAKLSAgICAgKiAgICAgICAgICAgIGFycmF5LgotICAgICAqIEBwYXJhbSBoYXNhbHBoYQotICAgICAqICAgICAgICAgICAgd2hldGhlciB0aGlzIGNvbG9yIG1vZGVsIHVzZXMgYWxwaGEuCi0gICAgICogQHBhcmFtIHRyYW5zCi0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNwYXJlbmN5IHN1cHBvcnRlZCwgQHNlZSBqYXZhLmF3dC5UcmFuc3BhcmVuY3kuCi0gICAgICogQHRocm93cyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgc2l6ZSBvZiB0aGUgY29sb3IgbWFwIGlzIGxlc3MgdGhhbiBvbmUuCi0gICAgICovCi0gICAgcHVibGljIEluZGV4Q29sb3JNb2RlbChpbnQgYml0cywgaW50IHNpemUsIGJ5dGUgY21hcFtdLCBpbnQgc3RhcnQsIGJvb2xlYW4gaGFzYWxwaGEsIGludCB0cmFucykgewotCi0gICAgICAgIHN1cGVyKGJpdHMsIEluZGV4Q29sb3JNb2RlbC5jcmVhdGVCaXRzKGhhc2FscGhhIHx8ICh0cmFucyA+PSAwKSksIENvbG9yU3BhY2UKLSAgICAgICAgICAgICAgICAuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19zUkdCKSwgKGhhc2FscGhhIHx8ICh0cmFucyA+PSAwKSksIGZhbHNlLAotICAgICAgICAgICAgICAgIFRyYW5zcGFyZW5jeS5PUEFRVUUsIHZhbGlkYXRlVHJhbnNmZXJUeXBlKENvbG9yTW9kZWwuZ2V0VHJhbnNmZXJUeXBlKGJpdHMpKSk7Ci0KLSAgICAgICAgaWYgKHNpemUgPCAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjY0PVNpemUgb2YgdGhlIGNvbG9yIG1hcCBpcyBsZXNzIHRoYW4gMQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNjQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIG1hcFNpemUgPSBzaXplOwotICAgICAgICBjb2xvck1hcCA9IG5ldyBpbnRbbWFwU2l6ZV07Ci0gICAgICAgIHRyYW5zcGFyZW50SW5kZXggPSAtMTsKLQotICAgICAgICB0cmFuc3BhcmVuY3kgPSBUcmFuc3BhcmVuY3kuT1BBUVVFOwotICAgICAgICBpbnQgYWxwaGEgPSAweGZmMDAwMDAwOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbWFwU2l6ZTsgaSsrKSB7Ci0gICAgICAgICAgICBjb2xvck1hcFtpXSA9IChjbWFwW3N0YXJ0KytdICYgMHhmZikgPDwgMTYgfCAoY21hcFtzdGFydCsrXSAmIDB4ZmYpIDw8IDgKLSAgICAgICAgICAgICAgICAgICAgfCAoY21hcFtzdGFydCsrXSAmIDB4ZmYpOwotICAgICAgICAgICAgaWYgKHRyYW5zID09IGkpIHsKLSAgICAgICAgICAgICAgICBpZiAodHJhbnNwYXJlbmN5ID09IFRyYW5zcGFyZW5jeS5PUEFRVUUpIHsKLSAgICAgICAgICAgICAgICAgICAgdHJhbnNwYXJlbmN5ID0gVHJhbnNwYXJlbmN5LkJJVE1BU0s7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChoYXNhbHBoYSkgewotICAgICAgICAgICAgICAgICAgICBzdGFydCsrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChoYXNhbHBoYSkgewotICAgICAgICAgICAgICAgIGFscGhhID0gY21hcFtzdGFydCsrXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgaWYgKGFscGhhID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHRyYW5zcGFyZW5jeSA9PSBUcmFuc3BhcmVuY3kuT1BBUVVFKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB0cmFuc3BhcmVuY3kgPSBUcmFuc3BhcmVuY3kuQklUTUFTSzsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmICh0cmFucyA8IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB0cmFucyA9IGk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBpZiAoYWxwaGEgIT0gMHhmZiAmJiB0cmFuc3BhcmVuY3kgIT0gVHJhbnNwYXJlbmN5LlRSQU5TTFVDRU5UKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB0cmFuc3BhcmVuY3kgPSBUcmFuc3BhcmVuY3kuVFJBTlNMVUNFTlQ7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYWxwaGEgPDw9IDI0OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY29sb3JNYXBbaV0gfD0gYWxwaGE7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAodHJhbnMgPj0gMCAmJiB0cmFucyA8IG1hcFNpemUpIHsKLSAgICAgICAgICAgIHRyYW5zcGFyZW50SW5kZXggPSB0cmFuczsKLSAgICAgICAgfQotICAgICAgICBjaGVja1BhbGV0dGUoKTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBpbmRleCBjb2xvciBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYml0cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGNvbXBvbmVudCBtYXNrcy4KLSAgICAgKiBAcGFyYW0gc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIHNpemUgb2YgdGhlIGNvbG9yIG1hcC4KLSAgICAgKiBAcGFyYW0gY21hcAotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IHRoYXQgZ2l2ZXMgdGhlIGNvbG9yIG1hcHBpbmcuCi0gICAgICogQHBhcmFtIHN0YXJ0Ci0gICAgICogICAgICAgICAgICB0aGUgc3RhcnQgaW5kZXggb2YgdGhlIGNvbG9yIG1hcHBpbmcgZGF0YSB3aXRoaW4gdGhlIGNtYXAKLSAgICAgKiAgICAgICAgICAgIGFycmF5LgotICAgICAqIEBwYXJhbSBoYXNhbHBoYQotICAgICAqICAgICAgICAgICAgd2hldGhlciB0aGlzIGNvbG9yIG1vZGVsIHVzZXMgYWxwaGEuCi0gICAgICogQHRocm93cyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgc2l6ZSBvZiB0aGUgY29sb3IgbWFwIGlzIGxlc3MgdGhhbiBvbmUuCi0gICAgICovCi0gICAgcHVibGljIEluZGV4Q29sb3JNb2RlbChpbnQgYml0cywgaW50IHNpemUsIGJ5dGUgY21hcFtdLCBpbnQgc3RhcnQsIGJvb2xlYW4gaGFzYWxwaGEpIHsKLQotICAgICAgICB0aGlzKGJpdHMsIHNpemUsIGNtYXAsIHN0YXJ0LCBoYXNhbHBoYSwgLTEpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgZ2V0RGF0YUVsZW1lbnRzKGludFtdIGNvbXBvbmVudHMsIGludCBvZmZzZXQsIE9iamVjdCBwaXhlbCkgewotICAgICAgICBpbnQgcmdiID0gKGNvbXBvbmVudHNbb2Zmc2V0XSA8PCAxNikgfCAoY29tcG9uZW50c1tvZmZzZXQgKyAxXSkgPDwgOAotICAgICAgICAgICAgICAgIHwgY29tcG9uZW50c1tvZmZzZXQgKyAyXTsKLSAgICAgICAgaWYgKGhhc0FscGhhKSB7Ci0gICAgICAgICAgICByZ2IgfD0gY29tcG9uZW50c1tvZmZzZXQgKyAzXSA8PCAyNDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJnYiB8PSAweGZmMDAwMDAwOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBnZXREYXRhRWxlbWVudHMocmdiLCBwaXhlbCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHN5bmNocm9uaXplZCBPYmplY3QgZ2V0RGF0YUVsZW1lbnRzKGludCByZ2IsIE9iamVjdCBwaXhlbCkgewotICAgICAgICBpbnQgcmVkID0gKHJnYiA+PiAxNikgJiAweGZmOwotICAgICAgICBpbnQgZ3JlZW4gPSAocmdiID4+IDgpICYgMHhmZjsKLSAgICAgICAgaW50IGJsdWUgPSByZ2IgJiAweGZmOwotICAgICAgICBpbnQgYWxwaGEgPSByZ2IgPj4+IDI0OwotICAgICAgICBpbnQgcGl4SWR4ID0gMDsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHRvdGFsSW5zZXJ0ZWQ7IGkrKykgewotICAgICAgICAgICAgaW50IGlkeCA9IGkgKiAyOwotICAgICAgICAgICAgaWYgKHJnYiA9PSBjYWNoZXRhYmxlW2lkeF0pIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gY3JlYXRlRGF0YU9iamVjdChjYWNoZXRhYmxlW2lkeCArIDFdLCBwaXhlbCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoIWhhc0FscGhhICYmIGdyYXlQYWxldHRlKSB7Ci0gICAgICAgICAgICBpbnQgZ3JleSA9IChyZWQgKiA3NyArIGdyZWVuICogMTUwICsgYmx1ZSAqIDI5ICsgMTI4KSA+Pj4gODsKLSAgICAgICAgICAgIGludCBtaW5FcnJvciA9IDI1NTsKLSAgICAgICAgICAgIGludCBlcnJvciA9IDA7Ci0KLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbWFwU2l6ZTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgZXJyb3IgPSBNYXRoLmFicygoY29sb3JNYXBbaV0gJiAweGZmKSAtIGdyZXkpOwotICAgICAgICAgICAgICAgIGlmIChlcnJvciA8IG1pbkVycm9yKSB7Ci0gICAgICAgICAgICAgICAgICAgIHBpeElkeCA9IGk7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChlcnJvciA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBtaW5FcnJvciA9IGVycm9yOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIGlmIChhbHBoYSA9PSAwICYmIHRyYW5zcGFyZW50SW5kZXggPiAtMSkgewotICAgICAgICAgICAgcGl4SWR4ID0gdHJhbnNwYXJlbnRJbmRleDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGludCBtaW5BbHBoYUVycm9yID0gMjU1OwotICAgICAgICAgICAgaW50IG1pbkVycm9yID0gMTk1MDc1OyAvLyAyNTVeMiArIDI1NV4yICsgMjU1XjIKLSAgICAgICAgICAgIGludCBhbHBoYUVycm9yOwotICAgICAgICAgICAgaW50IGVycm9yID0gMDsKLQotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBtYXBTaXplOyBpKyspIHsKLSAgICAgICAgICAgICAgICBpbnQgcGl4ID0gY29sb3JNYXBbaV07Ci0gICAgICAgICAgICAgICAgaWYgKHJnYiA9PSBwaXgpIHsKLSAgICAgICAgICAgICAgICAgICAgcGl4SWR4ID0gaTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGFscGhhRXJyb3IgPSBNYXRoLmFicyhhbHBoYSAtIChwaXggPj4+IDI0KSk7Ci0gICAgICAgICAgICAgICAgaWYgKGFscGhhRXJyb3IgPD0gbWluQWxwaGFFcnJvcikgewotICAgICAgICAgICAgICAgICAgICBtaW5BbHBoYUVycm9yID0gYWxwaGFFcnJvcjsKLQotICAgICAgICAgICAgICAgICAgICBpbnQgYnVmID0gKChwaXggPj4gMTYpICYgMHhmZikgLSByZWQ7Ci0gICAgICAgICAgICAgICAgICAgIGVycm9yID0gYnVmICogYnVmOwotCi0gICAgICAgICAgICAgICAgICAgIGlmIChlcnJvciA8IG1pbkVycm9yKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBidWYgPSAoKHBpeCA+PiA4KSAmIDB4ZmYpIC0gZ3JlZW47Ci0gICAgICAgICAgICAgICAgICAgICAgICBlcnJvciArPSBidWYgKiBidWY7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChlcnJvciA8IG1pbkVycm9yKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnVmID0gKHBpeCAmIDB4ZmYpIC0gYmx1ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBlcnJvciArPSBidWYgKiBidWY7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoZXJyb3IgPCBtaW5FcnJvcikgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwaXhJZHggPSBpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtaW5FcnJvciA9IGVycm9yOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGNhY2hldGFibGVbbmV4dEluc2VydElkeF0gPSByZ2I7Ci0gICAgICAgIGNhY2hldGFibGVbbmV4dEluc2VydElkeCArIDFdID0gcGl4SWR4OwotCi0gICAgICAgIG5leHRJbnNlcnRJZHggPSAobmV4dEluc2VydElkeCArIDIpICUgKENBQ0hFU0laRSAqIDIpOwotICAgICAgICBpZiAodG90YWxJbnNlcnRlZCA8IENBQ0hFU0laRSkgewotICAgICAgICAgICAgdG90YWxJbnNlcnRlZCsrOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGNyZWF0ZURhdGFPYmplY3QocGl4SWR4LCBwaXhlbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29udmVydHMgYW4gaW1hZ2UgZnJvbSBpbmRleGVkIHRvIFJHQiBmb3JtYXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJhc3RlcgotICAgICAqICAgICAgICAgICAgdGhlIHJhc3RlciBjb250YWluaW5nIHRoZSBzb3VyY2UgaW1hZ2UuCi0gICAgICogQHBhcmFtIGZvcmNlQVJHQgotICAgICAqICAgICAgICAgICAgd2hldGhlciB0byB1c2UgdGhlIGRlZmF1bHQgUkdCIGNvbG9yIG1vZGVsLgotICAgICAqIEByZXR1cm4gdGhlIGJ1ZmZlcmVkIGltYWdlLgotICAgICAqIEB0aHJvd3MgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHJhc3RlciBpcyBub3QgY29tcGF0aWJsZSB3aXRoIHRoaXMgY29sb3IgbW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UgY29udmVydFRvSW50RGlzY3JldGUoUmFzdGVyIHJhc3RlciwgYm9vbGVhbiBmb3JjZUFSR0IpIHsKLQotICAgICAgICBpZiAoIWlzQ29tcGF0aWJsZVJhc3RlcihyYXN0ZXIpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjY1PVRoZSByYXN0ZXIgYXJndW1lbnQgaXMgbm90IGNvbXBhdGlibGUgd2l0aCB0aGlzCi0gICAgICAgICAgICAvLyBJbmRleENvbG9yTW9kZWwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjY1IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBDb2xvck1vZGVsIG1vZGVsOwotICAgICAgICBpZiAoZm9yY2VBUkdCIHx8IHRyYW5zcGFyZW5jeSA9PSBUcmFuc3BhcmVuY3kuVFJBTlNMVUNFTlQpIHsKLSAgICAgICAgICAgIG1vZGVsID0gQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCk7Ci0gICAgICAgIH0gZWxzZSBpZiAodHJhbnNwYXJlbmN5ID09IFRyYW5zcGFyZW5jeS5CSVRNQVNLKSB7Ci0gICAgICAgICAgICBtb2RlbCA9IG5ldyBEaXJlY3RDb2xvck1vZGVsKDI1LCAweDAwZmYwMDAwLCAweDAwMDBmZjAwLCAweDAwMDAwMGZmLCAweDAxMDAwMDAwKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG1vZGVsID0gbmV3IERpcmVjdENvbG9yTW9kZWwoMjQsIDB4MDBmZjAwMDAsIDB4MDAwMGZmMDAsIDB4MDAwMDAwZmYpOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHcgPSByYXN0ZXIuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IGggPSByYXN0ZXIuZ2V0SGVpZ2h0KCk7Ci0KLSAgICAgICAgV3JpdGFibGVSYXN0ZXIgZGlzdFJhc3RlciA9IG1vZGVsLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3Rlcih3LCBoKTsKLQotICAgICAgICBpbnQgbWluWCA9IHJhc3Rlci5nZXRNaW5YKCk7Ci0gICAgICAgIGludCBtaW5ZID0gcmFzdGVyLmdldE1pblkoKTsKLQotICAgICAgICBPYmplY3Qgb2JqID0gbnVsbDsKLSAgICAgICAgaW50IHBpeGVsc1tdID0gbnVsbDsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGg7IGkrKywgbWluWSsrKSB7Ci0gICAgICAgICAgICBvYmogPSByYXN0ZXIuZ2V0RGF0YUVsZW1lbnRzKG1pblgsIG1pblksIHcsIDEsIG9iaik7Ci0gICAgICAgICAgICBpZiAob2JqIGluc3RhbmNlb2YgYnl0ZVtdKSB7Ci0gICAgICAgICAgICAgICAgYnl0ZSBiYVtdID0gKGJ5dGVbXSlvYmo7Ci0gICAgICAgICAgICAgICAgaWYgKHBpeGVscyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHBpeGVscyA9IG5ldyBpbnRbYmEubGVuZ3RoXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDA7IGogPCBiYS5sZW5ndGg7IGorKykgewotICAgICAgICAgICAgICAgICAgICBwaXhlbHNbal0gPSBjb2xvck1hcFtiYVtqXSAmIDB4ZmZdOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSBpZiAob2JqIGluc3RhbmNlb2Ygc2hvcnRbXSkgewotICAgICAgICAgICAgICAgIHNob3J0IHNhW10gPSAoc2hvcnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgaWYgKHBpeGVscyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHBpeGVscyA9IG5ldyBpbnRbc2EubGVuZ3RoXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IDA7IGogPCBzYS5sZW5ndGg7IGorKykgewotICAgICAgICAgICAgICAgICAgICBwaXhlbHNbal0gPSBjb2xvck1hcFtzYVtqXSAmIDB4ZmZmZl07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKG9iaiBpbnN0YW5jZW9mIGludFtdKSB7Ci0gICAgICAgICAgICAgICAgaW50IGlhW10gPSAoaW50W10pb2JqOwotICAgICAgICAgICAgICAgIGlmIChwaXhlbHMgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBwaXhlbHMgPSBuZXcgaW50W2lhLmxlbmd0aF07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSAwOyBqIDwgaWEubGVuZ3RoOyBqKyspIHsKLSAgICAgICAgICAgICAgICAgICAgcGl4ZWxzW2pdID0gY29sb3JNYXBbaWFbal1dOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZGlzdFJhc3Rlci5zZXREYXRhRWxlbWVudHMoMCwgaSwgdywgMSwgcGl4ZWxzKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBuZXcgQnVmZmVyZWRJbWFnZShtb2RlbCwgZGlzdFJhc3RlciwgZmFsc2UsIG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHZhbGlkIHBpeGVscy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB2YWxpZCBwaXhlbHMuCi0gICAgICovCi0gICAgcHVibGljIEJpZ0ludGVnZXIgZ2V0VmFsaWRQaXhlbHMoKSB7Ci0gICAgICAgIHJldHVybiB2YWxpZEJpdHM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsKLSAgICAgICAgLy8gVGhlIG91dHB1dCBmb3JtYXQgYmFzZWQgb24gMS41IHJlbGVhc2UgYmVoYXZpb3VyLgotICAgICAgICAvLyBJdCBjb3VsZCBiZSByZXZlbGVkIHN1Y2ggd2F5OgotICAgICAgICAvLyBCdWZmZXJlZEltYWdlIGJpID0gbmV3IEJ1ZmZlcmVkSW1hZ2UoMSwgMSwKLSAgICAgICAgLy8gQnVmZmVyZWRJbWFnZS5UWVBFX0JZVEVfSU5ERVhFRCk7Ci0gICAgICAgIC8vIENvbG9yTW9kZWwgY20gPSBiaS5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIC8vIFN5c3RlbS5vdXQucHJpbnRsbihjbS50b1N0cmluZygpKTsKLSAgICAgICAgU3RyaW5nIHN0ciA9ICJJbmRleENvbG9yTW9kZWw6ICNwaXhlbF9iaXRzID0gIiArIHBpeGVsX2JpdHMgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgIiBudW1Db21wb25lbnRzID0gIiArIG51bUNvbXBvbmVudHMgKyAiIGNvbG9yIHNwYWNlID0gIiArIGNzICsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgIiB0cmFuc3BhcmVuY3kgPSAiOyAvLyROT04tTkxTLTEkCi0KLSAgICAgICAgaWYgKHRyYW5zcGFyZW5jeSA9PSBUcmFuc3BhcmVuY3kuT1BBUVVFKSB7Ci0gICAgICAgICAgICBzdHIgPSBzdHIgKyAiVHJhbnNwYXJlbmN5Lk9QQVFVRSI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfSBlbHNlIGlmICh0cmFuc3BhcmVuY3kgPT0gVHJhbnNwYXJlbmN5LkJJVE1BU0spIHsKLSAgICAgICAgICAgIHN0ciA9IHN0ciArICJUcmFuc3BhcmVuY3kuQklUTUFTSyI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHN0ciA9IHN0ciArICJUcmFuc3BhcmVuY3kuVFJBTlNMVUNFTlQiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBzdHIgPSBzdHIgKyAiIHRyYW5zSW5kZXggPSAiICsgdHJhbnNwYXJlbnRJbmRleCArICIgaGFzIGFscGhhID0gIiArIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgICAgIGhhc0FscGhhICsgIiBpc0FscGhhUHJlID0gIiArIGlzQWxwaGFQcmVtdWx0aXBsaWVkOyAvLyROT04tTkxTLTEkCi0KLSAgICAgICAgcmV0dXJuIHN0cjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50W10gZ2V0Q29tcG9uZW50cyhPYmplY3QgcGl4ZWwsIGludCBjb21wb25lbnRzW10sIGludCBvZmZzZXQpIHsKLSAgICAgICAgaW50IHBpeElkeCA9IC0xOwotICAgICAgICBpZiAocGl4ZWwgaW5zdGFuY2VvZiBieXRlW10pIHsKLSAgICAgICAgICAgIGJ5dGUgYmFbXSA9IChieXRlW10pcGl4ZWw7Ci0gICAgICAgICAgICBwaXhJZHggPSBiYVswXSAmIDB4ZmY7Ci0gICAgICAgIH0gZWxzZSBpZiAocGl4ZWwgaW5zdGFuY2VvZiBzaG9ydFtdKSB7Ci0gICAgICAgICAgICBzaG9ydCBzYVtdID0gKHNob3J0W10pcGl4ZWw7Ci0gICAgICAgICAgICBwaXhJZHggPSBzYVswXSAmIDB4ZmZmZjsKLSAgICAgICAgfSBlbHNlIGlmIChwaXhlbCBpbnN0YW5jZW9mIGludFtdKSB7Ci0gICAgICAgICAgICBpbnQgaWFbXSA9IChpbnRbXSlwaXhlbDsKLSAgICAgICAgICAgIHBpeElkeCA9IGlhWzBdOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLy8gYXd0LjIxOT1UaGlzIHRyYW5zZmVyVHlwZSBpcyBub3Qgc3VwcG9ydGVkIGJ5IHRoaXMgY29sb3IgbW9kZWwKLSAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMTkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBnZXRDb21wb25lbnRzKHBpeElkeCwgY29tcG9uZW50cywgb2Zmc2V0KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKGludCB3LCBpbnQgaCkgewotICAgICAgICBXcml0YWJsZVJhc3RlciByYXN0ZXI7Ci0gICAgICAgIGlmIChwaXhlbF9iaXRzID09IDEgfHwgcGl4ZWxfYml0cyA9PSAyIHx8IHBpeGVsX2JpdHMgPT0gNCkgewotICAgICAgICAgICAgcmFzdGVyID0gUmFzdGVyLmNyZWF0ZVBhY2tlZFJhc3RlcihEYXRhQnVmZmVyLlRZUEVfQllURSwgdywgaCwgMSwgcGl4ZWxfYml0cywgbnVsbCk7Ci0gICAgICAgIH0gZWxzZSBpZiAocGl4ZWxfYml0cyA8PSA4KSB7Ci0gICAgICAgICAgICByYXN0ZXIgPSBSYXN0ZXIuY3JlYXRlSW50ZXJsZWF2ZWRSYXN0ZXIoRGF0YUJ1ZmZlci5UWVBFX0JZVEUsIHcsIGgsIDEsIG51bGwpOwotICAgICAgICB9IGVsc2UgaWYgKHBpeGVsX2JpdHMgPD0gMTYpIHsKLSAgICAgICAgICAgIHJhc3RlciA9IFJhc3Rlci5jcmVhdGVJbnRlcmxlYXZlZFJhc3RlcihEYXRhQnVmZmVyLlRZUEVfVVNIT1JULCB3LCBoLCAxLCBudWxsKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNjY9VGhlIG51bWJlciBvZiBiaXRzIGluIGEgcGl4ZWwgaXMgZ3JlYXRlciB0aGFuIDE2Ci0gICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjY2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcmFzdGVyOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGlzQ29tcGF0aWJsZVNhbXBsZU1vZGVsKFNhbXBsZU1vZGVsIHNtKSB7Ci0gICAgICAgIGlmIChzbSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoIShzbSBpbnN0YW5jZW9mIE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkgJiYgIShzbSBpbnN0YW5jZW9mIENvbXBvbmVudFNhbXBsZU1vZGVsKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHNtLmdldFRyYW5zZmVyVHlwZSgpICE9IHRyYW5zZmVyVHlwZSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChzbS5nZXROdW1CYW5kcygpICE9IDEpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTYW1wbGVNb2RlbCBjcmVhdGVDb21wYXRpYmxlU2FtcGxlTW9kZWwoaW50IHcsIGludCBoKSB7Ci0gICAgICAgIGlmIChwaXhlbF9iaXRzID09IDEgfHwgcGl4ZWxfYml0cyA9PSAyIHx8IHBpeGVsX2JpdHMgPT0gNCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwoRGF0YUJ1ZmZlci5UWVBFX0JZVEUsIHcsIGgsIHBpeGVsX2JpdHMpOwotICAgICAgICB9Ci0gICAgICAgIGludCBiYW5kT2Zmc2V0c1tdID0gbmV3IGludFsxXTsKLSAgICAgICAgYmFuZE9mZnNldHNbMF0gPSAwOwotICAgICAgICByZXR1cm4gbmV3IENvbXBvbmVudFNhbXBsZU1vZGVsKHRyYW5zZmVyVHlwZSwgdywgaCwgMSwgdywgYmFuZE9mZnNldHMpOwotCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb21wYXRpYmxlUmFzdGVyKFJhc3RlciByYXN0ZXIpIHsKLSAgICAgICAgaW50IHNhbXBsZVNpemUgPSByYXN0ZXIuZ2V0U2FtcGxlTW9kZWwoKS5nZXRTYW1wbGVTaXplKDApOwotICAgICAgICByZXR1cm4gKHJhc3Rlci5nZXRUcmFuc2ZlclR5cGUoKSA9PSB0cmFuc2ZlclR5cGUgJiYgcmFzdGVyLmdldE51bUJhbmRzKCkgPT0gMSAmJiAoMSA8PCBzYW1wbGVTaXplKSA+PSBtYXBTaXplKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldERhdGFFbGVtZW50KGludCBjb21wb25lbnRzW10sIGludCBvZmZzZXQpIHsKLSAgICAgICAgaW50IHJnYiA9IChjb21wb25lbnRzW29mZnNldF0gPDwgMTYpIHwgKGNvbXBvbmVudHNbb2Zmc2V0ICsgMV0pIDw8IDgKLSAgICAgICAgICAgICAgICB8IGNvbXBvbmVudHNbb2Zmc2V0ICsgMl07Ci0KLSAgICAgICAgaWYgKGhhc0FscGhhKSB7Ci0gICAgICAgICAgICByZ2IgfD0gY29tcG9uZW50c1tvZmZzZXQgKyAzXSA8PCAyNDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJnYiB8PSAweGZmMDAwMDAwOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHBpeGVsOwotCi0gICAgICAgIHN3aXRjaCAodHJhbnNmZXJUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGJ5dGUgYmFbXSA9IChieXRlW10pZ2V0RGF0YUVsZW1lbnRzKHJnYiwgbnVsbCk7Ci0gICAgICAgICAgICAgICAgcGl4ZWwgPSBiYVswXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2FbXSA9IChzaG9ydFtdKWdldERhdGFFbGVtZW50cyhyZ2IsIG51bGwpOwotICAgICAgICAgICAgICAgIHBpeGVsID0gc2FbMF0gJiAweGZmZmY7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4yNjc9VGhlIHRyYW5zZmVyVHlwZSBpcyBpbnZhbGlkCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2NyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHBpeGVsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvbG9yIG1hcC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmdiCi0gICAgICogICAgICAgICAgICB0aGUgZGVzdGluYXRpb24gYXJyYXkgd2hlcmUgdGhlIGNvbG9yIG1hcCBpcyB3cml0dGVuLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCB2b2lkIGdldFJHQnMoaW50IHJnYltdKSB7Ci0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkoY29sb3JNYXAsIDAsIHJnYiwgMCwgbWFwU2l6ZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcmVkIGNvbXBvbmVudCBvZiB0aGUgY29sb3IgbWFwLgotICAgICAqIAotICAgICAqIEBwYXJhbSByCi0gICAgICogICAgICAgICAgICB0aGUgZGVzdGluYXRpb24gYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIHZvaWQgZ2V0UmVkcyhieXRlIHJbXSkgewotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG1hcFNpemU7IGkrKykgewotICAgICAgICAgICAgcltpXSA9IChieXRlKShjb2xvck1hcFtpXSA+PiAxNik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBncmVlbiBjb21wb25lbnQgb2YgdGhlIGNvbG9yIG1hcC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZwotICAgICAqICAgICAgICAgICAgdGhlIGRlc3RpbmF0aW9uIGFycmF5LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCB2b2lkIGdldEdyZWVucyhieXRlIGdbXSkgewotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG1hcFNpemU7IGkrKykgewotICAgICAgICAgICAgZ1tpXSA9IChieXRlKShjb2xvck1hcFtpXSA+PiA4KTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJsdWUgY29tcG9uZW50IG9mIHRoZSBjb2xvciBtYXAuCi0gICAgICogCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXN0aW5hdGlvbiBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgdm9pZCBnZXRCbHVlcyhieXRlIGJbXSkgewotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG1hcFNpemU7IGkrKykgewotICAgICAgICAgICAgYltpXSA9IChieXRlKWNvbG9yTWFwW2ldOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYWxwaGEgY29tcG9uZW50IG9mIHRoZSBjb2xvciBtYXAuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXN0aW5hdGlvbiBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgdm9pZCBnZXRBbHBoYXMoYnl0ZSBhW10pIHsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBtYXBTaXplOyBpKyspIHsKLSAgICAgICAgICAgIGFbaV0gPSAoYnl0ZSkoY29sb3JNYXBbaV0gPj4gMjQpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldENvbXBvbmVudHMoaW50IHBpeGVsLCBpbnQgY29tcG9uZW50c1tdLCBpbnQgb2Zmc2V0KSB7Ci0gICAgICAgIGlmIChjb21wb25lbnRzID09IG51bGwpIHsKLSAgICAgICAgICAgIGNvbXBvbmVudHMgPSBuZXcgaW50W29mZnNldCArIG51bUNvbXBvbmVudHNdOwotICAgICAgICB9Ci0KLSAgICAgICAgY29tcG9uZW50c1tvZmZzZXQgKyAwXSA9IGdldFJlZChwaXhlbCk7Ci0gICAgICAgIGNvbXBvbmVudHNbb2Zmc2V0ICsgMV0gPSBnZXRHcmVlbihwaXhlbCk7Ci0gICAgICAgIGNvbXBvbmVudHNbb2Zmc2V0ICsgMl0gPSBnZXRCbHVlKHBpeGVsKTsKLSAgICAgICAgaWYgKGhhc0FscGhhICYmIChjb21wb25lbnRzLmxlbmd0aCAtIG9mZnNldCkgPiAzKSB7Ci0gICAgICAgICAgICBjb21wb25lbnRzW29mZnNldCArIDNdID0gZ2V0QWxwaGEocGl4ZWwpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGNvbXBvbmVudHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoZSBzcGVjaWZpZWQgcGl4ZWwgaXMgdmFsaWQgZm9yIHRoaXMgY29sb3IgbW9kZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBpeGVsCi0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgcGl4ZWwgaXMgdmFsaWQuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNWYWxpZChpbnQgcGl4ZWwpIHsKLSAgICAgICAgaWYgKHZhbGlkQml0cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gKHBpeGVsID49IDAgJiYgcGl4ZWwgPCBtYXBTaXplKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gKHBpeGVsIDwgbWFwU2l6ZSAmJiB2YWxpZEJpdHMudGVzdEJpdChwaXhlbCkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0UmVkKGludCBwaXhlbCkgewotICAgICAgICByZXR1cm4gKGNvbG9yTWFwW3BpeGVsXSA+PiAxNikgJiAweGZmOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0UkdCKGludCBwaXhlbCkgewotICAgICAgICByZXR1cm4gY29sb3JNYXBbcGl4ZWxdOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0R3JlZW4oaW50IHBpeGVsKSB7Ci0gICAgICAgIHJldHVybiAoY29sb3JNYXBbcGl4ZWxdID4+IDgpICYgMHhmZjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmluYWwgaW50IGdldEJsdWUoaW50IHBpeGVsKSB7Ci0gICAgICAgIHJldHVybiBjb2xvck1hcFtwaXhlbF0gJiAweGZmOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0QWxwaGEoaW50IHBpeGVsKSB7Ci0gICAgICAgIHJldHVybiAoY29sb3JNYXBbcGl4ZWxdID4+IDI0KSAmIDB4ZmY7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldENvbXBvbmVudFNpemUoKSB7Ci0gICAgICAgIHJldHVybiBiaXRzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoaXMgY29sb3IgbW9kZWwgdmFsaWRhdGVzIHBpeGVscy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGFsbCBwaXhlbHMgYXJlIHZhbGlkLCBvdGhlcndpc2UgZmFsc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNWYWxpZCgpIHsKLSAgICAgICAgcmV0dXJuICh2YWxpZEJpdHMgPT0gbnVsbCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZmluYWxpemUoKSB7Ci0gICAgICAgIC8vIFRPRE86IGltcGxlbWVudAotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW5kZXggdGhhdCByZXByZXNlbnRzIHRoZSB0cmFuc3BhcmVudCBwaXhlbC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBpbmRleCB0aGF0IHJlcHJlc2VudHMgdGhlIHRyYW5zcGFyZW50IHBpeGVsLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0VHJhbnNwYXJlbnRQaXhlbCgpIHsKLSAgICAgICAgcmV0dXJuIHRyYW5zcGFyZW50SW5kZXg7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRUcmFuc3BhcmVuY3koKSB7Ci0gICAgICAgIHJldHVybiB0cmFuc3BhcmVuY3k7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2l6ZSBvZiB0aGUgY29sb3IgbWFwLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG1hcCBzaXplLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0TWFwU2l6ZSgpIHsKLSAgICAgICAgcmV0dXJuIG1hcFNpemU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgY29sb3IgbWFwLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzaXplCi0gICAgICogICAgICAgICAgICB0aGUgc2l6ZS4KLSAgICAgKiBAcGFyYW0gcgotICAgICAqICAgICAgICAgICAgdGhlIHIuCi0gICAgICogQHBhcmFtIGcKLSAgICAgKiAgICAgICAgICAgIHRoZSBnLgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgYi4KLSAgICAgKiBAcGFyYW0gYQotICAgICAqICAgICAgICAgICAgdGhlIGEuCi0gICAgICogQHBhcmFtIHRyYW5zCi0gICAgICogICAgICAgICAgICB0aGUgdHJhbnMuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIGNyZWF0ZUNvbG9yTWFwKGludCBzaXplLCBieXRlIHJbXSwgYnl0ZSBnW10sIGJ5dGUgYltdLCBieXRlIGFbXSwgaW50IHRyYW5zKSB7Ci0gICAgICAgIGlmIChzaXplIDwgMSkgewotICAgICAgICAgICAgLy8gYXd0LjI2ND1TaXplIG9mIHRoZSBjb2xvciBtYXAgaXMgbGVzcyB0aGFuIDEKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjY0IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBtYXBTaXplID0gc2l6ZTsKLSAgICAgICAgY29sb3JNYXAgPSBuZXcgaW50W21hcFNpemVdOwotICAgICAgICBpZiAodHJhbnMgPj0gMCAmJiB0cmFucyA8IG1hcFNpemUpIHsKLSAgICAgICAgICAgIHRyYW5zcGFyZW5jeSA9IFRyYW5zcGFyZW5jeS5CSVRNQVNLOwotICAgICAgICAgICAgdHJhbnNwYXJlbnRJbmRleCA9IHRyYW5zOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdHJhbnNwYXJlbmN5ID0gVHJhbnNwYXJlbmN5Lk9QQVFVRTsKLSAgICAgICAgICAgIHRyYW5zcGFyZW50SW5kZXggPSAtMTsKLSAgICAgICAgfQotICAgICAgICBpbnQgYWxwaGEgPSAwOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbWFwU2l6ZTsgaSsrKSB7Ci0gICAgICAgICAgICBjb2xvck1hcFtpXSA9ICgocltpXSAmIDB4ZmYpIDw8IDE2KSB8ICgoZ1tpXSAmIDB4ZmYpIDw8IDgpIHwgKGJbaV0gJiAweGZmKTsKLQotICAgICAgICAgICAgaWYgKHRyYW5zID09IGkpIHsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKGEgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIGNvbG9yTWFwW2ldIHw9IDB4ZmYwMDAwMDA7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGFscGhhID0gYVtpXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgaWYgKGFscGhhID09IDB4ZmYpIHsKLSAgICAgICAgICAgICAgICAgICAgY29sb3JNYXBbaV0gfD0gMHhmZjAwMDAwMDsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGFscGhhID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHRyYW5zcGFyZW5jeSA9PSBUcmFuc3BhcmVuY3kuT1BBUVVFKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB0cmFuc3BhcmVuY3kgPSBUcmFuc3BhcmVuY3kuQklUTUFTSzsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpZiAodHJhbnNwYXJlbnRJbmRleCA8IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHRyYW5zcGFyZW50SW5kZXggPSBpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgY29sb3JNYXBbaV0gfD0gKGFbaV0gJiAweGZmKSA8PCAyNDsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHRyYW5zcGFyZW5jeSAhPSBUcmFuc3BhcmVuY3kuVFJBTlNMVUNFTlQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHRyYW5zcGFyZW5jeSA9IFRyYW5zcGFyZW5jeS5UUkFOU0xVQ0VOVDsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCBjaGVja2luZywgaWYgQ29sb3IgTWFwIGhhcyBHcmF5IHBhbGV0dGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIGNoZWNrUGFsZXR0ZSgpIHsKLSAgICAgICAgZ3JheVBhbGV0dGUgPSBmYWxzZTsKLSAgICAgICAgaWYgKHRyYW5zcGFyZW5jeSA+IFRyYW5zcGFyZW5jeS5PUEFRVUUpIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBpbnQgcmdiID0gMDsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG1hcFNpemU7IGkrKykgewotICAgICAgICAgICAgcmdiID0gY29sb3JNYXBbaV07Ci0gICAgICAgICAgICBpZiAoKChyZ2IgPj4gMTYpICYgMHhmZikgIT0gKChyZ2IgPj4gOCkgJiAweGZmKSB8fCAoKHJnYiA+PiA4KSAmIDB4ZmYpICE9IChyZ2IgJiAweGZmKSkgewotICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBncmF5UGFsZXR0ZSA9IHRydWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29uc3RydWN0aW9uIGFuIGFycmF5IHBpeGVsIHJlcHJlc2VudGF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvck1hcElkeAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IGludG8gQ29sb3IgTWFwLgotICAgICAqIEBwYXJhbSBwaXhlbAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsCi0gICAgICogQHJldHVybiB0aGUgcGl4ZWwgcmVwcmVzZW50YXRpb24gYXJyYXkuCi0gICAgICovCi0gICAgcHJpdmF0ZSBPYmplY3QgY3JlYXRlRGF0YU9iamVjdChpbnQgY29sb3JNYXBJZHgsIE9iamVjdCBwaXhlbCkgewotICAgICAgICBpZiAocGl4ZWwgPT0gbnVsbCkgewotICAgICAgICAgICAgc3dpdGNoICh0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgICAgICBieXRlW10gYmEgPSBuZXcgYnl0ZVsxXTsKLSAgICAgICAgICAgICAgICAgICAgYmFbMF0gPSAoYnl0ZSljb2xvck1hcElkeDsKLSAgICAgICAgICAgICAgICAgICAgcGl4ZWwgPSBiYTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgICAgICBzaG9ydFtdIHNhID0gbmV3IHNob3J0WzFdOwotICAgICAgICAgICAgICAgICAgICBzYVswXSA9IChzaG9ydCljb2xvck1hcElkeDsKLSAgICAgICAgICAgICAgICAgICAgcGl4ZWwgPSBzYTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjI2Nz1UaGUgdHJhbnNmZXJUeXBlIGlzIGludmFsaWQKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2NyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgaWYgKHBpeGVsIGluc3RhbmNlb2YgYnl0ZVtdICYmIHRyYW5zZmVyVHlwZSA9PSBEYXRhQnVmZmVyLlRZUEVfQllURSkgewotICAgICAgICAgICAgYnl0ZSBiYVtdID0gKGJ5dGVbXSlwaXhlbDsKLSAgICAgICAgICAgIGJhWzBdID0gKGJ5dGUpY29sb3JNYXBJZHg7Ci0gICAgICAgICAgICBwaXhlbCA9IGJhOwotICAgICAgICB9IGVsc2UgaWYgKHBpeGVsIGluc3RhbmNlb2Ygc2hvcnRbXSAmJiB0cmFuc2ZlclR5cGUgPT0gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCkgewotICAgICAgICAgICAgc2hvcnRbXSBzYSA9IChzaG9ydFtdKXBpeGVsOwotICAgICAgICAgICAgc2FbMF0gPSAoc2hvcnQpY29sb3JNYXBJZHg7Ci0gICAgICAgICAgICBwaXhlbCA9IHNhOwotICAgICAgICB9IGVsc2UgaWYgKHBpeGVsIGluc3RhbmNlb2YgaW50W10pIHsKLSAgICAgICAgICAgIGludCBpYVtdID0gKGludFtdKXBpeGVsOwotICAgICAgICAgICAgaWFbMF0gPSBjb2xvck1hcElkeDsKLSAgICAgICAgICAgIHBpeGVsID0gaWE7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjY4PVRoZSBwaXhlbCBpcyBub3QgYSBwcmltaXRpdmUgYXJyYXkgb2YgdHlwZSB0cmFuc2ZlclR5cGUKLSAgICAgICAgICAgIHRocm93IG5ldyBDbGFzc0Nhc3RFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjY4IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHBpeGVsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGJpdHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGhhc0FscGhhCi0gICAgICogICAgICAgICAgICB0aGUgaGFzIGFscGhhLgotICAgICAqIEByZXR1cm4gdGhlIGludFtdLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGludFtdIGNyZWF0ZUJpdHMoYm9vbGVhbiBoYXNBbHBoYSkgewotCi0gICAgICAgIGludCBudW1DaGFubmVsczsKLSAgICAgICAgaWYgKGhhc0FscGhhKSB7Ci0gICAgICAgICAgICBudW1DaGFubmVscyA9IDQ7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBudW1DaGFubmVscyA9IDM7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgYml0c1tdID0gbmV3IGludFtudW1DaGFubmVsc107Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQ2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgYml0c1tpXSA9IDg7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gYml0czsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFZhbGlkYXRlIHRyYW5zZmVyIHR5cGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRyYW5zZmVyVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zZmVyIHR5cGUuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGludCB2YWxpZGF0ZVRyYW5zZmVyVHlwZShpbnQgdHJhbnNmZXJUeXBlKSB7Ci0gICAgICAgIGlmICh0cmFuc2ZlclR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYgdHJhbnNmZXJUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9VU0hPUlQpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNjk9VGhlIHRyYW5zZmVyVHlwZSBpcyBub3Qgb25lIG9mIERhdGFCdWZmZXIuVFlQRV9CWVRFIG9yCi0gICAgICAgICAgICAvLyBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2OSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB0cmFuc2ZlclR5cGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIGlzIGdyYXkgcGFsZXR0ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGlzIGdyYXkgcGFsZXR0ZS4KLSAgICAgKi8KLSAgICBib29sZWFuIGlzR3JheVBhbGxldGUoKSB7Ci0gICAgICAgIHJldHVybiBncmF5UGFsZXR0ZTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9LZXJuZWwuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9LZXJuZWwuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTU5ZDI3YS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvS2VybmVsLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNTMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKgotICogQGRhdGU6IFNlcCAyOCwgMjAwNQotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgS2VybmVsIGNsYXNzIHByb3ZpZGVzIGEgbWF0cml4LiBUaGlzIG1hdHJpeCBpcyBzdG9yZWQgYXMgYSBmbG9hdCBhcnJheQotICogd2hpY2ggZGVzY3JpYmVzIGhvdyBhIHNwZWNpZmllZCBwaXhlbCBhZmZlY3RzIHRoZSB2YWx1ZSBjYWxjdWxhdGVkIGZvciB0aGUKLSAqIHBpeGVsJ3MgcG9zaXRpb24gaW4gdGhlIG91dHB1dCBpbWFnZSBvZiBhIGZpbHRlcmluZyBvcGVyYXRpb24uIFRoZSBYLCBZCi0gKiBvcmlnaW5zIGluZGljYXRlIHRoZSBrZXJuZWwgbWF0cml4IGVsZW1lbnQgd2hpY2ggY29ycmVzcG9uZHMgdG8gdGhlIHBpeGVsCi0gKiBwb3NpdGlvbiBmb3Igd2hpY2ggYW4gb3V0cHV0IHZhbHVlIGlzIGJlaW5nIGNhbGN1bGF0ZWQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgS2VybmVsIGltcGxlbWVudHMgQ2xvbmVhYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSB4IG9yaWdpbi4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIGludCB4T3JpZ2luOwotCi0gICAgLyoqCi0gICAgICogVGhlIHkgb3JpZ2luLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgaW50IHlPcmlnaW47Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgd2lkdGguCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgd2lkdGg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGVpZ2h0LgotICAgICAqLwotICAgIHByaXZhdGUgaW50IGhlaWdodDsKLQotICAgIC8qKgotICAgICAqIFRoZSBkYXRhLgotICAgICAqLwotICAgIGZsb2F0IGRhdGFbXTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBLZXJuZWwgd2l0aCB0aGUgc3BlY2lmaWVkIGZsb2F0IGFycmF5LiBUaGUKLSAgICAgKiB3aWR0aCpoZWlnaHQgZWxlbWVudHMgb2YgdGhlIGRhdGEgYXJyYXkgYXJlIGNvcGllZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgS2VybmVsLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIEtlcm5lbC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgb2YgS2VybmVsLgotICAgICAqLwotICAgIHB1YmxpYyBLZXJuZWwoaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBmbG9hdFtdIGRhdGEpIHsKLSAgICAgICAgaW50IGRhdGFMZW5ndGggPSB3aWR0aCAqIGhlaWdodDsKLSAgICAgICAgaWYgKGRhdGEubGVuZ3RoIDwgZGF0YUxlbmd0aCkgewotICAgICAgICAgICAgLy8gYXd0LjIyQj1MZW5ndGggb2YgZGF0YSBzaG91bGQgbm90IGJlIGxlc3MgdGhhbiB3aWR0aCpoZWlnaHQKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjJCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICB0aGlzLndpZHRoID0gd2lkdGg7Ci0gICAgICAgIHRoaXMuaGVpZ2h0ID0gaGVpZ2h0OwotCi0gICAgICAgIHRoaXMuZGF0YSA9IG5ldyBmbG9hdFtkYXRhTGVuZ3RoXTsKLSAgICAgICAgU3lzdGVtLmFycmF5Y29weShkYXRhLCAwLCB0aGlzLmRhdGEsIDAsIGRhdGFMZW5ndGgpOwotCi0gICAgICAgIHhPcmlnaW4gPSAod2lkdGggLSAxKSAvIDI7Ci0gICAgICAgIHlPcmlnaW4gPSAoaGVpZ2h0IC0gMSkgLyAyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHdpZHRoIG9mIHRoaXMgS2VybmVsLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHdpZHRoIG9mIHRoaXMgS2VybmVsLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0V2lkdGgoKSB7Ci0gICAgICAgIHJldHVybiB3aWR0aDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBoZWlnaHQgb2YgdGhpcyBLZXJuZWwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaGVpZ2h0IG9mIHRoaXMgS2VybmVsLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0SGVpZ2h0KCkgewotICAgICAgICByZXR1cm4gaGVpZ2h0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZsb2F0IGRhdGEgYXJyYXkgb2YgdGhpcyBLZXJuZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbG9hdCBhcnJheSB3aGVyZSB0aGUgcmVzdWx0ZWQgZGF0YSB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBmbG9hdCBkYXRhIGFycmF5IG9mIHRoaXMgS2VybmVsLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBmbG9hdFtdIGdldEtlcm5lbERhdGEoZmxvYXRbXSBkYXRhKSB7Ci0gICAgICAgIGlmIChkYXRhID09IG51bGwpIHsKLSAgICAgICAgICAgIGRhdGEgPSBuZXcgZmxvYXRbdGhpcy5kYXRhLmxlbmd0aF07Ci0gICAgICAgIH0KLSAgICAgICAgU3lzdGVtLmFycmF5Y29weSh0aGlzLmRhdGEsIDAsIGRhdGEsIDAsIHRoaXMuZGF0YS5sZW5ndGgpOwotCi0gICAgICAgIHJldHVybiBkYXRhOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFggb3JpZ2luIG9mIHRoaXMgS2VybmVsLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFggb3JpZ2luIG9mIHRoaXMgS2VybmVsLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0WE9yaWdpbigpIHsKLSAgICAgICAgcmV0dXJuIHhPcmlnaW47Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgWSBvcmlnaW4gb2YgdGhpcyBLZXJuZWwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgWSBvcmlnaW4gb2YgdGhpcyBLZXJuZWwuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGludCBnZXRZT3JpZ2luKCkgewotICAgICAgICByZXR1cm4geU9yaWdpbjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgY29weSBvZiB0aGlzIEtlcm5lbCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY29weSBvZiB0aGlzIEtlcm5lbCBvYmplY3QuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBjbG9uZSgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBLZXJuZWwod2lkdGgsIGhlaWdodCwgZGF0YSk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0xvb2t1cE9wLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvTG9va3VwT3AuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzM2MmM1Yy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvTG9va3VwT3AuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDY2MSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqCi0gKiBAZGF0ZTogT2N0IDE0LCAyMDA1Ci0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEuYXd0Lio7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBvaW50MkQ7Ci1pbXBvcnQgamF2YS51dGlsLkFycmF5czsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBMb29rdXBPcCBjbGFzcyBwZXJmb3JtcyBhIGxvb2t1cCBvcGVyYXRpb24gd2hpY2ggdHJhbnNmb3JtcyBhIHNvdXJjZQotICogaW1hZ2UgYnkgZmlsdGVyaW5nIGVhY2ggYmFuZCB1c2luZyBhIHRhYmxlIG9mIGRhdGEuIFRoZSB0YWJsZSBtYXkgY29udGFpbiBhCi0gKiBzaW5nbGUgYXJyYXkgb3IgaXQgbWF5IGNvbnRhaW4gYSBkaWZmZXJlbnQgZGF0YSBhcnJheSBmb3IgZWFjaCBiYW5kIG9mIHRoZQotICogaW1hZ2UuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgTG9va3VwT3AgaW1wbGVtZW50cyBCdWZmZXJlZEltYWdlT3AsIFJhc3Rlck9wIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBsdXQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBMb29rdXBUYWJsZSBsdXQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGludHMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBSZW5kZXJpbmdIaW50cyBoaW50czsKLQotICAgIC8vIFRPRE8gcmVtb3ZlIHdoZW4gdGhpcyBmaWVsZCBpcyB1c2VkCi0gICAgLyoqCi0gICAgICogVGhlIGNhbiB1c2UgaXBwLgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bnVzZWQiKQotICAgIHByaXZhdGUgZmluYWwgYm9vbGVhbiBjYW5Vc2VJcHA7Ci0KLSAgICAvLyBXZSBkb24ndCBjcmVhdGUgbGV2ZWxzL3ZhbHVlcyB3aGVuIGl0IGlzIHBvc3NpYmxlIHRvIHJldXNlIG9sZAotICAgIC8qKgotICAgICAqIFRoZSBjYWNoZWQgbGV2ZWxzLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IGNhY2hlZExldmVsc1tdOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNhY2hlZCB2YWx1ZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgY2FjaGVkVmFsdWVzW107Ci0KLSAgICAvLyBOdW1iZXIgb2YgY2hhbm5lbHMgZm9yIHdoaWNoIGNhY2hlIGlzIHZhbGlkLgotICAgIC8vIElmIG5lZ2F0aXZlIG51bWJlciBvZiBjaGFubmVscyBpcyBzYW1lIGFzIHBvc2l0aXZlIGJ1dCBza2lwQWxwaGEgd2FzCi0gICAgLy8gc3BlY2lmaWVkCi0gICAgLyoqCi0gICAgICogVGhlIHZhbGlkIGZvciBjaGFubmVscy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCB2YWxpZEZvckNoYW5uZWxzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGxldmVsIGluaXRpYWxpemVyLgotICAgICAqLwotICAgIHN0YXRpYyBpbnQgbGV2ZWxJbml0aWFsaXplcltdID0gbmV3IGludFsweDEwMDAwXTsKLQotICAgIHN0YXRpYyB7Ci0gICAgICAgIC8vIFRPRE8KLSAgICAgICAgLy8gU3lzdGVtLmxvYWRMaWJyYXJ5KCJpbWFnZW9wcyIpOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAxOyBpIDw9IDB4MTAwMDA7IGkrKykgewotICAgICAgICAgICAgbGV2ZWxJbml0aWFsaXplcltpIC0gMV0gPSBpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IExvb2t1cE9wIG9iamVjdCBmcm9tIHRoZSBzcGVjaWZpZWQgTG9va3VwVGFibGUgb2JqZWN0Ci0gICAgICogYW5kIGEgUmVuZGVyaW5nSGludHMgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsb29rdXAKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgTG9va3VwVGFibGUgb2JqZWN0LgotICAgICAqIEBwYXJhbSBoaW50cwotICAgICAqICAgICAgICAgICAgdGhlIFJlbmRlcmluZ0hpbnRzIG9iamVjdCBvciBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBMb29rdXBPcChMb29rdXBUYWJsZSBsb29rdXAsIFJlbmRlcmluZ0hpbnRzIGhpbnRzKSB7Ci0gICAgICAgIGlmIChsb29rdXAgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjAxIiwgImxvb2t1cCIpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLSAgICAgICAgbHV0ID0gbG9va3VwOwotICAgICAgICB0aGlzLmhpbnRzID0gaGludHM7Ci0gICAgICAgIGNhblVzZUlwcCA9IGx1dCBpbnN0YW5jZW9mIEJ5dGVMb29rdXBUYWJsZSB8fCBsdXQgaW5zdGFuY2VvZiBTaG9ydExvb2t1cFRhYmxlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIExvb2t1cFRhYmxlIG9mIHRoZSBzcGVjaWZpZWQgT2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIExvb2t1cFRhYmxlIG9mIHRoZSBzcGVjaWZpZWQgT2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBMb29rdXBUYWJsZSBnZXRUYWJsZSgpIHsKLSAgICAgICAgcmV0dXJuIGx1dDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUmVuZGVyaW5nSGludHMgZ2V0UmVuZGVyaW5nSGludHMoKSB7Ci0gICAgICAgIHJldHVybiBoaW50czsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUG9pbnQyRCBnZXRQb2ludDJEKFBvaW50MkQgc3JjUHQsIFBvaW50MkQgZHN0UHQpIHsKLSAgICAgICAgaWYgKGRzdFB0ID09IG51bGwpIHsKLSAgICAgICAgICAgIGRzdFB0ID0gbmV3IFBvaW50MkQuRmxvYXQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGRzdFB0LnNldExvY2F0aW9uKHNyY1B0KTsKLSAgICAgICAgcmV0dXJuIGRzdFB0OwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRChSYXN0ZXIgc3JjKSB7Ci0gICAgICAgIHJldHVybiBzcmMuZ2V0Qm91bmRzKCk7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKEJ1ZmZlcmVkSW1hZ2Ugc3JjKSB7Ci0gICAgICAgIHJldHVybiBnZXRCb3VuZHMyRChzcmMuZ2V0UmFzdGVyKCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBjcmVhdGVDb21wYXRpYmxlRGVzdFJhc3RlcihSYXN0ZXIgc3JjKSB7Ci0gICAgICAgIHJldHVybiBzcmMuY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKCk7Ci0gICAgfQotCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UgY3JlYXRlQ29tcGF0aWJsZURlc3RJbWFnZShCdWZmZXJlZEltYWdlIHNyYywgQ29sb3JNb2RlbCBkc3RDTSkgewotICAgICAgICBpZiAoZHN0Q00gPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0Q00gPSBzcmMuZ2V0Q29sb3JNb2RlbCgpOwotCi0gICAgICAgICAgICAvLyBTeW5jIHRyYW5zZmVyIHR5cGUgd2l0aCBMVVQgZm9yIGNvbXBvbmVudCBjb2xvciBtb2RlbAotICAgICAgICAgICAgaWYgKGRzdENNIGluc3RhbmNlb2YgQ29tcG9uZW50Q29sb3JNb2RlbCkgewotICAgICAgICAgICAgICAgIGludCB0cmFuc2ZlclR5cGUgPSBkc3RDTS5nZXRUcmFuc2ZlclR5cGUoKTsKLSAgICAgICAgICAgICAgICBpZiAobHV0IGluc3RhbmNlb2YgQnl0ZUxvb2t1cFRhYmxlKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRyYW5zZmVyVHlwZSA9IERhdGFCdWZmZXIuVFlQRV9CWVRFOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAobHV0IGluc3RhbmNlb2YgU2hvcnRMb29rdXBUYWJsZSkgewotICAgICAgICAgICAgICAgICAgICB0cmFuc2ZlclR5cGUgPSBEYXRhQnVmZmVyLlRZUEVfU0hPUlQ7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgZHN0Q00gPSBuZXcgQ29tcG9uZW50Q29sb3JNb2RlbChkc3RDTS5jcywgZHN0Q00uaGFzQWxwaGEoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdENNLmlzQWxwaGFQcmVtdWx0aXBsaWVkLCBkc3RDTS50cmFuc3BhcmVuY3ksIHRyYW5zZmVyVHlwZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBXcml0YWJsZVJhc3RlciByID0gZHN0Q00uaXNDb21wYXRpYmxlU2FtcGxlTW9kZWwoc3JjLmdldFNhbXBsZU1vZGVsKCkpID8gc3JjLmdldFJhc3RlcigpCi0gICAgICAgICAgICAgICAgLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3RlcihzcmMuZ2V0V2lkdGgoKSwgc3JjLmdldEhlaWdodCgpKSA6IGRzdENNCi0gICAgICAgICAgICAgICAgLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3RlcihzcmMuZ2V0V2lkdGgoKSwgc3JjLmdldEhlaWdodCgpKTsKLQotICAgICAgICByZXR1cm4gbmV3IEJ1ZmZlcmVkSW1hZ2UoZHN0Q00sIHIsIGRzdENNLmlzQWxwaGFQcmVtdWx0aXBsaWVkKCksIG51bGwpOwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBXcml0YWJsZVJhc3RlciBmaWx0ZXIoUmFzdGVyIHNyYywgV3JpdGFibGVSYXN0ZXIgZHN0KSB7Ci0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0ID0gY3JlYXRlQ29tcGF0aWJsZURlc3RSYXN0ZXIoc3JjKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGlmIChzcmMuZ2V0TnVtQmFuZHMoKSAhPSBkc3QuZ2V0TnVtQmFuZHMoKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjM3IikpOyAvLyROT04tTkxTLTEkICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHNyYy5nZXRXaWR0aCgpICE9IGRzdC5nZXRXaWR0aCgpKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOEYiKSk7IC8vJE5PTi1OTFMtMSQgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoc3JjLmdldEhlaWdodCgpICE9IGRzdC5nZXRIZWlnaHQoKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjkwIikpOyAvLyROT04tTkxTLTEkICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGx1dC5nZXROdW1Db21wb25lbnRzKCkgIT0gMSAmJiBsdXQuZ2V0TnVtQ29tcG9uZW50cygpICE9IHNyYy5nZXROdW1CYW5kcygpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjM4PVRoZSBudW1iZXIgb2YgYXJyYXlzIGluIHRoZSBMb29rdXBUYWJsZSBkb2VzIG5vdCBtZWV0IHRoZQotICAgICAgICAgICAgLy8gcmVzdHJpY3Rpb25zCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzOCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgLy8gVE9ETwotICAgICAgICAvLyBpZiAoIWNhblVzZUlwcCB8fCBpcHBGaWx0ZXIoc3JjLCBkc3QsIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9DVVNUT00sCi0gICAgICAgIC8vIGZhbHNlKSAhPSAwKQotICAgICAgICBpZiAoc2xvd0ZpbHRlcihzcmMsIGRzdCwgZmFsc2UpICE9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMUY9VW5hYmxlIHRvIHRyYW5zZm9ybSBzb3VyY2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbWFnaW5nT3BFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjFGIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZHN0OwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBCdWZmZXJlZEltYWdlIGZpbHRlcihCdWZmZXJlZEltYWdlIHNyYywgQnVmZmVyZWRJbWFnZSBkc3QpIHsKLSAgICAgICAgQ29sb3JNb2RlbCBzcmNDTSA9IHNyYy5nZXRDb2xvck1vZGVsKCk7Ci0KLSAgICAgICAgaWYgKHNyY0NNIGluc3RhbmNlb2YgSW5kZXhDb2xvck1vZGVsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjIwPVNvdXJjZSBzaG91bGQgbm90IGhhdmUgSW5kZXhDb2xvck1vZGVsCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIyMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgLy8gQ2hlY2sgaWYgdGhlIG51bWJlciBvZiBzY2FsaW5nIGZhY3RvcnMgbWF0Y2hlcyB0aGUgbnVtYmVyIG9mIGJhbmRzCi0gICAgICAgIGludCBuQ29tcG9uZW50cyA9IHNyY0NNLmdldE51bUNvbXBvbmVudHMoKTsKLSAgICAgICAgaW50IG5MVVRDb21wb25lbnRzID0gbHV0LmdldE51bUNvbXBvbmVudHMoKTsKLSAgICAgICAgYm9vbGVhbiBza2lwQWxwaGE7Ci0gICAgICAgIGlmIChzcmNDTS5oYXNBbHBoYSgpKSB7Ci0gICAgICAgICAgICBpZiAobkxVVENvbXBvbmVudHMgPT0gMSB8fCBuTFVUQ29tcG9uZW50cyA9PSBuQ29tcG9uZW50cyAtIDEpIHsKLSAgICAgICAgICAgICAgICBza2lwQWxwaGEgPSB0cnVlOwotICAgICAgICAgICAgfSBlbHNlIGlmIChuTFVUQ29tcG9uZW50cyA9PSBuQ29tcG9uZW50cykgewotICAgICAgICAgICAgICAgIHNraXBBbHBoYSA9IGZhbHNlOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjI5PU51bWJlciBvZiBjb21wb25lbnRzIGluIHRoZSBMVVQgZG9lcyBub3QgbWF0Y2ggdGhlCi0gICAgICAgICAgICAgICAgLy8gbnVtYmVyIG9mIGJhbmRzCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMjkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIGlmIChuTFVUQ29tcG9uZW50cyA9PSAxIHx8IG5MVVRDb21wb25lbnRzID09IG5Db21wb25lbnRzKSB7Ci0gICAgICAgICAgICBza2lwQWxwaGEgPSBmYWxzZTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMjk9TnVtYmVyIG9mIGNvbXBvbmVudHMgaW4gdGhlIExVVCBkb2VzIG5vdCBtYXRjaCB0aGUgbnVtYmVyCi0gICAgICAgICAgICAvLyBvZiBiYW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMjkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIEJ1ZmZlcmVkSW1hZ2UgZmluYWxEc3QgPSBudWxsOwotICAgICAgICBpZiAoZHN0ID09IG51bGwpIHsKLSAgICAgICAgICAgIGZpbmFsRHN0ID0gZHN0OwotICAgICAgICAgICAgZHN0ID0gY3JlYXRlQ29tcGF0aWJsZURlc3RJbWFnZShzcmMsIG51bGwpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaWYgKHNyYy5nZXRXaWR0aCgpICE9IGRzdC5nZXRXaWR0aCgpKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOTEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKHNyYy5nZXRIZWlnaHQoKSAhPSBkc3QuZ2V0SGVpZ2h0KCkpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI5MiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoIXNyY0NNLmVxdWFscyhkc3QuZ2V0Q29sb3JNb2RlbCgpKSkgewotICAgICAgICAgICAgICAgIC8vIFRyZWF0IEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfUkdCIGFuZAotICAgICAgICAgICAgICAgIC8vIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQiBhcyBzYW1lCi0gICAgICAgICAgICAgICAgaWYgKCEoKHNyYy5nZXRUeXBlKCkgPT0gQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9SR0IgfHwgc3JjLmdldFR5cGUoKSA9PSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0IpICYmIChkc3QKLSAgICAgICAgICAgICAgICAgICAgICAgIC5nZXRUeXBlKCkgPT0gQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9SR0IgfHwgZHN0LmdldFR5cGUoKSA9PSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0IpKSkgewotICAgICAgICAgICAgICAgICAgICBmaW5hbERzdCA9IGRzdDsKLSAgICAgICAgICAgICAgICAgICAgZHN0ID0gY3JlYXRlQ29tcGF0aWJsZURlc3RJbWFnZShzcmMsIG51bGwpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIFRPRE8KLSAgICAgICAgLy8gaWYgKCFjYW5Vc2VJcHAgfHwgaXBwRmlsdGVyKHNyYy5nZXRSYXN0ZXIoKSwgZHN0LmdldFJhc3RlcigpLAotICAgICAgICAvLyBzcmMuZ2V0VHlwZSgpLCBza2lwQWxwaGEpICE9IDApCi0gICAgICAgIGlmIChzbG93RmlsdGVyKHNyYy5nZXRSYXN0ZXIoKSwgZHN0LmdldFJhc3RlcigpLCBza2lwQWxwaGEpICE9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMUY9VW5hYmxlIHRvIHRyYW5zZm9ybSBzb3VyY2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbWFnaW5nT3BFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjFGIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoZmluYWxEc3QgIT0gbnVsbCkgewotICAgICAgICAgICAgR3JhcGhpY3MyRCBnID0gZmluYWxEc3QuY3JlYXRlR3JhcGhpY3MoKTsKLSAgICAgICAgICAgIGcuc2V0Q29tcG9zaXRlKEFscGhhQ29tcG9zaXRlLlNyYyk7Ci0gICAgICAgICAgICBnLmRyYXdJbWFnZShkc3QsIDAsIDAsIG51bGwpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZmluYWxEc3QgPSBkc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZHN0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNsb3cgZmlsdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMuCi0gICAgICogQHBhcmFtIGRzdAotICAgICAqICAgICAgICAgICAgdGhlIGRzdC4KLSAgICAgKiBAcGFyYW0gc2tpcEFscGhhCi0gICAgICogICAgICAgICAgICB0aGUgc2tpcCBhbHBoYS4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbnQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBpbnQgc2xvd0ZpbHRlcihSYXN0ZXIgc3JjLCBXcml0YWJsZVJhc3RlciBkc3QsIGJvb2xlYW4gc2tpcEFscGhhKSB7Ci0gICAgICAgIGludCBtaW5TcmNYID0gc3JjLmdldE1pblgoKTsKLSAgICAgICAgaW50IG1pbkRzdFggPSBkc3QuZ2V0TWluWCgpOwotICAgICAgICBpbnQgbWluU3JjWSA9IHNyYy5nZXRNaW5ZKCk7Ci0gICAgICAgIGludCBtaW5Ec3RZID0gZHN0LmdldE1pblkoKTsKLQotICAgICAgICBpbnQgc2tpcHBpbmdDaGFubmVscyA9IHNraXBBbHBoYSA/IDEgOiAwOwotICAgICAgICBpbnQgbnVtQmFuZHMyUHJvY2VzcyA9IHNyYy5nZXROdW1CYW5kcygpIC0gc2tpcHBpbmdDaGFubmVsczsKLQotICAgICAgICBpbnQgbnVtQmFuZHMgPSBzcmMuZ2V0TnVtQmFuZHMoKTsKLSAgICAgICAgaW50IHNyY0hlaWdodCA9IHNyYy5nZXRIZWlnaHQoKTsKLSAgICAgICAgaW50IHNyY1dpZHRoID0gc3JjLmdldFdpZHRoKCk7Ci0KLSAgICAgICAgaW50W10gcGl4ZWxzID0gbnVsbDsKLSAgICAgICAgaW50IG9mZnNldCA9IGx1dC5nZXRPZmZzZXQoKTsKLQotICAgICAgICBpZiAobHV0IGluc3RhbmNlb2YgQnl0ZUxvb2t1cFRhYmxlKSB7Ci0gICAgICAgICAgICBieXRlW11bXSBieXRlRGF0YSA9ICgoQnl0ZUxvb2t1cFRhYmxlKWx1dCkuZ2V0VGFibGUoKTsKLSAgICAgICAgICAgIHBpeGVscyA9IHNyYy5nZXRQaXhlbHMobWluU3JjWCwgbWluU3JjWSwgc3JjV2lkdGgsIHNyY0hlaWdodCwgcGl4ZWxzKTsKLQotICAgICAgICAgICAgaWYgKGx1dC5nZXROdW1Db21wb25lbnRzKCkgIT0gMSkgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcGl4ZWxzLmxlbmd0aDsgaSArPSBudW1CYW5kcykgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBiID0gMDsgYiA8IG51bUJhbmRzMlByb2Nlc3M7IGIrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgcGl4ZWxzW2kgKyBiXSA9IGJ5dGVEYXRhW2JdW3BpeGVsc1tpICsgYl0gLSBvZmZzZXRdICYgMHhGRjsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBwaXhlbHMubGVuZ3RoOyBpICs9IG51bUJhbmRzKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGIgPSAwOyBiIDwgbnVtQmFuZHMyUHJvY2VzczsgYisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwaXhlbHNbaSArIGJdID0gYnl0ZURhdGFbMF1bcGl4ZWxzW2kgKyBiXSAtIG9mZnNldF0gJiAweEZGOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBkc3Quc2V0UGl4ZWxzKG1pbkRzdFgsIG1pbkRzdFksIHNyY1dpZHRoLCBzcmNIZWlnaHQsIHBpeGVscyk7Ci0gICAgICAgIH0gZWxzZSBpZiAobHV0IGluc3RhbmNlb2YgU2hvcnRMb29rdXBUYWJsZSkgewotICAgICAgICAgICAgc2hvcnRbXVtdIHNob3J0RGF0YSA9ICgoU2hvcnRMb29rdXBUYWJsZSlsdXQpLmdldFRhYmxlKCk7Ci0gICAgICAgICAgICBwaXhlbHMgPSBzcmMuZ2V0UGl4ZWxzKG1pblNyY1gsIG1pblNyY1ksIHNyY1dpZHRoLCBzcmNIZWlnaHQsIHBpeGVscyk7Ci0KLSAgICAgICAgICAgIGlmIChsdXQuZ2V0TnVtQ29tcG9uZW50cygpICE9IDEpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHBpeGVscy5sZW5ndGg7IGkgKz0gbnVtQmFuZHMpIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgYiA9IDA7IGIgPCBudW1CYW5kczJQcm9jZXNzOyBiKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHBpeGVsc1tpICsgYl0gPSBzaG9ydERhdGFbYl1bcGl4ZWxzW2kgKyBiXSAtIG9mZnNldF0gJiAweEZGRkY7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcGl4ZWxzLmxlbmd0aDsgaSArPSBudW1CYW5kcykgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBiID0gMDsgYiA8IG51bUJhbmRzMlByb2Nlc3M7IGIrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgcGl4ZWxzW2kgKyBiXSA9IHNob3J0RGF0YVswXVtwaXhlbHNbaSArIGJdIC0gb2Zmc2V0XSAmIDB4RkZGRjsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZHN0LnNldFBpeGVscyhtaW5Ec3RYLCBtaW5Ec3RZLCBzcmNXaWR0aCwgc3JjSGVpZ2h0LCBwaXhlbHMpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaW50IHBpeGVsW10gPSBuZXcgaW50W3NyYy5nZXROdW1CYW5kcygpXTsKLSAgICAgICAgICAgIGludCBtYXhZID0gbWluU3JjWSArIHNyY0hlaWdodDsKLSAgICAgICAgICAgIGludCBtYXhYID0gbWluU3JjWCArIHNyY1dpZHRoOwotICAgICAgICAgICAgZm9yIChpbnQgc3JjWSA9IG1pblNyY1ksIGRzdFkgPSBtaW5Ec3RZOyBzcmNZIDwgbWF4WTsgc3JjWSsrLCBkc3RZKyspIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBzcmNYID0gbWluU3JjWCwgZHN0WCA9IG1pbkRzdFg7IHNyY1ggPCBtYXhYOyBzcmNYKyssIGRzdFgrKykgewotICAgICAgICAgICAgICAgICAgICBzcmMuZ2V0UGl4ZWwoc3JjWCwgc3JjWSwgcGl4ZWwpOwotICAgICAgICAgICAgICAgICAgICBsdXQubG9va3VwUGl4ZWwocGl4ZWwsIHBpeGVsKTsKLSAgICAgICAgICAgICAgICAgICAgZHN0LnNldFBpeGVsKGRzdFgsIGRzdFksIHBpeGVsKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBieXRlIGxldmVscy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2hhbm5lbHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBjaGFubmVscy4KLSAgICAgKiBAcGFyYW0gc2tpcEFscGhhCi0gICAgICogICAgICAgICAgICB0aGUgc2tpcCBhbHBoYS4KLSAgICAgKiBAcGFyYW0gbGV2ZWxzCi0gICAgICogICAgICAgICAgICB0aGUgbGV2ZWxzLgotICAgICAqIEBwYXJhbSB2YWx1ZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSB2YWx1ZXMuCi0gICAgICogQHBhcmFtIGNoYW5uZWxzT3JkZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBjaGFubmVscyBvcmRlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIHZvaWQgY3JlYXRlQnl0ZUxldmVscyhpbnQgY2hhbm5lbHMsIGJvb2xlYW4gc2tpcEFscGhhLCBpbnQgbGV2ZWxzW10sCi0gICAgICAgICAgICBpbnQgdmFsdWVzW10sIGludCBjaGFubmVsc09yZGVyW10pIHsKLSAgICAgICAgYnl0ZSBkYXRhW11bXSA9ICgoQnl0ZUxvb2t1cFRhYmxlKWx1dCkuZ2V0VGFibGUoKTsKLSAgICAgICAgaW50IG5MZXZlbHMgPSBkYXRhWzBdLmxlbmd0aDsKLSAgICAgICAgaW50IG9mZnNldCA9IGx1dC5nZXRPZmZzZXQoKTsKLQotICAgICAgICAvLyBVc2Ugb25lIGRhdGEgYXJyYXkgZm9yIGFsbCBjaGFubmVscyBvciB1c2Ugc2V2ZXJhbCBkYXRhIGFycmF5cwotICAgICAgICBpbnQgZGF0YUluY3JlbWVudCA9IGRhdGEubGVuZ3RoID4gMSA/IDEgOiAwOwotCi0gICAgICAgIGZvciAoaW50IGNoID0gMCwgZGF0YUlkeCA9IDA7IGNoIDwgY2hhbm5lbHM7IGRhdGFJZHggKz0gZGF0YUluY3JlbWVudCwgY2grKykgewotICAgICAgICAgICAgaW50IGNoYW5uZWxPZmZzZXQgPSBjaGFubmVsc09yZGVyID09IG51bGwgPyBjaCA6IGNoYW5uZWxzT3JkZXJbY2hdOwotICAgICAgICAgICAgaW50IGNoYW5uZWxCYXNlID0gbkxldmVscyAqIGNoYW5uZWxPZmZzZXQ7Ci0KLSAgICAgICAgICAgIC8vIFNraXAgbGFzdCBjaGFubmVsIGlmIG5lZWRlZCwgemVybyB2YWx1ZXMgYXJlIE9LIC0KLSAgICAgICAgICAgIC8vIG5vIGNoYW5nZXMgdG8gdGhlIGNoYW5uZWwgaW5mb3JtYXRpb24gd2lsbCBiZSBkb25lIGluIElQUAotICAgICAgICAgICAgaWYgKChjaGFubmVsT2Zmc2V0ID09IGNoYW5uZWxzIC0gMSAmJiBza2lwQWxwaGEpIHx8IChkYXRhSWR4ID49IGRhdGEubGVuZ3RoKSkgewotICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGxldmVsSW5pdGlhbGl6ZXIsIG9mZnNldCwgbGV2ZWxzLCBjaGFubmVsQmFzZSwgbkxldmVscyk7Ci0gICAgICAgICAgICBmb3IgKGludCBmcm9tID0gMCwgdG8gPSBjaGFubmVsQmFzZTsgZnJvbSA8IG5MZXZlbHM7IGZyb20rKywgdG8rKykgewotICAgICAgICAgICAgICAgIHZhbHVlc1t0b10gPSBkYXRhW2RhdGFJZHhdW2Zyb21dICYgMHhGRjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIHNob3J0IGxldmVscy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2hhbm5lbHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBjaGFubmVscy4KLSAgICAgKiBAcGFyYW0gc2tpcEFscGhhCi0gICAgICogICAgICAgICAgICB0aGUgc2tpcCBhbHBoYS4KLSAgICAgKiBAcGFyYW0gbGV2ZWxzCi0gICAgICogICAgICAgICAgICB0aGUgbGV2ZWxzLgotICAgICAqIEBwYXJhbSB2YWx1ZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSB2YWx1ZXMuCi0gICAgICogQHBhcmFtIGNoYW5uZWxzT3JkZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBjaGFubmVscyBvcmRlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIHZvaWQgY3JlYXRlU2hvcnRMZXZlbHMoaW50IGNoYW5uZWxzLCBib29sZWFuIHNraXBBbHBoYSwgaW50IGxldmVsc1tdLAotICAgICAgICAgICAgaW50IHZhbHVlc1tdLCBpbnQgY2hhbm5lbHNPcmRlcltdKSB7Ci0gICAgICAgIHNob3J0IGRhdGFbXVtdID0gKChTaG9ydExvb2t1cFRhYmxlKWx1dCkuZ2V0VGFibGUoKTsKLSAgICAgICAgaW50IG5MZXZlbHMgPSBkYXRhWzBdLmxlbmd0aDsKLSAgICAgICAgaW50IG9mZnNldCA9IGx1dC5nZXRPZmZzZXQoKTsKLQotICAgICAgICAvLyBVc2Ugb25lIGRhdGEgYXJyYXkgZm9yIGFsbCBjaGFubmVscyBvciB1c2Ugc2V2ZXJhbCBkYXRhIGFycmF5cwotICAgICAgICBpbnQgZGF0YUluY3JlbWVudCA9IGRhdGEubGVuZ3RoID4gMSA/IDEgOiAwOwotCi0gICAgICAgIGZvciAoaW50IGNoID0gMCwgZGF0YUlkeCA9IDA7IGNoIDwgY2hhbm5lbHM7IGRhdGFJZHggKz0gZGF0YUluY3JlbWVudCwgY2grKykgewotICAgICAgICAgICAgaW50IGNoYW5uZWxPZmZzZXQgPSBjaGFubmVsc09yZGVyID09IG51bGwgPyBjaCA6IGNoYW5uZWxzT3JkZXJbY2hdOwotCi0gICAgICAgICAgICAvLyBTa2lwIGxhc3QgY2hhbm5lbCBpZiBuZWVkZWQsIHplcm8gdmFsdWVzIGFyZSBPSyAtCi0gICAgICAgICAgICAvLyBubyBjaGFuZ2VzIHRvIHRoZSBjaGFubmVsIGluZm9ybWF0aW9uIHdpbGwgYmUgZG9uZSBpbiBJUFAKLSAgICAgICAgICAgIGlmICgoY2hhbm5lbE9mZnNldCA9PSBjaGFubmVscyAtIDEgJiYgc2tpcEFscGhhKSB8fCAoZGF0YUlkeCA+PSBkYXRhLmxlbmd0aCkpIHsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaW50IGNoYW5uZWxCYXNlID0gbkxldmVscyAqIGNoYW5uZWxPZmZzZXQ7Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGxldmVsSW5pdGlhbGl6ZXIsIG9mZnNldCwgbGV2ZWxzLCBjaGFubmVsQmFzZSwgbkxldmVscyk7Ci0gICAgICAgICAgICBmb3IgKGludCBmcm9tID0gMCwgdG8gPSBjaGFubmVsQmFzZTsgZnJvbSA8IG5MZXZlbHM7IGZyb20rKywgdG8rKykgewotICAgICAgICAgICAgICAgIHZhbHVlc1t0b10gPSBkYXRhW2RhdGFJZHhdW2Zyb21dICYgMHhGRkZGOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gVE9ETyByZW1vdmUgd2hlbiB0aGlzIG1ldGhvZCBpcyB1c2VkCi0gICAgLyoqCi0gICAgICogSXBwIGZpbHRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgc3JjLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3QuCi0gICAgICogQHBhcmFtIGltYWdlVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHR5cGUuCi0gICAgICogQHBhcmFtIHNraXBBbHBoYQotICAgICAqICAgICAgICAgICAgdGhlIHNraXAgYWxwaGEuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bnVzZWQiKQotICAgIHByaXZhdGUgZmluYWwgaW50IGlwcEZpbHRlcihSYXN0ZXIgc3JjLCBXcml0YWJsZVJhc3RlciBkc3QsIGludCBpbWFnZVR5cGUsIGJvb2xlYW4gc2tpcEFscGhhKSB7Ci0gICAgICAgIGludCByZXM7Ci0KLSAgICAgICAgaW50IHNyY1N0cmlkZSwgZHN0U3RyaWRlOwotICAgICAgICBpbnQgY2hhbm5lbHM7Ci0gICAgICAgIGludCBvZmZzZXRzW10gPSBudWxsOwotICAgICAgICBpbnQgY2hhbm5lbHNPcmRlcltdID0gbnVsbDsKLQotICAgICAgICBzd2l0Y2ggKGltYWdlVHlwZSkgewotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQl9QUkU6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfUkdCOiB7Ci0gICAgICAgICAgICAgICAgY2hhbm5lbHMgPSA0OwotICAgICAgICAgICAgICAgIHNyY1N0cmlkZSA9IHNyYy5nZXRXaWR0aCgpICogNDsKLSAgICAgICAgICAgICAgICBkc3RTdHJpZGUgPSBkc3QuZ2V0V2lkdGgoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgY2hhbm5lbHNPcmRlciA9IG5ldyBpbnRbXSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAyLCAxLCAwLCAzCi0gICAgICAgICAgICAgICAgfTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfNEJZVEVfQUJHUjoKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFXzRCWVRFX0FCR1JfUFJFOgotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0JHUjogewotICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gNDsKLSAgICAgICAgICAgICAgICBzcmNTdHJpZGUgPSBzcmMuZ2V0V2lkdGgoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCkgKiA0OwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0dSQVk6IHsKLSAgICAgICAgICAgICAgICBjaGFubmVscyA9IDE7Ci0gICAgICAgICAgICAgICAgc3JjU3RyaWRlID0gc3JjLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFXzNCWVRFX0JHUjogewotICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gMzsKLSAgICAgICAgICAgICAgICBzcmNTdHJpZGUgPSBzcmMuZ2V0V2lkdGgoKSAqIDM7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCkgKiAzOwotICAgICAgICAgICAgICAgIGNoYW5uZWxzT3JkZXIgPSBuZXcgaW50W10gewotICAgICAgICAgICAgICAgICAgICAgICAgMiwgMSwgMAotICAgICAgICAgICAgICAgIH07Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX1VTSE9SVF9HUkFZOgotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfVVNIT1JUXzU2NV9SR0I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9VU0hPUlRfNTU1X1JHQjoKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX0JZVEVfQklOQVJZOiB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QsIHNraXBBbHBoYSk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGRlZmF1bHQ6IHsKLSAgICAgICAgICAgICAgICBTYW1wbGVNb2RlbCBzcmNTTSA9IHNyYy5nZXRTYW1wbGVNb2RlbCgpOwotICAgICAgICAgICAgICAgIFNhbXBsZU1vZGVsIGRzdFNNID0gZHN0LmdldFNhbXBsZU1vZGVsKCk7Ci0KLSAgICAgICAgICAgICAgICBpZiAoc3JjU00gaW5zdGFuY2VvZiBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIGRzdFNNIGluc3RhbmNlb2YgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIENoZWNrIFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbAotICAgICAgICAgICAgICAgICAgICBpZiAoc3JjU00uZ2V0RGF0YVR5cGUoKSAhPSBEYXRhQnVmZmVyLlRZUEVfQllURQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8IGRzdFNNLmdldERhdGFUeXBlKCkgIT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0LCBza2lwQWxwaGEpOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgLy8gSGF2ZSBJUFAgZnVuY3Rpb25zIGZvciAxLCAzIGFuZCA0IGNoYW5uZWxzCi0gICAgICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gc3JjU00uZ2V0TnVtQmFuZHMoKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKCEoY2hhbm5lbHMgPT0gMSB8fCBjaGFubmVscyA9PSAzIHx8IGNoYW5uZWxzID09IDQpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCwgc2tpcEFscGhhKTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIHNyY1N0cmlkZSA9ICgoQ29tcG9uZW50U2FtcGxlTW9kZWwpc3JjU00pLmdldFNjYW5saW5lU3RyaWRlKCk7Ci0gICAgICAgICAgICAgICAgICAgIGRzdFN0cmlkZSA9ICgoQ29tcG9uZW50U2FtcGxlTW9kZWwpZHN0U00pLmdldFNjYW5saW5lU3RyaWRlKCk7Ci0KLSAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHNPcmRlciA9ICgoQ29tcG9uZW50U2FtcGxlTW9kZWwpc3JjU00pLmdldEJhbmRPZmZzZXRzKCk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChzcmNTTSBpbnN0YW5jZW9mIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIGRzdFNNIGluc3RhbmNlb2YgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkgewotICAgICAgICAgICAgICAgICAgICAvLyBDaGVjayBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsCi0gICAgICAgICAgICAgICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc3Bwc20xID0gKFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpc3JjU007Ci0gICAgICAgICAgICAgICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc3Bwc20yID0gKFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpZHN0U007Ci0KLSAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHMgPSBzcHBzbTEuZ2V0TnVtQmFuZHMoKTsKLQotICAgICAgICAgICAgICAgICAgICAvLyBUWVBFX0lOVF9SR0IsIFRZUEVfSU5UX0FSR0IuLi4KLSAgICAgICAgICAgICAgICAgICAgaWYgKHNwcHNtMS5nZXREYXRhVHlwZSgpICE9IERhdGFCdWZmZXIuVFlQRV9JTlQKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCBzcHBzbTIuZ2V0RGF0YVR5cGUoKSAhPSBEYXRhQnVmZmVyLlRZUEVfSU5UCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgIShjaGFubmVscyA9PSAzIHx8IGNoYW5uZWxzID09IDQpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCwgc2tpcEFscGhhKTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIC8vIENoZWNrIGNvbXBhdGliaWxpdHkgb2Ygc2FtcGxlIG1vZGVscwotICAgICAgICAgICAgICAgICAgICBpZiAoIUFycmF5cy5lcXVhbHMoc3Bwc20xLmdldEJpdE9mZnNldHMoKSwgc3Bwc20yLmdldEJpdE9mZnNldHMoKSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCAhQXJyYXlzLmVxdWFscyhzcHBzbTEuZ2V0Qml0TWFza3MoKSwgc3Bwc20yLmdldEJpdE1hc2tzKCkpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCwgc2tpcEFscGhhKTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHNwcHNtMS5nZXRTYW1wbGVTaXplKGkpICE9IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCwgc2tpcEFscGhhKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGNoYW5uZWxzT3JkZXIgPSBuZXcgaW50W2NoYW5uZWxzXTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGJpdE9mZnNldHNbXSA9IHNwcHNtMS5nZXRCaXRPZmZzZXRzKCk7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHNPcmRlcltpXSA9IGJpdE9mZnNldHNbaV0gLyA4OwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgaWYgKGNoYW5uZWxzID09IDMpIHsgLy8gRG9uJ3Qgc2tpcCBjaGFubmVsIG5vdywgY291bGQgYmUKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIG9wdGltaXplZAotICAgICAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHMgPSA0OwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgc3JjU3RyaWRlID0gc3Bwc20xLmdldFNjYW5saW5lU3RyaWRlKCkgKiA0OwotICAgICAgICAgICAgICAgICAgICBkc3RTdHJpZGUgPSBzcHBzbTIuZ2V0U2NhbmxpbmVTdHJpZGUoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QsIHNraXBBbHBoYSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgLy8gRmlsbCBvZmZzZXRzIGlmIHRoZXJlJ3MgYSBjaGlsZCByYXN0ZXIKLSAgICAgICAgICAgICAgICBpZiAoc3JjLmdldFBhcmVudCgpICE9IG51bGwgfHwgZHN0LmdldFBhcmVudCgpICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSAhPSAwIHx8IHNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSAhPSAwCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgZHN0LmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWCgpICE9IDAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCBkc3QuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVZKCkgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0cyA9IG5ldyBpbnRbNF07Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXRzWzBdID0gLXNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSArIHNyYy5nZXRNaW5YKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXRzWzFdID0gLXNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSArIHNyYy5nZXRNaW5ZKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXRzWzJdID0gLWRzdC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSArIGRzdC5nZXRNaW5YKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXRzWzNdID0gLWRzdC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSArIGRzdC5nZXRNaW5ZKCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgbGV2ZWxzW10gPSBudWxsLCB2YWx1ZXNbXSA9IG51bGw7Ci0gICAgICAgIGludCBjaGFubmVsTXVsdGlwbGllciA9IHNraXBBbHBoYSA/IC0xIDogMTsKLSAgICAgICAgaWYgKGNoYW5uZWxNdWx0aXBsaWVyICogY2hhbm5lbHMgPT0gdmFsaWRGb3JDaGFubmVscykgeyAvLyB1c2UgZXhpc3RpbmcKLSAgICAgICAgICAgIC8vIGxldmVscy92YWx1ZXMKLSAgICAgICAgICAgIGxldmVscyA9IGNhY2hlZExldmVsczsKLSAgICAgICAgICAgIHZhbHVlcyA9IGNhY2hlZFZhbHVlczsKLSAgICAgICAgfSBlbHNlIHsgLy8gY3JlYXRlIG5ldyBsZXZlbHMvdmFsdWVzCi0gICAgICAgICAgICBpZiAobHV0IGluc3RhbmNlb2YgQnl0ZUxvb2t1cFRhYmxlKSB7Ci0gICAgICAgICAgICAgICAgYnl0ZSBkYXRhW11bXSA9ICgoQnl0ZUxvb2t1cFRhYmxlKWx1dCkuZ2V0VGFibGUoKTsKLSAgICAgICAgICAgICAgICBsZXZlbHMgPSBuZXcgaW50W2NoYW5uZWxzICogZGF0YVswXS5sZW5ndGhdOwotICAgICAgICAgICAgICAgIHZhbHVlcyA9IG5ldyBpbnRbY2hhbm5lbHMgKiBkYXRhWzBdLmxlbmd0aF07Ci0gICAgICAgICAgICAgICAgY3JlYXRlQnl0ZUxldmVscyhjaGFubmVscywgc2tpcEFscGhhLCBsZXZlbHMsIHZhbHVlcywgY2hhbm5lbHNPcmRlcik7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGx1dCBpbnN0YW5jZW9mIFNob3J0TG9va3VwVGFibGUpIHsKLSAgICAgICAgICAgICAgICBzaG9ydCBkYXRhW11bXSA9ICgoU2hvcnRMb29rdXBUYWJsZSlsdXQpLmdldFRhYmxlKCk7Ci0gICAgICAgICAgICAgICAgbGV2ZWxzID0gbmV3IGludFtjaGFubmVscyAqIGRhdGFbMF0ubGVuZ3RoXTsKLSAgICAgICAgICAgICAgICB2YWx1ZXMgPSBuZXcgaW50W2NoYW5uZWxzICogZGF0YVswXS5sZW5ndGhdOwotICAgICAgICAgICAgICAgIGNyZWF0ZVNob3J0TGV2ZWxzKGNoYW5uZWxzLCBza2lwQWxwaGEsIGxldmVscywgdmFsdWVzLCBjaGFubmVsc09yZGVyKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gY2FjaGUgbGV2ZWxzL3ZhbHVlcwotICAgICAgICAgICAgdmFsaWRGb3JDaGFubmVscyA9IGNoYW5uZWxNdWx0aXBsaWVyICogY2hhbm5lbHM7Ci0gICAgICAgICAgICBjYWNoZWRMZXZlbHMgPSBsZXZlbHM7Ci0gICAgICAgICAgICBjYWNoZWRWYWx1ZXMgPSB2YWx1ZXM7Ci0gICAgICAgIH0KLQotICAgICAgICBPYmplY3Qgc3JjRGF0YSwgZHN0RGF0YTsKLSAgICAgICAgQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yIGRiQWNjZXNzID0gQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yLmdldEluc3RhbmNlKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBzcmNEYXRhID0gZGJBY2Nlc3MuZ2V0RGF0YShzcmMuZ2V0RGF0YUJ1ZmZlcigpKTsKLSAgICAgICAgICAgIGRzdERhdGEgPSBkYkFjY2Vzcy5nZXREYXRhKGRzdC5nZXREYXRhQnVmZmVyKCkpOwotICAgICAgICB9IGNhdGNoIChJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgcmV0dXJuIC0xOyAvLyBVbmtub3duIGRhdGEgYnVmZmVyIHR5cGUKLSAgICAgICAgfQotCi0gICAgICAgIHJlcyA9IGlwcExVVChzcmNEYXRhLCBzcmMuZ2V0V2lkdGgoKSwgc3JjLmdldEhlaWdodCgpLCBzcmNTdHJpZGUsIGRzdERhdGEsIGRzdC5nZXRXaWR0aCgpLAotICAgICAgICAgICAgICAgIGRzdC5nZXRIZWlnaHQoKSwgZHN0U3RyaWRlLCBsZXZlbHMsIHZhbHVlcywgY2hhbm5lbHMsIG9mZnNldHMsIGZhbHNlKTsKLQotICAgICAgICByZXR1cm4gcmVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIElwcCBsdXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIHNyYy4KLSAgICAgKiBAcGFyYW0gc3JjV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcmMgd2lkdGguCi0gICAgICogQHBhcmFtIHNyY0hlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIHNyYyBoZWlnaHQuCi0gICAgICogQHBhcmFtIHNyY1N0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIHNyYyBzdHJpZGUuCi0gICAgICogQHBhcmFtIGRzdAotICAgICAqICAgICAgICAgICAgdGhlIGRzdC4KLSAgICAgKiBAcGFyYW0gZHN0V2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3Qgd2lkdGguCi0gICAgICogQHBhcmFtIGRzdEhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGRzdCBoZWlnaHQuCi0gICAgICogQHBhcmFtIGRzdFN0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIGRzdCBzdHJpZGUuCi0gICAgICogQHBhcmFtIGxldmVscwotICAgICAqICAgICAgICAgICAgdGhlIGxldmVscy4KLSAgICAgKiBAcGFyYW0gdmFsdWVzCi0gICAgICogICAgICAgICAgICB0aGUgdmFsdWVzLgotICAgICAqIEBwYXJhbSBjaGFubmVscwotICAgICAqICAgICAgICAgICAgdGhlIGNoYW5uZWxzLgotICAgICAqIEBwYXJhbSBvZmZzZXRzCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0cy4KLSAgICAgKiBAcGFyYW0gbGluZWFyCi0gICAgICogICAgICAgICAgICB0aGUgbGluZWFyLgotICAgICAqIEByZXR1cm4gdGhlIGludC4KLSAgICAgKi8KLSAgICBmaW5hbCBzdGF0aWMgbmF0aXZlIGludCBpcHBMVVQoT2JqZWN0IHNyYywgaW50IHNyY1dpZHRoLCBpbnQgc3JjSGVpZ2h0LCBpbnQgc3JjU3RyaWRlLAotICAgICAgICAgICAgT2JqZWN0IGRzdCwgaW50IGRzdFdpZHRoLCBpbnQgZHN0SGVpZ2h0LCBpbnQgZHN0U3RyaWRlLCBpbnQgbGV2ZWxzW10sIGludCB2YWx1ZXNbXSwKLSAgICAgICAgICAgIGludCBjaGFubmVscywgaW50IG9mZnNldHNbXSwgYm9vbGVhbiBsaW5lYXIpOwotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL0xvb2t1cFRhYmxlLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvTG9va3VwVGFibGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZTQ2NWE1NC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvTG9va3VwVGFibGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEwMSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqCi0gKiBAZGF0ZTogT2N0IDE0LCAyMDA1Ci0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoaXMgYWJzdHJhY3QgTG9va3VwVGFibGUgY2xhc3MgcmVwcmVzZW50cyBsb29rdXAgdGFibGUgd2hpY2ggaXMgZGVmaW5lZCB3aXRoCi0gKiB0aGUgbnVtYmVyIG9mIGNvbXBvbmVudHMgYW5kIG9mZnNldCB2YWx1ZS4gQnl0ZUxvb2t1cFRhYmxlIGFuZAotICogU2hvcnRMb29rdXBUYWJsZSBjbGFzc2VzIGFyZSBzdWJjbGFzc2VzIG9mIExvb2t1cFRhYmxlIHdoaWNoIGNvbnRhaW5zIGJ5dGUKLSAqIGFuZCBzaG9ydCBkYXRhIHRhYmxlcyBhcyBhbiBpbnB1dCBhcnJheXMgZm9yIGJhbmRzIG9yIGNvbXBvbmVudHMgb2YgaW1hZ2UuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgTG9va3VwVGFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIG9mZnNldC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBvZmZzZXQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbnVtIGNvbXBvbmVudHMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgbnVtQ29tcG9uZW50czsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBMb29rdXBUYWJsZSB3aXRoIHRoZSBzcGVjaWZpZWQgb2Zmc2V0IHZhbHVlIGFuZCBudW1iZXIKLSAgICAgKiBvZiBjb21wb25lbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgdmFsdWUuCi0gICAgICogQHBhcmFtIG51bUNvbXBvbmVudHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgY29tcG9uZW50cy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgTG9va3VwVGFibGUoaW50IG9mZnNldCwgaW50IG51bUNvbXBvbmVudHMpIHsKLSAgICAgICAgaWYgKG9mZnNldCA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMzI9T2Zmc2V0IHNob3VsZCBiZSBub3QgbGVzcyB0aGFuIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjMyIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKG51bUNvbXBvbmVudHMgPCAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjMzPU51bWJlciBvZiBjb21wb25lbnRzIHNob3VsZCBiZSBwb3NpdGl2ZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMzMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMub2Zmc2V0ID0gb2Zmc2V0OwotICAgICAgICB0aGlzLm51bUNvbXBvbmVudHMgPSBudW1Db21wb25lbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG9mZnNldCB2YWx1ZSBvZiB0aGlzIExvb2t1cCB0YWJsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBvZmZzZXQgdmFsdWUgb2YgdGhpcyBMb29rdXAgdGFibGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRPZmZzZXQoKSB7Ci0gICAgICAgIHJldHVybiBvZmZzZXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGNvbXBvbmVudHMgb2YgdGhpcyBMb29rdXAgdGFibGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIGNvbXBvbmVudHMgb2YgdGhpcyBMb29rdXAgdGFibGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXROdW1Db21wb25lbnRzKCkgewotICAgICAgICByZXR1cm4gbnVtQ29tcG9uZW50czsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGFuIGludGVnZXIgYXJyYXkgd2hpY2ggY29udGFpbnMgc2FtcGxlcyBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsIHdoaWNoCi0gICAgICogaXMgdHJhbnNsYXRlZCB3aXRoIHRoZSBsb29rdXAgdGFibGUgb2YgdGhpcyBMb29rdXBUYWJsZS4gVGhlIHJlc3VsdGVkCi0gICAgICogYXJyYXkgaXMgc3RvcmVkIHRvIHRoZSBkc3QgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNyYwotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBhcnJheS4KLSAgICAgKiBAcGFyYW0gZHN0Ci0gICAgICogICAgICAgICAgICB0aGUgZGVzdGluYXRpb24gYXJyYXkgd2hlcmUgdGhlIHJlc3VsdCBjYW4gYmUgc3RvcmVkLgotICAgICAqIEByZXR1cm4gdGhlIGludGVnZXIgYXJyYXkgb2YgdHJhbnNsYXRlZCBzYW1wbGVzIG9mIGEgcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludFtdIGxvb2t1cFBpeGVsKGludFtdIHNyYywgaW50W10gZHN0KTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9NZW1vcnlJbWFnZVNvdXJjZS5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL01lbW9yeUltYWdlU291cmNlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDY0NGZkNDAuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL01lbW9yeUltYWdlU291cmNlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2MDMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLnV0aWwuSGFzaHRhYmxlOwotaW1wb3J0IGphdmEudXRpbC5WZWN0b3I7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgTWVtb3J5SW1hZ2VTb3VyY2UgY2xhc3MgaXMgdXNlZCB0byBwcm9kdWNlcyBwaXhlbHMgb2YgYW4gaW1hZ2UgZnJvbSBhbgotICogYXJyYXkuIFRoaXMgY2xhc3MgY2FuIG1hbmFnZSBhIG1lbW9yeSBpbWFnZSB3aGljaCBjb250YWlucyBhbiBhbmltYXRpb24gb3IKLSAqIGN1c3RvbSByZW5kZXJpbmcuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgTWVtb3J5SW1hZ2VTb3VyY2UgaW1wbGVtZW50cyBJbWFnZVByb2R1Y2VyIHsKLQotICAgIC8qKgotICAgICAqIFRoZSB3aWR0aC4KLSAgICAgKi8KLSAgICBpbnQgd2lkdGg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGVpZ2h0LgotICAgICAqLwotICAgIGludCBoZWlnaHQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY20uCi0gICAgICovCi0gICAgQ29sb3JNb2RlbCBjbTsKLQotICAgIC8qKgotICAgICAqIFRoZSBiIGRhdGEuCi0gICAgICovCi0gICAgYnl0ZSBiRGF0YVtdOwotCi0gICAgLyoqCi0gICAgICogVGhlIGkgZGF0YS4KLSAgICAgKi8KLSAgICBpbnQgaURhdGFbXTsKLQotICAgIC8qKgotICAgICAqIFRoZSBvZmZzZXQuCi0gICAgICovCi0gICAgaW50IG9mZnNldDsKLQotICAgIC8qKgotICAgICAqIFRoZSBzY2FubGluZS4KLSAgICAgKi8KLSAgICBpbnQgc2NhbmxpbmU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcHJvcGVydGllcy4KLSAgICAgKi8KLSAgICBIYXNodGFibGU8PywgPz4gcHJvcGVydGllczsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb25zdW1lcnMuCi0gICAgICovCi0gICAgVmVjdG9yPEltYWdlQ29uc3VtZXI+IGNvbnN1bWVyczsKLQotICAgIC8qKgotICAgICAqIFRoZSBhbmltYXRlZC4KLSAgICAgKi8KLSAgICBib29sZWFuIGFuaW1hdGVkOwotCi0gICAgLyoqCi0gICAgICogVGhlIGZ1bGxidWZmZXJzLgotICAgICAqLwotICAgIGJvb2xlYW4gZnVsbGJ1ZmZlcnM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGF0YSB0eXBlLgotICAgICAqLwotICAgIGludCBkYXRhVHlwZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBEQVRBX1RZUEVfQllURS4KLSAgICAgKi8KLSAgICBzdGF0aWMgZmluYWwgaW50IERBVEFfVFlQRV9CWVRFID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBEQVRBX1RZUEVfSU5ULgotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBpbnQgREFUQV9UWVBFX0lOVCA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgTWVtb3J5SW1hZ2VTb3VyY2Ugd2l0aCB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBjbQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBDb2xvck1vZGVsLgotICAgICAqIEBwYXJhbSBwaXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbCBhcnJheS4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBwaXhlbCBhcnJheS4KLSAgICAgKiBAcGFyYW0gc2NhbgotICAgICAqICAgICAgICAgICAgdGhlIGRpc3RhbmNlIGZyb20gb25lIHBpeGVsJ3Mgcm93IHRvIHRoZSBuZXh0IGluIHRoZSBwaXhlbAotICAgICAqICAgICAgICAgICAgYXJyYXkuCi0gICAgICogQHBhcmFtIHByb3BzCi0gICAgICogICAgICAgICAgICB0aGUgc2V0IG9mIHByb3BlcnRpZXMgdG8gYmUgdXNlZCBmb3IgaW1hZ2UgcHJvY2Vzc2luZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgTWVtb3J5SW1hZ2VTb3VyY2UoaW50IHcsIGludCBoLCBDb2xvck1vZGVsIGNtLCBpbnQgcGl4W10sIGludCBvZmYsIGludCBzY2FuLAotICAgICAgICAgICAgSGFzaHRhYmxlPD8sID8+IHByb3BzKSB7Ci0gICAgICAgIGluaXQodywgaCwgY20sIHBpeCwgb2ZmLCBzY2FuLCBwcm9wcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IE1lbW9yeUltYWdlU291cmNlIHdpdGggdGhlIHNwZWNpZmllZCBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gY20KLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgQ29sb3JNb2RlbC4KLSAgICAgKiBAcGFyYW0gcGl4Ci0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwgYXJyYXkuCi0gICAgICogQHBhcmFtIG9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgcGl4ZWwgYXJyYXkuCi0gICAgICogQHBhcmFtIHNjYW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSBmcm9tIG9uZSBwaXhlbCdzIHJvdyB0byB0aGUgbmV4dCBpbiB0aGUgcGl4ZWwKLSAgICAgKiAgICAgICAgICAgIGFycmF5LgotICAgICAqIEBwYXJhbSBwcm9wcwotICAgICAqICAgICAgICAgICAgdGhlIHNldCBvZiBwcm9wZXJ0aWVzIHRvIGJlIHVzZWQgZm9yIGltYWdlIHByb2Nlc3NpbmcuCi0gICAgICovCi0gICAgcHVibGljIE1lbW9yeUltYWdlU291cmNlKGludCB3LCBpbnQgaCwgQ29sb3JNb2RlbCBjbSwgYnl0ZSBwaXhbXSwgaW50IG9mZiwgaW50IHNjYW4sCi0gICAgICAgICAgICBIYXNodGFibGU8PywgPz4gcHJvcHMpIHsKLSAgICAgICAgaW5pdCh3LCBoLCBjbSwgcGl4LCBvZmYsIHNjYW4sIHByb3BzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgTWVtb3J5SW1hZ2VTb3VyY2Ugd2l0aCB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMgYW5kCi0gICAgICogZGVmYXVsdCBSR0IgQ29sb3JNb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHBpeAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsIGFycmF5LgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIHBpeGVsIGFycmF5LgotICAgICAqIEBwYXJhbSBzY2FuCi0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgZnJvbSBvbmUgcGl4ZWwncyByb3cgdG8gdGhlIG5leHQgaW4gdGhlIHBpeGVsCi0gICAgICogICAgICAgICAgICBhcnJheS4KLSAgICAgKiBAcGFyYW0gcHJvcHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzZXQgb2YgcHJvcGVydGllcyB0byBiZSB1c2VkIGZvciBpbWFnZSBwcm9jZXNzaW5nLgotICAgICAqLwotICAgIHB1YmxpYyBNZW1vcnlJbWFnZVNvdXJjZShpbnQgdywgaW50IGgsIGludCBwaXhbXSwgaW50IG9mZiwgaW50IHNjYW4sIEhhc2h0YWJsZTw/LCA/PiBwcm9wcykgewotICAgICAgICBpbml0KHcsIGgsIENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpLCBwaXgsIG9mZiwgc2NhbiwgcHJvcHMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBNZW1vcnlJbWFnZVNvdXJjZSB3aXRoIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGNtCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIENvbG9yTW9kZWwuCi0gICAgICogQHBhcmFtIHBpeAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsIGFycmF5LgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIHBpeGVsIGFycmF5LgotICAgICAqIEBwYXJhbSBzY2FuCi0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgZnJvbSBvbmUgcGl4ZWwncyByb3cgdG8gdGhlIG5leHQgaW4gdGhlIHBpeGVsCi0gICAgICogICAgICAgICAgICBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgTWVtb3J5SW1hZ2VTb3VyY2UoaW50IHcsIGludCBoLCBDb2xvck1vZGVsIGNtLCBpbnQgcGl4W10sIGludCBvZmYsIGludCBzY2FuKSB7Ci0gICAgICAgIGluaXQodywgaCwgY20sIHBpeCwgb2ZmLCBzY2FuLCBudWxsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgTWVtb3J5SW1hZ2VTb3VyY2Ugd2l0aCB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBjbQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBDb2xvck1vZGVsLgotICAgICAqIEBwYXJhbSBwaXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbCBhcnJheS4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBwaXhlbCBhcnJheS4KLSAgICAgKiBAcGFyYW0gc2NhbgotICAgICAqICAgICAgICAgICAgdGhlIGRpc3RhbmNlIGZyb20gb25lIHBpeGVsJ3Mgcm93IHRvIHRoZSBuZXh0IGluIHRoZSBwaXhlbAotICAgICAqICAgICAgICAgICAgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIE1lbW9yeUltYWdlU291cmNlKGludCB3LCBpbnQgaCwgQ29sb3JNb2RlbCBjbSwgYnl0ZSBwaXhbXSwgaW50IG9mZiwgaW50IHNjYW4pIHsKLSAgICAgICAgaW5pdCh3LCBoLCBjbSwgcGl4LCBvZmYsIHNjYW4sIG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBNZW1vcnlJbWFnZVNvdXJjZSB3aXRoIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycyBhbmQKLSAgICAgKiBkZWZhdWx0IFJHQiBDb2xvck1vZGVsLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gcGl4Ci0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWxzIGFycmF5LgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIHBpeGVsIGFycmF5LgotICAgICAqIEBwYXJhbSBzY2FuCi0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgZnJvbSBvbmUgcGl4ZWwncyByb3cgdG8gdGhlIG5leHQgaW4gdGhlIHBpeGVsCi0gICAgICogICAgICAgICAgICBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgTWVtb3J5SW1hZ2VTb3VyY2UoaW50IHcsIGludCBoLCBpbnQgcGl4W10sIGludCBvZmYsIGludCBzY2FuKSB7Ci0gICAgICAgIGluaXQodywgaCwgQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCksIHBpeCwgb2ZmLCBzY2FuLCBudWxsKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIGJvb2xlYW4gaXNDb25zdW1lcihJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIHJldHVybiBjb25zdW1lcnMuY29udGFpbnMoaWMpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHN0YXJ0UHJvZHVjdGlvbihJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIGlmICghaXNDb25zdW1lcihpYykgJiYgaWMgIT0gbnVsbCkgewotICAgICAgICAgICAgY29uc3VtZXJzLmFkZEVsZW1lbnQoaWMpOwotICAgICAgICB9Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBzZXRIZWFkZXIoaWMpOwotICAgICAgICAgICAgc2V0UGl4ZWxzKGljLCAwLCAwLCB3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgICAgIGlmIChhbmltYXRlZCkgewotICAgICAgICAgICAgICAgIGljLmltYWdlQ29tcGxldGUoSW1hZ2VDb25zdW1lci5TSU5HTEVGUkFNRURPTkUpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBpYy5pbWFnZUNvbXBsZXRlKEltYWdlQ29uc3VtZXIuU1RBVElDSU1BR0VET05FKTsKLSAgICAgICAgICAgICAgICBpZiAoaXNDb25zdW1lcihpYykpIHsKLSAgICAgICAgICAgICAgICAgICAgcmVtb3ZlQ29uc3VtZXIoaWMpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIGlmIChpc0NvbnN1bWVyKGljKSkgewotICAgICAgICAgICAgICAgIGljLmltYWdlQ29tcGxldGUoSW1hZ2VDb25zdW1lci5JTUFHRUVSUk9SKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChpc0NvbnN1bWVyKGljKSkgewotICAgICAgICAgICAgICAgIHJlbW92ZUNvbnN1bWVyKGljKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlcXVlc3RUb3BEb3duTGVmdFJpZ2h0UmVzZW5kKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgcmVtb3ZlQ29uc3VtZXIoSW1hZ2VDb25zdW1lciBpYykgewotICAgICAgICBjb25zdW1lcnMucmVtb3ZlRWxlbWVudChpYyk7Ci0gICAgfQotCi0gICAgcHVibGljIHN5bmNocm9uaXplZCB2b2lkIGFkZENvbnN1bWVyKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgaWYgKGljID09IG51bGwgfHwgY29uc3VtZXJzLmNvbnRhaW5zKGljKSkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIGNvbnN1bWVycy5hZGRFbGVtZW50KGljKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXBsYWNlcyB0aGUgcGl4ZWwgZGF0YSB3aXRoIGEgbmV3IHBpeGVsIGFycmF5IGZvciBob2xkaW5nIHRoZSBwaXhlbHMgZm9yCi0gICAgICogdGhpcyBpbWFnZS4gSWYgYW4gYW5pbWF0aW9uIGZsYWcgaXMgc2V0IHRvIHRydWUgdmFsdWUgYnkgdGhlCi0gICAgICogc2V0QW5pbWF0ZWQoKSBtZXRob2QsIHRoZSBuZXcgcGl4ZWxzIHdpbGwgYmUgaW1tZWRpYXRlbHkgZGVsaXZlcmVkIHRvIHRoZQotICAgICAqIEltYWdlQ29uc3VtZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuZXdwaXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgcGl4ZWwgYXJyYXkuCi0gICAgICogQHBhcmFtIG5ld21vZGVsCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IENvbG9yTW9kZWwuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCBpbiB0aGUgYXJyYXkuCi0gICAgICogQHBhcmFtIHNjYW5zaXplCi0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgZnJvbSBvbmUgcm93IG9mIHBpeGVscyB0byB0aGUgbmV4dCByb3cgaW4gdGhlCi0gICAgICogICAgICAgICAgICBwaXhlbCBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgbmV3UGl4ZWxzKGludCBuZXdwaXhbXSwgQ29sb3JNb2RlbCBuZXdtb2RlbCwgaW50IG9mZnNldCwgaW50IHNjYW5zaXplKSB7Ci0gICAgICAgIHRoaXMuZGF0YVR5cGUgPSBEQVRBX1RZUEVfSU5UOwotICAgICAgICB0aGlzLmlEYXRhID0gbmV3cGl4OwotICAgICAgICB0aGlzLmNtID0gbmV3bW9kZWw7Ci0gICAgICAgIHRoaXMub2Zmc2V0ID0gb2Zmc2V0OwotICAgICAgICB0aGlzLnNjYW5saW5lID0gc2NhbnNpemU7Ci0gICAgICAgIG5ld1BpeGVscygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcGxhY2VzIHRoZSBwaXhlbCBkYXRhIHdpdGggYSBuZXcgcGl4ZWwgYXJyYXkgZm9yIGhvbGRpbmcgdGhlIHBpeGVscyBmb3IKLSAgICAgKiB0aGlzIGltYWdlLiBJZiBhbiBhbmltYXRpb24gZmxhZyBpcyBzZXQgdG8gdHJ1ZSB2YWx1ZSBieSB0aGUKLSAgICAgKiBzZXRBbmltYXRlZCgpIG1ldGhvZCwgdGhlIG5ldyBwaXhlbHMgd2lsbCBiZSBpbW1lZGlhdGVseSBkZWxpdmVyZWQgdG8gdGhlCi0gICAgICogSW1hZ2VDb25zdW1lcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIG5ld3BpeAotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBwaXhlbCBhcnJheS4KLSAgICAgKiBAcGFyYW0gbmV3bW9kZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgQ29sb3JNb2RlbC4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBhcnJheS4KLSAgICAgKiBAcGFyYW0gc2NhbnNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSBmcm9tIG9uZSByb3cgb2YgcGl4ZWxzIHRvIHRoZSBuZXh0IHJvdyBpbiB0aGUKLSAgICAgKiAgICAgICAgICAgIHBpeGVsIGFycmF5LgotICAgICAqLwotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCBuZXdQaXhlbHMoYnl0ZSBuZXdwaXhbXSwgQ29sb3JNb2RlbCBuZXdtb2RlbCwgaW50IG9mZnNldCwgaW50IHNjYW5zaXplKSB7Ci0gICAgICAgIHRoaXMuZGF0YVR5cGUgPSBEQVRBX1RZUEVfQllURTsKLSAgICAgICAgdGhpcy5iRGF0YSA9IG5ld3BpeDsKLSAgICAgICAgdGhpcy5jbSA9IG5ld21vZGVsOwotICAgICAgICB0aGlzLm9mZnNldCA9IG9mZnNldDsKLSAgICAgICAgdGhpcy5zY2FubGluZSA9IHNjYW5zaXplOwotICAgICAgICBuZXdQaXhlbHMoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBmdWxsIGJ1ZmZlciB1cGRhdGVzIGZsYWcgdG8gdHJ1ZS4gSWYgdGhpcyBpcyBhbiBhbmltYXRlZCBpbWFnZSwKLSAgICAgKiB0aGUgaW1hZ2UgY29uc3VtZXJzIGhpbnRzIGFyZSB1cGRhdGVkIGFjY29yZGluZ2x5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmdWxsYnVmZmVycwotICAgICAqICAgICAgICAgICAgdGhlIHRydWUgaWYgdGhlIHBpeGVsIGJ1ZmZlciBzaG91bGQgYmUgc2VudCBhbHdheXMuCi0gICAgICovCi0gICAgcHVibGljIHN5bmNocm9uaXplZCB2b2lkIHNldEZ1bGxCdWZmZXJVcGRhdGVzKGJvb2xlYW4gZnVsbGJ1ZmZlcnMpIHsKLSAgICAgICAgaWYgKHRoaXMuZnVsbGJ1ZmZlcnMgPT0gZnVsbGJ1ZmZlcnMpIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICB0aGlzLmZ1bGxidWZmZXJzID0gZnVsbGJ1ZmZlcnM7Ci0gICAgICAgIGlmIChhbmltYXRlZCkgewotICAgICAgICAgICAgT2JqZWN0IGNvbnNBcltdID0gY29uc3VtZXJzLnRvQXJyYXkoKTsKLSAgICAgICAgICAgIGZvciAoT2JqZWN0IGVsZW1lbnQgOiBjb25zQXIpIHsKLSAgICAgICAgICAgICAgICBJbWFnZUNvbnN1bWVyIGNvbiA9IChJbWFnZUNvbnN1bWVyKWVsZW1lbnQ7Ci0gICAgICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGZ1bGxidWZmZXJzKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb24uc2V0SGludHMoSW1hZ2VDb25zdW1lci5UT1BET1dOTEVGVFJJR0hUCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgSW1hZ2VDb25zdW1lci5DT01QTEVURVNDQU5MSU5FUyk7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb24uc2V0SGludHMoSW1hZ2VDb25zdW1lci5SQU5ET01QSVhFTE9SREVSKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChpc0NvbnN1bWVyKGNvbikpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbi5pbWFnZUNvbXBsZXRlKEltYWdlQ29uc3VtZXIuSU1BR0VFUlJPUik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWYgKGlzQ29uc3VtZXIoY29uKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmVtb3ZlQ29uc3VtZXIoY29uKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGZsYWcgdGhhdCB0ZWxscyB3aGV0aGVyIHRoaXMgbWVtb3J5IGltYWdlIGhhcyBtb3JlIHRoYW4gb25lCi0gICAgICogZnJhbWUgKGZvciBhbmltYXRpb24pOiB0cnVlIGZvciBtdWx0aXBsZSBmcmFtZXMsIGZhbHNlIGlmIHRoaXMgY2xhc3MKLSAgICAgKiByZXByZXNlbnRzIGEgc2luZ2xlIGZyYW1lIGltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhbmltYXRlZAotICAgICAqICAgICAgICAgICAgd2hldGhlciB0aGlzIGltYWdlIHJlcHJlc2VudHMgYW4gYW5pbWF0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCBzZXRBbmltYXRlZChib29sZWFuIGFuaW1hdGVkKSB7Ci0gICAgICAgIGlmICh0aGlzLmFuaW1hdGVkID09IGFuaW1hdGVkKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgT2JqZWN0IGNvbnNBcltdID0gY29uc3VtZXJzLnRvQXJyYXkoKTsKLSAgICAgICAgZm9yIChPYmplY3QgZWxlbWVudCA6IGNvbnNBcikgewotICAgICAgICAgICAgSW1hZ2VDb25zdW1lciBjb24gPSAoSW1hZ2VDb25zdW1lcillbGVtZW50OwotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICBjb24uaW1hZ2VDb21wbGV0ZShJbWFnZUNvbnN1bWVyLlNUQVRJQ0lNQUdFRE9ORSk7Ci0gICAgICAgICAgICB9IGNhdGNoIChFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIGlmIChpc0NvbnN1bWVyKGNvbikpIHsKLSAgICAgICAgICAgICAgICAgICAgY29uLmltYWdlQ29tcGxldGUoSW1hZ2VDb25zdW1lci5JTUFHRUVSUk9SKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoaXNDb25zdW1lcihjb24pKSB7Ci0gICAgICAgICAgICAgICAgcmVtb3ZlQ29uc3VtZXIoY29uKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICB0aGlzLmFuaW1hdGVkID0gYW5pbWF0ZWQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2VuZHMgdGhlIHNwZWNpZmllZCByZWN0YW5ndWxhciBhcmVhIG9mIHRoZSBidWZmZXIgdG8gSW1hZ2VDb25zdW1lcnMgYW5kCi0gICAgICogbm90aWZpZXMgdGhlbSB0aGF0IGFuIGFuaW1hdGlvbiBmcmFtZSBpcyBjb21wbGV0ZWQgb25seSBpZiB0aGUge0Bjb2RlCi0gICAgICogZnJhbWVub3RpZnl9IHBhcmFtZXRlciBpcyB0cnVlLiBUaGF0IHdvcmtzIG9ubHkgaWYgdGhlIGFuaW1hdGVkIGZsYWcgaGFzCi0gICAgICogYmVlbiBzZXQgdG8gdHJ1ZSBieSB0aGUgc2V0QW5pbWF0ZWQoKSBtZXRob2QuIElmIHRoZSBmdWxsIGJ1ZmZlciB1cGRhdGUKLSAgICAgKiBmbGFnIGhhcyBiZWVuIHNldCB0byB0cnVlIGJ5IHRoZSBzZXRGdWxsQnVmZmVyVXBkYXRlcygpIG1ldGhvZCwgdGhlbiB0aGUKLSAgICAgKiBlbnRpcmUgYnVmZmVyIHdpbGwgYWx3YXlzIGJlIHNlbnQgaWdub3JpbmcgcGFyYW1ldGVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5ndWxhciBhcmVhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5ndWxhciBhcmVhLgotICAgICAqIEBwYXJhbSBmcmFtZW5vdGlmeQotICAgICAqICAgICAgICAgICAgdHJ1ZSBpZiBhIFNJTkdMRUZSQU1FRE9ORSBub3RpZmljYXRpb24gc2hvdWxkIGJlIHNlbnQgdG8gdGhlCi0gICAgICogICAgICAgICAgICByZWdpc3RlcmVkIGNvbnN1bWVycywgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCBuZXdQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGJvb2xlYW4gZnJhbWVub3RpZnkpIHsKLSAgICAgICAgaWYgKGFuaW1hdGVkKSB7Ci0gICAgICAgICAgICBpZiAoZnVsbGJ1ZmZlcnMpIHsKLSAgICAgICAgICAgICAgICB4ID0gMDsKLSAgICAgICAgICAgICAgICB5ID0gMDsKLSAgICAgICAgICAgICAgICB3ID0gd2lkdGg7Ci0gICAgICAgICAgICAgICAgaCA9IGhlaWdodDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgaWYgKHggPCAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHcgKz0geDsKLSAgICAgICAgICAgICAgICAgICAgeCA9IDA7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmICh3ID4gd2lkdGgpIHsKLSAgICAgICAgICAgICAgICAgICAgdyA9IHdpZHRoIC0geDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKHkgPCAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGggKz0geTsKLSAgICAgICAgICAgICAgICAgICAgeSA9IDA7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGggPiBoZWlnaHQpIHsKLSAgICAgICAgICAgICAgICBoID0gaGVpZ2h0IC0geTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIE9iamVjdCBjb25zQXJbXSA9IGNvbnN1bWVycy50b0FycmF5KCk7Ci0gICAgICAgICAgICBmb3IgKE9iamVjdCBlbGVtZW50IDogY29uc0FyKSB7Ci0gICAgICAgICAgICAgICAgSW1hZ2VDb25zdW1lciBjb24gPSAoSW1hZ2VDb25zdW1lcillbGVtZW50OwotICAgICAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmICh3ID4gMCAmJiBoID4gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgc2V0UGl4ZWxzKGNvbiwgeCwgeSwgdywgaCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWYgKGZyYW1lbm90aWZ5KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb24uaW1hZ2VDb21wbGV0ZShJbWFnZUNvbnN1bWVyLlNJTkdMRUZSQU1FRE9ORSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9IGNhdGNoIChFeGNlcHRpb24gZXgpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGlzQ29uc3VtZXIoY29uKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgY29uLmltYWdlQ29tcGxldGUoSW1hZ2VDb25zdW1lci5JTUFHRUVSUk9SKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpZiAoaXNDb25zdW1lcihjb24pKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZW1vdmVDb25zdW1lcihjb24pOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2VuZHMgdGhlIHNwZWNpZmllZCByZWN0YW5ndWxhciBhcmVhIG9mIHRoZSBidWZmZXIgdG8gdGhlIEltYWdlQ29uc3VtZXJzCi0gICAgICogYW5kIG5vdGlmaWVzIHRoZW0gdGhhdCBhbiBhbmltYXRpb24gZnJhbWUgaXMgY29tcGxldGVkIGlmIHRoZSBhbmltYXRlZAotICAgICAqIGZsYWcgaGFzIGJlZW4gc2V0IHRvIHRydWUgYnkgdGhlIHNldEFuaW1hdGVkKCkgbWV0aG9kLiBJZiB0aGUgZnVsbCBidWZmZXIKLSAgICAgKiB1cGRhdGUgZmxhZyBoYXMgYmVlbiBzZXQgdG8gdHJ1ZSBieSB0aGUgc2V0RnVsbEJ1ZmZlclVwZGF0ZXMoKSBtZXRob2QsCi0gICAgICogdGhlbiB0aGUgZW50aXJlIGJ1ZmZlciB3aWxsIGFsd2F5cyBiZSBzZW50IGlnbm9yaW5nIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgbmV3UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoKSB7Ci0gICAgICAgIG5ld1BpeGVscyh4LCB5LCB3LCBoLCB0cnVlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZW5kcyBhIG5ldyBidWZmZXIgb2YgcGl4ZWxzIHRvIHRoZSBJbWFnZUNvbnN1bWVycyBhbmQgbm90aWZpZXMgdGhlbSB0aGF0Ci0gICAgICogYW4gYW5pbWF0aW9uIGZyYW1lIGlzIGNvbXBsZXRlZCBpZiB0aGUgYW5pbWF0ZWQgZmxhZyBoYXMgYmVlbiBzZXQgdG8gdHJ1ZQotICAgICAqIGJ5IHRoZSBzZXRBbmltYXRlZCgpIG1ldGhvZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBuZXdQaXhlbHMoKSB7Ci0gICAgICAgIG5ld1BpeGVscygwLCAwLCB3aWR0aCwgaGVpZ2h0LCB0cnVlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbml0cyB0aGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGguCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodC4KLSAgICAgKiBAcGFyYW0gbW9kZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBtb2RlbC4KLSAgICAgKiBAcGFyYW0gcGl4ZWxzCi0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmYuCi0gICAgICogQHBhcmFtIHNjYW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FuLgotICAgICAqIEBwYXJhbSBwcm9wCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgaW5pdChpbnQgd2lkdGgsIGludCBoZWlnaHQsIENvbG9yTW9kZWwgbW9kZWwsIGJ5dGUgcGl4ZWxzW10sIGludCBvZmYsIGludCBzY2FuLAotICAgICAgICAgICAgSGFzaHRhYmxlPD8sID8+IHByb3ApIHsKLQotICAgICAgICB0aGlzLndpZHRoID0gd2lkdGg7Ci0gICAgICAgIHRoaXMuaGVpZ2h0ID0gaGVpZ2h0OwotICAgICAgICB0aGlzLmNtID0gbW9kZWw7Ci0gICAgICAgIHRoaXMuYkRhdGEgPSBwaXhlbHM7Ci0gICAgICAgIHRoaXMub2Zmc2V0ID0gb2ZmOwotICAgICAgICB0aGlzLnNjYW5saW5lID0gc2NhbjsKLSAgICAgICAgdGhpcy5wcm9wZXJ0aWVzID0gcHJvcDsKLSAgICAgICAgdGhpcy5kYXRhVHlwZSA9IERBVEFfVFlQRV9CWVRFOwotICAgICAgICB0aGlzLmNvbnN1bWVycyA9IG5ldyBWZWN0b3I8SW1hZ2VDb25zdW1lcj4oKTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluaXRzIHRoZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aC4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0LgotICAgICAqIEBwYXJhbSBtb2RlbAotICAgICAqICAgICAgICAgICAgdGhlIG1vZGVsLgotICAgICAqIEBwYXJhbSBwaXhlbHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbHMuCi0gICAgICogQHBhcmFtIG9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZi4KLSAgICAgKiBAcGFyYW0gc2NhbgotICAgICAqICAgICAgICAgICAgdGhlIHNjYW4uCi0gICAgICogQHBhcmFtIHByb3AKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcm9wLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBpbml0KGludCB3aWR0aCwgaW50IGhlaWdodCwgQ29sb3JNb2RlbCBtb2RlbCwgaW50IHBpeGVsc1tdLCBpbnQgb2ZmLCBpbnQgc2NhbiwKLSAgICAgICAgICAgIEhhc2h0YWJsZTw/LCA/PiBwcm9wKSB7Ci0KLSAgICAgICAgdGhpcy53aWR0aCA9IHdpZHRoOwotICAgICAgICB0aGlzLmhlaWdodCA9IGhlaWdodDsKLSAgICAgICAgdGhpcy5jbSA9IG1vZGVsOwotICAgICAgICB0aGlzLmlEYXRhID0gcGl4ZWxzOwotICAgICAgICB0aGlzLm9mZnNldCA9IG9mZjsKLSAgICAgICAgdGhpcy5zY2FubGluZSA9IHNjYW47Ci0gICAgICAgIHRoaXMucHJvcGVydGllcyA9IHByb3A7Ci0gICAgICAgIHRoaXMuZGF0YVR5cGUgPSBEQVRBX1RZUEVfSU5UOwotICAgICAgICB0aGlzLmNvbnN1bWVycyA9IG5ldyBWZWN0b3I8SW1hZ2VDb25zdW1lcj4oKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBwaXhlbHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbgotICAgICAqICAgICAgICAgICAgdGhlIGNvbi4KLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHguCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSB5LgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgdy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGguCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHNldFBpeGVscyhJbWFnZUNvbnN1bWVyIGNvbiwgaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgpIHsKLSAgICAgICAgaW50IHBpeGVsT2ZmID0gc2NhbmxpbmUgKiB5ICsgb2Zmc2V0ICsgeDsKLQotICAgICAgICBzd2l0Y2ggKGRhdGFUeXBlKSB7Ci0gICAgICAgICAgICBjYXNlIERBVEFfVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGNvbi5zZXRQaXhlbHMoeCwgeSwgdywgaCwgY20sIGJEYXRhLCBwaXhlbE9mZiwgc2NhbmxpbmUpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEQVRBX1RZUEVfSU5UOgotICAgICAgICAgICAgICAgIGNvbi5zZXRQaXhlbHMoeCwgeSwgdywgaCwgY20sIGlEYXRhLCBwaXhlbE9mZiwgc2NhbmxpbmUpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjJBPVdyb25nIHR5cGUgb2YgcGl4ZWxzIGFycmF5Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMkEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGhlYWRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29uCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IGhlYWRlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN5bmNocm9uaXplZCB2b2lkIHNldEhlYWRlcihJbWFnZUNvbnN1bWVyIGNvbikgewotICAgICAgICBjb24uc2V0RGltZW5zaW9ucyh3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgY29uLnNldFByb3BlcnRpZXMocHJvcGVydGllcyk7Ci0gICAgICAgIGNvbi5zZXRDb2xvck1vZGVsKGNtKTsKLSAgICAgICAgY29uCi0gICAgICAgICAgICAgICAgLnNldEhpbnRzKGFuaW1hdGVkID8gKGZ1bGxidWZmZXJzID8gKEltYWdlQ29uc3VtZXIuVE9QRE9XTkxFRlRSSUdIVCB8IEltYWdlQ29uc3VtZXIuQ09NUExFVEVTQ0FOTElORVMpCi0gICAgICAgICAgICAgICAgICAgICAgICA6IEltYWdlQ29uc3VtZXIuUkFORE9NUElYRUxPUkRFUikKLSAgICAgICAgICAgICAgICAgICAgICAgIDogKEltYWdlQ29uc3VtZXIuVE9QRE9XTkxFRlRSSUdIVCB8IEltYWdlQ29uc3VtZXIuQ09NUExFVEVTQ0FOTElORVMKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCBJbWFnZUNvbnN1bWVyLlNJTkdMRVBBU1MgfCBJbWFnZUNvbnN1bWVyLlNJTkdMRUZSQU1FKSk7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDNkYzEzZDguLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL011bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDc5ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBjbGFzcyByZXByZXNlbnRzIGltYWdlIGRhdGEgd2l0aCBvbmUgYmFuZC4KLSAqIFRoaXMgY2xhc3MgcGFja3MgbXVsdGlwbGUgcGl4ZWxzIHdpdGggb25lIHNhbXBsZSBpbiBvbmUgZGF0YSBlbGVtZW50IGFuZAotICogc3VwcG9ydHMgdGhlIGZvbGxvd2luZyBkYXRhIHR5cGVzOiBEYXRhQnVmZmVyLlRZUEVfQllURSwKLSAqIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQsIG9yIERhdGFCdWZmZXIuVFlQRV9JTlQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIGV4dGVuZHMgU2FtcGxlTW9kZWwgewotCi0gICAgLyoqCi0gICAgICogVGhlIHBpeGVsIGJpdCBzdHJpZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgcGl4ZWxCaXRTdHJpZGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc2NhbmxpbmUgc3RyaWRlLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IHNjYW5saW5lU3RyaWRlOwotCi0gICAgLyoqCi0gICAgICogVGhlIGRhdGEgYml0IG9mZnNldC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBkYXRhQml0T2Zmc2V0OwotCi0gICAgLyoqCi0gICAgICogVGhlIGJpdCBtYXNrLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IGJpdE1hc2s7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGF0YSBlbGVtZW50IHNpemUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgZGF0YUVsZW1lbnRTaXplOwotCi0gICAgLyoqCi0gICAgICogVGhlIHBpeGVscyBwZXIgZGF0YSBlbGVtZW50LgotICAgICAqLwotICAgIHByaXZhdGUgaW50IHBpeGVsc1BlckRhdGFFbGVtZW50OwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgdHlwZSBvZiB0aGUgc2FtcGxlcy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBudW1iZXJPZkJpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgYml0cyBwZXIgcGl4ZWwuCi0gICAgICogQHBhcmFtIHNjYW5saW5lU3RyaWRlCi0gICAgICogICAgICAgICAgICB0aGUgc2NhbmxpbmUgc3RyaWRlIG9mIHRoZSBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gZGF0YUJpdE9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHRoZSBiYW5kIG9mZnNldHMuCi0gICAgICovCi0gICAgcHVibGljIE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbChpbnQgZGF0YVR5cGUsIGludCB3LCBpbnQgaCwgaW50IG51bWJlck9mQml0cywKLSAgICAgICAgICAgIGludCBzY2FubGluZVN0cmlkZSwgaW50IGRhdGFCaXRPZmZzZXQpIHsKLQotICAgICAgICBzdXBlcihkYXRhVHlwZSwgdywgaCwgMSk7Ci0gICAgICAgIGlmIChkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfQllURSAmJiBkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUCi0gICAgICAgICAgICAgICAgJiYgZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX0lOVCkgewotICAgICAgICAgICAgLy8gYXd0LjYxPVVuc3VwcG9ydGVkIGRhdGEgdHlwZTogezB9Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYxIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICBkYXRhVHlwZSkpOwotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5zY2FubGluZVN0cmlkZSA9IHNjYW5saW5lU3RyaWRlOwotICAgICAgICBpZiAobnVtYmVyT2ZCaXRzID09IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMEM9TnVtYmVyIG9mIEJpdHMgZXF1YWxzIHRvIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjBDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy5waXhlbEJpdFN0cmlkZSA9IG51bWJlck9mQml0czsKLSAgICAgICAgdGhpcy5kYXRhRWxlbWVudFNpemUgPSBEYXRhQnVmZmVyLmdldERhdGFUeXBlU2l6ZShkYXRhVHlwZSk7Ci0gICAgICAgIGlmIChkYXRhRWxlbWVudFNpemUgJSBwaXhlbEJpdFN0cmlkZSAhPSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjBEPVRoZSBudW1iZXIgb2YgYml0cyBwZXIgcGl4ZWwgaXMgbm90IGEgcG93ZXIgb2YgMiBvcgotICAgICAgICAgICAgLy8gcGl4ZWxzIHNwYW4gZGF0YSBlbGVtZW50IGJvdW5kYXJpZXMKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjBEIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoZGF0YUJpdE9mZnNldCAlIG51bWJlck9mQml0cyAhPSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjBFPURhdGEgQml0IG9mZnNldCBpcyBub3QgYSBtdWx0aXBsZSBvZiBwaXhlbCBiaXQgc3RyaWRlCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIwRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHRoaXMuZGF0YUJpdE9mZnNldCA9IGRhdGFCaXRPZmZzZXQ7Ci0KLSAgICAgICAgdGhpcy5waXhlbHNQZXJEYXRhRWxlbWVudCA9IGRhdGFFbGVtZW50U2l6ZSAvIHBpeGVsQml0U3RyaWRlOwotICAgICAgICB0aGlzLmJpdE1hc2sgPSAoMSA8PCBudW1iZXJPZkJpdHMpIC0gMTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlIG9mIHRoZSBzYW1wbGVzLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIG51bWJlck9mQml0cwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBiaXRzIHBlciBwaXhlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKGludCBkYXRhVHlwZSwgaW50IHcsIGludCBoLCBpbnQgbnVtYmVyT2ZCaXRzKSB7Ci0KLSAgICAgICAgdGhpcyhkYXRhVHlwZSwgdywgaCwgbnVtYmVyT2ZCaXRzLAotICAgICAgICAgICAgICAgIChudW1iZXJPZkJpdHMgKiB3ICsgRGF0YUJ1ZmZlci5nZXREYXRhVHlwZVNpemUoZGF0YVR5cGUpIC0gMSkKLSAgICAgICAgICAgICAgICAgICAgICAgIC8gRGF0YUJ1ZmZlci5nZXREYXRhVHlwZVNpemUoZGF0YVR5cGUpLCAwKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgT2JqZWN0IGdldERhdGFFbGVtZW50cyhpbnQgeCwgaW50IHksIE9iamVjdCBvYmosIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHN3aXRjaCAoZ2V0VHJhbnNmZXJUeXBlKCkpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgYnl0ZSBiZGF0YVtdOwotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBiZGF0YSA9IG5ldyBieXRlWzFdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGJkYXRhID0gKGJ5dGVbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJkYXRhWzBdID0gKGJ5dGUpZ2V0U2FtcGxlKHgsIHksIDAsIGRhdGEpOwotICAgICAgICAgICAgICAgIG9iaiA9IGJkYXRhOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNkYXRhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNkYXRhID0gbmV3IHNob3J0WzFdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHNkYXRhID0gKHNob3J0W10pb2JqOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBzZGF0YVswXSA9IChzaG9ydClnZXRTYW1wbGUoeCwgeSwgMCwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgb2JqID0gc2RhdGE7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgaW50IGlkYXRhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlkYXRhID0gbmV3IGludFsxXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBpZGF0YSA9IChpbnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlkYXRhWzBdID0gZ2V0U2FtcGxlKHgsIHksIDAsIGRhdGEpOwotICAgICAgICAgICAgICAgIG9iaiA9IGlkYXRhOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG9iajsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXREYXRhRWxlbWVudHMoaW50IHgsIGludCB5LCBPYmplY3Qgb2JqLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgc2V0U2FtcGxlKHgsIHksIG9iaiwgZGF0YSwgMSwgMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgdGhpcyBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbwotICAgICAqICAgICAgICAgICAgdGhlIE9iamVjdCB0byBiZSBjb21wYXJlZC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBvYmplY3QgaXMgYSBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgd2l0aCB0aGUKLSAgICAgKiAgICAgICAgIHNhbWUgZGF0YSBwYXJhbWV0ZXIgdmFsdWVzIGFzIHRoaXMgTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsLAotICAgICAqICAgICAgICAgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3QgbykgewotICAgICAgICBpZiAoKG8gPT0gbnVsbCkgfHwgIShvIGluc3RhbmNlb2YgTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIG1vZGVsID0gKE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbClvOwotICAgICAgICByZXR1cm4gdGhpcy53aWR0aCA9PSBtb2RlbC53aWR0aCAmJiB0aGlzLmhlaWdodCA9PSBtb2RlbC5oZWlnaHQKLSAgICAgICAgICAgICAgICAmJiB0aGlzLm51bUJhbmRzID09IG1vZGVsLm51bUJhbmRzICYmIHRoaXMuZGF0YVR5cGUgPT0gbW9kZWwuZGF0YVR5cGUKLSAgICAgICAgICAgICAgICAmJiB0aGlzLnBpeGVsQml0U3RyaWRlID09IG1vZGVsLnBpeGVsQml0U3RyaWRlICYmIHRoaXMuYml0TWFzayA9PSBtb2RlbC5iaXRNYXNrCi0gICAgICAgICAgICAgICAgJiYgdGhpcy5waXhlbHNQZXJEYXRhRWxlbWVudCA9PSBtb2RlbC5waXhlbHNQZXJEYXRhRWxlbWVudAotICAgICAgICAgICAgICAgICYmIHRoaXMuZGF0YUVsZW1lbnRTaXplID09IG1vZGVsLmRhdGFFbGVtZW50U2l6ZQotICAgICAgICAgICAgICAgICYmIHRoaXMuZGF0YUJpdE9mZnNldCA9PSBtb2RlbC5kYXRhQml0T2Zmc2V0Ci0gICAgICAgICAgICAgICAgJiYgdGhpcy5zY2FubGluZVN0cmlkZSA9PSBtb2RlbC5zY2FubGluZVN0cmlkZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU2FtcGxlTW9kZWwgY3JlYXRlU3Vic2V0U2FtcGxlTW9kZWwoaW50IGJhbmRzW10pIHsKLSAgICAgICAgaWYgKGJhbmRzICE9IG51bGwgJiYgYmFuZHMubGVuZ3RoICE9IDEpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMEY9TnVtYmVyIG9mIGJhbmRzIG11c3QgYmUgb25seSAxCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIwRiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBjcmVhdGVDb21wYXRpYmxlU2FtcGxlTW9kZWwod2lkdGgsIGhlaWdodCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFNhbXBsZU1vZGVsIGNyZWF0ZUNvbXBhdGlibGVTYW1wbGVNb2RlbChpbnQgdywgaW50IGgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwoZGF0YVR5cGUsIHcsIGgsIHBpeGVsQml0U3RyaWRlKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50W10gZ2V0UGl4ZWwoaW50IHgsIGludCB5LCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGludCBwaXhlbFtdOwotICAgICAgICBpZiAoaUFycmF5ID09IG51bGwpIHsKLSAgICAgICAgICAgIHBpeGVsID0gbmV3IGludFtudW1CYW5kc107Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBwaXhlbCA9IGlBcnJheTsKLSAgICAgICAgfQotCi0gICAgICAgIHBpeGVsWzBdID0gZ2V0U2FtcGxlKHgsIHksIDAsIGRhdGEpOwotICAgICAgICByZXR1cm4gcGl4ZWw7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWwoaW50IHgsIGludCB5LCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBzZXRTYW1wbGUoeCwgeSwgaUFycmF5LCBkYXRhLCAyLCAwKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldFNhbXBsZShpbnQgeCwgaW50IHksIGludCBiLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0IHx8IGIgIT0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpbnQgYml0bnVtID0gZGF0YUJpdE9mZnNldCArIHggKiBwaXhlbEJpdFN0cmlkZTsKLSAgICAgICAgaW50IGVsZW0gPSBkYXRhLmdldEVsZW0oeSAqIHNjYW5saW5lU3RyaWRlICsgYml0bnVtIC8gZGF0YUVsZW1lbnRTaXplKTsKLSAgICAgICAgaW50IHNoaWZ0ID0gZGF0YUVsZW1lbnRTaXplIC0gKGJpdG51bSAmIChkYXRhRWxlbWVudFNpemUgLSAxKSkgLSBwaXhlbEJpdFN0cmlkZTsKLQotICAgICAgICByZXR1cm4gKGVsZW0gPj4gc2hpZnQpICYgYml0TWFzazsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgaW50IHMsIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoYiAhPSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHNldFNhbXBsZSh4LCB5LCBudWxsLCBkYXRhLCAzLCBzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgRGF0YUJ1ZmZlciBjcmVhdGVEYXRhQnVmZmVyKCkgewotICAgICAgICBEYXRhQnVmZmVyIGRhdGFCdWZmZXIgPSBudWxsOwotICAgICAgICBpbnQgc2l6ZSA9IHNjYW5saW5lU3RyaWRlICogaGVpZ2h0OwotCi0gICAgICAgIHN3aXRjaCAoZGF0YVR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgZGF0YUJ1ZmZlciA9IG5ldyBEYXRhQnVmZmVyQnl0ZShzaXplICsgKGRhdGFCaXRPZmZzZXQgKyA3KSAvIDgpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIGRhdGFCdWZmZXIgPSBuZXcgRGF0YUJ1ZmZlclVTaG9ydChzaXplICsgKGRhdGFCaXRPZmZzZXQgKyAxNSkgLyAxNik7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgZGF0YUJ1ZmZlciA9IG5ldyBEYXRhQnVmZmVySW50KHNpemUgKyAoZGF0YUJpdE9mZnNldCArIDMxKSAvIDMyKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZGF0YUJ1ZmZlcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBvZmZzZXQgb2YgdGhlIHNwZWNpZmllZCBwaXhlbCBpbiB0aGUgZGF0YSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBzcGVjaWZpZWQgcGl4ZWwuCi0gICAgICogQHJldHVybiB0aGUgb2Zmc2V0IG9mIHRoZSBzcGVjaWZpZWQgcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRPZmZzZXQoaW50IHgsIGludCB5KSB7Ci0gICAgICAgIHJldHVybiB5ICogc2NhbmxpbmVTdHJpZGUgKyAoeCAqIHBpeGVsQml0U3RyaWRlICsgZGF0YUJpdE9mZnNldCkgLyBkYXRhRWxlbWVudFNpemU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRTYW1wbGVTaXplKGludCBiYW5kKSB7Ci0gICAgICAgIHJldHVybiBwaXhlbEJpdFN0cmlkZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBiaXQgb2Zmc2V0IGluIHRoZSBkYXRhIGVsZW1lbnQgd2hpY2ggaXMgc3RvcmVkIGZvciB0aGUgc3BlY2lmaWVkCi0gICAgICogcGl4ZWwgb2YgYSBzY2FubGluZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsLgotICAgICAqIEByZXR1cm4gdGhlIGJpdCBvZmZzZXQgb2YgdGhlIHBpeGVsIGluIHRoZSBkYXRhIGVsZW1lbnQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRCaXRPZmZzZXQoaW50IHgpIHsKLSAgICAgICAgcmV0dXJuICh4ICogcGl4ZWxCaXRTdHJpZGUgKyBkYXRhQml0T2Zmc2V0KSAlIGRhdGFFbGVtZW50U2l6ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50W10gZ2V0U2FtcGxlU2l6ZSgpIHsKLSAgICAgICAgaW50IHNhbXBsZVNpemVzW10gPSB7Ci0gICAgICAgICAgICBwaXhlbEJpdFN0cmlkZQotICAgICAgICB9OwotICAgICAgICByZXR1cm4gc2FtcGxlU2l6ZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIGhhc2ggY29kZSBvZiB0aGlzIE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBjbGFzcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBoYXNoIGNvZGUgb2YgdGhpcyBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgY2xhc3MuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgaW50IGhhc2ggPSAwOwotICAgICAgICBpbnQgdG1wID0gMDsKLQotICAgICAgICBoYXNoID0gd2lkdGg7Ci0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICBoYXNoIHw9IHRtcDsKLSAgICAgICAgaGFzaCBePSBoZWlnaHQ7Ci0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICBoYXNoIHw9IHRtcDsKLSAgICAgICAgaGFzaCBePSBudW1CYW5kczsKLSAgICAgICAgdG1wID0gaGFzaCA+Pj4gMjQ7Ci0gICAgICAgIGhhc2ggPDw9IDg7Ci0gICAgICAgIGhhc2ggfD0gdG1wOwotICAgICAgICBoYXNoIF49IGRhdGFUeXBlOwotICAgICAgICB0bXAgPSBoYXNoID4+PiAyNDsKLSAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgaGFzaCB8PSB0bXA7Ci0gICAgICAgIGhhc2ggXj0gc2NhbmxpbmVTdHJpZGU7Ci0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICBoYXNoIHw9IHRtcDsKLSAgICAgICAgaGFzaCBePSBwaXhlbEJpdFN0cmlkZTsKLSAgICAgICAgdG1wID0gaGFzaCA+Pj4gMjQ7Ci0gICAgICAgIGhhc2ggPDw9IDg7Ci0gICAgICAgIGhhc2ggfD0gdG1wOwotICAgICAgICBoYXNoIF49IGRhdGFCaXRPZmZzZXQ7Ci0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICBoYXNoIHw9IHRtcDsKLSAgICAgICAgaGFzaCBePSBiaXRNYXNrOwotICAgICAgICB0bXAgPSBoYXNoID4+PiAyNDsKLSAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgaGFzaCB8PSB0bXA7Ci0gICAgICAgIGhhc2ggXj0gZGF0YUVsZW1lbnRTaXplOwotICAgICAgICB0bXAgPSBoYXNoID4+PiAyNDsKLSAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgaGFzaCB8PSB0bXA7Ci0gICAgICAgIGhhc2ggXj0gcGl4ZWxzUGVyRGF0YUVsZW1lbnQ7Ci0gICAgICAgIHJldHVybiBoYXNoOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0VHJhbnNmZXJUeXBlKCkgewotICAgICAgICBpZiAocGl4ZWxCaXRTdHJpZGUgPiAxNikgewotICAgICAgICAgICAgcmV0dXJuIERhdGFCdWZmZXIuVFlQRV9JTlQ7Ci0gICAgICAgIH0gZWxzZSBpZiAocGl4ZWxCaXRTdHJpZGUgPiA4KSB7Ci0gICAgICAgICAgICByZXR1cm4gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJldHVybiBEYXRhQnVmZmVyLlRZUEVfQllURTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNjYW5saW5lIHN0cmlkZSBvZiB0aGlzIE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzY2FubGluZSBzdHJpZGUgb2YgdGhpcyBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRTY2FubGluZVN0cmlkZSgpIHsKLSAgICAgICAgcmV0dXJuIHNjYW5saW5lU3RyaWRlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHBpeGVsIGJpdCBzdHJpZGUgb2YgdGhpcyBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgcGl4ZWwgYml0IHN0cmlkZSBvZiB0aGlzIE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFBpeGVsQml0U3RyaWRlKCkgewotICAgICAgICByZXR1cm4gcGl4ZWxCaXRTdHJpZGU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXROdW1EYXRhRWxlbWVudHMoKSB7Ci0gICAgICAgIHJldHVybiAxOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRhdGEgYml0IG9mZnNldC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhIGJpdCBvZmZzZXQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXREYXRhQml0T2Zmc2V0KCkgewotICAgICAgICByZXR1cm4gZGF0YUJpdE9mZnNldDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCBpcyB1c2VkIGJ5IG90aGVyIG1ldGhvZHMgb2YgdGhpcyBjbGFzcy4gVGhlIGJlaGF2aW9yIG9mIHRoaXMKLSAgICAgKiBtZXRob2QgZGVwZW5kcyBvbiB0aGUgbWV0aG9kIHdoaWNoIGhhcyBiZWVuIGludm9rZSB0aGlzIG9uZS4gVGhlIGFyZ3VtZW50Ci0gICAgICogbWV0aG9kSWQgaXMgdXNlZCB0byBjaG9vc2UgdmFsaWQgYmVoYXZpb3IgaW4gYSBwYXJ0aWN1bGFyIGNhc2UuIElmCi0gICAgICogbWV0aG9kSWQgaXMgZXF1YWwgdG8gMSBpdCBtZWFucyB0aGF0IHRoaXMgbWV0aG9kIGhhcyBiZWVuIGludm9rZWQgYnkgdGhlCi0gICAgICogc2V0RGF0YUVsZW1lbnRzKCkgbWV0aG9kLCAyIC0gbWVhbnMgc2V0UGl4ZWwoKSwgYW5kIHNldFNhbXBsZSgpIGluIGFueQotICAgICAqIG90aGVyIGNhc2VzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgeC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHkuCi0gICAgICogQHBhcmFtIG9iagotICAgICAqICAgICAgICAgICAgdGhlIG9iai4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEuCi0gICAgICogQHBhcmFtIG1ldGhvZElkCi0gICAgICogICAgICAgICAgICB0aGUgbWV0aG9kIGlkLgotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgc2V0U2FtcGxlKGZpbmFsIGludCB4LCBmaW5hbCBpbnQgeSwgZmluYWwgT2JqZWN0IG9iaiwgZmluYWwgRGF0YUJ1ZmZlciBkYXRhLAotICAgICAgICAgICAgZmluYWwgaW50IG1ldGhvZElkLCBpbnQgcykgewotICAgICAgICBpZiAoKHggPCAwKSB8fCAoeSA8IDApIHx8ICh4ID49IHRoaXMud2lkdGgpIHx8ICh5ID49IHRoaXMuaGVpZ2h0KSkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBmaW5hbCBpbnQgYml0bnVtID0gZGF0YUJpdE9mZnNldCArIHggKiBwaXhlbEJpdFN0cmlkZTsKLSAgICAgICAgZmluYWwgaW50IGlkeCA9IHkgKiBzY2FubGluZVN0cmlkZSArIGJpdG51bSAvIGRhdGFFbGVtZW50U2l6ZTsKLSAgICAgICAgZmluYWwgaW50IHNoaWZ0ID0gZGF0YUVsZW1lbnRTaXplIC0gKGJpdG51bSAmIChkYXRhRWxlbWVudFNpemUgLSAxKSkgLSBwaXhlbEJpdFN0cmlkZTsKLSAgICAgICAgZmluYWwgaW50IG1hc2sgPSB+KGJpdE1hc2sgPDwgc2hpZnQpOwotICAgICAgICBpbnQgZWxlbSA9IGRhdGEuZ2V0RWxlbShpZHgpOwotCi0gICAgICAgIHN3aXRjaCAobWV0aG9kSWQpIHsKLSAgICAgICAgICAgIGNhc2UgMTogeyAvLyBJbnZva2VkIGZyb20gc2V0RGF0YUVsZW1lbnRzKCkKLSAgICAgICAgICAgICAgICBzd2l0Y2ggKGdldFRyYW5zZmVyVHlwZSgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgICAgICAgICBzID0gKChieXRlW10pb2JqKVswXSAmIDB4ZmY7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgICAgICAgICAgcyA9ICgoc2hvcnRbXSlvYmopWzBdICYgMHhmZmZmOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICAgICAgICAgIHMgPSAoKGludFtdKW9iailbMF07Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjYXNlIDI6IHsgLy8gSW52b2tlZCBmcm9tIHNldFBpeGVsKCkKLSAgICAgICAgICAgICAgICBzID0gKChpbnRbXSlvYmopWzBdOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgZWxlbSAmPSBtYXNrOwotICAgICAgICBlbGVtIHw9IChzICYgYml0TWFzaykgPDwgc2hpZnQ7Ci0gICAgICAgIGRhdGEuc2V0RWxlbShpZHgsIGVsZW0pOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9QYWNrZWRDb2xvck1vZGVsLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvUGFja2VkQ29sb3JNb2RlbC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0ZDFjMmU1Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9QYWNrZWRDb2xvck1vZGVsLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0MDIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5UcmFuc3BhcmVuY3k7Ci1pbXBvcnQgamF2YS5hd3QuY29sb3IuQ29sb3JTcGFjZTsKLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlzOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIGNsYXNzIFBhY2tlZENvbG9yTW9kZWwgcmVwcmVzZW50cyBhIGNvbG9yIG1vZGVsIHdoZXJlIHRoZSBjb21wb25lbnRzIGFyZQotICoganVzdCB0aGUgcmVkLCBncmVlbiwgYW5kIGJsdWUgYmFuZHMsIHBsdXMgYW4gYWxwaGEgYmFuZCBpZiBhbHBoYSBpcwotICogc3VwcG9ydGVkLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIFBhY2tlZENvbG9yTW9kZWwgZXh0ZW5kcyBDb2xvck1vZGVsIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb21wb25lbnQgbWFza3MuCi0gICAgICovCi0gICAgaW50IGNvbXBvbmVudE1hc2tzW107Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgb2Zmc2V0cy4KLSAgICAgKi8KLSAgICBpbnQgb2Zmc2V0c1tdOwotCi0gICAgLyoqCi0gICAgICogVGhlIHNjYWxlcy4KLSAgICAgKi8KLSAgICBmbG9hdCBzY2FsZXNbXTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBwYWNrZWQgY29sb3IgbW9kZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgY29sb3Igc3BhY2UuCi0gICAgICogQHBhcmFtIGJpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBjb21wb25lbnQgbWFza3MuCi0gICAgICogQHBhcmFtIGNvbG9yTWFza0FycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgdGhhdCBnaXZlcyB0aGUgYml0bWFzayBjb3JyZXNwb25kaW5nIHRvIGVhY2ggY29sb3IKLSAgICAgKiAgICAgICAgICAgIGJhbmQgKHJlZCwgZ3JlZW4sIGFuZCBibHVlKS4KLSAgICAgKiBAcGFyYW0gYWxwaGFNYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYml0bWFzayBjb3JyZXNwb25kaW5nIHRvIHRoZSBhbHBoYSBiYW5kLgotICAgICAqIEBwYXJhbSBpc0FscGhhUHJlbXVsdGlwbGllZAotICAgICAqICAgICAgICAgICAgd2hldGhlciB0aGUgYWxwaGEgaXMgcHJlLW11bHRpcGxpZWQgaW4gdGhpcyBjb2xvciBtb2RlbC4KLSAgICAgKiBAcGFyYW0gdHJhbnMKLSAgICAgKiAgICAgICAgICAgIHRoZSB0cmFuc3BhcmVuY3kgc3RyYXRlZ3ksIEBzZWUgamF2YS5hd3QuVHJhbnNwYXJlbmN5LgotICAgICAqIEBwYXJhbSB0cmFuc2ZlclR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSB0cmFuc2ZlciB0eXBlIChwcmltaXRpdmUgamF2YSB0eXBlIHRvIHVzZSBmb3IgdGhlCi0gICAgICogICAgICAgICAgICBjb21wb25lbnRzKS4KLSAgICAgKiBAdGhyb3dzIElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBudW1iZXIgb2YgYml0cyBpbiB0aGUgY29tYmluZWQgYml0bWFza3MgZm9yIHRoZSBjb2xvcgotICAgICAqICAgICAgICAgICAgIGJhbmRzIGlzIGxlc3MgdGhhbiBvbmUgb3IgZ3JlYXRlciB0aGFuIDMyLgotICAgICAqLwotICAgIHB1YmxpYyBQYWNrZWRDb2xvck1vZGVsKENvbG9yU3BhY2Ugc3BhY2UsIGludCBiaXRzLCBpbnQgY29sb3JNYXNrQXJyYXlbXSwgaW50IGFscGhhTWFzaywKLSAgICAgICAgICAgIGJvb2xlYW4gaXNBbHBoYVByZW11bHRpcGxpZWQsIGludCB0cmFucywgaW50IHRyYW5zZmVyVHlwZSkgewotCi0gICAgICAgIHN1cGVyKGJpdHMsIGNyZWF0ZUJpdHMoY29sb3JNYXNrQXJyYXksIGFscGhhTWFzayksIHNwYWNlLCAoYWxwaGFNYXNrID09IDAgPyBmYWxzZSA6IHRydWUpLAotICAgICAgICAgICAgICAgIGlzQWxwaGFQcmVtdWx0aXBsaWVkLCB0cmFucywgdmFsaWRhdGVUcmFuc2ZlclR5cGUodHJhbnNmZXJUeXBlKSk7Ci0KLSAgICAgICAgaWYgKHBpeGVsX2JpdHMgPCAxIHx8IHBpeGVsX2JpdHMgPiAzMikgewotICAgICAgICAgICAgLy8gYXd0LjIzNj1UaGUgYml0cyBpcyBsZXNzIHRoYW4gMSBvciBncmVhdGVyIHRoYW4gMzIKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjM2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBjb21wb25lbnRNYXNrcyA9IG5ldyBpbnRbbnVtQ29tcG9uZW50c107Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQ29sb3JDb21wb25lbnRzOyBpKyspIHsKLSAgICAgICAgICAgIGNvbXBvbmVudE1hc2tzW2ldID0gY29sb3JNYXNrQXJyYXlbaV07Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIGNvbXBvbmVudE1hc2tzW251bUNvbG9yQ29tcG9uZW50c10gPSBhbHBoYU1hc2s7Ci0gICAgICAgICAgICBpZiAodGhpcy5iaXRzW251bUNvbG9yQ29tcG9uZW50c10gPT0gMSkgewotICAgICAgICAgICAgICAgIHRyYW5zcGFyZW5jeSA9IFRyYW5zcGFyZW5jeS5CSVRNQVNLOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcGFyc2VDb21wb25lbnRzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHBhY2tlZCBjb2xvciBtb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3BhY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBjb2xvciBzcGFjZS4KLSAgICAgKiBAcGFyYW0gYml0cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGNvbXBvbmVudCBtYXNrcy4KLSAgICAgKiBAcGFyYW0gcm1hc2sKLSAgICAgKiAgICAgICAgICAgIHRoZSBiaXRtYXNrIGNvcnJlc3BvbmRpbmcgdG8gdGhlIHJlZCBiYW5kLgotICAgICAqIEBwYXJhbSBnbWFzawotICAgICAqICAgICAgICAgICAgdGhlIGJpdG1hc2sgY29ycmVzcG9uZGluZyB0byB0aGUgZ3JlZW4gYmFuZC4KLSAgICAgKiBAcGFyYW0gYm1hc2sKLSAgICAgKiAgICAgICAgICAgIHRoZSBiaXRtYXNrIGNvcnJlc3BvbmRpbmcgdG8gdGhlIGJsdWUgYmFuZC4KLSAgICAgKiBAcGFyYW0gYW1hc2sKLSAgICAgKiAgICAgICAgICAgIHRoZSBiaXRtYXNrIGNvcnJlc3BvbmRpbmcgdG8gdGhlIGFscGhhIGJhbmQuCi0gICAgICogQHBhcmFtIGlzQWxwaGFQcmVtdWx0aXBsaWVkCi0gICAgICogICAgICAgICAgICB3aGV0aGVyIHRoZSBhbHBoYSBpcyBwcmUtbXVsdGlwbGllZCBpbiB0aGlzIGNvbG9yIG1vZGVsLgotICAgICAqIEBwYXJhbSB0cmFucwotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zcGFyZW5jeSBzdHJhdGVneSwgQHNlZSBqYXZhLmF3dC5UcmFuc3BhcmVuY3kuCi0gICAgICogQHBhcmFtIHRyYW5zZmVyVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIHRyYW5zZmVyIHR5cGUgKHByaW1pdGl2ZSBqYXZhIHR5cGUgdG8gdXNlIGZvciB0aGUKLSAgICAgKiAgICAgICAgICAgIGNvbXBvbmVudHMpLgotICAgICAqIEB0aHJvd3MgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIG51bWJlciBvZiBiaXRzIGluIHRoZSBjb21iaW5lZCBiaXRtYXNrcyBmb3IgdGhlIGNvbG9yCi0gICAgICogICAgICAgICAgICAgYmFuZHMgaXMgbGVzcyB0aGFuIG9uZSBvciBncmVhdGVyIHRoYW4gMzIuCi0gICAgICovCi0gICAgcHVibGljIFBhY2tlZENvbG9yTW9kZWwoQ29sb3JTcGFjZSBzcGFjZSwgaW50IGJpdHMsIGludCBybWFzaywgaW50IGdtYXNrLCBpbnQgYm1hc2ssIGludCBhbWFzaywKLSAgICAgICAgICAgIGJvb2xlYW4gaXNBbHBoYVByZW11bHRpcGxpZWQsIGludCB0cmFucywgaW50IHRyYW5zZmVyVHlwZSkgewotCi0gICAgICAgIHN1cGVyKGJpdHMsIGNyZWF0ZUJpdHMocm1hc2ssIGdtYXNrLCBibWFzaywgYW1hc2spLCBzcGFjZSwgKGFtYXNrID09IDAgPyBmYWxzZSA6IHRydWUpLAotICAgICAgICAgICAgICAgIGlzQWxwaGFQcmVtdWx0aXBsaWVkLCB0cmFucywgdmFsaWRhdGVUcmFuc2ZlclR5cGUodHJhbnNmZXJUeXBlKSk7Ci0KLSAgICAgICAgaWYgKHBpeGVsX2JpdHMgPCAxIHx8IHBpeGVsX2JpdHMgPiAzMikgewotICAgICAgICAgICAgLy8gYXd0LjIzNj1UaGUgYml0cyBpcyBsZXNzIHRoYW4gMSBvciBncmVhdGVyIHRoYW4gMzIKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjM2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoY3MuZ2V0VHlwZSgpICE9IENvbG9yU3BhY2UuVFlQRV9SR0IpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMzk9VGhlIHNwYWNlIGlzIG5vdCBhIFRZUEVfUkdCIHNwYWNlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzOSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1Db2xvckNvbXBvbmVudHM7IGkrKykgewotICAgICAgICAgICAgaWYgKGNzLmdldE1pblZhbHVlKGkpICE9IDAuMGYgfHwgY3MuZ2V0TWF4VmFsdWUoaSkgIT0gMS4wZikgewotICAgICAgICAgICAgICAgIC8vIGF3dC4yM0E9VGhlIG1pbi9tYXggbm9ybWFsaXplZCBjb21wb25lbnQgdmFsdWVzIGFyZSBub3QKLSAgICAgICAgICAgICAgICAvLyAwLjAvMS4wCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yM0EiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBjb21wb25lbnRNYXNrcyA9IG5ldyBpbnRbbnVtQ29tcG9uZW50c107Ci0gICAgICAgIGNvbXBvbmVudE1hc2tzWzBdID0gcm1hc2s7Ci0gICAgICAgIGNvbXBvbmVudE1hc2tzWzFdID0gZ21hc2s7Ci0gICAgICAgIGNvbXBvbmVudE1hc2tzWzJdID0gYm1hc2s7Ci0KLSAgICAgICAgaWYgKGhhc0FscGhhKSB7Ci0gICAgICAgICAgICBjb21wb25lbnRNYXNrc1szXSA9IGFtYXNrOwotICAgICAgICAgICAgaWYgKHRoaXMuYml0c1szXSA9PSAxKSB7Ci0gICAgICAgICAgICAgICAgdHJhbnNwYXJlbmN5ID0gVHJhbnNwYXJlbmN5LkJJVE1BU0s7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBwYXJzZUNvbXBvbmVudHMoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgZ2V0QWxwaGFSYXN0ZXIoV3JpdGFibGVSYXN0ZXIgcmFzdGVyKSB7Ci0gICAgICAgIGlmICghaGFzQWxwaGEpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHggPSByYXN0ZXIuZ2V0TWluWCgpOwotICAgICAgICBpbnQgeSA9IHJhc3Rlci5nZXRNaW5ZKCk7Ci0gICAgICAgIGludCB3ID0gcmFzdGVyLmdldFdpZHRoKCk7Ci0gICAgICAgIGludCBoID0gcmFzdGVyLmdldEhlaWdodCgpOwotICAgICAgICBpbnQgYmFuZFtdID0gbmV3IGludFsxXTsKLSAgICAgICAgYmFuZFswXSA9IHJhc3Rlci5nZXROdW1CYW5kcygpIC0gMTsKLSAgICAgICAgcmV0dXJuIHJhc3Rlci5jcmVhdGVXcml0YWJsZUNoaWxkKHgsIHksIHcsIGgsIHgsIHksIGJhbmQpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3Qgb2JqKSB7Ci0gICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIGlmICghKG9iaiBpbnN0YW5jZW9mIFBhY2tlZENvbG9yTW9kZWwpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgUGFja2VkQ29sb3JNb2RlbCBjbSA9IChQYWNrZWRDb2xvck1vZGVsKW9iajsKLQotICAgICAgICByZXR1cm4gKHBpeGVsX2JpdHMgPT0gY20uZ2V0UGl4ZWxTaXplKCkgJiYgdHJhbnNmZXJUeXBlID09IGNtLmdldFRyYW5zZmVyVHlwZSgpCi0gICAgICAgICAgICAgICAgJiYgY3MuZ2V0VHlwZSgpID09IGNtLmdldENvbG9yU3BhY2UoKS5nZXRUeXBlKCkgJiYgaGFzQWxwaGEgPT0gY20uaGFzQWxwaGEoKQotICAgICAgICAgICAgICAgICYmIGlzQWxwaGFQcmVtdWx0aXBsaWVkID09IGNtLmlzQWxwaGFQcmVtdWx0aXBsaWVkKCkKLSAgICAgICAgICAgICAgICAmJiB0cmFuc3BhcmVuY3kgPT0gY20uZ2V0VHJhbnNwYXJlbmN5KCkKLSAgICAgICAgICAgICAgICAmJiBudW1Db2xvckNvbXBvbmVudHMgPT0gY20uZ2V0TnVtQ29sb3JDb21wb25lbnRzKCkKLSAgICAgICAgICAgICAgICAmJiBudW1Db21wb25lbnRzID09IGNtLmdldE51bUNvbXBvbmVudHMoKQotICAgICAgICAgICAgICAgICYmIEFycmF5cy5lcXVhbHMoYml0cywgY20uZ2V0Q29tcG9uZW50U2l6ZSgpKSAmJiBBcnJheXMuZXF1YWxzKGNvbXBvbmVudE1hc2tzLCBjbQotICAgICAgICAgICAgICAgIC5nZXRNYXNrcygpKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb21wYXRpYmxlU2FtcGxlTW9kZWwoU2FtcGxlTW9kZWwgc20pIHsKLSAgICAgICAgaWYgKHNtID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgICAgICBpZiAoIShzbSBpbnN0YW5jZW9mIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBlc20gPSAoU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbClzbTsKLQotICAgICAgICByZXR1cm4gKChlc20uZ2V0TnVtQmFuZHMoKSA9PSBudW1Db21wb25lbnRzKSAmJiAoZXNtLmdldFRyYW5zZmVyVHlwZSgpID09IHRyYW5zZmVyVHlwZSkgJiYgQXJyYXlzCi0gICAgICAgICAgICAgICAgLmVxdWFscyhlc20uZ2V0Qml0TWFza3MoKSwgY29tcG9uZW50TWFza3MpKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU2FtcGxlTW9kZWwgY3JlYXRlQ29tcGF0aWJsZVNhbXBsZU1vZGVsKGludCB3LCBpbnQgaCkgewotICAgICAgICByZXR1cm4gbmV3IFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwodHJhbnNmZXJUeXBlLCB3LCBoLCBjb21wb25lbnRNYXNrcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYml0bWFzayBjb3JyZXNwb25kaW5nIHRvIHRoZSBzcGVjaWZpZWQgY29sb3IgY29tcG9uZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBkZXNpcmVkIGNvbG9yLgotICAgICAqIEByZXR1cm4gdGhlIG1hc2suCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGludCBnZXRNYXNrKGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gY29tcG9uZW50TWFza3NbaW5kZXhdOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJpdG1hc2tzIG9mIHRoZSBjb21wb25lbnRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG1hc2tzLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnRbXSBnZXRNYXNrcygpIHsKLSAgICAgICAgcmV0dXJuIChjb21wb25lbnRNYXNrcy5jbG9uZSgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBiaXRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvck1hc2tBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGNvbG9yIG1hc2sgYXJyYXkuCi0gICAgICogQHBhcmFtIGFscGhhTWFzawotICAgICAqICAgICAgICAgICAgdGhlIGFscGhhIG1hc2suCi0gICAgICogQHJldHVybiB0aGUgaW50W10uCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgaW50W10gY3JlYXRlQml0cyhpbnQgY29sb3JNYXNrQXJyYXlbXSwgaW50IGFscGhhTWFzaykgewotICAgICAgICBpbnQgYml0c1tdOwotICAgICAgICBpbnQgbnVtQ29tcDsKLSAgICAgICAgaWYgKGFscGhhTWFzayA9PSAwKSB7Ci0gICAgICAgICAgICBudW1Db21wID0gY29sb3JNYXNrQXJyYXkubGVuZ3RoOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbnVtQ29tcCA9IGNvbG9yTWFza0FycmF5Lmxlbmd0aCArIDE7Ci0gICAgICAgIH0KLQotICAgICAgICBiaXRzID0gbmV3IGludFtudW1Db21wXTsKLSAgICAgICAgaW50IGkgPSAwOwotICAgICAgICBmb3IgKDsgaSA8IGNvbG9yTWFza0FycmF5Lmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBiaXRzW2ldID0gY291bnRDb21wQml0cyhjb2xvck1hc2tBcnJheVtpXSk7Ci0gICAgICAgICAgICBpZiAoYml0c1tpXSA8IDApIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjNCPVRoZSBtYXNrIG9mIHRoZSB7MH0gY29tcG9uZW50IGlzIG5vdCBjb250aWd1b3VzCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yM0IiLCBpKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChpIDwgbnVtQ29tcCkgewotICAgICAgICAgICAgYml0c1tpXSA9IGNvdW50Q29tcEJpdHMoYWxwaGFNYXNrKTsKLQotICAgICAgICAgICAgaWYgKGJpdHNbaV0gPCAwKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjIzQz1UaGUgbWFzayBvZiB0aGUgYWxwaGEgY29tcG9uZW50IGlzIG5vdCBjb250aWd1b3VzCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yM0MiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBiaXRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGJpdHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgcm1hc2suCi0gICAgICogQHBhcmFtIGdtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgZ21hc2suCi0gICAgICogQHBhcmFtIGJtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYm1hc2suCi0gICAgICogQHBhcmFtIGFtYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYW1hc2suCi0gICAgICogQHJldHVybiB0aGUgaW50W10uCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgaW50W10gY3JlYXRlQml0cyhpbnQgcm1hc2ssIGludCBnbWFzaywgaW50IGJtYXNrLCBpbnQgYW1hc2spIHsKLQotICAgICAgICBpbnQgbnVtQ29tcDsKLSAgICAgICAgaWYgKGFtYXNrID09IDApIHsKLSAgICAgICAgICAgIG51bUNvbXAgPSAzOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbnVtQ29tcCA9IDQ7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IGJpdHNbXSA9IG5ldyBpbnRbbnVtQ29tcF07Ci0KLSAgICAgICAgYml0c1swXSA9IGNvdW50Q29tcEJpdHMocm1hc2spOwotICAgICAgICBpZiAoYml0c1swXSA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yM0Q9VGhlIG1hc2sgb2YgdGhlIHJlZCBjb21wb25lbnQgaXMgbm90IGNvbnRpZ3VvdXMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjNEIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBiaXRzWzFdID0gY291bnRDb21wQml0cyhnbWFzayk7Ci0gICAgICAgIGlmIChiaXRzWzFdIDwgMCkgewotICAgICAgICAgICAgLy8gYXd0LjIzRT1UaGUgbWFzayBvZiB0aGUgZ3JlZW4gY29tcG9uZW50IGlzIG5vdCBjb250aWd1b3VzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgYml0c1syXSA9IGNvdW50Q29tcEJpdHMoYm1hc2spOwotICAgICAgICBpZiAoYml0c1syXSA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yM0Y9VGhlIG1hc2sgb2YgdGhlIGJsdWUgY29tcG9uZW50IGlzIG5vdCBjb250aWd1b3VzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzRiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGFtYXNrICE9IDApIHsKLSAgICAgICAgICAgIGJpdHNbM10gPSBjb3VudENvbXBCaXRzKGFtYXNrKTsKLSAgICAgICAgICAgIGlmIChiaXRzWzNdIDwgMCkgewotICAgICAgICAgICAgICAgIC8vIGF3dC4yM0M9VGhlIG1hc2sgb2YgdGhlIGFscGhhIGNvbXBvbmVudCBpcyBub3QgY29udGlndW91cwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjNDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gYml0czsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb3VudCBjb21wIGJpdHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGNvbXBNYXNrCi0gICAgICogICAgICAgICAgICB0aGUgY29tcCBtYXNrLgotICAgICAqIEByZXR1cm4gdGhlIGludC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBpbnQgY291bnRDb21wQml0cyhpbnQgY29tcE1hc2spIHsKLSAgICAgICAgaW50IGJpdHMgPSAwOwotICAgICAgICBpZiAoY29tcE1hc2sgIT0gMCkgewotICAgICAgICAgICAgLy8gRGVsZXRpbmcgZmluYWwgemVyb3MKLSAgICAgICAgICAgIHdoaWxlICgoY29tcE1hc2sgJiAxKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgY29tcE1hc2sgPj4+PSAxOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgLy8gQ291bnRpbmcgY29tcG9uZW50IGJpdHMKLSAgICAgICAgICAgIHdoaWxlICgoY29tcE1hc2sgJiAxKSA9PSAxKSB7Ci0gICAgICAgICAgICAgICAgY29tcE1hc2sgPj4+PSAxOwotICAgICAgICAgICAgICAgIGJpdHMrKzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChjb21wTWFzayAhPSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gLTE7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gYml0czsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBWYWxpZGF0ZSB0cmFuc2ZlciB0eXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0cmFuc2ZlclR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSB0cmFuc2ZlciB0eXBlLgotICAgICAqIEByZXR1cm4gdGhlIGludC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBpbnQgdmFsaWRhdGVUcmFuc2ZlclR5cGUoaW50IHRyYW5zZmVyVHlwZSkgewotICAgICAgICBpZiAodHJhbnNmZXJUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9CWVRFICYmIHRyYW5zZmVyVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUCi0gICAgICAgICAgICAgICAgJiYgdHJhbnNmZXJUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9JTlQpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNDA9VGhlIHRyYW5zZmVyVHlwZSBub3QgaXMgb25lIG9mIERhdGFCdWZmZXIuVFlQRV9CWVRFLAotICAgICAgICAgICAgLy8gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCBvciBEYXRhQnVmZmVyLlRZUEVfSU5UCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI0MCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB0cmFuc2ZlclR5cGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUGFyc2VzIHRoZSBjb21wb25lbnRzLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBwYXJzZUNvbXBvbmVudHMoKSB7Ci0gICAgICAgIG9mZnNldHMgPSBuZXcgaW50W251bUNvbXBvbmVudHNdOwotICAgICAgICBzY2FsZXMgPSBuZXcgZmxvYXRbbnVtQ29tcG9uZW50c107Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQ29tcG9uZW50czsgaSsrKSB7Ci0gICAgICAgICAgICBpbnQgb2ZmID0gMDsKLSAgICAgICAgICAgIGludCBtYXNrID0gY29tcG9uZW50TWFza3NbaV07Ci0gICAgICAgICAgICB3aGlsZSAoKG1hc2sgJiAxKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgbWFzayA+Pj49IDE7Ci0gICAgICAgICAgICAgICAgb2ZmKys7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBvZmZzZXRzW2ldID0gb2ZmOwotICAgICAgICAgICAgaWYgKGJpdHNbaV0gPT0gMCkgewotICAgICAgICAgICAgICAgIHNjYWxlc1tpXSA9IDI1Ni4wZjsgLy8gTWF5IGJlIGFueSB2YWx1ZSBkaWZmZXJlbnQgZnJvbSB6ZXJvLAotICAgICAgICAgICAgICAgIC8vIGJlY2F1c2Ugd2lsbCBkaXZpZGluZyBieSB6ZXJvCi0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHNjYWxlc1tpXSA9IDI1NS4wZiAvIG1heFZhbHVlc1tpXTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvUGl4ZWxHcmFiYmVyLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvUGl4ZWxHcmFiYmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGNlY2Q1YzguLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL1BpeGVsR3JhYmJlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDA4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEuYXd0LkltYWdlOwotaW1wb3J0IGphdmEudXRpbC5IYXNodGFibGU7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotcHVibGljIGNsYXNzIFBpeGVsR3JhYmJlciBpbXBsZW1lbnRzIEltYWdlQ29uc3VtZXIgewotCi0gICAgaW50IHdpZHRoOwotICAgIGludCBoZWlnaHQ7Ci0gICAgaW50IFg7Ci0gICAgaW50IFk7Ci0gICAgaW50IG9mZnNldDsKLSAgICBpbnQgc2NhbmxpbmU7Ci0gICAgSW1hZ2VQcm9kdWNlciBwcm9kdWNlcjsKLQotICAgIGJ5dGUgYkRhdGFbXTsKLSAgICBpbnQgaURhdGFbXTsKLSAgICBDb2xvck1vZGVsIGNtOwotCi0gICAgcHJpdmF0ZSBpbnQgZ3JhYmJlclN0YXR1czsKLSAgICBwcml2YXRlIGludCBkYXRhVHlwZTsKLSAgICBwcml2YXRlIGJvb2xlYW4gaXNHcmFiYmluZzsKLSAgICBwcml2YXRlIGJvb2xlYW4gaXNSR0I7Ci0KLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBEQVRBX1RZUEVfQllURSA9IDA7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IERBVEFfVFlQRV9JTlQgPSAxOwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBEQVRBX1RZUEVfVU5ERUZJTkVEID0gMjsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBBTExfQklUUyA9IChJbWFnZU9ic2VydmVyLkZSQU1FQklUUyB8Ci0gICAgICAgICAgICBJbWFnZU9ic2VydmVyLkFMTEJJVFMpOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEdSQUJCSU5HX1NUT1AgPSBBTExfQklUUyB8IEltYWdlT2JzZXJ2ZXIuRVJST1I7Ci0KLQotCi0gICAgcHVibGljIFBpeGVsR3JhYmJlcihJbWFnZVByb2R1Y2VyIGlwLCBpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50W10gcGl4LAotICAgICAgICAgICAgaW50IG9mZiwgaW50IHNjYW5zaXplKSB7Ci0gICAgICAgIGluaXRpYWxpemUoaXAsIHgsIHksIHcsIGgsIHBpeCwgb2ZmLCBzY2Fuc2l6ZSwgdHJ1ZSk7Ci0gICAgfQotCi0gICAgcHVibGljIFBpeGVsR3JhYmJlcihJbWFnZSBpbWcsIGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnRbXSBwaXgsCi0gICAgICAgICAgICBpbnQgb2ZmLCBpbnQgc2NhbnNpemUpIHsKLSAgICAgICAgaW5pdGlhbGl6ZShpbWcuZ2V0U291cmNlKCksIHgsIHksIHcsIGgsIHBpeCwgb2ZmLCBzY2Fuc2l6ZSwgdHJ1ZSk7Ci0gICAgfQotCi0gICAgcHVibGljIFBpeGVsR3JhYmJlcihJbWFnZSBpbWcsIGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBib29sZWFuIGZvcmNlUkdCKSB7Ci0gICAgICAgIGluaXRpYWxpemUoaW1nLmdldFNvdXJjZSgpLCB4LCB5LCB3LCBoLCBudWxsLCAwLCAwLCBmb3JjZVJHQik7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0UHJvcGVydGllcyhIYXNodGFibGU8PywgPz4gcHJvcHMpIHsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgT2JqZWN0IGdldFBpeGVscygpIHsKLSAgICAgICAgc3dpdGNoKGRhdGFUeXBlKXsKLSAgICAgICAgY2FzZSBEQVRBX1RZUEVfQllURToKLSAgICAgICAgICAgIHJldHVybiBiRGF0YTsKLSAgICAgICAgY2FzZSBEQVRBX1RZUEVfSU5UOgotICAgICAgICAgICAgcmV0dXJuIGlEYXRhOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRDb2xvck1vZGVsKENvbG9yTW9kZWwgbW9kZWwpIHsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgc3JjWCwgaW50IHNyY1ksIGludCBzcmNXLCBpbnQgc3JjSCwKLSAgICAgICAgICAgIENvbG9yTW9kZWwgbW9kZWwsIGJ5dGVbXSBwaXhlbHMsIGludCBzcmNPZmYsIGludCBzcmNTY2FuKSB7Ci0gICAgICAgIGlmKHNyY1kgPCBZKXsKLSAgICAgICAgICAgIGludCBkZWx0YSA9IFkgLSBzcmNZOwotICAgICAgICAgICAgaWYoZGVsdGEgPj0gaGVpZ2h0KSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgc3JjWSArPSBkZWx0YTsKLSAgICAgICAgICAgIHNyY0ggLT0gZGVsdGE7Ci0gICAgICAgICAgICBzcmNPZmYgKz0gc3JjU2NhbiAqIGRlbHRhOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYoc3JjWSArIHNyY0ggPiBZICsgaGVpZ2h0KXsKLSAgICAgICAgICAgIHNyY0ggPSBZICsgaGVpZ2h0IC0gc3JjWTsKLSAgICAgICAgICAgIGlmKHNyY0ggPD0gMCkgewotICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmKHNyY1ggPCBYKXsKLSAgICAgICAgICAgIGludCBkZWx0YSA9IFggLSBzcmNYOwotICAgICAgICAgICAgaWYoZGVsdGEgPj0gd2lkdGgpIHsKLSAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBzcmNXIC09IGRlbHRhOwotICAgICAgICAgICAgc3JjWCArPSBkZWx0YTsKLSAgICAgICAgICAgIHNyY09mZiArPSBkZWx0YTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmKHNyY1ggKyBzcmNXID4gWCArIHdpZHRoKXsKLSAgICAgICAgICAgIHNyY1cgPSBYICsgd2lkdGggLSBzcmNYOwotICAgICAgICAgICAgaWYoc3JjVyA8PSAwKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGlmKHNjYW5saW5lID09IDApIHsKLSAgICAgICAgICAgIHNjYW5saW5lID0gd2lkdGg7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IHJlYWxPZmYgPSBvZmZzZXQgKyAoc3JjWSAtIFkpICogc2NhbmxpbmUgKyAoc3JjWCAtIFgpOwotICAgICAgICBzd2l0Y2goZGF0YVR5cGUpewotICAgICAgICBjYXNlIERBVEFfVFlQRV9VTkRFRklORUQ6Ci0gICAgICAgICAgICBjbSA9IG1vZGVsOwotICAgICAgICAgICAgaWYobW9kZWwgIT0gQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCkpewotICAgICAgICAgICAgICAgIGJEYXRhID0gbmV3IGJ5dGVbd2lkdGggKiBoZWlnaHRdOwotICAgICAgICAgICAgICAgIGlzUkdCID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgZGF0YVR5cGUgPSBEQVRBX1RZUEVfQllURTsKLSAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgIGlEYXRhID0gbmV3IGludFt3aWR0aCAqIGhlaWdodF07Ci0gICAgICAgICAgICAgICAgaXNSR0IgPSB0cnVlOwotICAgICAgICAgICAgICAgIGRhdGFUeXBlID0gREFUQV9UWVBFX0lOVDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgY2FzZSBEQVRBX1RZUEVfQllURToKLSAgICAgICAgICAgIGlmKCFpc1JHQiAmJiBjbSA9PSBtb2RlbCl7Ci0gICAgICAgICAgICAgICAgZm9yKGludCB5ID0gMDsgeSA8IHNyY0g7IHkrKyl7Ci0gICAgICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkocGl4ZWxzLCBzcmNPZmYsIGJEYXRhLCByZWFsT2ZmLCBzcmNXKTsKLSAgICAgICAgICAgICAgICAgICAgc3JjT2ZmICs9IHNyY1NjYW47Ci0gICAgICAgICAgICAgICAgICAgIHJlYWxPZmYgKz0gc2NhbmxpbmU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZm9yY2VUb1JHQigpOwotICAgICAgICBjYXNlIERBVEFfVFlQRV9JTlQ6Ci0gICAgICAgICAgICBmb3IoaW50IHkgPSAwOyB5IDwgc3JjSDsgeSsrKXsKLSAgICAgICAgICAgICAgICBmb3IoaW50IHggPSAwOyB4IDwgc3JjVzsgeCsrKXsKLSAgICAgICAgICAgICAgICAgICAgaURhdGFbcmVhbE9mZiArIHhdID0gY20uZ2V0UkdCKHBpeGVsc1tzcmNPZmYgKyB4XSAmIDB4ZmYpOyAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHNyY09mZiArPSBzcmNTY2FuOwotICAgICAgICAgICAgICAgIHJlYWxPZmYgKz0gc2NhbmxpbmU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCBzcmNYLCBpbnQgc3JjWSwgaW50IHNyY1csIGludCBzcmNILAotICAgICAgICAgICAgQ29sb3JNb2RlbCBtb2RlbCwgaW50W10gcGl4ZWxzLCBpbnQgc3JjT2ZmLCBpbnQgc3JjU2NhbikgewotCi0gICAgICAgIGlmKHNyY1kgPCBZKXsKLSAgICAgICAgICAgIGludCBkZWx0YSA9IFkgLSBzcmNZOwotICAgICAgICAgICAgaWYoZGVsdGEgPj0gaGVpZ2h0KSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgc3JjWSArPSBkZWx0YTsKLSAgICAgICAgICAgIHNyY0ggLT0gZGVsdGE7Ci0gICAgICAgICAgICBzcmNPZmYgKz0gc3JjU2NhbiAqIGRlbHRhOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYoc3JjWSArIHNyY0ggPiBZICsgaGVpZ2h0KXsKLSAgICAgICAgICAgIHNyY0ggPSBZICsgaGVpZ2h0IC0gc3JjWTsKLSAgICAgICAgICAgIGlmKHNyY0ggPD0gMCkgewotICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmKHNyY1ggPCBYKXsKLSAgICAgICAgICAgIGludCBkZWx0YSA9IFggLSBzcmNYOwotICAgICAgICAgICAgaWYoZGVsdGEgPj0gd2lkdGgpIHsKLSAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBzcmNXIC09IGRlbHRhOwotICAgICAgICAgICAgc3JjWCArPSBkZWx0YTsKLSAgICAgICAgICAgIHNyY09mZiArPSBkZWx0YTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmKHNyY1ggKyBzcmNXID4gWCArIHdpZHRoKXsKLSAgICAgICAgICAgIHNyY1cgPSBYICsgd2lkdGggLSBzcmNYOwotICAgICAgICAgICAgaWYoc3JjVyA8PSAwKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGlmKHNjYW5saW5lID09IDApIHsKLSAgICAgICAgICAgIHNjYW5saW5lID0gd2lkdGg7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IHJlYWxPZmYgPSBvZmZzZXQgKyAoc3JjWSAtIFkpICogc2NhbmxpbmUgKyAoc3JjWCAtIFgpOwotCi0gICAgICAgIGludCBtYXNrID0gMHhGRjsKLQotICAgICAgICBzd2l0Y2goZGF0YVR5cGUpewotICAgICAgICBjYXNlIERBVEFfVFlQRV9VTkRFRklORUQ6Ci0gICAgICAgICAgICBjbSA9IG1vZGVsOwotICAgICAgICAgICAgaURhdGEgPSBuZXcgaW50W3dpZHRoICogaGVpZ2h0XTsKLSAgICAgICAgICAgIGRhdGFUeXBlID0gREFUQV9UWVBFX0lOVDsKLSAgICAgICAgICAgIGlzUkdCID0gKGNtID09IENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpKTsKLQotICAgICAgICBjYXNlIERBVEFfVFlQRV9JTlQ6Ci0gICAgICAgICAgICBpZihjbSA9PSBtb2RlbCl7Ci0gICAgICAgICAgICAgICAgZm9yKGludCB5ID0gMDsgeSA8IHNyY0g7IHkrKyl7Ci0gICAgICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkocGl4ZWxzLCBzcmNPZmYsIGlEYXRhLCByZWFsT2ZmLCBzcmNXKTsKLSAgICAgICAgICAgICAgICAgICAgc3JjT2ZmICs9IHNyY1NjYW47Ci0gICAgICAgICAgICAgICAgICAgIHJlYWxPZmYgKz0gc2NhbmxpbmU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgbWFzayA9IDB4RkZGRkZGRkY7Ci0KLSAgICAgICAgY2FzZSBEQVRBX1RZUEVfQllURToKLSAgICAgICAgICAgIGZvcmNlVG9SR0IoKTsKLSAgICAgICAgICAgIGZvcihpbnQgeSA9IDA7IHkgPCBzcmNIOyB5KyspewotICAgICAgICAgICAgICAgIGZvcihpbnQgeCA9IDA7IHggPCBzcmNXOyB4KyspewotICAgICAgICAgICAgICAgICAgICBpRGF0YVtyZWFsT2ZmK3hdID0gY20uZ2V0UkdCKHBpeGVsc1tzcmNPZmYreF0gJiBtYXNrKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgc3JjT2ZmICs9IHNyY1NjYW47Ci0gICAgICAgICAgICAgICAgcmVhbE9mZiArPSBzY2FubGluZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgQ29sb3JNb2RlbCBnZXRDb2xvck1vZGVsKCkgewotICAgICAgICByZXR1cm4gY207Ci0gICAgfQotCi0gICAgcHVibGljIHN5bmNocm9uaXplZCBib29sZWFuIGdyYWJQaXhlbHMobG9uZyBtcykgCi0gICAgdGhyb3dzIEludGVycnVwdGVkRXhjZXB0aW9uIHsKLSAgICAgICAgaWYoKGdyYWJiZXJTdGF0dXMgJiBHUkFCQklOR19TVE9QKSAhPSAwKXsKLSAgICAgICAgICAgIHJldHVybiAoKGdyYWJiZXJTdGF0dXMgJiBBTExfQklUUykgIT0gMCk7Ci0gICAgICAgIH0KLQotICAgICAgICBsb25nIHN0YXJ0ID0gU3lzdGVtLmN1cnJlbnRUaW1lTWlsbGlzKCk7Ci0KLSAgICAgICAgaWYoIWlzR3JhYmJpbmcpewotICAgICAgICAgICAgaXNHcmFiYmluZyA9IHRydWU7Ci0gICAgICAgICAgICBncmFiYmVyU3RhdHVzICY9IH5JbWFnZU9ic2VydmVyLkFCT1JUOwotICAgICAgICAgICAgcHJvZHVjZXIuc3RhcnRQcm9kdWN0aW9uKHRoaXMpOwotICAgICAgICB9Ci0gICAgICAgIHdoaWxlKChncmFiYmVyU3RhdHVzICYgR1JBQkJJTkdfU1RPUCkgPT0gMCl7Ci0gICAgICAgICAgICBpZihtcyAhPSAwKXsKLSAgICAgICAgICAgICAgICBtcyA9IHN0YXJ0ICsgbXMgLSBTeXN0ZW0uY3VycmVudFRpbWVNaWxsaXMoKTsKLSAgICAgICAgICAgICAgICBpZihtcyA8PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHdhaXQobXMpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuICgoZ3JhYmJlclN0YXR1cyAmIEFMTF9CSVRTKSAhPSAwKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXREaW1lbnNpb25zKGludCB3LCBpbnQgaCkgewotICAgICAgICBpZih3aWR0aCA8IDApIHsKLSAgICAgICAgICAgIHdpZHRoID0gdyAtIFg7Ci0gICAgICAgIH0KLSAgICAgICAgaWYoaGVpZ2h0IDwgMCkgewotICAgICAgICAgICAgaGVpZ2h0ID0gaCAtIFk7Ci0gICAgICAgIH0KLQotICAgICAgICBncmFiYmVyU3RhdHVzIHw9IEltYWdlT2JzZXJ2ZXIuV0lEVEggfCBJbWFnZU9ic2VydmVyLkhFSUdIVDsKLQotICAgICAgICBpZih3aWR0aCA8PTAgfHwgaGVpZ2h0IDw9MCl7Ci0gICAgICAgICAgICBpbWFnZUNvbXBsZXRlKFNUQVRJQ0lNQUdFRE9ORSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpZihpc1JHQiAmJiBkYXRhVHlwZSA9PSBEQVRBX1RZUEVfVU5ERUZJTkVEKXsKLSAgICAgICAgICAgIGlEYXRhID0gbmV3IGludFt3aWR0aCAqIGhlaWdodF07Ci0gICAgICAgICAgICBkYXRhVHlwZSA9IERBVEFfVFlQRV9JTlQ7Ci0gICAgICAgICAgICBzY2FubGluZSA9IHdpZHRoOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0SGludHMoaW50IGhpbnRzKSB7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgaW1hZ2VDb21wbGV0ZShpbnQgc3RhdHVzKSB7Ci0gICAgICAgIHN3aXRjaChzdGF0dXMpewotICAgICAgICBjYXNlIElNQUdFQUJPUlRFRDoKLSAgICAgICAgICAgIGdyYWJiZXJTdGF0dXMgfD0gSW1hZ2VPYnNlcnZlci5BQk9SVDsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIElNQUdFRVJST1I6Ci0gICAgICAgICAgICBncmFiYmVyU3RhdHVzIHw9IEltYWdlT2JzZXJ2ZXIuRVJST1IgfCBJbWFnZU9ic2VydmVyLkFCT1JUOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgU0lOR0xFRlJBTUVET05FOgotICAgICAgICAgICAgZ3JhYmJlclN0YXR1cyB8PSBJbWFnZU9ic2VydmVyLkZSQU1FQklUUzsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFNUQVRJQ0lNQUdFRE9ORToKLSAgICAgICAgICAgIGdyYWJiZXJTdGF0dXMgfD0gSW1hZ2VPYnNlcnZlci5BTExCSVRTOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAvLyBhd3QuMjZBPUluY29ycmVjdCBJbWFnZUNvbnN1bWVyIGNvbXBsZXRpb24gc3RhdHVzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI2QSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGlzR3JhYmJpbmcgPSBmYWxzZTsKLSAgICAgICAgcHJvZHVjZXIucmVtb3ZlQ29uc3VtZXIodGhpcyk7Ci0gICAgICAgIG5vdGlmeUFsbCgpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGdyYWJQaXhlbHMoKSB0aHJvd3MgSW50ZXJydXB0ZWRFeGNlcHRpb24gewotICAgICAgICByZXR1cm4gZ3JhYlBpeGVscygwKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgc3RhcnRHcmFiYmluZygpIHsKLSAgICAgICAgaWYoKGdyYWJiZXJTdGF0dXMgJiBHUkFCQklOR19TVE9QKSAhPSAwKXsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBpZighaXNHcmFiYmluZyl7Ci0gICAgICAgICAgICBpc0dyYWJiaW5nID0gdHJ1ZTsKLSAgICAgICAgICAgIGdyYWJiZXJTdGF0dXMgJj0gfkltYWdlT2JzZXJ2ZXIuQUJPUlQ7Ci0gICAgICAgICAgICBwcm9kdWNlci5zdGFydFByb2R1Y3Rpb24odGhpcyk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgYWJvcnRHcmFiYmluZygpIHsKLSAgICAgICAgaW1hZ2VDb21wbGV0ZShJTUFHRUFCT1JURUQpOwotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgaW50IHN0YXR1cygpIHsKLSAgICAgICAgcmV0dXJuIGdyYWJiZXJTdGF0dXM7Ci0gICAgfQotCi0gICAgcHVibGljIHN5bmNocm9uaXplZCBpbnQgZ2V0V2lkdGgoKSB7Ci0gICAgICAgIGlmKHdpZHRoIDwgMCkgewotICAgICAgICAgICAgcmV0dXJuIC0xOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB3aWR0aDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIGludCBnZXRTdGF0dXMoKSB7Ci0gICAgICAgIHJldHVybiBncmFiYmVyU3RhdHVzOwotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgaW50IGdldEhlaWdodCgpIHsKLSAgICAgICAgaWYoaGVpZ2h0IDwgMCkgewotICAgICAgICAgICAgcmV0dXJuIC0xOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBoZWlnaHQ7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIGluaXRpYWxpemUoSW1hZ2VQcm9kdWNlciBpcCwgaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsCi0gICAgICAgICAgICBpbnQgcGl4ZWxzW10sIGludCBvZmYsIGludCBzY2Fuc2l6ZSwgYm9vbGVhbiBmb3JjZVJHQil7Ci0KLSAgICAgICAgcHJvZHVjZXIgPSBpcDsKLSAgICAgICAgWCA9IHg7Ci0gICAgICAgIFkgPSB5OwotICAgICAgICB3aWR0aCA9IHc7Ci0gICAgICAgIGhlaWdodCA9IGg7Ci0gICAgICAgIGlEYXRhID0gcGl4ZWxzOwotICAgICAgICBkYXRhVHlwZSA9IChwaXhlbHMgPT0gbnVsbCkgPyBEQVRBX1RZUEVfVU5ERUZJTkVEIDogREFUQV9UWVBFX0lOVDsKLSAgICAgICAgb2Zmc2V0ID0gb2ZmOwotICAgICAgICBzY2FubGluZSA9IHNjYW5zaXplOwotICAgICAgICBpZihmb3JjZVJHQil7Ci0gICAgICAgICAgICBjbSA9IENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpOwotICAgICAgICAgICAgaXNSR0IgPSB0cnVlOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRm9yY2UgcGl4ZWxzIHRvIElOVCBSR0IgbW9kZQotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBmb3JjZVRvUkdCKCl7Ci0gICAgICAgIGlmIChpc1JHQikKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAKLSAgICAgICAgc3dpdGNoKGRhdGFUeXBlKXsKLSAgICAgICAgY2FzZSBEQVRBX1RZUEVfQllURToKLSAgICAgICAgICAgIGlEYXRhID0gbmV3IGludFt3aWR0aCAqIGhlaWdodF07Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgaURhdGEubGVuZ3RoOyBpKyspewotICAgICAgICAgICAgICAgIGlEYXRhW2ldID0gY20uZ2V0UkdCKGJEYXRhW2ldICYgMHhmZik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkYXRhVHlwZSA9IERBVEFfVFlQRV9JTlQ7Ci0gICAgICAgICAgICBiRGF0YSA9IG51bGw7Ci0gICAgICAgICAgICBicmVhazsKLQotICAgICAgICBjYXNlIERBVEFfVFlQRV9JTlQ6Ci0gICAgICAgICAgICBpbnQgYnVmZltdID0gbmV3IGludFt3aWR0aCAqIGhlaWdodF07Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgaURhdGEubGVuZ3RoOyBpKyspewotICAgICAgICAgICAgICAgIGJ1ZmZbaV0gPSBjbS5nZXRSR0IoaURhdGFbaV0pOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaURhdGEgPSBidWZmOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgb2Zmc2V0ID0gMDsKLSAgICAgICAgc2NhbmxpbmUgPSB3aWR0aDsKLSAgICAgICAgY20gPSBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKTsKLSAgICAgICAgaXNSR0IgPSB0cnVlOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL1BpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbC5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL1BpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4ZTY0NmY4Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9QaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEzNCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwgY2xhc3MgcmVwcmVzZW50cyBpbWFnZSBkYXRhIGFzIHJlcHJlc2VudGVkIGFzCi0gKiBpbnRlcmxlYXZlZCBwaXhlbHMgYW5kIGZvciB3aGljaCBlYWNoIHNhbXBsZSBvZiBhIHBpeGVsIHRha2VzIG9uZSBkYXRhCi0gKiBlbGVtZW50IG9mIHRoZSBEYXRhQnVmZmVyLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbCBleHRlbmRzIENvbXBvbmVudFNhbXBsZU1vZGVsIHsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICogcGFyYW1ldGVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUgb2YgdGhlIHNhbXBsZXMuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gcGl4ZWxTdHJpZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbCBzdHJpZGUgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIHNjYW5saW5lU3RyaWRlCi0gICAgICogICAgICAgICAgICB0aGUgc2NhbmxpbmUgc3RyaWRlIG9mIHRoZSBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gYmFuZE9mZnNldHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiB0aGUgYmFuZCBvZmZzZXRzLgotICAgICAqLwotICAgIHB1YmxpYyBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwoaW50IGRhdGFUeXBlLCBpbnQgdywgaW50IGgsIGludCBwaXhlbFN0cmlkZSwKLSAgICAgICAgICAgIGludCBzY2FubGluZVN0cmlkZSwgaW50IGJhbmRPZmZzZXRzW10pIHsKLQotICAgICAgICBzdXBlcihkYXRhVHlwZSwgdywgaCwgcGl4ZWxTdHJpZGUsIHNjYW5saW5lU3RyaWRlLCBiYW5kT2Zmc2V0cyk7Ci0KLSAgICAgICAgaW50IG1heE9mZnNldCA9IGJhbmRPZmZzZXRzWzBdOwotICAgICAgICBpbnQgbWluT2Zmc2V0ID0gYmFuZE9mZnNldHNbMF07Ci0gICAgICAgIGZvciAoaW50IGkgPSAxOyBpIDwgYmFuZE9mZnNldHMubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIGlmIChiYW5kT2Zmc2V0c1tpXSA+IG1heE9mZnNldCkgewotICAgICAgICAgICAgICAgIG1heE9mZnNldCA9IGJhbmRPZmZzZXRzW2ldOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGJhbmRPZmZzZXRzW2ldIDwgbWluT2Zmc2V0KSB7Ci0gICAgICAgICAgICAgICAgbWluT2Zmc2V0ID0gYmFuZE9mZnNldHNbaV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBtYXhPZmZzZXQgLT0gbWluT2Zmc2V0OwotCi0gICAgICAgIGlmIChtYXhPZmZzZXQgPiBzY2FubGluZVN0cmlkZSkgewotICAgICAgICAgICAgLy8gYXd0LjI0MT1Bbnkgb2Zmc2V0IGJldHdlZW4gYmFuZHMgaXMgZ3JlYXRlciB0aGFuIHRoZSBTY2FubGluZQotICAgICAgICAgICAgLy8gc3RyaWRlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI0MSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG1heE9mZnNldCA+IHBpeGVsU3RyaWRlKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjQyPVBpeGVsIHN0cmlkZSBpcyBsZXNzIHRoYW4gYW55IG9mZnNldCBiZXR3ZWVuIGJhbmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI0MiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHBpeGVsU3RyaWRlICogdyA+IHNjYW5saW5lU3RyaWRlKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjQzPVByb2R1Y3Qgb2YgUGl4ZWwgc3RyaWRlIGFuZCB3IGlzIGdyZWF0ZXIgdGhhbiBTY2FubGluZQotICAgICAgICAgICAgLy8gc3RyaWRlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI0MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU2FtcGxlTW9kZWwgY3JlYXRlU3Vic2V0U2FtcGxlTW9kZWwoaW50IGJhbmRzW10pIHsKLSAgICAgICAgaW50IG5ld09mZnNldHNbXSA9IG5ldyBpbnRbYmFuZHMubGVuZ3RoXTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBiYW5kcy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgbmV3T2Zmc2V0c1tpXSA9IGJhbmRPZmZzZXRzW2JhbmRzW2ldXTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBuZXcgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsKGRhdGFUeXBlLCB3aWR0aCwgaGVpZ2h0LCBwaXhlbFN0cmlkZSwKLSAgICAgICAgICAgICAgICBzY2FubGluZVN0cmlkZSwgbmV3T2Zmc2V0cyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFNhbXBsZU1vZGVsIGNyZWF0ZUNvbXBhdGlibGVTYW1wbGVNb2RlbChpbnQgdywgaW50IGgpIHsKLSAgICAgICAgaW50IG5ld09mZnNldHNbXTsKLSAgICAgICAgaW50IG1pbk9mZnNldCA9IGJhbmRPZmZzZXRzWzBdOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAxOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgaWYgKGJhbmRPZmZzZXRzW2ldIDwgbWluT2Zmc2V0KSB7Ci0gICAgICAgICAgICAgICAgbWluT2Zmc2V0ID0gYmFuZE9mZnNldHNbaV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAobWluT2Zmc2V0ID4gMCkgewotICAgICAgICAgICAgbmV3T2Zmc2V0cyA9IG5ldyBpbnRbbnVtQmFuZHNdOwotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1CYW5kczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgbmV3T2Zmc2V0c1tpXSA9IGJhbmRPZmZzZXRzW2ldIC0gbWluT2Zmc2V0OwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbmV3T2Zmc2V0cyA9IGJhbmRPZmZzZXRzOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwoZGF0YVR5cGUsIHcsIGgsIHBpeGVsU3RyaWRlLCBwaXhlbFN0cmlkZSAqIHcsCi0gICAgICAgICAgICAgICAgbmV3T2Zmc2V0cyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgaW50IGhhc2ggPSBzdXBlci5oYXNoQ29kZSgpOwotICAgICAgICBpbnQgdG1wID0gaGFzaCA+Pj4gODsKLSAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgaGFzaCB8PSB0bXA7Ci0KLSAgICAgICAgcmV0dXJuIGhhc2ggXiAweDY2OwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL1JHQkltYWdlRmlsdGVyLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvUkdCSW1hZ2VGaWx0ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZjVmZTVkOS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvUkdCSW1hZ2VGaWx0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE5NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotLyoqCi0gKiBUaGUgUkdCSW1hZ2VGaWx0ZXIgY2xhc3MgcmVwcmVzZW50cyBhIGZpbHRlciB3aGljaCBtb2RpZmllcyBwaXhlbHMgb2YgYW4KLSAqIGltYWdlIGluIHRoZSBkZWZhdWx0IFJHQiBDb2xvck1vZGVsLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIFJHQkltYWdlRmlsdGVyIGV4dGVuZHMgSW1hZ2VGaWx0ZXIgewotCi0gICAgLyoqCi0gICAgICogVGhlIG9yaWdpbmFsIG1vZGVsIGlzIHRoZSBDb2xvck1vZGVsIHRvIGJlIHJlcGxhY2VkIGJ5IHRoZSBuZXcgbW9kZWwgd2hlbgotICAgICAqIHN1YnN0aXR1dGVDb2xvck1vZGVsIGlzIGNhbGxlZC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgQ29sb3JNb2RlbCBvcmlnbW9kZWw7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbmV3IG1vZGVsIGlzIHRoZSBDb2xvck1vZGVsIHdpdGggd2hpY2ggdG8gcmVwbGFjZSB0aGUgb3JpZ2luYWwgbW9kZWwKLSAgICAgKiB3aGVuIHN1YnN0aXR1dGVDb2xvck1vZGVsIGlzIGNhbGxlZC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgQ29sb3JNb2RlbCBuZXdtb2RlbDsKLQotICAgIC8qKgotICAgICAqIFRoZSBjYW5GaWx0ZXJJbmRleENvbG9yTW9kZWwgaW5kaWNhdGVzIGlmIGl0IGlzIGFjY2VwdGFibGUgdG8gYXBwbHkgdGhlCi0gICAgICogY29sb3IgZmlsdGVyaW5nIG9mIHRoZSBmaWx0ZXJSR0IgbWV0aG9kIHRvIHRoZSBjb2xvciB0YWJsZSBlbnRyaWVzIG9mIGFuCi0gICAgICogSW5kZXhDb2xvck1vZGVsIG9iamVjdC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgYm9vbGVhbiBjYW5GaWx0ZXJJbmRleENvbG9yTW9kZWw7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgUkdCSW1hZ2VGaWx0ZXIuCi0gICAgICovCi0gICAgcHVibGljIFJHQkltYWdlRmlsdGVyKCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbHRlcnMgYW4gSW5kZXhDb2xvck1vZGVsIG9iamVjdCBieSBjYWxsaW5nIGZpbHRlclJHQiBmdW5jdGlvbiBmb3IgZWFjaAotICAgICAqIGVudHJ5IG9mIEluZGV4Q29sb3JNb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWNtCi0gICAgICogICAgICAgICAgICB0aGUgSW5kZXhDb2xvck1vZGVsIHRvIGJlIGZpbHRlcmVkLgotICAgICAqIEByZXR1cm4gdGhlIEluZGV4Q29sb3JNb2RlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW5kZXhDb2xvck1vZGVsIGZpbHRlckluZGV4Q29sb3JNb2RlbChJbmRleENvbG9yTW9kZWwgaWNtKSB7Ci0gICAgICAgIGludCB0cmFuc2ZlclR5cGUgPSBpY20uZ2V0VHJhbnNmZXJUeXBlKCk7Ci0gICAgICAgIGludCBiaXRzID0gaWNtLmdldFBpeGVsU2l6ZSgpOwotICAgICAgICBpbnQgbWFwU2l6ZSA9IGljbS5nZXRNYXBTaXplKCk7Ci0gICAgICAgIGludCBjb2xvck1hcFtdID0gbmV3IGludFttYXBTaXplXTsKLSAgICAgICAgaW50IGZpbHRlcmVkQ29sb3JNYXBbXSA9IG5ldyBpbnRbbWFwU2l6ZV07Ci0gICAgICAgIGljbS5nZXRSR0JzKGNvbG9yTWFwKTsKLSAgICAgICAgaW50IHRyYW5zID0gLTE7Ci0gICAgICAgIGJvb2xlYW4gaGFzQWxwaGEgPSBmYWxzZTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBtYXBTaXplOyBpKyspIHsKLSAgICAgICAgICAgIGZpbHRlcmVkQ29sb3JNYXBbaV0gPSBmaWx0ZXJSR0IoLTEsIC0xLCBjb2xvck1hcFtpXSk7Ci0gICAgICAgICAgICBpbnQgYWxwaGEgPSBmaWx0ZXJlZENvbG9yTWFwW2ldID4+PiAyNDsKLSAgICAgICAgICAgIGlmIChhbHBoYSAhPSAweGZmKSB7Ci0gICAgICAgICAgICAgICAgaWYgKCFoYXNBbHBoYSkgewotICAgICAgICAgICAgICAgICAgICBoYXNBbHBoYSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChhbHBoYSA9PSAwICYmIHRyYW5zIDwgMCkgewotICAgICAgICAgICAgICAgICAgICB0cmFucyA9IGk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBJbmRleENvbG9yTW9kZWwoYml0cywgbWFwU2l6ZSwgZmlsdGVyZWRDb2xvck1hcCwgMCwgaGFzQWxwaGEsIHRyYW5zLAotICAgICAgICAgICAgICAgIHRyYW5zZmVyVHlwZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVwbGFjZXMgdGhlIG9yaWdpbmFsIGNvbG9yIG1vZGVsIGFuZCB0aGUgbmV3IG9uZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb2xkY20KLSAgICAgKiAgICAgICAgICAgIHRoZSBvbGQgQ29sb3JNb2RlbC4KLSAgICAgKiBAcGFyYW0gbmV3Y20KLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgQ29sb3JNb2RlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzdWJzdGl0dXRlQ29sb3JNb2RlbChDb2xvck1vZGVsIG9sZGNtLCBDb2xvck1vZGVsIG5ld2NtKSB7Ci0gICAgICAgIG9yaWdtb2RlbCA9IG9sZGNtOwotICAgICAgICBuZXdtb2RlbCA9IG5ld2NtOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldENvbG9yTW9kZWwoQ29sb3JNb2RlbCBtb2RlbCkgewotICAgICAgICBpZiAobW9kZWwgaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwgJiYgY2FuRmlsdGVySW5kZXhDb2xvck1vZGVsKSB7Ci0gICAgICAgICAgICBJbmRleENvbG9yTW9kZWwgaWNtID0gKEluZGV4Q29sb3JNb2RlbCltb2RlbDsKLSAgICAgICAgICAgIENvbG9yTW9kZWwgZmlsdGVyZWRNb2RlbCA9IGZpbHRlckluZGV4Q29sb3JNb2RlbChpY20pOwotICAgICAgICAgICAgc3Vic3RpdHV0ZUNvbG9yTW9kZWwobW9kZWwsIGZpbHRlcmVkTW9kZWwpOwotICAgICAgICAgICAgY29uc3VtZXIuc2V0Q29sb3JNb2RlbChmaWx0ZXJlZE1vZGVsKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbnN1bWVyLnNldENvbG9yTW9kZWwoQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCkpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBDb2xvck1vZGVsIG1vZGVsLCBpbnRbXSBwaXhlbHMsIGludCBvZmYsCi0gICAgICAgICAgICBpbnQgc2NhbnNpemUpIHsKLQotICAgICAgICBpZiAobW9kZWwgPT0gbnVsbCB8fCBtb2RlbCA9PSBvcmlnbW9kZWwpIHsKLSAgICAgICAgICAgIGNvbnN1bWVyLnNldFBpeGVscyh4LCB5LCB3LCBoLCBuZXdtb2RlbCwgcGl4ZWxzLCBvZmYsIHNjYW5zaXplKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGludCByZ2JQaXhlbHNbXSA9IG5ldyBpbnRbd107Ci0gICAgICAgICAgICBmb3IgKGludCBzeSA9IHksIHBpeGVsc09mZiA9IG9mZjsgc3kgPCB5ICsgaDsgc3krKywgcGl4ZWxzT2ZmICs9IHNjYW5zaXplKSB7Ci0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBzeCA9IHgsIGlkeCA9IDA7IHN4IDwgeCArIHc7IHN4KyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJnYlBpeGVsc1tpZHhdID0gbW9kZWwuZ2V0UkdCKHBpeGVsc1twaXhlbHNPZmYgKyBpZHhdKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgZmlsdGVyUkdCUGl4ZWxzKHgsIHN5LCB3LCAxLCByZ2JQaXhlbHMsIDAsIHcpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBDb2xvck1vZGVsIG1vZGVsLCBieXRlW10gcGl4ZWxzLCBpbnQgb2ZmLAotICAgICAgICAgICAgaW50IHNjYW5zaXplKSB7Ci0KLSAgICAgICAgaWYgKG1vZGVsID09IG51bGwgfHwgbW9kZWwgPT0gb3JpZ21vZGVsKSB7Ci0gICAgICAgICAgICBjb25zdW1lci5zZXRQaXhlbHMoeCwgeSwgdywgaCwgbmV3bW9kZWwsIHBpeGVscywgb2ZmLCBzY2Fuc2l6ZSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpbnQgcmdiUGl4ZWxzW10gPSBuZXcgaW50W3ddOwotICAgICAgICAgICAgZm9yIChpbnQgc3kgPSB5LCBwaXhlbHNPZmYgPSBvZmY7IHN5IDwgeSArIGg7IHN5KyssIHBpeGVsc09mZiArPSBzY2Fuc2l6ZSkgewotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgc3ggPSB4LCBpZHggPSAwOyBzeCA8IHggKyB3OyBzeCsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICByZ2JQaXhlbHNbaWR4XSA9IG1vZGVsLmdldFJHQihwaXhlbHNbcGl4ZWxzT2ZmICsgaWR4XSAmIDB4ZmYpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBmaWx0ZXJSR0JQaXhlbHMoeCwgc3ksIHcsIDEsIHJnYlBpeGVscywgMCwgdyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaWx0ZXJzIGEgcmVnaW9uIG9mIHBpeGVscyBpbiB0aGUgZGVmYXVsdCBSR0IgQ29sb3JNb2RlbCBieSBjYWxsaW5nIHRoZQotICAgICAqIGZpbHRlclJHQiBtZXRob2QgZm9yIHRoZW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgcmVnaW9uLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHJlZ2lvbi4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlZ2lvbi4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiByZWdpb24uCi0gICAgICogQHBhcmFtIHBpeGVscwotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVscyBhcnJheS4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IG9mIGFycmF5LgotICAgICAqIEBwYXJhbSBzY2Fuc2l6ZQotICAgICAqICAgICAgICAgICAgdGhlIGRpc3RhbmNlIGJldHdlZW4gcm93cyBvZiBwaXhlbHMgaW4gdGhlIGFycmF5LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGZpbHRlclJHQlBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50W10gcGl4ZWxzLCBpbnQgb2ZmLCBpbnQgc2NhbnNpemUpIHsKLQotICAgICAgICBmb3IgKGludCBzeSA9IHksIGxpbmVPZmYgPSBvZmY7IHN5IDwgeSArIGg7IHN5KyssIGxpbmVPZmYgKz0gc2NhbnNpemUpIHsKLSAgICAgICAgICAgIGZvciAoaW50IHN4ID0geCwgaWR4ID0gMDsgc3ggPCB4ICsgdzsgc3grKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICBwaXhlbHNbbGluZU9mZiArIGlkeF0gPSBmaWx0ZXJSR0Ioc3gsIHN5LCBwaXhlbHNbbGluZU9mZiArIGlkeF0pOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGNvbnN1bWVyLnNldFBpeGVscyh4LCB5LCB3LCBoLCBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKSwgcGl4ZWxzLCBvZmYsIHNjYW5zaXplKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb252ZXJ0cyBhIHNpbmdsZSBpbnB1dCBwaXhlbCBpbiB0aGUgZGVmYXVsdCBSR0IgQ29sb3JNb2RlbCB0byBhIHNpbmdsZQotICAgICAqIG91dHB1dCBwaXhlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggcGl4ZWwncyBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBwaXhlbCdzIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHJnYgotICAgICAqICAgICAgICAgICAgYSBwaXhlbCBpbiB0aGUgZGVmYXVsdCBSR0IgY29sb3IgbW9kZWwuCi0gICAgICogQHJldHVybiBhIGZpbHRlcmVkIHBpeGVsIGluIHRoZSBkZWZhdWx0IFJHQiBjb2xvciBtb2RlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgaW50IGZpbHRlclJHQihpbnQgeCwgaW50IHksIGludCByZ2IpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvUmFzdGVyLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvUmFzdGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDY3NDlmZGUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL1Jhc3Rlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTUxNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEuYXd0LlBvaW50OwotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2UuT3JkaW5hcnlXcml0YWJsZVJhc3RlcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBUaGUgUmFzdGVyIGNsYXNzIHJlcHJlc2VudHMgYSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4gVGhpcyBjbGFzcyBpcwotICogZGVmaW5lZCBieSBEYXRhQnVmZmVyIGFuZCBTYW1wbGVNb2RlbCBvYmplY3RzLiBUaGUgRGF0YUJ1ZmZlciBvYmplY3Qgc3RvcmVzCi0gKiBzYW1wbGUgdmFsdWVzIGFuZCBEU2FtcGxlTW9kZWwgZGVmaW5lcyB0aGUgbG9jYXRpb24gb2Ygc2FtcGxlIGluIHRoaXMKLSAqIERhdGFCdWZmZXIuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgUmFzdGVyIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBEYXRhQnVmZmVyIG9mIHRoaXMgUmFzdGVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBEYXRhQnVmZmVyIGRhdGFCdWZmZXI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaGVpZ2h0IG9mIHRoaXMgUmFzdGVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgaGVpZ2h0OwotCi0gICAgLyoqCi0gICAgICogVGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBwaXhlbCBpbiB0aGlzIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IG1pblg7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgWSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IHBpeGVsIGluIHRoaXMgUmFzdGVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgbWluWTsKLQotICAgIC8qKgotICAgICAqIFRoZSBudW1iZXIgb2YgYmFuZHMgaW4gdGhpcyBSYXN0ZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBudW1CYW5kczsKLQotICAgIC8qKgotICAgICAqIFRoZSBudW1iZXIgb2YgZGF0YSBlbGVtZW50cy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IG51bURhdGFFbGVtZW50czsKLQotICAgIC8qKgotICAgICAqIFRoZSBwYXJlbnQgb2YgdGhpcyBSYXN0ZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFJhc3RlciBwYXJlbnQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgU2FtcGxlTW9kZWwgb2YgdGhpcyBSYXN0ZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsOwotCi0gICAgLyoqCi0gICAgICogVGhlIFggdHJhbnNsYXRpb24gZnJvbSB0aGUgY29vcmRpbmF0ZSBzcGFjZSBvZiB0aGUgU2FtcGxlTW9kZWwgb2YgdGhpcwotICAgICAqIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IHNhbXBsZU1vZGVsVHJhbnNsYXRlWDsKLQotICAgIC8qKgotICAgICAqIFRoZSBZIHRyYW5zbGF0aW9uIGZyb20gdGhlIGNvb3JkaW5hdGUgc3BhY2Ugb2YgdGhlIFNhbXBsZU1vZGVsIG9mIHRoaXMKLSAgICAgKiBSYXN0ZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBzYW1wbGVNb2RlbFRyYW5zbGF0ZVk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgd2lkdGggb2YgdGhpcyBSYXN0ZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCB3aWR0aDsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBSYXN0ZXIgb2JqZWN0IHdpdGggYSBCYW5kZWRTYW1wbGVNb2RlbCBhbmQgdGhlIHNwZWNpZmllZAotICAgICAqIERhdGFCdWZmZXIuIFRoZSBudW1iZXIgb2YgYmFuZHMgaXMgZGVmaW5lZCBieSB0aGUgbGVuZ3RoIG9mIGJhbmRPZmZzZXRzCi0gICAgICogb3IgYmFua0luZGljZXMgYXJyYXlzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQnVmZmVyCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIERhdGFCdWZmZXIuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gc2NhbmxpbmVTdHJpZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FubGluZSBzdHJpZGUgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGJhbmtJbmRpY2VzCi0gICAgICogICAgICAgICAgICB0aGUgYmFuayBpbmRpY2VzIG9mIGJhbmRzLgotICAgICAqIEBwYXJhbSBiYW5kT2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIGJhbmQgb2Zmc2V0cyBvZiBiYW5kcy4KLSAgICAgKiBAcGFyYW0gbG9jYXRpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBsb2NhdGlvbiB3aGljaCBkZWZpbmVzIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiBSYXN0ZXIuCi0gICAgICogQHJldHVybiB0aGUgV3JpdGFibGVSYXN0ZXIgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlQmFuZGVkUmFzdGVyKERhdGFCdWZmZXIgZGF0YUJ1ZmZlciwgaW50IHcsIGludCBoLAotICAgICAgICAgICAgaW50IHNjYW5saW5lU3RyaWRlLCBpbnQgYmFua0luZGljZXNbXSwgaW50IGJhbmRPZmZzZXRzW10sIFBvaW50IGxvY2F0aW9uKSB7Ci0KLSAgICAgICAgaWYgKHcgPD0gMCB8fCBoIDw9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMkU9dyBvciBoIGlzIGxlc3MgdGhhbiBvciBlcXVhbCB0byB6ZXJvCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIyRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGxvY2F0aW9uID09IG51bGwpIHsKLSAgICAgICAgICAgIGxvY2F0aW9uID0gbmV3IFBvaW50KDAsIDApOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChsb25nKWxvY2F0aW9uLnggKyB3ID4gSW50ZWdlci5NQVhfVkFMVUUgfHwgKGxvbmcpbG9jYXRpb24ueSArIGggPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI3Nj1sb2NhdGlvbi54ICsgdyBvciBsb2NhdGlvbi55ICsgaCByZXN1bHRzIGluIGludGVnZXIKLSAgICAgICAgICAgIC8vIG92ZXJmbG93Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3NiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGJhbmtJbmRpY2VzID09IG51bGwgfHwgYmFuZE9mZnNldHMgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjI3Nz1iYW5rSW5kaWNlcyBvciBiYW5kT2Zmc2V0cyBpcyBudWxsCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3NyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRhdGFCdWZmZXIgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjI3OD1kYXRhQnVmZmVyIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNzgiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGludCBkYXRhVHlwZSA9IGRhdGFCdWZmZXIuZ2V0RGF0YVR5cGUoKTsKLQotICAgICAgICBpZiAoZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYgZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVAotICAgICAgICAgICAgICAgICYmIGRhdGFUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9JTlQpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMzA9ZGF0YVR5cGUgaXMgbm90IG9uZSBvZiB0aGUgc3VwcG9ydGVkIGRhdGEgdHlwZXMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjMwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBCYW5kZWRTYW1wbGVNb2RlbCBzYW1wbGVNb2RlbCA9IG5ldyBCYW5kZWRTYW1wbGVNb2RlbChkYXRhVHlwZSwgdywgaCwgc2NhbmxpbmVTdHJpZGUsCi0gICAgICAgICAgICAgICAgYmFua0luZGljZXMsIGJhbmRPZmZzZXRzKTsKLQotICAgICAgICByZXR1cm4gbmV3IE9yZGluYXJ5V3JpdGFibGVSYXN0ZXIoc2FtcGxlTW9kZWwsIGRhdGFCdWZmZXIsIGxvY2F0aW9uKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgUmFzdGVyIG9iamVjdCB3aXRoIGEgQmFuZGVkU2FtcGxlTW9kZWwgYW5kIHRoZSBzcGVjaWZpZWQgZGF0YQotICAgICAqIHR5cGUuIFRoZSBEYXRhIHR5cGUgY2FuIGJlIG9uZSBvZiB0aGUgZm9sbG93aW5nIHZhbHVlczogVFlQRV9CWVRFLAotICAgICAqIFRZUEVfVVNIT1JULCBvciBUWVBFX0lOVC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUgb2YgdGhlIHNhbXBsZXM6IFRZUEVfQllURSwgVFlQRV9VU0hPUlQsIG9yCi0gICAgICogICAgICAgICAgICBUWVBFX0lOVC4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBzY2FubGluZVN0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIHNjYW5saW5lIHN0cmlkZSBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gYmFua0luZGljZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBiYW5rIGluZGljZXMgb2YgYmFuZHMuCi0gICAgICogQHBhcmFtIGJhbmRPZmZzZXRzCi0gICAgICogICAgICAgICAgICB0aGUgYmFuZCBvZmZzZXRzIG9mIGJhbmRzLgotICAgICAqIEBwYXJhbSBsb2NhdGlvbgotICAgICAqICAgICAgICAgICAgdGhlIGxvY2F0aW9uIHdoaWNoIGRlZmluZXMgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZQotICAgICAqICAgICAgICAgICAgUmFzdGVyLgotICAgICAqIEByZXR1cm4gdGhlIFdyaXRhYmxlUmFzdGVyIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZUJhbmRlZFJhc3RlcihpbnQgZGF0YVR5cGUsIGludCB3LCBpbnQgaCwgaW50IHNjYW5saW5lU3RyaWRlLAotICAgICAgICAgICAgaW50IGJhbmtJbmRpY2VzW10sIGludCBiYW5kT2Zmc2V0c1tdLCBQb2ludCBsb2NhdGlvbikgewotCi0gICAgICAgIGlmICh3IDw9IDAgfHwgaCA8PSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjJFPXcgb3IgaCBpcyBsZXNzIHRoYW4gb3IgZXF1YWwgdG8gemVybwotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMkUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChsb2NhdGlvbiA9PSBudWxsKSB7Ci0gICAgICAgICAgICBsb2NhdGlvbiA9IG5ldyBQb2ludCgwLCAwKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICgobG9uZylsb2NhdGlvbi54ICsgdyA+IEludGVnZXIuTUFYX1ZBTFVFIHx8IChsb25nKWxvY2F0aW9uLnkgKyBoID4gSW50ZWdlci5NQVhfVkFMVUUpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNzY9bG9jYXRpb24ueCArIHcgb3IgbG9jYXRpb24ueSArIGggcmVzdWx0cyBpbiBpbnRlZ2VyCi0gICAgICAgICAgICAvLyBvdmVyZmxvdwotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNzYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChiYW5rSW5kaWNlcyA9PSBudWxsIHx8IGJhbmRPZmZzZXRzID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNzc9YmFua0luZGljZXMgb3IgYmFuZE9mZnNldHMgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNzciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfQllURSAmJiBkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUCi0gICAgICAgICAgICAgICAgJiYgZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX0lOVCkgewotICAgICAgICAgICAgLy8gYXd0LjIzMD1kYXRhVHlwZSBpcyBub3Qgb25lIG9mIHRoZSBzdXBwb3J0ZWQgZGF0YSB0eXBlcwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMzAiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGludCBtYXhPZmZzZXQgPSBiYW5kT2Zmc2V0c1swXTsKLSAgICAgICAgaW50IG1heEJhbmsgPSBiYW5rSW5kaWNlc1swXTsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGJhbmtJbmRpY2VzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBpZiAoYmFuZE9mZnNldHNbaV0gPiBtYXhPZmZzZXQpIHsKLSAgICAgICAgICAgICAgICBtYXhPZmZzZXQgPSBiYW5kT2Zmc2V0c1tpXTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChiYW5rSW5kaWNlc1tpXSA+IG1heEJhbmspIHsKLSAgICAgICAgICAgICAgICBtYXhCYW5rID0gYmFua0luZGljZXNbaV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgbnVtQmFua3MgPSBtYXhCYW5rICsgMTsKLSAgICAgICAgaW50IGRhdGFTaXplID0gc2NhbmxpbmVTdHJpZGUgKiAoaCAtIDEpICsgdyArIG1heE9mZnNldDsKLQotICAgICAgICBEYXRhQnVmZmVyIGRhdGEgPSBudWxsOwotCi0gICAgICAgIHN3aXRjaCAoZGF0YVR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgZGF0YSA9IG5ldyBEYXRhQnVmZmVyQnl0ZShkYXRhU2l6ZSwgbnVtQmFua3MpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlclVTaG9ydChkYXRhU2l6ZSwgbnVtQmFua3MpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlckludChkYXRhU2l6ZSwgbnVtQmFua3MpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBjcmVhdGVCYW5kZWRSYXN0ZXIoZGF0YSwgdywgaCwgc2NhbmxpbmVTdHJpZGUsIGJhbmtJbmRpY2VzLCBiYW5kT2Zmc2V0cywgbG9jYXRpb24pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBSYXN0ZXIgb2JqZWN0IHdpdGggYSBCYW5kZWRTYW1wbGVNb2RlbCBhbmQgdGhlIHNwZWNpZmllZCBkYXRhCi0gICAgICogdHlwZS4gVGhlIERhdGEgdHlwZSBjYW4gYmUgb25lIG9mIHRoZSBmb2xsb3dpbmcgdmFsdWVzOiBUWVBFX0JZVEUsCi0gICAgICogVFlQRV9VU0hPUlQsIG9yIFRZUEVfSU5ULgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgdHlwZSBvZiB0aGUgc2FtcGxlczogVFlQRV9CWVRFLCBUWVBFX1VTSE9SVCwgb3IKLSAgICAgKiAgICAgICAgICAgIFRZUEVfSU5ULgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGJhbmRzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGJhbmRzLgotICAgICAqIEBwYXJhbSBsb2NhdGlvbgotICAgICAqICAgICAgICAgICAgdGhlIGxvY2F0aW9uIHdoaWNoIGRlZmluZXMgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZQotICAgICAqICAgICAgICAgICAgUmFzdGVyLgotICAgICAqIEByZXR1cm4gdGhlIFdyaXRhYmxlUmFzdGVyIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZUJhbmRlZFJhc3RlcihpbnQgZGF0YVR5cGUsIGludCB3LCBpbnQgaCwgaW50IGJhbmRzLAotICAgICAgICAgICAgUG9pbnQgbG9jYXRpb24pIHsKLQotICAgICAgICBpZiAodyA8PSAwIHx8IGggPD0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjIyRT13IG9yIGggaXMgbGVzcyB0aGFuIG9yIGVxdWFsIHRvIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjJFIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAobG9jYXRpb24gPT0gbnVsbCkgewotICAgICAgICAgICAgbG9jYXRpb24gPSBuZXcgUG9pbnQoMCwgMCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoKGxvbmcpbG9jYXRpb24ueCArIHcgPiBJbnRlZ2VyLk1BWF9WQUxVRSB8fCAobG9uZylsb2NhdGlvbi55ICsgaCA+IEludGVnZXIuTUFYX1ZBTFVFKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjc2PWxvY2F0aW9uLnggKyB3IG9yIGxvY2F0aW9uLnkgKyBoIHJlc3VsdHMgaW4gaW50ZWdlcgotICAgICAgICAgICAgLy8gb3ZlcmZsb3cKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjc2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoYmFuZHMgPCAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjc5PWJhbmRzIGlzIGxlc3MgdGhhbiAxCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3OSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGJhbmRPZmZzZXRzW10gPSBuZXcgaW50W2JhbmRzXTsKLSAgICAgICAgaW50IGJhbmtJbmRpY2VzW10gPSBuZXcgaW50W2JhbmRzXTsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgIGJhbmRPZmZzZXRzW2ldID0gMDsKLSAgICAgICAgICAgIGJhbmtJbmRpY2VzW2ldID0gaTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gY3JlYXRlQmFuZGVkUmFzdGVyKGRhdGFUeXBlLCB3LCBoLCB3LCBiYW5rSW5kaWNlcywgYmFuZE9mZnNldHMsIGxvY2F0aW9uKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgUmFzdGVyIG9iamVjdCB3aXRoIGEgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsIGFuZCB0aGUKLSAgICAgKiBzcGVjaWZpZWQgRGF0YUJ1ZmZlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YUJ1ZmZlcgotICAgICAqICAgICAgICAgICAgdGhlIERhdGFCdWZmZXIuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIHNjYW5saW5lU3RyaWRlCi0gICAgICogICAgICAgICAgICB0aGUgc2NhbmxpbmUgc3RyaWRlIG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBwaXhlbFN0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsIHN0cmlkZSBvZiBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBiYW5kT2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIGJhbmQgb2Zmc2V0cyBvZiBiYW5kcy4KLSAgICAgKiBAcGFyYW0gbG9jYXRpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBsb2NhdGlvbiB3aGljaCBkZWZpbmVzIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUKLSAgICAgKiAgICAgICAgICAgIFJhc3Rlci4KLSAgICAgKiBAcmV0dXJuIHRoZSBXcml0YWJsZVJhc3RlciBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBXcml0YWJsZVJhc3RlciBjcmVhdGVJbnRlcmxlYXZlZFJhc3RlcihEYXRhQnVmZmVyIGRhdGFCdWZmZXIsIGludCB3LCBpbnQgaCwKLSAgICAgICAgICAgIGludCBzY2FubGluZVN0cmlkZSwgaW50IHBpeGVsU3RyaWRlLCBpbnQgYmFuZE9mZnNldHNbXSwgUG9pbnQgbG9jYXRpb24pIHsKLQotICAgICAgICBpZiAodyA8PSAwIHx8IGggPD0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjIyRT13IG9yIGggaXMgbGVzcyB0aGFuIG9yIGVxdWFsIHRvIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjJFIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAobG9jYXRpb24gPT0gbnVsbCkgewotICAgICAgICAgICAgbG9jYXRpb24gPSBuZXcgUG9pbnQoMCwgMCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoKGxvbmcpbG9jYXRpb24ueCArIHcgPiBJbnRlZ2VyLk1BWF9WQUxVRSB8fCAobG9uZylsb2NhdGlvbi55ICsgaCA+IEludGVnZXIuTUFYX1ZBTFVFKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjc2PWxvY2F0aW9uLnggKyB3IG9yIGxvY2F0aW9uLnkgKyBoIHJlc3VsdHMgaW4gaW50ZWdlcgotICAgICAgICAgICAgLy8gb3ZlcmZsb3cKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjc2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoZGF0YUJ1ZmZlciA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjc4PWRhdGFCdWZmZXIgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3OCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGRhdGFUeXBlID0gZGF0YUJ1ZmZlci5nZXREYXRhVHlwZSgpOwotICAgICAgICBpZiAoZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYgZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCkgewotICAgICAgICAgICAgLy8gYXd0LjIzMD1kYXRhVHlwZSBpcyBub3Qgb25lIG9mIHRoZSBzdXBwb3J0ZWQgZGF0YSB0eXBlcwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMzAiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkYXRhQnVmZmVyLmdldE51bUJhbmtzKCkgPiAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjdBPWRhdGFCdWZmZXIgaGFzIG1vcmUgdGhhbiBvbmUgYmFuawotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yN0EiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChiYW5kT2Zmc2V0cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjdCPWJhbmRPZmZzZXRzIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yN0IiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbCBzYW1wbGVNb2RlbCA9IG5ldyBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwoZGF0YVR5cGUsIHcsIGgsCi0gICAgICAgICAgICAgICAgcGl4ZWxTdHJpZGUsIHNjYW5saW5lU3RyaWRlLCBiYW5kT2Zmc2V0cyk7Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBPcmRpbmFyeVdyaXRhYmxlUmFzdGVyKHNhbXBsZU1vZGVsLCBkYXRhQnVmZmVyLCBsb2NhdGlvbik7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgUmFzdGVyIG9iamVjdCB3aXRoIGEgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsIGFuZCB0aGUKLSAgICAgKiBzcGVjaWZpZWQgZGF0YSB0eXBlLiBUaGUgRGF0YSB0eXBlIGNhbiBiZSBvbmUgb2YgdGhlIGZvbGxvd2luZyB2YWx1ZXM6Ci0gICAgICogVFlQRV9CWVRFLCBUWVBFX1VTSE9SVCwgb3IgVFlQRV9JTlQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlIG9mIHRoZSBzYW1wbGVzOiBUWVBFX0JZVEUsIFRZUEVfVVNIT1JULCBvcgotICAgICAqICAgICAgICAgICAgVFlQRV9JTlQuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIHNjYW5saW5lU3RyaWRlCi0gICAgICogICAgICAgICAgICB0aGUgc2NhbmxpbmUgc3RyaWRlIG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBwaXhlbFN0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsIHN0cmlkZSBvZiBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBiYW5kT2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIGJhbmQgb2Zmc2V0cyBvZiBiYW5kcy4KLSAgICAgKiBAcGFyYW0gbG9jYXRpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBsb2NhdGlvbiB3aGljaCBkZWZpbmVzIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUKLSAgICAgKiAgICAgICAgICAgIFJhc3Rlci4KLSAgICAgKiBAcmV0dXJuIHRoZSBXcml0YWJsZVJhc3RlciBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBXcml0YWJsZVJhc3RlciBjcmVhdGVJbnRlcmxlYXZlZFJhc3RlcihpbnQgZGF0YVR5cGUsIGludCB3LCBpbnQgaCwKLSAgICAgICAgICAgIGludCBzY2FubGluZVN0cmlkZSwgaW50IHBpeGVsU3RyaWRlLCBpbnQgYmFuZE9mZnNldHNbXSwgUG9pbnQgbG9jYXRpb24pIHsKLQotICAgICAgICBpZiAodyA8PSAwIHx8IGggPD0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjIyRT13IG9yIGggaXMgbGVzcyB0aGFuIG9yIGVxdWFsIHRvIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjJFIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAobG9jYXRpb24gPT0gbnVsbCkgewotICAgICAgICAgICAgbG9jYXRpb24gPSBuZXcgUG9pbnQoMCwgMCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoKGxvbmcpbG9jYXRpb24ueCArIHcgPiBJbnRlZ2VyLk1BWF9WQUxVRSB8fCAobG9uZylsb2NhdGlvbi55ICsgaCA+IEludGVnZXIuTUFYX1ZBTFVFKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjc2PWxvY2F0aW9uLnggKyB3IG9yIGxvY2F0aW9uLnkgKyBoIHJlc3VsdHMgaW4gaW50ZWdlcgotICAgICAgICAgICAgLy8gb3ZlcmZsb3cKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjc2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYgZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCkgewotICAgICAgICAgICAgLy8gYXd0LjIzMD1kYXRhVHlwZSBpcyBub3Qgb25lIG9mIHRoZSBzdXBwb3J0ZWQgZGF0YSB0eXBlcwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMzAiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChiYW5kT2Zmc2V0cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjdCPWJhbmRPZmZzZXRzIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yN0IiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGludCBtaW5PZmZzZXQgPSBiYW5kT2Zmc2V0c1swXTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDE7IGkgPCBiYW5kT2Zmc2V0cy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgaWYgKGJhbmRPZmZzZXRzW2ldIDwgbWluT2Zmc2V0KSB7Ci0gICAgICAgICAgICAgICAgbWluT2Zmc2V0ID0gYmFuZE9mZnNldHNbaV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgaW50IHNpemUgPSAoaCAtIDEpICogc2NhbmxpbmVTdHJpZGUgKyB3ICogcGl4ZWxTdHJpZGUgKyBtaW5PZmZzZXQ7Ci0gICAgICAgIERhdGFCdWZmZXIgZGF0YSA9IG51bGw7Ci0KLSAgICAgICAgc3dpdGNoIChkYXRhVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBkYXRhID0gbmV3IERhdGFCdWZmZXJCeXRlKHNpemUpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlclVTaG9ydChzaXplKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBjcmVhdGVJbnRlcmxlYXZlZFJhc3RlcihkYXRhLCB3LCBoLCBzY2FubGluZVN0cmlkZSwgcGl4ZWxTdHJpZGUsIGJhbmRPZmZzZXRzLAotICAgICAgICAgICAgICAgIGxvY2F0aW9uKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgUmFzdGVyIG9iamVjdCB3aXRoIGEgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsIGFuZCB0aGUKLSAgICAgKiBzcGVjaWZpZWQgZGF0YSB0eXBlLiBUaGUgRGF0YSB0eXBlIGNhbiBiZSBvbmUgb2YgdGhlIGZvbGxvd2luZyB2YWx1ZXM6Ci0gICAgICogVFlQRV9CWVRFLCBUWVBFX1VTSE9SVCwgb3IgVFlQRV9JTlQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlIG9mIHNhbXBsZXM6IFRZUEVfQllURSwgVFlQRV9VU0hPUlQsIG9yIFRZUEVfSU5ULgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBiYW5kcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBiYW5kcy4KLSAgICAgKiBAcGFyYW0gbG9jYXRpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBsb2NhdGlvbiB3aGljaCBkZWZpbmVzIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUKLSAgICAgKiAgICAgICAgICAgIFJhc3Rlci4KLSAgICAgKiBAcmV0dXJuIHRoZSBXcml0YWJsZVJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZUludGVybGVhdmVkUmFzdGVyKGludCBkYXRhVHlwZSwgaW50IHcsIGludCBoLCBpbnQgYmFuZHMsCi0gICAgICAgICAgICBQb2ludCBsb2NhdGlvbikgewotCi0gICAgICAgIGlmICh3IDw9IDAgfHwgaCA8PSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjJFPXcgb3IgaCBpcyBsZXNzIHRoYW4gb3IgZXF1YWwgdG8gemVybwotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMkUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChsb2NhdGlvbiA9PSBudWxsKSB7Ci0gICAgICAgICAgICBsb2NhdGlvbiA9IG5ldyBQb2ludCgwLCAwKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICgobG9uZylsb2NhdGlvbi54ICsgdyA+IEludGVnZXIuTUFYX1ZBTFVFIHx8IChsb25nKWxvY2F0aW9uLnkgKyBoID4gSW50ZWdlci5NQVhfVkFMVUUpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yNzY9bG9jYXRpb24ueCArIHcgb3IgbG9jYXRpb24ueSArIGggcmVzdWx0cyBpbiBpbnRlZ2VyCi0gICAgICAgICAgICAvLyBvdmVyZmxvdwotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNzYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfQllURSAmJiBkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjMwPWRhdGFUeXBlIGlzIG5vdCBvbmUgb2YgdGhlIHN1cHBvcnRlZCBkYXRhIHR5cGVzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGJhbmRPZmZzZXRzW10gPSBuZXcgaW50W2JhbmRzXTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBiYW5kczsgaSsrKSB7Ci0gICAgICAgICAgICBiYW5kT2Zmc2V0c1tpXSA9IGk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gY3JlYXRlSW50ZXJsZWF2ZWRSYXN0ZXIoZGF0YVR5cGUsIHcsIGgsIHcgKiBiYW5kcywgYmFuZHMsIGJhbmRPZmZzZXRzLCBsb2NhdGlvbik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIFJhc3RlciBvYmplY3Qgd2l0aCBhIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgYW5kIHRoZQotICAgICAqIHNwZWNpZmllZCBEYXRhQnVmZmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQnVmZmVyCi0gICAgICogICAgICAgICAgICB0aGUgRGF0YUJ1ZmZlci4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBzY2FubGluZVN0cmlkZQotICAgICAqICAgICAgICAgICAgdGhlIHNjYW5saW5lIHN0cmlkZSBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gYmFuZE1hc2tzCi0gICAgICogICAgICAgICAgICB0aGUgYmFuZCBtYXNrcy4KLSAgICAgKiBAcGFyYW0gbG9jYXRpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBsb2NhdGlvbiB3aGljaCBkZWZpbmVzIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUKLSAgICAgKiAgICAgICAgICAgIFJhc3Rlci4KLSAgICAgKiBAcmV0dXJuIHRoZSBXcml0YWJsZVJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZVBhY2tlZFJhc3RlcihEYXRhQnVmZmVyIGRhdGFCdWZmZXIsIGludCB3LCBpbnQgaCwKLSAgICAgICAgICAgIGludCBzY2FubGluZVN0cmlkZSwgaW50IGJhbmRNYXNrc1tdLCBQb2ludCBsb2NhdGlvbikgewotICAgICAgICBpZiAoZGF0YUJ1ZmZlciA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjc4PWRhdGFCdWZmZXIgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3OCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHcgPD0gMCB8fCBoIDw9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMkU9dyBvciBoIGlzIGxlc3MgdGhhbiBvciBlcXVhbCB0byB6ZXJvCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIyRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGxvY2F0aW9uID09IG51bGwpIHsKLSAgICAgICAgICAgIGxvY2F0aW9uID0gbmV3IFBvaW50KDAsIDApOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChsb25nKWxvY2F0aW9uLnggKyB3ID4gSW50ZWdlci5NQVhfVkFMVUUgfHwgKGxvbmcpbG9jYXRpb24ueSArIGggPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI3Nj1sb2NhdGlvbi54ICsgdyBvciBsb2NhdGlvbi55ICsgaCByZXN1bHRzIGluIGludGVnZXIKLSAgICAgICAgICAgIC8vIG92ZXJmbG93Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3NiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGJhbmRNYXNrcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjdDPWJhbmRNYXNrcyBpcyBudWxsCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3QyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRhdGFCdWZmZXIuZ2V0TnVtQmFua3MoKSA+IDEpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yN0E9ZGF0YUJ1ZmZlciBoYXMgbW9yZSB0aGFuIG9uZSBiYW5rCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3QSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGRhdGFUeXBlID0gZGF0YUJ1ZmZlci5nZXREYXRhVHlwZSgpOwotICAgICAgICBpZiAoZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYgZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVAotICAgICAgICAgICAgICAgICYmIGRhdGFUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9JTlQpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMzA9ZGF0YVR5cGUgaXMgbm90IG9uZSBvZiB0aGUgc3VwcG9ydGVkIGRhdGEgdHlwZXMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjMwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsID0gbmV3IFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwoZGF0YVR5cGUsIHcsIGgsCi0gICAgICAgICAgICAgICAgc2NhbmxpbmVTdHJpZGUsIGJhbmRNYXNrcyk7Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBPcmRpbmFyeVdyaXRhYmxlUmFzdGVyKHNhbXBsZU1vZGVsLCBkYXRhQnVmZmVyLCBsb2NhdGlvbik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIFJhc3RlciBvYmplY3Qgd2l0aCBhIE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBhbmQgdGhlCi0gICAgICogc3BlY2lmaWVkIERhdGFCdWZmZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFCdWZmZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBEYXRhQnVmZmVyLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGJpdHNQZXJQaXhlbAotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBiaXRzIHBlciBwaXhlbC4KLSAgICAgKiBAcGFyYW0gbG9jYXRpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBsb2NhdGlvbiB3aGljaCBkZWZpbmVzIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUKLSAgICAgKiAgICAgICAgICAgIFJhc3Rlci4KLSAgICAgKiBAcmV0dXJuIHRoZSBXcml0YWJsZVJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZVBhY2tlZFJhc3RlcihEYXRhQnVmZmVyIGRhdGFCdWZmZXIsIGludCB3LCBpbnQgaCwKLSAgICAgICAgICAgIGludCBiaXRzUGVyUGl4ZWwsIFBvaW50IGxvY2F0aW9uKSB7Ci0KLSAgICAgICAgaWYgKHcgPD0gMCB8fCBoIDw9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMkU9dyBvciBoIGlzIGxlc3MgdGhhbiBvciBlcXVhbCB0byB6ZXJvCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIyRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGxvY2F0aW9uID09IG51bGwpIHsKLSAgICAgICAgICAgIGxvY2F0aW9uID0gbmV3IFBvaW50KDAsIDApOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChsb25nKWxvY2F0aW9uLnggKyB3ID4gSW50ZWdlci5NQVhfVkFMVUUgfHwgKGxvbmcpbG9jYXRpb24ueSArIGggPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI3Nj1sb2NhdGlvbi54ICsgdyBvciBsb2NhdGlvbi55ICsgaCByZXN1bHRzIGluIGludGVnZXIKLSAgICAgICAgICAgIC8vIG92ZXJmbG93Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3NiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRhdGFCdWZmZXIgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjI3OD1kYXRhQnVmZmVyIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNzgiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkYXRhQnVmZmVyLmdldE51bUJhbmtzKCkgPiAxKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjdBPWRhdGFCdWZmZXIgaGFzIG1vcmUgdGhhbiBvbmUgYmFuawotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yN0EiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGludCBkYXRhVHlwZSA9IGRhdGFCdWZmZXIuZ2V0RGF0YVR5cGUoKTsKLSAgICAgICAgaWYgKGRhdGFUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9CWVRFICYmIGRhdGFUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9VU0hPUlQKLSAgICAgICAgICAgICAgICAmJiBkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfSU5UKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjMwPWRhdGFUeXBlIGlzIG5vdCBvbmUgb2YgdGhlIHN1cHBvcnRlZCBkYXRhIHR5cGVzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsID0gbmV3IE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbChkYXRhVHlwZSwgdywgaCwKLSAgICAgICAgICAgICAgICBiaXRzUGVyUGl4ZWwpOwotCi0gICAgICAgIHJldHVybiBuZXcgT3JkaW5hcnlXcml0YWJsZVJhc3RlcihzYW1wbGVNb2RlbCwgZGF0YUJ1ZmZlciwgbG9jYXRpb24pOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIFJhc3RlciBvYmplY3Qgd2l0aCBhIE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBhbmQgdGhlCi0gICAgICogc3BlY2lmaWVkIERhdGFCdWZmZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlIG9mIHNhbXBsZXM6IFRZUEVfQllURSwgVFlQRV9VU0hPUlQsIG9yIFRZUEVfSU5ULgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGJhbmRzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGJhbmRzLgotICAgICAqIEBwYXJhbSBiaXRzUGVyQmFuZAotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBiaXRzIHBlciBiYW5kLgotICAgICAqIEBwYXJhbSBsb2NhdGlvbgotICAgICAqICAgICAgICAgICAgdGhlIGxvY2F0aW9uIHdoaWNoIGRlZmluZXMgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZQotICAgICAqICAgICAgICAgICAgUmFzdGVyLgotICAgICAqIEByZXR1cm4gdGhlIFdyaXRhYmxlUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlUGFja2VkUmFzdGVyKGludCBkYXRhVHlwZSwgaW50IHcsIGludCBoLCBpbnQgYmFuZHMsCi0gICAgICAgICAgICBpbnQgYml0c1BlckJhbmQsIFBvaW50IGxvY2F0aW9uKSB7Ci0KLSAgICAgICAgaWYgKHcgPD0gMCB8fCBoIDw9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMkU9dyBvciBoIGlzIGxlc3MgdGhhbiBvciBlcXVhbCB0byB6ZXJvCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIyRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGxvY2F0aW9uID09IG51bGwpIHsKLSAgICAgICAgICAgIGxvY2F0aW9uID0gbmV3IFBvaW50KDAsIDApOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChsb25nKWxvY2F0aW9uLnggKyB3ID4gSW50ZWdlci5NQVhfVkFMVUUgfHwgKGxvbmcpbG9jYXRpb24ueSArIGggPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI3Nj1sb2NhdGlvbi54ICsgdyBvciBsb2NhdGlvbi55ICsgaCByZXN1bHRzIGluIGludGVnZXIKLSAgICAgICAgICAgIC8vIG92ZXJmbG93Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3NiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGJhbmRzIDwgMSB8fCBiaXRzUGVyQmFuZCA8IDEpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yN0Q9Yml0c1BlckJhbmQgb3IgYmFuZHMgaXMgbm90IGdyZWF0ZXIgdGhhbiB6ZXJvCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3RCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRhdGFUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9CWVRFICYmIGRhdGFUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9VU0hPUlQKLSAgICAgICAgICAgICAgICAmJiBkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfSU5UKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjMwPWRhdGFUeXBlIGlzIG5vdCBvbmUgb2YgdGhlIHN1cHBvcnRlZCBkYXRhIHR5cGVzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGJpdHNQZXJCYW5kICogYmFuZHMgPiBEYXRhQnVmZmVyLmdldERhdGFUeXBlU2l6ZShkYXRhVHlwZSkpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yN0U9VGhlIHByb2R1Y3Qgb2YgYml0c1BlckJhbmQgYW5kIGJhbmRzIGlzIGdyZWF0ZXIgdGhhbiB0aGUKLSAgICAgICAgICAgIC8vIG51bWJlciBvZiBiaXRzIGhlbGQgYnkgZGF0YVR5cGUKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjdFIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoYmFuZHMgPiAxKSB7Ci0KLSAgICAgICAgICAgIGludCBiYW5kTWFza3NbXSA9IG5ldyBpbnRbYmFuZHNdOwotICAgICAgICAgICAgaW50IG1hc2sgPSAoMSA8PCBiaXRzUGVyQmFuZCkgLSAxOwotCi0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICBiYW5kTWFza3NbaV0gPSBtYXNrIDw8IChiaXRzUGVyQmFuZCAqIChiYW5kcyAtIDEgLSBpKSk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBjcmVhdGVQYWNrZWRSYXN0ZXIoZGF0YVR5cGUsIHcsIGgsIGJhbmRNYXNrcywgbG9jYXRpb24pOwotICAgICAgICB9Ci0gICAgICAgIERhdGFCdWZmZXIgZGF0YSA9IG51bGw7Ci0gICAgICAgIGludCBzaXplID0gKChiaXRzUGVyQmFuZCAqIHcgKyBEYXRhQnVmZmVyLmdldERhdGFUeXBlU2l6ZShkYXRhVHlwZSkgLSAxKSAvIERhdGFCdWZmZXIKLSAgICAgICAgICAgICAgICAuZ2V0RGF0YVR5cGVTaXplKGRhdGFUeXBlKSkKLSAgICAgICAgICAgICAgICAqIGg7Ci0KLSAgICAgICAgc3dpdGNoIChkYXRhVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBkYXRhID0gbmV3IERhdGFCdWZmZXJCeXRlKHNpemUpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlclVTaG9ydChzaXplKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBkYXRhID0gbmV3IERhdGFCdWZmZXJJbnQoc2l6ZSk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGNyZWF0ZVBhY2tlZFJhc3RlcihkYXRhLCB3LCBoLCBiaXRzUGVyQmFuZCwgbG9jYXRpb24pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBSYXN0ZXIgb2JqZWN0IHdpdGggYSBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIGFuZCB0aGUKLSAgICAgKiBzcGVjaWZpZWQgRGF0YUJ1ZmZlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUgb2Ygc2FtcGxlczogVFlQRV9CWVRFLCBUWVBFX1VTSE9SVCwgb3IgVFlQRV9JTlQuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gYmFuZE1hc2tzCi0gICAgICogICAgICAgICAgICB0aGUgYmFuZCBtYXNrcy4KLSAgICAgKiBAcGFyYW0gbG9jYXRpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBsb2NhdGlvbiB3aGljaCBkZWZpbmVzIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUKLSAgICAgKiAgICAgICAgICAgIFJhc3Rlci4KLSAgICAgKiBAcmV0dXJuIHRoZSBXcml0YWJsZVJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZVBhY2tlZFJhc3RlcihpbnQgZGF0YVR5cGUsIGludCB3LCBpbnQgaCwgaW50IGJhbmRNYXNrc1tdLAotICAgICAgICAgICAgUG9pbnQgbG9jYXRpb24pIHsKLQotICAgICAgICBpZiAoZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYgZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVAotICAgICAgICAgICAgICAgICYmIGRhdGFUeXBlICE9IERhdGFCdWZmZXIuVFlQRV9JTlQpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMzA9ZGF0YVR5cGUgaXMgbm90IG9uZSBvZiB0aGUgc3VwcG9ydGVkIGRhdGEgdHlwZXMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjMwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAodyA8PSAwIHx8IGggPD0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjIyRT13IG9yIGggaXMgbGVzcyB0aGFuIG9yIGVxdWFsIHRvIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjJFIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAobG9jYXRpb24gPT0gbnVsbCkgewotICAgICAgICAgICAgbG9jYXRpb24gPSBuZXcgUG9pbnQoMCwgMCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoKGxvbmcpbG9jYXRpb24ueCArIHcgPiBJbnRlZ2VyLk1BWF9WQUxVRSB8fCAobG9uZylsb2NhdGlvbi55ICsgaCA+IEludGVnZXIuTUFYX1ZBTFVFKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjc2PWxvY2F0aW9uLnggKyB3IG9yIGxvY2F0aW9uLnkgKyBoIHJlc3VsdHMgaW4gaW50ZWdlcgotICAgICAgICAgICAgLy8gb3ZlcmZsb3cKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjc2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoYmFuZE1hc2tzID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yN0M9YmFuZE1hc2tzIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yN0MiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIERhdGFCdWZmZXIgZGF0YSA9IG51bGw7Ci0KLSAgICAgICAgc3dpdGNoIChkYXRhVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBkYXRhID0gbmV3IERhdGFCdWZmZXJCeXRlKHcgKiBoKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgICAgICBkYXRhID0gbmV3IERhdGFCdWZmZXJVU2hvcnQodyAqIGgpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlckludCh3ICogaCk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gY3JlYXRlUGFja2VkUmFzdGVyKGRhdGEsIHcsIGgsIHcsIGJhbmRNYXNrcywgbG9jYXRpb24pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBSYXN0ZXIgb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCBEYXRhQnVmZmVyIGFuZCBTYW1wbGVNb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc20KLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwuCi0gICAgICogQHBhcmFtIGRiCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIERhdGFCdWZmZXIuCi0gICAgICogQHBhcmFtIGxvY2F0aW9uCi0gICAgICogICAgICAgICAgICB0aGUgbG9jYXRpb24gd2hpY2ggZGVmaW5lcyB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlCi0gICAgICogICAgICAgICAgICBSYXN0ZXIuCi0gICAgICogQHJldHVybiB0aGUgUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgUmFzdGVyIGNyZWF0ZVJhc3RlcihTYW1wbGVNb2RlbCBzbSwgRGF0YUJ1ZmZlciBkYiwgUG9pbnQgbG9jYXRpb24pIHsKLQotICAgICAgICBpZiAoc20gPT0gbnVsbCB8fCBkYiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjdGPVNhbXBsZU1vZGVsIG9yIERhdGFCdWZmZXIgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3RiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGxvY2F0aW9uID09IG51bGwpIHsKLSAgICAgICAgICAgIGxvY2F0aW9uID0gbmV3IFBvaW50KDAsIDApOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBSYXN0ZXIoc20sIGRiLCBsb2NhdGlvbik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIFdyaXRhYmxlUmFzdGVyIHdpdGggdGhlIHNwZWNpZmllZCBTYW1wbGVNb2RlbCBhbmQgRGF0YUJ1ZmZlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc20KLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwuCi0gICAgICogQHBhcmFtIGRiCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIERhdGFCdWZmZXIuCi0gICAgICogQHBhcmFtIGxvY2F0aW9uCi0gICAgICogICAgICAgICAgICB0aGUgbG9jYXRpb24gd2hpY2ggZGVmaW5lcyB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlCi0gICAgICogICAgICAgICAgICBSYXN0ZXIuCi0gICAgICogQHJldHVybiB0aGUgV3JpdGFibGVSYXN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBXcml0YWJsZVJhc3RlciBjcmVhdGVXcml0YWJsZVJhc3RlcihTYW1wbGVNb2RlbCBzbSwgRGF0YUJ1ZmZlciBkYiwgUG9pbnQgbG9jYXRpb24pIHsKLQotICAgICAgICBpZiAoc20gPT0gbnVsbCB8fCBkYiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjdGPVNhbXBsZU1vZGVsIG9yIERhdGFCdWZmZXIgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI3RiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGxvY2F0aW9uID09IG51bGwpIHsKLSAgICAgICAgICAgIGxvY2F0aW9uID0gbmV3IFBvaW50KDAsIDApOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBPcmRpbmFyeVdyaXRhYmxlUmFzdGVyKHNtLCBkYiwgbG9jYXRpb24pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBXcml0YWJsZVJhc3RlciB3aXRoIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNtCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIFNhbXBsZU1vZGVsLgotICAgICAqIEBwYXJhbSBsb2NhdGlvbgotICAgICAqICAgICAgICAgICAgdGhlIGxvY2F0aW9uIHdoaWNoIGRlZmluZXMgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZQotICAgICAqICAgICAgICAgICAgUmFzdGVyLgotICAgICAqIEByZXR1cm4gdGhlIFdyaXRhYmxlUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlV3JpdGFibGVSYXN0ZXIoU2FtcGxlTW9kZWwgc20sIFBvaW50IGxvY2F0aW9uKSB7Ci0KLSAgICAgICAgaWYgKHNtID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yODA9U2FtcGxlTW9kZWwgaXMgbnVsbAotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI4MCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGxvY2F0aW9uID09IG51bGwpIHsKLSAgICAgICAgICAgIGxvY2F0aW9uID0gbmV3IFBvaW50KDAsIDApOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGNyZWF0ZVdyaXRhYmxlUmFzdGVyKHNtLCBzbS5jcmVhdGVEYXRhQnVmZmVyKCksIGxvY2F0aW9uKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgUmFzdGVyIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwgYW5kCi0gICAgICogRGF0YUJ1ZmZlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2FtcGxlTW9kZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwuCi0gICAgICogQHBhcmFtIGRhdGFCdWZmZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgRGF0YUJ1ZmZlci4KLSAgICAgKiBAcGFyYW0gb3JpZ2luCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIG9yaWdpbi4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgUmFzdGVyKFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsLCBEYXRhQnVmZmVyIGRhdGFCdWZmZXIsIFBvaW50IG9yaWdpbikgewotCi0gICAgICAgIHRoaXMoc2FtcGxlTW9kZWwsIGRhdGFCdWZmZXIsIG5ldyBSZWN0YW5nbGUob3JpZ2luLngsIG9yaWdpbi55LCBzYW1wbGVNb2RlbC5nZXRXaWR0aCgpLAotICAgICAgICAgICAgICAgIHNhbXBsZU1vZGVsLmdldEhlaWdodCgpKSwgb3JpZ2luLCBudWxsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgUmFzdGVyIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwsCi0gICAgICogRGF0YUJ1ZmZlciwgcmVjdGFuZ3VsYXIgcmVnaW9uIGFuZCBwYXJlbnQgUmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzYW1wbGVNb2RlbAotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBTYW1wbGVNb2RlbC4KLSAgICAgKiBAcGFyYW0gZGF0YUJ1ZmZlcgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBEYXRhQnVmZmVyLgotICAgICAqIEBwYXJhbSBhUmVnaW9uCi0gICAgICogICAgICAgICAgICB0aGUgYSByZWN0YW5ndWxhciByZWdpb24gd2hpY2ggZGVmaW5lcyB0aGUgbmV3IGltYWdlIGJvdW5kcy4KLSAgICAgKiBAcGFyYW0gc2FtcGxlTW9kZWxUcmFuc2xhdGUKLSAgICAgKiAgICAgICAgICAgIHRoaXMgcG9pbnQgZGVmaW5lcyB0aGUgdHJhbnNsYXRpb24gcG9pbnQgZnJvbSB0aGUgU2FtcGxlTW9kZWwKLSAgICAgKiAgICAgICAgICAgIGNvb3JkaW5hdGVzIHRvIHRoZSBuZXcgUmFzdGVyIGNvb3JkaW5hdGVzLgotICAgICAqIEBwYXJhbSBwYXJlbnQKLSAgICAgKiAgICAgICAgICAgIHRoZSBwYXJlbnQgb2YgdGhpcyBSYXN0ZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFJhc3RlcihTYW1wbGVNb2RlbCBzYW1wbGVNb2RlbCwgRGF0YUJ1ZmZlciBkYXRhQnVmZmVyLCBSZWN0YW5nbGUgYVJlZ2lvbiwKLSAgICAgICAgICAgIFBvaW50IHNhbXBsZU1vZGVsVHJhbnNsYXRlLCBSYXN0ZXIgcGFyZW50KSB7Ci0KLSAgICAgICAgaWYgKHNhbXBsZU1vZGVsID09IG51bGwgfHwgZGF0YUJ1ZmZlciA9PSBudWxsIHx8IGFSZWdpb24gPT0gbnVsbAotICAgICAgICAgICAgICAgIHx8IHNhbXBsZU1vZGVsVHJhbnNsYXRlID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yODE9c2FtcGxlTW9kZWwsIGRhdGFCdWZmZXIsIGFSZWdpb24gb3Igc2FtcGxlTW9kZWxUcmFuc2xhdGUKLSAgICAgICAgICAgIC8vIGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yODEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChhUmVnaW9uLndpZHRoIDw9IDAgfHwgYVJlZ2lvbi5oZWlnaHQgPD0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjI4Mj1hUmVnaW9uIGhhcyB3aWR0aCBvciBoZWlnaHQgbGVzcyB0aGFuIG9yIGVxdWFsIHRvIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjgyIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoKGxvbmcpYVJlZ2lvbi54ICsgKGxvbmcpYVJlZ2lvbi53aWR0aCA+IEludGVnZXIuTUFYX1ZBTFVFKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjgzPU92ZXJmbG93IFggY29vcmRpbmF0ZSBvZiBSYXN0ZXIKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjgzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoKGxvbmcpYVJlZ2lvbi55ICsgKGxvbmcpYVJlZ2lvbi5oZWlnaHQgPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI4ND1PdmVyZmxvdyBZIGNvb3JkaW5hdGUgb2YgUmFzdGVyCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI4NCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHNhbXBsZU1vZGVsIGluc3RhbmNlb2YgQ29tcG9uZW50U2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgIHZhbGlkYXRlRGF0YUJ1ZmZlcihkYXRhQnVmZmVyLCBhUmVnaW9uLndpZHRoLCBhUmVnaW9uLmhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgKChDb21wb25lbnRTYW1wbGVNb2RlbClzYW1wbGVNb2RlbCkuZ2V0U2NhbmxpbmVTdHJpZGUoKSk7Ci0gICAgICAgIH0gZWxzZSBpZiAoc2FtcGxlTW9kZWwgaW5zdGFuY2VvZiBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgIHZhbGlkYXRlRGF0YUJ1ZmZlcihkYXRhQnVmZmVyLCBhUmVnaW9uLndpZHRoLCBhUmVnaW9uLmhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgKChNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpc2FtcGxlTW9kZWwpLmdldFNjYW5saW5lU3RyaWRlKCkpOwotICAgICAgICB9IGVsc2UgaWYgKHNhbXBsZU1vZGVsIGluc3RhbmNlb2YgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkgewotICAgICAgICAgICAgdmFsaWRhdGVEYXRhQnVmZmVyKGRhdGFCdWZmZXIsIGFSZWdpb24ud2lkdGgsIGFSZWdpb24uaGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICAoKFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpc2FtcGxlTW9kZWwpLmdldFNjYW5saW5lU3RyaWRlKCkpOwotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5zYW1wbGVNb2RlbCA9IHNhbXBsZU1vZGVsOwotICAgICAgICB0aGlzLmRhdGFCdWZmZXIgPSBkYXRhQnVmZmVyOwotICAgICAgICB0aGlzLm1pblggPSBhUmVnaW9uLng7Ci0gICAgICAgIHRoaXMubWluWSA9IGFSZWdpb24ueTsKLSAgICAgICAgdGhpcy53aWR0aCA9IGFSZWdpb24ud2lkdGg7Ci0gICAgICAgIHRoaXMuaGVpZ2h0ID0gYVJlZ2lvbi5oZWlnaHQ7Ci0gICAgICAgIHRoaXMuc2FtcGxlTW9kZWxUcmFuc2xhdGVYID0gc2FtcGxlTW9kZWxUcmFuc2xhdGUueDsKLSAgICAgICAgdGhpcy5zYW1wbGVNb2RlbFRyYW5zbGF0ZVkgPSBzYW1wbGVNb2RlbFRyYW5zbGF0ZS55OwotICAgICAgICB0aGlzLnBhcmVudCA9IHBhcmVudDsKLSAgICAgICAgdGhpcy5udW1CYW5kcyA9IHNhbXBsZU1vZGVsLmdldE51bUJhbmRzKCk7Ci0gICAgICAgIHRoaXMubnVtRGF0YUVsZW1lbnRzID0gc2FtcGxlTW9kZWwuZ2V0TnVtRGF0YUVsZW1lbnRzKCk7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgUmFzdGVyIHdpdGggdGhlIHNwZWNpZmllZCBTYW1wbGVNb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2FtcGxlTW9kZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwuCi0gICAgICogQHBhcmFtIG9yaWdpbgotICAgICAqICAgICAgICAgICAgdGhlIG9yaWdpbi4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgUmFzdGVyKFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsLCBQb2ludCBvcmlnaW4pIHsKLSAgICAgICAgdGhpcyhzYW1wbGVNb2RlbCwgc2FtcGxlTW9kZWwuY3JlYXRlRGF0YUJ1ZmZlcigpLCBuZXcgUmVjdGFuZ2xlKG9yaWdpbi54LCBvcmlnaW4ueSwKLSAgICAgICAgICAgICAgICBzYW1wbGVNb2RlbC5nZXRXaWR0aCgpLCBzYW1wbGVNb2RlbC5nZXRIZWlnaHQoKSksIG9yaWdpbiwgbnVsbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgY2hpbGQgb2YgdGhpcyBSYXN0ZXIgYnkgc2hhcmluZyB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyCi0gICAgICogYXJlYSBpbiB0aGlzIHJhc3Rlci4gVGhlIHBhcmVudFgsIHBhcmVudFksIHdpZHRoIGFuZCBoZWlnaHQgcGFyYW1ldGVycwotICAgICAqIHNwZWNpZnkgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgdG8gYmUgc2hhcmVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXJlbnRYCi0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGlzIFJhc3Rlci4KLSAgICAgKiBAcGFyYW0gcGFyZW50WQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhpcyBSYXN0ZXIuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGNoaWxkIGFyZWEuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgY2hpbGQgYXJlYS4KLSAgICAgKiBAcGFyYW0gY2hpbGRNaW5YCi0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIGNoaWxkIGFyZWEgbWFwcGVkIHRvIHRoZSBwYXJlbnRYCi0gICAgICogICAgICAgICAgICBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSBjaGlsZE1pblkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgY2hpbGQgYXJlYSBtYXBwZWQgdG8gdGhlIHBhcmVudFkKLSAgICAgKiAgICAgICAgICAgIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIGJhbmRMaXN0Ci0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgYmFuZCBpbmRpY2VzLgotICAgICAqIEByZXR1cm4gdGhlIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmFzdGVyIGNyZWF0ZUNoaWxkKGludCBwYXJlbnRYLCBpbnQgcGFyZW50WSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgY2hpbGRNaW5YLAotICAgICAgICAgICAgaW50IGNoaWxkTWluWSwgaW50IGJhbmRMaXN0W10pIHsKLSAgICAgICAgaWYgKHdpZHRoIDw9IDAgfHwgaGVpZ2h0IDw9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yODU9V2lkdGggb3IgSGVpZ2h0IG9mIGNoaWxkIFJhc3RlciBpcyBsZXNzIHRoYW4gb3IgZXF1YWwgdG8KLSAgICAgICAgICAgIC8vIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjg1IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAocGFyZW50WCA8IHRoaXMubWluWCB8fCBwYXJlbnRYICsgd2lkdGggPiB0aGlzLm1pblggKyB0aGlzLndpZHRoKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjg2PXBhcmVudFggZGlzcG9zZXMgb3V0c2lkZSBSYXN0ZXIKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjg2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAocGFyZW50WSA8IHRoaXMubWluWSB8fCBwYXJlbnRZICsgaGVpZ2h0ID4gdGhpcy5taW5ZICsgdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yODc9cGFyZW50WSBkaXNwb3NlcyBvdXRzaWRlIFJhc3RlcgotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yODciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmICgobG9uZylwYXJlbnRYICsgd2lkdGggPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI4OD1wYXJlbnRYICsgd2lkdGggcmVzdWx0cyBpbiBpbnRlZ2VyIG92ZXJmbG93Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI4OCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChsb25nKXBhcmVudFkgKyBoZWlnaHQgPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI4OT1wYXJlbnRZICsgaGVpZ2h0IHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yODkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmICgobG9uZyljaGlsZE1pblggKyB3aWR0aCA+IEludGVnZXIuTUFYX1ZBTFVFKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjhBPWNoaWxkTWluWCArIHdpZHRoIHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOEEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmICgobG9uZyljaGlsZE1pblkgKyBoZWlnaHQgPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI4Qj1jaGlsZE1pblkgKyBoZWlnaHQgcmVzdWx0cyBpbiBpbnRlZ2VyIG92ZXJmbG93Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI4QiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgU2FtcGxlTW9kZWwgY2hpbGRNb2RlbDsKLQotICAgICAgICBpZiAoYmFuZExpc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgY2hpbGRNb2RlbCA9IHNhbXBsZU1vZGVsOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgY2hpbGRNb2RlbCA9IHNhbXBsZU1vZGVsLmNyZWF0ZVN1YnNldFNhbXBsZU1vZGVsKGJhbmRMaXN0KTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBjaGlsZFRyYW5zbGF0ZVggPSBjaGlsZE1pblggLSBwYXJlbnRYOwotICAgICAgICBpbnQgY2hpbGRUcmFuc2xhdGVZID0gY2hpbGRNaW5ZIC0gcGFyZW50WTsKLQotICAgICAgICByZXR1cm4gbmV3IFJhc3RlcihjaGlsZE1vZGVsLCBkYXRhQnVmZmVyLAotICAgICAgICAgICAgICAgIG5ldyBSZWN0YW5nbGUoY2hpbGRNaW5YLCBjaGlsZE1pblksIHdpZHRoLCBoZWlnaHQpLCBuZXcgUG9pbnQoY2hpbGRUcmFuc2xhdGVYCi0gICAgICAgICAgICAgICAgICAgICAgICArIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgY2hpbGRUcmFuc2xhdGVZICsgc2FtcGxlTW9kZWxUcmFuc2xhdGVZKSwgdGhpcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlIGEgY29tcGF0aWJsZSBXcml0YWJsZVJhc3RlciB3aXRoIHRoZSBzYW1lIHBhcmFtZXRlcnMgYXMgdGhpcwotICAgICAqIFJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBXcml0YWJsZVJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKCkgewotICAgICAgICByZXR1cm4gbmV3IE9yZGluYXJ5V3JpdGFibGVSYXN0ZXIoc2FtcGxlTW9kZWwsIG5ldyBQb2ludCgwLCAwKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlIGEgY29tcGF0aWJsZSBXcml0YWJsZVJhc3RlciB3aXRoIHRoZSBzYW1lIHBhcmFtZXRlcnMgYXMgdGhpcwotICAgICAqIFJhc3RlciBhbmQgdGhlIHNwZWNpZmllZCBzaXplLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIG5ldyBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgbmV3IFdyaXRhYmxlUmFzdGVyLgotICAgICAqIEByZXR1cm4gdGhlIFdyaXRhYmxlUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBjcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIoaW50IHcsIGludCBoKSB7Ci0gICAgICAgIGlmICh3IDw9IDAgfHwgaCA8PSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjJFPXcgb3IgaCBpcyBsZXNzIHRoYW4gb3IgZXF1YWwgdG8gemVybwotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMkUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIFNhbXBsZU1vZGVsIHNtID0gc2FtcGxlTW9kZWwuY3JlYXRlQ29tcGF0aWJsZVNhbXBsZU1vZGVsKHcsIGgpOwotCi0gICAgICAgIHJldHVybiBuZXcgT3JkaW5hcnlXcml0YWJsZVJhc3RlcihzbSwgbmV3IFBvaW50KDAsIDApKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGUgYSBjb21wYXRpYmxlIFdyaXRhYmxlUmFzdGVyIHdpdGggdGhlIHNhbWUgcGFyYW1ldGVycyBhcyB0aGlzCi0gICAgICogUmFzdGVyIGFuZCB0aGUgc3BlY2lmaWVkIHNpemUgYW5kIGxvY2F0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBuZXcgV3JpdGFibGVSYXN0ZXIuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIG5ldyBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBuZXcgV3JpdGFibGVSYXN0ZXIuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIG5ldyBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiBAcmV0dXJuIHRoZSBXcml0YWJsZVJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoKSB7Ci0KLSAgICAgICAgV3JpdGFibGVSYXN0ZXIgcmFzdGVyID0gY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKHcsIGgpOwotCi0gICAgICAgIHJldHVybiByYXN0ZXIuY3JlYXRlV3JpdGFibGVDaGlsZCgwLCAwLCB3LCBoLCB4LCB5LCBudWxsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGUgYSBjb21wYXRpYmxlIFdyaXRhYmxlUmFzdGVyIHdpdGggdGhlIHNhbWUgcGFyYW1ldGVycyBhcyB0aGlzCi0gICAgICogUmFzdGVyIGFuZCB0aGUgc3BlY2lmaWVkIHJlY3RhbmdsZSB3aGljaCBkZXRlcm1pbmVzIG5ldyBXcml0YWJsZVJhc3RlcidzCi0gICAgICogbG9jYXRpb24gYW5kIHNpemUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJlY3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgUmVjdGFuZ2xlLgotICAgICAqIEByZXR1cm4gdGhlIFdyaXRhYmxlUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBjcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIoUmVjdGFuZ2xlIHJlY3QpIHsKLSAgICAgICAgaWYgKHJlY3QgPT0gbnVsbCkgewotICAgICAgICAgICAgLy8gYXd0LjI4Qz1SZWN0IGlzIG51bGwKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOEMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBjcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIocmVjdC54LCByZWN0LnksIHJlY3Qud2lkdGgsIHJlY3QuaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSB0cmFuc2xhdGVkIGNoaWxkIG9mIHRoaXMgUmFzdGVyLiBUaGUgTmV3IFJhc3RlciBvYmplY3QgaXMgYQotICAgICAqIHJlZmVyZW5jZSB0byB0aGUgdGhpcyBSYXN0ZXIgd2l0aCBhIGRpZmZlcmVudCBsb2NhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2hpbGRNaW5YCi0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBuZXcgUmFzdGVyLgotICAgICAqIEBwYXJhbSBjaGlsZE1pblkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIG5ldyBSYXN0ZXIuCi0gICAgICogQHJldHVybiB0aGUgUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBSYXN0ZXIgY3JlYXRlVHJhbnNsYXRlZENoaWxkKGludCBjaGlsZE1pblgsIGludCBjaGlsZE1pblkpIHsKLSAgICAgICAgcmV0dXJuIGNyZWF0ZUNoaWxkKG1pblgsIG1pblksIHdpZHRoLCBoZWlnaHQsIGNoaWxkTWluWCwgY2hpbGRNaW5ZLCBudWxsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBib3VuZHMgb2YgdGhpcyBSYXN0ZXIgYXMgYSByZWN0YW5nbGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYm91bmRzIG9mIHRoaXMgUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Qm91bmRzKCkgewotICAgICAgICByZXR1cm4gbmV3IFJlY3RhbmdsZShtaW5YLCBtaW5ZLCB3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBEYXRhQnVmZmVyIGFzc29jaWF0ZWQgd2l0aCB0aGlzIFJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBEYXRhQnVmZmVyIGFzc29jaWF0ZWQgd2l0aCB0aGlzIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgRGF0YUJ1ZmZlciBnZXREYXRhQnVmZmVyKCkgewotICAgICAgICByZXR1cm4gZGF0YUJ1ZmZlcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkYXRhIGVsZW1lbnRzIHdoaWNoIHJlcHJlc2VudCB0aGUgcGl4ZWwgZGF0YSBvZiB0aGUgc3BlY2lmaWVkCi0gICAgICogcmVjdGFuZ2xlIGFyZWEgYXMgYSBwcmltaXRpdmUgYXJyYXkuIFRoZSBmb2xsb3dpbmcgaW1hZ2UgZGF0YSB0eXBlcyBhcmUKLSAgICAgKiBzdXBwb3J0ZWQ6IERhdGFCdWZmZXIuVFlQRV9CWVRFLCBEYXRhQnVmZmVyLlRZUEVfVVNIT1JULAotICAgICAqIERhdGFCdWZmZXIuVFlQRV9JTlQsIERhdGFCdWZmZXIuVFlQRV9TSE9SVCwgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FULCBvcgotICAgICAqIERhdGFCdWZmZXIuVFlQRV9ET1VCTEUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIG91dERhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSByZXN1bHRpbmcgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgZGF0YSBlbGVtZW50cyBvZiB0aGUgc3BlY2lmaWVkIGFyZWEgb2YgdGhpcyBSYXN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIE9iamVjdCBnZXREYXRhRWxlbWVudHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIE9iamVjdCBvdXREYXRhKSB7Ci0gICAgICAgIHJldHVybiBzYW1wbGVNb2RlbC5nZXREYXRhRWxlbWVudHMoeCAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgeSAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWSwgdywKLSAgICAgICAgICAgICAgICBoLCBvdXREYXRhLCBkYXRhQnVmZmVyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkYXRhIGVsZW1lbnRzIHdoaWNoIHJlcHJlc2VudCB0aGUgc3BlY2lmaWVkIHBpeGVsIG9mIHRoaXMgUmFzdGVyCi0gICAgICogYXMgYSBwcmltaXRpdmUgYXJyYXkuIFRoZSBmb2xsb3dpbmcgaW1hZ2UgZGF0YSB0eXBlcyBhcmUgc3VwcG9ydGVkOgotICAgICAqIERhdGFCdWZmZXIuVFlQRV9CWVRFLCBEYXRhQnVmZmVyLlRZUEVfVVNIT1JULCBEYXRhQnVmZmVyLlRZUEVfSU5ULAotICAgICAqIERhdGFCdWZmZXIuVFlQRV9TSE9SVCwgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FULCBvciBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBwaXhlbC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIG91dERhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSByZXN1bHRpbmcgZGF0YS4KLSAgICAgKiBAcmV0dXJuIHRoZSBkYXRhIGVsZW1lbnRzIG9mIHRoZSBzcGVjaWZpZWQgcGl4ZWwgb2YgdGhpcyBSYXN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIE9iamVjdCBnZXREYXRhRWxlbWVudHMoaW50IHgsIGludCB5LCBPYmplY3Qgb3V0RGF0YSkgewotICAgICAgICByZXR1cm4gc2FtcGxlTW9kZWwuZ2V0RGF0YUVsZW1lbnRzKHggLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVgsIHkgLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVksCi0gICAgICAgICAgICAgICAgb3V0RGF0YSwgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaGVpZ2h0IG9mIHRoaXMgUmFzdGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhlaWdodCBvZiB0aGlzIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldEhlaWdodCgpIHsKLSAgICAgICAgcmV0dXJuIGhlaWdodDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtaW5pbXVtIFggY29vcmRpbmF0ZSBvZiB0aGlzIFJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtaW5pbXVtIFggY29vcmRpbmF0ZSBvZiB0aGlzIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldE1pblgoKSB7Ci0gICAgICAgIHJldHVybiBtaW5YOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1pbmltdW0gWSBjb29yZGluYXRlIG9mIHRoaXMgUmFzdGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG1pbmltdW0gWSBjb29yZGluYXRlIG9mIHRoaXMgUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0TWluWSgpIHsKLSAgICAgICAgcmV0dXJuIG1pblk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGJhbmRzIGluIHRoaXMgUmFzdGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiBiYW5kcyBpbiB0aGlzIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldE51bUJhbmRzKCkgewotICAgICAgICByZXR1cm4gbnVtQmFuZHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGRhdGEgZWxlbWVudHMgZm9yIG9uZSBwaXhlbC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgZGF0YSBlbGVtZW50cyBmb3Igb25lIHBpeGVsLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0TnVtRGF0YUVsZW1lbnRzKCkgewotICAgICAgICByZXR1cm4gbnVtRGF0YUVsZW1lbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHBhcmVudCBSYXN0ZXIgZm9yIHRoaXMgUmFzdGVyIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBwYXJlbnQgUmFzdGVyIGZvciB0aGlzIFJhc3RlciBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIFJhc3RlciBnZXRQYXJlbnQoKSB7Ci0gICAgICAgIHJldHVybiBwYXJlbnQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIGRvdWJsZSBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIHBpeGVsIGluIHRoaXMgUmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwncyBYIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbCdzIFkgY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0gZEFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZG91YmxlIGFycmF5IHdoZXJlIHJlc3VsdCBhcnJheSB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBkb3VibGUgYXJyYXkgb2Ygc2FtcGxlcyBmb3IgdGhlIHNwZWNpZmllZCBwaXhlbCBpbiB0aGlzCi0gICAgICogICAgICAgICBSYXN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZVtdIGdldFBpeGVsKGludCB4LCBpbnQgeSwgZG91YmxlIGRBcnJheVtdKSB7Ci0gICAgICAgIHJldHVybiBzYW1wbGVNb2RlbC5nZXRQaXhlbCh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCBkQXJyYXksCi0gICAgICAgICAgICAgICAgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIGZsb2F0IGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgcGl4ZWwgaW4gdGhpcyBSYXN0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbCdzIFggY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsJ3MgWSBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSBmQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbG9hdCBhcnJheSB3aGVyZSB0aGUgcmVzdWx0IGFycmF5IHdpbGwgYmUgc3RvcmVkLgotICAgICAqIEByZXR1cm4gdGhlIGZsb2F0IGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgcGl4ZWwgaW4gdGhpcwotICAgICAqICAgICAgICAgUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldFBpeGVsKGludCB4LCBpbnQgeSwgZmxvYXQgZkFycmF5W10pIHsKLSAgICAgICAgcmV0dXJuIHNhbXBsZU1vZGVsLmdldFBpeGVsKHggLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVgsIHkgLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVksIGZBcnJheSwKLSAgICAgICAgICAgICAgICBkYXRhQnVmZmVyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGludGVnZXIgYXJyYXkgb2Ygc2FtcGxlcyBmb3IgdGhlIHNwZWNpZmllZCBwaXhlbCBpbiB0aGlzIFJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsJ3MgWCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwncyBZIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIGlBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGludGVnZXIgYXJyYXkgd2hlcmUgdGhlIHJlc3VsdCBhcnJheSB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbnRlZ2VyIGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgcGl4ZWwgaW4gdGhpcwotICAgICAqICAgICAgICAgUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXRQaXhlbChpbnQgeCwgaW50IHksIGludCBpQXJyYXlbXSkgewotICAgICAgICByZXR1cm4gc2FtcGxlTW9kZWwuZ2V0UGl4ZWwoeCAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgeSAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWSwgaUFycmF5LAotICAgICAgICAgICAgICAgIGRhdGFCdWZmZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gZG91YmxlIGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIgYXJlYSBvZgotICAgICAqIHBpeGVscyBpbiB0aGlzIFJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gZEFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgcmVzdWx0aW5nIGFycmF5LgotICAgICAqIEByZXR1cm4gdGhlIGRvdWJsZSBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YKLSAgICAgKiAgICAgICAgIHBpeGVscyBpbiB0aGlzIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlW10gZ2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBkb3VibGUgZEFycmF5W10pIHsKLSAgICAgICAgcmV0dXJuIHNhbXBsZU1vZGVsLmdldFBpeGVscyh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCB3LCBoLAotICAgICAgICAgICAgICAgIGRBcnJheSwgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBmbG9hdCBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YKLSAgICAgKiBwaXhlbHMgaW4gdGhpcyBSYXN0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGZBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIHJlc3VsdGluZyBhcnJheS4KLSAgICAgKiBAcmV0dXJuIHRoZSBmbG9hdCBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YKLSAgICAgKiAgICAgICAgIHBpeGVscyBpbiB0aGlzIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXRbXSBnZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGZsb2F0IGZBcnJheVtdKSB7Ci0gICAgICAgIHJldHVybiBzYW1wbGVNb2RlbC5nZXRQaXhlbHMoeCAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgeSAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWSwgdywgaCwKLSAgICAgICAgICAgICAgICBmQXJyYXksIGRhdGFCdWZmZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gaW50ZWdlciBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YKLSAgICAgKiBwaXhlbHMgaW4gdGhpcyByYXN0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHBpeGVsJ3MgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHBpeGVsJ3MgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBpQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSByZXN1bHRpbmcgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgaW50ZWdlciBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEKLSAgICAgKiAgICAgICAgIG9mIHBpeGVscyBpbiB0aGlzIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50W10gZ2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnQgaUFycmF5W10pIHsKLSAgICAgICAgcmV0dXJuIHNhbXBsZU1vZGVsLmdldFBpeGVscyh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCB3LCBoLAotICAgICAgICAgICAgICAgIGlBcnJheSwgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2FtcGxlIGZvciB0aGUgc3BlY2lmaWVkIGJhbmQgb2YgdGhlIHNwZWNpZmllZCBwaXhlbCBhcyBhbgotICAgICAqIGludGVnZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHBpeGVsLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBwaXhlbC4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJhbmQuCi0gICAgICogQHJldHVybiB0aGUgc2FtcGxlIGZvciB0aGUgc3BlY2lmaWVkIGJhbmQgb2YgdGhlIHNwZWNpZmllZCBwaXhlbCBhcyBhbgotICAgICAqICAgICAgICAgaW50ZWdlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFNhbXBsZShpbnQgeCwgaW50IHksIGludCBiKSB7Ci0gICAgICAgIHJldHVybiBzYW1wbGVNb2RlbC5nZXRTYW1wbGUoeCAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgeSAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWSwgYiwKLSAgICAgICAgICAgICAgICBkYXRhQnVmZmVyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzYW1wbGUgZm9yIHRoZSBzcGVjaWZpZWQgYmFuZCBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsIGFzIGEKLSAgICAgKiBkb3VibGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHBpeGVsLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBwaXhlbC4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJhbmQuCi0gICAgICogQHJldHVybiB0aGUgc2FtcGxlIGZvciB0aGUgc3BlY2lmaWVkIGJhbmQgb2YgdGhlIHNwZWNpZmllZCBwaXhlbCBhcyBhCi0gICAgICogICAgICAgICBkb3VibGUuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXRTYW1wbGVEb3VibGUoaW50IHgsIGludCB5LCBpbnQgYikgewotICAgICAgICByZXR1cm4gc2FtcGxlTW9kZWwuZ2V0U2FtcGxlRG91YmxlKHggLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVgsIHkgLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVksIGIsCi0gICAgICAgICAgICAgICAgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2FtcGxlIGZvciB0aGUgc3BlY2lmaWVkIGJhbmQgb2YgdGhlIHNwZWNpZmllZCBwaXhlbCBhcyBhIGZsb2F0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBwaXhlbC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBiYW5kLgotICAgICAqIEByZXR1cm4gdGhlIHNhbXBsZSBmb3IgdGhlIHNwZWNpZmllZCBiYW5kIG9mIHRoZSBzcGVjaWZpZWQgcGl4ZWwgYXMgYQotICAgICAqICAgICAgICAgZmxvYXQuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldFNhbXBsZUZsb2F0KGludCB4LCBpbnQgeSwgaW50IGIpIHsKLSAgICAgICAgcmV0dXJuIHNhbXBsZU1vZGVsLmdldFNhbXBsZUZsb2F0KHggLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVgsIHkgLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVksIGIsCi0gICAgICAgICAgICAgICAgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgU2FtcGxlTW9kZWwgYXNzb2NpYXRlZCB3aXRoIHRoaXMgUmFzdGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFNhbXBsZU1vZGVsIGFzc29jaWF0ZWQgd2l0aCB0aGlzIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgU2FtcGxlTW9kZWwgZ2V0U2FtcGxlTW9kZWwoKSB7Ci0gICAgICAgIHJldHVybiBzYW1wbGVNb2RlbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0cmFuc2xhdGlvbiBvZiB0aGUgWCBjb29yZGluYXRlIGZyb20gdGhlIFNhbXBsZU1vZGVsIGNvb3JkaW5hdGUKLSAgICAgKiBzeXN0ZW0gdG8gdGhlIFJhc3RlcnMncyBjb29yZGluYXRlIHN5c3RlbS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB2YWx1ZSBvZiB0aGUgdHJhbnNsYXRpb24gb2YgdGhlIFggY29vcmRpbmF0ZSBmcm9tIHRoZQotICAgICAqICAgICAgICAgU2FtcGxlTW9kZWwgY29vcmRpbmF0ZSBzeXN0ZW0gdG8gdGhlIFJhc3RlcnMncyBjb29yZGluYXRlIHN5c3RlbS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWCgpIHsKLSAgICAgICAgcmV0dXJuIHNhbXBsZU1vZGVsVHJhbnNsYXRlWDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0cmFuc2xhdGlvbiBvZiB0aGUgWSBjb29yZGluYXRlIGZyb20gdGhlIFNhbXBsZU1vZGVsIGNvb3JkaW5hdGUKLSAgICAgKiBzeXN0ZW0gdG8gdGhlIFJhc3RlcnMncyBjb29yZGluYXRlIHN5c3RlbS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB2YWx1ZSBvZiB0aGUgdHJhbnNsYXRpb24gb2YgdGhlIFkgY29vcmRpbmF0ZSBmcm9tIHRoZQotICAgICAqICAgICAgICAgU2FtcGxlTW9kZWwgY29vcmRpbmF0ZSBzeXN0ZW0gdG8gdGhlIFJhc3RlcnMncyBjb29yZGluYXRlIHN5c3RlbS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgaW50IGdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWSgpIHsKLSAgICAgICAgcmV0dXJuIHNhbXBsZU1vZGVsVHJhbnNsYXRlWTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkb3VibGUgYXJyYXkgb2Ygc2FtcGxlcyBmb3IgdGhlIHNwZWNpZmllZCBiYW5kIG9mIHRoZSBzcGVjaWZpZWQKLSAgICAgKiByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscyBpbiB0aGlzIFJhc3RlciBhcyBhIGRvdWJsZSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJhbmQuCi0gICAgICogQHBhcmFtIGRBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIHJlc3VsdGluZyBkb3VibGUgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgZG91YmxlIGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgYmFuZCBvZiB0aGUKLSAgICAgKiAgICAgICAgIHNwZWNpZmllZCByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4KLSAgICAgKi8KLSAgICBwdWJsaWMgZG91YmxlW10gZ2V0U2FtcGxlcyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGIsIGRvdWJsZSBkQXJyYXlbXSkgewotCi0gICAgICAgIHJldHVybiBzYW1wbGVNb2RlbC5nZXRTYW1wbGVzKHggLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVgsIHkgLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVksIHcsIGgsCi0gICAgICAgICAgICAgICAgYiwgZEFycmF5LCBkYXRhQnVmZmVyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmbG9hdCBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIGJhbmQgb2YgdGhlIHNwZWNpZmllZAotICAgICAqIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzIGluIHRoaXMgUmFzdGVyIGFzIGEgZmxvYXQgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBiYW5kLgotICAgICAqIEBwYXJhbSBmQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSByZXN1bHRpbmcgZmxvYXQgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgZmxvYXQgYXJyYXkgb2Ygc2FtcGxlcyBmb3IgdGhlIHNwZWNpZmllZCBiYW5kIG9mIHRoZQotICAgICAqICAgICAgICAgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBmbG9hdCBmQXJyYXlbXSkgewotCi0gICAgICAgIHJldHVybiBzYW1wbGVNb2RlbC5nZXRTYW1wbGVzKHggLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVgsIHkgLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVksIHcsIGgsCi0gICAgICAgICAgICAgICAgYiwgZkFycmF5LCBkYXRhQnVmZmVyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBpbnRlZ2VyIGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgYmFuZCBvZiB0aGUgc3BlY2lmaWVkCi0gICAgICogcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMgaW4gdGhpcyBSYXN0ZXIgYXMgYSBpbnRlZ2VyIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgYmFuZC4KLSAgICAgKiBAcGFyYW0gaUFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgcmVzdWx0aW5nIGludGVnZXIgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgaW50ZWdlciBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIGJhbmQgb2YgdGhlCi0gICAgICogICAgICAgICBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBpbnQgaUFycmF5W10pIHsKLSAgICAgICAgcmV0dXJuIHNhbXBsZU1vZGVsLmdldFNhbXBsZXMoeCAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgeSAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWSwgdywgaCwKLSAgICAgICAgICAgICAgICBiLCBpQXJyYXksIGRhdGFCdWZmZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRyYW5zZmVyIHR5cGUgZm9yIHBpeGVscyBvZiB0aGlzIFJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAc2VlIFNhbXBsZU1vZGVsI2dldFRyYW5zZmVyVHlwZSgpCi0gICAgICogQHJldHVybiB0aGUgdHJhbnNmZXIgdHlwZSBmb3IgcGl4ZWxzIG9mIHRoaXMgUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0VHJhbnNmZXJUeXBlKCkgewotICAgICAgICByZXR1cm4gc2FtcGxlTW9kZWwuZ2V0VHJhbnNmZXJUeXBlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgd2lkdGggb2YgdGhpcyBSYXN0ZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgd2lkdGggb2YgdGhpcyBSYXN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGludCBnZXRXaWR0aCgpIHsKLSAgICAgICAgcmV0dXJuIHdpZHRoOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFZhbGlkYXRlIGRhdGEgYnVmZmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhQnVmZmVyCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSBidWZmZXIuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3LgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaC4KLSAgICAgKiBAcGFyYW0gc2NhbmxpbmVTdHJpZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FubGluZSBzdHJpZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgdm9pZCB2YWxpZGF0ZURhdGFCdWZmZXIoZmluYWwgRGF0YUJ1ZmZlciBkYXRhQnVmZmVyLCBmaW5hbCBpbnQgdywgZmluYWwgaW50IGgsCi0gICAgICAgICAgICBmaW5hbCBpbnQgc2NhbmxpbmVTdHJpZGUpIHsKLSAgICAgICAgaWYgKGRhdGFCdWZmZXIuZ2V0U2l6ZSgpIDwgKHNjYW5saW5lU3RyaWRlICogKGggLSAxKSArIHcgLSAxKSkgewotICAgICAgICAgICAgLy8gYXd0LjI5OD1kYXRhQnVmZmVyIGlzIHRvbyBzbWFsbAotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yOTgiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9SYXN0ZXJGb3JtYXRFeGNlcHRpb24uamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9SYXN0ZXJGb3JtYXRFeGNlcHRpb24uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzY2NzE0MS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvUmFzdGVyRm9ybWF0RXhjZXB0aW9uLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0OCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotLyoqCi0gKiBUaGUgUmFzdGVyRm9ybWF0RXhjZXB0aW9uIGNsYXNzIHJlcHJlc2VudHMgdGhlIGV4Y2VwdGlvbiB0aGF0IGlzIHRocm93biB3aGVuCi0gKiB0aGVyZSdzIGFuIGludmFsaWQgbGF5b3V0IGluIHRoZSBSYXN0ZXIuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgUmFzdGVyRm9ybWF0RXhjZXB0aW9uIGV4dGVuZHMgUnVudGltZUV4Y2VwdGlvbiB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgc2VyaWFsVmVyc2lvblVJRC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSA5NjU5ODk5NjExNjE2NDMxNUw7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uIHdpdGggdGhlIHNwZWNpZmllZCBkZXRhaWwKLSAgICAgKiBtZXNzYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgZGV0YWlsIG1lc3NhZ2UuCi0gICAgICovCi0gICAgcHVibGljIFJhc3RlckZvcm1hdEV4Y2VwdGlvbihTdHJpbmcgcykgewotICAgICAgICBzdXBlcihzKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9SYXN0ZXJPcC5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL1Jhc3Rlck9wLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDE5YTg0YzkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL1Jhc3Rlck9wLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4OCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5SZW5kZXJpbmdIaW50czsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBvaW50MkQ7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLQotLyoqCi0gKiBUaGUgUmFzdGVyT3AgaW50ZXJmYWNlIHByb3ZpZGVzIG1ldGhvZHMgZm9yIHBlcmZvcm1pbmcgdHJhbnNmb3JtYXRpb25zIGZyb20KLSAqIHNvdXJjZSBkYXRhIHRvIGRlc3RpbmF0aW9uIGRhdGEgZm9yIFJhc3RlciBvYmplY3RzLiBUaGUgc291cmNlIGFuZAotICogZGVzdGluYXRpb24gb2JqZWN0cyBzaG91bGQgY29udGFpbiB0aGUgYXBwcm9wcmlhdGUgbnVtYmVyIG9mIGJhbmRzIGZvciB0aGUKLSAqIHBhcnRpY3VsYXIgY2xhc3NlcyB3aGljaCBpbXBsZW1lbnQgdGhpcyBpbnRlcmZhY2UuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIFJhc3Rlck9wIHsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBkZXN0aW5hdGlvbiBXcml0YWJsZVJhc3RlciB3aXRoIHRoZSBzcGVjaWZpZWQgUmFzdGVyOyB0aGlzCi0gICAgICogZGVzdGluYXRpb24gaW1hZ2UgZGF0YSBpcyBlbXB0eSBhbmQgaGFzIHRoZSBjb3JyZWN0IHNpemUgYW5kIG51bWJlciBvZgotICAgICAqIGJhbmRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgUmFzdGVyLgotICAgICAqIEByZXR1cm4gdGhlIFdyaXRhYmxlUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBjcmVhdGVDb21wYXRpYmxlRGVzdFJhc3RlcihSYXN0ZXIgc3JjKTsKLQotICAgIC8qKgotICAgICAqIFBlcmZvcm1zIGEgZmlsdGVyIG9wZXJhdGlvbiBvbiB0aGUgc291cmNlIFJhc3RlciBhbmQgc3RvcmVzIHRoZSByZXN1bHRpbmcKLSAgICAgKiBpbWFnZSBkYXRhIHRvIHRoZSBkZXN0aW5hdGlvbiBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgc291cmNlIFJhc3Rlci4KLSAgICAgKiBAcGFyYW0gZHN0Ci0gICAgICogICAgICAgICAgICB0aGUgZGVzdGluYXRpb24gV3JpdGFibGVSYXN0ZXIsIHdoZXJlIHRoZSByZXN1bHQgaXMgc3RvcmVkLgotICAgICAqIEByZXR1cm4gdGhlIGZpbHRlcmVkIFdyaXRhYmxlUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBmaWx0ZXIoUmFzdGVyIHNyYywgV3JpdGFibGVSYXN0ZXIgZHN0KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJvdW5kcyBvZiB0aGUgZmlsdGVyZWQgUmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgUmFzdGVyIHRvIGJlIGZpbHRlcmVkLgotICAgICAqIEByZXR1cm4gdGhlIHJlY3RhbmdsZSBib3VuZHMgb2YgdGhlIGZpbHRlcmVkIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoUmFzdGVyIHNyYyk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBwb2ludCBvZiB0aGUgZGVzdGluYXRpb24gaW1hZ2Ugd2hpY2ggY29ycmVzcG9uZHMgdG8gdGhlCi0gICAgICogc3BlY2lmaWVkIHBvaW50IGluIHRoZSBzb3VyY2UgcmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzcmNQb2ludAotICAgICAqICAgICAgICAgICAgdGhlIHBvaW50IG9mIHRoZSBzb3VyY2UgcmFzdGVyLgotICAgICAqIEBwYXJhbSBkc3RQb2ludAotICAgICAqICAgICAgICAgICAgdGhlIHBvaW50IHdoZXJlIHRoZSByZXN1bHQgd2lsbCBiZSBzdG9yZWQuCi0gICAgICogQHJldHVybiB0aGUgZGVzdGluYXRpb24gcG9pbnQuCi0gICAgICovCi0gICAgcHVibGljIFBvaW50MkQgZ2V0UG9pbnQyRChQb2ludDJEIHNyY1BvaW50LCBQb2ludDJEIGRzdFBvaW50KTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFJlbmRlcmluZ0hpbnRzIG9mIHRoZSBSYXN0ZXJPcC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBSZW5kZXJpbmdIaW50cyBvZiB0aGUgUmFzdGVyT3AuCi0gICAgICovCi0gICAgcHVibGljIFJlbmRlcmluZ0hpbnRzIGdldFJlbmRlcmluZ0hpbnRzKCk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvUmVuZGVyZWRJbWFnZS5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL1JlbmRlcmVkSW1hZ2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNWVhZmE2NC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvUmVuZGVyZWRJbWFnZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTk4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEudXRpbC5WZWN0b3I7Ci0KLS8qKgotICogVGhlIFJlbmRlcmVkSW1hZ2UgaW50ZXJmYWNlIHNob3VsZCBiZSBpbXBsZW1lbnRlZCBieSBhbGwgb2JqZWN0cyB3aGljaAotICogY29udGFpbnMgaW1hZ2UgZGF0YS4gVGhlIGltYWdlIGRhdGEgaXMgcmVwcmVzZW50ZWQgYXMgYSBzaW5nbGUgdGlsZSBvciBhbgotICogYXJyYXkgb2YgdGlsZXMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIFJlbmRlcmVkSW1hZ2UgewotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcHJvcGVydHkgd2l0aCB0aGUgc3BlY2lmaWVkIG5hbWUgZnJvbSB0aGUgcHJvcGVydHkgc2V0IG9mIHRoaXMKLSAgICAgKiBSZW5kZXJlZEltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcHJvcGVydHkncyBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIHByb3BlcnR5IHZhbHVlIGNvcnJlc3BvbmRlZCB0byB0aGlzIHByb3BlcnR5J3MgbmFtZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IGdldFByb3BlcnR5KFN0cmluZyBuYW1lKTsKLQotICAgIC8qKgotICAgICAqIENvcGllcyB0aGUgcmVnaW9uIG9mIHRoaXMgUmVuZGVyZWRJbWFnZSB0byB0aGUgc3BlY2lmaWVkIFdyaXRhYmxlUmFzdGVyLgotICAgICAqIFRoZSBib3VuZHMgb2YgdGhlIHJlZ2lvbiBhcmUgdGhlIGJvdW5kcyBvZiB0aGUgV3JpdGFibGVSYXN0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJhc3RlcgotICAgICAqICAgICAgICAgICAgdGhlIFdyaXRhYmxlUmFzdGVyLgotICAgICAqIEByZXR1cm4gdGhlIGNyZWF0ZWQgV3JpdGFibGVSYXN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIFdyaXRhYmxlUmFzdGVyIGNvcHlEYXRhKFdyaXRhYmxlUmFzdGVyIHJhc3Rlcik7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBpbWFnZSBkYXRhIG9mIHRoZSBpbWFnZSdzIHJlZ2lvbiBhcyBvbmUgdGlsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmVjdAotICAgICAqICAgICAgICAgICAgdGhlIHJlY3Rhbmd1bGFyIHJlZ2lvbiBvZiBSZW5kZXJlZEltYWdlLgotICAgICAqIEByZXR1cm4gdGhlIGltYWdlIGRhdGEgb2YgdGhlIGltYWdlJ3MgcmVnaW9uIGFzIG9uZSB0aWxlLgotICAgICAqLwotICAgIHB1YmxpYyBSYXN0ZXIgZ2V0RGF0YShSZWN0YW5nbGUgcmVjdCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFsbCBSZW5kZXJlZEltYWdlIG9iamVjdHMgd2hpY2ggYXJlIHRoZSBzb3VyY2Ugb2YgdGhpcyBSZW5kZXJlZEltYWdlCi0gICAgICogb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBWZWN0b3Igb2YgUmVuZGVyZWRJbWFnZSBvYmplY3RzIHdoaWNoIGFyZSB0aGUgc291cmNlIG9mIHRoaXMKLSAgICAgKiAgICAgICAgIFJlbmRlcmVkSW1hZ2Ugb2JqZWN0IG9yIG51bGwsIGlmIHRoZXJlIGlzIG5vIGluZm9ybWF0aW9uIGFib3V0Ci0gICAgICogICAgICAgICB0aGVtLgotICAgICAqLwotICAgIHB1YmxpYyBWZWN0b3I8UmVuZGVyZWRJbWFnZT4gZ2V0U291cmNlcygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2V0IG9mIGFsbCBwcm9wZXJ0eSBuYW1lcyBmb3IgdGhpcyBSZW5kZXJlZEltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGFsbCBwcm9wZXJ0eSBuYW1lcyBmb3IgdGhpcyBSZW5kZXJlZEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRQcm9wZXJ0eU5hbWVzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBTYW1wbGVNb2RlbCBvZiB0aGlzIFJlbmRlcmVkSW1hZ2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgU2FtcGxlTW9kZWwgb2YgdGhpcyBSZW5kZXJlZEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBTYW1wbGVNb2RlbCBnZXRTYW1wbGVNb2RlbCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdGlsZSBjb3JyZXNwb25kZWQgdG8gdGhlIHNwZWNpZmllZCBpbmRpY2VzIGluIHRoZSB0aWxlIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB0aWxlWAotICAgICAqICAgICAgICAgICAgdGhlIFggaW5kZXggb2YgdGhlIHRpbGUuCi0gICAgICogQHBhcmFtIHRpbGVZCi0gICAgICogICAgICAgICAgICB0aGUgWSBpbmRleCBvZiB0aGUgdGlsZS4KLSAgICAgKiBAcmV0dXJuIHRoZSB0aWxlIGNvcnJlc3BvbmRlZCB0byB0aGUgc3BlY2lmaWVkIGluZGljZXMgaW4gdGhlIHRpbGUgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIFJhc3RlciBnZXRUaWxlKGludCB0aWxlWCwgaW50IHRpbGVZKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGltYWdlIGRhdGEgb2YgdGhpcyBpbWFnZSBhcyBvbmUgdGlsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBpbWFnZSBkYXRhIG9mIHRoaXMgaW1hZ2UgYXMgb25lIHRpbGUuCi0gICAgICovCi0gICAgcHVibGljIFJhc3RlciBnZXREYXRhKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBDb2xvck1vZGVsIG9mIHRoaXMgUmVuZGVyZWRJbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBDb2xvck1vZGVsIG9mIHRoaXMgUmVuZGVyZWRJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ29sb3JNb2RlbCBnZXRDb2xvck1vZGVsKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB3aWR0aCBvZiB0aGUgUmVuZGVyZWRJbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB3aWR0aCBvZiB0aGUgUmVuZGVyZWRJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFdpZHRoKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0aWxlIHdpZHRoLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHRpbGUgd2lkdGggaW4gcGl4ZWxzLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VGlsZVdpZHRoKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0aWxlIGhlaWdodC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB0aWxlIGhlaWdodCBpbiBwaXhlbHMuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRUaWxlSGVpZ2h0KCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBZIG9mZnNldCBvZiB0aGUgdGlsZSBncmlkLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIFkgb2Zmc2V0IG9mIHRoZSB0aWxlIGdyaWQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRUaWxlR3JpZFlPZmZzZXQoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFggb2Zmc2V0IG9mIHRoZSB0aWxlIGdyaWQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgWCBvZmZzZXQgb2YgdGhlIHRpbGUgZ3JpZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFRpbGVHcmlkWE9mZnNldCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIHRpbGVzIGFsb25nIFkgZGlyZWN0aW9uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiB0aWxlcyBhbG9uZyBZIGRpcmVjdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE51bVlUaWxlcygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIHRpbGVzIGFsb25nIFggZGlyZWN0aW9uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiB0aWxlcyBhbG9uZyBYIGRpcmVjdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE51bVhUaWxlcygpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWluaW11bSBZIGNvb3JkaW5hdGUgb2YgdGhpcyBSZW5kZXJlZEltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG1pbmltdW0gWSBjb29yZGluYXRlIG9mIHRoaXMgUmVuZGVyZWRJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE1pblkoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1pbmltdW0gWCBjb29yZGluYXRlIG9mIHRoaXMgUmVuZGVyZWRJbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtaW5pbXVtIFggY29vcmRpbmF0ZSBvZiB0aGlzIFJlbmRlcmVkSW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRNaW5YKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtaW5pbXVtIHRpbGUncyBpbmRleCBhbG9uZyB0aGUgWSBkaXJlY3Rpb24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWluaW11bSB0aWxlJ3MgaW5kZXggYWxvbmcgdGhlIFkgZGlyZWN0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TWluVGlsZVkoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1pbmltdW0gdGlsZSdzIGluZGV4IGFsb25nIHRoZSBYIGRpcmVjdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtaW5pbXVtIHRpbGUncyBpbmRleCBhbG9uZyB0aGUgWCBkaXJlY3Rpb24uCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRNaW5UaWxlWCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaGVpZ2h0IG9mIHRoZSBSZW5kZXJlZEltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhlaWdodCBvZiB0aGUgUmVuZGVyZWRJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEhlaWdodCgpOwotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvUmVwbGljYXRlU2NhbGVGaWx0ZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9SZXBsaWNhdGVTY2FsZUZpbHRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1MWMwZjQ5Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9SZXBsaWNhdGVTY2FsZUZpbHRlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgamF2YS51dGlsLkhhc2h0YWJsZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBSZXBsaWNhdGVTY2FsZUZpbHRlciBjbGFzcyBzY2FsZXMgYW4gc291cmNlIGltYWdlIGJ5IHJlcGxpY2F0aW5nIHJvd3MgYW5kCi0gKiBjb2x1bW5zIG9mIHBpeGVscyB0byBzY2FsZSB1cCBvciBvbWl0dGluZyByb3dzIGFuZCBjb2x1bW5zIG9mIHBpeGVscyB0byBzY2FsZQotICogZG93bi4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBSZXBsaWNhdGVTY2FsZUZpbHRlciBleHRlbmRzIEltYWdlRmlsdGVyIHsKLQotICAgIC8qKgotICAgICAqIFRoZSB3aWR0aCBvZiBhIHNvdXJjZSBpbWFnZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IHNyY1dpZHRoOwotCi0gICAgLyoqCi0gICAgICogVGhlIGhlaWdodCBvZiBhIHNvdXJjZSBpbWFnZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IHNyY0hlaWdodDsKLQotICAgIC8qKgotICAgICAqIFRoZSB3aWR0aCBvZiBhIGRlc3RpbmF0aW9uIGltYWdlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgZGVzdFdpZHRoOwotCi0gICAgLyoqCi0gICAgICogVGhlIGhlaWdodCBvZiBhIGRlc3RpbmF0aW9uIGltYWdlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgZGVzdEhlaWdodDsKLQotICAgIC8qKgotICAgICAqIFRoZSBpbnRlZ2VyIGFycmF5IG9mIHNvdXJjZSByb3dzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnRbXSBzcmNyb3dzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGludGVnZXIgYXJyYXkgb2Ygc291cmNlIGNvbHVtbnMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludFtdIHNyY2NvbHM7Ci0KLSAgICAvKioKLSAgICAgKiBBbiBPYmplY3QgKGJ5dGUgYXJyYXkgd2l0aCBhIGRlc3RpbmF0aW9uIHdpZHRoKSBwcm92aWRlcyBhIHJvdyBvZiBwaXhlbAotICAgICAqIGRhdGEgdG8gdGhlIEltYWdlQ29uc3VtZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIE9iamVjdCBvdXRwaXhidWY7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgUmVwbGljYXRlU2NhbGVGaWx0ZXIgdGhhdCBmaWx0ZXJzIHRoZSBpbWFnZSB3aXRoIHRoZQotICAgICAqIHNwZWNpZmllZCB3aWR0aCBhbmQgaGVpZ2h0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHNjYWxlZCBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHNjYWxlZCBpbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVwbGljYXRlU2NhbGVGaWx0ZXIoaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIGlmICh3aWR0aCA9PSAwIHx8IGhlaWdodCA9PSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjM0PVdpZHRoIG9yIEhlaWdodCBlcXVhbHMgemVybwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMzQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMuZGVzdFdpZHRoID0gd2lkdGg7Ci0gICAgICAgIHRoaXMuZGVzdEhlaWdodCA9IGhlaWdodDsKLSAgICB9Ci0KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygidW5jaGVja2VkIikKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRQcm9wZXJ0aWVzKEhhc2h0YWJsZTw/LCA/PiBwcm9wcykgewotICAgICAgICBIYXNodGFibGU8T2JqZWN0LCBPYmplY3Q+IGZwcm9wczsKLSAgICAgICAgaWYgKHByb3BzID09IG51bGwpIHsKLSAgICAgICAgICAgIGZwcm9wcyA9IG5ldyBIYXNodGFibGU8T2JqZWN0LCBPYmplY3Q+KCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmcHJvcHMgPSAoSGFzaHRhYmxlPE9iamVjdCwgT2JqZWN0Pilwcm9wcy5jbG9uZSgpOwotICAgICAgICB9Ci0gICAgICAgIFN0cmluZyBwcm9wTmFtZSA9ICJSZXNjYWxlIEZpbHRlcnMiOyAvLyROT04tTkxTLTEkCi0gICAgICAgIFN0cmluZyBwcm9wID0gImRlc3RXaWR0aD0iICsgZGVzdFdpZHRoICsgIjsgIiArIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgICAgICJkZXN0SGVpZ2h0PSIgKyBkZXN0SGVpZ2h0OyAvLyROT04tTkxTLTEkCi0gICAgICAgIE9iamVjdCBvID0gZnByb3BzLmdldChwcm9wTmFtZSk7Ci0gICAgICAgIGlmIChvICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChvIGluc3RhbmNlb2YgU3RyaW5nKSB7Ci0gICAgICAgICAgICAgICAgcHJvcCA9IChTdHJpbmcpbyArICI7ICIgKyBwcm9wOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHByb3AgPSBvLnRvU3RyaW5nKCkgKyAiOyAiICsgcHJvcDsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGZwcm9wcy5wdXQocHJvcE5hbWUsIHByb3ApOwotICAgICAgICBjb25zdW1lci5zZXRQcm9wZXJ0aWVzKGZwcm9wcyk7Ci0gICAgfQotCi0gICAgLy8gc2V0UGl4ZWxzIG1ldGhvZHMgcHJvZHVjZSBwaXhlbHMgYWNjb3JkaW5nIHRvIEphdmEgQVBJIFNwYWNpZmljYXRpb24KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgQ29sb3JNb2RlbCBtb2RlbCwgaW50W10gcGl4ZWxzLCBpbnQgb2ZmLAotICAgICAgICAgICAgaW50IHNjYW5zaXplKSB7Ci0KLSAgICAgICAgaWYgKHNyY2NvbHMgPT0gbnVsbCkgewotICAgICAgICAgICAgaW5pdEFycmF5cygpOwotICAgICAgICB9Ci0gICAgICAgIGludCBidWZmW107Ci0gICAgICAgIGlmIChvdXRwaXhidWYgPT0gbnVsbCB8fCAhKG91dHBpeGJ1ZiBpbnN0YW5jZW9mIGludFtdKSkgewotICAgICAgICAgICAgYnVmZiA9IG5ldyBpbnRbZGVzdFdpZHRoXTsKLSAgICAgICAgICAgIG91dHBpeGJ1ZiA9IGJ1ZmY7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBidWZmID0gKGludFtdKW91dHBpeGJ1ZjsKLSAgICAgICAgfQotCi0gICAgICAgIGludCB3YSA9IChzcmNXaWR0aCAtIDEpID4+PiAxOwotICAgICAgICBpbnQgaGEgPSAoc3JjSGVpZ2h0IC0gMSkgPj4+IDE7Ci0gICAgICAgIGludCBkc3RYID0gKHggKiBkZXN0V2lkdGggKyB3YSkgLyBzcmNXaWR0aDsKLSAgICAgICAgaW50IGRzdFkgPSAoeSAqIGRlc3RIZWlnaHQgKyBoYSkgLyBzcmNIZWlnaHQ7Ci0KLSAgICAgICAgaW50IHN4LCBzeSwgZHgsIGR5OwotICAgICAgICBkeSA9IGRzdFk7Ci0gICAgICAgIHdoaWxlICgoZHkgPCBkZXN0SGVpZ2h0KSAmJiAoKHN5ID0gc3Jjcm93c1tkeV0pIDwgeSArIGgpKSB7Ci0gICAgICAgICAgICBkeCA9IGRzdFg7Ci0gICAgICAgICAgICBpbnQgc3JjT2ZmID0gb2ZmICsgKHN5IC0geSkgKiBzY2Fuc2l6ZTsKLSAgICAgICAgICAgIHdoaWxlICgoZHggPCBkZXN0V2lkdGgpICYmICgoc3ggPSBzcmNjb2xzW2R4XSkgPCB4ICsgdykpIHsKLSAgICAgICAgICAgICAgICBidWZmW2R4XSA9IHBpeGVsc1tzcmNPZmYgKyAoc3ggLSB4KV07Ci0gICAgICAgICAgICAgICAgZHgrKzsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY29uc3VtZXIuc2V0UGl4ZWxzKGRzdFgsIGR5LCBkeCAtIGRzdFgsIDEsIG1vZGVsLCBidWZmLCBkc3RYLCBkZXN0V2lkdGgpOwotICAgICAgICAgICAgZHkrKzsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgQ29sb3JNb2RlbCBtb2RlbCwgYnl0ZVtdIHBpeGVscywgaW50IG9mZiwKLSAgICAgICAgICAgIGludCBzY2Fuc2l6ZSkgewotCi0gICAgICAgIGlmIChzcmNjb2xzID09IG51bGwpIHsKLSAgICAgICAgICAgIGluaXRBcnJheXMoKTsKLSAgICAgICAgfQotICAgICAgICBieXRlIGJ1ZmZbXTsKLSAgICAgICAgaWYgKG91dHBpeGJ1ZiA9PSBudWxsIHx8ICEob3V0cGl4YnVmIGluc3RhbmNlb2YgYnl0ZVtdKSkgewotICAgICAgICAgICAgYnVmZiA9IG5ldyBieXRlW2Rlc3RXaWR0aF07Ci0gICAgICAgICAgICBvdXRwaXhidWYgPSBidWZmOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgYnVmZiA9IChieXRlW10pb3V0cGl4YnVmOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHdhID0gKHNyY1dpZHRoIC0gMSkgPj4+IDE7Ci0gICAgICAgIGludCBoYSA9IChzcmNIZWlnaHQgLSAxKSA+Pj4gMTsKLSAgICAgICAgaW50IGRzdFggPSAoeCAqIGRlc3RXaWR0aCArIHdhKSAvIHNyY1dpZHRoOwotICAgICAgICBpbnQgZHN0WSA9ICh5ICogZGVzdEhlaWdodCArIGhhKSAvIHNyY0hlaWdodDsKLQotICAgICAgICBpbnQgc3gsIHN5LCBkeCwgZHk7Ci0gICAgICAgIGR5ID0gZHN0WTsKLSAgICAgICAgd2hpbGUgKChkeSA8IGRlc3RIZWlnaHQpICYmICgoc3kgPSBzcmNyb3dzW2R5XSkgPCB5ICsgaCkpIHsKLSAgICAgICAgICAgIGR4ID0gZHN0WDsKLSAgICAgICAgICAgIGludCBzcmNPZmYgPSBvZmYgKyAoc3kgLSB5KSAqIHNjYW5zaXplOwotICAgICAgICAgICAgd2hpbGUgKChkeCA8IGRlc3RXaWR0aCkgJiYgKChzeCA9IHNyY2NvbHNbZHhdKSA8IHggKyB3KSkgewotICAgICAgICAgICAgICAgIGJ1ZmZbZHhdID0gcGl4ZWxzW3NyY09mZiArIChzeCAtIHgpXTsKLSAgICAgICAgICAgICAgICBkeCsrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjb25zdW1lci5zZXRQaXhlbHMoZHN0WCwgZHksIGR4IC0gZHN0WCwgMSwgbW9kZWwsIGJ1ZmYsIGRzdFgsIGRlc3RXaWR0aCk7Ci0gICAgICAgICAgICBkeSsrOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0RGltZW5zaW9ucyhpbnQgdywgaW50IGgpIHsKLSAgICAgICAgc3JjV2lkdGggPSB3OwotICAgICAgICBzcmNIZWlnaHQgPSBoOwotCi0gICAgICAgIGlmIChkZXN0V2lkdGggPCAwICYmIGRlc3RIZWlnaHQgPCAwKSB7Ci0gICAgICAgICAgICBkZXN0V2lkdGggPSBzcmNXaWR0aDsKLSAgICAgICAgICAgIGRlc3RIZWlnaHQgPSBzcmNIZWlnaHQ7Ci0gICAgICAgIH0gZWxzZSBpZiAoZGVzdFdpZHRoIDwgMCkgewotICAgICAgICAgICAgZGVzdFdpZHRoID0gZGVzdEhlaWdodCAqIHNyY1dpZHRoIC8gc3JjSGVpZ2h0OwotICAgICAgICB9IGVsc2UgaWYgKGRlc3RIZWlnaHQgPCAwKSB7Ci0gICAgICAgICAgICBkZXN0SGVpZ2h0ID0gZGVzdFdpZHRoICogc3JjSGVpZ2h0IC8gc3JjV2lkdGg7Ci0gICAgICAgIH0KLSAgICAgICAgY29uc3VtZXIuc2V0RGltZW5zaW9ucyhkZXN0V2lkdGgsIGRlc3RIZWlnaHQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluaXRpYWxpemF0aW9uIG9mIHNyY2NvbHMgYW5kIHNyY3Jvd3MgYXJyYXlzLgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBpbml0QXJyYXlzKCkgewotICAgICAgICBpZiAoKGRlc3RXaWR0aCA8IDApIHx8IChkZXN0SGVpZ2h0IDwgMCkpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBzcmNjb2xzID0gbmV3IGludFtkZXN0V2lkdGhdOwotICAgICAgICBpbnQgY2EgPSBzcmNXaWR0aCA+Pj4gMTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBkZXN0V2lkdGg7IGkrKykgewotICAgICAgICAgICAgc3JjY29sc1tpXSA9IChpICogc3JjV2lkdGggKyBjYSkgLyBkZXN0V2lkdGg7Ci0gICAgICAgIH0KLQotICAgICAgICBzcmNyb3dzID0gbmV3IGludFtkZXN0SGVpZ2h0XTsKLSAgICAgICAgaW50IHJhID0gc3JjSGVpZ2h0ID4+PiAxOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGRlc3RIZWlnaHQ7IGkrKykgewotICAgICAgICAgICAgc3Jjcm93c1tpXSA9IChpICogc3JjSGVpZ2h0ICsgcmEpIC8gZGVzdEhlaWdodDsKLSAgICAgICAgfQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL1Jlc2NhbGVPcC5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL1Jlc2NhbGVPcC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkN2UyYmQ3Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9SZXNjYWxlT3AuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDY1OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqCi0gKiBAZGF0ZTogT2N0IDYsIDIwMDUKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgamF2YS5hd3QuZ2VvbS5Qb2ludDJEOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS5hd3QuKjsKLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlzOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5Bd3RJbWFnZUJhY2tkb29yQWNjZXNzb3I7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIENsYXNzIFJlc2NhbGVPcCBwZXJmb3JtcyByZXNjYWxpbmcgb2YgdGhlIHNvdXJjZSBpbWFnZSBkYXRhIGJ5Ci0gKiBtdWx0aXBseWluZyB0aGUgcGl4ZWwgdmFsdWVzIHdpdGggYSBzY2FsZSBmYWN0b3IgYW5kIHRoZW4gYWRkaW5nIGFuIG9mZnNldC4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBSZXNjYWxlT3AgaW1wbGVtZW50cyBCdWZmZXJlZEltYWdlT3AsIFJhc3Rlck9wIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBzY2FsZSBmYWN0b3JzLgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgc2NhbGVGYWN0b3JzW107Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgb2Zmc2V0cy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IG9mZnNldHNbXTsKLQotICAgIC8qKgotICAgICAqIFRoZSBoaW50cy4KLSAgICAgKi8KLSAgICBwcml2YXRlIFJlbmRlcmluZ0hpbnRzIGhpbnRzOwotCi0gICAgc3RhdGljIHsKLSAgICAgICAgLy8gVE9ETwotICAgICAgICAvLyBTeXN0ZW0ubG9hZExpYnJhcnkoImltYWdlb3BzIik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IFJlc2NhbGVPcCBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIHNjYWxlIGZhY3RvcnMgYW5kCi0gICAgICogb2Zmc2V0cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2NhbGVGYWN0b3JzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2Ygc2NhbGUgZmFjdG9yIHZhbHVlcy4KLSAgICAgKiBAcGFyYW0gb2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIG9mZnNldCB2YWx1ZXMuCi0gICAgICogQHBhcmFtIGhpbnRzCi0gICAgICogICAgICAgICAgICB0aGUgUmVuZGVyaW5nSGludHMgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVzY2FsZU9wKGZsb2F0W10gc2NhbGVGYWN0b3JzLCBmbG9hdFtdIG9mZnNldHMsIFJlbmRlcmluZ0hpbnRzIGhpbnRzKSB7Ci0gICAgICAgIGludCBudW1GYWN0b3JzID0gTWF0aC5taW4oc2NhbGVGYWN0b3JzLmxlbmd0aCwgb2Zmc2V0cy5sZW5ndGgpOwotCi0gICAgICAgIHRoaXMuc2NhbGVGYWN0b3JzID0gbmV3IGZsb2F0W251bUZhY3RvcnNdOwotICAgICAgICB0aGlzLm9mZnNldHMgPSBuZXcgZmxvYXRbbnVtRmFjdG9yc107Ci0KLSAgICAgICAgU3lzdGVtLmFycmF5Y29weShzY2FsZUZhY3RvcnMsIDAsIHRoaXMuc2NhbGVGYWN0b3JzLCAwLCBudW1GYWN0b3JzKTsKLSAgICAgICAgU3lzdGVtLmFycmF5Y29weShvZmZzZXRzLCAwLCB0aGlzLm9mZnNldHMsIDAsIG51bUZhY3RvcnMpOwotCi0gICAgICAgIHRoaXMuaGludHMgPSBoaW50czsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgUmVzY2FsZU9wIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgc2NhbGUgZmFjdG9yIGFuZAotICAgICAqIG9mZnNldC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2NhbGVGYWN0b3IKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FsZSBmYWN0b3IuCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldC4KLSAgICAgKiBAcGFyYW0gaGludHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJpbmdIaW50cyBvciBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBSZXNjYWxlT3AoZmxvYXQgc2NhbGVGYWN0b3IsIGZsb2F0IG9mZnNldCwgUmVuZGVyaW5nSGludHMgaGludHMpIHsKLSAgICAgICAgc2NhbGVGYWN0b3JzID0gbmV3IGZsb2F0WzFdOwotICAgICAgICBvZmZzZXRzID0gbmV3IGZsb2F0WzFdOwotCi0gICAgICAgIHNjYWxlRmFjdG9yc1swXSA9IHNjYWxlRmFjdG9yOwotICAgICAgICBvZmZzZXRzWzBdID0gb2Zmc2V0OwotCi0gICAgICAgIHRoaXMuaGludHMgPSBoaW50czsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBudW1iZXIgb2Ygc2NhbGluZyBmYWN0b3JzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiBzY2FsaW5nIGZhY3RvcnMuCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGludCBnZXROdW1GYWN0b3JzKCkgewotICAgICAgICByZXR1cm4gc2NhbGVGYWN0b3JzLmxlbmd0aDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUmVuZGVyaW5nSGludHMgZ2V0UmVuZGVyaW5nSGludHMoKSB7Ci0gICAgICAgIHJldHVybiBoaW50czsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzY2FsZSBmYWN0b3JzIG9mIHRoaXMgUmVzY2FsZU9wLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzY2FsZUZhY3RvcnMKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXNpcmVkIHNjYWxlIGZhY3RvcnMgYXJyYXkgd2lsbCBiZSBjb3BpZWQgdG8gdGhpcyBhcnJheS4KLSAgICAgKiBAcmV0dXJuIHRoZSBzY2FsZSBmYWN0b3JzIGFycmF5LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBmbG9hdFtdIGdldFNjYWxlRmFjdG9ycyhmbG9hdFtdIHNjYWxlRmFjdG9ycykgewotICAgICAgICBpZiAoc2NhbGVGYWN0b3JzID09IG51bGwpIHsKLSAgICAgICAgICAgIHNjYWxlRmFjdG9ycyA9IG5ldyBmbG9hdFt0aGlzLnNjYWxlRmFjdG9ycy5sZW5ndGhdOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IG1pbkxlbmd0aCA9IE1hdGgubWluKHNjYWxlRmFjdG9ycy5sZW5ndGgsIHRoaXMuc2NhbGVGYWN0b3JzLmxlbmd0aCk7Ci0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkodGhpcy5zY2FsZUZhY3RvcnMsIDAsIHNjYWxlRmFjdG9ycywgMCwgbWluTGVuZ3RoKTsKLSAgICAgICAgcmV0dXJuIHNjYWxlRmFjdG9yczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBvZmZzZXRzIGFycmF5IG9mIHRoaXMgUmVzY2FsZU9wLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvZmZzZXRzCi0gICAgICogICAgICAgICAgICB0aGUgZGVzaXJlZCBvZmZzZXRzIGFycmF5IHdpbGwgYmUgY29waWVkIHRvIHRoaXMgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgb2Zmc2V0cyBhcnJheSBvZiB0aGlzIFJlc2NhbGVPcC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgZmxvYXRbXSBnZXRPZmZzZXRzKGZsb2F0W10gb2Zmc2V0cykgewotICAgICAgICBpZiAob2Zmc2V0cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICBvZmZzZXRzID0gbmV3IGZsb2F0W3RoaXMub2Zmc2V0cy5sZW5ndGhdOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IG1pbkxlbmd0aCA9IE1hdGgubWluKG9mZnNldHMubGVuZ3RoLCB0aGlzLm9mZnNldHMubGVuZ3RoKTsKLSAgICAgICAgU3lzdGVtLmFycmF5Y29weSh0aGlzLm9mZnNldHMsIDAsIG9mZnNldHMsIDAsIG1pbkxlbmd0aCk7Ci0gICAgICAgIHJldHVybiBvZmZzZXRzOwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBQb2ludDJEIGdldFBvaW50MkQoUG9pbnQyRCBzcmNQdCwgUG9pbnQyRCBkc3RQdCkgewotICAgICAgICBpZiAoZHN0UHQgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0UHQgPSBuZXcgUG9pbnQyRC5GbG9hdCgpOwotICAgICAgICB9Ci0KLSAgICAgICAgZHN0UHQuc2V0TG9jYXRpb24oc3JjUHQpOwotICAgICAgICByZXR1cm4gZHN0UHQ7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIFJlY3RhbmdsZTJEIGdldEJvdW5kczJEKFJhc3RlciBzcmMpIHsKLSAgICAgICAgcmV0dXJuIHNyYy5nZXRCb3VuZHMoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoQnVmZmVyZWRJbWFnZSBzcmMpIHsKLSAgICAgICAgcmV0dXJuIGdldEJvdW5kczJEKHNyYy5nZXRSYXN0ZXIoKSk7Ci0gICAgfQotCi0gICAgcHVibGljIFdyaXRhYmxlUmFzdGVyIGNyZWF0ZUNvbXBhdGlibGVEZXN0UmFzdGVyKFJhc3RlciBzcmMpIHsKLSAgICAgICAgcmV0dXJuIHNyYy5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQnVmZmVyZWRJbWFnZSBjcmVhdGVDb21wYXRpYmxlRGVzdEltYWdlKEJ1ZmZlcmVkSW1hZ2Ugc3JjLCBDb2xvck1vZGVsIGRzdENNKSB7Ci0gICAgICAgIGlmIChkc3RDTSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBkc3RDTSA9IHNyYy5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZHN0Q00gaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwpIHsKLSAgICAgICAgICAgIGRzdENNID0gQ29sb3JNb2RlbC5nZXRSR0JkZWZhdWx0KCk7Ci0gICAgICAgIH0KLQotICAgICAgICBXcml0YWJsZVJhc3RlciByID0gZHN0Q00uaXNDb21wYXRpYmxlU2FtcGxlTW9kZWwoc3JjLmdldFNhbXBsZU1vZGVsKCkpID8gc3JjLmdldFJhc3RlcigpCi0gICAgICAgICAgICAgICAgLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3RlcihzcmMuZ2V0V2lkdGgoKSwgc3JjLmdldEhlaWdodCgpKSA6IGRzdENNCi0gICAgICAgICAgICAgICAgLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3RlcihzcmMuZ2V0V2lkdGgoKSwgc3JjLmdldEhlaWdodCgpKTsKLQotICAgICAgICByZXR1cm4gbmV3IEJ1ZmZlcmVkSW1hZ2UoZHN0Q00sIHIsIGRzdENNLmlzQWxwaGFQcmVtdWx0aXBsaWVkKCksIG51bGwpOwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBXcml0YWJsZVJhc3RlciBmaWx0ZXIoUmFzdGVyIHNyYywgV3JpdGFibGVSYXN0ZXIgZHN0KSB7Ci0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0ID0gY3JlYXRlQ29tcGF0aWJsZURlc3RSYXN0ZXIoc3JjKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGlmIChzcmMuZ2V0TnVtQmFuZHMoKSAhPSBkc3QuZ2V0TnVtQmFuZHMoKSkgewotICAgICAgICAgICAgICAgIC8vIGF3dC4yMUQ9TnVtYmVyIG9mIHNyYyBiYW5kcyAoezB9KSBkb2VzIG5vdCBtYXRjaCBudW1iZXIgb2YKLSAgICAgICAgICAgICAgICAvLyBkc3QgYmFuZHMgKHsxfSkKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxRCIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgICAgIHNyYy5nZXROdW1CYW5kcygpLCBkc3QuZ2V0TnVtQmFuZHMoKSkpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHRoaXMuc2NhbGVGYWN0b3JzLmxlbmd0aCAhPSAxICYmIHRoaXMuc2NhbGVGYWN0b3JzLmxlbmd0aCAhPSBzcmMuZ2V0TnVtQmFuZHMoKSkgewotICAgICAgICAgICAgLy8gYXd0LjIxRT1OdW1iZXIgb2Ygc2NhbGluZyBjb25zdGFudHMgaXMgbm90IGVxdWFsIHRvIHRoZSBudW1iZXIgb2YKLSAgICAgICAgICAgIC8vIGJhbmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgLy8gVE9ETwotICAgICAgICAvLyBpZiAoaXBwRmlsdGVyKHNyYywgZHN0LCBCdWZmZXJlZEltYWdlLlRZUEVfQ1VTVE9NLCBmYWxzZSkgIT0gMCkKLSAgICAgICAgaWYgKHNsb3dGaWx0ZXIoc3JjLCBkc3QsIGZhbHNlKSAhPSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjFGPVVuYWJsZSB0byB0cmFuc2Zvcm0gc291cmNlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSW1hZ2luZ09wRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxRiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGRzdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTbG93IGZpbHRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgc3JjLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3QuCi0gICAgICogQHBhcmFtIHNraXBBbHBoYQotICAgICAqICAgICAgICAgICAgdGhlIHNraXAgYWxwaGEuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgaW50IHNsb3dGaWx0ZXIoUmFzdGVyIHNyYywgV3JpdGFibGVSYXN0ZXIgZHN0LCBib29sZWFuIHNraXBBbHBoYSkgewotICAgICAgICBTYW1wbGVNb2RlbCBzbSA9IHNyYy5nZXRTYW1wbGVNb2RlbCgpOwotCi0gICAgICAgIGludCBudW1CYW5kcyA9IHNyYy5nZXROdW1CYW5kcygpOwotICAgICAgICBpbnQgc3JjSGVpZ2h0ID0gc3JjLmdldEhlaWdodCgpOwotICAgICAgICBpbnQgc3JjV2lkdGggPSBzcmMuZ2V0V2lkdGgoKTsKLQotICAgICAgICBpbnQgc3JjTWluWCA9IHNyYy5nZXRNaW5YKCk7Ci0gICAgICAgIGludCBzcmNNaW5ZID0gc3JjLmdldE1pblkoKTsKLSAgICAgICAgaW50IGRzdE1pblggPSBkc3QuZ2V0TWluWCgpOwotICAgICAgICBpbnQgZHN0TWluWSA9IGRzdC5nZXRNaW5ZKCk7Ci0KLSAgICAgICAgaW50W10gbWF4VmFsdWVzID0gbmV3IGludFtudW1CYW5kc107Ci0gICAgICAgIGludFtdIG1hc2tzID0gbmV3IGludFtudW1CYW5kc107Ci0gICAgICAgIGludFtdIHNhbXBsZVNpemVzID0gc20uZ2V0U2FtcGxlU2l6ZSgpOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgbWF4VmFsdWVzW2ldID0gKDEgPDwgc2FtcGxlU2l6ZXNbaV0pIC0gMTsKLSAgICAgICAgICAgIG1hc2tzW2ldID0gfihtYXhWYWx1ZXNbaV0pOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gUHJvY2Vzc2luZyBib3VuZHMKLSAgICAgICAgZmxvYXRbXSBwaXhlbHMgPSBudWxsOwotICAgICAgICBwaXhlbHMgPSBzcmMuZ2V0UGl4ZWxzKHNyY01pblgsIHNyY01pblksIHNyY1dpZHRoLCBzcmNIZWlnaHQsIHBpeGVscyk7Ci0KLSAgICAgICAgLy8gQ3ljbGUgb3ZlciBwaXhlbHMgdG8gYmUgY2FsY3VsYXRlZAotICAgICAgICBpZiAoc2tpcEFscGhhKSB7IC8vIEFsd2F5cyBzdXBwb3NlIHRoYXQgYWxwaGEgY2hhbm5lbCBpcyB0aGUgbGFzdCBiYW5kCi0gICAgICAgICAgICBpZiAoc2NhbGVGYWN0b3JzLmxlbmd0aCA+IDEpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHBpeGVscy5sZW5ndGg7KSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGJhbmRJZHggPSAwOyBiYW5kSWR4IDwgbnVtQmFuZHMgLSAxOyBiYW5kSWR4KyssIGkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgcGl4ZWxzW2ldID0gcGl4ZWxzW2ldICogc2NhbGVGYWN0b3JzW2JhbmRJZHhdICsgb2Zmc2V0c1tiYW5kSWR4XTsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIENoZWNrIGZvciBvdmVyZmxvdyBub3cKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmICgoKGludClwaXhlbHNbaV0gJiBtYXNrc1tiYW5kSWR4XSkgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChwaXhlbHNbaV0gPCAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBpeGVsc1tpXSA9IDA7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcGl4ZWxzW2ldID0gbWF4VmFsdWVzW2JhbmRJZHhdOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGkrKzsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcGl4ZWxzLmxlbmd0aDspIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgYmFuZElkeCA9IDA7IGJhbmRJZHggPCBudW1CYW5kcyAtIDE7IGJhbmRJZHgrKywgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwaXhlbHNbaV0gPSBwaXhlbHNbaV0gKiBzY2FsZUZhY3RvcnNbMF0gKyBvZmZzZXRzWzBdOwotICAgICAgICAgICAgICAgICAgICAgICAgLy8gQ2hlY2sgZm9yIG92ZXJmbG93IG5vdwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCgoaW50KXBpeGVsc1tpXSAmIG1hc2tzW2JhbmRJZHhdKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHBpeGVsc1tpXSA8IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcGl4ZWxzW2ldID0gMDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwaXhlbHNbaV0gPSBtYXhWYWx1ZXNbYmFuZElkeF07Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgaSsrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGlmIChzY2FsZUZhY3RvcnMubGVuZ3RoID4gMSkgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcGl4ZWxzLmxlbmd0aDspIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgYmFuZElkeCA9IDA7IGJhbmRJZHggPCBudW1CYW5kczsgYmFuZElkeCsrLCBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHBpeGVsc1tpXSA9IHBpeGVsc1tpXSAqIHNjYWxlRmFjdG9yc1tiYW5kSWR4XSArIG9mZnNldHNbYmFuZElkeF07Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyBDaGVjayBmb3Igb3ZlcmZsb3cgbm93Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoKChpbnQpcGl4ZWxzW2ldICYgbWFza3NbYmFuZElkeF0pICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAocGl4ZWxzW2ldIDwgMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwaXhlbHNbaV0gPSAwOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBpeGVsc1tpXSA9IG1heFZhbHVlc1tiYW5kSWR4XTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcGl4ZWxzLmxlbmd0aDspIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgYmFuZElkeCA9IDA7IGJhbmRJZHggPCBudW1CYW5kczsgYmFuZElkeCsrLCBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHBpeGVsc1tpXSA9IHBpeGVsc1tpXSAqIHNjYWxlRmFjdG9yc1swXSArIG9mZnNldHNbMF07Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyBDaGVjayBmb3Igb3ZlcmZsb3cgbm93Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoKChpbnQpcGl4ZWxzW2ldICYgbWFza3NbYmFuZElkeF0pICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAocGl4ZWxzW2ldIDwgMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwaXhlbHNbaV0gPSAwOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBpeGVsc1tpXSA9IG1heFZhbHVlc1tiYW5kSWR4XTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBkc3Quc2V0UGl4ZWxzKGRzdE1pblgsIGRzdE1pblksIHNyY1dpZHRoLCBzcmNIZWlnaHQsIHBpeGVscyk7Ci0KLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIEJ1ZmZlcmVkSW1hZ2UgZmlsdGVyKEJ1ZmZlcmVkSW1hZ2Ugc3JjLCBCdWZmZXJlZEltYWdlIGRzdCkgewotICAgICAgICBDb2xvck1vZGVsIHNyY0NNID0gc3JjLmdldENvbG9yTW9kZWwoKTsKLQotICAgICAgICBpZiAoc3JjQ00gaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMjA9U291cmNlIHNob3VsZCBub3QgaGF2ZSBJbmRleENvbG9yTW9kZWwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjIwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICAvLyBDaGVjayBpZiB0aGUgbnVtYmVyIG9mIHNjYWxpbmcgZmFjdG9ycyBtYXRjaGVzIHRoZSBudW1iZXIgb2YgYmFuZHMKLSAgICAgICAgaW50IG5Db21wb25lbnRzID0gc3JjQ00uZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICBib29sZWFuIHNraXBBbHBoYTsKLSAgICAgICAgaWYgKHNyY0NNLmhhc0FscGhhKCkpIHsKLSAgICAgICAgICAgIGlmIChzY2FsZUZhY3RvcnMubGVuZ3RoID09IDEgfHwgc2NhbGVGYWN0b3JzLmxlbmd0aCA9PSBuQ29tcG9uZW50cyAtIDEpIHsKLSAgICAgICAgICAgICAgICBza2lwQWxwaGEgPSB0cnVlOwotICAgICAgICAgICAgfSBlbHNlIGlmIChzY2FsZUZhY3RvcnMubGVuZ3RoID09IG5Db21wb25lbnRzKSB7Ci0gICAgICAgICAgICAgICAgc2tpcEFscGhhID0gZmFsc2U7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIC8vIGF3dC4yMUU9TnVtYmVyIG9mIHNjYWxpbmcgY29uc3RhbnRzIGlzIG5vdCBlcXVhbCB0byB0aGUKLSAgICAgICAgICAgICAgICAvLyBudW1iZXIgb2YgYmFuZHMKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIxRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgaWYgKHNjYWxlRmFjdG9ycy5sZW5ndGggPT0gMSB8fCBzY2FsZUZhY3RvcnMubGVuZ3RoID09IG5Db21wb25lbnRzKSB7Ci0gICAgICAgICAgICBza2lwQWxwaGEgPSBmYWxzZTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMUU9TnVtYmVyIG9mIHNjYWxpbmcgY29uc3RhbnRzIGlzIG5vdCBlcXVhbCB0byB0aGUgbnVtYmVyIG9mCi0gICAgICAgICAgICAvLyBiYW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMUUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIEJ1ZmZlcmVkSW1hZ2UgZmluYWxEc3QgPSBudWxsOwotICAgICAgICBpZiAoZHN0ID09IG51bGwpIHsKLSAgICAgICAgICAgIGZpbmFsRHN0ID0gZHN0OwotICAgICAgICAgICAgZHN0ID0gY3JlYXRlQ29tcGF0aWJsZURlc3RJbWFnZShzcmMsIHNyY0NNKTsKLSAgICAgICAgfSBlbHNlIGlmICghc3JjQ00uZXF1YWxzKGRzdC5nZXRDb2xvck1vZGVsKCkpKSB7Ci0gICAgICAgICAgICAvLyBUcmVhdCBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX1JHQiBhbmQgQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9BUkdCCi0gICAgICAgICAgICAvLyBhcyBzYW1lCi0gICAgICAgICAgICBpZiAoISgoc3JjLmdldFR5cGUoKSA9PSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX1JHQiB8fCBzcmMuZ2V0VHlwZSgpID09IEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQikgJiYgKGRzdAotICAgICAgICAgICAgICAgICAgICAuZ2V0VHlwZSgpID09IEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfUkdCIHx8IGRzdC5nZXRUeXBlKCkgPT0gQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9BUkdCKSkpIHsKLSAgICAgICAgICAgICAgICBmaW5hbERzdCA9IGRzdDsKLSAgICAgICAgICAgICAgICBkc3QgPSBjcmVhdGVDb21wYXRpYmxlRGVzdEltYWdlKHNyYywgc3JjQ00pOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLy8gVE9ETwotICAgICAgICAvLyBpZiAoaXBwRmlsdGVyKHNyYy5nZXRSYXN0ZXIoKSwgZHN0LmdldFJhc3RlcigpLCBzcmMuZ2V0VHlwZSgpLAotICAgICAgICAvLyBza2lwQWxwaGEpICE9IDApCi0gICAgICAgIGlmIChzbG93RmlsdGVyKHNyYy5nZXRSYXN0ZXIoKSwgZHN0LmdldFJhc3RlcigpLCBza2lwQWxwaGEpICE9IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMUY9VW5hYmxlIHRvIHRyYW5zZm9ybSBzb3VyY2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbWFnaW5nT3BFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjFGIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoZmluYWxEc3QgIT0gbnVsbCkgewotICAgICAgICAgICAgR3JhcGhpY3MyRCBnID0gZmluYWxEc3QuY3JlYXRlR3JhcGhpY3MoKTsKLSAgICAgICAgICAgIGcuc2V0Q29tcG9zaXRlKEFscGhhQ29tcG9zaXRlLlNyYyk7Ci0gICAgICAgICAgICBnLmRyYXdJbWFnZShkc3QsIDAsIDAsIG51bGwpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZmluYWxEc3QgPSBkc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZmluYWxEc3Q7Ci0gICAgfQotCi0gICAgLy8gRG9uJ3QgZm9yZ2V0IHRvIHBhc3MgYWxsb2NhdGVkIGFycmF5cyBmb3IgbGV2ZWxzIGFuZCB2YWx1ZXMsIHNpemUgc2hvdWxkCi0gICAgLy8gYmUgbnVtQmFuZHMqNAotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGxldmVscy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc20KLSAgICAgKiAgICAgICAgICAgIHRoZSBzbS4KLSAgICAgKiBAcGFyYW0gbnVtQmFuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW0gYmFuZHMuCi0gICAgICogQHBhcmFtIHNraXBBbHBoYQotICAgICAqICAgICAgICAgICAgdGhlIHNraXAgYWxwaGEuCi0gICAgICogQHBhcmFtIGxldmVscwotICAgICAqICAgICAgICAgICAgdGhlIGxldmVscy4KLSAgICAgKiBAcGFyYW0gdmFsdWVzCi0gICAgICogICAgICAgICAgICB0aGUgdmFsdWVzLgotICAgICAqIEBwYXJhbSBjaGFubmVsc09yZGVyCi0gICAgICogICAgICAgICAgICB0aGUgY2hhbm5lbHMgb3JkZXIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCB2b2lkIGNyZWF0ZUxldmVscyhTYW1wbGVNb2RlbCBzbSwgaW50IG51bUJhbmRzLCBib29sZWFuIHNraXBBbHBoYSwgaW50IGxldmVsc1tdLAotICAgICAgICAgICAgaW50IHZhbHVlc1tdLCBpbnQgY2hhbm5lbHNPcmRlcltdKSB7Ci0gICAgICAgIC8vIFN1cHBvc2Ugc2FtZSBzYW1wbGUgc2l6ZSBmb3IgYWxsIGNoYW5uZWxzLCBvdGhlcndpc2UgdXNlIHNsb3cgZmlsdGVyCi0gICAgICAgIGludCBtYXhWYWx1ZSA9ICgxIDw8IHNtLmdldFNhbXBsZVNpemUoMCkpIC0gMTsKLQotICAgICAgICAvLyBGb3Igc2ltcGxpY2l0eSBpbnRyb2R1Y2UgdGhlc2UgYXJyYXlzCi0gICAgICAgIGZsb2F0IGV4dFNjYWxlRmFjdG9yc1tdID0gbmV3IGZsb2F0W251bUJhbmRzXTsKLSAgICAgICAgZmxvYXQgZXh0T2Zmc2V0c1tdID0gbmV3IGZsb2F0W251bUJhbmRzXTsKLQotICAgICAgICBpZiAoc2NhbGVGYWN0b3JzLmxlbmd0aCAhPSAxKSB7Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHNjYWxlRmFjdG9ycywgMCwgZXh0U2NhbGVGYWN0b3JzLCAwLCBzY2FsZUZhY3RvcnMubGVuZ3RoKTsKLSAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkob2Zmc2V0cywgMCwgZXh0T2Zmc2V0cywgMCwgc2NhbGVGYWN0b3JzLmxlbmd0aCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgICAgICBleHRTY2FsZUZhY3RvcnNbaV0gPSBzY2FsZUZhY3RvcnNbMF07Ci0gICAgICAgICAgICAgICAgZXh0T2Zmc2V0c1tpXSA9IG9mZnNldHNbMF07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoc2tpcEFscGhhKSB7Ci0gICAgICAgICAgICBleHRTY2FsZUZhY3RvcnNbbnVtQmFuZHMgLSAxXSA9IDE7Ci0gICAgICAgICAgICBleHRPZmZzZXRzW251bUJhbmRzIC0gMV0gPSAwOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gQ3JlYXRlIGEgbGV2ZWxzCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgaWYgKGV4dFNjYWxlRmFjdG9yc1tpXSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgbGV2ZWxzW2kgKiA0XSA9IDA7Ci0gICAgICAgICAgICAgICAgbGV2ZWxzW2kgKiA0ICsgMV0gPSAwOwotICAgICAgICAgICAgICAgIGxldmVsc1tpICogNCArIDJdID0gbWF4VmFsdWUgKyAxOwotICAgICAgICAgICAgICAgIGxldmVsc1tpICogNCArIDNdID0gbWF4VmFsdWUgKyAxOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBmbG9hdCBtaW5MZXZlbCA9IC1leHRPZmZzZXRzW2ldIC8gZXh0U2NhbGVGYWN0b3JzW2ldOwotICAgICAgICAgICAgZmxvYXQgbWF4TGV2ZWwgPSAobWF4VmFsdWUgLSBleHRPZmZzZXRzW2ldKSAvIGV4dFNjYWxlRmFjdG9yc1tpXTsKLQotICAgICAgICAgICAgaWYgKG1pbkxldmVsIDwgMCkgewotICAgICAgICAgICAgICAgIG1pbkxldmVsID0gMDsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAobWluTGV2ZWwgPiBtYXhWYWx1ZSkgewotICAgICAgICAgICAgICAgIG1pbkxldmVsID0gbWF4VmFsdWU7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChtYXhMZXZlbCA8IDApIHsKLSAgICAgICAgICAgICAgICBtYXhMZXZlbCA9IDA7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKG1heExldmVsID4gbWF4VmFsdWUpIHsKLSAgICAgICAgICAgICAgICBtYXhMZXZlbCA9IG1heFZhbHVlOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBsZXZlbHNbaSAqIDRdID0gMDsKLSAgICAgICAgICAgIGlmIChtaW5MZXZlbCA+IG1heExldmVsKSB7Ci0gICAgICAgICAgICAgICAgbGV2ZWxzW2kgKiA0ICsgMV0gPSAoaW50KW1heExldmVsOwotICAgICAgICAgICAgICAgIGxldmVsc1tpICogNCArIDJdID0gKGludCltaW5MZXZlbDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgbGV2ZWxzW2kgKiA0ICsgMV0gPSAoaW50KW1pbkxldmVsOwotICAgICAgICAgICAgICAgIGxldmVsc1tpICogNCArIDJdID0gKGludCltYXhMZXZlbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGxldmVsc1tpICogNCArIDNdID0gbWF4VmFsdWUgKyAxOwotCi0gICAgICAgICAgICAvLyBGaWxsIHZhbHVlcwotICAgICAgICAgICAgZm9yIChpbnQgayA9IDA7IGsgPCA0OyBrKyspIHsKLSAgICAgICAgICAgICAgICBpbnQgaWR4ID0gaSAqIDQgKyBrOwotICAgICAgICAgICAgICAgIHZhbHVlc1tpZHhdID0gKGludCkoZXh0U2NhbGVGYWN0b3JzW2ldICogbGV2ZWxzW2lkeF0gKyBleHRPZmZzZXRzW2ldKTsKLSAgICAgICAgICAgICAgICBpZiAodmFsdWVzW2lkeF0gPCAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHZhbHVlc1tpZHhdID0gMDsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKHZhbHVlc1tpZHhdID4gbWF4VmFsdWUpIHsKLSAgICAgICAgICAgICAgICAgICAgdmFsdWVzW2lkeF0gPSBtYXhWYWx1ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBSZW9yZGVyIGRhdGEgaWYgY2hhbm5lbHMgYXJlIHN0b3JlZCBpbiBkaWZmZXJlbnQgb3JkZXIKLSAgICAgICAgaWYgKGNoYW5uZWxzT3JkZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgaW50IGxlbiA9IG51bUJhbmRzICogNDsKLSAgICAgICAgICAgIGludCBzYXZlZExldmVsc1tdID0gbmV3IGludFtsZW5dOwotICAgICAgICAgICAgaW50IHNhdmVkVmFsdWVzW10gPSBuZXcgaW50W2xlbl07Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGxldmVscywgMCwgc2F2ZWRMZXZlbHMsIDAsIGxlbik7Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHZhbHVlcywgMCwgc2F2ZWRWYWx1ZXMsIDAsIGxlbik7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGNoYW5uZWxzT3JkZXIubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHNhdmVkTGV2ZWxzLCBpICogNCwgbGV2ZWxzLCBjaGFubmVsc09yZGVyW2ldICogNCwgNCk7Ci0gICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShzYXZlZFZhbHVlcywgaSAqIDQsIHZhbHVlcywgY2hhbm5lbHNPcmRlcltpXSAqIDQsIDQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gVE9ETyByZW1vdmUgd2hlbiB0aGlzIG1ldGhvZCBpcyB1c2VkCi0gICAgLyoqCi0gICAgICogSXBwIGZpbHRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgc3JjLgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkc3QuCi0gICAgICogQHBhcmFtIGltYWdlVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHR5cGUuCi0gICAgICogQHBhcmFtIHNraXBBbHBoYQotICAgICAqICAgICAgICAgICAgdGhlIHNraXAgYWxwaGEuCi0gICAgICogQHJldHVybiB0aGUgaW50LgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bnVzZWQiKQotICAgIHByaXZhdGUgZmluYWwgaW50IGlwcEZpbHRlcihSYXN0ZXIgc3JjLCBXcml0YWJsZVJhc3RlciBkc3QsIGludCBpbWFnZVR5cGUsIGJvb2xlYW4gc2tpcEFscGhhKSB7Ci0gICAgICAgIGludCByZXM7Ci0KLSAgICAgICAgaW50IHNyY1N0cmlkZSwgZHN0U3RyaWRlOwotICAgICAgICBpbnQgY2hhbm5lbHM7Ci0gICAgICAgIGludCBvZmZzZXRzW10gPSBudWxsOwotICAgICAgICBpbnQgY2hhbm5lbHNPcmRlcltdID0gbnVsbDsKLQotICAgICAgICBzd2l0Y2ggKGltYWdlVHlwZSkgewotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQl9QUkU6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfUkdCOiB7Ci0gICAgICAgICAgICAgICAgY2hhbm5lbHMgPSA0OwotICAgICAgICAgICAgICAgIHNyY1N0cmlkZSA9IHNyYy5nZXRXaWR0aCgpICogNDsKLSAgICAgICAgICAgICAgICBkc3RTdHJpZGUgPSBkc3QuZ2V0V2lkdGgoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgY2hhbm5lbHNPcmRlciA9IG5ldyBpbnRbXSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAyLCAxLCAwLCAzCi0gICAgICAgICAgICAgICAgfTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfNEJZVEVfQUJHUjoKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFXzRCWVRFX0FCR1JfUFJFOgotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0JHUjogewotICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gNDsKLSAgICAgICAgICAgICAgICBzcmNTdHJpZGUgPSBzcmMuZ2V0V2lkdGgoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCkgKiA0OwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0dSQVk6IHsKLSAgICAgICAgICAgICAgICBjaGFubmVscyA9IDE7Ci0gICAgICAgICAgICAgICAgc3JjU3RyaWRlID0gc3JjLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFXzNCWVRFX0JHUjogewotICAgICAgICAgICAgICAgIGNoYW5uZWxzID0gMzsKLSAgICAgICAgICAgICAgICBzcmNTdHJpZGUgPSBzcmMuZ2V0V2lkdGgoKSAqIDM7Ci0gICAgICAgICAgICAgICAgZHN0U3RyaWRlID0gZHN0LmdldFdpZHRoKCkgKiAzOwotICAgICAgICAgICAgICAgIGNoYW5uZWxzT3JkZXIgPSBuZXcgaW50W10gewotICAgICAgICAgICAgICAgICAgICAgICAgMiwgMSwgMAotICAgICAgICAgICAgICAgIH07Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX1VTSE9SVF9HUkFZOgotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfVVNIT1JUXzU2NV9SR0I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9VU0hPUlRfNTU1X1JHQjoKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX0JZVEVfQklOQVJZOiB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QsIHNraXBBbHBoYSk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGRlZmF1bHQ6IHsKLSAgICAgICAgICAgICAgICBTYW1wbGVNb2RlbCBzcmNTTSA9IHNyYy5nZXRTYW1wbGVNb2RlbCgpOwotICAgICAgICAgICAgICAgIFNhbXBsZU1vZGVsIGRzdFNNID0gZHN0LmdldFNhbXBsZU1vZGVsKCk7Ci0KLSAgICAgICAgICAgICAgICBpZiAoc3JjU00gaW5zdGFuY2VvZiBQaXhlbEludGVybGVhdmVkU2FtcGxlTW9kZWwKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIGRzdFNNIGluc3RhbmNlb2YgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIENoZWNrIFBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbAotICAgICAgICAgICAgICAgICAgICBpZiAoc3JjU00uZ2V0RGF0YVR5cGUoKSAhPSBEYXRhQnVmZmVyLlRZUEVfQllURQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8IGRzdFNNLmdldERhdGFUeXBlKCkgIT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBzbG93RmlsdGVyKHNyYywgZHN0LCBza2lwQWxwaGEpOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHMgPSBzcmNTTS5nZXROdW1CYW5kcygpOyAvLyBIYXZlIElQUCBmdW5jdGlvbnMgZm9yIDEsCi0gICAgICAgICAgICAgICAgICAgIC8vIDMgYW5kIDQgY2hhbm5lbHMKLSAgICAgICAgICAgICAgICAgICAgaWYgKCEoY2hhbm5lbHMgPT0gMSB8fCBjaGFubmVscyA9PSAzIHx8IGNoYW5uZWxzID09IDQpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCwgc2tpcEFscGhhKTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIHNyY1N0cmlkZSA9ICgoQ29tcG9uZW50U2FtcGxlTW9kZWwpc3JjU00pLmdldFNjYW5saW5lU3RyaWRlKCk7Ci0gICAgICAgICAgICAgICAgICAgIGRzdFN0cmlkZSA9ICgoQ29tcG9uZW50U2FtcGxlTW9kZWwpZHN0U00pLmdldFNjYW5saW5lU3RyaWRlKCk7Ci0KLSAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHNPcmRlciA9ICgoQ29tcG9uZW50U2FtcGxlTW9kZWwpc3JjU00pLmdldEJhbmRPZmZzZXRzKCk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChzcmNTTSBpbnN0YW5jZW9mIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwKLSAgICAgICAgICAgICAgICAgICAgICAgICYmIGRzdFNNIGluc3RhbmNlb2YgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkgewotICAgICAgICAgICAgICAgICAgICAvLyBDaGVjayBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsCi0gICAgICAgICAgICAgICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc3Bwc20xID0gKFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpc3JjU007Ci0gICAgICAgICAgICAgICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc3Bwc20yID0gKFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpZHN0U007Ci0KLSAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHMgPSBzcHBzbTEuZ2V0TnVtQmFuZHMoKTsKLQotICAgICAgICAgICAgICAgICAgICAvLyBUWVBFX0lOVF9SR0IsIFRZUEVfSU5UX0FSR0IuLi4KLSAgICAgICAgICAgICAgICAgICAgaWYgKHNwcHNtMS5nZXREYXRhVHlwZSgpICE9IERhdGFCdWZmZXIuVFlQRV9JTlQKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCBzcHBzbTIuZ2V0RGF0YVR5cGUoKSAhPSBEYXRhQnVmZmVyLlRZUEVfSU5UCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgIShjaGFubmVscyA9PSAzIHx8IGNoYW5uZWxzID09IDQpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCwgc2tpcEFscGhhKTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIC8vIENoZWNrIGNvbXBhdGliaWxpdHkgb2Ygc2FtcGxlIG1vZGVscwotICAgICAgICAgICAgICAgICAgICBpZiAoIUFycmF5cy5lcXVhbHMoc3Bwc20xLmdldEJpdE9mZnNldHMoKSwgc3Bwc20yLmdldEJpdE9mZnNldHMoKSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCAhQXJyYXlzLmVxdWFscyhzcHBzbTEuZ2V0Qml0TWFza3MoKSwgc3Bwc20yLmdldEJpdE1hc2tzKCkpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCwgc2tpcEFscGhhKTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHNwcHNtMS5nZXRTYW1wbGVTaXplKGkpICE9IDgpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gc2xvd0ZpbHRlcihzcmMsIGRzdCwgc2tpcEFscGhhKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGNoYW5uZWxzT3JkZXIgPSBuZXcgaW50W2NoYW5uZWxzXTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGJpdE9mZnNldHNbXSA9IHNwcHNtMS5nZXRCaXRPZmZzZXRzKCk7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHNPcmRlcltpXSA9IGJpdE9mZnNldHNbaV0gLyA4OwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgaWYgKGNoYW5uZWxzID09IDMpIHsgLy8gRG9uJ3Qgc2tpcCBjaGFubmVsIG5vdywgY291bGQgYmUKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIG9wdGltaXplZAotICAgICAgICAgICAgICAgICAgICAgICAgY2hhbm5lbHMgPSA0OwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgc3JjU3RyaWRlID0gc3Bwc20xLmdldFNjYW5saW5lU3RyaWRlKCkgKiA0OwotICAgICAgICAgICAgICAgICAgICBkc3RTdHJpZGUgPSBzcHBzbTIuZ2V0U2NhbmxpbmVTdHJpZGUoKSAqIDQ7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNsb3dGaWx0ZXIoc3JjLCBkc3QsIHNraXBBbHBoYSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgLy8gRmlsbCBvZmZzZXRzIGlmIHRoZXJlJ3MgYSBjaGlsZCByYXN0ZXIKLSAgICAgICAgICAgICAgICBpZiAoc3JjLmdldFBhcmVudCgpICE9IG51bGwgfHwgZHN0LmdldFBhcmVudCgpICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSAhPSAwIHx8IHNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSAhPSAwCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgZHN0LmdldFNhbXBsZU1vZGVsVHJhbnNsYXRlWCgpICE9IDAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCBkc3QuZ2V0U2FtcGxlTW9kZWxUcmFuc2xhdGVZKCkgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0cyA9IG5ldyBpbnRbNF07Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXRzWzBdID0gLXNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSArIHNyYy5nZXRNaW5YKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXRzWzFdID0gLXNyYy5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSArIHNyYy5nZXRNaW5ZKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXRzWzJdID0gLWRzdC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVgoKSArIGRzdC5nZXRNaW5YKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXRzWzNdID0gLWRzdC5nZXRTYW1wbGVNb2RlbFRyYW5zbGF0ZVkoKSArIGRzdC5nZXRNaW5ZKCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgbGV2ZWxzW10gPSBuZXcgaW50WzQgKiBjaGFubmVsc107Ci0gICAgICAgIGludCB2YWx1ZXNbXSA9IG5ldyBpbnRbNCAqIGNoYW5uZWxzXTsKLQotICAgICAgICBjcmVhdGVMZXZlbHMoc3JjLmdldFNhbXBsZU1vZGVsKCksIGNoYW5uZWxzLCBza2lwQWxwaGEsIGxldmVscywgdmFsdWVzLCBjaGFubmVsc09yZGVyKTsKLQotICAgICAgICBPYmplY3Qgc3JjRGF0YSwgZHN0RGF0YTsKLSAgICAgICAgQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yIGRiQWNjZXNzID0gQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yLmdldEluc3RhbmNlKCk7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBzcmNEYXRhID0gZGJBY2Nlc3MuZ2V0RGF0YShzcmMuZ2V0RGF0YUJ1ZmZlcigpKTsKLSAgICAgICAgICAgIGRzdERhdGEgPSBkYkFjY2Vzcy5nZXREYXRhKGRzdC5nZXREYXRhQnVmZmVyKCkpOwotICAgICAgICB9IGNhdGNoIChJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgcmV0dXJuIC0xOyAvLyBVbmtub3duIGRhdGEgYnVmZmVyIHR5cGUKLSAgICAgICAgfQotCi0gICAgICAgIHJlcyA9IExvb2t1cE9wLmlwcExVVChzcmNEYXRhLCBzcmMuZ2V0V2lkdGgoKSwgc3JjLmdldEhlaWdodCgpLCBzcmNTdHJpZGUsIGRzdERhdGEsIGRzdAotICAgICAgICAgICAgICAgIC5nZXRXaWR0aCgpLCBkc3QuZ2V0SGVpZ2h0KCksIGRzdFN0cmlkZSwgbGV2ZWxzLCB2YWx1ZXMsIGNoYW5uZWxzLCBvZmZzZXRzLCB0cnVlKTsKLQotICAgICAgICByZXR1cm4gcmVzOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9TYW1wbGVNb2RlbC5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL1NhbXBsZU1vZGVsLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM5NjdmYTYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL1NhbXBsZU1vZGVsLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMTY2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIFNhbXBsZU1vZGVsIGNsYXNzIGlzIGFic3RyYWN0IGNsYXNzIGZvciByZXRyaWV2aW5nIHBpeGVsJ3Mgc2FtcGxlcyBpbiB0aGUKLSAqIGRhdGEgb2YgYW4gaW1hZ2UuIEVhY2ggcGl4ZWwgY29udGFpbnMgc2V2ZXJhbCBzYW1wbGVzLiBBIHNhbXBsZSBpcyB0aGUgc2V0IG9mCi0gKiB2YWx1ZXMgb2YgdGhlIGJhbmRzIGZvciBzaW5nbGUgcGl4ZWwuIEZvciBleGFtcGxlLCBlYWNoIHBpeGVsIGluIHRoZSBSR0IKLSAqIG1vZGVsIGNvbnRhaW5zIHRocmVlIHNhbXBsZXMgYW5kIHRoZXJlIGFyZSB0aHJlZSBjb3JyZXNwb25kaW5nIGJhbmRzIGluIHRoZQotICogaW1hZ2UgZGF0YSBvZiBzdWNoIHBpeGVscyByZXByZXNlbnRpbmcgcmVkLCBncmVlbiBhbmQgYmx1ZSBjb21wb25lbnRzLgotICogPHA+Ci0gKiBUaGUgaW1hZ2UgZGF0YSBpcyByZXByZXNlbnRlZCBhcyBhIFJhc3RlciB3aXRoIGEgRGF0YUJ1ZmZlciBhbmQgYQotICogU2FtcGxlTW9kZWwuIFRoZSBTYW1wbGVNb2RlbCBhbGxvd3MgYWNjZXNzIHRvIHRoZSBzYW1wbGVzIGluIHRoZSBEYXRhQnVmZmVyLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIFNhbXBsZU1vZGVsIHsKLQotICAgIC8qKgotICAgICAqIFRoZSB3aWR0aCBvZiB0aGUgaW1hZ2UgZGF0YSB3aGljaCB0aGlzIFNhbXBsZU1vZGVsIGRlc2NyaWJlcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IHdpZHRoOwotCi0gICAgLyoqCi0gICAgICogVGhlIGhlaWdodCBvZiB0aGUgaW1hZ2UgZGF0YSB3aGljaCB0aGlzIFNhbXBsZU1vZGVsIGRlc2NyaWJlcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IGhlaWdodDsKLQotICAgIC8qKgotICAgICAqIFRoZSBudW1iZXIgb2YgYmFuZHMgb2YgaW1hZ2UgZGF0YSB3aGljaCB0aGlzIFNhbXBsZU1vZGVsIGRlc2NyaWJlcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IG51bUJhbmRzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGRhdGEgdHlwZSBvZiB0aGUgaW1hZ2UgZGF0YSB3aGljaCB0aGlzIFNhbXBsZU1vZGVsIGRlc2NyaWJlcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IGRhdGFUeXBlOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IFNhbXBsZU1vZGVsIHdpdGggdGhlIHNwZWNpZmllZCBkYXRhIHR5cGUsIHdpZHRoLAotICAgICAqIGhlaWdodCBhbmQgbnVtYmVyIG9mIGJhbmRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgdHlwZSBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEBwYXJhbSBudW1CYW5kcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBiYW5kcyBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgU2FtcGxlTW9kZWwoaW50IGRhdGFUeXBlLCBpbnQgdywgaW50IGgsIGludCBudW1CYW5kcykgewotICAgICAgICBpZiAodyA8PSAwIHx8IGggPD0gMCkgewotICAgICAgICAgICAgLy8gYXd0LjIyRT13IG9yIGggaXMgbGVzcyB0aGFuIG9yIGVxdWFsIHRvIHplcm8KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjJFIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgc3F1cmUgPSAoKGRvdWJsZSl3KSAqICgoZG91YmxlKWgpOwotICAgICAgICBpZiAoc3F1cmUgPj0gSW50ZWdlci5NQVhfVkFMVUUpIHsKLSAgICAgICAgICAgIC8vIGF3dC4yMkY9VGhlIHByb2R1Y3Qgb2YgdyBhbmQgaCBpcyBncmVhdGVyIHRoYW4gSW50ZWdlci5NQVhfVkFMVUUKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjJGIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoZGF0YVR5cGUgPCBEYXRhQnVmZmVyLlRZUEVfQllURSB8fCBkYXRhVHlwZSA+IERhdGFCdWZmZXIuVFlQRV9ET1VCTEUKLSAgICAgICAgICAgICAgICAmJiBkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfVU5ERUZJTkVEKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjMwPWRhdGFUeXBlIGlzIG5vdCBvbmUgb2YgdGhlIHN1cHBvcnRlZCBkYXRhIHR5cGVzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG51bUJhbmRzIDwgMSkgewotICAgICAgICAgICAgLy8gYXd0LjIzMT1OdW1iZXIgb2YgYmFuZHMgbXVzdCBiZSBtb3JlIHRoZW4gMAotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMzEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMuZGF0YVR5cGUgPSBkYXRhVHlwZTsKLSAgICAgICAgdGhpcy53aWR0aCA9IHc7Ci0gICAgICAgIHRoaXMuaGVpZ2h0ID0gaDsKLSAgICAgICAgdGhpcy5udW1CYW5kcyA9IG51bUJhbmRzOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGF0YSBhcnJheSBmb3IgdGhlIHNwZWNpZmllZCBwaXhlbCBvZiB0aGUgc3BlY2lmaWVkIERhdGFCdWZmZXIKLSAgICAgKiB3aXRoIG9uZSBvZiB0aGUgZm9sbG93aW5nIHR5cGVzOiBEYXRhQnVmZmVyLlRZUEVfQllURSwKLSAgICAgKiBEYXRhQnVmZmVyLlRZUEVfVVNIT1JULCBEYXRhQnVmZmVyLlRZUEVfSU5ULCBEYXRhQnVmZmVyLlRZUEVfU0hPUlQsCi0gICAgICogRGF0YUJ1ZmZlci5UWVBFX0ZMT0FULCBvciBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QgaXMgYSBkYXRhIHdoZXJlIHRoZSByZXN1bHQgd2lsbCBiZSBzdG9yZWQuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEByZXR1cm4gdGhlIGRhdGEgYXJyYXkgZm9yIHRoZSBzcGVjaWZpZWQgcGl4ZWwgb2YgdGhlIHNwZWNpZmllZAotICAgICAqICAgICAgICAgRGF0YUJ1ZmZlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgT2JqZWN0IGdldERhdGFFbGVtZW50cyhpbnQgeCwgaW50IHksIE9iamVjdCBvYmosIERhdGFCdWZmZXIgZGF0YSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhcnJheSBvZiBwaXhlbCBkYXRhIGZvciB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzCi0gICAgICogb2YgdGhlIHNwZWNpZmllZCBEYXRhQnVmZmVyIHdpdGggb25lIG9mIHRoZSBmb2xsb3dpbmcgdHlwZXM6Ci0gICAgICogRGF0YUJ1ZmZlci5UWVBFX0JZVEUsIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQsIERhdGFCdWZmZXIuVFlQRV9JTlQsCi0gICAgICogRGF0YUJ1ZmZlci5UWVBFX1NIT1JULCBEYXRhQnVmZmVyLlRZUEVfRkxPQVQsIG9yIERhdGFCdWZmZXIuVFlQRV9ET1VCTEUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3Rhbmd1bGFyIHBpeGVsIGFyZWEuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3Rhbmd1bGFyIHBpeGVsIGFyZWEuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ3VsYXIgcGl4ZWwgYXJlYS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ3VsYXIgcGl4ZWwgYXJlYS4KLSAgICAgKiBAcGFyYW0gb2JqCi0gICAgICogICAgICAgICAgICB0aGUgT2JqZWN0IGlzIGFuIGFycmF5IHdpdGggdGhlIHByaW1pdGl2ZSB0eXBlLCB3aGVyZSB0aGUKLSAgICAgKiAgICAgICAgICAgIHJlc3VsdCBhcnJheSB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgcGl4ZWwgZGF0YSBmb3IgdGhlIHNwZWNpZmllZCByZWN0YW5ndWxhciBhcmVhIG9mCi0gICAgICogICAgICAgICBwaXhlbHMgb2YgdGhlIHNwZWNpZmllZCBEYXRhQnVmZmVyIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IGdldERhdGFFbGVtZW50cyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgT2JqZWN0IG9iaiwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGludCBudW1EYXRhRWxlbWVudHMgPSBnZXROdW1EYXRhRWxlbWVudHMoKTsKLSAgICAgICAgaW50IGlkeCA9IDA7Ci0KLSAgICAgICAgc3dpdGNoIChnZXRUcmFuc2ZlclR5cGUoKSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBieXRlIGJkYXRhW107Ci0gICAgICAgICAgICAgICAgYnl0ZSBiYnVmW10gPSBudWxsOwotCi0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJkYXRhID0gbmV3IGJ5dGVbbnVtRGF0YUVsZW1lbnRzICogdyAqIGhdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGJkYXRhID0gKGJ5dGVbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgICAgICAgICAgYmJ1ZiA9IChieXRlW10pZ2V0RGF0YUVsZW1lbnRzKGosIGksIGJidWYsIGRhdGEpOwotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1EYXRhRWxlbWVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJkYXRhW2lkeCsrXSA9IGJidWZbbl07Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgb2JqID0gYmRhdGE7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1NIT1JUOgotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNkYXRhW107Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2J1ZltdID0gbnVsbDsKLQotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBzZGF0YSA9IG5ldyBzaG9ydFtudW1EYXRhRWxlbWVudHMgKiB3ICogaF07Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgc2RhdGEgPSAoc2hvcnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgICAgICAgICAgc2J1ZiA9IChzaG9ydFtdKWdldERhdGFFbGVtZW50cyhqLCBpLCBzYnVmLCBkYXRhKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtRGF0YUVsZW1lbnRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBzZGF0YVtpZHgrK10gPSBzYnVmW25dOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIG9iaiA9IHNkYXRhOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgaW50IGlkYXRhW107Ci0gICAgICAgICAgICAgICAgaW50IGlidWZbXSA9IG51bGw7Ci0KLSAgICAgICAgICAgICAgICBpZiAob2JqID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgaWRhdGEgPSBuZXcgaW50W251bURhdGFFbGVtZW50cyAqIHcgKiBoXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBpZGF0YSA9IChpbnRbXSlvYmo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgICAgICAgICAgaWJ1ZiA9IChpbnRbXSlnZXREYXRhRWxlbWVudHMoaiwgaSwgaWJ1ZiwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBuID0gMDsgbiA8IG51bURhdGFFbGVtZW50czsgbisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWRhdGFbaWR4KytdID0gaWJ1ZltuXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBvYmogPSBpZGF0YTsKLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfRkxPQVQ6Ci0gICAgICAgICAgICAgICAgZmxvYXQgZmRhdGFbXTsKLSAgICAgICAgICAgICAgICBmbG9hdCBmYnVmW10gPSBudWxsOwotCi0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZkYXRhID0gbmV3IGZsb2F0W251bURhdGFFbGVtZW50cyAqIHcgKiBoXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBmZGF0YSA9IChmbG9hdFtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IHg7IGogPCB4ICsgdzsgaisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmYnVmID0gKGZsb2F0W10pZ2V0RGF0YUVsZW1lbnRzKGosIGksIGZidWYsIGRhdGEpOwotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1EYXRhRWxlbWVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZkYXRhW2lkeCsrXSA9IGZidWZbbl07Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgb2JqID0gZmRhdGE7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRToKLSAgICAgICAgICAgICAgICBkb3VibGUgZGRhdGFbXTsKLSAgICAgICAgICAgICAgICBkb3VibGUgZGJ1ZltdID0gbnVsbDsKLQotICAgICAgICAgICAgICAgIGlmIChvYmogPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBkZGF0YSA9IG5ldyBkb3VibGVbbnVtRGF0YUVsZW1lbnRzICogdyAqIGhdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGRkYXRhID0gKGRvdWJsZVtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaiA9IHg7IGogPCB4ICsgdzsgaisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkYnVmID0gKGRvdWJsZVtdKWdldERhdGFFbGVtZW50cyhqLCBpLCBkYnVmLCBkYXRhKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtRGF0YUVsZW1lbnRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkZGF0YVtpZHgrK10gPSBkYnVmW25dOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIG9iaiA9IGRkYXRhOwotICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gb2JqOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgZm9yIGEgc2luZ2xlIHBpeGVsIGluIHRoZSBzcGVjaWZpZWQgRGF0YUJ1ZmZlciBmcm9tIGEKLSAgICAgKiBwcmltaXRpdmUgYXJyYXkgd2l0aCBvbmUgb2YgdGhlIGZvbGxvd2luZyB0eXBlczogRGF0YUJ1ZmZlci5UWVBFX0JZVEUsCi0gICAgICogRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCwgRGF0YUJ1ZmZlci5UWVBFX0lOVCwgRGF0YUJ1ZmZlci5UWVBFX1NIT1JULAotICAgICAqIERhdGFCdWZmZXIuVFlQRV9GTE9BVCwgb3IgRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiBwaXhlbC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiBwaXhlbC4KLSAgICAgKiBAcGFyYW0gb2JqCi0gICAgICogICAgICAgICAgICB0aGUgT2JqZWN0IC0gdGhlIGFycmF5IG9mIHByaW1pdGl2ZSBwaXhlbCBkYXRhIHRvIGJlIHNldC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0RGF0YUVsZW1lbnRzKGludCB4LCBpbnQgeSwgT2JqZWN0IG9iaiwgRGF0YUJ1ZmZlciBkYXRhKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgZWxlbWVudHMgZm9yIGEgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMgaW4gdGhlIHNwZWNpZmllZAotICAgICAqIERhdGFCdWZmZXIgZnJvbSBhIHByaW1pdGl2ZSBhcnJheSB3aXRoIG9uZSBvZiB0aGUgZm9sbG93aW5nIHR5cGVzOgotICAgICAqIERhdGFCdWZmZXIuVFlQRV9CWVRFLCBEYXRhQnVmZmVyLlRZUEVfVVNIT1JULCBEYXRhQnVmZmVyLlRZUEVfSU5ULAotICAgICAqIERhdGFCdWZmZXIuVFlQRV9TSE9SVCwgRGF0YUJ1ZmZlci5UWVBFX0ZMT0FULCBvciBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QgLSB0aGUgYXJyYXkgb2YgcHJpbWl0aXZlIHBpeGVsIGRhdGEgdG8gYmUgc2V0LgotICAgICAqIEBwYXJhbSBkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXREYXRhRWxlbWVudHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIE9iamVjdCBvYmosIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpbnQgbnVtRGF0YUVsZW1lbnRzID0gZ2V0TnVtRGF0YUVsZW1lbnRzKCk7Ci0gICAgICAgIGludCBpZHggPSAwOwotCi0gICAgICAgIHN3aXRjaCAoZ2V0VHJhbnNmZXJUeXBlKCkpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgYnl0ZSBiYnVmW10gPSBuZXcgYnl0ZVtudW1EYXRhRWxlbWVudHNdOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSB5OyBpIDwgeSArIGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtRGF0YUVsZW1lbnRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBiYnVmW25dID0gKChieXRlW10pb2JqKVtpZHgrK107Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBzZXREYXRhRWxlbWVudHMoaiwgaSwgYmJ1ZiwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfU0hPUlQ6Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQ6Ci0gICAgICAgICAgICAgICAgc2hvcnQgc2J1ZltdID0gbmV3IHNob3J0W251bURhdGFFbGVtZW50c107Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1EYXRhRWxlbWVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNidWZbbl0gPSAoKHNob3J0W10pb2JqKVtpZHgrK107Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBzZXREYXRhRWxlbWVudHMoaiwgaSwgc2J1ZiwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaWJ1ZltdID0gbmV3IGludFtudW1EYXRhRWxlbWVudHNdOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSB5OyBpIDwgeSArIGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtRGF0YUVsZW1lbnRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpYnVmW25dID0gKChpbnRbXSlvYmopW2lkeCsrXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIHNldERhdGFFbGVtZW50cyhqLCBpLCBpYnVmLCBkYXRhKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfRkxPQVQ6Ci0gICAgICAgICAgICAgICAgZmxvYXQgZmJ1ZltdID0gbmV3IGZsb2F0W251bURhdGFFbGVtZW50c107Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1EYXRhRWxlbWVudHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZidWZbbl0gPSAoKGZsb2F0W10pb2JqKVtpZHgrK107Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBzZXREYXRhRWxlbWVudHMoaiwgaSwgZmJ1ZiwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRToKLSAgICAgICAgICAgICAgICBkb3VibGUgZGJ1ZltdID0gbmV3IGRvdWJsZVtudW1EYXRhRWxlbWVudHNdOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSB5OyBpIDwgeSArIGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtRGF0YUVsZW1lbnRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkYnVmW25dID0gKChkb3VibGVbXSlvYmopW2lkeCsrXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIHNldERhdGFFbGVtZW50cyhqLCBpLCBkYnVmLCBkYXRhKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIG5ldyBTYW1wbGVNb2RlbCB3aXRoIHRoZSBzcGVjaWZpZWQgYmFuZHMgb2YgdGhpcyBTYW1wbGVNb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmFuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBiYW5kcyBmcm9tIHRoaXMgU2FtcGxlTW9kZWwuCi0gICAgICogQHJldHVybiB0aGUgU2FtcGxlTW9kZWwgd2l0aCB0aGUgc3BlY2lmaWVkIGJhbmRzIG9mIHRoaXMgU2FtcGxlTW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFNhbXBsZU1vZGVsIGNyZWF0ZVN1YnNldFNhbXBsZU1vZGVsKGludCBiYW5kc1tdKTsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIFNhbXBsZU1vZGVsIHdoaWNoIGhhcyB0aGUgc2FtZSBkYXRhIGFzIGluIHRoaXMgU2FtcGxlTW9kZWwKLSAgICAgKiB3aXRoIGEgZGlmZmVyZW50IHdpZHRoIGFuZCBoZWlnaHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGExCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEByZXR1cm4gdGhlIFNhbXBsZU1vZGVsIHdoaWNoIGhhcyB0aGUgc2FtZSBkYXRhIGFzIGluIHRoaXMgU2FtcGxlTW9kZWwKLSAgICAgKiAgICAgICAgIHdpdGggYSBkaWZmZXJlbnQgd2lkdGggYW5kIGhlaWdodC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgU2FtcGxlTW9kZWwgY3JlYXRlQ29tcGF0aWJsZVNhbXBsZU1vZGVsKGludCBhMCwgaW50IGExKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNhbXBsZXMgb2YgdGhlIHNwZWNpZmllZCBwaXhlbCBhcyBhbiBpbnRlZ2VyIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEBwYXJhbSBpQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnRlZ2VyIGFycmF5IHdoZXJlIHJlc3VsdCB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgaW50ZWdlciBhcnJheSB3aXRoIHRoZSBzYW1wbGVzIG9mIHRoZSBzcGVjaWZpZWQgcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldFBpeGVsKGludCB4LCBpbnQgeSwgaW50IGlBcnJheVtdLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpbnQgcGl4ZWxbXTsKLQotICAgICAgICBpZiAoaUFycmF5ID09IG51bGwpIHsKLSAgICAgICAgICAgIHBpeGVsID0gbmV3IGludFtudW1CYW5kc107Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBwaXhlbCA9IGlBcnJheTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgcGl4ZWxbaV0gPSBnZXRTYW1wbGUoeCwgeSwgaSwgZGF0YSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcGl4ZWw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhIHBpeGVsIG9mIHRoZSBEYXRhQnVmZmVyIGZyb20gYSBpbnRlZ2VyIGFycmF5IG9mIHNhbXBsZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIGlBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGludGVnZXIgYXJyYXkuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVsKGludCB4LCBpbnQgeSwgaW50IGlBcnJheVtdLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgIHNldFNhbXBsZSh4LCB5LCBpLCBpQXJyYXlbaV0sIGRhdGEpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2FtcGxlcyBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsIGFzIGEgZmxvYXQgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIGZBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGZsb2F0IGFycmF5IHdoZXJlIHJlc3VsdCB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgZmxvYXQgYXJyYXkgd2l0aCB0aGUgc2FtcGxlcyBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldFBpeGVsKGludCB4LCBpbnQgeSwgZmxvYXQgZkFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGZsb2F0IHBpeGVsW107Ci0KLSAgICAgICAgaWYgKGZBcnJheSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBwaXhlbCA9IG5ldyBmbG9hdFtudW1CYW5kc107Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBwaXhlbCA9IGZBcnJheTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgcGl4ZWxbaV0gPSBnZXRTYW1wbGVGbG9hdCh4LCB5LCBpLCBkYXRhKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBwaXhlbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIGEgcGl4ZWwgb2YgdGhlIERhdGFCdWZmZXIgZnJvbSBhIGZsb2F0IGFycmF5IG9mIHNhbXBsZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIGZBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGZsb2F0IGFycmF5LgotICAgICAqIEBwYXJhbSBkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbChpbnQgeCwgaW50IHksIGZsb2F0IGZBcnJheVtdLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgIHNldFNhbXBsZSh4LCB5LCBpLCBmQXJyYXlbaV0sIGRhdGEpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2FtcGxlcyBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsIGFzIGEgZG91YmxlIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEBwYXJhbSBkQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkb3VibGUgYXJyYXkgd2hlcmUgcmVzdWx0IHdpbGwgYmUgc3RvcmVkLgotICAgICAqIEBwYXJhbSBkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcmV0dXJuIHRoZSBkb3VibGUgYXJyYXkgd2l0aCB0aGUgc2FtcGxlcyBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsLgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGVbXSBnZXRQaXhlbChpbnQgeCwgaW50IHksIGRvdWJsZSBkQXJyYXlbXSwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ID49IHRoaXMud2lkdGggfHwgeSA+PSB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgZG91YmxlIHBpeGVsW107Ci0KLSAgICAgICAgaWYgKGRBcnJheSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBwaXhlbCA9IG5ldyBkb3VibGVbbnVtQmFuZHNdOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcGl4ZWwgPSBkQXJyYXk7Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgIHBpeGVsW2ldID0gZ2V0U2FtcGxlRG91YmxlKHgsIHksIGksIGRhdGEpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHBpeGVsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgYSBwaXhlbCBvZiB0aGUgRGF0YUJ1ZmZlciBmcm9tIGEgZG91YmxlIGFycmF5IG9mIHNhbXBsZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIGRBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGRvdWJsZSBhcnJheS4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWwoaW50IHgsIGludCB5LCBkb3VibGUgZEFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgc2V0U2FtcGxlKHgsIHksIGksIGRBcnJheVtpXSwgZGF0YSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzYW1wbGUgb2YgYSBzcGVjaWZpZWQgYmFuZCBmb3IgdGhlIHNwZWNpZmllZCBwaXhlbCBhcyBhbgotICAgICAqIGludGVnZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgc2FtcGxlIG9mIGEgc3BlY2lmaWVkIGJhbmQgZm9yIHRoZSBzcGVjaWZpZWQgcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgRGF0YUJ1ZmZlciBkYXRhKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNhbXBsZSBvZiBhIHNwZWNpZmllZCBiYW5kIGZvciB0aGUgc3BlY2lmaWVkIHBpeGVsIGFzIGEgZmxvYXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcGl4ZWwuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgc2FtcGxlIG9mIGEgc3BlY2lmaWVkIGJhbmQgZm9yIHRoZSBzcGVjaWZpZWQgcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldFNhbXBsZUZsb2F0KGludCB4LCBpbnQgeSwgaW50IGIsIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICByZXR1cm4gZ2V0U2FtcGxlKHgsIHksIGIsIGRhdGEpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNhbXBsZSBvZiBhIHNwZWNpZmllZCBiYW5kIGZvciB0aGUgc3BlY2lmaWVkIHBpeGVsIGFzIGEgZG91YmxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHBpeGVsLgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGJhbmQuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEByZXR1cm4gdGhlIHNhbXBsZSBvZiBhIHNwZWNpZmllZCBiYW5kIGZvciB0aGUgc3BlY2lmaWVkIHBpeGVsLgotICAgICAqLwotICAgIHB1YmxpYyBkb3VibGUgZ2V0U2FtcGxlRG91YmxlKGludCB4LCBpbnQgeSwgaW50IGIsIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICByZXR1cm4gZ2V0U2FtcGxlKHgsIHksIGIsIGRhdGEpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNhbXBsZXMgb2YgdGhlIHNwZWNpZmllZCByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscyBhcyBhbgotICAgICAqIGludGVnZXIgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaUFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgaW50ZWdlciBhcnJheSB3aGVyZSByZXN1bHQgd2lsbCBiZSBzdG9yZWQuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEByZXR1cm4gdGhlIGludGVnZXIgYXJyYXkgd2l0aCB0aGUgc2FtcGxlcyBvZiB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyCi0gICAgICogICAgICAgICBhcmVhIG9mIHBpeGVscy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50W10gZ2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCArIHcgPiB0aGlzLndpZHRoIHx8IHkgKyBoID4gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGludCBwaXhlbHNbXTsKLSAgICAgICAgaW50IGlkeCA9IDA7Ci0KLSAgICAgICAgaWYgKGlBcnJheSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBwaXhlbHMgPSBuZXcgaW50W3cgKiBoICogbnVtQmFuZHNdOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcGl4ZWxzID0gaUFycmF5OwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBuID0gMDsgbiA8IG51bUJhbmRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgcGl4ZWxzW2lkeCsrXSA9IGdldFNhbXBsZShqLCBpLCBuLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHBpeGVsczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIGFsbCBvZiB0aGUgc2FtcGxlcyBmb3IgYSByZWN0YW5ndWxhciBhcmVhIG9mIHBpeGVscyBvZiB0aGUKLSAgICAgKiBEYXRhQnVmZmVyIGZyb20gYW4gaW50ZWdlciBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBpQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnRlZ2VyIGFycmF5LgotICAgICAqIEBwYXJhbSBkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBpQXJyYXlbXSwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICh4IDwgMCB8fCB5IDwgMCB8fCB4ICsgdyA+IHRoaXMud2lkdGggfHwgeSArIGggPiB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaW50IGlkeCA9IDA7Ci0gICAgICAgIGZvciAoaW50IGkgPSB5OyBpIDwgeSArIGg7IGkrKykgewotICAgICAgICAgICAgZm9yIChpbnQgaiA9IHg7IGogPCB4ICsgdzsgaisrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1CYW5kczsgbisrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNldFNhbXBsZShqLCBpLCBuLCBpQXJyYXlbaWR4KytdLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzYW1wbGVzIG9mIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMgYXMgYSBmbG9hdAotICAgICAqIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGZBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGZsb2F0IGFycmF5IHdoZXJlIHJlc3VsdCB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgZmxvYXQgYXJyYXkgd2l0aCB0aGUgc2FtcGxlcyBvZiB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyCi0gICAgICogICAgICAgICBhcmVhIG9mIHBpeGVscy4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXRbXSBnZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGZsb2F0IGZBcnJheVtdLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggKyB3ID4gdGhpcy53aWR0aCB8fCB5ICsgaCA+IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBmbG9hdCBwaXhlbHNbXTsKLSAgICAgICAgaW50IGlkeCA9IDA7Ci0KLSAgICAgICAgaWYgKGZBcnJheSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBwaXhlbHMgPSBuZXcgZmxvYXRbdyAqIGggKiBudW1CYW5kc107Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBwaXhlbHMgPSBmQXJyYXk7Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtQmFuZHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICBwaXhlbHNbaWR4KytdID0gZ2V0U2FtcGxlRmxvYXQoaiwgaSwgbiwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBwaXhlbHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhbGwgb2YgdGhlIHNhbXBsZXMgZm9yIGEgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMgb2YgdGhlCi0gICAgICogRGF0YUJ1ZmZlciBmcm9tIGEgZmxvYXQgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gZkFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZmxvYXQgYXJyYXkuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgZmxvYXQgZkFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCArIHcgPiB0aGlzLndpZHRoIHx8IHkgKyBoID4gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGludCBpZHggPSAwOwotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtQmFuZHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICBzZXRTYW1wbGUoaiwgaSwgbiwgZkFycmF5W2lkeCsrXSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2FtcGxlcyBvZiB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YgcGl4ZWxzIGFzIGEgZG91YmxlCi0gICAgICogYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gZEFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZG91YmxlIGFycmF5IHdoZXJlIHJlc3VsdCB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgZG91YmxlIGFycmF5IHdpdGggdGhlIHNhbXBsZXMgb2YgdGhlIHNwZWNpZmllZCByZWN0YW5ndWxhcgotICAgICAqICAgICAgICAgYXJlYSBvZiBwaXhlbHMuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZVtdIGdldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgZG91YmxlIGRBcnJheVtdLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggKyB3ID4gdGhpcy53aWR0aCB8fCB5ICsgaCA+IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBkb3VibGUgcGl4ZWxzW107Ci0gICAgICAgIGludCBpZHggPSAwOwotCi0gICAgICAgIGlmIChkQXJyYXkgPT0gbnVsbCkgewotICAgICAgICAgICAgcGl4ZWxzID0gbmV3IGRvdWJsZVt3ICogaCAqIG51bUJhbmRzXTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHBpeGVscyA9IGRBcnJheTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSB5OyBpIDwgeSArIGg7IGkrKykgewotICAgICAgICAgICAgZm9yIChpbnQgaiA9IHg7IGogPCB4ICsgdzsgaisrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCBudW1CYW5kczsgbisrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHBpeGVsc1tpZHgrK10gPSBnZXRTYW1wbGVEb3VibGUoaiwgaSwgbiwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBwaXhlbHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhbGwgb2YgdGhlIHNhbXBsZXMgZm9yIGEgcmVjdGFuZ3VsYXIgYXJlYSBvZiBwaXhlbHMgb2YgdGhlCi0gICAgICogRGF0YUJ1ZmZlciBmcm9tIGEgZG91YmxlIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGRBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGRvdWJsZSBhcnJheS4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBkb3VibGUgZEFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCArIHcgPiB0aGlzLndpZHRoIHx8IHkgKyBoID4gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGludCBpZHggPSAwOwotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIGZvciAoaW50IG4gPSAwOyBuIDwgbnVtQmFuZHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICBzZXRTYW1wbGUoaiwgaSwgbiwgZEFycmF5W2lkeCsrXSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhIHNhbXBsZSBvZiB0aGUgc3BlY2lmaWVkIGJhbmQgZm9yIHRoZSBzcGVjaWZpZWQgcGl4ZWwgaW4gdGhlCi0gICAgICogRGF0YUJ1ZmZlciBhcyBpbnRlZ2VyIHZhbHVlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBwaXhlbC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIHNhbXBsZSBhcyBhbiBpbnRlZ2VyIHZhbHVlLgotICAgICAqIEBwYXJhbSBkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgaW50IHMsIERhdGFCdWZmZXIgZGF0YSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzYW1wbGVzIG9mIGEgc3BlY2lmaWVkIGJhbmQgZm9yIGEgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YKLSAgICAgKiBwaXhlbHMgYXMgYSBpbnRlZ2VyIGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBiYW5kLgotICAgICAqIEBwYXJhbSBpQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnRlZ2VyIGFycmF5IHdoZXJlIHJlc3VsdCB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgc2FtcGxlcyBvZiBhIHNwZWNpZmllZCBiYW5kIGZvciBhIHNwZWNpZmllZCByZWN0YW5ndWxhciBhcmVhCi0gICAgICogICAgICAgICBvZiBwaXhlbHMuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpbnQgc2FtcGxlc1tdOwotICAgICAgICBpbnQgaWR4ID0gMDsKLQotICAgICAgICBpZiAoaUFycmF5ID09IG51bGwpIHsKLSAgICAgICAgICAgIHNhbXBsZXMgPSBuZXcgaW50W3cgKiBoXTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHNhbXBsZXMgPSBpQXJyYXk7Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIHNhbXBsZXNbaWR4KytdID0gZ2V0U2FtcGxlKGosIGksIGIsIGRhdGEpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHNhbXBsZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2FtcGxlcyBmcm9tIGFuIGludGVnZXIgYXJyYXkgaW4gdGhlIHNwZWNpZmllZCBiYW5kIGZvciB0aGUKLSAgICAgKiBzcGVjaWZpZWQgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gaUFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgaW50ZWdlciBhcnJheS4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0U2FtcGxlcyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGIsIGludCBpQXJyYXlbXSwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGludCBpZHggPSAwOwotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIHNldFNhbXBsZShqLCBpLCBiLCBpQXJyYXlbaWR4KytdLCBkYXRhKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNhbXBsZXMgb2YgYSBzcGVjaWZpZWQgYmFuZCBmb3IgYSBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIgYXJlYSBvZgotICAgICAqIHBpeGVscyBhcyBhIGZsb2F0IGFycmF5LgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBiYW5kLgotICAgICAqIEBwYXJhbSBmQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbG9hdCBhcnJheSB3aGVyZSByZXN1bHQgd2lsbCBiZSBzdG9yZWQuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBkYXRhLgotICAgICAqIEByZXR1cm4gdGhlIHNhbXBsZXMgb2YgYSBzcGVjaWZpZWQgYmFuZCBmb3IgYSBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIgYXJlYQotICAgICAqICAgICAgICAgb2YgcGl4ZWxzLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdIGdldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBmbG9hdCBmQXJyYXlbXSwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGZsb2F0IHNhbXBsZXNbXTsKLSAgICAgICAgaW50IGlkeCA9IDA7Ci0KLSAgICAgICAgaWYgKGZBcnJheSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBzYW1wbGVzID0gbmV3IGZsb2F0W3cgKiBoXTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHNhbXBsZXMgPSBmQXJyYXk7Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIHNhbXBsZXNbaWR4KytdID0gZ2V0U2FtcGxlRmxvYXQoaiwgaSwgYiwgZGF0YSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gc2FtcGxlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBzYW1wbGVzIGZyb20gYW4gZmxvYXQgYXJyYXkgaW4gdGhlIHNwZWNpZmllZCBiYW5kIGZvciB0aGUKLSAgICAgKiBzcGVjaWZpZWQgcmVjdGFuZ2xlIG9mIHBpeGVscy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gZkFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZmxvYXQgYXJyYXkuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBmbG9hdCBmQXJyYXlbXSwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGludCBpZHggPSAwOwotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIHNldFNhbXBsZShqLCBpLCBiLCBmQXJyYXlbaWR4KytdLCBkYXRhKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNhbXBsZXMgb2YgYSBzcGVjaWZpZWQgYmFuZCBmb3IgYSBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIgYXJlYSBvZgotICAgICAqIHBpeGVscyBhcyBhIGRvdWJsZSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgcmVjdGFuZ2xlLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gZEFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZG91YmxlIGFycmF5IHdoZXJlIHJlc3VsdCB3aWxsIGJlIHN0b3JlZC4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHJldHVybiB0aGUgc2FtcGxlcyBvZiBhIHNwZWNpZmllZCBiYW5kIGZvciBhIHNwZWNpZmllZCByZWN0YW5ndWxhciBhcmVhCi0gICAgICogICAgICAgICBvZiBwaXhlbHMuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZVtdIGdldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBkb3VibGUgZEFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBkb3VibGUgc2FtcGxlc1tdOwotICAgICAgICBpbnQgaWR4ID0gMDsKLQotICAgICAgICBpZiAoZEFycmF5ID09IG51bGwpIHsKLSAgICAgICAgICAgIHNhbXBsZXMgPSBuZXcgZG91YmxlW3cgKiBoXTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHNhbXBsZXMgPSBkQXJyYXk7Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IgKGludCBpID0geTsgaSA8IHkgKyBoOyBpKyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGogPSB4OyBqIDwgeCArIHc7IGorKykgewotICAgICAgICAgICAgICAgIHNhbXBsZXNbaWR4KytdID0gZ2V0U2FtcGxlRG91YmxlKGosIGksIGIsIGRhdGEpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHNhbXBsZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2FtcGxlcyBmcm9tIGFuIGRvdWJsZSBhcnJheSBpbiB0aGUgc3BlY2lmaWVkIGJhbmQgZm9yIHRoZQotICAgICAqIHNwZWNpZmllZCByZWN0YW5nbGUgb2YgcGl4ZWxzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSByZWN0YW5nbGUuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZS4KLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBiYW5kLgotICAgICAqIEBwYXJhbSBkQXJyYXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkb3VibGUgYXJyYXkuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBkb3VibGUgZEFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpbnQgaWR4ID0gMDsKLSAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICBzZXRTYW1wbGUoaiwgaSwgYiwgZEFycmF5W2lkeCsrXSwgZGF0YSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIGEgc2FtcGxlIG9mIHRoZSBzcGVjaWZpZWQgYmFuZCBmb3IgdGhlIHNwZWNpZmllZCBwaXhlbCBpbiB0aGUKLSAgICAgKiBEYXRhQnVmZmVyIGFzIGZsb2F0IHZhbHVlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBwaXhlbC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIHNhbXBsZSBhcyBmbG9hdCB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0U2FtcGxlKGludCB4LCBpbnQgeSwgaW50IGIsIGZsb2F0IHMsIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBzZXRTYW1wbGUoeCwgeSwgYiwgKGludClzLCBkYXRhKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIGEgc2FtcGxlIG9mIHRoZSBzcGVjaWZpZWQgYmFuZCBmb3IgdGhlIHNwZWNpZmllZCBwaXhlbCBpbiB0aGUKLSAgICAgKiBEYXRhQnVmZmVyIGFzIGRvdWJsZSB2YWx1ZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHBpeGVsLgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGJhbmQuCi0gICAgICogQHBhcmFtIHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzYW1wbGUgYXMgZG91YmxlIHZhbHVlLgotICAgICAqIEBwYXJhbSBkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgZG91YmxlIHMsIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBzZXRTYW1wbGUoeCwgeSwgYiwgKGludClzLCBkYXRhKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgRGF0YUJ1ZmZlciBvYmplY3Qgd2hpY2ggY29ycmVzcG9uZHMgdG8gdGhlIFNhbXBsZU1vZGVsLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIERhdGFCdWZmZXIgb2JqZWN0IHdoaWNoIGNvcnJlc3BvbmRzIHRvIHRoZSBTYW1wbGVNb2RlbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgRGF0YUJ1ZmZlciBjcmVhdGVEYXRhQnVmZmVyKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzYW1wbGUgc2l6ZSBpbiBiaXRzIGZvciB0aGUgc3BlY2lmaWVkIGJhbmQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJhbmQKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBzYW1wbGUgc2l6ZSBpbiBiaXRzIGZvciB0aGUgc3BlY2lmaWVkIGJhbmQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXRTYW1wbGVTaXplKGludCBiYW5kKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgdGhlIHNhbXBsZSBzaXplIGluIGJpdHMgZm9yIGFsbCBiYW5kcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIHRoZSBzYW1wbGUgc2l6ZSBpbiBiaXRzIGZvciBhbGwgYmFuZHMuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludFtdIGdldFNhbXBsZVNpemUoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHdpZHRoIG9mIHRoZSBpbWFnZSBkYXRhIG9mIHRoaXMgU2FtcGxlTW9kZWwgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHdpZHRoIG9mIHRoZSBpbWFnZSBkYXRhIG9mIHRoaXMgU2FtcGxlTW9kZWwgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0V2lkdGgoKSB7Ci0gICAgICAgIHJldHVybiB3aWR0aDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0cmFuc2ZlciB0eXBlIHVzZWQgdG8gdHJhbnNmZXIgcGl4ZWxzIHZpYSB0aGUgZ2V0RGF0YUVsZW1lbnRzCi0gICAgICogYW5kIHNldERhdGFFbGVtZW50cyBtZXRob2RzLiBUcmFuc2ZlciB0eXBlIHZhbHVlIGNhbiBiZSBvbmUgb2YgdGhlCi0gICAgICogcHJlZGVmaW5lZCB0eXBlIGZyb20gRGF0YUJ1ZmZlciBjbGFzcyBvciBub3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdHJhbnNmZXIgdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFRyYW5zZmVyVHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIGRhdGFUeXBlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIG51bWJlciBvZiBkYXRhIGVsZW1lbnRzIGZvciBwaXhlbCB0cmFuc2ZlcnJpbmcgdmlhIHRoZQotICAgICAqIGdldERhdGFFbGVtZW50cyBhbmQgc2V0RGF0YUVsZW1lbnRzIG1ldGhvZHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIGRhdGEgZWxlbWVudHMgZm9yIHBpeGVsIHRyYW5zZmVycmluZyB2aWEgdGhlCi0gICAgICogICAgICAgICBnZXREYXRhRWxlbWVudHMgYW5kIHNldERhdGFFbGVtZW50cyBtZXRob2RzLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0TnVtRGF0YUVsZW1lbnRzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBudW1iZXIgb2YgYmFuZHMgaW4gdGhlIGltYWdlIGRhdGEgb2YgdGhpcyBTYW1wbGVNb2RlbCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIGJhbmRzIGluIHRoZSBpbWFnZSBkYXRhIG9mIHRoaXMgU2FtcGxlTW9kZWwgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0TnVtQmFuZHMoKSB7Ci0gICAgICAgIHJldHVybiBudW1CYW5kczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlIGRhdGEgb2YgdGhpcyBTYW1wbGVNb2RlbCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaGVpZ2h0IG9mIHRoZSBpbWFnZSBkYXRhIG9mIHRoaXMgU2FtcGxlTW9kZWwgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0SGVpZ2h0KCkgewotICAgICAgICByZXR1cm4gaGVpZ2h0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRhdGEgdHlwZSBvZiBpbWFnZSBkYXRhIG9mIHRoaXMgU2FtcGxlTW9kZWwgb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRhdGEgdHlwZSBvZiBpbWFnZSBkYXRhIG9mIHRoaXMgU2FtcGxlTW9kZWwgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBpbnQgZ2V0RGF0YVR5cGUoKSB7Ci0gICAgICAgIHJldHVybiBkYXRhVHlwZTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9TaG9ydExvb2t1cFRhYmxlLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvU2hvcnRMb29rdXBUYWJsZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0MzE5ZDU4Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9TaG9ydExvb2t1cFRhYmxlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMzcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKgotICogQGRhdGU6IE9jdCAxNCwgMjAwNQotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLS8qKgotICogVGhlIFNob3J0TG9va3VwVGFibGUgY2xhc3MgcHJvdmlkZXMgcHJvdmlkZXMgZnVuY3Rpb25hbGl0eSBmb3IgbG9va3VwCi0gKiBvcGVyYXRpb25zLCBhbmQgaXMgZGVmaW5lZCBieSBhbiBpbnB1dCBzaG9ydCBhcnJheSBmb3IgYmFuZHMgb3IgY29tcG9uZW50cyBvZgotICogaW1hZ2UgYW5kIGFuIG9mZnNldCB2YWx1ZS4gVGhlIG9mZnNldCB2YWx1ZSB3aWxsIGJlIHN1YnRyYWN0ZWQgZnJvbSB0aGUgaW5wdXQKLSAqIHZhbHVlcyBiZWZvcmUgaW5kZXhpbmcgdGhlIGlucHV0IGFycmF5cy4gVGhlIG91dHB1dCBvZiBhIGxvb2t1cCBvcGVyYXRpb24gaXMKLSAqIHJlcHJlc2VudGVkIGFzIGFuIHVuc2lnbmVkIHNob3J0IGFycmF5LgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIFNob3J0TG9va3VwVGFibGUgZXh0ZW5kcyBMb29rdXBUYWJsZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGF0YS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHNob3J0IGRhdGFbXVtdOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IFNob3J0TG9va3VwVGFibGUgd2l0aCB0aGUgc3BlY2lmaWVkIG9mZnNldCB2YWx1ZSBhbmQKLSAgICAgKiB0aGUgc3BlY2lmaWVkIHNob3J0IGFycmF5IHdoaWNoIHJlcHJlc2VudHMgbG9va3VwIHRhYmxlIGZvciBhbGwgYmFuZHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldCB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIFNob3J0TG9va3VwVGFibGUoaW50IG9mZnNldCwgc2hvcnRbXSBkYXRhKSB7Ci0gICAgICAgIHN1cGVyKG9mZnNldCwgMSk7Ci0gICAgICAgIHRoaXMuZGF0YSA9IG5ldyBzaG9ydFsxXVtkYXRhLmxlbmd0aF07Ci0gICAgICAgIC8vIFRoZSBkYXRhIGFycmF5IHN0b3JlZCBhcyBhIHJlZmVyZW5jZQotICAgICAgICB0aGlzLmRhdGFbMF0gPSBkYXRhOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBTaG9ydExvb2t1cFRhYmxlIHdpdGggdGhlIHNwZWNpZmllZCBvZmZzZXQgdmFsdWUgYW5kCi0gICAgICogdGhlIHNwZWNpZmllZCBzaG9ydCBhcnJheSBvZiBhcnJheXMgd2hpY2ggcmVwcmVzZW50cyBsb29rdXAgdGFibGUgZm9yCi0gICAgICogZWFjaCBiYW5kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgdmFsdWUuCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIGFycmF5IG9mIGFycmF5cyBmb3IgZWFjaCBiYW5kLgotICAgICAqLwotICAgIHB1YmxpYyBTaG9ydExvb2t1cFRhYmxlKGludCBvZmZzZXQsIHNob3J0W11bXSBkYXRhKSB7Ci0gICAgICAgIHN1cGVyKG9mZnNldCwgZGF0YS5sZW5ndGgpOwotICAgICAgICB0aGlzLmRhdGEgPSBuZXcgc2hvcnRbZGF0YS5sZW5ndGhdW2RhdGFbMF0ubGVuZ3RoXTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBkYXRhLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAvLyBUaGUgZGF0YSBhcnJheSBmb3IgZWFjaCBiYW5kIHN0b3JlZCBhcyBhIHJlZmVyZW5jZQotICAgICAgICAgICAgdGhpcy5kYXRhW2ldID0gZGF0YVtpXTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvb2t1cCB0YWJsZSBvZiB0aGlzIFNob3J0TG9va3VwVGFibGUgb2JqZWN0LiBJZiB0aGlzCi0gICAgICogU2hvcnRMb29rdXBUYWJsZSBvYmplY3QgaGFzIG9uZSBzaG9ydCBhcnJheSBmb3IgYWxsIGJhbmRzLCB0aGUgcmV0dXJuZWQKLSAgICAgKiBhcnJheSBsZW5ndGggaXMgb25lLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGxvb2t1cCB0YWJsZSBvZiB0aGlzIFNob3J0TG9va3VwVGFibGUgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBmaW5hbCBzaG9ydFtdW10gZ2V0VGFibGUoKSB7Ci0gICAgICAgIHJldHVybiBkYXRhOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBzaG9ydCBhcnJheSB3aGljaCBjb250YWlucyBzYW1wbGVzIG9mIHRoZSBzcGVjaWZpZWQgcGl4ZWwgd2hpY2gKLSAgICAgKiBpcyB0cmFuc2xhdGVkIHdpdGggdGhlIGxvb2t1cCB0YWJsZSBvZiB0aGlzIFNob3J0TG9va3VwVGFibGUgb2JqZWN0LiBUaGUKLSAgICAgKiByZXN1bHRlZCBhcnJheSBpcyBzdG9yZWQgdG8gdGhlIGRzdCBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjCi0gICAgICogICAgICAgICAgICB0aGUgc291cmNlIGFycmF5LgotICAgICAqIEBwYXJhbSBkc3QKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXN0aW5hdGlvbiBhcnJheSB3aGVyZSB0aGUgcmVzdWx0IGNhbiBiZSBzdG9yZWQuCi0gICAgICogQHJldHVybiB0aGUgc2hvcnQgYXJyYXkgb2YgdHJhbnNsYXRlZCBzYW1wbGVzIG9mIGEgcGl4ZWwuCi0gICAgICovCi0gICAgcHVibGljIHNob3J0W10gbG9va3VwUGl4ZWwoc2hvcnRbXSBzcmMsIHNob3J0W10gZHN0KSB7Ci0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0ID0gbmV3IHNob3J0W3NyYy5sZW5ndGhdOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IG9mZnNldCA9IGdldE9mZnNldCgpOwotICAgICAgICBpZiAoZ2V0TnVtQ29tcG9uZW50cygpID09IDEpIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgc3JjLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgZHN0W2ldID0gZGF0YVswXVtzcmNbaV0gLSBvZmZzZXRdOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBnZXROdW1Db21wb25lbnRzKCk7IGkrKykgewotICAgICAgICAgICAgICAgIGRzdFtpXSA9IGRhdGFbaV1bc3JjW2ldIC0gb2Zmc2V0XTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBkc3Q7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGxvb2t1cFBpeGVsKGludFtdIHNyYywgaW50W10gZHN0KSB7Ci0gICAgICAgIGlmIChkc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgZHN0ID0gbmV3IGludFtzcmMubGVuZ3RoXTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBvZmZzZXQgPSBnZXRPZmZzZXQoKTsKLSAgICAgICAgaWYgKGdldE51bUNvbXBvbmVudHMoKSA9PSAxKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHNyYy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgICAgIGRzdFtpXSA9IGRhdGFbMF1bc3JjW2ldIC0gb2Zmc2V0XTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgZ2V0TnVtQ29tcG9uZW50cygpOyBpKyspIHsKLSAgICAgICAgICAgICAgICBkc3RbaV0gPSBkYXRhW2ldW3NyY1tpXSAtIG9mZnNldF07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZHN0OwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9TaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2OWYzMzUzLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9TaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1MTkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlzOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgY2xhc3MgcmVwcmVzZW50cyBwaXhlbCBkYXRhIHdoZXJlIHNldmVyYWwKLSAqIHNhbXBsZXMgY29tYmluZSB0byBjcmVhdGUgYSBzaW5nbGUgcGl4ZWwgYW5kIGFyZSBzdG9yZWQgaW4gYSBzaW5nbGUgZGF0YQotICogYXJyYXkgZWxlbWVudC4gVGhpcyBjbGFzcyBzdXBwb3J0cyBUWVBFX0JZVEUsIFRZUEVfVVNIT1JULCBUWVBFX0lOVCBkYXRhCi0gKiB0eXBlcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIGV4dGVuZHMgU2FtcGxlTW9kZWwgewotCi0gICAgLyoqCi0gICAgICogVGhlIGJpdCBtYXNrcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBiaXRNYXNrc1tdOwotCi0gICAgLyoqCi0gICAgICogVGhlIGJpdCBvZmZzZXRzLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IGJpdE9mZnNldHNbXTsKLQotICAgIC8qKgotICAgICAqIFRoZSBiaXQgc2l6ZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgYml0U2l6ZXNbXTsKLQotICAgIC8qKgotICAgICAqIFRoZSBzY2FubGluZSBzdHJpZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgc2NhbmxpbmVTdHJpZGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbWF4IGJpdCBzaXplLgotICAgICAqLwotICAgIHByaXZhdGUgaW50IG1heEJpdFNpemU7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkYXRhVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGRhdGEgdHlwZSBvZiBzYW1wbGVzLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGJpdE1hc2tzCi0gICAgICogICAgICAgICAgICB0aGUgYml0IG1hc2tzIGZvciBhbGwgdGhlIGJhbmRzLgotICAgICAqLwotICAgIHB1YmxpYyBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKGludCBkYXRhVHlwZSwgaW50IHcsIGludCBoLCBpbnQgYml0TWFza3NbXSkgewotICAgICAgICB0aGlzKGRhdGFUeXBlLCB3LCBoLCB3LCBiaXRNYXNrcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICogcGFyYW1ldGVycy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUgb2YgdGhlIHNhbXBsZXMuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKiBAcGFyYW0gc2NhbmxpbmVTdHJpZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzY2FubGluZSBzdHJpZGUgb2YgdGhlIGltYWdlIGRhdGEuCi0gICAgICogQHBhcmFtIGJpdE1hc2tzCi0gICAgICogICAgICAgICAgICB0aGUgYml0IG1hc2tzIGZvciBhbGwgdGhlIGJhbmRzLgotICAgICAqLwotICAgIHB1YmxpYyBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKGludCBkYXRhVHlwZSwgaW50IHcsIGludCBoLCBpbnQgc2NhbmxpbmVTdHJpZGUsCi0gICAgICAgICAgICBpbnQgYml0TWFza3NbXSkgewotCi0gICAgICAgIHN1cGVyKGRhdGFUeXBlLCB3LCBoLCBiaXRNYXNrcy5sZW5ndGgpOwotCi0gICAgICAgIGlmIChkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfQllURSAmJiBkYXRhVHlwZSAhPSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUCi0gICAgICAgICAgICAgICAgJiYgZGF0YVR5cGUgIT0gRGF0YUJ1ZmZlci5UWVBFX0lOVCkgewotICAgICAgICAgICAgLy8gYXd0LjYxPVVuc3VwcG9ydGVkIGRhdGEgdHlwZTogezB9Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYxIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICBkYXRhVHlwZSkpOwotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5zY2FubGluZVN0cmlkZSA9IHNjYW5saW5lU3RyaWRlOwotICAgICAgICB0aGlzLmJpdE1hc2tzID0gYml0TWFza3MuY2xvbmUoKTsKLSAgICAgICAgdGhpcy5iaXRPZmZzZXRzID0gbmV3IGludFt0aGlzLm51bUJhbmRzXTsKLSAgICAgICAgdGhpcy5iaXRTaXplcyA9IG5ldyBpbnRbdGhpcy5udW1CYW5kc107Ci0KLSAgICAgICAgdGhpcy5tYXhCaXRTaXplID0gMDsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHRoaXMubnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgaW50IG9mZnNldCA9IDA7Ci0gICAgICAgICAgICBpbnQgc2l6ZSA9IDA7Ci0gICAgICAgICAgICBpbnQgbWFzayA9IGJpdE1hc2tzW2ldOwotCi0gICAgICAgICAgICBpZiAobWFzayAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgd2hpbGUgKChtYXNrICYgMSkgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICBtYXNrID4+Pj0gMTsKLSAgICAgICAgICAgICAgICAgICAgb2Zmc2V0Kys7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgd2hpbGUgKChtYXNrICYgMSkgPT0gMSkgewotICAgICAgICAgICAgICAgICAgICBtYXNrID4+Pj0gMTsKLSAgICAgICAgICAgICAgICAgICAgc2l6ZSsrOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChtYXNrICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjYyPVdyb25nIG1hc2sgOiB7MH0KLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MiIsIGJpdE1hc2tzW2ldKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHRoaXMuYml0T2Zmc2V0c1tpXSA9IG9mZnNldDsKLSAgICAgICAgICAgIHRoaXMuYml0U2l6ZXNbaV0gPSBzaXplOwotCi0gICAgICAgICAgICBpZiAodGhpcy5tYXhCaXRTaXplIDwgc2l6ZSkgewotICAgICAgICAgICAgICAgIHRoaXMubWF4Qml0U2l6ZSA9IHNpemU7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBnZXREYXRhRWxlbWVudHMoaW50IHgsIGludCB5LCBPYmplY3Qgb2JqLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBzd2l0Y2ggKGdldFRyYW5zZmVyVHlwZSgpKSB7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9CWVRFOgotICAgICAgICAgICAgICAgIGJ5dGUgYmRhdGFbXTsKLSAgICAgICAgICAgICAgICBpZiAob2JqID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgYmRhdGEgPSBuZXcgYnl0ZVsxXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBiZGF0YSA9IChieXRlW10pb2JqOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGJkYXRhWzBdID0gKGJ5dGUpZGF0YS5nZXRFbGVtKHkgKiBzY2FubGluZVN0cmlkZSArIHgpOwotICAgICAgICAgICAgICAgIG9iaiA9IGJkYXRhOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIHNob3J0IHNkYXRhW107Ci0gICAgICAgICAgICAgICAgaWYgKG9iaiA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNkYXRhID0gbmV3IHNob3J0WzFdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHNkYXRhID0gKHNob3J0W10pb2JqOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIHNkYXRhWzBdID0gKHNob3J0KWRhdGEuZ2V0RWxlbSh5ICogc2NhbmxpbmVTdHJpZGUgKyB4KTsKLSAgICAgICAgICAgICAgICBvYmogPSBzZGF0YTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaWRhdGFbXTsKLSAgICAgICAgICAgICAgICBpZiAob2JqID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgaWRhdGEgPSBuZXcgaW50WzFdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGlkYXRhID0gKGludFtdKW9iajsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBpZGF0YVswXSA9IGRhdGEuZ2V0RWxlbSh5ICogc2NhbmxpbmVTdHJpZGUgKyB4KTsKLSAgICAgICAgICAgICAgICBvYmogPSBpZGF0YTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gb2JqOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldERhdGFFbGVtZW50cyhpbnQgeCwgaW50IHksIE9iamVjdCBvYmosIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHN3aXRjaCAoZ2V0VHJhbnNmZXJUeXBlKCkpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgZGF0YS5zZXRFbGVtKHkgKiBzY2FubGluZVN0cmlkZSArIHgsICgoYnl0ZVtdKW9iailbMF0gJiAweGZmKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgICAgICBkYXRhLnNldEVsZW0oeSAqIHNjYW5saW5lU3RyaWRlICsgeCwgKChzaG9ydFtdKW9iailbMF0gJiAweGZmZmYpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfSU5UOgotICAgICAgICAgICAgICAgIGRhdGEuc2V0RWxlbSh5ICogc2NhbmxpbmVTdHJpZGUgKyB4LCAoKGludFtdKW9iailbMF0pOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgdGhpcyBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIG8KLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgb2JqZWN0IGlzIGVxdWFsIHRvIHRoZQotICAgICAqICAgICAgICAgc3BlY2lmaWVkIG9iamVjdCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3QgbykgewotICAgICAgICBpZiAoKG8gPT0gbnVsbCkgfHwgIShvIGluc3RhbmNlb2YgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkpIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgbW9kZWwgPSAoU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbClvOwotICAgICAgICByZXR1cm4gdGhpcy53aWR0aCA9PSBtb2RlbC53aWR0aCAmJiB0aGlzLmhlaWdodCA9PSBtb2RlbC5oZWlnaHQKLSAgICAgICAgICAgICAgICAmJiB0aGlzLm51bUJhbmRzID09IG1vZGVsLm51bUJhbmRzICYmIHRoaXMuZGF0YVR5cGUgPT0gbW9kZWwuZGF0YVR5cGUKLSAgICAgICAgICAgICAgICAmJiBBcnJheXMuZXF1YWxzKHRoaXMuYml0TWFza3MsIG1vZGVsLmJpdE1hc2tzKQotICAgICAgICAgICAgICAgICYmIEFycmF5cy5lcXVhbHModGhpcy5iaXRPZmZzZXRzLCBtb2RlbC5iaXRPZmZzZXRzKQotICAgICAgICAgICAgICAgICYmIEFycmF5cy5lcXVhbHModGhpcy5iaXRTaXplcywgbW9kZWwuYml0U2l6ZXMpCi0gICAgICAgICAgICAgICAgJiYgdGhpcy5zY2FubGluZVN0cmlkZSA9PSBtb2RlbC5zY2FubGluZVN0cmlkZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU2FtcGxlTW9kZWwgY3JlYXRlU3Vic2V0U2FtcGxlTW9kZWwoaW50IGJhbmRzW10pIHsKLSAgICAgICAgaWYgKGJhbmRzLmxlbmd0aCA+IHRoaXMubnVtQmFuZHMpIHsKLSAgICAgICAgICAgIC8vIGF3dC42ND1UaGUgbnVtYmVyIG9mIHRoZSBiYW5kcyBpbiB0aGUgc3Vic2V0IGlzIGdyZWF0ZXIgdGhhbiB0aGUKLSAgICAgICAgICAgIC8vIG51bWJlciBvZiBiYW5kcyBpbiB0aGUgc2FtcGxlIG1vZGVsCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjY0IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpbnQgbWFza3NbXSA9IG5ldyBpbnRbYmFuZHMubGVuZ3RoXTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBiYW5kcy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgbWFza3NbaV0gPSB0aGlzLmJpdE1hc2tzW2JhbmRzW2ldXTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbmV3IFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwodGhpcy5kYXRhVHlwZSwgdGhpcy53aWR0aCwgdGhpcy5oZWlnaHQsCi0gICAgICAgICAgICAgICAgdGhpcy5zY2FubGluZVN0cmlkZSwgbWFza3MpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTYW1wbGVNb2RlbCBjcmVhdGVDb21wYXRpYmxlU2FtcGxlTW9kZWwoaW50IHcsIGludCBoKSB7Ci0gICAgICAgIHJldHVybiBuZXcgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCh0aGlzLmRhdGFUeXBlLCB3LCBoLCB0aGlzLmJpdE1hc2tzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50W10gZ2V0UGl4ZWwoaW50IHgsIGludCB5LCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoeCA8IDAgfHwgeSA8IDAgfHwgeCA+PSB0aGlzLndpZHRoIHx8IHkgPj0gdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGludCBwaXhlbFtdOwotICAgICAgICBpZiAoaUFycmF5ID09IG51bGwpIHsKLSAgICAgICAgICAgIHBpeGVsID0gbmV3IGludFt0aGlzLm51bUJhbmRzXTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHBpeGVsID0gaUFycmF5OwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCB0aGlzLm51bUJhbmRzOyBpKyspIHsKLSAgICAgICAgICAgIHBpeGVsW2ldID0gZ2V0U2FtcGxlKHgsIHksIGksIGRhdGEpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHBpeGVsOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVsKGludCB4LCBpbnQgeSwgaW50IGlBcnJheVtdLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHRoaXMubnVtQmFuZHM7IGkrKykgewotICAgICAgICAgICAgc2V0U2FtcGxlKHgsIHksIGksIGlBcnJheVtpXSwgZGF0YSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldFNhbXBsZShpbnQgeCwgaW50IHksIGludCBiLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpbnQgc2FtcGxlID0gZGF0YS5nZXRFbGVtKHkgKiBzY2FubGluZVN0cmlkZSArIHgpOwotICAgICAgICByZXR1cm4gKChzYW1wbGUgJiB0aGlzLmJpdE1hc2tzW2JdKSA+Pj4gdGhpcy5iaXRPZmZzZXRzW2JdKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50W10gZ2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoKHggPCAwKSB8fCAoeSA8IDApIHx8ICgobG9uZyl4ICsgKGxvbmcpdyA+IHRoaXMud2lkdGgpCi0gICAgICAgICAgICAgICAgfHwgKChsb25nKXkgKyAobG9uZyloID4gdGhpcy5oZWlnaHQpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGludCBwaXhlbHNbXTsKLQotICAgICAgICBpZiAoaUFycmF5ID09IG51bGwpIHsKLSAgICAgICAgICAgIHBpeGVscyA9IG5ldyBpbnRbdyAqIGggKiB0aGlzLm51bUJhbmRzXTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHBpeGVscyA9IGlBcnJheTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBpZHggPSAwOwotCi0gICAgICAgIGZvciAoaW50IGkgPSB5OyBpIDwgeSArIGg7IGkrKykgewotICAgICAgICAgICAgZm9yIChpbnQgaiA9IHg7IGogPCB4ICsgdzsgaisrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgbiA9IDA7IG4gPCB0aGlzLm51bUJhbmRzOyBuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgcGl4ZWxzW2lkeCsrXSA9IGdldFNhbXBsZShqLCBpLCBuLCBkYXRhKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHBpeGVsczsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBpQXJyYXlbXSwgRGF0YUJ1ZmZlciBkYXRhKSB7Ci0gICAgICAgIGlmICgoeCA8IDApIHx8ICh5IDwgMCkgfHwgKChsb25nKXggKyAobG9uZyl3ID4gdGhpcy53aWR0aCkKLSAgICAgICAgICAgICAgICB8fCAoKGxvbmcpeSArIChsb25nKWggPiB0aGlzLmhlaWdodCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGlkeCA9IDA7Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBuID0gMDsgbiA8IHRoaXMubnVtQmFuZHM7IG4rKykgewotICAgICAgICAgICAgICAgICAgICBzZXRTYW1wbGUoaiwgaSwgbiwgaUFycmF5W2lkeCsrXSwgZGF0YSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0U2FtcGxlKGludCB4LCBpbnQgeSwgaW50IGIsIGludCBzLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKHggPCAwIHx8IHkgPCAwIHx8IHggPj0gdGhpcy53aWR0aCB8fCB5ID49IHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpbnQgdG1wID0gZGF0YS5nZXRFbGVtKHkgKiBzY2FubGluZVN0cmlkZSArIHgpOwotICAgICAgICB0bXAgJj0gfnRoaXMuYml0TWFza3NbYl07Ci0gICAgICAgIHRtcCB8PSAocyA8PCB0aGlzLmJpdE9mZnNldHNbYl0pICYgdGhpcy5iaXRNYXNrc1tiXTsKLSAgICAgICAgZGF0YS5zZXRFbGVtKHkgKiBzY2FubGluZVN0cmlkZSArIHgsIHRtcCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBpbnQgaUFycmF5W10sIERhdGFCdWZmZXIgZGF0YSkgewotICAgICAgICBpZiAoKHggPCAwKSB8fCAoeSA8IDApIHx8ICgobG9uZyl4ICsgKGxvbmcpdyA+IHRoaXMud2lkdGgpCi0gICAgICAgICAgICAgICAgfHwgKChsb25nKXkgKyAobG9uZyloID4gdGhpcy5oZWlnaHQpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNjM9Q29vcmRpbmF0ZXMgYXJlIG5vdCBpbiBib3VuZHMKLSAgICAgICAgICAgIHRocm93IG5ldyBBcnJheUluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNjMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGludCBzYW1wbGVzW107Ci0gICAgICAgIGludCBpZHggPSAwOwotCi0gICAgICAgIGlmIChpQXJyYXkgPT0gbnVsbCkgewotICAgICAgICAgICAgc2FtcGxlcyA9IG5ldyBpbnRbdyAqIGhdOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgc2FtcGxlcyA9IGlBcnJheTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSB5OyBpIDwgeSArIGg7IGkrKykgewotICAgICAgICAgICAgZm9yIChpbnQgaiA9IHg7IGogPCB4ICsgdzsgaisrKSB7Ci0gICAgICAgICAgICAgICAgc2FtcGxlc1tpZHgrK10gPSBnZXRTYW1wbGUoaiwgaSwgYiwgZGF0YSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gc2FtcGxlczsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGVzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnQgYiwgaW50IGlBcnJheVtdLCBEYXRhQnVmZmVyIGRhdGEpIHsKLSAgICAgICAgaWYgKCh4IDwgMCkgfHwgKHkgPCAwKSB8fCAoKGxvbmcpeCArIChsb25nKXcgPiB0aGlzLndpZHRoKQotICAgICAgICAgICAgICAgIHx8ICgobG9uZyl5ICsgKGxvbmcpaCA+IHRoaXMuaGVpZ2h0KSkgewotICAgICAgICAgICAgLy8gYXd0LjYzPUNvb3JkaW5hdGVzIGFyZSBub3QgaW4gYm91bmRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgQXJyYXlJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpbnQgaWR4ID0gMDsKLSAgICAgICAgZm9yIChpbnQgaSA9IHk7IGkgPCB5ICsgaDsgaSsrKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBqID0geDsgaiA8IHggKyB3OyBqKyspIHsKLSAgICAgICAgICAgICAgICBzZXRTYW1wbGUoeCArIGosIHkgKyBpLCBiLCBpQXJyYXlbaWR4KytdLCBkYXRhKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBEYXRhQnVmZmVyIGNyZWF0ZURhdGFCdWZmZXIoKSB7Ci0gICAgICAgIERhdGFCdWZmZXIgZGF0YSA9IG51bGw7Ci0gICAgICAgIGludCBzaXplID0gKHRoaXMuaGVpZ2h0IC0gMSkgKiBzY2FubGluZVN0cmlkZSArIHdpZHRoOwotCi0gICAgICAgIHN3aXRjaCAodGhpcy5kYXRhVHlwZSkgewotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfQllURToKLSAgICAgICAgICAgICAgICBkYXRhID0gbmV3IERhdGFCdWZmZXJCeXRlKHNpemUpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIGRhdGEgPSBuZXcgRGF0YUJ1ZmZlclVTaG9ydChzaXplKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBkYXRhID0gbmV3IERhdGFCdWZmZXJJbnQoc2l6ZSk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGRhdGE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgb2Zmc2V0IG9mIHRoZSBzcGVjaWZpZWQgcGl4ZWwgaW4gdGhlIGRhdGEgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIHNwZWNpZmllZCBwaXhlbC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsLgotICAgICAqIEByZXR1cm4gdGhlIG9mZnNldCBvZiB0aGUgc3BlY2lmaWVkIHBpeGVsLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0T2Zmc2V0KGludCB4LCBpbnQgeSkgewotICAgICAgICByZXR1cm4gKHkgKiBzY2FubGluZVN0cmlkZSArIHgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0U2FtcGxlU2l6ZShpbnQgYmFuZCkgewotICAgICAgICByZXR1cm4gYml0U2l6ZXNbYmFuZF07Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldFNhbXBsZVNpemUoKSB7Ci0gICAgICAgIHJldHVybiBiaXRTaXplcy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgdGhlIGJpdCBvZmZzZXRzIG9mIHRoZSBkYXRhIGFycmF5IGVsZW1lbnRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgb2YgdGhlIGJpdCBvZmZzZXRzLgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXRCaXRPZmZzZXRzKCkgewotICAgICAgICByZXR1cm4gYml0T2Zmc2V0cy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgdGhlIGJpdCBtYXNrcyBmb3IgYWxsIGJhbmRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgb2YgdGhlIGJpdCBtYXNrcyBmb3IgYWxsIGJhbmRzLgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXRCaXRNYXNrcygpIHsKLSAgICAgICAgcmV0dXJuIGJpdE1hc2tzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIGhhc2ggY29kZSBvZiB0aGlzIE11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBjbGFzcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBoYXNoIGNvZGUgb2YgdGhpcyBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgY2xhc3MuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsKLSAgICAgICAgaW50IGhhc2ggPSAwOwotICAgICAgICBpbnQgdG1wID0gMDsKLQotICAgICAgICBoYXNoID0gd2lkdGg7Ci0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICBoYXNoIHw9IHRtcDsKLSAgICAgICAgaGFzaCBePSBoZWlnaHQ7Ci0gICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICBoYXNoIHw9IHRtcDsKLSAgICAgICAgaGFzaCBePSBudW1CYW5kczsKLSAgICAgICAgdG1wID0gaGFzaCA+Pj4gMjQ7Ci0gICAgICAgIGhhc2ggPDw9IDg7Ci0gICAgICAgIGhhc2ggfD0gdG1wOwotICAgICAgICBoYXNoIF49IGRhdGFUeXBlOwotICAgICAgICB0bXAgPSBoYXNoID4+PiAyNDsKLSAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgaGFzaCB8PSB0bXA7Ci0gICAgICAgIGZvciAoaW50IGVsZW1lbnQgOiBiaXRNYXNrcykgewotICAgICAgICAgICAgaGFzaCBePSBlbGVtZW50OwotICAgICAgICAgICAgdG1wID0gaGFzaCA+Pj4gMjQ7Ci0gICAgICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICAgICAgaGFzaCB8PSB0bXA7Ci0gICAgICAgIH0KLSAgICAgICAgZm9yIChpbnQgZWxlbWVudCA6IGJpdE9mZnNldHMpIHsKLSAgICAgICAgICAgIGhhc2ggXj0gZWxlbWVudDsKLSAgICAgICAgICAgIHRtcCA9IGhhc2ggPj4+IDI0OwotICAgICAgICAgICAgaGFzaCA8PD0gODsKLSAgICAgICAgICAgIGhhc2ggfD0gdG1wOwotICAgICAgICB9Ci0gICAgICAgIGZvciAoaW50IGVsZW1lbnQgOiBiaXRTaXplcykgewotICAgICAgICAgICAgaGFzaCBePSBlbGVtZW50OwotICAgICAgICAgICAgdG1wID0gaGFzaCA+Pj4gMjQ7Ci0gICAgICAgICAgICBoYXNoIDw8PSA4OwotICAgICAgICAgICAgaGFzaCB8PSB0bXA7Ci0gICAgICAgIH0KLSAgICAgICAgaGFzaCBePSBzY2FubGluZVN0cmlkZTsKLSAgICAgICAgcmV0dXJuIGhhc2g7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc2NhbmxpbmUgc3RyaWRlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNjYW5saW5lIHN0cmlkZQotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0U2NhbmxpbmVTdHJpZGUoKSB7Ci0gICAgICAgIHJldHVybiB0aGlzLnNjYW5saW5lU3RyaWRlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0TnVtRGF0YUVsZW1lbnRzKCkgewotICAgICAgICByZXR1cm4gMTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9UaWxlT2JzZXJ2ZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9UaWxlT2JzZXJ2ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggN2RkOTdlMi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvVGlsZU9ic2VydmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotLyoqCi0gKiBBbiBhc3luY2hyb25vdXMgdXBkYXRlIGludGVyZmFjZSBmb3IgcmVjZWl2aW5nIG5vdGlmaWNhdGlvbnMgYWJvdXQgdGlsZQotICogaW5mb3JtYXRpb24gd2hlbiB0aWxlcyBvZiBhIFdyaXRhYmxlUmVuZGVyZWRJbWFnZSBiZWNvbWUgbW9kaWZpYWJsZSBvcgotICogdW5tb2RpZmlhYmxlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBUaWxlT2JzZXJ2ZXIgewotCi0gICAgLyoqCi0gICAgICogVGhpcyBtZXRob2QgaXMgY2FsbGVkIHdoZW4gaW5mb3JtYXRpb24gYWJvdXQgYSB0aWxlIHVwZGF0ZSBpcyBhdmFpbGFibGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gdGlsZVgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGluZGV4IG9mIHRoZSB0aWxlLgotICAgICAqIEBwYXJhbSB0aWxlWQotICAgICAqICAgICAgICAgICAgdGhlIFkgaW5kZXggb2YgdGhlIHRpbGUuCi0gICAgICogQHBhcmFtIHdpbGxCZVdyaXRhYmxlCi0gICAgICogICAgICAgICAgICBwYXJhbWV0ZXIgd2hpY2ggaW5kaWNhdGVzIHdoZXRoZXIgdGhlIHRpbGUgd2lsbCBiZSBncmFiYmVkIGZvcgotICAgICAqICAgICAgICAgICAgd3JpdGluZyBvciBiZSByZWxlYXNlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB0aWxlVXBkYXRlKFdyaXRhYmxlUmVuZGVyZWRJbWFnZSBzb3VyY2UsIGludCB0aWxlWCwgaW50IHRpbGVZLAotICAgICAgICAgICAgYm9vbGVhbiB3aWxsQmVXcml0YWJsZSk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9Wb2xhdGlsZUltYWdlLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvVm9sYXRpbGVJbWFnZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmMjRlODY2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9Wb2xhdGlsZUltYWdlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNDYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBBbGV4ZXkgQS4gUGV0cmVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlOwotCi1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3M7Ci1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3MyRDsKLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljc0NvbmZpZ3VyYXRpb247Ci1pbXBvcnQgamF2YS5hd3QuSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuSW1hZ2VDYXBhYmlsaXRpZXM7Ci1pbXBvcnQgamF2YS5hd3QuVHJhbnNwYXJlbmN5OwotCi0vKioKLSAqIFRoZSBWb2xhdGlsZUltYWdlIGFic3RyYWN0IGNsYXNzIHJlcHJlc2VudHMgYW4gaW1hZ2Ugd2hpY2ggY2FuIGxvc2UgaXRzCi0gKiBjb250ZW50cyBhdCBhbnkgcG9pbnQuIFZvbGF0aWxlSW1hZ2Ugb2JqZWN0cyBhcmUgZGV2aWNlIHNwZWNpZmljLiBUaGlzIGNsYXNzCi0gKiBwcm92aWRlcyBtZXRob2RzIGZvciBjaGVja2luZyBpZiBvcGVyYXRpb24gb2YgdGhpcyBpbWFnZSBhcmUgY29tcGF0aWJsZSBmb3IKLSAqIHRoZSBHcmFwaGljc0NvbmZpZ3VyYXRpb24uCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgVm9sYXRpbGVJbWFnZSBleHRlbmRzIEltYWdlCi0vLyBWb2xhdGlsZSBpbWFnZSBpbXBsZW1lbnRzIFRyYW5zcGFyZW5jeSBzaW5jZSAxLjUKLSAgICAgICAgaW1wbGVtZW50cyBUcmFuc3BhcmVuY3kgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IElNQUdFX0lOQ09NUEFUSUJMRSBpbmRpY2F0ZXMgdGhhdCB0aGlzIFZvbGF0aWxlSW1hZ2UgaXMgbm90Ci0gICAgICogYXBwbGljYWJsZSBmb3IgdGhlIEdyYXBoaWNzQ29uZmlndXJhdGlvbiBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSU1BR0VfSU5DT01QQVRJQkxFID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBJTUFHRV9PSyBpbmRpY2F0ZXMgdGhhdCBWb2xhdGlsZUltYWdlIGlzIHJlYWR5IGZvciB1c2luZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJTUFHRV9PSyA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgSU1BR0VfUkVTVE9SRUQgaW5kaWNhdGVzIHRoYXQgVm9sYXRpbGVJbWFnZSB3aWxsIGJlIHJlYWR5IHRvCi0gICAgICogdXNlIGFmdGVyIHJlc3RvcmluZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJTUFHRV9SRVNUT1JFRCA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdHJhbnNwYXJlbmN5IHZhbHVlIG9mIHRoaXMgaW1hZ2UuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCB0cmFuc3BhcmVuY3kgPSBPUEFRVUU7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgVm9sYXRpbGVJbWFnZSBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIFZvbGF0aWxlSW1hZ2UoKSB7Ci0gICAgICAgIHN1cGVyKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHJlbmRlcmluZyBkYXRhIGlzIGxvc3QgZHVyaW5nIHZhbGlkYXRpbmcuIFRoaXMgbWV0aG9kCi0gICAgICogc2hvdWxkIGJlIGNhbGxlZCBhZnRlciByZW5kZXJpbmcgb3BlcmF0aW9uIG9mIGltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgY29udGVudHMgbG9zdCBkdXJpbmcgdmFsaWRhdGluZywgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotCi0gICAgcHVibGljIGFic3RyYWN0IGJvb2xlYW4gY29udGVudHNMb3N0KCk7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgR3JhcGhpY3MyRCB1c2VkIHRvIGRyYXcgaW4gdGhpcyBWb2xhdGlsZUltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEdyYXBoaWNzMkQgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBHcmFwaGljczJEIGNyZWF0ZUdyYXBoaWNzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBJbWFnZUNhcGFiaWxpdGllcyBvZiB0aGlzIFZvbGF0aWxlSW1hZ2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VDYXBhYmlsaXRpZXMgb2YgdGhpcyBWb2xhdGlsZUltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJbWFnZUNhcGFiaWxpdGllcyBnZXRDYXBhYmlsaXRpZXMoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGhlaWdodCBvZiB0aGlzIFZvbGF0aWxlSW1hZ2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaGVpZ2h0IG9mIHRoaXMgVm9sYXRpbGVJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgaW50IGdldEhlaWdodCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyBhIEJ1ZmZlcmVkSW1hZ2UgcmVwcmVzZW50YXRpb24gb2YgY3VycmVudCBWb2xhdGlsZUltYWdlIHRoYXQgd29uJ3QKLSAgICAgKiBiZSBhZmZlY3RlZCBieSBhbnkgY2hhbmdlcyB0byB0aGlzIFZvbGF0aWxlSW1hZ2UuCi0gICAgICogCi0gICAgICogQHJldHVybiBhIEJ1ZmZlcmVkSW1hZ2UgcmVwcmVzZW50YXRpb24gb2YgY3VycmVudCBWb2xhdGlsZUltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBCdWZmZXJlZEltYWdlIGdldFNuYXBzaG90KCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB3aWR0aCBvZiB0aGlzIFZvbGF0aWxlSW1hZ2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgd2lkdGggb2YgdGhpcyBWb2xhdGlsZUltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0V2lkdGgoKTsKLQotICAgIC8qKgotICAgICAqIFZhbGlkYXRlcyB0aGUgZHJhd2luZyBzdXJmYWNlIG9mIHRoZSBpbWFnZSBpZiB0aGUgc3VyZmFjZSBoYWQgYmVlbiBsb3N0Ci0gICAgICogYW5kIGlmIHRoZSBzcGVjaWZpZWQgR3JhcGhpY3NDb25maWd1cmF0aW9uIG9iamVjdCBpcyBhcHBsaWNhYmxlIHRvIHRoaXMKLSAgICAgKiBpbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2MKLSAgICAgKiAgICAgICAgICAgIHRoZSBHcmFwaGljc0NvbmZpZ3VyYXRpb24gb2JqZWN0LgotICAgICAqIEByZXR1cm4gb25lIG9mIHRoZSBpbWFnZSBzdGF0dXMgY29uc3RhbnRzOiBJTUFHRV9PSywgSU1BR0VfUkVTVE9SRUQgb3IKLSAgICAgKiAgICAgICAgIElNQUdFX0lOQ09NUEFUSUJMRS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgaW50IHZhbGlkYXRlKEdyYXBoaWNzQ29uZmlndXJhdGlvbiBnYyk7Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmbHVzaCgpIHsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR3JhcGhpY3MgZ2V0R3JhcGhpY3MoKSB7Ci0gICAgICAgIHJldHVybiBjcmVhdGVHcmFwaGljcygpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJbWFnZVByb2R1Y2VyIGdldFNvdXJjZSgpIHsKLSAgICAgICAgcmV0dXJuIGdldFNuYXBzaG90KCkuZ2V0U291cmNlKCk7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRUcmFuc3BhcmVuY3koKSB7Ci0gICAgICAgIHJldHVybiB0cmFuc3BhcmVuY3k7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL1dyaXRhYmxlUmFzdGVyLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvV3JpdGFibGVSYXN0ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNTEzNjZlZS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvV3JpdGFibGVSYXN0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDU5MiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBqYXZhLmF3dC5pbWFnZTsKLQotaW1wb3J0IGphdmEuYXd0LlBvaW50OwotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFRoZSBXcml0YWJsZVJhc3RlciBjbGFzcyBwcm92aWRlcyBmdW5jdGlvbmFsaXR5IGZvciB3cml0aW5nIHNhbXBsZXMgYW5kIHBpeGVsCi0gKiBjYXBhYmlsaXRpZXMgdG8gdGhlIFJhc3Rlci4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBXcml0YWJsZVJhc3RlciBleHRlbmRzIFJhc3RlciB7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgV3JpdGFibGVSYXN0ZXIgb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCBTYW1wbGVNb2RlbCwKLSAgICAgKiBEYXRhQnVmZmVyLCByZWN0YW5ndWxhciByZWdpb24gYW5kIHBhcmVudCBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc2FtcGxlTW9kZWwKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwuCi0gICAgICogQHBhcmFtIGRhdGFCdWZmZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgRGF0YUJ1ZmZlci4KLSAgICAgKiBAcGFyYW0gYVJlZ2lvbgotICAgICAqICAgICAgICAgICAgdGhlIHJlY3Rhbmd1bGFyIHJlZ2lvbiB3aGljaCBkZWZpbmVzIHRoZSBuZXcgaW1hZ2UgYm91bmRzLgotICAgICAqIEBwYXJhbSBzYW1wbGVNb2RlbFRyYW5zbGF0ZQotICAgICAqICAgICAgICAgICAgdGhpcyBwb2ludCBkZWZpbmVzIHRoZSB0cmFuc2xhdGlvbiBwb2ludCBmcm9tIHRoZSBTYW1wbGVNb2RlbAotICAgICAqICAgICAgICAgICAgdG8gdGhlIG5ldyBXcml0YWJsZVJhc3RlciBjb29yZGluYXRlcy4KLSAgICAgKiBAcGFyYW0gcGFyZW50Ci0gICAgICogICAgICAgICAgICB0aGUgcGFyZW50IG9mIHRoaXMgV3JpdGFibGVSYXN0ZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFdyaXRhYmxlUmFzdGVyKFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsLCBEYXRhQnVmZmVyIGRhdGFCdWZmZXIsIFJlY3RhbmdsZSBhUmVnaW9uLAotICAgICAgICAgICAgUG9pbnQgc2FtcGxlTW9kZWxUcmFuc2xhdGUsIFdyaXRhYmxlUmFzdGVyIHBhcmVudCkgewotICAgICAgICBzdXBlcihzYW1wbGVNb2RlbCwgZGF0YUJ1ZmZlciwgYVJlZ2lvbiwgc2FtcGxlTW9kZWxUcmFuc2xhdGUsIHBhcmVudCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IFdyaXRhYmxlUmFzdGVyIG9iamVjdCB3aXRoIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwKLSAgICAgKiB3aGljaCBkZWZpbmVzIGEgbGF5b3V0IG9mIHRoaXMgV3JpdGFibGVSYXN0ZXIgYW5kIERhdGFCdWZmZXIgb2JqZWN0cwotICAgICAqIHdoaWNoIGRlZmluZXMgdGhlIGltYWdlIGRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNhbXBsZU1vZGVsCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIFNhbXBsZU1vZGVsLgotICAgICAqIEBwYXJhbSBkYXRhQnVmZmVyCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIERhdGFCdWZmZXIuCi0gICAgICogQHBhcmFtIG9yaWdpbgotICAgICAqICAgICAgICAgICAgdGhlIHBvaW50IG9mIG9yaWdpbi4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgV3JpdGFibGVSYXN0ZXIoU2FtcGxlTW9kZWwgc2FtcGxlTW9kZWwsIERhdGFCdWZmZXIgZGF0YUJ1ZmZlciwgUG9pbnQgb3JpZ2luKSB7Ci0gICAgICAgIHRoaXMoc2FtcGxlTW9kZWwsIGRhdGFCdWZmZXIsIG5ldyBSZWN0YW5nbGUob3JpZ2luLngsIG9yaWdpbi55LCBzYW1wbGVNb2RlbC53aWR0aCwKLSAgICAgICAgICAgICAgICBzYW1wbGVNb2RlbC5oZWlnaHQpLCBvcmlnaW4sIG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBXcml0YWJsZVJhc3RlciB3aXRoIHRoZSBzcGVjaWZpZWQgU2FtcGxlTW9kZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNhbXBsZU1vZGVsCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIFNhbXBsZU1vZGVsLgotICAgICAqIEBwYXJhbSBvcmlnaW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBvcmlnaW4uCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFdyaXRhYmxlUmFzdGVyKFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsLCBQb2ludCBvcmlnaW4pIHsKLSAgICAgICAgdGhpcyhzYW1wbGVNb2RlbCwgc2FtcGxlTW9kZWwuY3JlYXRlRGF0YUJ1ZmZlcigpLCBuZXcgUmVjdGFuZ2xlKG9yaWdpbi54LCBvcmlnaW4ueSwKLSAgICAgICAgICAgICAgICBzYW1wbGVNb2RlbC53aWR0aCwgc2FtcGxlTW9kZWwuaGVpZ2h0KSwgb3JpZ2luLCBudWxsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBkYXRhIGZvciBhIHNpbmdsZSBwaXhlbCBmcm9tIGFuIGlucHV0IE9iamVjdCB3aGljaCByZXByZXNlbnRzIGFuCi0gICAgICogYXJyYXkgb2YgcHJpbWl0aXZlIHR5cGVzOiBEYXRhQnVmZmVyLlRZUEVfQllURSwgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCwKLSAgICAgKiBEYXRhQnVmZmVyLlRZUEVfSU5ULCBEYXRhQnVmZmVyLlRZUEVfU0hPUlQsIERhdGFCdWZmZXIuVFlQRV9GTE9BVCwgb3IKLSAgICAgKiBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBwaXhlbC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIGluRGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGlucHV0IGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RGF0YUVsZW1lbnRzKGludCB4LCBpbnQgeSwgT2JqZWN0IGluRGF0YSkgewotICAgICAgICBzYW1wbGVNb2RlbC5zZXREYXRhRWxlbWVudHMoeCAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgeSAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWSwgaW5EYXRhLAotICAgICAgICAgICAgICAgIGRhdGFCdWZmZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGRhdGEgZWxlbWVudHMgd2hpY2ggcmVwcmVzZW50IHBpeGVsIGRhdGEgdG8gdGhlIHNwZWNpZmllZAotICAgICAqIHJlY3RhbmdsZSBhcmVhIGFzIGEgcHJpbWl0aXZlIGFycmF5LiBUaGUgZm9sbG93aW5nIGltYWdlIGRhdGEgdHlwZXMgYXJlCi0gICAgICogc3VwcG9ydGVkOiBEYXRhQnVmZmVyLlRZUEVfQllURSwgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCwKLSAgICAgKiBEYXRhQnVmZmVyLlRZUEVfSU5ULCBEYXRhQnVmZmVyLlRZUEVfU0hPUlQsIERhdGFCdWZmZXIuVFlQRV9GTE9BVCwgb3IKLSAgICAgKiBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSByZWN0YW5nbGUgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIHJlY3RhbmdsZSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGluRGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHByaW1pdGl2ZSB0eXBlIGRhdGEgdG8gYmUgc2V0IHRvIHRoZSBzcGVjaWZpZWQKLSAgICAgKiAgICAgICAgICAgIGFyZWEuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RGF0YUVsZW1lbnRzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBPYmplY3QgaW5EYXRhKSB7Ci0gICAgICAgIHNhbXBsZU1vZGVsLnNldERhdGFFbGVtZW50cyh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCB3LCBoLAotICAgICAgICAgICAgICAgIGluRGF0YSwgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgY2hpbGQgb2YgdGhpcyBXcml0YWJsZVJhc3RlciBieSBzaGFyaW5nIHRoZSBzcGVjaWZpZWQKLSAgICAgKiByZWN0YW5ndWxhciBhcmVhIGluIHRoaXMgV3JpdGFibGVSYXN0ZXIuIFRoZSBwYXJlbnRYLCBwYXJlbnRZLCB3aWR0aCBhbmQKLSAgICAgKiBoZWlnaHQgcGFyYW1ldGVycyBzcGVjaWZ5IHJlY3Rhbmd1bGFyIGFyZWEgdG8gYmUgc2hhcmVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXJlbnRYCi0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgc2hhcmVkCi0gICAgICogICAgICAgICAgICByZWN0YW5nbGUgd2l0aCByZXNwZWN0IHRvIHRoaXMgV3JpdGFibGVSYXN0ZXInIGNvb3JkaW5hdGVzLgotICAgICAqIEBwYXJhbSBwYXJlbnRZCi0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUgc2hhcmVkCi0gICAgICogICAgICAgICAgICByZWN0YW5nbGUgd2l0aCByZXNwZWN0IHRvIHRoaXMgV3JpdGFibGVSYXN0ZXInIGNvb3JkaW5hdGVzLgotICAgICAqIEBwYXJhbSB3Ci0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdGhlIGNoaWxkIGFyZWEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGNoaWxkIGFyZWEuCi0gICAgICogQHBhcmFtIGNoaWxkTWluWAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiBjaGlsZCBhcmVhIG1hcHBlZCB0byB0aGUgcGFyZW50WAotICAgICAqICAgICAgICAgICAgY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0gY2hpbGRNaW5ZCi0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIGNoaWxkIGFyZWEgbWFwcGVkIHRvIHRoZSBwYXJlbnRZCi0gICAgICogICAgICAgICAgICBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSBiYW5kTGlzdAotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGJhbmQgaW5kaWNlcy4KLSAgICAgKiBAcmV0dXJuIHRoZSBjaGlsZCBXcml0YWJsZVJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlV3JpdGFibGVDaGlsZChpbnQgcGFyZW50WCwgaW50IHBhcmVudFksIGludCB3LCBpbnQgaCwKLSAgICAgICAgICAgIGludCBjaGlsZE1pblgsIGludCBjaGlsZE1pblksIGludCBiYW5kTGlzdFtdKSB7Ci0gICAgICAgIGlmICh3IDw9IDAgfHwgaCA8PSAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjQ0PVdpZHRoIG9yIEhlaWdodCBvZiBjaGlsZCBSYXN0ZXIgaXMgbGVzcyB0aGFuIG9yIGVxdWFsIHRvCi0gICAgICAgICAgICAvLyB6ZXJvCi0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI0NCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHBhcmVudFggPCB0aGlzLm1pblggfHwgcGFyZW50WCArIHcgPiB0aGlzLm1pblggKyB0aGlzLndpZHRoKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjQ1PXBhcmVudFggZGlzcG9zZXMgb3V0c2lkZSBSYXN0ZXIKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjQ1IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAocGFyZW50WSA8IHRoaXMubWluWSB8fCBwYXJlbnRZICsgaCA+IHRoaXMubWluWSArIHRoaXMuaGVpZ2h0KSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjQ2PXBhcmVudFkgZGlzcG9zZXMgb3V0c2lkZSBSYXN0ZXIKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjQ2IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoKGxvbmcpcGFyZW50WCArIHcgPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI0Nz1wYXJlbnRYICsgdyByZXN1bHRzIGluIGludGVnZXIgb3ZlcmZsb3cKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjQ3IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoKGxvbmcpcGFyZW50WSArIGggPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI0OD1wYXJlbnRZICsgaCByZXN1bHRzIGluIGludGVnZXIgb3ZlcmZsb3cKLSAgICAgICAgICAgIHRocm93IG5ldyBSYXN0ZXJGb3JtYXRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjQ4IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoKGxvbmcpY2hpbGRNaW5YICsgdyA+IEludGVnZXIuTUFYX1ZBTFVFKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMjQ5PWNoaWxkTWluWCArIHcgcmVzdWx0cyBpbiBpbnRlZ2VyIG92ZXJmbG93Ci0gICAgICAgICAgICB0aHJvdyBuZXcgUmFzdGVyRm9ybWF0RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjI0OSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChsb25nKWNoaWxkTWluWSArIGggPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgLy8gYXd0LjI0QT1jaGlsZE1pblkgKyBoIHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotICAgICAgICAgICAgdGhyb3cgbmV3IFJhc3RlckZvcm1hdEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNEEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIFNhbXBsZU1vZGVsIGNoaWxkTW9kZWw7Ci0KLSAgICAgICAgaWYgKGJhbmRMaXN0ID09IG51bGwpIHsKLSAgICAgICAgICAgIGNoaWxkTW9kZWwgPSBzYW1wbGVNb2RlbDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNoaWxkTW9kZWwgPSBzYW1wbGVNb2RlbC5jcmVhdGVTdWJzZXRTYW1wbGVNb2RlbChiYW5kTGlzdCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgY2hpbGRUcmFuc2xhdGVYID0gY2hpbGRNaW5YIC0gcGFyZW50WDsKLSAgICAgICAgaW50IGNoaWxkVHJhbnNsYXRlWSA9IGNoaWxkTWluWSAtIHBhcmVudFk7Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBXcml0YWJsZVJhc3RlcihjaGlsZE1vZGVsLCBkYXRhQnVmZmVyLAotICAgICAgICAgICAgICAgIG5ldyBSZWN0YW5nbGUoY2hpbGRNaW5YLCBjaGlsZE1pblksIHcsIGgpLCBuZXcgUG9pbnQoY2hpbGRUcmFuc2xhdGVYCi0gICAgICAgICAgICAgICAgICAgICAgICArIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgY2hpbGRUcmFuc2xhdGVZICsgc2FtcGxlTW9kZWxUcmFuc2xhdGVZKSwgdGhpcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgdHJhbnNsYXRlZCBjaGlsZCBvZiB0aGlzIFdyaXRhYmxlUmFzdGVyLiBOZXcgV3JpdGFibGVSYXN0ZXIKLSAgICAgKiBvYmplY3QgaXMgYSByZWZlcmVuY2UgdG8gdGhlIHRoaXMgV3JpdGFibGVSYXN0ZXIgYW5kIHdpdGggZGlmZmVyZW50Ci0gICAgICogbG9jYXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIGNoaWxkTWluWAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgbmV3IFdyaXRhYmxlUmFzdGVyLgotICAgICAqIEBwYXJhbSBjaGlsZE1pblkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIG5ldyBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiBAcmV0dXJuIHRoZSBXcml0YWJsZVJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlV3JpdGFibGVUcmFuc2xhdGVkQ2hpbGQoaW50IGNoaWxkTWluWCwgaW50IGNoaWxkTWluWSkgewotICAgICAgICByZXR1cm4gY3JlYXRlV3JpdGFibGVDaGlsZChtaW5YLCBtaW5ZLCB3aWR0aCwgaGVpZ2h0LCBjaGlsZE1pblgsIGNoaWxkTWluWSwgbnVsbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcGFyZW50IFdyaXRhYmxlUmFzdGVyIGZvciB0aGlzIFdyaXRhYmxlUmFzdGVyIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBwYXJlbnQgV3JpdGFibGVSYXN0ZXIgZm9yIHRoaXMgV3JpdGFibGVSYXN0ZXIgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBnZXRXcml0YWJsZVBhcmVudCgpIHsKLSAgICAgICAgcmV0dXJuIChXcml0YWJsZVJhc3RlcilwYXJlbnQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBwaXhlbHMgZnJvbSB0aGUgc3BlY2lmaWVkIHNvdXJjZSBSYXN0ZXIgc3JjUmFzdGVyIHRvIHRoaXMKLSAgICAgKiBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3JjUmFzdGVyCi0gICAgICogICAgICAgICAgICB0aGUgc291cmNlIFJhc3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRSZWN0KFJhc3RlciBzcmNSYXN0ZXIpIHsKLSAgICAgICAgc2V0UmVjdCgwLCAwLCBzcmNSYXN0ZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgcGl4ZWxzIGZyb20gdGhlIHNwZWNpZmllZCBzb3VyY2UgUmFzdGVyIHNyY1Jhc3RlciB0byB0aGlzCi0gICAgICogV3JpdGFibGVSYXN0ZXIuIEVhY2ggcGl4ZWwgd2l0aCAoeCwgeSkgY29vcmRpbmF0ZXMgZnJvbSB0aGUgc291cmNlIFJhc3RlcgotICAgICAqIGlzIGNvcGllZCB0byBwaXhlbCB3aXRoICh4K2R4LCB5K2R5KSBjb29yZGluYXRlcyBpbiB0aGlzIFdyaXRhYmxlUmFzdGVyLgotICAgICAqIFRoZSBwaXhlbHMgd2l0aCAoeCtkeCwgeStkeSkgY29vcmRpbmF0ZXMgd2hpY2ggYXJlIG91dCB0aGUgYm91bmRzIG9mIHRoaXMKLSAgICAgKiByYXN0ZXIgYXJlIGlnbm9yZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGR4Ci0gICAgICogICAgICAgICAgICB0aGUgZGlzdGFuY2UgdGhlIHBpeGVsJ3MgWCBjb29yZGluYXRlIGluIHRoZSBzb3VyY2UgUmFzdGVyIGlzCi0gICAgICogICAgICAgICAgICB0cmFuc2xhdGVkIHdoZW4gd3JpdHRpZW4gdG8gdGhpcyBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiBAcGFyYW0gZHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBkaXN0YW5jZSB0aGUgcGl4ZWwncyBZIGNvb3JkaW5hdGUgaW4gdGhlIHNvdXJjZSBSYXN0ZXIgaXMKLSAgICAgKiAgICAgICAgICAgIHRyYW5zbGF0ZWQgd2hlbiB3cml0dGllbiB0byB0aGlzIFdyaXRhYmxlUmFzdGVyLgotICAgICAqIEBwYXJhbSBzcmNSYXN0ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFJlY3QoaW50IGR4LCBpbnQgZHksIFJhc3RlciBzcmNSYXN0ZXIpIHsKLSAgICAgICAgaW50IHcgPSBzcmNSYXN0ZXIuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IGggPSBzcmNSYXN0ZXIuZ2V0SGVpZ2h0KCk7Ci0KLSAgICAgICAgaW50IHNyY1ggPSBzcmNSYXN0ZXIuZ2V0TWluWCgpOwotICAgICAgICBpbnQgc3JjWSA9IHNyY1Jhc3Rlci5nZXRNaW5ZKCk7Ci0KLSAgICAgICAgaW50IGRzdFggPSBzcmNYICsgZHg7Ci0gICAgICAgIGludCBkc3RZID0gc3JjWSArIGR5OwotCi0gICAgICAgIGlmIChkc3RYIDwgdGhpcy5taW5YKSB7Ci0gICAgICAgICAgICBpbnQgbWluT2ZmWCA9IHRoaXMubWluWCAtIGRzdFg7Ci0gICAgICAgICAgICB3IC09IG1pbk9mZlg7Ci0gICAgICAgICAgICBkc3RYID0gdGhpcy5taW5YOwotICAgICAgICAgICAgc3JjWCArPSBtaW5PZmZYOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRzdFkgPCB0aGlzLm1pblkpIHsKLSAgICAgICAgICAgIGludCBtaW5PZmZZID0gdGhpcy5taW5ZIC0gZHN0WTsKLSAgICAgICAgICAgIGggLT0gbWluT2ZmWTsKLSAgICAgICAgICAgIGRzdFkgPSB0aGlzLm1pblk7Ci0gICAgICAgICAgICBzcmNZICs9IG1pbk9mZlk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZHN0WCArIHcgPiB0aGlzLm1pblggKyB0aGlzLndpZHRoKSB7Ci0gICAgICAgICAgICBpbnQgbWF4T2ZmWCA9IChkc3RYICsgdykgLSAodGhpcy5taW5YICsgdGhpcy53aWR0aCk7Ci0gICAgICAgICAgICB3IC09IG1heE9mZlg7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZHN0WSArIGggPiB0aGlzLm1pblkgKyB0aGlzLmhlaWdodCkgewotICAgICAgICAgICAgaW50IG1heE9mZlkgPSAoZHN0WSArIGgpIC0gKHRoaXMubWluWSArIHRoaXMuaGVpZ2h0KTsKLSAgICAgICAgICAgIGggLT0gbWF4T2ZmWTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICh3IDw9IDAgfHwgaCA8PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBzd2l0Y2ggKHNhbXBsZU1vZGVsLmdldERhdGFUeXBlKCkpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9TSE9SVDoKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0lOVDoKLSAgICAgICAgICAgICAgICBpbnQgaVBpeGVsc0xpbmVbXSA9IG51bGw7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBoOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgaVBpeGVsc0xpbmUgPSBzcmNSYXN0ZXIuZ2V0UGl4ZWxzKHNyY1gsIHNyY1kgKyBpLCB3LCAxLCBpUGl4ZWxzTGluZSk7Ci0gICAgICAgICAgICAgICAgICAgIHNldFBpeGVscyhkc3RYLCBkc3RZICsgaSwgdywgMSwgaVBpeGVsc0xpbmUpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfRkxPQVQ6Ci0gICAgICAgICAgICAgICAgZmxvYXQgZlBpeGVsc0xpbmVbXSA9IG51bGw7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBoOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgZlBpeGVsc0xpbmUgPSBzcmNSYXN0ZXIuZ2V0UGl4ZWxzKHNyY1gsIHNyY1kgKyBpLCB3LCAxLCBmUGl4ZWxzTGluZSk7Ci0gICAgICAgICAgICAgICAgICAgIHNldFBpeGVscyhkc3RYLCBkc3RZICsgaSwgdywgMSwgZlBpeGVsc0xpbmUpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLQotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfRE9VQkxFOgotICAgICAgICAgICAgICAgIGRvdWJsZSBkUGl4ZWxzTGluZVtdID0gbnVsbDsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBkUGl4ZWxzTGluZSA9IHNyY1Jhc3Rlci5nZXRQaXhlbHMoc3JjWCwgc3JjWSArIGksIHcsIDEsIGRQaXhlbHNMaW5lKTsKLSAgICAgICAgICAgICAgICAgICAgc2V0UGl4ZWxzKGRzdFgsIGRzdFkgKyBpLCB3LCAxLCBkUGl4ZWxzTGluZSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZGF0YSBmb3IgYSByZWN0YW5nbGUgb2YgcGl4ZWxzIGZyb20gYW4gaW5wdXQgUmFzdGVyIHRvIHRoaXMKLSAgICAgKiBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcG9pbnQgd2hlcmUgdGhlIGRhdGEgb2YgdGhlIGlucHV0Ci0gICAgICogICAgICAgICAgICBSYXN0ZXIgaXMgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcG9pbnQgd2hlcmUgdGhlIGRhdGEgb2YgdGhlIGlucHV0Ci0gICAgICogICAgICAgICAgICBSYXN0ZXIgaXMgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gaW5SYXN0ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnB1dCBSYXN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RGF0YUVsZW1lbnRzKGludCB4LCBpbnQgeSwgUmFzdGVyIGluUmFzdGVyKSB7Ci0gICAgICAgIGludCBkc3RYID0geCArIGluUmFzdGVyLmdldE1pblgoKTsKLSAgICAgICAgaW50IGRzdFkgPSB5ICsgaW5SYXN0ZXIuZ2V0TWluWSgpOwotCi0gICAgICAgIGludCB3ID0gaW5SYXN0ZXIuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IGggPSBpblJhc3Rlci5nZXRIZWlnaHQoKTsKLQotICAgICAgICBpZiAoZHN0WCA8IHRoaXMubWluWCB8fCBkc3RYICsgdyA+IHRoaXMubWluWCArIHRoaXMud2lkdGggfHwgZHN0WSA8IHRoaXMubWluWQotICAgICAgICAgICAgICAgIHx8IGRzdFkgKyBoID4gdGhpcy5taW5ZICsgdGhpcy5oZWlnaHQpIHsKLSAgICAgICAgICAgIC8vIGF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotICAgICAgICAgICAgdGhyb3cgbmV3IEFycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC42MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHNyY1ggPSBpblJhc3Rlci5nZXRNaW5YKCk7Ci0gICAgICAgIGludCBzcmNZID0gaW5SYXN0ZXIuZ2V0TWluWSgpOwotICAgICAgICBPYmplY3QgbGluZSA9IG51bGw7Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBoOyBpKyspIHsKLSAgICAgICAgICAgIGxpbmUgPSBpblJhc3Rlci5nZXREYXRhRWxlbWVudHMoc3JjWCwgc3JjWSArIGksIHcsIDEsIGxpbmUpOwotICAgICAgICAgICAgc2V0RGF0YUVsZW1lbnRzKGRzdFgsIGRzdFkgKyBpLCB3LCAxLCBsaW5lKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgYW4gaW50ZWdlciBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIHBpeGVsIGluIHRoaXMKLSAgICAgKiBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIHBpeGVsJ3MgWCBjb29yZGluYXRlLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwncyBZIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIGlBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGludGVnZXIgYXJyYXkgb2Ygc2FtcGxlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbChpbnQgeCwgaW50IHksIGludCBpQXJyYXlbXSkgewotICAgICAgICBzYW1wbGVNb2RlbC5zZXRQaXhlbCh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCBpQXJyYXksCi0gICAgICAgICAgICAgICAgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhIGZsb2F0IGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgcGl4ZWwgaW4gdGhpcwotICAgICAqIFdyaXRhYmxlUmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwncyBYIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbCdzIFkgY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0gZkFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZmxvYXQgYXJyYXkgb2Ygc2FtcGxlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbChpbnQgeCwgaW50IHksIGZsb2F0IGZBcnJheVtdKSB7Ci0gICAgICAgIHNhbXBsZU1vZGVsLnNldFBpeGVsKHggLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVgsIHkgLSBzYW1wbGVNb2RlbFRyYW5zbGF0ZVksIGZBcnJheSwKLSAgICAgICAgICAgICAgICBkYXRhQnVmZmVyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIGEgZG91YmxlIGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgcGl4ZWwgaW4gdGhpcwotICAgICAqIFdyaXRhYmxlUmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgcGl4ZWwncyBYIGNvb3JkaW5hdGUuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBwaXhlbCdzIFkgY29vcmRpbmF0ZS4KLSAgICAgKiBAcGFyYW0gZEFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZG91YmxlIGFycmF5IG9mIHNhbXBsZXMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWwoaW50IHgsIGludCB5LCBkb3VibGUgZEFycmF5W10pIHsKLSAgICAgICAgc2FtcGxlTW9kZWwuc2V0UGl4ZWwoeCAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgeSAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWSwgZEFycmF5LAotICAgICAgICAgICAgICAgIGRhdGFCdWZmZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgYSBpbnRlZ2VyIGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIgYXJlYSBvZgotICAgICAqIHBpeGVscyBpbiB0aGlzIFdyaXRhYmxlUmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0gaUFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgaW50ZWdlciBhcnJheSBvZiBzYW1wbGVzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGlBcnJheVtdKSB7Ci0gICAgICAgIHNhbXBsZU1vZGVsLnNldFBpeGVscyh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCB3LCBoLCBpQXJyYXksCi0gICAgICAgICAgICAgICAgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhIGZsb2F0IGFycmF5IG9mIHNhbXBsZXMgZm9yIHRoZSBzcGVjaWZpZWQgcmVjdGFuZ3VsYXIgYXJlYSBvZgotICAgICAqIHBpeGVscyBpbiB0aGlzIFdyaXRhYmxlUmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgcmVjdGFuZ3VsYXIgYXJlYS4KLSAgICAgKiBAcGFyYW0gZkFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZmxvYXQgYXJyYXkgb2Ygc2FtcGxlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbHMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGZsb2F0IGZBcnJheVtdKSB7Ci0gICAgICAgIHNhbXBsZU1vZGVsLnNldFBpeGVscyh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCB3LCBoLCBmQXJyYXksCi0gICAgICAgICAgICAgICAgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhIGRvdWJsZSBhcnJheSBvZiBzYW1wbGVzIGZvciB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyIGFyZWEgb2YKLSAgICAgKiBwaXhlbHMgaW4gdGhpcyBXcml0YWJsZVJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiByZWN0YW5ndWxhciBhcmVhLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiByZWN0YW5ndWxhciBhcmVhLgotICAgICAqIEBwYXJhbSBoCi0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHJlY3Rhbmd1bGFyIGFyZWEuCi0gICAgICogQHBhcmFtIGRBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGRvdWJsZSBhcnJheSBvZiBzYW1wbGVzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgZG91YmxlIGRBcnJheVtdKSB7Ci0gICAgICAgIHNhbXBsZU1vZGVsLnNldFBpeGVscyh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCB3LCBoLCBkQXJyYXksCi0gICAgICAgICAgICAgICAgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2FtcGxlcyBmb3IgdGhlIHNwZWNpZmllZCBiYW5kIGFuZCB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyCi0gICAgICogYXJlYSBvZiBwaXhlbHMgd2l0aCBhbiBpbnRlZ2VyIGFycmF5IG9mIHNhbXBsZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gaUFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgaW50ZWdlciBhcnJheSBvZiBzYW1wbGVzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBpbnQgaUFycmF5W10pIHsKLSAgICAgICAgc2FtcGxlTW9kZWwuc2V0U2FtcGxlcyh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCB3LCBoLCBiLAotICAgICAgICAgICAgICAgIGlBcnJheSwgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2FtcGxlcyBmb3IgdGhlIHNwZWNpZmllZCBiYW5kIGFuZCB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyCi0gICAgICogYXJlYSBvZiBwaXhlbHMgd2l0aCBhIGZsb2F0IGFycmF5IG9mIHNhbXBsZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2YgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSB5Ci0gICAgICogICAgICAgICAgICB0aGUgWSBjb29yZGluYXRlIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gdwotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gZkFycmF5Ci0gICAgICogICAgICAgICAgICB0aGUgZmxvYXQgYXJyYXkgb2Ygc2FtcGxlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGVzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnQgYiwgZmxvYXQgZkFycmF5W10pIHsKLSAgICAgICAgc2FtcGxlTW9kZWwuc2V0U2FtcGxlcyh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCB3LCBoLCBiLAotICAgICAgICAgICAgICAgIGZBcnJheSwgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2FtcGxlcyBmb3IgdGhlIHNwZWNpZmllZCBiYW5kIGFuZCB0aGUgc3BlY2lmaWVkIHJlY3Rhbmd1bGFyCi0gICAgICogYXJlYSBvZiBwaXhlbHMgd2l0aCBhIGRvdWJsZSBhcnJheSBvZiBzYW1wbGVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBhcmVhIG9mIHBpeGVscy4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgYXJlYSBvZiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGFyZWEgb2YgcGl4ZWxzLgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGJhbmQuCi0gICAgICogQHBhcmFtIGRBcnJheQotICAgICAqICAgICAgICAgICAgdGhlIGRvdWJsZSBhcnJheSBvZiBzYW1wbGVzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZXMoaW50IHgsIGludCB5LCBpbnQgdywgaW50IGgsIGludCBiLCBkb3VibGUgZEFycmF5W10pIHsKLSAgICAgICAgc2FtcGxlTW9kZWwuc2V0U2FtcGxlcyh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCB3LCBoLCBiLAotICAgICAgICAgICAgICAgIGRBcnJheSwgZGF0YUJ1ZmZlcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc2FtcGxlIGZvciB0aGUgc3BlY2lmaWVkIGJhbmQgYW5kIHRoZSBzcGVjaWZpZWQgcGl4ZWwgd2l0aCBhbgotICAgICAqIGludGVnZXIgc2FtcGxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB4Ci0gICAgICogICAgICAgICAgICB0aGUgWCBjb29yZGluYXRlIG9mIHRoZSBwaXhlbC4KLSAgICAgKiBAcGFyYW0geQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIHNhbXBsZSB0byBiZSBzZXQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0U2FtcGxlKGludCB4LCBpbnQgeSwgaW50IGIsIGludCBzKSB7Ci0gICAgICAgIHNhbXBsZU1vZGVsLnNldFNhbXBsZSh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCBiLCBzLAotICAgICAgICAgICAgICAgIGRhdGFCdWZmZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHNhbXBsZSBmb3IgdGhlIHNwZWNpZmllZCBiYW5kIGFuZCB0aGUgc3BlY2lmaWVkIHBpeGVsIHdpdGggYQotICAgICAqIGZsb2F0IHNhbXBsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHBpeGVsLgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGJhbmQuCi0gICAgICogQHBhcmFtIHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzYW1wbGUgdG8gYmUgc2V0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZShpbnQgeCwgaW50IHksIGludCBiLCBmbG9hdCBzKSB7Ci0gICAgICAgIHNhbXBsZU1vZGVsLnNldFNhbXBsZSh4IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVYLCB5IC0gc2FtcGxlTW9kZWxUcmFuc2xhdGVZLCBiLCBzLAotICAgICAgICAgICAgICAgIGRhdGFCdWZmZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHNhbXBsZSBmb3IgdGhlIHNwZWNpZmllZCBiYW5kIGFuZCB0aGUgc3BlY2lmaWVkIHBpeGVsIHdpdGggYW4KLSAgICAgKiBpbnRlZ2VyIHNhbXBsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0geAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWwuCi0gICAgICogQHBhcmFtIHkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHBpeGVsLgotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGJhbmQuCi0gICAgICogQHBhcmFtIHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzYW1wbGUgdG8gYmUgc2V0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZShpbnQgeCwgaW50IHksIGludCBiLCBkb3VibGUgcykgewotICAgICAgICBzYW1wbGVNb2RlbC5zZXRTYW1wbGUoeCAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWCwgeSAtIHNhbXBsZU1vZGVsVHJhbnNsYXRlWSwgYiwgcywKLSAgICAgICAgICAgICAgICBkYXRhQnVmZmVyKTsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9Xcml0YWJsZVJlbmRlcmVkSW1hZ2UuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9Xcml0YWJsZVJlbmRlcmVkSW1hZ2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMDUyMzUzYi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvV3JpdGFibGVSZW5kZXJlZEltYWdlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMDkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5Qb2ludDsKLQotLyoqCi0gKiBUaGUgV3JpdGVhYmxlUmVuZGVyZWRJbWFnZSBpbnRlcmZhY2UgaXMgaW50ZXJmYWNlIGZvciBvYmplY3RzIHdoaWNoIGNvbnRhaW5zCi0gKiBSYXN0ZXIgZGF0YSBvZiBvbmUgb3Igc2V2ZXJhbCB0aWxlcy4gVGhpcyBpbnRlcmZhY2UgcHJvdmlkZXMgbm90aWZpY2F0aW9uCi0gKiBtZWNoYW5pc20gZm9yIG9idGFpbmluZyB0aWxlJ3Mgd3JpdGluZyBzdGF0dXMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIFdyaXRhYmxlUmVuZGVyZWRJbWFnZSBleHRlbmRzIFJlbmRlcmVkSW1hZ2UgewotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbmQgY2hlY2tzIG91dCB0aGUgd3JpdGFibGUgdGlsZSBmb3Igd3JpdGluZy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdGlsZVgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGluZGV4IG9mIHRoZSB0aWxlLgotICAgICAqIEBwYXJhbSB0aWxlWQotICAgICAqICAgICAgICAgICAgdGhlIFkgaW5kZXggb2YgdGhlIHRpbGUuCi0gICAgICogQHJldHVybiB0aGUgV3JpdGFibGVSYXN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIFdyaXRhYmxlUmFzdGVyIGdldFdyaXRhYmxlVGlsZShpbnQgdGlsZVgsIGludCB0aWxlWSk7Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHRoZSByZWdpc3RlcmVkIFRpbGVPYnNlcnZlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdG8KLSAgICAgKiAgICAgICAgICAgIHRoZSBUaWxlT2JzZXJ2ZXIgd2hpY2ggaXMgcmVnaXN0ZXJlZCBmb3IgdGhpcwotICAgICAqICAgICAgICAgICAgV3JpdGFibGVSZW5kZXJlZEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZVRpbGVPYnNlcnZlcihUaWxlT2JzZXJ2ZXIgdG8pOwotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgc3BlY2lmaWVkIFRpbGVPYnNlcnZlciB0byB0aGlzIFdyaXRhYmxlUmVuZGVyZWRJbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdG8KLSAgICAgKiAgICAgICAgICAgIHRoZSBUaWxlT2JzZXJ2ZXIgb2JqZWN0IHRvIGJlIGFkZGVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGFkZFRpbGVPYnNlcnZlcihUaWxlT2JzZXJ2ZXIgdG8pOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGlzIGltYWdlIHRvIHRoZSBjb250ZW50cyBvZiB0aGUgc3BlY2lmaWVkIFJhc3Rlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBSYXN0ZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RGF0YShSYXN0ZXIgcik7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhcnJheSBvZiBwb2ludHMgd2hpY2ggcmVwcmVzZW50IGluZGljZXMgb2YgdGlsZXMgd2hpY2ggYXJlIGNoZWNrCi0gICAgICogb3V0IGZvciB3cml0aW5nLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIHBvaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgUG9pbnRbXSBnZXRXcml0YWJsZVRpbGVJbmRpY2VzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIHNwZWNpZmllZCB0aWxlIGlzIHdyaXRhYmxlIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdGlsZVgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGluZGV4IG9mIHRpbGUuCi0gICAgICogQHBhcmFtIHRpbGVZCi0gICAgICogICAgICAgICAgICB0aGUgWSBpbmRleCBvZiB0aWxlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCB0aWxlIGlzIHdyaXRhYmxlLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNUaWxlV3JpdGFibGUoaW50IHRpbGVYLCBpbnQgdGlsZVkpOwotCi0gICAgLyoqCi0gICAgICogUmVsZWFzZSB0aGUgc3BlY2lmaWVkIHdyaXRhYmxlIHRpbGUuIFRoaXMgbWV0aG9kIHJlbW92ZXMgdGhlIHdyaXRlciBmcm9tCi0gICAgICogdGhlIHRpbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRpbGVYCi0gICAgICogICAgICAgICAgICB0aGUgWCBpbmRleCBvZiB0aGUgdGlsZS4KLSAgICAgKiBAcGFyYW0gdGlsZVkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGluZGV4IG9mIHRoZSB0aWxlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbGVhc2VXcml0YWJsZVRpbGUoaW50IHRpbGVYLCBpbnQgdGlsZVkpOwotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoZXJlIGlzIGEgdGlsZSB3aGljaCBpcyBjaGVja2VkIG91dCBmb3Igd3JpdGluZy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGFueSB0aWxlIGlzIGNoZWNrZWQgb3V0IGZvciB3cml0aW5nLCBmYWxzZSBpZiB0aGVyZSBpcwotICAgICAqICAgICAgICAgbm8gc3VjaCB0aWxlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGhhc1RpbGVXcml0ZXJzKCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9wYWNrYWdlLmh0bWwgYi9hd3QvamF2YS9hd3QvaW1hZ2UvcGFja2FnZS5odG1sCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiNGQ2ZWYwLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9wYWNrYWdlLmh0bWwKKysrIC9kZXYvbnVsbApAQCAtMSw4ICswLDAgQEAKLTxodG1sPgotICA8Ym9keT4KLSAgICA8cD4KLSAgICAgIFRoaXMgcGFja2FnZSBjb250YWlucyBjbGFzc2VzIGFuZCBpbnRlcmZhY2VzIHRoYXQgYWxsb3cgdG8gbW9kaWZ5IGV4aXN0aW5nIGltYWdlcyBvciB0byBjcmVhdGUgYSBuZXcgaW1hZ2UgcmF0aGVyIHRoYW4gbG9hZGluZyBpdCBmcm9tIGEgZmlsZS4KLSAgICA8L3A+Ci0gICBAc2luY2UgQW5kcm9pZCAxLjAKLSAgPC9ib2R5PgotPC9odG1sPgpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L2ltYWdlL3JlbmRlcmFibGUvQ29udGV4dHVhbFJlbmRlcmVkSW1hZ2VGYWN0b3J5LmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9Db250ZXh0dWFsUmVuZGVyZWRJbWFnZUZhY3RvcnkuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTg4MWEwYy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9Db250ZXh0dWFsUmVuZGVyZWRJbWFnZUZhY3RvcnkuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDk3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlLnJlbmRlcmFibGU7Ci0KLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJlbmRlcmVkSW1hZ2U7Ci0KLS8qKgotICogQSBmYWN0b3J5IGZvciBjcmVhdGluZyBDb250ZXh0dWFsUmVuZGVyZWRJbWFnZSBvYmplY3RzIHdpdGggdXRpbGl0aWVzIGZvcgotICogbWFuaXB1bGF0aW5nIHRoZSBwcm9wZXJ0aWVzIGluIHRoZSBwYXJhbWV0ZXIgYmxvY2suCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIENvbnRleHR1YWxSZW5kZXJlZEltYWdlRmFjdG9yeSBleHRlbmRzIFJlbmRlcmVkSW1hZ2VGYWN0b3J5IHsKLQotICAgIC8qKgotICAgICAqIE1hcHMgYSByZW5kZXIgY29udGV4dCB0byBhIHBhcmFtZXRlciBibG9jayBhbmQgYSByZW5kZXJhYmxlIGltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4LgotICAgICAqIEBwYXJhbSBhMQotICAgICAqICAgICAgICAgICAgdGhlIFJlbmRlckNvbnRleHQuCi0gICAgICogQHBhcmFtIGEyCi0gICAgICogICAgICAgICAgICB0aGUgUGFyYW1ldGVyQmxvY2suCi0gICAgICogQHBhcmFtIGEzCi0gICAgICogICAgICAgICAgICB0aGUgUmVuZGVyYWJsZUltYWdlLgotICAgICAqIEByZXR1cm4gdGhlIHJlbmRlciBjb250ZXh0LgotICAgICAqLwotICAgIHB1YmxpYyBSZW5kZXJDb250ZXh0IG1hcFJlbmRlckNvbnRleHQoaW50IGEwLCBSZW5kZXJDb250ZXh0IGExLCBQYXJhbWV0ZXJCbG9jayBhMiwKLSAgICAgICAgICAgIFJlbmRlcmFibGVJbWFnZSBhMyk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB2YWx1ZSBvZiB0aGUgcHJvcGVydHkgZnJvbSB0aGUgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIHBhcmFtZXRlciBibG9jayB0byBleGFtaW5lIHRvIGZpbmQgdGhlIHByb3BlcnR5LgotICAgICAqIEBwYXJhbSBhMQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWUgb2YgdGhlIHByb3BlcnR5LgotICAgICAqIEByZXR1cm4gdGhlIHZhbHVlIG9mIHRoZSBwcm9wZXJ0eS4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IGdldFByb3BlcnR5KFBhcmFtZXRlckJsb2NrIGEwLCBTdHJpbmcgYTEpOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgcmVuZGVyZWQgaW1hZ2UgZGV0ZXJtaW5lZCBieSB0aGUgcmVuZGVyIGNvbnRleHQgYW5kIHBhcmFtZXRlcgotICAgICAqIGJsb2NrLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhMAotICAgICAqICAgICAgICAgICAgdGhlIFJlbmRlckNvbnRleHQuCi0gICAgICogQHBhcmFtIGExCi0gICAgICogICAgICAgICAgICB0aGUgUGFyYW1ldGVyQmxvY2suCi0gICAgICogQHJldHVybiB0aGUgcmVuZGVyZWQgaW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIFJlbmRlcmVkSW1hZ2UgY3JlYXRlKFJlbmRlckNvbnRleHQgYTAsIFBhcmFtZXRlckJsb2NrIGExKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJvdW5kaW5nIHJlY3RhbmdsZSBmcm9tIHRoZSBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUgcGFyYW1ldGVyIGJsb2NrIHRvIHJlYWQgdGhlIGJvdW5kcyBmcm9tLgotICAgICAqIEByZXR1cm4gdGhlIGJvdW5kaW5nIHJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoUGFyYW1ldGVyQmxvY2sgYTApOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbmFtZXMgb2YgYWxsIG9mIHRoZSBzdXBwb3J0ZWQgcHJvcGVydGllcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBwcm9wZXJ0eSBuYW1lcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0UHJvcGVydHlOYW1lcygpOwotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoaXMgaW1hZ2UgZmFjdG9yeSBpcyBkeW5hbWljLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBpbWFnZSBmYWN0b3J5IGlzIGR5bmFtaWMuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNEeW5hbWljKCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9yZW5kZXJhYmxlL1BhcmFtZXRlckJsb2NrLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9QYXJhbWV0ZXJCbG9jay5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3ZGRlNzNhLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9yZW5kZXJhYmxlL1BhcmFtZXRlckJsb2NrLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NjggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2UucmVuZGVyYWJsZTsKLQotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJlbmRlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5pby5TZXJpYWxpemFibGU7Ci1pbXBvcnQgamF2YS51dGlsLlZlY3RvcjsKLQotLyoqCi0gKiBUaGUgY2xhc3MgUGFyYW1ldGVyQmxvY2sgZ3JvdXBzIGFuIGluZGV4ZWQgc2V0IG9mIHBhcmFtZXRlciBkYXRhIHdpdGggYSBzZXQKLSAqIG9mIHJlbmRlcmFibGUgKHNvdXJjZSkgaW1hZ2VzLiBUaGUgbWFwcGluZyBiZXR3ZWVuIHRoZSBpbmRleGVkIHBhcmFtZXRlcnMgYW5kCi0gKiB0aGVpciBwcm9wZXJ0eSBuYW1lcyBpcyBwcm92aWRlZCBieSBhIHtAbGluayBDb250ZXh0dWFsUmVuZGVyZWRJbWFnZUZhY3Rvcnl9LgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIFBhcmFtZXRlckJsb2NrIGltcGxlbWVudHMgQ2xvbmVhYmxlLCBTZXJpYWxpemFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IHNlcmlhbFZlcnNpb25VSUQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gLTc1NzcxMTU1NTE3ODUyNDA3NTBMOwotCi0gICAgLyoqCi0gICAgICogVGhlIHNvdXJjZXMgKHJlbmRlcmFibGUgaW1hZ2VzKS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgVmVjdG9yPE9iamVjdD4gc291cmNlcyA9IG5ldyBWZWN0b3I8T2JqZWN0PigpOwotCi0gICAgLyoqCi0gICAgICogVGhlIHBhcmFtZXRlcnMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFZlY3RvcjxPYmplY3Q+IHBhcmFtZXRlcnMgPSBuZXcgVmVjdG9yPE9iamVjdD4oKTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZWN0b3Igb2Ygc291cmNlIGltYWdlcy4KLSAgICAgKiBAcGFyYW0gcGFyYW1ldGVycwotICAgICAqICAgICAgICAgICAgdGhlIHZlY3RvciBvZiBwYXJhbWV0ZXJzLgotICAgICAqLwotICAgIHB1YmxpYyBQYXJhbWV0ZXJCbG9jayhWZWN0b3I8T2JqZWN0PiBzb3VyY2VzLCBWZWN0b3I8T2JqZWN0PiBwYXJhbWV0ZXJzKSB7Ci0gICAgICAgIHNldFNvdXJjZXMoc291cmNlcyk7Ci0gICAgICAgIHNldFBhcmFtZXRlcnMocGFyYW1ldGVycyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHBhcmFtZXRlciBibG9jayB3aXRoIG5vIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZWN0b3Igb2Ygc291cmNlIGltYWdlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGFyYW1ldGVyQmxvY2soVmVjdG9yPE9iamVjdD4gc291cmNlcykgewotICAgICAgICBzZXRTb3VyY2VzKHNvdXJjZXMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBwYXJhbWV0ZXIgYmxvY2sgd2l0aCBubyBpbWFnZSBvciBwYXJhbWV0ZXIgdmVjdG9ycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGFyYW1ldGVyQmxvY2soKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc291cmNlIGltYWdlIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXguCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCB3aGVyZSB0aGUgc291cmNlIHdpbGwgYmUgcGxhY2VkLgotICAgICAqIEByZXR1cm4gdGhpcyBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICovCi0gICAgcHVibGljIFBhcmFtZXRlckJsb2NrIHNldFNvdXJjZShPYmplY3Qgc291cmNlLCBpbnQgaW5kZXgpIHsKLSAgICAgICAgaWYgKHNvdXJjZXMuc2l6ZSgpIDwgaW5kZXggKyAxKSB7Ci0gICAgICAgICAgICBzb3VyY2VzLnNldFNpemUoaW5kZXggKyAxKTsKLSAgICAgICAgfQotICAgICAgICBzb3VyY2VzLnNldEVsZW1lbnRBdChzb3VyY2UsIGluZGV4KTsKLSAgICAgICAgcmV0dXJuIHRoaXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgcGFyYW1ldGVyIHZhbHVlIG9iamVjdCBhdCB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBvYmoKLSAgICAgKiAgICAgICAgICAgIHRoZSBwYXJhbWV0ZXIgdmFsdWUgdG8gcGxhY2UgYXQgdGhlIGRlc2lyZWQgaW5kZXguCi0gICAgICogQHBhcmFtIGluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggd2hlcmUgdGhlIG9iamVjdCBpcyB0byBiZSBwbGFjZWQgaW4gdGhlIHZlY3RvciBvZgotICAgICAqICAgICAgICAgICAgcGFyYW1ldGVycy4KLSAgICAgKiBAcmV0dXJuIHRoaXMgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqLwotICAgIHB1YmxpYyBQYXJhbWV0ZXJCbG9jayBzZXQoT2JqZWN0IG9iaiwgaW50IGluZGV4KSB7Ci0gICAgICAgIGlmIChwYXJhbWV0ZXJzLnNpemUoKSA8IGluZGV4ICsgMSkgewotICAgICAgICAgICAgcGFyYW1ldGVycy5zZXRTaXplKGluZGV4ICsgMSk7Ci0gICAgICAgIH0KLSAgICAgICAgcGFyYW1ldGVycy5zZXRFbGVtZW50QXQob2JqLCBpbmRleCk7Ci0gICAgICAgIHJldHVybiB0aGlzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgYSBzb3VyY2UgdG8gdGhlIHZlY3RvciBvZiBzb3VyY2VzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgdG8gYWRkLgotICAgICAqIEByZXR1cm4gdGhpcyBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICovCi0gICAgcHVibGljIFBhcmFtZXRlckJsb2NrIGFkZFNvdXJjZShPYmplY3Qgc291cmNlKSB7Ci0gICAgICAgIHNvdXJjZXMuYWRkRWxlbWVudChzb3VyY2UpOwotICAgICAgICByZXR1cm4gdGhpczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBvYmplY3QgdG8gdGhlIHZlY3RvciBvZiBwYXJhbWV0ZXIgdmFsdWVzCi0gICAgICogCi0gICAgICogQHBhcmFtIG9iagotICAgICAqICAgICAgICAgICAgdGhlIG9iaiB0byBhZGQuCi0gICAgICogQHJldHVybiB0aGlzIHBhcmFtZXRlciBibG9jay4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGFyYW1ldGVyQmxvY2sgYWRkKE9iamVjdCBvYmopIHsKLSAgICAgICAgcGFyYW1ldGVycy5hZGRFbGVtZW50KG9iaik7Ci0gICAgICAgIHJldHVybiB0aGlzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHZlY3RvciBvZiBzb3VyY2VzLCByZXBsYWNpbmcgdGhlIGV4aXN0aW5nIHZlY3RvciBvZiBzb3VyY2VzLCBpZgotICAgICAqIGFueS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlcwotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBzb3VyY2VzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNvdXJjZXMoVmVjdG9yPE9iamVjdD4gc291cmNlcykgewotICAgICAgICB0aGlzLnNvdXJjZXMgPSBzb3VyY2VzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHZlY3RvciBvZiBwYXJhbWV0ZXJzLCByZXBsYWNpbmcgdGhlIGV4aXN0aW5nIHZlY3RvciBvZgotICAgICAqIHBhcmFtZXRlcnMsIGlmIGFueS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcGFyYW1ldGVycwotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBwYXJhbWV0ZXJzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFBhcmFtZXRlcnMoVmVjdG9yPE9iamVjdD4gcGFyYW1ldGVycykgewotICAgICAgICB0aGlzLnBhcmFtZXRlcnMgPSBwYXJhbWV0ZXJzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHZlY3RvciBvZiBzb3VyY2VzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNvdXJjZXMuCi0gICAgICovCi0gICAgcHVibGljIFZlY3RvcjxPYmplY3Q+IGdldFNvdXJjZXMoKSB7Ci0gICAgICAgIHJldHVybiBzb3VyY2VzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHZlY3RvciBvZiBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHBhcmFtZXRlcnMuCi0gICAgICovCi0gICAgcHVibGljIFZlY3RvcjxPYmplY3Q+IGdldFBhcmFtZXRlcnMoKSB7Ci0gICAgICAgIHJldHVybiBwYXJhbWV0ZXJzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHNvdXJjZSBhdCB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIHNvdXJjZSBvYmplY3QgZm91bmQgYXQgdGhlIHNwZWNpZmllZCBpbmRleC4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IGdldFNvdXJjZShpbnQgaW5kZXgpIHsKLSAgICAgICAgcmV0dXJuIHNvdXJjZXMuZWxlbWVudEF0KGluZGV4KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBvYmplY3QgcGFyYW1ldGVyIGZvdW5kIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXguCi0gICAgICogCi0gICAgICogQHBhcmFtIGluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXguCi0gICAgICogQHJldHVybiB0aGUgcGFyYW1ldGVyIG9iamVjdCBmb3VuZCBhdCB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqLwotICAgIHB1YmxpYyBPYmplY3QgZ2V0T2JqZWN0UGFyYW1ldGVyKGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gcGFyYW1ldGVycy5lbGVtZW50QXQoaW5kZXgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNoYWxsb3cgY2xvbmUgKGNsb25lcyB1c2luZyB0aGUgc3VwZXJjbGFzcyBjbG9uZSBtZXRob2QpLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGNsb25lIG9mIHRoaXMgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBPYmplY3Qgc2hhbGxvd0Nsb25lKCkgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIHN1cGVyLmNsb25lKCk7Ci0gICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBjb3B5IG9mIHRoaXMgUGFyYW1ldGVyQmxvY2sgaW5zdGFuY2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaWRlbnRpY2FsIGNvcHkgb2YgdGhpcyBpbnN0YW5jZS4KLSAgICAgKi8KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygidW5jaGVja2VkIikKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgT2JqZWN0IGNsb25lKCkgewotICAgICAgICBQYXJhbWV0ZXJCbG9jayByZXBsaWNhOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmVwbGljYSA9IChQYXJhbWV0ZXJCbG9jaylzdXBlci5jbG9uZSgpOwotICAgICAgICB9IGNhdGNoIChFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHNvdXJjZXMgIT0gbnVsbCkgewotICAgICAgICAgICAgcmVwbGljYS5zZXRTb3VyY2VzKChWZWN0b3I8T2JqZWN0Pikoc291cmNlcy5jbG9uZSgpKSk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHBhcmFtZXRlcnMgIT0gbnVsbCkgewotICAgICAgICAgICAgcmVwbGljYS5zZXRQYXJhbWV0ZXJzKChWZWN0b3I8T2JqZWN0PikocGFyYW1ldGVycy5jbG9uZSgpKSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlcGxpY2E7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiBjbGFzc2VzIGNvcnJlc3BvbmRpbmcgdG8gYWxsIG9mIHRoZSBwYXJhbWV0ZXIgdmFsdWVzCi0gICAgICogZm91bmQgaW4gdGhlIGFycmF5IG9mIHBhcmFtZXRlcnMsIGluIG9yZGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHBhcmFtZXRlciBjbGFzc2VzLgotICAgICAqLwotICAgIHB1YmxpYyBDbGFzc1tdIGdldFBhcmFtQ2xhc3NlcygpIHsKLSAgICAgICAgaW50IGNvdW50ID0gcGFyYW1ldGVycy5zaXplKCk7Ci0gICAgICAgIENsYXNzIHBhcmFtQ2xhc3Nlc1tdID0gbmV3IENsYXNzW2NvdW50XTsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGNvdW50OyBpKyspIHsKLSAgICAgICAgICAgIHBhcmFtQ2xhc3Nlc1tpXSA9IHBhcmFtZXRlcnMuZWxlbWVudEF0KGkpLmdldENsYXNzKCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHBhcmFtQ2xhc3NlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSByZW5kZXJhYmxlIHNvdXJjZSBpbWFnZSBmb3VuZCBhdCB0aGUgc3BlY2lmaWVkIGluZGV4IGluIHRoZQotICAgICAqIHNvdXJjZSBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSByZW5kZXJhYmxlIHNvdXJjZSBpbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVuZGVyYWJsZUltYWdlIGdldFJlbmRlcmFibGVTb3VyY2UoaW50IGluZGV4KSB7Ci0gICAgICAgIHJldHVybiAoUmVuZGVyYWJsZUltYWdlKXNvdXJjZXMuZWxlbWVudEF0KGluZGV4KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcmFwcyB0aGUgc2hvcnQgdmFsdWUgaW4gYSBTaG9ydCBhbmQgcGxhY2VzIGl0IGluIHRoZSBwYXJhbWV0ZXIgYmxvY2sgYXQKLSAgICAgKiB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgc2hvcnQgdmFsdWUgb2YgdGhlIHBhcmFtZXRlci4KLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoaXMgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqLwotICAgIHB1YmxpYyBQYXJhbWV0ZXJCbG9jayBzZXQoc2hvcnQgcywgaW50IGluZGV4KSB7Ci0gICAgICAgIHJldHVybiBzZXQobmV3IFNob3J0KHMpLCBpbmRleCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogV3JhcHMgdGhlIHNob3J0IHZhbHVlIGluIGEgU2hvcnQgYW5kIGFkZHMgaXQgdG8gdGhlIHBhcmFtZXRlciBibG9jay4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIHNob3J0IHZhbHVlIG9mIHRoZSBwYXJhbWV0ZXIuCi0gICAgICogQHJldHVybiB0aGlzIHBhcmFtZXRlciBibG9jay4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGFyYW1ldGVyQmxvY2sgYWRkKHNob3J0IHMpIHsKLSAgICAgICAgcmV0dXJuIGFkZChuZXcgU2hvcnQocykpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFdyYXBzIHRoZSBsb25nIHZhbHVlIGluIGEgTG9uZyBhbmQgcGxhY2VzIGl0IGluIHRoZSBwYXJhbWV0ZXIgYmxvY2sgYXQKLSAgICAgKiB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgbG9uZyB2YWx1ZSBvZiB0aGUgcGFyYW1ldGVyLgotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhpcyBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICovCi0gICAgcHVibGljIFBhcmFtZXRlckJsb2NrIHNldChsb25nIGwsIGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gc2V0KG5ldyBMb25nKGwpLCBpbmRleCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogV3JhcHMgdGhlIGxvbmcgdmFsdWUgaW4gYSBMb25nIGFuZCBhZGRzIGl0IHRvIHRoZSBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICogCi0gICAgICogQHBhcmFtIGwKLSAgICAgKiAgICAgICAgICAgIHRoZSBsb25nIHZhbHVlIG9mIHRoZSBwYXJhbWV0ZXIuCi0gICAgICogQHJldHVybiB0aGlzIHBhcmFtZXRlciBibG9jay4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGFyYW1ldGVyQmxvY2sgYWRkKGxvbmcgbCkgewotICAgICAgICByZXR1cm4gYWRkKG5ldyBMb25nKGwpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcmFwcyB0aGUgaW50ZWdlciB2YWx1ZSBpbiBhbiBJbnRlZ2VyIGFuZCBwbGFjZXMgaXQgaW4gdGhlIHBhcmFtZXRlcgotICAgICAqIGJsb2NrIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXguCi0gICAgICogCi0gICAgICogQHBhcmFtIGkKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnRlZ2VyIHZhbHVlIG9mIHRoZSBwYXJhbWV0ZXIuCi0gICAgICogQHBhcmFtIGluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXguCi0gICAgICogQHJldHVybiB0aGlzIHBhcmFtZXRlciBibG9jay4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGFyYW1ldGVyQmxvY2sgc2V0KGludCBpLCBpbnQgaW5kZXgpIHsKLSAgICAgICAgcmV0dXJuIHNldChuZXcgSW50ZWdlcihpKSwgaW5kZXgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFdyYXBzIHRoZSBpbnRlZ2VyIHZhbHVlIGluIGFuIEludGVnZXIgYW5kIGFkZHMgaXQgdG8gdGhlIHBhcmFtZXRlciBibG9jay4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaQotICAgICAqICAgICAgICAgICAgdGhlIGludGVnZXIgdmFsdWUgb2YgdGhlIHBhcmFtZXRlci4KLSAgICAgKiBAcmV0dXJuIHRoaXMgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqLwotICAgIHB1YmxpYyBQYXJhbWV0ZXJCbG9jayBhZGQoaW50IGkpIHsKLSAgICAgICAgcmV0dXJuIGFkZChuZXcgSW50ZWdlcihpKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogV3JhcHMgdGhlIGZsb2F0IHZhbHVlIGluIGEgRmxvYXQgYW5kIHBsYWNlcyBpdCBpbiB0aGUgcGFyYW1ldGVyIGJsb2NrIGF0Ci0gICAgICogdGhlIHNwZWNpZmllZCBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZgotICAgICAqICAgICAgICAgICAgdGhlIGZsb2F0IHZhbHVlIG9mIHRoZSBwYXJhbWV0ZXIuCi0gICAgICogQHBhcmFtIGluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXguCi0gICAgICogQHJldHVybiB0aGlzIHBhcmFtZXRlciBibG9jay4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGFyYW1ldGVyQmxvY2sgc2V0KGZsb2F0IGYsIGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gc2V0KG5ldyBGbG9hdChmKSwgaW5kZXgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFdyYXBzIHRoZSBmbG9hdCB2YWx1ZSBpbiBhIEZsb2F0IGFuZCBhZGRzIGl0IHRvIHRoZSBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICogCi0gICAgICogQHBhcmFtIGYKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbG9hdCB2YWx1ZSBvZiB0aGUgcGFyYW1ldGVyLgotICAgICAqIEByZXR1cm4gdGhpcyBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICovCi0gICAgcHVibGljIFBhcmFtZXRlckJsb2NrIGFkZChmbG9hdCBmKSB7Ci0gICAgICAgIHJldHVybiBhZGQobmV3IEZsb2F0KGYpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcmFwcyB0aGUgZG91YmxlIHZhbHVlIGluIGEgRG91YmxlIGFuZCBwbGFjZXMgaXQgaW4gdGhlIHBhcmFtZXRlciBibG9jawotICAgICAqIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXguCi0gICAgICogCi0gICAgICogQHBhcmFtIGQKLSAgICAgKiAgICAgICAgICAgIHRoZSBkb3VibGUgdmFsdWUgb2YgdGhlIHBhcmFtZXRlci4KLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoaXMgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqLwotICAgIHB1YmxpYyBQYXJhbWV0ZXJCbG9jayBzZXQoZG91YmxlIGQsIGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gc2V0KG5ldyBEb3VibGUoZCksIGluZGV4KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcmFwcyB0aGUgZG91YmxlIHZhbHVlIGluIGEgRG91YmxlIGFuZCBhZGRzIGl0IHRvIHRoZSBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICogCi0gICAgICogQHBhcmFtIGQKLSAgICAgKiAgICAgICAgICAgIHRoZSBkb3VibGUgdmFsdWUgb2YgdGhlIHBhcmFtZXRlci4KLSAgICAgKiBAcmV0dXJuIHRoaXMgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqLwotICAgIHB1YmxpYyBQYXJhbWV0ZXJCbG9jayBhZGQoZG91YmxlIGQpIHsKLSAgICAgICAgcmV0dXJuIGFkZChuZXcgRG91YmxlKGQpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcmFwcyB0aGUgY2hhciB2YWx1ZSBpbiBhIENoYXJhY3RlciBhbmQgcGxhY2VzIGl0IGluIHRoZSBwYXJhbWV0ZXIgYmxvY2sKLSAgICAgKiBhdCB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgY2hhciB2YWx1ZSBvZiB0aGUgcGFyYW1ldGVyLgotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhpcyBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICovCi0gICAgcHVibGljIFBhcmFtZXRlckJsb2NrIHNldChjaGFyIGMsIGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gc2V0KG5ldyBDaGFyYWN0ZXIoYyksIGluZGV4KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcmFwcyB0aGUgY2hhciB2YWx1ZSBpbiBhIENoYXJhY3RlciBhbmQgYWRkcyBpdCB0byB0aGUgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgY2hhciB2YWx1ZSBvZiB0aGUgcGFyYW1ldGVyLgotICAgICAqIEByZXR1cm4gdGhpcyBwYXJhbWV0ZXIgYmxvY2suCi0gICAgICovCi0gICAgcHVibGljIFBhcmFtZXRlckJsb2NrIGFkZChjaGFyIGMpIHsKLSAgICAgICAgcmV0dXJuIGFkZChuZXcgQ2hhcmFjdGVyKGMpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcmFwcyB0aGUgYnl0ZSB2YWx1ZSBpbiBhIEJ5dGUgYW5kIHBsYWNlcyBpdCBpbiB0aGUgcGFyYW1ldGVyIGJsb2NrIGF0Ci0gICAgICogdGhlIHNwZWNpZmllZCBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJ5dGUgdmFsdWUgb2YgdGhlIHBhcmFtZXRlci4KLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoaXMgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqLwotICAgIHB1YmxpYyBQYXJhbWV0ZXJCbG9jayBzZXQoYnl0ZSBiLCBpbnQgaW5kZXgpIHsKLSAgICAgICAgcmV0dXJuIHNldChuZXcgQnl0ZShiKSwgaW5kZXgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFdyYXBzIHRoZSBieXRlIHZhbHVlIGluIGEgQnl0ZSBhbmQgYWRkcyBpdCB0byB0aGUgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgYnl0ZSB2YWx1ZSBvZiB0aGUgcGFyYW1ldGVyLgotICAgICAqIEByZXR1cm4gdGhlIHBhcmFtZXRlciBibG9jay4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGFyYW1ldGVyQmxvY2sgYWRkKGJ5dGUgYikgewotICAgICAgICByZXR1cm4gYWRkKG5ldyBCeXRlKGIpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBSZW5kZXJlZEltYWdlIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXggZnJvbSB0aGUgdmVjdG9yIG9mIHNvdXJjZQotICAgICAqIGltYWdlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSByZW5kZXJlZCBpbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVuZGVyZWRJbWFnZSBnZXRSZW5kZXJlZFNvdXJjZShpbnQgaW5kZXgpIHsKLSAgICAgICAgcmV0dXJuIChSZW5kZXJlZEltYWdlKXNvdXJjZXMuZWxlbWVudEF0KGluZGV4KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzaG9ydC12YWx1ZWQgcGFyYW1ldGVyIGZvdW5kIGF0IHRoZSBkZXNpcmVkIGluZGV4IGluIHRoZSB2ZWN0b3IKLSAgICAgKiBvZiBwYXJhbWV0ZXIgdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIHNob3J0IHBhcmFtZXRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc2hvcnQgZ2V0U2hvcnRQYXJhbWV0ZXIoaW50IGluZGV4KSB7Ci0gICAgICAgIHJldHVybiAoKFNob3J0KXBhcmFtZXRlcnMuZWxlbWVudEF0KGluZGV4KSkuc2hvcnRWYWx1ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGxvbmctdmFsdWVkIHBhcmFtZXRlciBmb3VuZCBhdCB0aGUgZGVzaXJlZCBpbmRleCBpbiB0aGUgdmVjdG9yCi0gICAgICogb2YgcGFyYW1ldGVyIHZhbHVlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSBsb25nIHBhcmFtZXRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgbG9uZyBnZXRMb25nUGFyYW1ldGVyKGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gKChMb25nKXBhcmFtZXRlcnMuZWxlbWVudEF0KGluZGV4KSkubG9uZ1ZhbHVlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW50ZWdlci12YWx1ZWQgcGFyYW1ldGVyIGZvdW5kIGF0IHRoZSBkZXNpcmVkIGluZGV4IGluIHRoZQotICAgICAqIHZlY3RvciBvZiBwYXJhbWV0ZXIgdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIGludGVnZXIgcGFyYW1ldGVyLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0SW50UGFyYW1ldGVyKGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gKChJbnRlZ2VyKXBhcmFtZXRlcnMuZWxlbWVudEF0KGluZGV4KSkuaW50VmFsdWUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBmbG9hdC12YWx1ZWQgcGFyYW1ldGVyIGZvdW5kIGF0IHRoZSBkZXNpcmVkIGluZGV4IGluIHRoZSB2ZWN0b3IKLSAgICAgKiBvZiBwYXJhbWV0ZXIgdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIGZsb2F0IHBhcmFtZXRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0RmxvYXRQYXJhbWV0ZXIoaW50IGluZGV4KSB7Ci0gICAgICAgIHJldHVybiAoKEZsb2F0KXBhcmFtZXRlcnMuZWxlbWVudEF0KGluZGV4KSkuZmxvYXRWYWx1ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRvdWJsZS12YWx1ZWQgcGFyYW1ldGVyIGZvdW5kIGF0IHRoZSBkZXNpcmVkIGluZGV4IGluIHRoZSB2ZWN0b3IKLSAgICAgKiBvZiBwYXJhbWV0ZXIgdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIGRvdWJsZSBwYXJhbWV0ZXIuCi0gICAgICovCi0gICAgcHVibGljIGRvdWJsZSBnZXREb3VibGVQYXJhbWV0ZXIoaW50IGluZGV4KSB7Ci0gICAgICAgIHJldHVybiAoKERvdWJsZSlwYXJhbWV0ZXJzLmVsZW1lbnRBdChpbmRleCkpLmRvdWJsZVZhbHVlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY2hhci12YWx1ZWQgcGFyYW1ldGVyIGZvdW5kIGF0IHRoZSBkZXNpcmVkIGluZGV4IGluIHRoZSB2ZWN0b3IKLSAgICAgKiBvZiBwYXJhbWV0ZXIgdmFsdWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIGNoYXIgcGFyYW1ldGVyLgotICAgICAqLwotICAgIHB1YmxpYyBjaGFyIGdldENoYXJQYXJhbWV0ZXIoaW50IGluZGV4KSB7Ci0gICAgICAgIHJldHVybiAoKENoYXJhY3RlcilwYXJhbWV0ZXJzLmVsZW1lbnRBdChpbmRleCkpLmNoYXJWYWx1ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJ5dGUtdmFsdWVkIHBhcmFtZXRlciBmb3VuZCBhdCB0aGUgZGVzaXJlZCBpbmRleCBpbiB0aGUgdmVjdG9yCi0gICAgICogb2YgcGFyYW1ldGVyIHZhbHVlcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSBieXRlIHBhcmFtZXRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYnl0ZSBnZXRCeXRlUGFyYW1ldGVyKGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gKChCeXRlKXBhcmFtZXRlcnMuZWxlbWVudEF0KGluZGV4KSkuYnl0ZVZhbHVlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2xlYXJzIHRoZSB2ZWN0b3Igb2Ygc291cmNlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVTb3VyY2VzKCkgewotICAgICAgICBzb3VyY2VzLnJlbW92ZUFsbEVsZW1lbnRzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2xlYXJzIHRoZSB2ZWN0b3Igb2YgcGFyYW1ldGVycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVQYXJhbWV0ZXJzKCkgewotICAgICAgICBwYXJhbWV0ZXJzLnJlbW92ZUFsbEVsZW1lbnRzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGVsZW1lbnRzIGluIHRoZSB2ZWN0b3Igb2Ygc291cmNlcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgZWxlbWVudHMgaW4gdGhlIHZlY3RvciBvZiBzb3VyY2VzLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TnVtU291cmNlcygpIHsKLSAgICAgICAgcmV0dXJuIHNvdXJjZXMuc2l6ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG51bWJlciBvZiBlbGVtZW50cyBpbiB0aGUgdmVjdG9yIG9mIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIGVsZW1lbnRzIGluIHRoZSB2ZWN0b3Igb2YgcGFyYW1ldGVycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE51bVBhcmFtZXRlcnMoKSB7Ci0gICAgICAgIHJldHVybiBwYXJhbWV0ZXJzLnNpemUoKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9SZW5kZXJDb250ZXh0LmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9SZW5kZXJDb250ZXh0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBkYjUxMmYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L2ltYWdlL3JlbmRlcmFibGUvUmVuZGVyQ29udGV4dC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjE0ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlLnJlbmRlcmFibGU7Ci0KLWltcG9ydCBqYXZhLmF3dC5SZW5kZXJpbmdIaW50czsKLWltcG9ydCBqYXZhLmF3dC5TaGFwZTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgUmVuZGVyQ29udGV4dCBzdG9yZXMgZGF0YSBvbiBob3cgYW4gaW1hZ2UgaXMgdG8gYmUgcmVuZGVyZWQ6IHRoZQotICogYWZmaW5lIHRyYW5zZm9ybSwgdGhlIGFyZWEgb2YgaW50ZXJlc3QsIGFuZCB0aGUgcmVuZGVyaW5nIGhpbnRzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIFJlbmRlckNvbnRleHQgaW1wbGVtZW50cyBDbG9uZWFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIGFmZmluZSB0cmFuc2Zvcm0uCi0gICAgICovCi0gICAgQWZmaW5lVHJhbnNmb3JtIHRyYW5zZm9ybTsKLQotICAgIC8qKgotICAgICAqIFRoZSBhcmVhIG9mIGludGVyZXN0LgotICAgICAqLwotICAgIFNoYXBlIGFvaTsKLQotICAgIC8qKgotICAgICAqIFRoZSByZW5kZXJpbmcgaGludHMuCi0gICAgICovCi0gICAgUmVuZGVyaW5nSGludHMgaGludHM7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgcmVuZGVyIGNvbnRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHVzcjJkZXYKLSAgICAgKiAgICAgICAgICAgIHRoZSBhZmZpbmUgdHJhbnNmb3JtLgotICAgICAqIEBwYXJhbSBhb2kKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcmVhIG9mIGludGVyZXN0LgotICAgICAqIEBwYXJhbSBoaW50cwotICAgICAqICAgICAgICAgICAgdGhlIHJlbmRlcmluZyBoaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVuZGVyQ29udGV4dChBZmZpbmVUcmFuc2Zvcm0gdXNyMmRldiwgU2hhcGUgYW9pLCBSZW5kZXJpbmdIaW50cyBoaW50cykgewotICAgICAgICB0aGlzLnRyYW5zZm9ybSA9IChBZmZpbmVUcmFuc2Zvcm0pdXNyMmRldi5jbG9uZSgpOwotICAgICAgICB0aGlzLmFvaSA9IGFvaTsKLSAgICAgICAgdGhpcy5oaW50cyA9IGhpbnRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyByZW5kZXIgY29udGV4dCB3aXRoIG5vIHNwZWNpZmllZCBoaW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdXNyMmRldgotICAgICAqICAgICAgICAgICAgdGhlIGFmZmluZSB0cmFuc2Zvcm0uCi0gICAgICogQHBhcmFtIGFvaQotICAgICAqICAgICAgICAgICAgdGhlIGFyZWEgb2YgaW50ZXJlc3QuCi0gICAgICovCi0gICAgcHVibGljIFJlbmRlckNvbnRleHQoQWZmaW5lVHJhbnNmb3JtIHVzcjJkZXYsIFNoYXBlIGFvaSkgewotICAgICAgICB0aGlzKHVzcjJkZXYsIGFvaSwgbnVsbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHJlbmRlciBjb250ZXh0IHdpdGggbm8gc3BlY2lmaWVkIGFyZWEgb2YgaW50ZXJlc3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHVzcjJkZXYKLSAgICAgKiAgICAgICAgICAgIHRoZSBhZmZpbmUgdHJhbnNmb3JtLgotICAgICAqIEBwYXJhbSBoaW50cwotICAgICAqICAgICAgICAgICAgdGhlIHJlbmRlcmluZyBoaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVuZGVyQ29udGV4dChBZmZpbmVUcmFuc2Zvcm0gdXNyMmRldiwgUmVuZGVyaW5nSGludHMgaGludHMpIHsKLSAgICAgICAgdGhpcyh1c3IyZGV2LCBudWxsLCBoaW50cyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IHJlbmRlciBjb250ZXh0IHdpdGggbm8gcmVuZGVyaW5nIGhpbnRzIG9yIGFyZWEgb2YKLSAgICAgKiBpbnRlcmVzdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdXNyMmRldgotICAgICAqICAgICAgICAgICAgdGhlIGFmZmluZSB0cmFuc2Zvcm0uCi0gICAgICovCi0gICAgcHVibGljIFJlbmRlckNvbnRleHQoQWZmaW5lVHJhbnNmb3JtIHVzcjJkZXYpIHsKLSAgICAgICAgdGhpcyh1c3IyZGV2LCBudWxsLCBudWxsKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgT2JqZWN0IGNsb25lKCkgewotICAgICAgICByZXR1cm4gbmV3IFJlbmRlckNvbnRleHQodHJhbnNmb3JtLCBhb2ksIGhpbnRzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBhZmZpbmUgdHJhbnNmb3JtIGZvciB0aGlzIHJlbmRlciBjb250ZXh0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBuZXdUcmFuc2Zvcm0KLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgYWZmaW5lIHRyYW5zZm9ybS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRUcmFuc2Zvcm0oQWZmaW5lVHJhbnNmb3JtIG5ld1RyYW5zZm9ybSkgewotICAgICAgICB0cmFuc2Zvcm0gPSAoQWZmaW5lVHJhbnNmb3JtKW5ld1RyYW5zZm9ybS5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbmNhdGVuYXRlcyB0aGUgY3VycmVudCB0cmFuc2Zvcm0gd2l0aCB0aGUgc3BlY2lmaWVkIHRyYW5zZm9ybSAoc28gdGhleQotICAgICAqIGFyZSBhcHBsaWVkIHdpdGggdGhlIHNwZWNpZmllZCB0cmFuc2Zvcm0gYWN0aW5nIGZpcnN0KSBhbmQgc2V0cyB0aGUKLSAgICAgKiByZXN1bHRpbmcgdHJhbnNmb3JtIGFzIHRoZSBhZmZpbmUgdHJhbnNmb3JtIG9mIHRoaXMgcmVuZGVyaW5nIGNvbnRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1vZFRyYW5zZm9ybQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyB0cmFuc2Zvcm0gd2hpY2ggbW9kaWZpZXMgdGhlIGN1cnJlbnQgdHJhbnNmb3JtLgotICAgICAqIEBkZXByZWNhdGVkIHVzZQotICAgICAqICAgICAgICAgICAgIHtAbGluayBSZW5kZXJDb250ZXh0I3ByZUNvbmNhdGVuYXRlVHJhbnNmb3JtKEFmZmluZVRyYW5zZm9ybSl9Ci0gICAgICogICAgICAgICAgICAgLgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIHZvaWQgcHJlQ29uY2V0ZW5hdGVUcmFuc2Zvcm0oQWZmaW5lVHJhbnNmb3JtIG1vZFRyYW5zZm9ybSkgewotICAgICAgICBwcmVDb25jYXRlbmF0ZVRyYW5zZm9ybShtb2RUcmFuc2Zvcm0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbmNhdGVuYXRlcyB0aGUgY3VycmVudCB0cmFuc2Zvcm0gd2l0aCB0aGUgc3BlY2lmaWVkIHRyYW5zZm9ybSAoc28gdGhleQotICAgICAqIGFyZSBhcHBsaWVkIHdpdGggdGhlIHNwZWNpZmllZCB0cmFuc2Zvcm0gYWN0aW5nIGZpcnN0KSBhbmQgc2V0cyB0aGUKLSAgICAgKiByZXN1bHRpbmcgdHJhbnNmb3JtIGFzIHRoZSBhZmZpbmUgdHJhbnNmb3JtIG9mIHRoaXMgcmVuZGVyaW5nIGNvbnRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1vZFRyYW5zZm9ybQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyB0cmFuc2Zvcm0gd2hpY2ggbW9kaWZpZXMgdGhlIGN1cnJlbnQgdHJhbnNmb3JtLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHByZUNvbmNhdGVuYXRlVHJhbnNmb3JtKEFmZmluZVRyYW5zZm9ybSBtb2RUcmFuc2Zvcm0pIHsKLSAgICAgICAgdHJhbnNmb3JtLnByZUNvbmNhdGVuYXRlKG1vZFRyYW5zZm9ybSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29uY2F0ZW5hdGUgdGhlIHNwZWNpZmllZCB0cmFuc2Zvcm0gd2l0aCB0aGUgY3VycmVudCB0cmFuc2Zvcm0uCi0gICAgICogCi0gICAgICogQHBhcmFtIG1vZFRyYW5zZm9ybQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyB0cmFuc2Zvcm0gd2hpY2ggbW9kaWZpZXMgdGhlIGN1cnJlbnQgdHJhbnNmb3JtLgotICAgICAqIEBkZXByZWNhdGVkIHVzZQotICAgICAqICAgICAgICAgICAgIHtAbGluayBSZW5kZXJDb250ZXh0I2NvbmNhdGVuYXRlVHJhbnNmb3JtKEFmZmluZVRyYW5zZm9ybSl9LgotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgcHVibGljIHZvaWQgY29uY2V0ZW5hdGVUcmFuc2Zvcm0oQWZmaW5lVHJhbnNmb3JtIG1vZFRyYW5zZm9ybSkgewotICAgICAgICBjb25jYXRlbmF0ZVRyYW5zZm9ybShtb2RUcmFuc2Zvcm0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbmNhdGVuYXRlIHRoZSBzcGVjaWZpZWQgdHJhbnNmb3JtIHdpdGggdGhlIGN1cnJlbnQgdHJhbnNmb3JtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBtb2RUcmFuc2Zvcm0KLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgdHJhbnNmb3JtIHdoaWNoIG1vZGlmaWVzIHRoZSBjdXJyZW50IHRyYW5zZm9ybS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBjb25jYXRlbmF0ZVRyYW5zZm9ybShBZmZpbmVUcmFuc2Zvcm0gbW9kVHJhbnNmb3JtKSB7Ci0gICAgICAgIHRyYW5zZm9ybS5jb25jYXRlbmF0ZShtb2RUcmFuc2Zvcm0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRyYW5zZm9ybS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB0cmFuc2Zvcm0uCi0gICAgICovCi0gICAgcHVibGljIEFmZmluZVRyYW5zZm9ybSBnZXRUcmFuc2Zvcm0oKSB7Ci0gICAgICAgIHJldHVybiAoQWZmaW5lVHJhbnNmb3JtKXRyYW5zZm9ybS5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGFyZWEgb2YgaW50ZXJlc3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIG5ld0FvaQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBhcmVhIG9mIGludGVyZXN0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEFyZWFPZkludGVyZXN0KFNoYXBlIG5ld0FvaSkgewotICAgICAgICBhb2kgPSBuZXdBb2k7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYXJlYSBvZiBpbnRlcmVzdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcmVhIG9mIGludGVyZXN0LgotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZSBnZXRBcmVhT2ZJbnRlcmVzdCgpIHsKLSAgICAgICAgcmV0dXJuIGFvaTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSByZW5kZXJpbmcgaGludHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGhpbnRzCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHJlbmRlcmluZyBoaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRSZW5kZXJpbmdIaW50cyhSZW5kZXJpbmdIaW50cyBoaW50cykgewotICAgICAgICB0aGlzLmhpbnRzID0gaGludHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcmVuZGVyaW5nIGhpbnRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHJlbmRlcmluZyBoaW50cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVuZGVyaW5nSGludHMgZ2V0UmVuZGVyaW5nSGludHMoKSB7Ci0gICAgICAgIHJldHVybiBoaW50czsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9SZW5kZXJhYmxlSW1hZ2UuamF2YSBiL2F3dC9qYXZhL2F3dC9pbWFnZS9yZW5kZXJhYmxlL1JlbmRlcmFibGVJbWFnZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyMTMzMmY3Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9yZW5kZXJhYmxlL1JlbmRlcmFibGVJbWFnZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTM4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlLnJlbmRlcmFibGU7Ci0KLWltcG9ydCBqYXZhLmF3dC5SZW5kZXJpbmdIaW50czsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5SZW5kZXJlZEltYWdlOwotaW1wb3J0IGphdmEudXRpbC5WZWN0b3I7Ci0KLS8qKgotICogVGhlIEludGVyZmFjZSBSZW5kZXJhYmxlSW1hZ2UgaXMgaW1wbGVtZW50ZWQgYnkgYW4gb2JqZWN0IHRoYXQgY29sbGVjdHMgYWxsCi0gKiBvZiB0aGUgaW1hZ2Utc3BlY2lmaWMgZGF0YSB0aGF0IGRlZmluZXMgYSBzaW5nbGUgaW1hZ2UgdGhhdCBjb3VsZCBiZSByZW5kZXJlZAotICogdG8gZGlmZmVyZW50IHJlbmRlcmluZyB0YXJnZXRzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBSZW5kZXJhYmxlSW1hZ2UgewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEhJTlRTX09CU0VSVkVEIGluZGljYXRlcyB0aGF0IHRoZSByZW5kZXJpbmcgaGludHMgYXJlCi0gICAgICogYXBwbGllZCByYXRoZXIgdGhhbiBpZ25vcmVkLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIEhJTlRTX09CU0VSVkVEID0gIkhJTlRTX09CU0VSVkVEIjsgLy8kTk9OLU5MUy0xJAotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcHJvcGVydHkgZnJvbSB0aGUgUmVuZGVyYWJsZUltYWdlJ3MgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiB0aGUgcHJvcGVydHkgdG8gZ2V0LgotICAgICAqIEByZXR1cm4gdGhlIHZhbHVlIG9mIHRoZSBwcm9wZXJ0eS4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IGdldFByb3BlcnR5KFN0cmluZyBuYW1lKTsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIHJlbmRlcmVkIGltYWdlIGJhc2VkIG9uIHRoZSBpbmZvcm1hdGlvbiBjb250YWluZWQgaW4gdGhlCi0gICAgICogcGFyYW1ldGVycyBhbmQgdGhlIHJlbmRlciBjb250ZXh0LgotICAgICAqIAotICAgICAqIEBwYXJhbSByZW5kZXJDb250ZXh0Ci0gICAgICogICAgICAgICAgICB0aGUgcmVuZGVyIGNvbnRleHQgZ2l2aW5nIHJlbmRlcmluZyBzcGVjaWZpY2F0aW9ucyBzdWNoIGFzCi0gICAgICogICAgICAgICAgICB0cmFuc2Zvcm1hdGlvbnMuCi0gICAgICogQHJldHVybiB0aGUgcmVuZGVyZWQgaW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIFJlbmRlcmVkSW1hZ2UgY3JlYXRlUmVuZGVyaW5nKFJlbmRlckNvbnRleHQgcmVuZGVyQ29udGV4dCk7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBzY2FsZWQgcmVuZGVyZWQgaW1hZ2UgYmFzZWQgb24gdGhlIGluZm9ybWF0aW9uIGNvbnRhaW5lZCBpbgotICAgICAqIHRoZSBwYXJhbWV0ZXJzIGFuZCB0aGUgcmVuZGVyIGNvbnRleHQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHcKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXNpcmVkIHdpZHRoIGFmdGVyIHNjYWxpbmcgb3IgemVybyBpZiB0aGUgc2NhbGluZyBzaG91bGQKLSAgICAgKiAgICAgICAgICAgIGJlIHByb3BvcnRpb25hbCwgYmFzZWQgb24gdGhlIGhlaWdodC4KLSAgICAgKiBAcGFyYW0gaAotICAgICAqICAgICAgICAgICAgdGhlIGRlc2lyZWQgaGVpZ2h0IGFmdGVyIHNjYWxpbmcgb3IgemVybyBpZiB0aGUgc2NhbGluZyBzaG91bGQKLSAgICAgKiAgICAgICAgICAgIGJlIHByb3BvcnRpb25hbCwgYmFzZWQgb24gdGhlIHdpZHRoLgotICAgICAqIEBwYXJhbSBoaW50cwotICAgICAqICAgICAgICAgICAgdGhlIHJlbmRlcmluZyBoaW50cyB0byB1c2UuCi0gICAgICogQHJldHVybiB0aGUgcmVuZGVyZWQgaW1hZ2UuCi0gICAgICogQHRocm93cyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBib3RoIHRoZSBoZWlnaHQgYW5kIHdpZHRoIGFyZSB6ZXJvLgotICAgICAqLwotICAgIHB1YmxpYyBSZW5kZXJlZEltYWdlIGNyZWF0ZVNjYWxlZFJlbmRlcmluZyhpbnQgdywgaW50IGgsIFJlbmRlcmluZ0hpbnRzIGhpbnRzKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHZlY3RvciBvZiBzb3VyY2VzIGZyb20gdGhlIHBhcmFtZXRlciBibG9jay4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzb3VyY2VzLgotICAgICAqLwotICAgIHB1YmxpYyBWZWN0b3I8UmVuZGVyYWJsZUltYWdlPiBnZXRTb3VyY2VzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBuYW1lcyBvZiBhbGwgb2YgdGhlIHN1cHBvcnRlZCBwcm9wZXJ0aWVzIGluIHRoZSBjdXJyZW50IGNvbnRleHQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgcHJvcGVydHkgbmFtZXMuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZ1tdIGdldFByb3BlcnR5TmFtZXMoKTsKLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIGRlZmF1bHQgcmVuZGVyaW5nICh1c2luZyB0aGUgaWRlbnRpdHkgdHJhbnNmb3JtIGFuZCBkZWZhdWx0Ci0gICAgICogcmVuZGVyIGNvbnRleHQpLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHJlbmRlcmVkIGltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBSZW5kZXJlZEltYWdlIGNyZWF0ZURlZmF1bHRSZW5kZXJpbmcoKTsKLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGlzIGNvbnRleHQgc3VwcG9ydHMgZHluYW1pYyByZW5kZXJpbmcuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGlzIGNvbnRleHQgc3VwcG9ydHMgZHluYW1pYyByZW5kZXJpbmcuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNEeW5hbWljKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB3aWR0aCBvZiB0aGUgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgd2lkdGggb2YgdGhlIGltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRXaWR0aCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgeSBjb29yZGluYXRlIG9mIHRoZSB1cHBlciBsZWZ0IGNvcm5lci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB5IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRNaW5ZKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldE1pblgoKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGhlaWdodCBvZiB0aGUgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaGVpZ2h0IG9mIHRoZSBpbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0SGVpZ2h0KCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9yZW5kZXJhYmxlL1JlbmRlcmFibGVJbWFnZU9wLmphdmEgYi9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9SZW5kZXJhYmxlSW1hZ2VPcC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkYzQ1MzcyLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9pbWFnZS9yZW5kZXJhYmxlL1JlbmRlcmFibGVJbWFnZU9wLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxOTEgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2UucmVuZGVyYWJsZTsKLQotaW1wb3J0IGphdmEuYXd0LlJlbmRlcmluZ0hpbnRzOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmVuZGVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLnV0aWwuVmVjdG9yOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhlIENsYXNzIFJlbmRlcmFibGVJbWFnZU9wIGlzIGEgYmFzaWMgaW1wbGVtZW50YXRpb24gb2YgUmVuZGVyYWJsZUltYWdlLAotICogd2l0aCBtZXRob2RzIHRvIGFjY2VzcyB0aGUgcGFyYW1ldGVyIGRhdGEgYW5kIHBlcmZvcm0gcmVuZGVyaW5nIG9wZXJhdGlvbnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgUmVuZGVyYWJsZUltYWdlT3AgaW1wbGVtZW50cyBSZW5kZXJhYmxlSW1hZ2UgewotCi0gICAgLyoqCi0gICAgICogVGhlIENSSUYuCi0gICAgICovCi0gICAgQ29udGV4dHVhbFJlbmRlcmVkSW1hZ2VGYWN0b3J5IENSSUY7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcGFyYW0gYmxvY2suCi0gICAgICovCi0gICAgUGFyYW1ldGVyQmxvY2sgcGFyYW1CbG9jazsKLQotICAgIC8qKgotICAgICAqIFRoZSBoZWlnaHQuCi0gICAgICovCi0gICAgZmxvYXQgbWluWCwgbWluWSwgd2lkdGgsIGhlaWdodDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyByZW5kZXJhYmxlIGltYWdlIG9wLgotICAgICAqIAotICAgICAqIEBwYXJhbSBDUklGCi0gICAgICogICAgICAgICAgICB0aGUgY1JJRi4KLSAgICAgKiBAcGFyYW0gcGFyYW1CbG9jawotICAgICAqICAgICAgICAgICAgdGhlIHBhcmFtIGJsb2NrLgotICAgICAqLwotICAgIHB1YmxpYyBSZW5kZXJhYmxlSW1hZ2VPcChDb250ZXh0dWFsUmVuZGVyZWRJbWFnZUZhY3RvcnkgQ1JJRiwgUGFyYW1ldGVyQmxvY2sgcGFyYW1CbG9jaykgewotICAgICAgICB0aGlzLkNSSUYgPSBDUklGOwotICAgICAgICB0aGlzLnBhcmFtQmxvY2sgPSAoUGFyYW1ldGVyQmxvY2spcGFyYW1CbG9jay5jbG9uZSgpOwotICAgICAgICBSZWN0YW5nbGUyRCByID0gQ1JJRi5nZXRCb3VuZHMyRChwYXJhbUJsb2NrKTsKLSAgICAgICAgbWluWCA9IChmbG9hdClyLmdldE1pblgoKTsKLSAgICAgICAgbWluWSA9IChmbG9hdClyLmdldE1pblkoKTsKLSAgICAgICAgd2lkdGggPSAoZmxvYXQpci5nZXRXaWR0aCgpOwotICAgICAgICBoZWlnaHQgPSAoZmxvYXQpci5nZXRIZWlnaHQoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgT2JqZWN0IGdldFByb3BlcnR5KFN0cmluZyBuYW1lKSB7Ci0gICAgICAgIHJldHVybiBDUklGLmdldFByb3BlcnR5KHBhcmFtQmxvY2ssIG5hbWUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHBhcmFtZXRlciBibG9jay4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcGFyYW1CbG9jawotICAgICAqICAgICAgICAgICAgdGhlIHBhcmFtIGJsb2NrLgotICAgICAqIEByZXR1cm4gdGhlIHBhcmFtZXRlciBibG9jay4KLSAgICAgKi8KLSAgICBwdWJsaWMgUGFyYW1ldGVyQmxvY2sgc2V0UGFyYW1ldGVyQmxvY2soUGFyYW1ldGVyQmxvY2sgcGFyYW1CbG9jaykgewotICAgICAgICBQYXJhbWV0ZXJCbG9jayBvbGRQYXJhbSA9IHRoaXMucGFyYW1CbG9jazsKLSAgICAgICAgdGhpcy5wYXJhbUJsb2NrID0gKFBhcmFtZXRlckJsb2NrKXBhcmFtQmxvY2suY2xvbmUoKTsKLSAgICAgICAgcmV0dXJuIG9sZFBhcmFtOwotICAgIH0KLQotICAgIHB1YmxpYyBSZW5kZXJlZEltYWdlIGNyZWF0ZVJlbmRlcmluZyhSZW5kZXJDb250ZXh0IHJlbmRlckNvbnRleHQpIHsKLQotICAgICAgICBWZWN0b3I8UmVuZGVyYWJsZUltYWdlPiBzb3VyY2VzID0gZ2V0U291cmNlcygpOwotICAgICAgICBQYXJhbWV0ZXJCbG9jayByZFBhcmFtID0gKFBhcmFtZXRlckJsb2NrKXBhcmFtQmxvY2suY2xvbmUoKTsKLQotICAgICAgICBpZiAoc291cmNlcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICBWZWN0b3I8T2JqZWN0PiByZFNvdXJjZXMgPSBuZXcgVmVjdG9yPE9iamVjdD4oKTsKLSAgICAgICAgICAgIGludCBpID0gMDsKLSAgICAgICAgICAgIHdoaWxlIChpIDwgc291cmNlcy5zaXplKCkpIHsKLSAgICAgICAgICAgICAgICBSZW5kZXJDb250ZXh0IG5ld0NvbnRleHQgPSBDUklGCi0gICAgICAgICAgICAgICAgICAgICAgICAubWFwUmVuZGVyQ29udGV4dChpLCByZW5kZXJDb250ZXh0LCBwYXJhbUJsb2NrLCB0aGlzKTsKLSAgICAgICAgICAgICAgICBSZW5kZXJlZEltYWdlIHJkaW0gPSBzb3VyY2VzLmVsZW1lbnRBdChpKS5jcmVhdGVSZW5kZXJpbmcobmV3Q29udGV4dCk7Ci0KLSAgICAgICAgICAgICAgICBpZiAocmRpbSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJkU291cmNlcy5hZGRFbGVtZW50KHJkaW0pOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpKys7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAocmRTb3VyY2VzLnNpemUoKSA+IDApIHsKLSAgICAgICAgICAgICAgICByZFBhcmFtLnNldFNvdXJjZXMocmRTb3VyY2VzKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gQ1JJRi5jcmVhdGUocmVuZGVyQ29udGV4dCwgcmRQYXJhbSk7Ci0gICAgfQotCi0gICAgcHVibGljIFJlbmRlcmVkSW1hZ2UgY3JlYXRlU2NhbGVkUmVuZGVyaW5nKGludCB3LCBpbnQgaCwgUmVuZGVyaW5nSGludHMgaGludHMpIHsKLSAgICAgICAgaWYgKHcgPT0gMCAmJiBoID09IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC42MD1XaWR0aCBhbmQgSGVpZ2h0IG11c3RuJ3QgYmUgZXF1YWwgemVybyBib3RoCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjYwIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKHcgPT0gMCkgewotICAgICAgICAgICAgdyA9IE1hdGgucm91bmQoaCAqIChnZXRXaWR0aCgpIC8gZ2V0SGVpZ2h0KCkpKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChoID09IDApIHsKLSAgICAgICAgICAgIGggPSBNYXRoLnJvdW5kKHcgKiAoZ2V0SGVpZ2h0KCkgLyBnZXRXaWR0aCgpKSk7Ci0gICAgICAgIH0KLQotICAgICAgICBkb3VibGUgc3ggPSAoZG91YmxlKXcgLyBnZXRXaWR0aCgpOwotICAgICAgICBkb3VibGUgc3kgPSAoZG91YmxlKWggLyBnZXRIZWlnaHQoKTsKLQotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gYXQgPSBBZmZpbmVUcmFuc2Zvcm0uZ2V0U2NhbGVJbnN0YW5jZShzeCwgc3kpOwotICAgICAgICBSZW5kZXJDb250ZXh0IGNvbnRleHQgPSBuZXcgUmVuZGVyQ29udGV4dChhdCwgaGludHMpOwotICAgICAgICByZXR1cm4gY3JlYXRlUmVuZGVyaW5nKGNvbnRleHQpOwotICAgIH0KLQotICAgIHB1YmxpYyBWZWN0b3I8UmVuZGVyYWJsZUltYWdlPiBnZXRTb3VyY2VzKCkgewotICAgICAgICBpZiAocGFyYW1CbG9jay5nZXROdW1Tb3VyY2VzKCkgPT0gMCkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLSAgICAgICAgVmVjdG9yPFJlbmRlcmFibGVJbWFnZT4gdiA9IG5ldyBWZWN0b3I8UmVuZGVyYWJsZUltYWdlPigpOwotICAgICAgICBpbnQgaSA9IDA7Ci0gICAgICAgIHdoaWxlIChpIDwgcGFyYW1CbG9jay5nZXROdW1Tb3VyY2VzKCkpIHsKLSAgICAgICAgICAgIE9iamVjdCBvID0gcGFyYW1CbG9jay5nZXRTb3VyY2UoaSk7Ci0gICAgICAgICAgICBpZiAobyBpbnN0YW5jZW9mIFJlbmRlcmFibGVJbWFnZSkgewotICAgICAgICAgICAgICAgIHYuYWRkRWxlbWVudCgoUmVuZGVyYWJsZUltYWdlKW8pOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaSsrOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB2OwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRQcm9wZXJ0eU5hbWVzKCkgewotICAgICAgICByZXR1cm4gQ1JJRi5nZXRQcm9wZXJ0eU5hbWVzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcGFyYW1ldGVyIGJsb2NrLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHBhcmFtZXRlciBibG9jawotICAgICAqLwotICAgIHB1YmxpYyBQYXJhbWV0ZXJCbG9jayBnZXRQYXJhbWV0ZXJCbG9jaygpIHsKLSAgICAgICAgcmV0dXJuIHBhcmFtQmxvY2s7Ci0gICAgfQotCi0gICAgcHVibGljIFJlbmRlcmVkSW1hZ2UgY3JlYXRlRGVmYXVsdFJlbmRlcmluZygpIHsKLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIGF0ID0gbmV3IEFmZmluZVRyYW5zZm9ybSgpOwotICAgICAgICBSZW5kZXJDb250ZXh0IGNvbnRleHQgPSBuZXcgUmVuZGVyQ29udGV4dChhdCk7Ci0gICAgICAgIHJldHVybiBjcmVhdGVSZW5kZXJpbmcoY29udGV4dCk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNEeW5hbWljKCkgewotICAgICAgICByZXR1cm4gQ1JJRi5pc0R5bmFtaWMoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmxvYXQgZ2V0V2lkdGgoKSB7Ci0gICAgICAgIHJldHVybiB3aWR0aDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmxvYXQgZ2V0TWluWSgpIHsKLSAgICAgICAgcmV0dXJuIG1pblk7Ci0gICAgfQotCi0gICAgcHVibGljIGZsb2F0IGdldE1pblgoKSB7Ci0gICAgICAgIHJldHVybiBtaW5YOwotICAgIH0KLQotICAgIHB1YmxpYyBmbG9hdCBnZXRIZWlnaHQoKSB7Ci0gICAgICAgIHJldHVybiBoZWlnaHQ7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9SZW5kZXJhYmxlSW1hZ2VQcm9kdWNlci5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL3JlbmRlcmFibGUvUmVuZGVyYWJsZUltYWdlUHJvZHVjZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZTgzZWJjNy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9SZW5kZXJhYmxlSW1hZ2VQcm9kdWNlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTUxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIGphdmEuYXd0LmltYWdlLnJlbmRlcmFibGU7Ci0KLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkltYWdlQ29uc3VtZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW1hZ2VQcm9kdWNlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5SYXN0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmVuZGVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLnV0aWwuVmVjdG9yOwotCi0vKioKLSAqIFRoZSBDbGFzcyBSZW5kZXJhYmxlSW1hZ2VQcm9kdWNlciBwcm92aWRlcyB0aGUgaW1wbGVtZW50YXRpb24gZm9yIHRoZSBpbWFnZQotICogcmVuZGVyaW5nLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIFJlbmRlcmFibGVJbWFnZVByb2R1Y2VyIGltcGxlbWVudHMgSW1hZ2VQcm9kdWNlciwgUnVubmFibGUgewotCi0gICAgLyoqCi0gICAgICogVGhlIHJibC4KLSAgICAgKi8KLSAgICBSZW5kZXJhYmxlSW1hZ2UgcmJsOwotCi0gICAgLyoqCi0gICAgICogVGhlIHJjLgotICAgICAqLwotICAgIFJlbmRlckNvbnRleHQgcmM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29uc3VtZXJzLgotICAgICAqLwotICAgIFZlY3RvcjxJbWFnZUNvbnN1bWVyPiBjb25zdW1lcnMgPSBuZXcgVmVjdG9yPEltYWdlQ29uc3VtZXI+KCk7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgcmVuZGVyYWJsZSBpbWFnZSBwcm9kdWNlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmRibEltYWdlCi0gICAgICogICAgICAgICAgICB0aGUgcmRibCBpbWFnZS4KLSAgICAgKiBAcGFyYW0gcmMKLSAgICAgKiAgICAgICAgICAgIHRoZSByYy4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVuZGVyYWJsZUltYWdlUHJvZHVjZXIoUmVuZGVyYWJsZUltYWdlIHJkYmxJbWFnZSwgUmVuZGVyQ29udGV4dCByYykgewotICAgICAgICB0aGlzLnJibCA9IHJkYmxJbWFnZTsKLSAgICAgICAgdGhpcy5yYyA9IHJjOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHJlbmRlciBjb250ZXh0LgotICAgICAqIAotICAgICAqIEBwYXJhbSByYwotICAgICAqICAgICAgICAgICAgdGhlIG5ldyByZW5kZXIgY29udGV4dC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgc2V0UmVuZGVyQ29udGV4dChSZW5kZXJDb250ZXh0IHJjKSB7Ci0gICAgICAgIHRoaXMucmMgPSByYzsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIGJvb2xlYW4gaXNDb25zdW1lcihJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIHJldHVybiBjb25zdW1lcnMuY29udGFpbnMoaWMpOwotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCBzdGFydFByb2R1Y3Rpb24oSW1hZ2VDb25zdW1lciBpYykgewotICAgICAgICBhZGRDb25zdW1lcihpYyk7Ci0gICAgICAgIFRocmVhZCB0ID0gbmV3IFRocmVhZCh0aGlzLCAiUmVuZGVyYWJsZUltYWdlUHJvZHVjZXIgdGhyZWFkIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgdC5zdGFydCgpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlcXVlc3RUb3BEb3duTGVmdFJpZ2h0UmVzZW5kKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgcmVtb3ZlQ29uc3VtZXIoSW1hZ2VDb25zdW1lciBpYykgewotICAgICAgICBpZiAoaWMgIT0gbnVsbCkgewotICAgICAgICAgICAgY29uc3VtZXJzLnJlbW92ZUVsZW1lbnQoaWMpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIHN5bmNocm9uaXplZCB2b2lkIGFkZENvbnN1bWVyKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgaWYgKGljICE9IG51bGwgJiYgIWNvbnN1bWVycy5jb250YWlucyhpYykpIHsKLSAgICAgICAgICAgIGNvbnN1bWVycy5hZGRFbGVtZW50KGljKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIHJlbmRlcmVkIGltYWdlIGluIGEgbmV3IHRocmVhZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBydW4oKSB7Ci0gICAgICAgIGlmIChyYmwgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgUmVuZGVyZWRJbWFnZSByZDsKLSAgICAgICAgaWYgKHJjICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJkID0gcmJsLmNyZWF0ZVJlbmRlcmluZyhyYyk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICByZCA9IHJibC5jcmVhdGVEZWZhdWx0UmVuZGVyaW5nKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBDb2xvck1vZGVsIGNtID0gcmQuZ2V0Q29sb3JNb2RlbCgpOwotICAgICAgICBpZiAoY20gPT0gbnVsbCkgewotICAgICAgICAgICAgY20gPSBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIFJhc3RlciByID0gcmQuZ2V0RGF0YSgpOwotICAgICAgICBpbnQgdyA9IHIuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IGggPSByLmdldEhlaWdodCgpOwotCi0gICAgICAgIGZvciAoSW1hZ2VDb25zdW1lciBjIDogY29uc3VtZXJzKSB7Ci0gICAgICAgICAgICBjLnNldERpbWVuc2lvbnModywgaCk7Ci0gICAgICAgICAgICBjLnNldEhpbnRzKEltYWdlQ29uc3VtZXIuVE9QRE9XTkxFRlRSSUdIVCB8IEltYWdlQ29uc3VtZXIuQ09NUExFVEVTQ0FOTElORVMKLSAgICAgICAgICAgICAgICAgICAgfCBJbWFnZUNvbnN1bWVyLlNJTkdMRUZSQU1FIHwgSW1hZ2VDb25zdW1lci5TSU5HTEVQQVNTKTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBzY2FuTGluZVtdID0gbmV3IGludFt3XTsKLSAgICAgICAgaW50IHBpeGVsW10gPSBudWxsOwotCi0gICAgICAgIGZvciAoaW50IHkgPSAwOyB5IDwgaDsgeSsrKSB7Ci0gICAgICAgICAgICBmb3IgKGludCB4ID0gMDsgeCA8IHc7IHgrKykgewotICAgICAgICAgICAgICAgIHBpeGVsID0gci5nZXRQaXhlbCh4LCB5LCBwaXhlbCk7Ci0gICAgICAgICAgICAgICAgc2NhbkxpbmVbeF0gPSBjbS5nZXREYXRhRWxlbWVudChwaXhlbCwgMCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGZvciAoSW1hZ2VDb25zdW1lciBjIDogY29uc3VtZXJzKSB7Ci0gICAgICAgICAgICAgICAgYy5zZXRQaXhlbHMoMCwgeSwgdywgMSwgY20sIHNjYW5MaW5lLCAwLCB3KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoSW1hZ2VDb25zdW1lciBjIDogY29uc3VtZXJzKSB7Ci0gICAgICAgICAgICBjLmltYWdlQ29tcGxldGUoSW1hZ2VDb25zdW1lci5TVEFUSUNJTUFHRURPTkUpOwotICAgICAgICB9Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9SZW5kZXJlZEltYWdlRmFjdG9yeS5qYXZhIGIvYXd0L2phdmEvYXd0L2ltYWdlL3JlbmRlcmFibGUvUmVuZGVyZWRJbWFnZUZhY3RvcnkuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggODgxYTQwYS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9SZW5kZXJlZEltYWdlRmFjdG9yeS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2UgamF2YS5hd3QuaW1hZ2UucmVuZGVyYWJsZTsKLQotaW1wb3J0IGphdmEuYXd0LlJlbmRlcmluZ0hpbnRzOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJlbmRlcmVkSW1hZ2U7Ci0KLS8qKgotICogQSBmYWN0b3J5IGZvciBjcmVhdGluZyBSZW5kZXJlZEltYWdlIG9iamVjdHMgYmFzZWQgb24gcGFyYW1ldGVycyBhbmQKLSAqIHJlbmRlcmluZyBoaW50cy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgUmVuZGVyZWRJbWFnZUZhY3RvcnkgewotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgcmVuZGVyZWQgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGEwCi0gICAgICogICAgICAgICAgICB0aGUgUGFyYW1ldGVyQmxvY2suCi0gICAgICogQHBhcmFtIGExCi0gICAgICogICAgICAgICAgICB0aGUgUmVuZGVyaW5nSGludHMuCi0gICAgICogQHJldHVybiB0aGUgcmVuZGVyZWQgaW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIFJlbmRlcmVkSW1hZ2UgY3JlYXRlKFBhcmFtZXRlckJsb2NrIGEwLCBSZW5kZXJpbmdIaW50cyBhMSk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9pbWFnZS9yZW5kZXJhYmxlL3BhY2thZ2UuaHRtbCBiL2F3dC9qYXZhL2F3dC9pbWFnZS9yZW5kZXJhYmxlL3BhY2thZ2UuaHRtbApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDNhYWFiYy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvaW1hZ2UvcmVuZGVyYWJsZS9wYWNrYWdlLmh0bWwKKysrIC9kZXYvbnVsbApAQCAtMSw4ICswLDAgQEAKLTxodG1sPgotICA8Ym9keT4KLSAgICA8cD4KLSAgICAgIFRoaXMgcGFja2FnZSBjb250YWlucyBjbGFzc2VzIHRvIGNyZWF0ZSBpbWFnZXMgd2hpY2ggYXJlIHJlbmRlcmluZy1pbmRlcGVuZGVudC4KLSAgICA8L3A+Ci0gICAgQHNpbmNlIEFuZHJvaWQgMS4wCi0gIDwvYm9keT4KLTwvaHRtbD4KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wYWNrYWdlLmh0bWwgYi9hd3QvamF2YS9hd3QvcGFja2FnZS5odG1sCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1YTZmOWYwLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wYWNrYWdlLmh0bWwKKysrIC9kZXYvbnVsbApAQCAtMSw4ICswLDAgQEAKLTxodG1sPgotICA8Ym9keT4KLSAgICA8cD4KLSAgICAgIFRoaXMgcGFja2FnZSBjb250YWlucyBjbGFzc2VzIGFuZCBpbnRlcmZhY2VzIGZvciBjcmVhdGluZyAoZ3JhcGhpY2FsKSB1c2VyIGludGVyZmFjZXMgKEdVSSksIHBhaW50aW5nIDJEIGdyYXBoaWNzIGFuZCBjcmVhdGluZywgbWFuaXB1bGF0aW5nIGFuZCBkcmF3aW5nIGltYWdlcy4gCi0gICAgPC9wPgotICBAc2luY2UgQW5kcm9pZCAxLjAKLSAgPC9ib2R5PgotPC9odG1sPgpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L3BlZXIvQnV0dG9uUGVlci5qYXZhIGIvYXd0L2phdmEvYXd0L3BlZXIvQnV0dG9uUGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjYzQ1YjQ5Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL0J1dHRvblBlZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QucGVlcjsKLQotcHVibGljIGludGVyZmFjZSBCdXR0b25QZWVyIHsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L3BlZXIvQ2FudmFzUGVlci5qYXZhIGIvYXd0L2phdmEvYXd0L3BlZXIvQ2FudmFzUGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlMjc2MzY2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL0NhbnZhc1BlZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QucGVlcjsKLQotcHVibGljIGludGVyZmFjZSBDYW52YXNQZWVyIHsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L3BlZXIvQ2hlY2tib3hNZW51SXRlbVBlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL0NoZWNrYm94TWVudUl0ZW1QZWVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI5NmY0MjIuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L3BlZXIvQ2hlY2tib3hNZW51SXRlbVBlZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QucGVlcjsKLQotcHVibGljIGludGVyZmFjZSBDaGVja2JveE1lbnVJdGVtUGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wZWVyL0NoZWNrYm94UGVlci5qYXZhIGIvYXd0L2phdmEvYXd0L3BlZXIvQ2hlY2tib3hQZWVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGU5ZjhkZDEuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L3BlZXIvQ2hlY2tib3hQZWVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LnBlZXI7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgQ2hlY2tib3hQZWVyIHsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L3BlZXIvQ2hvaWNlUGVlci5qYXZhIGIvYXd0L2phdmEvYXd0L3BlZXIvQ2hvaWNlUGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1N2I3NjI5Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL0Nob2ljZVBlZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QucGVlcjsKLQotcHVibGljIGludGVyZmFjZSBDaG9pY2VQZWVyIHsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L3BlZXIvQ29tcG9uZW50UGVlci5qYXZhIGIvYXd0L2phdmEvYXd0L3BlZXIvQ29tcG9uZW50UGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiYzI2NzkxLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL0NvbXBvbmVudFBlZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QucGVlcjsKLQotcHVibGljIGludGVyZmFjZSBDb21wb25lbnRQZWVyIHsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L3BlZXIvRGlhbG9nUGVlci5qYXZhIGIvYXd0L2phdmEvYXd0L3BlZXIvRGlhbG9nUGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4YWUzMDQ5Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL0RpYWxvZ1BlZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QucGVlcjsKLQotcHVibGljIGludGVyZmFjZSBEaWFsb2dQZWVyIHsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L3BlZXIvRmlsZURpYWxvZ1BlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL0ZpbGVEaWFsb2dQZWVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBkMTVlNDguLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L3BlZXIvRmlsZURpYWxvZ1BlZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QucGVlcjsKLQotcHVibGljIGludGVyZmFjZSBGaWxlRGlhbG9nUGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wZWVyL0ZvbnRQZWVyLmphdmEgYi9hd3QvamF2YS9hd3QvcGVlci9Gb250UGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmZDk4MTVmLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL0ZvbnRQZWVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LnBlZXI7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgRm9udFBlZXIgewotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvcGVlci9GcmFtZVBlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL0ZyYW1lUGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5Y2ZjNDBiLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL0ZyYW1lUGVlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5wZWVyOwotCi1wdWJsaWMgaW50ZXJmYWNlIEZyYW1lUGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wZWVyL0xhYmVsUGVlci5qYXZhIGIvYXd0L2phdmEvYXd0L3BlZXIvTGFiZWxQZWVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDA1MmNhOWQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L3BlZXIvTGFiZWxQZWVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LnBlZXI7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgTGFiZWxQZWVyIHsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L3BlZXIvTGlnaHR3ZWlnaHRQZWVyLmphdmEgYi9hd3QvamF2YS9hd3QvcGVlci9MaWdodHdlaWdodFBlZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMWRlZTkwNS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvcGVlci9MaWdodHdlaWdodFBlZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QucGVlcjsKLQotcHVibGljIGludGVyZmFjZSBMaWdodHdlaWdodFBlZXIgewotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvcGVlci9MaXN0UGVlci5qYXZhIGIvYXd0L2phdmEvYXd0L3BlZXIvTGlzdFBlZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMGEyNzg4NS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvcGVlci9MaXN0UGVlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5wZWVyOwotCi1wdWJsaWMgaW50ZXJmYWNlIExpc3RQZWVyIHsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYXd0L3BlZXIvTWVudUJhclBlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL01lbnVCYXJQZWVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDNhZDJjMTYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L3BlZXIvTWVudUJhclBlZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2UgamF2YS5hd3QucGVlcjsKLQotcHVibGljIGludGVyZmFjZSBNZW51QmFyUGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wZWVyL01lbnVDb21wb25lbnRQZWVyLmphdmEgYi9hd3QvamF2YS9hd3QvcGVlci9NZW51Q29tcG9uZW50UGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzYWMzYjM0Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL01lbnVDb21wb25lbnRQZWVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LnBlZXI7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgTWVudUNvbXBvbmVudFBlZXIgewotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvcGVlci9NZW51SXRlbVBlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL01lbnVJdGVtUGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiMTMzODk3Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL01lbnVJdGVtUGVlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5wZWVyOwotCi1wdWJsaWMgaW50ZXJmYWNlIE1lbnVJdGVtUGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wZWVyL01lbnVQZWVyLmphdmEgYi9hd3QvamF2YS9hd3QvcGVlci9NZW51UGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkNjQzY2U3Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL01lbnVQZWVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LnBlZXI7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgTWVudVBlZXIgewotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvcGVlci9Nb3VzZUluZm9QZWVyLmphdmEgYi9hd3QvamF2YS9hd3QvcGVlci9Nb3VzZUluZm9QZWVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDkxNzNhNjIuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L3BlZXIvTW91c2VJbmZvUGVlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5wZWVyOwotCi1wdWJsaWMgaW50ZXJmYWNlIE1vdXNlSW5mb1BlZXIgewotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvcGVlci9QYW5lbFBlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL1BhbmVsUGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxZmFhMWZlLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL1BhbmVsUGVlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5wZWVyOwotCi1wdWJsaWMgaW50ZXJmYWNlIFBhbmVsUGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wZWVyL1BvcHVwTWVudVBlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL1BvcHVwTWVudVBlZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggY2YxZWY2MS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvcGVlci9Qb3B1cE1lbnVQZWVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LnBlZXI7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgUG9wdXBNZW51UGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wZWVyL1Njcm9sbFBhbmVQZWVyLmphdmEgYi9hd3QvamF2YS9hd3QvcGVlci9TY3JvbGxQYW5lUGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkZjNkZTgzLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL1Njcm9sbFBhbmVQZWVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LnBlZXI7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgU2Nyb2xsUGFuZVBlZXIgewotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvcGVlci9TY3JvbGxiYXJQZWVyLmphdmEgYi9hd3QvamF2YS9hd3QvcGVlci9TY3JvbGxiYXJQZWVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGVlYzg5NjEuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYXd0L3BlZXIvU2Nyb2xsYmFyUGVlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5wZWVyOwotCi1wdWJsaWMgaW50ZXJmYWNlIFNjcm9sbGJhclBlZXIgewotCi19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9hd3QvcGVlci9UZXh0QXJlYVBlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL1RleHRBcmVhUGVlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2MzY3MDdmLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2F3dC9wZWVyL1RleHRBcmVhUGVlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBqYXZhLmF3dC5wZWVyOwotCi1wdWJsaWMgaW50ZXJmYWNlIFRleHRBcmVhUGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wZWVyL1RleHRGaWVsZFBlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL1RleHRGaWVsZFBlZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMmI4MjMyYS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvcGVlci9UZXh0RmllbGRQZWVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LnBlZXI7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgVGV4dEZpZWxkUGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2F3dC9wZWVyL1dpbmRvd1BlZXIuamF2YSBiL2F3dC9qYXZhL2F3dC9wZWVyL1dpbmRvd1BlZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzg0NjQ2Zi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9hd3QvcGVlci9XaW5kb3dQZWVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIGphdmEuYXd0LnBlZXI7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgV2luZG93UGVlciB7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2JlYW5zL0ZlYXR1cmVEZXNjcmlwdG9yLmphdmEgYi9hd3QvamF2YS9iZWFucy9GZWF0dXJlRGVzY3JpcHRvci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyOTQ1YzY1Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2JlYW5zL0ZlYXR1cmVEZXNjcmlwdG9yLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMzQgKzAsMCBAQAotLyoKLSAqIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKiAKLSAqIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICogCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCBXSVRIT1VUCi0gKiBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIFNlZSB0aGUKLSAqIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIGxpbWl0YXRpb25zIHVuZGVyCi0gKiB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmEuYmVhbnM7Ci0KLWltcG9ydCBqYXZhLnV0aWwuQ29sbGVjdGlvbnM7Ci1pbXBvcnQgamF2YS51dGlsLkVudW1lcmF0aW9uOwotaW1wb3J0IGphdmEudXRpbC5IYXNoTWFwOwotaW1wb3J0IGphdmEudXRpbC5MaW5rZWRMaXN0OwotaW1wb3J0IGphdmEudXRpbC5NYXA7Ci0KLS8qKgotICogQ29tbW9uIGJhc2UgY2xhc3MgZm9yIERlc2NyaXB0b3JzLgotICovCi1wdWJsaWMgY2xhc3MgRmVhdHVyZURlc2NyaXB0b3IgewotCi0gICAgcHJpdmF0ZSBNYXA8U3RyaW5nLCBPYmplY3Q+IHZhbHVlczsKLQotICAgIGJvb2xlYW4gcHJlZmVycmVkLCBoaWRkZW4sIGV4cGVydDsKLQotICAgIFN0cmluZyBzaG9ydERlc2NyaXB0aW9uOwotCi0gICAgU3RyaW5nIG5hbWU7Ci0KLSAgICBTdHJpbmcgZGlzcGxheU5hbWU7Ci0KLSAgICAvKioKLSAgICAgKiA8cD4KLSAgICAgKiBDb25zdHJ1Y3RzIGFuIGluc3RhbmNlLgotICAgICAqIDwvcD4KLSAgICAgKi8KLSAgICBwdWJsaWMgRmVhdHVyZURlc2NyaXB0b3IoKSB7Ci0gICAgICAgIHRoaXMudmFsdWVzID0gbmV3IEhhc2hNYXA8U3RyaW5nLCBPYmplY3Q+KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogPHA+Ci0gICAgICogU2V0cyB0aGUgdmFsdWUgZm9yIHRoZSBuYW1lZCBhdHRyaWJ1dGUuCi0gICAgICogPC9wPgotICAgICAqIAotICAgICAqIEBwYXJhbSBhdHRyaWJ1dGVOYW1lCi0gICAgICogICAgICAgICAgICBUaGUgbmFtZSBvZiB0aGUgYXR0cmlidXRlIHRvIHNldCBhIHZhbHVlIHdpdGguCi0gICAgICogQHBhcmFtIHZhbHVlCi0gICAgICogICAgICAgICAgICBUaGUgdmFsdWUgdG8gc2V0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFZhbHVlKFN0cmluZyBhdHRyaWJ1dGVOYW1lLCBPYmplY3QgdmFsdWUpIHsKLSAgICAgICAgaWYgKGF0dHJpYnV0ZU5hbWUgPT0gbnVsbCB8fCB2YWx1ZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oKTsKLSAgICAgICAgfQotICAgICAgICB2YWx1ZXMucHV0KGF0dHJpYnV0ZU5hbWUsIHZhbHVlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8cD4KLSAgICAgKiBHZXRzIHRoZSB2YWx1ZSBhc3NvY2lhdGVkIHdpdGggdGhlIG5hbWVkIGF0dHJpYnV0ZS4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHBhcmFtIGF0dHJpYnV0ZU5hbWUKLSAgICAgKiAgICAgICAgICAgIFRoZSBuYW1lIG9mIHRoZSBhdHRyaWJ1dGUgdG8gZ2V0IGEgdmFsdWUgZm9yLgotICAgICAqIEByZXR1cm4gVGhlIGF0dHJpYnV0ZSdzIHZhbHVlLgotICAgICAqLwotICAgIHB1YmxpYyBPYmplY3QgZ2V0VmFsdWUoU3RyaW5nIGF0dHJpYnV0ZU5hbWUpIHsKLSAgICAgICAgT2JqZWN0IHJlc3VsdCA9IG51bGw7Ci0gICAgICAgIGlmIChhdHRyaWJ1dGVOYW1lICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJlc3VsdCA9IHZhbHVlcy5nZXQoYXR0cmlidXRlTmFtZSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8cD4KLSAgICAgKiBFbnVtZXJhdGVzIHRoZSBhdHRyaWJ1dGUgbmFtZXMuCi0gICAgICogPC9wPgotICAgICAqIAotICAgICAqIEByZXR1cm4gQW4gaW5zdGFuY2Ugb2Yge0BsaW5rIEVudW1lcmF0aW9ufS4KLSAgICAgKi8KLSAgICBwdWJsaWMgRW51bWVyYXRpb248U3RyaW5nPiBhdHRyaWJ1dGVOYW1lcygpIHsKLSAgICAgICAgLy8gQ3JlYXRlIGEgbmV3IGxpc3QsIHNvIHRoYXQgdGhlIHJlZmVyZW5jZXMgYXJlIGNvcGllZAotICAgICAgICByZXR1cm4gQ29sbGVjdGlvbnMuZW51bWVyYXRpb24obmV3IExpbmtlZExpc3Q8U3RyaW5nPih2YWx1ZXMua2V5U2V0KCkpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8cD4KLSAgICAgKiBTZXRzIHRoZSBzaG9ydCBkZXNjcmlwdGlvbi4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHBhcmFtIHRleHQKLSAgICAgKiAgICAgICAgICAgIFRoZSBkZXNjcmlwdGlvbiB0byBzZXQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0U2hvcnREZXNjcmlwdGlvbihTdHJpbmcgdGV4dCkgewotICAgICAgICB0aGlzLnNob3J0RGVzY3JpcHRpb24gPSB0ZXh0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIDxwPgotICAgICAqIFNldHMgdGhlIG5hbWUuCi0gICAgICogPC9wPgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lCi0gICAgICogICAgICAgICAgICBUaGUgbmFtZSB0byBzZXQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0TmFtZShTdHJpbmcgbmFtZSkgewotICAgICAgICB0aGlzLm5hbWUgPSBuYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIDxwPgotICAgICAqIFNldHMgdGhlIGRpc3BsYXkgbmFtZS4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHBhcmFtIGRpc3BsYXlOYW1lCi0gICAgICogICAgICAgICAgICBUaGUgZGlzcGxheSBuYW1lIHRvIHNldC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXREaXNwbGF5TmFtZShTdHJpbmcgZGlzcGxheU5hbWUpIHsKLSAgICAgICAgdGhpcy5kaXNwbGF5TmFtZSA9IGRpc3BsYXlOYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIDxwPgotICAgICAqIEdldHMgdGhlIHNob3J0IGRlc2NyaXB0aW9uIG9yIHtAbGluayAjZ2V0RGlzcGxheU5hbWUoKX0gaWYgbm90IHNldC4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHJldHVybiBUaGUgZGVzY3JpcHRpb24uCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZyBnZXRTaG9ydERlc2NyaXB0aW9uKCkgewotICAgICAgICByZXR1cm4gc2hvcnREZXNjcmlwdGlvbiA9PSBudWxsID8gZ2V0RGlzcGxheU5hbWUoKSA6IHNob3J0RGVzY3JpcHRpb247Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogPHA+Ci0gICAgICogR2V0cyB0aGUgbmFtZS4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHJldHVybiBUaGUgbmFtZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldE5hbWUoKSB7Ci0gICAgICAgIHJldHVybiBuYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIDxwPgotICAgICAqIEdldHMgdGhlIGRpc3BsYXkgbmFtZSBvciB7QGxpbmsgI2dldE5hbWUoKX0gaWYgbm90IHNldC4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHJldHVybiBUaGUgZGlzcGxheSBuYW1lLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RGlzcGxheU5hbWUoKSB7Ci0gICAgICAgIHJldHVybiBkaXNwbGF5TmFtZSA9PSBudWxsID8gZ2V0TmFtZSgpIDogZGlzcGxheU5hbWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogPHA+Ci0gICAgICogU2V0cyB0aGUgcHJlZmVycmVkIGluZGljYXRvci4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHBhcmFtIHByZWZlcnJlZAotICAgICAqICAgICAgICAgICAgPGNvZGU+dHJ1ZTwvY29kZT4gaWYgcHJlZmVycmVkLCA8Y29kZT5mYWxzZTwvY29kZT4KLSAgICAgKiAgICAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRQcmVmZXJyZWQoYm9vbGVhbiBwcmVmZXJyZWQpIHsKLSAgICAgICAgdGhpcy5wcmVmZXJyZWQgPSBwcmVmZXJyZWQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogPHA+Ci0gICAgICogU2V0cyB0aGUgaGlkZGVuIGluZGljYXRvci4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHBhcmFtIGhpZGRlbgotICAgICAqICAgICAgICAgICAgPGNvZGU+dHJ1ZTwvY29kZT4gaWYgaGlkZGVuLCA8Y29kZT5mYWxzZTwvY29kZT4gb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEhpZGRlbihib29sZWFuIGhpZGRlbikgewotICAgICAgICB0aGlzLmhpZGRlbiA9IGhpZGRlbjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8cD4KLSAgICAgKiBTZXRzIHRoZSBleHBlcnQgaW5kaWNhdG9yLgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXhwZXJ0Ci0gICAgICogICAgICAgICAgICA8Y29kZT50cnVlPC9jb2RlPiBpZiBleHBlcnQsIDxjb2RlPmZhbHNlPC9jb2RlPiBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RXhwZXJ0KGJvb2xlYW4gZXhwZXJ0KSB7Ci0gICAgICAgIHRoaXMuZXhwZXJ0ID0gZXhwZXJ0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIDxwPgotICAgICAqIEluZGljYXRlcyBpZiB0aGlzIGZlYXR1cmUgaXMgcHJlZmVycmVkLgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIDxjb2RlPnRydWU8L2NvZGU+IGlmIHByZWZlcnJlZCwgPGNvZGU+ZmFsc2U8L2NvZGU+IG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1ByZWZlcnJlZCgpIHsKLSAgICAgICAgcmV0dXJuIHByZWZlcnJlZDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8cD4KLSAgICAgKiBJbmRpY2F0ZXMgaWYgdGhpcyBmZWF0dXJlIGlzIGhpZGRlbi4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHJldHVybiA8Y29kZT50cnVlPC9jb2RlPiBpZiBoaWRkZW4sIDxjb2RlPmZhbHNlPC9jb2RlPiBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNIaWRkZW4oKSB7Ci0gICAgICAgIHJldHVybiBoaWRkZW47Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogPHA+Ci0gICAgICogSW5kaWNhdGVzIGlmIHRoaXMgZmVhdHVyZSBpcyBhbiBleHBlcnQgZmVhdHVyZS4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHJldHVybiA8Y29kZT50cnVlPC9jb2RlPiBpZiBoaWRkZW4sIDxjb2RlPmZhbHNlPC9jb2RlPiBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNFeHBlcnQoKSB7Ci0gICAgICAgIHJldHVybiBleHBlcnQ7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYmVhbnMvSW5kZXhlZFByb3BlcnR5RGVzY3JpcHRvci5qYXZhIGIvYXd0L2phdmEvYmVhbnMvSW5kZXhlZFByb3BlcnR5RGVzY3JpcHRvci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyNTY2N2Q5Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2JlYW5zL0luZGV4ZWRQcm9wZXJ0eURlc2NyaXB0b3IuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDIyNyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmEuYmVhbnM7Ci0KLWltcG9ydCBqYXZhLmxhbmcucmVmbGVjdC5NZXRob2Q7Ci1pbXBvcnQgamF2YS5sYW5nLnJlZmxlY3QuTW9kaWZpZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmJlYW5zLmludGVybmFsLm5scy5NZXNzYWdlczsKLQotcHVibGljIGNsYXNzIEluZGV4ZWRQcm9wZXJ0eURlc2NyaXB0b3IgZXh0ZW5kcyBQcm9wZXJ0eURlc2NyaXB0b3IgewotICAgIHByaXZhdGUgTWV0aG9kIGluZGV4ZWRHZXR0ZXI7Ci0KLSAgICBwcml2YXRlIE1ldGhvZCBpbmRleGVkU2V0dGVyOwotCi0gICAgcHVibGljIEluZGV4ZWRQcm9wZXJ0eURlc2NyaXB0b3IoU3RyaW5nIHByb3BlcnR5TmFtZSwgQ2xhc3M8Pz4gYmVhbkNsYXNzLAotICAgICAgICAgICAgU3RyaW5nIGdldHRlck5hbWUsIFN0cmluZyBzZXR0ZXJOYW1lLCBTdHJpbmcgaW5kZXhlZEdldHRlck5hbWUsCi0gICAgICAgICAgICBTdHJpbmcgaW5kZXhlZFNldHRlck5hbWUpIHRocm93cyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uIHsKLSAgICAgICAgc3VwZXIocHJvcGVydHlOYW1lLCBiZWFuQ2xhc3MsIGdldHRlck5hbWUsIHNldHRlck5hbWUpOwotCi0gICAgICAgIC8vIFJJIGJlaGF2ZXMgbGlrZSB0aGlzCi0gICAgICAgIGlmIChpbmRleGVkR2V0dGVyTmFtZSA9PSBudWxsICYmIGluZGV4ZWRTZXR0ZXJOYW1lID09IG51bGwgJiYKLSAgICAgICAgICAgICAgICAoZ2V0dGVyTmFtZSAhPSBudWxsIHx8IHNldHRlck5hbWUgIT0gbnVsbCkpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuNTAiKSk7Ci0gICAgICAgIH0KLSAgICAgICAgc2V0SW5kZXhlZFJlYWRNZXRob2QoYmVhbkNsYXNzLCBpbmRleGVkR2V0dGVyTmFtZSk7Ci0gICAgICAgIHNldEluZGV4ZWRXcml0ZU1ldGhvZChiZWFuQ2xhc3MsIGluZGV4ZWRTZXR0ZXJOYW1lKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgSW5kZXhlZFByb3BlcnR5RGVzY3JpcHRvcihTdHJpbmcgcHJvcGVydHlOYW1lLCBNZXRob2QgZ2V0dGVyLCBNZXRob2Qgc2V0dGVyLAotICAgICAgICAgICAgTWV0aG9kIGluZGV4ZWRHZXR0ZXIsIE1ldGhvZCBpbmRleGVkU2V0dGVyKSB0aHJvd3MgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbiB7Ci0gICAgICAgIHN1cGVyKHByb3BlcnR5TmFtZSwgZ2V0dGVyLCBzZXR0ZXIpOwotICAgICAgICAKLSAgICAgICAgLy8gd2UgbmVlZCB0aGlzIGluIG9yZGVyIHRvIGJlIGNvbXBhdGlibGUgd2l0aCBSSQotICAgICAgICBpZiAoaW5kZXhlZEdldHRlciA9PSBudWxsICYmIGluZGV4ZWRTZXR0ZXIgPT0gbnVsbCAmJgotICAgICAgICAgICAgICAgIChnZXR0ZXIgIT0gbnVsbCB8fCBzZXR0ZXIgIT0gbnVsbCkpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuNTAiKSk7Ci0gICAgICAgIH0KLSAgICAgICAgc2V0SW5kZXhlZFJlYWRNZXRob2QoaW5kZXhlZEdldHRlcik7Ci0gICAgICAgIHNldEluZGV4ZWRXcml0ZU1ldGhvZChpbmRleGVkU2V0dGVyKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgSW5kZXhlZFByb3BlcnR5RGVzY3JpcHRvcihTdHJpbmcgcHJvcGVydHlOYW1lLCBDbGFzczw/PiBiZWFuQ2xhc3MpCi0gICAgICAgICAgICB0aHJvd3MgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbiB7Ci0gICAgICAgIHN1cGVyKHByb3BlcnR5TmFtZSwgYmVhbkNsYXNzLCBudWxsLCBudWxsKTsKLSAgICAgICAgU3RyaW5nIGdldHRlck5hbWU7Ci0gICAgICAgIFN0cmluZyBzZXR0ZXJOYW1lOwotICAgICAgICBTdHJpbmcgaW5kZXhlZEdldHRlck5hbWU7Ci0gICAgICAgIFN0cmluZyBpbmRleGVkU2V0dGVyTmFtZTsKLQotICAgICAgICAvLyBhcnJheSBnZXR0ZXIKLSAgICAgICAgZ2V0dGVyTmFtZSA9IGNyZWF0ZURlZmF1bHRNZXRob2ROYW1lKHByb3BlcnR5TmFtZSwgImdldCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIGlmIChoYXNNZXRob2QoYmVhbkNsYXNzLCBnZXR0ZXJOYW1lKSkgewotICAgICAgICAgICAgc2V0UmVhZE1ldGhvZChiZWFuQ2xhc3MsIGdldHRlck5hbWUpOwotICAgICAgICB9Ci0gICAgICAgIC8vIGFycmF5IHNldHRlcgotICAgICAgICBzZXR0ZXJOYW1lID0gY3JlYXRlRGVmYXVsdE1ldGhvZE5hbWUocHJvcGVydHlOYW1lLCAic2V0Iik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgaWYgKGhhc01ldGhvZChiZWFuQ2xhc3MsIHNldHRlck5hbWUpKSB7Ci0gICAgICAgICAgICBzZXRXcml0ZU1ldGhvZChiZWFuQ2xhc3MsIHNldHRlck5hbWUpOwotICAgICAgICB9Ci0gICAgICAgIC8vIGluZGV4ZWQgZ2V0dGVyCi0gICAgICAgIGluZGV4ZWRHZXR0ZXJOYW1lID0gY3JlYXRlRGVmYXVsdE1ldGhvZE5hbWUocHJvcGVydHlOYW1lLCAiZ2V0Iik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgaWYgKGhhc01ldGhvZChiZWFuQ2xhc3MsIGluZGV4ZWRHZXR0ZXJOYW1lKSkgewotICAgICAgICAgICAgc2V0SW5kZXhlZFJlYWRNZXRob2QoYmVhbkNsYXNzLCBpbmRleGVkR2V0dGVyTmFtZSk7Ci0gICAgICAgIH0KLSAgICAgICAgLy8gaW5kZXhlZCBzZXR0ZXIKLSAgICAgICAgaW5kZXhlZFNldHRlck5hbWUgPSBjcmVhdGVEZWZhdWx0TWV0aG9kTmFtZShwcm9wZXJ0eU5hbWUsICJzZXQiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBpZiAoaGFzTWV0aG9kKGJlYW5DbGFzcywgaW5kZXhlZFNldHRlck5hbWUpKSB7Ci0gICAgICAgICAgICBzZXRJbmRleGVkV3JpdGVNZXRob2QoYmVhbkNsYXNzLCBpbmRleGVkU2V0dGVyTmFtZSk7Ci0gICAgICAgIH0KLSAgICAgICAgLy8gUkkgc2VlbXMgdG8gYmVoYXZlIGEgYml0IGRpZmZlcmVudGx5Ci0gICAgICAgIGlmIChpbmRleGVkR2V0dGVyID09IG51bGwgJiYgaW5kZXhlZFNldHRlciA9PSBudWxsICYmCi0gICAgICAgICAgICAgICAgZ2V0UmVhZE1ldGhvZCgpID09IG51bGwgJiYgZ2V0V3JpdGVNZXRob2QoKSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbigKLSAgICAgICAgICAgICAgICAgICAgTWVzc2FnZXMuZ2V0U3RyaW5nKCJiZWFucy4wMSIsIHByb3BlcnR5TmFtZSkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKGluZGV4ZWRHZXR0ZXIgPT0gbnVsbCAmJiBpbmRleGVkU2V0dGVyID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIG5vdCBhbiBpbmRleGVkIHByb3BlcnR5IGluZGVlZAotICAgICAgICAgICAgdGhyb3cgbmV3IEludHJvc3BlY3Rpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJiZWFucy41MCIpKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldEluZGV4ZWRSZWFkTWV0aG9kKE1ldGhvZCBpbmRleGVkR2V0dGVyKSB0aHJvd3MgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChpbmRleGVkR2V0dGVyICE9IG51bGwpIHsKLSAgICAgICAgICAgIGludCBtb2RpZmllcnMgPSBpbmRleGVkR2V0dGVyLmdldE1vZGlmaWVycygpOwotICAgICAgICAgICAgQ2xhc3M8Pz5bXSBwYXJhbWV0ZXJUeXBlczsKLSAgICAgICAgICAgIENsYXNzPD8+IHJldHVyblR5cGU7Ci0gICAgICAgICAgICBDbGFzczw/PiBpbmRleGVkUHJvcGVydHlUeXBlOwotCi0gICAgICAgICAgICBpZiAoIU1vZGlmaWVyLmlzUHVibGljKG1vZGlmaWVycykpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImJlYW5zLjIxIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwYXJhbWV0ZXJUeXBlcyA9IGluZGV4ZWRHZXR0ZXIuZ2V0UGFyYW1ldGVyVHlwZXMoKTsKLSAgICAgICAgICAgIGlmIChwYXJhbWV0ZXJUeXBlcy5sZW5ndGggIT0gMSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMjIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICghcGFyYW1ldGVyVHlwZXNbMF0uZXF1YWxzKGludC5jbGFzcykpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImJlYW5zLjIzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm5UeXBlID0gaW5kZXhlZEdldHRlci5nZXRSZXR1cm5UeXBlKCk7Ci0gICAgICAgICAgICBpbmRleGVkUHJvcGVydHlUeXBlID0gZ2V0SW5kZXhlZFByb3BlcnR5VHlwZSgpOwotICAgICAgICAgICAgaWYgKChpbmRleGVkUHJvcGVydHlUeXBlICE9IG51bGwpICYmICFyZXR1cm5UeXBlLmVxdWFscyhpbmRleGVkUHJvcGVydHlUeXBlKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMjQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICB0aGlzLmluZGV4ZWRHZXR0ZXIgPSBpbmRleGVkR2V0dGVyOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldEluZGV4ZWRXcml0ZU1ldGhvZChNZXRob2QgaW5kZXhlZFNldHRlcikgdGhyb3dzIEludHJvc3BlY3Rpb25FeGNlcHRpb24gewotICAgICAgICBpZiAoaW5kZXhlZFNldHRlciAhPSBudWxsKSB7Ci0gICAgICAgICAgICBpbnQgbW9kaWZpZXJzID0gaW5kZXhlZFNldHRlci5nZXRNb2RpZmllcnMoKTsKLSAgICAgICAgICAgIENsYXNzPD8+W10gcGFyYW1ldGVyVHlwZXM7Ci0gICAgICAgICAgICBDbGFzczw/PiBmaXJzdFBhcmFtZXRlclR5cGU7Ci0gICAgICAgICAgICBDbGFzczw/PiBzZWNvbmRQYXJhbWV0ZXJUeXBlOwotICAgICAgICAgICAgQ2xhc3M8Pz4gcHJvcFR5cGU7Ci0KLSAgICAgICAgICAgIGlmICghTW9kaWZpZXIuaXNQdWJsaWMobW9kaWZpZXJzKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMjUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHBhcmFtZXRlclR5cGVzID0gaW5kZXhlZFNldHRlci5nZXRQYXJhbWV0ZXJUeXBlcygpOwotICAgICAgICAgICAgaWYgKHBhcmFtZXRlclR5cGVzLmxlbmd0aCAhPSAyKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEludHJvc3BlY3Rpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJiZWFucy4yNiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgZmlyc3RQYXJhbWV0ZXJUeXBlID0gcGFyYW1ldGVyVHlwZXNbMF07Ci0gICAgICAgICAgICBpZiAoIWZpcnN0UGFyYW1ldGVyVHlwZS5lcXVhbHMoaW50LmNsYXNzKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMjciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHNlY29uZFBhcmFtZXRlclR5cGUgPSBwYXJhbWV0ZXJUeXBlc1sxXTsKLSAgICAgICAgICAgIHByb3BUeXBlID0gZ2V0SW5kZXhlZFByb3BlcnR5VHlwZSgpOwotICAgICAgICAgICAgaWYgKHByb3BUeXBlICE9IG51bGwgJiYgIXNlY29uZFBhcmFtZXRlclR5cGUuZXF1YWxzKHByb3BUeXBlKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMjgiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICB0aGlzLmluZGV4ZWRTZXR0ZXIgPSBpbmRleGVkU2V0dGVyOwotICAgIH0KLQotICAgIHB1YmxpYyBNZXRob2QgZ2V0SW5kZXhlZFdyaXRlTWV0aG9kKCkgewotICAgICAgICByZXR1cm4gaW5kZXhlZFNldHRlcjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTWV0aG9kIGdldEluZGV4ZWRSZWFkTWV0aG9kKCkgewotICAgICAgICByZXR1cm4gaW5kZXhlZEdldHRlcjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG9iaikgewotICAgICAgICBib29sZWFuIHJlc3VsdCA9IHN1cGVyLmVxdWFscyhvYmopOwotICAgICAgICAKLSAgICAgICAgaWYgKHJlc3VsdCkgewotICAgICAgICAgICAgSW5kZXhlZFByb3BlcnR5RGVzY3JpcHRvciBwZCA9IChJbmRleGVkUHJvcGVydHlEZXNjcmlwdG9yKSBvYmo7Ci0gICAgCi0gICAgICAgICAgICBpZiAoaW5kZXhlZEdldHRlciAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcmVzdWx0ID0gaW5kZXhlZEdldHRlci5lcXVhbHMocGQuZ2V0SW5kZXhlZFJlYWRNZXRob2QoKSk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKHJlc3VsdCAmJiBpbmRleGVkR2V0dGVyID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICByZXN1bHQgPSBwZC5nZXRJbmRleGVkUmVhZE1ldGhvZCgpID09IG51bGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICBpZiAocmVzdWx0KSB7Ci0gICAgICAgICAgICAgICAgaWYgKGluZGV4ZWRTZXR0ZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICByZXN1bHQgPSBpbmRleGVkU2V0dGVyLmVxdWFscyhwZC5nZXRJbmRleGVkV3JpdGVNZXRob2QoKSk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChpbmRleGVkU2V0dGVyID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgcmVzdWx0ID0gcGQuZ2V0SW5kZXhlZFdyaXRlTWV0aG9kKCkgPT0gbnVsbDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgICAgIAotICAgICAgICByZXR1cm4gcmVzdWx0OwotICAgIH0KLQotICAgIHB1YmxpYyBDbGFzczw/PiBnZXRJbmRleGVkUHJvcGVydHlUeXBlKCkgewotICAgICAgICBDbGFzczw/PiByZXN1bHQgPSBudWxsOwotCi0gICAgICAgIGlmIChpbmRleGVkR2V0dGVyICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJlc3VsdCA9IGluZGV4ZWRHZXR0ZXIuZ2V0UmV0dXJuVHlwZSgpOwotICAgICAgICB9IGVsc2UgaWYgKGluZGV4ZWRTZXR0ZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgQ2xhc3M8Pz5bXSBwYXJhbWV0ZXJUeXBlcyA9IGluZGV4ZWRTZXR0ZXIuZ2V0UGFyYW1ldGVyVHlwZXMoKTsKLQotICAgICAgICAgICAgcmVzdWx0ID0gcGFyYW1ldGVyVHlwZXNbMV07Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICB9Ci0KLSAgICBwcml2YXRlIHZvaWQgc2V0SW5kZXhlZFJlYWRNZXRob2QoQ2xhc3M8Pz4gYmVhbkNsYXNzLCBTdHJpbmcgaW5kZXhlZEdldHRlck5hbWUpIHsKLSAgICAgICAgTWV0aG9kW10gZ2V0dGVycyA9IGZpbmRNZXRob2RzKGJlYW5DbGFzcywgaW5kZXhlZEdldHRlck5hbWUpOwotICAgICAgICBib29sZWFuIHJlc3VsdCA9IGZhbHNlOwotCi0gICAgICAgIGZvciAoTWV0aG9kIGVsZW1lbnQgOiBnZXR0ZXJzKSB7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIHNldEluZGV4ZWRSZWFkTWV0aG9kKGVsZW1lbnQpOwotICAgICAgICAgICAgICAgIHJlc3VsdCA9IHRydWU7Ci0gICAgICAgICAgICB9IGNhdGNoIChJbnRyb3NwZWN0aW9uRXhjZXB0aW9uIGllKSB7fQotCi0gICAgICAgICAgICBpZiAocmVzdWx0KSB7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwcml2YXRlIHZvaWQgc2V0SW5kZXhlZFdyaXRlTWV0aG9kKENsYXNzPD8+IGJlYW5DbGFzcywgU3RyaW5nIGluZGV4ZWRTZXR0ZXJOYW1lKSB7Ci0gICAgICAgIE1ldGhvZFtdIHNldHRlcnMgPSBmaW5kTWV0aG9kcyhiZWFuQ2xhc3MsIGluZGV4ZWRTZXR0ZXJOYW1lKTsKLSAgICAgICAgYm9vbGVhbiByZXN1bHQgPSBmYWxzZTsKLQotICAgICAgICBmb3IgKE1ldGhvZCBlbGVtZW50IDogc2V0dGVycykgewotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICBzZXRJbmRleGVkV3JpdGVNZXRob2QoZWxlbWVudCk7Ci0gICAgICAgICAgICAgICAgcmVzdWx0ID0gdHJ1ZTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKEludHJvc3BlY3Rpb25FeGNlcHRpb24gaWUpIHt9Ci0KLSAgICAgICAgICAgIGlmIChyZXN1bHQpIHsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2JlYW5zL0ludHJvc3BlY3Rpb25FeGNlcHRpb24uamF2YSBiL2F3dC9qYXZhL2JlYW5zL0ludHJvc3BlY3Rpb25FeGNlcHRpb24uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzg5NWFmZS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9iZWFucy9JbnRyb3NwZWN0aW9uRXhjZXB0aW9uLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmEuYmVhbnM7Ci0KLXB1YmxpYyBjbGFzcyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uIGV4dGVuZHMgRXhjZXB0aW9uIHsKLQotICAgIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSAtMzcyODE1MDUzOTk2OTU0MjYxOUw7Ci0KLSAgICBwdWJsaWMgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbihTdHJpbmcgbWVzc2FnZSkgewotICAgICAgICBzdXBlcihtZXNzYWdlKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YS9iZWFucy9Qcm9wZXJ0eURlc2NyaXB0b3IuamF2YSBiL2F3dC9qYXZhL2JlYW5zL1Byb3BlcnR5RGVzY3JpcHRvci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5Mzg5MTUyLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2JlYW5zL1Byb3BlcnR5RGVzY3JpcHRvci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzAwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YS5iZWFuczsKLQotaW1wb3J0IGphdmEubGFuZy5yZWZsZWN0LkNvbnN0cnVjdG9yOwotaW1wb3J0IGphdmEubGFuZy5yZWZsZWN0Lk1ldGhvZDsKLWltcG9ydCBqYXZhLmxhbmcucmVmbGVjdC5Nb2RpZmllcjsKLWltcG9ydCBqYXZhLnV0aWwuVmVjdG9yOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5iZWFucy5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLXB1YmxpYyBjbGFzcyBQcm9wZXJ0eURlc2NyaXB0b3IgZXh0ZW5kcyBGZWF0dXJlRGVzY3JpcHRvciB7Ci0gICAgcHJpdmF0ZSBNZXRob2QgZ2V0dGVyOwotCi0gICAgcHJpdmF0ZSBNZXRob2Qgc2V0dGVyOwotCi0gICAgcHJpdmF0ZSBDbGFzczw/PiBwcm9wZXJ0eUVkaXRvckNsYXNzOwotCi0gICAgcHJpdmF0ZSBib29sZWFuIGNvbnN0cmFpbmVkOwotCi0gICAgcHJpdmF0ZSBib29sZWFuIGJvdW5kOwotCi0gICAgcHVibGljIFByb3BlcnR5RGVzY3JpcHRvcihTdHJpbmcgcHJvcGVydHlOYW1lLCBDbGFzczw/PiBiZWFuQ2xhc3MsIFN0cmluZyBnZXR0ZXJOYW1lLAotICAgICAgICAgICAgU3RyaW5nIHNldHRlck5hbWUpIHRocm93cyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uIHsKLSAgICAgICAgc3VwZXIoKTsKLSAgICAgICAgaWYgKGJlYW5DbGFzcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImJlYW5zLjAzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKHByb3BlcnR5TmFtZSA9PSBudWxsIHx8IHByb3BlcnR5TmFtZS5sZW5ndGgoKSA9PSAwKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImJlYW5zLjA0IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy5zZXROYW1lKHByb3BlcnR5TmFtZSk7Ci0gICAgICAgIHRoaXMuc2V0RGlzcGxheU5hbWUocHJvcGVydHlOYW1lKTsKLSAgICAgICAgaWYgKHNldHRlck5hbWUgIT0gbnVsbCkgewotICAgICAgICAgICAgaWYgKGhhc01ldGhvZChiZWFuQ2xhc3MsIHNldHRlck5hbWUpKSB7Ci0gICAgICAgICAgICAgICAgc2V0V3JpdGVNZXRob2QoYmVhbkNsYXNzLCBzZXR0ZXJOYW1lKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEludHJvc3BlY3Rpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJiZWFucy4yMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGlmIChnZXR0ZXJOYW1lICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChoYXNNZXRob2QoYmVhbkNsYXNzLCBnZXR0ZXJOYW1lKSkgewotICAgICAgICAgICAgICAgIHNldFJlYWRNZXRob2QoYmVhbkNsYXNzLCBnZXR0ZXJOYW1lKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEludHJvc3BlY3Rpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJiZWFucy4xRiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIFByb3BlcnR5RGVzY3JpcHRvcihTdHJpbmcgcHJvcGVydHlOYW1lLCBNZXRob2QgZ2V0dGVyLCBNZXRob2Qgc2V0dGVyKQotICAgICAgICAgICAgdGhyb3dzIEludHJvc3BlY3Rpb25FeGNlcHRpb24gewotICAgICAgICBzdXBlcigpOwotICAgICAgICBpZiAocHJvcGVydHlOYW1lID09IG51bGwgfHwgcHJvcGVydHlOYW1lLmxlbmd0aCgpID09IDApIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMDQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB0aGlzLnNldE5hbWUocHJvcGVydHlOYW1lKTsKLSAgICAgICAgdGhpcy5zZXREaXNwbGF5TmFtZShwcm9wZXJ0eU5hbWUpOwotICAgICAgICBzZXRXcml0ZU1ldGhvZChzZXR0ZXIpOwotICAgICAgICBzZXRSZWFkTWV0aG9kKGdldHRlcik7Ci0gICAgfQotCi0gICAgcHVibGljIFByb3BlcnR5RGVzY3JpcHRvcihTdHJpbmcgcHJvcGVydHlOYW1lLCBDbGFzczw/PiBiZWFuQ2xhc3MpCi0gICAgICAgICAgICB0aHJvd3MgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbiB7Ci0gICAgICAgIFN0cmluZyBnZXR0ZXJOYW1lOwotICAgICAgICBTdHJpbmcgc2V0dGVyTmFtZTsKLSAgICAgICAgaWYgKGJlYW5DbGFzcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImJlYW5zLjAzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaWYgKHByb3BlcnR5TmFtZSA9PSBudWxsIHx8IHByb3BlcnR5TmFtZS5sZW5ndGgoKSA9PSAwKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImJlYW5zLjA0IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy5zZXROYW1lKHByb3BlcnR5TmFtZSk7Ci0gICAgICAgIHRoaXMuc2V0RGlzcGxheU5hbWUocHJvcGVydHlOYW1lKTsKLSAgICAgICAgZ2V0dGVyTmFtZSA9IGNyZWF0ZURlZmF1bHRNZXRob2ROYW1lKHByb3BlcnR5TmFtZSwgImlzIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgaWYgKGhhc01ldGhvZChiZWFuQ2xhc3MsIGdldHRlck5hbWUpKSB7Ci0gICAgICAgICAgICBzZXRSZWFkTWV0aG9kKGJlYW5DbGFzcywgZ2V0dGVyTmFtZSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBnZXR0ZXJOYW1lID0gY3JlYXRlRGVmYXVsdE1ldGhvZE5hbWUocHJvcGVydHlOYW1lLCAiZ2V0Iik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIGlmIChoYXNNZXRob2QoYmVhbkNsYXNzLCBnZXR0ZXJOYW1lKSkgewotICAgICAgICAgICAgICAgIHNldFJlYWRNZXRob2QoYmVhbkNsYXNzLCBnZXR0ZXJOYW1lKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBzZXR0ZXJOYW1lID0gY3JlYXRlRGVmYXVsdE1ldGhvZE5hbWUocHJvcGVydHlOYW1lLCAic2V0Iik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgaWYgKGhhc01ldGhvZChiZWFuQ2xhc3MsIHNldHRlck5hbWUpKSB7Ci0gICAgICAgICAgICBzZXRXcml0ZU1ldGhvZChiZWFuQ2xhc3MsIHNldHRlck5hbWUpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChnZXR0ZXIgPT0gbnVsbCAmJiBzZXR0ZXIgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IEludHJvc3BlY3Rpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJiZWFucy4wMSIsIHByb3BlcnR5TmFtZSkpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRXcml0ZU1ldGhvZChNZXRob2Qgc2V0dGVyKSB0aHJvd3MgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChzZXR0ZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgaW50IG1vZGlmaWVycyA9IHNldHRlci5nZXRNb2RpZmllcnMoKTsKLSAgICAgICAgICAgIGlmICghTW9kaWZpZXIuaXNQdWJsaWMobW9kaWZpZXJzKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMDUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIENsYXNzPD8+W10gcGFyYW1ldGVyVHlwZXMgPSBzZXR0ZXIuZ2V0UGFyYW1ldGVyVHlwZXMoKTsKLSAgICAgICAgICAgIGlmIChwYXJhbWV0ZXJUeXBlcy5sZW5ndGggIT0gMSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMDYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIENsYXNzPD8+IHBhcmFtZXRlclR5cGUgPSBwYXJhbWV0ZXJUeXBlc1swXTsKLSAgICAgICAgICAgIENsYXNzPD8+IHByb3BlcnR5VHlwZSA9IGdldFByb3BlcnR5VHlwZSgpOwotICAgICAgICAgICAgaWYgKHByb3BlcnR5VHlwZSAhPSBudWxsICYmICFwcm9wZXJ0eVR5cGUuZXF1YWxzKHBhcmFtZXRlclR5cGUpKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEludHJvc3BlY3Rpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJiZWFucy4wNyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHRoaXMuc2V0dGVyID0gc2V0dGVyOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldFJlYWRNZXRob2QoTWV0aG9kIGdldHRlcikgdGhyb3dzIEludHJvc3BlY3Rpb25FeGNlcHRpb24gewotICAgICAgICBpZiAoZ2V0dGVyICE9IG51bGwpIHsKLSAgICAgICAgICAgIGludCBtb2RpZmllcnMgPSBnZXR0ZXIuZ2V0TW9kaWZpZXJzKCk7Ci0gICAgICAgICAgICBpZiAoIU1vZGlmaWVyLmlzUHVibGljKG1vZGlmaWVycykpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImJlYW5zLjBBIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBDbGFzczw/PltdIHBhcmFtZXRlclR5cGVzID0gZ2V0dGVyLmdldFBhcmFtZXRlclR5cGVzKCk7Ci0gICAgICAgICAgICBpZiAocGFyYW1ldGVyVHlwZXMubGVuZ3RoICE9IDApIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSW50cm9zcGVjdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImJlYW5zLjA4IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBDbGFzczw/PiByZXR1cm5UeXBlID0gZ2V0dGVyLmdldFJldHVyblR5cGUoKTsKLSAgICAgICAgICAgIGlmIChyZXR1cm5UeXBlLmVxdWFscyhWb2lkLlRZUEUpKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEludHJvc3BlY3Rpb25FeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJiZWFucy4zMyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgQ2xhc3M8Pz4gcHJvcGVydHlUeXBlID0gZ2V0UHJvcGVydHlUeXBlKCk7Ci0gICAgICAgICAgICBpZiAoKHByb3BlcnR5VHlwZSAhPSBudWxsKSAmJiAhcmV0dXJuVHlwZS5lcXVhbHMocHJvcGVydHlUeXBlKSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbnRyb3NwZWN0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMDkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICB0aGlzLmdldHRlciA9IGdldHRlcjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTWV0aG9kIGdldFdyaXRlTWV0aG9kKCkgewotICAgICAgICByZXR1cm4gc2V0dGVyOwotICAgIH0KLQotICAgIHB1YmxpYyBNZXRob2QgZ2V0UmVhZE1ldGhvZCgpIHsKLSAgICAgICAgcmV0dXJuIGdldHRlcjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG9iamVjdCkgewotICAgICAgICBib29sZWFuIHJlc3VsdCA9IChvYmplY3QgIT0gbnVsbCAmJiBvYmplY3QgaW5zdGFuY2VvZiBQcm9wZXJ0eURlc2NyaXB0b3IpOwotICAgICAgICBpZiAocmVzdWx0KSB7Ci0gICAgICAgICAgICBQcm9wZXJ0eURlc2NyaXB0b3IgcGQgPSAoUHJvcGVydHlEZXNjcmlwdG9yKSBvYmplY3Q7Ci0gICAgICAgICAgICBib29sZWFuIGdldHRlcnNBcmVFcXVhbCA9ICh0aGlzLmdldHRlciA9PSBudWxsKSAmJiAocGQuZ2V0UmVhZE1ldGhvZCgpID09IG51bGwpCi0gICAgICAgICAgICAgICAgICAgIHx8ICh0aGlzLmdldHRlciAhPSBudWxsKSAmJiAodGhpcy5nZXR0ZXIuZXF1YWxzKHBkLmdldFJlYWRNZXRob2QoKSkpOwotICAgICAgICAgICAgYm9vbGVhbiBzZXR0ZXJzQXJlRXF1YWwgPSAodGhpcy5zZXR0ZXIgPT0gbnVsbCkgJiYgKHBkLmdldFdyaXRlTWV0aG9kKCkgPT0gbnVsbCkKLSAgICAgICAgICAgICAgICAgICAgfHwgKHRoaXMuc2V0dGVyICE9IG51bGwpICYmICh0aGlzLnNldHRlci5lcXVhbHMocGQuZ2V0V3JpdGVNZXRob2QoKSkpOwotICAgICAgICAgICAgYm9vbGVhbiBwcm9wZXJ0eVR5cGVzQXJlRXF1YWwgPSB0aGlzLmdldFByb3BlcnR5VHlwZSgpID09IHBkLmdldFByb3BlcnR5VHlwZSgpOwotICAgICAgICAgICAgYm9vbGVhbiBwcm9wZXJ0eUVkaXRvckNsYXNzZXNBcmVFcXVhbCA9IHRoaXMuZ2V0UHJvcGVydHlFZGl0b3JDbGFzcygpID09IHBkCi0gICAgICAgICAgICAgICAgICAgIC5nZXRQcm9wZXJ0eUVkaXRvckNsYXNzKCk7Ci0gICAgICAgICAgICBib29sZWFuIGJvdW5kUHJvcGVydHlBcmVFcXVhbCA9IHRoaXMuaXNCb3VuZCgpID09IHBkLmlzQm91bmQoKTsKLSAgICAgICAgICAgIGJvb2xlYW4gY29uc3RyYWluZWRQcm9wZXJ0eUFyZUVxdWFsID0gdGhpcy5pc0NvbnN0cmFpbmVkKCkgPT0gcGQuaXNDb25zdHJhaW5lZCgpOwotICAgICAgICAgICAgcmVzdWx0ID0gZ2V0dGVyc0FyZUVxdWFsICYmIHNldHRlcnNBcmVFcXVhbCAmJiBwcm9wZXJ0eVR5cGVzQXJlRXF1YWwKLSAgICAgICAgICAgICAgICAgICAgJiYgcHJvcGVydHlFZGl0b3JDbGFzc2VzQXJlRXF1YWwgJiYgYm91bmRQcm9wZXJ0eUFyZUVxdWFsCi0gICAgICAgICAgICAgICAgICAgICYmIGNvbnN0cmFpbmVkUHJvcGVydHlBcmVFcXVhbDsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcmVzdWx0OwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldFByb3BlcnR5RWRpdG9yQ2xhc3MoQ2xhc3M8Pz4gcHJvcGVydHlFZGl0b3JDbGFzcykgewotICAgICAgICB0aGlzLnByb3BlcnR5RWRpdG9yQ2xhc3MgPSBwcm9wZXJ0eUVkaXRvckNsYXNzOwotICAgIH0KLQotICAgIHB1YmxpYyBDbGFzczw/PiBnZXRQcm9wZXJ0eVR5cGUoKSB7Ci0gICAgICAgIENsYXNzPD8+IHJlc3VsdCA9IG51bGw7Ci0gICAgICAgIGlmIChnZXR0ZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgcmVzdWx0ID0gZ2V0dGVyLmdldFJldHVyblR5cGUoKTsKLSAgICAgICAgfSBlbHNlIGlmIChzZXR0ZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgQ2xhc3M8Pz5bXSBwYXJhbWV0ZXJUeXBlcyA9IHNldHRlci5nZXRQYXJhbWV0ZXJUeXBlcygpOwotICAgICAgICAgICAgcmVzdWx0ID0gcGFyYW1ldGVyVHlwZXNbMF07Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQ2xhc3M8Pz4gZ2V0UHJvcGVydHlFZGl0b3JDbGFzcygpIHsKLSAgICAgICAgcmV0dXJuIHByb3BlcnR5RWRpdG9yQ2xhc3M7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0Q29uc3RyYWluZWQoYm9vbGVhbiBjb25zdHJhaW5lZCkgewotICAgICAgICB0aGlzLmNvbnN0cmFpbmVkID0gY29uc3RyYWluZWQ7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0Qm91bmQoYm9vbGVhbiBib3VuZCkgewotICAgICAgICB0aGlzLmJvdW5kID0gYm91bmQ7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb25zdHJhaW5lZCgpIHsKLSAgICAgICAgcmV0dXJuIGNvbnN0cmFpbmVkOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGlzQm91bmQoKSB7Ci0gICAgICAgIHJldHVybiBib3VuZDsKLSAgICB9Ci0KLSAgICBib29sZWFuIGhhc01ldGhvZChDbGFzczw/PiBiZWFuQ2xhc3MsIFN0cmluZyBtZXRob2ROYW1lKSB7Ci0gICAgICAgIE1ldGhvZFtdIG1ldGhvZHMgPSBmaW5kTWV0aG9kcyhiZWFuQ2xhc3MsIG1ldGhvZE5hbWUpOwotICAgICAgICByZXR1cm4gKG1ldGhvZHMubGVuZ3RoID4gMCk7Ci0gICAgfQotCi0gICAgU3RyaW5nIGNyZWF0ZURlZmF1bHRNZXRob2ROYW1lKFN0cmluZyBwcm9wZXJ0eU5hbWUsIFN0cmluZyBwcmVmaXgpIHsKLSAgICAgICAgU3RyaW5nIHJlc3VsdCA9IG51bGw7Ci0gICAgICAgIGlmIChwcm9wZXJ0eU5hbWUgIT0gbnVsbCkgewotICAgICAgICAgICAgU3RyaW5nIGJvcyA9IHByb3BlcnR5TmFtZS5zdWJzdHJpbmcoMCwgMSkudG9VcHBlckNhc2UoKTsKLSAgICAgICAgICAgIFN0cmluZyBlb3MgPSBwcm9wZXJ0eU5hbWUuc3Vic3RyaW5nKDEsIHByb3BlcnR5TmFtZS5sZW5ndGgoKSk7Ci0gICAgICAgICAgICByZXN1bHQgPSBwcmVmaXggKyBib3MgKyBlb3M7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICB9Ci0KLSAgICBNZXRob2RbXSBmaW5kTWV0aG9kcyhDbGFzczw/PiBhQ2xhc3MsIFN0cmluZyBtZXRob2ROYW1lKSB7Ci0gICAgICAgIE1ldGhvZFtdIGFsbE1ldGhvZHMgPSBhQ2xhc3MuZ2V0TWV0aG9kcygpOwotICAgICAgICBWZWN0b3I8TWV0aG9kPiBtYXRjaGVkTWV0aG9kcyA9IG5ldyBWZWN0b3I8TWV0aG9kPigpOwotICAgICAgICBNZXRob2RbXSByZXN1bHQ7Ci0gICAgICAgIGZvciAoTWV0aG9kIG1ldGhvZCA6IGFsbE1ldGhvZHMpIHsKLSAgICAgICAgICAgIGlmIChtZXRob2QuZ2V0TmFtZSgpLmVxdWFscyhtZXRob2ROYW1lKSkgewotICAgICAgICAgICAgICAgIG1hdGNoZWRNZXRob2RzLmFkZChtZXRob2QpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJlc3VsdCA9IG5ldyBNZXRob2RbbWF0Y2hlZE1ldGhvZHMuc2l6ZSgpXTsKLSAgICAgICAgZm9yIChpbnQgaiA9IDA7IGogPCBtYXRjaGVkTWV0aG9kcy5zaXplKCk7ICsraikgewotICAgICAgICAgICAgcmVzdWx0W2pdID0gbWF0Y2hlZE1ldGhvZHMuZWxlbWVudEF0KGopOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiByZXN1bHQ7Ci0gICAgfQotCi0gICAgdm9pZCBzZXRSZWFkTWV0aG9kKENsYXNzPD8+IGJlYW5DbGFzcywgU3RyaW5nIGdldHRlck5hbWUpIHsKLSAgICAgICAgYm9vbGVhbiByZXN1bHQgPSBmYWxzZTsKLSAgICAgICAgTWV0aG9kW10gZ2V0dGVycyA9IGZpbmRNZXRob2RzKGJlYW5DbGFzcywgZ2V0dGVyTmFtZSk7Ci0gICAgICAgIGZvciAoTWV0aG9kIGVsZW1lbnQgOiBnZXR0ZXJzKSB7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIHNldFJlYWRNZXRob2QoZWxlbWVudCk7Ci0gICAgICAgICAgICAgICAgcmVzdWx0ID0gdHJ1ZTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKEludHJvc3BlY3Rpb25FeGNlcHRpb24gaWUpIHsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChyZXN1bHQpIHsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHZvaWQgc2V0V3JpdGVNZXRob2QoQ2xhc3M8Pz4gYmVhbkNsYXNzLCBTdHJpbmcgc2V0dGVyTmFtZSkgdGhyb3dzIEludHJvc3BlY3Rpb25FeGNlcHRpb24gewotICAgICAgICBib29sZWFuIHJlc3VsdCA9IGZhbHNlOwotICAgICAgICBNZXRob2RbXSBzZXR0ZXJzID0gZmluZE1ldGhvZHMoYmVhbkNsYXNzLCBzZXR0ZXJOYW1lKTsKLSAgICAgICAgZm9yIChNZXRob2QgZWxlbWVudCA6IHNldHRlcnMpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgc2V0V3JpdGVNZXRob2QoZWxlbWVudCk7Ci0gICAgICAgICAgICAgICAgcmVzdWx0ID0gdHJ1ZTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKEludHJvc3BlY3Rpb25FeGNlcHRpb24gaWUpIHsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChyZXN1bHQpIHsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBQcm9wZXJ0eUVkaXRvciBjcmVhdGVQcm9wZXJ0eUVkaXRvcihPYmplY3QgYmVhbikgewotICAgICAgICBQcm9wZXJ0eUVkaXRvciBlZGl0b3I7Ci0gICAgICAgIGlmIChwcm9wZXJ0eUVkaXRvckNsYXNzID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgICAgIGlmICghUHJvcGVydHlFZGl0b3IuY2xhc3MuaXNBc3NpZ25hYmxlRnJvbShwcm9wZXJ0eUVkaXRvckNsYXNzKSkgewotICAgICAgICAgICAgLy8gYmVhbnMuNDg9UHJvcGVydHkgZWRpdG9yIGlzIG5vdCBhc3NpZ25hYmxlIGZyb20gdGhlCi0gICAgICAgICAgICAvLyBQcm9wZXJ0eUVkaXRvciBpbnRlcmZhY2UKLSAgICAgICAgICAgIHRocm93IG5ldyBDbGFzc0Nhc3RFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJiZWFucy40OCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBDb25zdHJ1Y3Rvcjw/PiBjb25zdHI7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIC8vIHRyeSB0byBsb29rIGZvciB0aGUgY29uc3RydWN0b3Igd2l0aCBzaW5nbGUgT2JqZWN0IGFyZ3VtZW50Ci0gICAgICAgICAgICAgICAgY29uc3RyID0gcHJvcGVydHlFZGl0b3JDbGFzcy5nZXRDb25zdHJ1Y3RvcihPYmplY3QuY2xhc3MpOwotICAgICAgICAgICAgICAgIGVkaXRvciA9IChQcm9wZXJ0eUVkaXRvcikgY29uc3RyLm5ld0luc3RhbmNlKGJlYW4pOwotICAgICAgICAgICAgfSBjYXRjaCAoTm9TdWNoTWV0aG9kRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICAvLyB0cnkgbm8tYXJndW1lbnQgY29uc3RydWN0b3IKLSAgICAgICAgICAgICAgICBjb25zdHIgPSBwcm9wZXJ0eUVkaXRvckNsYXNzLmdldENvbnN0cnVjdG9yKCk7Ci0gICAgICAgICAgICAgICAgZWRpdG9yID0gKFByb3BlcnR5RWRpdG9yKSBjb25zdHIubmV3SW5zdGFuY2UoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIC8vIGJlYW5zLjQ3PVVuYWJsZSB0byBpbnN0YW50aWF0ZSBwcm9wZXJ0eSBlZGl0b3IKLSAgICAgICAgICAgIFJ1bnRpbWVFeGNlcHRpb24gcmUgPSBuZXcgUnVudGltZUV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImJlYW5zLjQ3IiksIGUpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB0aHJvdyByZTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZWRpdG9yOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2JlYW5zL1Byb3BlcnR5RWRpdG9yLmphdmEgYi9hd3QvamF2YS9iZWFucy9Qcm9wZXJ0eUVkaXRvci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2NWJlZGVhLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZhL2JlYW5zL1Byb3BlcnR5RWRpdG9yLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmEuYmVhbnM7Ci0KLWltcG9ydCBqYXZhLmF3dC5Db21wb25lbnQ7Ci1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3M7Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotCi1wdWJsaWMgaW50ZXJmYWNlIFByb3BlcnR5RWRpdG9yIHsKLQotICAgIHB1YmxpYyB2b2lkIHBhaW50VmFsdWUoR3JhcGhpY3MgZ2Z4LCBSZWN0YW5nbGUgYm94KTsKLQotICAgIHB1YmxpYyB2b2lkIHNldEFzVGV4dChTdHJpbmcgdGV4dCkgdGhyb3dzIElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbjsKLQotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRUYWdzKCk7Ci0KLSAgICBwdWJsaWMgU3RyaW5nIGdldEphdmFJbml0aWFsaXphdGlvblN0cmluZygpOwotCi0gICAgcHVibGljIFN0cmluZyBnZXRBc1RleHQoKTsKLQotICAgIHB1YmxpYyB2b2lkIHNldFZhbHVlKE9iamVjdCB2YWx1ZSk7Ci0KLSAgICBwdWJsaWMgT2JqZWN0IGdldFZhbHVlKCk7Ci0KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyKFByb3BlcnR5Q2hhbmdlTGlzdGVuZXIgbGlzdGVuZXIpOwotCi0gICAgcHVibGljIHZvaWQgYWRkUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcihQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyIGxpc3RlbmVyKTsKLQotICAgIHB1YmxpYyBDb21wb25lbnQgZ2V0Q3VzdG9tRWRpdG9yKCk7Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBzdXBwb3J0c0N1c3RvbUVkaXRvcigpOwotCi0gICAgcHVibGljIGJvb2xlYW4gaXNQYWludGFibGUoKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2JlYW5zL1Byb3BlcnR5RWRpdG9yTWFuYWdlci5qYXZhIGIvYXd0L2phdmEvYmVhbnMvUHJvcGVydHlFZGl0b3JNYW5hZ2VyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGVkNTU4MjkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYmVhbnMvUHJvcGVydHlFZGl0b3JNYW5hZ2VyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMTQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotcGFja2FnZSBqYXZhLmJlYW5zOwotCi1pbXBvcnQgamF2YS51dGlsLkhhc2hNYXA7Ci1pbXBvcnQgamF2YS51dGlsLk1hcDsKLQotcHVibGljIGNsYXNzIFByb3BlcnR5RWRpdG9yTWFuYWdlciB7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBTdHJpbmdbXSBwYXRoID0geyAib3JnLmFwYWNoZS5oYXJtb255LmJlYW5zLmVkaXRvcnMiIH07IC8vJE5PTi1OTFMtMSQKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIE1hcDxDbGFzczw/PiwgQ2xhc3M8Pz4+IHJlZ2lzdGVyZWRFZGl0b3JzID0gbmV3IEhhc2hNYXA8Q2xhc3M8Pz4sIENsYXNzPD8+PigpOwotCi0gICAgcHVibGljIFByb3BlcnR5RWRpdG9yTWFuYWdlcigpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgcmVnaXN0ZXJFZGl0b3IoQ2xhc3M8Pz4gdGFyZ2V0VHlwZSwgQ2xhc3M8Pz4gZWRpdG9yQ2xhc3MpIHsKLSAgICAgICAgaWYgKHRhcmdldFR5cGUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBTZWN1cml0eU1hbmFnZXIgc20gPSBTeXN0ZW0uZ2V0U2VjdXJpdHlNYW5hZ2VyKCk7Ci0gICAgICAgIGlmIChzbSAhPSBudWxsKSB7Ci0gICAgICAgICAgICBzbS5jaGVja1Byb3BlcnRpZXNBY2Nlc3MoKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoZWRpdG9yQ2xhc3MgIT0gbnVsbCkgewotICAgICAgICAgICAgcmVnaXN0ZXJlZEVkaXRvcnMucHV0KHRhcmdldFR5cGUsIGVkaXRvckNsYXNzKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJlZ2lzdGVyZWRFZGl0b3JzLnJlbW92ZSh0YXJnZXRUeXBlKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgc3luY2hyb25pemVkIFByb3BlcnR5RWRpdG9yIGZpbmRFZGl0b3IoQ2xhc3M8Pz4gdGFyZ2V0VHlwZSkgewotICAgICAgICBpZiAodGFyZ2V0VHlwZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oKTsKLSAgICAgICAgfQotCi0gICAgICAgIENsYXNzPD8+IGVkaXRvckNsYXNzID0gbnVsbDsKLSAgICAgICAgUHJvcGVydHlFZGl0b3IgZWRpdG9yID0gbnVsbDsKLQotICAgICAgICBlZGl0b3JDbGFzcyA9IHJlZ2lzdGVyZWRFZGl0b3JzLmdldCh0YXJnZXRUeXBlKTsKLQotICAgICAgICBpZiAoZWRpdG9yQ2xhc3MgPT0gbnVsbCkgewotICAgICAgICAgICAgU3RyaW5nIGVkaXRvckNsYXNzTmFtZSA9IHRhcmdldFR5cGUuZ2V0TmFtZSgpICsgIkVkaXRvciI7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIENsYXNzTG9hZGVyIGxvYWRlciA9IHRhcmdldFR5cGUuZ2V0Q2xhc3NMb2FkZXIoKTsKLQotICAgICAgICAgICAgaWYgKGxvYWRlciA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgbG9hZGVyID0gVGhyZWFkLmN1cnJlbnRUaHJlYWQoKS5nZXRDb250ZXh0Q2xhc3NMb2FkZXIoKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICBlZGl0b3JDbGFzcyA9IENsYXNzLmZvck5hbWUoZWRpdG9yQ2xhc3NOYW1lLCB0cnVlLCBsb2FkZXIpOwotICAgICAgICAgICAgfSBjYXRjaCAoQ2xhc3NOb3RGb3VuZEV4Y2VwdGlvbiBjbmZlKSB7Ci0gICAgICAgICAgICAgICAgU3RyaW5nIHNob3J0RWRpdG9yQ2xhc3NOYW1lID0gZWRpdG9yQ2xhc3NOYW1lCi0gICAgICAgICAgICAgICAgICAgICAgICAuc3Vic3RyaW5nKGVkaXRvckNsYXNzTmFtZS5sYXN0SW5kZXhPZigiLiIpICsgMSk7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICAgICAgICAgIGlmICh0YXJnZXRUeXBlLmlzUHJpbWl0aXZlKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgc2hvcnRFZGl0b3JDbGFzc05hbWUgPSBzaG9ydEVkaXRvckNsYXNzTmFtZS5zdWJzdHJpbmcoMCwgMSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAudG9VcHBlckNhc2UoKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICsgc2hvcnRFZGl0b3JDbGFzc05hbWUuc3Vic3RyaW5nKDEpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGZvciAoU3RyaW5nIGVsZW1lbnQgOiBwYXRoKSB7Ci0gICAgICAgICAgICAgICAgICAgIGVkaXRvckNsYXNzTmFtZSA9IGVsZW1lbnQgKyAiLiIgKyBzaG9ydEVkaXRvckNsYXNzTmFtZTsgLy8kTk9OLU5MUy0xJAotCi0gICAgICAgICAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBlZGl0b3JDbGFzcyA9IENsYXNzLmZvck5hbWUoZWRpdG9yQ2xhc3NOYW1lLCB0cnVlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsb2FkZXIpOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGNhdGNoIChFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGVkaXRvckNsYXNzICE9IG51bGwpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgZWRpdG9yID0gKFByb3BlcnR5RWRpdG9yKSBlZGl0b3JDbGFzcy5uZXdJbnN0YW5jZSgpOwotICAgICAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBlZGl0b3I7Ci0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyBzeW5jaHJvbml6ZWQgdm9pZCBzZXRFZGl0b3JTZWFyY2hQYXRoKFN0cmluZ1tdIGFwYXRoKSB7Ci0gICAgICAgIFNlY3VyaXR5TWFuYWdlciBzbSA9IFN5c3RlbS5nZXRTZWN1cml0eU1hbmFnZXIoKTsKLSAgICAgICAgaWYgKHNtICE9IG51bGwpIHsKLSAgICAgICAgICAgIHNtLmNoZWNrUHJvcGVydGllc0FjY2VzcygpOwotICAgICAgICB9Ci0KLSAgICAgICAgcGF0aCA9IGFwYXRoOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgc3luY2hyb25pemVkIFN0cmluZ1tdIGdldEVkaXRvclNlYXJjaFBhdGgoKSB7Ci0gICAgICAgIHJldHVybiBwYXRoOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZhL2JlYW5zL1Byb3BlcnR5RWRpdG9yU3VwcG9ydC5qYXZhIGIvYXd0L2phdmEvYmVhbnMvUHJvcGVydHlFZGl0b3JTdXBwb3J0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGMzOTI5YTEuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmEvYmVhbnMvUHJvcGVydHlFZGl0b3JTdXBwb3J0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMjkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLXBhY2thZ2UgamF2YS5iZWFuczsKLQotaW1wb3J0IGphdmEuYXd0LkNvbXBvbmVudDsKLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljczsKLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci1pbXBvcnQgamF2YS51dGlsLkFycmF5TGlzdDsKLWltcG9ydCBqYXZhLnV0aWwuSXRlcmF0b3I7Ci1pbXBvcnQgamF2YS51dGlsLkxpc3Q7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYmVhbnMuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi1wdWJsaWMgY2xhc3MgUHJvcGVydHlFZGl0b3JTdXBwb3J0IGltcGxlbWVudHMgUHJvcGVydHlFZGl0b3IgewotCi0gICAgT2JqZWN0IHNvdXJjZSA9IG51bGw7Ci0KLSAgICBMaXN0PFByb3BlcnR5Q2hhbmdlTGlzdGVuZXI+IGxpc3RlbmVycyA9IG5ldyBBcnJheUxpc3Q8UHJvcGVydHlDaGFuZ2VMaXN0ZW5lcj4oKTsKLQotICAgIE9iamVjdCBvbGRWYWx1ZSA9IG51bGw7Ci0KLSAgICBPYmplY3QgbmV3VmFsdWUgPSBudWxsOwotCi0gICAgcHVibGljIFByb3BlcnR5RWRpdG9yU3VwcG9ydChPYmplY3Qgc291cmNlKSB7Ci0gICAgICAgIGlmIChzb3VyY2UgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMEMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB0aGlzLnNvdXJjZSA9IHNvdXJjZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgUHJvcGVydHlFZGl0b3JTdXBwb3J0KCkgewotICAgICAgICBzb3VyY2UgPSB0aGlzOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHBhaW50VmFsdWUoR3JhcGhpY3MgZ2Z4LCBSZWN0YW5nbGUgYm94KSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0QXNUZXh0KFN0cmluZyB0ZXh0KSB0aHJvd3MgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKG5ld1ZhbHVlIGluc3RhbmNlb2YgU3RyaW5nKSB7Ci0gICAgICAgICAgICBzZXRWYWx1ZSh0ZXh0KTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24odGV4dCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0VGFncygpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZyBnZXRKYXZhSW5pdGlhbGl6YXRpb25TdHJpbmcoKSB7Ci0gICAgICAgIHJldHVybiAiPz8/IjsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0QXNUZXh0KCkgewotICAgICAgICByZXR1cm4gbmV3VmFsdWUgPT0gbnVsbCA/ICJudWxsIiA6IG5ld1ZhbHVlLnRvU3RyaW5nKCk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRWYWx1ZShPYmplY3QgdmFsdWUpIHsKLSAgICAgICAgdGhpcy5vbGRWYWx1ZSA9IHRoaXMubmV3VmFsdWU7Ci0gICAgICAgIHRoaXMubmV3VmFsdWUgPSB2YWx1ZTsKLSAgICAgICAgZmlyZVByb3BlcnR5Q2hhbmdlKCk7Ci0gICAgfQotCi0gICAgcHVibGljIE9iamVjdCBnZXRWYWx1ZSgpIHsKLSAgICAgICAgcmV0dXJuIG5ld1ZhbHVlOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldFNvdXJjZShPYmplY3Qgc291cmNlKSB7Ci0gICAgICAgIGlmIChzb3VyY2UgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYmVhbnMuMEMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB0aGlzLnNvdXJjZSA9IHNvdXJjZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgT2JqZWN0IGdldFNvdXJjZSgpIHsKLSAgICAgICAgcmV0dXJuIHNvdXJjZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgcmVtb3ZlUHJvcGVydHlDaGFuZ2VMaXN0ZW5lcigKLSAgICAgICAgICAgIFByb3BlcnR5Q2hhbmdlTGlzdGVuZXIgbGlzdGVuZXIpIHsKLSAgICAgICAgaWYgKGxpc3RlbmVycyAhPSBudWxsKSB7Ci0gICAgICAgICAgICBsaXN0ZW5lcnMucmVtb3ZlKGxpc3RlbmVyKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCBhZGRQcm9wZXJ0eUNoYW5nZUxpc3RlbmVyKAotICAgICAgICAgICAgUHJvcGVydHlDaGFuZ2VMaXN0ZW5lciBsaXN0ZW5lcikgewotICAgICAgICBsaXN0ZW5lcnMuYWRkKGxpc3RlbmVyKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQ29tcG9uZW50IGdldEN1c3RvbUVkaXRvcigpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gc3VwcG9ydHNDdXN0b21FZGl0b3IoKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1BhaW50YWJsZSgpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGZpcmVQcm9wZXJ0eUNoYW5nZSgpIHsKLSAgICAgICAgaWYgKGxpc3RlbmVycy5zaXplKCkgPiAwKSB7Ci0gICAgICAgICAgICBQcm9wZXJ0eUNoYW5nZUV2ZW50IGV2ZW50ID0gbmV3IFByb3BlcnR5Q2hhbmdlRXZlbnQoc291cmNlLCBudWxsLAotICAgICAgICAgICAgICAgICAgICBvbGRWYWx1ZSwgbmV3VmFsdWUpOwotICAgICAgICAgICAgSXRlcmF0b3I8UHJvcGVydHlDaGFuZ2VMaXN0ZW5lcj4gaXRlcmF0b3IgPSBsaXN0ZW5lcnMuaXRlcmF0b3IoKTsKLQotICAgICAgICAgICAgd2hpbGUgKGl0ZXJhdG9yLmhhc05leHQoKSkgewotICAgICAgICAgICAgICAgIFByb3BlcnR5Q2hhbmdlTGlzdGVuZXIgbGlzdGVuZXIgPSBpdGVyYXRvci5uZXh0KCk7Ci0gICAgICAgICAgICAgICAgbGlzdGVuZXIucHJvcGVydHlDaGFuZ2UoZXZlbnQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmEvYmVhbnMvUHJvcGVydHlWZXRvRXhjZXB0aW9uLmphdmEgYi9hd3QvamF2YS9iZWFucy9Qcm9wZXJ0eVZldG9FeGNlcHRpb24uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzdmMDkyYS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YS9iZWFucy9Qcm9wZXJ0eVZldG9FeGNlcHRpb24uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDU0ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YS5iZWFuczsKLQotLyoqCi0gKiBJbmRpY2F0ZXMgdGhhdCBhIHByb3Bvc2VkIHByb3BlcnR5IGNoYW5nZSBpcyB1bmFjY2VwdGFibGUuCi0gKi8KLXB1YmxpYyBjbGFzcyBQcm9wZXJ0eVZldG9FeGNlcHRpb24gZXh0ZW5kcyBFeGNlcHRpb24gewotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gMTI5NTk2MDU3Njk0MTYyMTY0TDsKLQotICAgIHByaXZhdGUgZmluYWwgUHJvcGVydHlDaGFuZ2VFdmVudCBldnQ7Ci0KLSAgICAvKioKLSAgICAgKiA8cD4KLSAgICAgKiBDb25zdHJ1Y3RzIGFuIGluc3RhbmNlIHdpdGggYSBtZXNzYWdlIGFuZCB0aGUgY2hhbmdlIGV2ZW50LgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbWVzc2FnZQotICAgICAqICAgICAgICAgICAgQSBkZXNjcmlwdGlvbiBvZiB0aGUgdmV0by4KLSAgICAgKiBAcGFyYW0gZXZlbnQKLSAgICAgKiAgICAgICAgICAgIFRoZSBldmVudCB0aGF0IHdhcyB2ZXRvZWQuCi0gICAgICovCi0gICAgcHVibGljIFByb3BlcnR5VmV0b0V4Y2VwdGlvbihTdHJpbmcgbWVzc2FnZSwgUHJvcGVydHlDaGFuZ2VFdmVudCBldmVudCkgewotICAgICAgICBzdXBlcihtZXNzYWdlKTsKLSAgICAgICAgdGhpcy5ldnQgPSBldmVudDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8cD4KLSAgICAgKiBHZXRzIHRoZSBwcm9wZXJ0eSBjaGFuZ2UgZXZlbnQuCi0gICAgICogPC9wPgotICAgICAqIAotICAgICAqIEByZXR1cm4gQW4gaW5zdGFuY2Ugb2Yge0BsaW5rIFByb3BlcnR5Q2hhbmdlRXZlbnR9Ci0gICAgICovCi0gICAgcHVibGljIFByb3BlcnR5Q2hhbmdlRXZlbnQgZ2V0UHJvcGVydHlDaGFuZ2VFdmVudCgpIHsKLSAgICAgICAgcmV0dXJuIGV2dDsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9JSU9FeGNlcHRpb24uamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL0lJT0V4Y2VwdGlvbi5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjNzc3MTZjLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL0lJT0V4Y2VwdGlvbi5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNjAgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2VpbzsKLQotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci0KLS8qKgotICogVGhlIElJT0V4Y2VwdGlvbiBjbGFzcyBpbmRpY2F0ZXMgZXJyb3JzIGluIHJlYWRpbmcvd3JpdGluZyBvcGVyYXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIElJT0V4Y2VwdGlvbiBleHRlbmRzIElPRXhjZXB0aW9uIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzZXJpYWxWZXJzaW9uVUlELgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGxvbmcgc2VyaWFsVmVyc2lvblVJRCA9IC0zMjE2MjEwNzE4NjM4OTg1MjUxTDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJSU9FeGNlcHRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIG1lc3NhZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXRhaWxlZCBtZXNzYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBJSU9FeGNlcHRpb24oU3RyaW5nIG1lc3NhZ2UpIHsKLSAgICAgICAgc3VwZXIobWVzc2FnZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IElJT0V4Y2VwdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbWVzc2FnZQotICAgICAqICAgICAgICAgICAgdGhlIGRldGFpbGVkIG1lc3NhZ2UuCi0gICAgICogQHBhcmFtIGNhdXNlCi0gICAgICogICAgICAgICAgICB0aGUgY2F1c2Ugb2YgdGhpcyBleGNlcHRpb24uCi0gICAgICovCi0gICAgcHVibGljIElJT0V4Y2VwdGlvbihTdHJpbmcgbWVzc2FnZSwgVGhyb3dhYmxlIGNhdXNlKSB7Ci0gICAgICAgIHN1cGVyKG1lc3NhZ2UpOwotICAgICAgICBpbml0Q2F1c2UoY2F1c2UpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL0lJT0ltYWdlLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9JSU9JbWFnZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlOWU1MTMwLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL0lJT0ltYWdlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMjQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2VpbzsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8ubWV0YWRhdGEuSUlPTWV0YWRhdGE7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmVuZGVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5SYXN0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLnV0aWwuTGlzdDsKLQotLyoqCi0gKiBUaGUgSUlPSW1hZ2UgY2xhc3MgY29tYmluZXMgdGhlIGltYWdlLCBpbWFnZSdzIHRodW1ibmFpbCBhbmQgaW1hZ2UncwotICogbWV0YWRhdGEuIFRoZSBpbWFnZSBjYW4gYmUgcHJlc2VudGVkIGFzIFJlbmRlcmVkSW1hZ2Ugb3IgUmFzdGVyIG9iamVjdC4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJSU9JbWFnZSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaW1hZ2Ugb2YgdGhpcyBJSU9JbWFnZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgUmVuZGVyZWRJbWFnZSBpbWFnZTsKLQotICAgIC8qKgotICAgICAqIFRoZSByYXN0ZXIgb2YgdGhpcyBJSU9JbWFnZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgUmFzdGVyIHJhc3RlcjsKLQotICAgIC8qKgotICAgICAqIFRoZSBsaXN0IHdpdGggdGh1bWJuYWlscyBhc3NvY2lhdGVkIHdpdGggdGhlIGltYWdlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBMaXN0PD8gZXh0ZW5kcyBCdWZmZXJlZEltYWdlPiB0aHVtYm5haWxzOwotCi0gICAgLyoqCi0gICAgICogVGhlIG1ldGFkYXRhIGFzc29jaWF0ZWQgd2l0aCB0aGUgaW1hZ2UuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIElJT01ldGFkYXRhIG1ldGFkYXRhOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IElJT0ltYWdlIHdpdGggdGhlIHNwZWNpZmllZCBSZW5kZXJlZEltYWdlLCBsaXN0IG9mCi0gICAgICogdGh1bWJuYWlscyBhbmQgbWV0YWRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2Ugc3BlY2lmaWVkIGJ5IFJlbmRlcmVkSW1hZ2UuCi0gICAgICogQHBhcmFtIHRodW1ibmFpbHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBsaXN0IG9mIEJ1ZmZlcmVkSW1hZ2Ugb2JqZWN0cyB3aGljaCByZXByZXNlbnQgdGhlCi0gICAgICogICAgICAgICAgICB0aHVtYm5haWxzIG9mIHRoZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gbWV0YWRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBtZXRhZGF0YSBvZiB0aGUgaW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIElJT0ltYWdlKFJlbmRlcmVkSW1hZ2UgaW1hZ2UsIExpc3Q8PyBleHRlbmRzIEJ1ZmZlcmVkSW1hZ2U+IHRodW1ibmFpbHMsCi0gICAgICAgICAgICBJSU9NZXRhZGF0YSBtZXRhZGF0YSkgewotICAgICAgICBpZiAoaW1hZ2UgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiaW1hZ2Ugc2hvdWxkIG5vdCBiZSBOVUxMIik7Ci0gICAgICAgIH0KLSAgICAgICAgdGhpcy5yYXN0ZXIgPSBudWxsOwotICAgICAgICB0aGlzLmltYWdlID0gaW1hZ2U7Ci0gICAgICAgIHRoaXMudGh1bWJuYWlscyA9IHRodW1ibmFpbHM7Ci0gICAgICAgIHRoaXMubWV0YWRhdGEgPSBtZXRhZGF0YTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSUlPSW1hZ2Ugd2l0aCB0aGUgc3BlY2lmaWVkIFJhc3RlciwgbGlzdCBvZiB0aHVtYm5haWxzCi0gICAgICogYW5kIG1ldGFkYXRhLgotICAgICAqIAotICAgICAqIEBwYXJhbSByYXN0ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBSYXN0ZXIuCi0gICAgICogQHBhcmFtIHRodW1ibmFpbHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBsaXN0IG9mIEJ1ZmZlcmVkSW1hZ2Ugb2JqZWN0cyB3aGljaCByZXByZXNlbnQgdGhlCi0gICAgICogICAgICAgICAgICB0aHVtYm5haWxzIG9mIFJhc3RlciBkYXRhLgotICAgICAqIEBwYXJhbSBtZXRhZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIG1ldGFkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBJSU9JbWFnZShSYXN0ZXIgcmFzdGVyLCBMaXN0PD8gZXh0ZW5kcyBCdWZmZXJlZEltYWdlPiB0aHVtYm5haWxzLCBJSU9NZXRhZGF0YSBtZXRhZGF0YSkgewotICAgICAgICBpZiAocmFzdGVyID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInJhc3RlciBzaG91bGQgbm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotICAgICAgICB0aGlzLmltYWdlID0gbnVsbDsKLSAgICAgICAgdGhpcy5yYXN0ZXIgPSByYXN0ZXI7Ci0gICAgICAgIHRoaXMudGh1bWJuYWlscyA9IHRodW1ibmFpbHM7Ci0gICAgICAgIHRoaXMubWV0YWRhdGEgPSBtZXRhZGF0YTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBSZW5kZXJlZEltYWdlIG9iamVjdCBvciByZXR1cm5zIG51bGwgaWYgdGhpcyBJSU9JbWFnZSBvYmplY3QgaXMKLSAgICAgKiBhc3NvY2lhdGVkIHdpdGggYSBSYXN0ZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgUmVuZGVyZWRJbWFnZSBvYmplY3Qgb3IgbnVsbCBpZiB0aGlzIElJT0ltYWdlIG9iamVjdCBpcwotICAgICAqICAgICAgICAgYXNzb2NpYXRlZCB3aXRoIGEgUmFzdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBSZW5kZXJlZEltYWdlIGdldFJlbmRlcmVkSW1hZ2UoKSB7Ci0gICAgICAgIHJldHVybiBpbWFnZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBSZW5kZXJlZEltYWdlIHRvIHRoaXMgSUlPSW1hZ2Ugb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIFJlbmRlcmVkSW1hZ2UgdG8gYmUgc2V0IHRvIHRoaXMgSUlPSW1hZ2UuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0UmVuZGVyZWRJbWFnZShSZW5kZXJlZEltYWdlIGltYWdlKSB7Ci0gICAgICAgIGlmIChpbWFnZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJpbWFnZSBzaG91bGQgbm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotICAgICAgICByYXN0ZXIgPSBudWxsOwotICAgICAgICB0aGlzLmltYWdlID0gaW1hZ2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoZSBJSU9JbWFnZSBvYmplY3QgYXNzb2NpYXRlZCB3aXRoIGEgUmFzdGVyLCBvciBmYWxzZSBpZgotICAgICAqIGl0J3MgYXNzb2NpYXRlZCB3aXRoIGEgUmVuZGVyZWRJbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBJSU9JbWFnZSBvYmplY3QgYXNzb2NpYXRlZCB3aXRoIGEgUmFzdGVyLCBvciBmYWxzZQotICAgICAqICAgICAgICAgaWYgaXQncyBhc3NvY2lhdGVkIHdpdGggYSBSZW5kZXJlZEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGhhc1Jhc3RlcigpIHsKLSAgICAgICAgcmV0dXJuIHJhc3RlciAhPSBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFJhc3RlciBvYmplY3Qgb3IgcmV0dXJucyBudWxsIGlmIHRoaXMgSUlPSW1hZ2Ugb2JqZWN0IGlzCi0gICAgICogYXNzb2NpYXRlZCB3aXRoIGEgUmVuZGVyZWRJbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBSYXN0ZXIgb3IgbnVsbCBpZiB0aGlzIElJT0ltYWdlIG9iamVjdCBpcyBhc3NvY2lhdGVkIHdpdGggYQotICAgICAqICAgICAgICAgUmVuZGVyZWRJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmFzdGVyIGdldFJhc3RlcigpIHsKLSAgICAgICAgcmV0dXJuIHJhc3RlcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBSYXN0ZXIgdG8gdGhlIElJT0ltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSByYXN0ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgUmFzdGVyIHRvIHRoZSBJSU9JbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRSYXN0ZXIoUmFzdGVyIHJhc3RlcikgewotICAgICAgICBpZiAocmFzdGVyID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInJhc3RlciBzaG91bGQgbm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotICAgICAgICBpbWFnZSA9IG51bGw7Ci0gICAgICAgIHRoaXMucmFzdGVyID0gcmFzdGVyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG51bWJlciBvZiB0aHVtYm5haWxzIGZvciB0aGlzIElJT0ltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiB0aHVtYm5haWxzIGZvciB0aGlzIElJT0ltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TnVtVGh1bWJuYWlscygpIHsKLSAgICAgICAgcmV0dXJuIHRodW1ibmFpbHMgIT0gbnVsbCA/IHRodW1ibmFpbHMuc2l6ZSgpIDogMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0aHVtYm5haWwgd2l0aCB0aGUgc3BlY2lmaWVkIGluZGV4IGluIHRoZSBsaXN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSB0aHVtYm5haWwgaW4gdGhlIGxpc3QuCi0gICAgICogQHJldHVybiB0aGUgdGh1bWJuYWlsIHdpdGggdGhlIHNwZWNpZmllZCBpbmRleCBpbiB0aGUgbGlzdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQnVmZmVyZWRJbWFnZSBnZXRUaHVtYm5haWwoaW50IGluZGV4KSB7Ci0gICAgICAgIGlmICh0aHVtYm5haWxzICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiB0aHVtYm5haWxzLmdldChpbmRleCk7Ci0gICAgICAgIH0KLSAgICAgICAgdGhyb3cgbmV3IEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oIm5vIHRodW1ibmFpbHMgd2VyZSBzZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBsaXN0IG9mIHRodW1ibmFpbHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbGlzdCBvZiB0aHVtYm5haWxzLgotICAgICAqLwotICAgIHB1YmxpYyBMaXN0PD8gZXh0ZW5kcyBCdWZmZXJlZEltYWdlPiBnZXRUaHVtYm5haWxzKCkgewotICAgICAgICByZXR1cm4gdGh1bWJuYWlsczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBsaXN0IG9mIHRodW1ibmFpbHMgaW1hZ2VzIHRvIHRoaXMgSUlPSW1hZ2Ugb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB0aHVtYm5haWxzCi0gICAgICogICAgICAgICAgICB0aGUgbGlzdCBvZiBCdWZmZXJlZEltYWdlIHdoaWNoIHJlcHJlc2VudCB0aHVtYm5haWxzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFRodW1ibmFpbHMoTGlzdDw/IGV4dGVuZHMgQnVmZmVyZWRJbWFnZT4gdGh1bWJuYWlscykgewotICAgICAgICB0aGlzLnRodW1ibmFpbHMgPSB0aHVtYm5haWxzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1ldGFkYXRhIG9mIHRoaXMgSUlPSW1hZ2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWV0YWRhdGEgb2YgdGhpcyBJSU9JbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgSUlPTWV0YWRhdGEgZ2V0TWV0YWRhdGEoKSB7Ci0gICAgICAgIHJldHVybiBtZXRhZGF0YTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBtZXRhZGF0YSB0byB0aGlzIElJT0ltYWdlIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbWV0YWRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBJSU9NZXRhZGF0YSwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRNZXRhZGF0YShJSU9NZXRhZGF0YSBtZXRhZGF0YSkgewotICAgICAgICB0aGlzLm1ldGFkYXRhID0gbWV0YWRhdGE7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vSUlPUGFyYW0uamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL0lJT1BhcmFtLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJjY2M5NDUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vSUlPUGFyYW0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDM0MiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFJ1c3RlbSBWLiBSYWZpa292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMyAkCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvOwotCi1pbXBvcnQgamF2YS5hd3QuKjsKLQotLyoqCi0gKiBUaGUgSUlPUGFyYW0gYWJzdHJhY3QgY2xhc3MgaXMgc3VwZXJjbGFzcyBmb3IgSW1hZ2VSZWFkUGFyYW0gYW5kCi0gKiBJbWFnZVdyaXRlUGFyYW0gY2xhc3NlcyBhbmQgcHJvdmlkZXMgbWV0aG9kcyBhbmQgdmFyaWFibGVzIHdoaWNoIHRoZXkgc2hhcmUuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgSUlPUGFyYW0gewotCi0gICAgLyoqCi0gICAgICogVGhlIHNvdXJjZSByZWdpb24uCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFJlY3RhbmdsZSBzb3VyY2VSZWdpb247Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc291cmNlIHggc3Vic2FtcGxpbmcuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBzb3VyY2VYU3Vic2FtcGxpbmcgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIHNvdXJjZSB5IHN1YnNhbXBsaW5nLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgc291cmNlWVN1YnNhbXBsaW5nID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBzdWJzYW1wbGluZyB4IG9mZnNldC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IHN1YnNhbXBsaW5nWE9mZnNldDsKLQotICAgIC8qKgotICAgICAqIFRoZSBzdWJzYW1wbGluZyB5IG9mZnNldC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IHN1YnNhbXBsaW5nWU9mZnNldDsKLQotICAgIC8qKgotICAgICAqIFRoZSBzb3VyY2UgYmFuZHMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludFtdIHNvdXJjZUJhbmRzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGRlc3RpbmF0aW9uIHR5cGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEltYWdlVHlwZVNwZWNpZmllciBkZXN0aW5hdGlvblR5cGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGVzdGluYXRpb24gb2Zmc2V0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBQb2ludCBkZXN0aW5hdGlvbk9mZnNldCA9IG5ldyBQb2ludCgwLCAwKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBkZWZhdWx0IGNvbnRyb2xsZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIElJT1BhcmFtQ29udHJvbGxlciBkZWZhdWx0Q29udHJvbGxlcjsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb250cm9sbGVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBJSU9QYXJhbUNvbnRyb2xsZXIgY29udHJvbGxlcjsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJSU9QYXJhbS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgSUlPUGFyYW0oKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc291cmNlIHJlZ2lvbiBhcyBhIFJlY3RhbmdsZSBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZVJlZ2lvbgotICAgICAqICAgICAgICAgICAgdGhlIFJlY3RhbmdsZSB3aGljaCBzcGVjaWZpZXMgdGhlIHNvdXJjZSByZWdpb24uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0U291cmNlUmVnaW9uKFJlY3RhbmdsZSBzb3VyY2VSZWdpb24pIHsKLSAgICAgICAgaWYgKHNvdXJjZVJlZ2lvbiAhPSBudWxsKSB7Ci0gICAgICAgICAgICBpZiAoc291cmNlUmVnaW9uLnggPCAwKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigieCA8IDAiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChzb3VyY2VSZWdpb24ueSA8IDApIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJ5IDwgMCIpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHNvdXJjZVJlZ2lvbi53aWR0aCA8PSAwKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigid2lkdGggPD0gMCIpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHNvdXJjZVJlZ2lvbi5oZWlnaHQgPD0gMCkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImhlaWdodCA8PSAwIik7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChzb3VyY2VSZWdpb24ud2lkdGggPD0gc3Vic2FtcGxpbmdYT2Zmc2V0KSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigid2lkdGggPD0gc3Vic2FtcGxpbmdYT2Zmc2V0Iik7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChzb3VyY2VSZWdpb24uaGVpZ2h0IDw9IHN1YnNhbXBsaW5nWU9mZnNldCkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImhlaWdodCA8PSBzdWJzYW1wbGluZ1hPZmZzZXQiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIC8vIC0tIGNsb25lIGl0IHRvIGF2b2lkIHVuZXhwZWN0ZWQgbW9kaWZpY2F0aW9ucwotICAgICAgICAgICAgdGhpcy5zb3VyY2VSZWdpb24gPSAoUmVjdGFuZ2xlKXNvdXJjZVJlZ2lvbi5jbG9uZSgpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdGhpcy5zb3VyY2VSZWdpb24gPSBudWxsOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc291cmNlIHJlZ2lvbi4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzb3VyY2UgcmVnaW9uIGFzIFJlY3RhbmdsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldFNvdXJjZVJlZ2lvbigpIHsKLSAgICAgICAgaWYgKHNvdXJjZVJlZ2lvbiA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotICAgICAgICAvLyAtLSBjbG9uZSBpdCB0byBhdm9pZCB1bmV4cGVjdGVkIG1vZGlmaWNhdGlvbnMKLSAgICAgICAgcmV0dXJuIChSZWN0YW5nbGUpc291cmNlUmVnaW9uLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc291cmNlIHN1YnNhbXBsaW5nLiBUaGUgc291cmNlWFN1YnNhbXBsaW5nIGFuZAotICAgICAqIHNvdXJjZVlTdWJzYW1wbGluZyBwYXJhbWV0ZXJzIHNwZWNpZnkgdGhlIG51bWJlciBvZiByb3dzIGFuZCBjb2x1bW5zIHRvCi0gICAgICogYWR2YW5jZSBhZnRlciBldmVyeSBzb3VyY2UgcGl4ZWwuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZVhTdWJzYW1wbGluZwotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBYIHN1YnNhbXBsaW5nLgotICAgICAqIEBwYXJhbSBzb3VyY2VZU3Vic2FtcGxpbmcKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgWSBzdWJzYW1wbGluZy4KLSAgICAgKiBAcGFyYW0gc3Vic2FtcGxpbmdYT2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgc3Vic2FtcGxpbmcgWCBvZmZzZXQuCi0gICAgICogQHBhcmFtIHN1YnNhbXBsaW5nWU9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIHN1YnNhbXBsaW5nIFkgb2Zmc2V0LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFNvdXJjZVN1YnNhbXBsaW5nKGludCBzb3VyY2VYU3Vic2FtcGxpbmcsIGludCBzb3VyY2VZU3Vic2FtcGxpbmcsCi0gICAgICAgICAgICBpbnQgc3Vic2FtcGxpbmdYT2Zmc2V0LCBpbnQgc3Vic2FtcGxpbmdZT2Zmc2V0KSB7Ci0KLSAgICAgICAgaWYgKHNvdXJjZVhTdWJzYW1wbGluZyA8PSAwKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJzb3VyY2VYU3Vic2FtcGxpbmcgPD0gMCIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChzb3VyY2VZU3Vic2FtcGxpbmcgPD0gMCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigic291cmNlWVN1YnNhbXBsaW5nIDw9IDAiKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChzdWJzYW1wbGluZ1hPZmZzZXQgPD0gMCB8fCBzdWJzYW1wbGluZ1hPZmZzZXQgPj0gc291cmNlWFN1YnNhbXBsaW5nKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJzdWJzYW1wbGluZ1hPZmZzZXQgaXMgd3JvbmciKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChzdWJzYW1wbGluZ1lPZmZzZXQgPD0gMCB8fCBzdWJzYW1wbGluZ1lPZmZzZXQgPj0gc291cmNlWVN1YnNhbXBsaW5nKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJzdWJzYW1wbGluZ1lPZmZzZXQgaXMgd3JvbmciKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIC0tIGRvZXMgcmVnaW9uIGNvbnRhaW4gcGl4ZWxzCi0gICAgICAgIGlmIChzb3VyY2VSZWdpb24gIT0gbnVsbCkgewotICAgICAgICAgICAgaWYgKHNvdXJjZVJlZ2lvbi53aWR0aCA8PSBzdWJzYW1wbGluZ1hPZmZzZXQKLSAgICAgICAgICAgICAgICAgICAgfHwgc291cmNlUmVnaW9uLmhlaWdodCA8PSBzdWJzYW1wbGluZ1lPZmZzZXQpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJ0aGVyZSBhcmUgbm8gcGl4ZWxzIGluIHJlZ2lvbiIpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5zb3VyY2VYU3Vic2FtcGxpbmcgPSBzb3VyY2VYU3Vic2FtcGxpbmc7Ci0gICAgICAgIHRoaXMuc291cmNlWVN1YnNhbXBsaW5nID0gc291cmNlWVN1YnNhbXBsaW5nOwotICAgICAgICB0aGlzLnN1YnNhbXBsaW5nWE9mZnNldCA9IHN1YnNhbXBsaW5nWE9mZnNldDsKLSAgICAgICAgdGhpcy5zdWJzYW1wbGluZ1lPZmZzZXQgPSBzdWJzYW1wbGluZ1lPZmZzZXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc291cmNlIFggc3Vic2FtcGxpbmcgLSB0aGUgbnVtYmVyIG9mIHNvdXJjZSBjb2x1bW5zIHRvIGFkdmFuY2UKLSAgICAgKiBmb3IgZWFjaCBwaXhlbC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzb3VyY2UgWCBzdWJzYW1wbGluZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFNvdXJjZVhTdWJzYW1wbGluZygpIHsKLSAgICAgICAgcmV0dXJuIHNvdXJjZVhTdWJzYW1wbGluZzsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzb3VyY2UgWSBzdWJzYW1wbGluZyAtIHRoZSBudW1iZXIgb2Ygc291cmNlIHJvd3MgdG8gYWR2YW5jZSBmb3IKLSAgICAgKiBlYWNoIHBpeGVsLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNvdXJjZSBZIHN1YnNhbXBsaW5nLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0U291cmNlWVN1YnNhbXBsaW5nKCkgewotICAgICAgICByZXR1cm4gc291cmNlWVN1YnNhbXBsaW5nOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGhvcml6b250YWwgb2Zmc2V0IG9mIHRoZSBzdWJzYW1wbGluZyBncmlkLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGhvcml6b250YWwgb2Zmc2V0IG9mIHRoZSBzdWJzYW1wbGluZyBncmlkLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0U3Vic2FtcGxpbmdYT2Zmc2V0KCkgewotICAgICAgICByZXR1cm4gc3Vic2FtcGxpbmdYT2Zmc2V0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHZlcnRpY2FsIG9mZnNldCBvZiB0aGUgc3Vic2FtcGxpbmcgZ3JpZC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB2ZXJ0aWNhbCBvZmZzZXQgb2YgdGhlIHN1YnNhbXBsaW5nIGdyaWQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRTdWJzYW1wbGluZ1lPZmZzZXQoKSB7Ci0gICAgICAgIHJldHVybiBzdWJzYW1wbGluZ1lPZmZzZXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgaW5kaWNlcyBvZiB0aGUgc291cmNlIGJhbmRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2VCYW5kcwotICAgICAqICAgICAgICAgICAgdGhlIGluZGljZXMgb2YgdGhlIHNvdXJjZSBiYW5kcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRTb3VyY2VCYW5kcyhpbnRbXSBzb3VyY2VCYW5kcykgewotICAgICAgICAvLyBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhcnJheSBvZiBzb3VyY2UgYmFuZHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2Ygc291cmNlIGJhbmRzLgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXRTb3VyY2VCYW5kcygpIHsKLSAgICAgICAgLy8gVE9ETyBpbXBsZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJub3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc3BlY2lmaWVkIEltYWdlVHlwZVNwZWNpZmllciBmb3IgdGhlIGRlc3RpbmF0aW9uIGltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkZXN0aW5hdGlvblR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RGVzdGluYXRpb25UeXBlKEltYWdlVHlwZVNwZWNpZmllciBkZXN0aW5hdGlvblR5cGUpIHsKLSAgICAgICAgLy8gVE9ETyBpbXBsZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJub3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdHlwZSBvZiB0aGUgZGVzdGluYXRpb24gaW1hZ2UgYXMgYW4gSW1hZ2VUeXBlU3BlY2lmaWVyLiAuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VUeXBlU3BlY2lmaWVyLgotICAgICAqLwotICAgIHB1YmxpYyBJbWFnZVR5cGVTcGVjaWZpZXIgZ2V0RGVzdGluYXRpb25UeXBlKCkgewotICAgICAgICAvLyBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBvZmZzZXQgaW4gdGhlIGRlc3RpbmF0aW9uIGltYWdlIHdoZXJlIHRoZSBkZWNvZGVkIHBpeGVscyBhcmUKLSAgICAgKiBwbGFjZWQgYXMgYSByZXN1bHQgb2YgcmVhZGluZywgb3Igc3BlY2lmaWVkIGFuIGFyZWEgdG8gYmUgd3JpdHRlbiB3aGlsZQotICAgICAqIHdyaXRpbmcgb3BlcmF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBkZXN0aW5hdGlvbk9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIGRlc3RpbmF0aW9uIG9mZnNldC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXREZXN0aW5hdGlvbk9mZnNldChQb2ludCBkZXN0aW5hdGlvbk9mZnNldCkgewotICAgICAgICBpZiAoZGVzdGluYXRpb25PZmZzZXQgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiZGVzdGluYXRpb25PZmZzZXQgPT0gbnVsbCEiKTsKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMuZGVzdGluYXRpb25PZmZzZXQgPSAoUG9pbnQpZGVzdGluYXRpb25PZmZzZXQuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBvZmZzZXQgaW4gdGhlIGRlc3RpbmF0aW9uIGltYWdlIGZvciBwbGFjaW5nIHBpeGVscy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBvZmZzZXQgaW4gdGhlIGRlc3RpbmF0aW9uIGltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBQb2ludCBnZXREZXN0aW5hdGlvbk9mZnNldCgpIHsKLSAgICAgICAgcmV0dXJuIChQb2ludClkZXN0aW5hdGlvbk9mZnNldC5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIElJT1BhcmFtQ29udHJvbGxlciB0byB0aGlzIElJT1BhcmFtIG9iamVjdCBmb3IgcHJvdmlkaW5nCi0gICAgICogc2V0dGluZ3MgdG8gdGhpcyBJSU9QYXJhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29udHJvbGxlcgotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBJSU9QYXJhbUNvbnRyb2xsZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Q29udHJvbGxlcihJSU9QYXJhbUNvbnRyb2xsZXIgY29udHJvbGxlcikgewotICAgICAgICAvLyBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBjdXJyZW50IElJT1BhcmFtQ29udHJvbGxlciBjb250cm9sbGVyIGZvciB0aGlzIElJT1BhcmFtLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGN1cnJlbnQgSUlPUGFyYW1Db250cm9sbGVyIGNvbnRyb2xsZXIgZm9yIHRoaXMgSUlPUGFyYW0uCi0gICAgICovCi0gICAgcHVibGljIElJT1BhcmFtQ29udHJvbGxlciBnZXRDb250cm9sbGVyKCkgewotICAgICAgICAvLyBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkZWZhdWx0IElJT1BhcmFtQ29udHJvbGxlciBjb250cm9sbGVyIGZvciB0aGlzIElJT1BhcmFtLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRlZmF1bHQgSUlPUGFyYW1Db250cm9sbGVyIGNvbnRyb2xsZXIgZm9yIHRoaXMgSUlPUGFyYW0sIG9yCi0gICAgICogICAgICAgICBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBJSU9QYXJhbUNvbnRyb2xsZXIgZ2V0RGVmYXVsdENvbnRyb2xsZXIoKSB7Ci0gICAgICAgIC8vIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigibm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiBJSU9QYXJhbUNvbnRyb2xsZXIgaXMgaW5zdGFsbGVkIGZvciB0aGlzIElJT1BhcmFtLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgSUlPUGFyYW1Db250cm9sbGVyIGlzIGluc3RhbGxlZCBmb3IgdGhpcyBJSU9QYXJhbSwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBoYXNDb250cm9sbGVyKCkgewotICAgICAgICAvLyBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBY3RpdmF0ZXMgdGhlIGNvbnRyb2xsZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gYWN0aXZhdGVDb250cm9sbGVyKCkgewotICAgICAgICAvLyBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9JSU9QYXJhbUNvbnRyb2xsZXIuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL0lJT1BhcmFtQ29udHJvbGxlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzMzhjYjI1Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL0lJT1BhcmFtQ29udHJvbGxlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBTZXJnZXkgSS4gU2FsaXNoZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4yICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW87Ci0KLS8qIAotICogQGF1dGhvciBTZXJnZXkgSS4gU2FsaXNoZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4yICQKLSAqLwotCi0vKioKLSAqIFRoZSBJSU9QYXJhbUNvbnRyb2xsZXIgc3BlY2lmaWVzIGFuIGFjdGl2YXRlIG1ldGhvZCB0aGF0IGludm9rZXMgdGhlCi0gKiBjb250cm9sbGVyLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJSU9QYXJhbUNvbnRyb2xsZXIgewotCi0gICAgLyoqCi0gICAgICogQWN0aXZhdGVzIHRoZSBjb250cm9sbGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIElJT1BhcmFtLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIElJT1BhcmFtIGhhcyBiZWVuIG1vZGlmaWVkLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgYm9vbGVhbiBhY3RpdmF0ZShJSU9QYXJhbSBwYXJhbSk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9JbWFnZUlPLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9JbWFnZUlPLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGUwZDdlYzkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vSW1hZ2VJTy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsODAwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW87Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLnN0cmVhbS5JbWFnZUlucHV0U3RyZWFtOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3RyZWFtLkltYWdlT3V0cHV0U3RyZWFtOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLio7Ci1pbXBvcnQgamF2YS5pby5GaWxlOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLk91dHB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLnV0aWwuSXRlcmF0b3I7Ci1pbXBvcnQgamF2YS51dGlsLkFycmF5czsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJlbmRlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5uZXQuVVJMOwotCi0vKioKLSAqIFRoZSBJbWFnZUlPIGNsYXNzIHByb3ZpZGVzIHN0YXRpYyBtZXRob2RzIHRvIHBlcmZvcm0gcmVhZGluZyBhbmQgd3JpdGluZwotICogb3BlcmF0aW9ucyB1c2luZyByZWdpc3RlcmVkIEltYWdlUmVhZGVyIGFuZCBJbWFnZVdyaXRlciBvYmplY3RzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIEltYWdlSU8gewotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbnN0YW50IHJlZ2lzdHJ5LgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIElJT1JlZ2lzdHJ5IHJlZ2lzdHJ5ID0gSUlPUmVnaXN0cnkuZ2V0RGVmYXVsdEluc3RhbmNlKCk7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VJTy4KLSAgICAgKi8KLSAgICBwcml2YXRlIEltYWdlSU8oKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2NhbnMgZm9yIHBsdWctaW5zIGluIHRoZSBjbGFzcyBwYXRoLCBsb2FkcyBzcGkgY2xhc3NlcywgYW5kIHJlZ2lzdGVycwotICAgICAqIHRoZW0gd2l0aCB0aGUgSUlPUmVnaXN0cnkuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyB2b2lkIHNjYW5Gb3JQbHVnaW5zKCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBmbGFnIHdoaWNoIGluZGljYXRlcyB3aGV0aGVyIGEgY2FjaGUgZmlsZSBpcyB1c2VkIHdoZW4gY3JlYXRpbmcKLSAgICAgKiBJbWFnZUlucHV0U3RyZWFtcyBhbmQgSW1hZ2VPdXRwdXRTdHJlYW1zIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdXNlQ2FjaGUKLSAgICAgKiAgICAgICAgICAgIHRoZSB1c2UgY2FjaGUgZmxhZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgc2V0VXNlQ2FjaGUoYm9vbGVhbiB1c2VDYWNoZSkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZmxhZyB3aGljaCBpbmRpY2F0ZXMgd2hldGhlciBhIGNhY2hlIGZpbGUgaXMgdXNlZCB3aGVuIGNyZWF0aW5nCi0gICAgICogSW1hZ2VJbnB1dFN0cmVhbXMgYW5kIEltYWdlT3V0cHV0U3RyZWFtcyBvciBub3QuIFRoaXMgbWV0aG9kIHJldHVybnMgdGhlCi0gICAgICogY3VycmVudCB2YWx1ZSB3aGljaCBpcyBzZXQgYnkgc2V0VXNlQ2FjaGUgbWV0aG9kLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHVzZSBjYWNoZSBmbGFnLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgYm9vbGVhbiBnZXRVc2VDYWNoZSgpIHsKLSAgICAgICAgLy8gVE9ETyBpbXBsZW1lbnQKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGNhY2hlIGRpcmVjdG9yeS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2FjaGVEaXJlY3RvcnkKLSAgICAgKiAgICAgICAgICAgIHRoZSBGaWxlIHdoaWNoIHNwZWNpZmllcyBhIGNhY2hlIGRpcmVjdG9yeS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgc2V0Q2FjaGVEaXJlY3RvcnkoRmlsZSBjYWNoZURpcmVjdG9yeSkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGlyZWN0b3J5IHdoZXJlIGNhY2hlIGZpbGVzIGFyZSBjcmVhdGVkLCByZXR1cm5lZCB0aGUgZmlsZSB3aGljaAotICAgICAqIGlzIHNldCBieSBzZXRDYWNoZURpcmVjdG9yeSBtZXRob2QsIG9yIG51bGwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgRmlsZSBvYmplY3Qgd2hpY2ggaXMgc2V0IGJ5IHNldENhY2hlRGlyZWN0b3J5IG1ldGhvZCwgb3IKLSAgICAgKiAgICAgICAgIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBGaWxlIGdldENhY2hlRGlyZWN0b3J5KCkgewotICAgICAgICAvLyBUT0RPIGltcGxlbWVudAotICAgICAgICAvLyAtLSBudWxsIGluZGljYXRlcyBzeXN0ZW0tZGVwIGRlZmF1bHQgdGVtcG9yYXJ5IGRpcmVjdG9yeQotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGFuIEltYWdlSW5wdXRTdHJlYW0gZnJvbSB0aGUgc3BlY2lmaWVkIE9iamVjdC4gVGhlIHNwZWNpZmllZAotICAgICAqIE9iamVjdCBzaG91bGQgb2J0YWluIHRoZSBpbnB1dCBzb3VyY2Ugc3VjaCBhcyBGaWxlLCBvciBJbnB1dFN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5wdXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnB1dCBPYmplY3Qgc3VjaCBhcyBGaWxlLCBvciBJbnB1dFN0cmVhbS4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZUlucHV0U3RyZWFtIG9iamVjdCwgb3IgbnVsbC4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJbWFnZUlucHV0U3RyZWFtIGNyZWF0ZUltYWdlSW5wdXRTdHJlYW0oT2JqZWN0IGlucHV0KSB0aHJvd3MgSU9FeGNlcHRpb24gewotCi0gICAgICAgIGlmIChpbnB1dCA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJpbnB1dCBzb3VyY2UgY2Fubm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotCi0gICAgICAgIEl0ZXJhdG9yPEltYWdlSW5wdXRTdHJlYW1TcGk+IGl0ID0gcmVnaXN0cnkuZ2V0U2VydmljZVByb3ZpZGVycyhJbWFnZUlucHV0U3RyZWFtU3BpLmNsYXNzLAotICAgICAgICAgICAgICAgIHRydWUpOwotCi0gICAgICAgIHdoaWxlIChpdC5oYXNOZXh0KCkpIHsKLSAgICAgICAgICAgIEltYWdlSW5wdXRTdHJlYW1TcGkgc3BpID0gaXQubmV4dCgpOwotICAgICAgICAgICAgaWYgKHNwaS5nZXRJbnB1dENsYXNzKCkuaXNJbnN0YW5jZShpbnB1dCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gc3BpLmNyZWF0ZUlucHV0U3RyZWFtSW5zdGFuY2UoaW5wdXQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYW4gSW1hZ2VPdXRwdXRTdHJlYW0gdXNpbmcgdGhlIHNwZWNpZmllZCBPYmplY3QuIFRoZSBzcGVjaWZpZWQKLSAgICAgKiBPYmplY3Qgc2hvdWxkIG9idGFpbiB0aGUgb3V0cHV0IHNvdXJjZSBzdWNoIGFzIEZpbGUsIG9yIE91dHB1dFN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb3V0cHV0Ci0gICAgICogICAgICAgICAgICB0aGUgb3V0cHV0IE9iamVjdCBzdWNoIGFzIEZpbGUsIG9yIE91dHB1dFN0cmVhbS4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZU91dHB1dFN0cmVhbSBvYmplY3QsIG9yIG51bGwuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgSW1hZ2VPdXRwdXRTdHJlYW0gY3JlYXRlSW1hZ2VPdXRwdXRTdHJlYW0oT2JqZWN0IG91dHB1dCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKG91dHB1dCA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJvdXRwdXQgZGVzdGluYXRpb24gY2Fubm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotCi0gICAgICAgIEl0ZXJhdG9yPEltYWdlT3V0cHV0U3RyZWFtU3BpPiBpdCA9IHJlZ2lzdHJ5LmdldFNlcnZpY2VQcm92aWRlcnMoCi0gICAgICAgICAgICAgICAgSW1hZ2VPdXRwdXRTdHJlYW1TcGkuY2xhc3MsIHRydWUpOwotCi0gICAgICAgIHdoaWxlIChpdC5oYXNOZXh0KCkpIHsKLSAgICAgICAgICAgIEltYWdlT3V0cHV0U3RyZWFtU3BpIHNwaSA9IGl0Lm5leHQoKTsKLSAgICAgICAgICAgIGlmIChzcGkuZ2V0T3V0cHV0Q2xhc3MoKS5pc0luc3RhbmNlKG91dHB1dCkpIHsKLSAgICAgICAgICAgICAgICAvLyB0b2RvIC0gdXNlIGdldFVzZUNhY2hlIGFuZCBnZXRDYWNoZURpciBoZXJlCi0gICAgICAgICAgICAgICAgcmV0dXJuIHNwaS5jcmVhdGVPdXRwdXRTdHJlYW1JbnN0YW5jZShvdXRwdXQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFycmF5IG9mIGZvcm1hdCBuYW1lcyBhcyBTdHJpbmcgd2hpY2ggY2FuIGJlIGRlY29kZWQgYnkKLSAgICAgKiByZWdpc3RlcmVkIEltYWdlUmVhZGVyIG9iamVjdHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgZm9ybWF0IG5hbWVzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgU3RyaW5nW10gZ2V0UmVhZGVyRm9ybWF0TmFtZXMoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhcnJheSBvZiBNSU1FIHR5cGVzIGFzIFN0cmluZyB3aGljaCBjYW4gYmUgZGVjb2RlZCBieSByZWdpc3RlcmVkCi0gICAgICogSW1hZ2VSZWFkZXIgb2JqZWN0cy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBNSU1FIHR5cGVzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgU3RyaW5nW10gZ2V0UmVhZGVyTUlNRVR5cGVzKCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgSXRlcmF0b3Igb2YgcmVnaXN0ZXJlZCBJbWFnZVJlYWRlciB3aGljaCBhcmUgYWJsZSB0byBkZWNvZGUgYW4KLSAgICAgKiBpbnB1dCBkYXRhIHNwZWNpZmllZCBieSBpbnB1dCBPYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGlucHV0Ci0gICAgICogICAgICAgICAgICB0aGUgaW5wdXQgT2JqZWN0IHdpdGggZW5jb2RlZCBkYXRhIHN1Y2ggYXMgSW1hZ2VJbnB1dFN0cmVhbQotICAgICAqICAgICAgICAgICAgb2JqZWN0LgotICAgICAqIEByZXR1cm4gdGhlIEl0ZXJhdG9yIG9mIHJlZ2lzdGVyZWQgSW1hZ2VSZWFkZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJdGVyYXRvcjxJbWFnZVJlYWRlcj4gZ2V0SW1hZ2VSZWFkZXJzKE9iamVjdCBpbnB1dCkgewotICAgICAgICBpZiAoaW5wdXQgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJpbnB1dCBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgSXRlcmF0b3I8SW1hZ2VSZWFkZXJTcGk+IGl0ID0gcmVnaXN0cnkuZ2V0U2VydmljZVByb3ZpZGVycyhJbWFnZVJlYWRlclNwaS5jbGFzcywKLSAgICAgICAgICAgICAgICBuZXcgQ2FuUmVhZEZpbHRlcihpbnB1dCksIHRydWUpOwotCi0gICAgICAgIHJldHVybiBuZXcgU3BpSXRlcmF0b3JUb1JlYWRlcnNJdGVyYXRvcldyYXBwZXIoaXQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEl0ZXJhdG9yIG9mIHJlZ2lzdGVyZWQgSW1hZ2VSZWFkZXIgd2hpY2ggYXJlIGFibGUgdG8gZGVjb2RlIHRoZQotICAgICAqIHNwZWNpZmllZCBmb3JtYXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZvcm1hdE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBmb3JtYXQgbmFtZSBzdWNoIGFzICJqcGVnIiwgb3IgImdpZiIuCi0gICAgICogQHJldHVybiB0aGUgSXRlcmF0b3Igb2YgcmVnaXN0ZXJlZCBJbWFnZVJlYWRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEl0ZXJhdG9yPEltYWdlUmVhZGVyPiBnZXRJbWFnZVJlYWRlcnNCeUZvcm1hdE5hbWUoU3RyaW5nIGZvcm1hdE5hbWUpIHsKLSAgICAgICAgaWYgKGZvcm1hdE5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJmb3JtYXQgbmFtZSBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgSXRlcmF0b3I8SW1hZ2VSZWFkZXJTcGk+IGl0ID0gcmVnaXN0cnkuZ2V0U2VydmljZVByb3ZpZGVycyhJbWFnZVJlYWRlclNwaS5jbGFzcywKLSAgICAgICAgICAgICAgICBuZXcgRm9ybWF0RmlsdGVyKGZvcm1hdE5hbWUpLCB0cnVlKTsKLQotICAgICAgICByZXR1cm4gbmV3IFNwaUl0ZXJhdG9yVG9SZWFkZXJzSXRlcmF0b3JXcmFwcGVyKGl0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBJdGVyYXRvciB3aGljaCBsaXN0cyB0aGUgcmVnaXN0ZXJlZCBJbWFnZVJlYWRlciBvYmplY3RzIHRoYXQgYXJlCi0gICAgICogYWJsZSB0byBkZWNvZGUgZmlsZXMgd2l0aCB0aGUgc3BlY2lmaWVkIHN1ZmZpeC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZmlsZVN1ZmZpeAotICAgICAqICAgICAgICAgICAgdGhlIGZpbGUgc3VmZml4IHN1Y2ggYXMgImpwZyIuCi0gICAgICogQHJldHVybiB0aGUgSXRlcmF0b3Igb2YgcmVnaXN0ZXJlZCBJbWFnZVJlYWRlcnMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJdGVyYXRvcjxJbWFnZVJlYWRlcj4gZ2V0SW1hZ2VSZWFkZXJzQnlTdWZmaXgoU3RyaW5nIGZpbGVTdWZmaXgpIHsKLSAgICAgICAgaWYgKGZpbGVTdWZmaXggPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJzdWZmaXggY2Fubm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotICAgICAgICBJdGVyYXRvcjxJbWFnZVJlYWRlclNwaT4gaXQgPSByZWdpc3RyeS5nZXRTZXJ2aWNlUHJvdmlkZXJzKEltYWdlUmVhZGVyU3BpLmNsYXNzLAotICAgICAgICAgICAgICAgIG5ldyBTdWZmaXhGaWx0ZXIoZmlsZVN1ZmZpeCksIHRydWUpOwotCi0gICAgICAgIHJldHVybiBuZXcgU3BpSXRlcmF0b3JUb1JlYWRlcnNJdGVyYXRvcldyYXBwZXIoaXQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEl0ZXJhdG9yIG9mIHJlZ2lzdGVyZWQgSW1hZ2VSZWFkZXIgb2JqZWN0cyB0aGF0IGFyZSBhYmxlIHRvCi0gICAgICogZGVjb2RlIGZpbGVzIHdpdGggdGhlIHNwZWNpZmllZCBNSU1FIHR5cGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIE1JTUVUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgTUlNRSB0eXBlIHN1Y2ggYXMgImltYWdlL2pwZWciLgotICAgICAqIEByZXR1cm4gdGhlIEl0ZXJhdG9yIG9mIHJlZ2lzdGVyZWQgSW1hZ2VSZWFkZXJzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgSXRlcmF0b3I8SW1hZ2VSZWFkZXI+IGdldEltYWdlUmVhZGVyc0J5TUlNRVR5cGUoU3RyaW5nIE1JTUVUeXBlKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIFN0cmluZ3MgZ2l2aW5nIHRoZSBuYW1lcyBvZiB0aGUgZm9ybWF0cyBzdXBwb3J0ZWQgYnkKLSAgICAgKiByZWdpc3RlcmVkIEltYWdlV3JpdGVyIG9iamVjdHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgZm9ybWF0IG5hbWVzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgU3RyaW5nW10gZ2V0V3JpdGVyRm9ybWF0TmFtZXMoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIFN0cmluZ3MgZ2l2aW5nIHRoZSBNSU1FIHR5cGVzIG9mIHRoZSBmb3JtYXRzIHN1cHBvcnRlZAotICAgICAqIGJ5IHJlZ2lzdGVyZWQgSW1hZ2VXcml0ZXIgb2JqZWN0cy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBNSU1FIHR5cGVzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgU3RyaW5nW10gZ2V0V3JpdGVyTUlNRVR5cGVzKCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgSXRlcmF0b3Igd2hpY2ggbGlzdHMgdGhlIHJlZ2lzdGVyZWQgSW1hZ2VSZWFkZXIgb2JqZWN0cyB0aGF0IGFyZQotICAgICAqIGFibGUgdG8gZW5jb2RlIHRoZSBzcGVjaWZpZWQgaW1hZ2UgZm9ybWF0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgZm9ybWF0IG5hbWUgc3VjaCBhcyAianBlZyIuCi0gICAgICogQHJldHVybiB0aGUgSXRlcmF0b3Igb2YgcmVnaXN0ZXJlZCBJbWFnZVdyaXRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEl0ZXJhdG9yPEltYWdlV3JpdGVyPiBnZXRJbWFnZVdyaXRlcnNCeUZvcm1hdE5hbWUoU3RyaW5nIGZvcm1hdE5hbWUpIHsKLSAgICAgICAgaWYgKGZvcm1hdE5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJmb3JtYXQgbmFtZSBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgSXRlcmF0b3I8SW1hZ2VXcml0ZXJTcGk+IGl0ID0gcmVnaXN0cnkuZ2V0U2VydmljZVByb3ZpZGVycyhJbWFnZVdyaXRlclNwaS5jbGFzcywKLSAgICAgICAgICAgICAgICBuZXcgRm9ybWF0RmlsdGVyKGZvcm1hdE5hbWUpLCB0cnVlKTsKLQotICAgICAgICByZXR1cm4gbmV3IFNwaUl0ZXJhdG9yVG9Xcml0ZXJzSXRlcmF0b3JXcmFwcGVyKGl0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBJdGVyYXRvciB3aGljaCBsaXN0cyB0aGUgcmVnaXN0ZXJlZCBJbWFnZVJlYWRlciBvYmplY3RzIHRoYXQgYXJlCi0gICAgICogYWJsZSB0byBlbmNvZGUgdGhlIHNwZWNpZmllZCBzdWZmaXguCi0gICAgICogCi0gICAgICogQHBhcmFtIGZpbGVTdWZmaXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBmaWxlIHN1ZmZpeCBzdWNoIGFzICJqcGciLgotICAgICAqIEByZXR1cm4gdGhlIEl0ZXJhdG9yIG9mIHJlZ2lzdGVyZWQgSW1hZ2VXcml0ZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJdGVyYXRvcjxJbWFnZVdyaXRlcj4gZ2V0SW1hZ2VXcml0ZXJzQnlTdWZmaXgoU3RyaW5nIGZpbGVTdWZmaXgpIHsKLSAgICAgICAgaWYgKGZpbGVTdWZmaXggPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJzdWZmaXggY2Fubm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotICAgICAgICBJdGVyYXRvcjxJbWFnZVdyaXRlclNwaT4gaXQgPSByZWdpc3RyeS5nZXRTZXJ2aWNlUHJvdmlkZXJzKEltYWdlV3JpdGVyU3BpLmNsYXNzLAotICAgICAgICAgICAgICAgIG5ldyBTdWZmaXhGaWx0ZXIoZmlsZVN1ZmZpeCksIHRydWUpOwotICAgICAgICByZXR1cm4gbmV3IFNwaUl0ZXJhdG9yVG9Xcml0ZXJzSXRlcmF0b3JXcmFwcGVyKGl0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBJdGVyYXRvciB3aGljaCBsaXN0cyB0aGUgcmVnaXN0ZXJlZCBJbWFnZVJlYWRlciBvYmplY3RzIHRoYXQgYXJlCi0gICAgICogYWJsZSB0byBlbmNvZGUgdGhlIHNwZWNpZmllZCBNSU1FIHR5cGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIE1JTUVUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgTUlNRSB0eXBlIHN1Y2ggYXMgImltYWdlL2pwZWciLgotICAgICAqIEByZXR1cm4gdGhlIEl0ZXJhdG9yIG9mIHJlZ2lzdGVyZWQgSW1hZ2VXcml0ZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJdGVyYXRvcjxJbWFnZVdyaXRlcj4gZ2V0SW1hZ2VXcml0ZXJzQnlNSU1FVHlwZShTdHJpbmcgTUlNRVR5cGUpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gSW1hZ2VXcml0ZXIgb2JqZWN0IHdoaWNoIGNvcnJlc3BvbmRzIHRvIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBJbWFnZVJlYWRlciwgb3IgcmV0dXJucyBudWxsIGlmIHRoZSBzcGVjaWZpZWQgSW1hZ2VSZWFkZXIgaXMgbm90Ci0gICAgICogcmVnaXN0ZXJlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmVhZGVyCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIEltYWdlUmVhZGVyLgotICAgICAqIEByZXR1cm4gdGhlIEltYWdlV3JpdGVyLCBvciBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgSW1hZ2VXcml0ZXIgZ2V0SW1hZ2VXcml0ZXIoSW1hZ2VSZWFkZXIgcmVhZGVyKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIEltYWdlUmVhZGVyIG9iamVjdCB3aGljaCBjb3JyZXNwb25kcyB0byB0aGUgc3BlY2lmaWVkCi0gICAgICogSW1hZ2VXcml0ZXIsIG9yIHJldHVybnMgbnVsbCBpZiB0aGUgc3BlY2lmaWVkIEltYWdlV3JpdGVyIGlzIG5vdAotICAgICAqIHJlZ2lzdGVyZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdyaXRlcgotICAgICAqICAgICAgICAgICAgdGhlIHJlZ2lzdGVyZWQgSW1hZ2VXcml0ZXIgb2JqZWN0LgotICAgICAqIEByZXR1cm4gdGhlIEltYWdlUmVhZGVyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgSW1hZ2VSZWFkZXIgZ2V0SW1hZ2VSZWFkZXIoSW1hZ2VXcml0ZXIgd3JpdGVyKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBJdGVyYXRvciBvZiBJbWFnZVdyaXRlciBvYmplY3RzIHdoaWNoIGFyZSBhYmxlIHRvIGVuY29kZSBpbWFnZXMKLSAgICAgKiB3aXRoIHRoZSBzcGVjaWZpZWQgSW1hZ2VUeXBlU3BlY2lmaWVyIGFuZCBmb3JtYXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVR5cGVTcGVjaWZpZXIsIHdoaWNoIGRlZmluZXMgbGF5b3V0LgotICAgICAqIEBwYXJhbSBmb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZm9ybWF0IG5hbWUuCi0gICAgICogQHJldHVybiB0aGUgSXRlcmF0b3Igb2YgSW1hZ2VXcml0ZXIgb2JqZWN0cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEl0ZXJhdG9yPEltYWdlV3JpdGVyPiBnZXRJbWFnZVdyaXRlcnMoSW1hZ2VUeXBlU3BlY2lmaWVyIHR5cGUsIFN0cmluZyBmb3JtYXROYW1lKSB7Ci0gICAgICAgIGlmICh0eXBlID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbigidHlwZSBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGZvcm1hdE5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJmb3JtYXQgbmFtZSBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgSXRlcmF0b3I8SW1hZ2VXcml0ZXJTcGk+IGl0ID0gcmVnaXN0cnkuZ2V0U2VydmljZVByb3ZpZGVycyhJbWFnZVdyaXRlclNwaS5jbGFzcywKLSAgICAgICAgICAgICAgICBuZXcgRm9ybWF0QW5kRW5jb2RlRmlsdGVyKHR5cGUsIGZvcm1hdE5hbWUpLCB0cnVlKTsKLQotICAgICAgICByZXR1cm4gbmV3IFNwaUl0ZXJhdG9yVG9Xcml0ZXJzSXRlcmF0b3JXcmFwcGVyKGl0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBJdGVyYXRvciBvZiByZWdpc3RlcmVkIEltYWdlVHJhbnNjb2RlcnMgd2hpY2ggYXJlIGFibGUgdG8KLSAgICAgKiB0cmFuc2NvZGUgdGhlIG1ldGFkYXRhIG9mIHRoZSBzcGVjaWZpZWQgSW1hZ2VSZWFkZXIgb2JqZWN0IHRvIGEgc3VpdGFibGUKLSAgICAgKiBvYmplY3QgZm9yIGVuY29kaW5nIGJ5IHRoZSBzcGVjaWZpZWQgSW1hZ2VXcml0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJlYWRlcgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBJbWFnZVJlYWRlci4KLSAgICAgKiBAcGFyYW0gd3JpdGVyCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIEltYWdlV3JpdGVyLgotICAgICAqIEByZXR1cm4gdGhlIEl0ZXJhdG9yIG9mIHJlZ2lzdGVyZWQgSW1hZ2VUcmFuc2NvZGVycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEl0ZXJhdG9yPEltYWdlVHJhbnNjb2Rlcj4gZ2V0SW1hZ2VUcmFuc2NvZGVycyhJbWFnZVJlYWRlciByZWFkZXIsCi0gICAgICAgICAgICBJbWFnZVdyaXRlciB3cml0ZXIpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlYWRzIGltYWdlIGRhdGEgZnJvbSB0aGUgc3BlY2lmaWVkIEZpbGUgYW5kIGRlY29kZXMgaXQgdXNpbmcgdGhlCi0gICAgICogYXBwcm9wcmlhdGUgcmVnaXN0ZXJlZCBJbWFnZVJlYWRlciBvYmplY3QuIFRoZSBGaWxlIGlzIHdyYXBwZWQgaW4gYW4KLSAgICAgKiBJbWFnZUlucHV0U3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbnB1dAotICAgICAqICAgICAgICAgICAgdGhlIEZpbGUgdG8gYmUgcmVhZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBCdWZmZXJlZEltYWdlIGRlY29kZWQgZnJvbSB0aGUgc3BlY2lmaWVkIEZpbGUsIG9yIG51bGwuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgQnVmZmVyZWRJbWFnZSByZWFkKEZpbGUgaW5wdXQpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChpbnB1dCA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJpbnB1dCA9PSBudWxsISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgSW1hZ2VJbnB1dFN0cmVhbSBzdHJlYW0gPSBjcmVhdGVJbWFnZUlucHV0U3RyZWFtKGlucHV0KTsKLSAgICAgICAgcmV0dXJuIHJlYWQoc3RyZWFtKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyBpbWFnZSBkYXRhIGZyb20gdGhlIHNwZWNpZmllZCBJbnB1dFN0cmVhbSBhbmQgZGVjb2RlcyBpdCB1c2luZyBhbgotICAgICAqIGFwcHJvcHJpYXRlIHJlZ2lzdGVyZWQgYW4gSW1hZ2VSZWFkZXIgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbnB1dAotICAgICAqICAgICAgICAgICAgdGhlIElucHV0U3RyZWFtLgotICAgICAqIEByZXR1cm4gdGhlIEJ1ZmZlcmVkSW1hZ2UgZGVjb2RlZCBmcm9tIHRoZSBzcGVjaWZpZWQgSW5wdXRTdHJlYW0sIG9yCi0gICAgICogICAgICAgICBudWxsLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEJ1ZmZlcmVkSW1hZ2UgcmVhZChJbnB1dFN0cmVhbSBpbnB1dCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKGlucHV0ID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImlucHV0ID09IG51bGwhIik7Ci0gICAgICAgIH0KLQotICAgICAgICBJbWFnZUlucHV0U3RyZWFtIHN0cmVhbSA9IGNyZWF0ZUltYWdlSW5wdXRTdHJlYW0oaW5wdXQpOwotICAgICAgICByZXR1cm4gcmVhZChzdHJlYW0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlYWRzIGltYWdlIGRhdGEgZnJvbSB0aGUgc3BlY2lmaWVkIFVSTCBhbmQgZGVjb2RlcyBpdCB1c2luZyB0aGUKLSAgICAgKiBhcHByb3ByaWF0ZSByZWdpc3RlcmVkIEltYWdlUmVhZGVyIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5wdXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBVUkwgdG8gYmUgcmVhZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBCdWZmZXJlZEltYWdlIGRlY29kZWQgZnJvbSB0aGUgc3BlY2lmaWVkIFVSTCwgb3IgbnVsbC4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBCdWZmZXJlZEltYWdlIHJlYWQoVVJMIGlucHV0KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAoaW5wdXQgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiaW5wdXQgPT0gbnVsbCEiKTsKLSAgICAgICAgfQotCi0gICAgICAgIElucHV0U3RyZWFtIHN0cmVhbSA9IGlucHV0Lm9wZW5TdHJlYW0oKTsKLSAgICAgICAgQnVmZmVyZWRJbWFnZSByZXMgPSByZWFkKHN0cmVhbSk7Ci0gICAgICAgIHN0cmVhbS5jbG9zZSgpOwotCi0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVhZHMgaW1hZ2UgZGF0YSBmcm9tIHRoZSBzcGVjaWZpZWQgSW1hZ2VJbnB1dFN0cmVhbSBhbmQgZGVjb2RlcyBpdCB1c2luZwotICAgICAqIGFwcHJvcHJpYXRlIHJlZ2lzdGVyZWQgYW4gSW1hZ2VSZWFkZXIgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHJlYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZUlucHV0U3RyZWFtLgotICAgICAqIEByZXR1cm4gdGhlIEJ1ZmZlcmVkSW1hZ2UgZGVjb2RlZCBmcm9tIHRoZSBzcGVjaWZpZWQgSW1hZ2VJbnB1dFN0cmVhbSwgb3IKLSAgICAgKiAgICAgICAgIG51bGwuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgQnVmZmVyZWRJbWFnZSByZWFkKEltYWdlSW5wdXRTdHJlYW0gc3RyZWFtKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAoc3RyZWFtID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInN0cmVhbSA9PSBudWxsISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgSXRlcmF0b3I8SW1hZ2VSZWFkZXI+IGltYWdlUmVhZGVycyA9IGdldEltYWdlUmVhZGVycyhzdHJlYW0pOwotICAgICAgICBpZiAoIWltYWdlUmVhZGVycy5oYXNOZXh0KCkpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgSW1hZ2VSZWFkZXIgcmVhZGVyID0gaW1hZ2VSZWFkZXJzLm5leHQoKTsKLSAgICAgICAgcmVhZGVyLnNldElucHV0KHN0cmVhbSwgZmFsc2UsIHRydWUpOwotICAgICAgICBCdWZmZXJlZEltYWdlIHJlcyA9IHJlYWRlci5yZWFkKDApOwotICAgICAgICByZWFkZXIuZGlzcG9zZSgpOwotCi0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBzdHJlYW0uY2xvc2UoKTsKLSAgICAgICAgfSBjYXRjaCAoSU9FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgLy8gU3RyZWFtIGNvdWxkIGJlIGFscmVhZHkgY2xvc2VkLCBwcm9jZWVkIHNpbGVudGx5IGluIHRoaXMgY2FzZQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcml0ZXMgdGhlIHNwZWNpZmllZCBpbWFnZSBpbiB0aGUgc3BlY2lmaWVkIGZvcm1hdCAodXNpbmcgYW4gYXBwcm9wcmlhdGUKLSAgICAgKiBJbWFnZVdyaXRlcikgdG8gdGhlIHNwZWNpZmllZCBJbWFnZU91dHB1dFN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJlZEltYWdlLgotICAgICAqIEBwYXJhbSBmb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZm9ybWF0IG5hbWUuCi0gICAgICogQHBhcmFtIG91dHB1dAotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlT3V0cHV0U3RyZWFtIHdoZXJlIEltYWdlIHRvIGJlIHdyaXR0ZW4uCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBJbWFnZSBpcyB3cml0dGVuIHN1Y2Nlc3NmdWxseSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGJvb2xlYW4gd3JpdGUoUmVuZGVyZWRJbWFnZSBpbSwgU3RyaW5nIGZvcm1hdE5hbWUsIEltYWdlT3V0cHV0U3RyZWFtIG91dHB1dCkKLSAgICAgICAgICAgIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0KLSAgICAgICAgaWYgKGltID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImltYWdlIGNhbm5vdCBiZSBOVUxMIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGZvcm1hdE5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiZm9ybWF0IG5hbWUgY2Fubm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotICAgICAgICBpZiAob3V0cHV0ID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIm91dHB1dCBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgSXRlcmF0b3I8SW1hZ2VXcml0ZXI+IGl0ID0gZ2V0SW1hZ2VXcml0ZXJzKEltYWdlVHlwZVNwZWNpZmllci5jcmVhdGVGcm9tUmVuZGVyZWRJbWFnZShpbSksCi0gICAgICAgICAgICAgICAgZm9ybWF0TmFtZSk7Ci0gICAgICAgIGlmIChpdC5oYXNOZXh0KCkpIHsKLSAgICAgICAgICAgIEltYWdlV3JpdGVyIHdyaXRlciA9IGl0Lm5leHQoKTsKLSAgICAgICAgICAgIHdyaXRlci5zZXRPdXRwdXQob3V0cHV0KTsKLSAgICAgICAgICAgIHdyaXRlci53cml0ZShpbSk7Ci0gICAgICAgICAgICBvdXRwdXQuZmx1c2goKTsKLSAgICAgICAgICAgIHdyaXRlci5kaXNwb3NlKCk7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIHRoZSBzcGVjaWZpZWQgaW1hZ2UgaW4gdGhlIHNwZWNpZmllZCBmb3JtYXQgKHVzaW5nIGFuIGFwcHJvcHJpYXRlCi0gICAgICogSW1hZ2VXcml0ZXIpIHRvIHRoZSBzcGVjaWZpZWQgRmlsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJlZEltYWdlLgotICAgICAqIEBwYXJhbSBmb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZm9ybWF0IG5hbWUuCi0gICAgICogQHBhcmFtIG91dHB1dAotICAgICAqICAgICAgICAgICAgdGhlIG91dHB1dCBGaWxlIHdoZXJlIEltYWdlIHRvIGJlIHdyaXR0ZW4uCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBJbWFnZSBpcyB3cml0dGVuIHN1Y2Nlc3NmdWxseSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGJvb2xlYW4gd3JpdGUoUmVuZGVyZWRJbWFnZSBpbSwgU3RyaW5nIGZvcm1hdE5hbWUsIEZpbGUgb3V0cHV0KQotICAgICAgICAgICAgdGhyb3dzIElPRXhjZXB0aW9uIHsKLQotICAgICAgICBpZiAob3V0cHV0ID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIm91dHB1dCBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG91dHB1dC5leGlzdHMoKSkgewotICAgICAgICAgICAgb3V0cHV0LmRlbGV0ZSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgSW1hZ2VPdXRwdXRTdHJlYW0gaW9zID0gY3JlYXRlSW1hZ2VPdXRwdXRTdHJlYW0ob3V0cHV0KTsKLSAgICAgICAgYm9vbGVhbiBydCA9IHdyaXRlKGltLCBmb3JtYXROYW1lLCBpb3MpOwotICAgICAgICBpb3MuY2xvc2UoKTsKLSAgICAgICAgcmV0dXJuIHJ0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFdyaXRlcyB0aGUgc3BlY2lmaWVkIGltYWdlIGluIHRoZSBzcGVjaWZpZWQgZm9ybWF0ICh1c2luZyBhbiBhcHByb3ByaWF0ZQotICAgICAqIEltYWdlV3JpdGVyKSB0byB0aGUgc3BlY2lmaWVkIE91dHB1dFN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJlZEltYWdlLgotICAgICAqIEBwYXJhbSBmb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZm9ybWF0IG5hbWUuCi0gICAgICogQHBhcmFtIG91dHB1dAotICAgICAqICAgICAgICAgICAgdGhlIE91dHB1dFN0cmVhbSB3aGVyZSBJbWFnZSBpcyB0byBiZSB3cml0dGVuLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgSW1hZ2UgaXMgd3JpdHRlbiBzdWNjZXNzZnVsbHksIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBib29sZWFuIHdyaXRlKFJlbmRlcmVkSW1hZ2UgaW0sIFN0cmluZyBmb3JtYXROYW1lLCBPdXRwdXRTdHJlYW0gb3V0cHV0KQotICAgICAgICAgICAgdGhyb3dzIElPRXhjZXB0aW9uIHsKLQotICAgICAgICBpZiAob3V0cHV0ID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIm91dHB1dCBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgSW1hZ2VPdXRwdXRTdHJlYW0gaW9zID0gY3JlYXRlSW1hZ2VPdXRwdXRTdHJlYW0ob3V0cHV0KTsKLSAgICAgICAgYm9vbGVhbiBydCA9IHdyaXRlKGltLCBmb3JtYXROYW1lLCBpb3MpOwotICAgICAgICBpb3MuY2xvc2UoKTsKLSAgICAgICAgcmV0dXJuIHJ0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbHRlciB0byBtYXRjaCBzcGkgYnkgZm9ybWF0IG5hbWUuCi0gICAgICovCi0gICAgc3RhdGljIGNsYXNzIEZvcm1hdEZpbHRlciBpbXBsZW1lbnRzIFNlcnZpY2VSZWdpc3RyeS5GaWx0ZXIgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbmFtZS4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgU3RyaW5nIG5hbWU7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBmb3JtYXQgZmlsdGVyLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIG5hbWUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgbmFtZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBGb3JtYXRGaWx0ZXIoU3RyaW5nIG5hbWUpIHsKLSAgICAgICAgICAgIHRoaXMubmFtZSA9IG5hbWU7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBmaWx0ZXIoT2JqZWN0IHByb3ZpZGVyKSB7Ci0gICAgICAgICAgICBJbWFnZVJlYWRlcldyaXRlclNwaSBzcGkgPSAoSW1hZ2VSZWFkZXJXcml0ZXJTcGkpcHJvdmlkZXI7Ci0gICAgICAgICAgICByZXR1cm4gQXJyYXlzLmFzTGlzdChzcGkuZ2V0Rm9ybWF0TmFtZXMoKSkuY29udGFpbnMobmFtZSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaWx0ZXIgdG8gbWF0Y2ggc3BpIGJ5IGZvcm1hdCBuYW1lIGFuZCBlbmNvZGluZyBwb3NzaWJpbGl0eS4KLSAgICAgKi8KLSAgICBzdGF0aWMgY2xhc3MgRm9ybWF0QW5kRW5jb2RlRmlsdGVyIGV4dGVuZHMgRm9ybWF0RmlsdGVyIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHR5cGUuCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIEltYWdlVHlwZVNwZWNpZmllciB0eXBlOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZm9ybWF0IGFuZCBlbmNvZGUgZmlsdGVyLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHR5cGUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgdHlwZS4KLSAgICAgICAgICogQHBhcmFtIG5hbWUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgbmFtZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBGb3JtYXRBbmRFbmNvZGVGaWx0ZXIoSW1hZ2VUeXBlU3BlY2lmaWVyIHR5cGUsIFN0cmluZyBuYW1lKSB7Ci0gICAgICAgICAgICBzdXBlcihuYW1lKTsKLSAgICAgICAgICAgIHRoaXMudHlwZSA9IHR5cGU7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGJvb2xlYW4gZmlsdGVyKE9iamVjdCBwcm92aWRlcikgewotICAgICAgICAgICAgSW1hZ2VXcml0ZXJTcGkgc3BpID0gKEltYWdlV3JpdGVyU3BpKXByb3ZpZGVyOwotICAgICAgICAgICAgcmV0dXJuIHN1cGVyLmZpbHRlcihwcm92aWRlcikgJiYgc3BpLmNhbkVuY29kZUltYWdlKHR5cGUpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRmlsdGVyIHRvIG1hdGNoIHNwaSBieSBzdWZmaXguCi0gICAgICovCi0gICAgc3RhdGljIGNsYXNzIFN1ZmZpeEZpbHRlciBpbXBsZW1lbnRzIFNlcnZpY2VSZWdpc3RyeS5GaWx0ZXIgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgc3VmLgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBTdHJpbmcgc3VmOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgc3VmZml4IGZpbHRlci4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBzdWYKLSAgICAgICAgICogICAgICAgICAgICB0aGUgc3VmLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIFN1ZmZpeEZpbHRlcihTdHJpbmcgc3VmKSB7Ci0gICAgICAgICAgICB0aGlzLnN1ZiA9IHN1ZjsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGZpbHRlcihPYmplY3QgcHJvdmlkZXIpIHsKLSAgICAgICAgICAgIEltYWdlUmVhZGVyV3JpdGVyU3BpIHNwaSA9IChJbWFnZVJlYWRlcldyaXRlclNwaSlwcm92aWRlcjsKLSAgICAgICAgICAgIHJldHVybiBBcnJheXMuYXNMaXN0KHNwaS5nZXRGaWxlU3VmZml4ZXMoKSkuY29udGFpbnMoc3VmKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbHRlciB0byBtYXRjaCBzcGkgYnkgZGVjb2RpbmcgcG9zc2liaWxpdHkuCi0gICAgICovCi0gICAgc3RhdGljIGNsYXNzIENhblJlYWRGaWx0ZXIgaW1wbGVtZW50cyBTZXJ2aWNlUmVnaXN0cnkuRmlsdGVyIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGlucHV0LgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBPYmplY3QgaW5wdXQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBjYW4gcmVhZCBmaWx0ZXIuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gaW5wdXQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgaW5wdXQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgQ2FuUmVhZEZpbHRlcihPYmplY3QgaW5wdXQpIHsKLSAgICAgICAgICAgIHRoaXMuaW5wdXQgPSBpbnB1dDsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGZpbHRlcihPYmplY3QgcHJvdmlkZXIpIHsKLSAgICAgICAgICAgIEltYWdlUmVhZGVyU3BpIHNwaSA9IChJbWFnZVJlYWRlclNwaSlwcm92aWRlcjsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHNwaS5jYW5EZWNvZGVJbnB1dChpbnB1dCk7Ci0gICAgICAgICAgICB9IGNhdGNoIChJT0V4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogV3JhcHMgU3BpJ3MgaXRlcmF0b3IgdG8gSW1hZ2VXcml0ZXIgaXRlcmF0b3IuCi0gICAgICovCi0gICAgc3RhdGljIGNsYXNzIFNwaUl0ZXJhdG9yVG9Xcml0ZXJzSXRlcmF0b3JXcmFwcGVyIGltcGxlbWVudHMgSXRlcmF0b3I8SW1hZ2VXcml0ZXI+IHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGJhY2tlbmQuCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIEl0ZXJhdG9yPEltYWdlV3JpdGVyU3BpPiBiYWNrZW5kOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgc3BpIGl0ZXJhdG9yIHRvIHdyaXRlcnMgaXRlcmF0b3Igd3JhcHBlci4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBiYWNrZW5kCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGJhY2tlbmQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgU3BpSXRlcmF0b3JUb1dyaXRlcnNJdGVyYXRvcldyYXBwZXIoSXRlcmF0b3I8SW1hZ2VXcml0ZXJTcGk+IGJhY2tlbmQpIHsKLSAgICAgICAgICAgIHRoaXMuYmFja2VuZCA9IGJhY2tlbmQ7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogTmV4dC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdGhlIGltYWdlIHdyaXRlci4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBJbWFnZVdyaXRlciBuZXh0KCkgewotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICByZXR1cm4gYmFja2VuZC5uZXh0KCkuY3JlYXRlV3JpdGVySW5zdGFuY2UoKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICBlLnByaW50U3RhY2tUcmFjZSgpOwotICAgICAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENoZWNrcyBmb3IgbmV4dC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgc3VjY2Vzc2Z1bC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGhhc05leHQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gYmFja2VuZC5oYXNOZXh0KCk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmVtb3ZlcyB0aGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgdm9pZCByZW1vdmUoKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oCi0gICAgICAgICAgICAgICAgICAgICJVc2UgZGVyZWdpc3RlclNlcnZpY2Vwcm92aWRlciBpbnN0ZWFkIG9mIEl0ZXJhdG9yLnJlbW92ZSgpIik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcmFwcyBzcGkncyBpdGVyYXRvciB0byBJbWFnZVJlYWRlciBpdGVyYXRvci4KLSAgICAgKi8KLSAgICBzdGF0aWMgY2xhc3MgU3BpSXRlcmF0b3JUb1JlYWRlcnNJdGVyYXRvcldyYXBwZXIgaW1wbGVtZW50cyBJdGVyYXRvcjxJbWFnZVJlYWRlcj4gewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgYmFja2VuZC4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgSXRlcmF0b3I8SW1hZ2VSZWFkZXJTcGk+IGJhY2tlbmQ7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBzcGkgaXRlcmF0b3IgdG8gcmVhZGVycyBpdGVyYXRvciB3cmFwcGVyLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGJhY2tlbmQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgYmFja2VuZC4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBTcGlJdGVyYXRvclRvUmVhZGVyc0l0ZXJhdG9yV3JhcHBlcihJdGVyYXRvcjxJbWFnZVJlYWRlclNwaT4gYmFja2VuZCkgewotICAgICAgICAgICAgdGhpcy5iYWNrZW5kID0gYmFja2VuZDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBOZXh0LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0aGUgaW1hZ2UgcmVhZGVyLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEltYWdlUmVhZGVyIG5leHQoKSB7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIHJldHVybiBiYWNrZW5kLm5leHQoKS5jcmVhdGVSZWFkZXJJbnN0YW5jZSgpOwotICAgICAgICAgICAgfSBjYXRjaCAoSU9FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIGUucHJpbnRTdGFja1RyYWNlKCk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ2hlY2tzIGZvciBuZXh0LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGJvb2xlYW4gaGFzTmV4dCgpIHsKLSAgICAgICAgICAgIHJldHVybiBiYWNrZW5kLmhhc05leHQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBSZW1vdmVzIHRoZS4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyB2b2lkIHJlbW92ZSgpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigKLSAgICAgICAgICAgICAgICAgICAgIlVzZSBkZXJlZ2lzdGVyU2VydmljZXByb3ZpZGVyIGluc3RlYWQgb2YgSXRlcmF0b3IucmVtb3ZlKCkiKTsKLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL0ltYWdlUmVhZFBhcmFtLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9JbWFnZVJlYWRQYXJhbS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5Y2M1YzVmLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL0ltYWdlUmVhZFBhcmFtLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMDEgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBTZXJnZXkgSS4gU2FsaXNoZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4yICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW87Ci0KLWltcG9ydCBqYXZhLmF3dC5EaW1lbnNpb247Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLQotLyoKLSAqIEBhdXRob3IgU2VyZ2V5IEkuIFNhbGlzaGV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0gKi8KLQotLyoqCi0gKiBUaGUgSW1hZ2VSZWFkUGFyYW0gY2xhc3MgcHJvdmlkZXMgaW5mb3JtYXRpb24gdG8gdGhlIEltYWdlUmVhZGVyIGFib3V0IGhvdyBhbgotICogaW1hZ2UgaXMgdG8gYmUgZGVjb2RlZC4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJbWFnZVJlYWRQYXJhbSBleHRlbmRzIElJT1BhcmFtIHsKLQotICAgIC8qKgotICAgICAqIFRoaXMgZmxhZyBpbmRpY2F0ZXMgaWYgdGhpcyBJbWFnZVJlYWRQYXJhbSBzdXBwb3J0cyBzZXR0aW5nIHRoZSBzb3VyY2UKLSAgICAgKiByZW5kZXJpbmcgc2l6ZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgYm9vbGVhbiBjYW5TZXRTb3VyY2VSZW5kZXJTaXplOwotCi0gICAgLyoqCi0gICAgICogVGhlIGRlc3RpbmF0aW9uIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEJ1ZmZlcmVkSW1hZ2UgZGVzdGluYXRpb247Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGVzdGluYXRpb24gYmFuZHMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludFtdIGRlc3RpbmF0aW9uQmFuZHM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbWluaW11bSBwcm9ncmVzc2l2ZSBwYXNzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgbWluUHJvZ3Jlc3NpdmVQYXNzOwotCi0gICAgLyoqCi0gICAgICogVGhlIG51bWJlciBvZiBwcm9ncmVzc2l2ZSBwYXNzZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBudW1Qcm9ncmVzc2l2ZVBhc3NlczsKLQotICAgIC8qKgotICAgICAqIFRoZSBzb3VyY2UgcmVuZGVyIHNpemUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIERpbWVuc2lvbiBzb3VyY2VSZW5kZXJTaXplOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoaXMgSW1hZ2VSZWFkZXJQYXJhbSBzdXBwb3J0cyByZW5kZXJpbmcgYSBzb3VyY2UgaW1hZ2UKLSAgICAgKiBhdCBhbiBhcmJpdHJhcnkgc2l6ZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgSW1hZ2VSZWFkZXJQYXJhbSBzdXBwb3J0cyByZW5kZXJpbmcgYSBzb3VyY2UgaW1hZ2UKLSAgICAgKiAgICAgICAgIGF0IGFuIGFyYml0cmFyeSBzaXplLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY2FuU2V0U291cmNlUmVuZGVyU2l6ZSgpIHsKLSAgICAgICAgcmV0dXJuIGNhblNldFNvdXJjZVJlbmRlclNpemU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY3VycmVudCBkZXN0aW5hdGlvbiBpbWFnZSBhcyBCdWZmZXJlZEltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEJ1ZmZlcmVkSW1hZ2Ugd2hpY2ggcmVwcmVzZW50cyB0aGUgZGVzdGluYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UgZ2V0RGVzdGluYXRpb24oKSB7Ci0gICAgICAgIHJldHVybiBkZXN0aW5hdGlvbjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBpbmRpY2VzIG9mIGRlc3RpbmF0aW9uIGJhbmRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGRlc3RpbmF0aW9uIGJhbmRzLgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXREZXN0aW5hdGlvbkJhbmRzKCkgewotICAgICAgICByZXR1cm4gZGVzdGluYXRpb25CYW5kczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBpbmRleCBvZiB0aGUgbWF4aW11bSBwYXNzIHRvIGJlIGRlY29kZWQuIFRoaXMgbWV0aG9kIHJldHVybnMKLSAgICAgKiBJbnRlZ2VyLk1BWF9WQUxVRSwgaWYgZ2V0U291cmNlTnVtUHJvZ3Jlc3NpdmVQYXNzZXMoKSBtZXRob2QgcmV0dXJucwotICAgICAqIHZhbHVlIHRoYXQgaXMgZXF1YWwgdG8gSW50ZWdlci5NQVhfVkFMVUUuIE90aGVyd2lzZSB0aGlzIG1ldGhvZCByZXR1cm5zCi0gICAgICogZ2V0U291cmNlTWluUHJvZ3Jlc3NpdmVQYXNzKCkgKyBnZXRTb3VyY2VOdW1Qcm9ncmVzc2l2ZVBhc3NlcygpIC0gMS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBpbmRleCBvZiB0aGUgbWF4aW11bSBwYXNzIHRvIGJlIGRlY29kZWQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRTb3VyY2VNYXhQcm9ncmVzc2l2ZVBhc3MoKSB7Ci0gICAgICAgIGlmIChnZXRTb3VyY2VOdW1Qcm9ncmVzc2l2ZVBhc3NlcygpID09IEludGVnZXIuTUFYX1ZBTFVFKSB7Ci0gICAgICAgICAgICByZXR1cm4gSW50ZWdlci5NQVhfVkFMVUU7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGdldFNvdXJjZU1pblByb2dyZXNzaXZlUGFzcygpICsgZ2V0U291cmNlTnVtUHJvZ3Jlc3NpdmVQYXNzZXMoKSAtIDE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW5kZXggb2YgdGhlIG1pbmltdW0gcHJvZ3Jlc3NpdmUgcGFzcyB0aGF0IGlzIGRlY29kZWQsIGRlZmF1bHQKLSAgICAgKiBpcyAwLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGluZGV4IG9mIHRoZSBtaW5pbXVtIHByb2dyZXNzaXZlIHBhc3MgdGhhdCBpcyBkZWNvZGVkLAotICAgICAqICAgICAgICAgZGVmYXVsdCBpcyAwLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0U291cmNlTWluUHJvZ3Jlc3NpdmVQYXNzKCkgewotICAgICAgICByZXR1cm4gbWluUHJvZ3Jlc3NpdmVQYXNzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG51bWJlciBvZiBwcm9ncmVzc2l2ZSBwYXNzZXMuIFRoZSBkZWZhdWx0IHZhbHVlIGlzCi0gICAgICogSW50ZWdlci5NQVhfVkFMVUUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIHByb2dyZXNzaXZlIHBhc3Nlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFNvdXJjZU51bVByb2dyZXNzaXZlUGFzc2VzKCkgewotICAgICAgICByZXR1cm4gbnVtUHJvZ3Jlc3NpdmVQYXNzZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGltZW5zaW9uIG9mIHNvdXJjZSBpbWFnZSB3aGljaCB3aWxsIGJlIHJlbmRlcmVkIGR1cmluZyBkZWNvZGluZwotICAgICAqIHByb2Nlc3MuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc291cmNlIHJlbmRlciBzaXplLgotICAgICAqLwotICAgIHB1YmxpYyBEaW1lbnNpb24gZ2V0U291cmNlUmVuZGVyU2l6ZSgpIHsKLSAgICAgICAgcmV0dXJuIHNvdXJjZVJlbmRlclNpemU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgc3BlY2lmaWVkIGRlc3RpbmF0aW9uIGltYWdlLiBUaGlzIGltYWdlIHdpbGwgYmUgdXNlZCBieSByZWFkLAotICAgICAqIHJlYWRBbGwsIGFuZCByZWFkUmFzdGVyIG1ldGhvZHMsIGFuZCBhIHJlZmVyZW5jZSB0byBpdCB3aWxsIGJlIHJldHVybmVkCi0gICAgICogYnkgdGhvc2UgbWV0aG9kcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGVzdGluYXRpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXN0aW5hdGlvbiBpbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXREZXN0aW5hdGlvbihCdWZmZXJlZEltYWdlIGRlc3RpbmF0aW9uKSB7Ci0gICAgICAgIHRoaXMuZGVzdGluYXRpb24gPSBkZXN0aW5hdGlvbjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBpbmRpY2VzIG9mIHRoZSBkZXN0aW5hdGlvbiBiYW5kcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZGVzdGluYXRpb25CYW5kcwotICAgICAqICAgICAgICAgICAgdGhlIGluZGljZXMgb2YgdGhlIGRlc3RpbmF0aW9uIGJhbmRzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldERlc3RpbmF0aW9uQmFuZHMoaW50W10gZGVzdGluYXRpb25CYW5kcykgewotICAgICAgICB0aGlzLmRlc3RpbmF0aW9uQmFuZHMgPSBkZXN0aW5hdGlvbkJhbmRzOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldERlc3RpbmF0aW9uVHlwZShJbWFnZVR5cGVTcGVjaWZpZXIgZGVzdGluYXRpb25UeXBlKSB7Ci0gICAgICAgIHRoaXMuZGVzdGluYXRpb25UeXBlID0gZGVzdGluYXRpb25UeXBlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHNvdXJjZSBwcm9ncmVzc2l2ZSBwYXNzZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1pblBhc3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgbWluaW11bSBwYXNzIHRvIGJlIGRlY29kZWQuCi0gICAgICogQHBhcmFtIG51bVBhc3NlcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBwYXNzZXMgdG8gYmUgZGVjb2RlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRTb3VyY2VQcm9ncmVzc2l2ZVBhc3NlcyhpbnQgbWluUGFzcywgaW50IG51bVBhc3NlcykgewotICAgICAgICBtaW5Qcm9ncmVzc2l2ZVBhc3MgPSBtaW5QYXNzOwotICAgICAgICBudW1Qcm9ncmVzc2l2ZVBhc3NlcyA9IG51bVBhc3NlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBkaW1lbnNpb24gc2l6ZSBvZiBzb3VyY2UgaW1hZ2UgaWYgYW4gaW1hZ2UgY2FuIGJlIHJlbmRlcmVkIGF0IGFuCi0gICAgICogYXJiaXRyYXJ5IHNpemUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNpemUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzaXplIG9mIHJlbmRlcmVkIGltYWdlLgotICAgICAqIEB0aHJvd3MgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICB0aGUgdW5zdXBwb3J0ZWQgb3BlcmF0aW9uIGV4Y2VwdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRTb3VyY2VSZW5kZXJTaXplKERpbWVuc2lvbiBzaXplKSB0aHJvd3MgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24gewotICAgICAgICBpZiAoIWNhblNldFNvdXJjZVJlbmRlclNpemUpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiY2FuJ3Qgc2V0IHNvdXJjZSByZW5kZXJlciBzaXplIik7Ci0gICAgICAgIH0KLSAgICAgICAgc291cmNlUmVuZGVyU2l6ZSA9IHNpemU7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vSW1hZ2VSZWFkZXIuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL0ltYWdlUmVhZGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGNmMjgyZWQuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vSW1hZ2VSZWFkZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDExNjIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2VpbzsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLkltYWdlUmVhZGVyU3BpOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3RyZWFtLkltYWdlSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5tZXRhZGF0YS5JSU9NZXRhZGF0YTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLmV2ZW50LklJT1JlYWRXYXJuaW5nTGlzdGVuZXI7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5ldmVudC5JSU9SZWFkUHJvZ3Jlc3NMaXN0ZW5lcjsKLWltcG9ydCBqYXZheC5pbWFnZWlvLmV2ZW50LklJT1JlYWRVcGRhdGVMaXN0ZW5lcjsKLWltcG9ydCBqYXZhLnV0aWwuTG9jYWxlOwotaW1wb3J0IGphdmEudXRpbC5MaXN0OwotaW1wb3J0IGphdmEudXRpbC5JdGVyYXRvcjsKLWltcG9ydCBqYXZhLnV0aWwuU2V0OwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5SYXN0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmVuZGVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC4qOwotCi0vKioKLSAqIFRoZSBJbWFnZVJlYWRlciBjbGFzcyBpcyBhbiBhYnN0cmFjdCBjbGFzcyBmb3IgZGVjb2RpbmcgaW1hZ2VzLiBJbWFnZVJlYWRlcgotICogb2JqZWN0cyBhcmUgaW5zdGFudGlhdGVkIGJ5IHRoZSBzZXJ2aWNlIHByb3ZpZGVyIGludGVyZmFjZSwgSW1hZ2VSZWFkZXJTcGkKLSAqIGNsYXNzLCBmb3IgdGhlIHNwZWNpZmljIGZvcm1hdC4gSW1hZ2VSZWFkZXJTcGkgY2xhc3Mgc2hvdWxkIGJlIHJlZ2lzdGVyZWQKLSAqIHdpdGggdGhlIElJT1JlZ2lzdHJ5LCB3aGljaCB1c2VzIHRoZW0gZm9yIGZvcm1hdCByZWNvZ25pdGlvbiBhbmQgcHJlc2VudGF0aW9uCi0gKiBvZiBhdmFpbGFibGUgZm9ybWF0IHJlYWRlcnMgYW5kIHdyaXRlcnMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgSW1hZ2VSZWFkZXIgewotCi0gICAgLyoqCi0gICAgICogVGhlIG9yaWdpbmF0aW5nIHByb3ZpZGVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBJbWFnZVJlYWRlclNwaSBvcmlnaW5hdGluZ1Byb3ZpZGVyOwotCi0gICAgLyoqCi0gICAgICogVGhlIGlucHV0IG9iamVjdCBzdWNoIGFzIEltYWdlSW5wdXRTdHJlYW0uCi0gICAgICovCi0gICAgcHJvdGVjdGVkIE9iamVjdCBpbnB1dDsKLQotICAgIC8qKgotICAgICAqIFRoZSBzZWVrIGZvcndhcmQgb25seS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgYm9vbGVhbiBzZWVrRm9yd2FyZE9ubHk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaWdub3JlIG1ldGFkYXRhIGZsYWcgaW5kaWNhdGVzIHdoZXRoZXIgY3VycmVudCBpbnB1dCBzb3VyY2UgaGFzIGJlZW4KLSAgICAgKiBtYXJrZWQgYXMgbWV0YWRhdGEgaXMgYWxsb3dlZCB0byBiZSBpZ25vcmVkIGJ5IHNldElucHV0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBib29sZWFuIGlnbm9yZU1ldGFkYXRhOwotCi0gICAgLyoqCi0gICAgICogVGhlIG1pbmltdW0gaW5kZXguCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBtaW5JbmRleDsKLQotICAgIC8qKgotICAgICAqIFRoZSBhdmFpbGFibGUgbG9jYWxlcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgTG9jYWxlW10gYXZhaWxhYmxlTG9jYWxlczsKLQotICAgIC8qKgotICAgICAqIFRoZSBsb2NhbGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIExvY2FsZSBsb2NhbGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbGlzdCBvZiB3YXJuaW5nIGxpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgTGlzdDxJSU9SZWFkV2FybmluZ0xpc3RlbmVyPiB3YXJuaW5nTGlzdGVuZXJzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGxpc3Qgb2Ygd2FybmluZyBsb2NhbGVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBMaXN0PExvY2FsZT4gd2FybmluZ0xvY2FsZXM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbGlzdCBvZiBwcm9ncmVzcyBsaXN0ZW5lcnMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIExpc3Q8SUlPUmVhZFByb2dyZXNzTGlzdGVuZXI+IHByb2dyZXNzTGlzdGVuZXJzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGxpc3Qgb2YgdXBkYXRlIGxpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgTGlzdDxJSU9SZWFkVXBkYXRlTGlzdGVuZXI+IHVwZGF0ZUxpc3RlbmVyczsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJbWFnZVJlYWRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb3JpZ2luYXRpbmdQcm92aWRlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUmVhZGVyU3BpIHdoaWNoIGluc3RhbnRpYXRlcyB0aGlzIEltYWdlUmVhZGVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBJbWFnZVJlYWRlcihJbWFnZVJlYWRlclNwaSBvcmlnaW5hdGluZ1Byb3ZpZGVyKSB7Ci0gICAgICAgIHRoaXMub3JpZ2luYXRpbmdQcm92aWRlciA9IG9yaWdpbmF0aW5nUHJvdmlkZXI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZm9ybWF0IG5hbWUgb2YgdGhpcyBpbnB1dCBzb3VyY2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZm9ybWF0IG5hbWUgb2YgdGhpcyBpbnB1dCBzb3VyY2UuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0Rm9ybWF0TmFtZSgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBvcmlnaW5hdGluZ1Byb3ZpZGVyLmdldEZvcm1hdE5hbWVzKClbMF07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgSW1hZ2VSZWFkZXJTcGkgd2hpY2ggaW5zdGFudGlhdGVkIHRoaXMgSW1hZ2VSZWFkZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VSZWFkZXJTcGkuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlUmVhZGVyU3BpIGdldE9yaWdpbmF0aW5nUHJvdmlkZXIoKSB7Ci0gICAgICAgIHJldHVybiBvcmlnaW5hdGluZ1Byb3ZpZGVyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHNwZWNpZmllZCBPYmplY3QgYXMgdGhlIGlucHV0IHNvdXJjZSBvZiB0aGlzIEltYWdlUmVhZGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbnB1dAotICAgICAqICAgICAgICAgICAgdGhlIGlucHV0IHNvdXJjZSwgaXQgY2FuIGJlIGFuIEltYWdlSW5wdXRTdHJlYW0gb3Igb3RoZXIKLSAgICAgKiAgICAgICAgICAgIHN1cHBvcnRlZCBvYmplY3RzLgotICAgICAqIEBwYXJhbSBzZWVrRm9yd2FyZE9ubHkKLSAgICAgKiAgICAgICAgICAgIGluZGljYXRlcyB3aGV0aGVyIHRoZSBzdHJlYW0gbXVzdCBiZSByZWFkIHNlcXVlbnRpYWxseSBmcm9tCi0gICAgICogICAgICAgICAgICBpdHMgY3VycmVudCBzdGFydGluZyBwb2ludC4KLSAgICAgKiBAcGFyYW0gaWdub3JlTWV0YWRhdGEKLSAgICAgKiAgICAgICAgICAgIHBhcmFtZXRlciB3aGljaCBpbmRpY2F0ZXMgaWYgbWV0YWRhdGEgbWF5IGJlIGlnbm9yZWQgZHVyaW5nCi0gICAgICogICAgICAgICAgICByZWFkcyBvciBub3QuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0SW5wdXQoT2JqZWN0IGlucHV0LCBib29sZWFuIHNlZWtGb3J3YXJkT25seSwgYm9vbGVhbiBpZ25vcmVNZXRhZGF0YSkgewotICAgICAgICBpZiAoaW5wdXQgIT0gbnVsbCkgewotICAgICAgICAgICAgaWYgKCFpc1N1cHBvcnRlZChpbnB1dCkgJiYgIShpbnB1dCBpbnN0YW5jZW9mIEltYWdlSW5wdXRTdHJlYW0pKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiaW5wdXQgIiArIGlucHV0ICsgIiBpcyBub3Qgc3VwcG9ydGVkIik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgdGhpcy5taW5JbmRleCA9IDA7Ci0gICAgICAgIHRoaXMuc2Vla0ZvcndhcmRPbmx5ID0gc2Vla0ZvcndhcmRPbmx5OwotICAgICAgICB0aGlzLmlnbm9yZU1ldGFkYXRhID0gaWdub3JlTWV0YWRhdGE7Ci0gICAgICAgIHRoaXMuaW5wdXQgPSBpbnB1dDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgaXMgc3VwcG9ydGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbnB1dAotICAgICAqICAgICAgICAgICAgdGhlIGlucHV0LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgaXMgc3VwcG9ydGVkLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBpc1N1cHBvcnRlZChPYmplY3QgaW5wdXQpIHsKLSAgICAgICAgSW1hZ2VSZWFkZXJTcGkgc3BpID0gZ2V0T3JpZ2luYXRpbmdQcm92aWRlcigpOwotICAgICAgICBpZiAobnVsbCAhPSBzcGkpIHsKLSAgICAgICAgICAgIENsYXNzW10gb3V0VHlwZXMgPSBzcGkuZ2V0SW5wdXRUeXBlcygpOwotICAgICAgICAgICAgZm9yIChDbGFzczw/PiBlbGVtZW50IDogb3V0VHlwZXMpIHsKLSAgICAgICAgICAgICAgICBpZiAoZWxlbWVudC5pc0luc3RhbmNlKGlucHV0KSkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHNwZWNpZmllZCBPYmplY3QgYXMgdGhlIGlucHV0IHNvdXJjZSBvZiB0aGlzIEltYWdlUmVhZGVyLgotICAgICAqIE1ldGFkYXRhIGlzIG5vdCBpZ25vcmVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbnB1dAotICAgICAqICAgICAgICAgICAgdGhlIGlucHV0IHNvdXJjZSwgaXQgY2FuIGJlIGFuIEltYWdlSW5wdXRTdHJlYW0gb3Igb3RoZXIKLSAgICAgKiAgICAgICAgICAgIHN1cHBvcnRlZCBvYmplY3RzLgotICAgICAqIEBwYXJhbSBzZWVrRm9yd2FyZE9ubHkKLSAgICAgKiAgICAgICAgICAgIGluZGljYXRlcyB3aGV0aGVyIHRoZSBzdHJlYW0gbXVzdCBiZSByZWFkIHNlcXVlbnRpYWxseSBmcm9tCi0gICAgICogICAgICAgICAgICBpdHMgY3VycmVudCBzdGFydGluZyBwb2ludC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRJbnB1dChPYmplY3QgaW5wdXQsIGJvb2xlYW4gc2Vla0ZvcndhcmRPbmx5KSB7Ci0gICAgICAgIHNldElucHV0KGlucHV0LCBzZWVrRm9yd2FyZE9ubHksIGZhbHNlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBzcGVjaWZpZWQgT2JqZWN0IGFzIHRoZSBpbnB1dCBzb3VyY2Ugb2YgdGhpcyBJbWFnZVJlYWRlci4KLSAgICAgKiBNZXRhZGF0YSBpcyBub3QgaWdub3JlZCBhbmQgZm9yd2FyZCBzZWVraW5nIGlzIG5vdCByZXF1aXJlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5wdXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnB1dCBzb3VyY2UsIGl0IGNhbiBiZSBJbWFnZUlucHV0U3RyZWFtIG9yIG90aGVyIG9iamVjdHMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0SW5wdXQoT2JqZWN0IGlucHV0KSB7Ci0gICAgICAgIHNldElucHV0KGlucHV0LCBmYWxzZSwgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGlucHV0IHNvdXJjZSBvYmplY3Qgb2YgdGhpcyBJbWFnZVJlYWRlciwgb3IgcmV0dXJucyBudWxsLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGlucHV0IHNvdXJjZSBvYmplY3Qgc3VjaCBhcyBJbWFnZUlucHV0U3RyZWFtLCBvciBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBPYmplY3QgZ2V0SW5wdXQoKSB7Ci0gICAgICAgIHJldHVybiBpbnB1dDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIGlucHV0IHNvdXJjZSBzdXBwb3J0cyBvbmx5IGZvcndhcmQgcmVhZGluZywgb3Igbm90LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIGlucHV0IHNvdXJjZSBzdXBwb3J0cyBvbmx5IGZvcndhcmQgcmVhZGluZywgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1NlZWtGb3J3YXJkT25seSgpIHsKLSAgICAgICAgcmV0dXJuIHNlZWtGb3J3YXJkT25seTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIGN1cnJlbnQgaW5wdXQgc291cmNlIGFsbG93cyB0byBtZXRhZGF0YSB0byBiZSBpZ25vcmVkCi0gICAgICogYnkgcGFzc2luZyB0cnVlIGFzIHRoZSBpZ25vcmVNZXRhZGF0YSBhcmd1bWVudCB0byB0aGUgc2V0SW5wdXQgbWV0aG9kLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIGN1cnJlbnQgaW5wdXQgc291cmNlIGFsbG93cyB0byBtZXRhZGF0YSB0byBiZQotICAgICAqICAgICAgICAgaWdub3JlZCBieSBwYXNzaW5nIHRydWUgYXMgdGhlIGlnbm9yZU1ldGFkYXRhIGFyZ3VtZW50IHRvIHRoZQotICAgICAqICAgICAgICAgc2V0SW5wdXQgbWV0aG9kLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzSWdub3JpbmdNZXRhZGF0YSgpIHsKLSAgICAgICAgcmV0dXJuIGlnbm9yZU1ldGFkYXRhOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1pbmltdW0gdmFsaWQgaW5kZXggZm9yIHJlYWRpbmcgYW4gaW1hZ2UsIHRodW1ibmFpbCwgb3IgaW1hZ2UKLSAgICAgKiBtZXRhZGF0YS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtaW5pbXVtIHZhbGlkIGluZGV4IGZvciByZWFkaW5nIGFuIGltYWdlLCB0aHVtYm5haWwsIG9yIGltYWdlCi0gICAgICogICAgICAgICBtZXRhZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE1pbkluZGV4KCkgewotICAgICAgICByZXR1cm4gbWluSW5kZXg7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYXZhaWxhYmxlIGxvY2FsZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiBhbiBhcnJheSBvZiB0aGUgYXZhaWxhYmxlIGxvY2FsZXMuCi0gICAgICovCi0gICAgcHVibGljIExvY2FsZVtdIGdldEF2YWlsYWJsZUxvY2FsZXMoKSB7Ci0gICAgICAgIHJldHVybiBhdmFpbGFibGVMb2NhbGVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGxvY2FsZSB0byB0aGlzIEltYWdlUmVhZGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsb2NhbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBMb2NhbGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0TG9jYWxlKExvY2FsZSBsb2NhbGUpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbG9jYWxlIG9mIHRoaXMgSW1hZ2VSZWFkZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbG9jYWxlIG9mIHRoaXMgSW1hZ2VSZWFkZXIuCi0gICAgICovCi0gICAgcHVibGljIExvY2FsZSBnZXRMb2NhbGUoKSB7Ci0gICAgICAgIHJldHVybiBsb2NhbGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGltYWdlcyBhdmFpbGFibGUgaW4gdGhlIGN1cnJlbnQgaW5wdXQgc291cmNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBhbGxvd1NlYXJjaAotICAgICAqICAgICAgICAgICAgdGhlIHBhcmFtZXRlciB3aGljaCBpbmRpY2F0ZXMgd2hhdCBhIHNlYXJjaCBpcyByZXF1aXJlZDsgaWYKLSAgICAgKiAgICAgICAgICAgIGZhbHNlLCB0aGUgcmVhZGVyIG1heSByZXR1cm4gLTEgd2l0aG91dCBzZWFyY2hpbmcuCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIGltYWdlcy4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXROdW1JbWFnZXMoYm9vbGVhbiBhbGxvd1NlYXJjaCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgd2lkdGggb2YgdGhlIHNwZWNpZmllZCBpbWFnZSBpbiBpbnB1dCBzb3VyY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSB3aWR0aCBpbiBwaXhlbHMuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0V2lkdGgoaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGhlaWdodCBvZiB0aGUgc3BlY2lmaWVkIGltYWdlIGluIGlucHV0IHNvdXJjZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIGhlaWdodCBpbiBwaXhlbHMuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0SGVpZ2h0KGludCBpbWFnZUluZGV4KSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIHN0b3JhZ2UgZm9ybWF0IG9mIHRoZSBzcGVjaWZpZWQgaW1hZ2UgcGxhY2VzIGFuIGltcGVkaW1lbnQKLSAgICAgKiBvbiByYW5kb20gcGl4ZWxzIGFjY2VzcyBvciBub3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSdzIGluZGV4LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHN0b3JhZ2UgZm9ybWF0IG9mIHRoZSBzcGVjaWZpZWQgaW1hZ2UgcGxhY2VzIGFuCi0gICAgICogICAgICAgICBpbXBlZGltZW50IG9uIHJhbmRvbSBwaXhlbHMgYWNjZXNzLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzUmFuZG9tQWNjZXNzRWFzeShpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOyAvLyBkZWYKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhc3BlY3QgcmF0aW8gKHdpZHRoIGRldmlkZWQgYnkgaGVpZ2h0KSBvZiB0aGUgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSBhc3BlY3QgcmF0aW8gb2YgdGhlIGltYWdlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0QXNwZWN0UmF0aW8oaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiAoZmxvYXQpZ2V0V2lkdGgoaW1hZ2VJbmRleCkgLyBnZXRIZWlnaHQoaW1hZ2VJbmRleCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBJbWFnZVR5cGVTcGVjaWZpZXIgd2hpY2ggaW5kaWNhdGVzIHRoZSB0eXBlIG9mIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBpbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlJ3MgaW5kZXguCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VUeXBlU3BlY2lmaWVyLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VUeXBlU3BlY2lmaWVyIGdldFJhd0ltYWdlVHlwZShpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBJdGVyYXRvciBvZiBJbWFnZVR5cGVTcGVjaWZpZXIgb2JqZWN0cyB3aGljaCBhcmUgYXNzb2NpYXRlZCB3aXRoCi0gICAgICogaW1hZ2UgdHlwZXMgdGhhdCBtYXkgYmUgdXNlZCB3aGVuIGRlY29kaW5nIHNwZWNpZmllZCBpbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGluZGV4LgotICAgICAqIEByZXR1cm4gYW4gSXRlcmF0b3Igb2YgSW1hZ2VUeXBlU3BlY2lmaWVyIG9iamVjdHMuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJdGVyYXRvcjxJbWFnZVR5cGVTcGVjaWZpZXI+IGdldEltYWdlVHlwZXMoaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRlZmF1bHQgSW1hZ2VSZWFkUGFyYW0gb2JqZWN0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEltYWdlUmVhZFBhcmFtIG9iamVjdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VSZWFkUGFyYW0gZ2V0RGVmYXVsdFJlYWRQYXJhbSgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBJSU9NZXRhZGF0YSBvYmplY3QgZm9yIHRoaXMgaW5wdXQgc291cmNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIElJT01ldGFkYXRhLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgSUlPTWV0YWRhdGEgZ2V0U3RyZWFtTWV0YWRhdGEoKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIElJT01ldGFkYXRhIG9iamVjdCBmb3IgdGhpcyBpbnB1dCBzb3VyY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZvcm1hdE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXNpcmVkIG1ldGFkYXRhIGZvcm1hdCB0byBiZSB1c2VkIGluIHRoZSByZXR1cm5lZAotICAgICAqICAgICAgICAgICAgSUlPTWV0YWRhdGEgb2JqZWN0LgotICAgICAqIEBwYXJhbSBub2RlTmFtZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBub2RlIG5hbWVzIG9mIHRoZSBkb2N1bWVudC4KLSAgICAgKiBAcmV0dXJuIHRoZSBJSU9NZXRhZGF0YS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIElJT01ldGFkYXRhIGdldFN0cmVhbU1ldGFkYXRhKFN0cmluZyBmb3JtYXROYW1lLCBTZXQ8U3RyaW5nPiBub2RlTmFtZXMpCi0gICAgICAgICAgICB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBpbWFnZSBtZXRhZGF0YSBvZiB0aGUgc3BlY2lmaWVkIGltYWdlIGluIGlucHV0IHNvdXJjZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIElJT01ldGFkYXRhLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgSUlPTWV0YWRhdGEgZ2V0SW1hZ2VNZXRhZGF0YShpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgaW1hZ2UgbWV0YWRhdGEgb2YgdGhlIHNwZWNpZmllZCBpbWFnZSBpbnB1dCBzb3VyY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBpbmRleC4KLSAgICAgKiBAcGFyYW0gZm9ybWF0TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGRlc2lyZWQgbWV0YWRhdGEgZm9ybWF0IHRvIGJlIHVzZWQgaW4gdGhlIHJldHVybmVkCi0gICAgICogICAgICAgICAgICBJSU9NZXRhZGF0YSBvYmplY3QuCi0gICAgICogQHBhcmFtIG5vZGVOYW1lcwotICAgICAqICAgICAgICAgICAgdGhlIG5vZGUgbmFtZXMgd2hpY2ggY2FuIGJlIGNvbnRhaW5lZCBpbiB0aGUgZG9jdW1lbnQuCi0gICAgICogQHJldHVybiB0aGUgSUlPTWV0YWRhdGEuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBJSU9NZXRhZGF0YSBnZXRJbWFnZU1ldGFkYXRhKGludCBpbWFnZUluZGV4LCBTdHJpbmcgZm9ybWF0TmFtZSwgU2V0PFN0cmluZz4gbm9kZU5hbWVzKQotICAgICAgICAgICAgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVhZHMgdGhlIHNwZWNpZmllZCBpbWFnZSBhbmQgcmV0dXJucyBpdCBhcyBhIEJ1ZmZlcmVkSW1hZ2UgdXNpbmcgdGhlCi0gICAgICogZGVmYXVsdCBJbWFnZVJlYWRQYXJhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlIHJlYWQoaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiByZWFkKGltYWdlSW5kZXgsIG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlYWRzIHRoZSBzcGVjaWZpZWQgaW1hZ2UgYW5kIHJldHVybnMgaXQgYXMgYSBCdWZmZXJlZEltYWdlIHVzaW5nIHRoZQotICAgICAqIHNwZWNpZmllZCBJbWFnZVJlYWRQYXJhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGluZGV4LgotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUmVhZFBhcmFtLgotICAgICAqIEByZXR1cm4gdGhlIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBCdWZmZXJlZEltYWdlIHJlYWQoaW50IGltYWdlSW5kZXgsIEltYWdlUmVhZFBhcmFtIHBhcmFtKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyB0aGUgc3BlY2lmaWVkIGltYWdlIGFuZCByZXR1cm5zIGFuIElJT0ltYWdlIHdpdGggdGhpcyBpbWFnZSwKLSAgICAgKiB0aHVtYm5haWxzLCBhbmQgbWV0YWRhdGEgZm9yIHRoaXMgaW1hZ2UsIHVzaW5nIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBJbWFnZVJlYWRQYXJhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGluZGV4LgotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUmVhZFBhcmFtLgotICAgICAqIEByZXR1cm4gdGhlIElJT0ltYWdlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSUlPSW1hZ2UgcmVhZEFsbChpbnQgaW1hZ2VJbmRleCwgSW1hZ2VSZWFkUGFyYW0gcGFyYW0pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gSXRlcmF0b3Igb2YgSUlPSW1hZ2VzIGZyb20gdGhlIGlucHV0IHNvdXJjZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcGFyYW1zCi0gICAgICogICAgICAgICAgICB0aGUgSXRlcmF0b3Igb2YgSW1hZ2VSZWFkUGFyYW0gb2JqZWN0cy4KLSAgICAgKiBAcmV0dXJuIHRoZSBpdGVyYXRvciBvZiBJSU9JbWFnZXMuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBJdGVyYXRvcjxJSU9JbWFnZT4gcmVhZEFsbChJdGVyYXRvcjw/IGV4dGVuZHMgSW1hZ2VSZWFkUGFyYW0+IHBhcmFtcykgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IHRoaXMgcGx1Zy1pbiBzdXBwb3J0cyByZWFkaW5nIGEgUmFzdGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBwbHVnLWluIHN1cHBvcnRzIHJlYWRpbmcgYSBSYXN0ZXIsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjYW5SZWFkUmFzdGVyKCkgewotICAgICAgICByZXR1cm4gZmFsc2U7IC8vIGRlZgotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlYWRzIGEgbmV3IFJhc3RlciBvYmplY3Qgd2hpY2ggY29udGFpbnMgdGhlIHJhdyBwaXhlbCBkYXRhIGZyb20gdGhlCi0gICAgICogaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBpbmRleC4KLSAgICAgKiBAcGFyYW0gcGFyYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVJlYWRQYXJhbS4KLSAgICAgKiBAcmV0dXJuIHRoZSBSYXN0ZXIuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBSYXN0ZXIgcmVhZFJhc3RlcihpbnQgaW1hZ2VJbmRleCwgSW1hZ2VSZWFkUGFyYW0gcGFyYW0pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiVW5zdXBwb3J0ZWQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIHNwZWNpZmllZCBpbWFnZSBoYXMgdGlsZXMgb3Igbm90LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UncyBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgaW1hZ2UgaGFzIHRpbGVzLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzSW1hZ2VUaWxlZChpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOyAvLyBkZWYKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0aWxlIHdpZHRoIGluIHRoZSBzcGVjaWZpZWQgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSdzIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIHRpbGUgd2lkdGguCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VGlsZVdpZHRoKGludCBpbWFnZUluZGV4KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gZ2V0V2lkdGgoaW1hZ2VJbmRleCk7IC8vIGRlZgotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRpbGUgaGVpZ2h0IGluIHRoZSBzcGVjaWZpZWQgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSdzIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIHRpbGUgaGVpZ2h0LgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFRpbGVIZWlnaHQoaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBnZXRIZWlnaHQoaW1hZ2VJbmRleCk7IC8vIGRlZgotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIHRpbGUgZ3JpZCBpbiB0aGUKLSAgICAgKiBzcGVjaWZpZWQgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSdzIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBjb3JuZXIgb2YgdGhlIHRpbGUgZ3JpZC4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRUaWxlR3JpZFhPZmZzZXQoaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiAwOyAvLyBkZWYKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSB0aWxlIGdyaWQgaW4gdGhlCi0gICAgICogc3BlY2lmaWVkIGltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UncyBpbmRleC4KLSAgICAgKiBAcmV0dXJuIHRoZSBZIGNvb3JkaW5hdGUgb2YgdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZSB0aWxlIGdyaWQuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VGlsZUdyaWRZT2Zmc2V0KGludCBpbWFnZUluZGV4KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gMDsgLy8gZGVmCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVhZHMgdGhlIHRpbGUgc3BlY2lmaWVkIGJ5IHRoZSB0aWxlWCBhbmQgdGlsZVkgcGFyYW1ldGVycyBvZiB0aGUKLSAgICAgKiBzcGVjaWZpZWQgaW1hZ2UgYW5kIHJldHVybnMgaXQgYXMgYSBCdWZmZXJlZEltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgaW5kZXguCi0gICAgICogQHBhcmFtIHRpbGVYCi0gICAgICogICAgICAgICAgICB0aGUgWCBpbmRleCBvZiB0aWxlLgotICAgICAqIEBwYXJhbSB0aWxlWQotICAgICAqICAgICAgICAgICAgdGhlIFkgaW5kZXggb2YgdGlsZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBCdWZmZXJlZEltYWdlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgQnVmZmVyZWRJbWFnZSByZWFkVGlsZShpbnQgaW1hZ2VJbmRleCwgaW50IHRpbGVYLCBpbnQgdGlsZVkpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlYWRzIHRoZSB0aWxlIHNwZWNpZmllZCBieSB0aGUgdGlsZVggYW5kIHRpbGVZIHBhcmFtZXRlcnMgb2YgdGhlCi0gICAgICogc3BlY2lmaWVkIGltYWdlIGFuZCByZXR1cm5zIGl0IGFzIGEgUmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgaW5kZXguCi0gICAgICogQHBhcmFtIHRpbGVYCi0gICAgICogICAgICAgICAgICB0aGUgWCBpbmRleCBvZiB0aWxlLgotICAgICAqIEBwYXJhbSB0aWxlWQotICAgICAqICAgICAgICAgICAgdGhlIFkgaW5kZXggb2YgdGlsZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBSYXN0ZXIuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBSYXN0ZXIgcmVhZFRpbGVSYXN0ZXIoaW50IGltYWdlSW5kZXgsIGludCB0aWxlWCwgaW50IHRpbGVZKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyB0aGUgc3BlY2lmaWVkIGltYWdlIHVzaW5nIHRoZSBzcGVjaWZpZWQgSW1hZ2VSZWFkUGFyYW0gYW5kIHJldHVybnMKLSAgICAgKiBpdCBhcyBhIFJlbmRlcmVkSW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBpbmRleC4KLSAgICAgKiBAcGFyYW0gcGFyYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVJlYWRQYXJhbS4KLSAgICAgKiBAcmV0dXJuIHRoZSBSZW5kZXJlZEltYWdlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgUmVuZGVyZWRJbWFnZSByZWFkQXNSZW5kZXJlZEltYWdlKGludCBpbWFnZUluZGV4LCBJbWFnZVJlYWRQYXJhbSBwYXJhbSkKLSAgICAgICAgICAgIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiByZWFkKGltYWdlSW5kZXgsIHBhcmFtKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIGltYWdlIGZvcm1hdCBzdXBwb3J0ZWQgYnkgdGhpcyByZWFkZXIgc3VwcG9ydHMKLSAgICAgKiB0aHVtYm5haWwgcHJldmlldyBpbWFnZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgaW1hZ2UgZm9ybWF0IHN1cHBvcnRlZCBieSB0aGlzIHJlYWRlciBzdXBwb3J0cwotICAgICAqICAgICAgICAgdGh1bWJuYWlsIHByZXZpZXcgaW1hZ2VzLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gcmVhZGVyU3VwcG9ydHNUaHVtYm5haWxzKCkgewotICAgICAgICByZXR1cm4gZmFsc2U7IC8vIGRlZgotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGUgc3BlY2lmaWVkIGltYWdlIGhhcyB0aHVtYm5haWxzIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlJ3MgaW5kZXguCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIGltYWdlIGhhcyB0aHVtYm5haWxzLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGhhc1RodW1ibmFpbHMoaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBnZXROdW1UaHVtYm5haWxzKGltYWdlSW5kZXgpID4gMDsgLy8gZGVmCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIHRodW1ibmFpbHMgZm9yIHRoZSBzcGVjaWZpZWQgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSdzIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiB0aHVtYm5haWxzLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE51bVRodW1ibmFpbHMoaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiAwOyAvLyBkZWYKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB3aWR0aCBvZiB0aGUgc3BlY2lmaWVkIHRodW1ibmFpbCBmb3IgdGhlIHNwZWNpZmllZCBpbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlJ3MgaW5kZXguCi0gICAgICogQHBhcmFtIHRodW1ibmFpbEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgdGh1bWJuYWlsJ3MgaW5kZXguCi0gICAgICogQHJldHVybiB0aGUgdGh1bWJuYWlsIHdpZHRoLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFRodW1ibmFpbFdpZHRoKGludCBpbWFnZUluZGV4LCBpbnQgdGh1bWJuYWlsSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiByZWFkVGh1bWJuYWlsKGltYWdlSW5kZXgsIHRodW1ibmFpbEluZGV4KS5nZXRXaWR0aCgpOyAvLyBkZWYKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBoZWlnaHQgb2YgdGhlIHNwZWNpZmllZCB0aHVtYm5haWwgZm9yIHRoZSBzcGVjaWZpZWQgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSdzIGluZGV4LgotICAgICAqIEBwYXJhbSB0aHVtYm5haWxJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIHRodW1ibmFpbCdzIGluZGV4LgotICAgICAqIEByZXR1cm4gdGhlIHRodW1ibmFpbCBoZWlnaHQuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VGh1bWJuYWlsSGVpZ2h0KGludCBpbWFnZUluZGV4LCBpbnQgdGh1bWJuYWlsSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiByZWFkVGh1bWJuYWlsKGltYWdlSW5kZXgsIHRodW1ibmFpbEluZGV4KS5nZXRIZWlnaHQoKTsgLy8gZGVmCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVhZHMgdGhlIHRodW1ibmFpbCBpbWFnZSBmb3IgdGhlIHNwZWNpZmllZCBpbWFnZSBhcyBhIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBpbmRleC4KLSAgICAgKiBAcGFyYW0gdGh1bWJuYWlsSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aHVtYm5haWwgaW5kZXguCi0gICAgICogQHJldHVybiB0aGUgQnVmZmVyZWRJbWFnZS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UgcmVhZFRodW1ibmFpbChpbnQgaW1hZ2VJbmRleCwgaW50IHRodW1ibmFpbEluZGV4KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIlVuc3VwcG9ydGVkIik7IC8vIGRlZgotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcXVlc3RzIGFuIGFib3J0IG9wZXJhdGlvbiBmb3IgY3VycmVudCByZWFkaW5nIG9wZXJhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhYm9ydCgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgb3Igbm90IGEgcmVxdWVzdCB0byBhYm9ydCB0aGUgY3VycmVudCByZWFkIG9wZXJhdGlvbiBoYXMKLSAgICAgKiBiZWVuIG1hZGUgc3VjY2Vzc2Z1bGx5LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHJlcXVlc3QgdG8gYWJvcnQgdGhlIGN1cnJlbnQgcmVhZCBvcGVyYXRpb24gaGFzIGJlZW4KLSAgICAgKiAgICAgICAgIG1hZGUgc3VjY2Vzc2Z1bGx5LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGJvb2xlYW4gYWJvcnRSZXF1ZXN0ZWQoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENsZWFycyBhbGwgcHJldmlvdXMgYWJvcnQgcmVxdWVzdCwgYW5kIGFib3J0UmVxdWVzdGVkIHJldHVybnMgZmFsc2UgYWZ0ZXIKLSAgICAgKiBjYWxsaW5nIHRoaXMgbWV0aG9kLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIGNsZWFyQWJvcnRSZXF1ZXN0KCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHRoZSBJSU9SZWFkV2FybmluZ0xpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsaXN0ZW5lcgotICAgICAqICAgICAgICAgICAgdGhlIElJT1JlYWRXYXJuaW5nTGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkSUlPUmVhZFdhcm5pbmdMaXN0ZW5lcihJSU9SZWFkV2FybmluZ0xpc3RlbmVyIGxpc3RlbmVyKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIHNwZWNpZmllZCBJSU9SZWFkV2FybmluZ0xpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsaXN0ZW5lcgotICAgICAqICAgICAgICAgICAgdGhlIElJT1JlYWRXYXJuaW5nTGlzdGVuZXIgdG8gYmUgcmVtb3ZlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVJSU9SZWFkV2FybmluZ0xpc3RlbmVyKElJT1JlYWRXYXJuaW5nTGlzdGVuZXIgbGlzdGVuZXIpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyBhbGwgcmVnaXN0ZXJlZCBJSU9SZWFkV2FybmluZ0xpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVBbGxJSU9SZWFkV2FybmluZ0xpc3RlbmVycygpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgSUlPUmVhZFByb2dyZXNzTGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxpc3RlbmVyCi0gICAgICogICAgICAgICAgICB0aGUgSUlPUmVhZFByb2dyZXNzTGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkSUlPUmVhZFByb2dyZXNzTGlzdGVuZXIoSUlPUmVhZFByb2dyZXNzTGlzdGVuZXIgbGlzdGVuZXIpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgc3BlY2lmaWVkIElJT1JlYWRQcm9ncmVzc0xpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsaXN0ZW5lcgotICAgICAqICAgICAgICAgICAgdGhlIElJT1JlYWRQcm9ncmVzc0xpc3RlbmVyIHRvIGJlIHJlbW92ZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlSUlPUmVhZFByb2dyZXNzTGlzdGVuZXIoSUlPUmVhZFByb2dyZXNzTGlzdGVuZXIgbGlzdGVuZXIpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyByZWdpc3RlcmVkIElJT1JlYWRQcm9ncmVzc0xpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVBbGxJSU9SZWFkUHJvZ3Jlc3NMaXN0ZW5lcnMoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgdGhlIElJT1JlYWRVcGRhdGVMaXN0ZW5lci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbGlzdGVuZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBJSU9SZWFkVXBkYXRlTGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkSUlPUmVhZFVwZGF0ZUxpc3RlbmVyKElJT1JlYWRVcGRhdGVMaXN0ZW5lciBsaXN0ZW5lcikgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHRoZSBzcGVjaWZpZWQgSUlPUmVhZFVwZGF0ZUxpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsaXN0ZW5lcgotICAgICAqICAgICAgICAgICAgdGhlIElJT1JlYWRVcGRhdGVMaXN0ZW5lciB0byBiZSByZW1vdmVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlbW92ZUlJT1JlYWRVcGRhdGVMaXN0ZW5lcihJSU9SZWFkVXBkYXRlTGlzdGVuZXIgbGlzdGVuZXIpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyByZWdpc3RlcmVkIElJT1JlYWRVcGRhdGVMaXN0ZW5lcnMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlQWxsSUlPUmVhZFVwZGF0ZUxpc3RlbmVycygpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIHRoZSBzdGFydCBvZiBhbiBzZXF1ZW5jZSBvZiBpbWFnZSByZWFkcyBieSBjYWxsaW5nIHRoZQotICAgICAqIHNlcXVlbmNlU3RhcnRlZCBtZXRob2Qgb24gYWxsIHJlZ2lzdGVyZWQgSUlPUmVhZFByb2dyZXNzTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBtaW5JbmRleAotICAgICAqICAgICAgICAgICAgdGhlIG1pbmltdW0gaW5kZXguCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgcHJvY2Vzc1NlcXVlbmNlU3RhcnRlZChpbnQgbWluSW5kZXgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIHRoZSBjb21wbGV0aW9uIG9mIGFuIHNlcXVlbmNlIG9mIGltYWdlIHJlYWRzIGJ5IGNhbGxpbmcKLSAgICAgKiBzZXF1ZW5jZUNvbXBsZXRlIG1ldGhvZCBvbiBhbGwgcmVnaXN0ZXJlZCBJSU9SZWFkUHJvZ3Jlc3NMaXN0ZW5lcnMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgcHJvY2Vzc1NlcXVlbmNlQ29tcGxldGUoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyB0aGUgc3RhcnQgb2YgYW4gaW1hZ2UgcmVhZCBieSBjYWxsaW5nIHRoZSBpbWFnZVN0YXJ0ZWQgbWV0aG9kCi0gICAgICogb24gYWxsIHJlZ2lzdGVyZWQgSUlPUmVhZFByb2dyZXNzTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgaW5kZXguCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgcHJvY2Vzc0ltYWdlU3RhcnRlZChpbnQgaW1hZ2VJbmRleCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgdGhlIGN1cnJlbnQgcGVyY2VudGFnZSBvZiBpbWFnZSBjb21wbGV0aW9uIGJ5IGNhbGxpbmcgdGhlCi0gICAgICogaW1hZ2VQcm9ncmVzcyBtZXRob2Qgb24gYWxsIHJlZ2lzdGVyZWQgSUlPUmVhZFByb2dyZXNzTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwZXJjZW50YWdlRG9uZQotICAgICAqICAgICAgICAgICAgdGhlIHBlcmNlbnRhZ2UgZG9uZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzSW1hZ2VQcm9ncmVzcyhmbG9hdCBwZXJjZW50YWdlRG9uZSkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgaW1hZ2UgY29tcGxldGlvbiBieSBjYWxsaW5nIHRoZSBpbWFnZUNvbXBsZXRlIG1ldGhvZCBvbiBhbGwKLSAgICAgKiByZWdpc3RlcmVkIElJT1JlYWRQcm9ncmVzc0xpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzSW1hZ2VDb21wbGV0ZSgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIHRoZSBzdGFydCBvZiBhIHRodW1ibmFpbCByZWFkIGJ5IGNhbGxpbmcgdGhlIHRodW1ibmFpbFN0YXJ0ZWQKLSAgICAgKiBtZXRob2Qgb24gYWxsIHJlZ2lzdGVyZWQgSUlPUmVhZFByb2dyZXNzTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgaW5kZXguCi0gICAgICogQHBhcmFtIHRodW1ibmFpbEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgdGh1bWJuYWlsIGluZGV4LgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NUaHVtYm5haWxTdGFydGVkKGludCBpbWFnZUluZGV4LCBpbnQgdGh1bWJuYWlsSW5kZXgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIHRoZSBjdXJyZW50IHBlcmNlbnRhZ2Ugb2YgdGh1bWJuYWlsIGNvbXBsZXRpb24gYnkgY2FsbGluZyB0aGUKLSAgICAgKiB0aHVtYm5haWxQcm9ncmVzcyBtZXRob2Qgb24gYWxsIHJlZ2lzdGVyZWQgSUlPUmVhZFByb2dyZXNzTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwZXJjZW50YWdlRG9uZQotICAgICAqICAgICAgICAgICAgdGhlIHBlcmNlbnRhZ2UgZG9uZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzVGh1bWJuYWlsUHJvZ3Jlc3MoZmxvYXQgcGVyY2VudGFnZURvbmUpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIHRoZSBjb21wbGV0aW9uIG9mIGEgdGh1bWJuYWlsIHJlYWQgYnkgY2FsbGluZyB0aGUKLSAgICAgKiB0aHVtYm5haWxDb21wbGV0ZSBtZXRob2Qgb24gYWxsIHJlZ2lzdGVyZWQgSUlPUmVhZFByb2dyZXNzTGlzdGVuZXJzLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NUaHVtYm5haWxDb21wbGV0ZSgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIGEgcmVhZCBhYm9ydGVkIGV2ZW50IGJ5IGNhbGxpbmcgdGhlIHJlYWRBYm9ydGVkIG1ldGhvZCBvbiBhbGwKLSAgICAgKiByZWdpc3RlcmVkIElJT1JlYWRQcm9ncmVzc0xpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzUmVhZEFib3J0ZWQoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyB0aGUgYmVnaW5uaW5nIG9mIGEgcHJvZ3Jlc3NpdmUgcGFzcyBieSBjYWxsaW5nIHRoZSBwYXNzU3RhcnRlZAotICAgICAqIG1ldGhvZCBvbiBhbGwgcmVnaXN0ZXJlZCBJSU9SZWFkVXBkYXRlTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0aGVJbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHRvIGJlIHVwZGF0ZWQuCi0gICAgICogQHBhcmFtIHBhc3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBjdXJyZW50IHBhc3MgaW5kZXguCi0gICAgICogQHBhcmFtIG1pblBhc3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBtaW5pbXVtIHBhc3MgaW5kZXguCi0gICAgICogQHBhcmFtIG1heFBhc3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBtYXhpbXVtIHBhc3MgaW5kZXguCi0gICAgICogQHBhcmFtIG1pblgKLSAgICAgKiAgICAgICAgICAgIHRoZSBYIGNvb3JkaW5hdGUgb2Ygb2YgdGhlIHVwcGVyIGxlZnQgcGl4ZWwuCi0gICAgICogQHBhcmFtIG1pblkKLSAgICAgKiAgICAgICAgICAgIHRoZSBZIGNvb3JkaW5hdGUgb2Ygb2YgdGhlIHVwcGVyIGxlZnQgcGl4ZWwuCi0gICAgICogQHBhcmFtIHBlcmlvZFgKLSAgICAgKiAgICAgICAgICAgIHRoZSBob3Jpem9udGFsIHNlcGFyYXRpb24gYmV0d2VlbiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHBlcmlvZFkKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZXJ0aWNhbCBzZXBhcmF0aW9uIGJldHdlZW4gcGl4ZWxzLgotICAgICAqIEBwYXJhbSBiYW5kcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBhZmZlY3RlZCBiYW5kcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzUGFzc1N0YXJ0ZWQoQnVmZmVyZWRJbWFnZSB0aGVJbWFnZSwgaW50IHBhc3MsIGludCBtaW5QYXNzLCBpbnQgbWF4UGFzcywKLSAgICAgICAgICAgIGludCBtaW5YLCBpbnQgbWluWSwgaW50IHBlcmlvZFgsIGludCBwZXJpb2RZLCBpbnRbXSBiYW5kcykgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgdGhlIHVwZGF0ZSBvZiBhIHNldCBvZiBzYW1wbGVzIGJ5IGNhbGxpbmcgdGhlIGltYWdlVXBkYXRlCi0gICAgICogbWV0aG9kIG9uIGFsbCByZWdpc3RlcmVkIElJT1JlYWRVcGRhdGVMaXN0ZW5lcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRoZUltYWdlCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgdG8gYmUgdXBkYXRlZC4KLSAgICAgKiBAcGFyYW0gbWluWAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBwaXhlbC4KLSAgICAgKiBAcGFyYW0gbWluWQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBwaXhlbC4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB1cGRhdGVkIGFyZWEuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB1cGRhdGVkIGFyZWEuCi0gICAgICogQHBhcmFtIHBlcmlvZFgKLSAgICAgKiAgICAgICAgICAgIHRoZSBob3Jpem9udGFsIHNlcGFyYXRpb24gYmV0d2VlbiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHBlcmlvZFkKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZXJ0aWNhbCBzZXBhcmF0aW9uIGJldHdlZW4gcGl4ZWxzLgotICAgICAqIEBwYXJhbSBiYW5kcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBhZmZlY3RlZCBiYW5kcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzSW1hZ2VVcGRhdGUoQnVmZmVyZWRJbWFnZSB0aGVJbWFnZSwgaW50IG1pblgsIGludCBtaW5ZLCBpbnQgd2lkdGgsCi0gICAgICAgICAgICBpbnQgaGVpZ2h0LCBpbnQgcGVyaW9kWCwgaW50IHBlcmlvZFksIGludFtdIGJhbmRzKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyB0aGUgZW5kIG9mIGEgcHJvZ3Jlc3NpdmUgcGFzcyBieSBjYWxsaW5nIHBhc3NDb21wbGV0ZSBtZXRob2Qgb2YKLSAgICAgKiByZWdpc3RlcmVkIElJT1JlYWRVcGRhdGVMaXN0ZW5lcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRoZUltYWdlCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgdG8gYmUgdXBkYXRlZC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzUGFzc0NvbXBsZXRlKEJ1ZmZlcmVkSW1hZ2UgdGhlSW1hZ2UpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIHRoZSBiZWdpbm5pbmcgb2YgYSB0aHVtYm5haWwgcHJvZ3Jlc3NpdmUgcGFzcyBieSBjYWxsaW5nIHRoZQotICAgICAqIHRodW1ibmFpbFBhc3NTdGFydGVkIG1ldGhvZCBvbiBhbGwgcmVnaXN0ZXJlZCBJSU9SZWFkVXBkYXRlTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0aGVUaHVtYm5haWwKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aHVtYm5haWwgdG8gYmUgdXBkYXRlZC4KLSAgICAgKiBAcGFyYW0gcGFzcwotICAgICAqICAgICAgICAgICAgdGhlIGN1cnJlbnQgcGFzcyBpbmRleC4KLSAgICAgKiBAcGFyYW0gbWluUGFzcwotICAgICAqICAgICAgICAgICAgdGhlIG1pbmltdW0gcGFzcyBpbmRleC4KLSAgICAgKiBAcGFyYW0gbWF4UGFzcwotICAgICAqICAgICAgICAgICAgdGhlIG1heGltdW0gcGFzcyBpbmRleC4KLSAgICAgKiBAcGFyYW0gbWluWAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBwaXhlbC4KLSAgICAgKiBAcGFyYW0gbWluWQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBwaXhlbC4KLSAgICAgKiBAcGFyYW0gcGVyaW9kWAotICAgICAqICAgICAgICAgICAgdGhlIGhvcml6b250YWwgc2VwYXJhdGlvbiBiZXR3ZWVuIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gcGVyaW9kWQotICAgICAqICAgICAgICAgICAgdGhlIHZlcnRpY2FsIHNlcGFyYXRpb24gYmV0d2VlbiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGJhbmRzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGFmZmVjdGVkIGJhbmRzLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NUaHVtYm5haWxQYXNzU3RhcnRlZChCdWZmZXJlZEltYWdlIHRoZVRodW1ibmFpbCwgaW50IHBhc3MsIGludCBtaW5QYXNzLAotICAgICAgICAgICAgaW50IG1heFBhc3MsIGludCBtaW5YLCBpbnQgbWluWSwgaW50IHBlcmlvZFgsIGludCBwZXJpb2RZLCBpbnRbXSBiYW5kcykgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgdGhlIHVwZGF0ZSBvZiBhIHNldCBvZiBzYW1wbGVzIGluIGEgdGh1bWJuYWlsIGltYWdlIGJ5IGNhbGxpbmcKLSAgICAgKiB0aGUgdGh1bWJuYWlsVXBkYXRlIG1ldGhvZCBvbiBhbGwgcmVnaXN0ZXJlZCBJSU9SZWFkVXBkYXRlTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0aGVUaHVtYm5haWwKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aHVtYm5haWwgdG8gYmUgdXBkYXRlZC4KLSAgICAgKiBAcGFyYW0gbWluWAotICAgICAqICAgICAgICAgICAgdGhlIFggY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBwaXhlbC4KLSAgICAgKiBAcGFyYW0gbWluWQotICAgICAqICAgICAgICAgICAgdGhlIFkgY29vcmRpbmF0ZSBvZiB0aGUgdXBwZXIgbGVmdCBwaXhlbC4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB0b3RhbCB3aWR0aCBvZiB0aGUgdXBkYXRlZCBhcmVhLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSB0b3RhbCBoZWlnaHQgb2YgdGhlIHVwZGF0ZWQgYXJlYS4KLSAgICAgKiBAcGFyYW0gcGVyaW9kWAotICAgICAqICAgICAgICAgICAgdGhlIGhvcml6b250YWwgc2VwYXJhdGlvbiBiZXR3ZWVuIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gcGVyaW9kWQotICAgICAqICAgICAgICAgICAgdGhlIHZlcnRpY2FsIHNlcGFyYXRpb24gYmV0d2VlbiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGJhbmRzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGFmZmVjdGVkIGJhbmRzLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NUaHVtYm5haWxVcGRhdGUoQnVmZmVyZWRJbWFnZSB0aGVUaHVtYm5haWwsIGludCBtaW5YLCBpbnQgbWluWSwKLSAgICAgICAgICAgIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IHBlcmlvZFgsIGludCBwZXJpb2RZLCBpbnRbXSBiYW5kcykgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgdGhlIGVuZCBvZiBhIHRodW1ibmFpbCBwcm9ncmVzc2l2ZSBwYXNzIGJ5IGNhbGxpbmcgdGhlCi0gICAgICogdGh1bWJuYWlsUGFzc0NvbXBsZXRlIG1ldGhvZCBvbiBhbGwgcmVnaXN0ZXJlZCBJSU9SZWFkVXBkYXRlTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0aGVUaHVtYm5haWwKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aHVtYm5haWwgdG8gYmUgdXBkYXRlZC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzVGh1bWJuYWlsUGFzc0NvbXBsZXRlKEJ1ZmZlcmVkSW1hZ2UgdGhlVGh1bWJuYWlsKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyBhIHdhcm5pbmcgbWVzc2FnZSBieSBjYWxsaW5nIHdhcm5pbmdPY2N1cnJlZCBtZXRob2Qgb2YKLSAgICAgKiByZWdpc3RlcmVkIElJT1JlYWRXYXJuaW5nTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3YXJuaW5nCi0gICAgICogICAgICAgICAgICB0aGUgd2FybmluZy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzV2FybmluZ09jY3VycmVkKFN0cmluZyB3YXJuaW5nKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyBhIHdhcm5pbmcgYnkgY2FsbGluZyB0aGUgd2FybmluZ09jY3VycmVkIG1ldGhvZCBvZiBvbiBhbGwKLSAgICAgKiByZWdpc3RlcmVkIElJT1JlYWRXYXJuaW5nTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiYXNlTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGJhc2UgbmFtZSBvZiBSZXNvdXJjZUJ1bmRsZXMuCi0gICAgICogQHBhcmFtIGtleXdvcmQKLSAgICAgKiAgICAgICAgICAgIHRoZSBrZXl3b3JkIHRvIGluZGV4IHRoZSB3YXJuaW5nIGFtb25nIFJlc291cmNlQnVuZGxlcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzV2FybmluZ09jY3VycmVkKFN0cmluZyBiYXNlTmFtZSwgU3RyaW5nIGtleXdvcmQpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVzZXRzIHRoaXMgSW1hZ2VSZWFkZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVzZXQoKSB7Ci0gICAgICAgIC8vIGRlZgotICAgICAgICBzZXRJbnB1dChudWxsLCBmYWxzZSk7Ci0gICAgICAgIHNldExvY2FsZShudWxsKTsKLSAgICAgICAgcmVtb3ZlQWxsSUlPUmVhZFVwZGF0ZUxpc3RlbmVycygpOwotICAgICAgICByZW1vdmVBbGxJSU9SZWFkV2FybmluZ0xpc3RlbmVycygpOwotICAgICAgICByZW1vdmVBbGxJSU9SZWFkUHJvZ3Jlc3NMaXN0ZW5lcnMoKTsKLSAgICAgICAgY2xlYXJBYm9ydFJlcXVlc3QoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEaXNwb3NlcyBvZiBhbnkgcmVzb3VyY2VzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGRpc3Bvc2UoKSB7Ci0gICAgICAgIC8vIGRvIG5vdGhpbmcgYnkgZGVmCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgcmVnaW9uIG9mIHNvdXJjZSBpbWFnZSB0aGF0IHNob3VsZCBiZSByZWFkIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIHdpZHRoLCBoZWlnaHQgYW5kIEltYWdlUmVhZFBhcmFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUmVhZFBhcmFtIG9iamVjdCwgb3IgbnVsbC4KLSAgICAgKiBAcGFyYW0gc3JjV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgaW1hZ2UncyB3aWR0aC4KLSAgICAgKiBAcGFyYW0gc3JjSGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgc291cmNlIGltYWdlJ3MgaGVpZ2h0LgotICAgICAqIEByZXR1cm4gdGhlIFJlY3RhbmdsZSBvZiBzb3VyY2UgcmVnaW9uLgotICAgICAqLwotICAgIHByb3RlY3RlZCBzdGF0aWMgUmVjdGFuZ2xlIGdldFNvdXJjZVJlZ2lvbihJbWFnZVJlYWRQYXJhbSBwYXJhbSwgaW50IHNyY1dpZHRoLCBpbnQgc3JjSGVpZ2h0KSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbXB1dGVzIHRoZSBzcGVjaWZpZWQgc291cmNlIHJlZ2lvbiBhbmQgdGhlIHNwZWNpZmllZCBkZXN0aW5hdGlvbiByZWdpb24KLSAgICAgKiB3aXRoIHRoZSBzcGVjaWZpZWQgdGhlIHdpZHRoIGFuZCBoZWlnaHQgb2YgdGhlIHNvdXJjZSBpbWFnZSwgYW4gb3B0aW9uYWwKLSAgICAgKiBkZXN0aW5hdGlvbiBpbWFnZSwgYW5kIGFuIEltYWdlUmVhZFBhcmFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIGFuIEltYWdlUmVhZFBhcmFtIG9iamVjdCwgb3IgbnVsbC4KLSAgICAgKiBAcGFyYW0gc3JjV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgaW1hZ2UncyB3aWR0aC4KLSAgICAgKiBAcGFyYW0gc3JjSGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgc291cmNlIGltYWdlJ3MgaGVpZ2h0LgotICAgICAqIEBwYXJhbSBpbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIGRlc3RpbmF0aW9uIGltYWdlLgotICAgICAqIEBwYXJhbSBzcmNSZWdpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBzb3VyY2UgcmVnaW9uLgotICAgICAqIEBwYXJhbSBkZXN0UmVnaW9uCi0gICAgICogICAgICAgICAgICB0aGUgZGVzdGluYXRpb24gcmVnaW9uLgotICAgICAqLwotICAgIHByb3RlY3RlZCBzdGF0aWMgdm9pZCBjb21wdXRlUmVnaW9ucyhJbWFnZVJlYWRQYXJhbSBwYXJhbSwgaW50IHNyY1dpZHRoLCBpbnQgc3JjSGVpZ2h0LAotICAgICAgICAgICAgQnVmZmVyZWRJbWFnZSBpbWFnZSwgUmVjdGFuZ2xlIHNyY1JlZ2lvbiwgUmVjdGFuZ2xlIGRlc3RSZWdpb24pIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHRoZSB2YWxpZGl0eSBvZiB0aGUgc291cmNlIGFuZCBkZXN0aW5hdGlvbiBiYW5kIGFuZCBpcyBjYWxsZWQgd2hlbgotICAgICAqIHRoZSByZWFkZXIga25vd3MgdGhlIG51bWJlciBvZiBiYW5kcyBvZiB0aGUgc291cmNlIGltYWdlIGFuZCB0aGUgbnVtYmVyCi0gICAgICogb2YgYmFuZHMgb2YgdGhlIGRlc3RpbmF0aW9uIGltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUmVhZFBhcmFtIGZvciByZWFkaW5nIHRoZSBJbWFnZS4KLSAgICAgKiBAcGFyYW0gbnVtU3JjQmFuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgYmFuZHMgaW4gdGhlIHNvdXJjZS4KLSAgICAgKiBAcGFyYW0gbnVtRHN0QmFuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgYmFuZHMgaW4gdGhlIGRlc3RpbmF0aW9uLgotICAgICAqLwotICAgIHByb3RlY3RlZCBzdGF0aWMgdm9pZCBjaGVja1JlYWRQYXJhbUJhbmRTZXR0aW5ncyhJbWFnZVJlYWRQYXJhbSBwYXJhbSwgaW50IG51bVNyY0JhbmRzLAotICAgICAgICAgICAgaW50IG51bURzdEJhbmRzKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRlc3RpbmF0aW9uIGltYWdlIHdoZXJlIHRoZSBkZWNvZGVkIGRhdGEgaXMgd3JpdHRlbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcGFyYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVJlYWRQYXJhbS4KLSAgICAgKiBAcGFyYW0gaW1hZ2VUeXBlcwotICAgICAqICAgICAgICAgICAgdGhlIGl0ZXJhdG9yIG9mIEltYWdlVHlwZVNwZWNpZmllciBvYmplY3RzLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBpbWFnZSBiZWluZyBkZWNvZGVkLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIGltYWdlIGJlaW5nIGRlY29kZWQuCi0gICAgICogQHJldHVybiB0aGUgQnVmZmVyZWRJbWFnZSB3aGVyZSBkZWNvZGVkIHBpeGVscyBzaG91bGQgYmUgd3JpdHRlbi4KLSAgICAgKiBAdGhyb3dzIElJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIHRoZSBJSU9FeGNlcHRpb24gaXMgdGhyb3duIGlmIHRoZXJlIGlzIG5vIHN1aXRhYmxlCi0gICAgICogICAgICAgICAgICAgSW1hZ2VUeXBlU3BlY2lmaWVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBzdGF0aWMgQnVmZmVyZWRJbWFnZSBnZXREZXN0aW5hdGlvbihJbWFnZVJlYWRQYXJhbSBwYXJhbSwKLSAgICAgICAgICAgIEl0ZXJhdG9yPEltYWdlVHlwZVNwZWNpZmllcj4gaW1hZ2VUeXBlcywgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB0aHJvd3MgSUlPRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vSW1hZ2VUcmFuc2NvZGVyLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9JbWFnZVRyYW5zY29kZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjMyZDg5MC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9JbWFnZVRyYW5zY29kZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDY3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW87Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLm1ldGFkYXRhLklJT01ldGFkYXRhOwotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VUeXBlU3BlY2lmaWVyOwotCi0vKioKLSAqIFRoZSBJbWFnZVRyYW5zY29kZXIgaW50ZXJmYWNlIGlzIHRvIGJlIGltcGxlbWVudGVkIGJ5IGNsYXNzZXMgdGhhdCBwZXJmb3JtCi0gKiBpbWFnZSB0cmFuc2NvZGluZyBvcGVyYXRpb25zLCB0aGF0IGlzLCB0YWtlIGltYWdlcyB3cml0dGVuIGluIG9uZSBmb3JtYXQgYW5kCi0gKiB3cml0ZSB0aGVtIGluIGFub3RoZXIgZm9ybWF0IHVzaW5nIHJlYWQvd3JpdGUgb3BlcmF0aW9ucy4gU29tZSBpbWFnZSBkYXRhIGNhbgotICogYmUgbG9zdCBpbiBzdWNoIHByb2Nlc3Nlcy4gVGhlIEltYWdlVHJhbnNjb2RlciBpbnRlcmZhY2UgY29udmVydHMgbWV0YWRhdGEKLSAqIG9iamVjdHMgKElJT01ldGFkYXRhKSBvZiBJbWFnZVJlYWRlciB0byBhcHByb3ByaWF0ZSBtZXRhZGF0YSBvYmplY3QgZm9yCi0gKiBJbWFnZVdyaXRlci4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgSW1hZ2VUcmFuc2NvZGVyIHsKLQotICAgIC8qKgotICAgICAqIENvbnZlcnRzIHRoZSBzcGVjaWZpZWQgSUlPTWV0YWRhdGEgb2JqZWN0IHVzaW5nIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBJbWFnZVdyaXRlUGFyYW0gZm9yIG9idGFpbmluZyB3cml0ZXIncyBtZXRhZGF0YSBzdHJ1Y3R1cmUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGluRGF0YQotICAgICAqICAgICAgICAgICAgdGhlIElJT01ldGFkYXRhLgotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlV3JpdGVQYXJhbS4KLSAgICAgKiBAcmV0dXJuIHRoZSBJSU9NZXRhZGF0YSwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBJSU9NZXRhZGF0YSBjb252ZXJ0U3RyZWFtTWV0YWRhdGEoSUlPTWV0YWRhdGEgaW5EYXRhLCBJbWFnZVdyaXRlUGFyYW0gcGFyYW0pOwotCi0gICAgLyoqCi0gICAgICogQ29udmVydHMgdGhlIHNwZWNpZmllZCBJSU9NZXRhZGF0YSBvYmplY3QgdXNpbmcgdGhlIHNwZWNpZmllZAotICAgICAqIEltYWdlV3JpdGVQYXJhbSBmb3Igb2J0YWluaW5nIHdyaXRlcidzIG1ldGFkYXRhIHN0cnVjdHVyZSBhbmQKLSAgICAgKiBJbWFnZVR5cGVTcGVjaWZpZXIgb2JqZWN0IGZvciBvYnRhaW5pbmcgdGhlIGxheW91dCBhbmQgY29sb3IgaW5mb3JtYXRpb24KLSAgICAgKiBvZiB0aGUgaW1hZ2UgZm9yIHRoaXMgbWV0YWRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIGluRGF0YQotICAgICAqICAgICAgICAgICAgdGhlIElJT01ldGFkYXRhLgotICAgICAqIEBwYXJhbSBpbWFnZVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICogQHBhcmFtIHBhcmFtCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VXcml0ZVBhcmFtLgotICAgICAqIEByZXR1cm4gdGhlIElJT01ldGFkYXRhLCBvciBudWxsLgotICAgICAqLwotICAgIElJT01ldGFkYXRhIGNvbnZlcnRJbWFnZU1ldGFkYXRhKElJT01ldGFkYXRhIGluRGF0YSwgSW1hZ2VUeXBlU3BlY2lmaWVyIGltYWdlVHlwZSwKLSAgICAgICAgICAgIEltYWdlV3JpdGVQYXJhbSBwYXJhbSk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9JbWFnZVR5cGVTcGVjaWZpZXIuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL0ltYWdlVHlwZVNwZWNpZmllci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1MDViMWM0Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL0ltYWdlVHlwZVNwZWNpZmllci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzQ3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW87Ci0KLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlNhbXBsZU1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmVuZGVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5jb2xvci5Db2xvclNwYWNlOwotCi0vKioKLSAqIFRoZSBJbWFnZVR5cGVTcGVjaWZpZXIgY2xhc3MgcGVyZm9ybXMgY29udmVyc2lvbiBvcGVyYXRpb25zIG9uIHRoZQotICogU2FtcGxlTW9kZWwgYW5kIHRoZSBDb2xvck1vZGVsIG9mIGFuIGltYWdlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEltYWdlVHlwZVNwZWNpZmllciB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29sb3JNb2RlbCBvZiB0aGlzIEltYWdlVHlwZVNwZWNpZmllci4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgQ29sb3JNb2RlbCBjb2xvck1vZGVsOwotCi0gICAgLyoqCi0gICAgICogVGhlIFNhbXBsZU1vZGVsIG9mIHRoaXMgSW1hZ2VUeXBlU3BlY2lmaWVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTYW1wbGVNb2RlbCBzYW1wbGVNb2RlbDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJbWFnZVR5cGVTcGVjaWZpZXIgd2l0aCB0aGUgc3BlY2lmaWVkIENvbG9yTW9kZWwgYW5kCi0gICAgICogU2FtcGxlTW9kZWwgb2JqZWN0cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29sb3JNb2RlbAotICAgICAqICAgICAgICAgICAgdGhlIENvbG9yTW9kZWwuCi0gICAgICogQHBhcmFtIHNhbXBsZU1vZGVsCi0gICAgICogICAgICAgICAgICB0aGUgU2FtcGxlTW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlVHlwZVNwZWNpZmllcihDb2xvck1vZGVsIGNvbG9yTW9kZWwsIFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsKSB7Ci0gICAgICAgIGlmIChjb2xvck1vZGVsID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImNvbG9yIG1vZGVsIHNob3VsZCBub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChzYW1wbGVNb2RlbCA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJzYW1wbGUgbW9kZWwgc2hvdWxkIG5vdCBiZSBOVUxMIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKCFjb2xvck1vZGVsLmlzQ29tcGF0aWJsZVNhbXBsZU1vZGVsKHNhbXBsZU1vZGVsKSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiY29sb3IgYW5kIHNhbXBsZSBtb2RlbHMgYXJlIG5vdCBjb21wYXRpYmxlIik7Ci0gICAgICAgIH0KLQotICAgICAgICB0aGlzLmNvbG9yTW9kZWwgPSBjb2xvck1vZGVsOwotICAgICAgICB0aGlzLnNhbXBsZU1vZGVsID0gc2FtcGxlTW9kZWw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEltYWdlVHlwZVNwZWNpZmllciB1c2luZyB0aGUgc3BlY2lmaWVkIFJlbmRlcmVkSW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJlbmRlcmVkSW1hZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJlZEltYWdlLgotICAgICAqLwotICAgIHB1YmxpYyBJbWFnZVR5cGVTcGVjaWZpZXIoUmVuZGVyZWRJbWFnZSByZW5kZXJlZEltYWdlKSB7Ci0gICAgICAgIGlmIChyZW5kZXJlZEltYWdlID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImltYWdlIHNob3VsZCBub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0gICAgICAgIHRoaXMuY29sb3JNb2RlbCA9IHJlbmRlcmVkSW1hZ2UuZ2V0Q29sb3JNb2RlbCgpOwotICAgICAgICB0aGlzLnNhbXBsZU1vZGVsID0gcmVuZGVyZWRJbWFnZS5nZXRTYW1wbGVNb2RlbCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYW4gSW1hZ2VUeXBlU3BlY2lmaWVyIHdpdGggdGhlIHNwZWNpZmllZCBEaXJlY3RDb2xvck1vZGVsIGFuZCBhCi0gICAgICogcGFja2VkIFNhbXBsZU1vZGVsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvclNwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgQ29sb3JTcGFjZS4KLSAgICAgKiBAcGFyYW0gcmVkTWFzawotICAgICAqICAgICAgICAgICAgdGhlIHJlZCBtYXNrLgotICAgICAqIEBwYXJhbSBncmVlbk1hc2sKLSAgICAgKiAgICAgICAgICAgIHRoZSBncmVlbiBtYXNrLgotICAgICAqIEBwYXJhbSBibHVlTWFzawotICAgICAqICAgICAgICAgICAgdGhlIGJsdWUgbWFzay4KLSAgICAgKiBAcGFyYW0gYWxwaGFNYXNrCi0gICAgICogICAgICAgICAgICB0aGUgYWxwaGEgbWFzay4KLSAgICAgKiBAcGFyYW0gdHJhbnNmZXJUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgdHJhbnNmZXIgdHlwZS4KLSAgICAgKiBAcGFyYW0gaXNBbHBoYVByZW11bHRpcGxpZWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBwYXJhbWV0ZXIgaW5kaWNhdGVzIGlmIHRoZSBjb2xvciBjaGFubmVsIGlzIHByZS1tdWx0aXBsaWVkCi0gICAgICogICAgICAgICAgICBieSBhbHBoYS4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJbWFnZVR5cGVTcGVjaWZpZXIgY3JlYXRlUGFja2VkKENvbG9yU3BhY2UgY29sb3JTcGFjZSwgaW50IHJlZE1hc2ssCi0gICAgICAgICAgICBpbnQgZ3JlZW5NYXNrLCBpbnQgYmx1ZU1hc2ssIGludCBhbHBoYU1hc2ssIGludCB0cmFuc2ZlclR5cGUsCi0gICAgICAgICAgICBib29sZWFuIGlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGFuIEltYWdlVHlwZVNwZWNpZmllciB3aXRoIHNwZWNpZmllZCBDb21wb25lbnRDb2xvck1vZGVsIGFuZCBhCi0gICAgICogUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb2xvclNwYWNlCi0gICAgICogICAgICAgICAgICB0aGUgQ29sb3JTcGFjZS4KLSAgICAgKiBAcGFyYW0gYmFuZE9mZnNldHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBiYW5kIG9mZnNldHMuCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlLgotICAgICAqIEBwYXJhbSBoYXNBbHBoYQotICAgICAqICAgICAgICAgICAgdGhlIHBhcmFtZXRlciBpbmRpY2F0ZXMgaWYgYWxwaGEgY2hhbm5lbCBpcyBuZWVkZWQuCi0gICAgICogQHBhcmFtIGlzQWxwaGFQcmVtdWx0aXBsaWVkCi0gICAgICogICAgICAgICAgICB0aGUgcGFyYW1ldGVyIGluZGljYXRlcyBpZiB0aGUgY29sb3IgY2hhbm5lbCBpcyBwcmUtbXVsdGlwbGllZAotICAgICAqICAgICAgICAgICAgYnkgYWxwaGEuCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VUeXBlU3BlY2lmaWVyLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgSW1hZ2VUeXBlU3BlY2lmaWVyIGNyZWF0ZUludGVybGVhdmVkKENvbG9yU3BhY2UgY29sb3JTcGFjZSwgaW50W10gYmFuZE9mZnNldHMsCi0gICAgICAgICAgICBpbnQgZGF0YVR5cGUsIGJvb2xlYW4gaGFzQWxwaGEsIGJvb2xlYW4gaXNBbHBoYVByZW11bHRpcGxpZWQpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBJbWFnZVR5cGVTcGVjaWZpZXIgZm9yIGEgaW1hZ2Ugd2l0aCBhIEJhbmRlZFNhbXBsZU1vZGVsIGFuZCBhCi0gICAgICogQ29tcG9uZW50Q29sb3JNb2RlbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29sb3JTcGFjZQotICAgICAqICAgICAgICAgICAgdGhlIENvbG9yU3BhY2UuCi0gICAgICogQHBhcmFtIGJhbmtJbmRpY2VzCi0gICAgICogICAgICAgICAgICB0aGUgYmFuayBpbmRpY2VzLgotICAgICAqIEBwYXJhbSBiYW5kT2Zmc2V0cwotICAgICAqICAgICAgICAgICAgdGhlIGJhbmQgb2Zmc2V0cy4KLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUuCi0gICAgICogQHBhcmFtIGhhc0FscGhhCi0gICAgICogICAgICAgICAgICB0aGUgcGFyYW1ldGVyIGluZGljYXRlcyBhIHByZXNlbmNlIG9mIGFscGhhIGNoYW5uZWwuCi0gICAgICogQHBhcmFtIGlzQWxwaGFQcmVtdWx0aXBsaWVkCi0gICAgICogICAgICAgICAgICB0aGUgcGFyYW1ldGVyIGluZGljYXRlcyB3aGV0aGVyIG9yIG5vdCBjb2xvciBjaGFubmVsIGlzIGFscGhhCi0gICAgICogICAgICAgICAgICBwcmUtbXVsdGlwbGllZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBpbWFnZSB0eXBlIHNwZWNpZmllcgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgSW1hZ2VUeXBlU3BlY2lmaWVyIGNyZWF0ZUJhbmRlZChDb2xvclNwYWNlIGNvbG9yU3BhY2UsIGludFtdIGJhbmtJbmRpY2VzLAotICAgICAgICAgICAgaW50W10gYmFuZE9mZnNldHMsIGludCBkYXRhVHlwZSwgYm9vbGVhbiBoYXNBbHBoYSwgYm9vbGVhbiBpc0FscGhhUHJlbXVsdGlwbGllZCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIEltYWdlVHlwZVNwZWNpZmllciBmb3IgYSBncmF5c2NhbGUgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgYml0cyBwZXIgZ3JheSB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUuCi0gICAgICogQHBhcmFtIGlzU2lnbmVkCi0gICAgICogICAgICAgICAgICBhIHNpZ25lZCBmbGFnLgotICAgICAqIEByZXR1cm4gdGhlIEltYWdlVHlwZVNwZWNpZmllci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEltYWdlVHlwZVNwZWNpZmllciBjcmVhdGVHcmF5c2NhbGUoaW50IGJpdHMsIGludCBkYXRhVHlwZSwgYm9vbGVhbiBpc1NpZ25lZCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhIEltYWdlVHlwZVNwZWNpZmllciBmb3IgYSBncmF5c2NhbGUgaW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJpdHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgYml0cyBwZXIgZ3JheSB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUuCi0gICAgICogQHBhcmFtIGlzU2lnbmVkCi0gICAgICogICAgICAgICAgICBhIHNpZ25lZCBmbGFnLgotICAgICAqIEBwYXJhbSBpc0FscGhhUHJlbXVsdGlwbGllZAotICAgICAqICAgICAgICAgICAgdGhlIHBhcmFtZXRlciBpbmRpY2F0ZXMgaWYgY29sb3IgY2hhbm5lbCBpcyBwcmUtbXVsdGlwbGllZCBieQotICAgICAqICAgICAgICAgICAgYWxwaGEsIG9yIG5vdC4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJbWFnZVR5cGVTcGVjaWZpZXIgY3JlYXRlR3JheXNjYWxlKGludCBiaXRzLCBpbnQgZGF0YVR5cGUsIGJvb2xlYW4gaXNTaWduZWQsCi0gICAgICAgICAgICBib29sZWFuIGlzQWxwaGFQcmVtdWx0aXBsaWVkKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgSW1hZ2VUeXBlU3BlY2lmaWVyIHdpdGggdGhlIGluZGV4ZWQgaW1hZ2UgZm9ybWF0LgotICAgICAqIAotICAgICAqIEBwYXJhbSByZWRMVVQKLSAgICAgKiAgICAgICAgICAgIHRoZSByZWQgdmFsdWVzIG9mIGluZGljZXMuCi0gICAgICogQHBhcmFtIGdyZWVuTFVUCi0gICAgICogICAgICAgICAgICB0aGUgZ3JlZW4gdmFsdWVzIG9mIGluZGljZXMuCi0gICAgICogQHBhcmFtIGJsdWVMVVQKLSAgICAgKiAgICAgICAgICAgIHRoZSBibHVlIHZhbHVlcyBvZiBpbmRpY2VzLgotICAgICAqIEBwYXJhbSBhbHBoYUxVVAotICAgICAqICAgICAgICAgICAgdGhlIGFscGhhIHZhbHVlcyBvZiBpbmRpY2VzLgotICAgICAqIEBwYXJhbSBiaXRzCi0gICAgICogICAgICAgICAgICB0aGUgYml0cyBudW1iZXIgZm9yIGVhY2ggaW5kZXguCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlLgotICAgICAqIEByZXR1cm4gdGhlIEltYWdlVHlwZVNwZWNpZmllci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEltYWdlVHlwZVNwZWNpZmllciBjcmVhdGVJbmRleGVkKGJ5dGVbXSByZWRMVVQsIGJ5dGVbXSBncmVlbkxVVCwgYnl0ZVtdIGJsdWVMVVQsCi0gICAgICAgICAgICBieXRlW10gYWxwaGFMVVQsIGludCBiaXRzLCBpbnQgZGF0YVR5cGUpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIEltYWdlVHlwZVNwZWNpZmllciBmcm9tIHRoZSBzcGVjaWZpZWQgYnVmZmVyZWQgaW1hZ2UgdHlwZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYnVmZmVyZWRJbWFnZVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBidWZmZXJlZCBpbWFnZSB0eXBlLgotICAgICAqIEByZXR1cm4gdGhlIEltYWdlVHlwZVNwZWNpZmllci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEltYWdlVHlwZVNwZWNpZmllciBjcmVhdGVGcm9tQnVmZmVyZWRJbWFnZVR5cGUoaW50IGJ1ZmZlcmVkSW1hZ2VUeXBlKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBJbWFnZVR5cGVTcGVjaWZpZXIgZnJvbSB0aGUgc3BlY2lmaWVkIFJlbmRlcmVkSW1hZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlCi0gICAgICogICAgICAgICAgICB0aGUgUmVuZGVyZWRJbWFnZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJbWFnZVR5cGVTcGVjaWZpZXIgY3JlYXRlRnJvbVJlbmRlcmVkSW1hZ2UoUmVuZGVyZWRJbWFnZSBpbWFnZSkgewotICAgICAgICBpZiAobnVsbCA9PSBpbWFnZSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiaW1hZ2Ugc2hvdWxkIG5vdCBiZSBOVUxMIik7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBJbWFnZVR5cGVTcGVjaWZpZXIoaW1hZ2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIEJ1ZmZlcmVkSW1hZ2UgdHlwZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBCdWZmZXJlZEltYWdlIHR5cGUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRCdWZmZXJlZEltYWdlVHlwZSgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG51bWJlciBvZiBjb21wb25lbnRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG51bWJlciBvZiBjb21wb25lbnRzLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TnVtQ29tcG9uZW50cygpIHsKLSAgICAgICAgcmV0dXJuIGNvbG9yTW9kZWwuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG51bWJlciBvZiBiYW5kcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgYmFuZHMuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXROdW1CYW5kcygpIHsKLSAgICAgICAgcmV0dXJuIHNhbXBsZU1vZGVsLmdldE51bUJhbmRzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIGJpdHMgcGVyIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmFuZAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIGJhbmQuCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIGJpdHMgcGVyIHRoZSBzcGVjaWZpZWQgYmFuZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEJpdHNQZXJCYW5kKGludCBiYW5kKSB7Ci0gICAgICAgIGlmIChiYW5kIDwgMCB8fCBiYW5kID49IGdldE51bUJhbmRzKCkpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gc2FtcGxlTW9kZWwuZ2V0U2FtcGxlU2l6ZShiYW5kKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBTYW1wbGVNb2RlbCBhc3NvY2lhdGVkIHdpdGggdGhpcyBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgU2FtcGxlTW9kZWwgYXNzb2NpYXRlZCB3aXRoIHRoaXMgSW1hZ2VUeXBlU3BlY2lmaWVyLgotICAgICAqLwotICAgIHB1YmxpYyBTYW1wbGVNb2RlbCBnZXRTYW1wbGVNb2RlbCgpIHsKLSAgICAgICAgcmV0dXJuIHNhbXBsZU1vZGVsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBjb21wYXRpYmxlIFNhbXBsZU1vZGVsIHdpdGggdGhlIHNwZWNpZmllZCB3aWR0aCBhbmQgaGVpZ2h0LgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQuCi0gICAgICogQHJldHVybiB0aGUgU2FtcGxlTW9kZWwuCi0gICAgICovCi0gICAgcHVibGljIFNhbXBsZU1vZGVsIGdldFNhbXBsZU1vZGVsKGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICBpZiAoKGxvbmcpd2lkdGggKiBoZWlnaHQgPiBJbnRlZ2VyLk1BWF9WQUxVRSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigid2lkdGggKiBoZWlnaHQgPiBJbnRlZ2VyLk1BWF9WQUxVRSIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBzYW1wbGVNb2RlbC5jcmVhdGVDb21wYXRpYmxlU2FtcGxlTW9kZWwod2lkdGgsIGhlaWdodCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQ29sb3JNb2RlbCBhc3NvY2lhdGVkIHdpdGggdGhpcyBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgQ29sb3JNb2RlbCBhc3NvY2lhdGVkIHdpdGggdGhpcyBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICovCi0gICAgcHVibGljIENvbG9yTW9kZWwgZ2V0Q29sb3JNb2RlbCgpIHsKLSAgICAgICAgcmV0dXJuIGNvbG9yTW9kZWw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgQnVmZmVyZWRJbWFnZSB3aXRoIHRoZSBzcGVjaWZpZWQgd2lkdGggYW5kIGhlaWdodCBhbmQgdGhlCi0gICAgICogQ29sb3JNYWRlbCBhbmQgU2FtcGxlTW9kZWwgd2hpY2ggYXJlIHNwZWNpZmllZCBieSB0aGlzCi0gICAgICogSW1hZ2VUeXBlU3BlY2lmaWVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBCdWZmZXJlZEltYWdlLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdGhlIEJ1ZmZlcmVkSW1hZ2UuCi0gICAgICogQHJldHVybiB0aGUgQnVmZmVyZWRJbWFnZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgQnVmZmVyZWRJbWFnZSBjcmVhdGVCdWZmZXJlZEltYWdlKGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29tcGFyZXMgdGhpcyBJbWFnZVR5cGVTcGVjaWZpZXIgb2JqZWN0IHdpdGggdGhlIHNwZWNpZmllZCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIG8KLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3QgdG8gYmUgY29tcGFyZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgb2JqZWN0IGlzIGFuIEltYWdlVHlwZVNwZWNpZmllciB3aXRoIHRoZSBzYW1lIGRhdGEKLSAgICAgKiAgICAgICAgIGFzIHRoaXMgSW1hZ2VUeXBlU3BlY2lmaWVyLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvKSB7Ci0gICAgICAgIGJvb2xlYW4gcnQgPSBmYWxzZTsKLSAgICAgICAgaWYgKG8gaW5zdGFuY2VvZiBJbWFnZVR5cGVTcGVjaWZpZXIpIHsKLSAgICAgICAgICAgIEltYWdlVHlwZVNwZWNpZmllciB0cyA9IChJbWFnZVR5cGVTcGVjaWZpZXIpbzsKLSAgICAgICAgICAgIHJ0ID0gY29sb3JNb2RlbC5lcXVhbHModHMuY29sb3JNb2RlbCkgJiYgc2FtcGxlTW9kZWwuZXF1YWxzKHRzLnNhbXBsZU1vZGVsKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcnQ7Ci0gICAgfQotfQpcIE5vIG5ld2xpbmUgYXQgZW5kIG9mIGZpbGUKZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL0ltYWdlV3JpdGVQYXJhbS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vSW1hZ2VXcml0ZVBhcmFtLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGQ2NjE4ODkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vSW1hZ2VXcml0ZVBhcmFtLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2NjQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2VpbzsKLQotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci1pbXBvcnQgamF2YS5hd3QuKjsKLQotLyoqCi0gKiBUaGUgSW1hZ2VXcml0ZVBhcmFtIGNsYXNzIHByb3ZpZGVzIGluZm9ybWF0aW9uIHRvIGFuIEltYWdlV3JpdGVyIGFib3V0IGhvdyBhbgotICogaW1hZ2UgaXMgdG8gYmUgZW5jb2RlZC4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJbWFnZVdyaXRlUGFyYW0gZXh0ZW5kcyBJSU9QYXJhbSB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTU9ERV9ESVNBQkxFRCBpbmRpY2F0ZXMgdGhhdCBzdHJlYW0gaXMgbm90IHRpbGVkLAotICAgICAqIHByb2dyZXNzaXZlLCBvciBjb21wcmVzc2VkLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IE1PREVfRElTQUJMRUQgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE1PREVfREVGQVVMVCBpbmRpY2F0ZXMgdGhhdCB0aGUgc3RyZWFtIHdpbGwgYmUgdGlsZWQsCi0gICAgICogcHJvZ3Jlc3NpdmUsIG9yIGNvbXByZXNzZWQgYWNjb3JkaW5nIHRvIHRoZSBwbHVnLWluJ3MgZGVmYXVsdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNT0RFX0RFRkFVTFQgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IE1PREVfRVhQTElDSVQgaW5kaWNhdGVzIHRoYXQgdGhlIHN0cmVhbSB3aWxsIGJlIHRpbGVkLAotICAgICAqIHByb2dyZXNzaXZlLCBvciBjb21wcmVzc2VkIGFjY29yZGluZyB0byBjdXJyZW50IHNldHRpbmdzIHdoaWNoIGFyZQotICAgICAqIGRlZmluZWQgYnkgc2V0IG1ldGhvZHMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTU9ERV9FWFBMSUNJVCA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTU9ERV9DT1BZX0ZST01fTUVUQURBVEEgaW5kaWNhdGVzIHRoYXQgdGhlIHN0cmVhbSB3aWxsIGJlCi0gICAgICogdGlsZWQsIHByb2dyZXNzaXZlLCBvciBjb21wcmVzc2VkIGFjY29yZGluZyB0byBzdHJlYW0gb3IgaW1hZ2UgbWV0YWRhdGEuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTU9ERV9DT1BZX0ZST01fTUVUQURBVEEgPSAzOwotCi0gICAgLyoqCi0gICAgICogV2hldGhlciB0aGUgSW1hZ2VXcml0ZXIgY2FuIHdyaXRlIHRpbGVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBib29sZWFuIGNhbldyaXRlVGlsZXMgPSBmYWxzZTsKLQotICAgIC8qKgotICAgICAqIFRoZSB0aWxpbmcgbW9kZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IHRpbGluZ01vZGUgPSBNT0RFX0NPUFlfRlJPTV9NRVRBREFUQTsKLQotICAgIC8qKgotICAgICAqIFRoZSBwcmVmZXJyZWQgdGlsZSBzaXplcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgRGltZW5zaW9uW10gcHJlZmVycmVkVGlsZVNpemVzID0gbnVsbDsKLQotICAgIC8qKgotICAgICAqIFRoZSB0aWxpbmcgc2V0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBib29sZWFuIHRpbGluZ1NldCA9IGZhbHNlOwotCi0gICAgLyoqCi0gICAgICogVGhlIHRpbGUgd2lkdGguCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCB0aWxlV2lkdGggPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIHRpbGUgaGVpZ2h0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgdGlsZUhlaWdodCA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBXaGV0aGVyIHRoZSBJbWFnZVdyaXRlciBjYW4gb2Zmc2V0IHRpbGVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBib29sZWFuIGNhbk9mZnNldFRpbGVzID0gZmFsc2U7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgdGlsZSBncmlkIHggb2Zmc2V0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBpbnQgdGlsZUdyaWRYT2Zmc2V0ID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSB0aWxlIGdyaWQgeSBvZmZzZXQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCB0aWxlR3JpZFlPZmZzZXQgPSAwOwotCi0gICAgLyoqCi0gICAgICogV2hldGhlciB0aGUgSW1hZ2VXcml0ZXIgY2FuIHdyaXRlIGluIHByb2dyZXNzaXZlIG1vZGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGJvb2xlYW4gY2FuV3JpdGVQcm9ncmVzc2l2ZSA9IGZhbHNlOwotCi0gICAgLyoqCi0gICAgICogVGhlIHByb2dyZXNzaXZlIG1vZGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBwcm9ncmVzc2l2ZU1vZGUgPSBNT0RFX0NPUFlfRlJPTV9NRVRBREFUQTsKLQotICAgIC8qKgotICAgICAqIFdoZXRoZXIgdGhlIEltYWdlV3JpdGVyIGNhbiB3cml0ZSBpbiBjb21wcmVzc2VkIG1vZGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGJvb2xlYW4gY2FuV3JpdGVDb21wcmVzc2VkID0gZmFsc2U7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29tcHJlc3Npb24gbW9kZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgaW50IGNvbXByZXNzaW9uTW9kZSA9IE1PREVfQ09QWV9GUk9NX01FVEFEQVRBOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbXByZXNzaW9uIHR5cGVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmdbXSBjb21wcmVzc2lvblR5cGVzID0gbnVsbDsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb21wcmVzc2lvbiB0eXBlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmcgY29tcHJlc3Npb25UeXBlID0gbnVsbDsKLQotICAgIC8qKgotICAgICAqIFRoZSBjb21wcmVzc2lvbiBxdWFsaXR5LgotICAgICAqLwotICAgIHByb3RlY3RlZCBmbG9hdCBjb21wcmVzc2lvblF1YWxpdHkgPSAxLjBmOwotCi0gICAgLyoqCi0gICAgICogVGhlIGxvY2FsZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgTG9jYWxlIGxvY2FsZSA9IG51bGw7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VXcml0ZVBhcmFtLgotICAgICAqLwotICAgIHByb3RlY3RlZCBJbWFnZVdyaXRlUGFyYW0oKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEltYWdlV3JpdGVQYXJhbSB3aXRoIHRoZSBzcGVjaWZpZWQgTG9jYWxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsb2NhbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBMb2NhbGUuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlV3JpdGVQYXJhbShMb2NhbGUgbG9jYWxlKSB7Ci0gICAgICAgIHRoaXMubG9jYWxlID0gbG9jYWxlOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbW9kZSBmb3Igd3JpdGluZyB0aGUgc3RyZWFtIGluIGEgcHJvZ3Jlc3NpdmUgc2VxdWVuY2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY3VycmVudCBwcm9ncmVzc2l2ZSBtb2RlLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0UHJvZ3Jlc3NpdmVNb2RlKCkgewotICAgICAgICBpZiAoY2FuV3JpdGVQcm9ncmVzc2l2ZSgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gcHJvZ3Jlc3NpdmVNb2RlOwotICAgICAgICB9Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigicHJvZ3Jlc3NpdmUgbW9kZSBpcyBub3Qgc3VwcG9ydGVkIik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIGltYWdlcyBjYW4gYmUgd3JpdHRlbiB1c2luZyBpbmNyZWFzaW5nIHF1YWxpdHkgcGFzc2VzIGJ5Ci0gICAgICogcHJvZ3Jlc3NpdmUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlIGlmIGltYWdlcyBjYW4gYmUgd3JpdHRlbiB1c2luZyBpbmNyZWFzaW5nIHF1YWxpdHkgcGFzc2VzIGJ5Ci0gICAgICogICAgICAgICBwcm9ncmVzc2l2ZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNhbldyaXRlUHJvZ3Jlc3NpdmUoKSB7Ci0gICAgICAgIHJldHVybiBjYW5Xcml0ZVByb2dyZXNzaXZlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHByb2dyZXNzaXZlIG1vZGUgd2hpY2ggZGVmaW5lcyB3aGV0aGVyIHRoZSBzdHJlYW0gY29udGFpbnMgYQotICAgICAqIHByb2dyZXNzaXZlIHNlcXVlbmNlIG9mIGluY3JlYXNpbmcgcXVhbGl0eSBkdXJpbmcgd3JpdGluZy4gVGhlCi0gICAgICogcHJvZ3Jlc3NpdmUgbW9kZSBzaG91bGQgYmUgb25lIG9mIHRoZSBmb2xsb3dpbmcgdmFsdWVzOiBNT0RFX0RJU0FCTEVELAotICAgICAqIE1PREVfREVGQVVMVCwgb3IgTU9ERV9DT1BZX0ZST01fTUVUQURBVEEuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1vZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgcHJvZ3Jlc3NpdmUgbW9kZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRQcm9ncmVzc2l2ZU1vZGUoaW50IG1vZGUpIHsKLSAgICAgICAgaWYgKGNhbldyaXRlUHJvZ3Jlc3NpdmUoKSkgewotICAgICAgICAgICAgaWYgKG1vZGUgPCBNT0RFX0RJU0FCTEVEIHx8IG1vZGUgPiBNT0RFX0NPUFlfRlJPTV9NRVRBREFUQSB8fCBtb2RlID09IE1PREVfRVhQTElDSVQpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJtb2RlIGlzIG5vdCBzdXBwb3J0ZWQiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHRoaXMucHJvZ3Jlc3NpdmVNb2RlID0gbW9kZTsKLSAgICAgICAgfQotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oInByb2dyZXNzaXZlIG1vZGUgaXMgbm90IHN1cHBvcnRlZCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgd3JpdGVyIGNhbiB1c2UgdGlsZXMgd2l0aCBub24gemVybyBncmlkIG9mZnNldHMgd2hpbGUKLSAgICAgKiB3cml0aW5nLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHdyaXRlciBjYW4gdXNlIHRpbGVzIHdpdGggbm9uIHplcm8gZ3JpZCBvZmZzZXRzCi0gICAgICogICAgICAgICB3aGlsZSB3cml0aW5nLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY2FuT2Zmc2V0VGlsZXMoKSB7Ci0gICAgICAgIHJldHVybiBjYW5PZmZzZXRUaWxlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhpcyB3cml0ZXIgY2FuIHdyaXRlIGltYWdlcyB3aXRoIGNvbXByZXNzaW9uLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyB3cml0ZXIgY2FuIHdyaXRlIGltYWdlcyB3aXRoIGNvbXByZXNzaW9uLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNhbldyaXRlQ29tcHJlc3NlZCgpIHsKLSAgICAgICAgcmV0dXJuIGNhbldyaXRlQ29tcHJlc3NlZDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIHdyaXRlciBjYW4gd3JpdGUgdGlsZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgd3JpdGVyIGNhbiB3cml0ZSB0aWxlcywgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNhbldyaXRlVGlsZXMoKSB7Ci0gICAgICAgIHJldHVybiBjYW5Xcml0ZVRpbGVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrIHdyaXRlIGNvbXByZXNzZWQuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCB2b2lkIGNoZWNrV3JpdGVDb21wcmVzc2VkKCkgewotICAgICAgICBpZiAoIWNhbldyaXRlQ29tcHJlc3NlZCgpKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIkNvbXByZXNzaW9uIG5vdCBzdXBwb3J0ZWQuIik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVjayBjb21wcmVzc2lvbiBtb2RlLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgdm9pZCBjaGVja0NvbXByZXNzaW9uTW9kZSgpIHsKLSAgICAgICAgaWYgKGdldENvbXByZXNzaW9uTW9kZSgpICE9IE1PREVfRVhQTElDSVQpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oIkNvbXByZXNzaW9uIG1vZGUgbm90IE1PREVfRVhQTElDSVQhIik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVjayBjb21wcmVzc2lvbiB0eXBlLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgdm9pZCBjaGVja0NvbXByZXNzaW9uVHlwZSgpIHsKLSAgICAgICAgaWYgKGdldENvbXByZXNzaW9uVHlwZXMoKSAhPSBudWxsICYmIGdldENvbXByZXNzaW9uVHlwZSgpID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oIk5vIGNvbXByZXNzaW9uIHR5cGUgc2V0ISIpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29tcHJlc3Npb24gbW9kZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjb21wcmVzc2lvbiBtb2RlIGlmIGl0J3Mgc3VwcG9ydGVkLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0Q29tcHJlc3Npb25Nb2RlKCkgewotICAgICAgICBjaGVja1dyaXRlQ29tcHJlc3NlZCgpOwotICAgICAgICByZXR1cm4gY29tcHJlc3Npb25Nb2RlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFuIGFycmF5IG9mIHN1cHBvcnRlZCBjb21wcmVzc2lvbiB0eXBlcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhbiBhcnJheSBvZiBzdXBwb3J0ZWQgY29tcHJlc3Npb24gdHlwZXMuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZ1tdIGdldENvbXByZXNzaW9uVHlwZXMoKSB7Ci0gICAgICAgIGNoZWNrV3JpdGVDb21wcmVzc2VkKCk7Ci0gICAgICAgIGlmIChjb21wcmVzc2lvblR5cGVzICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBjb21wcmVzc2lvblR5cGVzLmNsb25lKCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY3VycmVudCBjb21wcmVzc2lvbiB0eXBlLCBvciByZXR1cm5zIG51bGwuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY3VycmVudCBjb21wcmVzc2lvbiB0eXBlLCBvciByZXR1cm5zIG51bGwgaWYgaXQgaXMgbm90IHNldC4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldENvbXByZXNzaW9uVHlwZSgpIHsKLSAgICAgICAgY2hlY2tXcml0ZUNvbXByZXNzZWQoKTsKLSAgICAgICAgY2hlY2tDb21wcmVzc2lvbk1vZGUoKTsKLSAgICAgICAgcmV0dXJuIGNvbXByZXNzaW9uVHlwZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGEgYml0IHJhdGUgd2hpY2ggcmVwcmVzZW50cyBhbiBlc3RpbWF0ZSBvZiB0aGUgbnVtYmVyIG9mIGJpdHMgb2YKLSAgICAgKiBvdXRwdXQgZGF0YSBmb3IgZWFjaCBiaXQgb2YgaW5wdXQgaW1hZ2UgZGF0YSB3aXRoIHRoZSBzcGVjaWZpZWQgcXVhbGl0eS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcXVhbGl0eQotICAgICAqICAgICAgICAgICAgdGhlIHF1YWxpdHkuCi0gICAgICogQHJldHVybiBhbiBlc3RpbWF0ZSBvZiB0aGUgYml0IHJhdGUsIG9yIC0xLjBGIGlmIHRoZXJlIGlzIG5vIGVzdGltYXRlLgotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRCaXRSYXRlKGZsb2F0IHF1YWxpdHkpIHsKLSAgICAgICAgY2hlY2tXcml0ZUNvbXByZXNzZWQoKTsKLSAgICAgICAgY2hlY2tDb21wcmVzc2lvbk1vZGUoKTsKLSAgICAgICAgY2hlY2tDb21wcmVzc2lvblR5cGUoKTsKLSAgICAgICAgaWYgKHF1YWxpdHkgPCAwIHx8IHF1YWxpdHkgPiAxKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJRdWFsaXR5IG91dC1vZi1ib3VuZHMhIik7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIC0xLjBmOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNvbXByZXNzaW9uIHF1YWxpdHkuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY29tcHJlc3Npb24gcXVhbGl0eS4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0Q29tcHJlc3Npb25RdWFsaXR5KCkgewotICAgICAgICBjaGVja1dyaXRlQ29tcHJlc3NlZCgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uTW9kZSgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uVHlwZSgpOwotICAgICAgICByZXR1cm4gY29tcHJlc3Npb25RdWFsaXR5OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGFycmF5IG9mIGNvbXByZXNzaW9uIHF1YWxpdHkgZGVzY3JpcHRpb25zLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0cmluZyBhcnJheSBvZiBjb21wcmVzc2lvbiBxdWFsaXR5IGRlc2NyaXB0aW9ucy4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0Q29tcHJlc3Npb25RdWFsaXR5RGVzY3JpcHRpb25zKCkgewotICAgICAgICBjaGVja1dyaXRlQ29tcHJlc3NlZCgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uTW9kZSgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uVHlwZSgpOwotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIGZsb2F0cyB3aGljaCBkZXNjcmliZXMgY29tcHJlc3Npb24gcXVhbGl0eSBsZXZlbHMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgY29tcHJlc3Npb24gcXVhbGl0eSB2YWx1ZXMuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0W10gZ2V0Q29tcHJlc3Npb25RdWFsaXR5VmFsdWVzKCkgewotICAgICAgICBjaGVja1dyaXRlQ29tcHJlc3NlZCgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uTW9kZSgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uVHlwZSgpOwotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBsb2NhbGUgb2YgdGhpcyBJbWFnZVdyaXRlUGFyYW0uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbG9jYWxlIG9mIHRoaXMgSW1hZ2VXcml0ZVBhcmFtLgotICAgICAqLwotICAgIHB1YmxpYyBMb2NhbGUgZ2V0TG9jYWxlKCkgewotICAgICAgICByZXR1cm4gbG9jYWxlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGN1cnJlbnQgY29tcHJlc3Npb24gdHlwZSB1c2luZyB0aGUgY3VycmVudCBMb2NhbGUuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY3VycmVudCBjb21wcmVzc2lvbiB0eXBlIHVzaW5nIHRoZSBjdXJyZW50IExvY2FsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldExvY2FsaXplZENvbXByZXNzaW9uVHlwZU5hbWUoKSB7Ci0gICAgICAgIGNoZWNrV3JpdGVDb21wcmVzc2VkKCk7Ci0gICAgICAgIGNoZWNrQ29tcHJlc3Npb25Nb2RlKCk7Ci0KLSAgICAgICAgU3RyaW5nIGNvbXByZXNzaW9uVHlwZSA9IGdldENvbXByZXNzaW9uVHlwZSgpOwotICAgICAgICBpZiAoY29tcHJlc3Npb25UeXBlID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oIk5vIGNvbXByZXNzaW9uIHR5cGUgc2V0ISIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBjb21wcmVzc2lvblR5cGU7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVjayB0aWxpbmcuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCB2b2lkIGNoZWNrVGlsaW5nKCkgewotICAgICAgICBpZiAoIWNhbldyaXRlVGlsZXMoKSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJUaWxpbmcgbm90IHN1cHBvcnRlZCEiKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrIHRpbGluZyBtb2RlLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgdm9pZCBjaGVja1RpbGluZ01vZGUoKSB7Ci0gICAgICAgIGlmIChnZXRUaWxpbmdNb2RlKCkgIT0gTU9ERV9FWFBMSUNJVCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxTdGF0ZUV4Y2VwdGlvbigiVGlsaW5nIG1vZGUgbm90IE1PREVfRVhQTElDSVQhIik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVjayB0aWxpbmcgcGFyYW1zLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgdm9pZCBjaGVja1RpbGluZ1BhcmFtcygpIHsKLSAgICAgICAgaWYgKCF0aWxpbmdTZXQpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oIlRpbGluZyBwYXJhbWV0ZXJzIG5vdCBzZXQhIik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0aWxpbmcgbW9kZSBpZiB0aWxpbmcgaXMgc3VwcG9ydGVkLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHRpbGluZyBtb2RlIGlmIHRpbGluZyBpcyBzdXBwb3J0ZWQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRUaWxpbmdNb2RlKCkgewotICAgICAgICBjaGVja1RpbGluZygpOwotICAgICAgICByZXR1cm4gdGlsaW5nTW9kZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIERpbWVuc2lvbnMgZ2l2aW5nIHRoZSBzaXplcyBvZiB0aGUgdGlsZXMgYXMgdGhleSBhcmUKLSAgICAgKiBlbmNvZGVkIGluIHRoZSBvdXRwdXQgZmlsZSBvciBzdHJlYW0uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgcHJlZmVycmVkIHRpbGUgc2l6ZXMuCi0gICAgICovCi0gICAgcHVibGljIERpbWVuc2lvbltdIGdldFByZWZlcnJlZFRpbGVTaXplcygpIHsKLSAgICAgICAgY2hlY2tUaWxpbmcoKTsKLSAgICAgICAgaWYgKHByZWZlcnJlZFRpbGVTaXplcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotCi0gICAgICAgIERpbWVuc2lvbltdIHJldHZhbCA9IG5ldyBEaW1lbnNpb25bcHJlZmVycmVkVGlsZVNpemVzLmxlbmd0aF07Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcHJlZmVycmVkVGlsZVNpemVzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICByZXR2YWxbaV0gPSBuZXcgRGltZW5zaW9uKHJldHZhbFtpXSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJldHZhbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0aWxlIGdyaWQgWCBvZmZzZXQgZm9yIGVuY29kaW5nLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHRpbGUgZ3JpZCBYIG9mZnNldCBmb3IgZW5jb2RpbmcuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRUaWxlR3JpZFhPZmZzZXQoKSB7Ci0gICAgICAgIGNoZWNrVGlsaW5nKCk7Ci0gICAgICAgIGNoZWNrVGlsaW5nTW9kZSgpOwotICAgICAgICBjaGVja1RpbGluZ1BhcmFtcygpOwotICAgICAgICByZXR1cm4gdGlsZUdyaWRYT2Zmc2V0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHRpbGUgZ3JpZCBZIG9mZnNldCBmb3IgZW5jb2RpbmcuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdGlsZSBncmlkIFkgb2Zmc2V0IGZvciBlbmNvZGluZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFRpbGVHcmlkWU9mZnNldCgpIHsKLSAgICAgICAgY2hlY2tUaWxpbmcoKTsKLSAgICAgICAgY2hlY2tUaWxpbmdNb2RlKCk7Ci0gICAgICAgIGNoZWNrVGlsaW5nUGFyYW1zKCk7Ci0gICAgICAgIHJldHVybiB0aWxlR3JpZFlPZmZzZXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgdGlsZSBoZWlnaHQgaW4gYW4gaW1hZ2UgYXMgaXQgaXMgd3JpdHRlbiB0byB0aGUgb3V0cHV0IHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB0aWxlIGhlaWdodCBpbiBhbiBpbWFnZSBhcyBpdCBpcyB3cml0dGVuIHRvIHRoZSBvdXRwdXQKLSAgICAgKiAgICAgICAgIHN0cmVhbS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFRpbGVIZWlnaHQoKSB7Ci0gICAgICAgIGNoZWNrVGlsaW5nKCk7Ci0gICAgICAgIGNoZWNrVGlsaW5nTW9kZSgpOwotICAgICAgICBjaGVja1RpbGluZ1BhcmFtcygpOwotICAgICAgICByZXR1cm4gdGlsZUhlaWdodDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB0aWxlIHdpZHRoIGluIGFuIGltYWdlIGFzIGl0IGlzIHdyaXR0ZW4gdG8gdGhlIG91dHB1dCBzdHJlYW0uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdGlsZSB3aWR0aCBpbiBhbiBpbWFnZSBhcyBpdCBpcyB3cml0dGVuIHRvIHRoZSBvdXRwdXQgc3RyZWFtLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0VGlsZVdpZHRoKCkgewotICAgICAgICBjaGVja1RpbGluZygpOwotICAgICAgICBjaGVja1RpbGluZ01vZGUoKTsKLSAgICAgICAgY2hlY2tUaWxpbmdQYXJhbXMoKTsKLSAgICAgICAgcmV0dXJuIHRpbGVXaWR0aDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIGN1cnJlbnQgY29tcHJlc3Npb24gdHlwZSBoYXMgbG9zc2xlc3MgY29tcHJlc3Npb24gb3Igbm90LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIGN1cnJlbnQgY29tcHJlc3Npb24gdHlwZSBoYXMgbG9zc2xlc3MgY29tcHJlc3Npb24sCi0gICAgICogICAgICAgICBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb21wcmVzc2lvbkxvc3NsZXNzKCkgewotICAgICAgICBjaGVja1dyaXRlQ29tcHJlc3NlZCgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uTW9kZSgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uVHlwZSgpOwotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIGN1cnJlbnQgY29tcHJlc3Npb24gdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB1bnNldENvbXByZXNzaW9uKCkgewotICAgICAgICBjaGVja1dyaXRlQ29tcHJlc3NlZCgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uTW9kZSgpOwotICAgICAgICBjb21wcmVzc2lvblR5cGUgPSBudWxsOwotICAgICAgICBjb21wcmVzc2lvblF1YWxpdHkgPSAxOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGNvbXByZXNzaW9uIG1vZGUgdG8gdGhlIHNwZWNpZmllZCB2YWx1ZS4gVGhlIHNwZWNpZmllZCBtb2RlIGNhbgotICAgICAqIGJlIG9uZSBvZiB0aGUgcHJlZGVmaW5lZCBjb25zdGFudHM6IE1PREVfREVGQVVMVCwgTU9ERV9ESVNBQkxFRCwKLSAgICAgKiBNT0RFX0VYUExJQ0lULCBvciBNT0RFX0NPUFlfRlJPTV9NRVRBREFUQS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbW9kZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBjb21wcmVzc2lvbiBtb2RlIHRvIGJlIHNldC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRDb21wcmVzc2lvbk1vZGUoaW50IG1vZGUpIHsKLSAgICAgICAgY2hlY2tXcml0ZUNvbXByZXNzZWQoKTsKLSAgICAgICAgc3dpdGNoIChtb2RlKSB7Ci0gICAgICAgICAgICBjYXNlIE1PREVfRVhQTElDSVQ6IHsKLSAgICAgICAgICAgICAgICBjb21wcmVzc2lvbk1vZGUgPSBtb2RlOwotICAgICAgICAgICAgICAgIHVuc2V0Q29tcHJlc3Npb24oKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNhc2UgTU9ERV9DT1BZX0ZST01fTUVUQURBVEE6Ci0gICAgICAgICAgICBjYXNlIE1PREVfRElTQUJMRUQ6Ci0gICAgICAgICAgICBjYXNlIE1PREVfREVGQVVMVDogewotICAgICAgICAgICAgICAgIGNvbXByZXNzaW9uTW9kZSA9IG1vZGU7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkZWZhdWx0OiB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiSWxsZWdhbCB2YWx1ZSBmb3IgbW9kZSEiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGNvbXByZXNzaW9uIHF1YWxpdHkuIFRoZSB2YWx1ZSBzaG91bGQgYmUgYmV0d2VlbiAwIGFuZCAxLgotICAgICAqIAotICAgICAqIEBwYXJhbSBxdWFsaXR5Ci0gICAgICogICAgICAgICAgICB0aGUgbmV3IGNvbXByZXNzaW9uIHF1YWxpdHksIGZsb2F0IHZhbHVlIGJldHdlZW4gMCBhbmQgMS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRDb21wcmVzc2lvblF1YWxpdHkoZmxvYXQgcXVhbGl0eSkgewotICAgICAgICBjaGVja1dyaXRlQ29tcHJlc3NlZCgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uTW9kZSgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uVHlwZSgpOwotICAgICAgICBpZiAocXVhbGl0eSA8IDAgfHwgcXVhbGl0eSA+IDEpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIlF1YWxpdHkgb3V0LW9mLWJvdW5kcyEiKTsKLSAgICAgICAgfQotICAgICAgICBjb21wcmVzc2lvblF1YWxpdHkgPSBxdWFsaXR5OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGNvbXByZXNzaW9uIHR5cGUuIFRoZSBzcGVjaWZpZWQgc3RyaW5nIHNob3VsZCBiZSBvbmUgb2YgdGhlCi0gICAgICogdmFsdWVzIHJldHVybmVkIGJ5IGdldENvbXByZXNzaW9uVHlwZXMgbWV0aG9kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjb21wcmVzc2lvblR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgY29tcHJlc3Npb24gdHlwZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRDb21wcmVzc2lvblR5cGUoU3RyaW5nIGNvbXByZXNzaW9uVHlwZSkgewotICAgICAgICBjaGVja1dyaXRlQ29tcHJlc3NlZCgpOwotICAgICAgICBjaGVja0NvbXByZXNzaW9uTW9kZSgpOwotCi0gICAgICAgIGlmIChjb21wcmVzc2lvblR5cGUgPT0gbnVsbCkgeyAvLyBEb24ndCBjaGVjayBhbnl0aGluZwotICAgICAgICAgICAgdGhpcy5jb21wcmVzc2lvblR5cGUgPSBudWxsOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgU3RyaW5nW10gY29tcHJlc3Npb25UeXBlcyA9IGdldENvbXByZXNzaW9uVHlwZXMoKTsKLSAgICAgICAgICAgIGlmIChjb21wcmVzc2lvblR5cGVzID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vIHNldHRhYmxlIGNvbXByZXNzaW9uIHR5cGVzIik7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY29tcHJlc3Npb25UeXBlcy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgICAgIGlmIChjb21wcmVzc2lvblR5cGVzW2ldLmVxdWFscyhjb21wcmVzc2lvblR5cGUpKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMuY29tcHJlc3Npb25UeXBlID0gY29tcHJlc3Npb25UeXBlOwotICAgICAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBDb21wcmVzc2lvbiB0eXBlIGlzIG5vdCBpbiB0aGUgbGlzdC4KLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIlVua25vd24gY29tcHJlc3Npb24gdHlwZSEiKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGluc3RydWN0aW9uIHRoYXQgdGlsaW5nIHNob3VsZCBiZSBwZXJmb3JtZWQgZm9yIHRoZSBpbWFnZSBpbiB0aGUKLSAgICAgKiBvdXRwdXQgc3RyZWFtIHdpdGggdGhlIHNwZWNpZmllZCBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0aWxlV2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aWxlJ3Mgd2lkdGguCi0gICAgICogQHBhcmFtIHRpbGVIZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aWxlJ3MgaGVpZ2h0LgotICAgICAqIEBwYXJhbSB0aWxlR3JpZFhPZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aWxlIGdyaWQncyB4IG9mZnNldC4KLSAgICAgKiBAcGFyYW0gdGlsZUdyaWRZT2Zmc2V0Ci0gICAgICogICAgICAgICAgICB0aGUgdGlsZSBncmlkJ3MgeSBvZmZzZXQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0VGlsaW5nKGludCB0aWxlV2lkdGgsIGludCB0aWxlSGVpZ2h0LCBpbnQgdGlsZUdyaWRYT2Zmc2V0LCBpbnQgdGlsZUdyaWRZT2Zmc2V0KSB7Ci0gICAgICAgIGNoZWNrVGlsaW5nKCk7Ci0gICAgICAgIGNoZWNrVGlsaW5nTW9kZSgpOwotCi0gICAgICAgIGlmICghY2FuT2Zmc2V0VGlsZXMoKSAmJiAodGlsZUdyaWRYT2Zmc2V0ICE9IDAgfHwgdGlsZUdyaWRZT2Zmc2V0ICE9IDApKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIkNhbid0IG9mZnNldCB0aWxlcyEiKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICh0aWxlV2lkdGggPD0gMCB8fCB0aWxlSGVpZ2h0IDw9IDApIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInRpbGUgZGltZW5zaW9ucyBhcmUgbm9uLXBvc2l0aXZlISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgRGltZW5zaW9uIHByZWZlcnJlZFRpbGVTaXplc1tdID0gZ2V0UHJlZmVycmVkVGlsZVNpemVzKCk7Ci0gICAgICAgIGlmIChwcmVmZXJyZWRUaWxlU2l6ZXMgIT0gbnVsbCkgewotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBwcmVmZXJyZWRUaWxlU2l6ZXMubGVuZ3RoOyBpICs9IDIpIHsKLSAgICAgICAgICAgICAgICBEaW1lbnNpb24gbWluU2l6ZSA9IHByZWZlcnJlZFRpbGVTaXplc1tpXTsKLSAgICAgICAgICAgICAgICBEaW1lbnNpb24gbWF4U2l6ZSA9IHByZWZlcnJlZFRpbGVTaXplc1tpICsgMV07Ci0gICAgICAgICAgICAgICAgaWYgKHRpbGVXaWR0aCA8IG1pblNpemUud2lkdGggfHwgdGlsZVdpZHRoID4gbWF4U2l6ZS53aWR0aAotICAgICAgICAgICAgICAgICAgICAgICAgfHwgdGlsZUhlaWdodCA8IG1pblNpemUuaGVpZ2h0IHx8IHRpbGVIZWlnaHQgPiBtYXhTaXplLmhlaWdodCkgewotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJJbGxlZ2FsIHRpbGUgc2l6ZSEiKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICB0aWxpbmdTZXQgPSB0cnVlOwotICAgICAgICB0aGlzLnRpbGVXaWR0aCA9IHRpbGVXaWR0aDsKLSAgICAgICAgdGhpcy50aWxlSGVpZ2h0ID0gdGlsZUhlaWdodDsKLSAgICAgICAgdGhpcy50aWxlR3JpZFhPZmZzZXQgPSB0aWxlR3JpZFhPZmZzZXQ7Ci0gICAgICAgIHRoaXMudGlsZUdyaWRZT2Zmc2V0ID0gdGlsZUdyaWRZT2Zmc2V0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENsZWFycyBhbGwgdGlsaW5nIHNldHRpbmdzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHVuc2V0VGlsaW5nKCkgewotICAgICAgICBjaGVja1RpbGluZygpOwotICAgICAgICBjaGVja1RpbGluZ01vZGUoKTsKLQotICAgICAgICB0aWxpbmdTZXQgPSBmYWxzZTsKLSAgICAgICAgdGlsZVdpZHRoID0gMDsKLSAgICAgICAgdGlsZUhlaWdodCA9IDA7Ci0gICAgICAgIHRpbGVHcmlkWE9mZnNldCA9IDA7Ci0gICAgICAgIHRpbGVHcmlkWU9mZnNldCA9IDA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgdGlsaW5nIG1vZGUuIFRoZSBzcGVjaWZpZWQgbW9kZSBzaG91bGQgYmUgb25lIG9mIHRoZSBmb2xsb3dpbmcKLSAgICAgKiB2YWx1ZXM6IE1PREVfRElTQUJMRUQsIE1PREVfREVGQVVMVCwgTU9ERV9FWFBMSUNJVCwgb3IKLSAgICAgKiBNT0RFX0NPUFlfRlJPTV9NRVRBREFUQS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbW9kZQotICAgICAqICAgICAgICAgICAgdGhlIG5ldyB0aWxpbmcgbW9kZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRUaWxpbmdNb2RlKGludCBtb2RlKSB7Ci0gICAgICAgIGNoZWNrVGlsaW5nKCk7Ci0KLSAgICAgICAgc3dpdGNoIChtb2RlKSB7Ci0gICAgICAgICAgICBjYXNlIE1PREVfRVhQTElDSVQ6IHsKLSAgICAgICAgICAgICAgICB0aWxpbmdNb2RlID0gbW9kZTsKLSAgICAgICAgICAgICAgICB1bnNldFRpbGluZygpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY2FzZSBNT0RFX0NPUFlfRlJPTV9NRVRBREFUQToKLSAgICAgICAgICAgIGNhc2UgTU9ERV9ESVNBQkxFRDoKLSAgICAgICAgICAgIGNhc2UgTU9ERV9ERUZBVUxUOiB7Ci0gICAgICAgICAgICAgICAgdGlsaW5nTW9kZSA9IG1vZGU7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkZWZhdWx0OiB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiSWxsZWdhbCB2YWx1ZSBmb3IgbW9kZSEiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL0ltYWdlV3JpdGVyLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9JbWFnZVdyaXRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4Njg3OWUwLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL0ltYWdlV3JpdGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMDAxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW87Ci0KLWltcG9ydCBqYXZhLmF3dC5EaW1lbnNpb247Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmFzdGVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJlbmRlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLnNlY3VyaXR5LkFjY2Vzc0NvbnRyb2xsZXI7Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5Qcml2aWxlZ2VkQWN0aW9uOwotaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLkxpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLWltcG9ydCBqYXZhLnV0aWwuTWlzc2luZ1Jlc291cmNlRXhjZXB0aW9uOwotaW1wb3J0IGphdmEudXRpbC5SZXNvdXJjZUJ1bmRsZTsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uZXZlbnQuSUlPV3JpdGVQcm9ncmVzc0xpc3RlbmVyOwotaW1wb3J0IGphdmF4LmltYWdlaW8uZXZlbnQuSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXI7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5tZXRhZGF0YS5JSU9NZXRhZGF0YTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLnNwaS5JbWFnZVdyaXRlclNwaTsKLQotLyoqCi0gKiBUaGUgSW1hZ2VXcml0ZXIgY2xhc3MgaXMgYW4gYWJzdHJhY3QgY2xhc3MgZm9yIGVuY29kaW5nIGltYWdlcy4gSW1hZ2VXcml0ZXIKLSAqIG9iamVjdHMgYXJlIGluc3RhbnRpYXRlZCBieSB0aGUgc2VydmljZSBwcm92aWRlciBpbnRlcmZhY2UsIEltYWdlV3JpdGVyU3BpCi0gKiBjbGFzcywgZm9yIHRoZSBzcGVjaWZpYyBmb3JtYXQuIEltYWdlV3JpdGVyU3BpIGNsYXNzIHNob3VsZCBiZSByZWdpc3RlcmVkCi0gKiB3aXRoIHRoZSBJSU9SZWdpc3RyeSwgd2hpY2ggdXNlcyB0aGVtIGZvciBmb3JtYXQgcmVjb2duaXRpb24gYW5kIHByZXNlbnRhdGlvbgotICogb2YgYXZhaWxhYmxlIGZvcm1hdCByZWFkZXJzIGFuZCB3cml0ZXJzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEltYWdlV3JpdGVyIGltcGxlbWVudHMgSW1hZ2VUcmFuc2NvZGVyIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBhdmFpbGFibGUgbG9jYWxlcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgTG9jYWxlW10gYXZhaWxhYmxlTG9jYWxlczsKLQotICAgIC8qKgotICAgICAqIFRoZSBsb2NhbGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIExvY2FsZSBsb2NhbGU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgb3JpZ2luYXRpbmcgcHJvdmlkZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEltYWdlV3JpdGVyU3BpIG9yaWdpbmF0aW5nUHJvdmlkZXI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgb3V0cHV0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBPYmplY3Qgb3V0cHV0OwotCi0gICAgLyoqCi0gICAgICogVGhlIHByb2dyZXNzIGxpc3RlbmVycy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgTGlzdDxJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXI+IHByb2dyZXNzTGlzdGVuZXJzOwotCi0gICAgLyoqCi0gICAgICogVGhlIHdhcm5pbmcgbGlzdGVuZXJzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBMaXN0PElJT1dyaXRlV2FybmluZ0xpc3RlbmVyPiB3YXJuaW5nTGlzdGVuZXJzOwotCi0gICAgLyoqCi0gICAgICogVGhlIHdhcm5pbmcgbG9jYWxlcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgTGlzdDxMb2NhbGU+IHdhcm5pbmdMb2NhbGVzOwotCi0gICAgLy8gSW5kaWNhdGVzIHRoYXQgYWJvcnQgb3BlcmF0aW9uIGlzIHJlcXVlc3RlZAotICAgIC8vIEFib3J0IG1lY2hhbmlzbSBzaG91bGQgYmUgdGhyZWFkLXNhZmUKLSAgICAvKiogVGhlIGFib3J0ZWQuICovCi0gICAgcHJpdmF0ZSBib29sZWFuIGFib3J0ZWQ7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VXcml0ZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9yaWdpbmF0aW5nUHJvdmlkZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVdyaXRlclNwaSB3aGljaCBpbnN0YW50aWF0ZXMgdGhpcyBJbWFnZVdyaXRlci4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgSW1hZ2VXcml0ZXIoSW1hZ2VXcml0ZXJTcGkgb3JpZ2luYXRpbmdQcm92aWRlcikgewotICAgICAgICB0aGlzLm9yaWdpbmF0aW5nUHJvdmlkZXIgPSBvcmlnaW5hdGluZ1Byb3ZpZGVyOwotICAgIH0KLQotICAgIHB1YmxpYyBhYnN0cmFjdCBJSU9NZXRhZGF0YSBjb252ZXJ0U3RyZWFtTWV0YWRhdGEoSUlPTWV0YWRhdGEgaWlvTWV0YWRhdGEsCi0gICAgICAgICAgICBJbWFnZVdyaXRlUGFyYW0gaW1hZ2VXcml0ZVBhcmFtKTsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCBJSU9NZXRhZGF0YSBjb252ZXJ0SW1hZ2VNZXRhZGF0YShJSU9NZXRhZGF0YSBpaW9NZXRhZGF0YSwKLSAgICAgICAgICAgIEltYWdlVHlwZVNwZWNpZmllciBpbWFnZVR5cGVTcGVjaWZpZXIsIEltYWdlV3JpdGVQYXJhbSBpbWFnZVdyaXRlUGFyYW0pOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgSW1hZ2VXcml0ZXJTcGkgd2hpY2ggaW5zdGFudGlhdGVkIHRoaXMgSW1hZ2VXcml0ZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VXcml0ZXJTcGkuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlV3JpdGVyU3BpIGdldE9yaWdpbmF0aW5nUHJvdmlkZXIoKSB7Ci0gICAgICAgIHJldHVybiBvcmlnaW5hdGluZ1Byb3ZpZGVyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyB0aGUgc3RhcnQgb2YgYW4gaW1hZ2UgcmVhZCBieSBjYWxsaW5nIHRoZWlyIGltYWdlU3RhcnRlZCBtZXRob2QKLSAgICAgKiBvZiByZWdpc3RlcmVkIElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBpbmRleC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBwcm9jZXNzSW1hZ2VTdGFydGVkKGludCBpbWFnZUluZGV4KSB7Ci0gICAgICAgIGlmIChudWxsICE9IHByb2dyZXNzTGlzdGVuZXJzKSB7Ci0gICAgICAgICAgICBmb3IgKElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lciBsaXN0ZW5lciA6IHByb2dyZXNzTGlzdGVuZXJzKSB7Ci0gICAgICAgICAgICAgICAgbGlzdGVuZXIuaW1hZ2VTdGFydGVkKHRoaXMsIGltYWdlSW5kZXgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIHRoZSBjdXJyZW50IHBlcmNlbnRhZ2Ugb2YgaW1hZ2UgY29tcGxldGlvbiBieSBjYWxsaW5nCi0gICAgICogaW1hZ2VQcm9ncmVzcyBtZXRob2Qgb2YgcmVnaXN0ZXJlZCBJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBlcmNlbnRhZ2VEb25lCi0gICAgICogICAgICAgICAgICB0aGUgcGVyY2VudGFnZSBkb25lLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NJbWFnZVByb2dyZXNzKGZsb2F0IHBlcmNlbnRhZ2VEb25lKSB7Ci0gICAgICAgIGlmIChudWxsICE9IHByb2dyZXNzTGlzdGVuZXJzKSB7Ci0gICAgICAgICAgICBmb3IgKElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lciBsaXN0ZW5lciA6IHByb2dyZXNzTGlzdGVuZXJzKSB7Ci0gICAgICAgICAgICAgICAgbGlzdGVuZXIuaW1hZ2VQcm9ncmVzcyh0aGlzLCBwZXJjZW50YWdlRG9uZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgaW1hZ2UgY29tcGxldGlvbiBieSBjYWxsaW5nIGltYWdlQ29tcGxldGUgbWV0aG9kIG9mIHJlZ2lzdGVyZWQKLSAgICAgKiBJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXJzLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NJbWFnZUNvbXBsZXRlKCkgewotICAgICAgICBpZiAobnVsbCAhPSBwcm9ncmVzc0xpc3RlbmVycykgewotICAgICAgICAgICAgZm9yIChJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIgbGlzdGVuZXIgOiBwcm9ncmVzc0xpc3RlbmVycykgewotICAgICAgICAgICAgICAgIGxpc3RlbmVyLmltYWdlQ29tcGxldGUodGhpcyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgYSB3YXJuaW5nIG1lc3NhZ2UgYnkgY2FsbGluZyB3YXJuaW5nT2NjdXJyZWQgbWV0aG9kIG9mCi0gICAgICogcmVnaXN0ZXJlZCBJSU9Xcml0ZVdhcm5pbmdMaXN0ZW5lcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBpbmRleC4KLSAgICAgKiBAcGFyYW0gd2FybmluZwotICAgICAqICAgICAgICAgICAgdGhlIHdhcm5pbmcuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgcHJvY2Vzc1dhcm5pbmdPY2N1cnJlZChpbnQgaW1hZ2VJbmRleCwgU3RyaW5nIHdhcm5pbmcpIHsKLSAgICAgICAgaWYgKG51bGwgPT0gd2FybmluZykgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJ3YXJuaW5nIG1lc3NhZ2Ugc2hvdWxkIG5vdCBiZSBOVUxMIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG51bGwgIT0gd2FybmluZ0xpc3RlbmVycykgewotICAgICAgICAgICAgZm9yIChJSU9Xcml0ZVdhcm5pbmdMaXN0ZW5lciBsaXN0ZW5lciA6IHdhcm5pbmdMaXN0ZW5lcnMpIHsKLSAgICAgICAgICAgICAgICBsaXN0ZW5lci53YXJuaW5nT2NjdXJyZWQodGhpcywgaW1hZ2VJbmRleCwgd2FybmluZyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcm9jZXNzZXMgYSB3YXJuaW5nIG1lc3NhZ2UgYnkgY2FsbGluZyB3YXJuaW5nT2NjdXJyZWQgbWV0aG9kIG9mCi0gICAgICogcmVnaXN0ZXJlZCBJSU9Xcml0ZVdhcm5pbmdMaXN0ZW5lcnMgd2l0aCBzdHJpbmcgZnJvbSBSZXNvdXJjZUJ1bmRsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGluZGV4LgotICAgICAqIEBwYXJhbSBidW5kbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYW1lIG9mIFJlc291cmNlQnVuZGxlLgotICAgICAqIEBwYXJhbSBrZXkKLSAgICAgKiAgICAgICAgICAgIHRoZSBrZXl3b3JkLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NXYXJuaW5nT2NjdXJyZWQoaW50IGltYWdlSW5kZXgsIFN0cmluZyBidW5kbGUsIFN0cmluZyBrZXkpIHsKLSAgICAgICAgaWYgKHdhcm5pbmdMaXN0ZW5lcnMgIT0gbnVsbCkgeyAvLyBEb24ndCBjaGVjayB0aGUgcGFyYW1ldGVycwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGJ1bmRsZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJiYXNlTmFtZSA9PSBudWxsISIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChrZXkgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigia2V5d29yZCA9PSBudWxsISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gR2V0IHRoZSBjb250ZXh0IGNsYXNzIGxvYWRlciBhbmQgdHJ5IHRvIGxvY2F0ZSB0aGUgYnVuZGxlIHdpdGggaXQKLSAgICAgICAgLy8gZmlyc3QKLSAgICAgICAgQ2xhc3NMb2FkZXIgY29udGV4dENsYXNzbG9hZGVyID0gQWNjZXNzQ29udHJvbGxlcgotICAgICAgICAgICAgICAgIC5kb1ByaXZpbGVnZWQobmV3IFByaXZpbGVnZWRBY3Rpb248Q2xhc3NMb2FkZXI+KCkgewotICAgICAgICAgICAgICAgICAgICBwdWJsaWMgQ2xhc3NMb2FkZXIgcnVuKCkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIFRocmVhZC5jdXJyZW50VGhyZWFkKCkuZ2V0Q29udGV4dENsYXNzTG9hZGVyKCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9KTsKLQotICAgICAgICAvLyBJdGVyYXRlIHRocm91Z2ggYm90aCBsaXN0ZW5lcnMgYW5kIGxvY2FsZXMKLSAgICAgICAgaW50IG4gPSB3YXJuaW5nTGlzdGVuZXJzLnNpemUoKTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBuOyBpKyspIHsKLSAgICAgICAgICAgIElJT1dyaXRlV2FybmluZ0xpc3RlbmVyIGxpc3RlbmVyID0gd2FybmluZ0xpc3RlbmVycy5nZXQoaSk7Ci0gICAgICAgICAgICBMb2NhbGUgbG9jYWxlID0gd2FybmluZ0xvY2FsZXMuZ2V0KGkpOwotCi0gICAgICAgICAgICAvLyBOb3cgdHJ5IHRvIGdldCB0aGUgcmVzb3VyY2UgYnVuZGxlCi0gICAgICAgICAgICBSZXNvdXJjZUJ1bmRsZSByYjsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgcmIgPSBSZXNvdXJjZUJ1bmRsZS5nZXRCdW5kbGUoYnVuZGxlLCBsb2NhbGUsIGNvbnRleHRDbGFzc2xvYWRlcik7Ci0gICAgICAgICAgICB9IGNhdGNoIChNaXNzaW5nUmVzb3VyY2VFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgICAgIHJiID0gUmVzb3VyY2VCdW5kbGUuZ2V0QnVuZGxlKGJ1bmRsZSwgbG9jYWxlKTsKLSAgICAgICAgICAgICAgICB9IGNhdGNoIChNaXNzaW5nUmVzb3VyY2VFeGNlcHRpb24gZTEpIHsKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiQnVuZGxlIG5vdCBmb3VuZCEiKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgU3RyaW5nIHdhcm5pbmcgPSByYi5nZXRTdHJpbmcoa2V5KTsKLSAgICAgICAgICAgICAgICBsaXN0ZW5lci53YXJuaW5nT2NjdXJyZWQodGhpcywgaW1hZ2VJbmRleCwgd2FybmluZyk7Ci0gICAgICAgICAgICB9IGNhdGNoIChNaXNzaW5nUmVzb3VyY2VFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIlJlc291cmNlIGlzIG1pc3NpbmchIik7Ci0gICAgICAgICAgICB9IGNhdGNoIChDbGFzc0Nhc3RFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIlJlc291cmNlIGlzIG5vdCBhIFN0cmluZyEiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHNwZWNpZmllZCBPYmplY3QgdG8gdGhlIG91dHB1dCBvZiB0aGlzIEltYWdlV3JpdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBvdXRwdXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBPYmplY3Qgd2hpY2ggcmVwcmVzZW50cyBkZXN0aW5hdGlvbiwgaXQgY2FuIGJlCi0gICAgICogICAgICAgICAgICBJbWFnZU91dHB1dFN0cmVhbSBvciBvdGhlciBvYmplY3RzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldE91dHB1dChPYmplY3Qgb3V0cHV0KSB7Ci0gICAgICAgIGlmIChvdXRwdXQgIT0gbnVsbCkgewotICAgICAgICAgICAgSW1hZ2VXcml0ZXJTcGkgc3BpID0gZ2V0T3JpZ2luYXRpbmdQcm92aWRlcigpOwotICAgICAgICAgICAgaWYgKG51bGwgIT0gc3BpKSB7Ci0gICAgICAgICAgICAgICAgQ2xhc3NbXSBvdXRUeXBlcyA9IHNwaS5nZXRPdXRwdXRUeXBlcygpOwotICAgICAgICAgICAgICAgIGJvb2xlYW4gc3VwcG9ydGVkID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgZm9yIChDbGFzczw/PiBlbGVtZW50IDogb3V0VHlwZXMpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGVsZW1lbnQuaXNJbnN0YW5jZShvdXRwdXQpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzdXBwb3J0ZWQgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKCFzdXBwb3J0ZWQpIHsKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigib3V0cHV0ICIgKyBvdXRwdXQgKyAiIGlzIG5vdCBzdXBwb3J0ZWQiKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgdGhpcy5vdXRwdXQgPSBvdXRwdXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIGEgY29tcGxldGVkIGltYWdlIHN0cmVhbSB0aGF0IGNvbnRhaW5zIHRoZSBzcGVjaWZpZWQgaW1hZ2UsCi0gICAgICogZGVmYXVsdCBtZXRhZGF0YSwgYW5kIHRodW1ibmFpbHMgdG8gdGhlIG91dHB1dC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgaW1hZ2UgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQgZHVyaW5nIHdyaXRpbmcuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgd3JpdGUoSUlPSW1hZ2UgaW1hZ2UpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHdyaXRlKG51bGwsIGltYWdlLCBudWxsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcml0ZXMgYSBjb21wbGV0ZWQgaW1hZ2Ugc3RyZWFtIHRoYXQgY29udGFpbnMgdGhlIHNwZWNpZmllZCByZW5kZXJlZAotICAgICAqIGltYWdlLCBkZWZhdWx0IG1ldGFkYXRhLCBhbmQgdGh1bWJuYWlscyB0byB0aGUgb3V0cHV0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBSZW5kZXJlZEltYWdlIHRvIGJlIHdyaXR0ZW4uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkIGR1cmluZyB3cml0aW5nLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHdyaXRlKFJlbmRlcmVkSW1hZ2UgaW1hZ2UpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHdyaXRlKG51bGwsIG5ldyBJSU9JbWFnZShpbWFnZSwgbnVsbCwgbnVsbCksIG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFdyaXRlcyBhIGNvbXBsZXRlZCBpbWFnZSBzdHJlYW0gdGhhdCBjb250YWlucyB0aGUgc3BlY2lmaWVkIGltYWdlLAotICAgICAqIG1ldGFkYXRhIGFuZCB0aHVtYm5haWxzIHRvIHRoZSBvdXRwdXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cmVhbU1ldGFkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgc3RyZWFtIG1ldGFkYXRhLCBvciBudWxsLgotICAgICAqIEBwYXJhbSBpbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBpbWFnZSB0byBiZSB3cml0dGVuLCBpZiBjYW5Xcml0ZVJhc3RlcigpIG1ldGhvZAotICAgICAqICAgICAgICAgICAgcmV0dXJucyBmYWxzZSwgdGhlbiBJbWFnZSBtdXN0IGNvbnRhaW4gb25seSBSZW5kZXJlZEltYWdlLgotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlV3JpdGVQYXJhbSwgb3IgbnVsbC4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gZXJyb3Igb2NjdXJzIGR1cmluZyB3cml0aW5nLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHdyaXRlKElJT01ldGFkYXRhIHN0cmVhbU1ldGFkYXRhLCBJSU9JbWFnZSBpbWFnZSwgSW1hZ2VXcml0ZVBhcmFtIHBhcmFtKQotICAgICAgICAgICAgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogRGlzcG9zZXMgb2YgYW55IHJlc291cmNlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBkaXNwb3NlKCkgewotICAgICAgICAvLyBkZWYgaW1wbC4gZG9lcyBub3RoaW5nIGFjY29yZGluZyB0byB0aGUgc3BlYy4KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXF1ZXN0cyBhbiBhYm9ydCBvcGVyYXRpb24gZm9yIGN1cnJlbnQgd3JpdGluZyBvcGVyYXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHN5bmNocm9uaXplZCB2b2lkIGFib3J0KCkgewotICAgICAgICBhYm9ydGVkID0gdHJ1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgYSByZXF1ZXN0IHRvIGFib3J0IHRoZSBjdXJyZW50IHdyaXRlIG9wZXJhdGlvbiBoYXMKLSAgICAgKiBiZWVuIG1hZGUgc3VjY2Vzc2Z1bGx5LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHJlcXVlc3QgdG8gYWJvcnQgdGhlIGN1cnJlbnQgd3JpdGUgb3BlcmF0aW9uIGhhcwotICAgICAqICAgICAgICAgYmVlbiBtYWRlIHN1Y2Nlc3NmdWxseSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBzeW5jaHJvbml6ZWQgYm9vbGVhbiBhYm9ydFJlcXVlc3RlZCgpIHsKLSAgICAgICAgcmV0dXJuIGFib3J0ZWQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2xlYXJzIGFsbCBwcmV2aW91cyBhYm9ydCByZXF1ZXN0LCBhbmQgYWJvcnRSZXF1ZXN0ZWQgcmV0dXJucyBmYWxzZSBhZnRlcgotICAgICAqIGNhbGxpbmcgdGhpcyBtZXRob2QuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHN5bmNocm9uaXplZCB2b2lkIGNsZWFyQWJvcnRSZXF1ZXN0KCkgewotICAgICAgICBhYm9ydGVkID0gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgSUlPV3JpdGVQcm9ncmVzc0xpc3RlbmVyIGxpc3RlbmVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsaXN0ZW5lcgotICAgICAqICAgICAgICAgICAgdGhlIElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lciBsaXN0ZW5lci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIoSUlPV3JpdGVQcm9ncmVzc0xpc3RlbmVyIGxpc3RlbmVyKSB7Ci0gICAgICAgIGlmIChsaXN0ZW5lciA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAocHJvZ3Jlc3NMaXN0ZW5lcnMgPT0gbnVsbCkgewotICAgICAgICAgICAgcHJvZ3Jlc3NMaXN0ZW5lcnMgPSBuZXcgQXJyYXlMaXN0PElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lcj4oKTsKLSAgICAgICAgfQotCi0gICAgICAgIHByb2dyZXNzTGlzdGVuZXJzLmFkZChsaXN0ZW5lcik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyB0aGUgSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxpc3RlbmVyCi0gICAgICogICAgICAgICAgICB0aGUgSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXIgbGlzdGVuZXIuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXIoSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXIgbGlzdGVuZXIpIHsKLSAgICAgICAgaWYgKGxpc3RlbmVyID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICh3YXJuaW5nTGlzdGVuZXJzID09IG51bGwpIHsKLSAgICAgICAgICAgIHdhcm5pbmdMaXN0ZW5lcnMgPSBuZXcgQXJyYXlMaXN0PElJT1dyaXRlV2FybmluZ0xpc3RlbmVyPigpOwotICAgICAgICAgICAgd2FybmluZ0xvY2FsZXMgPSBuZXcgQXJyYXlMaXN0PExvY2FsZT4oKTsKLSAgICAgICAgfQotCi0gICAgICAgIHdhcm5pbmdMaXN0ZW5lcnMuYWRkKGxpc3RlbmVyKTsKLSAgICAgICAgd2FybmluZ0xvY2FsZXMuYWRkKGdldExvY2FsZSgpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBvdXRwdXQgb2JqZWN0IHRoYXQgd2FzIHNldCBieSBzZXRPdXRwdXQgbWV0aG9kLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG91dHB1dCBvYmplY3Qgc3VjaCBhcyBJbWFnZU91dHB1dFN0cmVhbSwgb3IgbnVsbCBpZiBpdCBpcyBub3QKLSAgICAgKiAgICAgICAgIHNldC4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IGdldE91dHB1dCgpIHsKLSAgICAgICAgcmV0dXJuIG91dHB1dDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVjayBvdXRwdXQgcmV0dXJuIGZhbHNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgc3VjY2Vzc2Z1bC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIGJvb2xlYW4gY2hlY2tPdXRwdXRSZXR1cm5GYWxzZSgpIHsKLSAgICAgICAgaWYgKGdldE91dHB1dCgpID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oImdldE91dHB1dCgpID09IG51bGwhIik7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFVuc3VwcG9ydGVkIG9wZXJhdGlvbi4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZpbmFsIHZvaWQgdW5zdXBwb3J0ZWRPcGVyYXRpb24oKSB7Ci0gICAgICAgIGlmIChnZXRPdXRwdXQoKSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKCJnZXRPdXRwdXQoKSA9PSBudWxsISIpOwotICAgICAgICB9Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiVW5zdXBwb3J0ZWQgd3JpdGUgdmFyaWFudCEiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgYSBuZXcgZW1wdHkgaW1hZ2UgY2FuIGJlIGluc2VydGVkIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXguCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgaW5kZXggb2YgaW1hZ2UuCi0gICAgICogQHJldHVybiB0cnVlIGlmIGEgbmV3IGVtcHR5IGltYWdlIGNhbiBiZSBpbnNlcnRlZCBhdCB0aGUgc3BlY2lmaWVkIGluZGV4LAotICAgICAqICAgICAgICAgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBTaWduYWxzIHRoYXQgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY2FuSW5zZXJ0RW1wdHkoaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBjaGVja091dHB1dFJldHVybkZhbHNlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIGEgbmV3IGltYWdlIGNhbiBiZSBpbnNlcnRlZCBhdCB0aGUgc3BlY2lmaWVkIGluZGV4LgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGluZGV4IG9mIGltYWdlLgotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiBhIG5ldyBpbWFnZSBjYW4gYmUgaW5zZXJ0ZWQgYXQgdGhlIHNwZWNpZmllZCBpbmRleCwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgU2lnbmFscyB0aGF0IGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNhbkluc2VydEltYWdlKGludCBpbWFnZUluZGV4KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gY2hlY2tPdXRwdXRSZXR1cm5GYWxzZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgaW1hZ2Ugd2l0aCB0aGUgc3BlY2lmaWVkIGluZGV4IGNhbiBiZSByZW1vdmVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGluZGV4IG9mIGltYWdlLgotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgaW1hZ2Ugd2l0aCB0aGUgc3BlY2lmaWVkIGluZGV4IGNhbiBiZSByZW1vdmVkLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBTaWduYWxzIHRoYXQgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY2FuUmVtb3ZlSW1hZ2UoaW50IGltYWdlSW5kZXgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBjaGVja091dHB1dFJldHVybkZhbHNlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIG1ldGFkYXRhIG9mIHRoZSBpbWFnZSB3aXRoIHRoZSBzcGVjaWZpZWQgaW5kZXggY2FuIGJlCi0gICAgICogcmVwbGFjZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgaW1hZ2UgaW5kZXguCi0gICAgICogQHJldHVybiB0cnVlIGlmIG1ldGFkYXRhIG9mIHRoZSBpbWFnZSB3aXRoIHRoZSBzcGVjaWZpZWQgaW5kZXggY2FuIGJlCi0gICAgICogICAgICAgICByZXBsYWNlZCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjYW5SZXBsYWNlSW1hZ2VNZXRhZGF0YShpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIGNoZWNrT3V0cHV0UmV0dXJuRmFsc2UoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgcGl4ZWxzIG9mIHRoZSBpbWFnZSB3aXRoIHRoZSBzcGVjaWZpZWQgaW5kZXggY2FuIGJlCi0gICAgICogcmVwbGFjZWQgYnkgdGhlIHJlcGxhY2VQaXhlbHMgbWV0aG9kcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlJ3MgaW5kZXguCi0gICAgICogQHJldHVybiB0cnVlIGlmIHBpeGVscyBvZiB0aGUgaW1hZ2Ugd2l0aCB0aGUgc3BlY2lmaWVkIGluZGV4IGNhbiBiZQotICAgICAqICAgICAgICAgcmVwbGFjZWQgYnkgdGhlIHJlcGxhY2VQaXhlbHMgbWV0aG9kcywgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBTaWduYWxzIHRoYXQgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY2FuUmVwbGFjZVBpeGVscyhpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIGNoZWNrT3V0cHV0UmV0dXJuRmFsc2UoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIHN0cmVhbSBtZXRhZGF0YSBwcmVzZW50ZWQgaW4gdGhlIG91dHB1dCBjYW4gYmUKLSAgICAgKiByZW1vdmVkLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgc3RyZWFtIG1ldGFkYXRhIHByZXNlbnRlZCBpbiB0aGUgb3V0cHV0IGNhbiBiZQotICAgICAqICAgICAgICAgcmVtb3ZlZCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjYW5SZXBsYWNlU3RyZWFtTWV0YWRhdGEoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gY2hlY2tPdXRwdXRSZXR1cm5GYWxzZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgd3JpdGluZyBvZiBhIGNvbXBsZXRlIGltYWdlIHN0cmVhbSB3aGljaCBjb250YWlucyBhCi0gICAgICogc2luZ2xlIGltYWdlIGlzIHN1cHBvcnRlZCB3aXRoIHVuZGVmaW5lZCBwaXhlbCB2YWx1ZXMgYW5kIGFzc29jaWF0ZWQKLSAgICAgKiBtZXRhZGF0YSBhbmQgdGh1bWJuYWlscyB0byB0aGUgb3V0cHV0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgd3JpdGluZyBvZiBhIGNvbXBsZXRlIGltYWdlIHN0cmVhbSB3aGljaCBjb250YWlucyBhCi0gICAgICogICAgICAgICBzaW5nbGUgaW1hZ2UgaXMgc3VwcG9ydGVkLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNhbldyaXRlRW1wdHkoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gY2hlY2tPdXRwdXRSZXR1cm5GYWxzZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgbWV0aG9kcyB3aGljaCB0YWtlbiBhbiBJSU9JbWFnZVBhcmFtZXRlciBjYW4gZGVhbAotICAgICAqIHdpdGggYSBSYXN0ZXIgc291cmNlIGltYWdlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgbWV0aG9kcyB3aGljaCB0YWtlbiBhbiBJSU9JbWFnZVBhcmFtZXRlciBjYW4gZGVhbAotICAgICAqICAgICAgICAgd2l0aCBhIFJhc3RlciBzb3VyY2UgaW1hZ2UsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjYW5Xcml0ZVJhc3RlcnMoKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIHdyaXRlciBjYW4gYWRkIGFuIGltYWdlIHRvIHN0cmVhbSB0aGF0IGFscmVhZHkKLSAgICAgKiBjb250YWlucyBoZWFkZXIgaW5mb3JtYXRpb24uCi0gICAgICogCi0gICAgICogQHJldHVybiBpZiB0aGUgd3JpdGVyIGNhbiBhZGQgYW4gaW1hZ2UgdG8gc3RyZWFtIHRoYXQgYWxyZWFkeSBjb250YWlucwotICAgICAqICAgICAgICAgaGVhZGVyIGluZm9ybWF0aW9uLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY2FuV3JpdGVTZXF1ZW5jZSgpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEVuZHMgdGhlIGluc2VydGlvbiBvZiBhIG5ldyBpbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZW5kSW5zZXJ0RW1wdHkoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB1bnN1cHBvcnRlZE9wZXJhdGlvbigpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEVuZHMgdGhlIHJlcGxhY2UgcGl4ZWxzIG9wZXJhdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZW5kUmVwbGFjZVBpeGVscygpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHVuc3VwcG9ydGVkT3BlcmF0aW9uKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRW5kcyBhbiBlbXB0eSB3cml0ZSBvcGVyYXRpb24uCi0gICAgICogCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGVuZFdyaXRlRW1wdHkoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB1bnN1cHBvcnRlZE9wZXJhdGlvbigpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEVuZHMgdGhlIHNlcXVlbmNlIG9mIHdyaXRlIG9wZXJhdGlvbnMuCi0gICAgICogCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGVuZFdyaXRlU2VxdWVuY2UoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB1bnN1cHBvcnRlZE9wZXJhdGlvbigpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgYXZhaWxhYmxlIGxvY2FsZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiBhbiBvZiBhcnJheSBhdmFpbGFibGUgbG9jYWxlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgTG9jYWxlW10gZ2V0QXZhaWxhYmxlTG9jYWxlcygpIHsKLSAgICAgICAgaWYgKGF2YWlsYWJsZUxvY2FsZXMgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gYXZhaWxhYmxlTG9jYWxlcy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gSUlPTWV0YWRhdGEgb2JqZWN0IHRoYXQgY29udGFpbnMgZGVmYXVsdCB2YWx1ZXMgZm9yIGVuY29kaW5nIGFuCi0gICAgICogaW1hZ2Ugd2l0aCB0aGUgc3BlY2lmaWVkIHR5cGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlVHlwZVNwZWNpZmllci4KLSAgICAgKiBAcGFyYW0gcGFyYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVdyaXRlUGFyYW0uCi0gICAgICogQHJldHVybiB0aGUgSUlPTWV0YWRhdGEgb2JqZWN0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJSU9NZXRhZGF0YSBnZXREZWZhdWx0SW1hZ2VNZXRhZGF0YShJbWFnZVR5cGVTcGVjaWZpZXIgaW1hZ2VUeXBlLAotICAgICAgICAgICAgSW1hZ2VXcml0ZVBhcmFtIHBhcmFtKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gSUlPTWV0YWRhdGEgb2JqZWN0IHRoYXQgY29udGFpbnMgZGVmYXVsdCB2YWx1ZXMgZm9yIGVuY29kaW5nIGEKLSAgICAgKiBzdHJlYW0gb2YgaW1hZ2VzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlV3JpdGVQYXJhbS4KLSAgICAgKiBAcmV0dXJuIHRoZSBJSU9NZXRhZGF0YSBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IElJT01ldGFkYXRhIGdldERlZmF1bHRTdHJlYW1NZXRhZGF0YShJbWFnZVdyaXRlUGFyYW0gcGFyYW0pOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY3VycmVudCBsb2NhbGUgb2YgdGhpcyBJbWFnZVdyaXRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjdXJyZW50IGxvY2FsZSBvZiB0aGlzIEltYWdlV3JpdGVyLgotICAgICAqLwotICAgIHB1YmxpYyBMb2NhbGUgZ2V0TG9jYWxlKCkgewotICAgICAgICByZXR1cm4gbG9jYWxlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRlZmF1bHQgd3JpdGUgcGFyYW0uIEdldHMgYSBuZXcgSW1hZ2VXcml0ZVBhcmFtIG9iamVjdCBmb3IgdGhpcwotICAgICAqIEltYWdlV3JpdGVyIHdpdGggdGhlIGN1cnJlbnQgTG9jYWxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBuZXcgSW1hZ2VXcml0ZVBhcmFtIG9iamVjdCBmb3IgdGhpcyBJbWFnZVdyaXRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VXcml0ZVBhcmFtIGdldERlZmF1bHRXcml0ZVBhcmFtKCkgewotICAgICAgICByZXR1cm4gbmV3IEltYWdlV3JpdGVQYXJhbShnZXRMb2NhbGUoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbnVtYmVyIG9mIHRodW1ibmFpbHMgc3VwcG9ydGVkIGJ5IHRoZSBmb3JtYXQgYmVpbmcgd3JpdHRlbiB3aXRoCi0gICAgICogc3VwcG9ydGVkIGltYWdlIHR5cGUsIGltYWdlIHdyaXRlIHBhcmFtZXRlcnMsIHN0cmVhbSwgYW5kIGltYWdlIG1ldGFkYXRhCi0gICAgICogb2JqZWN0cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VUeXBlU3BlY2lmaWVyLgotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlJ3MgcGFyYW1ldGVycy4KLSAgICAgKiBAcGFyYW0gc3RyZWFtTWV0YWRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdHJlYW0gbWV0YWRhdGEuCi0gICAgICogQHBhcmFtIGltYWdlTWV0YWRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSBtZXRhZGF0YS4KLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgdGh1bWJuYWlscyBzdXBwb3J0ZWQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXROdW1UaHVtYm5haWxzU3VwcG9ydGVkKEltYWdlVHlwZVNwZWNpZmllciBpbWFnZVR5cGUsIEltYWdlV3JpdGVQYXJhbSBwYXJhbSwKLSAgICAgICAgICAgIElJT01ldGFkYXRhIHN0cmVhbU1ldGFkYXRhLCBJSU9NZXRhZGF0YSBpbWFnZU1ldGFkYXRhKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHByZWZlcnJlZCB0aHVtYm5haWwgc2l6ZXMuIEdldHMgYW4gYXJyYXkgb2YgRGltZW5zaW9ucyB3aXRoIHRoZQotICAgICAqIHNpemVzIGZvciB0aHVtYm5haWwgaW1hZ2VzIGFzIHRoZXkgYXJlIGVuY29kZWQgaW4gdGhlIG91dHB1dCBmaWxlIG9yCi0gICAgICogc3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICogQHBhcmFtIHBhcmFtCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VXcml0ZVBhcmFtLgotICAgICAqIEBwYXJhbSBzdHJlYW1NZXRhZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIHN0cmVhbSBtZXRhZGF0YS4KLSAgICAgKiBAcGFyYW0gaW1hZ2VNZXRhZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIG1ldGFkYXRhLgotICAgICAqIEByZXR1cm4gdGhlIHByZWZlcnJlZCB0aHVtYm5haWwgc2l6ZXMuCi0gICAgICovCi0gICAgcHVibGljIERpbWVuc2lvbltdIGdldFByZWZlcnJlZFRodW1ibmFpbFNpemVzKEltYWdlVHlwZVNwZWNpZmllciBpbWFnZVR5cGUsCi0gICAgICAgICAgICBJbWFnZVdyaXRlUGFyYW0gcGFyYW0sIElJT01ldGFkYXRhIHN0cmVhbU1ldGFkYXRhLCBJSU9NZXRhZGF0YSBpbWFnZU1ldGFkYXRhKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByZXBhcmVzIGluc2VydGlvbiBvZiBhbiBlbXB0eSBpbWFnZSBieSByZXF1ZXN0aW5nIHRoZSBpbnNlcnRpb24gb2YgYSBuZXcKLSAgICAgKiBpbWFnZSBpbnRvIGFuIGV4aXN0aW5nIGltYWdlIHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGluZGV4LgotICAgICAqIEBwYXJhbSBpbWFnZVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSB0eXBlLgotICAgICAqIEBwYXJhbSB3aWR0aAotICAgICAqICAgICAgICAgICAgdGhlIHdpZHRoIG9mIHRoZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaGVpZ2h0Ci0gICAgICogICAgICAgICAgICB0aGUgaGVpZ2h0IG9mIHRoZSBpbWFnZS4KLSAgICAgKiBAcGFyYW0gaW1hZ2VNZXRhZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIG1ldGFkYXRhLCBvciBudWxsLgotICAgICAqIEBwYXJhbSB0aHVtYm5haWxzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgdGh1bWJuYWlscyBmb3IgdGhpcyBpbWFnZSwgb3IgbnVsbC4KLSAgICAgKiBAcGFyYW0gcGFyYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVdyaXRlUGFyYW0sIG9yIG51bGwuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHByZXBhcmVJbnNlcnRFbXB0eShpbnQgaW1hZ2VJbmRleCwgSW1hZ2VUeXBlU3BlY2lmaWVyIGltYWdlVHlwZSwgaW50IHdpZHRoLAotICAgICAgICAgICAgaW50IGhlaWdodCwgSUlPTWV0YWRhdGEgaW1hZ2VNZXRhZGF0YSwgTGlzdDw/IGV4dGVuZHMgQnVmZmVyZWRJbWFnZT4gdGh1bWJuYWlscywKLSAgICAgICAgICAgIEltYWdlV3JpdGVQYXJhbSBwYXJhbSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdW5zdXBwb3J0ZWRPcGVyYXRpb24oKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcmVwYXJlcyB0aGUgd3JpdGVyIHRvIGNhbGwgdGhlIHJlcGxhY2VQaXhlbHMgbWV0aG9kIGZvciB0aGUgc3BlY2lmaWVkCi0gICAgICogcmVnaW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UncyBpbmRleC4KLSAgICAgKiBAcGFyYW0gcmVnaW9uCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIHJlZ2lvbi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcHJlcGFyZVJlcGxhY2VQaXhlbHMoaW50IGltYWdlSW5kZXgsIFJlY3RhbmdsZSByZWdpb24pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHVuc3VwcG9ydGVkT3BlcmF0aW9uKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJlcGFyZXMgdGhlIHdyaXRlciBmb3Igd3JpdGluZyBhbiBlbXB0eSBpbWFnZSBieSBiZWdpbm5pbmcgdGhlIHByb2Nlc3MKLSAgICAgKiBvZiB3cml0aW5nIGEgY29tcGxldGUgaW1hZ2Ugc3RyZWFtIHRoYXQgY29udGFpbnMgYSBzaW5nbGUgaW1hZ2Ugd2l0aAotICAgICAqIHVuZGVmaW5lZCBwaXhlbCB2YWx1ZXMsIG1ldGFkYXRhIGFuZCB0aHVtYm5haWxzLCB0byB0aGUgb3V0cHV0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHJlYW1NZXRhZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIHN0cmVhbSBtZXRhZGF0YS4KLSAgICAgKiBAcGFyYW0gaW1hZ2VUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgdHlwZS4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB0aGUgaW1hZ2UuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB0aGUgaW1hZ2UuCi0gICAgICogQHBhcmFtIGltYWdlTWV0YWRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSdzIG1ldGFkYXRhLCBvciBudWxsLgotICAgICAqIEBwYXJhbSB0aHVtYm5haWxzCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UncyB0aHVtYm5haWxzLCBvciBudWxsLgotICAgICAqIEBwYXJhbSBwYXJhbQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlJ3MgcGFyYW1ldGVycywgb3IgbnVsbC4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcHJlcGFyZVdyaXRlRW1wdHkoSUlPTWV0YWRhdGEgc3RyZWFtTWV0YWRhdGEsIEltYWdlVHlwZVNwZWNpZmllciBpbWFnZVR5cGUsCi0gICAgICAgICAgICBpbnQgd2lkdGgsIGludCBoZWlnaHQsIElJT01ldGFkYXRhIGltYWdlTWV0YWRhdGEsCi0gICAgICAgICAgICBMaXN0PD8gZXh0ZW5kcyBCdWZmZXJlZEltYWdlPiB0aHVtYm5haWxzLCBJbWFnZVdyaXRlUGFyYW0gcGFyYW0pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHVuc3VwcG9ydGVkT3BlcmF0aW9uKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJlcGFyZXMgYSBzdHJlYW0gdG8gYWNjZXB0IGNhbGxzIG9mIHdyaXRlVG9TZXF1ZW5jZSBtZXRob2QgdXNpbmcgdGhlCi0gICAgICogbWV0YWRhdGEgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHJlYW1NZXRhZGF0YQotICAgICAqICAgICAgICAgICAgdGhlIHN0cmVhbSBtZXRhZGF0YS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcHJlcGFyZVdyaXRlU2VxdWVuY2UoSUlPTWV0YWRhdGEgc3RyZWFtTWV0YWRhdGEpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHVuc3VwcG9ydGVkT3BlcmF0aW9uKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIHRoZSBjb21wbGV0aW9uIG9mIGEgdGh1bWJuYWlsIHJlYWQgYnkgY2FsbGluZyB0aGVpcgotICAgICAqIHRodW1ibmFpbENvbXBsZXRlIG1ldGhvZCBvZiByZWdpc3RlcmVkIElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lcnMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgcHJvY2Vzc1RodW1ibmFpbENvbXBsZXRlKCkgewotICAgICAgICBpZiAocHJvZ3Jlc3NMaXN0ZW5lcnMgIT0gbnVsbCkgewotICAgICAgICAgICAgZm9yIChJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIgbGlzdGVuZXIgOiBwcm9ncmVzc0xpc3RlbmVycykgewotICAgICAgICAgICAgICAgIGxpc3RlbmVyLnRodW1ibmFpbENvbXBsZXRlKHRoaXMpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUHJvY2Vzc2VzIHRoZSBjdXJyZW50IHBlcmNlbnRhZ2Ugb2YgdGh1bWJuYWlsIGNvbXBsZXRpb24gYnkgY2FsbGluZyB0aGVpcgotICAgICAqIHRodW1ibmFpbFByb2dyZXNzIG1ldGhvZCBvZiByZWdpc3RlcmVkIElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHBlcmNlbnRhZ2VEb25lCi0gICAgICogICAgICAgICAgICB0aGUgcGVyY2VudGFnZSBkb25lLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NUaHVtYm5haWxQcm9ncmVzcyhmbG9hdCBwZXJjZW50YWdlRG9uZSkgewotICAgICAgICBpZiAocHJvZ3Jlc3NMaXN0ZW5lcnMgIT0gbnVsbCkgewotICAgICAgICAgICAgZm9yIChJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIgbGlzdGVuZXIgOiBwcm9ncmVzc0xpc3RlbmVycykgewotICAgICAgICAgICAgICAgIGxpc3RlbmVyLnRodW1ibmFpbFByb2dyZXNzKHRoaXMsIHBlcmNlbnRhZ2VEb25lKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyB0aGUgc3RhcnQgb2YgYSB0aHVtYm5haWwgcmVhZCBieSBjYWxsaW5nIHRodW1ibmFpbFN0YXJ0ZWQKLSAgICAgKiBtZXRob2Qgb2YgcmVnaXN0ZXJlZCBJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgaW5kZXguCi0gICAgICogQHBhcmFtIHRodW1ibmFpbEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgdGh1bWJuYWlsIGluZGV4LgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NUaHVtYm5haWxTdGFydGVkKGludCBpbWFnZUluZGV4LCBpbnQgdGh1bWJuYWlsSW5kZXgpIHsKLSAgICAgICAgaWYgKHByb2dyZXNzTGlzdGVuZXJzICE9IG51bGwpIHsKLSAgICAgICAgICAgIGZvciAoSUlPV3JpdGVQcm9ncmVzc0xpc3RlbmVyIGxpc3RlbmVyIDogcHJvZ3Jlc3NMaXN0ZW5lcnMpIHsKLSAgICAgICAgICAgICAgICBsaXN0ZW5lci50aHVtYm5haWxTdGFydGVkKHRoaXMsIGltYWdlSW5kZXgsIHRodW1ibmFpbEluZGV4KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByb2Nlc3NlcyB0aGF0IHRoZSB3cml0aW5nIGhhcyBiZWVuIGFib3J0ZWQgYnkgY2FsbGluZyB3cml0ZUFib3J0ZWQKLSAgICAgKiBtZXRob2Qgb2YgcmVnaXN0ZXJlZCBJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXJzLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHByb2Nlc3NXcml0ZUFib3J0ZWQoKSB7Ci0gICAgICAgIGlmIChwcm9ncmVzc0xpc3RlbmVycyAhPSBudWxsKSB7Ci0gICAgICAgICAgICBmb3IgKElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lciBsaXN0ZW5lciA6IHByb2dyZXNzTGlzdGVuZXJzKSB7Ci0gICAgICAgICAgICAgICAgbGlzdGVuZXIud3JpdGVBYm9ydGVkKHRoaXMpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgYWxsIElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lciBsaXN0ZW5lcnMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlQWxsSUlPV3JpdGVQcm9ncmVzc0xpc3RlbmVycygpIHsKLSAgICAgICAgcHJvZ3Jlc3NMaXN0ZW5lcnMgPSBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIGFsbCBJSU9Xcml0ZVdhcm5pbmdMaXN0ZW5lciBsaXN0ZW5lcnMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlQWxsSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXJzKCkgewotICAgICAgICB3YXJuaW5nTGlzdGVuZXJzID0gbnVsbDsKLSAgICAgICAgd2FybmluZ0xvY2FsZXMgPSBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIHNwZWNpZmllZCBJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIgbGlzdGVuZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxpc3RlbmVyCi0gICAgICogICAgICAgICAgICB0aGUgcmVnaXN0ZXJlZCBJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIgdG8gYmUgcmVtb3ZlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIoSUlPV3JpdGVQcm9ncmVzc0xpc3RlbmVyIGxpc3RlbmVyKSB7Ci0gICAgICAgIGlmIChwcm9ncmVzc0xpc3RlbmVycyAhPSBudWxsICYmIGxpc3RlbmVyICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChwcm9ncmVzc0xpc3RlbmVycy5yZW1vdmUobGlzdGVuZXIpICYmIHByb2dyZXNzTGlzdGVuZXJzLmlzRW1wdHkoKSkgewotICAgICAgICAgICAgICAgIHByb2dyZXNzTGlzdGVuZXJzID0gbnVsbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIHNwZWNpZmllZCBJSU9Xcml0ZVdhcm5pbmdMaXN0ZW5lciBsaXN0ZW5lci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbGlzdGVuZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSByZWdpc3RlcmVkIElJT1dyaXRlV2FybmluZ0xpc3RlbmVyIGxpc3RlbmVyIHRvIGJlIHJlbW92ZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXIoSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXIgbGlzdGVuZXIpIHsKLSAgICAgICAgaWYgKHdhcm5pbmdMaXN0ZW5lcnMgPT0gbnVsbCB8fCBsaXN0ZW5lciA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgaWR4ID0gd2FybmluZ0xpc3RlbmVycy5pbmRleE9mKGxpc3RlbmVyKTsKLSAgICAgICAgaWYgKGlkeCA+IC0xKSB7Ci0gICAgICAgICAgICB3YXJuaW5nTGlzdGVuZXJzLnJlbW92ZShpZHgpOwotICAgICAgICAgICAgd2FybmluZ0xvY2FsZXMucmVtb3ZlKGlkeCk7Ci0KLSAgICAgICAgICAgIGlmICh3YXJuaW5nTGlzdGVuZXJzLmlzRW1wdHkoKSkgewotICAgICAgICAgICAgICAgIHdhcm5pbmdMaXN0ZW5lcnMgPSBudWxsOwotICAgICAgICAgICAgICAgIHdhcm5pbmdMb2NhbGVzID0gbnVsbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIGltYWdlIHdpdGggdGhlIHNwZWNpZmllZCBpbmRleCBmcm9tIHRoZSBzdHJlYW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSdzIGluZGV4LgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVJbWFnZShpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdW5zdXBwb3J0ZWRPcGVyYXRpb24oKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXBsYWNlcyBpbWFnZSBtZXRhZGF0YSBvZiB0aGUgaW1hZ2Ugd2l0aCBzcGVjaWZpZWQgaW5kZXguCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbWFnZSdzIGluZGV4LgotICAgICAqIEBwYXJhbSBpbWFnZU1ldGFkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgbWV0YWRhdGEuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlcGxhY2VJbWFnZU1ldGFkYXRhKGludCBpbWFnZUluZGV4LCBJSU9NZXRhZGF0YSBpbWFnZU1ldGFkYXRhKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB1bnN1cHBvcnRlZE9wZXJhdGlvbigpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlcGxhY2VzIGEgcGFydCBvZiBhbiBpbWFnZSBwcmVzZW50ZWQgaW4gdGhlIG91dHB1dCB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBSZW5kZXJlZEltYWdlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIFJlbmRlcmVkSW1hZ2UuCi0gICAgICogQHBhcmFtIHBhcmFtCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VXcml0ZVBhcmFtLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZXBsYWNlUGl4ZWxzKFJlbmRlcmVkSW1hZ2UgaW1hZ2UsIEltYWdlV3JpdGVQYXJhbSBwYXJhbSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdW5zdXBwb3J0ZWRPcGVyYXRpb24oKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXBsYWNlcyBhIHBhcnQgb2YgYW4gaW1hZ2UgcHJlc2VudGVkIGluIHRoZSBvdXRwdXQgd2l0aCB0aGUgc3BlY2lmaWVkCi0gICAgICogUmFzdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSByYXN0ZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBSYXN0ZXIuCi0gICAgICogQHBhcmFtIHBhcmFtCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VXcml0ZVBhcmFtLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZXBsYWNlUGl4ZWxzKFJhc3RlciByYXN0ZXIsIEltYWdlV3JpdGVQYXJhbSBwYXJhbSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdW5zdXBwb3J0ZWRPcGVyYXRpb24oKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXBsYWNlcyB0aGUgc3RyZWFtIG1ldGFkYXRhIG9mIHRoZSBvdXRwdXQgd2l0aCBuZXcgSUlPTWV0YWRhdGEuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cmVhbU1ldGFkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHN0cmVhbSBtZXRhZGF0YS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVwbGFjZVN0cmVhbU1ldGFkYXRhKElJT01ldGFkYXRhIHN0cmVhbU1ldGFkYXRhKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB1bnN1cHBvcnRlZE9wZXJhdGlvbigpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGxvY2FsZSBvZiB0aGlzIEltYWdlV3JpdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsb2NhbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgbG9jYWxlLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldExvY2FsZShMb2NhbGUgbG9jYWxlKSB7Ci0gICAgICAgIGlmIChsb2NhbGUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhpcy5sb2NhbGUgPSBudWxsOwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgTG9jYWxlW10gbG9jYWxlcyA9IGdldEF2YWlsYWJsZUxvY2FsZXMoKTsKLSAgICAgICAgYm9vbGVhbiB2YWxpZExvY2FsZSA9IGZhbHNlOwotICAgICAgICBpZiAobG9jYWxlcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGxvY2FsZXMubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgICAgICBpZiAobG9jYWxlLmVxdWFscyhsb2NhbGVzW2ldKSkgewotICAgICAgICAgICAgICAgICAgICB2YWxpZExvY2FsZSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmICh2YWxpZExvY2FsZSkgewotICAgICAgICAgICAgdGhpcy5sb2NhbGUgPSBsb2NhbGU7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJJbnZhbGlkIGxvY2FsZSEiKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlc2V0cyB0aGlzIEltYWdlV3JpdGVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlc2V0KCkgewotICAgICAgICBzZXRPdXRwdXQobnVsbCk7Ci0gICAgICAgIHNldExvY2FsZShudWxsKTsKLSAgICAgICAgcmVtb3ZlQWxsSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXJzKCk7Ci0gICAgICAgIHJlbW92ZUFsbElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lcnMoKTsKLSAgICAgICAgY2xlYXJBYm9ydFJlcXVlc3QoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnNlcnRzIGltYWdlIGludG8gZXhpc3Rpbmcgb3V0cHV0IHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIGluZGV4IHdoZXJlIGFuIGltYWdlIHdpbGwgYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gaW1hZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgaW1hZ2UgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gcGFyYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVdyaXRlUGFyYW0sIG9yIG51bGwuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHdyaXRlSW5zZXJ0KGludCBpbWFnZUluZGV4LCBJSU9JbWFnZSBpbWFnZSwgSW1hZ2VXcml0ZVBhcmFtIHBhcmFtKQotICAgICAgICAgICAgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdW5zdXBwb3J0ZWRPcGVyYXRpb24oKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXcml0ZXMgdGhlIHNwZWNpZmllZCBpbWFnZSB0byB0aGUgc2VxdWVuY2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGltYWdlCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gcGFyYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVdyaXRlUGFyYW0sIG9yIG51bGwuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkIGR1cmluZyB3cml0aW5nLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHdyaXRlVG9TZXF1ZW5jZShJSU9JbWFnZSBpbWFnZSwgSW1hZ2VXcml0ZVBhcmFtIHBhcmFtKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB1bnN1cHBvcnRlZE9wZXJhdGlvbigpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL2V2ZW50L0lJT1JlYWRQcm9ncmVzc0xpc3RlbmVyLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9ldmVudC9JSU9SZWFkUHJvZ3Jlc3NMaXN0ZW5lci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyOTQ0ODk2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL2V2ZW50L0lJT1JlYWRQcm9ncmVzc0xpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMjEgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBTZXJnZXkgSS4gU2FsaXNoZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4yICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlUmVhZGVyOwotCi0vKioKLSAqIFRoZSBJSU9SZWFkUHJvZ3Jlc3NMaXN0ZW5lciBpbnRlcmZhY2Ugbm90aWZpZXMgY2FsbGVycyBhYm91dCB0aGUgcHJvZ3Jlc3Mgb2YKLSAqIHRoZSBpbWFnZSBhbmQgdGh1bWJuYWlsIHJlYWRpbmcgbWV0aG9kcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgSUlPUmVhZFByb2dyZXNzTGlzdGVuZXIgZXh0ZW5kcyBFdmVudExpc3RlbmVyIHsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgdGhhdCB0aGUgaW1hZ2UgcmVhZGluZyBoYXMgYmVlbiBjb21wbGV0ZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUmVhZGVyIG9iamVjdCB3aGljaCBjYWxscyB0aGlzIG1ldGhvZC4KLSAgICAgKi8KLSAgICB2b2lkIGltYWdlQ29tcGxldGUoSW1hZ2VSZWFkZXIgc291cmNlKTsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgYWJvdXQgdGhlIGRlZ3JlZSBvZiBjb21wbGV0aW9uIG9mIHRoZSByZWFkIGNhbGwuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUmVhZGVyIG9iamVjdCB3aGljaCBjYWxscyB0aGlzIG1ldGhvZC4KLSAgICAgKiBAcGFyYW0gcGVyY2VudGFnZURvbmUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwZXJjZW50YWdlIG9mIGRlY29kaW5nIGRvbmUuCi0gICAgICovCi0gICAgdm9pZCBpbWFnZVByb2dyZXNzKEltYWdlUmVhZGVyIHNvdXJjZSwgZmxvYXQgcGVyY2VudGFnZURvbmUpOwotCi0gICAgLyoqCi0gICAgICogTm90aWZpZXMgdGhpcyBsaXN0ZW5lciB0aGF0IGFuIGltYWdlIHJlYWQgb3BlcmF0aW9uIGhhcyBiZWVuIHN0YXJ0ZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUmVhZGVyIG9iamVjdCB3aGljaCBjYWxscyB0aGlzIG1ldGhvZC4KLSAgICAgKiBAcGFyYW0gaW1hZ2VJbmRleAotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBpbWFnZSBpbiBhbiBpbnB1dCBmaWxlIG9yIHN0cmVhbSB0byBiZSByZWFkLgotICAgICAqLwotICAgIHZvaWQgaW1hZ2VTdGFydGVkKEltYWdlUmVhZGVyIHNvdXJjZSwgaW50IGltYWdlSW5kZXgpOwotCi0gICAgLyoqCi0gICAgICogTm90aWZpZXMgdGhpcyBsaXN0ZW5lciB0aGF0IGEgcmVhZCBvcGVyYXRpb24gaGFzIGJlZW4gYWJvcnRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VSZWFkZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqLwotICAgIHZvaWQgcmVhZEFib3J0ZWQoSW1hZ2VSZWFkZXIgc291cmNlKTsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgdGhhdCBhIHNlcXVlbmNlIG9mIHJlYWQgb3BlcmF0aW9ucyBoYXMgYmVlbgotICAgICAqIGNvbXBsZXRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VSZWFkZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqLwotICAgIHZvaWQgc2VxdWVuY2VDb21wbGV0ZShJbWFnZVJlYWRlciBzb3VyY2UpOwotCi0gICAgLyoqCi0gICAgICogTm90aWZpZXMgdGhpcyBsaXN0ZW5lciB0aGF0IGEgc2VxdWVuY2Ugb2YgcmVhZCBvcGVyYXRpb24gaGFzIGJlZW4KLSAgICAgKiBzdGFydGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVJlYWRlciBvYmplY3Qgd2hpY2ggY2FsbHMgdGhpcyBtZXRob2QuCi0gICAgICogQHBhcmFtIG1pbkluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggb2YgdGhlIGZpcnN0IGltYWdlIHRvIGJlIHJlYWQuCi0gICAgICovCi0gICAgdm9pZCBzZXF1ZW5jZVN0YXJ0ZWQoSW1hZ2VSZWFkZXIgc291cmNlLCBpbnQgbWluSW5kZXgpOwotCi0gICAgLyoqCi0gICAgICogTm90aWZpZXMgdGhhdCBhIHRodW1ibmFpbCByZWFkIG9wZXJhdGlvbiBoYXMgYmVlbiBjb21wbGV0ZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlUmVhZGVyIG9iamVjdCB3aGljaCBjYWxscyB0aGlzIG1ldGhvZC4KLSAgICAgKi8KLSAgICB2b2lkIHRodW1ibmFpbENvbXBsZXRlKEltYWdlUmVhZGVyIHNvdXJjZSk7Ci0KLSAgICAvKioKLSAgICAgKiBOb3RpZmllcyB0aGlzIGxpc3RlbmVyIGFib3V0IHRoZSBkZWdyZWUgb2YgY29tcGxldGlvbiBvZiB0aGUgcmVhZCBjYWxsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVJlYWRlciBvYmplY3Qgd2hpY2ggY2FsbHMgdGhpcyBtZXRob2QuCi0gICAgICogQHBhcmFtIHBlcmNlbnRhZ2VEb25lCi0gICAgICogICAgICAgICAgICB0aGUgcGVyY2VudGFnZSBvZiBkZWNvZGluZyBkb25lLgotICAgICAqLwotICAgIHZvaWQgdGh1bWJuYWlsUHJvZ3Jlc3MoSW1hZ2VSZWFkZXIgc291cmNlLCBmbG9hdCBwZXJjZW50YWdlRG9uZSk7Ci0KLSAgICAvKioKLSAgICAgKiBOb3RpZmllcyB0aGlzIGxpc3RlbmVyIHRoYXQgYSB0aHVtYm5haWwgcmVhZGluZyBvcGVyYXRpb24gaGFzIGJlZW4KLSAgICAgKiBzdGFydGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVJlYWRlciBvYmplY3Qgd2hpY2ggY2FsbHMgdGhpcyBtZXRob2QuCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgaW1hZ2UgaW4gYW4gaW5wdXQgZmlsZSBvciBzdHJlYW0gdG8gYmUgcmVhZC4KLSAgICAgKiBAcGFyYW0gdGh1bWJuYWlsSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgdGh1bWJuYWlsIHRvIGJlIHJlYWQuCi0gICAgICovCi0gICAgdm9pZCB0aHVtYm5haWxTdGFydGVkKEltYWdlUmVhZGVyIHNvdXJjZSwgaW50IGltYWdlSW5kZXgsIGludCB0aHVtYm5haWxJbmRleCk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9ldmVudC9JSU9SZWFkVXBkYXRlTGlzdGVuZXIuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL2V2ZW50L0lJT1JlYWRVcGRhdGVMaXN0ZW5lci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0OWJkYmNiLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL2V2ZW50L0lJT1JlYWRVcGRhdGVMaXN0ZW5lci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTgyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgU2VyZ2V5IEkuIFNhbGlzaGV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvLmV2ZW50OwotCi1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlUmVhZGVyOwotCi0vKgotICogQGF1dGhvciBTZXJnZXkgSS4gU2FsaXNoZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4yICQKLSAqLwotCi0vKioKLSAqIFRoZSBJSU9SZWFkVXBkYXRlTGlzdGVuZXIgaW50ZXJmYWNlIHByb3ZpZGVzIGZ1bmN0aW9uYWxpdHkgdG8gcmVjZWl2ZQotICogbm90aWZpY2F0aW9uIG9mIHBpeGVsIHVwZGF0ZXMgZHVyaW5nIGltYWdlIGFuZCB0aHVtYm5haWwgcmVhZGluZyBvcGVyYXRpb25zLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJSU9SZWFkVXBkYXRlTGlzdGVuZXIgZXh0ZW5kcyBFdmVudExpc3RlbmVyIHsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgdGhhdCB0aGUgc3BlY2lmaWVkIGFyZWEgb2YgdGhlIGltYWdlIGhhcyBiZWVuCi0gICAgICogdXBkYXRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VSZWFkZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqIEBwYXJhbSB0aGVJbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHRvIGJlIHVwZGF0ZWQuCi0gICAgICogQHBhcmFtIG1pblgKLSAgICAgKiAgICAgICAgICAgIHRoZSBtaW5pbXVtIFggY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWxzIGluIHRoZSB1cGRhdGVkIGFyZWEuCi0gICAgICogQHBhcmFtIG1pblkKLSAgICAgKiAgICAgICAgICAgIHRoZSBtaW5pbXVtIFkgY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWxzIGluIHRoZSB1cGRhdGVkIGFyZWEuCi0gICAgICogQHBhcmFtIHdpZHRoCi0gICAgICogICAgICAgICAgICB0aGUgd2lkdGggb2YgdXBkYXRlZCBhcmVhLgotICAgICAqIEBwYXJhbSBoZWlnaHQKLSAgICAgKiAgICAgICAgICAgIHRoZSBoZWlnaHQgb2YgdXBkYXRlZCBhcmVhLgotICAgICAqIEBwYXJhbSBwZXJpb2RYCi0gICAgICogICAgICAgICAgICB0aGUgaG9yaXpvbnRhbCBzcGFjaW5nIHBlcmlvZCBiZXR3ZWVuIHVwZGF0ZWQgcGl4ZWxzLCBpZiBpdAotICAgICAqICAgICAgICAgICAgZXF1YWxzIDEsIHRoZXJlIGlzIG5vIHNwYWNlIGJldHdlZW4gcGl4ZWxzLgotICAgICAqIEBwYXJhbSBwZXJpb2RZCi0gICAgICogICAgICAgICAgICB0aGUgdmVydGljYWwgc3BhY2luZyBwZXJpb2QgYmV0d2VlbiB1cGRhdGVkIHBpeGVscywgaWYgaXQKLSAgICAgKiAgICAgICAgICAgIGVxdWFscyAxLCB0aGVyZSBpcyBubyBzcGFjZSBiZXR3ZWVuIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gYmFuZHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBpbnRlZ2VyIHZhbHVlcyBpbmRpY2F0aW5nIHRoZSBiYW5kcyBiZWluZwotICAgICAqICAgICAgICAgICAgdXBkYXRlZC4KLSAgICAgKi8KLSAgICB2b2lkIGltYWdlVXBkYXRlKEltYWdlUmVhZGVyIHNvdXJjZSwgQnVmZmVyZWRJbWFnZSB0aGVJbWFnZSwgaW50IG1pblgsIGludCBtaW5ZLCBpbnQgd2lkdGgsCi0gICAgICAgICAgICBpbnQgaGVpZ2h0LCBpbnQgcGVyaW9kWCwgaW50IHBlcmlvZFksIGludFtdIGJhbmRzKTsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgdGhhdCB0aGUgY3VycmVudCByZWFkIG9wZXJhdGlvbiBoYXMgY29tcGxldGVkIGEKLSAgICAgKiBwcm9ncmVzc2l2ZSBwYXNzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVJlYWRlciBvYmplY3Qgd2hpY2ggY2FsbHMgdGhpcyBtZXRob2QuCi0gICAgICogQHBhcmFtIHRoZUltYWdlCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgdG8gYmUgdXBkYXRlZC4KLSAgICAgKi8KLSAgICB2b2lkIHBhc3NDb21wbGV0ZShJbWFnZVJlYWRlciBzb3VyY2UsIEJ1ZmZlcmVkSW1hZ2UgdGhlSW1hZ2UpOwotCi0gICAgLyoqCi0gICAgICogTm90aWZpZXMgdGhpcyBsaXN0ZW5lciB0aGF0IHRoZSBjdXJyZW50IHJlYWQgb3BlcmF0aW9uIGhhcyBiZWd1biBhCi0gICAgICogcHJvZ3Jlc3NpdmUgcGFzcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VSZWFkZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqIEBwYXJhbSB0aGVJbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIGltYWdlIHRvIGJlIHVwZGF0ZWQuCi0gICAgICogQHBhcmFtIHBhc3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgdGhlIHBhc3MuCi0gICAgICogQHBhcmFtIG1pblBhc3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgZmlyc3QgcGFzcyB0aGF0IHdpbGwgYmUgZGVjb2RlZC4KLSAgICAgKiBAcGFyYW0gbWF4UGFzcwotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBsYXN0IHBhc3MgdGhhdCB3aWxsIGJlIGRlY29kZWQuCi0gICAgICogQHBhcmFtIG1pblgKLSAgICAgKiAgICAgICAgICAgIHRoZSBtaW5pbXVtIFggY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWxzIGluIHRoZSB1cGRhdGVkIGFyZWEuCi0gICAgICogQHBhcmFtIG1pblkKLSAgICAgKiAgICAgICAgICAgIHRoZSBtaW5pbXVtIFkgY29vcmRpbmF0ZSBvZiB0aGUgcGl4ZWxzIGluIHRoZSB1cGRhdGVkIGFyZWEuCi0gICAgICogQHBhcmFtIHBlcmlvZFgKLSAgICAgKiAgICAgICAgICAgIHRoZSBob3Jpem9udGFsIHNwYWNpbmcgcGVyaW9kIGJldHdlZW4gdXBkYXRlZCBwaXhlbHMsIGlmIGl0Ci0gICAgICogICAgICAgICAgICBlcXVhbHMgMSwgdGhlcmUgaXMgbm8gc3BhY2UgYmV0d2VlbiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHBlcmlvZFkKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZXJ0aWNhbCBzcGFjaW5nIHBlcmlvZCBiZXR3ZWVuIHVwZGF0ZWQgcGl4ZWxzLCBpZiBpdAotICAgICAqICAgICAgICAgICAgZXF1YWxzIDEsIHRoZXJlIGlzIG5vIHNwYWNlIGJldHdlZW4gcGl4ZWxzLgotICAgICAqIEBwYXJhbSBiYW5kcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGludGVnZXIgdmFsdWVzIGluZGljYXRpbmcgdGhlIGJhbmRzIGJlaW5nCi0gICAgICogICAgICAgICAgICB1cGRhdGVkLgotICAgICAqLwotICAgIHZvaWQgcGFzc1N0YXJ0ZWQoSW1hZ2VSZWFkZXIgc291cmNlLCBCdWZmZXJlZEltYWdlIHRoZUltYWdlLCBpbnQgcGFzcywgaW50IG1pblBhc3MsCi0gICAgICAgICAgICBpbnQgbWF4UGFzcywgaW50IG1pblgsIGludCBtaW5ZLCBpbnQgcGVyaW9kWCwgaW50IHBlcmlvZFksIGludFtdIGJhbmRzKTsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgdGhhdCB0aGUgY3VycmVudCB0aHVtYm5haWwgcmVhZCBvcGVyYXRpb24gaGFzCi0gICAgICogY29tcGxldGVkIGEgcHJvZ3Jlc3NpdmUgcGFzcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VSZWFkZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqIEBwYXJhbSB0aGVJbWFnZQotICAgICAqICAgICAgICAgICAgdGhlIHRodW1ibmFpbCB0byBiZSB1cGRhdGVkLgotICAgICAqLwotICAgIHZvaWQgdGh1bWJuYWlsUGFzc0NvbXBsZXRlKEltYWdlUmVhZGVyIHNvdXJjZSwgQnVmZmVyZWRJbWFnZSB0aGVJbWFnZSk7Ci0KLSAgICAvKioKLSAgICAgKiBOb3RpZmllcyB0aGlzIGxpc3RlbmVyIHRoYXQgdGhlIGN1cnJlbnQgdGh1bWJuYWlsIHJlYWQgb3BlcmF0aW9uIGhhcwotICAgICAqIGJlZ3VuIGEgcHJvZ3Jlc3NpdmUgcGFzcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VSZWFkZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqIEBwYXJhbSB0aGVUaHVtYm5haWwKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aHVtYm5haWwgdG8gYmUgdXBkYXRlZC4KLSAgICAgKiBAcGFyYW0gcGFzcwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiB0aGUgcGFzcy4KLSAgICAgKiBAcGFyYW0gbWluUGFzcwotICAgICAqICAgICAgICAgICAgdGhlIGluZGV4IG9mIHRoZSBmaXJzdCBwYXNzIHRoYXQgd2lsbCBiZSBkZWNvZGVkLgotICAgICAqIEBwYXJhbSBtYXhQYXNzCi0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggb2YgdGhlIGxhc3QgcGFzcyB0aGF0IHdpbGwgYmUgZGVjb2RlZC4KLSAgICAgKiBAcGFyYW0gbWluWAotICAgICAqICAgICAgICAgICAgdGhlIG1pbmltdW0gWCBjb29yZGluYXRlIG9mIHRoZSBwaXhlbHMgaW4gdGhlIHVwZGF0ZWQgYXJlYS4KLSAgICAgKiBAcGFyYW0gbWluWQotICAgICAqICAgICAgICAgICAgdGhlIG1pbmltdW0gWSBjb29yZGluYXRlIG9mIHRoZSBwaXhlbHMgaW4gdGhlIHVwZGF0ZWQgYXJlYS4KLSAgICAgKiBAcGFyYW0gcGVyaW9kWAotICAgICAqICAgICAgICAgICAgdGhlIGhvcml6b250YWwgc3BhY2luZyBwZXJpb2QgYmV0d2VlbiB1cGRhdGVkIHBpeGVscywgaWYgaXQKLSAgICAgKiAgICAgICAgICAgIGVxdWFscyAxLCB0aGVyZSBpcyBubyBzcGFjZSBiZXR3ZWVuIHBpeGVscy4KLSAgICAgKiBAcGFyYW0gcGVyaW9kWQotICAgICAqICAgICAgICAgICAgdGhlIHZlcnRpY2FsIHNwYWNpbmcgcGVyaW9kIGJldHdlZW4gdXBkYXRlZCBwaXhlbHMsIGlmIGl0Ci0gICAgICogICAgICAgICAgICBlcXVhbHMgMSwgdGhlcmUgaXMgbm8gc3BhY2UgYmV0d2VlbiBwaXhlbHMuCi0gICAgICogQHBhcmFtIGJhbmRzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2YgaW50ZWdlciB2YWx1ZXMgaW5kaWNhdGluZyB0aGUgYmFuZHMgYmVpbmcKLSAgICAgKiAgICAgICAgICAgIHVwZGF0ZWQuCi0gICAgICovCi0gICAgdm9pZCB0aHVtYm5haWxQYXNzU3RhcnRlZChJbWFnZVJlYWRlciBzb3VyY2UsIEJ1ZmZlcmVkSW1hZ2UgdGhlVGh1bWJuYWlsLCBpbnQgcGFzcywKLSAgICAgICAgICAgIGludCBtaW5QYXNzLCBpbnQgbWF4UGFzcywgaW50IG1pblgsIGludCBtaW5ZLCBpbnQgcGVyaW9kWCwgaW50IHBlcmlvZFksIGludFtdIGJhbmRzKTsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgdGhhdCBhIHNwZWNpZmllZCBhcmVhIG9mIGEgdGh1bWJuYWlsIGltYWdlIGhhcwotICAgICAqIGJlZW4gdXBkYXRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VSZWFkZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqIEBwYXJhbSB0aGVUaHVtYm5haWwKLSAgICAgKiAgICAgICAgICAgIHRoZSB0aHVtYm5haWwgdG8gYmUgdXBkYXRlZC4KLSAgICAgKiBAcGFyYW0gbWluWAotICAgICAqICAgICAgICAgICAgdGhlIG1pbmltdW0gWCBjb29yZGluYXRlIG9mIHRoZSBwaXhlbHMgaW4gdGhlIHVwZGF0ZWQgYXJlYS4KLSAgICAgKiBAcGFyYW0gbWluWQotICAgICAqICAgICAgICAgICAgdGhlIG1pbmltdW0gWSBjb29yZGluYXRlIG9mIHRoZSBwaXhlbHMgaW4gdGhlIHVwZGF0ZWQgYXJlYS4KLSAgICAgKiBAcGFyYW0gd2lkdGgKLSAgICAgKiAgICAgICAgICAgIHRoZSB3aWR0aCBvZiB1cGRhdGVkIGFyZWEuCi0gICAgICogQHBhcmFtIGhlaWdodAotICAgICAqICAgICAgICAgICAgdGhlIGhlaWdodCBvZiB1cGRhdGVkIGFyZWEuCi0gICAgICogQHBhcmFtIHBlcmlvZFgKLSAgICAgKiAgICAgICAgICAgIHRoZSBob3Jpem9udGFsIHNwYWNpbmcgcGVyaW9kIGJldHdlZW4gdXBkYXRlZCBwaXhlbHMsIGlmIGl0Ci0gICAgICogICAgICAgICAgICBlcXVhbHMgMSwgdGhlcmUgaXMgbm8gc3BhY2UgYmV0d2VlbiBwaXhlbHMuCi0gICAgICogQHBhcmFtIHBlcmlvZFkKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZXJ0aWNhbCBzcGFjaW5nIHBlcmlvZCBiZXR3ZWVuIHVwZGF0ZWQgcGl4ZWxzLCBpZiBpdAotICAgICAqICAgICAgICAgICAgZXF1YWxzIDEsIHRoZXJlIGlzIG5vIHNwYWNlIGJldHdlZW4gcGl4ZWxzLgotICAgICAqIEBwYXJhbSBiYW5kcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIGludGVnZXIgdmFsdWVzIGluZGljYXRpbmcgdGhlIGJhbmRzIGJlaW5nCi0gICAgICogICAgICAgICAgICB1cGRhdGVkLgotICAgICAqLwotICAgIHZvaWQgdGh1bWJuYWlsVXBkYXRlKEltYWdlUmVhZGVyIHNvdXJjZSwgQnVmZmVyZWRJbWFnZSB0aGVUaHVtYm5haWwsIGludCBtaW5YLCBpbnQgbWluWSwKLSAgICAgICAgICAgIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IHBlcmlvZFgsIGludCBwZXJpb2RZLCBpbnRbXSBiYW5kcyk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9ldmVudC9JSU9SZWFkV2FybmluZ0xpc3RlbmVyLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9ldmVudC9JSU9SZWFkV2FybmluZ0xpc3RlbmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDMxOGE1ZGYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vZXZlbnQvSUlPUmVhZFdhcm5pbmdMaXN0ZW5lci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBTZXJnZXkgSS4gU2FsaXNoZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4yICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uZXZlbnQ7Ci0KLWltcG9ydCBqYXZhLnV0aWwuRXZlbnRMaXN0ZW5lcjsKLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlUmVhZGVyOwotCi0vKiAKLSAqIEBhdXRob3IgU2VyZ2V5IEkuIFNhbGlzaGV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0gKi8KLQotLyoqCi0gKiBUaGUgSUlPUmVhZFdhcm5pbmdMaXN0ZW5lciBwcm92aWRlcyBtZXRob2RzIHRvIHJlY2VpdmUgbm90aWZpY2F0aW9uIG9mCi0gKiB3YXJuaW5nIG1lc3NhZ2VzIGdlbmVyYXRlZCBieSBpbWFnZSBhbmQgdGh1bWJuYWlsIHJlYWRpbmcgbWV0aG9kcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgSUlPUmVhZFdhcm5pbmdMaXN0ZW5lciBleHRlbmRzIEV2ZW50TGlzdGVuZXIgewotCi0gICAgLyoqCi0gICAgICogTm90aWZpZXMgdGhpcyBsaXN0ZW5lciBhYm91dCBhIHdhcm5pbmcgKG5vbi1mYXRhbCBlcnJvcikgZHVyaW5nIGRlY29kaW5nLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVJlYWRlciBvYmplY3Qgd2hpY2ggY2FsbHMgdGhpcyBtZXRob2QuCi0gICAgICogQHBhcmFtIHdhcm5pbmcKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdHJpbmcgZGVzY3JpYmluZyB0aGUgd2FybmluZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB3YXJuaW5nT2NjdXJyZWQoSW1hZ2VSZWFkZXIgc291cmNlLCBTdHJpbmcgd2FybmluZyk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9ldmVudC9JSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL2V2ZW50L0lJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0YTJjNTk1Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL2V2ZW50L0lJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTAxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uZXZlbnQ7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlV3JpdGVyOwotaW1wb3J0IGphdmEudXRpbC5FdmVudExpc3RlbmVyOwotCi0vKioKLSAqIFRoZSBJSU9Xcml0ZVByb2dyZXNzTGlzdGVuZXIgaW50ZXJmYWNlIHByb3ZpZGVzIG1ldGhvZHMgdG8gcmVjZWl2ZQotICogbm90aWZpY2F0aW9uIGFib3V0IHRoZSBwcm9ncmVzcyBvZiB0aGUgaW1hZ2UgYW5kIHRodW1ibmFpbCB3cml0aW5nIG1ldGhvZHMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgaW50ZXJmYWNlIElJT1dyaXRlUHJvZ3Jlc3NMaXN0ZW5lciBleHRlbmRzIEV2ZW50TGlzdGVuZXIgewotCi0gICAgLyoqCi0gICAgICogTm90aWZpZXMgdGhpcyBsaXN0ZW5lciB0aGF0IGFuIGltYWdlIHdyaXRlIG9wZXJhdGlvbiBoYXMgYmVlbiBzdGFydGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVdyaXRlciBvYmplY3Qgd2hpY2ggY2FsbHMgdGhpcyBtZXRob2QuCi0gICAgICogQHBhcmFtIGltYWdlSW5kZXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbmRleCBvZiB0aGUgaW1hZ2UgYmVpbmcgd3JpdHRlbi4KLSAgICAgKi8KLSAgICB2b2lkIGltYWdlU3RhcnRlZChJbWFnZVdyaXRlciBzb3VyY2UsIGludCBpbWFnZUluZGV4KTsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgYWJvdXQgdGhlIGRlZ3JlZSBvZiBjb21wbGV0aW9uIG9mIHRoZSB3cml0ZSBjYWxsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzb3VyY2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVdyaXRlciBvYmplY3Qgd2hpY2ggY2FsbHMgdGhpcyBtZXRob2QuCi0gICAgICogQHBhcmFtIHBlcmNlbnRhZ2VEb25lCi0gICAgICogICAgICAgICAgICB0aGUgcGVyY2VudGFnZSBvZiBlbmNvZGluZyBkb25lLgotICAgICAqLwotICAgIHZvaWQgaW1hZ2VQcm9ncmVzcyhJbWFnZVdyaXRlciBzb3VyY2UsIGZsb2F0IHBlcmNlbnRhZ2VEb25lKTsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgdGhhdCB0aGUgaW1hZ2Ugd3JpdGluZyBoYXMgYmVlbiBjb21wbGV0ZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlV3JpdGVyIG9iamVjdCB3aGljaCBjYWxscyB0aGlzIG1ldGhvZC4KLSAgICAgKi8KLSAgICB2b2lkIGltYWdlQ29tcGxldGUoSW1hZ2VXcml0ZXIgc291cmNlKTsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgdGhhdCBhIHRodW1ibmFpbCB3cml0ZSBvcGVyYXRpb24gaGFzIGJlZW4gc3RhcnRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VXcml0ZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggb2YgdGhlIGltYWdlIGJlaW5nIHdyaXR0ZW4uCi0gICAgICogQHBhcmFtIHRodW1ibmFpbEluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggb2YgdGhlIHRodW1ibmFpbCBiZWluZyB3cml0dGVuLgotICAgICAqLwotICAgIHZvaWQgdGh1bWJuYWlsU3RhcnRlZChJbWFnZVdyaXRlciBzb3VyY2UsIGludCBpbWFnZUluZGV4LCBpbnQgdGh1bWJuYWlsSW5kZXgpOwotCi0gICAgLyoqCi0gICAgICogTm90aWZpZXMgdGhpcyBsaXN0ZW5lciBhYm91dCB0aGUgZGVncmVlIG9mIGNvbXBsZXRpb24gb2YgdGhlIHdyaXRlIGNhbGwuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlV3JpdGVyIG9iamVjdCB3aGljaCBjYWxscyB0aGlzIG1ldGhvZC4KLSAgICAgKiBAcGFyYW0gcGVyY2VudGFnZURvbmUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwZXJjZW50YWdlIG9mIGVuY29kaW5nIGRvbmUuCi0gICAgICovCi0gICAgdm9pZCB0aHVtYm5haWxQcm9ncmVzcyhJbWFnZVdyaXRlciBzb3VyY2UsIGZsb2F0IHBlcmNlbnRhZ2VEb25lKTsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgdGhhdCBhIHRodW1ibmFpbCB3cml0ZSBvcGVyYXRpb24gaGFzIGJlZW4KLSAgICAgKiBjb21wbGV0ZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlV3JpdGVyIG9iamVjdCB3aGljaCBjYWxscyB0aGlzIG1ldGhvZC4KLSAgICAgKi8KLSAgICB2b2lkIHRodW1ibmFpbENvbXBsZXRlKEltYWdlV3JpdGVyIHNvdXJjZSk7Ci0KLSAgICAvKioKLSAgICAgKiBOb3RpZmllcyB0aGlzIGxpc3RlbmVyIHRoYXQgd3JpdGluZyBvcGVyYXRpb24gaGFzIGJlZW4gYWJvcnRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VXcml0ZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqLwotICAgIHZvaWQgd3JpdGVBYm9ydGVkKEltYWdlV3JpdGVyIHNvdXJjZSk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9ldmVudC9JSU9Xcml0ZVdhcm5pbmdMaXN0ZW5lci5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vZXZlbnQvSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOGVlNDFjZC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9ldmVudC9JSU9Xcml0ZVdhcm5pbmdMaXN0ZW5lci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5ldmVudDsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VXcml0ZXI7Ci1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci0KLS8qKgotICogVGhlIElJT1dyaXRlV2FybmluZ0xpc3RlbmVyIHByb3ZpZGVzIG1ldGhvZHMgdG8gcmVjZWl2ZSBub3RpZmljYXRpb24gb2YKLSAqIHdhcm5pbmdzIGdlbmVyYXRlZCBieSBpbWFnZSBhbmQgdGh1bWJuYWlsIHdyaXRpbmcgbWV0aG9kcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgSUlPV3JpdGVXYXJuaW5nTGlzdGVuZXIgZXh0ZW5kcyBFdmVudExpc3RlbmVyIHsKLQotICAgIC8qKgotICAgICAqIE5vdGlmaWVzIHRoaXMgbGlzdGVuZXIgYWJvdXQgYSB3YXJuaW5nIChub24tZmF0YWwgZXJyb3IpIGR1cmluZyBlbmNvZGluZy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc291cmNlCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VXcml0ZXIgb2JqZWN0IHdoaWNoIGNhbGxzIHRoaXMgbWV0aG9kLgotICAgICAqIEBwYXJhbSBpbWFnZUluZGV4Ci0gICAgICogICAgICAgICAgICB0aGUgaW5kZXggb2YgdGhlIGltYWdlIGdlbmVyYXRpbmcgdGhlIHdhcm5pbmcuCi0gICAgICogQHBhcmFtIHdhcm5pbmcKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdHJpbmcgZGVzY3JpYmluZyB0aGUgd2FybmluZy4KLSAgICAgKi8KLSAgICB2b2lkIHdhcm5pbmdPY2N1cnJlZChJbWFnZVdyaXRlciBzb3VyY2UsIGludCBpbWFnZUluZGV4LCBTdHJpbmcgd2FybmluZyk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9ldmVudC9wYWNrYWdlLmh0bWwgYi9hd3QvamF2YXgvaW1hZ2Vpby9ldmVudC9wYWNrYWdlLmh0bWwKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGMyZmUzOWYuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vZXZlbnQvcGFja2FnZS5odG1sCisrKyAvZGV2L251bGwKQEAgLTEsOCArMCwwIEBACi08aHRtbD4KLSAgPGJvZHk+Ci0gICAgPHA+Ci0gICAgICBUaGlzIHBhY2thZ2UgcHJvdmlkZXMgaW50ZXJmYWNlcyB0byBoYW5kbGUgZXZlbnRzIHdoaWNoIGNhbiBiZSBmaXJlZCBkdXJpbmcgdGhlIHJlYWRpbmcgb3Igd3JpdGluZyBvZiBpbWFnZXMuCi0gICAgPC9wPgotICBAc2luY2UgQW5kcm9pZCAxLjAKLSAgPC9ib2R5PgotPC9odG1sPgpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vbWV0YWRhdGEvSUlPSW52YWxpZFRyZWVFeGNlcHRpb24uamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL21ldGFkYXRhL0lJT0ludmFsaWRUcmVlRXhjZXB0aW9uLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGJhOTA2NTcuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vbWV0YWRhdGEvSUlPSW52YWxpZFRyZWVFeGNlcHRpb24uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDc0ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5tZXRhZGF0YTsKLQotaW1wb3J0IG9yZy53M2MuZG9tLk5vZGU7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5JSU9FeGNlcHRpb247Ci0KLS8qKgotICogVGhlIElJT0ludmFsaWRUcmVlRXhjZXB0aW9uIHByb3ZpZGVzIG5vdGlmaWNhdGlvbiBhYm91dCBmYWlscyBvZgotICogSUlPTWV0YWRhdGFOb2RlcyB0cmVlIHBhcnNpbmcgYnkgSUlPTWV0YWRhdGEgb2JqZWN0LgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIElJT0ludmFsaWRUcmVlRXhjZXB0aW9uIGV4dGVuZHMgSUlPRXhjZXB0aW9uIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBvZmZlbmRpbmcgbm9kZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgTm9kZSBvZmZlbmRpbmdOb2RlID0gbnVsbDsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhbiBJSU9JbnZhbGlkVHJlZUV4Y2VwdGlvbiB3aXRoIHRoZSBzcGVjaWZpZWQgZGV0YWlsZWQKLSAgICAgKiBtZXNzYWdlIGFuZCBzcGVjaWZpZWQgb2ZmZW5kaW5nIE5vZGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1lc3NhZ2UKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZXRhaWxlZCBtZXNzYWdlLgotICAgICAqIEBwYXJhbSBvZmZlbmRpbmdOb2RlCi0gICAgICogICAgICAgICAgICB0aGUgb2ZmZW5kaW5nIG5vZGUuCi0gICAgICovCi0gICAgcHVibGljIElJT0ludmFsaWRUcmVlRXhjZXB0aW9uKFN0cmluZyBtZXNzYWdlLCBOb2RlIG9mZmVuZGluZ05vZGUpIHsKLSAgICAgICAgc3VwZXIobWVzc2FnZSk7Ci0gICAgICAgIHRoaXMub2ZmZW5kaW5nTm9kZSA9IG9mZmVuZGluZ05vZGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IElJT0ludmFsaWRUcmVlRXhjZXB0aW9uIHdpdGggdGhlIHNwZWNpZmllZCBkZXRhaWxlZAotICAgICAqIG1lc3NhZ2UgYW5kIHNwZWNpZmllZCBvZmZlbmRpbmcgTm9kZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbWVzc2FnZQotICAgICAqICAgICAgICAgICAgdGhlIGRldGFpbGVkIG1lc3NhZ2UuCi0gICAgICogQHBhcmFtIGNhdXNlCi0gICAgICogICAgICAgICAgICB0aGUgY2F1c2Ugb2YgdGhpcyBleGNlcHRpb24uCi0gICAgICogQHBhcmFtIG9mZmVuZGluZ05vZGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZlbmRpbmcgbm9kZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgSUlPSW52YWxpZFRyZWVFeGNlcHRpb24oU3RyaW5nIG1lc3NhZ2UsIFRocm93YWJsZSBjYXVzZSwgTm9kZSBvZmZlbmRpbmdOb2RlKSB7Ci0gICAgICAgIHN1cGVyKG1lc3NhZ2UsIGNhdXNlKTsKLSAgICAgICAgdGhpcy5vZmZlbmRpbmdOb2RlID0gb2ZmZW5kaW5nTm9kZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBvZmZlbmRpbmcgbm9kZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBvZmZlbmRpbmcgbm9kZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgTm9kZSBnZXRPZmZlbmRpbmdOb2RlKCkgewotICAgICAgICByZXR1cm4gb2ZmZW5kaW5nTm9kZTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vbWV0YWRhdGEvSUlPTWV0YWRhdGEuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTZjZWJmOS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzkxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5tZXRhZGF0YTsKLQotaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLm1ldGFkYXRhLklJT01ldGFkYXRhVXRpbHM7Ci1pbXBvcnQgb3JnLnczYy5kb20uTm9kZTsKLQotLyoqCi0gKiBUaGUgY2xhc3MgSUlPTWV0YWRhdGEgcmVwcmVzZW50cyB0aGUgbWV0YWRhdGEgKGJ1bmRsZWQgd2l0aCBhbiBpbWFnZSkgYXMgYQotICogRG9tLXR5cGUgdHJlZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBJSU9NZXRhZGF0YSB7Ci0KLSAgICAvKioKLSAgICAgKiBXaGV0aGVyIHRoZSBzdGFuZGFyZCBtZXRhZGF0YSBmb3JtYXQgaXMgc3VwcG9ydGVkLgotICAgICAqLwotICAgIHByb3RlY3RlZCBib29sZWFuIHN0YW5kYXJkRm9ybWF0U3VwcG9ydGVkOwotCi0gICAgLyoqCi0gICAgICogVGhlIG5hdGl2ZSBtZXRhZGF0YSBmb3JtYXQgbmFtZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgU3RyaW5nIG5hdGl2ZU1ldGFkYXRhRm9ybWF0TmFtZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBuYXRpdmUgbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZyBuYXRpdmVNZXRhZGF0YUZvcm1hdENsYXNzTmFtZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBleHRyYSBtZXRhZGF0YSBmb3JtYXQgbmFtZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZ1tdIGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lczsKLQotICAgIC8qKgotICAgICAqIFRoZSBleHRyYSBtZXRhZGF0YSBmb3JtYXQgY2xhc3MgbmFtZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZ1tdIGV4dHJhTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzOwotCi0gICAgLyoqCi0gICAgICogVGhlIGRlZmF1bHQgY29udHJvbGxlci4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgSUlPTWV0YWRhdGFDb250cm9sbGVyIGRlZmF1bHRDb250cm9sbGVyOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNvbnRyb2xsZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIElJT01ldGFkYXRhQ29udHJvbGxlciBjb250cm9sbGVyOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IElJT01ldGFkYXRhIHdpdGggbm8gZGF0YSBzZXQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIElJT01ldGFkYXRhKCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJSU9NZXRhZGF0YSB3aXRoIHRoZSBzcGVjaWZpZWQgZGF0YSBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdGFuZGFyZE1ldGFkYXRhRm9ybWF0U3VwcG9ydGVkCi0gICAgICogICAgICAgICAgICB3aGV0aGVyIHRoZSBzdGFuZGFyZCBtZXRhZGF0YSBmb3JtYXQgaXMgc3VwcG9ydGVkLgotICAgICAqIEBwYXJhbSBuYXRpdmVNZXRhZGF0YUZvcm1hdE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYXRpdmUgbWV0YWRhdGEgZm9ybWF0IG5hbWUuCi0gICAgICogQHBhcmFtIG5hdGl2ZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmF0aXZlIG1ldGFkYXRhIGZvcm1hdCBjbGFzcyBuYW1lLgotICAgICAqIEBwYXJhbSBleHRyYU1ldGFkYXRhRm9ybWF0TmFtZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBleHRyYSBtZXRhZGF0YSBmb3JtYXQgbmFtZXMuCi0gICAgICogQHBhcmFtIGV4dHJhTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzCi0gICAgICogICAgICAgICAgICB0aGUgZXh0cmEgbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBJSU9NZXRhZGF0YShib29sZWFuIHN0YW5kYXJkTWV0YWRhdGFGb3JtYXRTdXBwb3J0ZWQsIFN0cmluZyBuYXRpdmVNZXRhZGF0YUZvcm1hdE5hbWUsCi0gICAgICAgICAgICBTdHJpbmcgbmF0aXZlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsIFN0cmluZ1tdIGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcywKLSAgICAgICAgICAgIFN0cmluZ1tdIGV4dHJhTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzKSB7Ci0gICAgICAgIHN0YW5kYXJkRm9ybWF0U3VwcG9ydGVkID0gc3RhbmRhcmRNZXRhZGF0YUZvcm1hdFN1cHBvcnRlZDsKLSAgICAgICAgdGhpcy5uYXRpdmVNZXRhZGF0YUZvcm1hdE5hbWUgPSBuYXRpdmVNZXRhZGF0YUZvcm1hdE5hbWU7Ci0gICAgICAgIHRoaXMubmF0aXZlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUgPSBuYXRpdmVNZXRhZGF0YUZvcm1hdENsYXNzTmFtZTsKLSAgICAgICAgaWYgKGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICBpZiAoZXh0cmFNZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oCi0gICAgICAgICAgICAgICAgICAgICAgICAiZXh0cmFNZXRhZGF0YUZvcm1hdE5hbWVzID09IG51bGwgJiYgZXh0cmFNZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMgIT0gbnVsbCEiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGlmIChleHRyYU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigKLSAgICAgICAgICAgICAgICAgICAgICAgICJleHRyYU1ldGFkYXRhRm9ybWF0TmFtZXMgIT0gbnVsbCAmJiBleHRyYU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcyA9PSBudWxsISIpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcy5sZW5ndGggPT0gMCkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcy5sZW5ndGggPT0gMCEiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChleHRyYU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcy5sZW5ndGggIT0gZXh0cmFNZXRhZGF0YUZvcm1hdE5hbWVzLmxlbmd0aCkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oCi0gICAgICAgICAgICAgICAgICAgICAgICAiZXh0cmFNZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMubGVuZ3RoICE9IGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcy5sZW5ndGghIik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB0aGlzLmV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcyA9IGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcy5jbG9uZSgpOwotICAgICAgICAgICAgdGhpcy5leHRyYU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcyA9IGV4dHJhTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzLmNsb25lKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtZXRhZGF0YSBhcyB0cmVlLXR5cGUgZG9jdW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZvcm1hdE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBmb3JtYXQgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBub2RlIGluIHRyZWUgZm9ybWF0LgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBOb2RlIGdldEFzVHJlZShTdHJpbmcgZm9ybWF0TmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIG1ldGFkYXRhIGlzIHJlYWQgb25seS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBtZXRhZGF0YSBpcyByZWFkIG9ubHkuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGJvb2xlYW4gaXNSZWFkT25seSgpOwotCi0gICAgLyoqCi0gICAgICogTWVyZ2VzIHRoZSBzcGVjaWZpZWQgdHJlZSB3aXRoIHRoaXMgbWV0YWRhdGEgdHJlZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZm9ybWF0TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGZvcm1hdCBvZiB0aGUgc3BlY2lmaWVkIHRyZWUuCi0gICAgICogQHBhcmFtIHJvb3QKLSAgICAgKiAgICAgICAgICAgIHRoZSByb290IG5vZGUgb2YgdGhlIG1ldGFkYXRhIHRyZWUuCi0gICAgICogQHRocm93cyBJSU9JbnZhbGlkVHJlZUV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIHRoZSBzcGVjaWZpZWQgdHJlZSBpcyBpbmNvbXBhdGlibGUgd2l0aCB0aGUgdGhpcyBtZXRhZGF0YQotICAgICAqICAgICAgICAgICAgIHRyZWUuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgbWVyZ2VUcmVlKFN0cmluZyBmb3JtYXROYW1lLCBOb2RlIHJvb3QpIHRocm93cyBJSU9JbnZhbGlkVHJlZUV4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFJlc2V0cyB0aGUgY29udHJvbGxlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCByZXNldCgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29udHJvbGxlciBhc3NvY2lhdGVkIHdpdGggdGhpcyBtZXRhZGF0YSBkb2N1bWVudC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjb250cm9sbGVyLgotICAgICAqLwotICAgIHB1YmxpYyBJSU9NZXRhZGF0YUNvbnRyb2xsZXIgZ2V0Q29udHJvbGxlcigpIHsKLSAgICAgICAgcmV0dXJuIGNvbnRyb2xsZXI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgdGhpcyBtZXRhZGF0YSBoYXMgYSBjb250cm9sbGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBtZXRhZGF0YSBoYXMgYSBjb250cm9sbGVyLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGhhc0NvbnRyb2xsZXIoKSB7Ci0gICAgICAgIHJldHVybiBnZXRDb250cm9sbGVyKCkgIT0gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBY3RpdmF0ZSB0aGUgY29udHJvbGxlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN1Y2Nlc3NmdWwuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gYWN0aXZhdGVDb250cm9sbGVyKCkgewotICAgICAgICBpZiAoIWhhc0NvbnRyb2xsZXIoKSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxTdGF0ZUV4Y2VwdGlvbigiaGFzQ29udHJvbGxlcigpID09IGZhbHNlISIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBnZXRDb250cm9sbGVyKCkuYWN0aXZhdGUodGhpcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZGVmYXVsdCBjb250cm9sbGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRlZmF1bHQgY29udHJvbGxlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgSUlPTWV0YWRhdGFDb250cm9sbGVyIGdldERlZmF1bHRDb250cm9sbGVyKCkgewotICAgICAgICByZXR1cm4gZGVmYXVsdENvbnRyb2xsZXI7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgZXh0cmEgbWV0YWRhdGEgZm9ybWF0IG5hbWVzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGV4dHJhIG1ldGFkYXRhIGZvcm1hdCBuYW1lcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0RXh0cmFNZXRhZGF0YUZvcm1hdE5hbWVzKCkgewotICAgICAgICByZXR1cm4gZXh0cmFNZXRhZGF0YUZvcm1hdE5hbWVzID09IG51bGwgPyBudWxsIDogZXh0cmFNZXRhZGF0YUZvcm1hdE5hbWVzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWV0YWRhdGEgZm9ybWF0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZm9ybWF0IG5hbWUuCi0gICAgICogQHJldHVybiB0aGUgbWV0YWRhdGEgZm9ybWF0LgotICAgICAqLwotICAgIHB1YmxpYyBJSU9NZXRhZGF0YUZvcm1hdCBnZXRNZXRhZGF0YUZvcm1hdChTdHJpbmcgZm9ybWF0TmFtZSkgewotICAgICAgICByZXR1cm4gSUlPTWV0YWRhdGFVdGlscy5pbnN0YW50aWF0ZU1ldGFkYXRhRm9ybWF0KGZvcm1hdE5hbWUsIHN0YW5kYXJkRm9ybWF0U3VwcG9ydGVkLAotICAgICAgICAgICAgICAgIG5hdGl2ZU1ldGFkYXRhRm9ybWF0TmFtZSwgbmF0aXZlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsIGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcywKLSAgICAgICAgICAgICAgICBleHRyYU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbmF0aXZlIG1ldGFkYXRhIGZvcm1hdCBuYW1lLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG5hdGl2ZSBtZXRhZGF0YSBmb3JtYXQgbmFtZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldE5hdGl2ZU1ldGFkYXRhRm9ybWF0TmFtZSgpIHsKLSAgICAgICAgcmV0dXJuIG5hdGl2ZU1ldGFkYXRhRm9ybWF0TmFtZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgaWYgdGhlIHN0YW5kYXJkIG1ldGFkYXRhIGZvcm1hdCBpcyBzdXBwb3J0ZWQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3RhbmRhcmQgbWV0YWRhdGEgZm9ybWF0IGlzIHN1cHBvcnRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1N0YW5kYXJkTWV0YWRhdGFGb3JtYXRTdXBwb3J0ZWQoKSB7Ci0gICAgICAgIHJldHVybiBzdGFuZGFyZEZvcm1hdFN1cHBvcnRlZDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtZXRhZGF0YSBmb3JtYXQgbmFtZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbWV0YWRhdGEgZm9ybWF0IG5hbWVzLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRNZXRhZGF0YUZvcm1hdE5hbWVzKCkgewotICAgICAgICBBcnJheUxpc3Q8U3RyaW5nPiByZXMgPSBuZXcgQXJyYXlMaXN0PFN0cmluZz4oKTsKLQotICAgICAgICBTdHJpbmcgbmF0aXZlTWV0YWRhdGFGb3JtYXROYW1lID0gZ2V0TmF0aXZlTWV0YWRhdGFGb3JtYXROYW1lKCk7Ci0gICAgICAgIGJvb2xlYW4gc3RhbmRhcmRGb3JtYXRTdXBwb3J0ZWQgPSBpc1N0YW5kYXJkTWV0YWRhdGFGb3JtYXRTdXBwb3J0ZWQoKTsKLSAgICAgICAgU3RyaW5nIGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lc1tdID0gZ2V0RXh0cmFNZXRhZGF0YUZvcm1hdE5hbWVzKCk7Ci0KLSAgICAgICAgaWYgKHN0YW5kYXJkRm9ybWF0U3VwcG9ydGVkKSB7Ci0gICAgICAgICAgICByZXMuYWRkKElJT01ldGFkYXRhRm9ybWF0SW1wbC5zdGFuZGFyZE1ldGFkYXRhRm9ybWF0TmFtZSk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG5hdGl2ZU1ldGFkYXRhRm9ybWF0TmFtZSAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXMuYWRkKG5hdGl2ZU1ldGFkYXRhRm9ybWF0TmFtZSk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICBmb3IgKFN0cmluZyBleHRyYU1ldGFkYXRhRm9ybWF0TmFtZSA6IGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcykgewotICAgICAgICAgICAgICAgIHJlcy5hZGQoZXh0cmFNZXRhZGF0YUZvcm1hdE5hbWUpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHJlcy5zaXplKCkgPiAwID8gcmVzLnRvQXJyYXkobmV3IFN0cmluZ1swXSkgOiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN0YW5kYXJkIGNocm9tYSBub2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0YW5kYXJkIGNocm9tYSBub2RlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBJSU9NZXRhZGF0YU5vZGUgZ2V0U3RhbmRhcmRDaHJvbWFOb2RlKCkgewotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzdGFuZGFyZCBjb21wcmVzc2lvbiBub2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0YW5kYXJkIGNvbXByZXNzaW9uIG5vZGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIElJT01ldGFkYXRhTm9kZSBnZXRTdGFuZGFyZENvbXByZXNzaW9uTm9kZSgpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc3RhbmRhcmQgZGF0YSBub2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0YW5kYXJkIGRhdGEgbm9kZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgSUlPTWV0YWRhdGFOb2RlIGdldFN0YW5kYXJkRGF0YU5vZGUoKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN0YW5kYXJkIGRpbWVuc2lvbiBub2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0YW5kYXJkIGRpbWVuc2lvbiBub2RlLgotICAgICAqLwotICAgIHByb3RlY3RlZCBJSU9NZXRhZGF0YU5vZGUgZ2V0U3RhbmRhcmREaW1lbnNpb25Ob2RlKCkgewotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzdGFuZGFyZCBkb2N1bWVudCBub2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0YW5kYXJkIGRvY3VtZW50IG5vZGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIElJT01ldGFkYXRhTm9kZSBnZXRTdGFuZGFyZERvY3VtZW50Tm9kZSgpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgc3RhbmRhcmQgdGV4dCBub2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0YW5kYXJkIHRleHQgbm9kZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgSUlPTWV0YWRhdGFOb2RlIGdldFN0YW5kYXJkVGV4dE5vZGUoKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN0YW5kYXJkIHRpbGUgbm9kZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdGFuZGFyZCB0aWxlIG5vZGUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIElJT01ldGFkYXRhTm9kZSBnZXRTdGFuZGFyZFRpbGVOb2RlKCkgewotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzdGFuZGFyZCB0cmFuc3BhcmVuY3kgbm9kZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdGFuZGFyZCB0cmFuc3BhcmVuY3kgbm9kZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgSUlPTWV0YWRhdGFOb2RlIGdldFN0YW5kYXJkVHJhbnNwYXJlbmN5Tm9kZSgpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWV0YWRhdGEgYXMgYSB0cmVlIGluIHN0YW5kYXJkIGZvcm1hdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBtZXRhZGF0YSBhcyBhIHRyZWUgaW4gc3RhbmRhcmQgZm9ybWF0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBmaW5hbCBJSU9NZXRhZGF0YU5vZGUgZ2V0U3RhbmRhcmRUcmVlKCkgewotICAgICAgICAvLyBDcmVhdGUgcm9vdCBub2RlCi0gICAgICAgIElJT01ldGFkYXRhTm9kZSByb290ID0gbmV3IElJT01ldGFkYXRhTm9kZShJSU9NZXRhZGF0YUZvcm1hdEltcGwuc3RhbmRhcmRNZXRhZGF0YUZvcm1hdE5hbWUpOwotCi0gICAgICAgIE5vZGUgbm9kZTsKLSAgICAgICAgaWYgKChub2RlID0gZ2V0U3RhbmRhcmRDaHJvbWFOb2RlKCkpICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJvb3QuYXBwZW5kQ2hpbGQobm9kZSk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChub2RlID0gZ2V0U3RhbmRhcmRDb21wcmVzc2lvbk5vZGUoKSkgIT0gbnVsbCkgewotICAgICAgICAgICAgcm9vdC5hcHBlbmRDaGlsZChub2RlKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoKG5vZGUgPSBnZXRTdGFuZGFyZERhdGFOb2RlKCkpICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJvb3QuYXBwZW5kQ2hpbGQobm9kZSk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChub2RlID0gZ2V0U3RhbmRhcmREaW1lbnNpb25Ob2RlKCkpICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJvb3QuYXBwZW5kQ2hpbGQobm9kZSk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChub2RlID0gZ2V0U3RhbmRhcmREb2N1bWVudE5vZGUoKSkgIT0gbnVsbCkgewotICAgICAgICAgICAgcm9vdC5hcHBlbmRDaGlsZChub2RlKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoKG5vZGUgPSBnZXRTdGFuZGFyZFRleHROb2RlKCkpICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJvb3QuYXBwZW5kQ2hpbGQobm9kZSk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKChub2RlID0gZ2V0U3RhbmRhcmRUaWxlTm9kZSgpKSAhPSBudWxsKSB7Ci0gICAgICAgICAgICByb290LmFwcGVuZENoaWxkKG5vZGUpOwotICAgICAgICB9Ci0gICAgICAgIGlmICgobm9kZSA9IGdldFN0YW5kYXJkVHJhbnNwYXJlbmN5Tm9kZSgpKSAhPSBudWxsKSB7Ci0gICAgICAgICAgICByb290LmFwcGVuZENoaWxkKG5vZGUpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHJvb3Q7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgY29udHJvbGxlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29udHJvbGxlcgotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBjb250cm9sbGVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldENvbnRyb2xsZXIoSUlPTWV0YWRhdGFDb250cm9sbGVyIGNvbnRyb2xsZXIpIHsKLSAgICAgICAgdGhpcy5jb250cm9sbGVyID0gY29udHJvbGxlcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBmcm9tIHRyZWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZvcm1hdE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYW1lIG9mIHRoZSBtZXRhdGRhdGEgZm9ybWF0IG9mIHRoZSBmcm9tIHRyZWUuCi0gICAgICogQHBhcmFtIHJvb3QKLSAgICAgKiAgICAgICAgICAgIHRoZSByb290IG5vZGUgb2YgdGhlIGZyb20gdHJlZS4KLSAgICAgKiBAdGhyb3dzIElJT0ludmFsaWRUcmVlRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHRyZWUgb3IgaXRzIGZvcm1hdCBpcyBub3QgY29tcGF0aWJsZSB3aXRoIHRoaXMKLSAgICAgKiAgICAgICAgICAgICBtZXRhZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRGcm9tVHJlZShTdHJpbmcgZm9ybWF0TmFtZSwgTm9kZSByb290KSB0aHJvd3MgSUlPSW52YWxpZFRyZWVFeGNlcHRpb24gewotICAgICAgICByZXNldCgpOwotICAgICAgICBtZXJnZVRyZWUoZm9ybWF0TmFtZSwgcm9vdCk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vbWV0YWRhdGEvSUlPTWV0YWRhdGFDb250cm9sbGVyLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YUNvbnRyb2xsZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTQwNTk0OC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YUNvbnRyb2xsZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgU2VyZ2V5IEkuIFNhbGlzaGV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvLm1ldGFkYXRhOwotCi0vKiAKLSAqIEBhdXRob3IgU2VyZ2V5IEkuIFNhbGlzaGV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0gKi8KLQotLyoqCi0gKiBUaGUgSUlPTWV0YWRhdGFDb250cm9sbGVyIGludGVyZmFjZSBwcm92aWRlcyBhIG1ldGhvZCBmb3IgaW1wbGVtZW50aW5nCi0gKiBvYmplY3RzIHRvIGFjdGl2YXRlIHRoZSBjb250cm9sbGVyIHdpdGhvdXQgZGVmaW5pbmcgaG93IHRoZSBjb250cm9sbGVyCi0gKiBvYnRhaW5zIHZhbHVlcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgSUlPTWV0YWRhdGFDb250cm9sbGVyIHsKLQotICAgIC8qKgotICAgICAqIEFjdGl2YXRlcyBhIGNvbnRyb2xsZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1ldGFkYXRhCi0gICAgICogICAgICAgICAgICB0aGUgbWV0YWRhdGEgdG8gYmUgbW9kaWZpZWQuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgSUlPTWV0YWRhdGEgaGFzIGJlZW4gbW9kaWZpZWQsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBhY3RpdmF0ZShJSU9NZXRhZGF0YSBtZXRhZGF0YSk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YUZvcm1hdC5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vbWV0YWRhdGEvSUlPTWV0YWRhdGFGb3JtYXQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMGU3ZTY5Ny4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YUZvcm1hdC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDA0ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5tZXRhZGF0YTsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VUeXBlU3BlY2lmaWVyOwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci0KLS8qKgotICogVGhlIEludGVyZmFjZSBJSU9NZXRhZGF0YUZvcm1hdCBpcyBpbXBsZW1lbnRlZCBieSBjbGFzc2VzIHRoYXQgZGVzY3JpYmUgdGhlCi0gKiBydWxlcyBhbmQgYWxsb3dlZCBlbGVtZW50cyBmb3IgYSBtZXRhZGF0YSBkb2N1bWVudCB0cmVlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJSU9NZXRhZGF0YUZvcm1hdCB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ0hJTERfUE9MSUNZX0VNUFRZLgotICAgICAqLwotICAgIGludCBDSElMRF9QT0xJQ1lfRU1QVFkgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIENISUxEX1BPTElDWV9BTEwuCi0gICAgICovCi0gICAgaW50IENISUxEX1BPTElDWV9BTEwgPSAxOwotCi0gICAgLyoqCi0gICAgICogVGhlIENISUxEX1BPTElDWV9TT01FLgotICAgICAqLwotICAgIGludCBDSElMRF9QT0xJQ1lfU09NRSA9IDI7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ0hJTERfUE9MSUNZX0NIT0lDRS4KLSAgICAgKi8KLSAgICBpbnQgQ0hJTERfUE9MSUNZX0NIT0lDRSA9IDM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ0hJTERfUE9MSUNZX1NFUVVFTkNFLgotICAgICAqLwotICAgIGludCBDSElMRF9QT0xJQ1lfU0VRVUVOQ0UgPSA0OwotCi0gICAgLyoqCi0gICAgICogVGhlIENISUxEX1BPTElDWV9SRVBFQVQuCi0gICAgICovCi0gICAgaW50IENISUxEX1BPTElDWV9SRVBFQVQgPSA1OwotCi0gICAgLyoqCi0gICAgICogVGhlIG1heGltdW0gdmFsdWUgZm9yIHRoZSBjaGlsZCBwb2xpY3kuCi0gICAgICovCi0gICAgaW50IENISUxEX1BPTElDWV9NQVggPSBDSElMRF9QT0xJQ1lfUkVQRUFUOwotCi0gICAgLyoqCi0gICAgICogVGhlIERBVEFUWVBFX1NUUklORy4KLSAgICAgKi8KLSAgICBpbnQgREFUQVRZUEVfU1RSSU5HID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBEQVRBVFlQRV9CT09MRUFOLgotICAgICAqLwotICAgIGludCBEQVRBVFlQRV9CT09MRUFOID0gMTsKLQotICAgIC8qKgotICAgICAqIFRoZSBEQVRBVFlQRV9JTlRFR0VSLgotICAgICAqLwotICAgIGludCBEQVRBVFlQRV9JTlRFR0VSID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBEQVRBVFlQRV9GTE9BVC4KLSAgICAgKi8KLSAgICBpbnQgREFUQVRZUEVfRkxPQVQgPSAzOwotCi0gICAgLyoqCi0gICAgICogVGhlIERBVEFUWVBFX0RPVUJMRS4KLSAgICAgKi8KLSAgICBpbnQgREFUQVRZUEVfRE9VQkxFID0gNDsKLQotICAgIC8qKgotICAgICAqIFRoZSBWQUxVRV9OT05FLgotICAgICAqLwotICAgIGludCBWQUxVRV9OT05FID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBWQUxVRV9BUkJJVFJBUlkuCi0gICAgICovCi0gICAgaW50IFZBTFVFX0FSQklUUkFSWSA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgVkFMVUVfUkFOR0UuCi0gICAgICovCi0gICAgaW50IFZBTFVFX1JBTkdFID0gMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBWQUxVRV9SQU5HRV9NSU5fSU5DTFVTSVZFX01BU0suCi0gICAgICovCi0gICAgaW50IFZBTFVFX1JBTkdFX01JTl9JTkNMVVNJVkVfTUFTSyA9IDQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgVkFMVUVfUkFOR0VfTUFYX0lOQ0xVU0lWRV9NQVNLLgotICAgICAqLwotICAgIGludCBWQUxVRV9SQU5HRV9NQVhfSU5DTFVTSVZFX01BU0sgPSA4OwotCi0gICAgLyoqCi0gICAgICogVGhlIFZBTFVFX0VOVU1FUkFUSU9OLgotICAgICAqLwotICAgIGludCBWQUxVRV9FTlVNRVJBVElPTiA9IDE2OwotCi0gICAgLyoqCi0gICAgICogVGhlIFZBTFVFX0xJU1QuCi0gICAgICovCi0gICAgaW50IFZBTFVFX0xJU1QgPSAzMjsKLQotICAgIC8qKgotICAgICAqIFRoZSBWQUxVRV9SQU5HRV9NSU5fSU5DTFVTSVZFLgotICAgICAqLwotICAgIGludCBWQUxVRV9SQU5HRV9NSU5fSU5DTFVTSVZFID0gVkFMVUVfUkFOR0UgfCBWQUxVRV9SQU5HRV9NSU5fSU5DTFVTSVZFX01BU0s7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgVkFMVUVfUkFOR0VfTUFYX0lOQ0xVU0lWRS4KLSAgICAgKi8KLSAgICBpbnQgVkFMVUVfUkFOR0VfTUFYX0lOQ0xVU0lWRSA9IFZBTFVFX1JBTkdFIHwgVkFMVUVfUkFOR0VfTUFYX0lOQ0xVU0lWRV9NQVNLOwotCi0gICAgLyoqCi0gICAgICogVGhlIFZBTFVFX1JBTkdFX01JTl9NQVhfSU5DTFVTSVZFLgotICAgICAqLwotICAgIGludCBWQUxVRV9SQU5HRV9NSU5fTUFYX0lOQ0xVU0lWRSA9IFZBTFVFX1JBTkdFIHwgVkFMVUVfUkFOR0VfTUlOX0lOQ0xVU0lWRV9NQVNLCi0gICAgICAgICAgICB8IFZBTFVFX1JBTkdFX01BWF9JTkNMVVNJVkVfTUFTSzsKLQotICAgIC8qKgotICAgICAqIFRlbGxzIHdoZXRoZXIgdGhlIHNwZWNpZmllZCBlbGVtZW50IGlzIGFsbG93ZWQgZm9yIHRoZSBzcGVjaWZpZWQgaW1hZ2UKLSAgICAgKiB0eXBlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcGFyYW0gaW1hZ2VUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgaW1hZ2UgdHlwZS4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgZWxlbWVudCBpcyBhbGxvd2VkIGZvciB0aGUgc3BlY2lmaWVkIGltYWdlCi0gICAgICogICAgICAgICB0eXBlLgotICAgICAqLwotICAgIGJvb2xlYW4gY2FuTm9kZUFwcGVhcihTdHJpbmcgZWxlbWVudE5hbWUsIEltYWdlVHlwZVNwZWNpZmllciBpbWFnZVR5cGUpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyBkYXRhIHR5cGUgb2YgdGhlIHNwZWNpZmllZCBhdHRyaWJ1dGUgb2YgdGhlIHNwZWNpZmllZCBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcGFyYW0gYXR0ck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBhdHRyaWJ1dGUgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBhdHRyaWJ1dGUncyBkYXRhIHR5cGUuCi0gICAgICovCi0gICAgaW50IGdldEF0dHJpYnV0ZURhdGFUeXBlKFN0cmluZyBlbGVtZW50TmFtZSwgU3RyaW5nIGF0dHJOYW1lKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGRlZmF1bHQgdmFsdWUgb2YgdGhlIHNwZWNpZmllZCBhdHRyaWJ1dGUgb2YgdGhlIHNwZWNpZmllZAotICAgICAqIGVsZW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEBwYXJhbSBhdHRyTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGF0dHJpYnV0ZSBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIGF0dHJpYnV0ZSdzIGRlZmF1bHQgdmFsdWUuCi0gICAgICovCi0gICAgU3RyaW5nIGdldEF0dHJpYnV0ZURlZmF1bHRWYWx1ZShTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB1c2VyLWZyaWVuZGx5IGRlc2NyaXB0aW9uIG9mIHRoZSBhdHRyaWJ1dGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEBwYXJhbSBhdHRyTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGF0dHJpYnV0ZSBuYW1lLgotICAgICAqIEBwYXJhbSBsb2NhbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBsb2NhbGUgZ2l2aW5nIHRoZSBkZXNpcmVkIGxhbmd1YWdlIGZvciB0aGUgZGVzY3JpcHRpb24uCi0gICAgICogQHJldHVybiB0aGUgYXR0cmlidXRlIGRlc2NyaXB0aW9uLgotICAgICAqLwotICAgIFN0cmluZyBnZXRBdHRyaWJ1dGVEZXNjcmlwdGlvbihTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSwgTG9jYWxlIGxvY2FsZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBhdHRyaWJ1dGUgZW51bWVyYXRpb25zLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcGFyYW0gYXR0ck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBhdHRyaWJ1dGUgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBhdHRyaWJ1dGUgZW51bWVyYXRpb25zLgotICAgICAqLwotICAgIFN0cmluZ1tdIGdldEF0dHJpYnV0ZUVudW1lcmF0aW9ucyhTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtYXhpbXVtIGxlbmd0aCBvZiB0aGUgYXR0cmlidXRlIGxpc3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEBwYXJhbSBhdHRyTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGF0dHJpYnV0ZSBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIG1heGltdW0gbGVuZ3RoIG9mIHRoZSBhdHRyaWJ1dGUgbGlzdC4KLSAgICAgKi8KLSAgICBpbnQgZ2V0QXR0cmlidXRlTGlzdE1heExlbmd0aChTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtaW5pbXVtIGxlbmd0aCBvZiB0aGUgYXR0cmlidXRlIGxpc3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEBwYXJhbSBhdHRyTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGF0dHJpYnV0ZSBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIG1pbmltdW0gbGVuZ3RoIG9mIHRoZSBhdHRyaWJ1dGUgbGlzdC4KLSAgICAgKi8KLSAgICBpbnQgZ2V0QXR0cmlidXRlTGlzdE1pbkxlbmd0aChTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtYXhpbXVtIHZhbHVlIGFsbG93ZWQgZm9yIHRoZSBhdHRyaWJ1dGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEBwYXJhbSBhdHRyTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGF0dHJpYnV0ZSBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIG1heGltdW0gdmFsdWUgYWxsb3dlZCBmb3IgdGhlIGF0dHJpYnV0ZS4KLSAgICAgKi8KLSAgICBTdHJpbmcgZ2V0QXR0cmlidXRlTWF4VmFsdWUoU3RyaW5nIGVsZW1lbnROYW1lLCBTdHJpbmcgYXR0ck5hbWUpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWluaW11bSB2YWx1ZSBhbGxvd2VkIGZvciB0aGUgYXR0cmlidXRlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcGFyYW0gYXR0ck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBhdHRyaWJ1dGUgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBtaW5pbXVtIHZhbHVlIGFsbG93ZWQgZm9yIHRoZSBhdHRyaWJ1dGUuCi0gICAgICovCi0gICAgU3RyaW5nIGdldEF0dHJpYnV0ZU1pblZhbHVlKFN0cmluZyBlbGVtZW50TmFtZSwgU3RyaW5nIGF0dHJOYW1lKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGF0dHJpYnV0ZSBuYW1lcyBhbGxvd2VkIGZvciB0aGUgc3BlY2lmaWVkIGVsZW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIGF0dHJpYnV0ZSBuYW1lcy4KLSAgICAgKi8KLSAgICBTdHJpbmdbXSBnZXRBdHRyaWJ1dGVOYW1lcyhTdHJpbmcgZWxlbWVudE5hbWUpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgYXR0cmlidXRlIHZhbHVlIHR5cGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEBwYXJhbSBhdHRyTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGF0dHJpYnV0ZSBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIGF0dHJpYnV0ZSB2YWx1ZSB0eXBlLgotICAgICAqLwotICAgIGludCBnZXRBdHRyaWJ1dGVWYWx1ZVR5cGUoU3RyaW5nIGVsZW1lbnROYW1lLCBTdHJpbmcgYXR0ck5hbWUpOwotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgdGhlIHNwZWNpZmllZCBhdHRyaWJ1dGUgaXMgcmVxdWlyZWQgZm9yIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcGFyYW0gYXR0ck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBhdHRyaWJ1dGUgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgYXR0cmlidXRlIGlzIHJlcXVpcmVkIGZvciB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBlbGVtZW50LgotICAgICAqLwotICAgIGJvb2xlYW4gaXNBdHRyaWJ1dGVSZXF1aXJlZChTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBuYW1lcyBvZiB0aGUgcG9zc2libGUgY2hpbGQgZWxlbWVudHMgZm9yIHRoZSBnaXZlbiBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBjaGlsZCBuYW1lcy4KLSAgICAgKi8KLSAgICBTdHJpbmdbXSBnZXRDaGlsZE5hbWVzKFN0cmluZyBlbGVtZW50TmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBjb25zdGFudCBkZXNjcmliaW5nIHRoZSBlbGVtZW50J3MgY2hpbGQgcG9saWN5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBjaGlsZCBwb2xpY3kuCi0gICAgICovCi0gICAgaW50IGdldENoaWxkUG9saWN5KFN0cmluZyBlbGVtZW50TmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB1c2VyLWZyaWVuZGx5IGRlc2NyaXB0aW9uIG9mIHRoZSBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcGFyYW0gbG9jYWxlCi0gICAgICogICAgICAgICAgICB0aGUgbG9jYWxlIGdpdmluZyB0aGUgZGVzaXJlZCBsYW5ndWFnZSBmb3IgdGhlIGRlc2NyaXB0aW9uLgotICAgICAqIEByZXR1cm4gdGhlIGVsZW1lbnQgZGVzY3JpcHRpb24uCi0gICAgICovCi0gICAgU3RyaW5nIGdldEVsZW1lbnREZXNjcmlwdGlvbihTdHJpbmcgZWxlbWVudE5hbWUsIExvY2FsZSBsb2NhbGUpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgbWF4aW11bSBudW1iZXIgb2YgY2hpbGRyZW4gYWxsb3dlZCBmb3IgdGhlIGVsZW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIG1heGltdW0gbnVtYmVyIG9mIGNoaWxkcmVuIGFsbG93ZWQgZm9yIHRoZSBlbGVtZW50LgotICAgICAqLwotICAgIGludCBnZXRFbGVtZW50TWF4Q2hpbGRyZW4oU3RyaW5nIGVsZW1lbnROYW1lKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1pbmltdW0gbnVtYmVyIG9mIGNoaWxkcmVuIGFsbG93ZWQgZm9yIHRoZSBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBtaW5pbXVtIG51bWJlciBvZiBjaGlsZHJlbiBhbGxvd2VkIGZvciB0aGUgZWxlbWVudC4KLSAgICAgKi8KLSAgICBpbnQgZ2V0RWxlbWVudE1pbkNoaWxkcmVuKFN0cmluZyBlbGVtZW50TmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtYXhpbXVtIG9iamVjdCBhcnJheSBsZW5ndGggYWxsb3dlZCBmb3IgdGhlIGVsZW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIG1heGltdW0gb2JqZWN0IGFycmF5IGxlbmd0aCBhbGxvd2VkIGZvciB0aGUgZWxlbWVudC4KLSAgICAgKi8KLSAgICBpbnQgZ2V0T2JqZWN0QXJyYXlNYXhMZW5ndGgoU3RyaW5nIGVsZW1lbnROYW1lKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1pbmltdW0gb2JqZWN0IGFycmF5IGxlbmd0aCBhbGxvd2VkIGZvciB0aGUgZWxlbWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZWxlbWVudE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbGVtZW50IG5hbWUuCi0gICAgICogQHJldHVybiB0aGUgbWluaW11bSBvYmplY3QgYXJyYXkgbGVuZ3RoIGFsbG93ZWQgZm9yIHRoZSBlbGVtZW50LgotICAgICAqLwotICAgIGludCBnZXRPYmplY3RBcnJheU1pbkxlbmd0aChTdHJpbmcgZWxlbWVudE5hbWUpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgb2JqZWN0IGNsYXNzIGNvcnJlc3BvbmRpbmcgdG8gdGhlIHNwZWNpZmllZCBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBvYmplY3QgY2xhc3MgY29ycmVzcG9uZGluZyB0byB0aGUgc3BlY2lmaWVkIGVsZW1lbnQuCi0gICAgICovCi0gICAgQ2xhc3M8Pz4gZ2V0T2JqZWN0Q2xhc3MoU3RyaW5nIGVsZW1lbnROYW1lKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG9iamVjdCBkZWZhdWx0IHZhbHVlIGZvciB0aGUgZWxlbWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZWxlbWVudE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbGVtZW50IG5hbWUuCi0gICAgICogQHJldHVybiB0aGUgb2JqZWN0IGRlZmF1bHQgdmFsdWUgZm9yIHRoZSBlbGVtZW50LgotICAgICAqLwotICAgIE9iamVjdCBnZXRPYmplY3REZWZhdWx0VmFsdWUoU3RyaW5nIGVsZW1lbnROYW1lKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG9iamVjdCBlbnVtZXJhdGlvbnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIG9iamVjdCBlbnVtZXJhdGlvbnMuCi0gICAgICovCi0gICAgT2JqZWN0W10gZ2V0T2JqZWN0RW51bWVyYXRpb25zKFN0cmluZyBlbGVtZW50TmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBtYXhpbXVtIHZhbHVlIGFsbG93ZWQgZm9yIHRoZSBlbGVtZW50J3Mgb2JqZWN0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBtYXhpbXVtIHZhbHVlIGFsbG93ZWQgZm9yIHRoZSBlbGVtZW50J3Mgb2JqZWN0LgotICAgICAqLwotICAgIENvbXBhcmFibGU8Pz4gZ2V0T2JqZWN0TWF4VmFsdWUoU3RyaW5nIGVsZW1lbnROYW1lKTsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIG1pbmltdW0gdmFsdWUgYWxsb3dlZCBmb3IgdGhlIGVsZW1lbnQncyBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEByZXR1cm4gdGhlIG1pbmltdW0gdmFsdWUgYWxsb3dlZCBmb3IgdGhlIGVsZW1lbnQncyBvYmplY3QuCi0gICAgICovCi0gICAgQ29tcGFyYWJsZTw/PiBnZXRPYmplY3RNaW5WYWx1ZShTdHJpbmcgZWxlbWVudE5hbWUpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY29uc3RhbnQgdGhhdCBpbmRpY2F0ZXMgdGhlIHR5cGUgb2YgdGhlIGVsZW1lbnQncyB2YWx1ZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZWxlbWVudE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbGVtZW50IG5hbWUuCi0gICAgICogQHJldHVybiB0aGUgY29uc3RhbnQgdGhhdCBpbmRpY2F0ZXMgdGhlIHR5cGUgb2YgdGhlIGVsZW1lbnQncyB2YWx1ZS4KLSAgICAgKi8KLSAgICBpbnQgZ2V0T2JqZWN0VmFsdWVUeXBlKFN0cmluZyBlbGVtZW50TmFtZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBuYW1lIG9mIHRoZSByb290IGVsZW1lbnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbmFtZSBvZiB0aGUgcm9vdCBlbGVtZW50LgotICAgICAqLwotICAgIFN0cmluZyBnZXRSb290TmFtZSgpOwotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vbWV0YWRhdGEvSUlPTWV0YWRhdGFGb3JtYXRJbXBsLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YUZvcm1hdEltcGwuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMWE2ZTU2OC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YUZvcm1hdEltcGwuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEwNTYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvLm1ldGFkYXRhOwotCi1pbXBvcnQgamF2YXguaW1hZ2Vpby5JbWFnZVR5cGVTcGVjaWZpZXI7Ci1pbXBvcnQgamF2YS51dGlsLio7Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5BY2Nlc3NDb250cm9sbGVyOwotaW1wb3J0IGphdmEuc2VjdXJpdHkuUHJpdmlsZWdlZEFjdGlvbjsKLQotLyoqCi0gKiBUaGUgSUlPTWV0YWRhdGFGb3JtYXRJbXBsIGNsYXNzIHByb3ZpZGVzIGFuIGltcGxlbWVudGF0aW9uIG9mIHRoZQotICogSUlPTWV0YWRhdGFGb3JtYXQgaW50ZXJmYWNlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIElJT01ldGFkYXRhRm9ybWF0SW1wbCBpbXBsZW1lbnRzIElJT01ldGFkYXRhRm9ybWF0IHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBzdGFuZGFyZE1ldGFkYXRhRm9ybWF0TmFtZS4KLSAgICAgKi8KLSAgICBAU3VwcHJlc3NXYXJuaW5ncyggewotICAgICAgICAiQ29uc3RhbnREZWNsYXJlZEluQWJzdHJhY3RDbGFzcyIKLSAgICB9KQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIHN0YW5kYXJkTWV0YWRhdGFGb3JtYXROYW1lID0gImphdmF4X2ltYWdlaW9fMS4wIjsKLQotICAgIC8qKgotICAgICAqIFRoZSBzdGFuZGFyZCBmb3JtYXQuCi0gICAgICovCi0gICAgQFN1cHByZXNzV2FybmluZ3MoIHsKLSAgICAgICAgIlN0YXRpY05vbkZpbmFsRmllbGQiCi0gICAgfSkKLSAgICBwcml2YXRlIHN0YXRpYyBJSU9NZXRhZGF0YUZvcm1hdEltcGwgc3RhbmRhcmRGb3JtYXQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcm9vdCBuYW1lLgotICAgICAqLwotICAgIHByaXZhdGUgU3RyaW5nIHJvb3ROYW1lOwotCi0gICAgLyoqCi0gICAgICogVGhlIGVsZW1lbnQgaGFzaC4KLSAgICAgKi8KLSAgICBwcml2YXRlIEhhc2hNYXA8U3RyaW5nLCBFbGVtZW50PiBlbGVtZW50SGFzaCA9IG5ldyBIYXNoTWFwPFN0cmluZywgRWxlbWVudD4oKTsKLQotICAgIC8qKgotICAgICAqIFRoZSByZXNvdXJjZSBiYXNlIG5hbWUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBTdHJpbmcgcmVzb3VyY2VCYXNlTmFtZSA9IGdldENsYXNzKCkuZ2V0TmFtZSgpICsgIlJlc291cmNlcyI7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYW4gSUlPTWV0YWRhdGFGb3JtYXRJbXBsIHdpdGggdGhlIHNwZWNpZmllZCByb290IG5hbWUgYW5kCi0gICAgICogY2hpbGQgcG9saWN5IChub3QgQ0hJTERfUE9MSUNZX1JFUEVBVCkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJvb3ROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiByb290IGVsZW1lbnQuCi0gICAgICogQHBhcmFtIGNoaWxkUG9saWN5Ci0gICAgICogICAgICAgICAgICB0aGUgY2hpbGQgcG9saWN5IGRlZmluZWQgYnkgb25lIG9mIHRoZSBDSElMRF9QT0xJQ1lfKgotICAgICAqICAgICAgICAgICAgY29uc3RhbnRzIChleGNlcHQgQ0hJTERfUE9MSUNZX1JFUEVBVCkuCi0gICAgICovCi0gICAgcHVibGljIElJT01ldGFkYXRhRm9ybWF0SW1wbChTdHJpbmcgcm9vdE5hbWUsIGludCBjaGlsZFBvbGljeSkgewotICAgICAgICBpZiAocm9vdE5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigicm9vdE5hbWUgaXMgbnVsbCIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChjaGlsZFBvbGljeSA8IENISUxEX1BPTElDWV9FTVBUWSB8fCBjaGlsZFBvbGljeSA+IENISUxEX1BPTElDWV9NQVgKLSAgICAgICAgICAgICAgICB8fCBjaGlsZFBvbGljeSA9PSBDSElMRF9QT0xJQ1lfUkVQRUFUKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJjaGlsZFBvbGljeSBpcyBub3Qgb25lIG9mIHRoZSBwcmVkZWZpbmVkIGNvbnN0YW50cyIpOwotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5yb290TmFtZSA9IHJvb3ROYW1lOwotICAgICAgICBFbGVtZW50IHJvb3QgPSBuZXcgRWxlbWVudCgpOwotICAgICAgICByb290Lm5hbWUgPSByb290TmFtZTsKLSAgICAgICAgcm9vdC5jaGlsZFBvbGljeSA9IGNoaWxkUG9saWN5OwotICAgICAgICBlbGVtZW50SGFzaC5wdXQocm9vdE5hbWUsIHJvb3QpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhbiBJSU9NZXRhZGF0YUZvcm1hdEltcGwgd2l0aCB0aGUgc3BlY2lmaWVkIHJvb3QgbmFtZSBhbmQKLSAgICAgKiBDSElMRF9QT0xJQ1lfUkVQRUFUIGNoaWxkIHBvbGljeS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcm9vdE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYW1lIG9mIHJvb3QgZWxlbWVudC4KLSAgICAgKiBAcGFyYW0gbWluQ2hpbGRyZW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBtaW5pbXVtIG51bWJlciBvZiBjaGlsZHJlbi4KLSAgICAgKiBAcGFyYW0gbWF4Q2hpbGRyZW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBtYXhpbXVtIG51bWJlciBvZiBjaGlsZHJlbgotICAgICAqLwotICAgIHB1YmxpYyBJSU9NZXRhZGF0YUZvcm1hdEltcGwoU3RyaW5nIHJvb3ROYW1lLCBpbnQgbWluQ2hpbGRyZW4sIGludCBtYXhDaGlsZHJlbikgewotICAgICAgICBpZiAocm9vdE5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigicm9vdE5hbWUgaXMgbnVsbCIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChtaW5DaGlsZHJlbiA8IDApIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIm1pbkNoaWxkcmVuIDwgMCEiKTsKLSAgICAgICAgfQotICAgICAgICBpZiAobWluQ2hpbGRyZW4gPiBtYXhDaGlsZHJlbikgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigibWluQ2hpbGRyZW4gPiBtYXhDaGlsZHJlbiEiKTsKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMucm9vdE5hbWUgPSByb290TmFtZTsKLSAgICAgICAgRWxlbWVudCByb290ID0gbmV3IEVsZW1lbnQoKTsKLSAgICAgICAgcm9vdC5uYW1lID0gcm9vdE5hbWU7Ci0gICAgICAgIHJvb3QubWluQ2hpbGRyZW4gPSBtaW5DaGlsZHJlbjsKLSAgICAgICAgcm9vdC5tYXhDaGlsZHJlbiA9IG1heENoaWxkcmVuOwotICAgICAgICByb290LmNoaWxkUG9saWN5ID0gQ0hJTERfUE9MSUNZX1JFUEVBVDsKLSAgICAgICAgZWxlbWVudEhhc2gucHV0KHJvb3ROYW1lLCByb290KTsKLSAgICB9Ci0KLSAgICBAU3VwcHJlc3NXYXJuaW5ncyggewotICAgICAgICAiQWJzdHJhY3RNZXRob2RPdmVycmlkZXNBYnN0cmFjdE1ldGhvZCIKLSAgICB9KQotICAgIHB1YmxpYyBhYnN0cmFjdCBib29sZWFuIGNhbk5vZGVBcHBlYXIoU3RyaW5nIGVsZW1lbnROYW1lLCBJbWFnZVR5cGVTcGVjaWZpZXIgaW1hZ2VUeXBlKTsKLQotICAgIC8qKgotICAgICAqIEFkZHMgYSBuZXcgYXR0cmlidXRlIHRvIGFuIGV4aXN0aW5nIGVsZW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiB0aGUgZWxlbWVudCB0byB3aGljaCB0aGUgbmV3IGF0dHJpYnV0ZSB3aWxsIGJlCi0gICAgICogICAgICAgICAgICBhZGRlZC4KLSAgICAgKiBAcGFyYW0gYXR0ck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBhdHRyaWJ1dGUgbmFtZS4KLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUgb2YgdGhlIG5ldyBhdHRyaWJ1dGUuCi0gICAgICogQHBhcmFtIHJlcXVpcmVkCi0gICAgICogICAgICAgICAgICB0aGUgZmxhZyB3aGljaCBpbmRpY2F0ZXMgd2hldGhlciB0aGlzIGF0dHJpYnV0ZSBtdXN0IGJlCi0gICAgICogICAgICAgICAgICBwcmVzZW50LgotICAgICAqIEBwYXJhbSBsaXN0TWluTGVuZ3RoCi0gICAgICogICAgICAgICAgICB0aGUgbWluaW11bSBsZWdhbCBudW1iZXIgb2YgbGlzdCBpdGVtcy4KLSAgICAgKiBAcGFyYW0gbGlzdE1heExlbmd0aAotICAgICAqICAgICAgICAgICAgdGhlIHRoZSBtYXhpbXVtIGxlZ2FsIG51bWJlciBvZiBsaXN0IGl0ZW1zLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIGFkZEF0dHJpYnV0ZShTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSwgaW50IGRhdGFUeXBlLAotICAgICAgICAgICAgYm9vbGVhbiByZXF1aXJlZCwgaW50IGxpc3RNaW5MZW5ndGgsIGludCBsaXN0TWF4TGVuZ3RoKSB7Ci0gICAgICAgIGlmIChhdHRyTmFtZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJhdHRyTmFtZSA9PSBudWxsISIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChkYXRhVHlwZSA8IERBVEFUWVBFX1NUUklORyB8fCBkYXRhVHlwZSA+IERBVEFUWVBFX0RPVUJMRSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiSW52YWxpZCB2YWx1ZSBmb3IgZGF0YVR5cGUhIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGxpc3RNaW5MZW5ndGggPCAwIHx8IGxpc3RNaW5MZW5ndGggPiBsaXN0TWF4TGVuZ3RoKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJJbnZhbGlkIGxpc3QgYm91bmRzISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICBBdHRsaXN0IGF0dHIgPSBuZXcgQXR0bGlzdCgpOwotICAgICAgICBhdHRyLm5hbWUgPSBhdHRyTmFtZTsKLSAgICAgICAgYXR0ci5kYXRhVHlwZSA9IGRhdGFUeXBlOwotICAgICAgICBhdHRyLnJlcXVpcmVkID0gcmVxdWlyZWQ7Ci0gICAgICAgIGF0dHIubGlzdE1pbkxlbmd0aCA9IGxpc3RNaW5MZW5ndGg7Ci0gICAgICAgIGF0dHIubGlzdE1heExlbmd0aCA9IGxpc3RNYXhMZW5ndGg7Ci0gICAgICAgIGF0dHIudmFsdWVUeXBlID0gVkFMVUVfTElTVDsKLQotICAgICAgICBlbGVtZW50LmF0dHJpYnV0ZXMucHV0KGF0dHJOYW1lLCBhdHRyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGEgbmV3IGF0dHJpYnV0ZSB0byBhbiBleGlzdGluZyBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWUgb2YgdGhlIGVsZW1lbnQgdG8gd2hpY2ggdGhlIG5ldyBhdHRyaWJ1dGUgd2lsbCBiZQotICAgICAqICAgICAgICAgICAgYWRkZWQuCi0gICAgICogQHBhcmFtIGF0dHJOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgYXR0cmlidXRlIG5hbWUuCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlIG9mIHRoZSBuZXcgYXR0cmlidXRlLgotICAgICAqIEBwYXJhbSByZXF1aXJlZAotICAgICAqICAgICAgICAgICAgdGhlIGZsYWcgd2hpY2ggaW5kaWNhdGVzIHdoZXRoZXIgdGhpcyBhdHRyaWJ1dGUgbXVzdCBiZQotICAgICAqICAgICAgICAgICAgcHJlc2VudC4KLSAgICAgKiBAcGFyYW0gZGVmYXVsdFZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgZGVmYXVsdCB2YWx1ZSBvZiB0aGUgYXR0cmlidXRlLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIGFkZEF0dHJpYnV0ZShTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSwgaW50IGRhdGFUeXBlLAotICAgICAgICAgICAgYm9vbGVhbiByZXF1aXJlZCwgU3RyaW5nIGRlZmF1bHRWYWx1ZSkgewotICAgICAgICBpZiAoYXR0ck5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiYXR0ck5hbWUgPT0gbnVsbCEiKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoZGF0YVR5cGUgPCBEQVRBVFlQRV9TVFJJTkcgfHwgZGF0YVR5cGUgPiBEQVRBVFlQRV9ET1VCTEUpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkludmFsaWQgdmFsdWUgZm9yIGRhdGFUeXBlISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICBBdHRsaXN0IGF0dHIgPSBuZXcgQXR0bGlzdCgpOwotICAgICAgICBhdHRyLm5hbWUgPSBhdHRyTmFtZTsKLSAgICAgICAgYXR0ci5kYXRhVHlwZSA9IGRhdGFUeXBlOwotICAgICAgICBhdHRyLnJlcXVpcmVkID0gcmVxdWlyZWQ7Ci0gICAgICAgIGF0dHIuZGVmYXVsdFZhbHVlID0gZGVmYXVsdFZhbHVlOwotICAgICAgICBhdHRyLnZhbHVlVHlwZSA9IFZBTFVFX0FSQklUUkFSWTsKLQotICAgICAgICBlbGVtZW50LmF0dHJpYnV0ZXMucHV0KGF0dHJOYW1lLCBhdHRyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGEgbmV3IGF0dHJpYnV0ZSB0byBhbiBleGlzdGluZyBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWUgb2YgdGhlIGVsZW1lbnQgdG8gd2hpY2ggdGhlIG5ldyBhdHRyaWJ1dGUgd2lsbCBiZQotICAgICAqICAgICAgICAgICAgYWRkZWQuCi0gICAgICogQHBhcmFtIGF0dHJOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgYXR0cmlidXRlIG5hbWUuCi0gICAgICogQHBhcmFtIGRhdGFUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgZGF0YSB0eXBlIG9mIHRoZSBuZXcgYXR0cmlidXRlLgotICAgICAqIEBwYXJhbSByZXF1aXJlZAotICAgICAqICAgICAgICAgICAgdGhlIGZsYWcgd2hpY2ggaW5kaWNhdGVzIHdoZXRoZXIgdGhpcyBhdHRyaWJ1dGUgbXVzdCBiZQotICAgICAqICAgICAgICAgICAgcHJlc2VudC4KLSAgICAgKiBAcGFyYW0gZGVmYXVsdFZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgZGVmYXVsdCB2YWx1ZSBvZiB0aGUgYXR0cmlidXRlLgotICAgICAqIEBwYXJhbSBlbnVtZXJhdGVkVmFsdWVzCi0gICAgICogICAgICAgICAgICB0aGUgbGVnYWwgdmFsdWVzIGZvciB0aGUgYXR0cmlidXRlIGFzIGEgbGlzdCBvZiBzdHJpbmdzLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIGFkZEF0dHJpYnV0ZShTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSwgaW50IGRhdGFUeXBlLAotICAgICAgICAgICAgYm9vbGVhbiByZXF1aXJlZCwgU3RyaW5nIGRlZmF1bHRWYWx1ZSwgTGlzdDxTdHJpbmc+IGVudW1lcmF0ZWRWYWx1ZXMpIHsKLSAgICAgICAgaWYgKGF0dHJOYW1lID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImF0dHJOYW1lID09IG51bGwhIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGRhdGFUeXBlIDwgREFUQVRZUEVfU1RSSU5HIHx8IGRhdGFUeXBlID4gREFUQVRZUEVfRE9VQkxFKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJJbnZhbGlkIHZhbHVlIGZvciBkYXRhVHlwZSEiKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoZW51bWVyYXRlZFZhbHVlcyA9PSBudWxsIHx8IGVudW1lcmF0ZWRWYWx1ZXMuaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJlbnVtZXJhdGVkVmFsdWVzIGlzIGVtcHR5IG9yIG51bGwiKTsKLSAgICAgICAgfQotCi0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBmb3IgKFN0cmluZyBlbnVtZXJhdGVkVmFsdWUgOiBlbnVtZXJhdGVkVmFsdWVzKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGVudW1lcmF0ZWRWYWx1ZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImVudW1lcmF0ZWRWYWx1ZXMgY29udGFpbnMgYSBudWxsISIpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBjYXRjaCAoQ2xhc3NDYXN0RXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImVudW1lcmF0ZWRWYWx1ZXMgY29udGFpbnMgYSBub24tU3RyaW5nIHZhbHVlISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICBBdHRsaXN0IGF0dHIgPSBuZXcgQXR0bGlzdCgpOwotICAgICAgICBhdHRyLm5hbWUgPSBhdHRyTmFtZTsKLSAgICAgICAgYXR0ci5kYXRhVHlwZSA9IGRhdGFUeXBlOwotICAgICAgICBhdHRyLnJlcXVpcmVkID0gcmVxdWlyZWQ7Ci0gICAgICAgIGF0dHIuZGVmYXVsdFZhbHVlID0gZGVmYXVsdFZhbHVlOwotICAgICAgICBhdHRyLmVudW1lcmF0ZWRWYWx1ZXMgPSBlbnVtZXJhdGVkVmFsdWVzOwotICAgICAgICBhdHRyLnZhbHVlVHlwZSA9IFZBTFVFX0VOVU1FUkFUSU9OOwotCi0gICAgICAgIGVsZW1lbnQuYXR0cmlidXRlcy5wdXQoYXR0ck5hbWUsIGF0dHIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgYSBuZXcgYXR0cmlidXRlIHRvIGFuIGV4aXN0aW5nIGVsZW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZSBvZiB0aGUgZWxlbWVudCB0byB3aGljaCB0aGUgbmV3IGF0dHJpYnV0ZSB3aWxsIGJlCi0gICAgICogICAgICAgICAgICBhZGRlZC4KLSAgICAgKiBAcGFyYW0gYXR0ck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBhdHRyaWJ1dGUgbmFtZS4KLSAgICAgKiBAcGFyYW0gZGF0YVR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkYXRhIHR5cGUgb2YgdGhlIG5ldyBhdHRyaWJ1dGUuCi0gICAgICogQHBhcmFtIHJlcXVpcmVkCi0gICAgICogICAgICAgICAgICB0aGUgZmxhZyB3aGljaCBpbmRpY2F0ZXMgd2hldGhlciB0aGlzIGF0dHJpYnV0ZSBtdXN0IGJlCi0gICAgICogICAgICAgICAgICBwcmVzZW50LgotICAgICAqIEBwYXJhbSBkZWZhdWx0VmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZWZhdWx0IHZhbHVlIG9mIGF0dHJpYnV0ZS4KLSAgICAgKiBAcGFyYW0gbWluVmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBtaW5pbXVtIGxlZ2FsIHZhbHVlIG9mIGFuIGF0dHJpYnV0ZS4KLSAgICAgKiBAcGFyYW0gbWF4VmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBtYXhpbXVtIGxlZ2FsIHZhbHVlIG9mIGFuIGF0dHJpYnV0ZS4KLSAgICAgKiBAcGFyYW0gbWluSW5jbHVzaXZlCi0gICAgICogICAgICAgICAgICB0aGUgZmxhZyB3aGljaCBpbmRpY2F0ZXMgd2hldGhlciB0aGUgbWluVmFsdWUgaXMgaW5jbHVzaXZlLgotICAgICAqIEBwYXJhbSBtYXhJbmNsdXNpdmUKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbGFnIHdoaWNoIGluZGljYXRlcyB3aGV0aGVyIHRoZSBtYXhWYWx1ZSBpcyBpbmNsdXNpdmUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgYWRkQXR0cmlidXRlKFN0cmluZyBlbGVtZW50TmFtZSwgU3RyaW5nIGF0dHJOYW1lLCBpbnQgZGF0YVR5cGUsCi0gICAgICAgICAgICBib29sZWFuIHJlcXVpcmVkLCBTdHJpbmcgZGVmYXVsdFZhbHVlLCBTdHJpbmcgbWluVmFsdWUsIFN0cmluZyBtYXhWYWx1ZSwKLSAgICAgICAgICAgIGJvb2xlYW4gbWluSW5jbHVzaXZlLCBib29sZWFuIG1heEluY2x1c2l2ZSkgewotICAgICAgICBpZiAoYXR0ck5hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiYXR0ck5hbWUgPT0gbnVsbCEiKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoZGF0YVR5cGUgPCBEQVRBVFlQRV9TVFJJTkcgfHwgZGF0YVR5cGUgPiBEQVRBVFlQRV9ET1VCTEUpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkludmFsaWQgdmFsdWUgZm9yIGRhdGFUeXBlISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICBBdHRsaXN0IGF0dHIgPSBuZXcgQXR0bGlzdCgpOwotICAgICAgICBhdHRyLm5hbWUgPSBhdHRyTmFtZTsKLSAgICAgICAgYXR0ci5kYXRhVHlwZSA9IGRhdGFUeXBlOwotICAgICAgICBhdHRyLnJlcXVpcmVkID0gcmVxdWlyZWQ7Ci0gICAgICAgIGF0dHIuZGVmYXVsdFZhbHVlID0gZGVmYXVsdFZhbHVlOwotICAgICAgICBhdHRyLm1pblZhbHVlID0gbWluVmFsdWU7Ci0gICAgICAgIGF0dHIubWF4VmFsdWUgPSBtYXhWYWx1ZTsKLSAgICAgICAgYXR0ci5taW5JbmNsdXNpdmUgPSBtaW5JbmNsdXNpdmU7Ci0gICAgICAgIGF0dHIubWF4SW5jbHVzaXZlID0gbWF4SW5jbHVzaXZlOwotCi0gICAgICAgIGF0dHIudmFsdWVUeXBlID0gVkFMVUVfUkFOR0U7Ci0gICAgICAgIGF0dHIudmFsdWVUeXBlIHw9IG1pbkluY2x1c2l2ZSA/IFZBTFVFX1JBTkdFX01JTl9JTkNMVVNJVkVfTUFTSyA6IDA7Ci0gICAgICAgIGF0dHIudmFsdWVUeXBlIHw9IG1heEluY2x1c2l2ZSA/IFZBTFVFX1JBTkdFX01BWF9JTkNMVVNJVkVfTUFTSyA6IDA7Ci0KLSAgICAgICAgZWxlbWVudC5hdHRyaWJ1dGVzLnB1dChhdHRyTmFtZSwgYXR0cik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyBhIG5ldyBhdHRyaWJ1dGUgd2l0aCBib29sZWFuIGRhdGEgdHlwZSB0byBhbiBleGlzdGluZyBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWUgb2YgdGhlIGVsZW1lbnQgdG8gd2hpY2ggdGhlIG5ldyBhdHRyaWJ1dGUgd2lsbCBiZQotICAgICAqICAgICAgICAgICAgYWRkZWQuCi0gICAgICogQHBhcmFtIGF0dHJOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgYXR0cmlidXRlIG5hbWUuCi0gICAgICogQHBhcmFtIGhhc0RlZmF1bHRWYWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIGZsYWcgd2hpY2ggaW5kaWNhdGVzIHdoZXRoZXIgdGhpcyBhdHRyaWJ1dGUgbXVzdCBoYXZlIGEKLSAgICAgKiAgICAgICAgICAgIGRlZmF1bHQgdmFsdWUuCi0gICAgICogQHBhcmFtIGRlZmF1bHRWYWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIGRlZmF1bHQgdmFsdWUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgYWRkQm9vbGVhbkF0dHJpYnV0ZShTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSwKLSAgICAgICAgICAgIGJvb2xlYW4gaGFzRGVmYXVsdFZhbHVlLCBib29sZWFuIGRlZmF1bHRWYWx1ZSkgewotICAgICAgICBTdHJpbmcgZGVmYXVsdFZhbCA9IGhhc0RlZmF1bHRWYWx1ZSA/IChkZWZhdWx0VmFsdWUgPyAiVFJVRSIgOiAiRkFMU0UiKSA6IG51bGw7Ci0gICAgICAgIEFycmF5TGlzdDxTdHJpbmc+IHZhbHVlcyA9IG5ldyBBcnJheUxpc3Q8U3RyaW5nPigyKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiVFJVRSIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJGQUxTRSIpOwotCi0gICAgICAgIGFkZEF0dHJpYnV0ZShlbGVtZW50TmFtZSwgYXR0ck5hbWUsIERBVEFUWVBFX0JPT0xFQU4sIHRydWUsIGRlZmF1bHRWYWwsIHZhbHVlcyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyBhbiBleGlzdGluZyBlbGVtZW50IHRvIHRoZSBsaXN0IG9mIGNoaWxkIGVsZW1lbnRzIG9mIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBwYXJlbnQgZWxlbWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZWxlbWVudE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYW1lIG9mIHRoZSBlbGVtZW50IHRvIGJlIGFkZGVkLgotICAgICAqIEBwYXJhbSBwYXJlbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcGFyZW50IGVsZW1lbnQgbmFtZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBhZGRDaGlsZEVsZW1lbnQoU3RyaW5nIGVsZW1lbnROYW1lLCBTdHJpbmcgcGFyZW50TmFtZSkgewotICAgICAgICBFbGVtZW50IHBhcmVudCA9IGZpbmRFbGVtZW50KHBhcmVudE5hbWUpOwotICAgICAgICBFbGVtZW50IGVsZW1lbnQgPSBmaW5kRWxlbWVudChlbGVtZW50TmFtZSk7Ci0gICAgICAgIHBhcmVudC5jaGlsZHJlbi5hZGQoZWxlbWVudC5uYW1lKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGEgbmV3IGVsZW1lbnQgdHlwZSB0byB0aGlzIElJT01ldGFkYXRhRm9ybWF0IHdpdGggYSBjaGlsZCBwb2xpY3kgKGlmCi0gICAgICogcG9saWN5IGlzIG5vdCBDSElMRF9QT0xJQ1lfUkVQRUFUKS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZWxlbWVudE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYW1lIG9mIHRoZSBlbGVtZW50IHRvIGJlIGFkZGVkLgotICAgICAqIEBwYXJhbSBwYXJlbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcGFyZW50IGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcGFyYW0gY2hpbGRQb2xpY3kKLSAgICAgKiAgICAgICAgICAgIG9uZSBvZiB0aGUgQ0hJTERfUE9MSUNZXyogY29uc3RhbnRzIGRlZmluZWQgYnkKLSAgICAgKiAgICAgICAgICAgIElJT01ldGFkYXRhRm9ybWF0LgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIGFkZEVsZW1lbnQoU3RyaW5nIGVsZW1lbnROYW1lLCBTdHJpbmcgcGFyZW50TmFtZSwgaW50IGNoaWxkUG9saWN5KSB7Ci0gICAgICAgIGlmIChjaGlsZFBvbGljeSA8IENISUxEX1BPTElDWV9FTVBUWSB8fCBjaGlsZFBvbGljeSA+IENISUxEX1BPTElDWV9NQVgKLSAgICAgICAgICAgICAgICB8fCBjaGlsZFBvbGljeSA9PSBDSElMRF9QT0xJQ1lfUkVQRUFUKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJjaGlsZFBvbGljeSBpcyBub3Qgb25lIG9mIHRoZSBwcmVkZWZpbmVkIGNvbnN0YW50cyIpOwotICAgICAgICB9Ci0KLSAgICAgICAgRWxlbWVudCBwYXJlbnQgPSBmaW5kRWxlbWVudChwYXJlbnROYW1lKTsKLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gbmV3IEVsZW1lbnQoKTsKLSAgICAgICAgZWxlbWVudC5uYW1lID0gZWxlbWVudE5hbWU7Ci0gICAgICAgIGVsZW1lbnQuY2hpbGRQb2xpY3kgPSBjaGlsZFBvbGljeTsKLSAgICAgICAgZWxlbWVudEhhc2gucHV0KGVsZW1lbnROYW1lLCBlbGVtZW50KTsKLSAgICAgICAgcGFyZW50LmNoaWxkcmVuLmFkZChlbGVtZW50TmFtZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyBhIG5ldyBlbGVtZW50IHR5cGUgdG8gdGhpcyBJSU9NZXRhZGF0YUZvcm1hdCB3aXRoCi0gICAgICogQ0hJTERfUE9MSUNZX1JFUEVBVCBhbmQgdGhlIHNwZWNpZmllZCBtaW5pbXVtIGFuZCBtYXhpbXVtIG51bWJlciBvZiBjaGlsZAotICAgICAqIGVsZW1lbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZSB0byBiZSBhZGRlZC4KLSAgICAgKiBAcGFyYW0gcGFyZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIHBhcmVudCBlbGVtZW50IG5hbWUuCi0gICAgICogQHBhcmFtIG1pbkNoaWxkcmVuCi0gICAgICogICAgICAgICAgICB0aGUgbWluaW11bSBudW1iZXIgb2YgY2hpbGQgZWxlbWVudHMuCi0gICAgICogQHBhcmFtIG1heENoaWxkcmVuCi0gICAgICogICAgICAgICAgICB0aGUgbWF4aW11bSBudW1iZXIgb2YgY2hpbGQgZWxlbWVudHMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgYWRkRWxlbWVudChTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBwYXJlbnROYW1lLCBpbnQgbWluQ2hpbGRyZW4sCi0gICAgICAgICAgICBpbnQgbWF4Q2hpbGRyZW4pIHsKLSAgICAgICAgaWYgKG1pbkNoaWxkcmVuIDwgMCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigibWluQ2hpbGRyZW4gPCAwISIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChtaW5DaGlsZHJlbiA+IG1heENoaWxkcmVuKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJtaW5DaGlsZHJlbiA+IG1heENoaWxkcmVuISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgRWxlbWVudCBwYXJlbnQgPSBmaW5kRWxlbWVudChwYXJlbnROYW1lKTsKLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gbmV3IEVsZW1lbnQoKTsKLSAgICAgICAgZWxlbWVudC5uYW1lID0gZWxlbWVudE5hbWU7Ci0gICAgICAgIGVsZW1lbnQuY2hpbGRQb2xpY3kgPSBDSElMRF9QT0xJQ1lfUkVQRUFUOwotICAgICAgICBlbGVtZW50Lm1pbkNoaWxkcmVuID0gbWluQ2hpbGRyZW47Ci0gICAgICAgIGVsZW1lbnQubWF4Q2hpbGRyZW4gPSBtYXhDaGlsZHJlbjsKLSAgICAgICAgZWxlbWVudEhhc2gucHV0KGVsZW1lbnROYW1lLCBlbGVtZW50KTsKLSAgICAgICAgcGFyZW50LmNoaWxkcmVuLmFkZChlbGVtZW50TmFtZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyBhbiBPYmplY3QgcmVmZXJlbmNlIHdpdGggdGhlIHNwZWNpZmllZCBjbGFzcyB0eXBlIHRvIGJlIHN0b3JlZCBhcwotICAgICAqIGVsZW1lbnQncyB2YWx1ZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZWxlbWVudE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbGVtZW50IG5hbWUuCi0gICAgICogQHBhcmFtIGNsYXNzVHlwZQotICAgICAqICAgICAgICAgICAgdGhlIGNsYXNzIGluZGljYXRlcyB0aGUgbGVnYWwgdHlwZXMgZm9yIHRoZSBvYmplY3QncyB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gYXJyYXlNaW5MZW5ndGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBtaW5pbXVtIGxlZ2FsIGxlbmd0aCBmb3IgdGhlIGFycmF5LgotICAgICAqIEBwYXJhbSBhcnJheU1heExlbmd0aAotICAgICAqICAgICAgICAgICAgdGhlIG1heGltdW0gbGVnYWwgbGVuZ3RoIGZvciB0aGUgYXJyYXkuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgYWRkT2JqZWN0VmFsdWUoU3RyaW5nIGVsZW1lbnROYW1lLCBDbGFzczw/PiBjbGFzc1R5cGUsIGludCBhcnJheU1pbkxlbmd0aCwKLSAgICAgICAgICAgIGludCBhcnJheU1heExlbmd0aCkgewotICAgICAgICBFbGVtZW50IGVsZW1lbnQgPSBmaW5kRWxlbWVudChlbGVtZW50TmFtZSk7Ci0KLSAgICAgICAgT2JqZWN0VmFsdWUgb2JqVmFsID0gbmV3IE9iamVjdFZhbHVlKCk7Ci0gICAgICAgIG9ialZhbC5jbGFzc1R5cGUgPSBjbGFzc1R5cGU7Ci0gICAgICAgIG9ialZhbC5hcnJheU1heExlbmd0aCA9IGFycmF5TWF4TGVuZ3RoOwotICAgICAgICBvYmpWYWwuYXJyYXlNaW5MZW5ndGggPSBhcnJheU1pbkxlbmd0aDsKLSAgICAgICAgb2JqVmFsLnZhbHVlVHlwZSA9IFZBTFVFX0xJU1Q7Ci0KLSAgICAgICAgZWxlbWVudC5vYmplY3RWYWx1ZSA9IG9ialZhbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGFuIE9iamVjdCByZWZlcmVuY2Ugd2l0aCB0aGUgc3BlY2lmaWVkIGNsYXNzIHR5cGUgdG8gYmUgc3RvcmVkIGFzIGFuCi0gICAgICogZWxlbWVudCdzIHZhbHVlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcGFyYW0gY2xhc3NUeXBlCi0gICAgICogICAgICAgICAgICB0aGUgY2xhc3MgaW5kaWNhdGVzIHRoZSBsZWdhbCB0eXBlcyBmb3IgdGhlIG9iamVjdCdzIHZhbHVlLgotICAgICAqIEBwYXJhbSByZXF1aXJlZAotICAgICAqICAgICAgICAgICAgYSBmbGFnIGluZGljYXRlZCB0aGF0IHRoaXMgb2JqZWN0IHZhbHVlIG11c3QgYmUgcHJlc2VudC4KLSAgICAgKiBAcGFyYW0gZGVmYXVsdFZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgZGVmYXVsdCB2YWx1ZSwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgPFQ+IHZvaWQgYWRkT2JqZWN0VmFsdWUoU3RyaW5nIGVsZW1lbnROYW1lLCBDbGFzczxUPiBjbGFzc1R5cGUsIGJvb2xlYW4gcmVxdWlyZWQsCi0gICAgICAgICAgICBUIGRlZmF1bHRWYWx1ZSkgewotICAgICAgICAvLyBub3RlOiByZXFpcmVkIGlzIGFuIHVudXNlZCBwYXJhbWV0ZXIKLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotCi0gICAgICAgIE9iamVjdFZhbHVlPFQ+IG9ialZhbCA9IG5ldyBPYmplY3RWYWx1ZTxUPigpOwotICAgICAgICBvYmpWYWwuY2xhc3NUeXBlID0gY2xhc3NUeXBlOwotICAgICAgICBvYmpWYWwuZGVmYXVsdFZhbHVlID0gZGVmYXVsdFZhbHVlOwotICAgICAgICBvYmpWYWwudmFsdWVUeXBlID0gVkFMVUVfQVJCSVRSQVJZOwotCi0gICAgICAgIGVsZW1lbnQub2JqZWN0VmFsdWUgPSBvYmpWYWw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyBhbiBPYmplY3QgcmVmZXJlbmNlIHdpdGggdGhlIHNwZWNpZmllZCBjbGFzcyB0eXBlIHRvIGJlIHN0b3JlZCBhcwotICAgICAqIHRoZSBlbGVtZW50J3MgdmFsdWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEBwYXJhbSBjbGFzc1R5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBjbGFzcyBpbmRpY2F0ZXMgdGhlIGxlZ2FsIHR5cGVzIGZvciB0aGUgb2JqZWN0IHZhbHVlLgotICAgICAqIEBwYXJhbSByZXF1aXJlZAotICAgICAqICAgICAgICAgICAgYSBmbGFnIGluZGljYXRlZCB0aGF0IHRoaXMgb2JqZWN0IHZhbHVlIG11c3QgYmUgcHJlc2VudC4KLSAgICAgKiBAcGFyYW0gZGVmYXVsdFZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgZGVmYXVsdCB2YWx1ZSwgb3IgbnVsbC4KLSAgICAgKiBAcGFyYW0gZW51bWVyYXRlZFZhbHVlcwotICAgICAqICAgICAgICAgICAgdGhlIGxpc3Qgb2YgbGVnYWwgdmFsdWVzIGZvciB0aGUgb2JqZWN0LgotICAgICAqLwotICAgIHByb3RlY3RlZCA8VD4gdm9pZCBhZGRPYmplY3RWYWx1ZShTdHJpbmcgZWxlbWVudE5hbWUsIENsYXNzPFQ+IGNsYXNzVHlwZSwgYm9vbGVhbiByZXF1aXJlZCwKLSAgICAgICAgICAgIFQgZGVmYXVsdFZhbHVlLCBMaXN0PD8gZXh0ZW5kcyBUPiBlbnVtZXJhdGVkVmFsdWVzKSB7Ci0gICAgICAgIC8vIG5vdGU6IHJlcWlyZWQgaXMgYW4gdW51c2VkIHBhcmFtZXRlcgotICAgICAgICBpZiAoZW51bWVyYXRlZFZhbHVlcyA9PSBudWxsIHx8IGVudW1lcmF0ZWRWYWx1ZXMuaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJlbnVtZXJhdGVkVmFsdWVzIGlzIGVtcHR5IG9yIG51bGwiKTsKLSAgICAgICAgfQotCi0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBmb3IgKFQgZW51bWVyYXRlZFZhbHVlIDogZW51bWVyYXRlZFZhbHVlcykgewotICAgICAgICAgICAgICAgIGlmIChlbnVtZXJhdGVkVmFsdWUgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJlbnVtZXJhdGVkVmFsdWVzIGNvbnRhaW5zIGEgbnVsbCEiKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gY2F0Y2ggKENsYXNzQ2FzdEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKAotICAgICAgICAgICAgICAgICAgICAiZW51bWVyYXRlZFZhbHVlcyBjb250YWlucyBhIHZhbHVlIG5vdCBvZiBjbGFzcyBjbGFzc1R5cGUhIik7Ci0gICAgICAgIH0KLQotICAgICAgICBFbGVtZW50IGVsZW1lbnQgPSBmaW5kRWxlbWVudChlbGVtZW50TmFtZSk7Ci0KLSAgICAgICAgT2JqZWN0VmFsdWU8VD4gb2JqVmFsID0gbmV3IE9iamVjdFZhbHVlPFQ+KCk7Ci0gICAgICAgIG9ialZhbC5jbGFzc1R5cGUgPSBjbGFzc1R5cGU7Ci0gICAgICAgIG9ialZhbC5kZWZhdWx0VmFsdWUgPSBkZWZhdWx0VmFsdWU7Ci0gICAgICAgIG9ialZhbC5lbnVtZXJhdGVkVmFsdWVzID0gZW51bWVyYXRlZFZhbHVlczsKLSAgICAgICAgb2JqVmFsLnZhbHVlVHlwZSA9IFZBTFVFX0VOVU1FUkFUSU9OOwotCi0gICAgICAgIGVsZW1lbnQub2JqZWN0VmFsdWUgPSBvYmpWYWw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyBhbiBPYmplY3QgcmVmZXJlbmNlIHdpdGggdGhlIHNwZWNpZmllZCBjbGFzcyB0eXBlIHRvIGJlIHN0b3JlZCBhcwotICAgICAqIHRoZSBlbGVtZW50J3MgdmFsdWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEBwYXJhbSBjbGFzc1R5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBjbGFzcyBpbmRpY2F0ZXMgdGhlIGxlZ2FsIHR5cGVzIGZvciB0aGUgb2JqZWN0IHZhbHVlLgotICAgICAqIEBwYXJhbSBkZWZhdWx0VmFsdWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBkZWZhdWx0IHZhbHVlLCBvciBudWxsLgotICAgICAqIEBwYXJhbSBtaW5WYWx1ZQotICAgICAqICAgICAgICAgICAgdGhlIG1pbmltdW0gbGVnYWwgdmFsdWUgZm9yIHRoZSBvYmplY3QgdmFsdWUuCi0gICAgICogQHBhcmFtIG1heFZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgbWF4aW11bSBsZWdhbCB2YWx1ZSBmb3IgdGhlIG9iamVjdCB2YWx1ZS4KLSAgICAgKiBAcGFyYW0gbWluSW5jbHVzaXZlCi0gICAgICogICAgICAgICAgICB0aGUgZmxhZyB3aGljaCBpbmRpY2F0ZXMgd2hldGhlciB0aGUgbWluVmFsdWUgaXMgaW5jbHVzaXZlLgotICAgICAqIEBwYXJhbSBtYXhJbmNsdXNpdmUKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbGFnIHdoaWNoIGluZGljYXRlcyB3aGV0aGVyIHRoZSBtYXhWYWx1ZSBpcyBpbmNsdXNpdmUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIDxUIGV4dGVuZHMgT2JqZWN0ICYgQ29tcGFyYWJsZTw/IHN1cGVyIFQ+PiB2b2lkIGFkZE9iamVjdFZhbHVlKFN0cmluZyBlbGVtZW50TmFtZSwKLSAgICAgICAgICAgIENsYXNzPFQ+IGNsYXNzVHlwZSwgVCBkZWZhdWx0VmFsdWUsIENvbXBhcmFibGU8PyBzdXBlciBUPiBtaW5WYWx1ZSwKLSAgICAgICAgICAgIENvbXBhcmFibGU8PyBzdXBlciBUPiBtYXhWYWx1ZSwgYm9vbGVhbiBtaW5JbmNsdXNpdmUsIGJvb2xlYW4gbWF4SW5jbHVzaXZlKSB7Ci0gICAgICAgIEVsZW1lbnQgZWxlbWVudCA9IGZpbmRFbGVtZW50KGVsZW1lbnROYW1lKTsKLQotICAgICAgICBPYmplY3RWYWx1ZTxUPiBvYmpWYWwgPSBuZXcgT2JqZWN0VmFsdWU8VD4oKTsKLSAgICAgICAgb2JqVmFsLmNsYXNzVHlwZSA9IGNsYXNzVHlwZTsKLSAgICAgICAgb2JqVmFsLmRlZmF1bHRWYWx1ZSA9IGRlZmF1bHRWYWx1ZTsKLSAgICAgICAgb2JqVmFsLm1pblZhbHVlID0gbWluVmFsdWU7Ci0gICAgICAgIG9ialZhbC5tYXhWYWx1ZSA9IG1heFZhbHVlOwotICAgICAgICBvYmpWYWwubWluSW5jbHVzaXZlID0gbWluSW5jbHVzaXZlOwotICAgICAgICBvYmpWYWwubWF4SW5jbHVzaXZlID0gbWF4SW5jbHVzaXZlOwotCi0gICAgICAgIG9ialZhbC52YWx1ZVR5cGUgPSBWQUxVRV9SQU5HRTsKLSAgICAgICAgb2JqVmFsLnZhbHVlVHlwZSB8PSBtaW5JbmNsdXNpdmUgPyBWQUxVRV9SQU5HRV9NSU5fSU5DTFVTSVZFX01BU0sgOiAwOwotICAgICAgICBvYmpWYWwudmFsdWVUeXBlIHw9IG1heEluY2x1c2l2ZSA/IFZBTFVFX1JBTkdFX01BWF9JTkNMVVNJVkVfTUFTSyA6IDA7Ci0KLSAgICAgICAgZWxlbWVudC5vYmplY3RWYWx1ZSA9IG9ialZhbDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldEF0dHJpYnV0ZURhdGFUeXBlKFN0cmluZyBlbGVtZW50TmFtZSwgU3RyaW5nIGF0dHJOYW1lKSB7Ci0gICAgICAgIEF0dGxpc3QgYXR0ciA9IGZpbmRBdHRyaWJ1dGUoZWxlbWVudE5hbWUsIGF0dHJOYW1lKTsKLSAgICAgICAgcmV0dXJuIGF0dHIuZGF0YVR5cGU7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZyBnZXRBdHRyaWJ1dGVEZWZhdWx0VmFsdWUoU3RyaW5nIGVsZW1lbnROYW1lLCBTdHJpbmcgYXR0ck5hbWUpIHsKLSAgICAgICAgQXR0bGlzdCBhdHRyID0gZmluZEF0dHJpYnV0ZShlbGVtZW50TmFtZSwgYXR0ck5hbWUpOwotICAgICAgICByZXR1cm4gYXR0ci5kZWZhdWx0VmFsdWU7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZyBnZXRBdHRyaWJ1dGVEZXNjcmlwdGlvbihTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSwgTG9jYWxlIGxvY2FsZSkgewotICAgICAgICBmaW5kQXR0cmlidXRlKGVsZW1lbnROYW1lLCBhdHRyTmFtZSk7Ci0gICAgICAgIHJldHVybiBnZXRSZXNvdXJjZVN0cmluZyhlbGVtZW50TmFtZSArICIvIiArIGF0dHJOYW1lLCBsb2NhbGUpOwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRBdHRyaWJ1dGVFbnVtZXJhdGlvbnMoU3RyaW5nIGVsZW1lbnROYW1lLCBTdHJpbmcgYXR0ck5hbWUpIHsKLSAgICAgICAgQXR0bGlzdCBhdHRyID0gZmluZEF0dHJpYnV0ZShlbGVtZW50TmFtZSwgYXR0ck5hbWUpOwotICAgICAgICBpZiAoYXR0ci52YWx1ZVR5cGUgIT0gVkFMVUVfRU5VTUVSQVRJT04pIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkF0dHJpYnV0ZSBpcyBub3QgYW4gZW51bWVyYXRpb24hIik7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gYXR0ci5lbnVtZXJhdGVkVmFsdWVzLnRvQXJyYXkobmV3IFN0cmluZ1thdHRyLmVudW1lcmF0ZWRWYWx1ZXMuc2l6ZSgpXSk7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRBdHRyaWJ1dGVMaXN0TWF4TGVuZ3RoKFN0cmluZyBlbGVtZW50TmFtZSwgU3RyaW5nIGF0dHJOYW1lKSB7Ci0gICAgICAgIEF0dGxpc3QgYXR0ciA9IGZpbmRBdHRyaWJ1dGUoZWxlbWVudE5hbWUsIGF0dHJOYW1lKTsKLSAgICAgICAgaWYgKGF0dHIudmFsdWVUeXBlICE9IFZBTFVFX0xJU1QpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkF0dHJpYnV0ZSBpcyBub3QgYSBsaXN0ISIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBhdHRyLmxpc3RNYXhMZW5ndGg7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRBdHRyaWJ1dGVMaXN0TWluTGVuZ3RoKFN0cmluZyBlbGVtZW50TmFtZSwgU3RyaW5nIGF0dHJOYW1lKSB7Ci0gICAgICAgIEF0dGxpc3QgYXR0ciA9IGZpbmRBdHRyaWJ1dGUoZWxlbWVudE5hbWUsIGF0dHJOYW1lKTsKLSAgICAgICAgaWYgKGF0dHIudmFsdWVUeXBlICE9IFZBTFVFX0xJU1QpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkF0dHJpYnV0ZSBpcyBub3QgYSBsaXN0ISIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBhdHRyLmxpc3RNaW5MZW5ndGg7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZyBnZXRBdHRyaWJ1dGVNYXhWYWx1ZShTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSkgewotICAgICAgICBBdHRsaXN0IGF0dHIgPSBmaW5kQXR0cmlidXRlKGVsZW1lbnROYW1lLCBhdHRyTmFtZSk7Ci0gICAgICAgIGlmICgoYXR0ci52YWx1ZVR5cGUgJiBWQUxVRV9SQU5HRSkgPT0gMCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiQXR0cmlidXRlIGlzIG5vdCBhIHJhbmdlISIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBhdHRyLm1heFZhbHVlOwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0QXR0cmlidXRlTWluVmFsdWUoU3RyaW5nIGVsZW1lbnROYW1lLCBTdHJpbmcgYXR0ck5hbWUpIHsKLSAgICAgICAgQXR0bGlzdCBhdHRyID0gZmluZEF0dHJpYnV0ZShlbGVtZW50TmFtZSwgYXR0ck5hbWUpOwotICAgICAgICBpZiAoKGF0dHIudmFsdWVUeXBlICYgVkFMVUVfUkFOR0UpID09IDApIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkF0dHJpYnV0ZSBpcyBub3QgYSByYW5nZSEiKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gYXR0ci5taW5WYWx1ZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0QXR0cmlidXRlTmFtZXMoU3RyaW5nIGVsZW1lbnROYW1lKSB7Ci0gICAgICAgIEVsZW1lbnQgZWxlbWVudCA9IGZpbmRFbGVtZW50KGVsZW1lbnROYW1lKTsKLSAgICAgICAgcmV0dXJuIGVsZW1lbnQuYXR0cmlidXRlcy5rZXlTZXQoKS50b0FycmF5KG5ldyBTdHJpbmdbZWxlbWVudC5hdHRyaWJ1dGVzLnNpemUoKV0pOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0QXR0cmlidXRlVmFsdWVUeXBlKFN0cmluZyBlbGVtZW50TmFtZSwgU3RyaW5nIGF0dHJOYW1lKSB7Ci0gICAgICAgIEF0dGxpc3QgYXR0ciA9IGZpbmRBdHRyaWJ1dGUoZWxlbWVudE5hbWUsIGF0dHJOYW1lKTsKLSAgICAgICAgcmV0dXJuIGF0dHIudmFsdWVUeXBlOwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRDaGlsZE5hbWVzKFN0cmluZyBlbGVtZW50TmFtZSkgewotICAgICAgICBFbGVtZW50IGVsZW1lbnQgPSBmaW5kRWxlbWVudChlbGVtZW50TmFtZSk7Ci0gICAgICAgIGlmIChlbGVtZW50LmNoaWxkUG9saWN5ID09IENISUxEX1BPTElDWV9FTVBUWSkgeyAvLyBFbGVtZW50IGNhbm5vdCBoYXZlCi0gICAgICAgICAgICAvLyBjaGlsZHJlbgotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGVsZW1lbnQuY2hpbGRyZW4udG9BcnJheShuZXcgU3RyaW5nW2VsZW1lbnQuY2hpbGRyZW4uc2l6ZSgpXSk7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRDaGlsZFBvbGljeShTdHJpbmcgZWxlbWVudE5hbWUpIHsKLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICByZXR1cm4gZWxlbWVudC5jaGlsZFBvbGljeTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIGdldEVsZW1lbnREZXNjcmlwdGlvbihTdHJpbmcgZWxlbWVudE5hbWUsIExvY2FsZSBsb2NhbGUpIHsKLSAgICAgICAgZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOyAvLyBDaGVjayBpZiB0aGVyZSBpcyBzdWNoIGVsZW1lbnQKLSAgICAgICAgcmV0dXJuIGdldFJlc291cmNlU3RyaW5nKGVsZW1lbnROYW1lLCBsb2NhbGUpOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0RWxlbWVudE1heENoaWxkcmVuKFN0cmluZyBlbGVtZW50TmFtZSkgewotICAgICAgICBFbGVtZW50IGVsZW1lbnQgPSBmaW5kRWxlbWVudChlbGVtZW50TmFtZSk7Ci0gICAgICAgIGlmIChlbGVtZW50LmNoaWxkUG9saWN5ICE9IENISUxEX1BPTElDWV9SRVBFQVQpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkNoaWxkIHBvbGljeSBpcyBub3QgQ0hJTERfUE9MSUNZX1JFUEVBVCEiKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZWxlbWVudC5tYXhDaGlsZHJlbjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldEVsZW1lbnRNaW5DaGlsZHJlbihTdHJpbmcgZWxlbWVudE5hbWUpIHsKLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICBpZiAoZWxlbWVudC5jaGlsZFBvbGljeSAhPSBDSElMRF9QT0xJQ1lfUkVQRUFUKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJDaGlsZCBwb2xpY3kgaXMgbm90IENISUxEX1BPTElDWV9SRVBFQVQhIik7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGVsZW1lbnQubWluQ2hpbGRyZW47Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRPYmplY3RBcnJheU1heExlbmd0aChTdHJpbmcgZWxlbWVudE5hbWUpIHsKLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICBPYmplY3RWYWx1ZSB2ID0gZWxlbWVudC5vYmplY3RWYWx1ZTsKLSAgICAgICAgaWYgKHYgPT0gbnVsbCB8fCB2LnZhbHVlVHlwZSAhPSBWQUxVRV9MSVNUKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJOb3QgYSBsaXN0ISIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB2LmFycmF5TWF4TGVuZ3RoOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0T2JqZWN0QXJyYXlNaW5MZW5ndGgoU3RyaW5nIGVsZW1lbnROYW1lKSB7Ci0gICAgICAgIEVsZW1lbnQgZWxlbWVudCA9IGZpbmRFbGVtZW50KGVsZW1lbnROYW1lKTsKLSAgICAgICAgT2JqZWN0VmFsdWUgdiA9IGVsZW1lbnQub2JqZWN0VmFsdWU7Ci0gICAgICAgIGlmICh2ID09IG51bGwgfHwgdi52YWx1ZVR5cGUgIT0gVkFMVUVfTElTVCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiTm90IGEgbGlzdCEiKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdi5hcnJheU1pbkxlbmd0aDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQ2xhc3M8Pz4gZ2V0T2JqZWN0Q2xhc3MoU3RyaW5nIGVsZW1lbnROYW1lKSB7Ci0gICAgICAgIE9iamVjdFZhbHVlIHYgPSBmaW5kT2JqZWN0VmFsdWUoZWxlbWVudE5hbWUpOwotICAgICAgICByZXR1cm4gdi5jbGFzc1R5cGU7Ci0gICAgfQotCi0gICAgcHVibGljIE9iamVjdCBnZXRPYmplY3REZWZhdWx0VmFsdWUoU3RyaW5nIGVsZW1lbnROYW1lKSB7Ci0gICAgICAgIE9iamVjdFZhbHVlIHYgPSBmaW5kT2JqZWN0VmFsdWUoZWxlbWVudE5hbWUpOwotICAgICAgICByZXR1cm4gdi5kZWZhdWx0VmFsdWU7Ci0gICAgfQotCi0gICAgcHVibGljIE9iamVjdFtdIGdldE9iamVjdEVudW1lcmF0aW9ucyhTdHJpbmcgZWxlbWVudE5hbWUpIHsKLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICBPYmplY3RWYWx1ZSB2ID0gZWxlbWVudC5vYmplY3RWYWx1ZTsKLSAgICAgICAgaWYgKHYgPT0gbnVsbCB8fCB2LnZhbHVlVHlwZSAhPSBWQUxVRV9FTlVNRVJBVElPTikgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiTm90IGFuIGVudW1lcmF0aW9uISIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB2LmVudW1lcmF0ZWRWYWx1ZXMudG9BcnJheSgpOwotICAgIH0KLQotICAgIHB1YmxpYyBDb21wYXJhYmxlPD8+IGdldE9iamVjdE1heFZhbHVlKFN0cmluZyBlbGVtZW50TmFtZSkgewotICAgICAgICBFbGVtZW50IGVsZW1lbnQgPSBmaW5kRWxlbWVudChlbGVtZW50TmFtZSk7Ci0gICAgICAgIE9iamVjdFZhbHVlIHYgPSBlbGVtZW50Lm9iamVjdFZhbHVlOwotICAgICAgICBpZiAodiA9PSBudWxsIHx8ICh2LnZhbHVlVHlwZSAmIFZBTFVFX1JBTkdFKSA9PSAwKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJOb3QgYSByYW5nZSEiKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdi5tYXhWYWx1ZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQ29tcGFyYWJsZTw/PiBnZXRPYmplY3RNaW5WYWx1ZShTdHJpbmcgZWxlbWVudE5hbWUpIHsKLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICBPYmplY3RWYWx1ZSB2ID0gZWxlbWVudC5vYmplY3RWYWx1ZTsKLSAgICAgICAgaWYgKHYgPT0gbnVsbCB8fCAodi52YWx1ZVR5cGUgJiBWQUxVRV9SQU5HRSkgPT0gMCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiTm90IGEgcmFuZ2UhIik7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHYubWluVmFsdWU7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRPYmplY3RWYWx1ZVR5cGUoU3RyaW5nIGVsZW1lbnROYW1lKSB7Ci0gICAgICAgIEVsZW1lbnQgZWxlbWVudCA9IGZpbmRFbGVtZW50KGVsZW1lbnROYW1lKTsKLSAgICAgICAgaWYgKGVsZW1lbnQub2JqZWN0VmFsdWUgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIFZBTFVFX05PTkU7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGVsZW1lbnQub2JqZWN0VmFsdWUudmFsdWVUeXBlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHJlc291cmNlIGJhc2UgbmFtZSBmb3IgbG9jYXRpbmcgUmVzb3VyY2VCdW5kbGVzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGN1cnJlbnQgcmVzb3VyY2UgYmFzZSBuYW1lLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmcgZ2V0UmVzb3VyY2VCYXNlTmFtZSgpIHsKLSAgICAgICAgcmV0dXJuIHJlc291cmNlQmFzZU5hbWU7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZyBnZXRSb290TmFtZSgpIHsKLSAgICAgICAgcmV0dXJuIHJvb3ROYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN0YW5kYXJkIGZvcm1hdCBpbnN0YW5jZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBJSU9NZXRhZGF0YUZvcm1hdCBpbnN0YW5jZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIElJT01ldGFkYXRhRm9ybWF0IGdldFN0YW5kYXJkRm9ybWF0SW5zdGFuY2UoKSB7Ci0gICAgICAgIGlmIChzdGFuZGFyZEZvcm1hdCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBzdGFuZGFyZEZvcm1hdCA9IG5ldyBJSU9TdGFuZGFyZE1ldGFkYXRhRm9ybWF0KCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gc3RhbmRhcmRGb3JtYXQ7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNBdHRyaWJ1dGVSZXF1aXJlZChTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyTmFtZSkgewotICAgICAgICByZXR1cm4gZmluZEF0dHJpYnV0ZShlbGVtZW50TmFtZSwgYXR0ck5hbWUpLnJlcXVpcmVkOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgdGhlIHNwZWNpZmllZCBhdHRyaWJ1dGUgZnJvbSB0aGUgc3BlY2lmaWVkIGVsZW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGVsZW1lbnQgbmFtZS4KLSAgICAgKiBAcGFyYW0gYXR0ck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgYXR0cmlidXRlIG5hbWUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgcmVtb3ZlQXR0cmlidXRlKFN0cmluZyBlbGVtZW50TmFtZSwgU3RyaW5nIGF0dHJOYW1lKSB7Ci0gICAgICAgIEVsZW1lbnQgZWxlbWVudCA9IGZpbmRFbGVtZW50KGVsZW1lbnROYW1lKTsKLSAgICAgICAgZWxlbWVudC5hdHRyaWJ1dGVzLnJlbW92ZShhdHRyTmFtZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgc3BlY2lmaWVkIGVsZW1lbnQgZnJvbSB0aGlzIGZvcm1hdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZWxlbWVudE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgZWxlbWVudCBuYW1lLgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHJlbW92ZUVsZW1lbnQoU3RyaW5nIGVsZW1lbnROYW1lKSB7Ci0gICAgICAgIEVsZW1lbnQgZWxlbWVudDsKLSAgICAgICAgaWYgKChlbGVtZW50ID0gZWxlbWVudEhhc2guZ2V0KGVsZW1lbnROYW1lKSkgIT0gbnVsbCkgewotICAgICAgICAgICAgZWxlbWVudEhhc2gucmVtb3ZlKGVsZW1lbnROYW1lKTsKLSAgICAgICAgICAgIGZvciAoRWxlbWVudCBlIDogZWxlbWVudEhhc2gudmFsdWVzKCkpIHsKLSAgICAgICAgICAgICAgICBlLmNoaWxkcmVuLnJlbW92ZShlbGVtZW50Lm5hbWUpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyB0aGUgb2JqZWN0IHZhbHVlIGZyb20gdGhlIHNwZWNpZmllZCBlbGVtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBlbGVtZW50TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGVsZW1lbnQgbmFtZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCByZW1vdmVPYmplY3RWYWx1ZShTdHJpbmcgZWxlbWVudE5hbWUpIHsKLSAgICAgICAgRWxlbWVudCBlbGVtZW50ID0gZmluZEVsZW1lbnQoZWxlbWVudE5hbWUpOwotICAgICAgICBlbGVtZW50Lm9iamVjdFZhbHVlID0gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIGEgbmV3IGJhc2UgbmFtZSBmb3IgUmVzb3VyY2VCdW5kbGVzIGNvbnRhaW5pbmcgZGVzY3JpcHRpb25zIG9mCi0gICAgICogZWxlbWVudHMgYW5kIGF0dHJpYnV0ZXMgZm9yIHRoaXMgZm9ybWF0LgotICAgICAqIAotICAgICAqIEBwYXJhbSByZXNvdXJjZUJhc2VOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmV3IHJlc291cmNlIGJhc2UgbmFtZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgdm9pZCBzZXRSZXNvdXJjZUJhc2VOYW1lKFN0cmluZyByZXNvdXJjZUJhc2VOYW1lKSB7Ci0gICAgICAgIGlmIChyZXNvdXJjZUJhc2VOYW1lID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInJlc291cmNlQmFzZU5hbWUgPT0gbnVsbCEiKTsKLSAgICAgICAgfQotICAgICAgICB0aGlzLnJlc291cmNlQmFzZU5hbWUgPSByZXNvdXJjZUJhc2VOYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBFbGVtZW50LgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCB7Ci0gICAgICAgICJDbGFzc1dpdGhvdXRDb25zdHJ1Y3RvciIKLSAgICB9KQotICAgIHByaXZhdGUgY2xhc3MgRWxlbWVudCB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBuYW1lLgotICAgICAgICAgKi8KLSAgICAgICAgU3RyaW5nIG5hbWU7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjaGlsZHJlbi4KLSAgICAgICAgICovCi0gICAgICAgIEFycmF5TGlzdDxTdHJpbmc+IGNoaWxkcmVuID0gbmV3IEFycmF5TGlzdDxTdHJpbmc+KCk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBhdHRyaWJ1dGVzLgotICAgICAgICAgKi8KLSAgICAgICAgSGFzaE1hcDxTdHJpbmcsIEF0dGxpc3Q+IGF0dHJpYnV0ZXMgPSBuZXcgSGFzaE1hcDxTdHJpbmcsIEF0dGxpc3Q+KCk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBtaW4gY2hpbGRyZW4uCi0gICAgICAgICAqLwotICAgICAgICBpbnQgbWluQ2hpbGRyZW47Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBtYXggY2hpbGRyZW4uCi0gICAgICAgICAqLwotICAgICAgICBpbnQgbWF4Q2hpbGRyZW47Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBjaGlsZCBwb2xpY3kuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgY2hpbGRQb2xpY3k7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBvYmplY3QgdmFsdWUuCi0gICAgICAgICAqLwotICAgICAgICBPYmplY3RWYWx1ZSBvYmplY3RWYWx1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgQXR0bGlzdC4KLSAgICAgKi8KLSAgICBAU3VwcHJlc3NXYXJuaW5ncyggewotICAgICAgICAiQ2xhc3NXaXRob3V0Q29uc3RydWN0b3IiCi0gICAgfSkKLSAgICBwcml2YXRlIGNsYXNzIEF0dGxpc3QgewotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbmFtZS4KLSAgICAgICAgICovCi0gICAgICAgIFN0cmluZyBuYW1lOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgZGF0YSB0eXBlLgotICAgICAgICAgKi8KLSAgICAgICAgaW50IGRhdGFUeXBlOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcmVxdWlyZWQuCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIHJlcXVpcmVkOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbGlzdCBtaW4gbGVuZ3RoLgotICAgICAgICAgKi8KLSAgICAgICAgaW50IGxpc3RNaW5MZW5ndGg7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBsaXN0IG1heCBsZW5ndGguCi0gICAgICAgICAqLwotICAgICAgICBpbnQgbGlzdE1heExlbmd0aDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGRlZmF1bHQgdmFsdWUuCi0gICAgICAgICAqLwotICAgICAgICBTdHJpbmcgZGVmYXVsdFZhbHVlOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgZW51bWVyYXRlZCB2YWx1ZXMuCi0gICAgICAgICAqLwotICAgICAgICBMaXN0PFN0cmluZz4gZW51bWVyYXRlZFZhbHVlczsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG1pbiB2YWx1ZS4KLSAgICAgICAgICovCi0gICAgICAgIFN0cmluZyBtaW5WYWx1ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG1heCB2YWx1ZS4KLSAgICAgICAgICovCi0gICAgICAgIFN0cmluZyBtYXhWYWx1ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG1pbiBpbmNsdXNpdmUuCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIG1pbkluY2x1c2l2ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG1heCBpbmNsdXNpdmUuCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIG1heEluY2x1c2l2ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHZhbHVlIHR5cGUuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgdmFsdWVUeXBlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBPYmplY3RWYWx1ZS4KLSAgICAgKi8KLSAgICBAU3VwcHJlc3NXYXJuaW5ncyggewotICAgICAgICAiQ2xhc3NXaXRob3V0Q29uc3RydWN0b3IiCi0gICAgfSkKLSAgICBwcml2YXRlIGNsYXNzIE9iamVjdFZhbHVlPFQ+IHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGNsYXNzIHR5cGUuCi0gICAgICAgICAqLwotICAgICAgICBDbGFzczxUPiBjbGFzc1R5cGU7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBhcnJheSBtaW4gbGVuZ3RoLgotICAgICAgICAgKi8KLSAgICAgICAgaW50IGFycmF5TWluTGVuZ3RoOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgYXJyYXkgbWF4IGxlbmd0aC4KLSAgICAgICAgICovCi0gICAgICAgIGludCBhcnJheU1heExlbmd0aDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGRlZmF1bHQgdmFsdWUuCi0gICAgICAgICAqLwotICAgICAgICBUIGRlZmF1bHRWYWx1ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGVudW1lcmF0ZWQgdmFsdWVzLgotICAgICAgICAgKi8KLSAgICAgICAgTGlzdDw/IGV4dGVuZHMgVD4gZW51bWVyYXRlZFZhbHVlczsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG1pbiB2YWx1ZS4KLSAgICAgICAgICovCi0gICAgICAgIENvbXBhcmFibGU8PyBzdXBlciBUPiBtaW5WYWx1ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG1heCB2YWx1ZS4KLSAgICAgICAgICovCi0gICAgICAgIENvbXBhcmFibGU8PyBzdXBlciBUPiBtYXhWYWx1ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG1pbiBpbmNsdXNpdmUuCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIG1pbkluY2x1c2l2ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG1heCBpbmNsdXNpdmUuCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIG1heEluY2x1c2l2ZTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHZhbHVlIHR5cGUuCi0gICAgICAgICAqLwotICAgICAgICBpbnQgdmFsdWVUeXBlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbmQgZWxlbWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWUuCi0gICAgICogQHJldHVybiB0aGUgZWxlbWVudC4KLSAgICAgKi8KLSAgICBwcml2YXRlIEVsZW1lbnQgZmluZEVsZW1lbnQoU3RyaW5nIG5hbWUpIHsKLSAgICAgICAgRWxlbWVudCBlbGVtZW50OwotICAgICAgICBpZiAoKGVsZW1lbnQgPSBlbGVtZW50SGFzaC5nZXQobmFtZSkpID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImVsZW1lbnQgbmFtZSBpcyBudWxsIG9yIG5vIHN1Y2ggZWxlbWVudDogIiArIG5hbWUpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGVsZW1lbnQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRmluZCBhdHRyaWJ1dGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGVsZW1lbnROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZWxlbWVudCBuYW1lLgotICAgICAqIEBwYXJhbSBhdHRyaWJ1dGVOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgYXR0cmlidXRlIG5hbWUuCi0gICAgICogQHJldHVybiB0aGUgYXR0bGlzdC4KLSAgICAgKi8KLSAgICBwcml2YXRlIEF0dGxpc3QgZmluZEF0dHJpYnV0ZShTdHJpbmcgZWxlbWVudE5hbWUsIFN0cmluZyBhdHRyaWJ1dGVOYW1lKSB7Ci0gICAgICAgIEVsZW1lbnQgZWxlbWVudCA9IGZpbmRFbGVtZW50KGVsZW1lbnROYW1lKTsKLSAgICAgICAgQXR0bGlzdCBhdHRyaWJ1dGU7Ci0gICAgICAgIGlmICgoYXR0cmlidXRlID0gZWxlbWVudC5hdHRyaWJ1dGVzLmdldChhdHRyaWJ1dGVOYW1lKSkgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiYXR0cmlidXRlIG5hbWUgaXMgbnVsbCBvciBubyBzdWNoIGF0dHJpYnV0ZTogIgotICAgICAgICAgICAgICAgICAgICArIGF0dHJpYnV0ZU5hbWUpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGF0dHJpYnV0ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaW5kIG9iamVjdCB2YWx1ZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZWxlbWVudE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBlbGVtZW50IG5hbWUuCi0gICAgICogQHJldHVybiB0aGUgb2JqZWN0IHZhbHVlLgotICAgICAqLwotICAgIHByaXZhdGUgT2JqZWN0VmFsdWUgZmluZE9iamVjdFZhbHVlKFN0cmluZyBlbGVtZW50TmFtZSkgewotICAgICAgICBFbGVtZW50IGVsZW1lbnQgPSBmaW5kRWxlbWVudChlbGVtZW50TmFtZSk7Ci0gICAgICAgIE9iamVjdFZhbHVlIHYgPSBlbGVtZW50Lm9iamVjdFZhbHVlOwotICAgICAgICBpZiAodiA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJObyBvYmplY3Qgd2l0aGluIGVsZW1lbnQiKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSByZXNvdXJjZSBzdHJpbmcuCi0gICAgICogCi0gICAgICogQHBhcmFtIGtleQotICAgICAqICAgICAgICAgICAgdGhlIGtleS4KLSAgICAgKiBAcGFyYW0gbG9jYWxlCi0gICAgICogICAgICAgICAgICB0aGUgbG9jYWxlLgotICAgICAqIEByZXR1cm4gdGhlIHJlc291cmNlIHN0cmluZy4KLSAgICAgKi8KLSAgICBwcml2YXRlIFN0cmluZyBnZXRSZXNvdXJjZVN0cmluZyhTdHJpbmcga2V5LCBMb2NhbGUgbG9jYWxlKSB7Ci0gICAgICAgIGlmIChsb2NhbGUgPT0gbnVsbCkgewotICAgICAgICAgICAgbG9jYWxlID0gTG9jYWxlLmdldERlZmF1bHQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIEdldCB0aGUgY29udGV4dCBjbGFzcyBsb2FkZXIgYW5kIHRyeSB0byBsb2NhdGUgdGhlIGJ1bmRsZSB3aXRoIGl0Ci0gICAgICAgIC8vIGZpcnN0Ci0gICAgICAgIENsYXNzTG9hZGVyIGNvbnRleHRDbGFzc2xvYWRlciA9IEFjY2Vzc0NvbnRyb2xsZXIKLSAgICAgICAgICAgICAgICAuZG9Qcml2aWxlZ2VkKG5ldyBQcml2aWxlZ2VkQWN0aW9uPENsYXNzTG9hZGVyPigpIHsKLSAgICAgICAgICAgICAgICAgICAgcHVibGljIENsYXNzTG9hZGVyIHJ1bigpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBUaHJlYWQuY3VycmVudFRocmVhZCgpLmdldENvbnRleHRDbGFzc0xvYWRlcigpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSk7Ci0KLSAgICAgICAgLy8gTm93IHRyeSB0byBnZXQgdGhlIHJlc291cmNlIGJ1bmRsZQotICAgICAgICBSZXNvdXJjZUJ1bmRsZSByYjsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJiID0gUmVzb3VyY2VCdW5kbGUuZ2V0QnVuZGxlKHJlc291cmNlQmFzZU5hbWUsIGxvY2FsZSwgY29udGV4dENsYXNzbG9hZGVyKTsKLSAgICAgICAgfSBjYXRjaCAoTWlzc2luZ1Jlc291cmNlRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgcmIgPSBSZXNvdXJjZUJ1bmRsZS5nZXRCdW5kbGUocmVzb3VyY2VCYXNlTmFtZSwgbG9jYWxlKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKE1pc3NpbmdSZXNvdXJjZUV4Y2VwdGlvbiBlMSkgewotICAgICAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiByYi5nZXRTdHJpbmcoa2V5KTsKLSAgICAgICAgfSBjYXRjaCAoTWlzc2luZ1Jlc291cmNlRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9IGNhdGNoIChDbGFzc0Nhc3RFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7IC8vIE5vdCBhIHN0cmluZyByZXNvdXJjZQotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vbWV0YWRhdGEvSUlPTWV0YWRhdGFOb2RlLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YU5vZGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYWRjNmQ2Ny4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YU5vZGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEwNzAgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvLm1ldGFkYXRhOwotCi1pbXBvcnQgamF2YS51dGlsLkFycmF5TGlzdDsKLWltcG9ydCBqYXZhLnV0aWwuTGlzdDsKLQotaW1wb3J0IG9yZy53M2MuZG9tLkF0dHI7Ci1pbXBvcnQgb3JnLnczYy5kb20uRE9NRXhjZXB0aW9uOwotaW1wb3J0IG9yZy53M2MuZG9tLkRvY3VtZW50OwotaW1wb3J0IG9yZy53M2MuZG9tLkVsZW1lbnQ7Ci1pbXBvcnQgb3JnLnczYy5kb20uTmFtZWROb2RlTWFwOwotaW1wb3J0IG9yZy53M2MuZG9tLk5vZGU7Ci1pbXBvcnQgb3JnLnczYy5kb20uTm9kZUxpc3Q7Ci0KLS8vPz8/QVdUCi0vL2ltcG9ydCBvcmcudzNjLmRvbS5UeXBlSW5mbzsKLS8vaW1wb3J0IG9yZy53M2MuZG9tLlVzZXJEYXRhSGFuZGxlcjsKLQotLyoqCi0gKiBUaGUgQ2xhc3MgSUlPTWV0YWRhdGFOb2RlIHJlcHJlc2VudHMgYSBub2RlIG9mIHRoZSAoRE9NLXN0eWxlKSBtZXRhZGF0YSB0cmVlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIElJT01ldGFkYXRhTm9kZSBpbXBsZW1lbnRzIEVsZW1lbnQsIE5vZGVMaXN0IHsKLQotICAgIC8qKgotICAgICAqIFRoZSBub2RlIG5hbWUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBTdHJpbmcgbm9kZU5hbWU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbm9kZSB2YWx1ZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIFN0cmluZyBub2RlVmFsdWU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYXR0cmlidXRlcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIElJT01ldGFkYXRhTm9kZUxpc3QgYXR0cnMgPSBuZXcgSUlPTWV0YWRhdGFOb2RlTGlzdChuZXcgQXJyYXlMaXN0PElJT01ldGFkYXRhTm9kZT4oKSk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcGFyZW50IG5vZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBJSU9NZXRhZGF0YU5vZGUgcGFyZW50OwotCi0gICAgLyoqCi0gICAgICogVGhlIGZpcnN0IGNoaWxkIG5vZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBJSU9NZXRhZGF0YU5vZGUgZmlyc3RDaGlsZDsKLQotICAgIC8qKgotICAgICAqIFRoZSBsYXN0IGNoaWxkIG5vZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBJSU9NZXRhZGF0YU5vZGUgbGFzdENoaWxkOwotCi0gICAgLyoqCi0gICAgICogVGhlIHByZXZpb3VzIHNpYmxpbmcuCi0gICAgICovCi0gICAgcHJpdmF0ZSBJSU9NZXRhZGF0YU5vZGUgcHJldmlvdXNTaWJsaW5nOwotCi0gICAgLyoqCi0gICAgICogVGhlIG5leHQgc2libGluZy4KLSAgICAgKi8KLSAgICBwcml2YXRlIElJT01ldGFkYXRhTm9kZSBuZXh0U2libGluZzsKLQotICAgIC8qKgotICAgICAqIFRoZSBudW1iZXIgb2YgY2hpbGRyZW4uCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgbkNoaWxkcmVuOwotCi0gICAgLyoqCi0gICAgICogVGhlIHVzZXIgb2JqZWN0IGFzc29jaWF0ZWQgd2l0aCB0aGlzIG5vZGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBPYmplY3QgdXNlck9iamVjdDsKLQotICAgIC8qKgotICAgICAqIFRoZSB0ZXh0IGNvbnRlbnQgb2YgdGhpcyBub2RlLgotICAgICAqLwotICAgIHByaXZhdGUgU3RyaW5nIHRleHRDb250ZW50OwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGVtcHR5IG5vZGUuCi0gICAgICovCi0gICAgcHVibGljIElJT01ldGFkYXRhTm9kZSgpIHsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgZW1wdHkgbm9kZSB3aXRoIHRoZSBzcGVjaWZpZWQgbmFtZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbm9kZU5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBub2RlIG5hbWUuCi0gICAgICovCi0gICAgcHVibGljIElJT01ldGFkYXRhTm9kZShTdHJpbmcgbm9kZU5hbWUpIHsKLSAgICAgICAgdGhpcy5ub2RlTmFtZSA9IG5vZGVOYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJSU9NZXRhZGF0YU5vZGUgd2l0aCB0aGUgc3BlY2lmaWVkIG5hbWUgYW5kIHZhbHVlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBub2RlTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5vZGUgbmFtZS4KLSAgICAgKiBAcGFyYW0gbm9kZVZhbHVlCi0gICAgICogICAgICAgICAgICB0aGUgbm9kZSB2YWx1ZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIElJT01ldGFkYXRhTm9kZShTdHJpbmcgbm9kZU5hbWUsIFN0cmluZyBub2RlVmFsdWUpIHsKLSAgICAgICAgdGhpcy5ub2RlTmFtZSA9IG5vZGVOYW1lOwotICAgICAgICB0aGlzLm5vZGVWYWx1ZSA9IG5vZGVWYWx1ZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIGdldFRhZ05hbWUoKSB7Ci0gICAgICAgIHJldHVybiBub2RlTmFtZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIGdldEF0dHJpYnV0ZShTdHJpbmcgbmFtZSkgewotICAgICAgICBBdHRyIGF0dHJOb2RlID0gKEF0dHIpYXR0cnMuZ2V0TmFtZWRJdGVtKG5hbWUpOwotICAgICAgICByZXR1cm4gKGF0dHJOb2RlID09IG51bGwpID8gIiIgOiBhdHRyTm9kZS5nZXRWYWx1ZSgpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldEF0dHJpYnV0ZShTdHJpbmcgbmFtZSwgU3RyaW5nIHZhbHVlKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgQXR0ciBhdHRyID0gKEF0dHIpYXR0cnMuZ2V0TmFtZWRJdGVtKG5hbWUpOwotICAgICAgICBpZiAoYXR0ciAhPSBudWxsKSB7Ci0gICAgICAgICAgICBhdHRyLnNldFZhbHVlKHZhbHVlKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGF0dHJzLmxpc3QuYWRkKG5ldyBJSU9NZXRhZGF0YUF0dHIobmFtZSwgdmFsdWUsIHRoaXMpKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlbW92ZUF0dHJpYnV0ZShTdHJpbmcgbmFtZSkgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgIElJT01ldGFkYXRhQXR0ciBhdHRyID0gKElJT01ldGFkYXRhQXR0cilhdHRycy5nZXROYW1lZEl0ZW0obmFtZSk7Ci0gICAgICAgIGlmIChhdHRyICE9IG51bGwpIHsKLSAgICAgICAgICAgIGF0dHIuc2V0T3duZXJFbGVtZW50KG51bGwpOwotICAgICAgICAgICAgYXR0cnMubGlzdC5yZW1vdmUoYXR0cik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgQXR0ciBnZXRBdHRyaWJ1dGVOb2RlKFN0cmluZyBuYW1lKSB7Ci0gICAgICAgIHJldHVybiAoQXR0cilhdHRycy5nZXROYW1lZEl0ZW0obmFtZSk7Ci0gICAgfQotCi0gICAgcHVibGljIEF0dHIgc2V0QXR0cmlidXRlTm9kZShBdHRyIG5ld0F0dHIpIHRocm93cyBET01FeGNlcHRpb24gewotICAgICAgICAvLyBDaGVjayBpZiB0aGlzIGF0dHJpYnV0ZSBpcyBhbHJlYWR5IGluIHVzZS4KLSAgICAgICAgRWxlbWVudCBvd25lciA9IG5ld0F0dHIuZ2V0T3duZXJFbGVtZW50KCk7Ci0gICAgICAgIGlmIChvd25lciAhPSBudWxsKSB7Ci0gICAgICAgICAgICBpZiAob3duZXIgPT0gdGhpcykgeyAvLyBSZXBsYWNpbmcgYW4gYXR0cmlidXRlIG5vZGUgYnkgaXRzZWxmIGhhcyBubwotICAgICAgICAgICAgICAgIC8vIGVmZmVjdAotICAgICAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRE9NRXhjZXB0aW9uKERPTUV4Y2VwdGlvbi5JTlVTRV9BVFRSSUJVVEVfRVJSLAotICAgICAgICAgICAgICAgICAgICAgICAgIkF0dHJpYnV0ZSBpcyBhbHJlYWR5IGluIHVzZSIpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgU3RyaW5nIG5hbWUgPSBuZXdBdHRyLmdldE5hbWUoKTsKLSAgICAgICAgQXR0ciBvbGRBdHRyID0gZ2V0QXR0cmlidXRlTm9kZShuYW1lKTsKLSAgICAgICAgaWYgKG9sZEF0dHIgIT0gbnVsbCkgewotICAgICAgICAgICAgcmVtb3ZlQXR0cmlidXRlTm9kZShvbGRBdHRyKTsKLSAgICAgICAgfQotCi0gICAgICAgIElJT01ldGFkYXRhQXR0ciBpaW9BdHRyOwotICAgICAgICBpZiAobmV3QXR0ciBpbnN0YW5jZW9mIElJT01ldGFkYXRhQXR0cikgewotICAgICAgICAgICAgaWlvQXR0ciA9IChJSU9NZXRhZGF0YUF0dHIpbmV3QXR0cjsKLSAgICAgICAgICAgIGlpb0F0dHIuc2V0T3duZXJFbGVtZW50KHRoaXMpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaWlvQXR0ciA9IG5ldyBJSU9NZXRhZGF0YUF0dHIobmFtZSwgbmV3QXR0ci5nZXRWYWx1ZSgpLCB0aGlzKTsKLSAgICAgICAgfQotCi0gICAgICAgIGF0dHJzLmxpc3QuYWRkKGlpb0F0dHIpOwotCi0gICAgICAgIHJldHVybiBvbGRBdHRyOwotICAgIH0KLQotICAgIHB1YmxpYyBBdHRyIHJlbW92ZUF0dHJpYnV0ZU5vZGUoQXR0ciBvbGRBdHRyKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKCFhdHRycy5saXN0LnJlbW92ZShvbGRBdHRyKSkgeyAvLyBOb3QgZm91bmQKLSAgICAgICAgICAgIHRocm93IG5ldyBET01FeGNlcHRpb24oRE9NRXhjZXB0aW9uLk5PVF9GT1VORF9FUlIsICJObyBzdWNoIGF0dHJpYnV0ZSEiKTsKLSAgICAgICAgfQotCi0gICAgICAgICgoSUlPTWV0YWRhdGFBdHRyKW9sZEF0dHIpLnNldE93bmVyRWxlbWVudChudWxsKTsKLQotICAgICAgICByZXR1cm4gb2xkQXR0cjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTm9kZUxpc3QgZ2V0RWxlbWVudHNCeVRhZ05hbWUoU3RyaW5nIG5hbWUpIHsKLSAgICAgICAgQXJyYXlMaXN0PElJT01ldGFkYXRhTm9kZT4gbm9kZXMgPSBuZXcgQXJyYXlMaXN0PElJT01ldGFkYXRhTm9kZT4oKTsKLQotICAgICAgICAvLyBOb24tcmVjdXJzaXZlIHRyZWUgd2FsawotICAgICAgICBOb2RlIHBvcyA9IHRoaXM7Ci0KLSAgICAgICAgd2hpbGUgKHBvcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICBpZiAocG9zLmdldE5vZGVOYW1lKCkuZXF1YWxzKG5hbWUpKSB7Ci0gICAgICAgICAgICAgICAgbm9kZXMuYWRkKChJSU9NZXRhZGF0YU5vZGUpcG9zKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgTm9kZSBuZXh0Tm9kZSA9IHBvcy5nZXRGaXJzdENoaWxkKCk7Ci0KLSAgICAgICAgICAgIHdoaWxlIChuZXh0Tm9kZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHBvcyA9PSB0aGlzKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIG5leHROb2RlID0gcG9zLmdldE5leHRTaWJsaW5nKCk7Ci0KLSAgICAgICAgICAgICAgICBpZiAobmV4dE5vZGUgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBwb3MgPSBwb3MuZ2V0UGFyZW50Tm9kZSgpOwotCi0gICAgICAgICAgICAgICAgICAgIGlmIChwb3MgPT0gbnVsbCB8fCBwb3MgPT0gdGhpcykgewotICAgICAgICAgICAgICAgICAgICAgICAgbmV4dE5vZGUgPSBudWxsOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwb3MgPSBuZXh0Tm9kZTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBuZXcgSUlPTWV0YWRhdGFOb2RlTGlzdChub2Rlcyk7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZyBnZXRBdHRyaWJ1dGVOUyhTdHJpbmcgbmFtZXNwYWNlVVJJLCBTdHJpbmcgbG9jYWxOYW1lKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIGdldEF0dHJpYnV0ZShsb2NhbE5hbWUpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldEF0dHJpYnV0ZU5TKFN0cmluZyBuYW1lc3BhY2VVUkksIFN0cmluZyBxdWFsaWZpZWROYW1lLCBTdHJpbmcgdmFsdWUpCi0gICAgICAgICAgICB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgc2V0QXR0cmlidXRlKHF1YWxpZmllZE5hbWUsIHZhbHVlKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVBdHRyaWJ1dGVOUyhTdHJpbmcgbmFtZXNwYWNlVVJJLCBTdHJpbmcgbG9jYWxOYW1lKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgcmVtb3ZlQXR0cmlidXRlKGxvY2FsTmFtZSk7Ci0gICAgfQotCi0gICAgcHVibGljIEF0dHIgZ2V0QXR0cmlidXRlTm9kZU5TKFN0cmluZyBuYW1lc3BhY2VVUkksIFN0cmluZyBsb2NhbE5hbWUpIHRocm93cyBET01FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gZ2V0QXR0cmlidXRlTm9kZShsb2NhbE5hbWUpOwotICAgIH0KLQotICAgIHB1YmxpYyBBdHRyIHNldEF0dHJpYnV0ZU5vZGVOUyhBdHRyIG5ld0F0dHIpIHRocm93cyBET01FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gc2V0QXR0cmlidXRlTm9kZShuZXdBdHRyKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTm9kZUxpc3QgZ2V0RWxlbWVudHNCeVRhZ05hbWVOUyhTdHJpbmcgbmFtZXNwYWNlVVJJLCBTdHJpbmcgbG9jYWxOYW1lKQotICAgICAgICAgICAgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBnZXRFbGVtZW50c0J5VGFnTmFtZShsb2NhbE5hbWUpOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGhhc0F0dHJpYnV0ZShTdHJpbmcgbmFtZSkgewotICAgICAgICByZXR1cm4gYXR0cnMuZ2V0TmFtZWRJdGVtKG5hbWUpICE9IG51bGw7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaGFzQXR0cmlidXRlTlMoU3RyaW5nIG5hbWVzcGFjZVVSSSwgU3RyaW5nIGxvY2FsTmFtZSkgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBoYXNBdHRyaWJ1dGUobG9jYWxOYW1lKTsKLSAgICB9Ci0KLSAgICAvLyA/Pz9BV1QKLSAgICAvKgotICAgICAqIHB1YmxpYyBUeXBlSW5mbyBnZXRTY2hlbWFUeXBlSW5mbygpIHsgdGhyb3cgbmV3Ci0gICAgICogRE9NRXhjZXB0aW9uKERPTUV4Y2VwdGlvbi5OT1RfU1VQUE9SVEVEX0VSUiwgIk1ldGhvZCBub3Qgc3VwcG9ydGVkIik7IH0KLSAgICAgKi8KLQotICAgIC8qKgotICAgICAqIDxpPkRlc2NyaXB0aW9uIGNvcGllZCBmcm9tIGludGVyZmFjZTogb3JnLnczYy5kb20uRWxlbWVudCAoRE9NIExldmVsCi0gICAgICogMyk8L2k+Ci0gICAgICogPHA+Ci0gICAgICogSWYgdGhlIHBhcmFtZXRlciBpc0lkIGlzIHRydWUsIHRoaXMgbWV0aG9kIGRlY2xhcmVzIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBhdHRyaWJ1dGUgdG8gYmUgYSB1c2VyLWRldGVybWluZWQgSUQgYXR0cmlidXRlIC4gVGhpcyBhZmZlY3RzIHRoZSB2YWx1ZQotICAgICAqIG9mIEF0dHIuaXNJZCBhbmQgdGhlIGJlaGF2aW9yIG9mIERvY3VtZW50LmdldEVsZW1lbnRCeUlkLCBidXQgZG9lcyBub3QKLSAgICAgKiBjaGFuZ2UgYW55IHNjaGVtYSB0aGF0IG1heSBiZSBpbiB1c2UsIGluIHBhcnRpY3VsYXIgdGhpcyBkb2VzIG5vdCBhZmZlY3QKLSAgICAgKiB0aGUgQXR0ci5zY2hlbWFUeXBlSW5mbyBvZiB0aGUgc3BlY2lmaWVkIEF0dHIgbm9kZS4gVXNlIHRoZSB2YWx1ZSBmYWxzZQotICAgICAqIGZvciB0aGUgcGFyYW1ldGVyIGlzSWQgdG8gdW5kZWNsYXJlIGFuIGF0dHJpYnV0ZSBmb3IgYmVpbmcgYQotICAgICAqIHVzZXItZGV0ZXJtaW5lZCBJRCBhdHRyaWJ1dGUuIFRvIHNwZWNpZnkgYW4gYXR0cmlidXRlIGJ5IGxvY2FsIG5hbWUgYW5kCi0gICAgICogbmFtZXNwYWNlIFVSSSwgdXNlIHRoZSBzZXRJZEF0dHJpYnV0ZU5TIG1ldGhvZC4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHBhcmFtIG5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYW1lIG9mIHRoZSBhdHRyaWJ1dGUuCi0gICAgICogQHBhcmFtIGlzSWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbGFnIHdoaWNoIGRldGVybWluZXMgd2hldGhlciB0aGlzIGF0dHJpYnV0ZSBpcyBvZiB0eXBlCi0gICAgICogICAgICAgICAgICBJRC4KLSAgICAgKiBAdGhyb3dzIERPTUV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGEgRE9NIGVycm9yIG9jY3VycmVkIHdoaWxlIHNldHRpbmcgdGhlIGF0dHJpYnV0ZSB0eXBlLgotICAgICAqICAgICAgICAgICAgIDxwPgotICAgICAqICAgICAgICAgICAgIE5PX01PRElGSUNBVElPTl9BTExPV0VEX0VSUjogUmFpc2VkIGlmIHRoaXMgbm9kZSBpcyByZWFkb25seS4KLSAgICAgKiAgICAgICAgICAgICA8YnI+Ci0gICAgICogICAgICAgICAgICAgTk9UX0ZPVU5EX0VSUjogUmFpc2VkIGlmIHRoZSBzcGVjaWZpZWQgbm9kZSBpcyBub3QgYW4KLSAgICAgKiAgICAgICAgICAgICBhdHRyaWJ1dGUgb2YgdGhpcyBlbGVtZW50LgotICAgICAqICAgICAgICAgICAgIDwvcD4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRJZEF0dHJpYnV0ZShTdHJpbmcgbmFtZSwgYm9vbGVhbiBpc0lkKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IERPTUV4Y2VwdGlvbihET01FeGNlcHRpb24uTk9UX1NVUFBPUlRFRF9FUlIsICJNZXRob2Qgbm90IHN1cHBvcnRlZCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIDxpPkRlc2NyaXB0aW9uIGNvcGllZCBmcm9tIGludGVyZmFjZTogb3JnLnczYy5kb20uRWxlbWVudCAoRE9NIExldmVsCi0gICAgICogMyk8L2k+Ci0gICAgICogPHA+Ci0gICAgICogSWYgdGhlIHBhcmFtZXRlciBpc0lkIGlzIHRydWUsIHRoaXMgbWV0aG9kIGRlY2xhcmVzIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBhdHRyaWJ1dGUgdG8gYmUgYSB1c2VyLWRldGVybWluZWQgSUQgYXR0cmlidXRlIC4gVGhpcyBhZmZlY3RzIHRoZSB2YWx1ZQotICAgICAqIG9mIEF0dHIuaXNJZCBhbmQgdGhlIGJlaGF2aW9yIG9mIERvY3VtZW50LmdldEVsZW1lbnRCeUlkLCBidXQgZG9lcyBub3QKLSAgICAgKiBjaGFuZ2UgYW55IHNjaGVtYSB0aGF0IG1heSBiZSBpbiB1c2UsIGluIHBhcnRpY3VsYXIgdGhpcyBkb2VzIG5vdCBhZmZlY3QKLSAgICAgKiB0aGUgQXR0ci5zY2hlbWFUeXBlSW5mbyBvZiB0aGUgc3BlY2lmaWVkIEF0dHIgbm9kZS4gVXNlIHRoZSB2YWx1ZSBmYWxzZQotICAgICAqIGZvciB0aGUgcGFyYW1ldGVyIGlzSWQgdG8gdW5kZWNsYXJlIGFuIGF0dHJpYnV0ZSBmb3IgYmVpbmcgYQotICAgICAqIHVzZXItZGV0ZXJtaW5lZCBJRCBhdHRyaWJ1dGUuCi0gICAgICogPC9wPgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lc3BhY2VVUkkKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYW1lc3BhY2UgVVJJIG9mIHRoZSBhdHRyaWJ1dGUuCi0gICAgICogQHBhcmFtIGxvY2FsTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIGxvY2FsIG5hbWUgb2YgdGhlIGF0dHJpYnV0ZS4KLSAgICAgKiBAcGFyYW0gaXNJZAotICAgICAqICAgICAgICAgICAgdGhlIGZsYWcgd2hpY2ggZGV0ZXJtaW5lcyB3aGV0aGVyIHRoaXMgYXR0cmlidXRlIGlzIG9mIHR5cGUKLSAgICAgKiAgICAgICAgICAgIElELgotICAgICAqIEB0aHJvd3MgRE9NRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYSBET00gZXJyb3Igb2NjdXJyZWQgd2hpbGUgc2V0dGluZyB0aGUgYXR0cmlidXRlIHR5cGUuCi0gICAgICogICAgICAgICAgICAgPHA+Ci0gICAgICogICAgICAgICAgICAgTk9fTU9ESUZJQ0FUSU9OX0FMTE9XRURfRVJSOiBSYWlzZWQgaWYgdGhpcyBub2RlIGlzIHJlYWRvbmx5LgotICAgICAqICAgICAgICAgICAgIDxicj4KLSAgICAgKiAgICAgICAgICAgICBOT1RfRk9VTkRfRVJSOiBSYWlzZWQgaWYgdGhlIHNwZWNpZmllZCBub2RlIGlzIG5vdCBhbgotICAgICAqICAgICAgICAgICAgIGF0dHJpYnV0ZSBvZiB0aGlzIGVsZW1lbnQuCi0gICAgICogICAgICAgICAgICAgPC9wPgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldElkQXR0cmlidXRlTlMoU3RyaW5nIG5hbWVzcGFjZVVSSSwgU3RyaW5nIGxvY2FsTmFtZSwgYm9vbGVhbiBpc0lkKQotICAgICAgICAgICAgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBET01FeGNlcHRpb24oRE9NRXhjZXB0aW9uLk5PVF9TVVBQT1JURURfRVJSLCAiTWV0aG9kIG5vdCBzdXBwb3J0ZWQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8aT5EZXNjcmlwdGlvbiBjb3BpZWQgZnJvbSBpbnRlcmZhY2U6IG9yZy53M2MuZG9tLkVsZW1lbnQgKERPTSBMZXZlbAotICAgICAqIDMpPC9pPgotICAgICAqIDxwPgotICAgICAqIElmIHRoZSBwYXJhbWV0ZXIgaXNJZCBpcyB0cnVlLCB0aGlzIG1ldGhvZCBkZWNsYXJlcyB0aGUgc3BlY2lmaWVkCi0gICAgICogYXR0cmlidXRlIHRvIGJlIGEgdXNlci1kZXRlcm1pbmVkIElEIGF0dHJpYnV0ZSAuIFRoaXMgYWZmZWN0cyB0aGUgdmFsdWUKLSAgICAgKiBvZiBBdHRyLmlzSWQgYW5kIHRoZSBiZWhhdmlvciBvZiBEb2N1bWVudC5nZXRFbGVtZW50QnlJZCwgYnV0IGRvZXMgbm90Ci0gICAgICogY2hhbmdlIGFueSBzY2hlbWEgdGhhdCBtYXkgYmUgaW4gdXNlLCBpbiBwYXJ0aWN1bGFyIHRoaXMgZG9lcyBub3QgYWZmZWN0Ci0gICAgICogdGhlIEF0dHIuc2NoZW1hVHlwZUluZm8gb2YgdGhlIHNwZWNpZmllZCBBdHRyIG5vZGUuIFVzZSB0aGUgdmFsdWUgZmFsc2UKLSAgICAgKiBmb3IgdGhlIHBhcmFtZXRlciBpc0lkIHRvIHVuZGVjbGFyZSBhbiBhdHRyaWJ1dGUgZm9yIGJlaW5nIGEKLSAgICAgKiB1c2VyLWRldGVybWluZWQgSUQgYXR0cmlidXRlLgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaWRBdHRyCi0gICAgICogICAgICAgICAgICB0aGUgYXR0cmlidXRlIG5vZGUuCi0gICAgICogQHBhcmFtIGlzSWQKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbGFnIHdoaWNoIGRldGVybWluZXMgd2hldGhlciB0aGlzIGF0dHJpYnV0ZSBpcyBvZiB0eXBlCi0gICAgICogICAgICAgICAgICBJRC4KLSAgICAgKiBAdGhyb3dzIERPTUV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGEgRE9NIGVycm9yIG9jY3VycmVkIHdoaWxlIHNldHRpbmcgdGhlIGF0dHJpYnV0ZSB0eXBlLgotICAgICAqICAgICAgICAgICAgIDxwPgotICAgICAqICAgICAgICAgICAgIE5PX01PRElGSUNBVElPTl9BTExPV0VEX0VSUjogUmFpc2VkIGlmIHRoaXMgbm9kZSBpcyByZWFkb25seS4KLSAgICAgKiAgICAgICAgICAgICA8YnI+Ci0gICAgICogICAgICAgICAgICAgTk9UX0ZPVU5EX0VSUjogUmFpc2VkIGlmIHRoZSBzcGVjaWZpZWQgbm9kZSBpcyBub3QgYW4KLSAgICAgKiAgICAgICAgICAgICBhdHRyaWJ1dGUgb2YgdGhpcyBlbGVtZW50LgotICAgICAqICAgICAgICAgICAgIDwvcD4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRJZEF0dHJpYnV0ZU5vZGUoQXR0ciBpZEF0dHIsIGJvb2xlYW4gaXNJZCkgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgIHRocm93IG5ldyBET01FeGNlcHRpb24oRE9NRXhjZXB0aW9uLk5PVF9TVVBQT1JURURfRVJSLCAiTWV0aG9kIG5vdCBzdXBwb3J0ZWQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIGdldE5vZGVOYW1lKCkgewotICAgICAgICByZXR1cm4gbm9kZU5hbWU7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZyBnZXROb2RlVmFsdWUoKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIG5vZGVWYWx1ZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXROb2RlVmFsdWUoU3RyaW5nIG5vZGVWYWx1ZSkgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgIHRoaXMubm9kZVZhbHVlID0gbm9kZVZhbHVlOwotICAgIH0KLQotICAgIHB1YmxpYyBzaG9ydCBnZXROb2RlVHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIEVMRU1FTlRfTk9ERTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTm9kZSBnZXRQYXJlbnROb2RlKCkgewotICAgICAgICByZXR1cm4gcGFyZW50OwotICAgIH0KLQotICAgIHB1YmxpYyBOb2RlTGlzdCBnZXRDaGlsZE5vZGVzKCkgewotICAgICAgICByZXR1cm4gdGhpczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTm9kZSBnZXRGaXJzdENoaWxkKCkgewotICAgICAgICByZXR1cm4gZmlyc3RDaGlsZDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTm9kZSBnZXRMYXN0Q2hpbGQoKSB7Ci0gICAgICAgIHJldHVybiBsYXN0Q2hpbGQ7Ci0gICAgfQotCi0gICAgcHVibGljIE5vZGUgZ2V0UHJldmlvdXNTaWJsaW5nKCkgewotICAgICAgICByZXR1cm4gcHJldmlvdXNTaWJsaW5nOwotICAgIH0KLQotICAgIHB1YmxpYyBOb2RlIGdldE5leHRTaWJsaW5nKCkgewotICAgICAgICByZXR1cm4gbmV4dFNpYmxpbmc7Ci0gICAgfQotCi0gICAgcHVibGljIE5hbWVkTm9kZU1hcCBnZXRBdHRyaWJ1dGVzKCkgewotICAgICAgICByZXR1cm4gYXR0cnM7Ci0gICAgfQotCi0gICAgcHVibGljIERvY3VtZW50IGdldE93bmVyRG9jdW1lbnQoKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIHB1YmxpYyBOb2RlIGluc2VydEJlZm9yZShOb2RlIG5ld0NoaWxkLCBOb2RlIHJlZkNoaWxkKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKG5ld0NoaWxkID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIm5ld0NoaWxkID09IG51bGwhIik7Ci0gICAgICAgIH0KLQotICAgICAgICBJSU9NZXRhZGF0YU5vZGUgbmV3SUlPQ2hpbGQgPSAoSUlPTWV0YWRhdGFOb2RlKW5ld0NoaWxkOwotICAgICAgICBJSU9NZXRhZGF0YU5vZGUgcmVmSUlPQ2hpbGQgPSAoSUlPTWV0YWRhdGFOb2RlKXJlZkNoaWxkOwotCi0gICAgICAgIG5ld0lJT0NoaWxkLnBhcmVudCA9IHRoaXM7Ci0KLSAgICAgICAgaWYgKHJlZklJT0NoaWxkID09IG51bGwpIHsKLSAgICAgICAgICAgIG5ld0lJT0NoaWxkLm5leHRTaWJsaW5nID0gbnVsbDsKLSAgICAgICAgICAgIG5ld0lJT0NoaWxkLnByZXZpb3VzU2libGluZyA9IGxhc3RDaGlsZDsKLQotICAgICAgICAgICAgLy8gRml4IHRoaXMgbm9kZQotICAgICAgICAgICAgbGFzdENoaWxkID0gbmV3SUlPQ2hpbGQ7Ci0gICAgICAgICAgICBpZiAoZmlyc3RDaGlsZCA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgZmlyc3RDaGlsZCA9IG5ld0lJT0NoaWxkOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbmV3SUlPQ2hpbGQubmV4dFNpYmxpbmcgPSByZWZJSU9DaGlsZDsKLSAgICAgICAgICAgIG5ld0lJT0NoaWxkLnByZXZpb3VzU2libGluZyA9IHJlZklJT0NoaWxkLnByZXZpb3VzU2libGluZzsKLQotICAgICAgICAgICAgLy8gRml4IHRoaXMgbm9kZQotICAgICAgICAgICAgaWYgKGZpcnN0Q2hpbGQgPT0gcmVmSUlPQ2hpbGQpIHsKLSAgICAgICAgICAgICAgICBmaXJzdENoaWxkID0gbmV3SUlPQ2hpbGQ7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIEZpeCBuZXh0IG5vZGUKLSAgICAgICAgICAgIGlmIChyZWZJSU9DaGlsZCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcmVmSUlPQ2hpbGQucHJldmlvdXNTaWJsaW5nID0gbmV3SUlPQ2hpbGQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBGaXggcHJldiBub2RlCi0gICAgICAgIGlmIChuZXdJSU9DaGlsZC5wcmV2aW91c1NpYmxpbmcgIT0gbnVsbCkgewotICAgICAgICAgICAgbmV3SUlPQ2hpbGQucHJldmlvdXNTaWJsaW5nLm5leHRTaWJsaW5nID0gbmV3SUlPQ2hpbGQ7Ci0gICAgICAgIH0KLQotICAgICAgICBuQ2hpbGRyZW4rKzsKLQotICAgICAgICByZXR1cm4gbmV3SUlPQ2hpbGQ7Ci0gICAgfQotCi0gICAgcHVibGljIE5vZGUgcmVwbGFjZUNoaWxkKE5vZGUgbmV3Q2hpbGQsIE5vZGUgb2xkQ2hpbGQpIHRocm93cyBET01FeGNlcHRpb24gewotICAgICAgICBpZiAobmV3Q2hpbGQgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigibmV3Q2hpbGQgPT0gbnVsbCEiKTsKLSAgICAgICAgfQotCi0gICAgICAgIElJT01ldGFkYXRhTm9kZSBuZXdJSU9DaGlsZCA9IChJSU9NZXRhZGF0YU5vZGUpbmV3Q2hpbGQ7Ci0gICAgICAgIElJT01ldGFkYXRhTm9kZSBvbGRJSU9DaGlsZCA9IChJSU9NZXRhZGF0YU5vZGUpb2xkQ2hpbGQ7Ci0KLSAgICAgICAgSUlPTWV0YWRhdGFOb2RlIG5leHQgPSBvbGRJSU9DaGlsZC5uZXh0U2libGluZzsKLSAgICAgICAgSUlPTWV0YWRhdGFOb2RlIHByZXZpb3VzID0gb2xkSUlPQ2hpbGQucHJldmlvdXNTaWJsaW5nOwotCi0gICAgICAgIC8vIEZpeCBuZXcgbm9kZQotICAgICAgICBuZXdJSU9DaGlsZC5wYXJlbnQgPSB0aGlzOwotICAgICAgICBuZXdJSU9DaGlsZC5uZXh0U2libGluZyA9IG5leHQ7Ci0gICAgICAgIG5ld0lJT0NoaWxkLnByZXZpb3VzU2libGluZyA9IHByZXZpb3VzOwotCi0gICAgICAgIC8vIEZpeCB0aGlzIG5vZGUKLSAgICAgICAgaWYgKGxhc3RDaGlsZCA9PSBvbGRJSU9DaGlsZCkgewotICAgICAgICAgICAgbGFzdENoaWxkID0gbmV3SUlPQ2hpbGQ7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGZpcnN0Q2hpbGQgPT0gb2xkSUlPQ2hpbGQpIHsKLSAgICAgICAgICAgIGZpcnN0Q2hpbGQgPSBuZXdJSU9DaGlsZDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIEZpeCBzaWJsaW5ncwotICAgICAgICBpZiAobmV4dCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBuZXh0LnByZXZpb3VzU2libGluZyA9IG5ld0lJT0NoaWxkOwotICAgICAgICB9Ci0gICAgICAgIGlmIChwcmV2aW91cyAhPSBudWxsKSB7Ci0gICAgICAgICAgICBwcmV2aW91cy5uZXh0U2libGluZyA9IG5ld0lJT0NoaWxkOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gRml4IG9sZCBjaGlsZAotICAgICAgICBvbGRJSU9DaGlsZC5wYXJlbnQgPSBudWxsOwotICAgICAgICBvbGRJSU9DaGlsZC5uZXh0U2libGluZyA9IG5leHQ7Ci0gICAgICAgIG9sZElJT0NoaWxkLnByZXZpb3VzU2libGluZyA9IHByZXZpb3VzOwotCi0gICAgICAgIHJldHVybiBvbGRJSU9DaGlsZDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTm9kZSByZW1vdmVDaGlsZChOb2RlIG9sZENoaWxkKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKG9sZENoaWxkID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIm9sZENoaWxkID09IG51bGwhIik7Ci0gICAgICAgIH0KLQotICAgICAgICBJSU9NZXRhZGF0YU5vZGUgb2xkSUlPQ2hpbGQgPSAoSUlPTWV0YWRhdGFOb2RlKW9sZENoaWxkOwotCi0gICAgICAgIC8vIEZpeCBuZXh0IGFuZCBwcmV2aW91cwotICAgICAgICBJSU9NZXRhZGF0YU5vZGUgcHJldmlvdXMgPSBvbGRJSU9DaGlsZC5wcmV2aW91c1NpYmxpbmc7Ci0gICAgICAgIElJT01ldGFkYXRhTm9kZSBuZXh0ID0gb2xkSUlPQ2hpbGQubmV4dFNpYmxpbmc7Ci0KLSAgICAgICAgaWYgKHByZXZpb3VzICE9IG51bGwpIHsKLSAgICAgICAgICAgIHByZXZpb3VzLm5leHRTaWJsaW5nID0gbmV4dDsKLSAgICAgICAgfQotICAgICAgICBpZiAobmV4dCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBuZXh0LnByZXZpb3VzU2libGluZyA9IHByZXZpb3VzOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gRml4IHRoaXMgbm9kZQotICAgICAgICBpZiAobGFzdENoaWxkID09IG9sZElJT0NoaWxkKSB7Ci0gICAgICAgICAgICBsYXN0Q2hpbGQgPSBwcmV2aW91czsKLSAgICAgICAgfQotICAgICAgICBpZiAoZmlyc3RDaGlsZCA9PSBvbGRJSU9DaGlsZCkgewotICAgICAgICAgICAgZmlyc3RDaGlsZCA9IG5leHQ7Ci0gICAgICAgIH0KLSAgICAgICAgbkNoaWxkcmVuLS07Ci0KLSAgICAgICAgLy8gRml4IG9sZCBjaGlsZAotICAgICAgICBvbGRJSU9DaGlsZC5wYXJlbnQgPSBudWxsOwotICAgICAgICBvbGRJSU9DaGlsZC5wcmV2aW91c1NpYmxpbmcgPSBudWxsOwotICAgICAgICBvbGRJSU9DaGlsZC5uZXh0U2libGluZyA9IG51bGw7Ci0KLSAgICAgICAgcmV0dXJuIG9sZElJT0NoaWxkOwotICAgIH0KLQotICAgIHB1YmxpYyBOb2RlIGFwcGVuZENoaWxkKE5vZGUgbmV3Q2hpbGQpIHRocm93cyBET01FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gaW5zZXJ0QmVmb3JlKG5ld0NoaWxkLCBudWxsKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBoYXNDaGlsZE5vZGVzKCkgewotICAgICAgICByZXR1cm4gbkNoaWxkcmVuICE9IDA7Ci0gICAgfQotCi0gICAgcHVibGljIE5vZGUgY2xvbmVOb2RlKGJvb2xlYW4gZGVlcCkgewotICAgICAgICBJSU9NZXRhZGF0YU5vZGUgY2xvbmVkID0gbmV3IElJT01ldGFkYXRhTm9kZShub2RlTmFtZSk7Ci0gICAgICAgIGNsb25lZC5zZXRVc2VyT2JqZWN0KGdldFVzZXJPYmplY3QoKSk7Ci0KLSAgICAgICAgaWYgKGRlZXApIHsgLy8gQ2xvbmUgcmVjdXJzaXZlbHkKLSAgICAgICAgICAgIElJT01ldGFkYXRhTm9kZSBjID0gZmlyc3RDaGlsZDsKLSAgICAgICAgICAgIHdoaWxlIChjICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBjbG9uZWQuaW5zZXJ0QmVmb3JlKGMuY2xvbmVOb2RlKHRydWUpLCBudWxsKTsKLSAgICAgICAgICAgICAgICBjID0gYy5uZXh0U2libGluZzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBjbG9uZWQ7IC8vIFRvIGNoYW5nZSBib2R5IG9mIGltcGxlbWVudGVkIG1ldGhvZHMgdXNlIEZpbGUgfAotICAgICAgICAvLyBTZXR0aW5ncyB8IEZpbGUgVGVtcGxhdGVzLgotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIG5vcm1hbGl6ZSgpIHsKLSAgICAgICAgLy8gRG8gbm90aGluZwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGlzU3VwcG9ydGVkKFN0cmluZyBmZWF0dXJlLCBTdHJpbmcgdmVyc2lvbikgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZyBnZXROYW1lc3BhY2VVUkkoKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0UHJlZml4KCkgewotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRQcmVmaXgoU3RyaW5nIHByZWZpeCkgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIERvIG5vdGhpbmcKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIGdldExvY2FsTmFtZSgpIHsKLSAgICAgICAgcmV0dXJuIG5vZGVOYW1lOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGhhc0F0dHJpYnV0ZXMoKSB7Ci0gICAgICAgIHJldHVybiBhdHRycy5saXN0LnNpemUoKSA+IDA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogPGk+RGVzY3JpcHRpb24gY29waWVkIGZyb20gaW50ZXJmYWNlOiBvcmcudzNjLmRvbS5Ob2RlIChET00gTGV2ZWwgMyk8L2k+Ci0gICAgICogPHA+Ci0gICAgICogVGhlIGFic29sdXRlIGJhc2UgVVJJIG9mIHRoaXMgbm9kZSBvciBudWxsIGlmIHRoZSBpbXBsZW1lbnRhdGlvbiB3YXNuJ3QKLSAgICAgKiBhYmxlIHRvIG9idGFpbiBhbiBhYnNvbHV0ZSBVUkkuIFRoaXMgdmFsdWUgaXMgY29tcHV0ZWQgYXMgZGVzY3JpYmVkIGluLgotICAgICAqIEhvd2V2ZXIsIHdoZW4gdGhlIERvY3VtZW50IHN1cHBvcnRzIHRoZSBmZWF0dXJlICJIVE1MIiBbRE9NIExldmVsIDIKLSAgICAgKiBIVE1MXSwgdGhlIGJhc2UgVVJJIGlzIGNvbXB1dGVkIHVzaW5nIGZpcnN0IHRoZSB2YWx1ZSBvZiB0aGUgaHJlZgotICAgICAqIGF0dHJpYnV0ZSBvZiB0aGUgSFRNTCBCQVNFIGVsZW1lbnQgaWYgYW55LCBhbmQgdGhlIHZhbHVlIG9mIHRoZQotICAgICAqIGRvY3VtZW50VVJJIGF0dHJpYnV0ZSBmcm9tIHRoZSBEb2N1bWVudCBpbnRlcmZhY2Ugb3RoZXJ3aXNlLgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIGFic29sdXRlIGJhc2UgVVJJLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0QmFzZVVSSSgpIHsKLSAgICAgICAgdGhyb3cgbmV3IERPTUV4Y2VwdGlvbihET01FeGNlcHRpb24uTk9UX1NVUFBPUlRFRF9FUlIsICJNZXRob2Qgbm90IHN1cHBvcnRlZCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIDxpPkRlc2NyaXB0aW9uIGNvcGllZCBmcm9tIGludGVyZmFjZTogb3JnLnczYy5kb20uTm9kZSAoRE9NIExldmVsIDMpPC9pPgotICAgICAqIDxwPgotICAgICAqIENvbXBhcmVzIHRoZSByZWZlcmVuY2Ugbm9kZSwgaS5lLiB0aGUgbm9kZSBvbiB3aGljaCB0aGlzIG1ldGhvZCBpcyBiZWluZwotICAgICAqIGNhbGxlZCwgd2l0aCBhIG5vZGUsIGkuZS4gdGhlIG9uZSBwYXNzZWQgYXMgYSBwYXJhbWV0ZXIsIHdpdGggcmVnYXJkIHRvCi0gICAgICogdGhlaXIgcG9zaXRpb24gaW4gdGhlIGRvY3VtZW50IGFuZCBhY2NvcmRpbmcgdG8gdGhlIGRvY3VtZW50IG9yZGVyLgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb3RoZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBub2RlIHRvIGNvbXBhcmUgYWdhaW5zdCB0aGUgcmVmZXJlbmNlIG5vZGUuCi0gICAgICogQHJldHVybiBSZXR1cm5zIGhvdyB0aGUgbm9kZSBpcyBwb3NpdGlvbmVkIHJlbGF0aXZlbHkgdG8gdGhlIHJlZmVyZW5jZQotICAgICAqICAgICAgICAgbm9kZS4KLSAgICAgKiBAdGhyb3dzIERPTUV4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIE5PVF9TVVBQT1JURURfRVJSOiB3aGVuIHRoZSBjb21wYXJlZCBub2RlcyBhcmUgZnJvbSBkaWZmZXJlbnQKLSAgICAgKiAgICAgICAgICAgICBET00gaW1wbGVtZW50YXRpb25zIHRoYXQgZG8gbm90IGNvb3JkaW5hdGUgdG8gcmV0dXJuCi0gICAgICogICAgICAgICAgICAgY29uc2lzdGVudCBpbXBsZW1lbnRhdGlvbi1zcGVjaWZpYyByZXN1bHRzLgotICAgICAqLwotICAgIHB1YmxpYyBzaG9ydCBjb21wYXJlRG9jdW1lbnRQb3NpdGlvbihOb2RlIG90aGVyKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgdGhyb3cgbmV3IERPTUV4Y2VwdGlvbihET01FeGNlcHRpb24uTk9UX1NVUFBPUlRFRF9FUlIsICJNZXRob2Qgbm90IHN1cHBvcnRlZCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIDxpPkRlc2NyaXB0aW9uIGNvcGllZCBmcm9tIGludGVyZmFjZTogb3JnLnczYy5kb20uTm9kZSAoRE9NIExldmVsIDMpPC9pPgotICAgICAqIDxwPgotICAgICAqIFRoaXMgYXR0cmlidXRlIHJldHVybnMgdGhlIHRleHQgY29udGVudCBvZiB0aGlzIG5vZGUgYW5kIGl0cyBkZXNjZW5kYW50cy4KLSAgICAgKiBXaGVuIGl0IGlzIGRlZmluZWQgdG8gYmUgbnVsbCwgc2V0dGluZyBpdCBoYXMgbm8gZWZmZWN0LiBPbiBzZXR0aW5nLCBhbnkKLSAgICAgKiBwb3NzaWJsZSBjaGlsZHJlbiB0aGlzIG5vZGUgbWF5IGhhdmUgYXJlIHJlbW92ZWQgYW5kLCBpZiBpdCB0aGUgbmV3Ci0gICAgICogc3RyaW5nIGlzIG5vdCBlbXB0eSBvciBudWxsLCByZXBsYWNlZCBieSBhIHNpbmdsZSBUZXh0IG5vZGUgY29udGFpbmluZwotICAgICAqIHRoZSBzdHJpbmcgdGhpcyBhdHRyaWJ1dGUgaXMgc2V0IHRvLiBPbiBnZXR0aW5nLCBubyBzZXJpYWxpemF0aW9uIGlzCi0gICAgICogcGVyZm9ybWVkLCB0aGUgcmV0dXJuZWQgc3RyaW5nIGRvZXMgbm90IGNvbnRhaW4gYW55IG1hcmt1cC4gTm8gd2hpdGVzcGFjZQotICAgICAqIG5vcm1hbGl6YXRpb24gaXMgcGVyZm9ybWVkIGFuZCB0aGUgcmV0dXJuZWQgc3RyaW5nIGRvZXMgbm90IGNvbnRhaW4gdGhlCi0gICAgICogd2hpdGUgc3BhY2VzIGluIGVsZW1lbnQgY29udGVudCAoc2VlIHRoZSBhdHRyaWJ1dGUKLSAgICAgKiBUZXh0LmlzRWxlbWVudENvbnRlbnRXaGl0ZXNwYWNlKS4gU2ltaWxhcmx5LCBvbiBzZXR0aW5nLCBubyBwYXJzaW5nIGlzCi0gICAgICogcGVyZm9ybWVkIGVpdGhlciwgdGhlIGlucHV0IHN0cmluZyBpcyB0YWtlbiBhcyBwdXJlIHRleHR1YWwgY29udGVudC4gVGhlCi0gICAgICogc3RyaW5nIHJldHVybmVkIGlzIG1hZGUgb2YgdGhlIHRleHQgY29udGVudCBvZiB0aGlzIG5vZGUgZGVwZW5kaW5nIG9uIGl0cwotICAgICAqIHR5cGUsIGFzIGRlZmluZWQgYmVsb3c6Ci0gICAgICogPHRhYmxlPgotICAgICAqIDx0cj4KLSAgICAgKiA8dGQ+PHN0cm9uZz5Ob2RlIHR5cGU8L3N0cm9uZz48L3RkPgotICAgICAqIDx0ZD48c3Ryb25nPkNvbnRlbnQ8L3N0cm9uZz48L3RkPgotICAgICAqIDwvdHI+Ci0gICAgICogPHRyPgotICAgICAqIDx0ZD5FTEVNRU5UX05PREUsIEFUVFJJQlVURV9OT0RFLCBFTlRJVFlfTk9ERSwgRU5USVRZX1JFRkVSRU5DRV9OT0RFLAotICAgICAqIERPQ1VNRU5UX0ZSQUdNRU5UX05PREU8L3RkPgotICAgICAqIDx0ZD5jb25jYXRlbmF0aW9uIG9mIHRoZSB0ZXh0Q29udGVudCBhdHRyaWJ1dGUgdmFsdWUgb2YgZXZlcnkgY2hpbGQgbm9kZSwKLSAgICAgKiBleGNsdWRpbmcgQ09NTUVOVF9OT0RFIGFuZCBQUk9DRVNTSU5HX0lOU1RSVUNUSU9OX05PREUgbm9kZXMuIFRoaXMgaXMgdGhlCi0gICAgICogZW1wdHkgc3RyaW5nIGlmIHRoZSBub2RlIGhhcyBubyBjaGlsZHJlbi48L3RkPgotICAgICAqIDwvdHI+Ci0gICAgICogPHRyPgotICAgICAqIDx0ZD5URVhUX05PREUsIENEQVRBX1NFQ1RJT05fTk9ERSwgQ09NTUVOVF9OT0RFLAotICAgICAqIFBST0NFU1NJTkdfSU5TVFJVQ1RJT05fTk9ERTwvdGQ+Ci0gICAgICogPHRkPm5vZGVWYWx1ZTwvdGQ+Ci0gICAgICogPC90cj4KLSAgICAgKiA8dHI+Ci0gICAgICogPHRkPkRPQ1VNRU5UX05PREUsIERPQ1VNRU5UX1RZUEVfTk9ERSwgTk9UQVRJT05fTk9ERTwvdGQ+Ci0gICAgICogPHRkPm51bGw8L3RkPgotICAgICAqIDwvdHI+Ci0gICAgICogPC90YWJsZT4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgdGV4dCBjb250ZW50IGRlcGVuZGluZyBvbiB0aGUgdHlwZSBvZiB0aGlzIG5vZGUuCi0gICAgICogQHRocm93cyBET01FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBET01TVFJJTkdfU0laRV9FUlI6IFJhaXNlZCB3aGVuIGl0IHdvdWxkIHJldHVybiBtb3JlCi0gICAgICogICAgICAgICAgICAgY2hhcmFjdGVycyB0aGFuIGZpdCBpbiBhIERPTVN0cmluZyB2YXJpYWJsZSBvbiB0aGUKLSAgICAgKiAgICAgICAgICAgICBpbXBsZW1lbnRhdGlvbiBwbGF0Zm9ybS4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldFRleHRDb250ZW50KCkgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiB0ZXh0Q29udGVudDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8aT5EZXNjcmlwdGlvbiBjb3BpZWQgZnJvbSBpbnRlcmZhY2U6IG9yZy53M2MuZG9tLk5vZGUgKERPTSBMZXZlbCAzKTwvaT4KLSAgICAgKiA8cD4KLSAgICAgKiBUaGlzIGF0dHJpYnV0ZSByZXR1cm5zIHRoZSB0ZXh0IGNvbnRlbnQgb2YgdGhpcyBub2RlIGFuZCBpdHMgZGVzY2VuZGFudHMuCi0gICAgICogV2hlbiBpdCBpcyBkZWZpbmVkIHRvIGJlIG51bGwsIHNldHRpbmcgaXQgaGFzIG5vIGVmZmVjdC4gT24gc2V0dGluZywgYW55Ci0gICAgICogcG9zc2libGUgY2hpbGRyZW4gdGhpcyBub2RlIG1heSBoYXZlIGFyZSByZW1vdmVkIGFuZCwgaWYgaXQgdGhlIG5ldwotICAgICAqIHN0cmluZyBpcyBub3QgZW1wdHkgb3IgbnVsbCwgcmVwbGFjZWQgYnkgYSBzaW5nbGUgVGV4dCBub2RlIGNvbnRhaW5pbmcKLSAgICAgKiB0aGUgc3RyaW5nIHRoaXMgYXR0cmlidXRlIGlzIHNldCB0by4gT24gZ2V0dGluZywgbm8gc2VyaWFsaXphdGlvbiBpcwotICAgICAqIHBlcmZvcm1lZCwgdGhlIHJldHVybmVkIHN0cmluZyBkb2VzIG5vdCBjb250YWluIGFueSBtYXJrdXAuIE5vIHdoaXRlc3BhY2UKLSAgICAgKiBub3JtYWxpemF0aW9uIGlzIHBlcmZvcm1lZCBhbmQgdGhlIHJldHVybmVkIHN0cmluZyBkb2VzIG5vdCBjb250YWluIHRoZQotICAgICAqIHdoaXRlIHNwYWNlcyBpbiBlbGVtZW50IGNvbnRlbnQgKHNlZSB0aGUgYXR0cmlidXRlCi0gICAgICogVGV4dC5pc0VsZW1lbnRDb250ZW50V2hpdGVzcGFjZSkuIFNpbWlsYXJseSwgb24gc2V0dGluZywgbm8gcGFyc2luZyBpcwotICAgICAqIHBlcmZvcm1lZCBlaXRoZXIsIHRoZSBpbnB1dCBzdHJpbmcgaXMgdGFrZW4gYXMgcHVyZSB0ZXh0dWFsIGNvbnRlbnQuIFRoZQotICAgICAqIHN0cmluZyByZXR1cm5lZCBpcyBtYWRlIG9mIHRoZSB0ZXh0IGNvbnRlbnQgb2YgdGhpcyBub2RlIGRlcGVuZGluZyBvbiBpdHMKLSAgICAgKiB0eXBlLCBhcyBkZWZpbmVkIGJlbG93OgotICAgICAqIDx0YWJsZT4KLSAgICAgKiA8dHI+Ci0gICAgICogPHRkPjxzdHJvbmc+Tm9kZSB0eXBlPC9zdHJvbmc+PC90ZD4KLSAgICAgKiA8dGQ+PHN0cm9uZz5Db250ZW50PC9zdHJvbmc+PC90ZD4KLSAgICAgKiA8L3RyPgotICAgICAqIDx0cj4KLSAgICAgKiA8dGQ+RUxFTUVOVF9OT0RFLCBBVFRSSUJVVEVfTk9ERSwgRU5USVRZX05PREUsIEVOVElUWV9SRUZFUkVOQ0VfTk9ERSwKLSAgICAgKiBET0NVTUVOVF9GUkFHTUVOVF9OT0RFPC90ZD4KLSAgICAgKiA8dGQ+Y29uY2F0ZW5hdGlvbiBvZiB0aGUgdGV4dENvbnRlbnQgYXR0cmlidXRlIHZhbHVlIG9mIGV2ZXJ5IGNoaWxkIG5vZGUsCi0gICAgICogZXhjbHVkaW5nIENPTU1FTlRfTk9ERSBhbmQgUFJPQ0VTU0lOR19JTlNUUlVDVElPTl9OT0RFIG5vZGVzLiBUaGlzIGlzIHRoZQotICAgICAqIGVtcHR5IHN0cmluZyBpZiB0aGUgbm9kZSBoYXMgbm8gY2hpbGRyZW4uPC90ZD4KLSAgICAgKiA8L3RyPgotICAgICAqIDx0cj4KLSAgICAgKiA8dGQ+VEVYVF9OT0RFLCBDREFUQV9TRUNUSU9OX05PREUsIENPTU1FTlRfTk9ERSwKLSAgICAgKiBQUk9DRVNTSU5HX0lOU1RSVUNUSU9OX05PREU8L3RkPgotICAgICAqIDx0ZD5ub2RlVmFsdWU8L3RkPgotICAgICAqIDwvdHI+Ci0gICAgICogPHRyPgotICAgICAqIDx0ZD5ET0NVTUVOVF9OT0RFLCBET0NVTUVOVF9UWVBFX05PREUsIE5PVEFUSU9OX05PREU8L3RkPgotICAgICAqIDx0ZD5udWxsPC90ZD4KLSAgICAgKiA8L3RyPgotICAgICAqIDwvdGFibGU+Ci0gICAgICogPC9wPgotICAgICAqIAotICAgICAqIEBwYXJhbSB0ZXh0Q29udGVudAotICAgICAqICAgICAgICAgICAgdGhlIHRleHQgY29udGVudCBmb3IgdGhpcyBub2RlLgotICAgICAqIEB0aHJvd3MgRE9NRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgTk9fTU9ESUZJQ0FUSU9OX0FMTE9XRURfRVJSOiBSYWlzZWQgd2hlbiB0aGUgbm9kZSBpcwotICAgICAqICAgICAgICAgICAgIHJlYWRvbmx5LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFRleHRDb250ZW50KFN0cmluZyB0ZXh0Q29udGVudCkgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgIHRoaXMudGV4dENvbnRlbnQgPSB0ZXh0Q29udGVudDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8aT5EZXNjcmlwdGlvbiBjb3BpZWQgZnJvbSBpbnRlcmZhY2U6IG9yZy53M2MuZG9tLk5vZGUgKERPTSBMZXZlbCAzKTwvaT4KLSAgICAgKiA8cD4KLSAgICAgKiBSZXR1cm5zIHdoZXRoZXIgdGhpcyBub2RlIGlzIHRoZSBzYW1lIG5vZGUgYXMgdGhlIGdpdmVuIG9uZS4gVGhpcyBtZXRob2QKLSAgICAgKiBwcm92aWRlcyBhIHdheSB0byBkZXRlcm1pbmUgd2hldGhlciB0d28gTm9kZSByZWZlcmVuY2VzIHJldHVybmVkIGJ5IHRoZQotICAgICAqIGltcGxlbWVudGF0aW9uIHJlZmVyZW5jZSB0aGUgc2FtZSBvYmplY3QuIFdoZW4gdHdvIE5vZGUgcmVmZXJlbmNlcyBhcmUKLSAgICAgKiByZWZlcmVuY2VzIHRvIHRoZSBzYW1lIG9iamVjdCwgZXZlbiBpZiB0aHJvdWdoIGEgcHJveHksIHRoZSByZWZlcmVuY2VzCi0gICAgICogbWF5IGJlIHVzZWQgY29tcGxldGVseSBpbnRlcmNoYW5nZWFibHksIHN1Y2ggdGhhdCBhbGwgYXR0cmlidXRlcyBoYXZlIHRoZQotICAgICAqIHNhbWUgdmFsdWVzIGFuZCBjYWxsaW5nIHRoZSBzYW1lIERPTSBtZXRob2Qgb24gZWl0aGVyIHJlZmVyZW5jZSBhbHdheXMKLSAgICAgKiBoYXMgZXhhY3RseSB0aGUgc2FtZSBlZmZlY3QuCi0gICAgICogPC9wPgotICAgICAqIAotICAgICAqIEBwYXJhbSBvdGhlcgotICAgICAqICAgICAgICAgICAgdGhlIG5vZGUgdG8gdGVzdCBhZ2FpbnN0LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIG5vZGVzIGFyZSB0aGUgc2FtZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzU2FtZU5vZGUoTm9kZSBvdGhlcikgewotICAgICAgICB0aHJvdyBuZXcgRE9NRXhjZXB0aW9uKERPTUV4Y2VwdGlvbi5OT1RfU1VQUE9SVEVEX0VSUiwgIk1ldGhvZCBub3Qgc3VwcG9ydGVkIik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogPGk+RGVzY3JpcHRpb24gY29waWVkIGZyb20gaW50ZXJmYWNlOiBvcmcudzNjLmRvbS5Ob2RlIChET00gTGV2ZWwgMyk8L2k+Ci0gICAgICogPHA+Ci0gICAgICogTG9vayB1cCB0aGUgcHJlZml4IGFzc29jaWF0ZWQgdG8gdGhlIGdpdmVuIG5hbWVzcGFjZSBVUkksIHN0YXJ0aW5nIGZyb20KLSAgICAgKiB0aGlzIG5vZGUuIFRoZSBkZWZhdWx0IG5hbWVzcGFjZSBkZWNsYXJhdGlvbnMgYXJlIGlnbm9yZWQgYnkgdGhpcyBtZXRob2QuCi0gICAgICogU2VlIGZvciBkZXRhaWxzIG9uIHRoZSBhbGdvcml0aG0gdXNlZCBieSB0aGlzIG1ldGhvZC4KLSAgICAgKiA8L3A+Ci0gICAgICogCi0gICAgICogQHBhcmFtIG5hbWVzcGFjZVVSSQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWVzcGFjZSBVUkkgdG8gbG9vayBmb3IuCi0gICAgICogQHJldHVybiB0aGUgYXNzb2NpYXRlZCBuYW1lc3BhY2UgcHJlZml4IGlmIGZvdW5kIG9yIG51bGwgaWYgbm9uZSBpcwotICAgICAqICAgICAgICAgZm91bmQuIElmIG1vcmUgdGhhbiBvbmUgcHJlZml4IGFyZSBhc3NvY2lhdGVkIHRvIHRoZSBuYW1lc3BhY2UKLSAgICAgKiAgICAgICAgIHByZWZpeCwgdGhlIHJldHVybmVkIG5hbWVzcGFjZSBwcmVmaXggaXMgaW1wbGVtZW50YXRpb24KLSAgICAgKiAgICAgICAgIGRlcGVuZGVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGxvb2t1cFByZWZpeChTdHJpbmcgbmFtZXNwYWNlVVJJKSB7Ci0gICAgICAgIHRocm93IG5ldyBET01FeGNlcHRpb24oRE9NRXhjZXB0aW9uLk5PVF9TVVBQT1JURURfRVJSLCAiTWV0aG9kIG5vdCBzdXBwb3J0ZWQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiA8aT5EZXNjcmlwdGlvbiBjb3BpZWQgZnJvbSBpbnRlcmZhY2U6IG9yZy53M2MuZG9tLk5vZGUgKERPTSBMZXZlbCAzKTwvaT4KLSAgICAgKiA8cD4KLSAgICAgKiBUaGlzIG1ldGhvZCBjaGVja3MgaWYgdGhlIHNwZWNpZmllZCBuYW1lc3BhY2VVUkkgaXMgdGhlIGRlZmF1bHQgbmFtZXNwYWNlCi0gICAgICogb3Igbm90LgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbmFtZXNwYWNlVVJJCi0gICAgICogICAgICAgICAgICB0aGUgbmFtZXNwYWNlIFVSSSB0byBsb29rIGZvci4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgbmFtZXNwYWNlVVJJIGlzIHRoZSBkZWZhdWx0IG5hbWVzcGFjZSwKLSAgICAgKiAgICAgICAgIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0RlZmF1bHROYW1lc3BhY2UoU3RyaW5nIG5hbWVzcGFjZVVSSSkgewotICAgICAgICB0aHJvdyBuZXcgRE9NRXhjZXB0aW9uKERPTUV4Y2VwdGlvbi5OT1RfU1VQUE9SVEVEX0VSUiwgIk1ldGhvZCBub3Qgc3VwcG9ydGVkIik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogPGk+RGVzY3JpcHRpb24gY29waWVkIGZyb20gaW50ZXJmYWNlOiBvcmcudzNjLmRvbS5Ob2RlIChET00gTGV2ZWwgMyk8L2k+Ci0gICAgICogPHA+Ci0gICAgICogTG9vayB1cCB0aGUgbmFtZXNwYWNlIFVSSSBhc3NvY2lhdGVkIHRvIHRoZSBnaXZlbiBwcmVmaXgsIHN0YXJ0aW5nIGZyb20KLSAgICAgKiB0aGlzIG5vZGUuIFNlZSBmb3IgZGV0YWlscyBvbiB0aGUgYWxnb3JpdGhtIHVzZWQgYnkgdGhpcyBtZXRob2QuCi0gICAgICogPC9wPgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcmVmaXgKLSAgICAgKiAgICAgICAgICAgIHRoZSBwcmVmaXggdG8gbG9vayBmb3IuIElmIHRoaXMgcGFyYW1ldGVyIGlzIG51bGwsIHRoZSBtZXRob2QKLSAgICAgKiAgICAgICAgICAgIHdpbGwgcmV0dXJuIHRoZSBkZWZhdWx0IG5hbWVzcGFjZSBVUkkgaWYgYW55LgotICAgICAqIEByZXR1cm4gdGhlIGFzc29jaWF0ZWQgbmFtZXNwYWNlIFVSSSBvciBudWxsIGlmIG5vbmUgaXMgZm91bmQuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZyBsb29rdXBOYW1lc3BhY2VVUkkoU3RyaW5nIHByZWZpeCkgewotICAgICAgICB0aHJvdyBuZXcgRE9NRXhjZXB0aW9uKERPTUV4Y2VwdGlvbi5OT1RfU1VQUE9SVEVEX0VSUiwgIk1ldGhvZCBub3Qgc3VwcG9ydGVkIik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogPGk+RGVzY3JpcHRpb24gY29waWVkIGZyb20gaW50ZXJmYWNlOiBvcmcudzNjLmRvbS5Ob2RlIChET00gTGV2ZWwgMyk8L2k+Ci0gICAgICogPHA+Ci0gICAgICogVGVzdHMgd2hldGhlciB0d28gbm9kZXMgYXJlIGVxdWFsLiBUaGlzIG1ldGhvZCB0ZXN0cyBmb3IgZXF1YWxpdHkgb2YKLSAgICAgKiBub2Rlcywgbm90IHNhbWVuZXNzIChpLmUuLCB3aGV0aGVyIHRoZSB0d28gbm9kZXMgYXJlIHJlZmVyZW5jZXMgdG8gdGhlCi0gICAgICogc2FtZSBvYmplY3QpIHdoaWNoIGNhbiBiZSB0ZXN0ZWQgd2l0aCBOb2RlLmlzU2FtZU5vZGUoKS4gQWxsIG5vZGVzIHRoYXQKLSAgICAgKiBhcmUgdGhlIHNhbWUgd2lsbCBhbHNvIGJlIGVxdWFsLCB0aG91Z2ggdGhlIHJldmVyc2UgbWF5IG5vdCBiZSB0cnVlLiBUd28KLSAgICAgKiBub2RlcyBhcmUgZXF1YWwgaWYgYW5kIG9ubHkgaWYgdGhlIGZvbGxvd2luZyBjb25kaXRpb25zIGFyZSBzYXRpc2ZpZWQ6Ci0gICAgICogPHA+Ci0gICAgICogPGxpPlRoZSB0d28gbm9kZXMgYXJlIG9mIHRoZSBzYW1lIHR5cGUuPC9saT4KLSAgICAgKiA8bGk+VGhlIGZvbGxvd2luZyBzdHJpbmcgYXR0cmlidXRlcyBhcmUgZXF1YWw6IG5vZGVOYW1lLCBsb2NhbE5hbWUsCi0gICAgICogbmFtZXNwYWNlVVJJLCBwcmVmaXgsIG5vZGVWYWx1ZSAuIFRoaXMgaXM6IHRoZXkgYXJlIGJvdGggbnVsbCwgb3IgdGhleQotICAgICAqIGhhdmUgdGhlIHNhbWUgbGVuZ3RoIGFuZCBhcmUgY2hhcmFjdGVyIGZvciBjaGFyYWN0ZXIgaWRlbnRpY2FsLjwvbGk+Ci0gICAgICogPGxpPlRoZSBhdHRyaWJ1dGVzIE5hbWVkTm9kZU1hcHMgYXJlIGVxdWFsLiBUaGlzIGlzOiB0aGV5IGFyZSBib3RoIG51bGwsCi0gICAgICogb3IgdGhleSBoYXZlIHRoZSBzYW1lIGxlbmd0aCBhbmQgZm9yIGVhY2ggbm9kZSB0aGF0IGV4aXN0cyBpbiBvbmUgbWFwCi0gICAgICogdGhlcmUgaXMgYSBub2RlIHRoYXQgZXhpc3RzIGluIHRoZSBvdGhlciBtYXAgYW5kIGlzIGVxdWFsLCBhbHRob3VnaCBub3QKLSAgICAgKiBuZWNlc3NhcmlseSBhdCB0aGUgc2FtZSBpbmRleC48L2xpPgotICAgICAqIDxsaT5UaGUgY2hpbGROb2RlcyBOb2RlTGlzdHMgYXJlIGVxdWFsLiBUaGlzIGlzOiB0aGV5IGFyZSBib3RoIG51bGwsIG9yCi0gICAgICogdGhleSBoYXZlIHRoZSBzYW1lIGxlbmd0aCBhbmQgY29udGFpbiBlcXVhbCBub2RlcyBhdCB0aGUgc2FtZSBpbmRleC4gTm90ZQotICAgICAqIHRoYXQgbm9ybWFsaXphdGlvbiBjYW4gYWZmZWN0IGVxdWFsaXR5OyB0byBhdm9pZCB0aGlzLCBub2RlcyBzaG91bGQgYmUKLSAgICAgKiBub3JtYWxpemVkIGJlZm9yZSBiZWluZyBjb21wYXJlZC48L2xpPgotICAgICAqIDwvcD4KLSAgICAgKiBGb3IgdHdvIERvY3VtZW50VHlwZSBub2RlcyB0byBiZSBlcXVhbCwgdGhlIGZvbGxvd2luZyBjb25kaXRpb25zIG11c3QKLSAgICAgKiBhbHNvIGJlIHNhdGlzZmllZDoKLSAgICAgKiA8cD4KLSAgICAgKiA8bGk+VGhlIGZvbGxvd2luZyBzdHJpbmcgYXR0cmlidXRlcyBhcmUgZXF1YWw6IHB1YmxpY0lkLCBzeXN0ZW1JZCwKLSAgICAgKiBpbnRlcm5hbFN1YnNldC48L2xpPgotICAgICAqIDxsaT5UaGUgZW50aXRpZXMgTmFtZWROb2RlTWFwcyBhcmUgZXF1YWwuPC9saT4KLSAgICAgKiA8bGk+VGhlIG5vdGF0aW9ucyBOYW1lZE5vZGVNYXBzIGFyZSBlcXVhbC48L2xpPgotICAgICAqIDwvcD4KLSAgICAgKiBPbiB0aGUgb3RoZXIgaGFuZCwgdGhlIGZvbGxvd2luZyBkbyBub3QgYWZmZWN0IGVxdWFsaXR5OiB0aGUKLSAgICAgKiBvd25lckRvY3VtZW50LCBiYXNlVVJJLCBhbmQgcGFyZW50Tm9kZSBhdHRyaWJ1dGVzLCB0aGUgc3BlY2lmaWVkCi0gICAgICogYXR0cmlidXRlIGZvciBBdHRyIG5vZGVzLCB0aGUgc2NoZW1hVHlwZUluZm8gYXR0cmlidXRlIGZvciBBdHRyIGFuZAotICAgICAqIEVsZW1lbnQgbm9kZXMsIHRoZSBUZXh0LmlzRWxlbWVudENvbnRlbnRXaGl0ZXNwYWNlIGF0dHJpYnV0ZSBmb3IgVGV4dAotICAgICAqIG5vZGVzLCBhcyB3ZWxsIGFzIGFueSB1c2VyIGRhdGEgb3IgZXZlbnQgbGlzdGVuZXJzIHJlZ2lzdGVyZWQgb24gdGhlCi0gICAgICogbm9kZXMuIDwvcD4KLSAgICAgKiA8cD4KLSAgICAgKiBOb3RlOiBBcyBhIGdlbmVyYWwgcnVsZSwgYW55dGhpbmcgbm90IG1lbnRpb25lZCBpbiB0aGUgZGVzY3JpcHRpb24gYWJvdmUKLSAgICAgKiBpcyBub3Qgc2lnbmlmaWNhbnQgaW4gY29uc2lkZXJhdGlvbiBvZiBlcXVhbGl0eSBjaGVja2luZy4gTm90ZSB0aGF0Ci0gICAgICogZnV0dXJlIHZlcnNpb25zIG9mIHRoaXMgc3BlY2lmaWNhdGlvbiBtYXkgdGFrZSBpbnRvIGFjY291bnQgbW9yZQotICAgICAqIGF0dHJpYnV0ZXMgYW5kIGltcGxlbWVudGF0aW9ucyBjb25mb3JtIHRvIHRoaXMgc3BlY2lmaWNhdGlvbiBhcmUgZXhwZWN0ZWQKLSAgICAgKiB0byBiZSB1cGRhdGVkIGFjY29yZGluZ2x5LgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYXJnCi0gICAgICogICAgICAgICAgICB0aGUgbm9kZSB0byBjb21wYXJlIGVxdWFsaXR5IHdpdGguCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgbm9kZXMgYXJlIGVxdWFsLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNFcXVhbE5vZGUoTm9kZSBhcmcpIHsKLSAgICAgICAgdGhyb3cgbmV3IERPTUV4Y2VwdGlvbihET01FeGNlcHRpb24uTk9UX1NVUFBPUlRFRF9FUlIsICJNZXRob2Qgbm90IHN1cHBvcnRlZCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIDxpPkRlc2NyaXB0aW9uIGNvcGllZCBmcm9tIGludGVyZmFjZTogb3JnLnczYy5kb20uTm9kZSAoRE9NIExldmVsIDMpPC9pPgotICAgICAqIDxwPgotICAgICAqIFRoaXMgbWV0aG9kIHJldHVybnMgYSBzcGVjaWFsaXplZCBvYmplY3Qgd2hpY2ggaW1wbGVtZW50cyB0aGUgc3BlY2lhbGl6ZWQKLSAgICAgKiBBUElzIG9mIHRoZSBzcGVjaWZpZWQgZmVhdHVyZSBhbmQgdmVyc2lvbiwgYXMgc3BlY2lmaWVkIGluLiBUaGUKLSAgICAgKiBzcGVjaWFsaXplZCBvYmplY3QgbWF5IGFsc28gYmUgb2J0YWluZWQgYnkgdXNpbmcgYmluZGluZy1zcGVjaWZpYyBjYXN0aW5nCi0gICAgICogbWV0aG9kcyBidXQgaXMgbm90IG5lY2Vzc2FyaWx5IGV4cGVjdGVkIHRvLCBhcyBkaXNjdXNzZWQgaW4uIFRoaXMgbWV0aG9kCi0gICAgICogYWxzbyBhbGxvdyB0aGUgaW1wbGVtZW50YXRpb24gdG8gcHJvdmlkZSBzcGVjaWFsaXplZCBvYmplY3RzIHdoaWNoIGRvIG5vdAotICAgICAqIHN1cHBvcnQgdGhlIE5vZGUgaW50ZXJmYWNlLgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZmVhdHVyZQotICAgICAqICAgICAgICAgICAgdGhlIG5hbWUgb2YgdGhlIGZlYXR1cmUgcmVxdWVzdGVkLiBOb3RlIHRoYXQgYW55IHBsdXMgc2lnbiAiKyIKLSAgICAgKiAgICAgICAgICAgIHByZXBlbmRlZCB0byB0aGUgbmFtZSBvZiB0aGUgZmVhdHVyZSB3aWxsIGJlIGlnbm9yZWQgc2luY2UgaXQKLSAgICAgKiAgICAgICAgICAgIGlzIG5vdCBzaWduaWZpY2FudCBpbiB0aGUgY29udGV4dCBvZiB0aGlzIG1ldGhvZC4KLSAgICAgKiBAcGFyYW0gdmVyc2lvbgotICAgICAqICAgICAgICAgICAgdGhpcyBpcyB0aGUgdmVyc2lvbiBudW1iZXIgb2YgdGhlIGZlYXR1cmUgdG8gdGVzdC4KLSAgICAgKiBAcmV0dXJuIHRoZSBvYmplY3Qgd2hpY2ggaW1wbGVtZW50cyB0aGUgc3BlY2lhbGl6ZWQgQVBJcyBvZiB0aGUgc3BlY2lmaWVkCi0gICAgICogICAgICAgICBmZWF0dXJlIGFuZCB2ZXJzaW9uLCBpZiBhbnksIG9yIG51bGwgaWYgdGhlcmUgaXMgbm8gb2JqZWN0IHdoaWNoCi0gICAgICogICAgICAgICBpbXBsZW1lbnRzIGludGVyZmFjZXMgYXNzb2NpYXRlZCB3aXRoIHRoYXQgZmVhdHVyZS4gSWYgdGhlCi0gICAgICogICAgICAgICBET01PYmplY3QgcmV0dXJuZWQgYnkgdGhpcyBtZXRob2QgaW1wbGVtZW50cyB0aGUgTm9kZSBpbnRlcmZhY2UsCi0gICAgICogICAgICAgICBpdCBtdXN0IGRlbGVnYXRlIHRvIHRoZSBwcmltYXJ5IGNvcmUgTm9kZSBhbmQgbm90IHJldHVybiByZXN1bHRzCi0gICAgICogICAgICAgICBpbmNvbnNpc3RlbnQgd2l0aCB0aGUgcHJpbWFyeSBjb3JlIE5vZGUgc3VjaCBhcyBhdHRyaWJ1dGVzLAotICAgICAqICAgICAgICAgY2hpbGROb2RlcywgZXRjLgotICAgICAqLwotICAgIHB1YmxpYyBPYmplY3QgZ2V0RmVhdHVyZShTdHJpbmcgZmVhdHVyZSwgU3RyaW5nIHZlcnNpb24pIHsKLSAgICAgICAgdGhyb3cgbmV3IERPTUV4Y2VwdGlvbihET01FeGNlcHRpb24uTk9UX1NVUFBPUlRFRF9FUlIsICJNZXRob2Qgbm90IHN1cHBvcnRlZCIpOwotICAgIH0KLQotICAgIC8vID8/P0FXVAotICAgIC8qCi0gICAgICogcHVibGljIE9iamVjdCBzZXRVc2VyRGF0YShTdHJpbmcga2V5LCBPYmplY3QgZGF0YSwgVXNlckRhdGFIYW5kbGVyCi0gICAgICogaGFuZGxlcikgeyB0aHJvdyBuZXcgRE9NRXhjZXB0aW9uKERPTUV4Y2VwdGlvbi5OT1RfU1VQUE9SVEVEX0VSUiwKLSAgICAgKiAiTWV0aG9kIG5vdCBzdXBwb3J0ZWQiKTsgfQotICAgICAqLwotCi0gICAgLyoqCi0gICAgICogPGk+RGVzY3JpcHRpb24gY29waWVkIGZyb20gaW50ZXJmYWNlOiBvcmcudzNjLmRvbS5Ob2RlIChET00gTGV2ZWwgMyk8L2k+Ci0gICAgICogPHA+Ci0gICAgICogUmV0cmlldmVzIHRoZSBvYmplY3QgYXNzb2NpYXRlZCB0byBhIGtleSBvbiBhIHRoaXMgbm9kZS4gVGhlIG9iamVjdCBtdXN0Ci0gICAgICogZmlyc3QgaGF2ZSBiZWVuIHNldCB0byB0aGlzIG5vZGUgYnkgY2FsbGluZyBzZXRVc2VyRGF0YSB3aXRoIHRoZSBzYW1lCi0gICAgICoga2V5LgotICAgICAqIDwvcD4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0ga2V5Ci0gICAgICogICAgICAgICAgICB0aGUga2V5IHRoZSBvYmplY3QgaXMgYXNzb2NpYXRlZCB0by4KLSAgICAgKiBAcmV0dXJuIHRoZSBET01Vc2VyRGF0YSBhc3NvY2lhdGVkIHRvIHRoZSBnaXZlbiBrZXkgb24gdGhpcyBub2RlLCBvciBudWxsCi0gICAgICogICAgICAgICBpZiB0aGVyZSB3YXMgbm9uZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IGdldFVzZXJEYXRhKFN0cmluZyBrZXkpIHsKLSAgICAgICAgdGhyb3cgbmV3IERPTUV4Y2VwdGlvbihET01FeGNlcHRpb24uTk9UX1NVUFBPUlRFRF9FUlIsICJNZXRob2Qgbm90IHN1cHBvcnRlZCIpOwotICAgIH0KLQotICAgIHB1YmxpYyBOb2RlIGl0ZW0oaW50IGluZGV4KSB7Ci0gICAgICAgIGlmIChpbmRleCA8IDAgfHwgaW5kZXggPj0gbkNoaWxkcmVuKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotCi0gICAgICAgIE5vZGUgbjsKLSAgICAgICAgZm9yIChuID0gZ2V0Rmlyc3RDaGlsZCgpOyBpbmRleCA+IDA7IGluZGV4LS0pIHsKLSAgICAgICAgICAgIG4gPSBuLmdldE5leHRTaWJsaW5nKCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldExlbmd0aCgpIHsKLSAgICAgICAgcmV0dXJuIG5DaGlsZHJlbjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB1c2VyIG9iamVjdCBhc3NvY2lhdGVkIHdpdGggdGhpcyBub2RlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHVzZXIgb2JqZWN0IGFzc29jaWF0ZWQgd2l0aCB0aGlzIG5vZGUuCi0gICAgICovCi0gICAgcHVibGljIE9iamVjdCBnZXRVc2VyT2JqZWN0KCkgewotICAgICAgICByZXR1cm4gdXNlck9iamVjdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSB1c2VyIG9iamVjdCBhc3NvY2lhdGVkIHdpdGggdGhpcyBub2RlLgotICAgICAqIAotICAgICAqIEBwYXJhbSB1c2VyT2JqZWN0Ci0gICAgICogICAgICAgICAgICB0aGUgbmV3IHVzZXIgb2JqZWN0IGFzc29jaWF0ZWQgd2l0aCB0aGlzIG5vZGUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0VXNlck9iamVjdChPYmplY3QgdXNlck9iamVjdCkgewotICAgICAgICB0aGlzLnVzZXJPYmplY3QgPSB1c2VyT2JqZWN0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBJSU9NZXRhZGF0YUF0dHIuCi0gICAgICovCi0gICAgcHJpdmF0ZSBjbGFzcyBJSU9NZXRhZGF0YUF0dHIgZXh0ZW5kcyBJSU9NZXRhZGF0YU5vZGUgaW1wbGVtZW50cyBBdHRyIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIG93bmVyIGVsZW1lbnQuCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIEVsZW1lbnQgb3duZXJFbGVtZW50OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgaUlPIG1ldGFkYXRhIGF0dHIuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gbmFtZQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBuYW1lLgotICAgICAgICAgKiBAcGFyYW0gdmFsdWUKLSAgICAgICAgICogICAgICAgICAgICB0aGUgdmFsdWUuCi0gICAgICAgICAqIEBwYXJhbSBvd25lcgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBvd25lci4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBJSU9NZXRhZGF0YUF0dHIoU3RyaW5nIG5hbWUsIFN0cmluZyB2YWx1ZSwgRWxlbWVudCBvd25lcikgewotICAgICAgICAgICAgc3VwZXIobmFtZSwgdmFsdWUpOwotICAgICAgICAgICAgdGhpcy5vd25lckVsZW1lbnQgPSBvd25lcjsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBTdHJpbmcgZ2V0TmFtZSgpIHsKLSAgICAgICAgICAgIHJldHVybiBnZXROb2RlTmFtZSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGJvb2xlYW4gZ2V0U3BlY2lmaWVkKCkgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgU3RyaW5nIGdldFZhbHVlKCkgewotICAgICAgICAgICAgcmV0dXJuIG5vZGVWYWx1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyB2b2lkIHNldFZhbHVlKFN0cmluZyB2YWx1ZSkgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgICAgICBub2RlVmFsdWUgPSB2YWx1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBFbGVtZW50IGdldE93bmVyRWxlbWVudCgpIHsKLSAgICAgICAgICAgIHJldHVybiBvd25lckVsZW1lbnQ7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogU2V0cyB0aGUgb3duZXIgZWxlbWVudC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBvd25lckVsZW1lbnQKLSAgICAgICAgICogICAgICAgICAgICB0aGUgbmV3IG93bmVyIGVsZW1lbnQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgdm9pZCBzZXRPd25lckVsZW1lbnQoRWxlbWVudCBvd25lckVsZW1lbnQpIHsKLSAgICAgICAgICAgIHRoaXMub3duZXJFbGVtZW50ID0gb3duZXJFbGVtZW50OwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEByZXR1cm4KLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzSWQoKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgRE9NRXhjZXB0aW9uKERPTUV4Y2VwdGlvbi5OT1RfU1VQUE9SVEVEX0VSUiwgIk1ldGhvZCBub3Qgc3VwcG9ydGVkIik7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHNob3J0IGdldE5vZGVUeXBlKCkgewotICAgICAgICAgICAgcmV0dXJuIEFUVFJJQlVURV9OT0RFOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIElJT01ldGFkYXRhTm9kZUxpc3QuCi0gICAgICovCi0gICAgcHJpdmF0ZSBjbGFzcyBJSU9NZXRhZGF0YU5vZGVMaXN0IGltcGxlbWVudHMgTm9kZUxpc3QsIE5hbWVkTm9kZU1hcCB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBsaXN0LgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBMaXN0PElJT01ldGFkYXRhTm9kZT4gbGlzdDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGlJTyBtZXRhZGF0YSBub2RlIGxpc3QuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gbGlzdAotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBsaXN0LgotICAgICAgICAgKi8KLSAgICAgICAgSUlPTWV0YWRhdGFOb2RlTGlzdChMaXN0PElJT01ldGFkYXRhTm9kZT4gbGlzdCkgewotICAgICAgICAgICAgdGhpcy5saXN0ID0gbGlzdDsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBOb2RlIGl0ZW0oaW50IGluZGV4KSB7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIHJldHVybiBsaXN0LmdldChpbmRleCk7Ci0gICAgICAgICAgICB9IGNhdGNoIChJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgZ2V0TGVuZ3RoKCkgewotICAgICAgICAgICAgcmV0dXJuIGxpc3Quc2l6ZSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIE5vZGUgZ2V0TmFtZWRJdGVtKFN0cmluZyBuYW1lKSB7Ci0gICAgICAgICAgICBmb3IgKElJT01ldGFkYXRhTm9kZSBub2RlIDogbGlzdCkgewotICAgICAgICAgICAgICAgIGlmIChuYW1lLmVxdWFscyhub2RlLmdldE5vZGVOYW1lKCkpKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBub2RlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIE5vZGUgc2V0TmFtZWRJdGVtKE5vZGUgYXJnKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBET01FeGNlcHRpb24oRE9NRXhjZXB0aW9uLk5PX01PRElGSUNBVElPTl9BTExPV0VEX0VSUiwKLSAgICAgICAgICAgICAgICAgICAgIlRoaXMgTmFtZWROb2RlTWFwIGlzIHJlYWQtb25seSEiKTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBOb2RlIHJlbW92ZU5hbWVkSXRlbShTdHJpbmcgbmFtZSkgdGhyb3dzIERPTUV4Y2VwdGlvbiB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgRE9NRXhjZXB0aW9uKERPTUV4Y2VwdGlvbi5OT19NT0RJRklDQVRJT05fQUxMT1dFRF9FUlIsCi0gICAgICAgICAgICAgICAgICAgICJUaGlzIE5hbWVkTm9kZU1hcCBpcyByZWFkLW9ubHkhIik7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgTm9kZSBnZXROYW1lZEl0ZW1OUyhTdHJpbmcgbmFtZXNwYWNlVVJJLCBTdHJpbmcgbG9jYWxOYW1lKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgICAgIHJldHVybiBnZXROYW1lZEl0ZW0obG9jYWxOYW1lKTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBOb2RlIHNldE5hbWVkSXRlbU5TKE5vZGUgYXJnKSB0aHJvd3MgRE9NRXhjZXB0aW9uIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBET01FeGNlcHRpb24oRE9NRXhjZXB0aW9uLk5PX01PRElGSUNBVElPTl9BTExPV0VEX0VSUiwKLSAgICAgICAgICAgICAgICAgICAgIlRoaXMgTmFtZWROb2RlTWFwIGlzIHJlYWQtb25seSEiKTsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBOb2RlIHJlbW92ZU5hbWVkSXRlbU5TKFN0cmluZyBuYW1lc3BhY2VVUkksIFN0cmluZyBsb2NhbE5hbWUpIHRocm93cyBET01FeGNlcHRpb24gewotICAgICAgICAgICAgdGhyb3cgbmV3IERPTUV4Y2VwdGlvbihET01FeGNlcHRpb24uTk9fTU9ESUZJQ0FUSU9OX0FMTE9XRURfRVJSLAotICAgICAgICAgICAgICAgICAgICAiVGhpcyBOYW1lZE5vZGVNYXAgaXMgcmVhZC1vbmx5ISIpOwotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vbWV0YWRhdGEvSUlPU3RhbmRhcmRNZXRhZGF0YUZvcm1hdC5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vbWV0YWRhdGEvSUlPU3RhbmRhcmRNZXRhZGF0YUZvcm1hdC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3MDZjYjJmLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL21ldGFkYXRhL0lJT1N0YW5kYXJkTWV0YWRhdGFGb3JtYXQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI5NyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8ubWV0YWRhdGE7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlVHlwZVNwZWNpZmllcjsKLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlMaXN0OwotCi0vKioKLSAqIFRoZSBjbGFzcyBJSU9TdGFuZGFyZE1ldGFkYXRhRm9ybWF0IGRlc2NyaWJlcyB0aGUgcnVsZXMgb2YgdGhlIHN0YW5kYXJkCi0gKiBtZXRhZGF0YSBmb3JtYXQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1jbGFzcyBJSU9TdGFuZGFyZE1ldGFkYXRhRm9ybWF0IGV4dGVuZHMgSUlPTWV0YWRhdGFGb3JtYXRJbXBsIHsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJSU9TdGFuZGFyZE1ldGFkYXRhRm9ybWF0LgotICAgICAqLwotICAgIHB1YmxpYyBJSU9TdGFuZGFyZE1ldGFkYXRhRm9ybWF0KCkgewotICAgICAgICBzdXBlcihzdGFuZGFyZE1ldGFkYXRhRm9ybWF0TmFtZSwgQ0hJTERfUE9MSUNZX1NPTUUpOwotICAgICAgICBidWlsZERURCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGNhbk5vZGVBcHBlYXIoU3RyaW5nIGVsZW1lbnROYW1lLCBJbWFnZVR5cGVTcGVjaWZpZXIgaW1hZ2VUeXBlKSB7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEJ1aWxkcyB0aGUgRFREIHRoYXQgZGVzY3JpYmVzIHRoZSBzdGFuZGFyZCBtZXRhZGF0YSBmb3JtYXQuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIGJ1aWxkRFREKCkgewotICAgICAgICAvLyBDSFJPTUEKLSAgICAgICAgYWRkRWxlbWVudCgiQ2hyb21hIiwgc3RhbmRhcmRNZXRhZGF0YUZvcm1hdE5hbWUsIENISUxEX1BPTElDWV9TT01FKTsKLQotICAgICAgICBhZGRFbGVtZW50KCJDb2xvclNwYWNlVHlwZSIsICJDaHJvbWEiLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotCi0gICAgICAgIEFycmF5TGlzdDxTdHJpbmc+IHZhbHVlcyA9IG5ldyBBcnJheUxpc3Q8U3RyaW5nPigyNyk7Ci0gICAgICAgIHZhbHVlcy5hZGQoIlhZWiIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJMYWIiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiTHV2Iik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIllDYkNyIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIll4eSIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJZQ0NLIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIlBob3RvWUNDIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIlJHQiIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJHUkFZIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIkhTViIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJITFMiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiQ01ZSyIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJDTVkiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiMkNMUiIpOwotICAgICAgICB2YWx1ZXMuYWRkKCIzQ0xSIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIjRDTFIiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiNUNMUiIpOwotICAgICAgICB2YWx1ZXMuYWRkKCI2Q0xSIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIjdDTFIiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiOENMUiIpOwotICAgICAgICB2YWx1ZXMuYWRkKCI5Q0xSIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIkFDTFIiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiQkNMUiIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJDQ0xSIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIkRDTFIiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiRUNMUiIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJGQ0xSIik7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiQ29sb3JTcGFjZVR5cGUiLCAibmFtZSIsIERBVEFUWVBFX1NUUklORywgdHJ1ZSwgbnVsbCwgdmFsdWVzKTsKLQotICAgICAgICBhZGRFbGVtZW50KCJOdW1DaGFubmVscyIsICJDaHJvbWEiLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIk51bUNoYW5uZWxzIiwgInZhbHVlIiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgMCwgSW50ZWdlci5NQVhfVkFMVUUpOyAvLyBsaXN0Ci0gICAgICAgIC8vIC0KLSAgICAgICAgLy8gd2h5Ci0gICAgICAgIC8vID8KLQotICAgICAgICBhZGRFbGVtZW50KCJHYW1tYSIsICJDaHJvbWEiLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIkdhbW1hIiwgInZhbHVlIiwgREFUQVRZUEVfRkxPQVQsIHRydWUsIG51bGwpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIkJsYWNrSXNaZXJvIiwgIkNocm9tYSIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEJvb2xlYW5BdHRyaWJ1dGUoIkJsYWNrSXNaZXJvIiwgInZhbHVlIiwgdHJ1ZSwgdHJ1ZSk7Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiUGFsZXR0ZSIsICJDaHJvbWEiLCAwLCBJbnRlZ2VyLk1BWF9WQUxVRSk7IC8vIENISUxEX1BPTElDWV9SRVBFQVQKLSAgICAgICAgYWRkRWxlbWVudCgiUGFsZXR0ZUVudHJ5IiwgIlBhbGV0dGUiLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIlBhbGV0dGVFbnRyeSIsICJpbmRleCIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIG51bGwpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIlBhbGV0dGVFbnRyeSIsICJyZWQiLCBEQVRBVFlQRV9JTlRFR0VSLCB0cnVlLCBudWxsKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJQYWxldHRlRW50cnkiLCAiZ3JlZW4iLCBEQVRBVFlQRV9JTlRFR0VSLCB0cnVlLCBudWxsKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJQYWxldHRlRW50cnkiLCAiYmx1ZSIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIG51bGwpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIlBhbGV0dGVFbnRyeSIsICJhbHBoYSIsIERBVEFUWVBFX0lOVEVHRVIsIGZhbHNlLCAiMjU1Iik7Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiQmFja2dyb3VuZEluZGV4IiwgIkNocm9tYSIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiQmFja2dyb3VuZEluZGV4IiwgInZhbHVlIiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgbnVsbCk7Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiQmFja2dyb3VuZENvbG9yIiwgIkNocm9tYSIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiQmFja2dyb3VuZENvbG9yIiwgInJlZCIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIG51bGwpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIkJhY2tncm91bmRDb2xvciIsICJncmVlbiIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIG51bGwpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIkJhY2tncm91bmRDb2xvciIsICJibHVlIiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgbnVsbCk7Ci0KLSAgICAgICAgLy8gQ09NUFJFU1NJT04KLSAgICAgICAgYWRkRWxlbWVudCgiQ29tcHJlc3Npb24iLCBzdGFuZGFyZE1ldGFkYXRhRm9ybWF0TmFtZSwgQ0hJTERfUE9MSUNZX1NPTUUpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIkNvbXByZXNzaW9uVHlwZU5hbWUiLCAiQ29tcHJlc3Npb24iLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIkNvbXByZXNzaW9uVHlwZU5hbWUiLCAidmFsdWUiLCBEQVRBVFlQRV9TVFJJTkcsIHRydWUsIG51bGwpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIkxvc3NsZXNzIiwgIkNvbXByZXNzaW9uIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQm9vbGVhbkF0dHJpYnV0ZSgiTG9zc2xlc3MiLCAidmFsdWUiLCB0cnVlLCB0cnVlKTsKLQotICAgICAgICBhZGRFbGVtZW50KCJOdW1Qcm9ncmVzc2l2ZVNjYW5zIiwgIkNvbXByZXNzaW9uIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJOdW1Qcm9ncmVzc2l2ZVNjYW5zIiwgInZhbHVlIiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgbnVsbCk7Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiQml0UmF0ZSIsICJDb21wcmVzc2lvbiIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiQml0UmF0ZSIsICJ2YWx1ZSIsIERBVEFUWVBFX0ZMT0FULCB0cnVlLCBudWxsKTsKLQotICAgICAgICAvLyBEQVRBCi0gICAgICAgIGFkZEVsZW1lbnQoIkRhdGEiLCBzdGFuZGFyZE1ldGFkYXRhRm9ybWF0TmFtZSwgQ0hJTERfUE9MSUNZX1NPTUUpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIlBsYW5hckNvbmZpZ3VyYXRpb24iLCAiRGF0YSIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIHZhbHVlcyA9IG5ldyBBcnJheUxpc3Q8U3RyaW5nPig0KTsKLSAgICAgICAgdmFsdWVzLmFkZCgiUGl4ZWxJbnRlcmxlYXZlZCIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJQbGFuZUludGVybGVhdmVkIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIkxpbmVJbnRlcmxlYXZlZCIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJUaWxlSW50ZXJsZWF2ZWQiKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJQbGFuYXJDb25maWd1cmF0aW9uIiwgInZhbHVlIiwgREFUQVRZUEVfU1RSSU5HLCB0cnVlLCBudWxsLCB2YWx1ZXMpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIlNhbXBsZUZvcm1hdCIsICJEYXRhIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgdmFsdWVzID0gbmV3IEFycmF5TGlzdDxTdHJpbmc+KDQpOwotICAgICAgICB2YWx1ZXMuYWRkKCJTaWduZWRJbnRlZ3JhbCIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJVbnNpZ25lZEludGVncmFsIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIlJlYWwiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiSW5kZXgiKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJTYW1wbGVGb3JtYXQiLCAidmFsdWUiLCBEQVRBVFlQRV9TVFJJTkcsIHRydWUsIG51bGwsIHZhbHVlcyk7Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiQml0c1BlclNhbXBsZSIsICJEYXRhIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJCaXRzUGVyU2FtcGxlIiwgInZhbHVlIiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgMSwgSW50ZWdlci5NQVhfVkFMVUUpOyAvLyBsaXN0Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiU2lnbmlmaWNhbnRCaXRzUGVyU2FtcGxlIiwgIkRhdGEiLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIlNpZ25pZmljYW50Qml0c1BlclNhbXBsZSIsICJ2YWx1ZSIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIDEsCi0gICAgICAgICAgICAgICAgSW50ZWdlci5NQVhfVkFMVUUpOyAvLyBsaXN0Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiU2FtcGxlTVNCIiwgIkRhdGEiLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIlNhbXBsZU1TQiIsICJ2YWx1ZSIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIDEsIEludGVnZXIuTUFYX1ZBTFVFKTsgLy8gbGlzdAotCi0gICAgICAgIC8vIERJTUVOU0lPTgotICAgICAgICBhZGRFbGVtZW50KCJEaW1lbnNpb24iLCBzdGFuZGFyZE1ldGFkYXRhRm9ybWF0TmFtZSwgQ0hJTERfUE9MSUNZX1NPTUUpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIlBpeGVsQXNwZWN0UmF0aW8iLCAiRGltZW5zaW9uIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJQaXhlbEFzcGVjdFJhdGlvIiwgInZhbHVlIiwgREFUQVRZUEVfRkxPQVQsIHRydWUsIG51bGwpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIkltYWdlT3JpZW50YXRpb24iLCAiRGltZW5zaW9uIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgdmFsdWVzID0gbmV3IEFycmF5TGlzdDxTdHJpbmc+KDgpOwotICAgICAgICB2YWx1ZXMuYWRkKCJOb3JtYWwiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiUm90YXRlOTAiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiUm90YXRlMTgwIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIlJvdGF0ZTI3MCIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJGbGlwSCIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJGbGlwViIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJGbGlwSFJvdGF0ZTkwIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIkZsaXBWUm90YXRlOTAiKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJJbWFnZU9yaWVudGF0aW9uIiwgInZhbHVlIiwgREFUQVRZUEVfU1RSSU5HLCB0cnVlLCBudWxsLCB2YWx1ZXMpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIkhvcml6b250YWxQaXhlbFNpemUiLCAiRGltZW5zaW9uIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJIb3Jpem9udGFsUGl4ZWxTaXplIiwgInZhbHVlIiwgREFUQVRZUEVfRkxPQVQsIHRydWUsIG51bGwpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIlZlcnRpY2FsUGl4ZWxTaXplIiwgIkRpbWVuc2lvbiIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiVmVydGljYWxQaXhlbFNpemUiLCAidmFsdWUiLCBEQVRBVFlQRV9GTE9BVCwgdHJ1ZSwgbnVsbCk7Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiSG9yaXpvbnRhbFBoeXNpY2FsUGl4ZWxTcGFjaW5nIiwgIkRpbWVuc2lvbiIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiSG9yaXpvbnRhbFBoeXNpY2FsUGl4ZWxTcGFjaW5nIiwgInZhbHVlIiwgREFUQVRZUEVfRkxPQVQsIHRydWUsIG51bGwpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIlZlcnRpY2FsUGh5c2ljYWxQaXhlbFNwYWNpbmciLCAiRGltZW5zaW9uIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJWZXJ0aWNhbFBoeXNpY2FsUGl4ZWxTcGFjaW5nIiwgInZhbHVlIiwgREFUQVRZUEVfRkxPQVQsIHRydWUsIG51bGwpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIkhvcml6b250YWxQb3NpdGlvbiIsICJEaW1lbnNpb24iLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIkhvcml6b250YWxQb3NpdGlvbiIsICJ2YWx1ZSIsIERBVEFUWVBFX0ZMT0FULCB0cnVlLCBudWxsKTsKLQotICAgICAgICBhZGRFbGVtZW50KCJWZXJ0aWNhbFBvc2l0aW9uIiwgIkRpbWVuc2lvbiIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiVmVydGljYWxQb3NpdGlvbiIsICJ2YWx1ZSIsIERBVEFUWVBFX0ZMT0FULCB0cnVlLCBudWxsKTsKLQotICAgICAgICBhZGRFbGVtZW50KCJIb3Jpem9udGFsUGl4ZWxPZmZzZXQiLCAiRGltZW5zaW9uIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJIb3Jpem9udGFsUGl4ZWxPZmZzZXQiLCAidmFsdWUiLCBEQVRBVFlQRV9JTlRFR0VSLCB0cnVlLCBudWxsKTsKLQotICAgICAgICBhZGRFbGVtZW50KCJWZXJ0aWNhbFBpeGVsT2Zmc2V0IiwgIkRpbWVuc2lvbiIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiVmVydGljYWxQaXhlbE9mZnNldCIsICJ2YWx1ZSIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIG51bGwpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIkhvcml6b250YWxTY3JlZW5TaXplIiwgIkRpbWVuc2lvbiIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiSG9yaXpvbnRhbFNjcmVlblNpemUiLCAidmFsdWUiLCBEQVRBVFlQRV9JTlRFR0VSLCB0cnVlLCBudWxsKTsKLQotICAgICAgICBhZGRFbGVtZW50KCJWZXJ0aWNhbFNjcmVlblNpemUiLCAiRGltZW5zaW9uIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJWZXJ0aWNhbFNjcmVlblNpemUiLCAidmFsdWUiLCBEQVRBVFlQRV9JTlRFR0VSLCB0cnVlLCBudWxsKTsKLQotICAgICAgICAvLyBET0NVTUVOVAotICAgICAgICBhZGRFbGVtZW50KCJEb2N1bWVudCIsIHN0YW5kYXJkTWV0YWRhdGFGb3JtYXROYW1lLCBDSElMRF9QT0xJQ1lfU09NRSk7Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiRm9ybWF0VmVyc2lvbiIsICJEb2N1bWVudCIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiRm9ybWF0VmVyc2lvbiIsICJ2YWx1ZSIsIERBVEFUWVBFX1NUUklORywgdHJ1ZSwgbnVsbCk7Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiU3ViaW1hZ2VJbnRlcnByZXRhdGlvbiIsICJEb2N1bWVudCIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIHZhbHVlcyA9IG5ldyBBcnJheUxpc3Q8U3RyaW5nPigxNCk7Ci0gICAgICAgIHZhbHVlcy5hZGQoIlN0YW5kYWxvbmUiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiU2luZ2xlUGFnZSIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJGdWxsUmVzb2x1dGlvbiIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJSZWR1Y2VkUmVzb2x1dGlvbiIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJQeXJhbWlkTGF5ZXIiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiUHJldmlldyIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJWb2x1bWVTbGljZSIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJPYmplY3RWaWV3Iik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIlBhbm9yYW1hIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIkFuaW1hdGlvbkZyYW1lIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIlRyYW5zcGFyZW5jeU1hc2siKTsKLSAgICAgICAgdmFsdWVzLmFkZCgiQ29tcG9zaXRpbmdMYXllciIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJTcGVjdHJhbFNsaWNlIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIlVua25vd24iKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJTdWJpbWFnZUludGVycHJldGF0aW9uIiwgInZhbHVlIiwgREFUQVRZUEVfU1RSSU5HLCB0cnVlLCBudWxsLCB2YWx1ZXMpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIkltYWdlQ3JlYXRpb25UaW1lIiwgIkRvY3VtZW50IiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJJbWFnZUNyZWF0aW9uVGltZSIsICJ5ZWFyIiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgbnVsbCk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiSW1hZ2VDcmVhdGlvblRpbWUiLCAibW9udGgiLCBEQVRBVFlQRV9JTlRFR0VSLCB0cnVlLCBudWxsLCAiMSIsICIxMiIsIHRydWUsCi0gICAgICAgICAgICAgICAgdHJ1ZSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiSW1hZ2VDcmVhdGlvblRpbWUiLCAiZGF5IiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgbnVsbCwgIjEiLCAiMzEiLCB0cnVlLAotICAgICAgICAgICAgICAgIHRydWUpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIkltYWdlQ3JlYXRpb25UaW1lIiwgImhvdXIiLCBEQVRBVFlQRV9JTlRFR0VSLCBmYWxzZSwgIjAiLCAiMCIsICIyMyIsIHRydWUsCi0gICAgICAgICAgICAgICAgdHJ1ZSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiSW1hZ2VDcmVhdGlvblRpbWUiLCAibWludXRlIiwgREFUQVRZUEVfSU5URUdFUiwgZmFsc2UsICIwIiwgIjAiLCAiNTkiLCB0cnVlLAotICAgICAgICAgICAgICAgIHRydWUpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIkltYWdlQ3JlYXRpb25UaW1lIiwgInNlY29uZCIsIERBVEFUWVBFX0lOVEVHRVIsIGZhbHNlLCAiMCIsICIwIiwgIjYwIiwgdHJ1ZSwKLSAgICAgICAgICAgICAgICB0cnVlKTsKLQotICAgICAgICBhZGRFbGVtZW50KCJJbWFnZU1vZGlmaWNhdGlvblRpbWUiLCAiRG9jdW1lbnQiLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIkltYWdlTW9kaWZpY2F0aW9uVGltZSIsICJ5ZWFyIiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgbnVsbCk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiSW1hZ2VNb2RpZmljYXRpb25UaW1lIiwgIm1vbnRoIiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgbnVsbCwgIjEiLCAiMTIiLAotICAgICAgICAgICAgICAgIHRydWUsIHRydWUpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIkltYWdlTW9kaWZpY2F0aW9uVGltZSIsICJkYXkiLCBEQVRBVFlQRV9JTlRFR0VSLCB0cnVlLCBudWxsLCAiMSIsICIzMSIsIHRydWUsCi0gICAgICAgICAgICAgICAgdHJ1ZSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiSW1hZ2VNb2RpZmljYXRpb25UaW1lIiwgImhvdXIiLCBEQVRBVFlQRV9JTlRFR0VSLCBmYWxzZSwgIjAiLCAiMCIsICIyMyIsCi0gICAgICAgICAgICAgICAgdHJ1ZSwgdHJ1ZSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiSW1hZ2VNb2RpZmljYXRpb25UaW1lIiwgIm1pbnV0ZSIsIERBVEFUWVBFX0lOVEVHRVIsIGZhbHNlLCAiMCIsICIwIiwgIjU5IiwKLSAgICAgICAgICAgICAgICB0cnVlLCB0cnVlKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJJbWFnZU1vZGlmaWNhdGlvblRpbWUiLCAic2Vjb25kIiwgREFUQVRZUEVfSU5URUdFUiwgZmFsc2UsICIwIiwgIjAiLCAiNjAiLAotICAgICAgICAgICAgICAgIHRydWUsIHRydWUpOwotCi0gICAgICAgIC8vIFRFWFQKLSAgICAgICAgYWRkRWxlbWVudCgiVGV4dCIsIHN0YW5kYXJkTWV0YWRhdGFGb3JtYXROYW1lLCAwLCBJbnRlZ2VyLk1BWF9WQUxVRSk7IC8vIENISUxEX1BPTElDWV9SRVBFQVQKLQotICAgICAgICBhZGRFbGVtZW50KCJUZXh0RW50cnkiLCAiVGV4dCIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiVGV4dEVudHJ5IiwgImtleXdvcmQiLCBEQVRBVFlQRV9TVFJJTkcsIGZhbHNlLCBudWxsKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJUZXh0RW50cnkiLCAidmFsdWUiLCBEQVRBVFlQRV9TVFJJTkcsIHRydWUsIG51bGwpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIlRleHRFbnRyeSIsICJsYW5ndWFnZSIsIERBVEFUWVBFX1NUUklORywgZmFsc2UsIG51bGwpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIlRleHRFbnRyeSIsICJlbmNvZGluZyIsIERBVEFUWVBFX1NUUklORywgZmFsc2UsIG51bGwpOwotICAgICAgICB2YWx1ZXMgPSBuZXcgQXJyYXlMaXN0PFN0cmluZz4oNSk7Ci0gICAgICAgIHZhbHVlcy5hZGQoIm5vbmUiKTsKLSAgICAgICAgdmFsdWVzLmFkZCgibHp3Iik7Ci0gICAgICAgIHZhbHVlcy5hZGQoInppcCIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJiemlwIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIm90aGVyIik7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiVGV4dEVudHJ5IiwgImNvbXByZXNzaW9uIiwgREFUQVRZUEVfU1RSSU5HLCBmYWxzZSwgIm5vbmUiLCB2YWx1ZXMpOwotCi0gICAgICAgIC8vIFRSQU5TUEFSRU5DWQotICAgICAgICBhZGRFbGVtZW50KCJUcmFuc3BhcmVuY3kiLCBzdGFuZGFyZE1ldGFkYXRhRm9ybWF0TmFtZSwgQ0hJTERfUE9MSUNZX1NPTUUpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIkFscGhhIiwgIlRyYW5zcGFyZW5jeSIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIHZhbHVlcyA9IG5ldyBBcnJheUxpc3Q8U3RyaW5nPigzKTsKLSAgICAgICAgdmFsdWVzLmFkZCgibm9uZSIpOwotICAgICAgICB2YWx1ZXMuYWRkKCJwcmVtdWx0aXBsaWVkIik7Ci0gICAgICAgIHZhbHVlcy5hZGQoIm5vbnByZW11bHRpcGxpZWQiKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJBbHBoYSIsICJ2YWx1ZSIsIERBVEFUWVBFX1NUUklORywgZmFsc2UsICJub25lIiwgdmFsdWVzKTsKLQotICAgICAgICBhZGRFbGVtZW50KCJUcmFuc3BhcmVudEluZGV4IiwgIlRyYW5zcGFyZW5jeSIsIENISUxEX1BPTElDWV9FTVBUWSk7Ci0gICAgICAgIGFkZEF0dHJpYnV0ZSgiVHJhbnNwYXJlbnRJbmRleCIsICJ2YWx1ZSIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIG51bGwpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIlRyYW5zcGFyZW50Q29sb3IiLCAiVHJhbnNwYXJlbmN5IiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJUcmFuc3BhcmVudENvbG9yIiwgInZhbHVlIiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgMCwgSW50ZWdlci5NQVhfVkFMVUUpOwotCi0gICAgICAgIGFkZEVsZW1lbnQoIlRpbGVUcmFuc3BhcmVuY2llcyIsICJUcmFuc3BhcmVuY3kiLCAwLCBJbnRlZ2VyLk1BWF9WQUxVRSk7IC8vIENISUxEX1BPTElDWV9SRVBFQVQKLQotICAgICAgICBhZGRFbGVtZW50KCJUcmFuc3BhcmVudFRpbGUiLCAiVGlsZVRyYW5zcGFyZW5jaWVzIiwgQ0hJTERfUE9MSUNZX0VNUFRZKTsKLSAgICAgICAgYWRkQXR0cmlidXRlKCJUcmFuc3BhcmVudFRpbGUiLCAieCIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIG51bGwpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIlRyYW5zcGFyZW50VGlsZSIsICJ5IiwgREFUQVRZUEVfSU5URUdFUiwgdHJ1ZSwgbnVsbCk7Ci0KLSAgICAgICAgYWRkRWxlbWVudCgiVGlsZU9wYWNpdGllcyIsICJUcmFuc3BhcmVuY3kiLCAwLCBJbnRlZ2VyLk1BWF9WQUxVRSk7IC8vIENISUxEX1BPTElDWV9SRVBFQVQKLQotICAgICAgICBhZGRFbGVtZW50KCJPcGFxdWVUaWxlIiwgIlRpbGVPcGFjaXRpZXMiLCBDSElMRF9QT0xJQ1lfRU1QVFkpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIk9wYXF1ZVRpbGUiLCAieCIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIG51bGwpOwotICAgICAgICBhZGRBdHRyaWJ1dGUoIk9wYXF1ZVRpbGUiLCAieSIsIERBVEFUWVBFX0lOVEVHRVIsIHRydWUsIG51bGwpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL21ldGFkYXRhL0lJT1N0YW5kYXJkTWV0YWRhdGFGb3JtYXRSZXNvdXJjZXMucHJvcGVydGllcyBiL2F3dC9qYXZheC9pbWFnZWlvL21ldGFkYXRhL0lJT1N0YW5kYXJkTWV0YWRhdGFGb3JtYXRSZXNvdXJjZXMucHJvcGVydGllcwpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDE4NTgwOC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9JSU9TdGFuZGFyZE1ldGFkYXRhRm9ybWF0UmVzb3VyY2VzLnByb3BlcnRpZXMKKysrIC9kZXYvbnVsbApAQCAtMSwxMzMgKzAsMCBAQAotIyBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSMgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0jIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSMgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSMgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSMgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSMKLSMgICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSMKLSMgIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSMgIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0jICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSMgIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSMgIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotCi0jIERlc2NyaXB0aW9ucyBvZiBlbGVtZW50cyBhbmQgYXR0cmlidXRlcyBvZiB0aGUgcGx1Z2luIG5ldXRyYWwgbWV0YWRhdGEgZm9ybWF0Ci0jIChzZWUgSUlPU3RhbmRhcmRNZXRhZGF0YUZvcm1hdCkKLQotIyBNZXNzYWdlcyBmb3IgRU4gbG9jYWxlCi1DaHJvbWE9Q2hyb21hIChjb2xvcikgaW5mb3JtYXRpb24KLUNvbG9yU3BhY2VUeXBlPVRoZSByYXcgY29sb3Igc3BhY2Ugb2YgdGhlIGltYWdlCi1Db2xvclNwYWNlVHlwZS9uYW1lPVRoZSByYXcgY29sb3Igc3BhY2Ugb2YgdGhlIGltYWdlCi1OdW1DaGFubmVscz1UaGUgbnVtYmVyIG9mIGNoYW5uZWxzIGluIHRoZSByYXcgaW1hZ2UsIGluY2x1ZGluZyBhbHBoYQotTnVtQ2hhbm5lbHMvdmFsdWU9VGhlIG51bWJlciBvZiBjaGFubmVscyBpbiB0aGUgcmF3IGltYWdlLCBpbmNsdWRpbmcgYWxwaGEKLUdhbW1hPVRoZSBpbWFnZSBnYW1tYQotR2FtbWEvdmFsdWU9VGhlIGltYWdlIGdhbW1hCi1CbGFja0lzWmVybz1UcnVlIGlmIHNtYWxsZXIgdmFsdWVzIHJlcHJlc2VudCBkYXJrZXIgc2hhZGVzCi1CbGFja0lzWmVyby92YWx1ZT1UcnVlIGlmIHNtYWxsZXIgdmFsdWVzIHJlcHJlc2VudCBkYXJrZXIgc2hhZGVzCi1QYWxldHRlPVBhbGV0dGUtY29sb3IgaW5mb3JtYXRpb24KLVBhbGV0dGVFbnRyeT1BIHBhbGV0dGUgZW50cnkKLVBhbGV0dGVFbnRyeS9pbmRleD1UaGUgaW5kZXggb2YgdGhlIHBhbGV0dGUgZW50cnkKLVBhbGV0dGVFbnRyeS9yZWQ9VGhlIHJlZCB2YWx1ZSBmb3IgdGhlIHBhbGV0dGUgZW50cnkKLVBhbGV0dGVFbnRyeS9ncmVlbj1UaGUgZ3JlZW4gdmFsdWUgZm9yIHRoZSBwYWxldHRlIGVudHJ5Ci1QYWxldHRlRW50cnkvYmx1ZT1UaGUgYmx1ZSB2YWx1ZSBmb3IgdGhlIHBhbGV0dGUgZW50cnkKLVBhbGV0dGVFbnRyeS9hbHBoYT1UaGUgYWxwaGEgdmFsdWUgZm9yIHRoZSBwYWxldHRlIGVudHJ5Ci1CYWNrZ3JvdW5kSW5kZXg9QSBwYWxldHRlIGluZGV4IHRvIGJlIHVzZWQgYXMgYSBiYWNrZ3JvdW5kCi1CYWNrZ3JvdW5kSW5kZXgvdmFsdWU9QSBwYWxldHRlIGluZGV4IHRvIGJlIHVzZWQgYXMgYSBiYWNrZ3JvdW5kCi1CYWNrZ3JvdW5kQ29sb3I9QW4gUkdCIHRyaXBsZSB0byBiZSB1c2VkIGFzIGEgYmFja2dyb3VuZAotQmFja2dyb3VuZENvbG9yL3JlZD1UaGUgcmVkIGJhY2tncm91bmQgdmFsdWUKLUJhY2tncm91bmRDb2xvci9ncmVlbj1UaGUgZ3JlZW4gYmFja2dyb3VuZCB2YWx1ZQotQmFja2dyb3VuZENvbG9yL2JsdWU9VGhlIGJsdWUgYmFja2dyb3VuZCB2YWx1ZQotCi1Db21wcmVzc2lvbj1Db21wcmVzc2lvbiBpbmZvcm1hdGlvbgotQ29tcHJlc3Npb25UeXBlTmFtZT1UaGUgbmFtZSBvZiB0aGUgY29tcHJlc3Npb24gc2NoZW1lIGluIHVzZQotQ29tcHJlc3Npb25UeXBlTmFtZS92YWx1ZT1UaGUgbmFtZSBvZiB0aGUgY29tcHJlc3Npb24gc2NoZW1lIGluIHVzZQotTG9zc2xlc3M9VHJ1ZSBpZiB0aGUgY29tcHJlc3Npb24gc2NoZW1lIGlzIGxvc3NsZXNzCi1Mb3NzbGVzcy92YWx1ZT1UcnVlIGlmIHRoZSBjb21wcmVzc2lvbiBzY2hlbWUgaXMgbG9zc2xlc3MKLU51bVByb2dyZXNzaXZlU2NhbnM9VGhlIG51bWJlciBvZiBwcm9ncmVzc2l2ZSBzY2FucyB1c2VkIGluIHRoZSBpbWFnZSBlbmNvZGluZwotTnVtUHJvZ3Jlc3NpdmVTY2Fucy92YWx1ZT1UaGUgbnVtYmVyIG9mIHByb2dyZXNzaXZlIHNjYW5zIHVzZWQgaW4gdGhlIGltYWdlIGVuY29kaW5nCi1CaXRSYXRlPVRoZSBlc3RpbWF0ZWQgYml0IHJhdGUgb2YgdGhlIGNvbXByZXNzaW9uIHNjaGVtZQotQml0UmF0ZS92YWx1ZT1UaGUgZXN0aW1hdGVkIGJpdCByYXRlIG9mIHRoZSBjb21wcmVzc2lvbiBzY2hlbWUKLQotRGF0YT1JbmZvcm1hdGlvbiBvbiB0aGUgaW1hZ2UgbGF5b3V0Ci1QbGFuYXJDb25maWd1cmF0aW9uPVRoZSBvcmdhbml6YXRpb24gb2YgaW1hZ2Ugc2FtcGxlcyBpbiB0aGUgc3RyZWFtCi1QbGFuYXJDb25maWd1cmF0aW9uL3ZhbHVlPVRoZSBvcmdhbml6YXRpb24gb2YgaW1hZ2Ugc2FtcGxlcyBpbiB0aGUgc3RyZWFtCi1TYW1wbGVGb3JtYXQ9VGhlIG51bWVyaWMgZm9ybWF0IG9mIGltYWdlIHNhbXBsZXMKLVNhbXBsZUZvcm1hdC92YWx1ZT1UaGUgbnVtZXJpYyBmb3JtYXQgb2YgaW1hZ2Ugc2FtcGxlcwotQml0c1BlclNhbXBsZT1UaGUgbnVtYmVyIG9mIGJpdHMgcGVyIHNhbXBsZQotQml0c1BlclNhbXBsZS92YWx1ZT1BIGxpc3Qgb2YgaW50ZWdlcnMsIG9uZSBwZXIgY2hhbm5lbAotU2lnbmlmaWNhbnRCaXRzUGVyU2FtcGxlPVRoZSBudW1iZXIgb2Ygc2lnbmlmaWNhbnQgYml0cyBwZXIgc2FtcGxlCi1TaWduaWZpY2FudEJpdHNQZXJTYW1wbGUvdmFsdWU9QSBsaXN0IG9mIGludGVnZXJzLCBvbmUgcGVyIGNoYW5uZWwKLVNhbXBsZU1TQj1UaGUgcG9zaXRpb24gb2YgdGhlIG1vc3Qgc2lnbmlmaWNhbnQgYml0IG9mIGVhY2ggc2FtcGxlCi1TYW1wbGVNU0IvdmFsdWU9QSBsaXN0IG9mIGludGVnZXJzLCBvbmUgcGVyIGNoYW5uZWwKLQotRGltZW5zaW9uPURpbWVuc2lvbiBpbmZvcm1hdGlvbgotUGl4ZWxBc3BlY3RSYXRpbz1UaGUgd2lkdGggb2YgYSBwaXhlbCBkaXZpZGVkIGJ5IGl0cyBoZWlnaHQKLVBpeGVsQXNwZWN0UmF0aW8vdmFsdWU9VGhlIHdpZHRoIG9mIGEgcGl4ZWwgZGl2aWRlZCBieSBpdHMgaGVpZ2h0Ci1JbWFnZU9yaWVudGF0aW9uPVRoZSBkZXNpcmVkIG9yaWVudGF0aW9uIG9mIHRoZSBpbWFnZSBpbiB0ZXJtcyBvZiBmbGlwcyBhbmQgY291bnRlci1jbG9ja3dpc2Ugcm90YXRpb25zCi1JbWFnZU9yaWVudGF0aW9uL3ZhbHVlPVRoZSBkZXNpcmVkIG9yaWVudGF0aW9uIG9mIHRoZSBpbWFnZSBpbiB0ZXJtcyBvZiBmbGlwcyBhbmQgY291bnRlci1jbG9ja3dpc2Ugcm90YXRpb25zCi1Ib3Jpem9udGFsUGl4ZWxTaXplPVRoZSB3aWR0aCBvZiBhIHBpeGVsLCBpbiBtaWxsaW1ldGVycywgYXMgaXQgc2hvdWxkIGJlIHJlbmRlcmVkIG9uIG1lZGlhCi1Ib3Jpem9udGFsUGl4ZWxTaXplL3ZhbHVlPVRoZSB3aWR0aCBvZiBhIHBpeGVsLCBpbiBtaWxsaW1ldGVycywgYXMgaXQgc2hvdWxkIGJlIHJlbmRlcmVkIG9uIG1lZGlhCi1WZXJ0aWNhbFBpeGVsU2l6ZT1UaGUgaGVpZ2h0IG9mIGEgcGl4ZWwsIGluIG1pbGxpbWV0ZXJzLCBhcyBpdCBzaG91bGQgYmUgcmVuZGVyZWQgb24gbWVkaWEKLVZlcnRpY2FsUGl4ZWxTaXplL3ZhbHVlPVRoZSBoZWlnaHQgb2YgYSBwaXhlbCwgaW4gbWlsbGltZXRlcnMsIGFzIGl0IHNob3VsZCBiZSByZW5kZXJlZCBvbiBtZWRpYQotSG9yaXpvbnRhbFBoeXNpY2FsUGl4ZWxTcGFjaW5nPVRoZSBob3Jpem9udGFsIGRpc3RhbmNlIGluIHRoZSBzdWJqZWN0IG9mIHRoZSBpbWFnZSwgaW4gbWlsbGltZXRlcnMsIHJlcHJlc2VudGVkIGJ5IG9uZSBwaXhlbCBhdCB0aGUgY2VudGVyIG9mIHRoZSBpbWFnZQotSG9yaXpvbnRhbFBoeXNpY2FsUGl4ZWxTcGFjaW5nL3ZhbHVlPVRoZSBob3Jpem9udGFsIGRpc3RhbmNlIGluIHRoZSBzdWJqZWN0IG9mIHRoZSBpbWFnZSwgaW4gbWlsbGltZXRlcnMsIHJlcHJlc2VudGVkIGJ5IG9uZSBwaXhlbCBhdCB0aGUgY2VudGVyIG9mIHRoZSBpbWFnZQotVmVydGljYWxQaHlzaWNhbFBpeGVsU3BhY2luZz1UaGUgdmVydGljYWwgZGlzdGFuY2UgaW4gdGhlIHN1YmplY3Qgb2YgdGhlIGltYWdlLCBpbiBtaWxsaW1ldGVycywgcmVwcmVzZW50ZWQgYnkgb25lIHBpeGVsIGF0IHRoZSBjZW50ZXIgb2YgdGhlIGltYWdlCi1WZXJ0aWNhbFBoeXNpY2FsUGl4ZWxTcGFjaW5nL3ZhbHVlPVRoZSB2ZXJ0aWNhbCBkaXN0YW5jZSBpbiB0aGUgc3ViamVjdCBvZiB0aGUgaW1hZ2UsIGluIG1pbGxpbWV0ZXJzLCByZXByZXNlbnRlZCBieSBvbmUgcGl4ZWwgYXQgdGhlIGNlbnRlciBvZiB0aGUgaW1hZ2UKLUhvcml6b250YWxQb3NpdGlvbj1UaGUgaG9yaXpvbnRhbCBwb3NpdGlvbiwgaW4gbWlsbGltZXRlcnMsIHdoZXJlIHRoZSBpbWFnZSBzaG91bGQgYmUgcmVuZGVyZWQgb24gbWVkaWEKLUhvcml6b250YWxQb3NpdGlvbi92YWx1ZT1UaGUgaG9yaXpvbnRhbCBwb3NpdGlvbiwgaW4gbWlsbGltZXRlcnMsIHdoZXJlIHRoZSBpbWFnZSBzaG91bGQgYmUgcmVuZGVyZWQgb24gbWVkaWEKLVZlcnRpY2FsUG9zaXRpb249VGhlIHZlcnRpY2FsIHBvc2l0aW9uLCBpbiBtaWxsaW1ldGVycywgd2hlcmUgdGhlIGltYWdlIHNob3VsZCBiZSByZW5kZXJlZCBvbiBtZWRpYQotVmVydGljYWxQb3NpdGlvbi92YWx1ZT1UaGUgdmVydGljYWwgcG9zaXRpb24sIGluIG1pbGxpbWV0ZXJzLCB3aGVyZSB0aGUgaW1hZ2Ugc2hvdWxkIGJlIHJlbmRlcmVkIG9uIG1lZGlhCi1Ib3Jpem9udGFsUGl4ZWxPZmZzZXQ9VGhlIGhvcml6b25hbCBwb3NpdGlvbiwgaW4gcGl4ZWxzLCB3aGVyZSB0aGUgaW1hZ2Ugc2hvdWxkIGJlIHJlbmRlcmVkIG9udG8gYSByYXN0ZXIgZGlzcGxheQotSG9yaXpvbnRhbFBpeGVsT2Zmc2V0L3ZhbHVlPVRoZSBob3Jpem9uYWwgcG9zaXRpb24sIGluIHBpeGVscywgd2hlcmUgdGhlIGltYWdlIHNob3VsZCBiZSByZW5kZXJlZCBvbnRvIGEgcmFzdGVyIGRpc3BsYXkKLVZlcnRpY2FsUGl4ZWxPZmZzZXQ9VGhlIHZlcnRpY2FsIHBvc2l0aW9uLCBpbiBwaXhlbHMsIHdoZXJlIHRoZSBpbWFnZSBzaG91bGQgYmUgcmVuZGVyZWQgb250byBhIHJhc3RlciBkaXNwbGF5Ci1WZXJ0aWNhbFBpeGVsT2Zmc2V0L3ZhbHVlPVRoZSB2ZXJ0aWNhbCBwb3NpdGlvbiwgaW4gcGl4ZWxzLCB3aGVyZSB0aGUgaW1hZ2Ugc2hvdWxkIGJlIHJlbmRlcmVkIG9udG8gYSByYXN0ZXIgZGlzcGxheQotSG9yaXpvbnRhbFNjcmVlblNpemU9VGhlIHdpZHRoLCBpbiBwaXhlbHMsIG9mIHRoZSByYXN0ZXIgZGlzcGxheSBpbnRvIHdoaWNoIHRoZSBpbWFnZSBzaG91bGQgYmUgcmVuZGVyZWQKLUhvcml6b250YWxTY3JlZW5TaXplL3ZhbHVlPVRoZSB3aWR0aCwgaW4gcGl4ZWxzLCBvZiB0aGUgcmFzdGVyIGRpc3BsYXkgaW50byB3aGljaCB0aGUgaW1hZ2Ugc2hvdWxkIGJlIHJlbmRlcmVkCi1WZXJ0aWNhbFNjcmVlblNpemU9VGhlIGhlaWdodCwgaW4gcGl4ZWxzLCBvZiB0aGUgcmFzdGVyIGRpc3BsYXkgaW50byB3aGljaCB0aGUgaW1hZ2Ugc2hvdWxkIGJlIHJlbmRlcmVkCi1WZXJ0aWNhbFNjcmVlblNpemUvdmFsdWU9VGhlIGhlaWdodCwgaW4gcGl4ZWxzLCBvZiB0aGUgcmFzdGVyIGRpc3BsYXkgaW50byB3aGljaCB0aGUgaW1hZ2Ugc2hvdWxkIGJlIHJlbmRlcmVkCi0KLURvY3VtZW50PURvY3VtZW50IGluZm9ybWF0aW9uCi1Gb3JtYXRWZXJzaW9uPVRoZSB2ZXJzaW9uIG9mIHRoZSBmb3JtYXQgdXNlZCBieSB0aGUgc3RyZWFtCi1Gb3JtYXRWZXJzaW9uL3ZhbHVlPVRoZSB2ZXJzaW9uIG9mIHRoZSBmb3JtYXQgdXNlZCBieSB0aGUgc3RyZWFtCi1TdWJpbWFnZUludGVycHJldGF0aW9uPVRoZSBpbnRlcnByZXRhdGlvbiBvZiB0aGlzIGltYWdlIGluIHJlbGF0aW9uIHRvIHRoZSBvdGhlciBpbWFnZXMgc3RvcmVkIGluIHRoZSBzYW1lIHN0cmVhbQotU3ViaW1hZ2VJbnRlcnByZXRhdGlvbi92YWx1ZT1UaGUgaW50ZXJwcmV0YXRpb24gb2YgdGhpcyBpbWFnZSBpbiByZWxhdGlvbiB0byB0aGUgb3RoZXIgaW1hZ2VzIHN0b3JlZCBpbiB0aGUgc2FtZSBzdHJlYW0KLUltYWdlQ3JlYXRpb25UaW1lPVRoZSB0aW1lIG9mIGltYWdlIGNyZWF0aW9uCi1JbWFnZUNyZWF0aW9uVGltZS95ZWFyPVRoZSBmdWxsIHllYXIgKGUuZy4sIDE5NjcsIG5vdCA2NykKLUltYWdlQ3JlYXRpb25UaW1lL21vbnRoPVRoZSBtb250aCwgd2l0aCBKYW51YXJ5ID0gMQotSW1hZ2VDcmVhdGlvblRpbWUvZGF5PVRoZSBkYXkgb2YgdGhlIG1vbnRoCi1JbWFnZUNyZWF0aW9uVGltZS9ob3VyPVRoZSBob3VyIGZyb20gMCB0byAyMwotSW1hZ2VDcmVhdGlvblRpbWUvbWludXRlPVRoZSBtaW51dGUgZnJvbSAwIHRvIDU5Ci1JbWFnZUNyZWF0aW9uVGltZS9zZWNvbmQ9VGhlIHNlY29uZCBmcm9tIDAgdG8gNjAgKDYwID0gbGVhcCBzZWNvbmQpCi1JbWFnZU1vZGlmaWNhdGlvblRpbWU9VGhlIHRpbWUgb2YgdGhlIGxhc3QgaW1hZ2UgbW9kaWZpY2F0aW9uCi1JbWFnZU1vZGlmaWNhdGlvblRpbWUveWVhcj1UaGUgZnVsbCB5ZWFyIChlLmcuLCAxOTY3LCBub3QgNjcpCi1JbWFnZU1vZGlmaWNhdGlvblRpbWUvbW9udGg9VGhlIG1vbnRoLCB3aXRoIEphbnVhcnkgPSAxCi1JbWFnZU1vZGlmaWNhdGlvblRpbWUvZGF5PVRoZSBkYXkgb2YgdGhlIG1vbnRoCi1JbWFnZU1vZGlmaWNhdGlvblRpbWUvaG91cj1UaGUgaG91ciBmcm9tIDAgdG8gMjMKLUltYWdlTW9kaWZpY2F0aW9uVGltZS9taW51dGU9VGhlIG1pbnV0ZSBmcm9tIDAgdG8gNTkKLUltYWdlTW9kaWZpY2F0aW9uVGltZS9zZWNvbmQ9VGhlIHNlY29uZCBmcm9tIDAgdG8gNjAgKDYwID0gbGVhcCBzZWNvbmQpCi0KLVRleHQ9VGV4dCBpbmZvcm1hdGlvbgotVGV4dEVudHJ5PUEgdGV4dCBlbnRyeQotVGV4dEVudHJ5L2tleXdvcmQ9QSBrZXl3b3JkIGFzc29jaWF0ZWQgd2l0aCB0aGUgdGV4dCBlbnRyeQotVGV4dEVudHJ5L3ZhbHVlPXRoZSB0ZXh0IGVudHJ5Ci1UZXh0RW50cnkvbGFuZ3VhZ2U9VGhlIGxhbmd1YWdlIG9mIHRoZSB0ZXh0Ci1UZXh0RW50cnkvZW5jb2Rpbmc9VGhlIGVuY29kaW5nIG9mIHRoZSB0ZXh0Ci1UZXh0RW50cnkvY29tcHJlc3Npb249VGhlIG1ldGhvZCB1c2VkIHRvIGNvbXByZXNzIHRoZSB0ZXh0Ci0KLVRyYW5zcGFyZW5jeT1UcmFuc3BhcmVuY3kgaW5mb3JtYXRpb24KLUFscGhhPVRoZSB0eXBlIG9mIGFscGhhIGluZm9ybWF0aW9uIGNvbnRhaW5lZCBpbiB0aGUgaW1hZ2UKLUFscGhhL3ZhbHVlPVRoZSB0eXBlIG9mIGFscGhhIGluZm9ybWF0aW9uIGNvbnRhaW5lZCBpbiB0aGUgaW1hZ2UKLVRyYW5zcGFyZW50SW5kZXg9QSBwYWxldHRlIGluZGV4IHRvIGJlIHRyZWF0ZWQgYXMgdHJhbnNwYXJlbnQKLVRyYW5zcGFyZW50SW5kZXgvdmFsdWU9QSBwYWxldHRlIGluZGV4IHRvIGJlIHRyZWF0ZWQgYXMgdHJhbnNwYXJlbnQKLVRyYW5zcGFyZW50Q29sb3I9QW4gUkdCIGNvbG9yIHRvIGJlIHRyZWF0ZWQgYXMgdHJhbnNwYXJlbnQKLVRyYW5zcGFyZW50Q29sb3IvdmFsdWU9QW4gUkdCIGNvbG9yIHRvIGJlIHRyZWF0ZWQgYXMgdHJhbnNwYXJlbnQKLVRpbGVUcmFuc3BhcmVuY2llcz1BIGxpc3Qgb2YgY29tcGxldGVseSB0cmFuc3BhcmVudCB0aWxlcwotVHJhbnNwYXJlbnRUaWxlPVRoZSBpbmRleCBvZiBhIGNvbXBsZXRlbHkgdHJhbnNwYXJlbnQgdGlsZQotVHJhbnNwYXJlbnRUaWxlL3g9VGhlIHRpbGUncyBYIGluZGV4Ci1UcmFuc3BhcmVudFRpbGUveT1UaGUgdGlsZSdzIFkgaW5kZXgKLVRpbGVPcGFjaXRpZXM9QSBsaXN0IG9mIGNvbXBsZXRlbHkgb3BhcXVlIHRpbGVzCi1PcGFxdWVUaWxlPVRoZSBpbmRleCBvZiBhIGNvbXBsZXRlbHkgb3BhcXVlIHRpbGUKLU9wYXF1ZVRpbGUveD1UaGUgdGlsZSdzIFggaW5kZXgKLU9wYXF1ZVRpbGUveT1UaGUgdGlsZSdzIFkgaW5kZXgKZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL21ldGFkYXRhL3BhY2thZ2UuaHRtbCBiL2F3dC9qYXZheC9pbWFnZWlvL21ldGFkYXRhL3BhY2thZ2UuaHRtbApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjliZDUxYi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9tZXRhZGF0YS9wYWNrYWdlLmh0bWwKKysrIC9kZXYvbnVsbApAQCAtMSw4ICswLDAgQEAKLTxodG1sPgotICA8Ym9keT4KLSAgICA8cD4KLSAgICAgIFRoaXMgcGFja2FnZSBjb250YWlucyBjbGFzc2VzIHdoaWNoIGFsbG93cyB0byByZWFkIGFuZCB3cml0ZSBkZXNjcmliaW5nIG1ldGFkYXRhIG9mIGltYWdlIGZpbGVzLgotICAgIDwvcD4KLSAgQHNpbmNlIEFuZHJvaWQgMS4wCi0gIDwvYm9keT4KLTwvaHRtbD4KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3BhY2thZ2UuaHRtbCBiL2F3dC9qYXZheC9pbWFnZWlvL3BhY2thZ2UuaHRtbApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMmZkNjE0OC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9wYWNrYWdlLmh0bWwKKysrIC9kZXYvbnVsbApAQCAtMSw4ICswLDAgQEAKLTxodG1sPgotICA8Ym9keT4KLSAgICA8cD4KLSAgICAgIFRoaXMgcGFja2FnZSBjb250YWlucyBjbGFzc2VzIGFuZCBpbnRlcmZhY2VzIHdoaWNoIHByb3ZpZGVzIGFuIEltYWdlIEkvTyBBUEkuIFRoZSBjb250YWluZWQgY2xhc3NlcyBhbmQgaW50ZXJmYWNlcyBhbGxvdyByZWFkaW5nIGFuZCB3cml0aW5nIGltYWdlIGZpbGVzIG9mIGRpZmZlcmVudCBmb3JtYXRzLgotICAgIDwvcD4KLSAgQHNpbmNlIEFuZHJvaWQgMS4wCi0gIDwvYm9keT4KLTwvaHRtbD4KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3BsdWdpbnMvYm1wL0JNUEltYWdlV3JpdGVQYXJhbS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vcGx1Z2lucy9ibXAvQk1QSW1hZ2VXcml0ZVBhcmFtLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGVjZmIyMGEuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vcGx1Z2lucy9ibXAvQk1QSW1hZ2VXcml0ZVBhcmFtLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw3OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8ucGx1Z2lucy5ibXA7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlV3JpdGVQYXJhbTsKLWltcG9ydCBqYXZhLnV0aWwuTG9jYWxlOwotCi0vKioKLSAqIFRoZSBCTVBJbWFnZVdyaXRlUGFyYW0gY2xhc3MgYWxsb3dzIGVuY29kaW5nIGFuIGltYWdlIGluIEJNUCBmb3JtYXQuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgQk1QSW1hZ2VXcml0ZVBhcmFtIGV4dGVuZHMgSW1hZ2VXcml0ZVBhcmFtIHsKLQotICAgIC8qKgotICAgICAqIFRoZSB0b3AgZG93bi4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gdG9wRG93bjsgLy8gRGVmYXVsdCBpcyBib3R0b20tdXAKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBCTVBJbWFnZVdyaXRlUGFyYW0gd2l0aCBkZWZhdWx0IHZhbHVlcyBvZiBhbGwKLSAgICAgKiBwYXJhbWV0ZXJzLgotICAgICAqLwotICAgIHB1YmxpYyBCTVBJbWFnZVdyaXRlUGFyYW0oKSB7Ci0gICAgICAgIHRoaXMobnVsbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEJNUEltYWdlV3JpdGVQYXJhbSB3aXRoIHRoZSBzcGVjaWZpZWQgTG9jYWxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsb2NhbGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgTG9jYWxlLgotICAgICAqLwotICAgIHB1YmxpYyBCTVBJbWFnZVdyaXRlUGFyYW0oTG9jYWxlIGxvY2FsZSkgewotICAgICAgICBzdXBlcihsb2NhbGUpOwotCi0gICAgICAgIC8vIFNldCB0aGUgY29tcHJlc3Npb24KLSAgICAgICAgY2FuV3JpdGVDb21wcmVzc2VkID0gdHJ1ZTsKLSAgICAgICAgY29tcHJlc3Npb25UeXBlcyA9IG5ldyBTdHJpbmdbXSB7Ci0gICAgICAgICAgICAgICAgIkJJX1JHQiIsICJCSV9STEU4IiwgIkJJX1JMRTQiLCAiQklfQklURklFTERTIgotICAgICAgICB9OwotICAgICAgICBjb21wcmVzc2lvblR5cGUgPSBjb21wcmVzc2lvblR5cGVzWzBdOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdHJ1ZSBpZiB0aGUgZGF0YSB3aWxsIGJlIHdyaXR0ZW4gaW4gYSB0b3AtZG93biBvcmRlciwgZmFsc2UKLSAgICAgKiBvdGhlcndpc2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRvcERvd24KLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgdG9wLWRvd24gdmFsdWUuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0VG9wRG93bihib29sZWFuIHRvcERvd24pIHsKLSAgICAgICAgdGhpcy50b3BEb3duID0gdG9wRG93bjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIGRhdGEgaXMgd3JpdHRlbiBpbiB0b3AtZG93biBvcmRlciwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgZGF0YSBpcyB3cml0dGVuIGluIHRvcC1kb3duIG9yZGVyLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNUb3BEb3duKCkgewotICAgICAgICByZXR1cm4gdG9wRG93bjsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9wbHVnaW5zL2JtcC9wYWNrYWdlLmh0bWwgYi9hd3QvamF2YXgvaW1hZ2Vpby9wbHVnaW5zL2JtcC9wYWNrYWdlLmh0bWwKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDk0OTRhMTAuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vcGx1Z2lucy9ibXAvcGFja2FnZS5odG1sCisrKyAvZGV2L251bGwKQEAgLTEsOCArMCwwIEBACi08aHRtbD4KLSAgPGJvZHk+Ci0gICAgPHA+Ci0gICAgICBUaGlzIHBhY2thZ2UgY29udGFpbnMgYXV4aWxpYXJ5IGNsYXNzZXMgZm9yIHRoZSBidWlsdC1pbiBCTVAgaW1hZ2UgcGx1Zy1pbi4KLSAgICA8L3A+Ci0gIEBzaW5jZSBBbmRyb2lkIDEuMAotICA8L2JvZHk+Ci08L2h0bWw+CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR0h1ZmZtYW5UYWJsZS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vcGx1Z2lucy9qcGVnL0pQRUdIdWZmbWFuVGFibGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjdiNTA0Yi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR0h1ZmZtYW5UYWJsZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjI2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5wbHVnaW5zLmpwZWc7Ci0KLS8qKgotICogVGhlIEpQRUdIdWZmbWFuVGFibGUgY2xhc3MgcmVwcmVzZW50cyBhIHNpbmdsZSBKUEVHIEh1ZmZtYW4gdGFibGUuIEl0Ci0gKiBjb250YWlucyB0aGUgc3RhbmRhcmQgdGFibGVzIGZyb20gdGhlIEpQRUcgc3BlY2lmaWNhdGlvbi4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBKUEVHSHVmZm1hblRhYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBzdGFuZGFyZCBEQyBsdW1pbmFuY2UgSHVmZm1hbiB0YWJsZSAuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBKUEVHSHVmZm1hblRhYmxlIFN0ZERDTHVtaW5hbmNlID0gbmV3IEpQRUdIdWZmbWFuVGFibGUobmV3IHNob3J0W10gewotICAgICAgICAgICAgMCwgMSwgNSwgMSwgMSwgMSwgMSwgMSwgMSwgMCwgMCwgMCwgMCwgMCwgMCwgMAotICAgIH0sIG5ldyBzaG9ydFtdIHsKLSAgICAgICAgICAgIDAsIDEsIDIsIDMsIDQsIDUsIDYsIDcsIDgsIDksIDB4MEEsIDB4MEIKLSAgICB9LCBmYWxzZSk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc3RhbmRhcmQgREMgY2hyb21pbmFuY2UgSHVmZm1hbiB0YWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEpQRUdIdWZmbWFuVGFibGUgU3RkRENDaHJvbWluYW5jZSA9IG5ldyBKUEVHSHVmZm1hblRhYmxlKG5ldyBzaG9ydFtdIHsKLSAgICAgICAgICAgIDAsIDMsIDEsIDEsIDEsIDEsIDEsIDEsIDEsIDEsIDEsIDAsIDAsIDAsIDAsIDAKLSAgICB9LCBuZXcgc2hvcnRbXSB7Ci0gICAgICAgICAgICAwLCAxLCAyLCAzLCA0LCA1LCA2LCA3LCA4LCA5LCAweDBBLCAweDBCCi0gICAgfSwgZmFsc2UpOwotCi0gICAgLyoqCi0gICAgICogVGhlIHN0YW5kYXJkIEFDIGx1bWluYW5jZSBIdWZmbWFuIHRhYmxlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgSlBFR0h1ZmZtYW5UYWJsZSBTdGRBQ0x1bWluYW5jZSA9IG5ldyBKUEVHSHVmZm1hblRhYmxlKG5ldyBzaG9ydFtdIHsKLSAgICAgICAgICAgIDAsIDIsIDEsIDMsIDMsIDIsIDQsIDMsIDUsIDUsIDQsIDQsIDAsIDAsIDEsIDB4N0QKLSAgICB9LCBuZXcgc2hvcnRbXSB7Ci0gICAgICAgICAgICAweDAxLCAweDAyLCAweDAzLCAweDAwLCAweDA0LCAweDExLCAweDA1LCAweDEyLCAweDIxLCAweDMxLCAweDQxLCAweDA2LCAweDEzLCAweDUxLAotICAgICAgICAgICAgMHg2MSwgMHgwNywgMHgyMiwgMHg3MSwgMHgxNCwgMHgzMiwgMHg4MSwgMHg5MSwgMHhBMSwgMHgwOCwgMHgyMywgMHg0MiwgMHhCMSwgMHhDMSwKLSAgICAgICAgICAgIDB4MTUsIDB4NTIsIDB4RDEsIDB4RjAsIDB4MjQsIDB4MzMsIDB4NjIsIDB4NzIsIDB4ODIsIDB4MDksIDB4MEEsIDB4MTYsIDB4MTcsIDB4MTgsCi0gICAgICAgICAgICAweDE5LCAweDFBLCAweDI1LCAweDI2LCAweDI3LCAweDI4LCAweDI5LCAweDJBLCAweDM0LCAweDM1LCAweDM2LCAweDM3LCAweDM4LCAweDM5LAotICAgICAgICAgICAgMHgzQSwgMHg0MywgMHg0NCwgMHg0NSwgMHg0NiwgMHg0NywgMHg0OCwgMHg0OSwgMHg0QSwgMHg1MywgMHg1NCwgMHg1NSwgMHg1NiwgMHg1NywKLSAgICAgICAgICAgIDB4NTgsIDB4NTksIDB4NUEsIDB4NjMsIDB4NjQsIDB4NjUsIDB4NjYsIDB4NjcsIDB4NjgsIDB4NjksIDB4NkEsIDB4NzMsIDB4NzQsIDB4NzUsCi0gICAgICAgICAgICAweDc2LCAweDc3LCAweDc4LCAweDc5LCAweDdBLCAweDgzLCAweDg0LCAweDg1LCAweDg2LCAweDg3LCAweDg4LCAweDg5LCAweDhBLCAweDkyLAotICAgICAgICAgICAgMHg5MywgMHg5NCwgMHg5NSwgMHg5NiwgMHg5NywgMHg5OCwgMHg5OSwgMHg5QSwgMHhBMiwgMHhBMywgMHhBNCwgMHhBNSwgMHhBNiwgMHhBNywKLSAgICAgICAgICAgIDB4QTgsIDB4QTksIDB4QUEsIDB4QjIsIDB4QjMsIDB4QjQsIDB4QjUsIDB4QjYsIDB4QjcsIDB4QjgsIDB4QjksIDB4QkEsIDB4QzIsIDB4QzMsCi0gICAgICAgICAgICAweEM0LCAweEM1LCAweEM2LCAweEM3LCAweEM4LCAweEM5LCAweENBLCAweEQyLCAweEQzLCAweEQ0LCAweEQ1LCAweEQ2LCAweEQ3LCAweEQ4LAotICAgICAgICAgICAgMHhEOSwgMHhEQSwgMHhFMSwgMHhFMiwgMHhFMywgMHhFNCwgMHhFNSwgMHhFNiwgMHhFNywgMHhFOCwgMHhFOSwgMHhFQSwgMHhGMSwgMHhGMiwKLSAgICAgICAgICAgIDB4RjMsIDB4RjQsIDB4RjUsIDB4RjYsIDB4RjcsIDB4RjgsIDB4RjksIDB4RkEKLSAgICB9LCBmYWxzZSk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc3RhbmRhcmQgQUMgY2hyb21pbmFuY2UgSHVmZm1hbiB0YWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEpQRUdIdWZmbWFuVGFibGUgU3RkQUNDaHJvbWluYW5jZSA9IG5ldyBKUEVHSHVmZm1hblRhYmxlKG5ldyBzaG9ydFtdIHsKLSAgICAgICAgICAgIDAsIDIsIDEsIDIsIDQsIDQsIDMsIDQsIDcsIDUsIDQsIDQsIDAsIDEsIDIsIDB4NzcKLSAgICB9LCBuZXcgc2hvcnRbXSB7Ci0gICAgICAgICAgICAweDAwLCAweDAxLCAweDAyLCAweDAzLCAweDExLCAweDA0LCAweDA1LCAweDIxLCAweDMxLCAweDA2LCAweDEyLCAweDQxLCAweDUxLCAweDA3LAotICAgICAgICAgICAgMHg2MSwgMHg3MSwgMHgxMywgMHgyMiwgMHgzMiwgMHg4MSwgMHgwOCwgMHgxNCwgMHg0MiwgMHg5MSwgMHhBMSwgMHhCMSwgMHhDMSwgMHgwOSwKLSAgICAgICAgICAgIDB4MjMsIDB4MzMsIDB4NTIsIDB4RjAsIDB4MTUsIDB4NjIsIDB4NzIsIDB4RDEsIDB4MEEsIDB4MTYsIDB4MjQsIDB4MzQsIDB4RTEsIDB4MjUsCi0gICAgICAgICAgICAweEYxLCAweDE3LCAweDE4LCAweDE5LCAweDFBLCAweDI2LCAweDI3LCAweDI4LCAweDI5LCAweDJBLCAweDM1LCAweDM2LCAweDM3LCAweDM4LAotICAgICAgICAgICAgMHgzOSwgMHgzQSwgMHg0MywgMHg0NCwgMHg0NSwgMHg0NiwgMHg0NywgMHg0OCwgMHg0OSwgMHg0QSwgMHg1MywgMHg1NCwgMHg1NSwgMHg1NiwKLSAgICAgICAgICAgIDB4NTcsIDB4NTgsIDB4NTksIDB4NUEsIDB4NjMsIDB4NjQsIDB4NjUsIDB4NjYsIDB4NjcsIDB4NjgsIDB4NjksIDB4NkEsIDB4NzMsIDB4NzQsCi0gICAgICAgICAgICAweDc1LCAweDc2LCAweDc3LCAweDc4LCAweDc5LCAweDdBLCAweDgyLCAweDgzLCAweDg0LCAweDg1LCAweDg2LCAweDg3LCAweDg4LCAweDg5LAotICAgICAgICAgICAgMHg4QSwgMHg5MiwgMHg5MywgMHg5NCwgMHg5NSwgMHg5NiwgMHg5NywgMHg5OCwgMHg5OSwgMHg5QSwgMHhBMiwgMHhBMywgMHhBNCwgMHhBNSwKLSAgICAgICAgICAgIDB4QTYsIDB4QTcsIDB4QTgsIDB4QTksIDB4QUEsIDB4QjIsIDB4QjMsIDB4QjQsIDB4QjUsIDB4QjYsIDB4QjcsIDB4QjgsIDB4QjksIDB4QkEsCi0gICAgICAgICAgICAweEMyLCAweEMzLCAweEM0LCAweEM1LCAweEM2LCAweEM3LCAweEM4LCAweEM5LCAweENBLCAweEQyLCAweEQzLCAweEQ0LCAweEQ1LCAweEQ2LAotICAgICAgICAgICAgMHhENywgMHhEOCwgMHhEOSwgMHhEQSwgMHhFMiwgMHhFMywgMHhFNCwgMHhFNSwgMHhFNiwgMHhFNywgMHhFOCwgMHhFOSwgMHhFQSwgMHhGMiwKLSAgICAgICAgICAgIDB4RjMsIDB4RjQsIDB4RjUsIDB4RjYsIDB4RjcsIDB4RjgsIDB4RjksIDB4RkEKLSAgICB9LCBmYWxzZSk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbGVuZ3Rocy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHNob3J0IGxlbmd0aHNbXTsKLQotICAgIC8qKgotICAgICAqIFRoZSB2YWx1ZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzaG9ydCB2YWx1ZXNbXTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBqUEVHIGh1ZmZtYW4gdGFibGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxlbmd0aHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGhzCi0gICAgICogQHBhcmFtIHZhbHVlcwotICAgICAqICAgICAgICAgICAgdGhlIHZhbHVlcwotICAgICAqIEBwYXJhbSBjb3B5Ci0gICAgICogICAgICAgICAgICB0aGUgY29weQotICAgICAqLwotICAgIEpQRUdIdWZmbWFuVGFibGUoc2hvcnRbXSBsZW5ndGhzLCBzaG9ydFtdIHZhbHVlcywgYm9vbGVhbiBjb3B5KSB7Ci0gICAgICAgIC8vIENvbnN0cnVjdGlvbiBvZiBzdGFuZGFyZCB0YWJsZXMgd2l0aG91dCBjaGVja3MKLSAgICAgICAgLy8gVGhlIHRoaXJkIHBhcmFtIGlzIGR1bW15Ci0gICAgICAgIC8vIENvdWxkIGJlIGFsc28gdXNlZCBmb3IgY29weWluZyBvZiB0aGUgZXhpc3RpbmcgdGFibGVzCi0gICAgICAgIHRoaXMubGVuZ3RocyA9IGxlbmd0aHM7Ci0gICAgICAgIHRoaXMudmFsdWVzID0gdmFsdWVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBKUEVHSHVmZm1hblRhYmxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsZW5ndGhzCi0gICAgICogICAgICAgICAgICB0aGUgYXJyYXkgb2Ygc2hvcnRzIGxlbmd0aHMuCi0gICAgICogQHBhcmFtIHZhbHVlcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHNob3J0cyBjb250YWluaW5nIHRoZSB2YWx1ZXMgaW4gb3JkZXIgb2YKLSAgICAgKiAgICAgICAgICAgIGluY3JlYXNpbmcgY29kZSBsZW5ndGguCi0gICAgICovCi0gICAgcHVibGljIEpQRUdIdWZmbWFuVGFibGUoc2hvcnRbXSBsZW5ndGhzLCBzaG9ydFtdIHZhbHVlcykgewotICAgICAgICBpZiAobGVuZ3RocyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJsZW5ndGhzIGFycmF5IGlzIG51bGwhIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHZhbHVlcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJ2YWx1ZXMgYXJyYXkgaXMgbnVsbCEiKTsKLSAgICAgICAgfQotICAgICAgICBpZiAobGVuZ3Rocy5sZW5ndGggPiAxNikgeyAvLyBBY2NvcmRpbmcgdG8gdGhlIHNwZWMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImxlbmd0aHMgYXJyYXkgaXMgdG9vIGxvbmchIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHZhbHVlcy5sZW5ndGggPiAyNTYpIHsgLy8gQWNjb3JkaW5nIHRvIHRoZSBzcGVjCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJ2YWx1ZXMgYXJyYXkgaXMgdG9vIGxvbmciKTsKLSAgICAgICAgfQotICAgICAgICBmb3IgKHNob3J0IGxlbmd0aCA6IGxlbmd0aHMpIHsKLSAgICAgICAgICAgIGlmIChsZW5ndGggPCAwKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiVmFsdWVzIGluIGxlbmd0aHMgYXJyYXkgbXVzdCBiZSBub24tbmVnYXRpdmUuIik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgZm9yIChzaG9ydCB2YWx1ZSA6IHZhbHVlcykgewotICAgICAgICAgICAgaWYgKHZhbHVlIDwgMCkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIlZhbHVlcyBpbiB2YWx1ZXMgYXJyYXkgbXVzdCBiZSBub24tbmVnYXRpdmUuIik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBjaGVja0h1ZmZtYW5UYWJsZShsZW5ndGhzLCB2YWx1ZXMpOwotCi0gICAgICAgIHRoaXMubGVuZ3RocyA9IG5ldyBzaG9ydFtsZW5ndGhzLmxlbmd0aF07Ci0gICAgICAgIHRoaXMudmFsdWVzID0gbmV3IHNob3J0W3ZhbHVlcy5sZW5ndGhdOwotICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGxlbmd0aHMsIDAsIHRoaXMubGVuZ3RocywgMCwgbGVuZ3Rocy5sZW5ndGgpOwotICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHZhbHVlcywgMCwgdGhpcy52YWx1ZXMsIDAsIHZhbHVlcy5sZW5ndGgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2YgbGVuZ3RocyBpbiB0aGUgSHVmZm1hbiB0YWJsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBzaG9ydCB2YWx1ZXMgcmVwcmVzZW50aW5nIHRoZSBsZW5ndGggdmFsdWVzIGluIHRoZQotICAgICAqICAgICAgICAgSHVmZm1hbiB0YWJsZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc2hvcnRbXSBnZXRMZW5ndGhzKCkgewotICAgICAgICBzaG9ydCBuZXdMZW5ndGhzW10gPSBuZXcgc2hvcnRbbGVuZ3Rocy5sZW5ndGhdOwotICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGxlbmd0aHMsIDAsIG5ld0xlbmd0aHMsIDAsIGxlbmd0aHMubGVuZ3RoKTsKLSAgICAgICAgcmV0dXJuIG5ld0xlbmd0aHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiB2YWx1ZXMgcmVwcmVzZW50ZWQgYnkgaW5jcmVhc2luZyBsZW5ndGggb2YgdGhlaXIgY29kZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2YgdmFsdWVzLgotICAgICAqLwotICAgIHB1YmxpYyBzaG9ydFtdIGdldFZhbHVlcygpIHsKLSAgICAgICAgc2hvcnQgbmV3VmFsdWVzW10gPSBuZXcgc2hvcnRbdmFsdWVzLmxlbmd0aF07Ci0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkodmFsdWVzLCAwLCBuZXdWYWx1ZXMsIDAsIHZhbHVlcy5sZW5ndGgpOwotICAgICAgICByZXR1cm4gbmV3VmFsdWVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrIGh1ZmZtYW4gdGFibGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxlbmd0aHMKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGhzLgotICAgICAqIEBwYXJhbSB2YWx1ZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSB2YWx1ZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgdm9pZCBjaGVja0h1ZmZtYW5UYWJsZShzaG9ydFtdIGxlbmd0aHMsIHNob3J0W10gdmFsdWVzKSB7Ci0gICAgICAgIGludCBudW1MZWF2ZXMgPSAwOwotICAgICAgICBpbnQgcG9zc2libGVMZWF2ZXMgPSAyOwotICAgICAgICBmb3IgKHNob3J0IGxlbmd0aCA6IGxlbmd0aHMpIHsKLSAgICAgICAgICAgIG51bUxlYXZlcyArPSBsZW5ndGg7Ci0gICAgICAgICAgICBwb3NzaWJsZUxlYXZlcyAtPSBsZW5ndGg7Ci0gICAgICAgICAgICBpZiAocG9zc2libGVMZWF2ZXMgPCAwKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigKLSAgICAgICAgICAgICAgICAgICAgICAgICJJbnZhbGlkIEh1ZmZtYW4gdGFibGUgcHJvdmlkZWQsIGxlbmd0aHMgYXJlIGluY29ycmVjdC4iKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHBvc3NpYmxlTGVhdmVzIDw8PSAxOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHZhbHVlcy5sZW5ndGggIT0gbnVtTGVhdmVzKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKAotICAgICAgICAgICAgICAgICAgICAiSW52YWxpZCBIdWZmbWFuIHRhYmxlIHByb3ZpZGVkLCBzdW0gb2YgbGVuZ3RocyAhPSB2YWx1ZXMuIik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBKUEVHSHVmZm1hblRhYmxlIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBKUEVHSHVmZm1hblRhYmxlIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICBTdHJpbmdCdWZmZXIgc2IgPSBuZXcgU3RyaW5nQnVmZmVyKCk7Ci0KLSAgICAgICAgc2IuYXBwZW5kKCJKUEVHSHVmZm1hblRhYmxlOlxubGVuZ3RoczoiKTsKLSAgICAgICAgZm9yIChzaG9ydCBsZW5ndGggOiBsZW5ndGhzKSB7Ci0gICAgICAgICAgICBzYi5hcHBlbmQoJyAnKS5hcHBlbmQobGVuZ3RoKTsKLSAgICAgICAgfQotCi0gICAgICAgIHNiLmFwcGVuZCgiXG52YWx1ZXM6Iik7Ci0gICAgICAgIGZvciAoc2hvcnQgdmFsdWUgOiB2YWx1ZXMpIHsKLSAgICAgICAgICAgIHNiLmFwcGVuZCgnICcpLmFwcGVuZCh2YWx1ZSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gc2IudG9TdHJpbmcoKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR0ltYWdlUmVhZFBhcmFtLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR0ltYWdlUmVhZFBhcmFtLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJmM2E5YTguLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vcGx1Z2lucy9qcGVnL0pQRUdJbWFnZVJlYWRQYXJhbS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTIzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5wbHVnaW5zLmpwZWc7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlUmVhZFBhcmFtOwotCi0vKioKLSAqIFRoZSBKUEVHSW1hZ2VSZWFkUGFyYW0gY2xhc3MgcHJvdmlkZXMgZnVuY3Rpb25hbGl0eSB0byBzZXQgSHVmZm1hbiB0YWJsZXMgYW5kCi0gKiBxdWFudGl6YXRpb24gdGFibGVzIHdoZW4gdXNpbmcgdGhlIEpQRUcgcmVhZGVyIHBsdWctaW4uCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgY2xhc3MgSlBFR0ltYWdlUmVhZFBhcmFtIGV4dGVuZHMgSW1hZ2VSZWFkUGFyYW0gewotCi0gICAgLyoqCi0gICAgICogVGhlIHEgdGFibGVzLgotICAgICAqLwotICAgIHByaXZhdGUgSlBFR1FUYWJsZSBxVGFibGVzW107Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZGMgaHVmZm1hbiB0YWJsZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBKUEVHSHVmZm1hblRhYmxlIGRjSHVmZm1hblRhYmxlc1tdOwotCi0gICAgLyoqCi0gICAgICogVGhlIGFjIGh1ZmZtYW4gdGFibGVzLgotICAgICAqLwotICAgIHByaXZhdGUgSlBFR0h1ZmZtYW5UYWJsZSBhY0h1ZmZtYW5UYWJsZXNbXTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBKUEVHSW1hZ2VSZWFkUGFyYW0uCi0gICAgICovCi0gICAgcHVibGljIEpQRUdJbWFnZVJlYWRQYXJhbSgpIHsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGFibGVzIGFyZSBzZXQsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRhYmxlcyBhcmUgc2V0LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gYXJlVGFibGVzU2V0KCkgewotICAgICAgICByZXR1cm4gcVRhYmxlcyAhPSBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHF1YW50aXphdGlvbiBhbmQgSHVmZm1hbiB0YWJsZXMgZm9yIHVzaW5nIGluIGRlY29kaW5nIHN0cmVhbXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHFUYWJsZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBxdWFudGl6YXRpb24gdGFibGVzLgotICAgICAqIEBwYXJhbSBEQ0h1ZmZtYW5UYWJsZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdGFuZGFydCBEQyBIdWZmbWFuIHRhYmxlcy4KLSAgICAgKiBAcGFyYW0gQUNIdWZmbWFuVGFibGVzCi0gICAgICogICAgICAgICAgICB0aGUgc3RhbmRhcnQgQUMgaHVmZm1hbiB0YWJsZXMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RGVjb2RlVGFibGVzKEpQRUdRVGFibGVbXSBxVGFibGVzLCBKUEVHSHVmZm1hblRhYmxlW10gRENIdWZmbWFuVGFibGVzLAotICAgICAgICAgICAgSlBFR0h1ZmZtYW5UYWJsZVtdIEFDSHVmZm1hblRhYmxlcykgewotICAgICAgICBpZiAocVRhYmxlcyA9PSBudWxsIHx8IERDSHVmZm1hblRhYmxlcyA9PSBudWxsIHx8IEFDSHVmZm1hblRhYmxlcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJJbnZhbGlkIEpQRUcgdGFibGUgYXJyYXlzIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKERDSHVmZm1hblRhYmxlcy5sZW5ndGggIT0gQUNIdWZmbWFuVGFibGVzLmxlbmd0aCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiSW52YWxpZCBKUEVHIHRhYmxlIGFycmF5cyIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChxVGFibGVzLmxlbmd0aCA+IDQgfHwgRENIdWZmbWFuVGFibGVzLmxlbmd0aCA+IDQpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkludmFsaWQgSlBFRyB0YWJsZSBhcnJheXMiKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIERvIHRoZSBzaGFsbG93IGNvcHksIGl0IHNob3VsZCBiZSBlbm91Z2gKLSAgICAgICAgdGhpcy5xVGFibGVzID0gcVRhYmxlcy5jbG9uZSgpOwotICAgICAgICBkY0h1ZmZtYW5UYWJsZXMgPSBEQ0h1ZmZtYW5UYWJsZXMuY2xvbmUoKTsKLSAgICAgICAgYWNIdWZmbWFuVGFibGVzID0gQUNIdWZmbWFuVGFibGVzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVW5zZXQgYWxsIGRlY29kZWQgdGFibGVzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHVuc2V0RGVjb2RlVGFibGVzKCkgewotICAgICAgICBxVGFibGVzID0gbnVsbDsKLSAgICAgICAgZGNIdWZmbWFuVGFibGVzID0gbnVsbDsKLSAgICAgICAgYWNIdWZmbWFuVGFibGVzID0gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBxdWFudGl6YXRpb24gdGFibGVzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHF1YW50aXphdGlvbiB0YWJsZXMsIG9yIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIEpQRUdRVGFibGVbXSBnZXRRVGFibGVzKCkgewotICAgICAgICByZXR1cm4gcVRhYmxlcyA9PSBudWxsID8gbnVsbCA6IHFUYWJsZXMuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBEQyBIdWZmbWFuIHRhYmxlcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBEQyBIdWZmbWFuIHRhYmxlcyB3aGljaCBhcmUgc2V0LCBvciBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBKUEVHSHVmZm1hblRhYmxlW10gZ2V0RENIdWZmbWFuVGFibGVzKCkgewotICAgICAgICByZXR1cm4gZGNIdWZmbWFuVGFibGVzID09IG51bGwgPyBudWxsIDogZGNIdWZmbWFuVGFibGVzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQUMgSHVmZm1hbiB0YWJsZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgQUMgSHVmZm1hbiB0YWJsZXMgd2hpY2ggYXJlIHNldCwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSlBFR0h1ZmZtYW5UYWJsZVtdIGdldEFDSHVmZm1hblRhYmxlcygpIHsKLSAgICAgICAgcmV0dXJuIGFjSHVmZm1hblRhYmxlcyA9PSBudWxsID8gbnVsbCA6IGFjSHVmZm1hblRhYmxlcy5jbG9uZSgpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHSW1hZ2VXcml0ZVBhcmFtLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR0ltYWdlV3JpdGVQYXJhbS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiOTc5OTExLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHSW1hZ2VXcml0ZVBhcmFtLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMDkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvLnBsdWdpbnMuanBlZzsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ucGx1Z2lucy5qcGVnLkpQRUdDb25zdHM7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlV3JpdGVQYXJhbTsKLWltcG9ydCBqYXZhLnV0aWwuTG9jYWxlOwotCi0vKioKLSAqIFRoZSBKUEVHSW1hZ2VXcml0ZVBhcmFtIGNsYXNzIGFsbG93cyB0byBzZXQgSlBFRyBIdWZmbWFuIHRhYmxlcyBhbmQKLSAqIHF1YW50aXphdGlvbiB3aGVuIHVzaW5nIHRoZSBKUEVHIHdyaXRlciBwbHVnLWluLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEpQRUdJbWFnZVdyaXRlUGFyYW0gZXh0ZW5kcyBJbWFnZVdyaXRlUGFyYW0gewotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IENPTVBfUVVBTElUWV9WQUxVRVMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgZmxvYXRbXSBDT01QX1FVQUxJVFlfVkFMVUVTID0gewotICAgICAgICAgICAgMC4wNWYsIDAuNzVmLCAwLjk1ZgotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQ09NUF9RVUFMSVRZX0RFU0NSSVBUSU9OUy4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmdbXSBDT01QX1FVQUxJVFlfREVTQ1JJUFRJT05TID0gewotICAgICAgICAgICAgIk1pbmltdW0gdXNlZnVsIiwgIlZpc3VhbGx5IGxvc3NsZXNzIiwgIk1heGltdW0gdXNlZnVsIgotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcSB0YWJsZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBKUEVHUVRhYmxlW10gcVRhYmxlczsKLQotICAgIC8qKgotICAgICAqIFRoZSBkYyBodWZmbWFuIHRhYmxlcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIEpQRUdIdWZmbWFuVGFibGVbXSBkY0h1ZmZtYW5UYWJsZXM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgYWMgaHVmZm1hbiB0YWJsZXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBKUEVHSHVmZm1hblRhYmxlW10gYWNIdWZmbWFuVGFibGVzOwotCi0gICAgLyoqCi0gICAgICogVGhlIG9wdGltaXplIGh1ZmZtYW4gdGFibGVzLgotICAgICAqLwotICAgIHByaXZhdGUgYm9vbGVhbiBvcHRpbWl6ZUh1ZmZtYW5UYWJsZXM7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSlBFR0ltYWdlV3JpdGVQYXJhbSBvYmplY3Qgd2l0aCB0aGUgc3BlY2lmaWVkIExvY2FsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbG9jYWxlCi0gICAgICogICAgICAgICAgICB0aGUgTG9jYWxlLgotICAgICAqLwotICAgIHB1YmxpYyBKUEVHSW1hZ2VXcml0ZVBhcmFtKExvY2FsZSBsb2NhbGUpIHsKLSAgICAgICAgc3VwZXIobG9jYWxlKTsKLQotICAgICAgICBjYW5Xcml0ZVByb2dyZXNzaXZlID0gdHJ1ZTsKLSAgICAgICAgcHJvZ3Jlc3NpdmVNb2RlID0gSW1hZ2VXcml0ZVBhcmFtLk1PREVfRElTQUJMRUQ7Ci0KLSAgICAgICAgY2FuV3JpdGVDb21wcmVzc2VkID0gdHJ1ZTsKLSAgICAgICAgY29tcHJlc3Npb25UeXBlcyA9IG5ldyBTdHJpbmdbXSB7Ci0gICAgICAgICAgICAiSlBFRyIKLSAgICAgICAgfTsKLSAgICAgICAgY29tcHJlc3Npb25UeXBlID0gY29tcHJlc3Npb25UeXBlc1swXTsKLSAgICAgICAgY29tcHJlc3Npb25RdWFsaXR5ID0gSlBFR0NvbnN0cy5ERUZBVUxUX0pQRUdfQ09NUFJFU1NJT05fUVVBTElUWTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGFibGVzIGFyZSBzZXQsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRhYmxlcyBhcmUgc2V0LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gYXJlVGFibGVzU2V0KCkgewotICAgICAgICByZXR1cm4gcVRhYmxlcyAhPSBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIHF1YW50aXphdGlvbiBhbmQgSHVmZm1hbiB0YWJsZXMgZm9yIHVzaW5nIGluIGVuY29kaW5nIHN0cmVhbXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHFUYWJsZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBxdWFudGl6YXRpb24gdGFibGVzLgotICAgICAqIEBwYXJhbSBEQ0h1ZmZtYW5UYWJsZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBzdGFuZGFydCBEQyBIdWZmbWFuIHRhYmxlcy4KLSAgICAgKiBAcGFyYW0gQUNIdWZmbWFuVGFibGVzCi0gICAgICogICAgICAgICAgICB0aGUgc3RhbmRhcnQgQUMgaHVmZm1hbiB0YWJsZXMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0RW5jb2RlVGFibGVzKEpQRUdRVGFibGVbXSBxVGFibGVzLCBKUEVHSHVmZm1hblRhYmxlW10gRENIdWZmbWFuVGFibGVzLAotICAgICAgICAgICAgSlBFR0h1ZmZtYW5UYWJsZVtdIEFDSHVmZm1hblRhYmxlcykgewotICAgICAgICBpZiAocVRhYmxlcyA9PSBudWxsIHx8IERDSHVmZm1hblRhYmxlcyA9PSBudWxsIHx8IEFDSHVmZm1hblRhYmxlcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJJbnZhbGlkIEpQRUcgdGFibGUgYXJyYXlzIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKERDSHVmZm1hblRhYmxlcy5sZW5ndGggIT0gQUNIdWZmbWFuVGFibGVzLmxlbmd0aCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiSW52YWxpZCBKUEVHIHRhYmxlIGFycmF5cyIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChxVGFibGVzLmxlbmd0aCA+IDQgfHwgRENIdWZmbWFuVGFibGVzLmxlbmd0aCA+IDQpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkludmFsaWQgSlBFRyB0YWJsZSBhcnJheXMiKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIERvIHRoZSBzaGFsbG93IGNvcHksIGl0IHNob3VsZCBiZSBlbm91Z2gKLSAgICAgICAgdGhpcy5xVGFibGVzID0gcVRhYmxlcy5jbG9uZSgpOwotICAgICAgICBkY0h1ZmZtYW5UYWJsZXMgPSBEQ0h1ZmZtYW5UYWJsZXMuY2xvbmUoKTsKLSAgICAgICAgYWNIdWZmbWFuVGFibGVzID0gQUNIdWZmbWFuVGFibGVzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVW5zZXQgYWxsIGVuY29kZWQgdGFibGVzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHVuc2V0RW5jb2RlVGFibGVzKCkgewotICAgICAgICBxVGFibGVzID0gbnVsbDsKLSAgICAgICAgZGNIdWZmbWFuVGFibGVzID0gbnVsbDsKLSAgICAgICAgYWNIdWZmbWFuVGFibGVzID0gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBEQyBIdWZmbWFuIHRhYmxlcy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBEQyBIdWZmbWFuIHRhYmxlcyB3aGljaCBhcmUgc2V0LCBvciBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBKUEVHSHVmZm1hblRhYmxlW10gZ2V0RENIdWZmbWFuVGFibGVzKCkgewotICAgICAgICByZXR1cm4gZGNIdWZmbWFuVGFibGVzID09IG51bGwgPyBudWxsIDogZGNIdWZmbWFuVGFibGVzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgQUMgSHVmZm1hbiB0YWJsZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgQUMgSHVmZm1hbiB0YWJsZXMgd2hpY2ggYXJlIHNldCwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSlBFR0h1ZmZtYW5UYWJsZVtdIGdldEFDSHVmZm1hblRhYmxlcygpIHsKLSAgICAgICAgcmV0dXJuIGFjSHVmZm1hblRhYmxlcyA9PSBudWxsID8gbnVsbCA6IGFjSHVmZm1hblRhYmxlcy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHF1YW50aXphdGlvbiB0YWJsZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgcXVhbnRpemF0aW9uIHRhYmxlcywgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSlBFR1FUYWJsZVtdIGdldFFUYWJsZXMoKSB7Ci0gICAgICAgIHJldHVybiBxVGFibGVzID09IG51bGwgPyBudWxsIDogcVRhYmxlcy5jbG9uZSgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRDb21wcmVzc2lvblF1YWxpdHlEZXNjcmlwdGlvbnMoKSB7Ci0gICAgICAgIHN1cGVyLmdldENvbXByZXNzaW9uUXVhbGl0eURlc2NyaXB0aW9ucygpOwotICAgICAgICByZXR1cm4gQ09NUF9RVUFMSVRZX0RFU0NSSVBUSU9OUy5jbG9uZSgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdFtdIGdldENvbXByZXNzaW9uUXVhbGl0eVZhbHVlcygpIHsKLSAgICAgICAgc3VwZXIuZ2V0Q29tcHJlc3Npb25RdWFsaXR5VmFsdWVzKCk7Ci0gICAgICAgIHJldHVybiBDT01QX1FVQUxJVFlfVkFMVUVTLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgZmxhZyBpbmRpY2F0ZWQgdGhhdCB0aGUgd3JpdGVyIHdpbGwgZ2VuZXJhdGUgb3B0aW1pemVkIEh1ZmZtYW4KLSAgICAgKiB0YWJsZXMgZm9yIHRoZSBpbWFnZSBhcyBwYXJ0IG9mIHRoZSB3cml0aW5nIHByb2Nlc3MuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9wdGltaXplCi0gICAgICogICAgICAgICAgICB0aGUgZmxhZyBvZiBvcHRpbWl6aW5nIGh1ZmZtYW4gdGFibGVzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldE9wdGltaXplSHVmZm1hblRhYmxlcyhib29sZWFuIG9wdGltaXplKSB7Ci0gICAgICAgIG9wdGltaXplSHVmZm1hblRhYmxlcyA9IG9wdGltaXplOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgd3JpdGVyIGdlbmVyYXRlcyBvcHRpbWl6ZWQgSHVmZm1hbiB0YWJsZXMsIGZhbHNlCi0gICAgICogb3RoZXJ3aXNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHdyaXRlciBnZW5lcmF0ZXMgb3B0aW1pemVkIEh1ZmZtYW4gdGFibGVzLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGdldE9wdGltaXplSHVmZm1hblRhYmxlcygpIHsKLSAgICAgICAgcmV0dXJuIG9wdGltaXplSHVmZm1hblRhYmxlczsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NvbXByZXNzaW9uTG9zc2xlc3MoKSB7Ci0gICAgICAgIGlmIChnZXRDb21wcmVzc2lvbk1vZGUoKSAhPSBNT0RFX0VYUExJQ0lUKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKCJDb21wcmVzc2lvbiBtb2RlIG5vdCBNT0RFX0VYUExJQ0lUISIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB1bnNldENvbXByZXNzaW9uKCkgewotICAgICAgICBpZiAoZ2V0Q29tcHJlc3Npb25Nb2RlKCkgIT0gTU9ERV9FWFBMSUNJVCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxTdGF0ZUV4Y2VwdGlvbigiQ29tcHJlc3Npb24gbW9kZSBub3QgTU9ERV9FWFBMSUNJVCEiKTsKLSAgICAgICAgfQotICAgICAgICBjb21wcmVzc2lvblF1YWxpdHkgPSBKUEVHQ29uc3RzLkRFRkFVTFRfSlBFR19DT01QUkVTU0lPTl9RVUFMSVRZOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHUVRhYmxlLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR1FUYWJsZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzNDYxZDQ2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHUVRhYmxlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNjUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5wbHVnaW5zLmpwZWc7Ci0KLS8qKgotICogVGhlIEpQRUdRVGFibGUgY2xhc3MgcmVwcmVzZW50cyBhIHNpbmdsZSBKUEVHIHF1YW50aXphdGlvbiB0YWJsZSBhbmQgcHJvdmlkZXMKLSAqIGZvciB0aGUgc3RhbmRhcmQgdGFibGVzIHRha2VuIGZyb20gdGhlIEpQRUcgc3BlY2lmaWNhdGlvbi4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBKUEVHUVRhYmxlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBTSVpFLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgc3RhdGljIGludCBTSVpFID0gNjQ7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgQkFTRUxJTkVfTUFYLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgc3RhdGljIGludCBCQVNFTElORV9NQVggPSAyNTU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgTUFYLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgc3RhdGljIGludCBNQVggPSAzMjc2NzsKLQotICAgIC8qKgotICAgICAqIFRoZSB0YWJsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludFtdIHRoZVRhYmxlOwotCi0gICAgLyoKLSAgICAgKiBLMSAmIEsyIHRhYmxlcyBjYW4gYmUgZm91bmQgaW4gdGhlIEpQRUcgZm9ybWF0IHNwZWNpZmljYXRpb24gYXQKLSAgICAgKiBodHRwOi8vd3d3LnczLm9yZy9HcmFwaGljcy9KUEVHL2l0dS10ODEucGRmCi0gICAgICovCi0KLSAgICAvKioKLSAgICAgKiBUaGUgQ29uc3RhbnQgSzFMdW1UYWJsZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnRbXSBLMUx1bVRhYmxlID0gbmV3IGludFtdIHsKLSAgICAgICAgICAgIDE2LCAxMSwgMTAsIDE2LCAyNCwgNDAsIDUxLCA2MSwgMTIsIDEyLCAxNCwgMTksIDI2LCA1OCwgNjAsIDU1LCAxNCwgMTMsIDE2LCAyNCwgNDAsIDU3LAotICAgICAgICAgICAgNjksIDU2LCAxNCwgMTcsIDIyLCAyOSwgNTEsIDg3LCA4MCwgNjIsIDE4LCAyMiwgMzcsIDU2LCA2OCwgMTA5LCAxMDMsIDc3LCAyNCwgMzUsIDU1LAotICAgICAgICAgICAgNjQsIDgxLCAxMDQsIDExMywgOTIsIDQ5LCA2NCwgNzgsIDg3LCAxMDMsIDEyMSwgMTIwLCAxMDEsIDcyLCA5MiwgOTUsIDk4LCAxMTIsIDEwMCwKLSAgICAgICAgICAgIDEwMywgOTkKLSAgICB9OwotCi0gICAgLyoqCi0gICAgICogVGhlIENvbnN0YW50IEsyQ2hyVGFibGUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50W10gSzJDaHJUYWJsZSA9IG5ldyBpbnRbXSB7Ci0gICAgICAgICAgICAxNywgMTgsIDI0LCA0NywgOTksIDk5LCA5OSwgOTksIDE4LCAyMSwgMjYsIDY2LCA5OSwgOTksIDk5LCA5OSwgMjQsIDI2LCA1NiwgOTksIDk5LCA5OSwKLSAgICAgICAgICAgIDk5LCA5OSwgNDcsIDY2LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5LAotICAgICAgICAgICAgOTksIDk5LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5LCA5OSwgOTksIDk5Ci0gICAgfTsKLQotICAgIC8qKgotICAgICAqIFRoZSBLMUx1bWluYW5jZSBpbmRpY2F0ZXMgc3RhbmRhcmQgdGFibGUgSy4xIGZyb20gSlBFRyBzcGVjaWZpY2F0aW9uIGFuZAotICAgICAqIHByb2R1Y2VzICJnb29kIiBxdWFsaXR5IG91dHB1dC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIEpQRUdRVGFibGUgSzFMdW1pbmFuY2UgPSBuZXcgSlBFR1FUYWJsZShLMUx1bVRhYmxlKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBLMURpdjJMdW1pbmFuY2UgaW5kaWNhdGVzIEsuMSB0YWJsZSBmcm9tIEpQRUcgc3BlY2lmaWNhdGlvbiB3aXRoIGFsbAotICAgICAqIGVsZW1lbnRzIGRpdmlkZWQgYnkgMiBhbmQgcHJvZHVjZXMgInZlcnkgZ29vZCIgcXVhbGl0eSBvdXRwdXQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBKUEVHUVRhYmxlIEsxRGl2Mkx1bWluYW5jZSA9IEsxTHVtaW5hbmNlLmdldFNjYWxlZEluc3RhbmNlKDAuNWYsIHRydWUpOwotCi0gICAgLyoqCi0gICAgICogVGhlIEsyQ2hyb21pbmFuY2UgaW5kaWNhdGVzIEsuMiB0YWJsZSBmcm9tIEpQRUcgc3BlY2lmaWNhdGlvbiBhbmQKLSAgICAgKiBwcm9kdWNlcyAiZ29vZCIgcXVhbGl0eSBvdXRwdXQuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBKUEVHUVRhYmxlIEsyQ2hyb21pbmFuY2UgPSBuZXcgSlBFR1FUYWJsZShLMkNoclRhYmxlKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBLMkRpdjJDaHJvbWluYW5jZSBpbmRpY2F0ZXMgSy4yIHRhYmxlIGZyb20gSlBFRwotICAgICAqIHNwZWNpZmljYXRpb24gd2l0aCBhbGwgZWxlbWVudHMgZGl2aWRlZCBieSAyIGFuZCBwcm9kdWNlcyAidmVyeSBnb29kIgotICAgICAqIHF1YWxpdHkgb3V0cHV0LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgSlBFR1FUYWJsZSBLMkRpdjJDaHJvbWluYW5jZSA9IEsyQ2hyb21pbmFuY2UuZ2V0U2NhbGVkSW5zdGFuY2UoMC41ZiwgdHJ1ZSk7OwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEpQRUdRVGFibGUgZnJvbSB0aGUgYXJyYXksIHdoaWNoIHNob3VsZCBjb250YWluIDY0Ci0gICAgICogZWxlbWVudHMgaW4gbmF0dXJhbCBvcmRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdGFibGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBxdWFudGl6YXRpb24gdGFibGUuCi0gICAgICovCi0gICAgcHVibGljIEpQRUdRVGFibGUoaW50W10gdGFibGUpIHsKLSAgICAgICAgaWYgKHRhYmxlID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInRhYmxlIHNob3VsZCBub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0gICAgICAgIGlmICh0YWJsZS5sZW5ndGggIT0gU0laRSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiaWxsZWdhbCB0YWJsZSBzaXplOiAiICsgdGFibGUubGVuZ3RoKTsKLSAgICAgICAgfQotICAgICAgICB0aGVUYWJsZSA9IHRhYmxlLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY3VycmVudCBxdWFudGl6YXRpb24gdGFibGUgYXMgYW4gYXJyYXkgb2YgaW50ZWdlciB2YWx1ZXMuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgY3VycmVudCBxdWFudGl6YXRpb24gdGFibGUgYXMgYW4gYXJyYXkgb2YgaW50ZWdlciB2YWx1ZXMuCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldFRhYmxlKCkgewotICAgICAgICByZXR1cm4gdGhlVGFibGUuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBzY2FsZWQgaW5zdGFuY2UgYXMgcXVhbnRpemF0aW9uIHRhYmxlIHdoZXJlIHRoZSB2YWx1ZXMgYXJlCi0gICAgICogbXVsdGlwbGllZCBieSB0aGUgc2NhbGVGYWN0b3IgYW5kIHRoZW4gY2xhbXBlZCBpZiBmb3JjZUJhc2VsaW5lIGlzIHRydWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNjYWxlRmFjdG9yCi0gICAgICogICAgICAgICAgICB0aGUgc2NhbGUgZmFjdG9yIG9mIHRhYmxlLgotICAgICAqIEBwYXJhbSBmb3JjZUJhc2VsaW5lCi0gICAgICogICAgICAgICAgICB0aGUgZm9yY2UgYmFzZWxpbmUgZmxhZywgdGhlIHZhbHVlcyBzaG91bGQgYmUgY2xhbXBlZCBpZiB0cnVlLgotICAgICAqIEByZXR1cm4gdGhlIG5ldyBxdWFudGl6YXRpb24gdGFibGUuCi0gICAgICovCi0gICAgcHVibGljIEpQRUdRVGFibGUgZ2V0U2NhbGVkSW5zdGFuY2UoZmxvYXQgc2NhbGVGYWN0b3IsIGJvb2xlYW4gZm9yY2VCYXNlbGluZSkgewotICAgICAgICBpbnQgdGFibGVbXSA9IG5ldyBpbnRbU0laRV07Ci0KLSAgICAgICAgaW50IG1heFZhbHVlID0gZm9yY2VCYXNlbGluZSA/IEJBU0VMSU5FX01BWCA6IE1BWDsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHRoZVRhYmxlLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBpbnQgcm91bmRlZCA9IE1hdGgucm91bmQodGhlVGFibGVbaV0gKiBzY2FsZUZhY3Rvcik7Ci0gICAgICAgICAgICBpZiAocm91bmRlZCA8IDEpIHsKLSAgICAgICAgICAgICAgICByb3VuZGVkID0gMTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChyb3VuZGVkID4gbWF4VmFsdWUpIHsKLSAgICAgICAgICAgICAgICByb3VuZGVkID0gbWF4VmFsdWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB0YWJsZVtpXSA9IHJvdW5kZWQ7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBKUEVHUVRhYmxlKHRhYmxlKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBKUEVHUVRhYmxlIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhpcyBKUEVHUVRhYmxlIG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgewotICAgICAgICAvLyAtLSBUT0RPIG1vcmUgaW5mb3JtYXRpdmUgaW5mbwotICAgICAgICByZXR1cm4gIkpQRUdRVGFibGUiOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3BsdWdpbnMvanBlZy9wYWNrYWdlLmh0bWwgYi9hd3QvamF2YXgvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvcGFja2FnZS5odG1sCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxNDU3NWM0Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3BsdWdpbnMvanBlZy9wYWNrYWdlLmh0bWwKKysrIC9kZXYvbnVsbApAQCAtMSw4ICswLDAgQEAKLTxodG1sPgotICA8Ym9keT4KLSAgICA8cD4KLSAgICAgIFRoaXMgcGFja2FnZSBjb250YWlucyBhdXhpbGlhcnkgY2xhc3NlcyBmb3IgdGhlIGJ1aWx0LWluIEpQRUcgaW1hZ2UgcGx1Zy1pbi4KLSAgICA8L3A+Ci0gIEBzaW5jZSBBbmRyb2lkIDEuMAotICA8L2JvZHk+Ci08L2h0bWw+CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSUlPUmVnaXN0cnkuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL3NwaS9JSU9SZWdpc3RyeS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwMWRkZWFhLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3NwaS9JSU9SZWdpc3RyeS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTE1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uc3BpOwotCi1pbXBvcnQgamF2YS51dGlsLkFycmF5czsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ucGx1Z2lucy5qcGVnLkpQRUdJbWFnZVJlYWRlclNwaTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMuanBlZy5KUEVHSW1hZ2VXcml0ZXJTcGk7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5wbHVnaW5zLnBuZy5QTkdJbWFnZVJlYWRlclNwaTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMucG5nLlBOR0ltYWdlV3JpdGVyU3BpOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8uc3BpLkZpbGVJSVNTcGk7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5zcGkuRmlsZUlPU1NwaTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnNwaS5JbnB1dFN0cmVhbUlJU1NwaTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnNwaS5PdXRwdXRTdHJlYW1JT1NTcGk7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5zcGkuUkFGSUlTU3BpOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8uc3BpLlJBRklPU1NwaTsKLQotLyoKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YsIFZpc2tvdiBOaWtvbGF5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMyAkCi0gKi8KLQotLyoqCi0gKiBUaGUgSUlPUmVnaXN0cnkgY2xhc3MgcmVnaXN0ZXJzIHNlcnZpY2UgcHJvdmlkZXIgaW5zdGFuY2VzIChTUEkpLiBTZXJ2aWNlCi0gKiBwcm92aWRlciBpbnN0YW5jZXMgYXJlIHJlY29nbml6ZWQgYnkgc3BlY2lmaWMgbWV0YS1pbmZvcm1hdGlvbiBpbiB0aGUgSkFSCi0gKiBmaWxlcyBjb250YWluaW5nIHRoZW0uIFRoZSBKQVIgZmlsZXMgd2l0aCBTUEkgY2xhc3NlcyBhcmUgbG9hZGVkIGZyb20gdGhlCi0gKiBhcHBsaWNhdGlvbiBjbGFzcyBwYXRoLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIElJT1JlZ2lzdHJ5IGV4dGVuZHMgU2VydmljZVJlZ2lzdHJ5IHsKLQotICAgIC8qKgotICAgICAqIFRoZSBpbnN0YW5jZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBJSU9SZWdpc3RyeSBpbnN0YW5jZTsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBDQVRFR09SSUVTLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIENsYXNzW10gQ0FURUdPUklFUyA9IG5ldyBDbGFzc1tdIHsKLSAgICAgICAgICAgIGphdmF4LmltYWdlaW8uc3BpLkltYWdlV3JpdGVyU3BpLmNsYXNzLCBqYXZheC5pbWFnZWlvLnNwaS5JbWFnZVJlYWRlclNwaS5jbGFzcywKLSAgICAgICAgICAgIGphdmF4LmltYWdlaW8uc3BpLkltYWdlSW5wdXRTdHJlYW1TcGkuY2xhc3MsCi0gICAgICAgICAgICAvLyBqYXZheC5pbWFnZWlvLnNwaS5JbWFnZVRyYW5zY29kZXJTcGkuY2xhc3MsCi0gICAgICAgICAgICBqYXZheC5pbWFnZWlvLnNwaS5JbWFnZU91dHB1dFN0cmVhbVNwaS5jbGFzcwotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSUlPIHJlZ2lzdHJ5LgotICAgICAqLwotICAgIHByaXZhdGUgSUlPUmVnaXN0cnkoKSB7Ci0gICAgICAgIHN1cGVyKEFycmF5cy48Q2xhc3M8Pz4+IGFzTGlzdChDQVRFR09SSUVTKS5pdGVyYXRvcigpKTsKLSAgICAgICAgcmVnaXN0ZXJCdWlsdGluU3BpcygpOwotICAgICAgICByZWdpc3RlckFwcGxpY2F0aW9uQ2xhc3NwYXRoU3BpcygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlZ2lzdGVyIGJ1aWx0LWluIFNQSXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSB2b2lkIHJlZ2lzdGVyQnVpbHRpblNwaXMoKSB7Ci0gICAgICAgIHJlZ2lzdGVyU2VydmljZVByb3ZpZGVyKG5ldyBKUEVHSW1hZ2VXcml0ZXJTcGkoKSk7Ci0gICAgICAgIHJlZ2lzdGVyU2VydmljZVByb3ZpZGVyKG5ldyBKUEVHSW1hZ2VSZWFkZXJTcGkoKSk7Ci0gICAgICAgIHJlZ2lzdGVyU2VydmljZVByb3ZpZGVyKG5ldyBQTkdJbWFnZVJlYWRlclNwaSgpKTsKLSAgICAgICAgcmVnaXN0ZXJTZXJ2aWNlUHJvdmlkZXIobmV3IFBOR0ltYWdlV3JpdGVyU3BpKCkpOwotICAgICAgICByZWdpc3RlclNlcnZpY2VQcm92aWRlcihuZXcgRmlsZUlPU1NwaSgpKTsKLSAgICAgICAgcmVnaXN0ZXJTZXJ2aWNlUHJvdmlkZXIobmV3IEZpbGVJSVNTcGkoKSk7Ci0gICAgICAgIHJlZ2lzdGVyU2VydmljZVByb3ZpZGVyKG5ldyBSQUZJT1NTcGkoKSk7Ci0gICAgICAgIHJlZ2lzdGVyU2VydmljZVByb3ZpZGVyKG5ldyBSQUZJSVNTcGkoKSk7Ci0gICAgICAgIHJlZ2lzdGVyU2VydmljZVByb3ZpZGVyKG5ldyBPdXRwdXRTdHJlYW1JT1NTcGkoKSk7Ci0gICAgICAgIHJlZ2lzdGVyU2VydmljZVByb3ZpZGVyKG5ldyBJbnB1dFN0cmVhbUlJU1NwaSgpKTsKLSAgICAgICAgLy8gLS0gVE9ETyBpbXBsZW1lbnQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBkZWZhdWx0IElJT1JlZ2lzdHJ5IGluc3RhbmNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRlZmF1bHQgSUlPUmVnaXN0cnkgaW5zdGFuY2UuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBJSU9SZWdpc3RyeSBnZXREZWZhdWx0SW5zdGFuY2UoKSB7Ci0gICAgICAgIC8vIFRPRE8gaW1wbGVtZW50IG93biBpbnN0YW5jZSBmb3IgZWFjaCBUaHJlYWRHcm91cCAoc2VlIGFsc28KLSAgICAgICAgLy8gVGhyZWFkTG9jYWwpCi0gICAgICAgIHN5bmNocm9uaXplZCAoSUlPUmVnaXN0cnkuY2xhc3MpIHsKLSAgICAgICAgICAgIGlmIChpbnN0YW5jZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaW5zdGFuY2UgPSBuZXcgSUlPUmVnaXN0cnkoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBpbnN0YW5jZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlZ2lzdGVycyBhbGwgc2VydmljZSBwcm92aWRlcnMgZnJvbSB0aGUgYXBwbGljYXRpb24gY2xhc3MgcGF0aC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZWdpc3RlckFwcGxpY2F0aW9uQ2xhc3NwYXRoU3BpcygpIHsKLSAgICAgICAgLy8gLS0gVE9ETyBpbXBsZW1lbnQgZm9yIG5vbi1idWlsdGluIHBsdWdpbnMKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSUlPU2VydmljZVByb3ZpZGVyLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSUlPU2VydmljZVByb3ZpZGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGU5NDc2NzcuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vc3BpL0lJT1NlcnZpY2VQcm92aWRlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTA1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uc3BpOwotCi1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLQotLyoqCi0gKiBUaGUgSUlPU2VydmljZVByb3ZpZGVyIGFic3RyYWN0IGNsYXNzIHByb3ZpZGVzIGJhc2UgZnVuY3Rpb25hbGl0eSBmb3IgSW1hZ2VJTwotICogc2VydmljZSBwcm92aWRlciBpbnRlcmZhY2VzIChTUElzKS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBJSU9TZXJ2aWNlUHJvdmlkZXIgaW1wbGVtZW50cyBSZWdpc3RlcmFibGVTZXJ2aWNlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSB2ZW5kb3IgbmFtZSBvZiB0aGlzIHNlcnZpY2UgcHJvdmlkZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZyB2ZW5kb3JOYW1lOwotCi0gICAgLyoqCi0gICAgICogVGhlIHZlcnNpb24gb2YgdGhpcyBzZXJ2aWNlIHByb3ZpZGVyLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmcgdmVyc2lvbjsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJSU9TZXJ2aWNlUHJvdmlkZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHZlbmRvck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZW5kb3IgbmFtZSBvZiBzZXJ2aWNlIHByb3ZpZGVyLgotICAgICAqIEBwYXJhbSB2ZXJzaW9uCi0gICAgICogICAgICAgICAgICB0aGUgdmVyc2lvbiBvZiBzZXJ2aWNlIHByb3ZpZGVyLgotICAgICAqLwotICAgIHB1YmxpYyBJSU9TZXJ2aWNlUHJvdmlkZXIoU3RyaW5nIHZlbmRvck5hbWUsIFN0cmluZyB2ZXJzaW9uKSB7Ci0gICAgICAgIGlmICh2ZW5kb3JOYW1lID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbigidmVuZG9yIG5hbWUgY2Fubm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotICAgICAgICBpZiAodmVyc2lvbiA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oInZlcnNpb24gbmFtZSBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0gICAgICAgIHRoaXMudmVuZG9yTmFtZSA9IHZlbmRvck5hbWU7Ci0gICAgICAgIHRoaXMudmVyc2lvbiA9IHZlcnNpb247Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IElJT1NlcnZpY2VQcm92aWRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgSUlPU2VydmljZVByb3ZpZGVyKCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgb25SZWdpc3RyYXRpb24oU2VydmljZVJlZ2lzdHJ5IHJlZ2lzdHJ5LCBDbGFzczw/PiBjYXRlZ29yeSkgewotICAgICAgICAvLyB0aGUgZGVmYXVsdCBpbXBsLiBkb2VzIG5vdGhpbmcKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBvbkRlcmVnaXN0cmF0aW9uKFNlcnZpY2VSZWdpc3RyeSByZWdpc3RyeSwgQ2xhc3M8Pz4gY2F0ZWdvcnkpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHZlbmRvciBuYW1lIG9mIHRoaXMgc2VydmljZSBwcm92aWRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB2ZW5kb3IgbmFtZSBvZiB0aGlzIHNlcnZpY2UgcHJvdmlkZXIuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZyBnZXRWZW5kb3JOYW1lKCkgewotICAgICAgICByZXR1cm4gdmVuZG9yTmFtZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSB2ZXJzaW9uIG9mIHRoaXMgc2VydmljZSBwcm92aWRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB2ZXJzaW9uIG9mIHRoaXMgc2VydmljZSBwcm92aWRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldFZlcnNpb24oKSB7Ci0gICAgICAgIHJldHVybiB2ZXJzaW9uOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYSBkZXNjcmlwdGlvbiBvZiB0aGlzIHNlcnZpY2UgcHJvdmlkZXIuIFRoZSByZXN1bHQgc3RyaW5nIHNob3VsZCBiZQotICAgICAqIGxvY2FsaXplZCBmb3IgdGhlIHNwZWNpZmllZCBMb2NhbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGxvY2FsZQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBMb2NhbGUuCi0gICAgICogQHJldHVybiB0aGUgZGVzY3JpcHRpb24gb2YgdGhpcyBzZXJ2aWNlIHByb3ZpZGVyLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBTdHJpbmcgZ2V0RGVzY3JpcHRpb24oTG9jYWxlIGxvY2FsZSk7Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSW1hZ2VJbnB1dFN0cmVhbVNwaS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vc3BpL0ltYWdlSW5wdXRTdHJlYW1TcGkuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZmM4NTlhOC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSW1hZ2VJbnB1dFN0cmVhbVNwaS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTMxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uc3BpOwotCi1pbXBvcnQgamF2YS5pby5GaWxlOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uSW1hZ2VJbnB1dFN0cmVhbTsKLQotLyoqCi0gKiBUaGUgSW1hZ2VJbnB1dFN0cmVhbVNwaSBhYnN0cmFjdCBjbGFzcyBpcyBhIHNlcnZpY2UgcHJvdmlkZXIgaW50ZXJmYWNlIChTUEkpCi0gKiBmb3IgSW1hZ2VJbnB1dFN0cmVhbXMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgSW1hZ2VJbnB1dFN0cmVhbVNwaSBleHRlbmRzIElJT1NlcnZpY2VQcm92aWRlciBpbXBsZW1lbnRzIFJlZ2lzdGVyYWJsZVNlcnZpY2UgewotCi0gICAgLyoqCi0gICAgICogVGhlIGlucHV0IGNsYXNzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBDbGFzczw/PiBpbnB1dENsYXNzOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEltYWdlSW5wdXRTdHJlYW1TcGkuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEltYWdlSW5wdXRTdHJlYW1TcGkoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VJbnB1dFN0cmVhbVNwaS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdmVuZG9yTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIHZlbmRvciBuYW1lLgotICAgICAqIEBwYXJhbSB2ZXJzaW9uCi0gICAgICogICAgICAgICAgICB0aGUgdmVyc2lvbi4KLSAgICAgKiBAcGFyYW0gaW5wdXRDbGFzcwotICAgICAqICAgICAgICAgICAgdGhlIGlucHV0IGNsYXNzLgotICAgICAqLwotICAgIHB1YmxpYyBJbWFnZUlucHV0U3RyZWFtU3BpKFN0cmluZyB2ZW5kb3JOYW1lLCBTdHJpbmcgdmVyc2lvbiwgQ2xhc3M8Pz4gaW5wdXRDbGFzcykgewotICAgICAgICBzdXBlcih2ZW5kb3JOYW1lLCB2ZXJzaW9uKTsKLSAgICAgICAgdGhpcy5pbnB1dENsYXNzID0gaW5wdXRDbGFzczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGlucHV0IENsYXNzIG9iamVjdCB0aGF0IHJlcHJlc2VudHMgY2xhc3Mgb3IgaW50ZXJmYWNlIHRoYXQgbXVzdAotICAgICAqIGJlIGltcGxlbWVudGVkIGJ5IGFuIGlucHV0IHNvdXJjZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBpbnB1dCBjbGFzcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ2xhc3M8Pz4gZ2V0SW5wdXRDbGFzcygpIHsKLSAgICAgICAgcmV0dXJuIGlucHV0Q2xhc3M7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoZSBJbWFnZUlucHV0U3RyZWFtIGNhbiB1c2UgYSBjYWNoZSBmaWxlLiBJZiB0aGlzIG1ldGhvZAotICAgICAqIHJldHVybnMgZmFsc2UsIHRoZSB2YWx1ZSBvZiB0aGUgdXNlQ2FjaGUgcGFyYW1ldGVyIG9mCi0gICAgICogY3JlYXRlSW5wdXRTdHJlYW1JbnN0YW5jZSB3aWxsIGJlIGlnbm9yZWQuIFRoZSBkZWZhdWx0IGltcGxlbWVudGF0aW9uCi0gICAgICogcmV0dXJucyBmYWxzZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBJbWFnZUlucHV0U3RyZWFtIGNhbiB1c2UgYSBjYWNoZSBmaWxlLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNhblVzZUNhY2hlRmlsZSgpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOyAvLyAtLSBkZWYKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIEltYWdlSW5wdXRTdHJlYW0gaW1wbGVtZW50YXRpb24gcmVxdWlyZXMgdGhlIHVzZSBvZiBhCi0gICAgICogY2FjaGUgZmlsZS4gVGhlIGRlZmF1bHQgaW1wbGVtZW50YXRpb24gcmV0dXJucyBmYWxzZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBJbWFnZUlucHV0U3RyZWFtIGltcGxlbWVudGF0aW9uIHJlcXVpcmVzIHRoZSB1c2Ugb2YKLSAgICAgKiAgICAgICAgIGEgY2FjaGUgZmlsZSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIG5lZWRzQ2FjaGVGaWxlKCkgewotICAgICAgICByZXR1cm4gZmFsc2U7IC8vIGRlZgotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIEltYWdlSW5wdXRTdHJlYW0gYXNzb2NpYXRlZCB3aXRoIHRoaXMgc2VydmljZSBwcm92aWRlci4gVGhlCi0gICAgICogaW5wdXQgb2JqZWN0IHNob3VsZCBiZSBhbiBpbnN0YW5jZSBvZiB0aGUgY2xhc3MgcmV0dXJuZWQgYnkgdGhlCi0gICAgICogZ2V0SW5wdXRDbGFzcyBtZXRob2QuIFRoaXMgbWV0aG9kIHVzZXMgdGhlIHNwZWNpZmllZCBkaXJlY3RvcnkgZm9yIHRoZQotICAgICAqIGNhY2hlIGZpbGUgaWYgdGhlIHVzZUNhY2hlIHBhcmFtZXRlciBpcyB0cnVlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbnB1dAotICAgICAqICAgICAgICAgICAgdGhlIGlucHV0IE9iamVjdC4KLSAgICAgKiBAcGFyYW0gdXNlQ2FjaGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbGFnIGluZGljYXRpbmcgaWYgYSBjYWNoZSBmaWxlIGlzIG5lZWRlZCBvciBub3QuCi0gICAgICogQHBhcmFtIGNhY2hlRGlyCi0gICAgICogICAgICAgICAgICB0aGUgY2FjaGUgZGlyZWN0b3J5LgotICAgICAqIEByZXR1cm4gdGhlIEltYWdlSW5wdXRTdHJlYW0uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJbWFnZUlucHV0U3RyZWFtIGNyZWF0ZUlucHV0U3RyZWFtSW5zdGFuY2UoT2JqZWN0IGlucHV0LCBib29sZWFuIHVzZUNhY2hlLAotICAgICAgICAgICAgRmlsZSBjYWNoZURpcikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgSW1hZ2VJbnB1dFN0cmVhbSBhc3NvY2lhdGVkIHdpdGggdGhpcyBzZXJ2aWNlIHByb3ZpZGVyLiBUaGUKLSAgICAgKiBpbnB1dCBvYmplY3Qgc2hvdWxkIGJlIGFuIGluc3RhbmNlIG9mIHRoZSBjbGFzcyByZXR1cm5lZCBieSBnZXRJbnB1dENsYXNzCi0gICAgICogbWV0aG9kLiBUaGlzIG1ldGhvZCB1c2VzIHRoZSBkZWZhdWx0IHN5c3RlbSBkaXJlY3RvcnkgZm9yIHRoZSBjYWNoZSBmaWxlLAotICAgICAqIGlmIGl0IGlzIG5lZWRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5wdXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnB1dCBPYmplY3QuCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VJbnB1dFN0cmVhbS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlSW5wdXRTdHJlYW0gY3JlYXRlSW5wdXRTdHJlYW1JbnN0YW5jZShPYmplY3QgaW5wdXQpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBjcmVhdGVJbnB1dFN0cmVhbUluc3RhbmNlKGlucHV0LCB0cnVlLCBudWxsKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSW1hZ2VPdXRwdXRTdHJlYW1TcGkuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL3NwaS9JbWFnZU91dHB1dFN0cmVhbVNwaS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiN2E5YTVjLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3NwaS9JbWFnZU91dHB1dFN0cmVhbVNwaS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTMyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uc3BpOwotCi1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uSW1hZ2VPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLkZpbGU7Ci0KLS8qKgotICogVGhlIEltYWdlT3V0cHV0U3RyZWFtU3BpIGFic3RyYWN0IGNsYXNzIGlzIGEgc2VydmljZSBwcm92aWRlciBpbnRlcmZhY2UgKFNQSSkKLSAqIGZvciBJbWFnZU91dHB1dFN0cmVhbXMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgSW1hZ2VPdXRwdXRTdHJlYW1TcGkgZXh0ZW5kcyBJSU9TZXJ2aWNlUHJvdmlkZXIgaW1wbGVtZW50cwotICAgICAgICBSZWdpc3RlcmFibGVTZXJ2aWNlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBvdXRwdXQgY2xhc3MuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIENsYXNzPD8+IG91dHB1dENsYXNzOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEltYWdlT3V0cHV0U3RyZWFtU3BpLgotICAgICAqLwotICAgIHByb3RlY3RlZCBJbWFnZU91dHB1dFN0cmVhbVNwaSgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJbWFnZU91dHB1dFN0cmVhbVNwaS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdmVuZG9yTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIHZlbmRvciBuYW1lLgotICAgICAqIEBwYXJhbSB2ZXJzaW9uCi0gICAgICogICAgICAgICAgICB0aGUgdmVyc2lvbi4KLSAgICAgKiBAcGFyYW0gb3V0cHV0Q2xhc3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBvdXRwdXQgY2xhc3MuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlT3V0cHV0U3RyZWFtU3BpKFN0cmluZyB2ZW5kb3JOYW1lLCBTdHJpbmcgdmVyc2lvbiwgQ2xhc3M8Pz4gb3V0cHV0Q2xhc3MpIHsKLSAgICAgICAgc3VwZXIodmVuZG9yTmFtZSwgdmVyc2lvbik7Ci0gICAgICAgIHRoaXMub3V0cHV0Q2xhc3MgPSBvdXRwdXRDbGFzczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIG91dHB1dCBDbGFzcyBvYmplY3QgdGhhdCByZXByZXNlbnRzIHRoZSBjbGFzcyBvciBpbnRlcmZhY2UgdGhhdAotICAgICAqIG11c3QgYmUgaW1wbGVtZW50ZWQgYnkgYW4gb3V0cHV0IHNvdXJjZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBvdXRwdXQgY2xhc3MuCi0gICAgICovCi0gICAgcHVibGljIENsYXNzPD8+IGdldE91dHB1dENsYXNzKCkgewotICAgICAgICByZXR1cm4gb3V0cHV0Q2xhc3M7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoZSBJbWFnZU91dHB1dFN0cmVhbSBjYW4gdXNlIGEgY2FjaGUgZmlsZS4gSWYgdGhpcwotICAgICAqIG1ldGhvZCByZXR1cm5zIGZhbHNlLCB0aGUgdmFsdWUgb2YgdGhlIHVzZUNhY2hlIHBhcmFtZXRlciBvZgotICAgICAqIGNyZWF0ZU91dHB1dFN0cmVhbUluc3RhbmNlIHdpbGwgYmUgaWdub3JlZC4gVGhlIGRlZmF1bHQgaW1wbGVtZW50YXRpb24KLSAgICAgKiByZXR1cm5zIGZhbHNlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIEltYWdlT3V0cHV0U3RyZWFtIGNhbiB1c2UgYSBjYWNoZSBmaWxlLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNhblVzZUNhY2hlRmlsZSgpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOyAvLyBkZWYKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhlIEltYWdlT3V0cHV0U3RyZWFtIGltcGxlbWVudGF0aW9uIHJlcXVpcmVzIHRoZSB1c2Ugb2YKLSAgICAgKiBhIGNhY2hlIGZpbGUuIFRoZSBkZWZhdWx0IGltcGxlbWVudGF0aW9uIHJldHVybnMgZmFsc2UuCi0gICAgICogCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgSW1hZ2VPdXRwdXRTdHJlYW0gaW1wbGVtZW50YXRpb24gcmVxdWlyZXMgdGhlIHVzZSBvZgotICAgICAqICAgICAgICAgYSBjYWNoZSBmaWxlLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gbmVlZHNDYWNoZUZpbGUoKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsgLy8gZGVmCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyB0aGUgSW1hZ2VPdXRwdXRTdHJlYW0gYXNzb2NpYXRlZCB3aXRoIHRoaXMgc2VydmljZSBwcm92aWRlci4gVGhlCi0gICAgICogb3V0cHV0IG9iamVjdCBzaG91bGQgYmUgYW4gaW5zdGFuY2Ugb2YgdGhlIGNsYXNzIHJldHVybmVkIGJ5Ci0gICAgICogZ2V0T3V0cHV0Q2xhc3MgbWV0aG9kLiBUaGlzIG1ldGhvZCB1c2VzIHRoZSBkZWZhdWx0IHN5c3RlbSBkaXJlY3RvcnkgZm9yCi0gICAgICogdGhlIGNhY2hlIGZpbGUsIGlmIGl0IGlzIG5lZWRlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gb3V0cHV0Ci0gICAgICogICAgICAgICAgICB0aGUgb3V0cHV0IE9iamVjdC4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZU91dHB1dFN0cmVhbS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlT3V0cHV0U3RyZWFtIGNyZWF0ZU91dHB1dFN0cmVhbUluc3RhbmNlKE9iamVjdCBvdXRwdXQpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBjcmVhdGVPdXRwdXRTdHJlYW1JbnN0YW5jZShvdXRwdXQsIHRydWUsIG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdGhlIEltYWdlT3V0cHV0U3RyZWFtIGFzc29jaWF0ZWQgd2l0aCB0aGlzIHNlcnZpY2UgcHJvdmlkZXIuIFRoZQotICAgICAqIG91dHB1dCBvYmplY3Qgc2hvdWxkIGJlIGFuIGluc3RhbmNlIG9mIHRoZSBjbGFzcyByZXR1cm5lZCBieQotICAgICAqIGdldElucHV0Q2xhc3MgbWV0aG9kLiBUaGlzIG1ldGhvZCB1c2VzIHRoZSBzcGVjaWZpZWQgZGlyZWN0b3J5IGZvciB0aGUKLSAgICAgKiBjYWNoZSBmaWxlLCBpZiB0aGUgdXNlQ2FjaGUgcGFyYW1ldGVyIGlzIHRydWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIG91dHB1dAotICAgICAqICAgICAgICAgICAgdGhlIG91dHB1dCBPYmplY3QuCi0gICAgICogQHBhcmFtIHVzZUNhY2hlCi0gICAgICogICAgICAgICAgICB0aGUgZmxhZyBpbmRpY2F0aW5nIGlmIGNhY2hlIGZpbGUgaXMgbmVlZGVkIG9yIG5vdC4KLSAgICAgKiBAcGFyYW0gY2FjaGVEaXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBjYWNoZSBkaXJlY3RvcnkuCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VPdXRwdXRTdHJlYW0uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJbWFnZU91dHB1dFN0cmVhbSBjcmVhdGVPdXRwdXRTdHJlYW1JbnN0YW5jZShPYmplY3Qgb3V0cHV0LCBib29sZWFuIHVzZUNhY2hlLAotICAgICAgICAgICAgRmlsZSBjYWNoZURpcikgdGhyb3dzIElPRXhjZXB0aW9uOwotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vc3BpL0ltYWdlUmVhZGVyU3BpLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSW1hZ2VSZWFkZXJTcGkuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMDUyOGQyNS4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSW1hZ2VSZWFkZXJTcGkuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDIwNCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFJ1c3RlbSBWLiBSYWZpa292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMyAkCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvLnNwaTsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uc3RyZWFtLkltYWdlSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5JbWFnZVJlYWRlcjsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotCi0vKioKLSAqIFRoZSBJbWFnZVJlYWRlclNwaSBhYnN0cmFjdCBjbGFzcyBpcyBhIHNlcnZpY2UgcHJvdmlkZXIgaW50ZXJmYWNlIChTUEkpIGZvcgotICogSW1hZ2VSZWFkZXJzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEltYWdlUmVhZGVyU3BpIGV4dGVuZHMgSW1hZ2VSZWFkZXJXcml0ZXJTcGkgewotCi0gICAgLyoqCi0gICAgICogVGhlIFNUQU5EQVJEX0lOUFVUX1RZUEUgY29udGFpbnMgSW1hZ2VJbnB1dFN0cmVhbS5jbGFzcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIENsYXNzW10gU1RBTkRBUkRfSU5QVVRfVFlQRSA9IG5ldyBDbGFzc1tdIHsKLSAgICAgICAgSW1hZ2VJbnB1dFN0cmVhbS5jbGFzcwotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaW5wdXQgdHlwZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIENsYXNzW10gaW5wdXRUeXBlczsKLQotICAgIC8qKgotICAgICAqIFRoZSB3cml0ZXIgU1BJIG5hbWVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmdbXSB3cml0ZXJTcGlOYW1lczsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJbWFnZVJlYWRlclNwaS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgSW1hZ2VSZWFkZXJTcGkoKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VSZWFkZXJTcGkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHZlbmRvck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZW5kb3IgbmFtZS4KLSAgICAgKiBAcGFyYW0gdmVyc2lvbgotICAgICAqICAgICAgICAgICAgdGhlIHZlcnNpb24uCi0gICAgICogQHBhcmFtIG5hbWVzCi0gICAgICogICAgICAgICAgICB0aGUgZm9ybWF0IG5hbWVzLgotICAgICAqIEBwYXJhbSBzdWZmaXhlcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHN0cmluZ3MgcmVwcmVzZW50aW5nIHRoZSBmaWxlIHN1ZmZpeGVzLgotICAgICAqIEBwYXJhbSBNSU1FVHlwZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbiBhcnJheSBvZiBzdHJpbmdzIHJlcHJlc2VudGluZyBNSU1FIHR5cGVzLgotICAgICAqIEBwYXJhbSBwbHVnaW5DbGFzc05hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwbHVnLWluIGNsYXNzIG5hbWUuCi0gICAgICogQHBhcmFtIGlucHV0VHlwZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnB1dCB0eXBlcy4KLSAgICAgKiBAcGFyYW0gd3JpdGVyU3BpTmFtZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBzdHJpbmdzIHdpdGggY2xhc3MgbmFtZXMgb2YgYWxsIGFzc29jaWF0ZWQKLSAgICAgKiAgICAgICAgICAgIEltYWdlV3JpdGVycy4KLSAgICAgKiBAcGFyYW0gc3VwcG9ydHNTdGFuZGFyZFN0cmVhbU1ldGFkYXRhRm9ybWF0Ci0gICAgICogICAgICAgICAgICB0aGUgdmFsdWUgaW5kaWNhdGluZyBpZiBzdHJlYW0gbWV0YWRhdGEgY2FuIGJlIGRlc2NyaWJlZCBieQotICAgICAqICAgICAgICAgICAgc3RhbmRhcmQgbWV0YWRhdGEgZm9ybWF0LgotICAgICAqIEBwYXJhbSBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYXRpdmUgc3RyZWFtIG1ldGFkYXRhIGZvcm1hdCBuYW1lLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0TmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lLgotICAgICAqIEBwYXJhbSBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hdGl2ZSBzdHJlYW0gbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWUsIHJldHVybmVkIGJ5Ci0gICAgICogICAgICAgICAgICBnZXROYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdC4KLSAgICAgKiBAcGFyYW0gZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWVzCi0gICAgICogICAgICAgICAgICB0aGUgZXh0cmEgc3RyZWFtIG1ldGFkYXRhIGZvcm1hdCBuYW1lcywgcmV0dXJuZWQgYnkKLSAgICAgKiAgICAgICAgICAgIGdldEV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lcy4KLSAgICAgKiBAcGFyYW0gZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBleHRyYSBzdHJlYW0gbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWVzLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0U3RyZWFtTWV0YWRhdGFGb3JtYXQuCi0gICAgICogQHBhcmFtIHN1cHBvcnRzU3RhbmRhcmRJbWFnZU1ldGFkYXRhRm9ybWF0Ci0gICAgICogICAgICAgICAgICB0aGUgdmFsdWUgaW5kaWNhdGluZyBpZiBpbWFnZSBtZXRhZGF0YSBjYW4gYmUgZGVzY3JpYmVkIGJ5Ci0gICAgICogICAgICAgICAgICBzdGFuZGFyZCBtZXRhZGF0YSBmb3JtYXQuCi0gICAgICogQHBhcmFtIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmF0aXZlIGltYWdlIG1ldGFkYXRhIGZvcm1hdCBuYW1lLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0TmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWUuCi0gICAgICogQHBhcmFtIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYXRpdmUgaW1hZ2UgbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWUsIHJldHVybmVkIGJ5Ci0gICAgICogICAgICAgICAgICBnZXROYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0LgotICAgICAqIEBwYXJhbSBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXROYW1lcwotICAgICAqICAgICAgICAgICAgdGhlIGV4dHJhIGltYWdlIG1ldGFkYXRhIGZvcm1hdCBuYW1lcywgcmV0dXJuZWQgYnkKLSAgICAgKiAgICAgICAgICAgIGdldEV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWVzLgotICAgICAqIEBwYXJhbSBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzCi0gICAgICogICAgICAgICAgICB0aGUgZXh0cmEgaW1hZ2UgbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWVzLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0SW1hZ2VNZXRhZGF0YUZvcm1hdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VSZWFkZXJTcGkoU3RyaW5nIHZlbmRvck5hbWUsIFN0cmluZyB2ZXJzaW9uLCBTdHJpbmdbXSBuYW1lcywgU3RyaW5nW10gc3VmZml4ZXMsCi0gICAgICAgICAgICBTdHJpbmdbXSBNSU1FVHlwZXMsIFN0cmluZyBwbHVnaW5DbGFzc05hbWUsIENsYXNzW10gaW5wdXRUeXBlcywKLSAgICAgICAgICAgIFN0cmluZ1tdIHdyaXRlclNwaU5hbWVzLCBib29sZWFuIHN1cHBvcnRzU3RhbmRhcmRTdHJlYW1NZXRhZGF0YUZvcm1hdCwKLSAgICAgICAgICAgIFN0cmluZyBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWUsIFN0cmluZyBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZSwKLSAgICAgICAgICAgIFN0cmluZ1tdIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lcywgU3RyaW5nW10gZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMsCi0gICAgICAgICAgICBib29sZWFuIHN1cHBvcnRzU3RhbmRhcmRJbWFnZU1ldGFkYXRhRm9ybWF0LCBTdHJpbmcgbmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWUsCi0gICAgICAgICAgICBTdHJpbmcgbmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdENsYXNzTmFtZSwgU3RyaW5nW10gZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0TmFtZXMsCi0gICAgICAgICAgICBTdHJpbmdbXSBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzKSB7Ci0gICAgICAgIHN1cGVyKHZlbmRvck5hbWUsIHZlcnNpb24sIG5hbWVzLCBzdWZmaXhlcywgTUlNRVR5cGVzLCBwbHVnaW5DbGFzc05hbWUsCi0gICAgICAgICAgICAgICAgc3VwcG9ydHNTdGFuZGFyZFN0cmVhbU1ldGFkYXRhRm9ybWF0LCBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWUsCi0gICAgICAgICAgICAgICAgbmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lcywKLSAgICAgICAgICAgICAgICBleHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcywgc3VwcG9ydHNTdGFuZGFyZEltYWdlTWV0YWRhdGFGb3JtYXQsCi0gICAgICAgICAgICAgICAgbmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWUsIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsCi0gICAgICAgICAgICAgICAgZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0TmFtZXMsIGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMpOwotCi0gICAgICAgIGlmIChpbnB1dFR5cGVzID09IG51bGwgfHwgaW5wdXRUeXBlcy5sZW5ndGggPT0gMCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJpbnB1dCB0eXBlcyBhcnJheSBjYW5ub3QgYmUgTlVMTCBvciBlbXB0eSIpOwotICAgICAgICB9Ci0gICAgICAgIHRoaXMuaW5wdXRUeXBlcyA9IGlucHV0VHlwZXM7Ci0gICAgICAgIHRoaXMud3JpdGVyU3BpTmFtZXMgPSB3cml0ZXJTcGlOYW1lczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIENsYXNzIG9iamVjdHMgd2hvc2UgdHlwZXMgY2FuIGJlIHVzZWQgYXMgaW5wdXQgZm9yIHRoaXMKLSAgICAgKiByZWFkZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgaW5wdXQgdHlwZXMuCi0gICAgICovCi0gICAgcHVibGljIENsYXNzW10gZ2V0SW5wdXRUeXBlcygpIHsKLSAgICAgICAgcmV0dXJuIGlucHV0VHlwZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoZSBmb3JtYXQgb2Ygc291cmNlIG9iamVjdCBpcyBzdXBwb3J0ZWQgYnkgdGhpcyByZWFkZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNvdXJjZQotICAgICAqICAgICAgICAgICAgdGhlIHNvdXJjZSBvYmplY3QgdG8gYmUgZGVjb2RlZCAoZm9yIGV4YW1wbGUgYW4KLSAgICAgKiAgICAgICAgICAgIEltYWdlSW5wdXRTdHJlYW0pLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIGZvcm1hdCBvZiBzb3VyY2Ugb2JqZWN0IGlzIHN1cHBvcnRlZCBieSB0aGlzIHJlYWRlciwKLSAgICAgKiAgICAgICAgIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGJvb2xlYW4gY2FuRGVjb2RlSW5wdXQoT2JqZWN0IHNvdXJjZSkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhbiBpbnN0YW5jZSBvZiB0aGUgSW1hZ2VSZWFkZXIgaW1wbGVtZW50YXRpb24gZm9yIHRoaXMgc2VydmljZQotICAgICAqIHByb3ZpZGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEltYWdlUmVhZGVyLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VSZWFkZXIgY3JlYXRlUmVhZGVySW5zdGFuY2UoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gY3JlYXRlUmVhZGVySW5zdGFuY2UobnVsbCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhbiBpbnN0YW5jZSBvZiB0aGUgSW1hZ2VSZWFkZXIgaW1wbGVtZW50YXRpb24gZm9yIHRoaXMgc2VydmljZQotICAgICAqIHByb3ZpZGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBleHRlbnNpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSBhIHBsdWctaW4gc3BlY2lmaWMgZXh0ZW5zaW9uIG9iamVjdCwgb3IgbnVsbC4KLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZVJlYWRlci4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEltYWdlUmVhZGVyIGNyZWF0ZVJlYWRlckluc3RhbmNlKE9iamVjdCBleHRlbnNpb24pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGUgc3BlY2lmaWVkIEltYWdlUmVhZGVyIG9iamVjdCBpcyBhbiBpbnN0YW5jZSBvZgotICAgICAqIHRoZSBJbWFnZVJlYWRlciBhc3NvY2lhdGVkIHdpdGggdGhpcyBzZXJ2aWNlIHByb3ZpZGVyIG9yIG5vdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmVhZGVyCi0gICAgICogICAgICAgICAgICB0aGUgSW1hZ2VSZWFkZXIuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiB0aGUgc3BlY2lmaWVkIEltYWdlUmVhZGVyIG9iamVjdCBpcyBhbiBpbnN0YW5jZSBvZiB0aGUKLSAgICAgKiAgICAgICAgIEltYWdlUmVhZGVyIGFzc29jaWF0ZWQgd2l0aCB0aGlzIHNlcnZpY2UgcHJvdmlkZXIsIGZhbHNlCi0gICAgICogICAgICAgICBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNPd25SZWFkZXIoSW1hZ2VSZWFkZXIgcmVhZGVyKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIHN0cmluZ3Mgd2l0aCBuYW1lcyBvZiB0aGUgSW1hZ2VXcml0ZXJTcGkgY2xhc3NlcyB0aGF0Ci0gICAgICogc3VwcG9ydCB0aGUgaW50ZXJuYWwgbWV0YWRhdGEgcmVwcmVzZW50YXRpb24gdXNlZCBieSB0aGUgSW1hZ2VSZWFkZXIgb2YKLSAgICAgKiB0aGlzIHNlcnZpY2UgcHJvdmlkZXIsIG9yIG51bGwgaWYgdGhlcmUgYXJlIG5vIHN1Y2ggSW1hZ2VXcml0ZXJzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIHN0cmluZ3Mgd2l0aCBuYW1lcyBvZiB0aGUgSW1hZ2VXcml0ZXJTcGkgY2xhc3Nlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0SW1hZ2VXcml0ZXJTcGlOYW1lcygpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3NwaS9JbWFnZVJlYWRlcldyaXRlclNwaS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vc3BpL0ltYWdlUmVhZGVyV3JpdGVyU3BpLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDljYTA4YjUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vc3BpL0ltYWdlUmVhZGVyV3JpdGVyU3BpLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzNDQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5zcGk7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLm1ldGFkYXRhLklJT01ldGFkYXRhVXRpbHM7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLm1ldGFkYXRhLklJT01ldGFkYXRhRm9ybWF0OwotCi0vKioKLSAqIFRoZSBJbWFnZVJlYWRlcldyaXRlclNwaSBjbGFzcyBpcyBhIHN1cGVyY2xhc3MgZm9yIHRoZSBJbWFnZVJlYWRlclNwaSBhbmQKLSAqIEltYWdlV3JpdGVyU3BpIFNQSXMuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgSW1hZ2VSZWFkZXJXcml0ZXJTcGkgZXh0ZW5kcyBJSU9TZXJ2aWNlUHJvdmlkZXIgaW1wbGVtZW50cwotICAgICAgICBSZWdpc3RlcmFibGVTZXJ2aWNlIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBuYW1lcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgU3RyaW5nW10gbmFtZXM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc3VmZml4ZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZ1tdIHN1ZmZpeGVzOwotCi0gICAgLyoqCi0gICAgICogVGhlIE1JTUUgdHlwZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZ1tdIE1JTUVUeXBlczsKLQotICAgIC8qKgotICAgICAqIFRoZSBwbHVnLWluIGNsYXNzIG5hbWUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZyBwbHVnaW5DbGFzc05hbWU7Ci0KLSAgICAvKioKLSAgICAgKiBXaGV0aGVyIHRoZSByZWFkZXIvd3JpdGVyIHN1cHBvcnRzIHN0YW5kYXJkIHN0cmVhbSBtZXRhZGF0YSBmb3JtYXQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGJvb2xlYW4gc3VwcG9ydHNTdGFuZGFyZFN0cmVhbU1ldGFkYXRhRm9ybWF0OwotCi0gICAgLyoqCi0gICAgICogVGhlIG5hdGl2ZSBzdHJlYW0gbWV0YWRhdGEgZm9ybWF0IG5hbWUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZyBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgbmF0aXZlIHN0cmVhbSBtZXRhZGF0YSBmb3JtYXQgY2xhc3MgbmFtZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgU3RyaW5nIG5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lOwotCi0gICAgLyoqCi0gICAgICogVGhlIGV4dHJhIHN0cmVhbSBtZXRhZGF0YSBmb3JtYXQgbmFtZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZ1tdIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lczsKLQotICAgIC8qKgotICAgICAqIFRoZSBleHRyYSBzdHJlYW0gbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmdbXSBleHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lczsKLQotICAgIC8qKgotICAgICAqIFdoZXRoZXIgdGhlIHJlYWRlci93cml0ZXIgc3VwcG9ydHMgc3RhbmRhcmQgaW1hZ2UgbWV0YWRhdGEgZm9ybWF0LgotICAgICAqLwotICAgIHByb3RlY3RlZCBib29sZWFuIHN1cHBvcnRzU3RhbmRhcmRJbWFnZU1ldGFkYXRhRm9ybWF0OwotCi0gICAgLyoqCi0gICAgICogVGhlIG5hdGl2ZSBpbWFnZSBtZXRhZGF0YSBmb3JtYXQgbmFtZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgU3RyaW5nIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lOwotCi0gICAgLyoqCi0gICAgICogVGhlIG5hdGl2ZSBpbWFnZSBtZXRhZGF0YSBmb3JtYXQgY2xhc3MgbmFtZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgU3RyaW5nIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWU7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZXh0cmEgaW1hZ2UgbWV0YWRhdGEgZm9ybWF0IG5hbWVzLgotICAgICAqLwotICAgIHByb3RlY3RlZCBTdHJpbmdbXSBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXROYW1lczsKLQotICAgIC8qKgotICAgICAqIFRoZSBleHRyYSBpbWFnZSBtZXRhZGF0YSBmb3JtYXQgY2xhc3MgbmFtZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIFN0cmluZ1tdIGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdENsYXNzTmFtZXM7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VSZWFkZXJXcml0ZXJTcGkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHZlbmRvck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZW5kb3IgbmFtZS4KLSAgICAgKiBAcGFyYW0gdmVyc2lvbgotICAgICAqICAgICAgICAgICAgdGhlIHZlcnNpb24uCi0gICAgICogQHBhcmFtIG5hbWVzCi0gICAgICogICAgICAgICAgICB0aGUgZm9ybWF0IG5hbWVzLgotICAgICAqIEBwYXJhbSBzdWZmaXhlcwotICAgICAqICAgICAgICAgICAgdGhlIGFycmF5IG9mIHN0cmluZ3MgcmVwcmVzZW50aW5nIHRoZSBmaWxlIHN1ZmZpeGVzLgotICAgICAqIEBwYXJhbSBNSU1FVHlwZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhbiBhcnJheSBvZiBzdHJpbmdzIHJlcHJlc2VudGluZyBNSU1FIHR5cGVzLgotICAgICAqIEBwYXJhbSBwbHVnaW5DbGFzc05hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBwbHVnLWluIGNsYXNzIG5hbWUuCi0gICAgICogQHBhcmFtIHN1cHBvcnRzU3RhbmRhcmRTdHJlYW1NZXRhZGF0YUZvcm1hdAotICAgICAqICAgICAgICAgICAgdGhlIHZhbHVlIGluZGljYXRpbmcgaWYgc3RyZWFtIG1ldGFkYXRhIGNhbiBiZSBkZXNjcmliZWQgYnkKLSAgICAgKiAgICAgICAgICAgIHN0YW5kYXJkIG1ldGFkYXRhIGZvcm1hdC4KLSAgICAgKiBAcGFyYW0gbmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmF0aXZlIHN0cmVhbSBtZXRhZGF0YSBmb3JtYXQgbmFtZSwgcmV0dXJuZWQgYnkKLSAgICAgKiAgICAgICAgICAgIGdldE5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZS4KLSAgICAgKiBAcGFyYW0gbmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYXRpdmUgc3RyZWFtIG1ldGFkYXRhIGZvcm1hdCBjbGFzcyBuYW1lLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0TmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXQuCi0gICAgICogQHBhcmFtIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lcwotICAgICAqICAgICAgICAgICAgdGhlIGV4dHJhIHN0cmVhbSBtZXRhZGF0YSBmb3JtYXQgbmFtZXMsIHJldHVybmVkIGJ5Ci0gICAgICogICAgICAgICAgICBnZXRFeHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZXMuCi0gICAgICogQHBhcmFtIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzCi0gICAgICogICAgICAgICAgICB0aGUgZXh0cmEgc3RyZWFtIG1ldGFkYXRhIGZvcm1hdCBjbGFzcyBuYW1lcywgcmV0dXJuZWQgYnkKLSAgICAgKiAgICAgICAgICAgIGdldFN0cmVhbU1ldGFkYXRhRm9ybWF0LgotICAgICAqIEBwYXJhbSBzdXBwb3J0c1N0YW5kYXJkSW1hZ2VNZXRhZGF0YUZvcm1hdAotICAgICAqICAgICAgICAgICAgdGhlIHZhbHVlIGluZGljYXRpbmcgaWYgaW1hZ2UgbWV0YWRhdGEgY2FuIGJlIGRlc2NyaWJlZCBieQotICAgICAqICAgICAgICAgICAgc3RhbmRhcmQgbWV0YWRhdGEgZm9ybWF0LgotICAgICAqIEBwYXJhbSBuYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0TmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hdGl2ZSBpbWFnZSBtZXRhZGF0YSBmb3JtYXQgbmFtZSwgcmV0dXJuZWQgYnkKLSAgICAgKiAgICAgICAgICAgIGdldE5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lLgotICAgICAqIEBwYXJhbSBuYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmF0aXZlIGltYWdlIG1ldGFkYXRhIGZvcm1hdCBjbGFzcyBuYW1lLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0TmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdC4KLSAgICAgKiBAcGFyYW0gZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0TmFtZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBleHRyYSBpbWFnZSBtZXRhZGF0YSBmb3JtYXQgbmFtZXMsIHJldHVybmVkIGJ5Ci0gICAgICogICAgICAgICAgICBnZXRFeHRyYUltYWdlTWV0YWRhdGFGb3JtYXROYW1lcy4KLSAgICAgKiBAcGFyYW0gZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcwotICAgICAqICAgICAgICAgICAgdGhlIGV4dHJhIGltYWdlIG1ldGFkYXRhIGZvcm1hdCBjbGFzcyBuYW1lcywgcmV0dXJuZWQgYnkKLSAgICAgKiAgICAgICAgICAgIGdldEltYWdlTWV0YWRhdGFGb3JtYXQuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlUmVhZGVyV3JpdGVyU3BpKFN0cmluZyB2ZW5kb3JOYW1lLCBTdHJpbmcgdmVyc2lvbiwgU3RyaW5nW10gbmFtZXMsCi0gICAgICAgICAgICBTdHJpbmdbXSBzdWZmaXhlcywgU3RyaW5nW10gTUlNRVR5cGVzLCBTdHJpbmcgcGx1Z2luQ2xhc3NOYW1lLAotICAgICAgICAgICAgYm9vbGVhbiBzdXBwb3J0c1N0YW5kYXJkU3RyZWFtTWV0YWRhdGFGb3JtYXQsIFN0cmluZyBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWUsCi0gICAgICAgICAgICBTdHJpbmcgbmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsIFN0cmluZ1tdIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lcywKLSAgICAgICAgICAgIFN0cmluZ1tdIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzLAotICAgICAgICAgICAgYm9vbGVhbiBzdXBwb3J0c1N0YW5kYXJkSW1hZ2VNZXRhZGF0YUZvcm1hdCwgU3RyaW5nIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lLAotICAgICAgICAgICAgU3RyaW5nIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsIFN0cmluZ1tdIGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWVzLAotICAgICAgICAgICAgU3RyaW5nW10gZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcykgewotICAgICAgICBzdXBlcih2ZW5kb3JOYW1lLCB2ZXJzaW9uKTsKLQotICAgICAgICBpZiAobmFtZXMgPT0gbnVsbCB8fCBuYW1lcy5sZW5ndGggPT0gMCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJmb3JtYXQgbmFtZXMgYXJyYXkgY2Fubm90IGJlIE5VTEwgb3IgZW1wdHkiKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChwbHVnaW5DbGFzc05hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJQbHVnaW4gY2xhc3MgbmFtZSBjYW5ub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gV2UgY2xvbmUgYWxsIHRoZSBhcnJheXMgdG8gYmUgY29uc2lzdGVudCB3aXRoIHRoZSBmYWN0IHRoYXQKLSAgICAgICAgLy8gc29tZSBtZXRob2RzIG9mIHRoaXMgY2xhc3MgbXVzdCByZXR1cm4gY2xvbmVzIG9mIHRoZSBhcnJheXMKLSAgICAgICAgLy8gYXMgaXQgaXMgc3RhdGVkIGluIHRoZSBzcGVjLgotICAgICAgICB0aGlzLm5hbWVzID0gbmFtZXMuY2xvbmUoKTsKLSAgICAgICAgdGhpcy5zdWZmaXhlcyA9IHN1ZmZpeGVzID09IG51bGwgPyBudWxsIDogc3VmZml4ZXMuY2xvbmUoKTsKLSAgICAgICAgdGhpcy5NSU1FVHlwZXMgPSBNSU1FVHlwZXMgPT0gbnVsbCA/IG51bGwgOiBNSU1FVHlwZXMuY2xvbmUoKTsKLSAgICAgICAgdGhpcy5wbHVnaW5DbGFzc05hbWUgPSBwbHVnaW5DbGFzc05hbWU7Ci0gICAgICAgIHRoaXMuc3VwcG9ydHNTdGFuZGFyZFN0cmVhbU1ldGFkYXRhRm9ybWF0ID0gc3VwcG9ydHNTdGFuZGFyZFN0cmVhbU1ldGFkYXRhRm9ybWF0OwotICAgICAgICB0aGlzLm5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZSA9IG5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZTsKLSAgICAgICAgdGhpcy5uYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZSA9IG5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lOwotCi0gICAgICAgIHRoaXMuZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWVzID0gZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWVzID09IG51bGwgPyBudWxsCi0gICAgICAgICAgICAgICAgOiBleHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZXMuY2xvbmUoKTsKLQotICAgICAgICB0aGlzLmV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzID0gZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMgPT0gbnVsbCA/IG51bGwKLSAgICAgICAgICAgICAgICA6IGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzLmNsb25lKCk7Ci0KLSAgICAgICAgdGhpcy5zdXBwb3J0c1N0YW5kYXJkSW1hZ2VNZXRhZGF0YUZvcm1hdCA9IHN1cHBvcnRzU3RhbmRhcmRJbWFnZU1ldGFkYXRhRm9ybWF0OwotICAgICAgICB0aGlzLm5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lID0gbmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWU7Ci0gICAgICAgIHRoaXMubmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdENsYXNzTmFtZSA9IG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWU7Ci0KLSAgICAgICAgdGhpcy5leHRyYUltYWdlTWV0YWRhdGFGb3JtYXROYW1lcyA9IGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWVzID09IG51bGwgPyBudWxsCi0gICAgICAgICAgICAgICAgOiBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXROYW1lcy5jbG9uZSgpOwotCi0gICAgICAgIHRoaXMuZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcyA9IGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMgPT0gbnVsbCA/IG51bGwKLSAgICAgICAgICAgICAgICA6IGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VSZWFkZXJXcml0ZXJTcGkuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlUmVhZGVyV3JpdGVyU3BpKCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2Ygc3RyaW5ncyByZXByZXNlbnRpbmcgbmFtZXMgb2YgdGhlIGZvcm1hdHMgdGhhdCBjYW4gYmUKLSAgICAgKiB1c2VkIGJ5IHRoZSBJbWFnZVJlYWRlciBvciBJbWFnZVdyaXRlciBpbXBsZW1lbnRhdGlvbiBhc3NvY2lhdGVkIHdpdGgKLSAgICAgKiB0aGlzIHNlcnZpY2UgcHJvdmlkZXIuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgYXJyYXkgb2Ygc3VwcG9ydGVkIGZvcm1hdCBuYW1lcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0Rm9ybWF0TmFtZXMoKSB7Ci0gICAgICAgIHJldHVybiBuYW1lcy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2Ygc3RyaW5ncyByZXByZXNlbnRpbmcgZmlsZSBzdWZmaXhlcyBhc3NvY2lhdGVkIHdpdGggdGhlCi0gICAgICogZm9ybWF0cyB0aGF0IGNhbiBiZSB1c2VkIGJ5IHRoZSBJbWFnZVJlYWRlciBvciBJbWFnZVdyaXRlciBpbXBsZW1lbnRhdGlvbgotICAgICAqIG9mIHRoaXMgc2VydmljZSBwcm92aWRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBmaWxlIHN1ZmZpeGVzLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRGaWxlU3VmZml4ZXMoKSB7Ci0gICAgICAgIHJldHVybiBzdWZmaXhlcyA9PSBudWxsID8gbnVsbCA6IHN1ZmZpeGVzLmNsb25lKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiBzdHJpbmdzIHdpdGggdGhlIG5hbWVzIG9mIGFkZGl0aW9uYWwgZm9ybWF0cyBvZiB0aGUKLSAgICAgKiBpbWFnZSBtZXRhZGF0YSBvYmplY3RzIHByb2R1Y2VkIG9yIGNvbnN1bWVkIGJ5IHRoaXMgcGx1Zy1pbi4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBleHRyYSBpbWFnZSBtZXRhZGF0YSBmb3JtYXQgbmFtZXMuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZ1tdIGdldEV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWVzKCkgewotICAgICAgICByZXR1cm4gZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0TmFtZXMgPT0gbnVsbCA/IG51bGwgOiBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXROYW1lcy5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gYXJyYXkgb2Ygc3RyaW5ncyB3aXRoIHRoZSBuYW1lcyBvZiBhZGRpdGlvbmFsIGZvcm1hdHMgb2YgdGhlCi0gICAgICogc3RyZWFtIG1ldGFkYXRhIG9iamVjdHMgcHJvZHVjZWQgb3IgY29uc3VtZWQgYnkgdGhpcyBwbHVnLWluLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGFycmF5IG9mIGV4dHJhIHN0cmVhbSBtZXRhZGF0YSBmb3JtYXQgbmFtZXMuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZ1tdIGdldEV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lcygpIHsKLSAgICAgICAgcmV0dXJuIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lcyA9PSBudWxsID8gbnVsbCA6IGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lcwotICAgICAgICAgICAgICAgIC5jbG9uZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gSUlPTWV0YWRhdGFGb3JtYXQgb2JqZWN0IGZvciB0aGUgc3BlY2lmaWVkIGltYWdlIG1ldGFkYXRhIGZvcm1hdAotICAgICAqIG5hbWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZvcm1hdE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBmb3JtYXQgbmFtZS4KLSAgICAgKiBAcmV0dXJuIHRoZSBJSU9NZXRhZGF0YUZvcm1hdCwgb3IgbnVsbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSUlPTWV0YWRhdGFGb3JtYXQgZ2V0SW1hZ2VNZXRhZGF0YUZvcm1hdChTdHJpbmcgZm9ybWF0TmFtZSkgewotICAgICAgICByZXR1cm4gSUlPTWV0YWRhdGFVdGlscy5pbnN0YW50aWF0ZU1ldGFkYXRhRm9ybWF0KGZvcm1hdE5hbWUsCi0gICAgICAgICAgICAgICAgc3VwcG9ydHNTdGFuZGFyZEltYWdlTWV0YWRhdGFGb3JtYXQsIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lLAotICAgICAgICAgICAgICAgIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsIGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWVzLAotICAgICAgICAgICAgICAgIGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYW4gSUlPTWV0YWRhdGFGb3JtYXQgb2JqZWN0IGZvciB0aGUgc3BlY2lmaWVkIHN0cmVhbSBtZXRhZGF0YSBmb3JtYXQKLSAgICAgKiBuYW1lLgotICAgICAqIAotICAgICAqIEBwYXJhbSBmb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgZm9ybWF0IG5hbWUuCi0gICAgICogQHJldHVybiB0aGUgSUlPTWV0YWRhdGFGb3JtYXQsIG9yIG51bGwuCi0gICAgICovCi0gICAgcHVibGljIElJT01ldGFkYXRhRm9ybWF0IGdldFN0cmVhbU1ldGFkYXRhRm9ybWF0KFN0cmluZyBmb3JtYXROYW1lKSB7Ci0gICAgICAgIHJldHVybiBJSU9NZXRhZGF0YVV0aWxzLmluc3RhbnRpYXRlTWV0YWRhdGFGb3JtYXQoZm9ybWF0TmFtZSwKLSAgICAgICAgICAgICAgICBzdXBwb3J0c1N0YW5kYXJkU3RyZWFtTWV0YWRhdGFGb3JtYXQsIG5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZSwKLSAgICAgICAgICAgICAgICBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZSwgZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWVzLAotICAgICAgICAgICAgICAgIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIHN0cmluZ3MgcmVwcmVzZW50aW5nIHRoZSBNSU1FIHR5cGVzIG9mIHRoZSBmb3JtYXRzIHRoYXQKLSAgICAgKiBhcmUgc3VwcG9ydGVkIGJ5IHRoZSBJbWFnZVJlYWRlciBvciBJbWFnZVdyaXRlciBpbXBsZW1lbnRhdGlvbiBvZiB0aGlzCi0gICAgICogc2VydmljZSBwcm92aWRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBNSU1FIHR5cGVzLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRNSU1FVHlwZXMoKSB7Ci0gICAgICAgIHJldHVybiBNSU1FVHlwZXMgPT0gbnVsbCA/IG51bGwgOiBNSU1FVHlwZXMuY2xvbmUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBuYW1lIG9mIHRoZSBuYXRpdmUgaW1hZ2UgbWV0YWRhdGEgZm9ybWF0IGZvciB0aGlzIHJlYWRlci93cml0ZXIsCi0gICAgICogd2hpY2ggYWxsb3dzIGZvciBsb3NzbGVzcyBlbmNvZGluZyBvciBkZWNvZGluZyBvZiB0aGUgaW1hZ2UgbWV0YWRhdGEgd2l0aAotICAgICAqIHRoZSBmb3JtYXQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc3RyaW5nIHdpdGggbmF0aXZlIGltYWdlIG1ldGFkYXRhIGZvcm1hdCBuYW1lLCBvciBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0TmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWUoKSB7Ci0gICAgICAgIHJldHVybiBuYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0TmFtZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBuYW1lIG9mIHRoZSBuYXRpdmUgc3RyZWFtIG1ldGFkYXRhIGZvcm1hdCBmb3IgdGhpcwotICAgICAqIHJlYWRlci93cml0ZXIsIHdoaWNoIGFsbG93cyBmb3IgbG9zc2xlc3MgZW5jb2Rpbmcgb3IgZGVjb2Rpbmcgb2YgdGhlCi0gICAgICogc3RyZWFtIG1ldGFkYXRhIHdpdGggdGhlIGZvcm1hdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgd2l0aCBuYXRpdmUgc3RyZWFtIG1ldGFkYXRhIGZvcm1hdCBuYW1lLCBvciBudWxsLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0TmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lKCkgewotICAgICAgICByZXR1cm4gbmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGNsYXNzIG5hbWUgb2YgdGhlIEltYWdlUmVhZGVyIG9yIEltYWdlV3JpdGVyIGFzc29jaWF0ZWQgd2l0aAotICAgICAqIHRoaXMgc2VydmljZSBwcm92aWRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjbGFzcyBuYW1lLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0UGx1Z2luQ2xhc3NOYW1lKCkgewotICAgICAgICByZXR1cm4gcGx1Z2luQ2xhc3NOYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGUgc3RhbmRhcmQgbWV0YWRhdGEgZm9ybWF0IGlzIHN1cHBvcnRlZCBieSB0aGUgZ2V0QXNUcmVlIGFuZAotICAgICAqIHNldEZyb21UcmVlIG1ldGhvZHMgZm9yIHRoZSBpbWFnZSBtZXRhZGF0YSBvYmplY3RzIHByb2R1Y2VkIG9yIGNvbnN1bWVkCi0gICAgICogYnkgdGhpcyByZWFkZXIgb3Igd3JpdGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgc3RhbmRhcmQgaW1hZ2UgbWV0YWRhdGEgZm9ybWF0IGlzIHN1cHBvcnRlZCwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1N0YW5kYXJkSW1hZ2VNZXRhZGF0YUZvcm1hdFN1cHBvcnRlZCgpIHsKLSAgICAgICAgcmV0dXJuIHN1cHBvcnRzU3RhbmRhcmRJbWFnZU1ldGFkYXRhRm9ybWF0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGUgc3RhbmRhcmQgbWV0YWRhdGEgZm9ybWF0IGlzIHN1cHBvcnRlZCBieSB0aGUgZ2V0QXNUcmVlIGFuZAotICAgICAqIHNldEZyb21UcmVlIG1ldGhvZHMgZm9yIHRoZSBzdHJlYW0gbWV0YWRhdGEgb2JqZWN0cyBwcm9kdWNlZCBvciBjb25zdW1lZAotICAgICAqIGJ5IHRoaXMgcmVhZGVyIG9yIHdyaXRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN0YW5kYXJkIHN0cmVhbSBtZXRhZGF0YSBmb3JtYXQgaXMgc3VwcG9ydGVkLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzU3RhbmRhcmRTdHJlYW1NZXRhZGF0YUZvcm1hdFN1cHBvcnRlZCgpIHsKLSAgICAgICAgcmV0dXJuIHN1cHBvcnRzU3RhbmRhcmRTdHJlYW1NZXRhZGF0YUZvcm1hdDsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSW1hZ2VUcmFuc2NvZGVyU3BpLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvSW1hZ2VUcmFuc2NvZGVyU3BpLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDc0MmFmMTkuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vc3BpL0ltYWdlVHJhbnNjb2RlclNwaS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNzYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5zcGk7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlVHJhbnNjb2RlcjsKLQotLyoqCi0gKiBUaGUgSW1hZ2VUcmFuc2NvZGVyU3BpIGNsYXNzIGlzIGEgc2VydmljZSBwcm92aWRlciBpbnRlcmZhY2UgKFNQSSkgZm9yCi0gKiBJbWFnZVRyYW5zY29kZXJzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEltYWdlVHJhbnNjb2RlclNwaSBleHRlbmRzIElJT1NlcnZpY2VQcm92aWRlciBpbXBsZW1lbnRzIFJlZ2lzdGVyYWJsZVNlcnZpY2UgewotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEltYWdlVHJhbnNjb2RlclNwaS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgSW1hZ2VUcmFuc2NvZGVyU3BpKCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJbWFnZVRyYW5zY29kZXJTcGkgd2l0aCB0aGUgc3BlY2lmaWVkIHZlbmRvciBuYW1lIGFuZAotICAgICAqIHZlcnNpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHZlbmRvck5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZW5kb3IgbmFtZS4KLSAgICAgKiBAcGFyYW0gdmVyc2lvbgotICAgICAqICAgICAgICAgICAgdGhlIHZlcnNpb24uCi0gICAgICovCi0gICAgcHVibGljIEltYWdlVHJhbnNjb2RlclNwaShTdHJpbmcgdmVuZG9yTmFtZSwgU3RyaW5nIHZlcnNpb24pIHsKLSAgICAgICAgc3VwZXIodmVuZG9yTmFtZSwgdmVyc2lvbik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY2xhc3MgbmFtZSBvZiBhbiBJbWFnZVJlYWRlclNwaSB0aGF0IHByb2R1Y2VzIElJT01ldGFkYXRhCi0gICAgICogb2JqZWN0cyB0aGF0IGNhbiBiZSB1c2VkIGFzIGlucHV0IHRvIHRoaXMgdHJhbnNjb2Rlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjbGFzcyBuYW1lIG9mIGFuIEltYWdlUmVhZGVyU3BpLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBTdHJpbmcgZ2V0UmVhZGVyU2VydmljZVByb3ZpZGVyTmFtZSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgY2xhc3MgbmFtZSBvZiBhbiBJbWFnZVdyaXRlclNwaSB0aGF0IHByb2R1Y2VzIElJT01ldGFkYXRhCi0gICAgICogb2JqZWN0cyB0aGF0IGNhbiBiZSB1c2VkIGFzIGlucHV0IHRvIHRoaXMgdHJhbnNjb2Rlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBjbGFzcyBuYW1lIG9mIGFuIEltYWdlV3JpdGVyU3BpLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBTdHJpbmcgZ2V0V3JpdGVyU2VydmljZVByb3ZpZGVyTmFtZSgpOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBhbiBpbnN0YW5jZSBvZiB0aGUgSW1hZ2VUcmFuc2NvZGVyIGFzc29jaWF0ZWQgd2l0aCB0aGlzIHNlcnZpY2UKLSAgICAgKiBwcm92aWRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZVRyYW5zY29kZXIgaW5zdGFuY2UuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEltYWdlVHJhbnNjb2RlciBjcmVhdGVUcmFuc2NvZGVySW5zdGFuY2UoKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3NwaS9JbWFnZVdyaXRlclNwaS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vc3BpL0ltYWdlV3JpdGVyU3BpLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGJmMjU0NTUuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vc3BpL0ltYWdlV3JpdGVyU3BpLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMjcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5zcGk7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLnN0cmVhbS5JbWFnZUlucHV0U3RyZWFtOwotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VUeXBlU3BlY2lmaWVyOwotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VXcml0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmVuZGVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotCi0vKioKLSAqIFRoZSBJbWFnZVdyaXRlclNwaSBhYnN0cmFjdCBjbGFzcyBpcyBhIHNlcnZpY2UgcHJvdmlkZXIgaW50ZXJmYWNlIChTUEkpIGZvcgotICogSW1hZ2VXcml0ZXJzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEltYWdlV3JpdGVyU3BpIGV4dGVuZHMgSW1hZ2VSZWFkZXJXcml0ZXJTcGkgewotCi0gICAgLyoqCi0gICAgICogVGhlIFNUQU5EQVJEX09VVFBVVF9UWVBFIGNvbnRhaW5zIEltYWdlSW5wdXRTdHJlYW0uY2xhc3MuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBDbGFzc1tdIFNUQU5EQVJEX09VVFBVVF9UWVBFID0gbmV3IENsYXNzW10gewotICAgICAgICBJbWFnZUlucHV0U3RyZWFtLmNsYXNzCi0gICAgfTsKLQotICAgIC8qKgotICAgICAqIFRoZSBvdXRwdXQgdHlwZXMuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIENsYXNzW10gb3V0cHV0VHlwZXM7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcmVhZGVyIFNQSSBuYW1lcy4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgU3RyaW5nW10gcmVhZGVyU3BpTmFtZXM7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VXcml0ZXJTcGkuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEltYWdlV3JpdGVyU3BpKCkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEltYWdlV3JpdGVyU3BpIHdpdGggdGhlIHNwZWNpZmllZCBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSB2ZW5kb3JOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgdmVuZG9yIG5hbWUuCi0gICAgICogQHBhcmFtIHZlcnNpb24KLSAgICAgKiAgICAgICAgICAgIHRoZSB2ZXJzaW9uLgotICAgICAqIEBwYXJhbSBuYW1lcwotICAgICAqICAgICAgICAgICAgdGhlIGZvcm1hdCBuYW1lcy4KLSAgICAgKiBAcGFyYW0gc3VmZml4ZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBzdHJpbmdzIHJlcHJlc2VudGluZyB0aGUgZmlsZSBzdWZmaXhlcy4KLSAgICAgKiBAcGFyYW0gTUlNRVR5cGVzCi0gICAgICogICAgICAgICAgICB0aGUgYW4gYXJyYXkgb2Ygc3RyaW5ncyByZXByZXNlbnRpbmcgTUlNRSB0eXBlcy4KLSAgICAgKiBAcGFyYW0gcGx1Z2luQ2xhc3NOYW1lCi0gICAgICogICAgICAgICAgICB0aGUgcGx1Zy1pbiBjbGFzcyBuYW1lLgotICAgICAqIEBwYXJhbSBvdXRwdXRUeXBlcwotICAgICAqICAgICAgICAgICAgdGhlIG91dHB1dCB0eXBlcy4KLSAgICAgKiBAcGFyYW0gcmVhZGVyU3BpTmFtZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBhcnJheSBvZiBzdHJpbmdzIHdpdGggY2xhc3MgbmFtZXMgb2YgYWxsIGFzc29jaWF0ZWQKLSAgICAgKiAgICAgICAgICAgIEltYWdlUmVhZGVycy4KLSAgICAgKiBAcGFyYW0gc3VwcG9ydHNTdGFuZGFyZFN0cmVhbU1ldGFkYXRhRm9ybWF0Ci0gICAgICogICAgICAgICAgICB0aGUgdmFsdWUgaW5kaWNhdGluZyBpZiBzdHJlYW0gbWV0YWRhdGEgY2FuIGJlIGRlc2NyaWJlZCBieQotICAgICAqICAgICAgICAgICAgc3RhbmRhcmQgbWV0YWRhdGEgZm9ybWF0LgotICAgICAqIEBwYXJhbSBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYXRpdmUgc3RyZWFtIG1ldGFkYXRhIGZvcm1hdCBuYW1lLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0TmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lLgotICAgICAqIEBwYXJhbSBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZQotICAgICAqICAgICAgICAgICAgdGhlIG5hdGl2ZSBzdHJlYW0gbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWUsIHJldHVybmVkIGJ5Ci0gICAgICogICAgICAgICAgICBnZXROYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdC4KLSAgICAgKiBAcGFyYW0gZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWVzCi0gICAgICogICAgICAgICAgICB0aGUgZXh0cmEgc3RyZWFtIG1ldGFkYXRhIGZvcm1hdCBuYW1lcywgcmV0dXJuZWQgYnkKLSAgICAgKiAgICAgICAgICAgIGdldEV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lcy4KLSAgICAgKiBAcGFyYW0gZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMKLSAgICAgKiAgICAgICAgICAgIHRoZSBleHRyYSBzdHJlYW0gbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWVzLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0U3RyZWFtTWV0YWRhdGFGb3JtYXQuCi0gICAgICogQHBhcmFtIHN1cHBvcnRzU3RhbmRhcmRJbWFnZU1ldGFkYXRhRm9ybWF0Ci0gICAgICogICAgICAgICAgICB0aGUgdmFsdWUgaW5kaWNhdGluZyBpZiBpbWFnZSBtZXRhZGF0YSBjYW4gYmUgZGVzY3JpYmVkIGJ5Ci0gICAgICogICAgICAgICAgICBzdGFuZGFyZCBtZXRhZGF0YSBmb3JtYXQuCi0gICAgICogQHBhcmFtIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lCi0gICAgICogICAgICAgICAgICB0aGUgbmF0aXZlIGltYWdlIG1ldGFkYXRhIGZvcm1hdCBuYW1lLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0TmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWUuCi0gICAgICogQHBhcmFtIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUKLSAgICAgKiAgICAgICAgICAgIHRoZSBuYXRpdmUgaW1hZ2UgbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWUsIHJldHVybmVkIGJ5Ci0gICAgICogICAgICAgICAgICBnZXROYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0LgotICAgICAqIEBwYXJhbSBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXROYW1lcwotICAgICAqICAgICAgICAgICAgdGhlIGV4dHJhIGltYWdlIG1ldGFkYXRhIGZvcm1hdCBuYW1lcywgcmV0dXJuZWQgYnkKLSAgICAgKiAgICAgICAgICAgIGdldEV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWVzLgotICAgICAqIEBwYXJhbSBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzCi0gICAgICogICAgICAgICAgICB0aGUgZXh0cmEgaW1hZ2UgbWV0YWRhdGEgZm9ybWF0IGNsYXNzIG5hbWVzLCByZXR1cm5lZCBieQotICAgICAqICAgICAgICAgICAgZ2V0SW1hZ2VNZXRhZGF0YUZvcm1hdC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VXcml0ZXJTcGkoU3RyaW5nIHZlbmRvck5hbWUsIFN0cmluZyB2ZXJzaW9uLCBTdHJpbmdbXSBuYW1lcywgU3RyaW5nW10gc3VmZml4ZXMsCi0gICAgICAgICAgICBTdHJpbmdbXSBNSU1FVHlwZXMsIFN0cmluZyBwbHVnaW5DbGFzc05hbWUsIENsYXNzW10gb3V0cHV0VHlwZXMsCi0gICAgICAgICAgICBTdHJpbmdbXSByZWFkZXJTcGlOYW1lcywgYm9vbGVhbiBzdXBwb3J0c1N0YW5kYXJkU3RyZWFtTWV0YWRhdGFGb3JtYXQsCi0gICAgICAgICAgICBTdHJpbmcgbmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lLCBTdHJpbmcgbmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsCi0gICAgICAgICAgICBTdHJpbmdbXSBleHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZXMsIFN0cmluZ1tdIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzLAotICAgICAgICAgICAgYm9vbGVhbiBzdXBwb3J0c1N0YW5kYXJkSW1hZ2VNZXRhZGF0YUZvcm1hdCwgU3RyaW5nIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lLAotICAgICAgICAgICAgU3RyaW5nIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsIFN0cmluZ1tdIGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWVzLAotICAgICAgICAgICAgU3RyaW5nW10gZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcykgewotICAgICAgICBzdXBlcih2ZW5kb3JOYW1lLCB2ZXJzaW9uLCBuYW1lcywgc3VmZml4ZXMsIE1JTUVUeXBlcywgcGx1Z2luQ2xhc3NOYW1lLAotICAgICAgICAgICAgICAgIHN1cHBvcnRzU3RhbmRhcmRTdHJlYW1NZXRhZGF0YUZvcm1hdCwgbmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXROYW1lLAotICAgICAgICAgICAgICAgIG5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lLCBleHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZXMsCi0gICAgICAgICAgICAgICAgZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMsIHN1cHBvcnRzU3RhbmRhcmRJbWFnZU1ldGFkYXRhRm9ybWF0LAotICAgICAgICAgICAgICAgIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lLCBuYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lLAotICAgICAgICAgICAgICAgIGV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWVzLCBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzKTsKLQotICAgICAgICBpZiAob3V0cHV0VHlwZXMgPT0gbnVsbCB8fCBvdXRwdXRUeXBlcy5sZW5ndGggPT0gMCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKCJvdXRwdXQgdHlwZXMgYXJyYXkgY2Fubm90IGJlIE5VTEwgb3IgZW1wdHkiKTsKLSAgICAgICAgfQotCi0gICAgICAgIHRoaXMub3V0cHV0VHlwZXMgPSBvdXRwdXRUeXBlczsKLSAgICAgICAgdGhpcy5yZWFkZXJTcGlOYW1lcyA9IHJlYWRlclNwaU5hbWVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgZm9ybWF0IG9mIHRoZSB3cml0ZXIncyBvdXRwdXQgaXMgbG9zc2xlc3MuIFRoZQotICAgICAqIGRlZmF1bHQgaW1wbGVtZW50YXRpb24gcmV0dXJucyB0cnVlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgYSBmb3JtYXQgaXMgbG9zc2xlc3MsIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0Zvcm1hdExvc3NsZXNzKCkgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIGFycmF5IG9mIENsYXNzIG9iamVjdHMgd2hvc2UgdHlwZXMgY2FuIGJlIHVzZWQgYXMgb3V0cHV0IGZvciB0aGlzCi0gICAgICogd3JpdGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIG91dHB1dCB0eXBlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgQ2xhc3NbXSBnZXRPdXRwdXRUeXBlcygpIHsKLSAgICAgICAgcmV0dXJuIG91dHB1dFR5cGVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIG9yIG5vdCB0aGUgSW1hZ2VXcml0ZXIgaW1wbGVtZW50YXRpb24gYXNzb2NpYXRlZCB3aXRoIHRoaXMKLSAgICAgKiBzZXJ2aWNlIHByb3ZpZGVyIGNhbiBlbmNvZGUgYW4gaW1hZ2Ugd2l0aCB0aGUgc3BlY2lmaWVkIHR5cGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHR5cGUKLSAgICAgKiAgICAgICAgICAgIHRoZSBJbWFnZVR5cGVTcGVjaWZpZXIuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBhbiBpbWFnZSB3aXRoIHRoZSBzcGVjaWZpZWQgdHlwZSBjYW4gYmUgZW5jb2RlZCwgZmFsc2UKLSAgICAgKiAgICAgICAgIG90aGVyd2lzZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgYm9vbGVhbiBjYW5FbmNvZGVJbWFnZShJbWFnZVR5cGVTcGVjaWZpZXIgdHlwZSk7Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIEltYWdlV3JpdGVyIGltcGxlbWVudGF0aW9uIGFzc29jaWF0ZWQgd2l0aCB0aGlzCi0gICAgICogc2VydmljZSBwcm92aWRlciBjYW4gZW5jb2RlIHRoZSBzcGVjaWZpZWQgUmVuZGVyZWRJbWFnZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBSZW5kZXJlZEltYWdlLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgUmVuZGVyZWRJbWFnZSBjYW4gYmUgZW5jb2RlZCwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNhbkVuY29kZUltYWdlKFJlbmRlcmVkSW1hZ2UgaW0pIHsKLSAgICAgICAgcmV0dXJuIGNhbkVuY29kZUltYWdlKEltYWdlVHlwZVNwZWNpZmllci5jcmVhdGVGcm9tUmVuZGVyZWRJbWFnZShpbSkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gaW5zdGFuY2Ugb2YgdGhlIEltYWdlV3JpdGVyIGltcGxlbWVudGF0aW9uIGZvciB0aGlzIHNlcnZpY2UKLSAgICAgKiBwcm92aWRlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBJbWFnZVdyaXRlci4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgcHVibGljIEltYWdlV3JpdGVyIGNyZWF0ZVdyaXRlckluc3RhbmNlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIGNyZWF0ZVdyaXRlckluc3RhbmNlKG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gaW5zdGFuY2Ugb2YgdGhlIEltYWdlV3JpdGVyIGltcGxlbWVudGF0aW9uIGZvciB0aGlzIHNlcnZpY2UKLSAgICAgKiBwcm92aWRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZXh0ZW5zaW9uCi0gICAgICogICAgICAgICAgICB0aGUgYSBwbHVnLWluIHNwZWNpZmljIGV4dGVuc2lvbiBvYmplY3QsIG9yIG51bGwuCi0gICAgICogQHJldHVybiB0aGUgSW1hZ2VXcml0ZXIuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBJbWFnZVdyaXRlciBjcmVhdGVXcml0ZXJJbnN0YW5jZShPYmplY3QgZXh0ZW5zaW9uKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3Mgd2hldGhlciBvciBub3QgdGhlIHNwZWNpZmllZCBJbWFnZVdyaXRlciBvYmplY3QgaXMgYW4gaW5zdGFuY2Ugb2YKLSAgICAgKiB0aGUgSW1hZ2VXcml0ZXIgYXNzb2NpYXRlZCB3aXRoIHRoaXMgc2VydmljZSBwcm92aWRlciBvciBub3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIHdyaXRlcgotICAgICAqICAgICAgICAgICAgdGhlIEltYWdlV3JpdGVyLgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhlIHNwZWNpZmllZCBJbWFnZVdyaXRlciBvYmplY3QgaXMgYW4gaW5zdGFuY2Ugb2YgdGhlCi0gICAgICogICAgICAgICBJbWFnZVdyaXRlciBhc3NvY2lhdGVkIHdpdGggdGhpcyBzZXJ2aWNlIHByb3ZpZGVyLCBmYWxzZQotICAgICAqICAgICAgICAgb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzT3duV3JpdGVyKEltYWdlV3JpdGVyIHdyaXRlcikgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBhcnJheSBvZiBzdHJpbmdzIHdpdGggbmFtZXMgb2YgdGhlIEltYWdlUmVhZGVyU3BpIGNsYXNzZXMgdGhhdAotICAgICAqIHN1cHBvcnQgdGhlIGludGVybmFsIG1ldGFkYXRhIHJlcHJlc2VudGF0aW9uIHVzZWQgYnkgdGhlIEltYWdlV3JpdGVyIG9mCi0gICAgICogdGhpcyBzZXJ2aWNlIHByb3ZpZGVyLCBvciBudWxsIGlmIHRoZXJlIGFyZSBubyBzdWNoIEltYWdlUmVhZGVycy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBhcnJheSBvZiBzdHJpbmdzIHdpdGggbmFtZXMgb2YgdGhlIEltYWdlV3JpdGVyU3BpIGNsYXNzZXMuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZ1tdIGdldEltYWdlUmVhZGVyU3BpTmFtZXMoKSB7Ci0gICAgICAgIHJldHVybiByZWFkZXJTcGlOYW1lczsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvUmVnaXN0ZXJhYmxlU2VydmljZS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vc3BpL1JlZ2lzdGVyYWJsZVNlcnZpY2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYWUyZjRkMy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvUmVnaXN0ZXJhYmxlU2VydmljZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5zcGk7Ci0KLS8qKgotICogVGhlIFJlZ2lzdGVyYWJsZVNlcnZpY2UgaW50ZXJmYWNlIHByb3ZpZGVzIHNlcnZpY2UgcHJvdmlkZXIgb2JqZWN0cyB0aGF0IGNhbgotICogYmUgcmVnaXN0ZXJlZCBieSBhIFNlcnZpY2VSZWdpc3RyeSwgYW5kIG5vdGlmaWNhdGlvbnMgdGhhdCByZWdpc3RyYXRpb24gYW5kCi0gKiBkZXJlZ2lzdHJhdGlvbiBoYXZlIGJlZW4gcGVyZm9ybWVkLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBSZWdpc3RlcmFibGVTZXJ2aWNlIHsKLQotICAgIC8qKgotICAgICAqIFRoaXMgbWV0aG9kIGlzIGNhbGxlZCB3aGVuIHRoZSBvYmplY3Qgd2hpY2ggaW1wbGVtZW50cyB0aGlzIGludGVyZmFjZSBpcwotICAgICAqIHJlZ2lzdGVyZWQgdG8gdGhlIHNwZWNpZmllZCBjYXRlZ29yeSBvZiB0aGUgc3BlY2lmaWVkIHJlZ2lzdHJ5LgotICAgICAqIAotICAgICAqIEBwYXJhbSByZWdpc3RyeQotICAgICAqICAgICAgICAgICAgdGhlIFNlcnZpY2VSZWdpc3RyeSB0byBiZSByZWdpc3RlcmVkLgotICAgICAqIEBwYXJhbSBjYXRlZ29yeQotICAgICAqICAgICAgICAgICAgdGhlIGNsYXNzIHJlcHJlc2VudGluZyBhIGNhdGVnb3J5LgotICAgICAqLwotICAgIHZvaWQgb25SZWdpc3RyYXRpb24oU2VydmljZVJlZ2lzdHJ5IHJlZ2lzdHJ5LCBDbGFzczw/PiBjYXRlZ29yeSk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCBpcyBjYWxsZWQgd2hlbiB0aGUgb2JqZWN0IHdoaWNoIGltcGxlbWVudHMgdGhpcyBpbnRlcmZhY2UgaXMKLSAgICAgKiBkZXJlZ2lzdGVyZWQgdG8gdGhlIHNwZWNpZmllZCBjYXRlZ29yeSBvZiB0aGUgc3BlY2lmaWVkIHJlZ2lzdHJ5LgotICAgICAqIAotICAgICAqIEBwYXJhbSByZWdpc3RyeQotICAgICAqICAgICAgICAgICAgdGhlIFNlcnZpY2VSZWdpc3RyeSB0byBiZSByZWdpc3RlcmVkLgotICAgICAqIEBwYXJhbSBjYXRlZ29yeQotICAgICAqICAgICAgICAgICAgdGhlIGNsYXNzIHJlcHJlc2VudGluZyBhIGNhdGVnb3J5LgotICAgICAqLwotICAgIHZvaWQgb25EZXJlZ2lzdHJhdGlvbihTZXJ2aWNlUmVnaXN0cnkgcmVnaXN0cnksIENsYXNzPD8+IGNhdGVnb3J5KTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3NwaS9TZXJ2aWNlUmVnaXN0cnkuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL3NwaS9TZXJ2aWNlUmVnaXN0cnkuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzliMDJhMy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvU2VydmljZVJlZ2lzdHJ5LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NTIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5zcGk7Ci0KLWltcG9ydCBqYXZhLnV0aWwuKjsKLWltcG9ydCBqYXZhLnV0aWwuTWFwLkVudHJ5OwotCi0vKioKLSAqIFRoZSBTZXJ2aWNlUmVnaXN0cnkgY2xhc3MgcHJvdmlkZXMgYWJpbGl0eSB0byByZWdpc3RlciwgZGVyZWdpc3RlciwgbG9vayB1cAotICogYW5kIG9idGFpbiBzZXJ2aWNlIHByb3ZpZGVyIGluc3RhbmNlcyAoU1BJcykuIEEgc2VydmljZSBtZWFucyBhIHNldCBvZgotICogaW50ZXJmYWNlcyBhbmQgY2xhc3NlcywgYW5kIGEgc2VydmljZSBwcm92aWRlciBpcyBhbiBpbXBsZW1lbnRhdGlvbiBvZiBhCi0gKiBzZXJ2aWNlLiBTZXJ2aWNlIHByb3ZpZGVycyBjYW4gYmUgYXNzb2NpYXRlZCB3aXRoIG9uZSBvciBtb3JlIGNhdGVnb3JpZXMuCi0gKiBFYWNoIGNhdGVnb3J5IGlzIGRlZmluZWQgYnkgYSBjbGFzcyBvciBpbnRlcmZhY2UuIE9ubHkgYSBzaW5nbGUgaW5zdGFuY2Ugb2YgYQotICogZWFjaCBjbGFzcyBpcyBhbGxvd2VkIHRvIGJlIHJlZ2lzdGVyZWQgYXMgYSBjYXRlZ29yeS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBTZXJ2aWNlUmVnaXN0cnkgewotCi0gICAgLyoqCi0gICAgICogVGhlIGNhdGVnb3JpZXMuCi0gICAgICovCi0gICAgQ2F0ZWdvcmllc01hcCBjYXRlZ29yaWVzID0gbmV3IENhdGVnb3JpZXNNYXAodGhpcyk7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgU2VydmljZVJlZ2lzdHJ5IHdpdGggdGhlIHNwZWNpZmllZCBjYXRlZ29yaWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjYXRlZ29yaWVzSXRlcmF0b3IKLSAgICAgKiAgICAgICAgICAgIGFuIEl0ZXJhdG9yIG9mIENsYXNzIG9iamVjdHMgZm9yIGRlZmluaW5nIG9mIGNhdGVnb3JpZXMuCi0gICAgICovCi0gICAgcHVibGljIFNlcnZpY2VSZWdpc3RyeShJdGVyYXRvcjxDbGFzczw/Pj4gY2F0ZWdvcmllc0l0ZXJhdG9yKSB7Ci0gICAgICAgIGlmIChudWxsID09IGNhdGVnb3JpZXNJdGVyYXRvcikgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiY2F0ZWdvcmllcyBpdGVyYXRvciBzaG91bGQgbm90IGJlIE5VTEwiKTsKLSAgICAgICAgfQotICAgICAgICB3aGlsZSAoY2F0ZWdvcmllc0l0ZXJhdG9yLmhhc05leHQoKSkgewotICAgICAgICAgICAgQ2xhc3M8Pz4gYyA9IGNhdGVnb3JpZXNJdGVyYXRvci5uZXh0KCk7Ci0gICAgICAgICAgICBjYXRlZ29yaWVzLmFkZENhdGVnb3J5KGMpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogTG9va3MgdXAgYW5kIGluc3RhbnRpYXRlcyB0aGUgYXZhaWxhYmxlIHByb3ZpZGVycyBvZiB0aGlzIHNlcnZpY2UgdXNpbmcKLSAgICAgKiB0aGUgc3BlY2lmaWVkIGNsYXNzIGxvYWRlci4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvdmlkZXJDbGFzcwotICAgICAqICAgICAgICAgICAgdGhlIENsYXNzIG9iamVjdCBvZiB0aGUgcHJvdmlkZXIgdG8gYmUgbG9va2VkIHVwLgotICAgICAqIEBwYXJhbSBsb2FkZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBjbGFzcyBsb2FkZXIgdG8gYmUgdXNlZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBpdGVyYXRvciBvZiBwcm92aWRlcnMgb2JqZWN0cyBmb3IgdGhpcyBzZXJ2aWNlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgPFQ+IEl0ZXJhdG9yPFQ+IGxvb2t1cFByb3ZpZGVycyhDbGFzczxUPiBwcm92aWRlckNsYXNzLCBDbGFzc0xvYWRlciBsb2FkZXIpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIExvb2tzIHVwIGFuZCBpbnN0YW50aWF0ZXMgdGhlIGF2YWlsYWJsZSBwcm92aWRlcnMgb2YgdGhpcyBzZXJ2aWNlIHVzaW5nCi0gICAgICogdGhlIGNvbnRleHQgY2xhc3MgbG9hZGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm92aWRlckNsYXNzCi0gICAgICogICAgICAgICAgICB0aGUgQ2xhc3Mgb2JqZWN0IG9mIHRoZSBwcm92aWRlciB0byBiZSBsb29rZWQgdXAuCi0gICAgICogQHJldHVybiB0aGUgaXRlcmF0b3Igb2YgcHJvdmlkZXJzIG9iamVjdHMgZm9yIHRoaXMgc2VydmljZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIDxUPiBJdGVyYXRvcjxUPiBsb29rdXBQcm92aWRlcnMoQ2xhc3M8VD4gcHJvdmlkZXJDbGFzcykgewotICAgICAgICByZXR1cm4gbG9va3VwUHJvdmlkZXJzKHByb3ZpZGVyQ2xhc3MsIFRocmVhZC5jdXJyZW50VGhyZWFkKCkuZ2V0Q29udGV4dENsYXNzTG9hZGVyKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlZ2lzdGVycyB0aGUgc3BlY2lmaWVkIHNlcnZpY2UgcHJvdmlkZXIgb2JqZWN0IGluIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBjYXRlZ29yaWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm92aWRlcgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBwcm92aWRlciB0byBiZSByZWdpc3RlcmVkLgotICAgICAqIEBwYXJhbSBjYXRlZ29yeQotICAgICAqICAgICAgICAgICAgdGhlIGNhdGVnb3J5LgotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgbm8gcHJvdmlkZXIgb2YgdGhlIHNhbWUgY2xhc3MgaXMgcmVnaXN0ZXJlZCBpbiB0aGlzCi0gICAgICogICAgICAgICBjYXRlZ29yeSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyA8VD4gYm9vbGVhbiByZWdpc3RlclNlcnZpY2VQcm92aWRlcihUIHByb3ZpZGVyLCBDbGFzczxUPiBjYXRlZ29yeSkgewotICAgICAgICByZXR1cm4gY2F0ZWdvcmllcy5hZGRQcm92aWRlcihwcm92aWRlciwgY2F0ZWdvcnkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlZ2lzdGVycyBhIGxpc3Qgb2Ygc2VydmljZSBwcm92aWRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb3ZpZGVycwotICAgICAqICAgICAgICAgICAgdGhlIGxpc3Qgb2Ygc2VydmljZSBwcm92aWRlcnMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgcmVnaXN0ZXJTZXJ2aWNlUHJvdmlkZXJzKEl0ZXJhdG9yPD8+IHByb3ZpZGVycykgewotICAgICAgICBmb3IgKEl0ZXJhdG9yPD8+IGl0ZXJhdG9yID0gcHJvdmlkZXJzOyBpdGVyYXRvci5oYXNOZXh0KCk7KSB7Ci0gICAgICAgICAgICBjYXRlZ29yaWVzLmFkZFByb3ZpZGVyKGl0ZXJhdG9yLm5leHQoKSwgbnVsbCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZWdpc3RlcnMgdGhlIHNwZWNpZmllZCBzZXJ2aWNlIHByb3ZpZGVyIG9iamVjdCBpbiBhbGwgY2F0ZWdvcmllcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcHJvdmlkZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzZXJ2aWNlIHByb3ZpZGVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHJlZ2lzdGVyU2VydmljZVByb3ZpZGVyKE9iamVjdCBwcm92aWRlcikgewotICAgICAgICBjYXRlZ29yaWVzLmFkZFByb3ZpZGVyKHByb3ZpZGVyLCBudWxsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXJlZ2lzdGVycyB0aGUgc3BlY2lmaWVzIHNlcnZpY2UgcHJvdmlkZXIgZnJvbSB0aGUgc3BlY2lmaWVkIGNhdGVnb3J5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm92aWRlcgotICAgICAqICAgICAgICAgICAgdGhlIHNlcnZpY2UgcHJvdmlkZXIgdG8gYmUgZGVyZWdpc3RlcmVkLgotICAgICAqIEBwYXJhbSBjYXRlZ29yeQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBjYXRlZ29yeS4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBwcm92aWRlciB3YXMgYWxyZWFkeSByZWdpc3RlcmVkIGluIHRoZSBzcGVjaWZpZWQKLSAgICAgKiAgICAgICAgIGNhdGVnb3J5LCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgcHVibGljIDxUPiBib29sZWFuIGRlcmVnaXN0ZXJTZXJ2aWNlUHJvdmlkZXIoVCBwcm92aWRlciwgQ2xhc3M8VD4gY2F0ZWdvcnkpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERlcmVnaXN0ZXJzIHRoZSBzcGVjaWZpZWQgc2VydmljZSBwcm92aWRlciBmcm9tIGFsbCBjYXRlZ29yaWVzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwcm92aWRlcgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBzZXJ2aWNlIHByb3ZpZGVyLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGRlcmVnaXN0ZXJTZXJ2aWNlUHJvdmlkZXIoT2JqZWN0IHByb3ZpZGVyKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIEl0ZXJhdG9yIG9mIHJlZ2lzdGVyZWQgc2VydmljZSBwcm92aWRlcnMgaW4gdGhlIHNwZWNpZmllZAotICAgICAqIGNhdGVnb3J5IHdoaWNoIHNhdGlzZnkgdGhlIHNwZWNpZmllZCBGaWx0ZXIuIFRoZSB1c2VPcmRlcmluZyBwYXJhbWV0ZXIKLSAgICAgKiBpbmRpY2F0ZXMgd2hldGhlciB0aGUgaXRlcmF0b3Igd2lsbCByZXR1cm4gYWxsIG9mIHRoZSBzZXJ2ZXIgcHJvdmlkZXIKLSAgICAgKiBvYmplY3RzIGluIGEgc2V0IG9yZGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjYXRlZ29yeQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBjYXRlZ29yeS4KLSAgICAgKiBAcGFyYW0gZmlsdGVyCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGZpbHRlci4KLSAgICAgKiBAcGFyYW0gdXNlT3JkZXJpbmcKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbGFnIGluZGljYXRpbmcgdGhhdCBwcm92aWRlcnMgYXJlIG9yZGVyZWQgaW4gdGhlIHJldHVybmVkCi0gICAgICogICAgICAgICAgICBJdGVyYXRvci4KLSAgICAgKiBAcmV0dXJuIHRoZSBpdGVyYXRvciBvZiByZWdpc3RlcmVkIHNlcnZpY2UgcHJvdmlkZXJzLgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bmNoZWNrZWQiKQotICAgIHB1YmxpYyA8VD4gSXRlcmF0b3I8VD4gZ2V0U2VydmljZVByb3ZpZGVycyhDbGFzczxUPiBjYXRlZ29yeSwgRmlsdGVyIGZpbHRlciwgYm9vbGVhbiB1c2VPcmRlcmluZykgewotICAgICAgICByZXR1cm4gbmV3IEZpbHRlcmVkSXRlcmF0b3I8VD4oZmlsdGVyLCAoSXRlcmF0b3I8VD4pY2F0ZWdvcmllcy5nZXRQcm92aWRlcnMoY2F0ZWdvcnksCi0gICAgICAgICAgICAgICAgdXNlT3JkZXJpbmcpKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGFuIEl0ZXJhdG9yIG9mIGFsbCByZWdpc3RlcmVkIHNlcnZpY2UgcHJvdmlkZXJzIGluIHRoZSBzcGVjaWZpZWQKLSAgICAgKiBjYXRlZ29yeS4gVGhlIHVzZU9yZGVyaW5nIHBhcmFtZXRlciBpbmRpY2F0ZXMgd2hldGhlciB0aGUgaXRlcmF0b3Igd2lsbAotICAgICAqIHJldHVybiBhbGwgb2YgdGhlIHNlcnZlciBwcm92aWRlciBvYmplY3RzIGluIGEgc2V0IG9yZGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjYXRlZ29yeQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBjYXRlZ29yeS4KLSAgICAgKiBAcGFyYW0gdXNlT3JkZXJpbmcKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbGFnIGluZGljYXRpbmcgdGhhdCBwcm92aWRlcnMgYXJlIG9yZGVyZWQgaW4gdGhlIHJldHVybmVkCi0gICAgICogICAgICAgICAgICBJdGVyYXRvci4KLSAgICAgKiBAcmV0dXJuIHRoZSBJdGVyYXRvciBvZiBzZXJ2aWNlIHByb3ZpZGVycy4KLSAgICAgKi8KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygidW5jaGVja2VkIikKLSAgICBwdWJsaWMgPFQ+IEl0ZXJhdG9yPFQ+IGdldFNlcnZpY2VQcm92aWRlcnMoQ2xhc3M8VD4gY2F0ZWdvcnksIGJvb2xlYW4gdXNlT3JkZXJpbmcpIHsKLSAgICAgICAgcmV0dXJuIChJdGVyYXRvcjxUPiljYXRlZ29yaWVzLmdldFByb3ZpZGVycyhjYXRlZ29yeSwgdXNlT3JkZXJpbmcpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHJlZ2lzdGVyZWQgc2VydmljZSBwcm92aWRlciBvYmplY3QgdGhhdCBoYXMgdGhlIHNwZWNpZmllZCBjbGFzcwotICAgICAqIHR5cGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb3ZpZGVyQ2xhc3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgcHJvdmlkZXIgY2xhc3MuCi0gICAgICogQHJldHVybiB0aGUgc2VydmljZSBwcm92aWRlciBvYmplY3QuCi0gICAgICovCi0gICAgcHVibGljIDxUPiBUIGdldFNlcnZpY2VQcm92aWRlckJ5Q2xhc3MoQ2xhc3M8VD4gcHJvdmlkZXJDbGFzcykgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBhbiBvcmRlcmluZyBiZXR3ZWVuIHR3byBzZXJ2aWNlIHByb3ZpZGVyIG9iamVjdHMgd2l0aGluIHRoZQotICAgICAqIHNwZWNpZmllZCBjYXRlZ29yeS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2F0ZWdvcnkKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgY2F0ZWdvcnkuCi0gICAgICogQHBhcmFtIGZpcnN0UHJvdmlkZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBmaXJzdCBwcm92aWRlci4KLSAgICAgKiBAcGFyYW0gc2Vjb25kUHJvdmlkZXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBzZWNvbmQgcHJvdmlkZXIuCi0gICAgICogQHJldHVybiB0cnVlLCBpZiBhIHByZXZpb3VzbHkgdW5zZXQgb3JkZXIgd2FzIHNldC4KLSAgICAgKi8KLSAgICBwdWJsaWMgPFQ+IGJvb2xlYW4gc2V0T3JkZXJpbmcoQ2xhc3M8VD4gY2F0ZWdvcnksIFQgZmlyc3RQcm92aWRlciwgVCBzZWNvbmRQcm92aWRlcikgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVW5zZXRzIGFuIG9yZGVyaW5nIGJldHdlZW4gdHdvIHNlcnZpY2UgcHJvdmlkZXIgb2JqZWN0cyB3aXRoaW4gdGhlCi0gICAgICogc3BlY2lmaWVkIGNhdGVnb3J5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjYXRlZ29yeQotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBjYXRlZ29yeS4KLSAgICAgKiBAcGFyYW0gZmlyc3RQcm92aWRlcgotICAgICAqICAgICAgICAgICAgdGhlIGZpcnN0IHByb3ZpZGVyLgotICAgICAqIEBwYXJhbSBzZWNvbmRQcm92aWRlcgotICAgICAqICAgICAgICAgICAgdGhlIHNlY29uZCBwcm92aWRlci4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGEgcHJldmlvdXNseSB1bnNldCBvcmRlciB3YXMgcmVtb3ZlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgPFQ+IGJvb2xlYW4gdW5zZXRPcmRlcmluZyhDbGFzczxUPiBjYXRlZ29yeSwgVCBmaXJzdFByb3ZpZGVyLCBUIHNlY29uZFByb3ZpZGVyKSB7Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IHN1cHBvcnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZXJlZ2lzdGVycyBhbGwgcHJvdmlkZXJzIGZyb20gdGhlIHNwZWNpZmllZCBjYXRlZ29yeS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2F0ZWdvcnkKLSAgICAgKiAgICAgICAgICAgIHRoZSBzcGVjaWZpZWQgY2F0ZWdvcnkuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZGVyZWdpc3RlckFsbChDbGFzczw/PiBjYXRlZ29yeSkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGVyZWdpc3RlciBhbGwgcHJvdmlkZXJzIGZyb20gYWxsIGNhdGVnb3JpZXMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgZGVyZWdpc3RlckFsbCgpIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbmFsaXplcyB0aGlzIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAdGhyb3dzIFRocm93YWJsZQotICAgICAqICAgICAgICAgICAgIGlmIGFuIGVycm9yIG9jY3VycyBkdXJpbmcgZmluYWxpemF0aW9uLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGZpbmFsaXplKCkgdGhyb3dzIFRocm93YWJsZSB7Ci0gICAgICAgIC8vIFRPRE8gdW5jb21tZW50IHdoZW4gZGVyZWdpc3RlckFsbCBpcyBpbXBsZW1lbnRlZAotICAgICAgICAvLyBkZXJlZ2lzdGVyQWxsKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHdoZXRoZXIgdGhlIHNwZWNpZmllZCBwcm92aWRlciBoYXMgYmVlbiBhbHJlYWR5IHJlZ2lzdGVyZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHByb3ZpZGVyCi0gICAgICogICAgICAgICAgICB0aGUgcHJvdmlkZXIgdG8gYmUgY2hlY2tlZC4KLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgcHJvdmlkZXIgaGFzIGJlZW4gYWxyZWFkeSByZWdpc3RlcmVkLAotICAgICAqICAgICAgICAgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKE9iamVjdCBwcm92aWRlcikgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbiBpdGVyYXRvciBvZiBDbGFzcyBvYmplY3RzIHJlcHJlc2VudGluZyB0aGUgY3VycmVudCBjYXRlZ29yaWVzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIEl0ZXJhdG9yIG9mIENsYXNzIG9iamVjdHMuCi0gICAgICovCi0gICAgcHVibGljIEl0ZXJhdG9yPENsYXNzPD8+PiBnZXRDYXRlZ29yaWVzKCkgewotICAgICAgICByZXR1cm4gY2F0ZWdvcmllcy5saXN0KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIFNlcnZpY2VSZWdpc3RyeS5GaWx0ZXIgaW50ZXJmYWNlIGlzIHVzZWQgYnkKLSAgICAgKiBTZXJ2aWNlUmVnaXN0cnkuZ2V0U2VydmljZVByb3ZpZGVycyB0byBmaWx0ZXIgcHJvdmlkZXJzIGFjY29yZGluZyB0byB0aGUKLSAgICAgKiBzcGVjaWZpZWQgY3JpdGVyaW9uLgotICAgICAqIAotICAgICAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgaW50ZXJmYWNlIEZpbHRlciB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgc3BlY2lmaWVkIHByb3ZpZGVyIHNhdGlzZmllcyB0aGUgY3JpdGVyaW9uIG9mCi0gICAgICAgICAqIHRoaXMgRmlsdGVyLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHByb3ZpZGVyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHByb3ZpZGVyLgotICAgICAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZSBzcGVjaWZpZWQgcHJvdmlkZXIgc2F0aXNmaWVzIHRoZSBjcml0ZXJpb24gb2YKLSAgICAgICAgICogICAgICAgICB0aGlzIEZpbHRlciwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAgICAgKi8KLSAgICAgICAgYm9vbGVhbiBmaWx0ZXIoT2JqZWN0IHByb3ZpZGVyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgQ2xhc3MgQ2F0ZWdvcmllc01hcC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBjbGFzcyBDYXRlZ29yaWVzTWFwIHsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIGNhdGVnb3JpZXMuCi0gICAgICAgICAqLwotICAgICAgICBNYXA8Q2xhc3M8Pz4sIFByb3ZpZGVyc01hcD4gY2F0ZWdvcmllcyA9IG5ldyBIYXNoTWFwPENsYXNzPD8+LCBQcm92aWRlcnNNYXA+KCk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSByZWdpc3RyeS4KLSAgICAgICAgICovCi0gICAgICAgIFNlcnZpY2VSZWdpc3RyeSByZWdpc3RyeTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGNhdGVnb3JpZXMgbWFwLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHJlZ2lzdHJ5Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHJlZ2lzdHJ5LgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIENhdGVnb3JpZXNNYXAoU2VydmljZVJlZ2lzdHJ5IHJlZ2lzdHJ5KSB7Ci0gICAgICAgICAgICB0aGlzLnJlZ2lzdHJ5ID0gcmVnaXN0cnk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyAtLSBUT0RPOiB1c2VPcmRlcmluZwotICAgICAgICAvKioKLSAgICAgICAgICogR2V0cyB0aGUgcHJvdmlkZXJzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGNhdGVnb3J5Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGNhdGVnb3J5LgotICAgICAgICAgKiBAcGFyYW0gdXNlT3JkZXJpbmcKLSAgICAgICAgICogICAgICAgICAgICB0aGUgdXNlIG9yZGVyaW5nLgotICAgICAgICAgKiBAcmV0dXJuIHRoZSBwcm92aWRlcnMuCi0gICAgICAgICAqLwotICAgICAgICBJdGVyYXRvcjw/PiBnZXRQcm92aWRlcnMoQ2xhc3M8Pz4gY2F0ZWdvcnksIGJvb2xlYW4gdXNlT3JkZXJpbmcpIHsKLSAgICAgICAgICAgIFByb3ZpZGVyc01hcCBwcm92aWRlcnMgPSBjYXRlZ29yaWVzLmdldChjYXRlZ29yeSk7Ci0gICAgICAgICAgICBpZiAobnVsbCA9PSBwcm92aWRlcnMpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJVbmtub3duIGNhdGVnb3J5OiAiICsgY2F0ZWdvcnkpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIHByb3ZpZGVycy5nZXRQcm92aWRlcnModXNlT3JkZXJpbmcpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIExpc3QuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBpdGVyYXRvcjwgY2xhc3M8Pz4+LgotICAgICAgICAgKi8KLSAgICAgICAgSXRlcmF0b3I8Q2xhc3M8Pz4+IGxpc3QoKSB7Ci0gICAgICAgICAgICByZXR1cm4gY2F0ZWdvcmllcy5rZXlTZXQoKS5pdGVyYXRvcigpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEFkZHMgdGhlIGNhdGVnb3J5LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGNhdGVnb3J5Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIGNhdGVnb3J5LgotICAgICAgICAgKi8KLSAgICAgICAgdm9pZCBhZGRDYXRlZ29yeShDbGFzczw/PiBjYXRlZ29yeSkgewotICAgICAgICAgICAgY2F0ZWdvcmllcy5wdXQoY2F0ZWdvcnksIG5ldyBQcm92aWRlcnNNYXAoKSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQWRkcyBhIHByb3ZpZGVyIHRvIHRoZSBjYXRlZ29yeS4gSWYgPGNvZGU+Y2F0ZWdvcnk8L2NvZGU+IGlzCi0gICAgICAgICAqIDxjb2RlPm51bGw8L2NvZGU+IHRoZW4gdGhlIHByb3ZpZGVyIHdpbGwgYmUgYWRkZWQgdG8gYWxsIGNhdGVnb3JpZXMKLSAgICAgICAgICogd2hpY2ggdGhlIHByb3ZpZGVyIGlzIGFzc2lnbmFibGUgZnJvbS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBwcm92aWRlcgotICAgICAgICAgKiAgICAgICAgICAgIHByb3ZpZGVyIHRvIGFkZC4KLSAgICAgICAgICogQHBhcmFtIGNhdGVnb3J5Ci0gICAgICAgICAqICAgICAgICAgICAgY2F0ZWdvcnkgdG8gYWRkIHByb3ZpZGVyIHRvLgotICAgICAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoZXJlIHdlcmUgc3VjaCBwcm92aWRlciBpbiBzb21lIGNhdGVnb3J5LgotICAgICAgICAgKi8KLSAgICAgICAgYm9vbGVhbiBhZGRQcm92aWRlcihPYmplY3QgcHJvdmlkZXIsIENsYXNzPD8+IGNhdGVnb3J5KSB7Ci0gICAgICAgICAgICBpZiAocHJvdmlkZXIgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInByb3ZpZGVyIHNob3VsZCBiZSAhPSBOVUxMIik7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGJvb2xlYW4gcnQ7Ci0gICAgICAgICAgICBpZiAoY2F0ZWdvcnkgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHJ0ID0gZmluZEFuZEFkZChwcm92aWRlcik7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJ0ID0gYWRkVG9OYW1lZChwcm92aWRlciwgY2F0ZWdvcnkpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAocHJvdmlkZXIgaW5zdGFuY2VvZiBSZWdpc3RlcmFibGVTZXJ2aWNlKSB7Ci0gICAgICAgICAgICAgICAgKChSZWdpc3RlcmFibGVTZXJ2aWNlKXByb3ZpZGVyKS5vblJlZ2lzdHJhdGlvbihyZWdpc3RyeSwgY2F0ZWdvcnkpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZXR1cm4gcnQ7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQWRkcyB0aGUgdG8gbmFtZWQuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcGFyYW0gcHJvdmlkZXIKLSAgICAgICAgICogICAgICAgICAgICB0aGUgcHJvdmlkZXIuCi0gICAgICAgICAqIEBwYXJhbSBjYXRlZ29yeQotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBjYXRlZ29yeS4KLSAgICAgICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBib29sZWFuIGFkZFRvTmFtZWQoT2JqZWN0IHByb3ZpZGVyLCBDbGFzczw/PiBjYXRlZ29yeSkgewotICAgICAgICAgICAgT2JqZWN0IG9iaiA9IGNhdGVnb3JpZXMuZ2V0KGNhdGVnb3J5KTsKLQotICAgICAgICAgICAgaWYgKG51bGwgPT0gb2JqKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiVW5rbm93biBjYXRlZ29yeTogIiArIGNhdGVnb3J5KTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuICgoUHJvdmlkZXJzTWFwKW9iaikuYWRkUHJvdmlkZXIocHJvdmlkZXIpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEZpbmQgYW5kIGFkZC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBwcm92aWRlcgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBwcm92aWRlci4KLSAgICAgICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBib29sZWFuIGZpbmRBbmRBZGQoT2JqZWN0IHByb3ZpZGVyKSB7Ci0gICAgICAgICAgICBib29sZWFuIHJ0ID0gZmFsc2U7Ci0gICAgICAgICAgICBmb3IgKEVudHJ5PENsYXNzPD8+LCBQcm92aWRlcnNNYXA+IGUgOiBjYXRlZ29yaWVzLmVudHJ5U2V0KCkpIHsKLSAgICAgICAgICAgICAgICBpZiAoZS5nZXRLZXkoKS5pc0Fzc2lnbmFibGVGcm9tKHByb3ZpZGVyLmdldENsYXNzKCkpKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJ0IHw9IGUuZ2V0VmFsdWUoKS5hZGRQcm92aWRlcihwcm92aWRlcik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIHJ0OwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIFByb3ZpZGVyc01hcC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBjbGFzcyBQcm92aWRlcnNNYXAgewotICAgICAgICAvLyAtLSBUT0RPOiBwcm92aWRlcnMgb3JkZXJpbmcgc3VwcG9ydAotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcHJvdmlkZXJzLgotICAgICAgICAgKi8KLSAgICAgICAgTWFwPENsYXNzPD8+LCBPYmplY3Q+IHByb3ZpZGVycyA9IG5ldyBIYXNoTWFwPENsYXNzPD8+LCBPYmplY3Q+KCk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEFkZHMgdGhlIHByb3ZpZGVyLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHByb3ZpZGVyCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHByb3ZpZGVyLgotICAgICAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHN1Y2Nlc3NmdWwuCi0gICAgICAgICAqLwotICAgICAgICBib29sZWFuIGFkZFByb3ZpZGVyKE9iamVjdCBwcm92aWRlcikgewotICAgICAgICAgICAgcmV0dXJuIHByb3ZpZGVycy5wdXQocHJvdmlkZXIuZ2V0Q2xhc3MoKSwgcHJvdmlkZXIpICE9IG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogR2V0cyB0aGUgcHJvdmlkZXIgY2xhc3Nlcy4KLSAgICAgICAgICogCi0gICAgICAgICAqIEByZXR1cm4gdGhlIHByb3ZpZGVyIGNsYXNzZXMuCi0gICAgICAgICAqLwotICAgICAgICBJdGVyYXRvcjxDbGFzczw/Pj4gZ2V0UHJvdmlkZXJDbGFzc2VzKCkgewotICAgICAgICAgICAgcmV0dXJuIHByb3ZpZGVycy5rZXlTZXQoKS5pdGVyYXRvcigpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gLS0gVE9ETyBvcmRlcmluZwotICAgICAgICAvKioKLSAgICAgICAgICogR2V0cyB0aGUgcHJvdmlkZXJzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIHVzZXJPcmRlcmluZwotICAgICAgICAgKiAgICAgICAgICAgIHRoZSB1c2VyIG9yZGVyaW5nLgotICAgICAgICAgKiBAcmV0dXJuIHRoZSBwcm92aWRlcnMuCi0gICAgICAgICAqLwotICAgICAgICBJdGVyYXRvcjw/PiBnZXRQcm92aWRlcnMoYm9vbGVhbiB1c2VyT3JkZXJpbmcpIHsKLSAgICAgICAgICAgIHJldHVybiBwcm92aWRlcnMudmFsdWVzKCkuaXRlcmF0b3IoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoZSBDbGFzcyBGaWx0ZXJlZEl0ZXJhdG9yLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGNsYXNzIEZpbHRlcmVkSXRlcmF0b3I8RT4gaW1wbGVtZW50cyBJdGVyYXRvcjxFPiB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBmaWx0ZXIuCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIEZpbHRlciBmaWx0ZXI7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBiYWNrZW5kLgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBJdGVyYXRvcjxFPiBiYWNrZW5kOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgbmV4dCBvYmouCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIEUgbmV4dE9iajsKLQotICAgICAgICAvKioKLSAgICAgICAgICogSW5zdGFudGlhdGVzIGEgbmV3IGZpbHRlcmVkIGl0ZXJhdG9yLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGZpbHRlcgotICAgICAgICAgKiAgICAgICAgICAgIHRoZSBmaWx0ZXIuCi0gICAgICAgICAqIEBwYXJhbSBiYWNrZW5kCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIGJhY2tlbmQuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgRmlsdGVyZWRJdGVyYXRvcihGaWx0ZXIgZmlsdGVyLCBJdGVyYXRvcjxFPiBiYWNrZW5kKSB7Ci0gICAgICAgICAgICB0aGlzLmZpbHRlciA9IGZpbHRlcjsKLSAgICAgICAgICAgIHRoaXMuYmFja2VuZCA9IGJhY2tlbmQ7Ci0gICAgICAgICAgICBmaW5kTmV4dCgpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIE5leHQuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBlLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIEUgbmV4dCgpIHsKLSAgICAgICAgICAgIGlmIChuZXh0T2JqID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgTm9TdWNoRWxlbWVudEV4Y2VwdGlvbigpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgRSB0bXAgPSBuZXh0T2JqOwotICAgICAgICAgICAgZmluZE5leHQoKTsKLSAgICAgICAgICAgIHJldHVybiB0bXA7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ2hlY2tzIGZvciBuZXh0LgotICAgICAgICAgKiAKLSAgICAgICAgICogQHJldHVybiB0cnVlLCBpZiBzdWNjZXNzZnVsLgotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIGJvb2xlYW4gaGFzTmV4dCgpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXh0T2JqICE9IG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmVtb3ZlcyB0aGUuCi0gICAgICAgICAqLwotICAgICAgICBwdWJsaWMgdm9pZCByZW1vdmUoKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBTZXRzIG5leHRPYmogdG8gYSBuZXh0IHByb3ZpZGVyIG1hdGNoaW5nIHRoZSBjcml0ZXJpb24gZ2l2ZW4gYnkgdGhlCi0gICAgICAgICAqIGZpbHRlci4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgdm9pZCBmaW5kTmV4dCgpIHsKLSAgICAgICAgICAgIG5leHRPYmogPSBudWxsOwotICAgICAgICAgICAgd2hpbGUgKGJhY2tlbmQuaGFzTmV4dCgpKSB7Ci0gICAgICAgICAgICAgICAgRSBvID0gYmFja2VuZC5uZXh0KCk7Ci0gICAgICAgICAgICAgICAgaWYgKGZpbHRlci5maWx0ZXIobykpIHsKLSAgICAgICAgICAgICAgICAgICAgbmV4dE9iaiA9IG87Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvcGFja2FnZS5odG1sIGIvYXd0L2phdmF4L2ltYWdlaW8vc3BpL3BhY2thZ2UuaHRtbApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMThjZWZmNC4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zcGkvcGFja2FnZS5odG1sCisrKyAvZGV2L251bGwKQEAgLTEsOCArMCwwIEBACi08aHRtbD4KLSAgPGJvZHk+Ci0gICAgPHA+Ci0gICAgVGhpcyBwYWNrYWdlIHByb3ZpZGVzIHNldmVyYWwgU2VydmljZSBQcm92aWRlciBJbnRlcmZhY2UgKFNQSSkgY2xhc3NlcyBmb3IgcmVhZGVycywgd3JpdGVycywgdHJhbnNjb2RlcnMgYW5kIHN0cmVhbXMgdG8gaGFuZGxlIGltYWdlcy4KLSAgICA8L3A+Ci0gIEBzaW5jZSBBbmRyb2lkIDEuMAotICA8L2JvZHk+Ci08L2h0bWw+CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vRmlsZUNhY2hlSW1hZ2VJbnB1dFN0cmVhbS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL0ZpbGVDYWNoZUltYWdlSW5wdXRTdHJlYW0uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzEwYWM2Ni4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vRmlsZUNhY2hlSW1hZ2VJbnB1dFN0cmVhbS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTM3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5zdHJlYW07Ci0KLWltcG9ydCBqYXZhLmlvLio7Ci0KLS8qKgotICogVGhlIEZpbGVDYWNoZUltYWdlSW5wdXRTdHJlYW0gY2xhc3MgaXMgYW4gaW1wbGVtZW50YXRpb24gb2YgSW1hZ2VJbnB1dFN0cmVhbQotICogd2hpY2ggcmVhZHMgZnJvbSBpdHMgSW5wdXRTdHJlYW0gYW5kIHVzZXMgYSB0ZW1wb3JhcnkgZmlsZSBhcyBhIGNhY2hlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEZpbGVDYWNoZUltYWdlSW5wdXRTdHJlYW0gZXh0ZW5kcyBJbWFnZUlucHV0U3RyZWFtSW1wbCB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBJbnB1dFN0cmVhbSBpczsKLQotICAgIC8qKgotICAgICAqIFRoZSBmaWxlLgotICAgICAqLwotICAgIHByaXZhdGUgRmlsZSBmaWxlOwotCi0gICAgLyoqCi0gICAgICogVGhlIHJhZi4KLSAgICAgKi8KLSAgICBwcml2YXRlIFJhbmRvbUFjY2Vzc0ZpbGUgcmFmOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IEZpbGVDYWNoZUltYWdlSW5wdXRTdHJlYW0gZnJvbSB0aGUgc3BlY2lmaWVkCi0gICAgICogSW5wdXRTdHJlYW0gYW5kIHVzaW5nIHRoZSBzcGVjaWZpZWQgRmlsZSBhcyBpdHMgY2FjaGUgZGlyZWN0b3J5LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHJlYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBJbnB1dFN0cmVhbSBmb3IgcmVhZGluZy4KLSAgICAgKiBAcGFyYW0gY2FjaGVEaXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBjYWNoZSBkaXJlY3Rvcnkgd2hlcmUgdGhlIGNhY2hlIGZpbGUgd2lsbCBiZSBjcmVhdGVkLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgRmlsZUNhY2hlSW1hZ2VJbnB1dFN0cmVhbShJbnB1dFN0cmVhbSBzdHJlYW0sIEZpbGUgY2FjaGVEaXIpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChzdHJlYW0gPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigic3RyZWFtID09IG51bGwhIik7Ci0gICAgICAgIH0KLSAgICAgICAgaXMgPSBzdHJlYW07Ci0KLSAgICAgICAgaWYgKGNhY2hlRGlyID09IG51bGwgfHwgY2FjaGVEaXIuaXNEaXJlY3RvcnkoKSkgewotICAgICAgICAgICAgZmlsZSA9IEZpbGUuY3JlYXRlVGVtcEZpbGUoRmlsZUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0uSUlPX1RFTVBfRklMRV9QUkVGSVgsIG51bGwsCi0gICAgICAgICAgICAgICAgICAgIGNhY2hlRGlyKTsKLSAgICAgICAgICAgIGZpbGUuZGVsZXRlT25FeGl0KCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJOb3QgYSBkaXJlY3RvcnkhIik7Ci0gICAgICAgIH0KLQotICAgICAgICByYWYgPSBuZXcgUmFuZG9tQWNjZXNzRmlsZShmaWxlLCAicnciKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IHJlYWQoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBiaXRPZmZzZXQgPSAwOwotCi0gICAgICAgIGlmIChzdHJlYW1Qb3MgPj0gcmFmLmxlbmd0aCgpKSB7Ci0gICAgICAgICAgICBpbnQgYiA9IGlzLnJlYWQoKTsKLQotICAgICAgICAgICAgaWYgKGIgPCAwKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIC0xOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByYWYuc2VlayhzdHJlYW1Qb3MrKyk7Ci0gICAgICAgICAgICByYWYud3JpdGUoYik7Ci0gICAgICAgICAgICByZXR1cm4gYjsKLSAgICAgICAgfQotCi0gICAgICAgIHJhZi5zZWVrKHN0cmVhbVBvcysrKTsKLSAgICAgICAgcmV0dXJuIHJhZi5yZWFkKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCByZWFkKGJ5dGVbXSBiLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBiaXRPZmZzZXQgPSAwOwotCi0gICAgICAgIGlmIChzdHJlYW1Qb3MgPj0gcmFmLmxlbmd0aCgpKSB7Ci0gICAgICAgICAgICBpbnQgbkJ5dGVzID0gaXMucmVhZChiLCBvZmYsIGxlbik7Ci0KLSAgICAgICAgICAgIGlmIChuQnl0ZXMgPCAwKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIC0xOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByYWYuc2VlayhzdHJlYW1Qb3MpOwotICAgICAgICAgICAgcmFmLndyaXRlKGIsIG9mZiwgbkJ5dGVzKTsKLSAgICAgICAgICAgIHN0cmVhbVBvcyArPSBuQnl0ZXM7Ci0gICAgICAgICAgICByZXR1cm4gbkJ5dGVzOwotICAgICAgICB9Ci0KLSAgICAgICAgcmFmLnNlZWsoc3RyZWFtUG9zKTsKLSAgICAgICAgaW50IG5CeXRlcyA9IHJhZi5yZWFkKGIsIG9mZiwgbGVuKTsKLSAgICAgICAgc3RyZWFtUG9zICs9IG5CeXRlczsKLSAgICAgICAgcmV0dXJuIG5CeXRlczsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NhY2hlZCgpIHsKLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDYWNoZWRGaWxlKCkgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NhY2hlZE1lbW9yeSgpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGNsb3NlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgc3VwZXIuY2xvc2UoKTsKLSAgICAgICAgcmFmLmNsb3NlKCk7Ci0gICAgICAgIGZpbGUuZGVsZXRlKCk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL0ZpbGVDYWNoZUltYWdlT3V0cHV0U3RyZWFtLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vRmlsZUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTM1YWZhYi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vRmlsZUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE5NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uc3RyZWFtOwotCi1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLkZpbGU7Ci1pbXBvcnQgamF2YS5pby5PdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5SYW5kb21BY2Nlc3NGaWxlOwotCi0vKioKLSAqIFRoZSBGaWxlQ2FjaGVJbWFnZU91dHB1dFN0cmVhbSBjbGFzcyBpcyBhbiBpbXBsZW1lbnRhdGlvbiBvZgotICogSW1hZ2VPdXRwdXRTdHJlYW0gdGhhdCB3cml0ZXMgdG8gaXRzIE91dHB1dFN0cmVhbSB1c2luZyBhIHRlbXBvcmFyeSBmaWxlIGFzIGEKLSAqIGNhY2hlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEZpbGVDYWNoZUltYWdlT3V0cHV0U3RyZWFtIGV4dGVuZHMgSW1hZ2VPdXRwdXRTdHJlYW1JbXBsIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBJSU9fVEVNUF9GSUxFX1BSRUZJWC4KLSAgICAgKi8KLSAgICBzdGF0aWMgZmluYWwgU3RyaW5nIElJT19URU1QX0ZJTEVfUFJFRklYID0gImlpb0NhY2hlIjsKLQotICAgIC8qKgotICAgICAqIFRoZSBDb25zdGFudCBNQVhfQlVGRkVSX0xFTi4KLSAgICAgKi8KLSAgICBzdGF0aWMgZmluYWwgaW50IE1BWF9CVUZGRVJfTEVOID0gMTA0ODU3NTsgLy8gMSBNQiAtIGlzIGl0IG5vdCB0b28gbXVjaD8KLQotICAgIC8qKgotICAgICAqIFRoZSBvcy4KLSAgICAgKi8KLSAgICBwcml2YXRlIE91dHB1dFN0cmVhbSBvczsKLQotICAgIC8qKgotICAgICAqIFRoZSBmaWxlLgotICAgICAqLwotICAgIHByaXZhdGUgRmlsZSBmaWxlOwotCi0gICAgLyoqCi0gICAgICogVGhlIHJhZi4KLSAgICAgKi8KLSAgICBwcml2YXRlIFJhbmRvbUFjY2Vzc0ZpbGUgcmFmOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgRmlsZUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0cmVhbQotICAgICAqICAgICAgICAgICAgdGhlIE91dHB1dFN0cmVhbSBmb3Igd3JpdGluZy4KLSAgICAgKiBAcGFyYW0gY2FjaGVEaXIKLSAgICAgKiAgICAgICAgICAgIHRoZSBjYWNoZSBkaXJlY3Rvcnkgd2hlcmUgdGhlIGNhY2hlIGZpbGUgd2lsbCBiZSBjcmVhdGVkLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgRmlsZUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0oT3V0cHV0U3RyZWFtIHN0cmVhbSwgRmlsZSBjYWNoZURpcikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKHN0cmVhbSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJzdHJlYW0gPT0gbnVsbCEiKTsKLSAgICAgICAgfQotICAgICAgICBvcyA9IHN0cmVhbTsKLQotICAgICAgICBpZiAoY2FjaGVEaXIgPT0gbnVsbCB8fCBjYWNoZURpci5pc0RpcmVjdG9yeSgpKSB7Ci0gICAgICAgICAgICBmaWxlID0gRmlsZS5jcmVhdGVUZW1wRmlsZShJSU9fVEVNUF9GSUxFX1BSRUZJWCwgbnVsbCwgY2FjaGVEaXIpOwotICAgICAgICAgICAgZmlsZS5kZWxldGVPbkV4aXQoKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIk5vdCBhIGRpcmVjdG9yeSEiKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJhZiA9IG5ldyBSYW5kb21BY2Nlc3NGaWxlKGZpbGUsICJydyIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGNsb3NlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgZmx1c2hCZWZvcmUocmFmLmxlbmd0aCgpKTsKLSAgICAgICAgc3VwZXIuY2xvc2UoKTsKLSAgICAgICAgcmFmLmNsb3NlKCk7Ci0gICAgICAgIGZpbGUuZGVsZXRlKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDYWNoZWQoKSB7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGlzQ2FjaGVkRmlsZSgpIHsKLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDYWNoZWRNZW1vcnkoKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB3cml0ZShpbnQgYikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgZmx1c2hCaXRzKCk7IC8vIFNlZSB0aGUgZmx1c2hCaXRzIG1ldGhvZCBkZXNjcmlwdGlvbgotCi0gICAgICAgIHJhZi53cml0ZShiKTsKLSAgICAgICAgc3RyZWFtUG9zKys7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgd3JpdGUoYnl0ZVtdIGIsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGZsdXNoQml0cygpOyAvLyBTZWUgdGhlIGZsdXNoQml0cyBtZXRob2QgZGVzY3JpcHRpb24KLQotICAgICAgICByYWYud3JpdGUoYiwgb2ZmLCBsZW4pOwotICAgICAgICBzdHJlYW1Qb3MgKz0gbGVuOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgcmVhZCgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGJpdE9mZnNldCA9IDA7IC8vIFNob3VsZCByZXNldAotCi0gICAgICAgIGludCByZXMgPSByYWYucmVhZCgpOwotICAgICAgICBpZiAocmVzID49IDApIHsKLSAgICAgICAgICAgIHN0cmVhbVBvcysrOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IHJlYWQoYnl0ZVtdIGIsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGJpdE9mZnNldCA9IDA7Ci0KLSAgICAgICAgaW50IG51bVJlYWQgPSByYWYucmVhZChiLCBvZmYsIGxlbik7Ci0gICAgICAgIGlmIChudW1SZWFkID4gMCkgewotICAgICAgICAgICAgc3RyZWFtUG9zICs9IG51bVJlYWQ7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbnVtUmVhZDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmbHVzaEJlZm9yZShsb25nIHBvcykgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgbG9uZyByZWFkRnJvbVBvcyA9IGZsdXNoZWRQb3M7Ci0gICAgICAgIHN1cGVyLmZsdXNoQmVmb3JlKHBvcyk7Ci0KLSAgICAgICAgbG9uZyBieXRlc1RvUmVhZCA9IHBvcyAtIHJlYWRGcm9tUG9zOwotICAgICAgICByYWYuc2VlayhyZWFkRnJvbVBvcyk7Ci0KLSAgICAgICAgaWYgKGJ5dGVzVG9SZWFkIDwgTUFYX0JVRkZFUl9MRU4pIHsKLSAgICAgICAgICAgIGJ5dGUgYnVmZmVyW10gPSBuZXcgYnl0ZVsoaW50KWJ5dGVzVG9SZWFkXTsKLSAgICAgICAgICAgIHJhZi5yZWFkRnVsbHkoYnVmZmVyKTsKLSAgICAgICAgICAgIG9zLndyaXRlKGJ1ZmZlcik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBieXRlIGJ1ZmZlcltdID0gbmV3IGJ5dGVbTUFYX0JVRkZFUl9MRU5dOwotICAgICAgICAgICAgd2hpbGUgKGJ5dGVzVG9SZWFkID4gMCkgewotICAgICAgICAgICAgICAgIGludCBjb3VudCA9IChpbnQpTWF0aC5taW4oTUFYX0JVRkZFUl9MRU4sIGJ5dGVzVG9SZWFkKTsKLSAgICAgICAgICAgICAgICByYWYucmVhZEZ1bGx5KGJ1ZmZlciwgMCwgY291bnQpOwotICAgICAgICAgICAgICAgIG9zLndyaXRlKGJ1ZmZlciwgMCwgY291bnQpOwotICAgICAgICAgICAgICAgIGJ5dGVzVG9SZWFkIC09IGNvdW50OwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgb3MuZmx1c2goKTsKLQotICAgICAgICBpZiAocG9zICE9IHN0cmVhbVBvcykgewotICAgICAgICAgICAgcmFmLnNlZWsoc3RyZWFtUG9zKTsgLy8gUmVzZXQgdGhlIHBvc2l0aW9uCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZWVrKGxvbmcgcG9zKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAocG9zIDwgZmx1c2hlZFBvcykgewotICAgICAgICAgICAgdGhyb3cgbmV3IEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJhZi5zZWVrKHBvcyk7Ci0gICAgICAgIHN0cmVhbVBvcyA9IHJhZi5nZXRGaWxlUG9pbnRlcigpOwotICAgICAgICBiaXRPZmZzZXQgPSAwOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBsb25nIGxlbmd0aCgpIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIHJldHVybiByYWYubGVuZ3RoKCk7Ci0gICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHJldHVybiAtMUw7Ci0gICAgICAgIH0KLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vRmlsZUltYWdlSW5wdXRTdHJlYW0uamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9GaWxlSW1hZ2VJbnB1dFN0cmVhbS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiOWI2MDAyLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9GaWxlSW1hZ2VJbnB1dFN0cmVhbS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTIyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5zdHJlYW07Ci0KLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uUmFuZG9tQWNjZXNzRmlsZTsKLWltcG9ydCBqYXZhLmlvLkZpbGU7Ci1pbXBvcnQgamF2YS5pby5GaWxlTm90Rm91bmRFeGNlcHRpb247Ci0KLS8qKgotICogVGhlIEZpbGVJbWFnZUlucHV0U3RyZWFtIGNsYXNzIGltcGxlbWVudHMgSW1hZ2VJbnB1dFN0cmVhbSBhbmQgb2J0YWlucyBpdHMKLSAqIGlucHV0IGRhdGEgZnJvbSBhIEZpbGUgb3IgUmFuZG9tQWNjZXNzRmlsZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBGaWxlSW1hZ2VJbnB1dFN0cmVhbSBleHRlbmRzIEltYWdlSW5wdXRTdHJlYW1JbXBsIHsKLQotICAgIC8qKgotICAgICAqIFRoZSByYWYuCi0gICAgICovCi0gICAgUmFuZG9tQWNjZXNzRmlsZSByYWY7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRmlsZUltYWdlSW5wdXRTdHJlYW0gZnJvbSB0aGUgc3BlY2lmaWVkIEZpbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIGYKLSAgICAgKiAgICAgICAgICAgIHRoZSBGaWxlIG9mIGlucHV0IGRhdGEuCi0gICAgICogQHRocm93cyBGaWxlTm90Rm91bmRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgc3BlY2lmaWVkIGZpbGUgZG9lc24ndCBleGlzdC4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgQFN1cHByZXNzV2FybmluZ3MoIHsKLSAgICAgICAgIkR1cGxpY2F0ZVRocm93cyIKLSAgICB9KQotICAgIHB1YmxpYyBGaWxlSW1hZ2VJbnB1dFN0cmVhbShGaWxlIGYpIHRocm93cyBGaWxlTm90Rm91bmRFeGNlcHRpb24sIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKGYgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiZiA9PSBudWxsISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmFmID0gbmV3IFJhbmRvbUFjY2Vzc0ZpbGUoZiwgInIiKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRmlsZUltYWdlSW5wdXRTdHJlYW0gZnJvbSB0aGUgc3BlY2lmaWVkCi0gICAgICogUmFuZG9tQWNjZXNzRmlsZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcmFmCi0gICAgICogICAgICAgICAgICB0aGUgUmFuZG9tQWNjZXNzRmlsZSBvZiBpbnB1dCBkYXRhLgotICAgICAqLwotICAgIHB1YmxpYyBGaWxlSW1hZ2VJbnB1dFN0cmVhbShSYW5kb21BY2Nlc3NGaWxlIHJhZikgewotICAgICAgICBpZiAocmFmID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInJhZiA9PSBudWxsISIpOwotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5yYWYgPSByYWY7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCByZWFkKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgYml0T2Zmc2V0ID0gMDsKLQotICAgICAgICBpbnQgcmVzID0gcmFmLnJlYWQoKTsKLSAgICAgICAgaWYgKHJlcyAhPSAtMSkgewotICAgICAgICAgICAgc3RyZWFtUG9zKys7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IHJlYWQoYnl0ZVtdIGIsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGJpdE9mZnNldCA9IDA7Ci0KLSAgICAgICAgaW50IG51bVJlYWQgPSByYWYucmVhZChiLCBvZmYsIGxlbik7Ci0gICAgICAgIGlmIChudW1SZWFkID49IDApIHsKLSAgICAgICAgICAgIHN0cmVhbVBvcyArPSBudW1SZWFkOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG51bVJlYWQ7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGxvbmcgbGVuZ3RoKCkgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIHJhZi5sZW5ndGgoKTsKLSAgICAgICAgfSBjYXRjaCAoSU9FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgcmV0dXJuIC0xTDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNlZWsobG9uZyBwb3MpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChwb3MgPCBnZXRGbHVzaGVkUG9zaXRpb24oKSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJhZi5zZWVrKHBvcyk7Ci0gICAgICAgIHN0cmVhbVBvcyA9IHJhZi5nZXRGaWxlUG9pbnRlcigpOwotICAgICAgICBiaXRPZmZzZXQgPSAwOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGNsb3NlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgc3VwZXIuY2xvc2UoKTsKLSAgICAgICAgcmFmLmNsb3NlKCk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL0ZpbGVJbWFnZU91dHB1dFN0cmVhbS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL0ZpbGVJbWFnZU91dHB1dFN0cmVhbS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyNzMwYmE2Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9GaWxlSW1hZ2VPdXRwdXRTdHJlYW0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEyOCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFJ1c3RlbSBWLiBSYWZpa292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMyAkCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvLnN0cmVhbTsKLQotaW1wb3J0IGphdmEuaW8uKjsKLQotLyoqCi0gKiBUaGUgRmlsZUltYWdlT3V0cHV0U3RyZWFtIGNsYXNzIGltcGxlbWVudHMgSW1hZ2VPdXRwdXRTdHJlYW0gYW5kIHdyaXRlcyB0aGUKLSAqIG91dHB1dCBkYXRhIHRvIGEgRmlsZSBvciBSYW5kb21BY2Nlc3NGaWxlLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIEZpbGVJbWFnZU91dHB1dFN0cmVhbSBleHRlbmRzIEltYWdlT3V0cHV0U3RyZWFtSW1wbCB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgZmlsZS4KLSAgICAgKi8KLSAgICBSYW5kb21BY2Nlc3NGaWxlIGZpbGU7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRmlsZUltYWdlT3V0cHV0U3RyZWFtIHdpdGggdGhlIHNwZWNpZmllZCBGaWxlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBmCi0gICAgICogICAgICAgICAgICB0aGUgb3V0cHV0IEZpbGUuCi0gICAgICogQHRocm93cyBGaWxlTm90Rm91bmRFeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiB0aGUgZmlsZSBub3QgZm91bmQuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHB1YmxpYyBGaWxlSW1hZ2VPdXRwdXRTdHJlYW0oRmlsZSBmKSB0aHJvd3MgRmlsZU5vdEZvdW5kRXhjZXB0aW9uLCBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRoaXMoZiAhPSBudWxsID8gbmV3IFJhbmRvbUFjY2Vzc0ZpbGUoZiwgInJ3IikgOiBudWxsKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgRmlsZUltYWdlT3V0cHV0U3RyZWFtIHdpdGggdGhlIHNwZWNpZmllZAotICAgICAqIFJhbmRvbUFjY2Vzc0ZpbGUuCi0gICAgICogCi0gICAgICogQHBhcmFtIHJhZgotICAgICAqICAgICAgICAgICAgdGhlIG91dHB1dCBSYW5kb21BY2Nlc3NGaWxlLgotICAgICAqLwotICAgIHB1YmxpYyBGaWxlSW1hZ2VPdXRwdXRTdHJlYW0oUmFuZG9tQWNjZXNzRmlsZSByYWYpIHsKLSAgICAgICAgaWYgKHJhZiA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJmaWxlIHNob3VsZCBub3QgYmUgTlVMTCIpOwotICAgICAgICB9Ci0gICAgICAgIGZpbGUgPSByYWY7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgd3JpdGUoaW50IGIpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGNoZWNrQ2xvc2VkKCk7Ci0gICAgICAgIC8vIGFjY29yZGluZyB0byB0aGUgc3BlYyBmb3IgSW1hZ2VPdXRwdXRTdHJlYW1JbXBsI2ZsdXNoQml0cygpCi0gICAgICAgIGZsdXNoQml0cygpOwotICAgICAgICBmaWxlLndyaXRlKGIpOwotICAgICAgICBzdHJlYW1Qb3MrKzsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB3cml0ZShieXRlW10gYiwgaW50IG9mZiwgaW50IGxlbikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgY2hlY2tDbG9zZWQoKTsKLSAgICAgICAgLy8gYWNjb3JkaW5nIHRvIHRoZSBzcGVjIGZvciBJbWFnZU91dHB1dFN0cmVhbUltcGwjZmx1c2hCaXRzKCkKLSAgICAgICAgZmx1c2hCaXRzKCk7Ci0gICAgICAgIGZpbGUud3JpdGUoYiwgb2ZmLCBsZW4pOwotICAgICAgICBzdHJlYW1Qb3MgKz0gbGVuOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgcmVhZCgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGNoZWNrQ2xvc2VkKCk7Ci0gICAgICAgIGludCBydCA9IGZpbGUucmVhZCgpOwotICAgICAgICBpZiAocnQgIT0gLTEpIHsKLSAgICAgICAgICAgIHN0cmVhbVBvcysrOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBydDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IHJlYWQoYnl0ZVtdIGIsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGNoZWNrQ2xvc2VkKCk7Ci0gICAgICAgIGludCBydCA9IGZpbGUucmVhZChiLCBvZmYsIGxlbik7Ci0gICAgICAgIGlmIChydCAhPSAtMSkgewotICAgICAgICAgICAgc3RyZWFtUG9zICs9IHJ0OwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBydDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgbG9uZyBsZW5ndGgoKSB7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBjaGVja0Nsb3NlZCgpOwotICAgICAgICAgICAgcmV0dXJuIGZpbGUubGVuZ3RoKCk7Ci0gICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHJldHVybiBzdXBlci5sZW5ndGgoKTsgLy8gLTFMCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZWVrKGxvbmcgcG9zKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBjaGVja0Nsb3NlZCgpIGlzIHBlcmZvcm1lZCBpbiBzdXBlci5zZWVrKCkKLSAgICAgICAgc3VwZXIuc2Vlayhwb3MpOwotICAgICAgICBmaWxlLnNlZWsocG9zKTsKLSAgICAgICAgc3RyZWFtUG9zID0gZmlsZS5nZXRGaWxlUG9pbnRlcigpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGNsb3NlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgc3VwZXIuY2xvc2UoKTsKLSAgICAgICAgZmlsZS5jbG9zZSgpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9JSU9CeXRlQnVmZmVyLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vSUlPQnl0ZUJ1ZmZlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4NjdkODA4Li4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9JSU9CeXRlQnVmZmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMjQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBTZXJnZXkgSS4gU2FsaXNoZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4yICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uc3RyZWFtOwotCi0vLyAKLS8vIEBhdXRob3IgU2VyZ2V5IEkuIFNhbGlzaGV2Ci0vLyBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0vLwotCi0vKioKLSAqIFRoZSBJSU9CeXRlQnVmZmVyIGNsYXNzIHJlcHJlc2VudHMgYSBieXRlIGFycmF5IHdpdGggb2Zmc2V0IGFuZCBsZW5ndGggdGhhdAotICogaXMgdXNlZCBieSBJbWFnZUlucHV0U3RyZWFtIGZvciBvYnRhaW5pbmcgYSBzZXF1ZW5jZSBvZiBieXRlcy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBJSU9CeXRlQnVmZmVyIHsKLQotICAgIC8qKgotICAgICAqIFRoZSBkYXRhLgotICAgICAqLwotICAgIHByaXZhdGUgYnl0ZVtdIGRhdGE7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgb2Zmc2V0LgotICAgICAqLwotICAgIHByaXZhdGUgaW50IG9mZnNldDsKLQotICAgIC8qKgotICAgICAqIFRoZSBsZW5ndGguCi0gICAgICovCi0gICAgcHJpdmF0ZSBpbnQgbGVuZ3RoOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IElJT0J5dGVCdWZmZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBieXRlIGFycmF5LgotICAgICAqIEBwYXJhbSBvZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIGFycmF5LgotICAgICAqIEBwYXJhbSBsZW5ndGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggb2YgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIElJT0J5dGVCdWZmZXIoYnl0ZVtdIGRhdGEsIGludCBvZmZzZXQsIGludCBsZW5ndGgpIHsKLSAgICAgICAgdGhpcy5kYXRhID0gZGF0YTsKLSAgICAgICAgdGhpcy5vZmZzZXQgPSBvZmZzZXQ7Ci0gICAgICAgIHRoaXMubGVuZ3RoID0gbGVuZ3RoOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGJ5dGUgYXJyYXkgb2YgdGhpcyBJSU9CeXRlQnVmZmVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJ5dGUgYXJyYXkuCi0gICAgICovCi0gICAgcHVibGljIGJ5dGVbXSBnZXREYXRhKCkgewotICAgICAgICByZXR1cm4gZGF0YTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBsZW5ndGggaW4gdGhlIGFycmF5IHdoaWNoIHdpbGwgYmUgdXNlZC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBsZW5ndGggb2YgdGhlIGRhdGEuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRMZW5ndGgoKSB7Ci0gICAgICAgIHJldHVybiBsZW5ndGg7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyB0aGUgb2Zmc2V0IG9mIHRoaXMgSUlPQnl0ZUJ1ZmZlci4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBvZmZzZXQgb2YgdGhpcyBJSU9CeXRlQnVmZmVyLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0T2Zmc2V0KCkgewotICAgICAgICByZXR1cm4gb2Zmc2V0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIG5ldyBkYXRhIGFycmF5IHRvIHRoaXMgSUlPQnl0ZUJ1ZmZlciBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIGRhdGEKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgZGF0YSBhcnJheS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXREYXRhKGJ5dGVbXSBkYXRhKSB7Ci0gICAgICAgIHRoaXMuZGF0YSA9IGRhdGE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgbGVuZ3RoIG9mIGRhdGEgd2hpY2ggd2lsbCBiZSB1c2VkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsZW5ndGgKLSAgICAgKiAgICAgICAgICAgIHRoZSBuZXcgbGVuZ3RoLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldExlbmd0aChpbnQgbGVuZ3RoKSB7Ci0gICAgICAgIHRoaXMubGVuZ3RoID0gbGVuZ3RoOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIG9mZnNldCBpbiB0aGUgZGF0YSBhcnJheSBvZiB0aGlzIElJT0J5dGVCdWZmZXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIG9mZnNldAotICAgICAqICAgICAgICAgICAgdGhlIG5ldyBvZmZzZXQuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0T2Zmc2V0KGludCBvZmZzZXQpIHsKLSAgICAgICAgdGhpcy5vZmZzZXQgPSBvZmZzZXQ7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL0ltYWdlSW5wdXRTdHJlYW0uamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9JbWFnZUlucHV0U3RyZWFtLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDNkZWM1ZDIuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL0ltYWdlSW5wdXRTdHJlYW0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDUwMiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFJ1c3RlbSBWLiBSYWZpa292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvLnN0cmVhbTsKLQotaW1wb3J0IGphdmEuaW8uRGF0YUlucHV0OwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5uaW8uQnl0ZU9yZGVyOwotCi0vKioKLSAqIFRoZSBJbWFnZUlucHV0U3RyZWFtIHJlcHJlc2VudHMgaW5wdXQgc3RyZWFtIGludGVyZmFjZSB0aGF0IGlzIHVzZWQgYnkKLSAqIEltYWdlUmVhZGVycy4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgSW1hZ2VJbnB1dFN0cmVhbSBleHRlbmRzIERhdGFJbnB1dCB7Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBzcGVjaWZpZWQgYnl0ZSBvcmRlciBmb3IgcmVhZGluZyBvZiBkYXRhIHZhbHVlcyBmcm9tIHRoaXMKLSAgICAgKiBzdHJlYW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIGJ5dGVPcmRlcgotICAgICAqICAgICAgICAgICAgdGhlIGJ5dGUgb3JkZXIuCi0gICAgICovCi0gICAgdm9pZCBzZXRCeXRlT3JkZXIoQnl0ZU9yZGVyIGJ5dGVPcmRlcik7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBieXRlIG9yZGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJ5dGUgb3JkZXIuCi0gICAgICovCi0gICAgQnl0ZU9yZGVyIGdldEJ5dGVPcmRlcigpOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgYSBieXRlIGZyb20gdGhlIHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBieXRlIG9mIHRoZSBzdHJlYW0sIG9yIC0xIGZvciBFT0YgaW5kaWNhdGluZy4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgaW50IHJlYWQoKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyBudW1iZXIgb2YgYnl0ZXMgd2hpY2ggaXMgZXF1YWwgdG8gdGhlIHNwZWNpZmllZCBhcnJheSdzIGxlbmd0aCBhbmQKLSAgICAgKiBzdG9yZXMgYSByZXN1bHQgdG8gdGhpcyBhcnJheS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJ5dGUgYXJyYXkuCi0gICAgICogQHJldHVybiB0aGUgbnVtYmVyIG9mIHJlYWQgYnl0ZXMsIG9yIC0xIGluZGljYXRlZCBFT0YuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGludCByZWFkKGJ5dGVbXSBiKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyB0aGUgbnVtYmVyIG9mIGJ5dGVzIHNwZWNpZmllZCBieSBsZW4gcGFyYW1ldGVyIGZyb20gdGhlIHN0cmVhbSBhbmQKLSAgICAgKiBzdG9yZXMgYSByZXN1bHQgdG8gdGhlIHNwZWNpZmllZCBhcnJheSB3aXRoIHRoZSBzcGVjaWZpZWQgb2Zmc2V0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgYnl0ZSBhcnJheS4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBsZW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgYnl0ZXMgdG8gYmUgcmVhZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBudW1iZXIgb2YgcmVhZCBieXRlcywgb3IgLTEgaW5kaWNhdGVkIEVPRi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgaW50IHJlYWQoYnl0ZVtdIGIsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFJlYWRzIHRoZSBudW1iZXIgb2YgYnl0ZXMgc3BlY2lmaWVkIGJ5IGxlbiBwYXJhbWV0ZXIgZnJvbSB0aGUgc3RyZWFtLCBhbmQKLSAgICAgKiBtb2RpZmllcyB0aGUgc3BlY2lmaWVkIElJT0J5dGVCdWZmZXIgd2l0aCB0aGUgYnl0ZSBhcnJheSwgb2Zmc2V0LCBhbmQKLSAgICAgKiBsZW5ndGguCi0gICAgICogCi0gICAgICogQHBhcmFtIGJ1ZgotICAgICAqICAgICAgICAgICAgdGhlIElJT0J5dGVCdWZmZXIuCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBieXRlcyB0byBiZSByZWFkLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHJlYWRCeXRlcyhJSU9CeXRlQnVmZmVyIGJ1ZiwgaW50IGxlbikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgYSBieXRlIGZyb20gdGhlIHN0cmVhbSBhbmQgcmV0dXJucyBhIGJvb2xlYW4gdHJ1ZSB2YWx1ZSBpZiBpdCBpcwotICAgICAqIG5vbiB6ZXJvLCBmYWxzZSBpZiBpdCBpcyB6ZXJvLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJvb2xlYW4gdmFsdWUgZm9yIHJlYWQgYnl0ZS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgYm9vbGVhbiByZWFkQm9vbGVhbigpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFJlYWRzIGEgYnl0ZSBmcm9tIHRoZSBzdHJlYW0gYW5kIHJldHVybnMgaXRzIHZhbHVlIGFzIHNpZ25lZCBieXRlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNpZ25lZCBieXRlIHZhbHVlIGZvciByZWFkIGJ5dGUuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGJ5dGUgcmVhZEJ5dGUoKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyBhIGJ5dGUgZnJvbSB0aGUgc3RyZWFtIGFuZCByZXR1cm5zIGl0cyB2YWx1ZSBhcyBhbiBpbnRlZ2VyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHVuc2lnbmVkIGJ5dGUgdmFsdWUgZm9yIHJlYWQgYnl0ZSBhcyBhbiBpbnRlZ2VyLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBpbnQgcmVhZFVuc2lnbmVkQnl0ZSgpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFJlYWRzIDIgYnl0ZXMgZnJvbSB0aGUgc3RyZWFtLCBhbmQgcmV0dXJucyB0aGUgcmVzdWx0IGFzIGEgc2hvcnQuCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgc2lnbmVkIHNob3J0IHZhbHVlIGZyb20gdGhlIHN0cmVhbS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgc2hvcnQgcmVhZFNob3J0KCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgMiBieXRlcyBmcm9tIHRoZSBzdHJlYW0gYW5kIHJldHVybnMgaXRzIHZhbHVlIGFzIGFuIHVuc2lnbmVkIHNob3J0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSB1bnNpZ25lZCBzaG9ydCB2YWx1ZSBjb2RlZCBpbiBhbiBpbnRlZ2VyLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBpbnQgcmVhZFVuc2lnbmVkU2hvcnQoKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyAyIGJ5dGVzIGZyb20gdGhlIHN0cmVhbSBhbmQgcmV0dXJucyB0aGVpciB1bnNpZ25lZCBjaGFyIHZhbHVlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHVuc2lnbmVkIGNoYXIgdmFsdWUuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGNoYXIgcmVhZENoYXIoKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyA0IGJ5dGVzIGZyb20gdGhlIHN0cmVhbSwgYW5kIHJldHVybnMgdGhlIHJlc3VsdCBhcyBhbiBpbnRlZ2VyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHNpZ25lZCBpbnRlZ2VyIHZhbHVlIGZyb20gdGhlIHN0cmVhbS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgaW50IHJlYWRJbnQoKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyA0IGJ5dGVzIGZyb20gdGhlIHN0cmVhbSBhbmQgcmV0dXJucyBpdHMgdmFsdWUgYXMgbG9uZy4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB1bnNpZ25lZCBpbnRlZ2VyIHZhbHVlIGFzIGxvbmcuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGxvbmcgcmVhZFVuc2lnbmVkSW50KCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgOCBieXRlcyBmcm9tIHRoZSBzdHJlYW0sIGFuZCByZXR1cm5zIHRoZSByZXN1bHQgYXMgYSBsb25nLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGxvbmcgdmFsdWUgZnJvbSB0aGUgc3RyZWFtLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBsb25nIHJlYWRMb25nKCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgNCBieXRlcyBmcm9tIHRoZSBzdHJlYW0sIGFuZCByZXR1cm5zIHRoZSByZXN1bHQgYXMgYSBmbG9hdC4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBmbG9hdCB2YWx1ZSBmcm9tIHRoZSBzdHJlYW0uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGZsb2F0IHJlYWRGbG9hdCgpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFJlYWRzIDggYnl0ZXMgZnJvbSB0aGUgc3RyZWFtLCBhbmQgcmV0dXJucyB0aGUgcmVzdWx0IGFzIGEgZG91YmxlLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGRvdWJsZSB2YWx1ZSBmcm9tIHRoZSBzdHJlYW0uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGRvdWJsZSByZWFkRG91YmxlKCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgYSBsaW5lIGZyb20gdGhlIHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJpbmcgY29udGFpbmVkIHRoZSBsaW5lIGZyb20gdGhlIHN0cmVhbS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgU3RyaW5nIHJlYWRMaW5lKCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgYnl0ZXMgZnJvbSB0aGUgc3RyZWFtIGluIGEgc3RyaW5nIHRoYXQgaGFzIGJlZW4gZW5jb2RlZCBpbiBhCi0gICAgICogbW9kaWZpZWQgVVRGLTggZm9ybWF0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIHN0cmluZyByZWFkIGZyb20gc3RyZWFtIGFuZCBtb2RpZmllZCBVVEYtOCBmb3JtYXQuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIFN0cmluZyByZWFkVVRGKCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgdGhlIHNwZWNpZmllZCBudW1iZXIgb2YgYnl0ZXMgZnJvbSB0aGUgc3RyZWFtLCBhbmQgc3RvcmVzIHRoZQotICAgICAqIHJlc3VsdCBpbnRvIHRoZSBzcGVjaWZpZWQgYXJyYXkgc3RhcnRpbmcgYXQgdGhlIHNwZWNpZmllZCBpbmRleCBvZmZzZXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBieXRlIGFycmF5LgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQuCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBieXRlcyB0byBiZSByZWFkLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHJlYWRGdWxseShieXRlW10gYiwgaW50IG9mZiwgaW50IGxlbikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgbnVtYmVyIG9mIGJ5dGVzIGZyb20gdGhlIHN0cmVhbSB3aGljaCBpcyBlcXVhbCB0byB0aGUgc3BlY2lmaWVkCi0gICAgICogYXJyYXkncyBsZW5ndGgsIGFuZCBzdG9yZXMgdGhlbSBpbnRvIHRoaXMgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBieXRlIGFycmF5LgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHJlYWRGdWxseShieXRlW10gYikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZHMgdGhlIHNwZWNpZmllZCBudW1iZXIgb2Ygc2hvcnRzIGZyb20gdGhlIHN0cmVhbSwgYW5kIHN0b3JlcyB0aGUKLSAgICAgKiByZXN1bHQgaW50byB0aGUgc3BlY2lmaWVkIGFycmF5IHN0YXJ0aW5nIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXggb2Zmc2V0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzCi0gICAgICogICAgICAgICAgICB0aGUgc2hvcnQgYXJyYXkuCi0gICAgICogQHBhcmFtIG9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldC4KLSAgICAgKiBAcGFyYW0gbGVuCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIHNob3J0cyB0byBiZSByZWFkLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHJlYWRGdWxseShzaG9ydFtdIHMsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFJlYWRzIHRoZSBzcGVjaWZpZWQgbnVtYmVyIG9mIGNoYXJzIGZyb20gdGhlIHN0cmVhbSwgYW5kIHN0b3JlcyB0aGUKLSAgICAgKiByZXN1bHQgaW50byB0aGUgc3BlY2lmaWVkIGFycmF5IHN0YXJ0aW5nIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXggb2Zmc2V0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgY2hhciBhcnJheS4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBsZW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgY2hhcnMgdG8gYmUgcmVhZC4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCByZWFkRnVsbHkoY2hhcltdIGMsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFJlYWRzIHRoZSBzcGVjaWZpZWQgbnVtYmVyIG9mIGludGVnZXIgZnJvbSB0aGUgc3RyZWFtLCBhbmQgc3RvcmVzIHRoZQotICAgICAqIHJlc3VsdCBpbnRvIHRoZSBzcGVjaWZpZWQgYXJyYXkgc3RhcnRpbmcgYXQgdGhlIHNwZWNpZmllZCBpbmRleCBvZmZzZXQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGkKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnRlZ2VyIGFycmF5LgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQuCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBpbnRlZ2VyIHRvIGJlIHJlYWQuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHZvaWQgcmVhZEZ1bGx5KGludFtdIGksIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFJlYWRzIHRoZSBzcGVjaWZpZWQgbnVtYmVyIG9mIGxvbmdzIGZyb20gdGhlIHN0cmVhbSwgYW5kIHN0b3JlcyB0aGUKLSAgICAgKiByZXN1bHQgaW50byB0aGUgc3BlY2lmaWVkIGFycmF5IHN0YXJ0aW5nIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXggb2Zmc2V0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgbG9uZyBhcnJheS4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0LgotICAgICAqIEBwYXJhbSBsZW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgbG9uZ3MgdG8gYmUgcmVhZC4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCByZWFkRnVsbHkobG9uZ1tdIGwsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFJlYWRzIHRoZSBzcGVjaWZpZWQgbnVtYmVyIG9mIGZsb2F0cyBmcm9tIHRoZSBzdHJlYW0sIGFuZCBzdG9yZXMgdGhlCi0gICAgICogcmVzdWx0IGludG8gdGhlIHNwZWNpZmllZCBhcnJheSBzdGFydGluZyBhdCB0aGUgc3BlY2lmaWVkIGluZGV4IG9mZnNldC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZgotICAgICAqICAgICAgICAgICAgdGhlIGZsb2F0IGFycmF5LgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQuCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBmbG9hdHMgdG8gYmUgcmVhZC4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCByZWFkRnVsbHkoZmxvYXRbXSBmLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyB0aGUgc3BlY2lmaWVkIG51bWJlciBvZiBkb3VibGVzIGZyb20gdGhlIHN0cmVhbSwgYW5kIHN0b3JlcyB0aGUKLSAgICAgKiByZXN1bHQgaW50byB0aGUgc3BlY2lmaWVkIGFycmF5IHN0YXJ0aW5nIGF0IHRoZSBzcGVjaWZpZWQgaW5kZXggb2Zmc2V0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBkCi0gICAgICogICAgICAgICAgICB0aGUgZG91YmxlIGFycmF5LgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQuCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBkb3VibGVzIHRvIGJlIHJlYWQuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHZvaWQgcmVhZEZ1bGx5KGRvdWJsZVtdIGQsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIHN0cmVhbSBwb3NpdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSBzdHJlYW0gcG9zaXRpb24uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGxvbmcgZ2V0U3RyZWFtUG9zaXRpb24oKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHRoZSBiaXQgb2Zmc2V0LgotICAgICAqIAotICAgICAqIEByZXR1cm4gdGhlIGJpdCBvZmZzZXQuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGludCBnZXRCaXRPZmZzZXQoKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIHRoZSBiaXQgb2Zmc2V0IHRvIGFuIGludGVnZXIgYmV0d2VlbiAwIGFuZCA3LgotICAgICAqIAotICAgICAqIEBwYXJhbSBiaXRPZmZzZXQKLSAgICAgKiAgICAgICAgICAgIHRoZSBiaXQgb2Zmc2V0LgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHNldEJpdE9mZnNldChpbnQgYml0T2Zmc2V0KSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBSZWFkcyBhIGJpdCBmcm9tIHRoZSBzdHJlYW0gYW5kIHJldHVybnMgdGhlIHZhbHVlIDAgb3IgMS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRoZSB2YWx1ZSBvZiBzaW5nbGUgYml0OiAwIG9yIDEuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGludCByZWFkQml0KCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmVhZCB0aGUgc3BlY2lmaWVkIG51bWJlciBvZiBiaXRzIGFuZCByZXR1cm5zIHRoZWlyIHZhbHVlcyBhcyBsb25nLgotICAgICAqIAotICAgICAqIEBwYXJhbSBudW1CaXRzCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGJpdHMgdG8gYmUgcmVhZC4KLSAgICAgKiBAcmV0dXJuIHRoZSBiaXQgc3RyaW5nIGFzIGEgbG9uZy4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgbG9uZyByZWFkQml0cyhpbnQgbnVtQml0cykgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbGVuZ3RoIG9mIHRoZSBzdHJlYW0uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgbGVuZ3RoIG9mIHRoZSBzdHJlYW0sIG9yIC0xIGlmIHVua25vd24uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGxvbmcgbGVuZ3RoKCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogU2tpcHMgdGhlIHNwZWNpZmllZCBudW1iZXIgb2YgYnl0ZXMgYnkgbW92aW5nIHN0cmVhbSBwb3NpdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbgotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBieXRlcy4KLSAgICAgKiBAcmV0dXJuIHRoZSBhY3R1YWwgc2tpcHBlZCBudW1iZXIgb2YgYnl0ZXMuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIGludCBza2lwQnl0ZXMoaW50IG4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFNraXBzIHRoZSBzcGVjaWZpZWQgbnVtYmVyIG9mIGJ5dGVzIGJ5IG1vdmluZyBzdHJlYW0gcG9zaXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIG4KLSAgICAgKiAgICAgICAgICAgIHRoZSBudW1iZXIgb2YgYnl0ZXMuCi0gICAgICogQHJldHVybiB0aGUgYWN0dWFsIHNraXBwZWQgbnVtYmVyIG9mIGJ5dGVzLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICBsb25nIHNraXBCeXRlcyhsb25nIG4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlIGN1cnJlbnQgc3RyZWFtIHBvc2l0aW9uIHRvIHRoZSBzcGVjaWZpZWQgbG9jYXRpb24uCi0gICAgICogCi0gICAgICogQHBhcmFtIHBvcwotICAgICAqICAgICAgICAgICAgYSBmaWxlIHBvaW50ZXIgcG9zaXRpb24uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHZvaWQgc2Vlayhsb25nIHBvcykgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogTWFya3MgYSBwb3NpdGlvbiBpbiB0aGUgc3RyZWFtIHRvIGJlIHJldHVybmVkIHRvIGJ5IGEgc3Vic2VxdWVudCBjYWxsIHRvCi0gICAgICogcmVzZXQuCi0gICAgICovCi0gICAgdm9pZCBtYXJrKCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBmaWxlIHBvaW50ZXIgdG8gaXRzIHByZXZpb3VzIHBvc2l0aW9uLgotICAgICAqIAotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHJlc2V0KCkgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogRmx1c2hlcyB0aGUgaW5pdGlhbCBwb3NpdGlvbiBpbiB0aGlzIHN0cmVhbSBwcmlvciB0byB0aGUgc3BlY2lmaWVkIHN0cmVhbQotICAgICAqIHBvc2l0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwb3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb3NpdGlvbi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCBmbHVzaEJlZm9yZShsb25nIHBvcykgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogRmx1c2hlcyB0aGUgaW5pdGlhbCBwb3NpdGlvbiBpbiB0aGlzIHN0cmVhbSBwcmlvciB0byB0aGUgY3VycmVudCBzdHJlYW0KLSAgICAgKiBwb3NpdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCBmbHVzaCgpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIEdldHMgdGhlIGZsdXNoZWQgcG9zaXRpb24uCi0gICAgICogCi0gICAgICogQHJldHVybiB0aGUgZmx1c2hlZCBwb3NpdGlvbi4KLSAgICAgKi8KLSAgICBsb25nIGdldEZsdXNoZWRQb3NpdGlvbigpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoaXMgSW1hZ2VJbnB1dFN0cmVhbSBjYWNoZXMgZGF0YSBpbiBvcmRlciB0byBhbGxvdwotICAgICAqIHNlZWtpbmcgYmFja3dhcmRzLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSwgaWYgdGhpcyBJbWFnZUlucHV0U3RyZWFtIGNhY2hlcyBkYXRhIGluIG9yZGVyIHRvIGFsbG93Ci0gICAgICogICAgICAgICBzZWVraW5nIGJhY2t3YXJkcywgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqLwotICAgIGJvb2xlYW4gaXNDYWNoZWQoKTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiB0aGlzIEltYWdlSW5wdXRTdHJlYW0gY2FjaGVzIGRhdGEgaW4gb3JkZXIgdG8gYWxsb3cKLSAgICAgKiBzZWVraW5nIGJhY2t3YXJkcywgYW5kIGtlZXBzIGl0IGluIG1lbW9yeS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgSW1hZ2VJbnB1dFN0cmVhbSBjYWNoZXMgZGF0YSBpbiBvcmRlciB0byBhbGxvdwotICAgICAqICAgICAgICAgc2Vla2luZyBiYWNrd2FyZHMsIGFuZCBrZWVwcyBpdCBpbiBtZW1vcnkuCi0gICAgICovCi0gICAgYm9vbGVhbiBpc0NhY2hlZE1lbW9yeSgpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoaXMgSW1hZ2VJbnB1dFN0cmVhbSBjYWNoZXMgZGF0YSBpbiBvcmRlciB0byBhbGxvdwotICAgICAqIHNlZWtpbmcgYmFja3dhcmRzLCBhbmQga2VlcHMgaXQgaW4gYSB0ZW1wb3JhcnkgZmlsZS4KLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHRydWUsIGlmIHRoaXMgSW1hZ2VJbnB1dFN0cmVhbSBjYWNoZXMgZGF0YSBpbiBvcmRlciB0byBhbGxvdwotICAgICAqICAgICAgICAgc2Vla2luZyBiYWNrd2FyZHMsIGFuZCBrZWVwcyBpdCBpbiBhIHRlbXBvcmFyeSBmaWxlLgotICAgICAqLwotICAgIGJvb2xlYW4gaXNDYWNoZWRGaWxlKCk7Ci0KLSAgICAvKioKLSAgICAgKiBDbG9zZXMgdGhpcyBzdHJlYW0uCi0gICAgICogCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHZvaWQgY2xvc2UoKSB0aHJvd3MgSU9FeGNlcHRpb247Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vSW1hZ2VJbnB1dFN0cmVhbUltcGwuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9JbWFnZUlucHV0U3RyZWFtSW1wbC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkNzlkYTQxLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9JbWFnZUlucHV0U3RyZWFtSW1wbC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDE4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uc3RyZWFtOwotCi1pbXBvcnQgamF2YS5pby5FT0ZFeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLm5pby5CeXRlT3JkZXI7Ci0KLS8qKgotICogVGhlIEltYWdlSW5wdXRTdHJlYW1JbXBsIGFic3RyYWN0IGNsYXNzIGltcGxlbWVudHMgdGhlIEltYWdlSW5wdXRTdHJlYW0KLSAqIGludGVyZmFjZS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBJbWFnZUlucHV0U3RyZWFtSW1wbCBpbXBsZW1lbnRzIEltYWdlSW5wdXRTdHJlYW0gewotCi0gICAgLyoqCi0gICAgICogVGhlIGJ5dGUgb3JkZXIuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIEJ5dGVPcmRlciBieXRlT3JkZXIgPSBCeXRlT3JkZXIuQklHX0VORElBTjsKLQotICAgIC8qKgotICAgICAqIFRoZSBzdHJlYW0gcG9zaXRpb24uCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGxvbmcgc3RyZWFtUG9zID0gMDsKLQotICAgIC8qKgotICAgICAqIFRoZSBmbHVzaGVkIHBvc2l0aW9uLgotICAgICAqLwotICAgIHByb3RlY3RlZCBsb25nIGZsdXNoZWRQb3MgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIGJpdCBvZmZzZXQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBiaXRPZmZzZXQgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhlIGNsb3NlZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGJvb2xlYW4gY2xvc2VkID0gZmFsc2U7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgcG9zaXRpb24gc3RhY2suCi0gICAgICovCi0gICAgcHJpdmF0ZSBmaW5hbCBQb3NpdGlvblN0YWNrIHBvc1N0YWNrID0gbmV3IFBvc2l0aW9uU3RhY2soKTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBJbWFnZUlucHV0U3RyZWFtSW1wbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgSW1hZ2VJbnB1dFN0cmVhbUltcGwoKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2sgaWYgdGhlIHN0cmVhbSBpcyBjbG9zZWQgYW5kIGlmIHRydWUsIHRocm93cyBhbiBJT0V4Y2VwdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgdGhlIHN0cmVhbSBpcyBjbG9zZWQuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGZpbmFsIHZvaWQgY2hlY2tDbG9zZWQoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAoY2xvc2VkKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSU9FeGNlcHRpb24oInN0cmVhbSBpcyBjbG9zZWQiKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldEJ5dGVPcmRlcihCeXRlT3JkZXIgYnl0ZU9yZGVyKSB7Ci0gICAgICAgIHRoaXMuYnl0ZU9yZGVyID0gYnl0ZU9yZGVyOwotICAgIH0KLQotICAgIHB1YmxpYyBCeXRlT3JkZXIgZ2V0Qnl0ZU9yZGVyKCkgewotICAgICAgICByZXR1cm4gYnl0ZU9yZGVyOwotICAgIH0KLQotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgcmVhZCgpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIHB1YmxpYyBpbnQgcmVhZChieXRlW10gYikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIHJlYWQoYiwgMCwgYi5sZW5ndGgpOwotICAgIH0KLQotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgcmVhZChieXRlW10gYiwgaW50IG9mZiwgaW50IGxlbikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgcHVibGljIHZvaWQgcmVhZEJ5dGVzKElJT0J5dGVCdWZmZXIgYnVmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAoYnVmID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbigiYnVmZmVyIGlzIE5VTEwiKTsKLSAgICAgICAgfQotCi0gICAgICAgIGJ5dGVbXSBiID0gbmV3IGJ5dGVbbGVuXTsKLSAgICAgICAgbGVuID0gcmVhZChiLCAwLCBiLmxlbmd0aCk7Ci0KLSAgICAgICAgYnVmLnNldERhdGEoYik7Ci0gICAgICAgIGJ1Zi5zZXRPZmZzZXQoMCk7Ci0gICAgICAgIGJ1Zi5zZXRMZW5ndGgobGVuKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiByZWFkQm9vbGVhbigpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGludCBiID0gcmVhZCgpOwotICAgICAgICBpZiAoYiA8IDApIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBFT0ZFeGNlcHRpb24oIkVPRiByZWFjaGVkIik7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGIgIT0gMDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYnl0ZSByZWFkQnl0ZSgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGludCBiID0gcmVhZCgpOwotICAgICAgICBpZiAoYiA8IDApIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBFT0ZFeGNlcHRpb24oIkVPRiByZWFjaGVkIik7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIChieXRlKWI7Ci0gICAgfQotCi0gICAgcHVibGljIGludCByZWFkVW5zaWduZWRCeXRlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaW50IGIgPSByZWFkKCk7Ci0gICAgICAgIGlmIChiIDwgMCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IEVPRkV4Y2VwdGlvbigiRU9GIHJlYWNoZWQiKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gYjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc2hvcnQgcmVhZFNob3J0KCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaW50IGIxID0gcmVhZCgpOwotICAgICAgICBpbnQgYjIgPSByZWFkKCk7Ci0KLSAgICAgICAgaWYgKGIxIDwgMCB8fCBiMiA8IDApIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBFT0ZFeGNlcHRpb24oIkVPRiByZWFjaGVkIik7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gYnl0ZU9yZGVyID09IEJ5dGVPcmRlci5CSUdfRU5ESUFOID8gKHNob3J0KSgoYjEgPDwgOCkgfCAoYjIgJiAweGZmKSkKLSAgICAgICAgICAgICAgICA6IChzaG9ydCkoKGIyIDw8IDgpIHwgKGIxICYgMHhmZikpOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgcmVhZFVuc2lnbmVkU2hvcnQoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgY2hhciByZWFkQ2hhcigpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgcmVhZEludCgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyBsb25nIHJlYWRVbnNpZ25lZEludCgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyBsb25nIHJlYWRMb25nKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgLy8gLS0gVE9ETyBpbXBsZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgcHVibGljIGZsb2F0IHJlYWRGbG9hdCgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyBkb3VibGUgcmVhZERvdWJsZSgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmcgcmVhZExpbmUoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIHJlYWRVVEYoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZWFkRnVsbHkoYnl0ZVtdIGIsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlYWRGdWxseShieXRlW10gYikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcmVhZEZ1bGx5KGIsIDAsIGIubGVuZ3RoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZWFkRnVsbHkoc2hvcnRbXSBzLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZWFkRnVsbHkoY2hhcltdIGMsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlYWRGdWxseShpbnRbXSBpLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZWFkRnVsbHkobG9uZ1tdIGwsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlYWRGdWxseShmbG9hdFtdIGYsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlYWRGdWxseShkb3VibGVbXSBkLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgbG9uZyBnZXRTdHJlYW1Qb3NpdGlvbigpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGNoZWNrQ2xvc2VkKCk7Ci0gICAgICAgIHJldHVybiBzdHJlYW1Qb3M7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRCaXRPZmZzZXQoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBjaGVja0Nsb3NlZCgpOwotICAgICAgICByZXR1cm4gYml0T2Zmc2V0OwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldEJpdE9mZnNldChpbnQgYml0T2Zmc2V0KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBjaGVja0Nsb3NlZCgpOwotICAgICAgICB0aGlzLmJpdE9mZnNldCA9IGJpdE9mZnNldDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IHJlYWRCaXQoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgbG9uZyByZWFkQml0cyhpbnQgbnVtQml0cykgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgLy8gLS0gVE9ETyBpbXBsZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgcHVibGljIGxvbmcgbGVuZ3RoKCkgewotICAgICAgICByZXR1cm4gLTFMOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgc2tpcEJ5dGVzKGludCBuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgbG9uZyBza2lwQnl0ZXMobG9uZyBuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZWVrKGxvbmcgcG9zKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBjaGVja0Nsb3NlZCgpOwotICAgICAgICBpZiAocG9zIDwgZ2V0Rmx1c2hlZFBvc2l0aW9uKCkpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInRyeWluZyB0byBzZWVrIGJlZm9yZSBmbHVzaGVkIHBvcyIpOwotICAgICAgICB9Ci0gICAgICAgIGJpdE9mZnNldCA9IDA7Ci0gICAgICAgIHN0cmVhbVBvcyA9IHBvczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBtYXJrKCkgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgcG9zU3RhY2sucHVzaChnZXRTdHJlYW1Qb3NpdGlvbigpKTsKLSAgICAgICAgfSBjYXRjaCAoSU9FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgZS5wcmludFN0YWNrVHJhY2UoKTsKLSAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJTdHJlYW0gbWFya2luZyBlcnJvciIpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgcmVzZXQoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGJpdCBwb3MKLSAgICAgICAgaWYgKCFwb3NTdGFjay5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgIGxvbmcgcCA9IHBvc1N0YWNrLnBvcCgpOwotICAgICAgICAgICAgaWYgKHAgPCBmbHVzaGVkUG9zKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElPRXhjZXB0aW9uKCJtYXJrZWQgcG9zaXRpb24gbGllcyBpbiB0aGUgZmx1c2hlZCBwb3J0aW9uIG9mIHRoZSBzdHJlYW0iKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHNlZWsocCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBmbHVzaEJlZm9yZShsb25nIHBvcykgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKHBvcyA+IGdldFN0cmVhbVBvc2l0aW9uKCkpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKCJUcnlpbmcgdG8gZmx1c2ggb3V0c2lkZSBvZiBjdXJyZW50IHBvc2l0aW9uIik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHBvcyA8IGZsdXNoZWRQb3MpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKCJUcnlpbmcgdG8gZmx1c2ggd2l0aGluIGFscmVhZHkgZmx1c2hlZCBwb3J0aW9uIik7Ci0gICAgICAgIH0KLSAgICAgICAgZmx1c2hlZFBvcyA9IHBvczsKLSAgICAgICAgLy8gLS0gVE9ETyBpbXBsZW1lbnQKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBmbHVzaCgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGZsdXNoQmVmb3JlKGdldFN0cmVhbVBvc2l0aW9uKCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBsb25nIGdldEZsdXNoZWRQb3NpdGlvbigpIHsKLSAgICAgICAgcmV0dXJuIGZsdXNoZWRQb3M7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNDYWNoZWQoKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsgLy8gZGVmCi0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNDYWNoZWRNZW1vcnkoKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsgLy8gZGVmCi0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNDYWNoZWRGaWxlKCkgewotICAgICAgICByZXR1cm4gZmFsc2U7IC8vIGRlZgotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGNsb3NlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgY2hlY2tDbG9zZWQoKTsKLSAgICAgICAgY2xvc2VkID0gdHJ1ZTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbmFsaXplcyB0aGlzIG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAdGhyb3dzIFRocm93YWJsZQotICAgICAqICAgICAgICAgICAgIGlmIGFuIGVycm9yIG9jY3Vycy4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgdm9pZCBmaW5hbGl6ZSgpIHRocm93cyBUaHJvd2FibGUgewotICAgICAgICBpZiAoIWNsb3NlZCkgewotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICBjbG9zZSgpOwotICAgICAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgICAgICBzdXBlci5maW5hbGl6ZSgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIENsYXNzIFBvc2l0aW9uU3RhY2suCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgY2xhc3MgUG9zaXRpb25TdGFjayB7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFRoZSBDb25zdGFudCBTSVpFLgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFNJWkUgPSAxMDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhlIHZhbHVlcy4KLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgbG9uZ1tdIHZhbHVlcyA9IG5ldyBsb25nW1NJWkVdOwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUgcG9zLgotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSBpbnQgcG9zID0gMDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogUHVzaC4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSB2Ci0gICAgICAgICAqICAgICAgICAgICAgdGhlIHYuCi0gICAgICAgICAqLwotICAgICAgICB2b2lkIHB1c2gobG9uZyB2KSB7Ci0gICAgICAgICAgICBpZiAocG9zID49IHZhbHVlcy5sZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBlbnN1cmUocG9zICsgMSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB2YWx1ZXNbcG9zKytdID0gdjsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBQb3AuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRoZSBsb25nLgotICAgICAgICAgKi8KLSAgICAgICAgbG9uZyBwb3AoKSB7Ci0gICAgICAgICAgICByZXR1cm4gdmFsdWVzWy0tcG9zXTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDaGVja3MgaWYgaXMgZW1wdHkuCi0gICAgICAgICAqIAotICAgICAgICAgKiBAcmV0dXJuIHRydWUsIGlmIGlzIGVtcHR5LgotICAgICAgICAgKi8KLSAgICAgICAgYm9vbGVhbiBpc0VtcHR5KCkgewotICAgICAgICAgICAgcmV0dXJuIHBvcyA9PSAwOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIEVuc3VyZS4KLSAgICAgICAgICogCi0gICAgICAgICAqIEBwYXJhbSBzaXplCi0gICAgICAgICAqICAgICAgICAgICAgdGhlIHNpemUuCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIHZvaWQgZW5zdXJlKGludCBzaXplKSB7Ci0gICAgICAgICAgICBsb25nW10gYXJyID0gbmV3IGxvbmdbTWF0aC5tYXgoMiAqIHZhbHVlcy5sZW5ndGgsIHNpemUpXTsKLSAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkodmFsdWVzLCAwLCBhcnIsIDAsIHZhbHVlcy5sZW5ndGgpOwotICAgICAgICAgICAgdmFsdWVzID0gYXJyOwotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL0ltYWdlT3V0cHV0U3RyZWFtLmphdmEgYi9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vSW1hZ2VPdXRwdXRTdHJlYW0uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjhlYzkzMi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vSW1hZ2VPdXRwdXRTdHJlYW0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDMwNyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFJ1c3RlbSBWLiBSYWZpa292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0gKi8KLQotcGFja2FnZSBqYXZheC5pbWFnZWlvLnN0cmVhbTsKLQotaW1wb3J0IGphdmEuaW8uRGF0YU91dHB1dDsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotCi0vKioKLSAqIFRoZSBJbWFnZU91dHB1dFN0cmVhbSByZXByZXNlbnRzIG91dHB1dCBzdHJlYW0gaW50ZXJmYWNlIHRoYXQgaXMgdXNlZCBieQotICogSW1hZ2VXcml0ZXJzLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGludGVyZmFjZSBJbWFnZU91dHB1dFN0cmVhbSBleHRlbmRzIERhdGFPdXRwdXQsIEltYWdlSW5wdXRTdHJlYW0gewotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIGEgc2luZ2xlIGJ5dGUgdG8gdGhlIHN0cmVhbSBhdCB0aGUgY3VycmVudCBwb3NpdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGludGVnZXIgdmFsdWUsIG9mIHdoaWNoIHRoZSA4IGxvd2VzdCBiaXRzIHdpbGwgYmUgd3JpdHRlbi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCB3cml0ZShpbnQgYikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIHRoZSBieXRlcyBhcnJheSB0byB0aGUgc3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgYnl0ZSBhcnJheSB0byBiZSB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlKGJ5dGVbXSBiKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBXcml0ZXMgYSBudW1iZXIgb2YgYnl0ZXMgZnJvbSB0aGUgc3BlY2lmaWVkIGJ5dGUgYXJyYXkgYmVnaW5uaW5nIGZyb20gdGhlCi0gICAgICogc3BlY2lmaWVkIG9mZnNldC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYgotICAgICAqICAgICAgICAgICAgdGhlIGJ5dGUgYXJyYXkuCi0gICAgICogQHBhcmFtIG9mZgotICAgICAqICAgICAgICAgICAgdGhlIG9mZnNldC4KLSAgICAgKiBAcGFyYW0gbGVuCi0gICAgICogICAgICAgICAgICB0aGUgbnVtYmVyIG9mIGJ5dGVzIHRvIGJlIHdyaXR0ZW4uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHZvaWQgd3JpdGUoYnl0ZVtdIGIsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFdyaXRlcyB0aGUgc3BlY2lmaWVkIGJvb2xlYW4gdmFsdWUgdG8gdGhlIHN0cmVhbSwgMSBpZiBpdCBpcyB0cnVlLCAwIGlmCi0gICAgICogaXQgaXMgZmFsc2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIGIKLSAgICAgKiAgICAgICAgICAgIHRoZSBib29sZWFuIHZhbHVlIHRvIGJlIHdyaXR0ZW4uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHZvaWQgd3JpdGVCb29sZWFuKGJvb2xlYW4gYikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIHRoZSA4IGxvd2VzdCBiaXRzIG9mIHRoZSBzcGVjaWZpZWQgaW50ZWdlciB2YWx1ZSB0byB0aGUgc3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiCi0gICAgICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIGludGVnZXIgdmFsdWUuCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHZvaWQgd3JpdGVCeXRlKGludCBiKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBXcml0ZXMgYSBzaG9ydCB2YWx1ZSB0byB0aGUgb3V0cHV0IHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdgotICAgICAqICAgICAgICAgICAgdGhlIHNob3J0IHZhbHVlIHRvIGJlIHdyaXR0ZW4uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHZvaWQgd3JpdGVTaG9ydChpbnQgdikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIHRoZSAxNiBsb3dlc3QgYml0cyBvZiB0aGUgc3BlY2lmaWVkIGludGVnZXIgdmFsdWUgdG8gdGhlIHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdgotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBpbnRlZ2VyIHZhbHVlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlQ2hhcihpbnQgdikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIGFuIGludGVnZXIgdmFsdWUgdG8gdGhlIG91dHB1dCBzdHJlYW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIHYKLSAgICAgKiAgICAgICAgICAgIHRoZSBpbnRlZ2VyIHZhbHVlIHRvIGJlIHdyaXR0ZW4uCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHZvaWQgd3JpdGVJbnQoaW50IHYpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFdyaXRlIGxvbmcuCi0gICAgICogCi0gICAgICogQHBhcmFtIHYKLSAgICAgKiAgICAgICAgICAgIHRoZSBsb25nIHZhbHVlLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlTG9uZyhsb25nIHYpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFdyaXRlcyBhIGZsb2F0IHZhbHVlIHRvIHRoZSBvdXRwdXQgc3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSB2Ci0gICAgICogICAgICAgICAgICB0aGUgZmxvYXQgd2hpY2ggY29udGFpbnMgdmFsdWUgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCB3cml0ZUZsb2F0KGZsb2F0IHYpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFdyaXRlcyBhIGRvdWJsZSB2YWx1ZSB0byB0aGUgb3V0cHV0IHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdgotICAgICAqICAgICAgICAgICAgdGhlIGRvdWJsZSB3aGljaCBjb250YWlucyB2YWx1ZSB0byBiZSB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlRG91YmxlKGRvdWJsZSB2KSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBXcml0ZXMgdGhlIHNwZWNpZmllZCBzdHJpbmcgdG8gdGhlIHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIHN0cmluZyB0byBiZSB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlQnl0ZXMoU3RyaW5nIHMpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFdyaXRlcyB0aGUgc3BlY2lmaWVkIFN0cmluZyB0byB0aGUgb3V0cHV0IHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIFN0cmluZyB0byBiZSB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlQ2hhcnMoU3RyaW5nIHMpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFdyaXRlcyAyIGJ5dGVzIHRvIHRoZSBvdXRwdXQgc3RyZWFtIGluIHRoZSBtb2RpZmllZCBVVEYtOCByZXByZXNlbnRhdGlvbgotICAgICAqIG9mIGV2ZXJ5IGNoYXJhY3RlciBvZiB0aGUgc3BlY2lmaWVkIHN0cmluZy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIHNwZWNpZmllZCBzdHJpbmcgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCB3cml0ZVVURihTdHJpbmcgcykgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogRmx1c2hlcyB0aGUgaW5pdGlhbCBwb3NpdGlvbiBpbiB0aGlzIHN0cmVhbSBwcmlvciB0byB0aGUgc3BlY2lmaWVkIHN0cmVhbQotICAgICAqIHBvc2l0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBwb3MKLSAgICAgKiAgICAgICAgICAgIHRoZSBwb3NpdGlvbi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCBmbHVzaEJlZm9yZShsb25nIHBvcykgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIGEgbGVuIG51bWJlciBvZiBzaG9ydCB2YWx1ZXMgZnJvbSB0aGUgc3BlY2lmaWVkIGFycmF5IHRvIHRoZQotICAgICAqIHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gcwotICAgICAqICAgICAgICAgICAgdGhlIHNob3J0cyBhcnJheSB0byBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIGNoYXIgYXJyYXkuCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCBvZiBjaGFycyB0byBiZSB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlU2hvcnRzKHNob3J0W10gcywgaW50IG9mZiwgaW50IGxlbikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIGEgbGVuIG51bWJlciBvZiBjaGFycyB0byB0aGUgc3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjCi0gICAgICogICAgICAgICAgICB0aGUgY2hhciBhcnJheSB0byBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIGNoYXIgYXJyYXkuCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCBvZiBjaGFycyB0byBiZSB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlQ2hhcnMoY2hhcltdIGMsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFdyaXRlcyBhIGxlbiBudW1iZXIgb2YgaW50ZWdlciB2YWx1ZXMgZnJvbSB0aGUgc3BlY2lmaWVkIGFycmF5IHRvIHRoZQotICAgICAqIHN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaQotICAgICAqICAgICAgICAgICAgdGhlIGludGVnZXIgYXJyYXkgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBjaGFyIGFycmF5LgotICAgICAqIEBwYXJhbSBsZW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggb2YgY2hhcnMgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCB3cml0ZUludHMoaW50W10gaSwgaW50IG9mZiwgaW50IGxlbikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIGEgbGVuIG51bWJlciBvZiBsb25nIHZhbHVlcyBmcm9tIHRoZSBzcGVjaWZpZWQgYXJyYXkgdG8gdGhlCi0gICAgICogc3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsCi0gICAgICogICAgICAgICAgICB0aGUgbG9uZyBhcnJheSB0byBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIGNoYXIgYXJyYXkuCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCBvZiBjaGFycyB0byBiZSB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlTG9uZ3MobG9uZ1tdIGwsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIC8qKgotICAgICAqIFdyaXRlcyBhIGxlbiBudW1iZXIgb2YgZmxvYXQgdmFsdWVzIGZyb20gdGhlIHNwZWNpZmllZCBhcnJheSB0byB0aGUKLSAgICAgKiBzdHJlYW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIGYKLSAgICAgKiAgICAgICAgICAgIHRoZSBmbG9hdCBhcnJheSB0byBiZSB3cml0dGVuLgotICAgICAqIEBwYXJhbSBvZmYKLSAgICAgKiAgICAgICAgICAgIHRoZSBvZmZzZXQgaW4gdGhlIGNoYXIgYXJyYXkuCi0gICAgICogQHBhcmFtIGxlbgotICAgICAqICAgICAgICAgICAgdGhlIGxlbmd0aCBvZiBjaGFycyB0byBiZSB3cml0dGVuLgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlRmxvYXRzKGZsb2F0W10gZiwgaW50IG9mZiwgaW50IGxlbikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIGEgbGVuIG51bWJlciBvZiBkb3VibGUgdmFsdWVzIGZyb20gdGhlIHNwZWNpZmllZCBhcnJheSB0byB0aGUKLSAgICAgKiBzdHJlYW0uCi0gICAgICogCi0gICAgICogQHBhcmFtIGQKLSAgICAgKiAgICAgICAgICAgIHRoZSBkb3VibGUgYXJyYXkgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAcGFyYW0gb2ZmCi0gICAgICogICAgICAgICAgICB0aGUgb2Zmc2V0IGluIHRoZSBjaGFyIGFycmF5LgotICAgICAqIEBwYXJhbSBsZW4KLSAgICAgKiAgICAgICAgICAgIHRoZSBsZW5ndGggb2YgY2hhcnMgdG8gYmUgd3JpdHRlbi4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCB3cml0ZURvdWJsZXMoZG91YmxlW10gZCwgaW50IG9mZiwgaW50IGxlbikgdGhyb3dzIElPRXhjZXB0aW9uOwotCi0gICAgLyoqCi0gICAgICogV3JpdGVzIGEgc2luZ2xlIGJpdCBhdCB0aGUgY3VycmVudCBwb3NpdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYml0Ci0gICAgICogICAgICAgICAgICB0aGUgaW50ZWdlciB3aG9zZSBsZWFzdCBzaWduaWZpY2FudCBiaXQgaXMgdG8gYmUgd3JpdHRlbiB0bwotICAgICAqICAgICAgICAgICAgdGhlIHN0cmVhbS4KLSAgICAgKiBAdGhyb3dzIElPRXhjZXB0aW9uCi0gICAgICogICAgICAgICAgICAgaWYgYW4gSS9PIGV4Y2VwdGlvbiBoYXMgb2NjdXJyZWQuCi0gICAgICovCi0gICAgdm9pZCB3cml0ZUJpdChpbnQgYml0KSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLSAgICAvKioKLSAgICAgKiBXcml0ZXMgYSBzZXF1ZW5jZSBvZiBiaXRzIGJlZ2lubmluZyBmcm9tIHRoZSBjdXJyZW50IHBvc2l0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiaXRzCi0gICAgICogICAgICAgICAgICB0aGUgbG9uZyB2YWx1ZSBjb250YWluaW5nIHRoZSBiaXRzIHRvIGJlIHdyaXR0ZW4sIHN0YXJ0aW5nCi0gICAgICogICAgICAgICAgICB3aXRoIHRoZSBiaXQgaW4gcG9zaXRpb24gbnVtQml0cyAtIDEgZG93biB0byB0aGUgbGVhc3QKLSAgICAgKiAgICAgICAgICAgIHNpZ25pZmljYW50IGJpdC4KLSAgICAgKiBAcGFyYW0gbnVtQml0cwotICAgICAqICAgICAgICAgICAgdGhlIG51bWJlciBvZiBzaWduaWZpY2FudCBiaXQsIGl0IGNhbiBiZSBiZXR3ZWVuIDAgYW5kIDY0LgotICAgICAqIEB0aHJvd3MgSU9FeGNlcHRpb24KLSAgICAgKiAgICAgICAgICAgICBpZiBhbiBJL08gZXhjZXB0aW9uIGhhcyBvY2N1cnJlZC4KLSAgICAgKi8KLSAgICB2b2lkIHdyaXRlQml0cyhsb25nIGJpdHMsIGludCBudW1CaXRzKSB0aHJvd3MgSU9FeGNlcHRpb247Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9JbWFnZU91dHB1dFN0cmVhbUltcGwuamF2YSBiL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9JbWFnZU91dHB1dFN0cmVhbUltcGwuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMGZlZjc4Zi4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vSW1hZ2VPdXRwdXRTdHJlYW1JbXBsLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNzQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5zdHJlYW07Ci0KLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEubmlvLkJ5dGVPcmRlcjsKLQotLyogCi0gKiBAYXV0aG9yIFJ1c3RlbSBWLiBSYWZpa292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMyAkCi0gKi8KLQotLyoqCi0gKiBUaGUgSW1hZ2VPdXRwdXRTdHJlYW1JbXBsIGFic3RyYWN0IGNsYXNzIGltcGxlbWVudHMgdGhlIEltYWdlT3V0cHV0U3RyZWFtCi0gKiBpbnRlcmZhY2UuCi0gKiAKLSAqIEBzaW5jZSBBbmRyb2lkIDEuMAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgSW1hZ2VPdXRwdXRTdHJlYW1JbXBsIGV4dGVuZHMgSW1hZ2VJbnB1dFN0cmVhbUltcGwgaW1wbGVtZW50cwotICAgICAgICBJbWFnZU91dHB1dFN0cmVhbSB7Ci0KLSAgICAvKioKLSAgICAgKiBJbnN0YW50aWF0ZXMgYSBuZXcgSW1hZ2VPdXRwdXRTdHJlYW1JbXBsLgotICAgICAqLwotICAgIHB1YmxpYyBJbWFnZU91dHB1dFN0cmVhbUltcGwoKSB7Ci0gICAgfQotCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgd3JpdGUoaW50IGIpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlKGJ5dGVbXSBiKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB3cml0ZShiLCAwLCBiLmxlbmd0aCk7Ci0gICAgfQotCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgd3JpdGUoYnl0ZVtdIGIsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlQm9vbGVhbihib29sZWFuIHYpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHdyaXRlKHYgPyAxIDogMCk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgd3JpdGVCeXRlKGludCB2KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB3cml0ZSh2KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3cml0ZVNob3J0KGludCB2KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAoYnl0ZU9yZGVyID09IEJ5dGVPcmRlci5CSUdfRU5ESUFOKSB7Ci0KLSAgICAgICAgfSBlbHNlIHsKLQotICAgICAgICB9Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlQ2hhcihpbnQgdikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgd3JpdGVTaG9ydCh2KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3cml0ZUludChpbnQgdikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKGJ5dGVPcmRlciA9PSBCeXRlT3JkZXIuQklHX0VORElBTikgewotCi0gICAgICAgIH0gZWxzZSB7Ci0KLSAgICAgICAgfQotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3cml0ZUxvbmcobG9uZyB2KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAoYnl0ZU9yZGVyID09IEJ5dGVPcmRlci5CSUdfRU5ESUFOKSB7Ci0KLSAgICAgICAgfSBlbHNlIHsKLQotICAgICAgICB9Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlRmxvYXQoZmxvYXQgdikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgd3JpdGVJbnQoRmxvYXQuZmxvYXRUb0ludEJpdHModikpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlRG91YmxlKGRvdWJsZSB2KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICB3cml0ZUxvbmcoRG91YmxlLmRvdWJsZVRvTG9uZ0JpdHModikpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlQnl0ZXMoU3RyaW5nIHMpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHdyaXRlKHMuZ2V0Qnl0ZXMoKSk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgd3JpdGVDaGFycyhTdHJpbmcgcykgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgY2hhcltdIGNocyA9IHMudG9DaGFyQXJyYXkoKTsKLSAgICAgICAgd3JpdGVDaGFycyhjaHMsIDAsIGNocy5sZW5ndGgpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlVVRGKFN0cmluZyBzKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3cml0ZVNob3J0cyhzaG9ydFtdIHMsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlQ2hhcnMoY2hhcltdIGMsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlSW50cyhpbnRbXSBpLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3cml0ZUxvbmdzKGxvbmdbXSBsLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3cml0ZUZsb2F0cyhmbG9hdFtdIGYsIGludCBvZmYsIGludCBsZW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vIC0tIFRPRE8gaW1wbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHdyaXRlRG91Ymxlcyhkb3VibGVbXSBkLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3cml0ZUJpdChpbnQgYml0KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLyAtLSBUT0RPIGltcGxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCB3cml0ZUJpdHMobG9uZyBiaXRzLCBpbnQgbnVtQml0cykgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgLy8gLS0gVE9ETyBpbXBsZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRmx1c2hlcyB0aGUgYml0cy4gVGhpcyBtZXRob2Qgc2hvdWxkIGJlIGNhbGxlZCBpbiB0aGUgd3JpdGUgbWV0aG9kcyBieQotICAgICAqIHN1YmNsYXNzZXMuCi0gICAgICogCi0gICAgICogQHRocm93cyBJT0V4Y2VwdGlvbgotICAgICAqICAgICAgICAgICAgIGlmIGFuIEkvTyBleGNlcHRpb24gaGFzIG9jY3VycmVkLgotICAgICAqLwotICAgIHByb3RlY3RlZCBmaW5hbCB2b2lkIGZsdXNoQml0cygpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChiaXRPZmZzZXQgPT0gMCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gLS0gVE9ETyBpbXBsZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL01lbW9yeUNhY2hlSW1hZ2VJbnB1dFN0cmVhbS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL01lbW9yeUNhY2hlSW1hZ2VJbnB1dFN0cmVhbS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkN2ZjNzkxLi4wMDAwMDAwCi0tLSBhL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9NZW1vcnlDYWNoZUltYWdlSW5wdXRTdHJlYW0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDExOSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIGphdmF4LmltYWdlaW8uc3RyZWFtOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5zdHJlYW0uUmFuZG9tQWNjZXNzTWVtb3J5Q2FjaGU7Ci0KLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uSW5wdXRTdHJlYW07Ci0KLS8qKgotICogVGhlIE1lbW9yeUNhY2hlSW1hZ2VJbnB1dFN0cmVhbSBjbGFzcyBpbXBsZW1lbnRzIEltYWdlSW5wdXRTdHJlYW0gdXNpbmcgYQotICogbWVtb3J5IGJ1ZmZlciBmb3IgY2FjaGluZyB0aGUgZGF0YS4KLSAqIAotICogQHNpbmNlIEFuZHJvaWQgMS4wCi0gKi8KLXB1YmxpYyBjbGFzcyBNZW1vcnlDYWNoZUltYWdlSW5wdXRTdHJlYW0gZXh0ZW5kcyBJbWFnZUlucHV0U3RyZWFtSW1wbCB7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgaXMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBJbnB1dFN0cmVhbSBpczsKLQotICAgIC8qKgotICAgICAqIFRoZSByYW1jLgotICAgICAqLwotICAgIHByaXZhdGUgUmFuZG9tQWNjZXNzTWVtb3J5Q2FjaGUgcmFtYyA9IG5ldyBSYW5kb21BY2Nlc3NNZW1vcnlDYWNoZSgpOwotCi0gICAgLyoqCi0gICAgICogSW5zdGFudGlhdGVzIGEgbmV3IE1lbW9yeUNhY2hlSW1hZ2VJbnB1dFN0cmVhbSB3aGljaCByZWFkcyBmcm9tIHRoZQotICAgICAqIHNwZWNpZmllZCBJbnB1dFN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gc3RyZWFtCi0gICAgICogICAgICAgICAgICB0aGUgSW5wdXRTdHJlYW0gdG8gYmUgcmVhZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgTWVtb3J5Q2FjaGVJbWFnZUlucHV0U3RyZWFtKElucHV0U3RyZWFtIHN0cmVhbSkgewotICAgICAgICBpZiAoc3RyZWFtID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInN0cmVhbSA9PSBudWxsISIpOwotICAgICAgICB9Ci0gICAgICAgIGlzID0gc3RyZWFtOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgcmVhZCgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGJpdE9mZnNldCA9IDA7Ci0KLSAgICAgICAgaWYgKHN0cmVhbVBvcyA+PSByYW1jLmxlbmd0aCgpKSB7Ci0gICAgICAgICAgICBpbnQgY291bnQgPSAoaW50KShzdHJlYW1Qb3MgLSByYW1jLmxlbmd0aCgpICsgMSk7Ci0gICAgICAgICAgICBpbnQgYnl0ZXNBcHBlbmRlZCA9IHJhbWMuYXBwZW5kRGF0YShpcywgY291bnQpOwotCi0gICAgICAgICAgICBpZiAoYnl0ZXNBcHBlbmRlZCA8IGNvdW50KSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIC0xOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHJlcyA9IHJhbWMuZ2V0RGF0YShzdHJlYW1Qb3MpOwotICAgICAgICBpZiAocmVzID49IDApIHsKLSAgICAgICAgICAgIHN0cmVhbVBvcysrOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCByZWFkKGJ5dGVbXSBiLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBiaXRPZmZzZXQgPSAwOwotCi0gICAgICAgIGlmIChzdHJlYW1Qb3MgPj0gcmFtYy5sZW5ndGgoKSkgewotICAgICAgICAgICAgaW50IGNvdW50ID0gKGludCkoc3RyZWFtUG9zIC0gcmFtYy5sZW5ndGgoKSArIGxlbik7Ci0gICAgICAgICAgICByYW1jLmFwcGVuZERhdGEoaXMsIGNvdW50KTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCByZXMgPSByYW1jLmdldERhdGEoYiwgb2ZmLCBsZW4sIHN0cmVhbVBvcyk7Ci0gICAgICAgIGlmIChyZXMgPiAwKSB7Ci0gICAgICAgICAgICBzdHJlYW1Qb3MgKz0gcmVzOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDYWNoZWQoKSB7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGlzQ2FjaGVkRmlsZSgpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGlzQ2FjaGVkTWVtb3J5KCkgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBjbG9zZSgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHN1cGVyLmNsb3NlKCk7Ci0gICAgICAgIHJhbWMuY2xvc2UoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmbHVzaEJlZm9yZShsb25nIHBvcykgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgc3VwZXIuZmx1c2hCZWZvcmUocG9zKTsKLSAgICAgICAgcmFtYy5mcmVlQmVmb3JlKGdldEZsdXNoZWRQb3NpdGlvbigpKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vTWVtb3J5Q2FjaGVJbWFnZU91dHB1dFN0cmVhbS5qYXZhIGIvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL01lbW9yeUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMWRmNDBhMy4uMDAwMDAwMAotLS0gYS9hd3QvamF2YXgvaW1hZ2Vpby9zdHJlYW0vTWVtb3J5Q2FjaGVJbWFnZU91dHB1dFN0cmVhbS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTM1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgamF2YXguaW1hZ2Vpby5zdHJlYW07Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnN0cmVhbS5SYW5kb21BY2Nlc3NNZW1vcnlDYWNoZTsKLQotaW1wb3J0IGphdmEuaW8uT3V0cHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci0KLS8qKgotICogVGhlIE1lbW9yeUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0gY2xhc3MgaW1wbGVtZW50cyBJbWFnZU91dHB1dFN0cmVhbSB1c2luZyBhCi0gKiBtZW1vcnkgYnVmZmVyIGZvciBjYWNoaW5nIHRoZSBkYXRhLgotICogCi0gKiBAc2luY2UgQW5kcm9pZCAxLjAKLSAqLwotcHVibGljIGNsYXNzIE1lbW9yeUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0gZXh0ZW5kcyBJbWFnZU91dHB1dFN0cmVhbUltcGwgewotCi0gICAgLyoqCi0gICAgICogVGhlIG9zLgotICAgICAqLwotICAgIE91dHB1dFN0cmVhbSBvczsKLQotICAgIC8qKgotICAgICAqIFRoZSByYW1jLgotICAgICAqLwotICAgIFJhbmRvbUFjY2Vzc01lbW9yeUNhY2hlIHJhbWMgPSBuZXcgUmFuZG9tQWNjZXNzTWVtb3J5Q2FjaGUoKTsKLQotICAgIC8qKgotICAgICAqIEluc3RhbnRpYXRlcyBhIG5ldyBNZW1vcnlDYWNoZUltYWdlT3V0cHV0U3RyZWFtIHdoaWNoIHdyaXRlcyB0byB0aGUKLSAgICAgKiBzcGVjaWZpZWQgT3V0cHV0U3RyZWFtLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHJlYW0KLSAgICAgKiAgICAgICAgICAgIHRoZSBPdXRwdXRTdHJlYW0uCi0gICAgICovCi0gICAgcHVibGljIE1lbW9yeUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0oT3V0cHV0U3RyZWFtIHN0cmVhbSkgewotICAgICAgICBpZiAoc3RyZWFtID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oInN0cmVhbSA9PSBudWxsISIpOwotICAgICAgICB9Ci0gICAgICAgIG9zID0gc3RyZWFtOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHdyaXRlKGludCBiKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBmbHVzaEJpdHMoKTsgLy8gU2VlIHRoZSBmbHVzaEJpdHMgbWV0aG9kIGRlc2NyaXB0aW9uCi0KLSAgICAgICAgcmFtYy5wdXREYXRhKGIsIHN0cmVhbVBvcyk7Ci0gICAgICAgIHN0cmVhbVBvcysrOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHdyaXRlKGJ5dGVbXSBiLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBmbHVzaEJpdHMoKTsgLy8gU2VlIHRoZSBmbHVzaEJpdHMgbWV0aG9kIGRlc2NyaXB0aW9uCi0KLSAgICAgICAgcmFtYy5wdXREYXRhKGIsIG9mZiwgbGVuLCBzdHJlYW1Qb3MpOwotICAgICAgICBzdHJlYW1Qb3MgKz0gbGVuOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgcmVhZCgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGJpdE9mZnNldCA9IDA7Ci0KLSAgICAgICAgaW50IHJlcyA9IHJhbWMuZ2V0RGF0YShzdHJlYW1Qb3MpOwotICAgICAgICBpZiAocmVzID49IDApIHsKLSAgICAgICAgICAgIHN0cmVhbVBvcysrOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCByZWFkKGJ5dGVbXSBiLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBiaXRPZmZzZXQgPSAwOwotCi0gICAgICAgIGludCByZXMgPSByYW1jLmdldERhdGEoYiwgb2ZmLCBsZW4sIHN0cmVhbVBvcyk7Ci0gICAgICAgIGlmIChyZXMgPiAwKSB7Ci0gICAgICAgICAgICBzdHJlYW1Qb3MgKz0gcmVzOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGxvbmcgbGVuZ3RoKCkgewotICAgICAgICByZXR1cm4gcmFtYy5sZW5ndGgoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NhY2hlZCgpIHsKLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDYWNoZWRNZW1vcnkoKSB7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGlzQ2FjaGVkRmlsZSgpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGNsb3NlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgZmx1c2hCZWZvcmUobGVuZ3RoKCkpOwotICAgICAgICBzdXBlci5jbG9zZSgpOwotICAgICAgICByYW1jLmNsb3NlKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZmx1c2hCZWZvcmUobG9uZyBwb3MpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGxvbmcgZmx1c2hlZFBvc2l0aW9uID0gZ2V0Rmx1c2hlZFBvc2l0aW9uKCk7Ci0gICAgICAgIHN1cGVyLmZsdXNoQmVmb3JlKHBvcyk7Ci0KLSAgICAgICAgbG9uZyBuZXdGbHVzaGVkUG9zaXRpb24gPSBnZXRGbHVzaGVkUG9zaXRpb24oKTsKLSAgICAgICAgaW50IG5CeXRlcyA9IChpbnQpKG5ld0ZsdXNoZWRQb3NpdGlvbiAtIGZsdXNoZWRQb3NpdGlvbik7Ci0KLSAgICAgICAgcmFtYy5nZXREYXRhKG9zLCBuQnl0ZXMsIGZsdXNoZWRQb3NpdGlvbik7Ci0gICAgICAgIHJhbWMuZnJlZUJlZm9yZShuZXdGbHVzaGVkUG9zaXRpb24pOwotCi0gICAgICAgIG9zLmZsdXNoKCk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL3BhY2thZ2UuaHRtbCBiL2F3dC9qYXZheC9pbWFnZWlvL3N0cmVhbS9wYWNrYWdlLmh0bWwKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDZjZjUzYzMuLjAwMDAwMDAKLS0tIGEvYXd0L2phdmF4L2ltYWdlaW8vc3RyZWFtL3BhY2thZ2UuaHRtbAorKysgL2Rldi9udWxsCkBAIC0xLDggKzAsMCBAQAotPGh0bWw+Ci0gIDxib2R5PgotICAgIDxwPgotICAgICAgVGhpcyBwYWNrYWdlIGNvbnRhaW5zIGNsYXNzZXMgYW5kIGludGVyZmFjZXMgZm9yIGhhbmRsaW5nIGltYWdlcyB3aXRoIGxvdy1sZXZlbCBJL08gb3BlcmF0aW9ucy4gCi0gICAgPC9wPgotICBAc2luY2UgQW5kcm9pZCAxLjAKLSAgPC9ib2R5PgotPC9odG1sPgpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvQ2hvaWNlU3R5bGUuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L0Nob2ljZVN0eWxlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDkzYjdhYWQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvQ2hvaWNlU3R5bGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDMzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRG1pdHJ5IEEuIER1cm5ldgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Q7Ci0KLS8qKgotICogQ2hvaWNlU3R5bGUuCi0gKiBJcyB1c2VkIHRvIGRlZmluZSBjdXN0b20gY2hvaWNlIHByb3BlcnRpZXM6Ci0gKiB3aWR0aCBhbmQgeCBzY3JlZW4gY29vcmRpbmF0ZSBvZiB0aGUgbGlzdCBwb3B1cCB3aW5kb3cuIAotICovCi1wdWJsaWMgaW50ZXJmYWNlIENob2ljZVN0eWxlIHsKLQotICAgIGludCBnZXRQb3B1cFgoaW50IHgsIGludCB3aWR0aCwgaW50IGNob2ljZVdpZHRoLCBpbnQgc2NyZWVuV2lkdGgpOwotICAgIGludCBnZXRQb3B1cFdpZHRoKGludCBjaG9pY2VXaWR0aCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L0NsaXBSZWdpb24uamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L0NsaXBSZWdpb24uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzg5YTgxZC4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9DbGlwUmVnaW9uLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4NCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdiwgQW50b24gQXZ0YW1vbm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dDsKLQotaW1wb3J0IGphdmEuYXd0LkNvbXBvbmVudDsKLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLk11bHRpUmVjdEFyZWE7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLXB1YmxpYyBjbGFzcyBDbGlwUmVnaW9uIGV4dGVuZHMgUmVjdGFuZ2xlIHsKLSAgICBwcml2YXRlIGZpbmFsIE11bHRpUmVjdEFyZWEgY2xpcDsKLQotICAgIHB1YmxpYyBDbGlwUmVnaW9uKGZpbmFsIE11bHRpUmVjdEFyZWEgY2xpcCkgewotICAgICAgICB0aGlzLmNsaXAgPSBuZXcgTXVsdGlSZWN0QXJlYShjbGlwKTsKLSAgICAgICAgc2V0Qm91bmRzKGNsaXAuZ2V0Qm91bmRzKCkpOwotICAgIH0KLQotICAgIHB1YmxpYyBNdWx0aVJlY3RBcmVhIGdldENsaXAoKSB7Ci0gICAgICAgIHJldHVybiBjbGlwOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgdG9TdHJpbmcoKSB7Ci0gICAgICAgIFN0cmluZyBzdHIgPSBjbGlwLnRvU3RyaW5nKCk7Ci0gICAgICAgIGludCBpID0gc3RyLmluZGV4T2YoJ1snKTsKLSAgICAgICAgc3RyID0gc3RyLnN1YnN0cmluZyhpKTsKLSAgICAgICAgaWYgKGNsaXAuZ2V0UmVjdENvdW50KCkgPT0gMSkgewotICAgICAgICAgICAgc3RyID0gc3RyLnN1YnN0cmluZygxLCBzdHIubGVuZ3RoKCkgLSAxKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZ2V0Q2xhc3MoKS5nZXROYW1lKCkgKyBzdHI7Ci0gICAgfQotCi0KLSAgICBwdWJsaWMgdm9pZCBjb252ZXJ0UmVnaW9uKGZpbmFsIENvbXBvbmVudCBjaGlsZCwgZmluYWwgQ29tcG9uZW50IHBhcmVudCkgewotICAgICAgICBjb252ZXJ0UmVnaW9uKGNoaWxkLCBjbGlwLCBwYXJlbnQpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGludGVyc2VjdChmaW5hbCBSZWN0YW5nbGUgcmVjdCkgewotICAgICAgICBjbGlwLmludGVyc2VjdChyZWN0KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBpc0VtcHR5KCkgewotICAgICAgICByZXR1cm4gY2xpcC5pc0VtcHR5KCk7Ci0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyB2b2lkIGNvbnZlcnRSZWdpb24oZmluYWwgQ29tcG9uZW50IGNoaWxkLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZpbmFsIE11bHRpUmVjdEFyZWEgcmVnaW9uLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZpbmFsIENvbXBvbmVudCBwYXJlbnQpIHsKLSAgICAgICAgaW50IHggPSAwLCB5ID0gMDsKLSAgICAgICAgQ29tcG9uZW50IGMgPSBjaGlsZDsKLSAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgLyoKLSAgICAgICAgZm9yICg7IGMgIT0gbnVsbCAmJiBjICE9IHBhcmVudDsgYyA9IGMuZ2V0UGFyZW50KCkpIHsKLSAgICAgICAgICAgIHggKz0gYy5nZXRYKCk7Ci0gICAgICAgICAgICB5ICs9IGMuZ2V0WSgpOwotICAgICAgICB9Ci0gICAgICAgICovCi0gICAgICAgIGlmIChjID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC41MT1Db21wb25lbnQgZXhwZWN0ZWQgdG8gYmUgYSBwYXJlbnQKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNTEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICByZWdpb24udHJhbnNsYXRlKHgsIHkpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L0NvbXBvbmVudEludGVybmFscy5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvQ29tcG9uZW50SW50ZXJuYWxzLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGMzNTk3ODQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvQ29tcG9uZW50SW50ZXJuYWxzLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMTIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0OwotCi0vLz8/P0FXVAotLy9pbXBvcnQgamF2YS5hd3QuQ29tcG9uZW50OwotLy9pbXBvcnQgamF2YS5hd3QuQ29udGFpbmVyOwotLy9pbXBvcnQgamF2YS5hd3QuRGlhbG9nOwotaW1wb3J0IGphdmEuYXd0LkRpbWVuc2lvbjsKLS8vaW1wb3J0IGphdmEuYXd0LkltYWdlOwotaW1wb3J0IGphdmEuYXd0Lkluc2V0czsKLWltcG9ydCBqYXZhLmF3dC5Qb2ludDsKLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci0vL2ltcG9ydCBqYXZhLmF3dC5XaW5kb3c7Ci0vL2ltcG9ydCBqYXZhLmF3dC5DaG9pY2U7Ci1pbXBvcnQgamF2YS5sYW5nLnJlZmxlY3QuSW52b2NhdGlvblRhcmdldEV4Y2VwdGlvbjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuTXVsdGlSZWN0QXJlYTsKLS8vaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QudGV4dC5UZXh0RmllbGRLaXQ7Ci0vL2ltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LnRleHQuVGV4dEtpdDsKLS8vaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLk5hdGl2ZVdpbmRvdzsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuTm90SW1wbGVtZW50ZWRFeGNlcHRpb247Ci0KLS8qKgotICogIFRoZSBhY2Nlc3NvciB0byBBV1QgcHJpdmF0ZSBBUEkKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIENvbXBvbmVudEludGVybmFscyB7Ci0KLSAgICAvKioKLSAgICAgKiBAcmV0dXJuIHRoZSBDb21wb25lbnRJbnRlcm5hbHMgaW5zdGFuY2UgdG8gc2VydmUgdGhlIHJlcXVlc3RzCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBDb21wb25lbnRJbnRlcm5hbHMgZ2V0Q29tcG9uZW50SW50ZXJuYWxzKCkgewotICAgICAgICByZXR1cm4gQ29udGV4dFN0b3JhZ2UuZ2V0Q29tcG9uZW50SW50ZXJuYWxzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhpcyBtZXRob2QgbXVzdCBiZSBjYWxsZWQgYnkgQVdUIHRvIGVzdGFibGlzaCB0aGUgY29ubmVjdGlvbgotICAgICAqIEBwYXJhbSBpbnRlcm5hbHMgLSBpbXBsZW1lbnRhdGlvbiBvZiBDb21wb25lbnRJbnRlcm5hbHMgY3JlYXRlZCBieSBBV1QKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgc2V0Q29tcG9uZW50SW50ZXJuYWxzKENvbXBvbmVudEludGVybmFscyBpbnRlcm5hbHMpIHsKLSAgICAgICAgQ29udGV4dFN0b3JhZ2Uuc2V0Q29tcG9uZW50SW50ZXJuYWxzKGludGVybmFscyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIGFjY2Vzc29yIHRvIG5hdGl2ZSByZXNvdXJjZSBjb25uZWN0ZWQgdG8gYSBjb21wb25lbnQuCi0gICAgICogSXQgcmV0dXJucyBub24tPGNvZGU+bnVsbDwvY29kZT4gdmFsdWUgb25seSBpZiBjb21wb25lbnQKLSAgICAgKiBhbHJlYWR5IGhhcyB0aGUgbmF0aXZlIHJlc291cmNlCi0gICAgICovCi0gICAgLy9wdWJsaWMgYWJzdHJhY3QgTmF0aXZlV2luZG93IGdldE5hdGl2ZVdpbmRvdyhDb21wb25lbnQgY29tcG9uZW50KTsKLQotICAgIC8qKgotICAgICAqIENvbm5lY3QgV2luZG93IG9iamVjdCB0byBleGlzdGluZyBuYXRpdmUgcmVzb3VyY2UKLSAgICAgKiBAcGFyYW0gbmF0aXZlV2luZG93SWQgLSBpZCBvZiBuYXRpdmUgd2luZG93IHRvIGF0dGFjaAotICAgICAqIEByZXR1cm4gV2luZG93IG9iamVjdCB3aXRoIHNwZWNpYWwgYmVoYXZpb3VyIHRoYXQKLSAgICAgKiByZXN0cmljdHMgbWFudXB1bGF0aW9uIHdpdGggdGhhdCB3aW5kb3cKLSAgICAgKi8KLSAgICAvL3B1YmxpYyBhYnN0cmFjdCBXaW5kb3cgYXR0YWNoTmF0aXZlV2luZG93KGxvbmcgbmF0aXZlV2luZG93SWQpOwotCi0gICAgLyoqCi0gICAgICogU3RhcnQgbW91c2UgZ3JhYiBpbiAiY2xpZW50IiBtb2RlLgotICAgICAqIEFsbCBtb3VzZSBldmVudHMgaW4gQVdUIGNvbXBvbmVudHMgd2lsbCBiZSByZXBvcnRlZCBhcyB1c3VhbCwKLSAgICAgKiBtb3VzZSBldmVudHMgdGhhdCBvY2N1cmVkIG91dHNpZGUgb2YgQVdUIGNvbXBvbmVudHMgd2lsbCBiZSBzZW50IHRvCi0gICAgICogdGhlIHdpbmRvdyBwYXNzZWQgYXMgZ3JhYldpbmRvdyBwYXJhbWV0ZXIuIFdoZW4gbW91c2UgZ3JhYiBpcyBjYW5jZWxlZAotICAgICAqIChiZWNhdXNlIG9mIGNsaWNrIGluIG5vbi1BV1Qgd2luZG93IG9yIGJ5IHRhc2sgc3dpdGNoaW5nKQotICAgICAqIHRoZSB3aGVuQ2FuY2VsZWQgY2FsbGJhY2sgaXMgY2FsbGVkCi0gICAgICoKLSAgICAgKiBAcGFyYW0gZ3JhYldpbmRvdyAtIHdpbmRvdyB0aGF0IHdpbGwgb3duIHRoZSBncmFiCi0gICAgICogQHBhcmFtIHdoZW5DYW5jZWxlZCAtIGNhbGxiYWNrIGNhbGxlZCB3aGVuIGdyYWIgaXMgY2FuY2VsZWQgYnkgdXNlcidzIGFjdGlvbgotICAgICAqLwotICAgIC8vcHVibGljIGFic3RyYWN0IHZvaWQgc3RhcnRNb3VzZUdyYWIoV2luZG93IGdyYWJXaW5kb3csIFJ1bm5hYmxlIHdoZW5DYW5jZWxlZCk7Ci0KLSAgICAvKioKLSAgICAgKiBFbmQgbW91c2UgZ3JhYiBhbmQgcmVzdW1lIG5vcm1hbCBwcm9jZXNzaW5nIG9mIG1vdXNlIGV2ZW50cwotICAgICAqLwotICAgIC8vcHVibGljIGFic3RyYWN0IHZvaWQgZW5kTW91c2VHcmFiKCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXQgdGhlIDxjb2RlPnBvcHVwPC9jb2RlPiBmbGFnIG9mIHRoZSB3aW5kb3cgdG8gdHJ1ZS4KLSAgICAgKiBUaGlzIHdpbmRvdyB3b24ndCBiZSBjb250cm9sbGVkIGJ5IHdpbmRvdyBtYW5hZ2VyIG9uIExpbnV4LgotICAgICAqIENhbGwgdGhpcyBtZXRob2QgYmVmb3JlIHRoZSB3aW5kb3cgaXMgc2hvd24gZmlyc3QgdGltZQotICAgICAqIEBwYXJhbSB3aW5kb3cgLSB0aGUgd2luZG93IHRoYXQgc2hvdWxkIGJlY29tZSBwb3B1cCBvbmUKLSAgICAgKi8KLSAgICAvL3B1YmxpYyBhYnN0cmFjdCB2b2lkIG1ha2VQb3B1cChXaW5kb3cgd2luZG93KTsKLQotICAgIC8qKgotICAgICAqIFRoaXMgbWV0aG9kIG11c3QgYmUgY2FsbGVkIGJ5IEdyYXBoaWNzIGF0IHRoZSBiZWdpbm5pbmcgb2YgZHJhd0ltYWdlKCkKLSAgICAgKiB0byBzdG9yZSBpbWFnZSBkcmF3aW5nIHBhcmFtZXRlcnMgKGRlZmluZWQgYnkgYXBwbGljYXRpb24gZGV2ZWxvcGVyKSBpbiBjb21wb25lbnQKLSAgICAgKgotICAgICAqIEBwYXJhbSBjb21wIC0gY29tcG9uZW50IHRoYXQgZHJhd3MgdGhlIGltYWdlCi0gICAgICogQHBhcmFtIGltYWdlIC0gaW1hZ2UgdG8gYmUgZHJhd24KLSAgICAgKiBAcGFyYW0gZGVzdExvY2F0aW9uIC0gbG9jYXRpb24gb2YgdGhlIGltYWdlIHVwb24gdGhlIGNvbXBvbmVudCdzIHN1cmZhY2UuIE5ldmVyIG51bGwuCi0gICAgICogQHBhcmFtIGRlc3RTaXplIC0gc2l6ZSBvZiB0aGUgY29tcG9uZW50J3MgYXJlYSB0byBiZSBmaWxsZWQgd2l0aCB0aGUgaW1hZ2UuCi0gICAgICogICAgICAgICAgICAgICAgICBFcXVhbHMgdG8gbnVsbCBpZiBzaXplIHBhcmFtZXRlcnMgb21pdHRlZCBpbiBkcmF3SW1hZ2UuCi0gICAgICogQHBhcmFtIHNvdXJjZSAtIGFyZWEgb2YgdGhlIGltYWdlIHRvIGJlIGRyYXduIG9uIHRoZSBjb21wb25lbnQuCi0gICAgICogICAgICAgICAgICAgICAgICBFcXVhbHMgdG8gbnVsbCBpZiBzcmMgcGFyYW1ldGVycyBvbWl0dGVkIGluIGRyYXdJbWFnZS4KLSAgICAgKi8KLSAgICAvKgotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIG9uRHJhd0ltYWdlKENvbXBvbmVudCBjb21wLCBJbWFnZSBpbWFnZSwgUG9pbnQgZGVzdExvY2F0aW9uLAotICAgICAgICAgICAgRGltZW5zaW9uIGRlc3RTaXplLCBSZWN0YW5nbGUgc291cmNlKTsKLSovCi0gICAgLyoqCi0gICAgICogU2V0cyBzeXN0ZW0ncyBjYXJldCBwb3NpdGlvbi4KLSAgICAgKiBUaGlzIG1ldGhvZCBzaG91bGQgYmUgY2FsbGVkIGJ5IHRleHQgY29tcG9uZW50IHRvIHN5bmNocm9uaXplIG91ciBjYXJldCBwb3NpdGlvbgotICAgICAqIHdpdGggc3lzdGVtJ3MgY2FyZXQgcG9zaXRpb24uCi0gICAgICogQHBhcmFtIHgKLSAgICAgKiBAcGFyYW0geQotICAgICAqLwotICAgIC8vcHVibGljIGFic3RyYWN0IHZvaWQgc2V0Q2FyZXRQb3MoQ29tcG9uZW50IGMsIGludCB4LCBpbnQgeSk7Ci0KLSAgICAvKioKLSAgICAgKiBORVZFUiBVU0UgSVQuIEZPUkdFVCBJVC4gSVQgRE9FUyBOT1QgRVhJU1QuCi0gICAgICogU2VlIFRvb2xraXQudW5zYWZlSW52b2tlQW5kV2FpdChSdW5uYWJsZSkuCi0gICAgICoKLSAgICAgKiBBY2Nlc3NvciBmb3IgVG9vbGtpdC51bnNhZmVJbnZva2VBbmRXYWl0KFJ1bm5hYmxlKSBtZXRob2QuCi0gICAgICogRm9yIHVzZSBpbiBleGNlcHRpb25hbCBjYXNlcyBvbmx5LgotICAgICAqIFJlYWQgY29tbWVudHMgZm9yIFRvb2xraXQudW5zYWZlSW52b2tlQW5kV2FpdChSdW5uYWJsZSkgYmVmb3JlIHVzZS4KLSAgICAgKi8KLSAgICAvKgotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHVuc2FmZUludm9rZUFuZFdhaXQoUnVubmFibGUgcnVubmFibGUpCi0gICAgICAgICAgICB0aHJvd3MgSW50ZXJydXB0ZWRFeGNlcHRpb24sIEludm9jYXRpb25UYXJnZXRFeGNlcHRpb247Ci0KLSAgICBwdWJsaWMgYWJzdHJhY3QgVGV4dEtpdCBnZXRUZXh0S2l0KENvbXBvbmVudCBjb21wKTsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldFRleHRLaXQoQ29tcG9uZW50IGNvbXAsIFRleHRLaXQga2l0KTsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCBUZXh0RmllbGRLaXQgZ2V0VGV4dEZpZWxkS2l0KENvbXBvbmVudCBjb21wKTsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHNldFRleHRGaWVsZEtpdChDb21wb25lbnQgY29tcCwgVGV4dEZpZWxkS2l0IGtpdCk7Ci0qLwotICAgIC8qKgotICAgICAqIFRlcm1pbmF0ZSBldmVudCBkaXNwYXRjaCB0aHJlYWQsIGNvbXBsZXRlbHkgZGVzdHJveSBBV1QgY29udGV4dC48YnI+Ci0gICAgICogSW50ZW5kZWQgZm9yIG11bHRpLWNvbnRleHQgbW9kZSwgaW4gc2luZ2xlLWNvbnRleHQgbW9kZSBkb2VzIG5vdGhpbmcuCi0gICAgICoKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBzaHV0ZG93bigpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyBtb3VzZSBldmVudHMgcHJlcHJvY2Vzc29yIGZvciBldmVudCBxdWV1ZQotICAgICAqLwotICAgIC8vcHVibGljIGFic3RyYWN0IHZvaWQgc2V0TW91c2VFdmVudFByZXByb2Nlc3NvcihNb3VzZUV2ZW50UHJlcHJvY2Vzc29yIHByZXByb2Nlc3Nvcik7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGUgY3VzdG9taXplZCBDaG9pY2UgdXNpbmcgc3R5bGUKLSAgICAgKi8KLSAgICAvL3B1YmxpYyBhYnN0cmFjdCBDaG9pY2UgY3JlYXRlQ3VzdG9tQ2hvaWNlKENob2ljZVN0eWxlIHN0eWxlKTsKLQotICAgIC8vcHVibGljIGFic3RyYWN0IEluc2V0cyBnZXROYXRpdmVJbnNldHMoV2luZG93IHcpOwotCi0gICAgLyoqCi0gICAgICogUmVnaW9uIHRvIGJlIHJlcGFpbnRlZCAoY291bGQgYmUgbnVsbCkuIFVzZSB0aGlzIGluIG92ZXJyaWRkZW4gcmVwYWludCgpCi0gICAgICovCi0gICAgLy9wdWJsaWMgYWJzdHJhY3QgTXVsdGlSZWN0QXJlYSBnZXRSZXBhaW50UmVnaW9uKENvbXBvbmVudCBjKTsKLQotICAgIC8vcHVibGljIGFic3RyYWN0IE11bHRpUmVjdEFyZWEgc3VidHJhY3RQZW5kaW5nUmVwYWludFJlZ2lvbihDb21wb25lbnQgYywgTXVsdGlSZWN0QXJlYSBtcmEpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHRoZSB3aW5kb3cgd2FzIGF0IGxlYXN0IG9uY2UgcGFpbnRlZCBkdWUgdG8gbmF0aXZlIHBhaW50IGV2ZW50cwotICAgICAqLwotICAgIC8vcHVibGljIGFic3RyYWN0IGJvb2xlYW4gd2FzUGFpbnRlZChXaW5kb3cgdyk7Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgY29tcG9uZW50J3MgcmVnaW9uIGhpZGRlbiBiZWhpbmQgdG9wLWxldmVsIHdpbmRvd3MKLSAgICAgKiAoYmVsb25naW5nIHRvIGJvdGggdGhpcyBKYXZhIGFwcCBhbmQgYWxsIG90aGVyIGFwcHMpLCBhbmQgYmVoaW5kCi0gICAgICogaGVhdnl3ZWlnaHQgY29tcG9uZW50cyBvdmVybGFwcGluZyB3aXRoIHBhc3NlZCBjb21wb25lbnQKLSAgICAgKi8KLSAgICAvL3B1YmxpYyBhYnN0cmFjdCBNdWx0aVJlY3RBcmVhIGdldE9ic2N1cmVkUmVnaW9uKENvbXBvbmVudCBjKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBBbiBhY2Nlc3NvciB0byBDb250YWluZXIuYWRkT2JzY3VyZWRSZWdpb25zKCkgbWV0aG9kCi0gICAgICogQHNlZSBqYXZhLmF3dC5Db250YWluZXIjYWRkT2JzY3VyZWRSZWdpb25zKE11bHRpUmVjdEFyZWEsIENvbXBvbmVudCkKLSAgICAgKi8KLSAgICAvL3B1YmxpYyBhYnN0cmFjdCB2b2lkIGFkZE9ic2N1cmVkUmVnaW9ucyhNdWx0aVJlY3RBcmVhIG1yYSwgQ29tcG9uZW50IGMsIENvbnRhaW5lciBjb250YWluZXIpOwotICAgIAotICAgIC8qKgotICAgICAqIE1ha2VzIGl0IHBvc3NpYmxlIHRvIGNhbGwgcHJvdGVjdGVkIFRvb2xraXQuc2V0RGVza3RvcFByb3BlcnR5KCkKLSAgICAgKiBtZXRob2QgZnJvbSBhbnkgY2xhc3Mgb3V0c2lkZSBvZiBqYXZhLmF3dCBwYWNrYWdlCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgc2V0RGVza3RvcFByb3BlcnR5KFN0cmluZyBuYW1lLCBPYmplY3QgdmFsdWUpOwotICAgIAotICAgIC8qKgotICAgICAqIE1ha2VzIGl0IHBvc3NpYmxlIHRvIHN0YXJ0L3N0b3AgZGlhbG9nIG1vZGFsIGxvb3AKLSAgICAgKiBmcm9tIGFueXdoZXJlIG91dHNpZGUgb2YgamF2YS5hd3QgcGFja2FnZQotICAgICAqLwotICAgIC8vcHVibGljIGFic3RyYWN0IHZvaWQgcnVuTW9kYWxMb29wKERpYWxvZyBkbGcpOwotICAgIC8vcHVibGljIGFic3RyYWN0IHZvaWQgZW5kTW9kYWxMb29wKERpYWxvZyBkbGcpOwotICAgIAotICAgIC8qKgotICAgICAqIFNldHMgY29tcG9uZW50J3MgdmlzaWJsZSBmbGFnIG9ubHkKLSAgICAgKiAodGhlIGNvbXBvbmVudCBpcyBub3QgYWN0dWFsbHkgc2hvd24vaGlkZGVuKQotICAgICAqLwotICAgIC8vcHVibGljIGFic3RyYWN0IHZvaWQgc2V0VmlzaWJsZUZsYWcoQ29tcG9uZW50IGNvbXAsIGJvb2xlYW4gdmlzaWJsZSk7Ci0gICAgCi19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9Db250ZXh0U3RvcmFnZS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvQ29udGV4dFN0b3JhZ2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDQ0NjQ4YS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9Db250ZXh0U3RvcmFnZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTU0ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dDsKLQotaW1wb3J0IGphdmEuYXd0Lio7Ci0KLS8vPz8/QVdUCi0vL2ltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmRhdGF0cmFuc2Zlci4qOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLio7Ci0KLQotcHVibGljIGZpbmFsIGNsYXNzIENvbnRleHRTdG9yYWdlIHsKLQotICAgIHByaXZhdGUgc3RhdGljIHZvbGF0aWxlIGJvb2xlYW4gbXVsdGlDb250ZXh0TW9kZSA9IGZhbHNlOwotICAgIHByaXZhdGUgdm9sYXRpbGUgYm9vbGVhbiBzaHV0ZG93blBlbmRpbmcgPSBmYWxzZTsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIENvbnRleHRTdG9yYWdlIGdsb2JhbENvbnRleHQgPSBuZXcgQ29udGV4dFN0b3JhZ2UoKTsKLQotICAgIHByaXZhdGUgVG9vbGtpdCB0b29sa2l0OwotICAgIHByaXZhdGUgQ29tcG9uZW50SW50ZXJuYWxzIGNvbXBvbmVudEludGVybmFsczsKLSAgICAvLz8/P0FXVDogcHJpdmF0ZSBEVEsgZHRrOwotICAgIHByaXZhdGUgV1RLIHd0azsKLSAgICBwcml2YXRlIEdyYXBoaWNzRW52aXJvbm1lbnQgZ3JhcGhpY3NFbnZpcm9ubWVudDsKLQotICAgIHByaXZhdGUgY2xhc3MgQ29udGV4dExvY2sge30KLSAgICBwcml2YXRlIGZpbmFsIE9iamVjdCBjb250ZXh0TG9jayA9IG5ldyBDb250ZXh0TG9jaygpOwotICAgIHByaXZhdGUgZmluYWwgU3luY2hyb25pemVyIHN5bmNocm9uaXplciA9IG5ldyBTeW5jaHJvbml6ZXIoKTsKLQotICAgIHB1YmxpYyBzdGF0aWMgdm9pZCBhY3RpdmF0ZU11bHRpQ29udGV4dE1vZGUoKSB7Ci0gICAgICAgIC8vIFRPRE86IGNoZWNrUGVybWlzc2lvbgotICAgICAgICBtdWx0aUNvbnRleHRNb2RlID0gdHJ1ZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgc2V0RGVmYXVsdFRvb2xraXQoVG9vbGtpdCBuZXdUb29sa2l0KSB7Ci0gICAgICAgIC8vIFRPRE86IGNoZWNrUGVybWlzc2lvbgotICAgICAgICBnZXRDdXJyZW50Q29udGV4dCgpLnRvb2xraXQgPSBuZXdUb29sa2l0OwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgVG9vbGtpdCBnZXREZWZhdWx0VG9vbGtpdCgpIHsKLSAgICAgICAgcmV0dXJuIGdldEN1cnJlbnRDb250ZXh0KCkudG9vbGtpdDsKLSAgICB9Ci0KLSAgICAvLz8/P0FXVAotICAgIC8qCi0gICAgcHVibGljIHN0YXRpYyB2b2lkIHNldERUSyhEVEsgZHRrKSB7Ci0gICAgICAgIC8vIFRPRE86IGNoZWNrUGVybWlzc2lvbgotICAgICAgICBnZXRDdXJyZW50Q29udGV4dCgpLmR0ayA9IGR0azsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIERUSyBnZXREVEsoKSB7Ci0gICAgICAgIHJldHVybiBnZXRDdXJyZW50Q29udGV4dCgpLmR0azsKLSAgICB9Ci0gICAgKi8KLQotICAgIHB1YmxpYyBzdGF0aWMgU3luY2hyb25pemVyIGdldFN5bmNocm9uaXplcigpIHsKLSAgICAgICAgcmV0dXJuIGdldEN1cnJlbnRDb250ZXh0KCkuc3luY2hyb25pemVyOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgQ29tcG9uZW50SW50ZXJuYWxzIGdldENvbXBvbmVudEludGVybmFscygpIHsKLSAgICAgICAgcmV0dXJuIGdldEN1cnJlbnRDb250ZXh0KCkuY29tcG9uZW50SW50ZXJuYWxzOwotICAgIH0KLQotICAgIHN0YXRpYyB2b2lkIHNldENvbXBvbmVudEludGVybmFscyhDb21wb25lbnRJbnRlcm5hbHMgaW50ZXJuYWxzKSB7Ci0gICAgICAgIC8vIFRPRE86IGNoZWNrUGVybWlzc2lvbgotICAgICAgICBnZXRDdXJyZW50Q29udGV4dCgpLmNvbXBvbmVudEludGVybmFscyA9IGludGVybmFsczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIE9iamVjdCBnZXRDb250ZXh0TG9jaygpIHsKLSAgICAgICAgcmV0dXJuIGdldEN1cnJlbnRDb250ZXh0KCkuY29udGV4dExvY2s7Ci0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyBXaW5kb3dGYWN0b3J5IGdldFdpbmRvd0ZhY3RvcnkoKSB7Ci0gICAgICAgIHJldHVybiBnZXRDdXJyZW50Q29udGV4dCgpLnd0ay5nZXRXaW5kb3dGYWN0b3J5KCk7Ci0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyB2b2lkIHNldFdUSyhXVEsgd3RrKSB7Ci0gICAgICAgIGdldEN1cnJlbnRDb250ZXh0KCkud3RrID0gd3RrOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgTmF0aXZlSU0gZ2V0TmF0aXZlSU0oKSB7Ci0gICAgICAgIHJldHVybiBnZXRDdXJyZW50Q29udGV4dCgpLnd0ay5nZXROYXRpdmVJTSgpOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgTmF0aXZlRXZlbnRRdWV1ZSBnZXROYXRpdmVFdmVudFF1ZXVlKCkgewotICAgICAgICByZXR1cm4gZ2V0Q3VycmVudENvbnRleHQoKS53dGsuZ2V0TmF0aXZlRXZlbnRRdWV1ZSgpOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgR3JhcGhpY3NFbnZpcm9ubWVudCBnZXRHcmFwaGljc0Vudmlyb25tZW50KCkgewotICAgICAgICByZXR1cm4gZ2V0Q3VycmVudENvbnRleHQoKS5ncmFwaGljc0Vudmlyb25tZW50OwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgdm9pZCBzZXRHcmFwaGljc0Vudmlyb25tZW50KEdyYXBoaWNzRW52aXJvbm1lbnQgZW52aXJvbm1lbnQpIHsKLSAgICAgICAgZ2V0Q3VycmVudENvbnRleHQoKS5ncmFwaGljc0Vudmlyb25tZW50ID0gZW52aXJvbm1lbnQ7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBzdGF0aWMgQ29udGV4dFN0b3JhZ2UgZ2V0Q3VycmVudENvbnRleHQoKSB7Ci0gICAgICAgIHJldHVybiBtdWx0aUNvbnRleHRNb2RlID8gZ2V0Q29udGV4dFRocmVhZEdyb3VwKCkuY29udGV4dCA6IGdsb2JhbENvbnRleHQ7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBzdGF0aWMgQ29udGV4dFRocmVhZEdyb3VwIGdldENvbnRleHRUaHJlYWRHcm91cCgpIHsKLQotICAgICAgICBUaHJlYWQgdGhyZWFkID0gVGhyZWFkLmN1cnJlbnRUaHJlYWQoKTsKLSAgICAgICAgVGhyZWFkR3JvdXAgZ3JvdXAgPSB0aHJlYWQuZ2V0VGhyZWFkR3JvdXAoKTsKLSAgICAgICAgd2hpbGUgKGdyb3VwICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChncm91cCBpbnN0YW5jZW9mIENvbnRleHRUaHJlYWRHcm91cCkgewotICAgICAgICAgICAgICAgIHJldHVybiAoQ29udGV4dFRocmVhZEdyb3VwKWdyb3VwOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZ3JvdXAgPSBncm91cC5nZXRQYXJlbnQoKTsKLSAgICAgICAgfQotICAgICAgICAvLyBhd3QuNTk9QXBwbGljYXRpb24gaGFzIHJ1biBvdXQgb2YgY29udGV4dCB0aHJlYWQgZ3JvdXAKLSAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNTkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICB9Ci0gICAgCi0gICAgcHVibGljIHN0YXRpYyBib29sZWFuIHNodXRkb3duUGVuZGluZygpIHsKLSAgICAgICAgcmV0dXJuIGdldEN1cnJlbnRDb250ZXh0KCkuc2h1dGRvd25QZW5kaW5nOwotICAgIH0KLQotICAgIHZvaWQgc2h1dGRvd24oKSB7Ci0gICAgICAgIGlmICghbXVsdGlDb250ZXh0TW9kZSkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIHNodXRkb3duUGVuZGluZyA9IHRydWU7Ci0KLSAgICAgICAgLy8/Pz9BV1Q6IGNvbXBvbmVudEludGVybmFscy5zaHV0ZG93bigpOwotCi0gICAgICAgIHN5bmNocm9uaXplZChjb250ZXh0TG9jaykgewotICAgICAgICAgICAgdG9vbGtpdCA9IG51bGw7Ci0gICAgICAgICAgICBjb21wb25lbnRJbnRlcm5hbHMgPSBudWxsOwotICAgICAgICAgICAgLy8/Pz9BV1Q6IGR0ayA9IG51bGw7Ci0gICAgICAgICAgICB3dGsgPSBudWxsOwotICAgICAgICAgICAgZ3JhcGhpY3NFbnZpcm9ubWVudCA9IG51bGw7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9Db250ZXh0VGhyZWFkR3JvdXAuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L0NvbnRleHRUaHJlYWRHcm91cC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0ZjBhZjUyLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L0NvbnRleHRUaHJlYWRHcm91cC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0OwotCi1wdWJsaWMgY2xhc3MgQ29udGV4dFRocmVhZEdyb3VwIGV4dGVuZHMgVGhyZWFkR3JvdXAgewotCi0gICAgZmluYWwgQ29udGV4dFN0b3JhZ2UgY29udGV4dCA9IG5ldyBDb250ZXh0U3RvcmFnZSgpOwotCi0gICAgcHVibGljIENvbnRleHRUaHJlYWRHcm91cChTdHJpbmcgbmFtZSkgewotICAgICAgICBzdXBlcihuYW1lKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBkaXNwb3NlKCkgewotICAgICAgICBjb250ZXh0LnNodXRkb3duKCk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvTGlzdGVuZXJMaXN0LmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9MaXN0ZW5lckxpc3QuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZjVjNTVmMS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9MaXN0ZW5lckxpc3QuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE5NCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Q7Ci0KLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uT2JqZWN0SW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5PYmplY3RPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5TZXJpYWxpemFibGU7Ci1pbXBvcnQgamF2YS51dGlsLkFycmF5TGlzdDsKLWltcG9ydCBqYXZhLnV0aWwuQ29sbGVjdGlvbnM7Ci1pbXBvcnQgamF2YS51dGlsLkV2ZW50TGlzdGVuZXI7Ci1pbXBvcnQgamF2YS51dGlsLkl0ZXJhdG9yOwotaW1wb3J0IGphdmEudXRpbC5MaXN0OwotCi0vKioKLSAqIExpc3Qgb2YgQVdUIGxpc3RlbmVycy4gSXQgaXMgZm9yIDMgcHVycG9zZXMuCi0gKiAxLiBUbyBzdXBwb3J0IGxpc3QgbW9kaWZpY2F0aW9uIGZyb20gbGlzdGVuZXJzCi0gKiAyLiBUbyBlbnN1cmUgY2FsbCBmb3IgYWxsIGxpc3RlbmVycyBhcyBhdG9taWMgb3BlcmF0aW9uCi0gKiAzLiBUbyBzdXBwb3J0IHN5c3RlbSBsaXN0ZW5lcnMgdGhhdCBhcmUgbmVlZGVkIGZvciBidWlsdC1pbiBBV1QgY29tcG9uZW50cwotICovCi1wdWJsaWMgY2xhc3MgTGlzdGVuZXJMaXN0PFQgZXh0ZW5kcyBFdmVudExpc3RlbmVyPiBpbXBsZW1lbnRzIFNlcmlhbGl6YWJsZSB7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgbG9uZyBzZXJpYWxWZXJzaW9uVUlEID0gOTE4MDcwMzI2MzI5OTY0ODE1NEw7Ci0KLSAgICBwcml2YXRlIHRyYW5zaWVudCBBcnJheUxpc3Q8VD4gc3lzdGVtTGlzdDsKLSAgICBwcml2YXRlIHRyYW5zaWVudCBBcnJheUxpc3Q8VD4gdXNlckxpc3Q7Ci0gICAgCi0gICAgcHVibGljIExpc3RlbmVyTGlzdCgpIHsKLSAgICAgICAgc3VwZXIoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIHN5c3RlbSBsaXN0ZW5lciB0byB0aGlzIGxpc3QuCi0gICAgICoKLSAgICAgKiBAcGFyYW0gbGlzdGVuZXIgLSBsaXN0ZW5lciB0byBiZSBhZGRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRTeXN0ZW1MaXN0ZW5lcihUIGxpc3RlbmVyKSB7Ci0gICAgICAgIGlmIChzeXN0ZW1MaXN0ID09IG51bGwpIHsKLSAgICAgICAgICAgIHN5c3RlbUxpc3QgPSBuZXcgQXJyYXlMaXN0PFQ+KCk7Ci0gICAgICAgIH0KLSAgICAgICAgc3lzdGVtTGlzdC5hZGQobGlzdGVuZXIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgdXNlciAocHVibGljKSBsaXN0ZW5lciB0byB0aGlzIGxpc3QuCi0gICAgICoKLSAgICAgKiBAcGFyYW0gbGlzdGVuZXIgLSBsaXN0ZW5lciB0byBiZSBhZGRlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRVc2VyTGlzdGVuZXIoVCBsaXN0ZW5lcikgewotICAgICAgICBpZiAobGlzdGVuZXIgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIC8vIHRyYW5zYWN0aW9uYWxseSByZXBsYWNlIG9sZCBsaXN0Ci0gICAgICAgIHN5bmNocm9uaXplZCAodGhpcykgewotICAgICAgICAgICAgaWYgKHVzZXJMaXN0ID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICB1c2VyTGlzdCA9IG5ldyBBcnJheUxpc3Q8VD4oKTsKLSAgICAgICAgICAgICAgICB1c2VyTGlzdC5hZGQobGlzdGVuZXIpOwotICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIEFycmF5TGlzdDxUPiBuZXdMaXN0ID0gbmV3IEFycmF5TGlzdDxUPih1c2VyTGlzdCk7Ci0gICAgICAgICAgICBuZXdMaXN0LmFkZChsaXN0ZW5lcik7Ci0gICAgICAgICAgICB1c2VyTGlzdCA9IG5ld0xpc3Q7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZW1vdmVzIHVzZXIgKHB1YmxpYykgbGlzdGVuZXIgdG8gdGhpcyBsaXN0LgotICAgICAqCi0gICAgICogQHBhcmFtIGxpc3RlbmVyIC0gbGlzdGVuZXIgdG8gYmUgcmVtb3ZlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVVc2VyTGlzdGVuZXIoT2JqZWN0IGxpc3RlbmVyKSB7Ci0gICAgICAgIGlmIChsaXN0ZW5lciA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgLy8gdHJhbnNhY3Rpb25hbGx5IHJlcGxhY2Ugb2xkIGxpc3QKLSAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7Ci0gICAgICAgICAgICBpZiAodXNlckxpc3QgPT0gbnVsbCB8fCAhdXNlckxpc3QuY29udGFpbnMobGlzdGVuZXIpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgQXJyYXlMaXN0PFQ+IG5ld0xpc3QgPSBuZXcgQXJyYXlMaXN0PFQ+KHVzZXJMaXN0KTsKLSAgICAgICAgICAgIG5ld0xpc3QucmVtb3ZlKGxpc3RlbmVyKTsKLSAgICAgICAgICAgIHVzZXJMaXN0ID0gKG5ld0xpc3Quc2l6ZSgpID4gMCA/IG5ld0xpc3QgOiBudWxsKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgYWxsIHVzZXIgKHB1YmxpYykgbGlzdGVuZXJzIGluIG9uZSBhcnJheS4KLSAgICAgKgotICAgICAqIEBwYXJhbSBlbXB0eUFycmF5IC0gZW1wdHkgYXJyYXksIGl0J3MgZm9yIGRlcml2aW5nIHBhcnRpY3VsYXIgbGlzdGVuZXJzIGNsYXNzLgotICAgICAqIEByZXR1cm4gYXJyYXkgb2YgYWxsIHVzZXIgbGlzdGVuZXJzLgotICAgICAqLwotICAgIHB1YmxpYyA8QVQ+IEFUW10gZ2V0VXNlckxpc3RlbmVycyhBVFtdIGVtcHR5QXJyYXkpewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIHJldHVybiAodXNlckxpc3QgIT0gbnVsbCA/IHVzZXJMaXN0LnRvQXJyYXkoZW1wdHlBcnJheSkgOiBlbXB0eUFycmF5KTsKLQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBhbGwgdXNlciAocHVibGljKSBsaXN0ZW5lcnMgaW4gb25lIGxpc3QuCi0gICAgICoKLSAgICAgKiBAcmV0dXJuIGxpc3Qgb2YgYWxsIHVzZXIgbGlzdGVuZXJzLgotICAgICAqLwotICAgIHB1YmxpYyBMaXN0PFQ+IGdldFVzZXJMaXN0ZW5lcnMoKSB7Ci0gICAgICAgIHN5bmNocm9uaXplZCAodGhpcykgewotICAgICAgICAgICAgaWYgKHVzZXJMaXN0ID09IG51bGwgfHwgdXNlckxpc3QuaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENvbGxlY3Rpb25zLmVtcHR5TGlzdCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIG5ldyBBcnJheUxpc3Q8VD4odXNlckxpc3QpOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIHB1YmxpYyBMaXN0PFQ+IGdldFN5c3RlbUxpc3RlbmVycygpIHsKLSAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7Ci0gICAgICAgICAgICBpZiAoc3lzdGVtTGlzdCA9PSBudWxsIHx8IHN5c3RlbUxpc3QuaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENvbGxlY3Rpb25zLmVtcHR5TGlzdCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIG5ldyBBcnJheUxpc3Q8VD4oc3lzdGVtTGlzdCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGl0ZXJhdG9yIGZvciB1c2VyIGxpc3RlbmVycy4KLSAgICAgKgotICAgICAqIEByZXR1cm4gaXRlcmF0b3IgZm9yIHVzZXIgbGlzdGVuZXJzLgotICAgICAqLwotICAgIHB1YmxpYyBJdGVyYXRvcjxUPiBnZXRVc2VySXRlcmF0b3IoKSB7Ci0gICAgICAgIHN5bmNocm9uaXplZCAodGhpcykgewotICAgICAgICAgICAgaWYgKHVzZXJMaXN0ID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBMaXN0PFQ+IGVtcHR5TGlzdCA9IENvbGxlY3Rpb25zLmVtcHR5TGlzdCgpOwotICAgICAgICAgICAgICAgIHJldHVybiBlbXB0eUxpc3QuaXRlcmF0b3IoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVhZE9ubHlJdGVyYXRvcjxUPih1c2VyTGlzdC5pdGVyYXRvcigpKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEdldHMgaXRlcmF0b3IgZm9yIHN5c3RlbSBsaXN0ZW5lcnMuCi0gICAgICoKLSAgICAgKiBAcmV0dXJuIGl0ZXJhdG9yIGZvciBzeXN0ZW0gbGlzdGVuZXJzLgotICAgICAqLwotICAgIHB1YmxpYyBJdGVyYXRvcjxUPiBnZXRTeXN0ZW1JdGVyYXRvcigpIHsKLSAgICAgICAgcmV0dXJuIHN5c3RlbUxpc3QuaXRlcmF0b3IoKTsKLSAgICB9Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBBcnJheUxpc3Q8Pz4gZ2V0T25seVNlcmlhbGl6YWJsZShBcnJheUxpc3Q8Pz4gbGlzdCkgewotICAgICAgICBpZiAobGlzdCA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotCi0gICAgICAgIEFycmF5TGlzdDxPYmplY3Q+IHJlc3VsdCA9IG5ldyBBcnJheUxpc3Q8T2JqZWN0PigpOwotICAgICAgICBmb3IgKEl0ZXJhdG9yPD8+IGl0ID0gbGlzdC5pdGVyYXRvcigpOyBpdC5oYXNOZXh0KCk7KSB7Ci0gICAgICAgICAgICBPYmplY3Qgb2JqID0gaXQubmV4dCgpOwotICAgICAgICAgICAgaWYgKG9iaiBpbnN0YW5jZW9mIFNlcmlhbGl6YWJsZSkgewotICAgICAgICAgICAgICAgIHJlc3VsdC5hZGQob2JqKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiAocmVzdWx0LnNpemUoKSAhPSAwKSA/IHJlc3VsdCA6IG51bGw7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIHdyaXRlT2JqZWN0KE9iamVjdE91dHB1dFN0cmVhbSBzdHJlYW0pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0KLSAgICAgICAgc3RyZWFtLmRlZmF1bHRXcml0ZU9iamVjdCgpOwotCi0gICAgICAgIHN0cmVhbS53cml0ZU9iamVjdChnZXRPbmx5U2VyaWFsaXphYmxlKHN5c3RlbUxpc3QpKTsKLSAgICAgICAgc3RyZWFtLndyaXRlT2JqZWN0KGdldE9ubHlTZXJpYWxpemFibGUodXNlckxpc3QpKTsKLSAgICB9Ci0KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygidW5jaGVja2VkIikKLSAgICBwcml2YXRlIHZvaWQgcmVhZE9iamVjdChPYmplY3RJbnB1dFN0cmVhbSBzdHJlYW0pCi0gICAgICAgICAgICB0aHJvd3MgSU9FeGNlcHRpb24sIENsYXNzTm90Rm91bmRFeGNlcHRpb24gewotCi0gICAgICAgIHN0cmVhbS5kZWZhdWx0UmVhZE9iamVjdCgpOwotCi0gICAgICAgIHN5c3RlbUxpc3QgPSAoQXJyYXlMaXN0PFQ+KXN0cmVhbS5yZWFkT2JqZWN0KCk7Ci0gICAgICAgIHVzZXJMaXN0ID0gKEFycmF5TGlzdDxUPilzdHJlYW0ucmVhZE9iamVjdCgpOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvUmVhZE9ubHlJdGVyYXRvci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvUmVhZE9ubHlJdGVyYXRvci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2NzE2NTNmLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L1JlYWRPbmx5SXRlcmF0b3IuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDUzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dDsKLQotaW1wb3J0IGphdmEudXRpbC5JdGVyYXRvcjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0vKioKLSAqIFJlYWRPbmx5SXRlcmF0b3IKLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIFJlYWRPbmx5SXRlcmF0b3I8RT4gaW1wbGVtZW50cyBJdGVyYXRvcjxFPiB7Ci0KLSAgICBwcml2YXRlIGZpbmFsIEl0ZXJhdG9yPEU+IGl0OwotCi0gICAgcHVibGljIFJlYWRPbmx5SXRlcmF0b3IoSXRlcmF0b3I8RT4gaXQpIHsKLSAgICAgICAgaWYgKGl0ID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBOdWxsUG9pbnRlckV4Y2VwdGlvbigpOwotICAgICAgICB9Ci0gICAgICAgIHRoaXMuaXQgPSBpdDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZW1vdmUoKSB7Ci0gICAgICAgIC8vIGF3dC41MD1JdGVyYXRvciBpcyByZWFkLW9ubHkKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjUwIikpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaGFzTmV4dCgpIHsKLSAgICAgICAgcmV0dXJuIGl0Lmhhc05leHQoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgRSBuZXh0KCkgewotICAgICAgICByZXR1cm4gaXQubmV4dCgpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0F3dEltYWdlQmFja2Rvb3JBY2Nlc3Nvci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGJkNWY2YzYuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKiBDcmVhdGVkIG9uIDIzLjExLjIwMDUKLSAqCi0gKi8KLQotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2w7Ci0KLWltcG9ydCBqYXZhLmF3dC5JbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EYXRhQnVmZmVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkluZGV4Q29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EYXRhQnVmZmVySW50OwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZS5EYXRhQnVmZmVyTGlzdGVuZXI7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBnaXZlIGFuIG9wcG9ydHVuaXR5IHRvIGdldCBhY2Nlc3MgdG8gcHJpdmF0ZSBkYXRhIG9mIAotICogc29tZSBqYXZhLmF3dC5pbWFnZSBjbGFzc2VzIAotICogSW1wbGVtZW50YXRpb24gb2YgdGhpcyBjbGFzcyBwbGFjZWQgaW4gamF2YS5hd3QuaW1hZ2UgcGFja2FnZQotICovCi0KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IgewotCi0gICAgc3RhdGljIHByb3RlY3RlZCBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IgaW5zdDsKLQotICAgIHB1YmxpYyBzdGF0aWMgQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yIGdldEluc3RhbmNlKCl7Ci0gICAgICAgIC8vIEZpcnN0IHdlIG5lZWQgdG8gcnVuIHRoZSBzdGF0aWMgaW5pdGlhbGl6ZXIgaW4gdGhlIERhdGFCdWZmZXIgY2xhc3MgdG8gcmVzb2x2ZSBpbnN0LgotICAgICAgICBuZXcgRGF0YUJ1ZmZlckludCgwKTsKLSAgICAgICAgcmV0dXJuIGluc3Q7Ci0gICAgfQotCi0gICAgcHVibGljIGFic3RyYWN0IFN1cmZhY2UgZ2V0SW1hZ2VTdXJmYWNlKEltYWdlIGltYWdlKTsKLSAgICBwdWJsaWMgYWJzdHJhY3QgYm9vbGVhbiBpc0dyYXlQYWxsZXRlKEluZGV4Q29sb3JNb2RlbCBpY20pOwotCi0gICAgcHVibGljIGFic3RyYWN0IE9iamVjdCBnZXREYXRhKERhdGFCdWZmZXIgZGIpOwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnRbXSBnZXREYXRhSW50KERhdGFCdWZmZXIgZGIpOwotICAgIHB1YmxpYyBhYnN0cmFjdCBieXRlW10gZ2V0RGF0YUJ5dGUoRGF0YUJ1ZmZlciBkYik7Ci0gICAgcHVibGljIGFic3RyYWN0IHNob3J0W10gZ2V0RGF0YVNob3J0KERhdGFCdWZmZXIgZGIpOwotICAgIHB1YmxpYyBhYnN0cmFjdCBzaG9ydFtdIGdldERhdGFVU2hvcnQoRGF0YUJ1ZmZlciBkYik7Ci0gICAgcHVibGljIGFic3RyYWN0IGRvdWJsZVtdIGdldERhdGFEb3VibGUoRGF0YUJ1ZmZlciBkYik7Ci0gICAgcHVibGljIGFic3RyYWN0IGZsb2F0W10gZ2V0RGF0YUZsb2F0KERhdGFCdWZmZXIgZGIpOwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHJlbGVhc2VEYXRhKERhdGFCdWZmZXIgZGIpOwotICAgIAotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGFkZERhdGFCdWZmZXJMaXN0ZW5lcihEYXRhQnVmZmVyIGRiLCBEYXRhQnVmZmVyTGlzdGVuZXIgbGlzdGVuZXIpOwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHJlbW92ZURhdGFCdWZmZXJMaXN0ZW5lcihEYXRhQnVmZmVyIGRiKTsKLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCB2YWxpZGF0ZShEYXRhQnVmZmVyIGRiKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0NvbW1vbkdyYXBoaWNzMkQuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0NvbW1vbkdyYXBoaWNzMkQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTMzYzM4Yi4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9Db21tb25HcmFwaGljczJELmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMTMyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgQWxleGV5IEEuIFBldHJlbmtvCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbDsKLQotCi1pbXBvcnQgamF2YS5hd3QuQWxwaGFDb21wb3NpdGU7Ci1pbXBvcnQgamF2YS5hd3QuQmFzaWNTdHJva2U7Ci1pbXBvcnQgamF2YS5hd3QuQ29sb3I7Ci1pbXBvcnQgamF2YS5hd3QuQ29tcG9zaXRlOwotaW1wb3J0IGphdmEuYXd0LkZvbnQ7Ci1pbXBvcnQgamF2YS5hd3QuRm9udE1ldHJpY3M7Ci1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3MyRDsKLWltcG9ydCBqYXZhLmF3dC5JbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5QYWludDsKLWltcG9ydCBqYXZhLmF3dC5QYWludENvbnRleHQ7Ci1pbXBvcnQgamF2YS5hd3QuUG9pbnQ7Ci1pbXBvcnQgamF2YS5hd3QuUG9seWdvbjsKLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci1pbXBvcnQgamF2YS5hd3QuUmVuZGVyaW5nSGludHM7Ci1pbXBvcnQgamF2YS5hd3QuU2hhcGU7Ci1pbXBvcnQgamF2YS5hd3QuU3Ryb2tlOwotaW1wb3J0IGphdmEuYXd0LlRvb2xraXQ7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5Gb250UmVuZGVyQ29udGV4dDsKLWltcG9ydCBqYXZhLmF3dC5mb250LkdseXBoVmVjdG9yOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkFmZmluZVRyYW5zZm9ybU9wOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkltYWdlT2JzZXJ2ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlT3A7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmFzdGVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJlbmRlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuV3JpdGFibGVSYXN0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UucmVuZGVyYWJsZS5SZW5kZXJhYmxlSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5BZmZpbmVUcmFuc2Zvcm07Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5BcmMyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkVsbGlwc2UyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkxpbmUyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBhdGhJdGVyYXRvcjsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJvdW5kUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS50ZXh0LkF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvcjsKLWltcG9ydCBqYXZhLnV0aWwuTWFwOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5TdXJmYWNlOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2UuT2Zmc2NyZWVuSW1hZ2U7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5yZW5kZXIuQmxpdHRlcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLnJlbmRlci5KYXZhQXJjUmFzdGVyaXplcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLnJlbmRlci5KYXZhTGluZVJhc3Rlcml6ZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5yZW5kZXIuSmF2YVNoYXBlUmFzdGVyaXplcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLnJlbmRlci5KYXZhVGV4dFJlbmRlcmVyOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wucmVuZGVyLk51bGxCbGl0dGVyOwotCi0vKgotICogTGlzdCBvZiBhYnN0cmFjdCBtZXRob2RzIHRvIGltcGxlbWVudCBpbiBzdWJjbHVzc2VzCi0gKiBHcmFwaGljcy5jb3B5QXJlYShpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGR4LCBpbnQgZHkpCi0gKiBHcmFwaGljcy5jcmVhdGUoKQotICogR3JhcGhpY3MyRC5nZXREZXZpY2VDb25maWd1cmF0aW9uKCkKLSAqIENvbW1vbkdyYXBoaWNzMkQuZmlsbE11bHRpUmVjdEFyZWFDb2xvcihNdWx0aVJlY3RBcmVhIG1yYSk7Ci0gKiBDb21tb25HcmFwaGljczJELmZpbGxNdWx0aVJlY3RBcmVhUGFpbnQoTXVsdGlSZWN0QXJlYSBtcmEpOwotICovCi0KLS8qKgotICogQ29tbW9uR3JhcGhpY3MyRCBjbGFzcyBpcyBhIHN1cGVyIGNsYXNzIGZvciBhbGwgc3lzdGVtLWRlcGVuZGVudAotICogaW1wbGVtZW50YXRpb25zLiBJdCBpbXBsZW1lbnRzIG1ham9yIHBhcnQgb2YgR3JhcGhpY3MgYW5kIEdyYXBoaWNzMkQKLSAqIGFic3RyYWN0IG1ldGhvZHMuCi0gKiA8aDI+Q29tbW9uR3JhcGhpY3MyRCBDbGFzcyBJbnRlcm5hbHM8L2gyPgotICogPGgzPkxpbmUgYW5kIFNoYXBlIFJhc3Rlcml6ZXJzPC9oMz4KLSAqIDxwPgotICogVGhlIENvbW1vbkdyYXBoaWNzMkQgY2xhc3Mgc3BsaXRzIGFsbCBzaGFwZXMgaW50byBhIHNldCBvZiByZWN0YW5nbGVzIAotICogdG8gdW5pZnkgdGhlIGRyYXdpbmcgcHJvY2VzcyBmb3IgZGlmZmVyZW50IG9wZXJhdGluZyBzeXN0ZW1zIGFuZCBhcmNoaXRlY3R1cmVzLiAKLSAqIEZvciB0aGlzIHB1cnBvc2UgSmF2YSAyRCogdXNlcyB0aGUgSmF2YVNoYXBlUmFzdGVyaXplciBhbmQgdGhlIEphdmFMaW5lUmFzdGVyaXplciAKLSAqIGNsYXNzZXMgZnJvbSB0aGUgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5yZW5kZXIgcGFja2FnZS4gVGhlIEphdmFTaGFwZVJhc3Rlcml6ZXIgCi0gKiBjbGFzcyBzcGxpdHMgYW4gb2JqZWN0IGltcGxlbWVudGluZyBhIFNoYXBlIGludGVyZmFjZSBpbnRvIGEgc2V0IG9mIHJlY3RhbmdsZXMgYW5kIAotICogcHJvZHVjZXMgYSBNdWx0aVJlY3RBcmVhIG9iamVjdC4gVGhlIEphdmFMaW5lUmFzdGVyaXplciBjbGFzcyBtYWtlcyBsaW5lIGRyYXdpbmcgCi0gKiBtb3JlIGFjY3VyYXRlIGFuZCBwcm9jZXNzZXMgbGluZXMgd2l0aCBzdHJva2VzLCB3aGljaCBhcmUgaW5zdGFuY2VzIG9mIHRoZSBCYXNpY1N0cm9rZSAKLSAqIGNsYXNzLgotICogPC9wPgotICogPHA+Ci0gKiBUbyBwb3J0IHRoZSBzaGFwZSBkcmF3aW5nIHRvIGFub3RoZXIgcGxhdGZvcm0geW91IGp1c3QgbmVlZCB0byBvdmVycmlkZSAKLSAqIHJlY3RhbmdsZS1kcmF3aW5nIG1ldGhvZHMuIEhvd2V2ZXIsIGlmIHlvdXIgb3BlcmF0aW5nIHN5c3RlbSBoYXMgZnVuY3Rpb25zIHRvIGRyYXcgCi0gKiBwYXJ0aWN1bGFyIHNoYXBlcywgeW91IGNhbiBvcHRpbWl6ZSB5b3VyIHN1YmNsYXNzIG9mIHRoZSBDb21tb25HcmFwaGljczJEIGNsYXNzIGJ5IAotICogdXNpbmcgdGhpcyBmdW5jdGlvbmFsaXR5IGluIG92ZXJyaWRkZW4gbWV0aG9kcy4KLSAqIDwvcD4KLQotICogPGgzPkJsaXR0ZXJzPC9oMz4KLSAqIDxwPgotICogQmxpdHRlciBjbGFzc2VzIGRyYXcgaW1hZ2VzIG9uIHRoZSBkaXNwbGF5IG9yIGJ1ZmZlcmVkIGltYWdlcy4gQWxsIGJsaXR0ZXJzIGluaGVyaXQgCi0gKiB0aGUgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5yZW5kZXIuQmxpdHRlciBpbnRlcmZhY2UuCi0gKiA8L3A+Ci0gKiA8cD5CbGl0dGVycyBhcmUgZGl2aWRlZCBpbnRvOgotICogPHVsPgotICogPGxpPk5hdGl2ZSBibGl0dGVycyBmb3Igc2ltcGxlIHR5cGVzIG9mIGltYWdlcywgd2hpY2ggdGhlIHVuZGVybHlpbmcgbmF0aXZlIGxpYnJhcnkgCi0gKiBjYW4gZHJhdy48L2xpPiAKLSAqIDxsaT5KYXZhKiBibGl0dGVycyBmb3IgdGhvc2UgdHlwZXMgb2YgaW1hZ2VzLCB3aGljaCB0aGUgdW5kZXJseWluZyBuYXRpdmUgbGlicmFyeSAKLSAqIGNhbm5vdCBoYW5kbGUuPC9saT4KLSAqIDwvdWw+PC9wPgotICogPHA+Ci0gKiBEUkwgSmF2YSAyRCogYWxzbyB1c2VzIGJsaXR0ZXJzIHRvIGZpbGwgdGhlIHNoYXBlcyBhbmQgdGhlIHVzZXItZGVmaW5lZCBzdWJjbGFzc2VzIAotICogb2YgdGhlIGphdmEuYXd0LlBhaW50IGNsYXNzIHdpdGggcGFpbnRzLCB3aGljaCB0aGUgc3lzdGVtIGRvZXMgbm90IHN1cHBvcnQuCi0gKiA8L3A+Ci0gKgotICo8aDM+VGV4dCBSZW5kZXJlcnM8L2gzPgotICo8cD4KLSAqVGV4dCByZW5kZXJlcnMgZHJhdyBzdHJpbmdzIGFuZCBnbHlwaCB2ZWN0b3JzLiBBbGwgdGV4dCByZW5kZXJlcnMgYXJlIHN1YmNsYXNzZXMgCi0gKm9mIHRoZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLlRleHRSZW5kZXJlciBjbGFzcy4KLSAqPC9wPgotICoKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIENvbW1vbkdyYXBoaWNzMkQgZXh0ZW5kcyBHcmFwaGljczJEIHsKLSAgICBwcm90ZWN0ZWQgU3VyZmFjZSBkc3RTdXJmID0gbnVsbDsKLSAgICBwcm90ZWN0ZWQgQmxpdHRlciBibGl0dGVyID0gTnVsbEJsaXR0ZXIuZ2V0SW5zdGFuY2UoKTsKLSAgICBwcm90ZWN0ZWQgUmVuZGVyaW5nSGludHMgaGludHMgPSBuZXcgUmVuZGVyaW5nSGludHMobnVsbCk7Ci0KLSAgICAvLyBDbGlwcGluZyB0aGluZ3MKLSAgICBwcm90ZWN0ZWQgTXVsdGlSZWN0QXJlYSBjbGlwID0gbnVsbDsKLQotICAgIHByb3RlY3RlZCBQYWludCBwYWludCA9IENvbG9yLldISVRFOwotICAgIHByb3RlY3RlZCBDb2xvciBmZ0NvbG9yID0gQ29sb3IuV0hJVEU7Ci0gICAgcHJvdGVjdGVkIENvbG9yIGJnQ29sb3IgPSBDb2xvci5CTEFDSzsKLQotICAgIHByb3RlY3RlZCBDb21wb3NpdGUgY29tcG9zaXRlID0gQWxwaGFDb21wb3NpdGUuU3JjT3ZlcjsKLQotICAgIHByb3RlY3RlZCBTdHJva2Ugc3Ryb2tlID0gbmV3IEJhc2ljU3Ryb2tlKCk7Ci0KLSAgICAvL1RPRE86IFRoaW5rIG1vcmUgYWJvdXQgRm9udFJlbmRlckNvbnRleHQKLSAgICBwcm90ZWN0ZWQgRm9udFJlbmRlckNvbnRleHQgZnJjID0gbmV3IEZvbnRSZW5kZXJDb250ZXh0KG51bGwsIGZhbHNlLCBmYWxzZSk7Ci0KLSAgICBwcm90ZWN0ZWQgSmF2YVNoYXBlUmFzdGVyaXplciBqc3IgPSBuZXcgSmF2YVNoYXBlUmFzdGVyaXplcigpOwotCi0gICAgcHJvdGVjdGVkIEZvbnQgZm9udCA9IG5ldyBGb250KCJEaWFsb2ciLCBGb250LlBMQUlOLCAxMik7OyAvLyROT04tTkxTLTEkCi0KLSAgICBwcm90ZWN0ZWQgVGV4dFJlbmRlcmVyIGp0ciA9IEphdmFUZXh0UmVuZGVyZXIuaW5zdDsKLQotICAgIC8vIEN1cnJlbnQgZ3JhcGhpY3MgdHJhbnNmb3JtCi0gICAgcHJvdGVjdGVkIEFmZmluZVRyYW5zZm9ybSB0cmFuc2Zvcm0gPSBuZXcgQWZmaW5lVHJhbnNmb3JtKCk7Ci0gICAgcHJvdGVjdGVkIGRvdWJsZVtdIG1hdHJpeCA9IG5ldyBkb3VibGVbNl07Ci0KLSAgICAvLyBPcmlnaW5hbCB1c2VyLT5kZXZpY2UgdHJhbnNsYXRpb24gYXMgdHJhbnNmb3JtIGFuZCBwb2ludAotICAgIC8vcHVibGljIEFmZmluZVRyYW5zZm9ybSBvcmlnVHJhbnNmb3JtID0gbmV3IEFmZmluZVRyYW5zZm9ybSgpOwotICAgIHB1YmxpYyBQb2ludCBvcmlnUG9pbnQgPSBuZXcgUG9pbnQoMCwgMCk7Ci0KLQotICAgIC8vIFByaW50IGRlYnVnIG91dHB1dCBvciBub3QKLSAgICBwcm90ZWN0ZWQgc3RhdGljIGZpbmFsIGJvb2xlYW4gZGVidWdPdXRwdXQgPSAiMSIuZXF1YWxzKFN5c3RlbS5nZXRQcm9wZXJ0eSgiZzJkLmRlYnVnIikpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLQotICAgIC8vIENvbnN0cnVjdG9ycwotICAgIHByb3RlY3RlZCBDb21tb25HcmFwaGljczJEKCkgewotICAgIH0KLQotICAgIHByb3RlY3RlZCBDb21tb25HcmFwaGljczJEKGludCB0eCwgaW50IHR5KSB7Ci0gICAgICAgIHRoaXModHgsIHR5LCBudWxsKTsKLSAgICB9Ci0KLSAgICBwcm90ZWN0ZWQgQ29tbW9uR3JhcGhpY3MyRChpbnQgdHgsIGludCB0eSwgTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0gICAgICAgIHNldFRyYW5zZm9ybShBZmZpbmVUcmFuc2Zvcm0uZ2V0VHJhbnNsYXRlSW5zdGFuY2UodHgsIHR5KSk7Ci0gICAgICAgIC8vb3JpZ1RyYW5zZm9ybSA9IEFmZmluZVRyYW5zZm9ybS5nZXRUcmFuc2xhdGVJbnN0YW5jZSh0eCwgdHkpOwotICAgICAgICBvcmlnUG9pbnQgPSBuZXcgUG9pbnQodHgsIHR5KTsKLSAgICAgICAgc2V0Q2xpcChjbGlwKTsKLSAgICB9Ci0KLSAgICAvLyBQdWJsaWMgbWV0aG9kcwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGFkZFJlbmRlcmluZ0hpbnRzKE1hcDw/LD8+IGhpbnRzKSB7Ci0gICAgICAgIHRoaXMuaGludHMucHV0QWxsKGhpbnRzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBjbGVhclJlY3QoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgQ29sb3IgYyA9IGdldENvbG9yKCk7Ci0gICAgICAgIFBhaW50IHAgPSBnZXRQYWludCgpOwotICAgICAgICBzZXRDb2xvcihnZXRCYWNrZ3JvdW5kKCkpOwotICAgICAgICBmaWxsUmVjdCh4LCB5LCB3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgc2V0Q29sb3IoYyk7Ci0gICAgICAgIHNldFBhaW50KHApOwotICAgICAgICBpZiAoZGVidWdPdXRwdXQpIHsKLSAgICAgICAgICAgIFN5c3RlbS5lcnIucHJpbnRsbigiQ29tbW9uR3JhcGhpY3MyRC5jbGVhclJlY3QoIit4KyIsICIreSsiLCAiK3dpZHRoKyIsICIraGVpZ2h0KyIpIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgY2xpcFJlY3QoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgY2xpcChuZXcgUmVjdGFuZ2xlKHgsIHksIHdpZHRoLCBoZWlnaHQpKTsKLSAgICB9Ci0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGNsaXAoU2hhcGUgcykgewotICAgICAgICBpZiAocyA9PSBudWxsKSB7Ci0gICAgICAgICAgICBjbGlwID0gbnVsbDsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIE11bHRpUmVjdEFyZWEgbXJhID0gbnVsbDsKLSAgICAgICAgaWYgKHMgaW5zdGFuY2VvZiBNdWx0aVJlY3RBcmVhKSB7Ci0gICAgICAgICAgICBtcmEgPSBuZXcgTXVsdGlSZWN0QXJlYSgoTXVsdGlSZWN0QXJlYSlzKTsKLSAgICAgICAgICAgIG1yYS50cmFuc2xhdGUoKGludCl0cmFuc2Zvcm0uZ2V0VHJhbnNsYXRlWCgpLCAoaW50KXRyYW5zZm9ybS5nZXRUcmFuc2xhdGVZKCkpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaW50IHR5cGUgPSB0cmFuc2Zvcm0uZ2V0VHlwZSgpOwotICAgICAgICAgICAgaWYocyBpbnN0YW5jZW9mIFJlY3RhbmdsZSAmJiAodHlwZSAmIChBZmZpbmVUcmFuc2Zvcm0uVFlQRV9JREVOVElUWSB8Ci0gICAgICAgICAgICAgICAgQWZmaW5lVHJhbnNmb3JtLlRZUEVfVFJBTlNMQVRJT04pKSAhPSAwKXsKLSAgICAgICAgICAgICAgICAgICAgbXJhID0gbmV3IE11bHRpUmVjdEFyZWEoKFJlY3RhbmdsZSlzKTsKLSAgICAgICAgICAgICAgICAgICAgaWYodHlwZSA9PSBBZmZpbmVUcmFuc2Zvcm0uVFlQRV9UUkFOU0xBVElPTil7Ci0gICAgICAgICAgICAgICAgICAgICAgICBtcmEudHJhbnNsYXRlKChpbnQpdHJhbnNmb3JtLmdldFRyYW5zbGF0ZVgoKSwgKGludCl0cmFuc2Zvcm0uZ2V0VHJhbnNsYXRlWSgpKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBzID0gdHJhbnNmb3JtLmNyZWF0ZVRyYW5zZm9ybWVkU2hhcGUocyk7Ci0gICAgICAgICAgICAgICAgbXJhID0ganNyLnJhc3Rlcml6ZShzLCAwLjUpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGNsaXAgPT0gbnVsbCkgewotICAgICAgICAgICAgc2V0VHJhbnNmb3JtZWRDbGlwKG1yYSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBjbGlwLmludGVyc2VjdChtcmEpOwotICAgICAgICAgICAgc2V0VHJhbnNmb3JtZWRDbGlwKGNsaXApOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZGlzcG9zZSgpIHsKLSAgICAgICAgLy8gRG8gbm90aGluZyBmb3IgSmF2YSBvbmx5IGNsYXNzZXMKLSAgICB9Ci0KLQotCi0KLSAgICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCi0gICAgICoKLSAgICAgKiAgRHJhdyBtZXRob2RzCi0gICAgICoKLSAgICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhdyhTaGFwZSBzKSB7Ci0gICAgICAgIGlmIChzdHJva2UgaW5zdGFuY2VvZiBCYXNpY1N0cm9rZSAmJiAoKEJhc2ljU3Ryb2tlKXN0cm9rZSkuZ2V0TGluZVdpZHRoKCkgPD0gMSkgewotICAgICAgICAgICAgLy9UT0RPOiBUaGluayBhYm91dCBkcmF3aW5nIHRoZSBzaGFwZSBpbiBvbmUgZmlsbE11bHRpUmVjdEFyZWEgY2FsbAotICAgICAgICAgICAgQmFzaWNTdHJva2UgYnN0cm9rZSA9IChCYXNpY1N0cm9rZSlzdHJva2U7Ci0gICAgICAgICAgICBKYXZhTGluZVJhc3Rlcml6ZXIuTGluZURhc2hlciBsZCA9IChic3Ryb2tlLmdldERhc2hBcnJheSgpID09IG51bGwpP251bGw6bmV3IEphdmFMaW5lUmFzdGVyaXplci5MaW5lRGFzaGVyKGJzdHJva2UuZ2V0RGFzaEFycmF5KCksIGJzdHJva2UuZ2V0RGFzaFBoYXNlKCkpOwotICAgICAgICAgICAgUGF0aEl0ZXJhdG9yIHBpID0gcy5nZXRQYXRoSXRlcmF0b3IodHJhbnNmb3JtLCAwLjUpOwotICAgICAgICAgICAgZmxvYXQgW11wb2ludHMgPSBuZXcgZmxvYXRbNl07Ci0gICAgICAgICAgICBpbnQgeDEgPSBJbnRlZ2VyLk1JTl9WQUxVRTsKLSAgICAgICAgICAgIGludCB5MSA9IEludGVnZXIuTUlOX1ZBTFVFOwotICAgICAgICAgICAgaW50IGN4MSA9IEludGVnZXIuTUlOX1ZBTFVFOwotICAgICAgICAgICAgaW50IGN5MSA9IEludGVnZXIuTUlOX1ZBTFVFOwotICAgICAgICAgICAgd2hpbGUgKCFwaS5pc0RvbmUoKSkgewotICAgICAgICAgICAgICAgIHN3aXRjaCAocGkuY3VycmVudFNlZ21lbnQocG9pbnRzKSkgewotICAgICAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfTU9WRVRPOgotICAgICAgICAgICAgICAgICAgICAgICAgeDEgPSAoaW50KU1hdGguZmxvb3IocG9pbnRzWzBdKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHkxID0gKGludClNYXRoLmZsb29yKHBvaW50c1sxXSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjeDEgPSB4MTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGN5MSA9IHkxOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19MSU5FVE86Ci0gICAgICAgICAgICAgICAgICAgICAgICBpbnQgeDIgPSAoaW50KU1hdGguZmxvb3IocG9pbnRzWzBdKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGludCB5MiA9IChpbnQpTWF0aC5mbG9vcihwb2ludHNbMV0pOwotICAgICAgICAgICAgICAgICAgICAgICAgZmlsbE11bHRpUmVjdEFyZWEoSmF2YUxpbmVSYXN0ZXJpemVyLnJhc3Rlcml6ZSh4MSwgeTEsIHgyLCB5MiwgbnVsbCwgbGQsIGZhbHNlKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB4MSA9IHgyOwotICAgICAgICAgICAgICAgICAgICAgICAgeTEgPSB5MjsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfQ0xPU0U6Ci0gICAgICAgICAgICAgICAgICAgICAgICB4MiA9IGN4MTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHkyID0gY3kxOwotICAgICAgICAgICAgICAgICAgICAgICAgZmlsbE11bHRpUmVjdEFyZWEoSmF2YUxpbmVSYXN0ZXJpemVyLnJhc3Rlcml6ZSh4MSwgeTEsIHgyLCB5MiwgbnVsbCwgbGQsIGZhbHNlKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB4MSA9IHgyOwotICAgICAgICAgICAgICAgICAgICAgICAgeTEgPSB5MjsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBwaS5uZXh0KCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBzID0gc3Ryb2tlLmNyZWF0ZVN0cm9rZWRTaGFwZShzKTsKLSAgICAgICAgICAgIHMgPSB0cmFuc2Zvcm0uY3JlYXRlVHJhbnNmb3JtZWRTaGFwZShzKTsKLSAgICAgICAgICAgIGZpbGxNdWx0aVJlY3RBcmVhKGpzci5yYXN0ZXJpemUocywgMC41KSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3QXJjKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgc2EsIGludCBlYSkgewotICAgICAgICBpZiAoc3Ryb2tlIGluc3RhbmNlb2YgQmFzaWNTdHJva2UgJiYgKChCYXNpY1N0cm9rZSlzdHJva2UpLmdldExpbmVXaWR0aCgpIDw9IDEgJiYKLSAgICAgICAgICAgICAgICAoKEJhc2ljU3Ryb2tlKXN0cm9rZSkuZ2V0RGFzaEFycmF5KCkgPT0gbnVsbCAmJiAKLSAgICAgICAgICAgICAgICAodHJhbnNmb3JtLmlzSWRlbnRpdHkoKSB8fCB0cmFuc2Zvcm0uZ2V0VHlwZSgpID09IEFmZmluZVRyYW5zZm9ybS5UWVBFX1RSQU5TTEFUSU9OKSkgewotICAgICAgICAgICAgUG9pbnQgcCA9IG5ldyBQb2ludCh4LCB5KTsKLSAgICAgICAgICAgIHRyYW5zZm9ybS50cmFuc2Zvcm0ocCwgcCk7Ci0gICAgICAgICAgICBNdWx0aVJlY3RBcmVhIG1yYSA9IEphdmFBcmNSYXN0ZXJpemVyLnJhc3Rlcml6ZSh4LCB5LCB3aWR0aCwgaGVpZ2h0LCBzYSwgZWEsIGNsaXApOwotICAgICAgICAgICAgZmlsbE11bHRpUmVjdEFyZWEobXJhKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBkcmF3KG5ldyBBcmMyRC5GbG9hdCh4LCB5LCB3aWR0aCwgaGVpZ2h0LCBzYSwgZWEsIEFyYzJELk9QRU4pKTsKLSAgICB9Ci0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGRyYXdJbWFnZShJbWFnZSBpbWFnZSwgaW50IHgsIGludCB5LCBDb2xvciBiZ2NvbG9yLAotICAgICAgICAgICAgSW1hZ2VPYnNlcnZlciBpbWFnZU9ic2VydmVyKSB7Ci0KLSAgICAgICAgaWYoaW1hZ2UgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICBib29sZWFuIGRvbmUgPSBmYWxzZTsKLSAgICAgICAgYm9vbGVhbiBzb21lYml0cyA9IGZhbHNlOwotICAgICAgICBTdXJmYWNlIHNyY1N1cmYgPSBudWxsOwotICAgICAgICBpZihpbWFnZSBpbnN0YW5jZW9mIE9mZnNjcmVlbkltYWdlKXsKLSAgICAgICAgICAgIE9mZnNjcmVlbkltYWdlIG9pID0gKE9mZnNjcmVlbkltYWdlKSBpbWFnZTsKLSAgICAgICAgICAgIGlmKChvaS5nZXRTdGF0ZSgpICYgSW1hZ2VPYnNlcnZlci5FUlJPUikgIT0gMCkgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGRvbmUgPSBvaS5wcmVwYXJlSW1hZ2UoaW1hZ2VPYnNlcnZlcik7Ci0gICAgICAgICAgICBzb21lYml0cyA9IChvaS5nZXRTdGF0ZSgpICYgSW1hZ2VPYnNlcnZlci5TT01FQklUUykgIT0gMDsKLSAgICAgICAgICAgIHNyY1N1cmYgPSBvaS5nZXRJbWFnZVN1cmZhY2UoKTsKLSAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICBkb25lID0gdHJ1ZTsKLSAgICAgICAgICAgIHNyY1N1cmYgPSBTdXJmYWNlLmdldEltYWdlU3VyZmFjZShpbWFnZSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZihkb25lIHx8IHNvbWViaXRzKSB7Ci0gICAgICAgICAgICBpbnQgdyA9IHNyY1N1cmYuZ2V0V2lkdGgoKTsKLSAgICAgICAgICAgIGludCBoID0gc3JjU3VyZi5nZXRIZWlnaHQoKTsKLSAgICAgICAgICAgIGJsaXR0ZXIuYmxpdCgwLCAwLCBzcmNTdXJmLCB4LCB5LCBkc3RTdXJmLCB3LCBoLCAoQWZmaW5lVHJhbnNmb3JtKSB0cmFuc2Zvcm0uY2xvbmUoKSwKLSAgICAgICAgICAgICAgICAgICAgY29tcG9zaXRlLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZG9uZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1hZ2UsIGludCB4LCBpbnQgeSwgSW1hZ2VPYnNlcnZlciBpbWFnZU9ic2VydmVyKSB7Ci0gICAgICAgIHJldHVybiBkcmF3SW1hZ2UoaW1hZ2UsIHgsIHksIG51bGwsIGltYWdlT2JzZXJ2ZXIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGRyYXdJbWFnZShJbWFnZSBpbWFnZSwgaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQsCi0gICAgICAgICAgICBDb2xvciBiZ2NvbG9yLCBJbWFnZU9ic2VydmVyIGltYWdlT2JzZXJ2ZXIpIHsKLQotICAgICAgICBpZihpbWFnZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgICAgICBpZih3aWR0aCA9PSAwIHx8IGhlaWdodCA9PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGJvb2xlYW4gZG9uZSA9IGZhbHNlOwotICAgICAgICBib29sZWFuIHNvbWViaXRzID0gZmFsc2U7Ci0gICAgICAgIFN1cmZhY2Ugc3JjU3VyZiA9IG51bGw7Ci0KLSAgICAgICAgaWYoaW1hZ2UgaW5zdGFuY2VvZiBPZmZzY3JlZW5JbWFnZSl7Ci0gICAgICAgICAgICBPZmZzY3JlZW5JbWFnZSBvaSA9IChPZmZzY3JlZW5JbWFnZSkgaW1hZ2U7Ci0gICAgICAgICAgICBpZigob2kuZ2V0U3RhdGUoKSAmIEltYWdlT2JzZXJ2ZXIuRVJST1IpICE9IDApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkb25lID0gb2kucHJlcGFyZUltYWdlKGltYWdlT2JzZXJ2ZXIpOwotICAgICAgICAgICAgc29tZWJpdHMgPSAob2kuZ2V0U3RhdGUoKSAmIEltYWdlT2JzZXJ2ZXIuU09NRUJJVFMpICE9IDA7Ci0gICAgICAgICAgICBzcmNTdXJmID0gb2kuZ2V0SW1hZ2VTdXJmYWNlKCk7Ci0gICAgICAgIH1lbHNlewotICAgICAgICAgICAgZG9uZSA9IHRydWU7Ci0gICAgICAgICAgICBzcmNTdXJmID0gU3VyZmFjZS5nZXRJbWFnZVN1cmZhY2UoaW1hZ2UpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYoZG9uZSB8fCBzb21lYml0cykgewotICAgICAgICAgICAgaW50IHcgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgaCA9IHNyY1N1cmYuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICBpZih3ID09IHdpZHRoICYmIGggPT0gaGVpZ2h0KXsKLSAgICAgICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgeCwgeSwgZHN0U3VyZiwgdywgaCwKLSAgICAgICAgICAgICAgICAgICAgICAgIChBZmZpbmVUcmFuc2Zvcm0pIHRyYW5zZm9ybS5jbG9uZSgpLAotICAgICAgICAgICAgICAgICAgICAgICAgY29tcG9zaXRlLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgIEFmZmluZVRyYW5zZm9ybSB4Zm9ybSA9IG5ldyBBZmZpbmVUcmFuc2Zvcm0oKTsKLSAgICAgICAgICAgICAgICB4Zm9ybS5zZXRUb1NjYWxlKChmbG9hdCl3aWR0aCAvIHcsIChmbG9hdCloZWlnaHQgLyBoKTsKLSAgICAgICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgeCwgeSwgZHN0U3VyZiwgdywgaCwKLSAgICAgICAgICAgICAgICAgICAgICAgIChBZmZpbmVUcmFuc2Zvcm0pIHRyYW5zZm9ybS5jbG9uZSgpLAotICAgICAgICAgICAgICAgICAgICAgICAgeGZvcm0sIGNvbXBvc2l0ZSwgYmdjb2xvciwgY2xpcCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGRvbmU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZHJhd0ltYWdlKEltYWdlIGltYWdlLCBpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCwKLSAgICAgICAgICAgIEltYWdlT2JzZXJ2ZXIgaW1hZ2VPYnNlcnZlcikgewotICAgICAgICByZXR1cm4gZHJhd0ltYWdlKGltYWdlLCB4LCB5LCB3aWR0aCwgaGVpZ2h0LCBudWxsLCBpbWFnZU9ic2VydmVyKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1hZ2UsIGludCBkeDEsIGludCBkeTEsIGludCBkeDIsIGludCBkeTIsCi0gICAgICAgICAgICBpbnQgc3gxLCBpbnQgc3kxLCBpbnQgc3gyLCBpbnQgc3kyLCBDb2xvciBiZ2NvbG9yLAotICAgICAgICAgICAgSW1hZ2VPYnNlcnZlciBpbWFnZU9ic2VydmVyKSB7Ci0KLSAgICAgICAgaWYoaW1hZ2UgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgaWYoZHgxID09IGR4MiB8fCBkeTEgPT0gZHkyIHx8IHN4MSA9PSBzeDIgfHwgc3kxID09IHN5MikgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICBib29sZWFuIGRvbmUgPSBmYWxzZTsKLSAgICAgICAgYm9vbGVhbiBzb21lYml0cyA9IGZhbHNlOwotICAgICAgICBTdXJmYWNlIHNyY1N1cmYgPSBudWxsOwotICAgICAgICBpZihpbWFnZSBpbnN0YW5jZW9mIE9mZnNjcmVlbkltYWdlKXsKLSAgICAgICAgICAgIE9mZnNjcmVlbkltYWdlIG9pID0gKE9mZnNjcmVlbkltYWdlKSBpbWFnZTsKLSAgICAgICAgICAgIGlmKChvaS5nZXRTdGF0ZSgpICYgSW1hZ2VPYnNlcnZlci5FUlJPUikgIT0gMCkgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGRvbmUgPSBvaS5wcmVwYXJlSW1hZ2UoaW1hZ2VPYnNlcnZlcik7Ci0gICAgICAgICAgICBzb21lYml0cyA9IChvaS5nZXRTdGF0ZSgpICYgSW1hZ2VPYnNlcnZlci5TT01FQklUUykgIT0gMDsKLSAgICAgICAgICAgIHNyY1N1cmYgPSBvaS5nZXRJbWFnZVN1cmZhY2UoKTsKLSAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICBkb25lID0gdHJ1ZTsKLSAgICAgICAgICAgIHNyY1N1cmYgPSBTdXJmYWNlLmdldEltYWdlU3VyZmFjZShpbWFnZSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZihkb25lIHx8IHNvbWViaXRzKSB7Ci0KLSAgICAgICAgICAgIGludCBkc3RYID0gZHgxOwotICAgICAgICAgICAgaW50IGRzdFkgPSBkeTE7Ci0gICAgICAgICAgICBpbnQgc3JjWCA9IHN4MTsKLSAgICAgICAgICAgIGludCBzcmNZID0gc3kxOwotCi0gICAgICAgICAgICBpbnQgZHN0VyA9IGR4MiAtIGR4MTsKLSAgICAgICAgICAgIGludCBkc3RIID0gZHkyIC0gZHkxOwotICAgICAgICAgICAgaW50IHNyY1cgPSBzeDIgLSBzeDE7Ci0gICAgICAgICAgICBpbnQgc3JjSCA9IHN5MiAtIHN5MTsKLQotICAgICAgICAgICAgaWYoc3JjVyA9PSBkc3RXICYmIHNyY0ggPT0gZHN0SCl7Ci0gICAgICAgICAgICAgICAgYmxpdHRlci5ibGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIGRzdFgsIGRzdFksIGRzdFN1cmYsIHNyY1csIHNyY0gsCi0gICAgICAgICAgICAgICAgICAgICAgICAoQWZmaW5lVHJhbnNmb3JtKSB0cmFuc2Zvcm0uY2xvbmUoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbXBvc2l0ZSwgYmdjb2xvciwgY2xpcCk7Ci0gICAgICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0gPSBuZXcgQWZmaW5lVHJhbnNmb3JtKCk7Ci0gICAgICAgICAgICAgICAgeGZvcm0uc2V0VG9TY2FsZSgoZmxvYXQpZHN0VyAvIHNyY1csIChmbG9hdClkc3RIIC8gc3JjSCk7Ci0gICAgICAgICAgICAgICAgYmxpdHRlci5ibGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIGRzdFgsIGRzdFksIGRzdFN1cmYsIHNyY1csIHNyY0gsCi0gICAgICAgICAgICAgICAgICAgICAgICAoQWZmaW5lVHJhbnNmb3JtKSB0cmFuc2Zvcm0uY2xvbmUoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHhmb3JtLCBjb21wb3NpdGUsIGJnY29sb3IsIGNsaXApOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBkb25lOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGRyYXdJbWFnZShJbWFnZSBpbWFnZSwgaW50IGR4MSwgaW50IGR5MSwgaW50IGR4MiwgaW50IGR5MiwKLSAgICAgICAgICAgIGludCBzeDEsIGludCBzeTEsIGludCBzeDIsIGludCBzeTIsIEltYWdlT2JzZXJ2ZXIgaW1hZ2VPYnNlcnZlcikgewotCi0gICAgICAgIHJldHVybiBkcmF3SW1hZ2UoaW1hZ2UsIGR4MSwgZHkxLCBkeDIsIGR5Miwgc3gxLCBzeTEsIHN4Miwgc3kyLCBudWxsLAotICAgICAgICAgICAgICAgIGltYWdlT2JzZXJ2ZXIpOwotICAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3SW1hZ2UoQnVmZmVyZWRJbWFnZSBidWZJbWFnZSwgQnVmZmVyZWRJbWFnZU9wIG9wLAotICAgICAgICAgICAgaW50IHgsIGludCB5KSB7Ci0KLSAgICAgICAgaWYoYnVmSW1hZ2UgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYob3AgPT0gbnVsbCkgewotICAgICAgICAgICAgZHJhd0ltYWdlKGJ1ZkltYWdlLCB4LCB5LCBudWxsKTsKLSAgICAgICAgfSBlbHNlIGlmKG9wIGluc3RhbmNlb2YgQWZmaW5lVHJhbnNmb3JtT3ApewotICAgICAgICAgICAgQWZmaW5lVHJhbnNmb3JtT3AgYXRvcCA9IChBZmZpbmVUcmFuc2Zvcm1PcCkgb3A7Ci0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0gPSBhdG9wLmdldFRyYW5zZm9ybSgpOwotICAgICAgICAgICAgU3VyZmFjZSBzcmNTdXJmID0gU3VyZmFjZS5nZXRJbWFnZVN1cmZhY2UoYnVmSW1hZ2UpOwotICAgICAgICAgICAgaW50IHcgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgaCA9IHNyY1N1cmYuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgeCwgeSwgZHN0U3VyZiwgdywgaCwKLSAgICAgICAgICAgICAgICAgICAgKEFmZmluZVRyYW5zZm9ybSkgdHJhbnNmb3JtLmNsb25lKCksIHhmb3JtLAotICAgICAgICAgICAgICAgICAgICBjb21wb3NpdGUsIG51bGwsIGNsaXApOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgYnVmSW1hZ2UgPSBvcC5maWx0ZXIoYnVmSW1hZ2UsIG51bGwpOwotICAgICAgICAgICAgU3VyZmFjZSBzcmNTdXJmID0gU3VyZmFjZS5nZXRJbWFnZVN1cmZhY2UoYnVmSW1hZ2UpOwotICAgICAgICAgICAgaW50IHcgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgaCA9IHNyY1N1cmYuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgeCwgeSwgZHN0U3VyZiwgdywgaCwKLSAgICAgICAgICAgICAgICAgICAgKEFmZmluZVRyYW5zZm9ybSkgdHJhbnNmb3JtLmNsb25lKCksCi0gICAgICAgICAgICAgICAgICAgIGNvbXBvc2l0ZSwgbnVsbCwgY2xpcCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBkcmF3SW1hZ2UoSW1hZ2UgaW1hZ2UsIEFmZmluZVRyYW5zZm9ybSB0cmFucywKLSAgICAgICAgICAgIEltYWdlT2JzZXJ2ZXIgaW1hZ2VPYnNlcnZlcikgewotCi0gICAgICAgIGlmKGltYWdlID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgICAgIGlmKHRyYW5zID09IG51bGwgfHwgdHJhbnMuaXNJZGVudGl0eSgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZHJhd0ltYWdlKGltYWdlLCAwLCAwLCBpbWFnZU9ic2VydmVyKTsKLSAgICAgICAgfQotCi0gICAgICAgIGJvb2xlYW4gZG9uZSA9IGZhbHNlOwotICAgICAgICBib29sZWFuIHNvbWViaXRzID0gZmFsc2U7Ci0gICAgICAgIFN1cmZhY2Ugc3JjU3VyZiA9IG51bGw7Ci0gICAgICAgIGlmKGltYWdlIGluc3RhbmNlb2YgT2Zmc2NyZWVuSW1hZ2UpewotICAgICAgICAgICAgT2Zmc2NyZWVuSW1hZ2Ugb2kgPSAoT2Zmc2NyZWVuSW1hZ2UpIGltYWdlOwotICAgICAgICAgICAgaWYoKG9pLmdldFN0YXRlKCkgJiBJbWFnZU9ic2VydmVyLkVSUk9SKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZG9uZSA9IG9pLnByZXBhcmVJbWFnZShpbWFnZU9ic2VydmVyKTsKLSAgICAgICAgICAgIHNvbWViaXRzID0gKG9pLmdldFN0YXRlKCkgJiBJbWFnZU9ic2VydmVyLlNPTUVCSVRTKSAhPSAwOwotICAgICAgICAgICAgc3JjU3VyZiA9IG9pLmdldEltYWdlU3VyZmFjZSgpOwotICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgIGRvbmUgPSB0cnVlOwotICAgICAgICAgICAgc3JjU3VyZiA9IFN1cmZhY2UuZ2V0SW1hZ2VTdXJmYWNlKGltYWdlKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmKGRvbmUgfHwgc29tZWJpdHMpIHsKLSAgICAgICAgICAgIGludCB3ID0gc3JjU3VyZi5nZXRXaWR0aCgpOwotICAgICAgICAgICAgaW50IGggPSBzcmNTdXJmLmdldEhlaWdodCgpOwotICAgICAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHhmb3JtID0gKEFmZmluZVRyYW5zZm9ybSkgdHJhbnNmb3JtLmNsb25lKCk7Ci0gICAgICAgICAgICB4Zm9ybS5jb25jYXRlbmF0ZSh0cmFucyk7Ci0gICAgICAgICAgICBibGl0dGVyLmJsaXQoMCwgMCwgc3JjU3VyZiwgMCwgMCwgZHN0U3VyZiwgdywgaCwgeGZvcm0sIGNvbXBvc2l0ZSwKLSAgICAgICAgICAgICAgICAgICAgbnVsbCwgY2xpcCk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGRvbmU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd0xpbmUoaW50IHgxLCBpbnQgeTEsIGludCB4MiwgaW50IHkyKSB7Ci0gICAgICAgIGlmIChkZWJ1Z091dHB1dCkgewotICAgICAgICAgICAgU3lzdGVtLmVyci5wcmludGxuKCJDb21tb25HcmFwaGljczJELmRyYXdMaW5lKCIreDErIiwgIit5MSsiLCAiK3gyKyIsICIreTIrIikiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQgLy8kTk9OLU5MUy00JCAvLyROT04tTkxTLTUkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoc3Ryb2tlIGluc3RhbmNlb2YgQmFzaWNTdHJva2UgJiYgKChCYXNpY1N0cm9rZSlzdHJva2UpLmdldExpbmVXaWR0aCgpIDw9IDEpIHsKLSAgICAgICAgICAgIEJhc2ljU3Ryb2tlIGJzdHJva2UgPSAoQmFzaWNTdHJva2Upc3Ryb2tlOwotICAgICAgICAgICAgUG9pbnQgcDEgPSBuZXcgUG9pbnQoeDEsIHkxKTsKLSAgICAgICAgICAgIFBvaW50IHAyID0gbmV3IFBvaW50KHgyLCB5Mik7Ci0gICAgICAgICAgICB0cmFuc2Zvcm0udHJhbnNmb3JtKHAxLCBwMSk7Ci0gICAgICAgICAgICB0cmFuc2Zvcm0udHJhbnNmb3JtKHAyLCBwMik7Ci0gICAgICAgICAgICBKYXZhTGluZVJhc3Rlcml6ZXIuTGluZURhc2hlciBsZCA9IChic3Ryb2tlLmdldERhc2hBcnJheSgpID09IG51bGwpP251bGw6bmV3IEphdmFMaW5lUmFzdGVyaXplci5MaW5lRGFzaGVyKGJzdHJva2UuZ2V0RGFzaEFycmF5KCksIGJzdHJva2UuZ2V0RGFzaFBoYXNlKCkpOwotICAgICAgICAgICAgTXVsdGlSZWN0QXJlYSBtcmEgPSBKYXZhTGluZVJhc3Rlcml6ZXIucmFzdGVyaXplKHAxLngsIHAxLnksIHAyLngsIHAyLnksIG51bGwsIGxkLCBmYWxzZSk7Ci0gICAgICAgICAgICBmaWxsTXVsdGlSZWN0QXJlYShtcmEpOwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIGRyYXcobmV3IExpbmUyRC5GbG9hdCh4MSwgeTEsIHgyLCB5MikpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRyYXdPdmFsKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIGlmIChzdHJva2UgaW5zdGFuY2VvZiBCYXNpY1N0cm9rZSAmJiAoKEJhc2ljU3Ryb2tlKXN0cm9rZSkuZ2V0TGluZVdpZHRoKCkgPD0gMSAmJgotICAgICAgICAgICAgICAgICgoQmFzaWNTdHJva2Upc3Ryb2tlKS5nZXREYXNoQXJyYXkoKSA9PSBudWxsICYmIAotICAgICAgICAgICAgICAgICh0cmFuc2Zvcm0uaXNJZGVudGl0eSgpIHx8IHRyYW5zZm9ybS5nZXRUeXBlKCkgPT0gQWZmaW5lVHJhbnNmb3JtLlRZUEVfVFJBTlNMQVRJT04pKSB7Ci0gICAgICAgICAgICBQb2ludCBwID0gbmV3IFBvaW50KHgsIHkpOwotICAgICAgICAgICAgdHJhbnNmb3JtLnRyYW5zZm9ybShwLCBwKTsKLSAgICAgICAgICAgIE11bHRpUmVjdEFyZWEgbXJhID0gSmF2YUFyY1Jhc3Rlcml6ZXIucmFzdGVyaXplKHgsIHksIHdpZHRoLCBoZWlnaHQsIDAsIDM2MCwgY2xpcCk7Ci0gICAgICAgICAgICBmaWxsTXVsdGlSZWN0QXJlYShtcmEpOwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIGRyYXcobmV3IEVsbGlwc2UyRC5GbG9hdCh4LCB5LCB3aWR0aCwgaGVpZ2h0KSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd1BvbHlnb24oaW50W10geHBvaW50cywgaW50W10geXBvaW50cywgaW50IG5wb2ludHMpIHsKLSAgICAgICAgZHJhdyhuZXcgUG9seWdvbih4cG9pbnRzLCB5cG9pbnRzLCBucG9pbnRzKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd1BvbHlnb24oUG9seWdvbiBwb2x5Z29uKSB7Ci0gICAgICAgIGRyYXcocG9seWdvbik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd1BvbHlsaW5lKGludFtdIHhwb2ludHMsIGludFtdIHlwb2ludHMsIGludCBucG9pbnRzKSB7Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnBvaW50cy0xOyBpKyspIHsKLSAgICAgICAgICAgIGRyYXdMaW5lKHhwb2ludHNbaV0sIHlwb2ludHNbaV0sIHhwb2ludHNbaSsxXSwgeXBvaW50c1tpKzFdKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRyYXdSZW5kZXJhYmxlSW1hZ2UoUmVuZGVyYWJsZUltYWdlIGltZywgQWZmaW5lVHJhbnNmb3JtIHhmb3JtKSB7Ci0gICAgICAgIGlmIChpbWcgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIHNjYWxlWCA9IHhmb3JtLmdldFNjYWxlWCgpOwotICAgICAgICBkb3VibGUgc2NhbGVZID0geGZvcm0uZ2V0U2NhbGVZKCk7Ci0gICAgICAgIGlmIChzY2FsZVggPT0gMSAmJiBzY2FsZVkgPT0gMSkgewotICAgICAgICAgICAgZHJhd1JlbmRlcmVkSW1hZ2UoaW1nLmNyZWF0ZURlZmF1bHRSZW5kZXJpbmcoKSwgeGZvcm0pOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaW50IHdpZHRoID0gKGludClNYXRoLnJvdW5kKGltZy5nZXRXaWR0aCgpKnNjYWxlWCk7Ci0gICAgICAgICAgICBpbnQgaGVpZ2h0ID0gKGludClNYXRoLnJvdW5kKGltZy5nZXRIZWlnaHQoKSpzY2FsZVkpOwotICAgICAgICAgICAgeGZvcm0gPSAoQWZmaW5lVHJhbnNmb3JtKXhmb3JtLmNsb25lKCk7Ci0gICAgICAgICAgICB4Zm9ybS5zY2FsZSgxLCAxKTsKLSAgICAgICAgICAgIGRyYXdSZW5kZXJlZEltYWdlKGltZy5jcmVhdGVTY2FsZWRSZW5kZXJpbmcod2lkdGgsIGhlaWdodCwgbnVsbCksIHhmb3JtKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRyYXdSZW5kZXJlZEltYWdlKFJlbmRlcmVkSW1hZ2UgcmltZywgQWZmaW5lVHJhbnNmb3JtIHhmb3JtKSB7Ci0gICAgICAgIGlmIChyaW1nID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIEltYWdlIGltZyA9IG51bGw7Ci0KLSAgICAgICAgaWYgKHJpbWcgaW5zdGFuY2VvZiBJbWFnZSkgewotICAgICAgICAgICAgaW1nID0gKEltYWdlKXJpbWc7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvL1RPRE86IENyZWF0ZSBuZXcgY2xhc3MgdG8gcHJvdmlkZSBJbWFnZSBpbnRlcmZhY2UgZm9yIFJlbmRlcmVkSW1hZ2Ugb3IgcmV3cml0ZSB0aGlzIG1ldGhvZAotICAgICAgICAgICAgaW1nID0gbmV3IEJ1ZmZlcmVkSW1hZ2UocmltZy5nZXRDb2xvck1vZGVsKCksIHJpbWcuY29weURhdGEobnVsbCksIGZhbHNlLCBudWxsKTsKLSAgICAgICAgfQotCi0gICAgICAgIGRyYXdJbWFnZShpbWcsIHhmb3JtLCBudWxsKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3Um91bmRSZWN0KGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgYXJjV2lkdGgsIGludCBhcmNIZWlnaHQpIHsKLSAgICAgICAgaWYgKGRlYnVnT3V0cHV0KSB7Ci0gICAgICAgICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIkNvbW1vbkdyYXBoaWNzMkQuZHJhd1JvdW5kUmVjdCgiK3grIiwgIit5KyIsICIrd2lkdGgrIiwgIitoZWlnaHQrIiwiK2FyY1dpZHRoKyIsICIrYXJjSGVpZ2h0KyIpIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JCAvLyROT04tTkxTLTYkIC8vJE5PTi1OTFMtNyQKLSAgICAgICAgfQotCi0gICAgICAgIGRyYXcobmV3IFJvdW5kUmVjdGFuZ2xlMkQuRmxvYXQoeCwgeSwgd2lkdGgsIGhlaWdodCwgYXJjV2lkdGgsIGFyY0hlaWdodCkpOwotICAgIH0KLQotCi0KLQotCi0gICAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgotICAgICAqCi0gICAgICogIFN0cmluZyBtZXRob2RzCi0gICAgICoKLSAgICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd1N0cmluZyhBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgaXRlcmF0b3IsIGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgR2x5cGhWZWN0b3IgZ3YgPSBmb250LmNyZWF0ZUdseXBoVmVjdG9yKGZyYywgaXRlcmF0b3IpOwotICAgICAgICBkcmF3R2x5cGhWZWN0b3IoZ3YsIHgsIHkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRyYXdTdHJpbmcoQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIGl0ZXJhdG9yLCBpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgZHJhd1N0cmluZyhpdGVyYXRvciwgKGZsb2F0KXgsIChmbG9hdCl5KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3U3RyaW5nKFN0cmluZyBzdHIsIGludCB4LCBpbnQgeSkgewotICAgICAgICBkcmF3U3RyaW5nKHN0ciwgKGZsb2F0KXgsIChmbG9hdCl5KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3U3RyaW5nKFN0cmluZyBzdHIsIGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgaWYgKGRlYnVnT3V0cHV0KSB7Ci0gICAgICAgICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIkNvbW1vbkdyYXBoaWNzMkQuZHJhd1N0cmluZygiK3N0cisiLCAiK3grIiwgIit5KyIpIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQKLSAgICAgICAgfQotCi0gICAgICAgIEFmZmluZVRyYW5zZm9ybSBhdCA9IChBZmZpbmVUcmFuc2Zvcm0pdGhpcy5nZXRUcmFuc2Zvcm0oKS5jbG9uZSgpOwotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gZm9udFRyYW5zZm9ybSA9IGZvbnQuZ2V0VHJhbnNmb3JtKCk7Ci0gICAgICAgIGF0LmNvbmNhdGVuYXRlKGZvbnRUcmFuc2Zvcm0pOwotCi0gICAgICAgIGRvdWJsZVtdIG1hdHJpeCA9IG5ldyBkb3VibGVbNl07Ci0gICAgICAgIGlmICghYXQuaXNJZGVudGl0eSgpKXsKLQotICAgICAgICAgICAgaW50IGF0VHlwZSA9IGF0LmdldFR5cGUoKTsKLSAgICAgICAgICAgIGF0LmdldE1hdHJpeChtYXRyaXgpOwotCi0gICAgICAgICAgICAvLyBUWVBFX1RSQU5TTEFUSU9OCi0gICAgICAgICAgICBpZiAoYXRUeXBlID09IEFmZmluZVRyYW5zZm9ybS5UWVBFX1RSQU5TTEFUSU9OKXsKLSAgICAgICAgICAgICAgICBqdHIuZHJhd1N0cmluZyh0aGlzLCBzdHIsCi0gICAgICAgICAgICAgICAgICAgICAgICAoZmxvYXQpKHgrZm9udFRyYW5zZm9ybS5nZXRUcmFuc2xhdGVYKCkpLAotICAgICAgICAgICAgICAgICAgICAgICAgKGZsb2F0KSh5K2ZvbnRUcmFuc2Zvcm0uZ2V0VHJhbnNsYXRlWSgpKSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgLy8gVE9ETzogd2UgdXNlIHNsb3cgdHlwZSBvZiBkcmF3aW5nIHN0cmluZ3Mgd2hlbiBGb250IG9iamVjdAotICAgICAgICAgICAgLy8gaW4gR3JhcGhpY3MgaGFzIHRyYW5zZm9ybXMsIHdlIGp1c3QgZmlsbCBvdXRsaW5lcy4gTmV3IHRleHRyZW5kZXJlcgotICAgICAgICAgICAgLy8gaXMgdG8gYmUgaW1wbGVtZW50ZWQuCi0gICAgICAgICAgICBTaGFwZSBzaCA9IGZvbnQuY3JlYXRlR2x5cGhWZWN0b3IodGhpcy5nZXRGb250UmVuZGVyQ29udGV4dCgpLCBzdHIpLmdldE91dGxpbmUoeCwgeSk7Ci0gICAgICAgICAgICB0aGlzLmZpbGwoc2gpOwotCi0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBqdHIuZHJhd1N0cmluZyh0aGlzLCBzdHIsIHgsIHkpOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkcmF3R2x5cGhWZWN0b3IoR2x5cGhWZWN0b3IgZ3YsIGZsb2F0IHgsIGZsb2F0IHkpIHsKLQotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gYXQgPSBndi5nZXRGb250KCkuZ2V0VHJhbnNmb3JtKCk7Ci0KLSAgICAgICAgZG91YmxlW10gbWF0cml4ID0gbmV3IGRvdWJsZVs2XTsKLSAgICAgICAgaWYgKChhdCAhPSBudWxsKSAmJiAoIWF0LmlzSWRlbnRpdHkoKSkpewotCi0gICAgICAgICAgICBpbnQgYXRUeXBlID0gYXQuZ2V0VHlwZSgpOwotICAgICAgICAgICAgYXQuZ2V0TWF0cml4KG1hdHJpeCk7Ci0KLSAgICAgICAgICAgIC8vIFRZUEVfVFJBTlNMQVRJT04KLSAgICAgICAgICAgIGlmICgoYXRUeXBlID09IEFmZmluZVRyYW5zZm9ybS5UWVBFX1RSQU5TTEFUSU9OKSAmJgotICAgICAgICAgICAgICAgICgoZ3YuZ2V0TGF5b3V0RmxhZ3MoKSAmIEdseXBoVmVjdG9yLkZMQUdfSEFTX1RSQU5TRk9STVMpID09IDApKXsKLSAgICAgICAgICAgICAgICBqdHIuZHJhd0dseXBoVmVjdG9yKHRoaXMsIGd2LCAoaW50KSh4K21hdHJpeFs0XSksIChpbnQpKHkrbWF0cml4WzVdKSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaWYgKCgoZ3YuZ2V0TGF5b3V0RmxhZ3MoKSAmIEdseXBoVmVjdG9yLkZMQUdfSEFTX1RSQU5TRk9STVMpID09IDApKXsKLSAgICAgICAgICAgICAgICBqdHIuZHJhd0dseXBoVmVjdG9yKHRoaXMsIGd2LCB4LCB5KTsKLSAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBUT0RPOiB3ZSB1c2Ugc2xvdyB0eXBlIG9mIGRyYXdpbmcgc3RyaW5ncyB3aGVuIEZvbnQgb2JqZWN0Ci0gICAgICAgIC8vIGluIEdyYXBoaWNzIGhhcyB0cmFuc2Zvcm1zLCB3ZSBqdXN0IGZpbGwgb3V0bGluZXMuIE5ldyB0ZXh0cmVuZGVyZXIKLSAgICAgICAgLy8gaXMgdG8gYmUgaW1wbGVtZW50ZWQuCi0KLSAgICAgICAgU2hhcGUgc2ggPSBndi5nZXRPdXRsaW5lKHgsIHkpOwotICAgICAgICB0aGlzLmZpbGwoc2gpOwotCi0gICAgICAgIH0KLQotCi0KLQotICAgIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKLSAgICAgKgotICAgICAqICBGaWxsIG1ldGhvZHMKLSAgICAgKgotICAgICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmaWxsKFNoYXBlIHMpIHsKLSAgICAgICAgcyA9IHRyYW5zZm9ybS5jcmVhdGVUcmFuc2Zvcm1lZFNoYXBlKHMpOwotICAgICAgICBNdWx0aVJlY3RBcmVhIG1yYSA9IGpzci5yYXN0ZXJpemUocywgMC41KTsKLSAgICAgICAgZmlsbE11bHRpUmVjdEFyZWEobXJhKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmaWxsQXJjKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgc2EsIGludCBlYSkgewotICAgICAgICBmaWxsKG5ldyBBcmMyRC5GbG9hdCh4LCB5LCB3aWR0aCwgaGVpZ2h0LCBzYSwgZWEsIEFyYzJELlBJRSkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGZpbGxPdmFsKGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7Ci0gICAgICAgIGZpbGwobmV3IEVsbGlwc2UyRC5GbG9hdCh4LCB5LCB3aWR0aCwgaGVpZ2h0KSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZmlsbFBvbHlnb24oaW50W10geHBvaW50cywgaW50W10geXBvaW50cywgaW50IG5wb2ludHMpIHsKLSAgICAgICAgZmlsbChuZXcgUG9seWdvbih4cG9pbnRzLCB5cG9pbnRzLCBucG9pbnRzKSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZmlsbFBvbHlnb24oUG9seWdvbiBwb2x5Z29uKSB7Ci0gICAgICAgIGZpbGwocG9seWdvbik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZmlsbFJlY3QoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgaWYgKGRlYnVnT3V0cHV0KSB7Ci0gICAgICAgICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIkNvbW1vbkdyYXBoaWNzMkQuZmlsbFJlY3QoIit4KyIsICIreSsiLCAiK3dpZHRoKyIsICIraGVpZ2h0KyIpIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgICAgICB9Ci0KLSAgICAgICAgZmlsbChuZXcgUmVjdGFuZ2xlKHgsIHksIHdpZHRoLCBoZWlnaHQpKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBmaWxsUm91bmRSZWN0KGludCB4LCBpbnQgeSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgYXJjV2lkdGgsIGludCBhcmNIZWlnaHQpIHsKLSAgICAgICAgaWYgKGRlYnVnT3V0cHV0KSB7Ci0gICAgICAgICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIkNvbW1vbkdyYXBoaWNzMkQuZmlsbFJvdW5kUmVjdCgiK3grIiwgIit5KyIsICIrd2lkdGgrIiwgIitoZWlnaHQrIiwiK2FyY1dpZHRoKyIsICIrYXJjSGVpZ2h0KyIpIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JCAvLyROT04tTkxTLTYkIC8vJE5PTi1OTFMtNyQKLSAgICAgICAgfQotCi0gICAgICAgIGZpbGwobmV3IFJvdW5kUmVjdGFuZ2xlMkQuRmxvYXQoeCwgeSwgd2lkdGgsIGhlaWdodCwgYXJjV2lkdGgsIGFyY0hlaWdodCkpOwotICAgIH0KLQotCi0KLQotICAgIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKLSAgICAgKgotICAgICAqICBHZXQgbWV0aG9kcwotICAgICAqCi0gICAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBDb2xvciBnZXRCYWNrZ3JvdW5kKCkgewotICAgICAgICByZXR1cm4gYmdDb2xvcjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU2hhcGUgZ2V0Q2xpcCgpIHsKLSAgICAgICAgaWYgKGNsaXAgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICBNdWx0aVJlY3RBcmVhIHJlcyA9IG5ldyBNdWx0aVJlY3RBcmVhKGNsaXApOwotICAgICAgICByZXMudHJhbnNsYXRlKC1NYXRoLnJvdW5kKChmbG9hdCl0cmFuc2Zvcm0uZ2V0VHJhbnNsYXRlWCgpKSwgLU1hdGgucm91bmQoKGZsb2F0KXRyYW5zZm9ybS5nZXRUcmFuc2xhdGVZKCkpKTsKLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldENsaXBCb3VuZHMoKSB7Ci0gICAgICAgIGlmIChjbGlwID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgUmVjdGFuZ2xlIHJlcyA9IChSZWN0YW5nbGUpIGNsaXAuZ2V0Qm91bmRzKCkuY2xvbmUoKTsKLSAgICAgICAgcmVzLnRyYW5zbGF0ZSgtTWF0aC5yb3VuZCgoZmxvYXQpdHJhbnNmb3JtLmdldFRyYW5zbGF0ZVgoKSksIC1NYXRoLnJvdW5kKChmbG9hdCl0cmFuc2Zvcm0uZ2V0VHJhbnNsYXRlWSgpKSk7Ci0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIENvbG9yIGdldENvbG9yKCkgewotICAgICAgICByZXR1cm4gZmdDb2xvcjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgQ29tcG9zaXRlIGdldENvbXBvc2l0ZSgpIHsKLSAgICAgICAgcmV0dXJuIGNvbXBvc2l0ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgRm9udCBnZXRGb250KCkgewotICAgICAgICByZXR1cm4gZm9udDsKLSAgICB9Ci0KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygiZGVwcmVjYXRpb24iKQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBGb250TWV0cmljcyBnZXRGb250TWV0cmljcyhGb250IGZvbnQpIHsKLSAgICAgICAgcmV0dXJuIFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKS5nZXRGb250TWV0cmljcyhmb250KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgRm9udFJlbmRlckNvbnRleHQgZ2V0Rm9udFJlbmRlckNvbnRleHQoKSB7Ci0gICAgICAgIHJldHVybiBmcmM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFBhaW50IGdldFBhaW50KCkgewotICAgICAgICByZXR1cm4gcGFpbnQ7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBnZXRSZW5kZXJpbmdIaW50KFJlbmRlcmluZ0hpbnRzLktleSBrZXkpIHsKLSAgICAgICAgcmV0dXJuIGhpbnRzLmdldChrZXkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBSZW5kZXJpbmdIaW50cyBnZXRSZW5kZXJpbmdIaW50cygpIHsKLSAgICAgICAgcmV0dXJuIGhpbnRzOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJva2UgZ2V0U3Ryb2tlKCkgewotICAgICAgICByZXR1cm4gc3Ryb2tlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0gZ2V0VHJhbnNmb3JtKCkgewotICAgICAgICByZXR1cm4gKEFmZmluZVRyYW5zZm9ybSl0cmFuc2Zvcm0uY2xvbmUoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBoaXQoUmVjdGFuZ2xlIHJlY3QsIFNoYXBlIHMsIGJvb2xlYW4gb25TdHJva2UpIHsKLSAgICAgICAgLy9UT0RPOiBJbXBsZW1lbnQgbWV0aG9kLi4uLgotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0KLQotCi0gICAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgotICAgICAqCi0gICAgICogIFRyYW5zZm9ybWF0aW9uIG1ldGhvZHMKLSAgICAgKgotICAgICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCByb3RhdGUoZG91YmxlIHRoZXRhKSB7Ci0gICAgICAgIHRyYW5zZm9ybS5yb3RhdGUodGhldGEpOwotICAgICAgICB0cmFuc2Zvcm0uZ2V0TWF0cml4KG1hdHJpeCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgcm90YXRlKGRvdWJsZSB0aGV0YSwgZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0gICAgICAgIHRyYW5zZm9ybS5yb3RhdGUodGhldGEsIHgsIHkpOwotICAgICAgICB0cmFuc2Zvcm0uZ2V0TWF0cml4KG1hdHJpeCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2NhbGUoZG91YmxlIHN4LCBkb3VibGUgc3kpIHsKLSAgICAgICAgdHJhbnNmb3JtLnNjYWxlKHN4LCBzeSk7Ci0gICAgICAgIHRyYW5zZm9ybS5nZXRNYXRyaXgobWF0cml4KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzaGVhcihkb3VibGUgc2h4LCBkb3VibGUgc2h5KSB7Ci0gICAgICAgIHRyYW5zZm9ybS5zaGVhcihzaHgsIHNoeSk7Ci0gICAgICAgIHRyYW5zZm9ybS5nZXRNYXRyaXgobWF0cml4KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB0cmFuc2Zvcm0oQWZmaW5lVHJhbnNmb3JtIGF0KSB7Ci0gICAgICAgIHRyYW5zZm9ybS5jb25jYXRlbmF0ZShhdCk7Ci0gICAgICAgIHRyYW5zZm9ybS5nZXRNYXRyaXgobWF0cml4KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB0cmFuc2xhdGUoZG91YmxlIHR4LCBkb3VibGUgdHkpIHsKLSAgICAgICAgaWYgKGRlYnVnT3V0cHV0KSB7Ci0gICAgICAgICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIkNvbW1vbkdyYXBoaWNzMkQudHJhbnNsYXRlKCIrdHgrIiwgIit0eSsiKSIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgICAgICB9Ci0KLSAgICAgICAgdHJhbnNmb3JtLnRyYW5zbGF0ZSh0eCwgdHkpOwotICAgICAgICB0cmFuc2Zvcm0uZ2V0TWF0cml4KG1hdHJpeCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgdHJhbnNsYXRlKGludCB0eCwgaW50IHR5KSB7Ci0gICAgICAgIGlmIChkZWJ1Z091dHB1dCkgewotICAgICAgICAgICAgU3lzdGVtLmVyci5wcmludGxuKCJDb21tb25HcmFwaGljczJELnRyYW5zbGF0ZSgiK3R4KyIsICIrdHkrIikiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQKLSAgICAgICAgfQotCi0gICAgICAgIHRyYW5zZm9ybS50cmFuc2xhdGUodHgsIHR5KTsKLSAgICAgICAgdHJhbnNmb3JtLmdldE1hdHJpeChtYXRyaXgpOwotICAgIH0KLQotCi0KLQotICAgIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKLSAgICAgKgotICAgICAqICBTZXQgbWV0aG9kcwotICAgICAqCi0gICAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEJhY2tncm91bmQoQ29sb3IgY29sb3IpIHsKLSAgICAgICAgYmdDb2xvciA9IGNvbG9yOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldENsaXAoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQpIHsKLSAgICAgICAgc2V0Q2xpcChuZXcgUmVjdGFuZ2xlKHgsIHksIHdpZHRoLCBoZWlnaHQpKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRDbGlwKFNoYXBlIHMpIHsKLSAgICAgICAgaWYgKHMgPT0gbnVsbCkgewotICAgICAgICAgICAgc2V0VHJhbnNmb3JtZWRDbGlwKG51bGwpOwotICAgICAgICAgICAgaWYgKGRlYnVnT3V0cHV0KSB7Ci0gICAgICAgICAgICAgICAgU3lzdGVtLmVyci5wcmludGxuKCJDb21tb25HcmFwaGljczJELnNldENsaXAobnVsbCkiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRlYnVnT3V0cHV0KSB7Ci0gICAgICAgICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIkNvbW1vbkdyYXBoaWNzMkQuc2V0Q2xpcCgiK3MuZ2V0Qm91bmRzKCkrIikiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAocyBpbnN0YW5jZW9mIE11bHRpUmVjdEFyZWEpIHsKLSAgICAgICAgICAgIE11bHRpUmVjdEFyZWEgbmNsaXAgPSBuZXcgTXVsdGlSZWN0QXJlYSgoTXVsdGlSZWN0QXJlYSlzKTsKLSAgICAgICAgICAgIG5jbGlwLnRyYW5zbGF0ZShNYXRoLnJvdW5kKChmbG9hdCl0cmFuc2Zvcm0uZ2V0VHJhbnNsYXRlWCgpKSwgTWF0aC5yb3VuZCgoZmxvYXQpdHJhbnNmb3JtLmdldFRyYW5zbGF0ZVkoKSkpOwotICAgICAgICAgICAgc2V0VHJhbnNmb3JtZWRDbGlwKG5jbGlwKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGludCB0eXBlID0gdHJhbnNmb3JtLmdldFR5cGUoKTsKLSAgICAgICAgICAgIGlmKHMgaW5zdGFuY2VvZiBSZWN0YW5nbGUgJiYgKHR5cGUgJiAoQWZmaW5lVHJhbnNmb3JtLlRZUEVfSURFTlRJVFkgfAotICAgICAgICAgICAgICAgIEFmZmluZVRyYW5zZm9ybS5UWVBFX1RSQU5TTEFUSU9OKSkgIT0gMCl7Ci0gICAgICAgICAgICAgICAgICAgIE11bHRpUmVjdEFyZWEgbmNsaXAgPSBuZXcgTXVsdGlSZWN0QXJlYSgoUmVjdGFuZ2xlKXMpOwotICAgICAgICAgICAgICAgICAgICBpZih0eXBlID09IEFmZmluZVRyYW5zZm9ybS5UWVBFX1RSQU5TTEFUSU9OKXsKLSAgICAgICAgICAgICAgICAgICAgICAgIG5jbGlwLnRyYW5zbGF0ZSgoaW50KXRyYW5zZm9ybS5nZXRUcmFuc2xhdGVYKCksIChpbnQpdHJhbnNmb3JtLmdldFRyYW5zbGF0ZVkoKSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgc2V0VHJhbnNmb3JtZWRDbGlwKG5jbGlwKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgcyA9IHRyYW5zZm9ybS5jcmVhdGVUcmFuc2Zvcm1lZFNoYXBlKHMpOwotICAgICAgICAgICAgICAgIHNldFRyYW5zZm9ybWVkQ2xpcChqc3IucmFzdGVyaXplKHMsIDAuNSkpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0Q29sb3IoQ29sb3IgY29sb3IpIHsKLSAgICAgICAgaWYgKGNvbG9yICE9IG51bGwpIHsKLSAgICAgICAgICAgIGZnQ29sb3IgPSBjb2xvcjsKLSAgICAgICAgICAgIHBhaW50ID0gY29sb3I7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRDb21wb3NpdGUoQ29tcG9zaXRlIGNvbXBvc2l0ZSkgewotICAgICAgICB0aGlzLmNvbXBvc2l0ZSA9IGNvbXBvc2l0ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRGb250KEZvbnQgZm9udCkgewotICAgICAgICB0aGlzLmZvbnQgPSBmb250OwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFBhaW50KFBhaW50IHBhaW50KSB7Ci0gICAgICAgIGlmIChwYWludCA9PSBudWxsKQotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgCi0gICAgICAgIHRoaXMucGFpbnQgPSBwYWludDsKLSAgICAgICAgaWYgKHBhaW50IGluc3RhbmNlb2YgQ29sb3IpIHsKLSAgICAgICAgICAgIGZnQ29sb3IgPSAoQ29sb3IpcGFpbnQ7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRQYWludE1vZGUoKSB7Ci0gICAgICAgIGNvbXBvc2l0ZSA9IEFscGhhQ29tcG9zaXRlLlNyY092ZXI7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UmVuZGVyaW5nSGludChSZW5kZXJpbmdIaW50cy5LZXkga2V5LCBPYmplY3QgdmFsdWUpIHsKLSAgICAgICAgaGludHMucHV0KGtleSwgdmFsdWUpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFJlbmRlcmluZ0hpbnRzKE1hcDw/LD8+IGhpbnRzKSB7Ci0gICAgICAgIHRoaXMuaGludHMuY2xlYXIoKTsKLSAgICAgICAgdGhpcy5oaW50cy5wdXRBbGwoaGludHMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFN0cm9rZShTdHJva2Ugc3Ryb2tlKSB7Ci0gICAgICAgIHRoaXMuc3Ryb2tlID0gc3Ryb2tlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFRyYW5zZm9ybShBZmZpbmVUcmFuc2Zvcm0gdHJhbnNmb3JtKSB7Ci0gICAgICAgIHRoaXMudHJhbnNmb3JtID0gdHJhbnNmb3JtOwotCi0gICAgICAgIHRyYW5zZm9ybS5nZXRNYXRyaXgobWF0cml4KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRYT1JNb2RlKENvbG9yIGNvbG9yKSB7Ci0gICAgICAgIGNvbXBvc2l0ZSA9IG5ldyBYT1JDb21wb3NpdGUoY29sb3IpOwotICAgIH0KLQotCi0gICAgLy8gUHJvdGVjdGVkIG1ldGhvZHMKLSAgICBwcm90ZWN0ZWQgdm9pZCBzZXRUcmFuc2Zvcm1lZENsaXAoTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0gICAgICAgIHRoaXMuY2xpcCA9IGNsaXA7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhpcyBtZXRob2QgZmlsbHMgdGhlIGdpdmVuIE11bHRpUmVjdEFyZWEgd2l0aCBjdXJyZW50IHBhaW50LgotICAgICAqIEl0IGNhbGxzIGZpbGxNdWx0aVJlY3RBcmVhQ29sb3IgYW5kIGZpbGxNdWx0aVJlY3RBcmVhUGFpbnQgCi0gICAgICogbWV0aG9kcyBkZXBlbmRpbmcgb24gdGhlIHR5cGUgb2YgY3VycmVudCBwYWludC4KLSAgICAgKiBAcGFyYW0gbXJhIE11bHRpUmVjdEFyZWEgdG8gZmlsbAotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIGZpbGxNdWx0aVJlY3RBcmVhKE11bHRpUmVjdEFyZWEgbXJhKSB7Ci0gICAgICAgIGlmIChjbGlwICE9IG51bGwpIHsKLSAgICAgICAgICAgIG1yYS5pbnRlcnNlY3QoY2xpcCk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBSZXR1cm4gaWYgYWxsIHN0dWZmIGlzIGNsaXBwZWQKLSAgICAgICAgaWYgKG1yYS5yZWN0WzBdIDwgNSkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRlYnVnT3V0cHV0KSB7Ci0gICAgICAgICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIkNvbW1vbkdyYXBoaWNzMkQuZmlsbE11bHRpUmVjdEFyZWEoIittcmErIikiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAocGFpbnQgaW5zdGFuY2VvZiBDb2xvcil7Ci0gICAgICAgICAgICBmaWxsTXVsdGlSZWN0QXJlYUNvbG9yKG1yYSk7Ci0gICAgICAgIH1lbHNlewotICAgICAgICAgICAgZmlsbE11bHRpUmVjdEFyZWFQYWludChtcmEpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhpcyBtZXRob2QgZmlsbHMgdGhlIGdpdmVuIE11bHRpUmVjdEFyZWEgd2l0aCBzb2xpZCBjb2xvci4KLSAgICAgKiBAcGFyYW0gbXJhIE11bHRpUmVjdEFyZWEgdG8gZmlsbAotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIGZpbGxNdWx0aVJlY3RBcmVhQ29sb3IoTXVsdGlSZWN0QXJlYSBtcmEpIHsKLSAgICAgICAgZmlsbE11bHRpUmVjdEFyZWFQYWludChtcmEpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoaXMgbWV0aG9kIGZpbGxzIHRoZSBnaXZlbiBNdWx0aVJlY3RBcmVhIHdpdGggYW55IHBhaW50LgotICAgICAqIEBwYXJhbSBtcmEgTXVsdGlSZWN0QXJlYSB0byBmaWxsCi0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgZmlsbE11bHRpUmVjdEFyZWFQYWludChNdWx0aVJlY3RBcmVhIG1yYSkgewotICAgICAgICBSZWN0YW5nbGUgcmVjID0gbXJhLmdldEJvdW5kcygpOwotICAgICAgICBpbnQgeCA9IHJlYy54OwotICAgICAgICBpbnQgeSA9IHJlYy55OwotICAgICAgICBpbnQgdyA9IHJlYy53aWR0aDsKLSAgICAgICAgaW50IGggPSByZWMuaGVpZ2h0OwotICAgICAgICBpZih3IDw9IDAgfHwgaCA8PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgUGFpbnRDb250ZXh0IHBjID0gcGFpbnQuY3JlYXRlQ29udGV4dChudWxsLCByZWMsIHJlYywgdHJhbnNmb3JtLCBoaW50cyk7Ci0gICAgICAgIFJhc3RlciByID0gcGMuZ2V0UmFzdGVyKHgsIHksIHcsIGgpOwotICAgICAgICBXcml0YWJsZVJhc3RlciB3cjsKLSAgICAgICAgaWYociBpbnN0YW5jZW9mIFdyaXRhYmxlUmFzdGVyKXsKLSAgICAgICAgICAgIHdyID0gKFdyaXRhYmxlUmFzdGVyKSByOwotICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgIHdyID0gci5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIoKTsKLSAgICAgICAgICAgIHdyLnNldFJlY3Qocik7Ci0gICAgICAgIH0KLSAgICAgICAgU3VyZmFjZSBzcmNTdXJmID0gbmV3IEltYWdlU3VyZmFjZShwYy5nZXRDb2xvck1vZGVsKCksIHdyKTsKLSAgICAgICAgYmxpdHRlci5ibGl0KDAsIDAsIHNyY1N1cmYsIHgsIHksIGRzdFN1cmYsIHcsIGgsCi0gICAgICAgICAgICAgICAgY29tcG9zaXRlLCBudWxsLCBtcmEpOwotICAgICAgICBzcmNTdXJmLmRpc3Bvc2UoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb3BpZXMgZ3JhcGhpY3MgY2xhc3MgZmllbGRzLiAKLSAgICAgKiBVc2VkIGluIGNyZWF0ZSBtZXRob2QKLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY29weSBHcmFwaGljcyBjbGFzcyB0byBjb3B5Ci0gICAgICovCi0gICAgcHJvdGVjdGVkIHZvaWQgY29weUludGVybmFsRmllbGRzKENvbW1vbkdyYXBoaWNzMkQgY29weSkgewotICAgICAgICBpZiAoY2xpcCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBjb3B5LnNldFRyYW5zZm9ybWVkQ2xpcChudWxsKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvcHkuc2V0VHJhbnNmb3JtZWRDbGlwKG5ldyBNdWx0aVJlY3RBcmVhKGNsaXApKTsKLSAgICAgICAgfQotICAgICAgICBjb3B5LnNldEJhY2tncm91bmQoYmdDb2xvcik7Ci0gICAgICAgIGNvcHkuc2V0Q29sb3IoZmdDb2xvcik7Ci0gICAgICAgIGNvcHkuc2V0UGFpbnQocGFpbnQpOwotICAgICAgICBjb3B5LnNldENvbXBvc2l0ZShjb21wb3NpdGUpOwotICAgICAgICBjb3B5LnNldFN0cm9rZShzdHJva2UpOwotICAgICAgICBjb3B5LnNldEZvbnQoZm9udCk7Ci0gICAgICAgIGNvcHkuc2V0VHJhbnNmb3JtKG5ldyBBZmZpbmVUcmFuc2Zvcm0odHJhbnNmb3JtKSk7Ci0gICAgICAgIC8vY29weS5vcmlnVHJhbnNmb3JtID0gbmV3IEFmZmluZVRyYW5zZm9ybShvcmlnVHJhbnNmb3JtKTsKLSAgICAgICAgY29weS5vcmlnUG9pbnQgPSBuZXcgUG9pbnQob3JpZ1BvaW50KTsKLSAgICB9Ci19ClwgTm8gbmV3bGluZSBhdCBlbmQgb2YgZmlsZQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvQ29tbW9uR3JhcGhpY3MyREZhY3RvcnkuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0NvbW1vbkdyYXBoaWNzMkRGYWN0b3J5LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI3ZTNlZjAuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvQ29tbW9uR3JhcGhpY3MyREZhY3RvcnkuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDc4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgQWxleGV5IEEuIFBldHJlbmtvLCBJbHlhIFMuIE9rb21pbgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2w7Ci0KLWltcG9ydCBqYXZhLmF3dC5Gb250OwotaW1wb3J0IGphdmEuYXd0LkZvbnRNZXRyaWNzOwotaW1wb3J0IGphdmEuYXd0LnBlZXIuRm9udFBlZXI7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuRm9udE1ldHJpY3NJbXBsOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLkdyYXBoaWNzRmFjdG9yeTsKLQotLyoqCi0gKiBDb21tb24gR3JhcGhpY3NGYWN0b3J5IGltcGxlbWVudGF0aW9uCi0gKgotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgQ29tbW9uR3JhcGhpY3MyREZhY3RvcnkgaW1wbGVtZW50cyBHcmFwaGljc0ZhY3RvcnkgewotICAgIAotICAgIC8vIHN0YXRpYyBpbnN0YW5jZSBvZiBDb21tb25HcmFwaGljczJERmFjdG9yeQotICAgIHB1YmxpYyBzdGF0aWMgQ29tbW9uR3JhcGhpY3MyREZhY3RvcnkgaW5zdDsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgRm9udE1ldHJpY3Mgb2JqZWN0IHRoYXQga2VlcHMgbWV0cmljcyBvZiB0aGUgc3BlY2lmaWVkIGZvbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGZvbnQgc3BlY2lmaWVkIEZvbnQKLSAgICAgKiBAcmV0dXJuIEZvbnRNZXRyaWNzIG9iamVjdCBjb3JyZXNwb25kaW5nIHRvIHRoZSBzcGVjaWZpZWQgRm9udCBvYmplY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgRm9udE1ldHJpY3MgZ2V0Rm9udE1ldHJpY3MoRm9udCBmb250KSB7Ci0gICAgICAgIEZvbnRNZXRyaWNzIGZtOwotICAgICAgICBmb3IgKEZvbnRNZXRyaWNzIGVsZW1lbnQgOiBjYWNoZUZNKSB7Ci0gICAgICAgICAgICBmbSA9IGVsZW1lbnQ7Ci0gICAgICAgICAgICBpZiAoZm0gPT0gbnVsbCl7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChmbS5nZXRGb250KCkuZXF1YWxzKGZvbnQpKXsKLSAgICAgICAgICAgICAgICByZXR1cm4gZm07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgZm0gPSBuZXcgRm9udE1ldHJpY3NJbXBsKGZvbnQpOwotCi0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkoY2FjaGVGTSwgMCwgY2FjaGVGTSwgMSwgY2FjaGVGTS5sZW5ndGggLTEpOwotICAgICAgICBjYWNoZUZNWzBdID0gZm07Ci0KLSAgICAgICAgcmV0dXJuIGZtOwotICAgIH0KLSAgICAvLyBGb250IG1ldGhvZHMKLQotICAgIHB1YmxpYyBGb250UGVlciBnZXRGb250UGVlcihGb250IGZvbnQpIHsKLSAgICAgICAgcmV0dXJuIGdldEZvbnRNYW5hZ2VyKCkuZ2V0Rm9udFBlZXIoZm9udC5nZXROYW1lKCksIGZvbnQuZ2V0U3R5bGUoKSwgZm9udC5nZXRTaXplKCkpOwotICAgIH0KLSAgICAKLSAgICAvKioKLSAgICAgKiBFbWJlZHMgZm9udCBmcm9tIGdpbGUgd2l0aCBzcGVjaWZpZWQgcGF0aCBpbnRvIHRoZSBzeXN0ZW0uIAotICAgICAqIAotICAgICAqIEBwYXJhbSBmb250RmlsZVBhdGggcGF0aCB0byB0aGUgZm9udCBmaWxlIAotICAgICAqIEByZXR1cm4gRm9udCBvYmplY3QgdGhhdCB3YXMgY3JlYXRlZCBmcm9tIHRoZSBmaWxlLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBGb250IGVtYmVkRm9udChTdHJpbmcgZm9udEZpbGVQYXRoKTsKLQotfQpcIE5vIG5ld2xpbmUgYXQgZW5kIG9mIGZpbGUKZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0NvbW1vbkdyYXBoaWNzRW52aXJvbm1lbnQuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0NvbW1vbkdyYXBoaWNzRW52aXJvbm1lbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNWM3OGU1MC4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9Db21tb25HcmFwaGljc0Vudmlyb25tZW50LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2NyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbywgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2w7Ci0KLWltcG9ydCBqYXZhLmF3dC5Gb250OwotaW1wb3J0IGphdmEuYXd0LkdyYXBoaWNzMkQ7Ci1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3NFbnZpcm9ubWVudDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2UuQnVmZmVyZWRJbWFnZUdyYXBoaWNzMkQ7Ci0KLS8qKgotICogQ29tbW9uIEdyYXBoaWNzRW52aXJvbm1lbnQgaW1wbGVtZW50YXRpb24KLSAqCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBDb21tb25HcmFwaGljc0Vudmlyb25tZW50IGV4dGVuZHMgR3JhcGhpY3NFbnZpcm9ubWVudCB7Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR3JhcGhpY3MyRCBjcmVhdGVHcmFwaGljcyhCdWZmZXJlZEltYWdlIGJ1ZmZlcmVkSW1hZ2UpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBCdWZmZXJlZEltYWdlR3JhcGhpY3MyRChidWZmZXJlZEltYWdlKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0QXZhaWxhYmxlRm9udEZhbWlseU5hbWVzKExvY2FsZSBsb2NhbGUpIHsKLSAgICAgICAgRm9udFtdIGZvbnRzID0gZ2V0QWxsRm9udHMoKTsKLSAgICAgICAgQXJyYXlMaXN0PFN0cmluZz4gZmFtaWx5TmFtZXMgPSBuZXcgQXJyYXlMaXN0PFN0cmluZz4oKTsKLQotICAgICAgICBmb3IgKEZvbnQgZWxlbWVudCA6IGZvbnRzKSB7Ci0gICAgICAgICAgICBTdHJpbmcgbmFtZSA9IGVsZW1lbnQuZ2V0RmFtaWx5KGxvY2FsZSk7Ci0gICAgICAgICAgICBpZiAoIWZhbWlseU5hbWVzLmNvbnRhaW5zKG5hbWUpKSB7Ci0gICAgICAgICAgICAgICAgZmFtaWx5TmFtZXMuYWRkKG5hbWUpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGZhbWlseU5hbWVzLnRvQXJyYXkobmV3IFN0cmluZ1tmYW1pbHlOYW1lcy5zaXplKCldKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgRm9udFtdIGdldEFsbEZvbnRzKCkgewotICAgICAgICByZXR1cm4gQ29tbW9uR3JhcGhpY3MyREZhY3RvcnkuaW5zdC5nZXRGb250TWFuYWdlcigpLmdldEFsbEZvbnRzKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZ1tdIGdldEF2YWlsYWJsZUZvbnRGYW1pbHlOYW1lcygpIHsKLSAgICAgICAgcmV0dXJuIENvbW1vbkdyYXBoaWNzMkRGYWN0b3J5Lmluc3QuZ2V0Rm9udE1hbmFnZXIoKS5nZXRBbGxGYW1pbGllcygpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0Nyb3NzaW5nLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9Dcm9zc2luZy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBhZTdmYjBlLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0Nyb3NzaW5nLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4ODkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2w7Ci0KLWltcG9ydCBqYXZhLmF3dC5TaGFwZTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBhdGhJdGVyYXRvcjsKLQotcHVibGljIGNsYXNzIENyb3NzaW5nIHsKLQotICAgIC8qKgotICAgICAqIEFsbG93YWJsZSB0b2xlcmFuY2UgZm9yIGJvdW5kcyBjb21wYXJpc29uCi0gICAgICovCi0gICAgc3RhdGljIGZpbmFsIGRvdWJsZSBERUxUQSA9IDFFLTU7Ci0gICAgCi0gICAgLyoqCi0gICAgICogSWYgcm9vdHMgaGF2ZSBkaXN0YW5jZSBsZXNzIHRoZW4gPGNvZGU+Uk9PVF9ERUxUQTwvY29kZT4gdGhleSBhcmUgZG91YmxlCi0gICAgICovCi0gICAgc3RhdGljIGZpbmFsIGRvdWJsZSBST09UX0RFTFRBID0gMUUtMTA7Ci0gICAgCi0gICAgLyoqCi0gICAgICogUmVjdGFuZ2xlIGNyb3NzIHNlZ21lbnQKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBDUk9TU0lORyA9IDI1NTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBVbmtub3duIGNyb3NzaW5nIHJlc3VsdAotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBpbnQgVU5LTk9XTiA9IDI1NDsKLQotICAgIC8qKgotICAgICAqIFNvbHZlcyBxdWFkcmF0aWMgZXF1YXRpb24KLSAgICAgKiBAcGFyYW0gZXFuIC0gdGhlIGNvZWZmaWNpZW50cyBvZiB0aGUgZXF1YXRpb24KLSAgICAgKiBAcGFyYW0gcmVzIC0gdGhlIHJvb3RzIG9mIHRoZSBlcXVhdGlvbgotICAgICAqIEByZXR1cm4gYSBudW1iZXIgb2Ygcm9vdHMKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBzb2x2ZVF1YWQoZG91YmxlIGVxbltdLCBkb3VibGUgcmVzW10pIHsKLSAgICAgICAgZG91YmxlIGEgPSBlcW5bMl07Ci0gICAgICAgIGRvdWJsZSBiID0gZXFuWzFdOwotICAgICAgICBkb3VibGUgYyA9IGVxblswXTsKLSAgICAgICAgaW50IHJjID0gMDsKLSAgICAgICAgaWYgKGEgPT0gMC4wKSB7Ci0gICAgICAgICAgICBpZiAoYiA9PSAwLjApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gLTE7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXNbcmMrK10gPSAtYyAvIGI7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBkb3VibGUgZCA9IGIgKiBiIC0gNC4wICogYSAqIGM7Ci0gICAgICAgICAgICAvLyBkIDwgMC4wCi0gICAgICAgICAgICBpZiAoZCA8IDAuMCkgewotICAgICAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZCA9IE1hdGguc3FydChkKTsKLSAgICAgICAgICAgIHJlc1tyYysrXSA9ICgtIGIgKyBkKSAvIChhICogMi4wKTsKLSAgICAgICAgICAgIC8vIGQgIT0gMC4wCi0gICAgICAgICAgICBpZiAoZCAhPSAwLjApIHsKLSAgICAgICAgICAgICAgICByZXNbcmMrK10gPSAoLSBiIC0gZCkgLyAoYSAqIDIuMCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZpeFJvb3RzKHJlcywgcmMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNvbHZlcyBjdWJpYyBlcXVhdGlvbgotICAgICAqIEBwYXJhbSBlcW4gLSB0aGUgY29lZmZpY2llbnRzIG9mIHRoZSBlcXVhdGlvbgotICAgICAqIEBwYXJhbSByZXMgLSB0aGUgcm9vdHMgb2YgdGhlIGVxdWF0aW9uCi0gICAgICogQHJldHVybiBhIG51bWJlciBvZiByb290cwotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgaW50IHNvbHZlQ3ViaWMoZG91YmxlIGVxbltdLCBkb3VibGUgcmVzW10pIHsKLSAgICAgICAgZG91YmxlIGQgPSBlcW5bM107Ci0gICAgICAgIGlmIChkID09IDApIHsKLSAgICAgICAgICAgIHJldHVybiBzb2x2ZVF1YWQoZXFuLCByZXMpOwotICAgICAgICB9Ci0gICAgICAgIGRvdWJsZSBhID0gZXFuWzJdIC8gZDsKLSAgICAgICAgZG91YmxlIGIgPSBlcW5bMV0gLyBkOwotICAgICAgICBkb3VibGUgYyA9IGVxblswXSAvIGQ7Ci0gICAgICAgIGludCByYyA9IDA7Ci0KLSAgICAgICAgZG91YmxlIFEgPSAoYSAqIGEgLSAzLjAgKiBiKSAvIDkuMDsKLSAgICAgICAgZG91YmxlIFIgPSAoMi4wICogYSAqIGEgKiBhIC0gOS4wICogYSAqIGIgKyAyNy4wICogYykgLyA1NC4wOwotICAgICAgICBkb3VibGUgUTMgPSBRICogUSAqIFE7Ci0gICAgICAgIGRvdWJsZSBSMiA9IFIgKiBSOwotICAgICAgICBkb3VibGUgbiA9IC0gYSAvIDMuMDsKLQotICAgICAgICBpZiAoUjIgPCBRMykgewotICAgICAgICAgICAgZG91YmxlIHQgPSBNYXRoLmFjb3MoUiAvIE1hdGguc3FydChRMykpIC8gMy4wOwotICAgICAgICAgICAgZG91YmxlIHAgPSAyLjAgKiBNYXRoLlBJIC8gMy4wOwotICAgICAgICAgICAgZG91YmxlIG0gPSAtMi4wICogTWF0aC5zcXJ0KFEpOwotICAgICAgICAgICAgcmVzW3JjKytdID0gbSAqIE1hdGguY29zKHQpICsgbjsKLSAgICAgICAgICAgIHJlc1tyYysrXSA9IG0gKiBNYXRoLmNvcyh0ICsgcCkgKyBuOwotICAgICAgICAgICAgcmVzW3JjKytdID0gbSAqIE1hdGguY29zKHQgLSBwKSArIG47Ci0gICAgICAgIH0gZWxzZSB7Ci0vLyAgICAgICAgICBEZWJ1Zy5wcmludGxuKCJSMiA+PSBRMyAoIiArIFIyICsgIi8iICsgUTMgKyAiKSIpOwotICAgICAgICAgICAgZG91YmxlIEEgPSBNYXRoLnBvdyhNYXRoLmFicyhSKSArIE1hdGguc3FydChSMiAtIFEzKSwgMS4wIC8gMy4wKTsKLSAgICAgICAgICAgIGlmIChSID4gMC4wKSB7Ci0gICAgICAgICAgICAgICAgQSA9IC1BOwotICAgICAgICAgICAgfQotLy8gICAgICAgICAgaWYgKEEgPT0gMC4wKSB7Ci0gICAgICAgICAgICBpZiAoLVJPT1RfREVMVEEgPCBBICYmIEEgPCBST09UX0RFTFRBKSB7Ci0gICAgICAgICAgICAgICAgcmVzW3JjKytdID0gbjsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZG91YmxlIEIgPSBRIC8gQTsKLSAgICAgICAgICAgICAgICByZXNbcmMrK10gPSBBICsgQiArIG47Ci0vLyAgICAgICAgICAgICAgaWYgKFIyID09IFEzKSB7Ci0gICAgICAgICAgICAgICAgZG91YmxlIGRlbHRhID0gUjIgLSBRMzsKLSAgICAgICAgICAgICAgICBpZiAoLVJPT1RfREVMVEEgPCBkZWx0YSAmJiBkZWx0YSA8IFJPT1RfREVMVEEpIHsKLSAgICAgICAgICAgICAgICAgICAgcmVzW3JjKytdID0gLSAoQSArIEIpIC8gMi4wICsgbjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZml4Um9vdHMocmVzLCByYyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRXhjbHVkZXMgZG91YmxlIHJvb3RzLiBSb290cyBhcmUgZG91YmxlIGlmIHRoZXkgbGllcyBlbm91Z2ggY2xvc2Ugd2l0aCBlYWNoIG90aGVyLiAKLSAgICAgKiBAcGFyYW0gcmVzIC0gdGhlIHJvb3RzIAotICAgICAqIEBwYXJhbSByYyAtIHRoZSByb290cyBjb3VudAotICAgICAqIEByZXR1cm4gbmV3IHJvb3RzIGNvdW50Ci0gICAgICovCi0gICAgc3RhdGljIGludCBmaXhSb290cyhkb3VibGUgcmVzW10sIGludCByYykgewotICAgICAgICBpbnQgdGMgPSAwOwotICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgcmM7IGkrKykgewotICAgICAgICAgICAgb3V0OiB7Ci0gICAgICAgICAgICAgICAgZm9yKGludCBqID0gaSArIDE7IGogPCByYzsgaisrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChpc1plcm8ocmVzW2ldIC0gcmVzW2pdKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWsgb3V0OwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJlc1t0YysrXSA9IHJlc1tpXTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdGM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUXVhZEN1cnZlIGNsYXNzIHByb3ZpZGVzIGJhc2ljIGZ1bmN0aW9uYWxpdHkgdG8gZmluZCBjdXJ2ZSBjcm9zc2luZyBhbmQgY2FsY3VsYXRpbmcgYm91bmRzCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBjbGFzcyBRdWFkQ3VydmUgewotCi0gICAgICAgIGRvdWJsZSBheCwgYXksIGJ4LCBieTsKLSAgICAgICAgZG91YmxlIEF4LCBBeSwgQngsIEJ5OwotCi0gICAgICAgIHB1YmxpYyBRdWFkQ3VydmUoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSBjeCwgZG91YmxlIGN5LCBkb3VibGUgeDIsIGRvdWJsZSB5MikgewotICAgICAgICAgICAgYXggPSB4MiAtIHgxOwotICAgICAgICAgICAgYXkgPSB5MiAtIHkxOwotICAgICAgICAgICAgYnggPSBjeCAtIHgxOwotICAgICAgICAgICAgYnkgPSBjeSAtIHkxOwotCi0gICAgICAgICAgICBCeCA9IGJ4ICsgYng7ICAgLy8gQnggPSAyLjAgKiBieAotICAgICAgICAgICAgQXggPSBheCAtIEJ4OyAgIC8vIEF4ID0gYXggLSAyLjAgKiBieAotCi0gICAgICAgICAgICBCeSA9IGJ5ICsgYnk7ICAgLy8gQnkgPSAyLjAgKiBieQotICAgICAgICAgICAgQXkgPSBheSAtIEJ5OyAgIC8vIEF5ID0gYXkgLSAyLjAgKiBieQotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGNyb3NzKGRvdWJsZSByZXNbXSwgaW50IHJjLCBkb3VibGUgcHkxLCBkb3VibGUgcHkyKSB7Ci0gICAgICAgICAgICBpbnQgY3Jvc3MgPSAwOwotCi0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHJjOyBpKyspIHsKLSAgICAgICAgICAgICAgICBkb3VibGUgdCA9IHJlc1tpXTsKLQotICAgICAgICAgICAgICAgIC8vIENVUlZFLU9VVFNJREUKLSAgICAgICAgICAgICAgICBpZiAodCA8IC1ERUxUQSB8fCB0ID4gMSArIERFTFRBKSB7Ci0gICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAvLyBDVVJWRS1TVEFSVAotICAgICAgICAgICAgICAgIGlmICh0IDwgREVMVEEpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHB5MSA8IDAuMCAmJiAoYnggIT0gMC4wID8gYnggOiBheCAtIGJ4KSA8IDAuMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgY3Jvc3MtLTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgLy8gQ1VSVkUtRU5ECi0gICAgICAgICAgICAgICAgaWYgKHQgPiAxIC0gREVMVEEpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHB5MSA8IGF5ICYmIChheCAhPSBieCA/IGF4IC0gYnggOiBieCkgPiAwLjApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNyb3NzKys7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIC8vIENVUlZFLUlOU0lERQotICAgICAgICAgICAgICAgIGRvdWJsZSByeSA9IHQgKiAodCAqIEF5ICsgQnkpOwotICAgICAgICAgICAgICAgIC8vIHJ5ID0gdCAqIHQgKiBBeSArIHQgKiBCeQotICAgICAgICAgICAgICAgIGlmIChyeSA+IHB5MikgewotICAgICAgICAgICAgICAgICAgICBkb3VibGUgcnh0ID0gdCAqIEF4ICsgYng7Ci0gICAgICAgICAgICAgICAgICAgIC8vIHJ4dCA9IDIuMCAqIHQgKiBBeCArIEJ4ID0gMi4wICogdCAqIEF4ICsgMi4wICogYngKLSAgICAgICAgICAgICAgICAgICAgaWYgKHJ4dCA+IC1ERUxUQSAmJiByeHQgPCBERUxUQSkgewotICAgICAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgY3Jvc3MgKz0gcnh0ID4gMC4wID8gMSA6IC0xOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gLy8gZm9yCi0KLSAgICAgICAgICAgIHJldHVybiBjcm9zczsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBzb2x2ZVBvaW50KGRvdWJsZSByZXNbXSwgZG91YmxlIHB4KSB7Ci0gICAgICAgICAgICBkb3VibGUgZXFuW10gPSB7LXB4LCBCeCwgQXh9OwotICAgICAgICAgICAgcmV0dXJuIHNvbHZlUXVhZChlcW4sIHJlcyk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgc29sdmVFeHRyZW0oZG91YmxlIHJlc1tdKSB7Ci0gICAgICAgICAgICBpbnQgcmMgPSAwOwotICAgICAgICAgICAgaWYgKEF4ICE9IDAuMCkgewotICAgICAgICAgICAgICAgIHJlc1tyYysrXSA9IC0gQnggLyAoQXggKyBBeCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoQXkgIT0gMC4wKSB7Ci0gICAgICAgICAgICAgICAgcmVzW3JjKytdID0gLSBCeSAvIChBeSArIEF5KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiByYzsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBhZGRCb3VuZChkb3VibGUgYm91bmRbXSwgaW50IGJjLCBkb3VibGUgcmVzW10sIGludCByYywgZG91YmxlIG1pblgsIGRvdWJsZSBtYXhYLCBib29sZWFuIGNoYW5nZUlkLCBpbnQgaWQpIHsKLSAgICAgICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCByYzsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHQgPSByZXNbaV07Ci0gICAgICAgICAgICAgICAgaWYgKHQgPiAtREVMVEEgJiYgdCA8IDEgKyBERUxUQSkgewotICAgICAgICAgICAgICAgICAgICBkb3VibGUgcnggPSB0ICogKHQgKiBBeCArIEJ4KTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKG1pblggPD0gcnggJiYgcnggPD0gbWF4WCkgewotICAgICAgICAgICAgICAgICAgICAgICAgYm91bmRbYmMrK10gPSB0OwotICAgICAgICAgICAgICAgICAgICAgICAgYm91bmRbYmMrK10gPSByeDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJvdW5kW2JjKytdID0gdCAqICh0ICogQXkgKyBCeSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBib3VuZFtiYysrXSA9IGlkOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGNoYW5nZUlkKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWQrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBiYzsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3ViaWNDdXJ2ZSBjbGFzcyBwcm92aWRlcyBiYXNpYyBmdW5jdGlvbmFsaXR5IHRvIGZpbmQgY3VydmUgY3Jvc3NpbmcgYW5kIGNhbGN1bGF0aW5nIGJvdW5kcwotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgY2xhc3MgQ3ViaWNDdXJ2ZSB7Ci0KLSAgICAgICAgZG91YmxlIGF4LCBheSwgYngsIGJ5LCBjeCwgY3k7Ci0gICAgICAgIGRvdWJsZSBBeCwgQXksIEJ4LCBCeSwgQ3gsIEN5OwotICAgICAgICBkb3VibGUgQXgzLCBCeDI7Ci0KLSAgICAgICAgcHVibGljIEN1YmljQ3VydmUoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSBjeDEsIGRvdWJsZSBjeTEsIGRvdWJsZSBjeDIsIGRvdWJsZSBjeTIsIGRvdWJsZSB4MiwgZG91YmxlIHkyKSB7Ci0gICAgICAgICAgICBheCA9IHgyIC0geDE7Ci0gICAgICAgICAgICBheSA9IHkyIC0geTE7Ci0gICAgICAgICAgICBieCA9IGN4MSAtIHgxOwotICAgICAgICAgICAgYnkgPSBjeTEgLSB5MTsKLSAgICAgICAgICAgIGN4ID0gY3gyIC0geDE7Ci0gICAgICAgICAgICBjeSA9IGN5MiAtIHkxOwotCi0gICAgICAgICAgICBDeCA9IGJ4ICsgYnggKyBieDsgICAgICAgICAgIC8vIEN4ID0gMy4wICogYngKLSAgICAgICAgICAgIEJ4ID0gY3ggKyBjeCArIGN4IC0gQ3ggLSBDeDsgLy8gQnggPSAzLjAgKiBjeCAtIDYuMCAqIGJ4Ci0gICAgICAgICAgICBBeCA9IGF4IC0gQnggLSBDeDsgICAgICAgICAgIC8vIEF4ID0gYXggLSAzLjAgKiBjeCArIDMuMCAqIGJ4Ci0KLSAgICAgICAgICAgIEN5ID0gYnkgKyBieSArIGJ5OyAgICAgICAgICAgLy8gQ3kgPSAzLjAgKiBieQotICAgICAgICAgICAgQnkgPSBjeSArIGN5ICsgY3kgLSBDeSAtIEN5OyAvLyBCeSA9IDMuMCAqIGN5IC0gNi4wICogYnkKLSAgICAgICAgICAgIEF5ID0gYXkgLSBCeSAtIEN5OyAgICAgICAgICAgLy8gQXkgPSBheSAtIDMuMCAqIGN5ICsgMy4wICogYnkKLQotICAgICAgICAgICAgQXgzID0gQXggKyBBeCArIEF4OwotICAgICAgICAgICAgQngyID0gQnggKyBCeDsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBjcm9zcyhkb3VibGUgcmVzW10sIGludCByYywgZG91YmxlIHB5MSwgZG91YmxlIHB5MikgewotICAgICAgICAgICAgaW50IGNyb3NzID0gMDsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcmM7IGkrKykgewotICAgICAgICAgICAgICAgIGRvdWJsZSB0ID0gcmVzW2ldOwotCi0gICAgICAgICAgICAgICAgLy8gQ1VSVkUtT1VUU0lERQotICAgICAgICAgICAgICAgIGlmICh0IDwgLURFTFRBIHx8IHQgPiAxICsgREVMVEEpIHsKLSAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIC8vIENVUlZFLVNUQVJUCi0gICAgICAgICAgICAgICAgaWYgKHQgPCBERUxUQSkgewotICAgICAgICAgICAgICAgICAgICBpZiAocHkxIDwgMC4wICYmIChieCAhPSAwLjAgPyBieCA6IChjeCAhPSBieCA/IGN4IC0gYnggOiBheCAtIGN4KSkgPCAwLjApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNyb3NzLS07Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIC8vIENVUlZFLUVORAotICAgICAgICAgICAgICAgIGlmICh0ID4gMSAtIERFTFRBKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChweTEgPCBheSAmJiAoYXggIT0gY3ggPyBheCAtIGN4IDogKGN4ICE9IGJ4ID8gY3ggLSBieCA6IGJ4KSkgPiAwLjApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNyb3NzKys7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIC8vIENVUlZFLUlOU0lERQotICAgICAgICAgICAgICAgIGRvdWJsZSByeSA9IHQgKiAodCAqICh0ICogQXkgKyBCeSkgKyBDeSk7Ci0gICAgICAgICAgICAgICAgLy8gcnkgPSB0ICogdCAqIHQgKiBBeSArIHQgKiB0ICogQnkgKyB0ICogQ3kKLSAgICAgICAgICAgICAgICBpZiAocnkgPiBweTIpIHsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHJ4dCA9IHQgKiAodCAqIEF4MyArIEJ4MikgKyBDeDsKLSAgICAgICAgICAgICAgICAgICAgLy8gcnh0ID0gMy4wICogdCAqIHQgKiBBeCArIDIuMCAqIHQgKiBCeCArIEN4Ci0gICAgICAgICAgICAgICAgICAgIGlmIChyeHQgPiAtREVMVEEgJiYgcnh0IDwgREVMVEEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJ4dCA9IHQgKiAoQXgzICsgQXgzKSArIEJ4MjsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIHJ4dCA9IDYuMCAqIHQgKiBBeCArIDIuMCAqIEJ4Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAocnh0IDwgLURFTFRBIHx8IHJ4dCA+IERFTFRBKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gSW5mbGVjdGlvbiBwb2ludAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgcnh0ID0gYXg7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgY3Jvc3MgKz0gcnh0ID4gMC4wID8gMSA6IC0xOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gLy9mb3IKLQotICAgICAgICAgICAgcmV0dXJuIGNyb3NzOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHNvbHZlUG9pbnQoZG91YmxlIHJlc1tdLCBkb3VibGUgcHgpIHsKLSAgICAgICAgICAgIGRvdWJsZSBlcW5bXSA9IHstcHgsIEN4LCBCeCwgQXh9OwotICAgICAgICAgICAgcmV0dXJuIHNvbHZlQ3ViaWMoZXFuLCByZXMpOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHNvbHZlRXh0cmVtWChkb3VibGUgcmVzW10pIHsKLSAgICAgICAgICAgIGRvdWJsZSBlcW5bXSA9IHtDeCwgQngyLCBBeDN9OwotICAgICAgICAgICAgcmV0dXJuIHNvbHZlUXVhZChlcW4sIHJlcyk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgc29sdmVFeHRyZW1ZKGRvdWJsZSByZXNbXSkgewotICAgICAgICAgICAgZG91YmxlIGVxbltdID0ge0N5LCBCeSArIEJ5LCBBeSArIEF5ICsgQXl9OwotICAgICAgICAgICAgcmV0dXJuIHNvbHZlUXVhZChlcW4sIHJlcyk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgYWRkQm91bmQoZG91YmxlIGJvdW5kW10sIGludCBiYywgZG91YmxlIHJlc1tdLCBpbnQgcmMsIGRvdWJsZSBtaW5YLCBkb3VibGUgbWF4WCwgYm9vbGVhbiBjaGFuZ2VJZCwgaW50IGlkKSB7Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgcmM7IGkrKykgewotICAgICAgICAgICAgICAgIGRvdWJsZSB0ID0gcmVzW2ldOwotICAgICAgICAgICAgICAgIGlmICh0ID4gLURFTFRBICYmIHQgPCAxICsgREVMVEEpIHsKLSAgICAgICAgICAgICAgICAgICAgZG91YmxlIHJ4ID0gdCAqICh0ICogKHQgKiBBeCArIEJ4KSArIEN4KTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKG1pblggPD0gcnggJiYgcnggPD0gbWF4WCkgewotICAgICAgICAgICAgICAgICAgICAgICAgYm91bmRbYmMrK10gPSB0OwotICAgICAgICAgICAgICAgICAgICAgICAgYm91bmRbYmMrK10gPSByeDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJvdW5kW2JjKytdID0gdCAqICh0ICogKHQgKiBBeSArIEJ5KSArIEN5KTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJvdW5kW2JjKytdID0gaWQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoY2hhbmdlSWQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZCsrOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGJjOwotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGhvdyBtYW55IHRpbWVzIHJheSBmcm9tIHBvaW50ICh4LHkpIGNyb3NzIGxpbmUuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnQgY3Jvc3NMaW5lKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgeDIsIGRvdWJsZSB5MiwgZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0KLSAgICAgICAgLy8gTEVGVC9SSUdIVC9VUC9FTVBUWQotICAgICAgICBpZiAoKHggPCB4MSAmJiB4IDwgeDIpIHx8Ci0gICAgICAgICAgICAoeCA+IHgxICYmIHggPiB4MikgfHwKLSAgICAgICAgICAgICh5ID4geTEgJiYgeSA+IHkyKSB8fAotICAgICAgICAgICAgKHgxID09IHgyKSkKLSAgICAgICAgewotICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBET1dOCi0gICAgICAgIGlmICh5IDwgeTEgJiYgeSA8IHkyKSB7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBJTlNJREUKLSAgICAgICAgICAgIGlmICgoeTIgLSB5MSkgKiAoeCAtIHgxKSAvICh4MiAtIHgxKSA8PSB5IC0geTEpIHsKLSAgICAgICAgICAgICAgICAvLyBJTlNJREUtVVAKLSAgICAgICAgICAgICAgICByZXR1cm4gMDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIFNUQVJUCi0gICAgICAgIGlmICh4ID09IHgxKSB7Ci0gICAgICAgICAgICByZXR1cm4geDEgPCB4MiA/IDAgOiAtMTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIEVORAotICAgICAgICBpZiAoeCA9PSB4MikgewotICAgICAgICAgICAgcmV0dXJuIHgxIDwgeDIgPyAxIDogMDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIElOU0lERS1ET1dOCi0gICAgICAgIHJldHVybiB4MSA8IHgyID8gMSA6IC0xOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgaG93IG1hbnkgdGltZXMgcmF5IGZyb20gcG9pbnQgKHgseSkgY3Jvc3MgcXVhcmQgY3VydmUKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBjcm9zc1F1YWQoZG91YmxlIHgxLCBkb3VibGUgeTEsIGRvdWJsZSBjeCwgZG91YmxlIGN5LCBkb3VibGUgeDIsIGRvdWJsZSB5MiwgZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0KLSAgICAgICAgLy8gTEVGVC9SSUdIVC9VUC9FTVBUWQotICAgICAgICBpZiAoKHggPCB4MSAmJiB4IDwgY3ggJiYgeCA8IHgyKSB8fAotICAgICAgICAgICAgKHggPiB4MSAmJiB4ID4gY3ggJiYgeCA+IHgyKSB8fAotICAgICAgICAgICAgKHkgPiB5MSAmJiB5ID4gY3kgJiYgeSA+IHkyKSB8fAotICAgICAgICAgICAgKHgxID09IGN4ICYmIGN4ID09IHgyKSkKLSAgICAgICAgewotICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBET1dOCi0gICAgICAgIGlmICh5IDwgeTEgJiYgeSA8IGN5ICYmIHkgPCB5MiAmJiB4ICE9IHgxICYmIHggIT0geDIpIHsKLSAgICAgICAgICAgIGlmICh4MSA8IHgyKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHgxIDwgeCAmJiB4IDwgeDIgPyAxIDogMDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB4MiA8IHggJiYgeCA8IHgxID8gLTEgOiAwOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gSU5TSURFCi0gICAgICAgIFF1YWRDdXJ2ZSBjID0gbmV3IFF1YWRDdXJ2ZSh4MSwgeTEsIGN4LCBjeSwgeDIsIHkyKTsKLSAgICAgICAgZG91YmxlIHB4ID0geCAtIHgxOwotICAgICAgICBkb3VibGUgcHkgPSB5IC0geTE7Ci0gICAgICAgIGRvdWJsZSByZXNbXSA9IG5ldyBkb3VibGVbM107Ci0gICAgICAgIGludCByYyA9IGMuc29sdmVQb2ludChyZXMsIHB4KTsKLQotICAgICAgICByZXR1cm4gYy5jcm9zcyhyZXMsIHJjLCBweSwgcHkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgaG93IG1hbnkgdGltZXMgcmF5IGZyb20gcG9pbnQgKHgseSkgY3Jvc3MgY3ViaWMgY3VydmUKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBjcm9zc0N1YmljKGRvdWJsZSB4MSwgZG91YmxlIHkxLCBkb3VibGUgY3gxLCBkb3VibGUgY3kxLCBkb3VibGUgY3gyLCBkb3VibGUgY3kyLCBkb3VibGUgeDIsIGRvdWJsZSB5MiwgZG91YmxlIHgsIGRvdWJsZSB5KSB7Ci0KLSAgICAgICAgLy8gTEVGVC9SSUdIVC9VUC9FTVBUWQotICAgICAgICBpZiAoKHggPCB4MSAmJiB4IDwgY3gxICYmIHggPCBjeDIgJiYgeCA8IHgyKSB8fAotICAgICAgICAgICAgKHggPiB4MSAmJiB4ID4gY3gxICYmIHggPiBjeDIgJiYgeCA+IHgyKSB8fAotICAgICAgICAgICAgKHkgPiB5MSAmJiB5ID4gY3kxICYmIHkgPiBjeTIgJiYgeSA+IHkyKSB8fAotICAgICAgICAgICAgKHgxID09IGN4MSAmJiBjeDEgPT0gY3gyICYmIGN4MiA9PSB4MikpCi0gICAgICAgIHsKLSAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gRE9XTgotICAgICAgICBpZiAoeSA8IHkxICYmIHkgPCBjeTEgJiYgeSA8IGN5MiAmJiB5IDwgeTIgJiYgeCAhPSB4MSAmJiB4ICE9IHgyKSB7Ci0gICAgICAgICAgICBpZiAoeDEgPCB4MikgewotICAgICAgICAgICAgICAgIHJldHVybiB4MSA8IHggJiYgeCA8IHgyID8gMSA6IDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4geDIgPCB4ICYmIHggPCB4MSA/IC0xIDogMDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIElOU0lERQotICAgICAgICBDdWJpY0N1cnZlIGMgPSBuZXcgQ3ViaWNDdXJ2ZSh4MSwgeTEsIGN4MSwgY3kxLCBjeDIsIGN5MiwgeDIsIHkyKTsKLSAgICAgICAgZG91YmxlIHB4ID0geCAtIHgxOwotICAgICAgICBkb3VibGUgcHkgPSB5IC0geTE7Ci0gICAgICAgIGRvdWJsZSByZXNbXSA9IG5ldyBkb3VibGVbM107Ci0gICAgICAgIGludCByYyA9IGMuc29sdmVQb2ludChyZXMsIHB4KTsKLSAgICAgICAgcmV0dXJuIGMuY3Jvc3MocmVzLCByYywgcHksIHB5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGhvdyBtYW55IHRpbWVzIHJheSBmcm9tIHBvaW50ICh4LHkpIGNyb3NzIHBhdGgKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBjcm9zc1BhdGgoUGF0aEl0ZXJhdG9yIHAsIGRvdWJsZSB4LCBkb3VibGUgeSkgewotICAgICAgICBpbnQgY3Jvc3MgPSAwOwotICAgICAgICBkb3VibGUgbXgsIG15LCBjeCwgY3k7Ci0gICAgICAgIG14ID0gbXkgPSBjeCA9IGN5ID0gMC4wOwotICAgICAgICBkb3VibGUgY29vcmRzW10gPSBuZXcgZG91YmxlWzZdOwotCi0gICAgICAgIHdoaWxlICghcC5pc0RvbmUoKSkgewotICAgICAgICAgICAgc3dpdGNoIChwLmN1cnJlbnRTZWdtZW50KGNvb3JkcykpIHsKLSAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19NT1ZFVE86Ci0gICAgICAgICAgICAgICAgaWYgKGN4ICE9IG14IHx8IGN5ICE9IG15KSB7Ci0gICAgICAgICAgICAgICAgICAgIGNyb3NzICs9IGNyb3NzTGluZShjeCwgY3ksIG14LCBteSwgeCwgeSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIG14ID0gY3ggPSBjb29yZHNbMF07Ci0gICAgICAgICAgICAgICAgbXkgPSBjeSA9IGNvb3Jkc1sxXTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19MSU5FVE86Ci0gICAgICAgICAgICAgICAgY3Jvc3MgKz0gY3Jvc3NMaW5lKGN4LCBjeSwgY3ggPSBjb29yZHNbMF0sIGN5ID0gY29vcmRzWzFdLCB4LCB5KTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19RVUFEVE86Ci0gICAgICAgICAgICAgICAgY3Jvc3MgKz0gY3Jvc3NRdWFkKGN4LCBjeSwgY29vcmRzWzBdLCBjb29yZHNbMV0sIGN4ID0gY29vcmRzWzJdLCBjeSA9IGNvb3Jkc1szXSwgeCwgeSk7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfQ1VCSUNUTzoKLSAgICAgICAgICAgICAgICBjcm9zcyArPSBjcm9zc0N1YmljKGN4LCBjeSwgY29vcmRzWzBdLCBjb29yZHNbMV0sIGNvb3Jkc1syXSwgY29vcmRzWzNdLCBjeCA9IGNvb3Jkc1s0XSwgY3kgPSBjb29yZHNbNV0sIHgsIHkpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0NMT1NFOgotICAgICAgICAgICAgICAgIGlmIChjeSAhPSBteSB8fCBjeCAhPSBteCkgewotICAgICAgICAgICAgICAgICAgICBjcm9zcyArPSBjcm9zc0xpbmUoY3gsIGN5LCBjeCA9IG14LCBjeSA9IG15LCB4LCB5KTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwLm5leHQoKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoY3kgIT0gbXkpIHsKLSAgICAgICAgICAgIGNyb3NzICs9IGNyb3NzTGluZShjeCwgY3ksIG14LCBteSwgeCwgeSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGNyb3NzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgaG93IG1hbnkgdGltZXMgcmF5IGZyb20gcG9pbnQgKHgseSkgY3Jvc3Mgc2hhcGUKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBjcm9zc1NoYXBlKFNoYXBlIHMsIGRvdWJsZSB4LCBkb3VibGUgeSkgewotICAgICAgICBpZiAoIXMuZ2V0Qm91bmRzMkQoKS5jb250YWlucyh4LCB5KSkgewotICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGNyb3NzUGF0aChzLmdldFBhdGhJdGVyYXRvcihudWxsKSwgeCwgeSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHZhbHVlIGVub3VnaCBzbWFsbAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgYm9vbGVhbiBpc1plcm8oZG91YmxlIHZhbCkgewotICAgICAgICByZXR1cm4gLURFTFRBIDwgdmFsICYmIHZhbCA8IERFTFRBOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNvcnQgYm91bmQgYXJyYXkKLSAgICAgKi8KLSAgICBzdGF0aWMgdm9pZCBzb3J0Qm91bmQoZG91YmxlIGJvdW5kW10sIGludCBiYykgewotICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgYmMgLSA0OyBpICs9IDQpIHsKLSAgICAgICAgICAgIGludCBrID0gaTsKLSAgICAgICAgICAgIGZvcihpbnQgaiA9IGkgKyA0OyBqIDwgYmM7IGogKz0gNCkgewotICAgICAgICAgICAgICAgIGlmIChib3VuZFtrXSA+IGJvdW5kW2pdKSB7Ci0gICAgICAgICAgICAgICAgICAgIGsgPSBqOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChrICE9IGkpIHsKLSAgICAgICAgICAgICAgICBkb3VibGUgdG1wID0gYm91bmRbaV07Ci0gICAgICAgICAgICAgICAgYm91bmRbaV0gPSBib3VuZFtrXTsKLSAgICAgICAgICAgICAgICBib3VuZFtrXSA9IHRtcDsKLSAgICAgICAgICAgICAgICB0bXAgPSBib3VuZFtpICsgMV07Ci0gICAgICAgICAgICAgICAgYm91bmRbaSArIDFdID0gYm91bmRbayArIDFdOwotICAgICAgICAgICAgICAgIGJvdW5kW2sgKyAxXSA9IHRtcDsKLSAgICAgICAgICAgICAgICB0bXAgPSBib3VuZFtpICsgMl07Ci0gICAgICAgICAgICAgICAgYm91bmRbaSArIDJdID0gYm91bmRbayArIDJdOwotICAgICAgICAgICAgICAgIGJvdW5kW2sgKyAyXSA9IHRtcDsKLSAgICAgICAgICAgICAgICB0bXAgPSBib3VuZFtpICsgM107Ci0gICAgICAgICAgICAgICAgYm91bmRbaSArIDNdID0gYm91bmRbayArIDNdOwotICAgICAgICAgICAgICAgIGJvdW5kW2sgKyAzXSA9IHRtcDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGFyZSBib3VuZHMgaW50ZXJzZWN0IG9yIG5vdCBpbnRlcnNlY3QgcmVjdGFuZ2xlIAotICAgICAqLwotICAgIHN0YXRpYyBpbnQgY3Jvc3NCb3VuZChkb3VibGUgYm91bmRbXSwgaW50IGJjLCBkb3VibGUgcHkxLCBkb3VibGUgcHkyKSB7Ci0KLSAgICAgICAgLy8gTEVGVC9SSUdIVAotICAgICAgICBpZiAoYmMgPT0gMCkgewotICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBDaGVjayBZIGNvb3JkaW5hdGUKLSAgICAgICAgaW50IHVwID0gMDsKLSAgICAgICAgaW50IGRvd24gPSAwOwotICAgICAgICBmb3IoaW50IGkgPSAyOyBpIDwgYmM7IGkgKz0gNCkgewotICAgICAgICAgICAgaWYgKGJvdW5kW2ldIDwgcHkxKSB7Ci0gICAgICAgICAgICAgICAgdXArKzsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChib3VuZFtpXSA+IHB5MikgewotICAgICAgICAgICAgICAgIGRvd24rKzsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBDUk9TU0lORzsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIFVQCi0gICAgICAgIGlmIChkb3duID09IDApIHsKLSAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHVwICE9IDApIHsKLSAgICAgICAgICAgIC8vIGJjID49IDIKLSAgICAgICAgICAgIHNvcnRCb3VuZChib3VuZCwgYmMpOwotICAgICAgICAgICAgYm9vbGVhbiBzaWduID0gYm91bmRbMl0gPiBweTI7Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSA2OyBpIDwgYmM7IGkgKz0gNCkgewotICAgICAgICAgICAgICAgIGJvb2xlYW4gc2lnbjIgPSBib3VuZFtpXSA+IHB5MjsKLSAgICAgICAgICAgICAgICBpZiAoc2lnbiAhPSBzaWduMiAmJiBib3VuZFtpICsgMV0gIT0gYm91bmRbaSAtIDNdKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBDUk9TU0lORzsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgc2lnbiA9IHNpZ24yOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBVTktOT1dOOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgaG93IG1hbnkgdGltZXMgcmVjdGFuZ2xlIHN0cmlwZSBjcm9zcyBsaW5lIG9yIHRoZSBhcmUgaW50ZXJzZWN0Ci0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnQgaW50ZXJzZWN0TGluZShkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIHgyLCBkb3VibGUgeTIsIGRvdWJsZSByeDEsIGRvdWJsZSByeTEsIGRvdWJsZSByeDIsIGRvdWJsZSByeTIpIHsKLQotICAgICAgICAvLyBMRUZUL1JJR0hUL1VQCi0gICAgICAgIGlmICgocngyIDwgeDEgJiYgcngyIDwgeDIpIHx8Ci0gICAgICAgICAgICAocngxID4geDEgJiYgcngxID4geDIpIHx8Ci0gICAgICAgICAgICAocnkxID4geTEgJiYgcnkxID4geTIpKQotICAgICAgICB7Ci0gICAgICAgICAgICByZXR1cm4gMDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIERPV04KLSAgICAgICAgaWYgKHJ5MiA8IHkxICYmIHJ5MiA8IHkyKSB7Ci0gICAgICAgIH0gZWxzZSB7Ci0KLSAgICAgICAgICAgIC8vIElOU0lERQotICAgICAgICAgICAgaWYgKHgxID09IHgyKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENST1NTSU5HOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBCdWlsZCBib3VuZAotICAgICAgICAgICAgZG91YmxlIGJ4MSwgYngyOwotICAgICAgICAgICAgaWYgKHgxIDwgeDIpIHsKLSAgICAgICAgICAgICAgICBieDEgPSB4MSA8IHJ4MSA/IHJ4MSA6IHgxOwotICAgICAgICAgICAgICAgIGJ4MiA9IHgyIDwgcngyID8geDIgOiByeDI7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGJ4MSA9IHgyIDwgcngxID8gcngxIDogeDI7Ci0gICAgICAgICAgICAgICAgYngyID0geDEgPCByeDIgPyB4MSA6IHJ4MjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGRvdWJsZSBrID0gKHkyIC0geTEpIC8gKHgyIC0geDEpOwotICAgICAgICAgICAgZG91YmxlIGJ5MSA9IGsgKiAoYngxIC0geDEpICsgeTE7Ci0gICAgICAgICAgICBkb3VibGUgYnkyID0gayAqIChieDIgLSB4MSkgKyB5MTsKLQotICAgICAgICAgICAgLy8gQk9VTkQtVVAKLSAgICAgICAgICAgIGlmIChieTEgPCByeTEgJiYgYnkyIDwgcnkxKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIEJPVU5ELURPV04KLSAgICAgICAgICAgIGlmIChieTEgPiByeTIgJiYgYnkyID4gcnkyKSB7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJldHVybiBDUk9TU0lORzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIEVNUFRZCi0gICAgICAgIGlmICh4MSA9PSB4MikgewotICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBDVVJWRS1TVEFSVAotICAgICAgICBpZiAocngxID09IHgxKSB7Ci0gICAgICAgICAgICByZXR1cm4geDEgPCB4MiA/IDAgOiAtMTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIENVUlZFLUVORAotICAgICAgICBpZiAocngxID09IHgyKSB7Ci0gICAgICAgICAgICByZXR1cm4geDEgPCB4MiA/IDEgOiAwOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHgxIDwgeDIpIHsKLSAgICAgICAgICAgIHJldHVybiB4MSA8IHJ4MSAmJiByeDEgPCB4MiA/IDEgOiAwOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB4MiA8IHJ4MSAmJiByeDEgPCB4MSA/IC0xIDogMDsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgaG93IG1hbnkgdGltZXMgcmVjdGFuZ2xlIHN0cmlwZSBjcm9zcyBxdWFkIGN1cnZlIG9yIHRoZSBhcmUgaW50ZXJzZWN0Ci0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnQgaW50ZXJzZWN0UXVhZChkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIGN4LCBkb3VibGUgY3ksIGRvdWJsZSB4MiwgZG91YmxlIHkyLCBkb3VibGUgcngxLCBkb3VibGUgcnkxLCBkb3VibGUgcngyLCBkb3VibGUgcnkyKSB7Ci0KLSAgICAgICAgLy8gTEVGVC9SSUdIVC9VUCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSAgICAgICAgaWYgKChyeDIgPCB4MSAmJiByeDIgPCBjeCAmJiByeDIgPCB4MikgfHwKLSAgICAgICAgICAgIChyeDEgPiB4MSAmJiByeDEgPiBjeCAmJiByeDEgPiB4MikgfHwKLSAgICAgICAgICAgIChyeTEgPiB5MSAmJiByeTEgPiBjeSAmJiByeTEgPiB5MikpCi0gICAgICAgIHsKLSAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gRE9XTiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSAgICAgICAgaWYgKHJ5MiA8IHkxICYmIHJ5MiA8IGN5ICYmIHJ5MiA8IHkyICYmIHJ4MSAhPSB4MSAmJiByeDEgIT0geDIpIHsKLSAgICAgICAgICAgIGlmICh4MSA8IHgyKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHgxIDwgcngxICYmIHJ4MSA8IHgyID8gMSA6IDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4geDIgPCByeDEgJiYgcngxIDwgeDEgPyAtMSA6IDA7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBJTlNJREUgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICAgICAgICBRdWFkQ3VydmUgYyA9IG5ldyBRdWFkQ3VydmUoeDEsIHkxLCBjeCwgY3ksIHgyLCB5Mik7Ci0gICAgICAgIGRvdWJsZSBweDEgPSByeDEgLSB4MTsKLSAgICAgICAgZG91YmxlIHB5MSA9IHJ5MSAtIHkxOwotICAgICAgICBkb3VibGUgcHgyID0gcngyIC0geDE7Ci0gICAgICAgIGRvdWJsZSBweTIgPSByeTIgLSB5MTsKLQotICAgICAgICBkb3VibGUgcmVzMVtdID0gbmV3IGRvdWJsZVszXTsKLSAgICAgICAgZG91YmxlIHJlczJbXSA9IG5ldyBkb3VibGVbM107Ci0gICAgICAgIGludCByYzEgPSBjLnNvbHZlUG9pbnQocmVzMSwgcHgxKTsKLSAgICAgICAgaW50IHJjMiA9IGMuc29sdmVQb2ludChyZXMyLCBweDIpOwotCi0gICAgICAgIC8vIElOU0lERS1MRUZUL1JJR0hUCi0gICAgICAgIGlmIChyYzEgPT0gMCAmJiByYzIgPT0gMCkgewotICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBCdWlsZCBib3VuZCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICAgICAgICBkb3VibGUgbWluWCA9IHB4MSAtIERFTFRBOwotICAgICAgICBkb3VibGUgbWF4WCA9IHB4MiArIERFTFRBOwotICAgICAgICBkb3VibGUgYm91bmRbXSA9IG5ldyBkb3VibGVbMjhdOwotICAgICAgICBpbnQgYmMgPSAwOwotICAgICAgICAvLyBBZGQgcm9vdHMKLSAgICAgICAgYmMgPSBjLmFkZEJvdW5kKGJvdW5kLCBiYywgcmVzMSwgcmMxLCBtaW5YLCBtYXhYLCBmYWxzZSwgMCk7Ci0gICAgICAgIGJjID0gYy5hZGRCb3VuZChib3VuZCwgYmMsIHJlczIsIHJjMiwgbWluWCwgbWF4WCwgZmFsc2UsIDEpOwotICAgICAgICAvLyBBZGQgZXh0cmVtYWwgcG9pbnRzYAotICAgICAgICByYzIgPSBjLnNvbHZlRXh0cmVtKHJlczIpOwotICAgICAgICBiYyA9IGMuYWRkQm91bmQoYm91bmQsIGJjLCByZXMyLCByYzIsIG1pblgsIG1heFgsIHRydWUsIDIpOwotICAgICAgICAvLyBBZGQgc3RhcnQgYW5kIGVuZAotICAgICAgICBpZiAocngxIDwgeDEgJiYgeDEgPCByeDIpIHsKLSAgICAgICAgICAgIGJvdW5kW2JjKytdID0gMC4wOwotICAgICAgICAgICAgYm91bmRbYmMrK10gPSAwLjA7Ci0gICAgICAgICAgICBib3VuZFtiYysrXSA9IDAuMDsKLSAgICAgICAgICAgIGJvdW5kW2JjKytdID0gNDsKLSAgICAgICAgfQotICAgICAgICBpZiAocngxIDwgeDIgJiYgeDIgPCByeDIpIHsKLSAgICAgICAgICAgIGJvdW5kW2JjKytdID0gMS4wOwotICAgICAgICAgICAgYm91bmRbYmMrK10gPSBjLmF4OwotICAgICAgICAgICAgYm91bmRbYmMrK10gPSBjLmF5OwotICAgICAgICAgICAgYm91bmRbYmMrK10gPSA1OwotICAgICAgICB9Ci0gICAgICAgIC8vIEVuZCBidWlsZCBib3VuZCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSAgICAgICAgaW50IGNyb3NzID0gY3Jvc3NCb3VuZChib3VuZCwgYmMsIHB5MSwgcHkyKTsKLSAgICAgICAgaWYgKGNyb3NzICE9IFVOS05PV04pIHsKLSAgICAgICAgICAgIHJldHVybiBjcm9zczsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gYy5jcm9zcyhyZXMxLCByYzEsIHB5MSwgcHkyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGhvdyBtYW55IHRpbWVzIHJlY3RhbmdsZSBzdHJpcGUgY3Jvc3MgY3ViaWMgY3VydmUgb3IgdGhlIGFyZSBpbnRlcnNlY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBpbnRlcnNlY3RDdWJpYyhkb3VibGUgeDEsIGRvdWJsZSB5MSwgZG91YmxlIGN4MSwgZG91YmxlIGN5MSwgZG91YmxlIGN4MiwgZG91YmxlIGN5MiwgZG91YmxlIHgyLCBkb3VibGUgeTIsIGRvdWJsZSByeDEsIGRvdWJsZSByeTEsIGRvdWJsZSByeDIsIGRvdWJsZSByeTIpIHsKLQotICAgICAgICAvLyBMRUZUL1JJR0hUL1VQCi0gICAgICAgIGlmICgocngyIDwgeDEgJiYgcngyIDwgY3gxICYmIHJ4MiA8IGN4MiAmJiByeDIgPCB4MikgfHwKLSAgICAgICAgICAgIChyeDEgPiB4MSAmJiByeDEgPiBjeDEgJiYgcngxID4gY3gyICYmIHJ4MSA+IHgyKSB8fAotICAgICAgICAgICAgKHJ5MSA+IHkxICYmIHJ5MSA+IGN5MSAmJiByeTEgPiBjeTIgJiYgcnkxID4geTIpKQotICAgICAgICB7Ci0gICAgICAgICAgICByZXR1cm4gMDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIERPV04KLSAgICAgICAgaWYgKHJ5MiA8IHkxICYmIHJ5MiA8IGN5MSAmJiByeTIgPCBjeTIgJiYgcnkyIDwgeTIgJiYgcngxICE9IHgxICYmIHJ4MSAhPSB4MikgewotICAgICAgICAgICAgaWYgKHgxIDwgeDIpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4geDEgPCByeDEgJiYgcngxIDwgeDIgPyAxIDogMDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB4MiA8IHJ4MSAmJiByeDEgPCB4MSA/IC0xIDogMDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIElOU0lERQotICAgICAgICBDdWJpY0N1cnZlIGMgPSBuZXcgQ3ViaWNDdXJ2ZSh4MSwgeTEsIGN4MSwgY3kxLCBjeDIsIGN5MiwgeDIsIHkyKTsKLSAgICAgICAgZG91YmxlIHB4MSA9IHJ4MSAtIHgxOwotICAgICAgICBkb3VibGUgcHkxID0gcnkxIC0geTE7Ci0gICAgICAgIGRvdWJsZSBweDIgPSByeDIgLSB4MTsKLSAgICAgICAgZG91YmxlIHB5MiA9IHJ5MiAtIHkxOwotCi0gICAgICAgIGRvdWJsZSByZXMxW10gPSBuZXcgZG91YmxlWzNdOwotICAgICAgICBkb3VibGUgcmVzMltdID0gbmV3IGRvdWJsZVszXTsKLSAgICAgICAgaW50IHJjMSA9IGMuc29sdmVQb2ludChyZXMxLCBweDEpOwotICAgICAgICBpbnQgcmMyID0gYy5zb2x2ZVBvaW50KHJlczIsIHB4Mik7Ci0KLSAgICAgICAgLy8gTEVGVC9SSUdIVAotICAgICAgICBpZiAocmMxID09IDAgJiYgcmMyID09IDApIHsKLSAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlIG1pblggPSBweDEgLSBERUxUQTsKLSAgICAgICAgZG91YmxlIG1heFggPSBweDIgKyBERUxUQTsKLQotICAgICAgICAvLyBCdWlsZCBib3VuZCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICAgICAgICBkb3VibGUgYm91bmRbXSA9IG5ldyBkb3VibGVbNDBdOwotICAgICAgICBpbnQgYmMgPSAwOwotICAgICAgICAvLyBBZGQgcm9vdHMKLSAgICAgICAgYmMgPSBjLmFkZEJvdW5kKGJvdW5kLCBiYywgcmVzMSwgcmMxLCBtaW5YLCBtYXhYLCBmYWxzZSwgMCk7Ci0gICAgICAgIGJjID0gYy5hZGRCb3VuZChib3VuZCwgYmMsIHJlczIsIHJjMiwgbWluWCwgbWF4WCwgZmFsc2UsIDEpOwotICAgICAgICAvLyBBZGQgZXh0cmltYWwgcG9pbnRzCi0gICAgICAgIHJjMiA9IGMuc29sdmVFeHRyZW1YKHJlczIpOwotICAgICAgICBiYyA9IGMuYWRkQm91bmQoYm91bmQsIGJjLCByZXMyLCByYzIsIG1pblgsIG1heFgsIHRydWUsIDIpOwotICAgICAgICByYzIgPSBjLnNvbHZlRXh0cmVtWShyZXMyKTsKLSAgICAgICAgYmMgPSBjLmFkZEJvdW5kKGJvdW5kLCBiYywgcmVzMiwgcmMyLCBtaW5YLCBtYXhYLCB0cnVlLCA0KTsKLSAgICAgICAgLy8gQWRkIHN0YXJ0IGFuZCBlbmQKLSAgICAgICAgaWYgKHJ4MSA8IHgxICYmIHgxIDwgcngyKSB7Ci0gICAgICAgICAgICBib3VuZFtiYysrXSA9IDAuMDsKLSAgICAgICAgICAgIGJvdW5kW2JjKytdID0gMC4wOwotICAgICAgICAgICAgYm91bmRbYmMrK10gPSAwLjA7Ci0gICAgICAgICAgICBib3VuZFtiYysrXSA9IDY7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHJ4MSA8IHgyICYmIHgyIDwgcngyKSB7Ci0gICAgICAgICAgICBib3VuZFtiYysrXSA9IDEuMDsKLSAgICAgICAgICAgIGJvdW5kW2JjKytdID0gYy5heDsKLSAgICAgICAgICAgIGJvdW5kW2JjKytdID0gYy5heTsKLSAgICAgICAgICAgIGJvdW5kW2JjKytdID0gNzsKLSAgICAgICAgfQotICAgICAgICAvLyBFbmQgYnVpbGQgYm91bmQgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0gICAgICAgIGludCBjcm9zcyA9IGNyb3NzQm91bmQoYm91bmQsIGJjLCBweTEsIHB5Mik7Ci0gICAgICAgIGlmIChjcm9zcyAhPSBVTktOT1dOKSB7Ci0gICAgICAgICAgICByZXR1cm4gY3Jvc3M7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGMuY3Jvc3MocmVzMSwgcmMxLCBweTEsIHB5Mik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBob3cgbWFueSB0aW1lcyByZWN0YW5nbGUgc3RyaXBlIGNyb3NzIHBhdGggb3IgdGhlIGFyZSBpbnRlcnNlY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBpbnRlcnNlY3RQYXRoKFBhdGhJdGVyYXRvciBwLCBkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3LCBkb3VibGUgaCkgewotCi0gICAgICAgIGludCBjcm9zcyA9IDA7Ci0gICAgICAgIGludCBjb3VudDsKLSAgICAgICAgZG91YmxlIG14LCBteSwgY3gsIGN5OwotICAgICAgICBteCA9IG15ID0gY3ggPSBjeSA9IDAuMDsKLSAgICAgICAgZG91YmxlIGNvb3Jkc1tdID0gbmV3IGRvdWJsZVs2XTsKLQotICAgICAgICBkb3VibGUgcngxID0geDsKLSAgICAgICAgZG91YmxlIHJ5MSA9IHk7Ci0gICAgICAgIGRvdWJsZSByeDIgPSB4ICsgdzsKLSAgICAgICAgZG91YmxlIHJ5MiA9IHkgKyBoOwotCi0gICAgICAgIHdoaWxlICghcC5pc0RvbmUoKSkgewotICAgICAgICAgICAgY291bnQgPSAwOwotICAgICAgICAgICAgc3dpdGNoIChwLmN1cnJlbnRTZWdtZW50KGNvb3JkcykpIHsKLSAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19NT1ZFVE86Ci0gICAgICAgICAgICAgICAgaWYgKGN4ICE9IG14IHx8IGN5ICE9IG15KSB7Ci0gICAgICAgICAgICAgICAgICAgIGNvdW50ID0gaW50ZXJzZWN0TGluZShjeCwgY3ksIG14LCBteSwgcngxLCByeTEsIHJ4MiwgcnkyKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbXggPSBjeCA9IGNvb3Jkc1swXTsKLSAgICAgICAgICAgICAgICBteSA9IGN5ID0gY29vcmRzWzFdOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0xJTkVUTzoKLSAgICAgICAgICAgICAgICBjb3VudCA9IGludGVyc2VjdExpbmUoY3gsIGN5LCBjeCA9IGNvb3Jkc1swXSwgY3kgPSBjb29yZHNbMV0sIHJ4MSwgcnkxLCByeDIsIHJ5Mik7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfUVVBRFRPOgotICAgICAgICAgICAgICAgIGNvdW50ID0gaW50ZXJzZWN0UXVhZChjeCwgY3ksIGNvb3Jkc1swXSwgY29vcmRzWzFdLCBjeCA9IGNvb3Jkc1syXSwgY3kgPSBjb29yZHNbM10sIHJ4MSwgcnkxLCByeDIsIHJ5Mik7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfQ1VCSUNUTzoKLSAgICAgICAgICAgICAgICBjb3VudCA9IGludGVyc2VjdEN1YmljKGN4LCBjeSwgY29vcmRzWzBdLCBjb29yZHNbMV0sIGNvb3Jkc1syXSwgY29vcmRzWzNdLCBjeCA9IGNvb3Jkc1s0XSwgY3kgPSBjb29yZHNbNV0sIHJ4MSwgcnkxLCByeDIsIHJ5Mik7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfQ0xPU0U6Ci0gICAgICAgICAgICAgICAgaWYgKGN5ICE9IG15IHx8IGN4ICE9IG14KSB7Ci0gICAgICAgICAgICAgICAgICAgIGNvdW50ID0gaW50ZXJzZWN0TGluZShjeCwgY3ksIG14LCBteSwgcngxLCByeTEsIHJ4MiwgcnkyKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgY3ggPSBteDsKLSAgICAgICAgICAgICAgICBjeSA9IG15OwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGNvdW50ID09IENST1NTSU5HKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIENST1NTSU5HOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY3Jvc3MgKz0gY291bnQ7Ci0gICAgICAgICAgICBwLm5leHQoKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoY3kgIT0gbXkpIHsKLSAgICAgICAgICAgIGNvdW50ID0gaW50ZXJzZWN0TGluZShjeCwgY3ksIG14LCBteSwgcngxLCByeTEsIHJ4MiwgcnkyKTsKLSAgICAgICAgICAgIGlmIChjb3VudCA9PSBDUk9TU0lORykgewotICAgICAgICAgICAgICAgIHJldHVybiBDUk9TU0lORzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNyb3NzICs9IGNvdW50OwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBjcm9zczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGhvdyBtYW55IHRpbWVzIHJlY3RhbmdsZSBzdHJpcGUgY3Jvc3Mgc2hhcGUgb3IgdGhlIGFyZSBpbnRlcnNlY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBpbnRlcnNlY3RTaGFwZShTaGFwZSBzLCBkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3LCBkb3VibGUgaCkgewotICAgICAgICBpZiAoIXMuZ2V0Qm91bmRzMkQoKS5pbnRlcnNlY3RzKHgsIHksIHcsIGgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gMDsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gaW50ZXJzZWN0UGF0aChzLmdldFBhdGhJdGVyYXRvcihudWxsKSwgeCwgeSwgdywgaCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIGNyb3NzIGNvdW50IGNvcnJlc3BvbmQgaW5zaWRlIGxvY2F0aW9uIGZvciBub24gemVybyBwYXRoIHJ1bGUKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGJvb2xlYW4gaXNJbnNpZGVOb25aZXJvKGludCBjcm9zcykgewotICAgICAgICByZXR1cm4gY3Jvc3MgIT0gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgY3Jvc3MgY291bnQgY29ycmVzcG9uZCBpbnNpZGUgbG9jYXRpb24gZm9yIGV2ZW4tb2RkIHBhdGggcnVsZQotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgYm9vbGVhbiBpc0luc2lkZUV2ZW5PZGQoaW50IGNyb3NzKSB7Ci0gICAgICAgIHJldHVybiAoY3Jvc3MgJiAxKSAhPSAwOwotICAgIH0KLX0KXCBObyBuZXdsaW5lIGF0IGVuZCBvZiBmaWxlCmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9HTFZvbGF0aWxlSW1hZ2UuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0dMVm9sYXRpbGVJbWFnZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxNzdiZTIzLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0dMVm9sYXRpbGVJbWFnZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzAgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2w7Ci0KLWltcG9ydCBqYXZhLmF3dC5pbWFnZS4qOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5TdXJmYWNlOwotCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgR0xWb2xhdGlsZUltYWdlIGV4dGVuZHMgVm9sYXRpbGVJbWFnZSB7Ci0KLSAgICBwdWJsaWMgYWJzdHJhY3QgU3VyZmFjZSBnZXRJbWFnZVN1cmZhY2UoKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0lDb21wb3NpdGVDb250ZXh0LmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9JQ29tcG9zaXRlQ29udGV4dC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmYzU2MzFmLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL0lDb21wb3NpdGVDb250ZXh0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw5MCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbDsKLQotaW1wb3J0IGphdmEuYXd0LkNvbXBvc2l0ZTsKLWltcG9ydCBqYXZhLmF3dC5Db21wb3NpdGVDb250ZXh0OwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmFzdGVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLldyaXRhYmxlUmFzdGVyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5JbWFnZVN1cmZhY2U7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5yZW5kZXIuTmF0aXZlSW1hZ2VCbGl0dGVyOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi0KLS8qKgotICogVGhpcyBjbGFzcyByZXByZXNlbnQgaW1wbGVtZW50YXRpb24gb2YgdGhlIENvbXBvc2l0ZUNvbnRleHQgaW50ZXJmYWNlCi0gKi8KLXB1YmxpYyBjbGFzcyBJQ29tcG9zaXRlQ29udGV4dCBpbXBsZW1lbnRzIENvbXBvc2l0ZUNvbnRleHQgewotICAgIENvbXBvc2l0ZSBjb21wb3NpdGU7Ci0gICAgQ29sb3JNb2RlbCBzcmNDTSwgZHN0Q007Ci0gICAgSW1hZ2VTdXJmYWNlIHNyY1N1cmYsIGRzdFN1cmY7Ci0KLSAgICBwdWJsaWMgSUNvbXBvc2l0ZUNvbnRleHQoQ29tcG9zaXRlIGNvbXAsIENvbG9yTW9kZWwgc3JjLCBDb2xvck1vZGVsIGRzdCl7Ci0gICAgICAgIGNvbXBvc2l0ZSA9IGNvbXA7Ci0gICAgICAgIHNyY0NNID0gc3JjOwotICAgICAgICBkc3RDTSA9IGRzdDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBkaXNwb3NlKCkgewotICAgICAgICBzcmNTdXJmLmRpc3Bvc2UoKTsKLSAgICAgICAgZHN0U3VyZi5kaXNwb3NlKCk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgY29tcG9zZShSYXN0ZXIgc3JjSW4sIFJhc3RlciBkc3RJbiwgV3JpdGFibGVSYXN0ZXIgZHN0T3V0KSB7Ci0KLSAgICAgICAgaWYoIXNyY0NNLmlzQ29tcGF0aWJsZVJhc3RlcihzcmNJbikpIHsKLSAgICAgICAgICAgIC8vIGF3dC40OD1UaGUgc3JjSW4gcmFzdGVyIGlzIGluY29tcGF0aWJsZSB3aXRoIHNyYyBDb2xvck1vZGVsCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjQ4IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZighZHN0Q00uaXNDb21wYXRpYmxlUmFzdGVyKGRzdEluKSkgewotICAgICAgICAgICAgLy8gYXd0LjQ5PVRoZSBkc3RJbiByYXN0ZXIgaXMgaW5jb21wYXRpYmxlIHdpdGggZHN0IENvbG9yTW9kZWwKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmKGRzdEluICE9IGRzdE91dCl7Ci0gICAgICAgICAgICBpZighZHN0Q00uaXNDb21wYXRpYmxlUmFzdGVyKGRzdE91dCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuNEE9VGhlIGRzdE91dCByYXN0ZXIgaXMgaW5jb21wYXRpYmxlIHdpdGggZHN0IENvbG9yTW9kZWwKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRBIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkc3RPdXQuc2V0RGF0YUVsZW1lbnRzKDAsIDAsIGRzdEluKTsKLSAgICAgICAgfQotICAgICAgICBXcml0YWJsZVJhc3RlciBzcmM7Ci0gICAgICAgIGlmKHNyY0luIGluc3RhbmNlb2YgV3JpdGFibGVSYXN0ZXIpewotICAgICAgICAgICAgc3JjID0gKFdyaXRhYmxlUmFzdGVyKSBzcmNJbjsKLSAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICBzcmMgPSBzcmNJbi5jcmVhdGVDb21wYXRpYmxlV3JpdGFibGVSYXN0ZXIoKTsKLSAgICAgICAgICAgIHNyYy5zZXREYXRhRWxlbWVudHMoMCwgMCwgc3JjSW4pOwotICAgICAgICB9Ci0gICAgICAgIHNyY1N1cmYgPSBuZXcgSW1hZ2VTdXJmYWNlKHNyY0NNLCBzcmMpOwotICAgICAgICBkc3RTdXJmID0gbmV3IEltYWdlU3VyZmFjZShkc3RDTSwgZHN0T3V0KTsKLQotICAgICAgICBpbnQgdyA9IE1hdGgubWluKHNyY0luLmdldFdpZHRoKCksIGRzdE91dC5nZXRXaWR0aCgpKTsKLSAgICAgICAgaW50IGggPSBNYXRoLm1pbihzcmNJbi5nZXRIZWlnaHQoKSwgZHN0T3V0LmdldEhlaWdodCgpKTsKLQotICAgICAgICBOYXRpdmVJbWFnZUJsaXR0ZXIuZ2V0SW5zdGFuY2UoKS5ibGl0KDAsIDAsIHNyY1N1cmYsIDAsIDAsIGRzdFN1cmYsCi0gICAgICAgICAgICAgICAgdywgaCwgY29tcG9zaXRlLCBudWxsLCBudWxsKTsKLQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvSW1hZ2VTdXJmYWNlLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9JbWFnZVN1cmZhY2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjM2OGRkOC4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9JbWFnZVN1cmZhY2UuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDMyMyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKiBDcmVhdGVkIG9uIDEwLjExLjIwMDUKLSAqCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbDsKLQotaW1wb3J0IGphdmEuYXd0LmNvbG9yLkNvbG9yU3BhY2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQmFuZGVkU2FtcGxlTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbXBvbmVudENvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29tcG9uZW50U2FtcGxlTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGlyZWN0Q29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbmRleENvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlBpeGVsSW50ZXJsZWF2ZWRTYW1wbGVNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5TYW1wbGVNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5TaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLldyaXRhYmxlUmFzdGVyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvci5MVVRDb2xvckNvbnZlcnRlcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmltYWdlLkRhdGFCdWZmZXJMaXN0ZW5lcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotCi0vKioKLSAqIFRoaXMgY2xhc3MgcmVwcmVzZW50IFN1cmZhY2UgZm9yIGRpZmZlcmVudCB0eXBlcyBvZiBJbWFnZXMgKEJ1ZmZlcmVkSW1hZ2UsIAotICogT2Zmc2NyZWVuSW1hZ2UgYW5kIHNvIG9uKSAKLSAqLwotcHVibGljIGNsYXNzIEltYWdlU3VyZmFjZSBleHRlbmRzIFN1cmZhY2UgaW1wbGVtZW50cyBEYXRhQnVmZmVyTGlzdGVuZXIgewotCi0gICAgYm9vbGVhbiBuYXRpdmVEcmF3YWJsZSA9IHRydWU7Ci0gICAgaW50IHN1cmZhY2VUeXBlOwotICAgIGludCBjc1R5cGU7Ci0gICAgQ29sb3JNb2RlbCBjbTsKLSAgICBXcml0YWJsZVJhc3RlciByYXN0ZXI7Ci0gICAgT2JqZWN0IGRhdGE7Ci0gICAgCi0gICAgYm9vbGVhbiBuZWVkVG9SZWZyZXNoID0gdHJ1ZTsKLSAgICBib29sZWFuIGRhdGFUYWtlbiA9IGZhbHNlOwotICAgIAotICAgIHByaXZhdGUgbG9uZyBjYWNoZWREYXRhUHRyOyAgICAgICAvLyBQb2ludGVyIGZvciBjYWNoZWQgSW1hZ2UgRGF0YQotICAgIHByaXZhdGUgYm9vbGVhbiBhbHBoYVByZTsgICAgICAgICAvLyBDYWNoZWQgSW1hZ2UgRGF0YSBhbHBoYSBwcmVtdWx0aXBsaWVkIAotCi0gICAgcHVibGljIEltYWdlU3VyZmFjZShDb2xvck1vZGVsIGNtLCBXcml0YWJsZVJhc3RlciByYXN0ZXIpewotICAgICAgICB0aGlzKGNtLCByYXN0ZXIsIFN1cmZhY2UuZ2V0VHlwZShjbSwgcmFzdGVyKSk7Ci0gICAgfQotCi0gICAgcHVibGljIEltYWdlU3VyZmFjZShDb2xvck1vZGVsIGNtLCBXcml0YWJsZVJhc3RlciByYXN0ZXIsIGludCB0eXBlKXsKLSAgICAgICAgaWYgKCFjbS5pc0NvbXBhdGlibGVSYXN0ZXIocmFzdGVyKSkgewotICAgICAgICAgICAgLy8gYXd0LjREPVRoZSByYXN0ZXIgaXMgaW5jb21wYXRpYmxlIHdpdGggdGhpcyBDb2xvck1vZGVsCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjREIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgdGhpcy5jbSA9IGNtOwotICAgICAgICB0aGlzLnJhc3RlciA9IHJhc3RlcjsKLSAgICAgICAgc3VyZmFjZVR5cGUgPSB0eXBlOwotCi0gICAgICAgIGRhdGEgPSBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IuZ2V0SW5zdGFuY2UoKS4KLSAgICAgICAgZ2V0RGF0YShyYXN0ZXIuZ2V0RGF0YUJ1ZmZlcigpKTsKLSAgICAgICAgQ29sb3JTcGFjZSBjcyA9IGNtLmdldENvbG9yU3BhY2UoKTsKLSAgICAgICAgdHJhbnNwYXJlbmN5ID0gY20uZ2V0VHJhbnNwYXJlbmN5KCk7Ci0gICAgICAgIHdpZHRoID0gcmFzdGVyLmdldFdpZHRoKCk7Ci0gICAgICAgIGhlaWdodCA9IHJhc3Rlci5nZXRIZWlnaHQoKTsKLQotICAgICAgICAvLyBGb3IgdGhlIG1vbWVudCB3ZSBjYW4gYnVpbGQgbmF0aXZlbHkgb25seSBpbWFnZXMgd2hpY2ggaGF2ZSAKLSAgICAgICAgLy8gc1JHQiwgTGluZWFyX1JHQiwgTGluZWFyX0dyYXkgQ29sb3IgU3BhY2UgYW5kIHR5cGUgZGlmZmVyZW50Ci0gICAgICAgIC8vIGZyb20gQnVmZmVyZWRJbWFnZS5UWVBFX0NVU1RPTQotICAgICAgICBpZihjcyA9PSBMVVRDb2xvckNvbnZlcnRlci5zUkdCX0NTKXsKLSAgICAgICAgICAgIGNzVHlwZSA9IHNSR0JfQ1M7Ci0gICAgICAgIH1lbHNlIGlmKGNzID09IExVVENvbG9yQ29udmVydGVyLkxJTkVBUl9SR0JfQ1MpewotICAgICAgICAgICAgY3NUeXBlID0gTGluZWFyX1JHQl9DUzsKLSAgICAgICAgfWVsc2UgaWYoY3MgPT0gTFVUQ29sb3JDb252ZXJ0ZXIuTElORUFSX0dSQVlfQ1MpewotICAgICAgICAgICAgY3NUeXBlID0gTGluZWFyX0dyYXlfQ1M7Ci0gICAgICAgIH1lbHNlewotICAgICAgICAgICAgY3NUeXBlID0gQ3VzdG9tX0NTOwotICAgICAgICAgICAgbmF0aXZlRHJhd2FibGUgPSBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmKHR5cGUgPT0gQnVmZmVyZWRJbWFnZS5UWVBFX0NVU1RPTSl7Ci0gICAgICAgICAgICBuYXRpdmVEcmF3YWJsZSA9IGZhbHNlOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIENvbG9yTW9kZWwgZ2V0Q29sb3JNb2RlbCgpIHsKLSAgICAgICAgcmV0dXJuIGNtOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBXcml0YWJsZVJhc3RlciBnZXRSYXN0ZXIoKSB7Ci0gICAgICAgIHJldHVybiByYXN0ZXI7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGxvbmcgZ2V0U3VyZmFjZURhdGFQdHIoKSB7Ci0gICAgICAgIGlmKHN1cmZhY2VEYXRhUHRyID09IDBMICYmIG5hdGl2ZURyYXdhYmxlKXsKLSAgICAgICAgICAgIGNyZWF0ZVN1ZmFjZVN0cnVjdHVyZSgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBzdXJmYWNlRGF0YVB0cjsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgT2JqZWN0IGdldERhdGEoKXsKLSAgICAgICAgcmV0dXJuIGRhdGE7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNOYXRpdmVEcmF3YWJsZSgpewotICAgICAgICByZXR1cm4gbmF0aXZlRHJhd2FibGU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRTdXJmYWNlVHlwZSgpIHsKLSAgICAgICAgcmV0dXJuIHN1cmZhY2VUeXBlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgbmF0aXZlIFN1cmZhY2Ugc3RydWN0dXJlIHdoaWNoIHVzZWQgZm9yIG5hdGl2ZSBibGl0dGluZwotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBjcmVhdGVTdWZhY2VTdHJ1Y3R1cmUoKXsKLSAgICAgICAgaW50IGNtVHlwZSA9IDA7Ci0gICAgICAgIGludCBudW1Db21wb25lbnRzID0gY20uZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICBib29sZWFuIGhhc0FscGhhID0gY20uaGFzQWxwaGEoKTsKLSAgICAgICAgYm9vbGVhbiBpc0FscGhhUHJlID0gY20uaXNBbHBoYVByZW11bHRpcGxpZWQoKTsKLSAgICAgICAgaW50IHRyYW5zcGFyZW5jeSA9IGNtLmdldFRyYW5zcGFyZW5jeSgpOwotICAgICAgICBpbnQgYml0c1tdID0gY20uZ2V0Q29tcG9uZW50U2l6ZSgpOwotICAgICAgICBpbnQgcGl4ZWxTdHJpZGUgPSBjbS5nZXRQaXhlbFNpemUoKTsKLSAgICAgICAgaW50IG1hc2tzW10gPSBudWxsOwotICAgICAgICBpbnQgY29sb3JNYXBbXSA9IG51bGw7Ci0gICAgICAgIGludCBjb2xvck1hcFNpemUgPSAwOwotICAgICAgICBpbnQgdHJhbnNwUGl4ZWwgPSAtMTsKLSAgICAgICAgYm9vbGVhbiBpc0dyYXlQYWxsZXRlID0gZmFsc2U7Ci0gICAgICAgIFNhbXBsZU1vZGVsIHNtID0gcmFzdGVyLmdldFNhbXBsZU1vZGVsKCk7Ci0gICAgICAgIGludCBzbVR5cGUgPSAwOwotICAgICAgICBpbnQgZGF0YVR5cGUgPSBzbS5nZXREYXRhVHlwZSgpOwotICAgICAgICBpbnQgc2NhbmxpbmVTdHJpZGUgPSAwOwotICAgICAgICBpbnQgYmFua0luZGVjZXNbXSA9IG51bGw7Ci0gICAgICAgIGludCBiYW5kT2Zmc2V0c1tdID0gbnVsbDsKLSAgICAgICAgaW50IG9mZnNldCA9IHJhc3Rlci5nZXREYXRhQnVmZmVyKCkuZ2V0T2Zmc2V0KCk7Ci0KLSAgICAgICAgaWYoY20gaW5zdGFuY2VvZiBEaXJlY3RDb2xvck1vZGVsKXsKLSAgICAgICAgICAgIGNtVHlwZSA9IERDTTsKLSAgICAgICAgICAgIERpcmVjdENvbG9yTW9kZWwgZGNtID0gKERpcmVjdENvbG9yTW9kZWwpIGNtOwotICAgICAgICAgICAgbWFza3MgPSBkY20uZ2V0TWFza3MoKTsKLSAgICAgICAgICAgIHNtVHlwZSA9IFNQUFNNOwotICAgICAgICAgICAgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBzcHBzbSA9IChTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKSBzbTsKLSAgICAgICAgICAgIHNjYW5saW5lU3RyaWRlID0gc3Bwc20uZ2V0U2NhbmxpbmVTdHJpZGUoKTsKLQotICAgICAgICB9ZWxzZSBpZihjbSBpbnN0YW5jZW9mIEluZGV4Q29sb3JNb2RlbCl7Ci0gICAgICAgICAgICBjbVR5cGUgPSBJQ007Ci0gICAgICAgICAgICBJbmRleENvbG9yTW9kZWwgaWNtID0gKEluZGV4Q29sb3JNb2RlbCkgY207Ci0gICAgICAgICAgICBjb2xvck1hcFNpemUgPSBpY20uZ2V0TWFwU2l6ZSgpOwotICAgICAgICAgICAgY29sb3JNYXAgPSBuZXcgaW50W2NvbG9yTWFwU2l6ZV07Ci0gICAgICAgICAgICBpY20uZ2V0UkdCcyhjb2xvck1hcCk7Ci0gICAgICAgICAgICB0cmFuc3BQaXhlbCA9IGljbS5nZXRUcmFuc3BhcmVudFBpeGVsKCk7Ci0gICAgICAgICAgICBpc0dyYXlQYWxsZXRlID0gU3VyZmFjZS5pc0dyYXlQYWxsZXRlKGljbSk7Ci0KLSAgICAgICAgICAgIGlmKHNtIGluc3RhbmNlb2YgTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKXsKLSAgICAgICAgICAgICAgICBzbVR5cGUgPSBNUFBTTTsKLSAgICAgICAgICAgICAgICBNdWx0aVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgbXBwc20gPQotICAgICAgICAgICAgICAgICAgICAoTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKSBzbTsKLSAgICAgICAgICAgICAgICBzY2FubGluZVN0cmlkZSA9IG1wcHNtLmdldFNjYW5saW5lU3RyaWRlKCk7Ci0gICAgICAgICAgICB9ZWxzZSBpZihzbSBpbnN0YW5jZW9mIENvbXBvbmVudFNhbXBsZU1vZGVsKXsKLSAgICAgICAgICAgICAgICBzbVR5cGUgPSBDU007Ci0gICAgICAgICAgICAgICAgQ29tcG9uZW50U2FtcGxlTW9kZWwgY3NtID0KLSAgICAgICAgICAgICAgICAgICAgKENvbXBvbmVudFNhbXBsZU1vZGVsKSBzbTsKLSAgICAgICAgICAgICAgICBzY2FubGluZVN0cmlkZSA9IGNzbS5nZXRTY2FubGluZVN0cmlkZSgpOwotICAgICAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjREPVRoZSByYXN0ZXIgaXMgaW5jb21wYXRpYmxlIHdpdGggdGhpcyBDb2xvck1vZGVsCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40RCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotCi0gICAgICAgIH1lbHNlIGlmKGNtIGluc3RhbmNlb2YgQ29tcG9uZW50Q29sb3JNb2RlbCl7Ci0gICAgICAgICAgICBjbVR5cGUgPSBDQ007Ci0gICAgICAgICAgICBpZihzbSBpbnN0YW5jZW9mIENvbXBvbmVudFNhbXBsZU1vZGVsKXsKLSAgICAgICAgICAgICAgICBDb21wb25lbnRTYW1wbGVNb2RlbCBjc20gPSAoQ29tcG9uZW50U2FtcGxlTW9kZWwpIHNtOwotICAgICAgICAgICAgICAgIHNjYW5saW5lU3RyaWRlID0gY3NtLmdldFNjYW5saW5lU3RyaWRlKCk7Ci0gICAgICAgICAgICAgICAgYmFua0luZGVjZXMgPSBjc20uZ2V0QmFua0luZGljZXMoKTsKLSAgICAgICAgICAgICAgICBiYW5kT2Zmc2V0cyA9IGNzbS5nZXRCYW5kT2Zmc2V0cygpOwotICAgICAgICAgICAgICAgIGlmKHNtIGluc3RhbmNlb2YgUGl4ZWxJbnRlcmxlYXZlZFNhbXBsZU1vZGVsKXsKLSAgICAgICAgICAgICAgICAgICAgc21UeXBlID0gUElTTTsKLSAgICAgICAgICAgICAgICB9ZWxzZSBpZihzbSBpbnN0YW5jZW9mIEJhbmRlZFNhbXBsZU1vZGVsKXsKLSAgICAgICAgICAgICAgICAgICAgc21UeXBlID0gQlNNOwotICAgICAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgICAgICBzbVR5cGUgPSBDU007Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjREPVRoZSByYXN0ZXIgaXMgaW5jb21wYXRpYmxlIHdpdGggdGhpcyBDb2xvck1vZGVsCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40RCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotCi0gICAgICAgIH1lbHNlewotICAgICAgICAgICAgc3VyZmFjZURhdGFQdHIgPSAwTDsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBzdXJmYWNlRGF0YVB0ciA9IGNyZWF0ZVN1cmZTdHJ1Y3Qoc3VyZmFjZVR5cGUsIHdpZHRoLCBoZWlnaHQsIGNtVHlwZSwgY3NUeXBlLCBzbVR5cGUsIGRhdGFUeXBlLAotICAgICAgICAgICAgICAgIG51bUNvbXBvbmVudHMsIHBpeGVsU3RyaWRlLCBzY2FubGluZVN0cmlkZSwgYml0cywgbWFza3MsIGNvbG9yTWFwU2l6ZSwKLSAgICAgICAgICAgICAgICBjb2xvck1hcCwgdHJhbnNwUGl4ZWwsIGlzR3JheVBhbGxldGUsIGJhbmtJbmRlY2VzLCBiYW5kT2Zmc2V0cywKLSAgICAgICAgICAgICAgICBvZmZzZXQsIGhhc0FscGhhLCBpc0FscGhhUHJlLCB0cmFuc3BhcmVuY3kpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRpc3Bvc2UoKSB7Ci0gICAgICAgIGlmKHN1cmZhY2VEYXRhUHRyICE9IDBMKXsKLSAgICAgICAgICAgIGRpc3Bvc2Uoc3VyZmFjZURhdGFQdHIpOwotICAgICAgICAgICAgc3VyZmFjZURhdGFQdHIgPSAwTDsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBwdWJsaWMgbG9uZyBnZXRDYWNoZWREYXRhKGJvb2xlYW4gYWxwaGFQcmUpewotICAgICAgICBpZihuYXRpdmVEcmF3YWJsZSl7Ci0gICAgICAgICAgICBpZihjYWNoZWREYXRhUHRyID09IDBMIHx8IG5lZWRUb1JlZnJlc2ggfHwgdGhpcy5hbHBoYVByZSAhPSBhbHBoYVByZSl7Ci0gICAgICAgICAgICAgICAgY2FjaGVkRGF0YVB0ciA9IHVwZGF0ZUNhY2hlKGdldFN1cmZhY2VEYXRhUHRyKCksIGRhdGEsIGFscGhhUHJlKTsKLSAgICAgICAgICAgICAgICB0aGlzLmFscGhhUHJlID0gYWxwaGFQcmU7Ci0gICAgICAgICAgICAgICAgdmFsaWRhdGUoKTsgCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGNhY2hlZERhdGFQdHI7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBuYXRpdmUgbG9uZyBjcmVhdGVTdXJmU3RydWN0KGludCBzdXJmYWNlVHlwZSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCAKLSAgICAgICAgICAgIGludCBjbVR5cGUsIGludCBjc1R5cGUsIGludCBzbVR5cGUsIGludCBkYXRhVHlwZSwKLSAgICAgICAgICAgIGludCBudW1Db21wb25lbnRzLCBpbnQgcGl4ZWxTdHJpZGUsIGludCBzY2FubGluZVN0cmlkZSwKLSAgICAgICAgICAgIGludCBiaXRzW10sIGludCBtYXNrc1tdLCBpbnQgY29sb3JNYXBTaXplLCBpbnQgY29sb3JNYXBbXSwKLSAgICAgICAgICAgIGludCB0cmFuc3BQaXhlbCwgYm9vbGVhbiBpc0dyYXlQYWxldHRlLCBpbnQgYmFua0luZGVjZXNbXSwgCi0gICAgICAgICAgICBpbnQgYmFuZE9mZnNldHNbXSwgaW50IG9mZnNldCwgYm9vbGVhbiBoYXNBbHBoYSwgYm9vbGVhbiBpc0FscGhhUHJlLAotICAgICAgICAgICAgaW50IHRyYW5zcGFyZW5jeSk7Ci0KLSAgICBwcml2YXRlIG5hdGl2ZSB2b2lkIGRpc3Bvc2UobG9uZyBzdHJ1Y3RQdHIpOwotCi0gICAgcHJpdmF0ZSBuYXRpdmUgdm9pZCBzZXRJbWFnZVNpemUobG9uZyBzdHJ1Y3RQdHIsIGludCB3aWR0aCwgaW50IGhlaWdodCk7Ci0KLSAgICBwcml2YXRlIG5hdGl2ZSBsb25nIHVwZGF0ZUNhY2hlKGxvbmcgc3RydWN0UHRyLCBPYmplY3QgZGF0YSwgYm9vbGVhbiBhbHBoYVByZSk7Ci0gICAgCi0gICAgLyoqCi0gICAgICogU3VwcG9zZXMgdGhhdCBuZXcgcmFzdGVyIGlzIGNvbXBhdGlibGUgd2l0aCBhbiBvbGQgb25lCi0gICAgICogQHBhcmFtIHIKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRSYXN0ZXIoV3JpdGFibGVSYXN0ZXIgcikgewotICAgICAgICByYXN0ZXIgPSByOwotICAgICAgICBkYXRhID0gQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yLmdldEluc3RhbmNlKCkuZ2V0RGF0YShyLmdldERhdGFCdWZmZXIoKSk7Ci0gICAgICAgIGlmIChzdXJmYWNlRGF0YVB0ciAhPSAwKSB7Ci0gICAgICAgICAgICBzZXRJbWFnZVNpemUoc3VyZmFjZURhdGFQdHIsIHIuZ2V0V2lkdGgoKSwgci5nZXRIZWlnaHQoKSk7Ci0gICAgICAgIH0KLSAgICAgICAgdGhpcy53aWR0aCA9IHIuZ2V0V2lkdGgoKTsKLSAgICAgICAgdGhpcy5oZWlnaHQgPSByLmdldEhlaWdodCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBsb25nIGxvY2soKSB7Ci0gICAgICAgIC8vIFRPRE8KLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgdW5sb2NrKCkgewotICAgICAgICAvL1RPRE8KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgU3VyZmFjZSBnZXRJbWFnZVN1cmZhY2UoKSB7Ci0gICAgICAgIHJldHVybiB0aGlzOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGRhdGFDaGFuZ2VkKCkgewotICAgICAgICBuZWVkVG9SZWZyZXNoID0gdHJ1ZTsKLSAgICAgICAgY2xlYXJWYWxpZENhY2hlcygpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGRhdGFUYWtlbigpIHsKLSAgICAgICAgZGF0YVRha2VuID0gdHJ1ZTsKLSAgICAgICAgbmVlZFRvUmVmcmVzaCA9IHRydWU7Ci0gICAgICAgIGNsZWFyVmFsaWRDYWNoZXMoKTsKLSAgICB9Ci0gICAgCi0gICAgcHVibGljIHZvaWQgZGF0YVJlbGVhc2VkKCl7Ci0gICAgICAgIGRhdGFUYWtlbiA9IGZhbHNlOwotICAgICAgICBuZWVkVG9SZWZyZXNoID0gdHJ1ZTsKLSAgICAgICAgY2xlYXJWYWxpZENhY2hlcygpOwotICAgIH0KLSAgICAKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBpbnZhbGlkYXRlKCl7Ci0gICAgICAgIG5lZWRUb1JlZnJlc2ggPSB0cnVlOwotICAgICAgICBjbGVhclZhbGlkQ2FjaGVzKCk7Ci0gICAgfQotICAgIAotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHZhbGlkYXRlKCl7Ci0gICAgICAgIGlmKCFuZWVkVG9SZWZyZXNoKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgaWYoIWRhdGFUYWtlbil7Ci0gICAgICAgICAgICBuZWVkVG9SZWZyZXNoID0gZmFsc2U7Ci0gICAgICAgICAgICBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IgYmEgPSBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IuZ2V0SW5zdGFuY2UoKTsKLSAgICAgICAgICAgIGJhLnZhbGlkYXRlKHJhc3Rlci5nZXREYXRhQnVmZmVyKCkpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgIH0KLSAgICAKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBpbnZhbGlkYXRlZCgpewotICAgICAgICByZXR1cm4gbmVlZFRvUmVmcmVzaDsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9NdWx0aVJlY3RBcmVhLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9NdWx0aVJlY3RBcmVhLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM0MjY3ZjMuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvTXVsdGlSZWN0QXJlYS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsODM2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRGVuaXMgTS4gS2lzaGVua28KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsOwotCi1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LlNoYXBlOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUGF0aEl0ZXJhdG9yOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUG9pbnQyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLk5vU3VjaEVsZW1lbnRFeGNlcHRpb247Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotcHVibGljIGNsYXNzIE11bHRpUmVjdEFyZWEgaW1wbGVtZW50cyBTaGFwZSB7Ci0KLSAgICAvKioKLSAgICAgKiBJZiBDSEVDSyBpcyB0cnVlIHZhbGlkYXRpb24gY2hlY2sgYWN0aXZlCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgYm9vbGVhbiBDSEVDSyA9IGZhbHNlOwotCi0gICAgYm9vbGVhbiBzb3J0ZWQgPSB0cnVlOwotICAgIAotICAgIC8qKgotICAgICAqIFJlY3RhbmdsZSBidWZmZXIKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50W10gcmVjdDsKLSAgICAKLSAgICAvKioKLSAgICAgKiBCb3VuZGluZyBib3gKLSAgICAgKi8KLSAgICBSZWN0YW5nbGUgYm91bmRzOwotICAgIAotICAgIC8qKgotICAgICAqIFJlc3VsdCByZWN0YW5nbGUgYXJyYXkKLSAgICAgKi8KLSAgICBSZWN0YW5nbGVbXSByZWN0YW5nbGVzOwotCi0gICAgLyoqCi0gICAgICogTGluZUNhc2ggcHJvdmlkZXMgY3JlYXRpbmcgTXVsdGlSZWN0QXJlYSBsaW5lIGJ5IGxpbmUuIFVzZWQgaW4gSmF2YVNoYXBlUmFzdGVyaXplci4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGNsYXNzIExpbmVDYXNoIGV4dGVuZHMgTXVsdGlSZWN0QXJlYSB7Ci0KLSAgICAgICAgaW50IGxpbmVZOwotICAgICAgICBpbnQgYm90dG9tQ291bnQ7Ci0gICAgICAgIGludFtdIGJvdHRvbTsKLQotICAgICAgICBwdWJsaWMgTGluZUNhc2goaW50IHNpemUpIHsKLSAgICAgICAgICAgIHN1cGVyKCk7Ci0gICAgICAgICAgICBib3R0b20gPSBuZXcgaW50W3NpemVdOwotICAgICAgICAgICAgYm90dG9tQ291bnQgPSAwOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIHZvaWQgc2V0TGluZShpbnQgeSkgewotICAgICAgICAgICAgbGluZVkgPSB5OwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIHZvaWQgc2tpcExpbmUoKSB7Ci0gICAgICAgICAgICBsaW5lWSsrOwotICAgICAgICAgICAgYm90dG9tQ291bnQgPSAwOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIHZvaWQgYWRkTGluZShpbnRbXSBwb2ludHMsIGludCBwb2ludENvdW50KSB7Ci0gICAgICAgICAgICBpbnQgYm90dG9tSW5kZXggPSAwOwotICAgICAgICAgICAgaW50IHBvaW50SW5kZXggPSAwOwotICAgICAgICAgICAgaW50IHJlY3RJbmRleCA9IDA7Ci0gICAgICAgICAgICBpbnQgcG9pbnRYMSA9IDA7Ci0gICAgICAgICAgICBpbnQgcG9pbnRYMiA9IDA7Ci0gICAgICAgICAgICBpbnQgYm90dG9tWDEgPSAwOwotICAgICAgICAgICAgaW50IGJvdHRvbVgyID0gMDsKLSAgICAgICAgICAgIGJvb2xlYW4gYXBwZW5kUmVjdCA9IGZhbHNlOwotICAgICAgICAgICAgYm9vbGVhbiBkZWxldGVSZWN0ID0gZmFsc2U7Ci0gICAgICAgICAgICBpbnQgbGFzdENvdW50ID0gYm90dG9tQ291bnQ7Ci0KLSAgICAgICAgICAgIHdoaWxlIChib3R0b21JbmRleCA8IGxhc3RDb3VudCB8fCBwb2ludEluZGV4IDwgcG9pbnRDb3VudCkgewotCi0gICAgICAgICAgICAgICAgYXBwZW5kUmVjdCA9IGZhbHNlOwotICAgICAgICAgICAgICAgIGRlbGV0ZVJlY3QgPSBmYWxzZTsKLQotICAgICAgICAgICAgICAgIGlmIChib3R0b21JbmRleCA8IGxhc3RDb3VudCkgewotICAgICAgICAgICAgICAgICAgICByZWN0SW5kZXggPSBib3R0b21bYm90dG9tSW5kZXhdOwotICAgICAgICAgICAgICAgICAgICBib3R0b21YMSA9IHJlY3RbcmVjdEluZGV4XTsKLSAgICAgICAgICAgICAgICAgICAgYm90dG9tWDIgPSByZWN0W3JlY3RJbmRleCArIDJdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGFwcGVuZFJlY3QgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChwb2ludEluZGV4IDwgcG9pbnRDb3VudCkgewotICAgICAgICAgICAgICAgICAgICBwb2ludFgxID0gcG9pbnRzW3BvaW50SW5kZXhdOwotICAgICAgICAgICAgICAgICAgICBwb2ludFgyID0gcG9pbnRzW3BvaW50SW5kZXggKyAxXTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBkZWxldGVSZWN0ID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBpZiAoIWRlbGV0ZVJlY3QgJiYgIWFwcGVuZFJlY3QpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHBvaW50WDEgPT0gYm90dG9tWDEgJiYgcG9pbnRYMiA9PSBib3R0b21YMikgewotICAgICAgICAgICAgICAgICAgICAgICAgcmVjdFtyZWN0SW5kZXggKyAzXSA9IHJlY3RbcmVjdEluZGV4ICsgM10gKyAxOwotICAgICAgICAgICAgICAgICAgICAgICAgcG9pbnRJbmRleCArPSAyOwotICAgICAgICAgICAgICAgICAgICAgICAgYm90dG9tSW5kZXgrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGRlbGV0ZVJlY3QgPSBwb2ludFgyID49IGJvdHRvbVgxOwotICAgICAgICAgICAgICAgICAgICBhcHBlbmRSZWN0ID0gcG9pbnRYMSA8PSBib3R0b21YMjsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBpZiAoZGVsZXRlUmVjdCkgewotICAgICAgICAgICAgICAgICAgICBpZiAoYm90dG9tSW5kZXggPCBib3R0b21Db3VudCAtIDEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkoYm90dG9tLCBib3R0b21JbmRleCArIDEsIGJvdHRvbSwgYm90dG9tSW5kZXgsIGJvdHRvbUNvdW50IC0gYm90dG9tSW5kZXggLSAxKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlY3RJbmRleCAtPSA0OwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGJvdHRvbUNvdW50LS07Ci0gICAgICAgICAgICAgICAgICAgIGxhc3RDb3VudC0tOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChhcHBlbmRSZWN0KSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCBpID0gcmVjdFswXTsKLSAgICAgICAgICAgICAgICAgICAgYm90dG9tW2JvdHRvbUNvdW50KytdID0gaTsKLSAgICAgICAgICAgICAgICAgICAgcmVjdCA9IE11bHRpUmVjdEFyZWFPcC5jaGVja0J1ZlNpemUocmVjdCwgNCk7Ci0gICAgICAgICAgICAgICAgICAgIHJlY3RbaSsrXSA9IHBvaW50WDE7Ci0gICAgICAgICAgICAgICAgICAgIHJlY3RbaSsrXSA9IGxpbmVZOwotICAgICAgICAgICAgICAgICAgICByZWN0W2krK10gPSBwb2ludFgyOwotICAgICAgICAgICAgICAgICAgICByZWN0W2krK10gPSBsaW5lWTsKLSAgICAgICAgICAgICAgICAgICAgcG9pbnRJbmRleCArPSAyOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGxpbmVZKys7Ci0KLSAgICAgICAgICAgIGludmFsaWRhdGUoKTsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVjdENhc2ggcHJvdmlkZXMgc2ltcGxlIGNyZWF0aW5nIE11bHRpUmVjdEFyZWEKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGNsYXNzIFJlY3RDYXNoIGV4dGVuZHMgTXVsdGlSZWN0QXJlYSB7Ci0KLSAgICAgICAgaW50W10gY2FzaDsKLQotICAgICAgICBwdWJsaWMgUmVjdENhc2goKSB7Ci0gICAgICAgICAgICBzdXBlcigpOwotICAgICAgICAgICAgY2FzaCA9IG5ldyBpbnRbTXVsdGlSZWN0QXJlYU9wLlJFQ1RfQ0FQQUNJVFldOwotICAgICAgICAgICAgY2FzaFswXSA9IDE7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgdm9pZCBhZGRSZWN0Q2FzaGVkKGludCB4MSwgaW50IHkxLCBpbnQgeDIsIGludCB5MikgewotICAgICAgICAgICAgYWRkUmVjdCh4MSwgeTEsIHgyLCB5Mik7Ci0gICAgICAgICAgICBpbnZhbGlkYXRlKCk7Ci0vKgotICAgICAgICAgICAgLy8gRXhjbHVkZSBmcm9tIGNhc2ggdW5uZWNlc3NhcnkgcmVjdGFuZ2xlcwotICAgICAgICAgICAgaW50IGkgPSAxOwotICAgICAgICAgICAgd2hpbGUoaSA8IGNhc2hbMF0pIHsKLSAgICAgICAgICAgICAgICBpZiAocmVjdFtjYXNoW2ldICsgM10gPj0geTEgLSAxKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChpID4gMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShjYXNoLCBpLCBjYXNoLCAxLCBjYXNoWzBdIC0gaSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGkrKzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNhc2hbMF0gLT0gaSAtIDE7Ci0KLSAgICAgICAgICAgIC8vIEZpbmQgaW4gY2FzaCByZWN0YW5nbGUgdG8gY29uY2F0aW5hdGUKLSAgICAgICAgICAgIGkgPSAxOwotICAgICAgICAgICAgd2hpbGUoaSA8IGNhc2hbMF0pIHsKLSAgICAgICAgICAgICAgICBpbnQgaW5kZXggPSBjYXNoW2ldOwotICAgICAgICAgICAgICAgIGlmIChyZWN0W2luZGV4ICsgM10gIT0geTEgLSAxKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAocmVjdFtpbmRleF0gPT0geDEgJiYgcmVjdFtpbmRleCArIDJdID09IHgyKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJlY3RbaW5kZXggKyAzXSArPSB5MiAtIHkxICsgMTsKLQotICAgICAgICAgICAgICAgICAgICBpbnQgcG9zID0gaSArIDE7Ci0gICAgICAgICAgICAgICAgICAgIHdoaWxlKHBvcyA8IGNhc2hbMF0pIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChyZWN0W2luZGV4ICsgM10gPD0gcmVjdFtjYXNoW2ldICsgM10pIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGNhc2gsIGkgKyAxLCBjYXNoLCBpLCBwb3MgLSBpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIGkrKzsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBjYXNoW3BvcyAtIDFdID0gaW5kZXg7Ci0KLSAgICAgICAgICAgICAgICAgICAgaW52YWxpZGF0ZSgpOwotICAgICAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGkrKzsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gQWRkIHJlY3RhbmdsZSB0byBidWZmZXIKLSAgICAgICAgICAgIGludCBpbmRleCA9IHJlY3RbMF07Ci0gICAgICAgICAgICByZWN0ID0gTXVsdGlSZWN0QXJlYU9wLmNoZWNrQnVmU2l6ZShyZWN0LCA0KTsKLSAgICAgICAgICAgIHJlY3RbaW5kZXggKyAwXSA9IHgxOwotICAgICAgICAgICAgcmVjdFtpbmRleCArIDFdID0geTE7Ci0gICAgICAgICAgICByZWN0W2luZGV4ICsgMl0gPSB4MjsKLSAgICAgICAgICAgIHJlY3RbaW5kZXggKyAzXSA9IHkyOwotCi0gICAgICAgICAgICAvLyBBZGQgcmVjdGFuZ2xlIHRvIGNhc2gKLSAgICAgICAgICAgIGludCBsZW5ndGggPSBjYXNoWzBdOwotICAgICAgICAgICAgY2FzaCA9IE11bHRpUmVjdEFyZWFPcC5jaGVja0J1ZlNpemUoY2FzaCwgMSk7Ci0gICAgICAgICAgICB3aGlsZShpIDwgbGVuZ3RoKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHkyIDw9IHJlY3RbY2FzaFtpXSArIDNdKSB7Ci0gICAgICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkoY2FzaCwgaSwgY2FzaCwgaSArIDEsIGxlbmd0aCAtIGkpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaSsrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY2FzaFtpXSA9IGluZGV4OwotICAgICAgICAgICAgaW52YWxpZGF0ZSgpOwotKi8KLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyB2b2lkIGFkZFJlY3RDYXNoZWQoaW50W10gcmVjdCwgaW50IHJlY3RPZmYsIGludCByZWN0TGVuZ3RoKSB7Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSByZWN0T2ZmOyBpIDwgcmVjdE9mZiArIHJlY3RMZW5ndGg7KSB7Ci0gICAgICAgICAgICAgICAgYWRkUmVjdChyZWN0W2krK10sIHJlY3RbaSsrXSwgcmVjdFtpKytdLCByZWN0W2krK10pOwotLy8gICAgICAgICAgICAgIGFkZFJlY3RDYXNoZWQocmVjdFtpKytdLCByZWN0W2krK10sIHJlY3RbaSsrXSwgcmVjdFtpKytdKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogTXVsdGlSZWN0QXJlYSBwYXRoIGl0ZXJhdG9yCi0gICAgICovCi0gICAgY2xhc3MgSXRlcmF0b3IgaW1wbGVtZW50cyBQYXRoSXRlcmF0b3IgewotCi0gICAgICAgIGludCB0eXBlOwotICAgICAgICBpbnQgaW5kZXg7Ci0gICAgICAgIGludCBwb3M7Ci0KLSAgICAgICAgaW50W10gcmVjdDsKLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHQ7Ci0KLSAgICAgICAgSXRlcmF0b3IoTXVsdGlSZWN0QXJlYSBtcmEsIEFmZmluZVRyYW5zZm9ybSB0KSB7Ci0gICAgICAgICAgICByZWN0ID0gbmV3IGludFttcmEucmVjdFswXSAtIDFdOwotICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShtcmEucmVjdCwgMSwgcmVjdCwgMCwgcmVjdC5sZW5ndGgpOwotICAgICAgICAgICAgdGhpcy50ID0gdDsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBpbnQgZ2V0V2luZGluZ1J1bGUoKSB7Ci0gICAgICAgICAgICByZXR1cm4gV0lORF9OT05fWkVSTzsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBib29sZWFuIGlzRG9uZSgpIHsKLSAgICAgICAgICAgIHJldHVybiBwb3MgPj0gcmVjdC5sZW5ndGg7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgdm9pZCBuZXh0KCkgewotICAgICAgICAgICAgaWYgKGluZGV4ID09IDQpIHsKLSAgICAgICAgICAgICAgICBwb3MgKz0gNDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGluZGV4ID0gKGluZGV4ICsgMSkgJSA1OwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGludCBjdXJyZW50U2VnbWVudChkb3VibGVbXSBjb29yZHMpIHsKLSAgICAgICAgICAgIGlmIChpc0RvbmUoKSkgewotICAgICAgICAgICAgICAgIC8vIGF3dC40Qj1JaXRlcmF0b3Igb3V0IG9mIGJvdW5kcwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBOb1N1Y2hFbGVtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjRCIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbnQgdHlwZSA9IDA7Ci0KLSAgICAgICAgICAgIHN3aXRjaChpbmRleCkgewotICAgICAgICAgICAgY2FzZSAwIDoKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX01PVkVUTzsKLSAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSByZWN0W3BvcyArIDBdOwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IHJlY3RbcG9zICsgMV07Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIDE6Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19MSU5FVE87Ci0gICAgICAgICAgICAgICAgY29vcmRzWzBdID0gcmVjdFtwb3MgKyAyXTsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSByZWN0W3BvcyArIDFdOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSAyOgotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTElORVRPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IHJlY3RbcG9zICsgMl07Ci0gICAgICAgICAgICAgICAgY29vcmRzWzFdID0gcmVjdFtwb3MgKyAzXTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgMzoKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX0xJTkVUTzsKLSAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSByZWN0W3BvcyArIDBdOwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IHJlY3RbcG9zICsgM107Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIDQ6Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19DTE9TRTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKHQgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHQudHJhbnNmb3JtKGNvb3JkcywgMCwgY29vcmRzLCAwLCAxKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB0eXBlOwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIGludCBjdXJyZW50U2VnbWVudChmbG9hdFtdIGNvb3JkcykgewotICAgICAgICAgICAgaWYgKGlzRG9uZSgpKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjRCPUlpdGVyYXRvciBvdXQgb2YgYm91bmRzCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IE5vU3VjaEVsZW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNEIiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGludCB0eXBlID0gMDsKLQotICAgICAgICAgICAgc3dpdGNoKGluZGV4KSB7Ci0gICAgICAgICAgICBjYXNlIDAgOgotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTU9WRVRPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IHJlY3RbcG9zICsgMF07Ci0gICAgICAgICAgICAgICAgY29vcmRzWzFdID0gcmVjdFtwb3MgKyAxXTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgMToKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX0xJTkVUTzsKLSAgICAgICAgICAgICAgICBjb29yZHNbMF0gPSByZWN0W3BvcyArIDJdOwotICAgICAgICAgICAgICAgIGNvb3Jkc1sxXSA9IHJlY3RbcG9zICsgMV07Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIDI6Ci0gICAgICAgICAgICAgICAgdHlwZSA9IFNFR19MSU5FVE87Ci0gICAgICAgICAgICAgICAgY29vcmRzWzBdID0gcmVjdFtwb3MgKyAyXTsKLSAgICAgICAgICAgICAgICBjb29yZHNbMV0gPSByZWN0W3BvcyArIDNdOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSAzOgotICAgICAgICAgICAgICAgIHR5cGUgPSBTRUdfTElORVRPOwotICAgICAgICAgICAgICAgIGNvb3Jkc1swXSA9IHJlY3RbcG9zICsgMF07Ci0gICAgICAgICAgICAgICAgY29vcmRzWzFdID0gcmVjdFtwb3MgKyAzXTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgNDoKLSAgICAgICAgICAgICAgICB0eXBlID0gU0VHX0NMT1NFOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAodCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgdC50cmFuc2Zvcm0oY29vcmRzLCAwLCBjb29yZHMsIDAsIDEpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIHR5cGU7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbnN0cnVjdHMgYSBuZXcgZW1wdHkgTXVsdGlSZWN0QXJlYSAKLSAgICAgKi8KLSAgICBwdWJsaWMgTXVsdGlSZWN0QXJlYSgpIHsKLSAgICAgICAgcmVjdCA9IE11bHRpUmVjdEFyZWFPcC5jcmVhdGVCdWYoMCk7Ci0gICAgfQotCi0gICAgcHVibGljIE11bHRpUmVjdEFyZWEoYm9vbGVhbiBzb3J0ZWQpIHsKLSAgICAgICB0aGlzKCk7Ci0gICAgICAgdGhpcy5zb3J0ZWQgPSBzb3J0ZWQ7Ci0gICAgfQotICAgIAotICAgIC8qKgotICAgICAqIENvbnN0cnVjdHMgYSBuZXcgTXVsdGlSZWN0QXJlYSBhcyBhIGNvcHkgb2YgYW5vdGhlciBvbmUgCi0gICAgICovCi0gICAgcHVibGljIE11bHRpUmVjdEFyZWEoTXVsdGlSZWN0QXJlYSBtcmEpIHsKLSAgICAgICAgaWYgKG1yYSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZWN0ID0gTXVsdGlSZWN0QXJlYU9wLmNyZWF0ZUJ1ZigwKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJlY3QgPSBuZXcgaW50W21yYS5yZWN0Lmxlbmd0aF07Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KG1yYS5yZWN0LCAwLCByZWN0LCAwLCBtcmEucmVjdC5sZW5ndGgpOwotICAgICAgICAgICAgY2hlY2sodGhpcywgIk11bHRpUmVjdEFyZWEoTVJBKSIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb25zdHJ1Y3RzIGEgbmV3IE11bHRpUmVjdEFyZWEgY29uc2lzdHMgb2Ygc2luZ2xlIHJlY3RhbmdsZSAKLSAgICAgKi8KLSAgICBwdWJsaWMgTXVsdGlSZWN0QXJlYShSZWN0YW5nbGUgcikgewotICAgICAgICByZWN0ID0gTXVsdGlSZWN0QXJlYU9wLmNyZWF0ZUJ1ZigwKTsKLSAgICAgICAgaWYgKHIgIT0gbnVsbCAmJiAhci5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgIHJlY3RbMF0gPSA1OwotICAgICAgICAgICAgcmVjdFsxXSA9IHIueDsKLSAgICAgICAgICAgIHJlY3RbMl0gPSByLnk7Ci0gICAgICAgICAgICByZWN0WzNdID0gci54ICsgci53aWR0aCAtIDE7Ci0gICAgICAgICAgICByZWN0WzRdID0gci55ICsgci5oZWlnaHQgLSAxOwotICAgICAgICB9Ci0gICAgICAgIGNoZWNrKHRoaXMsICJNdWx0aVJlY3RBcmVhKFJlY3RhbmdsZSkiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbnN0cnVjdHMgYSBuZXcgTXVsdGlSZWN0QXJlYSBjb25zaXN0cyBvZiBzaW5nbGUgcmVjdGFuZ2xlCi0gICAgICovCi0gICAgcHVibGljIE11bHRpUmVjdEFyZWEoaW50IHgwLCBpbnQgeTAsIGludCB4MSwgaW50IHkxKSB7Ci0gICAgICAgIHJlY3QgPSBNdWx0aVJlY3RBcmVhT3AuY3JlYXRlQnVmKDApOwotICAgICAgICBpZiAoeDEgPj0geDAgJiYgeTEgPj0geTApIHsKLSAgICAgICAgICAgIHJlY3RbMF0gPSA1OwotICAgICAgICAgICAgcmVjdFsxXSA9IHgwOwotICAgICAgICAgICAgcmVjdFsyXSA9IHkwOwotICAgICAgICAgICAgcmVjdFszXSA9IHgxOwotICAgICAgICAgICAgcmVjdFs0XSA9IHkxOwotICAgICAgICB9Ci0gICAgICAgIGNoZWNrKHRoaXMsICJNdWx0aVJlY3RBcmVhKFJlY3RhbmdsZSkiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbnN0cnVjdHMgYSBuZXcgTXVsdGlSZWN0QXJlYSBhbmQgYXBwZW5kIHJlY3RhbmdsZSBmcm9tIGJ1ZmZlcgotICAgICAqLwotICAgIHB1YmxpYyBNdWx0aVJlY3RBcmVhKFJlY3RhbmdsZVtdIGJ1ZikgewotICAgICAgICB0aGlzKCk7Ci0gICAgICAgIGZvciAoUmVjdGFuZ2xlIGVsZW1lbnQgOiBidWYpIHsKLSAgICAgICAgICAgIGFkZChlbGVtZW50KTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbnN0cnVjdHMgYSBuZXcgTXVsdGlSZWN0QXJlYSBhbmQgYXBwZW5kIHJlY3RhbmdsZSBmcm9tIGFycmF5Ci0gICAgICovCi0gICAgcHVibGljIE11bHRpUmVjdEFyZWEoQXJyYXlMaXN0PFJlY3RhbmdsZT4gYnVmKSB7Ci0gICAgICAgIHRoaXMoKTsKLSAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IGJ1Zi5zaXplKCk7IGkrKykgewotICAgICAgICAgICAgYWRkKGJ1Zi5nZXQoaSkpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU29ydCByZWN0YW5nbGUgYnVmZmVyCi0gICAgICovCi0gICAgdm9pZCByZXNvcnQoKSB7Ci0gICAgICAgIGludFtdIGJ1ZiA9IG5ldyBpbnRbNF07Ci0gICAgICAgIGZvcihpbnQgaSA9IDE7IGkgPCByZWN0WzBdOyBpICs9IDQpIHsKLSAgICAgICAgICAgIGludCBrID0gaTsKLSAgICAgICAgICAgIGludCB4MSA9IHJlY3Rba107Ci0gICAgICAgICAgICBpbnQgeTEgPSByZWN0W2sgKyAxXTsKLSAgICAgICAgICAgIGZvcihpbnQgaiA9IGkgKyA0OyBqIDwgcmVjdFswXTsgaiArPSA0KSB7Ci0gICAgICAgICAgICAgICAgaW50IHgyID0gcmVjdFtqXTsKLSAgICAgICAgICAgICAgICBpbnQgeTIgPSByZWN0W2ogKyAxXTsKLSAgICAgICAgICAgICAgICBpZiAoeTEgPiB5MiB8fCAoeTEgPT0geTIgJiYgeDEgPiB4MikpIHsKLSAgICAgICAgICAgICAgICAgICAgeDEgPSB4MjsKLSAgICAgICAgICAgICAgICAgICAgeTEgPSB5MjsKLSAgICAgICAgICAgICAgICAgICAgayA9IGo7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGsgIT0gaSkgewotICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkocmVjdCwgaSwgYnVmLCAwLCA0KTsKLSAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHJlY3QsIGssIHJlY3QsIGksIDQpOwotICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkoYnVmLCAwLCByZWN0LCBrLCA0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBpbnZhbGlkYXRlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGVzdHMgZXF1YWxzIHdpdGggYW5vdGhlciBvYmplY3QKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG9iaikgewotICAgICAgICBpZiAob2JqID09IHRoaXMpIHsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChvYmogaW5zdGFuY2VvZiBNdWx0aVJlY3RBcmVhKSB7Ci0gICAgICAgICAgICBNdWx0aVJlY3RBcmVhIG1yYSA9IChNdWx0aVJlY3RBcmVhKSBvYmo7Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgcmVjdFswXTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHJlY3RbaV0gIT0gbXJhLnJlY3RbaV0pIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDaGVja3MgdmFsaWRhdGlvbiBvZiBNdWx0aVJlY3RBcmVhIG9iamVjdAotICAgICAqLwotICAgIHN0YXRpYyBNdWx0aVJlY3RBcmVhIGNoZWNrKE11bHRpUmVjdEFyZWEgbXJhLCBTdHJpbmcgbXNnKSB7Ci0gICAgICAgIGlmIChDSEVDSyAmJiBtcmEgIT0gbnVsbCkgewotICAgICAgICAgICAgaWYgKE11bHRpUmVjdEFyZWEuY2hlY2tWYWxpZGF0aW9uKG1yYS5nZXRSZWN0YW5nbGVzKCksIG1yYS5zb3J0ZWQpICE9IC0xKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjRDPUludmFsaWQgTXVsdGlSZWN0QXJlYSBpbiBtZXRob2QgezB9Ci0gICAgICAgICAgICAgICAgbmV3IFJ1bnRpbWVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNEMiLCBtc2cpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBtcmE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIHZhbGlkYXRpb24gb2YgTXVsdGlSZWN0QXJlYSBvYmplY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBjaGVja1ZhbGlkYXRpb24oUmVjdGFuZ2xlW10gciwgYm9vbGVhbiBzb3J0ZWQpIHsKLQotICAgICAgICAvLyBDaGVjayB3aWR0aCBhbmQgaGVpZ2h0Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCByLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBpZiAocltpXS53aWR0aCA8PSAwIHx8IHJbaV0uaGVpZ2h0IDw9IDApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gaTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIENoZWNrIG9yZGVyCi0gICAgICAgIGlmIChzb3J0ZWQpIHsKLSAgICAgICAgICAgIGZvcihpbnQgaSA9IDE7IGkgPCByLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHJbaSAtIDFdLnkgPiByW2ldLnkpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChyW2kgLSAxXS55ID09IHJbaV0ueSkgewotICAgICAgICAgICAgICAgICAgICBpZiAocltpIC0gMV0ueCA+IHJbaV0ueCkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBDaGVjayBvdmVycmlkZQotICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgci5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgZm9yKGludCBqID0gaSArIDE7IGogPCByLmxlbmd0aDsgaisrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHJbaV0uaW50ZXJzZWN0cyhyW2pdKSkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gaTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXNzaWducyByZWN0YW5nbGUgZnJvbSBhbm90aGVyIGJ1ZmZlcgotICAgICAqLwotICAgIHByb3RlY3RlZCB2b2lkIHNldFJlY3QoaW50W10gYnVmLCBib29sZWFuIGNvcHkpIHsKLSAgICAgICAgaWYgKGNvcHkpIHsKLSAgICAgICAgICAgIHJlY3QgPSBuZXcgaW50W2J1Zi5sZW5ndGhdOwotICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShidWYsIDAsIHJlY3QsIDAsIGJ1Zi5sZW5ndGgpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcmVjdCA9IGJ1ZjsKLSAgICAgICAgfQotICAgICAgICBpbnZhbGlkYXRlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVW5pb24gd2l0aCBhbm90aGVyIE11bHRpUmVjdEFyZWEgb2JqZWN0Ci0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkKE11bHRpUmVjdEFyZWEgbXJhKSB7Ci0gICAgICAgIHNldFJlY3QodW5pb24odGhpcywgbXJhKS5yZWN0LCBmYWxzZSk7Ci0gICAgICAgIGludmFsaWRhdGUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnRlcnNlY3Qgd2l0aCBhbm90aGVyIE11bHRpUmVjdEFyZWEgb2JqZWN0Ci0gICAgICovCi0gICAgcHVibGljIHZvaWQgaW50ZXJzZWN0KE11bHRpUmVjdEFyZWEgbXJhKSB7Ci0gICAgICAgIHNldFJlY3QoaW50ZXJzZWN0KHRoaXMsIG1yYSkucmVjdCwgZmFsc2UpOwotICAgICAgICBpbnZhbGlkYXRlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU3VidHJhY3QgYW5vdGhlciBNdWx0aVJlY3RBcmVhIG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHN1YnN0cmFjdChNdWx0aVJlY3RBcmVhIG1yYSkgewotICAgICAgICBzZXRSZWN0KHN1YnRyYWN0KHRoaXMsIG1yYSkucmVjdCwgZmFsc2UpOwotICAgICAgICBpbnZhbGlkYXRlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVW5pb24gd2l0aCBSZWN0YW5nbGUgb2JqZWN0Ci0gICAgICovCi0gICAgcHVibGljIHZvaWQgYWRkKFJlY3RhbmdsZSByZWN0KSB7Ci0gICAgICAgIHNldFJlY3QodW5pb24odGhpcywgbmV3IE11bHRpUmVjdEFyZWEocmVjdCkpLnJlY3QsIGZhbHNlKTsKLSAgICAgICAgaW52YWxpZGF0ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEludGVyc2VjdCB3aXRoIFJlY3RhbmdsZSBvYmplY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBpbnRlcnNlY3QoUmVjdGFuZ2xlIHJlY3QpIHsKLSAgICAgICAgc2V0UmVjdChpbnRlcnNlY3QodGhpcywgbmV3IE11bHRpUmVjdEFyZWEocmVjdCkpLnJlY3QsIGZhbHNlKTsKLSAgICAgICAgaW52YWxpZGF0ZSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFN1YnRyYWN0IHJlY3RhbmdsZSBvYmplY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzdWJzdHJhY3QoUmVjdGFuZ2xlIHJlY3QpIHsKLSAgICAgICAgc2V0UmVjdChzdWJ0cmFjdCh0aGlzLCBuZXcgTXVsdGlSZWN0QXJlYShyZWN0KSkucmVjdCwgZmFsc2UpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFVuaW9uIHR3byBNdXRsaVJlY3RhcmVBcmVhIG9iamVjdHMKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIE11bHRpUmVjdEFyZWEgaW50ZXJzZWN0KE11bHRpUmVjdEFyZWEgc3JjMSwgTXVsdGlSZWN0QXJlYSBzcmMyKSB7Ci0gICAgICAgIE11bHRpUmVjdEFyZWEgcmVzID0gY2hlY2soTXVsdGlSZWN0QXJlYU9wLkludGVyc2VjdGlvbi5nZXRSZXN1bHQoc3JjMSwgc3JjMiksICJpbnRlcnNlY3QoTVJBLE1SQSkiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICByZXR1cm4gcmVzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEludGVyc2VjdCB0d28gTXVsdGlSZWN0QXJlYSBvYmplY3RzCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBNdWx0aVJlY3RBcmVhIHVuaW9uKE11bHRpUmVjdEFyZWEgc3JjMSwgTXVsdGlSZWN0QXJlYSBzcmMyKSB7Ci0gICAgICAgIE11bHRpUmVjdEFyZWEgcmVzID0gY2hlY2sobmV3IE11bHRpUmVjdEFyZWFPcC5VbmlvbigpLmdldFJlc3VsdChzcmMxLCBzcmMyKSwgInVuaW9uKE1SQSxNUkEpIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTdWJ0cmFjdCB0d28gTXVsdGlSZWN0QXJlYSBvYmplY3RzCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBNdWx0aVJlY3RBcmVhIHN1YnRyYWN0KE11bHRpUmVjdEFyZWEgc3JjMSwgTXVsdGlSZWN0QXJlYSBzcmMyKSB7Ci0gICAgICAgIE11bHRpUmVjdEFyZWEgcmVzID0gY2hlY2soTXVsdGlSZWN0QXJlYU9wLlN1YnRyYWN0aW9uLmdldFJlc3VsdChzcmMxLCBzcmMyKSwgInN1YnRyYWN0KE1SQSxNUkEpIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBQcmludCBNdWx0aVJlY3RBcmVhIG9iamVjdCB0byBvdXRwdXQgc3RyZWFtCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyB2b2lkIHByaW50KE11bHRpUmVjdEFyZWEgbXJhLCBTdHJpbmcgbXNnKSB7Ci0gICAgICAgIGlmIChtcmEgPT0gbnVsbCkgewotICAgICAgICAgICAgU3lzdGVtLm91dC5wcmludGxuKG1zZyArICI9bnVsbCIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBSZWN0YW5nbGVbXSByZWN0cyA9IG1yYS5nZXRSZWN0YW5nbGVzKCk7Ci0gICAgICAgICAgICBTeXN0ZW0ub3V0LnByaW50bG4obXNnICsgIigiICsgcmVjdHMubGVuZ3RoICsgIikiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICBmb3IgKFJlY3RhbmdsZSBlbGVtZW50IDogcmVjdHMpIHsKLSAgICAgICAgICAgICAgICBTeXN0ZW0ub3V0LnByaW50bG4oCi0gICAgICAgICAgICAgICAgICAgICAgICBlbGVtZW50LnggKyAiLCIgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgICAgICBlbGVtZW50LnkgKyAiLCIgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgICAgICAoZWxlbWVudC54ICsgZWxlbWVudC53aWR0aCAtIDEpICsgIiwiICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICAgICAgKGVsZW1lbnQueSArIGVsZW1lbnQuaGVpZ2h0IC0gMSkpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVHJhbnNsYXRlIE11bHRpUmVjdEFyZWEgb2JqZWN0IGJ5ICh4LCB5KQotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHRyYW5zbGF0ZShpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgZm9yKGludCBpID0gMTsgaSA8IHJlY3RbMF07KSB7Ci0gICAgICAgICAgICByZWN0W2krK10gKz0geDsKLSAgICAgICAgICAgIHJlY3RbaSsrXSArPSB5OwotICAgICAgICAgICAgcmVjdFtpKytdICs9IHg7Ci0gICAgICAgICAgICByZWN0W2krK10gKz0geTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChib3VuZHMgIT0gbnVsbCAmJiAhYm91bmRzLmlzRW1wdHkoKSkgewotICAgICAgICAgICAgYm91bmRzLnRyYW5zbGF0ZSh4LCB5KTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChyZWN0YW5nbGVzICE9IG51bGwpIHsKLSAgICAgICAgICAgIGZvciAoUmVjdGFuZ2xlIGVsZW1lbnQgOiByZWN0YW5nbGVzKSB7Ci0gICAgICAgICAgICAgICAgZWxlbWVudC50cmFuc2xhdGUoeCwgeSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGQgcmVjdGFuZ2xlIHRvIHRoZSBidWZmZXIgd2l0aG91dCBhbnkgY2hlY2tpbmcKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRSZWN0KGludCB4MSwgaW50IHkxLCBpbnQgeDIsIGludCB5MikgewotICAgICAgICBpbnQgaSA9IHJlY3RbMF07Ci0gICAgICAgIHJlY3QgPSBNdWx0aVJlY3RBcmVhT3AuY2hlY2tCdWZTaXplKHJlY3QsIDQpOwotICAgICAgICByZWN0W2krK10gPSB4MTsKLSAgICAgICAgcmVjdFtpKytdID0geTE7Ci0gICAgICAgIHJlY3RbaSsrXSA9IHgyOwotICAgICAgICByZWN0W2krK10gPSB5MjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUZXN0cyBpcyBNdWx0aVJlY3RBcmVhIGVtcHR5IAotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzRW1wdHkoKSB7Ci0gICAgICAgIHJldHVybiByZWN0WzBdID09IDE7Ci0gICAgfQotCi0gICAgdm9pZCBpbnZhbGlkYXRlKCkgewotICAgICAgICBib3VuZHMgPSBudWxsOwotICAgICAgICByZWN0YW5nbGVzID0gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGJvdW5kcyBvZiBNdWx0aVJlY3RBcmVhIG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0Qm91bmRzKCkgewotICAgICAgICBpZiAoYm91bmRzICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBib3VuZHM7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gYm91bmRzID0gbmV3IFJlY3RhbmdsZSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHgxID0gcmVjdFsxXTsKLSAgICAgICAgaW50IHkxID0gcmVjdFsyXTsKLSAgICAgICAgaW50IHgyID0gcmVjdFszXTsKLSAgICAgICAgaW50IHkyID0gcmVjdFs0XTsKLSAgICAgICAgCi0gICAgICAgIGZvcihpbnQgaSA9IDU7IGkgPCByZWN0WzBdOyBpICs9IDQpIHsKLSAgICAgICAgICAgIGludCByeDEgPSByZWN0W2kgKyAwXTsKLSAgICAgICAgICAgIGludCByeTEgPSByZWN0W2kgKyAxXTsKLSAgICAgICAgICAgIGludCByeDIgPSByZWN0W2kgKyAyXTsKLSAgICAgICAgICAgIGludCByeTIgPSByZWN0W2kgKyAzXTsKLSAgICAgICAgICAgIGlmIChyeDEgPCB4MSkgewotICAgICAgICAgICAgICAgIHgxID0gcngxOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHJ4MiA+IHgyKSB7Ci0gICAgICAgICAgICAgICAgeDIgPSByeDI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAocnkxIDwgeTEpIHsKLSAgICAgICAgICAgICAgICB5MSA9IHJ5MTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChyeTIgPiB5MikgewotICAgICAgICAgICAgICAgIHkyID0gcnkyOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICByZXR1cm4gYm91bmRzID0gbmV3IFJlY3RhbmdsZSh4MSwgeTEsIHgyIC0geDEgKyAxLCB5MiAtIHkxICsgMSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVjdHVybiByZWN0YW5nbGUgY291bnQgaW4gdGhlIGJ1ZmZlcgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0UmVjdENvdW50KCkgewotICAgICAgICByZXR1cm4gKHJlY3RbMF0gLSAxKSAvIDQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBSZWN0YW5nbGUgYXJyYXkgCi0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZVtdIGdldFJlY3RhbmdsZXMoKSB7Ci0gICAgICAgIGlmIChyZWN0YW5nbGVzICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiByZWN0YW5nbGVzOwotICAgICAgICB9Ci0KLSAgICAgICAgcmVjdGFuZ2xlcyA9IG5ldyBSZWN0YW5nbGVbKHJlY3RbMF0gLSAxKSAvIDRdOwotICAgICAgICBpbnQgaiA9IDA7Ci0gICAgICAgIGZvcihpbnQgaSA9IDE7IGkgPCByZWN0WzBdOyBpICs9IDQpIHsKLSAgICAgICAgICAgIHJlY3RhbmdsZXNbaisrXSA9IG5ldyBSZWN0YW5nbGUoCi0gICAgICAgICAgICAgICAgICAgIHJlY3RbaV0sCi0gICAgICAgICAgICAgICAgICAgIHJlY3RbaSArIDFdLAotICAgICAgICAgICAgICAgICAgICByZWN0W2kgKyAyXSAtIHJlY3RbaV0gKyAxLAotICAgICAgICAgICAgICAgICAgICByZWN0W2kgKyAzXSAtIHJlY3RbaSArIDFdICsgMSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlY3RhbmdsZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBCb3VuZHMyRAotICAgICAqLwotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRCb3VuZHMyRCgpIHsKLSAgICAgICAgcmV0dXJuIGdldEJvdW5kcygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRlc3RzIGRvZXMgcG9pbnQgbGllIGluc2lkZSBNdWx0aVJlY3RBcmVhIG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSB4LCBkb3VibGUgeSkgewotICAgICAgICBmb3IoaW50IGkgPSAxOyBpIDwgcmVjdFswXTsgaSs9IDQpIHsKLSAgICAgICAgICAgIGlmIChyZWN0W2ldIDw9IHggJiYgeCA8PSByZWN0W2kgKyAyXSAmJiByZWN0W2kgKyAxXSA8PSB5ICYmIHkgPD0gcmVjdFtpICsgM10pIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGVzdHMgZG9lcyBQb2ludDJEIGxpZSBpbnNpZGUgTXVsdGlSZWN0QXJlYSBvYmplY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBjb250YWlucyhQb2ludDJEIHApIHsKLSAgICAgICAgcmV0dXJuIGNvbnRhaW5zKHAuZ2V0WCgpLCBwLmdldFkoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGVzdHMgZG9lcyByZWN0YW5nbGUgbGllIGluc2lkZSBNdWx0aVJlY3RBcmVhIG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKGRvdWJsZSB4LCBkb3VibGUgeSwgZG91YmxlIHcsIGRvdWJsZSBoKSB7Ci0gICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRlc3RzIGRvZXMgUmVjdGFuZ2xlMkQgbGllIGluc2lkZSBNdWx0aVJlY3RBcmVhIG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGNvbnRhaW5zKFJlY3RhbmdsZTJEIHIpIHsKLSAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCIpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGVzdHMgZG9lcyByZWN0YW5nbGUgaW50ZXJzZWN0IE11bHRpUmVjdEFyZWEgb2JqZWN0Ci0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaW50ZXJzZWN0cyhkb3VibGUgeCwgZG91YmxlIHksIGRvdWJsZSB3LCBkb3VibGUgaCkgewotICAgICAgICBSZWN0YW5nbGUgciA9IG5ldyBSZWN0YW5nbGUoKTsKLSAgICAgICAgci5zZXRSZWN0KHgsIHksIHcsIGgpOwotICAgICAgICByZXR1cm4gaW50ZXJzZWN0cyhyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUZXN0cyBkb2VzIFJlY3RhbmdsZTJEIGludGVyc2VjdCBNdWx0aVJlY3RBcmVhIG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGludGVyc2VjdHMoUmVjdGFuZ2xlMkQgcikgewotICAgICAgICBpZiAociA9PSBudWxsIHx8IHIuaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgZm9yKGludCBpID0gMTsgaSA8IHJlY3RbMF07IGkrPSA0KSB7Ci0gICAgICAgICAgICBpZiAoci5pbnRlcnNlY3RzKHJlY3RbaV0sIHJlY3RbaSsxXSwgcmVjdFtpICsgMl0tcmVjdFtpXSsxLCByZWN0W2kgKyAzXS1yZWN0W2kgKyAxXSsxKSkgewotICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHBhdGggaXRlcmF0b3IKLSAgICAgKi8KLSAgICBwdWJsaWMgUGF0aEl0ZXJhdG9yIGdldFBhdGhJdGVyYXRvcihBZmZpbmVUcmFuc2Zvcm0gdCwgZG91YmxlIGZsYXRuZXNzKSB7Ci0gICAgICAgIHJldHVybiBuZXcgSXRlcmF0b3IodGhpcywgdCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBwYXRoIGl0ZXJhdG9yCi0gICAgICovCi0gICAgcHVibGljIFBhdGhJdGVyYXRvciBnZXRQYXRoSXRlcmF0b3IoQWZmaW5lVHJhbnNmb3JtIHQpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJdGVyYXRvcih0aGlzLCB0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIE11bHRpUmVjdEFyZWEgb2JqZWN0IGNvbnZlcnRlZCB0byBzdHJpbmcgCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsKLSAgICAgICAgaW50IGNudCA9IGdldFJlY3RDb3VudCgpOwotICAgICAgICBTdHJpbmdCdWZmZXIgc2IgPSBuZXcgU3RyaW5nQnVmZmVyKChjbnQgPDwgNSkgKyAxMjgpOwotICAgICAgICBzYi5hcHBlbmQoZ2V0Q2xhc3MoKS5nZXROYW1lKCkpLmFwcGVuZCgiIFsiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBmb3IoaW50IGkgPSAxOyBpIDwgcmVjdFswXTsgaSArPSA0KSB7Ci0gICAgICAgICAgICBzYi5hcHBlbmQoaSA+IDEgPyAiLCBbIiA6ICJbIikuYXBwZW5kKHJlY3RbaV0pLmFwcGVuZCgiLCAiKS5hcHBlbmQocmVjdFtpICsgMV0pLiAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgICAgICAgICAgYXBwZW5kKCIsICIpLmFwcGVuZChyZWN0W2kgKyAyXSAtIHJlY3RbaV0gKyAxKS5hcHBlbmQoIiwgIikuIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgICAgICAgICAgYXBwZW5kKHJlY3RbaSArIDNdIC0gcmVjdFtpICsgMV0gKyAxKS5hcHBlbmQoIl0iKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBzYi5hcHBlbmQoIl0iKS50b1N0cmluZygpOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi19Ci0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL011bHRpUmVjdEFyZWFPcC5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvTXVsdGlSZWN0QXJlYU9wLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM3NWUyMDMuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvTXVsdGlSZWN0QXJlYU9wLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4MzcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2w7Ci0KLWltcG9ydCBqYXZhLmF3dC5SZWN0YW5nbGU7Ci0KLXB1YmxpYyBjbGFzcyBNdWx0aVJlY3RBcmVhT3AgewotCi0gICAgLyoqCi0gICAgICogUmVjdGFuZ2xlIGJ1ZmZlciBjYXBhY2l0eQotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFJFQ1RfQ0FQQUNJVFkgPSAxNjsKLSAgICAKLSAgICAvKioKLSAgICAgKiBJZiBudW1iZXIgb2YgcmVjdGFuZ2xlIGluIE11bHRpUmVjdEFyZWEgb2JqZWN0IGxlc3MgdGhhbiBNQVhfU0lNUExFIHNpbXBsZSBhbGdvcml0aG0gYXBwbGllcyAKLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgTUFYX1NJTVBMRSA9IDg7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGUgYnVmZmVyCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnRbXSBjcmVhdGVCdWYoaW50IGNhcGFjaXR5KSB7Ci0gICAgICAgIGlmIChjYXBhY2l0eSA9PSAwKSB7Ci0gICAgICAgICAgICBjYXBhY2l0eSA9IFJFQ1RfQ0FQQUNJVFk7Ci0gICAgICAgIH0KLSAgICAgICAgaW50W10gYnVmID0gbmV3IGludFtjYXBhY2l0eV07Ci0gICAgICAgIGJ1ZlswXSA9IDE7Ci0gICAgICAgIHJldHVybiBidWY7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGJ1ZmZlciBzaXplIGFuZCByZWFsbG9jYXRlIGlmIG5lY2Vzc2FyeSAgCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnRbXSBjaGVja0J1ZlNpemUoaW50W10gYnVmLCBpbnQgY2FwYWNpdHkpIHsKLSAgICAgICAgaWYgKGJ1ZlswXSArIGNhcGFjaXR5ID49IGJ1Zi5sZW5ndGgpIHsKLSAgICAgICAgICAgIGludCBsZW5ndGggPSBidWZbMF0gKyAoY2FwYWNpdHkgPiBSRUNUX0NBUEFDSVRZID8gY2FwYWNpdHkgOiBSRUNUX0NBUEFDSVRZKTsKLSAgICAgICAgICAgIGludFtdIHRtcCA9IG5ldyBpbnRbbGVuZ3RoXTsKLSAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkoYnVmLCAwLCB0bXAsIDAsIGJ1ZlswXSk7Ci0gICAgICAgICAgICBidWYgPSB0bXA7Ci0gICAgICAgIH0KLSAgICAgICAgYnVmWzBdICs9IGNhcGFjaXR5OwotICAgICAgICByZXR1cm4gYnVmOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlZ2lvbiBjbGFzcyBwcm92aWRlcyBiYXNpYyBmdW5jdGlvbmxpdHkgZm9yIE11bHRpUmVjdEFyZWEgb2JqZWN0cyB0byBtYWtlIGxvZ2ljYWwgb3BlcmF0aW9ucyAKLSAgICAgKi8KLSAgICBzdGF0aWMgY2xhc3MgUmVnaW9uIHsKLQotICAgICAgICBpbnRbXSByZWdpb247Ci0gICAgICAgIGludFtdIGFjdGl2ZTsKLSAgICAgICAgaW50W10gYm90dG9tOwotICAgICAgICBpbnQgaW5kZXg7Ci0KLSAgICAgICAgcHVibGljIFJlZ2lvbihpbnRbXSByZWdpb24pIHsKLSAgICAgICAgICAgIHRoaXMucmVnaW9uID0gcmVnaW9uOwotICAgICAgICAgICAgYWN0aXZlID0gbmV3IGludFtSRUNUX0NBUEFDSVRZXTsKLSAgICAgICAgICAgIGJvdHRvbSA9IG5ldyBpbnRbUkVDVF9DQVBBQ0lUWV07Ci0gICAgICAgICAgICBhY3RpdmVbMF0gPSAxOwotICAgICAgICAgICAgYm90dG9tWzBdID0gMTsKLSAgICAgICAgICAgIGluZGV4ID0gMTsKLSAgICAgICAgfQotCi0gICAgICAgIHZvaWQgYWRkQWN0aXZlKGludCBpbmRleCkgewotICAgICAgICAgICAgaW50IGxlbmd0aCA9IGFjdGl2ZVswXTsKLSAgICAgICAgICAgIGFjdGl2ZSA9IGNoZWNrQnVmU2l6ZShhY3RpdmUsIDQpOwotICAgICAgICAgICAgaW50IGkgPSAxOwotCi0gICAgICAgICAgICB3aGlsZShpIDwgbGVuZ3RoKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHJlZ2lvbltpbmRleF0gPCBhY3RpdmVbaV0pIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gSW5zZXJ0Ci0gICAgICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkoYWN0aXZlLCBpLCBhY3RpdmUsIGkgKyA0LCBsZW5ndGggLSBpKTsKLSAgICAgICAgICAgICAgICAgICAgbGVuZ3RoID0gaTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGkgKz0gNDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkocmVnaW9uLCBpbmRleCwgYWN0aXZlLCBsZW5ndGgsIDQpOwotCi0gICAgICAgIH0KLQotICAgICAgICB2b2lkIGZpbmRBY3RpdmUoaW50IHRvcCwgaW50IGJvdHRvbSkgewotICAgICAgICAgICAgd2hpbGUoaW5kZXggPCByZWdpb25bMF0pIHsKLSAgICAgICAgICAgICAgICBpZiAocmVnaW9uW2luZGV4ICsgMV0gPiBib3R0b20pIHsgLy8geTEgPiBib3R0b20KLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAocmVnaW9uW2luZGV4ICsgM10gPj0gdG9wKSB7IC8vIHkyID49IHRvcAotICAgICAgICAgICAgICAgICAgICBhZGRBY3RpdmUoaW5kZXgpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpbmRleCArPSA0OwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgdm9pZCBkZWxldGVBY3RpdmUoaW50IGJvdHRvbSkgewotICAgICAgICAgICAgaW50IGxlbmd0aCA9IGFjdGl2ZVswXTsKLSAgICAgICAgICAgIGZvcihpbnQgaSA9IDE7IGkgPCBsZW5ndGg7KSB7Ci0gICAgICAgICAgICAgICAgaWYgKGFjdGl2ZVtpICsgM10gPT0gYm90dG9tKSB7Ci0gICAgICAgICAgICAgICAgICAgIGxlbmd0aCAtPSA0OwotICAgICAgICAgICAgICAgICAgICBpZiAoaSA8IGxlbmd0aCkgewotICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShhY3RpdmUsIGkgKyA0LCBhY3RpdmUsIGksIGxlbmd0aCAtIGkpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgIGkgKz0gNDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBhY3RpdmVbMF0gPSBsZW5ndGg7Ci0gICAgICAgIH0KLQotICAgICAgICB2b2lkIGRlbGV0ZUFjdGl2ZSgpIHsKLSAgICAgICAgICAgIGludCBsZW5ndGggPSBhY3RpdmVbMF07Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSBsZW5ndGggLSA0OyBpID4gMDsgaSAtPSA0KSB7Ci0gICAgICAgICAgICAgICAgaWYgKGFjdGl2ZVtpICsgMV0gPiBhY3RpdmVbaSArIDNdKSB7Ci0gICAgICAgICAgICAgICAgICAgIGxlbmd0aCAtPSA0OwotICAgICAgICAgICAgICAgICAgICBpZiAoaSA8IGxlbmd0aCkgewotICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShhY3RpdmUsIGkgKyA0LCBhY3RpdmUsIGksIGxlbmd0aCAtIGkpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgYWN0aXZlWzBdID0gbGVuZ3RoOwotICAgICAgICB9Ci0KLSAgICAgICAgdm9pZCBjcmVhdGVMZXZlbChpbnRbXSBsZXZlbCkgewotICAgICAgICAgICAgaW50IGxldmVsQ291bnQgPSAxOwotICAgICAgICAgICAgaW50IHRvcEluZGV4ID0gMTsKLSAgICAgICAgICAgIGludCBpID0gMTsKLSAgICAgICAgICAgIHdoaWxlKGkgPCByZWdpb25bMF0pIHsKLQotICAgICAgICAgICAgICAgIGludCB0b3AgPSByZWdpb25baSArIDFdOwotICAgICAgICAgICAgICAgIGludCBib3R0b20gPSByZWdpb25baSArIDNdICsgMTsKLSAgICAgICAgICAgICAgICBpbnQgaiA9IHRvcEluZGV4OwotCi0gICAgICAgICAgICAgICAgYWRkVG9wOiB7Ci0gICAgICAgICAgICAgICAgICAgIHdoaWxlKGogPCBsZXZlbENvdW50KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAobGV2ZWxbal0gPT0gdG9wKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWsgYWRkVG9wOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGxldmVsW2pdID4gdG9wKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShsZXZlbCwgaiwgbGV2ZWwsIGogKyAxLCBsZXZlbENvdW50IC0gaik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBqKys7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBsZXZlbFtqXSA9IHRvcDsKLSAgICAgICAgICAgICAgICAgICAgbGV2ZWxDb3VudCsrOwotICAgICAgICAgICAgICAgICAgICB0b3BJbmRleCA9IGo7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgYWRkQm90dG9tOiB7Ci0gICAgICAgICAgICAgICAgICAgIHdoaWxlKGogPCBsZXZlbENvdW50KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAobGV2ZWxbal0gPT0gYm90dG9tKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWsgYWRkQm90dG9tOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGxldmVsW2pdID4gYm90dG9tKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShsZXZlbCwgaiwgbGV2ZWwsIGogKyAxLCBsZXZlbENvdW50IC0gaik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBqKys7Ci0gICAgICAgICAgICAgICAgICAgIH07Ci0KLSAgICAgICAgICAgICAgICAgICAgbGV2ZWxbal0gPSBib3R0b207Ci0gICAgICAgICAgICAgICAgICAgIGxldmVsQ291bnQrKzsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBpICs9IDQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBsZXZlbFswXSA9IGxldmVsQ291bnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBzdGF0aWMgdm9pZCBzb3J0T3JkZXJlZChpbnRbXSBzcmMxLCBpbnRbXSBzcmMyLCBpbnRbXSBkc3QpIHsKLSAgICAgICAgICAgIGludCBsZW5ndGgxID0gc3JjMVswXTsKLSAgICAgICAgICAgIGludCBsZW5ndGgyID0gc3JjMlswXTsKLSAgICAgICAgICAgIGludCBjb3VudCA9IDE7Ci0gICAgICAgICAgICBpbnQgaTEgPSAxOwotICAgICAgICAgICAgaW50IGkyID0gMTsKLSAgICAgICAgICAgIGludCB2MSA9IHNyYzFbMV07Ci0gICAgICAgICAgICBpbnQgdjIgPSBzcmMyWzFdOwotICAgICAgICAgICAgd2hpbGUodHJ1ZSkgewotCi0gICAgICAgICAgICAgICAgTEVGVDogewotICAgICAgICAgICAgICAgICAgICB3aGlsZShpMSA8IGxlbmd0aDEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHYxID0gc3JjMVtpMV07Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAodjEgPj0gdjIpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhayBMRUZUOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgZHN0W2NvdW50KytdID0gdjE7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpMSsrOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHdoaWxlKGkyIDwgbGVuZ3RoMikgewotICAgICAgICAgICAgICAgICAgICAgICAgZHN0W2NvdW50KytdID0gc3JjMltpMisrXTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBkc3RbMF0gPSBjb3VudDsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIFJJR0hUOiB7Ci0gICAgICAgICAgICAgICAgICAgIHdoaWxlKGkyIDwgbGVuZ3RoMikgewotICAgICAgICAgICAgICAgICAgICAgICAgdjIgPSBzcmMyW2kyXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmICh2MiA+PSB2MSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrIFJJR0hUOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgZHN0W2NvdW50KytdID0gdjI7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpMisrOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHdoaWxlKGkxIDwgbGVuZ3RoMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgZHN0W2NvdW50KytdID0gc3JjMVtpMSsrXTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBkc3RbMF0gPSBjb3VudDsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmICh2MSA9PSB2MikgewotICAgICAgICAgICAgICAgICAgICBkc3RbY291bnQrK10gPSB2MTsKLSAgICAgICAgICAgICAgICAgICAgaTErKzsKLSAgICAgICAgICAgICAgICAgICAgaTIrKzsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGkxIDwgbGVuZ3RoMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgdjEgPSBzcmMxW2kxXTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpZiAoaTIgPCBsZW5ndGgyIC0gMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgdjIgPSBzcmMyW2kyXTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIC8vIFVOUkVBQ0hBQkxFCi0gICAgICAgIH0KLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEludGVyc2VjdGlvbiBjbGFzcyBwcm92aWRlcyBpbnRlcnNlY3Rpb24gb2YgdHdvIE11bHRpUmVjdEFyZSBhb2JqZWN0cwotICAgICAqLwotICAgIHN0YXRpYyBjbGFzcyBJbnRlcnNlY3Rpb24gewotCi0gICAgICAgIHN0YXRpYyB2b2lkIGludGVyc2VjdFJlZ2lvbnMoaW50W10gcmVnMSwgaW50W10gcmVnMiwgTXVsdGlSZWN0QXJlYS5SZWN0Q2FzaCBkc3QsIGludCBoZWlnaHQxLCBpbnQgaGVpZ2h0MikgewotCi0gICAgICAgICAgICBSZWdpb24gZDEgPSBuZXcgUmVnaW9uKHJlZzEpOwotICAgICAgICAgICAgUmVnaW9uIGQyID0gbmV3IFJlZ2lvbihyZWcyKTsKLQotICAgICAgICAgICAgaW50W10gbGV2ZWwgPSBuZXcgaW50W2hlaWdodDEgKyBoZWlnaHQyXTsKLSAgICAgICAgICAgIGludFtdIGxldmVsMSA9IG5ldyBpbnRbaGVpZ2h0MV07Ci0gICAgICAgICAgICBpbnRbXSBsZXZlbDIgPSBuZXcgaW50W2hlaWdodDJdOwotICAgICAgICAgICAgZDEuY3JlYXRlTGV2ZWwobGV2ZWwxKTsKLSAgICAgICAgICAgIGQyLmNyZWF0ZUxldmVsKGxldmVsMik7Ci0gICAgICAgICAgICBSZWdpb24uc29ydE9yZGVyZWQobGV2ZWwxLCBsZXZlbDIsIGxldmVsKTsKLQotICAgICAgICAgICAgaW50IHRvcDsKLSAgICAgICAgICAgIGludCBib3R0b20gPSBsZXZlbFsxXSAtIDE7Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSAyOyBpIDwgbGV2ZWxbMF07IGkrKykgewotCi0gICAgICAgICAgICAgICAgdG9wID0gYm90dG9tICsgMTsKLSAgICAgICAgICAgICAgICBib3R0b20gPSBsZXZlbFtpXSAtIDE7Ci0KLSAgICAgICAgICAgICAgICBkMS5maW5kQWN0aXZlKHRvcCwgYm90dG9tKTsKLSAgICAgICAgICAgICAgICBkMi5maW5kQWN0aXZlKHRvcCwgYm90dG9tKTsKLQotICAgICAgICAgICAgICAgIGludCBpMSA9IDE7Ci0gICAgICAgICAgICAgICAgaW50IGkyID0gMTsKLQotICAgICAgICAgICAgICAgIHdoaWxlKGkxIDwgZDEuYWN0aXZlWzBdICYmIGkyIDwgZDIuYWN0aXZlWzBdKSB7Ci0KLSAgICAgICAgICAgICAgICAgICAgaW50IHgxMSA9IGQxLmFjdGl2ZVtpMV07Ci0gICAgICAgICAgICAgICAgICAgIGludCB4MTIgPSBkMS5hY3RpdmVbaTEgKyAyXTsKLSAgICAgICAgICAgICAgICAgICAgaW50IHgyMSA9IGQyLmFjdGl2ZVtpMl07Ci0gICAgICAgICAgICAgICAgICAgIGludCB4MjIgPSBkMi5hY3RpdmVbaTIgKyAyXTsKLQotICAgICAgICAgICAgICAgICAgICBpZiAoeDExIDw9IHgyMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHgxMiA+PSB4MjEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoeDEyIDw9IHgyMikgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdENhc2hlZCh4MjEsIHRvcCwgeDEyLCBib3R0b20pOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpMSArPSA0OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5hZGRSZWN0Q2FzaGVkKHgyMSwgdG9wLCB4MjIsIGJvdHRvbSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGkyICs9IDQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpMSArPSA0OwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHgyMiA+PSB4MTEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoeDIyIDw9IHgxMikgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdENhc2hlZCh4MTEsIHRvcCwgeDIyLCBib3R0b20pOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpMiArPSA0OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5hZGRSZWN0Q2FzaGVkKHgxMSwgdG9wLCB4MTIsIGJvdHRvbSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGkxICs9IDQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpMiArPSA0OwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgZDEuZGVsZXRlQWN0aXZlKGJvdHRvbSk7Ci0gICAgICAgICAgICAgICAgZDIuZGVsZXRlQWN0aXZlKGJvdHRvbSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBzdGF0aWMgaW50W10gc2ltcGxlSW50ZXJzZWN0KE11bHRpUmVjdEFyZWEgc3JjMSwgTXVsdGlSZWN0QXJlYSBzcmMyKSB7Ci0gICAgICAgICAgICBpbnRbXSByZWN0MSA9IHNyYzEucmVjdDsKLSAgICAgICAgICAgIGludFtdIHJlY3QyID0gc3JjMi5yZWN0OwotICAgICAgICAgICAgaW50W10gcmVjdCA9IGNyZWF0ZUJ1ZigwKTsKLQotICAgICAgICAgICAgaW50IGsgPSAxOwotICAgICAgICAgICAgZm9yKGludCBpID0gMTsgaSA8IHJlY3QxWzBdOykgewotCi0gICAgICAgICAgICAgICAgaW50IHgxMSA9IHJlY3QxW2krK107Ci0gICAgICAgICAgICAgICAgaW50IHkxMSA9IHJlY3QxW2krK107Ci0gICAgICAgICAgICAgICAgaW50IHgxMiA9IHJlY3QxW2krK107Ci0gICAgICAgICAgICAgICAgaW50IHkxMiA9IHJlY3QxW2krK107Ci0KLSAgICAgICAgICAgICAgICBmb3IoaW50IGogPSAxOyBqIDwgcmVjdDJbMF07KSB7Ci0KLSAgICAgICAgICAgICAgICAgICAgaW50IHgyMSA9IHJlY3QyW2orK107Ci0gICAgICAgICAgICAgICAgICAgIGludCB5MjEgPSByZWN0MltqKytdOwotICAgICAgICAgICAgICAgICAgICBpbnQgeDIyID0gcmVjdDJbaisrXTsKLSAgICAgICAgICAgICAgICAgICAgaW50IHkyMiA9IHJlY3QyW2orK107Ci0KLSAgICAgICAgICAgICAgICAgICAgaWYgKHgxMSA8PSB4MjIgJiYgeDEyID49IHgyMSAmJgotICAgICAgICAgICAgICAgICAgICAgICAgeTExIDw9IHkyMiAmJiB5MTIgPj0geTIxKQotICAgICAgICAgICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZWN0ID0gY2hlY2tCdWZTaXplKHJlY3QsIDQpOwotICAgICAgICAgICAgICAgICAgICAgICAgcmVjdFtrKytdID0geDExID4geDIxID8geDExIDogeDIxOwotICAgICAgICAgICAgICAgICAgICAgICAgcmVjdFtrKytdID0geTExID4geTIxID8geTExIDogeTIxOwotICAgICAgICAgICAgICAgICAgICAgICAgcmVjdFtrKytdID0geDEyID4geDIyID8geDIyIDogeDEyOwotICAgICAgICAgICAgICAgICAgICAgICAgcmVjdFtrKytdID0geTEyID4geTIyID8geTIyIDogeTEyOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZWN0WzBdID0gazsKLSAgICAgICAgICAgIHJldHVybiByZWN0OwotICAgICAgICB9Ci0KLSAgICAgICAgcHVibGljIHN0YXRpYyBNdWx0aVJlY3RBcmVhIGdldFJlc3VsdChNdWx0aVJlY3RBcmVhIHNyYzEsIE11bHRpUmVjdEFyZWEgc3JjMikgewotCi0gICAgICAgICAgICBpZiAoc3JjMSA9PSBudWxsIHx8IHNyYzIgPT0gbnVsbCB8fCBzcmMxLmlzRW1wdHkoKSB8fCBzcmMyLmlzRW1wdHkoKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBuZXcgTXVsdGlSZWN0QXJlYSgpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBNdWx0aVJlY3RBcmVhLlJlY3RDYXNoIGRzdCA9IG5ldyBNdWx0aVJlY3RBcmVhLlJlY3RDYXNoKCk7Ci0KLSAgICAgICAgICAgIGlmICghc3JjMS5zb3J0ZWQgfHwgIXNyYzIuc29ydGVkIHx8IAotICAgICAgICAgICAgICAgc3JjMS5nZXRSZWN0Q291bnQoKSA8PSBNQVhfU0lNUExFIHx8IHNyYzIuZ2V0UmVjdENvdW50KCkgPD0gTUFYX1NJTVBMRSkgCi0gICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgZHN0LnNldFJlY3Qoc2ltcGxlSW50ZXJzZWN0KHNyYzEsIHNyYzIpLCBmYWxzZSk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIFJlY3RhbmdsZSBib3VuZHMxID0gc3JjMS5nZXRCb3VuZHMoKTsKLSAgICAgICAgICAgICAgICBSZWN0YW5nbGUgYm91bmRzMiA9IHNyYzIuZ2V0Qm91bmRzKCk7Ci0gICAgICAgICAgICAgICAgUmVjdGFuZ2xlIGJvdW5kczMgPSBib3VuZHMxLmludGVyc2VjdGlvbihib3VuZHMyKTsKLSAgICAgICAgICAgICAgICBpZiAoYm91bmRzMy53aWR0aCA+IDAgJiYgYm91bmRzMy5oZWlnaHQgPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGludGVyc2VjdFJlZ2lvbnMoc3JjMS5yZWN0LCBzcmMyLnJlY3QsIGRzdCwgYm91bmRzMS5oZWlnaHQgKyAyLCBib3VuZHMyLmhlaWdodCArIDIpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIGRzdDsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogVW5pb24gY2xhc3MgcHJvdmlkZXMgdW5pb24gb2YgdHdvIE11bHRpUmVjdEFyZSBhb2JqZWN0cwotICAgICAqLwotICAgIHN0YXRpYyBjbGFzcyBVbmlvbiB7Ci0KLSAgICAgICAgaW50IHJ4MSwgcngyOwotICAgICAgICBpbnQgdG9wLCBib3R0b207Ci0gICAgICAgIE11bHRpUmVjdEFyZWEuUmVjdENhc2ggZHN0OwotCi0gICAgICAgIGJvb2xlYW4gbmV4dChSZWdpb24gZCwgaW50IGluZGV4KSB7Ci0gICAgICAgICAgICBpbnQgeDEgPSBkLmFjdGl2ZVtpbmRleF07Ci0gICAgICAgICAgICBpbnQgeDIgPSBkLmFjdGl2ZVtpbmRleCArIDJdOwotICAgICAgICAgICAgYm9vbGVhbiByZXMgPSBmYWxzZTsKLQotICAgICAgICAgICAgaWYgKHgyIDwgcngxIC0gMSkgewotICAgICAgICAgICAgICAgIHJlcyA9IHRydWU7Ci0gICAgICAgICAgICAgICAgZHN0LmFkZFJlY3RDYXNoZWQoeDEsIHRvcCwgeDIsIGJvdHRvbSk7Ci0gICAgICAgICAgICB9IGVsc2UKLSAgICAgICAgICAgICAgICBpZiAoeDEgPiByeDIgKyAxKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJlcyA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdENhc2hlZChyeDEsIHRvcCwgcngyLCBib3R0b20pOwotICAgICAgICAgICAgICAgICAgICByeDEgPSB4MTsKLSAgICAgICAgICAgICAgICAgICAgcngyID0geDI7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgcmVzID0geDIgPD0gcngyOwotICAgICAgICAgICAgICAgICAgICByeDEgPSBNYXRoLm1pbih4MSwgcngxKTsKLSAgICAgICAgICAgICAgICAgICAgcngyID0gTWF0aC5tYXgoeDIsIHJ4Mik7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBUb3AKLSAgICAgICAgICAgIGlmIChkLmFjdGl2ZVtpbmRleCArIDFdIDwgdG9wKSB7Ci0gICAgICAgICAgICAgICAgZHN0LmFkZFJlY3RDYXNoZWQoeDEsIGQuYWN0aXZlW2luZGV4ICsgMV0sIHgyLCB0b3AgLSAxKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIC8vIEJvdHRvbQotICAgICAgICAgICAgaWYgKGQuYWN0aXZlW2luZGV4ICsgM10gPiBib3R0b20pIHsKLSAgICAgICAgICAgICAgICBkLmFjdGl2ZVtpbmRleCArIDFdID0gYm90dG9tICsgMTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiByZXM7Ci0gICAgICAgIH0KLQotICAgICAgICB2b2lkIGNoZWNrKFJlZ2lvbiBkLCBpbnQgaW5kZXgsIGJvb2xlYW4gdCkgewotICAgICAgICAgICAgaW50IHgxID0gZC5hY3RpdmVbaW5kZXhdOwotICAgICAgICAgICAgaW50IHgyID0gZC5hY3RpdmVbaW5kZXggKyAyXTsKLSAgICAgICAgICAgIC8vIFRvcAotICAgICAgICAgICAgaWYgKGQuYWN0aXZlW2luZGV4ICsgMV0gPCB0b3ApIHsKLSAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdENhc2hlZCh4MSwgZC5hY3RpdmVbaW5kZXggKyAxXSwgeDIsIHRvcCAtIDEpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHQpIHsKLSAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdENhc2hlZCh4MSwgdG9wLCB4MiwgYm90dG9tKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIC8vIEJvdHRvbQotICAgICAgICAgICAgaWYgKGQuYWN0aXZlW2luZGV4ICsgM10gPiBib3R0b20pIHsKLSAgICAgICAgICAgICAgICBkLmFjdGl2ZVtpbmRleCArIDFdID0gYm90dG9tICsgMTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHZvaWQgdW5pb25SZWdpb25zKGludFtdIHJlZzEsIGludFtdIHJlZzIsIGludCBoZWlnaHQxLCBpbnQgaGVpZ2h0MikgewotICAgICAgICAgICAgUmVnaW9uIGQxID0gbmV3IFJlZ2lvbihyZWcxKTsKLSAgICAgICAgICAgIFJlZ2lvbiBkMiA9IG5ldyBSZWdpb24ocmVnMik7Ci0KLSAgICAgICAgICAgIGludFtdIGxldmVsID0gbmV3IGludFtoZWlnaHQxICsgaGVpZ2h0Ml07Ci0gICAgICAgICAgICBpbnRbXSBsZXZlbDEgPSBuZXcgaW50W2hlaWdodDFdOwotICAgICAgICAgICAgaW50W10gbGV2ZWwyID0gbmV3IGludFtoZWlnaHQyXTsKLSAgICAgICAgICAgIGQxLmNyZWF0ZUxldmVsKGxldmVsMSk7Ci0gICAgICAgICAgICBkMi5jcmVhdGVMZXZlbChsZXZlbDIpOwotICAgICAgICAgICAgUmVnaW9uLnNvcnRPcmRlcmVkKGxldmVsMSwgbGV2ZWwyLCBsZXZlbCk7Ci0KLSAgICAgICAgICAgIGJvdHRvbSA9IGxldmVsWzFdIC0gMTsKLSAgICAgICAgICAgIGZvcihpbnQgaSA9IDI7IGkgPCBsZXZlbFswXTsgaSsrKSB7Ci0KLSAgICAgICAgICAgICAgICB0b3AgPSBib3R0b20gKyAxOwotICAgICAgICAgICAgICAgIGJvdHRvbSA9IGxldmVsW2ldIC0gMTsKLQotICAgICAgICAgICAgICAgIGQxLmZpbmRBY3RpdmUodG9wLCBib3R0b20pOwotICAgICAgICAgICAgICAgIGQyLmZpbmRBY3RpdmUodG9wLCBib3R0b20pOwotCi0gICAgICAgICAgICAgICAgaW50IGkxID0gMTsKLSAgICAgICAgICAgICAgICBpbnQgaTIgPSAxOwotICAgICAgICAgICAgICAgIGJvb2xlYW4gcmVzMSwgcmVzMjsKLQotICAgICAgICAgICAgICAgIGlmIChkMS5hY3RpdmVbMF0gPiAxKSB7Ci0gICAgICAgICAgICAgICAgICAgIGNoZWNrKGQxLCAxLCBmYWxzZSk7Ci0gICAgICAgICAgICAgICAgICAgIHJ4MSA9IGQxLmFjdGl2ZVsxXTsKLSAgICAgICAgICAgICAgICAgICAgcngyID0gZDEuYWN0aXZlWzNdOwotICAgICAgICAgICAgICAgICAgICBpMSArPSA0OwotICAgICAgICAgICAgICAgICAgICByZXMxID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgICAgIHJlczIgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0gZWxzZQotICAgICAgICAgICAgICAgICAgICBpZiAoZDIuYWN0aXZlWzBdID4gMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgY2hlY2soZDIsIDEsIGZhbHNlKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJ4MSA9IGQyLmFjdGl2ZVsxXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJ4MiA9IGQyLmFjdGl2ZVszXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGkyICs9IDQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXMxID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlczIgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIG91dGVyOgotICAgICAgICAgICAgICAgIHdoaWxlKHRydWUpIHsKLQotICAgICAgICAgICAgICAgICAgICB3aGlsZSAocmVzMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGkxID49IGQxLmFjdGl2ZVswXSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5hZGRSZWN0Q2FzaGVkKHJ4MSwgdG9wLCByeDIsIGJvdHRvbSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgd2hpbGUoaTIgPCBkMi5hY3RpdmVbMF0pIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY2hlY2soZDIsIGkyLCB0cnVlKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaTIgKz0gNDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWsgb3V0ZXI7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXMxID0gbmV4dChkMSwgaTEpOwotICAgICAgICAgICAgICAgICAgICAgICAgaTEgKz0gNDsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIHdoaWxlIChyZXMyKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoaTIgPj0gZDIuYWN0aXZlWzBdKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0LmFkZFJlY3RDYXNoZWQocngxLCB0b3AsIHJ4MiwgYm90dG9tKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB3aGlsZShpMSA8IGQxLmFjdGl2ZVswXSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjaGVjayhkMSwgaTEsIHRydWUpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpMSArPSA0OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhayBvdXRlcjsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIHJlczIgPSBuZXh0KGQyLCBpMik7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpMiArPSA0OwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgcmVzMSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIHJlczIgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0gLy8gd2hpbGUKLQotICAgICAgICAgICAgICAgIGQxLmRlbGV0ZUFjdGl2ZShib3R0b20pOwotICAgICAgICAgICAgICAgIGQyLmRlbGV0ZUFjdGl2ZShib3R0b20pOwotCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBzdGF0aWMgdm9pZCBzaW1wbGVVbmlvbihNdWx0aVJlY3RBcmVhIHNyYzEsIE11bHRpUmVjdEFyZWEgc3JjMiwgTXVsdGlSZWN0QXJlYSBkc3QpIHsKLSAgICAgICAgICAgIGlmIChzcmMxLmdldFJlY3RDb3VudCgpIDwgc3JjMi5nZXRSZWN0Q291bnQoKSkgewotICAgICAgICAgICAgICAgIHNpbXBsZVVuaW9uKHNyYzIsIHNyYzEsIGRzdCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIFN1YnRyYWN0aW9uLnNpbXBsZVN1YnRyYWN0KHNyYzEsIHNyYzIsIGRzdCk7Ci0gICAgICAgICAgICAgICAgaW50IHBvcyA9IGRzdC5yZWN0WzBdOwotICAgICAgICAgICAgICAgIGludCBzaXplID0gc3JjMi5yZWN0WzBdIC0gMTsKLSAgICAgICAgICAgICAgICBkc3QucmVjdCA9IGNoZWNrQnVmU2l6ZShkc3QucmVjdCwgc2l6ZSk7Ci0gICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShzcmMyLnJlY3QsMSwgZHN0LnJlY3QsIHBvcywgc2l6ZSk7Ci0gICAgICAgICAgICAgICAgZHN0LnJlc29ydCgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgTXVsdGlSZWN0QXJlYSBnZXRSZXN1bHQoTXVsdGlSZWN0QXJlYSBzcmMxLCBNdWx0aVJlY3RBcmVhIHNyYzIpIHsKLQotICAgICAgICAgICAgaWYgKHNyYzEgPT0gbnVsbCB8fCBzcmMxLmlzRW1wdHkoKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBuZXcgTXVsdGlSZWN0QXJlYShzcmMyKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKHNyYzIgPT0gbnVsbCB8fCBzcmMyLmlzRW1wdHkoKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBuZXcgTXVsdGlSZWN0QXJlYShzcmMxKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZHN0ID0gbmV3IE11bHRpUmVjdEFyZWEuUmVjdENhc2goKTsKLQotICAgICAgICAgICAgaWYgKCFzcmMxLnNvcnRlZCB8fCAhc3JjMi5zb3J0ZWQgfHwKLSAgICAgICAgICAgICAgIHNyYzEuZ2V0UmVjdENvdW50KCkgPD0gTUFYX1NJTVBMRSB8fCBzcmMyLmdldFJlY3RDb3VudCgpIDw9IE1BWF9TSU1QTEUpIAotICAgICAgICAgICAgewotICAgICAgICAgICAgICAgIHNpbXBsZVVuaW9uKHNyYzEsIHNyYzIsIGRzdCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIFJlY3RhbmdsZSBib3VuZHMxID0gc3JjMS5nZXRCb3VuZHMoKTsKLSAgICAgICAgICAgICAgICBSZWN0YW5nbGUgYm91bmRzMiA9IHNyYzIuZ2V0Qm91bmRzKCk7Ci0gICAgICAgICAgICAgICAgUmVjdGFuZ2xlIGJvdW5kczMgPSBib3VuZHMxLmludGVyc2VjdGlvbihib3VuZHMyKTsKLQotICAgICAgICAgICAgICAgIGlmIChib3VuZHMzLndpZHRoIDwgMCB8fCBib3VuZHMzLmhlaWdodCA8IDApIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGJvdW5kczEueSArIGJvdW5kczEuaGVpZ2h0IDwgYm91bmRzMi55KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkc3Quc2V0UmVjdChhZGRWZXJSZWdpb24oc3JjMS5yZWN0LCBzcmMyLnJlY3QpLCBmYWxzZSk7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZQotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGJvdW5kczIueSArIGJvdW5kczIuaGVpZ2h0IDwgYm91bmRzMS55KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0LnNldFJlY3QoYWRkVmVyUmVnaW9uKHNyYzIucmVjdCwgc3JjMS5yZWN0KSwgZmFsc2UpOwotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGJvdW5kczEueCA8IGJvdW5kczIueCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3Quc2V0UmVjdChhZGRIb3JSZWdpb24oc3JjMS5yZWN0LCBzcmMyLnJlY3QpLCBmYWxzZSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0LnNldFJlY3QoYWRkSG9yUmVnaW9uKHNyYzIucmVjdCwgc3JjMS5yZWN0KSwgZmFsc2UpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICB1bmlvblJlZ2lvbnMoc3JjMS5yZWN0LCBzcmMyLnJlY3QsIGJvdW5kczEuaGVpZ2h0ICsgMiwgYm91bmRzMi5oZWlnaHQgKyAyKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBkc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnRbXSBhZGRWZXJSZWdpb24oaW50W10gdG9wLCBpbnRbXSBib3R0b20pIHsKLSAgICAgICAgICAgIGludCBsZW5ndGggPSB0b3BbMF0gKyBib3R0b21bMF0gLSAxOwotICAgICAgICAgICAgaW50W10gZHN0ID0gbmV3IGludFtsZW5ndGhdOwotICAgICAgICAgICAgZHN0WzBdID0gbGVuZ3RoOwotICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weSh0b3AsIDEsIGRzdCwgMSwgdG9wWzBdIC0gMSk7Ci0gICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGJvdHRvbSwgMSwgZHN0LCB0b3BbMF0sIGJvdHRvbVswXSAtIDEpOwotICAgICAgICAgICAgcmV0dXJuIGRzdDsKLSAgICAgICAgfQotCi0gICAgICAgIGludFtdIGFkZEhvclJlZ2lvbihpbnRbXSBsZWZ0LCBpbnRbXSByaWdodCkgewotICAgICAgICAgICAgaW50IGNvdW50MSA9IGxlZnRbMF07Ci0gICAgICAgICAgICBpbnQgY291bnQyID0gcmlnaHRbMF07Ci0gICAgICAgICAgICBpbnRbXSBkc3QgPSBuZXcgaW50W2NvdW50MSArIGNvdW50MiArIDFdOwotICAgICAgICAgICAgaW50IGNvdW50ID0gMTsKLSAgICAgICAgICAgIGludCBpbmRleDEgPSAxOwotICAgICAgICAgICAgaW50IGluZGV4MiA9IDE7Ci0KLSAgICAgICAgICAgIGludCB0b3AxID0gbGVmdFsyXTsKLSAgICAgICAgICAgIGludCB0b3AyID0gcmlnaHRbMl07Ci0gICAgICAgICAgICBpbnQgcG9zMSwgcG9zMjsKLQotICAgICAgICAgICAgd2hpbGUodHJ1ZSkgewotCi0gICAgICAgICAgICAgICAgaWYgKGluZGV4MSA+PSBjb3VudDEpIHsKLSAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShyaWdodCwgaW5kZXgyLCBkc3QsIGNvdW50LCBjb3VudDIgLSBpbmRleDIpOwotICAgICAgICAgICAgICAgICAgICBjb3VudCArPSBjb3VudDIgLSBpbmRleDI7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAoaW5kZXgyID49IGNvdW50MikgewotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGxlZnQsIGluZGV4MSwgZHN0LCBjb3VudCwgY291bnQxIC0gaW5kZXgxKTsKLSAgICAgICAgICAgICAgICAgICAgY291bnQgKz0gY291bnQxIC0gaW5kZXgxOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBpZiAodG9wMSA8IHRvcDIpIHsKLSAgICAgICAgICAgICAgICAgICAgcG9zMSA9IGluZGV4MTsKLSAgICAgICAgICAgICAgICAgICAgZG8gewotICAgICAgICAgICAgICAgICAgICAgICAgaW5kZXgxICs9IDQ7Ci0gICAgICAgICAgICAgICAgICAgIH0gd2hpbGUgKGluZGV4MSA8IGNvdW50MSAmJiAodG9wMSA9IGxlZnRbaW5kZXgxICsgMV0pIDwgdG9wMik7Ci0gICAgICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkobGVmdCwgcG9zMSwgZHN0LCBjb3VudCwgaW5kZXgxIC0gcG9zMSk7Ci0gICAgICAgICAgICAgICAgICAgIGNvdW50ICs9IGluZGV4MSAtIHBvczE7Ci0gICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmICh0b3AxID4gdG9wMikgewotICAgICAgICAgICAgICAgICAgICBwb3MyID0gaW5kZXgyOwotICAgICAgICAgICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpbmRleDIgKz0gNDsKLSAgICAgICAgICAgICAgICAgICAgfSB3aGlsZSAoaW5kZXgyIDwgY291bnQyICYmICh0b3AyID0gcmlnaHRbaW5kZXgyICsgMV0pIDwgdG9wMSk7Ci0gICAgICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkocmlnaHQsIHBvczIsIGRzdCwgY291bnQsIGluZGV4MiAtIHBvczIpOwotICAgICAgICAgICAgICAgICAgICBjb3VudCArPSBpbmRleDIgLSBwb3MyOwotICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBpbnQgdG9wID0gdG9wMTsKLSAgICAgICAgICAgICAgICBwb3MxID0gaW5kZXgxOwotICAgICAgICAgICAgICAgIHBvczIgPSBpbmRleDI7Ci0gICAgICAgICAgICAgICAgZG8gIHsKLSAgICAgICAgICAgICAgICAgICAgaW5kZXgxICs9IDQ7Ci0gICAgICAgICAgICAgICAgfSB3aGlsZShpbmRleDEgPCBjb3VudDEgJiYgKHRvcDEgPSBsZWZ0W2luZGV4MSArIDFdKSA9PSB0b3ApOwotICAgICAgICAgICAgICAgIGRvIHsKLSAgICAgICAgICAgICAgICAgICAgaW5kZXgyICs9IDQ7Ci0gICAgICAgICAgICAgICAgfSB3aGlsZShpbmRleDIgPCBjb3VudDIgJiYgKHRvcDIgPSByaWdodFtpbmRleDIgKyAxXSkgPT0gdG9wKTsKLQotICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkobGVmdCwgcG9zMSwgZHN0LCBjb3VudCwgaW5kZXgxIC0gcG9zMSk7Ci0gICAgICAgICAgICAgICAgY291bnQgKz0gaW5kZXgxIC0gcG9zMTsKLSAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHJpZ2h0LCBwb3MyLCBkc3QsIGNvdW50LCBpbmRleDIgLSBwb3MyKTsKLSAgICAgICAgICAgICAgICBjb3VudCArPSBpbmRleDIgLSBwb3MyOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBkc3RbMF0gPSBjb3VudDsKLSAgICAgICAgICAgIHJldHVybiBkc3Q7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFN1YnRyYWN0aW9uIGNsYXNzIHByb3ZpZGVzIHN1YnRyYWN0aW9uIG9mIHR3byBNdWx0aVJlY3RBcmUgYW9iamVjdHMKLSAgICAgKi8KLSAgICBzdGF0aWMgY2xhc3MgU3VidHJhY3Rpb24gewotCi0gICAgICAgIHN0YXRpYyB2b2lkIHN1YnRyYWN0UmVnaW9ucyhpbnRbXSByZWcxLCBpbnRbXSByZWcyLCBNdWx0aVJlY3RBcmVhLlJlY3RDYXNoIGRzdCwgaW50IGhlaWdodDEsIGludCBoZWlnaHQyKSB7Ci0gICAgICAgICAgICBSZWdpb24gZDEgPSBuZXcgUmVnaW9uKHJlZzEpOwotICAgICAgICAgICAgUmVnaW9uIGQyID0gbmV3IFJlZ2lvbihyZWcyKTsKLQotICAgICAgICAgICAgaW50W10gbGV2ZWwgPSBuZXcgaW50W2hlaWdodDEgKyBoZWlnaHQyXTsKLSAgICAgICAgICAgIGludFtdIGxldmVsMSA9IG5ldyBpbnRbaGVpZ2h0MV07Ci0gICAgICAgICAgICBpbnRbXSBsZXZlbDIgPSBuZXcgaW50W2hlaWdodDJdOwotICAgICAgICAgICAgZDEuY3JlYXRlTGV2ZWwobGV2ZWwxKTsKLSAgICAgICAgICAgIGQyLmNyZWF0ZUxldmVsKGxldmVsMik7Ci0gICAgICAgICAgICBSZWdpb24uc29ydE9yZGVyZWQobGV2ZWwxLCBsZXZlbDIsIGxldmVsKTsKLQotICAgICAgICAgICAgaW50IHRvcDsKLSAgICAgICAgICAgIGludCBib3R0b20gPSBsZXZlbFsxXSAtIDE7Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSAyOyBpIDwgbGV2ZWxbMF07IGkrKykgewotCi0gICAgICAgICAgICAgICAgdG9wID0gYm90dG9tICsgMTsKLSAgICAgICAgICAgICAgICBib3R0b20gPSBsZXZlbFtpXSAtIDE7Ci0KLSAgICAgICAgICAgICAgICBkMS5maW5kQWN0aXZlKHRvcCwgYm90dG9tKTsKLSAgICAgICAgICAgICAgICBpZiAoZDEuYWN0aXZlWzBdID09IDEpIHsKLSAgICAgICAgICAgICAgICAgICAgZDIuZGVsZXRlQWN0aXZlKGJvdHRvbSk7Ci0gICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGQyLmZpbmRBY3RpdmUodG9wLCBib3R0b20pOwotCi0gICAgICAgICAgICAgICAgaW50IGkxID0gMTsKLSAgICAgICAgICAgICAgICBpbnQgaTIgPSAxOwotCi0gICAgICAgICAgICAgICAgaW50IHJ4MSA9IDA7Ci0gICAgICAgICAgICAgICAgaW50IHJ4MiA9IDA7Ci0KLSAgICAgICAgICAgICAgICBib29sZWFuIG5leHQgPSB0cnVlOwotCi0gICAgICAgICAgICAgICAgd2hpbGUodHJ1ZSkgewotCi0gICAgICAgICAgICAgICAgICAgIGlmIChuZXh0KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBuZXh0ID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoaTEgPj0gZDEuYWN0aXZlWzBdKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyBCb3R0b20KLSAgICAgICAgICAgICAgICAgICAgICAgIGQxLmFjdGl2ZVtpMSArIDFdID0gYm90dG9tICsgMTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJ4MSA9IGQxLmFjdGl2ZVtpMV07Ci0gICAgICAgICAgICAgICAgICAgICAgICByeDIgPSBkMS5hY3RpdmVbaTEgKyAyXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGkxICs9IDQ7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBpZiAoaTIgPj0gZDIuYWN0aXZlWzBdKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdENhc2hlZChyeDEsIHRvcCwgcngyLCBib3R0b20pOwotICAgICAgICAgICAgICAgICAgICAgICAgZm9yKGludCBqID0gaTE7IGogPCBkMS5hY3RpdmVbMF07IGogKz0gNCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5hZGRSZWN0Q2FzaGVkKGQxLmFjdGl2ZVtqXSwgdG9wLCBkMS5hY3RpdmVbaiArIDJdLCBib3R0b20pOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGQxLmFjdGl2ZVtqICsgMV0gPSBib3R0b20gKyAxOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBpbnQgeDEgPSBkMi5hY3RpdmVbaTJdOwotICAgICAgICAgICAgICAgICAgICBpbnQgeDIgPSBkMi5hY3RpdmVbaTIgKyAyXTsKLQotICAgICAgICAgICAgICAgICAgICBpZiAocngxIDwgeDEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChyeDIgPj0geDEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAocngyIDw9IHgyKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vICBbLS0tLS0tLS0tLS1dCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vICAgICAgIFstLS0tLS0tLS0tLS0tXQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdENhc2hlZChyeDEsIHRvcCwgeDEgLSAxLCBib3R0b20pOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXh0ID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBbLS0tLS0tLS0tLS0tLS0tLS1dCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vICAgICAgWy0tLS0tLV0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0LmFkZFJlY3RDYXNoZWQocngxLCB0b3AsIHgxIC0gMSwgYm90dG9tKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcngxID0geDIgKyAxOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpMiArPSA0OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gWy0tLS0tXQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vICAgICAgICAgWy0tLS1dCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0LmFkZFJlY3RDYXNoZWQocngxLCB0b3AsIHJ4MiwgYm90dG9tKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXh0ID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChyeDEgPD0geDIpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAocngyIDw9IHgyKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vICAgIFstLS0tLS1dCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vICBbLS0tLS0tLS0tLS1dCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5leHQgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vICAgICBbLS0tLS0tLS0tLS0tXQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBbLS0tLS0tLS0tXQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByeDEgPSB4MiArIDE7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGkyICs9IDQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyAgICAgICAgIFstLS0tXQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIFstLS0tLV0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpMiArPSA0OwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgZDEuZGVsZXRlQWN0aXZlKCk7Ci0gICAgICAgICAgICAgICAgZDIuZGVsZXRlQWN0aXZlKGJvdHRvbSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBzdGF0aWMgdm9pZCBzdWJ0cmFjdFJlY3QoaW50IHgxMSwgaW50IHkxMSwgaW50IHgxMiwgaW50IHkxMiwgaW50W10gcmVjdCwgaW50IGluZGV4LCBNdWx0aVJlY3RBcmVhIGRzdCkgewotCi0gICAgICAgICAgICBmb3IoaW50IGkgPSBpbmRleDsgaSA8IHJlY3RbMF07IGkgKz0gNCkgewotICAgICAgICAgICAgICAgIGludCB4MjEgPSByZWN0W2kgKyAwXTsKLSAgICAgICAgICAgICAgICBpbnQgeTIxID0gcmVjdFtpICsgMV07Ci0gICAgICAgICAgICAgICAgaW50IHgyMiA9IHJlY3RbaSArIDJdOwotICAgICAgICAgICAgICAgIGludCB5MjIgPSByZWN0W2kgKyAzXTsKLQotICAgICAgICAgICAgICAgIGlmICh4MTEgPD0geDIyICYmIHgxMiA+PSB4MjEgJiYgeTExIDw9IHkyMiAmJiB5MTIgPj0geTIxKSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCB0b3AsIGJvdHRvbTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHkxMSA8IHkyMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgc3VidHJhY3RSZWN0KHgxMSwgeTExLCB4MTIsIHkyMSAtIDEsIHJlY3QsIGkgKyA0LCBkc3QpOwotICAgICAgICAgICAgICAgICAgICAgICAgdG9wID0geTIxOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgdG9wID0geTExOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGlmICh5MTIgPiB5MjIpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHN1YnRyYWN0UmVjdCh4MTEsIHkyMiArIDEsIHgxMiwgeTEyLCByZWN0LCBpICsgNCwgZHN0KTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJvdHRvbSA9IHkyMjsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJvdHRvbSA9IHkxMjsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpZiAoeDExIDwgeDIxKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzdWJ0cmFjdFJlY3QoeDExLCB0b3AsIHgyMSAtIDEsIGJvdHRvbSwgcmVjdCwgaSArIDQsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWYgKHgxMiA+IHgyMikgewotICAgICAgICAgICAgICAgICAgICAgICAgc3VidHJhY3RSZWN0KHgyMiArIDEsIHRvcCwgeDEyLCBib3R0b20sIHJlY3QsIGkgKyA0LCBkc3QpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkc3QuYWRkUmVjdCh4MTEsIHkxMSwgeDEyLCB5MTIpOwotICAgICAgICB9Ci0KLSAgICAgICAgc3RhdGljIHZvaWQgc2ltcGxlU3VidHJhY3QoTXVsdGlSZWN0QXJlYSBzcmMxLCBNdWx0aVJlY3RBcmVhIHNyYzIsIE11bHRpUmVjdEFyZWEgZHN0KSB7Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSAxOyBpIDwgc3JjMS5yZWN0WzBdOyBpICs9IDQpIHsKLSAgICAgICAgICAgICAgICBzdWJ0cmFjdFJlY3QoCi0gICAgICAgICAgICAgICAgICAgICAgICBzcmMxLnJlY3RbaSArIDBdLAotICAgICAgICAgICAgICAgICAgICAgICAgc3JjMS5yZWN0W2kgKyAxXSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHNyYzEucmVjdFtpICsgMl0sCi0gICAgICAgICAgICAgICAgICAgICAgICBzcmMxLnJlY3RbaSArIDNdLAotICAgICAgICAgICAgICAgICAgICAgICAgc3JjMi5yZWN0LAotICAgICAgICAgICAgICAgICAgICAgICAgMSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkc3QucmVzb3J0KCk7Ci0gICAgICAgIH0KLQotICAgICAgICBwdWJsaWMgc3RhdGljIE11bHRpUmVjdEFyZWEgZ2V0UmVzdWx0KE11bHRpUmVjdEFyZWEgc3JjMSwgTXVsdGlSZWN0QXJlYSBzcmMyKSB7Ci0KLSAgICAgICAgICAgIGlmIChzcmMxID09IG51bGwgfHwgc3JjMS5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbmV3IE11bHRpUmVjdEFyZWEoKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKHNyYzIgPT0gbnVsbCB8fCBzcmMyLmlzRW1wdHkoKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBuZXcgTXVsdGlSZWN0QXJlYShzcmMxKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgTXVsdGlSZWN0QXJlYS5SZWN0Q2FzaCBkc3QgPSBuZXcgTXVsdGlSZWN0QXJlYS5SZWN0Q2FzaCgpOwotCi0gICAgICAgICAgICBpZiAoIXNyYzEuc29ydGVkIHx8ICFzcmMyLnNvcnRlZCB8fAotICAgICAgICAgICAgICAgc3JjMS5nZXRSZWN0Q291bnQoKSA8PSBNQVhfU0lNUExFIHx8IHNyYzIuZ2V0UmVjdENvdW50KCkgPD0gTUFYX1NJTVBMRSkgCi0gICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgc2ltcGxlU3VidHJhY3Qoc3JjMSwgc3JjMiwgZHN0KTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgUmVjdGFuZ2xlIGJvdW5kczEgPSBzcmMxLmdldEJvdW5kcygpOwotICAgICAgICAgICAgICAgIFJlY3RhbmdsZSBib3VuZHMyID0gc3JjMi5nZXRCb3VuZHMoKTsKLSAgICAgICAgICAgICAgICBSZWN0YW5nbGUgYm91bmRzMyA9IGJvdW5kczEuaW50ZXJzZWN0aW9uKGJvdW5kczIpOwotCi0gICAgICAgICAgICAgICAgaWYgKGJvdW5kczMud2lkdGggPiAwICYmIGJvdW5kczMuaGVpZ2h0ID4gMCkgewotICAgICAgICAgICAgICAgICAgICBzdWJ0cmFjdFJlZ2lvbnMoc3JjMS5yZWN0LCBzcmMyLnJlY3QsIGRzdCwgYm91bmRzMS5oZWlnaHQgKyAyLCBib3VuZHMyLmhlaWdodCArIDIpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGRzdC5zZXRSZWN0KHNyYzEucmVjdCwgdHJ1ZSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZXR1cm4gZHN0OwotICAgICAgICB9Ci0KLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL1N1cmZhY2UuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL1N1cmZhY2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOGIwYWUzOC4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9TdXJmYWNlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzMDkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICogQ3JlYXRlZCBvbiAxMC4xMS4yMDA1Ci0gKgotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2w7Ci0KLWltcG9ydCBqYXZhLmF3dC5JbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5UcmFuc3BhcmVuY3k7Ci1pbXBvcnQgamF2YS5hd3QuY29sb3IuQ29sb3JTcGFjZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29tcG9uZW50Q29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db21wb25lbnRTYW1wbGVNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EYXRhQnVmZmVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRpcmVjdENvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW5kZXhDb2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLk11bHRpUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5TYW1wbGVNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Xcml0YWJsZVJhc3RlcjsKLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlMaXN0OwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvci5MVVRDb2xvckNvbnZlcnRlcjsKLQotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgc3VwZXIgY2xhc3MgZm9yIG90aGVycyB0eXBlcyBvZiBTdXJmYWNlcy4gCi0gKiBTdXJmYWNlIGlzIHN0b3JpbmcgZGF0YSBhbmQgZGF0YSBmb3JtYXQgZGVzY3JpcHRpb24sIHRoYXQgYXJlIHVzaW5nCi0gKiBpbiBibGl0dGluZyBvcGVyYXRpb25zICAgIAotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgU3VyZmFjZSBpbXBsZW1lbnRzIFRyYW5zcGFyZW5jeXsKLQotICAgIC8vIENvbG9yIFNwYWNlIFR5cGVzCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgc1JHQl9DUyA9IDE7Ci0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTGluZWFyX1JHQl9DUyA9IDI7Ci0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgTGluZWFyX0dyYXlfQ1MgPSAzOwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEN1c3RvbV9DUyA9IDA7Ci0gICAgCi0gICAgLy8gQ29sb3IgTW9kZWwgVHlwZXMKLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBEQ00gPSAxOyAgLy8gRGlyZWN0IENvbG9yIE1vZGVsCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSUNNID0gMjsgIC8vIEluZGV4IENvbG9yIE1vZGVsCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ0NNID0gMzsgIC8vIENvbXBvbmVudCBDb2xvciBNb2RlbAotCi0gICAgLy8gU2FtcGxlIE1vZGVsIFR5cGVzCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgU1BQU00gPSAxOyAgLy8gU2luZ2xlIFBpeGVsIFBhY2tlZCBTYW1wbGUgTW9kZWwKLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBNUFBTTSA9IDI7ICAvLyBNdWx0aSBQaXhlbCBQYWNrZWQgU2FtcGxlIE1vZGVsCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgQ1NNICAgPSAzOyAgLy8gQ29tcG9uZW50IFNhbXBsZSBNb2RlbAotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFBJU00gID0gNDsgIC8vIFBpeGVsIEludGVybGVhdmVkIFNhbXBsZSBNb2RlbAotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEJTTSAgID0gNTsgIC8vIEJhbmRlZCBTYW1wbGUgTW9kZWwKLQotICAgIC8vIFN1cmZhY2UgVHlwZXMKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgQUxQSEFfTUFTSyA9IDB4ZmYwMDAwMDA7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFJFRF9NQVNLID0gMHgwMGZmMDAwMDsKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgR1JFRU5fTUFTSyA9IDB4MDAwMGZmMDA7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJMVUVfTUFTSyA9IDB4MDAwMDAwZmY7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFJFRF9CR1JfTUFTSyA9IDB4MDAwMDAwZmY7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEdSRUVOX0JHUl9NQVNLID0gMHgwMDAwZmYwMDsKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgQkxVRV9CR1JfTUFTSyA9IDB4MDBmZjAwMDA7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFJFRF81NjVfTUFTSyA9IDB4ZjgwMDsKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgR1JFRU5fNTY1X01BU0sgPSAweDA3ZTA7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJMVUVfNTY1X01BU0sgPSAweDAwMWY7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFJFRF81NTVfTUFTSyA9IDB4N2MwMDsKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgR1JFRU5fNTU1X01BU0sgPSAweDAzZTA7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJMVUVfNTU1X01BU0sgPSAweDAwMWY7Ci0KLSAgICBzdGF0aWN7Ci0gICAgICAgIC8vPz8/QVdUCi0gICAgICAgIC8qCi0gICAgICAgIFN5c3RlbS5sb2FkTGlicmFyeSgiZ2wiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBpbml0SURzKCk7Ci0gICAgICAgICovCi0gICAgfQotCi0KLSAgICBwcm90ZWN0ZWQgbG9uZyBzdXJmYWNlRGF0YVB0cjsgICAgICAgIC8vIFBvaW50ZXIgZm9yIE5hdGl2ZSBTdXJmYWNlIGRhdGEKLSAgICBwcm90ZWN0ZWQgaW50IHRyYW5zcGFyZW5jeSA9IE9QQVFVRTsKLSAgICBwcm90ZWN0ZWQgaW50IHdpZHRoOwotICAgIHByb3RlY3RlZCBpbnQgaGVpZ2h0OwotCi0gICAgLyoqCi0gICAgICogVGhpcyBsaXN0IGNvbnRhaW5zIGNhY2hlcyB3aXRoIHRoZSBkYXRhIG9mIHRoaXMgc3VyZmFjZSB0aGF0IGFyZSB2YWxpZCBhdCB0aGUgbW9tZW50LgotICAgICAqIFN1cmZhY2Ugc2hvdWxkIGNsZWFyIHRoaXMgbGlzdCB3aGVuIGl0cyBkYXRhIGlzIHVwZGF0ZWQuCi0gICAgICogQ2FjaGVzIG1heSBjaGVjayBpZiB0aGV5IGFyZSBzdGlsbCB2YWxpZCB1c2luZyBpc0NhY2hlVmFsaWQgbWV0aG9kLgotICAgICAqIFdoZW4gY2FjaGUgZ2V0cyBkYXRhIGZyb20gdGhlIHN1cmZhY2UsIGl0IHNob3VsZCBjYWxsIGFkZFZhbGlkQ2FjaGUgbWV0aG9kIG9mIHRoZSBzdXJmYWNlLgotICAgICAqLwotICAgIHByaXZhdGUgZmluYWwgQXJyYXlMaXN0PE9iamVjdD4gdmFsaWRDYWNoZXMgPSBuZXcgQXJyYXlMaXN0PE9iamVjdD4oKTsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoKTsKLSAgICBwdWJsaWMgYWJzdHJhY3QgV3JpdGFibGVSYXN0ZXIgZ2V0UmFzdGVyKCk7Ci0gICAgcHVibGljIGFic3RyYWN0IGludCBnZXRTdXJmYWNlVHlwZSgpOyAvLyBTeXJmYWNlIHR5cGUuIEl0IGlzIGVxdWFsIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gQnVmZmVyZWRJbWdlIHR5cGUKLSAgICAvKioKLSAgICAgKiBMb2NrIE5hdGl2ZSBTdXJmYWNlIGRhdGEKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgbG9uZyBsb2NrKCk7ICAgICAKLSAgICAKLSAgICAvKioKLSAgICAgKiBVbmxvY2sgTmF0aXZlIFN1cmZhY2UgZGF0YSAKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCB1bmxvY2soKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBEaXNwb3NlIE5hdGl2ZSBTdXJmYWNlIGRhdGEKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBkaXNwb3NlKCk7Ci0gICAgcHVibGljIGFic3RyYWN0IFN1cmZhY2UgZ2V0SW1hZ2VTdXJmYWNlKCk7Ci0KLSAgICBwdWJsaWMgbG9uZyBnZXRTdXJmYWNlRGF0YVB0cigpewotICAgICAgICByZXR1cm4gc3VyZmFjZURhdGFQdHI7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIGJvb2xlYW4gaXNDYWhlVmFsaWQoT2JqZWN0IGNhY2hlKSB7Ci0gICAgICAgIHJldHVybiB2YWxpZENhY2hlcy5jb250YWlucyhjYWNoZSk7Ci0gICAgfQotCi0gICAgcHVibGljIGZpbmFsIHZvaWQgYWRkVmFsaWRDYWNoZShPYmplY3QgY2FjaGUpIHsKLSAgICAgICAgdmFsaWRDYWNoZXMuYWRkKGNhY2hlKTsKLSAgICB9Ci0KLSAgICBwcm90ZWN0ZWQgZmluYWwgdm9pZCBjbGVhclZhbGlkQ2FjaGVzKCkgewotICAgICAgICB2YWxpZENhY2hlcy5jbGVhcigpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgY291bGQgb3IgY29sZG4ndCB0aGUgU3VyZmFjZSBiZSBibGl0IGJ5IE5hdGl2ZSBibGl0dGVyIAotICAgICAqIEByZXR1cm4gLSB0cnVlIGlmIHRoZSBTdXJmYWNlIGNvdWxkIGJlIGJsaXQgYnkgTmF0aXZlIGJsaXR0ZXIsIAotICAgICAqICAgICAgICAgICBmYWxzZSBpbiBvdGhlciBjYXNlCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNOYXRpdmVEcmF3YWJsZSgpewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFRyYW5zcGFyZW5jeSgpIHsKLSAgICAgICAgcmV0dXJuIHRyYW5zcGFyZW5jeTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldFdpZHRoKCl7Ci0gICAgICAgIHJldHVybiB3aWR0aDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldEhlaWdodCgpewotICAgICAgICByZXR1cm4gaGVpZ2h0OwotICAgIH0KLSAgICAKLSAgICAvKioKLSAgICAgKiBJZiBTdXJmYWNlIGhhcyBSYXN0ZXIsIHRoaXMgbWV0aG9kIHJldHVybnMgZGF0YSBhcnJheSBvZiBSYXN0ZXIncyBEYXRhQnVmZmVyCi0gICAgICogQHJldHVybiAtIGRhdGEgYXJyYXkKLSAgICAgKi8KLSAgICBwdWJsaWMgT2JqZWN0IGdldERhdGEoKXsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotICAgIAotICAgIHB1YmxpYyBib29sZWFuIGludmFsaWRhdGVkKCl7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLSAgICAKLSAgICBwdWJsaWMgdm9pZCB2YWxpZGF0ZSgpe30KLSAgICAKLSAgICBwdWJsaWMgdm9pZCBpbnZhbGlkYXRlKCl7fQotCi0gICAgLyoqCi0gICAgICogQ29tcHV0YXRpb24gdHlwZSBvZiBCdWZmZXJlZEltYWdlIG9yIFN1cmZhY2UKLSAgICAgKiBAcGFyYW0gY20gLSBDb2xvck1vZGVsCi0gICAgICogQHBhcmFtIHJhc3RlciAtIFdyaXRhYmxlUmFzdGUKLSAgICAgKiBAcmV0dXJuIC0gdHlwZSBvZiBCdWZmZXJlZEltYWdlCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnQgZ2V0VHlwZShDb2xvck1vZGVsIGNtLCBXcml0YWJsZVJhc3RlciByYXN0ZXIpewotICAgICAgICBpbnQgdHJhbnNmZXJUeXBlID0gY20uZ2V0VHJhbnNmZXJUeXBlKCk7Ci0gICAgICAgIGJvb2xlYW4gaGFzQWxwaGEgPSBjbS5oYXNBbHBoYSgpOwotICAgICAgICBDb2xvclNwYWNlIGNzID0gY20uZ2V0Q29sb3JTcGFjZSgpOwotICAgICAgICBpbnQgY3NUeXBlID0gY3MuZ2V0VHlwZSgpOwotICAgICAgICBTYW1wbGVNb2RlbCBzbSA9IHJhc3Rlci5nZXRTYW1wbGVNb2RlbCgpOwotCi0gICAgICAgIGlmKGNzVHlwZSA9PSBDb2xvclNwYWNlLlRZUEVfUkdCKXsKLSAgICAgICAgICAgIGlmKGNtIGluc3RhbmNlb2YgRGlyZWN0Q29sb3JNb2RlbCl7Ci0gICAgICAgICAgICAgICAgRGlyZWN0Q29sb3JNb2RlbCBkY20gPSAoRGlyZWN0Q29sb3JNb2RlbCkgY207Ci0gICAgICAgICAgICAgICAgc3dpdGNoICh0cmFuc2ZlclR5cGUpIHsKLSAgICAgICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgICAgIGlmIChkY20uZ2V0UmVkTWFzaygpID09IFJFRF9NQVNLICYmCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZGNtLmdldEdyZWVuTWFzaygpID09IEdSRUVOX01BU0sgJiYKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkY20uZ2V0Qmx1ZU1hc2soKSA9PSBCTFVFX01BU0spIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmICghaGFzQWxwaGEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9SR0I7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoZGNtLmdldEFscGhhTWFzaygpID09IEFMUEhBX01BU0spIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoZGNtLmlzQWxwaGFQcmVtdWx0aXBsaWVkKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQl9QUkU7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0I7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gQnVmZmVyZWRJbWFnZS5UWVBFX0NVU1RPTTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIGlmIChkY20uZ2V0UmVkTWFzaygpID09IFJFRF9CR1JfTUFTSyAmJgotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRjbS5nZXRHcmVlbk1hc2soKSA9PSBHUkVFTl9CR1JfTUFTSyAmJgotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRjbS5nZXRCbHVlTWFzaygpID09IEJMVUVfQkdSX01BU0spIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmICghaGFzQWxwaGEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9CR1I7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gQnVmZmVyZWRJbWFnZS5UWVBFX0NVU1RPTTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVDoKLSAgICAgICAgICAgICAgICAgICAgaWYgKGRjbS5nZXRSZWRNYXNrKCkgPT0gUkVEXzU1NV9NQVNLICYmCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZGNtLmdldEdyZWVuTWFzaygpID09IEdSRUVOXzU1NV9NQVNLICYmCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZGNtLmdldEJsdWVNYXNrKCkgPT0gQkxVRV81NTVfTUFTSyAmJiAhaGFzQWxwaGEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBCdWZmZXJlZEltYWdlLlRZUEVfVVNIT1JUXzU1NV9SR0I7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoZGNtLmdldFJlZE1hc2soKSA9PSBSRURfNTY1X01BU0sgJiYKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkY20uZ2V0R3JlZW5NYXNrKCkgPT0gR1JFRU5fNTY1X01BU0sgJiYKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkY20uZ2V0Qmx1ZU1hc2soKSA9PSBCTFVFXzU2NV9NQVNLKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gQnVmZmVyZWRJbWFnZS5UWVBFX1VTSE9SVF81NjVfUkdCOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9DVVNUT007Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfWVsc2UgaWYoY20gaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwpewotICAgICAgICAgICAgICAgIEluZGV4Q29sb3JNb2RlbCBpY20gPSAoSW5kZXhDb2xvck1vZGVsKSBjbTsKLSAgICAgICAgICAgICAgICBpbnQgcGl4ZWxCaXRzID0gaWNtLmdldFBpeGVsU2l6ZSgpOwotICAgICAgICAgICAgICAgIGlmKHRyYW5zZmVyVHlwZSA9PSBEYXRhQnVmZmVyLlRZUEVfQllURSl7Ci0gICAgICAgICAgICAgICAgICAgIGlmKHNtIGluc3RhbmNlb2YgTXVsdGlQaXhlbFBhY2tlZFNhbXBsZU1vZGVsICYmICFoYXNBbHBoYSAmJgotICAgICAgICAgICAgICAgICAgICAgICAgcGl4ZWxCaXRzIDwgNSl7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0JJTkFSWTsKLSAgICAgICAgICAgICAgICAgICAgfWVsc2UgaWYocGl4ZWxCaXRzID09IDgpewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0lOREVYRUQ7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9DVVNUT007Ci0gICAgICAgICAgICB9ZWxzZSBpZihjbSBpbnN0YW5jZW9mIENvbXBvbmVudENvbG9yTW9kZWwpewotICAgICAgICAgICAgICAgIENvbXBvbmVudENvbG9yTW9kZWwgY2NtID0gKENvbXBvbmVudENvbG9yTW9kZWwpIGNtOwotICAgICAgICAgICAgICAgIGlmKHRyYW5zZmVyVHlwZSA9PSBEYXRhQnVmZmVyLlRZUEVfQllURSAmJgotICAgICAgICAgICAgICAgICAgICAgICAgc20gaW5zdGFuY2VvZiBDb21wb25lbnRTYW1wbGVNb2RlbCl7Ci0gICAgICAgICAgICAgICAgICAgIENvbXBvbmVudFNhbXBsZU1vZGVsIGNzbSA9Ci0gICAgICAgICAgICAgICAgICAgICAgICAoQ29tcG9uZW50U2FtcGxlTW9kZWwpIHNtOwotICAgICAgICAgICAgICAgICAgICBpbnRbXSBvZmZzZXRzID0gY3NtLmdldEJhbmRPZmZzZXRzKCk7Ci0gICAgICAgICAgICAgICAgICAgIGludFtdIGJpdHMgPSBjY20uZ2V0Q29tcG9uZW50U2l6ZSgpOwotICAgICAgICAgICAgICAgICAgICBib29sZWFuIGlzQ3VzdG9tID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgYml0cy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGJpdHNbaV0gIT0gOCB8fAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9mZnNldHNbaV0gIT0gb2Zmc2V0cy5sZW5ndGggLSAxIC0gaSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlzQ3VzdG9tID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpZiAoIWlzQ3VzdG9tKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoIWNjbS5oYXNBbHBoYSgpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV8zQllURV9CR1I7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGNjbS5pc0FscGhhUHJlbXVsdGlwbGllZCgpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV80QllURV9BQkdSX1BSRTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV80QllURV9BQkdSOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBCdWZmZXJlZEltYWdlLlRZUEVfQ1VTVE9NOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9DVVNUT007Ci0gICAgICAgIH1lbHNlIGlmKGNzID09IExVVENvbG9yQ29udmVydGVyLkxJTkVBUl9HUkFZX0NTKXsKLSAgICAgICAgICAgIGlmKGNtIGluc3RhbmNlb2YgQ29tcG9uZW50Q29sb3JNb2RlbCAmJgotICAgICAgICAgICAgICAgICAgICBjbS5nZXROdW1Db21wb25lbnRzKCkgPT0gMSl7Ci0gICAgICAgICAgICAgICAgaW50IGJpdHNbXSA9IGNtLmdldENvbXBvbmVudFNpemUoKTsKLSAgICAgICAgICAgICAgICBpZih0cmFuc2ZlclR5cGUgPT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYKLSAgICAgICAgICAgICAgICAgICAgICAgIGJpdHNbMF0gPT0gOCl7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBCdWZmZXJlZEltYWdlLlRZUEVfQllURV9HUkFZOwotICAgICAgICAgICAgICAgIH1lbHNlIGlmKHRyYW5zZmVyVHlwZSA9PSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUICYmCi0gICAgICAgICAgICAgICAgICAgICAgICBiaXRzWzBdID09IDE2KXsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9VU0hPUlRfR1JBWTsKLSAgICAgICAgICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9DVVNUT007Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9DVVNUT007Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9DVVNUT007Ci0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyBTdXJmYWNlIGdldEltYWdlU3VyZmFjZShJbWFnZSBpbWFnZSl7Ci0gICAgICAgIHJldHVybiBBd3RJbWFnZUJhY2tkb29yQWNjZXNzb3IuZ2V0SW5zdGFuY2UoKS5nZXRJbWFnZVN1cmZhY2UoaW1hZ2UpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHByb3RlY3RlZCB2b2lkIGZpbmFsaXplKCkgdGhyb3dzIFRocm93YWJsZXsKLSAgICAgICAgZGlzcG9zZSgpOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgYm9vbGVhbiBpc0dyYXlQYWxsZXRlKEluZGV4Q29sb3JNb2RlbCBpY20pewotICAgICAgICByZXR1cm4gQXd0SW1hZ2VCYWNrZG9vckFjY2Vzc29yLmdldEluc3RhbmNlKCkuaXNHcmF5UGFsbGV0ZShpY20pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEluaXRpYWxpemF0aW9uIG9mIE5hdGl2ZSBkYXRhCi0gICAgICogCi0gICAgICovCi0gICAgLy8/Pz9BV1Q6IHByaXZhdGUgc3RhdGljIG5hdGl2ZSB2b2lkIGluaXRJRHMoKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL1RleHRSZW5kZXJlci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvVGV4dFJlbmRlcmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGY1Nzk1MmQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvVGV4dFJlbmRlcmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbDsKLQotaW1wb3J0IGphdmEuYXd0LkdyYXBoaWNzMkQ7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5HbHlwaFZlY3RvcjsKLQotcHVibGljIGFic3RyYWN0IGNsYXNzIFRleHRSZW5kZXJlciB7Ci0gICAgCi0gICAgLyoqCi0gICAgICogRHJhd3Mgc3RyaW5nIG9uIHNwZWNpZmllZCBHcmFwaGljcyBhdCBkZXNpcmVkIHBvc2l0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnIHNwZWNpZmllZCBHcmFwaGljczJEIG9iamVjdAotICAgICAqIEBwYXJhbSBzdHIgU3RyaW5nIG9iamVjdCB0byBkcmF3Ci0gICAgICogQHBhcmFtIHggc3RhcnQgWCBwb3NpdGlvbiB0byBkcmF3Ci0gICAgICogQHBhcmFtIHkgc3RhcnQgWSBwb3NpdGlvbiB0byBkcmF3Ci0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZHJhd1N0cmluZyhHcmFwaGljczJEIGcsIFN0cmluZyBzdHIsIGZsb2F0IHgsIGZsb2F0IHkpOwotCi0gICAgLyoqCi0gICAgICogRHJhd3Mgc3RyaW5nIG9uIHNwZWNpZmllZCBHcmFwaGljcyBhdCBkZXNpcmVkIHBvc2l0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnIHNwZWNpZmllZCBHcmFwaGljczJEIG9iamVjdAotICAgICAqIEBwYXJhbSBzdHIgU3RyaW5nIG9iamVjdCB0byBkcmF3Ci0gICAgICogQHBhcmFtIHggc3RhcnQgWCBwb3NpdGlvbiB0byBkcmF3Ci0gICAgICogQHBhcmFtIHkgc3RhcnQgWSBwb3NpdGlvbiB0byBkcmF3Ci0gICAgICovICAgIAotICAgIHB1YmxpYyB2b2lkIGRyYXdTdHJpbmcoR3JhcGhpY3MyRCBnLCBTdHJpbmcgc3RyLCBpbnQgeCwgaW50IHkpewotICAgICAgICBkcmF3U3RyaW5nKGcsIHN0ciwgKGZsb2F0KXgsIChmbG9hdCl5KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEcmF3cyBHbHlwaFZlY3RvciBvbiBzcGVjaWZpZWQgR3JhcGhpY3MgYXQgZGVzaXJlZCBwb3NpdGlvbi4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZyBzcGVjaWZpZWQgR3JhcGhpY3MyRCBvYmplY3QKLSAgICAgKiBAcGFyYW0gZ2x5cGhWZWN0b3IgR2x5cGhWZWN0b3Igb2JqZWN0IHRvIGRyYXcKLSAgICAgKiBAcGFyYW0geCBzdGFydCBYIHBvc2l0aW9uIHRvIGRyYXcKLSAgICAgKiBAcGFyYW0geSBzdGFydCBZIHBvc2l0aW9uIHRvIGRyYXcKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBkcmF3R2x5cGhWZWN0b3IoR3JhcGhpY3MyRCBnLCBHbHlwaFZlY3RvciBnbHlwaFZlY3RvciwgZmxvYXQgeCwgZmxvYXQgeSk7Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9YT1JDb21wb3NpdGUuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL1hPUkNvbXBvc2l0ZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlMjdlMWQzLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL1hPUkNvbXBvc2l0ZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDggKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICogQ3JlYXRlZCBvbiAyMS4xMS4yMDA1Ci0gKgotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2w7Ci0KLWltcG9ydCBqYXZhLmF3dC5Db2xvcjsKLWltcG9ydCBqYXZhLmF3dC5Db21wb3NpdGU7Ci1pbXBvcnQgamF2YS5hd3QuQ29tcG9zaXRlQ29udGV4dDsKLWltcG9ydCBqYXZhLmF3dC5SZW5kZXJpbmdIaW50czsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsOwotCi1wdWJsaWMgY2xhc3MgWE9SQ29tcG9zaXRlIGltcGxlbWVudHMgQ29tcG9zaXRlIHsKLQotICAgIENvbG9yIHhvcmNvbG9yOwotCi0gICAgcHVibGljIFhPUkNvbXBvc2l0ZShDb2xvciB4b3Jjb2xvcil7Ci0gICAgICAgIHRoaXMueG9yY29sb3IgPSB4b3Jjb2xvcjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQ29tcG9zaXRlQ29udGV4dCBjcmVhdGVDb250ZXh0KENvbG9yTW9kZWwgc3JjQ00sIENvbG9yTW9kZWwgZHN0Q00sCi0gICAgICAgICAgICBSZW5kZXJpbmdIaW50cyBoaW50cykgewotCi0gICAgICAgIHJldHVybiBuZXcgSUNvbXBvc2l0ZUNvbnRleHQodGhpcywgc3JjQ00sIGRzdENNKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQ29sb3IgZ2V0WE9SQ29sb3IoKXsKLSAgICAgICAgcmV0dXJuIHhvcmNvbG9yOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2NvbG9yL0NvbG9yQ29udmVydGVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9jb2xvci9Db2xvckNvbnZlcnRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjOThlMTE0Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2NvbG9yL0NvbG9yQ29udmVydGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNTcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvcjsKLQotaW1wb3J0IGphdmEuYXd0LmNvbG9yLkNvbG9yU3BhY2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJhc3RlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Xcml0YWJsZVJhc3RlcjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGNvbWJpbmVzIENvbG9yU2NhbGVyLCBJQ0NfVHJhbnNmb3JtIGFuZCBOYXRpdmVJbWFnZUZvcm1hdCBmdW5jdGlvbmFsaXR5Ci0gKiBpbiB0aGUgd29ya2Zsb3dzIGZvciBkaWZmZXJlbnQgdHlwZXMgb2YgaW5wdXQvb3V0cHV0IHBpeGVsIGRhdGEuCi0gKi8KLXB1YmxpYyBjbGFzcyBDb2xvckNvbnZlcnRlciB7Ci0gICAgcHJpdmF0ZSBDb2xvclNjYWxlciBzY2FsZXIgPSBuZXcgQ29sb3JTY2FsZXIoKTsKLQotICAgIHB1YmxpYyB2b2lkIGxvYWRTY2FsaW5nRGF0YShDb2xvclNwYWNlIGNzKSB7Ci0gICAgICAgIHNjYWxlci5sb2FkU2NhbGluZ0RhdGEoY3MpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRyYW5zbGF0ZXMgcGl4ZWxzLCBzdG9yZWQgaW4gc291cmNlIGJ1ZmZlcmVkIGltYWdlIGFuZCB3cml0ZXMgdGhlIGRhdGEKLSAgICAgKiB0byB0aGUgZGVzdGluYXRpb24gaW1hZ2UuCi0gICAgICogQHBhcmFtIHQgLSBJQ0MgdHJhbnNmb3JtCi0gICAgICogQHBhcmFtIHNyYyAtIHNvdXJjZSBpbWFnZQotICAgICAqIEBwYXJhbSBkc3QgLSBkZXN0aW5hdGlvbiBpbWFnZQotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHRyYW5zbGF0ZUNvbG9yKElDQ19UcmFuc2Zvcm0gdCwKLSAgICAgICAgICAgIEJ1ZmZlcmVkSW1hZ2Ugc3JjLCBCdWZmZXJlZEltYWdlIGRzdCkgewotICAgICAgTmF0aXZlSW1hZ2VGb3JtYXQgc3JjSUYgPSBOYXRpdmVJbWFnZUZvcm1hdC5jcmVhdGVOYXRpdmVJbWFnZUZvcm1hdChzcmMpOwotICAgICAgTmF0aXZlSW1hZ2VGb3JtYXQgZHN0SUYgPSBOYXRpdmVJbWFnZUZvcm1hdC5jcmVhdGVOYXRpdmVJbWFnZUZvcm1hdChkc3QpOwotCi0gICAgICBpZiAoc3JjSUYgIT0gbnVsbCAmJiBkc3RJRiAhPSBudWxsKSB7Ci0gICAgICAgICAgdC50cmFuc2xhdGVDb2xvcnMoc3JjSUYsIGRzdElGKTsKLSAgICAgICAgICByZXR1cm47Ci0gICAgICB9Ci0KLSAgICAgICAgc3JjSUYgPSBjcmVhdGVJbWFnZUZvcm1hdChzcmMpOwotICAgICAgICBkc3RJRiA9IGNyZWF0ZUltYWdlRm9ybWF0KGRzdCk7Ci0KLSAgICAgICAgc2hvcnQgc3JjQ2hhbkRhdGFbXSA9IChzaG9ydFtdKSBzcmNJRi5nZXRDaGFubmVsRGF0YSgpOwotICAgICAgICBzaG9ydCBkc3RDaGFuRGF0YVtdID0gKHNob3J0W10pIGRzdElGLmdldENoYW5uZWxEYXRhKCk7Ci0KLSAgICAgICAgQ29sb3JNb2RlbCBzcmNDTSA9IHNyYy5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIGludCBuQ29sb3JDaGFubmVscyA9IHNyY0NNLmdldE51bUNvbG9yQ29tcG9uZW50cygpOwotICAgICAgICBzY2FsZXIubG9hZFNjYWxpbmdEYXRhKHNyY0NNLmdldENvbG9yU3BhY2UoKSk7IC8vIGlucHV0IHNjYWxpbmcgZGF0YQotICAgICAgICBDb2xvck1vZGVsIGRzdENNID0gZHN0LmdldENvbG9yTW9kZWwoKTsKLQotICAgICAgICAvLyBQcmVwYXJlIGFycmF5IGZvciBhbHBoYSBjaGFubmVsCi0gICAgICAgIGZsb2F0IGFscGhhW10gPSBudWxsOwotICAgICAgICBib29sZWFuIHNhdmVBbHBoYSA9IHNyY0NNLmhhc0FscGhhKCkgJiYgZHN0Q00uaGFzQWxwaGEoKTsKLSAgICAgICAgaWYgKHNhdmVBbHBoYSkgewotICAgICAgICAgICAgYWxwaGEgPSBuZXcgZmxvYXRbc3JjLmdldFdpZHRoKCkqc3JjLmdldEhlaWdodCgpXTsKLSAgICAgICAgfQotCi0gICAgICAgIFdyaXRhYmxlUmFzdGVyIHdyID0gc3JjLmdldFJhc3RlcigpOwotICAgICAgICBpbnQgc3JjRGF0YVBvcyA9IDAsIGFscGhhUG9zID0gMDsKLSAgICAgICAgZmxvYXQgbm9ybWFsaXplZFZhbFtdOwotICAgICAgICBmb3IgKGludCByb3c9MCwgblJvd3MgPSBzcmNJRi5nZXROdW1Sb3dzKCk7IHJvdzxuUm93czsgcm93KyspIHsKLSAgICAgICAgICAgIGZvciAoaW50IGNvbD0wLCBuQ29scyA9IHNyY0lGLmdldE51bUNvbHMoKTsgY29sPG5Db2xzOyBjb2wrKykgewotICAgICAgICAgICAgICAgIG5vcm1hbGl6ZWRWYWwgPSBzcmNDTS5nZXROb3JtYWxpemVkQ29tcG9uZW50cygKLSAgICAgICAgICAgICAgICAgICAgd3IuZ2V0RGF0YUVsZW1lbnRzKGNvbCwgcm93LCBudWxsKSwKLSAgICAgICAgICAgICAgICAgICAgbnVsbCwgMCk7Ci0gICAgICAgICAgICAgICAgLy8gU2F2ZSBhbHBoYSBjaGFubmVsCi0gICAgICAgICAgICAgICAgaWYgKHNhdmVBbHBoYSkgewotICAgICAgICAgICAgICAgICAgICAvLyBXZSBuZWVkIG5Db2xvckNoYW5uZWxzJ3RoIGVsZW1lbnQgY2F1c2UgaXQncyBuQ2hhbm5lbHMgLSAxCi0gICAgICAgICAgICAgICAgICAgIGFscGhhW2FscGhhUG9zKytdID0gbm9ybWFsaXplZFZhbFtuQ29sb3JDaGFubmVsc107Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHNjYWxlci5zY2FsZShub3JtYWxpemVkVmFsLCBzcmNDaGFuRGF0YSwgc3JjRGF0YVBvcyk7Ci0gICAgICAgICAgICAgICAgc3JjRGF0YVBvcyArPSBuQ29sb3JDaGFubmVsczsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHQudHJhbnNsYXRlQ29sb3JzKHNyY0lGLCBkc3RJRik7Ci0KLSAgICAgICAgbkNvbG9yQ2hhbm5lbHMgPSBkc3RDTS5nZXROdW1Db2xvckNvbXBvbmVudHMoKTsKLSAgICAgICAgYm9vbGVhbiBmaWxsQWxwaGEgPSBkc3RDTS5oYXNBbHBoYSgpOwotICAgICAgICBzY2FsZXIubG9hZFNjYWxpbmdEYXRhKGRzdENNLmdldENvbG9yU3BhY2UoKSk7IC8vIG91dHB1dCBzY2FsaW5nIGRhdGEKLSAgICAgICAgZmxvYXQgZHN0UGl4ZWxbXSA9IG5ldyBmbG9hdFtkc3RDTS5nZXROdW1Db21wb25lbnRzKCldOwotICAgICAgICBpbnQgZHN0RGF0YVBvcyA9IDA7Ci0gICAgICAgIGFscGhhUG9zID0gMDsKLSAgICAgICAgd3IgPSBkc3QuZ2V0UmFzdGVyKCk7Ci0KLSAgICAgICAgZm9yIChpbnQgcm93PTAsIG5Sb3dzID0gZHN0SUYuZ2V0TnVtUm93cygpOyByb3c8blJvd3M7IHJvdysrKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBjb2w9MCwgbkNvbHMgPSBkc3RJRi5nZXROdW1Db2xzKCk7IGNvbDxuQ29sczsgY29sKyspIHsKLSAgICAgICAgICAgICAgICBzY2FsZXIudW5zY2FsZShkc3RQaXhlbCwgZHN0Q2hhbkRhdGEsIGRzdERhdGFQb3MpOwotICAgICAgICAgICAgICAgIGRzdERhdGFQb3MgKz0gbkNvbG9yQ2hhbm5lbHM7Ci0gICAgICAgICAgICAgICAgaWYgKGZpbGxBbHBoYSkgewotICAgICAgICAgICAgICAgICAgICBpZiAoc2F2ZUFscGhhKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkc3RQaXhlbFtuQ29sb3JDaGFubmVsc10gPSBhbHBoYVthbHBoYVBvcysrXTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdFBpeGVsW25Db2xvckNoYW5uZWxzXSA9IDFmOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHdyLnNldERhdGFFbGVtZW50cyhjb2wsIHJvdywKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdENNLmdldERhdGFFbGVtZW50cyhkc3RQaXhlbCwgMCAsIG51bGwpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRyYW5zbGF0ZXMgcGl4ZWxzLCBzdG9yZWQgaW4gdGhlIGZsb2F0IGRhdGEgYnVmZmVyLgotICAgICAqIEVhY2ggcGl4ZWwgb2NjdXBpZXMgc2VwYXJhdGUgYXJyYXkuIElucHV0IHBpeGVscyBwYXNzZWQgaW4gdGhlIGJ1ZmZlcgotICAgICAqIGFyZSByZXBsYWNlZCBieSBvdXRwdXQgcGl4ZWxzIGFuZCB0aGVuIHRoZSBidWZmZXIgaXMgcmV0dXJuZWQKLSAgICAgKiBAcGFyYW0gdCAtIElDQyB0cmFuc2Zvcm0KLSAgICAgKiBAcGFyYW0gYnVmZmVyIC0gZGF0YSBidWZmZXIKLSAgICAgKiBAcGFyYW0gc3JjQ1MgLSBzb3VyY2UgY29sb3Igc3BhY2UKLSAgICAgKiBAcGFyYW0gZHN0Q1MgLSBkZXN0aW5hdGlvbiBjb2xvciBzcGFjZQotICAgICAqIEBwYXJhbSBuUGl4ZWxzIC0gbnVtYmVyIG9mIHBpeGVscwotICAgICAqIEByZXR1cm4gdHJhbnNsYXRlZCBwaXhlbHMKLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXRbXVtdIHRyYW5zbGF0ZUNvbG9yKElDQ19UcmFuc2Zvcm0gdCwKLSAgICAgICAgICAgIGZsb2F0IGJ1ZmZlcltdW10sCi0gICAgICAgICAgICBDb2xvclNwYWNlIHNyY0NTLAotICAgICAgICAgICAgQ29sb3JTcGFjZSBkc3RDUywKLSAgICAgICAgICAgIGludCBuUGl4ZWxzKSB7Ci0gICAgICAgIC8vIFNjYWxlIHNvdXJjZSBkYXRhCi0gICAgICAgIGlmIChzcmNDUyAhPSBudWxsKSB7IC8vIGlmIGl0IGlzIG51bGwgdXNlIG9sZCBzY2FsaW5nIGRhdGEKLSAgICAgICAgICAgIHNjYWxlci5sb2FkU2NhbGluZ0RhdGEoc3JjQ1MpOwotICAgICAgICB9Ci0gICAgICAgIGludCBuU3JjQ2hhbm5lbHMgPSB0LmdldE51bUlucHV0Q2hhbm5lbHMoKTsKLSAgICAgICAgc2hvcnQgc3JjU2hvcnREYXRhW10gPSBuZXcgc2hvcnRbblBpeGVscypuU3JjQ2hhbm5lbHNdOwotICAgICAgICBmb3IgKGludCBpPTAsIHNyY0RhdGFQb3MgPSAwOyBpPG5QaXhlbHM7IGkrKykgewotICAgICAgICAgICAgc2NhbGVyLnNjYWxlKGJ1ZmZlcltpXSwgc3JjU2hvcnREYXRhLCBzcmNEYXRhUG9zKTsKLSAgICAgICAgICAgIHNyY0RhdGFQb3MgKz0gblNyY0NoYW5uZWxzOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gQXBwbHkgdHJhbnNmb3JtCi0gICAgICAgIHNob3J0IGRzdFNob3J0RGF0YVtdID0gdGhpcy50cmFuc2xhdGVDb2xvcih0LCBzcmNTaG9ydERhdGEsIG51bGwpOwotCi0gICAgICAgIGludCBuRHN0Q2hhbm5lbHMgPSB0LmdldE51bU91dHB1dENoYW5uZWxzKCk7Ci0gICAgICAgIGludCBidWZmZXJTaXplID0gYnVmZmVyWzBdLmxlbmd0aDsKLSAgICAgICAgaWYgKGJ1ZmZlclNpemUgPCBuRHN0Q2hhbm5lbHMgKyAxKSB7IC8vIFJlLWFsbG9jYXRlIGJ1ZmZlciBpZiBuZWVkZWQKLSAgICAgICAgICAgIGZvciAoaW50IGk9MDsgaTxuUGl4ZWxzOyBpKyspIHsKLSAgICAgICAgICAgICAgICAvLyBPbmUgZXh0cmEgZWxlbWVudCByZXNlcnZlZCBmb3IgYWxwaGEKLSAgICAgICAgICAgICAgICBidWZmZXJbaV0gPSBuZXcgZmxvYXRbbkRzdENoYW5uZWxzICsgMV07Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBVbnNjYWxlIGRlc3RpbmF0aW9uIGRhdGEKLSAgICAgICAgaWYgKGRzdENTICE9IG51bGwpIHsgLy8gaWYgaXQgaXMgbnVsbCB1c2Ugb2xkIHNjYWxpbmcgZGF0YQotICAgICAgICAgICAgc2NhbGVyLmxvYWRTY2FsaW5nRGF0YShkc3RDUyk7Ci0gICAgICAgIH0KLSAgICAgICAgZm9yIChpbnQgaT0wLCBkc3REYXRhUG9zID0gMDsgaTxuUGl4ZWxzOyBpKyspIHsKLSAgICAgICAgICAgIHNjYWxlci51bnNjYWxlKGJ1ZmZlcltpXSwgZHN0U2hvcnREYXRhLCBkc3REYXRhUG9zKTsKLSAgICAgICAgICAgIGRzdERhdGFQb3MgKz0gbkRzdENoYW5uZWxzOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGJ1ZmZlcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUcmFuc2xhdGVzIHBpeGVscyBzdG9yZWQgaW4gYSByYXN0ZXIuCi0gICAgICogQWxsIGRhdGEgdHlwZXMgYXJlIHN1cHBvcnRlZAotICAgICAqIEBwYXJhbSB0IC0gSUNDIHRyYW5zZm9ybQotICAgICAqIEBwYXJhbSBzcmMgLSBzb3VyY2UgcGl4ZWxzCi0gICAgICogQHBhcmFtIGRzdCAtIGRlc3RpbmF0aW9uIHBpeGVscwotICAgICAqLwotICAgcHVibGljIHZvaWQgdHJhbnNsYXRlQ29sb3IoSUNDX1RyYW5zZm9ybSB0LCBSYXN0ZXIgc3JjLCBXcml0YWJsZVJhc3RlciBkc3QpIHsKLSAgICAgICAgdHJ5ewotICAgICAgICAgICAgTmF0aXZlSW1hZ2VGb3JtYXQgc3JjRm10ID0gTmF0aXZlSW1hZ2VGb3JtYXQuY3JlYXRlTmF0aXZlSW1hZ2VGb3JtYXQoc3JjKTsKLSAgICAgICAgICAgIE5hdGl2ZUltYWdlRm9ybWF0IGRzdEZtdCA9IE5hdGl2ZUltYWdlRm9ybWF0LmNyZWF0ZU5hdGl2ZUltYWdlRm9ybWF0KGRzdCk7Ci0KLSAgICAgICAgICBpZiAoc3JjRm10ICE9IG51bGwgJiYgZHN0Rm10ICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgdC50cmFuc2xhdGVDb2xvcnMoc3JjRm10LCBkc3RGbXQpOwotICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgfQotICAgICAgICB9IGNhdGNoIChJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24gZSkgewotICAgICAgfQotCi0gICAgICAgIC8vIEdvIGFoZWFkIGFuZCByZXNjYWxlIHRoZSBzb3VyY2UgaW1hZ2UKLSAgICAgICAgc2NhbGVyLmxvYWRTY2FsaW5nRGF0YShzcmMsIHQuZ2V0U3JjKCkpOwotICAgICAgICBzaG9ydCBzcmNEYXRhW10gPSBzY2FsZXIuc2NhbGUoc3JjKTsKLQotICAgICAgICBzaG9ydCBkc3REYXRhW10gPSB0cmFuc2xhdGVDb2xvcih0LCBzcmNEYXRhLCBudWxsKTsKLQotICAgICAgICBzY2FsZXIubG9hZFNjYWxpbmdEYXRhKGRzdCwgdC5nZXREc3QoKSk7Ci0gICAgICAgIHNjYWxlci51bnNjYWxlKGRzdERhdGEsIGRzdCk7Ci0gICB9Ci0KLSAgICAvKioKLSAgICAgKiBUcmFuc2xhdGVzIHBpeGVscyBzdG9yZWQgaW4gYW4gYXJyYXkgb2Ygc2hvcnRzLgotICAgICAqIFNhbXBsZXMgYXJlIHN0b3JlZCBvbmUtYnktb25lLCBpLmUuIGFycmF5IHN0cnVjdHVyZSBpcyBsaWtlIGZvbGxvd2luZzogUkdCUkdCUkdCLi4uCi0gICAgICogVGhlIG51bWJlciBvZiBwaXhlbHMgaXMgKHNpemUgb2YgdGhlIGFycmF5KSAvIChudW1iZXIgb2YgY29tcG9uZW50cykuCi0gICAgICogQHBhcmFtIHQgLSBJQ0MgdHJhbnNmb3JtCi0gICAgICogQHBhcmFtIHNyYyAtIHNvdXJjZSBwaXhlbHMKLSAgICAgKiBAcGFyYW0gZHN0IC0gZGVzdGluYXRpb24gcGl4ZWxzCi0gICAgICogQHJldHVybiBkZXN0aW5hdGlvbiBwaXhlbHMsIHN0b3JlZCBpbiB0aGUgYXJyYXksIHBhc3NlZCBpbiBkc3QKLSAgICAgKi8KLSAgICBwdWJsaWMgc2hvcnRbXSB0cmFuc2xhdGVDb2xvcihJQ0NfVHJhbnNmb3JtIHQsIHNob3J0IHNyY1tdLCBzaG9ydCBkc3RbXSkgewotICAgICAgICBOYXRpdmVJbWFnZUZvcm1hdCBzcmNGbXQgPSBjcmVhdGVJbWFnZUZvcm1hdCh0LCBzcmMsIDAsIHRydWUpOwotICAgICAgICBOYXRpdmVJbWFnZUZvcm1hdCBkc3RGbXQgPSBjcmVhdGVJbWFnZUZvcm1hdCh0LCBkc3QsIHNyY0ZtdC5nZXROdW1Db2xzKCksIGZhbHNlKTsKLQotICAgICAgICB0LnRyYW5zbGF0ZUNvbG9ycyhzcmNGbXQsIGRzdEZtdCk7Ci0KLSAgICAgICAgcmV0dXJuIChzaG9ydFtdKSBkc3RGbXQuZ2V0Q2hhbm5lbERhdGEoKTsKLSAgICB9Ci0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgTmF0aXZlSW1hZ2VGb3JtYXQgZnJvbSBidWZmZXJlZCBpbWFnZS4KLSAgICAgKiBAcGFyYW0gYmkgLSBidWZmZXJlZCBpbWFnZQotICAgICAqIEByZXR1cm4gY3JlYXRlZCBOYXRpdmVJbWFnZUZvcm1hdAotICAgICAqLwotICAgIHByaXZhdGUgTmF0aXZlSW1hZ2VGb3JtYXQgY3JlYXRlSW1hZ2VGb3JtYXQoQnVmZmVyZWRJbWFnZSBiaSkgewotICAgICAgICBpbnQgblJvd3MgPSBiaS5nZXRIZWlnaHQoKTsKLSAgICAgICAgaW50IG5Db2xzID0gYmkuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IG5Db21wcyA9IGJpLmdldENvbG9yTW9kZWwoKS5nZXROdW1Db2xvckNvbXBvbmVudHMoKTsKLSAgICAgICAgc2hvcnQgaW1nRGF0YVtdID0gbmV3IHNob3J0W25Sb3dzKm5Db2xzKm5Db21wc107Ci0gICAgICAgIHJldHVybiBuZXcgTmF0aXZlSW1hZ2VGb3JtYXQoCi0gICAgICAgICAgICAgICAgaW1nRGF0YSwgbkNvbXBzLCBuUm93cywgbkNvbHMpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgb25lLXJvdyBOYXRpdmVJbWFnZUZvcm1hdCwgdXNpbmcgZWl0aGVyIG5Db2xzIGlmIGl0IGlzIHBvc2l0aXZlLAotICAgICAqIG9yIGFyci5sZW5ndGggdG8gZGV0ZXJtaW5lIHRoZSBudW1iZXIgb2YgcGl4ZWxzCi0gICAgICoKLSAgICAgKiBAcGFyYW0gdCAtIHRyYW5zZm9ybQotICAgICAqIEBwYXJhbSBhcnIgLSBzaG9ydCBhcnJheSBvciBudWxsIGlmIG5Db2xzIGlzIHBvc2l0aXZlCi0gICAgICogQHBhcmFtIG5Db2xzIC0gbnVtYmVyIG9mIHBpeGVscyBpbiB0aGUgYXJyYXkgb3IgMCBpZiBhcnJheSBpcyBub3QgbnVsbAotICAgICAqIEBwYXJhbSBpbiAtIGlzIGl0IGFuIGlucHV0IG9yIG91dHB1dCBhcnJheQotICAgICAqIEByZXR1cm4gb25lLXJvdyBOYXRpdmVJbWFnZUZvcm1hdAotICAgICAqLwotICAgIHByaXZhdGUgTmF0aXZlSW1hZ2VGb3JtYXQgY3JlYXRlSW1hZ2VGb3JtYXQoCi0gICAgICAgICAgICBJQ0NfVHJhbnNmb3JtIHQsIHNob3J0IGFycltdLCBpbnQgbkNvbHMsIGJvb2xlYW4gaW4KLSAgICApIHsKLSAgICAgICAgaW50IG5Db21wb25lbnRzID0gaW4gPyB0LmdldE51bUlucHV0Q2hhbm5lbHMoKSA6IHQuZ2V0TnVtT3V0cHV0Q2hhbm5lbHMoKTsKLQotICAgICAgICBpZiAoYXJyID09IG51bGwgfHwgYXJyLmxlbmd0aCA8IG5Db2xzKm5Db21wb25lbnRzKSB7Ci0gICAgICAgICAgICBhcnIgPSBuZXcgc2hvcnRbbkNvbHMqbkNvbXBvbmVudHNdOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG5Db2xzID09IDApCi0gICAgICAgICAgICBuQ29scyA9IGFyci5sZW5ndGggLyBuQ29tcG9uZW50czsKLQotICAgICAgICByZXR1cm4gbmV3IE5hdGl2ZUltYWdlRm9ybWF0KGFyciwgbkNvbXBvbmVudHMsIDEsIG5Db2xzKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9jb2xvci9Db2xvclNjYWxlci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvY29sb3IvQ29sb3JTY2FsZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTFjYzE2OS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9jb2xvci9Db2xvclNjYWxlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMzU1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuY29sb3I7Ci0KLWltcG9ydCBqYXZhLmF3dC5jb2xvci5Db2xvclNwYWNlOwotaW1wb3J0IGphdmEuYXd0LmNvbG9yLklDQ19Qcm9maWxlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmFzdGVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlNhbXBsZU1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLldyaXRhYmxlUmFzdGVyOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgcHJvdmlkZXMgZnVuY3Rpb25hbGl0eSBmb3Igc2NhbGluZyBjb2xvciBkYXRhIHdoZW4KLSAqIHJhbmdlcyBvZiB0aGUgc291cmNlIGFuZCBkZXN0aW5hdGlvbiBjb2xvciB2YWx1ZXMgZGlmZmVycy4gCi0gKi8KLXB1YmxpYyBjbGFzcyBDb2xvclNjYWxlciB7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgZmxvYXQgTUFYX1NIT1JUID0gMHhGRkZGOwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGZsb2F0IE1BWF9TSUdORURfU0hPUlQgPSAweDdGRkY7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBmbG9hdCBNQVhfWFlaID0gMWYgKyAoMzI3NjdmLzMyNzY4Zik7Ci0KLSAgICAvLyBDYWNoZWQgdmFsdWVzIGZvciBzY2FsaW5nIGNvbG9yIGRhdGEKLSAgICBwcml2YXRlIGZsb2F0W10gY2hhbm5lbE1pblZhbHVlcyA9IG51bGw7Ci0gICAgcHJpdmF0ZSBmbG9hdFtdIGNoYW5uZWxNdWxpcGxpZXJzID0gbnVsbDsgLy8gZm9yIHNjYWxlCi0gICAgcHJpdmF0ZSBmbG9hdFtdIGludkNoYW5uZWxNdWxpcGxpZXJzID0gbnVsbDsgLy8gZm9yIHVuc2NhbGUKLQotICAgIGludCBuQ29sb3JDaGFubmVscyA9IDA7Ci0KLSAgICAvLyBGb3Igc2NhbGluZyByYXN0ZXJzLCBmYWxzZSBpZiB0cmFuc2ZlciB0eXBlIGlzIGRvdWJsZSBvciBmbG9hdAotICAgIGJvb2xlYW4gaXNUVHlwZUludGVncmFsID0gZmFsc2U7Ci0KLSAgICAvKioKLSAgICAgKiBMb2FkcyBzY2FsaW5nIGRhdGEgZm9yIHJhc3Rlci4gTm90ZSwgaWYgcHJvZmlsZSBwZiBpcyBudWxsLAotICAgICAqIGZvciBub24taW50ZWdyYWwgZGF0YSB0eXBlcyBtdWx0aXBsaWVycyBhcmUgbm90IGluaXRpYWxpemVkLgotICAgICAqIEBwYXJhbSByIC0gcmFzdGVyCi0gICAgICogQHBhcmFtIHBmIC0gcHJvZmlsZSB3aGljaCBoZWxwcyB0byBkZXRlcm1pbmUgdGhlIHJhbmdlcyBvZiB0aGUgY29sb3IgZGF0YQotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGxvYWRTY2FsaW5nRGF0YShSYXN0ZXIgciwgSUNDX1Byb2ZpbGUgcGYpIHsKLSAgICAgICAgYm9vbGVhbiBpc1NyY1RUeXBlSW50ZWdyYWwgPQotICAgICAgICAgICAgci5nZXRUcmFuc2ZlclR5cGUoKSAhPSBEYXRhQnVmZmVyLlRZUEVfRkxPQVQgJiYKLSAgICAgICAgICAgIHIuZ2V0VHJhbnNmZXJUeXBlKCkgIT0gRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRTsKLSAgICAgICAgaWYgKGlzU3JjVFR5cGVJbnRlZ3JhbCkKLSAgICAgICAgICAgIGxvYWRTY2FsaW5nRGF0YShyLmdldFNhbXBsZU1vZGVsKCkpOwotICAgICAgICBlbHNlIGlmIChwZiAhPSBudWxsKQotICAgICAgICAgICAgbG9hZFNjYWxpbmdEYXRhKHBmKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBVc2UgdGhpcyBtZXRob2Qgb25seSBmb3IgaW50ZWdyYWwgdHJhbnNmZXIgdHlwZXMuCi0gICAgICogRXh0cmFjdHMgbWluL21heCB2YWx1ZXMgZnJvbSB0aGUgc2FtcGxlIG1vZGVsCi0gICAgICogQHBhcmFtIHNtIC0gc2FtcGxlIG1vZGVsCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgbG9hZFNjYWxpbmdEYXRhKFNhbXBsZU1vZGVsIHNtKSB7Ci0gICAgICAgIC8vIFN1cHBvc2luZyBpbnRlZ3JhbCB0cmFuc2ZlciB0eXBlCi0gICAgICAgIGlzVFR5cGVJbnRlZ3JhbCA9IHRydWU7Ci0KLSAgICAgICAgbkNvbG9yQ2hhbm5lbHMgPSBzbS5nZXROdW1CYW5kcygpOwotCi0gICAgICAgIGNoYW5uZWxNaW5WYWx1ZXMgPSBuZXcgZmxvYXRbbkNvbG9yQ2hhbm5lbHNdOwotICAgICAgICBjaGFubmVsTXVsaXBsaWVycyA9IG5ldyBmbG9hdFtuQ29sb3JDaGFubmVsc107Ci0gICAgICAgIGludkNoYW5uZWxNdWxpcGxpZXJzID0gbmV3IGZsb2F0W25Db2xvckNoYW5uZWxzXTsKLQotICAgICAgICBib29sZWFuIGlzU2lnbmVkU2hvcnQgPQotICAgICAgICAgICAgKHNtLmdldFRyYW5zZmVyVHlwZSgpID09IERhdGFCdWZmZXIuVFlQRV9TSE9SVCk7Ci0KLSAgICAgICAgZmxvYXQgbWF4VmFsOwotICAgICAgICBmb3IgKGludCBpPTA7IGk8bkNvbG9yQ2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgY2hhbm5lbE1pblZhbHVlc1tpXSA9IDA7Ci0gICAgICAgICAgICBpZiAoaXNTaWduZWRTaG9ydCkgewotICAgICAgICAgICAgICAgIGNoYW5uZWxNdWxpcGxpZXJzW2ldID0gTUFYX1NIT1JUIC8gTUFYX1NJR05FRF9TSE9SVDsKLSAgICAgICAgICAgICAgICBpbnZDaGFubmVsTXVsaXBsaWVyc1tpXSA9IE1BWF9TSUdORURfU0hPUlQgLyBNQVhfU0hPUlQ7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIG1heFZhbCA9ICgoMSA8PCBzbS5nZXRTYW1wbGVTaXplKGkpKSAtIDEpOwotICAgICAgICAgICAgICAgIGNoYW5uZWxNdWxpcGxpZXJzW2ldID0gTUFYX1NIT1JUIC8gbWF4VmFsOwotICAgICAgICAgICAgICAgIGludkNoYW5uZWxNdWxpcGxpZXJzW2ldID0gbWF4VmFsIC8gTUFYX1NIT1JUOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVXNlIHRoaXMgbWV0aG9kIG9ubHkgZm9yIGRvdWJsZSBvZiBmbG9hdCB0cmFuc2ZlciB0eXBlcy4KLSAgICAgKiBFeHRyYWN0cyBzY2FsaW5nIGRhdGEgZnJvbSB0aGUgY29sb3Igc3BhY2Ugc2lnbmF0dXJlCi0gICAgICogYW5kIG90aGVyIHRhZ3MsIHN0b3JlZCBpbiB0aGUgcHJvZmlsZQotICAgICAqIEBwYXJhbSBwZiAtIElDQyBwcm9maWxlCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgbG9hZFNjYWxpbmdEYXRhKElDQ19Qcm9maWxlIHBmKSB7Ci0gICAgICAgIC8vIFN1cHBvc2luZyBkb3VibGUgb3IgZmxvYXQgdHJhbnNmZXIgdHlwZQotICAgICAgICBpc1RUeXBlSW50ZWdyYWwgPSBmYWxzZTsKLQotICAgICAgICBuQ29sb3JDaGFubmVscyA9IHBmLmdldE51bUNvbXBvbmVudHMoKTsKLQotICAgICAgICAvLyBHZXQgbWluL21heCB2YWx1ZXMgZGlyZWN0bHkgZnJvbSB0aGUgcHJvZmlsZQotICAgICAgICAvLyBWZXJ5IG11Y2ggbGlrZSBmaWxsTWluTWF4VmFsdWVzIGluIElDQ19Db2xvclNwYWNlCi0gICAgICAgIGZsb2F0IG1heFZhbHVlc1tdID0gbmV3IGZsb2F0W25Db2xvckNoYW5uZWxzXTsKLSAgICAgICAgZmxvYXQgbWluVmFsdWVzW10gPSBuZXcgZmxvYXRbbkNvbG9yQ2hhbm5lbHNdOwotCi0gICAgICAgIHN3aXRjaCAocGYuZ2V0Q29sb3JTcGFjZVR5cGUoKSkgewotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLlRZUEVfWFlaOgotICAgICAgICAgICAgICAgIG1pblZhbHVlc1swXSA9IDA7Ci0gICAgICAgICAgICAgICAgbWluVmFsdWVzWzFdID0gMDsKLSAgICAgICAgICAgICAgICBtaW5WYWx1ZXNbMl0gPSAwOwotICAgICAgICAgICAgICAgIG1heFZhbHVlc1swXSA9IE1BWF9YWVo7Ci0gICAgICAgICAgICAgICAgbWF4VmFsdWVzWzFdID0gTUFYX1hZWjsKLSAgICAgICAgICAgICAgICBtYXhWYWx1ZXNbMl0gPSBNQVhfWFlaOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLlRZUEVfTGFiOgotICAgICAgICAgICAgICAgIG1pblZhbHVlc1swXSA9IDA7Ci0gICAgICAgICAgICAgICAgbWluVmFsdWVzWzFdID0gLTEyODsKLSAgICAgICAgICAgICAgICBtaW5WYWx1ZXNbMl0gPSAtMTI4OwotICAgICAgICAgICAgICAgIG1heFZhbHVlc1swXSA9IDEwMDsKLSAgICAgICAgICAgICAgICBtYXhWYWx1ZXNbMV0gPSAxMjc7Ci0gICAgICAgICAgICAgICAgbWF4VmFsdWVzWzJdID0gMTI3OwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpPTA7IGk8bkNvbG9yQ2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBtaW5WYWx1ZXNbaV0gPSAwOwotICAgICAgICAgICAgICAgICAgICBtYXhWYWx1ZXNbaV0gPSAxOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGNoYW5uZWxNaW5WYWx1ZXMgPSBtaW5WYWx1ZXM7Ci0gICAgICAgIGNoYW5uZWxNdWxpcGxpZXJzID0gbmV3IGZsb2F0W25Db2xvckNoYW5uZWxzXTsKLSAgICAgICAgaW52Q2hhbm5lbE11bGlwbGllcnMgPSBuZXcgZmxvYXRbbkNvbG9yQ2hhbm5lbHNdOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbkNvbG9yQ2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgY2hhbm5lbE11bGlwbGllcnNbaV0gPQotICAgICAgICAgICAgICAgIE1BWF9TSE9SVCAvIChtYXhWYWx1ZXNbaV0gLSBjaGFubmVsTWluVmFsdWVzW2ldKTsKLQotICAgICAgICAgICAgaW52Q2hhbm5lbE11bGlwbGllcnNbaV0gPQotICAgICAgICAgICAgICAgIChtYXhWYWx1ZXNbaV0gLSBjaGFubmVsTWluVmFsdWVzW2ldKSAvIE1BWF9TSE9SVDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEV4dHJhY3RzIHNjYWxpbmcgZGF0YSBmcm9tIHRoZSBjb2xvciBzcGFjZQotICAgICAqIEBwYXJhbSBjcyAtIGNvbG9yIHNwYWNlCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgbG9hZFNjYWxpbmdEYXRhKENvbG9yU3BhY2UgY3MpIHsKLSAgICAgICAgbkNvbG9yQ2hhbm5lbHMgPSBjcy5nZXROdW1Db21wb25lbnRzKCk7Ci0KLSAgICAgICAgY2hhbm5lbE1pblZhbHVlcyA9IG5ldyBmbG9hdFtuQ29sb3JDaGFubmVsc107Ci0gICAgICAgIGNoYW5uZWxNdWxpcGxpZXJzID0gbmV3IGZsb2F0W25Db2xvckNoYW5uZWxzXTsKLSAgICAgICAgaW52Q2hhbm5lbE11bGlwbGllcnMgPSBuZXcgZmxvYXRbbkNvbG9yQ2hhbm5lbHNdOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbkNvbG9yQ2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgY2hhbm5lbE1pblZhbHVlc1tpXSA9IGNzLmdldE1pblZhbHVlKGkpOwotCi0gICAgICAgICAgICBjaGFubmVsTXVsaXBsaWVyc1tpXSA9Ci0gICAgICAgICAgICAgICAgTUFYX1NIT1JUIC8gKGNzLmdldE1heFZhbHVlKGkpIC0gY2hhbm5lbE1pblZhbHVlc1tpXSk7Ci0KLSAgICAgICAgICAgIGludkNoYW5uZWxNdWxpcGxpZXJzW2ldID0KLSAgICAgICAgICAgICAgICAoY3MuZ2V0TWF4VmFsdWUoaSkgLSBjaGFubmVsTWluVmFsdWVzW2ldKSAvIE1BWF9TSE9SVDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNjYWxlcyBhbmQgbm9ybWFsaXplcyB0aGUgd2hvbGUgcmFzdGVyIGFuZCByZXR1cm5zIHRoZSByZXN1bHQKLSAgICAgKiBpbiB0aGUgZmxvYXQgYXJyYXkKLSAgICAgKiBAcGFyYW0gciAtIHNvdXJjZSByYXN0ZXIKLSAgICAgKiBAcmV0dXJuIHNjYWxlZCBhbmQgbm9ybWFsaXplZCByYXN0ZXIgZGF0YQotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdFtdW10gc2NhbGVOb3JtYWxpemUoUmFzdGVyIHIpIHsKLSAgICAgICAgaW50IHdpZHRoID0gci5nZXRXaWR0aCgpOwotICAgICAgICBpbnQgaGVpZ2h0ID0gci5nZXRIZWlnaHQoKTsKLSAgICAgICAgZmxvYXQgcmVzdWx0W11bXSA9IG5ldyBmbG9hdFt3aWR0aCpoZWlnaHRdW25Db2xvckNoYW5uZWxzXTsKLSAgICAgICAgZmxvYXQgbm9ybU11bHRpcGxpZXJzW10gPSBuZXcgZmxvYXRbbkNvbG9yQ2hhbm5lbHNdOwotCi0gICAgICAgIGludCBwb3MgPSAwOwotICAgICAgICBpZiAoaXNUVHlwZUludGVncmFsKSB7Ci0gICAgICAgICAgICAvLyBDaGFuZ2UgbWF4IHZhbHVlIGZyb20gTUFYX1NIT1JUIHRvIDFmCi0gICAgICAgICAgICBmb3IgKGludCBpPTA7IGk8bkNvbG9yQ2hhbm5lbHM7IGkrKykgewotICAgICAgICAgICAgICAgIG5vcm1NdWx0aXBsaWVyc1tpXSA9IGNoYW5uZWxNdWxpcGxpZXJzW2ldIC8gTUFYX1NIT1JUOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpbnQgc2FtcGxlOwotICAgICAgICAgICAgZm9yIChpbnQgcm93PXIuZ2V0TWluWCgpOyByb3c8d2lkdGg7IHJvdysrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgY29sPXIuZ2V0TWluWSgpOyBjb2w8aGVpZ2h0OyBjb2wrKykgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBjaGFuID0gMDsgY2hhbiA8IG5Db2xvckNoYW5uZWxzOyBjaGFuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHNhbXBsZSA9IHIuZ2V0U2FtcGxlKHJvdywgY29sLCBjaGFuKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlc3VsdFtwb3NdW2NoYW5dID0gKHNhbXBsZSAqIG5vcm1NdWx0aXBsaWVyc1tjaGFuXSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcG9zKys7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgeyAvLyBKdXN0IGdldCB0aGUgc2FtcGxlcy4uLgotICAgICAgICAgICAgZm9yIChpbnQgcm93PXIuZ2V0TWluWCgpOyByb3c8d2lkdGg7IHJvdysrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgY29sPXIuZ2V0TWluWSgpOyBjb2w8aGVpZ2h0OyBjb2wrKykgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBjaGFuID0gMDsgY2hhbiA8IG5Db2xvckNoYW5uZWxzOyBjaGFuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlc3VsdFtwb3NdW2NoYW5dID0gci5nZXRTYW1wbGVGbG9hdChyb3csIGNvbCwgY2hhbik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcG9zKys7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiByZXN1bHQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVW5zY2FsZSB0aGUgd2hvbGUgZmxvYXQgYXJyYXkgYW5kIHB1dCB0aGUgcmVzdWx0Ci0gICAgICogaW4gdGhlIHJhc3RlcgotICAgICAqIEBwYXJhbSByIC0gZGVzdGluYXRpb24gcmFzdGVyCi0gICAgICogQHBhcmFtIGRhdGEgLSBpbnB1dCBwaXhlbHMKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB1bnNjYWxlTm9ybWFsaXplZChXcml0YWJsZVJhc3RlciByLCBmbG9hdCBkYXRhW11bXSkgewotICAgICAgICBpbnQgd2lkdGggPSByLmdldFdpZHRoKCk7Ci0gICAgICAgIGludCBoZWlnaHQgPSByLmdldEhlaWdodCgpOwotICAgICAgICBmbG9hdCBub3JtTXVsdGlwbGllcnNbXSA9IG5ldyBmbG9hdFtuQ29sb3JDaGFubmVsc107Ci0KLSAgICAgICAgaW50IHBvcyA9IDA7Ci0gICAgICAgIGlmIChpc1RUeXBlSW50ZWdyYWwpIHsKLSAgICAgICAgICAgIC8vIENoYW5nZSBtYXggdmFsdWUgZnJvbSBNQVhfU0hPUlQgdG8gMWYKLSAgICAgICAgICAgIGZvciAoaW50IGk9MDsgaTxuQ29sb3JDaGFubmVsczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgbm9ybU11bHRpcGxpZXJzW2ldID0gaW52Q2hhbm5lbE11bGlwbGllcnNbaV0gKiBNQVhfU0hPUlQ7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGludCBzYW1wbGU7Ci0gICAgICAgICAgICBmb3IgKGludCByb3c9ci5nZXRNaW5YKCk7IHJvdzx3aWR0aDsgcm93KyspIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBjb2w9ci5nZXRNaW5ZKCk7IGNvbDxoZWlnaHQ7IGNvbCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGNoYW4gPSAwOyBjaGFuIDwgbkNvbG9yQ2hhbm5lbHM7IGNoYW4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgc2FtcGxlID0gKGludCkgKGRhdGFbcG9zXVtjaGFuXSAqIG5vcm1NdWx0aXBsaWVyc1tjaGFuXSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICAgICAgci5zZXRTYW1wbGUocm93LCBjb2wsIGNoYW4sIHNhbXBsZSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcG9zKys7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgeyAvLyBKdXN0IHNldCB0aGUgc2FtcGxlcy4uLgotICAgICAgICAgICAgZm9yIChpbnQgcm93PXIuZ2V0TWluWCgpOyByb3c8d2lkdGg7IHJvdysrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgY29sPXIuZ2V0TWluWSgpOyBjb2w8aGVpZ2h0OyBjb2wrKykgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBjaGFuID0gMDsgY2hhbiA8IG5Db2xvckNoYW5uZWxzOyBjaGFuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHIuc2V0U2FtcGxlKHJvdywgY29sLCBjaGFuLCBkYXRhW3Bvc11bY2hhbl0pOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHBvcysrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNjYWxlcyB0aGUgd2hvbGUgcmFzdGVyIHRvIHNob3J0IGFuZCByZXR1cm5zIHRoZSByZXN1bHQKLSAgICAgKiBpbiB0aGUgYXJyYXkKLSAgICAgKiBAcGFyYW0gciAtIHNvdXJjZSByYXN0ZXIKLSAgICAgKiBAcmV0dXJuIHNjYWxlZCBhbmQgbm9ybWFsaXplZCByYXN0ZXIgZGF0YQotICAgICAqLwotICAgIHB1YmxpYyBzaG9ydFtdIHNjYWxlKFJhc3RlciByKSB7Ci0gICAgICAgIGludCB3aWR0aCA9IHIuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IGhlaWdodCA9IHIuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgIHNob3J0IHJlc3VsdFtdID0gbmV3IHNob3J0W3dpZHRoKmhlaWdodCpuQ29sb3JDaGFubmVsc107Ci0KLSAgICAgICAgaW50IHBvcyA9IDA7Ci0gICAgICAgIGlmIChpc1RUeXBlSW50ZWdyYWwpIHsKLSAgICAgICAgICAgIGludCBzYW1wbGU7Ci0gICAgICAgICAgICBmb3IgKGludCByb3c9ci5nZXRNaW5YKCk7IHJvdzx3aWR0aDsgcm93KyspIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBjb2w9ci5nZXRNaW5ZKCk7IGNvbDxoZWlnaHQ7IGNvbCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZvciAoaW50IGNoYW4gPSAwOyBjaGFuIDwgbkNvbG9yQ2hhbm5lbHM7IGNoYW4rKykgewotICAgICAgICAgICAgICAgICAgICAgICAgc2FtcGxlID0gci5nZXRTYW1wbGUocm93LCBjb2wsIGNoYW4pOwotICAgICAgICAgICAgICAgICAgICAgICAgcmVzdWx0W3BvcysrXSA9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKHNob3J0KSAoc2FtcGxlICogY2hhbm5lbE11bGlwbGllcnNbY2hhbl0gKyAwLjVmKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZsb2F0IHNhbXBsZTsKLSAgICAgICAgICAgIGZvciAoaW50IHJvdz1yLmdldE1pblgoKTsgcm93PHdpZHRoOyByb3crKykgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGNvbD1yLmdldE1pblkoKTsgY29sPGhlaWdodDsgY29sKyspIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgY2hhbiA9IDA7IGNoYW4gPCBuQ29sb3JDaGFubmVsczsgY2hhbisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzYW1wbGUgPSByLmdldFNhbXBsZUZsb2F0KHJvdywgY29sLCBjaGFuKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlc3VsdFtwb3MrK10gPSAoc2hvcnQpICgoc2FtcGxlIC0gY2hhbm5lbE1pblZhbHVlc1tjaGFuXSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAqIGNoYW5uZWxNdWxpcGxpZXJzW2NoYW5dICsgMC41Zik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBVbnNjYWxlcyB0aGUgd2hvbGUgZGF0YSBhcnJheSBhbmQgcHV0cyBvYnRhaW5lZCB2YWx1ZXMgdG8gdGhlIHJhc3RlcgotICAgICAqIEBwYXJhbSBkYXRhIC0gaW5wdXQgZGF0YQotICAgICAqIEBwYXJhbSB3ciAtIGRlc3RpbmF0aW9uIHJhc3RlcgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHVuc2NhbGUoc2hvcnRbXSBkYXRhLCBXcml0YWJsZVJhc3RlciB3cikgewotICAgICAgICBpbnQgd2lkdGggPSB3ci5nZXRXaWR0aCgpOwotICAgICAgICBpbnQgaGVpZ2h0ID0gd3IuZ2V0SGVpZ2h0KCk7Ci0KLSAgICAgICAgaW50IHBvcyA9IDA7Ci0gICAgICAgIGlmIChpc1RUeXBlSW50ZWdyYWwpIHsKLSAgICAgICAgICAgIGludCBzYW1wbGU7Ci0gICAgICAgICAgICBmb3IgKGludCByb3c9d3IuZ2V0TWluWCgpOyByb3c8d2lkdGg7IHJvdysrKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgY29sPXdyLmdldE1pblkoKTsgY29sPGhlaWdodDsgY29sKyspIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgY2hhbiA9IDA7IGNoYW4gPCBuQ29sb3JDaGFubmVsczsgY2hhbisrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgc2FtcGxlID0gKGludCkgKChkYXRhW3BvcysrXSAmIDB4RkZGRikgKgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnZDaGFubmVsTXVsaXBsaWVyc1tjaGFuXSArIDAuNWYpOwotICAgICAgICAgICAgICAgICAgICAgICAgIHdyLnNldFNhbXBsZShyb3csIGNvbCwgY2hhbiwgc2FtcGxlKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGZsb2F0IHNhbXBsZTsKLSAgICAgICAgICAgIGZvciAoaW50IHJvdz13ci5nZXRNaW5YKCk7IHJvdzx3aWR0aDsgcm93KyspIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBjb2w9d3IuZ2V0TWluWSgpOyBjb2w8aGVpZ2h0OyBjb2wrKykgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBjaGFuID0gMDsgY2hhbiA8IG5Db2xvckNoYW5uZWxzOyBjaGFuKyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICBzYW1wbGUgPSAoZGF0YVtwb3MrK10gJiAweEZGRkYpICoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnZDaGFubmVsTXVsaXBsaWVyc1tjaGFuXSArIGNoYW5uZWxNaW5WYWx1ZXNbY2hhbl07Ci0gICAgICAgICAgICAgICAgICAgICAgICAgd3Iuc2V0U2FtcGxlKHJvdywgY29sLCBjaGFuLCBzYW1wbGUpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2NhbGVzIG9uZSBwaXhlbCBhbmQgcHV0cyBvYnRhaW5lZCB2YWx1ZXMgdG8gdGhlIGNoYW5EYXRhCi0gICAgICogQHBhcmFtIHBpeGVsRGF0YSAtIGlucHV0IHBpeGVsCi0gICAgICogQHBhcmFtIGNoYW5EYXRhIC0gb3V0cHV0IGJ1ZmZlcgotICAgICAqIEBwYXJhbSBjaGFuRGF0YU9mZnNldCAtIG91dHB1dCBidWZmZXIgb2Zmc2V0Ci0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2NhbGUoZmxvYXRbXSBwaXhlbERhdGEsIHNob3J0W10gY2hhbkRhdGEsIGludCBjaGFuRGF0YU9mZnNldCkgewotICAgICAgICBmb3IgKGludCBjaGFuID0gMDsgY2hhbiA8IG5Db2xvckNoYW5uZWxzOyBjaGFuKyspIHsKLSAgICAgICAgICAgIGNoYW5EYXRhW2NoYW5EYXRhT2Zmc2V0ICsgY2hhbl0gPQotICAgICAgICAgICAgICAgICAgICAoc2hvcnQpICgocGl4ZWxEYXRhW2NoYW5dIC0gY2hhbm5lbE1pblZhbHVlc1tjaGFuXSkgKgotICAgICAgICAgICAgICAgICAgICAgICAgY2hhbm5lbE11bGlwbGllcnNbY2hhbl0gKyAwLjVmKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFVuc2NhbGVzIG9uZSBwaXhlbCBhbmQgcHV0cyBvYnRhaW5lZCB2YWx1ZXMgdG8gdGhlIHBpeGVsRGF0YQotICAgICAqIEBwYXJhbSBwaXhlbERhdGEgLSBvdXRwdXQgcGl4ZWwKLSAgICAgKiBAcGFyYW0gY2hhbkRhdGEgLSBpbnB1dCBidWZmZXIKLSAgICAgKiBAcGFyYW0gY2hhbkRhdGFPZmZzZXQgLSBpbnB1dCBidWZmZXIgb2Zmc2V0Ci0gICAgICovCi0gICAgcHVibGljIHZvaWQgdW5zY2FsZShmbG9hdFtdIHBpeGVsRGF0YSwgc2hvcnRbXSBjaGFuRGF0YSwgaW50IGNoYW5EYXRhT2Zmc2V0KSB7Ci0gICAgICAgIGZvciAoaW50IGNoYW4gPSAwOyBjaGFuIDwgbkNvbG9yQ2hhbm5lbHM7IGNoYW4rKykgewotICAgICAgICAgICAgcGl4ZWxEYXRhW2NoYW5dID0gKGNoYW5EYXRhW2NoYW5EYXRhT2Zmc2V0ICsgY2hhbl0gJiAweEZGRkYpCi0gICAgICAgICAgICAgICAgKiBpbnZDaGFubmVsTXVsaXBsaWVyc1tjaGFuXSArIGNoYW5uZWxNaW5WYWx1ZXNbY2hhbl07Ci0gICAgICAgIH0KLSAgICB9Ci19ClwgTm8gbmV3bGluZSBhdCBlbmQgb2YgZmlsZQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvY29sb3IvSUNDX1Byb2ZpbGVIZWxwZXIuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2NvbG9yL0lDQ19Qcm9maWxlSGVscGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJmN2U1MTkuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvY29sb3IvSUNDX1Byb2ZpbGVIZWxwZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDgyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuY29sb3I7Ci0KLWltcG9ydCBqYXZhLmF3dC5jb2xvci5JQ0NfUHJvZmlsZTsKLQotLyoqCi0gKiBJbmNsdWRlcyB1dGlsaXR5IG1ldGhvZHMgZm9yIHJlYWRpbmcgSUNDIHByb2ZpbGUgZGF0YS4KLSAqIENyZWF0ZWQgdG8gcHJvdmlkZSBwdWJsaWMgYWNjZXNzIHRvIElDQ19Qcm9maWxlIG1ldGhvZHMKLSAqIGZvciBjbGFzc2VzIG91dHNpZGUgb2YgamF2YS5hd3QuY29sb3IKLSAqLwotcHVibGljIGNsYXNzIElDQ19Qcm9maWxlSGVscGVyIHsKLSAgICAvKioKLSAgICAgKiBVdGlsaXR5IG1ldGhvZC4KLSAgICAgKiBHZXRzIGludGVnZXIgdmFsdWUgZnJvbSB0aGUgYnl0ZSBhcnJheQotICAgICAqIEBwYXJhbSBieXRlQXJyYXkgLSBieXRlIGFycmF5Ci0gICAgICogQHBhcmFtIGlkeCAtIGJ5dGUgb2Zmc2V0Ci0gICAgICogQHJldHVybiBpbnRlZ2VyIHZhbHVlCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnQgZ2V0SW50RnJvbUJ5dGVBcnJheShieXRlW10gYnl0ZUFycmF5LCBpbnQgaWR4KSB7Ci0gICAgICAgIHJldHVybiAoYnl0ZUFycmF5W2lkeF0gJiAweEZGKXwKLSAgICAgICAgICAgICAgICgoYnl0ZUFycmF5W2lkeCsxXSAmIDB4RkYpIDw8IDgpIHwKLSAgICAgICAgICAgICAgICgoYnl0ZUFycmF5W2lkeCsyXSAmIDB4RkYpIDw8IDE2KXwKLSAgICAgICAgICAgICAgICgoYnl0ZUFycmF5W2lkeCszXSAmIDB4RkYpIDw8IDI0KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBVdGlsaXR5IG1ldGhvZC4KLSAgICAgKiBHZXRzIGJpZyBlbmRpYW4gaW50ZWdlciB2YWx1ZSBmcm9tIHRoZSBieXRlIGFycmF5Ci0gICAgICogQHBhcmFtIGJ5dGVBcnJheSAtIGJ5dGUgYXJyYXkKLSAgICAgKiBAcGFyYW0gaWR4IC0gYnl0ZSBvZmZzZXQKLSAgICAgKiBAcmV0dXJuIGludGVnZXIgdmFsdWUKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBnZXRCaWdFbmRpYW5Gcm9tQnl0ZUFycmF5KGJ5dGVbXSBieXRlQXJyYXksIGludCBpZHgpIHsKLSAgICAgICAgcmV0dXJuICgoYnl0ZUFycmF5W2lkeF0gJiAweEZGKSA8PCAyNCkgICB8Ci0gICAgICAgICAgICAgICAoKGJ5dGVBcnJheVtpZHgrMV0gJiAweEZGKSA8PCAxNikgfAotICAgICAgICAgICAgICAgKChieXRlQXJyYXlbaWR4KzJdICYgMHhGRikgPDwgOCkgIHwKLSAgICAgICAgICAgICAgICggYnl0ZUFycmF5W2lkeCszXSAmIDB4RkYpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFV0aWxpdHkgbWV0aG9kLgotICAgICAqIEdldHMgc2hvcnQgdmFsdWUgZnJvbSB0aGUgYnl0ZSBhcnJheQotICAgICAqIEBwYXJhbSBieXRlQXJyYXkgLSBieXRlIGFycmF5Ci0gICAgICogQHBhcmFtIGlkeCAtIGJ5dGUgb2Zmc2V0Ci0gICAgICogQHJldHVybiBzaG9ydCB2YWx1ZQotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgc2hvcnQgZ2V0U2hvcnRGcm9tQnl0ZUFycmF5KGJ5dGVbXSBieXRlQXJyYXksIGludCBpZHgpIHsKLSAgICAgICAgcmV0dXJuIChzaG9ydCkgKChieXRlQXJyYXlbaWR4XSAmIDB4RkYpIHwKLSAgICAgICAgICAgICAgICAgICAgICAgKChieXRlQXJyYXlbaWR4KzFdICYgMHhGRikgPDwgOCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFVzZWQgaW4gSUNDX1RyYW5zZm9ybSBjbGFzcyB0byBjaGVjayB0aGUgcmVuZGVyaW5nIGludGVudCBvZiB0aGUgcHJvZmlsZQotICAgICAqIEBwYXJhbSBwcm9maWxlIC0gSUNDIHByb2ZpbGUKLSAgICAgKiBAcmV0dXJuIHJlbmRlcmluZyBpbnRlbnQKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBnZXRSZW5kZXJpbmdJbnRlbnQoSUNDX1Byb2ZpbGUgcHJvZmlsZSkgewotICAgICAgICByZXR1cm4gZ2V0SW50RnJvbUJ5dGVBcnJheSgKLSAgICAgICAgICAgICAgICBwcm9maWxlLmdldERhdGEoSUNDX1Byb2ZpbGUuaWNTaWdIZWFkKSwgLy8gcGYgaGVhZGVyCi0gICAgICAgICAgICAgICAgSUNDX1Byb2ZpbGUuaWNIZHJSZW5kZXJpbmdJbnRlbnQKLSAgICAgICAgICAgICk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvY29sb3IvSUNDX1RyYW5zZm9ybS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvY29sb3IvSUNDX1RyYW5zZm9ybS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyNzY0NmM0Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2NvbG9yL0lDQ19UcmFuc2Zvcm0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE1NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmNvbG9yOwotCi1pbXBvcnQgamF2YS5hd3QuY29sb3IuSUNDX1Byb2ZpbGU7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmNvbG9yLk5hdGl2ZUNNTTsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGVuY2Fwc3VsYXRlcyBuYXRpdmUgSUNDIHRyYW5zZm9ybSBvYmplY3QsIGlzIHJlc3BvbnNpYmxlIGZvciBpdHMKLSAqIGNyZWF0aW9uLCBkZXN0cnVjdGlvbiBhbmQgcGFzc2luZyBpdHMgaGFuZGxlIHRvIHRoZSBuYXRpdmUgQ01NLgotICovCi1wdWJsaWMgY2xhc3MgSUNDX1RyYW5zZm9ybSB7Ci0gICAgcHJpdmF0ZSBsb25nIHRyYW5zZm9ybUhhbmRsZTsKLSAgICBwcml2YXRlIGludCBudW1JbnB1dENoYW5uZWxzOwotICAgIHByaXZhdGUgaW50IG51bU91dHB1dENoYW5uZWxzOwotICAgIHByaXZhdGUgSUNDX1Byb2ZpbGUgc3JjOwotICAgIHByaXZhdGUgSUNDX1Byb2ZpbGUgZHN0OwotCi0KLSAgICAvKioKLSAgICAgKiBAcmV0dXJuIFJldHVybnMgdGhlIG51bWJlciBvZiBpbnB1dCBjaGFubmVscy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE51bUlucHV0Q2hhbm5lbHMoKSB7Ci0gICAgICAgIHJldHVybiBudW1JbnB1dENoYW5uZWxzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEByZXR1cm4gUmV0dXJucyB0aGUgbnVtYmVyIG9mIG91dHB1dCBjaGFubmVscy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldE51bU91dHB1dENoYW5uZWxzKCkgewotICAgICAgICByZXR1cm4gbnVtT3V0cHV0Q2hhbm5lbHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQHJldHVybiBSZXR1cm5zIHRoZSBkc3QuCi0gICAgICovCi0gICAgcHVibGljIElDQ19Qcm9maWxlIGdldERzdCgpIHsKLSAgICAgICAgcmV0dXJuIGRzdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBAcmV0dXJuIFJldHVybnMgdGhlIHNyYy4KLSAgICAgKi8KLSAgICBwdWJsaWMgSUNDX1Byb2ZpbGUgZ2V0U3JjKCkgewotICAgICAgICByZXR1cm4gc3JjOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENvbnN0cnVjdHMgYSBtdWx0aXByb2ZpbGUgSUNDIHRyYW5zZm9ybQotICAgICAqIEBwYXJhbSBwcm9maWxlcyAtIGxpc3Qgb2YgSUNDIHByb2ZpbGVzCi0gICAgICogQHBhcmFtIHJlbmRlckludGVudHMgLSBvbmx5IGhpbnRzIGZvciBDTU0KLSAgICAgKi8KLSAgICBwdWJsaWMgSUNDX1RyYW5zZm9ybShJQ0NfUHJvZmlsZVtdIHByb2ZpbGVzLCBpbnRbXSByZW5kZXJJbnRlbnRzKSB7Ci0gICAgICAgIGludCBudW1Qcm9maWxlcyA9IHByb2ZpbGVzLmxlbmd0aDsKLQotICAgICAgICBsb25nW10gcHJvZmlsZUhhbmRsZXMgPSBuZXcgbG9uZ1tudW1Qcm9maWxlc107Ci0gICAgICAgIGZvciAoaW50IGk9MDsgaTxudW1Qcm9maWxlczsgaSsrKSB7Ci0gICAgICAgICAgICBwcm9maWxlSGFuZGxlc1tpXSA9IE5hdGl2ZUNNTS5nZXRIYW5kbGUocHJvZmlsZXNbaV0pOwotICAgICAgICB9Ci0KLSAgICAgICAgdHJhbnNmb3JtSGFuZGxlID0gTmF0aXZlQ01NLmNtbUNyZWF0ZU11bHRpcHJvZmlsZVRyYW5zZm9ybSgKLSAgICAgICAgICAgICAgICBwcm9maWxlSGFuZGxlcywKLSAgICAgICAgICAgICAgICByZW5kZXJJbnRlbnRzKTsKLQotICAgICAgICBzcmMgPSBwcm9maWxlc1swXTsKLSAgICAgICAgZHN0ID0gcHJvZmlsZXNbbnVtUHJvZmlsZXMtMV07Ci0gICAgICAgIG51bUlucHV0Q2hhbm5lbHMgPSBzcmMuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgICAgICBudW1PdXRwdXRDaGFubmVscyA9IGRzdC5nZXROdW1Db21wb25lbnRzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhpcyBjb25zdHJ1Y3RvciBpcyBhYmxlIHRvIHNldCBpbnRlbnRzIGJ5IGRlZmF1bHQKLSAgICAgKiBAcGFyYW0gcHJvZmlsZXMgLSBsaXN0IG9mIElDQyBwcm9maWxlcwotICAgICAqLwotICAgIHB1YmxpYyBJQ0NfVHJhbnNmb3JtKElDQ19Qcm9maWxlW10gcHJvZmlsZXMpIHsKLSAgICAgICAgaW50IG51bVByb2ZpbGVzID0gcHJvZmlsZXMubGVuZ3RoOwotICAgICAgICBpbnRbXSByZW5kZXJpbmdJbnRlbnRzID0gbmV3IGludFtudW1Qcm9maWxlc107Ci0KLSAgICAgICAgLy8gRGVmYXVsdCBpcyBwZXJjZXB0dWFsCi0gICAgICAgIGludCBjdXJyUmVuZGVyaW5nSW50ZW50ID0gSUNDX1Byb2ZpbGUuaWNQZXJjZXB0dWFsOwotCi0gICAgICAgIC8vIHJlbmRlciBhcyBjb2xvcmltZXRyaWMgZm9yIG91dHB1dCBkZXZpY2UKLSAgICAgICAgaWYgKHByb2ZpbGVzWzBdLmdldFByb2ZpbGVDbGFzcygpID09IElDQ19Qcm9maWxlLkNMQVNTX09VVFBVVCkgewotICAgICAgICAgICAgY3VyclJlbmRlcmluZ0ludGVudCA9IElDQ19Qcm9maWxlLmljUmVsYXRpdmVDb2xvcmltZXRyaWM7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBnZXQgdGhlIHRyYW5zZm9ybXMgZnJvbSBlYWNoIHByb2ZpbGUKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1Qcm9maWxlczsgaSsrKSB7Ci0gICAgICAgICAgICAvLyBmaXJzdCBvciBsYXN0IHByb2ZpbGUgY2Fubm90IGJlIGFic3RyYWN0Ci0gICAgICAgICAgICAvLyBpZiBwcm9maWxlIGlzIGFic3RyYWN0LCB0aGUgb25seSBwb3NzaWJsZSB3YXkgaXMKLSAgICAgICAgICAgIC8vIHVzZSBBVG9CMFRhZyAocGVyY2VwdHVhbCksIHNlZSBJQ0Mgc3BlYwotICAgICAgICAgICAgaWYgKGkgIT0gMCAmJgotICAgICAgICAgICAgICAgaSAhPSBudW1Qcm9maWxlcyAtIDEgJiYKLSAgICAgICAgICAgICAgIHByb2ZpbGVzW2ldLmdldFByb2ZpbGVDbGFzcygpID09IElDQ19Qcm9maWxlLkNMQVNTX0FCU1RSQUNUCi0gICAgICAgICAgICApIHsKLSAgICAgICAgICAgICAgICBjdXJyUmVuZGVyaW5nSW50ZW50ID0gSUNDX1Byb2ZpbGUuaWNQZXJjZXB0dWFsOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZW5kZXJpbmdJbnRlbnRzW2ldID0gY3VyclJlbmRlcmluZ0ludGVudDsKLSAgICAgICAgICAgIC8vIHVzZSBjdXJyZW50IHJlbmRlcmluZyBpbnRlbnQKLSAgICAgICAgICAgIC8vIHRvIHNlbGVjdCBMVVQgZnJvbSB0aGUgbmV4dCBwcm9maWxlIChjaGFpbmluZykKLSAgICAgICAgICAgIGN1cnJSZW5kZXJpbmdJbnRlbnQgPQotICAgICAgICAgICAgICAgIElDQ19Qcm9maWxlSGVscGVyLmdldFJlbmRlcmluZ0ludGVudChwcm9maWxlc1tpXSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBHZXQgdGhlIHByb2ZpbGUgaGFuZGxlcyBhbmQgZ28gYWhlYWQKLSAgICAgICAgbG9uZ1tdIHByb2ZpbGVIYW5kbGVzID0gbmV3IGxvbmdbbnVtUHJvZmlsZXNdOwotICAgICAgICBmb3IgKGludCBpPTA7IGk8bnVtUHJvZmlsZXM7IGkrKykgewotICAgICAgICAgICAgcHJvZmlsZUhhbmRsZXNbaV0gPSBOYXRpdmVDTU0uZ2V0SGFuZGxlKHByb2ZpbGVzW2ldKTsKLSAgICAgICAgfQotCi0gICAgICAgIHRyYW5zZm9ybUhhbmRsZSA9IE5hdGl2ZUNNTS5jbW1DcmVhdGVNdWx0aXByb2ZpbGVUcmFuc2Zvcm0oCi0gICAgICAgICAgICAgICAgcHJvZmlsZUhhbmRsZXMsCi0gICAgICAgICAgICAgICAgcmVuZGVyaW5nSW50ZW50cyk7Ci0KLSAgICAgICAgc3JjID0gcHJvZmlsZXNbMF07Ci0gICAgICAgIGRzdCA9IHByb2ZpbGVzW251bVByb2ZpbGVzLTFdOwotICAgICAgICBudW1JbnB1dENoYW5uZWxzID0gc3JjLmdldE51bUNvbXBvbmVudHMoKTsKLSAgICAgICAgbnVtT3V0cHV0Q2hhbm5lbHMgPSBkc3QuZ2V0TnVtQ29tcG9uZW50cygpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHByb3RlY3RlZCB2b2lkIGZpbmFsaXplKCkgewotICAgICAgICBpZiAodHJhbnNmb3JtSGFuZGxlICE9IDApIHsKLSAgICAgICAgICAgIE5hdGl2ZUNNTS5jbW1EZWxldGVUcmFuc2Zvcm0odHJhbnNmb3JtSGFuZGxlKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEludm9rZXMgbmF0aXZlIGNvbG9yIGNvbnZlcnNpb24KLSAgICAgKiBAcGFyYW0gc3JjIC0gc291cmNlIGltYWdlIGZvcm1hdAotICAgICAqIEBwYXJhbSBkc3QgLSBkZXN0aW5hdGlvbiBpbWFnZSBmb3JtYXQKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB0cmFuc2xhdGVDb2xvcnMoTmF0aXZlSW1hZ2VGb3JtYXQgc3JjLCBOYXRpdmVJbWFnZUZvcm1hdCBkc3QpIHsKLSAgICAgICAgTmF0aXZlQ01NLmNtbVRyYW5zbGF0ZUNvbG9ycyh0cmFuc2Zvcm1IYW5kbGUsIHNyYywgZHN0KTsKLSAgICB9Ci19ClwgTm8gbmV3bGluZSBhdCBlbmQgb2YgZmlsZQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvY29sb3IvTFVUQ29sb3JDb252ZXJ0ZXIuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2NvbG9yL0xVVENvbG9yQ29udmVydGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDVlYTZkMjUuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvY29sb3IvTFVUQ29sb3JDb252ZXJ0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE0OCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLS8qCi0gKiBDcmVhdGVkIG9uIDAyLjExLjIwMDQKLSAqCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvcjsKLQotaW1wb3J0IGphdmEuYXd0LmNvbG9yLkNvbG9yU3BhY2U7Ci0KLXB1YmxpYyBjbGFzcyBMVVRDb2xvckNvbnZlcnRlciB7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBieXRlIGZyb204bFJHQnRvc1JHQl9MVVRbXTsKLQotICAgIHByaXZhdGUgc3RhdGljIGJ5dGUgZnJvbTE2bFJHQnRvc1JHQl9MVVRbXTsKLQotICAgIHByaXZhdGUgc3RhdGljIGJ5dGUgZnJvbXNSR0J0bzhsUkdCX0xVVFtdOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgc2hvcnQgZnJvbXNSR0J0bzE2bFJHQl9MVVRbXTsKLQotICAgIHByaXZhdGUgc3RhdGljIGJ5dGUgZnJvbXNSR0J0bzhzUkdCX0xVVHNbXVtdOwotCi0gICAgcHVibGljIHN0YXRpYyBDb2xvclNwYWNlIExJTkVBUl9SR0JfQ1M7Ci0KLSAgICBwdWJsaWMgc3RhdGljIENvbG9yU3BhY2UgTElORUFSX0dSQVlfQ1M7Ci0KLSAgICBwdWJsaWMgc3RhdGljIENvbG9yU3BhY2Ugc1JHQl9DUzsKLQotICAgIHB1YmxpYyBMVVRDb2xvckNvbnZlcnRlcigpIHsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFRoaXMgY2xhc3MgcHJlcGFyZWQgYW5kIHJldHVybmVkIGxvb2t1cCB0YWJsZXMgZm9yIGNvbnZlcnNpb24gY29sb3IgCi0gICAgICogdmFsdWVzIGZyb20gTGluZWFyIFJHQiBDb2xvciBTcGFjZSB0byBzUkdCIGFuZCB2aWNlIHZlcnNhLgotICAgICAqIENvbnZlcnNpb24gaXMgcHJvZHVjaW5nIGFjY29yZGluZyB0byBzUkdCIENvbG9yIFNwYWNlIGRlZmluaXRpb24uCi0gICAgICogIkEgU3RhbmRhcmQgRGVmYXVsdCBDb2xvciBTcGFjZSBmb3IgdGhlIEludGVybmV0IC0gc1JHQiIsCi0gICAgICogIE1pY2hhZWwgU3Rva2VzIChIZXdsZXR0LVBhY2thcmQpLCBNYXR0aGV3IEFuZGVyc29uIChNaWNyb3NvZnQpLCAKLSAgICAgKiBTcmluaXZhc2FuIENoYW5kcmFzZWthciAoTWljcm9zb2Z0KSwgUmljYXJkbyBNb3R0YSAoSGV3bGV0dC1QYWNrYXJkKSAKLSAgICAgKiBWZXJzaW9uIDEuMTAsIE5vdmVtYmVyIDUsIDE5OTYgCi0gICAgICogVGhpcyBkb2N1bWVudCBpcyBhdmFpbGFibGU6IGh0dHA6Ly93d3cudzMub3JnL0dyYXBoaWNzL0NvbG9yL3NSR0IKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGJ5dGVbXSBnZXRGcm9tOGxSR0J0b3NSR0JfTFVUKCkgewotICAgICAgICBpZiAoZnJvbThsUkdCdG9zUkdCX0xVVCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBmcm9tOGxSR0J0b3NSR0JfTFVUID0gbmV3IGJ5dGVbMjU2XTsKLSAgICAgICAgICAgIGZsb2F0IHY7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IDI1NjsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgdiA9IChmbG9hdClpIC8gMjU1OwotICAgICAgICAgICAgICAgIHYgPSAodiA8PSAwLjA0MDQ1ZikgPyB2IC8gMTIuOTJmIDoKLSAgICAgICAgICAgICAgICAgICAgKGZsb2F0KSBNYXRoLnBvdygodiArIDAuMDU1KSAvIDEuMDU1LCAyLjQpOwotICAgICAgICAgICAgICAgIGZyb204bFJHQnRvc1JHQl9MVVRbaV0gPSAoYnl0ZSkgTWF0aC5yb3VuZCh2ICogMjU1LjBmKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZnJvbThsUkdCdG9zUkdCX0xVVDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIGJ5dGVbXSBnZXRGcm9tMTZsUkdCdG9zUkdCX0xVVCgpIHsKLSAgICAgICAgaWYgKGZyb20xNmxSR0J0b3NSR0JfTFVUID09IG51bGwpIHsKLSAgICAgICAgICAgIGZyb20xNmxSR0J0b3NSR0JfTFVUID0gbmV3IGJ5dGVbNjU1MzZdOwotICAgICAgICAgICAgZmxvYXQgdjsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgNjU1MzY7IGkrKykgewotICAgICAgICAgICAgICAgIHYgPSAoZmxvYXQpIGkgLyA2NTUzNTsKLSAgICAgICAgICAgICAgICB2ID0gKHYgPD0gMC4wNDA0NWYpID8gdiAvIDEyLjkyZiA6Ci0gICAgICAgICAgICAgICAgICAgIChmbG9hdCkgTWF0aC5wb3coKHYgKyAwLjA1NSkgLyAxLjA1NSwgMi40KTsKLSAgICAgICAgICAgICAgICBmcm9tMTZsUkdCdG9zUkdCX0xVVFtpXSA9IChieXRlKSBNYXRoLnJvdW5kKHYgKiAyNTUuMGYpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmcm9tMTZsUkdCdG9zUkdCX0xVVDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIGJ5dGVbXSBnZXRGcm9tc1JHQnRvOGxSR0JfTFVUKCkgewotICAgICAgICBpZiAoZnJvbXNSR0J0bzhsUkdCX0xVVCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBmcm9tc1JHQnRvOGxSR0JfTFVUID0gbmV3IGJ5dGVbMjU2XTsKLSAgICAgICAgICAgIGZsb2F0IHY7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IDI1NjsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgdiA9IChmbG9hdCkgaSAvIDI1NTsKLSAgICAgICAgICAgICAgICB2ID0gKHYgPD0gMC4wMDMxMzA4ZikgPyB2ICogMTIuOTJmIDoKLSAgICAgICAgICAgICAgICAgICAgKChmbG9hdCkgTWF0aC5wb3codiwgMS4wIC8gMi40KSkgKiAxLjA1NWYgLSAwLjA1NWY7Ci0gICAgICAgICAgICAgICAgZnJvbXNSR0J0bzhsUkdCX0xVVFtpXSA9IChieXRlKSBNYXRoLnJvdW5kKHYgKiAyNTUuMGYpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmcm9tc1JHQnRvOGxSR0JfTFVUOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgc2hvcnRbXSBnZXRGcm9tc1JHQnRvMTZsUkdCX0xVVCgpIHsKLSAgICAgICAgaWYgKGZyb21zUkdCdG8xNmxSR0JfTFVUID09IG51bGwpIHsKLSAgICAgICAgICAgIGZyb21zUkdCdG8xNmxSR0JfTFVUID0gbmV3IHNob3J0WzI1Nl07Ci0gICAgICAgICAgICBmbG9hdCB2OwotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCAyNTY7IGkrKykgewotICAgICAgICAgICAgICAgIHYgPSAoZmxvYXQpIGkgLyAyNTU7Ci0gICAgICAgICAgICAgICAgdiA9ICh2IDw9IDAuMDAzMTMwOGYpID8gdiAqIDEyLjkyZiA6Ci0gICAgICAgICAgICAgICAgICAgICgoZmxvYXQpIE1hdGgucG93KHYsIDEuMCAvIDIuNCkpICogMS4wNTVmIC0gMC4wNTVmOwotICAgICAgICAgICAgICAgIGZyb21zUkdCdG8xNmxSR0JfTFVUW2ldID0gKHNob3J0KSBNYXRoLnJvdW5kKHYgKiA2NTUzNS4wZik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZyb21zUkdCdG8xNmxSR0JfTFVUOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgYnl0ZVtdIGdldHNSR0JMVVQoaW50IGJpdHMpIHsKLSAgICAgICAgaWYgKGJpdHMgPCAxKSByZXR1cm4gbnVsbDsKLSAgICAgICAgaW50IGlkeCA9IGJpdHMgLTE7Ci0gICAgICAgIGlmKGZyb21zUkdCdG84c1JHQl9MVVRzID09IG51bGwpIGZyb21zUkdCdG84c1JHQl9MVVRzID0gbmV3IGJ5dGVbMTZdW107Ci0KLSAgICAgICAgaWYoZnJvbXNSR0J0bzhzUkdCX0xVVHNbaWR4XSA9PSBudWxsKXsKLSAgICAgICAgICAgIGZyb21zUkdCdG84c1JHQl9MVVRzW2lkeF0gPSBjcmVhdGVMVVQoYml0cyk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZyb21zUkdCdG84c1JHQl9MVVRzW2lkeF07Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBzdGF0aWMgYnl0ZVtdIGNyZWF0ZUxVVChpbnQgYml0cykgewotICAgICAgICBpbnQgbHV0U2l6ZSA9ICgxIDw8IGJpdHMpOwotICAgICAgICBieXRlIGx1dFtdID0gbmV3IGJ5dGVbbHV0U2l6ZV07Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbHV0U2l6ZTsgaSsrKSB7Ci0gICAgICAgICAgICBsdXRbaV0gPSAoYnl0ZSkgKDI1NS4wZiAvIChsdXRTaXplIC0gMSkgKyAwLjVmKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbHV0OwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgYm9vbGVhbiBpc19MSU5FQVJfUkdCX0NTKENvbG9yU3BhY2UgY3MpIHsKLSAgICAgICAgcmV0dXJuIChjcyA9PSBMSU5FQVJfUkdCX0NTKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIGJvb2xlYW4gaXNfTElORUFSX0dSQVlfQ1MoQ29sb3JTcGFjZSBjcykgewotICAgICAgICByZXR1cm4gKGNzID09IExJTkVBUl9HUkFZX0NTKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3RhdGljIGJvb2xlYW4gaXNfc1JHQl9DUyhDb2xvclNwYWNlIGNzKSB7Ci0gICAgICAgIHJldHVybiAoY3MgPT0gc1JHQl9DUyk7Ci0gICAgfQotCi19ClwgTm8gbmV3bGluZSBhdCBlbmQgb2YgZmlsZQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvY29sb3IvTmF0aXZlQ01NLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9jb2xvci9OYXRpdmVDTU0uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggN2Y4YzdlNi4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9jb2xvci9OYXRpdmVDTU0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDkyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuY29sb3I7Ci0KLWltcG9ydCBqYXZhLmF3dC5jb2xvci5JQ0NfUHJvZmlsZTsKLWltcG9ydCBqYXZhLnNlY3VyaXR5LkFjY2Vzc0NvbnRyb2xsZXI7Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5Qcml2aWxlZ2VkQWN0aW9uOwotaW1wb3J0IGphdmEudXRpbC5IYXNoTWFwOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgYSB3cmFwcGVyIGZvciB0aGUgbmF0aXZlIENNTSBsaWJyYXJ5Ci0gKi8KLXB1YmxpYyBjbGFzcyBOYXRpdmVDTU0gewotCi0gICAgLyoqCi0gICAgICogU3RvcmFnZSBmb3IgcHJvZmlsZSBoYW5kbGVzLCBzaW5jZSB0aGV5IGFyZSBwcml2YXRlCi0gICAgICogaW4gSUNDX1Byb2ZpbGUsIGJ1dCB3ZSBuZWVkIGFjY2VzcyB0byB0aGVtLgotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIEhhc2hNYXA8SUNDX1Byb2ZpbGUsIExvbmc+IHByb2ZpbGVIYW5kbGVzID0gbmV3IEhhc2hNYXA8SUNDX1Byb2ZpbGUsIExvbmc+KCk7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBib29sZWFuIGlzQ01NTG9hZGVkOwotCi0gICAgcHVibGljIHN0YXRpYyB2b2lkIGFkZEhhbmRsZShJQ0NfUHJvZmlsZSBrZXksIGxvbmcgaGFuZGxlKSB7Ci0gICAgICAgIHByb2ZpbGVIYW5kbGVzLnB1dChrZXksIG5ldyBMb25nKGhhbmRsZSkpOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgdm9pZCByZW1vdmVIYW5kbGUoSUNDX1Byb2ZpbGUga2V5KSB7Ci0gICAgICAgIHByb2ZpbGVIYW5kbGVzLnJlbW92ZShrZXkpOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgbG9uZyBnZXRIYW5kbGUoSUNDX1Byb2ZpbGUga2V5KSB7Ci0gICAgICAgIHJldHVybiBwcm9maWxlSGFuZGxlcy5nZXQoa2V5KS5sb25nVmFsdWUoKTsKLSAgICB9Ci0KLSAgICAvKiBJQ0MgcHJvZmlsZSBtYW5hZ2VtZW50ICovCi0gICAgcHVibGljIHN0YXRpYyBuYXRpdmUgbG9uZyBjbW1PcGVuUHJvZmlsZShieXRlW10gZGF0YSk7Ci0gICAgcHVibGljIHN0YXRpYyBuYXRpdmUgdm9pZCBjbW1DbG9zZVByb2ZpbGUobG9uZyBwcm9maWxlSUQpOwotICAgIHB1YmxpYyBzdGF0aWMgbmF0aXZlIGludCBjbW1HZXRQcm9maWxlU2l6ZShsb25nIHByb2ZpbGVJRCk7Ci0gICAgcHVibGljIHN0YXRpYyBuYXRpdmUgdm9pZCBjbW1HZXRQcm9maWxlKGxvbmcgcHJvZmlsZUlELCBieXRlW10gZGF0YSk7Ci0gICAgcHVibGljIHN0YXRpYyBuYXRpdmUgaW50IGNtbUdldFByb2ZpbGVFbGVtZW50U2l6ZShsb25nIHByb2ZpbGVJRCwgaW50IHNpZ25hdHVyZSk7Ci0gICAgcHVibGljIHN0YXRpYyBuYXRpdmUgdm9pZCBjbW1HZXRQcm9maWxlRWxlbWVudChsb25nIHByb2ZpbGVJRCwgaW50IHNpZ25hdHVyZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBieXRlW10gZGF0YSk7Ci0gICAgcHVibGljIHN0YXRpYyBuYXRpdmUgdm9pZCBjbW1TZXRQcm9maWxlRWxlbWVudChsb25nIHByb2ZpbGVJRCwgaW50IHRhZ1NpZ25hdHVyZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBieXRlW10gZGF0YSk7Ci0KLQotICAgIC8qIElDQyB0cmFuc2Zvcm1zICovCi0gICAgcHVibGljIHN0YXRpYyBuYXRpdmUgbG9uZyBjbW1DcmVhdGVNdWx0aXByb2ZpbGVUcmFuc2Zvcm0oCi0gICAgICAgICAgICBsb25nW10gcHJvZmlsZUhhbmRsZXMsCi0gICAgICAgICAgICBpbnRbXSByZW5kZXJpbmdJbnRlbnRzCi0gICAgICAgICk7Ci0gICAgcHVibGljIHN0YXRpYyBuYXRpdmUgdm9pZCBjbW1EZWxldGVUcmFuc2Zvcm0obG9uZyB0cmFuc2Zvcm1IYW5kbGUpOwotICAgIHB1YmxpYyBzdGF0aWMgbmF0aXZlIHZvaWQgY21tVHJhbnNsYXRlQ29sb3JzKGxvbmcgdHJhbnNmb3JtSGFuZGxlLAotICAgICAgICAgICAgTmF0aXZlSW1hZ2VGb3JtYXQgc3JjLAotICAgICAgICAgICAgTmF0aXZlSW1hZ2VGb3JtYXQgZGVzdCk7Ci0KLSAgICBzdGF0aWMgdm9pZCBsb2FkQ01NKCkgewotICAgICAgICBpZiAoIWlzQ01NTG9hZGVkKSB7Ci0gICAgICAgICAgICBBY2Nlc3NDb250cm9sbGVyLmRvUHJpdmlsZWdlZCgKLSAgICAgICAgICAgICAgICAgIG5ldyBQcml2aWxlZ2VkQWN0aW9uPFZvaWQ+KCkgewotICAgICAgICAgICAgICAgICAgICBwdWJsaWMgVm9pZCBydW4oKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBTeXN0ZW0ubG9hZExpYnJhcnkoImxjbW0iKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gKTsKLSAgICAgICAgICAgIGlzQ01NTG9hZGVkID0gdHJ1ZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qIGxvYWQgbmF0aXZlIENNTSBsaWJyYXJ5ICovCi0gICAgc3RhdGljIHsKLSAgICAgICAgbG9hZENNTSgpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2NvbG9yL05hdGl2ZUltYWdlRm9ybWF0LmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9jb2xvci9OYXRpdmVJbWFnZUZvcm1hdC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5NTk0MDQ3Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2NvbG9yL05hdGl2ZUltYWdlRm9ybWF0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2NDIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5jb2xvcjsKLQotaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db21wb25lbnRTYW1wbGVNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EYXRhQnVmZmVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJhc3RlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5TYW1wbGVNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5TaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsOwotaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLkF3dEltYWdlQmFja2Rvb3JBY2Nlc3NvcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotCi0vKioKLSAqIFRoaXMgY2xhc3MgY29udmVydHMgamF2YSBjb2xvci9zYW1wbGUgbW9kZWxzIHRvIHRoZSBMQ01TIHBpeGVsIGZvcm1hdHMuCi0gKiBJdCBhbHNvIGVuY2Fwc3VsYXRlcyBhbGwgdGhlIGluZm9ybWF0aW9uIGFib3V0IHRoZSBpbWFnZSBmb3JtYXQsIHdoaWNoIG5hdGl2ZSBDTU0KLSAqIG5lZWRzIHRvIGhhdmUgaW4gb3JkZXIgdG8gcmVhZC93cml0ZSBkYXRhLgotICoKLSAqIEF0IHByZXNlbnQgcGxhbmFyIGZvcm1hdHMgKG11bHRpcGxlIGJhbmRzKSBhcmUgbm90IHN1cHBvcnRlZAotICogYW5kIHRoZXkgYXJlIGhhbmRsZWQgYXMgYSBjb21tb24gKGN1c3RvbSkgY2FzZS4KLSAqIFNhbXBsZXMgb3RoZXIgdGhhbiAxIC0gNyBieXRlcyBhbmQgbXVsdGlwbGUgb2YgOCBiaXRzIGFyZQotICogYWxzbyBoYW5kbGVkIGFzIGN1c3RvbSAoYW5kIHdvbid0IGJlIHN1cHBvcnRlZCBpbiB0aGUgbmVhcmVzdCBmdXR1cmUpLgotICovCi1jbGFzcyBOYXRpdmVJbWFnZUZvcm1hdCB7Ci0gICAgLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLwotICAgIC8vICBMQ01TIFBpeGVsIHR5cGVzCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFBUX0FOWSA9IDA7ICAgIC8vIERvbid0IGNoZWNrIGNvbG9yc3BhY2UKLSAgICAvLyAxICYgMiBhcmUgcmVzZXJ2ZWQKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgUFRfR1JBWSAgICAgPSAzOwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBQVF9SR0IgICAgICA9IDQ7Ci0gICAgLy8gU2tpcHBpbmcgb3RoZXIgc2luY2Ugd2UgZG9uJ3QgdXNlIHRoZW0gaGVyZQotICAgIC8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vCi0KLSAgICAvLyBDb252ZXJzaW9uIG9mIHByZWRlZmluZWQgQnVmZmVyZWRJbWFnZSBmb3JtYXRzIHRvIExDTVMgZm9ybWF0cwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBJTlRfUkdCX0xDTVNfRk1UID0KLSAgICAgICAgY29sb3JzcGFjZVNoKFBUX1JHQil8Ci0gICAgICAgIGV4dHJhU2goMSl8Ci0gICAgICAgIGNoYW5uZWxzU2goMyl8Ci0gICAgICAgIGJ5dGVzU2goMSl8Ci0gICAgICAgIGRvc3dhcFNoKDEpfAotICAgICAgICBzd2FwZmlyc3RTaCgxKTsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBJTlRfQVJHQl9MQ01TX0ZNVCA9IElOVF9SR0JfTENNU19GTVQ7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgSU5UX0JHUl9MQ01TX0ZNVCA9Ci0gICAgICAgIGNvbG9yc3BhY2VTaChQVF9SR0IpfAotICAgICAgICBleHRyYVNoKDEpfAotICAgICAgICBjaGFubmVsc1NoKDMpfAotICAgICAgICBieXRlc1NoKDEpOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFRIUkVFX0JZVEVfQkdSX0xDTVNfRk1UID0KLSAgICAgICAgY29sb3JzcGFjZVNoKFBUX1JHQil8Ci0gICAgICAgIGNoYW5uZWxzU2goMyl8Ci0gICAgICAgIGJ5dGVzU2goMSl8Ci0gICAgICAgIGRvc3dhcFNoKDEpOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEZPVVJfQllURV9BQkdSX0xDTVNfRk1UID0KLSAgICAgICAgY29sb3JzcGFjZVNoKFBUX1JHQil8Ci0gICAgICAgIGV4dHJhU2goMSl8Ci0gICAgICAgIGNoYW5uZWxzU2goMyl8Ci0gICAgICAgIGJ5dGVzU2goMSl8Ci0gICAgICAgIGRvc3dhcFNoKDEpOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJZVEVfR1JBWV9MQ01TX0ZNVCA9Ci0gICAgICAgIGNvbG9yc3BhY2VTaChQVF9HUkFZKXwKLSAgICAgICAgY2hhbm5lbHNTaCgxKXwKLSAgICAgICAgYnl0ZXNTaCgxKTsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBVU0hPUlRfR1JBWV9MQ01TX0ZNVCA9Ci0gICAgICAgIGNvbG9yc3BhY2VTaChQVF9HUkFZKXwKLSAgICAgICAgY2hhbm5lbHNTaCgxKXwKLSAgICAgICAgYnl0ZXNTaCgyKTsKLQotICAgIC8vIExDTVMgZm9ybWF0IHBhY2tlZCBpbnRvIDMyIGJpdCB2YWx1ZS4gRm9yIGRlc2NyaXB0aW9uCi0gICAgLy8gb2YgdGhpcyBmb3JtYXQgcmVmZXIgdG8gTENNUyBkb2N1bWVudGF0aW9uLgotICAgIHByaXZhdGUgaW50IGNtbUZvcm1hdCA9IDA7Ci0KLSAgICAvLyBEaW1lbnNpb25zCi0gICAgcHJpdmF0ZSBpbnQgcm93cyA9IDA7Ci0gICAgcHJpdmF0ZSBpbnQgY29scyA9IDA7Ci0KLSAgICAvLyAgU2NhbmxpbmUgbWF5IGNvbnRhaW4gc29tZSBwYWRkaW5nIGluIHRoZSBlbmQKLSAgICBwcml2YXRlIGludCBzY2FubGluZVN0cmlkZSA9IC0xOwotCi0gICAgcHJpdmF0ZSBPYmplY3QgaW1hZ2VEYXRhOwotICAgIC8vIEl0J3MgcG9zc2libGUgdG8gaGF2ZSBvZmZzZXQgZnJvbSB0aGUgYmVnaW5uaW5nIG9mIHRoZSBhcnJheQotICAgIHByaXZhdGUgaW50IGRhdGFPZmZzZXQ7Ci0KLSAgICAvLyBIYXMgdGhlIGltYWdlIGFscGhhIGNoYW5uZWw/IElmIGhhcyAtIGhlcmUgaXRzIGJhbmQgYmFuZCBvZmZzZXQgZ29lcwotICAgIHByaXZhdGUgaW50IGFscGhhT2Zmc2V0ID0gLTE7Ci0KLSAgICAvLyBpbml0aWFsaXplcyBwcm9wZXIgZmllbGQgSURzCi0gICAgcHJpdmF0ZSBzdGF0aWMgbmF0aXZlIHZvaWQgaW5pdElEcygpOwotCi0gICAgc3RhdGljIHsKLSAgICAgICAgTmF0aXZlQ01NLmxvYWRDTU0oKTsKLSAgICAgICAgaW5pdElEcygpOwotICAgIH0KLQotICAgIC8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLwotICAgIC8vIExDTVMgaW1hZ2UgZm9ybWF0IGVuY29kZXJzCi0gICAgLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vCi0gICAgcHJpdmF0ZSBzdGF0aWMgaW50IGNvbG9yc3BhY2VTaChpbnQgcykgewotICAgICAgICByZXR1cm4gKHMgPDwgMTYpOwotICAgIH0KLQotICAgIHByaXZhdGUgc3RhdGljIGludCBzd2FwZmlyc3RTaChpbnQgcykgewotICAgICAgICByZXR1cm4gKHMgPDwgMTQpOwotICAgIH0KLQotICAgIHByaXZhdGUgc3RhdGljIGludCBmbGF2b3JTaChpbnQgcykgewotICAgICAgICByZXR1cm4gKHMgPDwgMTMpOwotICAgIH0KLQotICAgIHByaXZhdGUgc3RhdGljIGludCBwbGFuYXJTaChpbnQgcykgewotICAgICAgICByZXR1cm4gKHMgPDwgMTIpOwotICAgIH0KLQotICAgIHByaXZhdGUgc3RhdGljIGludCBlbmRpYW5TaChpbnQgcykgewotICAgICAgICByZXR1cm4gKHMgPDwgMTEpOwotICAgIH0KLQotICAgIHByaXZhdGUgc3RhdGljIGludCBkb3N3YXBTaChpbnQgcykgewotICAgICAgICByZXR1cm4gKHMgPDwgMTApOwotICAgIH0KLQotICAgIHByaXZhdGUgc3RhdGljIGludCBleHRyYVNoKGludCBzKSB7Ci0gICAgICAgIHJldHVybiAocyA8PCA3KTsKLSAgICB9Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBpbnQgY2hhbm5lbHNTaChpbnQgcykgewotICAgICAgICByZXR1cm4gKHMgPDwgMyk7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBzdGF0aWMgaW50IGJ5dGVzU2goaW50IHMpIHsKLSAgICAgICAgcmV0dXJuIHM7Ci0gICAgfQotICAgIC8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLwotICAgIC8vIEVuZCBvZiBMQ01TIGltYWdlIGZvcm1hdCBlbmNvZGVycwotICAgIC8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLwotCi0gICAgLy8gQWNjZXNzb3JzCi0gICAgT2JqZWN0IGdldENoYW5uZWxEYXRhKCkgewotICAgICAgICByZXR1cm4gaW1hZ2VEYXRhOwotICAgIH0KLQotICAgIGludCBnZXROdW1Db2xzKCkgewotICAgICAgICByZXR1cm4gY29sczsKLSAgICB9Ci0KLSAgICBpbnQgZ2V0TnVtUm93cygpIHsKLSAgICAgICAgcmV0dXJuIHJvd3M7Ci0gICAgfQotCi0gICAgLy8gQ29uc3RydWN0b3JzCi0gICAgcHVibGljIE5hdGl2ZUltYWdlRm9ybWF0KCkgewotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNpbXBsZSBpbWFnZSBsYXlvdXQgZm9yIGNvbW1vbiBjYXNlIHdpdGgKLSAgICAgKiBub3Qgb3B0aW1pemVkIHdvcmtmbG93LgotICAgICAqCi0gICAgICogRm9yIGhpZmkgY29sb3JzcGFjZXMgd2l0aCA1KyBjb2xvciBjaGFubmVscyBpbWdEYXRhCi0gICAgICogc2hvdWxkIGJlIDxjb2RlPmJ5dGU8L2NvZGU+IGFycmF5LgotICAgICAqCi0gICAgICogRm9yIGNvbW1vbiBjb2xvcnNwYWNlcyB3aXRoIHVwIHRvIDQgY29sb3IgY2hhbm5lbHMgaXQKLSAgICAgKiBzaG91bGQgYmUgPGNvZGU+c2hvcnQ8L2NvZGU+IGFycmF5LgotICAgICAqCi0gICAgICogQWxwaGEgY2hhbm5lbCBpcyBoYW5kbGVkIGJ5IGNhbGxlciwgbm90IGJ5IENNUy4KLSAgICAgKgotICAgICAqIENvbG9yIGNoYW5uZWxzIGFyZSBpbiB0aGVpciBuYXR1cmFsIG9yZGVyIChub3QgQkdSIGJ1dCBSR0IpLgotICAgICAqCi0gICAgICogQHBhcmFtIGltZ0RhdGEgLSBhcnJheSBvZiA8Y29kZT5ieXRlPC9jb2RlPiBvciA8Y29kZT5zaG9ydDwvY29kZT4KLSAgICAgKiBAcGFyYW0gbkNoYW5uZWxzIC0gbnVtYmVyIG9mIGNoYW5uZWxzCi0gICAgICogQHBhcmFtIG5Sb3dzIC0gbnVtYmVyIG9mIHNjYW5saW5lcyBpbiB0aGUgaW1hZ2UKLSAgICAgKiBAcGFyYW0gbkNvbHMgLSBudW1iZXIgb2YgcGl4ZWxzIGluIG9uZSByb3cgb2YgdGhlIGltYWdlCi0gICAgICovCi0gICAgcHVibGljIE5hdGl2ZUltYWdlRm9ybWF0KE9iamVjdCBpbWdEYXRhLCBpbnQgbkNoYW5uZWxzLCBpbnQgblJvd3MsIGludCBuQ29scykgewotICAgICAgICBpZiAoaW1nRGF0YSBpbnN0YW5jZW9mIHNob3J0W10pIHsKLSAgICAgICAgICAgIGNtbUZvcm1hdCB8PSBieXRlc1NoKDIpOwotICAgICAgICB9Ci0gICAgICAgIGVsc2UgaWYgKGltZ0RhdGEgaW5zdGFuY2VvZiBieXRlW10pIHsKLSAgICAgICAgICAgIGNtbUZvcm1hdCB8PSBieXRlc1NoKDEpOwotICAgICAgICB9Ci0gICAgICAgIGVsc2UKLSAgICAgICAgICAgIC8vIGF3dC40Nz1GaXJzdCBhcmd1bWVudCBzaG91bGQgYmUgYnl0ZSBvciBzaG9ydCBhcnJheQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40NyIpKTsgLy8kTk9OLU5MUy0xJAotCi0gICAgICAgIGNtbUZvcm1hdCB8PSBjaGFubmVsc1NoKG5DaGFubmVscyk7Ci0KLSAgICAgICAgcm93cyA9IG5Sb3dzOwotICAgICAgICBjb2xzID0gbkNvbHM7Ci0KLSAgICAgICAgaW1hZ2VEYXRhID0gaW1nRGF0YTsKLQotICAgICAgICBkYXRhT2Zmc2V0ID0gMDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEZWR1Y2VzIGltYWdlIGZvcm1hdCBmcm9tIHRoZSBidWZmZXJlZCBpbWFnZSB0eXBlCi0gICAgICogb3IgY29sb3IgYW5kIHNhbXBsZSBtb2RlbHMuCi0gICAgICogQHBhcmFtIGJpIC0gaW1hZ2UKLSAgICAgKiBAcmV0dXJuIGltYWdlIGZvcm1hdCBvYmplY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIE5hdGl2ZUltYWdlRm9ybWF0IGNyZWF0ZU5hdGl2ZUltYWdlRm9ybWF0KEJ1ZmZlcmVkSW1hZ2UgYmkpIHsKLSAgICAgICAgTmF0aXZlSW1hZ2VGb3JtYXQgZm10ID0gbmV3IE5hdGl2ZUltYWdlRm9ybWF0KCk7Ci0KLSAgICAgICAgc3dpdGNoIChiaS5nZXRUeXBlKCkpIHsKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX0lOVF9SR0I6IHsKLSAgICAgICAgICAgICAgICBmbXQuY21tRm9ybWF0ID0gSU5UX1JHQl9MQ01TX0ZNVDsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0FSR0I6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9JTlRfQVJHQl9QUkU6IHsKLSAgICAgICAgICAgICAgICBmbXQuY21tRm9ybWF0ID0gSU5UX0FSR0JfTENNU19GTVQ7Ci0gICAgICAgICAgICAgICAgZm10LmFscGhhT2Zmc2V0ID0gMzsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX0JHUjogewotICAgICAgICAgICAgICAgIGZtdC5jbW1Gb3JtYXQgPSBJTlRfQkdSX0xDTVNfRk1UOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV8zQllURV9CR1I6IHsKLSAgICAgICAgICAgICAgICBmbXQuY21tRm9ybWF0ID0gVEhSRUVfQllURV9CR1JfTENNU19GTVQ7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFXzRCWVRFX0FCR1JfUFJFOgotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfNEJZVEVfQUJHUjogewotICAgICAgICAgICAgICAgIGZtdC5jbW1Gb3JtYXQgPSBGT1VSX0JZVEVfQUJHUl9MQ01TX0ZNVDsKLSAgICAgICAgICAgICAgICBmbXQuYWxwaGFPZmZzZXQgPSAwOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9CWVRFX0dSQVk6IHsKLSAgICAgICAgICAgICAgICBmbXQuY21tRm9ybWF0ID0gQllURV9HUkFZX0xDTVNfRk1UOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9VU0hPUlRfR1JBWTogewotICAgICAgICAgICAgICAgIGZtdC5jbW1Gb3JtYXQgPSBVU0hPUlRfR1JBWV9MQ01TX0ZNVDsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfQllURV9CSU5BUlk6Ci0gICAgICAgICAgICBjYXNlIEJ1ZmZlcmVkSW1hZ2UuVFlQRV9VU0hPUlRfNTY1X1JHQjoKLSAgICAgICAgICAgIGNhc2UgQnVmZmVyZWRJbWFnZS5UWVBFX1VTSE9SVF81NTVfUkdCOgotICAgICAgICAgICAgY2FzZSBCdWZmZXJlZEltYWdlLlRZUEVfQllURV9JTkRFWEVEOiB7Ci0gICAgICAgICAgICAgICAgLy8gQSBidW5jaCBvZiB1bnN1cHBvcnRlZCBmb3JtYXRzCi0gICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgYnJlYWs7IC8vIFRyeSB0byBsb29rIGF0IHNhbXBsZSBtb2RlbCBhbmQgY29sb3IgbW9kZWwKLSAgICAgICAgfQotCi0KLSAgICAgICAgaWYgKGZtdC5jbW1Gb3JtYXQgPT0gMCkgewotICAgICAgICAgICAgQ29sb3JNb2RlbCBjbSA9IGJpLmdldENvbG9yTW9kZWwoKTsKLSAgICAgICAgICAgIFNhbXBsZU1vZGVsIHNtID0gYmkuZ2V0U2FtcGxlTW9kZWwoKTsKLQotICAgICAgICAgICAgaWYgKHNtIGluc3RhbmNlb2YgQ29tcG9uZW50U2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgICAgICBDb21wb25lbnRTYW1wbGVNb2RlbCBjc20gPSAoQ29tcG9uZW50U2FtcGxlTW9kZWwpIHNtOwotICAgICAgICAgICAgICAgIGZtdC5jbW1Gb3JtYXQgPSBnZXRGb3JtYXRGcm9tQ29tcG9uZW50TW9kZWwoY3NtLCBjbS5oYXNBbHBoYSgpKTsKLSAgICAgICAgICAgICAgICBmbXQuc2NhbmxpbmVTdHJpZGUgPSBjYWxjdWxhdGVTY2FubGluZVN0cmlkZUNTTShjc20sIGJpLmdldFJhc3RlcigpKTsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoc20gaW5zdGFuY2VvZiBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKSB7Ci0gICAgICAgICAgICAgICAgU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBzcHBzbSA9IChTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKSBzbTsKLSAgICAgICAgICAgICAgICBmbXQuY21tRm9ybWF0ID0gZ2V0Rm9ybWF0RnJvbVNQUFNhbXBsZU1vZGVsKHNwcHNtLCBjbS5oYXNBbHBoYSgpKTsKLSAgICAgICAgICAgICAgICBmbXQuc2NhbmxpbmVTdHJpZGUgPSBjYWxjdWxhdGVTY2FubGluZVN0cmlkZVNQUFNNKHNwcHNtLCBiaS5nZXRSYXN0ZXIoKSk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChjbS5oYXNBbHBoYSgpKQotICAgICAgICAgICAgICAgIGZtdC5hbHBoYU9mZnNldCA9IGNhbGN1bGF0ZUFscGhhT2Zmc2V0KHNtLCBiaS5nZXRSYXN0ZXIoKSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZm10LmNtbUZvcm1hdCA9PSAwKQotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0KLSAgICAgICAgaWYgKCFmbXQuc2V0SW1hZ2VEYXRhKGJpLmdldFJhc3RlcigpLmdldERhdGFCdWZmZXIoKSkpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgZm10LnJvd3MgPSBiaS5nZXRIZWlnaHQoKTsKLSAgICAgICAgZm10LmNvbHMgPSBiaS5nZXRXaWR0aCgpOwotCi0gICAgICAgIGZtdC5kYXRhT2Zmc2V0ID0gYmkuZ2V0UmFzdGVyKCkuZ2V0RGF0YUJ1ZmZlcigpLmdldE9mZnNldCgpOwotCi0gICAgICAgIHJldHVybiBmbXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGVkdWNlcyBpbWFnZSBmb3JtYXQgZnJvbSB0aGUgcmFzdGVyIHNhbXBsZSBtb2RlbC4KLSAgICAgKiBAcGFyYW0gciAtIHJhc3RlcgotICAgICAqIEByZXR1cm4gaW1hZ2UgZm9ybWF0IG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgTmF0aXZlSW1hZ2VGb3JtYXQgY3JlYXRlTmF0aXZlSW1hZ2VGb3JtYXQoUmFzdGVyIHIpIHsKLSAgICAgICAgTmF0aXZlSW1hZ2VGb3JtYXQgZm10ID0gbmV3IE5hdGl2ZUltYWdlRm9ybWF0KCk7Ci0gICAgICAgIFNhbXBsZU1vZGVsIHNtID0gci5nZXRTYW1wbGVNb2RlbCgpOwotCi0gICAgICAgIC8vIEFzc3VtZSB0aGF0IHRoZXJlJ3Mgbm8gYWxwaGEKLSAgICAgICAgaWYgKHNtIGluc3RhbmNlb2YgQ29tcG9uZW50U2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgIENvbXBvbmVudFNhbXBsZU1vZGVsIGNzbSA9IChDb21wb25lbnRTYW1wbGVNb2RlbCkgc207Ci0gICAgICAgICAgICBmbXQuY21tRm9ybWF0ID0gZ2V0Rm9ybWF0RnJvbUNvbXBvbmVudE1vZGVsKGNzbSwgZmFsc2UpOwotICAgICAgICAgICAgZm10LnNjYW5saW5lU3RyaWRlID0gY2FsY3VsYXRlU2NhbmxpbmVTdHJpZGVDU00oY3NtLCByKTsKLSAgICAgICAgfSBlbHNlIGlmIChzbSBpbnN0YW5jZW9mIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc3Bwc20gPSAoU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkgc207Ci0gICAgICAgICAgICBmbXQuY21tRm9ybWF0ID0gZ2V0Rm9ybWF0RnJvbVNQUFNhbXBsZU1vZGVsKHNwcHNtLCBmYWxzZSk7Ci0gICAgICAgICAgICBmbXQuc2NhbmxpbmVTdHJpZGUgPSBjYWxjdWxhdGVTY2FubGluZVN0cmlkZVNQUFNNKHNwcHNtLCByKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChmbXQuY21tRm9ybWF0ID09IDApCi0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLQotICAgICAgICBmbXQuY29scyA9IHIuZ2V0V2lkdGgoKTsKLSAgICAgICAgZm10LnJvd3MgPSByLmdldEhlaWdodCgpOwotICAgICAgICBmbXQuZGF0YU9mZnNldCA9IHIuZ2V0RGF0YUJ1ZmZlcigpLmdldE9mZnNldCgpOwotCi0gICAgICAgIGlmICghZm10LnNldEltYWdlRGF0YShyLmdldERhdGFCdWZmZXIoKSkpCi0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLQotICAgICAgICByZXR1cm4gZm10OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIE9idGFpbnMgTENNUyBmb3JtYXQgZnJvbSB0aGUgY29tcG9uZW50IHNhbXBsZSBtb2RlbAotICAgICAqIEBwYXJhbSBzbSAtIHNhbXBsZSBtb2RlbAotICAgICAqIEBwYXJhbSBoYXNBbHBoYSAtIHRydWUgaWYgdGhlcmUncyBhbiBhbHBoYSBjaGFubmVsCi0gICAgICogQHJldHVybiBMQ01TIGZvcm1hdAotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGludCBnZXRGb3JtYXRGcm9tQ29tcG9uZW50TW9kZWwoQ29tcG9uZW50U2FtcGxlTW9kZWwgc20sIGJvb2xlYW4gaGFzQWxwaGEpIHsKLSAgICAgICAgLy8gTXVsdGlwbGUgZGF0YSBhcnJheXMgKGJhbmtzKSBub3Qgc3VwcG9ydGVkCi0gICAgICAgIGludCBiYW5rSW5kZXggPSBzbS5nZXRCYW5rSW5kaWNlcygpWzBdOwotICAgICAgICBmb3IgKGludCBpPTE7IGkgPCBzbS5nZXROdW1CYW5kcygpOyBpKyspIHsKLSAgICAgICAgICAgIGlmIChzbS5nZXRCYW5rSW5kaWNlcygpW2ldICE9IGJhbmtJbmRleCkgewotICAgICAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGNoYW5uZWxzID0gaGFzQWxwaGEgPyBzbS5nZXROdW1CYW5kcygpLTEgOiBzbS5nZXROdW1CYW5kcygpOwotICAgICAgICBpbnQgZXh0cmEgPSBoYXNBbHBoYSA/IDEgOiAwOwotICAgICAgICBpbnQgYnl0ZXMgPSAxOwotICAgICAgICBzd2l0Y2ggKHNtLmdldERhdGFUeXBlKCkpIHsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0JZVEU6Ci0gICAgICAgICAgICAgICAgYnl0ZXMgPSAxOyBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX1NIT1JUOgotICAgICAgICAgICAgY2FzZSBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUOgotICAgICAgICAgICAgICAgIGJ5dGVzID0gMjsgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIERhdGFCdWZmZXIuVFlQRV9JTlQ6Ci0gICAgICAgICAgICAgICAgYnl0ZXMgPSA0OyBicmVhazsKLSAgICAgICAgICAgIGNhc2UgRGF0YUJ1ZmZlci5UWVBFX0RPVUJMRToKLSAgICAgICAgICAgICAgICBieXRlcyA9IDA7IGJyZWFrOwotICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICByZXR1cm4gMDsgLy8gVW5zdXBwb3J0ZWQgZGF0YSB0eXBlCi0gICAgICAgIH0KLQotICAgICAgICBpbnQgZG9Td2FwID0gMDsKLSAgICAgICAgaW50IHN3YXBGaXJzdCA9IDA7Ci0gICAgICAgIGJvb2xlYW4ga25vd25Gb3JtYXQgPSBmYWxzZTsKLQotICAgICAgICBpbnQgaTsKLQotICAgICAgICAvLyAiUkdCQSIKLSAgICAgICAgZm9yIChpPTA7IGkgPCBzbS5nZXROdW1CYW5kcygpOyBpKyspIHsKLSAgICAgICAgICAgIGlmIChzbS5nZXRCYW5kT2Zmc2V0cygpW2ldICE9IGkpIGJyZWFrOwotICAgICAgICB9Ci0gICAgICAgIGlmIChpID09IHNtLmdldE51bUJhbmRzKCkpIHsgLy8gT2ssIGl0IGlzIGl0Ci0gICAgICAgICAgICBkb1N3YXAgPSAwOwotICAgICAgICAgICAgc3dhcEZpcnN0ID0gMDsKLSAgICAgICAgICAgIGtub3duRm9ybWF0ID0gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vICJBUkdCIgotICAgICAgICBpZiAoIWtub3duRm9ybWF0KSB7Ci0gICAgICAgICAgICBmb3IgKGk9MDsgaSA8IHNtLmdldE51bUJhbmRzKCktMTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHNtLmdldEJhbmRPZmZzZXRzKClbaV0gIT0gaSsxKSBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChzbS5nZXRCYW5kT2Zmc2V0cygpW2ldID09IDApIGkrKzsKLSAgICAgICAgICAgIGlmIChpID09IHNtLmdldE51bUJhbmRzKCkpIHsgLy8gT2ssIGl0IGlzIGl0Ci0gICAgICAgICAgICAgICAgZG9Td2FwID0gMDsKLSAgICAgICAgICAgICAgICBzd2FwRmlyc3QgPSAxOwotICAgICAgICAgICAgICAgIGtub3duRm9ybWF0ID0gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vICJCR1JBIgotICAgICAgICBpZiAoIWtub3duRm9ybWF0KSB7Ci0gICAgICAgICAgICBmb3IgKGk9MDsgaSA8IHNtLmdldE51bUJhbmRzKCktMTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHNtLmdldEJhbmRPZmZzZXRzKClbaV0gIT0gc20uZ2V0TnVtQmFuZHMoKSAtIDIgLSBpKSBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChzbS5nZXRCYW5kT2Zmc2V0cygpW2ldID09IHNtLmdldE51bUJhbmRzKCktMSkgaSsrOwotICAgICAgICAgICAgaWYgKGkgPT0gc20uZ2V0TnVtQmFuZHMoKSkgeyAvLyBPaywgaXQgaXMgaXQKLSAgICAgICAgICAgICAgICBkb1N3YXAgPSAxOwotICAgICAgICAgICAgICAgIHN3YXBGaXJzdCA9IDE7Ci0gICAgICAgICAgICAgICAga25vd25Gb3JtYXQgPSB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLy8gIkFCR1IiCi0gICAgICAgIGlmICgha25vd25Gb3JtYXQpIHsKLSAgICAgICAgICAgIGZvciAoaT0wOyBpIDwgc20uZ2V0TnVtQmFuZHMoKTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHNtLmdldEJhbmRPZmZzZXRzKClbaV0gIT0gc20uZ2V0TnVtQmFuZHMoKSAtIDEgLSBpKSBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChpID09IHNtLmdldE51bUJhbmRzKCkpIHsgLy8gT2ssIGl0IGlzIGl0Ci0gICAgICAgICAgICAgICAgZG9Td2FwID0gMTsKLSAgICAgICAgICAgICAgICBzd2FwRmlyc3QgPSAwOwotICAgICAgICAgICAgICAgIGtub3duRm9ybWF0ID0gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIFhYWCAtIFBsYW5hciBmb3JtYXRzIGFyZSBub3Qgc3VwcG9ydGVkIHlldAotICAgICAgICBpZiAoIWtub3duRm9ybWF0KQotICAgICAgICAgICAgcmV0dXJuIDA7Ci0KLSAgICAgICAgcmV0dXJuCi0gICAgICAgICAgICBjaGFubmVsc1NoKGNoYW5uZWxzKSB8Ci0gICAgICAgICAgICBieXRlc1NoKGJ5dGVzKSB8Ci0gICAgICAgICAgICBleHRyYVNoKGV4dHJhKSB8Ci0gICAgICAgICAgICBkb3N3YXBTaChkb1N3YXApIHwKLSAgICAgICAgICAgIHN3YXBmaXJzdFNoKHN3YXBGaXJzdCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogT2J0YWlucyBMQ01TIGZvcm1hdCBmcm9tIHRoZSBzaW5nbGUgcGl4ZWwgcGFja2VkIHNhbXBsZSBtb2RlbAotICAgICAqIEBwYXJhbSBzbSAtIHNhbXBsZSBtb2RlbAotICAgICAqIEBwYXJhbSBoYXNBbHBoYSAtIHRydWUgaWYgdGhlcmUncyBhbiBhbHBoYSBjaGFubmVsCi0gICAgICogQHJldHVybiBMQ01TIGZvcm1hdAotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGludCBnZXRGb3JtYXRGcm9tU1BQU2FtcGxlTW9kZWwoU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCBzbSwKLSAgICAgICAgICAgIGJvb2xlYW4gaGFzQWxwaGEpIHsKLSAgICAgICAgLy8gQ2FuIHdlIGV4dHJhY3QgYnl0ZXM/Ci0gICAgICAgIGludCBtYXNrID0gc20uZ2V0Qml0TWFza3MoKVswXSA+Pj4gc20uZ2V0Qml0T2Zmc2V0cygpWzBdOwotICAgICAgICBpZiAoIShtYXNrID09IDB4RkYgfHwgbWFzayA9PSAweEZGRkYgfHwgbWFzayA9PSAweEZGRkZGRkZGKSkKLSAgICAgICAgICAgIHJldHVybiAwOwotCi0gICAgICAgIC8vIEFsbCBtYXNrcyBhcmUgc2FtZT8KLSAgICAgICAgZm9yIChpbnQgaSA9IDE7IGkgPCBzbS5nZXROdW1CYW5kcygpOyBpKyspIHsKLSAgICAgICAgICAgIGlmICgoc20uZ2V0Qml0TWFza3MoKVtpXSA+Pj4gc20uZ2V0Qml0T2Zmc2V0cygpW2ldKSAhPSBtYXNrKQotICAgICAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IHBpeGVsU2l6ZSA9IDA7Ci0gICAgICAgIC8vIENoZWNrIGlmIGRhdGEgdHlwZSBpcyBzdXBwb3J0ZWQKLSAgICAgICAgaWYgKHNtLmdldERhdGFUeXBlKCkgPT0gRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCkKLSAgICAgICAgICAgIHBpeGVsU2l6ZSA9IDI7Ci0gICAgICAgIGVsc2UgaWYgKHNtLmdldERhdGFUeXBlKCkgPT0gRGF0YUJ1ZmZlci5UWVBFX0lOVCkKLSAgICAgICAgICAgIHBpeGVsU2l6ZSA9IDQ7Ci0gICAgICAgIGVsc2UKLSAgICAgICAgICAgIHJldHVybiAwOwotCi0KLSAgICAgICAgaW50IGJ5dGVzID0gMDsKLSAgICAgICAgc3dpdGNoIChtYXNrKSB7Ci0gICAgICAgICAgICBjYXNlIDB4RkY6Ci0gICAgICAgICAgICAgICAgYnl0ZXMgPSAxOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSAweEZGRkY6Ci0gICAgICAgICAgICAgICAgYnl0ZXMgPSAyOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSAweEZGRkZGRkZGOgotICAgICAgICAgICAgICAgIGJ5dGVzID0gNDsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGRlZmF1bHQ6IHJldHVybiAwOwotICAgICAgICB9Ci0KLQotICAgICAgICBpbnQgY2hhbm5lbHMgPSBoYXNBbHBoYSA/IHNtLmdldE51bUJhbmRzKCktMSA6IHNtLmdldE51bUJhbmRzKCk7Ci0gICAgICAgIGludCBleHRyYSA9IGhhc0FscGhhID8gMSA6IDA7Ci0gICAgICAgIGV4dHJhICs9ICBwaXhlbFNpemUvYnl0ZXMgLSBzbS5nZXROdW1CYW5kcygpOyAvLyBVbnVzZWQgYnl0ZXM/Ci0KLSAgICAgICAgLy8gRm9ybSBhbiBBcnJheUxpc3QgY29udGFpbmluZyBvZmZzZXQgZm9yIGVhY2ggYmFuZAotICAgICAgICBBcnJheUxpc3Q8SW50ZWdlcj4gb2Zmc2V0c0xzdCA9IG5ldyBBcnJheUxpc3Q8SW50ZWdlcj4oKTsKLSAgICAgICAgZm9yIChpbnQgaz0wOyBrIDwgc20uZ2V0TnVtQmFuZHMoKTsgaysrKSB7Ci0gICAgICAgICAgICBvZmZzZXRzTHN0LmFkZChuZXcgSW50ZWdlcihzbS5nZXRCaXRPZmZzZXRzKClba10vKGJ5dGVzKjgpKSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBBZGQgb2Zmc2V0cyBmb3IgdW51c2VkIHNwYWNlCi0gICAgICAgIGZvciAoaW50IGk9MDsgaTxwaXhlbFNpemUvYnl0ZXM7IGkrKykgewotICAgICAgICAgICAgaWYgKG9mZnNldHNMc3QuaW5kZXhPZihuZXcgSW50ZWdlcihpKSkgPCAwKQotICAgICAgICAgICAgICAgIG9mZnNldHNMc3QuYWRkKG5ldyBJbnRlZ2VyKGkpKTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBvZmZzZXRzW10gPSBuZXcgaW50W3BpeGVsU2l6ZS9ieXRlc107Ci0gICAgICAgIGZvciAoaW50IGk9MDsgaTxvZmZzZXRzTHN0LnNpemUoKTsgaSsrKSB7Ci0gICAgICAgICAgICBvZmZzZXRzW2ldID0gb2Zmc2V0c0xzdC5nZXQoaSkuaW50VmFsdWUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBkb1N3YXAgPSAwOwotICAgICAgICBpbnQgc3dhcEZpcnN0ID0gMDsKLSAgICAgICAgYm9vbGVhbiBrbm93bkZvcm1hdCA9IGZhbHNlOwotCi0gICAgICAgIGludCBpOwotCi0gICAgICAgIC8vICJSR0JBIgotICAgICAgICBmb3IgKGk9MDsgaSA8IHBpeGVsU2l6ZTsgaSsrKSB7Ci0gICAgICAgICAgICBpZiAob2Zmc2V0c1tpXSAhPSBpKSBicmVhazsKLSAgICAgICAgfQotICAgICAgICBpZiAoaSA9PSBwaXhlbFNpemUpIHsgLy8gT2ssIGl0IGlzIGl0Ci0gICAgICAgICAgICBkb1N3YXAgPSAwOwotICAgICAgICAgICAgc3dhcEZpcnN0ID0gMDsKLSAgICAgICAgICAgIGtub3duRm9ybWF0ID0gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vICJBUkdCIgotICAgICAgICBpZiAoIWtub3duRm9ybWF0KSB7Ci0gICAgICAgICAgICBmb3IgKGk9MDsgaSA8IHBpeGVsU2l6ZS0xOyBpKyspIHsKLSAgICAgICAgICAgICAgICBpZiAob2Zmc2V0c1tpXSAhPSBpKzEpIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKG9mZnNldHNbaV0gPT0gMCkgaSsrOwotICAgICAgICAgICAgaWYgKGkgPT0gcGl4ZWxTaXplKSB7IC8vIE9rLCBpdCBpcyBpdAotICAgICAgICAgICAgICAgIGRvU3dhcCA9IDA7Ci0gICAgICAgICAgICAgICAgc3dhcEZpcnN0ID0gMTsKLSAgICAgICAgICAgICAgICBrbm93bkZvcm1hdCA9IHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyAiQkdSQSIKLSAgICAgICAgaWYgKCFrbm93bkZvcm1hdCkgewotICAgICAgICAgICAgZm9yIChpPTA7IGkgPCBwaXhlbFNpemUtMTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKG9mZnNldHNbaV0gIT0gcGl4ZWxTaXplIC0gMiAtIGkpIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKG9mZnNldHNbaV0gPT0gcGl4ZWxTaXplLTEpIGkrKzsKLSAgICAgICAgICAgIGlmIChpID09IHBpeGVsU2l6ZSkgeyAvLyBPaywgaXQgaXMgaXQKLSAgICAgICAgICAgICAgICBkb1N3YXAgPSAxOwotICAgICAgICAgICAgICAgIHN3YXBGaXJzdCA9IDE7Ci0gICAgICAgICAgICAgICAga25vd25Gb3JtYXQgPSB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLy8gIkFCR1IiCi0gICAgICAgIGlmICgha25vd25Gb3JtYXQpIHsKLSAgICAgICAgICAgIGZvciAoaT0wOyBpIDwgcGl4ZWxTaXplOyBpKyspIHsKLSAgICAgICAgICAgICAgICBpZiAob2Zmc2V0c1tpXSAhPSBwaXhlbFNpemUgLSAxIC0gaSkgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoaSA9PSBwaXhlbFNpemUpIHsgLy8gT2ssIGl0IGlzIGl0Ci0gICAgICAgICAgICAgICAgZG9Td2FwID0gMTsKLSAgICAgICAgICAgICAgICBzd2FwRmlyc3QgPSAwOwotICAgICAgICAgICAgICAgIGtub3duRm9ybWF0ID0gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIFhYWCAtIFBsYW5hciBmb3JtYXRzIGFyZSBub3Qgc3VwcG9ydGVkIHlldAotICAgICAgICBpZiAoIWtub3duRm9ybWF0KQotICAgICAgICAgICAgcmV0dXJuIDA7Ci0KLSAgICAgICAgcmV0dXJuCi0gICAgICAgICAgICBjaGFubmVsc1NoKGNoYW5uZWxzKSB8Ci0gICAgICAgICAgICBieXRlc1NoKGJ5dGVzKSB8Ci0gICAgICAgICAgICBleHRyYVNoKGV4dHJhKSB8Ci0gICAgICAgICAgICBkb3N3YXBTaChkb1N3YXApIHwKLSAgICAgICAgICAgIHN3YXBmaXJzdFNoKHN3YXBGaXJzdCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogT2J0YWlucyBkYXRhIGFycmF5IGZyb20gdGhlIERhdGFCdWZmZXIgb2JqZWN0Ci0gICAgICogQHBhcmFtIGRiIC0gZGF0YSBidWZmZXIKLSAgICAgKiBAcmV0dXJuIC0gdHJ1ZSBpZiBzdWNjZXNzZnVsCi0gICAgICovCi0gICAgcHJpdmF0ZSBib29sZWFuIHNldEltYWdlRGF0YShEYXRhQnVmZmVyIGRiKSB7Ci0gICAgICAgIEF3dEltYWdlQmFja2Rvb3JBY2Nlc3NvciBkYkFjY2VzcyA9IEF3dEltYWdlQmFja2Rvb3JBY2Nlc3Nvci5nZXRJbnN0YW5jZSgpOwotICAgICAgICB0cnkgewotICAgICAgICAgICAgaW1hZ2VEYXRhID0gZGJBY2Nlc3MuZ2V0RGF0YShkYik7Ci0gICAgICAgIH0gY2F0Y2ggKElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7IC8vIFVua25vd24gZGF0YSBidWZmZXIgdHlwZQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2FsY3VsYXRlcyBzY2FubGluZSBzdHJpZGUgaW4gYnl0ZXMKLSAgICAgKiBAcGFyYW0gY3NtIC0gY29tcG9uZW50IHNhbXBsZSBtb2RlbAotICAgICAqIEBwYXJhbSByIC0gcmFzdGVyCi0gICAgICogQHJldHVybiBzY2FubGluZSBzdHJpZGUgaW4gYnl0ZXMKLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBpbnQgY2FsY3VsYXRlU2NhbmxpbmVTdHJpZGVDU00oQ29tcG9uZW50U2FtcGxlTW9kZWwgY3NtLCBSYXN0ZXIgcikgewotICAgICAgICBpZiAoY3NtLmdldFNjYW5saW5lU3RyaWRlKCkgIT0gY3NtLmdldFBpeGVsU3RyaWRlKCkqY3NtLmdldFdpZHRoKCkpIHsKLSAgICAgICAgICAgIGludCBkYXRhVHlwZVNpemUgPSBEYXRhQnVmZmVyLmdldERhdGFUeXBlU2l6ZShyLmdldERhdGFCdWZmZXIoKS5nZXREYXRhVHlwZSgpKSAvIDg7Ci0gICAgICAgICAgICByZXR1cm4gY3NtLmdldFNjYW5saW5lU3RyaWRlKCkqZGF0YVR5cGVTaXplOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDYWxjdWxhdGVzIHNjYW5saW5lIHN0cmlkZSBpbiBieXRlcwotICAgICAqIEBwYXJhbSBzcHBzbSAtIHNhbXBsZSBtb2RlbAotICAgICAqIEBwYXJhbSByIC0gcmFzdGVyCi0gICAgICogQHJldHVybiBzY2FubGluZSBzdHJpZGUgaW4gYnl0ZXMKLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBpbnQgY2FsY3VsYXRlU2NhbmxpbmVTdHJpZGVTUFBTTShTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsIHNwcHNtLCBSYXN0ZXIgcikgewotICAgICAgICBpZiAoc3Bwc20uZ2V0U2NhbmxpbmVTdHJpZGUoKSAhPSBzcHBzbS5nZXRXaWR0aCgpKSB7Ci0gICAgICAgICAgICBpbnQgZGF0YVR5cGVTaXplID0gRGF0YUJ1ZmZlci5nZXREYXRhVHlwZVNpemUoci5nZXREYXRhQnVmZmVyKCkuZ2V0RGF0YVR5cGUoKSkgLyA4OwotICAgICAgICAgICAgcmV0dXJuIHNwcHNtLmdldFNjYW5saW5lU3RyaWRlKCkqZGF0YVR5cGVTaXplOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDYWxjdWxhdGVzIGJ5dGUgb2Zmc2V0IG9mIHRoZSBhbHBoYSBjaGFubmVsIGZyb20gdGhlIGJlZ2lubmluZyBvZiB0aGUgcGl4ZWwgZGF0YQotICAgICAqIEBwYXJhbSBzbSAtIHNhbXBsZSBtb2RlbAotICAgICAqIEBwYXJhbSByIC0gcmFzdGVyCi0gICAgICogQHJldHVybiBieXRlIG9mZnNldCBvZiB0aGUgYWxwaGEgY2hhbm5lbAotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIGludCBjYWxjdWxhdGVBbHBoYU9mZnNldChTYW1wbGVNb2RlbCBzbSwgUmFzdGVyIHIpIHsKLSAgICAgICAgaWYgKHNtIGluc3RhbmNlb2YgQ29tcG9uZW50U2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgIENvbXBvbmVudFNhbXBsZU1vZGVsIGNzbSA9IChDb21wb25lbnRTYW1wbGVNb2RlbCkgc207Ci0gICAgICAgICAgICBpbnQgZGF0YVR5cGVTaXplID0KLSAgICAgICAgICAgICAgICBEYXRhQnVmZmVyLmdldERhdGFUeXBlU2l6ZShyLmdldERhdGFCdWZmZXIoKS5nZXREYXRhVHlwZSgpKSAvIDg7Ci0gICAgICAgICAgICByZXR1cm4KLSAgICAgICAgICAgICAgICBjc20uZ2V0QmFuZE9mZnNldHMoKVtjc20uZ2V0QmFuZE9mZnNldHMoKS5sZW5ndGggLSAxXSAqIGRhdGFUeXBlU2l6ZTsKLSAgICAgICAgfSBlbHNlIGlmIChzbSBpbnN0YW5jZW9mIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwgc3Bwc20gPSAoU2luZ2xlUGl4ZWxQYWNrZWRTYW1wbGVNb2RlbCkgc207Ci0gICAgICAgICAgICByZXR1cm4gc3Bwc20uZ2V0Qml0T2Zmc2V0cygpW3NwcHNtLmdldEJpdE9mZnNldHMoKS5sZW5ndGggLSAxXSAvIDg7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICByZXR1cm4gLTE7IC8vIE5vIG9mZnNldCwgZG9uJ3QgY29weSBhbHBoYQotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9BbmRyb2lkRm9udC5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9BbmRyb2lkRm9udC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlOGFkMWJiLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvQW5kcm9pZEZvbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI1NCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuRm9udDsKLWltcG9ydCBqYXZhLmF3dC5Ub29sa2l0OwotaW1wb3J0IGphdmEuYXd0LmZvbnQuRm9udFJlbmRlckNvbnRleHQ7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5MaW5lTWV0cmljczsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEuaW8uRmlsZTsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaHRhYmxlOwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuRm9udE1hbmFnZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250LkZvbnRQZWVySW1wbDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuR2x5cGg7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250LkxpbmVNZXRyaWNzSW1wbDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBMaW51eCBwbGF0Zm9ybSBmb250IHBlZXIgaW1wbGVtZW50YXRpb24gYmFzZWQgb24gWGZ0IGFuZCBGcmVlVHlwZSBsaWJyYXJpZXMuCi0gKi8KLXB1YmxpYyBjbGFzcyBBbmRyb2lkRm9udCBleHRlbmRzIEZvbnRQZWVySW1wbCB7Ci0KLSAgICAvLyBQYWlycyBvZiBbYmVnaW4sIGVuZF0sWy4uXS4uIHVuaWNvZGUgcmFuZ2VzIHZhbHVlcyAKLSAgICBwcml2YXRlIGludFtdIGZvbnRVbmljb2RlUmFuZ2VzOwotICAgIAotICAgIC8vIHRhYmxlIHdpdGggbG9hZGVkIGNhY2hlZCBHbHlwaHMKLSAgICBwcml2YXRlIEhhc2h0YWJsZSBnbHlwaHMgPSBuZXcgSGFzaHRhYmxlKCk7Ci0gICAgCi0gICAgLy8gWDExIGRpc3BsYXkgdmFsdWUKLSAgICBwcml2YXRlIGxvbmcgZGlzcGxheSA9IDA7Ci0KLSAgICAvLyBYMTEgc2NyZWVuIHZhbHVlCi0gICAgcHJpdmF0ZSBpbnQgc2NyZWVuID0gMDsKLSAgICAKLSAgICBwdWJsaWMgQW5kcm9pZEZvbnQoU3RyaW5nIGZvbnROYW1lLCBpbnQgZm9udFN0eWxlLCBpbnQgZm9udFNpemUpIHsKLSAgICAgICAgLyoKLSAgICAgICAgICogV29ya2Fyb3VuZCA6IHRvIGluaXRpYWxpemUgYXd0IHBsYXRmb3JtLWRlcGVuZGVudCBmaWVsZHMgYW5kIGxpYnJhcmllcy4KLSAgICAgICAgICovCi0gICAgICAgIFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKTsKLSAgICAgICAgdGhpcy5uYW1lID0gZm9udE5hbWU7Ci0gICAgICAgIHRoaXMuc2l6ZSA9IGZvbnRTaXplOwotICAgICAgICB0aGlzLnN0eWxlID0gZm9udFN0eWxlOwotICAgICAgIAotICAgICAgICBpbml0QW5kcm9pZEZvbnQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbml0aWFsaXplcyBzb21lIG5hdGl2ZSBkZXBlbmRlbnQgZm9udCBpbmZvcm1hdGlvbiwgZS5nLiBudW1iZXIgb2YgZ2x5cGhzLCAKLSAgICAgKiBmb250IG1ldHJpY3MsIGl0YWxpYyBhbmdsZSBldGMuIAotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGluaXRBbmRyb2lkRm9udCgpewotICAgICAgICB0aGlzLm5sbSA9IG5ldyBBbmRyb2lkTGluZU1ldHJpY3ModGhpcywgbnVsbCwgIiAiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB0aGlzLmFzY2VudCA9IG5sbS5nZXRMb2dpY2FsQXNjZW50KCk7Ci0gICAgICAgIHRoaXMuZGVzY2VudCA9IG5sbS5nZXRMb2dpY2FsRGVzY2VudCgpOwotICAgICAgICB0aGlzLmhlaWdodCA9IG5sbS5nZXRIZWlnaHQoKTsKLSAgICAgICAgdGhpcy5sZWFkaW5nID0gbmxtLmdldExvZ2ljYWxMZWFkaW5nKCk7Ci0gICAgICAgIHRoaXMubWF4QWR2YW5jZSA9IG5sbS5nZXRMb2dpY2FsTWF4Q2hhcldpZHRoKCk7Ci0KLSAgICAgICAgaWYgKHRoaXMuZm9udFR5cGUgPT0gRm9udE1hbmFnZXIuRk9OVF9UWVBFX1QxKXsKLSAgICAgICAgICAgIHRoaXMuZGVmYXVsdENoYXIgPSAxOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdGhpcy5kZWZhdWx0Q2hhciA9IDA7Ci0gICAgICAgIH0KLQotICAgICAgICB0aGlzLm1heENoYXJCb3VuZHMgPSBuZXcgUmVjdGFuZ2xlMkQuRmxvYXQoMCwgLW5sbS5nZXRBc2NlbnQoKSwgbmxtLmdldE1heENoYXJXaWR0aCgpLCB0aGlzLmhlaWdodCk7Ci0gICAgfQotCi0KLSAgICBwdWJsaWMgYm9vbGVhbiBjYW5EaXNwbGF5KGNoYXIgY2hyKSB7Ci0gICAgICAgIC8vIFRPRE86IHRvIGltcHJvdmUgcGVyZm9ybWFuY2UgdGhlcmUgaXMgYSBzZW5jZSB0byBpbXBsZW1lbnQgZ2V0Ci0gICAgICAgIC8vIHVuaWNvZGUgcmFuZ2VzIHRvIGNoZWNrIGlmIGNoYXIgY2FuIGJlIGRpc3BsYXllZCB3aXRob3V0Ci0gICAgICAgIC8vIG5hdGl2ZSBjYWxscyBpbiBpc0dseXBoRXhpc3RzKCkgbWV0aG9kCi0KLSAgICAgICAgcmV0dXJuIGlzR2x5cGhFeGlzdHMoY2hyKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTGluZU1ldHJpY3MgZ2V0TGluZU1ldHJpY3MoU3RyaW5nIHN0ciwgRm9udFJlbmRlckNvbnRleHQgZnJjLCBBZmZpbmVUcmFuc2Zvcm0gYXQpIHsKLQotICAgICAgICAvLyBJbml0aWFsaXplIGJhc2VsaW5lIG9mZnNldHMKLSAgICAgICAgbmxtLmdldEJhc2VsaW5lT2Zmc2V0cygpOwotICAgICAgICAKLSAgICAgICAgTGluZU1ldHJpY3NJbXBsIGxtID0gKExpbmVNZXRyaWNzSW1wbCkodGhpcy5ubG0uY2xvbmUoKSk7Ci0gICAgICAgIGxtLnNldE51bUNoYXJzKHN0ci5sZW5ndGgoKSk7Ci0KLSAgICAgICAgaWYgKChhdCAhPSBudWxsKSAmJiAoIWF0LmlzSWRlbnRpdHkoKSkpewotICAgICAgICAgICAgbG0uc2NhbGUoKGZsb2F0KWF0LmdldFNjYWxlWCgpLCAoZmxvYXQpYXQuZ2V0U2NhbGVZKCkpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGxtOwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0UFNOYW1lKCkgewotICAgICAgICByZXR1cm4gcHNOYW1lOwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RmFtaWx5KExvY2FsZSBsKSB7Ci0gICAgICAgIC8vIFRPRE86IGltcGxlbWVudCBsb2NhbGl6ZWQgZmFtaWx5Ci0gICAgICAgIGlmIChmb250VHlwZSA9PSBGb250TWFuYWdlci5GT05UX1RZUEVfVFQpewotICAgICAgICAgICAgcmV0dXJuIHRoaXMuZ2V0RmFtaWx5KCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gdGhpcy5mb250RmFtaWx5TmFtZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIGdldEZvbnROYW1lKExvY2FsZSBsKSB7Ci0gICAgICAgIGlmICgocEZvbnQgPT0gMCkgfHwgKHRoaXMuZm9udFR5cGUgPT0gRm9udE1hbmFnZXIuRk9OVF9UWVBFX1QxKSl7Ci0gICAgICAgICAgICByZXR1cm4gdGhpcy5uYW1lOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHRoaXMuZ2V0Rm9udE5hbWUoKTsKLSAgICB9Ci0KLQotICAgIHB1YmxpYyBpbnQgZ2V0TWlzc2luZ0dseXBoQ29kZSgpIHsKLSAgICAgICAgcmV0dXJuIGdldERlZmF1bHRHbHlwaCgpLmdldEdseXBoQ29kZSgpOwotICAgIH0KLQotICAgIHB1YmxpYyBHbHlwaCBnZXRHbHlwaChjaGFyIGluZGV4KSB7Ci0gICAgICAgIEdseXBoIHJlc3VsdCA9IG51bGw7Ci0KLSAgICAgICAgT2JqZWN0IGtleSA9IG5ldyBJbnRlZ2VyKGluZGV4KTsKLSAgICAgICAgaWYgKGdseXBocy5jb250YWluc0tleShrZXkpKSB7Ci0gICAgICAgICAgICByZXN1bHQgPSAoR2x5cGgpIGdseXBocy5nZXQoa2V5KTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGlmICh0aGlzLmFkZEdseXBoKGluZGV4KSkgewotICAgICAgICAgICAgICAgIHJlc3VsdCA9IChHbHlwaCkgZ2x5cGhzLmdldChrZXkpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICByZXN1bHQgPSB0aGlzLmdldERlZmF1bHRHbHlwaCgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgR2x5cGggZ2V0RGVmYXVsdEdseXBoKCkgewotICAgIAl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiRGVmYXVsdEdseXBocyBub3QgaW1wbGVtZW50ZWQhIik7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGlzcG9zZXMgbmF0aXZlIGZvbnQgaGFuZGxlLiBJZiB0aGlzIGZvbnQgcGVlciB3YXMgY3JlYXRlZCBmcm9tIElucHV0U3RyZWFtIAotICAgICAqIHRlbXBvcmFyeSBjcmVhdGVkIGZvbnQgcmVzb3VyY2UgZmlsZSBpcyBkZWxldGVkLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGRpc3Bvc2UoKXsKLSAgICAgICAgU3RyaW5nIHRlbXBEaXJOYW1lOwotICAgICAgICBpZiAocEZvbnQgIT0gMCl7Ci0gICAgICAgICAgICBwRm9udCA9IDA7Ci0KLSAgICAgICAgICAgIGlmIChpc0NyZWF0ZWRGcm9tU3RyZWFtKCkpIHsKLSAgICAgICAgICAgICAgICBGaWxlIGZvbnRGaWxlID0gbmV3IEZpbGUoZ2V0VGVtcEZvbnRGaWxlTmFtZSgpKTsKLSAgICAgICAgICAgICAgICB0ZW1wRGlyTmFtZSA9IGZvbnRGaWxlLmdldFBhcmVudCgpOwotICAgICAgICAgICAgICAgIGZvbnRGaWxlLmRlbGV0ZSgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkIGdseXBoIHRvIGNhY2hlZCBHbHlwaCBvYmplY3RzIGluIHRoaXMgTGludXhGb250IG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdUNoYXIgdGhlIHNwZWNpZmllZCBjaGFyYWN0ZXIKLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgZ2x5cGggb2YgdGhlIHNwZWNpZmllZCBjaGFyYWN0ZXIgZXhpc3RzIGluIHRoaXMKLSAgICAgKiBMaW51eEZvbnQgb3IgdGhpcyBjaGFyYWN0ZXIgaXMgZXNjYXBlIHNlcXVlbmNlIGNoYXJhY3Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBhZGRHbHlwaChjaGFyIHVDaGFyKSB7Ci0gICAgCXRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQhIik7ICAgIAkKLSAgICB9Ci0KLSAgIC8qKgotICAgICogQWRkcyByYW5nZSBvZiBleGlzdGluZyBnbHlwaHMgdG8gdGhpcyBMaW51eEZvbnQgb2JqZWN0Ci0gICAgKiAKLSAgICAqIEBwYXJhbSB1Rmlyc3QgdGhlIGxvd2VzdCByYW5nZSdzIGJvdW5kLCBpbmNsdXNpdmUgCi0gICAgKiBAcGFyYW0gdUxhc3QgdGhlIGhpZ2hlc3QgcmFuZ2UncyBib3VuZCwgZXhjbHVzaXZlCi0gICAgKi8KLSAgICBwdWJsaWMgdm9pZCBhZGRHbHlwaHMoY2hhciB1Rmlyc3QsIGNoYXIgdUxhc3QpIHsKLSAgICAJCi0gICAgICAgIGNoYXIgaW5kZXggPSB1Rmlyc3Q7Ci0gICAgICAgIGlmICh1TGFzdCA8IHVGaXJzdCkgewotICAgICAgICAgICAgLy8gYXd0LjA5PW1pbiByYW5nZSBib3VuZCB2YWx1ZSBpcyBncmF0ZXIgdGhhbiBtYXggcmFuZ2UgYm91bmQKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMDkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICB3aGlsZSAoaW5kZXggPCB1TGFzdCkgewotICAgICAgICAgICAgYWRkR2x5cGgoaW5kZXgpOwotICAgICAgICAgICAgaW5kZXgrKzsKLSAgICAgICAgfQotICAgICAgICAKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgc3BlY2lmaWVkIGNoYXJhY3RlciBoYXMgY29ycmVzb3BuZGluZyBnbHlwaCwgZmFsc2Ugb3RoZXJ3aXNlLiAgCi0gICAgICogCi0gICAgICogQHBhcmFtIHVJbmRleCBzcGVjaWZpZWQgY2hhcgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzR2x5cGhFeGlzdHMoY2hhciB1SW5kZXgpIHsKLSAgICAJdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIkRlZmF1bHRHbHlwaHMgbm90IGltcGxlbWVudGVkISIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqICBSZXR1cm5zIGFuIGFycmF5IG9mIHVuaWNvZGUgcmFuZ2VzIHRoYXQgYXJlIHN1cHBvcnRlZCBieSB0aGlzIExpbnV4Rm9udC4gCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldFVuaWNvZGVSYW5nZXMoKSB7Ci0gICAgICAgIGludFtdIHJhbmdlcyA9IG5ldyBpbnRbZm9udFVuaWNvZGVSYW5nZXMubGVuZ3RoXTsKLSAgICAgICAgU3lzdGVtLmFycmF5Y29weShmb250VW5pY29kZVJhbmdlcywgMCwgcmFuZ2VzLCAwLAotICAgICAgICAgICAgICAgIGZvbnRVbmljb2RlUmFuZ2VzLmxlbmd0aCk7Ci0KLSAgICAgICAgcmV0dXJuIHJhbmdlczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm4gRm9udCBvYmplY3QgaWYgaXQgd2FzIHN1Y2Nlc3NmdWxseSBlbWJlZGRlZCBpbiBTeXN0ZW0KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEZvbnQgZW1iZWRGb250KFN0cmluZyBhYnNvbHV0ZVBhdGgpewotICAgIAl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiZW1iZWRGb250IG5vdCBpbXBsZW1lbnRlZCEiKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIGdldEZvbnROYW1lKCl7Ci0gICAgICAgIGlmICgocEZvbnQgIT0gMCkgJiYgKGZhY2VOYW1lID09IG51bGwpKXsKLSAgICAgICAgICAgIGlmICh0aGlzLmZvbnRUeXBlID09IEZvbnRNYW5hZ2VyLkZPTlRfVFlQRV9UMSl7Ci0gICAgICAgICAgICAgICAgZmFjZU5hbWUgPSBnZXRGYW1pbHkoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZmFjZU5hbWU7Ci0gICAgfQotCi0gICAgcHVibGljIFN0cmluZyBnZXRGYW1pbHkoKSB7Ci0gICAgICAgIHJldHVybiBmb250RmFtaWx5TmFtZTsKLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBpbml0aWF0ZWQgRm9udEV4dHJhTWV0cmljcyBpbnN0YW5jZSBvZiB0aGlzIFdpbmRvd3NGb250LgotICAgICAqLwotICAgIHB1YmxpYyBGb250RXh0cmFNZXRyaWNzIGdldEV4dHJhTWV0cmljcygpewotICAgIAl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvQW5kcm9pZEZvbnRNYW5hZ2VyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0FuZHJvaWRGb250TWFuYWdlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwNjNhMjU2Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvQW5kcm9pZEZvbnRNYW5hZ2VyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNzcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJbHlhIFMuIE9rb21pbgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udDsKLQotaW1wb3J0IGphdmEuYXd0LkZvbnQ7Ci1pbXBvcnQgamF2YS5hd3QucGVlci5Gb250UGVlcjsKLWltcG9ydCBqYXZhLmlvLkZpbGU7Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuUHJvcGVydGllczsKLWltcG9ydCBqYXZhLnV0aWwuVmVjdG9yOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250LkZvbnRNYW5hZ2VyOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5Gb250UHJvcGVydHk7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLWltcG9ydCBhbmRyb2lkLnV0aWwuTG9nOwotCi1wdWJsaWMgY2xhc3MgQW5kcm9pZEZvbnRNYW5hZ2VyIGV4dGVuZHMgRm9udE1hbmFnZXIgewotCi0gICAgLy8gc2V0IG9mIGFsbCBhdmFpbGFibGUgZmFjZXMgc3VwcG9ydGVkIGJ5IGEgc3lzdGVtCi0gICAgU3RyaW5nIGZhY2VzW107Ci0KLSAgICAvLyB3ZWlnaHQgbmFtZXMgYWNjb3JkaW5nIHRvIHhsZmQgc3RydWN0dXJlCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBTdHJpbmdbXSBMSU5VWF9XRUlHSFRfTkFNRVMgPSB7Ci0gICAgICAgICAgICAiYmxhY2siLCAiYm9sZCIsICJkZW1pYm9sZCIsICJtZWRpdW0iLCAibGlnaHQiIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgIH07Ci0KLSAgICAvLyBzbGFudCBuYW1lcyBhY2NvcmRpbmcgdG8geGxmZCBzdHJ1Y3R1cmUKLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIFN0cmluZ1tdIExJTlVYX1NMQU5UX05BTUVTID0gewotICAgICAgICAgICAgImkiLCAibyIsICJyIiAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJAotICAgIH07Ci0KLSAgICAvKiogU2luZ2xldG9uIEFuZHJvaWRGb250TWFuYWdlciBpbnN0YW5jZSAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgQW5kcm9pZEZvbnRNYW5hZ2VyIGluc3QgPSBuZXcgQW5kcm9pZEZvbnRNYW5hZ2VyKCk7Ci0KLSAgICBwcml2YXRlIEFuZHJvaWRGb250TWFuYWdlcigpIHsKLSAgICAgICAgc3VwZXIoKTsKLSAgICAgICAgZmFjZXMgPSBuZXcgU3RyaW5nW10gey8qIlBMQUlOIiwqLyAiTk9STUFMIiwgIkJPTEQiLCAiSVRBTElDIiwgIkJPTERJVEFMSUMifTsKLSAgICAgICAgaW5pdEZvbnRQcm9wZXJ0aWVzKCk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgaW5pdExDSURUYWJsZSgpewotICAgIAl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGVtcG9yYXJ5IEZpbGUgb2JqZWN0IHRvIHN0b3JlIGRhdGEgZnJvbSBJbnB1dFN0cmVhbS4KLSAgICAgKiBUaGlzIEZpbGUgb2JqZWN0IHNhdmVkIHRvIGB+Ly5mb250cy8nIGZvbGRlciB0aGF0IGlzIGluY2x1ZGVkIGluIHRoZSAKLSAgICAgKiBsaXN0IG9mIGZvbGRlcnMgc2VhcmNoZWQgZm9yIGZvbnQgZmlsZXMsIGFuZCB0aGlzIGlzIHdoZXJlIHVzZXItc3BlY2lmaWMgCi0gICAgICogZm9udCBmaWxlcyBzaG91bGQgYmUgaW5zdGFsbGVkLgotICAgICAqLwotICAgIHB1YmxpYyBGaWxlIGdldFRlbXBGb250RmlsZSgpdGhyb3dzIElPRXhjZXB0aW9uewotICAgICAgICBGaWxlIGZvbnRGaWxlID0gRmlsZS5jcmVhdGVUZW1wRmlsZSgiakZvbnQiLCAiLnR0ZiIsIG5ldyBGaWxlKFN5c3RlbS5nZXRQcm9wZXJ0eSgidXNlci5ob21lIikgKyIvLmZvbnRzIikpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJCAvLyROT04tTkxTLTQkCi0gICAgICAgIGZvbnRGaWxlLmRlbGV0ZU9uRXhpdCgpOwotCi0gICAgICAgIHJldHVybiBmb250RmlsZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbml0aWFsaXplcyBmUHJvcGVydGllcyBhcnJheSBmaWVsZCBmb3IgdGhlIGN1cnJlbnQgc3lzdGVtIGNvbmZpZ3VyYXRpb24gZm9udAotICAgICAqIHByb3BlcnR5IGZpbGUuCi0gICAgICogCi0gICAgICogUnVudGltZUV4Y2VwdGlvbiBpcyB0aHJvd24gaWYgZm9udCBwcm9wZXJ0eSBjb250YWlucyBpbmNvcnJlY3QgZm9ybWF0IG9mIAotICAgICAqIHhsZmQgc3RyaW5nLgotICAgICAqIAotICAgICAqIEByZXR1cm4gdHJ1ZSBpcyBzdWNjZXNzLCBmYWxzZSBpZiBmb250IHByb3BlcnR5IGRvZXNuJ3QgZXhpc3Qgb3IgZG9lc24ndAotICAgICAqIGNvbnRhaW4gcm9wZXJ0aWVzLiAKLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpbml0Rm9udFByb3BlcnRpZXMoKXsKLSAgICAgICAgRmlsZSBmcEZpbGUgPSBnZXRGb250UHJvcGVydHlGaWxlKCk7Ci0gICAgICAgIGlmIChmcEZpbGUgPT0gbnVsbCl7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBQcm9wZXJ0aWVzIHByb3BzID0gZ2V0UHJvcGVydGllcyhmcEZpbGUpOwotICAgICAgICBpZiAocHJvcHMgPT0gbnVsbCl7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IgKGludCBpPTA7IGkgPCBMT0dJQ0FMX0ZPTlRfTkFNRVMubGVuZ3RoOyBpKyspewotICAgICAgICAgICAgU3RyaW5nIGxOYW1lID0gTE9HSUNBTF9GT05UX05BTUVTW2ldOwotICAgICAgICAgICAgZm9yIChpbnQgaj0wOyBqIDwgU1RZTEVfTkFNRVMubGVuZ3RoOyBqKyspewotICAgICAgICAgICAgICAgIFN0cmluZyBzdHlsZU5hbWUgPSBTVFlMRV9OQU1FU1tqXTsKLSAgICAgICAgICAgICAgICBWZWN0b3IgcHJvcHNWZWN0b3IgPSBuZXcgVmVjdG9yKCk7Ci0KLSAgICAgICAgICAgICAgICAvLyBOdW1iZXIgb2YgZW50cmllcyBmb3IgYSBsb2dpY2FsIGZvbnQKLSAgICAgICAgICAgICAgICBpbnQgbnVtQ29tcCA9IDA7Ci0gICAgICAgICAgICAgICAgLy8gSXMgbW9yZSBlbnRyaWVzIGZvciB0aGlzIHN0eWxlIGFuZCBsb2dpY2FsIGZvbnQgbmFtZSBsZWZ0Ci0gICAgICAgICAgICAgICAgYm9vbGVhbiBtb3JlRW50cmllcyA9IHRydWU7Ci0gICAgICAgICAgICAgICAgU3RyaW5nIHZhbHVlID0gbnVsbDsKLQotICAgICAgICAgICAgICAgIHdoaWxlKG1vcmVFbnRyaWVzKXsKLSAgICAgICAgICAgICAgICAgICAgLy8gQ29tcG9uZW50IEZvbnQgTWFwcGluZ3MgcHJvcGVydHkgbmFtZQotICAgICAgICAgICAgICAgICAgICBTdHJpbmcgcHJvcGVydHkgPSBGT05UX01BUFBJTkdfS0VZU1swXS5yZXBsYWNlQWxsKCJMb2dpY2FsRm9udE5hbWUiLCBsTmFtZSkucmVwbGFjZUFsbCgiU3R5bGVOYW1lIiwgc3R5bGVOYW1lKS5yZXBsYWNlQWxsKCJDb21wb25lbnRJbmRleCIsIFN0cmluZy52YWx1ZU9mKG51bUNvbXApKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQKLSAgICAgICAgICAgICAgICAgICAgdmFsdWUgPSBwcm9wcy5nZXRQcm9wZXJ0eShwcm9wZXJ0eSk7Ci0KLSAgICAgICAgICAgICAgICAgICAgLy8gSWYgdGhlIFN0eWxlTmFtZSBpcyBvbWl0dGVkLCBpdCdzIGFzc3VtZWQgdG8gYmUgcGxhaW4KLSAgICAgICAgICAgICAgICAgICAgaWYgKChqID09IDApICYmICh2YWx1ZSA9PSBudWxsKSl7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwcm9wZXJ0eSA9IEZPTlRfTUFQUElOR19LRVlTWzFdLnJlcGxhY2VBbGwoIkxvZ2ljYWxGb250TmFtZSIsIGxOYW1lKS5yZXBsYWNlQWxsKCJDb21wb25lbnRJbmRleCIsIFN0cmluZy52YWx1ZU9mKG51bUNvbXApKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgICAgICAgICB2YWx1ZSA9IHByb3BzLmdldFByb3BlcnR5KHByb3BlcnR5KTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGlmICh2YWx1ZSAhPSBudWxsKXsKLSAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZ1tdIGZpZWxkcyA9IHBhcnNlWExGRCh2YWx1ZSk7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChmaWVsZHMgPT0gbnVsbCl7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjA4PXhmbGQgcGFyc2Ugc3RyaW5nIGVycm9yOiB7MH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4wOCIsIHZhbHVlKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nIGZvbnROYW1lID0gZmllbGRzWzFdOwotICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nIHdlaWdodCA9IGZpZWxkc1syXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZyBpdGFsaWMgPSBmaWVsZHNbM107Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIGludCBzdHlsZSA9IGdldEJvbGRTdHlsZSh3ZWlnaHQpIHwgZ2V0SXRhbGljU3R5bGUoaXRhbGljKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIENvbXBvbmVudCBGb250IENoYXJhY3RlciBFbmNvZGluZ3MgcHJvcGVydHkgdmFsdWUKLSAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZyBlbmNvZGluZyA9IHByb3BzLmdldFByb3BlcnR5KEZPTlRfQ0hBUkFDVEVSX0VOQ09ESU5HLnJlcGxhY2VBbGwoIkxvZ2ljYWxGb250TmFtZSIsIGxOYW1lKS5yZXBsYWNlQWxsKCJDb21wb25lbnRJbmRleCIsIFN0cmluZy52YWx1ZU9mKG51bUNvbXApKSk7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotCi0gICAgICAgICAgICAgICAgICAgICAgICAvLyBFeGNsdXNpb24gUmFuZ2VzIHByb3BlcnR5IHZhbHVlCi0gICAgICAgICAgICAgICAgICAgICAgICBTdHJpbmcgZXhjbFN0cmluZyA9IHByb3BzLmdldFByb3BlcnR5KEVYQ0xVU0lPTl9SQU5HRVMucmVwbGFjZUFsbCgiTG9naWNhbEZvbnROYW1lIiwgbE5hbWUpLnJlcGxhY2VBbGwoIkNvbXBvbmVudEluZGV4IiwgU3RyaW5nLnZhbHVlT2YobnVtQ29tcCkpKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgICAgICAgICAgICAgICAgICBpbnRbXSBleGNsUmFuZ2UgPSBwYXJzZUludGVydmFscyhleGNsU3RyaW5nKTsKLQotICAgICAgICAgICAgICAgICAgICAgICAgRm9udFByb3BlcnR5IGZwID0gbmV3IEFuZHJvaWRGb250UHJvcGVydHkobE5hbWUsIHN0eWxlTmFtZSwgbnVsbCwgZm9udE5hbWUsIHZhbHVlLCBzdHlsZSwgZXhjbFJhbmdlLCBlbmNvZGluZyk7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIHByb3BzVmVjdG9yLmFkZChmcCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBudW1Db21wKys7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBtb3JlRW50cmllcyA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGZQcm9wZXJ0aWVzLnB1dChMT0dJQ0FMX0ZPTlRfTkFNRVNbaV0gKyAiLiIgKyBqLCBwcm9wc1ZlY3Rvcik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiB0cnVlOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBzdHlsZSBhY2NvcmRpbmcgdG8gdGhlIHhsZmQgd2VpZ2h0IHN0cmluZy4KLSAgICAgKiBJZiB3ZWlnaHQgc3RyaW5nIGlzIGluY29ycmVjdCByZXR1cm5lZCB2YWx1ZSBpcyBGb250LlBMQUlOCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0ciB3ZWlnaHQgbmFtZSBTdHJpbmcKLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBnZXRCb2xkU3R5bGUoU3RyaW5nIHN0cil7Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgTElOVVhfV0VJR0hUX05BTUVTLmxlbmd0aDtpKyspewotICAgICAgICAgICAgaWYgKHN0ci5lcXVhbHNJZ25vcmVDYXNlKExJTlVYX1dFSUdIVF9OQU1FU1tpXSkpewotICAgICAgICAgICAgICAgIHJldHVybiAoaSA8IDMpID8gRm9udC5CT0xEIDogRm9udC5QTEFJTjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gRm9udC5QTEFJTjsKLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBzdHlsZSBhY2NvcmRpbmcgdG8gdGhlIHhsZmQgc2xhbnQgc3RyaW5nLgotICAgICAqIElmIHNsYW50IHN0cmluZyBpcyBpbmNvcnJlY3QgcmV0dXJuZWQgdmFsdWUgaXMgRm9udC5QTEFJTgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHIgc2xhbnQgbmFtZSBTdHJpbmcKLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBnZXRJdGFsaWNTdHlsZShTdHJpbmcgc3RyKXsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBMSU5VWF9TTEFOVF9OQU1FUy5sZW5ndGg7aSsrKXsKLSAgICAgICAgICAgIGlmIChzdHIuZXF1YWxzSWdub3JlQ2FzZShMSU5VWF9TTEFOVF9OQU1FU1tpXSkpewotICAgICAgICAgICAgICAgIHJldHVybiAoaSA8IDIpID8gRm9udC5JVEFMSUMgOiBGb250LlBMQUlOOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBGb250LlBMQUlOOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBhcnNlIHhsZmQgc3RyaW5nIGFuZCByZXR1cm5zIGFycmF5IG9mIFN0cmluZ3Mgd2l0aCBzZXBhcmF0ZSB4bGZkIAotICAgICAqIGVsZW1lbnRzLjxwPgotICAgICAqIAotICAgICAqIHhsZmQgZm9ybWF0OgotICAgICAqICAgICAgLUZvdW5kcnktRmFtaWx5LVdlaWdodC1TbGFudC1XaWR0aC1TdHlsZS1QaXhlbFNpemUtUG9pbnRTaXplLVJlc1gtUmVzWS1TcGFjaW5nLUF2Z1dpZHRoLVJlZ2lzdHJ5LUVuY29kaW5nCi0gICAgICogQHBhcmFtIHhsZmQgU3RyaW5nIHBhcmFtZXRlciBpbiB4bGZkIGZvcm1hdAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgU3RyaW5nW10gcGFyc2VYTEZEKFN0cmluZyB4bGZkKXsKLSAgICAgICAgaW50IGZpZWxkc0NvdW50ID0gMTQ7Ci0gICAgICAgIFN0cmluZyBmaWVsZHNEZWxpbSA9ICItIjsgLy8kTk9OLU5MUy0xJAotICAgICAgICBTdHJpbmdbXSByZXMgPSBuZXcgU3RyaW5nW2ZpZWxkc0NvdW50XTsKLSAgICAgICAgaWYgKCF4bGZkLnN0YXJ0c1dpdGgoZmllbGRzRGVsaW0pKXsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgeGxmZCA9IHhsZmQuc3Vic3RyaW5nKDEpOwotICAgICAgICBpbnQgaT0wOwotICAgICAgICBpbnQgcG9zOwotICAgICAgICBmb3IgKGk9MDsgaSA8IGZpZWxkc0NvdW50LTE7IGkrKyl7Ci0gICAgICAgICAgICBwb3MgPSB4bGZkLmluZGV4T2YoZmllbGRzRGVsaW0pOwotICAgICAgICAgICAgaWYgKHBvcyAhPSAtMSl7Ci0gICAgICAgICAgICAgICAgcmVzW2ldID0geGxmZC5zdWJzdHJpbmcoMCwgcG9zKTsKLSAgICAgICAgICAgICAgICB4bGZkID0geGxmZC5zdWJzdHJpbmcocG9zICsgMSk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHBvcyA9IHhsZmQuaW5kZXhPZihmaWVsZHNEZWxpbSk7Ci0KLSAgICAgICAgLy8gY2hlY2sgaWYgbm8gZmllbGRzIGxlZnQKLSAgICAgICAgaWYocG9zICE9IC0xKXsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgICAgIHJlc1tmaWVsZHNDb3VudC0xXSA9IHhsZmQ7Ci0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldEZhY2VJbmRleChTdHJpbmcgZmFjZU5hbWUpewotICAgIAkKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBmYWNlcy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgaWYoZmFjZXNbaV0uZXF1YWxzKGZhY2VOYW1lKSl7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIC0xOwotICAgIH0KLQotICAgIHB1YmxpYyBTdHJpbmdbXSBnZXRBbGxGYW1pbGllcygpewotICAgICAgICBpZiAoYWxsRmFtaWxpZXMgPT0gbnVsbCl7Ci0gICAgICAgIAlhbGxGYW1pbGllcyA9IG5ldyBTdHJpbmdbXXsic2Fucy1zZXJpZiIsICJzZXJpZiIsICJtb25vc3BhY2UifTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gYWxsRmFtaWxpZXM7Ci0gICAgfQotCi0gICAgcHVibGljIEZvbnRbXSBnZXRBbGxGb250cygpewotICAgICAgICBGb250W10gZm9udHMgPSBuZXcgRm9udFtmYWNlcy5sZW5ndGhdOwotICAgICAgICBmb3IgKGludCBpID0wOyBpIDwgZm9udHMubGVuZ3RoO2krKyl7Ci0gICAgICAgICAgICBmb250c1tpXSA9IG5ldyBGb250KGZhY2VzW2ldLCBGb250LlBMQUlOLCAxKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZm9udHM7Ci0gICAgfQotCi0gICAgcHVibGljIEZvbnRQZWVyIGNyZWF0ZVBoeXNpY2FsRm9udFBlZXIoU3RyaW5nIG5hbWUsIGludCBzdHlsZSwgaW50IHNpemUpIHsKLSAgICAgICAgQW5kcm9pZEZvbnQgcGVlcjsKLSAgICAgICAgaW50IGZhbWlseUluZGV4ID0gZ2V0RmFtaWx5SW5kZXgobmFtZSk7Ci0gICAgICAgIGlmIChmYW1pbHlJbmRleCAhPSAtMSl7Ci0gICAgICAgICAgICAvLyAhISB3ZSB1c2UgZmFtaWx5IG5hbWVzIGZyb20gdGhlIGxpc3Qgd2l0aCBjYWNoZWQgZmFtaWxpZXMgYmVjYXVzZSAKLSAgICAgICAgICAgIC8vIHRoZXkgYXJlIGRpZmZlciBmcm9tIHRoZSBmYW1pbHkgbmFtZXMgaW4geGxmZCBzdHJ1Y3R1cmUsIGluIHhsZmQgCi0gICAgICAgICAgICAvLyBmYW1pbHkgbmFtZXMgbW9zdGx5IGluIGxvd2VyIGNhc2UuCi0gICAgICAgICAgICBwZWVyID0gbmV3IEFuZHJvaWRGb250KGdldEZhbWlseShmYW1pbHlJbmRleCksIHN0eWxlLCBzaXplKTsKLSAgICAgICAgICAgIHBlZXIuc2V0RmFtaWx5KGdldEZhbWlseShmYW1pbHlJbmRleCkpOwotICAgICAgICAgICAgcmV0dXJuIHBlZXI7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IGZhY2VJbmRleCA9IGdldEZhY2VJbmRleChuYW1lKTsgCi0gICAgICAgIGlmIChmYWNlSW5kZXggIT0gLTEpewotCi0gICAgICAgICAgICBwZWVyID0gbmV3IEFuZHJvaWRGb250KG5hbWUsIHN0eWxlLCBzaXplKTsKLSAgICAgICAgICAgIHJldHVybiBwZWVyOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgRm9udFBlZXIgY3JlYXRlRGVmYXVsdEZvbnQoaW50IHN0eWxlLCBpbnQgc2l6ZSkgewotICAgIAlMb2cuaSgiREVGQVVMVCBGT05UIiwgSW50ZWdlci50b1N0cmluZyhzdHlsZSkpOwotICAgICAgICByZXR1cm4gbmV3IEFuZHJvaWRGb250KERFRkFVTFRfTkFNRSwgc3R5bGUsIHNpemUpOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9BbmRyb2lkRm9udFByb3BlcnR5LmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0FuZHJvaWRGb250UHJvcGVydHkuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMGNmZGM0My4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0FuZHJvaWRGb250UHJvcGVydHkuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDgxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi0vKioKLSAqIEFuZHJvaWQgRm9udFByb3BlcnR5IGltcGxlbWVudGF0aW9uLCBhcHBsaWNhYmxlIGZvciBMaW51eCBmb3JtYXRzIG9mIAotICogZm9udCBwcm9wZXJ0eSBmaWxlcy4gCi0gKi8KLXB1YmxpYyBjbGFzcyBBbmRyb2lkRm9udFByb3BlcnR5IGV4dGVuZHMgRm9udFByb3BlcnR5IHsKLSAgICAKLSAgICAvKiogeGxmZCBzdHJpbmcgdGhhdCBpcyBhcHBsaWNhYmxlIGZvciBMaW51eCBmb250LnByb3BlcnRpZXMgKi8gCi0gICAgU3RyaW5nIHhsZmQ7Ci0KLSAgICAvKiogbG9naWNhbCBuYW1lIG9mIHRoZSBmb250IGNvcnJlc3BvbmRpbmcgdG8gdGhpcyBGb250UHJvcGVydHkgKi8gCi0gICAgU3RyaW5nIGxvZ2ljYWxOYW1lOwotICAgIAotICAgIC8qKiBzdHlsZSBuYW1lIG9mIHRoZSBmb250IGNvcnJlc3BvbmRpbmcgdG8gdGhpcyBGb250UHJvcGVydHkgKi8KLSAgICBTdHJpbmcgc3R5bGVOYW1lOwotCi0gICAgcHVibGljIEFuZHJvaWRGb250UHJvcGVydHkoU3RyaW5nIF9sb2dpY2FsTmFtZSwgU3RyaW5nIF9zdHlsZU5hbWUsIFN0cmluZyBfZmlsZU5hbWUsIFN0cmluZyBfbmFtZSwgU3RyaW5nIF94bGZkLCBpbnQgX3N0eWxlLCBpbnRbXSBleGNsdXNpb25SYW5nZSwgU3RyaW5nIF9lbmNvZGluZyl7Ci0gICAgICAgIHRoaXMubG9naWNhbE5hbWUgPSBfbG9naWNhbE5hbWU7Ci0gICAgICAgIHRoaXMuc3R5bGVOYW1lID0gX3N0eWxlTmFtZTsKLSAgICAgICAgdGhpcy5uYW1lID0gX25hbWU7Ci0gICAgICAgIHRoaXMuZW5jb2RpbmcgPSBfZW5jb2Rpbmc7Ci0gICAgICAgIHRoaXMuZXhjbFJhbmdlID0gZXhjbHVzaW9uUmFuZ2U7Ci0gICAgICAgIHRoaXMuZmlsZU5hbWUgPSBfZmlsZU5hbWU7Ci0gICAgICAgIHRoaXMueGxmZCA9IF94bGZkOwotICAgICAgICB0aGlzLnN0eWxlID0gX3N0eWxlOwotICAgIH0KLSAgICAKLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGxvZ2ljYWwgbmFtZSBvZiB0aGUgZm9udCBjb3JyZXNwb25kaW5nIHRvIHRoaXMgRm9udFByb3BlcnR5LiAKLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldExvZ2ljYWxOYW1lKCl7Ci0gICAgICAgIHJldHVybiBsb2dpY2FsTmFtZTsKLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBzdHlsZSBuYW1lIG9mIHRoZSBmb250IGNvcnJlc3BvbmRpbmcgdG8gdGhpcyBGb250UHJvcGVydHkuIAotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0U3R5bGVOYW1lKCl7Ci0gICAgICAgIHJldHVybiBzdHlsZU5hbWU7Ci0gICAgfQotICAgIAotICAgIC8qKgotICAgICAqIFJldHVybnMgeGxmZCBzdHJpbmcgb2YgdGhpcyBGb250UHJvcGVydHkuIAotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0WExGRCgpewotICAgICAgICByZXR1cm4geGxmZDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCl7Ci0gICAgICAgIHJldHVybiBuZXcgU3RyaW5nKHRoaXMuZ2V0Q2xhc3MoKS5nZXROYW1lKCkgKwotICAgICAgICAgICAgICAgICJbbmFtZT0iICsgbmFtZSArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAiLGZpbGVOYW1lPSIrIGZpbGVOYW1lICsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICIsQ2hhcnNldD0iICsgZW5jb2RpbmcgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgIixleGNsUmFuZ2U9IiArIGV4Y2xSYW5nZSArIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAiLHhsZmQ9IiArIHhsZmQgKyAiXSIpOyAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQKLQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9BbmRyb2lkR2x5cGhWZWN0b3IuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvQW5kcm9pZEdseXBoVmVjdG9yLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDRjZTVhZWQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9BbmRyb2lkR2x5cGhWZWN0b3IuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDIxOSArMCwwIEBACi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udDsKLQotaW1wb3J0IGNvbS5hbmRyb2lkLmludGVybmFsLmF3dC5BbmRyb2lkR3JhcGhpY3MyRDsKLQotaW1wb3J0IGphdmEuYXd0LkZvbnQ7Ci1pbXBvcnQgamF2YS5hd3QuU2hhcGU7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5Gb250UmVuZGVyQ29udGV4dDsKLWltcG9ydCBqYXZhLmF3dC5mb250LkdseXBoSnVzdGlmaWNhdGlvbkluZm87Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5HbHlwaE1ldHJpY3M7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5HbHlwaFZlY3RvcjsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBvaW50MkQ7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLQotaW1wb3J0IGFuZHJvaWQudXRpbC5Mb2c7Ci1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5QYXRoOwotCi1wdWJsaWMgY2xhc3MgQW5kcm9pZEdseXBoVmVjdG9yIGV4dGVuZHMgR2x5cGhWZWN0b3IgewotCi0gICAgLy8gYXJyYXkgb2YgY2hhcnMgZGVmaW5lZCBpbiBjb25zdHJ1Y3RvcgotICAgIHB1YmxpYyBjaGFyW10gY2hhclZlY3RvcjsKLQotICAgIC8vIGFycmF5IG9mIEdseXBoIG9iamVjdHMsIHRoYXQgZGVzY3JpYmUgaW5mb3JtYXRpb24gYWJvdXQgZ2x5cGhzCi0gICAgcHVibGljIEdseXBoW10gdmVjdG9yOwotCi0gICAgLy8gYXJyYXkgb2YgZGVmYXVsdCBwb3NpdGlvbnMgb2YgZ2x5cGhzIGluIEdseXBoVmVjdG9yCi0gICAgLy8gd2l0aG91dCBhcHBseWluZyBHbHlwaFZlY3RvcidzIHRyYW5zZm9ybQotICAgIGZsb2F0W10gZGVmYXVsdFBvc2l0aW9uczsKLQotICAgIC8vIGFycmF5IG9mIGxvZ2ljYWwgcG9zaXRpb25zIG9mIGdseXBocyBpbiBHbHlwaFZlY3RvcgotCi0gICAgZmxvYXRbXSBsb2dpY2FsUG9zaXRpb25zOwotCi0gICAgLy8gYXJyYXkgb2YgdmlzdWFsIChyZWFsKSBwb3NpdGlvbnMgb2YgZ2x5cGhzIGluIEdseXBoVmVjdG9yCi0gICAgcHVibGljIGZsb2F0W10gdmlzdWFsUG9zaXRpb25zOwotCi0gICAgLy8gRm9udFJlbmRlckNvbnRleHQgZm9yIHRoaXMgdmVjdG9yLgotICAgIHByb3RlY3RlZCBGb250UmVuZGVyQ29udGV4dCB2ZWN0b3JGUkM7Ci0KLSAgICAvLyBsYXlvdXQgZmxhZ3MgbWFzawotICAgIHByb3RlY3RlZCBpbnQgbGF5b3V0RmxhZ3MgPSAwOwotCi0gICAgLy8gYXJyYXkgb2YgY2FjaGVkIGdseXBoIG91dGxpbmVzIAotICAgIHByb3RlY3RlZCBTaGFwZVtdIGd2U2hhcGVzOwotCi0gICAgRm9udFBlZXJJbXBsIHBlZXI7Ci0KLSAgICAvLyBmb250IGNvcnJlc3BvbmRpbmcgdG8gdGhlIEdseXBoVmVjdG9yIAotICAgIEZvbnQgZm9udDsKLQotICAgIC8vIGFzY2VudCBvZiB0aGUgZm9udAotICAgIGZsb2F0IGFzY2VudDsKLQotICAgIC8vIGhlaWdodCBvZiB0aGUgZm9udAotICAgIGZsb2F0IGhlaWdodDsKLSAgICAKLSAgICAvLyBsZWFkaW5nIG9mIHRoZSBmb250Ci0gICAgZmxvYXQgbGVhZGluZzsKLSAgICAKLSAgICAvLyBkZXNjZW50IG9mIHRoZSBmb250Ci0gICAgZmxvYXQgZGVzY2VudDsKLQotICAgIC8vIHRyYW5zZm9ybSBvZiB0aGUgR2x5cGhWZWN0b3IKLSAgICBBZmZpbmVUcmFuc2Zvcm0gdHJhbnNmb3JtOwotCi0gICAgQFN1cHByZXNzV2FybmluZ3MoImRlcHJlY2F0aW9uIikKLSAgICBwdWJsaWMgQW5kcm9pZEdseXBoVmVjdG9yKGNoYXJbXSBjaGFycywgRm9udFJlbmRlckNvbnRleHQgZnJjLCBGb250IGZudCwKLSAgICAgICAgICAgIGludCBmbGFncykgewotICAgICAgICBpbnQgbGVuID0gY2hhcnMubGVuZ3RoOwotICAgICAgICB0aGlzLmZvbnQgPSBmbnQ7Ci0gICAgICAgIExpbmVNZXRyaWNzSW1wbCBsbUltcGwgPSAoTGluZU1ldHJpY3NJbXBsKWZudC5nZXRMaW5lTWV0cmljcyhTdHJpbmcudmFsdWVPZihjaGFycyksIGZyYyk7ICAgICAJCi0gICAgICAgIHRoaXMuYXNjZW50ID0gbG1JbXBsLmdldEFzY2VudCgpOwotICAgICAgICB0aGlzLmhlaWdodCA9IGxtSW1wbC5nZXRIZWlnaHQoKTsKLSAgICAgICAgdGhpcy5sZWFkaW5nID0gbG1JbXBsLmdldExlYWRpbmcoKTsKLSAgICAgICAgdGhpcy5kZXNjZW50ID0gbG1JbXBsLmdldERlc2NlbnQoKTsKLSAgICAgICAgdGhpcy5jaGFyVmVjdG9yID0gY2hhcnM7Ci0gICAgICAgIHRoaXMudmVjdG9yRlJDID0gZnJjOwotICAgIH0KLQotICAgIHB1YmxpYyBBbmRyb2lkR2x5cGhWZWN0b3IoY2hhcltdIGNoYXJzLCBGb250UmVuZGVyQ29udGV4dCBmcmMsIEZvbnQgZm50KSB7Ci0gICAgICAgIHRoaXMoY2hhcnMsIGZyYywgZm50LCAwKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQW5kcm9pZEdseXBoVmVjdG9yKFN0cmluZyBzdHIsIEZvbnRSZW5kZXJDb250ZXh0IGZyYywgRm9udCBmbnQpIHsKLSAgICAgICAgdGhpcyhzdHIudG9DaGFyQXJyYXkoKSwgZnJjLCBmbnQsIDApOwotICAgIH0KLQotICAgIHB1YmxpYyBBbmRyb2lkR2x5cGhWZWN0b3IoU3RyaW5nIHN0ciwgRm9udFJlbmRlckNvbnRleHQgZnJjLCBGb250IGZudCwgaW50IGZsYWdzKSB7Ci0gICAgICAgIHRoaXMoc3RyLnRvQ2hhckFycmF5KCksIGZyYywgZm50LCBmbGFncyk7Ci0gICAgfQotCi0JQE92ZXJyaWRlCi0JcHVibGljIGJvb2xlYW4gZXF1YWxzKEdseXBoVmVjdG9yIGdseXBoVmVjdG9yKSB7Ci0JCXJldHVybiBmYWxzZTsKLQl9Ci0KLQlwdWJsaWMgY2hhcltdIGdldEdseXBocygpIHsKLQkJcmV0dXJuIHRoaXMuY2hhclZlY3RvcjsKLQl9Ci0JCi0JQE92ZXJyaWRlCi0JcHVibGljIEZvbnQgZ2V0Rm9udCgpIHsKLQkJcmV0dXJuIHRoaXMuZm9udDsKLQl9Ci0KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgRm9udFJlbmRlckNvbnRleHQgZ2V0Rm9udFJlbmRlckNvbnRleHQoKSB7Ci0JCXJldHVybiB0aGlzLnZlY3RvckZSQzsKLQl9Ci0KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgaW50IGdldEdseXBoQ29kZShpbnQgZ2x5cGhJbmRleCkgewotCQlyZXR1cm4gY2hhclZlY3RvcltnbHlwaEluZGV4XTsKLQl9Ci0KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgaW50W10gZ2V0R2x5cGhDb2RlcyhpbnQgYmVnaW5HbHlwaEluZGV4LCBpbnQgbnVtRW50cmllcywKLQkJCWludFtdIGNvZGVSZXR1cm4pIHsKLQkJdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCEiKTsKLQl9Ci0KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBnZXRHbHlwaEp1c3RpZmljYXRpb25JbmZvKGludCBnbHlwaEluZGV4KSB7Ci0JCXRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQhIik7Ci0JfQotCi0JQE92ZXJyaWRlCi0JcHVibGljIFNoYXBlIGdldEdseXBoTG9naWNhbEJvdW5kcyhpbnQgZ2x5cGhJbmRleCkgewotCQl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotCX0KLQotCUBPdmVycmlkZQotCXB1YmxpYyBHbHlwaE1ldHJpY3MgZ2V0R2x5cGhNZXRyaWNzKGludCBnbHlwaEluZGV4KSB7Ci0JCXRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQhIik7Ci0JfQotCi0JcHVibGljIFBhdGggZ2V0QW5kcm9pZEdseXBoT3V0bGluZShpbnQgZ2x5cGhJbmRleCkgewotCQlBbmRyb2lkR3JhcGhpY3MyRCBnID0gQW5kcm9pZEdyYXBoaWNzMkQuZ2V0SW5zdGFuY2UoKTsKLSAgICAgICAgUGF0aCBwYXRoID0gbmV3IFBhdGgoKTsKLSAgICAgICAgY2hhciB0bXBbXSA9IG5ldyBjaGFyWzFdOwotICAgICAgICB0bXBbMF0gPSBjaGFyVmVjdG9yW2dseXBoSW5kZXhdOwotICAgICAgICAoKEFuZHJvaWRHcmFwaGljczJEKWcpLmdldEFuZHJvaWRQYWludCgpLmdldFRleHRQYXRoKG5ldyBTdHJpbmcodG1wKSwgMCwgMSwgMCwgMCwgcGF0aCk7Ci0gICAgICAgIHJldHVybiBwYXRoOwotCX0KLQkKLQlAT3ZlcnJpZGUKLQlwdWJsaWMgU2hhcGUgZ2V0R2x5cGhPdXRsaW5lKGludCBnbHlwaEluZGV4KSB7Ci0JCXRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQhIik7Ci0JfQotCi0JQE92ZXJyaWRlCi0JcHVibGljIFBvaW50MkQgZ2V0R2x5cGhQb3NpdGlvbihpbnQgZ2x5cGhJbmRleCkgewotCQl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotCX0KLQotCUBPdmVycmlkZQotCXB1YmxpYyBmbG9hdFtdIGdldEdseXBoUG9zaXRpb25zKGludCBiZWdpbkdseXBoSW5kZXgsIGludCBudW1FbnRyaWVzLAotCQkJZmxvYXRbXSBwb3NpdGlvblJldHVybikgewotCQl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotCX0KLQotCUBPdmVycmlkZQotCXB1YmxpYyBBZmZpbmVUcmFuc2Zvcm0gZ2V0R2x5cGhUcmFuc2Zvcm0oaW50IGdseXBoSW5kZXgpIHsKLQkJdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCEiKTsKLQl9Ci0KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgU2hhcGUgZ2V0R2x5cGhWaXN1YWxCb3VuZHMoaW50IGdseXBoSW5kZXgpIHsKLQkJdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCEiKTsKLQl9Ci0KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0TG9naWNhbEJvdW5kcygpIHsKLQkJdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCEiKTsKLQl9Ci0KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgaW50IGdldE51bUdseXBocygpIHsKLQkJcmV0dXJuIGNoYXJWZWN0b3IubGVuZ3RoOwotCX0KLQotCUBPdmVycmlkZQotCXB1YmxpYyBTaGFwZSBnZXRPdXRsaW5lKGZsb2F0IHgsIGZsb2F0IHkpIHsKLQkJdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCEiKTsKLQl9Ci0KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgU2hhcGUgZ2V0T3V0bGluZSgpIHsKLQkJdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk5vdCBpbXBsZW1lbnRlZCEiKTsKLQl9Ci0KLQlwdWJsaWMgUGF0aCBnZXRBbmRyb2lkT3V0bGluZSgpIHsKLQkJQW5kcm9pZEdyYXBoaWNzMkQgZyA9IEFuZHJvaWRHcmFwaGljczJELmdldEluc3RhbmNlKCk7Ci0gICAgICAgIFBhdGggcGF0aCA9IG5ldyBQYXRoKCk7Ci0gICAgICAgICgoQW5kcm9pZEdyYXBoaWNzMkQpZykuZ2V0QW5kcm9pZFBhaW50KCkuZ2V0VGV4dFBhdGgobmV3IFN0cmluZyhjaGFyVmVjdG9yKSwgMCwgY2hhclZlY3Rvci5sZW5ndGgsIDAsIDAsIHBhdGgpOwotICAgICAgICByZXR1cm4gcGF0aDsKLQl9Ci0KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0VmlzdWFsQm91bmRzKCkgewotCQl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotCX0KLQotCUBPdmVycmlkZQotCXB1YmxpYyB2b2lkIHBlcmZvcm1EZWZhdWx0TGF5b3V0KCkgewotCQl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotCX0KLQotCUBPdmVycmlkZQotCXB1YmxpYyB2b2lkIHNldEdseXBoUG9zaXRpb24oaW50IGdseXBoSW5kZXgsIFBvaW50MkQgbmV3UG9zKSB7Ci0JCXRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJOb3QgaW1wbGVtZW50ZWQhIik7Ci0JfQotCi0JQE92ZXJyaWRlCi0JcHVibGljIHZvaWQgc2V0R2x5cGhUcmFuc2Zvcm0oaW50IGdseXBoSW5kZXgsIEFmZmluZVRyYW5zZm9ybSB0cmFucykgewotCQl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiTm90IGltcGxlbWVudGVkISIpOwotCX0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9BbmRyb2lkTGluZU1ldHJpY3MuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvQW5kcm9pZExpbmVNZXRyaWNzLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGYzN2JlNmQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9BbmRyb2lkTGluZU1ldHJpY3MuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEyMCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuZm9udC5Gb250UmVuZGVyQ29udGV4dDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuTGluZU1ldHJpY3NJbXBsOwotCi0KLS8qKgotICoKLSAqIExpbnV4IGltcGxlbWVudGF0aW9uIG9mIExpbmVNZXRyaWNzIGNsYXNzCi0gKi8KLXB1YmxpYyBjbGFzcyBBbmRyb2lkTGluZU1ldHJpY3MgZXh0ZW5kcyBMaW5lTWV0cmljc0ltcGwgewotICAgIAotICAgIC8qKgotICAgICAqIENvbnN0cnVjdG9yCi0gICAgICovCi0gICAgcHVibGljIEFuZHJvaWRMaW5lTWV0cmljcyggICAgQW5kcm9pZEZvbnQgZm50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBGb250UmVuZGVyQ29udGV4dCBmcmMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZyBzdHIpewotICAgICAgICBudW1DaGFycyA9IHN0ci5sZW5ndGgoKTsKLSAgICAgICAgYmFzZUxpbmVJbmRleCA9IDA7Ci0KLSAgICAgICAgYXNjZW50ID0gZm50LmFzY2VudDsgICAgLy8gQXNjZW50IG9mIHRoZSBmb250Ci0gICAgICAgIGRlc2NlbnQgPSAtZm50LmRlc2NlbnQ7ICAvLyBEZXNjZW50IG9mIHRoZSBmb250Ci0gICAgICAgIGxlYWRpbmcgPSBmbnQubGVhZGluZzsgIC8vIEV4dGVybmFsIGxlYWRpbmcKLQotICAgICAgICBoZWlnaHQgPSBhc2NlbnQgKyBkZXNjZW50ICsgbGVhZGluZzsgICAgLy8gSGVpZ2h0IG9mIHRoZSBmb250ICggPT0gKGFzY2VudCArIGRlc2NlbnQgKyBsZWFkaW5nKSkKLSAgICAgICAgdW5kZXJsaW5lVGhpY2tuZXNzID0gMC4wZjsKLSAgICAgICAgdW5kZXJsaW5lT2Zmc2V0ID0gMC4wZjsKLSAgICAgICAgc3RyaWtldGhyb3VnaFRoaWNrbmVzcyA9IDAuMGY7Ci0gICAgICAgIHN0cmlrZXRocm91Z2hPZmZzZXQgPSAwLjBmOwotICAgICAgICBtYXhDaGFyV2lkdGggPSAwLjBmOwotCi0gICAgICAgIC8vICAgIFRPRE86IEZpbmQgb3V0IHBpeGVsIG1ldHJpY3MKLSAgICAgICAgLyoKLSAgICAgICAgICogcG9zaXRpdmUgbWV0cmljcyByb3VuZGVkIHRvIHRoZSBzbWFsbGVzdCBpbnQgdGhhdCBpcyBiaWdnZXIgdGhhbiB2YWx1ZQotICAgICAgICAgKiBuZWdhdGl2ZSBtZXRyaWNzIHJvdW5kZWQgdG8gdGhlIHNtYWxsZXN0IGludCB0aGF0IGlzIGxlc3NlciB0aGFuIHZhbHVlCi0gICAgICAgICAqIHRoaWNrbmVzc2VzIHJvdW5kZWQgdG8gaW50ICgoaW50KXJvdW5kKHZhbHVlICsgMC41KSkKLSAgICAgICAgICoKLSAgICAgICAgICovCi0KLSAgICAgICAgbEFzY2VudCA9IChpbnQpTWF0aC5jZWlsKGZudC5hc2NlbnQpOy8vICAgLy8gQXNjZW50IG9mIHRoZSBmb250Ci0gICAgICAgIGxEZXNjZW50ID0gLShpbnQpTWF0aC5jZWlsKGZudC5kZXNjZW50KTsvLyBEZXNjZW50IG9mIHRoZSBmb250Ci0gICAgICAgIGxMZWFkaW5nID0gKGludClNYXRoLmNlaWwobGVhZGluZyk7ICAvLyBFeHRlcm5hbCBsZWFkaW5nCi0KLSAgICAgICAgbEhlaWdodCA9IGxBc2NlbnQgKyBsRGVzY2VudCArIGxMZWFkaW5nOyAgICAvLyBIZWlnaHQgb2YgdGhlIGZvbnQgKCA9PSAoYXNjZW50ICsgZGVzY2VudCArIGxlYWRpbmcpKQotCi0gICAgICAgIGxVbmRlcmxpbmVUaGlja25lc3MgPSBNYXRoLnJvdW5kKHVuZGVybGluZVRoaWNrbmVzcyk7Ly8oaW50KW1ldHJpY3NbMTFdOwotCi0gICAgICAgIGlmICh1bmRlcmxpbmVPZmZzZXQgPj0gMCl7Ci0gICAgICAgICAgICBsVW5kZXJsaW5lT2Zmc2V0ID0gKGludClNYXRoLmNlaWwodW5kZXJsaW5lT2Zmc2V0KTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGxVbmRlcmxpbmVPZmZzZXQgPSAoaW50KU1hdGguZmxvb3IodW5kZXJsaW5lT2Zmc2V0KTsKLSAgICAgICAgfQotCi0gICAgICAgIGxTdHJpa2V0aHJvdWdoVGhpY2tuZXNzID0gTWF0aC5yb3VuZChzdHJpa2V0aHJvdWdoVGhpY2tuZXNzKTsgLy8oaW50KW1ldHJpY3NbMTNdOwotCi0gICAgICAgIGlmIChzdHJpa2V0aHJvdWdoT2Zmc2V0ID49IDApewotICAgICAgICAgICAgbFN0cmlrZXRocm91Z2hPZmZzZXQgPSAoaW50KU1hdGguY2VpbChzdHJpa2V0aHJvdWdoT2Zmc2V0KTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGxTdHJpa2V0aHJvdWdoT2Zmc2V0ID0gKGludClNYXRoLmZsb29yKHN0cmlrZXRocm91Z2hPZmZzZXQpOwotICAgICAgICB9Ci0KLSAgICAgICAgbE1heENoYXJXaWR0aCA9IChpbnQpTWF0aC5jZWlsKG1heENoYXJXaWR0aCk7IC8vKGludCltZXRyaWNzWzE1XTsKLSAgICAgICAgdW5pdHNfcGVyX0VNID0gMDsKLQotICAgIH0KLQotICAgIHB1YmxpYyBmbG9hdFtdIGdldEJhc2VsaW5lT2Zmc2V0cygpIHsKLSAgICAgICAgLy8gVE9ETzogaW1wbGVtZW50IGJhc2VsaW5lIG9mZnNldHMgZm9yIFRydWVUeXBlIGZvbnRzCi0gICAgICAgIGlmIChiYXNlbGluZU9mZnNldHMgPT0gbnVsbCl7Ci0gICAgICAgICAgICBmbG9hdFtdIGJhc2VsaW5lRGF0YSA9IG51bGw7Ci0KLSAgICAgICAgICAgIC8vIFRlbXBvcmFyeSB3b3JrYXJvdW5kOgotICAgICAgICAgICAgLy8gQ29tbWVudGVkIG91dCBuYXRpdmUgZGF0YSBpbml0aWFsaXphdGlvbiwgc2luY2UgaXQgY2FuIAotICAgICAgICAgICAgLy8gY2F1c2UgZmFpbHVyZXMgd2l0aCBvcGVuaW5nIGZpbGVzIGluIG11bHRpdGhyZWFkZWQgYXBwbGljYXRpb25zLgotICAgICAgICAgICAgLy8KLSAgICAgICAgICAgIC8vIFRPRE86IHN1cHBvcnQgd29yayB3aXRoIHRydWV0eXBlIGRhdGEgaW4gbXVsdGl0aHJlYWRlZAotICAgICAgICAgICAgLy8gYXBwbGljYXRpb25zLgotCi0gICAgICAgICAgICAvLyBJZiBmb250IFRydWVUeXBlIGRhdGEgaXMgdGFrZW4gZnJvbSBCQVNFIHRhYmxlCi0vLyAgICAgICAgICAgIGlmICgodGhpcy5mb250LmdldEZvbnRIYW5kbGUoKSAhPSAwKSAmJiAoZm9udC5nZXRGb250VHlwZSgpID09IEZvbnRNYW5hZ2VyLkZPTlRfVFlQRV9UVCkpewotLy8gICAgICAgICAgICAgICAgYmFzZWxpbmVEYXRhID0gTGludXhOYXRpdmVGb250LmdldEJhc2VsaW5lT2Zmc2V0c05hdGl2ZShmb250LmdldEZvbnRIYW5kbGUoKSwgZm9udC5nZXRTaXplKCksIGFzY2VudCwgZGVzY2VudCwgdW5pdHNfcGVyX0VNKTsKLS8vICAgICAgICAgICAgfQotLy8KLSAgICAgICAgICAgICAgICBiYXNlTGluZUluZGV4ID0gMDsKLSAgICAgICAgICAgICAgICBiYXNlbGluZU9mZnNldHMgPSBuZXcgZmxvYXRbXXswLCAoLWFzY2VudCtkZXNjZW50KS8yLCAtYXNjZW50fTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBiYXNlbGluZU9mZnNldHM7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRCYXNlbGluZUluZGV4KCkgewotICAgICAgICBpZiAoYmFzZWxpbmVPZmZzZXRzID09IG51bGwpewotICAgICAgICAgICAgLy8gZ2V0IG9mZnNldHMgYW5kIHNldCBjb3JyZWN0IGluZGV4Ci0gICAgICAgICAgICBnZXRCYXNlbGluZU9mZnNldHMoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gYmFzZUxpbmVJbmRleDsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvQmFzaWNNZXRyaWNzLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0Jhc2ljTWV0cmljcy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjMGZiMzkwLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvQmFzaWNNZXRyaWNzLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMzQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKgotICovCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuZm9udC5MaW5lTWV0cmljczsKLWltcG9ydCBqYXZhLmF3dC5mb250LkdyYXBoaWNBdHRyaWJ1dGU7Ci1pbXBvcnQgamF2YS5hd3QuKjsKLQotLyoqCi0gKiBEYXRlOiBNYXkgMTQsIDIwMDUKLSAqIFRpbWU6IDc6NDQ6MTMgUE0KLSAqCi0gKiBUaGlzIGNsYXNzIGluY2Fwc3VsYXRlcyB0ZXh0IG1ldHJpY3Mgc3BlY2lmaWMgZm9yIHRoZSB0ZXh0IGxheW91dCBvcgotICogZm9yIHRoZSBzZXBhcmF0ZSB0ZXh0IHNlZ21lbnQuIFRleHQgc2VnbWVudCBpcyBhIHRleHQgcnVuIHdpdGggdGhlIGNvbnN0YW50IGRpcmVjdGlvbgotICogYW5kIGF0dHJpYnV0ZXMgbGlrZSBmb250LCBkZWNvcmF0aW9ucywgZXRjLiBCYXNpY01ldHJpY3MgaXMgYWxzbyB1c2VkIHRvIHN0b3JlCi0gKiBjYWxjdWxhdGVkIHRleHQgbWV0cmljcyBsaWtlIGFkdmFuY2UsIGFzY2VudCBvciBkZXNjZW50LiB0aGlzIGNsYXNzIGlzIHZlcnkgc2ltaWxhciB0bwotICogTGluZU1ldHJpY3MsIGJ1dCBwcm92aWRlcyBzb21lIGFkZGl0aW9uYWwgaW5mbywgY29uc3RydWN0b3JzIGFuZCBpcyBtb3JlIHRyYW5zcGFyZW50LgotICovCi1wdWJsaWMgY2xhc3MgQmFzaWNNZXRyaWNzIHsKLSAgICBpbnQgYmFzZUxpbmVJbmRleDsKLQotICAgIGZsb2F0IGFzY2VudDsgICAvLyBBc2NlbnQgb2YgdGhlIGZvbnQKLSAgICBmbG9hdCBkZXNjZW50OyAgLy8gRGVzY2VudCBvZiB0aGUgZm9udAotICAgIGZsb2F0IGxlYWRpbmc7ICAvLyBFeHRlcm5hbCBsZWFkaW5nCi0gICAgZmxvYXQgYWR2YW5jZTsKLQotICAgIGZsb2F0IGl0YWxpY0FuZ2xlOwotICAgIGZsb2F0IHN1cGVyU2NyaXB0T2Zmc2V0OwotCi0gICAgZmxvYXQgdW5kZXJsaW5lT2Zmc2V0OwotICAgIGZsb2F0IHVuZGVybGluZVRoaWNrbmVzczsKLQotICAgIGZsb2F0IHN0cmlrZXRocm91Z2hPZmZzZXQ7Ci0gICAgZmxvYXQgc3RyaWtldGhyb3VnaFRoaWNrbmVzczsKLQotICAgIC8qKgotICAgICAqIENvbnN0cnVjdHMgQmFzaWNNZXRyaWNzIGZyb20gTGluZU1ldHJpY3MgYW5kIGZvbnQKLSAgICAgKiBAcGFyYW0gbG0KLSAgICAgKiBAcGFyYW0gZm9udAotICAgICAqLwotICAgIEJhc2ljTWV0cmljcyhMaW5lTWV0cmljcyBsbSwgRm9udCBmb250KSB7Ci0gICAgICAgIGFzY2VudCA9IGxtLmdldEFzY2VudCgpOwotICAgICAgICBkZXNjZW50ID0gbG0uZ2V0RGVzY2VudCgpOwotICAgICAgICBsZWFkaW5nID0gbG0uZ2V0TGVhZGluZygpOwotCi0gICAgICAgIHVuZGVybGluZU9mZnNldCA9IGxtLmdldFVuZGVybGluZU9mZnNldCgpOwotICAgICAgICB1bmRlcmxpbmVUaGlja25lc3MgPSBsbS5nZXRVbmRlcmxpbmVUaGlja25lc3MoKTsKLQotICAgICAgICBzdHJpa2V0aHJvdWdoT2Zmc2V0ID0gbG0uZ2V0U3RyaWtldGhyb3VnaE9mZnNldCgpOwotICAgICAgICBzdHJpa2V0aHJvdWdoVGhpY2tuZXNzID0gbG0uZ2V0U3RyaWtldGhyb3VnaFRoaWNrbmVzcygpOwotCi0gICAgICAgIGJhc2VMaW5lSW5kZXggPSBsbS5nZXRCYXNlbGluZUluZGV4KCk7Ci0KLSAgICAgICAgaXRhbGljQW5nbGUgPSBmb250LmdldEl0YWxpY0FuZ2xlKCk7Ci0gICAgICAgIHN1cGVyU2NyaXB0T2Zmc2V0ID0gKGZsb2F0KSBmb250LmdldFRyYW5zZm9ybSgpLmdldFRyYW5zbGF0ZVkoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb25zdHJ1Y3RzIEJhc2ljTWV0cmljcyBmcm9tIEdyYXBoaWNBdHRyaWJ1dGUuCi0gICAgICogSXQgZ2V0cyBhc2NlbnQgYW5kIGRlc2NlbnQgZnJvbSB0aGUgZ3JhcGhpYyBhdHRyaWJ1dGUgYW5kCi0gICAgICogY29tcHV0ZXMgcmVhc29uYWJsZSBkZWZhdWx0cyBmb3Igb3RoZXIgbWV0cmljcy4KLSAgICAgKiBAcGFyYW0gZ2EgLSBncmFwaGljIGF0dHJpYnV0ZQotICAgICAqLwotICAgIEJhc2ljTWV0cmljcyhHcmFwaGljQXR0cmlidXRlIGdhKSB7Ci0gICAgICAgIGFzY2VudCA9IGdhLmdldEFzY2VudCgpOwotICAgICAgICBkZXNjZW50ID0gZ2EuZ2V0RGVzY2VudCgpOwotICAgICAgICBsZWFkaW5nID0gMjsKLQotICAgICAgICBiYXNlTGluZUluZGV4ID0gZ2EuZ2V0QWxpZ25tZW50KCk7Ci0KLSAgICAgICAgaXRhbGljQW5nbGUgPSAwOwotICAgICAgICBzdXBlclNjcmlwdE9mZnNldCA9IDA7Ci0KLSAgICAgICAgdW5kZXJsaW5lT2Zmc2V0ID0gTWF0aC5tYXgoZGVzY2VudC8yLCAxKTsKLQotICAgICAgICAvLyBKdXN0IHN1Z2dlc3RlZCwgc2hvdWxkIGJlIGNhcF9zdGVtX3dpZHRoIG9yIHNvbWV0aGluZyBsaWtlIHRoYXQKLSAgICAgICAgdW5kZXJsaW5lVGhpY2tuZXNzID0gTWF0aC5tYXgoYXNjZW50LzEzLCAxKTsKLQotICAgICAgICBzdHJpa2V0aHJvdWdoT2Zmc2V0ID0gLWFzY2VudC8yOyAvLyBTb21ldGhpbmcgbGlrZSBtaWRkbGUgb2YgdGhlIGxpbmUKLSAgICAgICAgc3RyaWtldGhyb3VnaFRoaWNrbmVzcyA9IHVuZGVybGluZVRoaWNrbmVzczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb3BpZXMgbWV0cmljcyBmcm9tIHRoZSBUZXh0TWV0cmljc0NhbGN1bGF0b3Igb2JqZWN0LgotICAgICAqIEBwYXJhbSB0bWMgLSBUZXh0TWV0cmljc0NhbGN1bGF0b3Igb2JqZWN0Ci0gICAgICovCi0gICAgQmFzaWNNZXRyaWNzKFRleHRNZXRyaWNzQ2FsY3VsYXRvciB0bWMpIHsKLSAgICAgICAgYXNjZW50ID0gdG1jLmFzY2VudDsKLSAgICAgICAgZGVzY2VudCA9IHRtYy5kZXNjZW50OwotICAgICAgICBsZWFkaW5nID0gdG1jLmxlYWRpbmc7Ci0gICAgICAgIGFkdmFuY2UgPSB0bWMuYWR2YW5jZTsKLSAgICAgICAgYmFzZUxpbmVJbmRleCA9IHRtYy5iYXNlbGluZUluZGV4OwotICAgIH0KLQotICAgIHB1YmxpYyBmbG9hdCBnZXRBc2NlbnQoKSB7Ci0gICAgICAgIHJldHVybiBhc2NlbnQ7Ci0gICAgfQotCi0gICAgcHVibGljIGZsb2F0IGdldERlc2NlbnQoKSB7Ci0gICAgICAgIHJldHVybiBkZXNjZW50OwotICAgIH0KLQotICAgIHB1YmxpYyBmbG9hdCBnZXRMZWFkaW5nKCkgewotICAgICAgICByZXR1cm4gbGVhZGluZzsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmxvYXQgZ2V0QWR2YW5jZSgpIHsKLSAgICAgICAgcmV0dXJuIGFkdmFuY2U7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRCYXNlTGluZUluZGV4KCkgewotICAgICAgICByZXR1cm4gYmFzZUxpbmVJbmRleDsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0NhcmV0TWFuYWdlci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9DYXJldE1hbmFnZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYjE4YmRkNS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0NhcmV0TWFuYWdlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTMwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICoKLSAqIEBkYXRlOiBKdW4gMTQsIDIwMDUKLSAqLwotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udDsKLQotaW1wb3J0IGphdmEuYXd0LmZvbnQuVGV4dEhpdEluZm87Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5UZXh0TGF5b3V0OwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5HZW5lcmFsUGF0aDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkxpbmUyRDsKLWltcG9ydCBqYXZhLmF3dC4qOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBwcm92aWRlcyBmdW5jdGlvbmFsaXR5IGZvciBjcmVhdGluZyBjYXJldCBhbmQgaGlnaGxpZ2h0IHNoYXBlcwotICogKGJpZGlyZWN0aW9uYWwgdGV4dCBpcyBhbHNvIHN1cHBvcnRlZCwgYnV0LCB1bmZvcnR1bmF0ZWx5LCBub3QgdGVzdGVkIHlldCkuCi0gKi8KLXB1YmxpYyBjbGFzcyBDYXJldE1hbmFnZXIgewotICAgIHByaXZhdGUgVGV4dFJ1bkJyZWFrZXIgYnJlYWtlcjsKLQotICAgIHB1YmxpYyBDYXJldE1hbmFnZXIoVGV4dFJ1bkJyZWFrZXIgYnJlYWtlcikgewotICAgICAgICB0aGlzLmJyZWFrZXIgPSBicmVha2VyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiBUZXh0SGl0SW5mbyBpcyBub3Qgb3V0IG9mIHRoZSB0ZXh0IHJhbmdlIGFuZCB0aHJvd3MgdGhlCi0gICAgICogSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uIGlmIGl0IGlzLgotICAgICAqIEBwYXJhbSBpbmZvIC0gdGV4dCBoaXQgaW5mbwotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBjaGVja0hpdChUZXh0SGl0SW5mbyBpbmZvKSB7Ci0gICAgICAgIGludCBpZHggPSBpbmZvLmdldEluc2VydGlvbkluZGV4KCk7Ci0KLSAgICAgICAgaWYgKGlkeCA8IDAgfHwgaWR4ID4gYnJlYWtlci5nZXRDaGFyQ291bnQoKSkgewotICAgICAgICAgICAgLy8gYXd0LjQyPVRleHRIaXRJbmZvIG91dCBvZiByYW5nZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40MiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2FsY3VsYXRlcyBhbmQgcmV0dXJucyB2aXN1YWwgcG9zaXRpb24gZnJvbSB0aGUgdGV4dCBoaXQgaW5mby4KLSAgICAgKiBAcGFyYW0gaGl0SW5mbyAtIHRleHQgaGl0IGluZm8KLSAgICAgKiBAcmV0dXJuIHZpc3VhbCBpbmRleAotICAgICAqLwotICAgIHByaXZhdGUgaW50IGdldFZpc3VhbEZyb21IaXRJbmZvKFRleHRIaXRJbmZvIGhpdEluZm8pIHsKLSAgICAgICAgZmluYWwgaW50IGlkeCA9IGhpdEluZm8uZ2V0Q2hhckluZGV4KCk7Ci0KLSAgICAgICAgaWYgKGlkeCA+PSAwICYmIGlkeCA8IGJyZWFrZXIuZ2V0Q2hhckNvdW50KCkpIHsKLSAgICAgICAgICAgIGludCB2aXN1YWwgPSBicmVha2VyLmdldFZpc3VhbEZyb21Mb2dpY2FsKGlkeCk7Ci0gICAgICAgICAgICAvLyBXZSB0YWtlIG5leHQgY2hhcmFjdGVyIGZvciAoTFRSIGNoYXIgKyBUUkFJTElORyBpbmZvKSBhbmQgKFJUTCArIExFQURJTkcpCi0gICAgICAgICAgICBpZiAoaGl0SW5mby5pc0xlYWRpbmdFZGdlKCkgXiAoKGJyZWFrZXIuZ2V0TGV2ZWwoaWR4KSAmIDB4MSkgPT0gMHgwKSkgewotICAgICAgICAgICAgICAgIHZpc3VhbCsrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIHZpc3VhbDsKLSAgICAgICAgfSBlbHNlIGlmIChpZHggPCAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gYnJlYWtlci5pc0xUUigpID8gMDogYnJlYWtlci5nZXRDaGFyQ291bnQoKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJldHVybiBicmVha2VyLmlzTFRSKCkgPyBicmVha2VyLmdldENoYXJDb3VudCgpIDogMDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENhbGN1bGF0ZXMgdGV4dCBoaXQgaW5mbyBmcm9tIHRoZSB2aXN1YWwgcG9zaXRpb24KLSAgICAgKiBAcGFyYW0gdmlzdWFsIC0gdmlzdWFsIHBvc2l0aW9uCi0gICAgICogQHJldHVybiB0ZXh0IGhpdCBpbmZvCi0gICAgICovCi0gICAgcHJpdmF0ZSBUZXh0SGl0SW5mbyBnZXRIaXRJbmZvRnJvbVZpc3VhbChpbnQgdmlzdWFsKSB7Ci0gICAgICAgIGZpbmFsIGJvb2xlYW4gZmlyc3QgPSB2aXN1YWwgPT0gMDsKLQotICAgICAgICBpZiAoIShmaXJzdCB8fCB2aXN1YWwgPT0gYnJlYWtlci5nZXRDaGFyQ291bnQoKSkpIHsKLSAgICAgICAgICAgIGludCBsb2dpY2FsID0gYnJlYWtlci5nZXRMb2dpY2FsRnJvbVZpc3VhbCh2aXN1YWwpOwotICAgICAgICAgICAgcmV0dXJuIChicmVha2VyLmdldExldmVsKGxvZ2ljYWwpICYgMHgxKSA9PSAweDAgPwotICAgICAgICAgICAgICAgICAgICBUZXh0SGl0SW5mby5sZWFkaW5nKGxvZ2ljYWwpIDogLy8gTFRSCi0gICAgICAgICAgICAgICAgICAgIFRleHRIaXRJbmZvLnRyYWlsaW5nKGxvZ2ljYWwpOyAvLyBSVEwKLSAgICAgICAgfSBlbHNlIGlmIChmaXJzdCkgewotICAgICAgICAgICAgcmV0dXJuIGJyZWFrZXIuaXNMVFIoKSA/Ci0gICAgICAgICAgICAgICAgICAgIFRleHRIaXRJbmZvLnRyYWlsaW5nKC0xKSA6Ci0gICAgICAgICAgICAgICAgICAgIFRleHRIaXRJbmZvLmxlYWRpbmcoYnJlYWtlci5nZXRDaGFyQ291bnQoKSk7Ci0gICAgICAgIH0gZWxzZSB7IC8vIExhc3QKLSAgICAgICAgICAgIHJldHVybiBicmVha2VyLmlzTFRSKCkgPwotICAgICAgICAgICAgICAgICAgICBUZXh0SGl0SW5mby5sZWFkaW5nKGJyZWFrZXIuZ2V0Q2hhckNvdW50KCkpIDoKLSAgICAgICAgICAgICAgICAgICAgVGV4dEhpdEluZm8udHJhaWxpbmcoLTEpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBjYXJldCBpbmZvLiBSZXF1aXJlZCBmb3IgdGhlIGdldENhcmV0SW5mbwotICAgICAqIG1ldGhvZHMgb2YgdGhlIFRleHRMYXlvdXQKLSAgICAgKiBAcGFyYW0gaGl0SW5mbyAtIHNwZWNpZmllcyBjYXJldCBwb3NpdGlvbgotICAgICAqIEByZXR1cm4gY2FyZXQgaW5mbywgc2VlIFRleHRMYXlvdXQuZ2V0Q2FyZXRJbmZvIGRvY3VtZW50YXRpb24KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXRbXSBnZXRDYXJldEluZm8oVGV4dEhpdEluZm8gaGl0SW5mbykgewotICAgICAgICBjaGVja0hpdChoaXRJbmZvKTsKLSAgICAgICAgZmxvYXQgcmVzW10gPSBuZXcgZmxvYXRbMl07Ci0KLSAgICAgICAgaW50IHZpc3VhbCA9IGdldFZpc3VhbEZyb21IaXRJbmZvKGhpdEluZm8pOwotICAgICAgICBmbG9hdCBhZHZhbmNlLCBhbmdsZTsKLSAgICAgICAgVGV4dFJ1blNlZ21lbnQgc2VnOwotCi0gICAgICAgIGlmICh2aXN1YWwgPCBicmVha2VyLmdldENoYXJDb3VudCgpKSB7Ci0gICAgICAgICAgICBpbnQgbG9nSWR4ID0gYnJlYWtlci5nZXRMb2dpY2FsRnJvbVZpc3VhbCh2aXN1YWwpOwotICAgICAgICAgICAgaW50IHNlZ21lbnRJZHggPSBicmVha2VyLmxvZ2ljYWwyc2VnbWVudFtsb2dJZHhdOwotICAgICAgICAgICAgc2VnID0gYnJlYWtlci5ydW5TZWdtZW50cy5nZXQoc2VnbWVudElkeCk7Ci0gICAgICAgICAgICBhZHZhbmNlID0gc2VnLnggKyBzZWcuZ2V0QWR2YW5jZURlbHRhKHNlZy5nZXRTdGFydCgpLCBsb2dJZHgpOwotICAgICAgICAgICAgYW5nbGUgPSBzZWcubWV0cmljcy5pdGFsaWNBbmdsZTsKLQotICAgICAgICB9IGVsc2UgeyAvLyBMYXN0IGNoYXJhY3RlcgotICAgICAgICAgICAgaW50IGxvZ0lkeCA9IGJyZWFrZXIuZ2V0TG9naWNhbEZyb21WaXN1YWwodmlzdWFsLTEpOwotICAgICAgICAgICAgaW50IHNlZ21lbnRJZHggPSBicmVha2VyLmxvZ2ljYWwyc2VnbWVudFtsb2dJZHhdOwotICAgICAgICAgICAgc2VnID0gYnJlYWtlci5ydW5TZWdtZW50cy5nZXQoc2VnbWVudElkeCk7Ci0gICAgICAgICAgICBhZHZhbmNlID0gc2VnLnggKyBzZWcuZ2V0QWR2YW5jZURlbHRhKHNlZy5nZXRTdGFydCgpLCBsb2dJZHgrMSk7Ci0gICAgICAgIH0KLQotICAgICAgICBhbmdsZSA9IHNlZy5tZXRyaWNzLml0YWxpY0FuZ2xlOwotCi0gICAgICAgIHJlc1swXSA9IGFkdmFuY2U7Ci0gICAgICAgIHJlc1sxXSA9IGFuZ2xlOwotCi0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbmV4dCBwb3NpdGlvbiB0byB0aGUgcmlnaHQgZnJvbSB0aGUgY3VycmVudCBjYXJldCBwb3NpdGlvbgotICAgICAqIEBwYXJhbSBoaXRJbmZvIC0gY3VycmVudCBwb3NpdGlvbgotICAgICAqIEByZXR1cm4gbmV4dCBwb3NpdGlvbiB0byB0aGUgcmlnaHQKLSAgICAgKi8KLSAgICBwdWJsaWMgVGV4dEhpdEluZm8gZ2V0TmV4dFJpZ2h0SGl0KFRleHRIaXRJbmZvIGhpdEluZm8pIHsKLSAgICAgICAgY2hlY2tIaXQoaGl0SW5mbyk7Ci0gICAgICAgIGludCB2aXN1YWwgPSBnZXRWaXN1YWxGcm9tSGl0SW5mbyhoaXRJbmZvKTsKLQotICAgICAgICBpZiAodmlzdWFsID09IGJyZWFrZXIuZ2V0Q2hhckNvdW50KCkpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgVGV4dEhpdEluZm8gbmV3SW5mbzsKLQotICAgICAgICB3aGlsZSh2aXN1YWwgPD0gYnJlYWtlci5nZXRDaGFyQ291bnQoKSkgewotICAgICAgICAgICAgdmlzdWFsKys7Ci0gICAgICAgICAgICBuZXdJbmZvID0gZ2V0SGl0SW5mb0Zyb21WaXN1YWwodmlzdWFsKTsKLQotICAgICAgICAgICAgaWYgKG5ld0luZm8uZ2V0Q2hhckluZGV4KCkgPj0gYnJlYWtlci5sb2dpY2FsMnNlZ21lbnQubGVuZ3RoKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG5ld0luZm87Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChoaXRJbmZvLmdldENoYXJJbmRleCgpID49IDApIHsgLy8gRG9uJ3QgY2hlY2sgZm9yIGxlZnRtb3N0IGluZm8KLSAgICAgICAgICAgICAgICBpZiAoCi0gICAgICAgICAgICAgICAgICAgICAgICBicmVha2VyLmxvZ2ljYWwyc2VnbWVudFtuZXdJbmZvLmdldENoYXJJbmRleCgpXSAhPQotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWtlci5sb2dpY2FsMnNlZ21lbnRbaGl0SW5mby5nZXRDaGFySW5kZXgoKV0KLSAgICAgICAgICAgICAgICApIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIG5ld0luZm87IC8vIFdlIGNyb3NzZWQgc2VnbWVudCBib3VuZGFyeQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgVGV4dFJ1blNlZ21lbnQgc2VnID0gYnJlYWtlci5ydW5TZWdtZW50cy5nZXQoYnJlYWtlci5sb2dpY2FsMnNlZ21lbnRbbmV3SW5mbwotICAgICAgICAgICAgICAgICAgICAuZ2V0Q2hhckluZGV4KCldKTsKLSAgICAgICAgICAgIGlmICghc2VnLmNoYXJIYXNaZXJvQWR2YW5jZShuZXdJbmZvLmdldENoYXJJbmRleCgpKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBuZXdJbmZvOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbmV4dCBwb3NpdGlvbiB0byB0aGUgbGVmdCBmcm9tIHRoZSBjdXJyZW50IGNhcmV0IHBvc2l0aW9uCi0gICAgICogQHBhcmFtIGhpdEluZm8gLSBjdXJyZW50IHBvc2l0aW9uCi0gICAgICogQHJldHVybiBuZXh0IHBvc2l0aW9uIHRvIHRoZSBsZWZ0Ci0gICAgICovCi0gICAgcHVibGljIFRleHRIaXRJbmZvIGdldE5leHRMZWZ0SGl0KFRleHRIaXRJbmZvIGhpdEluZm8pIHsKLSAgICAgICAgY2hlY2tIaXQoaGl0SW5mbyk7Ci0gICAgICAgIGludCB2aXN1YWwgPSBnZXRWaXN1YWxGcm9tSGl0SW5mbyhoaXRJbmZvKTsKLQotICAgICAgICBpZiAodmlzdWFsID09IDApIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgVGV4dEhpdEluZm8gbmV3SW5mbzsKLQotICAgICAgICB3aGlsZSh2aXN1YWwgPj0gMCkgewotICAgICAgICAgICAgdmlzdWFsLS07Ci0gICAgICAgICAgICBuZXdJbmZvID0gZ2V0SGl0SW5mb0Zyb21WaXN1YWwodmlzdWFsKTsKLQotICAgICAgICAgICAgaWYgKG5ld0luZm8uZ2V0Q2hhckluZGV4KCkgPCAwKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG5ld0luZm87Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIERvbid0IGNoZWNrIGZvciByaWdodG1vc3QgaW5mbwotICAgICAgICAgICAgaWYgKGhpdEluZm8uZ2V0Q2hhckluZGV4KCkgPCBicmVha2VyLmxvZ2ljYWwyc2VnbWVudC5sZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBpZiAoCi0gICAgICAgICAgICAgICAgICAgICAgICBicmVha2VyLmxvZ2ljYWwyc2VnbWVudFtuZXdJbmZvLmdldENoYXJJbmRleCgpXSAhPQotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWtlci5sb2dpY2FsMnNlZ21lbnRbaGl0SW5mby5nZXRDaGFySW5kZXgoKV0KLSAgICAgICAgICAgICAgICApIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIG5ld0luZm87IC8vIFdlIGNyb3NzZWQgc2VnbWVudCBib3VuZGFyeQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgVGV4dFJ1blNlZ21lbnQgc2VnID0gYnJlYWtlci5ydW5TZWdtZW50cy5nZXQoYnJlYWtlci5sb2dpY2FsMnNlZ21lbnRbbmV3SW5mbwotICAgICAgICAgICAgICAgICAgICAuZ2V0Q2hhckluZGV4KCldKTsKLSAgICAgICAgICAgIGlmICghc2VnLmNoYXJIYXNaZXJvQWR2YW5jZShuZXdJbmZvLmdldENoYXJJbmRleCgpKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBuZXdJbmZvOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRm9yIGVhY2ggdmlzdWFsIGNhcmV0IHBvc2l0aW9uIHRoZXJlIGFyZSB0d28gaGl0cy4gRm9yIHRoZSBzaW1wbGUgTFRSIHRleHQgb25lIGlzCi0gICAgICogYSB0cmFpbGluZyBvZiB0aGUgcHJldmlvdXMgY2hhciBhbmQgYW5vdGhlciBpcyB0aGUgbGVhZGluZyBvZiB0aGUgbmV4dCBjaGFyLiBUaGlzCi0gICAgICogbWV0aG9kIHJldHVybnMgdGhlIG9wcG9zaXRlIGhpdCBmb3IgdGhlIGdpdmVuIGhpdC4KLSAgICAgKiBAcGFyYW0gaGl0SW5mbyAtIGdpdmVuIGhpdAotICAgICAqIEByZXR1cm4gb3Bwb3NpdGUgaGl0Ci0gICAgICovCi0gICAgcHVibGljIFRleHRIaXRJbmZvIGdldFZpc3VhbE90aGVySGl0KFRleHRIaXRJbmZvIGhpdEluZm8pIHsKLSAgICAgICAgY2hlY2tIaXQoaGl0SW5mbyk7Ci0KLSAgICAgICAgaW50IGlkeCA9IGhpdEluZm8uZ2V0Q2hhckluZGV4KCk7Ci0KLSAgICAgICAgaW50IHJlc0lkeDsKLSAgICAgICAgYm9vbGVhbiByZXNJc0xlYWRpbmc7Ci0KLSAgICAgICAgaWYgKGlkeCA+PSAwICYmIGlkeCA8IGJyZWFrZXIuZ2V0Q2hhckNvdW50KCkpIHsgLy8gSGl0IGluZm8gaW4gdGhlIG1pZGRsZQotICAgICAgICAgICAgaW50IHZpc3VhbCA9IGJyZWFrZXIuZ2V0VmlzdWFsRnJvbUxvZ2ljYWwoaWR4KTsKLQotICAgICAgICAgICAgLy8gQ2hhciBpcyBMVFIgKyBMRUFESU5HIGluZm8KLSAgICAgICAgICAgIGlmICgoKGJyZWFrZXIuZ2V0TGV2ZWwoaWR4KSAmIDB4MSkgPT0gMHgwKSBeIGhpdEluZm8uaXNMZWFkaW5nRWRnZSgpKSB7Ci0gICAgICAgICAgICAgICAgdmlzdWFsKys7Ci0gICAgICAgICAgICAgICAgaWYgKHZpc3VhbCA9PSBicmVha2VyLmdldENoYXJDb3VudCgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChicmVha2VyLmlzTFRSKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlc0lkeCA9IGJyZWFrZXIuZ2V0Q2hhckNvdW50KCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXNJc0xlYWRpbmcgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgcmVzSWR4ID0gLTE7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXNJc0xlYWRpbmcgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHJlc0lkeCA9IGJyZWFrZXIuZ2V0TG9naWNhbEZyb21WaXN1YWwodmlzdWFsKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKChicmVha2VyLmdldExldmVsKHJlc0lkeCkgJiAweDEpID09IDB4MCkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmVzSXNMZWFkaW5nID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlc0lzTGVhZGluZyA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICB2aXN1YWwtLTsKLSAgICAgICAgICAgICAgICBpZiAodmlzdWFsID09IC0xKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChicmVha2VyLmlzTFRSKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlc0lkeCA9IC0xOwotICAgICAgICAgICAgICAgICAgICAgICAgcmVzSXNMZWFkaW5nID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXNJZHggPSBicmVha2VyLmdldENoYXJDb3VudCgpOwotICAgICAgICAgICAgICAgICAgICAgICAgcmVzSXNMZWFkaW5nID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHJlc0lkeCA9IGJyZWFrZXIuZ2V0TG9naWNhbEZyb21WaXN1YWwodmlzdWFsKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKChicmVha2VyLmdldExldmVsKHJlc0lkeCkgJiAweDEpID09IDB4MCkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmVzSXNMZWFkaW5nID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXNJc0xlYWRpbmcgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgaWYgKGlkeCA8IDApIHsgLy8gYmVmb3JlICJzdGFydCIKLSAgICAgICAgICAgIGlmIChicmVha2VyLmlzTFRSKCkpIHsKLSAgICAgICAgICAgICAgICByZXNJZHggPSBicmVha2VyLmdldExvZ2ljYWxGcm9tVmlzdWFsKDApOwotICAgICAgICAgICAgICAgIHJlc0lzTGVhZGluZyA9IChicmVha2VyLmdldExldmVsKHJlc0lkeCkgJiAweDEpID09IDB4MDsgLy8gTFRSIGNoYXI/Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJlc0lkeCA9IGJyZWFrZXIuZ2V0TG9naWNhbEZyb21WaXN1YWwoYnJlYWtlci5nZXRDaGFyQ291bnQoKSAtIDEpOwotICAgICAgICAgICAgICAgIHJlc0lzTGVhZGluZyA9IChicmVha2VyLmdldExldmVsKHJlc0lkeCkgJiAweDEpICE9IDB4MDsgLy8gUlRMIGNoYXI/Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7IC8vIGlkeCA9PSBicmVha2VyLmdldENoYXJDb3VudCgpCi0gICAgICAgICAgICBpZiAoYnJlYWtlci5pc0xUUigpKSB7Ci0gICAgICAgICAgICAgICAgcmVzSWR4ID0gYnJlYWtlci5nZXRMb2dpY2FsRnJvbVZpc3VhbChicmVha2VyLmdldENoYXJDb3VudCgpIC0gMSk7Ci0gICAgICAgICAgICAgICAgcmVzSXNMZWFkaW5nID0gKGJyZWFrZXIuZ2V0TGV2ZWwocmVzSWR4KSAmIDB4MSkgIT0gMHgwOyAvLyBMVFIgY2hhcj8KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgcmVzSWR4ID0gYnJlYWtlci5nZXRMb2dpY2FsRnJvbVZpc3VhbCgwKTsKLSAgICAgICAgICAgICAgICByZXNJc0xlYWRpbmcgPSAoYnJlYWtlci5nZXRMZXZlbChyZXNJZHgpICYgMHgxKSA9PSAweDA7IC8vIFJUTCBjaGFyPwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHJlc0lzTGVhZGluZyA/IFRleHRIaXRJbmZvLmxlYWRpbmcocmVzSWR4KSA6IFRleHRIaXRJbmZvLnRyYWlsaW5nKHJlc0lkeCk7Ci0gICAgfQotCi0gICAgcHVibGljIExpbmUyRCBnZXRDYXJldFNoYXBlKFRleHRIaXRJbmZvIGhpdEluZm8sIFRleHRMYXlvdXQgbGF5b3V0KSB7Ci0gICAgICAgIHJldHVybiBnZXRDYXJldFNoYXBlKGhpdEluZm8sIGxheW91dCwgdHJ1ZSwgZmFsc2UsIG51bGwpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBjYXJldCBzaGFwZS4KLSAgICAgKiBAcGFyYW0gaGl0SW5mbyAtIGhpdCB3aGVyZSB0byBwbGFjZSBhIGNhcmV0Ci0gICAgICogQHBhcmFtIGxheW91dCAtIHRleHQgbGF5b3V0Ci0gICAgICogQHBhcmFtIHVzZUl0YWxpYyAtIHVudXNlZCBmb3Igbm93LCB3YXMgdXNlZCB0byBjcmVhdGUKLSAgICAgKiBzbGFudGVkIGNhcmV0cyBmb3IgaXRhbGljIHRleHQKLSAgICAgKiBAcGFyYW0gdXNlQm91bmRzIC0gdHJ1ZSBpZiB0aGUgY2FyZWQgc2hvdWxkIGZpdCBpbnRvIHRoZSBwcm92aWRlZCBib3VuZHMKLSAgICAgKiBAcGFyYW0gYm91bmRzIC0gYm91bmRzIGZvciB0aGUgY2FyZXQKLSAgICAgKiBAcmV0dXJuIGNhcmV0IHNoYXBlCi0gICAgICovCi0gICAgcHVibGljIExpbmUyRCBnZXRDYXJldFNoYXBlKAotICAgICAgICAgICAgVGV4dEhpdEluZm8gaGl0SW5mbywgVGV4dExheW91dCBsYXlvdXQsCi0gICAgICAgICAgICBib29sZWFuIHVzZUl0YWxpYywgYm9vbGVhbiB1c2VCb3VuZHMsIFJlY3RhbmdsZTJEIGJvdW5kcwotICAgICkgewotICAgICAgICBjaGVja0hpdChoaXRJbmZvKTsKLQotICAgICAgICBmbG9hdCB4MSwgeDIsIHkxLCB5MjsKLQotICAgICAgICBpbnQgY2hhcklkeCA9IGhpdEluZm8uZ2V0Q2hhckluZGV4KCk7Ci0KLSAgICAgICAgaWYgKGNoYXJJZHggPj0gMCAmJiBjaGFySWR4IDwgYnJlYWtlci5nZXRDaGFyQ291bnQoKSkgewotICAgICAgICAgICAgVGV4dFJ1blNlZ21lbnQgc2VnbWVudCA9IGJyZWFrZXIucnVuU2VnbWVudHMuZ2V0KGJyZWFrZXIubG9naWNhbDJzZWdtZW50W2NoYXJJZHhdKTsKLSAgICAgICAgICAgIHkxID0gc2VnbWVudC5tZXRyaWNzLmRlc2NlbnQ7Ci0gICAgICAgICAgICB5MiA9IC0gc2VnbWVudC5tZXRyaWNzLmFzY2VudCAtIHNlZ21lbnQubWV0cmljcy5sZWFkaW5nOwotCi0gICAgICAgICAgICB4MSA9IHgyID0gc2VnbWVudC5nZXRDaGFyUG9zaXRpb24oY2hhcklkeCkgKyAoaGl0SW5mby5pc0xlYWRpbmdFZGdlKCkgPwotICAgICAgICAgICAgICAgICAgICAwIDogc2VnbWVudC5nZXRDaGFyQWR2YW5jZShjaGFySWR4KSk7Ci0gICAgICAgICAgICAvLyBEZWNpZGVkIHRoYXQgc3RyYWlnaHQgY3Vyc29yIGxvb2tzIGJldHRlciBldmVuIGZvciBpdGFsaWMgZm9udHMsCi0gICAgICAgICAgICAvLyBlc3BlY2lhbGx5IGNvbWJpbmVkIHdpdGggaGlnaGxpZ2h0aW5nCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgLy8gTm90IGdyYXBoaWNzLCBuZWVkIHRvIGNoZWNrIGl0YWxpYyBhbmdsZSBhbmQgYmFzZWxpbmUKLSAgICAgICAgICAgIGlmIChsYXlvdXQuZ2V0QmFzZWxpbmUoKSA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHNlZ21lbnQubWV0cmljcy5pdGFsaWNBbmdsZSAhPSAwICYmIHVzZUl0YWxpYykgewotICAgICAgICAgICAgICAgICAgICB4MSAtPSBzZWdtZW50Lm1ldHJpY3MuaXRhbGljQW5nbGUgKiBzZWdtZW50Lm1ldHJpY3MuZGVzY2VudDsKLSAgICAgICAgICAgICAgICAgICAgeDIgKz0gc2VnbWVudC5tZXRyaWNzLml0YWxpY0FuZ2xlICoKLSAgICAgICAgICAgICAgICAgICAgICAgIChzZWdtZW50Lm1ldHJpY3MuYXNjZW50ICsgc2VnbWVudC5tZXRyaWNzLmxlYWRpbmcpOwotCi0gICAgICAgICAgICAgICAgICAgIGZsb2F0IGJhc2VsaW5lT2Zmc2V0ID0KLSAgICAgICAgICAgICAgICAgICAgICAgIGxheW91dC5nZXRCYXNlbGluZU9mZnNldHMoKVtsYXlvdXQuZ2V0QmFzZWxpbmUoKV07Ci0gICAgICAgICAgICAgICAgICAgIHkxICs9IGJhc2VsaW5lT2Zmc2V0OwotICAgICAgICAgICAgICAgICAgICB5MiArPSBiYXNlbGluZU9mZnNldDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAqLwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgeTEgPSBsYXlvdXQuZ2V0RGVzY2VudCgpOwotICAgICAgICAgICAgeTIgPSAtIGxheW91dC5nZXRBc2NlbnQoKSAtIGxheW91dC5nZXRMZWFkaW5nKCk7Ci0gICAgICAgICAgICB4MSA9IHgyID0gKChicmVha2VyLmdldEJhc2VMZXZlbCgpICYgMHgxKSA9PSAwIF4gY2hhcklkeCA8IDApID8KLSAgICAgICAgICAgICAgICAgICAgbGF5b3V0LmdldEFkdmFuY2UoKSA6IDA7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAodXNlQm91bmRzKSB7Ci0gICAgICAgICAgICB5MSA9IChmbG9hdCkgYm91bmRzLmdldE1heFkoKTsKLSAgICAgICAgICAgIHkyID0gKGZsb2F0KSBib3VuZHMuZ2V0TWluWSgpOwotCi0gICAgICAgICAgICBpZiAoeDIgPiBib3VuZHMuZ2V0TWF4WCgpKSB7Ci0gICAgICAgICAgICAgICAgeDEgPSB4MiA9IChmbG9hdCkgYm91bmRzLmdldE1heFgoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh4MSA8IGJvdW5kcy5nZXRNaW5YKCkpIHsKLSAgICAgICAgICAgICAgICB4MSA9IHgyID0gKGZsb2F0KSBib3VuZHMuZ2V0TWluWCgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBMaW5lMkQuRmxvYXQoeDEsIHkxLCB4MiwgeTIpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgY2FyZXQgc2hhcGVzIGZvciB0aGUgc3BlY2lmaWVkIG9mZnNldC4gT24gdGhlIGJvdW5kYXJpZXMgd2hlcmUKLSAgICAgKiB0aGUgdGV4dCBpcyBjaGFuZ2luZyBpdHMgZGlyZWN0aW9uIHRoaXMgbWV0aG9kIG1heSByZXR1cm4gdHdvIHNoYXBlcwotICAgICAqIGZvciB0aGUgc3Ryb25nIGFuZCB0aGUgd2VhayBjYXJldHMsIGluIG90aGVyIGNhc2VzIGl0IHdvdWxkIHJldHVybiBvbmUuCi0gICAgICogQHBhcmFtIG9mZnNldCAtIG9mZnNldCBpbiB0aGUgdGV4dC4KLSAgICAgKiBAcGFyYW0gYm91bmRzIC0gYm91bmRzIHRvIGZpdCB0aGUgY2FyZXRzIGludG8KLSAgICAgKiBAcGFyYW0gcG9saWN5IC0gY2FyZXQgcG9saWN5Ci0gICAgICogQHBhcmFtIGxheW91dCAtIHRleHQgbGF5b3V0Ci0gICAgICogQHJldHVybiBvbmUgb3IgdHdvIGNhcmV0IHNoYXBlcwotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZVtdIGdldENhcmV0U2hhcGVzKAotICAgICAgICAgICAgaW50IG9mZnNldCwgUmVjdGFuZ2xlMkQgYm91bmRzLAotICAgICAgICAgICAgVGV4dExheW91dC5DYXJldFBvbGljeSBwb2xpY3ksIFRleHRMYXlvdXQgbGF5b3V0Ci0gICAgKSB7Ci0gICAgICAgIFRleHRIaXRJbmZvIGhpdDEgPSBUZXh0SGl0SW5mby5hZnRlck9mZnNldChvZmZzZXQpOwotICAgICAgICBUZXh0SGl0SW5mbyBoaXQyID0gZ2V0VmlzdWFsT3RoZXJIaXQoaGl0MSk7Ci0KLSAgICAgICAgU2hhcGUgY2FyZXQxID0gZ2V0Q2FyZXRTaGFwZShoaXQxLCBsYXlvdXQpOwotCi0gICAgICAgIGlmIChnZXRWaXN1YWxGcm9tSGl0SW5mbyhoaXQxKSA9PSBnZXRWaXN1YWxGcm9tSGl0SW5mbyhoaXQyKSkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBTaGFwZVtdIHtjYXJldDEsIG51bGx9OwotICAgICAgICB9Ci0gICAgICAgIFNoYXBlIGNhcmV0MiA9IGdldENhcmV0U2hhcGUoaGl0MiwgbGF5b3V0KTsKLQotICAgICAgICBUZXh0SGl0SW5mbyBzdHJvbmdIaXQgPSBwb2xpY3kuZ2V0U3Ryb25nQ2FyZXQoaGl0MSwgaGl0MiwgbGF5b3V0KTsKLSAgICAgICAgcmV0dXJuIHN0cm9uZ0hpdC5lcXVhbHMoaGl0MSkgPwotICAgICAgICAgICAgICAgIG5ldyBTaGFwZVtdIHtjYXJldDEsIGNhcmV0Mn0gOgotICAgICAgICAgICAgICAgIG5ldyBTaGFwZVtdIHtjYXJldDIsIGNhcmV0MX07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ29ubmVjdHMgdHdvIGNhcmV0cyB0byBwcm9kdWNlIGEgaGlnaGxpZ2h0IHNoYXBlLgotICAgICAqIEBwYXJhbSBjYXJldDEgLSAxc3QgY2FyZXQKLSAgICAgKiBAcGFyYW0gY2FyZXQyIC0gMm5kIGNhcmV0Ci0gICAgICogQHJldHVybiBoaWdobGlnaHQgc2hhcGUKLSAgICAgKi8KLSAgICBHZW5lcmFsUGF0aCBjb25uZWN0Q2FyZXRzKExpbmUyRCBjYXJldDEsIExpbmUyRCBjYXJldDIpIHsKLSAgICAgICAgR2VuZXJhbFBhdGggcGF0aCA9IG5ldyBHZW5lcmFsUGF0aChHZW5lcmFsUGF0aC5XSU5EX05PTl9aRVJPKTsKLSAgICAgICAgcGF0aC5tb3ZlVG8oKGZsb2F0KSBjYXJldDEuZ2V0WDEoKSwgKGZsb2F0KSBjYXJldDEuZ2V0WTEoKSk7Ci0gICAgICAgIHBhdGgubGluZVRvKChmbG9hdCkgY2FyZXQyLmdldFgxKCksIChmbG9hdCkgY2FyZXQyLmdldFkxKCkpOwotICAgICAgICBwYXRoLmxpbmVUbygoZmxvYXQpIGNhcmV0Mi5nZXRYMigpLCAoZmxvYXQpIGNhcmV0Mi5nZXRZMigpKTsKLSAgICAgICAgcGF0aC5saW5lVG8oKGZsb2F0KSBjYXJldDEuZ2V0WDIoKSwgKGZsb2F0KSBjYXJldDEuZ2V0WTIoKSk7Ci0KLSAgICAgICAgcGF0aC5jbG9zZVBhdGgoKTsKLQotICAgICAgICByZXR1cm4gcGF0aDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGEgaGlnaGxpZ2h0IHNoYXBlIGZyb20gZ2l2ZW4gdHdvIGhpdHMuIFRoaXMgc2hhcGUKLSAgICAgKiB3aWxsIGFsd2F5cyBiZSB2aXN1YWxseSBjb250aWd1b3VzCi0gICAgICogQHBhcmFtIGhpdDEgLSAxc3QgaGl0Ci0gICAgICogQHBhcmFtIGhpdDIgLSAybmQgaGl0Ci0gICAgICogQHBhcmFtIGJvdW5kcyAtIGJvdW5kcyB0byBmaXQgdGhlIHNoYXBlIGludG8KLSAgICAgKiBAcGFyYW0gbGF5b3V0IC0gdGV4dCBsYXlvdXQKLSAgICAgKiBAcmV0dXJuIGhpZ2hsaWdodCBzaGFwZQotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZSBnZXRWaXN1YWxIaWdobGlnaHRTaGFwZSgKLSAgICAgICAgICAgIFRleHRIaXRJbmZvIGhpdDEsIFRleHRIaXRJbmZvIGhpdDIsCi0gICAgICAgICAgICBSZWN0YW5nbGUyRCBib3VuZHMsIFRleHRMYXlvdXQgbGF5b3V0Ci0gICAgKSB7Ci0gICAgICAgIGNoZWNrSGl0KGhpdDEpOwotICAgICAgICBjaGVja0hpdChoaXQyKTsKLQotICAgICAgICBMaW5lMkQgY2FyZXQxID0gZ2V0Q2FyZXRTaGFwZShoaXQxLCBsYXlvdXQsIGZhbHNlLCB0cnVlLCBib3VuZHMpOwotICAgICAgICBMaW5lMkQgY2FyZXQyID0gZ2V0Q2FyZXRTaGFwZShoaXQyLCBsYXlvdXQsIGZhbHNlLCB0cnVlLCBib3VuZHMpOwotCi0gICAgICAgIHJldHVybiBjb25uZWN0Q2FyZXRzKGNhcmV0MSwgY2FyZXQyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTdXBwb3NlIHRoYXQgdGhlIHVzZXIgdmlzdWFsbHkgc2VsZWN0ZWQgYSBibG9jayBvZiB0ZXh0IHdoaWNoIGhhcwotICAgICAqIHNldmVyYWwgZGlmZmVyZW50IGxldmVscyAobWl4ZWQgUlRMIGFuZCBMVFIpLCBzbywgaW4gdGhlIGxvZ2ljYWwKLSAgICAgKiByZXByZXNlbnRhdGlvbiBvZiB0aGUgdGV4dCB0aGlzIHNlbGVjdGlvbiBtYXkgYmUgbm90IGNvbnRpZ291cy4KLSAgICAgKiBUaGlzIG1ldGhvZHMgcmV0dXJucyBhIHNldCBvZiBsb2dpY2FsIHJhbmdlcyBmb3IgdGhlIGFyYml0cmFyeQotICAgICAqIHZpc3VhbCBzZWxlY3Rpb24gcmVwcmVzZW50ZWQgYnkgdHdvIGhpdHMuCi0gICAgICogQHBhcmFtIGhpdDEgLSAxc3QgaGl0Ci0gICAgICogQHBhcmFtIGhpdDIgLSAybmQgaGl0Ci0gICAgICogQHJldHVybiBsb2dpY2FsIHJhbmdlcyBmb3IgdGhlIHNlbGVjdGlvbgotICAgICAqLwotICAgIHB1YmxpYyBpbnRbXSBnZXRMb2dpY2FsUmFuZ2VzRm9yVmlzdWFsU2VsZWN0aW9uKFRleHRIaXRJbmZvIGhpdDEsIFRleHRIaXRJbmZvIGhpdDIpIHsKLSAgICAgICAgY2hlY2tIaXQoaGl0MSk7Ci0gICAgICAgIGNoZWNrSGl0KGhpdDIpOwotCi0gICAgICAgIGludCB2aXN1YWwxID0gZ2V0VmlzdWFsRnJvbUhpdEluZm8oaGl0MSk7Ci0gICAgICAgIGludCB2aXN1YWwyID0gZ2V0VmlzdWFsRnJvbUhpdEluZm8oaGl0Mik7Ci0KLSAgICAgICAgaWYgKHZpc3VhbDEgPiB2aXN1YWwyKSB7Ci0gICAgICAgICAgICBpbnQgdG1wID0gdmlzdWFsMjsKLSAgICAgICAgICAgIHZpc3VhbDIgPSB2aXN1YWwxOwotICAgICAgICAgICAgdmlzdWFsMSA9IHRtcDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIE1heCBsZXZlbCBpcyAyNTUsIHNvIHdlIGRvbid0IG5lZWQgbW9yZSB0aGFuIDUxMiBlbnRyaWVzCi0gICAgICAgIGludCByZXN1bHRzW10gPSBuZXcgaW50WzUxMl07Ci0KLSAgICAgICAgaW50IHByZXZMb2dpY2FsLCBsb2dpY2FsLCBydW5TdGFydCwgbnVtUnVucyA9IDA7Ci0KLSAgICAgICAgbG9naWNhbCA9IHJ1blN0YXJ0ID0gcHJldkxvZ2ljYWwgPSBicmVha2VyLmdldExvZ2ljYWxGcm9tVmlzdWFsKHZpc3VhbDEpOwotCi0gICAgICAgIC8vIEdldCBhbGwgdGhlIHJ1bnMuIFdlIHVzZSB0aGUgZmFjdCB0aGF0IGRpcmVjdGlvbiBpcyBjb25zdGFudCBpbiBhbGwgcnVucy4KLSAgICAgICAgZm9yIChpbnQgaT12aXN1YWwxKzE7IGk8PXZpc3VhbDI7IGkrKykgewotICAgICAgICAgICAgbG9naWNhbCA9IGJyZWFrZXIuZ2V0TG9naWNhbEZyb21WaXN1YWwoaSk7Ci0gICAgICAgICAgICBpbnQgZGlmZiA9IGxvZ2ljYWwtcHJldkxvZ2ljYWw7Ci0KLSAgICAgICAgICAgIC8vIFN0YXJ0IG9mIHRoZSBuZXh0IHJ1biBlbmNvdW50ZXJlZAotICAgICAgICAgICAgaWYgKGRpZmYgPiAxIHx8IGRpZmYgPCAtMSkgewotICAgICAgICAgICAgICAgIHJlc3VsdHNbKG51bVJ1bnMpKjJdID0gTWF0aC5taW4ocnVuU3RhcnQsIHByZXZMb2dpY2FsKTsKLSAgICAgICAgICAgICAgICByZXN1bHRzWyhudW1SdW5zKSoyICsgMV0gPSBNYXRoLm1heChydW5TdGFydCwgcHJldkxvZ2ljYWwpOwotICAgICAgICAgICAgICAgIG51bVJ1bnMrKzsKLSAgICAgICAgICAgICAgICBydW5TdGFydCA9IGxvZ2ljYWw7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHByZXZMb2dpY2FsID0gbG9naWNhbDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIFRoZSBsYXN0IHVuc2F2ZWQgcnVuCi0gICAgICAgIHJlc3VsdHNbKG51bVJ1bnMpKjJdID0gTWF0aC5taW4ocnVuU3RhcnQsIGxvZ2ljYWwpOwotICAgICAgICByZXN1bHRzWyhudW1SdW5zKSoyICsgMV0gPSBNYXRoLm1heChydW5TdGFydCwgbG9naWNhbCk7Ci0gICAgICAgIG51bVJ1bnMrKzsKLQotICAgICAgICBpbnQgcmV0dmFsW10gPSBuZXcgaW50W251bVJ1bnMqMl07Ci0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkocmVzdWx0cywgMCwgcmV0dmFsLCAwLCBudW1SdW5zKjIpOwotICAgICAgICByZXR1cm4gcmV0dmFsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYSBoaWdobGlnaHQgc2hhcGUgZnJvbSBnaXZlbiB0d28gZW5kcG9pbnRzIGluIHRoZSBsb2dpY2FsCi0gICAgICogcmVwcmVzZW50YXRpb24uIFRoaXMgc2hhcGUgaXMgbm90IGFsd2F5cyB2aXN1YWxseSBjb250aWd1b3VzCi0gICAgICogQHBhcmFtIGZpcnN0RW5kcG9pbnQgLSAxc3QgbG9naWNhbCBlbmRwb2ludAotICAgICAqIEBwYXJhbSBzZWNvbmRFbmRwb2ludCAtIDJuZCBsb2dpY2FsIGVuZHBvaW50Ci0gICAgICogQHBhcmFtIGJvdW5kcyAtIGJvdW5kcyB0byBmaXQgdGhlIHNoYXBlIGludG8KLSAgICAgKiBAcGFyYW0gbGF5b3V0IC0gdGV4dCBsYXlvdXQKLSAgICAgKiBAcmV0dXJuIGhpZ2hsaWdodCBzaGFwZQotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZSBnZXRMb2dpY2FsSGlnaGxpZ2h0U2hhcGUoCi0gICAgICAgICAgICBpbnQgZmlyc3RFbmRwb2ludCwgaW50IHNlY29uZEVuZHBvaW50LAotICAgICAgICAgICAgUmVjdGFuZ2xlMkQgYm91bmRzLCBUZXh0TGF5b3V0IGxheW91dAotICAgICkgewotICAgICAgICBHZW5lcmFsUGF0aCByZXMgPSBuZXcgR2VuZXJhbFBhdGgoKTsKLQotICAgICAgICBmb3IgKGludCBpPWZpcnN0RW5kcG9pbnQ7IGk8PXNlY29uZEVuZHBvaW50OyBpKyspIHsKLSAgICAgICAgICAgIGludCBlbmRSdW4gPSBicmVha2VyLmdldExldmVsUnVuTGltaXQoaSwgc2Vjb25kRW5kcG9pbnQpOwotICAgICAgICAgICAgVGV4dEhpdEluZm8gaGl0MSA9IFRleHRIaXRJbmZvLmxlYWRpbmcoaSk7Ci0gICAgICAgICAgICBUZXh0SGl0SW5mbyBoaXQyID0gVGV4dEhpdEluZm8udHJhaWxpbmcoZW5kUnVuLTEpOwotCi0gICAgICAgICAgICBMaW5lMkQgY2FyZXQxID0gZ2V0Q2FyZXRTaGFwZShoaXQxLCBsYXlvdXQsIGZhbHNlLCB0cnVlLCBib3VuZHMpOwotICAgICAgICAgICAgTGluZTJEIGNhcmV0MiA9IGdldENhcmV0U2hhcGUoaGl0MiwgbGF5b3V0LCBmYWxzZSwgdHJ1ZSwgYm91bmRzKTsKLQotICAgICAgICAgICAgcmVzLmFwcGVuZChjb25uZWN0Q2FyZXRzKGNhcmV0MSwgY2FyZXQyKSwgZmFsc2UpOwotCi0gICAgICAgICAgICBpID0gZW5kUnVuOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0NvbW1vbkdseXBoVmVjdG9yLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0NvbW1vbkdseXBoVmVjdG9yLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQwNDBhNjAuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Db21tb25HbHlwaFZlY3Rvci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsOTU0ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5Gb250OwotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLWltcG9ydCBqYXZhLmF3dC5TaGFwZTsKLWltcG9ydCBqYXZhLmF3dC5mb250LkZvbnRSZW5kZXJDb250ZXh0OwotaW1wb3J0IGphdmEuYXd0LmZvbnQuR2x5cGhKdXN0aWZpY2F0aW9uSW5mbzsKLWltcG9ydCBqYXZhLmF3dC5mb250LkdseXBoTWV0cmljczsKLWltcG9ydCBqYXZhLmF3dC5mb250LkdseXBoVmVjdG9yOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uR2VuZXJhbFBhdGg7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5Qb2ludDJEOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBHbHlwaFZlY3RvciBpbXBsZW1lbnRhdGlvbgotICovCi1wdWJsaWMgY2xhc3MgQ29tbW9uR2x5cGhWZWN0b3IgZXh0ZW5kcyBHbHlwaFZlY3RvciB7Ci0KLSAgICAvLyBhcnJheSBvZiB0cmFuc2Zvcm1zIG9mIGdseXBocyBpbiBHbHlwaFZlY3RvcgotICAgIHByb3RlY3RlZCBBZmZpbmVUcmFuc2Zvcm1bXSBnbHNUcmFuc2Zvcm1zOwotCi0gICAgLy8gYXJyYXkgb2YgY2hhcnMgZGVmaW5lZCBpbiBjb25zdHJ1Y3RvcgotICAgIHB1YmxpYyBjaGFyW10gY2hhclZlY3RvcjsKLQotICAgIC8vIGFycmF5IG9mIEdseXBoIG9iamVjdHMsIHRoYXQgZGVzY3JpYmUgaW5mb3JtYXRpb24gYWJvdXQgZ2x5cGhzCi0gICAgcHVibGljIEdseXBoW10gdmVjdG9yOwotCi0gICAgLy8gYXJyYXkgb2YgZGVmYXVsdCBwb3NpdGlvbnMgb2YgZ2x5cGhzIGluIEdseXBoVmVjdG9yCi0gICAgLy8gd2l0aG91dCBhcHBseWluZyBHbHlwaFZlY3RvcidzIHRyYW5zZm9ybQotICAgIGZsb2F0W10gZGVmYXVsdFBvc2l0aW9uczsKLQotICAgIC8vIGFycmF5IG9mIGxvZ2ljYWwgcG9zaXRpb25zIG9mIGdseXBocyBpbiBHbHlwaFZlY3RvcgotCi0gICAgZmxvYXRbXSBsb2dpY2FsUG9zaXRpb25zOwotCi0gICAgLy8gYXJyYXkgb2YgdmlzdWFsIChyZWFsKSBwb3NpdGlvbnMgb2YgZ2x5cGhzIGluIEdseXBoVmVjdG9yCi0gICAgcHVibGljIGZsb2F0W10gdmlzdWFsUG9zaXRpb25zOwotCi0gICAgLy8gRm9udFJlbmRlckNvbnRleHQgZm9yIHRoaXMgdmVjdG9yLgotICAgIHByb3RlY3RlZCBGb250UmVuZGVyQ29udGV4dCB2ZWN0b3JGUkM7Ci0KLSAgICAvLyBsYXlvdXQgZmxhZ3MgbWFzawotICAgIHByb3RlY3RlZCBpbnQgbGF5b3V0RmxhZ3MgPSAwOwotCi0gICAgLy8gYXJyYXkgb2YgY2FjaGVkIGdseXBoIG91dGxpbmVzIAotICAgIHByb3RlY3RlZCBTaGFwZVtdIGd2U2hhcGVzOwotCi0gICAgRm9udFBlZXJJbXBsIHBlZXI7Ci0KLSAgICAvLyBmb250IGNvcnJlc3BvbmRpbmcgdG8gdGhlIEdseXBoVmVjdG9yIAotICAgIEZvbnQgZm9udDsKLQotICAgIC8vIGFzY2VudCBvZiB0aGUgZm9udAotICAgIGZsb2F0IGFzY2VudDsKLQotICAgIC8vIGhlaWdodCBvZiB0aGUgZm9udAotICAgIGZsb2F0IGhlaWdodDsKLSAgICAKLSAgICAvLyBsZWFkaW5nIG9mIHRoZSBmb250Ci0gICAgZmxvYXQgbGVhZGluZzsKLSAgICAKLSAgICAvLyBkZXNjZW50IG9mIHRoZSBmb250Ci0gICAgZmxvYXQgZGVzY2VudDsKLQotICAgIC8vIHRyYW5zZm9ybSBvZiB0aGUgR2x5cGhWZWN0b3IKLSAgICBBZmZpbmVUcmFuc2Zvcm0gdHJhbnNmb3JtOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBuZXcgQ29tbW9uR2x5cGhWZWN0b3Igb2JqZWN0IGZyb20gdGhlIHNwZWNpZmllZCBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaGFycyBhbiBhcnJheSBvZiBjaGFycwotICAgICAqIEBwYXJhbSBmcmMgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0Ci0gICAgICogQHBhcmFtIGZudCBGb250IG9iamVjdAotICAgICAqIEBwYXJhbSBmbGFncyBsYXlvdXQgZmxhZ3MKLSAgICAgKi8KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygiZGVwcmVjYXRpb24iKQotICAgIHB1YmxpYyBDb21tb25HbHlwaFZlY3RvcihjaGFyW10gY2hhcnMsIEZvbnRSZW5kZXJDb250ZXh0IGZyYywgRm9udCBmbnQsCi0gICAgICAgICAgICBpbnQgZmxhZ3MpIHsKLSAgICAgICAgaW50IGxlbiA9IGNoYXJzLmxlbmd0aDsKLQotICAgICAgICB0aGlzLmZvbnQgPSBmbnQ7Ci0gICAgICAgIHRoaXMudHJhbnNmb3JtID0gZm50LmdldFRyYW5zZm9ybSgpOwotICAgICAgICB0aGlzLnBlZXIgPSAoRm9udFBlZXJJbXBsKSBmbnQuZ2V0UGVlcigpOwotCi0gICAgICAgIGd2U2hhcGVzID0gbmV3IFNoYXBlW2xlbl07Ci0KLSAgICAgICAgLy8gISEgQXMgcG9pbnRlZCBpbiBBUEkgZG9jdW1lbnRhdGlvbiBmb3IgdGhlIAotICAgICAgICAvLyBnZXRHbHlwaFBvc2lzaXRpb25zKGludCBpbmRleCxpbnQgbnVtRW50cmllcywgZmxvYXRbXSBwb3NpdGlvblJldHVybikgCi0gICAgICAgIC8vIGFuZCBnZXRHbHlwaFBvc2l0aW9uKGludCBpbmRleCkgbWV0aG9kcywgaWYgdGhlIGluZGV4IGlzIGVxdWFscyB0byAKLSAgICAgICAgLy8gdGhlIG51bWJlciBvZiBnbHlwaHMgdGhlIHBvc2l0aW9uIGFmdGVyIHRoZSBsYXN0IGdseXBoIG11c3QgYmUgCi0gICAgICAgIC8vIHJldHVybmVkLCB0aHVzIHRoZXJlIGFyZSBuKzEgcG9zaXRpb25zIGFuZCBsYXN0IChuKzEpIHBvc2l0aW9uIAotICAgICAgICAvLyBwb2ludHMgdG8gdGhlIGVuZCBvZiBHbHlwaFZlY3Rvci4KLQotICAgICAgICBsb2dpY2FsUG9zaXRpb25zID0gbmV3IGZsb2F0WyhsZW4rMSk8PDFdOwotICAgICAgICB2aXN1YWxQb3NpdGlvbnMgPSBuZXcgZmxvYXRbKGxlbisxKTw8MV07Ci0gICAgICAgIGRlZmF1bHRQb3NpdGlvbnMgPSBuZXcgZmxvYXRbKGxlbisxKTw8MV07Ci0KLSAgICAgICAgZ2xzVHJhbnNmb3JtcyA9IG5ldyBBZmZpbmVUcmFuc2Zvcm1bbGVuXTsKLQotICAgICAgICB0aGlzLmNoYXJWZWN0b3IgPSBjaGFyczsKLSAgICAgICAgdGhpcy52ZWN0b3JGUkMgPSBmcmM7Ci0gICAgICAgIC8vTGluZU1ldHJpY3NJbXBsIGxtSW1wbCA9IChMaW5lTWV0cmljc0ltcGwpcGVlci5nZXRMaW5lTWV0cmljcygpOwotCi0gICAgICAgIExpbmVNZXRyaWNzSW1wbCBsbUltcGwgPSAoTGluZU1ldHJpY3NJbXBsKWZudC5nZXRMaW5lTWV0cmljcyhTdHJpbmcudmFsdWVPZihjaGFycyksIGZyYyk7Ci0KLSAgICAgICAgdGhpcy5hc2NlbnQgPSBsbUltcGwuZ2V0QXNjZW50KCk7Ci0gICAgICAgIHRoaXMuaGVpZ2h0ID0gbG1JbXBsLmdldEhlaWdodCgpOwotICAgICAgICB0aGlzLmxlYWRpbmcgPSBsbUltcGwuZ2V0TGVhZGluZygpOwotICAgICAgICB0aGlzLmRlc2NlbnQgPSBsbUltcGwuZ2V0RGVzY2VudCgpOwotICAgICAgICB0aGlzLmxheW91dEZsYWdzID0gZmxhZ3M7Ci0KLSAgICAgICAgaWYgKChmbGFncyAmIEZvbnQuTEFZT1VUX1JJR0hUX1RPX0xFRlQpICE9IDApewotICAgICAgICAgICAgY2hhciB2ZWN0b3JbXSA9IG5ldyBjaGFyW2xlbl07Ci0gICAgICAgICAgICBmb3IoaW50IGk9MDsgaSA8IGxlbjsgaSsrKXsKLSAgICAgICAgICAgICAgICB2ZWN0b3JbaV0gPSBjaGFyc1tsZW4taS0xXTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHRoaXMudmVjdG9yID0gcGVlci5nZXRHbHlwaHModmVjdG9yKTsKLQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdGhpcy52ZWN0b3IgPSBwZWVyLmdldEdseXBocyhjaGFycyk7Ci0gICAgICAgIH0KLQotICAgICAgICB0aGlzLmdsc1RyYW5zZm9ybXMgPSBuZXcgQWZmaW5lVHJhbnNmb3JtW2xlbl07Ci0KLSAgICAgICAgc2V0RGVmYXVsdFBvc2l0aW9ucygpOwotICAgICAgICBwZXJmb3JtRGVmYXVsdExheW91dCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgbmV3IENvbW1vbkdseXBoVmVjdG9yIG9iamVjdCBmcm9tIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycy4gCi0gICAgICogTGF5b3V0IGZsYWdzIHNldCB0byBkZWZhdWx0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaGFycyBhbiBhcnJheSBvZiBjaGFycwotICAgICAqIEBwYXJhbSBmcmMgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0Ci0gICAgICogQHBhcmFtIGZudCBGb250IG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBDb21tb25HbHlwaFZlY3RvcihjaGFyW10gY2hhcnMsIEZvbnRSZW5kZXJDb250ZXh0IGZyYywgRm9udCBmbnQpIHsKLSAgICAgICAgdGhpcyhjaGFycywgZnJjLCBmbnQsIDApOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgbmV3IENvbW1vbkdseXBoVmVjdG9yIG9iamVjdCBmcm9tIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycy4gCi0gICAgICogTGF5b3V0IGZsYWdzIHNldCB0byBkZWZhdWx0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHIgc3BlY2lmaWVkIHN0cmluZwotICAgICAqIEBwYXJhbSBmcmMgRm9udFJlbmRlckNvbnRleHQgb2JqZWN0Ci0gICAgICogQHBhcmFtIGZudCBGb250IG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBDb21tb25HbHlwaFZlY3RvcihTdHJpbmcgc3RyLCBGb250UmVuZGVyQ29udGV4dCBmcmMsIEZvbnQgZm50KSB7Ci0gICAgICAgIHRoaXMoc3RyLnRvQ2hhckFycmF5KCksIGZyYywgZm50LCAwKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIG5ldyBDb21tb25HbHlwaFZlY3RvciBvYmplY3QgZnJvbSB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0ciBzcGVjaWZpZWQgc3RyaW5nCi0gICAgICogQHBhcmFtIGZyYyBGb250UmVuZGVyQ29udGV4dCBvYmplY3QKLSAgICAgKiBAcGFyYW0gZm50IEZvbnQgb2JqZWN0Ci0gICAgICogQHBhcmFtIGZsYWdzIGxheW91dCBmbGFncwotICAgICAqLwotICAgIHB1YmxpYyBDb21tb25HbHlwaFZlY3RvcihTdHJpbmcgc3RyLCBGb250UmVuZGVyQ29udGV4dCBmcmMsIEZvbnQgZm50LCBpbnQgZmxhZ3MpIHsKLSAgICAgICAgdGhpcyhzdHIudG9DaGFyQXJyYXkoKSwgZnJjLCBmbnQsIGZsYWdzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXQgYXJyYXkgb2YgbG9naWNhbCBwb3NpdGlvbnMgb2YgdGhlIGdseXBocyB0bwotICAgICAqIGRlZmF1bHQgd2l0aCB0aGVpciBkZWZhdWx0IGFkdmFuY2VzIGFuZCBoZWlnaHQuCi0gICAgICovCi0gICAgdm9pZCBzZXREZWZhdWx0UG9zaXRpb25zKCl7Ci0gICAgICAgIGludCBsZW4gPSBnZXROdW1HbHlwaHMoKTsKLQotICAgICAgICAvLyBGaXJzdCBbeCx5XSBpcyBzZXQgaW50byBbMCwwXSBwb3NpdGlvbgotICAgICAgICAvLyBmb3IgdGhpcyByZWFzb24gc3RhcnQgaW5kZXggaXMgMQotICAgICAgICBmb3IgKGludCBpPTE7IGkgPD0gbGVuOyBpKysgKXsKLSAgICAgICAgICAgICAgICBpbnQgaWR4ID0gaSA8PCAxOwotICAgICAgICAgICAgICAgIGZsb2F0IGFkdmFuY2VYID0gdmVjdG9yW2ktMV0uZ2V0R2x5cGhQb2ludE1ldHJpY3MoKS5nZXRBZHZhbmNlWCgpOwotICAgICAgICAgICAgICAgIGZsb2F0IGFkdmFuY2VZID0gdmVjdG9yW2ktMV0uZ2V0R2x5cGhQb2ludE1ldHJpY3MoKS5nZXRBZHZhbmNlWSgpOwotCi0gICAgICAgICAgICAgICAgZGVmYXVsdFBvc2l0aW9uc1tpZHhdID0gZGVmYXVsdFBvc2l0aW9uc1tpZHgtMl0gKyBhZHZhbmNlWDsKLSAgICAgICAgICAgICAgICBkZWZhdWx0UG9zaXRpb25zW2lkeCsxXSA9IGRlZmF1bHRQb3NpdGlvbnNbaWR4LTFdICsgYWR2YW5jZVk7Ci0KLSAgICAgICAgfQotICAgICAgICB0cmFuc2Zvcm0udHJhbnNmb3JtKGRlZmF1bHRQb3NpdGlvbnMsIDAsIGxvZ2ljYWxQb3NpdGlvbnMsIDAsIGdldE51bUdseXBocygpKzEpOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJuZXMgdGhlIHBpeGVsIGJvdW5kcyBvZiB0aGlzIEdseXBoVmVjdG9yIHJlbmRlcmVkIGF0IHRoZSAKLSAgICAgKiBzcGVjaWZpZWQgeCx5IGxvY2F0aW9uIHdpdGggdGhlIGdpdmVuIEZvbnRSZW5kZXJDb250ZXh0LgotICAgICAqICAKLSAgICAgKiBAcGFyYW0gZnJjIGEgRm9udFJlbmRlckNvbnRleHQgdGhhdCBpcyB1c2VkCi0gICAgICogQHBhcmFtIHggc3BlY2lmaWVkIHggY29vcmRpbmF0ZSB2YWx1ZQotICAgICAqIEBwYXJhbSB5IHNwZWNpZmllZCB5IGNvb3JkaW5hdGUgdmFsdWUKLSAgICAgKiBAcmV0dXJuIGEgUmVjdGFuZ2xlIHRoYXQgYm91bmRzIHBpeGVscyBvZiB0aGlzIEdseXBoVmVjdG9yCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFJlY3RhbmdsZSBnZXRQaXhlbEJvdW5kcyhGb250UmVuZGVyQ29udGV4dCBmcmMsIGZsb2F0IHgsIGZsb2F0IHkpIHsKLQotICAgICAgICBkb3VibGUgeE0sIHlNLCB4bSwgeW07Ci0KLSAgICAgICAgZG91YmxlIG1pblggPSAwOwotICAgICAgICBkb3VibGUgbWluWSA9IDA7Ci0gICAgICAgIGRvdWJsZSBtYXhYID0gMDsKLSAgICAgICAgZG91YmxlIG1heFkgPSAwOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgdGhpcy5nZXROdW1HbHlwaHMoKTsgaSsrKSB7Ci0gICAgICAgICAgICBSZWN0YW5nbGUgZ2x5cGhCb3VuZHMgPSB0aGlzLmdldEdseXBoUGl4ZWxCb3VuZHMoaSwgZnJjLCAwLCAwKTsKLSAgICAgICAgICAgIHhtID0gZ2x5cGhCb3VuZHMuZ2V0TWluWCgpOwotICAgICAgICAgICAgeW0gPSBnbHlwaEJvdW5kcy5nZXRNaW5ZKCk7Ci0gICAgICAgICAgICB4TSA9IGdseXBoQm91bmRzLmdldE1heFgoKTsKLSAgICAgICAgICAgIHlNID0gZ2x5cGhCb3VuZHMuZ2V0TWF4WSgpOwotCi0gICAgICAgICAgICBpZiAoaSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgbWluWCA9IHhtOwotICAgICAgICAgICAgICAgIG1pblkgPSB5bTsKLSAgICAgICAgICAgICAgICBtYXhYID0geE07Ci0gICAgICAgICAgICAgICAgbWF4WSA9IHlNOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAobWluWCA+IHhtKSB7Ci0gICAgICAgICAgICAgICAgbWluWCA9IHhtOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKG1pblkgPiB5bSkgewotICAgICAgICAgICAgICAgIG1pblkgPSB5bTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChtYXhYIDwgeE0pIHsKLSAgICAgICAgICAgICAgICBtYXhYID0geE07Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAobWF4WSA8IHlNKSB7Ci0gICAgICAgICAgICAgICAgbWF4WSA9IHlNOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlKChpbnQpKG1pblggKyB4KSwgKGludCkobWluWSArIHkpLCAoaW50KShtYXhYIC0gbWluWCksIChpbnQpKG1heFkgLSBtaW5ZKSk7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSB2aXN1YWwgYm91bmRzIG9mIHRoaXMgR2x5cGhWZWN0b3IuCi0gICAgICogVGhlIHZpc3VhbCBib3VuZHMgaXMgdGhlIGJvdW5kcyBvZiB0aGUgdG90YWwgb3V0bGluZSBvZiAKLSAgICAgKiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqIEByZXR1cm4gYSBSZWN0YW5nbGUyRCB0aGF0IGlkIHRoZSB2aXN1YWwgYm91bmRzIG9mIHRoaXMgR2x5cGhWZWN0b3IKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0VmlzdWFsQm91bmRzKCkgewotICAgICAgICBmbG9hdCB4TSwgeU0sIHhtLCB5bTsKLSAgICAgICAgZmxvYXQgbWluWCA9IDA7Ci0gICAgICAgIGZsb2F0IG1pblkgPSAwOwotICAgICAgICBmbG9hdCBtYXhYID0gMDsKLSAgICAgICAgZmxvYXQgbWF4WSA9IDA7Ci0gICAgICAgIGJvb2xlYW4gZmlyc3RJdGVyYXRpb24gPSB0cnVlOwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgdGhpcy5nZXROdW1HbHlwaHMoKTsgaSsrKSB7Ci0gICAgICAgICAgICBSZWN0YW5nbGUyRCBib3VuZHMgPSB0aGlzLmdldEdseXBoVmlzdWFsQm91bmRzKGkpLmdldEJvdW5kczJEKCk7Ci0gICAgICAgICAgICBpZiAoYm91bmRzLmdldFdpZHRoKCkgPT0gMCl7Ci0gICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB4bSA9IChmbG9hdClib3VuZHMuZ2V0WCgpOwotICAgICAgICAgICAgeW0gPSAoZmxvYXQpYm91bmRzLmdldFkoKTsKLQotICAgICAgICAgICAgeE0gPSAoZmxvYXQpKHhtICsgYm91bmRzLmdldFdpZHRoKCkpOwotCi0gICAgICAgICAgICB5TSA9IHltICsgKGZsb2F0KSBib3VuZHMuZ2V0SGVpZ2h0KCk7Ci0KLSAgICAgICAgICAgIGlmIChmaXJzdEl0ZXJhdGlvbikgewotICAgICAgICAgICAgICAgIG1pblggPSB4bTsKLSAgICAgICAgICAgICAgICBtaW5ZID0geW07Ci0gICAgICAgICAgICAgICAgbWF4WCA9IHhNOwotICAgICAgICAgICAgICAgIG1heFkgPSB5TTsKLSAgICAgICAgICAgICAgICBmaXJzdEl0ZXJhdGlvbiA9IGZhbHNlOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBpZiAobWluWCA+IHhtKSB7Ci0gICAgICAgICAgICAgICAgICAgIG1pblggPSB4bTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKG1pblkgPiB5bSkgewotICAgICAgICAgICAgICAgICAgICBtaW5ZID0geW07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChtYXhYIDwgeE0pIHsKLSAgICAgICAgICAgICAgICAgICAgbWF4WCA9IHhNOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAobWF4WSA8IHlNKSB7Ci0gICAgICAgICAgICAgICAgICAgIG1heFkgPSB5TTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiAodGhpcy5nZXROdW1HbHlwaHMoKSAhPSAwKSA/IG5ldyBSZWN0YW5nbGUyRC5GbG9hdChtaW5YLCBtaW5ZLAotICAgICAgICAgICAgICAgIChtYXhYIC0gbWluWCksIChtYXhZIC0gbWluWSkpIDogbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIG5ldyBwb3NpdGlvbiB0byB0aGUgc3BlY2lmaWVkIGdseXBoLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEdseXBoUG9zaXRpb24oaW50IGdseXBoSW5kZXgsIFBvaW50MkQgbmV3UG9zKSB7Ci0gICAgICAgIGlmICgoZ2x5cGhJbmRleCA+IHZlY3Rvci5sZW5ndGgpIHx8IChnbHlwaEluZGV4IDwgMCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC40Mz1nbHlwaEluZGV4IGlzIG91dCBvZiB2ZWN0b3IncyBsaW1pdHMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjQzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgZmxvYXQgeCA9IChmbG9hdCluZXdQb3MuZ2V0WCgpOwotICAgICAgICBmbG9hdCB5ID0gKGZsb2F0KW5ld1Bvcy5nZXRZKCk7Ci0gICAgICAgIGludCBpbmRleCA9IGdseXBoSW5kZXggPDwgMTsKLQotICAgICAgICBpZiAoKHggIT0gdmlzdWFsUG9zaXRpb25zW2luZGV4XSkgfHwgKHkgIT0gdmlzdWFsUG9zaXRpb25zW2luZGV4ICsgMV0pKXsKLSAgICAgICAgICAgIHZpc3VhbFBvc2l0aW9uc1tpbmRleF0gPSB4OwotICAgICAgICAgICAgdmlzdWFsUG9zaXRpb25zW2luZGV4KzFdID0geTsKLSAgICAgICAgICAgIGxheW91dEZsYWdzID0gbGF5b3V0RmxhZ3MgfCBGTEFHX0hBU19QT1NJVElPTl9BREpVU1RNRU5UUzsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgcG9zaXRpb24gb2YgdGhlIHNwZWNpZmllZCBnbHlwaCByZWxhdGl2ZSB0byB0aGUgb3JpZ2luIG9mCi0gICAgICogdGhpcyBHbHlwaFZlY3RvcgotICAgICAqIEByZXR1cm4gYSBQb2ludDJEIHRoYXQgdGhlIG9yaWdpbiBvZiB0aGUgZ2x5cGggd2l0aCBzcGVjaWZpZWQgaW5kZXgKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgUG9pbnQyRCBnZXRHbHlwaFBvc2l0aW9uKGludCBnbHlwaEluZGV4KSB7Ci0gICAgICAgIGlmICgoZ2x5cGhJbmRleCA+IHZlY3Rvci5sZW5ndGgpIHx8IChnbHlwaEluZGV4IDwgMCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC40Mz1nbHlwaEluZGV4IGlzIG91dCBvZiB2ZWN0b3IncyBsaW1pdHMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjQzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgaW50IGluZGV4ID0gZ2x5cGhJbmRleCA8PCAxOwotICAgICAgICBQb2ludDJEIHBvcyA9IG5ldyBQb2ludDJELkZsb2F0KHZpc3VhbFBvc2l0aW9uc1tpbmRleF0sIHZpc3VhbFBvc2l0aW9uc1tpbmRleCsxXSk7Ci0KLSAgICAgICAgLy8gRm9yIGxhc3QgcG9zaXRpb24gd2UgZG9uJ3QgaGF2ZSB0byB0cmFuc2Zvcm0gISEKLSAgICAgICAgaWYoZ2x5cGhJbmRleD09dmVjdG9yLmxlbmd0aCl7Ci0gICAgICAgICAgICByZXR1cm4gcG9zOwotICAgICAgICB9Ci0KLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIGF0ID0gZ2V0R2x5cGhUcmFuc2Zvcm0oZ2x5cGhJbmRleCk7Ci0gICAgICAgIGlmICgoYXQgPT0gbnVsbCkgfHwgKGF0LmlzSWRlbnRpdHkoKSkpewotICAgICAgICAgICAgcmV0dXJuIHBvczsKLSAgICAgICAgfQotCi0gICAgICAgIHBvcy5zZXRMb2NhdGlvbihwb3MuZ2V0WCgpICsgYXQuZ2V0VHJhbnNsYXRlWCgpLCBwb3MuZ2V0WSgpICsgYXQuZ2V0VHJhbnNsYXRlWSgpKTsKLQotICAgICAgICByZXR1cm4gcG9zOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgbmV3IHRyYW5zZm9ybSB0byB0aGUgc3BlY2lmaWVkIGdseXBoLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4IHNwZWNpZmllZCBpbmRleCBvZiB0aGUgZ2x5cGgKLSAgICAgKiBAcGFyYW0gdHJhbnMgQWZmaW5lVHJhbnNmb3JtIG9mIHRoZSBnbHlwaCB3aXRoIHNwZWNpZmllZCBpbmRleAotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldEdseXBoVHJhbnNmb3JtKGludCBnbHlwaEluZGV4LCBBZmZpbmVUcmFuc2Zvcm0gdHJhbnMpIHsKLSAgICAgICAgaWYgKChnbHlwaEluZGV4ID49IHZlY3Rvci5sZW5ndGgpIHx8IChnbHlwaEluZGV4IDwgMCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC40Mz1nbHlwaEluZGV4IGlzIG91dCBvZiB2ZWN0b3IncyBsaW1pdHMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjQzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoKHRyYW5zID09IG51bGwpIHx8ICh0cmFucy5pc0lkZW50aXR5KCkpKSB7Ci0gICAgICAgICAgICBnbHNUcmFuc2Zvcm1zW2dseXBoSW5kZXhdID0gbnVsbDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGdsc1RyYW5zZm9ybXNbZ2x5cGhJbmRleF0gPSBuZXcgQWZmaW5lVHJhbnNmb3JtKHRyYW5zKTsKLSAgICAgICAgICAgIGxheW91dEZsYWdzID0gbGF5b3V0RmxhZ3MgfCBGTEFHX0hBU19UUkFOU0ZPUk1TOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgYWZmaW5lIHRyYW5zZm9ybSBvZiB0aGUgc3BlY2lmaWVkIGdseXBoLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4IHNwZWNpZmllZCBpbmRleCBvZiB0aGUgZ2x5cGgKLSAgICAgKiBAcmV0dXJuIGFuIEFmZmluZVRyYW5zZm9ybSBvZiB0aGUgZ2x5cGggd2l0aCBzcGVjaWZpZWQgaW5kZXgKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgQWZmaW5lVHJhbnNmb3JtIGdldEdseXBoVHJhbnNmb3JtKGludCBnbHlwaEluZGV4KSB7Ci0gICAgICAgIGlmICgoZ2x5cGhJbmRleCA+PSB0aGlzLnZlY3Rvci5sZW5ndGgpIHx8IChnbHlwaEluZGV4IDwgMCkpIHsKLSAgICAgICAgICAgIC8vIGF3dC40Mz1nbHlwaEluZGV4IGlzIG91dCBvZiB2ZWN0b3IncyBsaW1pdHMKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjQzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHRoaXMuZ2xzVHJhbnNmb3Jtc1tnbHlwaEluZGV4XTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBtZXRyaWNzIG9mIHRoZSBzcGVjaWZpZWQgZ2x5cGguCi0gICAgICogCi0gICAgICogQHBhcmFtIGdseXBoSW5kZXggc3BlY2lmaWVkIGluZGV4IG9mIHRoZSBnbHlwaAotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBHbHlwaE1ldHJpY3MgZ2V0R2x5cGhNZXRyaWNzKGludCBnbHlwaEluZGV4KSB7Ci0KLSAgICAgICAgaWYgKChnbHlwaEluZGV4IDwgMCkgfHwgKChnbHlwaEluZGV4KSA+PSB0aGlzLmdldE51bUdseXBocygpKSkgewotICAgICAgICAgICAgLy8gYXd0LjQzPWdseXBoSW5kZXggaXMgb3V0IG9mIHZlY3RvcidzIGxpbWl0cwotICAgICAgICAgICAgdGhyb3cgbmV3IEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICAvLyBUT0RPOiBpcyB0aGVyZSBhIHNlbmNlIGluIEdseXBoTWV0cmljcwotICAgICAgICAvLyBpZiBjZXJ0YWluIGdseXBoIG9yIEZvbnQgaGFzIGEgdHJhbnNmb3JtPz8KLSAgICAgICAgcmV0dXJuIHRoaXMudmVjdG9yW2dseXBoSW5kZXhdLmdldEdseXBoTWV0cmljcygpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBqdXN0aWZpY2F0aW9uIGluZm9ybWF0aW9uIGZvciB0aGUgZ2x5cGggd2l0aCBzcGVjaWZpZWQgZ2x5cGggCi0gICAgICogaW5kZXguCi0gICAgICogQHBhcmFtIGdseXBoSW5kZXggaW5kZXggb2YgYSBnbHlwaCB3aGljaCBHbHlwaEp1c3RpZmljYXRpb25JbmZvIGlzIHRvIGJlIAotICAgICAqIHJlY2VpdmVkICAgCi0gICAgICogQHJldHVybiBhIEdseXBoSnVzdGlmaWNhdGlvbkluZm8gb2JqZWN0IHRoYXQgY29udGFpbnMgZ2x5cGgganVzdGlmaWNhdGlvbiAKLSAgICAgKiBwcm9wZXJ0aWVzIG9mIHRoZSBzcGVjaWZpZWQgZ2x5cGgKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBnZXRHbHlwaEp1c3RpZmljYXRpb25JbmZvKGludCBnbHlwaEluZGV4KSB7Ci0gICAgICAgIC8vIFRPRE8gOiBGaW5kIG91dCB0aGUgc291cmNlIG9mIEp1c3RpZmljYXRpb24gaW5mbwotICAgICAgICBpZiAodHJ1ZSkgewotICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIk1ldGhvZCBpcyBub3QgaW1wbGVtZW50ZWQiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIEZvbnRSZW5kZXJDb250ZXh0IHBhcmFtZXRlciBvZiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBGb250UmVuZGVyQ29udGV4dCBnZXRGb250UmVuZGVyQ29udGV4dCgpIHsKLSAgICAgICAgcmV0dXJuIHRoaXMudmVjdG9yRlJDOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIHZpc3VhbCBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBnbHlwaC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2x5cGhJbmRleCBzcGVjaWZpZWQgaW5kZXggb2YgdGhlIGdseXBoCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFNoYXBlIGdldEdseXBoVmlzdWFsQm91bmRzKGludCBnbHlwaEluZGV4KSB7Ci0gICAgICAgIGlmICgoZ2x5cGhJbmRleCA8IDApIHx8IChnbHlwaEluZGV4ID49IHRoaXMuZ2V0TnVtR2x5cGhzKCkpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNDM9Z2x5cGhJbmRleCBpcyBvdXQgb2YgdmVjdG9yJ3MgbGltaXRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGlkeCAgPSBnbHlwaEluZGV4IDw8IDE7Ci0KLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIGZvbnRUcmFuc2Zvcm0gPSB0aGlzLnRyYW5zZm9ybTsKLSAgICAgICAgZG91YmxlIHhPZmZzID0gZm9udFRyYW5zZm9ybS5nZXRUcmFuc2xhdGVYKCk7Ci0gICAgICAgIGRvdWJsZSB5T2ZmcyA9IGZvbnRUcmFuc2Zvcm0uZ2V0VHJhbnNsYXRlWSgpOwotCi0gICAgICAgIGlmICh2ZWN0b3JbZ2x5cGhJbmRleF0uZ2V0V2lkdGgoKSA9PSAwKXsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlMkQuRmxvYXQoKGZsb2F0KXhPZmZzLCAoZmxvYXQpeU9mZnMsIDAsIDApOwotICAgICAgICB9Ci0KLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIGF0ID0gQWZmaW5lVHJhbnNmb3JtLmdldFRyYW5zbGF0ZUluc3RhbmNlKHhPZmZzLCB5T2Zmcyk7Ci0gICAgICAgIEFmZmluZVRyYW5zZm9ybSBnbHlwaFRyYW5zZm9ybSA9IGdldEdseXBoVHJhbnNmb3JtKGdseXBoSW5kZXgpOwotCi0gICAgICAgIGlmICh0cmFuc2Zvcm0uaXNJZGVudGl0eSgpICYmICgoZ2x5cGhUcmFuc2Zvcm0gPT0gbnVsbCkgfHwgZ2x5cGhUcmFuc2Zvcm0uaXNJZGVudGl0eSgpKSl7Ci0gICAgICAgICAgICBSZWN0YW5nbGUyRCBibGFja0JveCA9IHZlY3RvcltnbHlwaEluZGV4XS5nZXRHbHlwaE1ldHJpY3MoKS5nZXRCb3VuZHMyRCgpOwotICAgICAgICAgICAgYXQudHJhbnNsYXRlKHZpc3VhbFBvc2l0aW9uc1tpZHhdLCB2aXN1YWxQb3NpdGlvbnNbaWR4KzFdKTsKLSAgICAgICAgICAgIHJldHVybihhdC5jcmVhdGVUcmFuc2Zvcm1lZFNoYXBlKGJsYWNrQm94KSk7Ci0gICAgICAgIH0KLQotICAgICAgICBHZW5lcmFsUGF0aCBzaGFwZSA9IChHZW5lcmFsUGF0aCl0aGlzLmdldEdseXBoT3V0bGluZShnbHlwaEluZGV4KTsKLSAgICAgICAgc2hhcGUudHJhbnNmb3JtKGF0KTsKLSAgICAgICAgcmV0dXJuIHNoYXBlLmdldEJvdW5kczJEKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJuZXMgdGhlIHBpeGVsIGJvdW5kcyBvZiB0aGUgc3BlY2lmaWVkIGdseXBoIHdpdGhpbiBHbHlwaFZlY3RvciAKLSAgICAgKiByZW5kZXJlZCBhdCB0aGUgc3BlY2lmaWVkIHgseSBsb2NhdGlvbi4KLSAgICAgKiAgCi0gICAgICogQHBhcmFtIGdseXBoSW5kZXggaW5kZXggb2YgdGhlIGdseXBoCi0gICAgICogQHBhcmFtIGZyYyBhIEZvbnRSZW5kZXJDb250ZXh0IHRoYXQgaXMgdXNlZAotICAgICAqIEBwYXJhbSB4IHNwZWNpZmllZCB4IGNvb3JkaW5hdGUgdmFsdWUKLSAgICAgKiBAcGFyYW0geSBzcGVjaWZpZWQgeSBjb29yZGluYXRlIHZhbHVlCi0gICAgICogQHJldHVybiBhIFJlY3RhbmdsZSB0aGF0IGJvdW5kcyBwaXhlbHMgb2YgdGhlIHNwZWNpZmllZCBnbHlwaAotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBSZWN0YW5nbGUgZ2V0R2x5cGhQaXhlbEJvdW5kcyhpbnQgZ2x5cGhJbmRleCwgRm9udFJlbmRlckNvbnRleHQgZnJjLAotICAgICAgICAgICAgZmxvYXQgeCwgZmxvYXQgeSkgewotICAgICAgICAvLyBUT0RPIDogbmVlZCB0byBiZSBpbXBsZW1lbnRlZCB3aXRoIEZvbnRSZW5kZXJDb250ZXh0Ci0gICAgICAgIGlmICgoZ2x5cGhJbmRleCA8IDApIHx8IChnbHlwaEluZGV4ID49IHRoaXMuZ2V0TnVtR2x5cGhzKCkpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNDM9Z2x5cGhJbmRleCBpcyBvdXQgb2YgdmVjdG9yJ3MgbGltaXRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGlkeCAgPSBnbHlwaEluZGV4IDw8IDE7Ci0KLSAgICAgICAgaWYgKHZlY3RvcltnbHlwaEluZGV4XS5nZXRXaWR0aCgpID09IDApewotICAgICAgICAgICAgQWZmaW5lVHJhbnNmb3JtIGZvbnRUcmFuc2Zvcm0gPSB0aGlzLnRyYW5zZm9ybTsKLSAgICAgICAgICAgIGRvdWJsZSB4T2ZmcyA9IHggKyB2aXN1YWxQb3NpdGlvbnNbaWR4XSArIGZvbnRUcmFuc2Zvcm0uZ2V0VHJhbnNsYXRlWCgpOwotICAgICAgICAgICAgZG91YmxlIHlPZmZzID0geSArIHZpc3VhbFBvc2l0aW9uc1tpZHgrMV0gKyBmb250VHJhbnNmb3JtLmdldFRyYW5zbGF0ZVkoKTsKLSAgICAgICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlKChpbnQpeE9mZnMsIChpbnQpeU9mZnMsIDAsIDApOwotICAgICAgICB9Ci0KLSAgICAgICAgR2VuZXJhbFBhdGggc2hhcGUgPSAoR2VuZXJhbFBhdGgpdGhpcy5nZXRHbHlwaE91dGxpbmUoZ2x5cGhJbmRleCk7Ci0KLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIGF0ID0gQWZmaW5lVHJhbnNmb3JtLmdldFRyYW5zbGF0ZUluc3RhbmNlKHgsIHkpOwotCi0gICAgICAgIGlmIChmcmMgIT0gbnVsbCl7Ci0gICAgICAgICAgICBhdC5jb25jYXRlbmF0ZShmcmMuZ2V0VHJhbnNmb3JtKCkpOwotICAgICAgICB9Ci0KLSAgICAgICAgc2hhcGUudHJhbnNmb3JtKGF0KTsKLQotICAgICAgICBSZWN0YW5nbGUgYm91bmRzID0gc2hhcGUuZ2V0Qm91bmRzKCk7Ci0gICAgICAgIHJldHVybiBuZXcgUmVjdGFuZ2xlKChpbnQpYm91bmRzLmdldFgoKSwgKGludClib3VuZHMuZ2V0WSgpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIChpbnQpYm91bmRzLmdldFdpZHRoKCktMSwgKGludClib3VuZHMuZ2V0SGVpZ2h0KCktMSk7Ci0gICAgICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBTaGFwZSB0aGF0IGVuY2xvc2VzIHNwZWNpZmllZCBnbHlwaC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2x5cGhJbmRleCBzcGVjaWZpZWQgaW5kZXggb2YgdGhlIGdseXBoCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFNoYXBlIGdldEdseXBoT3V0bGluZShpbnQgZ2x5cGhJbmRleCkgewotICAgICAgICBpZiAoKGdseXBoSW5kZXggPCAwKSB8fCAoZ2x5cGhJbmRleCA+PSB0aGlzLmdldE51bUdseXBocygpKSkgewotICAgICAgICAgICAgLy8gYXd0LjQzPWdseXBoSW5kZXggaXMgb3V0IG9mIHZlY3RvcidzIGxpbWl0cwotICAgICAgICAgICAgdGhyb3cgbmV3IEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChndlNoYXBlc1tnbHlwaEluZGV4XSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBndlNoYXBlc1tnbHlwaEluZGV4XSA9IHZlY3RvcltnbHlwaEluZGV4XS5nZXRTaGFwZSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgR2VuZXJhbFBhdGggZ3AgPSAoR2VuZXJhbFBhdGgpKChHZW5lcmFsUGF0aClndlNoYXBlc1tnbHlwaEluZGV4XSkuY2xvbmUoKTsKLQotICAgICAgICAvKiBBcHBseWluZyBHbHlwaFZlY3RvciBmb250IHRyYW5zZm9ybSAqLwotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gYXQgPSAoQWZmaW5lVHJhbnNmb3JtKXRoaXMudHJhbnNmb3JtLmNsb25lKCk7Ci0KLSAgICAgICAgLyogQXBwbHlpbmcgR2x5cGggdHJhbnNmb3JtICovCi0gICAgICAgIEFmZmluZVRyYW5zZm9ybSBnbHlwaEFUID0gZ2V0R2x5cGhUcmFuc2Zvcm0oZ2x5cGhJbmRleCk7Ci0gICAgICAgIGlmIChnbHlwaEFUICE9IG51bGwpewotICAgICAgICAgICAgYXQucHJlQ29uY2F0ZW5hdGUoZ2x5cGhBVCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgaWR4ICA9IGdseXBoSW5kZXggPDwgMTsKLQotICAgICAgICBncC50cmFuc2Zvcm0oYXQpOwotICAgICAgICBncC50cmFuc2Zvcm0oQWZmaW5lVHJhbnNmb3JtLmdldFRyYW5zbGF0ZUluc3RhbmNlKHZpc3VhbFBvc2l0aW9uc1tpZHhdLCB2aXN1YWxQb3NpdGlvbnNbaWR4KzFdKSk7Ci0gICAgICAgIHJldHVybiBncDsKLSAgICB9Ci0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYSBTaGFwZSB0aGF0IGlzIHRoZSBvdXRsaW5lIHJlcHJlc2VudGF0aW9uIG9mIHRoaXMgR2x5cGhWZWN0b3IgCi0gICAgICogcmVuZGVyZWQgYXQgdGhlIHNwZWNpZmllZCB4LHkgY29vcmRpbmF0ZXMuCi0gICAgICogCi0gICAgICogQHBhcmFtIHggc3BlY2lmaWVkIHggY29vcmRpbmF0ZSB2YWx1ZQotICAgICAqIEBwYXJhbSB5IHNwZWNpZmllZCB5IGNvb3JkaW5hdGUgdmFsdWUKLSAgICAgKiBAcmV0dXJuIGEgU2hhcGUgb2JqZWN0IHRoYXQgaXMgdGhlIG91dGxpbmUgb2YgdGhpcyBHbHlwaFZlY3RvcgotICAgICAqIGF0IHRoZSBzcGVjaWZpZWQgY29vcmRpbmF0ZXMuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFNoYXBlIGdldE91dGxpbmUoZmxvYXQgeCwgZmxvYXQgeSkgewotICAgICAgICBHZW5lcmFsUGF0aCBncCA9IG5ldyBHZW5lcmFsUGF0aChHZW5lcmFsUGF0aC5XSU5EX0VWRU5fT0REKTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCB0aGlzLnZlY3Rvci5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgR2VuZXJhbFBhdGggb3V0bGluZSA9IChHZW5lcmFsUGF0aClnZXRHbHlwaE91dGxpbmUoaSk7Ci0KLSAgICAgICAgICAgIC8qIEFwcGx5aW5nIHRyYW5zbGF0aW9uIHRvIGFjdHVhbCB2aXN1YWwgYm91bmRzICovCi0gICAgICAgICAgICBvdXRsaW5lLnRyYW5zZm9ybShBZmZpbmVUcmFuc2Zvcm0uZ2V0VHJhbnNsYXRlSW5zdGFuY2UoeCwgeSkpOwotICAgICAgICAgICAgZ3AuYXBwZW5kKG91dGxpbmUsIGZhbHNlKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBncDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGEgU2hhcGUgdGhhdCBpcyB0aGUgb3V0bGluZSByZXByZXNlbnRhdGlvbiBvZiB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqIAotICAgICAqIEByZXR1cm4gYSBTaGFwZSBvYmplY3QgdGhhdCBpcyB0aGUgb3V0bGluZSBvZiB0aGlzIEdseXBoVmVjdG9yCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFNoYXBlIGdldE91dGxpbmUoKSB7Ci0gICAgICAgIHJldHVybiB0aGlzLmdldE91dGxpbmUoMCwgMCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhbiBhcnJheSBvZiBnbHlwaGNvZGVzIGZvciB0aGUgc3BlY2lmaWVkIGdseXBocy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYmVnaW5HbHlwaEluZGV4IHRoZSBzdGFydCBpbmRleAotICAgICAqIEBwYXJhbSBudW1FbnRyaWVzIHRoZSBudW1iZXIgb2YgZ2x5cGggY29kZXMgdG8gZ2V0Ci0gICAgICogQHBhcmFtIGNvZGVSZXR1cm4gdGhlIGFycmF5IHRoYXQgcmVjZWl2ZXMgZ2x5cGggY29kZXMnIHZhbHVlcwotICAgICAqIEByZXR1cm4gYW4gYXJyYXkgdGhhdCByZWNlaXZlcyBnbHlwaCBjb2RlcycgdmFsdWVzCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldEdseXBoQ29kZXMoaW50IGJlZ2luR2x5cGhJbmRleCwgaW50IG51bUVudHJpZXMsCi0gICAgICAgICAgICBpbnRbXSBjb2RlUmV0dXJuKSB7Ci0KLSAgICAgICAgaWYgKChiZWdpbkdseXBoSW5kZXggPCAwKSB8fCAoKG51bUVudHJpZXMgKyBiZWdpbkdseXBoSW5kZXgpID4gdGhpcy5nZXROdW1HbHlwaHMoKSkpIHsKLSAgICAgICAgICAgIC8vIGF3dC40ND1iZWdpbkdseXBoSW5kZXggaXMgb3V0IG9mIHZlY3RvcidzIHJhbmdlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40NCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG51bUVudHJpZXMgPCAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNDU9bnVtRW50cmllcyBpcyBvdXQgb2YgdmVjdG9yJ3MgcmFuZ2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChjb2RlUmV0dXJuID09IG51bGwpIHsKLSAgICAgICAgICAgIGNvZGVSZXR1cm4gPSBuZXcgaW50W251bUVudHJpZXNdOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IGJlZ2luR2x5cGhJbmRleDsgaSA8IGJlZ2luR2x5cGhJbmRleCArIG51bUVudHJpZXM7IGkrKykgewotICAgICAgICAgICAgY29kZVJldHVybltpLWJlZ2luR2x5cGhJbmRleF0gPSB0aGlzLnZlY3RvcltpXS5nZXRHbHlwaENvZGUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBjb2RlUmV0dXJuOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gYXJyYXkgb2YgbnVtRW50cmllcyBjaGFyYWN0ZXIgaW5kaWNlcyBmb3IgdGhlIHNwZWNpZmllZCBnbHlwaHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJlZ2luR2x5cGhJbmRleCB0aGUgc3RhcnQgaW5kZXgKLSAgICAgKiBAcGFyYW0gbnVtRW50cmllcyB0aGUgbnVtYmVyIG9mIGdseXBoIGNvZGVzIHRvIGdldAotICAgICAqIEBwYXJhbSBjb2RlUmV0dXJuIHRoZSBhcnJheSB0aGF0IHJlY2VpdmVzIGdseXBoIGNvZGVzJyB2YWx1ZXMKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IHRoYXQgcmVjZWl2ZXMgZ2x5cGggY2hhciBpbmRpY2VzCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludFtdIGdldEdseXBoQ2hhckluZGljZXMoaW50IGJlZ2luR2x5cGhJbmRleCwgaW50IG51bUVudHJpZXMsCi0gICAgICAgICAgICBpbnRbXSBjb2RlUmV0dXJuKSB7Ci0gICAgICAgIGlmICgoYmVnaW5HbHlwaEluZGV4IDwgMCkgfHwgKGJlZ2luR2x5cGhJbmRleCA+PSB0aGlzLmdldE51bUdseXBocygpKSkgewotICAgICAgICAgICAgLy8gYXd0LjQ0PWJlZ2luR2x5cGhJbmRleCBpcyBvdXQgb2YgdmVjdG9yJ3MgcmFuZ2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmICgobnVtRW50cmllcyA8IDApCi0gICAgICAgICAgICAgICAgfHwgKChudW1FbnRyaWVzICsgYmVnaW5HbHlwaEluZGV4KSA+IHRoaXMuZ2V0TnVtR2x5cGhzKCkpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNDU9bnVtRW50cmllcyBpcyBvdXQgb2YgdmVjdG9yJ3MgcmFuZ2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChjb2RlUmV0dXJuID09IG51bGwpIHsKLSAgICAgICAgICAgIGNvZGVSZXR1cm4gPSBuZXcgaW50W251bUVudHJpZXNdOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1FbnRyaWVzOyBpKyspIHsKLSAgICAgICAgICAgIGNvZGVSZXR1cm5baV0gPSB0aGlzLmdldEdseXBoQ2hhckluZGV4KGkgKyBiZWdpbkdseXBoSW5kZXgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBjb2RlUmV0dXJuOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gYXJyYXkgb2YgbnVtRW50cmllcyBnbHlwaHMgcG9zaXRpb25zIGZyb20gYmVnaW5HbHlwaEluZGV4Ci0gICAgICogZ2x5cGggaW4gR2x5cGggVmVjdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBiZWdpbkdseXBoSW5kZXggdGhlIHN0YXJ0IGluZGV4Ci0gICAgICogQHBhcmFtIG51bUVudHJpZXMgdGhlIG51bWJlciBvZiBnbHlwaCBjb2RlcyB0byBnZXQKLSAgICAgKiBAcGFyYW0gcG9zaXRpb25SZXR1cm4gdGhlIGFycmF5IHRoYXQgcmVjZWl2ZXMgZ2x5cGhzJyBwb3NpdGlvbnMKLSAgICAgKiBAcmV0dXJuIGFuIGFycmF5IG9mIGZsb2F0cyB0aGF0IHJlY2VpdmVzIGdseXBoIGNoYXIgaW5kaWNlcwotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdFtdIGdldEdseXBoUG9zaXRpb25zKGludCBiZWdpbkdseXBoSW5kZXgsIGludCBudW1FbnRyaWVzLAotICAgICAgICAgICAgZmxvYXRbXSBwb3NpdGlvblJldHVybikgewotCi0gICAgICAgIGludCBsZW4gPSAodGhpcy5nZXROdW1HbHlwaHMoKSsxKSA8PCAxOwotICAgICAgICBiZWdpbkdseXBoSW5kZXggKj0gMjsKLSAgICAgICAgbnVtRW50cmllcyAqPSAyOwotCi0gICAgICAgIGlmICgoYmVnaW5HbHlwaEluZGV4IDwgMCkgfHwgKChudW1FbnRyaWVzICsgYmVnaW5HbHlwaEluZGV4KSA+IGxlbikpIHsKLSAgICAgICAgICAgIC8vIGF3dC40ND1iZWdpbkdseXBoSW5kZXggaXMgb3V0IG9mIHZlY3RvcidzIHJhbmdlCi0gICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40NCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG51bUVudHJpZXMgPCAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNDU9bnVtRW50cmllcyBpcyBvdXQgb2YgdmVjdG9yJ3MgcmFuZ2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDUiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChwb3NpdGlvblJldHVybiA9PSBudWxsKSB7Ci0gICAgICAgICAgICBwb3NpdGlvblJldHVybiA9IG5ldyBmbG9hdFtudW1FbnRyaWVzXTsKLSAgICAgICAgfQotCi0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkodmlzdWFsUG9zaXRpb25zLCBiZWdpbkdseXBoSW5kZXgsIHBvc2l0aW9uUmV0dXJuLCAwLCBudW1FbnRyaWVzKTsKLQotICAgICAgICByZXR1cm4gcG9zaXRpb25SZXR1cm47Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0IG51bUVudHJpZXMgZWxlbWVudHMgb2YgdGhlIHZpc3VhbFBvc2l0aW9ucyBhcnJheSBmcm9tIGJlZ2luR2x5cGhJbmRleAotICAgICAqIG9mIG51bUVudHJpZXMgZ2x5cGhzIHBvc2l0aW9ucyBmcm9tIGJlZ2luR2x5cGhJbmRleCBnbHlwaCBpbiBHbHlwaCBWZWN0b3IuCi0gICAgICogCi0gICAgICogQHBhcmFtIGJlZ2luR2x5cGhJbmRleCB0aGUgc3RhcnQgaW5kZXgKLSAgICAgKiBAcGFyYW0gbnVtRW50cmllcyB0aGUgbnVtYmVyIG9mIGdseXBoIGNvZGVzIHRvIGdldAotICAgICAqIEBwYXJhbSBzZXRQb3NpdGlvbnMgdGhlIGFycmF5IG9mIHBvc2l0aW9ucyB0byBzZXQKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRHbHlwaFBvc2l0aW9ucyhpbnQgYmVnaW5HbHlwaEluZGV4LCBpbnQgbnVtRW50cmllcywKLSAgICAgICAgICAgIGZsb2F0W10gc2V0UG9zaXRpb25zKSB7Ci0KLSAgICAgICAgaW50IGxlbiA9ICh0aGlzLmdldE51bUdseXBocygpKzEpIDw8IDE7Ci0gICAgICAgIGJlZ2luR2x5cGhJbmRleCAqPSAyOwotICAgICAgICBudW1FbnRyaWVzICo9IDI7Ci0KLSAgICAgICAgaWYgKChiZWdpbkdseXBoSW5kZXggPCAwKSB8fCAoKG51bUVudHJpZXMgKyBiZWdpbkdseXBoSW5kZXgpID4gbGVuKSkgewotICAgICAgICAgICAgLy8gYXd0LjQ0PWJlZ2luR2x5cGhJbmRleCBpcyBvdXQgb2YgdmVjdG9yJ3MgcmFuZ2UKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjQ0IikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAobnVtRW50cmllcyA8IDApIHsKLSAgICAgICAgICAgIC8vIGF3dC40NT1udW1FbnRyaWVzIGlzIG91dCBvZiB2ZWN0b3IncyByYW5nZQotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40NSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgU3lzdGVtLmFycmF5Y29weShzZXRQb3NpdGlvbnMsIDAsIHZpc3VhbFBvc2l0aW9ucywgYmVnaW5HbHlwaEluZGV4LCBudW1FbnRyaWVzKTsKLSAgICAgICAgbGF5b3V0RmxhZ3MgPSBsYXlvdXRGbGFncyAmIEZMQUdfSEFTX1BPU0lUSU9OX0FESlVTVE1FTlRTOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0IGVsZW1lbnRzIG9mIHRoZSB2aXN1YWxQb3NpdGlvbnMgYXJyYXkuCi0gICAgICogCi0gICAgICogQHBhcmFtIHNldFBvc2l0aW9ucyB0aGUgYXJyYXkgb2YgcG9zaXRpb25zIHRvIHNldAotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEdseXBoUG9zaXRpb25zKGZsb2F0W10gc2V0UG9zaXRpb25zKSB7Ci0KLSAgICAgICAgaW50IGxlbiA9ICh0aGlzLmdldE51bUdseXBocygpKzEpIDw8IDE7Ci0gICAgICAgIGlmIChsZW4gIT0gc2V0UG9zaXRpb25zLmxlbmd0aCl7Ci0gICAgICAgICAgICAvLyBhd3QuNDY9bGVuZ3RoIG9mIHNldFBvc2l0aW9ucyBhcnJheSBkaWZmZXJzIGZyb20gdGhlIGxlbmd0aCBvZiBwb3NpdGlvbnMgYXJyYXkKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkoc2V0UG9zaXRpb25zLCAwLCB2aXN1YWxQb3NpdGlvbnMsIDAsIGxlbik7Ci0gICAgICAgIGxheW91dEZsYWdzID0gbGF5b3V0RmxhZ3MgJiBGTEFHX0hBU19QT1NJVElPTl9BREpVU1RNRU5UUzsKLQotICAgIH0KLQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBnbHlwaCBjb2RlIG9mIHRoZSBzcGVjaWZpZWQgZ2x5cGguCi0gICAgICogCi0gICAgICogQHBhcmFtIGdseXBoSW5kZXggc3BlY2lmaWVkIGluZGV4IG9mIHRoZSBnbHlwaAotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0R2x5cGhDb2RlKGludCBnbHlwaEluZGV4KSB7Ci0gICAgICAgIGlmIChnbHlwaEluZGV4ID49IHRoaXMudmVjdG9yLmxlbmd0aCB8fCBnbHlwaEluZGV4IDwgMCkgewotICAgICAgICAgICAgLy8gYXd0LjQzPWdseXBoSW5kZXggaXMgb3V0IG9mIHZlY3RvcidzIGxpbWl0cwotICAgICAgICAgICAgdGhyb3cgbmV3IEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDMiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdGhpcy52ZWN0b3JbZ2x5cGhJbmRleF0uZ2V0R2x5cGhDb2RlKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBjaGFyYWN0ZXIgaW5kZXggb2YgdGhlIHNwZWNpZmllZCBnbHlwaC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZ2x5cGhJbmRleCBzcGVjaWZpZWQgaW5kZXggb2YgdGhlIGdseXBoCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRHbHlwaENoYXJJbmRleChpbnQgZ2x5cGhJbmRleCkgewotCi0gICAgICAgIGlmICgoZ2x5cGhJbmRleCA8IDApIHx8IChnbHlwaEluZGV4ID49IHRoaXMuZ2V0TnVtR2x5cGhzKCkpKSB7Ci0gICAgICAgICAgICAvLyBhd3QuNDM9Z2x5cGhJbmRleCBpcyBvdXQgb2YgdmVjdG9yJ3MgbGltaXRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjQzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLQotICAgICAgICBpZiAoKHRoaXMubGF5b3V0RmxhZ3MgJiBGb250LkxBWU9VVF9SSUdIVF9UT19MRUZUKSAhPSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gdGhpcy5jaGFyVmVjdG9yLmxlbmd0aCAtIGdseXBoSW5kZXggLSAxOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGdseXBoSW5kZXg7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIGNoYXJhY3RlciB2YWx1ZSBvZiB0aGUgc3BlY2lmaWVkIGdseXBoLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4IHNwZWNpZmllZCBpbmRleCBvZiB0aGUgZ2x5cGgKLSAgICAgKi8KLSAgICBwdWJsaWMgY2hhciBnZXRHbHlwaENoYXIoaW50IGdseXBoSW5kZXgpIHsKLQotICAgICAgICBpZiAoKGdseXBoSW5kZXggPCAwKSB8fCAoZ2x5cGhJbmRleCA+PSB0aGlzLmdldE51bUdseXBocygpKSkgewotICAgICAgICAgICAgLy8gYXd0LjQzPWdseXBoSW5kZXggaXMgb3V0IG9mIHZlY3RvcidzIGxpbWl0cwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB0aGlzLmNoYXJWZWN0b3JbZ2x5cGhJbmRleF07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQXNzaWducyBkZWZhdWx0IHBvc2l0aW9ucyB0byBlYWNoIGdseXBoIGluIHRoaXMgR2x5cGhWZWN0b3IuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgcGVyZm9ybURlZmF1bHRMYXlvdXQoKSB7Ci0KLSAgICAgICAgU3lzdGVtLmFycmF5Y29weShsb2dpY2FsUG9zaXRpb25zLCAwLCB2aXN1YWxQb3NpdGlvbnMsIDAsIGxvZ2ljYWxQb3NpdGlvbnMubGVuZ3RoKTsKLQotICAgICAgICAvLyBTZXQgcG9zaXRpb24gY2hhbmdlcyBmbGFnIHRvIHplcm8KLSAgICAgICAgY2xlYXJMYXlvdXRGbGFncyhHbHlwaFZlY3Rvci5GTEFHX0hBU19QT1NJVElPTl9BREpVU1RNRU5UUyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbnVtYmVyIG9mIGdseXBocyBpbiB0aGlzIEdseXBoIFZlY3RvcgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0TnVtR2x5cGhzKCkgewotICAgICAgICByZXR1cm4gdmVjdG9yLmxlbmd0aDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBsb2dpY2FsIGJvdW5kcyBvZiB0aGlzIEdseXBoVmVjdG9yCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGdldExvZ2ljYWxCb3VuZHMoKXsKLSAgICAgICAgLy8gWFhYOiBmb3IgdHJhbnNmb3JtcyB3aGVyZSBhbiBhbmdsZSBiZXR3ZWVuIGJhc2lzIHZlY3RvcnMgaXMgbm90IDkwIGRlZ3JlZXMKLSAgICAgICAgLy8gUmVjdGFubGdlMkQgY2xhc3MgZG9lc24ndCBmaXQgYXMgTG9naWNhbCBib3VuZHMuIEZvciB0aGlzIHJlYXNvbiB3ZSB1c2UKLSAgICAgICAgLy8gb25seSBub24tdHJhbnNmb3JtZWQgYm91bmRzISEKLQotICAgICAgICBmbG9hdCB4ID0gdmlzdWFsUG9zaXRpb25zWzBdOwotICAgICAgICBmbG9hdCB3aWR0aCA9IHZpc3VhbFBvc2l0aW9uc1t2aXN1YWxQb3NpdGlvbnMubGVuZ3RoLTJdOwotCi0gICAgICAgIGRvdWJsZSBzY2FsZVkgPSAgdHJhbnNmb3JtLmdldFNjYWxlWSgpOwotCi0gICAgICAgIFJlY3RhbmdsZTJEIGJvdW5kcyA9IG5ldyBSZWN0YW5nbGUyRC5GbG9hdCh4LCAoZmxvYXQpKCgtdGhpcy5hc2NlbnQtdGhpcy5sZWFkaW5nKSpzY2FsZVkpLCB3aWR0aCwgKGZsb2F0KSh0aGlzLmhlaWdodCpzY2FsZVkpKTsKLSAgICAgICAgcmV0dXJuIGJvdW5kczsKLSAgICB9Ci0KLQotICAgIC8qKgotICAgICAqIENoZWNrcyB3aGV0aGVyIGdpdmVuIEdseXBoVmVjdG9yIGVxdWFscyB0byB0aGlzIEdseXBoVmVjdG9yLgotICAgICAqIEBwYXJhbSBnbHlwaFZlY3RvciBHbHlwaFZlY3RvciBvYmplY3QgdG8gY29tcGFyZQotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGVxdWFscyhHbHlwaFZlY3RvciBnbHlwaFZlY3Rvcil7Ci0gICAgICAgIGlmIChnbHlwaFZlY3RvciA9PSB0aGlzKXsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGdseXBoVmVjdG9yICE9IG51bGwpIHsKLQotICAgICAgICAgICAgaWYgKCEoZ2x5cGhWZWN0b3IuZ2V0Rm9udFJlbmRlckNvbnRleHQoKS5lcXVhbHModGhpcy52ZWN0b3JGUkMpICYmCi0gICAgICAgICAgICAgICAgICAgICAgZ2x5cGhWZWN0b3IuZ2V0Rm9udCgpLmVxdWFscyh0aGlzLmZvbnQpKSl7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIGJvb2xlYW4gZXEgPSB0cnVlOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgZ2V0TnVtR2x5cGhzKCk7IGkrKykgewotCi0gICAgICAgICAgICAgICAgICAgIGludCBpZHggPSBpKjI7Ci0gICAgICAgICAgICAgICAgICAgIGVxID0gKCgoQ29tbW9uR2x5cGhWZWN0b3IpZ2x5cGhWZWN0b3IpLnZpc3VhbFBvc2l0aW9uc1tpZHhdID09IHRoaXMudmlzdWFsUG9zaXRpb25zW2lkeF0pICYmCi0gICAgICAgICAgICAgICAgICAgICAgICAoKChDb21tb25HbHlwaFZlY3RvcilnbHlwaFZlY3RvcikudmlzdWFsUG9zaXRpb25zW2lkeCsxXSA9PSB0aGlzLnZpc3VhbFBvc2l0aW9uc1tpZHgrMV0pICYmCi0gICAgICAgICAgICAgICAgICAgICAgICAoZ2x5cGhWZWN0b3IuZ2V0R2x5cGhDaGFySW5kZXgoaSkgPT0gdGhpcy5nZXRHbHlwaENoYXJJbmRleChpKSk7Ci0KLSAgICAgICAgICAgICAgICAgICAgaWYgKGVxKXsKLSAgICAgICAgICAgICAgICAgICAgICAgIEFmZmluZVRyYW5zZm9ybSB0cmFucyA9IGdseXBoVmVjdG9yLmdldEdseXBoVHJhbnNmb3JtKGkpOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHRyYW5zID09IG51bGwpewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGVxID0gKHRoaXMuZ2xzVHJhbnNmb3Jtc1tpXSA9PSBudWxsKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGVxID0gdGhpcy5nbHNUcmFuc2Zvcm1zW2ldLmVxdWFscyh0cmFucyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBpZiAoIWVxKXsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIHJldHVybiAgZXE7Ci0gICAgICAgICAgICB9IGNhdGNoIChDbGFzc0Nhc3RFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmbGFncyBkZXNjcmliaW5nIHRoZSBzdGF0ZSBvZiB0aGUgR2x5cGhWZWN0b3IuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRMYXlvdXRGbGFncygpIHsKLSAgICAgICAgcmV0dXJuIGxheW91dEZsYWdzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgY2hhciB3aXRoIHRoZSBzcGVjaWZpZWQgaW5kZXguCi0gICAgICogCi0gICAgICogQHBhcmFtIGluZGV4IHNwZWNpZmllZCBpbmRleCBvZiB0aGUgY2hhcgotICAgICAqIAotICAgICAqLwotICAgIHB1YmxpYyBjaGFyIGdldENoYXIoaW50IGluZGV4KSB7Ci0gICAgICAgIHJldHVybiB0aGlzLmNoYXJWZWN0b3JbaW5kZXhdOwotCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2xlYXIgZGVzaXJlZCBmbGFncyBpbiBsYXlvdXQgZmxhZ3MgZGVzY3JpYmluZyB0aGUgc3RhdGUuIAotICAgICAqIAotICAgICAqIEBwYXJhbSBjbGVhckZsYWdzIGZsYWdzIG1hc2sgdG8gY2xlYXIgCi0gICAgICovCi0gICAgCi0gICAgcHJpdmF0ZSB2b2lkIGNsZWFyTGF5b3V0RmxhZ3MoaW50IGNsZWFyRmxhZ3MpewotICAgICAgICBsYXlvdXRGbGFncyAmPSB+Y2xlYXJGbGFnczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBsb2dpY2FsIGJvdW5kcyBvZiB0aGUgc3BlY2lmaWVkIGdseXBoIHdpdGhpbiB0aGlzIENvbW1vbkdseXBoVmVjdG9yLgotICAgICAqIAotICAgICAqIEBwYXJhbSBnbHlwaEluZGV4IGluZGV4IG9mIHRoZSBnbHlwaCB0byBnZXQgaXQncyBsb2dpY2FsIGJvdW5kcwotICAgICAqIEByZXR1cm4gbG9naWNhbCBib3VuZHMgb2YgdGhlIHNwZWNpZmllZCBnbHlwaAotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTaGFwZSBnZXRHbHlwaExvZ2ljYWxCb3VuZHMoaW50IGdseXBoSW5kZXgpewotICAgICAgICBpZiAoKGdseXBoSW5kZXggPCAwKSB8fCAoZ2x5cGhJbmRleCA+PSB0aGlzLmdldE51bUdseXBocygpKSl7Ci0gICAgICAgICAgICAvLyBhd3QuNDM9Z2x5cGhJbmRleCBpcyBvdXQgb2YgdmVjdG9yJ3MgbGltaXRzCi0gICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC40MyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIEdseXBoIGdseXBoID0gdGhpcy52ZWN0b3JbZ2x5cGhJbmRleF07Ci0KLSAgICAgICAgZmxvYXQgeDAgPSB2aXN1YWxQb3NpdGlvbnNbZ2x5cGhJbmRleCoyXTsKLSAgICAgICAgZmxvYXQgeTAgPSB2aXN1YWxQb3NpdGlvbnNbZ2x5cGhJbmRleCoyKzFdOwotICAgICAgICBmbG9hdCBhZHZhbmNlWCA9IGdseXBoLmdldEdseXBoUG9pbnRNZXRyaWNzKCkuZ2V0QWR2YW5jZVgoKTsKLQotICAgICAgICBHZW5lcmFsUGF0aCBncCA9IG5ldyBHZW5lcmFsUGF0aCgpOwotICAgICAgICBncC5tb3ZlVG8oMCwgLWFzY2VudCAtIGxlYWRpbmcpOwotICAgICAgICBncC5saW5lVG8oYWR2YW5jZVggLC1hc2NlbnQgLSBsZWFkaW5nKTsKLSAgICAgICAgZ3AubGluZVRvKGFkdmFuY2VYLCBkZXNjZW50KTsKLSAgICAgICAgZ3AubGluZVRvKDAsIGRlc2NlbnQpOwotICAgICAgICBncC5saW5lVG8oMCwgLWFzY2VudCAtIGxlYWRpbmcpOwotICAgICAgICBncC5jbG9zZVBhdGgoKTsKLQotICAgICAgICAvKiBBcHBseWluZyBHbHlwaFZlY3RvciBmb250IHRyYW5zZm9ybSAqLwotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gYXQgPSAoQWZmaW5lVHJhbnNmb3JtKXRoaXMudHJhbnNmb3JtLmNsb25lKCk7Ci0KLSAgICAgICAgLyogQXBwbHlpbmcgR2x5cGggdHJhbnNmb3JtICovCi0gICAgICAgIEFmZmluZVRyYW5zZm9ybSBnbHlwaFRyYW5zZm9ybSA9IGdldEdseXBoVHJhbnNmb3JtKGdseXBoSW5kZXgpOwotICAgICAgICBpZiAoZ2x5cGhUcmFuc2Zvcm0gIT0gbnVsbCl7Ci0gICAgICAgICAgICBhdC5jb25jYXRlbmF0ZShnbHlwaFRyYW5zZm9ybSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKiBBcHBseWluZyB0cmFuc2xhdGlvbiB0byBhY3R1YWwgdmlzdWFsIGJvdW5kcyAqLwotICAgICAgICBhdC5wcmVDb25jYXRlbmF0ZShBZmZpbmVUcmFuc2Zvcm0uZ2V0VHJhbnNsYXRlSW5zdGFuY2UoeDAsIHkwKSk7Ci0gICAgICAgIGdwLnRyYW5zZm9ybShhdCk7Ci0gICAgICAgIHJldHVybiBncDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBGb250IHBhcmFtZXRlciBvZiB0aGlzIEdseXBoVmVjdG9yCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEZvbnQgZ2V0Rm9udCgpewotICAgICAgICByZXR1cm4gdGhpcy5mb250OwotICAgIH0KLQotCi19ClwgTm8gbmV3bGluZSBhdCBlbmQgb2YgZmlsZQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Db21wb3NpdGVGb250LmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0NvbXBvc2l0ZUZvbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzBjYjMzNC4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0NvbXBvc2l0ZUZvbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ4NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuZm9udC5Gb250UmVuZGVyQ29udGV4dDsKLWltcG9ydCBqYXZhLmF3dC5mb250LkxpbmVNZXRyaWNzOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuRm9udFBlZXJJbXBsOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5Gb250UHJvcGVydHk7Ci0KLS8qKgotICogQ29tcG9zaXRlRm9udCBjbGFzcyBpcyB0aGUgaW1wbGVtZW50YXRpb24gb2YgbG9naWNhbCBmb250IGNsYXNzZXMuIAotICogRXZlcnkgbG9naWNhbCBmb250IGNvbnNpc3RzIG9mIHNldmVyYWwgcGh5c2ljYWwgZm9udHMgdGhhdCBkZXNjcmliZWQgCi0gKiBpbiBmb250LnByb3BlcnRpZXMgZmlsZSBhY2NvcmRpbmcgdG8gdGhlIGZhY2UgbmFtZSBvZiB0aGlzIGxvZ2ljYWwgZm9udC4KLSAqLwotcHVibGljIGNsYXNzIENvbXBvc2l0ZUZvbnQgZXh0ZW5kcyBGb250UGVlckltcGx7Ci0gICAgCi0gICAgLy8gYSBudW1iZXIgb2YgcGh5c2ljYWwgZm9udHMgdGhhdCBDb21wb3NpdGVGb250IGNvbnNpc3Qgb2YgCi0gICAgaW50IG51bUZvbnRzOwotCi0gICAgLy8gZm9udCBmYW1pbHkgbmFtZQotICAgIFN0cmluZyBmYW1pbHk7Ci0KLSAgICAvLyBmb250IGZhY2UgbmFtZQotICAgIFN0cmluZyBmYWNlOwotCi0gICAgU3RyaW5nW10gZm9udE5hbWVzOwotICAgIAotICAgIC8vIGFuIGFycmF5IG9mIGZvbnQgcHJvcGVydGllcyBhcHBsaWNhYmxlIHRvIHRoaXMgQ29tcG9zaXRlRm9udAotICAgIEZvbnRQcm9wZXJ0eVtdIGZvbnRQcm9wZXJ0aWVzOwotICAgIAotICAgIC8vIGFuIGFycmF5IG9mIGZvbnQgcGVlcnMgYXBwbGljYWJsZSB0byB0aGlzIENvbXBvc2l0ZUZvbnQKLSAgICBwdWJsaWMgRm9udFBlZXJJbXBsW10gZlBoeXNpY2FsRm9udHM7Ci0gICAgCi0gICAgLy8gbWlzc2luZyBnbHlwaCBjb2RlIGZpZWxkCi0gICAgaW50IG1pc3NpbmdHbHlwaENvZGUgPSAtMTsKLSAgICAKLSAgICAvLyBsaW5lIG1ldHJpY3Mgb2YgdGhpcyBmb250Ci0gICAgTGluZU1ldHJpY3NJbXBsIG5sbSA9IG51bGw7Ci0gICAgCi0gICAgLy8gY2FjaGVkIG51bSBnbHlwaHMgcGFyYW1ldGVyIG9mIHRoaXMgZm9udCB0aGF0IGlzIHRoZSBzdW0gb2YgbnVtIGdseXBocyBvZiAKLSAgICAvLyBmb250IHBlZXJzIGNvbXBvc2luZyB0aGlzIGZvbnQKLSAgICBpbnQgY2FjaGVkTnVtR2x5cGhzID0gLTE7Ci0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBDb21wb3NpdGVGb250IG9iamVjdCB0aGF0IGlzIGNvcnJlc3BvbmRpbmcgdG8gdGhlIHNwZWNpZmllZCBsb2dpY2FsIAotICAgICAqIGZhbWlseSBuYW1lLgotICAgICAqIAotICAgICAqIEBwYXJhbSBmYW1pbHlOYW1lIGxvZ2ljYWwgZmFtaWx5IG5hbWUgQ29tcG9zaXRlRm9udCBpcyB0byBiZSBjcmVhdGVkIGZyb20KLSAgICAgKiBAcGFyYW0gZmFjZU5hbWUgbG9naWNhbCBmYWNlIG5hbWUgQ29tcG9zaXRlRm9udCBpcyB0byBiZSBjcmVhdGVkIGZyb20KLSAgICAgKiBAcGFyYW0gX3N0eWxlIHN0eWxlIG9mIHRoZSBDb21wb3NpdGVGb250IHRvIGJlIGNyZWF0ZWQKLSAgICAgKiBAcGFyYW0gX3NpemUgc2l6ZSBvZiB0aGUgQ29tcG9zaXRlRm9udCB0byBiZSBjcmVhdGVkIAotICAgICAqIEBwYXJhbSBmUHJvcGVydGllcyBhbiBhcnJheSBvZiBGb250UHJvcGVydGllcyBkZXNjcmliaW5nIHBoeXNpY2FsIGZvbnRzIC0gCi0gICAgICogcGFydHMgb2YgbG9naWNhbCBmb250Ci0gICAgICogQHBhcmFtIHBoeXNGb250cyBhbiBhcnJheSBvZiBwaHlzaWNhbCBmb250IHBlZXJzIHJlbGF0ZWQgdG8gdGhlIENvbXBvc2l0ZUZvbnQKLSAgICAgKiB0byBiZSBjcmVhdGVkCi0gICAgICovCi0gICAgcHVibGljIENvbXBvc2l0ZUZvbnQoU3RyaW5nIGZhbWlseU5hbWUsIFN0cmluZyBmYWNlTmFtZSwgaW50IF9zdHlsZSwgaW50IF9zaXplLCBGb250UHJvcGVydHlbXSBmUHJvcGVydGllcywgRm9udFBlZXJJbXBsW10gcGh5c0ZvbnRzKXsKLSAgICAgICAgdGhpcy5zaXplID0gX3NpemU7Ci0gICAgICAgIHRoaXMubmFtZSA9IGZhY2VOYW1lOwotICAgICAgICB0aGlzLmZhbWlseSA9IGZhbWlseU5hbWU7Ci0gICAgICAgIHRoaXMuc3R5bGUgPSBfc3R5bGU7Ci0gICAgICAgIHRoaXMuZmFjZSA9IGZhY2VOYW1lOwotICAgICAgICB0aGlzLnBzTmFtZSA9IGZhY2VOYW1lOwotICAgICAgICB0aGlzLmZvbnRQcm9wZXJ0aWVzID0gZlByb3BlcnRpZXM7Ly8gISEgU3VwcG9zZWQgdGhhdCBmUHJvcGVydGllcyBwYXJhbWV0ZXIgIT0gbnVsbAotICAgICAgICBmUGh5c2ljYWxGb250cyA9IHBoeXNGb250czsKLSAgICAgICAgbnVtRm9udHMgPSBmUGh5c2ljYWxGb250cy5sZW5ndGg7IAotICAgICAgICBzZXREZWZhdWx0TGluZU1ldHJpY3MoIiIsIG51bGwpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIHRoaXMudW5pZm9ybUxNID0gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgaW5kZXggb2YgdGhlIEZvbnRQZWVyIGluIGFycmF5IG9mIHBoeXNpY2FsIGZvbnRzIHRoYXQgaXMgYXBwbGljYWJsZSAKLSAgICAgKiBmb3IgdGhlIGdpdmVuIGNoYXJhY3Rlci4gVGhpcyBmb250IGhhcyB0byBoYXZlIHRoZSBoaWdoZXN0IHByaW9yaXR5IGFtb25nIGZvbnRzCi0gICAgICogdGhhdCBjYW4gZGlzcGxheSB0aGlzIGNoYXJhY3RlciBhbmQgZG9uJ3QgaGF2ZSBleGNsdXNpb24gcmFuZ2UgY292ZXJpbmcgCi0gICAgICogc3BlY2lmaWVkIGNoYXJhY3Rlci4gSWYgdGhlcmUgaXMgbm8gZGVzaXJlZCBmb250cyAtMSBpcyByZXR1cm5lZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gY2hyIHNwZWNpZmllZCBjaGFyYWN0ZXIKLSAgICAgKiBAcmV0dXJuIGluZGV4IG9mIHRoZSBmb250IGZyb20gdGhlIGFycmF5IG9mIHBoeXNpY2FsIGZvbnRzIHRoYXQgd2lsbCBiZSB1c2VkIAotICAgICAqIGR1cmluZyBwcm9jZXNzaW5nIG9mIHRoZSBzcGVjaWZpZWQgY2hhcmFjdGVyLiAKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldENoYXJGb250SW5kZXgoY2hhciBjaHIpewotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUZvbnRzOyBpKyspewotICAgICAgICAgICAgaWYgKGZvbnRQcm9wZXJ0aWVzW2ldLmlzQ2hhckV4Y2x1ZGVkKGNocikpewotICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGZQaHlzaWNhbEZvbnRzW2ldLmNhbkRpc3BsYXkoY2hyKSl7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgaW5kZXggb2YgdGhlIEZvbnRQZWVyIGluIGFycmF5IG9mIHBoeXNpY2FsIGZvbnRzIHRoYXQgaXMgYXBwbGljYWJsZSAKLSAgICAgKiBmb3IgdGhlIGdpdmVuIGNoYXJhY3Rlci4gVGhpcyBmb250IGhhcyB0byBoYXZlIHRoZSBoaWdoZXN0IHByaW9yaXR5IGFtb25nIGZvbnRzCi0gICAgICogdGhhdCBjYW4gZGlzcGxheSB0aGlzIGNoYXJhY3RlciBhbmQgZG9uJ3QgaGF2ZSBleGNsdXNpb24gcmFuZ2UgY292ZXJpbmcgCi0gICAgICogc3BlY2lmaWVkIGNoYXJhY3Rlci4gSWYgdGhlcmUgaXMgbm8gZGVzaXJlZCBmb250cyBkZWZhdWx0IHZhbHVlIGlzIHJldHVybmVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaHIgc3BlY2lmaWVkIGNoYXJhY3RlcgotICAgICAqIEBwYXJhbSBkZWZhdWx0VmFsdWUgZGVmYXVsdCBpbmRleCB0aGF0IGlzIHJldHVybmVkIGlmIHRoZSBuZWNlc3NhcnkgZm9udCBjb3VsZG4ndCBiZSBmb3VuZC4KLSAgICAgKiBAcmV0dXJuIGluZGV4IG9mIHRoZSBmb250IGZyb20gdGhlIGFycmF5IG9mIHBoeXNpY2FsIGZvbnRzIHRoYXQgd2lsbCBiZSB1c2VkIAotICAgICAqIGR1cmluZyBwcm9jZXNzaW5nIG9mIHRoZSBzcGVjaWZpZWQgY2hhcmFjdGVyLiAKLSAgICAgKi8KLSAgICAgcHVibGljIGludCBnZXRDaGFyRm9udEluZGV4KGNoYXIgY2hyLCBpbnQgZGVmYXVsdFZhbHVlKXsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1Gb250czsgaSsrKXsKLSAgICAgICAgICAgIGlmIChmb250UHJvcGVydGllc1tpXS5pc0NoYXJFeGNsdWRlZChjaHIpKXsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChmUGh5c2ljYWxGb250c1tpXS5jYW5EaXNwbGF5KGNocikpewotICAgICAgICAgICAgICAgIHJldHVybiBpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGRlZmF1bHRWYWx1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgb25lIG9mIHRoZSBwaHlzaWNhbCBmb250cyBjb21wb3NpbmcgdGhpcyBmb250IENvbXBvc2l0ZUZvbnQgCi0gICAgICogY2FuIGRpc3BsYXkgc3BlY2lmaWVkIGNoYXJhY3Rlci4KLSAgICAgKiAgIAotICAgICAqIEBwYXJhbSBjaHIgc3BlY2lmaWVkIGNoYXJhY3RlcgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBib29sZWFuIGNhbkRpc3BsYXkoY2hhciBjaHIpewotICAgICAgICByZXR1cm4gKGdldENoYXJGb250SW5kZXgoY2hyKSAhPSAtMSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBsb2dpY2FsIGFzY2VudCAoaW4gcGl4ZWxzKQotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0QXNjZW50KCl7Ci0gICAgICAgIHJldHVybiBubG0uZ2V0TG9naWNhbEFzY2VudCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgTGluZU1ldHJpY3MgaW5zdGFuY2Ugc2NhbGVkIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkIHRyYW5zZm9ybS4gIAotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHIgc3BlY2lmaWVkIFN0cmluZyAKLSAgICAgKiBAcGFyYW0gZnJjIHNwZWNpZmllZCBGb250UmVuZGVyQ29udGV4dCAKLSAgICAgKiBAcGFyYW0gYXQgc3BlY2lmaWVkIEFmZmluZVRyYW5zZm9ybQotICAgICAqLwotICAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgTGluZU1ldHJpY3MgZ2V0TGluZU1ldHJpY3MoU3RyaW5nIHN0ciwgRm9udFJlbmRlckNvbnRleHQgZnJjICwgQWZmaW5lVHJhbnNmb3JtIGF0KXsKLSAgICAgICAgTGluZU1ldHJpY3NJbXBsIGxtID0gKExpbmVNZXRyaWNzSW1wbCkodGhpcy5ubG0uY2xvbmUoKSk7Ci0gICAgICAgIGxtLnNldE51bUNoYXJzKHN0ci5sZW5ndGgoKSk7Ci0KLSAgICAgICAgaWYgKChhdCAhPSBudWxsKSAmJiAoIWF0LmlzSWRlbnRpdHkoKSkpewotICAgICAgICAgICAgbG0uc2NhbGUoKGZsb2F0KWF0LmdldFNjYWxlWCgpLCAoZmxvYXQpYXQuZ2V0U2NhbGVZKCkpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGxtOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgY2FjaGVkIExpbmVNZXRyaWNzIGluc3RhbmNlIGZvciB0aGUgbnVsbCBzdHJpbmcgb3IgY3JlYXRlcyBpdCBpZgotICAgICAqIGl0IHdhc24ndCBjYWNoZWQgeWV0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBMaW5lTWV0cmljcyBnZXRMaW5lTWV0cmljcygpewotICAgICAgICBpZiAobmxtID09IG51bGwpewotICAgICAgICAgICAgc2V0RGVmYXVsdExpbmVNZXRyaWNzKCIiLCBudWxsKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHRoaXMubmxtOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgTGluZU1ldHJpY3MgaW5zdGFuY2UgYW5kIHNldCBjYWNoZWQgTGluZU1ldHJpY3MgZmllbGQgdG8gaXQuCi0gICAgICogQ3JlYXRlZCBMaW5lTWV0cmljcyBoYXMgbWF4aW11bSB2YWx1ZXMgb2YgdGhlIGlkaXZpZHVhbCBtZXRyaWNzIG9mIGFsbAotICAgICAqIGNvbXBvc2luZyBwaHlzaWNhbCBmb250cy4gSWYgdGhlcmUgaXMgb25seSBvbmUgcGh5c2ljYWwgZm9udCAtIGl0J3MgCi0gICAgICogTGluZU1ldHJpY3Mgb2JqZWN0IGlzIHJldHVybmVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHIgc3BlY2lmaWVkIFN0cmluZyAKLSAgICAgKiBAcGFyYW0gZnJjIHNwZWNpZmllZCBGb250UmVuZGVyQ29udGV4dCAKLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgc2V0RGVmYXVsdExpbmVNZXRyaWNzKFN0cmluZyBzdHIsIEZvbnRSZW5kZXJDb250ZXh0IGZyYyl7Ci0gICAgICAgIExpbmVNZXRyaWNzIGxtID0gZlBoeXNpY2FsRm9udHNbMF0uZ2V0TGluZU1ldHJpY3Moc3RyLCBmcmMsIG51bGwpOwotICAgICAgICBmbG9hdCBtYXhDaGFyV2lkdGggPSAoZmxvYXQpZlBoeXNpY2FsRm9udHNbMF0uZ2V0TWF4Q2hhckJvdW5kcyhmcmMpLmdldFdpZHRoKCk7Ci0KLSAgICAgICAgaWYgKG51bUZvbnRzID09IDEpIHsKLSAgICAgICAgICAgIHRoaXMubmxtID0gKExpbmVNZXRyaWNzSW1wbClsbTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGZsb2F0W10gYmFzZWxpbmVPZmZzZXRzID0gbG0uZ2V0QmFzZWxpbmVPZmZzZXRzKCk7Ci0gICAgICAgIGludCBudW1DaGFycyA9IHN0ci5sZW5ndGgoKTsKLQotICAgICAgICAvLyBYWFg6IGRlZmF1bHQgdmFsdWUgLSBjb21tb24gZm9yIGFsbCBGb250cwotICAgICAgICBpbnQgYmFzZUxpbmVJbmRleCA9IGxtLmdldEJhc2VsaW5lSW5kZXgoKTsKLQotICAgICAgICBmbG9hdCBtYXhVbmRlcmxpbmVUaGlja25lc3MgPSBsbS5nZXRVbmRlcmxpbmVUaGlja25lc3MoKTsKLSAgICAgICAgZmxvYXQgbWF4VW5kZXJsaW5lT2Zmc2V0ID0gbG0uZ2V0VW5kZXJsaW5lT2Zmc2V0KCk7Ci0gICAgICAgIGZsb2F0IG1heFN0cmlrZXRocm91Z2hUaGlja25lc3MgPSBsbS5nZXRTdHJpa2V0aHJvdWdoVGhpY2tuZXNzKCk7Ci0gICAgICAgIGZsb2F0IG1pblN0cmlrZXRocm91Z2hPZmZzZXQgPSBsbS5nZXRTdHJpa2V0aHJvdWdoT2Zmc2V0KCk7Ci0gICAgICAgIGZsb2F0IG1heExlYWRpbmcgPSBsbS5nZXRMZWFkaW5nKCk7ICAvLyBFeHRlcm5hbCBsZWFkaW5nCi0gICAgICAgIGZsb2F0IG1heEhlaWdodCA9IGxtLmdldEhlaWdodCgpOyAgIC8vIEhlaWdodCBvZiB0aGUgZm9udCAoID09IChhc2NlbnQgKyBkZXNjZW50ICsgbGVhZGluZykpCi0gICAgICAgIGZsb2F0IG1heEFzY2VudCA9IGxtLmdldEFzY2VudCgpOyAgIC8vIEFzY2VudCBvZiB0aGUgZm9udAotICAgICAgICBmbG9hdCBtYXhEZXNjZW50ID0gbG0uZ2V0RGVzY2VudCgpOyAvLyBEZXNjZW50IG9mIHRoZSBmb250Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDE7IGkgPCBudW1Gb250czsgaSsrKXsKLSAgICAgICAgICAgIGxtID0gZlBoeXNpY2FsRm9udHNbaV0uZ2V0TGluZU1ldHJpY3Moc3RyLCBmcmMsIG51bGwpOwotICAgICAgICAgICAgaWYgKG1heFVuZGVybGluZVRoaWNrbmVzcyA8IGxtLmdldFVuZGVybGluZVRoaWNrbmVzcygpKXsKLSAgICAgICAgICAgICAgICBtYXhVbmRlcmxpbmVUaGlja25lc3MgPSBsbS5nZXRVbmRlcmxpbmVUaGlja25lc3MoKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKG1heFVuZGVybGluZU9mZnNldCA8IGxtLmdldFVuZGVybGluZU9mZnNldCgpKXsKLSAgICAgICAgICAgICAgICBtYXhVbmRlcmxpbmVPZmZzZXQgPSBsbS5nZXRVbmRlcmxpbmVPZmZzZXQoKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKG1heFN0cmlrZXRocm91Z2hUaGlja25lc3MgPCBsbS5nZXRTdHJpa2V0aHJvdWdoVGhpY2tuZXNzKCkpewotICAgICAgICAgICAgICAgIG1heFN0cmlrZXRocm91Z2hUaGlja25lc3MgPSBsbS5nZXRTdHJpa2V0aHJvdWdoVGhpY2tuZXNzKCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChtaW5TdHJpa2V0aHJvdWdoT2Zmc2V0ID4gbG0uZ2V0U3RyaWtldGhyb3VnaE9mZnNldCgpKXsKLSAgICAgICAgICAgICAgICBtaW5TdHJpa2V0aHJvdWdoT2Zmc2V0ID0gbG0uZ2V0U3RyaWtldGhyb3VnaE9mZnNldCgpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAobWF4TGVhZGluZyA8IGxtLmdldExlYWRpbmcoKSl7Ci0gICAgICAgICAgICAgICAgbWF4TGVhZGluZyA9IGxtLmdldExlYWRpbmcoKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKG1heEFzY2VudCA8IGxtLmdldEFzY2VudCgpKXsKLSAgICAgICAgICAgICAgICBtYXhBc2NlbnQgPSBsbS5nZXRBc2NlbnQoKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKG1heERlc2NlbnQgPCBsbS5nZXREZXNjZW50KCkpewotICAgICAgICAgICAgICAgIG1heERlc2NlbnQgPSBsbS5nZXREZXNjZW50KCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGZsb2F0IHdpZHRoID0gKGZsb2F0KWZQaHlzaWNhbEZvbnRzW2ldLmdldE1heENoYXJCb3VuZHMoZnJjKS5nZXRXaWR0aCgpOwotICAgICAgICAgICAgaWYobWF4Q2hhcldpZHRoIDwgd2lkdGgpewotICAgICAgICAgICAgICAgIG1heENoYXJXaWR0aCA9IHdpZHRoOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZm9yIChpbnQgaiA9MDsgaiA8IGJhc2VsaW5lT2Zmc2V0cy5sZW5ndGg7IGorKyl7Ci0gICAgICAgICAgICAgICAgZmxvYXRbXSBvZmZzZXRzID0gbG0uZ2V0QmFzZWxpbmVPZmZzZXRzKCk7Ci0gICAgICAgICAgICAgICAgaWYgKGJhc2VsaW5lT2Zmc2V0c1tqXSA+IG9mZnNldHNbal0pewotICAgICAgICAgICAgICAgICAgICBiYXNlbGluZU9mZnNldHNbal0gPSBvZmZzZXRzW2pdOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICB9Ci0gICAgICAgIG1heEhlaWdodCA9IG1heEFzY2VudCArIG1heERlc2NlbnQgKyBtYXhMZWFkaW5nOwotCi0gICAgICAgIHRoaXMubmxtID0gIG5ldyBMaW5lTWV0cmljc0ltcGwoCi0gICAgICAgICAgICAgICAgbnVtQ2hhcnMsCi0gICAgICAgICAgICAgICAgYmFzZUxpbmVJbmRleCwKLSAgICAgICAgICAgICAgICBiYXNlbGluZU9mZnNldHMsCi0gICAgICAgICAgICAgICAgbWF4VW5kZXJsaW5lVGhpY2tuZXNzLAotICAgICAgICAgICAgICAgIG1heFVuZGVybGluZU9mZnNldCwKLSAgICAgICAgICAgICAgICBtYXhTdHJpa2V0aHJvdWdoVGhpY2tuZXNzLAotICAgICAgICAgICAgICAgIG1pblN0cmlrZXRocm91Z2hPZmZzZXQsCi0gICAgICAgICAgICAgICAgbWF4TGVhZGluZywKLSAgICAgICAgICAgICAgICBtYXhIZWlnaHQsCi0gICAgICAgICAgICAgICAgbWF4QXNjZW50LAotICAgICAgICAgICAgICAgIG1heERlc2NlbnQsCi0gICAgICAgICAgICAgICAgbWF4Q2hhcldpZHRoKTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIG51bWJlciBvZiBnbHlwaHMgaW4gdGhpcyBDb21wb3NpdGVGb250IG9iamVjdC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldE51bUdseXBocygpewotICAgICAgICBpZiAodGhpcy5jYWNoZWROdW1HbHlwaHMgPT0gLTEpewotCi0gICAgICAgICAgICB0aGlzLmNhY2hlZE51bUdseXBocyA9IDA7Ci0KLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtRm9udHM7IGkrKyl7Ci0gICAgICAgICAgICAgICAgdGhpcy5jYWNoZWROdW1HbHlwaHMgKz0gZlBoeXNpY2FsRm9udHNbaV0uZ2V0TnVtR2x5cGhzKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gdGhpcy5jYWNoZWROdW1HbHlwaHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgaXRhbGljIGFuZ2xlIG9mIHRoaXMgb2JqZWN0LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRJdGFsaWNBbmdsZSgpewotICAgICAgICAvLyAhISBvbmx5IGZpcnN0IHBoeXNpY2FsIGZvbnQgdXNlZCB0byBnZXQgdGhpcyB2YWx1ZQotICAgICAgICByZXR1cm4gZlBoeXNpY2FsRm9udHNbMF0uZ2V0SXRhbGljQW5nbGUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHJlY3RhbmdsZSB0aGF0IGJvdW5kcyB0aGUgc3BlY2lmaWVkIHN0cmluZyBpbiB0ZXJtcyBvZiBjb21wb3NpdGUgbGluZSBtZXRyaWNzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaGFycyBhbiBhcnJheSBvZiBjaGFycwotICAgICAqIEBwYXJhbSBzdGFydCB0aGUgaW5pdGlhbCBvZmZzZXQgaW4gYXJyYXkgb2YgY2hhcnMKLSAgICAgKiBAcGFyYW0gZW5kIHRoZSBlbmQgb2Zmc2V0IGluIGFycmF5IG9mIGNoYXJzCi0gICAgICogQHBhcmFtIGZyYyBzcGVjaWZpZWQgRm9udFJlbmRlckNvbnRleHQKLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0U3RyaW5nQm91bmRzKGNoYXJbXSBjaGFycywgaW50IHN0YXJ0LCBpbnQgZW5kLCBGb250UmVuZGVyQ29udGV4dCBmcmMpewotCi0gICAgICAgIExpbmVNZXRyaWNzIGxtID0gZ2V0TGluZU1ldHJpY3MoKTsKLSAgICAgICAgZmxvYXQgbWluWSA9IC1sbS5nZXRBc2NlbnQoKTsKLSAgICAgICAgZmxvYXQgbWluWCA9IDA7Ci0gICAgICAgIGZsb2F0IGhlaWdodCA9IGxtLmdldEhlaWdodCgpOwotICAgICAgICBmbG9hdCB3aWR0aCA9IDA7Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IHN0YXJ0OyBpIDwgZW5kOyBpKyspewotICAgICAgICAgICAgd2lkdGggKz0gY2hhcldpZHRoKGNoYXJzW2ldKTsKLSAgICAgICAgfQotCi0gICAgICAgIFJlY3RhbmdsZTJEIHJlY3QyRCA9IG5ldyBSZWN0YW5nbGUyRC5GbG9hdChtaW5YLCBtaW5ZLCB3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgcmV0dXJuIHJlY3QyRDsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgbWF4aW11bSByZWN0YW5nbGUgdGhhdCBlbmNsb3NlcyBhbGwgbWF4aW11bSBjaGFyIGJvdW5kcyBvZiAKLSAgICAgKiBwaHlzaWNhbCBmb250cyBjb21wb3NpbmcgdGhpcyBDb21wb3NpdGVGb250LgotICAgICAqICAKLSAgICAgKiBAcGFyYW0gZnJjIHNwZWNpZmllZCBGb250UmVuZGVyQ29udGV4dAotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBSZWN0YW5nbGUyRCBnZXRNYXhDaGFyQm91bmRzKEZvbnRSZW5kZXJDb250ZXh0IGZyYyl7Ci0KLSAgICAgICAgUmVjdGFuZ2xlMkQgcmVjdDJEID0gZlBoeXNpY2FsRm9udHNbMF0uZ2V0TWF4Q2hhckJvdW5kcyhmcmMpOwotICAgICAgICBmbG9hdCBtaW5ZID0gKGZsb2F0KXJlY3QyRC5nZXRZKCk7Ci0gICAgICAgIGZsb2F0IG1heFdpZHRoID0gKGZsb2F0KXJlY3QyRC5nZXRXaWR0aCgpOwotICAgICAgICBmbG9hdCBtYXhIZWlnaHQgPSAoZmxvYXQpcmVjdDJELmdldEhlaWdodCgpOwotICAgICAgICBpZiAobnVtRm9udHMgPT0gMSl7Ci0gICAgICAgICAgICByZXR1cm4gcmVjdDJEOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaSA9IDE7IGkgPCBudW1Gb250czsgaSsrKXsKLSAgICAgICAgICAgIGlmIChmUGh5c2ljYWxGb250c1tpXSAhPSBudWxsKXsKLSAgICAgICAgICAgICAgICByZWN0MkQgPSBmUGh5c2ljYWxGb250c1tpXS5nZXRNYXhDaGFyQm91bmRzKGZyYyk7Ci0gICAgICAgICAgICAgICAgZmxvYXQgeSA9IChmbG9hdClyZWN0MkQuZ2V0WSgpOwotICAgICAgICAgICAgICAgIGZsb2F0IG1XaWR0aCA9IChmbG9hdClyZWN0MkQuZ2V0V2lkdGgoKTsKLSAgICAgICAgICAgICAgICBmbG9hdCBtSGVpZ2h0ID0gKGZsb2F0KXJlY3QyRC5nZXRIZWlnaHQoKTsKLSAgICAgICAgICAgICAgICBpZiAoeSA8IG1pblkpewotICAgICAgICAgICAgICAgICAgICBtaW5ZID0geTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKG1XaWR0aCA+IG1heFdpZHRoKXsKLSAgICAgICAgICAgICAgICAgICAgbWF4SGVpZ2h0ID0gbVdpZHRoOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICBpZiAobUhlaWdodCA+IG1heEhlaWdodCl7Ci0gICAgICAgICAgICAgICAgICAgIG1heEhlaWdodCA9IG1IZWlnaHQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmVjdDJEID0gbmV3IFJlY3RhbmdsZTJELkZsb2F0KDAsIG1pblksIG1heFdpZHRoLCBtYXhIZWlnaHQpOwotCi0gICAgICAgIHJldHVybiByZWN0MkQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmb250IG5hbWUuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyBnZXRGb250TmFtZSgpewotICAgICAgICByZXR1cm4gZmFjZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGZvbnQgcG9zdHNjcmlwdCBuYW1lLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0UFNOYW1lKCl7Ci0gICAgICAgIHJldHVybiBwc05hbWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmb250IGZhbWlseSBuYW1lLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RmFtaWx5KCl7Ci0gICAgICAgIHJldHVybiBmYW1pbHk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgY29kZSBvZiB0aGUgbWlzc2luZyBnbHlwaC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGdldE1pc3NpbmdHbHlwaENvZGUoKXsKLSAgICAgICAgLy8gISEgb25seSBmaXJzdCBwaHlzaWNhbCBmb250IHVzZWQgdG8gZ2V0IHRoaXMgdmFsdWUKLSAgICAgICAgcmV0dXJuIGZQaHlzaWNhbEZvbnRzWzBdLmdldE1pc3NpbmdHbHlwaENvZGUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIEdseXBoIG9iamVjdCBjb3JyZXNwb25kaW5nIHRvIHRoZSBzcGVjaWZpZWQgY2hhcmFjdGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaCBzcGVjaWZpZWQgY2hhcgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBHbHlwaCBnZXRHbHlwaChjaGFyIGNoKXsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1Gb250czsgaSsrKXsKLSAgICAgICAgICAgIGlmIChmb250UHJvcGVydGllc1tpXS5pc0NoYXJFeGNsdWRlZChjaCkpewotICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIAotICAgICAgICAgICAgLyogQ29udHJvbCBzeW1ib2xzIGNvbnNpZGVyZWQgdG8gYmUgc3VwcG9ydGVkIGJ5IHRoZSBmb250IHBlZXIgKi8KLSAgICAgICAgICAgIGlmICgoY2ggPCAweDIwKSB8fCBmUGh5c2ljYWxGb250c1tpXS5jYW5EaXNwbGF5KGNoKSl7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZQaHlzaWNhbEZvbnRzW2ldLmdldEdseXBoKGNoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZ2V0RGVmYXVsdEdseXBoKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB3aWR0aCBvZiB0aGUgY2hhciB3aXRoIHNwZWNpZmllZCBpbmRleC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gaW5kIHNwZWNpZmllZCBpbmRleCBvZiB0aGUgY2hhcmFjdGVyIAotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgY2hhcldpZHRoKGludCBpbmQpewotICAgICAgICByZXR1cm4gY2hhcldpZHRoKChjaGFyKWluZCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB3aWR0aCBvZiB0aGUgc3BlY2lmaWVkIGNoYXIuCi0gICAgICogCi0gICAgICogQHBhcmFtIGMgc3BlY2lmaWVkIGNoYXJhY3RlciAKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgaW50IGNoYXJXaWR0aChjaGFyIGMpewotICAgICAgICBHbHlwaCBnbCA9IHRoaXMuZ2V0R2x5cGgoYyk7Ci0gICAgICAgIHJldHVybiAoaW50KWdsLmdldEdseXBoUG9pbnRNZXRyaWNzKCkuZ2V0QWR2YW5jZVgoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGRlYnVnIGluZm9ybWF0aW9uIGFib3V0IHRoaXMgY2xhc3MuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpewotICAgIHJldHVybiBuZXcgU3RyaW5nKHRoaXMuZ2V0Q2xhc3MoKS5nZXROYW1lKCkgKwotICAgICAgICAgICAgIltuYW1lPSIgKyB0aGlzLm5hbWUgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAiLHN0eWxlPSIrIHRoaXMuc3R5bGUgKyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAiLGZwcz0iICsgdGhpcy5mb250UHJvcGVydGllcyArICJdIik7IC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgR2x5cGggb2JqZWN0IGNvcnJlc3BvbmRpbmcgdG8gdGhlIGRlZmF1bHQgZ2x5cGguCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEdseXBoIGdldERlZmF1bHRHbHlwaCgpewotICAgICAgICAvLyAhISBvbmx5IGZpcnN0IHBoeXNpY2FsIGZvbnQgdXNlZCB0byBnZXQgdGhpcyB2YWx1ZQotICAgICAgICByZXR1cm4gZlBoeXNpY2FsRm9udHNbMF0uZ2V0RGVmYXVsdEdseXBoKCk7Ci0gICAgfQotICAgIAotICAgIC8qKgotICAgICAqIFJldHVybnMgRm9udEV4dHJhTWV0cmljcyBvYmplY3Qgd2l0aCBleHRyYSBtZXRyaWNzCi0gICAgICogcmVsYXRlZCB0byB0aGlzIENvbXBvc2l0ZUZvbnQuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEZvbnRFeHRyYU1ldHJpY3MgZ2V0RXh0cmFNZXRyaWNzKCl7Ci0gICAgICAgIC8vIFJldHVybnMgRm9udEV4dHJhTWV0cmljcyBpbnN0YW5zZSBvZiB0aGUgZmlyc3QgcGh5c2ljYWwgCi0gICAgICAgIC8vIEZvbnQgZnJvbSB0aGUgYXJyYXkgb2YgZm9udHMuCi0gICAgICAgIHJldHVybiBmUGh5c2ljYWxGb250c1swXS5nZXRFeHRyYU1ldHJpY3MoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBEaXNwb3NlcyBDb21wb3NpdGVGb250IG9iamVjdCdzIHJlc291cmNlcy4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkaXNwb3NlKCkgewotICAgICAgICAvLyBOb3RoaW5nIHRvIGRpc3Bvc2UKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0ZvbnRFeHRyYU1ldHJpY3MuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvRm9udEV4dHJhTWV0cmljcy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwNDdiYTZkLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvRm9udEV4dHJhTWV0cmljcy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTQ1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqIAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udDsKLQotLyoqCi0gKiBFeHRyYSBmb250IG1ldHJpY3M6IHN1Yi9zdXBlcnNjcmlwdHMgc2l6ZXMsIG9mZnNldHMsIGF2ZXJhZ2UgY2hhciB3aWR0aC4KLSAqLwotcHVibGljIGNsYXNzIEZvbnRFeHRyYU1ldHJpY3MgewotICAgIAotICAgIC8qICEhIFN1YnNjcmlwdC9zdXBlcnNjcmlwdCBtZXRyaWNzIGFyZSB1bmRlZmluZWQgZm9yIFR5cGUxLiBBcyBhIHBvc3NpYmxlIAotICAgICAqIHNvbHV0aW9uIHdlIGNhbiB1c2UgdmFsdWVzIGZvciBUeXBlMSwgdGhhdCBhcmUgcHJvcG9ydGlvbmF0ZSB0byBUcnVlVHlwZQotICAgICAqIG9uZXM6Ci0gICAgICogIFN1YnNjcmlwdFNpemVYID09IDAuNyAqIGZvbnRTaXplCi0gICAgICogIFN1YnNjcmlwdFNpemVZID09IDAuNjUgKiBmb250U2l6ZQotICAgICAqICBTdWJzY3JpcHRPZmZzZXRYID09IDA7Ci0gICAgICogIFN1YnNjcmlwdE9mZnNldFkgPT0gMC4xNSAqIGZvbnRTaXplOwotICAgICAqICBTdXBlcnNjcmlwdFNpemVYID09IDAuNyAqIGZvbnRTaXplCi0gICAgICogIFN1cGVyc2NyaXB0U2l6ZVkgPT0gMC42NSAqIGZvbnRTaXplCi0gICAgICogIFN1cGVyc2NyaXB0T2Zmc2V0WCA9PSAwOwotICAgICAqICBTdXBlcnNjcmlwdE9mZnNldFkgPT0gMC40NSAqIGZvbnRTaXplCi0gICAgICogIAotICAgICAqLwotICAgIAotICAgIC8qCi0gICAgICogVGhlIGF2ZXJhZ2Ugd2lkdGggb2YgY2hhcmFjdGVycyBpbiB0aGUgZm9udC4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGxBdmVyYWdlQ2hhcldpZHRoOwotICAgIAotICAgIC8qCi0gICAgICogSG9yaXpvbnRhbCBzaXplIGZvciBzdWJzY3JpcHRzLgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgbFN1YnNjcmlwdFNpemVYOwotCi0gICAgLyoKLSAgICAgKiBWZXJ0aWNhbCBzaXplIGZvciBzdWJzY3JpcHRzLgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgbFN1YnNjcmlwdFNpemVZOyAKLSAgICAKLSAgICAvKgotICAgICAqIEhvcml6b250YWwgb2Zmc2V0IGZvciBzdWJzY3JpcHRzLCB0aGUgb2Zmc2V0IGZyb20gdGhlIGNoYXJhY3RlciBvcmlnaW4gCi0gICAgICogdG8gdGhlIG9yaWdpbiBvZiB0aGUgc3Vic2NyaXB0IGNoYXJhY3Rlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGxTdWJzY3JpcHRPZmZzZXRYOyAKLQotICAgIC8qCi0gICAgICogVmVydGljYWwgb2Zmc2V0IGZvciBzdWJzY3JpcHRzLCB0aGUgb2Zmc2V0IGZyb20gdGhlIGNoYXJhY3RlciBvcmlnaW4gCi0gICAgICogdG8gdGhlIG9yaWdpbiBvZiB0aGUgc3Vic2NyaXB0IGNoYXJhY3Rlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGxTdWJzY3JpcHRPZmZzZXRZOwotICAgIAotICAgIC8qCi0gICAgICogSG9yaXpvbnRhbCBzaXplIGZvciBzdXBlcnNjcmlwdHMuCi0gICAgICovCi0gICAgcHJpdmF0ZSBmbG9hdCBsU3VwZXJzY3JpcHRTaXplWDsgCi0KLSAgICAvKgotICAgICAqIFZlcnRpY2FsIHNpemUgZm9yIHN1cGVyc2NyaXB0cy4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGxTdXBlcnNjcmlwdFNpemVZOwotICAgIAotICAgIC8qCi0gICAgICogSG9yaXpvbnRhbCBvZmZzZXQgZm9yIHN1cGVyc2NyaXB0cywgdGhlIG9mZnNldCBmcm9tIHRoZSBjaGFyYWN0ZXIgCi0gICAgICogYmFzZSBsaW5lIHRvIHRoZSBiYXNlIGxpbmUgb2YgdGhlIHN1cGVyc2NyaXB0IGNoYXJhY3Rlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIGZsb2F0IGxTdXBlcnNjcmlwdE9mZnNldFg7Ci0KLSAgICAvKgotICAgICAqIFZlcnRpY2FsIG9mZnNldCBmb3Igc3VwZXJzY3JpcHRzLCB0aGUgb2Zmc2V0IGZyb20gdGhlIGNoYXJhY3RlciAKLSAgICAgKiBiYXNlIGxpbmUgdG8gdGhlIGJhc2UgbGluZSBvZiB0aGUgc3VwZXJzY3JpcHQgY2hhcmFjdGVyLgotICAgICAqLwotICAgIHByaXZhdGUgZmxvYXQgbFN1cGVyc2NyaXB0T2Zmc2V0WTsKLSAgICAKLSAgICBwdWJsaWMgRm9udEV4dHJhTWV0cmljcygpewotICAgICAgICAvLyBkZWZhdWx0IGNvbnN0cnVjdG9yCi0gICAgfQotCi0gICAgcHVibGljIEZvbnRFeHRyYU1ldHJpY3MoZmxvYXRbXSBtZXRyaWNzKXsKLSAgICAgICAgbEF2ZXJhZ2VDaGFyV2lkdGggPSBtZXRyaWNzWzBdOwotICAgICAgICBsU3Vic2NyaXB0U2l6ZVggPSBtZXRyaWNzWzFdOwotICAgICAgICBsU3Vic2NyaXB0U2l6ZVkgPSBtZXRyaWNzWzJdOwotICAgICAgICBsU3Vic2NyaXB0T2Zmc2V0WCA9IG1ldHJpY3NbM107Ci0gICAgICAgIGxTdWJzY3JpcHRPZmZzZXRZID0gbWV0cmljc1s0XTsKLSAgICAgICAgbFN1cGVyc2NyaXB0U2l6ZVggPSBtZXRyaWNzWzVdOwotICAgICAgICBsU3VwZXJzY3JpcHRTaXplWSA9IG1ldHJpY3NbNl07Ci0gICAgICAgIGxTdXBlcnNjcmlwdE9mZnNldFggPSBtZXRyaWNzWzddOwotICAgICAgICBsU3VwZXJzY3JpcHRPZmZzZXRZID0gbWV0cmljc1s4XTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmxvYXQgZ2V0QXZlcmFnZUNoYXJXaWR0aCgpewotICAgICAgICByZXR1cm4gbEF2ZXJhZ2VDaGFyV2lkdGg7Ci0gICAgfQotICAgIAotICAgIHB1YmxpYyBmbG9hdCBnZXRTdWJzY3JpcHRTaXplWCgpewotICAgICAgICByZXR1cm4gbFN1YnNjcmlwdFNpemVYOwotICAgIH0KLQotICAgIHB1YmxpYyBmbG9hdCBnZXRTdWJzY3JpcHRTaXplWSgpewotICAgICAgICByZXR1cm4gbFN1YnNjcmlwdFNpemVZOwotICAgIH0KLQotICAgIHB1YmxpYyBmbG9hdCBnZXRTdWJzY3JpcHRPZmZzZXRYKCl7Ci0gICAgICAgIHJldHVybiBsU3Vic2NyaXB0T2Zmc2V0WDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmxvYXQgZ2V0U3Vic2NyaXB0T2Zmc2V0WSgpewotICAgICAgICByZXR1cm4gbFN1YnNjcmlwdE9mZnNldFk7Ci0gICAgfQotCi0gICAgcHVibGljIGZsb2F0IGdldFN1cGVyc2NyaXB0U2l6ZVgoKXsKLSAgICAgICAgcmV0dXJuIGxTdXBlcnNjcmlwdFNpemVYOwotICAgIH0KLQotICAgIHB1YmxpYyBmbG9hdCBnZXRTdXBlcnNjcmlwdFNpemVZKCl7Ci0gICAgICAgIHJldHVybiBsU3VwZXJzY3JpcHRTaXplWTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgZmxvYXQgZ2V0U3VwZXJzY3JpcHRPZmZzZXRYKCl7Ci0gICAgICAgIHJldHVybiBsU3VwZXJzY3JpcHRPZmZzZXRYOwotICAgIH0KLQotICAgIHB1YmxpYyBmbG9hdCBnZXRTdXBlcnNjcmlwdE9mZnNldFkoKXsKLSAgICAgICAgcmV0dXJuIGxTdXBlcnNjcmlwdE9mZnNldFk7Ci0gICAgfQotICAgIAotICAgIAotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Gb250RmluZGVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0ZvbnRGaW5kZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMDliY2Y1Yy4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0ZvbnRGaW5kZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEyMSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqCi0gKiBAZGF0ZTogSnVsIDEyLCAyMDA1Ci0gKi8KLQotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5Gb250OwotaW1wb3J0IGphdmEuYXd0LkdyYXBoaWNzRW52aXJvbm1lbnQ7Ci1pbXBvcnQgamF2YS51dGlsLkxpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLk1hcDsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGNob29zZXMgdGhlIGRlZmF1bHQgZm9udCBmb3IgdGhlIGdpdmVuIHRleHQuCi0gKiBJZiBpdCBmaW5kcyB0aGUgY2hhcmFjdGVyIHdoaWNoIGN1cnJlbnQgZm9udCBpcyB1bmFibGUgdG8gZGlzcGxheQotICogaXQgc3RhcnRzIHRoZSBuZXh0IGZvbnQgcnVuIGFuZCBsb29rcyBmb3IgdGhlIGZvbnQgd2hpY2ggaXMgYWJsZSB0bwotICogZGlzcGxheSB0aGUgY3VycmVudCBjaGFyYWN0ZXIuIEl0IGFsc28gY2FjaGVzIHRoZSBmb250IG1hcHBpbmdzCi0gKiAoaW5kZXggaW4gdGhlIGFycmF5IGNvbnRhaW5pbmcgYWxsIGZvbnRzKSBmb3IgdGhlIGNoYXJhY3RlcnMsCi0gKiB1c2luZyB0aGF0IGZhY3QgdGhhdCBzY3JpcHRzIGFyZSBtYWlubHkgY29udGlndW91cyBpbiB0aGUgVVRGLTE2IGVuY29kaW5nCi0gKiBhbmQgdGhlcmUncyBhIGhpZ2ggcHJvYmFiaWxpdHkgdGhhdCB0aGUgdXBwZXIgYnl0ZSB3aWxsIGJlIHRoZSBzYW1lIGZvciB0aGUKLSAqIG5leHQgY2hhcmFjdGVyIGFzIGZvciB0aGUgcHJldmlvdXMuIFRoaXMgYWxsb3dzIHRvIHNhdmUgdGhlIHNwYWNlIHVzZWQgZm9yIHRoZSBjYWNoZS4KLSAqLwotcHVibGljIGNsYXNzIEZvbnRGaW5kZXIgewotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGZsb2F0IERFRkFVTFRfRk9OVF9TSVpFID0gMTI7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBGb250IGZvbnRzW10gPQotICAgICAgICAgICAgR3JhcGhpY3NFbnZpcm9ubWVudC5nZXRMb2NhbEdyYXBoaWNzRW52aXJvbm1lbnQoKS5nZXRBbGxGb250cygpOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IE5VTV9CTE9DS1MgPSAyNTY7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJMT0NLX1NJWkUgPSAyNTY7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IElOREVYX01BU0sgPSAweEZGOwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBCTE9DS19TSElGVCA9IDg7Ci0KLSAgICAvLyBNYXBzIGNoYXJhY3RlcnMgaW50byB0aGUgZm9udHMgYXJyYXkKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgYmxvY2tzW11bXSA9IG5ldyBpbnRbTlVNX0JMT0NLU11bXTsKLQotICAgIC8qKgotICAgICAqIEZpbmRzIHRoZSBmb250IHdoaWNoIGlzIGFibGUgdG8gZGlzcGxheSB0aGUgZ2l2ZW4gY2hhcmFjdGVyCi0gICAgICogYW5kIHNhdmVzIHRoZSBmb250IG1hcHBpbmcgZm9yIHRoaXMgY2hhcmFjdGVyCi0gICAgICogQHBhcmFtIGMgLSBjaGFyYWN0ZXIKLSAgICAgKiBAcmV0dXJuIGZvbnQKLSAgICAgKi8KLSAgICBzdGF0aWMgRm9udCBmaW5kRm9udEZvckNoYXIoY2hhciBjKSB7Ci0gICAgICAgIGludCBibG9ja051bSA9IGMgPj4gQkxPQ0tfU0hJRlQ7Ci0gICAgICAgIGludCBpbmRleCA9IGMgJiBJTkRFWF9NQVNLOwotCi0gICAgICAgIGlmIChibG9ja3NbYmxvY2tOdW1dID09IG51bGwpIHsKLSAgICAgICAgICAgIGJsb2Nrc1tibG9ja051bV0gPSBuZXcgaW50W0JMT0NLX1NJWkVdOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGJsb2Nrc1tibG9ja051bV1baW5kZXhdID09IDApIHsKLSAgICAgICAgICAgIGJsb2Nrc1tibG9ja051bV1baW5kZXhdID0gMTsKLQotICAgICAgICAgICAgZm9yIChpbnQgaT0wOyBpPGZvbnRzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGZvbnRzW2ldLmNhbkRpc3BsYXkoYykpIHsKLSAgICAgICAgICAgICAgICAgICAgYmxvY2tzW2Jsb2NrTnVtXVtpbmRleF0gPSBpKzE7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBnZXREZWZhdWx0U2l6ZUZvbnQoYmxvY2tzW2Jsb2NrTnVtXVtpbmRleF0tMSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRGVyaXZlcyB0aGUgZGVmYXVsdCBzaXplIGZvbnQKLSAgICAgKiBAcGFyYW0gaSAtIGluZGV4IGluIHRoZSBhcnJheSBvZiBhbGwgZm9udHMKLSAgICAgKiBAcmV0dXJuIGRlcml2ZWQgZm9udAotICAgICAqLwotICAgIHN0YXRpYyBGb250IGdldERlZmF1bHRTaXplRm9udChpbnQgaSkgewotICAgICAgICBpZiAoZm9udHNbaV0uZ2V0U2l6ZSgpICE9IERFRkFVTFRfRk9OVF9TSVpFKSB7Ci0gICAgICAgICAgICBmb250c1tpXSA9IGZvbnRzW2ldLmRlcml2ZUZvbnQoREVGQVVMVF9GT05UX1NJWkUpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGZvbnRzW2ldOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFzc2lnbnMgZGVmYXVsdCBmb250cyBmb3IgdGhlIGdpdmVuIHRleHQgcnVuLgotICAgICAqIEZpcnN0IHRocmVlIHBhcmFtZXRlcnMgYXJlIGlucHV0LCBsYXN0IHRocmVlIGFyZSBvdXRwdXQuCi0gICAgICogQHBhcmFtIHRleHQgLSBnaXZlbiB0ZXh0Ci0gICAgICogQHBhcmFtIHJ1blN0YXJ0IC0gc3RhcnQgb2YgdGhlIHRleHQgcnVuCi0gICAgICogQHBhcmFtIHJ1bkxpbWl0IC0gZW5kIG9mIHRoZSB0ZXh0IHJ1bgotICAgICAqIEBwYXJhbSBydW5TdGFydHMgLSBzdGFydHMgb2YgdGhlIHJlc3VsdGluZyBmb250IHJ1bnMKLSAgICAgKiBAcGFyYW0gZm9udHMgLSBtYXBwaW5nIG9mIHRoZSBmb250IHJ1biBzdGFydHMgdG8gdGhlIGZvbnRzCi0gICAgICovCi0gICAgc3RhdGljIHZvaWQgZmluZEZvbnRzKGNoYXIgdGV4dFtdLCBpbnQgcnVuU3RhcnQsIGludCBydW5MaW1pdCwgTGlzdDxJbnRlZ2VyPiBydW5TdGFydHMsCi0gICAgICAgICAgICBNYXA8SW50ZWdlciwgRm9udD4gZm9udHMpIHsKLSAgICAgICAgRm9udCBwcmV2Rm9udCA9IG51bGw7Ci0gICAgICAgIEZvbnQgY3VyckZvbnQ7Ci0gICAgICAgIGZvciAoaW50IGkgPSBydW5TdGFydDsgaSA8IHJ1bkxpbWl0OyBpKyspIHsKLSAgICAgICAgICAgIGN1cnJGb250ID0gZmluZEZvbnRGb3JDaGFyKHRleHRbaV0pOwotICAgICAgICAgICAgaWYgKGN1cnJGb250ICE9IHByZXZGb250KSB7Ci0gICAgICAgICAgICAgICAgcHJldkZvbnQgPSBjdXJyRm9udDsKLSAgICAgICAgICAgICAgICBJbnRlZ2VyIGlkeCA9IG5ldyBJbnRlZ2VyKGkpOwotICAgICAgICAgICAgICAgIGZvbnRzLnB1dChpZHgsIGN1cnJGb250KTsKLSAgICAgICAgICAgICAgICBpZiAoaSAhPSBydW5TdGFydCkgewotICAgICAgICAgICAgICAgICAgICBydW5TdGFydHMuYWRkKGlkeCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Gb250TWFuYWdlci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Gb250TWFuYWdlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4MzU0ZTI1Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvRm9udE1hbmFnZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDgxOSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuRm9udDsKLWltcG9ydCBqYXZhLmF3dC5wZWVyLkZvbnRQZWVyOwotaW1wb3J0IGphdmEuaW8uRmlsZTsKLWltcG9ydCBqYXZhLmlvLkZpbGVJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEubGFuZy5yZWYuUmVmZXJlbmNlUXVldWU7Ci1pbXBvcnQgamF2YS5sYW5nLnJlZi5Tb2Z0UmVmZXJlbmNlOwotaW1wb3J0IGphdmEudXRpbC5FbnVtZXJhdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaHRhYmxlOwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci1pbXBvcnQgamF2YS51dGlsLlByb3BlcnRpZXM7Ci1pbXBvcnQgamF2YS51dGlsLlZlY3RvcjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuQ29tbW9uR3JhcGhpY3MyREZhY3Rvcnk7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Ob3RJbXBsZW1lbnRlZEV4Y2VwdGlvbjsKLQotCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgRm9udE1hbmFnZXIgewotICAgIAotICAgIC8vPz8/QVdUCi0gICAgYm9vbGVhbiBOT1RfSU1QID0gZmFsc2U7Ci0gICAgCi0gICAgLyoqCi0gICAgICogYXJyYXkgb2YgZm9udCBmYW1pbGllcyBuYW1lcwotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmdbXSBhbGxGYW1pbGllczsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIERFRkFVTFRfTkFNRSA9ICJEZWZhdWx0IjsgLyogRGVmYXVsdCBmb250IG5hbWUgKi8gLy8kTk9OLU5MUy0xJAotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIERJQUxPR19OQU1FID0gIkRpYWxvZyI7ICAvKiBEaWFsb2cgZm9udCBuYW1lICovIC8vJE5PTi1OTFMtMSQKLQotICAgIC8qKgotICAgICAqIFNldCBvZiBjb25zdGFudHMgYXBwbGljYWJsZSB0byB0aGUgVHJ1ZVR5cGUgJ25hbWUnIHRhYmxlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgYnl0ZSAgRkFNSUxZX05BTUVfSUQgID0gMTsgICAgICAvKiBGYW1pbHkgbmFtZSBpZGVudGlmaWVyICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGJ5dGUgIEZPTlRfTkFNRV9JRCAgPSA0OyAgICAgICAgLyogRnVsbCBmb250IG5hbWUgaWRlbnRpZmllciAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgYnl0ZSAgUE9TVFNDUklQVF9OQU1FX0lEID0gNjsgICAvKiBQb3N0U2NyaXB0IG5hbWUgaWRlbnRpZmllciAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBzaG9ydCBFTkdMSVNIX0xBTkdJRCA9IDB4MDQwOTsgIC8qIEVuZ2xpc2ggKFVuaXRlZCBTdGF0ZXMpbGFuZ3VhZ2UgaWRlbnRpZmllciAgICovCi0KLSAgICAvKioKLSAgICAgKiBTZXQgb2YgY29uc3RhbnRzIGRlc2NyaWJpbmcgZm9udCB0eXBlLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgYnl0ZSAgRk9OVF9UWVBFX1RUICA9IDQ7ICAgICAgICAvKiBUcnVlVHlwZSB0eXBlIChUUlVFVFlQRV9GT05UVFlQRSkgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGJ5dGUgIEZPTlRfVFlQRV9UMSAgPSAyOyAgICAgICAgLyogVHlwZTEgdHlwZSAgICAoREVWSUNFX0ZPTlRUWVBFKSAgICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBieXRlICBGT05UX1RZUEVfVU5ERUYgID0gMDsgICAgIC8qIFVuZGVmaW5lZCB0eXBlICAgICAgICAgICAgICAgICAgICAgICAqLwotCi0gICAgLy8gbG9naWNhbCBmYW1pbHkgdHlwZXMgKGluZGljZXMgaW4gRm9udE1hbmFnZXIuTE9HSUNBTF9GT05UX05BTUVTKQotICAgIHN0YXRpYyBmaW5hbCBpbnQgRElBTE9HID0gMzsgICAgICAgIC8vIEZGX1NXSVNTCi0gICAgc3RhdGljIGZpbmFsIGludCBTQU5TU0VSSUYgPSAxOyAgICAgLy8gRkZfU1dJU1MKLSAgICBzdGF0aWMgZmluYWwgaW50IERJQUxPR0lOUFVUID0gNDsgICAvLyBGRl9NT0RFUk4KLSAgICBzdGF0aWMgZmluYWwgaW50IE1PTk9TUEFDRUQgPSAyOyAgICAvLyBGRl9NT0RFUk4KLSAgICBzdGF0aWMgZmluYWwgaW50IFNFUklGID0gMDsgICAgICAgICAvLyBGRl9ST01BTgotCi0KLSAgICAvKioKLSAgICAgKiBGb250UHJvcGVydHkgcmVsYXRlZCBjb25zdGFudHMuIAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIFBMQVRGT1JNX0ZPTlRfTkFNRSA9ICJQbGF0Zm9ybUZvbnROYW1lIjsgLy8kTk9OLU5MUy0xJAotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIExPR0lDQUxfRk9OVF9OQU1FID0gIkxvZ2ljYWxGb250TmFtZSI7IC8vJE5PTi1OTFMtMSQKLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIFN0cmluZyBDT01QT05FTlRfSU5ERVggPSAiQ29tcG9uZW50SW5kZXgiOyAvLyROT04tTkxTLTEkCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBTdHJpbmcgU1RZTEVfSU5ERVggPSAiU3R5bGVJbmRleCI7IC8vJE5PTi1OTFMtMSQKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nW10gRk9OVF9NQVBQSU5HX0tFWVMgPSB7Ci0gICAgICAgICAgICAiTG9naWNhbEZvbnROYW1lLlN0eWxlTmFtZS5Db21wb25lbnRJbmRleCIsICJMb2dpY2FsRm9udE5hbWUuQ29tcG9uZW50SW5kZXgiIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJAotICAgIH07Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIFN0cmluZyBGT05UX0NIQVJBQ1RFUl9FTkNPRElORyA9ICJmb250Y2hhcnNldC5Mb2dpY2FsRm9udE5hbWUuQ29tcG9uZW50SW5kZXgiOyAvLyROT04tTkxTLTEkCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIFN0cmluZyBFWENMVVNJT05fUkFOR0VTID0gImV4Y2x1c2lvbi5Mb2dpY2FsRm9udE5hbWUuQ29tcG9uZW50SW5kZXgiOyAvLyROT04tTkxTLTEkCi0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIFN0cmluZyBGT05UX0ZJTEVfTkFNRSA9ICJmaWxlbmFtZS5QbGF0Zm9ybUZvbnROYW1lIjsgLy8kTk9OLU5MUy0xJAotCi0gICAgLyoqCi0gICAgICogQXZhaWxhYmxlIGxvZ2ljYWwgZm9udCBmYW1pbGllcyBuYW1lcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIFN0cmluZ1tdIExPR0lDQUxfRk9OVF9GQU1JTElFUyA9IHsKLSAgICAgICAgICAgICJTZXJpZiIsICJTYW5zU2VyaWYiLCAiTW9ub3NwYWNlZCIsICJEaWFsb2ciLCAiRGlhbG9nSW5wdXQiIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBBdmFpbGFibGUgbG9naWNhbCBmb250IG5hbWVzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nW10gTE9HSUNBTF9GT05UX05BTUVTID0gewotICAgICAgICAgICAgInNlcmlmIiwgInNlcmlmLnBsYWluIiwgInNlcmlmLmJvbGQiLCAic2VyaWYuaXRhbGljIiwgInNlcmlmLmJvbGRpdGFsaWMiLCAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJCAvLyROT04tTkxTLTQkIC8vJE5PTi1OTFMtNSQKLSAgICAgICAgICAgICJzYW5zc2VyaWYiLCAic2Fuc3NlcmlmLnBsYWluIiwgInNhbnNzZXJpZi5ib2xkIiwgInNhbnNzZXJpZi5pdGFsaWMiLCAic2Fuc3NlcmlmLmJvbGRpdGFsaWMiLCAvLyROT04tTkxTLTEkIC8vJE5PTi1OTFMtMiQgLy8kTk9OLU5MUy0zJCAvLyROT04tTkxTLTQkIC8vJE5PTi1OTFMtNSQKLSAgICAgICAgICAgICJtb25vc3BhY2VkIiwgIm1vbm9zcGFjZWQucGxhaW4iLCAibW9ub3NwYWNlZC5ib2xkIiwgIm1vbm9zcGFjZWQuaXRhbGljIiwgIm1vbm9zcGFjZWQuYm9sZGl0YWxpYyIsIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgICAgICAgICAgImRpYWxvZyIsICJkaWFsb2cucGxhaW4iLCAiZGlhbG9nLmJvbGQiLCAiZGlhbG9nLml0YWxpYyIsICJkaWFsb2cuYm9sZGl0YWxpYyIsIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgICAgICAgICAgImRpYWxvZ2lucHV0IiwgImRpYWxvZ2lucHV0LnBsYWluIiwgImRpYWxvZ2lucHV0LmJvbGQiLCAiZGlhbG9naW5wdXQuaXRhbGljIiwgImRpYWxvZ2lucHV0LmJvbGRpdGFsaWMiIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBBdmFpbGFibGUgbG9naWNhbCBmb250IGZhY2UgbmFtZXMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBTdHJpbmdbXSBMT0dJQ0FMX0ZPTlRfRkFDRVMgPSB7Ci0gICAgICAgICAgICAiU2VyaWYiLCAiU2VyaWYucGxhaW4iLCAiU2VyaWYuYm9sZCIsICJTZXJpZi5pdGFsaWMiLCAiU2VyaWYuYm9sZGl0YWxpYyIsIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgICAgICAgICAgIlNhbnNzZXJpZiIsICJTYW5zc2VyaWYucGxhaW4iLCAiU2Fuc3NlcmlmLmJvbGQiLCAiU2Fuc3NlcmlmLml0YWxpYyIsICJTYW5zc2VyaWYuYm9sZGl0YWxpYyIsIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgICAgICAgICAgIk1vbm9zcGFjZWQiLCAiTW9ub3NwYWNlZC5wbGFpbiIsICJNb25vc3BhY2VkLmJvbGQiLCAiTW9ub3NwYWNlZC5pdGFsaWMiLCAiTW9ub3NwYWNlZC5ib2xkaXRhbGljIiwgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQgLy8kTk9OLU5MUy00JCAvLyROT04tTkxTLTUkCi0gICAgICAgICAgICAiRGlhbG9nIiwgIkRpYWxvZy5wbGFpbiIsICJEaWFsb2cuYm9sZCIsICJEaWFsb2cuaXRhbGljIiwgIkRpYWxvZy5ib2xkaXRhbGljIiwgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQgLy8kTk9OLU5MUy00JCAvLyROT04tTkxTLTUkCi0gICAgICAgICAgICAiRGlhbG9naW5wdXQiLCAiRGlhbG9naW5wdXQucGxhaW4iLCAiRGlhbG9naW5wdXQuYm9sZCIsICJEaWFsb2dpbnB1dC5pdGFsaWMiLCAiRGlhbG9naW5wdXQuYm9sZGl0YWxpYyIgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQgLy8kTk9OLU5MUy00JCAvLyROT04tTkxTLTUkCi0gICAgfTsKLQotICAgIC8qKgotICAgICAqIFNldCBvZiBmb250IHN0eWxlIG5hbWVzLgotICAgICAqIEZvbnQuZ2V0U3R5bGUoKSBjb3JyZXNwb25kcyB0byBpbmRleGVzIGluIFNUWUxFX05BTUVTIGFycmF5LgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nW10gU1RZTEVfTkFNRVMgPSB7Ci0gICAgICAgICAgICAicGxhaW4iLCAiYm9sZCIsICJpdGFsaWMiLCAiYm9sZGl0YWxpYyIgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQgLy8kTk9OLU5MUy00JAotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBMb2dpY2FsIGZvbnQgc3R5bGVzIG5hbWVzIHRhYmxlIHdoZXJlIGZvbnQgc3R5bGVzIG5hbWVzIHVzZWQgCi0gICAgICogYXMgdGhlIGtleSBhbmQgdGhlIHZhbHVlIGlzIHRoZSBpbmRleCBvZiB0aGlzIHN0eWxlIG5hbWUuCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgSGFzaHRhYmxlPFN0cmluZywgSW50ZWdlcj4gc3R5bGVfa2V5cyA9IG5ldyBIYXNodGFibGU8U3RyaW5nLCBJbnRlZ2VyPig0KTsKLQotICAgIC8qKgotICAgICAqIEluaXRpYWxpemUgZm9udCBzdHlsZXMga2V5cyB0YWJsZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgewotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IFNUWUxFX05BTUVTLmxlbmd0aDsgaSsrKXsKLSAgICAgICAgICAgIHN0eWxlX2tleXMucHV0KFNUWUxFX05BTUVTW2ldLCBJbnRlZ2VyLnZhbHVlT2YoaSkpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJuIGZvbnQgc3R5bGUgZnJvbSB0aGUgbG9naWNhbCBzdHlsZSBuYW1lLgotICAgICAqIAotICAgICAqIEBwYXJhbSBsTmFtZSBzdHlsZSBuYW1lIG9mIHRoZSBsb2dpY2FsIGZhY2UKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGludCBnZXRMb2dpY2FsU3R5bGUoU3RyaW5nIGxOYW1lKXsKLSAgICAgICAgSW50ZWdlciB2YWx1ZSA9IHN0eWxlX2tleXMuZ2V0KGxOYW1lKTsKLSAgICAgICAgcmV0dXJuIHZhbHVlICE9IG51bGwgPyB2YWx1ZS5pbnRWYWx1ZSgpOiAtMTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXQgb2YgcG9zc2libGUgIm9zIiBwcm9wZXJ0eSB2YWx1ZXMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBTdHJpbmdbXSBPU19WQUxVRVMgPSB7Ci0gICAgICAgICAgICAiTlQiLCAiOTgiLCAiMjAwMCIsICJNZSIsICJYUCIsIC8vIEZvciBXaW5kb3dzIC8vJE5PTi1OTFMtMSQgLy8kTk9OLU5MUy0yJCAvLyROT04tTkxTLTMkIC8vJE5PTi1OTFMtNCQgLy8kTk9OLU5MUy01JAotICAgICAgICAgICAgIlJlZGhhdCIsICJUdXJibyIsICJTdVNFIiAgICAgICAvLyBGb3IgTGludXggLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkIC8vJE5PTi1OTFMtMyQKLSAgICB9OwotCi0gICAgLyoqCi0gICAgICogU2V0IG9mIHBvc3NpYmxlIGZvbnQucHJvcGVydHkgZmlsZSBuYW1lcy4KLSAgICAgKiBMYW5ndWFnZSwgQ291bnRyeSwgRW5jb2RpbmcsIE9TLCBWZXJzaW9uIHNob3VsZCBiZSByZXBsYWNlZCB3aXRoCi0gICAgICogdGhlIHZhbHVlcyBmcm9tIGN1cnJlbnQgY29uZmlndXJhdGlvbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIFN0cmluZ1tdIEZQX0ZJTEVfTkFNRVMgPSB7Ci0gICAgICAgICAgICAiL2xpYi9mb250LnByb3BlcnRpZXMuTGFuZ3VhZ2VfQ291bnRyeV9FbmNvZGluZy5PU1ZlcnNpb24iLCAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAiL2xpYi9mb250LnByb3BlcnRpZXMuTGFuZ3VhZ2VfQ291bnRyeV9FbmNvZGluZy5PUyIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5MYW5ndWFnZV9Db3VudHJ5X0VuY29kaW5nLlZlcnNpb24iLCAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAiL2xpYi9mb250LnByb3BlcnRpZXMuTGFuZ3VhZ2VfQ291bnRyeV9FbmNvZGluZyIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5MYW5ndWFnZV9Db3VudHJ5Lk9TVmVyc2lvbiIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5MYW5ndWFnZV9Db3VudHJ5Lk9TIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgIi9saWIvZm9udC5wcm9wZXJ0aWVzLkxhbmd1YWdlX0NvdW50cnkuVmVyc2lvbiIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5MYW5ndWFnZV9Db3VudHJ5IiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgIi9saWIvZm9udC5wcm9wZXJ0aWVzLkxhbmd1YWdlX0VuY29kaW5nLk9TVmVyc2lvbiIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5MYW5ndWFnZV9FbmNvZGluZy5PUyIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5MYW5ndWFnZV9FbmNvZGluZy5WZXJzaW9uIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgIi9saWIvZm9udC5wcm9wZXJ0aWVzLkxhbmd1YWdlX0VuY29kaW5nIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgIi9saWIvZm9udC5wcm9wZXJ0aWVzLkxhbmd1YWdlLk9TVmVyc2lvbiIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5MYW5ndWFnZS5PUyIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5MYW5ndWFnZS5WZXJzaW9uIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgIi9saWIvZm9udC5wcm9wZXJ0aWVzLkxhbmd1YWdlIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgIi9saWIvZm9udC5wcm9wZXJ0aWVzLkVuY29kaW5nLk9TVmVyc2lvbiIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5FbmNvZGluZy5PUyIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5FbmNvZGluZy5WZXJzaW9uIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgIi9saWIvZm9udC5wcm9wZXJ0aWVzLkVuY29kaW5nIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgIi9saWIvZm9udC5wcm9wZXJ0aWVzLk9TVmVyc2lvbiIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5PUyIsIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICIvbGliL2ZvbnQucHJvcGVydGllcy5WZXJzaW9uIiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgIi9saWIvZm9udC5wcm9wZXJ0aWVzIiAvLyROT04tTkxTLTEkCi0gICAgfTsKLQotICAgIC8qKgotICAgICAqIFRhYmxlIHdpdGggYWxsIGF2YWlsYWJsZSBmb250IHByb3BlcnRpZXMgY29ycmVzcG9uZGluZwotICAgICAqIHRvIHRoZSBjdXJyZW50IHN5c3RlbSBjb25maWd1cmF0aW9uLgotICAgICAqLwotICAgIHB1YmxpYyBIYXNodGFibGU8U3RyaW5nLCBWZWN0b3I8Rm9udFByb3BlcnR5Pj4gZlByb3BlcnRpZXMgPSBuZXcgSGFzaHRhYmxlPFN0cmluZywgVmVjdG9yPEZvbnRQcm9wZXJ0eT4+KCk7Ci0gICAgCi0gICAgcHVibGljIEZvbnRNYW5hZ2VyKCl7Ci0gICAgICAgIGFsbEZhbWlsaWVzID0gZ2V0QWxsRmFtaWxpZXMoKTsKLSAgICAgICAgLyoKLSAgICAgICAgICogQ3JlYXRpbmcgYW5kIHJlZ2lzdGVyaW5nIHNodXRkb3duIGhvb2sgdG8gZnJlZSByZXNvdXJjZXMKLSAgICAgICAgICogYmVmb3JlIG9iamVjdCBpcyBkZXN0cm95ZWQuCi0gICAgICAgICAqLwotICAgICAgICAvLz8/P0FXVAotICAgICAgICAvL0Rpc3Bvc2VOYXRpdmVIb29rIHNodXRkb3duSG9vayA9IG5ldyBEaXNwb3NlTmF0aXZlSG9vaygpOwotICAgICAgICAvL1J1bnRpbWUuZ2V0UnVudGltZSgpLmFkZFNodXRkb3duSG9vayhzaHV0ZG93bkhvb2spOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIE1heGltdW0gbnVtYmVyIG9mIHVucmVmZXJlbmNlZCBmb250IHBlZXJzIHRvIGtlZXAuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRU1QVFlfRk9OVFNfQ0FQQUNJVFkgPSAxMDsKLQotICAgIC8qKgotICAgICAqIExvY2FsZSAtIExhbmd1YWdlIElEIGhhc2ggdGFibGUuCi0gICAgICovCi0gICAgSGFzaHRhYmxlPFN0cmluZywgU2hvcnQ+IHRhYmxlTENJRCA9IG5ldyBIYXNodGFibGU8U3RyaW5nLCBTaG9ydD4oKTsKLQotICAgIC8qKgotICAgICAqIEhhc2ggdGFibGUgdGhhdCBjb250YWlucyBGb250UGVlcnMgaW5zdGFuY2VzLgotICAgICAqLwotICAgIHB1YmxpYyBIYXNodGFibGU8U3RyaW5nLCBIYXNoTWFwUmVmZXJlbmNlPiBmb250c1RhYmxlID0gbmV3IEhhc2h0YWJsZTxTdHJpbmcsIEhhc2hNYXBSZWZlcmVuY2U+KCk7Ci0gICAgCi0gICAgLyoqCi0gICAgICogUmVmZXJlbmNlUXVldWUgZm9yIEhhc2hNYXBSZWZlcmVuY2Ugb2JqZWN0cyB0byBjaGVjawotICAgICAqIGlmIHRoZXkgd2VyZSBjb2xsZWN0ZWQgYnkgZ2FyYmFnZSBjb2xsZWN0b3IuIAotICAgICAqLwotICAgIHB1YmxpYyBSZWZlcmVuY2VRdWV1ZTxGb250UGVlcj4gcXVldWUgPSBuZXcgUmVmZXJlbmNlUXVldWU8Rm9udFBlZXI+KCk7Ci0KLSAgICAvKioKLSAgICAgKiBTaW5nbGV0b24gaW5zdGFuY2UKLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgc3RhdGljIEZvbnRNYW5hZ2VyIGluc3QgPSBDb21tb25HcmFwaGljczJERmFjdG9yeS5pbnN0LmdldEZvbnRNYW5hZ2VyKCk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIHNpbmdsZXRvbiBpbnN0YW5jZSBvZiBGb250TWFuYWdlcgotICAgICAqIAotICAgICAqIEByZXR1cm4gaW5zdGFuY2Ugb2YgRm9udE1hbmFnZXIgaW1wbGVtZW50YXRpb24KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIEZvbnRNYW5hZ2VyIGdldEluc3RhbmNlKCkgewotICAgICAgICByZXR1cm4gaW5zdDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHBsYXRmb3JtLWRlcGVuZGVudCBGb250IHBlZXIgY3JlYXRlZCBmcm9tIHRoZSBzcGVjaWZpZWQgCi0gICAgICogRm9udCBvYmplY3QgZnJvbSB0aGUgdGFibGUgd2l0aCBjYWNoZWQgRm9udFBlZXJzIGluc3RhbmNlcy4KLSAgICAgKiAKLSAgICAgKiBOb3RlLCB0aGlzIG1ldGhvZCBjaGVja3Mgd2hldGhlciBGb250UGVlciB3aXRoIHNwZWNpZmllZCBwYXJhbWV0ZXJzIAotICAgICAqIGV4aXN0cyBpbiB0aGUgdGFibGUgd2l0aCBjYWNoZWQgRm9udFBlZXJzJyBpbnN0YW5jZXMuIElmIHRoZXJlIGlzIG5vIG5lZWRlZCAKLSAgICAgKiBpbnN0YW5jZSAtIGl0IGlzIGNyZWF0ZWQgYW5kIGNhY2hlZC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZm9udE5hbWUgbmFtZSBvZiB0aGUgZm9udCAKLSAgICAgKiBAcGFyYW0gX2ZvbnRTdHlsZSBzdHlsZSBvZiB0aGUgZm9udCAKLSAgICAgKiBAcGFyYW0gc2l6ZSBmb250IHNpemUKLSAgICAgKiAKLSAgICAgKiBAcmV0dXJuIHBsYXRmb3JtIGRlcGVuZGVudCBGb250UGVlciBpbXBsZW1lbnRhdGlvbiBjcmVhdGVkIGZyb20gCi0gICAgICogdGhlIHNwZWNpZmllZCBwYXJhbWV0ZXJzCi0gICAgICovCi0gICAgcHVibGljIEZvbnRQZWVyIGdldEZvbnRQZWVyKFN0cmluZyBmb250TmFtZSwgaW50IF9mb250U3R5bGUsIGludCBzaXplKSB7Ci0gICAgICAgIHVwZGF0ZUZvbnRzVGFibGUoKTsKLSAgICAgICAgCi0gICAgICAgIEZvbnRQZWVyIHBlZXIgPSBudWxsOwotICAgICAgICBTdHJpbmcga2V5OyAKLSAgICAgICAgU3RyaW5nIG5hbWU7Ci0gICAgICAgIGludCBmb250U3R5bGUgPSBfZm9udFN0eWxlOwotICAgICAgICAKLSAgICAgICAgaW50IGxvZ2ljYWxJbmRleCA9IGdldExvZ2ljYWxGYWNlSW5kZXgoZm9udE5hbWUpOwotICAgICAgICAKLSAgICAgICAgaWYgKGxvZ2ljYWxJbmRleCAhPSAtMSl7Ci0gICAgICAgICAgICBuYW1lID0gZ2V0TG9naWNhbEZhY2VGcm9tRm9udChmb250U3R5bGUsIGxvZ2ljYWxJbmRleCk7Ci0gICAgICAgICAgICBmb250U3R5bGUgPSBnZXRTdHlsZUZyb21Mb2dpY2FsRmFjZShuYW1lKTsKLSAgICAgICAgICAgIGtleSA9IG5hbWUuY29uY2F0KFN0cmluZy52YWx1ZU9mKHNpemUpKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG5hbWUgPSBmb250TmFtZTsKLSAgICAgICAgICAgIGtleSA9IG5hbWUuY29uY2F0KFN0cmluZy52YWx1ZU9mKGZvbnRTdHlsZSkpLgotICAgICAgICAgICAgICAgICAgICBjb25jYXQoU3RyaW5nLnZhbHVlT2Yoc2l6ZSkpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBIYXNoTWFwUmVmZXJlbmNlIGhtciAgID0gZm9udHNUYWJsZS5nZXQoa2V5KTsKLSAgICAgICAgaWYgKGhtciAhPSBudWxsKSB7Ci0gICAgICAgICAgICBwZWVyID0gaG1yLmdldCgpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHBlZXIgPT0gbnVsbCkgewotICAgICAgICAgICAgcGVlciA9IGNyZWF0ZUZvbnRQZWVyKG5hbWUsIGZvbnRTdHlsZSwgc2l6ZSwgbG9naWNhbEluZGV4KTsKLSAgICAgICAgICAgIGlmIChwZWVyID09IG51bGwpewotICAgICAgICAgICAgICAgIHBlZXIgPSBnZXRGb250UGVlcihESUFMT0dfTkFNRSwgZm9udFN0eWxlLCBzaXplKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGZvbnRzVGFibGUucHV0KGtleSwgbmV3IEhhc2hNYXBSZWZlcmVuY2Uoa2V5LCBwZWVyLCBxdWV1ZSkpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHBlZXI7Ci0gICAgfQotICAgIAotICAgIC8qKgotICAgICAqIFJldHVybnMgaW5zdGFuY2Ugb2YgZm9udCBwZWVyIChsb2dpY2FsIG9yIHBoeXNpY2FsKSBhY2NvcmRpbmcgdG8gdGhlIAotICAgICAqIHNwZWNpZmllZCBwYXJhbWV0ZXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuYW1lIGZvbnQgZmFjZSBuYW1lCi0gICAgICogQHBhcmFtIHN0eWxlIHN0eWxlIG9mIHRoZSBmb250Ci0gICAgICogQHBhcmFtIHNpemUgc2l6ZSBvZiB0aGUgZm9udAotICAgICAqIEBwYXJhbSBsb2dpY2FsSW5kZXggaW5kZXggb2YgdGhlIGxvZ2ljYWwgZmFjZSBuYW1lIGluIExPR0lDQUxfRk9OVF9GQUNFUyAKLSAgICAgKiBhcnJheSBvciAtMSBpZiBkZXNpcmVkIGZvbnQgcGVlciBpcyBub3QgbG9naWNhbC4KLSAgICAgKi8KLSAgICBwcml2YXRlIEZvbnRQZWVyIGNyZWF0ZUZvbnRQZWVyKFN0cmluZyBuYW1lLCBpbnQgc3R5bGUsIGludCBzaXplLCBpbnQgbG9naWNhbEluZGV4KXsKLSAgICAgICAgRm9udFBlZXIgcGVlcjsKLSAgICAgICAgaWYgKGxvZ2ljYWxJbmRleCAhPSAtMSl7Ci0gICAgICAgICAgICBwZWVyID0gY3JlYXRlTG9naWNhbEZvbnRQZWVyKG5hbWUsIHN0eWxlLCBzaXplKTsKLSAgICAgICAgfWVsc2UgewotICAgICAgICAgICAgcGVlciA9IGNyZWF0ZVBoeXNpY2FsRm9udFBlZXIobmFtZSwgc3R5bGUsIHNpemUpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICByZXR1cm4gcGVlcjsKLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmYW1pbHkgbmFtZSBmb3IgbG9naWNhbCBmYWNlIG5hbWVzIGFzIGEgcGFyYW1ldGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBmYWNlTmFtZSBsb2dpY2FsIGZvbnQgZmFjZSBuYW1lCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZyBnZXRGYW1pbHlGcm9tTG9naWNhbEZhY2UoU3RyaW5nIGZhY2VOYW1lKXsKLSAgICAgICAgaW50IHBvcyA9IGZhY2VOYW1lLmluZGV4T2YoIi4iKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBpZiAocG9zID09IC0xKXsKLSAgICAgICAgICAgIHJldHVybiBmYWNlTmFtZTsKLSAgICAgICAgfQotICAgICAgICAgICAgCi0gICAgICAgIHJldHVybiBmYWNlTmFtZS5zdWJzdHJpbmcoMCwgcG9zKTsKLSAgICB9Ci0gICAgICAgICAgICAKLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIG5ldyBsb2dpY2FsIGZvbnQgcGVlciBmb3IgdGhlIHBhcmFtZXRlcnMgc3BlY2lmaWVkIHVzaW5nIGZvbnQgCi0gICAgICogcHJvcGVydGllcy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZmFjZU5hbWUgZmFjZSBuYW1lIG9mIHRoZSBsb2dpY2FsIGZvbnQgCi0gICAgICogQHBhcmFtIHN0eWxlIHN0eWxlIG9mIHRoZSBmb250IAotICAgICAqIEBwYXJhbSBzaXplIGZvbnQgc2l6ZQotICAgICAqIAotICAgICAqLwotICAgIHByaXZhdGUgRm9udFBlZXIgY3JlYXRlTG9naWNhbEZvbnRQZWVyKFN0cmluZyBmYWNlTmFtZSwgaW50IHN0eWxlLCBpbnQgc2l6ZSl7Ci0gICAgICAgIFN0cmluZyBmYW1pbHkgPSBnZXRGYW1pbHlGcm9tTG9naWNhbEZhY2UoZmFjZU5hbWUpOwotICAgICAgICBGb250UHJvcGVydHlbXSBmcHMgPSBnZXRGb250UHJvcGVydGllcyhmYW1pbHkudG9Mb3dlckNhc2UoKSArICIuIiArIHN0eWxlKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBpZiAoZnBzICE9IG51bGwpewotICAgICAgICAgICAgaW50IG51bUZvbnRzID0gZnBzLmxlbmd0aDsKLSAgICAgICAgICAgIEZvbnRQZWVySW1wbFtdIHBoeXNpY2FsRm9udHMgPSBuZXcgRm9udFBlZXJJbXBsW251bUZvbnRzXTsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtRm9udHM7IGkrKyl7Ci0gICAgICAgICAgICAgICAgRm9udFByb3BlcnR5IGZwID0gZnBzW2ldOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIFN0cmluZyBuYW1lID0gZnAuZ2V0TmFtZSgpOwotICAgICAgICAgICAgICAgIGludCBmcFN0eWxlID0gZnAuZ2V0U3R5bGUoKTsKLSAgICAgICAgICAgICAgICBTdHJpbmcga2V5ID0gbmFtZS5jb25jYXQoU3RyaW5nLnZhbHVlT2YoZnBTdHlsZSkpLgotICAgICAgICAgICAgICAgICAgICBjb25jYXQoU3RyaW5nLnZhbHVlT2Yoc2l6ZSkpOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIEhhc2hNYXBSZWZlcmVuY2UgaG1yICAgPSBmb250c1RhYmxlLmdldChrZXkpOwotICAgICAgICAgICAgICAgIGlmIChobXIgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBwaHlzaWNhbEZvbnRzW2ldID0gKEZvbnRQZWVySW1wbClobXIuZ2V0KCk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgaWYgKHBoeXNpY2FsRm9udHNbaV0gPT0gbnVsbCl7Ci0gICAgICAgICAgICAgICAgICAgIHBoeXNpY2FsRm9udHNbaV0gPSAoRm9udFBlZXJJbXBsKWNyZWF0ZVBoeXNpY2FsRm9udFBlZXIobmFtZSwgZnBTdHlsZSwgc2l6ZSk7Ci0gICAgICAgICAgICAgICAgICAgIGZvbnRzVGFibGUucHV0KGtleSwgbmV3IEhhc2hNYXBSZWZlcmVuY2Uoa2V5LCBwaHlzaWNhbEZvbnRzW2ldLCBxdWV1ZSkpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChwaHlzaWNhbEZvbnRzW2ldID09IG51bGwpewotICAgICAgICAgICAgICAgICAgICBwaHlzaWNhbEZvbnRzW2ldID0gKEZvbnRQZWVySW1wbClnZXREZWZhdWx0Rm9udChzdHlsZSwgc2l6ZSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIG5ldyBDb21wb3NpdGVGb250KGZhbWlseSwgZmFjZU5hbWUsIHN0eWxlLCBzaXplLCBmcHMsIHBoeXNpY2FsRm9udHMpOyAKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgLy8gaWYgdGhlcmUgaXMgbm8gcHJvcGVydHkgZm9yIHRoaXMgbG9naWNhbCBmb250IC0gZGVmYXVsdCBmb250IGlzIHRvIGJlCi0gICAgICAgIC8vIGNyZWF0ZWQKLSAgICAgICAgRm9udFBlZXJJbXBsIHBlZXIgPSAoRm9udFBlZXJJbXBsKWdldERlZmF1bHRGb250KHN0eWxlLCBzaXplKTsKLSAgICAgICAgCi0gICAgICAgIHJldHVybiBwZWVyOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgbmV3IHBoeXNpY2FsIGZvbnQgcGVlciBmb3IgdGhlIHBhcmFtZXRlcnMgc3BlY2lmaWVkIHVzaW5nIGZvbnQgcHJvcGVydGllcwotICAgICAqIFRoaXMgbWV0aG9kIG11c3QgYmUgb3ZlcnJpZGRlbiBieSBzdWJjbGFzc2VzIGltcGxlbWVudGF0aW9ucy4KLSAgICAgKiAgCi0gICAgICogQHBhcmFtIGZhY2VOYW1lIGZhY2UgbmFtZSBvciBmYW1pbHkgbmFtZSBvZiB0aGUgZm9udCAKLSAgICAgKiBAcGFyYW0gc3R5bGUgc3R5bGUgb2YgdGhlIGZvbnQgCi0gICAgICogQHBhcmFtIHNpemUgZm9udCBzaXplCi0gICAgICogCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEZvbnRQZWVyIGNyZWF0ZVBoeXNpY2FsRm9udFBlZXIoU3RyaW5nIG5hbWUsIGludCBzdHlsZSwgaW50IHNpemUpOwotICAgIAotICAgIC8qKgotICAgICAqIFJldHVybnMgZGVmYXVsdCBmb250IHBlZXIgY2xhc3Mgd2l0aCAiRGVmYXVsdCIgbmFtZSB0aGF0IGlzIHVzdWFsbHkgCi0gICAgICogdXNlZCB3aGVuIGZvbnQgd2l0aCBzcGVjaWZpZWQgZm9udCBuYW1lcyBhbmQgc3R5bGUgZG9lc24ndCBleHNpc3QgCi0gICAgICogb24gYSBzeXN0ZW0uIAotICAgICAqIAotICAgICAqIEBwYXJhbSBzdHlsZSBzdHlsZSBvZiB0aGUgZm9udAotICAgICAqIEBwYXJhbSBzaXplIHNpemUgb2YgdGhlIGZvbnQKLSAgICAgKi8KLSAgICBwdWJsaWMgRm9udFBlZXIgZ2V0RGVmYXVsdEZvbnQoaW50IHN0eWxlLCBpbnQgc2l6ZSl7Ci0gICAgICAgIHVwZGF0ZUZvbnRzVGFibGUoKTsKLSAgICAgICAgCi0gICAgICAgIEZvbnRQZWVyIHBlZXIgPSBudWxsOwotICAgICAgICBTdHJpbmcga2V5ID0gREVGQVVMVF9OQU1FLmNvbmNhdChTdHJpbmcudmFsdWVPZihzdHlsZSkpLgotICAgICAgICAgICAgICAgICAgICBjb25jYXQoU3RyaW5nLnZhbHVlT2Yoc2l6ZSkpOwotICAgICAgICAKLSAgICAgICAgSGFzaE1hcFJlZmVyZW5jZSBobXIgICA9IGZvbnRzVGFibGUuZ2V0KGtleSk7Ci0gICAgICAgIGlmIChobXIgIT0gbnVsbCkgewotICAgICAgICAgICAgcGVlciA9IGhtci5nZXQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChwZWVyID09IG51bGwpIHsKLSAgICAgICAgICAgIHBlZXIgPSBjcmVhdGVEZWZhdWx0Rm9udChzdHlsZSwgc2l6ZSk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgICgoRm9udFBlZXJJbXBsKXBlZXIpLnNldEZhbWlseShERUZBVUxUX05BTUUpOwotICAgICAgICAgICAgKChGb250UGVlckltcGwpcGVlcikuc2V0UFNOYW1lKERFRkFVTFRfTkFNRSk7Ci0gICAgICAgICAgICAoKEZvbnRQZWVySW1wbClwZWVyKS5zZXRGb250TmFtZShERUZBVUxUX05BTUUpOwotCi0gICAgICAgICAgICBmb250c1RhYmxlLnB1dChrZXksIG5ldyBIYXNoTWFwUmVmZXJlbmNlKGtleSwgcGVlciwgcXVldWUpKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBwZWVyOwotICAgIH0KLSAgICAKLSAgICAvKioKLSAgICAgKiAKLSAgICAgKiBSZXR1cm5zIG5ldyBkZWZhdWx0IGZvbnQgcGVlciB3aXRoICJEZWZhdWx0IiBuYW1lIGZvciB0aGUgcGFyYW1ldGVycyAKLSAgICAgKiBzcGVjaWZpZWQuIFRoaXMgbWV0aG9kIG11c3QgYmUgb3ZlcnJpZGRlbiBieSBzdWJjbGFzc2VzIGltcGxlbWVudGF0aW9ucy4KLSAgICAgKiAgCi0gICAgICogQHBhcmFtIHN0eWxlIHN0eWxlIG9mIHRoZSBmb250Ci0gICAgICogQHBhcmFtIHNpemUgc2l6ZSBvZiB0aGUgZm9udAotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBGb250UGVlciBjcmVhdGVEZWZhdWx0Rm9udChpbnQgc3R5bGUsIGludCBzaXplKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGZhY2UgbmFtZSBvZiB0aGUgbG9naWNhbCBmb250LCB3aGljaCBpcyB0aGUgcmVzdWx0Ci0gICAgICogb2Ygc3BlY2lmaWVkIGZvbnQgc3R5bGUgYW5kIGZhY2Ugc3R5bGUgdW5pb24uICAgCi0gICAgICogCi0gICAgICogQHBhcmFtIGZvbnRTdHlsZSBzcGVjaWZpZWQgc3R5bGUgb2YgdGhlIGZvbnQKLSAgICAgKiBAcGFyYW0gbG9naWNhbEluZGV4IGluZGV4IG9mIHRoZSBzcGVjaWZpZWQgZmFjZSBmcm9tIHRoZSAKLSAgICAgKiBMT0dJQ0FMX0ZPTlRfRkFDRVMgYXJyYXkKLSAgICAgKiBAcmV0dXJuIHJlc3VsdGluZyBmYWNlIG5hbWUKLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldExvZ2ljYWxGYWNlRnJvbUZvbnQoaW50IGZvbnRTdHlsZSwgaW50IGxvZ2ljYWxJbmRleCl7Ci0gICAgICAgIGludCBzdHlsZSA9IDA7Ci0gICAgICAgIFN0cmluZyBuYW1lID0gTE9HSUNBTF9GT05UX0ZBQ0VTW2xvZ2ljYWxJbmRleF07Ci0gICAgICAgIGludCBwb3MgPSBuYW1lLmluZGV4T2YoIi4iKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAKLSAgICAgICAgaWYgKHBvcyA9PSAtMSl7Ci0gICAgICAgICAgICByZXR1cm4gY3JlYXRlTG9naWNhbEZhY2UobmFtZSwgZm9udFN0eWxlKTsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgU3RyaW5nIHN0eWxlTmFtZSA9IG5hbWUuc3Vic3RyaW5nKHBvcysxKTsKLSAgICAgICAgbmFtZSA9IG5hbWUuc3Vic3RyaW5nKDAsIHBvcyk7Ci0gICAgICAgIAotICAgICAgICAvLyBhcHBlbmRpbmcgZm9udCBzdHlsZSB0byB0aGUgZmFjZSBzdHlsZQotICAgICAgICBzdHlsZSA9IGZvbnRTdHlsZSB8IGdldExvZ2ljYWxTdHlsZShzdHlsZU5hbWUpOwotICAgICAgICAKLSAgICAgICAgcmV0dXJuIGNyZWF0ZUxvZ2ljYWxGYWNlKG5hbWUsIHN0eWxlKTsKLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogRnVuY3Rpb24gcmV0dXJucyBzdHlsZSB2YWx1ZSBmcm9tIGxvZ2ljYWwgZmFjZSBuYW1lLgotICAgICAqICAKLSAgICAgKiBAcGFyYW0gbmFtZSBmYWNlIG5hbWUKLSAgICAgKiBAcmV0dXJuIGZvbnQgc3R5bGUKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFN0eWxlRnJvbUxvZ2ljYWxGYWNlKFN0cmluZyBuYW1lKXsKLSAgICAgICAgaW50IHN0eWxlOwotICAgICAgICBpbnQgcG9zID0gbmFtZS5pbmRleE9mKCIuIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgCi0gICAgICAgIGlmIChwb3MgPT0gLTEpewotICAgICAgICAgICAgcmV0dXJuIEZvbnQuUExBSU47Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIFN0cmluZyBzdHlsZU5hbWUgPSBuYW1lLnN1YnN0cmluZyhwb3MrMSk7Ci0gICAgICAgIAotICAgICAgICBzdHlsZSA9IGdldExvZ2ljYWxTdHlsZShzdHlsZU5hbWUpOwotICAgICAgICAKLSAgICAgICAgcmV0dXJuIHN0eWxlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgbG9naWNhbCBmYWNlIG5hbWUgY29ycmVzcG9uZGluZyB0byB0aGUgbG9naWNhbAotICAgICAqIGZhbWlseSBuYW1lIGFuZCBzdHlsZSBvZiB0aGUgZm9udC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZmFtaWx5IGZvbnQgZmFtaWx5Ci0gICAgICogQHBhcmFtIHN0eWxlSW5kZXggaW5kZXggb2YgdGhlIHN0eWxlIG5hbWUgZnJvbSB0aGUgU1RZTEVfTkFNRVMgYXJyYXkgCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZyBjcmVhdGVMb2dpY2FsRmFjZShTdHJpbmcgZmFtaWx5LCBpbnQgc3R5bGVJbmRleCl7Ci0gICAgICAgIHJldHVybiBmYW1pbHkgKyAiLiIgKyBTVFlMRV9OQU1FU1tzdHlsZUluZGV4XTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLSAgICAKLSAgICAvKioKLSAgICAgKiBSZXR1cm4gbGFuZ3VhZ2UgSWQgZnJvbSBMQ0lEIGhhc2ggY29ycmVzcG9uZGluZyB0byB0aGUgc3BlY2lmaWVkIGxvY2FsZQotICAgICAqIAotICAgICAqIEBwYXJhbSBsIHNwZWNpZmllZCBsb2NhbGUKLSAgICAgKi8KLSAgICBwdWJsaWMgU2hvcnQgZ2V0TENJRChMb2NhbGUgbCl7Ci0gICAgICAgIGlmICh0aGlzLnRhYmxlTENJRC5zaXplKCkgPT0gMCl7Ci0gICAgICAgICAgICBpbml0TENJRFRhYmxlKCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gdGFibGVMQ0lELmdldChsLnRvU3RyaW5nKCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBsYXRmb3JtLWRlcGVuZGVudCBMQ0lEIHRhYmxlIGluaXQuCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgaW5pdExDSURUYWJsZSgpOwotCi0gICAgLyoqCi0gICAgICogRnJlZWluZyBuYXRpdmUgcmVzb3VyY2VzLiBUaGlzIGhvb2sgaXMgdXNlZCB0byBhdm9pZCAKLSAgICAgKiBzdWRkZW4gYXBwbGljYXRpb24gZXhpdCBhbmQgdG8gZnJlZSByZXNvdXJjZXMgY3JlYXRlZCBpbiBuYXRpdmUgY29kZS4KLSAgICAgKi8KLSAgICBwcml2YXRlIGNsYXNzIERpc3Bvc2VOYXRpdmVIb29rIGV4dGVuZHMgVGhyZWFkIHsKLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgcnVuKCkgewotICAgICAgICAgICAgdHJ5ewotICAgICAgICAgICAgICAgIC8qIERpc3Bvc2luZyBuYXRpdmUgZm9udCBwZWVyJ3MgcmVzb3VyY2VzICovCi0gICAgICAgICAgICAgICAgRW51bWVyYXRpb248U3RyaW5nPiBrRW51bSA9IGZvbnRzVGFibGUua2V5cygpOwotCi0gICAgICAgICAgICAgICAgd2hpbGUoa0VudW0uaGFzTW9yZUVsZW1lbnRzKCkpewotICAgICAgICAgICAgICAgICAgICBPYmplY3Qga2V5ID0ga0VudW0ubmV4dEVsZW1lbnQoKTsKLSAgICAgICAgICAgICAgICAgICAgSGFzaE1hcFJlZmVyZW5jZSBobXIgPSBmb250c1RhYmxlLnJlbW92ZShrZXkpOwotICAgICAgICAgICAgICAgICAgICBGb250UGVlckltcGwgZGVsUGVlciA9IChGb250UGVlckltcGwpaG1yLmdldCgpOwotICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgaWYgKChkZWxQZWVyICE9IG51bGwpICYmIChkZWxQZWVyLmdldENsYXNzKCkgIT0gQ29tcG9zaXRlRm9udC5jbGFzcykpewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gdGhlcmUncyBub3RoaW5nIHRvIGRpc3Bvc2UgaW4gQ29tcG9zaXRlRm9udCBvYmplY3RzCi0gICAgICAgICAgICAgICAgICAgICAgICBkZWxQZWVyLmRpc3Bvc2UoKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gY2F0Y2ggKFRocm93YWJsZSB0KXsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbih0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBGaWxlIG9iamVjdCwgY3JlYXRlZCBpbiBhIGRpcmVjdG9yeQotICAgICAqIGFjY29yZGluZyB0byB0aGUgU3lzdGVtLCB3aGVyZSBKVk0gaXMgYmVpbmcgcmFuLgotICAgICAqCi0gICAgICogSW4gTGludXggY2FzZSB3ZSB1c2UgIi5mb250cyIgZGlyZWN0b3J5IChmb3IgZm9udGNvbmZpZyBwdXJwb3NlKSwKLSAgICAgKiB3aGVyZSBmb250IGZpbGUgZnJvbSB0aGUgc3RyZWFtIHdpbGwgYmUgc3RvcmVkLCBoZW5jZSBpbiBMaW51eEZvbnRNYW5hZ2VyIHRoaXMKLSAgICAgKiBtZXRob2QgaXMgb3ZlcnJpZGRlbi4KLSAgICAgKiBJbiBXaW5kb3dzIGNhc2Ugd2UgdXNlIFdpbmRvd3MgdGVtcCBkaXJlY3RvcnkgKGRlZmF1bHQgaW1wbGVtZW50YXRpb24pCi0gICAgICoKLSAgICAgKi8KLSAgICBwdWJsaWMgRmlsZSBnZXRUZW1wRm9udEZpbGUoKXRocm93cyBJT0V4Y2VwdGlvbnsKLSAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgLyoKLSAgICAgICAgRmlsZSBmb250RmlsZSA9IEZpbGUuY3JlYXRlVGVtcEZpbGUoImpGb250IiwgIi50dGYiKTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIGZvbnRGaWxlLmRlbGV0ZU9uRXhpdCgpOwotCi0gICAgICAgIHJldHVybiBmb250RmlsZTsKLSAgICAgICAgICovCi0gICAgICAgIGlmKE5PVF9JTVApCi0gICAgICAgICAgICB0aHJvdyBuZXcgTm90SW1wbGVtZW50ZWRFeGNlcHRpb24oImdldFRlbXBGb250RmlsZSBub3QgSW1wbGVtZW50ZWQiKTsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBGaWxlIG9iamVjdCB3aXRoIGZvbnQgcHJvcGVydGllcy4gSXQncyBuYW1lIG9idGFpbmVkIHVzaW5nIGN1cnJlbnQgCi0gICAgICogc3lzdGVtIGNvbmZpZ3VyYXRpb24gcHJvcGVydGllcyBhbmQgbG9jYWxlIHNldHRpbmdzLiBJZiBubyBhcHByb3ByaWF0ZSAKLSAgICAgKiBmaWxlIGlzIGZvdW5kIG1ldGhvZCByZXR1cm5zIG51bGwuIAotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgRmlsZSBnZXRGb250UHJvcGVydHlGaWxlKCl7Ci0gICAgICAgIEZpbGUgZmlsZSA9IG51bGw7Ci0KLSAgICAgICAgU3RyaW5nIGphdmFIb21lID0gU3lzdGVtLmdldFByb3BlcnR5KCJqYXZhLmhvbWUiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBMb2NhbGUgbCA9IExvY2FsZS5nZXREZWZhdWx0KCk7Ci0gICAgICAgIFN0cmluZyBsYW5ndWFnZSA9IGwuZ2V0TGFuZ3VhZ2UoKTsKLSAgICAgICAgU3RyaW5nIGNvdW50cnkgPSBsLmdldENvdW50cnkoKTsKLSAgICAgICAgU3RyaW5nIGZpbGVFbmNvZGluZyA9IFN5c3RlbS5nZXRQcm9wZXJ0eSgiZmlsZS5lbmNvZGluZyIpOyAvLyROT04tTkxTLTEkCi0KLSAgICAgICAgU3RyaW5nIG9zID0gU3lzdGVtLmdldFByb3BlcnR5KCJvcy5uYW1lIik7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICBpbnQgaSA9IDA7Ci0KLSAgICAgICAgLy8gT1MgbmFtZXMgZnJvbSBzeXN0ZW0gcHJvcGVydGllcyBkb24ndCBtYXRjaAotICAgICAgICAvLyBPUyBpZGVudGlmaWVycyB1c2VkIGluIGZvbnQucHJvcGVydHkgZmlsZXMKLSAgICAgICAgZm9yICg7IGkgPCBPU19WQUxVRVMubGVuZ3RoOyBpKyspewotICAgICAgICAgICAgaWYgKG9zLmVuZHNXaXRoKE9TX1ZBTFVFU1tpXSkpewotICAgICAgICAgICAgICAgIG9zID0gT1NfVkFMVUVTW2ldOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGkgPT0gT1NfVkFMVUVTLmxlbmd0aCl7Ci0gICAgICAgICAgICBvcyA9IG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICBTdHJpbmcgdmVyc2lvbiA9IFN5c3RlbS5nZXRQcm9wZXJ0eSgib3MudmVyc2lvbiIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIFN0cmluZyBwYXRobmFtZTsKLQotICAgICAgICBmb3IgKGkgPSAwOyBpIDwgRlBfRklMRV9OQU1FUy5sZW5ndGg7IGkrKyl7Ci0gICAgICAgICAgICBwYXRobmFtZSA9IEZQX0ZJTEVfTkFNRVNbaV07Ci0gICAgICAgICAgICBpZiAob3MgIT0gbnVsbCl7Ci0gICAgICAgICAgICAgICAgcGF0aG5hbWUgPSBwYXRobmFtZS5yZXBsYWNlRmlyc3QoIk9TIiwgb3MpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHBhdGhuYW1lID0gamF2YUhvbWUgKyBwYXRobmFtZTsKLQotICAgICAgICAgICAgcGF0aG5hbWUgPSBwYXRobmFtZS5yZXBsYWNlQWxsKCJMYW5ndWFnZSIsIGxhbmd1YWdlKS4gLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXBsYWNlQWxsKCJDb3VudHJ5IiwgY291bnRyeSkuIC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVwbGFjZUFsbCgiRW5jb2RpbmciLCBmaWxlRW5jb2RpbmcpLiAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlcGxhY2VBbGwoIlZlcnNpb24iLCB2ZXJzaW9uKTsgLy8kTk9OLU5MUy0xJAotCi0gICAgICAgICAgICBmaWxlID0gbmV3IEZpbGUocGF0aG5hbWUpOwotCi0gICAgICAgICAgICBpZiAoZmlsZS5leGlzdHMoKSl7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZmlsZS5leGlzdHMoKSA/IGZpbGUgOiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gYXJyYXkgb2YgaW50ZWdlciByYW5nZSB2YWx1ZXMKLSAgICAgKiBpZiB0aGUgcGFyYW1ldGVyIGV4Y2x1c2lvblN0cmluZyBoYXMgZm9ybWF0OgotICAgICAqICAgICAgICAgIFJhbmdlCi0gICAgICogICAgICAgICAgUmFuZ2UgWywgZXhjbHVzaW9uU3RyaW5nXQotICAgICAqCi0gICAgICogICAgICAgICAgUmFuZ2U6Ci0gICAgICogICAgICAgICAgICAgIENoYXItQ2hhcgotICAgICAqCi0gICAgICogICAgICAgICAgQ2hhcjoKLSAgICAgKiAgICAgICAgICAgICAgSGV4RGlnaXQgSGV4RGlnaXQgSGV4RGlnaXQgSGV4RGlnaXQKLSAgICAgKiAKLSAgICAgKiBNZXRob2QgcmV0dXJucyBudWxsIGlmIHRoZSBzcGVjaWZpZWQgc3RyaW5nIGlzIG51bGwuCi0gICAgICogIAotICAgICAqIEBwYXJhbSBleGNsdXNpb25TdHJpbmcgc3RyaW5nIHBhcmFtZXRlciBpbiBzcGVjaWZpZWQgZm9ybWF0Ci0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBpbnRbXSBwYXJzZUludGVydmFscyhTdHJpbmcgZXhjbHVzaW9uU3RyaW5nKXsKLSAgICAgICAgaW50W10gcmVzdWx0cyA9IG51bGw7Ci0KLSAgICAgICAgaWYgKGV4Y2x1c2lvblN0cmluZyA9PSBudWxsKXsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgU3RyaW5nW10gaW50ZXJ2YWxzID0gZXhjbHVzaW9uU3RyaW5nLnNwbGl0KCIsIik7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICBpZiAoaW50ZXJ2YWxzICE9IG51bGwpewotICAgICAgICAgICAgaW50IG51bSA9IGludGVydmFscy5sZW5ndGg7Ci0gICAgICAgICAgICBpZiAobnVtID4gMCl7Ci0gICAgICAgICAgICAgICAgcmVzdWx0cyA9IG5ldyBpbnRbaW50ZXJ2YWxzLmxlbmd0aCA8PCAxXTsKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGludGVydmFscy5sZW5ndGg7IGkrKyl7Ci0gICAgICAgICAgICAgICAgICAgIFN0cmluZyByYW5nZXNbXSA9IGludGVydmFsc1tpXS5zcGxpdCgiLSIpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgICAgIHJlc3VsdHNbaSoyXSA9IEludGVnZXIucGFyc2VJbnQocmFuZ2VzWzBdLCAxNik7Ci0gICAgICAgICAgICAgICAgICAgIHJlc3VsdHNbaSoyKzFdID0gSW50ZWdlci5wYXJzZUludChyYW5nZXNbMV0sIDE2KTsKLQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcmVzdWx0czsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIFByb3BlcnRpZXMgZnJvbSB0aGUgcHJvcGVydGllcyBmaWxlIG9yIG51bGwgaWYgCi0gICAgICogdGhlcmUgaXMgYW4gZXJyb3Igd2l0aCBGaWxlSW5wdXRTdHJlYW0gcHJvY2Vzc2luZy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZmlsZSBGaWxlIG9iamVjdCBjb250YWluaW5nIHByb3BlcnRpZXMKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIFByb3BlcnRpZXMgZ2V0UHJvcGVydGllcyhGaWxlIGZpbGUpewotICAgICAgICBQcm9wZXJ0aWVzIHByb3BzID0gbnVsbDsKLSAgICAgICAgRmlsZUlucHV0U3RyZWFtIGZpcyA9IG51bGw7Ci0gICAgICAgIHRyeXsKLSAgICAgICAgICAgIGZpcyA9IG5ldyBGaWxlSW5wdXRTdHJlYW0oZmlsZSk7Ci0gICAgICAgICAgICBwcm9wcyA9IG5ldyBQcm9wZXJ0aWVzKCk7Ci0gICAgICAgICAgICBwcm9wcy5sb2FkKGZpcyk7Ci0gICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKXsKLSAgICAgICAgICAgIFN5c3RlbS5vdXQucHJpbnRsbihlKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcHJvcHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhbiBhcnJheSBvZiBGb250UHJvcGVydGllcyBmcm9tIHRoZSBwcm9wZXJ0aWVzIGZpbGUKLSAgICAgKiB3aXRoIHRoZSBzcGVjaWZpZWQgcHJvcGVydHkgbmFtZSAibG9naWNhbCBmYWNlLnN0eWxlIi4gRS5nLiAKLSAgICAgKiAiZGlhbG9nLjIiIGNvcnJlc3BvbmRzIHRvIHRoZSBmb250IGZhbWlseSBEaWFsb2cgd2l0aCBib2xkIHN0eWxlLiAKLSAgICAgKgotICAgICAqIEBwYXJhbSBmcE5hbWUga2V5IG9mIHRoZSBmb250IHByb3BlcnRpZXMgaW4gdGhlIHByb3BlcnRpZXMgc2V0Ci0gICAgICovCi0gICAgcHVibGljIEZvbnRQcm9wZXJ0eVtdIGdldEZvbnRQcm9wZXJ0aWVzKFN0cmluZyBmcE5hbWUpewotICAgICAgICBWZWN0b3I8Rm9udFByb3BlcnR5PiBwcm9wcyA9IGZQcm9wZXJ0aWVzLmdldChmcE5hbWUpOwotICAgICAgICAKLSAgICAgICAgaWYgKHByb3BzID09IG51bGwpewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgc2l6ZSA9ICBwcm9wcy5zaXplKCk7Ci0gICAgICAgIAotICAgICAgICBpZiAoc2l6ZSA9PSAwKXsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgRm9udFByb3BlcnR5W10gZnBzID0gbmV3IEZvbnRQcm9wZXJ0eVtzaXplXTsKLSAgICAgICAgZm9yIChpbnQgaT0wOyBpIDwgZnBzLmxlbmd0aDsgaSsrKXsKLSAgICAgICAgICAgIGZwc1tpXSA9IHByb3BzLmVsZW1lbnRBdChpKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZnBzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgaW5kZXggb2YgdGhlIGZvbnQgbmFtZSBpbiBhcnJheSBvZiBmb250IG5hbWVzIG9yIC0xIGlmIAotICAgICAqIHRoaXMgZm9udCBpcyBub3QgbG9naWNhbC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gZm9udE5hbWUgc3BlY2lmaWVkIGZvbnQgbmFtZQotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgaW50IGdldExvZ2ljYWxGYWNlSW5kZXgoU3RyaW5nIGZvbnROYW1lKXsKLSAgICAgICAgZm9yIChpbnQgaT0wOyBpPExPR0lDQUxfRk9OVF9OQU1FUy5sZW5ndGg7IGkrKyApewotICAgICAgICAgICAgaWYgKExPR0lDQUxfRk9OVF9OQU1FU1tpXS5lcXVhbHNJZ25vcmVDYXNlKGZvbnROYW1lKSl7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIC0xOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiBzcGVjaWZpZWQgZmFtaWx5IG5hbWUgaXMgYXZhaWxhYmxlIGluIHRoaXMgCi0gICAgICogR3JhcGhpY3NFbnZpcm9ubWVudC4gCi0gICAgICogCi0gICAgICogQHBhcmFtIGZhbWlseU5hbWUgdGhlIHNwZWNpZmllZCBmb250IGZhbWlseSBuYW1lCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaXNGYW1pbHlFeGlzdChTdHJpbmcgZmFtaWx5TmFtZSl7Ci0gICAgICAgIHJldHVybiAoZ2V0RmFtaWx5SW5kZXgoZmFtaWx5TmFtZSkgIT0gLTEpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgaW5kZXggb2YgZmFtaWx5IG5hbWUgZnJvbSB0aGUgYXJyYXkgb2YgZmFtaWx5IG5hbWVzIGF2YWlsYWJsZSBpbiAKLSAgICAgKiB0aGlzIEdyYXBoaWNzRW52aXJvbm1lbnQgb3IgLTEgaWYgbm8gZmFtaWx5IG5hbWUgd2FzIGZvdW5kLgotICAgICAqIAotICAgICAqIEBwYXJhbSBmYW1pbHlOYW1lIHNwZWNpZmllZCBmb250IGZhbWlseSBuYW1lIAotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0RmFtaWx5SW5kZXgoU3RyaW5nIGZhbWlseU5hbWUpewotICAgICAgICBmb3IgKGludCBpPTA7IGk8YWxsRmFtaWxpZXMubGVuZ3RoOyBpKysgKXsKLSAgICAgICAgICAgIGlmIChmYW1pbHlOYW1lLmVxdWFsc0lnbm9yZUNhc2UoYWxsRmFtaWxpZXNbaV0pKXsKLSAgICAgICAgICAgICAgICByZXR1cm4gaTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmYW1pbHkgd2l0aCBpbmRleCBzcGVjaWZpZWQgZnJvbSB0aGUgYXJyYXkgb2YgZmFtaWx5IG5hbWVzIGF2YWlsYWJsZSBpbiAKLSAgICAgKiB0aGlzIEdyYXBoaWNzRW52aXJvbm1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIGluZGV4IGluZGV4IG9mIHRoZSBmYW1pbHkgaW4gZmFtaWxpZXMgbmFtZXMgYXJyYXkgCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZyBnZXRGYW1pbHkoaW50IGluZGV4KXsKLSAgICAgICAgcmV0dXJuIGFsbEZhbWlsaWVzW2luZGV4XTsKLSAgICB9Ci0gICAgLyoqCi0gICAgICogUmV0dXJucyBpbmRleCBvZiBmYWNlIG5hbWUgZnJvbSB0aGUgYXJyYXkgb2YgZmFjZSBuYW1lcyBhdmFpbGFibGUgaW4gCi0gICAgICogdGhpcyBHcmFwaGljc0Vudmlyb25tZW50IG9yIC0xIGlmIG5vIGZhY2UgbmFtZSB3YXMgZm91bmQuIERlZmF1bHQgcmV0dXJuIAotICAgICAqIHZhbHVlIGlzIC0xLCBtZXRob2QgbXVzdCBiZSBvdmVycmlkZGVuIGJ5IEZvbnRNYW5hZ2VyIGltcGxlbWVudGF0aW9uLgotICAgICAqIAotICAgICAqIEBwYXJhbSBmYWNlTmFtZSBmb250IGZhY2UgbmFtZSB3aGljaCBpbmRleCBpcyB0byBiZSBzZWFyY2hlZAotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0RmFjZUluZGV4KFN0cmluZyBmYWNlTmFtZSl7Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYWJzdHJhY3QgU3RyaW5nW10gZ2V0QWxsRmFtaWxpZXMoKTsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCBGb250W10gZ2V0QWxsRm9udHMoKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBDbGFzcyBjb250YWlucyBTb2Z0UmVmZXJlbmNlIGluc3RhbmNlIHRoYXQgY2FuIGJlIHN0b3JlZCBpbiB0aGUgCi0gICAgICogSGFzaHRhYmxlIGJ5IG1lYW5zIG9mIGtleSBmaWVsZCBjb3JyZXNwb25kaW5nIHRvIGl0LgotICAgICAqLwotICAgIHByaXZhdGUgY2xhc3MgSGFzaE1hcFJlZmVyZW5jZSBleHRlbmRzIFNvZnRSZWZlcmVuY2U8Rm9udFBlZXI+IHsKLSAgICAgICAgCi0gICAgICAgIC8qKgotICAgICAgICAgKiBUaGUga2V5IGZvciBIYXNodGFibGUuCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIGZpbmFsIFN0cmluZyBrZXk7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENyZWF0ZXMgYSBuZXcgc29mdCByZWZlcmVuY2Ugd2l0aCB0aGUga2V5IHNwZWNpZmllZCBhbmQgCi0gICAgICAgICAqIGFkZGluZyB0aGlzIHJlZmVyZW5jZSBpbiB0aGUgcmVmZXJlbmNlIHF1ZXVlIHNwZWNpZmllZC4KLSAgICAgICAgICoKLSAgICAgICAgICogQHBhcmFtIGtleSB0aGUga2V5IGluIEhhc2h0YWJsZQotICAgICAgICAgKiBAcGFyYW0gdmFsdWUgb2JqZWN0IHRoYXQgY29ycmVzcG9uZHMgdG8gdGhlIGtleQotICAgICAgICAgKiBAcGFyYW0gcXVldWUgcmVmZXJlbmNlIHF1ZXVlIHdoZXJlIHJlZmVyZW5jZSBpcyB0byBiZSBhZGRlZCAKLSAgICAgICAgICovCi0gICAgICAgIHB1YmxpYyBIYXNoTWFwUmVmZXJlbmNlKGZpbmFsIFN0cmluZyBrZXksIGZpbmFsIEZvbnRQZWVyIHZhbHVlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmluYWwgUmVmZXJlbmNlUXVldWU8Rm9udFBlZXI+IHF1ZXVlKSB7Ci0gICAgICAgICAgICBzdXBlcih2YWx1ZSwgcXVldWUpOwotICAgICAgICAgICAgdGhpcy5rZXkgPSBrZXk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyB0aGUga2V5IHRoYXQgY29ycmVzcG9uZHMgdG8gdGhlIFNvZnRSZWZlcmVuY2UgaW5zdGFuY2UgCi0gICAgICAgICAqCi0gICAgICAgICAqIEByZXR1cm4gdGhlIGtleSBpbiBIYXNodGFibGUgd2l0aCBjYWNoZWQgcmVmZXJlbmNlcwotICAgICAgICAgKi8KLSAgICAgICAgcHVibGljIE9iamVjdCBnZXRLZXkoKSB7Ci0gICAgICAgICAgICByZXR1cm4ga2V5OwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVtb3ZlcyBrZXlzIGZyb20gdGhlIEhhc2h0YWJsZSB3aXRoIGZvbnQgcGVlcnMgd2hpY2ggY29ycmVzcG9uZGluZyAKLSAgICAgKiBIYXNoTWFwUmVmZXJlbmNlIG9iamVjdHMgd2VyZSBnYXJiYWdlIGNvbGxlY3RlZC4KLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgdXBkYXRlRm9udHNUYWJsZSgpIHsKLSAgICAgICAgSGFzaE1hcFJlZmVyZW5jZSByOwotICAgICAgICAvLz8/P0FXVAotICAgICAgICAvL3doaWxlICgociA9IChIYXNoTWFwUmVmZXJlbmNlKXF1ZXVlLnBvbGwoKSkgIT0gbnVsbCkgewotICAgICAgICAvLyAgICBmb250c1RhYmxlLnJlbW92ZShyLmdldEtleSgpKTsKLSAgICAgICAgLy99Ci0gICAgfQotCi19Ci0KLQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Gb250TWV0cmljc0ltcGwuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvRm9udE1ldHJpY3NJbXBsLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDc3ODMzMTcuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Gb250TWV0cmljc0ltcGwuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI4MiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi1pbXBvcnQgY29tLmFuZHJvaWQuaW50ZXJuYWwuYXd0LkFuZHJvaWRHcmFwaGljczJEOwotCi1pbXBvcnQgamF2YS5hd3QuRm9udDsKLWltcG9ydCBqYXZhLmF3dC5Gb250TWV0cmljczsKLS8vaW1wb3J0IGphdmEuYXd0LlBhaW50OwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotCi1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5QYWludDsKLQotLyoqCi0gKiBGb250TWV0cmljcyBpbXBsZW1lbnRhdGlvbgotICovCi0KLXB1YmxpYyBjbGFzcyBGb250TWV0cmljc0ltcGwgZXh0ZW5kcyBGb250TWV0cmljcyB7Ci0KLQlwcml2YXRlIHN0YXRpYyBmaW5hbCBsb25nIHNlcmlhbFZlcnNpb25VSUQgPSA4NDQ2OTU2MTUyMDE5MjUxMzhMOwotCi0JLy8gYXNjZW50IG9mIHRoZSBmb250Ci0JcHJpdmF0ZSBpbnQgYXNjZW50OwotCi0JLy8gZGVzY2VudCBvZiB0aGUgZm9udAotCXByaXZhdGUgaW50IGRlc2NlbnQ7Ci0KLQkvLyBsZWFkaW5nIG9mIHRoZSBmb250Ci0JcHJpdmF0ZSBpbnQgbGVhZGluZzsKLQotCS8vIG1heGltdW0gYXNjZW50IG9mIHRoZSBmb250Ci0JcHJpdmF0ZSBpbnQgbWF4QXNjZW50OwotCi0JLy8gbWF4aW11bSBkZXNjZW50IG9mIHRoZSBmb250Ci0JcHJpdmF0ZSBpbnQgbWF4RGVzY2VudDsKLQotCS8vIG1heGltdW0gYWR2YW5jZSBvZiB0aGUgZm9udAotCXByaXZhdGUgaW50IG1heEFkdmFuY2U7Ci0KLQkvLyBhcnJheSBvZiBjaGFyIGFkdmFuY2Ugd2lkdGhzCi0JcHJpdmF0ZSBpbnRbXSB3aWR0aHMgPSBuZXcgaW50WzI1Nl07Ci0KLQkvLyBmb250IHBlZXIgY29ycmVzcG9uZGluZyB0byB0aGlzIEZvbnRQZWVySW1wbAotCXByaXZhdGUgdHJhbnNpZW50IEZvbnRQZWVySW1wbCBwZWVyOwotCi0JLy8gWCBzY2FsZSBwYXJhbWV0ZXIgb2YgdGhlIGZvbnQgdHJhbnNmb3JtCi0JcHJpdmF0ZSBmbG9hdCBzY2FsZVggPSAxOwotCi0JcHVibGljIEFuZHJvaWRHcmFwaGljczJEIG1TZzsKLQotCXByaXZhdGUgRm9udCBtRm47Ci0KLQkvLyBZIHNjYWxlIHBhcmFtZXRlciBvZiB0aGUgZm9udCB0cmFuc2Zvcm0KLQlwcml2YXRlIGZsb2F0IHNjYWxlWSA9IDE7Ci0KLQkvKioKLQkgKiBDcmVhdGVzIG5ldyBGb250TWVyaWNzSW1wbCBvYmplY3QgZGVzY3JpYmVkIGJ5IHRoZSBzcGVjaWZpZWQgRm9udC4KLQkgKiAKLQkgKiBAcGFyYW0gZm50Ci0JICogICAgICAgICAgICB0aGUgc3BlY2lmaWVkIEZvbnQgb2JqZWN0Ci0JICovCi0JcHVibGljIEZvbnRNZXRyaWNzSW1wbChGb250IGZudCkgewotCQlzdXBlcihmbnQpOwotCQl0aGlzLm1GbiA9IGZudDsKLQkJCi0JCW1TZyA9IEFuZHJvaWRHcmFwaGljczJELmdldEluc3RhbmNlKCk7Ci0JCVBhaW50IHAgPSBtU2cuZ2V0QW5kcm9pZFBhaW50KCk7Ci0JCQotCQl0aGlzLmFzY2VudCA9IChpbnQpLXAuYXNjZW50KCk7Ci0JCXRoaXMuZGVzY2VudCA9IChpbnQpcC5kZXNjZW50KCk7Ci0JCXRoaXMubGVhZGluZyA9IHAuZ2V0Rm9udE1ldHJpY3NJbnQoKS5sZWFkaW5nOwotCQkKLQkJQWZmaW5lVHJhbnNmb3JtIGF0ID0gZm50LmdldFRyYW5zZm9ybSgpOwotCQlpZiAoIWF0LmlzSWRlbnRpdHkoKSkgewotCQkJc2NhbGVYID0gKGZsb2F0KSBhdC5nZXRTY2FsZVgoKTsKLQkJCXNjYWxlWSA9IChmbG9hdCkgYXQuZ2V0U2NhbGVZKCk7Ci0JCX0KLQkJCQkKLQkgICAgLyoKLQkgICAgICogbWV0cmljc1s1XSAtIHN0cmlrZXRocm91Z2ggdGhpY2tuZXNzPHA+Ci0JICAgICAqIC1tZXRyaWNzWzZdIC0gc3RyaWtldGhyb3VnaCBvZmZzZXQ8cD4KLQkgICAgICogbWV0cmljc1s3XSAtIG1heGltdW0gY2hhciB3aWR0aDxwPgotCSAgICAgKiBtZXRyaWNzWzhdIC0gYXNjZW50IGluIHBpeGVsczxwPgotCSAgICAgKiBtZXRyaWNzWzldIC0gZGVzY2VudCBpbiBwaXhsZXM8cD4KLQkgICAgICogbWV0cmljc1sxMF0gLSBleHRlcm5hbCBsZWFkaW5nIGluIHBpeGVsczxwPgotCSAgICAgKiBtZXRyaWNzWzExXSAtIHVuZGVybGluZSB0aGlja25lc3MgaW4gcGl4ZWxzPHA+Ci0JICAgICAqIC1tZXRyaWNzWzEyXSAtIHVuZGVybGluZSBvZmZzZXQgaW4gcGl4ZWxzPHA+Ci0JICAgICAqIG1ldHJpY3NbMTNdIC0gc3RyaWtldGhyb3VnaCB0aGlja25lc3MgaW4gcGl4ZWxzPHA+Ci0JICAgICAqIC1tZXRyaWNzWzE0XSAtIHN0cmlrZXRocm91Z2ggb2Zmc2V0IGluIHBpeGVsczxwPgotCSAgICAgKiBtZXRyaWNzWzE1XSAtIG1heGltdW0gY2hhciB3aWR0aCBpbiBwaXhlbHM8cD4KLQotCSAgICAgKiBAcGFyYW0gX2Jhc2VsaW5lRGF0YSBhbiBhcnJheSBvZiAzIGVsZW1lbnRzIHdpdGggYmFzZWxpbmUgb2Zmc2V0cyBtZXRyaWNzPHA+Ci0JICAgICAqIF9iYXNlbGluZURhdGFbMF0gLSByb21hbiBiYXNlbGluZSBvZmZzZXQ8cD4gCi0JICAgICAqIF9iYXNlbGluZURhdGFbMV0gLSBjZW50ZXIgYmFzZWxpbmUgb2Zmc2V0PHA+Ci0JICAgICAqIF9iYXNlbGluZURhdGFbMl0gLSBoYW5naW5nIGJhc2VsaW5lIG9mZnNldDxwPgotCSAgICAgKi8KLQl9Ci0KLQotCS8qKgotCSAqIEluaXRpYWxpemUgdGhlIGFycmF5IG9mIHRoZSBmaXJzdCAyNTYgY2hhcnMnIGFkdmFuY2Ugd2lkdGhzIG9mIHRoZSBGb250Ci0JICogZGVzY3JpYmluZyB0aGlzIEZvbnRNZXRyaWNzSW1wbCBvYmplY3QuCi0JICovCi0JcHJpdmF0ZSB2b2lkIGluaXRXaWR0aHMoKSB7Ci0KLQkJdGhpcy53aWR0aHMgPSBuZXcgaW50WzI1Nl07Ci0JCWZvciAoaW50IGNociA9IDA7IGNociA8IDI1NjsgY2hyKyspIHsKLQkJCXdpZHRoc1tjaHJdID0gKGludCkgKGdldEZvbnRQZWVyKCkuY2hhcldpZHRoKChjaGFyKSBjaHIpICogc2NhbGVYKTsKLQkJfQotCi0JfQotCi0JLyoqCi0JICogUmV0dXJucyB0aGUgYXNjZW50IG9mIHRoZSBGb250IGRlc2NyaWJpbmcgdGhpcyBGb250TWV0cmljc0ltcGwgb2JqZWN0LgotCSAqLwotCUBPdmVycmlkZQotCXB1YmxpYyBpbnQgZ2V0QXNjZW50KCkgewotCQlyZXR1cm4gdGhpcy5hc2NlbnQ7Ci0JfQotCi0JLyoqCi0JICogUmV0dXJucyB0aGUgZGVzY2VudCBvZiB0aGUgRm9udCBkZXNjcmliaW5nIHRoaXMgRm9udE1ldHJpY3NJbXBsIG9iamVjdC4KLQkgKi8KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgaW50IGdldERlc2NlbnQoKSB7Ci0JCXJldHVybiB0aGlzLmRlc2NlbnQ7Ci0JfQotCi0JLyoqCi0JICogUmV0dXJucyB0aGUgbGVhZGluZyBvZiB0aGUgRm9udCBkZXNjcmliaW5nIHRoaXMgRm9udE1ldHJpY3NJbXBsIG9iamVjdC4KLQkgKi8KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgaW50IGdldExlYWRpbmcoKSB7Ci0JCXJldHVybiB0aGlzLmxlYWRpbmc7Ci0JfQotCi0JLyoqCi0JICogUmV0dXJucyB0aGUgYWR2YW5jZSB3aWR0aCBvZiB0aGUgc3BlY2lmaWVkIGNoYXIgb2YgdGhlIEZvbnQgZGVzY3JpYmluZwotCSAqIHRoaXMgRm9udE1ldHJpY3NJbXBsIG9iamVjdC4KLQkgKiAKLQkgKiBAcGFyYW0gY2gKLQkgKiAgICAgICAgICAgIHRoZSBjaGFyIHdoaWNoIHdpZHRoIGlzIHRvIGJlIHJldHVybmVkCi0JICogQHJldHVybiB0aGUgYWR2YW5jZSB3aWR0aCBvZiB0aGUgc3BlY2lmaWVkIGNoYXIgb2YgdGhlIEZvbnQgZGVzY3JpYmluZwotCSAqICAgICAgICAgdGhpcyBGb250TWV0cmljc0ltcGwgb2JqZWN0Ci0JICovCi0JQE92ZXJyaWRlCi0JcHVibGljIGludCBjaGFyV2lkdGgoaW50IGNoKSB7Ci0JCWlmIChjaCA8IDI1NikgewotCQkJcmV0dXJuIHdpZHRoc1tjaF07Ci0JCX0KLQotCQlyZXR1cm4gZ2V0Rm9udFBlZXIoKS5jaGFyV2lkdGgoKGNoYXIpIGNoKTsKLQl9Ci0KLQkvKioKLQkgKiBSZXR1cm5zIHRoZSBhZHZhbmNlIHdpZHRoIG9mIHRoZSBzcGVjaWZpZWQgY2hhciBvZiB0aGUgRm9udCBkZXNjcmliaW5nCi0JICogdGhpcyBGb250TWV0cmljc0ltcGwgb2JqZWN0LgotCSAqIAotCSAqIEBwYXJhbSBjaAotCSAqICAgICAgICAgICAgdGhlIGNoYXIgd2hpY2ggd2lkdGggaXMgdG8gYmUgcmV0dXJuZWQKLQkgKiBAcmV0dXJuIHRoZSBhZHZhbmNlIHdpZHRoIG9mIHRoZSBzcGVjaWZpZWQgY2hhciBvZiB0aGUgRm9udCBkZXNjcmliaW5nCi0JICogICAgICAgICB0aGlzIEZvbnRNZXRyaWNzSW1wbCBvYmplY3QKLQkgKi8KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgaW50IGNoYXJXaWR0aChjaGFyIGNoKSB7Ci0JCWlmIChjaCA8IDI1NikgewotCQkJcmV0dXJuIHdpZHRoc1tjaF07Ci0JCX0KLQkJcmV0dXJuIChpbnQpIChnZXRGb250UGVlcigpLmNoYXJXaWR0aChjaCkgKiBzY2FsZVgpOwotCX0KLQotCS8qKgotCSAqIFJldHVybnMgdGhlIG1heGltdW0gYWR2YW5jZSBvZiB0aGUgRm9udCBkZXNjcmliaW5nIHRoaXMgRm9udE1ldHJpY3NJbXBsCi0JICogb2JqZWN0LgotCSAqLwotCUBPdmVycmlkZQotCXB1YmxpYyBpbnQgZ2V0TWF4QWR2YW5jZSgpIHsKLQkJcmV0dXJuIHRoaXMubWF4QWR2YW5jZTsKLQl9Ci0KLQkvKioKLQkgKiBSZXR1cm5zIHRoZSBtYXhpbXVtIGFzY2VudCBvZiB0aGUgRm9udCBkZXNjcmliaW5nIHRoaXMgRm9udE1ldHJpY3NJbXBsCi0JICogb2JqZWN0LgotCSAqLwotCUBPdmVycmlkZQotCXB1YmxpYyBpbnQgZ2V0TWF4QXNjZW50KCkgewotCQlyZXR1cm4gdGhpcy5tYXhBc2NlbnQ7Ci0JfQotCi0JLyoqCi0JICogUmV0dXJucyB0aGUgbWF4aW11bSBkZXNjZW50IG9mIHRoZSBGb250IGRlc2NyaWJpbmcgdGhpcyBGb250TWV0cmljc0ltcGwKLQkgKiBvYmplY3QuCi0JICovCi0JQFN1cHByZXNzV2FybmluZ3MoImRlcHJlY2F0aW9uIikKLQlARGVwcmVjYXRlZAotCUBPdmVycmlkZQotCXB1YmxpYyBpbnQgZ2V0TWF4RGVjZW50KCkgewotCQlyZXR1cm4gdGhpcy5tYXhEZXNjZW50OwotCX0KLQotCS8qKgotCSAqIFJldHVybnMgdGhlIG1heGltdW0gZGVzY2VudCBvZiB0aGUgRm9udCBkZXNjcmliaW5nIHRoaXMgRm9udE1ldHJpY3NJbXBsCi0JICogb2JqZWN0LgotCSAqLwotCUBPdmVycmlkZQotCXB1YmxpYyBpbnQgZ2V0TWF4RGVzY2VudCgpIHsKLQkJcmV0dXJuIHRoaXMubWF4RGVzY2VudDsKLQl9Ci0KLQkvKioKLQkgKiBSZXR1cm5zIHRoZSBhZHZhbmNlIHdpZHRocyBvZiB0aGUgZmlyc3QgMjU2IGNoYXJhY3RlcnMgaW4gdGhlIEZvbnQKLQkgKiBkZXNjcmliaW5nIHRoaXMgRm9udE1ldHJpY3NJbXBsIG9iamVjdC4KLQkgKi8KLQlAT3ZlcnJpZGUKLQlwdWJsaWMgaW50W10gZ2V0V2lkdGhzKCkgewotCQlyZXR1cm4gdGhpcy53aWR0aHM7Ci0JfQotCi0JLyoqCi0JICogUmV0dXJucyB0aGUgdG90YWwgYWR2YW5jZSB3aWR0aCBvZiB0aGUgc3BlY2lmaWVkIHN0cmluZyBpbiB0aGUgbWV0cmljcyBvZgotCSAqIHRoZSBGb250IGRlc2NyaWJpbmcgdGhpcyBGb250TWV0cmljc0ltcGwgb2JqZWN0LgotCSAqIAotCSAqIEBwYXJhbSBzdHIKLQkgKiAgICAgICAgICAgIHRoZSBTdHJpbmcgd2hpY2ggd2lkdGggaXMgdG8gYmUgbWVhc3VyZWQKLQkgKiBAcmV0dXJuIHRoZSB0b3RhbCBhZHZhbmNlIHdpZHRoIG9mIHRoZSBzcGVjaWZpZWQgc3RyaW5nIGluIHRoZSBtZXRyaWNzIG9mCi0JICogICAgICAgICB0aGUgRm9udCBkZXNjcmliaW5nIHRoaXMgRm9udE1ldHJpY3NJbXBsIG9iamVjdAotCSAqLwotCUBPdmVycmlkZQotCXB1YmxpYyBpbnQgc3RyaW5nV2lkdGgoU3RyaW5nIHN0cikgewotCi0JCWludCB3aWR0aCA9IDA7Ci0JCWNoYXIgY2hyOwotCi0JCWZvciAoaW50IGkgPSAwOyBpIDwgc3RyLmxlbmd0aCgpOyBpKyspIHsKLQkJCWNociA9IHN0ci5jaGFyQXQoaSk7Ci0JCQl3aWR0aCArPSBjaGFyV2lkdGgoY2hyKTsKLQkJfQotCQlyZXR1cm4gd2lkdGg7Ci0KLQkJLyoKLQkJICogZmxvYXQgcmVzID0gMDsgaW50IGxuID0gc3RyLmxlbmd0aCgpOyBjaGFyW10gYyA9IG5ldyBjaGFyW2xuXTsgZmxvYXRbXSBmID0KLQkJICogbmV3IGZsb2F0W2xuXTsgc3RyLmdldENoYXJzKDAsIGxuLCBjLCAwKTsgbVNnLmdldFBhaW50KCkuZ2V0VGV4dFdpZHRocyhjLCAwLAotCQkgKiBsbiwgZik7Ci0JCSAqIAotCQkgKiBmb3IoaW50IGkgPSAwOyBpIDwgZi5sZW5ndGg7IGkrKykgeyByZXMgKz0gZltpXTsgfSByZXR1cm4gKGludClyZXM7Ci0JCSAqLwotCX0KLQotCS8qKgotCSAqIFJldHVybnMgRm9udFBlZXIgaW1wbGVtZW50YXRpb24gb2YgdGhlIEZvbnQgZGVzY3JpYmluZyB0aGlzCi0JICogRm9udE1ldHJpY3NJbXBsIG9iamVjdC4KLQkgKiAKLQkgKiBAcmV0dXJuIGEgRm9udFBlZXIgb2JqZWN0LCB0aGF0IGlzIHRoZSBwbGF0Zm9ybSBkZXBlbmRlbnQgRm9udFBlZXIKLQkgKiAgICAgICAgIGltcGxlbWVudGF0aW9uIGZvciB0aGUgRm9udCBkZXNjcmliaW5nIHRoaXMgRm9udE1ldHJpY3NJbXBsCi0JICogICAgICAgICBvYmplY3QuCi0JICovCi0JQFN1cHByZXNzV2FybmluZ3MoImRlcHJlY2F0aW9uIikKLQlwdWJsaWMgRm9udFBlZXJJbXBsIGdldEZvbnRQZWVyKCkgewotCQlpZiAocGVlciA9PSBudWxsKSB7Ci0JCQlwZWVyID0gKEZvbnRQZWVySW1wbCkgZm9udC5nZXRQZWVyKCk7Ci0JCX0KLQkJcmV0dXJuIHBlZXI7Ci0JfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Gb250UGVlckltcGwuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvRm9udFBlZXJJbXBsLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDE0ZmY5OTcuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Gb250UGVlckltcGwuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQ5OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi0KLWltcG9ydCBjb20uYW5kcm9pZC5pbnRlcm5hbC5hd3QuQW5kcm9pZEdyYXBoaWNzMkQ7Ci1pbXBvcnQgY29tLmFuZHJvaWQuaW50ZXJuYWwuYXd0LkFuZHJvaWRHcmFwaGljc0ZhY3Rvcnk7Ci0KLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljczJEOwotaW1wb3J0IGphdmEuYXd0LlRvb2xraXQ7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5BZmZpbmVUcmFuc2Zvcm07Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLWltcG9ydCBqYXZhLmF3dC5wZWVyLkZvbnRQZWVyOwotCi1pbXBvcnQgamF2YS5hd3QuZm9udC5Gb250UmVuZGVyQ29udGV4dDsKLWltcG9ydCBqYXZhLmF3dC5mb250LkxpbmVNZXRyaWNzOwotaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5QYWludDsKLQotLyoqCi0gKiBBYnN0cmFjdCBjbGFzcyBmb3IgcGxhdGZvcm0gZGVwZW5kZW50IHBlZXIgaW1wbGVtZW50YXRpb24gb2YgdGhlIEZvbnQgY2xhc3MuCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBGb250UGVlckltcGwgaW1wbGVtZW50cyBGb250UGVlcnsKLQotICAgIC8vIGFzY2VudCBvZiB0aGlzIGZvbnQgcGVlciAoaW4gcGl4ZWxzKQotICAgIGludCBhc2NlbnQ7Ci0KLSAgICAvLyBkZXNjZW50IG9mIHRoaXMgZm9udCBwZWVyIChpbiBwaXhlbHMpCi0gICAgaW50IGRlc2NlbnQ7Ci0KLSAgICAvLyBsZWFkaW5nIG9mIHRoaXMgZm9udCBwZWVyIChpbiBwaXhlbHMpIAotICAgIGludCBsZWFkaW5nOwotCi0gICAgLy8gbG9naWNhbCBtYXhpbXVtIGFkdmFuY2Ugb2YgdGhpcyBmb250IHBlZXIgKGluIHBpeGVscykKLSAgICBpbnQgbWF4QWR2YW5jZTsKLQotICAgIC8vIHRoZSBoZWlnaHQgb2YgdGhpcyBmb250IHBlZXIKLSAgICBmbG9hdCBoZWlnaHQ7Ci0KLSAgICAvLyB0aGUgc3R5bGUgb2YgdGhpcyBmb250IHBlZXIKLSAgICBpbnQgc3R5bGU7Ci0KLSAgICAvLyB0aGUgcG9pbnQgc2l6ZSBvZiB0aGlzIGZvbnQgcGVlciAoaW4gcGl4ZWxzKQotICAgIGludCBzaXplOwotCi0gICAgLy8gdGhlIGxvZ2ljYWwgaGlnaHQgb2YgdGhpcyBmb250IHBlZXIgKGluIHBpeGVscykKLSAgICBpbnQgbG9naWNhbEhlaWdodDsKLQotICAgIC8vIHRoZSBuYW1lIG9mIHRoaXMgZm9udCBwZWVyCi0gICAgU3RyaW5nIG5hbWU7Ci0KLSAgICAvLyBmYW1pbHkgbmFtZSBvZiB0aGlzIGZvbnQgcGVlcgotICAgIFN0cmluZyBmb250RmFtaWx5TmFtZTsKLQotICAgIC8vIHRoZSBGYWNlIG5hbWUgb2YgdGhpcyBmb250IHBlZXIKLSAgICBTdHJpbmcgZmFjZU5hbWU7Ci0KLSAgICAvLyBib3VuZHMgcmVjdGFubGdlIG9mIHRoZSBsYXJnZXN0IGNoYXJhY3RlciBpbiB0aGlzIGZvbnQgcGVlcgotICAgIFJlY3RhbmdsZTJEIG1heENoYXJCb3VuZHM7Ci0KLSAgICAvLyBpdGFsaWMgYW5nbGUgdmFsdWUgb2YgdGhpcyBmb250IHBlZXIKLSAgICBmbG9hdCBpdGFsaWNBbmdsZSA9IDAuMGY7Ci0KLSAgICAvLyB0aGUgbnVtYmVyIG9mIGdseXBocyBzdXBwb3J0ZWQgYnkgdGhpcyBmb250IHBlZXIKLSAgICBpbnQgbnVtR2x5cGhzID0gMDsKLQotICAgIC8vIG5hdGl2ZSBmb250IGhhbmRsZQotICAgIGxvbmcgcEZvbnQ7Ci0KLSAgICAvLyBjYWNoZWQgbGluZSBtZXRyaWNzIG9iamVjdAotICAgIExpbmVNZXRyaWNzSW1wbCBubG07Ci0KLSAgICAvLyB0aGUgcG9zdHNjcmlwdCBuYW1lIG9mIHRoaXMgZm9udCBwZWVyCi0gICAgU3RyaW5nIHBzTmFtZSA9IG51bGw7Ci0KLSAgICAvKioKLSAgICAgKiBEZWZhdWx0IGdseXBoIGluZGV4LCB0aGF0IGlzIHVzZWQsIHdoZW4gdGhlIGRlc2lyZWQgZ2x5cGgKLSAgICAgKiBpcyB1bnN1cHBvcnRlZCBpbiB0aGlzIEZvbnQuCi0gICAgICovCi0gICAgcHVibGljIGNoYXIgZGVmYXVsdENoYXIgPSAoY2hhcikweEZGRkY7Ci0KLSAgICAvKioKLSAgICAgKiBVbmlmb3JtIExpbmVNZXRyaWNzIGZsYWcsIHRoYXQgaXMgZmFsc2UgZm9yIENvbXBvc2l0ZUZvbnQuICAKLSAgICAgKiBEZWZhdWx0IHZhbHVlIGlzIHRydWUuCi0gICAgICovCi0gICAgYm9vbGVhbiB1bmlmb3JtTE0gPSB0cnVlOwotCi0gICAgLyoqCi0gICAgICogRmxhZyBvZiB0aGUgdHlwZSBvZiB0aGlzIEZvbnQgdGhhdCBpcyBpbmRpY2F0ZSBpcyB0aGUgRm9udAotICAgICAqIGhhcyBUcnVlVHlwZSBvciBUeXBlMSB0eXBlLiBEZWZhdWx0IHZhbHVlIGlzIEZPTlRfVFlQRV9VTkRFRi4gCi0gICAgICovCi0gICAgaW50IGZvbnRUeXBlID0gRm9udE1hbmFnZXIuRk9OVF9UWVBFX1VOREVGOwotCi0gICAgLyoqCi0gICAgICogRmxhZyBpZiB0aGlzIEZvbnQgd2FzIGNyZWF0ZWQgZnJvbSBzdHJlYW0sIAotICAgICAqIHRoaXMgcGFyYW1ldGVyIHVzZWQgaW4gZmluaWxpemUgbWV0aG9kLgotICAgICAqLyAKLSAgICBwcml2YXRlIGJvb2xlYW4gY3JlYXRlZEZyb21TdHJlYW0gPSBmYWxzZTsgIAotICAgIAotICAgIC8vIHRlbW9yYXJ5IEZvbnQgZmlsZSBuYW1lLCBpZiB0aGlzIEZvbnRQZWVySW1wbCB3YXMgY3JlYXRlZCBmcm9tIElucHV0U3RyZWFtIAotICAgIHByaXZhdGUgU3RyaW5nIHRlbXBGb250RmlsZU5hbWUgPSBudWxsOyAgICAgCi0gICAgCi0gICAgLy8gY2FjaGVkIEZvbnRFeHRyYU1ldHJpY3Mgb2JqZWN0IHJlbGF0ZWQgdG8gdGhpcyBmb250IHBlZXIKLSAgICBGb250RXh0cmFNZXRyaWNzIGV4dHJhTWV0cml4ID0gbnVsbDsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCBGb250RXh0cmFNZXRyaWNzIGdldEV4dHJhTWV0cmljcygpOwotICAgIAotICAgIC8qKgotICAgICAqIFJldHVybnMgTGluZU1ldHJpY3Mgb2JqZWN0IHdpdGggc3BlY2lmaWVkIHBhcmFtZXRlcnMKLSAgICAgKiBAcGFyYW0gc3RyIHNwZWNpZmllZCBTdHJpbmcKLSAgICAgKiBAcGFyYW0gZnJjIHNwZWNpZmllZCByZW5kZXIgY29udGV4dAotICAgICAqIEBwYXJhbSBhdCBzcGVjaWZpZWQgYWZmaW5lIHRyYW5zZm9ybQotICAgICAqIEByZXR1cm4KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgTGluZU1ldHJpY3MgZ2V0TGluZU1ldHJpY3MoU3RyaW5nIHN0ciwgRm9udFJlbmRlckNvbnRleHQgZnJjLCBBZmZpbmVUcmFuc2Zvcm0gYXQpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBwb3N0c2NyaXB0IG5hbWUgb2YgdGhlIGZvbnQuICAKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgU3RyaW5nIGdldFBTTmFtZSgpOwotICAgIAotCS8vcHJpdmF0ZSBHcmFwaGljczJEIGcgPSAoKEFuZHJvaWRHcmFwaGljc0ZhY3RvcnkpVG9vbGtpdC5nZXREZWZhdWx0VG9vbGtpdCgpLmdldEdyYXBoaWNzRmFjdG9yeSgpKS5nZXRHcmFwaGljczJEKCk7Ci0gICAgLy9wcml2YXRlIEdyYXBoaWNzMkQgZyA9IEFuZHJvaWRHcmFwaGljczJELmdldEluc3RhbmNlKCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXQgcG9zdHNjcmlwdCBuYW1lIG9mIHRoZSBmb250IHRvIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVyLiAgCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0UFNOYW1lKFN0cmluZyBuYW1lKXsKLSAgICAgICAgdGhpcy5wc05hbWUgPSBuYW1lOwotICAgIH0KLSAgICAKLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGNvZGUgb2YgdGhlIG1pc3NpbmcgZ2x5cGguIAotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0TWlzc2luZ0dseXBoQ29kZSgpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBHbHlwaCByZXByZXNlbnRhdGlvbiBvZiB0aGUgZ2l2ZW4gY2hhci4KLSAgICAgKiBAcGFyYW0gY2ggc3BlY2lmaWVkIGNoYXIKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgR2x5cGggZ2V0R2x5cGgoY2hhciBjaCk7Ci0KLSAgICAvKioKLSAgICAgKiBEaXNwb3NlcyBuZXNlc3NhcnkgcmVzb3VyY2VzLgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGRpc3Bvc2UoKTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgR2x5cGggcmVwcmVzZXRpbmcgbWlzc2luZyBjaGFyLiAKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgR2x5cGggZ2V0RGVmYXVsdEdseXBoKCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhpcyBGb250UGVlckltcGwgY2FuIGRpc3BsYXkgdGhlIHNwZWNpZmllZCBjaGFyCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IGJvb2xlYW4gY2FuRGlzcGxheShjaGFyIGMpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmYW1pbHkgbmFtZSBvZiB0aGUgZm9udCBpbiBzcGVjaWZpZWQgbG9jYWxlIHNldHRpbmdzLgotICAgICAqIEBwYXJhbSBsIHNwZWNpZmllZCBMb2NhbGUKLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldEZhbWlseShMb2NhbGUgbCl7Ci0gICAgICAgIHJldHVybiB0aGlzLmdldEZhbWlseSgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgZmFtaWx5IG5hbWUgb2YgdGhlIGZvbnQgaW4gc3BlY2lmaWVkIGxvY2FsZSBzZXR0aW5ncy4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRGYW1pbHkoU3RyaW5nIGZhbWlseU5hbWUpewotICAgICAgICB0aGlzLmZvbnRGYW1pbHlOYW1lID0gZmFtaWx5TmFtZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGZhY2UgbmFtZSBvZiB0aGUgZm9udCBpbiBzcGVjaWZpZWQgbG9jYWxlIHNldHRpbmdzLgotICAgICAqIEBwYXJhbSBsIHNwZWNpZmllZCBMb2NhbGUKLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldEZvbnROYW1lKExvY2FsZSBsKXsKLSAgICAgICAgcmV0dXJuIHRoaXMuZ2V0Rm9udE5hbWUoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIGZvbnQgbmFtZSBvZiB0aGUgZm9udCBpbiBzcGVjaWZpZWQgbG9jYWxlIHNldHRpbmdzLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEZvbnROYW1lKFN0cmluZyBmb250TmFtZSl7Ci0gICAgICAgIHRoaXMuZmFjZU5hbWUgPSBmb250TmFtZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUsIGlmIHRoaXMgZm9udCBwZWVyIHdhcyBjcmVhdGVkIGZyb20gSW5wdXRTdHJlYW0sIGZhbHNlIG90aGVyd2lzZS4KLSAgICAgKiBJbiBjYXNlIG9mIGNyZWF0aW5nIGZvbnRzIGZyb20gSW5wdXRTdHJlYW0gc29tZSBmb250IHBlZXIgaW1wbGVtZW50YXRpb25zIAotICAgICAqIG1heSBuZWVkIHRvIGZyZWUgdGVtcG9yYXJ5IHJlc291cmNlcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0NyZWF0ZWRGcm9tU3RyZWFtKCl7Ci0gICAgICAgIHJldHVybiB0aGlzLmNyZWF0ZWRGcm9tU3RyZWFtOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgY3JlYXRlZEZyb21TdHJlYW0gZmxhZyB0byB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlci4KLSAgICAgKiBJZiBwYXJhbWV0ZXIgaXMgdHJ1ZSBpdCBtZWFucyBmb250IHBlZXIgd2FzIGNyZWF0ZWQgZnJvbSBJbnB1dFN0cmVhbS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gdmFsdWUgdHJ1ZSwgaWYgZm9udCBwZWVyIHdhcyBjcmVhdGVkIGZyb20gSW5wdXRTdHJlYW0gCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0Q3JlYXRlZEZyb21TdHJlYW0oYm9vbGVhbiB2YWx1ZSl7Ci0gICAgICAgIHRoaXMuY3JlYXRlZEZyb21TdHJlYW0gPSB2YWx1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGZvbnQgZmlsZSBuYW1lIG9mIHRoaXMgZm9udC4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldFRlbXBGb250RmlsZU5hbWUoKXsKLSAgICAgICAgcmV0dXJuIHRoaXMudGVtcEZvbnRGaWxlTmFtZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTZXRzIGZvbnQgZmlsZSBuYW1lIG9mIHRoaXMgZm9udCB0byB0aGUgc3BlY2lmaWVkIG9uZS4KLSAgICAgKiBAcGFyYW0gdmFsdWUgU3RyaW5nIHJlcHJlc2VudGluZyBmb250IGZpbGUgbmFtZQotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEZvbnRGaWxlTmFtZShTdHJpbmcgdmFsdWUpewotICAgICAgICB0aGlzLnRlbXBGb250RmlsZU5hbWUgPSB2YWx1ZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBhZHZhbmNlIHdpZHRoIG9mIHRoZSBzcGVjaWZpZWQgY2hhciBvZiB0aGlzIEZvbnRQZWVySW1wbC4KLSAgICAgKiBOb3RlLCBpZiBnbHlwaCBpcyBhYnNlbnQgaW4gdGhlIGZvbnQncyBnbHlwaHNldCAtIHJldHVybmVkIHZhbHVlIAotICAgICAqIGlzIHRoZSBhZHZhbmNlIG9mIHRoZSBkZWFmdWFsdCBnbHlwaC4gRm9yIGVzY2FwZS1jaGFycyByZXR1cm5lZCAKLSAgICAgKiB3aWR0aCB2YWx1ZSBpcyAwLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaCB0aGUgY2hhciB3aGljaCB3aWR0aCBpcyB0byBiZSByZXR1cm5lZAotICAgICAqIEByZXR1cm4gdGhlIGFkdmFuY2Ugd2lkdGggb2YgdGhlIHNwZWNpZmllZCBjaGFyIG9mIHRoaXMgRm9udFBlZXJJbXBsCi0gICAgICovCi0gICAgcHVibGljIGludCBjaGFyV2lkdGgoY2hhciBjaCkgewotICAgIAlQYWludCBwOwotICAgIAlBbmRyb2lkR3JhcGhpY3MyRCBnID0gQW5kcm9pZEdyYXBoaWNzMkQuZ2V0SW5zdGFuY2UoKTsKLSAgICAJaWYoZyA9PSBudWxsKSB7Ci0gICAgCQl0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbigiQW5kcm9pZEdyYXBoaWNzMkQgbm90IGluc3RhbnRpYXRlZCEiKTsKLSAgICAJfQotICAgCQlwID0gKChBbmRyb2lkR3JhcGhpY3MyRClnKS5nZXRBbmRyb2lkUGFpbnQoKTsKLSAgIAkJY2hhcltdIGNhID0ge2NofTsKLSAgIAkJZmxvYXRbXSBmYSA9IG5ldyBmbG9hdFsxXTsKLSAgIAkJcC5nZXRUZXh0V2lkdGhzKGNhLCAwLCAxLCBmYSk7Ci0gICAJCXJldHVybiAoaW50KWZhWzBdOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGFkdmFuY2Ugd2lkdGggb2YgdGhlIHNwZWNpZmllZCBjaGFyIG9mIHRoaXMgRm9udFBlZXJJbXBsLgotICAgICAqIAotICAgICAqIEBwYXJhbSBpbmQgdGhlIGNoYXIgd2hpY2ggd2lkdGggaXMgdG8gYmUgcmV0dXJuZWQKLSAgICAgKiBAcmV0dXJuIHRoZSBhZHZhbmNlIHdpZHRoIG9mIHRoZSBzcGVjaWZpZWQgY2hhciBvZiB0aGlzIEZvbnRQZWVySW1wbAotICAgICAqLwotICAgIHB1YmxpYyBpbnQgY2hhcldpZHRoKGludCBpbmQpIHsKLSAgICAgICAgcmV0dXJuIGNoYXJXaWR0aCgoY2hhcilpbmQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gYXJyYXkgb2YgR2x5cGhzIHRoYXQgcmVwcmVzZW50IGNoYXJhY3RlcnMgZnJvbSB0aGUgc3BlY2lmaWVkIAotICAgICAqIFVuaWNvZGUgcmFuZ2UuCi0gICAgICogCi0gICAgICogQHBhcmFtIHVGaXJzdCBzdGFydCBwb3NpdGlvbiBpbiBVbmljb2RlIHJhbmdlCi0gICAgICogQHBhcmFtIHVMYXN0IGVuZCBwb3NpdGlvbiBpbiBVbmljb2RlIHJhbmdlCi0gICAgICogQHJldHVybgotICAgICAqLwotICAgIHB1YmxpYyBHbHlwaFtdIGdldEdseXBocyhjaGFyIHVGaXJzdCwgY2hhciB1TGFzdCkgewotCi0gICAgICAgIGNoYXIgaSA9IHVGaXJzdDsKLSAgICAgICAgaW50IGxlbiA9IHVMYXN0IC0gdUZpcnN0OwotICAgICAgICBBcnJheUxpc3Q8R2x5cGg+IGxzdCA9IG5ldyBBcnJheUxpc3Q8R2x5cGg+KGxlbik7Ci0KLSAgICAgICAgaWYgKHNpemUgPCAwKSB7Ci0gICAgICAgICAgICAvLyBhd3QuMDk9bWluIHJhbmdlIGJvdW5kIHZhbHVlIGlzIGdyZWF0ZXIgdGhhbiBtYXggcmFuZ2UgYm91bmQKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMDkiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotCi0gICAgICAgIHdoaWxlIChpIDwgdUxhc3QpIHsKLSAgICAgICAgICAgIGxzdC5hZGQodGhpcy5nZXRHbHlwaChpKSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gKEdseXBoW10pIGxzdC50b0FycmF5KCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhbiBhcnJheSBvZiBHbHlwaHMgcmVwcmVzZW50aW5nIGdpdmVuIGFycmF5IG9mIGNoYXJzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaGFycyBzcGVjaWZpZWQgYXJyYXkgb2YgY2hhcnMKLSAgICAgKi8KLSAgICBwdWJsaWMgR2x5cGhbXSBnZXRHbHlwaHMoY2hhcltdIGNoYXJzKSB7Ci0gICAgICAgIGlmIChjaGFycyA9PSBudWxsKXsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgR2x5cGhbXSByZXN1bHQgPSBuZXcgR2x5cGhbY2hhcnMubGVuZ3RoXTsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGNoYXJzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICByZXN1bHRbaV0gPSB0aGlzLmdldEdseXBoKGNoYXJzW2ldKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcmVzdWx0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gYXJyYXkgb2YgR2x5cGhzIHJlcHJlc2VudGluZyBnaXZlbiBzdHJpbmcuCi0gICAgICogCi0gICAgICogQHBhcmFtIHN0ciBzcGVjaWZpZWQgc3RyaW5nCi0gICAgICovCi0gICAgcHVibGljIEdseXBoW10gZ2V0R2x5cGhzKFN0cmluZyBzdHIpIHsKLSAgICAgICAgY2hhcltdIGNoYXJzID0gc3RyLnRvQ2hhckFycmF5KCk7Ci0gICAgICAgIHJldHVybiB0aGlzLmdldEdseXBocyhjaGFycyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmYW1pbHkgbmFtZSBvZiB0aGlzIEZvbnRQZWVySW1wbC4KLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nIGdldEZhbWlseSgpIHsKLSAgICAgICAgcmV0dXJuIGZvbnRGYW1pbHlOYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgZmFjZSBuYW1lIG9mIHRoaXMgRm9udFBlZXJJbXBsLgotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0Rm9udE5hbWUoKSB7Ci0gICAgICAgIGlmICh0aGlzLmZvbnRUeXBlID09IEZvbnRNYW5hZ2VyLkZPTlRfVFlQRV9UMSl7Ci0gICAgICAgICAgICByZXR1cm4gdGhpcy5mb250RmFtaWx5TmFtZTsKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBmYWNlTmFtZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGhlaWdodCBvZiB0aGlzIGZvbnQgcGVlciBpbiBwaXhlbHMuIAotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TG9naWNhbEhlaWdodCgpIHsKLSAgICAgICAgcmV0dXJuIGxvZ2ljYWxIZWlnaHQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBoZWlnaHQgb2YgdGhpcyBmb250IHBlZXIgaW4gcGl4ZWxzIHRvIHRoZSBnaXZlbiB2YWx1ZS4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbmV3SGVpZ2h0IG5ldyBoZWlnaHQgaW4gcGl4ZWxzIHZhbHVlCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0TG9naWNhbEhlaWdodChpbnQgbmV3SGVpZ2h0KSB7Ci0gICAgICAgIGxvZ2ljYWxIZWlnaHQgPSBuZXdIZWlnaHQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmb250IHNpemUuIAotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0U2l6ZSgpIHsKLSAgICAgICAgcmV0dXJuIHNpemU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmb250IHN0eWxlLiAKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFN0eWxlKCkgewotICAgICAgICByZXR1cm4gc3R5bGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmb250IG5hbWUuIAotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0TmFtZSgpIHsKLSAgICAgICAgcmV0dXJuIG5hbWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgYm91bmRzIG9mIHRoZSBsYXJnZXN0IGNoYXIgaW4gdGhpcyBGb250UGVlckltcGwgaW4gCi0gICAgICogc3BlY2lmaWVkIHJlbmRlciBjb250ZXh0LgotICAgICAqIAotICAgICAqIEBwYXJhbSBmcmMgc3BlY2lmaWVkIEZvbnRSZW5kZXJDb250ZXh0Ci0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGdldE1heENoYXJCb3VuZHMoRm9udFJlbmRlckNvbnRleHQgZnJjKSB7Ci0gICAgICAgIHJldHVybiBtYXhDaGFyQm91bmRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIG51bWJlciBvZiBnbHlwaHMgaW4gdGhpcyBGb250UGVlckltcGwuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXROdW1HbHlwaHMoKSB7Ci0gICAgICAgIHJldHVybiAgbnVtR2x5cGhzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGFuZ2VucyBvZiB0aGUgaXRhbGljIGFuZ2xlIG9mIHRoaXMgRm9udFBlZXJJbXBsLgotICAgICAqIElmIHRoZSBGb250UGVlckltcGwgaGFzIFRydWVUeXBlIGZvbnQgdHlwZSwgaXRhbGljIGFuZ2xlIHZhbHVlIGNhbiBiZSAKLSAgICAgKiBjYWxjdWxhdGVkIGFzIChDaGFyU2xvcGVSdW4gLyBDaGFyU2xvcGVSaXNlKSBpbiB0ZXJtcyBvZiBHREkuCi0gICAgICovCi0gICAgcHVibGljIGZsb2F0IGdldEl0YWxpY0FuZ2xlKCkgewotICAgICAgICByZXR1cm4gaXRhbGljQW5nbGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBoZWlnaHQgb2YgdGhpcyBmb250IHBlZXIuIAotICAgICAqLwotICAgIHB1YmxpYyBmbG9hdCBnZXRIZWlnaHQoKXsKLSAgICAgICAgcmV0dXJuIGhlaWdodDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGNhY2hlZCBMaW5lTWV0cmljcyBvYmplY3Qgb2YgdGhpcyBmb250IHBlZXIuIAotICAgICAqLwotICAgIHB1YmxpYyBMaW5lTWV0cmljcyBnZXRMaW5lTWV0cmljcygpewotICAgICAgICByZXR1cm4gbmxtOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgbmF0aXZlIGZvbnQgaGFuZGxlIG9mIHRoaXMgZm9udCBwZWVyLiAKLSAgICAgKi8KLSAgICBwdWJsaWMgbG9uZyBnZXRGb250SGFuZGxlKCl7Ci0gICAgICAgIHJldHVybiBwRm9udDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGFzY2VudCBvZiB0aGlzIGZvbnQgcGVlci4gCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRBc2NlbnQoKXsKLSAgICAJUGFpbnQgcDsKLSAgICAJQW5kcm9pZEdyYXBoaWNzMkQgZyA9IEFuZHJvaWRHcmFwaGljczJELmdldEluc3RhbmNlKCk7Ci0gICAgCWlmKGcgPT0gbnVsbCkgewotICAgIAkJdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIkFuZHJvaWRHcmFwaGljczJEIG5vdCBpbnN0YW50aWF0ZWQhIik7Ci0gICAgCX0KLSAgIAkJcCA9ICgoQW5kcm9pZEdyYXBoaWNzMkQpZykuZ2V0QW5kcm9pZFBhaW50KCk7Ci0gICAJCXJldHVybiAoaW50KXAuYXNjZW50KCk7Ci0gICAgICAgIC8vcmV0dXJuIGFzY2VudDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGRlc2NlbnQgb2YgdGhpcyBmb250IHBlZXIuIAotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0RGVzY2VudCgpewotICAgICAgICByZXR1cm4gZGVzY2VudDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGxlYWRpbmcgb2YgdGhpcyBmb250IHBlZXIuIAotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TGVhZGluZygpewotICAgICAgICByZXR1cm4gbGVhZGluZzsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhpcyBmb250IHBlZXIgaGFzIHVuaWZvcm0gbGluZSBtZXRyaWNzLiAKLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBoYXNVbmlmb3JtTGluZU1ldHJpY3MoKXsKLSAgICAgICAgcmV0dXJuIHVuaWZvcm1MTTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHR5cGUgb2YgdGhpcyBmb250LgotICAgICAqICAKLSAgICAgKiBAcmV0dXJuIG9uZSBvZiBjb25zdGFudCBmb250IHR5cGUgdmFsdWVzLiAKLSAgICAgKi8gICAgCi0gICAgcHVibGljIGludCBnZXRGb250VHlwZSgpewotICAgICAgICByZXR1cm4gZm9udFR5cGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyBuZXcgZm9udCB0eXBlIHRvIHRoZSBmb250IG9iamVjdC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbmV3VHlwZSBuZXcgdHlwZSB2YWx1ZQotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldEZvbnRUeXBlKGludCBuZXdUeXBlKXsKLSAgICAgICAgaWYgKG5ld1R5cGUgPT0gRm9udE1hbmFnZXIuRk9OVF9UWVBFX1QxIHx8IG5ld1R5cGUgPT0gRm9udE1hbmFnZXIuRk9OVF9UWVBFX1RUKXsKLSAgICAgICAgICAgIGZvbnRUeXBlID0gbmV3VHlwZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgbmV3IGZvbnQgdHlwZSB0byB0aGUgZm9udCBvYmplY3QuCi0gICAgICogCi0gICAgICogQHBhcmFtIG5ld1R5cGUgbmV3IHR5cGUgdmFsdWUKLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgdm9pZCBmaW5hbGl6ZSgpIHRocm93cyBUaHJvd2FibGUgewotICAgICAgc3VwZXIuZmluYWxpemUoKTsKLSAgICAgIAotICAgICAgZGlzcG9zZSgpOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Gb250UHJvcGVydHkuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvRm9udFByb3BlcnR5LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDRlYjdjYmIuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9Gb250UHJvcGVydHkuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEwNiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElseWEgUy4gT2tvbWluCi0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLQotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQ7Ci0KLQotLyoqCi0gKiBDbGFzcyBjb250YWluaW5nIGZvbnQgcHJvcGVydHkgaW5mb3JtYXRpb24uIFRoaXMgaW5mb3JtYXRpb24gY2FuIGJlIGZvdW5kIAotICogaW4gZm9udC5wcm9wZXJ0eSBmaWxlcy4gU2VlIEFQSSBkb2N1bWVudGF0aW9uLCBsb2dpY2FsIGZvbnRzIGRlc2NyaXB0aW9uIHBhcnQuIAotICoKLSAqLwotcHVibGljIGNsYXNzIEZvbnRQcm9wZXJ0eSB7Ci0KLSAgICAvLyBmb250IGZpbGUgbmFtZSAKLSAgICBTdHJpbmcgZmlsZU5hbWUgPSBudWxsOwotICAgIAotICAgIC8vIG5hbWUgb2YgdGhlIGVuY29kaW5nIHRvIGJlIHVzZWQgCi0gICAgU3RyaW5nIGVuY29kaW5nID0gbnVsbDsKLSAgICAKLSAgICAvLyBhcnJheSBvZiBleGNsdXNpb24gcmFuZ2VzIChwYWlycyBvZiBsb3cgYW5kIGhpZ2ggdW5pY29kZSBleGNsdXNpb24gYm91bmRzKQotICAgIGludFtdIGV4Y2xSYW5nZSA9IG51bGw7Ci0gICAgCi0gICAgLy8gZm9udCBmYWNlIG5hbWUKLSAgICBTdHJpbmcgbmFtZSA9IG51bGw7Ci0gICAgCi0gICAgLy8gZm9udCBzdHlsZQotICAgIGludCBzdHlsZSA9IC0xOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBmb250IHN0eWxlIG9mIHRoaXMgZm9udCBwcm9wZXJ0eS4gCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRTdHlsZSgpewotICAgICAgICByZXR1cm4gdGhpcy5zdHlsZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGZvbnQgbmFtZSBvZiB0aGlzIGZvbnQgcHJvcGVydHkuIAotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0TmFtZSgpewotICAgICAgICByZXR1cm4gdGhpcy5uYW1lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgZW5jb2RpbmcgdXNlZCBpbiB0aGlzIGZvbnQgcHJvcGVydHkuIAotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RW5jb2RpbmcoKXsKLSAgICAgICAgcmV0dXJuIHRoaXMuZW5jb2Rpbmc7Ci0gICAgfQotICAgIAotICAgIC8qKgotICAgICAqIFJldHVybnMgYW4gYXJyYXkgb2YgZXhjbHVzaW9uIHJhbmdlcy4gVGhpcyBhcnJheSBjb250YWluIHBhaXJzIG9mIAotICAgICAqIGxvdyBhbmQgaGlnaCBib3VuZHMgb2YgdGhlIGludGVydmFscyBvZiBjaGFyYWN0ZXJzIHRvIGlnbm9yZSBpbiAKLSAgICAgKiB0b3RhbCBVbmljb2RlIGNoYXJhY3RlcnMgcmFuZ2UuICAgCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldEV4Y2x1c2lvblJhbmdlKCl7Ci0gICAgICAgIHJldHVybiB0aGlzLmV4Y2xSYW5nZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGZpbGUgbmFtZSBvZiB0aGUgZm9udCB0aGF0IGlzIGRlc2NyaWJlZCBieSB0aGlzIGZvbnQgcHJvcGVydHkuIAotICAgICAqLwotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RmlsZU5hbWUoKXsKLSAgICAgICAgcmV0dXJuIHRoaXMuZmlsZU5hbWU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0cnVlIGlmIHNwZWNpZmllZCBjaGFyYWN0ZXIgY292ZXJlZCBieSBleGNsdXNpb24gcmFuZ2VzIG9mIHRoaXMgCi0gICAgICogZm9udCBwcm9wZXJ0eSwgZmFsc2Ugb3RoZXJ3aXNlLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaCBzcGVjaWZpZWQgY2hhciB0byBjaGVjawotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGlzQ2hhckV4Y2x1ZGVkKGNoYXIgY2gpewotICAgICAgICBpZiAoZXhjbFJhbmdlID09IG51bGwgKXsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgZXhjbFJhbmdlLmxlbmd0aDspewotICAgICAgICAgICAgaW50IGxiID0gZXhjbFJhbmdlW2krK107Ci0gICAgICAgICAgICBpbnQgaGIgPSBleGNsUmFuZ2VbaSsrXTsKLQotICAgICAgICAgICAgaWYgKGNoID49IGxiICYmIGNoIDw9IGhiKXsKLSAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0dseXBoLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0dseXBoLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQ0Yjg4MDkuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9HbHlwaC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjM2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5TaGFwZTsKLWltcG9ydCBqYXZhLmF3dC5mb250LkdseXBoSnVzdGlmaWNhdGlvbkluZm87Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5HbHlwaE1ldHJpY3M7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLQotcHVibGljIGFic3RyYWN0IGNsYXNzIEdseXBoewotCi0gICAgLy8gY2hhcmFjdGVyIG9mIHRoZSBnbHlwaAotICAgIGNoYXIgZ2xDaGFyOwotICAgIAotICAgIC8vIHByZWNpc2UgZ2x5cGggbWV0cmljcwotICAgIEdseXBoTWV0cmljcyBnbE1ldHJpY3M7Ci0gICAgCi0gICAgLy8gZ2x5cGggbWV0cmljcyBpbiBwaXhlbHMKLSAgICBHbHlwaE1ldHJpY3MgZ2xQb2ludE1ldHJpY3M7Ci0gICAgCi0gICAgLy8gIGdseXBoIGNvZGUgb2YgdGhpcyBHbHlwaAotICAgIGludCBnbENvZGU7Ci0gICAgCi0gICAgLy8ganVzdGlmaWNhdGlvbiBpbmZvIG9mIHRoaXMgZ2x5cGgKLSAgICBHbHlwaEp1c3RpZmljYXRpb25JbmZvIGdsSnVzdEluZm87Ci0gICAgCi0gICAgLy8gbmF0aXZlIGZvbnQgaGFuZGxlIG9mIHRoZSBmb250IGNvcnJlc3BvbmRpbmcgdG8gdGhpcyBnbHlwaAotICAgIGxvbmcgcEZvbnQ7Ci0gICAgCi0gICAgLy8gc2l6ZSBvZiB0aGUgZm9udCBjb3JyZXNwb25kaW5nIHRvIHRoaXMgZ2x5cGgKLSAgICBpbnQgZm9udFNpemU7Ci0gICAgCi0gICAgLy8gYml0bWFwIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBnbHlwaAotICAgIGJ5dGVbXSBiaXRtYXAgPSBudWxsOwotICAgIAotICAgIC8vIEJ1ZmZlcmVkIGltYWdlIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBnbHlwaAotICAgIEJ1ZmZlcmVkSW1hZ2UgaW1hZ2U7Ci0gICAgCi0gICAgLy8gc2hhcGUgdGhhdCByZXByZXNlbnRpbmcgdGhlIG91dGxpbmUgb2YgdGhpcyBnbHlwaAotICAgIFNoYXBlIGdsT3V0bGluZSA9IG51bGw7Ci0KLSAgICAvKioKLSAgICAgKiBpbWFnZSBiaXRtYXAgcGFyYW1ldGVycwotICAgICAqLwotICAgIAotICAgIC8vICB0b3Agc2lkZSBiZWFyaW5nCi0gICAgcHVibGljIGludCBibXBfdG9wID0gMDsKLSAgICAKLSAgICAvLyBsZWZ0IHNpZGUgYmVhcmluZwotICAgIHB1YmxpYyBpbnQgYm1wX2xlZnQgPSAwOwotCi0gICAgLy8gbnVtYmVyIG9mIGJ5dGVzIGluIHJvdwotICAgIHB1YmxpYyBpbnQgYm1wX3BpdGNoOwotICAgIAotICAgIC8vIG51bWJlciBvZiByb3dzCi0gICAgcHVibGljIGludCBibXBfcm93czsKLSAgICAKLSAgICAvLyB3aWR0aCBvZiB0aGUgcm93Ci0gICAgcHVibGljIGludCBibXBfd2lkdGg7Ci0KLSAgICAvKioKLSAgICAgKiAgUmV0cnVucyBoYW5kbGUgdG8gTmF0aXZlIEZvbnQgb2JqZWN0Ci0gICAgICovCi0gICAgcHVibGljIGxvbmcgZ2V0UEZvbnQoKXsKLSAgICAgICAgcmV0dXJuIHRoaXMucEZvbnQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogIFJldHJ1bnMgY2hhciB2YWx1ZSBvZiB0aGlzIGdseXBoIG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBjaGFyIGdldENoYXIoKXsKLSAgICAgICAgcmV0dXJuIGdsQ2hhcjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiAgUmV0cnVucyBwcmVjaXNlIHdpZHRoIG9mIHRoaXMgZ2x5cGggb2JqZWN0Ci0gICAgICovCi0gICAgcHVibGljIGludCBnZXRXaWR0aCgpewotICAgICAgICByZXR1cm4gTWF0aC5yb3VuZCgoZmxvYXQpZ2xNZXRyaWNzLmdldEJvdW5kczJEKCkuZ2V0V2lkdGgoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogIFJldHJ1bnMgcHJlY2lzZSBoZWlnaHQgb2YgdGhpcyBnbHlwaCBvYmplY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEhlaWdodCgpewotICAgICAgICByZXR1cm4gTWF0aC5yb3VuZCgoZmxvYXQpZ2xNZXRyaWNzLmdldEJvdW5kczJEKCkuZ2V0SGVpZ2h0KCkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqICBSZXRydW5zIGdseXBoIGNvZGUgb2YgdGhpcyBnbHlwaCBvYmplY3QKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldEdseXBoQ29kZSgpewotICAgICAgICByZXR1cm4gZ2xDb2RlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqICBSZXRydW5zIEdseXBoTWV0cmljcyBvZiB0aGlzIGdseXBoIG9iamVjdCB3aXRoIHByZWNpc2UgbWV0cmljcy4KLSAgICAgKi8KLSAgICBwdWJsaWMgR2x5cGhNZXRyaWNzIGdldEdseXBoTWV0cmljcygpewotICAgICAgICByZXR1cm4gZ2xNZXRyaWNzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqICBSZXRydW5zIEdseXBoTWV0cmljcyBvZiB0aGlzIGdseXBoIG9iamVjdCBpbiBwaXhlbHMuCi0gICAgICovCi0gICAgcHVibGljIEdseXBoTWV0cmljcyBnZXRHbHlwaFBvaW50TWV0cmljcygpewotICAgICAgICByZXR1cm4gZ2xQb2ludE1ldHJpY3M7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogIFJldHJ1bnMgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBvZiB0aGlzIGdseXBoIG9iamVjdAotICAgICAqLwotICAgIHB1YmxpYyBHbHlwaEp1c3RpZmljYXRpb25JbmZvIGdldEdseXBoSnVzdGlmaWNhdGlvbkluZm8oKXsKLSAgICAgICAgcmV0dXJuIGdsSnVzdEluZm87Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogIFNldHMgSnVzdGlmaWNhdGlvbkluZm8gb2YgdGhpcyBnbHlwaCBvYmplY3QKLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbmV3SnVzdEluZm8gR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBvYmplY3QgdG8gc2V0IHRvIHRoZSBHbHlwaCBvYmplY3QgCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0R2x5cGhKdXN0aWZpY2F0aW9uSW5mbyhHbHlwaEp1c3RpZmljYXRpb25JbmZvIG5ld0p1c3RJbmZvKXsKLSAgICAgICAgdGhpcy5nbEp1c3RJbmZvID0gbmV3SnVzdEluZm87Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhbiBpbnQgYXJyYXkgb2YgMyBlbGVtZW50cywgc28tY2FsbGVkIEFCQyBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucyAKLSAgICAgKiB0aGUgd2lkdGggb2YgdGhlIGNoYXJhY3RlcjoKLSAgICAgKiAxc3QgZWxlbWVudCA9IGxlZnQgc2lkZSBiZWFyaW5nIG9mIHRoZSBnbHlwaAotICAgICAqIDJuZCBlbGVtZW50ID0gd2lkdGggb2YgdGhlIGdseXBoCi0gICAgICogM2QgZWxlbWVudCA9IHJpZ2h0IHNpZGUgYmVhcmluZyBvZiB0aGUgZ2x5cGggCi0gICAgICovCi0gICAgcHVibGljIGludFtdIGdldEFCQygpewotICAgICAgICBpbnRbXSBhYmMgPSBuZXcgaW50WzNdOwotICAgICAgICBhYmNbMF0gPSAoaW50KWdsTWV0cmljcy5nZXRMU0IoKTsKLSAgICAgICAgYWJjWzFdID0gKGludClnbE1ldHJpY3MuZ2V0Qm91bmRzMkQoKS5nZXRXaWR0aCgpOwotICAgICAgICBhYmNbMl0gPSAoaW50KWdsTWV0cmljcy5nZXRSU0IoKTsKLQotICAgICAgICByZXR1cm4gYWJjOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgQnVmZmVyZWRJbWFnZSByZXByZXNlbnRhdGlvbiBvZiB0aGlzIGdseXBoIHRvIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVyLgotICAgICAqIAotICAgICAqIEBwYXJhbSBuZXdJbWFnZSBuZXcgQnVmZmVyZWRJbWFnZSBvYmplY3QgdG8gYmUgc2V0IGFzIEJ1ZmZlcmVkSW1hZ2UgCi0gICAgICogcmVwcmVzZW50YXRpb24uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0SW1hZ2UoQnVmZmVyZWRJbWFnZSBuZXdJbWFnZSl7Ci0gICAgICAgIHRoaXMuaW1hZ2UgPSBuZXdJbWFnZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgdGhpcyBHbHlwaCBhbmQgc3BlY2lmaWVkIG9iamVjdCBhcmUgZXF1YWwuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvYmopewotICAgICAgICAgaWYgKG9iaiA9PSB0aGlzKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChvYmogIT0gbnVsbCkgewotICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBHbHlwaCBnbCA9IChHbHlwaClvYmo7Ci0KLSAgICAgICAgICAgIHJldHVybiAgKCh0aGlzLmdldENoYXIoKSA9PSBnbC5nZXRDaGFyKCkpCi0gICAgICAgICAgICAgICYmICh0aGlzLmdldEdseXBoTWV0cmljcygpLmVxdWFscyhnbC5nZXRHbHlwaE1ldHJpY3MoKSkpCi0gICAgICAgICAgICAgICYmICh0aGlzLmdldEdseXBoQ29kZSgpID09IGdsLmdldEdseXBoQ29kZSgpKSk7Ci0gICAgICAgICAgfSBjYXRjaCAoQ2xhc3NDYXN0RXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBoZWlnaHQgb2YgdGhlIGdseXBoIGluIHBvaW50cy4gCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRQb2ludEhlaWdodCgpewotICAgICAgICByZXR1cm4gKGludClnbFBvaW50TWV0cmljcy5nZXRCb3VuZHMyRCgpLmdldEhlaWdodCgpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgd2lkdGggb2YgdGhlIGdseXBoIGluIHBvaW50cy4gCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRQb2ludFdpZHRoKCl7Ci0gICAgICAgIHJldHVybiAoaW50KWdsUG9pbnRNZXRyaWNzLmdldEJvdW5kczJEKCkuZ2V0V2lkdGgoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU2hhcGUgZ2V0U2hhcGUoKXsKLSAgICAgICAgaWYgKGdsT3V0bGluZSA9PSBudWxsKXsKLSAgICAgICAgICAgIGdsT3V0bGluZSA9IGluaXRPdXRsaW5lKHRoaXMuZ2xDaGFyKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZ2xPdXRsaW5lOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldHMgQnVmZmVyZWRJbWFnZSByZXByZXNlbnRhdGlvbiBvZiB0aGlzIGdseXBoLgotICAgICAqLwotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlIGdldEltYWdlKCl7Ci0gICAgICAgIC8vISEgSW1wbGVtZW50YXRpb24gY2xhc3NlcyBtdXN0IG92ZXJyaWRlIHRoaXMgbWV0aG9kCi0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIC8qKgotICAgICAqICBSZXR1cm5zIGFycmF5IG9mIGJ5dGVzLCByZXByZXNlbnRpbmcgaW1hZ2Ugb2YgdGhpcyBnbHlwaAotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBieXRlW10gZ2V0Qml0bWFwKCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHNoYXBlIHRoYXQgcmVwcmVzZW50cyBvdXRsaW5lIG9mIHRoZSBzcGVjaWZpZWQgY2hhcmFjdGVyLiAKLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gYyBzcGVjaWZpZWQgY2hhcmFjdGVyCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFNoYXBlIGluaXRPdXRsaW5lKGNoYXIgYyk7Ci0KLX0KLQotCmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0xpbmVNZXRyaWNzSW1wbC5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9MaW5lTWV0cmljc0ltcGwuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzcwMTQ2ZC4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L0xpbmVNZXRyaWNzSW1wbC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDEyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQ7Ci0KLWltcG9ydCBqYXZhLmF3dC5mb250LkxpbmVNZXRyaWNzOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICoKLSAqIExpbmVNZXRyaWNzIGltcGxlbWVudGF0aW9uIGNsYXNzLgotICovCi0KLXB1YmxpYyBjbGFzcyBMaW5lTWV0cmljc0ltcGwgZXh0ZW5kcyBMaW5lTWV0cmljcyBpbXBsZW1lbnRzIENsb25lYWJsZXsKLQotICAgIC8vIGFycmF5IG9mIGJhc2VsaW5lIG9mZnNldHMKLSAgICBmbG9hdFtdIGJhc2VsaW5lT2Zmc2V0czsKLQotICAgIC8vIHRoZSBudW1iZXIgb2YgY2hhcmFjdGVycyB0byBtZWFzdXJlCi0gICAgaW50IG51bUNoYXJzOwotCi0gICAgLy8gYmFzZWxpbmUgaW5kZXggb2YgdGhlIGZvbnQgY29ycmVzcG9uZGluZyB0byB0aGlzIGxpbmUgbWV0cmljcwotICAgIGludCBiYXNlTGluZUluZGV4OwotCi0gICAgLy8gdW5kZXJsaW5lIHRoaWNrbmVzcwotICAgIGZsb2F0IHVuZGVybGluZVRoaWNrbmVzczsKLQotICAgIC8vIHVuZGVybGluZSBvZmZzZXQKLSAgICBmbG9hdCB1bmRlcmxpbmVPZmZzZXQ7Ci0KLSAgICAvLyBzdHJpa2V0aHJvdWdoIHRoaWNrbmVzcwotICAgIGZsb2F0IHN0cmlrZXRocm91Z2hUaGlja25lc3M7Ci0KLSAgICAvLyBzdHJpa2V0aHJvdWdoIG9mZnNldAotICAgIGZsb2F0IHN0cmlrZXRocm91Z2hPZmZzZXQ7Ci0KLSAgICAvLyBFeHRlcm5hbCBsZWFkaW5nCi0gICAgZmxvYXQgbGVhZGluZzsKLQotICAgIC8vIEhlaWdodCBvZiB0aGUgZm9udCAoID09IChhc2NlbnQrZGVzY2VudCtsZWFkaW5nKSkKLSAgICBmbG9hdCBoZWlnaHQ7Ci0KLSAgICAvLyBBc2NlbnQgb2YgdGhlIGZvbnQKLSAgICBmbG9hdCBhc2NlbnQ7Ci0KLSAgICAvLyBEZXNjZW50IG9mIHRoZSBmb250Ci0gICAgZmxvYXQgZGVzY2VudDsKLQotICAgIC8vIFdpZHRoIG9mIHRoZSB3aWRlc3QgY2hhciBpbiB0aGUgZm9udAotICAgIGZsb2F0IG1heENoYXJXaWR0aDsKLQotICAgIC8vIHVuZGVybGluZSB0aGlja25lc3MgKGluIHBpeGVscykKLSAgICBpbnQgbFVuZGVybGluZVRoaWNrbmVzczsKLQotICAgIC8vIHVuZGVybGluZSBvZmZzZXQgKGluIHBpeGVscykKLSAgICBpbnQgbFVuZGVybGluZU9mZnNldDsKLQotICAgIC8vIHN0cmlrZXRocm91Z2ggdGhpY2tuZXNzIChpbiBwaXhlbHMpCi0gICAgaW50IGxTdHJpa2V0aHJvdWdoVGhpY2tuZXNzOwotCi0gICAgLy8gc3RyaWtldGhyb3VnaCBvZmZzZXQgKGluIHBpeGVscykKLSAgICBpbnQgbFN0cmlrZXRocm91Z2hPZmZzZXQ7Ci0KLSAgICAvLyBFeHRlcm5hbCBsZWFkaW5nIChpbiBwaXhlbHMpCi0gICAgaW50IGxMZWFkaW5nOwotCi0gICAgLy8gSGVpZ2h0IG9mIHRoZSBmb250ICggPT0gKGFzY2VudCtkZXNjZW50K2xlYWRpbmcpKSAoaW4gcGl4ZWxzKQotICAgIGludCBsSGVpZ2h0OwotCi0gICAgLy8gQXNjZW50IG9mIHRoZSBmb250IChpbiBwaXhlbHMpCi0gICAgaW50IGxBc2NlbnQ7Ci0gICAgCi0gICAgLy8gRGVzY2VudCBvZiB0aGUgZm9udCAoaW4gcGl4ZWxzKQotICAgIGludCBsRGVzY2VudDsKLQotICAgIC8vICBXaWR0aCBvZiB0aGUgd2lkZXN0IGNoYXIgaW4gdGhlIGZvbnQgKGluIHBpeGVscykKLSAgICBpbnQgbE1heENoYXJXaWR0aDsKLQotICAgIC8vIHVuaXRzIHBlciBFTSBzcXVhcmUgaW4gZm9udCB2YWx1ZQotICAgIGludCB1bml0c19wZXJfRU0gPSAwOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBMaW5lTWV0cmljc0ltcGwgb2JqZWN0IGZyb20gc3BlY2lmaWVkIHBhcmFtZXRlcnMuIElmIGJhc2VsaW5lIGRhdGEgcGFyYW1ldGVyCi0gICAgICogaXMgbnVsbCB0aGFuIHswLCAoLWFzY2VudCtkZXNjZW50KS8yLCAtYXNjZW50fSB2YWx1ZXMgYXJlIHVzZWQgZm9yIGJhc2VsaW5lIG9mZnNldHMuCi0gICAgICogIAotICAgICAqIEBwYXJhbSBsZW4gYSBudW1iZXIgb2YgY2hhcmFjdGVycyAKLSAgICAgKiBAcGFyYW0gbWV0cmljcyBhbiBhcnJheSBvZiAxNiBlbGVtZW50cyB3aXRoIG1ldHJpY3MgdmFsdWVzIHRoYXQgY2FuIGJlIAotICAgICAqIGluaXRpYWxpemVkIGluIG5hdGl2ZSBjb2RlLjxwPgotICAgICAqIG1ldHJpY3NbMF0gLSBhc2NlbnQ8cD4KLSAgICAgKiBtZXRyaWNzWzFdIC0gZGVzY2VudDxwPgotICAgICAqIG1ldHJpY3NbMl0gLSBleHRlcm5hbCBsZWFkaW5nPHA+Ci0gICAgICogbWV0cmljc1szXSAtIHVuZGVybGluZSB0aGlja25lc3M8cD4KLSAgICAgKiAtbWV0cmljc1s0XSAtIHVuZGVybGluZSBvZmZzZXQ8cD4KLSAgICAgKiBtZXRyaWNzWzVdIC0gc3RyaWtldGhyb3VnaCB0aGlja25lc3M8cD4KLSAgICAgKiAtbWV0cmljc1s2XSAtIHN0cmlrZXRocm91Z2ggb2Zmc2V0PHA+Ci0gICAgICogbWV0cmljc1s3XSAtIG1heGltdW0gY2hhciB3aWR0aDxwPgotICAgICAqIG1ldHJpY3NbOF0gLSBhc2NlbnQgaW4gcGl4ZWxzPHA+Ci0gICAgICogbWV0cmljc1s5XSAtIGRlc2NlbnQgaW4gcGl4bGVzPHA+Ci0gICAgICogbWV0cmljc1sxMF0gLSBleHRlcm5hbCBsZWFkaW5nIGluIHBpeGVsczxwPgotICAgICAqIG1ldHJpY3NbMTFdIC0gdW5kZXJsaW5lIHRoaWNrbmVzcyBpbiBwaXhlbHM8cD4KLSAgICAgKiAtbWV0cmljc1sxMl0gLSB1bmRlcmxpbmUgb2Zmc2V0IGluIHBpeGVsczxwPgotICAgICAqIG1ldHJpY3NbMTNdIC0gc3RyaWtldGhyb3VnaCB0aGlja25lc3MgaW4gcGl4ZWxzPHA+Ci0gICAgICogLW1ldHJpY3NbMTRdIC0gc3RyaWtldGhyb3VnaCBvZmZzZXQgaW4gcGl4ZWxzPHA+Ci0gICAgICogbWV0cmljc1sxNV0gLSBtYXhpbXVtIGNoYXIgd2lkdGggaW4gcGl4ZWxzPHA+Ci0KLSAgICAgKiBAcGFyYW0gX2Jhc2VsaW5lRGF0YSBhbiBhcnJheSBvZiAzIGVsZW1lbnRzIHdpdGggYmFzZWxpbmUgb2Zmc2V0cyBtZXRyaWNzPHA+Ci0gICAgICogX2Jhc2VsaW5lRGF0YVswXSAtIHJvbWFuIGJhc2VsaW5lIG9mZnNldDxwPiAKLSAgICAgKiBfYmFzZWxpbmVEYXRhWzFdIC0gY2VudGVyIGJhc2VsaW5lIG9mZnNldDxwPgotICAgICAqIF9iYXNlbGluZURhdGFbMl0gLSBoYW5naW5nIGJhc2VsaW5lIG9mZnNldDxwPgotICAgICAqLwotICAgIHB1YmxpYyBMaW5lTWV0cmljc0ltcGwoaW50IGxlbiwgZmxvYXRbXSBtZXRyaWNzLCBmbG9hdFtdIF9iYXNlbGluZURhdGEpewotICAgICAgICBudW1DaGFycyA9IGxlbjsKLQotICAgICAgICBhc2NlbnQgPSBtZXRyaWNzWzBdOyAgICAvLyBBc2NlbnQgb2YgdGhlIGZvbnQKLSAgICAgICAgZGVzY2VudCA9IG1ldHJpY3NbMV07ICAgLy8gRGVzY2VudCBvZiB0aGUgZm9udAotICAgICAgICBsZWFkaW5nID0gbWV0cmljc1syXTsgIC8vIEV4dGVybmFsIGxlYWRpbmcKLSAgICAgICAgaGVpZ2h0ID0gbWV0cmljc1swXSArIG1ldHJpY3NbMV0gKyBtZXRyaWNzWzJdOyAgLy8gSGVpZ2h0IG9mIHRoZSBmb250ICggPT0gKGFzY2VudCArIGRlc2NlbnQgKyBsZWFkaW5nKSkKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIExpbmVNZXRyaWNzSW1wbCBvYmplY3QgZnJvbSBzcGVjaWZpZWQgcGFyYW1ldGVycy4gSWYgYmFzZWxpbmUgZGF0YSBwYXJhbWV0ZXIKLSAgICAgKiBpcyBudWxsIHRoYW4gezAsICgtYXNjZW50K2Rlc2NlbnQpLzIsIC1hc2NlbnR9IHZhbHVlcyBhcmUgdXNlZCBmb3IgYmFzZWxpbmUgb2Zmc2V0cy4KLSAgICAgKiAgCi0gICAgICogQHBhcmFtIF9udW1DaGFycyBudW1iZXIgb2YgY2hhcnMgCi0gICAgICogQHBhcmFtIF9iYXNlTGluZUluZGV4IGluZGV4IG9mIHRoZSBiYXNlbGluZSBvZmZzZXQKLSAgICAgKiBAcGFyYW0gX2Jhc2VsaW5lT2Zmc2V0cyBhbiBhcnJheSBvZiBiYXNlbGluZSBvZmZzZXRzCi0gICAgICogQHBhcmFtIF91bmRlcmxpbmVUaGlja25lc3MgdW5kZXJsaW5lIHRoaWNrbmVzcwotICAgICAqIEBwYXJhbSBfdW5kZXJsaW5lT2Zmc2V0IHVuZGVybGluZSBvZmZzZXQKLSAgICAgKiBAcGFyYW0gX3N0cmlrZXRocm91Z2hUaGlja25lc3Mgc3RyaWtldGhyb3VnaCB0aGlja25lc3MKLSAgICAgKiBAcGFyYW0gX3N0cmlrZXRocm91Z2hPZmZzZXQgc3RyaW5rZXRocm91Z2ggb2Zmc2V0Ci0gICAgICogQHBhcmFtIF9sZWFkaW5nIGxlYWRpbmcgb2YgdGhlIGZvbnQKLSAgICAgKiBAcGFyYW0gX2hlaWdodCBmb250IGhlaWdodAotICAgICAqIEBwYXJhbSBfYXNjZW50IGFzY2VudCBvZiB0aGUgZm9udAotICAgICAqIEBwYXJhbSBfZGVzY2VudCBkZXNjZW50IG9mIHRoZSBmb250Ci0gICAgICogQHBhcmFtIF9tYXhDaGFyV2lkdGggbWF4IGNoYXIgd2lkdGgKLSAgICAgKi8KLSAgICBwdWJsaWMgTGluZU1ldHJpY3NJbXBsKGludCBfbnVtQ2hhcnMsIGludCBfYmFzZUxpbmVJbmRleCwKLSAgICAgICAgICAgIGZsb2F0W10gX2Jhc2VsaW5lT2Zmc2V0cywgZmxvYXQgX3VuZGVybGluZVRoaWNrbmVzcywKLSAgICAgICAgICAgIGZsb2F0IF91bmRlcmxpbmVPZmZzZXQsIGZsb2F0IF9zdHJpa2V0aHJvdWdoVGhpY2tuZXNzLAotICAgICAgICAgICAgZmxvYXQgX3N0cmlrZXRocm91Z2hPZmZzZXQsIGZsb2F0IF9sZWFkaW5nLCBmbG9hdCBfaGVpZ2h0LAotICAgICAgICAgICAgZmxvYXQgX2FzY2VudCwgZmxvYXQgX2Rlc2NlbnQsIGZsb2F0IF9tYXhDaGFyV2lkdGgpIHsKLQotICAgICAgICBudW1DaGFycyA9IF9udW1DaGFyczsKLSAgICAgICAgYmFzZUxpbmVJbmRleCA9IF9iYXNlTGluZUluZGV4OwotICAgICAgICB1bmRlcmxpbmVUaGlja25lc3MgPSBfdW5kZXJsaW5lVGhpY2tuZXNzOwotICAgICAgICB1bmRlcmxpbmVPZmZzZXQgPSBfdW5kZXJsaW5lT2Zmc2V0OwotICAgICAgICBzdHJpa2V0aHJvdWdoVGhpY2tuZXNzID0gX3N0cmlrZXRocm91Z2hUaGlja25lc3M7Ci0gICAgICAgIHN0cmlrZXRocm91Z2hPZmZzZXQgPSBfc3RyaWtldGhyb3VnaE9mZnNldDsKLSAgICAgICAgbGVhZGluZyA9IF9sZWFkaW5nOwotICAgICAgICBoZWlnaHQgPSBfaGVpZ2h0OwotICAgICAgICBhc2NlbnQgPSBfYXNjZW50OwotICAgICAgICBkZXNjZW50ID0gX2Rlc2NlbnQ7Ci0gICAgICAgIGJhc2VsaW5lT2Zmc2V0cyA9IF9iYXNlbGluZU9mZnNldHM7Ci0gICAgICAgIGxVbmRlcmxpbmVUaGlja25lc3MgPSAoaW50KSB1bmRlcmxpbmVUaGlja25lc3M7Ci0gICAgICAgIGxVbmRlcmxpbmVPZmZzZXQgPSAoaW50KSB1bmRlcmxpbmVPZmZzZXQ7Ci0gICAgICAgIGxTdHJpa2V0aHJvdWdoVGhpY2tuZXNzID0gKGludCkgc3RyaWtldGhyb3VnaFRoaWNrbmVzczsKLSAgICAgICAgbFN0cmlrZXRocm91Z2hPZmZzZXQgPSAoaW50KSBzdHJpa2V0aHJvdWdoT2Zmc2V0OwotICAgICAgICBsTGVhZGluZyA9IChpbnQpIGxlYWRpbmc7Ci0gICAgICAgIGxIZWlnaHQgPSAoaW50KSBoZWlnaHQ7Ci0gICAgICAgIGxBc2NlbnQgPSAoaW50KSBhc2NlbnQ7Ci0gICAgICAgIGxEZXNjZW50ID0gKGludCkgZGVzY2VudDsKLSAgICAgICAgbWF4Q2hhcldpZHRoID0gX21heENoYXJXaWR0aDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgTGluZU1ldHJpY3NJbXBsKCl7Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBbGwgbWV0cmljcyBhcmUgc2NhbGVkIGFjY29yZGluZyB0byBzY2FsZVggYW5kIHNjYWxlWSB2YWx1ZXMuIAotICAgICAqIFRoaXMgZnVuY3Rpb24gaGVscHMgdG8gcmVjb21wdXRlIG1ldHJpY3MgYWNjb3JkaW5nIHRvIHRoZSBzY2FsZSBmYWN0b3JzCi0gICAgICogb2YgZGVzaXJlZCBBZmZpbmVUcmFuc2Zvcm0uCi0gICAgICogCi0gICAgICogQHBhcmFtIHNjYWxlWCBzY2FsZSBYIGZhY3RvcgotICAgICAqIEBwYXJhbSBzY2FsZVkgc2NhbGUgWSBmYWN0b3IKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzY2FsZShmbG9hdCBzY2FsZVgsIGZsb2F0IHNjYWxlWSl7Ci0gICAgICAgIGZsb2F0IGFic1NjYWxlWCA9IE1hdGguYWJzKHNjYWxlWCk7Ci0gICAgICAgIGZsb2F0IGFic1NjYWxlWSA9IE1hdGguYWJzKHNjYWxlWSk7Ci0KLSAgICAgICAgdW5kZXJsaW5lVGhpY2tuZXNzICo9IGFic1NjYWxlWTsKLSAgICAgICAgdW5kZXJsaW5lT2Zmc2V0ICo9IHNjYWxlWTsKLSAgICAgICAgc3RyaWtldGhyb3VnaFRoaWNrbmVzcyAqPSBhYnNTY2FsZVk7Ci0gICAgICAgIHN0cmlrZXRocm91Z2hPZmZzZXQgKj0gc2NhbGVZOwotICAgICAgICBsZWFkaW5nICo9IGFic1NjYWxlWTsKLSAgICAgICAgaGVpZ2h0ICo9IGFic1NjYWxlWTsKLSAgICAgICAgYXNjZW50ICo9IGFic1NjYWxlWTsKLSAgICAgICAgZGVzY2VudCAqPSBhYnNTY2FsZVk7Ci0KLSAgICAgICAgaWYoYmFzZWxpbmVPZmZzZXRzID09IG51bGwpIHsKLSAgICAgICAgICAgIGdldEJhc2VsaW5lT2Zmc2V0cygpOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChpbnQgaT0wOyBpPCBiYXNlbGluZU9mZnNldHMubGVuZ3RoOyBpKyspewotICAgICAgICAgICAgYmFzZWxpbmVPZmZzZXRzW2ldICo9IHNjYWxlWTsKLSAgICAgICAgfQotCi0gICAgICAgIGxVbmRlcmxpbmVUaGlja25lc3MgKj0gYWJzU2NhbGVZOwotICAgICAgICBsVW5kZXJsaW5lT2Zmc2V0ICo9IHNjYWxlWTsKLSAgICAgICAgbFN0cmlrZXRocm91Z2hUaGlja25lc3MgKj0gYWJzU2NhbGVZOwotICAgICAgICBsU3RyaWtldGhyb3VnaE9mZnNldCAqPSBzY2FsZVk7Ci0gICAgICAgIGxMZWFkaW5nICAqPSBhYnNTY2FsZVk7Ci0gICAgICAgIGxIZWlnaHQgKj0gYWJzU2NhbGVZOwotICAgICAgICBsQXNjZW50ICo9IGFic1NjYWxlWTsKLSAgICAgICAgbERlc2NlbnQgKj0gYWJzU2NhbGVZOwotICAgICAgICBtYXhDaGFyV2lkdGggKj0gYWJzU2NhbGVYOwotCi0gICAgfQotCi0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIG9mZnNldCBvZiB0aGUgYmFzZWxpbmUuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZsb2F0W10gZ2V0QmFzZWxpbmVPZmZzZXRzKCkgewotICAgICAgICAvLyBYWFg6IGF0IHRoZSBtb21lbnQgdGhlcmUgb25seSBob3Jpem9udGFsIG1ldHJpY3MgYXJlIHRha2VuIGludG8KLSAgICAgICAgLy8gYWNjb3VudC4gSWYgdGhlcmUgaXMgbm8gYmFzZWxpbmUgaW5mb3JtYXRpb24gaW4gVHJ1ZVR5cGUgZm9udAotICAgICAgICAvLyBmaWxlIGRlZmF1bHQgdmFsdWVzIHVzZWQ6IHswLCAtYXNjZW50LCAoLWFzY2VudCtkZXNjZW50KS8yfQotCi0gICAgICAgIHJldHVybiBiYXNlbGluZU9mZnNldHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIG51bWJlciBvZiBjaGFycyBpbiBzcGVjaWZpZWQgdGV4dAotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0TnVtQ2hhcnMoKSB7Ci0gICAgICAgIHJldHVybiBudW1DaGFyczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGluZGV4IG9mIHRoZSBiYXNlbGluZSwgb25lIG9mIHByZWRlZmluZWQgY29uc3RhbnRzLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0QmFzZWxpbmVJbmRleCgpIHsKLSAgICAgICAgLy8gQmFzZWxpbmUgaW5kZXggaXMgdGhlIGRlYWZ1bHQgYmFzZWxpbmUgaW5kZXggdmFsdWUKLSAgICAgICAgLy8gdGFrZW4gZnJvbSB0aGUgVHJ1ZVR5cGUgdGFibGUgIkJBU0UiLgotICAgICAgICByZXR1cm4gYmFzZUxpbmVJbmRleDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoaWNrbmVzcyBvZiB0aGUgVW5kZXJsaW5lLgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRVbmRlcmxpbmVUaGlja25lc3MoKSB7Ci0gICAgICAgIHJldHVybiB1bmRlcmxpbmVUaGlja25lc3M7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBvZmZzZXQgb2YgdGhlIFVuZGVybGluZS4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXQgZ2V0VW5kZXJsaW5lT2Zmc2V0KCkgewotICAgICAgICByZXR1cm4gdW5kZXJsaW5lT2Zmc2V0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhpY2tuZXNzIG9mIHRoZSBTdHJpa2V0aHJvdWdoIGxpbmUuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZsb2F0IGdldFN0cmlrZXRocm91Z2hUaGlja25lc3MoKSB7Ci0gICAgICAgIHJldHVybiBzdHJpa2V0aHJvdWdoVGhpY2tuZXNzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgb2Zmc2V0IG9mIHRoZSBTdHJpa2V0aHJvdWdoIGxpbmUuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZsb2F0IGdldFN0cmlrZXRocm91Z2hPZmZzZXQoKSB7Ci0gICAgICAgIHJldHVybiBzdHJpa2V0aHJvdWdoT2Zmc2V0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGxlYWRpbmcuCi0gICAgICovCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGZsb2F0IGdldExlYWRpbmcoKSB7Ci0gICAgICAgIHJldHVybiBsZWFkaW5nOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGhlaWdodCBvZiB0aGUgZm9udC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXQgZ2V0SGVpZ2h0KCkgewotICAgICAgICAvL3JldHVybiBoZWlnaHQ7IC8vIGVxdWFscyB0byAoYXNjZW50ICsgZGVzY2VudCArIGxlYWRpbmcpOwotICAgIAlyZXR1cm4gYXNjZW50ICsgZGVzY2VudCArIGxlYWRpbmc7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgZGVzY2VudC4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgZmxvYXQgZ2V0RGVzY2VudCgpIHsKLSAgICAgICAgcmV0dXJuIGRlc2NlbnQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgYXNjZW50LgotICAgICAqLwotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBmbG9hdCBnZXRBc2NlbnQoKSB7Ci0gICAgICAgIHJldHVybiBhc2NlbnQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBsb2dpY2FsIHRoaWNrbmVzcyBvZiB0aGUgVW5kZXJsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TG9naWNhbFVuZGVybGluZVRoaWNrbmVzcygpIHsKLSAgICAgICAgcmV0dXJuIGxVbmRlcmxpbmVUaGlja25lc3M7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBsb2dpY2FsIG9mZnNldCBvZiB0aGUgVW5kZXJsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TG9naWNhbFVuZGVybGluZU9mZnNldCgpIHsKLSAgICAgICAgcmV0dXJuIGxVbmRlcmxpbmVPZmZzZXQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBsb2dpY2FsIHRoaWNrbmVzcyBvZiB0aGUgU3RyaWtldGhyb3VnaCBsaW5lLgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TG9naWNhbFN0cmlrZXRocm91Z2hUaGlja25lc3MoKSB7Ci0gICAgICAgIHJldHVybiBsU3RyaWtldGhyb3VnaFRoaWNrbmVzczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGxvZ2ljYWwgb2Zmc2V0IG9mIHRoZSBTdHJpa2V0aHJvdWdoIGxpbmUuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRMb2dpY2FsU3RyaWtldGhyb3VnaE9mZnNldCgpIHsKLSAgICAgICAgcmV0dXJuIGxTdHJpa2V0aHJvdWdoT2Zmc2V0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGxvZ2ljYWwgbGVhZGluZy4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldExvZ2ljYWxMZWFkaW5nKCkgewotICAgICAgICByZXR1cm4gbExlYWRpbmc7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbG9naWNhbCBoZWlnaHQgb2YgdGhlIGZvbnQuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRMb2dpY2FsSGVpZ2h0KCkgewotICAgICAgICByZXR1cm4gbEhlaWdodDsgLy8gZXF1YWxzIHRvIChhc2NlbnQgKyBkZXNjZW50ICsgbGVhZGluZyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbG9naWNhbCBkZXNjZW50LgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TG9naWNhbERlc2NlbnQoKSB7Ci0gICAgICAgIHJldHVybiBsRGVzY2VudDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBsb2dpY2FsIGFzY2VudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldExvZ2ljYWxBc2NlbnQoKSB7Ci0gICAgICAgIHJldHVybiBsQXNjZW50OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGxvZ2ljYWwgc2l6ZSBvZiB0aGUgd2lkZXN0IGNoYXIuCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRMb2dpY2FsTWF4Q2hhcldpZHRoKCkgewotICAgICAgICByZXR1cm4gbE1heENoYXJXaWR0aDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBzaXplIG9mIHRoZSB3aWRlc3QgY2hhci4KLSAgICAgKi8KLSAgICBwdWJsaWMgZmxvYXQgZ2V0TWF4Q2hhcldpZHRoKCkgewotICAgICAgICByZXR1cm4gbWF4Q2hhcldpZHRoOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNldCBudW0gY2hhcnMgdG8gdGhlIGRlc2lyZWQgdmFsdWUuCi0gICAgICogCi0gICAgICogQHBhcmFtIG51bSBzcGVjaWZpZWQgbnVtYmVyIG9mIGNoYXJzCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgc2V0TnVtQ2hhcnMoaW50IG51bSl7Ci0gICAgICAgIG51bUNoYXJzID0gbnVtOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgY2xvbmUoKXsKLSAgICAgICAgdHJ5ewotICAgICAgICAgICAgcmV0dXJuIHN1cGVyLmNsb25lKCk7Ci0gICAgICAgIH1jYXRjaCAoQ2xvbmVOb3RTdXBwb3J0ZWRFeGNlcHRpb24gZSl7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotICAgIH0KLQotfQpcIE5vIG5ld2xpbmUgYXQgZW5kIG9mIGZpbGUKZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvVGV4dERlY29yYXRvci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9UZXh0RGVjb3JhdG9yLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDgxOTA1ZmQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9UZXh0RGVjb3JhdG9yLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0MzMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udDsKLQotaW1wb3J0IGphdmEuYXd0LkJhc2ljU3Ryb2tlOwotaW1wb3J0IGphdmEuYXd0LkNvbG9yOwotaW1wb3J0IGphdmEuYXd0LkdyYXBoaWNzMkQ7Ci1pbXBvcnQgamF2YS5hd3QuUGFpbnQ7Ci1pbXBvcnQgamF2YS5hd3QuU2hhcGU7Ci1pbXBvcnQgamF2YS5hd3QuU3Ryb2tlOwotaW1wb3J0IGphdmEuYXd0LmZvbnQuVGV4dEF0dHJpYnV0ZTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFyZWE7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5MaW5lMkQ7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLWltcG9ydCBqYXZhLnRleHQuQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yLkF0dHJpYnV0ZTsKLWltcG9ydCBqYXZhLnV0aWwuTWFwOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgaXMgcmVzcG9uc2libGUgZm9yIHJlbmRlcmluZyB0ZXh0IGRlY29yYXRpb25zIGxpa2UKLSAqIHVuZGVybGluZSwgc3RyaWtldGhyb3VnaCwgdGV4dCB3aXRoIGJhY2tncm91bmQsIGV0Yy4KLSAqLwotcHVibGljIGNsYXNzIFRleHREZWNvcmF0b3IgewotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFRleHREZWNvcmF0b3IgaW5zdCA9IG5ldyBUZXh0RGVjb3JhdG9yKCk7Ci0gICAgcHJpdmF0ZSBUZXh0RGVjb3JhdG9yKCkge30KLSAgICBzdGF0aWMgVGV4dERlY29yYXRvciBnZXRJbnN0YW5jZSgpIHsKLSAgICAgICAgcmV0dXJuIGluc3Q7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhpcyBjbGFzcyBlbmNhcHN1bGF0ZXMgYSBzZXQgb2YgZGVjb3JhdGlvbiBhdHRyaWJ1dGVzIGZvciBhIHNpbmdsZSB0ZXh0IHJ1bi4KLSAgICAgKi8KLSAgICBzdGF0aWMgY2xhc3MgRGVjb3JhdGlvbiB7Ci0gICAgICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIEJhc2ljU3Ryb2tlIFVOREVSTElORV9MT1dfT05FX1BJWEVMX1NUUk9LRSA9Ci0gICAgICAgICAgICAgICAgbmV3IEJhc2ljU3Ryb2tlKDEsIEJhc2ljU3Ryb2tlLkNBUF9CVVRULCBCYXNpY1N0cm9rZS5KT0lOX01JVEVSLCAxMCk7Ci0KLSAgICAgICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgQmFzaWNTdHJva2UgVU5ERVJMSU5FX0xPV19UV09fUElYRUxfU1RST0tFID0KLSAgICAgICAgICAgICAgICBuZXcgQmFzaWNTdHJva2UoMiwgQmFzaWNTdHJva2UuQ0FQX0JVVFQsIEJhc2ljU3Ryb2tlLkpPSU5fTUlURVIsIDEwKTsKLQotICAgICAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBCYXNpY1N0cm9rZSBVTkRFUkxJTkVfTE9XX0RPVFRFRF9TVFJPS0UgPQotICAgICAgICAgICAgICAgIG5ldyBCYXNpY1N0cm9rZSgKLSAgICAgICAgICAgICAgICAgICAgICAgIDEsIEJhc2ljU3Ryb2tlLkNBUF9CVVRULCBCYXNpY1N0cm9rZS5KT0lOX01JVEVSLCAxMCwKLSAgICAgICAgICAgICAgICAgICAgICAgIG5ldyBmbG9hdFtdIHsgMSwgMSB9LCAwCi0gICAgICAgICAgICAgICAgKTsKLQotICAgICAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBCYXNpY1N0cm9rZSBVTkRFUkxJTkVfTE9XX0RPVFRFRF9TVFJPS0UyID0KLSAgICAgICAgICAgICAgICBuZXcgQmFzaWNTdHJva2UoCi0gICAgICAgICAgICAgICAgICAgICAgICAxLCBCYXNpY1N0cm9rZS5DQVBfQlVUVCwgQmFzaWNTdHJva2UuSk9JTl9NSVRFUiwgMTAsCi0gICAgICAgICAgICAgICAgICAgICAgICBuZXcgZmxvYXRbXSB7IDEsIDEgfSwgMQotICAgICAgICAgICAgICAgICk7Ci0KLSAgICAgICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgQmFzaWNTdHJva2UgVU5ERVJMSU5FX0xPV19EQVNIRURfU1RST0tFID0KLSAgICAgICAgICAgICAgICBuZXcgQmFzaWNTdHJva2UoCi0gICAgICAgICAgICAgICAgICAgICAgICAxLCBCYXNpY1N0cm9rZS5DQVBfQlVUVCwgQmFzaWNTdHJva2UuSk9JTl9NSVRFUiwgMTAsCi0gICAgICAgICAgICAgICAgICAgICAgICBuZXcgZmxvYXRbXSB7IDQsIDQgfSwgMAotICAgICAgICAgICAgICAgICk7Ci0KLSAgICAgICAgYm9vbGVhbiB1bE9uID0gZmFsc2U7IC8vIEhhdmUgc3RhbmRhcmQgdW5kZXJsaW5lPwotICAgICAgICBCYXNpY1N0cm9rZSB1bFN0cm9rZTsKLQotICAgICAgICBCYXNpY1N0cm9rZSBpbVVsU3Ryb2tlOyAgLy8gU3Ryb2tlIGZvciBJTlBVVF9NRVRIT0RfVU5ERVJMSU5FCi0gICAgICAgIEJhc2ljU3Ryb2tlIGltVWxTdHJva2UyOyAvLyBTcGVjaWFsbHkgZm9yIFVOREVSTElORV9MT1dfR1JBWQotCi0gICAgICAgIGJvb2xlYW4gc3RyaWtlVGhyb3VnaDsKLSAgICAgICAgQmFzaWNTdHJva2Ugc3RyaWtlVGhyb3VnaFN0cm9rZTsKLQotICAgICAgICBib29sZWFuIGhhdmVTdHJva2VzID0gZmFsc2U7IC8vIFN0cm9rZXMgYWxyZWFkeSBjcmVhdGVkPwotCi0gICAgICAgIGJvb2xlYW4gc3dhcEJmRmc7Ci0gICAgICAgIFBhaW50IGJnOyAvLyBiYWNrZ3JvdW5kIGNvbG9yCi0gICAgICAgIFBhaW50IGZnOyAvLyBmb3JlZ3JvdW5kIGNvbG9yCi0KLSAgICAgICAgUGFpbnQgZ3JhcGhpY3NQYWludDsgLy8gU2xvdCBmb3Igc2F2aW5nIGN1cnJlbnQgcGFpbnQKLQotICAgICAgICBEZWNvcmF0aW9uKAotICAgICAgICAgICAgICAgIEludGVnZXIgaW1VbCwKLSAgICAgICAgICAgICAgICBib29sZWFuIHN3YXAsCi0gICAgICAgICAgICAgICAgYm9vbGVhbiBzdGgsCi0gICAgICAgICAgICAgICAgUGFpbnQgYmcsIFBhaW50IGZnLAotICAgICAgICAgICAgICAgIGJvb2xlYW4gdWxPbikgewotCi0gICAgICAgICAgICBpZiAoaW1VbCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgLy8gRGV0ZXJtaW5lIHdoaWNoIHN0cm9rZSB0byB1c2UKLSAgICAgICAgICAgICAgICBpZiAoaW1VbCA9PSBUZXh0QXR0cmlidXRlLlVOREVSTElORV9MT1dfT05FX1BJWEVMKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMuaW1VbFN0cm9rZSA9IERlY29yYXRpb24uVU5ERVJMSU5FX0xPV19PTkVfUElYRUxfU1RST0tFOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoaW1VbCA9PSBUZXh0QXR0cmlidXRlLlVOREVSTElORV9MT1dfVFdPX1BJWEVMKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMuaW1VbFN0cm9rZSA9IERlY29yYXRpb24uVU5ERVJMSU5FX0xPV19UV09fUElYRUxfU1RST0tFOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoaW1VbCA9PSBUZXh0QXR0cmlidXRlLlVOREVSTElORV9MT1dfRE9UVEVEKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRoaXMuaW1VbFN0cm9rZSA9IERlY29yYXRpb24uVU5ERVJMSU5FX0xPV19ET1RURURfU1RST0tFOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoaW1VbCA9PSBUZXh0QXR0cmlidXRlLlVOREVSTElORV9MT1dfR1JBWSkgewotICAgICAgICAgICAgICAgICAgICB0aGlzLmltVWxTdHJva2UgPSBEZWNvcmF0aW9uLlVOREVSTElORV9MT1dfRE9UVEVEX1NUUk9LRTsKLSAgICAgICAgICAgICAgICAgICAgdGhpcy5pbVVsU3Ryb2tlMiA9IERlY29yYXRpb24uVU5ERVJMSU5FX0xPV19ET1RURURfU1RST0tFMjsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGltVWwgPT0gVGV4dEF0dHJpYnV0ZS5VTkRFUkxJTkVfTE9XX0RBU0hFRCkgewotICAgICAgICAgICAgICAgICAgICB0aGlzLmltVWxTdHJva2UgPSBEZWNvcmF0aW9uLlVOREVSTElORV9MT1dfREFTSEVEX1NUUk9LRTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHRoaXMudWxPbiA9IHVsT247IC8vIEhhcyB1bmRlcmxpbmUKLSAgICAgICAgICAgIHRoaXMuc3dhcEJmRmcgPSBzd2FwOwotICAgICAgICAgICAgdGhpcy5zdHJpa2VUaHJvdWdoID0gc3RoOwotICAgICAgICAgICAgdGhpcy5iZyA9IGJnOwotICAgICAgICAgICAgdGhpcy5mZyA9IGZnOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENyZWF0ZXMgc3Ryb2tlcyBvZiBwcm9wZXIgd2lkdGggYWNjb3JkaW5nIHRvIHRoZSBpbmZvCi0gICAgICAgICAqIHN0b3JlZCBpbiB0aGUgQmFzaWNNZXRyaWNzCi0gICAgICAgICAqIEBwYXJhbSBtZXRyaWNzIC0gYmFzaWMgbWV0cmljcwotICAgICAgICAgKi8KLSAgICAgICAgcHJpdmF0ZSB2b2lkIGdldFN0cm9rZXMoQmFzaWNNZXRyaWNzIG1ldHJpY3MpIHsKLSAgICAgICAgICAgIGlmICghaGF2ZVN0cm9rZXMpIHsKLSAgICAgICAgICAgICAgICBpZiAoc3RyaWtlVGhyb3VnaCkgewotICAgICAgICAgICAgICAgICAgICBzdHJpa2VUaHJvdWdoU3Ryb2tlID0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXcgQmFzaWNTdHJva2UoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtZXRyaWNzLnN0cmlrZXRocm91Z2hUaGlja25lc3MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBCYXNpY1N0cm9rZS5DQVBfQlVUVCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEJhc2ljU3Ryb2tlLkpPSU5fTUlURVIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAxMAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgaWYgKHVsT24pIHsKLSAgICAgICAgICAgICAgICAgICAgdWxTdHJva2UgPQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5ldyBCYXNpY1N0cm9rZSgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1ldHJpY3MudW5kZXJsaW5lVGhpY2tuZXNzLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQmFzaWNTdHJva2UuQ0FQX0JVVFQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBCYXNpY1N0cm9rZS5KT0lOX01JVEVSLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMTAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGhhdmVTdHJva2VzID0gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgRGVjb3JhdGlvbiBvYmplY3QgZnJvbSB0aGUgc2V0IG9mIHRleHQgYXR0cmlidXRlcwotICAgICAqIEBwYXJhbSBhdHRyaWJ1dGVzIC0gdGV4dCBhdHRyaWJ1dGVzCi0gICAgICogQHJldHVybiBEZWNvcmF0aW9uIG9iamVjdAotICAgICAqLwotICAgIHN0YXRpYyBEZWNvcmF0aW9uIGdldERlY29yYXRpb24oTWFwPD8gZXh0ZW5kcyBBdHRyaWJ1dGUsID8+IGF0dHJpYnV0ZXMpIHsKLSAgICAgICAgaWYgKGF0dHJpYnV0ZXMgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7IC8vIEl0IGlzIGZvciBwbGFpbiB0ZXh0Ci0gICAgICAgIH0KLQotICAgICAgICBPYmplY3QgdW5kZXJsaW5lID0gYXR0cmlidXRlcy5nZXQoVGV4dEF0dHJpYnV0ZS5VTkRFUkxJTkUpOwotICAgICAgICBib29sZWFuIGhhc1N0YW5kYXJkVW5kZXJsaW5lID0gdW5kZXJsaW5lID09IFRleHRBdHRyaWJ1dGUuVU5ERVJMSU5FX09OOwotCi0gICAgICAgIE9iamVjdCBpbVVuZGVybGluZSA9IGF0dHJpYnV0ZXMuZ2V0KFRleHRBdHRyaWJ1dGUuSU5QVVRfTUVUSE9EX1VOREVSTElORSk7Ci0gICAgICAgIEludGVnZXIgaW1VbCA9IChJbnRlZ2VyKSBpbVVuZGVybGluZTsKLQotICAgICAgICBib29sZWFuIHN3YXBCZ0ZnID0KLSAgICAgICAgICAgICAgICBUZXh0QXR0cmlidXRlLlNXQVBfQ09MT1JTX09OLmVxdWFscygKLSAgICAgICAgICAgICAgICAgICAgICAgIGF0dHJpYnV0ZXMuZ2V0KFRleHRBdHRyaWJ1dGUuU1dBUF9DT0xPUlMpCi0gICAgICAgICAgICAgICAgKTsKLQotICAgICAgICBib29sZWFuIHN0cmlrZVRocm91Z2ggPQotICAgICAgICAgICAgICAgIFRleHRBdHRyaWJ1dGUuU1RSSUtFVEhST1VHSF9PTi5lcXVhbHMoCi0gICAgICAgICAgICAgICAgICAgICAgICBhdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLlNUUklLRVRIUk9VR0gpCi0gICAgICAgICAgICAgICAgKTsKLQotICAgICAgICBQYWludCBmZyA9IChQYWludCkgYXR0cmlidXRlcy5nZXQoVGV4dEF0dHJpYnV0ZS5GT1JFR1JPVU5EKTsKLSAgICAgICAgUGFpbnQgYmcgPSAoUGFpbnQpIGF0dHJpYnV0ZXMuZ2V0KFRleHRBdHRyaWJ1dGUuQkFDS0dST1VORCk7Ci0KLSAgICAgICAgaWYgKAotICAgICAgICAgICAgICAgICFoYXNTdGFuZGFyZFVuZGVybGluZSAmJgotICAgICAgICAgICAgICAgIGltVW5kZXJsaW5lID09IG51bGwgJiYKLSAgICAgICAgICAgICAgICBmZyA9PSBudWxsICYmCi0gICAgICAgICAgICAgICAgYmcgPT0gbnVsbCAmJgotICAgICAgICAgICAgICAgICFzd2FwQmdGZyAmJgotICAgICAgICAgICAgICAgICFzdHJpa2VUaHJvdWdoCi0gICAgICAgICkgewotICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG5ldyBEZWNvcmF0aW9uKGltVWwsIHN3YXBCZ0ZnLCBzdHJpa2VUaHJvdWdoLCBiZywgZmcsIGhhc1N0YW5kYXJkVW5kZXJsaW5lKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGaWxscyB0aGUgYmFja2dyb3VuZCBiZWZvcmUgZHJhd2luZyBpZiBuZWVkZWQuCi0gICAgICogCi0gICAgICogQHBhcmFtIHRycyAtIHRleHQgc2VnbWVudAotICAgICAqIEBwYXJhbSBnMmQgLSBncmFwaGljcyB0byBkcmF3IHRvCi0gICAgICogQHBhcmFtIHhPZmZzZXQgLSBvZmZzZXQgaW4gWCBkaXJlY3Rpb24gdG8gdGhlIHVwcGVyIGxlZnQgY29ybmVyIG9mIHRoZQotICAgICAqICAgICAgICBsYXlvdXQgZnJvbSB0aGUgb3JpZ2luIG9mIHRoZSBncmFwaGljcwotICAgICAqIEBwYXJhbSB5T2Zmc2V0IC0gb2Zmc2V0IGluIFkgZGlyZWN0aW9uIHRvIHRoZSB1cHBlciBsZWZ0IGNvcm5lciBvZiB0aGUKLSAgICAgKiAgICAgICAgbGF5b3V0IGZyb20gdGhlIG9yaWdpbiBvZiB0aGUgZ3JhcGhpY3MKLSAgICAgKi8KLSAgICBzdGF0aWMgdm9pZCBwcmVwYXJlR3JhcGhpY3MoCi0gICAgICAgICAgICBUZXh0UnVuU2VnbWVudCB0cnMsIEdyYXBoaWNzMkQgZzJkLAotICAgICAgICAgICAgZmxvYXQgeE9mZnNldCwgZmxvYXQgeU9mZnNldAotICAgICkgewotICAgICAgICBEZWNvcmF0aW9uIGQgPSB0cnMuZGVjb3JhdGlvbjsKLQotICAgICAgICBpZiAoZC5mZyA9PSBudWxsICYmIGQuYmcgPT0gbnVsbCAmJiBkLnN3YXBCZkZnID09IGZhbHNlKSB7Ci0gICAgICAgICAgICByZXR1cm47IC8vIE5vdGhpbmcgdG8gZG8KLSAgICAgICAgfQotCi0gICAgICAgIGQuZ3JhcGhpY3NQYWludCA9IGcyZC5nZXRQYWludCgpOwotCi0gICAgICAgIGlmIChkLmZnID09IG51bGwpIHsKLSAgICAgICAgICAgIGQuZmcgPSBkLmdyYXBoaWNzUGFpbnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZC5zd2FwQmZGZykgewotICAgICAgICAgICAgLy8gRmlsbCBiYWNrZ3JvdW5kIGFyZWEKLSAgICAgICAgICAgIGcyZC5zZXRQYWludChkLmZnKTsKLSAgICAgICAgICAgIFJlY3RhbmdsZTJEIGJnQXJlYSA9IHRycy5nZXRMb2dpY2FsQm91bmRzKCk7Ci0gICAgICAgICAgICBSZWN0YW5nbGUyRCB0b0ZpbGwgPQotICAgICAgICAgICAgICAgICAgICBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJnQXJlYS5nZXRYKCkgKyB4T2Zmc2V0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJnQXJlYS5nZXRZKCkgKyB5T2Zmc2V0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJnQXJlYS5nZXRXaWR0aCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJnQXJlYS5nZXRIZWlnaHQoKQotICAgICAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgZzJkLmZpbGwodG9GaWxsKTsKLQotICAgICAgICAgICAgLy8gU2V0IGZvcmVncm91bmQgY29sb3IKLSAgICAgICAgICAgIGcyZC5zZXRQYWludChkLmJnID09IG51bGwgPyBDb2xvci5XSElURSA6IGQuYmcpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaWYgKGQuYmcgIT0gbnVsbCkgeyAvLyBGaWxsIGJhY2tncm91bmQgYXJlYQotICAgICAgICAgICAgICAgIGcyZC5zZXRQYWludChkLmJnKTsKLSAgICAgICAgICAgICAgICBSZWN0YW5nbGUyRCBiZ0FyZWEgPSB0cnMuZ2V0TG9naWNhbEJvdW5kcygpOwotICAgICAgICAgICAgICAgIFJlY3RhbmdsZTJEIHRvRmlsbCA9Ci0gICAgICAgICAgICAgICAgICAgICAgICBuZXcgUmVjdGFuZ2xlMkQuRG91YmxlKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBiZ0FyZWEuZ2V0WCgpICsgeE9mZnNldCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYmdBcmVhLmdldFkoKSArIHlPZmZzZXQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJnQXJlYS5nZXRXaWR0aCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBiZ0FyZWEuZ2V0SGVpZ2h0KCkKLSAgICAgICAgICAgICAgICAgICAgICAgICk7Ci0gICAgICAgICAgICAgICAgZzJkLmZpbGwodG9GaWxsKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gU2V0IGZvcmVncm91bmQgY29sb3IKLSAgICAgICAgICAgIGcyZC5zZXRQYWludChkLmZnKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlc3RvcmVzIHRoZSBvcmlnaW5hbCBzdGF0ZSBvZiB0aGUgZ3JhcGhpY3MgaWYgbmVlZGVkCi0gICAgICogQHBhcmFtIGQgLSBkZWNvcmF0aW9uCi0gICAgICogQHBhcmFtIGcyZCAtIGdyYXBoaWNzCi0gICAgICovCi0gICAgc3RhdGljIHZvaWQgcmVzdG9yZUdyYXBoaWNzKERlY29yYXRpb24gZCwgR3JhcGhpY3MyRCBnMmQpIHsKLSAgICAgICAgaWYgKGQuZmcgPT0gbnVsbCAmJiBkLmJnID09IG51bGwgJiYgZC5zd2FwQmZGZyA9PSBmYWxzZSkgewotICAgICAgICAgICAgcmV0dXJuOyAvLyBOb3RoaW5nIHRvIGRvCi0gICAgICAgIH0KLQotICAgICAgICBnMmQuc2V0UGFpbnQoZC5ncmFwaGljc1BhaW50KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZW5kZXJzIHRoZSB0ZXh0IGRlY29yYXRpb25zCi0gICAgICogQHBhcmFtIHRycyAtIHRleHQgcnVuIHNlZ21lbnQKLSAgICAgKiBAcGFyYW0gZzJkIC0gZ3JhcGhpY3MgdG8gcmVuZGVyIHRvCi0gICAgICogQHBhcmFtIHhPZmZzZXQgLSBvZmZzZXQgaW4gWCBkaXJlY3Rpb24gdG8gdGhlIHVwcGVyIGxlZnQgY29ybmVyCi0gICAgICogb2YgdGhlIGxheW91dCBmcm9tIHRoZSBvcmlnaW4gb2YgdGhlIGdyYXBoaWNzCi0gICAgICogQHBhcmFtIHlPZmZzZXQgLSBvZmZzZXQgaW4gWSBkaXJlY3Rpb24gdG8gdGhlIHVwcGVyIGxlZnQgY29ybmVyCi0gICAgICogb2YgdGhlIGxheW91dCBmcm9tIHRoZSBvcmlnaW4gb2YgdGhlIGdyYXBoaWNzCi0gICAgICovCi0gICAgc3RhdGljIHZvaWQgZHJhd1RleHREZWNvcmF0aW9ucygKLSAgICAgICAgICAgIFRleHRSdW5TZWdtZW50IHRycywgR3JhcGhpY3MyRCBnMmQsCi0gICAgICAgICAgICBmbG9hdCB4T2Zmc2V0LCBmbG9hdCB5T2Zmc2V0Ci0gICAgKSB7Ci0gICAgICAgIERlY29yYXRpb24gZCA9IHRycy5kZWNvcmF0aW9uOwotCi0gICAgICAgIGlmICghZC51bE9uICYmIGQuaW1VbFN0cm9rZSA9PSBudWxsICYmICFkLnN0cmlrZVRocm91Z2gpIHsKLSAgICAgICAgICAgIHJldHVybjsgLy8gTm90aGluZyB0byBkbwotICAgICAgICB9Ci0KLSAgICAgICAgZmxvYXQgbGVmdCA9IHhPZmZzZXQgKyAoZmxvYXQpIHRycy5nZXRMb2dpY2FsQm91bmRzKCkuZ2V0TWluWCgpOwotICAgICAgICBmbG9hdCByaWdodCA9IHhPZmZzZXQgKyAoZmxvYXQpIHRycy5nZXRMb2dpY2FsQm91bmRzKCkuZ2V0TWF4WCgpOwotCi0gICAgICAgIFN0cm9rZSBzYXZlZFN0cm9rZSA9IGcyZC5nZXRTdHJva2UoKTsKLQotICAgICAgICBkLmdldFN0cm9rZXModHJzLm1ldHJpY3MpOwotCi0gICAgICAgIGlmIChkLnN0cmlrZVRocm91Z2gpIHsKLSAgICAgICAgICAgIGZsb2F0IHkgPSB0cnMueSArIHlPZmZzZXQgKyB0cnMubWV0cmljcy5zdHJpa2V0aHJvdWdoT2Zmc2V0OwotICAgICAgICAgICAgZzJkLnNldFN0cm9rZShkLnN0cmlrZVRocm91Z2hTdHJva2UpOwotICAgICAgICAgICAgZzJkLmRyYXcobmV3IExpbmUyRC5GbG9hdChsZWZ0LCB5LCByaWdodCwgeSkpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGQudWxPbikgewotICAgICAgICAgICAgZmxvYXQgeSA9IHRycy55ICsgeU9mZnNldCArIHRycy5tZXRyaWNzLnVuZGVybGluZU9mZnNldDsKLSAgICAgICAgICAgIGcyZC5zZXRTdHJva2UoZC51bFN0cm9rZSk7Ci0gICAgICAgICAgICBnMmQuZHJhdyhuZXcgTGluZTJELkZsb2F0KGxlZnQsIHksIHJpZ2h0LCB5KSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZC5pbVVsU3Ryb2tlICE9IG51bGwpIHsKLSAgICAgICAgICAgIGZsb2F0IHkgPSB0cnMueSArIHlPZmZzZXQgKyB0cnMubWV0cmljcy51bmRlcmxpbmVPZmZzZXQ7Ci0gICAgICAgICAgICBnMmQuc2V0U3Ryb2tlKGQuaW1VbFN0cm9rZSk7Ci0gICAgICAgICAgICBnMmQuZHJhdyhuZXcgTGluZTJELkZsb2F0KGxlZnQsIHksIHJpZ2h0LCB5KSk7Ci0gICAgICAgICAgICBpZiAoZC5pbVVsU3Ryb2tlMiAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgeSsrOwotICAgICAgICAgICAgICAgIGcyZC5zZXRTdHJva2UoZC5pbVVsU3Ryb2tlMik7Ci0gICAgICAgICAgICAgICAgZzJkLmRyYXcobmV3IExpbmUyRC5GbG9hdChsZWZ0LCB5LCByaWdodCwgeSkpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgZzJkLnNldFN0cm9rZShzYXZlZFN0cm9rZSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRXh0ZW5kcyB0aGUgdmlzdWFsIGJvdW5kcyBvZiB0aGUgdGV4dCBydW4gc2VnbWVudCB0bwotICAgICAqIGluY2x1ZGUgdGV4dCBkZWNvcmF0aW9ucy4KLSAgICAgKiBAcGFyYW0gdHJzIC0gdGV4dCBzZWdtZW50Ci0gICAgICogQHBhcmFtIHNlZ21lbnRCb3VuZHMgLSBib3VuZHMgb2YgdGhlIHVuZGVjb3JhdGVkIHRleHQKLSAgICAgKiBAcGFyYW0gZCAtIGRlY29yYXRpb24KLSAgICAgKiBAcmV0dXJuIGV4dGVuZGVkIGJvdW5kcwotICAgICAqLwotICAgIHN0YXRpYyBSZWN0YW5nbGUyRCBleHRlbmRWaXN1YWxCb3VuZHMoCi0gICAgICAgICAgICBUZXh0UnVuU2VnbWVudCB0cnMsCi0gICAgICAgICAgICBSZWN0YW5nbGUyRCBzZWdtZW50Qm91bmRzLAotICAgICAgICAgICAgRGVjb3JhdGlvbiBkCi0gICAgKSB7Ci0gICAgICAgIGlmIChkID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBzZWdtZW50Qm91bmRzOwotICAgICAgICB9Ci0gICAgICAgIGRvdWJsZSBtaW54ID0gc2VnbWVudEJvdW5kcy5nZXRNaW5YKCk7Ci0gICAgICAgIGRvdWJsZSBtaW55ID0gc2VnbWVudEJvdW5kcy5nZXRNaW5ZKCk7Ci0gICAgICAgIGRvdWJsZSBtYXh4ID0gc2VnbWVudEJvdW5kcy5nZXRNYXhYKCk7Ci0gICAgICAgIGRvdWJsZSBtYXh5ID0gc2VnbWVudEJvdW5kcy5nZXRNYXhZKCk7Ci0KLSAgICAgICAgUmVjdGFuZ2xlMkQgbGIgPSB0cnMuZ2V0TG9naWNhbEJvdW5kcygpOwotCi0gICAgICAgIGlmIChkLnN3YXBCZkZnIHx8IGQuYmcgIT0gbnVsbCkgewotICAgICAgICAgICAgbWlueCA9IE1hdGgubWluKGxiLmdldE1pblgoKSAtIHRycy54LCBtaW54KTsKLSAgICAgICAgICAgIG1pbnkgPSBNYXRoLm1pbihsYi5nZXRNaW5ZKCkgLSB0cnMueSwgbWlueSk7Ci0gICAgICAgICAgICBtYXh4ID0gTWF0aC5tYXgobGIuZ2V0TWF4WCgpIC0gdHJzLngsIG1heHgpOwotICAgICAgICAgICAgbWF4eSA9IE1hdGgubWF4KGxiLmdldE1heFkoKSAtIHRycy55LCBtYXh5KTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkLnVsT24gfHwgZC5pbVVsU3Ryb2tlICE9IG51bGwgfHwgZC5zdHJpa2VUaHJvdWdoKSB7Ci0gICAgICAgICAgICBtaW54ID0gTWF0aC5taW4obGIuZ2V0TWluWCgpIC0gdHJzLngsIG1pbngpOwotICAgICAgICAgICAgbWF4eCA9IE1hdGgubWF4KGxiLmdldE1heFgoKSAtIHRycy54LCBtYXh4KTsKLQotICAgICAgICAgICAgZC5nZXRTdHJva2VzKHRycy5tZXRyaWNzKTsKLQotICAgICAgICAgICAgaWYgKGQudWxTdHJva2UgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIG1heHkgPSBNYXRoLm1heCgKLSAgICAgICAgICAgICAgICAgICAgICAgIG1heHksCi0gICAgICAgICAgICAgICAgICAgICAgICB0cnMubWV0cmljcy51bmRlcmxpbmVPZmZzZXQgKwotICAgICAgICAgICAgICAgICAgICAgICAgZC51bFN0cm9rZS5nZXRMaW5lV2lkdGgoKQotICAgICAgICAgICAgICAgICk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChkLmltVWxTdHJva2UgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIG1heHkgPSBNYXRoLm1heCgKLSAgICAgICAgICAgICAgICAgICAgICAgIG1heHksCi0gICAgICAgICAgICAgICAgICAgICAgICB0cnMubWV0cmljcy51bmRlcmxpbmVPZmZzZXQgKwotICAgICAgICAgICAgICAgICAgICAgICAgZC5pbVVsU3Ryb2tlLmdldExpbmVXaWR0aCgpICsKLSAgICAgICAgICAgICAgICAgICAgICAgIChkLmltVWxTdHJva2UyID09IG51bGwgPyAwIDogZC5pbVVsU3Ryb2tlMi5nZXRMaW5lV2lkdGgoKSkKLSAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG5ldyBSZWN0YW5nbGUyRC5Eb3VibGUobWlueCwgbWlueSwgbWF4eC1taW54LCBtYXh5LW1pbnkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEV4dGVuZHMgdGhlIG91dGxpbmUgb2YgdGhlIHRleHQgcnVuIHNlZ21lbnQgdG8KLSAgICAgKiBpbmNsdWRlIHRleHQgZGVjb3JhdGlvbnMuCi0gICAgICogQHBhcmFtIHRycyAtIHRleHQgc2VnbWVudAotICAgICAqIEBwYXJhbSBzZWdtZW50T3V0bGluZSAtIG91dGxpbmUgb2YgdGhlIHVuZGVjb3JhdGVkIHRleHQKLSAgICAgKiBAcGFyYW0gZCAtIGRlY29yYXRpb24KLSAgICAgKiBAcmV0dXJuIGV4dGVuZGVkIG91dGxpbmUKLSAgICAgKi8KLSAgICBzdGF0aWMgU2hhcGUgZXh0ZW5kT3V0bGluZSgKLSAgICAgICAgICAgIFRleHRSdW5TZWdtZW50IHRycywKLSAgICAgICAgICAgIFNoYXBlIHNlZ21lbnRPdXRsaW5lLAotICAgICAgICAgICAgRGVjb3JhdGlvbiBkCi0gICAgKSB7Ci0gICAgICAgIGlmIChkID09IG51bGwgfHwgIWQudWxPbiAmJiBkLmltVWxTdHJva2UgPT0gbnVsbCAmJiAhZC5zdHJpa2VUaHJvdWdoKSB7Ci0gICAgICAgICAgICByZXR1cm4gc2VnbWVudE91dGxpbmU7IC8vIE5vdGhpbmcgdG8gZG8KLSAgICAgICAgfQotCi0gICAgICAgIEFyZWEgcmVzID0gbmV3IEFyZWEoc2VnbWVudE91dGxpbmUpOwotCi0gICAgICAgIGZsb2F0IGxlZnQgPSAoZmxvYXQpIHRycy5nZXRMb2dpY2FsQm91bmRzKCkuZ2V0TWluWCgpIC0gdHJzLng7Ci0gICAgICAgIGZsb2F0IHJpZ2h0ID0gKGZsb2F0KSB0cnMuZ2V0TG9naWNhbEJvdW5kcygpLmdldE1heFgoKSAtIHRycy54OwotCi0gICAgICAgIGQuZ2V0U3Ryb2tlcyh0cnMubWV0cmljcyk7Ci0KLSAgICAgICAgaWYgKGQuc3RyaWtlVGhyb3VnaCkgewotICAgICAgICAgICAgZmxvYXQgeSA9IHRycy5tZXRyaWNzLnN0cmlrZXRocm91Z2hPZmZzZXQ7Ci0gICAgICAgICAgICByZXMuYWRkKG5ldyBBcmVhKGQuc3RyaWtlVGhyb3VnaFN0cm9rZS5jcmVhdGVTdHJva2VkU2hhcGUoCi0gICAgICAgICAgICAgICAgICAgIG5ldyBMaW5lMkQuRmxvYXQobGVmdCwgeSwgcmlnaHQsIHkpCi0gICAgICAgICAgICApKSk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZC51bE9uKSB7Ci0gICAgICAgICAgICBmbG9hdCB5ID0gdHJzLm1ldHJpY3MudW5kZXJsaW5lT2Zmc2V0OwotICAgICAgICAgICAgcmVzLmFkZChuZXcgQXJlYShkLnVsU3Ryb2tlLmNyZWF0ZVN0cm9rZWRTaGFwZSgKLSAgICAgICAgICAgICAgICAgICAgbmV3IExpbmUyRC5GbG9hdChsZWZ0LCB5LCByaWdodCwgeSkKLSAgICAgICAgICAgICkpKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChkLmltVWxTdHJva2UgIT0gbnVsbCkgewotICAgICAgICAgICAgZmxvYXQgeSA9IHRycy5tZXRyaWNzLnVuZGVybGluZU9mZnNldDsKLSAgICAgICAgICAgIHJlcy5hZGQobmV3IEFyZWEoZC5pbVVsU3Ryb2tlLmNyZWF0ZVN0cm9rZWRTaGFwZSgKLSAgICAgICAgICAgICAgICAgICAgbmV3IExpbmUyRC5GbG9hdChsZWZ0LCB5LCByaWdodCwgeSkKLSAgICAgICAgICAgICkpKTsKLQotICAgICAgICAgICAgaWYgKGQuaW1VbFN0cm9rZTIgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHkrKzsKLSAgICAgICAgICAgICAgICByZXMuYWRkKG5ldyBBcmVhKGQuaW1VbFN0cm9rZTIuY3JlYXRlU3Ryb2tlZFNoYXBlKAotICAgICAgICAgICAgICAgICAgICAgICAgbmV3IExpbmUyRC5GbG9hdChsZWZ0LCB5LCByaWdodCwgeSkKLSAgICAgICAgICAgICAgICApKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcmVzOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvVGV4dE1ldHJpY3NDYWxjdWxhdG9yLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L1RleHRNZXRyaWNzQ2FsY3VsYXRvci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiZTU3NjJhLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvVGV4dE1ldHJpY3NDYWxjdWxhdG9yLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMDkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKgotICovCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuZm9udC5MaW5lTWV0cmljczsKLWltcG9ydCBqYXZhLmF3dC5mb250LkdyYXBoaWNBdHRyaWJ1dGU7Ci1pbXBvcnQgamF2YS5hd3QuRm9udDsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaE1hcDsKLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlMaXN0OwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBvcGVyYXRlcyB3aXRoIGFuIGFyYml0cmFyeSB0ZXh0IHN0cmluZyB3aGljaCBjYW4gaW5jbHVkZQotICogYW55IG51bWJlciBvZiBzdHlsZSwgZm9udCBhbmQgZGlyZWN0aW9uIHJ1bnMuIEl0IGlzIHJlc3BvbnNpYmxlIGZvciBjb21wdXRhdGlvbgotICogb2YgdGhlIHRleHQgbWV0cmljcywgc3VjaCBhcyBhc2NlbnQsIGRlc2NlbnQsIGxlYWRpbmcgYW5kIGFkdmFuY2UuIEFjdHVhbGx5LAotICogZWFjaCB0ZXh0IHJ1biBzZWdtZW50IGNvbnRhaW5zIGxvZ2ljIHdoaWNoIGFsbG93cyBpdCB0byBjb21wdXRlIGl0cyBvd24gbWV0cmljcyBhbmQKLSAqIHJlc3BvbnNpYmlsaXR5IG9mIHRoaXMgY2xhc3MgaXMgdG8gY29tYmluZSBtZXRyaWNzIGZvciBhbGwgc2VnbWVudHMgaW5jbHVkZWQgaW4gdGhlIHRleHQsCi0gKiBtYW5hZ2VkIGJ5IHRoZSBhc3NvY2lhdGVkIFRleHRSdW5CcmVha2VyIG9iamVjdC4KLSAqLwotcHVibGljIGNsYXNzIFRleHRNZXRyaWNzQ2FsY3VsYXRvciB7Ci0gICAgVGV4dFJ1bkJyZWFrZXIgYnJlYWtlcjsgLy8gQXNzb2NpYXRlZCBydW4gYnJlYWtlcgotCi0gICAgLy8gTWV0cmljcwotICAgIGZsb2F0IGFzY2VudCA9IDA7Ci0gICAgZmxvYXQgZGVzY2VudCA9IDA7Ci0gICAgZmxvYXQgbGVhZGluZyA9IDA7Ci0gICAgZmxvYXQgYWR2YW5jZSA9IDA7Ci0KLSAgICBwcml2YXRlIGZsb2F0IGJhc2VsaW5lT2Zmc2V0c1tdOwotICAgIGludCBiYXNlbGluZUluZGV4OwotCi0gICAgcHVibGljIFRleHRNZXRyaWNzQ2FsY3VsYXRvcihUZXh0UnVuQnJlYWtlciBicmVha2VyKSB7Ci0gICAgICAgIHRoaXMuYnJlYWtlciA9IGJyZWFrZXI7Ci0gICAgICAgIGNoZWNrQmFzZWxpbmVzKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBlaXRoZXIgdmFsdWVzIGNhY2hlZCBieSBjaGVja0Jhc2VsaW5lcyBtZXRob2Qgb3IgcmVhc29uYWJsZQotICAgICAqIHZhbHVlcyBmb3IgdGhlIFRPUCBhbmQgQk9UVE9NIGFsaWdubWVudHMuCi0gICAgICogQHBhcmFtIGJhc2VsaW5lSW5kZXggLSBiYXNlbGluZSBpbmRleAotICAgICAqIEByZXR1cm4gYmFzZWxpbmUgb2Zmc2V0Ci0gICAgICovCi0gICAgZmxvYXQgZ2V0QmFzZWxpbmVPZmZzZXQoaW50IGJhc2VsaW5lSW5kZXgpIHsKLSAgICAgICAgaWYgKGJhc2VsaW5lSW5kZXggPj0gMCkgewotICAgICAgICAgICAgcmV0dXJuIGJhc2VsaW5lT2Zmc2V0c1tiYXNlbGluZUluZGV4XTsKLSAgICAgICAgfSBlbHNlIGlmIChiYXNlbGluZUluZGV4ID09IEdyYXBoaWNBdHRyaWJ1dGUuQk9UVE9NX0FMSUdOTUVOVCkgewotICAgICAgICAgICAgcmV0dXJuIGRlc2NlbnQ7Ci0gICAgICAgIH0gZWxzZSBpZiAoYmFzZWxpbmVJbmRleCA9PSBHcmFwaGljQXR0cmlidXRlLlRPUF9BTElHTk1FTlQpIHsKLSAgICAgICAgICAgIHJldHVybiAtYXNjZW50OwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLy8gYXd0LjNGPUludmFsaWQgYmFzZWxpbmUgaW5kZXgKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuM0YiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBmbG9hdFtdIGdldEJhc2VsaW5lT2Zmc2V0cygpIHsKLSAgICAgICAgZmxvYXQgcmV0W10gPSBuZXcgZmxvYXRbYmFzZWxpbmVPZmZzZXRzLmxlbmd0aF07Ci0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkoYmFzZWxpbmVPZmZzZXRzLCAwLCByZXQsIDAsIGJhc2VsaW5lT2Zmc2V0cy5sZW5ndGgpOwotICAgICAgICByZXR1cm4gcmV0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRha2UgYmFzZWxpbmUgb2Zmc2V0cyBmcm9tIHRoZSBmaXJzdCBmb250IG9yIGdyYXBoaWMgYXR0cmlidXRlCi0gICAgICogYW5kIG5vcm1hbGl6ZXMgdGhlbSwgdGhhbiBjYWNoZXMgdGhlIHJlc3VsdHMuCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgY2hlY2tCYXNlbGluZXMoKSB7Ci0gICAgICAgIC8vIFRha2UgYmFzZWxpbmUgb2Zmc2V0cyBvZiB0aGUgZmlyc3QgZm9udCBhbmQgbm9ybWFsaXplIHRoZW0KLSAgICAgICAgSGFzaE1hcDxJbnRlZ2VyLCBGb250PiBmb250cyA9IGJyZWFrZXIuZm9udHM7Ci0KLSAgICAgICAgT2JqZWN0IHZhbCA9IGZvbnRzLmdldChuZXcgSW50ZWdlcigwKSk7Ci0KLSAgICAgICAgaWYgKHZhbCBpbnN0YW5jZW9mIEZvbnQpIHsKLSAgICAgICAgICAgIEZvbnQgZmlyc3RGb250ID0gKEZvbnQpIHZhbDsKLSAgICAgICAgICAgIExpbmVNZXRyaWNzIGxtID0gZmlyc3RGb250LmdldExpbmVNZXRyaWNzKGJyZWFrZXIudGV4dCwgMCwgMSwgYnJlYWtlci5mcmMpOwotICAgICAgICAgICAgYmFzZWxpbmVPZmZzZXRzID0gbG0uZ2V0QmFzZWxpbmVPZmZzZXRzKCk7Ci0gICAgICAgICAgICBiYXNlbGluZUluZGV4ID0gbG0uZ2V0QmFzZWxpbmVJbmRleCgpOwotICAgICAgICB9IGVsc2UgaWYgKHZhbCBpbnN0YW5jZW9mIEdyYXBoaWNBdHRyaWJ1dGUpIHsKLSAgICAgICAgICAgIC8vIEdldCBmaXJzdCBncmFwaGljIGF0dHJpYnV0ZSBhbmQgdXNlIGl0Ci0gICAgICAgICAgICBHcmFwaGljQXR0cmlidXRlIGdhID0gKEdyYXBoaWNBdHRyaWJ1dGUpIHZhbDsKLQotICAgICAgICAgICAgaW50IGFsaWduID0gZ2EuZ2V0QWxpZ25tZW50KCk7Ci0KLSAgICAgICAgICAgIGlmICgKLSAgICAgICAgICAgICAgICAgICAgYWxpZ24gPT0gR3JhcGhpY0F0dHJpYnV0ZS5UT1BfQUxJR05NRU5UIHx8Ci0gICAgICAgICAgICAgICAgICAgIGFsaWduID09IEdyYXBoaWNBdHRyaWJ1dGUuQk9UVE9NX0FMSUdOTUVOVAotICAgICAgICAgICAgKSB7Ci0gICAgICAgICAgICAgICAgYmFzZWxpbmVJbmRleCA9IEdyYXBoaWNBdHRyaWJ1dGUuUk9NQU5fQkFTRUxJTkU7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGJhc2VsaW5lSW5kZXggPSBhbGlnbjsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgYmFzZWxpbmVPZmZzZXRzID0gbmV3IGZsb2F0WzNdOwotICAgICAgICAgICAgYmFzZWxpbmVPZmZzZXRzWzBdID0gMDsKLSAgICAgICAgICAgIGJhc2VsaW5lT2Zmc2V0c1sxXSA9IChnYS5nZXREZXNjZW50KCkgLSBnYS5nZXRBc2NlbnQoKSkgLyAyLmY7Ci0gICAgICAgICAgICBiYXNlbGluZU9mZnNldHNbMl0gPSAtZ2EuZ2V0QXNjZW50KCk7Ci0gICAgICAgIH0gZWxzZSB7IC8vIFVzZSBkZWZhdWx0cyAtIFJvbWFuIGJhc2VsaW5lIGFuZCB6ZXJvIG9mZnNldHMKLSAgICAgICAgICAgIGJhc2VsaW5lSW5kZXggPSBHcmFwaGljQXR0cmlidXRlLlJPTUFOX0JBU0VMSU5FOwotICAgICAgICAgICAgYmFzZWxpbmVPZmZzZXRzID0gbmV3IGZsb2F0WzNdOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gTm9ybWFsaXplIG9mZnNldHMgaWYgbmVlZGVkCi0gICAgICAgIGlmIChiYXNlbGluZU9mZnNldHNbYmFzZWxpbmVJbmRleF0gIT0gMCkgewotICAgICAgICAgICAgZmxvYXQgYmFzZU9mZnNldCA9IGJhc2VsaW5lT2Zmc2V0c1tiYXNlbGluZUluZGV4XTsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgYmFzZWxpbmVPZmZzZXRzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgYmFzZWxpbmVPZmZzZXRzW2ldIC09IGJhc2VPZmZzZXQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wdXRlcyBtZXRyaWNzIGZvciB0aGUgdGV4dCBtYW5hZ2VkIGJ5IHRoZSBhc3NvY2lhdGVkIFRleHRSdW5CcmVha2VyCi0gICAgICovCi0gICAgdm9pZCBjb21wdXRlTWV0cmljcygpIHsKLQotICAgICAgICBBcnJheUxpc3Q8VGV4dFJ1blNlZ21lbnQ+IHNlZ21lbnRzID0gYnJlYWtlci5ydW5TZWdtZW50czsKLQotICAgICAgICBmbG9hdCBtYXhIZWlnaHQgPSAwOwotICAgICAgICBmbG9hdCBtYXhIZWlnaHRMZWFkaW5nID0gMDsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHNlZ21lbnRzLnNpemUoKTsgaSsrKSB7Ci0gICAgICAgICAgICBUZXh0UnVuU2VnbWVudCBzZWdtZW50ID0gc2VnbWVudHMuZ2V0KGkpOwotICAgICAgICAgICAgQmFzaWNNZXRyaWNzIG1ldHJpY3MgPSBzZWdtZW50Lm1ldHJpY3M7Ci0gICAgICAgICAgICBpbnQgYmFzZWxpbmUgPSBtZXRyaWNzLmJhc2VMaW5lSW5kZXg7Ci0KLSAgICAgICAgICAgIGlmIChiYXNlbGluZSA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgZmxvYXQgYmFzZWxpbmVPZmZzZXQgPSBiYXNlbGluZU9mZnNldHNbbWV0cmljcy5iYXNlTGluZUluZGV4XTsKLSAgICAgICAgICAgICAgICBmbG9hdCBmaXhlZERlc2NlbnQgPSBtZXRyaWNzLmRlc2NlbnQgKyBiYXNlbGluZU9mZnNldDsKLQotICAgICAgICAgICAgICAgIGFzY2VudCA9IE1hdGgubWF4KGFzY2VudCwgbWV0cmljcy5hc2NlbnQgLSBiYXNlbGluZU9mZnNldCk7Ci0gICAgICAgICAgICAgICAgZGVzY2VudCA9IE1hdGgubWF4KGRlc2NlbnQsIGZpeGVkRGVzY2VudCk7Ci0gICAgICAgICAgICAgICAgbGVhZGluZyA9IE1hdGgubWF4KGxlYWRpbmcsIGZpeGVkRGVzY2VudCArIG1ldHJpY3MubGVhZGluZyk7Ci0gICAgICAgICAgICB9IGVsc2UgeyAvLyBQb3NpdGlvbiBpcyBub3QgZml4ZWQgYnkgdGhlIGJhc2VsaW5lLCBuZWVkIHN1bSBvZiBhc2NlbnQgYW5kIGRlc2NlbnQKLSAgICAgICAgICAgICAgICBmbG9hdCBoZWlnaHQgPSBtZXRyaWNzLmFzY2VudCArIG1ldHJpY3MuZGVzY2VudDsKLQotICAgICAgICAgICAgICAgIG1heEhlaWdodCA9IE1hdGgubWF4KG1heEhlaWdodCwgaGVpZ2h0KTsKLSAgICAgICAgICAgICAgICBtYXhIZWlnaHRMZWFkaW5nID0gTWF0aC5tYXgobWF4SGVpZ2h0TGVhZGluZywgaGVpZ2h0ICsgbWV0cmljcy5sZWFkaW5nKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIE5lZWQgdG8gaW5jcmVhc2Ugc2l6ZXMgZm9yIGdyYXBoaWNzPwotICAgICAgICBpZiAobWF4SGVpZ2h0TGVhZGluZyAhPSAwKSB7Ci0gICAgICAgICAgICBkZXNjZW50ID0gTWF0aC5tYXgoZGVzY2VudCwgbWF4SGVpZ2h0IC0gYXNjZW50KTsKLSAgICAgICAgICAgIGxlYWRpbmcgPSBNYXRoLm1heChsZWFkaW5nLCBtYXhIZWlnaHRMZWFkaW5nIC0gYXNjZW50KTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIE5vcm1hbGl6ZSBsZWFkaW5nCi0gICAgICAgIGxlYWRpbmcgLT0gZGVzY2VudDsKLQotICAgICAgICBCYXNpY01ldHJpY3MgY3Vyck1ldHJpY3M7Ci0gICAgICAgIGZsb2F0IGN1cnJBZHZhbmNlID0gMDsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHNlZ21lbnRzLnNpemUoKTsgaSsrKSB7Ci0gICAgICAgICAgICBUZXh0UnVuU2VnbWVudCBzZWdtZW50ID0gc2VnbWVudHMuZ2V0KGJyZWFrZXIuZ2V0U2VnbWVudEZyb21WaXN1YWxPcmRlcihpKSk7Ci0gICAgICAgICAgICBjdXJyTWV0cmljcyA9IHNlZ21lbnQubWV0cmljczsKLQotICAgICAgICAgICAgc2VnbWVudC55ID0gZ2V0QmFzZWxpbmVPZmZzZXQoY3Vyck1ldHJpY3MuYmFzZUxpbmVJbmRleCkKLSAgICAgICAgICAgICAgICAgICAgKyBjdXJyTWV0cmljcy5zdXBlclNjcmlwdE9mZnNldDsKLSAgICAgICAgICAgIHNlZ21lbnQueCA9IGN1cnJBZHZhbmNlOwotCi0gICAgICAgICAgICBjdXJyQWR2YW5jZSArPSBzZWdtZW50LmdldEFkdmFuY2UoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGFkdmFuY2UgPSBjdXJyQWR2YW5jZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21wdXRlcyBtZXRyaWNzIGFuZCBjcmVhdGVzIEJhc2ljTWV0cmljcyBvYmplY3QgZnJvbSB0aGVtCi0gICAgICogQHJldHVybiBiYXNpYyBtZXRyaWNzCi0gICAgICovCi0gICAgcHVibGljIEJhc2ljTWV0cmljcyBjcmVhdGVNZXRyaWNzKCkgewotICAgICAgICBjb21wdXRlTWV0cmljcygpOwotICAgICAgICByZXR1cm4gbmV3IEJhc2ljTWV0cmljcyh0aGlzKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb3JyZWN0cyBhZHZhbmNlIGFmdGVyIGp1c3RpZmljYXRpb24uIEdldHMgQmFzaWNNZXRyaWNzIG9iamVjdAotICAgICAqIGFuZCB1cGRhdGVzIGFkdmFuY2Ugc3RvcmVkIGludG8gaXQuCi0gICAgICogQHBhcmFtIG1ldHJpY3MgLSBtZXRyaWNzIHdpdGggb3V0ZGF0ZWQgYWR2YW5jZSB3aGljaCBzaG91bGQgYmUgY29ycmVjdGVkIAotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGNvcnJlY3RBZHZhbmNlKEJhc2ljTWV0cmljcyBtZXRyaWNzKSB7Ci0gICAgICAgIEFycmF5TGlzdDxUZXh0UnVuU2VnbWVudD4gc2VnbWVudHMgPSBicmVha2VyLnJ1blNlZ21lbnRzOwotICAgICAgICBUZXh0UnVuU2VnbWVudCBzZWdtZW50ID0gc2VnbWVudHMuZ2V0KGJyZWFrZXIKLSAgICAgICAgICAgICAgICAuZ2V0U2VnbWVudEZyb21WaXN1YWxPcmRlcihzZWdtZW50cy5zaXplKCkgLSAxKSk7Ci0KLSAgICAgICAgYWR2YW5jZSA9IHNlZ21lbnQueCArIHNlZ21lbnQuZ2V0QWR2YW5jZSgpOwotICAgICAgICBtZXRyaWNzLmFkdmFuY2UgPSBhZHZhbmNlOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvVGV4dFJ1bkJyZWFrZXIuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvVGV4dFJ1bkJyZWFrZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYmU2MDZmNy4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L1RleHRSdW5CcmVha2VyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4NjEgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKgotICovCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi0KLWltcG9ydCBqYXZhLmF3dC5nZW9tLkdlbmVyYWxQYXRoOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci1pbXBvcnQgamF2YS5hd3QuaW0uSW5wdXRNZXRob2RIaWdobGlnaHQ7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC4qOwotaW1wb3J0IGphdmEuYXd0Lio7Ci1pbXBvcnQgamF2YS50ZXh0LkF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvcjsKLWltcG9ydCBqYXZhLnRleHQuQW5ub3RhdGlvbjsKLWltcG9ydCBqYXZhLnRleHQuQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yLkF0dHJpYnV0ZTsKLWltcG9ydCBqYXZhLnV0aWwuKjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5UZXh0RGVjb3JhdG9yLkRlY29yYXRpb247Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lm1pc2MuSGFzaENvZGU7Ci0vLyBUT0RPIC0gYmlkaSBub3QgaW1wbGVtZW50ZWQgeWV0Ci0KLS8qKgotICogVGhpcyBjbGFzcyBpcyByZXNwb25zaWJsZSBmb3IgYnJlYWtpbmcgdGhlIHRleHQgaW50byB0aGUgcnVuIHNlZ21lbnRzCi0gKiB3aXRoIGNvbnN0YW50IGZvbnQsIHN0eWxlLCBvdGhlciB0ZXh0IGF0dHJpYnV0ZXMgYW5kIGRpcmVjdGlvbi4KLSAqIEl0IGFsc28gc3RvcmVzIHRoZSBjcmVhdGVkIHRleHQgcnVuIHNlZ21lbnRzIGFuZCBjb3ZlcnMgZnVuY3Rpb25hbGl0eQotICogcmVsYXRlZCB0byB0aGUgb3BlcmF0aW9ucyBvbiB0aGUgc2V0IG9mIHNlZ21lbnRzLCBsaWtlIGNhbGN1bGF0aW5nIG1ldHJpY3MsCi0gKiByZW5kZXJpbmcsIGp1c3RpZmljYXRpb24sIGhpdCB0ZXN0aW5nLCBldGMuCi0gKi8KLXB1YmxpYyBjbGFzcyBUZXh0UnVuQnJlYWtlciBpbXBsZW1lbnRzIENsb25lYWJsZSB7Ci0gICAgQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIGFjaTsKLSAgICBGb250UmVuZGVyQ29udGV4dCBmcmM7Ci0KLSAgICBjaGFyW10gdGV4dDsKLQotICAgIGJ5dGVbXSBsZXZlbHM7Ci0KLSAgICBIYXNoTWFwPEludGVnZXIsIEZvbnQ+IGZvbnRzOwotICAgIEhhc2hNYXA8SW50ZWdlciwgRGVjb3JhdGlvbj4gZGVjb3JhdGlvbnM7Ci0KLSAgICAvLyBSZWxhdGVkIHRvIGRlZmF1bHQgZm9udCBzdWJzdGl0dXRpb24KLSAgICBpbnQgZm9yY2VkRm9udFJ1blN0YXJ0c1tdOwotCi0gICAgQXJyYXlMaXN0PFRleHRSdW5TZWdtZW50PiBydW5TZWdtZW50cyA9IG5ldyBBcnJheUxpc3Q8VGV4dFJ1blNlZ21lbnQ+KCk7Ci0KLSAgICAvLyBGb3IgZmFzdCByZXRyaWV2aW5nIG9mIHRoZSBzZWdtZW50IGNvbnRhaW5pbmcKLSAgICAvLyBjaGFyYWN0ZXIgd2l0aCBrbm93biBsb2dpY2FsIGluZGV4Ci0gICAgaW50IGxvZ2ljYWwyc2VnbWVudFtdOwotICAgIGludCBzZWdtZW50MnZpc3VhbFtdOyAvLyBWaXN1YWwgb3JkZXIgb2Ygc2VnbWVudHMgVE9ETyAtIGltcGxlbWVudAotICAgIGludCB2aXN1YWwyc2VnbWVudFtdOwotICAgIGludCBsb2dpY2FsMnZpc3VhbFtdOwotICAgIGludCB2aXN1YWwybG9naWNhbFtdOwotCi0gICAgU2VnbWVudHNJbmZvIHN0b3JlZFNlZ21lbnRzOwotICAgIHByaXZhdGUgYm9vbGVhbiBoYXZlQWxsU2VnbWVudHMgPSBmYWxzZTsKLSAgICBpbnQgc2VnbWVudHNTdGFydCwgc2VnbWVudHNFbmQ7Ci0KLSAgICBmbG9hdCBqdXN0aWZpY2F0aW9uID0gMS4wZjsKLQotICAgIHB1YmxpYyBUZXh0UnVuQnJlYWtlcihBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgYWNpLCBGb250UmVuZGVyQ29udGV4dCBmcmMpIHsKLSAgICAgICAgdGhpcy5hY2kgPSBhY2k7Ci0gICAgICAgIHRoaXMuZnJjID0gZnJjOwotCi0gICAgICAgIHNlZ21lbnRzU3RhcnQgPSBhY2kuZ2V0QmVnaW5JbmRleCgpOwotICAgICAgICBzZWdtZW50c0VuZCA9IGFjaS5nZXRFbmRJbmRleCgpOwotCi0gICAgICAgIGludCBsZW4gPSBzZWdtZW50c0VuZCAtIHNlZ21lbnRzU3RhcnQ7Ci0gICAgICAgIHRleHQgPSBuZXcgY2hhcltsZW5dOwotICAgICAgICBhY2kuc2V0SW5kZXgoc2VnbWVudHNFbmQpOwotICAgICAgICB3aGlsZSAobGVuLS0gIT0gMCkgeyAvLyBHb2luZyBpbiBiYWNrd2FyZCBkaXJlY3Rpb24gaXMgZmFzdGVyPyBTaW1wbGllciBjaGVja3MgaGVyZT8KLSAgICAgICAgICAgIHRleHRbbGVuXSA9IGFjaS5wcmV2aW91cygpOwotICAgICAgICB9Ci0KLSAgICAgICAgY3JlYXRlU3R5bGVSdW5zKCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVmlzdWFsIG9yZGVyIG9mIHRleHQgc2VnbWVudHMgbWF5IGRpZmZlciBmcm9tIHRoZSBsb2dpY2FsIG9yZGVyLgotICAgICAqIFRoaXMgbWV0aG9kIGNhbGN1bGF0ZXMgdmlzdWFsIHBvc2l0aW9uIG9mIHRoZSBzZWdtZW50IGZyb20gaXRzIGxvZ2ljYWwgcG9zaXRpb24uCi0gICAgICogQHBhcmFtIHNlZ21lbnROdW0gLSBsb2dpY2FsIHBvc2l0aW9uIG9mIHRoZSBzZWdtZW50Ci0gICAgICogQHJldHVybiB2aXN1YWwgcG9zaXRpb24gb2YgdGhlIHNlZ21lbnQKLSAgICAgKi8KLSAgICBpbnQgZ2V0VmlzdWFsRnJvbVNlZ21lbnRPcmRlcihpbnQgc2VnbWVudE51bSkgewotICAgICAgICByZXR1cm4gKHNlZ21lbnQydmlzdWFsID09IG51bGwpID8gc2VnbWVudE51bSA6IHNlZ21lbnQydmlzdWFsW3NlZ21lbnROdW1dOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFZpc3VhbCBvcmRlciBvZiB0ZXh0IHNlZ21lbnRzIG1heSBkaWZmZXIgZnJvbSB0aGUgbG9naWNhbCBvcmRlci4KLSAgICAgKiBUaGlzIG1ldGhvZCBjYWxjdWxhdGVzIGxvZ2ljYWwgcG9zaXRpb24gb2YgdGhlIHNlZ21lbnQgZnJvbSBpdHMgdmlzdWFsIHBvc2l0aW9uLgotICAgICAqIEBwYXJhbSB2aXN1YWwgLSB2aXN1YWwgcG9zaXRpb24gb2YgdGhlIHNlZ21lbnQKLSAgICAgKiBAcmV0dXJuIGxvZ2ljYWwgcG9zaXRpb24gb2YgdGhlIHNlZ21lbnQKLSAgICAgKi8KLSAgICBpbnQgZ2V0U2VnbWVudEZyb21WaXN1YWxPcmRlcihpbnQgdmlzdWFsKSB7Ci0gICAgICAgIHJldHVybiAodmlzdWFsMnNlZ21lbnQgPT0gbnVsbCkgPyB2aXN1YWwgOiB2aXN1YWwyc2VnbWVudFt2aXN1YWxdOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFZpc3VhbCBvcmRlciBvZiB0aGUgY2hhcmFjdGVycyBtYXkgZGlmZmVyIGZyb20gdGhlIGxvZ2ljYWwgb3JkZXIuCi0gICAgICogVGhpcyBtZXRob2QgY2FsY3VsYXRlcyB2aXN1YWwgcG9zaXRpb24gb2YgdGhlIGNoYXJhY3RlciBmcm9tIGl0cyBsb2dpY2FsIHBvc2l0aW9uLgotICAgICAqIEBwYXJhbSBsb2dpY2FsIC0gbG9naWNhbCBwb3NpdGlvbiBvZiB0aGUgY2hhcmFjdGVyCi0gICAgICogQHJldHVybiB2aXN1YWwgcG9zaXRpb24KLSAgICAgKi8KLSAgICBpbnQgZ2V0VmlzdWFsRnJvbUxvZ2ljYWwoaW50IGxvZ2ljYWwpIHsKLSAgICAgICAgcmV0dXJuIChsb2dpY2FsMnZpc3VhbCA9PSBudWxsKSA/IGxvZ2ljYWwgOiBsb2dpY2FsMnZpc3VhbFtsb2dpY2FsXTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBWaXN1YWwgb3JkZXIgb2YgdGhlIGNoYXJhY3RlcnMgbWF5IGRpZmZlciBmcm9tIHRoZSBsb2dpY2FsIG9yZGVyLgotICAgICAqIFRoaXMgbWV0aG9kIGNhbGN1bGF0ZXMgbG9naWNhbCBwb3NpdGlvbiBvZiB0aGUgY2hhcmFjdGVyIGZyb20gaXRzIHZpc3VhbCBwb3NpdGlvbi4KLSAgICAgKiBAcGFyYW0gdmlzdWFsIC0gdmlzdWFsIHBvc2l0aW9uCi0gICAgICogQHJldHVybiBsb2dpY2FsIHBvc2l0aW9uCi0gICAgICovCi0gICAgaW50IGdldExvZ2ljYWxGcm9tVmlzdWFsKGludCB2aXN1YWwpIHsKLSAgICAgICAgcmV0dXJuICh2aXN1YWwybG9naWNhbCA9PSBudWxsKSA/IHZpc3VhbCA6IHZpc3VhbDJsb2dpY2FsW3Zpc3VhbF07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2FsY3VsYXRlcyB0aGUgZW5kIGluZGV4IG9mIHRoZSBsZXZlbCBydW4sIGxpbWl0ZWQgYnkgdGhlIGdpdmVuIHRleHQgcnVuLgotICAgICAqIEBwYXJhbSBydW5TdGFydCAtIHJ1biBzdGFydAotICAgICAqIEBwYXJhbSBydW5FbmQgLSBydW4gZW5kCi0gICAgICogQHJldHVybiBlbmQgaW5kZXggb2YgdGhlIGxldmVsIHJ1bgotICAgICAqLwotICAgIGludCBnZXRMZXZlbFJ1bkxpbWl0KGludCBydW5TdGFydCwgaW50IHJ1bkVuZCkgewotICAgICAgICBpZiAobGV2ZWxzID09IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBydW5FbmQ7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IGVuZExldmVsUnVuID0gcnVuU3RhcnQgKyAxOwotICAgICAgICBieXRlIGxldmVsID0gbGV2ZWxzW3J1blN0YXJ0XTsKLQotICAgICAgICB3aGlsZSAoZW5kTGV2ZWxSdW4gPD0gcnVuRW5kICYmIGxldmVsc1tlbmRMZXZlbFJ1bl0gPT0gbGV2ZWwpIHsKLSAgICAgICAgICAgIGVuZExldmVsUnVuKys7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZW5kTGV2ZWxSdW47Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyBJbnB1dE1ldGhvZEhpZ2hsaWdodCB0byB0aGUgYXR0cmlidXRlcwotICAgICAqIEBwYXJhbSBhdHRycyAtIHRleHQgYXR0cmlidXRlcwotICAgICAqIEByZXR1cm4gcGF0Y2hlZCB0ZXh0IGF0dHJpYnV0ZXMKLSAgICAgKi8KLSAgICBNYXA8PyBleHRlbmRzIEF0dHJpYnV0ZSwgPz4gdW5wYWNrQXR0cmlidXRlcyhNYXA8PyBleHRlbmRzIEF0dHJpYnV0ZSwgPz4gYXR0cnMpIHsKLSAgICAgICAgaWYgKGF0dHJzLmNvbnRhaW5zS2V5KFRleHRBdHRyaWJ1dGUuSU5QVVRfTUVUSE9EX0hJR0hMSUdIVCkpIHsKLSAgICAgICAgICAgIE1hcDxUZXh0QXR0cmlidXRlLCA/PiBzdHlsZXMgPSBudWxsOwotCi0gICAgICAgICAgICBPYmplY3QgdmFsID0gYXR0cnMuZ2V0KFRleHRBdHRyaWJ1dGUuSU5QVVRfTUVUSE9EX0hJR0hMSUdIVCk7Ci0KLSAgICAgICAgICAgIGlmICh2YWwgaW5zdGFuY2VvZiBBbm5vdGF0aW9uKSB7Ci0gICAgICAgICAgICAgICAgdmFsID0gKChBbm5vdGF0aW9uKSB2YWwpLmdldFZhbHVlKCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmICh2YWwgaW5zdGFuY2VvZiBJbnB1dE1ldGhvZEhpZ2hsaWdodCkgewotICAgICAgICAgICAgICAgIElucHV0TWV0aG9kSGlnaGxpZ2h0IGlobCA9ICgoSW5wdXRNZXRob2RIaWdobGlnaHQpIHZhbCk7Ci0gICAgICAgICAgICAgICAgc3R5bGVzID0gaWhsLmdldFN0eWxlKCk7Ci0KLSAgICAgICAgICAgICAgICBpZiAoc3R5bGVzID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgVG9vbGtpdCB0ayA9IFRvb2xraXQuZ2V0RGVmYXVsdFRvb2xraXQoKTsKLSAgICAgICAgICAgICAgICAgICAgc3R5bGVzID0gdGsubWFwSW5wdXRNZXRob2RIaWdobGlnaHQoaWhsKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChzdHlsZXMgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIEhhc2hNYXA8QXR0cmlidXRlLCBPYmplY3Q+IG5ld0F0dHJzID0gbmV3IEhhc2hNYXA8QXR0cmlidXRlLCBPYmplY3Q+KCk7Ci0gICAgICAgICAgICAgICAgbmV3QXR0cnMucHV0QWxsKGF0dHJzKTsKLSAgICAgICAgICAgICAgICBuZXdBdHRycy5wdXRBbGwoc3R5bGVzKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gbmV3QXR0cnM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gYXR0cnM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQnJlYWtzIHRoZSB0ZXh0IGludG8gc2VwYXJhdGUgc3R5bGUgcnVucy4KLSAgICAgKi8KLSAgICB2b2lkIGNyZWF0ZVN0eWxlUnVucygpIHsKLSAgICAgICAgLy8gVE9ETyAtIGltcGxlbWVudCBmYXN0IGFuZCBzaW1wbGUgY2FzZQotICAgICAgICBmb250cyA9IG5ldyBIYXNoTWFwPEludGVnZXIsIEZvbnQ+KCk7Ci0gICAgICAgIGRlY29yYXRpb25zID0gbmV3IEhhc2hNYXA8SW50ZWdlciwgRGVjb3JhdGlvbj4oKTsKLSAgICAgICAgLy8vLwotCi0gICAgICAgIEFycmF5TGlzdDxJbnRlZ2VyPiBmb3JjZWRGb250UnVuU3RhcnRzTGlzdCA9IG51bGw7Ci0KLSAgICAgICAgTWFwPD8gZXh0ZW5kcyBBdHRyaWJ1dGUsID8+IGF0dHJpYnV0ZXMgPSBudWxsOwotCi0gICAgICAgIC8vIENoZWNrIGp1c3RpZmljYXRpb24gYXR0cmlidXRlCi0gICAgICAgIE9iamVjdCB2YWwgPSBhY2kuZ2V0QXR0cmlidXRlKFRleHRBdHRyaWJ1dGUuSlVTVElGSUNBVElPTik7Ci0gICAgICAgIGlmICh2YWwgIT0gbnVsbCkgewotICAgICAgICAgICAganVzdGlmaWNhdGlvbiA9ICgoRmxvYXQpIHZhbCkuZmxvYXRWYWx1ZSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yICgKLSAgICAgICAgICAgIGludCBpbmRleCA9IHNlZ21lbnRzU3RhcnQsIG5leHRSdW5TdGFydCA9IHNlZ21lbnRzU3RhcnQ7Ci0gICAgICAgICAgICBpbmRleCA8IHNlZ21lbnRzRW5kOwotICAgICAgICAgICAgaW5kZXggPSBuZXh0UnVuU3RhcnQsIGFjaS5zZXRJbmRleChpbmRleCkKLSAgICAgICAgICAgKSAgewotICAgICAgICAgICAgbmV4dFJ1blN0YXJ0ID0gYWNpLmdldFJ1bkxpbWl0KCk7Ci0gICAgICAgICAgICBhdHRyaWJ1dGVzID0gdW5wYWNrQXR0cmlidXRlcyhhY2kuZ2V0QXR0cmlidXRlcygpKTsKLQotICAgICAgICAgICAgVGV4dERlY29yYXRvci5EZWNvcmF0aW9uIGQgPSBUZXh0RGVjb3JhdG9yLmdldERlY29yYXRpb24oYXR0cmlidXRlcyk7Ci0gICAgICAgICAgICBkZWNvcmF0aW9ucy5wdXQobmV3IEludGVnZXIoaW5kZXgpLCBkKTsKLQotICAgICAgICAgICAgLy8gRmluZCBhcHByb3ByaWF0ZSBmb250IG9yIHBsYWNlIEdyYXBoaWNBdHRyaWJ1dGUgdGhlcmUKLQotICAgICAgICAgICAgLy8gMS4gVHJ5IHRvIHBpY2sgdXAgQ0hBUl9SRVBMQUNFTUVOVCAoY29tcGF0aWJpbGl0eSkKLSAgICAgICAgICAgIEZvbnQgdmFsdWUgPSAoRm9udClhdHRyaWJ1dGVzLmdldChUZXh0QXR0cmlidXRlLkNIQVJfUkVQTEFDRU1FTlQpOwotCi0gICAgICAgICAgICBpZiAodmFsdWUgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIC8vIDIuIFRyeSB0byBHZXQgRk9OVAotICAgICAgICAgICAgICAgIHZhbHVlID0gKEZvbnQpYXR0cmlidXRlcy5nZXQoVGV4dEF0dHJpYnV0ZS5GT05UKTsKLQotICAgICAgICAgICAgICAgIGlmICh2YWx1ZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIDMuIFRyeSB0byBjcmVhdGUgZm9udCBmcm9tIEZBTUlMWQotICAgICAgICAgICAgICAgICAgICBpZiAoYXR0cmlidXRlcy5nZXQoVGV4dEF0dHJpYnV0ZS5GQU1JTFkpICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlID0gRm9udC5nZXRGb250KGF0dHJpYnV0ZXMpOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgaWYgKHZhbHVlID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIDQuIE5vIGF0dHJpYnV0ZXMgZm91bmQsIHVzaW5nIGRlZmF1bHQuCi0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoZm9yY2VkRm9udFJ1blN0YXJ0c0xpc3QgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZvcmNlZEZvbnRSdW5TdGFydHNMaXN0ID0gbmV3IEFycmF5TGlzdDxJbnRlZ2VyPigpOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgRm9udEZpbmRlci5maW5kRm9udHMoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRleHQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGluZGV4LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXh0UnVuU3RhcnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZvcmNlZEZvbnRSdW5TdGFydHNMaXN0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmb250cwotICAgICAgICAgICAgICAgICAgICAgICAgKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlID0gZm9udHMuZ2V0KG5ldyBJbnRlZ2VyKGluZGV4KSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGZvbnRzLnB1dChuZXcgSW50ZWdlcihpbmRleCksIHZhbHVlKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIFdlIGhhdmUgYWRkZWQgc29tZSBkZWZhdWx0IGZvbnRzLCBzbyB3ZSBoYXZlIHNvbWUgZXh0cmEgcnVucyBpbiB0ZXh0Ci0gICAgICAgIGlmIChmb3JjZWRGb250UnVuU3RhcnRzTGlzdCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBmb3JjZWRGb250UnVuU3RhcnRzID0gbmV3IGludFtmb3JjZWRGb250UnVuU3RhcnRzTGlzdC5zaXplKCldOwotICAgICAgICAgICAgZm9yIChpbnQgaT0wOyBpPGZvcmNlZEZvbnRSdW5TdGFydHNMaXN0LnNpemUoKTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgZm9yY2VkRm9udFJ1blN0YXJ0c1tpXSA9Ci0gICAgICAgICAgICAgICAgICAgICAgICBmb3JjZWRGb250UnVuU3RhcnRzTGlzdC5nZXQoaSkuaW50VmFsdWUoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFN0YXJ0aW5nIGZyb20gdGhlIGN1cnJlbnQgcG9zaXRpb24gbG9va3MgZm9yIHRoZSBlbmQgb2YgdGhlIHRleHQgcnVuIHdpdGgKLSAgICAgKiBjb25zdGFudCB0ZXh0IGF0dHJpYnV0ZXMuCi0gICAgICogQHBhcmFtIHJ1blN0YXJ0IC0gc3RhcnQgcG9zaXRpb24KLSAgICAgKiBAcGFyYW0gbWF4UG9zIC0gcG9zaXRpb24gd2hlcmUgdG8gc3RvcCBpZiBubyBydW4gbGltaXQgZm91bmQKLSAgICAgKiBAcmV0dXJuIHN0eWxlIHJ1biBsaW1pdAotICAgICAqLwotICAgIGludCBnZXRTdHlsZVJ1bkxpbWl0KGludCBydW5TdGFydCwgaW50IG1heFBvcykgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgYWNpLnNldEluZGV4KHJ1blN0YXJ0KTsKLSAgICAgICAgfSBjYXRjaChJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24gZSkgeyAvLyBJbmRleCBvdXQgb2YgYm91bmRzCi0gICAgICAgICAgICBpZiAocnVuU3RhcnQgPCBzZWdtZW50c1N0YXJ0KSB7Ci0gICAgICAgICAgICAgICAgYWNpLmZpcnN0KCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGFjaS5sYXN0KCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBJZiB3ZSBoYXZlIHNvbWUgZXh0cmEgcnVucyB3ZSBuZWVkIHRvIGNoZWNrIGZvciB0aGVpciBsaW1pdHMKLSAgICAgICAgaWYgKGZvcmNlZEZvbnRSdW5TdGFydHMgIT0gbnVsbCkgewotICAgICAgICAgICAgZm9yIChpbnQgZWxlbWVudCA6IGZvcmNlZEZvbnRSdW5TdGFydHMpIHsKLSAgICAgICAgICAgICAgICBpZiAoZWxlbWVudCA+IHJ1blN0YXJ0KSB7Ci0gICAgICAgICAgICAgICAgICAgIG1heFBvcyA9IE1hdGgubWluKGVsZW1lbnQsIG1heFBvcyk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBNYXRoLm1pbihhY2kuZ2V0UnVuTGltaXQoKSwgbWF4UG9zKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHNlZ21lbnRzIGZvciB0aGUgdGV4dCBydW4gd2l0aAotICAgICAqIGNvbnN0YW50IGRlY29yYXRpb24sIGZvbnQgYW5kIGJpZGkgbGV2ZWwKLSAgICAgKiBAcGFyYW0gcnVuU3RhcnQgLSBydW4gc3RhcnQKLSAgICAgKiBAcGFyYW0gcnVuRW5kIC0gcnVuIGVuZAotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGNyZWF0ZVNlZ21lbnRzKGludCBydW5TdGFydCwgaW50IHJ1bkVuZCkgewotICAgICAgICBpbnQgZW5kU3R5bGVSdW4sIGVuZExldmVsUnVuOwotCi0gICAgICAgIC8vIFRPRE8gLSB1cGRhdGUgbGV2ZWxzCi0KLSAgICAgICAgaW50IHBvcyA9IHJ1blN0YXJ0LCBsZXZlbFBvczsKLQotICAgICAgICBhY2kuc2V0SW5kZXgocG9zKTsKLSAgICAgICAgZmluYWwgaW50IGZpcnN0UnVuU3RhcnQgPSBhY2kuZ2V0UnVuU3RhcnQoKTsKLSAgICAgICAgT2JqZWN0IHRkZCA9IGRlY29yYXRpb25zLmdldChuZXcgSW50ZWdlcihmaXJzdFJ1blN0YXJ0KSk7Ci0gICAgICAgIE9iamVjdCBmb250T3JHQXR0ciA9IGZvbnRzLmdldChuZXcgSW50ZWdlcihmaXJzdFJ1blN0YXJ0KSk7Ci0KLSAgICAgICAgbG9naWNhbDJzZWdtZW50ID0gbmV3IGludFtydW5FbmQgLSBydW5TdGFydF07Ci0KLSAgICAgICAgZG8gewotICAgICAgICAgICAgZW5kU3R5bGVSdW4gPSBnZXRTdHlsZVJ1bkxpbWl0KHBvcywgcnVuRW5kKTsKLQotICAgICAgICAgICAgLy8gcnVuU3RhcnQgY2FuIGJlIG5vbi16ZXJvLCBidXQgYWxsIGFycmF5cyB3aWxsIGJlIGluZGV4ZWQgZnJvbSAwCi0gICAgICAgICAgICBpbnQgYWp1c3RlZFBvcyA9IHBvcyAtIHJ1blN0YXJ0OwotICAgICAgICAgICAgaW50IGFqdXN0ZWRFbmRTdHlsZVJ1biA9IGVuZFN0eWxlUnVuIC0gcnVuU3RhcnQ7Ci0gICAgICAgICAgICBsZXZlbFBvcyA9IGFqdXN0ZWRQb3M7Ci0gICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgZW5kTGV2ZWxSdW4gPSBnZXRMZXZlbFJ1bkxpbWl0KGxldmVsUG9zLCBhanVzdGVkRW5kU3R5bGVSdW4pOwotCi0gICAgICAgICAgICAgICAgaWYgKGZvbnRPckdBdHRyIGluc3RhbmNlb2YgR3JhcGhpY0F0dHJpYnV0ZSkgewotICAgICAgICAgICAgICAgICAgICBydW5TZWdtZW50cy5hZGQoCi0gICAgICAgICAgICAgICAgICAgICAgICBuZXcgVGV4dFJ1blNlZ21lbnRJbXBsLlRleHRSdW5TZWdtZW50R3JhcGhpYygKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKEdyYXBoaWNBdHRyaWJ1dGUpZm9udE9yR0F0dHIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGVuZExldmVsUnVuIC0gbGV2ZWxQb3MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldmVsUG9zICsgcnVuU3RhcnQpCi0gICAgICAgICAgICAgICAgICAgICk7Ci0gICAgICAgICAgICAgICAgICAgIEFycmF5cy5maWxsKGxvZ2ljYWwyc2VnbWVudCwgbGV2ZWxQb3MsIGVuZExldmVsUnVuLCBydW5TZWdtZW50cy5zaXplKCktMSk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgVGV4dFJ1blNlZ21lbnRJbXBsLlRleHRTZWdtZW50SW5mbyBpID0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXcgVGV4dFJ1blNlZ21lbnRJbXBsLlRleHRTZWdtZW50SW5mbygKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldmVscyA9PSBudWxsID8gMCA6IGxldmVsc1thanVzdGVkUG9zXSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChGb250KSBmb250T3JHQXR0ciwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZyYywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRleHQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsZXZlbFBvcyArIHJ1blN0YXJ0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZW5kTGV2ZWxSdW4gKyBydW5TdGFydAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICk7Ci0KLSAgICAgICAgICAgICAgICAgICAgcnVuU2VnbWVudHMuYWRkKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5ldyBUZXh0UnVuU2VnbWVudEltcGwuVGV4dFJ1blNlZ21lbnRDb21tb24oCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKFRleHREZWNvcmF0b3IuRGVjb3JhdGlvbikgdGRkCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKQotICAgICAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgICAgICAgICBBcnJheXMuZmlsbChsb2dpY2FsMnNlZ21lbnQsIGxldmVsUG9zLCBlbmRMZXZlbFJ1biwgcnVuU2VnbWVudHMuc2l6ZSgpLTEpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGxldmVsUG9zID0gZW5kTGV2ZWxSdW47Ci0gICAgICAgICAgICB9IHdoaWxlIChsZXZlbFBvcyA8IGFqdXN0ZWRFbmRTdHlsZVJ1bik7Ci0KLSAgICAgICAgICAgIC8vIFByZXBhcmUgbmV4dCBpdGVyYXRpb24KLSAgICAgICAgICAgIHBvcyA9IGVuZFN0eWxlUnVuOwotICAgICAgICAgICAgdGRkID0gZGVjb3JhdGlvbnMuZ2V0KG5ldyBJbnRlZ2VyKHBvcykpOwotICAgICAgICAgICAgZm9udE9yR0F0dHIgPSBmb250cy5nZXQobmV3IEludGVnZXIocG9zKSk7Ci0gICAgICAgIH0gd2hpbGUgKHBvcyA8IHJ1bkVuZCk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRleHQgcnVuIHNlZ21lbnRzIGFyZSB1cCB0byBkYXRlIGFuZCBjcmVhdGVzIHRoZSBuZXcgc2VnbWVudHMgaWYgbm90LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGNyZWF0ZUFsbFNlZ21lbnRzKCkgewotICAgICAgICBpZiAoICFoYXZlQWxsU2VnbWVudHMgJiYKLSAgICAgICAgICAgIChsb2dpY2FsMnNlZ21lbnQgPT0gbnVsbCB8fAotICAgICAgICAgICAgIGxvZ2ljYWwyc2VnbWVudC5sZW5ndGggIT0gc2VnbWVudHNFbmQgLSBzZWdtZW50c1N0YXJ0KQotICAgICAgICApIHsgLy8gQ2hlY2sgaWYgd2UgZG9uJ3QgaGF2ZSBhbGwgc2VnbWVudHMgeWV0Ci0gICAgICAgICAgICByZXNldFNlZ21lbnRzKCk7Ci0gICAgICAgICAgICBjcmVhdGVTZWdtZW50cyhzZWdtZW50c1N0YXJ0LCBzZWdtZW50c0VuZCk7Ci0gICAgICAgIH0KLQotICAgICAgICBoYXZlQWxsU2VnbWVudHMgPSB0cnVlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENhbGN1bGF0ZXMgcG9zaXRpb24gd2hlcmUgbGluZSBzaG91bGQgYmUgYnJva2VuIHdpdGhvdXQKLSAgICAgKiB0YWtpbmcgaW50byBhY2NvdW50IHdvcmQgYm91bmRhcmllcy4KLSAgICAgKiBAcGFyYW0gc3RhcnQgLSBzdGFydCBpbmRleAotICAgICAqIEBwYXJhbSBtYXhBZHZhbmNlIC0gbWF4aW11bSBhZHZhbmNlLCB3aWR0aCBvZiB0aGUgbGluZQotICAgICAqIEByZXR1cm4gcG9zaXRpb24gd2hlcmUgdG8gYnJlYWsKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldExpbmVCcmVha0luZGV4KGludCBzdGFydCwgZmxvYXQgbWF4QWR2YW5jZSkgewotICAgICAgICBpbnQgYnJlYWtJbmRleDsKLSAgICAgICAgVGV4dFJ1blNlZ21lbnQgcyA9IG51bGw7Ci0KLSAgICAgICAgZm9yICgKLSAgICAgICAgICAgICAgICBpbnQgc2VnbWVudEluZGV4ID0gbG9naWNhbDJzZWdtZW50W3N0YXJ0XTsKLSAgICAgICAgICAgICAgICBzZWdtZW50SW5kZXggPCBydW5TZWdtZW50cy5zaXplKCk7Ci0gICAgICAgICAgICAgICAgc2VnbWVudEluZGV4KysKLSAgICAgICAgICAgKSB7Ci0gICAgICAgICAgICBzID0gcnVuU2VnbWVudHMuZ2V0KHNlZ21lbnRJbmRleCk7Ci0gICAgICAgICAgICBicmVha0luZGV4ID0gcy5nZXRDaGFySW5kZXhGcm9tQWR2YW5jZShtYXhBZHZhbmNlLCBzdGFydCk7Ci0KLSAgICAgICAgICAgIGlmIChicmVha0luZGV4IDwgcy5nZXRFbmQoKSkgewotICAgICAgICAgICAgICAgIHJldHVybiBicmVha0luZGV4OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgbWF4QWR2YW5jZSAtPSBzLmdldEFkdmFuY2VEZWx0YShzdGFydCwgcy5nZXRFbmQoKSk7Ci0gICAgICAgICAgICBzdGFydCA9IHMuZ2V0RW5kKCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gcy5nZXRFbmQoKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBJbnNlcnRzIGNoYXJhY3RlciBpbnRvIHRoZSBtYW5hZ2VkIHRleHQuCi0gICAgICogQHBhcmFtIG5ld1BhcmFncmFwaCAtIG5ldyBjaGFyYWN0ZXIgaXRlcmF0b3IKLSAgICAgKiBAcGFyYW0gaW5zZXJ0UG9zIC0gaW5zZXJ0aW9uIHBvc2l0aW9uCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgaW5zZXJ0Q2hhcihBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgbmV3UGFyYWdyYXBoLCBpbnQgaW5zZXJ0UG9zKSB7Ci0gICAgICAgIGFjaSA9IG5ld1BhcmFncmFwaDsKLQotICAgICAgICBjaGFyIGluc0NoYXIgPSBhY2kuc2V0SW5kZXgoaW5zZXJ0UG9zKTsKLQotICAgICAgICBJbnRlZ2VyIGtleSA9IG5ldyBJbnRlZ2VyKGluc2VydFBvcyk7Ci0KLSAgICAgICAgaW5zZXJ0UG9zIC09IGFjaS5nZXRCZWdpbkluZGV4KCk7Ci0KLSAgICAgICAgY2hhciBuZXdUZXh0W10gPSBuZXcgY2hhclt0ZXh0Lmxlbmd0aCArIDFdOwotICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHRleHQsIDAsIG5ld1RleHQsIDAsIGluc2VydFBvcyk7Ci0gICAgICAgIG5ld1RleHRbaW5zZXJ0UG9zXSA9IGluc0NoYXI7Ci0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkodGV4dCwgaW5zZXJ0UG9zLCBuZXdUZXh0LCBpbnNlcnRQb3MrMSwgdGV4dC5sZW5ndGggLSBpbnNlcnRQb3MpOwotICAgICAgICB0ZXh0ID0gbmV3VGV4dDsKLQotICAgICAgICBpZiAoYWNpLmdldFJ1blN0YXJ0KCkgPT0ga2V5LmludFZhbHVlKCkgJiYgYWNpLmdldFJ1bkxpbWl0KCkgPT0ga2V5LmludFZhbHVlKCkgKyAxKSB7Ci0gICAgICAgICAgICBjcmVhdGVTdHlsZVJ1bnMoKTsgLy8gV2UgaGF2ZSB0byBjcmVhdGUgb25lIG5ldyBydW4sIGNvdWxkIGJlIG9wdGltaXplZAotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgc2hpZnRTdHlsZVJ1bnMoa2V5LCAxKTsKLSAgICAgICAgfQotCi0gICAgICAgIHJlc2V0U2VnbWVudHMoKTsKLQotICAgICAgICBzZWdtZW50c0VuZCsrOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIERlbGV0ZXMgY2hhcmFjdGVyIGZyb20gdGhlIG1hbmFnZWQgdGV4dC4KLSAgICAgKiBAcGFyYW0gbmV3UGFyYWdyYXBoIC0gbmV3IGNoYXJhY3RlciBpdGVyYXRvcgotICAgICAqIEBwYXJhbSBkZWxldGVQb3MgLSBkZWxldGlvbiBwb3NpdGlvbgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGRlbGV0ZUNoYXIoQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIG5ld1BhcmFncmFwaCwgaW50IGRlbGV0ZVBvcykgewotICAgICAgICBhY2kgPSBuZXdQYXJhZ3JhcGg7Ci0KLSAgICAgICAgSW50ZWdlciBrZXkgPSBuZXcgSW50ZWdlcihkZWxldGVQb3MpOwotCi0gICAgICAgIGRlbGV0ZVBvcyAtPSBhY2kuZ2V0QmVnaW5JbmRleCgpOwotCi0gICAgICAgIGNoYXIgbmV3VGV4dFtdID0gbmV3IGNoYXJbdGV4dC5sZW5ndGggLSAxXTsKLSAgICAgICAgU3lzdGVtLmFycmF5Y29weSh0ZXh0LCAwLCBuZXdUZXh0LCAwLCBkZWxldGVQb3MpOwotICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHRleHQsIGRlbGV0ZVBvcysxLCBuZXdUZXh0LCBkZWxldGVQb3MsIG5ld1RleHQubGVuZ3RoIC0gZGVsZXRlUG9zKTsKLSAgICAgICAgdGV4dCA9IG5ld1RleHQ7Ci0KLSAgICAgICAgaWYgKGZvbnRzLmdldChrZXkpICE9IG51bGwpIHsKLSAgICAgICAgICAgIGZvbnRzLnJlbW92ZShrZXkpOwotICAgICAgICB9Ci0KLSAgICAgICAgc2hpZnRTdHlsZVJ1bnMoa2V5LCAtMSk7Ci0KLSAgICAgICAgcmVzZXRTZWdtZW50cygpOwotCi0gICAgICAgIHNlZ21lbnRzRW5kLS07Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2hpZnQgYWxsIHJ1bnMgYWZ0ZXIgc3BlY2lmaWVkIHBvc2l0aW9uLCBuZWVkZWQgdG8gcGVyZm9tIGluc2VydGlvbgotICAgICAqIG9yIGRlbGV0aW9uIGluIHRoZSBtYW5hZ2VkIHRleHQKLSAgICAgKiBAcGFyYW0gcG9zIC0gcG9zaXRpb24gd2hlcmUgdG8gc3RhcnQKLSAgICAgKiBAcGFyYW0gc2hpZnQgLSBzaGlmdCwgY291bGQgYmUgbmVnYXRpdmUKLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgc2hpZnRTdHlsZVJ1bnMoSW50ZWdlciBwb3MsIGZpbmFsIGludCBzaGlmdCkgewotICAgICAgICBBcnJheUxpc3Q8SW50ZWdlcj4ga2V5cyA9IG5ldyBBcnJheUxpc3Q8SW50ZWdlcj4oKTsKLQotICAgICAgICBJbnRlZ2VyIGtleSwgb2xka2V5OwotICAgICAgICBmb3IgKEl0ZXJhdG9yPEludGVnZXI+IGl0ID0gZm9udHMua2V5U2V0KCkuaXRlcmF0b3IoKTsgaXQuaGFzTmV4dCgpOyApIHsKLSAgICAgICAgICAgIG9sZGtleSA9IGl0Lm5leHQoKTsKLSAgICAgICAgICAgIGlmIChvbGRrZXkuaW50VmFsdWUoKSA+IHBvcy5pbnRWYWx1ZSgpKSB7Ci0gICAgICAgICAgICAgICAga2V5cy5hZGQob2xka2V5KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGk9MDsgaTxrZXlzLnNpemUoKTsgaSsrKSB7Ci0gICAgICAgICAgICBvbGRrZXkgPSBrZXlzLmdldChpKTsKLSAgICAgICAgICAgIGtleSA9IG5ldyBJbnRlZ2VyKHNoaWZ0ICsgb2xka2V5LmludFZhbHVlKCkpOwotICAgICAgICAgICAgZm9udHMucHV0KGtleSwgZm9udHMucmVtb3ZlKG9sZGtleSkpOwotICAgICAgICAgICAgZGVjb3JhdGlvbnMucHV0KGtleSwgZGVjb3JhdGlvbnMucmVtb3ZlKG9sZGtleSkpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVzZXRzIHN0YXRlIG9mIHRoZSBjbGFzcwotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCByZXNldFNlZ21lbnRzKCkgewotICAgICAgICBydW5TZWdtZW50cyA9IG5ldyBBcnJheUxpc3Q8VGV4dFJ1blNlZ21lbnQ+KCk7Ci0gICAgICAgIGxvZ2ljYWwyc2VnbWVudCA9IG51bGw7Ci0gICAgICAgIHNlZ21lbnQydmlzdWFsID0gbnVsbDsKLSAgICAgICAgdmlzdWFsMnNlZ21lbnQgPSBudWxsOwotICAgICAgICBsZXZlbHMgPSBudWxsOwotICAgICAgICBoYXZlQWxsU2VnbWVudHMgPSBmYWxzZTsKLSAgICB9Ci0KLSAgICBwcml2YXRlIGNsYXNzIFNlZ21lbnRzSW5mbyB7Ci0gICAgICAgIEFycmF5TGlzdDxUZXh0UnVuU2VnbWVudD4gcnVuU2VnbWVudHM7Ci0gICAgICAgIGludCBsb2dpY2FsMnNlZ21lbnRbXTsKLSAgICAgICAgaW50IHNlZ21lbnQydmlzdWFsW107Ci0gICAgICAgIGludCB2aXN1YWwyc2VnbWVudFtdOwotICAgICAgICBieXRlIGxldmVsc1tdOwotICAgICAgICBpbnQgc2VnbWVudHNTdGFydDsKLSAgICAgICAgaW50IHNlZ21lbnRzRW5kOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNhdmVzIHRoZSBpbnRlcm5hbCBzdGF0ZSBvZiB0aGUgY2xhc3MKLSAgICAgKiBAcGFyYW0gbmV3U2VnU3RhcnQgLSBuZXcgc3RhcnQgaW5kZXggaW4gdGhlIHRleHQKLSAgICAgKiBAcGFyYW0gbmV3U2VnRW5kIC0gbmV3IGVuZCBpbmRleCBpbiB0aGUgdGV4dAotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHB1c2hTZWdtZW50cyhpbnQgbmV3U2VnU3RhcnQsIGludCBuZXdTZWdFbmQpIHsKLSAgICAgICAgc3RvcmVkU2VnbWVudHMgPSBuZXcgU2VnbWVudHNJbmZvKCk7Ci0gICAgICAgIHN0b3JlZFNlZ21lbnRzLnJ1blNlZ21lbnRzID0gdGhpcy5ydW5TZWdtZW50czsKLSAgICAgICAgc3RvcmVkU2VnbWVudHMubG9naWNhbDJzZWdtZW50ID0gdGhpcy5sb2dpY2FsMnNlZ21lbnQ7Ci0gICAgICAgIHN0b3JlZFNlZ21lbnRzLnNlZ21lbnQydmlzdWFsID0gdGhpcy5zZWdtZW50MnZpc3VhbDsKLSAgICAgICAgc3RvcmVkU2VnbWVudHMudmlzdWFsMnNlZ21lbnQgPSB0aGlzLnZpc3VhbDJzZWdtZW50OwotICAgICAgICBzdG9yZWRTZWdtZW50cy5sZXZlbHMgPSB0aGlzLmxldmVsczsKLSAgICAgICAgc3RvcmVkU2VnbWVudHMuc2VnbWVudHNTdGFydCA9IHNlZ21lbnRzU3RhcnQ7Ci0gICAgICAgIHN0b3JlZFNlZ21lbnRzLnNlZ21lbnRzRW5kID0gc2VnbWVudHNFbmQ7Ci0KLSAgICAgICAgcmVzZXRTZWdtZW50cygpOwotCi0gICAgICAgIHNlZ21lbnRzU3RhcnQgPSBuZXdTZWdTdGFydDsKLSAgICAgICAgc2VnbWVudHNFbmQgPSBuZXdTZWdFbmQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVzdG9yZXMgdGhlIGludGVybmFsIHN0YXRlIG9mIHRoZSBjbGFzcwotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHBvcFNlZ21lbnRzKCkgewotICAgICAgICBpZiAoc3RvcmVkU2VnbWVudHMgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgdGhpcy5ydW5TZWdtZW50cyA9IHN0b3JlZFNlZ21lbnRzLnJ1blNlZ21lbnRzOwotICAgICAgICB0aGlzLmxvZ2ljYWwyc2VnbWVudCA9IHN0b3JlZFNlZ21lbnRzLmxvZ2ljYWwyc2VnbWVudDsKLSAgICAgICAgdGhpcy5zZWdtZW50MnZpc3VhbCA9IHN0b3JlZFNlZ21lbnRzLnNlZ21lbnQydmlzdWFsOwotICAgICAgICB0aGlzLnZpc3VhbDJzZWdtZW50ID0gc3RvcmVkU2VnbWVudHMudmlzdWFsMnNlZ21lbnQ7Ci0gICAgICAgIHRoaXMubGV2ZWxzID0gc3RvcmVkU2VnbWVudHMubGV2ZWxzOwotICAgICAgICB0aGlzLnNlZ21lbnRzU3RhcnQgPSBzdG9yZWRTZWdtZW50cy5zZWdtZW50c1N0YXJ0OwotICAgICAgICB0aGlzLnNlZ21lbnRzRW5kID0gc3RvcmVkU2VnbWVudHMuc2VnbWVudHNFbmQ7Ci0gICAgICAgIHN0b3JlZFNlZ21lbnRzID0gbnVsbDsKLQotICAgICAgICBpZiAocnVuU2VnbWVudHMuc2l6ZSgpID09IDAgJiYgbG9naWNhbDJzZWdtZW50ID09IG51bGwpIHsKLSAgICAgICAgICAgIGhhdmVBbGxTZWdtZW50cyA9IGZhbHNlOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaGF2ZUFsbFNlZ21lbnRzID0gdHJ1ZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBPYmplY3QgY2xvbmUoKSB7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBUZXh0UnVuQnJlYWtlciByZXMgPSAoVGV4dFJ1bkJyZWFrZXIpIHN1cGVyLmNsb25lKCk7Ci0gICAgICAgICAgICByZXMuc3RvcmVkU2VnbWVudHMgPSBudWxsOwotICAgICAgICAgICAgQXJyYXlMaXN0PFRleHRSdW5TZWdtZW50PiBuZXdTZWdtZW50cyA9IG5ldyBBcnJheUxpc3Q8VGV4dFJ1blNlZ21lbnQ+KHJ1blNlZ21lbnRzLnNpemUoKSk7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHJ1blNlZ21lbnRzLnNpemUoKTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgVGV4dFJ1blNlZ21lbnQgc2VnID0gIHJ1blNlZ21lbnRzLmdldChpKTsKLSAgICAgICAgICAgICAgICBuZXdTZWdtZW50cy5hZGQoKFRleHRSdW5TZWdtZW50KXNlZy5jbG9uZSgpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJlcy5ydW5TZWdtZW50cyA9IG5ld1NlZ21lbnRzOwotICAgICAgICAgICAgcmV0dXJuIHJlczsKLSAgICAgICAgfSBjYXRjaCAoQ2xvbmVOb3RTdXBwb3J0ZWRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgLy8gYXd0LjNFPUNsb25lIG5vdCBzdXBwb3J0ZWQKLSAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4zRSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvYmopIHsKLSAgICAgICAgaWYgKCEob2JqIGluc3RhbmNlb2YgVGV4dFJ1bkJyZWFrZXIpKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBUZXh0UnVuQnJlYWtlciBiciA9IChUZXh0UnVuQnJlYWtlcikgb2JqOwotCi0gICAgICAgIGlmIChici5nZXRBQ0koKS5lcXVhbHMoYWNpKSAmJiBici5mcmMuZXF1YWxzKGZyYykpIHsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgaGFzaENvZGUoKSB7Ci0gICAgICAgIHJldHVybiBIYXNoQ29kZS5jb21iaW5lKGFjaS5oYXNoQ29kZSgpLCBmcmMuaGFzaENvZGUoKSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVuZGVycyB0aGUgbWFuYWdlZCB0ZXh0Ci0gICAgICogQHBhcmFtIGcyZCAtIGdyYXBoaWNzIHdoZXJlIHRvIHJlbmRlcgotICAgICAqIEBwYXJhbSB4T2Zmc2V0IC0gb2Zmc2V0IGluIFggZGlyZWN0aW9uIHRvIHRoZSB1cHBlciBsZWZ0IGNvcm5lcgotICAgICAqIG9mIHRoZSBsYXlvdXQgZnJvbSB0aGUgb3JpZ2luIG9mIHRoZSBncmFwaGljcwotICAgICAqIEBwYXJhbSB5T2Zmc2V0IC0gb2Zmc2V0IGluIFkgZGlyZWN0aW9uIHRvIHRoZSB1cHBlciBsZWZ0IGNvcm5lcgotICAgICAqIG9mIHRoZSBsYXlvdXQgZnJvbSB0aGUgb3JpZ2luIG9mIHRoZSBncmFwaGljcwotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIGRyYXdTZWdtZW50cyhHcmFwaGljczJEIGcyZCwgZmxvYXQgeE9mZnNldCwgZmxvYXQgeU9mZnNldCkgewotICAgICAgICBmb3IgKGludCBpPTA7IGk8cnVuU2VnbWVudHMuc2l6ZSgpOyBpKyspIHsKLSAgICAgICAgICAgIHJ1blNlZ21lbnRzLmdldChpKS5kcmF3KGcyZCwgeE9mZnNldCwgeU9mZnNldCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRoZSBibGFjayBib3ggYm91bmRzIHNoYXBlCi0gICAgICogQHBhcmFtIGZpcnN0RW5kcG9pbnQgLSBzdGFydCBwb3NpdGlvbgotICAgICAqIEBwYXJhbSBzZWNvbmRFbmRwb2ludCAtIGVuZCBwb3NpdGlvbgotICAgICAqIEByZXR1cm4gYmxhY2sgYm94IGJvdW5kcyBzaGFwZQotICAgICAqLwotICAgIHB1YmxpYyBTaGFwZSBnZXRCbGFja0JveEJvdW5kcyhpbnQgZmlyc3RFbmRwb2ludCwgaW50IHNlY29uZEVuZHBvaW50KSB7Ci0gICAgICAgIEdlbmVyYWxQYXRoIGJvdW5kcyA9IG5ldyBHZW5lcmFsUGF0aCgpOwotCi0gICAgICAgIFRleHRSdW5TZWdtZW50IHNlZ21lbnQ7Ci0KLSAgICAgICAgZm9yIChpbnQgaWR4ID0gZmlyc3RFbmRwb2ludDsgaWR4IDwgc2Vjb25kRW5kcG9pbnQ7IGlkeD1zZWdtZW50LmdldEVuZCgpKSB7Ci0gICAgICAgICAgICBzZWdtZW50ID0gcnVuU2VnbWVudHMuZ2V0KGxvZ2ljYWwyc2VnbWVudFtpZHhdKTsKLSAgICAgICAgICAgIGJvdW5kcy5hcHBlbmQoc2VnbWVudC5nZXRDaGFyc0JsYWNrQm94Qm91bmRzKGlkeCwgc2Vjb25kRW5kcG9pbnQpLCBmYWxzZSk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gYm91bmRzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgdmlzdWFsIGJvdW5kcyBzaGFwZQotICAgICAqIEByZXR1cm4gdmlzdWFsIGJvdW5kcyByZWN0YW5nbGUKLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlMkQgZ2V0VmlzdWFsQm91bmRzKCkgewotICAgICAgICBSZWN0YW5nbGUyRCBib3VuZHMgPSBudWxsOwotCi0gICAgICAgIGZvciAoaW50IGk9MDsgaTxydW5TZWdtZW50cy5zaXplKCk7IGkrKykgewotICAgICAgICAgICAgVGV4dFJ1blNlZ21lbnQgcyA9IHJ1blNlZ21lbnRzLmdldChpKTsKLSAgICAgICAgICAgIGlmIChib3VuZHMgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIFJlY3RhbmdsZTJELnVuaW9uKGJvdW5kcywgcy5nZXRWaXN1YWxCb3VuZHMoKSwgYm91bmRzKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgYm91bmRzID0gcy5nZXRWaXN1YWxCb3VuZHMoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBib3VuZHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBsb2dpY2FsIGJvdW5kcyBzaGFwZQotICAgICAqIEByZXR1cm4gbG9naWNhbCBib3VuZHMgcmVjdGFuZ2xlCi0gICAgICovCi0gICAgcHVibGljIFJlY3RhbmdsZTJEIGdldExvZ2ljYWxCb3VuZHMoKSB7Ci0gICAgICAgIFJlY3RhbmdsZTJEIGJvdW5kcyA9IG51bGw7Ci0KLSAgICAgICAgZm9yIChpbnQgaT0wOyBpPHJ1blNlZ21lbnRzLnNpemUoKTsgaSsrKSB7Ci0gICAgICAgICAgICBUZXh0UnVuU2VnbWVudCBzID0gcnVuU2VnbWVudHMuZ2V0KGkpOwotICAgICAgICAgICAgaWYgKGJvdW5kcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgUmVjdGFuZ2xlMkQudW5pb24oYm91bmRzLCBzLmdldExvZ2ljYWxCb3VuZHMoKSwgYm91bmRzKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgYm91bmRzID0gcy5nZXRMb2dpY2FsQm91bmRzKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gYm91bmRzOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0Q2hhckNvdW50KCkgewotICAgICAgICByZXR1cm4gc2VnbWVudHNFbmQgLSBzZWdtZW50c1N0YXJ0OwotICAgIH0KLQotICAgIHB1YmxpYyBieXRlIGdldExldmVsKGludCBpZHgpIHsKLSAgICAgICAgaWYgKGxldmVscyA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gMDsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gbGV2ZWxzW2lkeF07Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRCYXNlTGV2ZWwoKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGlzTFRSKCkgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgY2hhciBnZXRDaGFyKGludCBpbmRleCkgewotICAgICAgICByZXR1cm4gdGV4dFtpbmRleF07Ci0gICAgfQotCi0gICAgcHVibGljIEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvciBnZXRBQ0koKSB7Ci0gICAgICAgIHJldHVybiBhY2k7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBvdXRsaW5lIHNoYXBlIGZvciB0aGUgbWFuYWdlZCB0ZXh0Ci0gICAgICogQHJldHVybiBvdXRsaW5lCi0gICAgICovCi0gICAgcHVibGljIEdlbmVyYWxQYXRoIGdldE91dGxpbmUoKSB7Ci0gICAgICAgIEdlbmVyYWxQYXRoIG91dGxpbmUgPSBuZXcgR2VuZXJhbFBhdGgoKTsKLQotICAgICAgICBUZXh0UnVuU2VnbWVudCBzZWdtZW50OwotCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcnVuU2VnbWVudHMuc2l6ZSgpOyBpKyspIHsKLSAgICAgICAgICAgIHNlZ21lbnQgPSBydW5TZWdtZW50cy5nZXQoaSk7Ci0gICAgICAgICAgICBvdXRsaW5lLmFwcGVuZChzZWdtZW50LmdldE91dGxpbmUoKSwgZmFsc2UpOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG91dGxpbmU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2FsY3VsYXRlcyB0ZXh0IGhpdCBpbmZvIGZyb20gdGhlIHNjcmVlbiBjb29yZGluYXRlcy4KLSAgICAgKiBDdXJyZW50IGltcGxlbWVudGF0aW9uIHRvdGFsbHkgaWdub3JlcyBZIGNvb3JkaW5hdGUuCi0gICAgICogSWYgWCBjb29yZGluYXRlIGlzIG91dHNpZGUgb2YgdGhlIGxheW91dCBib3VuZGFyaWVzLCB0aGlzCi0gICAgICogbWV0aG9kIHJldHVybnMgbGVmdG1vc3Qgb3IgcmlnaHRtb3N0IGhpdC4KLSAgICAgKiBAcGFyYW0geCAtIHggY29vcmRpbmF0ZSBvZiB0aGUgaGl0Ci0gICAgICogQHBhcmFtIHkgLSB5IGNvb3JkaW5hdGUgb2YgdGhlIGhpdAotICAgICAqIEByZXR1cm4gaGl0IGluZm8KLSAgICAgKi8KLSAgICBwdWJsaWMgVGV4dEhpdEluZm8gaGl0VGVzdChmbG9hdCB4LCBmbG9hdCB5KSB7Ci0gICAgICAgIFRleHRSdW5TZWdtZW50IHNlZ21lbnQ7Ci0KLSAgICAgICAgZG91YmxlIGVuZE9mUHJldlNlZyA9IC0xOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHJ1blNlZ21lbnRzLnNpemUoKTsgaSsrKSB7Ci0gICAgICAgICAgICBzZWdtZW50ID0gcnVuU2VnbWVudHMuZ2V0KGkpOwotICAgICAgICAgICAgUmVjdGFuZ2xlMkQgYm91bmRzID0gc2VnbWVudC5nZXRWaXN1YWxCb3VuZHMoKTsKLSAgICAgICAgICAgIGlmICgoYm91bmRzLmdldE1pblgoKSA8PSB4ICYmIGJvdW5kcy5nZXRNYXhYKCkgPj0geCkgfHwgLy8gV2UgYXJlIGluIHRoZSBzZWdtZW50Ci0gICAgICAgICAgICAgICAoZW5kT2ZQcmV2U2VnIDwgeCAmJiBib3VuZHMuZ2V0TWluWCgpID4geCkpIHsgLy8gV2UgYXJlIHNvbWV3aGVyZSBiZXR3ZWVuIHRoZSBzZWdtZW50cwotICAgICAgICAgICAgICAgIHJldHVybiBzZWdtZW50LmhpdFRlc3QoeCx5KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGVuZE9mUHJldlNlZyA9IGJvdW5kcy5nZXRNYXhYKCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gaXNMVFIoKSA/IFRleHRIaXRJbmZvLnRyYWlsaW5nKHRleHQubGVuZ3RoKSA6IFRleHRIaXRJbmZvLmxlYWRpbmcoMCk7Ci0gICAgfQotCi0gICAgcHVibGljIGZsb2F0IGdldEp1c3RpZmljYXRpb24oKSB7Ci0gICAgICAgIHJldHVybiBqdXN0aWZpY2F0aW9uOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENhbGN1bGF0ZXMgcG9zaXRpb24gb2YgdGhlIGxhc3Qgbm9uIHdoaXRlc3BhY2UgY2hhcmFjdGVyCi0gICAgICogaW4gdGhlIG1hbmFnZWQgdGV4dC4KLSAgICAgKiBAcmV0dXJuIHBvc2l0aW9uIG9mIHRoZSBsYXN0IG5vbiB3aGl0ZXNwYWNlIGNoYXJhY3RlcgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0TGFzdE5vbldoaXRlc3BhY2UoKSB7Ci0gICAgICAgIGludCBsYXN0Tm9uV2hpdGVzcGFjZSA9IHRleHQubGVuZ3RoOwotCi0gICAgICAgIHdoaWxlIChsYXN0Tm9uV2hpdGVzcGFjZSA+PSAwKSB7Ci0gICAgICAgICAgICBsYXN0Tm9uV2hpdGVzcGFjZS0tOwotICAgICAgICAgICAgaWYgKCFDaGFyYWN0ZXIuaXNXaGl0ZXNwYWNlKHRleHRbbGFzdE5vbldoaXRlc3BhY2VdKSkgewotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGxhc3ROb25XaGl0ZXNwYWNlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFBlcmZvcm1zIGp1c3RpZmljYXRpb24gb2YgdGhlIG1hbmFnZWQgdGV4dCBieSBjaGFuZ2luZyBzZWdtZW50IHBvc2l0aW9ucwotICAgICAqIGFuZCBwb3NpdGlvbnMgb2YgdGhlIGdseXBocyBpbnNpZGUgb2YgdGhlIHNlZ21lbnRzLgotICAgICAqIEBwYXJhbSBnYXAgLSBhbW91bnQgb2Ygc3BhY2Ugd2hpY2ggc2hvdWxkIGJlIGNvbXBlbnNhdGVkIGJ5IGp1c3RpZmljYXRpb24KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBqdXN0aWZ5KGZsb2F0IGdhcCkgewotICAgICAgICAvLyBJZ25vcmUgdHJhaWxpbmcgbG9naWNhbCB3aGl0ZXNwYWNlCi0gICAgICAgIGludCBmaXJzdElkeCA9IHNlZ21lbnRzU3RhcnQ7Ci0gICAgICAgIGludCBsYXN0SWR4ID0gZ2V0TGFzdE5vbldoaXRlc3BhY2UoKSArIHNlZ21lbnRzU3RhcnQ7Ci0gICAgICAgIEp1c3RpZmljYXRpb25JbmZvIGpJbmZvc1tdID0gbmV3IEp1c3RpZmljYXRpb25JbmZvWzVdOwotICAgICAgICBmbG9hdCBnYXBMZWZ0ID0gZ2FwOwotCi0gICAgICAgIGludCBoaWdoZXN0UHJpb3JpdHkgPSAtMTsKLSAgICAgICAgLy8gR2x5cGhKdXN0aWZpY2F0aW9uSW5mby5QUklPUklUWV9LQVNISURBIGlzIDAKLSAgICAgICAgLy8gR2x5cGhKdXN0aWZpY2F0aW9uSW5mby5QUklPUklUWV9OT05FIGlzIDMKLSAgICAgICAgZm9yIChpbnQgcHJpb3JpdHkgPSAwOyBwcmlvcml0eSA8PSBHbHlwaEp1c3RpZmljYXRpb25JbmZvLlBSSU9SSVRZX05PTkUgKyAxOyBwcmlvcml0eSsrKSB7Ci0gICAgICAgICAgICBKdXN0aWZpY2F0aW9uSW5mbyBqSW5mbyA9IG5ldyBKdXN0aWZpY2F0aW9uSW5mbygpOwotICAgICAgICAgICAgakluZm8ubGFzdElkeCA9IGxhc3RJZHg7Ci0gICAgICAgICAgICBqSW5mby5maXJzdElkeCA9IGZpcnN0SWR4OwotICAgICAgICAgICAgakluZm8uZ3JvdyA9IGdhcCA+IDA7Ci0gICAgICAgICAgICBqSW5mby5nYXBUb0ZpbGwgPSBnYXBMZWZ0OwotCi0gICAgICAgICAgICBpZiAocHJpb3JpdHkgPD0gR2x5cGhKdXN0aWZpY2F0aW9uSW5mby5QUklPUklUWV9OT05FKSB7Ci0gICAgICAgICAgICAgICAgakluZm8ucHJpb3JpdHkgPSBwcmlvcml0eTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgakluZm8ucHJpb3JpdHkgPSBoaWdoZXN0UHJpb3JpdHk7IC8vIExhc3QgcGFzcwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IHJ1blNlZ21lbnRzLnNpemUoKTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgVGV4dFJ1blNlZ21lbnQgc2VnbWVudCA9IHJ1blNlZ21lbnRzLmdldChpKTsKLSAgICAgICAgICAgICAgICBpZiAoc2VnbWVudC5nZXRTdGFydCgpIDw9IGxhc3RJZHgpIHsKLSAgICAgICAgICAgICAgICAgICAgc2VnbWVudC51cGRhdGVKdXN0aWZpY2F0aW9uSW5mbyhqSW5mbyk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoakluZm8ucHJpb3JpdHkgPT0gaGlnaGVzdFByaW9yaXR5KSB7Ci0gICAgICAgICAgICAgICAgakluZm8uYWJzb3JiID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICBqSW5mby5hYnNvcmJlZFdlaWdodCA9IGpJbmZvLndlaWdodDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKGpJbmZvLndlaWdodCAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGhpZ2hlc3RQcmlvcml0eSA8IDApIHsKLSAgICAgICAgICAgICAgICAgICAgaGlnaGVzdFByaW9yaXR5ID0gcHJpb3JpdHk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGpJbmZvc1twcmlvcml0eV0gPSBqSW5mbzsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGdhcExlZnQgLT0gakluZm8uZ3Jvd0xpbWl0OwotCi0gICAgICAgICAgICBpZiAoKChnYXBMZWZ0ID4gMCkgXiBqSW5mby5ncm93KSB8fCBnYXBMZWZ0ID09IDApIHsKLSAgICAgICAgICAgICAgICBnYXBMZWZ0ID0gMDsKLSAgICAgICAgICAgICAgICBqSW5mby5nYXBQZXJVbml0ID0gakluZm8uZ2FwVG9GaWxsL2pJbmZvLndlaWdodDsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGpJbmZvLnVzZUxpbWl0cyA9IHRydWU7Ci0KLSAgICAgICAgICAgIGlmIChqSW5mby5hYnNvcmJlZFdlaWdodCA+IDApIHsKLSAgICAgICAgICAgICAgICBqSW5mby5hYnNvcmIgPSB0cnVlOwotICAgICAgICAgICAgICAgIGpJbmZvLmFic29yYmVkR2FwUGVyVW5pdCA9Ci0gICAgICAgICAgICAgICAgICAgICAgICAoakluZm8uZ2FwVG9GaWxsLWpJbmZvLmdyb3dMaW1pdCkvakluZm8uYWJzb3JiZWRXZWlnaHQ7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBmbG9hdCBjdXJySnVzdGlmaWNhdGlvbk9mZnNldCA9IDA7Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgcnVuU2VnbWVudHMuc2l6ZSgpOyBpKyspIHsKLSAgICAgICAgICAgIFRleHRSdW5TZWdtZW50IHNlZ21lbnQgPQotICAgICAgICAgICAgICAgICAgICBydW5TZWdtZW50cy5nZXQoZ2V0U2VnbWVudEZyb21WaXN1YWxPcmRlcihpKSk7Ci0gICAgICAgICAgICBzZWdtZW50LnggKz0gY3Vyckp1c3RpZmljYXRpb25PZmZzZXQ7Ci0gICAgICAgICAgICBjdXJySnVzdGlmaWNhdGlvbk9mZnNldCArPSBzZWdtZW50LmRvSnVzdGlmaWNhdGlvbihqSW5mb3MpOwotICAgICAgICB9Ci0KLSAgICAgICAganVzdGlmaWNhdGlvbiA9IC0xOyAvLyBNYWtlIGZ1cnRoZXIganVzdGlmaWNhdGlvbiBpbXBvc3NpYmxlCi0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhpcyBjbGFzcyByZXByZXNlbnRzIHRoZSBpbmZvcm1hdGlvbiBjb2xsZWN0ZWQgYmVmb3JlIHRoZSBhY3R1YWwKLSAgICAgKiBqdXN0aWZpY2F0aW9uIGlzIHN0YXJ0ZWQgYW5kIG5lZWRlZCB0byBwZXJmb3JtIHRoZSBqdXN0aWZpY2F0aW9uLgotICAgICAqIFRoaXMgaW5mb3JtYXRpb24gaXMgY2xvc2VseSByZWxhdGVkIHRvIHRoZSBpbmZvcm1hdGlvbiBzdG9yZWQgaW4gdGhlCi0gICAgICogR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBmb3IgdGhlIHRleHQgcmVwcmVzZW50ZWQgYnkgZ2x5cGggdmVjdG9ycy4KLSAgICAgKi8KLSAgICBjbGFzcyBKdXN0aWZpY2F0aW9uSW5mbyB7Ci0gICAgICAgIGJvb2xlYW4gZ3JvdzsKLSAgICAgICAgYm9vbGVhbiBhYnNvcmIgPSBmYWxzZTsKLSAgICAgICAgYm9vbGVhbiB1c2VMaW1pdHMgPSBmYWxzZTsKLSAgICAgICAgaW50IHByaW9yaXR5ID0gMDsKLSAgICAgICAgZmxvYXQgd2VpZ2h0ID0gMDsKLSAgICAgICAgZmxvYXQgYWJzb3JiZWRXZWlnaHQgPSAwOwotICAgICAgICBmbG9hdCBncm93TGltaXQgPSAwOwotCi0gICAgICAgIGludCBsYXN0SWR4OwotICAgICAgICBpbnQgZmlyc3RJZHg7Ci0KLSAgICAgICAgZmxvYXQgZ2FwVG9GaWxsOwotCi0gICAgICAgIGZsb2F0IGdhcFBlclVuaXQgPSAwOyAvLyBQcmVjYWxjdWxhdGVkIHZhbHVlLCBnYXBUb0ZpbGwgLyB3ZWlnaHQKLSAgICAgICAgZmxvYXQgYWJzb3JiZWRHYXBQZXJVbml0ID0gMDsgLy8gUHJlY2FsY3VsYXRlZCB2YWx1ZSwgZ2FwVG9GaWxsIC8gd2VpZ2h0Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9UZXh0UnVuU2VnbWVudC5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvZm9udC9UZXh0UnVuU2VnbWVudC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxY2QyYzA1Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvVGV4dFJ1blNlZ21lbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE2NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3MyRDsKLWltcG9ydCBqYXZhLmF3dC5TaGFwZTsKLWltcG9ydCBqYXZhLmF3dC5mb250LlRleHRIaXRJbmZvOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUmVjdGFuZ2xlMkQ7Ci0KLS8qKgotICogQWJzdHJhY3QgY2xhc3Mgd2hpY2ggcmVwcmVzZW50cyB0aGUgc2VnbWVudCBvZiB0aGUgdGV4dCB3aXRoIGNvbnN0YW50IGF0dHJpYnV0ZXMKLSAqIHJ1bm5pbmcgaW4gb25lIGRpcmVjdGlvbiAoaS5lLiBjb25zdGFudCBsZXZlbCkuCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBUZXh0UnVuU2VnbWVudCBpbXBsZW1lbnRzIENsb25lYWJsZSB7Ci0gICAgZmxvYXQgeDsgLy8gQ2FsY3VsYXRlZCB4IGxvY2F0aW9uIG9mIHRoaXMgc2VnbWVudCBvbiB0aGUgc2NyZWVuCi0gICAgZmxvYXQgeTsgLy8gQ2FsY3VsYXRlZCB5IGxvY2F0aW9uIG9mIHRoaXMgc2VnbWVudCBvbiB0aGUgc2NyZWVuCi0KLSAgICBCYXNpY01ldHJpY3MgbWV0cmljczsgLy8gTWV0cmljcyBvZiB0aGlzIHRleHQgcnVuIHNlZ21lbnQKLSAgICBUZXh0RGVjb3JhdG9yLkRlY29yYXRpb24gZGVjb3JhdGlvbjsgLy8gVW5kZXJsaW5lLCBzcmlrZXRocm91Z2gsIGV0Yy4KLSAgICBSZWN0YW5nbGUyRCBsb2dpY2FsQm91bmRzID0gbnVsbDsgLy8gTG9naWNhbCBib3VuZGluZyBib3ggZm9yIHRoZSBzZWdtZW50Ci0gICAgUmVjdGFuZ2xlMkQgdmlzdWFsQm91bmRzID0gbnVsbDsgLy8gVmlzdWFsIGJvdW5kaW5nIGJveCBmb3IgdGhlIHNlZ21lbnQKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgc3RhcnQgaW5kZXggb2YgdGhlIHNlZ21lbnQKLSAgICAgKiBAcmV0dXJuIHN0YXJ0IGluZGV4Ci0gICAgICovCi0gICAgYWJzdHJhY3QgaW50IGdldFN0YXJ0KCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGVuZCBpbmRleCBvZiB0aGUgc2VnbWVudAotICAgICAqIEByZXR1cm4gZW5kIGluZGV4Ci0gICAgICovCi0gICAgYWJzdHJhY3QgaW50IGdldEVuZCgpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMgaW4gdGhlIHNlZ21lbnQKLSAgICAgKiBAcmV0dXJuIG51bWJlciBvZiBjaGFyYWN0ZXJzCi0gICAgICovCi0gICAgYWJzdHJhY3QgaW50IGdldExlbmd0aCgpOwotCi0gICAgLyoqCi0gICAgICogUmVuZGVycyB0aGlzIHRleHQgcnVuIHNlZ21lbnQKLSAgICAgKiBAcGFyYW0gZzJkIC0gZ3JhcGhpY3MgdG8gcmVuZGVyIHRvCi0gICAgICogQHBhcmFtIHhPZmZzZXQgLSBYIG9mZnNldCBmcm9tIHRoZSBncmFwaGljcyBvcmlnaW4gdG8gdGhlCi0gICAgICogb3JpZ2luIG9mIHRoZSB0ZXh0IGxheW91dAotICAgICAqIEBwYXJhbSB5T2Zmc2V0IC0gWSBvZmZzZXQgZnJvbSB0aGUgZ3JhcGhpY3Mgb3JpZ2luIHRvIHRoZQotICAgICAqIG9yaWdpbiBvZiB0aGUgdGV4dCBsYXlvdXQKLSAgICAgKi8KLSAgICBhYnN0cmFjdCB2b2lkIGRyYXcoR3JhcGhpY3MyRCBnMmQsIGZsb2F0IHhPZmZzZXQsIGZsb2F0IHlPZmZzZXQpOwotCi0gICAgLyoqCi0gICAgICogQ3JlYXRlcyBibGFjayBib3ggYm91bmRzIHNoYXBlIGZvciB0aGUgc3BlY2lmaWVkIHJhbmdlCi0gICAgICogQHBhcmFtIHN0YXJ0IC0gcmFuZ2Ugc2FydAotICAgICAqIEBwYXJhbSBsaW1pdCAtIHJhbmdlIGVuZAotICAgICAqIEByZXR1cm4gYmxhY2sgYm94IGJvdW5kcyBzaGFwZQotICAgICAqLwotICAgIGFic3RyYWN0IFNoYXBlIGdldENoYXJzQmxhY2tCb3hCb3VuZHMoaW50IHN0YXJ0LCBpbnQgbGltaXQpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgb3V0bGluZSBzaGFwZQotICAgICAqIEByZXR1cm4gb3V0bGluZQotICAgICAqLwotICAgIGFic3RyYWN0IFNoYXBlIGdldE91dGxpbmUoKTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdmlzdWFsIGJvdW5kcyBvZiB0aGlzIHNlZ21lbnQKLSAgICAgKiBAcmV0dXJuIHZpc3VhbCBib3VuZHMKLSAgICAgKi8KLSAgICBhYnN0cmFjdCBSZWN0YW5nbGUyRCBnZXRWaXN1YWxCb3VuZHMoKTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgbG9naWNhbCBib3VuZHMgb2YgdGhpcyBzZWdtZW50Ci0gICAgICogQHJldHVybiBsb2dpY2FsIGJvdW5kcwotICAgICAqLwotICAgIGFic3RyYWN0IFJlY3RhbmdsZTJEIGdldExvZ2ljYWxCb3VuZHMoKTsKLQotICAgIC8qKgotICAgICAqIENhbGN1bGF0ZXMgYWR2YW5jZSBvZiB0aGUgc2VnbWVudAotICAgICAqIEByZXR1cm4gYWR2YW5jZQotICAgICAqLwotICAgIGFic3RyYWN0IGZsb2F0IGdldEFkdmFuY2UoKTsKLQotICAgIC8qKgotICAgICAqIENhbGN1bGF0ZXMgYWR2YW5jZSBkZWx0YSBiZXR3ZWVuIHR3byBjaGFyYWN0ZXJzCi0gICAgICogQHBhcmFtIHN0YXJ0IC0gMXN0IHBvc2l0aW9uCi0gICAgICogQHBhcmFtIGVuZCAtIDJuZCBwb3NpdGlvbgotICAgICAqIEByZXR1cm4gYWR2YW5jZSBpbmNyZW1lbnQgYmV0d2VlbiBzcGVjaWZpZWQgcG9zaXRpb25zCi0gICAgICovCi0gICAgYWJzdHJhY3QgZmxvYXQgZ2V0QWR2YW5jZURlbHRhKGludCBzdGFydCwgaW50IGVuZCk7Ci0KLSAgICAvKioKLSAgICAgKiBDYWxjdWxhdGVzIGluZGV4IG9mIHRoZSBjaGFyYWN0ZXIgd2hpY2ggYWR2YW5jZSBpcyBlcXVhbCB0bwotICAgICAqIHRoZSBnaXZlbi4gSWYgdGhlIGdpdmVuIGFkdmFuY2UgaXMgZ3JlYXRlciB0aGVuIHRoZSBzZWdtZW50Ci0gICAgICogYWR2YW5jZSBpdCByZXR1cm5zIHRoZSBwb3NpdGlvbiBhZnRlciB0aGUgbGFzdCBjaGFyYWN0ZXIuCi0gICAgICogQHBhcmFtIGFkdmFuY2UgLSBnaXZlbiBhZHZhbmNlCi0gICAgICogQHBhcmFtIHN0YXJ0IC0gY2hhcmFjdGVyLCBmcm9tIHdoaWNoIHRvIHN0YXJ0IG1lYXN1cmluZyBhZHZhbmNlCi0gICAgICogQHJldHVybiBjaGFyYWN0ZXIgaW5kZXgKLSAgICAgKi8KLSAgICBhYnN0cmFjdCBpbnQgZ2V0Q2hhckluZGV4RnJvbUFkdmFuY2UoZmxvYXQgYWR2YW5jZSwgaW50IHN0YXJ0KTsKLQotICAgIC8qKgotICAgICAqIENoZWNrcyBpZiB0aGUgY2hhcmFjdGVyIGRvZXNuJ3QgY29udHJpYnV0ZSB0byB0aGUgdGV4dCBhZHZhbmNlCi0gICAgICogQHBhcmFtIGluZGV4IC0gY2hhcmFjdGVyIGluZGV4Ci0gICAgICogQHJldHVybiB0cnVlIGlmIHRoZSBjaGFyYWN0ZXIgaGFzIHplcm8gYWR2YW5jZQotICAgICAqLwotICAgIGFic3RyYWN0IGJvb2xlYW4gY2hhckhhc1plcm9BZHZhbmNlKGludCBpbmRleCk7Ci0KLSAgICAvKioKLSAgICAgKiBDYWxjdWxhdGVzIHBvc2l0aW9uIG9mIHRoZSBjaGFyYWN0ZXIgb24gdGhlIHNjcmVlbgotICAgICAqIEBwYXJhbSBpbmRleCAtIGNoYXJhY3RlciBpbmRleAotICAgICAqIEByZXR1cm4gWCBjb29yZGluYXRlIG9mIHRoZSBjaGFyYWN0ZXIgcG9zaXRpb24KLSAgICAgKi8KLSAgICBhYnN0cmFjdCBmbG9hdCBnZXRDaGFyUG9zaXRpb24oaW50IGluZGV4KTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGFkdmFuY2Ugb2YgdGhlIGluZGl2aWR1YWwgY2hhcmFjdGVyCi0gICAgICogQHBhcmFtIGluZGV4IC0gY2hhcmFjdGVyIGluZGV4Ci0gICAgICogQHJldHVybiBjaGFyYWN0ZXIgYWR2YW5jZQotICAgICAqLwotICAgIGFic3RyYWN0IGZsb2F0IGdldENoYXJBZHZhbmNlKGludCBpbmRleCk7Ci0KLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIHRleHQgaGl0IGluZm8gZnJvbSB0aGUgaGl0IHBvc2l0aW9uCi0gICAgICogQHBhcmFtIHggLSBYIGNvb3JkaW5hdGUgcmVsYXRpdmUgdG8gdGhlIG9yaWdpbiBvZiB0aGUgbGF5b3V0Ci0gICAgICogQHBhcmFtIHkgLSBZIGNvb3JkaW5hdGUgcmVsYXRpdmUgdG8gdGhlIG9yaWdpbiBvZiB0aGUgbGF5b3V0Ci0gICAgICogQHJldHVybiBoaXQgaW5mbwotICAgICAqLwotICAgIGFic3RyYWN0IFRleHRIaXRJbmZvIGhpdFRlc3QoZmxvYXQgeCwgZmxvYXQgeSk7Ci0KLSAgICAvKioKLSAgICAgKiBDb2xsZWN0cyBqdXN0aWZpY2F0aW9uIGluZm9ybWF0aW9uIGludG8gSnVzdGlmaWNhdGlvbkluZm8gb2JqZWN0Ci0gICAgICogQHBhcmFtIGpJbmZvIC0gSnVzdGlmaWNhdGlvbkluZm8gb2JqZWN0Ci0gICAgICovCi0gICAgYWJzdHJhY3Qgdm9pZCB1cGRhdGVKdXN0aWZpY2F0aW9uSW5mbyhUZXh0UnVuQnJlYWtlci5KdXN0aWZpY2F0aW9uSW5mbyBqSW5mbyk7Ci0KLSAgICAvKioKLSAgICAgKiBQZXJmb3JtcyBqdXN0aWZpY2F0aW9uIG9mIHRoZSBzZWdtZW50LgotICAgICAqIFVwZGF0ZXMgcG9zaXRpb25zIG9mIGluZGl2aWR1YWwgY2hhcmFjdGVycy4KLSAgICAgKiBAcGFyYW0gakluZm9zIC0ganVzdGlmaWNhdGlvbiBpbmZvcm1hdGlvbiwgZ2F0aGVyZWQgYnkgdGhlIHByZXZpb3VzIHBhc3NlcwotICAgICAqIEByZXR1cm4gYW1vdW50IG9mIGdyb3d0aCBvciBzaHJpbmsgb2YgdGhlIHNlZ21lbnQKLSAgICAgKi8gICAgCi0gICAgYWJzdHJhY3QgZmxvYXQgZG9KdXN0aWZpY2F0aW9uKFRleHRSdW5CcmVha2VyLkp1c3RpZmljYXRpb25JbmZvIGpJbmZvc1tdKTsKLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBhYnN0cmFjdCBPYmplY3QgY2xvbmUoKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvVGV4dFJ1blNlZ21lbnRJbXBsLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9mb250L1RleHRSdW5TZWdtZW50SW1wbC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwZWMyZDA1Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ZvbnQvVGV4dFJ1blNlZ21lbnRJbXBsLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw5NzkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKgotICovCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250OwotCi1pbXBvcnQgamF2YS5hd3QuKjsKLWltcG9ydCBqYXZhLmF3dC5mb250Lio7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5SZWN0YW5nbGUyRDsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkdlbmVyYWxQYXRoOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uUG9pbnQyRDsKLS8vIFhYWCAtIFRPRE8gLSBiaWRpIG5vdCBpbXBsZW1lbnRlZCB5ZXQKLS8vaW1wb3J0IGphdmEudGV4dC5CaWRpOwotaW1wb3J0IGphdmEudXRpbC5BcnJheXM7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBEYXRlOiBBcHIgMjUsIDIwMDUKLSAqIFRpbWU6IDQ6MzM6MTggUE0KLSAqCi0gKiBUaGlzIGNsYXNzIGNvbnRhaW5zIHRoZSBpbXBsZW1lbnRhdGlvbiBvZiB0aGUgYmVoYXZpb3Igb2YgdGhlCi0gKiB0ZXh0IHJ1biBzZWdtZW50IHdpdGggY29uc3RhbnQgdGV4dCBhdHRyaWJ1dGVzIGFuZCBkaXJlY3Rpb24uCi0gKi8KLXB1YmxpYyBjbGFzcyBUZXh0UnVuU2VnbWVudEltcGwgewotCi0gICAgLyoqCi0gICAgICogVGhpcyBjbGFzcyBjb250YWlucyBiYXNpYyBpbmZvcm1hdGlvbiByZXF1aXJlZCBmb3IgY3JlYXRpb24KLSAgICAgKiBvZiB0aGUgZ2x5cGgtYmFzZWQgdGV4dCBydW4gc2VnbWVudC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGNsYXNzIFRleHRTZWdtZW50SW5mbyB7Ci0gICAgICAgIC8vIFhYWCAtIFRPRE8gLSBiaWRpIG5vdCBpbXBsZW1lbnRlZCB5ZXQKLSAgICAgICAgLy9CaWRpIGJpZGk7Ci0KLSAgICAgICAgRm9udCBmb250OwotICAgICAgICBGb250UmVuZGVyQ29udGV4dCBmcmM7Ci0KLSAgICAgICAgY2hhciB0ZXh0W107Ci0KLSAgICAgICAgaW50IHN0YXJ0OwotICAgICAgICBpbnQgZW5kOwotICAgICAgICBpbnQgbGVuZ3RoOwotCi0gICAgICAgIGludCBmbGFncyA9IDA7Ci0KLSAgICAgICAgYnl0ZSBsZXZlbCA9IDA7Ci0KLSAgICAgICAgVGV4dFNlZ21lbnRJbmZvKAotICAgICAgICAgICAgICAgIGJ5dGUgbGV2ZWwsCi0gICAgICAgICAgICAgICAgRm9udCBmb250LCBGb250UmVuZGVyQ29udGV4dCBmcmMsCi0gICAgICAgICAgICAgICAgY2hhciB0ZXh0W10sIGludCBzdGFydCwgaW50IGVuZAotICAgICAgICApIHsKLSAgICAgICAgICAgIHRoaXMuZm9udCA9IGZvbnQ7Ci0gICAgICAgICAgICB0aGlzLmZyYyA9IGZyYzsKLSAgICAgICAgICAgIHRoaXMudGV4dCA9IHRleHQ7Ci0gICAgICAgICAgICB0aGlzLnN0YXJ0ID0gc3RhcnQ7Ci0gICAgICAgICAgICB0aGlzLmVuZCA9IGVuZDsKLSAgICAgICAgICAgIHRoaXMubGV2ZWwgPSBsZXZlbDsKLSAgICAgICAgICAgIGxlbmd0aCA9IGVuZCAtIHN0YXJ0OwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhpcyBjbGFzcyByZXByZXNlbnRzIGEgc2ltcGxlIHRleHQgc2VnbWVudCBiYWNrZWQgYnkgdGhlIGdseXBoIHZlY3RvcgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgY2xhc3MgVGV4dFJ1blNlZ21lbnRDb21tb24gZXh0ZW5kcyBUZXh0UnVuU2VnbWVudCB7Ci0gICAgICAgIFRleHRTZWdtZW50SW5mbyBpbmZvOwotICAgICAgICBwcml2YXRlIEdseXBoVmVjdG9yIGd2OwotICAgICAgICBwcml2YXRlIGZsb2F0IGFkdmFuY2VJbmNyZW1lbnRzW107Ci0gICAgICAgIHByaXZhdGUgaW50IGNoYXIyZ2x5cGhbXTsKLSAgICAgICAgcHJpdmF0ZSBHbHlwaEp1c3RpZmljYXRpb25JbmZvIGdqaXNbXTsgLy8gR2x5cGgganVzdGlmaWNhdGlvbiBpbmZvCi0KLSAgICAgICAgVGV4dFJ1blNlZ21lbnRDb21tb24oVGV4dFNlZ21lbnRJbmZvIGksIFRleHREZWNvcmF0b3IuRGVjb3JhdGlvbiBkKSB7Ci0gICAgICAgICAgICAvLyBYWFggLSB0b2RvIC0gY2hlY2sgc3VwcG9ydCBiaWRpCi0gICAgICAgICAgICBpLmZsYWdzICY9IH4weDA5OyAvLyBDbGVhciBiaWRpIGZsYWdzCi0KLSAgICAgICAgICAgIGlmICgoaS5sZXZlbCAmIDB4MSkgIT0gMCkgewotICAgICAgICAgICAgICAgIGkuZmxhZ3MgfD0gRm9udC5MQVlPVVRfUklHSFRfVE9fTEVGVDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaW5mbyA9IGk7Ci0gICAgICAgICAgICB0aGlzLmRlY29yYXRpb24gPSBkOwotCi0gICAgICAgICAgICBMaW5lTWV0cmljcyBsbSA9IGkuZm9udC5nZXRMaW5lTWV0cmljcyhpLnRleHQsIGkuc3RhcnQsIGkuZW5kLCBpLmZyYyk7Ci0gICAgICAgICAgICB0aGlzLm1ldHJpY3MgPSBuZXcgQmFzaWNNZXRyaWNzKGxtLCBpLmZvbnQpOwotCi0gICAgICAgICAgICBpZiAobG0uZ2V0TnVtQ2hhcnMoKSAhPSBpLmxlbmd0aCkgeyAvLyBYWFggdG9kbyAtIFRoaXMgc2hvdWxkIGJlIGhhbmRsZWQKLSAgICAgICAgICAgICAgICAvLyBhd3QuNDE9Rm9udCByZXR1cm5lZCB1bnN1cHBvcnRlZCB0eXBlIG9mIGxpbmUgbWV0cmljcy4gVGhpcyBjYXNlIGlzIGtub3duLCBidXQgbm90IHN1cHBvcnRlZCB5ZXQuCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKAotICAgICAgICAgICAgICAgICAgICAgICAgTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuNDEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgT2JqZWN0IGNsb25lKCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBUZXh0UnVuU2VnbWVudENvbW1vbihpbmZvLCBkZWNvcmF0aW9uKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDcmVhdGVzIGdseXBoIHZlY3RvciBmcm9tIHRoZSBtYW5hZ2VkIHRleHQgaWYgbmVlZGVkCi0gICAgICAgICAqIEByZXR1cm4gZ2x5cGggdmVjdG9yCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIEdseXBoVmVjdG9yIGdldEdseXBoVmVjdG9yKCkgewotICAgICAgICAgICAgaWYgKGd2PT1udWxsKSB7Ci0gICAgICAgICAgICAgICAgZ3YgPSBpbmZvLmZvbnQubGF5b3V0R2x5cGhWZWN0b3IoCi0gICAgICAgICAgICAgICAgICAgICAgICBpbmZvLmZyYywKLSAgICAgICAgICAgICAgICAgICAgICAgIGluZm8udGV4dCwKLSAgICAgICAgICAgICAgICAgICAgICAgIGluZm8uc3RhcnQsCi0gICAgICAgICAgICAgICAgICAgICAgICBpbmZvLmVuZCAtIGluZm8uc3RhcnQsIC8vIE5PVEU6IFRoaXMgcGFyYW1ldGVyIHZpb2xhdGVzCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIHNwZWMsIGl0IGlzIGNvdW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBub3QgbGltaXQgYXMgc3BlYyBzdGF0ZXMKLSAgICAgICAgICAgICAgICAgICAgICAgIGluZm8uZmxhZ3MKLSAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZXR1cm4gZ3Y7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmVuZGVycyB0aGlzIHRleHQgcnVuIHNlZ21lbnQKLSAgICAgICAgICogQHBhcmFtIGcyZCAtIGdyYXBoaWNzIHRvIHJlbmRlciB0bwotICAgICAgICAgKiBAcGFyYW0geE9mZnNldCAtIFggb2Zmc2V0IGZyb20gdGhlIGdyYXBoaWNzIG9yaWdpbiB0byB0aGUKLSAgICAgICAgICogb3JpZ2luIG9mIHRoZSB0ZXh0IGxheW91dAotICAgICAgICAgKiBAcGFyYW0geU9mZnNldCAtIFkgb2Zmc2V0IGZyb20gdGhlIGdyYXBoaWNzIG9yaWdpbiB0byB0aGUKLSAgICAgICAgICogb3JpZ2luIG9mIHRoZSB0ZXh0IGxheW91dAotICAgICAgICAgKi8KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHZvaWQgZHJhdyhHcmFwaGljczJEIGcyZCwgZmxvYXQgeE9mZnNldCwgZmxvYXQgeU9mZnNldCkgewotICAgICAgICAgICAgaWYgKGRlY29yYXRpb24gPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIGcyZC5kcmF3R2x5cGhWZWN0b3IoZ2V0R2x5cGhWZWN0b3IoKSwgeE9mZnNldCArIHgsIHlPZmZzZXQgKyB5KTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgVGV4dERlY29yYXRvci5wcmVwYXJlR3JhcGhpY3ModGhpcywgZzJkLCB4T2Zmc2V0LCB5T2Zmc2V0KTsKLSAgICAgICAgICAgICAgICBnMmQuZHJhd0dseXBoVmVjdG9yKGdldEdseXBoVmVjdG9yKCksIHhPZmZzZXQgKyB4LCB5T2Zmc2V0ICsgeSk7Ci0gICAgICAgICAgICAgICAgVGV4dERlY29yYXRvci5kcmF3VGV4dERlY29yYXRpb25zKHRoaXMsIGcyZCwgeE9mZnNldCwgeU9mZnNldCk7Ci0gICAgICAgICAgICAgICAgVGV4dERlY29yYXRvci5yZXN0b3JlR3JhcGhpY3MoZGVjb3JhdGlvbiwgZzJkKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBSZXR1cm5zIHZpc3VhbCBib3VuZHMgb2YgdGhpcyBzZWdtZW50Ci0gICAgICAgICAqIEByZXR1cm4gdmlzdWFsIGJvdW5kcwotICAgICAgICAgKi8KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIFJlY3RhbmdsZTJEIGdldFZpc3VhbEJvdW5kcygpIHsKLSAgICAgICAgICAgIGlmICh2aXN1YWxCb3VuZHMgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHZpc3VhbEJvdW5kcyA9Ci0gICAgICAgICAgICAgICAgICAgICAgICBUZXh0RGVjb3JhdG9yLmV4dGVuZFZpc3VhbEJvdW5kcygKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2V0R2x5cGhWZWN0b3IoKS5nZXRWaXN1YWxCb3VuZHMoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGVjb3JhdGlvbgotICAgICAgICAgICAgICAgICAgICAgICAgKTsKLQotICAgICAgICAgICAgICAgIHZpc3VhbEJvdW5kcy5zZXRSZWN0KAotICAgICAgICAgICAgICAgICAgICAgICAgeCArIHZpc3VhbEJvdW5kcy5nZXRYKCksCi0gICAgICAgICAgICAgICAgICAgICAgICB5ICsgdmlzdWFsQm91bmRzLmdldFkoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHZpc3VhbEJvdW5kcy5nZXRXaWR0aCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgdmlzdWFsQm91bmRzLmdldEhlaWdodCgpCi0gICAgICAgICAgICAgICAgKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIChSZWN0YW5nbGUyRCkgdmlzdWFsQm91bmRzLmNsb25lKCk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyBsb2dpY2FsIGJvdW5kcyBvZiB0aGlzIHNlZ21lbnQKLSAgICAgICAgICogQHJldHVybiBsb2dpY2FsIGJvdW5kcwotICAgICAgICAgKi8KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIFJlY3RhbmdsZTJEIGdldExvZ2ljYWxCb3VuZHMoKSB7Ci0gICAgICAgICAgICBpZiAobG9naWNhbEJvdW5kcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgbG9naWNhbEJvdW5kcyA9IGdldEdseXBoVmVjdG9yKCkuZ2V0TG9naWNhbEJvdW5kcygpOwotCi0gICAgICAgICAgICAgICAgbG9naWNhbEJvdW5kcy5zZXRSZWN0KAotICAgICAgICAgICAgICAgICAgICAgICAgeCArIGxvZ2ljYWxCb3VuZHMuZ2V0WCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgeSArIGxvZ2ljYWxCb3VuZHMuZ2V0WSgpLAotICAgICAgICAgICAgICAgICAgICAgICAgbG9naWNhbEJvdW5kcy5nZXRXaWR0aCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgbG9naWNhbEJvdW5kcy5nZXRIZWlnaHQoKQotICAgICAgICAgICAgICAgICk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiAoUmVjdGFuZ2xlMkQpIGxvZ2ljYWxCb3VuZHMuY2xvbmUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBmbG9hdCBnZXRBZHZhbmNlKCkgewotICAgICAgICAgICAgcmV0dXJuIChmbG9hdCkgZ2V0TG9naWNhbEJvdW5kcygpLmdldFdpZHRoKCk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQXR0ZW10cyB0byBtYXAgZWFjaCBjaGFyYWN0ZXIgdG8gdGhlIGNvcnJlc3BvbmRpbmcgYWR2YW5jZSBpbmNyZW1lbnQKLSAgICAgICAgICovCi0gICAgICAgIHZvaWQgaW5pdEFkdmFuY2VNYXBwaW5nKCkgewotICAgICAgICAgICAgR2x5cGhWZWN0b3IgZ3YgPSBnZXRHbHlwaFZlY3RvcigpOwotICAgICAgICAgICAgaW50IGNoYXJJbmRpY2llc1tdID0gZ3YuZ2V0R2x5cGhDaGFySW5kaWNlcygwLCBndi5nZXROdW1HbHlwaHMoKSwgbnVsbCk7Ci0gICAgICAgICAgICBhZHZhbmNlSW5jcmVtZW50cyA9IG5ldyBmbG9hdFtpbmZvLmxlbmd0aF07Ci0KLSAgICAgICAgICAgIGZvciAoaW50IGk9MDsgaTxjaGFySW5kaWNpZXMubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgICAgICBhZHZhbmNlSW5jcmVtZW50c1tjaGFySW5kaWNpZXNbaV1dID0gZ3YuZ2V0R2x5cGhNZXRyaWNzKGkpLmdldEFkdmFuY2UoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDYWxjdWxhdGVzIGFkdmFuY2UgZGVsdGEgYmV0d2VlbiB0d28gY2hhcmFjdGVycwotICAgICAgICAgKiBAcGFyYW0gc3RhcnQgLSAxc3QgcG9zaXRpb24KLSAgICAgICAgICogQHBhcmFtIGVuZCAtIDJuZCBwb3NpdGlvbgotICAgICAgICAgKiBAcmV0dXJuIGFkdmFuY2UgaW5jcmVtZW50IGJldHdlZW4gc3BlY2lmaWVkIHBvc2l0aW9ucwotICAgICAgICAgKi8KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIGZsb2F0IGdldEFkdmFuY2VEZWx0YShpbnQgc3RhcnQsIGludCBlbmQpIHsKLSAgICAgICAgICAgIC8vIEdldCBjb29yZGluYXRlcyBpbiB0aGUgc2VnbWVudCBjb250ZXh0Ci0gICAgICAgICAgICBzdGFydCAtPSBpbmZvLnN0YXJ0OwotICAgICAgICAgICAgZW5kIC09IGluZm8uc3RhcnQ7Ci0KLSAgICAgICAgICAgIGlmIChhZHZhbmNlSW5jcmVtZW50cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaW5pdEFkdmFuY2VNYXBwaW5nKCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChzdGFydCA8IDApIHsKLSAgICAgICAgICAgICAgICBzdGFydCA9IDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoZW5kID4gaW5mby5sZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBlbmQgPSBpbmZvLmxlbmd0aDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZmxvYXQgc3VtID0gMDsKLSAgICAgICAgICAgIGZvciAoaW50IGk9c3RhcnQ7IGk8ZW5kOyBpKyspIHsKLSAgICAgICAgICAgICAgICBzdW0gKz0gYWR2YW5jZUluY3JlbWVudHNbaV07Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBzdW07Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ2FsY3VsYXRlcyBpbmRleCBvZiB0aGUgY2hhcmFjdGVyIHdoaWNoIGFkdmFuY2UgaXMgZXF1YWwgdG8KLSAgICAgICAgICogdGhlIGdpdmVuLiBJZiB0aGUgZ2l2ZW4gYWR2YW5jZSBpcyBncmVhdGVyIHRoZW4gdGhlIHNlZ21lbnQKLSAgICAgICAgICogYWR2YW5jZSBpdCByZXR1cm5zIHRoZSBwb3NpdGlvbiBhZnRlciB0aGUgbGFzdCBjaGFyYWN0ZXIuCi0gICAgICAgICAqIEBwYXJhbSBhZHZhbmNlIC0gZ2l2ZW4gYWR2YW5jZQotICAgICAgICAgKiBAcGFyYW0gc3RhcnQgLSBjaGFyYWN0ZXIsIGZyb20gd2hpY2ggdG8gc3RhcnQgbWVhc3VyaW5nIGFkdmFuY2UKLSAgICAgICAgICogQHJldHVybiBjaGFyYWN0ZXIgaW5kZXgKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBpbnQgZ2V0Q2hhckluZGV4RnJvbUFkdmFuY2UoZmxvYXQgYWR2YW5jZSwgaW50IHN0YXJ0KSB7Ci0gICAgICAgICAgICAvLyBYWFggLSB0b2RvIC0gcHJvYmFibHksIHBvc3NpYmxlIHRvIG9wdGltaXplCi0gICAgICAgICAgICAvLyBBZGQgY2hlY2sgaWYgdGhlIGdpdmVuIGFkdmFuY2UgaXMgZ3JlYXRlciB0aGVuCi0gICAgICAgICAgICAvLyB0aGUgc2VnbWVudCBhZHZhbmNlIGluIHRoZSBiZWdpbm5pbmcuIEluIHRoaXMgY2FzZQotICAgICAgICAgICAgLy8gd2UgZG9uJ3QgbmVlZCB0byBydW4gdGhyb3VnaCBhbGwgaW5jcmVtZW50cwotICAgICAgICAgICAgaWYgKGFkdmFuY2VJbmNyZW1lbnRzID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBpbml0QWR2YW5jZU1hcHBpbmcoKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgc3RhcnQgLT0gaW5mby5zdGFydDsKLQotICAgICAgICAgICAgaWYgKHN0YXJ0IDwgMCkgewotICAgICAgICAgICAgICAgIHN0YXJ0ID0gMDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaW50IGkgPSBzdGFydDsKLSAgICAgICAgICAgIGZvciAoOyBpPGluZm8ubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgICAgICBhZHZhbmNlIC09IGFkdmFuY2VJbmNyZW1lbnRzW2ldOwotICAgICAgICAgICAgICAgIGlmIChhZHZhbmNlIDwgMCkgewotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBpICsgaW5mby5zdGFydDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBpbnQgZ2V0U3RhcnQoKSB7Ci0gICAgICAgICAgICByZXR1cm4gaW5mby5zdGFydDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBpbnQgZ2V0RW5kKCkgewotICAgICAgICAgICAgcmV0dXJuIGluZm8uZW5kOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIGludCBnZXRMZW5ndGgoKSB7Ci0gICAgICAgICAgICByZXR1cm4gaW5mby5sZW5ndGg7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQXR0ZW10cyB0byBjcmVhdGUgbWFwcGluZyBvZiB0aGUgY2hhcmFjdGVycyB0byBnbHlwaHMgaW4gdGhlIGdseXBoIHZlY3Rvci4KLSAgICAgICAgICogQHJldHVybiBhcnJheSB3aGVyZSBmb3IgZWFjaCBjaGFyYWN0ZXIgaW5kZXggc3RvcmVkIGNvcnJlc3BvbmRpbmcgZ2x5cGggaW5kZXgKLSAgICAgICAgICovCi0gICAgICAgIHByaXZhdGUgaW50W10gZ2V0Q2hhcjJHbHlwaCgpIHsKLSAgICAgICAgICAgIGlmIChjaGFyMmdseXBoID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBHbHlwaFZlY3RvciBndiA9IGdldEdseXBoVmVjdG9yKCk7Ci0gICAgICAgICAgICAgICAgY2hhcjJnbHlwaCA9IG5ldyBpbnRbaW5mby5sZW5ndGhdOwotICAgICAgICAgICAgICAgIEFycmF5cy5maWxsKGNoYXIyZ2x5cGgsIC0xKTsKLQotICAgICAgICAgICAgICAgIC8vIEZpbGwgZ2x5cGggaW5kaWNpZXMgZm9yIGZpcnN0IGNoYXJhY3RlcnMgY29ycmVzcG9uZGluZyB0byBlYWNoIGdseXBoCi0gICAgICAgICAgICAgICAgaW50IGNoYXJJbmRpY2llc1tdID0gZ3YuZ2V0R2x5cGhDaGFySW5kaWNlcygwLCBndi5nZXROdW1HbHlwaHMoKSwgbnVsbCk7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaT0wOyBpPGNoYXJJbmRpY2llcy5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBjaGFyMmdseXBoW2NoYXJJbmRpY2llc1tpXV0gPSBpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIC8vIElmIHNldmVyYWwgY2hhcmFjdGVycyBjb3JyZXNwb25kcyB0byBvbmUgZ2x5cGgsIGNyZWF0ZSBtYXBwaW5nIGZvciB0aGVtCi0gICAgICAgICAgICAgICAgLy8gU3VwcG9zZSB0aGF0IHRoZXNlIGNoYXJhY3RlcnMgYXJlIGdvaW5nIGFsbCB0b2dldGhlcgotICAgICAgICAgICAgICAgIGludCBjdXJySW5kZXggPSAwOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGk9MDsgaTxjaGFyMmdseXBoLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChjaGFyMmdseXBoW2ldIDwgMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgY2hhcjJnbHlwaFtpXSA9IGN1cnJJbmRleDsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGN1cnJJbmRleCA9IGNoYXIyZ2x5cGhbaV07Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBjaGFyMmdseXBoOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENyZWF0ZXMgYmxhY2sgYm94IGJvdW5kcyBzaGFwZSBmb3IgdGhlIHNwZWNpZmllZCByYW5nZQotICAgICAgICAgKiBAcGFyYW0gc3RhcnQgLSByYW5nZSBzYXJ0Ci0gICAgICAgICAqIEBwYXJhbSBsaW1pdCAtIHJhbmdlIGVuZAotICAgICAgICAgKiBAcmV0dXJuIGJsYWNrIGJveCBib3VuZHMgc2hhcGUKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBTaGFwZSBnZXRDaGFyc0JsYWNrQm94Qm91bmRzKGludCBzdGFydCwgaW50IGxpbWl0KSB7Ci0gICAgICAgICAgICBzdGFydCAtPSBpbmZvLnN0YXJ0OwotICAgICAgICAgICAgbGltaXQgLT0gaW5mby5zdGFydDsKLQotICAgICAgICAgICAgaWYgKGxpbWl0ID4gaW5mby5sZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBsaW1pdCA9IGluZm8ubGVuZ3RoOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBHZW5lcmFsUGF0aCByZXN1bHQgPSBuZXcgR2VuZXJhbFBhdGgoKTsKLQotICAgICAgICAgICAgaW50IGdseXBoSW5kZXggPSAwOwotCi0gICAgICAgICAgICBmb3IgKGludCBpPXN0YXJ0OyBpPGxpbWl0OyBpKyspIHsKLSAgICAgICAgICAgICAgICBnbHlwaEluZGV4ID0gZ2V0Q2hhcjJHbHlwaCgpW2ldOwotICAgICAgICAgICAgICAgIHJlc3VsdC5hcHBlbmQoZ2V0R2x5cGhWZWN0b3IoKS5nZXRHbHlwaFZpc3VhbEJvdW5kcyhnbHlwaEluZGV4KSwgZmFsc2UpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBTaGlmdCB0byB0aGUgc2VnbWVudCdzIGNvb3JkaW5hdGVzCi0gICAgICAgICAgICByZXN1bHQudHJhbnNmb3JtKEFmZmluZVRyYW5zZm9ybS5nZXRUcmFuc2xhdGVJbnN0YW5jZSh4LCB5KSk7Ci0KLSAgICAgICAgICAgIHJldHVybiByZXN1bHQ7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ2FsY3VsYXRlcyBwb3NpdGlvbiBvZiB0aGUgY2hhcmFjdGVyIG9uIHRoZSBzY3JlZW4KLSAgICAgICAgICogQHBhcmFtIGluZGV4IC0gY2hhcmFjdGVyIGluZGV4Ci0gICAgICAgICAqIEByZXR1cm4gWCBjb29yZGluYXRlIG9mIHRoZSBjaGFyYWN0ZXIgcG9zaXRpb24KLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBmbG9hdCBnZXRDaGFyUG9zaXRpb24oaW50IGluZGV4KSB7Ci0gICAgICAgICAgICBpbmRleCAtPSBpbmZvLnN0YXJ0OwotCi0gICAgICAgICAgICBpZiAoaW5kZXggPiBpbmZvLmxlbmd0aCkgewotICAgICAgICAgICAgICAgIGluZGV4ID0gaW5mby5sZW5ndGg7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGZsb2F0IHJlc3VsdCA9IDA7Ci0KLSAgICAgICAgICAgIGludCBnbHlwaEluZGV4ID0gZ2V0Q2hhcjJHbHlwaCgpW2luZGV4XTsKLSAgICAgICAgICAgIHJlc3VsdCA9IChmbG9hdCkgZ2V0R2x5cGhWZWN0b3IoKS5nZXRHbHlwaFBvc2l0aW9uKGdseXBoSW5kZXgpLmdldFgoKTsKLQotICAgICAgICAgICAgLy8gU2hpZnQgdG8gdGhlIHNlZ21lbnQncyBjb29yZGluYXRlcwotICAgICAgICAgICAgcmVzdWx0ICs9IHg7Ci0KLSAgICAgICAgICAgIHJldHVybiByZXN1bHQ7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyB0aGUgYWR2YW5jZSBvZiB0aGUgaW5kaXZpZHVhbCBjaGFyYWN0ZXIKLSAgICAgICAgICogQHBhcmFtIGluZGV4IC0gY2hhcmFjdGVyIGluZGV4Ci0gICAgICAgICAqIEByZXR1cm4gY2hhcmFjdGVyIGFkdmFuY2UKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBmbG9hdCBnZXRDaGFyQWR2YW5jZShpbnQgaW5kZXgpIHsKLSAgICAgICAgICAgIGlmIChhZHZhbmNlSW5jcmVtZW50cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaW5pdEFkdmFuY2VNYXBwaW5nKCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBhZHZhbmNlSW5jcmVtZW50c1tpbmRleCAtIHRoaXMuZ2V0U3RhcnQoKV07Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyB0aGUgb3V0bGluZSBzaGFwZQotICAgICAgICAgKiBAcmV0dXJuIG91dGxpbmUKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBTaGFwZSBnZXRPdXRsaW5lKCkgewotICAgICAgICAgICAgQWZmaW5lVHJhbnNmb3JtIHQgPSBBZmZpbmVUcmFuc2Zvcm0uZ2V0VHJhbnNsYXRlSW5zdGFuY2UoeCwgeSk7Ci0gICAgICAgICAgICByZXR1cm4gdC5jcmVhdGVUcmFuc2Zvcm1lZFNoYXBlKAotICAgICAgICAgICAgICAgICAgICBUZXh0RGVjb3JhdG9yLmV4dGVuZE91dGxpbmUoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBnZXRHbHlwaFZlY3RvcigpLmdldE91dGxpbmUoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkZWNvcmF0aW9uCi0gICAgICAgICAgICAgICAgICAgICkKLSAgICAgICAgICAgICk7Ci0gICAgICAgIH0KLQotICAgICAgICAvKioKLSAgICAgICAgICogQ2hlY2tzIGlmIHRoZSBjaGFyYWN0ZXIgZG9lc24ndCBjb250cmlidXRlIHRvIHRoZSB0ZXh0IGFkdmFuY2UKLSAgICAgICAgICogQHBhcmFtIGluZGV4IC0gY2hhcmFjdGVyIGluZGV4Ci0gICAgICAgICAqIEByZXR1cm4gdHJ1ZSBpZiB0aGUgY2hhcmFjdGVyIGhhcyB6ZXJvIGFkdmFuY2UKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBib29sZWFuIGNoYXJIYXNaZXJvQWR2YW5jZShpbnQgaW5kZXgpIHsKLSAgICAgICAgICAgIGlmIChhZHZhbmNlSW5jcmVtZW50cyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaW5pdEFkdmFuY2VNYXBwaW5nKCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBhZHZhbmNlSW5jcmVtZW50c1tpbmRleCAtIHRoaXMuZ2V0U3RhcnQoKV0gPT0gMDsKLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDcmVhdGVzIHRleHQgaGl0IGluZm8gZnJvbSB0aGUgaGl0IHBvc2l0aW9uCi0gICAgICAgICAqIEBwYXJhbSBoaXRYIC0gWCBjb29yZGluYXRlIHJlbGF0aXZlIHRvIHRoZSBvcmlnaW4gb2YgdGhlIGxheW91dAotICAgICAgICAgKiBAcGFyYW0gaGl0WSAtIFkgY29vcmRpbmF0ZSByZWxhdGl2ZSB0byB0aGUgb3JpZ2luIG9mIHRoZSBsYXlvdXQKLSAgICAgICAgICogQHJldHVybiBoaXQgaW5mbwotICAgICAgICAgKi8KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIFRleHRIaXRJbmZvIGhpdFRlc3QoZmxvYXQgaGl0WCwgZmxvYXQgaGl0WSkgewotICAgICAgICAgICAgaGl0WCAtPSB4OwotCi0gICAgICAgICAgICBmbG9hdCBnbHlwaFBvc2l0aW9uc1tdID0KLSAgICAgICAgICAgICAgICAgICAgZ2V0R2x5cGhWZWN0b3IoKS5nZXRHbHlwaFBvc2l0aW9ucygwLCBpbmZvLmxlbmd0aCsxLCBudWxsKTsKLQotICAgICAgICAgICAgaW50IGdseXBoSWR4OwotICAgICAgICAgICAgYm9vbGVhbiBsZWFkaW5nID0gZmFsc2U7Ci0gICAgICAgICAgICBmb3IgKGdseXBoSWR4ID0gMTsgZ2x5cGhJZHggPD0gaW5mby5sZW5ndGg7IGdseXBoSWR4KyspIHsKLSAgICAgICAgICAgICAgICBpZiAoZ2x5cGhQb3NpdGlvbnNbKGdseXBoSWR4KSoyXSA+PSBoaXRYKSB7Ci0gICAgICAgICAgICAgICAgICAgIGZsb2F0IGFkdmFuY2UgPQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdseXBoUG9zaXRpb25zWyhnbHlwaElkeCkqMl0gLSBnbHlwaFBvc2l0aW9uc1soZ2x5cGhJZHgtMSkqMl07Ci0gICAgICAgICAgICAgICAgICAgIGxlYWRpbmcgPSBnbHlwaFBvc2l0aW9uc1soZ2x5cGhJZHgtMSkqMl0gKyBhZHZhbmNlLzIgPiBoaXRYID8gdHJ1ZSA6IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICBnbHlwaElkeC0tOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChnbHlwaElkeCA9PSBpbmZvLmxlbmd0aCkgewotICAgICAgICAgICAgICAgIGdseXBoSWR4LS07Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGludCBjaGFySWR4ID0gZ2V0R2x5cGhWZWN0b3IoKS5nZXRHbHlwaENoYXJJbmRleChnbHlwaElkeCk7Ci0KLSAgICAgICAgICAgIHJldHVybiAobGVhZGluZykgXiAoKGluZm8ubGV2ZWwgJiAweDEpID09IDB4MSk/Ci0gICAgICAgICAgICAgICAgICAgIFRleHRIaXRJbmZvLmxlYWRpbmcoY2hhcklkeCArIGluZm8uc3RhcnQpIDoKLSAgICAgICAgICAgICAgICAgICAgVGV4dEhpdEluZm8udHJhaWxpbmcoY2hhcklkeCArIGluZm8uc3RhcnQpOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENvbGxlY3RzIEdseXBoSnVzdGlmaWNhdGlvbkluZm8gb2JqZWN0cyBmcm9tIHRoZSBnbHlwaCB2ZWN0b3IKLSAgICAgICAgICogQHJldHVybiBhcnJheSBvZiBhbGwgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBvYmplY3RzCi0gICAgICAgICAqLwotICAgICAgICBwcml2YXRlIEdseXBoSnVzdGlmaWNhdGlvbkluZm9bXSBnZXRHbHlwaEp1c3RpZmljYXRpb25JbmZvcygpIHsKLSAgICAgICAgICAgIGlmIChnamlzID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBHbHlwaFZlY3RvciBndiA9IGdldEdseXBoVmVjdG9yKCk7Ci0gICAgICAgICAgICAgICAgaW50IG5HbHlwaHMgPSBndi5nZXROdW1HbHlwaHMoKTsKLSAgICAgICAgICAgICAgICBpbnQgY2hhckluZGljaWVzW10gPSBndi5nZXRHbHlwaENoYXJJbmRpY2VzKDAsIG5HbHlwaHMsIG51bGwpOwotICAgICAgICAgICAgICAgIGdqaXMgPSBuZXcgR2x5cGhKdXN0aWZpY2F0aW9uSW5mb1tuR2x5cGhzXTsKLQotICAgICAgICAgICAgICAgIC8vIFBhdGNoOiB0ZW1wb3JhcnkgcGF0Y2gsIGdldEdseXBoSnVzdGlmaWNhdGlvbkluZm8gaXMgbm90IGltcGxlbWVudGVkCi0gICAgICAgICAgICAgICAgZmxvYXQgZm9udFNpemUgPSBpbmZvLmZvbnQuZ2V0U2l6ZTJEKCk7Ci0gICAgICAgICAgICAgICAgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBkZWZhdWx0SW5mbyA9Ci0gICAgICAgICAgICAgICAgICAgICAgICBuZXcgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbygKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwgLy8gd2VpZ2h0Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZhbHNlLCBHbHlwaEp1c3RpZmljYXRpb25JbmZvLlBSSU9SSVRZX05PTkUsIDAsIDAsIC8vIGdyb3cKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmFsc2UsIEdseXBoSnVzdGlmaWNhdGlvbkluZm8uUFJJT1JJVFlfTk9ORSwgMCwgMCk7IC8vIHNocmluawotICAgICAgICAgICAgICAgIEdseXBoSnVzdGlmaWNhdGlvbkluZm8gc3BhY2VJbmZvID0gbmV3IEdseXBoSnVzdGlmaWNhdGlvbkluZm8oCi0gICAgICAgICAgICAgICAgICAgICAgICBmb250U2l6ZSwgLy8gd2VpZ2h0Ci0gICAgICAgICAgICAgICAgICAgICAgICB0cnVlLCBHbHlwaEp1c3RpZmljYXRpb25JbmZvLlBSSU9SSVRZX1dISVRFU1BBQ0UsIDAsIGZvbnRTaXplLCAvLyBncm93Ci0gICAgICAgICAgICAgICAgICAgICAgICB0cnVlLCBHbHlwaEp1c3RpZmljYXRpb25JbmZvLlBSSU9SSVRZX1dISVRFU1BBQ0UsIDAsIGZvbnRTaXplKTsgLy8gc2hyaW5rCi0KLSAgICAgICAgICAgICAgICAvLy8vLy8vLwotICAgICAgICAgICAgICAgIC8vIFRlbXBvcmFyeSBwYXRjaCwgZ2V0R2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBpcyBub3QgaW1wbGVtZW50ZWQKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG5HbHlwaHM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICAvL2dqaXNbaV0gPSBnZXRHbHlwaFZlY3RvcigpLmdldEdseXBoSnVzdGlmaWNhdGlvbkluZm8oaSk7Ci0KLSAgICAgICAgICAgICAgICAgICAgY2hhciBjID0gaW5mby50ZXh0W2NoYXJJbmRpY2llc1tpXSArIGluZm8uc3RhcnRdOwotICAgICAgICAgICAgICAgICAgICBpZiAoQ2hhcmFjdGVyLmlzV2hpdGVzcGFjZShjKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgZ2ppc1tpXSA9IHNwYWNlSW5mbzsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGdqaXNbaV0gPSBkZWZhdWx0SW5mbzsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAvLyBFbmQgcGF0Y2gKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBnamlzOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIENvbGxlY3RzIGp1c3RpZmljYXRpb24gaW5mb3JtYXRpb24gaW50byBKdXN0aWZpY2F0aW9uSW5mbyBvYmplY3QKLSAgICAgICAgICogQHBhcmFtIGpJbmZvIC0gSnVzdGlmaWNhdGlvbkluZm8gb2JqZWN0Ci0gICAgICAgICAqLwotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgdm9pZCB1cGRhdGVKdXN0aWZpY2F0aW9uSW5mbyhUZXh0UnVuQnJlYWtlci5KdXN0aWZpY2F0aW9uSW5mbyBqSW5mbykgewotICAgICAgICAgICAgaW50IGxhc3RDaGFyID0gTWF0aC5taW4oakluZm8ubGFzdElkeCwgaW5mby5lbmQpIC0gaW5mby5zdGFydDsKLSAgICAgICAgICAgIGJvb2xlYW4gaGF2ZUZpcnN0ID0gaW5mby5zdGFydCA8PSBqSW5mby5maXJzdElkeDsKLSAgICAgICAgICAgIGJvb2xlYW4gaGF2ZUxhc3QgPSBpbmZvLmVuZCA+PSAoakluZm8ubGFzdElkeCArIDEpOwotCi0gICAgICAgICAgICBpbnQgcHJldkdseXBoSWR4ID0gLTE7Ci0gICAgICAgICAgICBpbnQgY3VyckdseXBoSWR4OwotCi0gICAgICAgICAgICBpZiAoakluZm8uZ3JvdykgeyAvLyBDaGVjayBob3cgbXVjaCB3ZSBjYW4gZ3Jvdy9zaHJpbmsgb24gY3VycmVudCBwcmlvcml0eSBsZXZlbAotICAgICAgICAgICAgICAgIGZvciAoaW50IGk9MDsgaTxsYXN0Q2hhcjsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGN1cnJHbHlwaElkeCA9IGdldENoYXIyR2x5cGgoKVtpXTsKLQotICAgICAgICAgICAgICAgICAgICBpZiAoY3VyckdseXBoSWR4ID09IHByZXZHbHlwaElkeCkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gU2V2ZXJhbCBjaGFycyBjb3VsZCBiZSByZXByZXNlbnRlZCBieSBvbmUgZ2x5cGgsCi0gICAgICAgICAgICAgICAgICAgICAgICAvLyBzdXBwb3NlIHRoZXkgYXJlIGNvbnRpZ3VvdXMKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHByZXZHbHlwaElkeCA9IGN1cnJHbHlwaElkeDsKLQotICAgICAgICAgICAgICAgICAgICBHbHlwaEp1c3RpZmljYXRpb25JbmZvIGdqaSA9IGdldEdseXBoSnVzdGlmaWNhdGlvbkluZm9zKClbY3VyckdseXBoSWR4XTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGdqaS5ncm93UHJpb3JpdHkgPT0gakluZm8ucHJpb3JpdHkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGpJbmZvLndlaWdodCArPSBnamkud2VpZ2h0ICogMjsKLSAgICAgICAgICAgICAgICAgICAgICAgIGpJbmZvLmdyb3dMaW1pdCArPSBnamkuZ3Jvd0xlZnRMaW1pdDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGpJbmZvLmdyb3dMaW1pdCArPSBnamkuZ3Jvd1JpZ2h0TGltaXQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoZ2ppLmdyb3dBYnNvcmIpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBqSW5mby5hYnNvcmJlZFdlaWdodCArPSBnamkud2VpZ2h0ICogMjsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaT0wOyBpPGxhc3RDaGFyOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgY3VyckdseXBoSWR4ID0gZ2V0Q2hhcjJHbHlwaCgpW2ldOwotICAgICAgICAgICAgICAgICAgICBpZiAoY3VyckdseXBoSWR4ID09IHByZXZHbHlwaElkeCkgewotICAgICAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcHJldkdseXBoSWR4ID0gY3VyckdseXBoSWR4OwotCi0gICAgICAgICAgICAgICAgICAgIEdseXBoSnVzdGlmaWNhdGlvbkluZm8gZ2ppID0gZ2V0R2x5cGhKdXN0aWZpY2F0aW9uSW5mb3MoKVtjdXJyR2x5cGhJZHhdOwotICAgICAgICAgICAgICAgICAgICBpZiAoZ2ppLnNocmlua1ByaW9yaXR5ID09IGpJbmZvLnByaW9yaXR5KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBqSW5mby53ZWlnaHQgKz0gZ2ppLndlaWdodCAqIDI7Ci0gICAgICAgICAgICAgICAgICAgICAgICBqSW5mby5ncm93TGltaXQgLT0gZ2ppLnNocmlua0xlZnRMaW1pdDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGpJbmZvLmdyb3dMaW1pdCAtPSBnamkuc2hyaW5rUmlnaHRMaW1pdDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChnamkuc2hyaW5rQWJzb3JiKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgakluZm8uYWJzb3JiZWRXZWlnaHQgKz0gZ2ppLndlaWdodCAqIDI7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChoYXZlRmlyc3QpIHsgIC8vIERvbid0IGFkZCBwYWRkaW5nIGJlZm9yZSBmaXJzdCBjaGFyCi0gICAgICAgICAgICAgICAgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBnamkgPSBnZXRHbHlwaEp1c3RpZmljYXRpb25JbmZvcygpW2dldENoYXIyR2x5cGgoKVswXV07Ci0gICAgICAgICAgICAgICAgakluZm8ud2VpZ2h0IC09IGdqaS53ZWlnaHQ7Ci0gICAgICAgICAgICAgICAgaWYgKGpJbmZvLmdyb3cpIHsKLSAgICAgICAgICAgICAgICAgICAgakluZm8uZ3Jvd0xpbWl0IC09IGdqaS5ncm93TGVmdExpbWl0OwotICAgICAgICAgICAgICAgICAgICBpZiAoZ2ppLmdyb3dBYnNvcmIpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGpJbmZvLmFic29yYmVkV2VpZ2h0IC09IGdqaS53ZWlnaHQ7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBqSW5mby5ncm93TGltaXQgKz0gZ2ppLnNocmlua0xlZnRMaW1pdDsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGdqaS5zaHJpbmtBYnNvcmIpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGpJbmZvLmFic29yYmVkV2VpZ2h0IC09IGdqaS53ZWlnaHQ7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChoYXZlTGFzdCkgeyAgIC8vIERvbid0IGFkZCBwYWRkaW5nIGFmdGVyIGxhc3QgY2hhcgotICAgICAgICAgICAgICAgIEdseXBoSnVzdGlmaWNhdGlvbkluZm8gZ2ppID0KLSAgICAgICAgICAgICAgICAgICAgICAgIGdldEdseXBoSnVzdGlmaWNhdGlvbkluZm9zKClbZ2V0Q2hhcjJHbHlwaCgpW2xhc3RDaGFyXV07Ci0gICAgICAgICAgICAgICAgakluZm8ud2VpZ2h0IC09IGdqaS53ZWlnaHQ7Ci0gICAgICAgICAgICAgICAgaWYgKGpJbmZvLmdyb3cpIHsKLSAgICAgICAgICAgICAgICAgICAgakluZm8uZ3Jvd0xpbWl0IC09IGdqaS5ncm93UmlnaHRMaW1pdDsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGdqaS5ncm93QWJzb3JiKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBqSW5mby5hYnNvcmJlZFdlaWdodCAtPSBnamkud2VpZ2h0OwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgakluZm8uZ3Jvd0xpbWl0ICs9IGdqaS5zaHJpbmtSaWdodExpbWl0OwotICAgICAgICAgICAgICAgICAgICBpZiAoZ2ppLnNocmlua0Fic29yYikgewotICAgICAgICAgICAgICAgICAgICAgICAgakluZm8uYWJzb3JiZWRXZWlnaHQgLT0gZ2ppLndlaWdodDsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBQZXJmb3JtcyBqdXN0aWZpY2F0aW9uIG9mIHRoZSBzZWdtZW50LgotICAgICAgICAgKiBVcGRhdGVzIHBvc2l0aW9ucyBvZiBpbmRpdmlkdWFsIGNoYXJhY3RlcnMuCi0gICAgICAgICAqIEBwYXJhbSBqSW5mb3MgLSBqdXN0aWZpY2F0aW9uIGluZm9ybWF0aW9uLCBnYXRoZXJlZCBieSB0aGUgcHJldmlvdXMgcGFzc2VzCi0gICAgICAgICAqIEByZXR1cm4gYW1vdW50IG9mIGdyb3d0aCBvciBzaHJpbmsgb2YgdGhlIHNlZ21lbnQKLSAgICAgICAgICovCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBmbG9hdCBkb0p1c3RpZmljYXRpb24oVGV4dFJ1bkJyZWFrZXIuSnVzdGlmaWNhdGlvbkluZm8gakluZm9zW10pIHsKLSAgICAgICAgICAgIGludCBsYXN0UHJpb3JpdHkgPQotICAgICAgICAgICAgICAgICAgICBqSW5mb3NbakluZm9zLmxlbmd0aC0xXSA9PSBudWxsID8KLSAgICAgICAgICAgICAgICAgICAgLTEgOiBqSW5mb3NbakluZm9zLmxlbmd0aC0xXS5wcmlvcml0eTsKLQotICAgICAgICAgICAgLy8gR2V0IHRoZSBoaWdoZXN0IHByaW9yaXR5Ci0gICAgICAgICAgICBpbnQgaGlnaGVzdFByaW9yaXR5ID0gMDsKLSAgICAgICAgICAgIGZvciAoOyBoaWdoZXN0UHJpb3JpdHk8akluZm9zLmxlbmd0aDsgaGlnaGVzdFByaW9yaXR5KyspIHsKLSAgICAgICAgICAgICAgICBpZiAoakluZm9zW2hpZ2hlc3RQcmlvcml0eV0gIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChoaWdoZXN0UHJpb3JpdHkgPT0gakluZm9zLmxlbmd0aCkgewotICAgICAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBUZXh0UnVuQnJlYWtlci5KdXN0aWZpY2F0aW9uSW5mbyBmaXJzdEluZm8gPSBqSW5mb3NbaGlnaGVzdFByaW9yaXR5XTsKLSAgICAgICAgICAgIFRleHRSdW5CcmVha2VyLkp1c3RpZmljYXRpb25JbmZvIGxhc3RJbmZvID0KLSAgICAgICAgICAgICAgICAgICAgbGFzdFByaW9yaXR5ID4gMCA/IGpJbmZvc1tsYXN0UHJpb3JpdHldIDogbnVsbDsKLQotICAgICAgICAgICAgYm9vbGVhbiBoYXZlRmlyc3QgPSBpbmZvLnN0YXJ0IDw9IGZpcnN0SW5mby5maXJzdElkeDsKLSAgICAgICAgICAgIGJvb2xlYW4gaGF2ZUxhc3QgPSBpbmZvLmVuZCA+PSAoZmlyc3RJbmZvLmxhc3RJZHggKyAxKTsKLQotICAgICAgICAgICAgLy8gSGVyZSB3ZSBzdXBwb3NlIHRoYXQgR0xZUEhTIGFyZSBvcmRlcmVkIExFRlQgVE8gUklHSFQKLSAgICAgICAgICAgIGludCBmaXJzdEdseXBoID0gaGF2ZUZpcnN0ID8KLSAgICAgICAgICAgICAgICAgICAgZ2V0Q2hhcjJHbHlwaCgpW2ZpcnN0SW5mby5maXJzdElkeCAtIGluZm8uc3RhcnRdIDoKLSAgICAgICAgICAgICAgICAgICAgZ2V0Q2hhcjJHbHlwaCgpWzBdOwotCi0gICAgICAgICAgICBpbnQgbGFzdEdseXBoID0gaGF2ZUxhc3QgPwotICAgICAgICAgICAgICAgICAgICBnZXRDaGFyMkdseXBoKClbZmlyc3RJbmZvLmxhc3RJZHggLSBpbmZvLnN0YXJ0XSA6Ci0gICAgICAgICAgICAgICAgICAgIGdldENoYXIyR2x5cGgoKVtpbmZvLmxlbmd0aCAtIDFdOwotICAgICAgICAgICAgaWYgKGhhdmVMYXN0KSB7Ci0gICAgICAgICAgICAgICAgbGFzdEdseXBoLS07Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIFRleHRSdW5CcmVha2VyLkp1c3RpZmljYXRpb25JbmZvIGN1cnJJbmZvOwotICAgICAgICAgICAgZmxvYXQgZ2x5cGhPZmZzZXQgPSAwOwotICAgICAgICAgICAgZmxvYXQgcG9zaXRpb25JbmNyZW1lbnQgPSAwOwotICAgICAgICAgICAgZmxvYXQgc2lkZUluY3JlbWVudCA9IDA7Ci0KLSAgICAgICAgICAgIGlmIChoYXZlRmlyc3QpIHsgIC8vIERvbid0IGFkZCBwYWRkaW5nIGJlZm9yZSBmaXJzdCBjaGFyCi0gICAgICAgICAgICAgICAgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBnamkgPSBnZXRHbHlwaEp1c3RpZmljYXRpb25JbmZvcygpW2ZpcnN0R2x5cGhdOwotICAgICAgICAgICAgICAgIGN1cnJJbmZvID0gakluZm9zW2dqaS5ncm93UHJpb3JpdHldOwotICAgICAgICAgICAgICAgIGlmIChjdXJySW5mbyAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChjdXJySW5mby51c2VMaW1pdHMpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjdXJySW5mby5hYnNvcmIpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBnbHlwaE9mZnNldCArPSBnamkud2VpZ2h0ICogY3VyckluZm8uYWJzb3JiZWRHYXBQZXJVbml0OwotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIGlmICgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbGFzdEluZm8gIT0gbnVsbCAmJgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsYXN0SW5mby5wcmlvcml0eSA9PSBjdXJySW5mby5wcmlvcml0eQotICAgICAgICAgICAgICAgICAgICAgICAgKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2x5cGhPZmZzZXQgKz0gZ2ppLndlaWdodCAqIGxhc3RJbmZvLmFic29yYmVkR2FwUGVyVW5pdDsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIGdseXBoT2Zmc2V0ICs9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZpcnN0SW5mby5ncm93ID8KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2ppLmdyb3dSaWdodExpbWl0IDoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLWdqaS5zaHJpbmtSaWdodExpbWl0OwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgZ2x5cGhPZmZzZXQgKz0gZ2ppLndlaWdodCAqIGN1cnJJbmZvLmdhcFBlclVuaXQ7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBmaXJzdEdseXBoKys7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChmaXJzdEluZm8uZ3JvdykgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGk9Zmlyc3RHbHlwaDsgaTw9bGFzdEdseXBoOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBnamkgPSBnZXRHbHlwaEp1c3RpZmljYXRpb25JbmZvcygpW2ldOwotICAgICAgICAgICAgICAgICAgICBjdXJySW5mbyA9IGpJbmZvc1tnamkuZ3Jvd1ByaW9yaXR5XTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGN1cnJJbmZvID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIFdlIHN0aWxsIGhhdmUgdG8gaW5jcmVtZW50IGdseXBoIHBvc2l0aW9uCi0gICAgICAgICAgICAgICAgICAgICAgICBQb2ludDJEIGdseXBoUG9zID0gZ2V0R2x5cGhWZWN0b3IoKS5nZXRHbHlwaFBvc2l0aW9uKGkpOwotICAgICAgICAgICAgICAgICAgICAgICAgZ2x5cGhQb3Muc2V0TG9jYXRpb24oZ2x5cGhQb3MuZ2V0WCgpICsgZ2x5cGhPZmZzZXQsIGdseXBoUG9zLmdldFkoKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBnZXRHbHlwaFZlY3RvcigpLnNldEdseXBoUG9zaXRpb24oaSwgZ2x5cGhQb3MpOwotCi0gICAgICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGlmIChjdXJySW5mby51c2VMaW1pdHMpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGdseXBoT2Zmc2V0ICs9IGdqaS5ncm93TGVmdExpbWl0OwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGN1cnJJbmZvLmFic29yYikgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpZGVJbmNyZW1lbnQgPSBnamkud2VpZ2h0ICogY3VyckluZm8uYWJzb3JiZWRHYXBQZXJVbml0OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdseXBoT2Zmc2V0ICs9IHNpZGVJbmNyZW1lbnQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcG9zaXRpb25JbmNyZW1lbnQgPSBnbHlwaE9mZnNldDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBnbHlwaE9mZnNldCArPSBzaWRlSW5jcmVtZW50OwotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIGlmIChsYXN0SW5mbyAhPSBudWxsICYmIGxhc3RJbmZvLnByaW9yaXR5ID09IGN1cnJJbmZvLnByaW9yaXR5KSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgc2lkZUluY3JlbWVudCA9IGdqaS53ZWlnaHQgKiBsYXN0SW5mby5hYnNvcmJlZEdhcFBlclVuaXQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2x5cGhPZmZzZXQgKz0gc2lkZUluY3JlbWVudDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBwb3NpdGlvbkluY3JlbWVudCA9IGdseXBoT2Zmc2V0OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdseXBoT2Zmc2V0ICs9IHNpZGVJbmNyZW1lbnQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBvc2l0aW9uSW5jcmVtZW50ID0gZ2x5cGhPZmZzZXQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBnbHlwaE9mZnNldCArPSBnamkuZ3Jvd1JpZ2h0TGltaXQ7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzaWRlSW5jcmVtZW50ID0gZ2ppLndlaWdodCAqIGN1cnJJbmZvLmdhcFBlclVuaXQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICBnbHlwaE9mZnNldCArPSBzaWRlSW5jcmVtZW50OwotICAgICAgICAgICAgICAgICAgICAgICAgcG9zaXRpb25JbmNyZW1lbnQgPSBnbHlwaE9mZnNldDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGdseXBoT2Zmc2V0ICs9IHNpZGVJbmNyZW1lbnQ7Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICBQb2ludDJEIGdseXBoUG9zID0gZ2V0R2x5cGhWZWN0b3IoKS5nZXRHbHlwaFBvc2l0aW9uKGkpOwotICAgICAgICAgICAgICAgICAgICBnbHlwaFBvcy5zZXRMb2NhdGlvbihnbHlwaFBvcy5nZXRYKCkgKyBwb3NpdGlvbkluY3JlbWVudCwgZ2x5cGhQb3MuZ2V0WSgpKTsKLSAgICAgICAgICAgICAgICAgICAgZ2V0R2x5cGhWZWN0b3IoKS5zZXRHbHlwaFBvc2l0aW9uKGksIGdseXBoUG9zKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGZvciAoaW50IGk9Zmlyc3RHbHlwaDsgaTw9bGFzdEdseXBoOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBnamkgPSBnZXRHbHlwaEp1c3RpZmljYXRpb25JbmZvcygpW2ldOwotICAgICAgICAgICAgICAgICAgICBjdXJySW5mbyA9IGpJbmZvc1tnamkuc2hyaW5rUHJpb3JpdHldOwotICAgICAgICAgICAgICAgICAgICBpZiAoY3VyckluZm8gPT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gV2Ugc3RpbGwgaGF2ZSB0byBpbmNyZW1lbnQgZ2x5cGggcG9zaXRpb24KLSAgICAgICAgICAgICAgICAgICAgICAgIFBvaW50MkQgZ2x5cGhQb3MgPSBnZXRHbHlwaFZlY3RvcigpLmdldEdseXBoUG9zaXRpb24oaSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBnbHlwaFBvcy5zZXRMb2NhdGlvbihnbHlwaFBvcy5nZXRYKCkgKyBnbHlwaE9mZnNldCwgZ2x5cGhQb3MuZ2V0WSgpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGdldEdseXBoVmVjdG9yKCkuc2V0R2x5cGhQb3NpdGlvbihpLCBnbHlwaFBvcyk7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgaWYgKGN1cnJJbmZvLnVzZUxpbWl0cykgewotICAgICAgICAgICAgICAgICAgICAgICAgZ2x5cGhPZmZzZXQgLT0gZ2ppLnNocmlua0xlZnRMaW1pdDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjdXJySW5mby5hYnNvcmIpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaWRlSW5jcmVtZW50ID0gZ2ppLndlaWdodCAqIGN1cnJJbmZvLmFic29yYmVkR2FwUGVyVW5pdDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBnbHlwaE9mZnNldCArPSBzaWRlSW5jcmVtZW50OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBvc2l0aW9uSW5jcmVtZW50ID0gZ2x5cGhPZmZzZXQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2x5cGhPZmZzZXQgKz0gc2lkZUluY3JlbWVudDsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAobGFzdEluZm8gIT0gbnVsbCAmJiBsYXN0SW5mby5wcmlvcml0eSA9PSBjdXJySW5mby5wcmlvcml0eSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpZGVJbmNyZW1lbnQgPSBnamkud2VpZ2h0ICogbGFzdEluZm8uYWJzb3JiZWRHYXBQZXJVbml0OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdseXBoT2Zmc2V0ICs9IHNpZGVJbmNyZW1lbnQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcG9zaXRpb25JbmNyZW1lbnQgPSBnbHlwaE9mZnNldDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBnbHlwaE9mZnNldCArPSBzaWRlSW5jcmVtZW50OwotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBwb3NpdGlvbkluY3JlbWVudCA9IGdseXBoT2Zmc2V0OwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgZ2x5cGhPZmZzZXQgLT0gZ2ppLnNocmlua1JpZ2h0TGltaXQ7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzaWRlSW5jcmVtZW50ID0gIGdqaS53ZWlnaHQgKiBjdXJySW5mby5nYXBQZXJVbml0OwotICAgICAgICAgICAgICAgICAgICAgICAgZ2x5cGhPZmZzZXQgKz0gc2lkZUluY3JlbWVudDsKLSAgICAgICAgICAgICAgICAgICAgICAgIHBvc2l0aW9uSW5jcmVtZW50ID0gZ2x5cGhPZmZzZXQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICBnbHlwaE9mZnNldCArPSBzaWRlSW5jcmVtZW50OwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgUG9pbnQyRCBnbHlwaFBvcyA9IGdldEdseXBoVmVjdG9yKCkuZ2V0R2x5cGhQb3NpdGlvbihpKTsKLSAgICAgICAgICAgICAgICAgICAgZ2x5cGhQb3Muc2V0TG9jYXRpb24oZ2x5cGhQb3MuZ2V0WCgpICsgcG9zaXRpb25JbmNyZW1lbnQsIGdseXBoUG9zLmdldFkoKSk7Ci0gICAgICAgICAgICAgICAgICAgIGdldEdseXBoVmVjdG9yKCkuc2V0R2x5cGhQb3NpdGlvbihpLCBnbHlwaFBvcyk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0KLSAgICAgICAgICAgIGlmIChoYXZlTGFzdCkgeyAgIC8vIERvbid0IGFkZCBwYWRkaW5nIGFmdGVyIGxhc3QgY2hhcgotICAgICAgICAgICAgICAgIGxhc3RHbHlwaCsrOwotCi0gICAgICAgICAgICAgICAgR2x5cGhKdXN0aWZpY2F0aW9uSW5mbyBnamkgPSBnZXRHbHlwaEp1c3RpZmljYXRpb25JbmZvcygpW2xhc3RHbHlwaF07Ci0gICAgICAgICAgICAgICAgY3VyckluZm8gPSBqSW5mb3NbZ2ppLmdyb3dQcmlvcml0eV07Ci0KLSAgICAgICAgICAgICAgICBpZiAoY3VyckluZm8gIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBpZiAoY3VyckluZm8udXNlTGltaXRzKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBnbHlwaE9mZnNldCArPSBmaXJzdEluZm8uZ3JvdyA/IGdqaS5ncm93TGVmdExpbWl0IDogLWdqaS5zaHJpbmtMZWZ0TGltaXQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoY3VyckluZm8uYWJzb3JiKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2x5cGhPZmZzZXQgKz0gZ2ppLndlaWdodCAqIGN1cnJJbmZvLmFic29yYmVkR2FwUGVyVW5pdDsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAobGFzdEluZm8gIT0gbnVsbCAmJiBsYXN0SW5mby5wcmlvcml0eSA9PSBjdXJySW5mby5wcmlvcml0eSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdseXBoT2Zmc2V0ICs9IGdqaS53ZWlnaHQgKiBsYXN0SW5mby5hYnNvcmJlZEdhcFBlclVuaXQ7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBnbHlwaE9mZnNldCArPSBnamkud2VpZ2h0ICogY3VyckluZm8uZ2FwUGVyVW5pdDsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIC8vIEFqdXN0IHBvc2l0aW9ucyBvZiBhbGwgZ2x5cGhzIGFmdGVyIGxhc3QgZ2x5cGgKLSAgICAgICAgICAgICAgICBmb3IgKGludCBpPWxhc3RHbHlwaDsgaTxnZXRHbHlwaFZlY3RvcigpLmdldE51bUdseXBocygpKzE7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBQb2ludDJEIGdseXBoUG9zID0gZ2V0R2x5cGhWZWN0b3IoKS5nZXRHbHlwaFBvc2l0aW9uKGkpOwotICAgICAgICAgICAgICAgICAgICBnbHlwaFBvcy5zZXRMb2NhdGlvbihnbHlwaFBvcy5nZXRYKCkgKyBnbHlwaE9mZnNldCwgZ2x5cGhQb3MuZ2V0WSgpKTsKLSAgICAgICAgICAgICAgICAgICAgZ2V0R2x5cGhWZWN0b3IoKS5zZXRHbHlwaFBvc2l0aW9uKGksIGdseXBoUG9zKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgeyAvLyBVcGRhdGUgcG9zaXRpb24gYWZ0ZXIgbGFzdCBnbHlwaCBpbiBnbHlwaCB2ZWN0b3IgLQotICAgICAgICAgICAgICAgIC8vIHRvIGdldCBjb3JyZWN0IGFkdmFuY2UgZm9yIGl0Ci0gICAgICAgICAgICAgICAgUG9pbnQyRCBnbHlwaFBvcyA9IGdldEdseXBoVmVjdG9yKCkuZ2V0R2x5cGhQb3NpdGlvbihsYXN0R2x5cGgrMSk7Ci0gICAgICAgICAgICAgICAgZ2x5cGhQb3Muc2V0TG9jYXRpb24oZ2x5cGhQb3MuZ2V0WCgpICsgZ2x5cGhPZmZzZXQsIGdseXBoUG9zLmdldFkoKSk7Ci0gICAgICAgICAgICAgICAgZ2V0R2x5cGhWZWN0b3IoKS5zZXRHbHlwaFBvc2l0aW9uKGxhc3RHbHlwaCsxLCBnbHlwaFBvcyk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGdqaXMgPSBudWxsOyAvLyBXZSBkb24ndCBuZWVkIGp1c3RpZmljYXRpb24gaW5mb3MgYW55IG1vcmUKLSAgICAgICAgICAgIC8vIEFsc28gd2UgaGF2ZSB0byByZXNldCBjYWNoZWQgYm91bmRzIGFuZCBtZXRyaWNzCi0gICAgICAgICAgICB0aGlzLnZpc3VhbEJvdW5kcyA9IG51bGw7Ci0gICAgICAgICAgICB0aGlzLmxvZ2ljYWxCb3VuZHMgPSBudWxsOwotCi0gICAgICAgICAgICByZXR1cm4gZ2x5cGhPZmZzZXQ7IC8vIEhvdyBtdWNoIG91ciBzZWdtZW50IGdyb3duIG9yIHNocnVuawotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIHN0YXRpYyBjbGFzcyBUZXh0UnVuU2VnbWVudEdyYXBoaWMgZXh0ZW5kcyBUZXh0UnVuU2VnbWVudCB7Ci0gICAgICAgIEdyYXBoaWNBdHRyaWJ1dGUgZ2E7Ci0gICAgICAgIGludCBzdGFydDsKLSAgICAgICAgaW50IGxlbmd0aDsKLSAgICAgICAgZmxvYXQgZnVsbEFkdmFuY2U7Ci0KLSAgICAgICAgVGV4dFJ1blNlZ21lbnRHcmFwaGljKEdyYXBoaWNBdHRyaWJ1dGUgYXR0ciwgaW50IGxlbiwgaW50IHN0YXJ0KSB7Ci0gICAgICAgICAgICB0aGlzLnN0YXJ0ID0gc3RhcnQ7Ci0gICAgICAgICAgICBsZW5ndGggPSBsZW47Ci0gICAgICAgICAgICBnYSA9IGF0dHI7Ci0gICAgICAgICAgICBtZXRyaWNzID0gbmV3IEJhc2ljTWV0cmljcyhnYSk7Ci0gICAgICAgICAgICBmdWxsQWR2YW5jZSA9IGdhLmdldEFkdmFuY2UoKSAqIGxlbmd0aDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgT2JqZWN0IGNsb25lKCkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBUZXh0UnVuU2VnbWVudEdyYXBoaWMoZ2EsIGxlbmd0aCwgc3RhcnQpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gUmVuZGVycyB0aGlzIHRleHQgcnVuIHNlZ21lbnQKLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHZvaWQgZHJhdyhHcmFwaGljczJEIGcyZCwgZmxvYXQgeE9mZnNldCwgZmxvYXQgeU9mZnNldCkgewotICAgICAgICAgICAgaWYgKGRlY29yYXRpb24gIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIFRleHREZWNvcmF0b3IucHJlcGFyZUdyYXBoaWNzKHRoaXMsIGcyZCwgeE9mZnNldCwgeU9mZnNldCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGZsb2F0IHhQb3MgPSB4ICsgeE9mZnNldDsKLSAgICAgICAgICAgIGZsb2F0IHlQb3MgPSB5ICsgeU9mZnNldDsKLQotICAgICAgICAgICAgZm9yIChpbnQgaT0wOyBpIDwgbGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgICAgICBnYS5kcmF3KGcyZCwgeFBvcywgeVBvcyk7Ci0gICAgICAgICAgICAgICAgeFBvcyArPSBnYS5nZXRBZHZhbmNlKCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChkZWNvcmF0aW9uICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBUZXh0RGVjb3JhdG9yLmRyYXdUZXh0RGVjb3JhdGlvbnModGhpcywgZzJkLCB4T2Zmc2V0LCB5T2Zmc2V0KTsKLSAgICAgICAgICAgICAgICBUZXh0RGVjb3JhdG9yLnJlc3RvcmVHcmFwaGljcyhkZWNvcmF0aW9uLCBnMmQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLy8gUmV0dXJucyB2aXN1YWwgYm91bmRzIG9mIHRoaXMgc2VnbWVudAotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgUmVjdGFuZ2xlMkQgZ2V0VmlzdWFsQm91bmRzKCkgewotICAgICAgICAgICAgaWYgKHZpc3VhbEJvdW5kcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgUmVjdGFuZ2xlMkQgYm91bmRzID0gZ2EuZ2V0Qm91bmRzKCk7Ci0KLSAgICAgICAgICAgICAgICAvLyBGaXJzdCBhbmQgbGFzdCBjaGFycyBjYW4gYmUgb3V0IG9mIGxvZ2ljYWwgYm91bmRzLCBzbyB3ZSBjYWxjdWxhdGUKLSAgICAgICAgICAgICAgICAvLyAoYm91bmRzLmdldFdpZHRoKCkgLSBnYS5nZXRBZHZhbmNlKCkpIHdoaWNoIGlzIGV4YWN0bHkgdGhlIGRpZmZlcmVuY2UKLSAgICAgICAgICAgICAgICBib3VuZHMuc2V0UmVjdCgKLSAgICAgICAgICAgICAgICAgICAgICAgIGJvdW5kcy5nZXRNaW5YKCkgKyB4LAotICAgICAgICAgICAgICAgICAgICAgICAgYm91bmRzLmdldE1pblkoKSArIHksCi0gICAgICAgICAgICAgICAgICAgICAgICBib3VuZHMuZ2V0V2lkdGgoKSAtIGdhLmdldEFkdmFuY2UoKSArIGdldEFkdmFuY2UoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGJvdW5kcy5nZXRIZWlnaHQoKQotICAgICAgICAgICAgICAgICk7Ci0gICAgICAgICAgICAgICAgdmlzdWFsQm91bmRzID0gVGV4dERlY29yYXRvci5leHRlbmRWaXN1YWxCb3VuZHModGhpcywgYm91bmRzLCBkZWNvcmF0aW9uKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIChSZWN0YW5nbGUyRCkgdmlzdWFsQm91bmRzLmNsb25lKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgUmVjdGFuZ2xlMkQgZ2V0TG9naWNhbEJvdW5kcygpIHsKLSAgICAgICAgICAgIGlmIChsb2dpY2FsQm91bmRzID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBsb2dpY2FsQm91bmRzID0KLSAgICAgICAgICAgICAgICAgICAgICAgIG5ldyBSZWN0YW5nbGUyRC5GbG9hdCgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgeCwgeSAtIG1ldHJpY3MuYXNjZW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnZXRBZHZhbmNlKCksIG1ldHJpY3MuYXNjZW50ICsgbWV0cmljcy5kZXNjZW50Ci0gICAgICAgICAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZXR1cm4gKFJlY3RhbmdsZTJEKSBsb2dpY2FsQm91bmRzLmNsb25lKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgZmxvYXQgZ2V0QWR2YW5jZSgpIHsKLSAgICAgICAgICAgIHJldHVybiBmdWxsQWR2YW5jZTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBmbG9hdCBnZXRBZHZhbmNlRGVsdGEoaW50IHN0YXJ0LCBpbnQgZW5kKSB7Ci0gICAgICAgICAgICByZXR1cm4gZ2EuZ2V0QWR2YW5jZSgpICogKGVuZCAtIHN0YXJ0KTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBpbnQgZ2V0Q2hhckluZGV4RnJvbUFkdmFuY2UoZmxvYXQgYWR2YW5jZSwgaW50IHN0YXJ0KSB7Ci0gICAgICAgICAgICBzdGFydCAtPSB0aGlzLnN0YXJ0OwotCi0gICAgICAgICAgICBpZiAoc3RhcnQgPCAwKSB7Ci0gICAgICAgICAgICAgICAgc3RhcnQgPSAwOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpbnQgY2hhck9mZnNldCA9IChpbnQpIChhZHZhbmNlL2dhLmdldEFkdmFuY2UoKSk7Ci0KLSAgICAgICAgICAgIGlmIChjaGFyT2Zmc2V0ICsgc3RhcnQgPiBsZW5ndGgpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbGVuZ3RoICsgdGhpcy5zdGFydDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBjaGFyT2Zmc2V0ICsgc3RhcnQgKyB0aGlzLnN0YXJ0OwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIGludCBnZXRTdGFydCgpIHsKLSAgICAgICAgICAgIHJldHVybiBzdGFydDsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBpbnQgZ2V0RW5kKCkgewotICAgICAgICAgICAgcmV0dXJuIHN0YXJ0ICsgbGVuZ3RoOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIGludCBnZXRMZW5ndGgoKSB7Ci0gICAgICAgICAgICByZXR1cm4gbGVuZ3RoOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIFNoYXBlIGdldENoYXJzQmxhY2tCb3hCb3VuZHMoaW50IHN0YXJ0LCBpbnQgbGltaXQpIHsKLSAgICAgICAgICAgIHN0YXJ0IC09IHRoaXMuc3RhcnQ7Ci0gICAgICAgICAgICBsaW1pdCAtPSB0aGlzLnN0YXJ0OwotCi0gICAgICAgICAgICBpZiAobGltaXQgPiBsZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBsaW1pdCA9IGxlbmd0aDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgUmVjdGFuZ2xlMkQgY2hhckJvdW5kcyA9IGdhLmdldEJvdW5kcygpOwotICAgICAgICAgICAgY2hhckJvdW5kcy5zZXRSZWN0KAotICAgICAgICAgICAgICAgICAgICBjaGFyQm91bmRzLmdldFgoKSArIGdhLmdldEFkdmFuY2UoKSAqIHN0YXJ0ICsgeCwKLSAgICAgICAgICAgICAgICAgICAgY2hhckJvdW5kcy5nZXRZKCkgKyB5LAotICAgICAgICAgICAgICAgICAgICBjaGFyQm91bmRzLmdldFdpZHRoKCkgKyBnYS5nZXRBZHZhbmNlKCkgKiAobGltaXQgLSBzdGFydCksCi0gICAgICAgICAgICAgICAgICAgIGNoYXJCb3VuZHMuZ2V0SGVpZ2h0KCkKLSAgICAgICAgICAgICk7Ci0KLSAgICAgICAgICAgIHJldHVybiBjaGFyQm91bmRzOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIGZsb2F0IGdldENoYXJQb3NpdGlvbihpbnQgaW5kZXgpIHsKLSAgICAgICAgICAgIGluZGV4IC09IHN0YXJ0OwotICAgICAgICAgICAgaWYgKGluZGV4ID4gbGVuZ3RoKSB7Ci0gICAgICAgICAgICAgICAgaW5kZXggPSBsZW5ndGg7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBnYS5nZXRBZHZhbmNlKCkgKiBpbmRleCArIHg7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgZmxvYXQgZ2V0Q2hhckFkdmFuY2UoaW50IGluZGV4KSB7Ci0gICAgICAgICAgICByZXR1cm4gZ2EuZ2V0QWR2YW5jZSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIFNoYXBlIGdldE91dGxpbmUoKSB7Ci0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gdCA9IEFmZmluZVRyYW5zZm9ybS5nZXRUcmFuc2xhdGVJbnN0YW5jZSh4LCB5KTsKLSAgICAgICAgICAgIHJldHVybiB0LmNyZWF0ZVRyYW5zZm9ybWVkU2hhcGUoCi0gICAgICAgICAgICAgICAgICAgIFRleHREZWNvcmF0b3IuZXh0ZW5kT3V0bGluZSh0aGlzLCBnZXRWaXN1YWxCb3VuZHMoKSwgZGVjb3JhdGlvbikKLSAgICAgICAgICAgICk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgYm9vbGVhbiBjaGFySGFzWmVyb0FkdmFuY2UoaW50IGluZGV4KSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgVGV4dEhpdEluZm8gaGl0VGVzdChmbG9hdCBoaXRYLCBmbG9hdCBoaXRZKSB7Ci0gICAgICAgICAgICBoaXRYIC09IHg7Ci0KLSAgICAgICAgICAgIGZsb2F0IHRtcCA9IGhpdFggLyBnYS5nZXRBZHZhbmNlKCk7Ci0gICAgICAgICAgICBpbnQgaGl0SW5kZXggPSBNYXRoLnJvdW5kKHRtcCk7Ci0KLSAgICAgICAgICAgIGlmICh0bXAgPiBoaXRJbmRleCkgewotICAgICAgICAgICAgICAgIHJldHVybiBUZXh0SGl0SW5mby5sZWFkaW5nKGhpdEluZGV4ICsgdGhpcy5zdGFydCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gVGV4dEhpdEluZm8udHJhaWxpbmcoaGl0SW5kZXggKyB0aGlzLnN0YXJ0KTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICB2b2lkIHVwZGF0ZUp1c3RpZmljYXRpb25JbmZvKFRleHRSdW5CcmVha2VyLkp1c3RpZmljYXRpb25JbmZvIGpJbmZvKSB7Ci0gICAgICAgICAgICAvLyBEbyBub3RoaW5nCi0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgZmxvYXQgZG9KdXN0aWZpY2F0aW9uKFRleHRSdW5CcmVha2VyLkp1c3RpZmljYXRpb25JbmZvIGpJbmZvc1tdKSB7Ci0gICAgICAgICAgICAvLyBEbyBub3RoaW5nCi0gICAgICAgICAgICByZXR1cm4gMDsKLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0J1ZmZlcmVkSW1hZ2VHcmFwaGljczJELmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9CdWZmZXJlZEltYWdlR3JhcGhpY3MyRC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmMWQ2NGZiLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0J1ZmZlcmVkSW1hZ2VHcmFwaGljczJELmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw3OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIEFsZXhleSBBLiBQZXRyZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljczsKLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljc0NvbmZpZ3VyYXRpb247Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Xcml0YWJsZVJhc3RlcjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuQ29tbW9uR3JhcGhpY3MyRDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLlN1cmZhY2U7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5yZW5kZXIuSmF2YUJsaXR0ZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5yZW5kZXIuTmF0aXZlSW1hZ2VCbGl0dGVyOwotCi0vKioKLSAqIEJ1ZmZlcmVkSW1hZ2VHcmFwaGljczJEIGlzIGltcGxlbWVudGF0aW9uIG9mIENvbW1vbkdyYXBoaWNzMkQgZm9yCi0gKiBkcmF3aW5nIG9uIGJ1ZmZlcmVkIGltYWdlcy4gCi0gKi8KLXB1YmxpYyBjbGFzcyBCdWZmZXJlZEltYWdlR3JhcGhpY3MyRCBleHRlbmRzIENvbW1vbkdyYXBoaWNzMkQgewotICAgIHByaXZhdGUgQnVmZmVyZWRJbWFnZSBiaSA9IG51bGw7Ci0gICAgcHJpdmF0ZSBSZWN0YW5nbGUgYm91bmRzID0gbnVsbDsKLQotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlR3JhcGhpY3MyRChCdWZmZXJlZEltYWdlIGJpKSB7Ci0gICAgICAgIHN1cGVyKCk7Ci0gICAgICAgIHRoaXMuYmkgPSBiaTsKLSAgICAgICAgdGhpcy5ib3VuZHMgPSBuZXcgUmVjdGFuZ2xlKDAsIDAsIGJpLmdldFdpZHRoKCksIGJpLmdldEhlaWdodCgpKTsKLSAgICAgICAgY2xpcChib3VuZHMpOwotICAgICAgICBkc3RTdXJmID0gU3VyZmFjZS5nZXRJbWFnZVN1cmZhY2UoYmkpOwotICAgICAgICBpZihkc3RTdXJmLmlzTmF0aXZlRHJhd2FibGUoKSl7Ci0gICAgICAgICAgICBibGl0dGVyID0gTmF0aXZlSW1hZ2VCbGl0dGVyLmdldEluc3RhbmNlKCk7Ci0gICAgICAgIH1lbHNlewotICAgICAgICAgICAgYmxpdHRlciA9IEphdmFCbGl0dGVyLmdldEluc3RhbmNlKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBjb3B5QXJlYShpbnQgeCwgaW50IHksIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGR4LCBpbnQgZHkpIHsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR3JhcGhpY3MgY3JlYXRlKCkgewotICAgICAgICBCdWZmZXJlZEltYWdlR3JhcGhpY3MyRCByZXMgPSBuZXcgQnVmZmVyZWRJbWFnZUdyYXBoaWNzMkQoYmkpOwotICAgICAgICBjb3B5SW50ZXJuYWxGaWVsZHMocmVzKTsKLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgR3JhcGhpY3NDb25maWd1cmF0aW9uIGdldERldmljZUNvbmZpZ3VyYXRpb24oKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIHB1YmxpYyBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoKSB7Ci0gICAgICAgIHJldHVybiBiaS5nZXRDb2xvck1vZGVsKCk7Ci0gICAgfQotCi0gICAgcHVibGljIFdyaXRhYmxlUmFzdGVyIGdldFdyaXRhYmxlUmFzdGVyKCkgewotICAgICAgICByZXR1cm4gYmkuZ2V0UmFzdGVyKCk7Ci0gICAgfQotfQpcIE5vIG5ld2xpbmUgYXQgZW5kIG9mIGZpbGUKZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0J1ZmZlcmVkSW1hZ2VTb3VyY2UuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0J1ZmZlcmVkSW1hZ2VTb3VyY2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMGZlMjVhMi4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9CdWZmZXJlZEltYWdlU291cmNlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMzYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZTsKLQotaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db21wb25lbnRDb2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGF0YUJ1ZmZlckJ5dGU7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGF0YUJ1ZmZlckludDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EaXJlY3RDb2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkltYWdlQ29uc3VtZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW1hZ2VQcm9kdWNlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbmRleENvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuV3JpdGFibGVSYXN0ZXI7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2h0YWJsZTsKLQotcHVibGljIGNsYXNzIEJ1ZmZlcmVkSW1hZ2VTb3VyY2UgaW1wbGVtZW50cyBJbWFnZVByb2R1Y2VyIHsKLQotICAgIHByaXZhdGUgSGFzaHRhYmxlPD8sID8+IHByb3BlcnRpZXM7Ci0gICAgcHJpdmF0ZSBDb2xvck1vZGVsIGNtOwotICAgIHByaXZhdGUgV3JpdGFibGVSYXN0ZXIgcmFzdGVyOwotICAgIHByaXZhdGUgaW50IHdpZHRoOwotICAgIHByaXZhdGUgaW50IGhlaWdodDsKLQotICAgIHByaXZhdGUgSW1hZ2VDb25zdW1lciBpYzsKLQotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlU291cmNlKEJ1ZmZlcmVkSW1hZ2UgaW1hZ2UsIEhhc2h0YWJsZTw/LCA/PiBwcm9wZXJ0aWVzKXsKLSAgICAgICAgaWYocHJvcGVydGllcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aGlzLnByb3BlcnRpZXMgPSBuZXcgSGFzaHRhYmxlPE9iamVjdCwgT2JqZWN0PigpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdGhpcy5wcm9wZXJ0aWVzID0gcHJvcGVydGllczsKLSAgICAgICAgfQotCi0gICAgICAgIHdpZHRoID0gaW1hZ2UuZ2V0V2lkdGgoKTsKLSAgICAgICAgaGVpZ2h0ID0gaW1hZ2UuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgIGNtID0gaW1hZ2UuZ2V0Q29sb3JNb2RlbCgpOwotICAgICAgICByYXN0ZXIgPSBpbWFnZS5nZXRSYXN0ZXIoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgQnVmZmVyZWRJbWFnZVNvdXJjZShCdWZmZXJlZEltYWdlIGltYWdlKXsKLSAgICAgICAgdGhpcyhpbWFnZSwgbnVsbCk7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb25zdW1lcihJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIHJldHVybiAodGhpcy5pYyA9PSBpYyk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc3RhcnRQcm9kdWN0aW9uKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgYWRkQ29uc3VtZXIoaWMpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlcXVlc3RUb3BEb3duTGVmdFJpZ2h0UmVzZW5kKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZW1vdmVDb25zdW1lcihJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIGlmICh0aGlzLmljID09IGljKSB7Ci0gICAgICAgICAgICB0aGlzLmljID0gbnVsbDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGFkZENvbnN1bWVyKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgdGhpcy5pYyA9IGljOwotICAgICAgICBzdGFydFByb2R1Y3Rpb24oKTsKLSAgICB9Ci0KLSAgICBwcml2YXRlIHZvaWQgc3RhcnRQcm9kdWN0aW9uKCl7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpYy5zZXREaW1lbnNpb25zKHdpZHRoLCBoZWlnaHQpOwotICAgICAgICAgICAgaWMuc2V0UHJvcGVydGllcyhwcm9wZXJ0aWVzKTsKLSAgICAgICAgICAgIGljLnNldENvbG9yTW9kZWwoY20pOwotICAgICAgICAgICAgaWMuc2V0SGludHMoSW1hZ2VDb25zdW1lci5UT1BET1dOTEVGVFJJR0hUIHwKLSAgICAgICAgICAgICAgICAgICAgSW1hZ2VDb25zdW1lci5DT01QTEVURVNDQU5MSU5FUyB8Ci0gICAgICAgICAgICAgICAgICAgIEltYWdlQ29uc3VtZXIuU0lOR0xFRlJBTUUgfAotICAgICAgICAgICAgICAgICAgICBJbWFnZUNvbnN1bWVyLlNJTkdMRVBBU1MpOwotICAgICAgICAgICAgaWYoY20gaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwgJiYKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyLmdldFRyYW5zZmVyVHlwZSgpID09IERhdGFCdWZmZXIuVFlQRV9CWVRFIHx8Ci0gICAgICAgICAgICAgICAgICAgIGNtIGluc3RhbmNlb2YgQ29tcG9uZW50Q29sb3JNb2RlbCAmJgotICAgICAgICAgICAgICAgICAgICByYXN0ZXIuZ2V0VHJhbnNmZXJUeXBlKCkgPT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyLmdldE51bURhdGFFbGVtZW50cygpID09IDEpewotICAgICAgICAgICAgICAgIERhdGFCdWZmZXJCeXRlIGRiYiA9IChEYXRhQnVmZmVyQnl0ZSkgcmFzdGVyLmdldERhdGFCdWZmZXIoKTsKLSAgICAgICAgICAgICAgICBieXRlIGRhdGFbXSA9IGRiYi5nZXREYXRhKCk7Ci0gICAgICAgICAgICAgICAgaW50IG9mZiA9IGRiYi5nZXRPZmZzZXQoKTsKLSAgICAgICAgICAgICAgICBpYy5zZXRQaXhlbHMoMCwgMCwgd2lkdGgsIGhlaWdodCwgY20sIGRhdGEsIG9mZiwgd2lkdGgpOwotICAgICAgICAgICAgfWVsc2UgaWYoY20gaW5zdGFuY2VvZiBEaXJlY3RDb2xvck1vZGVsICYmCi0gICAgICAgICAgICAgICAgICAgIHJhc3Rlci5nZXRUcmFuc2ZlclR5cGUoKSA9PSBEYXRhQnVmZmVyLlRZUEVfSU5UKXsKLSAgICAgICAgICAgICAgICBEYXRhQnVmZmVySW50IGRiaSA9IChEYXRhQnVmZmVySW50KSByYXN0ZXIuZ2V0RGF0YUJ1ZmZlcigpOwotICAgICAgICAgICAgICAgIGludCBkYXRhW10gPSBkYmkuZ2V0RGF0YSgpOwotICAgICAgICAgICAgICAgIGludCBvZmYgPSBkYmkuZ2V0T2Zmc2V0KCk7Ci0gICAgICAgICAgICAgICAgaWMuc2V0UGl4ZWxzKDAsIDAsIHdpZHRoLCBoZWlnaHQsIGNtLCBkYXRhLCBvZmYsIHdpZHRoKTsKLSAgICAgICAgICAgIH1lbHNlIGlmKGNtIGluc3RhbmNlb2YgRGlyZWN0Q29sb3JNb2RlbCAmJgotICAgICAgICAgICAgICAgICAgICByYXN0ZXIuZ2V0VHJhbnNmZXJUeXBlKCkgPT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUpewotICAgICAgICAgICAgICAgIERhdGFCdWZmZXJCeXRlIGRiYiA9IChEYXRhQnVmZmVyQnl0ZSkgcmFzdGVyLmdldERhdGFCdWZmZXIoKTsKLSAgICAgICAgICAgICAgICBieXRlIGRhdGFbXSA9IGRiYi5nZXREYXRhKCk7Ci0gICAgICAgICAgICAgICAgaW50IG9mZiA9IGRiYi5nZXRPZmZzZXQoKTsKLSAgICAgICAgICAgICAgICBpYy5zZXRQaXhlbHMoMCwgMCwgd2lkdGgsIGhlaWdodCwgY20sIGRhdGEsIG9mZiwgd2lkdGgpOwotICAgICAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICAgICAgQ29sb3JNb2RlbCByZ2JDTSA9IENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpOwotICAgICAgICAgICAgICAgIGludCBwaXhlbHNbXSA9IG5ldyBpbnRbd2lkdGhdOwotICAgICAgICAgICAgICAgIE9iamVjdCBwaXggPSBudWxsOwotICAgICAgICAgICAgICAgIGZvcihpbnQgeSA9IDA7IHkgPCBoZWlnaHQ7IHkrKyl7Ci0gICAgICAgICAgICAgICAgICAgIGZvcihpbnQgeCA9IDAgOyB4IDwgd2lkdGg7IHgrKyl7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwaXggPSByYXN0ZXIuZ2V0RGF0YUVsZW1lbnRzKHgsIHksIHBpeCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwaXhlbHNbeF0gPSBjbS5nZXRSR0IocGl4KTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpYy5zZXRQaXhlbHMoMCwgeSwgd2lkdGgsIDEsIHJnYkNNLCBwaXhlbHMsIDAsIHdpZHRoKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpYy5pbWFnZUNvbXBsZXRlKEltYWdlQ29uc3VtZXIuU1RBVElDSU1BR0VET05FKTsKLSAgICAgICAgfWNhdGNoIChOdWxsUG9pbnRlckV4Y2VwdGlvbiBlKXsKLSAgICAgICAgICAgIGlmIChpYyAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaWMuaW1hZ2VDb21wbGV0ZShJbWFnZUNvbnN1bWVyLklNQUdFRVJST1IpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9CeXRlQXJyYXlEZWNvZGluZ0ltYWdlU291cmNlLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9CeXRlQXJyYXlEZWNvZGluZ0ltYWdlU291cmNlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGNjNmQ3Y2YuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvQnl0ZUFycmF5RGVjb2RpbmdJbWFnZVNvdXJjZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNjIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0vKgotICogQ3JlYXRlZCBvbiAxMC4wMi4yMDA1Ci0gKgotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmlvLkJ1ZmZlcmVkSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5CeXRlQXJyYXlJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLklucHV0U3RyZWFtOwotCi1wdWJsaWMgY2xhc3MgQnl0ZUFycmF5RGVjb2RpbmdJbWFnZVNvdXJjZSBleHRlbmRzIERlY29kaW5nSW1hZ2VTb3VyY2UgewotCi0gICAgYnl0ZSBpbWFnZWRhdGFbXTsKLSAgICBpbnQgaW1hZ2VvZmZzZXQ7Ci0gICAgaW50IGltYWdlbGVuZ3RoOwotCi0gICAgcHVibGljIEJ5dGVBcnJheURlY29kaW5nSW1hZ2VTb3VyY2UoYnl0ZSBpbWFnZWRhdGFbXSwgaW50IGltYWdlb2Zmc2V0LAotICAgICAgICAgICAgaW50IGltYWdlbGVuZ3RoKXsKLSAgICAgICAgdGhpcy5pbWFnZWRhdGEgPSBpbWFnZWRhdGE7Ci0gICAgICAgIHRoaXMuaW1hZ2VvZmZzZXQgPSBpbWFnZW9mZnNldDsKLSAgICAgICAgdGhpcy5pbWFnZWxlbmd0aCA9IGltYWdlbGVuZ3RoOwotICAgIH0KLQotICAgIHB1YmxpYyBCeXRlQXJyYXlEZWNvZGluZ0ltYWdlU291cmNlKGJ5dGUgaW1hZ2VkYXRhW10pewotICAgICAgICB0aGlzKGltYWdlZGF0YSwgMCwgaW1hZ2VkYXRhLmxlbmd0aCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHJvdGVjdGVkIGJvb2xlYW4gY2hlY2tDb25uZWN0aW9uKCkgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgSW5wdXRTdHJlYW0gZ2V0SW5wdXRTdHJlYW0oKSB7Ci0gICAgICAgIC8vIEJFR0lOIGFuZHJvaWQtbW9kaWZpZWQKLSAgICAgICAgLy8gVE9ETzogV2h5IGRvZXMgYSBCeXRlQXJyYXlJbnB1dFN0cmVhbSBuZWVkIHRvIGJlIGJ1ZmZlcmVkIGF0IGFsbD8KLSAgICAgICAgcmV0dXJuIG5ldyBCdWZmZXJlZElucHV0U3RyZWFtKG5ldyBCeXRlQXJyYXlJbnB1dFN0cmVhbShpbWFnZWRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZW9mZnNldCwgaW1hZ2VsZW5ndGgpLCAxMDI0KTsKLSAgICAgICAgLy8gRU5EIGFuZHJvaWQtbW9kaWZpZWQKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0RhdGFCdWZmZXJMaXN0ZW5lci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvRGF0YUJ1ZmZlckxpc3RlbmVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDg3OTMwNTAuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvRGF0YUJ1ZmZlckxpc3RlbmVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwzMSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIElnb3IgVi4gU3RvbHlhcm92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKiBDcmVhdGVkIG9uIDEzLjAzLjIwMDYKLSAqCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZTsKLQotcHVibGljIGludGVyZmFjZSBEYXRhQnVmZmVyTGlzdGVuZXIgewotICAgIAotICAgIHZvaWQgZGF0YUNoYW5nZWQoKTsKLSAgICB2b2lkIGRhdGFUYWtlbigpOwotICAgIHZvaWQgZGF0YVJlbGVhc2VkKCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0RlY29kaW5nSW1hZ2VTb3VyY2UuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0RlY29kaW5nSW1hZ2VTb3VyY2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTU4ZDY5MS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9EZWNvZGluZ0ltYWdlU291cmNlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNjEgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBPbGVnIFYuIEtoYXNjaGFuc2t5Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLS8qCi0gKiBDcmVhdGVkIG9uIDE4LjAxLjIwMDUKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmltYWdlOwotCi1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW1hZ2VDb25zdW1lcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbWFnZVByb2R1Y2VyOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlMaXN0OwotaW1wb3J0IGphdmEudXRpbC5JdGVyYXRvcjsKLWltcG9ydCBqYXZhLnV0aWwuTGlzdDsKLQotLyoqCi0gKiBUaGlzIGlzIGFuIGFic3RyYWN0IGNsYXNzIHRoYXQgZW5jYXBzdWxhdGVzIGEgbWFpbiBwYXJ0IG9mIEltYWdlUHJvZHVjZXIgZnVuY3Rpb25hbGl0eQotICogZm9yIHRoZSBpbWFnZXMgYmVpbmcgZGVjb2RlZCBieSB0aGUgbmF0aXZlIGRlY29kZXJzLCBsaWtlIFBORywgSlBFRyBhbmQgR0lGLgotICogSXQgaGVscHMgdG8gaW50ZWdyYXRlIGltYWdlIGRlY29kZXJzIGludG8gcHJvZHVjZXIvY29uc3VtZXIgbW9kZWwuIEl0IHByb3ZpZGVzCi0gKiBmdW5jdGlvbmFsaXR5IGZvciB3b3JraW5nIHdpdGggc2V2ZXJhbCBkZWNvZGVyIGluc3RhbmNlcyBhbmQgc2V2ZXJhbCBpbWFnZSBjb25zdW1lcnMKLSAqIHNpbXVsdGFuZW91c2x5LgotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgRGVjb2RpbmdJbWFnZVNvdXJjZSBpbXBsZW1lbnRzIEltYWdlUHJvZHVjZXIgewotICAgIExpc3Q8SW1hZ2VDb25zdW1lcj4gY29uc3VtZXJzID0gbmV3IEFycmF5TGlzdDxJbWFnZUNvbnN1bWVyPig1KTsKLSAgICBMaXN0PEltYWdlRGVjb2Rlcj4gZGVjb2RlcnMgPSBuZXcgQXJyYXlMaXN0PEltYWdlRGVjb2Rlcj4oNSk7Ci0gICAgYm9vbGVhbiBsb2FkaW5nOwotCi0gICAgSW1hZ2VEZWNvZGVyIGRlY29kZXI7Ci0KLSAgICBwcm90ZWN0ZWQgYWJzdHJhY3QgYm9vbGVhbiBjaGVja0Nvbm5lY3Rpb24oKTsKLQotICAgIHByb3RlY3RlZCBhYnN0cmFjdCBJbnB1dFN0cmVhbSBnZXRJbnB1dFN0cmVhbSgpOwotCi0gICAgcHVibGljIHN5bmNocm9uaXplZCB2b2lkIGFkZENvbnN1bWVyKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgaWYgKCFjaGVja0Nvbm5lY3Rpb24oKSkgeyAvLyBObyBwZXJtaXNzaW9uIGZvciB0aGlzIGNvbnN1bWVyCi0gICAgICAgICAgICBpYy5pbWFnZUNvbXBsZXRlKEltYWdlQ29uc3VtZXIuSU1BR0VFUlJPUik7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBJbWFnZUNvbnN1bWVyIGNvbnMgPSBmaW5kQ29uc3VtZXIoY29uc3VtZXJzLCBpYyk7Ci0KLSAgICAgICAgaWYgKGNvbnMgPT0gbnVsbCkgeyAvLyBUcnkgdG8gbG9vayBpbiB0aGUgZGVjb2RlcnMKLSAgICAgICAgICAgIEltYWdlRGVjb2RlciBkID0gbnVsbDsKLQotICAgICAgICAgICAgLy8gQ2hlY2sgZm9yIGFsbCBleGlzdGluZyBkZWNvZGVycwotICAgICAgICAgICAgZm9yIChJdGVyYXRvcjxJbWFnZURlY29kZXI+IGkgPSBkZWNvZGVycy5pdGVyYXRvcigpOyBpLmhhc05leHQoKTspIHsKLSAgICAgICAgICAgICAgICBkID0gaS5uZXh0KCk7Ci0gICAgICAgICAgICAgICAgY29ucyA9IGZpbmRDb25zdW1lcihkLmNvbnN1bWVycywgaWMpOwotICAgICAgICAgICAgICAgIGlmIChjb25zICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGNvbnMgPT0gbnVsbCkgeyAvLyBOb3QgZm91bmQsIGFkZCB0aGlzIGNvbnN1bWVyCi0gICAgICAgICAgICBjb25zdW1lcnMuYWRkKGljKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoaXMgbWV0aG9kIHN0b3BzIHNlbmRpbmcgZGF0YSB0byB0aGUgZ2l2ZW4gY29uc3VtZXIKLSAgICAgKiBAcGFyYW0gaWMgLSBjb25zdW1lcgotICAgICAqLwotICAgIHByaXZhdGUgdm9pZCBhYm9ydENvbnN1bWVyKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgaWMuaW1hZ2VDb21wbGV0ZShJbWFnZUNvbnN1bWVyLklNQUdFRVJST1IpOwotICAgICAgICBjb25zdW1lcnMucmVtb3ZlKGljKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCBzdG9wcyBzZW5kaW5nIGRhdGEgdG8gdGhlIGxpc3Qgb2YgY29uc3VtZXJzLgotICAgICAqIEBwYXJhbSBjb25zdW1lcnNMaXN0IC0gbGlzdCBvZiBjb25zdW1lcnMKLSAgICAgKi8KLSAgICBwcml2YXRlIHZvaWQgYWJvcnRBbGxDb25zdW1lcnMoTGlzdDxJbWFnZUNvbnN1bWVyPiBjb25zdW1lcnNMaXN0KSB7Ci0gICAgICAgIGZvciAoSW1hZ2VDb25zdW1lciBpbWFnZUNvbnN1bWVyIDogY29uc3VtZXJzTGlzdCkgewotICAgICAgICAgICAgYWJvcnRDb25zdW1lcihpbWFnZUNvbnN1bWVyKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCByZW1vdmVDb25zdW1lcihJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIEltYWdlRGVjb2RlciBkID0gbnVsbDsKLQotICAgICAgICAvLyBSZW1vdmUgaW4gYWxsIGV4aXN0aW5nIGRlY29kZXJzCi0gICAgICAgIGZvciAoSXRlcmF0b3I8SW1hZ2VEZWNvZGVyPiBpID0gZGVjb2RlcnMuaXRlcmF0b3IoKTsgaS5oYXNOZXh0KCk7KSB7Ci0gICAgICAgICAgICBkID0gaS5uZXh0KCk7Ci0gICAgICAgICAgICByZW1vdmVDb25zdW1lcihkLmNvbnN1bWVycywgaWMpOwotICAgICAgICAgICAgaWYgKGQuY29uc3VtZXJzLnNpemUoKSA8PSAwKSB7Ci0gICAgICAgICAgICAgICAgZC50ZXJtaW5hdGUoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIFJlbW92ZSBpbiB0aGUgY3VycmVudCBxdWV1ZSBvZiBjb25zdW1lcnMKLSAgICAgICAgcmVtb3ZlQ29uc3VtZXIoY29uc3VtZXJzLCBpYyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU3RhdGljIGltcGxlbWVudGF0aW9uIG9mIHJlbW92ZUNvbnN1bWVyIG1ldGhvZAotICAgICAqIEBwYXJhbSBjb25zdW1lcnNMaXN0IC0gbGlzdCBvZiBjb25zdW1lcnMKLSAgICAgKiBAcGFyYW0gaWMgLSBjb25zdW1lciB0byBiZSByZW1vdmVkCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgdm9pZCByZW1vdmVDb25zdW1lcihMaXN0PEltYWdlQ29uc3VtZXI+IGNvbnN1bWVyc0xpc3QsIEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgSW1hZ2VDb25zdW1lciBjb25zID0gbnVsbDsKLQotICAgICAgICBmb3IgKEl0ZXJhdG9yPEltYWdlQ29uc3VtZXI+IGkgPSBjb25zdW1lcnNMaXN0Lml0ZXJhdG9yKCk7IGkuaGFzTmV4dCgpOykgewotICAgICAgICAgICAgY29ucyA9IGkubmV4dCgpOwotICAgICAgICAgICAgaWYgKGNvbnMuZXF1YWxzKGljKSkgewotICAgICAgICAgICAgICAgIGkucmVtb3ZlKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZXF1ZXN0VG9wRG93bkxlZnRSaWdodFJlc2VuZChJbWFnZUNvbnN1bWVyIGNvbnN1bWVyKSB7Ci0gICAgICAgIC8vIERvIG5vdGhpbmcKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgc3RhcnRQcm9kdWN0aW9uKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgaWYgKGljICE9IG51bGwpIHsKLSAgICAgICAgICAgIGFkZENvbnN1bWVyKGljKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICghbG9hZGluZyAmJiBjb25zdW1lcnMuc2l6ZSgpID4gMCkgewotICAgICAgICAgICAgSW1hZ2VMb2FkZXIuYWRkSW1hZ2VTb3VyY2UodGhpcyk7Ci0gICAgICAgICAgICBsb2FkaW5nID0gdHJ1ZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgYm9vbGVhbiBpc0NvbnN1bWVyKEltYWdlQ29uc3VtZXIgaWMpIHsKLSAgICAgICAgSW1hZ2VEZWNvZGVyIGQgPSBudWxsOwotCi0gICAgICAgIC8vIENoZWNrIGZvciBhbGwgZXhpc3RpbmcgZGVjb2RlcnMKLSAgICAgICAgZm9yIChJdGVyYXRvcjxJbWFnZURlY29kZXI+IGkgPSBkZWNvZGVycy5pdGVyYXRvcigpOyBpLmhhc05leHQoKTspIHsKLSAgICAgICAgICAgIGQgPSBpLm5leHQoKTsKLSAgICAgICAgICAgIGlmIChmaW5kQ29uc3VtZXIoZC5jb25zdW1lcnMsIGljKSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBDaGVjayBjdXJyZW50IHF1ZXVlIG9mIGNvbnN1bWVycwotICAgICAgICByZXR1cm4gZmluZENvbnN1bWVyKGNvbnN1bWVycywgaWMpICE9IG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGlmIHRoZSBjb25zdW1lciBpcyBpbiB0aGUgbGlzdCBhbmQgcmV0dXJucyBpdCBpdCBpcyB0aGVyZQotICAgICAqIEBwYXJhbSBjb25zdW1lcnNMaXN0IC0gbGlzdCBvZiBjb25zdW1lcnMKLSAgICAgKiBAcGFyYW0gaWMgLSBjb25zdW1lcgotICAgICAqIEByZXR1cm4gY29uc3VtZXIgaWYgZm91bmQsIG51bGwgb3RoZXJ3aXNlCi0gICAgICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgSW1hZ2VDb25zdW1lciBmaW5kQ29uc3VtZXIoTGlzdDxJbWFnZUNvbnN1bWVyPiBjb25zdW1lcnNMaXN0LCBJbWFnZUNvbnN1bWVyIGljKSB7Ci0gICAgICAgIEltYWdlQ29uc3VtZXIgcmVzID0gbnVsbDsKLQotICAgICAgICBmb3IgKEl0ZXJhdG9yPEltYWdlQ29uc3VtZXI+IGkgPSBjb25zdW1lcnNMaXN0Lml0ZXJhdG9yKCk7IGkuaGFzTmV4dCgpOykgewotICAgICAgICAgICAgcmVzID0gaS5uZXh0KCk7Ci0gICAgICAgICAgICBpZiAocmVzLmVxdWFscyhpYykpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gcmVzOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVXNlIHRoaXMgbWV0aG9kIHRvIGZpbmlzaCBkZWNvZGluZyBvciBsb2NrIHRoZSBsaXN0IG9mIGNvbnN1bWVycwotICAgICAqIGZvciBhIHBhcnRpY3VsYXIgZGVjb2RlcgotICAgICAqIEBwYXJhbSBkIC0gZGVjb2RlcgotICAgICAqLwotICAgIHN5bmNocm9uaXplZCB2b2lkIGxvY2tEZWNvZGVyKEltYWdlRGVjb2RlciBkKSB7Ci0gICAgICAgIGlmIChkID09IGRlY29kZXIpIHsKLSAgICAgICAgICAgIGRlY29kZXIgPSBudWxsOwotICAgICAgICAgICAgc3RhcnRQcm9kdWN0aW9uKG51bGwpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVHJpZXMgdG8gZmluZCBhbiBhcHByb3ByaWF0ZSBkZWNvZGVyIGZvciB0aGUgaW5wdXQgc3RyZWFtIGFuZCBhZGRzIGl0Ci0gICAgICogdG8gdGhlIGxpc3Qgb2YgZGVjb2RlcnMKLSAgICAgKiBAcmV0dXJuIGNyZWF0ZWQgZGVjb2RlcgotICAgICAqLwotICAgIHByaXZhdGUgSW1hZ2VEZWNvZGVyIGNyZWF0ZURlY29kZXIoKSB7Ci0gICAgICAgIElucHV0U3RyZWFtIGlzID0gZ2V0SW5wdXRTdHJlYW0oKTsKLQotICAgICAgICBJbWFnZURlY29kZXIgZGVjb2RlcjsKLQotICAgICAgICBpZiAoaXMgPT0gbnVsbCkgewotICAgICAgICAgICAgZGVjb2RlciA9IG51bGw7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBkZWNvZGVyID0gSW1hZ2VEZWNvZGVyLmNyZWF0ZURlY29kZXIodGhpcywgaXMpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGRlY29kZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7Ci0gICAgICAgICAgICAgICAgZGVjb2RlcnMuYWRkKGRlY29kZXIpOwotICAgICAgICAgICAgICAgIHRoaXMuZGVjb2RlciA9IGRlY29kZXI7Ci0gICAgICAgICAgICAgICAgbG9hZGluZyA9IGZhbHNlOwotICAgICAgICAgICAgICAgIGNvbnN1bWVycyA9IG5ldyBBcnJheUxpc3Q8SW1hZ2VDb25zdW1lcj4oNSk7IC8vIFJlc2V0IHF1ZXVlCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBkZWNvZGVyOwotICAgICAgICB9Ci0gICAgICAgIC8vIFdlIHdlcmUgbm90IGFibGUgdG8gZmluZCBhcHByb3ByaWF0ZSBkZWNvZGVyCi0gICAgICAgIExpc3Q8SW1hZ2VDb25zdW1lcj4gY3M7Ci0gICAgICAgIHN5bmNocm9uaXplZCAodGhpcykgewotICAgICAgICAgICAgY3MgPSBjb25zdW1lcnM7Ci0gICAgICAgICAgICBjb25zdW1lcnMgPSBuZXcgQXJyYXlMaXN0PEltYWdlQ29uc3VtZXI+KDUpOwotICAgICAgICAgICAgbG9hZGluZyA9IGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIGFib3J0QWxsQ29uc3VtZXJzKGNzKTsKLQotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTdG9wIHRoZSBnaXZlbiBkZWNvZGVyIGFuZCByZW1vdmUgaXQgZnJvbSB0aGUgbGlzdAotICAgICAqIEBwYXJhbSBkciAtIGRlY29kZXIKLSAgICAgKi8KLSAgICBwcml2YXRlIHN5bmNocm9uaXplZCB2b2lkIHJlbW92ZURlY29kZXIoSW1hZ2VEZWNvZGVyIGRyKSB7Ci0gICAgICAgIGxvY2tEZWNvZGVyKGRyKTsKLSAgICAgICAgZGVjb2RlcnMucmVtb3ZlKGRyKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIG1ldGhvZCBzZXJ2ZXMgYXMgYW4gZW50cnkgcG9pbnQuCi0gICAgICogSXQgc3RhcnRzIHRoZSBkZWNvZGVyIGFuZCBsb2FkcyB0aGUgaW1hZ2UgZGF0YS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBsb2FkKCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIGlmIChjb25zdW1lcnMuc2l6ZSgpID09IDApIHsKLSAgICAgICAgICAgICAgICBsb2FkaW5nID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgSW1hZ2VEZWNvZGVyIGQgPSBjcmVhdGVEZWNvZGVyKCk7Ci0gICAgICAgIGlmIChkICE9IG51bGwpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgZGVjb2Rlci5kZWNvZGVJbWFnZSgpOwotICAgICAgICAgICAgfSBjYXRjaCAoSU9FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIGUucHJpbnRTdGFja1RyYWNlKCk7Ci0gICAgICAgICAgICB9IGZpbmFsbHkgewotICAgICAgICAgICAgICAgIHJlbW92ZURlY29kZXIoZCk7Ci0gICAgICAgICAgICAgICAgYWJvcnRBbGxDb25zdW1lcnMoZC5jb25zdW1lcnMpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvRmlsZURlY29kaW5nSW1hZ2VTb3VyY2UuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0ZpbGVEZWNvZGluZ0ltYWdlU291cmNlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDU0ZDQ2NjQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvRmlsZURlY29kaW5nSW1hZ2VTb3VyY2UuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDY4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0vKgotICogQ3JlYXRlZCBvbiAyMC4wMS4yMDA1Ci0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZTsKLQotaW1wb3J0IGphdmEuaW8uQnVmZmVyZWRJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLkZpbGVJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLkZpbGVOb3RGb3VuZEV4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLklucHV0U3RyZWFtOwotCi1wdWJsaWMgY2xhc3MgRmlsZURlY29kaW5nSW1hZ2VTb3VyY2UgZXh0ZW5kcyBEZWNvZGluZ0ltYWdlU291cmNlIHsKLSAgU3RyaW5nIGZpbGVuYW1lOwotCi0gIHB1YmxpYyBGaWxlRGVjb2RpbmdJbWFnZVNvdXJjZShTdHJpbmcgZmlsZSkgewotICAgIFNlY3VyaXR5TWFuYWdlciBzZWN1cml0eSA9IFN5c3RlbS5nZXRTZWN1cml0eU1hbmFnZXIoKTsKLSAgICBpZiAoc2VjdXJpdHkgIT0gbnVsbCkgewotICAgICAgICBzZWN1cml0eS5jaGVja1JlYWQoZmlsZSk7Ci0gICAgfQotCi0gICAgZmlsZW5hbWUgPSBmaWxlOwotICB9Ci0KLSAgQE92ZXJyaWRlCi1wcm90ZWN0ZWQgYm9vbGVhbiBjaGVja0Nvbm5lY3Rpb24oKSB7Ci0gICAgICBTZWN1cml0eU1hbmFnZXIgc2VjdXJpdHkgPSBTeXN0ZW0uZ2V0U2VjdXJpdHlNYW5hZ2VyKCk7Ci0gICAgICBpZiAoc2VjdXJpdHkgIT0gbnVsbCkgewotICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBzZWN1cml0eS5jaGVja1JlYWQoZmlsZW5hbWUpOwotICAgICAgICAgIH0gY2F0Y2ggKFNlY3VyaXR5RXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgIH0KLSAgICAgIH0KLQotICAgICAgcmV0dXJuIHRydWU7Ci0gIH0KLQotICBAT3ZlcnJpZGUKLXByb3RlY3RlZCBJbnB1dFN0cmVhbSBnZXRJbnB1dFN0cmVhbSgpIHsKLSAgICB0cnkgewotICAgICAgLy8gQkVHSU4gYW5kcm9pZC1tb2RpZmllZAotICAgICAgcmV0dXJuIG5ldyBCdWZmZXJlZElucHV0U3RyZWFtKG5ldyBGaWxlSW5wdXRTdHJlYW0oZmlsZW5hbWUpLCA4MTkyKTsKLSAgICAgIC8vIEVORCBhbmRyb2lkLW1vZGlmaWVkCi0gICAgfSBjYXRjaCAoRmlsZU5vdEZvdW5kRXhjZXB0aW9uIGUpIHsKLSAgICAgIHJldHVybiBudWxsOwotICAgIH0KLSAgfQotCi19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9HaWZEZWNvZGVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9HaWZEZWNvZGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDdlY2IxNWIuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvR2lmRGVjb2Rlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNjkyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0vKgotKiBDcmVhdGVkIG9uIDI3LjAxLjIwMDUKLSovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkltYWdlQ29uc3VtZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW5kZXhDb2xvck1vZGVsOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlMaXN0OwotaW1wb3J0IGphdmEudXRpbC5BcnJheXM7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2h0YWJsZTsKLWltcG9ydCBqYXZhLnV0aWwuTGlzdDsKLQotcHVibGljIGNsYXNzIEdpZkRlY29kZXIgZXh0ZW5kcyBJbWFnZURlY29kZXIgewotICAgIC8vIGluaXRpYWxpemVzIHByb3BlciBmaWVsZCBJRHMKLSAgICBwcml2YXRlIHN0YXRpYyBuYXRpdmUgdm9pZCBpbml0SURzKCk7Ci0KLSAgICBzdGF0aWMgewotICAgICAgICBTeXN0ZW0ubG9hZExpYnJhcnkoImdsIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgaW5pdElEcygpOwotICAgIH0KLQotICAgIC8vIEltYWdlQ29uc3VtZXIgaGludHM6IGNvbW1vbgotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBiYXNlSGludHMgPQotICAgICAgICAgICAgSW1hZ2VDb25zdW1lci5TSU5HTEVQQVNTIHwgSW1hZ2VDb25zdW1lci5DT01QTEVURVNDQU5MSU5FUyB8Ci0gICAgICAgICAgICBJbWFnZUNvbnN1bWVyLlNJTkdMRUZSQU1FOwotICAgIC8vIEltYWdlQ29uc3VtZXIgaGludHM6IGludGVybGFjZWQKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgaW50ZXJsYWNlZEhpbnRzID0KLSAgICAgICAgICAgIGJhc2VIaW50cyB8IEltYWdlQ29uc3VtZXIuUkFORE9NUElYRUxPUkRFUjsKLQotICAgIC8vIEltcG9zc2libGUgY29sb3IgdmFsdWUgLSBubyB0cmFuc2x1Y2VudCBwaXhlbHMgYWxsb3dlZAotICAgIHN0YXRpYyBmaW5hbCBpbnQgSU1QT1NTSUJMRV9WQUxVRSA9IDB4MEZGRkZGRkY7Ci0KLSAgICAvLyBJL08gYnVmZmVyCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJVRkZFUl9TSVpFID0gMTAyNDsKLSAgICBwcml2YXRlIGJ5dGUgYnVmZmVyW10gPSBuZXcgYnl0ZVtCVUZGRVJfU0laRV07Ci0KLSAgICBHaWZEYXRhU3RyZWFtIGdpZkRhdGFTdHJlYW0gPSBuZXcgR2lmRGF0YVN0cmVhbSgpOwotICAgIEdpZkdyYXBoaWNCbG9jayBjdXJyQmxvY2s7Ci0KLSAgICAvLyBQb2ludGVyIHRvIG5hdGl2ZSBzdHJ1Y3R1cmUgd2hpY2ggc3RvcmUgZGVjb2Rpbmcgc3RhdGUKLSAgICAvLyBiZXR3ZWVuIHN1YnNlcXVlbnQgZGVjb2RpbmcvSU8tc3VzcGVuc2lvbiBjeWNsZXMKLSAgICBwcml2YXRlIGxvbmcgaE5hdGl2ZURlY29kZXI7IC8vIE5VTEwgaW5pdGlhbGx5Ci0KLSAgICAvLyBOdW1iZXIgb2YgYnl0ZXMgZWF0ZW4gYnkgdGhlIG5hdGl2ZSBkZWNvZGVyCi0gICAgcHJpdmF0ZSBpbnQgYnl0ZXNDb25zdW1lZDsKLQotICAgIHByaXZhdGUgYm9vbGVhbiBjb25zdW1lcnNQcmVwYXJlZDsKLSAgICBwcml2YXRlIEhhc2h0YWJsZTxTdHJpbmcsIFN0cmluZz4gcHJvcGVydGllcyA9IG5ldyBIYXNodGFibGU8U3RyaW5nLCBTdHJpbmc+KCk7Ci0KLSAgICAvLyBDb3VsZCBiZSBzZXQgdXAgYnkgamF2YSBjb2RlIG9yIG5hdGl2ZSBtZXRob2Qgd2hlbgotICAgIC8vIHRyYW5zcGFyZW50IHBpeGVsIGluZGV4IGNoYW5nZXMgb3IgbG9jYWwgY29sb3IgdGFibGUgZW5jb3VudGVyZWQKLSAgICBwcml2YXRlIGJvb2xlYW4gZm9yY2VSR0I7Ci0KLSAgICBwcml2YXRlIGJ5dGUgc2NyZWVuQnVmZmVyW107Ci0gICAgcHJpdmF0ZSBpbnQgc2NyZWVuUkdCQnVmZmVyW107Ci0KLSAgICBwdWJsaWMgR2lmRGVjb2RlcihEZWNvZGluZ0ltYWdlU291cmNlIHNyYywgSW5wdXRTdHJlYW0gaXMpIHsKLSAgICAgICAgc3VwZXIoc3JjLCBpcyk7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBzdGF0aWMgbmF0aXZlIGludFtdIHRvUkdCKGJ5dGUgaW1hZ2VEYXRhW10sIGJ5dGUgY29sb3JtYXBbXSwgaW50IHRyYW5zcGFyZW50Q29sb3IpOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgbmF0aXZlIHZvaWQgcmVsZWFzZU5hdGl2ZURlY29kZXIobG9uZyBoRGVjb2Rlcik7Ci0KLSAgICBwcml2YXRlIG5hdGl2ZSBpbnQgZGVjb2RlKAotICAgICAgICAgICAgYnl0ZSBpbnB1dFtdLAotICAgICAgICAgICAgaW50IGJ5dGVzSW5CdWZmZXIsCi0gICAgICAgICAgICBsb25nIGhEZWNvZGVyLAotICAgICAgICAgICAgR2lmRGF0YVN0cmVhbSBkYXRhU3RyZWFtLAotICAgICAgICAgICAgR2lmR3JhcGhpY0Jsb2NrIGN1cnJCbG9jawotICAgICAgICAgICAgKTsKLQotICAgIHByaXZhdGUgaW50W10gZ2V0U2NyZWVuUkdCQnVmZmVyKCkgewotICAgICAgICBpZiAoc2NyZWVuUkdCQnVmZmVyID09IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChzY3JlZW5CdWZmZXIgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIGludCB0cmFuc3BhcmVudENvbG9yID0KLSAgICAgICAgICAgICAgICAgICAgICAgIGdpZkRhdGFTdHJlYW0ubG9naWNhbFNjcmVlbi5nbG9iYWxDb2xvclRhYmxlLmNtLmdldFRyYW5zcGFyZW50UGl4ZWwoKTsKLSAgICAgICAgICAgICAgICB0cmFuc3BhcmVudENvbG9yID0gdHJhbnNwYXJlbnRDb2xvciA+IDAgPyB0cmFuc3BhcmVudENvbG9yIDogSU1QT1NTSUJMRV9WQUxVRTsKLSAgICAgICAgICAgICAgICBzY3JlZW5SR0JCdWZmZXIgPQotICAgICAgICAgICAgICAgICAgICAgICAgdG9SR0IoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNjcmVlbkJ1ZmZlciwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2lmRGF0YVN0cmVhbS5sb2dpY2FsU2NyZWVuLmdsb2JhbENvbG9yVGFibGUuY29sb3JzLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0cmFuc3BhcmVudENvbG9yCi0gICAgICAgICAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBpbnQgc2l6ZSA9IGdpZkRhdGFTdHJlYW0ubG9naWNhbFNjcmVlbi5sb2dpY2FsU2NyZWVuSGVpZ2h0ICoKLSAgICAgICAgICAgICAgICAgICAgICAgIGdpZkRhdGFTdHJlYW0ubG9naWNhbFNjcmVlbi5sb2dpY2FsU2NyZWVuV2lkdGg7Ci0gICAgICAgICAgICAgICAgc2NyZWVuUkdCQnVmZmVyID0gbmV3IGludFtzaXplXTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBzY3JlZW5SR0JCdWZmZXI7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIHByZXBhcmVDb25zdW1lcnMoKSB7Ci0gICAgICAgIEdpZkxvZ2ljYWxTY3JlZW4gZ2xzID0gZ2lmRGF0YVN0cmVhbS5sb2dpY2FsU2NyZWVuOwotICAgICAgICBzZXREaW1lbnNpb25zKGdscy5sb2dpY2FsU2NyZWVuV2lkdGgsCi0gICAgICAgICAgICAgICAgZ2xzLmxvZ2ljYWxTY3JlZW5IZWlnaHQpOwotICAgICAgICBzZXRQcm9wZXJ0aWVzKHByb3BlcnRpZXMpOwotCi0gICAgICAgIGN1cnJCbG9jayA9IGdpZkRhdGFTdHJlYW0uZ3JhcGhpY0Jsb2Nrcy5nZXQoMCk7Ci0gICAgICAgIGlmIChmb3JjZVJHQikgewotICAgICAgICAgICAgc2V0Q29sb3JNb2RlbChDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBzZXRDb2xvck1vZGVsKGdscy5nbG9iYWxDb2xvclRhYmxlLmdldENvbG9yTW9kZWwoY3VyckJsb2NrLnRyYW5zcGFyZW50Q29sb3IpKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIEZpbGwgc2NyZWVuIGJ1ZmZlciB3aXRoIHRoZSBiYWNrZ3JvdW5kIG9yIHRyYW5zcGFyZW50IGNvbG9yCi0gICAgICAgIGlmIChmb3JjZVJHQikgewotICAgICAgICAgICAgaW50IGZpbGxDb2xvciA9IDB4RkYwMDAwMDA7Ci0gICAgICAgICAgICBpZiAoZ2xzLmJhY2tncm91bmRDb2xvciAhPSBJTVBPU1NJQkxFX1ZBTFVFKSB7Ci0gICAgICAgICAgICAgICAgZmlsbENvbG9yID0gZ2xzLmJhY2tncm91bmRDb2xvcjsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgQXJyYXlzLmZpbGwoZ2V0U2NyZWVuUkdCQnVmZmVyKCksIGZpbGxDb2xvcik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpbnQgZmlsbENvbG9yID0gMDsKLQotICAgICAgICAgICAgaWYgKGdscy5iYWNrZ3JvdW5kQ29sb3IgIT0gSU1QT1NTSUJMRV9WQUxVRSkgewotICAgICAgICAgICAgICAgIGZpbGxDb2xvciA9IGdscy5iYWNrZ3JvdW5kQ29sb3I7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGZpbGxDb2xvciA9IGdscy5nbG9iYWxDb2xvclRhYmxlLmNtLmdldFRyYW5zcGFyZW50UGl4ZWwoKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgc2NyZWVuQnVmZmVyID0gbmV3IGJ5dGVbZ2xzLmxvZ2ljYWxTY3JlZW5IZWlnaHQqZ2xzLmxvZ2ljYWxTY3JlZW5XaWR0aF07Ci0gICAgICAgICAgICBBcnJheXMuZmlsbChzY3JlZW5CdWZmZXIsIChieXRlKSBmaWxsQ29sb3IpOwotICAgICAgICB9Ci0KLSAgICAgICAgc2V0SGludHMoaW50ZXJsYWNlZEhpbnRzKTsgLy8gWFhYIC0gYWx3YXlzIHJhbmRvbSBwaXhlbCBvcmRlcgotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRlY29kZUltYWdlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGludCBieXRlc1JlYWQgPSAwOwotICAgICAgICAgICAgaW50IG5lZWRCeXRlcywgb2Zmc2V0LCBieXRlc0luQnVmZmVyID0gMDsKLSAgICAgICAgICAgIGJvb2xlYW4gZW9zUmVhY2hlZCA9IGZhbHNlOwotICAgICAgICAgICAgR2lmR3JhcGhpY0Jsb2NrIGJsb2NrVG9EaXNwb3NlID0gbnVsbDsKLQotICAgICAgICAgICAgLy8gQ3JlYXRlIG5ldyBncmFwaGljIGJsb2NrCi0gICAgICAgICAgICBpZiAoY3VyckJsb2NrID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBjdXJyQmxvY2sgPSBuZXcgR2lmR3JhcGhpY0Jsb2NrKCk7Ci0gICAgICAgICAgICAgICAgZ2lmRGF0YVN0cmVhbS5ncmFwaGljQmxvY2tzLmFkZChjdXJyQmxvY2spOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBSZWFkIGZyb20gdGhlIGlucHV0IHN0cmVhbQotICAgICAgICAgICAgZm9yICg7OykgewotICAgICAgICAgICAgICAgIG5lZWRCeXRlcyA9IEJVRkZFUl9TSVpFIC0gYnl0ZXNJbkJ1ZmZlcjsKLSAgICAgICAgICAgICAgICBvZmZzZXQgPSBieXRlc0luQnVmZmVyOwotCi0gICAgICAgICAgICAgICAgYnl0ZXNSZWFkID0gaW5wdXRTdHJlYW0ucmVhZChidWZmZXIsIG9mZnNldCwgbmVlZEJ5dGVzKTsKLQotICAgICAgICAgICAgICAgIGlmIChieXRlc1JlYWQgPCAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGVvc1JlYWNoZWQgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICBieXRlc1JlYWQgPSAwOwotICAgICAgICAgICAgICAgIH0gLy8gRG9uJ3QgYnJlYWssIG1heWJlIHNvbWV0aGluZyBsZWZ0IGluIGJ1ZmZlcgotCi0gICAgICAgICAgICAgICAgLy8gS2VlcCB0cmFjayBvbiBob3cgbXVjaCBieXRlcyBsZWZ0IGluIGJ1ZmZlcgotICAgICAgICAgICAgICAgIGJ5dGVzSW5CdWZmZXIgKz0gYnl0ZXNSZWFkOwotCi0gICAgICAgICAgICAgICAgLy8gSGVyZSB3ZSBwYXNzIG51bWJlciBvZiBuZXcgYnl0ZXMgcmVhZCBmcm9tIHRoZSBpbnB1dCBzdHJlYW0gKGJ5dGVzUmVhZCkKLSAgICAgICAgICAgICAgICAvLyBzaW5jZSBuYXRpdmUgZGVjb2RlciB1c2VzIGphdmEgYnVmZmVyIGFuZCBkb2Vzbid0IGhhdmUgaXRzIG93bgotICAgICAgICAgICAgICAgIC8vIGJ1ZmZlci4gU28gaXQgYWRkcyB0aGlzIG51bWJlciB0byB0aGUgbnVtYmVyIG9mIGJ5dGVzIGxlZnQKLSAgICAgICAgICAgICAgICAvLyBpbiBidWZmZXIgZnJvbSB0aGUgcHJldmlvdXMgY2FsbC4KLSAgICAgICAgICAgICAgICBpbnQgbnVtTGluZXMgPSBkZWNvZGUoCi0gICAgICAgICAgICAgICAgICAgICAgICBidWZmZXIsCi0gICAgICAgICAgICAgICAgICAgICAgICBieXRlc1JlYWQsCi0gICAgICAgICAgICAgICAgICAgICAgICBoTmF0aXZlRGVjb2RlciwKLSAgICAgICAgICAgICAgICAgICAgICAgIGdpZkRhdGFTdHJlYW0sCi0gICAgICAgICAgICAgICAgICAgICAgICBjdXJyQmxvY2spOwotCi0gICAgICAgICAgICAgICAgLy8gS2VlcCB0cmFjayBvbiBob3cgbXVjaCBieXRlcyBsZWZ0IGluIGJ1ZmZlcgotICAgICAgICAgICAgICAgIGJ5dGVzSW5CdWZmZXIgLT0gYnl0ZXNDb25zdW1lZDsKLQotICAgICAgICAgICAgICAgIGlmICgKLSAgICAgICAgICAgICAgICAgICAgICAgICFjb25zdW1lcnNQcmVwYXJlZCAmJgotICAgICAgICAgICAgICAgICAgICAgICAgZ2lmRGF0YVN0cmVhbS5sb2dpY2FsU2NyZWVuLmNvbXBsZXRlZCAmJgotICAgICAgICAgICAgICAgICAgICAgICAgZ2lmRGF0YVN0cmVhbS5sb2dpY2FsU2NyZWVuLmdsb2JhbENvbG9yVGFibGUuY29tcGxldGVkICYmCi0gICAgICAgICAgICAgICAgICAgICAgICAoY3VyckJsb2NrLmltYWdlRGF0YSAhPSBudWxsIHx8IC8vIEhhdmUgdHJhbnNwYXJlbnQgcGl4ZWwgZmlsbGVkCi0gICAgICAgICAgICAgICAgICAgICAgICBjdXJyQmxvY2sucmdiSW1hZ2VEYXRhICE9IG51bGwpCi0gICAgICAgICAgICAgICAgKSB7Ci0gICAgICAgICAgICAgICAgICAgIHByZXBhcmVDb25zdW1lcnMoKTsKLSAgICAgICAgICAgICAgICAgICAgY29uc3VtZXJzUHJlcGFyZWQgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChieXRlc0NvbnN1bWVkIDwgMCkgewotICAgICAgICAgICAgICAgICAgICBicmVhazsgLy8gRXJyb3IgZXhpdAotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChjdXJyQmxvY2sgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICBpZiAobnVtTGluZXMgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gRGlzcG9zZSBwcmV2aW91cyBpbWFnZSBvbmx5IGJlZm9yZSBzaG93aW5nIG5leHQKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChibG9ja1RvRGlzcG9zZSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYmxvY2tUb0Rpc3Bvc2UuZGlzcG9zZSgpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJsb2NrVG9EaXNwb3NlID0gbnVsbDsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgICAgICAgICAgY3VyckJsb2NrLnNlbmROZXdEYXRhKHRoaXMsIG51bUxpbmVzKTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIGlmIChjdXJyQmxvY2suY29tcGxldGVkICYmIGhOYXRpdmVEZWNvZGVyICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJsb2NrVG9EaXNwb3NlID0gY3VyckJsb2NrOyAvLyBEaXNwb3NlIG9ubHkgYmVmb3JlIHNob3dpbmcgbmV3IHBpeGVscwotICAgICAgICAgICAgICAgICAgICAgICAgY3VyckJsb2NrID0gbmV3IEdpZkdyYXBoaWNCbG9jaygpOwotICAgICAgICAgICAgICAgICAgICAgICAgZ2lmRGF0YVN0cmVhbS5ncmFwaGljQmxvY2tzLmFkZChjdXJyQmxvY2spOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgaWYgKGhOYXRpdmVEZWNvZGVyID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgaWYgKGVvc1JlYWNoZWQgJiYgbnVtTGluZXMgPT0gMCkgeyAvLyBNYXliZSBpbWFnZSBpcyB0cnVuY2F0ZWQuLi4KLSAgICAgICAgICAgICAgICAgICAgcmVsZWFzZU5hdGl2ZURlY29kZXIoaE5hdGl2ZURlY29kZXIpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICBjbG9zZVN0cmVhbSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gSGVyZSBhbGwgYW5pbWF0aW9uIGdvZXMKLSAgICAgICAgLy8gUmVwZWF0IGltYWdlIGxvb3BDb3VudC0xIHRpbWVzIG9yIGluZmluaXRlbHkgaWYgbG9vcENvdW50ID0gMAotICAgICAgICBpZiAoZ2lmRGF0YVN0cmVhbS5sb29wQ291bnQgIT0gMSkgewotICAgICAgICAgICAgaWYgKGN1cnJCbG9jay5jb21wbGV0ZWQgPT0gZmFsc2UpIHsKLSAgICAgICAgICAgICAgICBnaWZEYXRhU3RyZWFtLmdyYXBoaWNCbG9ja3MucmVtb3ZlKGN1cnJCbG9jayk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGludCBudW1GcmFtZXMgPSBnaWZEYXRhU3RyZWFtLmdyYXBoaWNCbG9ja3Muc2l6ZSgpOwotICAgICAgICAgICAgLy8gQXQgZmlyc3QgbGFzdCBibG9jayB3aWxsIGJlIGRpc3Bvc2VkCi0gICAgICAgICAgICBHaWZHcmFwaGljQmxvY2sgZ2IgPQotICAgICAgICAgICAgICAgICAgICBnaWZEYXRhU3RyZWFtLmdyYXBoaWNCbG9ja3MuZ2V0KG51bUZyYW1lcy0xKTsKLQotICAgICAgICAgICAgSW1hZ2VMb2FkZXIuYmVnaW5BbmltYXRpb24oKTsKLQotICAgICAgICAgICAgd2hpbGUgKGdpZkRhdGFTdHJlYW0ubG9vcENvdW50ICE9IDEpIHsKLSAgICAgICAgICAgICAgICBpZiAoZ2lmRGF0YVN0cmVhbS5sb29wQ291bnQgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICBnaWZEYXRhU3RyZWFtLmxvb3BDb3VudC0tOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIC8vIFNob3cgYWxsIGZyYW1lcwotICAgICAgICAgICAgICAgIGZvciAoaW50IGk9MDsgaTxudW1GcmFtZXM7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBnYi5kaXNwb3NlKCk7Ci0gICAgICAgICAgICAgICAgICAgIGdiID0gZ2lmRGF0YVN0cmVhbS5ncmFwaGljQmxvY2tzLmdldChpKTsKLQotICAgICAgICAgICAgICAgICAgICAvLyBTaG93IG9uZSBmcmFtZQotICAgICAgICAgICAgICAgICAgICBpZiAoZm9yY2VSR0IpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHNldFBpeGVscygKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2IuaW1hZ2VMZWZ0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnYi5pbWFnZVRvcCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2IuaW1hZ2VXaWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2IuaW1hZ2VIZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnYi5nZXRSZ2JJbWFnZURhdGEoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2IuaW1hZ2VXaWR0aAotICAgICAgICAgICAgICAgICAgICAgICAgKTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHNldFBpeGVscygKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2IuaW1hZ2VMZWZ0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnYi5pbWFnZVRvcCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2IuaW1hZ2VXaWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2IuaW1hZ2VIZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG51bGwsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdiLmltYWdlRGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2IuaW1hZ2VXaWR0aAotICAgICAgICAgICAgICAgICAgICAgICAgKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIEltYWdlTG9hZGVyLmVuZEFuaW1hdGlvbigpOwotICAgICAgICB9Ci0KLSAgICAgICAgaW1hZ2VDb21wbGV0ZShJbWFnZUNvbnN1bWVyLlNUQVRJQ0lNQUdFRE9ORSk7Ci0gICAgfQotCi0gICAgdm9pZCBzZXRDb21tZW50KFN0cmluZyBuZXdDb21tZW50KSB7Ci0gICAgICAgIE9iamVjdCBjdXJyQ29tbWVudCA9IHByb3BlcnRpZXMuZ2V0KCJjb21tZW50Iik7IC8vJE5PTi1OTFMtMSQKLQotICAgICAgICBpZiAoY3VyckNvbW1lbnQgPT0gbnVsbCkgewotICAgICAgICAgICAgcHJvcGVydGllcy5wdXQoImNvbW1lbnQiLCBuZXdDb21tZW50KTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcHJvcGVydGllcy5wdXQoImNvbW1lbnQiLCAoU3RyaW5nKSBjdXJyQ29tbWVudCArICJcbiIgKyBuZXdDb21tZW50KTsgLy8kTk9OLU5MUy0xJCAvLyROT04tTkxTLTIkCi0gICAgICAgIH0KLQotICAgICAgICBzZXRQcm9wZXJ0aWVzKHByb3BlcnRpZXMpOwotICAgIH0KLQotICAgIGNsYXNzIEdpZkRhdGFTdHJlYW0gewotICAgICAgICAvLyAgSW5kaWNhdGVzIHRoYXQgcmVhZGluZyBvZiB0aGUgd2hvbGUgZGF0YSBzdHJlYW0gYWNjb21wbGlzaGVkCi0gICAgICAgIGJvb2xlYW4gY29tcGxldGVkID0gZmFsc2U7Ci0KLSAgICAgICAgLy8gQWRkZWQgdG8gc3VwcG9ydCBOZXRzY2FwZSAyLjAgYXBwbGljYXRpb24KLSAgICAgICAgLy8gZXh0ZW5zaW9uIGJsb2NrLgotICAgICAgICBpbnQgbG9vcENvdW50ID0gMTsKLQotICAgICAgICBHaWZMb2dpY2FsU2NyZWVuIGxvZ2ljYWxTY3JlZW4gPSBuZXcgR2lmTG9naWNhbFNjcmVlbigpOwotICAgICAgICBMaXN0PEdpZkdyYXBoaWNCbG9jaz4gZ3JhcGhpY0Jsb2NrcyA9IG5ldyBBcnJheUxpc3Q8R2lmR3JhcGhpY0Jsb2NrPigxMCk7IC8vIE9mIEdpZkdyYXBoaWNCbG9ja3MKLQotICAgICAgICAvLyBDb21tZW50cyBmcm9tIHRoZSBpbWFnZQotICAgICAgICBTdHJpbmcgY29tbWVudHNbXTsKLSAgICB9Ci0KLSAgICBjbGFzcyBHaWZMb2dpY2FsU2NyZWVuIHsKLSAgICAgICAgLy8gIEluZGljYXRlcyB0aGF0IHJlYWRpbmcgb2YgdGhpcyBibG9jayBhY2NvbXBsaXNoZWQKLSAgICAgICAgYm9vbGVhbiBjb21wbGV0ZWQgPSBmYWxzZTsKLQotICAgICAgICBpbnQgbG9naWNhbFNjcmVlbldpZHRoOwotICAgICAgICBpbnQgbG9naWNhbFNjcmVlbkhlaWdodDsKLQotICAgICAgICBpbnQgYmFja2dyb3VuZENvbG9yID0gSU1QT1NTSUJMRV9WQUxVRTsKLQotICAgICAgICBHaWZDb2xvclRhYmxlIGdsb2JhbENvbG9yVGFibGUgPSBuZXcgR2lmQ29sb3JUYWJsZSgpOwotICAgIH0KLQotICAgIGNsYXNzIEdpZkdyYXBoaWNCbG9jayB7Ci0gICAgICAgIC8vICBJbmRpY2F0ZXMgdGhhdCByZWFkaW5nIG9mIHRoaXMgYmxvY2sgYWNjb21wbGlzaGVkCi0gICAgICAgIGJvb2xlYW4gY29tcGxldGVkID0gZmFsc2U7Ci0KLSAgICAgICAgZmluYWwgc3RhdGljIGludCBESVNQT1NBTF9OT05FID0gMDsKLSAgICAgICAgZmluYWwgc3RhdGljIGludCBESVNQT1NBTF9OT0RJU1BPU0FMID0gMTsKLSAgICAgICAgZmluYWwgc3RhdGljIGludCBESVNQT1NBTF9CQUNLR1JPVU5EID0gMjsKLSAgICAgICAgZmluYWwgc3RhdGljIGludCBESVNQT1NBTF9SRVNUT1JFID0gMzsKLQotICAgICAgICBpbnQgZGlzcG9zYWxNZXRob2Q7Ci0gICAgICAgIGludCBkZWxheVRpbWU7IC8vIE11bHRpcGxpZWQgYnkgMTAgYWxyZWFkeQotICAgICAgICBpbnQgdHJhbnNwYXJlbnRDb2xvciA9IElNUE9TU0lCTEVfVkFMVUU7Ci0KLSAgICAgICAgaW50IGltYWdlTGVmdDsKLSAgICAgICAgaW50IGltYWdlVG9wOwotICAgICAgICBpbnQgaW1hZ2VXaWR0aDsKLSAgICAgICAgaW50IGltYWdlSGVpZ2h0OwotCi0gICAgICAgIC8vIEF1eGlsbGlhcnkgdmFyaWFibGVzIHRvIG1pbmltaXplIGNvbXB1dGF0aW9ucwotICAgICAgICBpbnQgaW1hZ2VSaWdodDsKLSAgICAgICAgaW50IGltYWdlQm90dG9tOwotCi0gICAgICAgIGJvb2xlYW4gaW50ZXJsYWNlOwotCi0gICAgICAgIC8vIERvbid0IG5lZWQgbG9jYWwgY29sb3IgdGFibGUgLSBpZiBpdCBpcyBzcGVjaWZpZWQKLSAgICAgICAgLy8gaW1hZ2UgZGF0YSBhcmUgY29udmVydGVkIHRvIFJHQiBpbiB0aGUgbmF0aXZlIGNvZGUKLQotICAgICAgICBieXRlIGltYWdlRGF0YVtdID0gbnVsbDsKLSAgICAgICAgaW50IHJnYkltYWdlRGF0YVtdID0gbnVsbDsKLQotICAgICAgICBwcml2YXRlIGludCBjdXJyWSA9IDA7IC8vIEN1cnJlbnQgb3V0cHV0IHNjYW5saW5lCi0KLSAgICAgICAgaW50W10gZ2V0UmdiSW1hZ2VEYXRhKCkgewotICAgICAgICAgICAgaWYgKHJnYkltYWdlRGF0YSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgcmdiSW1hZ2VEYXRhID0KLSAgICAgICAgICAgICAgICAgICAgICAgIHRvUkdCKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbWFnZURhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdpZkRhdGFTdHJlYW0ubG9naWNhbFNjcmVlbi5nbG9iYWxDb2xvclRhYmxlLmNvbG9ycywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdHJhbnNwYXJlbnRDb2xvcgotICAgICAgICAgICAgICAgICAgICAgICAgKTsKLSAgICAgICAgICAgICAgICBpZiAodHJhbnNwYXJlbnRDb2xvciAhPSBJTVBPU1NJQkxFX1ZBTFVFKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRyYW5zcGFyZW50Q29sb3IgPQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdpZkRhdGFTdHJlYW0ubG9naWNhbFNjcmVlbi5nbG9iYWxDb2xvclRhYmxlLmNtLmdldFJHQih0cmFuc3BhcmVudENvbG9yKTsKLSAgICAgICAgICAgICAgICAgICAgdHJhbnNwYXJlbnRDb2xvciAmPSAweDAwRkZGRkZGOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiByZ2JJbWFnZURhdGE7Ci0gICAgICAgIH0KLQotICAgICAgICBwcml2YXRlIHZvaWQgcmVwbGFjZVRyYW5zcGFyZW50UGl4ZWxzKGludCBudW1MaW5lcykgewotICAgICAgICAgICAgTGlzdDxHaWZHcmFwaGljQmxvY2s+IGdyYXBoaWNCbG9ja3MgPSBnaWZEYXRhU3RyZWFtLmdyYXBoaWNCbG9ja3M7Ci0gICAgICAgICAgICBpbnQgcHJldkJsb2NrSW5kZXggPSBncmFwaGljQmxvY2tzLmluZGV4T2YodGhpcykgLSAxOwotCi0gICAgICAgICAgICBpZiAocHJldkJsb2NrSW5kZXggPj0gMCkgewotICAgICAgICAgICAgICAgIGludCBtYXhZID0gY3VyclkgKyBudW1MaW5lcyArIGltYWdlVG9wOwotICAgICAgICAgICAgICAgIGludCBvZmZzZXQgPSBjdXJyWSAqIGltYWdlV2lkdGg7Ci0KLSAgICAgICAgICAgICAgICAvLyBVcGRhdGUgcmlnaHQgYW5kIGJvdHRvbSBjb29yZGluYXRlcwotICAgICAgICAgICAgICAgIGltYWdlUmlnaHQgPSBpbWFnZUxlZnQgKyBpbWFnZVdpZHRoOwotICAgICAgICAgICAgICAgIGltYWdlQm90dG9tID0gaW1hZ2VUb3AgKyBpbWFnZUhlaWdodDsKLQotICAgICAgICAgICAgICAgIGludCBnbG9iYWxXaWR0aCA9IGdpZkRhdGFTdHJlYW0ubG9naWNhbFNjcmVlbi5sb2dpY2FsU2NyZWVuV2lkdGg7Ci0gICAgICAgICAgICAgICAgaW50IHBpeGVsVmFsdWUsIGltYWdlT2Zmc2V0OwotICAgICAgICAgICAgICAgIGludCByZ2JEYXRhW10gPSBmb3JjZVJHQiA/IGdldFJnYkltYWdlRGF0YSgpIDogbnVsbDsKLQotICAgICAgICAgICAgICAgIGZvciAoaW50IHkgPSBjdXJyWSArIGltYWdlVG9wOyB5IDwgbWF4WTsgeSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGltYWdlT2Zmc2V0ID0gZ2xvYmFsV2lkdGggKiB5ICsgaW1hZ2VMZWZ0OwotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCB4ID0gaW1hZ2VMZWZ0OyB4IDwgaW1hZ2VSaWdodDsgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwaXhlbFZhbHVlID0gZm9yY2VSR0IgPwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZ2JEYXRhW29mZnNldF0gOgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbWFnZURhdGFbb2Zmc2V0XSAmIDB4RkY7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAocGl4ZWxWYWx1ZSA9PSB0cmFuc3BhcmVudENvbG9yKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGZvcmNlUkdCKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBpeGVsVmFsdWUgPSBnZXRTY3JlZW5SR0JCdWZmZXIoKSBbaW1hZ2VPZmZzZXRdOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZ2JEYXRhW29mZnNldF0gPSBwaXhlbFZhbHVlOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBpeGVsVmFsdWUgPSBzY3JlZW5CdWZmZXIgW2ltYWdlT2Zmc2V0XTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VEYXRhW29mZnNldF0gPSAoYnl0ZSkgcGl4ZWxWYWx1ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXQrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlT2Zmc2V0Kys7Ci0gICAgICAgICAgICAgICAgICAgIH0gLy8gZm9yCi0gICAgICAgICAgICAgICAgfSAvLyBmb3IKLQotICAgICAgICAgICAgfSAvLyBpZiAocHJldkJsb2NrSW5kZXggPj0gMCkKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyB2b2lkIHNlbmROZXdEYXRhKEdpZkRlY29kZXIgZGVjb2RlciwgaW50IG51bUxpbmVzKSB7Ci0gICAgICAgICAgICAvLyBHZXQgdmFsdWVzIGZvciB0cmFuc3BhcmVudCBwaXhlbHMKLSAgICAgICAgICAgIC8vIGZyb20gdGhlIHBlcmV2aW91cyBmcmFtZXMKLSAgICAgICAgICAgIGlmICh0cmFuc3BhcmVudENvbG9yICE9IElNUE9TU0lCTEVfVkFMVUUpIHsKLSAgICAgICAgICAgICAgICByZXBsYWNlVHJhbnNwYXJlbnRQaXhlbHMobnVtTGluZXMpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoZm9yY2VSR0IpIHsKLSAgICAgICAgICAgICAgICBkZWNvZGVyLnNldFBpeGVscygKLSAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlTGVmdCwKLSAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlVG9wICsgY3VyclksCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZVdpZHRoLAotICAgICAgICAgICAgICAgICAgICAgICAgbnVtTGluZXMsCi0gICAgICAgICAgICAgICAgICAgICAgICBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGdldFJnYkltYWdlRGF0YSgpLAotICAgICAgICAgICAgICAgICAgICAgICAgY3VyclkqaW1hZ2VXaWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlV2lkdGgKLSAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBkZWNvZGVyLnNldFBpeGVscygKLSAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlTGVmdCwKLSAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlVG9wICsgY3VyclksCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZVdpZHRoLAotICAgICAgICAgICAgICAgICAgICAgICAgbnVtTGluZXMsCi0gICAgICAgICAgICAgICAgICAgICAgICBudWxsLAotICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VEYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgY3VyclkqaW1hZ2VXaWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlV2lkdGgKLSAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjdXJyWSArPSBudW1MaW5lczsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyB2b2lkIGRpc3Bvc2UoKSB7Ci0gICAgICAgICAgICBpbWFnZUNvbXBsZXRlKEltYWdlQ29uc3VtZXIuU0lOR0xFRlJBTUVET05FKTsKLQotICAgICAgICAgICAgLy8gU2hvdyBjdXJyZW50IGZyYW1lIHVudGlsIGRlbGF5SW50ZXJ2YWwgd2lsbCBub3QgZWxhcHNlCi0gICAgICAgICAgICBpZiAoZGVsYXlUaW1lID4gMCkgewotICAgICAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgICAgIFRocmVhZC5zbGVlcChkZWxheVRpbWUpOwotICAgICAgICAgICAgICAgIH0gY2F0Y2ggKEludGVycnVwdGVkRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICAgICAgZS5wcmludFN0YWNrVHJhY2UoKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIFRocmVhZC55aWVsZCgpOyAvLyBBbGxvdyBjb25zdW1lcnMgdG8gY29uc3VtZSBkYXRhCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIERvbid0IGRpc3Bvc2UgaWYgaW1hZ2UgaXMgb3V0c2lkZSBvZiB0aGUgdmlzaWJsZSBhcmVhCi0gICAgICAgICAgICBpZiAoaW1hZ2VMZWZ0ID4gZ2lmRGF0YVN0cmVhbS5sb2dpY2FsU2NyZWVuLmxvZ2ljYWxTY3JlZW5XaWR0aCB8fAotICAgICAgICAgICAgICAgICAgICBpbWFnZVRvcCA+IGdpZkRhdGFTdHJlYW0ubG9naWNhbFNjcmVlbi5sb2dpY2FsU2NyZWVuSGVpZ2h0KSB7Ci0gICAgICAgICAgICAgICAgZGlzcG9zYWxNZXRob2QgPSBESVNQT1NBTF9OT05FOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBzd2l0Y2goZGlzcG9zYWxNZXRob2QpIHsKLSAgICAgICAgICAgICAgICBjYXNlIERJU1BPU0FMX0JBQ0tHUk9VTkQ6IHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGZvcmNlUkdCKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBnZXRSZ2JJbWFnZURhdGEoKTsgLy8gRW5zdXJlIHRoYXQgdHJhbnNwYXJlbnRDb2xvciBpcyBSR0IsIG5vdCBpbmRleAotCi0gICAgICAgICAgICAgICAgICAgICAgICBpbnQgZGF0YVtdID0gbmV3IGludFtpbWFnZVdpZHRoKmltYWdlSGVpZ2h0XTsKLQotICAgICAgICAgICAgICAgICAgICAgICAgLy8gQ29tcGF0aWJpbGl0eTogRmlsbCB3aXRoIHRyYW5zcGFyZW50IGNvbG9yIGlmIHdlIGhhdmUgb25lCi0gICAgICAgICAgICAgICAgICAgICAgICBpZiAodHJhbnNwYXJlbnRDb2xvciAhPSBJTVBPU1NJQkxFX1ZBTFVFKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgQXJyYXlzLmZpbGwoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdHJhbnNwYXJlbnRDb2xvcgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEFycmF5cy5maWxsKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdpZkRhdGFTdHJlYW0ubG9naWNhbFNjcmVlbi5iYWNrZ3JvdW5kQ29sb3IKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgICAgICBzZXRQaXhlbHMoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlTGVmdCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VUb3AsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlV2lkdGgsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlSGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VXaWR0aAotICAgICAgICAgICAgICAgICAgICAgICAgKTsKLQotICAgICAgICAgICAgICAgICAgICAgICAgc2VuZFRvU2NyZWVuQnVmZmVyKGRhdGEpOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgYnl0ZSBkYXRhW10gPSBuZXcgYnl0ZVtpbWFnZVdpZHRoKmltYWdlSGVpZ2h0XTsKLQotICAgICAgICAgICAgICAgICAgICAgICAgLy8gQ29tcGF0aWJpbGl0eTogRmlsbCB3aXRoIHRyYW5zcGFyZW50IGNvbG9yIGlmIHdlIGhhdmUgb25lCi0gICAgICAgICAgICAgICAgICAgICAgICBpZiAodHJhbnNwYXJlbnRDb2xvciAhPSBJTVBPU1NJQkxFX1ZBTFVFKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgQXJyYXlzLmZpbGwoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGJ5dGUpIHRyYW5zcGFyZW50Q29sb3IKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBBcnJheXMuZmlsbCgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoYnl0ZSkgZ2lmRGF0YVN0cmVhbS5sb2dpY2FsU2NyZWVuLmJhY2tncm91bmRDb2xvcgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIHNldFBpeGVscygKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VMZWZ0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbWFnZVRvcCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VXaWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VIZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG51bGwsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlV2lkdGgKLSAgICAgICAgICAgICAgICAgICAgICAgICk7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIHNlbmRUb1NjcmVlbkJ1ZmZlcihkYXRhKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgY2FzZSBESVNQT1NBTF9SRVNUT1JFOiB7Ci0gICAgICAgICAgICAgICAgICAgIHNjcmVlbkJ1ZmZlclRvU2NyZWVuKCk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBjYXNlIERJU1BPU0FMX05PTkU6Ci0gICAgICAgICAgICAgICAgY2FzZSBESVNQT1NBTF9OT0RJU1BPU0FMOgotICAgICAgICAgICAgICAgIGRlZmF1bHQ6IHsKLSAgICAgICAgICAgICAgICAgICAgLy8gQ29weSB0cmFuc21pdHRlZCBkYXRhIHRvIHRoZSBzY3JlZW4gYnVmZmVyCi0gICAgICAgICAgICAgICAgICAgIE9iamVjdCBkYXRhID0gZm9yY2VSR0IgPyAoT2JqZWN0KSBnZXRSZ2JJbWFnZURhdGEoKSA6IGltYWdlRGF0YTsKLSAgICAgICAgICAgICAgICAgICAgc2VuZFRvU2NyZWVuQnVmZmVyKGRhdGEpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBwcml2YXRlIHZvaWQgc2VuZFRvU2NyZWVuQnVmZmVyKE9iamVjdCBkYXRhKSB7Ci0gICAgICAgICAgICBpbnQgZGF0YUludFtdOwotICAgICAgICAgICAgYnl0ZSBkYXRhQnl0ZVtdOwotCi0gICAgICAgICAgICBpbnQgd2lkdGggPSBnaWZEYXRhU3RyZWFtLmxvZ2ljYWxTY3JlZW4ubG9naWNhbFNjcmVlbldpZHRoOwotCi0KLSAgICAgICAgICAgIGlmIChmb3JjZVJHQikgewotICAgICAgICAgICAgICAgIGRhdGFJbnQgPSAoaW50W10pIGRhdGE7Ci0KLSAgICAgICAgICAgICAgICBpZiAoaW1hZ2VXaWR0aCA9PSB3aWR0aCkgewotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGRhdGFJbnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBnZXRTY3JlZW5SR0JCdWZmZXIoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbWFnZUxlZnQgKyBpbWFnZVRvcCp3aWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkYXRhSW50Lmxlbmd0aAotICAgICAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7IC8vIEVhY2ggc2NhbmxpbmUKLSAgICAgICAgICAgICAgICAgICAgY29weVNjYW5saW5lcyhkYXRhSW50LCBnZXRTY3JlZW5SR0JCdWZmZXIoKSwgd2lkdGgpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZGF0YUJ5dGUgPSAoYnl0ZVtdKSBkYXRhOwotCi0gICAgICAgICAgICAgICAgaWYgKGltYWdlV2lkdGggPT0gd2lkdGgpIHsKLSAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShkYXRhQnl0ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNjcmVlbkJ1ZmZlciwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbWFnZUxlZnQgKyBpbWFnZVRvcCp3aWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkYXRhQnl0ZS5sZW5ndGgKLSAgICAgICAgICAgICAgICAgICAgKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgeyAvLyBFYWNoIHNjYW5saW5lCi0gICAgICAgICAgICAgICAgICAgIGNvcHlTY2FubGluZXMoZGF0YUJ5dGUsIHNjcmVlbkJ1ZmZlciwgd2lkdGgpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSAvLyBzZW5kVG9TY3JlZW5CdWZmZXIKLQotICAgICAgICBwcml2YXRlIHZvaWQgY29weVNjYW5saW5lcyhPYmplY3Qgc3JjLCBPYmplY3QgZHN0LCBpbnQgd2lkdGgpIHsKLSAgICAgICAgICAgIGZvciAoaW50IGk9MDsgaTxpbWFnZUhlaWdodDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShzcmMsCi0gICAgICAgICAgICAgICAgICAgICAgICBpKmltYWdlV2lkdGgsCi0gICAgICAgICAgICAgICAgICAgICAgICBkc3QsCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZUxlZnQgKyBpKndpZHRoICsgaW1hZ2VUb3Aqd2lkdGgsCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZVdpZHRoCi0gICAgICAgICAgICAgICAgKTsKLSAgICAgICAgICAgIH0gLy8gZm9yCi0gICAgICAgIH0KLQotICAgICAgICBwcml2YXRlIHZvaWQgc2NyZWVuQnVmZmVyVG9TY3JlZW4oKSB7Ci0gICAgICAgICAgICBpbnQgd2lkdGggPSBnaWZEYXRhU3RyZWFtLmxvZ2ljYWxTY3JlZW4ubG9naWNhbFNjcmVlbldpZHRoOwotCi0gICAgICAgICAgICBPYmplY3QgZHN0ID0gZm9yY2VSR0IgPwotICAgICAgICAgICAgICAgICAgICAoT2JqZWN0KSBuZXcgaW50W2ltYWdlV2lkdGgqaW1hZ2VIZWlnaHRdIDoKLSAgICAgICAgICAgICAgICAgICAgbmV3IGJ5dGVbaW1hZ2VXaWR0aCppbWFnZUhlaWdodF07Ci0KLSAgICAgICAgICAgIE9iamVjdCBzcmMgPSBmb3JjZVJHQiA/Ci0gICAgICAgICAgICAgICAgICAgIGdldFNjcmVlblJHQkJ1ZmZlcigpIDoKLSAgICAgICAgICAgICAgICAgICAgKE9iamVjdCkgc2NyZWVuQnVmZmVyOwotCi0gICAgICAgICAgICBpbnQgb2Zmc2V0ID0gMDsKLSAgICAgICAgICAgIE9iamVjdCB0b1NlbmQ7Ci0KLSAgICAgICAgICAgIGlmICh3aWR0aCA9PSBpbWFnZVdpZHRoKSB7Ci0gICAgICAgICAgICAgICAgb2Zmc2V0ID0gaW1hZ2VXaWR0aCAqIGltYWdlVG9wOwotICAgICAgICAgICAgICAgIHRvU2VuZCA9IHNyYzsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgaT0wOyBpPGltYWdlSGVpZ2h0OyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShzcmMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VMZWZ0ICsgaSp3aWR0aCArIGltYWdlVG9wKndpZHRoLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpKmltYWdlV2lkdGgsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VXaWR0aAotICAgICAgICAgICAgICAgICAgICApOwotICAgICAgICAgICAgICAgIH0gLy8gZm9yCi0gICAgICAgICAgICAgICAgdG9TZW5kID0gZHN0OwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoZm9yY2VSR0IpIHsKLSAgICAgICAgICAgICAgICBzZXRQaXhlbHMoCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZUxlZnQsCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZVRvcCwKLSAgICAgICAgICAgICAgICAgICAgICAgIGltYWdlV2lkdGgsCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZUhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgICAgIENvbG9yTW9kZWwuZ2V0UkdCZGVmYXVsdCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgKGludCBbXSl0b1NlbmQsCi0gICAgICAgICAgICAgICAgICAgICAgICBvZmZzZXQsCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZVdpZHRoCi0gICAgICAgICAgICAgICAgKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgc2V0UGl4ZWxzKAotICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VMZWZ0LAotICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VUb3AsCi0gICAgICAgICAgICAgICAgICAgICAgICBpbWFnZVdpZHRoLAotICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VIZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgICAgICBudWxsLAotICAgICAgICAgICAgICAgICAgICAgICAgKGJ5dGUgW10pdG9TZW5kLAotICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0LAotICAgICAgICAgICAgICAgICAgICAgICAgaW1hZ2VXaWR0aAotICAgICAgICAgICAgICAgICk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBjbGFzcyBHaWZDb2xvclRhYmxlIHsKLSAgICAgICAgLy8gIEluZGljYXRlcyB0aGF0IHJlYWRpbmcgb2YgdGhpcyBibG9jayBhY2NvbXBsaXNoZWQKLSAgICAgICAgYm9vbGVhbiBjb21wbGV0ZWQgPSBmYWxzZTsKLQotICAgICAgICBJbmRleENvbG9yTW9kZWwgY20gPSBudWxsOwotICAgICAgICBpbnQgc2l6ZSA9IDA7IC8vIEFjdHVhbCBudW1iZXIgb2YgY29sb3JzIGluIHRoZSBjb2xvciB0YWJsZQotICAgICAgICBieXRlIGNvbG9yc1tdID0gbmV3IGJ5dGVbMjU2KjNdOwotCi0gICAgICAgIEluZGV4Q29sb3JNb2RlbCBnZXRDb2xvck1vZGVsKGludCB0cmFuc3BhcmVudENvbG9yKSB7Ci0gICAgICAgICAgICBpZiAoY20gIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIGlmICh0cmFuc3BhcmVudENvbG9yICE9IGNtLmdldFRyYW5zcGFyZW50UGl4ZWwoKSkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gY20gPSBudWxsOyAvLyBGb3JjZSBkZWZhdWx0IEFSR0IgY29sb3IgbW9kZWwKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGNtOwotICAgICAgICAgICAgfSBlbHNlCi0gICAgICAgICAgICAgICAgaWYgKGNvbXBsZXRlZCAmJiBzaXplID4gMCkgewotICAgICAgICAgICAgICAgICAgICBpZiAodHJhbnNwYXJlbnRDb2xvciA9PSBJTVBPU1NJQkxFX1ZBTFVFKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gY20gPQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXcgSW5kZXhDb2xvck1vZGVsKDgsIHNpemUsIGNvbG9ycywgMCwgZmFsc2UpOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgaWYgKHRyYW5zcGFyZW50Q29sb3IgPiBzaXplKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzaXplID0gdHJhbnNwYXJlbnRDb2xvciArIDE7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGNtID0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXcgSW5kZXhDb2xvck1vZGVsKDgsIHNpemUsIGNvbG9ycywgMCwgZmFsc2UsIHRyYW5zcGFyZW50Q29sb3IpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIGNtID0gbnVsbDsgLy8gRm9yY2UgZGVmYXVsdCBBUkdCIGNvbG9yIG1vZGVsCi0gICAgICAgIH0KLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9JbWFnZURlY29kZXIuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0ltYWdlRGVjb2Rlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkMTYxMjhlLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0ltYWdlRGVjb2Rlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjU4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0vKgotICogQ3JlYXRlZCBvbiAxOC4wMS4yMDA1Ci0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZTsKLQotaW1wb3J0IGNvbS5hbmRyb2lkLmludGVybmFsLmF3dC5BbmRyb2lkSW1hZ2VEZWNvZGVyOwotCi1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbWFnZUNvbnN1bWVyOwotaW1wb3J0IGphdmEuaW8uQnVmZmVyZWRJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5BY2Nlc3NDb250cm9sbGVyOwotaW1wb3J0IGphdmEuc2VjdXJpdHkuUHJpdmlsZWdlZEFjdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuQ29uY3VycmVudE1vZGlmaWNhdGlvbkV4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaHRhYmxlOwotaW1wb3J0IGphdmEudXRpbC5JdGVyYXRvcjsKLWltcG9ydCBqYXZhLnV0aWwuTGlzdDsKLQotCi0vKioKLSAqIFRoaXMgY2xhc3MgY29udGFpbnMgY29tbW9uIGZ1bmN0aW9uYWxpdHkgZm9yIGFsbCBpbWFnZSBkZWNvZGVycy4KLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIEltYWdlRGVjb2RlciB7Ci0gICAgCi0gICAgLyoqIEltYWdlIHR5cGVzICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgR0VORVJJQ19ERUNPREVSID0gMDsKLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBKUEdfREVDT0RFUiA9IDE7Ci0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgR0lGX0RFQ09ERVIgPSAyOwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IFBOR19ERUNPREVSID0gMzsKLSAgICAKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgTUFYX0JZVEVTX0lOX1NJR05BVFVSRSA9IDg7Ci0KLSAgICBwcm90ZWN0ZWQgTGlzdDxJbWFnZUNvbnN1bWVyPiBjb25zdW1lcnM7Ci0gICAgcHJvdGVjdGVkIElucHV0U3RyZWFtIGlucHV0U3RyZWFtOwotICAgIHByb3RlY3RlZCBEZWNvZGluZ0ltYWdlU291cmNlIHNyYzsKLQotICAgIHByb3RlY3RlZCBib29sZWFuIHRlcm1pbmF0ZWQ7Ci0KLSAgICAvKioKLSAgICAgKiBDaG9vc2VzIGFwcHJvcHJpYXRlIGltYWdlIGRlY29kZXIgYnkgbG9va2luZyBpbnRvIGlucHV0IHN0cmVhbSBhbmQgY2hlY2tpbmcKLSAgICAgKiB0aGUgaW1hZ2Ugc2lnbmF0dXJlLgotICAgICAqIEBwYXJhbSBzcmMgLSBpbWFnZSBwcm9kdWNlciwgcmVxdWlyZWQgZm9yIHBhc3NpbmcgZGF0YSB0byBpdCBmcm9tIHRoZQotICAgICAqIGNyZWF0ZWQgZGVjb2RlciB2aWEgY2FsbGJhY2tzCi0gICAgICogQHBhcmFtIGlzIC0gc3RyZWFtCi0gICAgICogQHJldHVybiBkZWNvZGVyCi0gICAgICovCi0gICAgc3RhdGljIEltYWdlRGVjb2RlciBjcmVhdGVEZWNvZGVyKERlY29kaW5nSW1hZ2VTb3VyY2Ugc3JjLCBJbnB1dFN0cmVhbSBpcykgewotICAgICAgICBJbnB1dFN0cmVhbSBtYXJrYWJsZTsKLQotICAgICAgICBpZiAoIWlzLm1hcmtTdXBwb3J0ZWQoKSkgewotICAgICAgICAgICAgLy8gQkVHSU4gYW5kcm9pZC1tb2RpZmllZAotICAgICAgICAgICAgbWFya2FibGUgPSBuZXcgQnVmZmVyZWRJbnB1dFN0cmVhbShpcywgODE5Mik7Ci0gICAgICAgICAgICAvLyBFTkQgYW5kcm9pZC1tb2RpZmllZAotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbWFya2FibGUgPSBpczsKLSAgICAgICAgfQotICAgICAgICAgICAgCi0gICAgICAgIC8vIFJlYWQgdGhlIHNpZ25hdHVyZSBmcm9tIHRoZSBzdHJlYW0gYW5kIHRoZW4gcmVzZXQgaXQgYmFjawotICAgICAgICB0cnkgewotICAgICAgICAgICAgbWFya2FibGUubWFyayhNQVhfQllURVNfSU5fU0lHTkFUVVJFKTsKLQotICAgICAgICAgICAgYnl0ZVtdIHNpZ25hdHVyZSA9IG5ldyBieXRlW01BWF9CWVRFU19JTl9TSUdOQVRVUkVdOwotICAgICAgICAgICAgbWFya2FibGUucmVhZChzaWduYXR1cmUsIDAsIE1BWF9CWVRFU19JTl9TSUdOQVRVUkUpOwotICAgICAgICAgICAgbWFya2FibGUucmVzZXQoKTsKLQotICAgICAgICAgICAgaWYgKChzaWduYXR1cmVbMF0gJiAweEZGKSA9PSAweEZGICYmCi0gICAgICAgICAgICAgICAgICAgIChzaWduYXR1cmVbMV0gJiAweEZGKSA9PSAweEQ4ICYmCi0gICAgICAgICAgICAgICAgICAgIChzaWduYXR1cmVbMl0gJiAweEZGKSA9PSAweEZGKSB7IC8vIEpQRUcKLSAgICAgICAgICAgICAgICByZXR1cm4gbG9hZERlY29kZXIoUE5HX0RFQ09ERVIsIHNyYywgaXMpOwotICAgICAgICAgICAgfSBlbHNlIGlmICgoc2lnbmF0dXJlWzBdICYgMHhGRikgPT0gMHg0NyAmJiAvLyBHCi0gICAgICAgICAgICAgICAgICAgIChzaWduYXR1cmVbMV0gJiAweEZGKSA9PSAweDQ5ICYmIC8vIEkKLSAgICAgICAgICAgICAgICAgICAgKHNpZ25hdHVyZVsyXSAmIDB4RkYpID09IDB4NDYpIHsgLy8gRgotICAgICAgICAgICAgICAgIHJldHVybiBsb2FkRGVjb2RlcihHSUZfREVDT0RFUiwgc3JjLCBpcyk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKChzaWduYXR1cmVbMF0gJiAweEZGKSA9PSAxMzcgJiYgLy8gUE5HIHNpZ25hdHVyZTogMTM3IDgwIDc4IDcxIDEzIDEwIDI2IDEwCi0gICAgICAgICAgICAgICAgICAgIChzaWduYXR1cmVbMV0gJiAweEZGKSA9PSA4MCAmJgotICAgICAgICAgICAgICAgICAgICAoc2lnbmF0dXJlWzJdICYgMHhGRikgPT0gNzggJiYKLSAgICAgICAgICAgICAgICAgICAgKHNpZ25hdHVyZVszXSAmIDB4RkYpID09IDcxICYmCi0gICAgICAgICAgICAgICAgICAgIChzaWduYXR1cmVbNF0gJiAweEZGKSA9PSAxMyAmJgotICAgICAgICAgICAgICAgICAgICAoc2lnbmF0dXJlWzVdICYgMHhGRikgPT0gMTAgJiYKLSAgICAgICAgICAgICAgICAgICAgKHNpZ25hdHVyZVs2XSAmIDB4RkYpID09IDI2ICYmCi0gICAgICAgICAgICAgICAgICAgIChzaWduYXR1cmVbN10gJiAweEZGKSA9PSAxMCkgewotICAgICAgICAgICAgICAgIHJldHVybiBsb2FkRGVjb2RlcihKUEdfREVDT0RFUiwgc3JjLCBpcyk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHJldHVybiBsb2FkRGVjb2RlcihHRU5FUklDX0RFQ09ERVIsIHNyYywgaXMpOwotICAgICAgICAgICAgCi0gICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsgLy8gU2lsZW50bHkKLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLSAgICAKLSAgICAvKgotICAgICAqIEluIHRoZSBmdXR1cmUsIHdlIG1pZ2h0IHJldHVybiBkaWZmZXJlbnQgZGVjb2RlcnMgZm9yIGRpZmZlcmVuIGltYWdlIHR5cGVzLgotICAgICAqIEJ1dCBmb3Igbm93LCB3ZSBhbHdheXMgcmV0dXJuIHRoZSBnZW5lcmljIG9uZS4KLSAgICAgKiBBbHNvOiB3ZSBjb3VsZCBhZGQgYSBmYWN0b3J5IHRvIGxvYWQgaW1hZ2UgZGVjb2Rlci4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBJbWFnZURlY29kZXIgbG9hZERlY29kZXIoaW50IHR5cGUsIERlY29kaW5nSW1hZ2VTb3VyY2Ugc3JjLCAKLSAgICAgICAgICAgIElucHV0U3RyZWFtIGlzKSB7Ci0gICAgICAgIHJldHVybiBuZXcgQW5kcm9pZEltYWdlRGVjb2RlcihzcmMsIGlzKTsKLSAgICB9Ci0KLSAgICBwcm90ZWN0ZWQgSW1hZ2VEZWNvZGVyKERlY29kaW5nSW1hZ2VTb3VyY2UgX3NyYywgSW5wdXRTdHJlYW0gaXMpIHsKLSAgICAgICAgc3JjID0gX3NyYzsKLSAgICAgICAgY29uc3VtZXJzID0gc3JjLmNvbnN1bWVyczsKLSAgICAgICAgaW5wdXRTdHJlYW0gPSBpczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBkZWNvZGVJbWFnZSgpIHRocm93cyBJT0V4Y2VwdGlvbjsKLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCBjbG9zZVN0cmVhbSgpIHsKLSAgICAgICAgaWYgKGlucHV0U3RyZWFtICE9IG51bGwpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgaW5wdXRTdHJlYW0uY2xvc2UoKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFN0b3BzIHRoZSBkZWNvZGluZyBieSBpbnRlcnJ1cHRpbmcgdGhlIGN1cnJlbnQgZGVjb2RpbmcgdGhyZWFkLgotICAgICAqIFVzZWQgd2hlbiBhbGwgY29uc3VtZXJzIGFyZSByZW1vdmVkIGFuZCB0aGVyZSdzIG5vIG1vcmUgbmVlZCB0bwotICAgICAqIHJ1biB0aGUgZGVjb2Rlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCB0ZXJtaW5hdGUoKSB7Ci0gICAgICAgIHNyYy5sb2NrRGVjb2Rlcih0aGlzKTsKLSAgICAgICAgY2xvc2VTdHJlYW0oKTsKLQotICAgICAgICBBY2Nlc3NDb250cm9sbGVyLmRvUHJpdmlsZWdlZCgKLSAgICAgICAgICAgICAgICBuZXcgUHJpdmlsZWdlZEFjdGlvbjxWb2lkPigpIHsKLSAgICAgICAgICAgICAgICAgICAgcHVibGljIFZvaWQgcnVuKCkgewotICAgICAgICAgICAgICAgICAgICAgICAgVGhyZWFkLmN1cnJlbnRUaHJlYWQoKS5pbnRlcnJ1cHQoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICApOwotCi0gICAgICAgIHRlcm1pbmF0ZWQgPSB0cnVlOwotICAgIH0KLQotICAgIHByb3RlY3RlZCB2b2lkIHNldERpbWVuc2lvbnMoaW50IHcsIGludCBoKSB7Ci0gICAgICAgIGlmICh0ZXJtaW5hdGVkKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IgKEltYWdlQ29uc3VtZXIgaWMgOiBjb25zdW1lcnMpIHsKLSAgICAgICAgICAgIGljLnNldERpbWVuc2lvbnModywgaCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwcm90ZWN0ZWQgdm9pZCBzZXRQcm9wZXJ0aWVzKEhhc2h0YWJsZTw/LCA/PiBwcm9wcykgewotICAgICAgICBpZiAodGVybWluYXRlZCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChJbWFnZUNvbnN1bWVyIGljIDogY29uc3VtZXJzKSB7Ci0gICAgICAgICAgICBpYy5zZXRQcm9wZXJ0aWVzKHByb3BzKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHByb3RlY3RlZCB2b2lkIHNldENvbG9yTW9kZWwoQ29sb3JNb2RlbCBjbSkgewotICAgICAgICBpZiAodGVybWluYXRlZCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChJbWFnZUNvbnN1bWVyIGljIDogY29uc3VtZXJzKSB7Ci0gICAgICAgICAgICBpYy5zZXRDb2xvck1vZGVsKGNtKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHByb3RlY3RlZCB2b2lkIHNldEhpbnRzKGludCBoaW50cykgewotICAgICAgICBpZiAodGVybWluYXRlZCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yIChJbWFnZUNvbnN1bWVyIGljIDogY29uc3VtZXJzKSB7Ci0gICAgICAgICAgICBpYy5zZXRIaW50cyhoaW50cyk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwcm90ZWN0ZWQgdm9pZCBzZXRQaXhlbHMoCi0gICAgICAgICAgICBpbnQgeCwgaW50IHksCi0gICAgICAgICAgICBpbnQgdywgaW50IGgsCi0gICAgICAgICAgICBDb2xvck1vZGVsIG1vZGVsLAotICAgICAgICAgICAgYnl0ZSBwaXhbXSwKLSAgICAgICAgICAgIGludCBvZmYsIGludCBzY2Fuc2l6ZQotICAgICAgICAgICAgKSB7Ci0gICAgICAgIGlmICh0ZXJtaW5hdGVkKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBzcmMubG9ja0RlY29kZXIodGhpcyk7Ci0KLSAgICAgICAgZm9yIChJbWFnZUNvbnN1bWVyIGljIDogY29uc3VtZXJzKSB7Ci0gICAgICAgICAgICBpYy5zZXRQaXhlbHMoeCwgeSwgdywgaCwgbW9kZWwsIHBpeCwgb2ZmLCBzY2Fuc2l6ZSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwcm90ZWN0ZWQgdm9pZCBzZXRQaXhlbHMoCi0gICAgICAgICAgICBpbnQgeCwgaW50IHksCi0gICAgICAgICAgICBpbnQgdywgaW50IGgsCi0gICAgICAgICAgICBDb2xvck1vZGVsIG1vZGVsLAotICAgICAgICAgICAgaW50IHBpeFtdLAotICAgICAgICAgICAgaW50IG9mZiwgaW50IHNjYW5zaXplCi0gICAgICAgICAgICApIHsKLSAgICAgICAgaWYgKHRlcm1pbmF0ZWQpIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIHNyYy5sb2NrRGVjb2Rlcih0aGlzKTsKLQotICAgICAgICBmb3IgKEltYWdlQ29uc3VtZXIgaWMgOiBjb25zdW1lcnMpIHsKLSAgICAgICAgICAgIGljLnNldFBpeGVscyh4LCB5LCB3LCBoLCBtb2RlbCwgcGl4LCBvZmYsIHNjYW5zaXplKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHByb3RlY3RlZCB2b2lkIGltYWdlQ29tcGxldGUoaW50IHN0YXR1cykgewotICAgICAgICBpZiAodGVybWluYXRlZCkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgc3JjLmxvY2tEZWNvZGVyKHRoaXMpOwotCi0gICAgICAgIEltYWdlQ29uc3VtZXIgaWMgPSBudWxsOwotCi0gICAgICAgIGZvciAoSXRlcmF0b3I8SW1hZ2VDb25zdW1lcj4gaSA9IGNvbnN1bWVycy5pdGVyYXRvcigpOyBpLmhhc05leHQoKTspIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgaWMgPSBpLm5leHQoKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKENvbmN1cnJlbnRNb2RpZmljYXRpb25FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIGkgPSBjb25zdW1lcnMuaXRlcmF0b3IoKTsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGljLmltYWdlQ29tcGxldGUoc3RhdHVzKTsKLSAgICAgICAgfQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvSW1hZ2VMb2FkZXIuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL0ltYWdlTG9hZGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDVjN2QxODAuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvSW1hZ2VMb2FkZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDIwOCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotLyoKLSAqIENyZWF0ZWQgb24gMTguMDEuMjAwNQotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLnNlY3VyaXR5LkFjY2Vzc0NvbnRyb2xsZXI7Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5Qcml2aWxlZ2VkQWN0aW9uOwotaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLkxpbmtlZExpc3Q7Ci1pbXBvcnQgamF2YS51dGlsLkxpc3Q7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBwcm92aWRlcyBmdW5jdGlvbmFsaXR5IGZvciBzaW11bHRhbmVvdXMgbG9hZGluZyBvZgotICogc2V2ZXJhbCBpbWFnZXMgYW5kIHJ1bm5pbmcgYW5pbWF0aW9uLgotICovCi1wdWJsaWMgY2xhc3MgSW1hZ2VMb2FkZXIgZXh0ZW5kcyBUaHJlYWQgewotICAgIC8vIENvbnRhaW5zIEltYWdlTG9hZGVyIG9iamVjdHMKLSAgICAvLyBhbmQgcXVldWUgb2YgaW1hZ2Ugc291cmNlcyB3YWl0aW5nIHRvIGJlIGxvYWRlZAotICAgIHN0YXRpYyBjbGFzcyBJbWFnZUxvYWRlcnNTdG9yYWdlIHsKLSAgICAgICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IE1BWF9USFJFQURTID0gNTsKLSAgICAgICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFRJTUVPVVQgPSA0MDAwOwotICAgICAgICBzdGF0aWMgSW1hZ2VMb2FkZXJzU3RvcmFnZSBpbnN0YW5jZTsKLQotICAgICAgICBMaXN0PERlY29kaW5nSW1hZ2VTb3VyY2U+IHF1ZXVlID0gbmV3IExpbmtlZExpc3Q8RGVjb2RpbmdJbWFnZVNvdXJjZT4oKTsKLSAgICAgICAgTGlzdDxUaHJlYWQ+IGxvYWRlcnMgPSBuZXcgQXJyYXlMaXN0PFRocmVhZD4oTUFYX1RIUkVBRFMpOwotCi0gICAgICAgIHByaXZhdGUgaW50IGZyZWVMb2FkZXJzOwotCi0gICAgICAgIHByaXZhdGUgSW1hZ2VMb2FkZXJzU3RvcmFnZSgpIHt9Ci0KLSAgICAgICAgc3RhdGljIEltYWdlTG9hZGVyc1N0b3JhZ2UgZ2V0U3RvcmFnZSgpIHsKLSAgICAgICAgICAgIGlmIChpbnN0YW5jZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgaW5zdGFuY2UgPSBuZXcgSW1hZ2VMb2FkZXJzU3RvcmFnZSgpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByZXR1cm4gaW5zdGFuY2U7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBJbWFnZUxvYWRlcigpIHsKLSAgICAgICAgc3VwZXIoKTsKLSAgICAgICAgc2V0RGFlbW9uKHRydWUpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFRoaXMgbWV0aG9kIGNyZWF0ZXMgYSBuZXcgdGhyZWFkIHdoaWNoIGlzIGFibGUgdG8gbG9hZCBhbiBpbWFnZQotICAgICAqIG9yIHJ1biBhbmltYXRpb24gKGlmIHRoZSBudW1iZXIgb2YgZXhpc3RpbmcgbG9hZGVyIHRocmVhZHMgZG9lcyBub3QKLSAgICAgKiBleGNlZWQgdGhlIGxpbWl0KS4KLSAgICAgKi8KLSAgICBwcml2YXRlIHN0YXRpYyB2b2lkIGNyZWF0ZUxvYWRlcigpIHsKLSAgICAgICAgZmluYWwgSW1hZ2VMb2FkZXJzU3RvcmFnZSBzdG9yYWdlID0gSW1hZ2VMb2FkZXJzU3RvcmFnZS5nZXRTdG9yYWdlKCk7Ci0KLSAgICAgICAgc3luY2hyb25pemVkKHN0b3JhZ2UubG9hZGVycykgewotICAgICAgICAgICAgaWYgKHN0b3JhZ2UubG9hZGVycy5zaXplKCkgPCBJbWFnZUxvYWRlcnNTdG9yYWdlLk1BWF9USFJFQURTKSB7Ci0gICAgICAgICAgICAgICAgQWNjZXNzQ29udHJvbGxlci5kb1ByaXZpbGVnZWQoCi0gICAgICAgICAgICAgICAgICAgICAgICBuZXcgUHJpdmlsZWdlZEFjdGlvbjxWb2lkPigpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBwdWJsaWMgVm9pZCBydW4oKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEltYWdlTG9hZGVyIGxvYWRlciA9IG5ldyBJbWFnZUxvYWRlcigpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzdG9yYWdlLmxvYWRlcnMuYWRkKGxvYWRlcik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxvYWRlci5zdGFydCgpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICB9KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIEFkZHMgYSBuZXcgaW1hZ2Ugc291cmNlIHRvIHRoZSBxdWV1ZSBhbmQgc3RhcnRzIGEgbmV3IGxvYWRlcgotICAgICAqIHRocmVhZCBpZiByZXF1aXJlZAotICAgICAqIEBwYXJhbSBpbWdTcmMgLSBpbWFnZSBzb3VyY2UKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIHZvaWQgYWRkSW1hZ2VTb3VyY2UoRGVjb2RpbmdJbWFnZVNvdXJjZSBpbWdTcmMpIHsKLSAgICAgICAgSW1hZ2VMb2FkZXJzU3RvcmFnZSBzdG9yYWdlID0gSW1hZ2VMb2FkZXJzU3RvcmFnZS5nZXRTdG9yYWdlKCk7Ci0gICAgICAgIHN5bmNocm9uaXplZChzdG9yYWdlLnF1ZXVlKSB7Ci0gICAgICAgICAgICBpZiAoIXN0b3JhZ2UucXVldWUuY29udGFpbnMoaW1nU3JjKSkgewotICAgICAgICAgICAgICAgIHN0b3JhZ2UucXVldWUuYWRkKGltZ1NyYyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoc3RvcmFnZS5mcmVlTG9hZGVycyA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgY3JlYXRlTG9hZGVyKCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHN0b3JhZ2UucXVldWUubm90aWZ5KCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBXYWl0cyBmb3IgYSBuZXcgSW1hZ2VTb3VyY2UgdW50aWwgdGltb3V0IGV4cGlyZXMuCi0gICAgICogTG9hZGVyIHRocmVhZCB3aWxsIHRlcm1pbmF0ZSBhZnRlciByZXR1cm5pbmcgZnJvbSB0aGlzIG1ldGhvZAotICAgICAqIGlmIHRpbWVvdXQgZXhwaXJlZCBhbmQgaW1hZ2Ugc291cmNlIHdhcyBub3QgcGlja2VkIHVwIGZyb20gdGhlIHF1ZXVlLgotICAgICAqIEByZXR1cm4gaW1hZ2Ugc291cmNlIHBpY2tlZCB1cCBmcm9tIHRoZSBxdWV1ZSBvciBudWxsIGlmIHRpbWVvdXQgZXhwaXJlZAotICAgICAqLwotICAgIHByaXZhdGUgc3RhdGljIERlY29kaW5nSW1hZ2VTb3VyY2UgZ2V0V2FpdGluZ0ltYWdlU291cmNlKCkgewotICAgICAgICBJbWFnZUxvYWRlcnNTdG9yYWdlIHN0b3JhZ2UgPSBJbWFnZUxvYWRlcnNTdG9yYWdlLmdldFN0b3JhZ2UoKTsKLQotICAgICAgICBzeW5jaHJvbml6ZWQoc3RvcmFnZS5xdWV1ZSkgewotICAgICAgICAgICAgRGVjb2RpbmdJbWFnZVNvdXJjZSBpc3JjID0gbnVsbDsKLQotICAgICAgICAgICAgaWYgKHN0b3JhZ2UucXVldWUuc2l6ZSgpID09IDApIHsKLSAgICAgICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgICAgICBzdG9yYWdlLmZyZWVMb2FkZXJzKys7Ci0gICAgICAgICAgICAgICAgICAgIHN0b3JhZ2UucXVldWUud2FpdChJbWFnZUxvYWRlcnNTdG9yYWdlLlRJTUVPVVQpOwotICAgICAgICAgICAgICAgIH0gY2F0Y2ggKEludGVycnVwdGVkRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgICAgICAgICAgc3RvcmFnZS5mcmVlTG9hZGVycy0tOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKHN0b3JhZ2UucXVldWUuc2l6ZSgpID4gMCkgewotICAgICAgICAgICAgICAgIGlzcmMgPSBzdG9yYWdlLnF1ZXVlLmdldCgwKTsKLSAgICAgICAgICAgICAgICBzdG9yYWdlLnF1ZXVlLnJlbW92ZSgwKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcmV0dXJuIGlzcmM7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBFbnRyeSBwb2ludCBvZiB0aGUgbG9hZGVyIHRocmVhZC4gUGlja3MgdXAgaW1hZ2Ugc291cmNlcyBhbmQKLSAgICAgKiBydW5zIGRlY29kZXJzIGZvciB0aGVtIHdoaWxlIHRoZXJlIGFyZSBhdmFpbGFibGUgaW1hZ2Ugc291cmNlcyBpbiB0aGUgcXVldWUuCi0gICAgICogSWYgdGhlcmUgYXJlIG5vIGFuZCB0aW1lb3V0IGV4cGlyZXMgaXQgdGVybWluYXRlcy4KLSAgICAgKi8KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBydW4oKSB7Ci0gICAgICAgIEltYWdlTG9hZGVyc1N0b3JhZ2Ugc3RvcmFnZSA9IEltYWdlTG9hZGVyc1N0b3JhZ2UuZ2V0U3RvcmFnZSgpOwotCi0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICB3aGlsZSAoc3RvcmFnZS5sb2FkZXJzLmNvbnRhaW5zKHRoaXMpKSB7Ci0gICAgICAgICAgICAgICAgVGhyZWFkLmludGVycnVwdGVkKCk7IC8vIFJlc2V0IHRoZSBpbnRlcnJ1cHRlZCBmbGFnCi0gICAgICAgICAgICAgICAgRGVjb2RpbmdJbWFnZVNvdXJjZSBpc3JjID0gZ2V0V2FpdGluZ0ltYWdlU291cmNlKCk7Ci0gICAgICAgICAgICAgICAgaWYgKGlzcmMgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgICAgICAgICAgaXNyYy5sb2FkKCk7Ci0gICAgICAgICAgICAgICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBlLnByaW50U3RhY2tUcmFjZSgpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7IC8vIERvbid0IHdhaXQgaWYgdGltZW91dCBleHBpcmVkIC0gdGVybWluYXRlIGxvYWRlcgotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIGUucHJpbnRTdGFja1RyYWNlKCk7Ci0gICAgICAgIH0gZmluYWxseSB7Ci0gICAgICAgICAgICBzeW5jaHJvbml6ZWQoc3RvcmFnZS5sb2FkZXJzKSB7Ci0gICAgICAgICAgICAgICAgc3RvcmFnZS5sb2FkZXJzLnJlbW92ZShUaHJlYWQuY3VycmVudFRocmVhZCgpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJlbW92ZXMgY3VycmVudCB0aHJlYWQgZnJvbSBsb2FkZXJzIChzbyB3ZSBhcmUgYWJsZQotICAgICAqIHRvIGNyZWF0ZSBtb3JlIGxvYWRlcnMpIGFuZCBkZWNyZWFzZXMgaXRzIHByaW9yaXR5LgotICAgICAqLwotICAgIHN0YXRpYyB2b2lkIGJlZ2luQW5pbWF0aW9uKCkgewotICAgICAgICBJbWFnZUxvYWRlcnNTdG9yYWdlIHN0b3JhZ2UgPSBJbWFnZUxvYWRlcnNTdG9yYWdlLmdldFN0b3JhZ2UoKTsKLSAgICAgICAgVGhyZWFkIGN1cnJUaHJlYWQgPSBUaHJlYWQuY3VycmVudFRocmVhZCgpOwotCi0gICAgICAgIHN5bmNocm9uaXplZChzdG9yYWdlKSB7Ci0gICAgICAgICAgICBzdG9yYWdlLmxvYWRlcnMucmVtb3ZlKGN1cnJUaHJlYWQpOwotCi0gICAgICAgICAgICBpZiAoc3RvcmFnZS5mcmVlTG9hZGVycyA8IHN0b3JhZ2UucXVldWUuc2l6ZSgpKSB7Ci0gICAgICAgICAgICAgICAgY3JlYXRlTG9hZGVyKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBjdXJyVGhyZWFkLnNldFByaW9yaXR5KFRocmVhZC5NSU5fUFJJT1JJVFkpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFNlbmRzIHRoZSBjdXJyZW50IHRocmVhZCB0byB3YWl0IGZvciB0aGUgbmV3IGltYWdlcyB0byBsb2FkCi0gICAgICogaWYgdGhlcmUgYXJlIGZyZWUgcGxhY2Vob2xkZXJzIGZvciBsb2FkZXJzCi0gICAgICovCi0gICAgc3RhdGljIHZvaWQgZW5kQW5pbWF0aW9uKCkgewotICAgICAgICBJbWFnZUxvYWRlcnNTdG9yYWdlIHN0b3JhZ2UgPSBJbWFnZUxvYWRlcnNTdG9yYWdlLmdldFN0b3JhZ2UoKTsKLSAgICAgICAgVGhyZWFkIGN1cnJUaHJlYWQgPSBUaHJlYWQuY3VycmVudFRocmVhZCgpOwotCi0gICAgICAgIHN5bmNocm9uaXplZChzdG9yYWdlKSB7Ci0gICAgICAgICAgICBpZiAoc3RvcmFnZS5sb2FkZXJzLnNpemUoKSA8IEltYWdlTG9hZGVyc1N0b3JhZ2UuTUFYX1RIUkVBRFMgJiYKLSAgICAgICAgICAgICAgICAgICAgIXN0b3JhZ2UubG9hZGVycy5jb250YWlucyhjdXJyVGhyZWFkKQotICAgICAgICAgICAgKSB7Ci0gICAgICAgICAgICAgICAgc3RvcmFnZS5sb2FkZXJzLmFkZChjdXJyVGhyZWFkKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGN1cnJUaHJlYWQuc2V0UHJpb3JpdHkoVGhyZWFkLk5PUk1fUFJJT1JJVFkpOwotICAgIH0KLX0KXCBObyBuZXdsaW5lIGF0IGVuZCBvZiBmaWxlCmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9KcGVnRGVjb2Rlci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvSnBlZ0RlY29kZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMmU2NDQyNy4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9KcGVnRGVjb2Rlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjMxICswLDAgQEAKLS8qCi0qICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0qICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotKgotKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0qCi0qICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0qICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0qICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0qICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5pbWFnZS4qOwotaW1wb3J0IGphdmEuYXd0LmNvbG9yLkNvbG9yU3BhY2U7Ci1pbXBvcnQgamF2YS5hd3QuKjsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS51dGlsLkhhc2h0YWJsZTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi1wdWJsaWMgY2xhc3MgSnBlZ0RlY29kZXIgZXh0ZW5kcyBJbWFnZURlY29kZXIgewotICAgIC8vIE9ubHkgMiBvdXRwdXQgY29sb3JzcGFjZXMgZXhwZWN0ZWQuIE90aGVycyBhcmUgY29udmVydGVkIGludG8KLSAgICAvLyB0aGVzZSBvbmVzLgotICAgIC8vIDEuIEdyYXlzY2FsZQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEpDU19HUkFZU0NBTEUgPSAxOwotICAgIC8vIDIuIFJHQgotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEpDU19SR0IgPSAyOwotCi0gICAgLy8gRmxhZ3MgZm9yIHRoZSBjb25zdW1lciwgcHJvZ3Jlc3NpdmUgSlBFRwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBoaW50ZmxhZ3NQcm9ncmVzc2l2ZSA9Ci0gICAgICAgICAgICBJbWFnZUNvbnN1bWVyLlNJTkdMRUZSQU1FIHwgLy8gSlBFRyBpcyBhIHN0YXRpYyBpbWFnZQotICAgICAgICAgICAgSW1hZ2VDb25zdW1lci5UT1BET1dOTEVGVFJJR0hUIHwgLy8gVGhpcyBvcmRlciBpcyBvbmx5IG9uZSBwb3NzaWJsZQotICAgICAgICAgICAgSW1hZ2VDb25zdW1lci5DT01QTEVURVNDQU5MSU5FUzsgLy8gRG9uJ3QgZGVsaXZlciBpbmNvbXBsZXRlIHNjYW5saW5lcwotICAgIC8vIEZsYWdzIGZvciB0aGUgY29uc3VtZXIsIHNpbmdsZXBhc3MgSlBFRwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBoaW50ZmxhZ3NTaW5nbGUgPQotICAgICAgICAgICAgSW1hZ2VDb25zdW1lci5TSU5HTEVQQVNTIHwKLSAgICAgICAgICAgIGhpbnRmbGFnc1Byb2dyZXNzaXZlOwotCi0gICAgLy8gQnVmZmVyIGZvciB0aGUgc3RyZWFtCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IEJVRkZFUl9TSVpFID0gMTAyNDsKLSAgICBwcml2YXRlIGJ5dGUgYnVmZmVyW10gPSBuZXcgYnl0ZVtCVUZGRVJfU0laRV07Ci0KLSAgICAvLyAzIHBvc3NpYmxlIGNvbG9yIG1vZGVscyBvbmx5Ci0gICAgcHJpdmF0ZSBzdGF0aWMgQ29sb3JNb2RlbCBjbVJHQjsKLSAgICBwcml2YXRlIHN0YXRpYyBDb2xvck1vZGVsIGNtR3JheTsKLQotICAgIC8vIGluaXRpYWxpemVzIHByb3BlciBmaWVsZCBJRHMKLSAgICBwcml2YXRlIHN0YXRpYyBuYXRpdmUgdm9pZCBpbml0SURzKCk7Ci0KLSAgICAvLyBQb2ludGVyIHRvIG5hdGl2ZSBzdHJ1Y3R1cmUgd2hpY2ggc3RvcmUgZGVjb2Rpbmcgc3RhdGUKLSAgICAvLyBiZXR3ZWVuIHN1YnNlcXVlbnQgZGVjb2RpbmcvSU8tc3VzcGVuc2lvbiBjeWNsZXMKLSAgICBwcml2YXRlIGxvbmcgaE5hdGl2ZURlY29kZXIgPSAwOyAvLyBOVUxMIGluaXRpYWxseQotCi0gICAgcHJpdmF0ZSBib29sZWFuIGhlYWRlckRvbmUgPSBmYWxzZTsKLQotICAgIC8vIE5leHQgNCBtZW1iZXJzIGFyZSBmaWxsZWQgYnkgdGhlIG5hdGl2ZSBtZXRob2QgKGRlY29tcHJlc3MpLgotICAgIC8vIFdlIGNhbiBzaW1wbHkgY2hlY2sgaWYgaW1hZ2VXaWR0aCBpcyBzdGlsbCBuZWdhdGl2ZSB0byBmaW5kCi0gICAgLy8gb3V0IGlmIHRoZXkgYXJlIGFscmVhZHkgZmlsbGVkLgotICAgIHByaXZhdGUgaW50IGltYWdlV2lkdGggPSAtMTsKLSAgICBwcml2YXRlIGludCBpbWFnZUhlaWdodCA9IC0xOwotICAgIHByaXZhdGUgYm9vbGVhbiBwcm9ncmVzc2l2ZSA9IGZhbHNlOwotICAgIHByaXZhdGUgaW50IGpwZWdDb2xvclNwYWNlID0gMDsKLQotICAgIC8vIFN0b3JlcyBudW1iZXIgb2YgYnl0ZXMgY29uc3VtZWQgYnkgdGhlIG5hdGl2ZSBkZWNvZGVyCi0gICAgcHJpdmF0ZSBpbnQgYnl0ZXNDb25zdW1lZCA9IDA7Ci0gICAgLy8gU3RvcmVzIGN1cnJlbnQgc2NhbmxpbmUgcmV0dXJuZWQgYnkgdGhlIGRlY29kZXIKLSAgICBwcml2YXRlIGludCBjdXJyU2NhbmxpbmUgPSAwOwotCi0gICAgcHJpdmF0ZSBDb2xvck1vZGVsIGNtID0gbnVsbDsKLQotICAgIHN0YXRpYyB7Ci0gICAgICAgIFN5c3RlbS5sb2FkTGlicmFyeSgianBlZ2RlY29kZXIiKTsgLy8kTk9OLU5MUy0xJAotCi0gICAgICAgIGNtR3JheSA9IG5ldyBDb21wb25lbnRDb2xvck1vZGVsKAotICAgICAgICAgICAgICAgIENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19HUkFZKSwKLSAgICAgICAgICAgICAgICBmYWxzZSwgZmFsc2UsCi0gICAgICAgICAgICAgICAgVHJhbnNwYXJlbmN5Lk9QQVFVRSwgRGF0YUJ1ZmZlci5UWVBFX0JZVEUKLSAgICAgICAgKTsKLQotICAgICAgICAvLyBDcmVhdGUgUkdCIGNvbG9yIG1vZGVsCi0gICAgICAgIGNtUkdCID0gbmV3IERpcmVjdENvbG9yTW9kZWwoMjQsIDB4RkYwMDAwLCAweEZGMDAsIDB4RkYpOwotCi0gICAgICAgIGluaXRJRHMoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgSnBlZ0RlY29kZXIoRGVjb2RpbmdJbWFnZVNvdXJjZSBzcmMsIElucHV0U3RyZWFtIGlzKSB7Ci0gICAgICAgIHN1cGVyKHNyYywgaXMpOwotICAgIH0KLQotICAgIC8qCi0gICAgcHVibGljIEpwZWdEZWNvZGVyKElucHV0U3RyZWFtIGlTdHJlYW0sIEltYWdlQ29uc3VtZXIgaUNvbnN1bWVyKSB7Ci0gICAgaW5wdXRTdHJlYW0gPSBpU3RyZWFtOwotICAgIGNvbnN1bWVyID0gaUNvbnN1bWVyOwotICAgIH0KLSAgICAqLwotCi0gICAgLyoqCi0gICAgICogQHJldHVybiAtIG5vdCBOVUxMIGlmIGNhbGwgaXMgc3VjY2Vzc2Z1bAotICAgICAqLwotICAgIHByaXZhdGUgbmF0aXZlIE9iamVjdCBkZWNvZGUoCi0gICAgICAgICAgICBieXRlW10gaW5wdXQsCi0gICAgICAgICAgICBpbnQgYnl0ZXNJbkJ1ZmZlciwKLSAgICAgICAgICAgIGxvbmcgaERlY29kZXIpOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgbmF0aXZlIHZvaWQgcmVsZWFzZU5hdGl2ZURlY29kZXIobG9uZyBoRGVjb2Rlcik7Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkZWNvZGVJbWFnZSgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBpbnQgYnl0ZXNSZWFkID0gMCwgZGF0YUxlbmd0aCA9IDA7Ci0gICAgICAgICAgICBib29sZWFuIGVvc1JlYWNoZWQgPSBmYWxzZTsKLSAgICAgICAgICAgIGludCBuZWVkQnl0ZXMsIG9mZnNldCwgYnl0ZXNJbkJ1ZmZlciA9IDA7Ci0gICAgICAgICAgICBieXRlIGJ5dGVPdXRbXSA9IG51bGw7Ci0gICAgICAgICAgICBpbnQgaW50T3V0W10gPSBudWxsOwotICAgICAgICAgICAgLy8gUmVhZCBmcm9tIHRoZSBpbnB1dCBzdHJlYW0KLSAgICAgICAgICAgIGZvciAoOzspIHsKLSAgICAgICAgICAgICAgICBuZWVkQnl0ZXMgPSBCVUZGRVJfU0laRSAtIGJ5dGVzSW5CdWZmZXI7Ci0gICAgICAgICAgICAgICAgb2Zmc2V0ID0gYnl0ZXNJbkJ1ZmZlcjsKLQotICAgICAgICAgICAgICAgIGJ5dGVzUmVhZCA9IGlucHV0U3RyZWFtLnJlYWQoYnVmZmVyLCBvZmZzZXQsIG5lZWRCeXRlcyk7Ci0KLSAgICAgICAgICAgICAgICBpZiAoYnl0ZXNSZWFkIDwgMCkgewotICAgICAgICAgICAgICAgICAgICBieXRlc1JlYWQgPSAwOy8vYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGVvc1JlYWNoZWQgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0gLy8gRG9uJ3QgYnJlYWssIG1heWJlIHNvbWV0aGluZyBsZWZ0IGluIGJ1ZmZlcgotCi0gICAgICAgICAgICAgICAgLy8gS2VlcCB0cmFjayBvbiBob3cgbXVjaCBieXRlcyBsZWZ0IGluIGJ1ZmZlcgotICAgICAgICAgICAgICAgIGJ5dGVzSW5CdWZmZXIgKz0gYnl0ZXNSZWFkOwotCi0gICAgICAgICAgICAgICAgLy8gSGVyZSB3ZSBwYXNzIG92ZXJhbGwgbnVtYmVyIG9mIGJ5dGVzIGxlZnQgaW4gdGhlIGphdmEgYnVmZmVyCi0gICAgICAgICAgICAgICAgLy8gKGJ5dGVzSW5CdWZmZXIpIHNpbmNlIGpwZWcgZGVjb2RlciBoYXMgaXRzIG93biBidWZmZXIgYW5kIGNvbnN1bWVzCi0gICAgICAgICAgICAgICAgLy8gYXMgbWFueSBieXRlcyBhcyBpdCBjYW4uIElmIHRoZXJlIGFyZSBhbnkgdW5jb25zdW1lZCBieXRlcwotICAgICAgICAgICAgICAgIC8vIGl0IGRpZG4ndCBhZGQgdGhlbSB0byBpdHMgYnVmZmVyLi4uCi0gICAgICAgICAgICAgICAgT2JqZWN0IGFyciA9IGRlY29kZSgKLSAgICAgICAgICAgICAgICAgICAgICAgIGJ1ZmZlciwKLSAgICAgICAgICAgICAgICAgICAgICAgIGJ5dGVzSW5CdWZmZXIsCi0gICAgICAgICAgICAgICAgICAgICAgICBoTmF0aXZlRGVjb2Rlcik7Ci0KLSAgICAgICAgICAgICAgICAvLyBLZWVwIHRyYWNrIG9uIGhvdyBtdWNoIGJ5dGVzIGxlZnQgaW4gYnVmZmVyCi0gICAgICAgICAgICAgICAgYnl0ZXNJbkJ1ZmZlciAtPSBieXRlc0NvbnN1bWVkOwotCi0gICAgICAgICAgICAgICAgaWYgKCFoZWFkZXJEb25lICYmIGltYWdlV2lkdGggIT0gLTEpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuSGVhZGVyKCk7Ci0gICAgICAgICAgICAgICAgICAgIGhlYWRlckRvbmUgPSB0cnVlOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChieXRlc0NvbnN1bWVkIDwgMCkgewotICAgICAgICAgICAgICAgICAgICBicmVhazsgLy8gRXJyb3IgZXhpdAotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChhcnIgaW5zdGFuY2VvZiBieXRlW10pIHsKLSAgICAgICAgICAgICAgICAgICAgYnl0ZU91dCA9IChieXRlW10pIGFycjsKLSAgICAgICAgICAgICAgICAgICAgZGF0YUxlbmd0aCA9IGJ5dGVPdXQubGVuZ3RoOwotICAgICAgICAgICAgICAgICAgICByZXR1cm5EYXRhKGJ5dGVPdXQsIGN1cnJTY2FubGluZSk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChhcnIgaW5zdGFuY2VvZiBpbnRbXSkgewotICAgICAgICAgICAgICAgICAgICBpbnRPdXQgPSAoaW50W10pIGFycjsKLSAgICAgICAgICAgICAgICAgICAgZGF0YUxlbmd0aCA9IGludE91dC5sZW5ndGg7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybkRhdGEoaW50T3V0LCBjdXJyU2NhbmxpbmUpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGRhdGFMZW5ndGggPSAwOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChoTmF0aXZlRGVjb2RlciA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChkYXRhTGVuZ3RoID09IDAgJiYgZW9zUmVhY2hlZCkgewotICAgICAgICAgICAgICAgICAgICByZWxlYXNlTmF0aXZlRGVjb2RlcihoTmF0aXZlRGVjb2Rlcik7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOyAvLyBQcm9iYWJseSBpbWFnZSBpcyB0cnVuY2F0ZWQKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbWFnZUNvbXBsZXRlKEltYWdlQ29uc3VtZXIuU1RBVElDSU1BR0VET05FKTsKLSAgICAgICAgfSBjYXRjaCAoSU9FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgdGhyb3cgZTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIGNsb3NlU3RyZWFtKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZXR1cm5IZWFkZXIoKSB7Ci0gICAgICAgIHNldERpbWVuc2lvbnMoaW1hZ2VXaWR0aCwgaW1hZ2VIZWlnaHQpOwotCi0gICAgICAgIHN3aXRjaCAoanBlZ0NvbG9yU3BhY2UpIHsKLSAgICAgICAgICAgIGNhc2UgSkNTX0dSQVlTQ0FMRTogY20gPSBjbUdyYXk7IGJyZWFrOwotICAgICAgICAgICAgY2FzZSBKQ1NfUkdCOiBjbSA9IGNtUkdCOyBicmVhazsKLSAgICAgICAgICAgIGRlZmF1bHQ6IAotICAgICAgICAgICAgICAgIC8vIGF3dC4zRD1Vbmtub3duIGNvbG9yc3BhY2UKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjNEIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgIH0KLSAgICAgICAgc2V0Q29sb3JNb2RlbChjbSk7Ci0KLSAgICAgICAgc2V0SGludHMocHJvZ3Jlc3NpdmUgPyBoaW50ZmxhZ3NQcm9ncmVzc2l2ZSA6IGhpbnRmbGFnc1NpbmdsZSk7Ci0KLSAgICAgICAgc2V0UHJvcGVydGllcyhuZXcgSGFzaHRhYmxlPE9iamVjdCwgT2JqZWN0PigpKTsgLy8gRW1wdHkKLSAgICB9Ci0KLSAgICAvLyBTZW5kIHRoZSBkYXRhIHRvIHRoZSBjb25zdW1lcgotICAgIHB1YmxpYyB2b2lkIHJldHVybkRhdGEoaW50IGRhdGFbXSwgaW50IGN1cnJTY2FuTGluZSkgewotICAgICAgICAvLyBTZW5kIDEgb3IgbW9yZSBzY2FubGluZXMgdG8gdGhlIGNvbnN1bWVyLgotICAgICAgICBpbnQgbnVtU2NhbmxpbmVzID0gZGF0YS5sZW5ndGggLyBpbWFnZVdpZHRoOwotICAgICAgICBpZiAobnVtU2NhbmxpbmVzID4gMCkgewotICAgICAgICAgICAgc2V0UGl4ZWxzKAotICAgICAgICAgICAgICAgICAgICAwLCBjdXJyU2NhbkxpbmUgLSBudW1TY2FubGluZXMsCi0gICAgICAgICAgICAgICAgICAgIGltYWdlV2lkdGgsIG51bVNjYW5saW5lcywKLSAgICAgICAgICAgICAgICAgICAgY20sIGRhdGEsIDAsIGltYWdlV2lkdGgKLSAgICAgICAgICAgICk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCByZXR1cm5EYXRhKGJ5dGUgZGF0YVtdLCBpbnQgY3VyclNjYW5MaW5lKSB7Ci0gICAgICAgIGludCBudW1TY2FubGluZXMgPSBkYXRhLmxlbmd0aCAvIGltYWdlV2lkdGg7Ci0gICAgICAgIGlmIChudW1TY2FubGluZXMgPiAwKSB7Ci0gICAgICAgICAgICBzZXRQaXhlbHMoCi0gICAgICAgICAgICAgICAgICAgIDAsIGN1cnJTY2FuTGluZSAtIG51bVNjYW5saW5lcywKLSAgICAgICAgICAgICAgICAgICAgaW1hZ2VXaWR0aCwgbnVtU2NhbmxpbmVzLAotICAgICAgICAgICAgICAgICAgICBjbSwgZGF0YSwgMCwgaW1hZ2VXaWR0aAotICAgICAgICAgICAgKTsKLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL09mZnNjcmVlbkltYWdlLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9PZmZzY3JlZW5JbWFnZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzNDQ1ZjhlLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL09mZnNjcmVlbkltYWdlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1MzIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi0vKgotICogQ3JlYXRlZCBvbiAyMi4xMi4yMDA0Ci0gKgotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljczsKLWltcG9ydCBqYXZhLmF3dC5JbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29tcG9uZW50Q29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EYXRhQnVmZmVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJCeXRlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJJbnQ7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuRGlyZWN0Q29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbWFnZUNvbnN1bWVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkltYWdlT2JzZXJ2ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW1hZ2VQcm9kdWNlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5JbmRleENvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuV3JpdGFibGVSYXN0ZXI7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2h0YWJsZTsKLWltcG9ydCBqYXZhLnV0aWwuVmVjdG9yOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5JbWFnZVN1cmZhY2U7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLQotLyoqCi0gKiBUaGlzIGNsYXNzIHJlcHJlc2VudCBpbXBsZW1lbnRhdGlvbiBvZiBhYnN0cmFjdCBJbWFnZSBjbGFzcwotICovCi1wdWJsaWMgY2xhc3MgT2Zmc2NyZWVuSW1hZ2UgZXh0ZW5kcyBJbWFnZSBpbXBsZW1lbnRzIEltYWdlQ29uc3VtZXIgewotCi0gICAgc3RhdGljIGZpbmFsIENvbG9yTW9kZWwgcmdiQ00gPSBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKTsKLSAgICBJbWFnZVByb2R1Y2VyIHNyYzsKLSAgICBCdWZmZXJlZEltYWdlIGltYWdlOwotICAgIENvbG9yTW9kZWwgY207Ci0gICAgV3JpdGFibGVSYXN0ZXIgcmFzdGVyOwotICAgIGJvb2xlYW4gaXNJbnRSR0I7Ci0gICAgSGFzaHRhYmxlPD8sID8+IHByb3BlcnRpZXM7Ci0gICAgVmVjdG9yPEltYWdlT2JzZXJ2ZXI+IG9ic2VydmVyczsKLSAgICBpbnQgd2lkdGg7Ci0gICAgaW50IGhlaWdodDsKLSAgICBpbnQgaW1hZ2VTdGF0ZTsKLSAgICBpbnQgaGludHM7Ci0gICAgcHJpdmF0ZSBib29sZWFuIHByb2R1Y2luZzsKLSAgICBwcml2YXRlIEltYWdlU3VyZmFjZSBpbWFnZVN1cmY7Ci0KLSAgICBwdWJsaWMgT2Zmc2NyZWVuSW1hZ2UoSW1hZ2VQcm9kdWNlciBpcCl7Ci0gICAgICAgIGltYWdlU3RhdGUgPSAwOwotICAgICAgICBzcmMgPSBpcDsKLSAgICAgICAgd2lkdGggPSAtMTsKLSAgICAgICAgaGVpZ2h0ID0gLTE7Ci0gICAgICAgIG9ic2VydmVycyA9IG5ldyBWZWN0b3I8SW1hZ2VPYnNlcnZlcj4oKTsKLSAgICAgICAgcHJvZHVjaW5nID0gZmFsc2U7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBnZXRQcm9wZXJ0eShTdHJpbmcgbmFtZSwgSW1hZ2VPYnNlcnZlciBvYnNlcnZlcikgewotICAgICAgICBpZihuYW1lID09IG51bGwpIHsKLSAgICAgICAgICAgIC8vIGF3dC4zOD1Qcm9wZXJ0eSBuYW1lIGlzIG5vdCBkZWZpbmVkCi0gICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMzgiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgfQotICAgICAgICBpZihwcm9wZXJ0aWVzID09IG51bGwpewotICAgICAgICAgICAgYWRkT2JzZXJ2ZXIob2JzZXJ2ZXIpOwotICAgICAgICAgICAgc3RhcnRQcm9kdWN0aW9uKCk7Ci0gICAgICAgICAgICBpZihwcm9wZXJ0aWVzID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBPYmplY3QgcHJvcCA9IHByb3BlcnRpZXMuZ2V0KG5hbWUpOwotICAgICAgICBpZihwcm9wID09IG51bGwpIHsKLSAgICAgICAgICAgIHByb3AgPSBVbmRlZmluZWRQcm9wZXJ0eTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcHJvcDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSW1hZ2VQcm9kdWNlciBnZXRTb3VyY2UoKSB7Ci0gICAgICAgIHJldHVybiBzcmM7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRXaWR0aChJbWFnZU9ic2VydmVyIG9ic2VydmVyKSB7Ci0gICAgICAgIGlmKChpbWFnZVN0YXRlICYgSW1hZ2VPYnNlcnZlci5XSURUSCkgPT0gMCl7Ci0gICAgICAgICAgICBhZGRPYnNlcnZlcihvYnNlcnZlcik7Ci0gICAgICAgICAgICBzdGFydFByb2R1Y3Rpb24oKTsKLSAgICAgICAgICAgIGlmKChpbWFnZVN0YXRlICYgSW1hZ2VPYnNlcnZlci5XSURUSCkgPT0gMCkgewotICAgICAgICAgICAgICAgIHJldHVybiAtMTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gd2lkdGg7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRIZWlnaHQoSW1hZ2VPYnNlcnZlciBvYnNlcnZlcikgewotICAgICAgICBpZigoaW1hZ2VTdGF0ZSAmIEltYWdlT2JzZXJ2ZXIuSEVJR0hUKSA9PSAwKXsKLSAgICAgICAgICAgIGFkZE9ic2VydmVyKG9ic2VydmVyKTsKLSAgICAgICAgICAgIHN0YXJ0UHJvZHVjdGlvbigpOwotICAgICAgICAgICAgaWYoKGltYWdlU3RhdGUgJiBJbWFnZU9ic2VydmVyLkhFSUdIVCkgPT0gMCkgewotICAgICAgICAgICAgICAgIHJldHVybiAtMTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gaGVpZ2h0OwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBHcmFwaGljcyBnZXRHcmFwaGljcygpIHsKLSAgICAgICAgLy8gYXd0LjM5PVRoaXMgbWV0aG9kIGlzIG5vdCBpbXBsZW1lbnRlZCBmb3IgaW1hZ2Ugb2J0YWluZWQgZnJvbSBJbWFnZVByb2R1Y2VyCi0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4zOSIpKTsgLy8kTk9OLU5MUy0xJAotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGZsdXNoKCkgewotICAgICAgICBzdG9wUHJvZHVjdGlvbigpOwotICAgICAgICBpbWFnZVVwZGF0ZSh0aGlzLCBJbWFnZU9ic2VydmVyLkFCT1JULCAtMSwgLTEsIC0xLCAtMSk7Ci0gICAgICAgIGltYWdlU3RhdGUgJj0gfkltYWdlT2JzZXJ2ZXIuRVJST1I7Ci0gICAgICAgIGltYWdlU3RhdGUgPSAwOwotICAgICAgICBpbWFnZSA9IG51bGw7Ci0gICAgICAgIGNtID0gbnVsbDsKLSAgICAgICAgcmFzdGVyID0gbnVsbDsKLSAgICAgICAgaGludHMgPSAwOwotICAgICAgICB3aWR0aCA9IC0xOwotICAgICAgICBoZWlnaHQgPSAtMTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRQcm9wZXJ0aWVzKEhhc2h0YWJsZTw/LCA/PiBwcm9wZXJ0aWVzKSB7Ci0gICAgICAgIHRoaXMucHJvcGVydGllcyA9IHByb3BlcnRpZXM7Ci0gICAgICAgIGltYWdlVXBkYXRlKHRoaXMsIEltYWdlT2JzZXJ2ZXIuUFJPUEVSVElFUywgMCwgMCwgd2lkdGgsIGhlaWdodCk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0Q29sb3JNb2RlbChDb2xvck1vZGVsIGNtKSB7Ci0gICAgICAgIHRoaXMuY20gPSBjbTsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFdlIHN1cHBvc2Ugd2hhdCBpbiBjYXNlIGxvYWRpbmcgSlBFRyBpbWFnZSB0aGVuIGltYWdlIGhhcyBEaXJlY3RDb2xvck1vZGVsCi0gICAgICogYW5kIGZvciBpbmZpbGwgaW1hZ2UgUmFzdGVyIHdpbGwgdXNlIHNldFBpeGVscyBtZXRob2Qgd2l0aCBpbnQgYXJyYXkuCi0gICAgICoKLSAgICAgKiBJbiBjYXNlIGxvYWRpbmcgR0lGIGltYWdlLCBmb3IgcmFzdGVyIGluZmlsbCwgaXMgdXNlZCBzZXRQaXhlbHMgbWV0aG9kIHdpdGgKLSAgICAgKiBieXRlIGFycmF5IGFuZCBDb2xvciBNb2RlbCBpcyBJbmRleENvbG9yTW9kZWwuIEJ1dCBDb2xvciBNb2RlbCBtYXkKLSAgICAgKiBiZSBjaGFuZ2VkIGR1cmluZyB0aGlzIHByb2Nlc3MuIFRoZW4gaXMgY2FsbGVkIHNldFBpeGVscyBtZXRob2Qgd2l0aAotICAgICAqIGludCBhcnJheSBhbmQgaW1hZ2UgZm9yY2UgdG8gZGVmYXVsdCBjb2xvciBtb2RlbCAtIGludCBBUkdCLiBUaGUgcmVzdAotICAgICAqIHBpeGVscyBhcmUgc2VuZGluZyBpbiBEaXJlY3RDb2xvck1vZGVsLgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgQ29sb3JNb2RlbCBtb2RlbCwKLSAgICAgICAgICAgIGludFtdIHBpeGVscywgaW50IG9mZiwgaW50IHNjYW5zaXplKSB7Ci0gICAgICAgIGlmKHJhc3RlciA9PSBudWxsKXsKLSAgICAgICAgICAgIGlmKGNtID09IG51bGwpewotICAgICAgICAgICAgICAgIGlmKG1vZGVsID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjNBPUNvbG9yIE1vZGVsIGlzIG51bGwKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjNBIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGNtID0gbW9kZWw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjcmVhdGVSYXN0ZXIoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmKG1vZGVsID09IG51bGwpIHsKLSAgICAgICAgICAgIG1vZGVsID0gY207Ci0gICAgICAgIH0KLSAgICAgICAgaWYoY20gIT0gbW9kZWwpewotICAgICAgICAgICAgZm9yY2VUb0ludEFSR0IoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmKGNtID09IG1vZGVsICYmIG1vZGVsLmdldFRyYW5zZmVyVHlwZSgpID09IERhdGFCdWZmZXIuVFlQRV9JTlQgJiYKLSAgICAgICAgICAgICAgICByYXN0ZXIuZ2V0TnVtRGF0YUVsZW1lbnRzKCkgPT0gMSl7Ci0KLSAgICAgICAgICAgIERhdGFCdWZmZXJJbnQgZGJpID0gKERhdGFCdWZmZXJJbnQpIHJhc3Rlci5nZXREYXRhQnVmZmVyKCk7Ci0gICAgICAgICAgICBpbnQgZGF0YVtdID0gZGJpLmdldERhdGEoKTsKLSAgICAgICAgICAgIGludCBzY2FubGluZSA9IHJhc3Rlci5nZXRXaWR0aCgpOwotICAgICAgICAgICAgaW50IHJvZiA9IGRiaS5nZXRPZmZzZXQoKSArIHkgKiBzY2FubGluZSArIHg7Ci0gICAgICAgICAgICBmb3IoaW50IGxpbmVPZmYgPSBvZmYsIGxpbmUgPSB5OyBsaW5lIDwgeSArIGg7Ci0gICAgICAgICAgICAgICAgbGluZSsrLCBsaW5lT2ZmICs9IHNjYW5zaXplLCByb2YgKz0gc2NhbmxpbmUpewotCi0gICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShwaXhlbHMsIGxpbmVPZmYsIGRhdGEsIHJvZiwgdyk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgfWVsc2UgaWYoaXNJbnRSR0IpewotICAgICAgICAgICAgaW50IGJ1ZmZbXSA9IG5ldyBpbnRbd107Ci0gICAgICAgICAgICBEYXRhQnVmZmVySW50IGRiaSA9IChEYXRhQnVmZmVySW50KSByYXN0ZXIuZ2V0RGF0YUJ1ZmZlcigpOwotICAgICAgICAgICAgaW50IGRhdGFbXSA9IGRiaS5nZXREYXRhKCk7Ci0gICAgICAgICAgICBpbnQgc2NhbmxpbmUgPSByYXN0ZXIuZ2V0V2lkdGgoKTsKLSAgICAgICAgICAgIGludCByb2YgPSBkYmkuZ2V0T2Zmc2V0KCkgKyB5ICogc2NhbmxpbmUgKyB4OwotICAgICAgICAgICAgZm9yIChpbnQgc3kgPSB5LCBzT2ZmID0gb2ZmOyBzeSA8IHkgKyBoOyBzeSsrLCBzT2ZmICs9IHNjYW5zaXplLAotICAgICAgICAgICAgICAgIHJvZiArPSBzY2FubGluZSkgewotCi0gICAgICAgICAgICAgICAgZm9yIChpbnQgc3ggPSB4LCBpZHggPSAwOyBzeCA8IHggKyB3OyBzeCsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICBidWZmW2lkeF0gPSBtb2RlbC5nZXRSR0IocGl4ZWxzW3NPZmYgKyBpZHhdKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShidWZmLCAwLCBkYXRhLCByb2YsIHcpOwotICAgICAgICAgICAgfQotICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgIE9iamVjdCBidWYgPSBudWxsOwotICAgICAgICAgICAgZm9yIChpbnQgc3kgPSB5LCBzT2ZmID0gb2ZmOyBzeSA8IHkgKyBoOyBzeSsrLCBzT2ZmICs9IHNjYW5zaXplKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgc3ggPSB4LCBpZHggPSAwOyBzeCA8IHggKyB3OyBzeCsrLCBpZHgrKykgewotICAgICAgICAgICAgICAgICAgICBpbnQgcmdiID0gbW9kZWwuZ2V0UkdCKHBpeGVsc1tzT2ZmICsgaWR4XSk7Ci0gICAgICAgICAgICAgICAgICAgIGJ1ZiA9IGNtLmdldERhdGFFbGVtZW50cyhyZ2IsIGJ1Zik7Ci0gICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXREYXRhRWxlbWVudHMoc3gsIHN5LCBidWYpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChpbWFnZVN1cmYgIT0gbnVsbCkgewotICAgICAgICAgICAgaW1hZ2VTdXJmLmludmFsaWRhdGUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGltYWdlVXBkYXRlKHRoaXMsIEltYWdlT2JzZXJ2ZXIuU09NRUJJVFMsIDAsIDAsIHdpZHRoLCBoZWlnaHQpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgQ29sb3JNb2RlbCBtb2RlbCwKLSAgICAgICAgICAgIGJ5dGVbXSBwaXhlbHMsIGludCBvZmYsIGludCBzY2Fuc2l6ZSkgewotCi0gICAgICAgIGlmKHJhc3RlciA9PSBudWxsKXsKLSAgICAgICAgICAgIGlmKGNtID09IG51bGwpewotICAgICAgICAgICAgICAgIGlmKG1vZGVsID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjNBPUNvbG9yIE1vZGVsIGlzIG51bGwKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IE51bGxQb2ludGVyRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjNBIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGNtID0gbW9kZWw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjcmVhdGVSYXN0ZXIoKTsKLSAgICAgICAgfQotICAgICAgICBpZihtb2RlbCA9PSBudWxsKSB7Ci0gICAgICAgICAgICBtb2RlbCA9IGNtOwotICAgICAgICB9Ci0gICAgICAgIGlmKG1vZGVsICE9IGNtKXsKLSAgICAgICAgICAgIGZvcmNlVG9JbnRBUkdCKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZihpc0ludFJHQil7Ci0gICAgICAgICAgICBpbnQgYnVmZltdID0gbmV3IGludFt3XTsKLSAgICAgICAgICAgIEluZGV4Q29sb3JNb2RlbCBpY20gPSAoSW5kZXhDb2xvck1vZGVsKSBtb2RlbDsKLSAgICAgICAgICAgIGludCBjb2xvck1hcFtdID0gbmV3IGludFtpY20uZ2V0TWFwU2l6ZSgpXTsKLSAgICAgICAgICAgIGljbS5nZXRSR0JzKGNvbG9yTWFwKTsKLSAgICAgICAgICAgIERhdGFCdWZmZXJJbnQgZGJpID0gKERhdGFCdWZmZXJJbnQpIHJhc3Rlci5nZXREYXRhQnVmZmVyKCk7Ci0gICAgICAgICAgICBpbnQgZGF0YVtdID0gZGJpLmdldERhdGEoKTsKLSAgICAgICAgICAgIGludCBzY2FubGluZSA9IHJhc3Rlci5nZXRXaWR0aCgpOwotICAgICAgICAgICAgaW50IHJvZiA9IGRiaS5nZXRPZmZzZXQoKSArIHkgKiBzY2FubGluZSArIHg7Ci0gICAgICAgICAgICBpZihtb2RlbCBpbnN0YW5jZW9mIEluZGV4Q29sb3JNb2RlbCl7Ci0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBzeSA9IHksIHNPZmYgPSBvZmY7IHN5IDwgeSArIGg7IHN5KyssIHNPZmYgKz0gc2NhbnNpemUsCi0gICAgICAgICAgICAgICAgICAgIHJvZiArPSBzY2FubGluZSkgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBzeCA9IHgsIGlkeCA9IDA7IHN4IDwgeCArIHc7IHN4KyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBidWZmW2lkeF0gPSBjb2xvck1hcFtwaXhlbHNbc09mZiArIGlkeF0gJiAweGZmXTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGJ1ZmYsIDAsIGRhdGEsIHJvZiwgdyk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfWVsc2V7Ci0KLSAgICAgICAgICAgICAgICBmb3IgKGludCBzeSA9IHksIHNPZmYgPSBvZmY7IHN5IDwgeSArIGg7IHN5KyssIHNPZmYgKz0gc2NhbnNpemUsCi0gICAgICAgICAgICAgICAgICAgIHJvZiArPSBzY2FubGluZSkgewotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBzeCA9IHgsIGlkeCA9IDA7IHN4IDwgeCArIHc7IHN4KyssIGlkeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBidWZmW2lkeF0gPSBtb2RlbC5nZXRSR0IocGl4ZWxzW3NPZmYgKyBpZHhdICYgMHhmZik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShidWZmLCAwLCBkYXRhLCByb2YsIHcpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfWVsc2UgaWYobW9kZWwgPT0gY20gJiYgbW9kZWwuZ2V0VHJhbnNmZXJUeXBlKCkgPT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYKLSAgICAgICAgICAgICAgICByYXN0ZXIuZ2V0TnVtRGF0YUVsZW1lbnRzKCkgPT0gMSl7Ci0KLSAgICAgICAgICAgIERhdGFCdWZmZXJCeXRlIGRiYiA9IChEYXRhQnVmZmVyQnl0ZSlyYXN0ZXIuZ2V0RGF0YUJ1ZmZlcigpOwotICAgICAgICAgICAgYnl0ZSBkYXRhW10gPSBkYmIuZ2V0RGF0YSgpOwotICAgICAgICAgICAgaW50IHNjYW5saW5lID0gcmFzdGVyLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgcm9mID0gZGJiLmdldE9mZnNldCgpICsgeSAqIHNjYW5saW5lICsgeDsKLSAgICAgICAgICAgIGZvcihpbnQgbGluZU9mZiA9IG9mZiwgbGluZSA9IHk7IGxpbmUgPCB5ICsgaDsKLSAgICAgICAgICAgICAgICBsaW5lKyssIGxpbmVPZmYgKz0gc2NhbnNpemUsIHJvZiArPSBzY2FubGluZSl7Ci0gICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShwaXhlbHMsIGxpbmVPZmYsIGRhdGEsIHJvZiwgdyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIC8vIEJFR0lOIGFuZHJvaWQtYWRkZWQgKHRha2VuIGZyb20gbmV3ZXIgSGFybW9ueSkKLSAgICAgICAgfWVsc2UgaWYobW9kZWwgPT0gY20gJiYgbW9kZWwuZ2V0VHJhbnNmZXJUeXBlKCkgPT0gRGF0YUJ1ZmZlci5UWVBFX0JZVEUgJiYKLSAgICAgICAgICAgICAgICBjbSBpbnN0YW5jZW9mIENvbXBvbmVudENvbG9yTW9kZWwpewotCi0gICAgICAgICAgICBpbnQgbmMgPSBjbS5nZXROdW1Db21wb25lbnRzKCk7Ci0gICAgICAgICAgICBieXRlIHN0cmlkZVtdID0gbmV3IGJ5dGVbc2NhbnNpemVdOwotICAgICAgICAgICAgZm9yIChpbnQgc3kgPSB5LCBzT2ZmID0gb2ZmOyBzeSA8IHkgKyBoOyBzeSsrLCBzT2ZmICs9IHNjYW5zaXplKSB7Ci0gICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShwaXhlbHMsIHNPZmYsIHN0cmlkZSwgMCwgc2NhbnNpemUpOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIHJhc3Rlci5zZXREYXRhRWxlbWVudHMoeCwgc3ksIHcsIDEsIHN0cmlkZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIC8vIEVORCBhbmRyb2lkLWFkZGVkCi0gICAgICAgIH1lbHNlIHsKLSAgICAgICAgICAgIGZvciAoaW50IHN5ID0geSwgc09mZiA9IG9mZjsgc3kgPCB5ICsgaDsgc3krKywgc09mZiArPSBzY2Fuc2l6ZSkgewotICAgICAgICAgICAgICAgIGZvciAoaW50IHN4ID0geCwgaWR4ID0gMDsgc3ggPCB4ICsgdzsgc3grKywgaWR4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IHJnYiA9IG1vZGVsLmdldFJHQihwaXhlbHNbc09mZiArIGlkeF0gJiAweGZmKTsKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyLnNldERhdGFFbGVtZW50cyhzeCwgc3ksIGNtLmdldERhdGFFbGVtZW50cyhyZ2IsIG51bGwpKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoaW1hZ2VTdXJmICE9IG51bGwpIHsKLSAgICAgICAgICAgIGltYWdlU3VyZi5pbnZhbGlkYXRlKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbWFnZVVwZGF0ZSh0aGlzLCBJbWFnZU9ic2VydmVyLlNPTUVCSVRTLCAwLCAwLCB3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXREaW1lbnNpb25zKGludCB3aWR0aCwgaW50IGhlaWdodCkgewotICAgICAgICBpZih3aWR0aCA8PSAwIHx8IGhlaWdodCA8PSAwKXsKLSAgICAgICAgICAgIGltYWdlQ29tcGxldGUoSW1hZ2VPYnNlcnZlci5FUlJPUik7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICB0aGlzLndpZHRoID0gd2lkdGg7Ci0gICAgICAgIHRoaXMuaGVpZ2h0ID0gaGVpZ2h0OwotICAgICAgICBpbWFnZVVwZGF0ZSh0aGlzLCAoSW1hZ2VPYnNlcnZlci5IRUlHSFQgfCBJbWFnZU9ic2VydmVyLldJRFRIKSwKLSAgICAgICAgICAgICAgICAwLCAwLCB3aWR0aCwgaGVpZ2h0KTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRIaW50cyhpbnQgaGludHMpIHsKLSAgICAgICAgdGhpcy5oaW50cyA9IGhpbnRzOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGltYWdlQ29tcGxldGUoaW50IHN0YXRlKSB7Ci0gICAgICAgIGludCBmbGFnOwotICAgICAgICBzd2l0Y2goc3RhdGUpewotICAgICAgICBjYXNlIElNQUdFQUJPUlRFRDoKLSAgICAgICAgICAgIGZsYWcgPSBJbWFnZU9ic2VydmVyLkFCT1JUOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgSU1BR0VFUlJPUjoKLSAgICAgICAgICAgIGZsYWcgPSBJbWFnZU9ic2VydmVyLkVSUk9SIHwgSW1hZ2VPYnNlcnZlci5BQk9SVDsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFNJTkdMRUZSQU1FRE9ORToKLSAgICAgICAgICAgIGZsYWcgPSBJbWFnZU9ic2VydmVyLkZSQU1FQklUUzsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFNUQVRJQ0lNQUdFRE9ORToKLSAgICAgICAgICAgIGZsYWcgPSBJbWFnZU9ic2VydmVyLkFMTEJJVFM7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIC8vIGF3dC4zQj1JbmNvcnJlY3QgSW1hZ2VDb25zdW1lciBjb21wbGV0aW9uIHN0YXR1cwotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4zQiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0gICAgICAgIGltYWdlVXBkYXRlKHRoaXMsIGZsYWcsIDAsIDAsIHdpZHRoLCBoZWlnaHQpOwotCi0gICAgICAgIGlmKChmbGFnICYgKEltYWdlT2JzZXJ2ZXIuRVJST1IgfCBJbWFnZU9ic2VydmVyLkFCT1JUIHwKLSAgICAgICAgICAgICAgICBJbWFnZU9ic2VydmVyLkFMTEJJVFMpKSAhPSAwICkgewotICAgICAgICAgICAgc3RvcFByb2R1Y3Rpb24oKTsKLSAgICAgICAgICAgIG9ic2VydmVycy5yZW1vdmVBbGxFbGVtZW50cygpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIC8qc3luY2hyb25pemVkKi8gQnVmZmVyZWRJbWFnZSBnZXRCdWZmZXJlZEltYWdlKCl7Ci0gICAgICAgIGlmKGltYWdlID09IG51bGwpewotICAgICAgICAgICAgQ29sb3JNb2RlbCBtb2RlbCA9IGdldENvbG9yTW9kZWwoKTsKLSAgICAgICAgICAgIFdyaXRhYmxlUmFzdGVyIHdyID0gZ2V0UmFzdGVyKCk7Ci0gICAgICAgICAgICBpZihtb2RlbCAhPSBudWxsICYmIHdyICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBpbWFnZSA9IG5ldyBCdWZmZXJlZEltYWdlKG1vZGVsLCB3ciwgbW9kZWwuaXNBbHBoYVByZW11bHRpcGxpZWQoKSwgbnVsbCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGltYWdlOwotICAgIH0KLQotICAgIHB1YmxpYyAvKnN5bmNocm9uaXplZCovIGludCBjaGVja0ltYWdlKEltYWdlT2JzZXJ2ZXIgb2JzZXJ2ZXIpewotICAgICAgICBhZGRPYnNlcnZlcihvYnNlcnZlcik7Ci0gICAgICAgIHJldHVybiBpbWFnZVN0YXRlOwotICAgIH0KLQotICAgIHB1YmxpYyAvKnN5bmNocm9uaXplZCovIGJvb2xlYW4gcHJlcGFyZUltYWdlKEltYWdlT2JzZXJ2ZXIgb2JzZXJ2ZXIpewotICAgICAgICBpZigoaW1hZ2VTdGF0ZSAmIEltYWdlT2JzZXJ2ZXIuRVJST1IpICE9IDApewotICAgICAgICAgICAgaWYob2JzZXJ2ZXIgIT0gbnVsbCl7Ci0gICAgICAgICAgICAgICAgb2JzZXJ2ZXIuaW1hZ2VVcGRhdGUodGhpcywgSW1hZ2VPYnNlcnZlci5FUlJPUiB8Ci0gICAgICAgICAgICAgICAgICAgICAgICBJbWFnZU9ic2VydmVyLkFCT1JULCAtMSwgLTEsIC0xLCAtMSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgaWYoKGltYWdlU3RhdGUgJiBJbWFnZU9ic2VydmVyLkFMTEJJVFMpICE9IDApIHsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgICAgIGFkZE9ic2VydmVyKG9ic2VydmVyKTsKLSAgICAgICAgc3RhcnRQcm9kdWN0aW9uKCk7Ci0gICAgICAgIHJldHVybiAoKGltYWdlU3RhdGUgJiBJbWFnZU9ic2VydmVyLkFMTEJJVFMpICE9IDApOwotICAgIH0KLQotICAgIHB1YmxpYyAvKnN5bmNocm9uaXplZCovIENvbG9yTW9kZWwgZ2V0Q29sb3JNb2RlbCgpewotICAgICAgICBpZihjbSA9PSBudWxsKSB7Ci0gICAgICAgICAgICBzdGFydFByb2R1Y3Rpb24oKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gY207Ci0gICAgfQotCi0gICAgcHVibGljIC8qc3luY2hyb25pemVkKi8gV3JpdGFibGVSYXN0ZXIgZ2V0UmFzdGVyKCl7Ci0gICAgICAgIGlmKHJhc3RlciA9PSBudWxsKSB7Ci0gICAgICAgICAgICBzdGFydFByb2R1Y3Rpb24oKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcmFzdGVyOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0U3RhdGUoKXsKLSAgICAgICAgcmV0dXJuIGltYWdlU3RhdGU7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSAvKnN5bmNocm9uaXplZCovIHZvaWQgYWRkT2JzZXJ2ZXIoSW1hZ2VPYnNlcnZlciBvYnNlcnZlcil7Ci0gICAgICAgIGlmKG9ic2VydmVyICE9IG51bGwpewotICAgICAgICAgIGlmKG9ic2VydmVycy5jb250YWlucyhvYnNlcnZlcikpIHsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICAgIGlmKChpbWFnZVN0YXRlICYgSW1hZ2VPYnNlcnZlci5FUlJPUikgIT0gMCl7Ci0gICAgICAgICAgICAgIG9ic2VydmVyLmltYWdlVXBkYXRlKHRoaXMsIEltYWdlT2JzZXJ2ZXIuRVJST1IgfAotICAgICAgICAgICAgICAgICAgICAgIEltYWdlT2JzZXJ2ZXIuQUJPUlQsIC0xLCAtMSwgLTEsIC0xKTsKLSAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgIH0KLSAgICAgICAgICBpZigoaW1hZ2VTdGF0ZSAmIEltYWdlT2JzZXJ2ZXIuQUxMQklUUykgIT0gMCl7Ci0gICAgICAgICAgICAgIG9ic2VydmVyLmltYWdlVXBkYXRlKHRoaXMsIGltYWdlU3RhdGUsIDAsIDAsIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgfQotICAgICAgICAgIG9ic2VydmVycy5hZGRFbGVtZW50KG9ic2VydmVyKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHByaXZhdGUgc3luY2hyb25pemVkIHZvaWQgc3RhcnRQcm9kdWN0aW9uKCl7Ci0gICAgICAgIGlmKCFwcm9kdWNpbmcpewotICAgICAgICAgICAgaW1hZ2VTdGF0ZSAmPSB+SW1hZ2VPYnNlcnZlci5BQk9SVDsKLSAgICAgICAgICAgIHByb2R1Y2luZyA9IHRydWU7Ci0gICAgICAgICAgICBzcmMuc3RhcnRQcm9kdWN0aW9uKHRoaXMpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBzeW5jaHJvbml6ZWQgdm9pZCBzdG9wUHJvZHVjdGlvbigpewotICAgICAgICBwcm9kdWNpbmcgPSBmYWxzZTsKLSAgICAgICAgc3JjLnJlbW92ZUNvbnN1bWVyKHRoaXMpOwotICAgIH0KLQotICAgIHByaXZhdGUgdm9pZCBjcmVhdGVSYXN0ZXIoKXsKLSAgICAgICAgdHJ5ewotICAgICAgICAgICAgcmFzdGVyID0gY20uY3JlYXRlQ29tcGF0aWJsZVdyaXRhYmxlUmFzdGVyKHdpZHRoLCBoZWlnaHQpOwotICAgICAgICAgICAgaXNJbnRSR0IgPSBmYWxzZTsKLSAgICAgICAgICAgIGlmKGNtIGluc3RhbmNlb2YgRGlyZWN0Q29sb3JNb2RlbCl7Ci0gICAgICAgICAgICAgICAgRGlyZWN0Q29sb3JNb2RlbCBkY20gPSAoRGlyZWN0Q29sb3JNb2RlbCkgY207Ci0gICAgICAgICAgICAgICAgaWYoZGNtLmdldFRyYW5zZmVyVHlwZSgpID09IERhdGFCdWZmZXIuVFlQRV9JTlQgJiYKLSAgICAgICAgICAgICAgICAgICAgICAgIGRjbS5nZXRSZWRNYXNrKCkgPT0gMHhmZjAwMDAgJiYKLSAgICAgICAgICAgICAgICAgICAgICAgIGRjbS5nZXRHcmVlbk1hc2soKSA9PSAweGZmMDAgJiYKLSAgICAgICAgICAgICAgICAgICAgICAgIGRjbS5nZXRCbHVlTWFzaygpID09IDB4ZmYpewotICAgICAgICAgICAgICAgICAgICBpc0ludFJHQiA9IHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Y2F0Y2goRXhjZXB0aW9uIGUpewotICAgICAgICAgICAgY20gPSBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKTsKLSAgICAgICAgICAgIHJhc3RlciA9IGNtLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3Rlcih3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgICAgIGlzSW50UkdCID0gdHJ1ZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHByaXZhdGUgLypzeW5jaHJvbml6ZWQqLyB2b2lkIGltYWdlVXBkYXRlKEltYWdlIGltZywgaW50IGluZm9mbGFncywgaW50IHgsIGludCB5LAotICAgICAgICAgICAgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KXsKLQotICAgICAgICBpbWFnZVN0YXRlIHw9IGluZm9mbGFnczsKLSAgICAgICAgZm9yIChJbWFnZU9ic2VydmVyIG9ic2VydmVyIDogb2JzZXJ2ZXJzKSB7Ci0gICAgICAgICAgICBvYnNlcnZlci5pbWFnZVVwZGF0ZSh0aGlzLCBpbmZvZmxhZ3MsIHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgICAgICB9Ci0KLS8vICAgICAgICAgICAgbm90aWZ5QWxsKCk7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIGZvcmNlVG9JbnRBUkdCKCl7Ci0KLSAgICAgICAgaW50IHcgPSByYXN0ZXIuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IGggPSByYXN0ZXIuZ2V0SGVpZ2h0KCk7Ci0KLSAgICAgICAgV3JpdGFibGVSYXN0ZXIgZGVzdFJhc3RlciA9IHJnYkNNLmNyZWF0ZUNvbXBhdGlibGVXcml0YWJsZVJhc3Rlcih3LCBoKTsKLQotICAgICAgICBPYmplY3Qgb2JqID0gbnVsbDsKLSAgICAgICAgaW50IHBpeGVsc1tdID0gbmV3IGludFt3XTsKLQotICAgICAgICBpZihjbSBpbnN0YW5jZW9mIEluZGV4Q29sb3JNb2RlbCl7Ci0gICAgICAgICAgICBJbmRleENvbG9yTW9kZWwgaWNtID0gKEluZGV4Q29sb3JNb2RlbCkgY207Ci0gICAgICAgICAgICBpbnQgY29sb3JNYXBbXSA9IG5ldyBpbnRbaWNtLmdldE1hcFNpemUoKV07Ci0gICAgICAgICAgICBpY20uZ2V0UkdCcyhjb2xvck1hcCk7Ci0KLSAgICAgICAgICAgIGZvciAoaW50IHkgPSAwOyB5IDwgaDsgeSsrKSB7Ci0gICAgICAgICAgICAgICAgb2JqID0gcmFzdGVyLmdldERhdGFFbGVtZW50cygwLCB5LCB3LCAxLCBvYmopOwotICAgICAgICAgICAgICAgIGJ5dGUgYmFbXSA9IChieXRlW10pIG9iajsKLSAgICAgICAgICAgICAgICBmb3IgKGludCB4ID0gMDsgeCA8IGJhLmxlbmd0aDsgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHBpeGVsc1t4XSA9IGNvbG9yTWFwW2JhW3hdICYgMHhmZl07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGRlc3RSYXN0ZXIuc2V0RGF0YUVsZW1lbnRzKDAsIHksIHcsIDEsIHBpeGVscyk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICBmb3IoaW50IHkgPSAwOyB5IDwgaDsgeSsrKXsKLSAgICAgICAgICAgICAgICBmb3IoaW50IHggPSAwOyB4IDwgdzsgeCsrKXsKLSAgICAgICAgICAgICAgICAgICAgb2JqID0gcmFzdGVyLmdldERhdGFFbGVtZW50cyh4LCB5LCBvYmopOwotICAgICAgICAgICAgICAgICAgICBwaXhlbHNbeF0gPSBjbS5nZXRSR0Iob2JqKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgZGVzdFJhc3Rlci5zZXREYXRhRWxlbWVudHMoMCwgeSwgdywgMSwgcGl4ZWxzKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHN5bmNocm9uaXplZCh0aGlzKXsKLSAgICAgICAgICAgIGlmKGltYWdlU3VyZiAhPSBudWxsKXsKLSAgICAgICAgICAgICAgICBpbWFnZVN1cmYuZGlzcG9zZSgpOwotICAgICAgICAgICAgICAgIGltYWdlU3VyZiA9IG51bGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZihpbWFnZSAhPSBudWxsKXsKLSAgICAgICAgICAgICAgICBpbWFnZS5mbHVzaCgpOwotICAgICAgICAgICAgICAgIGltYWdlID0gbnVsbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNtID0gcmdiQ007Ci0gICAgICAgICAgICByYXN0ZXIgPSBkZXN0UmFzdGVyOwotICAgICAgICAgICAgaXNJbnRSR0IgPSB0cnVlOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIEltYWdlU3VyZmFjZSBnZXRJbWFnZVN1cmZhY2UoKSB7Ci0gICAgICAgIGlmIChpbWFnZVN1cmYgPT0gbnVsbCkgewotICAgICAgICAgICAgQ29sb3JNb2RlbCBtb2RlbCA9IGdldENvbG9yTW9kZWwoKTsKLSAgICAgICAgICAgIFdyaXRhYmxlUmFzdGVyIHdyID0gZ2V0UmFzdGVyKCk7Ci0gICAgICAgICAgICBpZihtb2RlbCAhPSBudWxsICYmIHdyICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBpbWFnZVN1cmYgPSBuZXcgSW1hZ2VTdXJmYWNlKG1vZGVsLCB3cik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGltYWdlU3VyZjsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9PcmRpbmFyeVdyaXRhYmxlUmFzdGVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9PcmRpbmFyeVdyaXRhYmxlUmFzdGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDE3NDhlMWIuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvT3JkaW5hcnlXcml0YWJsZVJhc3Rlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTUzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotLyoKLSAqIENyZWF0ZWQgb24gMzAuMDkuMjAwNAotICoKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmltYWdlOwotCi1pbXBvcnQgamF2YS5hd3QuUG9pbnQ7Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmFzdGVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlNhbXBsZU1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLldyaXRhYmxlUmFzdGVyOwotCi1wdWJsaWMgY2xhc3MgT3JkaW5hcnlXcml0YWJsZVJhc3RlciBleHRlbmRzIFdyaXRhYmxlUmFzdGVyIHsKLQotICAgIHB1YmxpYyBPcmRpbmFyeVdyaXRhYmxlUmFzdGVyKFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsLAotICAgICAgICAgICAgRGF0YUJ1ZmZlciBkYXRhQnVmZmVyLCBSZWN0YW5nbGUgYVJlZ2lvbiwKLSAgICAgICAgICAgIFBvaW50IHNhbXBsZU1vZGVsVHJhbnNsYXRlLCBXcml0YWJsZVJhc3RlciBwYXJlbnQpIHsKLSAgICAgICAgc3VwZXIoc2FtcGxlTW9kZWwsIGRhdGFCdWZmZXIsIGFSZWdpb24sIHNhbXBsZU1vZGVsVHJhbnNsYXRlLCBwYXJlbnQpOwotICAgIH0KLQotICAgIHB1YmxpYyBPcmRpbmFyeVdyaXRhYmxlUmFzdGVyKFNhbXBsZU1vZGVsIHNhbXBsZU1vZGVsLAotICAgICAgICAgICAgRGF0YUJ1ZmZlciBkYXRhQnVmZmVyLCBQb2ludCBvcmlnaW4pIHsKLSAgICAgICAgc3VwZXIoc2FtcGxlTW9kZWwsIGRhdGFCdWZmZXIsIG9yaWdpbik7Ci0gICAgfQotCi0gICAgcHVibGljIE9yZGluYXJ5V3JpdGFibGVSYXN0ZXIoU2FtcGxlTW9kZWwgc2FtcGxlTW9kZWwsIFBvaW50IG9yaWdpbikgewotICAgICAgICBzdXBlcihzYW1wbGVNb2RlbCwgb3JpZ2luKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXREYXRhRWxlbWVudHMoaW50IHgsIGludCB5LCBPYmplY3QgaW5EYXRhKSB7Ci0gICAgICAgIHN1cGVyLnNldERhdGFFbGVtZW50cyh4LCB5LCBpbkRhdGEpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldERhdGFFbGVtZW50cyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgT2JqZWN0IGluRGF0YSkgewotICAgICAgICBzdXBlci5zZXREYXRhRWxlbWVudHMoeCwgeSwgdywgaCwgaW5EYXRhKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlV3JpdGFibGVDaGlsZChpbnQgcGFyZW50WCwgaW50IHBhcmVudFksIGludCB3LAotICAgICAgICAgICAgaW50IGgsIGludCBjaGlsZE1pblgsIGludCBjaGlsZE1pblksIGludFtdIGJhbmRMaXN0KSB7Ci0gICAgICAgIHJldHVybiBzdXBlci5jcmVhdGVXcml0YWJsZUNoaWxkKHBhcmVudFgsIHBhcmVudFksIHcsIGgsIGNoaWxkTWluWCwKLSAgICAgICAgICAgICAgICBjaGlsZE1pblksIGJhbmRMaXN0KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgY3JlYXRlV3JpdGFibGVUcmFuc2xhdGVkQ2hpbGQoaW50IGNoaWxkTWluWCwKLSAgICAgICAgICAgIGludCBjaGlsZE1pblkpIHsKLSAgICAgICAgcmV0dXJuIHN1cGVyLmNyZWF0ZVdyaXRhYmxlVHJhbnNsYXRlZENoaWxkKGNoaWxkTWluWCwgY2hpbGRNaW5ZKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgV3JpdGFibGVSYXN0ZXIgZ2V0V3JpdGFibGVQYXJlbnQoKSB7Ci0gICAgICAgIHJldHVybiBzdXBlci5nZXRXcml0YWJsZVBhcmVudCgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFJlY3QoUmFzdGVyIHNyY1Jhc3RlcikgewotICAgICAgICBzdXBlci5zZXRSZWN0KHNyY1Jhc3Rlcik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UmVjdChpbnQgZHgsIGludCBkeSwgUmFzdGVyIHNyY1Jhc3RlcikgewotICAgICAgICBzdXBlci5zZXRSZWN0KGR4LCBkeSwgc3JjUmFzdGVyKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXREYXRhRWxlbWVudHMoaW50IHgsIGludCB5LCBSYXN0ZXIgaW5SYXN0ZXIpIHsKLSAgICAgICAgc3VwZXIuc2V0RGF0YUVsZW1lbnRzKHgsIHksIGluUmFzdGVyKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbChpbnQgeCwgaW50IHksIGludFtdIGlBcnJheSkgewotICAgICAgICBzdXBlci5zZXRQaXhlbCh4LCB5LCBpQXJyYXkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVsKGludCB4LCBpbnQgeSwgZmxvYXRbXSBmQXJyYXkpIHsKLSAgICAgICAgc3VwZXIuc2V0UGl4ZWwoeCwgeSwgZkFycmF5KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRQaXhlbChpbnQgeCwgaW50IHksIGRvdWJsZVtdIGRBcnJheSkgewotICAgICAgICBzdXBlci5zZXRQaXhlbCh4LCB5LCBkQXJyYXkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50W10gaUFycmF5KSB7Ci0gICAgICAgIHN1cGVyLnNldFBpeGVscyh4LCB5LCB3LCBoLCBpQXJyYXkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFBpeGVscyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgZmxvYXRbXSBmQXJyYXkpIHsKLSAgICAgICAgc3VwZXIuc2V0UGl4ZWxzKHgsIHksIHcsIGgsIGZBcnJheSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0UGl4ZWxzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBkb3VibGVbXSBkQXJyYXkpIHsKLSAgICAgICAgc3VwZXIuc2V0UGl4ZWxzKHgsIHksIHcsIGgsIGRBcnJheSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0U2FtcGxlcyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGIsIGludFtdIGlBcnJheSkgewotICAgICAgICBzdXBlci5zZXRTYW1wbGVzKHgsIHksIHcsIGgsIGIsIGlBcnJheSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0U2FtcGxlcyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGIsIGZsb2F0W10gZkFycmF5KSB7Ci0gICAgICAgIHN1cGVyLnNldFNhbXBsZXMoeCwgeSwgdywgaCwgYiwgZkFycmF5KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGVzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoLCBpbnQgYiwgZG91YmxlW10gZEFycmF5KSB7Ci0gICAgICAgIHN1cGVyLnNldFNhbXBsZXMoeCwgeSwgdywgaCwgYiwgZEFycmF5KTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgaW50IHMpIHsKLSAgICAgICAgc3VwZXIuc2V0U2FtcGxlKHgsIHksIGIsIHMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldFNhbXBsZShpbnQgeCwgaW50IHksIGludCBiLCBmbG9hdCBzKSB7Ci0gICAgICAgIHN1cGVyLnNldFNhbXBsZSh4LCB5LCBiLCBzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRTYW1wbGUoaW50IHgsIGludCB5LCBpbnQgYiwgZG91YmxlIHMpIHsKLSAgICAgICAgc3VwZXIuc2V0U2FtcGxlKHgsIHksIGIsIHMpOwotICAgIH0KLX0KXCBObyBuZXdsaW5lIGF0IGVuZCBvZiBmaWxlCmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9QbmdEZWNvZGVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9pbWFnZS9QbmdEZWNvZGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDdlODU2MDAuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvUG5nRGVjb2Rlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjcwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgT2xlZyBWLiBLaGFzY2hhbnNreQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICoKLSAqIEBkYXRlOiBKdWwgMjIsIDIwMDUKLSAqLwotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2U7Ci0KLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS51dGlsLkhhc2h0YWJsZTsKLWltcG9ydCBqYXZhLmF3dC5jb2xvci5Db2xvclNwYWNlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLio7Ci1pbXBvcnQgamF2YS5hd3QuKjsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi1wdWJsaWMgY2xhc3MgUG5nRGVjb2RlciBleHRlbmRzIEltYWdlRGVjb2RlciB7Ci0gICAgLy8gaW5pdGlhbGl6ZXMgcHJvcGVyIGZpZWxkIElEcwotICAgIHByaXZhdGUgc3RhdGljIG5hdGl2ZSB2b2lkIGluaXRJRHMoKTsKLQotICAgIHN0YXRpYyB7Ci0gICAgICAgIFN5c3RlbS5sb2FkTGlicmFyeSgiZ2wiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBpbml0SURzKCk7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IGhpbnRmbGFncyA9Ci0gICAgICAgICAgICBJbWFnZUNvbnN1bWVyLlNJTkdMRUZSQU1FIHwgLy8gUE5HIGlzIGEgc3RhdGljIGltYWdlCi0gICAgICAgICAgICBJbWFnZUNvbnN1bWVyLlRPUERPV05MRUZUUklHSFQgfCAvLyBUaGlzIG9yZGVyIGlzIG9ubHkgb25lIHBvc3NpYmxlCi0gICAgICAgICAgICBJbWFnZUNvbnN1bWVyLkNPTVBMRVRFU0NBTkxJTkVTOyAvLyBEb24ndCBkZWxpdmVyIGluY29tcGxldGUgc2NhbmxpbmVzCi0KLSAgICAvLyBFYWNoIHBpeGVsIGlzIGEgZ3JheXNjYWxlIHNhbXBsZS4KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgUE5HX0NPTE9SX1RZUEVfR1JBWSA9IDA7Ci0gICAgLy8gRWFjaCBwaXhlbCBpcyBhbiBSLEcsQiB0cmlwbGUuCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFBOR19DT0xPUl9UWVBFX1JHQiA9IDI7Ci0gICAgLy8gRWFjaCBwaXhlbCBpcyBhIHBhbGV0dGUgaW5kZXgsIGEgUExURSBjaHVuayBtdXN0IGFwcGVhci4KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgUE5HX0NPTE9SX1RZUEVfUExURSA9IDM7Ci0gICAgLy8gRWFjaCBwaXhlbCBpcyBhIGdyYXlzY2FsZSBzYW1wbGUsIGZvbGxvd2VkIGJ5IGFuIGFscGhhIHNhbXBsZS4KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgUE5HX0NPTE9SX1RZUEVfR1JBWV9BTFBIQSA9IDQ7Ci0gICAgLy8gRWFjaCBwaXhlbCBpcyBhbiBSLEcsQiB0cmlwbGUsIGZvbGxvd2VkIGJ5IGFuIGFscGhhIHNhbXBsZS4KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgUE5HX0NPTE9SX1RZUEVfUkdCQSA9IDY7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgSU5QVVRfQlVGRkVSX1NJWkUgPSA0MDk2OwotICAgIHByaXZhdGUgYnl0ZSBidWZmZXJbXSA9IG5ldyBieXRlW0lOUFVUX0JVRkZFUl9TSVpFXTsKLQotICAgIC8vIEJ1ZmZlcnMgZm9yIGRlY29kZWQgaW1hZ2UgZGF0YQotICAgIGJ5dGUgYnl0ZU91dFtdOwotICAgIGludCBpbnRPdXRbXTsKLQotICAgIC8vIE5hdGl2ZSBwb2ludGVyIHRvIHBuZyBkZWNvZGVyIGRhdGEKLSAgICBwcml2YXRlIGxvbmcgaE5hdGl2ZURlY29kZXI7Ci0KLSAgICBpbnQgaW1hZ2VXaWR0aCwgaW1hZ2VIZWlnaHQ7Ci0gICAgaW50IGNvbG9yVHlwZTsKLSAgICBpbnQgYml0RGVwdGg7Ci0gICAgYnl0ZSBjbWFwW107Ci0KLSAgICBib29sZWFuIHRyYW5zZmVySW50czsgLy8gSXMgdHJhbnNmZXIgdHlwZSBpbnQ/Li4gb3IgYnl0ZT8KLSAgICBpbnQgZGF0YUVsZW1lbnRzUGVyUGl4ZWwgPSAxOwotCi0gICAgQ29sb3JNb2RlbCBjbTsKLQotICAgIGludCB1cGRhdGVGcm9tU2NhbmxpbmU7IC8vIEZpcnN0IHNjYW5saW5lIHRvIHVwZGF0ZQotICAgIGludCBudW1TY2FubGluZXM7IC8vIE51bWJlciBvZiBzY2FubGluZXMgdG8gdXBkYXRlCi0KLSAgICBwcml2YXRlIG5hdGl2ZSBsb25nIGRlY29kZShieXRlW10gaW5wdXQsIGludCBieXRlc0luQnVmZmVyLCBsb25nIGhEZWNvZGVyKTsKLQotICAgIHByaXZhdGUgc3RhdGljIG5hdGl2ZSB2b2lkIHJlbGVhc2VOYXRpdmVEZWNvZGVyKGxvbmcgaERlY29kZXIpOwotCi0gICAgcHVibGljIFBuZ0RlY29kZXIoRGVjb2RpbmdJbWFnZVNvdXJjZSBzcmMsIElucHV0U3RyZWFtIGlzKSB7Ci0gICAgICAgIHN1cGVyKHNyYywgaXMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRlY29kZUltYWdlKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGludCBieXRlc1JlYWQgPSAwOwotICAgICAgICAgICAgaW50IG5lZWRCeXRlcywgb2Zmc2V0LCBieXRlc0luQnVmZmVyID0gMDsKLSAgICAgICAgICAgIC8vIFJlYWQgZnJvbSB0aGUgaW5wdXQgc3RyZWFtCi0gICAgICAgICAgICBmb3IgKDs7KSB7Ci0gICAgICAgICAgICAgICAgbmVlZEJ5dGVzID0gSU5QVVRfQlVGRkVSX1NJWkUgLSBieXRlc0luQnVmZmVyOwotICAgICAgICAgICAgICAgIG9mZnNldCA9IGJ5dGVzSW5CdWZmZXI7Ci0KLSAgICAgICAgICAgICAgICBieXRlc1JlYWQgPSBpbnB1dFN0cmVhbS5yZWFkKGJ1ZmZlciwgb2Zmc2V0LCBuZWVkQnl0ZXMpOwotCi0gICAgICAgICAgICAgICAgaWYgKGJ5dGVzUmVhZCA8IDApIHsgLy8gQnJlYWssIG5vdGhpbmcgdG8gcmVhZCBmcm9tIGJ1ZmZlciwgaW1hZ2UgdHJ1bmNhdGVkPwotICAgICAgICAgICAgICAgICAgICByZWxlYXNlTmF0aXZlRGVjb2RlcihoTmF0aXZlRGVjb2Rlcik7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIC8vIEtlZXAgdHJhY2sgb24gaG93IG11Y2ggYnl0ZXMgbGVmdCBpbiBidWZmZXIKLSAgICAgICAgICAgICAgICBieXRlc0luQnVmZmVyICs9IGJ5dGVzUmVhZDsKLSAgICAgICAgICAgICAgICBoTmF0aXZlRGVjb2RlciA9IGRlY29kZShidWZmZXIsIGJ5dGVzSW5CdWZmZXIsIGhOYXRpdmVEZWNvZGVyKTsKLSAgICAgICAgICAgICAgICAvLyBQTkcgZGVjb2RlciBhbHdheXMgY29uc3VtZXMgYWxsIGJ5dGVzIGF0IG9uY2UKLSAgICAgICAgICAgICAgICBieXRlc0luQnVmZmVyID0gMDsKLQotICAgICAgICAgICAgICAgIC8vIGlmIChieXRlc0NvbnN1bWVkIDwgMCkKLSAgICAgICAgICAgICAgICAvL2JyZWFrOyAvLyBFcnJvciBleGl0Ci0KLSAgICAgICAgICAgICAgICByZXR1cm5EYXRhKCk7Ci0KLSAgICAgICAgICAgICAgICAvLyBPSywgd2UgZGVjb2RlZCBhbGwgdGhlIHBpY3R1cmUgaW4gdGhlIHJpZ2h0IHdheS4uLgotICAgICAgICAgICAgICAgIGlmIChoTmF0aXZlRGVjb2RlciA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaW1hZ2VDb21wbGV0ZShJbWFnZUNvbnN1bWVyLlNUQVRJQ0lNQUdFRE9ORSk7Ci0gICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHRocm93IGU7Ci0gICAgICAgIH0gY2F0Y2ggKFJ1bnRpbWVFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgaW1hZ2VDb21wbGV0ZShJbWFnZUNvbnN1bWVyLklNQUdFRVJST1IpOwotICAgICAgICAgICAgdGhyb3cgZTsKLSAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgIGNsb3NlU3RyZWFtKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBAU3VwcHJlc3NXYXJuaW5ncygidW51c2VkIikKLSAgICBwcml2YXRlIHZvaWQgcmV0dXJuSGVhZGVyKCkgeyAvLyBDYWxsZWQgZnJvbSBuYXRpdmUgY29kZQotICAgICAgICBzZXREaW1lbnNpb25zKGltYWdlV2lkdGgsIGltYWdlSGVpZ2h0KTsKLQotICAgICAgICBzd2l0Y2ggKGNvbG9yVHlwZSkgewotICAgICAgICAgICAgY2FzZSBQTkdfQ09MT1JfVFlQRV9HUkFZOiB7Ci0gICAgICAgICAgICAgICAgaWYgKGJpdERlcHRoICE9IDggJiYgYml0RGVwdGggIT0gNCAmJiBiaXREZXB0aCAhPSAyICYmIGJpdERlcHRoICE9IDEpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjNDPVVua25vd24gUE5HIGNvbG9yIHR5cGUKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4zQyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIC8vIENyZWF0ZSBncmF5IGNvbG9yIG1vZGVsCi0gICAgICAgICAgICAgICAgaW50IG51bUVudHJpZXMgPSAxIDw8IGJpdERlcHRoOwotICAgICAgICAgICAgICAgIGludCBzY2FsZUZhY3RvciA9IDI1NSAvIChudW1FbnRyaWVzLTEpOwotICAgICAgICAgICAgICAgIGJ5dGUgY29tcHNbXSA9IG5ldyBieXRlW251bUVudHJpZXNdOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtRW50cmllczsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGNvbXBzW2ldID0gKGJ5dGUpIChpICogc2NhbGVGYWN0b3IpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBjbSA9IG5ldyBJbmRleENvbG9yTW9kZWwoLypiaXREZXB0aCovOCwgbnVtRW50cmllcywgY29tcHMsIGNvbXBzLCBjb21wcyk7Ci0KLSAgICAgICAgICAgICAgICB0cmFuc2ZlckludHMgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY2FzZSBQTkdfQ09MT1JfVFlQRV9SR0I6IHsKLSAgICAgICAgICAgICAgICBpZiAoYml0RGVwdGggIT0gOCkgewotICAgICAgICAgICAgICAgICAgICAvLyBhd3QuM0M9VW5rbm93biBQTkcgY29sb3IgdHlwZQotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjNDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgY20gPSBuZXcgRGlyZWN0Q29sb3JNb2RlbCgyNCwgMHhGRjAwMDAsIDB4RkYwMCwgMHhGRik7Ci0KLSAgICAgICAgICAgICAgICB0cmFuc2ZlckludHMgPSB0cnVlOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIFBOR19DT0xPUl9UWVBFX1BMVEU6IHsKLSAgICAgICAgICAgICAgICBpZiAoYml0RGVwdGggIT0gOCAmJiBiaXREZXB0aCAhPSA0ICYmIGJpdERlcHRoICE9IDIgJiYgYml0RGVwdGggIT0gMSkgewotICAgICAgICAgICAgICAgICAgICAvLyBhd3QuM0M9VW5rbm93biBQTkcgY29sb3IgdHlwZQotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjNDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgY20gPSBuZXcgSW5kZXhDb2xvck1vZGVsKC8qYml0RGVwdGgqLzgsIGNtYXAubGVuZ3RoIC8gMywgY21hcCwgMCwgZmFsc2UpOwotCi0gICAgICAgICAgICAgICAgdHJhbnNmZXJJbnRzID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNhc2UgUE5HX0NPTE9SX1RZUEVfR1JBWV9BTFBIQTogewotICAgICAgICAgICAgICAgIGlmIChiaXREZXB0aCAhPSA4KSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIGF3dC4zQz1Vbmtub3duIFBORyBjb2xvciB0eXBlCi0gICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuM0MiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBjbSA9IG5ldyBDb21wb25lbnRDb2xvck1vZGVsKENvbG9yU3BhY2UuZ2V0SW5zdGFuY2UoQ29sb3JTcGFjZS5DU19HUkFZKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHRydWUsIGZhbHNlLAotICAgICAgICAgICAgICAgICAgICAgICAgVHJhbnNwYXJlbmN5LlRSQU5TTFVDRU5ULAotICAgICAgICAgICAgICAgICAgICAgICAgRGF0YUJ1ZmZlci5UWVBFX0JZVEUpOwotCi0gICAgICAgICAgICAgICAgdHJhbnNmZXJJbnRzID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgZGF0YUVsZW1lbnRzUGVyUGl4ZWwgPSAyOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjYXNlIFBOR19DT0xPUl9UWVBFX1JHQkE6IHsKLSAgICAgICAgICAgICAgICBpZiAoYml0RGVwdGggIT0gOCkgewotICAgICAgICAgICAgICAgICAgICAvLyBhd3QuM0M9VW5rbm93biBQTkcgY29sb3IgdHlwZQotICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjNDIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgY20gPSBDb2xvck1vZGVsLmdldFJHQmRlZmF1bHQoKTsKLQotICAgICAgICAgICAgICAgIHRyYW5zZmVySW50cyA9IHRydWU7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4zQz1Vbmtub3duIFBORyBjb2xvciB0eXBlCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4zQyIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICB9Ci0KLSAgICAgICAgLy8gQ3JlYXRlIG91dHB1dCBidWZmZXIKLSAgICAgICAgaWYgKHRyYW5zZmVySW50cykgewotICAgICAgICAgICAgaW50T3V0ID0gbmV3IGludFtpbWFnZVdpZHRoICogaW1hZ2VIZWlnaHRdOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgYnl0ZU91dCA9IG5ldyBieXRlW2ltYWdlV2lkdGggKiBpbWFnZUhlaWdodCAqIGRhdGFFbGVtZW50c1BlclBpeGVsXTsKLSAgICAgICAgfQotCi0gICAgICAgIHNldENvbG9yTW9kZWwoY20pOwotCi0gICAgICAgIHNldEhpbnRzKGhpbnRmbGFncyk7Ci0gICAgICAgIHNldFByb3BlcnRpZXMobmV3IEhhc2h0YWJsZTxPYmplY3QsIE9iamVjdD4oKSk7IC8vIEVtcHR5Ci0gICAgfQotCi0gICAgLy8gU2VuZCB0aGUgZGF0YSB0byB0aGUgY29uc3VtZXIKLSAgICBwcml2YXRlIHZvaWQgcmV0dXJuRGF0YSgpIHsKLSAgICAgICAgLy8gU2VuZCAxIG9yIG1vcmUgc2NhbmxpbmVzIHRvIHRoZSBjb25zdW1lci4KLSAgICAgICAgaWYgKG51bVNjYW5saW5lcyA+IDApIHsKLSAgICAgICAgICAgIC8vIE5hdGl2ZSBkZWNvZGVyIGNvdWxkIGhhdmUgcmV0dXJuZWQKLSAgICAgICAgICAgIC8vIHNvbWUgZGF0YSBmcm9tIHRoZSBuZXh0IHBhc3MsIGhhbmRsZSBpdCBoZXJlCi0gICAgICAgICAgICBpbnQgcGFzczEsIHBhc3MyOwotICAgICAgICAgICAgaWYgKHVwZGF0ZUZyb21TY2FubGluZSArIG51bVNjYW5saW5lcyA+IGltYWdlSGVpZ2h0KSB7Ci0gICAgICAgICAgICAgICAgcGFzczEgPSBpbWFnZUhlaWdodCAtIHVwZGF0ZUZyb21TY2FubGluZTsKLSAgICAgICAgICAgICAgICBwYXNzMiA9IHVwZGF0ZUZyb21TY2FubGluZSArIG51bVNjYW5saW5lcyAtIGltYWdlSGVpZ2h0OwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBwYXNzMSA9IG51bVNjYW5saW5lczsKLSAgICAgICAgICAgICAgICBwYXNzMiA9IDA7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHRyYW5zZmVyKHVwZGF0ZUZyb21TY2FubGluZSwgcGFzczEpOwotICAgICAgICAgICAgaWYgKHBhc3MyICE9IDApIHsKLSAgICAgICAgICAgICAgICB0cmFuc2ZlcigwLCBwYXNzMik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwcml2YXRlIHZvaWQgdHJhbnNmZXIoaW50IHVwZGF0ZUZyb21TY2FubGluZSwgaW50IG51bVNjYW5saW5lcykgewotICAgICAgICBpZiAodHJhbnNmZXJJbnRzKSB7Ci0gICAgICAgICAgICBzZXRQaXhlbHMoCi0gICAgICAgICAgICAgICAgICAgIDAsIHVwZGF0ZUZyb21TY2FubGluZSwKLSAgICAgICAgICAgICAgICAgICAgaW1hZ2VXaWR0aCwgbnVtU2NhbmxpbmVzLAotICAgICAgICAgICAgICAgICAgICBjbSwgaW50T3V0LAotICAgICAgICAgICAgICAgICAgICB1cGRhdGVGcm9tU2NhbmxpbmUgKiBpbWFnZVdpZHRoLAotICAgICAgICAgICAgICAgICAgICBpbWFnZVdpZHRoCi0gICAgICAgICAgICApOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgc2V0UGl4ZWxzKAotICAgICAgICAgICAgICAgICAgICAwLCB1cGRhdGVGcm9tU2NhbmxpbmUsCi0gICAgICAgICAgICAgICAgICAgIGltYWdlV2lkdGgsIG51bVNjYW5saW5lcywKLSAgICAgICAgICAgICAgICAgICAgY20sIGJ5dGVPdXQsCi0gICAgICAgICAgICAgICAgICAgIHVwZGF0ZUZyb21TY2FubGluZSAqIGltYWdlV2lkdGggKiBkYXRhRWxlbWVudHNQZXJQaXhlbCwKLSAgICAgICAgICAgICAgICAgICAgaW1hZ2VXaWR0aCAqIGRhdGFFbGVtZW50c1BlclBpeGVsCi0gICAgICAgICAgICApOwotICAgICAgICB9Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvUG5nRGVjb2RlckphdmEuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL1BuZ0RlY29kZXJKYXZhLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQ2NTQ1ZjkuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvUG5nRGVjb2RlckphdmEuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI4MiArMCwwIEBACi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2U7Ci0KLS8vIEEgc2ltcGxlIFBORyBkZWNvZGVyIHNvdXJjZSBjb2RlIGluIEphdmEuCi1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3M7Ci1pbXBvcnQgamF2YS5hd3QuSW5zZXRzOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQ29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5EYXRhQnVmZmVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJCeXRlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkluZGV4Q29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5SYXN0ZXI7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuV3JpdGFibGVSYXN0ZXI7Ci1pbXBvcnQgamF2YS5pby5CeXRlQXJyYXlJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLkJ5dGVBcnJheU91dHB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLkRhdGFJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLkVPRkV4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5VbnN1cHBvcnRlZEVuY29kaW5nRXhjZXB0aW9uOwotaW1wb3J0IGphdmEudXRpbC56aXAuQ1JDMzI7Ci1pbXBvcnQgamF2YS51dGlsLnppcC5JbmZsYXRlcklucHV0U3RyZWFtOwotCi0vL2ltcG9ydCBqYXZheC5zd2luZy5KRnJhbWU7Ci0KLXB1YmxpYyBjbGFzcyBQbmdEZWNvZGVySmF2YSB7Ci0gCi0vKgotICBwdWJsaWMgc3RhdGljIHZvaWQgbWFpbihTdHJpbmdbXSBhcmdzKSB0aHJvd3MgRXhjZXB0aW9uIHsKLSAgICBTdHJpbmcgbmFtZSA9ICJsb2dvLnBuZyI7Ci0gICAgaWYgKGFyZ3MubGVuZ3RoID4gMCkKLSAgICAgIG5hbWUgPSBhcmdzWzBdOwotICAgIElucHV0U3RyZWFtIGluID0gUG5nRGVjb2RlckphdmEuY2xhc3MuZ2V0UmVzb3VyY2VBc1N0cmVhbShuYW1lKTsKLSAgICBmaW5hbCBCdWZmZXJlZEltYWdlIGltYWdlID0gUG5nRGVjb2RlckphdmEuZGVjb2RlKGluKTsKLSAgICBpbi5jbG9zZSgpOwotCi0gICAgSkZyYW1lIGYgPSBuZXcgSkZyYW1lKCkgewotICAgICAgcHVibGljIHZvaWQgcGFpbnQoR3JhcGhpY3MgZykgewotICAgICAgICBJbnNldHMgaW5zZXRzID0gZ2V0SW5zZXRzKCk7Ci0gICAgICAgIGcuZHJhd0ltYWdlKGltYWdlLCBpbnNldHMubGVmdCwgaW5zZXRzLnRvcCwgbnVsbCk7Ci0gICAgICB9Ci0gICAgfTsKLSAgICBmLnNldFZpc2libGUodHJ1ZSk7Ci0gICAgSW5zZXRzIGluc2V0cyA9IGYuZ2V0SW5zZXRzKCk7Ci0gICAgZi5zZXRTaXplKGltYWdlLmdldFdpZHRoKCkgKyBpbnNldHMubGVmdCArIGluc2V0cy5yaWdodCwgaW1hZ2UKLSAgICAgICAgLmdldEhlaWdodCgpCi0gICAgICAgICsgaW5zZXRzLnRvcCArIGluc2V0cy5ib3R0b20pOwotICB9Ci0gICovCi0KLSAgcHVibGljIHN0YXRpYyBCdWZmZXJlZEltYWdlIGRlY29kZShJbnB1dFN0cmVhbSBpbikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICBEYXRhSW5wdXRTdHJlYW0gZGF0YUluID0gbmV3IERhdGFJbnB1dFN0cmVhbShpbik7Ci0gICAgcmVhZFNpZ25hdHVyZShkYXRhSW4pOwotICAgIFBOR0RhdGEgY2h1bmtzID0gcmVhZENodW5rcyhkYXRhSW4pOwotCi0gICAgbG9uZyB3aWR0aExvbmcgPSBjaHVua3MuZ2V0V2lkdGgoKTsKLSAgICBsb25nIGhlaWdodExvbmcgPSBjaHVua3MuZ2V0SGVpZ2h0KCk7Ci0gICAgaWYgKHdpZHRoTG9uZyA+IEludGVnZXIuTUFYX1ZBTFVFIHx8IGhlaWdodExvbmcgPiBJbnRlZ2VyLk1BWF9WQUxVRSkKLSAgICAgIHRocm93IG5ldyBJT0V4Y2VwdGlvbigiVGhhdCBpbWFnZSBpcyB0b28gd2lkZSBvciB0YWxsLiIpOwotICAgIGludCB3aWR0aCA9IChpbnQpIHdpZHRoTG9uZzsKLSAgICBpbnQgaGVpZ2h0ID0gKGludCkgaGVpZ2h0TG9uZzsKLQotICAgIENvbG9yTW9kZWwgY20gPSBjaHVua3MuZ2V0Q29sb3JNb2RlbCgpOwotICAgIFdyaXRhYmxlUmFzdGVyIHJhc3RlciA9IGNodW5rcy5nZXRSYXN0ZXIoKTsKLQotICAgIEJ1ZmZlcmVkSW1hZ2UgaW1hZ2UgPSBuZXcgQnVmZmVyZWRJbWFnZShjbSwgcmFzdGVyLCBmYWxzZSwgbnVsbCk7Ci0KLSAgICByZXR1cm4gaW1hZ2U7Ci0gIH0KLQotICBwcm90ZWN0ZWQgc3RhdGljIHZvaWQgcmVhZFNpZ25hdHVyZShEYXRhSW5wdXRTdHJlYW0gaW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgbG9uZyBzaWduYXR1cmUgPSBpbi5yZWFkTG9uZygpOwotICAgIGlmIChzaWduYXR1cmUgIT0gMHg4OTUwNGU0NzBkMGExYTBhTCkKLSAgICAgIHRocm93IG5ldyBJT0V4Y2VwdGlvbigiUE5HIHNpZ25hdHVyZSBub3QgZm91bmQhIik7Ci0gIH0KLQotICBwcm90ZWN0ZWQgc3RhdGljIFBOR0RhdGEgcmVhZENodW5rcyhEYXRhSW5wdXRTdHJlYW0gaW4pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgUE5HRGF0YSBjaHVua3MgPSBuZXcgUE5HRGF0YSgpOwotCi0gICAgYm9vbGVhbiB0cnVja2luZyA9IHRydWU7Ci0gICAgd2hpbGUgKHRydWNraW5nKSB7Ci0gICAgICB0cnkgewotICAgICAgICAvLyBSZWFkIHRoZSBsZW5ndGguCi0gICAgICAgIGludCBsZW5ndGggPSBpbi5yZWFkSW50KCk7Ci0gICAgICAgIGlmIChsZW5ndGggPCAwKQotICAgICAgICAgIHRocm93IG5ldyBJT0V4Y2VwdGlvbigiU29ycnksIHRoYXQgZmlsZSBpcyB0b28gbG9uZy4iKTsKLSAgICAgICAgLy8gUmVhZCB0aGUgdHlwZS4KLSAgICAgICAgYnl0ZVtdIHR5cGVCeXRlcyA9IG5ldyBieXRlWzRdOwotICAgICAgICBpbi5yZWFkRnVsbHkodHlwZUJ5dGVzKTsKLSAgICAgICAgLy8gUmVhZCB0aGUgZGF0YS4KLSAgICAgICAgYnl0ZVtdIGRhdGEgPSBuZXcgYnl0ZVtsZW5ndGhdOwotICAgICAgICBpbi5yZWFkRnVsbHkoZGF0YSk7Ci0gICAgICAgIC8vIFJlYWQgdGhlIENSQy4KLSAgICAgICAgbG9uZyBjcmMgPSBpbi5yZWFkSW50KCkgJiAweDAwMDAwMDAwZmZmZmZmZmZMOyAvLyBNYWtlIGl0Ci0gICAgICAgIC8vIHVuc2lnbmVkLgotICAgICAgICBpZiAodmVyaWZ5Q1JDKHR5cGVCeXRlcywgZGF0YSwgY3JjKSA9PSBmYWxzZSkKLSAgICAgICAgICB0aHJvdyBuZXcgSU9FeGNlcHRpb24oIlRoYXQgZmlsZSBhcHBlYXJzIHRvIGJlIGNvcnJ1cHRlZC4iKTsKLQotICAgICAgICBQTkdDaHVuayBjaHVuayA9IG5ldyBQTkdDaHVuayh0eXBlQnl0ZXMsIGRhdGEpOwotICAgICAgICBjaHVua3MuYWRkKGNodW5rKTsKLSAgICAgIH0gY2F0Y2ggKEVPRkV4Y2VwdGlvbiBlb2ZlKSB7Ci0gICAgICAgIHRydWNraW5nID0gZmFsc2U7Ci0gICAgICB9Ci0gICAgfQotICAgIHJldHVybiBjaHVua3M7Ci0gIH0KLQotICBwcm90ZWN0ZWQgc3RhdGljIGJvb2xlYW4gdmVyaWZ5Q1JDKGJ5dGVbXSB0eXBlQnl0ZXMsIGJ5dGVbXSBkYXRhLCBsb25nIGNyYykgewotICAgIENSQzMyIGNyYzMyID0gbmV3IENSQzMyKCk7Ci0gICAgY3JjMzIudXBkYXRlKHR5cGVCeXRlcyk7Ci0gICAgY3JjMzIudXBkYXRlKGRhdGEpOwotICAgIGxvbmcgY2FsY3VsYXRlZCA9IGNyYzMyLmdldFZhbHVlKCk7Ci0gICAgcmV0dXJuIChjYWxjdWxhdGVkID09IGNyYyk7Ci0gIH0KLX0KLQotY2xhc3MgUE5HRGF0YSB7Ci0gIHByaXZhdGUgaW50IG1OdW1iZXJPZkNodW5rczsKLQotICBwcml2YXRlIFBOR0NodW5rW10gbUNodW5rczsKLQotICBwdWJsaWMgUE5HRGF0YSgpIHsKLSAgICBtTnVtYmVyT2ZDaHVua3MgPSAwOwotICAgIG1DaHVua3MgPSBuZXcgUE5HQ2h1bmtbMTBdOwotICB9Ci0KLSAgcHVibGljIHZvaWQgYWRkKFBOR0NodW5rIGNodW5rKSB7Ci0gICAgbUNodW5rc1ttTnVtYmVyT2ZDaHVua3MrK10gPSBjaHVuazsKLSAgICBpZiAobU51bWJlck9mQ2h1bmtzID49IG1DaHVua3MubGVuZ3RoKSB7Ci0gICAgICBQTkdDaHVua1tdIGxhcmdlckFycmF5ID0gbmV3IFBOR0NodW5rW21DaHVua3MubGVuZ3RoICsgMTBdOwotICAgICAgU3lzdGVtLmFycmF5Y29weShtQ2h1bmtzLCAwLCBsYXJnZXJBcnJheSwgMCwgbUNodW5rcy5sZW5ndGgpOwotICAgICAgbUNodW5rcyA9IGxhcmdlckFycmF5OwotICAgIH0KLSAgfQotCi0gIHB1YmxpYyBsb25nIGdldFdpZHRoKCkgewotICAgIHJldHVybiBnZXRDaHVuaygiSUhEUiIpLmdldFVuc2lnbmVkSW50KDApOwotICB9Ci0KLSAgcHVibGljIGxvbmcgZ2V0SGVpZ2h0KCkgeyAgICByZXR1cm4gZ2V0Q2h1bmsoIklIRFIiKS5nZXRVbnNpZ25lZEludCg0KTsKLSAgfQotCi0gIHB1YmxpYyBzaG9ydCBnZXRCaXRzUGVyUGl4ZWwoKSB7Ci0gICAgcmV0dXJuIGdldENodW5rKCJJSERSIikuZ2V0VW5zaWduZWRCeXRlKDgpOwotICB9Ci0KLSAgcHVibGljIHNob3J0IGdldENvbG9yVHlwZSgpIHsKLSAgICByZXR1cm4gZ2V0Q2h1bmsoIklIRFIiKS5nZXRVbnNpZ25lZEJ5dGUoOSk7Ci0gIH0KLQotICBwdWJsaWMgc2hvcnQgZ2V0Q29tcHJlc3Npb24oKSB7Ci0gICAgcmV0dXJuIGdldENodW5rKCJJSERSIikuZ2V0VW5zaWduZWRCeXRlKDEwKTsKLSAgfQotCi0gIHB1YmxpYyBzaG9ydCBnZXRGaWx0ZXIoKSB7Ci0gICAgcmV0dXJuIGdldENodW5rKCJJSERSIikuZ2V0VW5zaWduZWRCeXRlKDExKTsKLSAgfQotCi0gIHB1YmxpYyBzaG9ydCBnZXRJbnRlcmxhY2UoKSB7Ci0gICAgcmV0dXJuIGdldENodW5rKCJJSERSIikuZ2V0VW5zaWduZWRCeXRlKDEyKTsKLSAgfQotCi0gIHB1YmxpYyBDb2xvck1vZGVsIGdldENvbG9yTW9kZWwoKSB7Ci0gICAgc2hvcnQgY29sb3JUeXBlID0gZ2V0Q29sb3JUeXBlKCk7Ci0gICAgaW50IGJpdHNQZXJQaXhlbCA9IGdldEJpdHNQZXJQaXhlbCgpOwotCi0gICAgaWYgKGNvbG9yVHlwZSA9PSAzKSB7Ci0gICAgICBieXRlW10gcGFsZXR0ZURhdGEgPSBnZXRDaHVuaygiUExURSIpLmdldERhdGEoKTsKLSAgICAgIGludCBwYWxldHRlTGVuZ3RoID0gcGFsZXR0ZURhdGEubGVuZ3RoIC8gMzsKLSAgICAgIHJldHVybiBuZXcgSW5kZXhDb2xvck1vZGVsKGJpdHNQZXJQaXhlbCwgcGFsZXR0ZUxlbmd0aCwKLSAgICAgICAgICBwYWxldHRlRGF0YSwgMCwgZmFsc2UpOwotICAgIH0KLSAgICBTeXN0ZW0ub3V0LnByaW50bG4oIlVuc3VwcG9ydGVkIGNvbG9yIHR5cGU6ICIgKyBjb2xvclR5cGUpOwotICAgIHJldHVybiBudWxsOwotICB9Ci0KLSAgcHVibGljIFdyaXRhYmxlUmFzdGVyIGdldFJhc3RlcigpIHsKLSAgICBpbnQgd2lkdGggPSAoaW50KSBnZXRXaWR0aCgpOwotICAgIGludCBoZWlnaHQgPSAoaW50KSBnZXRIZWlnaHQoKTsKLSAgICBpbnQgYml0c1BlclBpeGVsID0gZ2V0Qml0c1BlclBpeGVsKCk7Ci0gICAgc2hvcnQgY29sb3JUeXBlID0gZ2V0Q29sb3JUeXBlKCk7Ci0KLSAgICBpZiAoY29sb3JUeXBlID09IDMpIHsKLSAgICAgIGJ5dGVbXSBpbWFnZURhdGEgPSBnZXRJbWFnZURhdGEoKTsKLSAgICAgIC8vT3JpZzogRGF0YUJ1ZmZlciBkYiA9IG5ldyBEYXRhQnVmZmVyQnl0ZShpbWFnZURhdGEsIGltYWdlRGF0YS5sZW5ndGgpOwotICAgICAgaW50IGxlbiA9IE1hdGgubWF4KGltYWdlRGF0YS5sZW5ndGgsICh3aWR0aCAtIDEpICogKGhlaWdodCAtMSkpOwotICAgICAgRGF0YUJ1ZmZlciBkYiA9IG5ldyBEYXRhQnVmZmVyQnl0ZShpbWFnZURhdGEsIGxlbik7Ci0gICAgICBXcml0YWJsZVJhc3RlciByYXN0ZXIgPSBSYXN0ZXIuY3JlYXRlUGFja2VkUmFzdGVyKGRiLCB3aWR0aCwKLSAgICAgICAgICBoZWlnaHQsIGJpdHNQZXJQaXhlbCwgbnVsbCk7Ci0gICAgICByZXR1cm4gcmFzdGVyOwotICAgIH0gZWxzZQotICAgICAgU3lzdGVtLm91dC5wcmludGxuKCJVbnN1cHBvcnRlZCBjb2xvciB0eXBlISIpOwotICAgIHJldHVybiBudWxsOwotICB9Ci0KLSAgcHVibGljIGJ5dGVbXSBnZXRJbWFnZURhdGEoKSB7Ci0gICAgdHJ5IHsKLSAgICAgIEJ5dGVBcnJheU91dHB1dFN0cmVhbSBvdXQgPSBuZXcgQnl0ZUFycmF5T3V0cHV0U3RyZWFtKCk7Ci0gICAgICAvLyBXcml0ZSBhbGwgdGhlIElEQVQgZGF0YSBpbnRvIHRoZSBhcnJheS4KLSAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbU51bWJlck9mQ2h1bmtzOyBpKyspIHsKLSAgICAgICAgUE5HQ2h1bmsgY2h1bmsgPSBtQ2h1bmtzW2ldOwotICAgICAgICBpZiAoY2h1bmsuZ2V0VHlwZVN0cmluZygpLmVxdWFscygiSURBVCIpKSB7Ci0gICAgICAgICAgb3V0LndyaXRlKGNodW5rLmdldERhdGEoKSk7Ci0gICAgICAgIH0KLSAgICAgIH0KLSAgICAgIG91dC5mbHVzaCgpOwotICAgICAgLy8gTm93IGRlZmxhdGUgdGhlIGRhdGEuCi0gICAgICBJbmZsYXRlcklucHV0U3RyZWFtIGluID0gbmV3IEluZmxhdGVySW5wdXRTdHJlYW0oCi0gICAgICAgICAgbmV3IEJ5dGVBcnJheUlucHV0U3RyZWFtKG91dC50b0J5dGVBcnJheSgpKSk7Ci0gICAgICBCeXRlQXJyYXlPdXRwdXRTdHJlYW0gaW5mbGF0ZWRPdXQgPSBuZXcgQnl0ZUFycmF5T3V0cHV0U3RyZWFtKCk7Ci0gICAgICBpbnQgcmVhZExlbmd0aDsKLSAgICAgIGJ5dGVbXSBibG9jayA9IG5ldyBieXRlWzgxOTJdOwotICAgICAgd2hpbGUgKChyZWFkTGVuZ3RoID0gaW4ucmVhZChibG9jaykpICE9IC0xKQotICAgICAgICBpbmZsYXRlZE91dC53cml0ZShibG9jaywgMCwgcmVhZExlbmd0aCk7Ci0gICAgICBpbmZsYXRlZE91dC5mbHVzaCgpOwotICAgICAgYnl0ZVtdIGltYWdlRGF0YSA9IGluZmxhdGVkT3V0LnRvQnl0ZUFycmF5KCk7Ci0gICAgICAvLyBDb21wdXRlIHRoZSByZWFsIGxlbmd0aC4KLSAgICAgIGludCB3aWR0aCA9IChpbnQpIGdldFdpZHRoKCk7Ci0gICAgICBpbnQgaGVpZ2h0ID0gKGludCkgZ2V0SGVpZ2h0KCk7Ci0gICAgICBpbnQgYml0c1BlclBpeGVsID0gZ2V0Qml0c1BlclBpeGVsKCk7Ci0gICAgICBpbnQgbGVuZ3RoID0gd2lkdGggKiBoZWlnaHQgKiBiaXRzUGVyUGl4ZWwgLyA4OwotCi0gICAgICBieXRlW10gcHJ1bmVkRGF0YSA9IG5ldyBieXRlW2xlbmd0aF07Ci0KLSAgICAgIC8vIFdlIGNhbiBvbmx5IGRlYWwgd2l0aCBub24taW50ZXJsYWNlZCBpbWFnZXMuCi0gICAgICBpZiAoZ2V0SW50ZXJsYWNlKCkgPT0gMCkgewotICAgICAgICBpbnQgaW5kZXggPSAwOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgaWYgKChpICogOCAvIGJpdHNQZXJQaXhlbCkgJSB3aWR0aCA9PSAwKSB7Ci0gICAgICAgICAgICBpbmRleCsrOyAvLyBTa2lwIHRoZSBmaWx0ZXIgYnl0ZS4KLSAgICAgICAgICB9Ci0gICAgICAgICAgcHJ1bmVkRGF0YVtpXSA9IGltYWdlRGF0YVtpbmRleCsrXTsKLSAgICAgICAgfQotICAgICAgfSBlbHNlCi0gICAgICAgIFN5c3RlbS5vdXQucHJpbnRsbigiQ291bGRuJ3QgdW5kbyBpbnRlcmxhY2luZy4iKTsKLQotICAgICAgcmV0dXJuIHBydW5lZERhdGE7Ci0gICAgfSBjYXRjaCAoSU9FeGNlcHRpb24gaW9lKSB7Ci0gICAgfQotICAgIHJldHVybiBudWxsOwotICB9Ci0KLSAgcHVibGljIFBOR0NodW5rIGdldENodW5rKFN0cmluZyB0eXBlKSB7Ci0gICAgZm9yIChpbnQgaSA9IDA7IGkgPCBtTnVtYmVyT2ZDaHVua3M7IGkrKykKLSAgICAgIGlmIChtQ2h1bmtzW2ldLmdldFR5cGVTdHJpbmcoKS5lcXVhbHModHlwZSkpCi0gICAgICAgIHJldHVybiBtQ2h1bmtzW2ldOwotICAgIHJldHVybiBudWxsOwotICB9Ci19Ci0KLWNsYXNzIFBOR0NodW5rIHsKLSAgcHJpdmF0ZSBieXRlW10gbVR5cGU7Ci0KLSAgcHJpdmF0ZSBieXRlW10gbURhdGE7Ci0KLSAgcHVibGljIFBOR0NodW5rKGJ5dGVbXSB0eXBlLCBieXRlW10gZGF0YSkgewotICAgIG1UeXBlID0gdHlwZTsKLSAgICBtRGF0YSA9IGRhdGE7Ci0gIH0KLQotICBwdWJsaWMgU3RyaW5nIGdldFR5cGVTdHJpbmcoKSB7Ci0gICAgdHJ5IHsKLSAgICAgIHJldHVybiBuZXcgU3RyaW5nKG1UeXBlLCAiVVRGOCIpOwotICAgIH0gY2F0Y2ggKFVuc3VwcG9ydGVkRW5jb2RpbmdFeGNlcHRpb24gdWVlKSB7Ci0gICAgICByZXR1cm4gIiI7Ci0gICAgfQotICB9Ci0KLSAgcHVibGljIGJ5dGVbXSBnZXREYXRhKCkgewotICAgIHJldHVybiBtRGF0YTsKLSAgfQotCi0gIHB1YmxpYyBsb25nIGdldFVuc2lnbmVkSW50KGludCBvZmZzZXQpIHsKLSAgICBsb25nIHZhbHVlID0gMDsKLSAgICBmb3IgKGludCBpID0gMDsgaSA8IDQ7IGkrKykKLSAgICAgIHZhbHVlICs9IChtRGF0YVtvZmZzZXQgKyBpXSAmIDB4ZmYpIDw8ICgoMyAtIGkpICogOCk7Ci0gICAgcmV0dXJuIHZhbHVlOwotICB9Ci0KLSAgcHVibGljIHNob3J0IGdldFVuc2lnbmVkQnl0ZShpbnQgb2Zmc2V0KSB7Ci0gICAgcmV0dXJuIChzaG9ydCkgKG1EYXRhW29mZnNldF0gJiAweDAwZmYpOwotICB9Ci19ClwgTm8gbmV3bGluZSBhdCBlbmQgb2YgZmlsZQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvVVJMRGVjb2RpbmdJbWFnZVNvdXJjZS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvaW1hZ2UvVVJMRGVjb2RpbmdJbWFnZVNvdXJjZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBhMTg5OWQ2Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL2ltYWdlL1VSTERlY29kaW5nSW1hZ2VTb3VyY2UuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDc3ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotLyoKLSAqIENyZWF0ZWQgb24gMTAuMDIuMjAwNQotICoKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmltYWdlOwotCi1pbXBvcnQgamF2YS5pby5CdWZmZXJlZElucHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLm5ldC5VUkw7Ci1pbXBvcnQgamF2YS5uZXQuVVJMQ29ubmVjdGlvbjsKLWltcG9ydCBqYXZhLnNlY3VyaXR5LlBlcm1pc3Npb247Ci0KLXB1YmxpYyBjbGFzcyBVUkxEZWNvZGluZ0ltYWdlU291cmNlIGV4dGVuZHMgRGVjb2RpbmdJbWFnZVNvdXJjZSB7Ci0KLSAgICBVUkwgdXJsOwotCi0gICAgcHVibGljIFVSTERlY29kaW5nSW1hZ2VTb3VyY2UoVVJMIHVybCl7Ci0gICAgICAgIFNlY3VyaXR5TWFuYWdlciBzZWN1cml0eSA9IFN5c3RlbS5nZXRTZWN1cml0eU1hbmFnZXIoKTsKLSAgICAgICAgaWYgKHNlY3VyaXR5ICE9IG51bGwpIHsKLSAgICAgICAgICAgIHNlY3VyaXR5LmNoZWNrQ29ubmVjdCh1cmwuZ2V0SG9zdCgpLCB1cmwuZ2V0UG9ydCgpKTsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgUGVybWlzc2lvbiBwID0gdXJsLm9wZW5Db25uZWN0aW9uKCkuZ2V0UGVybWlzc2lvbigpOwotICAgICAgICAgICAgICAgIHNlY3VyaXR5LmNoZWNrUGVybWlzc2lvbihwKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICB0aGlzLnVybCA9IHVybDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgYm9vbGVhbiBjaGVja0Nvbm5lY3Rpb24oKSB7Ci0gICAgICAgIFNlY3VyaXR5TWFuYWdlciBzZWN1cml0eSA9IFN5c3RlbS5nZXRTZWN1cml0eU1hbmFnZXIoKTsKLSAgICAgICAgaWYgKHNlY3VyaXR5ICE9IG51bGwpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgc2VjdXJpdHkuY2hlY2tDb25uZWN0KHVybC5nZXRIb3N0KCksIHVybC5nZXRQb3J0KCkpOwotICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgfSBjYXRjaCAoU2VjdXJpdHlFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgSW5wdXRTdHJlYW0gZ2V0SW5wdXRTdHJlYW0oKSB7Ci0gICAgICAgIHRyeXsKLSAgICAgICAgICAgIFVSTENvbm5lY3Rpb24gdWMgPSB1cmwub3BlbkNvbm5lY3Rpb24oKTsKLSAgICAgICAgICAgIC8vIEJFR0lOIGFuZHJvaWQtbW9kaWZpZWQKLSAgICAgICAgICAgIHJldHVybiBuZXcgQnVmZmVyZWRJbnB1dFN0cmVhbSh1Yy5nZXRJbnB1dFN0cmVhbSgpLCA4MTkyKTsKLSAgICAgICAgICAgIC8vIEVORCBhbmRyb2lkLW1vZGlmaWVkCi0gICAgICAgIH1jYXRjaChJT0V4Y2VwdGlvbiBlKXsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvQmxpdHRlci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvcmVuZGVyL0JsaXR0ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggM2I4MDEyZS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvQmxpdHRlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICogQ3JlYXRlZCBvbiAxNC4xMS4yMDA1Ci0gKgotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wucmVuZGVyOwotCi1pbXBvcnQgamF2YS5hd3QuQ29sb3I7Ci1pbXBvcnQgamF2YS5hd3QuQ29tcG9zaXRlOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5NdWx0aVJlY3RBcmVhOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuU3VyZmFjZTsKLQotLyoqCi0gKiBUaGUgaW50ZXJmYWNlIGZvciBvYmplY3RzIHdoaWNoIGNhbiBkcmF3aW5nIEltYWdlcyBvbiBvdGhlciBJbWFnZXMgd2hpY2ggaGF2ZSAKLSAqIEdyYXBoaWNzIG9yIG9uIHRoZSBkaXNwbGF5LiAgCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgQmxpdHRlciB7Ci0KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBibGl0KGludCBzcmNYLCBpbnQgc3JjWSwgU3VyZmFjZSBzcmNTdXJmLAotICAgICAgICAgICAgaW50IGRzdFgsIGludCBkc3RZLCBTdXJmYWNlIGRzdFN1cmYsIGludCB3aWR0aCwgaW50IGhlaWdodCwKLSAgICAgICAgICAgIEFmZmluZVRyYW5zZm9ybSBzeXN4Zm9ybSwgQWZmaW5lVHJhbnNmb3JtIHhmb3JtLAotICAgICAgICAgICAgQ29tcG9zaXRlIGNvbXAsIENvbG9yIGJnY29sb3IsCi0gICAgICAgICAgICBNdWx0aVJlY3RBcmVhIGNsaXApOwotCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgYmxpdChpbnQgc3JjWCwgaW50IHNyY1ksIFN1cmZhY2Ugc3JjU3VyZiwKLSAgICAgICAgICAgIGludCBkc3RYLCBpbnQgZHN0WSwgU3VyZmFjZSBkc3RTdXJmLCBpbnQgd2lkdGgsIGludCBoZWlnaHQsCi0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gc3lzeGZvcm0sIENvbXBvc2l0ZSBjb21wLCBDb2xvciBiZ2NvbG9yLAotICAgICAgICAgICAgTXVsdGlSZWN0QXJlYSBjbGlwKTsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIGJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsCi0gICAgICAgICAgICBpbnQgZHN0WCwgaW50IGRzdFksIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LAotICAgICAgICAgICAgQ29tcG9zaXRlIGNvbXAsIENvbG9yIGJnY29sb3IsIE11bHRpUmVjdEFyZWEgY2xpcCk7Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL3JlbmRlci9KYXZhQXJjUmFzdGVyaXplci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvcmVuZGVyL0phdmFBcmNSYXN0ZXJpemVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGI2NDNiNDEuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvcmVuZGVyL0phdmFBcmNSYXN0ZXJpemVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1MDIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wucmVuZGVyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5NdWx0aVJlY3RBcmVhOwotCi1wdWJsaWMgY2xhc3MgSmF2YUFyY1Jhc3Rlcml6ZXIgewotCi0gICAgLyoqCi0gICAgICogQWRkcyBwYXJ0aWN1bGFyIGFyYyBzZWdtZW50IHRvIG1yYSAKLSAgICAgKi8KLSAgICBzdGF0aWMgdm9pZCBhZGRYMExpbmVTZWcoTXVsdGlSZWN0QXJlYSBtcmEsIGludFtdIGxpbmUsIGludCBjeCwgaW50IGN5LCBpbnQgYiwgaW50IHN0YXJ0LCBpbnQgZmluaXNoKSB7Ci0gICAgICAgIGludCB4MSA9IDA7Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBsaW5lLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBpbnQgeDIgPSBsaW5lW2ldOwotICAgICAgICAgICAgaW50IHkgPSBjeSArIChiIC0gaSk7Ci0gICAgICAgICAgICBpZiAoeDEgPD0gZmluaXNoICYmIHgyID49IHN0YXJ0KSB7Ci0gICAgICAgICAgICAgICAgbXJhLmFkZFJlY3QoY3ggKyBNYXRoLm1heCh4MSwgc3RhcnQpLCB5LCBjeCArIE1hdGgubWluKHgyLCBmaW5pc2gpLCB5KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHgxID0geDIgKyAxOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgc3RhdGljIHZvaWQgYWRkWDFMaW5lU2VnKE11bHRpUmVjdEFyZWEgbXJhLCBpbnRbXSBsaW5lLCBpbnQgY3gsIGludCBjeSwgaW50IGIsIGludCBzdGFydCwgaW50IGZpbmlzaCkgewotICAgICAgICBpbnQgeDEgPSAwOwotICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgbGluZS5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgaW50IHgyID0gbGluZVtpXTsKLSAgICAgICAgICAgIGludCB5ID0gY3kgLSAoYiAtIGkpOwotICAgICAgICAgICAgaWYgKHgxIDw9IGZpbmlzaCAmJiB4MiA+PSBzdGFydCkgewotICAgICAgICAgICAgICAgIG1yYS5hZGRSZWN0KGN4ICsgTWF0aC5tYXgoeDEsIHN0YXJ0KSwgeSwgY3ggKyBNYXRoLm1pbih4MiwgZmluaXNoKSwgeSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB4MSA9IHgyICsgMTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHN0YXRpYyB2b2lkIGFkZFgyTGluZVNlZyhNdWx0aVJlY3RBcmVhIG1yYSwgaW50W10gbGluZSwgaW50IGN4LCBpbnQgY3ksIGludCBiLCBpbnQgc3RhcnQsIGludCBmaW5pc2gpIHsKLSAgICAgICAgaW50IHgxID0gMDsKLSAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IGxpbmUubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIGludCB4MiA9IGxpbmVbaV07Ci0gICAgICAgICAgICBpbnQgeSA9IGN5IC0gKGIgLSBpKTsKLSAgICAgICAgICAgIGlmICh4MSA8PSBmaW5pc2ggJiYgeDIgPj0gc3RhcnQpIHsKLSAgICAgICAgICAgICAgICBtcmEuYWRkUmVjdChjeCAtIE1hdGgubWluKHgyLCBmaW5pc2gpLCB5LCBjeCAtIE1hdGgubWF4KHgxLCBzdGFydCksIHkpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgeDEgPSB4MiArIDE7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBzdGF0aWMgdm9pZCBhZGRYM0xpbmVTZWcoTXVsdGlSZWN0QXJlYSBtcmEsIGludFtdIGxpbmUsIGludCBjeCwgaW50IGN5LCBpbnQgYiwgaW50IHN0YXJ0LCBpbnQgZmluaXNoKSB7Ci0gICAgICAgIGludCB4MSA9IDA7Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBsaW5lLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBpbnQgeDIgPSBsaW5lW2ldOwotICAgICAgICAgICAgaW50IHkgPSBjeSArIChiIC0gaSk7Ci0gICAgICAgICAgICBpZiAoeDEgPD0gZmluaXNoICYmIHgyID49IHN0YXJ0KSB7Ci0gICAgICAgICAgICAgICAgbXJhLmFkZFJlY3QoY3ggLSBNYXRoLm1pbih4MiwgZmluaXNoKSwgeSwgY3ggLSBNYXRoLm1heCh4MSwgc3RhcnQpLCB5KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHgxID0geDIgKyAxOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgc3RhdGljIHZvaWQgYWRkWTBMaW5lU2VnKE11bHRpUmVjdEFyZWEgbXJhLCBpbnRbXSBsaW5lLCBpbnQgY3gsIGludCBjeSwgaW50IGIsIGludCBzdGFydCwgaW50IGZpbmlzaCkgewotICAgICAgICBpbnQgeTEgPSAwOwotICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgbGluZS5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgaW50IHggPSBjeCArIChiIC0gaSk7Ci0gICAgICAgICAgICBpbnQgeTIgPSBsaW5lW2ldOwotICAgICAgICAgICAgaWYgKHkxIDw9IGZpbmlzaCAmJiB5MiA+PSBzdGFydCkgewotICAgICAgICAgICAgICAgIG1yYS5hZGRSZWN0KHgsIGN5ICsgTWF0aC5tYXgoeTEsIHN0YXJ0KSwgeCwgY3kgKyBNYXRoLm1pbih5MiwgZmluaXNoKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB5MSA9IHkyICsgMTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHN0YXRpYyB2b2lkIGFkZFkxTGluZVNlZyhNdWx0aVJlY3RBcmVhIG1yYSwgaW50W10gbGluZSwgaW50IGN4LCBpbnQgY3ksIGludCBiLCBpbnQgc3RhcnQsIGludCBmaW5pc2gpIHsKLSAgICAgICAgaW50IHkxID0gMDsKLSAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IGxpbmUubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIGludCB4ID0gY3ggLSAoYiAtIGkpOwotICAgICAgICAgICAgaW50IHkyID0gbGluZVtpXTsKLSAgICAgICAgICAgIGlmICh5MSA8PSBmaW5pc2ggJiYgeTIgPj0gc3RhcnQpIHsKLSAgICAgICAgICAgICAgICBtcmEuYWRkUmVjdCh4LCBjeSArIE1hdGgubWF4KHkxLCBzdGFydCksIHgsIGN5ICsgTWF0aC5taW4oeTIsIGZpbmlzaCkpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgeTEgPSB5MiArIDE7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBzdGF0aWMgdm9pZCBhZGRZMkxpbmVTZWcoTXVsdGlSZWN0QXJlYSBtcmEsIGludFtdIGxpbmUsIGludCBjeCwgaW50IGN5LCBpbnQgYiwgaW50IHN0YXJ0LCBpbnQgZmluaXNoKSB7Ci0gICAgICAgIGludCB5MSA9IDA7Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBsaW5lLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBpbnQgeCA9IGN4IC0gKGIgLSBpKTsKLSAgICAgICAgICAgIGludCB5MiA9IGxpbmVbaV07Ci0gICAgICAgICAgICBpZiAoeTEgPD0gZmluaXNoICYmIHkyID49IHN0YXJ0KSB7Ci0gICAgICAgICAgICAgICAgbXJhLmFkZFJlY3QoeCwgY3kgLSBNYXRoLm1pbih5MiwgZmluaXNoKSwgeCwgY3kgLSBNYXRoLm1heCh5MSwgc3RhcnQpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHkxID0geTIgKyAxOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgc3RhdGljIHZvaWQgYWRkWTNMaW5lU2VnKE11bHRpUmVjdEFyZWEgbXJhLCBpbnRbXSBsaW5lLCBpbnQgY3gsIGludCBjeSwgaW50IGIsIGludCBzdGFydCwgaW50IGZpbmlzaCkgewotICAgICAgICBpbnQgeTEgPSAwOwotICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgbGluZS5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgaW50IHggPSBjeCArIChiIC0gaSk7Ci0gICAgICAgICAgICBpbnQgeTIgPSBsaW5lW2ldOwotICAgICAgICAgICAgaWYgKHkxIDw9IGZpbmlzaCAmJiB5MiA+PSBzdGFydCkgewotICAgICAgICAgICAgICAgIG1yYS5hZGRSZWN0KHgsIGN5IC0gTWF0aC5taW4oeTIsIGZpbmlzaCksIHgsIGN5IC0gTWF0aC5tYXgoeTEsIHN0YXJ0KSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB5MSA9IHkyICsgMTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHN0YXRpYyB2b2lkIGFkZFgwTGluZShNdWx0aVJlY3RBcmVhIG1yYSwgaW50W10gbGluZSwgaW50IGN4LCBpbnQgY3ksIGludCBiKSB7Ci0gICAgICAgIGludCBwcmV2ID0gMDsKLSAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IGxpbmUubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIG1yYS5hZGRSZWN0KGN4ICsgcHJldiwgY3kgKyAoYiAtIGkpLCBjeCArIGxpbmVbaV0sIGN5ICsgKGIgLSBpKSk7Ci0gICAgICAgICAgICBwcmV2ID0gbGluZVtpXSArIDE7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBzdGF0aWMgdm9pZCBhZGRYMUxpbmUoTXVsdGlSZWN0QXJlYSBtcmEsIGludFtdIGxpbmUsIGludCBjeCwgaW50IGN5LCBpbnQgYikgewotICAgICAgICBpbnQgcHJldiA9IDA7Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBsaW5lLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBtcmEuYWRkUmVjdChjeCArIHByZXYsIGN5IC0gKGIgLSBpKSwgY3ggKyBsaW5lW2ldLCBjeSAtIChiIC0gaSkpOwotICAgICAgICAgICAgcHJldiA9IGxpbmVbaV0gKyAxOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgc3RhdGljIHZvaWQgYWRkWDJMaW5lKE11bHRpUmVjdEFyZWEgbXJhLCBpbnRbXSBsaW5lLCBpbnQgY3gsIGludCBjeSwgaW50IGIpIHsKLSAgICAgICAgaW50IHByZXYgPSAwOwotICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgbGluZS5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgbXJhLmFkZFJlY3QoY3ggLSBsaW5lW2ldLCBjeSAtIChiIC0gaSksIGN4IC0gcHJldiwgY3kgLSAoYiAtIGkpKTsKLSAgICAgICAgICAgIHByZXYgPSBsaW5lW2ldICsgMTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHN0YXRpYyB2b2lkIGFkZFgzTGluZShNdWx0aVJlY3RBcmVhIG1yYSwgaW50W10gbGluZSwgaW50IGN4LCBpbnQgY3ksIGludCBiKSB7Ci0gICAgICAgIGludCBwcmV2ID0gMDsKLSAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IGxpbmUubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIG1yYS5hZGRSZWN0KGN4IC0gbGluZVtpXSwgY3kgKyAoYiAtIGkpLCBjeCAtIHByZXYsIGN5ICsgKGIgLSBpKSk7Ci0gICAgICAgICAgICBwcmV2ID0gbGluZVtpXSArIDE7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBzdGF0aWMgdm9pZCBhZGRZMExpbmUoTXVsdGlSZWN0QXJlYSBtcmEsIGludFtdIGxpbmUsIGludCBjeCwgaW50IGN5LCBpbnQgYSkgewotICAgICAgICBpbnQgcHJldiA9IDA7Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBsaW5lLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBtcmEuYWRkUmVjdChjeCArIChhIC0gaSksIGN5ICsgcHJldiwgY3ggKyAoYSAtIGkpLCBjeSArIGxpbmVbaV0pOwotICAgICAgICAgICAgcHJldiA9IGxpbmVbaV0gKyAxOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgc3RhdGljIHZvaWQgYWRkWTFMaW5lKE11bHRpUmVjdEFyZWEgbXJhLCBpbnRbXSBsaW5lLCBpbnQgY3gsIGludCBjeSwgaW50IGEpIHsKLSAgICAgICAgaW50IHByZXYgPSAwOwotICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgbGluZS5sZW5ndGg7IGkrKykgewotICAgICAgICAgICAgbXJhLmFkZFJlY3QoY3ggLSAoYSAtIGkpLCBjeSArIHByZXYsIGN4IC0gKGEgLSBpKSwgY3kgKyBsaW5lW2ldKTsKLSAgICAgICAgICAgIHByZXYgPSBsaW5lW2ldICsgMTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHN0YXRpYyB2b2lkIGFkZFkyTGluZShNdWx0aVJlY3RBcmVhIG1yYSwgaW50W10gbGluZSwgaW50IGN4LCBpbnQgY3ksIGludCBhKSB7Ci0gICAgICAgIGludCBwcmV2ID0gMDsKLSAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IGxpbmUubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgIG1yYS5hZGRSZWN0KGN4IC0gKGEgLSBpKSwgY3kgLSBsaW5lW2ldLCBjeCAtIChhIC0gaSksIGN5IC0gcHJldik7Ci0gICAgICAgICAgICBwcmV2ID0gbGluZVtpXSArIDE7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBzdGF0aWMgdm9pZCBhZGRZM0xpbmUoTXVsdGlSZWN0QXJlYSBtcmEsIGludFtdIGxpbmUsIGludCBjeCwgaW50IGN5LCBpbnQgYSkgewotICAgICAgICBpbnQgcHJldiA9IDA7Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBsaW5lLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICBtcmEuYWRkUmVjdChjeCArIChhIC0gaSksIGN5IC0gbGluZVtpXSwgY3ggKyAoYSAtIGkpLCBjeSAtIHByZXYpOwotICAgICAgICAgICAgcHJldiA9IGxpbmVbaV0gKyAxOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBub3JtYWxpemVkIGFuZ2xlIChmcm9tIDAgdG8gMzYwIGRlZ3JlZXMpCi0gICAgICovCi0gICAgc3RhdGljIGRvdWJsZSBnZXROb3JtQW5nbGUoZG91YmxlIGFuZ2xlKSB7Ci0gICAgICAgIGFuZ2xlIC09IE1hdGguZmxvb3IoYW5nbGUgLyAzNjApICogMzYwOwotICAgICAgICBpZiAoYW5nbGUgPCAwKSB7Ci0gICAgICAgICAgICBhbmdsZSArPSAzNjA7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGFuZ2xlOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYXJjIGxvb2t1cCB0YWJsZQotICAgICAqLwotICAgIHN0YXRpYyBpbnRbXSBjcmVhdGVMaW5lKGludCBhLCBpbnQgYiwgaW50IHhjb3VudCwgaW50IHljb3VudCkgewotICAgICAgICBpbnRbXSBidWYgPSBuZXcgaW50W2IgLSB5Y291bnQgKyAxXTsKLSAgICAgICAgaW50IGQgPSBhICogYSArIDIgKiBiICogYiAtIDIgKiBhICogYSAqIGI7Ci0gICAgICAgIGludCB4ID0gMDsKLSAgICAgICAgaW50IHkgPSBiOwotICAgICAgICB3aGlsZSAoeSA+PSB5Y291bnQpIHsKLSAgICAgICAgICAgIGlmIChkIDwgMCkgewotICAgICAgICAgICAgICAgIGQgPSBkICsgYiAqIGIgKiAoNCAqIHggKyA2KTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgYnVmW2IgLSB5XSA9IHg7Ci0gICAgICAgICAgICAgICAgZCA9IGQgKyBiICogYiAqICg0ICogeCArIDYpICsgNCAqIGEgKiBhICogKDEgLSB5KTsKLSAgICAgICAgICAgICAgICB5LS07Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB4Kys7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGJ1ZjsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBBZGRzIGhlYWQvdGFpbCBhcmMgc2VnbWVudCB0byBNdWx0aVJlY3RBcmVhCi0gICAgICovCi0gICAgc3RhdGljIHZvaWQgYWRkU2VnKE11bHRpUmVjdEFyZWEgbXJhLCBpbnQgY3gxLCBpbnQgY3kxLCBpbnQgY3gyLCBpbnQgY3kyLCBpbnQgYSwgaW50IGIsIGludFtdIHhsaW5lLCBpbnRbXSB5bGluZSwgaW50W10gYm91bmRzKSB7Ci0gICAgICAgIHN3aXRjaChib3VuZHNbMF0pIHsKLSAgICAgICAgY2FzZSAwOgotICAgICAgICAgICAgYWRkWTNMaW5lU2VnKG1yYSwgeWxpbmUsIGN4MiwgY3kxLCBhLCBib3VuZHNbMV0sIGJvdW5kc1syXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSAxOgotICAgICAgICAgICAgYWRkWDFMaW5lU2VnKG1yYSwgeGxpbmUsIGN4MiwgY3kxLCBiLCBib3VuZHNbMV0sIGJvdW5kc1syXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSAyOgotICAgICAgICAgICAgYWRkWDJMaW5lU2VnKG1yYSwgeGxpbmUsIGN4MSwgY3kxLCBiLCBib3VuZHNbMV0sIGJvdW5kc1syXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSAzOgotICAgICAgICAgICAgYWRkWTJMaW5lU2VnKG1yYSwgeWxpbmUsIGN4MSwgY3kxLCBhLCBib3VuZHNbMV0sIGJvdW5kc1syXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSA0OgotICAgICAgICAgICAgYWRkWTFMaW5lU2VnKG1yYSwgeWxpbmUsIGN4MSwgY3kyLCBhLCBib3VuZHNbMV0sIGJvdW5kc1syXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSA1OgotICAgICAgICAgICAgYWRkWDNMaW5lU2VnKG1yYSwgeGxpbmUsIGN4MSwgY3kyLCBiLCBib3VuZHNbMV0sIGJvdW5kc1syXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSA2OgotICAgICAgICAgICAgYWRkWDBMaW5lU2VnKG1yYSwgeGxpbmUsIGN4MiwgY3kyLCBiLCBib3VuZHNbMV0sIGJvdW5kc1syXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSA3OgotICAgICAgICAgICAgYWRkWTBMaW5lU2VnKG1yYSwgeWxpbmUsIGN4MiwgY3kyLCBhLCBib3VuZHNbMV0sIGJvdW5kc1syXSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgYm91bmRzIGZvciBub24gcXVhZHJhdGljIGFyYyBoZWFkCi0gICAgICovCi0gICAgc3RhdGljIGludFtdIGdldFNlZ21lbnQxKGRvdWJsZSBhbmdsZSwgaW50IGF4LCBpbnQgYXksIGludCB4Y291bnQsIGludCB5Y291bnQpIHsKLSAgICAgICAgaW50W10gYm91bmRzID0gbmV3IGludFszXTsKLSAgICAgICAgc3dpdGNoKChpbnQpKGFuZ2xlIC8gOTApKSB7Ci0gICAgICAgIGNhc2UgMDoKLSAgICAgICAgICAgIGlmICh4Y291bnQgPCAgYXgpIHsKLSAgICAgICAgICAgICAgICBib3VuZHNbMF0gPSAwOyAvLyBZMwotICAgICAgICAgICAgICAgIGJvdW5kc1sxXSA9IC1heTsKLSAgICAgICAgICAgICAgICBib3VuZHNbMl0gPSB5Y291bnQ7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGJvdW5kc1swXSA9IDE7IC8vIFgxCi0gICAgICAgICAgICAgICAgYm91bmRzWzFdID0gMDsKLSAgICAgICAgICAgICAgICBib3VuZHNbMl0gPSBheDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIDE6Ci0gICAgICAgICAgICBpZiAoeGNvdW50ID4gLWF4KSB7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzBdID0gMjsgLy8gWDIKLSAgICAgICAgICAgICAgICBib3VuZHNbMV0gPSAtYXg7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzJdID0geGNvdW50OwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBib3VuZHNbMF0gPSAzOyAvLyBZMgotICAgICAgICAgICAgICAgIGJvdW5kc1sxXSA9IDA7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzJdID0gLWF5OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgMjoKLSAgICAgICAgICAgIGlmICh4Y291bnQgPCAtYXgpIHsKLSAgICAgICAgICAgICAgICBib3VuZHNbMF0gPSA0OyAvLyBZMQotICAgICAgICAgICAgICAgIGJvdW5kc1sxXSA9IGF5OwotICAgICAgICAgICAgICAgIGJvdW5kc1syXSA9IHljb3VudDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzBdID0gNTsgLy8gWDMKLSAgICAgICAgICAgICAgICBib3VuZHNbMV0gPSAwOwotICAgICAgICAgICAgICAgIGJvdW5kc1syXSA9IC1heDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIDM6Ci0gICAgICAgICAgICBpZiAoeGNvdW50ID4gIGF4KSB7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzBdID0gNjsgLy8gWDAKLSAgICAgICAgICAgICAgICBib3VuZHNbMV0gPSBheDsKLSAgICAgICAgICAgICAgICBib3VuZHNbMl0gPSB4Y291bnQ7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGJvdW5kc1swXSA9IDc7IC8vIFkwCi0gICAgICAgICAgICAgICAgYm91bmRzWzFdID0gMDsKLSAgICAgICAgICAgICAgICBib3VuZHNbMl0gPSBheTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBib3VuZHM7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBib3VuZHMgZm9yIG5vbiBxdWFkcmF0aWMgYXJjIHRhaWwKLSAgICAgKi8KLSAgICBzdGF0aWMgaW50W10gZ2V0U2VnbWVudDIoZG91YmxlIGFuZ2xlLCBpbnQgYXgsIGludCBheSwgaW50IHhjb3VudCwgaW50IHljb3VudCkgewotICAgICAgICBpbnRbXSBib3VuZHMgPSBuZXcgaW50WzNdOwotICAgICAgICBzd2l0Y2goKGludCkoYW5nbGUgLyA5MCkpIHsKLSAgICAgICAgY2FzZSAwOgotICAgICAgICAgICAgaWYgKHhjb3VudCA8ICBheCkgewotICAgICAgICAgICAgICAgIGJvdW5kc1swXSA9IDA7IC8vIFkzCi0gICAgICAgICAgICAgICAgYm91bmRzWzFdID0gMDsKLSAgICAgICAgICAgICAgICBib3VuZHNbMl0gPSAtYXk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGJvdW5kc1swXSA9IDE7IC8vIFgxCi0gICAgICAgICAgICAgICAgYm91bmRzWzFdID0gYXg7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzJdID0geGNvdW50OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgMToKLSAgICAgICAgICAgIGlmICh4Y291bnQgPiAtYXgpIHsKLSAgICAgICAgICAgICAgICBib3VuZHNbMF0gPSAyOyAvLyBYMgotICAgICAgICAgICAgICAgIGJvdW5kc1sxXSA9IDA7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzJdID0gLWF4OwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBib3VuZHNbMF0gPSAzOyAvLyBZMgotICAgICAgICAgICAgICAgIGJvdW5kc1sxXSA9IC1heTsKLSAgICAgICAgICAgICAgICBib3VuZHNbMl0gPSB5Y291bnQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSAyOgotICAgICAgICAgICAgaWYgKHhjb3VudCA8IC1heCkgewotICAgICAgICAgICAgICAgIGJvdW5kc1swXSA9IDQ7IC8vIFkxCi0gICAgICAgICAgICAgICAgYm91bmRzWzFdID0gMDsKLSAgICAgICAgICAgICAgICBib3VuZHNbMl0gPSBheTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzBdID0gNTsgLy8gWDMKLSAgICAgICAgICAgICAgICBib3VuZHNbMV0gPSAtYXg7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzJdID0geGNvdW50OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgMzoKLSAgICAgICAgICAgIGlmICh4Y291bnQgPiAgYXgpIHsKLSAgICAgICAgICAgICAgICBib3VuZHNbMF0gPSA2OyAvLyBYMAotICAgICAgICAgICAgICAgIGJvdW5kc1sxXSA9IDA7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzJdID0gYXg7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGJvdW5kc1swXSA9IDc7IC8vIFkwCi0gICAgICAgICAgICAgICAgYm91bmRzWzFdID0gYXk7Ci0gICAgICAgICAgICAgICAgYm91bmRzWzJdID0geWNvdW50OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGJvdW5kczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSYXN0ZXJpemVzIGFyYyB1c2luZyBjbGlwcGluZCBhbmQgZGFzaGluZyBzdHlsZQotICAgICAqIEBwYXJhbSB4MSAtIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIGxlZnQtdXBwZXIgY29ybmVyIG9mIHRoZSBhcmMgYm91bmRzCi0gICAgICogQHBhcmFtIHkxIC0gdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgbGVmdC11cHBlciBjb3JuZXIgb2YgdGhlIGFyYyBib3VuZHMKLSAgICAgKiBAcGFyYW0gd2lkdGggLSB0aGUgd2lkdGggb2YgdGhlIGFyYyBib3VuZHMKLSAgICAgKiBAcGFyYW0gaGVpZ2h0IC0gdGhlIGhlaWdodCBvZiB0aGUgYXJjIGJvdW5kcwotICAgICAqIEBwYXJhbSBhbmdsZVN0YXJ0IC0gdGhlIHN0YXJ0IGFuZ2xlIG9mIHRoZSBhcmMgaW4gZGVncmVlcwotICAgICAqIEBwYXJhbSBhbmdsZUV4dGVudCAtIHRoZSBhbmdsZSBleHRlbnQgaW4gZGVncmVlcwotICAgICAqIEBwYXJhbSBjbGlwIC0gdGhlIE11bHRpUmVjdEFyZWEgb2JqZWN0IG9mIGNsaXBwaW5nIGFyZWEKLSAgICAgKiBAcmV0dXJuIGEgTXVsdGlSZWN0QXJlYSBvZiByYXN0ZXJpemVyIGFyYwotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgTXVsdGlSZWN0QXJlYSByYXN0ZXJpemUoaW50IHgsIGludCB5LCBpbnQgd2lkdGgsIGludCBoZWlnaHQsIGRvdWJsZSBhbmdsZVN0YXJ0LCBkb3VibGUgYW5nbGVFeHRlbnQsIE11bHRpUmVjdEFyZWEgY2xpcCkgewotCi0gICAgICAgIE11bHRpUmVjdEFyZWEgbXJhID0gbmV3IE11bHRpUmVjdEFyZWEoZmFsc2UpOwotCi0gICAgICAgIGludCBjeDEsIGN4MiwgY3kxLCBjeTI7Ci0gICAgICAgIGN4MSA9IGN4MiA9IHggKyB3aWR0aCAvIDI7Ci0gICAgICAgIGN5MSA9IGN5MiA9IHkgKyBoZWlnaHQgLyAyOwotCi0gICAgICAgIGlmICh3aWR0aCAlIDIgPT0gMCkgewotICAgICAgICAgICAgY3gyLS07Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoaGVpZ2h0ICUgMiA9PSAwKSB7Ci0gICAgICAgICAgICBjeTItLTsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBhID0gd2lkdGggLyAyOwotICAgICAgICBpbnQgYiA9IGhlaWdodCAvIDI7Ci0gICAgICAgIGRvdWJsZSBjID0gTWF0aC5zcXJ0KGEgKiBhICsgYiAqIGIpOwotCi0gICAgICAgIGludCB4Y291bnQsIHljb3VudDsKLSAgICAgICAgaWYgKGEgPCBiKSB7Ci0gICAgICAgICAgICB4Y291bnQgPSAoaW50KU1hdGguY2VpbChhICogYSAvIGMpOwotICAgICAgICAgICAgeWNvdW50ID0gKGludClNYXRoLmZsb29yKGIgKiBiIC8gYyk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICB4Y291bnQgPSAoaW50KU1hdGguZmxvb3IoYSAqIGEgLyBjKTsKLSAgICAgICAgICAgIHljb3VudCA9IChpbnQpTWF0aC5jZWlsKGIgKiBiIC8gYyk7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnRbXSB4bGluZSA9IGNyZWF0ZUxpbmUoYSwgYiwgeGNvdW50LCB5Y291bnQpOwotICAgICAgICBpbnRbXSB5bGluZSA9IGNyZWF0ZUxpbmUoYiwgYSwgeWNvdW50LCB4Y291bnQpOwotCi0gICAgICAgIC8vIENvcnJlY3QgbGluZXMKLSAgICAgICAgaW50IGkgPSB4bGluZS5sZW5ndGg7Ci0gICAgICAgIHdoaWxlKHhsaW5lWy0taV0gPiB4Y291bnQpIHsKLSAgICAgICAgICAgIHhsaW5lW2ldID0geGNvdW50OwotICAgICAgICB9Ci0KLSAgICAgICAgaSA9IHlsaW5lLmxlbmd0aDsKLSAgICAgICAgd2hpbGUoeWxpbmVbLS1pXSA+IHljb3VudCkgewotICAgICAgICAgICAgeWxpbmVbaV0gPSB5Y291bnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoTWF0aC5hYnMoYW5nbGVFeHRlbnQpID49IDM2MCkgewotICAgICAgICAgICAgLy8gUmFzdGVyaXplIENJUkNMRQotICAgICAgICAgICAgYWRkWDBMaW5lKG1yYSwgeGxpbmUsIGN4MiwgY3kyLCBiKTsKLSAgICAgICAgICAgIGFkZFgxTGluZShtcmEsIHhsaW5lLCBjeDIsIGN5MSwgYik7Ci0gICAgICAgICAgICBhZGRYMkxpbmUobXJhLCB4bGluZSwgY3gxLCBjeTEsIGIpOwotICAgICAgICAgICAgYWRkWDNMaW5lKG1yYSwgeGxpbmUsIGN4MSwgY3kyLCBiKTsKLSAgICAgICAgICAgIGFkZFkwTGluZShtcmEsIHlsaW5lLCBjeDIsIGN5MiwgYSk7Ci0gICAgICAgICAgICBhZGRZMUxpbmUobXJhLCB5bGluZSwgY3gxLCBjeTIsIGEpOwotICAgICAgICAgICAgYWRkWTJMaW5lKG1yYSwgeWxpbmUsIGN4MSwgY3kxLCBhKTsKLSAgICAgICAgICAgIGFkZFkzTGluZShtcmEsIHlsaW5lLCBjeDIsIGN5MSwgYSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBSYXN0ZXJpemUgQVJDCi0gICAgICAgICAgICBhbmdsZVN0YXJ0ID0gZ2V0Tm9ybUFuZ2xlKGFuZ2xlU3RhcnQpOwotICAgICAgICAgICAgZG91YmxlIGFuZ2xlRmluaXNoID0gZ2V0Tm9ybUFuZ2xlKGFuZ2xlU3RhcnQgKyBhbmdsZUV4dGVudCk7Ci0KLSAgICAgICAgICAgIGlmIChhbmdsZUV4dGVudCA8IDApIHsKLSAgICAgICAgICAgICAgICBkb3VibGUgdG1wID0gYW5nbGVTdGFydDsKLSAgICAgICAgICAgICAgICBhbmdsZVN0YXJ0ID0gYW5nbGVGaW5pc2g7Ci0gICAgICAgICAgICAgICAgYW5nbGVGaW5pc2ggPSB0bXA7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGRvdWJsZSByYWRTdGFydCA9IC1NYXRoLnRvUmFkaWFucyhhbmdsZVN0YXJ0KTsKLSAgICAgICAgICAgIGRvdWJsZSByYWRGaW5pc2ggPSAtTWF0aC50b1JhZGlhbnMoYW5nbGVGaW5pc2gpOwotICAgICAgICAgICAgaW50IGF4MSA9IChpbnQpKGEgKiBNYXRoLmNvcyhyYWRTdGFydCkpOwotICAgICAgICAgICAgaW50IGF5MSA9IChpbnQpKGIgKiBNYXRoLnNpbihyYWRTdGFydCkpOwotICAgICAgICAgICAgaW50IGF4MiA9IChpbnQpKGEgKiBNYXRoLmNvcyhyYWRGaW5pc2gpKTsKLSAgICAgICAgICAgIGludCBheTIgPSAoaW50KShiICogTWF0aC5zaW4ocmFkRmluaXNoKSk7Ci0KLSAgICAgICAgICAgIGludFtdIHNlZzEgPSBnZXRTZWdtZW50MShhbmdsZVN0YXJ0LCBheDEsIGF5MSwgeGNvdW50LCB5Y291bnQpOwotICAgICAgICAgICAgaW50W10gc2VnMiA9IGdldFNlZ21lbnQyKGFuZ2xlRmluaXNoLCBheDIsIGF5MiwgeGNvdW50LCB5Y291bnQpOwotCi0gICAgICAgICAgICAvLyBTdGFydCBhbmQgRmluaXNoIGxvY2F0ZWQgaW4gdGhlIHNhbWUgcXVhdGVyCi0gICAgICAgICAgICBpZiAoYW5nbGVTdGFydCA8IGFuZ2xlRmluaXNoICYmIHNlZzFbMF0gPT0gc2VnMlswXSkgewotICAgICAgICAgICAgICAgIGlmIChzZWcxWzBdICUgMiA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHNlZzFbMl0gPSBzZWcyWzJdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHNlZzFbMV0gPSBzZWcyWzFdOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBhZGRTZWcobXJhLCBjeDEsIGN5MSwgY3gyLCBjeTIsIGEsIGIsIHhsaW5lLCB5bGluZSwgc2VnMSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG1yYTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgYWRkU2VnKG1yYSwgY3gxLCBjeTEsIGN4MiwgY3kyLCBhLCBiLCB4bGluZSwgeWxpbmUsIHNlZzEpOwotICAgICAgICAgICAgYWRkU2VnKG1yYSwgY3gxLCBjeTEsIGN4MiwgY3kyLCBhLCBiLCB4bGluZSwgeWxpbmUsIHNlZzIpOwotCi0gICAgICAgICAgICBpbnQgc3RhcnRTZWcgPSAoc2VnMVswXSArIDEpICUgODsKLSAgICAgICAgICAgIGludCBmaW5pc2hTZWcgPSBzZWcyWzBdOwotCi0gICAgICAgICAgICB3aGlsZSAoc3RhcnRTZWcgIT0gZmluaXNoU2VnKSB7Ci0gICAgICAgICAgICAgICAgc3dpdGNoKHN0YXJ0U2VnKSB7Ci0gICAgICAgICAgICAgICAgY2FzZSAwOgotICAgICAgICAgICAgICAgICAgICBhZGRZM0xpbmUobXJhLCB5bGluZSwgY3gyLCBjeTEsIGEpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIDE6Ci0gICAgICAgICAgICAgICAgICAgIGFkZFgxTGluZShtcmEsIHhsaW5lLCBjeDIsIGN5MSwgYik7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgMjoKLSAgICAgICAgICAgICAgICAgICAgYWRkWDJMaW5lKG1yYSwgeGxpbmUsIGN4MSwgY3kxLCBiKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSAzOgotICAgICAgICAgICAgICAgICAgICBhZGRZMkxpbmUobXJhLCB5bGluZSwgY3gxLCBjeTEsIGEpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIDQ6Ci0gICAgICAgICAgICAgICAgICAgIGFkZFkxTGluZShtcmEsIHlsaW5lLCBjeDEsIGN5MiwgYSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIGNhc2UgNToKLSAgICAgICAgICAgICAgICAgICAgYWRkWDNMaW5lKG1yYSwgeGxpbmUsIGN4MSwgY3kyLCBiKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSA2OgotICAgICAgICAgICAgICAgICAgICBhZGRYMExpbmUobXJhLCB4bGluZSwgY3gyLCBjeTIsIGIpOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICBjYXNlIDc6Ci0gICAgICAgICAgICAgICAgICAgIGFkZFkwTGluZShtcmEsIHlsaW5lLCBjeDIsIGN5MiwgYSk7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBzdGFydFNlZyA9IChzdGFydFNlZyArIDEpICUgODsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChjbGlwICE9IG51bGwpIHsKLSAgICAgICAgICAgIG1yYS5pbnRlcnNlY3QoY2xpcCk7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbXJhOwotICAgIH0KLQotfQpcIE5vIG5ld2xpbmUgYXQgZW5kIG9mIGZpbGUKZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL3JlbmRlci9KYXZhQmxpdHRlci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvcmVuZGVyL0phdmFCbGl0dGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDY3ZTBhNTkuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvcmVuZGVyL0phdmFCbGl0dGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw2MTEgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBJZ29yIFYuIFN0b2x5YXJvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICogQ3JlYXRlZCBvbiAxOC4xMS4yMDA1Ci0gKgotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wucmVuZGVyOwotCi1pbXBvcnQgamF2YS5hd3QuQWxwaGFDb21wb3NpdGU7Ci1pbXBvcnQgamF2YS5hd3QuQ29sb3I7Ci1pbXBvcnQgamF2YS5hd3QuQ29tcG9zaXRlOwotaW1wb3J0IGphdmEuYXd0LkNvbXBvc2l0ZUNvbnRleHQ7Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uQWZmaW5lVHJhbnNmb3JtOwotaW1wb3J0IGphdmEuYXd0Lmdlb20uTm9uaW52ZXJ0aWJsZVRyYW5zZm9ybUV4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlJlY3RhbmdsZTJEOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkNvbG9yTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuUmFzdGVyOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLldyaXRhYmxlUmFzdGVyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5NdWx0aVJlY3RBcmVhOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuU3VyZmFjZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLlhPUkNvbXBvc2l0ZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBKYXZhIGltcGxlbmV0YXRpb24gb2YgdGhlIEJsaXR0ZXIgaW50ZXJmYWNlLiBVc2luZyB3aGVuIHdlIGNhbid0IAotICogZHJhdyBpbWFnZXMgbmF0aXZlbHkuCi0gKi8KLXB1YmxpYyBjbGFzcyBKYXZhQmxpdHRlciBpbXBsZW1lbnRzIEJsaXR0ZXIgewotCi0gICAgLyoqCi0gICAgICogSW5zdGVhZCBvZiBtdWx0aXBsaWNhdGlvbiBhbmQgZGl2aXNpb24gd2UgYXJlIHVzaW5nIHZhbHVlcyBmcm9tCi0gICAgICogTG9va3VwIHRhYmxlcy4KLSAgICAgKi8KLSAgICBzdGF0aWMgYnl0ZSBtdWxMVVRbXVtdOyAvLyBMb29rdXAgdGFibGUgZm9yIG11bHRpcGxpY2F0aW9uCi0gICAgc3RhdGljIGJ5dGUgZGl2TFVUW11bXTsgLy8gTG9va3VwIHRhYmxlIGZvciBkaXZpc2lvbgotCi0gICAgc3RhdGljewotICAgICAgICBtdWxMVVQgPSBuZXcgYnl0ZVsyNTZdWzI1Nl07Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCAyNTY7IGkrKyl7Ci0gICAgICAgICAgICBmb3IoaW50IGogPSAwOyBqIDwgMjU2OyBqKyspewotICAgICAgICAgICAgICAgIG11bExVVFtpXVtqXSA9IChieXRlKSgoZmxvYXQpKGkgKiBqKS8yNTUgKyAwLjVmKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBkaXZMVVQgPSBuZXcgYnl0ZVsyNTZdWzI1Nl07Ci0gICAgICAgIGZvcihpbnQgaSA9IDE7IGkgPCAyNTY7IGkrKyl7Ci0gICAgICAgICAgICBmb3IoaW50IGogPSAwOyBqIDwgaTsgaisrKXsKLSAgICAgICAgICAgICAgICBkaXZMVVRbaV1bal0gPSAoYnl0ZSkoKChmbG9hdClqIC8gMjU1KSAvICgoZmxvYXQpaS8gMjU1KSAqIDI1NSArIDAuNWYpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZm9yKGludCBqID0gaTsgaiA8IDI1NjsgaisrKXsKLSAgICAgICAgICAgICAgICBkaXZMVVRbaV1bal0gPSAoYnl0ZSkyNTU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBmaW5hbCBzdGF0aWMgaW50IEFscGhhQ29tcG9zaXRlTW9kZSA9IDE7Ci0gICAgZmluYWwgc3RhdGljIGludCBYT1JNb2RlID0gMjsKLQotICAgIGZpbmFsIHN0YXRpYyBKYXZhQmxpdHRlciBpbnN0ID0gbmV3IEphdmFCbGl0dGVyKCk7Ci0KLSAgICBwdWJsaWMgc3RhdGljIEphdmFCbGl0dGVyIGdldEluc3RhbmNlKCl7Ci0gICAgICAgIHJldHVybiBpbnN0OwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBBZmZpbmVUcmFuc2Zvcm0gc3lzeGZvcm0sCi0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0sIENvbXBvc2l0ZSBjb21wLCBDb2xvciBiZ2NvbG9yLAotICAgICAgICAgICAgTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0KLSAgICAgICAgaWYoeGZvcm0gPT0gbnVsbCl7Ci0gICAgICAgICAgICBibGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIGRzdFgsIGRzdFksIGRzdFN1cmYsIHdpZHRoLCBoZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgIHN5c3hmb3JtLCBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICBkb3VibGUgc2NhbGVYID0geGZvcm0uZ2V0U2NhbGVYKCk7Ci0gICAgICAgICAgICBkb3VibGUgc2NhbGVZID0geGZvcm0uZ2V0U2NhbGVZKCk7Ci0gICAgICAgICAgICBkb3VibGUgc2NhbGVkWCA9IGRzdFggLyBzY2FsZVg7Ci0gICAgICAgICAgICBkb3VibGUgc2NhbGVkWSA9IGRzdFkgLyBzY2FsZVk7Ci0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gYXQgPSBuZXcgQWZmaW5lVHJhbnNmb3JtKCk7Ci0gICAgICAgICAgICBhdC5zZXRUb1RyYW5zbGF0aW9uKHNjYWxlZFgsIHNjYWxlZFkpOwotICAgICAgICAgICAgeGZvcm0uY29uY2F0ZW5hdGUoYXQpOwotICAgICAgICAgICAgc3lzeGZvcm0uY29uY2F0ZW5hdGUoeGZvcm0pOwotICAgICAgICAgICAgYmxpdChzcmNYLCBzcmNZLCBzcmNTdXJmLCAwLCAwLCBkc3RTdXJmLCB3aWR0aCwgaGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICBzeXN4Zm9ybSwgY29tcCwgYmdjb2xvciwgY2xpcCk7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBBZmZpbmVUcmFuc2Zvcm0gc3lzeGZvcm0sCi0gICAgICAgICAgICBDb21wb3NpdGUgY29tcCwgQ29sb3IgYmdjb2xvciwgTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0KLSAgICAgICAgaWYoc3lzeGZvcm0gPT0gbnVsbCkgewotICAgICAgICAgICAgc3lzeGZvcm0gPSBuZXcgQWZmaW5lVHJhbnNmb3JtKCk7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IHR5cGUgPSBzeXN4Zm9ybS5nZXRUeXBlKCk7Ci0gICAgICAgIHN3aXRjaCh0eXBlKXsKLSAgICAgICAgICAgIGNhc2UgQWZmaW5lVHJhbnNmb3JtLlRZUEVfVFJBTlNMQVRJT046Ci0gICAgICAgICAgICAgICAgZHN0WCArPSBzeXN4Zm9ybS5nZXRUcmFuc2xhdGVYKCk7Ci0gICAgICAgICAgICAgICAgZHN0WSArPSBzeXN4Zm9ybS5nZXRUcmFuc2xhdGVZKCk7Ci0gICAgICAgICAgICBjYXNlIEFmZmluZVRyYW5zZm9ybS5UWVBFX0lERU5USVRZOgotICAgICAgICAgICAgICAgICBibGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIGRzdFgsIGRzdFksIGRzdFN1cmYsCi0gICAgICAgICAgICAgICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgaW50IHNyY1cgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgaW50IHNyY0ggPSBzcmNTdXJmLmdldEhlaWdodCgpOwotCi0gICAgICAgICAgICAgICAgaW50IHcgPSBzcmNYICsgd2lkdGggPCBzcmNXID8gd2lkdGggOiBzcmNXIC0gc3JjWDsKLSAgICAgICAgICAgICAgICBpbnQgaCA9IHNyY1kgKyBoZWlnaHQgPCBzcmNIID8gaGVpZ2h0IDogc3JjSCAtIHNyY1k7Ci0KLSAgICAgICAgICAgICAgICBDb2xvck1vZGVsIHNyY0NNID0gc3JjU3VyZi5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgICAgICAgICAgUmFzdGVyIHNyY1IgPSBzcmNTdXJmLmdldFJhc3RlcigpLmNyZWF0ZUNoaWxkKHNyY1gsIHNyY1ksCi0gICAgICAgICAgICAgICAgICAgICAgICB3LCBoLCAwLCAwLCBudWxsKTsKLQotICAgICAgICAgICAgICAgIENvbG9yTW9kZWwgZHN0Q00gPSBkc3RTdXJmLmdldENvbG9yTW9kZWwoKTsKLSAgICAgICAgICAgICAgICBXcml0YWJsZVJhc3RlciBkc3RSID0gZHN0U3VyZi5nZXRSYXN0ZXIoKTsKLQotICAgICAgICAgICAgICAgIHRyYW5zZm9ybWVkQmxpdChzcmNDTSwgc3JjUiwgMCwgMCwgZHN0Q00sIGRzdFIsIGRzdFgsIGRzdFksIHcsIGgsCi0gICAgICAgICAgICAgICAgICAgICAgICBzeXN4Zm9ybSwgY29tcCwgYmdjb2xvciwgY2xpcCk7Ci0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBDb21wb3NpdGUgY29tcCwKLSAgICAgICAgICAgIENvbG9yIGJnY29sb3IsIE11bHRpUmVjdEFyZWEgY2xpcCkgewotCi0gICAgICAgIGphdmFCbHQoc3JjWCwgc3JjWSwgc3JjU3VyZi5nZXRXaWR0aCgpLCBzcmNTdXJmLmdldEhlaWdodCgpLAotICAgICAgICAgICAgICAgIHNyY1N1cmYuZ2V0Q29sb3JNb2RlbCgpLCBzcmNTdXJmLmdldFJhc3RlcigpLCBkc3RYLCBkc3RZLAotICAgICAgICAgICAgICAgIGRzdFN1cmYuZ2V0V2lkdGgoKSwgZHN0U3VyZi5nZXRIZWlnaHQoKSwKLSAgICAgICAgICAgICAgICBkc3RTdXJmLmdldENvbG9yTW9kZWwoKSwgZHN0U3VyZi5nZXRSYXN0ZXIoKSwKLSAgICAgICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLQotICAgIH0KLSAgICBwdWJsaWMgdm9pZCBqYXZhQmx0KGludCBzcmNYLCBpbnQgc3JjWSwgaW50IHNyY1csIGludCBzcmNILAotICAgICAgICAgICAgQ29sb3JNb2RlbCBzcmNDTSwgUmFzdGVyIHNyY1Jhc3QsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIGludCBkc3RXLCBpbnQgZHN0SCwgQ29sb3JNb2RlbCBkc3RDTSwgV3JpdGFibGVSYXN0ZXIgZHN0UmFzdCwKLSAgICAgICAgICAgIGludCB3aWR0aCwgaW50IGhlaWdodCwgQ29tcG9zaXRlIGNvbXAsIENvbG9yIGJnY29sb3IsCi0gICAgICAgICAgICBNdWx0aVJlY3RBcmVhIGNsaXApewotCi0gICAgICAgIGludCBzcmNYMiA9IHNyY1cgLSAxOwotICAgICAgICBpbnQgc3JjWTIgPSBzcmNIIC0gMTsKLSAgICAgICAgaW50IGRzdFgyID0gZHN0VyAtIDE7Ci0gICAgICAgIGludCBkc3RZMiA9IGRzdEggLSAxOwotCi0gICAgICAgIGlmKHNyY1ggPCAwKXsKLSAgICAgICAgICAgIHdpZHRoICs9IHNyY1g7Ci0gICAgICAgICAgICBzcmNYID0gMDsKLSAgICAgICAgfQotICAgICAgICBpZihzcmNZIDwgMCl7Ci0gICAgICAgICAgICBoZWlnaHQgKz0gc3JjWTsKLSAgICAgICAgICAgIHNyY1kgPSAwOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYoZHN0WCA8IDApewotICAgICAgICAgICAgd2lkdGggKz0gZHN0WDsKLSAgICAgICAgICAgIHNyY1ggLT0gZHN0WDsKLSAgICAgICAgICAgIGRzdFggPSAwOwotICAgICAgICB9Ci0gICAgICAgIGlmKGRzdFkgPCAwKXsKLSAgICAgICAgICAgIGhlaWdodCArPSBkc3RZOwotICAgICAgICAgICAgc3JjWSAtPSBkc3RZOwotICAgICAgICAgICAgZHN0WSA9IDA7Ci0gICAgICAgIH0KLQotICAgICAgICBpZihzcmNYID4gc3JjWDIgfHwgc3JjWSA+IHNyY1kyKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgaWYoZHN0WCA+IGRzdFgyIHx8IGRzdFkgPiBkc3RZMikgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYoc3JjWCArIHdpZHRoID4gc3JjWDIpIHsKLSAgICAgICAgICAgIHdpZHRoID0gc3JjWDIgLSBzcmNYICsgMTsKLSAgICAgICAgfQotICAgICAgICBpZihzcmNZICsgaGVpZ2h0ID4gc3JjWTIpIHsKLSAgICAgICAgICAgIGhlaWdodCA9IHNyY1kyIC0gc3JjWSArIDE7Ci0gICAgICAgIH0KLSAgICAgICAgaWYoZHN0WCArIHdpZHRoID4gZHN0WDIpIHsKLSAgICAgICAgICAgIHdpZHRoID0gZHN0WDIgLSBkc3RYICsgMTsKLSAgICAgICAgfQotICAgICAgICBpZihkc3RZICsgaGVpZ2h0ID4gZHN0WTIpIHsKLSAgICAgICAgICAgIGhlaWdodCA9IGRzdFkyIC0gZHN0WSArIDE7Ci0gICAgICAgIH0KLQotICAgICAgICBpZih3aWR0aCA8PSAwIHx8IGhlaWdodCA8PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgY2xpcFJlY3RzW107Ci0gICAgICAgIGlmKGNsaXAgIT0gbnVsbCkgewotICAgICAgICAgICAgY2xpcFJlY3RzID0gY2xpcC5yZWN0OwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgY2xpcFJlY3RzID0gbmV3IGludFtdezUsIDAsIDAsIGRzdFcgLSAxLCBkc3RIIC0gMX07Ci0gICAgICAgIH0KLQotICAgICAgICBib29sZWFuIGlzQWxwaGFDb21wID0gZmFsc2U7Ci0gICAgICAgIGludCBydWxlID0gMDsKLSAgICAgICAgZmxvYXQgYWxwaGEgPSAwOwotICAgICAgICBib29sZWFuIGlzWE9SQ29tcCA9IGZhbHNlOwotICAgICAgICBDb2xvciB4b3Jjb2xvciA9IG51bGw7Ci0gICAgICAgIENvbXBvc2l0ZUNvbnRleHQgY29udCA9IG51bGw7Ci0KLSAgICAgICAgaWYoY29tcCBpbnN0YW5jZW9mIEFscGhhQ29tcG9zaXRlKXsKLSAgICAgICAgICAgIGlzQWxwaGFDb21wID0gdHJ1ZTsKLSAgICAgICAgICAgIEFscGhhQ29tcG9zaXRlIGFjID0gKEFscGhhQ29tcG9zaXRlKSBjb21wOwotICAgICAgICAgICAgcnVsZSA9IGFjLmdldFJ1bGUoKTsKLSAgICAgICAgICAgIGFscGhhID0gYWMuZ2V0QWxwaGEoKTsKLSAgICAgICAgfWVsc2UgaWYoY29tcCBpbnN0YW5jZW9mIFhPUkNvbXBvc2l0ZSl7Ci0gICAgICAgICAgICBpc1hPUkNvbXAgPSB0cnVlOwotICAgICAgICAgICAgWE9SQ29tcG9zaXRlIHhjb21wID0gKFhPUkNvbXBvc2l0ZSkgY29tcDsKLSAgICAgICAgICAgIHhvcmNvbG9yID0geGNvbXAuZ2V0WE9SQ29sb3IoKTsKLSAgICAgICAgfWVsc2V7Ci0gICAgICAgICAgICBjb250ID0gY29tcC5jcmVhdGVDb250ZXh0KHNyY0NNLCBkc3RDTSwgbnVsbCk7Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IoaW50IGkgPSAxOyBpIDwgY2xpcFJlY3RzWzBdOyBpICs9IDQpewotICAgICAgICAgICAgaW50IF9zeCA9IHNyY1g7Ci0gICAgICAgICAgICBpbnQgX3N5ID0gc3JjWTsKLQotICAgICAgICAgICAgaW50IF9keCA9IGRzdFg7Ci0gICAgICAgICAgICBpbnQgX2R5ID0gZHN0WTsKLQotICAgICAgICAgICAgaW50IF93ID0gd2lkdGg7Ci0gICAgICAgICAgICBpbnQgX2ggPSBoZWlnaHQ7Ci0KLSAgICAgICAgICAgIGludCBjeCA9IGNsaXBSZWN0c1tpXTsgICAgICAgICAgLy8gQ2xpcHBpbmcgbGVmdCB0b3AgWAotICAgICAgICAgICAgaW50IGN5ID0gY2xpcFJlY3RzW2kgKyAxXTsgICAgICAvLyBDbGlwcGluZyBsZWZ0IHRvcCBZCi0gICAgICAgICAgICBpbnQgY3gyID0gY2xpcFJlY3RzW2kgKyAyXTsgICAgIC8vIENsaXBwaW5nIHJpZ2h0IGJvdHRvbSBYCi0gICAgICAgICAgICBpbnQgY3kyID0gY2xpcFJlY3RzW2kgKyAzXTsgICAgIC8vIENsaXBwaW5nIHJpZ2h0IGJvdHRvbSBZCi0KLSAgICAgICAgICAgIGlmKF9keCA+IGN4MiB8fCBfZHkgPiBjeTIgfHwgZHN0WDIgPCBjeCB8fCBkc3RZMiA8IGN5KSB7Ci0gICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmKGN4ID4gX2R4KXsKLSAgICAgICAgICAgICAgICBpbnQgc2h4ID0gY3ggLSBfZHg7Ci0gICAgICAgICAgICAgICAgX3cgLT0gc2h4OwotICAgICAgICAgICAgICAgIF9keCA9IGN4OwotICAgICAgICAgICAgICAgIF9zeCArPSBzaHg7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmKGN5ID4gX2R5KXsKLSAgICAgICAgICAgICAgICBpbnQgc2h5ID0gY3kgLSBfZHk7Ci0gICAgICAgICAgICAgICAgX2ggLT0gc2h5OwotICAgICAgICAgICAgICAgIF9keSA9IGN5OwotICAgICAgICAgICAgICAgIF9zeSArPSBzaHk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmKF9keCArIF93ID4gY3gyICsgMSl7Ci0gICAgICAgICAgICAgICAgX3cgPSBjeDIgLSBfZHggKyAxOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZihfZHkgKyBfaCA+IGN5MiArIDEpewotICAgICAgICAgICAgICAgIF9oID0gY3kyIC0gX2R5ICsgMTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYoX3N4ID4gc3JjWDIgfHwgX3N5ID4gc3JjWTIpIHsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYoaXNBbHBoYUNvbXApewotICAgICAgICAgICAgICAgIGFscGhhQ29tcG9zZShfc3gsIF9zeSwgc3JjQ00sIHNyY1Jhc3QsIF9keCwgX2R5LAotICAgICAgICAgICAgICAgICAgICAgICAgZHN0Q00sIGRzdFJhc3QsIF93LCBfaCwgcnVsZSwgYWxwaGEsIGJnY29sb3IpOwotICAgICAgICAgICAgfWVsc2UgaWYoaXNYT1JDb21wKXsKLSAgICAgICAgICAgICAgICB4b3JDb21wb3NlKF9zeCwgX3N5LCBzcmNDTSwgc3JjUmFzdCwgX2R4LCBfZHksCi0gICAgICAgICAgICAgICAgICAgICAgICBkc3RDTSwgZHN0UmFzdCwgX3csIF9oLCB4b3Jjb2xvcik7Ci0gICAgICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgICAgICBSYXN0ZXIgc3IgPSBzcmNSYXN0LmNyZWF0ZUNoaWxkKF9zeCwgX3N5LCBfdywgX2gsIDAsIDAsIG51bGwpOwotICAgICAgICAgICAgICAgIFdyaXRhYmxlUmFzdGVyIGRyID0gZHN0UmFzdC5jcmVhdGVXcml0YWJsZUNoaWxkKF9keCwgX2R5LAotICAgICAgICAgICAgICAgICAgICAgICAgX3csIF9oLCAwLCAwLCBudWxsKTsKLSAgICAgICAgICAgICAgICBjb250LmNvbXBvc2Uoc3IsIGRyLCBkcik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICB2b2lkIGFscGhhQ29tcG9zZShpbnQgc3JjWCwgaW50IHNyY1ksIENvbG9yTW9kZWwgc3JjQ00sIFJhc3RlciBzcmNSYXN0LAotICAgICAgICAgICAgaW50IGRzdFgsIGludCBkc3RZLCBDb2xvck1vZGVsIGRzdENNLCBXcml0YWJsZVJhc3RlciBkc3RSYXN0LAotICAgICAgICAgICAgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgcnVsZSwgZmxvYXQgYWxwaGEsIENvbG9yIGJnY29sb3IpewotCi0gICAgICAgIE9iamVjdCBzcmNQaXhlbCwgZHN0UGl4ZWw7Ci0gICAgICAgIGludCBzcmNDb25zdEFsbHBoYSA9IChpbnQpKGFscGhhICogMjU1ICsgMC41Zik7Ci0gICAgICAgIGludCBzcmNSR0IsIGRzdFJHQiA9IDA7Ci0KLSAgICAgICAgaWYoYmdjb2xvciAhPSBudWxsKXsKLSAgICAgICAgICAgIGRzdFJHQiA9IGJnY29sb3IuZ2V0UkdCKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBmb3IoaW50IHN5ID0gc3JjWSwgZHkgPSBkc3RZLCBzcmNZTWF4ID0gc3JjWSArIGhlaWdodDsgc3kgPCBzcmNZTWF4OyBzeSsrLCBkeSsrKXsKLSAgICAgICAgICAgIGZvcihpbnQgc3ggPSBzcmNYLCBkeCA9IGRzdFgsIHNyY1hNYXggPSBzcmNYICsgd2lkdGg7IHN4IDwgc3JjWE1heDsgc3grKywgZHgrKyl7Ci0gICAgICAgICAgICAgICAgc3JjUGl4ZWwgPSBzcmNSYXN0LmdldERhdGFFbGVtZW50cyhzeCwgc3ksIG51bGwpOwotICAgICAgICAgICAgICAgIHNyY1JHQiA9IHNyY0NNLmdldFJHQihzcmNQaXhlbCk7Ci0gICAgICAgICAgICAgICAgaWYoYmdjb2xvciA9PSBudWxsKXsKLSAgICAgICAgICAgICAgICAgICAgZHN0UGl4ZWwgPSBkc3RSYXN0LmdldERhdGFFbGVtZW50cyhkeCwgZHksIG51bGwpOwotICAgICAgICAgICAgICAgICAgICBkc3RSR0IgPSBkc3RDTS5nZXRSR0IoZHN0UGl4ZWwpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGRzdFJHQiA9IGNvbXBvc2Uoc3JjUkdCLCBzcmNDTS5pc0FscGhhUHJlbXVsdGlwbGllZCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgZHN0UkdCLCBkc3RDTS5oYXNBbHBoYSgpLCBkc3RDTS5pc0FscGhhUHJlbXVsdGlwbGllZCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgcnVsZSwgc3JjQ29uc3RBbGxwaGEpOwotCi0gICAgICAgICAgICAgICAgZHN0UGl4ZWwgPSBkc3RDTS5nZXREYXRhRWxlbWVudHMoZHN0UkdCLCBudWxsKTsKLSAgICAgICAgICAgICAgICBkc3RSYXN0LnNldERhdGFFbGVtZW50cyhkeCxkeSxkc3RQaXhlbCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICB2b2lkIHhvckNvbXBvc2UoaW50IHNyY1gsIGludCBzcmNZLCBDb2xvck1vZGVsIHNyY0NNLCBSYXN0ZXIgc3JjUmFzdCwKLSAgICAgICAgICAgIGludCBkc3RYLCBpbnQgZHN0WSwgQ29sb3JNb2RlbCBkc3RDTSwgV3JpdGFibGVSYXN0ZXIgZHN0UmFzdCwKLSAgICAgICAgICAgIGludCB3aWR0aCwgaW50IGhlaWdodCwgQ29sb3IgeG9yY29sb3IpewotCi0gICAgICAgIE9iamVjdCBzcmNQaXhlbCwgZHN0UGl4ZWw7Ci0gICAgICAgIGludCB4b3JSR0IgPSB4b3Jjb2xvci5nZXRSR0IoKTsKLSAgICAgICAgaW50IHNyY1JHQiwgZHN0UkdCOwotCi0gICAgICAgIGZvcihpbnQgc3kgPSBzcmNZLCBkeSA9IGRzdFksIHNyY1lNYXggPSBzcmNZICsgaGVpZ2h0OyBzeSA8IHNyY1lNYXg7IHN5KyssIGR5KyspewotICAgICAgICAgICAgZm9yKGludCBzeCA9IHNyY1gsIGR4ID0gZHN0WCwgc3JjWE1heCA9IHNyY1ggKyB3aWR0aDsgc3ggPCBzcmNYTWF4OyBzeCsrLCBkeCsrKXsKLSAgICAgICAgICAgICAgICBzcmNQaXhlbCA9IHNyY1Jhc3QuZ2V0RGF0YUVsZW1lbnRzKHN4LCBzeSwgbnVsbCk7Ci0gICAgICAgICAgICAgICAgZHN0UGl4ZWwgPSBkc3RSYXN0LmdldERhdGFFbGVtZW50cyhkeCwgZHksIG51bGwpOwotCi0gICAgICAgICAgICAgICAgc3JjUkdCID0gc3JjQ00uZ2V0UkdCKHNyY1BpeGVsKTsKLSAgICAgICAgICAgICAgICBkc3RSR0IgPSBkc3RDTS5nZXRSR0IoZHN0UGl4ZWwpOwotICAgICAgICAgICAgICAgIGRzdFJHQiA9IHNyY1JHQiBeIHhvclJHQiBeIGRzdFJHQjsKLQotICAgICAgICAgICAgICAgIGRzdFJHQiA9IDB4ZmYwMDAwMDAgfCBkc3RSR0I7Ci0gICAgICAgICAgICAgICAgZHN0UGl4ZWwgPSBkc3RDTS5nZXREYXRhRWxlbWVudHMoZHN0UkdCLCBkc3RQaXhlbCk7Ci0gICAgICAgICAgICAgICAgZHN0UmFzdC5zZXREYXRhRWxlbWVudHMoZHgsZHksZHN0UGl4ZWwpOwotCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIHByaXZhdGUgdm9pZCB0cmFuc2Zvcm1lZEJsaXQoQ29sb3JNb2RlbCBzcmNDTSwgUmFzdGVyIHNyY1IsIGludCBzcmNYLCBpbnQgc3JjWSwKLSAgICAgICAgICAgIENvbG9yTW9kZWwgZHN0Q00sIFdyaXRhYmxlUmFzdGVyIGRzdFIsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIGludCB3aWR0aCwgaW50IGhlaWdodCwgQWZmaW5lVHJhbnNmb3JtIGF0LCBDb21wb3NpdGUgY29tcCwKLSAgICAgICAgICAgIENvbG9yIGJnY29sb3IsTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0KLSAgICAgICAgUmVjdGFuZ2xlIHNyY0JvdW5kcyA9IG5ldyBSZWN0YW5nbGUod2lkdGgsIGhlaWdodCk7Ci0gICAgICAgIFJlY3RhbmdsZSBkc3RCbGl0Qm91bmRzID0gbmV3IFJlY3RhbmdsZShkc3RYLCBkc3RZLCBzcmNSLmdldFdpZHRoKCksIHNyY1IuZ2V0SGVpZ2h0KCkpOwotCi0gICAgICAgIFJlY3RhbmdsZSB0cmFuc1NyY0JvdW5kcyA9IGdldEJvdW5kczJEKGF0LCBzcmNCb3VuZHMpLmdldEJvdW5kcygpOwotICAgICAgICBSZWN0YW5nbGUgdHJhbnNEc3RCbGl0Qm91bmRzID0gZ2V0Qm91bmRzMkQoYXQsIGRzdEJsaXRCb3VuZHMpLmdldEJvdW5kcygpOwotCi0gICAgICAgIGludCB0cmFuc2xhdGVYID0gdHJhbnNEc3RCbGl0Qm91bmRzLnggLSB0cmFuc1NyY0JvdW5kcy54OwotICAgICAgICBpbnQgdHJhbnNsYXRlWSA9IHRyYW5zRHN0QmxpdEJvdW5kcy55IC0gdHJhbnNTcmNCb3VuZHMueTsKLQotICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gaW52ID0gbnVsbDsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICBpbnYgPSBhdC5jcmVhdGVJbnZlcnNlKCk7Ci0gICAgICAgIH0gY2F0Y2ggKE5vbmludmVydGlibGVUcmFuc2Zvcm1FeGNlcHRpb24gZSkgewotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgZG91YmxlW10gbSA9IG5ldyBkb3VibGVbNl07Ci0gICAgICAgIGludi5nZXRNYXRyaXgobSk7Ci0KLSAgICAgICAgaW50IGNsaXBSZWN0c1tdOwotICAgICAgICBpZihjbGlwICE9IG51bGwpIHsKLSAgICAgICAgICAgIGNsaXBSZWN0cyA9IGNsaXAucmVjdDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNsaXBSZWN0cyA9IG5ldyBpbnRbXXs1LCAwLCAwLCBkc3RSLmdldFdpZHRoKCksIGRzdFIuZ2V0SGVpZ2h0KCl9OwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGNvbXBUeXBlID0gMDsKLSAgICAgICAgaW50IHNyY0NvbnN0QWxwaGEgPSAwOwotICAgICAgICBpbnQgcnVsZSA9IDA7Ci0gICAgICAgIGludCBiZ1JHQiA9IGJnY29sb3IgPT0gbnVsbCA/IDAgOiBiZ2NvbG9yLmdldFJHQigpOwotICAgICAgICBpbnQgc3JjUkdCID0gMCwgZHN0UkdCID0gMDsKLSAgICAgICAgT2JqZWN0IHNyY1ZhbCA9IG51bGwsIGRzdFZhbCA9IG51bGw7Ci0gICAgICAgIGlmKGNvbXAgaW5zdGFuY2VvZiBBbHBoYUNvbXBvc2l0ZSl7Ci0gICAgICAgICAgICBjb21wVHlwZSA9IEFscGhhQ29tcG9zaXRlTW9kZTsKLSAgICAgICAgICAgIEFscGhhQ29tcG9zaXRlIGFjID0gKEFscGhhQ29tcG9zaXRlKSBjb21wOwotICAgICAgICAgICAgcnVsZSA9IGFjLmdldFJ1bGUoKTsKLSAgICAgICAgICAgIHNyY0NvbnN0QWxwaGEgPSAoaW50KShhYy5nZXRBbHBoYSgpICogMjU1ICsgMC41Zik7Ci0gICAgICAgIH1lbHNlIGlmKGNvbXAgaW5zdGFuY2VvZiBYT1JDb21wb3NpdGUpewotICAgICAgICAgICAgY29tcFR5cGUgPSBYT1JNb2RlOwotICAgICAgICAgICAgWE9SQ29tcG9zaXRlIHhvciA9IChYT1JDb21wb3NpdGUpIGNvbXA7Ci0gICAgICAgICAgICBiZ1JHQiA9IHhvci5nZXRYT1JDb2xvcigpLmdldFJHQigpOwotICAgICAgICB9Ci0KLSAgICAgICAgZm9yKGludCBpID0gMTsgaSA8IGNsaXBSZWN0c1swXTsgaSArPSA0KXsKLSAgICAgICAgICAgIFJlY3RhbmdsZSBkc3RCb3VuZHMgPSBuZXcgUmVjdGFuZ2xlKGNsaXBSZWN0c1tpXSwgY2xpcFJlY3RzW2kgKyAxXSwgMCwgMCk7Ci0gICAgICAgICAgICBkc3RCb3VuZHMuYWRkKGNsaXBSZWN0c1tpICsgMl0gKyAxLCBjbGlwUmVjdHNbaSArIDFdKTsKLSAgICAgICAgICAgIGRzdEJvdW5kcy5hZGQoY2xpcFJlY3RzW2kgKyAyXSArIDEsIGNsaXBSZWN0c1tpICsgM10gKyAxKTsKLSAgICAgICAgICAgIGRzdEJvdW5kcy5hZGQoY2xpcFJlY3RzW2ldLCBjbGlwUmVjdHNbaSArIDNdICsgMSk7Ci0KLSAgICAgICAgICAgIFJlY3RhbmdsZSBib3VuZHMgPSBkc3RCb3VuZHMuaW50ZXJzZWN0aW9uKHRyYW5zRHN0QmxpdEJvdW5kcyk7Ci0KLSAgICAgICAgICAgIGludCBtaW5TcmNYID0gc3JjQm91bmRzLng7Ci0gICAgICAgICAgICBpbnQgbWluU3JjWSA9IHNyY0JvdW5kcy55OwotICAgICAgICAgICAgaW50IG1heFNyY1ggPSBtaW5TcmNYICsgc3JjQm91bmRzLndpZHRoOwotICAgICAgICAgICAgaW50IG1heFNyY1kgPSBtaW5TcmNZICsgc3JjQm91bmRzLmhlaWdodDsKLQotICAgICAgICAgICAgaW50IG1pblggPSBib3VuZHMueDsKLSAgICAgICAgICAgIGludCBtaW5ZID0gYm91bmRzLnk7Ci0gICAgICAgICAgICBpbnQgbWF4WCA9IG1pblggKyBib3VuZHMud2lkdGg7Ci0gICAgICAgICAgICBpbnQgbWF4WSA9IG1pblkgKyBib3VuZHMuaGVpZ2h0OwotCi0gICAgICAgICAgICBpbnQgaHggPSAoaW50KSgobVswXSAqIDI1NikgKyAwLjUpOwotICAgICAgICAgICAgaW50IGh5ID0gKGludCkoKG1bMV0gKiAyNTYpICsgMC41KTsKLSAgICAgICAgICAgIGludCB2eCA9IChpbnQpKChtWzJdICogMjU2KSArIDAuNSk7Ci0gICAgICAgICAgICBpbnQgdnkgPSAoaW50KSgobVszXSAqIDI1NikgKyAwLjUpOwotICAgICAgICAgICAgaW50IHN4ID0gKGludCkoKG1bNF0gKyBtWzBdICogKGJvdW5kcy54IC0gdHJhbnNsYXRlWCkgKyBtWzJdICogKGJvdW5kcy55IC0gdHJhbnNsYXRlWSkpICogMjU2ICsgMC41KTsKLSAgICAgICAgICAgIGludCBzeSA9IChpbnQpKChtWzVdICsgbVsxXSAqIChib3VuZHMueCAtIHRyYW5zbGF0ZVgpICsgbVszXSAqIChib3VuZHMueSAtIHRyYW5zbGF0ZVkpKSAqIDI1NiArIDAuNSk7Ci0KLSAgICAgICAgICAgIHZ4IC09IGh4ICogYm91bmRzLndpZHRoOwotICAgICAgICAgICAgdnkgLT0gaHkgKiBib3VuZHMud2lkdGg7Ci0KLSAgICAgICAgICAgIGZvcihpbnQgeSA9IG1pblk7IHkgPCBtYXhZOyB5KyspIHsKLSAgICAgICAgICAgICAgICBmb3IoaW50IHggPSBtaW5YOyB4IDwgbWF4WDsgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCBweCA9IHN4ID4+IDg7Ci0gICAgICAgICAgICAgICAgICAgIGludCBweSA9IHN5ID4+IDg7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChweCA+PSBtaW5TcmNYICYmIHB5ID49IG1pblNyY1kgJiYgcHggPCBtYXhTcmNYICYmIHB5IDwgbWF4U3JjWSkgewotICAgICAgICAgICAgICAgICAgICAgICAgc3dpdGNoKGNvbXBUeXBlKXsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjYXNlIEFscGhhQ29tcG9zaXRlTW9kZToKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3JjVmFsID0gc3JjUi5nZXREYXRhRWxlbWVudHMocHggLCBweSAsIG51bGwpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzcmNSR0IgPSBzcmNDTS5nZXRSR0Ioc3JjVmFsKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYoYmdjb2xvciAhPSBudWxsKXsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdFJHQiA9IGJnUkdCOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdFZhbCA9IGRzdFIuZ2V0RGF0YUVsZW1lbnRzKHgsIHksIG51bGwpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0UkdCID0gZHN0Q00uZ2V0UkdCKGRzdFZhbCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0UkdCID0gY29tcG9zZShzcmNSR0IsIHNyY0NNLmlzQWxwaGFQcmVtdWx0aXBsaWVkKCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0UkdCLCBkc3RDTS5oYXNBbHBoYSgpLCBkc3RDTS5pc0FscGhhUHJlbXVsdGlwbGllZCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJ1bGUsIHNyY0NvbnN0QWxwaGEpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3RWYWwgPSBkc3RDTS5nZXREYXRhRWxlbWVudHMoZHN0UkdCLCBudWxsKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0Ui5zZXREYXRhRWxlbWVudHMoeCwgeSwgZHN0VmFsKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjYXNlIFhPUk1vZGU6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNyY1ZhbCA9IHNyY1IuZ2V0RGF0YUVsZW1lbnRzKHB4ICwgcHkgLCBudWxsKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3JjUkdCID0gc3JjQ00uZ2V0UkdCKHNyY1ZhbCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdFZhbCA9IGRzdFIuZ2V0RGF0YUVsZW1lbnRzKHgsIHksIG51bGwpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3RSR0IgPSBkc3RDTS5nZXRSR0IoZHN0VmFsKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0UkdCID0gc3JjUkdCIF4gYmdSR0I7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0UkdCID0gMHhmZjAwMDAwMCB8IGRzdFJHQjsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0VmFsID0gZHN0Q00uZ2V0RGF0YUVsZW1lbnRzKGRzdFJHQiwgbnVsbCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdFIuc2V0RGF0YUVsZW1lbnRzKHgsIHksIGRzdFZhbCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gYXd0LjM3PVVua25vd24gIGNvbXBvc2l0ZSB0eXBlIHswfQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjM3IiwgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbXAuZ2V0Q2xhc3MoKSkpOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHN4ICs9IGh4OwotICAgICAgICAgICAgICAgICAgICBzeSArPSBoeTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgc3ggKz0gdng7Ci0gICAgICAgICAgICAgICAgc3kgKz0gdnk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIHByaXZhdGUgUmVjdGFuZ2xlMkQgZ2V0Qm91bmRzMkQoQWZmaW5lVHJhbnNmb3JtIGF0LCBSZWN0YW5nbGUgcikgewotICAgICAgICBpbnQgeCA9IHIueDsKLSAgICAgICAgaW50IHkgPSByLnk7Ci0gICAgICAgIGludCB3aWR0aCA9IHIud2lkdGg7Ci0gICAgICAgIGludCBoZWlnaHQgPSByLmhlaWdodDsKLQotICAgICAgICBmbG9hdFtdIGNvcm5lcnMgPSB7Ci0gICAgICAgICAgICB4LCB5LAotICAgICAgICAgICAgeCArIHdpZHRoLCB5LAotICAgICAgICAgICAgeCArIHdpZHRoLCB5ICsgaGVpZ2h0LAotICAgICAgICAgICAgeCwgeSArIGhlaWdodAotICAgICAgICB9OwotCi0gICAgICAgIGF0LnRyYW5zZm9ybShjb3JuZXJzLCAwLCBjb3JuZXJzLCAwLCA0KTsKLQotICAgICAgICBSZWN0YW5nbGUyRC5GbG9hdCBib3VuZHMgPSBuZXcgUmVjdGFuZ2xlMkQuRmxvYXQoY29ybmVyc1swXSwgY29ybmVyc1sxXSwgMCAsIDApOwotICAgICAgICBib3VuZHMuYWRkKGNvcm5lcnNbMl0sIGNvcm5lcnNbM10pOwotICAgICAgICBib3VuZHMuYWRkKGNvcm5lcnNbNF0sIGNvcm5lcnNbNV0pOwotICAgICAgICBib3VuZHMuYWRkKGNvcm5lcnNbNl0sIGNvcm5lcnNbN10pOwotCi0gICAgICAgIHJldHVybiBib3VuZHM7Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBpbnQgY29tcG9zZShpbnQgc3JjUkdCLCBib29sZWFuIGlzU3JjQWxwaGFQcmUsCi0gICAgICAgICAgICBpbnQgZHN0UkdCLCBib29sZWFuIGRzdEhhc0FscGhhLCBib29sZWFuIGlzRHN0QWxwaGFQcmUsCi0gICAgICAgICAgICBpbnQgcnVsZSwgaW50IHNyY0NvbnN0QWxwaGEpewotCi0gICAgICAgIGludCBzYSwgc3IsIHNnLCBzYiwgZGEsIGRyLCBkZywgZGI7Ci0KLSAgICAgICAgc2EgPSAoc3JjUkdCID4+IDI0KSAmIDB4ZmY7Ci0gICAgICAgIHNyID0gKHNyY1JHQiA+PiAxNikgJiAweGZmOwotICAgICAgICBzZyA9IChzcmNSR0IgPj4gOCkgJiAweGZmOwotICAgICAgICBzYiA9IHNyY1JHQiAmIDB4ZmY7Ci0KLSAgICAgICAgaWYoaXNTcmNBbHBoYVByZSl7Ci0gICAgICAgICAgICBzYSA9IG11bExVVFtzcmNDb25zdEFscGhhXVtzYV0gJiAweGZmOwotICAgICAgICAgICAgc3IgPSBtdWxMVVRbc3JjQ29uc3RBbHBoYV1bc3JdICYgMHhmZjsKLSAgICAgICAgICAgIHNnID0gbXVsTFVUW3NyY0NvbnN0QWxwaGFdW3NnXSAmIDB4ZmY7Ci0gICAgICAgICAgICBzYiA9IG11bExVVFtzcmNDb25zdEFscGhhXVtzYl0gJiAweGZmOwotICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgIHNhID0gbXVsTFVUW3NyY0NvbnN0QWxwaGFdW3NhXSAmIDB4ZmY7Ci0gICAgICAgICAgICBzciA9IG11bExVVFtzYV1bc3JdICYgMHhmZjsKLSAgICAgICAgICAgIHNnID0gbXVsTFVUW3NhXVtzZ10gJiAweGZmOwotICAgICAgICAgICAgc2IgPSBtdWxMVVRbc2FdW3NiXSAmIDB4ZmY7Ci0gICAgICAgIH0KLQotICAgICAgICBkYSA9IChkc3RSR0IgPj4gMjQpICYgMHhmZjsKLSAgICAgICAgZHIgPSAoZHN0UkdCID4+IDE2KSAmIDB4ZmY7Ci0gICAgICAgIGRnID0gKGRzdFJHQiA+PiA4KSAmIDB4ZmY7Ci0gICAgICAgIGRiID0gZHN0UkdCICYgMHhmZjsKLQotICAgICAgICBpZighaXNEc3RBbHBoYVByZSl7Ci0gICAgICAgICAgICBkciA9IG11bExVVFtkYV1bZHJdICYgMHhmZjsKLSAgICAgICAgICAgIGRnID0gbXVsTFVUW2RhXVtkZ10gJiAweGZmOwotICAgICAgICAgICAgZGIgPSBtdWxMVVRbZGFdW2RiXSAmIDB4ZmY7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgRnMgPSAwOwotICAgICAgICBpbnQgRmQgPSAwOwotICAgICAgICBzd2l0Y2gocnVsZSl7Ci0gICAgICAgIGNhc2UgQWxwaGFDb21wb3NpdGUuQ0xFQVI6Ci0gICAgICAgICAgICBicmVhazsKLQotICAgICAgICBjYXNlIEFscGhhQ29tcG9zaXRlLkRTVDoKLSAgICAgICAgICAgIEZkID0gMjU1OwotICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgY2FzZSBBbHBoYUNvbXBvc2l0ZS5EU1RfQVRPUDoKLSAgICAgICAgICAgIEZzID0gMjU1IC0gZGE7Ci0gICAgICAgICAgICBGZCA9IHNhOwotICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgY2FzZSBBbHBoYUNvbXBvc2l0ZS5EU1RfSU46Ci0gICAgICAgICAgICBGZCA9IHNhOwotICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgY2FzZSBBbHBoYUNvbXBvc2l0ZS5EU1RfT1VUOgotICAgICAgICAgICAgRmQgPSAyNTUgLSBzYTsKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGNhc2UgQWxwaGFDb21wb3NpdGUuRFNUX09WRVI6Ci0gICAgICAgICAgICBGcyA9IDI1NSAtIGRhOwotICAgICAgICAgICAgRmQgPSAyNTU7Ci0gICAgICAgICAgICBicmVhazsKLQotICAgICAgICBjYXNlIEFscGhhQ29tcG9zaXRlLlNSQzoKLSAgICAgICAgICAgIEZzID0gMjU1OwotICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgY2FzZSBBbHBoYUNvbXBvc2l0ZS5TUkNfQVRPUDoKLSAgICAgICAgICAgIEZzID0gZGE7Ci0gICAgICAgICAgICBGZCA9IDI1NSAtIHNhOwotICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgY2FzZSBBbHBoYUNvbXBvc2l0ZS5TUkNfSU46Ci0gICAgICAgICAgICBGcyA9IGRhOwotICAgICAgICAgICAgYnJlYWs7Ci0KLSAgICAgICAgY2FzZSBBbHBoYUNvbXBvc2l0ZS5TUkNfT1VUOgotICAgICAgICAgICAgRnMgPSAyNTUgLSBkYTsKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGNhc2UgQWxwaGFDb21wb3NpdGUuU1JDX09WRVI6Ci0gICAgICAgICAgICBGcyA9IDI1NTsKLSAgICAgICAgICAgIEZkID0gMjU1IC0gc2E7Ci0gICAgICAgICAgICBicmVhazsKLQotICAgICAgICBjYXNlIEFscGhhQ29tcG9zaXRlLlhPUjoKLSAgICAgICAgICAgIEZzID0gMjU1IC0gZGE7Ci0gICAgICAgICAgICBGZCA9IDI1NSAtIHNhOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgZHIgPSAobXVsTFVUW3NyXVtGc10gJiAweGZmKSArIChtdWxMVVRbZHJdW0ZkXSAmIDB4ZmYpOwotICAgICAgICBkZyA9IChtdWxMVVRbc2ddW0ZzXSAmIDB4ZmYpICsgKG11bExVVFtkZ11bRmRdICYgMHhmZik7Ci0gICAgICAgIGRiID0gKG11bExVVFtzYl1bRnNdICYgMHhmZikgKyAobXVsTFVUW2RiXVtGZF0gJiAweGZmKTsKLQotICAgICAgICBkYSA9IChtdWxMVVRbc2FdW0ZzXSAmIDB4ZmYpICsgKG11bExVVFtkYV1bRmRdICYgMHhmZik7Ci0KLSAgICAgICAgaWYoIWlzRHN0QWxwaGFQcmUpewotICAgICAgICAgICAgaWYoZGEgIT0gMjU1KXsKLSAgICAgICAgICAgICAgICBkciA9IGRpdkxVVFtkYV1bZHJdICYgMHhmZjsKLSAgICAgICAgICAgICAgICBkZyA9IGRpdkxVVFtkYV1bZGddICYgMHhmZjsKLSAgICAgICAgICAgICAgICBkYiA9IGRpdkxVVFtkYV1bZGJdICYgMHhmZjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBpZighZHN0SGFzQWxwaGEpIHsKLSAgICAgICAgICAgIGRhID0gMHhmZjsKLSAgICAgICAgfQotICAgICAgICBkc3RSR0IgPSAoZGEgPDwgMjQpIHwgKGRyIDw8IDE2KSB8IChkZyA8PCA4KSB8IGRiOwotCi0gICAgICAgIHJldHVybiBkc3RSR0I7Ci0KLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL3JlbmRlci9KYXZhTGluZVJhc3Rlcml6ZXIuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL3JlbmRlci9KYXZhTGluZVJhc3Rlcml6ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZWI2ZjdiNS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvSmF2YUxpbmVSYXN0ZXJpemVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw3NjAgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wucmVuZGVyOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5NdWx0aVJlY3RBcmVhOwotCi0KLXB1YmxpYyBjbGFzcyBKYXZhTGluZVJhc3Rlcml6ZXIgewotCi0gICAgLyoqCi0gICAgICogIExpbmVEYXNoZXIgY2xhc3MgcHJvdmlkZXMgZGFzaGluZyBmb3IgcGFydGljdWxhciBkYXNoIHN0eWxlCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBjbGFzcyBMaW5lRGFzaGVyIHsKLQotICAgICAgICBpbnQgaW5kZXg7Ci0gICAgICAgIGZsb2F0IHBvczsKLSAgICAgICAgZmxvYXQgcGhhc2U7Ci0gICAgICAgIGZsb2F0IGRhc2hbXTsKLSAgICAgICAgZmxvYXQgaW52W107Ci0gICAgICAgIGJvb2xlYW4gdmlzaWJsZTsKLQotICAgICAgICBwdWJsaWMgTGluZURhc2hlcigpIHsKLSAgICAgICAgfQotCi0gICAgICAgIHB1YmxpYyBMaW5lRGFzaGVyKGZsb2F0IGRhc2hbXSwgZmxvYXQgcGhhc2UpIHsKLSAgICAgICAgICAgIHRoaXMuZGFzaCA9IGRhc2g7Ci0gICAgICAgICAgICB0aGlzLnBoYXNlID0gcGhhc2U7Ci0KLSAgICAgICAgICAgIGludiA9IG5ldyBmbG9hdFtkYXNoLmxlbmd0aF07Ci0gICAgICAgICAgICBpbnQgaiA9IGRhc2gubGVuZ3RoOwotICAgICAgICAgICAgZm9yIChmbG9hdCBlbGVtZW50IDogZGFzaCkgewotICAgICAgICAgICAgICAgIGludlstLWpdID0gZWxlbWVudDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGluZGV4ID0gMDsKLSAgICAgICAgICAgIHdoaWxlIChwaGFzZSA+IGRhc2hbaW5kZXhdKSB7Ci0gICAgICAgICAgICAgICAgcGhhc2UgLT0gZGFzaFtpbmRleF07Ci0gICAgICAgICAgICAgICAgaW5kZXggPSAoaW5kZXggKyAxKSAlIGRhc2gubGVuZ3RoOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgdmlzaWJsZSA9IGluZGV4ICUgMiA9PSAwOwotICAgICAgICB9Ci0KLSAgICAgICAgdm9pZCBtb3ZlKGZsb2F0IHN0ZXApIHsgLy8gbWFpbiBkYXNoZXIKLSAgICAgICAgICAgIHBvcyArPSBzdGVwOwotICAgICAgICAgICAgc3RlcCArPSBwaGFzZTsKLSAgICAgICAgICAgIHdoaWxlKHN0ZXAgPj0gZGFzaFtpbmRleF0pIHsKLSAgICAgICAgICAgICAgICBzdGVwIC09IGRhc2hbaW5kZXhdOwotICAgICAgICAgICAgICAgIGluZGV4ID0gKGluZGV4ICsgMSkgJSBkYXNoLmxlbmd0aDsKLSAgICAgICAgICAgICAgICB2aXNpYmxlID0gIXZpc2libGU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwaGFzZSA9IHN0ZXA7Ci0gICAgICAgIH0KLQotICAgICAgICBmbG9hdCBuZXh0RGFzaCgpIHsKLSAgICAgICAgICAgIHBoYXNlID0gMC4wZjsKLSAgICAgICAgICAgIGluZGV4ID0gKGluZGV4ICsgMSkgJSBkYXNoLmxlbmd0aDsKLSAgICAgICAgICAgIHZpc2libGUgPSAhdmlzaWJsZTsKLSAgICAgICAgICAgIHJldHVybiBkYXNoW2luZGV4XTsKLSAgICAgICAgfQotCi0gICAgICAgIExpbmVEYXNoZXIgY3JlYXRlRGlhZ29uYWwoZG91YmxlIGssIGZsb2F0IGxlbmd0aCwgYm9vbGVhbiBpbnZlcnQpIHsKLSAgICAgICAgICAgIExpbmVEYXNoZXIgbG9jYWwgPSBuZXcgTGluZURhc2hlcigpOwotICAgICAgICAgICAgbG9jYWwuZGFzaCA9IG5ldyBmbG9hdFtkYXNoLmxlbmd0aF07Ci0gICAgICAgICAgICBpZiAoaW52ZXJ0KSB7IC8vIGludmVydGVkIGRhc2hlcgotICAgICAgICAgICAgICAgIG1vdmUobGVuZ3RoKTsKLSAgICAgICAgICAgICAgICBsb2NhbC5waGFzZSA9IChmbG9hdCkoKGRhc2hbaW5kZXhdIC0gcGhhc2UpICogayk7Ci0gICAgICAgICAgICAgICAgbG9jYWwudmlzaWJsZSA9IHZpc2libGU7Ci0gICAgICAgICAgICAgICAgbG9jYWwuaW5kZXggPSBpbnYubGVuZ3RoIC0gaW5kZXggLSAxOwotICAgICAgICAgICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBpbnYubGVuZ3RoOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgbG9jYWwuZGFzaFtpXSA9IChmbG9hdCkoaW52W2ldICogayk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBsb2NhbC5waGFzZSA9IChmbG9hdCkocGhhc2UgKiBrKTsKLSAgICAgICAgICAgICAgICBsb2NhbC52aXNpYmxlID0gdmlzaWJsZTsKLSAgICAgICAgICAgICAgICBsb2NhbC5pbmRleCA9IGluZGV4OwotICAgICAgICAgICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBkYXNoLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGxvY2FsLmRhc2hbaV0gPSAoZmxvYXQpKGRhc2hbaV0gKiBrKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbW92ZShsZW5ndGgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGxvY2FsOwotICAgICAgICB9Ci0KLSAgICAgICAgTGluZURhc2hlciBjcmVhdGVPcnRvZ29uYWwoZmxvYXQgbGVuZ3RoLCBib29sZWFuIGludmVydCkgewotICAgICAgICAgICAgTGluZURhc2hlciBsb2NhbCA9IG5ldyBMaW5lRGFzaGVyKCk7Ci0gICAgICAgICAgICBsb2NhbC5kYXNoID0gbmV3IGZsb2F0W2Rhc2gubGVuZ3RoXTsKLSAgICAgICAgICAgIGlmIChpbnZlcnQpIHsgLy8gaW52ZXJ0ZWQgZGFzaGVyCi0gICAgICAgICAgICAgICAgbW92ZShsZW5ndGgpOwotICAgICAgICAgICAgICAgIGxvY2FsLnBoYXNlID0gZGFzaFtpbmRleF0gLSBwaGFzZTsKLSAgICAgICAgICAgICAgICBsb2NhbC52aXNpYmxlID0gdmlzaWJsZTsKLSAgICAgICAgICAgICAgICBsb2NhbC5pbmRleCA9IGludi5sZW5ndGggLSBpbmRleCAtIDE7Ci0gICAgICAgICAgICAgICAgbG9jYWwuZGFzaCA9IGludjsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgbG9jYWwucGhhc2UgPSBwaGFzZTsKLSAgICAgICAgICAgICAgICBsb2NhbC52aXNpYmxlID0gdmlzaWJsZTsKLSAgICAgICAgICAgICAgICBsb2NhbC5pbmRleCA9IGluZGV4OwotICAgICAgICAgICAgICAgIGxvY2FsLmRhc2ggPSBkYXNoOwotICAgICAgICAgICAgICAgIG1vdmUobGVuZ3RoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBsb2NhbDsKLSAgICAgICAgfQotCi0gICAgICAgIExpbmVEYXNoZXIgY3JlYXRlQ2hpbGQoZmxvYXQgc3RhcnQpIHsKLSAgICAgICAgICAgIExpbmVEYXNoZXIgY2hpbGQgPSBuZXcgTGluZURhc2hlcigpOwotICAgICAgICAgICAgY2hpbGQucGhhc2UgPSBwaGFzZTsKLSAgICAgICAgICAgIGNoaWxkLnZpc2libGUgPSB2aXNpYmxlOwotICAgICAgICAgICAgY2hpbGQuaW5kZXggPSBpbmRleDsKLSAgICAgICAgICAgIGNoaWxkLmRhc2ggPSBkYXNoOwotICAgICAgICAgICAgY2hpbGQubW92ZShzdGFydCk7Ci0gICAgICAgICAgICByZXR1cm4gY2hpbGQ7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIExpbmUgY2xhc3MgcHJvdmlkZXMgcmFzdGVyaXphdGlvbiBmb3IgZGlmZmVyZW50IGxpbmUgdHlwZXMKLSAgICAgKi8KLSAgICBhYnN0cmFjdCBzdGF0aWMgY2xhc3MgTGluZSB7Ci0KLSAgICAgICAgaW50IHgxLCB5MSwgeDIsIHkyOwotICAgICAgICBpbnQgeCwgeTsKLSAgICAgICAgTXVsdGlSZWN0QXJlYSBkc3Q7Ci0KLSAgICAgICAgTGluZShpbnQgeDEsIGludCB5MSwgaW50IHgyLCBpbnQgeTIsIE11bHRpUmVjdEFyZWEgZHN0KSB7Ci0gICAgICAgICAgICB0aGlzLngxID0geDE7Ci0gICAgICAgICAgICB0aGlzLnkxID0geTE7Ci0gICAgICAgICAgICB0aGlzLngyID0geDI7Ci0gICAgICAgICAgICB0aGlzLnkyID0geTI7Ci0gICAgICAgICAgICB0aGlzLmRzdCA9IGRzdDsKLSAgICAgICAgfQotCi0gICAgICAgIHN0YXRpYyBhYnN0cmFjdCBjbGFzcyBEaWFnIGV4dGVuZHMgTGluZSB7Ci0gICAgICAgICAgICBpbnQgZHgsIGR5LCBhZHgsIGFkeSwgc3gsIHN5OwotICAgICAgICAgICAgaW50IGVCYXNlLCBlUG9zLCBlTmVnOwotICAgICAgICAgICAgaW50IHhjb3VudDsKLSAgICAgICAgICAgIGludCBlOwotCi0gICAgICAgICAgICBEaWFnKGludCB4MSwgaW50IHkxLCBpbnQgeDIsIGludCB5MiwgTXVsdGlSZWN0QXJlYSBkc3QpIHsKLSAgICAgICAgICAgICAgICBzdXBlcih4MSwgeTEsIHgyLCB5MiwgZHN0KTsKLSAgICAgICAgICAgICAgICBkeCA9IHgyIC0geDE7Ci0gICAgICAgICAgICAgICAgZHkgPSB5MiAtIHkxOwotICAgICAgICAgICAgICAgIHN5ID0gMTsKLSAgICAgICAgICAgICAgICBpZiAoZHggPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGFkeCA9IGR4OwotICAgICAgICAgICAgICAgICAgICBzeCA9IDE7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgYWR4ID0gLWR4OwotICAgICAgICAgICAgICAgICAgICBzeCA9IC0xOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBhZHkgPSBkeTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgZmxvYXQgZ2V0TGVuZ3RoKCkgewotICAgICAgICAgICAgICAgIHJldHVybiAoZmxvYXQpTWF0aC5zcXJ0KGR4ICogZHggKyBkeSAqIGR5KTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgc3RhdGljIGNsYXNzIEhvciBleHRlbmRzIERpYWcgewotCi0gICAgICAgICAgICAgICAgSG9yKGludCB4MSwgaW50IHkxLCBpbnQgeDIsIGludCB5MiwgTXVsdGlSZWN0QXJlYSBkc3QpIHsKLSAgICAgICAgICAgICAgICAgICAgc3VwZXIoeDEsIHkxLCB4MiwgeTIsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIGVCYXNlID0gYWR5ICsgYWR5IC0gYWR4OwotICAgICAgICAgICAgICAgICAgICBlUG9zID0gMiAqIChhZHkgLSBhZHgpOwotICAgICAgICAgICAgICAgICAgICBlTmVnID0gYWR5ICsgYWR5OwotICAgICAgICAgICAgICAgICAgICB4Y291bnQgPSBhZHg7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgICAgICAgICAgdm9pZCByYXN0ZXJpemUoKSB7Ci0gICAgICAgICAgICAgICAgICAgIGUgPSBlQmFzZTsKLSAgICAgICAgICAgICAgICAgICAgeCA9IHgxOwotICAgICAgICAgICAgICAgICAgICB5ID0geTE7Ci0gICAgICAgICAgICAgICAgICAgIHJhc3Rlcml6ZSh4Y291bnQpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplQ2xpcHBlZChpbnQgbngxLCBpbnQgbnkxLCBpbnQgbngyLCBpbnQgbnkyKSB7Ci0gICAgICAgICAgICAgICAgICAgIGUgPSBlQmFzZSArIDIgKiAoYWR5ICogTWF0aC5hYnMobngxIC0geDEpIC0gYWR4ICogTWF0aC5hYnMobnkxIC0geTEpKTsKLSAgICAgICAgICAgICAgICAgICAgeCA9IG54MTsKLSAgICAgICAgICAgICAgICAgICAgeSA9IG55MTsKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyaXplKGR4ID4gMCA/IG54MiAtIG54MSA6IG54MSAtIG54Mik7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgICAgICAgICAgdm9pZCByYXN0ZXJpemUoaW50IGNvdW50KSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCBweCA9IHg7Ci0gICAgICAgICAgICAgICAgICAgIHdoaWxlIChjb3VudC0tID4gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGUgPj0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChzeCA+IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0LmFkZFJlY3QocHgsIHksIHgsIHkpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5hZGRSZWN0KHgsIHksIHB4LCB5KTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgeCArPSBzeDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB5ICs9IHN5OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGUgKz0gZVBvczsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBweCA9IHg7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGUgKz0gZU5lZzsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB4ICs9IHN4OwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGlmIChzeCA+IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5hZGRSZWN0KHB4LCB5LCB4LCB5KTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5hZGRSZWN0KHgsIHksIHB4LCB5KTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgc2tpcChpbnQgY291bnQpIHsKLSAgICAgICAgICAgICAgICAgICAgd2hpbGUgKGNvdW50LS0gPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB4ICs9IHN4OwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGUgPj0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHkgKz0gc3k7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZSArPSBlUG9zOwotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBlICs9IGVOZWc7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgc3RhdGljIGNsYXNzIFZlciBleHRlbmRzIERpYWcgewotCi0gICAgICAgICAgICAgICAgVmVyKGludCB4MSwgaW50IHkxLCBpbnQgeDIsIGludCB5MiwgTXVsdGlSZWN0QXJlYSBkc3QpIHsKLSAgICAgICAgICAgICAgICAgICAgc3VwZXIoeDEsIHkxLCB4MiwgeTIsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIGVCYXNlID0gYWR4ICsgYWR4IC0gYWR5OwotICAgICAgICAgICAgICAgICAgICBlUG9zID0gMiAqIChhZHggLSBhZHkpOwotICAgICAgICAgICAgICAgICAgICBlTmVnID0gYWR4ICsgYWR4OwotICAgICAgICAgICAgICAgICAgICB4Y291bnQgPSBhZHk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgICAgICAgICAgdm9pZCByYXN0ZXJpemUoKSB7Ci0gICAgICAgICAgICAgICAgICAgIGUgPSBlQmFzZTsKLSAgICAgICAgICAgICAgICAgICAgeCA9IHgxOwotICAgICAgICAgICAgICAgICAgICB5ID0geTE7Ci0gICAgICAgICAgICAgICAgICAgIHJhc3Rlcml6ZSh4Y291bnQpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplQ2xpcHBlZChpbnQgbngxLCBpbnQgbnkxLCBpbnQgbngyLCBpbnQgbnkyKSB7Ci0gICAgICAgICAgICAgICAgICAgIGUgPSBlQmFzZSArIDIgKiAoYWR4ICogTWF0aC5hYnMobnkxIC0geTEpIC0gYWR5ICogTWF0aC5hYnMobngxIC0geDEpKTsKLSAgICAgICAgICAgICAgICAgICAgeCA9IG54MTsKLSAgICAgICAgICAgICAgICAgICAgeSA9IG55MTsKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyaXplKG55MiAtIG55MSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgICAgICAgICAgdm9pZCByYXN0ZXJpemUoaW50IGNvdW50KSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCBweSA9IHk7Ci0gICAgICAgICAgICAgICAgICAgIHdoaWxlIChjb3VudC0tID4gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGUgPj0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRzdC5hZGRSZWN0KHgsIHB5LCB4LCB5KTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB4ICs9IHN4OwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHkgKz0gc3k7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZSArPSBlUG9zOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHB5ID0geTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgeSArPSBzeTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBlICs9IGVOZWc7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgZHN0LmFkZFJlY3QoeCwgcHksIHgsIHkpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgc2tpcChpbnQgY291bnQpIHsKLSAgICAgICAgICAgICAgICAgICAgd2hpbGUgKGNvdW50LS0gPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB5ICs9IHN5OwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGUgPj0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHggKz0gc3g7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZSArPSBlUG9zOwotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBlICs9IGVOZWc7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgc3RhdGljIGNsYXNzIEhvckRhc2hlZCBleHRlbmRzIEhvciB7Ci0KLSAgICAgICAgICAgICAgICBMaW5lRGFzaGVyIGxvY2FsOwotCi0gICAgICAgICAgICAgICAgSG9yRGFzaGVkKGludCB4MSwgaW50IHkxLCBpbnQgeDIsIGludCB5MiwgTXVsdGlSZWN0QXJlYSBkc3QsIExpbmVEYXNoZXIgZGFzaGVyLCBib29sZWFuIGludmVydCkgewotICAgICAgICAgICAgICAgICAgICBzdXBlcih4MSwgeTEsIHgyLCB5MiwgZHN0KTsKLSAgICAgICAgICAgICAgICAgICAgZmxvYXQgbGVuZ3RoID0gZ2V0TGVuZ3RoKCk7Ci0gICAgICAgICAgICAgICAgICAgIGxvY2FsID0gZGFzaGVyLmNyZWF0ZURpYWdvbmFsKHhjb3VudCAvIGxlbmd0aCwgbGVuZ3RoLCBpbnZlcnQpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplKCkgewotICAgICAgICAgICAgICAgICAgICBlID0gZUJhc2U7Ci0gICAgICAgICAgICAgICAgICAgIHggPSB4MTsKLSAgICAgICAgICAgICAgICAgICAgeSA9IHkxOwotICAgICAgICAgICAgICAgICAgICByYXN0ZXJpemVEYXNoKHhjb3VudCwgbG9jYWwpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplQ2xpcHBlZChpbnQgbngxLCBpbnQgbnkxLCBpbnQgbngyLCBpbnQgbnkyKSB7Ci0gICAgICAgICAgICAgICAgICAgIGUgPSBlQmFzZSArIDIgKiAoYWR5ICogTWF0aC5hYnMobngxIC0geDEpIC0gYWR4ICogTWF0aC5hYnMobnkxIC0geTEpKTsKLSAgICAgICAgICAgICAgICAgICAgeCA9IG54MTsKLSAgICAgICAgICAgICAgICAgICAgeSA9IG55MTsKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyaXplRGFzaChNYXRoLmFicyhueDIgLSBueDEpLCBsb2NhbC5jcmVhdGVDaGlsZChNYXRoLmFicyhueDEgLSB4MSkpKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgc3RhdGljIGNsYXNzIFZlckRhc2hlZCBleHRlbmRzIFZlciB7Ci0KLSAgICAgICAgICAgICAgICBMaW5lRGFzaGVyIGxvY2FsOwotCi0gICAgICAgICAgICAgICAgVmVyRGFzaGVkKGludCB4MSwgaW50IHkxLCBpbnQgeDIsIGludCB5MiwgTXVsdGlSZWN0QXJlYSBkc3QsIExpbmVEYXNoZXIgZGFzaGVyLCBib29sZWFuIGludmVydCkgewotICAgICAgICAgICAgICAgICAgICBzdXBlcih4MSwgeTEsIHgyLCB5MiwgZHN0KTsKLSAgICAgICAgICAgICAgICAgICAgZmxvYXQgbGVuZ3RoID0gZ2V0TGVuZ3RoKCk7Ci0gICAgICAgICAgICAgICAgICAgIGxvY2FsID0gZGFzaGVyLmNyZWF0ZURpYWdvbmFsKHhjb3VudCAvIGxlbmd0aCwgbGVuZ3RoLCBpbnZlcnQpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplKCkgewotICAgICAgICAgICAgICAgICAgICBlID0gZUJhc2U7Ci0gICAgICAgICAgICAgICAgICAgIHggPSB4MTsKLSAgICAgICAgICAgICAgICAgICAgeSA9IHkxOwotICAgICAgICAgICAgICAgICAgICByYXN0ZXJpemVEYXNoKHhjb3VudCwgbG9jYWwpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplQ2xpcHBlZChpbnQgbngxLCBpbnQgbnkxLCBpbnQgbngyLCBpbnQgbnkyKSB7Ci0gICAgICAgICAgICAgICAgICAgIGUgPSBlQmFzZSArIDIgKiAoYWR4ICogTWF0aC5hYnMobnkxIC0geTEpIC0gYWR5ICogTWF0aC5hYnMobngxIC0geDEpKTsKLSAgICAgICAgICAgICAgICAgICAgeCA9IG54MTsKLSAgICAgICAgICAgICAgICAgICAgeSA9IG55MTsKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyaXplRGFzaChueTIgLSBueTEsIGxvY2FsLmNyZWF0ZUNoaWxkKG55MSAtIHkxKSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgdm9pZCByYXN0ZXJpemUoaW50W10gY2xpcCwgaW50IGluZGV4KSB7Ci0gICAgICAgICAgICAgICAgaW50IGN4MSA9IGNsaXBbaW5kZXggKyAwXTsKLSAgICAgICAgICAgICAgICBpbnQgY3kxID0gY2xpcFtpbmRleCArIDFdOwotICAgICAgICAgICAgICAgIGludCBjeDIgPSBjbGlwW2luZGV4ICsgMl0gKyAxOwotICAgICAgICAgICAgICAgIGludCBjeTIgPSBjbGlwW2luZGV4ICsgM10gKyAxOwotCi0gICAgICAgICAgICAgICAgaW50IGNvZGUxID0KLSAgICAgICAgICAgICAgICAgICAgKHgxIDwgY3gxID8gMSA6IDApIHwgKHgxID49IGN4MiA/IDIgOiAwKSB8Ci0gICAgICAgICAgICAgICAgICAgICh5MSA8IGN5MSA/IDggOiAwKSB8ICh5MSA+PSBjeTIgPyA0IDogMCk7Ci0gICAgICAgICAgICAgICAgaW50IGNvZGUyID0KLSAgICAgICAgICAgICAgICAgICAgKHgyIDwgY3gxID8gMSA6IDApIHwgKHgyID49IGN4MiA/IDIgOiAwKSB8Ci0gICAgICAgICAgICAgICAgICAgICh5MiA8IGN5MSA/IDggOiAwKSB8ICh5MiA+PSBjeTIgPyA0IDogMCk7Ci0KLSAgICAgICAgICAgICAgICAvLyBPdXRzaWRlCi0gICAgICAgICAgICAgICAgaWYgKChjb2RlMSAmIGNvZGUyKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAvLyBJbnNpZGUKLSAgICAgICAgICAgICAgICBpZiAoY29kZTEgPT0gMCAmJiBjb2RlMiA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJhc3Rlcml6ZSgpOwotICAgICAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgLy8gQ2xpcAotICAgICAgICAgICAgICAgIGludCBueDEgPSB4MTsKLSAgICAgICAgICAgICAgICBpbnQgbnkxID0geTE7Ci0gICAgICAgICAgICAgICAgaW50IG54MiA9IHgyOwotICAgICAgICAgICAgICAgIGludCBueTIgPSB5MjsKLSAgICAgICAgICAgICAgICAvLyBuZWVkIHRvIGNsaXAKLSAgICAgICAgICAgICAgICBjeDEgLT0geDE7IGN4MiAtPSB4MTsKLSAgICAgICAgICAgICAgICBjeTEgLT0geTE7IGN5MiAtPSB5MTsKLS8vICAgICAgICAgICAgICAgIGludCBkOwotICAgICAgICAgICAgICAgIGludCBuZXd4MSA9IDAsIG5ld3kxID0gMCwgbmV3eDIgPSAwLCBuZXd5MiA9IDA7Ci0gICAgICAgICAgICAgICAgaWYgKGNvZGUxICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgbmV3eDEgPSBJbnRlZ2VyLk1BWF9WQUxVRTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKChjb2RlMSAmIDgpICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIGNsaXAgcG9pbnQgMSB3aXRoIHRvcCBjbGlwIGJvdW5kCi0gICAgICAgICAgICAgICAgICAgICAgICBuZXd5MSA9IGN5MTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG5ld3gxID0gY2xpcFkoZHgsIGR5LCBuZXd5MSwgdHJ1ZSk7Ci0KLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIGlmICgoY29kZTEgJiA0KSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyBjbGlwIHBvaW50IDEgd2l0aCBib3R0b20gY2xpcCBib3VuZAotICAgICAgICAgICAgICAgICAgICAgICAgbmV3eTEgPSBjeTIgLSAxOwotICAgICAgICAgICAgICAgICAgICAgICAgbmV3eDEgPSBjbGlwWShkeCwgZHksIG5ld3kxLCBmYWxzZSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWYgKChjb2RlMSAmIDEpICE9IDAgJiYgKGN4MSA+IG5ld3gxIHx8IG5ld3gxID09IEludGVnZXIuTUFYX1ZBTFVFKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gY2xpcCBwb2ludCAxIHdpdGggbGVmdCBjbGlwIGJvdW5kCi0gICAgICAgICAgICAgICAgICAgICAgICBuZXd4MSA9IGN4MTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG5ld3kxID0gY2xpcFgoZHgsIGR5LCBuZXd4MSwgZmFsc2UpOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKChjb2RlMSAmIDIpICE9IDAgJiYgKG5ld3gxID49IGN4MiB8fCBuZXd4MSA9PSBJbnRlZ2VyLk1BWF9WQUxVRSkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIGNsaXAgcG9pbnQgMSB3aXRoIHJpZ2h0IGNsaXAgYm91bmQKLSAgICAgICAgICAgICAgICAgICAgICAgIG5ld3gxID0gY3gyIC0gMTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG5ld3kxID0gY2xpcFgoZHgsIGR5LCBuZXd4MSwgZmFsc2UpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGlmIChuZXd4MSA8IGN4MSB8fCBuZXd4MSA+PSBjeDIgfHwgbmV3eTEgPCBjeTEgfHwgbmV3eTEgPj0gY3kyKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICAgICAgICAgIH0KLS8vICAgICAgICAgICAgICAgICAgICBkID0gMiAqIChhZHkgKiBNYXRoLmFicyhuZXd4MSkgLSBhZHggKiBNYXRoLmFicyhuZXd5MSkpICsgMiAqIGFkeSAtIGFkeDsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotLy8gICAgICAgICAgICAgICAgICAgIGQgPSAoYWR5IDw8IDEpIC0gYWR4OwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChjb2RlMiAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIG5ld3gyPUludGVnZXIuTUFYX1ZBTFVFOwotICAgICAgICAgICAgICAgICAgICBpZiAoKGNvZGUyICYgOCkgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gY2xpcCBwb2ludCAyIHdpdGggdG9wIGNsaXAgYm91bmQKLSAgICAgICAgICAgICAgICAgICAgICAgIG5ld3kyID0gY3kxOwotICAgICAgICAgICAgICAgICAgICAgICAgbmV3eDIgPSBjbGlwWShkeCwgZHksIG5ld3kyLCB0cnVlKTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIGlmICgoY29kZTIgJiA0KSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyBjbGlwIHBvaW50IDIgd2l0aCBib3R0b20gY2xpcCBib3VuZAotICAgICAgICAgICAgICAgICAgICAgICAgbmV3eTIgPSBjeTIgLSAxOwotICAgICAgICAgICAgICAgICAgICAgICAgbmV3eDIgPSBjbGlwWShkeCwgZHksIG5ld3kyLCBmYWxzZSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWYgKChjb2RlMiAmIDEpICE9IDAgJiYgKGN4MSA+IG5ld3gyIHx8IG5ld3gyID09IEludGVnZXIuTUFYX1ZBTFVFKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gY2xpcCBwb2ludCAyIHdpdGggbGVmdCBjbGlwIGJvdW5kCi0gICAgICAgICAgICAgICAgICAgICAgICBuZXd4MiA9IGN4MTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG5ld3kyID0gY2xpcFgoZHgsIGR5LCBuZXd4MiwgZmFsc2UpOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKChjb2RlMiAmIDIpICE9IDAgJiYgKG5ld3gyID49IGN4MiB8fCBuZXd4MiA9PSBJbnRlZ2VyLk1BWF9WQUxVRSkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIGNsaXAgcG9pbnQgMiB3aXRoIHJpZ2h0IGNsaXAgYm91bmQKLSAgICAgICAgICAgICAgICAgICAgICAgIG5ld3gyID0gY3gyIC0gMTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG5ld3kyID0gY2xpcFgoZHgsIGR5LCBuZXd4MiwgZmFsc2UpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGlmIChuZXd4MiA8IGN4MSB8fCBuZXd4MiA+PSBjeDIgfHwgbmV3eTIgPCBjeTEgfHwgbmV3eTIgPj0gY3kyKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgbngyID0geDEgKyBuZXd4MjsKLSAgICAgICAgICAgICAgICAgICAgbnkyID0geTEgKyBuZXd5MjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbngxID0geDEgKyBuZXd4MTsKLSAgICAgICAgICAgICAgICBueTEgPSB5MSArIG5ld3kxOwotCi0gICAgICAgICAgICAgICAgcmFzdGVyaXplQ2xpcHBlZChueDEsIG55MSwgbngyLCBueTIpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBhYnN0cmFjdCB2b2lkIHJhc3Rlcml6ZUNsaXBwZWQoaW50IG54MSwgaW50IG55MSwgaW50IG54MiwgaW50IG55Mik7Ci0KLSAgICAgICAgfQotCi0gICAgICAgIHN0YXRpYyBhYnN0cmFjdCBjbGFzcyBPcnRvZyBleHRlbmRzIExpbmUgewotCi0gICAgICAgICAgICBPcnRvZyhpbnQgeDEsIGludCB5MSwgaW50IHgyLCBpbnQgeTIsIE11bHRpUmVjdEFyZWEgZHN0KSB7Ci0gICAgICAgICAgICAgICAgc3VwZXIoeDEsIHkxLCB4MiwgeTIsIGRzdCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHN0YXRpYyBjbGFzcyBIb3IgZXh0ZW5kcyBPcnRvZyB7Ci0KLSAgICAgICAgICAgICAgICBpbnQgZHg7Ci0KLSAgICAgICAgICAgICAgICBIb3IoaW50IHgxLCBpbnQgeTEsIGludCB4MiwgaW50IHkyLCBNdWx0aVJlY3RBcmVhIGRzdCkgewotICAgICAgICAgICAgICAgICAgICBzdXBlcih4MSwgeTEsIHgyLCB5MiwgZHN0KTsKLSAgICAgICAgICAgICAgICAgICAgZHggPSB4MiAtIHgxOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplKCkgewotICAgICAgICAgICAgICAgICAgICBpZiAoZHggPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdCh4MSwgeTEsIHgyLCB5Mik7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdCh4MiwgeTIsIHgxLCB5MSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgICAgICAgICB2b2lkIHJhc3Rlcml6ZShpbnQgc3RlcCkgewotICAgICAgICAgICAgICAgICAgICBpbnQgcHggPSB4OwotICAgICAgICAgICAgICAgICAgICBpZiAoZHggPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB4ICs9IHN0ZXA7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdChweCwgeTEsIHggLSAxLCB5Mik7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB4IC09IHN0ZXA7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdCh4ICsgMSwgeTIsIHB4LCB5MSk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgICAgICAgICB2b2lkIHNraXAoaW50IHN0ZXApIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGR4ID4gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgeCArPSBzdGVwOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgeCAtPSBzdGVwOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgdm9pZCByYXN0ZXJpemVDbGlwcGVkKGludCBueDEsIGludCBueDIpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKG54MSA8IG54MikgewotICAgICAgICAgICAgICAgICAgICAgICAgZHN0LmFkZFJlY3QobngxLCB5MSwgbngyLCB5MSk7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdChueDIsIHkxLCBueDEsIHkxKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplKGludFtdIGNsaXAsIGludCBpbmRleCkgewotICAgICAgICAgICAgICAgICAgICBpZiAoeTEgPj0gY2xpcFtpbmRleCArIDFdICYmIHkxIDw9IGNsaXBbaW5kZXggKyAzXSkgewotICAgICAgICAgICAgICAgICAgICAgICAgaW50IGN4MSA9IGNsaXBbaW5kZXggKyAwXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGludCBjeDIgPSBjbGlwW2luZGV4ICsgMl07Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoeDEgPD0gY3gyICYmIHgyID49IGN4MSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBueDEsIG54MjsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoZHggPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG54MSA9IE1hdGgubWF4KHgxLCBjeDEpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBueDIgPSBNYXRoLm1pbih4MiwgY3gyKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBueDIgPSBNYXRoLm1heCh4MiwgY3gxKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbngxID0gTWF0aC5taW4oeDEsIGN4Mik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlcml6ZUNsaXBwZWQobngxLCBueDIpOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHN0YXRpYyBjbGFzcyBWZXIgZXh0ZW5kcyBPcnRvZyB7Ci0KLSAgICAgICAgICAgICAgICBpbnQgZHk7Ci0KLSAgICAgICAgICAgICAgICBWZXIoaW50IHgxLCBpbnQgeTEsIGludCB4MiwgaW50IHkyLCBNdWx0aVJlY3RBcmVhIGRzdCkgewotICAgICAgICAgICAgICAgICAgICBzdXBlcih4MSwgeTEsIHgyLCB5MiwgZHN0KTsKLSAgICAgICAgICAgICAgICAgICAgZHkgPSB5MiAtIHkxOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplKCkgewotICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdCh4MSwgeTEsIHgyLCB5Mik7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgICAgICAgICAgdm9pZCByYXN0ZXJpemUoaW50IHN0ZXApIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IHB5ID0geTsKLSAgICAgICAgICAgICAgICAgICAgeSArPSBzdGVwOwotICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdCh4MSwgcHksIHgyLCB5IC0gMSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgICAgICAgICAgdm9pZCBza2lwKGludCBzdGVwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHkgKz0gc3RlcDsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICB2b2lkIHJhc3Rlcml6ZUNsaXBwZWQoaW50IG55MSwgaW50IG55MikgewotICAgICAgICAgICAgICAgICAgICBkc3QuYWRkUmVjdCh4MSwgbnkxLCB4MSwgbnkyKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgICAgICAgICB2b2lkIHJhc3Rlcml6ZShpbnRbXSBjbGlwLCBpbnQgaW5kZXgpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHgxID49IGNsaXBbaW5kZXhdICYmIHgxIDw9IGNsaXBbaW5kZXggKyAyXSkgewotICAgICAgICAgICAgICAgICAgICAgICAgaW50IGN5MSA9IGNsaXBbaW5kZXggKyAxXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGludCBjeTIgPSBjbGlwW2luZGV4ICsgM107Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoeTEgPD0gY3kyICYmIHkyID49IGN5MSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlcml6ZUNsaXBwZWQoTWF0aC5tYXgoeTEsIGN5MSksIE1hdGgubWluKHkyLCBjeTIpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBzdGF0aWMgY2xhc3MgSG9yRGFzaGVkIGV4dGVuZHMgSG9yIHsKLQotICAgICAgICAgICAgICAgIExpbmVEYXNoZXIgbG9jYWw7Ci0KLSAgICAgICAgICAgICAgICBIb3JEYXNoZWQoaW50IHgxLCBpbnQgeTEsIGludCB4MiwgaW50IHkyLCBNdWx0aVJlY3RBcmVhIGRzdCwgTGluZURhc2hlciBkYXNoZXIpIHsKLSAgICAgICAgICAgICAgICAgICAgc3VwZXIoeDEsIHkxLCB4MiwgeTIsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIGR4ID0geDIgLSB4MTsKLSAgICAgICAgICAgICAgICAgICAgbG9jYWwgPSBkYXNoZXIuY3JlYXRlT3J0b2dvbmFsKE1hdGguYWJzKGR4KSwgZmFsc2UpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplKCkgewotICAgICAgICAgICAgICAgICAgICB4ID0geDE7Ci0gICAgICAgICAgICAgICAgICAgIHkgPSB5MTsKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyaXplRGFzaChNYXRoLmFicyhkeCksIGxvY2FsKTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgICAgICAgICB2b2lkIHJhc3Rlcml6ZUNsaXBwZWQoaW50IG54MSwgaW50IG54MikgewotICAgICAgICAgICAgICAgICAgICB4ID0gbngxOwotICAgICAgICAgICAgICAgICAgICB5ID0geTE7Ci0gICAgICAgICAgICAgICAgICAgIHJhc3Rlcml6ZURhc2goTWF0aC5hYnMobngyIC0gbngxKSwgbG9jYWwuY3JlYXRlQ2hpbGQoTWF0aC5hYnMobngxIC0geDEpKSk7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHN0YXRpYyBjbGFzcyBWZXJEYXNoZWQgZXh0ZW5kcyBWZXIgewotCi0gICAgICAgICAgICAgICAgTGluZURhc2hlciBsb2NhbDsKLQotICAgICAgICAgICAgICAgIFZlckRhc2hlZChpbnQgeDEsIGludCB5MSwgaW50IHgyLCBpbnQgeTIsIE11bHRpUmVjdEFyZWEgZHN0LCBMaW5lRGFzaGVyIGRhc2hlciwgYm9vbGVhbiBpbnZlcnQpIHsKLSAgICAgICAgICAgICAgICAgICAgc3VwZXIoeDEsIHkxLCB4MiwgeTIsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIGR5ID0geTIgLSB5MTsKLSAgICAgICAgICAgICAgICAgICAgbG9jYWwgPSBkYXNoZXIuY3JlYXRlT3J0b2dvbmFsKGR5LCBpbnZlcnQpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplKCkgewotICAgICAgICAgICAgICAgICAgICB4ID0geDE7Ci0gICAgICAgICAgICAgICAgICAgIHkgPSB5MTsKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyaXplRGFzaChkeSwgbG9jYWwpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgICAgIHZvaWQgcmFzdGVyaXplQ2xpcHBlZChpbnQgbnkxLCBpbnQgbnkyKSB7Ci0gICAgICAgICAgICAgICAgICAgIHggPSB4MTsKLSAgICAgICAgICAgICAgICAgICAgeSA9IG55MTsKLSAgICAgICAgICAgICAgICAgICAgcmFzdGVyaXplRGFzaChueTIgLSBueTEsIGxvY2FsLmNyZWF0ZUNoaWxkKG55MSkpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLQotICAgICAgICBhYnN0cmFjdCB2b2lkIHJhc3Rlcml6ZSgpOwotICAgICAgICBhYnN0cmFjdCB2b2lkIHJhc3Rlcml6ZShpbnRbXSBjbGlwLCBpbnQgaW5kZXgpOwotICAgICAgICBhYnN0cmFjdCB2b2lkIHJhc3Rlcml6ZShpbnQgY291bnQpOwotICAgICAgICBhYnN0cmFjdCB2b2lkIHNraXAoaW50IGNvdW50KTsKLQotICAgICAgICB2b2lkIHJhc3Rlcml6ZURhc2goaW50IGNvdW50LCBMaW5lRGFzaGVyIGRhc2hlcikgewotICAgICAgICAgICAgZmxvYXQgZGVsdGEgPSBkYXNoZXIuZGFzaFtkYXNoZXIuaW5kZXhdIC0gZGFzaGVyLnBoYXNlOwotICAgICAgICAgICAgaW50IHN0ZXAgPSAoaW50KWRlbHRhOwotICAgICAgICAgICAgZGVsdGEgLT0gc3RlcDsKLSAgICAgICAgICAgIHdoaWxlKGNvdW50ID4gc3RlcCkgewotICAgICAgICAgICAgICAgIGlmIChkYXNoZXIudmlzaWJsZSkgewotICAgICAgICAgICAgICAgICAgICByYXN0ZXJpemUoc3RlcCk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgc2tpcChzdGVwKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgY291bnQgLT0gc3RlcDsKLSAgICAgICAgICAgICAgICBkZWx0YSArPSBkYXNoZXIubmV4dERhc2goKTsKLSAgICAgICAgICAgICAgICBzdGVwID0gKGludClkZWx0YTsKLSAgICAgICAgICAgICAgICBkZWx0YSAtPSBzdGVwOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGNvdW50ID4gMCAmJiBkYXNoZXIudmlzaWJsZSkgewotICAgICAgICAgICAgICAgIHJhc3Rlcml6ZShjb3VudCk7Ci0gICAgICAgICAgICAgICAgZGFzaGVyLm1vdmUoY291bnQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBDb21tb24gY2xpcHBpbmcgbWV0aG9kCi0gICAgICovCi0gICAgc3RhdGljIGludCBjbGlwKGludCBkWDEsIGludCBkWDIsIGludCBjWCwgYm9vbGVhbiB0b3ApIHsKLSAgICAgICAgaW50IGFkWDEgPSBkWDEgPCAwID8gLWRYMSA6IGRYMTsKLSAgICAgICAgaW50IGFkWDIgPSBkWDIgPCAwID8gLWRYMiA6IGRYMjsKLSAgICAgICAgaWYgKGFkWDEgPD0gYWRYMikgewotICAgICAgICAgICAgLy8gb2J0dXNlIGludGVyc2VjdGlvbiBhbmdsZQotICAgICAgICAgICAgcmV0dXJuICgoZFgxIDw8IDEpICogY1ggKyAoZFgxID4gMCA/IGRYMiA6IC1kWDIpKSAvIChkWDIgPDwgMSk7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IGs7Ci0gICAgICAgIGlmICh0b3ApIHsKLSAgICAgICAgICAgIGsgPSAtZFgxICsgKGRYMiA8IDAgPyAwIDogZFgxID4gMCA/IChkWDIgPDwgMSkgOiAtKGRYMiA8PCAxKSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBrID0gZFgxICsgKGRYMiA+IDAgPyAwIDogZFgxID4gMCA/IChkWDIgPDwgMSkgOiAtKGRYMiA8PCAxKSk7Ci0gICAgICAgIH0KLQotICAgICAgICBrICs9IGRYMSA+IDAgPT0gZFgyID4gMCA/IC0xIDogMTsKLSAgICAgICAgcmV0dXJuICgoZFgxIDw8IDEpICogY1ggKyBrKSAvIChkWDIgPDwgMSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2xpcHBpbmcgYWxvbmcgWCBheGlzCi0gICAgICovCi0gICAgc3RhdGljIGludCBjbGlwWChpbnQgZHgsIGludCBkeSwgaW50IGN5LCBib29sZWFuIHRvcCkgewotICAgICAgICByZXR1cm4gY2xpcChkeSwgZHgsIGN5LCB0b3ApOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIENsaXBwaW5nIGFsb25nIFkgYXhpcwotICAgICAqLwotICAgIHN0YXRpYyBpbnQgY2xpcFkoaW50IGR4LCBpbnQgZHksIGludCBjeCwgYm9vbGVhbiB0b3ApIHsKLSAgICAgICAgcmV0dXJuIGNsaXAoZHgsIGR5LCBjeCwgdG9wKTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSYXN0ZXJpemVzIGxpbmUgdXNpbmcgY2xpcHBpbmQgYW5kIGRhc2hpbmcgc3R5bGUKLSAgICAgKiBAcGFyYW0geDEgLSB0aGUgeCBjb29yZGluYXRlIG9mIHRoZSBmaXJzdCBjb250cm9sIHBvaW50Ci0gICAgICogQHBhcmFtIHkxIC0gdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgZmlyc3QgY29udHJvbCBwb2ludAotICAgICAqIEBwYXJhbSB4MiAtIHRoZSB4IGNvb3JkaW5hdGUgb2YgdGhlIHNlY29uZCBjb250cm9sIHBvaW50Ci0gICAgICogQHBhcmFtIHkyIC0gdGhlIHkgY29vcmRpbmF0ZSBvZiB0aGUgc2Vjb25kIGNvbnRyb2wgcG9pbnQKLSAgICAgKiBAcGFyYW0gY2xpcCAtIHRoZSBNdWx0aVJlY3RBcmVhIG9iamVjdCBvZiBjbGlwcGluZyBhcmVhCi0gICAgICogQHBhcmFtIGRhc2hlciAtIHRoZSBkYXNoZXIgc3R5bGUKLSAgICAgKiBAcGFyYW0gaW52ZXJ0IC0gdGhlIGludmVydCBpbmRpY2F0b3IsIGFsd2F5cyBmYWxzZQotICAgICAqIEByZXR1cm4gYSBNdWx0aVJlY3RBcmVhIG9mIHJhc3Rlcml6ZXIgbGluZQotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgTXVsdGlSZWN0QXJlYSByYXN0ZXJpemUoaW50IHgxLCBpbnQgeTEsIGludCB4MiwgaW50IHkyLCBNdWx0aVJlY3RBcmVhIGNsaXAsIExpbmVEYXNoZXIgZGFzaGVyLCBib29sZWFuIGludmVydCkgewotCi0gICAgICAgIE11bHRpUmVjdEFyZWEgZHN0ID0gbmV3IE11bHRpUmVjdEFyZWEoZmFsc2UpOwotICAgICAgICBpbnQgZHggPSB4MiAtIHgxOwotICAgICAgICBpbnQgZHkgPSB5MiAtIHkxOwotCi0gICAgICAgIC8vIFBvaW50Ci0gICAgICAgIGlmIChkeCA9PSAwICYmIGR5ID09IDApIHsKLSAgICAgICAgICAgIGlmICgoY2xpcCA9PSBudWxsIHx8IGNsaXAuY29udGFpbnMoeDEsIHkxKSkgJiYgKGRhc2hlciA9PSBudWxsIHx8IGRhc2hlci52aXNpYmxlKSkgewotICAgICAgICAgICAgICAgIGRzdCA9IG5ldyBNdWx0aVJlY3RBcmVhKHgxLCB5MSwgeDEsIHkxKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBkc3Q7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZHkgPCAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gcmFzdGVyaXplKHgyLCB5MiwgeDEsIHkxLCBjbGlwLCBkYXNoZXIsIHRydWUpOwotICAgICAgICB9Ci0KLSAgICAgICAgTGluZSBsaW5lOwotICAgICAgICBpZiAoZGFzaGVyID09IG51bGwpIHsKLSAgICAgICAgICAgIGlmIChkeCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgbGluZSA9IG5ldyBMaW5lLk9ydG9nLlZlcih4MSwgeTEsIHgyLCB5MiwgZHN0KTsKLSAgICAgICAgICAgIH0gZWxzZQotICAgICAgICAgICAgICAgIGlmIChkeSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGxpbmUgPSBuZXcgTGluZS5PcnRvZy5Ib3IoeDEsIHkxLCB4MiwgeTIsIGRzdCk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGR5IDwgTWF0aC5hYnMoZHgpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBsaW5lID0gbmV3IExpbmUuRGlhZy5Ib3IoeDEsIHkxLCB4MiwgeTIsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBsaW5lID0gbmV3IExpbmUuRGlhZy5WZXIoeDEsIHkxLCB4MiwgeTIsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpZiAoZHggPT0gMCkgewotICAgICAgICAgICAgICAgIGxpbmUgPSBuZXcgTGluZS5PcnRvZy5WZXJEYXNoZWQoeDEsIHkxLCB4MiwgeTIsIGRzdCwgZGFzaGVyLCBpbnZlcnQpOwotICAgICAgICAgICAgfSBlbHNlCi0gICAgICAgICAgICAgICAgaWYgKGR5ID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgbGluZSA9IG5ldyBMaW5lLk9ydG9nLkhvckRhc2hlZCh4MSwgeTEsIHgyLCB5MiwgZHN0LCBkYXNoZXIpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChkeSA8IE1hdGguYWJzKGR4KSkgewotICAgICAgICAgICAgICAgICAgICAgICAgbGluZSA9IG5ldyBMaW5lLkRpYWcuSG9yRGFzaGVkKHgxLCB5MSwgeDIsIHkyLCBkc3QsIGRhc2hlciwgaW52ZXJ0KTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGxpbmUgPSBuZXcgTGluZS5EaWFnLlZlckRhc2hlZCh4MSwgeTEsIHgyLCB5MiwgZHN0LCBkYXNoZXIsIGludmVydCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotCi0gICAgICAgIGlmIChjbGlwID09IG51bGwgfHwgY2xpcC5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgIGxpbmUucmFzdGVyaXplKCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmb3IoaW50IGkgPSAxOyBpIDwgY2xpcC5yZWN0WzBdOyBpICs9IDQpIHsKLSAgICAgICAgICAgICAgICBsaW5lLnJhc3Rlcml6ZShjbGlwLnJlY3QsIGkpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIGRzdDsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL3JlbmRlci9KYXZhU2hhcGVSYXN0ZXJpemVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvSmF2YVNoYXBlUmFzdGVyaXplci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkYmFhZjUzLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2dsL3JlbmRlci9KYXZhU2hhcGVSYXN0ZXJpemVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0NzUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEZW5pcyBNLiBLaXNoZW5rbwotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wucmVuZGVyOwotCi1pbXBvcnQgamF2YS5hd3QuU2hhcGU7Ci1pbXBvcnQgamF2YS5hd3QuZ2VvbS5QYXRoSXRlcmF0b3I7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLk11bHRpUmVjdEFyZWE7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLXB1YmxpYyBjbGFzcyBKYXZhU2hhcGVSYXN0ZXJpemVyIHsKLQotICAgIHN0YXRpYyBmaW5hbCBpbnQgUE9JTlRfQ0FQQUNJVFkgPSAxNjsKLQotICAgIGludCBlZGdlc0NvdW50OwotICAgIGludCBlZGdlQ3VyOwotICAgIGludFtdIGVkZ2VzWDsKLSAgICBpbnRbXSBlZGdlc1k7Ci0gICAgaW50W10gZWRnZXNZUzsgLy8gWSBjb29yZGluYXRlIG9mIGVkZ2UgU1RBUlQgcG9pbnQKLSAgICBpbnRbXSBlZGdlc047Ci0gICAgaW50W10gZWRnZXNEWTsKLSAgICBpbnRbXSBib3VuZHM7Ci0gICAgaW50IGJvdW5kQ291bnQ7Ci0gICAgYm9vbGVhbltdIGVkZ2VzRXh0OyAvLyBFeHRyZW1hbCBwb2ludHMKLQotICAgIGludCBhY3RpdmVDb3VudDsKLSAgICBmbG9hdFtdIGFjdGl2ZVg7Ci0gICAgaW50W10gYWN0aXZlWUVuZDsKLSAgICBmbG9hdFtdIGFjdGl2ZVhTdGVwOwotICAgIGludFtdIGFjdGl2ZURZOwotICAgIGJvb2xlYW5bXSBhY3RpdmVFeHQ7Ci0KLSAgICBpbnRbXSBjcm9zc1g7Ci0gICAgaW50W10gY3Jvc3NEWTsKLQotICAgIEZpbGxlciBmaWxsZXI7Ci0KLSAgICAvKioKLSAgICAgKiBSYXN0ZXJpemF0aW9uIGZpbGxlciBmb3IgZGlmZmVyZW50IHBhdGggcnVsZXMKLSAgICAgKi8KLSAgICBzdGF0aWMgYWJzdHJhY3QgY2xhc3MgRmlsbGVyIHsKLQotICAgICAgICBzdGF0aWMgY2xhc3MgTm9uWmVybyBleHRlbmRzIEZpbGxlciB7Ci0gICAgICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgICAgIHZvaWQgYWRkKE11bHRpUmVjdEFyZWEuTGluZUNhc2ggcmVjdCwgaW50W10gcG9pbnRzLCBpbnRbXSBvcmllbnQsIGludCBsZW5ndGgsIGludCB5KSB7Ci0KLSAgICAgICAgICAgICAgICBpbnRbXSBkc3QgPSBuZXcgaW50W2xlbmd0aF07Ci0gICAgICAgICAgICAgICAgaW50IGRzdExlbmd0aCA9IDE7Ci0gICAgICAgICAgICAgICAgZHN0WzBdID0gcG9pbnRzWzBdOwotICAgICAgICAgICAgICAgIGludCBjb3VudCA9IDA7Ci0gICAgICAgICAgICAgICAgYm9vbGVhbiBpbnNpZGUgPSB0cnVlOwotICAgICAgICAgICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBsZW5ndGg7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBjb3VudCArPSBvcmllbnRbaV0gPiAwID8gMSA6IC0xOwotICAgICAgICAgICAgICAgICAgICBpZiAoY291bnQgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgZHN0W2RzdExlbmd0aCsrXSA9IHBvaW50c1tpXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGluc2lkZSA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCFpbnNpZGUpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3RbZHN0TGVuZ3RoKytdID0gcG9pbnRzW2ldOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGluc2lkZSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGZvcihpbnQgaSA9IDE7IGkgPCBkc3RMZW5ndGg7IGkgKz0gMikgewotICAgICAgICAgICAgICAgICAgICBkc3RbaV0tLTsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICBkc3RMZW5ndGggPSBleGNsdWRlRW1wdHkoZHN0LCBkc3RMZW5ndGgpOwotLy8gICAgICAgICAgICAgIFN5c3RlbS5vdXQucHJpbnRsbigidGVzdCIpOwotCi0gICAgICAgICAgICAgICAgZHN0TGVuZ3RoID0gdW5pb24oZHN0LCBkc3RMZW5ndGgpOwotCi0gICAgICAgICAgICAgICAgcmVjdC5hZGRMaW5lKGRzdCwgZHN0TGVuZ3RoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHN0YXRpYyBjbGFzcyBFdmVuT2RkIGV4dGVuZHMgRmlsbGVyIHsKLSAgICAgICAgICAgIEBPdmVycmlkZQotICAgICAgICAgICAgdm9pZCBhZGQoTXVsdGlSZWN0QXJlYS5MaW5lQ2FzaCByZWN0LCBpbnRbXSBwb2ludHMsIGludFtdIG9yaWVudCwgaW50IGxlbmd0aCwgaW50IHkpIHsKLSAgICAvKgotICAgICAgICAgICAgICAgIGludFtdIGJ1ZiA9IG5ldyBpbnRbbGVuZ3RoXTsKLSAgICAgICAgICAgICAgICBpbnQgaiA9IDA7Ci0gICAgICAgICAgICAgICAgZm9yKGludCBpID0gMDsgaSA8IGxlbmd0aCAtIDE7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBpZiAocG9pbnRzW2ldICE9IHBvaW50c1tpICsgMV0pIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJ1ZltqKytdID0gcG9pbnRzW2ldOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICovCi0gICAgICAgICAgICAgICAgZm9yKGludCBpID0gMTsgaSA8IGxlbmd0aDsgaSArPSAyKSB7Ci0gICAgICAgICAgICAgICAgICAgIHBvaW50c1tpXS0tOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGxlbmd0aCA9IGV4Y2x1ZGVFbXB0eShwb2ludHMsIGxlbmd0aCk7Ci0vLyAgICAgICAgICAgICAgU3lzdGVtLm91dC5wcmludGxuKCJ0ZXN0Iik7Ci0KLSAgICAgICAgICAgICAgICBsZW5ndGggPSB1bmlvbihwb2ludHMsIGxlbmd0aCk7Ci0gICAgICAgICAgICAgICAgcmVjdC5hZGRMaW5lKHBvaW50cywgbGVuZ3RoKTsKLSAgICAvKgotICAgICAgICAgICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBsZW5ndGg7KSB7Ci0gICAgICAgICAgICAgICAgICAgIHJlY3QuYWRkKHBvaW50c1tpKytdLCB5LCBwb2ludHNbaSsrXSwgeSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICovCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBhYnN0cmFjdCB2b2lkIGFkZChNdWx0aVJlY3RBcmVhLkxpbmVDYXNoIHJlY3QsIGludFtdIHBvaW50cywgaW50W10gb3JpZW50LCBpbnQgbGVuZ3RoLCBpbnQgeSk7Ci0KLSAgICAgICAgc3RhdGljIGludCBleGNsdWRlRW1wdHkoaW50W10gcG9pbnRzLCBpbnQgbGVuZ3RoKSB7Ci0gICAgICAgICAgICBpbnQgaSA9IDA7Ci0gICAgICAgICAgICB3aGlsZShpIDwgbGVuZ3RoKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHBvaW50c1tpXSA8PSBwb2ludHNbaSArIDFdKSB7Ci0gICAgICAgICAgICAgICAgICAgIGkgKz0gMjsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBsZW5ndGggLT0gMjsKLSAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShwb2ludHMsIGkgKyAyLCBwb2ludHMsIGksIGxlbmd0aCAtIGkpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBsZW5ndGg7Ci0gICAgICAgIH0KLQotICAgICAgICBzdGF0aWMgaW50IHVuaW9uKGludFtdIHBvaW50cywgaW50IGxlbmd0aCkgewotICAgICAgICAgICAgaW50IGkgPSAxOwotICAgICAgICAgICAgd2hpbGUoaSA8IGxlbmd0aCAtIDEpIHsKLSAgICAgICAgICAgICAgICBpZiAocG9pbnRzW2ldIDwgcG9pbnRzW2kgLSAxXSkgewotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHBvaW50cywgaSArIDEsIHBvaW50cywgaSAtIDEsIGxlbmd0aCAtIGkgLSAxKTsKLSAgICAgICAgICAgICAgICAgICAgbGVuZ3RoIC09IDI7Ci0gICAgICAgICAgICAgICAgfSBlbHNlCi0gICAgICAgICAgICAgICAgaWYgKHBvaW50c1tpXSA+PSBwb2ludHNbaSArIDFdIC0gMSkgewotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KHBvaW50cywgaSArIDIsIHBvaW50cywgaSwgbGVuZ3RoIC0gaSAtIDIpOwotICAgICAgICAgICAgICAgICAgICBsZW5ndGggLT0gMjsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBpICs9IDI7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGxlbmd0aDsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgcHVibGljIEphdmFTaGFwZVJhc3Rlcml6ZXIoKSB7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQ2hlY2tzIGJ1ZmZlciBzaXplIGFuZCByZWFsbG9jIGlmIG5lY2Vzc2FyeQotICAgICAqLwotICAgIGludFtdIGNoZWNrQnVmU2l6ZShpbnRbXSBidWYsIGludCBzaXplKSB7Ci0gICAgICAgIGlmIChzaXplID09IGJ1Zi5sZW5ndGgpIHsKLSAgICAgICAgICAgIGludFtdIHRtcDsKLSAgICAgICAgICAgIHRtcCA9IG5ldyBpbnRbc2l6ZSArIFBPSU5UX0NBUEFDSVRZXTsKLSAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkoYnVmLCAwLCB0bXAsIDAsIGJ1Zi5sZW5ndGgpOwotICAgICAgICAgICAgYnVmID0gdG1wOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBidWY7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogQWRkcyB0byB0aGUgYnVmZmVycyBuZXcgZWRnZSAKLSAgICAgKi8KLSAgICB2b2lkIGFkZEVkZ2UoaW50IHgsIGludCB5LCBpbnQgbnVtKSB7Ci0gICAgICAgIGVkZ2VzWCA9IGNoZWNrQnVmU2l6ZShlZGdlc1gsIGVkZ2VzQ291bnQpOwotICAgICAgICBlZGdlc1kgPSBjaGVja0J1ZlNpemUoZWRnZXNZLCBlZGdlc0NvdW50KTsKLSAgICAgICAgZWRnZXNOID0gY2hlY2tCdWZTaXplKGVkZ2VzTiwgZWRnZXNDb3VudCk7Ci0gICAgICAgIGVkZ2VzWFtlZGdlc0NvdW50XSA9IHg7Ci0gICAgICAgIGVkZ2VzWVtlZGdlc0NvdW50XSA9IHk7Ci0gICAgICAgIGVkZ2VzTltlZGdlc0NvdW50XSA9IChudW0gPDwgMTYpIHwgZWRnZXNDb3VudDsKLSAgICAgICAgZWRnZXNDb3VudCsrOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFByZXBhcmUgYWxsIGJ1ZmZlcnMgYW5kIHZhcmlhYmxlIHRvIHJhc3Rlcml6ZSBzaGFwZSAKLSAgICAgKi8KLSAgICB2b2lkIG1ha2VCdWZmZXIoUGF0aEl0ZXJhdG9yIHBhdGgsIGRvdWJsZSBmbGF0bmVzcykgewotICAgICAgICBlZGdlc1ggPSBuZXcgaW50W1BPSU5UX0NBUEFDSVRZXTsKLSAgICAgICAgZWRnZXNZID0gbmV3IGludFtQT0lOVF9DQVBBQ0lUWV07Ci0gICAgICAgIGVkZ2VzTiA9IG5ldyBpbnRbUE9JTlRfQ0FQQUNJVFldOwotICAgICAgICBib3VuZHMgPSBuZXcgaW50W1BPSU5UX0NBUEFDSVRZXTsKLSAgICAgICAgYm91bmRDb3VudCA9IDA7Ci0gICAgICAgIGVkZ2VzQ291bnQgPSAwOwotCi0gICAgICAgIGlmIChwYXRoLmdldFdpbmRpbmdSdWxlKCkgPT0gUGF0aEl0ZXJhdG9yLldJTkRfRVZFTl9PREQpIHsKLSAgICAgICAgICAgIGZpbGxlciA9IG5ldyBGaWxsZXIuRXZlbk9kZCgpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZmlsbGVyID0gbmV3IEZpbGxlci5Ob25aZXJvKCk7Ci0gICAgICAgIH0KLSAgICAgICAgZmxvYXRbXSBjb29yZHMgPSBuZXcgZmxvYXRbMl07Ci0gICAgICAgIGJvb2xlYW4gY2xvc2VkID0gdHJ1ZTsKLSAgICAgICAgd2hpbGUgKCFwYXRoLmlzRG9uZSgpKSB7Ci0gICAgICAgICAgICBzd2l0Y2gocGF0aC5jdXJyZW50U2VnbWVudChjb29yZHMpKSB7Ci0gICAgICAgICAgICBjYXNlIFBhdGhJdGVyYXRvci5TRUdfTU9WRVRPOgotICAgICAgICAgICAgICAgIGlmICghY2xvc2VkKSB7Ci0gICAgICAgICAgICAgICAgICAgIGJvdW5kQ291bnQrKzsKLSAgICAgICAgICAgICAgICAgICAgYm91bmRzID0gY2hlY2tCdWZTaXplKGJvdW5kcywgYm91bmRDb3VudCk7Ci0gICAgICAgICAgICAgICAgICAgIGJvdW5kc1tib3VuZENvdW50XSA9IGVkZ2VzQ291bnQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGFkZEVkZ2UoKGludCljb29yZHNbMF0sIChpbnQpY29vcmRzWzFdLCBib3VuZENvdW50KTsKLSAgICAgICAgICAgICAgICBjbG9zZWQgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgUGF0aEl0ZXJhdG9yLlNFR19MSU5FVE86Ci0gICAgICAgICAgICAgICAgYWRkRWRnZSgoaW50KWNvb3Jkc1swXSwgKGludCljb29yZHNbMV0sIGJvdW5kQ291bnQpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBQYXRoSXRlcmF0b3IuU0VHX0NMT1NFOgotICAgICAgICAgICAgICAgIGJvdW5kQ291bnQrKzsKLSAgICAgICAgICAgICAgICBib3VuZHMgPSBjaGVja0J1ZlNpemUoYm91bmRzLCBib3VuZENvdW50KTsKLSAgICAgICAgICAgICAgICBib3VuZHNbYm91bmRDb3VudF0gPSBlZGdlc0NvdW50OwotICAgICAgICAgICAgICAgIGNsb3NlZCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIC8vIGF3dC4zNj1Xcm9uZyBzZWdtZW50Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMzYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHBhdGgubmV4dCgpOwotICAgICAgICB9Ci0gICAgICAgIGlmICghY2xvc2VkKSB7Ci0gICAgICAgICAgICBib3VuZENvdW50Kys7Ci0gICAgICAgICAgICBib3VuZHMgPSBjaGVja0J1ZlNpemUoYm91bmRzLCBib3VuZENvdW50KTsKLSAgICAgICAgICAgIGJvdW5kc1tib3VuZENvdW50XSA9IGVkZ2VzQ291bnQ7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBTb3J0IGJ1ZmZlcnMKLSAgICAgKi8KLSAgICB2b2lkIHNvcnQoaW50W10gbWFzdGVyLCBpbnRbXSBzbGF2ZSwgaW50IGxlbmd0aCkgewotICAgICAgICBmb3IoaW50IGkgPSAwOyBpIDwgbGVuZ3RoIC0gMTsgaSsrKSB7Ci0gICAgICAgICAgICBpbnQgbnVtID0gaTsKLSAgICAgICAgICAgIGludCBtaW4gPSBtYXN0ZXJbbnVtXTsKLSAgICAgICAgICAgIGZvcihpbnQgaiA9IGkgKyAxOyBqIDwgbGVuZ3RoOyBqKyspIHsKLSAgICAgICAgICAgICAgICBpZiAobWFzdGVyW2pdIDwgbWluKSB7Ci0gICAgICAgICAgICAgICAgICAgIG51bSA9IGo7Ci0gICAgICAgICAgICAgICAgICAgIG1pbiA9IG1hc3RlcltudW1dOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChudW0gIT0gaSkgewotICAgICAgICAgICAgICAgIG1hc3RlcltudW1dID0gbWFzdGVyW2ldOwotICAgICAgICAgICAgICAgIG1hc3RlcltpXSA9IG1pbjsKLSAgICAgICAgICAgICAgICBtaW4gPSBzbGF2ZVtudW1dOwotICAgICAgICAgICAgICAgIHNsYXZlW251bV0gPSBzbGF2ZVtpXTsKLSAgICAgICAgICAgICAgICBzbGF2ZVtpXSA9IG1pbjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIGludCBnZXROZXh0KGludCBjdXIpIHsKLSAgICAgICAgaW50IG4gPSBlZGdlc05bY3VyXTsKLSAgICAgICAgaW50IGJvdW5kID0gbiA+PiAxNjsKLSAgICAgICAgaW50IG51bSA9IChuICYgMHhGRkZGKSArIDE7Ci0gICAgICAgIGlmIChudW0gPT0gYm91bmRzW2JvdW5kICsgMV0pIHsKLSAgICAgICAgICAgIHJldHVybiBib3VuZHNbYm91bmRdOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudW07Ci0gICAgfQotCi0gICAgaW50IGdldFByZXYoaW50IGN1cikgewotICAgICAgICBpbnQgbiA9IGVkZ2VzTltjdXJdOwotICAgICAgICBpbnQgYm91bmQgPSBuID4+IDE2OwotICAgICAgICBpbnQgbnVtID0gKG4gJiAweEZGRkYpIC0gMTsKLSAgICAgICAgaWYgKG51bSA8IGJvdW5kc1tib3VuZF0pIHsKLSAgICAgICAgICAgIHJldHVybiBib3VuZHNbYm91bmQgKyAxXSAtIDE7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bTsKLSAgICB9Ci0KLSAgICBpbnQgZ2V0TmV4dFNoYXBlKGludCBjdXIpIHsKLSAgICAgICAgaW50IGJvdW5kID0gZWRnZXNOW2N1cl0gPj4gMTY7Ci0gICAgICAgIHJldHVybiBib3VuZHNbYm91bmQgKyAxXTsKLSAgICB9Ci0KLSAgICB2b2lkIGluaXQoKSB7Ci0KLSAgICAgICAgZWRnZXNZUyA9IG5ldyBpbnRbZWRnZXNDb3VudF07Ci0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkoZWRnZXNZLCAwLCBlZGdlc1lTLCAwLCBlZGdlc0NvdW50KTsKLSAgICAgICAgLy8gQ3JlYXRlIGVkZ2VzRFkKLSAgICAgICAgZWRnZXNEWSA9IG5ldyBpbnRbZWRnZXNDb3VudF07Ci0gICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBlZGdlc0NvdW50OyBpKyspIHsKLSAgICAgICAgICAgIGludCBkeSA9IGVkZ2VzWVtnZXROZXh0KGkpXSAtIGVkZ2VzWVtpXTsKLSAgICAgICAgICAgIGVkZ2VzRFlbaV0gPSBkeTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIENyZWF0ZSBlZGdlc0V4dAotICAgICAgICBlZGdlc0V4dCA9IG5ldyBib29sZWFuW2VkZ2VzQ291bnRdOwotICAgICAgICBpbnQgcHJldiA9IC0xOwotICAgICAgICBpbnQgaSA9IDA7Ci0gICAgICAgIGludCBwb3MgPSAwOwotICAgICAgICB3aGlsZShpIDwgZWRnZXNDb3VudCkgewotCi0gICAgICAgICAgICBUT1A6IHsKLSAgICAgICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChlZGdlc0RZW2ldID4gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWsgVE9QOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGkgPSBnZXROZXh0KGkpOwotICAgICAgICAgICAgICAgIH0gd2hpbGUgKGkgIT0gcG9zKTsKLSAgICAgICAgICAgICAgICBpID0gcG9zID0gZ2V0TmV4dFNoYXBlKGkpOwotICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBCT1RUT006IHsKLSAgICAgICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChlZGdlc0RZW2ldIDwgMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWsgQk9UVE9NOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGlmIChlZGdlc0RZW2ldID4gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgcHJldiA9IGk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaSA9IGdldE5leHQoaSk7Ci0gICAgICAgICAgICAgICAgfSB3aGlsZSAoaSAhPSBwb3MpOwotICAgICAgICAgICAgICAgIGkgPSBwb3MgPSBnZXROZXh0U2hhcGUoaSk7Ci0gICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChwcmV2ICE9IC0xKSB7Ci0gICAgICAgICAgICAgICAgZWRnZXNFeHRbcHJldl0gPSB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZWRnZXNFeHRbaV0gPSB0cnVlOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gU29ydCBlZGdlc1kgYW5kIGVkZ2VzTgotICAgICAgICBzb3J0KGVkZ2VzWVMsIGVkZ2VzTiwgZWRnZXNDb3VudCk7Ci0KLSAgICAgICAgZWRnZUN1ciA9IDA7Ci0gICAgICAgIGFjdGl2ZUNvdW50ID0gMDsKLSAgICAgICAgYWN0aXZlWCA9IG5ldyBmbG9hdFtlZGdlc0NvdW50XTsKLSAgICAgICAgYWN0aXZlWUVuZCA9IG5ldyBpbnRbZWRnZXNDb3VudF07Ci0gICAgICAgIGFjdGl2ZVhTdGVwID0gbmV3IGZsb2F0W2VkZ2VzQ291bnRdOwotICAgICAgICBhY3RpdmVEWSA9IG5ldyBpbnRbZWRnZXNDb3VudF07Ci0gICAgICAgIGFjdGl2ZUV4dCA9IG5ldyBib29sZWFuW2VkZ2VzQ291bnRdOwotCi0gICAgICAgIGNyb3NzWCA9IG5ldyBpbnRbZWRnZXNDb3VudF07Ci0gICAgICAgIGNyb3NzRFkgPSBuZXcgaW50W2VkZ2VzQ291bnRdOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIE1hcmtzIGVkZ2UgYXMgYWN0aXZlCi0gICAgICovCi0gICAgdm9pZCBhZGRBY3RpdmVFZGdlKGludCBsZXZlbFksIGludCBzdGFydCwgaW50IGVuZCwgYm9vbGVhbiBiYWNrKSB7Ci0gICAgICAgIGludCBkeSA9IGJhY2sgPyAtZWRnZXNEWVtlbmRdIDogZWRnZXNEWVtzdGFydF07Ci0gICAgICAgIGlmIChkeSA8PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgaW50IHgxID0gZWRnZXNYW3N0YXJ0XTsKLSAgICAgICAgaW50IGR4ID0gZWRnZXNYW2VuZF0gLSB4MTsKLSAgICAgICAgYWN0aXZlWFthY3RpdmVDb3VudF0gPSB4MTsKLSAgICAgICAgYWN0aXZlWUVuZFthY3RpdmVDb3VudF0gPSBlZGdlc1lbZW5kXTsKLSAgICAgICAgYWN0aXZlWFN0ZXBbYWN0aXZlQ291bnRdID0gZHggLyAoZmxvYXQpZHk7Ci0gICAgICAgIGFjdGl2ZURZW2FjdGl2ZUNvdW50XSA9IGJhY2sgPyAtZHkgOiBkeTsKLSAgICAgICAgYWN0aXZlRXh0W2FjdGl2ZUNvdW50XSA9IGJhY2sgPyBlZGdlc0V4dFtlbmRdIDogZWRnZXNFeHRbc3RhcnRdOwotICAgICAgICBhY3RpdmVDb3VudCsrOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIEZpbmQgbmV3IGFjdGl2ZSBlZGdlcwotICAgICAqLwotICAgIGludCBmaW5kQWN0aXZlRWRnZXMoaW50IGxldmVsWSkgewotCi0gICAgICAgIGludCBlZGdlQWN0aXZlID0gZWRnZUN1cjsKLSAgICAgICAgd2hpbGUgKGVkZ2VBY3RpdmUgPCBlZGdlc0NvdW50ICYmIGVkZ2VzWVNbZWRnZUFjdGl2ZV0gPT0gbGV2ZWxZKSB7Ci0gICAgICAgICAgICBlZGdlQWN0aXZlKys7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnQgYWN0aXZlTmV4dCA9IGVkZ2VBY3RpdmU7Ci0KLSAgICAgICAgd2hpbGUgKGVkZ2VBY3RpdmUgPiBlZGdlQ3VyKSB7Ci0gICAgICAgICAgICBlZGdlQWN0aXZlLS07Ci0gICAgICAgICAgICBpbnQgbnVtID0gZWRnZXNOW2VkZ2VBY3RpdmVdICYgMHhGRkZGOwotICAgICAgICAgICAgYWRkQWN0aXZlRWRnZShsZXZlbFksIG51bSwgZ2V0UHJldihlZGdlQWN0aXZlKSwgdHJ1ZSk7Ci0gICAgICAgICAgICBhZGRBY3RpdmVFZGdlKGxldmVsWSwgbnVtLCBnZXROZXh0KGVkZ2VBY3RpdmUpLCBmYWxzZSk7Ci0gICAgICAgIH0KLQotICAgICAgICBlZGdlQ3VyID0gYWN0aXZlTmV4dDsKLQotICAgICAgICBpZiAoYWN0aXZlTmV4dCA9PSBlZGdlc0NvdW50KSB7Ci0gICAgICAgICAgICByZXR1cm4gZWRnZXNZW2VkZ2VzQ291bnQgLSAxXTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZWRnZXNZU1thY3RpdmVOZXh0XTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSYXN0ZXJpemVzIHNoYXBlIHdpdGggcGFydGljdWxhciBmbGF0bmVzcwotICAgICAqIEBwYXJhbSBzaGFwZSAtIHRoZSBzb3V6ZSBTaGFwZSB0byBiZSByYXN0ZXJpemVkCi0gICAgICogQHBhcmFtIGZsYXRuZXNzIC0gdGhlIHJhc3Rlcml6YXRpb24gZmxhdG5lc3MKLSAgICAgKiBAcmV0dXJuIGEgTXVsdGlSZWN0QXJlYSBvZiByYXN0ZXJpemVkIHNoYXBlCi0gICAgICovCi0gICAgcHVibGljIE11bHRpUmVjdEFyZWEgcmFzdGVyaXplKFNoYXBlIHNoYXBlLCBkb3VibGUgZmxhdG5lc3MpIHsKLQotICAgICAgICBQYXRoSXRlcmF0b3IgcGF0aCA9IHNoYXBlLmdldFBhdGhJdGVyYXRvcihudWxsLCBmbGF0bmVzcyk7Ci0KLSAgICAgICAgLy8gU2hhcGUgaXMgZW1wdHkKLSAgICAgICAgaWYgKHBhdGguaXNEb25lKCkpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgTXVsdGlSZWN0QXJlYSgpOwotICAgICAgICB9Ci0KLSAgICAgICAgbWFrZUJ1ZmZlcihwYXRoLCBmbGF0bmVzcyk7Ci0KLSAgICAgICAgaW5pdCgpOwotCi0gICAgICAgIGludCB5ID0gZWRnZXNZU1swXTsKLSAgICAgICAgaW50IG5leHRZID0geTsKLSAgICAgICAgaW50IGNyb3NzQ291bnQ7Ci0KLSAgICAgICAgTXVsdGlSZWN0QXJlYS5MaW5lQ2FzaCByZWN0ID0gbmV3IE11bHRpUmVjdEFyZWEuTGluZUNhc2goZWRnZXNDb3VudCk7Ci0gICAgICAgIHJlY3Quc2V0TGluZSh5KTsKLQotICAgICAgICB3aGlsZSh5IDw9IG5leHRZKSB7Ci0KLSAgICAgICAgICAgIGNyb3NzQ291bnQgPSAwOwotCi0gICAgICAgICAgICBpZiAoeSA9PSBuZXh0WSkgewotCi0gICAgICAgICAgICAgICAgaW50IGkgPSBhY3RpdmVDb3VudDsKLSAgICAgICAgICAgICAgICB3aGlsZShpID4gMCkgewotICAgICAgICAgICAgICAgICAgICBpLS07Ci0gICAgICAgICAgICAgICAgICAgIGlmIChhY3RpdmVZRW5kW2ldID09IHkpIHsKLQotICAgICAgICAgICAgICAgICAgICAgICAgYWN0aXZlQ291bnQtLTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGludCBsZW5ndGggPSBhY3RpdmVDb3VudCAtIGk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAobGVuZ3RoICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgcG9zID0gaSArIDE7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShhY3RpdmVYLCBwb3MsIGFjdGl2ZVgsIGksIGxlbmd0aCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShhY3RpdmVZRW5kLCBwb3MsIGFjdGl2ZVlFbmQsIGksIGxlbmd0aCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShhY3RpdmVYU3RlcCwgcG9zLCBhY3RpdmVYU3RlcCwgaSwgbGVuZ3RoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uYXJyYXljb3B5KGFjdGl2ZURZLCBwb3MsIGFjdGl2ZURZLCBpLCBsZW5ndGgpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN5c3RlbS5hcnJheWNvcHkoYWN0aXZlRXh0LCBwb3MsIGFjdGl2ZUV4dCwgaSwgbGVuZ3RoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIG5leHRZID0gZmluZEFjdGl2ZUVkZ2VzKHkpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBHZXQgWCBjcm9zc2luZ3MKLSAgICAgICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBhY3RpdmVDb3VudDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgY3Jvc3NYW2Nyb3NzQ291bnRdID0gKGludClNYXRoLmNlaWwoYWN0aXZlWFtpXSk7Ci0gICAgICAgICAgICAgICAgY3Jvc3NEWVtjcm9zc0NvdW50XSA9IGFjdGl2ZURZW2ldOwotICAgICAgICAgICAgICAgIGNyb3NzQ291bnQrKzsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKGNyb3NzQ291bnQgPT0gMCkgewotICAgICAgICAgICAgICAgIHJlY3Quc2tpcExpbmUoKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgLy8gU29ydCBYIGNyb3NzaW5ncwotICAgICAgICAgICAgICAgIHNvcnQoY3Jvc3NYLCBjcm9zc0RZLCBjcm9zc0NvdW50KTsKLSAgICAgICAgICAgICAgICBmaWxsZXIuYWRkKHJlY3QsIGNyb3NzWCwgY3Jvc3NEWSwgY3Jvc3NDb3VudCwgeSk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGZvcihpbnQgaSA9IDA7IGkgPCBhY3RpdmVDb3VudDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgYWN0aXZlWFtpXSArPSBhY3RpdmVYU3RlcFtpXTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgeSsrOwotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIHJlY3Q7Ci0gICAgfQotCi19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvSmF2YVRleHRSZW5kZXJlci5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvcmVuZGVyL0phdmFUZXh0UmVuZGVyZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzIyYmE1Ny4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvSmF2YVRleHRSZW5kZXJlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjYzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWx5YSBTLiBPa29taW4KLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLnJlbmRlcjsKLQotaW1wb3J0IGphdmEuYXd0Lio7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuKjsKLQotCi1pbXBvcnQgamF2YS5hd3QuZm9udC5HbHlwaE1ldHJpY3M7Ci1pbXBvcnQgamF2YS5hd3QuZm9udC5HbHlwaFZlY3RvcjsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLlBvaW50MkQ7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLlRleHRSZW5kZXJlcjsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuQ29tbW9uR2x5cGhWZWN0b3I7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5mb250LkZvbnRQZWVySW1wbDsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmZvbnQuR2x5cGg7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZS5CdWZmZXJlZEltYWdlR3JhcGhpY3MyRDsKLQotcHVibGljIGNsYXNzIEphdmFUZXh0UmVuZGVyZXIgZXh0ZW5kcyBUZXh0UmVuZGVyZXIgewotCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBKYXZhVGV4dFJlbmRlcmVyIGluc3QgPSBuZXcgSmF2YVRleHRSZW5kZXJlcigpOwotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd0dseXBoVmVjdG9yKEdyYXBoaWNzMkQgZywgR2x5cGhWZWN0b3IgZ2x5cGhWZWN0b3IsCi0gICAgICAgICAgICBmbG9hdCB4LCBmbG9hdCB5KSB7Ci0KLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIGF0ID0gZy5nZXRUcmFuc2Zvcm0oKTsKLSAgICAgICAgUmVjdGFuZ2xlIGMgPSBnLmdldENsaXBCb3VuZHMoKTsKLSAgICAgICAgaWYgKGF0ICE9IG51bGwpewotICAgICAgICAgICAgaW50IGF0VHlwZSA9IGF0LmdldFR5cGUoKTsKLSAgICAgICAgICAgIGlmIChhdFR5cGUgPT0gQWZmaW5lVHJhbnNmb3JtLlRZUEVfVFJBTlNMQVRJT04pIHsKLSAgICAgICAgICAgICAgICBjLnRyYW5zbGF0ZSgoaW50KU1hdGgucm91bmQoYXQuZ2V0VHJhbnNsYXRlWCgpKSwgKGludClNYXRoLnJvdW5kKGF0LmdldFRyYW5zbGF0ZVkoKSkpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgV3JpdGFibGVSYXN0ZXIgd3IgPSAoKEJ1ZmZlcmVkSW1hZ2VHcmFwaGljczJEKWcpLmdldFdyaXRhYmxlUmFzdGVyKCk7Ci0gICAgICAgIENvbG9yTW9kZWwgY20gPSAoKEJ1ZmZlcmVkSW1hZ2VHcmFwaGljczJEKWcpLmdldENvbG9yTW9kZWwoKTsKLQotICAgICAgICBSZWN0YW5nbGUgckJvdW5kcyA9IHdyLmdldEJvdW5kcygpOwotCi0gICAgICAgIE9iamVjdCBjb2xvciA9IGNtLmdldERhdGFFbGVtZW50cyhnLmdldENvbG9yKCkuZ2V0UkdCKCksIG51bGwpOwotCi0gICAgICAgIGRyYXdDbGlwR2x5cGhWZWN0b3Iod3IsIGNvbG9yLCBnbHlwaFZlY3RvciwgKGludClNYXRoLnJvdW5kKHggKyBhdC5nZXRUcmFuc2xhdGVYKCkpLCAoaW50KU1hdGgucm91bmQoeSArIGF0LmdldFRyYW5zbGF0ZVkoKSksCi0gICAgICAgIE1hdGgubWF4KGMueCxyQm91bmRzLngpLAotICAgICAgICBNYXRoLm1heChjLnksckJvdW5kcy55KSwKLSAgICAgICAgTWF0aC5taW4oKGludClNYXRoLnJvdW5kKGMuZ2V0TWF4WCgpKSwgKGludClNYXRoLnJvdW5kKHJCb3VuZHMuZ2V0TWF4WCgpKSksCi0gICAgICAgIE1hdGgubWluKChpbnQpTWF0aC5yb3VuZChjLmdldE1heFkoKSksIChpbnQpTWF0aC5yb3VuZChyQm91bmRzLmdldE1heFkoKSkpKTsKLQotICAgIH0KLQotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJkZXByZWNhdGlvbiIpCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZHJhd1N0cmluZyhHcmFwaGljczJEIGcsIFN0cmluZyBzdHIsIGZsb2F0IHgsIGZsb2F0IHkpIHsKLSAgICAgICAgQWZmaW5lVHJhbnNmb3JtIGF0ID0gZy5nZXRUcmFuc2Zvcm0oKTsKLSAgICAgICAgUmVjdGFuZ2xlIGMgPSBnLmdldENsaXBCb3VuZHMoKTsKLSAgICAgICAgaWYgKGF0ICE9IG51bGwpewotICAgICAgICAgICAgaW50IGF0VHlwZSA9IGF0LmdldFR5cGUoKTsKLSAgICAgICAgICAgIGlmIChhdFR5cGUgPT0gQWZmaW5lVHJhbnNmb3JtLlRZUEVfVFJBTlNMQVRJT04pIHsKLSAgICAgICAgICAgICAgICBjLnRyYW5zbGF0ZSgoaW50KU1hdGgucm91bmQoYXQuZ2V0VHJhbnNsYXRlWCgpKSwgKGludClNYXRoLnJvdW5kKGF0LmdldFRyYW5zbGF0ZVkoKSkpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIFdyaXRhYmxlUmFzdGVyIHdyID0gKChCdWZmZXJlZEltYWdlR3JhcGhpY3MyRClnKS5nZXRXcml0YWJsZVJhc3RlcigpOwotICAgICAgICBDb2xvck1vZGVsIGNtID0gKChCdWZmZXJlZEltYWdlR3JhcGhpY3MyRClnKS5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIFJlY3RhbmdsZSByQm91bmRzID0gd3IuZ2V0Qm91bmRzKCk7Ci0KLSAgICAgICAgT2JqZWN0IGNvbG9yID0gY20uZ2V0RGF0YUVsZW1lbnRzKGcuZ2V0Q29sb3IoKS5nZXRSR0IoKSwgbnVsbCk7Ci0KLSAgICAgICAgZHJhd0NsaXBTdHJpbmcod3IsIGNvbG9yLCBzdHIsIChGb250UGVlckltcGwpIChnLmdldEZvbnQoKS5nZXRQZWVyKCkpLAotICAgICAgICAgICAgICAgIChpbnQpTWF0aC5yb3VuZCh4ICsgYXQuZ2V0VHJhbnNsYXRlWCgpKSwgKGludClNYXRoLnJvdW5kKHkgKyBhdC5nZXRUcmFuc2xhdGVZKCkpLAotICAgICAgICAgICAgICAgIE1hdGgubWF4KGMueCxyQm91bmRzLngpLAotICAgICAgICAgICAgICAgIE1hdGgubWF4KGMueSxyQm91bmRzLnkpLAotICAgICAgICAgICAgICAgIE1hdGgubWluKChpbnQpTWF0aC5yb3VuZChjLmdldE1heFgoKSksIChpbnQpTWF0aC5yb3VuZChyQm91bmRzLmdldE1heFgoKSkpLAotICAgICAgICAgICAgICAgIE1hdGgubWluKChpbnQpTWF0aC5yb3VuZChjLmdldE1heFkoKSksIChpbnQpTWF0aC5yb3VuZChyQm91bmRzLmdldE1heFkoKSkpKTsKLQotICAgIH0KLQotICAgIC8qKgotICAgICAqIAotICAgICAqIERyYXdzIHN0cmluZyBvbiBzcGVjaWZpZWQgcmFzdGVyIGF0IGRlc2lyZWQgcG9zaXRpb24uCi0gICAgICogIAotICAgICAqIEBwYXJhbSByYXN0ZXIgc3BlY2lmaWVkIFdyaXRhYmxlUmFzdGVyIHRvIGRyYXcgYXQKLSAgICAgKiBAcGFyYW0gY29sb3IgY29sb3Igb2YgdGhlIHRleHQKLSAgICAgKiBAcGFyYW0gZ2x5cGhWZWN0b3IgR2x5cGhWZWN0b3Igb2JqZWN0IHRvIGRyYXcKLSAgICAgKiBAcGFyYW0geCBzdGFydCBYIHBvc2l0aW9uIHRvIGRyYXcKLSAgICAgKiBAcGFyYW0geSBzdGFydCBZIHBvc2l0aW9uIHRvIGRyYXcKLSAgICAgKiBAcGFyYW0gY01pblggbWluaW11bSB4IG9mIHRoZSByYXN0ZXIgYXJlYSB0byBkcmF3Ci0gICAgICogQHBhcmFtIGNNaW5ZIG1pbmltdW0geSBvZiB0aGUgcmFzdGVyIGFyZWEgdG8gZHJhdwotICAgICAqIEBwYXJhbSBjTWF4WCBtYXhpbXVtIHggb2YgdGhlIHJhc3RlciBhcmVhIHRvIGRyYXcKLSAgICAgKiBAcGFyYW0gY01heFkgbWF4aW11bSB5IG9mIHRoZSByYXN0ZXIgYXJlYSB0byBkcmF3Ci0gICAgICovCi0gICAgcHVibGljIHZvaWQgZHJhd0NsaXBHbHlwaFZlY3RvcihXcml0YWJsZVJhc3RlciByYXN0ZXIsIE9iamVjdCBjb2xvciwKLSAgICAgICAgICAgIEdseXBoVmVjdG9yIGdseXBoVmVjdG9yLCBpbnQgeCwgaW50IHksCi0gICAgICAgICAgICBpbnQgY01pblgsIGludCBjTWluWSwgaW50IGNNYXhYLCBpbnQgY01heFkpIHsKLSAgICAgICAgLy8gVE9ETzogaW1wbGVtZW50IGNvbXBsZXggY2xpcHBpbmcKLQotICAgICAgICBpbnQgeFNyY1N1cmYsIHlTcmNTdXJmOyAvLyBTdGFydCBwb2ludCBpbiBTdHJpbmcgcmVjdGFuZ2xlCi0gICAgICAgIGludCB4RHN0U3VyZiwgeURzdFN1cmY7IC8vIFN0YXJ0IHBvaW50IGluIFN1cmZhY2UgcmVjdGFuZ2xlCi0gICAgICAgIGludCBjbFdpZHRoLCBjbEhlaWdodDsKLQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGdseXBoVmVjdG9yLmdldE51bUdseXBocygpOyBpKyspIHsKLSAgICAgICAgICAgIEdseXBoIGdsID0gKChDb21tb25HbHlwaFZlY3RvcikgZ2x5cGhWZWN0b3IpLnZlY3RvcltpXTsKLQotICAgICAgICAgICAgaWYgKGdsLmdldFBvaW50V2lkdGgoKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGJ5dGVbXSBkYXRhID0gZ2wuZ2V0Qml0bWFwKCk7Ci0gICAgICAgICAgICBpZiAoZGF0YSAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgUG9pbnQyRCBwb3MgPSBnbHlwaFZlY3Rvci5nZXRHbHlwaFBvc2l0aW9uKGkpOwotCi0gICAgICAgICAgICAgICAgeFNyY1N1cmYgPSAwOy8vZ2wuYm1wX2xlZnQ7Ci0gICAgICAgICAgICAgICAgeVNyY1N1cmYgPSAwOy8vZ2wuYm1wX3Jvd3MgLSBnbC5ibXBfdG9wOwotCi0gICAgICAgICAgICAgICAgeERzdFN1cmYgPSB4ICsgKGludClwb3MuZ2V0WCgpICsgKGludCkgZ2wuZ2V0R2x5cGhQb2ludE1ldHJpY3MoKS5nZXRMU0IoKTsvLyArIGdsLmJtcF9sZWZ0OwotICAgICAgICAgICAgICAgIHlEc3RTdXJmID0geSAtIGdsLmJtcF90b3AvKmdldFBvaW50SGVpZ2h0KCkqLyAgKyAoaW50KSBwb3MuZ2V0WSgpOy8vIC0gKGdsLmJtcF9yb3dzLWdsLmJtcF90b3ApOwotCi0gICAgICAgICAgICAgICAgaW50IHRleHRXaWR0aCA9IGdsLmJtcF93aWR0aDsKLSAgICAgICAgICAgICAgICBpbnQgdGV4dEhlaWdodCA9IGdsLmdldFBvaW50SGVpZ2h0KCk7Ci0KLSAgICAgICAgICAgICAgICAvLyBpZiBSZWdpb25zIGRvbid0IGludGVyc2VjdAotICAgICAgICAgICAgICAgIGlmICgoeERzdFN1cmYgPiBjTWF4WCkgfHwgKHlEc3RTdXJmID4gY01heFkpIHx8ICh4RHN0U3VyZiArIHRleHRXaWR0aCA8IGNNaW5YKQotICAgICAgICAgICAgICAgICAgICAgICAgfHwgKHlEc3RTdXJmICsgdGV4dEhlaWdodCA8IGNNaW5ZKSkgewotICAgICAgICAgICAgICAgICAgICAvLyBOb3RoaW5nIHRvIGRvCi0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHhEc3RTdXJmID49IGNNaW5YKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjbFdpZHRoID0gTWF0aC5taW4odGV4dFdpZHRoLCBjTWF4WCAtIHhEc3RTdXJmKTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHhTcmNTdXJmICs9IGNNaW5YIC0geERzdFN1cmY7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjbFdpZHRoID0gTWF0aC5taW4oY01heFggLSBjTWluWCwgdGV4dFdpZHRoIC0gKGNNaW5YIC0geERzdFN1cmYpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHhEc3RTdXJmID0gY01pblg7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgaWYgKHlEc3RTdXJmID49IGNNaW5ZKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjbEhlaWdodCA9IE1hdGgubWluKHRleHRIZWlnaHQsIGNNYXhZIC0geURzdFN1cmYpOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgeVNyY1N1cmYgKz0gY01pblkgLSB5RHN0U3VyZjsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNsSGVpZ2h0ID0gTWF0aC5taW4oY01heFkgLSBjTWluWSwgdGV4dEhlaWdodCAtIChjTWluWSAtIHlEc3RTdXJmKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB5RHN0U3VyZiA9IGNNaW5ZOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIC8vICAgICBEcmF3aW5nIG9uIHRoZSBSYXN0ZXIKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaD0wOyBoPGNsSGVpZ2h0OyBoKyspewotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgdz0wOyB3IDwgY2xXaWR0aCA7IHcrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJ5dGUgY3VyckJ5dGUgPSBkYXRhWyh5U3JjU3VyZiArIGgpKmdsLmJtcF9waXRjaCArICh4U3JjU3VyZit3KS84XTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBib29sZWFuIGVtcHR5Qnl0ZSA9ICgoY3VyckJ5dGUgJiAoMSA8PCAoNyAtICgoeFNyY1N1cmYrdykgJSA4KSkpKSAhPSAwKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoZW1wdHlCeXRlKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhc3Rlci5zZXREYXRhRWxlbWVudHMoeERzdFN1cmYrdywgeURzdFN1cmYraCwgY29sb3IpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIE5vdGhpbmcgdG8gZG8KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogRHJhd3Mgc3RyaW5nIG9uIHNwZWNpZmllZCByYXN0ZXIgYXQgZGVzaXJlZCBwb3NpdGlvbi4KLSAgICAgKiAgCi0gICAgICogQHBhcmFtIHJhc3RlciBzcGVjaWZpZWQgV3JpdGFibGVSYXN0ZXIgdG8gZHJhdyBhdAotICAgICAqIEBwYXJhbSBjb2xvciBjb2xvciBvZiB0aGUgdGV4dAotICAgICAqIEBwYXJhbSBzdHIgdGV4dCB0byBkcmF3Ci0gICAgICogQHBhcmFtIGZvbnQgZm9udCBwZWVyIHRvIHVzZSBmb3IgZHJhd2luZyB0ZXh0Ci0gICAgICogQHBhcmFtIHggc3RhcnQgWCBwb3NpdGlvbiB0byBkcmF3Ci0gICAgICogQHBhcmFtIHkgc3RhcnQgWSBwb3NpdGlvbiB0byBkcmF3Ci0gICAgICogQHBhcmFtIGNNaW5YIG1pbmltdW0geCBvZiB0aGUgcmFzdGVyIGFyZWEgdG8gZHJhdwotICAgICAqIEBwYXJhbSBjTWluWSBtaW5pbXVtIHkgb2YgdGhlIHJhc3RlciBhcmVhIHRvIGRyYXcKLSAgICAgKiBAcGFyYW0gY01heFggbWF4aW11bSB4IG9mIHRoZSByYXN0ZXIgYXJlYSB0byBkcmF3Ci0gICAgICogQHBhcmFtIGNNYXhZIG1heGltdW0geSBvZiB0aGUgcmFzdGVyIGFyZWEgdG8gZHJhdwotICAgICAqLyAgICAKLSAgICBwdWJsaWMgdm9pZCBkcmF3Q2xpcFN0cmluZyhXcml0YWJsZVJhc3RlciByYXN0ZXIsIE9iamVjdCBjb2xvciwgU3RyaW5nIHN0ciwKLSAgICAgICAgICAgIEZvbnRQZWVySW1wbCBmb250LCBpbnQgeCwgaW50IHksIGludCBjTWluWCwgaW50IGNNaW5ZLCBpbnQgY01heFgsCi0gICAgICAgICAgICBpbnQgY01heFkpIHsKLSAgICAgICAgLy8gVE9ETzogaW1wbGVtZW50IGNvbXBsZXggY2xpcHBpbmcKLQotICAgICAgICBpbnQgeFNyY1N1cmYsIHlTcmNTdXJmOyAvLyBTdGFydCBwb2ludCBpbiBTdHJpbmcgcmVjdGFuZ2xlCi0gICAgICAgIGludCB4RHN0U3VyZiwgeURzdFN1cmY7IC8vIFN0YXJ0IHBvaW50IGluIFN1cmZhY2UgcmVjdGFuZ2xlCi0gICAgICAgIGludCBjbFdpZHRoLCBjbEhlaWdodDsKLQotICAgICAgICBjaGFyW10gY2hhcnMgPSBzdHIudG9DaGFyQXJyYXkoKTsKLQotICAgICAgICBpbnQgeEJhc2VMaW5lID0geDsKLSAgICAgICAgaW50IHlCYXNlTGluZSA9IHk7Ci0KLSAgICAgICAgZm9yIChjaGFyIGVsZW1lbnQgOiBjaGFycykgewotICAgICAgICAgICAgR2x5cGggZ2wgPSBmb250LmdldEdseXBoKGVsZW1lbnQpOwotICAgICAgICAgICAgR2x5cGhNZXRyaWNzIHBvaW50TWV0cmljcyA9IGdsLmdldEdseXBoUG9pbnRNZXRyaWNzKCk7Ci0gICAgICAgICAgICBpZiAoZ2wuZ2V0V2lkdGgoKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgeEJhc2VMaW5lICs9IHBvaW50TWV0cmljcy5nZXRBZHZhbmNlWCgpOwotICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBieXRlW10gZGF0YSA9IGdsLmdldEJpdG1hcCgpOwotICAgICAgICAgICAgaWYgKGRhdGEgPT0gbnVsbCkgewotICAgICAgICAgICAgICAgIHhCYXNlTGluZSArPSBwb2ludE1ldHJpY3MuZ2V0QWR2YW5jZVgoKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0KLSAgICAgICAgICAgICAgICB4U3JjU3VyZiA9IDA7Ci0gICAgICAgICAgICAgICAgeVNyY1N1cmYgPSAwOwotCi0gICAgICAgICAgICAgICAgeERzdFN1cmYgPSBNYXRoLnJvdW5kKHhCYXNlTGluZSArIGdsLmdldEdseXBoUG9pbnRNZXRyaWNzKCkuZ2V0TFNCKCkpOwotICAgICAgICAgICAgICAgIHlEc3RTdXJmID0geUJhc2VMaW5lIC0gZ2wuYm1wX3RvcDsKLQotICAgICAgICAgICAgICAgIGludCB0ZXh0V2lkdGggPSBnbC5ibXBfd2lkdGg7Ci0gICAgICAgICAgICAgICAgaW50IHRleHRIZWlnaHQgPSBnbC5nZXRQb2ludEhlaWdodCgpOwotCi0gICAgICAgICAgICAgICAgLy8gaWYgUmVnaW9ucyBkb24ndCBpbnRlcnNlY3QKLSAgICAgICAgICAgICAgICBpZiAoKHhEc3RTdXJmID4gY01heFgpIHx8ICh5RHN0U3VyZiA+IGNNYXhZKSB8fCAoeERzdFN1cmYgKyB0ZXh0V2lkdGggPCBjTWluWCkKLSAgICAgICAgICAgICAgICAgICAgICAgIHx8ICh5RHN0U3VyZiArIHRleHRIZWlnaHQgPCBjTWluWSkpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gTm90aGluZyB0byBkbwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmICh4RHN0U3VyZiA+PSBjTWluWCkgewotICAgICAgICAgICAgICAgICAgICAgICAgY2xXaWR0aCA9IE1hdGgubWluKHRleHRXaWR0aCwgY01heFggLSB4RHN0U3VyZik7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB4U3JjU3VyZiArPSBjTWluWCAtIHhEc3RTdXJmOwotICAgICAgICAgICAgICAgICAgICAgICAgY2xXaWR0aCA9IE1hdGgubWluKGNNYXhYIC0gY01pblgsIHRleHRXaWR0aCAtIChjTWluWCAtIHhEc3RTdXJmKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB4RHN0U3VyZiA9IGNNaW5YOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGlmICh5RHN0U3VyZiA+PSBjTWluWSkgewotICAgICAgICAgICAgICAgICAgICAgICAgY2xIZWlnaHQgPSBNYXRoLm1pbih0ZXh0SGVpZ2h0LCBjTWF4WSAtIHlEc3RTdXJmKTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHlTcmNTdXJmICs9IGNNaW5ZIC0geURzdFN1cmY7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjbEhlaWdodCA9IE1hdGgubWluKGNNYXhZIC0gY01pblksIHRleHRIZWlnaHQgLSAoY01pblkgLSB5RHN0U3VyZikpOwotICAgICAgICAgICAgICAgICAgICAgICAgeURzdFN1cmYgPSBjTWluWTsKLSAgICAgICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgICAgIC8vIERyYXdpbmcgb24gdGhlIFJhc3RlcgotICAgICAgICAgICAgICAgICAgICBmb3IgKGludCBoPTA7IGg8Y2xIZWlnaHQ7IGgrKyl7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGludCB3PTA7IHcgPCBjbFdpZHRoIDsgdysrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnl0ZSBjdXJyQnl0ZSA9IGRhdGFbKHlTcmNTdXJmICsgaCkqZ2wuYm1wX3BpdGNoICsgKHhTcmNTdXJmK3cpLzhdOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2xlYW4gZW1wdHlCeXRlID0gKChjdXJyQnl0ZSAmICgxIDw8ICg3IC0gKCh4U3JjU3VyZit3KSAlIDgpKSkpICE9IDApOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChlbXB0eUJ5dGUpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmFzdGVyLnNldERhdGFFbGVtZW50cyh4RHN0U3VyZit3LCB5RHN0U3VyZitoLCBjb2xvcik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gTm90aGluZyB0byBkbwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB4QmFzZUxpbmUgKz0gcG9pbnRNZXRyaWNzLmdldEFkdmFuY2VYKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLX0KXCBObyBuZXdsaW5lIGF0IGVuZCBvZiBmaWxlCmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvTmF0aXZlSW1hZ2VCbGl0dGVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvTmF0aXZlSW1hZ2VCbGl0dGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGIwZWJjOTcuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvcmVuZGVyL05hdGl2ZUltYWdlQmxpdHRlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjE4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqIENyZWF0ZWQgb24gMjYuMTEuMjAwNQotICoKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLnJlbmRlcjsKLQotaW1wb3J0IGphdmEuYXd0LkFscGhhQ29tcG9zaXRlOwotaW1wb3J0IGphdmEuYXd0LkNvbG9yOwotaW1wb3J0IGphdmEuYXd0LkNvbXBvc2l0ZTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5JbWFnZVN1cmZhY2U7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5NdWx0aVJlY3RBcmVhOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuU3VyZmFjZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLlhPUkNvbXBvc2l0ZTsKLQotLyoqCi0gKiBUaGlzIGtpbmQgb2YgYmxpdHRlcnMgaXMgaW50ZW5kZWQgZm9yIGRyYXdpbmcgb25lIGltYWdlIG9uIHRoZSBidWZmZXJlZAotICogb3Igdm9sYXRpbGUgaW1hZ2UuIEZvciB0aGUgbW9tZW50IHdlIGNhbiBibGl0IG5hdGl2ZWx5IEJ1ZmZlcmVkIEltYWdlcyB3aGljaCAKLSAqIGhhdmUgc1JHQiwgTGluZWFyX1JHQiwgTGluZWFyX0dyYXkgQ29sb3IgU3BhY2UgYW5kIHR5cGUgZGlmZmVyZW50IAotICogZnJvbSBCdWZmZXJlZEltYWdlLlRZUEVfQ1VTVE9NLCBWb2xhdGlsZSBJbWFnZXMgYW5kIEltYWdlcyB3aGljaCByZWNlaXZlZCAKLSAqIHVzaW5nIFRvb2xraXQgYW5kIENvbXBvbmVudCBjbGFzc2VzLgotICovCi1wdWJsaWMgY2xhc3MgTmF0aXZlSW1hZ2VCbGl0dGVyIGltcGxlbWVudHMgQmxpdHRlciB7Ci0KLQotICAgIGZpbmFsIHN0YXRpYyBOYXRpdmVJbWFnZUJsaXR0ZXIgaW5zdCA9IG5ldyBOYXRpdmVJbWFnZUJsaXR0ZXIoKTsKLQotICAgIHB1YmxpYyBzdGF0aWMgTmF0aXZlSW1hZ2VCbGl0dGVyIGdldEluc3RhbmNlKCl7Ci0gICAgICAgIHJldHVybiBpbnN0OwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBBZmZpbmVUcmFuc2Zvcm0gc3lzeGZvcm0sCi0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0sIENvbXBvc2l0ZSBjb21wLCBDb2xvciBiZ2NvbG9yLAotICAgICAgICAgICAgTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0KLSAgICAgICAgaWYoIXNyY1N1cmYuaXNOYXRpdmVEcmF3YWJsZSgpKXsKLSAgICAgICAgICAgIEphdmFCbGl0dGVyLmluc3QuYmxpdChzcmNYLCBzcmNZLCBzcmNTdXJmLCBkc3RYLCBkc3RZLCBkc3RTdXJmLCB3aWR0aCwgaGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICBzeXN4Zm9ybSwgeGZvcm0sIGNvbXAsIGJnY29sb3IsIGNsaXApOwotICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgIGlmKHhmb3JtID09IG51bGwpewotICAgICAgICAgICAgICAgIGJsaXQoc3JjWCwgc3JjWSwgc3JjU3VyZiwgZHN0WCwgZHN0WSwgZHN0U3VyZiwgd2lkdGgsIGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgICAgIHN5c3hmb3JtLCBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgIGRvdWJsZSBzY2FsZVggPSB4Zm9ybS5nZXRTY2FsZVgoKTsKLSAgICAgICAgICAgICAgICBkb3VibGUgc2NhbGVZID0geGZvcm0uZ2V0U2NhbGVZKCk7Ci0gICAgICAgICAgICAgICAgZG91YmxlIHNjYWxlZFggPSBkc3RYIC8gc2NhbGVYOwotICAgICAgICAgICAgICAgIGRvdWJsZSBzY2FsZWRZID0gZHN0WSAvIHNjYWxlWTsKLSAgICAgICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0gYXQgPSBuZXcgQWZmaW5lVHJhbnNmb3JtKCk7Ci0gICAgICAgICAgICAgICAgYXQuc2V0VG9UcmFuc2xhdGlvbihzY2FsZWRYLCBzY2FsZWRZKTsKLSAgICAgICAgICAgICAgICB4Zm9ybS5jb25jYXRlbmF0ZShhdCk7Ci0gICAgICAgICAgICAgICAgc3lzeGZvcm0uY29uY2F0ZW5hdGUoeGZvcm0pOwotICAgICAgICAgICAgICAgIGJsaXQoc3JjWCwgc3JjWSwgc3JjU3VyZiwgMCwgMCwgZHN0U3VyZiwgd2lkdGgsIGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgICAgIHN5c3hmb3JtLCBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBBZmZpbmVUcmFuc2Zvcm0gc3lzeGZvcm0sCi0gICAgICAgICAgICBDb21wb3NpdGUgY29tcCwgQ29sb3IgYmdjb2xvciwgTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0KLSAgICAgICAgaWYoIXNyY1N1cmYuaXNOYXRpdmVEcmF3YWJsZSgpKXsKLSAgICAgICAgICAgIEphdmFCbGl0dGVyLmluc3QuYmxpdChzcmNYLCBzcmNZLCBzcmNTdXJmLCBkc3RYLCBkc3RZLCBkc3RTdXJmLCB3aWR0aCwgaGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICBzeXN4Zm9ybSwgY29tcCwgYmdjb2xvciwgY2xpcCk7Ci0gICAgICAgIH1lbHNlewotICAgICAgICAgICAgaW50IHR5cGUgPSBzeXN4Zm9ybS5nZXRUeXBlKCk7Ci0gICAgICAgICAgICBzd2l0Y2godHlwZSl7Ci0gICAgICAgICAgICAgICAgY2FzZSBBZmZpbmVUcmFuc2Zvcm0uVFlQRV9UUkFOU0xBVElPTjoKLSAgICAgICAgICAgICAgICAgICAgZHN0WCArPSBzeXN4Zm9ybS5nZXRUcmFuc2xhdGVYKCk7Ci0gICAgICAgICAgICAgICAgICAgIGRzdFkgKz0gc3lzeGZvcm0uZ2V0VHJhbnNsYXRlWSgpOwotICAgICAgICAgICAgICAgIGNhc2UgQWZmaW5lVHJhbnNmb3JtLlRZUEVfSURFTlRJVFk6Ci0gICAgICAgICAgICAgICAgICAgIGJsaXQoc3JjWCwgc3JjWSwgc3JjU3VyZiwgZHN0WCwgZHN0WSwgZHN0U3VyZiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgICAgICAgICAgLy8gVE9ETyBOZWVkIHRvIHJlYWxpemUgQWZmaW5lIFRyYW5zZm9ybWF0aW9uCi0gICAgICAgICAgICAgICAgICAgIGlmKHNyY1N1cmYgaW5zdGFuY2VvZiBJbWFnZVN1cmZhY2UpewotICAgICAgICAgICAgICAgICAgICAgICAgSmF2YUJsaXR0ZXIuaW5zdC5ibGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIGRzdFgsIGRzdFksIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3RTdXJmLCB3aWR0aCwgaGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzeXN4Zm9ybSwgY29tcCwgYmdjb2xvciwgY2xpcCk7Ci0gICAgICAgICAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgICAgICAgICAgaW50IHcgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpbnQgaCA9IHNyY1N1cmYuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBCdWZmZXJlZEltYWdlIHRtcCA9IG5ldyBCdWZmZXJlZEltYWdlKHcsIGgsIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX1JHQik7Ci0gICAgICAgICAgICAgICAgICAgICAgICBTdXJmYWNlIHRtcFN1cmYgPSBTdXJmYWNlLmdldEltYWdlU3VyZmFjZSh0bXApOwotICAgICAgICAgICAgICAgICAgICAgICAgYmxpdCgwLCAwLCBzcmNTdXJmLCAwLCAwLCB0bXBTdXJmLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB3LCBoLCBBbHBoYUNvbXBvc2l0ZS5TcmNPdmVyLCBudWxsLCBudWxsKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIEphdmFCbGl0dGVyLmluc3QuYmxpdChzcmNYLCBzcmNZLCB0bXBTdXJmLCBkc3RYLCBkc3RZLCAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0U3VyZiwgd2lkdGgsIGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3lzeGZvcm0sIGNvbXAsIGJnY29sb3IsIGNsaXApOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBibGl0KGludCBzcmNYLCBpbnQgc3JjWSwgU3VyZmFjZSBzcmNTdXJmLCBpbnQgZHN0WCwgaW50IGRzdFksCi0gICAgICAgICAgICBTdXJmYWNlIGRzdFN1cmYsIGludCB3aWR0aCwgaW50IGhlaWdodCwgQ29tcG9zaXRlIGNvbXAsCi0gICAgICAgICAgICBDb2xvciBiZ2NvbG9yLCBNdWx0aVJlY3RBcmVhIGNsaXApIHsKLQotICAgICAgICBpZighc3JjU3VyZi5pc05hdGl2ZURyYXdhYmxlKCkpewotICAgICAgICAgICAgSmF2YUJsaXR0ZXIuaW5zdC5ibGl0KHNyY1gsIHNyY1ksIHNyY1N1cmYsIGRzdFgsIGRzdFksIGRzdFN1cmYsIHdpZHRoLCBoZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgIGNvbXAsIGJnY29sb3IsIGNsaXApOwotICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgIGxvbmcgZHN0U3VyZlN0cnVjdCA9IGRzdFN1cmYuZ2V0U3VyZmFjZURhdGFQdHIoKTsKLSAgICAgICAgICAgIE9iamVjdCBkc3REYXRhID0gZHN0U3VyZi5nZXREYXRhKCk7Ci0gICAgICAgICAgICBpbnQgY2xpcFJlY3RzW107Ci0gICAgICAgICAgICBpZihjbGlwICE9IG51bGwpewotICAgICAgICAgICAgICAgIGNsaXBSZWN0cyA9IGNsaXAucmVjdDsKLSAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgIGNsaXBSZWN0cyA9IG5ldyBpbnRbXXs1LCAwLCAwLCBkc3RTdXJmLmdldFdpZHRoKCksCi0gICAgICAgICAgICAgICAgICAgICAgICBkc3RTdXJmLmdldEhlaWdodCgpfTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYoIShzcmNTdXJmIGluc3RhbmNlb2YgSW1hZ2VTdXJmYWNlKSl7Ci0gICAgICAgICAgICAgICAgc3JjU3VyZiA9IHNyY1N1cmYuZ2V0SW1hZ2VTdXJmYWNlKCk7Ci0gICAgICAgICAgICAgICAgaWYoYmdjb2xvciAhPSBudWxsKXsKLSAgICAgICAgICAgICAgICAgICAgYmdjb2xvciA9IG51bGw7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBsb25nIHNyY1N1cmZTdHJ1Y3QgPSBzcmNTdXJmLmdldFN1cmZhY2VEYXRhUHRyKCk7Ci0gICAgICAgICAgICBPYmplY3Qgc3JjRGF0YSA9IHNyY1N1cmYuZ2V0RGF0YSgpOwotICAgICAgICAgICAgaWYoY29tcCBpbnN0YW5jZW9mIEFscGhhQ29tcG9zaXRlKXsKLSAgICAgICAgICAgICAgICBBbHBoYUNvbXBvc2l0ZSBhYyA9IChBbHBoYUNvbXBvc2l0ZSkgY29tcDsKLSAgICAgICAgICAgICAgICBpbnQgY29tcFR5cGUgPSBhYy5nZXRSdWxlKCk7Ci0gICAgICAgICAgICAgICAgZmxvYXQgYWxwaGEgPSBhYy5nZXRBbHBoYSgpOwotICAgICAgICAgICAgICAgIGlmKGJnY29sb3IgIT0gbnVsbCl7Ci0gICAgICAgICAgICAgICAgICAgIGJsdEJHKHNyY1gsIHNyY1ksIHNyY1N1cmZTdHJ1Y3QsIHNyY0RhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0WCwgZHN0WSwgZHN0U3VyZlN0cnVjdCwgZHN0RGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCBiZ2NvbG9yLmdldFJHQigpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbXBUeXBlLCBhbHBoYSwgY2xpcFJlY3RzLCBzcmNTdXJmLmludmFsaWRhdGVkKCkpOwotICAgICAgICAgICAgICAgICAgICBkc3RTdXJmLmludmFsaWRhdGUoKTsKLSAgICAgICAgICAgICAgICAgICAgc3JjU3VyZi52YWxpZGF0ZSgpOwotICAgICAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgICAgICBibHQoc3JjWCwgc3JjWSwgc3JjU3VyZlN0cnVjdCwgc3JjRGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3RYLCBkc3RZLCBkc3RTdXJmU3RydWN0LCBkc3REYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHdpZHRoLCBoZWlnaHQsIGNvbXBUeXBlLCBhbHBoYSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGlwUmVjdHMsIHNyY1N1cmYuaW52YWxpZGF0ZWQoKSk7Ci0gICAgICAgICAgICAgICAgICAgIGRzdFN1cmYuaW52YWxpZGF0ZSgpOwotICAgICAgICAgICAgICAgICAgICBzcmNTdXJmLnZhbGlkYXRlKCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfWVsc2UgaWYoY29tcCBpbnN0YW5jZW9mIFhPUkNvbXBvc2l0ZSl7Ci0gICAgICAgICAgICAgICAgWE9SQ29tcG9zaXRlIHhjb21wID0gKFhPUkNvbXBvc2l0ZSkgY29tcDsKLSAgICAgICAgICAgICAgICB4b3Ioc3JjWCwgc3JjWSwgc3JjU3VyZlN0cnVjdCwgc3JjRGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdFgsIGRzdFksIGRzdFN1cmZTdHJ1Y3QsIGRzdERhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCB4Y29tcC5nZXRYT1JDb2xvcigpLmdldFJHQigpLAotICAgICAgICAgICAgICAgICAgICAgICAgY2xpcFJlY3RzLCBzcmNTdXJmLmludmFsaWRhdGVkKCkpOwotICAgICAgICAgICAgICAgIGRzdFN1cmYuaW52YWxpZGF0ZSgpOwotICAgICAgICAgICAgICAgIHNyY1N1cmYudmFsaWRhdGUoKTsKLSAgICAgICAgICAgIH1lbHNlewotICAgICAgICAgICAgICAgIGlmKHNyY1N1cmYgaW5zdGFuY2VvZiBJbWFnZVN1cmZhY2UpewotICAgICAgICAgICAgICAgICAgICBKYXZhQmxpdHRlci5pbnN0LmJsaXQoc3JjWCwgc3JjWSwgc3JjU3VyZiwgZHN0WCwgZHN0WSwgCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZHN0U3VyZiwgd2lkdGgsIGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb21wLCBiZ2NvbG9yLCBjbGlwKTsKLSAgICAgICAgICAgICAgICB9ZWxzZXsKLSAgICAgICAgICAgICAgICAgICAgaW50IHcgPSBzcmNTdXJmLmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgICAgIGludCBoID0gc3JjU3VyZi5nZXRIZWlnaHQoKTsKLSAgICAgICAgICAgICAgICAgICAgQnVmZmVyZWRJbWFnZSB0bXAgPSBuZXcgQnVmZmVyZWRJbWFnZSh3LCBoLCAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBCdWZmZXJlZEltYWdlLlRZUEVfSU5UX1JHQik7Ci0gICAgICAgICAgICAgICAgICAgIFN1cmZhY2UgdG1wU3VyZiA9IFN1cmZhY2UuZ2V0SW1hZ2VTdXJmYWNlKHRtcCk7Ci0gICAgICAgICAgICAgICAgICAgIGxvbmcgdG1wU3VyZlN0cnVjdCA9IHRtcFN1cmYuZ2V0U3VyZmFjZURhdGFQdHIoKTsKLSAgICAgICAgICAgICAgICAgICAgT2JqZWN0IHRtcERhdGEgPSB0bXBTdXJmLmdldERhdGEoKTsKLSAgICAgICAgICAgICAgICAgICAgaW50IHRtcENsaXBbXSA9IG5ldyBpbnRbXXs1LCAwLCAwLCBzcmNTdXJmLmdldFdpZHRoKCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgc3JjU3VyZi5nZXRIZWlnaHQoKX07Ci0gICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICBibHQoMCwgMCwgc3JjU3VyZlN0cnVjdCwgc3JjRGF0YSwgMCwgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB0bXBTdXJmU3RydWN0LCB0bXBEYXRhLCB3LCBoLCAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBBbHBoYUNvbXBvc2l0ZS5TUkNfT1ZFUiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAxLjBmLCB0bXBDbGlwLCBzcmNTdXJmLmludmFsaWRhdGVkKCkpOwotICAgICAgICAgICAgICAgICAgICBzcmNTdXJmLnZhbGlkYXRlKCk7Ci0gICAgICAgICAgICAgICAgICAgIEphdmFCbGl0dGVyLmluc3QuYmxpdChzcmNYLCBzcmNZLCB0bXBTdXJmLCBkc3RYLCBkc3RZLCAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkc3RTdXJmLCB3aWR0aCwgaGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbXAsIGJnY29sb3IsIGNsaXApOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgcHJpdmF0ZSBuYXRpdmUgdm9pZCBibHRCRyhpbnQgc3JjWCwgaW50IHNyY1ksIGxvbmcgc3JzU3VyZkRhdGFQdHIsCi0gICAgICAgICAgICBPYmplY3Qgc3JjRGF0YSwgaW50IGRzdFgsIGludCBkc3RZLCBsb25nIGRzdFN1cmZEYXRhUHRyLAotICAgICAgICAgICAgT2JqZWN0IGRzdERhdGEsIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IGJnY29sb3IsCi0gICAgICAgICAgICBpbnQgY29tcFR5cGUsIGZsb2F0IGFscGhhLCBpbnQgY2xpcFtdLCBib29sZWFuIGludmFsaWRhdGVkKTsKLQotICAgIHByaXZhdGUgbmF0aXZlIHZvaWQgYmx0KGludCBzcmNYLCBpbnQgc3JjWSwgbG9uZyBzcnNTdXJmRGF0YVB0ciwKLSAgICAgICAgICAgIE9iamVjdCBzcmNEYXRhLCBpbnQgZHN0WCwgaW50IGRzdFksIGxvbmcgZHN0U3VyZkRhdGFQdHIsCi0gICAgICAgICAgICBPYmplY3QgZHN0RGF0YSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBpbnQgY29tcFR5cGUsCi0gICAgICAgICAgICBmbG9hdCBhbHBoYSwgaW50IGNsaXBbXSwgYm9vbGVhbiBpbnZhbGlkYXRlZCk7Ci0KLSAgICBwcml2YXRlIG5hdGl2ZSB2b2lkIHhvcihpbnQgc3JjWCwgaW50IHNyY1ksIGxvbmcgc3JzU3VyZkRhdGFQdHIsCi0gICAgICAgICAgICBPYmplY3Qgc3JjRGF0YSwgaW50IGRzdFgsIGludCBkc3RZLCBsb25nIGRzdFN1cmZEYXRhUHRyLAotICAgICAgICAgICAgT2JqZWN0IGRzdERhdGEsIGludCB3aWR0aCwgaW50IGhlaWdodCwgaW50IHhvcmNvbG9yLAotICAgICAgICAgICAgaW50IGNsaXBbXSwgYm9vbGVhbiBpbnZhbGlkYXRlZCk7Ci0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvZ2wvcmVuZGVyL051bGxCbGl0dGVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvTnVsbEJsaXR0ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTAzMmU0ZS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9nbC9yZW5kZXIvTnVsbEJsaXR0ZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDU2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgSWdvciBWLiBTdG9seWFyb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqIENyZWF0ZWQgb24gMDcuMTIuMjAwNQotICoKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLnJlbmRlcjsKLQotaW1wb3J0IGphdmEuYXd0LkNvbG9yOwotaW1wb3J0IGphdmEuYXd0LkNvbXBvc2l0ZTsKLWltcG9ydCBqYXZhLmF3dC5nZW9tLkFmZmluZVRyYW5zZm9ybTsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuTXVsdGlSZWN0QXJlYTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLlN1cmZhY2U7Ci0KLQotcHVibGljIGNsYXNzIE51bGxCbGl0dGVyIGltcGxlbWVudHMgQmxpdHRlciB7Ci0KLSAgICBzdGF0aWMgQmxpdHRlciBpbnN0ID0gbmV3IE51bGxCbGl0dGVyKCk7Ci0gICAgcHVibGljIHN0YXRpYyBCbGl0dGVyIGdldEluc3RhbmNlKCl7Ci0gICAgICAgIHJldHVybiBpbnN0OwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGJsaXQoaW50IHNyY1gsIGludCBzcmNZLCBTdXJmYWNlIHNyY1N1cmYsIGludCBkc3RYLCBpbnQgZHN0WSwKLSAgICAgICAgICAgIFN1cmZhY2UgZHN0U3VyZiwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBBZmZpbmVUcmFuc2Zvcm0gc3lzeGZvcm0sCi0gICAgICAgICAgICBBZmZpbmVUcmFuc2Zvcm0geGZvcm0sIENvbXBvc2l0ZSBjb21wLCBDb2xvciBiZ2NvbG9yLAotICAgICAgICAgICAgTXVsdGlSZWN0QXJlYSBjbGlwKSB7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgYmxpdChpbnQgc3JjWCwgaW50IHNyY1ksIFN1cmZhY2Ugc3JjU3VyZiwgaW50IGRzdFgsIGludCBkc3RZLAotICAgICAgICAgICAgU3VyZmFjZSBkc3RTdXJmLCBpbnQgd2lkdGgsIGludCBoZWlnaHQsIEFmZmluZVRyYW5zZm9ybSBzeXN4Zm9ybSwKLSAgICAgICAgICAgIENvbXBvc2l0ZSBjb21wLCBDb2xvciBiZ2NvbG9yLCBNdWx0aVJlY3RBcmVhIGNsaXApIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBibGl0KGludCBzcmNYLCBpbnQgc3JjWSwgU3VyZmFjZSBzcmNTdXJmLCBpbnQgZHN0WCwgaW50IGRzdFksCi0gICAgICAgICAgICBTdXJmYWNlIGRzdFN1cmYsIGludCB3aWR0aCwgaW50IGhlaWdodCwgQ29tcG9zaXRlIGNvbXAsCi0gICAgICAgICAgICBDb2xvciBiZ2NvbG9yLCBNdWx0aVJlY3RBcmVhIGNsaXApIHsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2ltL0lucHV0TWV0aG9kQ29udGV4dC5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvaW0vSW5wdXRNZXRob2RDb250ZXh0LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQ1ZWQxMWYuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvaW0vSW5wdXRNZXRob2RDb250ZXh0LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NjMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKiAKLSAqIEBhdXRob3IgRG1pdHJ5IEEuIER1cm5ldgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW07Ci0KLS8vPz8/QVdUCi1pbXBvcnQgamF2YS5hd3QuQVdURXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuQ29tcG9uZW50OwotLy9pbXBvcnQgamF2YS5hd3QuS2V5Ym9hcmRGb2N1c01hbmFnZXI7Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotLy9pbXBvcnQgamF2YS5hd3QuV2luZG93OwotaW1wb3J0IGphdmEuYXd0LmV2ZW50LkZvY3VzRXZlbnQ7Ci1pbXBvcnQgamF2YS5hd3QuZXZlbnQuSW5wdXRNZXRob2RFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5LZXlFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5mb250LlRleHRIaXRJbmZvOwotaW1wb3J0IGphdmEuYXd0LmltLklucHV0Q29udGV4dDsKLWltcG9ydCBqYXZhLmF3dC5pbS5JbnB1dE1ldGhvZFJlcXVlc3RzOwotaW1wb3J0IGphdmEuYXd0LmltLnNwaS5JbnB1dE1ldGhvZDsKLWltcG9ydCBqYXZhLmF3dC5pbS5zcGkuSW5wdXRNZXRob2REZXNjcmlwdG9yOwotaW1wb3J0IGphdmEubGFuZy5DaGFyYWN0ZXIuU3Vic2V0OwotaW1wb3J0IGphdmEudGV4dC5BdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3I7Ci1pbXBvcnQgamF2YS50ZXh0LkF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvci5BdHRyaWJ1dGU7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2hNYXA7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2hTZXQ7Ci1pbXBvcnQgamF2YS51dGlsLkl0ZXJhdG9yOwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci1pbXBvcnQgamF2YS51dGlsLk1hcDsKLWltcG9ydCBqYXZhLnV0aWwuU2V0OwotCi0vLz8/P0FXVAotLy9pbXBvcnQgamF2YXguc3dpbmcuSkZyYW1lOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuTmF0aXZlSU07Ci0KLS8qKgotICogSW1wbGVtZW50YXRpb24gb2YgSW5wdXRNZXRob2RDb250ZXh0Ci0gKiBpbnRlcmZhY2UsIGFsc28gcHJvdmlkZXMgYWxsIHVzZWZ1bAotICogZnVuY3Rpb25hbGl0eSBvZiBJbnB1dENvbnRleHQKLSAqIAotICovCi1wdWJsaWMgY2xhc3MgSW5wdXRNZXRob2RDb250ZXh0IGV4dGVuZHMgSW5wdXRDb250ZXh0IGltcGxlbWVudHMKLSAgICAgICAgamF2YS5hd3QuaW0uc3BpLklucHV0TWV0aG9kQ29udGV4dCB7ICAgIAotCi0gICAgLy8/Pz9BV1QKLSAgICBwcml2YXRlIElucHV0TWV0aG9kIGlucHV0TWV0aG9kOyAvLyBjdXJyZW50IElNCi0gICAgcHJpdmF0ZSBDb21wb25lbnQgY2xpZW50OyAvLyBjdXJyZW50ICJhY3RpdmUiIGNsaWVudCBjb21wb25lbnQKLSAgICAvLz8/P0FXVDogcHJpdmF0ZSBDb21wb3NpdGlvbldpbmRvdyBjb21wb3NlV2luZG93OyAvLyBjb21wb3NpdGlvbiBXaW5kb3cgICAgCi0gICAgcHJpdmF0ZSBmaW5hbCBNYXA8SW5wdXRNZXRob2REZXNjcmlwdG9yLCBJbnB1dE1ldGhvZD4gaW1JbnN0YW5jZXM7IC8vIE1hcDxJbnB1dE1ldGhvZERlc2NyaXB0b3IsIElucHV0TWV0aG9kPgotICAgIHByaXZhdGUgZmluYWwgTWFwPExvY2FsZSwgSW5wdXRNZXRob2Q+IGxvY2FsZUlNOyAvLyBNYXA8TG9jYWxlLCBJbnB1dE1ldGhvZD4gbGFzdCB1c2VyLXNlbGVjdGVkIElNIGZvciBsb2NhbGUKLSAgICBwcml2YXRlIGZpbmFsIFNldDxJbnB1dE1ldGhvZD4gbm90aWZ5SU07IC8vIHNldCBvZiBJTXMgdG8gbm90aWZ5IG9mIGNsaWVudCB3aW5kb3cgYm91bmRzIGNoYW5nZXMKLSAgICAKLSAgICAvKioKLSAgICAgKiBhIGZsYWcgaW5kaWNhdGluZyB0aGF0IElNIHNob3VsZCBiZSBub3RpZmllZCBvZiBjbGllbnQgd2luZG93Ci0gICAgICogcG9zaXRpb24vdmlzaWJpbGl0eSBjaGFuZ2VzIGFzIHNvb24gYXMgaXQgaXMgYWN0aXZhdGVkKG5ldyBjbGllbnQKLSAgICAgKiBhcHBlYXJzKQotICAgICAqLyAgICAKLSAgICBwcml2YXRlIGJvb2xlYW4gcGVuZGluZ0NsaWVudE5vdGlmeTsKLSAgICBwcml2YXRlIENvbXBvbmVudCBuZXh0Q29tcDsgLy8gY29tcG9uZW50IHRvIGdhaW4gZm9jdXMgYWZ0ZXIgZW5kQ29tcG9zaXRpb24oKQotICAgIC8vPz8/QVdUOiBwcml2YXRlIGZpbmFsIFNldDxXaW5kb3c+IGltV2luZG93czsgLy8gc2V0IG9mIGFsbCBJTSB3aW5kb3dzIGNyZWF0ZWQgYnkgdGhpcyBpbnN0YW5jZQotICAgIHByaXZhdGUgZmluYWwgTmF0aXZlSU0gbmF0aXZlSU07Ci0gICAgCi0KLSAKLSAgICBwdWJsaWMgSW5wdXRNZXRob2RDb250ZXh0KCkgewotICAgICAgICBub3RpZnlJTSA9IG5ldyBIYXNoU2V0PElucHV0TWV0aG9kPigpOwotLy8/Pz9BV1Q6ICAgICAgICBpbVdpbmRvd3MgPSBuZXcgSGFzaFNldDxXaW5kb3c+KCk7Ci0gICAgICAgIGltSW5zdGFuY2VzID0gbmV3IEhhc2hNYXA8SW5wdXRNZXRob2REZXNjcmlwdG9yLCBJbnB1dE1ldGhvZD4oKTsKLSAgICAgICAgbG9jYWxlSU0gPSBuZXcgSGFzaE1hcDxMb2NhbGUsIElucHV0TWV0aG9kPigpOwotICAgICAgICBzZWxlY3RJbnB1dE1ldGhvZChMb2NhbGUuVVMpOyAvLyBub3QgZGVmYXVsdD8KLSAgICAgICAgbmF0aXZlSU0gPSAoTmF0aXZlSU0pIGlucHV0TWV0aG9kOwotICAgIH0KLQotICAgIC8vPz8/QVdUCi0gICAgLyoKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBkaXNwYXRjaEV2ZW50KEFXVEV2ZW50IGV2ZW50KSB7Ci0gICAgICAgIGludCBpZCA9IGV2ZW50LmdldElEKCk7IAotICAgICAgICBpZiAoKGlkID49IEZvY3VzRXZlbnQuRk9DVVNfRklSU1QpICYmIChpZCA8PUZvY3VzRXZlbnQuRk9DVVNfTEFTVCkpIHsKLSAgICAgICAgICAgIGRpc3BhdGNoRm9jdXNFdmVudCgoRm9jdXNFdmVudCkgZXZlbnQpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLy8gaGFuZGxlIHNwZWNpYWwgS0VZX1BSRVNTRUQKLSAgICAgICAgICAgIC8vIGV2ZW50IHRvIHNob3cgSU0gc2VsZWN0aW9uIG1lbnUKLSAgICAgICAgICAgIGlmIChpZCA9PSBLZXlFdmVudC5LRVlfUFJFU1NFRCkgewotICAgICAgICAgICAgICAgIEtleUV2ZW50IGtlID0gKEtleUV2ZW50KSBldmVudDsKLSAgICAgICAgICAgICAgICBJTU1hbmFnZXIuc2VsZWN0SU0oa2UsIHRoaXMsIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBJTU1hbmFnZXIuZ2V0V2luZG93KGtlLmdldENvbXBvbmVudCgpKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAvLyBkaXNwYXRjaCBhbGwgaW5wdXQgZXZlbnRzIHRvIHRoZSBjdXJyZW50IElNOgotICAgICAgICAgICAgaWYgKGlucHV0TWV0aG9kICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBpbnB1dE1ldGhvZC5kaXNwYXRjaEV2ZW50KGV2ZW50KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBwcml2YXRlIHZvaWQgZGlzcGF0Y2hGb2N1c0V2ZW50KEZvY3VzRXZlbnQgZmUpIHsKLSAgICAgICAgc3dpdGNoIChmZS5nZXRJRCgpKSB7Ci0gICAgICAgIGNhc2UgRm9jdXNFdmVudC5GT0NVU19MT1NUOiAgICAgICAgICAgIAotICAgICAgICAgICAgaWYgKGlucHV0TWV0aG9kICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBpbnB1dE1ldGhvZC5kZWFjdGl2YXRlKGZlLmlzVGVtcG9yYXJ5KCkpOyAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEZvY3VzRXZlbnQuRk9DVVNfR0FJTkVEOgotICAgICAgICAgICAgCi0gICAgICAgICAgICBDb21wb25lbnQgY29tcCA9IGZlLmdldENvbXBvbmVudCgpOwotICAgICAgICAgICAgaWYgKGltV2luZG93cy5jb250YWlucyhjb21wKSkgewotICAgICAgICAgICAgICAgIC8vIHByZXZlbnQgYWN0aXZhdGluZyB3aGVuIElNIHdpbmRvd3MKLSAgICAgICAgICAgICAgICAvLyBhdHRhY2hlZCB0byB0aGlzIGNvbnRleHQgZ2FpbiBmb2N1cyAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBJbnB1dE1ldGhvZENvbnRleHQgbGFzdEFjdGl2ZSA9IElNTWFuYWdlci5nZXRMYXN0QWN0aXZlSU1DKCk7Ci0gICAgICAgICAgICBpZiAoKGxhc3RBY3RpdmUgIT0gdGhpcykgJiYgKGxhc3RBY3RpdmUgIT0gbnVsbCkpIHsKLSAgICAgICAgICAgICAgICBsYXN0QWN0aXZlLmhpZGVXaW5kb3dzKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoaW5wdXRNZXRob2QgIT0gbnVsbCkgewotICAgICAgICAgICAgICAgIGFjdGl2YXRlSU0oaW5wdXRNZXRob2QpOwotICAgICAgICAgICAgICAgIGlmICghZ2V0Q29tcG9zaXRpb25XaW5kb3coKS5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgICAgICAgICAgSU1NYW5hZ2VyLnNob3dDb21wb3NpdGlvbldpbmRvdyhjb21wb3NlV2luZG93KTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKGNsaWVudCA9PSBjb21wKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChuZXh0Q29tcCAhPSBudWxsKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyB0ZW1wb3JhcmlseSBnb3QgZm9jdXMgdG8KLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIGVuZCBjb21wb3NpdGlvbgotICAgICAgICAgICAgICAgICAgICAgICAgZW5kQ29tcG9zaXRpb24oKTsKLQotICAgICAgICAgICAgICAgICAgICAgICAgLy8gdHJhbnNmZXIgZm9jdXMgdG8gbmV3IGNsaWVudAotICAgICAgICAgICAgICAgICAgICAgICAgY2xpZW50ID0gbmV4dENvbXA7Ci0gICAgICAgICAgICAgICAgICAgICAgICBuZXh0Q29tcCA9IG51bGw7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjbGllbnQucmVxdWVzdEZvY3VzSW5XaW5kb3coKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoKGNsaWVudCAhPSBudWxsKSAmJiBnZXRDb21wb3NpdGlvbldpbmRvdygpLmlzVmlzaWJsZSgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIHRlbXBvcmFyaWx5IHJldHVybiBmb2N1cyBiYWNrCi0gICAgICAgICAgICAgICAgICAgIC8vIHRvIHByZXZpb3VzIGNsaWVudCB0byBiZSBhYmxlCi0gICAgICAgICAgICAgICAgICAgIC8vIHRvIGVuZCBjb21wb3NpdGlvbgotICAgICAgICAgICAgICAgICAgICBuZXh0Q29tcCA9IGNvbXA7Ci0gICAgICAgICAgICAgICAgICAgIGNsaWVudC5yZXF1ZXN0Rm9jdXNJbldpbmRvdygpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGNsaWVudCA9IGNvbXA7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHBlbmRpbmdDbGllbnROb3RpZnkpIHsKLSAgICAgICAgICAgICAgICBub3RpZnlDbGllbnRXaW5kb3dDaGFuZ2UoSU1NYW5hZ2VyLmdldFdpbmRvdyhjb21wKS5nZXRCb3VuZHMoKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotCi0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIGFjdGl2YXRlSU0oSW5wdXRNZXRob2QgaW0pIHsKLSAgICAgICAgaW0uYWN0aXZhdGUoKTsKLSAgICAgICAgaWYgKChuYXRpdmVJTSAhPSBudWxsKSAmJiAoaW0gIT0gbmF0aXZlSU0pKSB7Ci0gICAgICAgICAgICAvLyB3aGVuIEphdmEgSU0gaXMgYWN0aXZlCi0gICAgICAgICAgICAvLyBuYXRpdmUgaW5wdXQgbWV0aG9kIGVkaXRvciBtdXN0IGJlCi0gICAgICAgICAgICAvLyBleHBsaWNpdGx5IGRpc2FibGVkCi0gICAgICAgICAgICBuYXRpdmVJTS5kaXNhYmxlSU1FKCk7Ci0gICAgICAgIH0KLSAgICAgICAgSU1NYW5hZ2VyLnNldExhc3RBY3RpdmVJTUModGhpcyk7Ci0gICAgfQotCi0gICAgQFN1cHByZXNzV2FybmluZ3MoImRlcHJlY2F0aW9uIikKLSAgICBwcml2YXRlIHZvaWQgaGlkZVdpbmRvd3MoKSB7Ci0gICAgICAgIGlmIChpbnB1dE1ldGhvZCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBpbnB1dE1ldGhvZC5oaWRlV2luZG93cygpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChjb21wb3NlV2luZG93ICE9IG51bGwpIHsKLSAgICAgICAgICAgIGNvbXBvc2VXaW5kb3cuaGlkZSgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIGNyZWF0ZUNvbXBvc2l0aW9uV2luZG93KCkgewotICAgICAgICBjb21wb3NlV2luZG93ID0gbmV3IENvbXBvc2l0aW9uV2luZG93KGNsaWVudCk7ICAgICAgICAKLSAgICB9Ci0gICAgCi0gICAgcHJpdmF0ZSBDb21wb3NpdGlvbldpbmRvdyBnZXRDb21wb3NpdGlvbldpbmRvdygpIHsKLSAgICAgICAgaWYgKGNvbXBvc2VXaW5kb3cgPT0gbnVsbCkgewotICAgICAgICAgICAgY3JlYXRlQ29tcG9zaXRpb25XaW5kb3coKTsKLSAgICAgICAgfQotICAgICAgICBjb21wb3NlV2luZG93LnNldENsaWVudChjbGllbnQpOwotICAgICAgICByZXR1cm4gY29tcG9zZVdpbmRvdzsgICAgICAgIAotICAgIH0KLSAgICAqLwotICAgIAotICAgIC8qKgotICAgICAqIEdldHMgaW5wdXQgbWV0aG9kIHJlcXVlc3RzIGZvciB0aGUgY3VycmVudCBjbGllbnQKLSAgICAgKiBpcnJlc3BlY3RpdmUgb2YgaW5wdXQgc3R5bGUuCi0gICAgICogQHJldHVybiBpbnB1dCBtZXRob2QgcmVxdWVzdHMgb2YgY29tcG9zaXRpb24gd2luZG93IGlmCi0gICAgICogY2xpZW50IGlzIHBhc3NpdmUsCi0gICAgICogb3RoZXJ3aXNlIGlucHV0IG1ldGhvZCByZXF1ZXN0cyBvZiBjbGllbnQKLSAgICAgKi8KLSAgICBwcml2YXRlIElucHV0TWV0aG9kUmVxdWVzdHMgZ2V0SU1SZXF1ZXN0cygpIHsKLSAgICAgICAgSW5wdXRNZXRob2RSZXF1ZXN0cyBpbVJlcXVlc3RzID0gbnVsbDsKLSAgICAKLSAgICAgICAgaWYgKGNsaWVudCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBpbVJlcXVlc3RzID0gY2xpZW50LmdldElucHV0TWV0aG9kUmVxdWVzdHMoKTsKLSAgICAgICAgICAgIC8vPz8/QVdUCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgaWYgKGltUmVxdWVzdHMgPT0gbnVsbCkgeyAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICBpbVJlcXVlc3RzID0gZ2V0Q29tcG9zaXRpb25XaW5kb3coKS5nZXRJbnB1dE1ldGhvZFJlcXVlc3RzKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAqLwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICByZXR1cm4gaW1SZXF1ZXN0czsKLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogR2V0cyBpbnB1dCBtZXRob2QgcmVxdWVzdHMgZm9yIHRoZSBjdXJyZW50IGNsaWVudCAmIGlucHV0IHN0eWxlLgotICAgICAqIEByZXR1cm4gaW5wdXQgbWV0aG9kIHJlcXVlc3RzIG9mIGNvbXBvc2l0aW9uIHdpbmRvdyBpZgotICAgICAqIGlucHV0IHN0eWxlIGlzICJiZWxvdy10aGUtc3BvdCIob3IgY2xpZW50IGlzIHBhc3NpdmUpLAotICAgICAqIG90aGVyd2lzZSBjbGllbnQgaW5wdXQgbWV0aG9kIHJlcXVlc3RzCi0gICAgICovCi0gICAgcHJpdmF0ZSBJbnB1dE1ldGhvZFJlcXVlc3RzIGdldFN0eWxlSU1SZXF1ZXN0cygpIHsKLSAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgLyoKLSAgICAgICAgaWYgKElNTWFuYWdlci5iZWxvd1RoZVNwb3QoKSkgewotICAgICAgICAgICAgcmV0dXJuIGdldENvbXBvc2l0aW9uV2luZG93KCkuZ2V0SW5wdXRNZXRob2RSZXF1ZXN0cygpOwotICAgICAgICB9Ci0gICAgICAgICovCi0gICAgICAgIHJldHVybiBnZXRJTVJlcXVlc3RzKCk7Ci0gICAgfQotICAgIAotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRpc3Bvc2UoKSB7Ci0gICAgICAgIGlmIChpbnB1dE1ldGhvZCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBjbG9zZUlNKGlucHV0TWV0aG9kKTsKLSAgICAgICAgICAgIGlucHV0TWV0aG9kLmRpc3Bvc2UoKTsKLSAgICAgICAgfQotICAgICAgICBub3RpZnlJTS5jbGVhcigpOwotICAgICAgICBzdXBlci5kaXNwb3NlKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgZW5kQ29tcG9zaXRpb24oKSB7Ci0gICAgICAgIGlmIChpbnB1dE1ldGhvZCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBpbnB1dE1ldGhvZC5lbmRDb21wb3NpdGlvbigpOwotICAgICAgICB9Ci0gICAgICAgIHN1cGVyLmVuZENvbXBvc2l0aW9uKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIE9iamVjdCBnZXRJbnB1dE1ldGhvZENvbnRyb2xPYmplY3QoKSB7Ci0gICAgICAgIGlmIChpbnB1dE1ldGhvZCAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gaW5wdXRNZXRob2QuZ2V0Q29udHJvbE9iamVjdCgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBzdXBlci5nZXRJbnB1dE1ldGhvZENvbnRyb2xPYmplY3QoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgTG9jYWxlIGdldExvY2FsZSgpIHsKLSAgICAgICAgaWYgKGlucHV0TWV0aG9kICE9IG51bGwpIHsKLSAgICAgICAgICAgIHJldHVybiBpbnB1dE1ldGhvZC5nZXRMb2NhbGUoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gc3VwZXIuZ2V0TG9jYWxlKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gaXNDb21wb3NpdGlvbkVuYWJsZWQoKSB7Ci0gICAgICAgIGlmIChpbnB1dE1ldGhvZCAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gaW5wdXRNZXRob2QuaXNDb21wb3NpdGlvbkVuYWJsZWQoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gc3VwZXIuaXNDb21wb3NpdGlvbkVuYWJsZWQoKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCByZWNvbnZlcnQoKSB7Ci0gICAgICAgIGlmIChpbnB1dE1ldGhvZCAhPSBudWxsKSB7Ci0gICAgICAgICAgICBpbnB1dE1ldGhvZC5yZWNvbnZlcnQoKTsKLSAgICAgICAgfQotICAgICAgICBzdXBlci5yZWNvbnZlcnQoKTsKLSAgICB9Ci0KLSAgICAvLz8/P0FXVAotICAgIC8qCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgcmVtb3ZlTm90aWZ5KENvbXBvbmVudCBjbGllbnQpIHsKLSAgICAgICAgaWYgKChpbnB1dE1ldGhvZCAhPSBudWxsKSAmJiAoY2xpZW50ID09IHRoaXMuY2xpZW50KSkgewotICAgICAgICAgICAgaW5wdXRNZXRob2QucmVtb3ZlTm90aWZ5KCk7Ci0gICAgICAgICAgICBjbGllbnQgPSBudWxsOwotICAgICAgICAgICAgLy8gc2V0IGZsYWcgaW5kaWNhdGluZyB0aGF0IElNIHNob3VsZCBiZSBub3RpZmllZAotICAgICAgICAgICAgLy8gYXMgc29vbiBhcyBpdCBpcyBhY3RpdmF0ZWQobmV3IGNsaWVudCBhcHBlYXJzKQotICAgICAgICAgICAgcGVuZGluZ0NsaWVudE5vdGlmeSA9IHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIHN1cGVyLnJlbW92ZU5vdGlmeShjbGllbnQpOwotICAgIH0KLSAgICAqLwotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gc2VsZWN0SW5wdXRNZXRob2QoTG9jYWxlIGxvY2FsZSkgeyAgICAgICAgCi0gICAgICAgIAotICAgICAgICBpZiAoKGlucHV0TWV0aG9kICE9IG51bGwpICYmIGlucHV0TWV0aG9kLnNldExvY2FsZShsb2NhbGUpKSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgICAgICAvLyBmaXJzdAotICAgICAgICAvLyB0YWtlIGxhc3QgdXNlci1zZWxlY3RlZCBJTSBmb3IgbG9jYWxlICAgICAgICAgICAgCi0gICAgICAgIElucHV0TWV0aG9kIG5ld0lNID0gbG9jYWxlSU0uZ2V0KGxvY2FsZSk7Ci0gICAgICAgIAotICAgICAgICAvLyBpZiBub3QgZm91bmQgc2VhcmNoIHRocm91Z2ggSU0gZGVzY3JpcHRvcnMKLSAgICAgICAgLy8gYW5kIHRha2UgYWxyZWFkeSBjcmVhdGVkIGluc3RhbmNlIGlmIGV4aXN0cwotICAgICAgICAvLyBvciBjcmVhdGUsIHN0b3JlIG5ldyBJTSBpbnN0YW5jZSBpbiBkZXNjcmlwdG9yLT5pbnN0YW5jZSBtYXAKLSAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgLyoKLSAgICAgICAgaWYgKG5ld0lNID09IG51bGwpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgbmV3SU0gPSBnZXRJTUluc3RhbmNlKElNTWFuYWdlci5nZXRJTURlc2NyaXB0b3JzKCkuaXRlcmF0b3IoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbG9jYWxlKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgLy8gaWdub3JlIGV4Y2VwdGlvbnMgLSBqdXN0IHJldHVybiBmYWxzZQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgICovCi0gICAgICAgIAotICAgICAgICByZXR1cm4gc3dpdGNoVG9JTShsb2NhbGUsIG5ld0lNKTsKLSAgICB9Ci0KLSAgICBwcml2YXRlIGJvb2xlYW4gc3dpdGNoVG9JTShMb2NhbGUgbG9jYWxlLCBJbnB1dE1ldGhvZCBuZXdJTSkgewotICAgICAgICAvLz8/P0FXVAotICAgICAgICAvKgotICAgICAgICBpZiAobmV3SU0gIT0gbnVsbCkgewotICAgICAgICAgICAgY2xvc2VJTShpbnB1dE1ldGhvZCk7Ci0gICAgICAgICAgICBjbGllbnQgPSBLZXlib2FyZEZvY3VzTWFuYWdlci4KLSAgICAgICAgICAgIGdldEN1cnJlbnRLZXlib2FyZEZvY3VzTWFuYWdlcigpLmdldEZvY3VzT3duZXIoKTsKLSAgICAgICAgICAgIGluaXRJTShuZXdJTSwgbG9jYWxlKTsKLSAgICAgICAgICAgIGlucHV0TWV0aG9kID0gbmV3SU07Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgICAgICovCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogSXMgY2FsbGVkIHdoZW4gSU0gaXMgc2VsZWN0ZWQgZnJvbSBVSQotICAgICAqLwotICAgIHZvaWQgc2VsZWN0SU0oSW5wdXRNZXRob2REZXNjcmlwdG9yIGltZCwgTG9jYWxlIGxvY2FsZSkgewotICAgICAgICB0cnkgewotICAgICAgICAgICAgc3dpdGNoVG9JTShsb2NhbGUsIGdldElNSW5zdGFuY2UoaW1kKSk7ICAgICAgICAgICAgCi0gICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICBlLnByaW50U3RhY2tUcmFjZSgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogR2V0cyBpbnB1dCBtZXRob2QgaW5zdGFuY2UgZm9yIHRoZSBnaXZlbgotICAgICAqIGxvY2FsZSBmcm9tIHRoZSBnaXZlbiBsaXN0IG9mIGRlc2NyaXB0b3JzCi0gICAgICogQHBhcmFtIGRlc2NyaXB0b3JzIGl0ZXJhdG9yIG9mIHRoZSBsaXN0IG9mIElNIGRlc2NyaXB0b3JzCi0gICAgICogQHBhcmFtIGxvY2FsZSB0aGUgbG9jYWxlIHRvIGJlIHN1cHBvcnRlZCBieSB0aGUgSU0KLSAgICAgKiBAcmV0dXJuIGlucHV0IG1ldGhvZCBpbnN0YW5jZQotICAgICAqIEB0aHJvd3MgRXhjZXB0aW9uCi0gICAgICovCi0gICAgcHJpdmF0ZSBJbnB1dE1ldGhvZCBnZXRJTUluc3RhbmNlKEl0ZXJhdG9yPElucHV0TWV0aG9kRGVzY3JpcHRvcj4gZGVzY3JpcHRvcnMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExvY2FsZSBsb2NhbGUpIHRocm93cyBFeGNlcHRpb24gewotICAgICAgICB3aGlsZSAoZGVzY3JpcHRvcnMuaGFzTmV4dCgpKSB7Ci0gICAgICAgICAgICBJbnB1dE1ldGhvZERlc2NyaXB0b3IgZGVzYyA9IGRlc2NyaXB0b3JzLm5leHQoKTsKLSAgICAgICAgICAgIExvY2FsZVtdIGxvY3MgPSBkZXNjLmdldEF2YWlsYWJsZUxvY2FsZXMoKTsKLSAgICAgICAgICAgIGZvciAoTG9jYWxlIGVsZW1lbnQgOiBsb2NzKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGxvY2FsZS5lcXVhbHMoZWxlbWVudCkpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGdldElNSW5zdGFuY2UoZGVzYyk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIHByaXZhdGUgSW5wdXRNZXRob2QgZ2V0SU1JbnN0YW5jZShJbnB1dE1ldGhvZERlc2NyaXB0b3IgaW1kKSB0aHJvd3MgRXhjZXB0aW9uIHsKLSAgICAgICAgSW5wdXRNZXRob2QgaW0gPSBpbUluc3RhbmNlcy5nZXQoaW1kKTsKLSAgICAgICAgaWYgKGltID09IG51bGwpIHsKLSAgICAgICAgICAgIGltID0gaW1kLmNyZWF0ZUlucHV0TWV0aG9kKCk7Ci0gICAgICAgICAgICBpbS5zZXRJbnB1dE1ldGhvZENvbnRleHQodGhpcyk7Ci0gICAgICAgICAgICBpbUluc3RhbmNlcy5wdXQoaW1kLCBpbSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGltOwotICAgIH0KLSAgICAKLSAgICBwcml2YXRlIHZvaWQgaW5pdElNKElucHV0TWV0aG9kIGltLCBMb2NhbGUgbG9jYWxlKSB7Ci0gICAgICAgIGlmIChpbSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgaW0uc2V0TG9jYWxlKGxvY2FsZSk7Ci0gICAgICAgIGltLnNldENoYXJhY3RlclN1YnNldHMobnVsbCk7Ci0gICAgICAgIC8vPz8/QVdUOiBhY3RpdmF0ZUlNKGltKTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIGltLnNldENvbXBvc2l0aW9uRW5hYmxlZChpbnB1dE1ldGhvZCAhPSBudWxsID8gCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW5wdXRNZXRob2QuaXNDb21wb3NpdGlvbkVuYWJsZWQoKSA6IHRydWUpOwotICAgICAgICB9IGNhdGNoIChVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbiB1b2UpIHsKLQotICAgICAgICB9Ci0gICAgICAgIAotICAgIH0KLQotICAgIHByaXZhdGUgdm9pZCBjbG9zZUlNKElucHV0TWV0aG9kIGltKSB7Ci0gICAgICAgIGlmIChpbSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGltLmlzQ29tcG9zaXRpb25FbmFibGVkKCkpIHsKLSAgICAgICAgICAgIGltLmVuZENvbXBvc2l0aW9uKCk7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIGltLmRlYWN0aXZhdGUodHJ1ZSk7Ci0gICAgICAgIGltLmhpZGVXaW5kb3dzKCk7Ci0gICAgICAgIAotICAgIH0KLSAgICAKLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRDaGFyYWN0ZXJTdWJzZXRzKFN1YnNldFtdIHN1YnNldHMpIHsKLSAgICAgICAgaWYgKGlucHV0TWV0aG9kICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlucHV0TWV0aG9kLnNldENoYXJhY3RlclN1YnNldHMoc3Vic2V0cyk7Ci0gICAgICAgIH0KLSAgICAgICAgc3VwZXIuc2V0Q2hhcmFjdGVyU3Vic2V0cyhzdWJzZXRzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCBzZXRDb21wb3NpdGlvbkVuYWJsZWQoYm9vbGVhbiBlbmFibGUpIHsKLSAgICAgICAgaWYgKGlucHV0TWV0aG9kICE9IG51bGwpIHsKLSAgICAgICAgICAgIGlucHV0TWV0aG9kLnNldENvbXBvc2l0aW9uRW5hYmxlZChlbmFibGUpOwotICAgICAgICB9Ci0gICAgICAgIHN1cGVyLnNldENvbXBvc2l0aW9uRW5hYmxlZChlbmFibGUpOwotICAgIH0KLQotICAgIC8vPz8/QVdUCi0gICAgLyoKLSAgICBwdWJsaWMgSkZyYW1lIGNyZWF0ZUlucHV0TWV0aG9kSkZyYW1lKFN0cmluZyB0aXRsZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2xlYW4gYXR0YWNoVG9JbnB1dENvbnRleHQpIHsKLSAgICAgICAgSkZyYW1lIGpmID0gbmV3IElNSkZyYW1lKHRpdGxlLCBhdHRhY2hUb0lucHV0Q29udGV4dCA/IHRoaXMgOiBudWxsKTsKLSAgICAgICAgaW1XaW5kb3dzLmFkZChqZik7Ci0gICAgICAgIHJldHVybiBqZjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgV2luZG93IGNyZWF0ZUlucHV0TWV0aG9kV2luZG93KFN0cmluZyB0aXRsZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2xlYW4gYXR0YWNoVG9JbnB1dENvbnRleHQpIHsKLSAgICAgICAgV2luZG93IHcgPSBuZXcgSU1XaW5kb3codGl0bGUsIGF0dGFjaFRvSW5wdXRDb250ZXh0ID8gdGhpcyA6IG51bGwpOwotICAgICAgICBpbVdpbmRvd3MuYWRkKHcpOwotICAgICAgICByZXR1cm4gdzsKLSAgICB9Ci0gICAgKi8KLSAgICAKLSAgICBAU3VwcHJlc3NXYXJuaW5ncygiZGVwcmVjYXRpb24iKQotICAgIHB1YmxpYyB2b2lkIGRpc3BhdGNoSW5wdXRNZXRob2RFdmVudChpbnQgaWQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEF0dHJpYnV0ZWRDaGFyYWN0ZXJJdGVyYXRvciB0ZXh0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgY29tbWl0dGVkQ2hhcmFjdGVyQ291bnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRleHRIaXRJbmZvIGNhcmV0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBUZXh0SGl0SW5mbyB2aXNpYmxlUG9zaXRpb24pIHsKLSAgICAgICAgaWYgKGNsaWVudCA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgLyoKLSAgICAgICAgSW5wdXRNZXRob2RFdmVudCBpbWUgPSBuZXcgSW5wdXRNZXRob2RFdmVudChjbGllbnQsIGlkLCB0ZXh0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbW1pdHRlZENoYXJhY3RlckNvdW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNhcmV0LCB2aXNpYmxlUG9zaXRpb24pOwotICAgICAgICAKLQotICAgICAgICBpZiAoKGNsaWVudC5nZXRJbnB1dE1ldGhvZFJlcXVlc3RzKCkgIT0gbnVsbCkgJiYKLSAgICAgICAgICAgICFJTU1hbmFnZXIuYmVsb3dUaGVTcG90KCkpIHsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgY2xpZW50LmRpc3BhdGNoRXZlbnQoaW1lKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8gc2hvdy9oaWRlIGNvbXBvc2l0aW9uIHdpbmRvdyBpZiBuZWNlc3NhcnkKLSAgICAgICAgICAgIGlmIChjb21taXR0ZWRDaGFyYWN0ZXJDb3VudCA8IHRleHQuZ2V0RW5kSW5kZXgoKSkgewotICAgICAgICAgICAgICAgIElNTWFuYWdlci5zaG93Q29tcG9zaXRpb25XaW5kb3coZ2V0Q29tcG9zaXRpb25XaW5kb3coKSk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGdldENvbXBvc2l0aW9uV2luZG93KCkuaGlkZSgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY29tcG9zZVdpbmRvdy5nZXRBY3RpdmVDbGllbnQoKS5kaXNwYXRjaEV2ZW50KGltZSk7Ci0gICAgICAgIH0KLSAgICAgICAgKi8KLSAgICAgICAgCi0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZW5hYmxlQ2xpZW50V2luZG93Tm90aWZpY2F0aW9uKElucHV0TWV0aG9kIGlucHV0TWV0aG9kLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBib29sZWFuIGVuYWJsZSkgewotICAgICAgICBpZiAoZW5hYmxlKSB7Ci0gICAgICAgICAgICBub3RpZnlJTS5hZGQoaW5wdXRNZXRob2QpOwotICAgICAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgICAgIC8qCi0gICAgICAgICAgICBpZiAoY2xpZW50ICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBub3RpZnlDbGllbnRXaW5kb3dDaGFuZ2UoSU1NYW5hZ2VyLmdldFdpbmRvdyhjbGllbnQpLmdldEJvdW5kcygpKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgcGVuZGluZ0NsaWVudE5vdGlmeSA9IHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAqLwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbm90aWZ5SU0ucmVtb3ZlKGlucHV0TWV0aG9kKTsKLSAgICAgICAgfQotICAgICAgICAKLSAgICB9Ci0KLSAgICBwdWJsaWMgQXR0cmlidXRlZENoYXJhY3Rlckl0ZXJhdG9yIGNhbmNlbExhdGVzdENvbW1pdHRlZFRleHQoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEF0dHJpYnV0ZVtdIGF0dHJpYnV0ZXMpIHsKLSAgICAgICAgcmV0dXJuIGdldElNUmVxdWVzdHMoKS5jYW5jZWxMYXRlc3RDb21taXR0ZWRUZXh0KGF0dHJpYnV0ZXMpOwotICAgIH0KLQotICAgIHB1YmxpYyBBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgZ2V0Q29tbWl0dGVkVGV4dChpbnQgYmVnaW5JbmRleCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGVuZEluZGV4LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBBdHRyaWJ1dGVbXSBhdHRyaWJ1dGVzKSB7Ci0gICAgICAgIHJldHVybiBnZXRJTVJlcXVlc3RzKCkuZ2V0Q29tbWl0dGVkVGV4dChiZWdpbkluZGV4LCBlbmRJbmRleCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGF0dHJpYnV0ZXMpOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0Q29tbWl0dGVkVGV4dExlbmd0aCgpIHsKLSAgICAgICAgcmV0dXJuIGdldElNUmVxdWVzdHMoKS5nZXRDb21taXR0ZWRUZXh0TGVuZ3RoKCk7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRJbnNlcnRQb3NpdGlvbk9mZnNldCgpIHsKLSAgICAgICAgcmV0dXJuIGdldElNUmVxdWVzdHMoKS5nZXRJbnNlcnRQb3NpdGlvbk9mZnNldCgpOwotICAgIH0KLQotICAgIHB1YmxpYyBUZXh0SGl0SW5mbyBnZXRMb2NhdGlvbk9mZnNldChpbnQgeCwgaW50IHkpIHsKLSAgICAgICAgSW5wdXRNZXRob2RSZXF1ZXN0cyBpbXIgPSBnZXRTdHlsZUlNUmVxdWVzdHMoKTsKLSAgICAgICAgaWYgKGltciAhPSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gaW1yLmdldExvY2F0aW9uT2Zmc2V0KHgsIHkpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIHB1YmxpYyBBdHRyaWJ1dGVkQ2hhcmFjdGVySXRlcmF0b3IgZ2V0U2VsZWN0ZWRUZXh0KEF0dHJpYnV0ZVtdIGF0dHJpYnV0ZXMpIHsKLSAgICAgICAgcmV0dXJuIGdldElNUmVxdWVzdHMoKS5nZXRTZWxlY3RlZFRleHQoYXR0cmlidXRlcyk7Ci0gICAgfQotCi0gICAgcHVibGljIFJlY3RhbmdsZSBnZXRUZXh0TG9jYXRpb24oVGV4dEhpdEluZm8gb2Zmc2V0KSB7ICAgICAgICAKLSAgICAgICAgcmV0dXJuIGdldFN0eWxlSU1SZXF1ZXN0cygpLmdldFRleHRMb2NhdGlvbihvZmZzZXQpOwotICAgIH0KLSAgICAKLSAgICAvKioKLSAgICAgKiBUbyBiZSBjYWxsZWQgYnkgQVdUIHdoZW4gY2xpZW50IFdpbmRvdydzIGJvdW5kcy92aXNpYmlsaXR5L3N0YXRlCi0gICAgICogY2hhbmdlCi0gICAgICovCi0gICAgcHVibGljIHZvaWQgbm90aWZ5Q2xpZW50V2luZG93Q2hhbmdlKFJlY3RhbmdsZSBib3VuZHMpIHsKLSAgICAgICAgaWYgKG5vdGlmeUlNLmNvbnRhaW5zKGlucHV0TWV0aG9kKSkgewotICAgICAgICAgICAgaW5wdXRNZXRob2Qubm90aWZ5Q2xpZW50V2luZG93Q2hhbmdlKGJvdW5kcyk7Ci0gICAgICAgIH0KLSAgICAgICAgcGVuZGluZ0NsaWVudE5vdGlmeSA9IGZhbHNlOwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBJbnB1dE1ldGhvZCBnZXRJbnB1dE1ldGhvZCgpIHsKLSAgICAgICAgcmV0dXJuIGlucHV0TWV0aG9kOwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBDb21wb25lbnQgZ2V0Q2xpZW50KCkgewotICAgICAgICByZXR1cm4gY2xpZW50OwotICAgIH0KLQotICAgIHB1YmxpYyBmaW5hbCBOYXRpdmVJTSBnZXROYXRpdmVJTSgpIHsKLSAgICAgICAgcmV0dXJuIG5hdGl2ZUlNOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2ludGVybmFsL25scy9NZXNzYWdlcy5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvaW50ZXJuYWwvbmxzL01lc3NhZ2VzLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGMzNDAzNTguLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvaW50ZXJuYWwvbmxzL01lc3NhZ2VzLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNTEgKzAsMCBAQAotLyogCi0gKiBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICogCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKiAKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8qCi0gKiBUSEUgRklMRSBIQVMgQkVFTiBBVVRPR0VORVJBVEVEIEJZIE1TR1RPT0wgVE9PTC4KLSAqIEFsbCBjaGFuZ2VzIG1hZGUgdG8gdGhpcyBmaWxlIG1hbnVhbGx5IHdpbGwgYmUgb3ZlcndyaXR0ZW4gCi0gKiBpZiB0aGlzIHRvb2wgcnVucyBhZ2Fpbi4gQmV0dGVyIG1ha2UgY2hhbmdlcyBpbiB0aGUgdGVtcGxhdGUgZmlsZS4KLSAqLwotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzOwotCi0KLWltcG9ydCBqYXZhLnNlY3VyaXR5LkFjY2Vzc0NvbnRyb2xsZXI7Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5Qcml2aWxlZ2VkQWN0aW9uOwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci1pbXBvcnQgamF2YS51dGlsLk1pc3NpbmdSZXNvdXJjZUV4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuUmVzb3VyY2VCdW5kbGU7Ci0KLS8vIEJFR0lOIGFuZHJvaWQtZGVsZXRlZAotLyoKLSAqIEZvciBBbmRyb2lkLCB0aGlzIG1vZHVsZSBpcyBhIHNlcGFyYXRlIGxpYnJhcnkgYW5kIG5vdCBwYXJ0IG9mIHRoZQotICogYm9vdCBjbGFzc3BhdGgsIHNvIGl0cyByZXNvdXJjZXMgd29uJ3QgYmUgZm91bmQgb24gdGhlIGJvb3QgY2xhc3NwYXRoCi0gKiBhcyBpcyBhc3N1bWVkIGJ5IE1zZ0hlbHAuZ2V0U3RyaW5nKCkuIFdlIGluc3RlYWQgdXNlIGEgbG9jYWwgTXNnSGVscAotICogd2hpY2ggYm90dG9tcyBvdXQgaW4gYSBjYWxsIHRvIHRoZSB1c2VmdWwgcGFydCBvZiBpdHMgbG93ZXItbGV2ZWwKLSAqIG5hbWVzYWtlLgotICovCi0vL2ltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkua2VybmVsLnZtLlZNOwotLy9pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Nc2dIZWxwOwotLy8gRU5EIGFuZHJvaWQtZGVsZXRlZAotCi0vKioKLSAqIFRoaXMgY2xhc3MgcmV0cmlldmVzIHN0cmluZ3MgZnJvbSBhIHJlc291cmNlIGJ1bmRsZSBhbmQgcmV0dXJucyB0aGVtLAotICogZm9ybWF0dGluZyB0aGVtIHdpdGggTWVzc2FnZUZvcm1hdCB3aGVuIHJlcXVpcmVkLgotICogPHA+Ci0gKiBJdCBpcyB1c2VkIGJ5IHRoZSBzeXN0ZW0gY2xhc3NlcyB0byBwcm92aWRlIG5hdGlvbmFsIGxhbmd1YWdlIHN1cHBvcnQsIGJ5Ci0gKiBsb29raW5nIHVwIG1lc3NhZ2VzIGluIHRoZSA8Y29kZT4KLSAqICAgIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLm1lc3NhZ2VzCi0gKiA8L2NvZGU+Ci0gKiByZXNvdXJjZSBidW5kbGUuIE5vdGUgdGhhdCBpZiB0aGlzIGZpbGUgaXMgbm90IGF2YWlsYWJsZSwgb3IgYW4gaW52YWxpZCBrZXkKLSAqIGlzIGxvb2tlZCB1cCwgb3IgcmVzb3VyY2UgYnVuZGxlIHN1cHBvcnQgaXMgbm90IGF2YWlsYWJsZSwgdGhlIGtleSBpdHNlbGYKLSAqIHdpbGwgYmUgcmV0dXJuZWQgYXMgdGhlIGFzc29jaWF0ZWQgbWVzc2FnZS4gVGhpcyBtZWFucyB0aGF0IHRoZSA8ZW0+S0VZPC9lbT4KLSAqIHNob3VsZCBhIHJlYXNvbmFibGUgaHVtYW4tcmVhZGFibGUgKGVuZ2xpc2gpIHN0cmluZy4KLSAqIAotICovCi1wdWJsaWMgY2xhc3MgTWVzc2FnZXMgewotCi0gICAgLy8gQkVHSU4gYW5kcm9pZC1kZWxldGVkCi0gICAgLy9wcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgc1Jlc291cmNlID0KLSAgICAvLyAgICAib3JnLmFwYWNoZS5oYXJtb255LmF3dC5pbnRlcm5hbC5ubHMubWVzc2FnZXMiOwotICAgIC8vIEVORCBhbmRyb2lkLWRlbGV0ZWQKLQotICAgIC8qKgotICAgICAqIFJldHJpZXZlcyBhIG1lc3NhZ2Ugd2hpY2ggaGFzIG5vIGFyZ3VtZW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXNnCi0gICAgICogICAgICAgICAgICBTdHJpbmcgdGhlIGtleSB0byBsb29rIHVwLgotICAgICAqIEByZXR1cm4gU3RyaW5nIHRoZSBtZXNzYWdlIGZvciB0aGF0IGtleSBpbiB0aGUgc3lzdGVtIG1lc3NhZ2UgYnVuZGxlLgotICAgICAqLwotICAgIHN0YXRpYyBwdWJsaWMgU3RyaW5nIGdldFN0cmluZyhTdHJpbmcgbXNnKSB7Ci0gICAgICAgIC8vIEJFR0lOIGFuZHJvaWQtY2hhbmdlZAotICAgICAgICByZXR1cm4gTXNnSGVscC5nZXRTdHJpbmcobXNnKTsKLSAgICAgICAgLy8gRU5EIGFuZHJvaWQtY2hhbmdlZAotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHJpZXZlcyBhIG1lc3NhZ2Ugd2hpY2ggdGFrZXMgMSBhcmd1bWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXNnCi0gICAgICogICAgICAgICAgICBTdHJpbmcgdGhlIGtleSB0byBsb29rIHVwLgotICAgICAqIEBwYXJhbSBhcmcKLSAgICAgKiAgICAgICAgICAgIE9iamVjdCB0aGUgb2JqZWN0IHRvIGluc2VydCBpbiB0aGUgZm9ybWF0dGVkIG91dHB1dC4KLSAgICAgKiBAcmV0dXJuIFN0cmluZyB0aGUgbWVzc2FnZSBmb3IgdGhhdCBrZXkgaW4gdGhlIHN5c3RlbSBtZXNzYWdlIGJ1bmRsZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgcHVibGljIFN0cmluZyBnZXRTdHJpbmcoU3RyaW5nIG1zZywgT2JqZWN0IGFyZykgewotICAgICAgICByZXR1cm4gZ2V0U3RyaW5nKG1zZywgbmV3IE9iamVjdFtdIHsgYXJnIH0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHJpZXZlcyBhIG1lc3NhZ2Ugd2hpY2ggdGFrZXMgMSBpbnRlZ2VyIGFyZ3VtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBtc2cKLSAgICAgKiAgICAgICAgICAgIFN0cmluZyB0aGUga2V5IHRvIGxvb2sgdXAuCi0gICAgICogQHBhcmFtIGFyZwotICAgICAqICAgICAgICAgICAgaW50IHRoZSBpbnRlZ2VyIHRvIGluc2VydCBpbiB0aGUgZm9ybWF0dGVkIG91dHB1dC4KLSAgICAgKiBAcmV0dXJuIFN0cmluZyB0aGUgbWVzc2FnZSBmb3IgdGhhdCBrZXkgaW4gdGhlIHN5c3RlbSBtZXNzYWdlIGJ1bmRsZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgcHVibGljIFN0cmluZyBnZXRTdHJpbmcoU3RyaW5nIG1zZywgaW50IGFyZykgewotICAgICAgICByZXR1cm4gZ2V0U3RyaW5nKG1zZywgbmV3IE9iamVjdFtdIHsgSW50ZWdlci50b1N0cmluZyhhcmcpIH0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHJpZXZlcyBhIG1lc3NhZ2Ugd2hpY2ggdGFrZXMgMSBjaGFyYWN0ZXIgYXJndW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1zZwotICAgICAqICAgICAgICAgICAgU3RyaW5nIHRoZSBrZXkgdG8gbG9vayB1cC4KLSAgICAgKiBAcGFyYW0gYXJnCi0gICAgICogICAgICAgICAgICBjaGFyIHRoZSBjaGFyYWN0ZXIgdG8gaW5zZXJ0IGluIHRoZSBmb3JtYXR0ZWQgb3V0cHV0LgotICAgICAqIEByZXR1cm4gU3RyaW5nIHRoZSBtZXNzYWdlIGZvciB0aGF0IGtleSBpbiB0aGUgc3lzdGVtIG1lc3NhZ2UgYnVuZGxlLgotICAgICAqLwotICAgIHN0YXRpYyBwdWJsaWMgU3RyaW5nIGdldFN0cmluZyhTdHJpbmcgbXNnLCBjaGFyIGFyZykgewotICAgICAgICByZXR1cm4gZ2V0U3RyaW5nKG1zZywgbmV3IE9iamVjdFtdIHsgU3RyaW5nLnZhbHVlT2YoYXJnKSB9KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXRyaWV2ZXMgYSBtZXNzYWdlIHdoaWNoIHRha2VzIDIgYXJndW1lbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBtc2cKLSAgICAgKiAgICAgICAgICAgIFN0cmluZyB0aGUga2V5IHRvIGxvb2sgdXAuCi0gICAgICogQHBhcmFtIGFyZzEKLSAgICAgKiAgICAgICAgICAgIE9iamVjdCBhbiBvYmplY3QgdG8gaW5zZXJ0IGluIHRoZSBmb3JtYXR0ZWQgb3V0cHV0LgotICAgICAqIEBwYXJhbSBhcmcyCi0gICAgICogICAgICAgICAgICBPYmplY3QgYW5vdGhlciBvYmplY3QgdG8gaW5zZXJ0IGluIHRoZSBmb3JtYXR0ZWQgb3V0cHV0LgotICAgICAqIEByZXR1cm4gU3RyaW5nIHRoZSBtZXNzYWdlIGZvciB0aGF0IGtleSBpbiB0aGUgc3lzdGVtIG1lc3NhZ2UgYnVuZGxlLgotICAgICAqLwotICAgIHN0YXRpYyBwdWJsaWMgU3RyaW5nIGdldFN0cmluZyhTdHJpbmcgbXNnLCBPYmplY3QgYXJnMSwgT2JqZWN0IGFyZzIpIHsKLSAgICAgICAgcmV0dXJuIGdldFN0cmluZyhtc2csIG5ldyBPYmplY3RbXSB7IGFyZzEsIGFyZzIgfSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0cmlldmVzIGEgbWVzc2FnZSB3aGljaCB0YWtlcyBzZXZlcmFsIGFyZ3VtZW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXNnCi0gICAgICogICAgICAgICAgICBTdHJpbmcgdGhlIGtleSB0byBsb29rIHVwLgotICAgICAqIEBwYXJhbSBhcmdzCi0gICAgICogICAgICAgICAgICBPYmplY3RbXSB0aGUgb2JqZWN0cyB0byBpbnNlcnQgaW4gdGhlIGZvcm1hdHRlZCBvdXRwdXQuCi0gICAgICogQHJldHVybiBTdHJpbmcgdGhlIG1lc3NhZ2UgZm9yIHRoYXQga2V5IGluIHRoZSBzeXN0ZW0gbWVzc2FnZSBidW5kbGUuCi0gICAgICovCi0gICAgc3RhdGljIHB1YmxpYyBTdHJpbmcgZ2V0U3RyaW5nKFN0cmluZyBtc2csIE9iamVjdFtdIGFyZ3MpIHsKLSAgICAgICAgLy8gQkVHSU4gYW5kcm9pZC1jaGFuZ2VkCi0gICAgICAgIHJldHVybiBNc2dIZWxwLmdldFN0cmluZyhtc2csIGFyZ3MpOwotICAgICAgICAvLyBFTkQgYW5kcm9pZC1jaGFuZ2VkCi0gICAgfQotCi0gICAgLy8gQkVHSU4gYW5kcm9pZC1ub3RlCi0gICAgLy8gRHVwbGljYXRlIGNvZGUgd2FzIGRyb3BwZWQgaW4gZmF2b3Igb2YgdXNpbmcgTXNnSGVscC4KLSAgICAvLyBFTkQgYW5kcm9pZC1ub3RlCi19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9pbnRlcm5hbC9ubHMvTXNnSGVscC5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3QvaW50ZXJuYWwvbmxzL01zZ0hlbHAuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYjU3ZmUxMS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9pbnRlcm5hbC9ubHMvTXNnSGVscC5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsODYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLyoKLSAqIFRoaXMgaW1wbGVtZW50YXRpb24gaXMgYmFzZWQgb24gdGhlIGNsYXNzIG9mIHRoZSBzYW1lIG5hbWUgaW4KLSAqIG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuCi0gKi8KLQotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5sczsKLQotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLnV0aWwubG9nZ2luZy5Mb2dnZXI7Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLWltcG9ydCBqYXZhLnV0aWwuUHJvcGVydHlSZXNvdXJjZUJ1bmRsZTsKLWltcG9ydCBqYXZhLnV0aWwuUmVzb3VyY2VCdW5kbGU7Ci1pbXBvcnQgamF2YS51dGlsLk1pc3NpbmdSZXNvdXJjZUV4Y2VwdGlvbjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGNvbnRhaW5zIGhlbHBlciBtZXRob2RzIGZvciBsb2FkaW5nIHJlc291cmNlIGJ1bmRsZXMgYW5kCi0gKiBmb3JtYXR0aW5nIGV4dGVybmFsIG1lc3NhZ2Ugc3RyaW5ncy4KLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIE1zZ0hlbHAgewotICAgIC8qKiBuYW1lIG9mIHRoZSByZXNvdXJjZSBmb3IgdGhpcyBjbGFzcyAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBSRVNPVVJDRV9OQU1FID0KLSAgICAgICAgIi9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L2ludGVybmFsL25scy9tZXNzYWdlcy5wcm9wZXJ0aWVzIjsKLQotICAgIC8qKiB0aGUgcmVzb3VyY2UgYnVuZGxlIGZvciB0aGlzIGNsYXNzICovCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgUmVzb3VyY2VCdW5kbGUgVEhFX0JVTkRMRTsKLQotICAgIHN0YXRpYyB7Ci0gICAgICAgIFJlc291cmNlQnVuZGxlIHJiID0gbnVsbDsKLQotICAgICAgICB0cnkgewotICAgICAgICAgICAgSW5wdXRTdHJlYW0gaW4gPSBNc2dIZWxwLmNsYXNzLmdldFJlc291cmNlQXNTdHJlYW0oCi0gICAgICAgICAgICAgICAgICAgIFJFU09VUkNFX05BTUUpOwotICAgICAgICAgICAgcmIgPSBuZXcgUHJvcGVydHlSZXNvdXJjZUJ1bmRsZShpbik7Ci0gICAgICAgIH0gY2F0Y2ggKElPRXhjZXB0aW9uIGV4KSB7Ci0gICAgICAgICAgICBMb2dnZXIuZ2xvYmFsLndhcm5pbmcoIkNvdWxkbid0IHJlYWQgcmVzb3VyY2UgYnVuZGxlOiAiICsKLSAgICAgICAgICAgICAgICAgICAgZXgpOwotICAgICAgICB9IGNhdGNoIChSdW50aW1lRXhjZXB0aW9uIGV4KSB7Ci0gICAgICAgICAgICAvLyBTaG91bGRuJ3QgaGFwcGVuLCBidXQgZGVhbCBhdCBsZWFzdCBzb21ld2hhdCBncmFjZWZ1bGx5LgotICAgICAgICAgICAgTG9nZ2VyLmdsb2JhbC53YXJuaW5nKCJDb3VsZG4ndCBmaW5kIHJlc291cmNlIGJ1bmRsZTogIiArCi0gICAgICAgICAgICAgICAgICAgIGV4KTsKLSAgICAgICAgfQotCi0gICAgICAgIFRIRV9CVU5ETEUgPSByYjsKLSAgICB9Ci0gICAgCi0gICAgcHVibGljIHN0YXRpYyBTdHJpbmcgZ2V0U3RyaW5nKFN0cmluZyBtc2cpIHsKLSAgICAgICAgaWYgKFRIRV9CVU5ETEUgPT0gbnVsbCkgewotICAgICAgICAgICAgcmV0dXJuIG1zZzsKLSAgICAgICAgfQotICAgICAgICB0cnkgewotICAgICAgICAgICAgcmV0dXJuIFRIRV9CVU5ETEUuZ2V0U3RyaW5nKG1zZyk7Ci0gICAgICAgIH0gY2F0Y2ggKE1pc3NpbmdSZXNvdXJjZUV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICByZXR1cm4gIk1pc3NpbmcgbWVzc2FnZTogIiArIG1zZzsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBzdGF0aWMgcHVibGljIFN0cmluZyBnZXRTdHJpbmcoU3RyaW5nIG1zZywgT2JqZWN0W10gYXJncykgewotICAgICAgICBTdHJpbmcgZm9ybWF0ID0gbXNnOwotICAgICAgICBpZiAoVEhFX0JVTkRMRSAhPSBudWxsKSB7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIGZvcm1hdCA9IFRIRV9CVU5ETEUuZ2V0U3RyaW5nKG1zZyk7Ci0gICAgICAgICAgICB9IGNhdGNoIChNaXNzaW5nUmVzb3VyY2VFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgcmV0dXJuIG9yZy5hcGFjaGUuaGFybW9ueS5sdW5pLnV0aWwuTXNnSGVscC5mb3JtYXQoZm9ybWF0LCBhcmdzKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9zdGF0ZS9NZW51SXRlbVN0YXRlLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9zdGF0ZS9NZW51SXRlbVN0YXRlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGIxM2U1MGIuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvc3RhdGUvTWVudUl0ZW1TdGF0ZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTEgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LnN0YXRlOwotCi1pbXBvcnQgamF2YS5hd3QuRGltZW5zaW9uOwotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLQotLyoqCi0gKiBTdGF0ZSBvZiBtZW51IGl0ZW0KLSAqLwotCi1wdWJsaWMgaW50ZXJmYWNlIE1lbnVJdGVtU3RhdGUgewotCi0gICAgU3RyaW5nIGdldFRleHQoKTsKLSAgICBSZWN0YW5nbGUgZ2V0VGV4dEJvdW5kcygpOwotICAgIHZvaWQgc2V0VGV4dEJvdW5kcyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCk7Ci0KLSAgICBTdHJpbmcgZ2V0U2hvcnRjdXQoKTsKLSAgICBSZWN0YW5nbGUgZ2V0U2hvcnRjdXRCb3VuZHMoKTsKLSAgICB2b2lkIHNldFNob3J0Y3V0Qm91bmRzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoKTsKLQotICAgIFJlY3RhbmdsZSBnZXRJdGVtQm91bmRzKCk7Ci0gICAgdm9pZCBzZXRJdGVtQm91bmRzKGludCB4LCBpbnQgeSwgaW50IHcsIGludCBoKTsKLQotICAgIGJvb2xlYW4gaXNNZW51KCk7Ci0gICAgYm9vbGVhbiBpc0NoZWNrZWQoKTsKLSAgICBib29sZWFuIGlzRW5hYmxlZCgpOwotCi0gICAgYm9vbGVhbiBpc0NoZWNrQm94KCk7Ci0gICAgYm9vbGVhbiBpc1NlcGFyYXRvcigpOwotCi0gICAgRGltZW5zaW9uIGdldE1lbnVTaXplKCk7Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9zdGF0ZS9NZW51U3RhdGUuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3N0YXRlL01lbnVTdGF0ZS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1NjRhNDlhLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3N0YXRlL01lbnVTdGF0ZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDYgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LnN0YXRlOwotCi1pbXBvcnQgamF2YS5hd3QuRm9udDsKLWltcG9ydCBqYXZhLmF3dC5Gb250TWV0cmljczsKLWltcG9ydCBqYXZhLmF3dC5Qb2ludDsKLQotLyoqCi0gKiBTdGF0ZSBvZiBwb3AtdXAgb3IgZHJvcC1kb3duIG1lbnUKLSAqLwotCi1wdWJsaWMgaW50ZXJmYWNlIE1lbnVTdGF0ZSB7Ci0gICAgaW50IGdldFdpZHRoKCk7Ci0gICAgaW50IGdldEhlaWdodCgpOwotICAgIFBvaW50IGdldExvY2F0aW9uKCk7Ci0KLSAgICB2b2lkIHNldFNpemUoaW50IHcsIGludCBoKTsKLQotICAgIEZvbnQgZ2V0Rm9udCgpOwotICAgIGJvb2xlYW4gaXNGb250U2V0KCk7Ci0gICAgRm9udE1ldHJpY3MgZ2V0Rm9udE1ldHJpY3MoRm9udCBmKTsKLQotICAgIGludCBnZXRJdGVtQ291bnQoKTsKLSAgICBpbnQgZ2V0U2VsZWN0ZWRJdGVtSW5kZXgoKTsKLQotICAgIE1lbnVJdGVtU3RhdGUgZ2V0SXRlbShpbnQgaW5kZXgpOwotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvc3RhdGUvU3RhdGUuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3N0YXRlL1N0YXRlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDRiODcwNmQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvc3RhdGUvU3RhdGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDU1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC5zdGF0ZTsKLQotaW1wb3J0IGphdmEuYXd0LkNvbG9yOwotaW1wb3J0IGphdmEuYXd0LkRpbWVuc2lvbjsKLWltcG9ydCBqYXZhLmF3dC5Gb250OwotaW1wb3J0IGphdmEuYXd0LkZvbnRNZXRyaWNzOwotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLQotLyoqCi0gKiBTdGF0ZSBvZiB0aGUgY29tcG9uZW50Ci0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgU3RhdGUgewotCi0gICAgYm9vbGVhbiBpc0VuYWJsZWQoKTsKLSAgICBib29sZWFuIGlzVmlzaWJsZSgpOwotICAgIGJvb2xlYW4gaXNGb2N1c2VkKCk7Ci0KLSAgICBGb250IGdldEZvbnQoKTsKLSAgICBib29sZWFuIGlzRm9udFNldCgpOwotICAgIEZvbnRNZXRyaWNzIGdldEZvbnRNZXRyaWNzKCk7Ci0KLSAgICBDb2xvciBnZXRCYWNrZ3JvdW5kKCk7Ci0gICAgYm9vbGVhbiBpc0JhY2tncm91bmRTZXQoKTsKLQotICAgIENvbG9yIGdldFRleHRDb2xvcigpOwotICAgIGJvb2xlYW4gaXNUZXh0Q29sb3JTZXQoKTsKLQotICAgIFJlY3RhbmdsZSBnZXRCb3VuZHMoKTsKLSAgICBEaW1lbnNpb24gZ2V0U2l6ZSgpOwotCi0gICAgRGltZW5zaW9uIGdldERlZmF1bHRNaW5pbXVtU2l6ZSgpOwotICAgIHZvaWQgc2V0RGVmYXVsdE1pbmltdW1TaXplKERpbWVuc2lvbiBzaXplKTsKLQotICAgIGxvbmcgZ2V0V2luZG93SWQoKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9DcmVhdGlvblBhcmFtcy5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL0NyZWF0aW9uUGFyYW1zLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDYzYzU4MWQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL0NyZWF0aW9uUGFyYW1zLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMzMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEbWl0cnkgQS4gRHVybmV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGs7Ci0KLS8qKgotICogVGhpcyBjbGFzcyBkZXNjcmliZXMgY3Jvc3MtcGxhdGZvcm0gTmF0aXZlV2luZG93IGNyZWF0aW9uIHBhcmFtcwotICogU2VlIGFsc28gV2luZG93RmFjdG9yeS5jcmVhdGVXaW5kb3cKLSAqLwotcHVibGljIGNsYXNzIENyZWF0aW9uUGFyYW1zIHsKLSAgICAvKioKLSAgICAgKiBJbml0aWFsIHN0YXRlIGlzIG1heGltaXplZCB2ZXJ0aWNhbHkKLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgbG9uZyBNQVhJTUlaRURfVkVSVCA9IDE7Ci0gICAgLyoqCi0gICAgICogSW5pdGlhbCBzdGF0ZSBpcyBtYXhpbWl6ZWQgaG9yaXpvbnRhbHkKLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgbG9uZyBNQVhJTUlaRURfSE9SSVogPSAyOwotICAgIC8qKgotICAgICAqIEluaXRpYWwgc3RhdGUgaXMgbWF4aW1pemVkIGJvdGgKLSAgICAgKiBob3Jpem9udGFseSBhbmQgdmVydGljYWx5Ci0gICAgICovCi0gICAgcHVibGljIGZpbmFsIGxvbmcgTUFYSU1JWkVEID0gMzsKLQotICAgIC8qKgotICAgICAqIFRoZSB0b3AtbGV2ZWwgd2luZG93IHRoYXQgaGFzIGFsbCBwb3NzaWJsZSBkZWNvcmF0aW9ucywKLSAgICAgKiBoYXMgbm8gb3duZXIgYW5kIGlzIGRpc3BsYXllZCBpbiB0YXNrYmFyCi0gICAgICovCi0gICAgcHVibGljIGZpbmFsIHN0YXRpYyBpbnQgREVDT1JfVFlQRV9GUkFNRSA9IDE7Ci0gICAgLyoqCi0gICAgICogVGhlIGRpYWxvZyB3aW5kb3cKLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgc3RhdGljIGludCBERUNPUl9UWVBFX0RJQUxPRyA9IDI7Ci0gICAgLyoqCi0gICAgICogVGhlIHRyYW5zaWVudCB1bmRlY29yYXRlZCBwb3AtdXAgd2luZG93Ci0gICAgICovCi0gICAgcHVibGljIGZpbmFsIHN0YXRpYyBpbnQgREVDT1JfVFlQRV9QT1BVUCA9IDM7Ci0gICAgLyoqCi0gICAgICogVGhlIHVuZGVjb3JhZGVkIHBvcC11cCB3aW5kb3cKLSAgICAgKi8KLSAgICBwdWJsaWMgZmluYWwgc3RhdGljIGludCBERUNPUl9UWVBFX1VOREVDT1IgPSA0OwotICAgIC8qKgotICAgICAqIE5vbi1NREkgY2hpbGQgd2luZG93Ci0gICAgICovCi0gICAgcHVibGljIGZpbmFsIHN0YXRpYyBpbnQgREVDT1JfVFlQRV9OT05FID0gMDsKLQotICAgIC8qKgotICAgICAqIEluaXRpYWwgeC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHggPSAwOwotICAgIC8qKgotICAgICAqIEluaXRpYWwgeS4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IHkgPSAwOwotICAgIC8qKgotICAgICAqIEluaXRpYWwgd2lkdGguCi0gICAgICovCi0gICAgcHVibGljIGludCB3ID0gMTsKLSAgICAvKioKLSAgICAgKiBJbml0aWFsIGhlaWdodC4KLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGggPSAxOwotICAgIC8qKgotICAgICAqIFRoZSBkZWNvcmF0aW9uIHR5cGUgb2YgdGhlIHRvcC1sZXZlbCB3aW5kb3cuIFRoZSBwb3NzaWJsZSB2YWx1ZXMgYXJlOgotICAgICAqIERFQ09SX1RZUEVfRlJBTUUsIERFQ09SX1RZUEVfRElBTE9HLCBERUNPUl9UWVBFX1BPUFVQIGFuZCBERUNPUl9UWVBFX1VOREVDT1IKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGRlY29yVHlwZSA9IERFQ09SX1RZUEVfTk9ORTsKLSAgICAvKioKLSAgICAgKiBXaW5kb3cgaXMgY2hpbGQgb2YgcGFyZW50LCBvdGhlcndpc2UgaXQncwotICAgICAqIHRvcGxldmVsKGNoaWxkIG9mIGRlc2t0b3ApIHdpbmRvdyBvd25lZCBieSBwYXJlbnQuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gY2hpbGQgPSBmYWxzZTsKLSAgICAvKioKLSAgICAgKiBXaW5kb3cgaXMgcmVzaXphYmxlCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gcmVzaXphYmxlID0gdHJ1ZTsKLSAgICAvKioKLSAgICAgKiBUaGUgd2luZG93IGhhcyBubyBkZWNvcmF0aW9ucwotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIHVuZGVjb3JhdGVkID0gZmFsc2U7Ci0gICAgLyoqCi0gICAgICogSW5pdGlhbCB2aXNpYmlsaXR5IHN0YXRlLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIHZpc2libGUgPSBmYWxzZTsKLSAgICAvKioKLSAgICAgKiBXaW5kb3cgaXMgQUxXQVlTIHRvcG1vc3QgaW4gWiBvcmRlci4KLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiB0b3Btb3N0ID0gZmFsc2U7Ci0gICAgLyoqCi0gICAgICogV2luZG93IGlzIGRpc2FibGVkLgotICAgICAqLwotICAgIHB1YmxpYyBib29sZWFuIGRpc2FibGVkID0gZmFsc2U7Ci0gICAgLyoqCi0gICAgICogV2luZG93IGluaXRpYWxseSBpY29uaWZpZWQuCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gaWNvbmlmaWVkID0gZmFsc2U7Ci0gICAgLyoqCi0gICAgICogQml0d2lzZSBPUiBvZiBNQVhJTUlaRURfKiBjb25zdGFudHMuCi0gICAgICogTWVhbnMgaWYgd2luZG93IGlzIGluaXRpYWxseSBtYXhpbWl6ZWQuCi0gICAgICovCi0gICAgcHVibGljIGludCBtYXhpbWl6ZWRTdGF0ZSA9IDA7Ci0gICAgLyoqCi0gICAgICogVGVsbHMgdGhhdCB3aW5kb3cgcG9zaXRpb24gc2hvdWxkIGJlIGRldGVybWluZWQgYnkgbmF0aXZlIHdpbmRvd2luZyBzeXN0ZW0gCi0gICAgICovCi0gICAgcHVibGljIGJvb2xlYW4gbG9jYXRpb25CeVBsYXRmb3JtID0gZmFsc2U7Ci0gICAgLyoqCi0gICAgICogSWQgb2YgcGFyZW50IG9yIG93bmVyIHdpbmRvdywgc2VlIGNoaWxkIGZpZWxkCi0gICAgICogRm9yIG5vbi1jaGlsZCB3aW5kb3cgd2l0aG91dCBvd25lciBlcXVhbHMgMC4KLSAgICAgKi8KLSAgICBwdWJsaWMgbG9uZyBwYXJlbnRJZCA9IDA7Ci0gICAgLyoqCi0gICAgICogTmFtZSB3aWNoIGlzIGRpc3BsYXllZCBvbiB0aXRsZWJhciwgdGFza2JhciBhbmQgdmlzaWJsZQotICAgICAqIGZvciBzeXN0ZW0gcmVxdWVzdHMuCi0gICAgICovCi0gICAgcHVibGljIFN0cmluZyBuYW1lID0gbnVsbDsKLX0KXCBObyBuZXdsaW5lIGF0IGVuZCBvZiBmaWxlCmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvQ3Vyc29yRmFjdG9yeS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL0N1cnNvckZhY3RvcnkuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzVlN2QzMy4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvQ3Vyc29yRmFjdG9yeS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsODUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEbWl0cnkgQS4gRHVybmV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGs7Ci0KLWltcG9ydCBqYXZhLmF3dC5EaW1lbnNpb247Ci1pbXBvcnQgamF2YS5hd3QuSW1hZ2U7Ci0KLS8qKgotICogUHJvdmlkZXMgZmFjdG9yeSBmb3IgTmF0aXZlQ3Vyc29yCi0gKi8KLXB1YmxpYyBhYnN0cmFjdCBjbGFzcyBDdXJzb3JGYWN0b3J5IHsKLSAgICBwcm90ZWN0ZWQgTmF0aXZlQ3Vyc29yW10gc3lzdGVtQ3Vyc29ycyA9IHsKLSAgICAgICAgICAgIG51bGwsIG51bGwsIG51bGwsIG51bGwsCi0gICAgICAgICAgICBudWxsLCBudWxsLCBudWxsLCBudWxsLAotICAgICAgICAgICAgbnVsbCwgbnVsbCwgbnVsbCwgbnVsbCwKLSAgICAgICAgICAgIG51bGwsIG51bGwsCi0gICAgfTsKLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGFuZCByZXR1cm5zIE5hdGl2ZUN1cnNvciBmb3IgcHJlZGVmaW5lZAotICAgICAqIEphdmEgQ3Vyc29yCi0gICAgICoKLSAgICAgKiBAcGFyYW0gdHlwZSAtIHR5cGUgb2YgcHJlZGVmaW5lZCBKYXZhIEN1cnNvcgotICAgICAqIEByZXR1cm4gY3JlYXRlZCBjdXJzb3IKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgTmF0aXZlQ3Vyc29yIGNyZWF0ZUN1cnNvcihpbnQgdHlwZSk7Ci0KLSAgICAvKioKLSAgICAgKiBHZXRzIGEgY2FjaGVkIGluc3RhbmNlIG9mIHN5c3RlbShwcmVkZWZpbmVkKSBuYXRpdmUgY3Vyc29yCi0gICAgICogb3IgY3JlYXRlcyBhIG5ldyBvbmUuIFRoaXMgaXMgYSBwbGF0Zm9ybS1pbmRlcGVuZGVudCBtZXRob2QuCi0gICAgICoKLSAgICAgKiBAcGFyYW0gdHlwZSAtIHR5cGUgb2YgcHJlZGVmaW5lZCBKYXZhIEN1cnNvcgotICAgICAqIEByZXR1cm4gY3JlYXRlZCBjdXJzb3IKLSAgICAgKi8KLSAgICBwdWJsaWMgTmF0aXZlQ3Vyc29yIGdldEN1cnNvcihpbnQgdHlwZSkgewotICAgICAgICBpZiAodHlwZSA+PSAwICYmIHR5cGUgPCBzeXN0ZW1DdXJzb3JzLmxlbmd0aCkgewotICAgICAgICAgICAgTmF0aXZlQ3Vyc29yIGN1cnNvciA9IHN5c3RlbUN1cnNvcnNbdHlwZV07Ci0gICAgICAgICAgICBpZiAoY3Vyc29yID09IG51bGwpIHsKLSAgICAgICAgICAgICAgICBjdXJzb3IgPSBjcmVhdGVDdXJzb3IodHlwZSk7Ci0gICAgICAgICAgICAgICAgc3lzdGVtQ3Vyc29yc1t0eXBlXSA9IGN1cnNvcjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBjdXJzb3I7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYW5kIHJldHVybnMgY3VzdG9tIE5hdGl2ZUN1cnNvciBmcm9tIGltYWdlCi0gICAgICoKLSAgICAgKiBAcGFyYW0gaW1nIC0gaW1hZ2Uoc291cmNlKSB0byBjcmVhdGUgY3Vyc29yIGZyb20KLSAgICAgKiBAcGFyYW0geEhvdFNwb3QgLSB4IGNvb3JkaW5hdGUgb2YgdGhlIGhvdHNwb3QgcmVsYXRpdmUgdG8gdGhlIHNvdXJjZSdzIG9yaWdpbgotICAgICAqIEBwYXJhbSB5SG90U3BvdCAtIHkgY29vcmRpbmF0ZSBvZiB0aGUgaG90c3BvdCByZWxhdGl2ZSB0byB0aGUgc291cmNlJ3Mgb3JpZ2luCi0gICAgICogQHJldHVybiBjcmVhdGVkIGN1cnNvcgotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBOYXRpdmVDdXJzb3IgY3JlYXRlQ3VzdG9tQ3Vyc29yKEltYWdlIGltZywgaW50IHhIb3RTcG90LCBpbnQgeUhvdFNwb3QpOwotCi0gICAgLyoqCi0gICAgICogUXVlcnkgbmF0aXZlIHN5c3RlbSBmb3IgdGhlIGJlc3QgY3Vyc29yIHNpemUgY2xvc2VzdCB0byBzcGVjaWZpZWQgZGltZW5zaW9ucwotICAgICAqIEBwYXJhbSBwcmVmV2lkdGggLSBwcmVmZXJyZWQgd2lkdGgKLSAgICAgKiBAcGFyYW0gcHJlZkhlaWdodCAtIHByZWZlcnJlZCBoZWlnaHQKLSAgICAgKiBAcmV0dXJuIGNsb3Nlc3Qgc3VwcG9ydGVkIGRpbWVuc2lvbnMgdG8gb25lcyBzcGVjaWZpZWQKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgRGltZW5zaW9uIGdldEJlc3RDdXJzb3JTaXplKGludCBwcmVmV2lkdGgsIGludCBwcmVmSGVpZ2h0KTsKLQotICAgIC8qKgotICAgICAqIEByZXR1cm4gbWF4aW11bSBudW1iZXIgb2YgY29sb3JzIHN1cHBvcnRlZCBieSBjdXN0b20gY3Vyc29ycwotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBpbnQgZ2V0TWF4aW11bUN1cnNvckNvbG9ycygpOwotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL0dyYXBoaWNzRmFjdG9yeS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL0dyYXBoaWNzRmFjdG9yeS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwZDdjODRmLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9HcmFwaGljc0ZhY3RvcnkuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDgyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292LCBBbGV4ZXkgQS4gUGV0cmVua28sIE9sZWcgVi4gS2hhc2NoYW5za3kKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0azsKLQotaW1wb3J0IGphdmEuYXd0LkZvbnQ7Ci1pbXBvcnQgamF2YS5hd3QuRm9udE1ldHJpY3M7Ci1pbXBvcnQgamF2YS5hd3QuR3JhcGhpY3MyRDsKLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljc0Vudmlyb25tZW50OwotaW1wb3J0IGphdmEuYXd0LnBlZXIuRm9udFBlZXI7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5NdWx0aVJlY3RBcmVhOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuZm9udC5Gb250TWFuYWdlcjsKLQotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuQ2FudmFzOwotaW1wb3J0IGFuZHJvaWQuZ3JhcGhpY3MuUGFpbnQ7Ci0KLQotLyoqCi0gKiBHcmFwaGljc0ZhY3RvcnkgaW50ZXJmYWNlIGRlZmluZXMgbWV0aG9kcyBmb3IgR3JhcGhpY3MyRCAKLSAqIGFuZCBmb250IHN0dWZmIGluc3RhbmNlcyBmYWN0b3JpZXMuCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgR3JhcGhpY3NGYWN0b3J5IHsKLSAgICBzdGF0aWMgZmluYWwgRm9udE1ldHJpY3MgY2FjaGVGTVtdID0gIG5ldyBGb250TWV0cmljc1sxMF07Ci0gICAgCi0gICAgLyoqCi0gICAgICogVGhpcyBtZXRob2QgY3JlYXRlcyBHcmFwaGljczJEIGluc3RhbmNlIGZvciBzcGVjaWZpZWQgbmF0aXZlIHdpbmRvdy4KLSAgICAgKiAgCi0gICAgICogQHBhcmFtIHdpbiBOYXRpdmUgd2luZG93IHRvIGRyYXcKLSAgICAgKiBAcGFyYW0gdHJhbnNsYXRlWCBUcmFuc2xhdGlvbiBhbG9uZyBYIGF4aXMKLSAgICAgKiBAcGFyYW0gdHJhbnNsYXRlWSBUcmFuc2xhdGlvbiBhbG9uZyBZIGF4aXMKLSAgICAgKiBAcGFyYW0gY2xpcCBDbGlwcGluZyBhcmVhIGZvciBhIG5ldyBHcmFwaGljczJEIGluc3RhbmNlCi0gICAgICogQHJldHVybiBOZXcgR3JhcGhpY3MyRCBpbnN0YW5jZSBmb3Igc3BlY2lmaWVkIG5hdGl2ZSB3aW5kb3cKLSAgICAgKiBAZGVwcmVjYXRlZAotICAgICAqLwotICAgIEBEZXByZWNhdGVkCi0gICAgR3JhcGhpY3MyRCBnZXRHcmFwaGljczJEKE5hdGl2ZVdpbmRvdyB3aW4sIGludCB0cmFuc2xhdGVYLCBpbnQgdHJhbnNsYXRlWSwgTXVsdGlSZWN0QXJlYSBjbGlwKTsKLQotICAgIC8qKgotICAgICAqIFRoaXMgbWV0aG9kIGNyZWF0ZXMgR3JhcGhpY3MyRCBpbnN0YW5jZSBmb3Igc3BlY2lmaWVkIG5hdGl2ZSB3aW5kb3cuCi0gICAgICogIAotICAgICAqIEBwYXJhbSB3aW4gTmF0aXZlIHdpbmRvdyB0byBkcmF3Ci0gICAgICogQHBhcmFtIHRyYW5zbGF0ZVggVHJhbnNsYXRpb24gYWxvbmcgWCBheGlzCi0gICAgICogQHBhcmFtIHRyYW5zbGF0ZVkgVHJhbnNsYXRpb24gYWxvbmcgWSBheGlzCi0gICAgICogQHBhcmFtIHdpZHRoIFdpZHRoIG9mIGRyYXdpbmcgYXJlYQotICAgICAqIEBwYXJhbSBoZWlnaHQgSGVpZ2h0IG9mIGRyYXdpbmcgYXJlYQotICAgICAqIEByZXR1cm4gTmV3IEdyYXBoaWNzMkQgaW5zdGFuY2UgZm9yIHNwZWNpZmllZCBuYXRpdmUgd2luZG93Ci0gICAgICovCi0gICAgR3JhcGhpY3MyRCBnZXRHcmFwaGljczJEKE5hdGl2ZVdpbmRvdyB3aW4sIGludCB0cmFuc2xhdGVYLCBpbnQgdHJhbnNsYXRlWSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KTsKLSAgICAvLyA/Pz9BV1Q6IG5vdCBzdGFuZGFyZCBoYXJtb255Ci0gICAgR3JhcGhpY3MyRCBnZXRHcmFwaGljczJEKENhbnZhcyBjLCBQYWludCBwKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBDcmVhdGVzIGluc3RhbmNlIG9mIEdyYXBoaWNzRW52aXJvbm1lbnQgZm9yIHNwZWNpZmllZCBXaW5kb3dGYWN0b3J5Ci0gICAgICogIAotICAgICAqIEBwYXJhbSB3ZiBXaW5kb3dGYWN0b3J5Ci0gICAgICogQHJldHVybiBOZXcgaW5zdGFuY2Ugb2YgR3JhcGhpY3NFbnZpcm9ubWVudAotICAgICAqLwotICAgIEdyYXBoaWNzRW52aXJvbm1lbnQgY3JlYXRlR3JhcGhpY3NFbnZpcm9ubWVudChXaW5kb3dGYWN0b3J5IHdmKTsKLSAgICAKLSAgICAvLyBGb250IG1ldGhvZHMKLSAgICBGb250TWV0cmljcyBnZXRGb250TWV0cmljcyhGb250IGZvbnQpOwotICAgIEZvbnRNYW5hZ2VyIGdldEZvbnRNYW5hZ2VyKCk7Ci0gICAgRm9udFBlZXIgZ2V0Rm9udFBlZXIoRm9udCBmb250KTsKLSAgICBGb250IGVtYmVkRm9udChTdHJpbmcgZm9udEZpbGVQYXRoKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9LZXlJbmZvLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvS2V5SW5mby5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxZjhhMjlhLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9LZXlJbmZvLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1MyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrOwotCi1pbXBvcnQgamF2YS5hd3QuZXZlbnQuS2V5RXZlbnQ7Ci0KLS8qKgotICogS2V5c3Ryb2tlIGluZm9ybWF0aW9uCi0gKi8KLQotcHVibGljIGZpbmFsIGNsYXNzIEtleUluZm8gewotCi0gICAgcHVibGljIGludCB2S2V5OwotICAgIHB1YmxpYyBpbnQga2V5TG9jYXRpb247Ci0gICAgcHVibGljIGZpbmFsIFN0cmluZ0J1ZmZlciBrZXlDaGFyczsKLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IERFRkFVTFRfVktFWSA9IEtleUV2ZW50LlZLX1VOREVGSU5FRDsKLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBERUZBVUxUX0xPQ0FUSU9OID0gS2V5RXZlbnQuS0VZX0xPQ0FUSU9OX1NUQU5EQVJEOwotCi0gICAgcHVibGljIEtleUluZm8oKSB7Ci0gICAgICAgIHZLZXkgPSBERUZBVUxUX1ZLRVk7Ci0gICAgICAgIGtleUxvY2F0aW9uID0gREVGQVVMVF9MT0NBVElPTjsKLSAgICAgICAga2V5Q2hhcnMgPSBuZXcgU3RyaW5nQnVmZmVyKCk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0S2V5Q2hhcnMoY2hhciBjaCkgewotICAgICAgICBrZXlDaGFycy5zZXRMZW5ndGgoMCk7Ci0gICAgICAgIGtleUNoYXJzLmFwcGVuZChjaCk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0S2V5Q2hhcnMoU3RyaW5nQnVmZmVyIHNiKSB7Ci0gICAgICAgIGtleUNoYXJzLnNldExlbmd0aCgwKTsKLSAgICAgICAga2V5Q2hhcnMuYXBwZW5kKHNiKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlQ3Vyc29yLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlQ3Vyc29yLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJjNmViMWUuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZUN1cnNvci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNDUgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBEbWl0cnkgQS4gRHVybmV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGs7Ci0KLS8qKgotICogVGhlIGludGVyZmFjZSBwcm92aWRlcyBhY2Nlc3MgdG8gcGxhdGZvcm0gZGVwZW5kZW50IGZ1bmN0aW9uYWxpdHkKLSAqIGZvciB0aGUgY2xhc3MgamF2YS5hd3QuQ3Vyc29yLgotICovCi1wdWJsaWMgaW50ZXJmYWNlIE5hdGl2ZUN1cnNvciB7Ci0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgY3VycmVudCBjdXJzb3Igc2hhcGUKLSAgICAgKiB0byB0aGlzIGN1cnNvciB3aGVuIGEgcG9pbnRlciBpcyBpbnNpZGUKLSAgICAgKiBAcGFyYW0gd2luSUQgLSB3aW5kb3coY3VycmVudGx5IHVzZWQgb25seSBvbiBYMTEpCi0gICAgICovCi0gICAgdm9pZCBzZXRDdXJzb3IobG9uZyB3aW5JRCk7Ci0gICAgLyoqCi0gICAgICogRGVzdHJveXMgdGhlIG5hdGl2ZSByZXNvdXJjZSBhc3NvY2lhdGVkIHdpdGgKLSAgICAgKiB0aGlzIGN1cnNvcgotICAgICAqLwotICAgIHZvaWQgZGVzdHJveUN1cnNvcigpOwotCi0gICAgLyoqCi0gICAgICogQHJldHVybiBOYXRpdmUgaGFuZGxlIGFzc29jaWF0ZWQgd2l0aCB0aGlzIGN1cnNvcgotICAgICAqLwotICAgIGxvbmcgZ2V0SWQoKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZUV2ZW50LmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlRXZlbnQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTQ3MWMxYS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlRXZlbnQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI2OCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pa2hhaWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrOwotCi1pbXBvcnQgamF2YS5hd3QuSW5zZXRzOwotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLWltcG9ydCBqYXZhLmF3dC5Qb2ludDsKLWltcG9ydCBqYXZhLmF3dC5ldmVudC5LZXlFdmVudDsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuTXVsdGlSZWN0QXJlYTsKLQotCi0vKioKLSAqIFRoZSBpbnRlcmZhY2UgZGVzY3JpYmluZyBjcm9zcy1wbGF0Zm9ybSB0cmFuc2xhdGlvbiBvZiBzeXN0ZW0KLSAqIG1lc3NhZ2VzLgotICoKLSAqIDxwLz5Tb21lIG1lc3NhZ2VzIGNhbiBhcHBlYXIgb25seSBvbiBzcGVjaWZpYyBwbGF0Zm9ybSwKLSAqIGJ1dCB0aGV5IHN0aWxsIGNhbiBoYXZlIGNyb3NzLXBsYXRmb3JtIGludGVycHJldGF0aW9uIGlmIHRoZQotICogYXBwbGljYXRpb24gc2hvdWxkIGJlIGF3YXJlIG9mIHRoZW0gYW5kIGNhbiByZWFjdCB1c2luZwotICogY3Jvc3MtcGxhdGZvcm0gQVBJLgotICoKLSAqLwotcHVibGljIGFic3RyYWN0IGNsYXNzIE5hdGl2ZUV2ZW50IHsKLQotICAgIC8qKgotICAgICAqIE1lc3NhZ2UgaGFzIG5vIGNvbW1vbiBjcm9zcy1wbGF0Zm9ybQotICAgICAqIGludGVycHJldGF0aW9uIGFuZCBzaG91bGQgYmUgc2tpcHBlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJRF9QTEFURk9STSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBXaW5kb3cgYm91bmRzIGhhdmUgY2hhbmdlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJRF9CT1VORFNfQ0hBTkdFRCA9IC0xOwotCi0gICAgLyoqCi0gICAgICogV2luZG93IGRlY29yYXRpb24gc2l6ZSBoYXMgY2hhbmdlZC4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJRF9JTlNFVFNfQ0hBTkdFRCA9IC0yOwotCi0gICAgLyoqCi0gICAgICogV2luZG93IHdhcyBqdXN0IGNyZWF0ZWQgKFdNX0NSRUFURSBvbiBXaW5kb3dzKQotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IElEX0NSRUFURUQgPSAtMzsKLQotICAgIC8qKgotICAgICAqIE1vdXNlIGdyYWIgd2FzIGNhbmNlbGVkIGJ5IHRoZSBuYXRpdmUgc3lzdGVtCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSURfTU9VU0VfR1JBQl9DQU5DRUxFRCA9IC00OwotCi0gICAgLyoqCi0gICAgICogU3lzdGVtIGNvbG9yIHNjaGVtZSBvciB2aXN1YWwgdGhlbWUgd2FzIGNoYW5nZWQKLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBJRF9USEVNRV9DSEFOR0VEID0gLTU7Ci0KLSAgICBwcm90ZWN0ZWQgbG9uZyB3aW5kb3dJZDsKLSAgICBwcm90ZWN0ZWQgaW50IGV2ZW50SWQ7Ci0gICAgcHJvdGVjdGVkIGxvbmcgb3RoZXJXaW5kb3dJZDsKLQotICAgIHByb3RlY3RlZCBQb2ludCBzY3JlZW5Qb3M7Ci0gICAgcHJvdGVjdGVkIFBvaW50IGxvY2FsUG9zOwotICAgIHByb3RlY3RlZCBSZWN0YW5nbGUgd2luZG93UmVjdDsKLQotICAgIHByb3RlY3RlZCBpbnQgbW9kaWZpZXJzOwotICAgIHByb3RlY3RlZCBpbnQgbW91c2VCdXR0b247Ci0gICAgcHJvdGVjdGVkIGludCB3aGVlbFJvdGF0aW9uOwotCi0gICAgcHJvdGVjdGVkIEtleUluZm8ga2V5SW5mbyA9IG5ldyBLZXlJbmZvKCk7Ci0KLSAgICBwcm90ZWN0ZWQgaW50IHdpbmRvd1N0YXRlID0gLTE7Ci0gICAgcHJvdGVjdGVkIGxvbmcgdGltZTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIHN5c3RlbSB3aW5kb3cgaWQgb2YgdGhlIGV2ZW50IHJlY2lwaWVudC4KLSAgICAgKiBAcmV0dXJuIEhXTkQgb24gV2luZG93cywgeHdpbmRub3cgb24gWAotICAgICAqLwotICAgIHB1YmxpYyBsb25nIGdldFdpbmRvd0lkKCkgewotICAgICAgICByZXR1cm4gd2luZG93SWQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBjcm9zcy1wbGF0Zm9ybSBldmVudCBpZAotICAgICAqIHNob3VsZCBiZSBvbmUgb2YgSURfKiBjb25zdGFudHMgb3IKLSAgICAgKiBpZCBjb25zdGFudHMgZnJvbSBqYXZhLmF3dC5BV1RFdmVudCBzdWJjbGFzZXNzCi0gICAgICogQHJldHVybiBjcm9zcy1wbGF0Zm9ybSBldmVudCBpZAotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0RXZlbnRJZCgpIHsKLSAgICAgICAgcmV0dXJuIGV2ZW50SWQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgcG9zaXRpb24gb2YgY3Vyc29yIHdoZW4gZXZlbnQgb2NjdXJlZCByZWxhdGl2ZSB0bwotICAgICAqIHRvcC1sZWZ0IGNvcm5lciBvZiByZWNpcGllbnQgd2luZG93Ci0gICAgICogQHJldHVybiBwb3NpdGlvbiBvZiBjdXJzb3IgaW4gbG9jYWwgY29vcmRpbmF0ZXMKLSAgICAgKi8KLSAgICBwdWJsaWMgUG9pbnQgZ2V0TG9jYWxQb3MoKSB7Ci0gICAgICAgIHJldHVybiBsb2NhbFBvczsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBwb3NpdGlvbiBvZiBjdXJzb3Igd2hlbiBldmVudCBvY2N1cmVkCi0gICAgICogaW4gc2NyZWVuIGNvb3JkaW5hdGVzLgotICAgICAqIEByZXR1cm4gcG9zaXRpb24gb2YgY3Vyc29yIGluIHNjcmVlbiBjb29yZGluYXRlcwotICAgICAqLwotICAgIHB1YmxpYyBQb2ludCBnZXRTY3JlZW5Qb3MoKSB7Ci0gICAgICAgIHJldHVybiBzY3JlZW5Qb3M7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIHJlY2lwaWVudCB3aW5kb3cgYm91bmRzIHdoZW4gdGhlIGV2ZW50IG9jY3VyZWQKLSAgICAgKiBAcmV0dXJuIHdpbmRvdyBib3VuZHMKLSAgICAgKi8KLSAgICBwdWJsaWMgUmVjdGFuZ2xlIGdldFdpbmRvd1JlY3QoKSB7Ci0gICAgICAgIHJldHVybiB3aW5kb3dSZWN0OwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIHN0YXRlIG9mIGtleWJvYXJkIGFuZCBtb3VzZSBidXR0b25zIHdoZW4gdGhlIGV2ZW50Ci0gICAgICogb2NjdXJlZCBpZiBldmVudCBmcm9tIG1vdXNlIG9yIGtleWJvYXJkLCBmb3Igb3RoZXIgZXZlbnRzIGNhbgotICAgICAqIHJldHVybiBqdW5rIHZhbHVlcy4gVGhlIHZhbHVlIGlzIGJpdHdpc2UgT1Igb2YKLSAgICAgKiBqYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50ICpfRE9XTiBjb25zdGFudHMuCi0gICAgICoKLSAgICAgKiBNZXRob2QgaXMgYXdhcmUgb2Ygc3lzdGVtIG1vdXNlIGJ1dHRvbiBzd2FwIGZvciBsZWZ0LWhhbmQKLSAgICAgKiBtb3VzZSBhbmQgcmV0dXJuIHN3YXBwZWQgdmFsdWVzLgotICAgICAqIEByZXR1cm4gYml0d2lzZSBPUiBvZiBqYXZhLmF3dC5ldmVudC5JbnB1dEV2ZW50ICpfRE9XTiBjb25zdGFudHMKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldElucHV0TW9kaWZpZXJzKCkgewotICAgICAgICByZXR1cm4gbW9kaWZpZXJzOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIGljb25pZmllZC9tYXhpbWl6ZWQgc3RhdGUgb2YgcmVjaXBpZW50IHdpbmRvdyBpZgotICAgICAqIGV2ZW50IGlzIHN0YXRlIHJlbGF0ZWQsIGZvciBvdGhlciBldmVudHMgY2FuIGp1bmsgdmFsdWVzLgotICAgICAqIFRoZSB2YWx1ZSBoYXMgdGhlIHNhbWUgbWVhbmluZyBhcyBGcmFtZS5nZXRFeHRlbmRlZFN0YXRlCi0gICAgICogSXQncyBiaXR3aXNlIE9SIG9mIElDT05JRklFRCwgTUFYSU1JWkVEX0hPUklaLCBNQVhJTUlaRURfVkVSVAotICAgICAqIEByZXR1cm4gYml0d2lzZSBPUiBvZiBJQ09OSUZJRUQsIE1BWElNSVpFRF9IT1JJWiwgTUFYSU1JWkVEX1ZFUlQKLSAgICAgKi8KLSAgICBwdWJsaWMgaW50IGdldFdpbmRvd1N0YXRlKCkgewotICAgICAgICByZXR1cm4gd2luZG93U3RhdGU7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogVGhlIHNhbWUgbWVhbmluZyBhcyBqYXZhLmF3dC5ldmVudC5nZXRLZXlDb2RlCi0gICAgICogQHJldHVybiBqYXZhLmF3dC5ldmVudCBWS18qIGNvbnN0YW50Ci0gICAgICovCi0gICAgcHVibGljIGludCBnZXRWS2V5KCkgewotICAgICAgICByZXR1cm4gKGtleUluZm8gIT0gbnVsbCkgPyBrZXlJbmZvLnZLZXkgOiBLZXlJbmZvLkRFRkFVTFRfVktFWTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBUaGUgc2FtZSBtZWFuaW5nIGFzIGphdmEuYXd0LmV2ZW50LmdldEtleUxvY2F0aW9uCi0gICAgICogQHJldHVybiBqYXZhLmF3dC5ldmVudCBLRVlfTE9DQVRJT05fKiBjb25zdGFudAotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0S2V5TG9jYXRpb24oKSB7Ci0gICAgICAgIHJldHVybiAoa2V5SW5mbyAhPSBudWxsKSA/IGtleUluZm8ua2V5TG9jYXRpb24gOiBLZXlJbmZvLkRFRkFVTFRfTE9DQVRJT047Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJuIHRoZSBzdHJpbmcgb2YgY2hhcmFjdGVycyBhc3NvY2lhdGVkIHdpdGggdGhlIGV2ZW50Ci0gICAgICogSGFzIG1lYW5pbmcgb25seSBmb3IgS0VZX1BSRVNTRUQgYXMgc2hvdWxkIGJlIHRyYW5zbGF0ZWQgdG8KLSAgICAgKiBzZXJpZSBvZiBLRVlfVFlQRUQgZXZlbnRzLiBGb3IgZGVhZCBrZXlzIGFuZCBpbnB1dCBtZXRob2RzCi0gICAgICogb25lIGtleSBwcmVzcyBjYW4gZ2VuZXJhdGUgbXVsdGlwbGUga2V5IGNoYXJzLgotICAgICAqIEByZXR1cm4gc3RyaW5nIG9mIGNoYXJhY3RlcnMKLSAgICAgKi8KLSAgICBwdWJsaWMgU3RyaW5nQnVmZmVyIGdldEtleUNoYXJzKCkgewotICAgICAgICBpZiAoa2V5SW5mbyA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfQotICAgICAgICBpZiAoa2V5SW5mby52S2V5ID09IEtleUV2ZW50LlZLX0VOVEVSKSB7Ci0gICAgICAgICAgICBrZXlJbmZvLmtleUNoYXJzLnNldExlbmd0aCgwKTsKLSAgICAgICAgICAgIGtleUluZm8uc2V0S2V5Q2hhcnMoJ1xuJyk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGtleUluZm8ua2V5Q2hhcnM7Ci0gICAgfQotCi0gICAgcHVibGljIGNoYXIgZ2V0TGFzdENoYXIoKSB7Ci0gICAgICAgIGlmIChrZXlJbmZvID09IG51bGwgfHwga2V5SW5mby5rZXlDaGFycy5sZW5ndGgoKSA9PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gS2V5RXZlbnQuQ0hBUl9VTkRFRklORUQ7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGtleUluZm8ua2V5Q2hhcnMuY2hhckF0KGtleUluZm8ua2V5Q2hhcnMubGVuZ3RoKCktMSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbnVtYmVyIG9mIG1vdXNlIGJ1dHRvbiB3aGljaCBjaGFuZ2VkIGl0J3Mgc3RhdGUsCi0gICAgICogb3RoZXJ3aXNlIDAuCi0gICAgICogTGVmdCBidXR0b24gaXMgMSwgbWlkZGxlIGJ1dHRvbiBpcyAyLCByaWdodCBidXR0b24gaXMgMy4KLSAgICAgKgotICAgICAqIE1ldGhvZCBpcyBhd2FyZSBvZiBzeXN0ZW0gbW91c2UgYnV0dG9uIHN3YXAgZm9yIGxlZnQtaGFuZAotICAgICAqIG1vdXNlIGFuZCByZXR1cm4gc3dhcHBlZCB2YWx1ZXMuCi0gICAgICogQHJldHVybiBtb3VzZSBidXR0b24gbnVtYmVyCi0gICAgICovCi0gICAgcHVibGljIGludCBnZXRNb3VzZUJ1dHRvbigpIHsKLSAgICAgICAgcmV0dXJuIG1vdXNlQnV0dG9uOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGltZSB3aGVuIHRoZSBtZXNzYWdlIHdhcyByZWNlaXZlZAotICAgICAqIEByZXR1cm4gdGltZSBpbiBtaWxsaXNlY29uZHMKLSAgICAgKi8KLSAgICBwdWJsaWMgbG9uZyBnZXRUaW1lKCkgewotICAgICAgICByZXR1cm4gdGltZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBGb3IgdGhlIGZvY3VzIGV2ZW50IGNvbnRhaW5zIHRoZSBvcG9zaXRlIHdpbmRvdy4KLSAgICAgKiBUaGlzIG1lYW5zIGl0IGxvc3QgZm9jdXMgaWYgcmVjaXBpZW50IGdhaW5zIGl0LAotICAgICAqIG9yIHdpbGwgZ2FpbiBmb2N1cyBpZiByZWNpcGllbnQgbG9vc2VzIGl0LgotICAgICAqIEByZXR1cm4gSFdORCBvbiBXaW5kb3dzLCB4d2luZG5vdyBvbiBYCi0gICAgICovCi0gICAgcHVibGljIGxvbmcgZ2V0T3RoZXJXaW5kb3dJZCgpIHsKLSAgICAgICAgcmV0dXJuIG90aGVyV2luZG93SWQ7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgImRpcnR5IiBhcmVhIG9mIHRoZSB3aW5kb3cgYXMgc2V0IG9mIG5vbi1pbnRlcnNlY3RpbmcKLSAgICAgKiByZWN0YW5nbGVzLiBUaGlzIGFyZWEgaXMgdG8gYmUgcGFpbnRlZC4KLSAgICAgKiBAcmV0dXJuIG5vbi1lbXB0eSBhcnJheSBvZiBudWxsIGlmIGVtcHR5Ci0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IE11bHRpUmVjdEFyZWEgZ2V0Q2xpcFJlY3RzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSAiZGlydHkiIGFyZWEgb2YgdGhlIHdpbmRvdyBhcyBvbmUgcmVjdGFuZ2xlLgotICAgICAqIFRoaXMgYXJlYSBpcyB0byBiZSBwYWludGVkLgotICAgICAqIEByZXR1cm4gbm9uLW51bGwgUmVjdGFuZ2xlCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IFJlY3RhbmdsZSBnZXRDbGlwQm91bmRzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSB3aW5kb3cgaW5zZXRzLiBJbnNldHMgaXMgYXJlYSB3aGljaCBiZWxvbmdzIHRvCi0gICAgICogd2luZG93IHNvbWVob3cgYnV0IGlzIG91dHNpZGUgb2YgaXQncyBjbGllbnQgYXJlYSwKLSAgICAgKiBpdCB1c3VhbGx5IGNvbnRhaW5zIHN5c3RlbSBwcm92aWRlZCBib3JkZXIgYW5kIHRpdGxlYmFyLgotICAgICAqIEByZXR1cm4gbm9uLW51bGwgamF2YS5hd3QuSW5zZXRzCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEluc2V0cyBnZXRJbnNldHMoKTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiBldmVudCBpcyBwb3B1cCBtZW51IHRyaWdnZXIuCi0gICAgICogQHJldHVybiBib29sZWFuIGZsYWcKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgYm9vbGVhbiBnZXRUcmlnZ2VyKCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBudW1iZXIgb2YgImNsaWNrcyIgdGhlIG1vdXNlIHdoZWVsIHdhcyByb3RhdGVkLgotICAgICAqIEByZXR1cm4gbmVnYXRpdmUgdmFsdWVzIGlmIHRoZSBtb3VzZSB3aGVlbCB3YXMgcm90YXRlZCB1cC9hd2F5IGZyb20gdGhlIHVzZXIsCi0gICAgICogYW5kIHBvc2l0aXZlIHZhbHVlcyBpZiB0aGUgbW91c2Ugd2hlZWwgd2FzIHJvdGF0ZWQgZG93bi8gdG93YXJkcyB0aGUgdXNlcgotICAgICAqLwotICAgIHB1YmxpYyBpbnQgZ2V0V2hlZWxSb3RhdGlvbigpIHsKLSAgICAgICAgcmV0dXJuIHdoZWVsUm90YXRpb247Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZUV2ZW50UXVldWUuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9OYXRpdmVFdmVudFF1ZXVlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDA3MzhjZDEuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZUV2ZW50UXVldWUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDExNyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pa2hhaWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrOwotCi1pbXBvcnQgamF2YS51dGlsLkxpbmtlZExpc3Q7Ci0KLQotLyoqCi0gKiBEZXNjcmliZXMgdGhlIGNyb3NzLXBsYXRmb3JtIG5hdGl2ZSBldmVudCBxdWV1ZSBpbnRlcmZhY2UKLSAqCi0gKiA8cC8+IFRoZSBpbXBsZW1lbnRhdGlvbiBjb25zdHJ1Y3RvciBzaG91bGQgcmVtZW1iZXIgdGhyZWFkIGl0IHdhcwotICogY3JlYXRlZC4gQWxsIG90aGVyIG1ldGhvZHMgd291bGQgYmUgY2FsbGVkIG9ibHkgZnJvbSB0aGlzIHRocmVhZCwKLSAqIGV4Y2VwdCBhd2FrZSgpLgotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgTmF0aXZlRXZlbnRRdWV1ZSB7Ci0gICAgCi0gICAgcHJpdmF0ZSBTaHV0ZG93bldhdGNoZG9nIHNodXRkb3duV2F0Y2hkb2c7Ci0gICAgcHJpdmF0ZSBjbGFzcyBFdmVudE1vbml0b3Ige30KLSAgICBwcml2YXRlIGZpbmFsIE9iamVjdCBldmVudE1vbml0b3IgPSBuZXcgRXZlbnRNb25pdG9yKCk7Ci0gICAgcHJpdmF0ZSBmaW5hbCBMaW5rZWRMaXN0PE5hdGl2ZUV2ZW50PiBldmVudFF1ZXVlID0gbmV3IExpbmtlZExpc3Q8TmF0aXZlRXZlbnQ+KCk7Ci0KLSAgICBwdWJsaWMgc3RhdGljIGFic3RyYWN0IGNsYXNzIFRhc2sgewotICAgICAgICBwdWJsaWMgdm9sYXRpbGUgT2JqZWN0IHJldHVyblZhbHVlOwotCi0gICAgICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHBlcmZvcm0oKTsKLSAgICB9Ci0gICAgCi0gICAgLyoqCi0gICAgICogQmxvY2tzIGN1cnJlbnQgdGhyZWFkIHVudGlsIG5hdGl2ZSBldmVudCBxdWV1ZSBpcyBub3QgZW1wdHkKLSAgICAgKiBvciBhd2FrZW4gZnJvbSBvdGhlciB0aHJlYWQgYnkgYXdha2UoKS4KLSAgICAgKgotICAgICAqIDxwLz5TaG91bGQgYmUgY2FsbGVkIG9ubHkgb24gdHJlYWQgd2hpY2gKLSAgICAgKiB3aWxsIHByb2Nlc3MgbmF0aXZlIGV2ZW50cy4KLSAgICAgKgotICAgICAqIEByZXR1cm4gaWYgZXZlbnQgbG9vcCBzaG91bGQgYmUgc3RvcHBlZAotICAgICAqLwotICAgIHB1YmxpYyBhYnN0cmFjdCBib29sZWFuIHdhaXRFdmVudCgpOwotCi0gICAgLyoqCi0gICAgICogRGV0ZXJtaW5lcyB3aGV0aGVyIG9yIG5vdCB0aGUgbmF0aXZlIGV2ZW50IHF1ZXVlIGlzIGVtcHR5LgotICAgICAqIEFuIHF1ZXVlIGlzIGVtcHR5IGlmIGl0IGNvbnRhaW5zIG5vIG1lc3NhZ2VzIHdhaXRpbmcuCi0gICAgICoKLSAgICAgKiBAcmV0dXJuIHRydWUgaWYgdGhlIHF1ZXVlIGlzIGVtcHR5OyBmYWxzZSBvdGhlcndpc2UKLSAgICAgKi8KLSAgICBwdWJsaWMgYm9vbGVhbiBpc0VtcHR5KCkgewotICAgICAgICBzeW5jaHJvbml6ZWQoZXZlbnRRdWV1ZSkgewotICAgICAgICAgICAgcmV0dXJuIGV2ZW50UXVldWUuaXNFbXB0eSgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIE5hdGl2ZUV2ZW50IGdldE5leHRFdmVudCgpIHsKLSAgICAgICAgc3luY2hyb25pemVkIChldmVudFF1ZXVlKSB7Ci0gICAgICAgICAgICBpZiAoZXZlbnRRdWV1ZS5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgICAgICBzaHV0ZG93bldhdGNoZG9nLnNldE5hdGl2ZVF1ZXVlRW1wdHkodHJ1ZSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gZXZlbnRRdWV1ZS5yZW1vdmUoMCk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgcHJvdGVjdGVkIHZvaWQgYWRkRXZlbnQoTmF0aXZlRXZlbnQgZXZlbnQpIHsKLSAgICAgICAgc3luY2hyb25pemVkIChldmVudFF1ZXVlKSB7Ci0gICAgICAgICAgICBldmVudFF1ZXVlLmFkZChldmVudCk7Ci0gICAgICAgICAgICBzaHV0ZG93bldhdGNoZG9nLnNldE5hdGl2ZVF1ZXVlRW1wdHkoZmFsc2UpOwotICAgICAgICB9Ci0gICAgICAgIHN5bmNocm9uaXplZCAoZXZlbnRNb25pdG9yKSB7Ci0gICAgICAgICAgICBldmVudE1vbml0b3Iubm90aWZ5KCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgZmluYWwgT2JqZWN0IGdldEV2ZW50TW9uaXRvcigpIHsKLSAgICAgICAgcmV0dXJuIGV2ZW50TW9uaXRvcjsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYWJzdHJhY3Qgdm9pZCBhd2FrZSgpOwotCi0gICAgLyoqCi0gICAgICogR2V0cyBBV1Qgc3lzdGVtIHdpbmRvdyBJRC4KLSAgICAgKgotICAgICAqIEByZXR1cm4gQVdUIHN5c3RlbSB3aW5kb3cgSUQKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgbG9uZyBnZXRKYXZhV2luZG93KCk7Ci0KLSAgICAvKioKLSAgICAgKiBBZGQgTmF0aXZlRXZlbnQgdG8gdGhlIHF1ZXVlCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZGlzcGF0Y2hFdmVudCgpOwotCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgcGVyZm9ybVRhc2soVGFzayB0YXNrKTsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCB2b2lkIHBlcmZvcm1MYXRlcihUYXNrIHRhc2spOwotICAgIAotICAgIHB1YmxpYyBmaW5hbCB2b2lkIHNldFNodXRkb3duV2F0Y2hkb2coU2h1dGRvd25XYXRjaGRvZyB3YXRjaGRvZykgewotICAgICAgICBzeW5jaHJvbml6ZWQgKGV2ZW50UXVldWUpIHsKLSAgICAgICAgICAgIHNodXRkb3duV2F0Y2hkb2cgPSB3YXRjaGRvZzsKLSAgICAgICAgfQotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZUV2ZW50VGhyZWFkLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlRXZlbnRUaHJlYWQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDUwYWRkNC4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlRXZlbnRUaHJlYWQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDc4ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGs7Ci0KLQotLyoqCi0gKiBOYXRpdmVFdmVudFRocmVhZAotICovCi1wdWJsaWMgY2xhc3MgTmF0aXZlRXZlbnRUaHJlYWQgZXh0ZW5kcyBUaHJlYWQgewotICAgIAotICAgIHB1YmxpYyBpbnRlcmZhY2UgSW5pdCB7Ci0gICAgICAgIFdUSyBpbml0KCk7Ci0gICAgfQotICAgIAotICAgIE5hdGl2ZUV2ZW50UXVldWUgbmF0aXZlUXVldWU7Ci0gICAgSW5pdCBpbml0OwotICAgIAotICAgIHByaXZhdGUgV1RLIHd0azsKLSAgICAKLSAgICBwdWJsaWMgTmF0aXZlRXZlbnRUaHJlYWQoKSB7Ci0gICAgICAgIHN1cGVyKCJBV1QtTmF0aXZlRXZlbnRUaHJlYWQiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBzZXREYWVtb24odHJ1ZSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgcnVuKCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgd3RrID0gaW5pdC5pbml0KCk7Ci0gICAgICAgICAgICAgICAgbmF0aXZlUXVldWUgPSB3dGsuZ2V0TmF0aXZlRXZlbnRRdWV1ZSgpOwotICAgICAgICAgICAgfSBmaW5hbGx5IHsKLSAgICAgICAgICAgICAgICBub3RpZnlBbGwoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgcnVuTW9kYWxMb29wKCk7Ci0gICAgfQotCi0gICAgdm9pZCBydW5Nb2RhbExvb3AoKSB7Ci0gICAgICAgIHdoaWxlIChuYXRpdmVRdWV1ZS53YWl0RXZlbnQoKSkgewotICAgICAgICAgICAgbmF0aXZlUXVldWUuZGlzcGF0Y2hFdmVudCgpOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIHB1YmxpYyB2b2lkIHN0YXJ0KEluaXQgaW5pdCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIHRoaXMuaW5pdCA9IGluaXQ7Ci0gICAgICAgICAgICBzdXBlci5zdGFydCgpOwotICAgICAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgICAgICB3YWl0KCk7Ci0gICAgICAgICAgICB9IGNhdGNoIChJbnRlcnJ1cHRlZEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgcHVibGljIFdUSyBnZXRXVEsoKSB7Ci0gICAgICAgIHJldHVybiB3dGs7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZUlNLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlSU0uamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTYyNmY0YS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlSU0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEzMCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqIAotICogQGF1dGhvciBEbWl0cnkgQS4gRHVybmV2Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGs7Ci0KLWltcG9ydCBqYXZhLmF3dC5BV1RFdmVudDsKLWltcG9ydCBqYXZhLmF3dC5BV1RFeGNlcHRpb247Ci1pbXBvcnQgamF2YS5hd3QuSW1hZ2U7Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotaW1wb3J0IGphdmEuYXd0LmltLnNwaS5JbnB1dE1ldGhvZDsKLWltcG9ydCBqYXZhLmF3dC5pbS5zcGkuSW5wdXRNZXRob2RDb250ZXh0OwotaW1wb3J0IGphdmEuYXd0LmltLnNwaS5JbnB1dE1ldGhvZERlc2NyaXB0b3I7Ci1pbXBvcnQgamF2YS5sYW5nLkNoYXJhY3Rlci5TdWJzZXQ7Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLQotLyoqCi0gKiBBIGNyb3NzLXBsYXRmb3JtIGludGVyZmFjZSBmb3IgbmF0aXZlIGlucHV0Ci0gKiBtZXRob2Qgc3ViLXN5c3RlbSBmdW5jdGlvbmFsaXR5LgotICovCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgTmF0aXZlSU0gaW1wbGVtZW50cyBJbnB1dE1ldGhvZCwgSW5wdXRNZXRob2REZXNjcmlwdG9yIHsKLSAgICBwcm90ZWN0ZWQgSW5wdXRNZXRob2RDb250ZXh0IGltYzsKLQotICAgIHB1YmxpYyB2b2lkIGFjdGl2YXRlKCkgewotCi0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZGVhY3RpdmF0ZShib29sZWFuIGlzVGVtcG9yYXJ5KSB7Ci0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBkaXNwYXRjaEV2ZW50KEFXVEV2ZW50IGV2ZW50KSB7Ci0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBkaXNwb3NlKCkgewotCi0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZW5kQ29tcG9zaXRpb24oKSB7Ci0KLSAgICB9Ci0KLSAgICBwdWJsaWMgT2JqZWN0IGdldENvbnRyb2xPYmplY3QoKSB7Ci0gICAgICAgIHJldHVybiBudWxsOwotICAgIH0KLQotICAgIHB1YmxpYyBMb2NhbGUgZ2V0TG9jYWxlKCkgewotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBoaWRlV2luZG93cygpIHsKLQotICAgIH0KLQotICAgIHB1YmxpYyBib29sZWFuIGlzQ29tcG9zaXRpb25FbmFibGVkKCkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgbm90aWZ5Q2xpZW50V2luZG93Q2hhbmdlKFJlY3RhbmdsZSBib3VuZHMpIHsKLQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlY29udmVydCgpIHsKLQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlbW92ZU5vdGlmeSgpIHsKLQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNldENoYXJhY3RlclN1YnNldHMoU3Vic2V0W10gc3Vic2V0cykgewotCi0gICAgfQotICAgIAotICAgIHB1YmxpYyB2b2lkIHNldENvbXBvc2l0aW9uRW5hYmxlZChib29sZWFuIGVuYWJsZSkgewotCi0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0SW5wdXRNZXRob2RDb250ZXh0KElucHV0TWV0aG9kQ29udGV4dCBjb250ZXh0KSB7Ci0gICAgICAgIGltYyA9IGNvbnRleHQ7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gc2V0TG9jYWxlKExvY2FsZSBsb2NhbGUpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIHB1YmxpYyBMb2NhbGVbXSBnZXRBdmFpbGFibGVMb2NhbGVzKCkgdGhyb3dzIEFXVEV4Y2VwdGlvbiB7Ci0gICAgCXJldHVybiBuZXcgTG9jYWxlW117TG9jYWxlLmdldERlZmF1bHQoKSwgTG9jYWxlLkVOR0xJU0h9OwotICAgICAgICAvL3JldHVybiBuZXcgTG9jYWxlW117TG9jYWxlLmdldERlZmF1bHQoKSwgTG9jYWxlLlVTfTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgSW5wdXRNZXRob2QgY3JlYXRlSW5wdXRNZXRob2QoKSB0aHJvd3MgRXhjZXB0aW9uIHsgICAgICAgIAotICAgICAgICByZXR1cm4gdGhpczsKLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nIGdldElucHV0TWV0aG9kRGlzcGxheU5hbWUoTG9jYWxlIGlucHV0TG9jYWxlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMb2NhbGUgZGlzcGxheUxhbmd1YWdlKSB7Ci0gICAgICAgIHJldHVybiAiU3lzdGVtIGlucHV0IG1ldGhvZHMiOyAvLyROT04tTkxTLTEkCi0gICAgfQotCi0gICAgcHVibGljIEltYWdlIGdldElucHV0TWV0aG9kSWNvbihMb2NhbGUgaW5wdXRMb2NhbGUpIHsKLSAgICAgICAgcmV0dXJuIG51bGw7Ci0gICAgfQotCi0gICAgcHVibGljIGJvb2xlYW4gaGFzRHluYW1pY0xvY2FsZUxpc3QoKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgCi0gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZGlzYWJsZUlNRSgpOwotICAgIAotLy8gICAgcHVibGljIGFic3RyYWN0IHZvaWQgZGlzYWJsZUlNRShsb25nIGlkKTsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZU1vdXNlSW5mby5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZU1vdXNlSW5mby5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwNjk2OTc1Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9OYXRpdmVNb3VzZUluZm8uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQyICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgRG1pdHJ5IEEuIER1cm5ldgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrOwotCi1pbXBvcnQgamF2YS5hd3QuUG9pbnQ7Ci0KLS8qKgotICogVGhlIGludGVyZmFjZSBwcm92aWRlcyBhY2Nlc3MgdG8gcGxhdGZvcm0gZGVwZW5kZW50IGZ1bmN0aW9uYWxpdHkKLSAqIGZvciBjbGFzc2VzIGphdmEuYXd0LlBvaW50ZXJJbmZvICYgamF2YS5hd3QuTW91c2VJbmZvLgotICovCi1wdWJsaWMgaW50ZXJmYWNlIE5hdGl2ZU1vdXNlSW5mbyB7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRoZSBQb2ludCB0aGF0IHJlcHJlc2VudHMKLSAgICAgKiB0aGUgY29vcmRpbmF0ZXMgb2YgdGhlIHBvaW50ZXIgb24gdGhlIHNjcmVlbi4KLSAgICAgKi8KLSAgICBQb2ludCBnZXRMb2NhdGlvbigpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyB0aGUgbnVtYmVyIG9mIGJ1dHRvbnMgb24gdGhlIG1vdXNlLgotICAgICAqIElmIG5vIG1vdXNlIGlzIGluc3RhbGxlZCByZXR1cm5zIC0xLgotICAgICAqLwotICAgIGludCBnZXROdW1iZXJPZkJ1dHRvbnMoKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9OYXRpdmVSb2JvdC5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZVJvYm90LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBiMzU0ZDAuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZVJvYm90LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw3NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIERtaXRyeSBBLiBEdXJuZXYKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0azsKLQotaW1wb3J0IGphdmEuYXd0LkNvbG9yOwotaW1wb3J0IGphdmEuYXd0LlJlY3RhbmdsZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotCi0vKioKLSAqIEEgY3Jvc3MtcGxhdGZvcm0gaW50ZXJmYWNlIGZvciBqYXZhLmF3dC5Sb2JvdCBpbXBsZW1lbnRhdGlvbgotICovCi1wdWJsaWMgaW50ZXJmYWNlIE5hdGl2ZVJvYm90IHsKLQotICAgIC8qKgotICAgICAqIEBzZWUgamF2YS5hd3QuUm9ib3QjY3JlYXRlU2NyZWVuQ2FwdHVyZShSZWN0YW5nbGUpCi0gICAgICogQHBhcmFtIHNjcmVlblJlY3QgcmVjdGFuZ2xlIHRvIGNhcHR1cmUgaW4gc2NyZWVuIGNvb3JkaW5hdGVzCi0gICAgICogQHJldHVybiB0aGUgY2FwdHVyZWQgaW1hZ2Ugb3IgbnVsbCBpZgotICAgICAqIGNhcHR1cmUgZmFpbGVkLgotICAgICAqLwotICAgIEJ1ZmZlcmVkSW1hZ2UgY3JlYXRlU2NyZWVuQ2FwdHVyZShSZWN0YW5nbGUgc2NyZWVuUmVjdCk7Ci0KLSAgICAvKioKLSAgICAgKiBAc2VlIGphdmEuYXd0LlJvYm90I2dldFBpeGVsQ29sb3IoaW50LCBpbnQpCi0gICAgICovCi0gICAgQ29sb3IgZ2V0UGl4ZWwoaW50IHgsIGludCB5KTsKLQotICAgIC8qKgotICAgICAqIEdlbmVyYXRlIGEgbmF0aXZlIHN5c3RlbSBrZXlib2FyZCBpbnB1dCBldmVudC4KLSAgICAgKiBAcGFyYW0ga2V5Y29kZSBBIEphdmEgdmlydHVhbCBrZXkgY29kZQotICAgICAqIEBwYXJhbSBwcmVzcyBBIGtleSBpcyBwcmVzc2VkIGlmIHRydWUsIHJlbGVhc2VkIG90aGVyd2lzZQotICAgICAqIEBzZWUgamF2YS5hd3QuUm9ib3Qja2V5UHJlc3MoaW50KQotICAgICAqIEB0aHJvd3MgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uIGlmIGtleWNvZGUgaXMgaW52YWxpZCBpbiB0aGUgbmF0aXZlIHN5c3RlbQotICAgICAqLwotICAgIHZvaWQga2V5RXZlbnQoaW50IGtleWNvZGUsIGJvb2xlYW4gcHJlc3MpOwotCi0gICAgLyoqCi0gICAgICogR2VuZXJhdGUgYSBuYXRpdmUgc3lzdGVtIG1vdXNlIGJ1dHRvbihzKSBwcmVzcyBvciByZWxlYXNlIGV2ZW50LgotICAgICAqIEBwYXJhbSBidXR0b25zIEEgbWFzayBvZiBKYXZhIG1vdXNlIGJ1dHRvbiBmbGFncwotICAgICAqIEBwYXJhbSBwcmVzcyBidXR0b25zIGFyZSBwcmVzc2VkIGlmIHRydWUsIHJlbGVhc2VkIG90aGVyd2lzZQotICAgICAqIEBzZWUgamF2YS5hd3QuUm9ib3QjbW91c2VQcmVzcyhpbnQpCi0gICAgICovCi0gICAgdm9pZCBtb3VzZUJ1dHRvbihpbnQgYnV0dG9ucywgYm9vbGVhbiBwcmVzcyk7Ci0KLSAgICAvKioKLSAgICAgKiBHZW5lcmF0ZSBhIG5hdGl2ZSBzeXN0ZW0gbW91c2UgbW90aW9uIGV2ZW50LgotICAgICAqCi0gICAgICogQHNlZSBqYXZhLmF3dC5Sb2JvdCNtb3VzZU1vdmUoaW50LCBpbnQpCi0gICAgICovCi0gICAgdm9pZCBtb3VzZU1vdmUoaW50IHgsIGludCB5KTsKLQotICAgIC8qKgotICAgICAqIEdlbmVyYXRlIGEgbmF0aXZlIHN5c3RlbSBtb3VzZSB3aGVlbCBldmVudC4KLSAgICAgKgotICAgICAqIEBzZWUgamF2YS5hd3QuUm9ib3QjbW91c2VXaGVlbChpbnQpCi0gICAgICovCi0gICAgdm9pZCBtb3VzZVdoZWVsKGludCB3aGVlbEFtdCk7Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlV2luZG93LmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvTmF0aXZlV2luZG93LmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDczZmQ2YzAuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL05hdGl2ZVdpbmRvdy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjIwICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWlraGFpbCBEYW5pbG92Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGs7Ci0KLWltcG9ydCBqYXZhLmF3dC5JbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5JbnNldHM7Ci1pbXBvcnQgamF2YS5hd3QuUG9pbnQ7Ci1pbXBvcnQgamF2YS5hd3QuUmVjdGFuZ2xlOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5NdWx0aVJlY3RBcmVhOwotCi0KLS8qKgotICogUHJvdmlkZXMgY3Jvc3MtcGxhdGZvcm0gd2F5IHRvIG1hbmlwdWxhdGUgbmF0aXZlIHdpbmRvdy4KLSAqCi0gKiBSZXN1bHRzIG9mIG1ldGhvZHMgYXJlIHJlcG9ydGVkIHRocm91Z2ggbmF0aXZlIG1lc3NhZ2VzLgotICovCi1wdWJsaWMgaW50ZXJmYWNlIE5hdGl2ZVdpbmRvdyB7Ci0gICAgLyoqCi0gICAgICogUmV0dXJucyBzeXN0ZW0gaWQgb2YgdGhlIGFzc29jaWF0ZWQgd2luZG93Ci0gICAgICogQHJldHVybiBIV05EIG9uIFdpbmRvd3MsIHh3aW5kb3cgb24gWAotICAgICAqLwotICAgIGxvbmcgZ2V0SWQoKTsKLQotICAgIC8qKgotICAgICAqIFNob3dzL2hpZGVzIHdpbmRvdwotICAgICAqIEBwYXJhbSB2IC0gbmV3IHZpc2liaWxpdHkKLSAgICAgKi8KLSAgICB2b2lkIHNldFZpc2libGUoYm9vbGVhbiB2KTsKLQotICAgIC8qKgotICAgICAqIE1lYW5zIG9ubHkgc2l6ZSBzaG91bGQgYmUgY2hhbmdlZAotICAgICAqLwotICAgIHN0YXRpYyBmaW5hbCBpbnQgQk9VTkRTX05PTU9WRSA9IDE7Ci0KLSAgICAvKioKLSAgICAgKiBNZWFucyBvbmx5IHBvc2l0aW9uIHNob3VsZCBiZSBjaGFuZ2VkCi0gICAgICovCi0gICAgc3RhdGljIGZpbmFsIGludCBCT1VORFNfTk9TSVpFID0gMjsKLQotICAgIC8qKgotICAgICAqIFRyaWVzIHRvIHNldCBkZXNpcmVkIHdpbmRvdyBib3VuZHMuIEl0J3Mgbm90IGd1cmFudGllZCB0aGUKLSAgICAgKiBwcm9wZXJ0eSB3aWxsIGhhdmUgdGhlIGRlc2lyZWQgdmFsdWUuIFRoZSB2YWx1ZSBjaGFuZ2UKLSAgICAgKiBzaG91bGQgYmUgcmVwb3J0ZWQgYnkgc3lzdGVtIGV2ZW50IChhcyBmb3Igb3RoZXIgcHJvcGVydGllcykuCi0gICAgICoKLSAgICAgKiA8cC8+ICBJZiBjaGlsZCwgcG9zaXRpb24gaXMgcmVsYXRpdmUgdG8gcGFyZW50IHdpbmRvdy4KLSAgICAgKiBAcGFyYW0geCAtIGRlc2lyZWQgeAotICAgICAqIEBwYXJhbSB5IC0gZGVzaXJlZCB5Ci0gICAgICogQHBhcmFtIHcgLSBkZXNpcmVkIHdpZHRoCi0gICAgICogQHBhcmFtIGggLSBkZXNpcmVkIGhlaWdodAotICAgICAqIEBwYXJhbSBib3VuZHNNYXNrIC0gYml0d2lzZSBPUiBvZiBCT1VORFNfKiBjb25zdGFudHMuCi0gICAgICogR292ZXJucyB0aGUgbmV3IGJvdW5kcyBpbnRlcnByZXRhdGlvbi4KLSAgICAgKi8KLSAgICB2b2lkIHNldEJvdW5kcyhpbnQgeCwgaW50IHksIGludCB3LCBpbnQgaCwgaW50IGJvdW5kc01hc2spOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBsYXN0IG5vdGlmaWVkIHdpbmRvdyBib3VuZHMuIFRoaXMgbWVhbnMgdGhlIGxhc3QgYm91bmRzCi0gICAgICogcmVwb3J0ZWQgYnkgc3lzdGVtIGV2ZW50LgotICAgICAqCi0gICAgICogPHAvPiAgSWYgY2hpbGQsIHBvc2l0aW9uIGlzIHJlbGF0aXZlIHRvIHBhcmVudCB3aW5kb3cuCi0gICAgICogQHJldHVybiBsYXN0IG5vdGlmaWVkIHdpbmRvdyBib3VuZHMKLSAgICAgKi8KLSAgICBSZWN0YW5nbGUgZ2V0Qm91bmRzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGxhc3Qgbm90aWZpZWQgaW5zZXRzLiBUaGlzIG1lYW5zIHRoZSBsYXN0IGluc2V0cwotICAgICAqIHJlcG9ydGVkIGJ5IHN5c3RlbSBldmVudC4gSW5zZXRzIGFyZSBtYXJnaW5zIGFyb3VuZCBjbGllbnQgYXJlYQotICAgICAqIG9jdXBpZWQgYnkgc3lzdGVtIHByb3ZpZGVkIGRlY29yLCB1c3VzYWxseSBib3JkZXIgYW5kIHRpdGxlYmFyLgotICAgICAqIEByZXR1cm4gbGFzdCBub3RpZmllZCBpbnNldHMKLSAgICAgKi8KLSAgICBJbnNldHMgZ2V0SW5zZXRzKCk7Ci0KLSAgICAvKioKLSAgICAgKiBFbmFibGVzL2Rpc2FibGVzIHByb2Nlc3Npbmcgb2YgaW5wdXQgKGtleSwgbW91c2UpIGV2ZW50Ci0gICAgICogYnkgd2luZG93LiBJZiBkaXNhYmxlZCBpbnB1dCBldmVudHMgYXJlIGlnbm9yZWQuCi0gICAgICogQHBhcmFtIHZhbHVlIC0gaWYgZW5hYmxlZAotICAgICAqLwotICAgIHZvaWQgc2V0RW5hYmxlZChib29sZWFuIHZhbHVlKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgdGhlICJmb2N1c2FibGUiIHdpbmRvdyBzdGF0ZS4KLSAgICAgKiBAcGFyYW0gdmFsdWUgLSBpZiB0cnVlIG1ha2VzIHdpbmRvdyBmb2N1c2FibGUKLSAgICAgKi8KLSAgICB2b2lkIHNldEZvY3VzYWJsZShib29sZWFuIHZhbHVlKTsKLQotICAgIC8qKgotICAgICAqCi0gICAgICogQHJldHVybiBjdXJyZW50IGZvY3VzYWJsZSB3aW5kb3cgc3RhdGUKLSAgICAgKi8KLSAgICBib29sZWFuIGlzRm9jdXNhYmxlKCk7Ci0KLSAgICAvKioKLSAgICAgKiBUcmllcyB0byBzZXQgYXBwbGljYXRpb24gaW5wdXQgZm9jdXMgdG8gdGhlIHdpbmRvdyBvciBjbGVhcgotICAgICAqIGN1cnJlbnQgZm9jdXMgZnJvbSBmb2N1c2VkIHdpbmRvdy4KLSAgICAgKgotICAgICAqIDxwLz4gRm9yIHRvcGxldmVsIHdpbmRvd3MgaXQncyBub3QgZ3VyYW50aWVkIGZvY3VzIHdpbGwgbGFuZCBpbgotICAgICAqIGRlc2lyZWQgd2luZG93IGV2ZW4gaWYgZnVuY3Rpb24gcmV0dXJucyB0cnVlLiBGb2N1cyB0cmF2ZXJzYWwgc2hvdWxkIGJlIHRyYWNrZWQKLSAgICAgKiBieSBwcm9jZXNzaW5nIHN5c3RlbSBldmVudHMuCi0gICAgICoKLSAgICAgKiBAcGFyYW0gZm9jdXMgIC0gaWYgdHJ1ZSBzZXRzIGZvY3VzLCBlbHNlIGNsZWFycyBmb2N1cwotICAgICAqIEByZXR1cm4gaWYgc3VjY2VzcwotICAgICAqLwotICAgIGJvb2xlYW4gc2V0Rm9jdXMoYm9vbGVhbiBmb2N1cyk7Ci0KLSAgICAvKioKLSAgICAgKiBEZXN0cm95cyB0aGUgYXNzY29pYXRlZCB3aW5kb3cuCi0gICAgICogQXR0ZW1wdHMgdG8gdXNlIGl0IHRoZXJlYWZ0ZXIgY2FuIHJlc3VsdCBpbgotICAgICAqIHVucHJlZGljdGFibGUgYmVjaGF2aW9yLgotICAgICAqLwotICAgIHZvaWQgZGlzcG9zZSgpOwotCi0gICAgLyoqCi0gICAgICogQ2hhbmdlcyB3aW5kb3cgWi1vcmRlciB0byBwbGFjZSB0aGlzIHdpbmRvdyB1bmRlciwgSWYgdyBpcyBudWxsCi0gICAgICogcGxhY2VzIHBsYWNlcyB0aGlzIHdpbmRvdyBvbiB0aGUgdG9wLiBaLW9yZGVyIGlzIHBlciBwYXJlbnQuCi0gICAgICogVG9wbGV2ZWxzIGEgY2hpbGRyZW4gb2YgZGVza3RvcCBpbiB0ZXJtcyBvZiBaLW9yZGVyLgotICAgICAqIEBwYXJhbSB3IC0gd2luZG93IHRvIHBsYWNlIHVuZGVyLgotICAgICAqLwotICAgIHZvaWQgcGxhY2VBZnRlcihOYXRpdmVXaW5kb3cgdyk7Ci0KLSAgICAvKioKLSAgICAgKiBQbGFjZXMgd2luZG93IG9uIHRvcCBvZiBaLW9yZGVyCi0gICAgICovCi0gICAgdm9pZCB0b0Zyb250KCk7Ci0KLSAgICAvKioKLSAgICAgKiBQbGFjZXMgd2luZG93IG9uIGJvdHRvbSBvZiBaLW9yZGVyCi0gICAgICovCi0gICAgdm9pZCB0b0JhY2soKTsKLQotICAgIC8qKgotICAgICAqIE1ha2VzIHRoZSB3aW5kb3cgcmVzaXphYmxlL25vdCByZXNpemFibGUgYnkgdXNlcgotICAgICAqIEBwYXJhbSB2YWx1ZSAtIGlmIHJlc2l6YWJsZQotICAgICAqLwotICAgIHZvaWQgc2V0UmVzaXphYmxlKGJvb2xlYW4gdmFsdWUpOwotCi0gICAgLyoqCi0gICAgICogU2V0cyB0aGUgd2luZG93IGNhcHRpb24KLSAgICAgKiBAcGFyYW0gdGl0bGUgLSBjYXB0aW9uIHRleHQKLSAgICAgKi8KLSAgICB2b2lkIHNldFRpdGxlKFN0cmluZyB0aXRsZSk7Ci0KLSAgICAvKioKLSAgICAgKiBBY3RpdmF0ZSB0aGUgbW91c2UgZXZlbnQgY2FwdHVyaW5nCi0gICAgICovCi0gICAgdm9pZCBncmFiTW91c2UoKTsKLQotICAgIC8qKgotICAgICAqIERlYWN0aXZhdGUgbW91c2UgZXZlbnQgY2FwdHVyaW5nCi0gICAgICovCi0gICAgdm9pZCB1bmdyYWJNb3VzZSgpOwotCi0gICAgLyoqCi0gICAgICogU2V0IGV4dGVuZGVkIHN0YXRlIGZvciB0b3AtbGV2ZWwgd2luZG93LgotICAgICAqCi0gICAgICogQHBhcmFtIHN0YXRlIC0gbmV3IHN0YXRlLCBiaXRtYXNrIG9mIElDT05JRklFRCwgTUFYSU1JWkVEX0JPVEgsIGV0Yy4KLSAgICAgKi8KLSAgICB2b2lkIHNldFN0YXRlKGludCBzdGF0ZSk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXQgdGhlIGltYWdlIHRvIGJlIGRpc3BsYXllZCBpbiB0aGUgbWluaW1pemVkIGljb24gZm9yCi0gICAgICogdG9wLWxldmVsIFtkZWNvcmF0ZWRdIHdpbmRvdy4KLSAgICAgKiBAcGFyYW0gaW1hZ2UgdGhlIGljb24gaW1hZ2UgdG8gYmUgZGlzcGxheWVkCi0gICAgICovCi0gICAgdm9pZCBzZXRJY29uSW1hZ2UoSW1hZ2UgaW1hZ2UpOwotCi0gICAgLyoqCi0gICAgICogTWFrZXMgd2luZG93IHRvcC1tb3N0IGlmIHZhbHVlIGlzIHRydWUsCi0gICAgICogbm9uLXRvcG1vc3Qobm9ybWFsKSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgdm9pZCBzZXRBbHdheXNPblRvcChib29sZWFuIHZhbHVlKTsKLQotICAgIC8qKgotICAgICAqIFNldCBkZXNpcmVkIFt0b3AtbGV2ZWxdIHdpbmRvdyBib3VuZHMgd2hlbiBiZWluZyBpbiBtYXhpbWl6ZWQgc3RhdGUuCi0gICAgICogRmllbGRzIHNldCB0byBJbnRlZ2VyLk1BWF9WQUxVRSBhcmUgaWdub3JlZFtzeXN0ZW0tc3VwcGxpZWQgdmFsdWVzIGFyZQotICAgICAqIHVzZWQgaW5zdGVhZF0KLSAgICAgKi8KLSAgICB2b2lkIHNldE1heGltaXplZEJvdW5kcyhSZWN0YW5nbGUgYm91bmRzKTsKLQotICAgIC8qKgotICAgICAqIEdldCBhYnNvbHV0ZSBwb3NpdGlvbiBvbiB0aGUgc2NyZWVuCi0gICAgICovCi0gICAgUG9pbnQgZ2V0U2NyZWVuUG9zKCk7Ci0KLSAgICAvKioKLSAgICAgKiBTZXQgYSB3aW5kb3cgInBhY2tlZCIgZmxhZzoKLSAgICAgKiB0aGUgZmxhZyBpbmRpY2F0ZXMgdGhhdCBpZiBpbnNldHMgY2hhbmdlCi0gICAgICogY2xpZW50IGFyZWEgc2hvdWxkbid0IGJlIHJlc2l6ZWQsIGJ1dCBmcmFtZQotICAgICAqIG11c3QgYmUgcmVzaXplZCBpbnN0ZWFkCi0gICAgICovCi0gICAgdm9pZCBzZXRQYWNrZWQoYm9vbGVhbiBwYWNrZWQpOwotICAgIAotICAgIC8qKgotICAgICAqIE1ha2Ugd2luZG93IGFuICJpbnB1dCBtZXRob2Qgd2luZG93IiBieSBzZXR0aW5nCi0gICAgICogc3BlY2lhbCB3aW5kb3cgc3R5bGUsIGUuIGcuIHNtYWxsIHRpdGxlIGJhciwgbm8KLSAgICAgKiBjbG9zZSwgbWluaW1pemUvbWF4aW1pemUgYnV0dG9ucy4gRm9yIGludGVybmFsCi0gICAgICogdXNlIGJ5IGlucHV0IG1ldGhvZCBmcmFtZXdvcmsuCi0gICAgICoKLSAgICAgKi8KLSAgICB2b2lkIHNldElNU3R5bGUoKTsKLQotICAgIE11bHRpUmVjdEFyZWEgZ2V0T2JzY3VyZWRSZWdpb24oUmVjdGFuZ2xlIHBhcnQpOwotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL1NodXRkb3duVGhyZWFkLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvU2h1dGRvd25UaHJlYWQuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzAxZWI0Ni4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvU2h1dGRvd25UaHJlYWQuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDgzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgTWljaGFlbCBEYW5pbG92LCBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0azsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuaW50ZXJuYWwubmxzLk1lc3NhZ2VzOwotCi1wdWJsaWMgZmluYWwgY2xhc3MgU2h1dGRvd25UaHJlYWQgZXh0ZW5kcyBUaHJlYWQgewotICAgIAotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgY2xhc3MgV2F0Y2hkb2cgewotICAgIH0KLQotICAgIHB1YmxpYyBTaHV0ZG93blRocmVhZCgpIHsKLSAgICAgICAgc2V0TmFtZSgiQVdULVNodXRkb3duIik7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgc2V0RGFlbW9uKGZhbHNlKTsKLSAgICB9Ci0gICAgCi0gICAgcHJpdmF0ZSBib29sZWFuIHNob3VsZFN0b3AgPSBmYWxzZTsKLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHJ1bigpIHsKLSAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7Ci0gICAgICAgICAgICBub3RpZnlBbGwoKTsgLy8gc3luY2hyb25pemUgdGhlIHN0YXJ0dXAKLQotICAgICAgICAgICAgd2hpbGUgKHRydWUpIHsKLSAgICAgICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgICAgICB3YWl0KCk7Ci0gICAgICAgICAgICAgICAgfSBjYXRjaCAoSW50ZXJydXB0ZWRFeGNlcHRpb24gZSkgewotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIGlmIChzaG91bGRTdG9wKSB7Ci0gICAgICAgICAgICAgICAgICAgIG5vdGlmeUFsbCgpOyAvLyBzeW5jaHJvbml6ZSB0aGUgc2h1dGRvd24KLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHN0YXJ0KCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIHN1cGVyLnN0YXJ0KCk7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIHdhaXQoKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKEludGVycnVwdGVkRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjY9U2h1dGRvd24gdGhyZWFkIHdhcyBpbnRlcnJ1cHRlZCB3aGlsZSBzdGFydGluZwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKAotICAgICAgICAgICAgICAgICAgICAgICAgTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjYiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHNodXRkb3duKCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIHNob3VsZFN0b3AgPSB0cnVlOwotICAgICAgICAgICAgbm90aWZ5QWxsKCk7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIHdhaXQoKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKEludGVycnVwdGVkRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjc9U2h1dGRvd24gdGhyZWFkIHdhcyBpbnRlcnJ1cHRlZCB3aGlsZSBzdG9wcGluZwotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKAotICAgICAgICAgICAgICAgICAgICAgICAgTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjciKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9TaHV0ZG93bldhdGNoZG9nLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvU2h1dGRvd25XYXRjaGRvZy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2ZWZhNTE5Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9TaHV0ZG93bldhdGNoZG9nLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4NiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqIAotICogQGF1dGhvciBQYXZlbCBEb2xnb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0azsKLQotLyoqCi0gKiBTaHV0ZG93biBXYXRjaGRvZwotICovCi1wdWJsaWMgZmluYWwgY2xhc3MgU2h1dGRvd25XYXRjaGRvZyB7Ci0gICAgCi0gICAgcHJpdmF0ZSBib29sZWFuIG5hdGl2ZVF1ZXVlRW1wdHkgPSB0cnVlOwotICAgIHByaXZhdGUgYm9vbGVhbiBhd3RRdWV1ZUVtcHR5ID0gdHJ1ZTsKLSAgICBwcml2YXRlIGJvb2xlYW4gd2luZG93TGlzdEVtcHR5ID0gdHJ1ZTsKLQotICAgIHByaXZhdGUgYm9vbGVhbiBmb3JjZWRTaHV0ZG93biA9IGZhbHNlOwotICAgIAotICAgIHByaXZhdGUgU2h1dGRvd25UaHJlYWQgdGhyZWFkOwotCi0gICAgcHVibGljIHN5bmNocm9uaXplZCB2b2lkIHNldE5hdGl2ZVF1ZXVlRW1wdHkoYm9vbGVhbiBlbXB0eSkgewotICAgICAgICBuYXRpdmVRdWV1ZUVtcHR5ID0gZW1wdHk7Ci0gICAgICAgIGNoZWNrU2h1dGRvd24oKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgc2V0QXd0UXVldWVFbXB0eShib29sZWFuIGVtcHR5KSB7Ci0gICAgICAgIGF3dFF1ZXVlRW1wdHkgPSBlbXB0eTsKLSAgICAgICAgY2hlY2tTaHV0ZG93bigpOwotICAgIH0KLQotICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgdm9pZCBzZXRXaW5kb3dMaXN0RW1wdHkoYm9vbGVhbiBlbXB0eSkgewotICAgICAgICB3aW5kb3dMaXN0RW1wdHkgPSBlbXB0eTsKLSAgICAgICAgY2hlY2tTaHV0ZG93bigpOwotICAgIH0KLSAgICAKLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgZm9yY2VTaHV0ZG93bigpIHsKLSAgICAgICAgZm9yY2VkU2h1dGRvd24gPSB0cnVlOwotICAgICAgICBzaHV0ZG93bigpOwotICAgIH0KLSAgICAKLSAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgc3RhcnQoKSB7Ci0gICAgICAgIGtlZXBBbGl2ZSgpOwotICAgIH0KLQotICAgIHByaXZhdGUgdm9pZCBjaGVja1NodXRkb3duKCkgewotICAgICAgICBpZiAoY2FuU2h1dGRvd24oKSkgewotICAgICAgICAgICAgc2h1dGRvd24oKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGtlZXBBbGl2ZSgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHJpdmF0ZSBib29sZWFuIGNhblNodXRkb3duKCkgewotICAgICAgICByZXR1cm4gKG5hdGl2ZVF1ZXVlRW1wdHkgJiYgYXd0UXVldWVFbXB0eSAmJiB3aW5kb3dMaXN0RW1wdHkpIHx8Ci0gICAgICAgICAgICAgICAgZm9yY2VkU2h1dGRvd247Ci0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIGtlZXBBbGl2ZSgpIHsKLSAgICAgICAgaWYgKHRocmVhZCA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJlYWQgPSBuZXcgU2h1dGRvd25UaHJlYWQoKTsKLSAgICAgICAgICAgIHRocmVhZC5zdGFydCgpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHJpdmF0ZSB2b2lkIHNodXRkb3duKCkgewotICAgICAgICBpZiAodGhyZWFkICE9IG51bGwpIHsKLSAgICAgICAgICAgIHRocmVhZC5zaHV0ZG93bigpOwotICAgICAgICAgICAgdGhyZWFkID0gbnVsbDsKLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9TeW5jaHJvbml6ZXIuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9TeW5jaHJvbml6ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggM2VlYWEwYi4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvU3luY2hyb25pemVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyMDAgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBNaWtoYWlsIERhbmlsb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0Lnd0azsKLQotaW1wb3J0IGphdmEudXRpbC5IYXNodGFibGU7Ci1pbXBvcnQgamF2YS51dGlsLkxpbmtlZExpc3Q7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmludGVybmFsLm5scy5NZXNzYWdlczsKLQotLyoqCi0gKiBDbGFzcyBzeW5jaHJvbml6ZXIgaXMgdG8gcHJvdGVjdCBBV1Qgc3RhdGUgaW50ZWdyaXR5IGluIG11bHRpdGhyZWFkaW5nIGVudmlyb25tZW50LgotICogSXQgaXMgc3VwcG9zZWQgdG8gaGF2ZSBhIGNoaWxkIGNsYXNzIHBlciBuYXRpdmUgcGxhdGZvcm0uCi0gKiBUaGUgb25seSBpbnN0YW5jZSBpcyBjcmVhdGVkIG9uIHRoZSBmaXJzdCB1c2Ugb2Ygb25lIG9mIHRoZSBjb3JlIEFXVCBjbGFzc2VzLgotICogUmVnaXN0ZXJzIFdUSyBvbiB0aGUgZGlzcGF0Y2ggdGhyZWFkIHN0YXJ0dXAuCi0gKiBJdCBpcyBqdXN0IGEgc3BlY2lhbCBraW5kIG9mIG11dGV4LgotICoKLSAqLwotCi1wdWJsaWMgY2xhc3MgU3luY2hyb25pemVyIHsKLSAgICAvL1RPRE86IHRoaW5rIGFib3V0IGphdmEudXRpbC5jb25jdXJyZW50IHVzZSBmb3IgZmFzdGVyIGJsb2NraW5nL2F3YWtpbmcgb3BlcmF0aW9ucwotICAgIC8vVE9ETzogdGhpbmsgYWJvdXQgYWxsIHN5bmNocm9uaXplZCBtZXRob2RzLiBJcyB0aGVyZSBuZWVkIHRvIHN5bmNocm9uaXplIGV2ZXJ5dGhpbmc/Ci0KLSAgICAvKioKLSAgICAgKiBUaGlzIGZpZWxkIGhvbGRzIHRoZSBjb3VudGVyIG9mIGxvY2sgb3BlcmF0aW9uLgotICAgICAqIFRvIGZyZWUgc3luY2hyb25pemVyIHVubG9jayBtZXRob2QgbXVzdCBiZSBjYWxsZWQgJGFjcXVlc3RDb3VudGVyIHRpbWVzLgotICAgICAqIEVxdWFscyB0byAwIHdoZW4gc3luY2hyb25pemVyIGlzIGZyZWUuCi0gICAgICovCi0gICAgcHJvdGVjdGVkIGludCBhY3F1ZXN0Q291bnRlcjsKLQotICAgIC8qKgotICAgICAqIFRoaXMgZmllbGQgaG9sZHMgdGhlIG93bmVyIG9mIHN5bmNocm9uaXplci4KLSAgICAgKiBPd25lciBvZiBzeW5jaHJvbml6ZXIgaXMgYSBsYXN0IHRocmVhZCB0aGF0IHN1Y2Nlc3NmdWxseSBsb2NrZWQgc3luY2hyb25pemVyIGFuZAotICAgICAqIHN0aWxsIGhhdm4ndCBmcmVlZCBpdC4gRXF1YWxzIHRvIG51bGwgd2hlbiBzeW5jaHJvbml6ZXIgaXMgZnJlZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgVGhyZWFkIG93bmVyOwotCi0gICAgLyoqCi0gICAgICogVGhpcyBmaWVsZCBob2xkcyB0aGUgd2FpdCBxdWV1ZS4KLSAgICAgKiBXYWl0IHF1ZXVlIGlzIGEgcXVldWUgd2hlcmUgdGhyZWFkIHdhaXQgZm9yIHN5bmNocm9uaXplciBhY2Nlc3MuCi0gICAgICogRW1wdHkgd2hlbiBzeW5jaHJvbml6ZXIgaXMgZnJlZS4KLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgZmluYWwgTGlua2VkTGlzdDxUaHJlYWQ+IHdhaXRRdWV1ZSA9IG5ldyBMaW5rZWRMaXN0PFRocmVhZD4oKTsKLQotICAgIC8qKgotICAgICAqIFRoZSBldmVudCBkaXNwYXRjaCB0aHJlYWQKLSAgICAgKi8KLSAgICBwcm90ZWN0ZWQgVGhyZWFkIGRpc3BhdGNoVGhyZWFkOwotCi0gICAgcHJpdmF0ZSBmaW5hbCBIYXNodGFibGU8VGhyZWFkLCBJbnRlZ2VyPiBzdG9yZWRTdGF0ZXMgPSBuZXcgSGFzaHRhYmxlPFRocmVhZCwgSW50ZWdlcj4oKTsKLQotICAgIC8qKgotICAgICAqIEFjcXVpcmUgdGhlIGxvY2sgZm9yIHRoaXMgc3luY2hyb25pemVyLiBOZXN0ZWQgbG9jayBpcyBzdXBwb3J0ZWQuCi0gICAgICogSWYgdGhlIG11dGV4IGlzIGFscmVhZHkgbG9ja2VkIGJ5IGFub3RoZXIgdGhyZWFkLCB0aGUgY3VycmVudCB0aHJlYWQgd2lsbCBiZSBwdXQKLSAgICAgKiBpbnRvIHdhaXQgcXVldWUgdW50aWwgdGhlIGxvY2sgYmVjb21lcyBhdmFpbGFibGUuCi0gICAgICogQWxsIHVzZXIgdGhyZWFkcyBhcmUgc2VydmVkIGluIEZJRk8gb3JkZXIuIERpc3BhdGNoIHRocmVhZCBoYXMgaGlnaGVyIHByaW9yaXR5LgotICAgICAqIFN1cHBvc2VkIHRvIGJlIHVzZWQgaW4gVG9vbGtpdC5sb2NrQVdUKCkgb25seS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBsb2NrKCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIFRocmVhZCBjdXJUaHJlYWQgPSBUaHJlYWQuY3VycmVudFRocmVhZCgpOwotCi0gICAgICAgICAgICBpZiAoYWNxdWVzdENvdW50ZXIgPT0gMCkgewotICAgICAgICAgICAgICAgIGFjcXVlc3RDb3VudGVyID0gMTsKLSAgICAgICAgICAgICAgICBvd25lciA9IGN1clRocmVhZDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgaWYgKG93bmVyID09IGN1clRocmVhZCkgewotICAgICAgICAgICAgICAgICAgICBhY3F1ZXN0Q291bnRlcisrOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChjdXJUaHJlYWQgPT0gZGlzcGF0Y2hUaHJlYWQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHdhaXRRdWV1ZS5hZGRGaXJzdChjdXJUaHJlYWQpOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgd2FpdFF1ZXVlLmFkZExhc3QoY3VyVGhyZWFkKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgICAgICAgICAgd2FpdCgpOwotICAgICAgICAgICAgICAgICAgICB9IGNhdGNoIChJbnRlcnJ1cHRlZEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAob3duZXIgIT0gY3VyVGhyZWFkKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgd2FpdFF1ZXVlLnJlbW92ZShjdXJUaHJlYWQpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIGF3dC4xRj1XYWl0aW5nIGZvciByZXNvdXJjZSBhY2Nlc3MgdGhyZWFkIGludGVycnVwdGVkIG5vdCBmcm9tIHVubG9jayBtZXRob2QuCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oTWVzc2FnZXMKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC5nZXRTdHJpbmcoImF3dC4xRiIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmVsZWFzZSB0aGUgbG9jayBmb3IgdGhpcyBzeW5jaHJvbml6ZXIuCi0gICAgICogSWYgd2FpdCBxdWV1ZSBpcyBub3QgZW1wdHkgdGhlIGZpcnN0IHdhaXRpbmcgdGhyZWFkIGFjcXVpcmVzIHRoZSBsb2NrLgotICAgICAqIFN1cHBvc2VkIHRvIGJlIHVzZWQgaW4gVG9vbGtpdC51bmxvY2tBV1QoKSBvbmx5LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHVubG9jaygpIHsKLSAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7Ci0gICAgICAgICAgICBpZiAoYWNxdWVzdENvdW50ZXIgPT0gMCkgewotICAgICAgICAgICAgICAgIC8vIGF3dC4yMD1DYW4ndCB1bmxvY2sgbm90IGxvY2tlZCByZXNvdXJjZS4KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMCIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKG93bmVyICE9IFRocmVhZC5jdXJyZW50VGhyZWFkKCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjE9Tm90IG93bmVyIGNhbid0IHVubG9jayByZXNvdXJjZS4KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yMSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBhY3F1ZXN0Q291bnRlci0tOwotICAgICAgICAgICAgaWYgKGFjcXVlc3RDb3VudGVyID09IDApIHsKLSAgICAgICAgICAgICAgICBpZiAod2FpdFF1ZXVlLnNpemUoKSA+IDApIHsKLSAgICAgICAgICAgICAgICAgICAgYWNxdWVzdENvdW50ZXIgPSAxOwotICAgICAgICAgICAgICAgICAgICBvd25lciA9IHdhaXRRdWV1ZS5yZW1vdmVGaXJzdCgpOwotICAgICAgICAgICAgICAgICAgICBvd25lci5pbnRlcnJ1cHQoKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBvd25lciA9IG51bGw7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU3RvcmVzIHN0YXRlIG9mIHRoaXMgc3luY2hyb25pemVyIGFuZCBmcmVlcyBpdC4KLSAgICAgKiBTdXBwb3NlZCB0byBiZSB1c2VkIGluIFRvb2xraXQudW5zYWZlSW52b2tlQW5kV2FpdFVuZGVyQVdUTG9jaygpIG9ubHkgaW4gcGFpciB3aXRoCi0gICAgICogbG9ja0FuZFJlc3RvcmVTdGF0ZSgpLgotICAgICAqIERvIG5vdCBjYWxsIGl0IGRpcmVjdGx5LgotICAgICAqLwotICAgIHB1YmxpYyB2b2lkIHN0b3JlU3RhdGVBbmRGcmVlKCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIFRocmVhZCBjdXJUaHJlYWQgPSBUaHJlYWQuY3VycmVudFRocmVhZCgpOwotCi0gICAgICAgICAgICBpZiAob3duZXIgIT0gY3VyVGhyZWFkKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjIyPU5vdCBvd25lciBjYW4ndCBmcmVlIHJlc291cmNlLgotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIyIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoc3RvcmVkU3RhdGVzLmNvbnRhaW5zS2V5KGN1clRocmVhZCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjM9T25lIHRocmVhZCBjYW4ndCBzdG9yZSBzdGF0ZSBzZXZlcmFsIHRpbWVzIGluIGEgcm93LgotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiYXd0LjIzIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHN0b3JlZFN0YXRlcy5wdXQoY3VyVGhyZWFkLCBuZXcgSW50ZWdlcihhY3F1ZXN0Q291bnRlcikpOwotICAgICAgICAgICAgYWNxdWVzdENvdW50ZXIgPSAxOwotICAgICAgICAgICAgdW5sb2NrKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBMb2NrcyB0aGlzIHN5bmNocm9uaXplciBhbmQgcmVzdG9yZXMgaXQncyBzdGF0ZS4KLSAgICAgKiBTdXBwb3NlZCB0byBiZSB1c2VkIGluIFRvb2xraXQudW5zYWZlSW52b2tlQW5kV2FpdFVuZGVyQVdUTG9jaygpIG9ubHkgaW4gcGFpciB3aXRoCi0gICAgICogc3RvcmVTdGF0ZUFuZEZyZWUoKS4KLSAgICAgKiBEbyBub3QgY2FsbCBpdCBkaXJlY3RseS4KLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBsb2NrQW5kUmVzdG9yZVN0YXRlKCkgewotICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKLSAgICAgICAgICAgIFRocmVhZCBjdXJUaHJlYWQgPSBUaHJlYWQuY3VycmVudFRocmVhZCgpOwotCi0gICAgICAgICAgICBpZiAob3duZXIgPT0gY3VyVGhyZWFkKSB7Ci0gICAgICAgICAgICAgICAgLy8gYXd0LjI0PU93bmVyIGNhbid0IG92ZXJ3cml0ZSByZXNvdXJjZSBzdGF0ZS4gTG9jayBvcGVyYXRpb25zIG1heSBiZSBsb3N0LgotICAgICAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKAotICAgICAgICAgICAgICAgICAgICAgICAgTWVzc2FnZXMuZ2V0U3RyaW5nKCJhd3QuMjQiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICghc3RvcmVkU3RhdGVzLmNvbnRhaW5zS2V5KGN1clRocmVhZCkpIHsKLSAgICAgICAgICAgICAgICAvLyBhd3QuMjU9Tm8gc3RhdGUgc3RvcmVkIGZvciBjdXJyZW50IHRocmVhZC4KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUnVudGltZUV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImF3dC4yNSIpKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBsb2NrKCk7Ci0gICAgICAgICAgICBhY3F1ZXN0Q291bnRlciA9IHN0b3JlZFN0YXRlcy5nZXQoY3VyVGhyZWFkKS5pbnRWYWx1ZSgpOwotICAgICAgICAgICAgc3RvcmVkU3RhdGVzLnJlbW92ZShjdXJUaHJlYWQpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogU2V0cyByZWZlcmVuY2VzIHRvIFdUSyBhbmQgZXZlbnQgZGlzcGF0Y2ggdGhyZWFkLgotICAgICAqIENhbGxlZCBvbiB0b29sa2l0IHN0YXJ0dXAuCi0gICAgICoKLSAgICAgKiBAcGFyYW0gd3RrIC0gcmVmZXJlbmNlIHRvIFdUSyBpbnN0YW5jZQotICAgICAqIEBwYXJhbSBkaXNwYXRjaFRocmVhZCAtIHJlZmVyZW5jZSB0byBldmVudCBkaXNwYXRjaCB0aHJlYWQKLSAgICAgKi8KLSAgICBwdWJsaWMgdm9pZCBzZXRFbnZpcm9ubWVudChXVEsgd3RrLCBUaHJlYWQgZGlzcGF0Y2hUaHJlYWQpIHsKLSAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7Ci0gICAgICAgICAgICB0aGlzLmRpc3BhdGNoVGhyZWFkID0gZGlzcGF0Y2hUaHJlYWQ7Ci0gICAgICAgIH0KLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9TeXN0ZW1Qcm9wZXJ0aWVzLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvU3lzdGVtUHJvcGVydGllcy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2YjU5ZjBlLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9TeXN0ZW1Qcm9wZXJ0aWVzLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1OSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFBhdmVsIERvbGdvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrOwotCi1pbXBvcnQgamF2YS5hd3QuRm9udDsKLWltcG9ydCBqYXZhLmF3dC5mb250LlRleHRBdHRyaWJ1dGU7Ci1pbXBvcnQgamF2YS5hd3QuaW0uSW5wdXRNZXRob2RIaWdobGlnaHQ7Ci1pbXBvcnQgamF2YS51dGlsLk1hcDsKLQotLyoqCi0gKiBOYXRpdmVQcm9wZXJ0aWVzCi0gKi8KLQotcHVibGljIGludGVyZmFjZSBTeXN0ZW1Qcm9wZXJ0aWVzIHsKLQotICAgIC8qKgotICAgICAqIEdldCBjdXJyZW50IHZhbHVlIG9mIGEgc3lzdGVtIGNvbG9yCi0gICAgICogQHBhcmFtIGluZGV4IC0gb25lIG9mIGphdmEuYXd0LlN5c3RlbUNvbG9yIGNvbnN0YW50cwotICAgICAqIEByZXR1cm4gQVJHQiB2YWx1ZSBvZiByZXF1ZXN0ZWQgc3lzdGVtIGNvbG9yCi0gICAgICovCi0gICAgaW50IGdldFN5c3RlbUNvbG9yQVJHQihpbnQgaW5kZXgpOwotCi0gICAgLyoqCi0gICAgICogR2V0IGRlZmF1bHQgZm9udCBmb3IgR1VJIGVsZW1lbnRzIHN1Y2ggYXMgbWVudXMgYW5kIGJ1dHRvbnMKLSAgICAgKiBAcmV0dXJuIHRoZSBmb250IG9iamVjdAotICAgICAqLwotICAgIEZvbnQgZ2V0RGVmYXVsdEZvbnQoKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBGaWxsIHRoZSBnaXZlbiBNYXAgd2l0aCBzeXN0ZW0gcHJvcGVydGllcwotICAgICAqLwotICAgIHZvaWQgaW5pdChNYXA8U3RyaW5nLCA/PiBkZXNrdG9wUHJvcGVydGllcyk7Ci0KLSAgICAvKioKLSAgICAgKiBGaWxscyB0aGUgZ2l2ZW4gbWFwIHdpdGggc3lzdGVtLWRlcGVuZGVudCB2aXN1YWwgdGV4dAotICAgICAqIGF0dHJpYnV0ZXMgZm9yIHRoZSBhYnN0cmFjdCBkZXNjcmlwdGlvbiAKLSAgICAgKiBvZiB0aGUgZ2l2ZW4gaW5wdXQgbWV0aG9kIGhpZ2hsaWdodAotICAgICAqIEBzZWUgamF2YS5hd3QuVG9vbGtpdC5tYXBJbnB1dE1ldGhvZEhpZ2hsaWdodCgpCi0gICAgICovCi0gICAgdm9pZCBtYXBJbnB1dE1ldGhvZEhpZ2hsaWdodChJbnB1dE1ldGhvZEhpZ2hsaWdodCBoaWdobGlnaHQsIE1hcDxUZXh0QXR0cmlidXRlLCA/PiBtYXApOwotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL1dUSy5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9hd3Qvd3RrL1dUSy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0MTYyZmJkLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9XVEsuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDYxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUGF2ZWwgRG9sZ292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb24kCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGs7Ci0KLWltcG9ydCBqYXZhLmF3dC5HcmFwaGljc0RldmljZTsKLQotCi1wdWJsaWMgYWJzdHJhY3QgY2xhc3MgV1RLIHsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCBHcmFwaGljc0ZhY3RvcnkgZ2V0R3JhcGhpY3NGYWN0b3J5KCk7Ci0gICAgcHVibGljIGFic3RyYWN0IE5hdGl2ZUV2ZW50UXVldWUgZ2V0TmF0aXZlRXZlbnRRdWV1ZSgpOwotICAgIHB1YmxpYyBhYnN0cmFjdCBXaW5kb3dGYWN0b3J5IGdldFdpbmRvd0ZhY3RvcnkoKTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgcGxhdGZvcm0gc3BlY2lmaWMgaW1wbGVtZW50YXRpb24gb2YgdGhlIGludGVyZmFjZQotICAgICAqIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLkN1cnNvckZhY3RvcnkuCi0gICAgICogQHJldHVybiBpbXBsZW1lbnRhdGlvbiBvZiBDdXJzb3JGYWN0b3J5Ci0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IEN1cnNvckZhY3RvcnkgZ2V0Q3Vyc29yRmFjdG9yeSgpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBwbGF0Zm9ybSBzcGVjaWZpYyBpbXBsZW1lbnRhdGlvbiBvZiB0aGUgaW50ZXJmYWNlCi0gICAgICogb3JnLmFwYWNoZS5oYXJtb255LmF3dC53dGsuTmF0aXZlTW91c2VJbmZvLgotICAgICAqIEByZXR1cm4gaW1wbGVtZW50YXRpb24gb2YgTmF0aXZlTW91c2VJbmZvCi0gICAgICovCi0gICAgcHVibGljIGFic3RyYWN0IE5hdGl2ZU1vdXNlSW5mbyBnZXROYXRpdmVNb3VzZUluZm8oKTsKLQotICAgIHB1YmxpYyBhYnN0cmFjdCBTeXN0ZW1Qcm9wZXJ0aWVzIGdldFN5c3RlbVByb3BlcnRpZXMoKTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgcGxhdGZvcm0gc3BlY2lmaWMgaW1wbGVtZW50YXRpb24gb2YgdGhlIGludGVyZmFjZQotICAgICAqIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLk5hdGl2ZVJvYm90LgotICAgICAqIEByZXR1cm4gaW1wbGVtZW50YXRpb24gb2YgTmF0aXZlUm9ib3QKLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgTmF0aXZlUm9ib3QgZ2V0TmF0aXZlUm9ib3QoR3JhcGhpY3NEZXZpY2Ugc2NyZWVuKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHBsYXRmb3JtIHNwZWNpZmljIGltcGxlbWVudGF0aW9uIG9mIHRoZSBhYnN0cmFjdAotICAgICAqIGNsYXNzIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrLk5hdGl2ZUlNLgotICAgICAqIEByZXR1cm4gaW1wbGVtZW50YXRpb24gb2YgTmF0aXZlSU0KLSAgICAgKi8KLSAgICBwdWJsaWMgYWJzdHJhY3QgTmF0aXZlSU0gZ2V0TmF0aXZlSU0oKTsKLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9XaW5kb3dGYWN0b3J5LmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2F3dC93dGsvV2luZG93RmFjdG9yeS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyMzYwNGRhLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYXd0L3d0ay9XaW5kb3dGYWN0b3J5LmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4NSArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIE1pa2hhaWwgRGFuaWxvdgotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3Qud3RrOwotCi1pbXBvcnQgamF2YS5hd3QuRGltZW5zaW9uOwotaW1wb3J0IGphdmEuYXd0LlBvaW50OwotCi0vKioKLSAqIFByb3ZpZGVzIGZhY3RvcnkgZm9yIE5hdGl2ZVdpbmRvdwotICovCi1wdWJsaWMgaW50ZXJmYWNlIFdpbmRvd0ZhY3RvcnkgewotICAgIC8qKgotICAgICAqIENyZWF0ZXMgYW5kIHJldHVybnMgTmF0aXZlV2luZG93IHdpdGggZGVzaXJlZAotICAgICAqIGNyZWF0aW9uIHBhcmFtcwotICAgICAqCi0gICAgICogQHBhcmFtIHAgLSBpbml0aWFsIHdpbmRvdyBwcm9wZXJ0aWVzCi0gICAgICogQHJldHVybiBjcmVhdGVkIHdpbmRvdwotICAgICAqLwotICAgIE5hdGl2ZVdpbmRvdyBjcmVhdGVXaW5kb3coQ3JlYXRpb25QYXJhbXMgcCk7Ci0gICAgLyoqCi0gICAgICogQ3JlYXRlIE5hdGl2ZVdpbmRvdyBpbnN0YW5jZSBjb25uZWN0ZWQgdG8gZXhpc3RpbmcgbmF0aXZlIHJlc291cmNlCi0gICAgICogQHBhcmFtIG5hdGl2ZVdpbmRvd0lkIC0gaWQgb2YgZXhpc3Rpbmcgd2luZG93Ci0gICAgICogQHJldHVybiBjcmVhdGVkIE5hdGl2ZVdpbmRvdyBpbnN0YW5jZQotICAgICAqLwotICAgIE5hdGl2ZVdpbmRvdyBhdHRhY2hXaW5kb3cobG9uZyBuYXRpdmVXaW5kb3dJZCk7Ci0gICAgLyoqCi0gICAgICogUmV0dXJucyBOYXRpdmVXaW5kb3cgaW5zdGFuY2UgaWYgY3JlYXRlZCBieSB0aGlzIGluc3RhbmNlIG9mCi0gICAgICogV2luZG93RmFjdG9yeSwgb3RoZXJ3aXNlIG51bGwKLSAgICAgKgotICAgICAqIEBwYXJhbSBpZCAtIEhXTkQgb24gV2luZG93cyB4d2luZG93IG9uIFgKLSAgICAgKiBAcmV0dXJuIE5hdGl2ZVdpbmRvdyBvciBudWxsIGlmIHVua25vd24KLSAgICAgKi8KLSAgICBOYXRpdmVXaW5kb3cgZ2V0V2luZG93QnlJZChsb25nIGlkKTsKLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIE5hdGl2ZVdpbmRvdyBpbnN0YW5jZSBvZiB0aGUgdG9wLWxldmVsIHdpbmRvdwotICAgICAqIHRoYXQgY29udGFpbnMgYSBzcGVjaWZpZWQgcG9pbnQgYW5kIHdhcwotICAgICAqIGNyZWF0ZWQgYnkgdGhpcyBpbnN0YW5jZSBvZiBXaW5kb3dGYWN0b3J5Ci0gICAgICogQHBhcmFtIHAgLSBQb2ludCB0byBjaGVjawotICAgICAqIEByZXR1cm4gTmF0aXZlV2luZG93IG9yIG51bGwgaWYgdGhlIHBvaW50IGlzCi0gICAgICogbm90IHdpdGhpbiBhIHdpbmRvdyBjcmVhdGVkIGJ5IHRoaXMgV2luZG93RmFjdG9yeQotICAgICAqLwotICAgIE5hdGl2ZVdpbmRvdyBnZXRXaW5kb3dGcm9tUG9pbnQoUG9pbnQgcCk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHdoZXRoZXIgbmF0aXZlIHN5c3RlbSBzdXBwb3J0cyB0aGUgc3RhdGUgZm9yIHdpbmRvd3MuCi0gICAgICogVGhpcyBtZXRob2QgdGVsbHMgd2hldGhlciB0aGUgVUkgY29uY2VwdCBvZiwgc2F5LCBtYXhpbWl6YXRpb24gb3IgaWNvbmlmaWNhdGlvbiBpcyBzdXBwb3J0ZWQuCi0gICAgICogSXQgd2lsbCBhbHdheXMgcmV0dXJuIGZhbHNlIGZvciAiY29tcG91bmQiIHN0YXRlcyBsaWtlIEZyYW1lLklDT05JRklFRHxGcmFtZS5NQVhJTUlaRURfVkVSVC4KLSAgICAgKiBJbiBvdGhlciB3b3JkcywgdGhlIHJ1bGUgb2YgdGh1bWIgaXMgdGhhdCBvbmx5IHF1ZXJpZXMgd2l0aCBhIHNpbmdsZSBmcmFtZSBzdGF0ZQotICAgICAqIGNvbnN0YW50IGFzIGFuIGFyZ3VtZW50IGFyZSBtZWFuaW5nZnVsLgotICAgICAqCi0gICAgICogQHBhcmFtIHN0YXRlIC0gb25lIG9mIG5hbWVkIGZyYW1lIHN0YXRlIGNvbnN0YW50cy4KLSAgICAgKiBAcmV0dXJuIHRydWUgaXMgdGhpcyBmcmFtZSBzdGF0ZSBpcyBzdXBwb3J0ZWQgYnkgdGhpcyBUb29sa2l0IGltcGxlbWVudGF0aW9uLCBmYWxzZSBvdGhlcndpc2UuCi0gICAgICovCi0gICAgYm9vbGVhbiBpc1dpbmRvd1N0YXRlU3VwcG9ydGVkKGludCBzdGF0ZSk7Ci0KLSAgICAvKioKLSAgICAgKiBAc2VlIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuQ29tcG9uZW50SW50ZXJuYWxzCi0gICAgICovCi0gICAgdm9pZCBzZXRDYXJldFBvc2l0aW9uKGludCB4LCBpbnQgeSk7Ci0KLSAgICAvKioKLSAgICAgKiBSZXF1ZXN0IHNpemUgb2YgYXJiaXRyYXJ5IG5hdGl2ZSB3aW5kb3cKLSAgICAgKiBAcGFyYW0gaWQgLSB3aW5kb3cgSUQKLSAgICAgKiBAcmV0dXJuIHdpbmRvdyBzaXplCi0gICAgICovCi0gICAgRGltZW5zaW9uIGdldFdpbmRvd1NpemVCeUlkKGxvbmcgaWQpOwotfQpcIE5vIG5ld2xpbmUgYXQgZW5kIG9mIGZpbGUKZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYmVhbnMvaW50ZXJuYWwvbmxzL01lc3NhZ2VzLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L2JlYW5zL2ludGVybmFsL25scy9NZXNzYWdlcy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1MWU4MTY4Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYmVhbnMvaW50ZXJuYWwvbmxzL01lc3NhZ2VzLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxNTEgKzAsMCBAQAotLyogCi0gKiBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICogCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKiAKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8qCi0gKiBUSEUgRklMRSBIQVMgQkVFTiBBVVRPR0VORVJBVEVEIEJZIE1TR1RPT0wgVE9PTC4KLSAqIEFsbCBjaGFuZ2VzIG1hZGUgdG8gdGhpcyBmaWxlIG1hbnVhbGx5IHdpbGwgYmUgb3ZlcndyaXR0ZW4gCi0gKiBpZiB0aGlzIHRvb2wgcnVucyBhZ2Fpbi4gQmV0dGVyIG1ha2UgY2hhbmdlcyBpbiB0aGUgdGVtcGxhdGUgZmlsZS4KLSAqLwotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS5iZWFucy5pbnRlcm5hbC5ubHM7Ci0KLQotaW1wb3J0IGphdmEuc2VjdXJpdHkuQWNjZXNzQ29udHJvbGxlcjsKLWltcG9ydCBqYXZhLnNlY3VyaXR5LlByaXZpbGVnZWRBY3Rpb247Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLWltcG9ydCBqYXZhLnV0aWwuTWlzc2luZ1Jlc291cmNlRXhjZXB0aW9uOwotaW1wb3J0IGphdmEudXRpbC5SZXNvdXJjZUJ1bmRsZTsKLQotLy8gQkVHSU4gYW5kcm9pZC1kZWxldGVkCi0vKgotICogRm9yIEFuZHJvaWQsIHRoaXMgbW9kdWxlIGlzIGEgc2VwYXJhdGUgbGlicmFyeSBhbmQgbm90IHBhcnQgb2YgdGhlCi0gKiBib290IGNsYXNzcGF0aCwgc28gaXRzIHJlc291cmNlcyB3b24ndCBiZSBmb3VuZCBvbiB0aGUgYm9vdCBjbGFzc3BhdGgKLSAqIGFzIGlzIGFzc3VtZWQgYnkgTXNnSGVscC5nZXRTdHJpbmcoKS4gV2UgaW5zdGVhZCB1c2UgYSBsb2NhbCBNc2dIZWxwCi0gKiB3aGljaCBib3R0b21zIG91dCBpbiBhIGNhbGwgdG8gdGhlIHVzZWZ1bCBwYXJ0IG9mIGl0cyBsb3dlci1sZXZlbAotICogbmFtZXNha2UuCi0gKi8KLS8vaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5rZXJuZWwudm0uVk07Ci0vL2ltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkubHVuaS51dGlsLk1zZ0hlbHA7Ci0vLyBFTkQgYW5kcm9pZC1kZWxldGVkCi0KLS8qKgotICogVGhpcyBjbGFzcyByZXRyaWV2ZXMgc3RyaW5ncyBmcm9tIGEgcmVzb3VyY2UgYnVuZGxlIGFuZCByZXR1cm5zIHRoZW0sCi0gKiBmb3JtYXR0aW5nIHRoZW0gd2l0aCBNZXNzYWdlRm9ybWF0IHdoZW4gcmVxdWlyZWQuCi0gKiA8cD4KLSAqIEl0IGlzIHVzZWQgYnkgdGhlIHN5c3RlbSBjbGFzc2VzIHRvIHByb3ZpZGUgbmF0aW9uYWwgbGFuZ3VhZ2Ugc3VwcG9ydCwgYnkKLSAqIGxvb2tpbmcgdXAgbWVzc2FnZXMgaW4gdGhlIDxjb2RlPgotICogICAgb3JnLmFwYWNoZS5oYXJtb255LmJlYW5zLmludGVybmFsLm5scy5tZXNzYWdlcwotICogPC9jb2RlPgotICogcmVzb3VyY2UgYnVuZGxlLiBOb3RlIHRoYXQgaWYgdGhpcyBmaWxlIGlzIG5vdCBhdmFpbGFibGUsIG9yIGFuIGludmFsaWQga2V5Ci0gKiBpcyBsb29rZWQgdXAsIG9yIHJlc291cmNlIGJ1bmRsZSBzdXBwb3J0IGlzIG5vdCBhdmFpbGFibGUsIHRoZSBrZXkgaXRzZWxmCi0gKiB3aWxsIGJlIHJldHVybmVkIGFzIHRoZSBhc3NvY2lhdGVkIG1lc3NhZ2UuIFRoaXMgbWVhbnMgdGhhdCB0aGUgPGVtPktFWTwvZW0+Ci0gKiBzaG91bGQgYSByZWFzb25hYmxlIGh1bWFuLXJlYWRhYmxlIChlbmdsaXNoKSBzdHJpbmcuCi0gKiAKLSAqLwotcHVibGljIGNsYXNzIE1lc3NhZ2VzIHsKLQotICAgIC8vIEJFR0lOIGFuZHJvaWQtZGVsZXRlZAotICAgIC8vIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBzUmVzb3VyY2UgPQotICAgIC8vICAgICAib3JnLmFwYWNoZS5oYXJtb255LmJlYW5zLmludGVybmFsLm5scy5tZXNzYWdlcyI7IC8vJE5PTi1OTFMtMSQKLSAgICAvLyBFTkQgYW5kcm9pZC1kZWxldGVkCi0KLSAgICAvKioKLSAgICAgKiBSZXRyaWV2ZXMgYSBtZXNzYWdlIHdoaWNoIGhhcyBubyBhcmd1bWVudHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1zZwotICAgICAqICAgICAgICAgICAgU3RyaW5nIHRoZSBrZXkgdG8gbG9vayB1cC4KLSAgICAgKiBAcmV0dXJuIFN0cmluZyB0aGUgbWVzc2FnZSBmb3IgdGhhdCBrZXkgaW4gdGhlIHN5c3RlbSBtZXNzYWdlIGJ1bmRsZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgcHVibGljIFN0cmluZyBnZXRTdHJpbmcoU3RyaW5nIG1zZykgewotICAgICAgICAvLyBCRUdJTiBhbmRyb2lkLWNoYW5nZWQKLSAgICAgICAgcmV0dXJuIE1zZ0hlbHAuZ2V0U3RyaW5nKG1zZyk7Ci0gICAgICAgIC8vIEVORCBhbmRyb2lkLWNoYW5nZWQKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXRyaWV2ZXMgYSBtZXNzYWdlIHdoaWNoIHRha2VzIDEgYXJndW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1zZwotICAgICAqICAgICAgICAgICAgU3RyaW5nIHRoZSBrZXkgdG8gbG9vayB1cC4KLSAgICAgKiBAcGFyYW0gYXJnCi0gICAgICogICAgICAgICAgICBPYmplY3QgdGhlIG9iamVjdCB0byBpbnNlcnQgaW4gdGhlIGZvcm1hdHRlZCBvdXRwdXQuCi0gICAgICogQHJldHVybiBTdHJpbmcgdGhlIG1lc3NhZ2UgZm9yIHRoYXQga2V5IGluIHRoZSBzeXN0ZW0gbWVzc2FnZSBidW5kbGUuCi0gICAgICovCi0gICAgc3RhdGljIHB1YmxpYyBTdHJpbmcgZ2V0U3RyaW5nKFN0cmluZyBtc2csIE9iamVjdCBhcmcpIHsKLSAgICAgICAgcmV0dXJuIGdldFN0cmluZyhtc2csIG5ldyBPYmplY3RbXSB7IGFyZyB9KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXRyaWV2ZXMgYSBtZXNzYWdlIHdoaWNoIHRha2VzIDEgaW50ZWdlciBhcmd1bWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXNnCi0gICAgICogICAgICAgICAgICBTdHJpbmcgdGhlIGtleSB0byBsb29rIHVwLgotICAgICAqIEBwYXJhbSBhcmcKLSAgICAgKiAgICAgICAgICAgIGludCB0aGUgaW50ZWdlciB0byBpbnNlcnQgaW4gdGhlIGZvcm1hdHRlZCBvdXRwdXQuCi0gICAgICogQHJldHVybiBTdHJpbmcgdGhlIG1lc3NhZ2UgZm9yIHRoYXQga2V5IGluIHRoZSBzeXN0ZW0gbWVzc2FnZSBidW5kbGUuCi0gICAgICovCi0gICAgc3RhdGljIHB1YmxpYyBTdHJpbmcgZ2V0U3RyaW5nKFN0cmluZyBtc2csIGludCBhcmcpIHsKLSAgICAgICAgcmV0dXJuIGdldFN0cmluZyhtc2csIG5ldyBPYmplY3RbXSB7IEludGVnZXIudG9TdHJpbmcoYXJnKSB9KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXRyaWV2ZXMgYSBtZXNzYWdlIHdoaWNoIHRha2VzIDEgY2hhcmFjdGVyIGFyZ3VtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBtc2cKLSAgICAgKiAgICAgICAgICAgIFN0cmluZyB0aGUga2V5IHRvIGxvb2sgdXAuCi0gICAgICogQHBhcmFtIGFyZwotICAgICAqICAgICAgICAgICAgY2hhciB0aGUgY2hhcmFjdGVyIHRvIGluc2VydCBpbiB0aGUgZm9ybWF0dGVkIG91dHB1dC4KLSAgICAgKiBAcmV0dXJuIFN0cmluZyB0aGUgbWVzc2FnZSBmb3IgdGhhdCBrZXkgaW4gdGhlIHN5c3RlbSBtZXNzYWdlIGJ1bmRsZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgcHVibGljIFN0cmluZyBnZXRTdHJpbmcoU3RyaW5nIG1zZywgY2hhciBhcmcpIHsKLSAgICAgICAgcmV0dXJuIGdldFN0cmluZyhtc2csIG5ldyBPYmplY3RbXSB7IFN0cmluZy52YWx1ZU9mKGFyZykgfSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0cmlldmVzIGEgbWVzc2FnZSB3aGljaCB0YWtlcyAyIGFyZ3VtZW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXNnCi0gICAgICogICAgICAgICAgICBTdHJpbmcgdGhlIGtleSB0byBsb29rIHVwLgotICAgICAqIEBwYXJhbSBhcmcxCi0gICAgICogICAgICAgICAgICBPYmplY3QgYW4gb2JqZWN0IHRvIGluc2VydCBpbiB0aGUgZm9ybWF0dGVkIG91dHB1dC4KLSAgICAgKiBAcGFyYW0gYXJnMgotICAgICAqICAgICAgICAgICAgT2JqZWN0IGFub3RoZXIgb2JqZWN0IHRvIGluc2VydCBpbiB0aGUgZm9ybWF0dGVkIG91dHB1dC4KLSAgICAgKiBAcmV0dXJuIFN0cmluZyB0aGUgbWVzc2FnZSBmb3IgdGhhdCBrZXkgaW4gdGhlIHN5c3RlbSBtZXNzYWdlIGJ1bmRsZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgcHVibGljIFN0cmluZyBnZXRTdHJpbmcoU3RyaW5nIG1zZywgT2JqZWN0IGFyZzEsIE9iamVjdCBhcmcyKSB7Ci0gICAgICAgIHJldHVybiBnZXRTdHJpbmcobXNnLCBuZXcgT2JqZWN0W10geyBhcmcxLCBhcmcyIH0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHJpZXZlcyBhIG1lc3NhZ2Ugd2hpY2ggdGFrZXMgc2V2ZXJhbCBhcmd1bWVudHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1zZwotICAgICAqICAgICAgICAgICAgU3RyaW5nIHRoZSBrZXkgdG8gbG9vayB1cC4KLSAgICAgKiBAcGFyYW0gYXJncwotICAgICAqICAgICAgICAgICAgT2JqZWN0W10gdGhlIG9iamVjdHMgdG8gaW5zZXJ0IGluIHRoZSBmb3JtYXR0ZWQgb3V0cHV0LgotICAgICAqIEByZXR1cm4gU3RyaW5nIHRoZSBtZXNzYWdlIGZvciB0aGF0IGtleSBpbiB0aGUgc3lzdGVtIG1lc3NhZ2UgYnVuZGxlLgotICAgICAqLwotICAgIHN0YXRpYyBwdWJsaWMgU3RyaW5nIGdldFN0cmluZyhTdHJpbmcgbXNnLCBPYmplY3RbXSBhcmdzKSB7Ci0gICAgICAgIC8vIEJFR0lOIGFuZHJvaWQtY2hhbmdlZAotICAgICAgICByZXR1cm4gTXNnSGVscC5nZXRTdHJpbmcobXNnLCBhcmdzKTsKLSAgICAgICAgLy8gRU5EIGFuZHJvaWQtY2hhbmdlZAotICAgIH0KLQotICAgIC8vIEJFR0lOIGFuZHJvaWQtbm90ZQotICAgIC8vIER1cGxpY2F0ZSBjb2RlIHdhcyBkcm9wcGVkIGluIGZhdm9yIG9mIHVzaW5nIE1zZ0hlbHAuCi0gICAgLy8gRU5EIGFuZHJvaWQtbm90ZQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9iZWFucy9pbnRlcm5hbC9ubHMvTXNnSGVscC5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS9iZWFucy9pbnRlcm5hbC9ubHMvTXNnSGVscC5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2OGZhYWJmLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkvYmVhbnMvaW50ZXJuYWwvbmxzL01zZ0hlbHAuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDg2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8qCi0gKiBUaGlzIGltcGxlbWVudGF0aW9uIGlzIGJhc2VkIG9uIHRoZSBjbGFzcyBvZiB0aGUgc2FtZSBuYW1lIGluCi0gKiBvcmcuYXBhY2hlLmhhcm1vbnkubHVuaS51dGlsLgotICovCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LmJlYW5zLmludGVybmFsLm5sczsKLQotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLnV0aWwubG9nZ2luZy5Mb2dnZXI7Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLWltcG9ydCBqYXZhLnV0aWwuUHJvcGVydHlSZXNvdXJjZUJ1bmRsZTsKLWltcG9ydCBqYXZhLnV0aWwuUmVzb3VyY2VCdW5kbGU7Ci1pbXBvcnQgamF2YS51dGlsLk1pc3NpbmdSZXNvdXJjZUV4Y2VwdGlvbjsKLQotLyoqCi0gKiBUaGlzIGNsYXNzIGNvbnRhaW5zIGhlbHBlciBtZXRob2RzIGZvciBsb2FkaW5nIHJlc291cmNlIGJ1bmRsZXMgYW5kCi0gKiBmb3JtYXR0aW5nIGV4dGVybmFsIG1lc3NhZ2Ugc3RyaW5ncy4KLSAqLwotcHVibGljIGZpbmFsIGNsYXNzIE1zZ0hlbHAgewotICAgIC8qKiBuYW1lIG9mIHRoZSByZXNvdXJjZSBmb3IgdGhpcyBjbGFzcyAqLwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBSRVNPVVJDRV9OQU1FID0KLSAgICAgICAgIi9vcmcvYXBhY2hlL2hhcm1vbnkvYmVhbnMvaW50ZXJuYWwvbmxzL21lc3NhZ2VzLnByb3BlcnRpZXMiOwotCi0gICAgLyoqIHRoZSByZXNvdXJjZSBidW5kbGUgZm9yIHRoaXMgY2xhc3MgKi8KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBSZXNvdXJjZUJ1bmRsZSBUSEVfQlVORExFOwotCi0gICAgc3RhdGljIHsKLSAgICAgICAgUmVzb3VyY2VCdW5kbGUgcmIgPSBudWxsOwotCi0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBJbnB1dFN0cmVhbSBpbiA9IE1zZ0hlbHAuY2xhc3MuZ2V0UmVzb3VyY2VBc1N0cmVhbSgKLSAgICAgICAgICAgICAgICAgICAgUkVTT1VSQ0VfTkFNRSk7Ci0gICAgICAgICAgICByYiA9IG5ldyBQcm9wZXJ0eVJlc291cmNlQnVuZGxlKGluKTsKLSAgICAgICAgfSBjYXRjaCAoSU9FeGNlcHRpb24gZXgpIHsKLSAgICAgICAgICAgIExvZ2dlci5nbG9iYWwud2FybmluZygiQ291bGRuJ3QgcmVhZCByZXNvdXJjZSBidW5kbGU6ICIgKwotICAgICAgICAgICAgICAgICAgICBleCk7Ci0gICAgICAgIH0gY2F0Y2ggKFJ1bnRpbWVFeGNlcHRpb24gZXgpIHsKLSAgICAgICAgICAgIC8vIFNob3VsZG4ndCBoYXBwZW4sIGJ1dCBkZWFsIGF0IGxlYXN0IHNvbWV3aGF0IGdyYWNlZnVsbHkuCi0gICAgICAgICAgICBMb2dnZXIuZ2xvYmFsLndhcm5pbmcoIkNvdWxkbid0IGZpbmQgcmVzb3VyY2UgYnVuZGxlOiAiICsKLSAgICAgICAgICAgICAgICAgICAgZXgpOwotICAgICAgICB9Ci0KLSAgICAgICAgVEhFX0JVTkRMRSA9IHJiOwotICAgIH0KLSAgICAKLSAgICBwdWJsaWMgc3RhdGljIFN0cmluZyBnZXRTdHJpbmcoU3RyaW5nIG1zZykgewotICAgICAgICBpZiAoVEhFX0JVTkRMRSA9PSBudWxsKSB7Ci0gICAgICAgICAgICByZXR1cm4gbXNnOwotICAgICAgICB9Ci0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICByZXR1cm4gVEhFX0JVTkRMRS5nZXRTdHJpbmcobXNnKTsKLSAgICAgICAgfSBjYXRjaCAoTWlzc2luZ1Jlc291cmNlRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIHJldHVybiAiTWlzc2luZyBtZXNzYWdlOiAiICsgbXNnOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIHN0YXRpYyBwdWJsaWMgU3RyaW5nIGdldFN0cmluZyhTdHJpbmcgbXNnLCBPYmplY3RbXSBhcmdzKSB7Ci0gICAgICAgIFN0cmluZyBmb3JtYXQgPSBtc2c7Ci0gICAgICAgIGlmIChUSEVfQlVORExFICE9IG51bGwpIHsKLSAgICAgICAgICAgIHRyeSB7Ci0gICAgICAgICAgICAgICAgZm9ybWF0ID0gVEhFX0JVTkRMRS5nZXRTdHJpbmcobXNnKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKE1pc3NpbmdSZXNvdXJjZUV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Nc2dIZWxwLmZvcm1hdChmb3JtYXQsIGFyZ3MpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL2ludGVybmFsL25scy9NZXNzYWdlcy5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vaW50ZXJuYWwvbmxzL01lc3NhZ2VzLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQ5OGUxYmIuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vaW50ZXJuYWwvbmxzL01lc3NhZ2VzLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMjQgKzAsMCBAQAotLyogCi0gKiBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICogCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKiAKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8qCi0gKiBUSEUgRklMRSBIQVMgQkVFTiBBVVRPR0VORVJBVEVEIEJZIE1TR1RPT0wgVE9PTC4KLSAqIEFsbCBjaGFuZ2VzIG1hZGUgdG8gdGhpcyBmaWxlIG1hbnVhbGx5IHdpbGwgYmUgb3ZlcndyaXR0ZW4gCi0gKiBpZiB0aGlzIHRvb2wgcnVucyBhZ2Fpbi4gQmV0dGVyIG1ha2UgY2hhbmdlcyBpbiB0aGUgdGVtcGxhdGUgZmlsZS4KLSAqLwotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8uaW50ZXJuYWwubmxzOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255Lmx1bmkudXRpbC5Nc2dIZWxwOwotCi0vKioKLSAqIFRoaXMgY2xhc3MgcmV0cmlldmVzIHN0cmluZ3MgZnJvbSBhIHJlc291cmNlIGJ1bmRsZSBhbmQgcmV0dXJucyB0aGVtLAotICogZm9ybWF0dGluZyB0aGVtIHdpdGggTWVzc2FnZUZvcm1hdCB3aGVuIHJlcXVpcmVkLgotICogPHA+Ci0gKiBJdCBpcyB1c2VkIGJ5IHRoZSBzeXN0ZW0gY2xhc3NlcyB0byBwcm92aWRlIG5hdGlvbmFsIGxhbmd1YWdlIHN1cHBvcnQsIGJ5Ci0gKiBsb29raW5nIHVwIG1lc3NhZ2VzIGluIHRoZSA8Y29kZT4KLSAqICAgIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8uaW50ZXJuYWwubmxzLm1lc3NhZ2VzCi0gKiA8L2NvZGU+Ci0gKiByZXNvdXJjZSBidW5kbGUuIE5vdGUgdGhhdCBpZiB0aGlzIGZpbGUgaXMgbm90IGF2YWlsYWJsZSwgb3IgYW4gaW52YWxpZCBrZXkKLSAqIGlzIGxvb2tlZCB1cCwgb3IgcmVzb3VyY2UgYnVuZGxlIHN1cHBvcnQgaXMgbm90IGF2YWlsYWJsZSwgdGhlIGtleSBpdHNlbGYKLSAqIHdpbGwgYmUgcmV0dXJuZWQgYXMgdGhlIGFzc29jaWF0ZWQgbWVzc2FnZS4gVGhpcyBtZWFucyB0aGF0IHRoZSA8ZW0+S0VZPC9lbT4KLSAqIHNob3VsZCBhIHJlYXNvbmFibGUgaHVtYW4tcmVhZGFibGUgKGVuZ2xpc2gpIHN0cmluZy4KLSAqIAotICovCi1wdWJsaWMgY2xhc3MgTWVzc2FnZXMgewotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIHNSZXNvdXJjZSA9Ci0gICAgICAgICJvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLmludGVybmFsLm5scy5tZXNzYWdlcyI7IC8vJE5PTi1OTFMtMSQKLQotICAgIC8qKgotICAgICAqIFJldHJpZXZlcyBhIG1lc3NhZ2Ugd2hpY2ggaGFzIG5vIGFyZ3VtZW50cy4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXNnCi0gICAgICogICAgICAgICAgICBTdHJpbmcgdGhlIGtleSB0byBsb29rIHVwLgotICAgICAqIEByZXR1cm4gU3RyaW5nIHRoZSBtZXNzYWdlIGZvciB0aGF0IGtleSBpbiB0aGUgc3lzdGVtIG1lc3NhZ2UgYnVuZGxlLgotICAgICAqLwotICAgIHN0YXRpYyBwdWJsaWMgU3RyaW5nIGdldFN0cmluZyhTdHJpbmcgbXNnKSB7Ci0gICAgICAgIHJldHVybiBNc2dIZWxwLmdldFN0cmluZyhzUmVzb3VyY2UsIG1zZyk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0cmlldmVzIGEgbWVzc2FnZSB3aGljaCB0YWtlcyAxIGFyZ3VtZW50LgotICAgICAqIAotICAgICAqIEBwYXJhbSBtc2cKLSAgICAgKiAgICAgICAgICAgIFN0cmluZyB0aGUga2V5IHRvIGxvb2sgdXAuCi0gICAgICogQHBhcmFtIGFyZwotICAgICAqICAgICAgICAgICAgT2JqZWN0IHRoZSBvYmplY3QgdG8gaW5zZXJ0IGluIHRoZSBmb3JtYXR0ZWQgb3V0cHV0LgotICAgICAqIEByZXR1cm4gU3RyaW5nIHRoZSBtZXNzYWdlIGZvciB0aGF0IGtleSBpbiB0aGUgc3lzdGVtIG1lc3NhZ2UgYnVuZGxlLgotICAgICAqLwotICAgIHN0YXRpYyBwdWJsaWMgU3RyaW5nIGdldFN0cmluZyhTdHJpbmcgbXNnLCBPYmplY3QgYXJnKSB7Ci0gICAgICAgIHJldHVybiBnZXRTdHJpbmcobXNnLCBuZXcgT2JqZWN0W10geyBhcmcgfSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0cmlldmVzIGEgbWVzc2FnZSB3aGljaCB0YWtlcyAxIGludGVnZXIgYXJndW1lbnQuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1zZwotICAgICAqICAgICAgICAgICAgU3RyaW5nIHRoZSBrZXkgdG8gbG9vayB1cC4KLSAgICAgKiBAcGFyYW0gYXJnCi0gICAgICogICAgICAgICAgICBpbnQgdGhlIGludGVnZXIgdG8gaW5zZXJ0IGluIHRoZSBmb3JtYXR0ZWQgb3V0cHV0LgotICAgICAqIEByZXR1cm4gU3RyaW5nIHRoZSBtZXNzYWdlIGZvciB0aGF0IGtleSBpbiB0aGUgc3lzdGVtIG1lc3NhZ2UgYnVuZGxlLgotICAgICAqLwotICAgIHN0YXRpYyBwdWJsaWMgU3RyaW5nIGdldFN0cmluZyhTdHJpbmcgbXNnLCBpbnQgYXJnKSB7Ci0gICAgICAgIHJldHVybiBnZXRTdHJpbmcobXNnLCBuZXcgT2JqZWN0W10geyBJbnRlZ2VyLnRvU3RyaW5nKGFyZykgfSk7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogUmV0cmlldmVzIGEgbWVzc2FnZSB3aGljaCB0YWtlcyAxIGNoYXJhY3RlciBhcmd1bWVudC4KLSAgICAgKiAKLSAgICAgKiBAcGFyYW0gbXNnCi0gICAgICogICAgICAgICAgICBTdHJpbmcgdGhlIGtleSB0byBsb29rIHVwLgotICAgICAqIEBwYXJhbSBhcmcKLSAgICAgKiAgICAgICAgICAgIGNoYXIgdGhlIGNoYXJhY3RlciB0byBpbnNlcnQgaW4gdGhlIGZvcm1hdHRlZCBvdXRwdXQuCi0gICAgICogQHJldHVybiBTdHJpbmcgdGhlIG1lc3NhZ2UgZm9yIHRoYXQga2V5IGluIHRoZSBzeXN0ZW0gbWVzc2FnZSBidW5kbGUuCi0gICAgICovCi0gICAgc3RhdGljIHB1YmxpYyBTdHJpbmcgZ2V0U3RyaW5nKFN0cmluZyBtc2csIGNoYXIgYXJnKSB7Ci0gICAgICAgIHJldHVybiBnZXRTdHJpbmcobXNnLCBuZXcgT2JqZWN0W10geyBTdHJpbmcudmFsdWVPZihhcmcpIH0pOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIFJldHJpZXZlcyBhIG1lc3NhZ2Ugd2hpY2ggdGFrZXMgMiBhcmd1bWVudHMuCi0gICAgICogCi0gICAgICogQHBhcmFtIG1zZwotICAgICAqICAgICAgICAgICAgU3RyaW5nIHRoZSBrZXkgdG8gbG9vayB1cC4KLSAgICAgKiBAcGFyYW0gYXJnMQotICAgICAqICAgICAgICAgICAgT2JqZWN0IGFuIG9iamVjdCB0byBpbnNlcnQgaW4gdGhlIGZvcm1hdHRlZCBvdXRwdXQuCi0gICAgICogQHBhcmFtIGFyZzIKLSAgICAgKiAgICAgICAgICAgIE9iamVjdCBhbm90aGVyIG9iamVjdCB0byBpbnNlcnQgaW4gdGhlIGZvcm1hdHRlZCBvdXRwdXQuCi0gICAgICogQHJldHVybiBTdHJpbmcgdGhlIG1lc3NhZ2UgZm9yIHRoYXQga2V5IGluIHRoZSBzeXN0ZW0gbWVzc2FnZSBidW5kbGUuCi0gICAgICovCi0gICAgc3RhdGljIHB1YmxpYyBTdHJpbmcgZ2V0U3RyaW5nKFN0cmluZyBtc2csIE9iamVjdCBhcmcxLCBPYmplY3QgYXJnMikgewotICAgICAgICByZXR1cm4gZ2V0U3RyaW5nKG1zZywgbmV3IE9iamVjdFtdIHsgYXJnMSwgYXJnMiB9KTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXRyaWV2ZXMgYSBtZXNzYWdlIHdoaWNoIHRha2VzIHNldmVyYWwgYXJndW1lbnRzLgotICAgICAqIAotICAgICAqIEBwYXJhbSBtc2cKLSAgICAgKiAgICAgICAgICAgIFN0cmluZyB0aGUga2V5IHRvIGxvb2sgdXAuCi0gICAgICogQHBhcmFtIGFyZ3MKLSAgICAgKiAgICAgICAgICAgIE9iamVjdFtdIHRoZSBvYmplY3RzIHRvIGluc2VydCBpbiB0aGUgZm9ybWF0dGVkIG91dHB1dC4KLSAgICAgKiBAcmV0dXJuIFN0cmluZyB0aGUgbWVzc2FnZSBmb3IgdGhhdCBrZXkgaW4gdGhlIHN5c3RlbSBtZXNzYWdlIGJ1bmRsZS4KLSAgICAgKi8KLSAgICBzdGF0aWMgcHVibGljIFN0cmluZyBnZXRTdHJpbmcoU3RyaW5nIG1zZywgT2JqZWN0W10gYXJncykgewotICAgICAgICByZXR1cm4gTXNnSGVscC5nZXRTdHJpbmcoc1Jlc291cmNlLCBtc2csIGFyZ3MpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL2ludGVybmFsL25scy9tZXNzYWdlcy5wcm9wZXJ0aWVzIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vaW50ZXJuYWwvbmxzL21lc3NhZ2VzLnByb3BlcnRpZXMKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDhhNDlkZDguLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vaW50ZXJuYWwvbmxzL21lc3NhZ2VzLnByb3BlcnRpZXMKKysrIC9kZXYvbnVsbApAQCAtMSwxOCArMCwwIEBACi0jIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotIyBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSMgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotIyBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotIyAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotIyB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotIyAgCi0jICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0jICAKLSMgIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSMgIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0jICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSMgIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSMgIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotIyAKLQotIyBtZXNzYWdlcyBmb3IgRU4gbG9jYWxlCi1pbWFnZWlvLjE9V3JvbmcgYml0RGVwdGgtbnVtQmFuZHMgY29tcG9zaXRpb24KXCBObyBuZXdsaW5lIGF0IGVuZCBvZiBmaWxlCmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YVV0aWxzLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9tZXRhZGF0YS9JSU9NZXRhZGF0YVV0aWxzLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGNhZWVmZGQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vbWV0YWRhdGEvSUlPTWV0YWRhdGFVdGlscy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsOTQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ubWV0YWRhdGE7Ci0KLWltcG9ydCBqYXZhLmxhbmcucmVmbGVjdC5NZXRob2Q7Ci1pbXBvcnQgamF2YS5zZWN1cml0eS5BY2Nlc3NDb250cm9sbGVyOwotaW1wb3J0IGphdmEuc2VjdXJpdHkuUHJpdmlsZWdlZEFjdGlvbjsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8ubWV0YWRhdGEuSUlPTWV0YWRhdGFGb3JtYXQ7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5tZXRhZGF0YS5JSU9NZXRhZGF0YUZvcm1hdEltcGw7Ci0KLXB1YmxpYyBjbGFzcyBJSU9NZXRhZGF0YVV0aWxzIHsKLSAgICBwcml2YXRlIElJT01ldGFkYXRhVXRpbHMoKSB7fSAKLQotICAgIHB1YmxpYyBzdGF0aWMgSUlPTWV0YWRhdGFGb3JtYXQgaW5zdGFudGlhdGVNZXRhZGF0YUZvcm1hdCgKLSAgICAgICAgICAgIFN0cmluZyBmb3JtYXROYW1lLCBib29sZWFuIHN0YW5kYXJkRm9ybWF0U3VwcG9ydGVkLAotICAgICAgICAgICAgU3RyaW5nIG5hdGl2ZU1ldGFkYXRhRm9ybWF0TmFtZSwgU3RyaW5nIG5hdGl2ZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lLAotICAgICAgICAgICAgU3RyaW5nIFtdIGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lcywgU3RyaW5nIFtdIGV4dHJhTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzCi0gICAgKSB7Ci0gICAgICAgIGlmIChmb3JtYXROYW1lID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImZvcm1hdE5hbWUgPT0gbnVsbCEiKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoZm9ybWF0TmFtZS5lcXVhbHMoSUlPTWV0YWRhdGFGb3JtYXRJbXBsLnN0YW5kYXJkTWV0YWRhdGFGb3JtYXROYW1lKSkgewotICAgICAgICAgICAgaWYgKHN0YW5kYXJkRm9ybWF0U3VwcG9ydGVkKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIElJT01ldGFkYXRhRm9ybWF0SW1wbC5nZXRTdGFuZGFyZEZvcm1hdEluc3RhbmNlKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBTdHJpbmcgY2xhc3NOYW1lID0gbnVsbDsKLQotICAgICAgICBpZiAoZm9ybWF0TmFtZS5lcXVhbHMobmF0aXZlTWV0YWRhdGFGb3JtYXROYW1lKSkgewotICAgICAgICAgICAgY2xhc3NOYW1lID0gbmF0aXZlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWU7Ci0gICAgICAgIH0gZWxzZSBpZiAoZXh0cmFNZXRhZGF0YUZvcm1hdE5hbWVzICE9IG51bGwpIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgZXh0cmFNZXRhZGF0YUZvcm1hdE5hbWVzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGZvcm1hdE5hbWUuZXF1YWxzKGV4dHJhTWV0YWRhdGFGb3JtYXROYW1lc1tpXSkpIHsKLSAgICAgICAgICAgICAgICAgICAgY2xhc3NOYW1lID0gZXh0cmFNZXRhZGF0YUZvcm1hdENsYXNzTmFtZXNbaV07Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChjbGFzc05hbWUgPT0gbnVsbCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiVW5zdXBwb3J0ZWQgZm9ybWF0IG5hbWUiKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIEdldCB0aGUgY29udGV4dCBjbGFzcyBsb2FkZXIgYW5kIHRyeSB0byB1c2UgaXQgZmlyc3QKLSAgICAgICAgQ2xhc3NMb2FkZXIgY29udGV4dENsYXNzbG9hZGVyID0gQWNjZXNzQ29udHJvbGxlci5kb1ByaXZpbGVnZWQoCi0gICAgICAgICAgICAgICAgbmV3IFByaXZpbGVnZWRBY3Rpb248Q2xhc3NMb2FkZXI+KCkgewotICAgICAgICAgICAgICAgICAgICBwdWJsaWMgQ2xhc3NMb2FkZXIgcnVuKCkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIFRocmVhZC5jdXJyZW50VGhyZWFkKCkuZ2V0Q29udGV4dENsYXNzTG9hZGVyKCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgfSk7Ci0KLSAgICAgICAgQ2xhc3MgY2xzOwotCi0gICAgICAgIHRyeSB7Ci0gICAgICAgICAgICBjbHMgPSBDbGFzcy5mb3JOYW1lKGNsYXNzTmFtZSwgdHJ1ZSwgY29udGV4dENsYXNzbG9hZGVyKTsKLSAgICAgICAgfSBjYXRjaCAoQ2xhc3NOb3RGb3VuZEV4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICB0cnkgewotICAgICAgICAgICAgICAgIC8vIFVzZSBjdXJyZW50IGNsYXNzIGxvYWRlcgotICAgICAgICAgICAgICAgIGNscyA9IENsYXNzLmZvck5hbWUoY2xhc3NOYW1lKTsKLSAgICAgICAgICAgIH0gY2F0Y2ggKENsYXNzTm90Rm91bmRFeGNlcHRpb24gZTEpIHsKLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uICgiQ2FuJ3Qgb2J0YWluIGZvcm1hdCIpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIC8vPz8/QVdUOgotICAgICAgICAgICAgLy9NZXRob2QgZ2V0SW5zdGFuY2UgPSBjbHMuZ2V0TWV0aG9kKCJnZXRJbnN0YW5jZSIpOwotICAgICAgICAgICAgLy9yZXR1cm4gKElJT01ldGFkYXRhRm9ybWF0KSBnZXRJbnN0YW5jZS5pbnZva2UobnVsbCk7Ci0gICAgICAgICAgICByZXR1cm4gbnVsbDsKLSAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKLSAgICAgICAgICAgIElsbGVnYWxTdGF0ZUV4Y2VwdGlvbiBlMSA9IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oIkNhbid0IG9idGFpbiBmb3JtYXQiKTsKLSAgICAgICAgICAgIGUxLmluaXRDYXVzZShlKTsgLy8gQWRkIHNvbWUgZGV0YWlscyB0byB0aGUgbWVzc2FnZQotICAgICAgICAgICAgdGhyb3cgZTE7Ci0gICAgICAgIH0KLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSUlTRGVjb2RpbmdJbWFnZVNvdXJjZS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9qcGVnL0lJU0RlY29kaW5nSW1hZ2VTb3VyY2UuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMDUxZjkwNi4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSUlTRGVjb2RpbmdJbWFnZVNvdXJjZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTE1ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4yICQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMuanBlZzsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uc3RyZWFtLkltYWdlSW5wdXRTdHJlYW07Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmltYWdlLkRlY29kaW5nSW1hZ2VTb3VyY2U7Ci0KLWltcG9ydCBqYXZhLmlvLklucHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci0KLS8qKgotICogVGhpcyBhbGxvd3MgdXNhZ2Ugb2YgdGhlIGphdmEyZCBqcGVnZGVjb2RlciB3aXRoIEltYWdlSW5wdXRTdHJlYW0gaW4KLSAqIHRoZSBKUEVHSW1hZ2VSZWFkZXIuIFRlbXBvcmFyeSwgb25seSB0byBtYWtlIEpQRUdJbWFnZVJlYWRlciNyZWFkKC4uKQotICogd29ya2luZy4KLSAqCi0gKi8KLXB1YmxpYyBjbGFzcyBJSVNEZWNvZGluZ0ltYWdlU291cmNlIGV4dGVuZHMgRGVjb2RpbmdJbWFnZVNvdXJjZSB7Ci0KLSAgICBwcml2YXRlIGZpbmFsIElucHV0U3RyZWFtIGlzOwotCi0gICAgcHVibGljIElJU0RlY29kaW5nSW1hZ2VTb3VyY2UoSW1hZ2VJbnB1dFN0cmVhbSBpaXMpIHsKLSAgICAgICAgaXMgPSBuZXcgSUlTVG9JbnB1dFN0cmVhbVdyYXBwZXIoaWlzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwcm90ZWN0ZWQgYm9vbGVhbiBjaGVja0Nvbm5lY3Rpb24oKSB7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHByb3RlY3RlZCBJbnB1dFN0cmVhbSBnZXRJbnB1dFN0cmVhbSgpIHsKLSAgICAgICAgcmV0dXJuIGlzOwotICAgIH0KLQotICAgIHN0YXRpYyBjbGFzcyBJSVNUb0lucHV0U3RyZWFtV3JhcHBlciBleHRlbmRzIElucHV0U3RyZWFtIHsKLQotICAgICAgICBwcml2YXRlIEltYWdlSW5wdXRTdHJlYW0gaW5wdXQ7Ci0KLSAgICAgICAgcHVibGljIElJU1RvSW5wdXRTdHJlYW1XcmFwcGVyKEltYWdlSW5wdXRTdHJlYW0gaW5wdXQpIHsKLSAgICAgICAgICAgIHRoaXMuaW5wdXQ9aW5wdXQ7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGludCByZWFkKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgICAgIHJldHVybiBpbnB1dC5yZWFkKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGludCByZWFkKGJ5dGVbXSBiKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAgICAgcmV0dXJuIGlucHV0LnJlYWQoYik7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIGludCByZWFkKGJ5dGVbXSBiLCBpbnQgb2ZmLCBpbnQgbGVuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAgICAgcmV0dXJuIGlucHV0LnJlYWQoYiwgb2ZmLCBsZW4pOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyBsb25nIHNraXAobG9uZyBuKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAgICAgcmV0dXJuIGlucHV0LnNraXBCeXRlcyhuKTsKLSAgICAgICAgfQotCi0gICAgICAgIEBPdmVycmlkZQotICAgICAgICBwdWJsaWMgYm9vbGVhbiBtYXJrU3VwcG9ydGVkKCkgewotICAgICAgICAJcmV0dXJuIHRydWU7ICAvLyBUaGlzIGlzIG9yaWcKLSAgICAgICAgCQotICAgICAgICAgICAgLy8gPz8/QVdUOiBGSVhNRQotICAgICAgICAJLy8gVGhpcyBpcyBhbiBlcnJvciBpbiBIYXJtb255LiBOb3QgYWxsIGlucHV0IHN0cmVhbXMKLSAgICAgICAgCS8vIGhhdmUgbWFyayBzdXBwb3J0IGFuZCBpdCBpcyBub3Qgb2sgdG8ganVzdCByZXR1cm4gdHJ1ZS4gCi0gICAgICAgIAkvLyBUaGVyZSBzaG91bGQgYmUgYW4gaW5wdXQubWFya1N1cHBvcnRlZCgpLiBIb3dldmVyLCBpZiAKLSAgICAgICAgCS8vIHRoaXMgY2FsbCByZXR1cm5zIGZhbHNlLCBub3RoaW5nIHdvcmtzIGFueW1vcmUuCi0gICAgICAgIAkKLSAgICAgICAgCS8vIFRoZSBiYWNrc2lkZSBpcyB0aGF0IEJpdG1hcEZhY3RvcnkgdXNlcyBhIGNhbGwgdG8gbWFya1N1cHBvcnQoKQotICAgICAgICAJLy8gdG8gZmluZCBvdXQgaWYgaXQgbmVlZHMgdG8gd2FycCB0aGUgc3RyZWFtIGluIGEKLSAgICAgICAgCS8vIEJ1ZmZlcmVkSW5wdXRTdHJlYW0gdG8gZ2V0IG1hcmsgc3VwcG9ydCwgYW5kIHRoaXMgZmFpbHMhCi0gICAgICAgIAkKLSAgICAgICAgCS8vIEN1cnJlbnRseSwgdGhlIGhhY2sgaXMgaW4gQml0bWFwRmFjdG9yeSwgd2hlcmUgd2UgYWx3YXlzCi0gICAgICAgIAkvLyB3cmFwIHRoZSBzdHJlYW0gaW4gYSBCdWZmZXJlZElucHV0U3RyZWFtLgotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyB2b2lkIG1hcmsoaW50IHJlYWRsaW1pdCkgewotICAgICAgICAgICAgaW5wdXQubWFyaygpOwotICAgICAgICB9Ci0KLSAgICAgICAgQE92ZXJyaWRlCi0gICAgICAgIHB1YmxpYyB2b2lkIHJlc2V0KCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgICAgIGlucHV0LnJlc2V0KCk7Ci0gICAgICAgIH0KLQotICAgICAgICBAT3ZlcnJpZGUKLSAgICAgICAgcHVibGljIHZvaWQgY2xvc2UoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAgICAgaW5wdXQuY2xvc2UoKTsKLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHQ29uc3RzLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR0NvbnN0cy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwNjdhODI1Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHQ29uc3RzLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0NCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFJ1c3RlbSBWLiBSYWZpa292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5wbHVnaW5zLmpwZWc7Ci0KLXB1YmxpYyBjbGFzcyBKUEVHQ29uc3RzIHsKLQotICAgIHByaXZhdGUgSlBFR0NvbnN0cygpIHt9Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTT0kgPSAweEQ4OwotCi0gICAgLy8tLSBJSkcgKEluZGVwZW5kZWQgSlBFRyBHcm91cCkgY29sb3Igc3BhY2VzCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSkNTX1VOS05PVyA9IDA7Ci0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSkNTX0dSQVlTQ0FMRSA9IDE7Ci0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSkNTX1JHQiA9IDI7Ci0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSkNTX1lDYkNyID0gMzsKLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBKQ1NfQ01ZSyA9IDQ7Ci0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSkNTX1lDQyA9IDU7Ci0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSkNTX1JHQkEgPSA2OwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEpDU19ZQ2JDckEgPSA3OwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IEpDU19ZQ0NBID0gMTA7Ci0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgSkNTX1lDQ0sgPSAxMTsKLQotICAgIHB1YmxpYyBzdGF0aWMgaW50W11bXSBCQU5EX09GRlNFVFMgPSB7e30sIHswfSwgezAsIDF9LCB7MCwgMSwgMn0sIHswLCAxLCAyLCAzfX07Ci0KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGZsb2F0IERFRkFVTFRfSlBFR19DT01QUkVTU0lPTl9RVUFMSVRZID0gMC43NWY7Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR0ltYWdlUmVhZGVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR0ltYWdlUmVhZGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDExMGVkMjMuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9qcGVnL0pQRUdJbWFnZVJlYWRlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTI2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS40ICQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMuanBlZzsKLQotCi1pbXBvcnQgamF2YXguaW1hZ2Vpby5JbWFnZVJlYWRlcjsKLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlUmVhZFBhcmFtOwotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VUeXBlU3BlY2lmaWVyOwotaW1wb3J0IGphdmF4LmltYWdlaW8ucGx1Z2lucy5qcGVnLkpQRUdJbWFnZVJlYWRQYXJhbTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLnN0cmVhbS5JbWFnZUlucHV0U3RyZWFtOwotaW1wb3J0IGphdmF4LmltYWdlaW8ubWV0YWRhdGEuSUlPTWV0YWRhdGE7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zcGkuSW1hZ2VSZWFkZXJTcGk7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmltYWdlLkRlY29kaW5nSW1hZ2VTb3VyY2U7Ci1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZS5PZmZzY3JlZW5JbWFnZTsKLQotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS51dGlsLkl0ZXJhdG9yOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkJ1ZmZlcmVkSW1hZ2U7Ci0KLS8qKgotICogVGhpcyBpbXBsZW1lbnRhdGlvbiB1c2VzIG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2UuSnBlZ0RlY29kZXIgdG8gcmVhZAotICogYW4gaW1hZ2UuIFRoZSBvbmx5IGltcGxlbWVudGVkIG1ldGhvZCBpcyByZWFkKC4uKTsKLSAqCi0gKiBUT0RPOiBJbXBsZW1lbnRzIGdlbmVyaWMgZGVjb2RlciB0byBiZSB1c2VkIGJ5IGphdmFkMiBhbmQgaW1hZ2VpbwotICoKLSAqIEBzZWUgb3JnLmFwYWNoZS5oYXJtb255LmF3dC5nbC5pbWFnZS5KcGVnRGVjb2RlcgotICogQHNlZSBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMuanBlZy5JSVNEZWNvZGluZ0ltYWdlU291cmNlCi0gKi8KLXB1YmxpYyBjbGFzcyBKUEVHSW1hZ2VSZWFkZXIgZXh0ZW5kcyBJbWFnZVJlYWRlciB7Ci0KLSAgICBJbWFnZUlucHV0U3RyZWFtIGlpczsKLQotICAgIHB1YmxpYyBKUEVHSW1hZ2VSZWFkZXIoSW1hZ2VSZWFkZXJTcGkgaW1hZ2VSZWFkZXJTcGkpIHsKLSAgICAgICAgc3VwZXIoaW1hZ2VSZWFkZXJTcGkpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0SGVpZ2h0KGludCBpKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLy0tIFRPRE8gaW1sZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJub3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGludCBnZXRXaWR0aChpbnQgaSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgLy8tLSBUT0RPIGltbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigibm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBpbnQgZ2V0TnVtSW1hZ2VzKGJvb2xlYW4gYikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgLy8tLSBUT0RPIGltbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigibm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJdGVyYXRvcjxJbWFnZVR5cGVTcGVjaWZpZXI+IGdldEltYWdlVHlwZXMoaW50IGkpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vLS0gVE9ETyBpbWxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSUlPTWV0YWRhdGEgZ2V0U3RyZWFtTWV0YWRhdGEoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLy0tIFRPRE8gaW1sZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJub3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIElJT01ldGFkYXRhIGdldEltYWdlTWV0YWRhdGEoaW50IGkpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vLS0gVE9ETyBpbWxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgQnVmZmVyZWRJbWFnZSByZWFkKGludCBpLCBJbWFnZVJlYWRQYXJhbSBpbWFnZVJlYWRQYXJhbSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKGlpcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJpbnB1dCBzdHJlYW0gPT0gbnVsbCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgRGVjb2RpbmdJbWFnZVNvdXJjZSBzb3VyY2UgPSBuZXcgSUlTRGVjb2RpbmdJbWFnZVNvdXJjZShpaXMpOwotICAgICAgICBPZmZzY3JlZW5JbWFnZSBpbWFnZSA9IG5ldyBPZmZzY3JlZW5JbWFnZShzb3VyY2UpOwotICAgICAgICBzb3VyY2UuYWRkQ29uc3VtZXIoaW1hZ2UpOwotICAgICAgICBzb3VyY2UubG9hZCgpOwotICAgICAgICAvLyBUaGUgaW50ZXJydXB0ZWQgZmxhZyBzaG91bGQgYmUgY2xlYXJlZCBiZWNhdXNlIEltYWdlRGVjb2RlciBpbnRlcnJ1cHRzCi0gICAgICAgIC8vIGN1cnJlbnQgdGhyZWFkIHdoaWxlIGRlY29kaW5nLiBUaGUgc2FtZSB0ZWNobmlxdWUgaXMgdXNlZCBpbgotICAgICAgICAvLyBJbWFnZUxvYWRlciNydW4oKS4gQW5vdGhlciBzb2x1dGlvbiBjYW4gYmUgdG8gY3JlYXRlCi0gICAgICAgIC8vIGEgc2VwYXJhdGUgZGVjb2RpbmcgdGhyZWFkLiBIb3dldmVyLCBkZWNvZGVyIGtlZXBzIGl0cyBvd24gcG9vbAotICAgICAgICAvLyBvZiB0aHJlYWRzIHNvIGNyZWF0aW5nIGEgbmV3IHRocmVhZCB3aWxsIGJlIGp1c3QgYSB3YXN0ZSBvZiByZXNvdXJjZXMuCi0gICAgICAgIFRocmVhZC5pbnRlcnJ1cHRlZCgpOwotICAgICAgICByZXR1cm4gaW1hZ2UuZ2V0QnVmZmVyZWRJbWFnZSgpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlIHJlYWQoaW50IGkpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiByZWFkKGksIG51bGwpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIHNldElucHV0KE9iamVjdCBpbnB1dCwgYm9vbGVhbiBzZWVrRm9yd2FyZE9ubHksIGJvb2xlYW4gaWdub3JlTWV0YWRhdGEpIHsKLSAgICAgICAgc3VwZXIuc2V0SW5wdXQoaW5wdXQsIHNlZWtGb3J3YXJkT25seSwgaWdub3JlTWV0YWRhdGEpOwotICAgICAgICBpaXMgPSAoSW1hZ2VJbnB1dFN0cmVhbSkgaW5wdXQ7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEltYWdlUmVhZFBhcmFtIGdldERlZmF1bHRSZWFkUGFyYW0oKSB7Ci0gICAgICAgIHJldHVybiBuZXcgSlBFR0ltYWdlUmVhZFBhcmFtKCk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9qcGVnL0pQRUdJbWFnZVJlYWRlclNwaS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9qcGVnL0pQRUdJbWFnZVJlYWRlclNwaS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjNzE5Y2U3Li4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHSW1hZ2VSZWFkZXJTcGkuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDg2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMuanBlZzsKLQotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlUmVhZGVyOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLkltYWdlUmVhZGVyU3BpOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLlNlcnZpY2VSZWdpc3RyeTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLnN0cmVhbS5JbWFnZUlucHV0U3RyZWFtOwotCi1wdWJsaWMgY2xhc3MgSlBFR0ltYWdlUmVhZGVyU3BpIGV4dGVuZHMgSW1hZ2VSZWFkZXJTcGkgewotCi0gICAgcHVibGljIEpQRUdJbWFnZVJlYWRlclNwaSgpIHsKLSAgICAgICAgc3VwZXIoSlBFR1NwaUNvbnN0cy52ZW5kb3JOYW1lLCBKUEVHU3BpQ29uc3RzLnZlcnNpb24sCi0gICAgICAgICAgICAgICAgSlBFR1NwaUNvbnN0cy5uYW1lcywgSlBFR1NwaUNvbnN0cy5zdWZmaXhlcywKLSAgICAgICAgICAgICAgICBKUEVHU3BpQ29uc3RzLk1JTUVUeXBlcywgSlBFR1NwaUNvbnN0cy5yZWFkZXJDbGFzc05hbWUsCi0gICAgICAgICAgICAgICAgU1RBTkRBUkRfSU5QVVRfVFlQRSwgSlBFR1NwaUNvbnN0cy53cml0ZXJTcGlOYW1lcywKLSAgICAgICAgICAgICAgICBKUEVHU3BpQ29uc3RzLnN1cHBvcnRzU3RhbmRhcmRTdHJlYW1NZXRhZGF0YUZvcm1hdCwKLSAgICAgICAgICAgICAgICBKUEVHU3BpQ29uc3RzLm5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZSwKLSAgICAgICAgICAgICAgICBKUEVHU3BpQ29uc3RzLm5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lLAotICAgICAgICAgICAgICAgIEpQRUdTcGlDb25zdHMuZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWVzLAotICAgICAgICAgICAgICAgIEpQRUdTcGlDb25zdHMuZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMsCi0gICAgICAgICAgICAgICAgSlBFR1NwaUNvbnN0cy5zdXBwb3J0c1N0YW5kYXJkSW1hZ2VNZXRhZGF0YUZvcm1hdCwKLSAgICAgICAgICAgICAgICBKUEVHU3BpQ29uc3RzLm5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lLAotICAgICAgICAgICAgICAgIEpQRUdTcGlDb25zdHMubmF0aXZlSW1hZ2VNZXRhZGF0YUZvcm1hdENsYXNzTmFtZSwKLSAgICAgICAgICAgICAgICBKUEVHU3BpQ29uc3RzLmV4dHJhSW1hZ2VNZXRhZGF0YUZvcm1hdE5hbWVzLAotICAgICAgICAgICAgICAgIEpQRUdTcGlDb25zdHMuZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcyk7Ci0gICAgfQotCi0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgYm9vbGVhbiBjYW5EZWNvZGVJbnB1dChPYmplY3Qgc291cmNlKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBJbWFnZUlucHV0U3RyZWFtIG1hcmthYmxlID0gKEltYWdlSW5wdXRTdHJlYW0pIHNvdXJjZTsKLSAgICAgICAgdHJ5IHsKLSAgICAgICAgICAgIG1hcmthYmxlLm1hcmsoKTsKLQotICAgICAgICAgICAgYnl0ZVtdIHNpZ25hdHVyZSA9IG5ldyBieXRlWzNdOwotICAgICAgICAgICAgbWFya2FibGUuc2VlaygwKTsKLSAgICAgICAgICAgIG1hcmthYmxlLnJlYWQoc2lnbmF0dXJlLCAwLCAzKTsKLSAgICAgICAgICAgIG1hcmthYmxlLnJlc2V0KCk7Ci0KLSAgICAgICAgICAgIGlmICgoc2lnbmF0dXJlWzBdICYgMHhGRikgPT0gMHhGRiAmJgotICAgICAgICAgICAgICAgICAgICAoc2lnbmF0dXJlWzFdICYgMHhGRikgPT0gSlBFR0NvbnN0cy5TT0kgJiYKLSAgICAgICAgICAgICAgICAgICAgKHNpZ25hdHVyZVsyXSAmIDB4RkYpID09IDB4RkYpIHsgLy8gSlBFRwotICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9IGNhdGNoIChJT0V4Y2VwdGlvbiBlKSB7Ci0gICAgICAgICAgICBlLnByaW50U3RhY2tUcmFjZSgpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSW1hZ2VSZWFkZXIgY3JlYXRlUmVhZGVySW5zdGFuY2UoT2JqZWN0IGV4dGVuc2lvbikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIG5ldyBKUEVHSW1hZ2VSZWFkZXIodGhpcyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyBnZXREZXNjcmlwdGlvbihMb2NhbGUgbG9jYWxlKSB7Ci0gICAgICAgIHJldHVybiAiRFJMIEpQRUcgZGVjb2RlciI7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgb25SZWdpc3RyYXRpb24oU2VydmljZVJlZ2lzdHJ5IHJlZ2lzdHJ5LCBDbGFzczw/PiBjYXRlZ29yeSkgewotICAgICAgICAvLyBzdXBlci5vblJlZ2lzdHJhdGlvbihyZWdpc3RyeSwgY2F0ZWdvcnkpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHSW1hZ2VXcml0ZXIuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHSW1hZ2VXcml0ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYWUzZTg3Ni4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR0ltYWdlV3JpdGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw0MDIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ucGx1Z2lucy5qcGVnOwotCi1pbXBvcnQgY29tLmFuZHJvaWQuaW50ZXJuYWwuYXd0LkltYWdlT3V0cHV0U3RyZWFtV3JhcHBlcjsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VXcml0ZXI7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5JSU9JbWFnZTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlVHlwZVNwZWNpZmllcjsKLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlV3JpdGVQYXJhbTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLnBsdWdpbnMuanBlZy5KUEVHSW1hZ2VXcml0ZVBhcmFtOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3RyZWFtLkltYWdlT3V0cHV0U3RyZWFtOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLkltYWdlV3JpdGVyU3BpOwotaW1wb3J0IGphdmF4LmltYWdlaW8ubWV0YWRhdGEuSUlPTWV0YWRhdGE7Ci0KLWltcG9ydCBhbmRyb2lkLmdyYXBoaWNzLkJpdG1hcDsKLWltcG9ydCBhbmRyb2lkLmdyYXBoaWNzLkJpdG1hcEZhY3Rvcnk7Ci1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5CaXRtYXAuQ29tcHJlc3NGb3JtYXQ7Ci0KLWltcG9ydCBqYXZhLmlvLkZpbGU7Ci1pbXBvcnQgamF2YS5pby5GaWxlT3V0cHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuKjsKLWltcG9ydCBqYXZhLmF3dC4qOwotaW1wb3J0IGphdmEuYXd0LmNvbG9yLkNvbG9yU3BhY2U7Ci0KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjMgJAotICovCi1wdWJsaWMgY2xhc3MgSlBFR0ltYWdlV3JpdGVyIGV4dGVuZHMgSW1hZ2VXcml0ZXIgewotCi0gICAgLy8gLyogPz8/QVdUOiBEZWJ1Z2dpbmcKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBib29sZWFuIERFQlVHID0gZmFsc2U7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgQml0bWFwIGJtOwotICAgIHB1YmxpYyBzdGF0aWMgQml0bWFwIGdldEJpdG1hcCgpIHsKLSAgICAgICAgcmV0dXJuIGJtOwotICAgIH0KLSAgICBwcml2YXRlIHN0YXRpYyBCdWZmZXJlZEltYWdlIGJ1ZkltZzsKLSAgICBwdWJsaWMgc3RhdGljIEJ1ZmZlcmVkSW1hZ2UgZ2V0QnVmSW1hZ2UoKSB7Ci0gICAgICAgIHJldHVybiBidWZJbWc7Ci0gICAgfQotICAgIHN0YXRpYyBwcml2YXRlIFJlbmRlcmVkSW1hZ2UgcmVuSW1nOwotICAgIHN0YXRpYyBwdWJsaWMgUmVuZGVyZWRJbWFnZSBnZXRSZW5JbWFnZSgpIHsKLSAgICAgICAgcmV0dXJuIHJlbkltZzsKLSAgICB9Ci0gICAgLy8gKi8KLSAgICAKLSAgICBwcml2YXRlIGxvbmcgY2luZm87Ci0gICAgcHJpdmF0ZSBSZW5kZXJlZEltYWdlIGltYWdlOwotICAgIHByaXZhdGUgUmFzdGVyIHNvdXJjZVJhc3RlcjsKLSAgICBwcml2YXRlIFdyaXRhYmxlUmFzdGVyIHNjYW5SYXN0ZXI7Ci0gICAgcHJpdmF0ZSBpbnQgc3JjWE9mZiA9IDA7Ci0gICAgcHJpdmF0ZSBpbnQgc3JjWU9mZiA9IDA7Ci0gICAgcHJpdmF0ZSBpbnQgc3JjV2lkdGg7Ci0gICAgcHJpdmF0ZSBpbnQgc3JjSGVpZ2h0OwotCi0gICAgLy8tLSB5IHN0ZXAgZm9yIGltYWdlIHN1YnNhbXBsaW5nCi0gICAgcHJpdmF0ZSBpbnQgZGVsdGFZID0gMTsKLSAgICAvLy0tIHggc3RlcCBmb3IgaW1hZ2Ugc3Vic2FtcGxpbmcKLSAgICBwcml2YXRlIGludCBkZWx0YVggPSAxOwotCi0gICAgcHJpdmF0ZSBJbWFnZU91dHB1dFN0cmVhbSBpb3M7Ci0KLSAgICBwdWJsaWMgSlBFR0ltYWdlV3JpdGVyKEltYWdlV3JpdGVyU3BpIGltYWdlV3JpdGVyU3BpKSB7Ci0gICAgICAgIHN1cGVyKGltYWdlV3JpdGVyU3BpKTsKLSAgICAgICAgLy8/Pz9BV1Q6IGNpbmZvID0gaW5pdENvbXByZXNzaW9uT2JqKCk7Ci0gICAgICAgIGNpbmZvID0gU3lzdGVtLmN1cnJlbnRUaW1lTWlsbGlzKCk7Ci0gICAgfQotCi0gICAgc3RhdGljIHsKLSAgICAgICAgLy8/Pz9BV1QKLSAgICAgICAgLyoKLSAgICAgICAgU3lzdGVtLmxvYWRMaWJyYXJ5KCJqcGVnZW5jb2RlciIpOwotICAgICAgICBpbml0V3JpdGVySWRzKEltYWdlT3V0cHV0U3RyZWFtLmNsYXNzKTsKLSAgICAgICAgKi8KLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB3cml0ZShJSU9NZXRhZGF0YSBpaW9NZXRhZGF0YSwgSUlPSW1hZ2UgaWlvSW1hZ2UsIEltYWdlV3JpdGVQYXJhbSBwYXJhbSkKLSAgICAgICAgICAgIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0KLSAgICAgICAgaWYgKGlvcyA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJpb3MgPT0gbnVsbCIpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChpaW9JbWFnZSA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJJbWFnZSBlcXVhbHMgbnVsbCIpOwotICAgICAgICB9Ci0KLSAgICAgICAgUmVuZGVyZWRJbWFnZSBpbWcgPSBudWxsOwotICAgICAgICBpZiAoIWlpb0ltYWdlLmhhc1Jhc3RlcigpKSB7Ci0gICAgICAgICAgICBpbWcgPSBpaW9JbWFnZS5nZXRSZW5kZXJlZEltYWdlKCk7Ci0gICAgICAgICAgICBpZiAoaW1nIGluc3RhbmNlb2YgQnVmZmVyZWRJbWFnZSkgewotICAgICAgICAgICAgICAgIHNvdXJjZVJhc3RlciA9ICgoQnVmZmVyZWRJbWFnZSkgaW1nKS5nZXRSYXN0ZXIoKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgc291cmNlUmFzdGVyID0gaW1nLmdldERhdGEoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHNvdXJjZVJhc3RlciA9IGlpb0ltYWdlLmdldFJhc3RlcigpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICAvLyBBV1Q/Pz86IERlYnVnZ2luZwotICAgICAgICBpZiAoREVCVUcpIHsKLSAgICAgICAgICAgIGlmKCBpbWc9PW51bGwgKSB7Ci0gICAgICAgICAgICAgICAgU3lzdGVtLm91dC5wcmludGxuKCIqKioqSjogSW1hZ2UgaXMgTlVMTCIpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICByZW5JbWcgPSBpbWc7Ci0gICAgICAgICAgICAgICAgYnVmSW1nID0gKEJ1ZmZlcmVkSW1hZ2UpaW1nOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaW50IG51bUJhbmRzID0gc291cmNlUmFzdGVyLmdldE51bUJhbmRzKCk7Ci0gICAgICAgIGludCBzb3VyY2VJSkdDcyA9IGltZyA9PSBudWxsID8gSlBFR0NvbnN0cy5KQ1NfVU5LTk9XIDogZ2V0U291cmNlQ1NUeXBlKGltZyk7Ci0KLSAgICAgICAgc3JjV2lkdGggPSBzb3VyY2VSYXN0ZXIuZ2V0V2lkdGgoKTsKLSAgICAgICAgc3JjSGVpZ2h0ID0gc291cmNlUmFzdGVyLmdldEhlaWdodCgpOwotCi0gICAgICAgIGludCBkZXN0V2lkdGggPSBzcmNXaWR0aDsKLSAgICAgICAgaW50IGRlc3RIZWlnaHQgPSBzcmNIZWlnaHQ7Ci0KLSAgICAgICAgYm9vbGVhbiBwcm9ncmVzc2l2ZSA9IGZhbHNlOwotICAgICAgICAgCi0gICAgICAgIGlmIChwYXJhbSAhPSBudWxsKSB7Ci0gICAgICAgICAgICBSZWN0YW5nbGUgcmVnID0gcGFyYW0uZ2V0U291cmNlUmVnaW9uKCk7Ci0gICAgICAgICAgICBpZiAocmVnICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBzcmNYT2ZmID0gcmVnLng7Ci0gICAgICAgICAgICAgICAgc3JjWU9mZiA9IHJlZy55OwotCi0gICAgICAgICAgICAgICAgc3JjV2lkdGggPSByZWcud2lkdGggKyBzcmNYT2ZmID4gc3JjV2lkdGgKLSAgICAgICAgICAgICAgICAgICAgICAgID8gc3JjV2lkdGggLSBzcmNYT2ZmCi0gICAgICAgICAgICAgICAgICAgICAgICA6IHJlZy53aWR0aDsKLSAgICAgICAgICAgICAgICBzcmNIZWlnaHQgPSByZWcuaGVpZ2h0ICsgc3JjWU9mZiA+IHNyY0hlaWdodAotICAgICAgICAgICAgICAgICAgICAgICAgPyBzcmNIZWlnaHQgLSBzcmNZT2ZmCi0gICAgICAgICAgICAgICAgICAgICAgICA6IHJlZy5oZWlnaHQ7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vLS0gVE9ETyB1bmNvbW1lbnQgd2hlbiBKUEVHSW1hZ2VXcml0ZVBhcmFtIGJlIGltcGxlbWVudGVkCi0gICAgICAgICAgICAvLy0tIE9ubHkgZGVmYXVsdCBwcm9ncmVzc2l2ZSBtb2RlIHlldAotICAgICAgICAgICAgLy8gcHJvZ3Jlc3NpdmUgPSBwYXJhbS5nZXRQcm9ncmVzc2l2ZU1vZGUoKSA9PSAgSW1hZ2VXcml0ZVBhcmFtLk1PREVfREVGQVVMVDsKLQotICAgICAgICAgICAgLy8tLSBkZWYgaXMgMQotICAgICAgICAgICAgZGVsdGFYID0gcGFyYW0uZ2V0U291cmNlWFN1YnNhbXBsaW5nKCk7Ci0gICAgICAgICAgICBkZWx0YVkgPSBwYXJhbS5nZXRTb3VyY2VZU3Vic2FtcGxpbmcoKTsKLQotICAgICAgICAgICAgLy8tLSBkZWYgaXMgMAotICAgICAgICAgICAgaW50IG9mZnNldFggPSBwYXJhbS5nZXRTdWJzYW1wbGluZ1hPZmZzZXQoKTsKLSAgICAgICAgICAgIGludCBvZmZzZXRZID0gcGFyYW0uZ2V0U3Vic2FtcGxpbmdZT2Zmc2V0KCk7Ci0KLSAgICAgICAgICAgIHNyY1hPZmYgKz0gb2Zmc2V0WDsKLSAgICAgICAgICAgIHNyY1lPZmYgKz0gb2Zmc2V0WTsKLSAgICAgICAgICAgIHNyY1dpZHRoIC09IG9mZnNldFg7Ci0gICAgICAgICAgICBzcmNIZWlnaHQgLT0gb2Zmc2V0WTsKLQotICAgICAgICAgICAgZGVzdFdpZHRoID0gKHNyY1dpZHRoICsgZGVsdGFYIC0gMSkgLyBkZWx0YVg7Ci0gICAgICAgICAgICBkZXN0SGVpZ2h0ID0gKHNyY0hlaWdodCArIGRlbHRhWSAtIDEpIC8gZGVsdGFZOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8tLSBkZWZhdWx0IERRVHMgKHNlZSBKUEVHUVRhYmxlIGphdmEgZG9jIGFuZCBKUEVHIHNwZWMgSzEgJiBLMiB0YWJsZXMpCi0gICAgICAgIC8vLS0gYXQgaHR0cDovL3d3dy53My5vcmcvR3JhcGhpY3MvSlBFRy9pdHUtdDgxLnBkZgotICAgICAgICAvLy0tIE9ubHkgZmlndXJpbmcgb3V0IGhvdyB0byBzZXQgRFFUIGluIElKRyBsaWJyYXJ5IGZvciBmdXR1cmUgbWV0YWRhdGEKLSAgICAgICAgLy8tLSBzdXBwb3J0LiBJSkcgZGVmIHRhYmxlcyBhcmUgdGhlIHNhbWUuCi0gICAgICAgIC8vSlBFR1FUYWJsZVtdIGRxdCA9IG5ldyBKUEVHUVRhYmxlWzJdOwotLy8gICAgICAgIGludFtdW10gZHF0ID0gbnVsbDsKLS8vICAgICAgICBpbnRbXVtdIGRxdCA9IG5ldyBpbnRbMl1bXTsKLS8vICAgICAgICBkcXRbMF0gPSBKUEVHUVRhYmxlLksxRGl2Mkx1bWluYW5jZS5nZXRUYWJsZSgpOwotLy8gICAgICAgIGRxdFsxXSA9IEpQRUdRVGFibGUuSzJEaXYyQ2hyb21pbmFuY2UuZ2V0VGFibGUoKTsKLSAgICAgICAgCi0gICAgICAgIC8vPz8/QVdUOiBJIHRoaW5rIHdlIGRvbid0IG5lZWQgdGhpcyBhbXltb3JlCi0gICAgICAgIC8qCi0gICAgICAgIC8vLS0gdXNpbmcgZGVmYXVsdCBjb2xvciBzcGFjZQotICAgICAgICAvLy0tIFRPRE86IFRha2UgZGVzdGluYXRpb24gY3MgZnJvbSBwYXJhbSBvciB1c2UgZGVmYXVsdCBpZiB0aGVyZSBpcyBubyBjcwotICAgICAgICBpbnQgZGVzdElKR0NzID0gaW1nID09IG51bGwgPyBKUEVHQ29uc3RzLkpDU19VTktOT1cgOiBnZXREZXN0aW5hdGlvbkNTVHlwZShpbWcpOwotCi0gICAgICAgIERhdGFCdWZmZXJCeXRlIGRidWZmZXIgPSBuZXcgRGF0YUJ1ZmZlckJ5dGUobnVtQmFuZHMgKiBzcmNXaWR0aCk7Ci0KLSAgICAgICAgc2NhblJhc3RlciA9IFJhc3Rlci5jcmVhdGVJbnRlcmxlYXZlZFJhc3RlcihkYnVmZmVyLCBzcmNXaWR0aCwgMSwKLSAgICAgICAgICAgICAgICBudW1CYW5kcyAqIHNyY1dpZHRoLCBudW1CYW5kcywgSlBFR0NvbnN0cy5CQU5EX09GRlNFVFNbbnVtQmFuZHNdLCBudWxsKTsKLQotICAgICAgICBlbmNvZGUoZGJ1ZmZlci5nZXREYXRhKCksIHNyY1dpZHRoLCBkZXN0V2lkdGgsIGRlc3RIZWlnaHQsIGRlbHRhWCwKLSAgICAgICAgICAgICAgICBzb3VyY2VJSkdDcywgZGVzdElKR0NzLCBudW1CYW5kcywgcHJvZ3Jlc3NpdmUsCi0gICAgICAgICAgICAgICAgbnVsbCwgY2luZm8pOwotICAgICAgICAqLwotICAgICAgICAKLSAgICAgICAgU2FtcGxlTW9kZWwgbW9kZWwgPSBzb3VyY2VSYXN0ZXIuZ2V0U2FtcGxlTW9kZWwoKTsKLSAgICAgICAgCi0gICAgICAgIGlmIChtb2RlbCBpbnN0YW5jZW9mIFNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWwpIHsKLSAgICAgICAgICAgIERhdGFCdWZmZXJJbnQgaWJ1ZiA9IChEYXRhQnVmZmVySW50KXNvdXJjZVJhc3Rlci5nZXREYXRhQnVmZmVyKCk7Ci0gICAgICAgICAgICBpbnRbXSBwaXhlbHMgPSBpYnVmLmdldERhdGEoKTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8gQ3JlYXRlIGEgYml0bWFwIHdpdGggdGhlIHBpeGVsCi0gICAgICAgICAgICBibSA9IEJpdG1hcC5jcmVhdGVCaXRtYXAocGl4ZWxzLCBzcmNXaWR0aCwgc3JjSGVpZ2h0LCBCaXRtYXAuQ29uZmlnLkFSR0JfODg4OCk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8vIFVzZSBCaXRtYXAuY29tcHJlc3MoKSB0byB3cml0ZSB0aGUgaW1hZ2UKLSAgICAgICAgICAgIEltYWdlT3V0cHV0U3RyZWFtV3JhcHBlciBpb3N3ID0gbmV3IEltYWdlT3V0cHV0U3RyZWFtV3JhcHBlcihpb3MpOwotICAgICAgICAgICAgYm0uY29tcHJlc3MoQ29tcHJlc3NGb3JtYXQuSlBFRywgMTAwLCBpb3N3KTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vID8/P0FXVDogQWRkIHN1cHBvcnQgZm9yIG90aGVyIGNvbG9yIG1vZGVscwotICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIkNvbG9yIG1vZGVsIG5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgICAgIH0KLQotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyB2b2lkIGRpc3Bvc2UoKSB7Ci0gICAgICAgIHN1cGVyLmRpc3Bvc2UoKTsKLSAgICAgICAgaWYgKGNpbmZvICE9IDApIHsKLSAgICAgICAgICAgIC8vPz8/QVdUOiBkaXNwb3NlKGNpbmZvKTsKLSAgICAgICAgICAgIGNpbmZvID0gMDsKLSAgICAgICAgICAgIGlvcyA9IG51bGw7Ci0gICAgICAgIH0KLSAgICB9Ci0KLQotICAgIHB1YmxpYyBJSU9NZXRhZGF0YSBnZXREZWZhdWx0U3RyZWFtTWV0YWRhdGEoSW1hZ2VXcml0ZVBhcmFtIGltYWdlV3JpdGVQYXJhbSkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgcHVibGljIElJT01ldGFkYXRhIGdldERlZmF1bHRJbWFnZU1ldGFkYXRhKEltYWdlVHlwZVNwZWNpZmllciBpbWFnZVR5cGVTcGVjaWZpZXIsIEltYWdlV3JpdGVQYXJhbSBpbWFnZVdyaXRlUGFyYW0pIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJub3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJSU9NZXRhZGF0YSBjb252ZXJ0U3RyZWFtTWV0YWRhdGEoSUlPTWV0YWRhdGEgaWlvTWV0YWRhdGEsIEltYWdlV3JpdGVQYXJhbSBpbWFnZVdyaXRlUGFyYW0pIHsKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJub3Qgc3VwcG9ydGVkIHlldCIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJSU9NZXRhZGF0YSBjb252ZXJ0SW1hZ2VNZXRhZGF0YShJSU9NZXRhZGF0YSBpaW9NZXRhZGF0YSwgSW1hZ2VUeXBlU3BlY2lmaWVyIGltYWdlVHlwZVNwZWNpZmllciwgSW1hZ2VXcml0ZVBhcmFtIGltYWdlV3JpdGVQYXJhbSkgewotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0T3V0cHV0KE9iamVjdCBvdXRwdXQpIHsKLSAgICAgICAgc3VwZXIuc2V0T3V0cHV0KG91dHB1dCk7Ci0gICAgICAgIGlvcyA9IChJbWFnZU91dHB1dFN0cmVhbSkgb3V0cHV0OwotICAgICAgICAvLz8/P0FXVDogc2V0SU9TKGlvcywgY2luZm8pOwotICAgICAgICBzb3VyY2VSYXN0ZXIgPSBudWxsOwotICAgICAgICBzY2FuUmFzdGVyID0gbnVsbDsKLSAgICAgICAgc3JjWE9mZiA9IDA7Ci0gICAgICAgIHNyY1lPZmYgPSAwOwotICAgICAgICBzcmNXaWR0aCA9IDA7Ci0gICAgICAgIHNyY0hlaWdodCA9IDA7Ci0gICAgICAgIGRlbHRhWSA9IDE7Ci0gICAgfQotCi0gICAgLyoqCi0gICAgICogRnJlZXMgcmVzb3VyY2VzCi0gICAgICogQHBhcmFtIHN0cnVjdFBvaW50ZXIKLSAgICAgKi8KLSAgICAvLz8/P0FXVDogcHJpdmF0ZSBuYXRpdmUgdm9pZCBkaXNwb3NlKGxvbmcgc3RydWN0UG9pbnRlcik7Ci0KLSAgICAvKioKLSAgICAgKiBJbml0cyBtZXRob2RzIElkcyBmb3IgbmF0aXZlIHRvIGphdmEgY2FsbGJhY2tzCi0gICAgICogQHBhcmFtIGlvc0NsYXNzCi0gICAgICovCi0gICAgLy8/Pz9BV1Q6IHByaXZhdGUgbmF0aXZlIHN0YXRpYyB2b2lkIGluaXRXcml0ZXJJZHMoQ2xhc3M8SW1hZ2VPdXRwdXRTdHJlYW0+IGlvc0NsYXNzKTsKLQotICAgIC8qKgotICAgICAqIEluaXRzIGNvbXByZXNzaW9uIG9iamVjdHMKLSAgICAgKiBAcmV0dXJuIHBvaW50ZXIgdG8gdGhlIG5hdGl2ZSBzdHJ1Y3R1cmUKLSAgICAgKi8KLSAgICAvLz8/P0FXVDogcHJpdmF0ZSBuYXRpdmUgbG9uZyBpbml0Q29tcHJlc3Npb25PYmooKTsKLQotICAgIC8qKgotICAgICAqIFNldHMgaW1hZ2Ugb3V0cHV0IHN0cmVhbSBpbiBJSkcgbGF5ZXIKLSAgICAgKiBAcGFyYW0gc3RyZWFtCi0gICAgICovCi0gICAgLy8/Pz9BV1Q6IHByaXZhdGUgbmF0aXZlIHZvaWQgc2V0SU9TKEltYWdlT3V0cHV0U3RyZWFtIHN0cmVhbSwgbG9uZyBzdHJ1Y3RQb2ludGVyKTsKLQotICAgIC8qKgotICAgICAqIFJ1bnMgZW5jb2RpbmcgcHJvY2Vzcy4KLSAgICAgKgotICAgICAqIEBwYXJhbSBkYXRhIGltYWdlIGRhdGEgYnVmZmVyIHRvIGVuY29kZQotICAgICAqIEBwYXJhbSBzcmNXaWR0aCAtIHNvdXJjZSB3aWR0aAotICAgICAqIEBwYXJhbSB3aWR0aCAtIGRlc3RpbmF0aW9uIHdpZHRoCi0gICAgICogQHBhcmFtIGhlaWdodCBkZXN0aW5hdGlvbiBoZWlnaHQKLSAgICAgKiBAcGFyYW0gZGVsdGFYIC0geCBzdWJzYW1wbGluZyBzdGVwCi0gICAgICogQHBhcmFtIGluQ29sb3JTcGFjZSAtIG9yaWdpbmFsIGNvbG9yIHNwYWNlCi0gICAgICogQHBhcmFtIG91dENvbG9yU3BhY2UgLSBkZXN0aW5hdGlvbiBjb2xvciBzcGFjZQotICAgICAqIEBwYXJhbSBudW1CYW5kcyAtIG51bWJlciBvZiBiYW5kcwotICAgICAqIEBwYXJhbSBjaW5mbyAtIG5hdGl2ZSBoYW5kbGVyCi0gICAgICogQHJldHVybgotICAgICAqLwotICAgIC8vPz8/QVdUOgotICAgIC8qCi0gICAgcHJpdmF0ZSBuYXRpdmUgYm9vbGVhbiBlbmNvZGUoYnl0ZVtdIGRhdGEsIGludCBzcmNXaWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgd2lkdGgsIGludCBoZWlnaHQsIGludCBkZWx0YVgsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGluQ29sb3JTcGFjZSwgaW50IG91dENvbG9yU3BhY2UsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IG51bUJhbmRzLCBib29sZWFuIHByb2dyZXNzaXZlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludFtdW10gZHF0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxvbmcgY2luZm8pOwotICAgICovCi0KLSAgICAvKioKLSAgICAgKiBDYWxsYmFjayBmb3IgZ2V0dGluZyBhIG5leHQgc2NhbmxpbmUKLSAgICAgKiBAcGFyYW0gc2NhbmxpbmUgc2NhbiBsaW5lIG51bWJlcgotICAgICAqLwotICAgIEBTdXBwcmVzc1dhcm5pbmdzKCJ1bnVzZWQiKQotICAgIHByaXZhdGUgdm9pZCBnZXRTY2FuTGluZShpbnQgc2NhbmxpbmUpIHsKLSAgICAgICAgLy8tLSBUT0RPOiBwcm9jZXNzSW1hZ2VQcm9ncmVzcyBpbiBJbWFnZVdyaXRlcgotICAgICAgICBSYXN0ZXIgY2hpbGQgPSBzb3VyY2VSYXN0ZXIuY3JlYXRlQ2hpbGQoc3JjWE9mZiwKLSAgICAgICAgICAgICAgICBzcmNZT2ZmICsgc2NhbmxpbmUgKiBkZWx0YVksIHNyY1dpZHRoLCAxLCAwLCAwLCBudWxsKTsKLQotICAgICAgICBzY2FuUmFzdGVyLnNldFJlY3QoY2hpbGQpOwotICAgIH0KLQotICAgIC8qKgotICAgICAqIE1hcHMgY29sb3Igc3BhY2UgdHlwZXMgdG8gSUpHIGNvbG9yIHNwYWNlcwotICAgICAqIEBwYXJhbSBpbWFnZQotICAgICAqIEByZXR1cm4KLSAgICAgKi8KLSAgICBwcml2YXRlIGludCBnZXRTb3VyY2VDU1R5cGUoUmVuZGVyZWRJbWFnZSBpbWFnZSkgewotICAgICAgICBpbnQgdHlwZSA9IEpQRUdDb25zdHMuSkNTX1VOS05PVzsKLSAgICAgICAgQ29sb3JNb2RlbCBjbSA9IGltYWdlLmdldENvbG9yTW9kZWwoKTsKLQotICAgICAgICBpZiAobnVsbCA9PSBjbSkgewotICAgICAgICAgICAgcmV0dXJuIHR5cGU7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoY20gaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiSW5kZXhDb2xvck1vZGVsIGlzIG5vdCBzdXBwb3J0ZWQgeWV0Iik7Ci0gICAgICAgIH0KLQotICAgICAgICBib29sZWFuIGhhc0FscGhhID0gY20uaGFzQWxwaGEoKTsKLSAgICAgICAgQ29sb3JTcGFjZSBjcyA9IGNtLmdldENvbG9yU3BhY2UoKTsKLSAgICAgICAgc3dpdGNoKGNzLmdldFR5cGUoKSkgewotICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLlRZUEVfR1JBWToKLSAgICAgICAgICAgICAgICB0eXBlID0gSlBFR0NvbnN0cy5KQ1NfR1JBWVNDQUxFOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuVFlQRV9SR0I6Ci0gICAgICAgICAgICAgICAgdHlwZSA9IGhhc0FscGhhID8gSlBFR0NvbnN0cy5KQ1NfUkdCQSA6IEpQRUdDb25zdHMuSkNTX1JHQjsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLlRZUEVfWUNiQ3I6Ci0gICAgICAgICAgICAgICAgdHlwZSA9IGhhc0FscGhhID8gSlBFR0NvbnN0cy5KQ1NfWUNiQ3JBIDogSlBFR0NvbnN0cy5KQ1NfWUNiQ3I7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5UWVBFXzNDTFI6Ci0gICAgICAgICAgICAgICAgIHR5cGUgPSBoYXNBbHBoYSA/IEpQRUdDb25zdHMuSkNTX1lDQ0EgOiBKUEVHQ29uc3RzLkpDU19ZQ0M7Ci0gICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuVFlQRV9DTVlLOgotICAgICAgICAgICAgICAgICAgdHlwZSA9IEpQRUdDb25zdHMuSkNTX0NNWUs7Ci0gICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdHlwZTsKLSAgICB9Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIGRlc3RpbmF0aW9uIGNvbG9yIHNwYWNlLgotICAgICAqIChZQ2JDcltBXSBmb3IgUkdCKQotICAgICAqCi0gICAgICogQHBhcmFtIGltYWdlCi0gICAgICogQHJldHVybgotICAgICAqLwotICAgIHByaXZhdGUgaW50IGdldERlc3RpbmF0aW9uQ1NUeXBlKFJlbmRlcmVkSW1hZ2UgaW1hZ2UpIHsKLSAgICAgICAgaW50IHR5cGUgPSBKUEVHQ29uc3RzLkpDU19VTktOT1c7Ci0gICAgICAgIENvbG9yTW9kZWwgY20gPSBpbWFnZS5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIGlmIChudWxsICE9IGNtKSB7Ci0gICAgICAgICAgICBib29sZWFuIGhhc0FscGhhID0gY20uaGFzQWxwaGEoKTsKLSAgICAgICAgICAgIENvbG9yU3BhY2UgY3MgPSBjbS5nZXRDb2xvclNwYWNlKCk7Ci0KLSAgICAgICAgICAgIHN3aXRjaChjcy5nZXRUeXBlKCkpIHsKLSAgICAgICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuVFlQRV9HUkFZOgotICAgICAgICAgICAgICAgICAgICB0eXBlID0gSlBFR0NvbnN0cy5KQ1NfR1JBWVNDQUxFOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5UWVBFX1JHQjoKLSAgICAgICAgICAgICAgICAgICAgdHlwZSA9IGhhc0FscGhhID8gSlBFR0NvbnN0cy5KQ1NfWUNiQ3JBIDogSlBFR0NvbnN0cy5KQ1NfWUNiQ3I7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgY2FzZSBDb2xvclNwYWNlLlRZUEVfWUNiQ3I6Ci0gICAgICAgICAgICAgICAgICAgIHR5cGUgPSBoYXNBbHBoYSA/IEpQRUdDb25zdHMuSkNTX1lDYkNyQSA6IEpQRUdDb25zdHMuSkNTX1lDYkNyOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgIGNhc2UgQ29sb3JTcGFjZS5UWVBFXzNDTFI6Ci0gICAgICAgICAgICAgICAgICAgICB0eXBlID0gaGFzQWxwaGEgPyBKUEVHQ29uc3RzLkpDU19ZQ0NBIDogSlBFR0NvbnN0cy5KQ1NfWUNDOwotICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICBjYXNlIENvbG9yU3BhY2UuVFlQRV9DTVlLOgotICAgICAgICAgICAgICAgICAgICAgIHR5cGUgPSBKUEVHQ29uc3RzLkpDU19DTVlLOwotICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiB0eXBlOwotICAgIH0KLQotICAgIHB1YmxpYyBJbWFnZVdyaXRlUGFyYW0gZ2V0RGVmYXVsdFdyaXRlUGFyYW0oKSB7Ci0gICAgICAgIHJldHVybiBuZXcgSlBFR0ltYWdlV3JpdGVQYXJhbShnZXRMb2NhbGUoKSk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9qcGVnL0pQRUdJbWFnZVdyaXRlclNwaS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9qcGVnL0pQRUdJbWFnZVdyaXRlclNwaS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiNzk5MGUwLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHSW1hZ2VXcml0ZXJTcGkuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDU2ICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4zICQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMuanBlZzsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLkltYWdlV3JpdGVyU3BpOwotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VXcml0ZXI7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5JbWFnZVR5cGVTcGVjaWZpZXI7Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLnV0aWwuTG9jYWxlOwotCi1wdWJsaWMgY2xhc3MgSlBFR0ltYWdlV3JpdGVyU3BpIGV4dGVuZHMgSW1hZ2VXcml0ZXJTcGkgewotCi0gICAgcHVibGljIEpQRUdJbWFnZVdyaXRlclNwaSgpIHsKLSAgICAgICAgc3VwZXIoSlBFR1NwaUNvbnN0cy52ZW5kb3JOYW1lLCBKUEVHU3BpQ29uc3RzLnZlcnNpb24sCi0gICAgICAgICAgICAgICAgSlBFR1NwaUNvbnN0cy5uYW1lcywgSlBFR1NwaUNvbnN0cy5zdWZmaXhlcywgSlBFR1NwaUNvbnN0cy5NSU1FVHlwZXMsCi0gICAgICAgICAgICAgICAgSlBFR1NwaUNvbnN0cy53cml0ZXJDbGFzc05hbWUsIFNUQU5EQVJEX09VVFBVVF9UWVBFLAotICAgICAgICAgICAgICAgIEpQRUdTcGlDb25zdHMucmVhZGVyU3BpTmFtZXMsIEpQRUdTcGlDb25zdHMuc3VwcG9ydHNTdGFuZGFyZFN0cmVhbU1ldGFkYXRhRm9ybWF0IC8qVE9ETzogc3VwcG9ydCBzdC4gbWV0YWRhdGEgZm9ybWF0Ki8sCi0gICAgICAgICAgICAgICAgSlBFR1NwaUNvbnN0cy5uYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWUsIEpQRUdTcGlDb25zdHMubmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUsCi0gICAgICAgICAgICAgICAgSlBFR1NwaUNvbnN0cy5leHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZXMsIEpQRUdTcGlDb25zdHMuZXh0cmFTdHJlYW1NZXRhZGF0YUZvcm1hdENsYXNzTmFtZXMsCi0gICAgICAgICAgICAgICAgSlBFR1NwaUNvbnN0cy5zdXBwb3J0c1N0YW5kYXJkSW1hZ2VNZXRhZGF0YUZvcm1hdCwgSlBFR1NwaUNvbnN0cy5uYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0TmFtZSwgSlBFR1NwaUNvbnN0cy5uYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lLAotICAgICAgICAgICAgICAgIEpQRUdTcGlDb25zdHMuZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0TmFtZXMsIEpQRUdTcGlDb25zdHMuZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gY2FuRW5jb2RlSW1hZ2UoSW1hZ2VUeXBlU3BlY2lmaWVyIGltYWdlVHlwZVNwZWNpZmllcikgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSW1hZ2VXcml0ZXIgY3JlYXRlV3JpdGVySW5zdGFuY2UoT2JqZWN0IG8pIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJldHVybiBuZXcgSlBFR0ltYWdlV3JpdGVyKHRoaXMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RGVzY3JpcHRpb24oTG9jYWxlIGxvY2FsZSkgewotICAgICAgICByZXR1cm4gIkRSTCBKUEVHIEVuY29kZXIiOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHU3BpQ29uc3RzLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL2pwZWcvSlBFR1NwaUNvbnN0cy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjM2I0YTUwLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvanBlZy9KUEVHU3BpQ29uc3RzLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw1NyArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotLyoqCi0gKiBAYXV0aG9yIFJ1c3RlbSBWLiBSYWZpa292Ci0gKiBAdmVyc2lvbiAkUmV2aXNpb246IDEuMiAkCi0gKi8KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5wbHVnaW5zLmpwZWc7Ci0KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjIgJAotICovCi1wdWJsaWMgY2xhc3MgSlBFR1NwaUNvbnN0cyB7Ci0gICAgcHJpdmF0ZSBKUEVHU3BpQ29uc3RzKCkge30KLQotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIHZlbmRvck5hbWUgPSAiSW50ZWwgQ29ycG9yYXRpb24iOwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIHZlcnNpb24gPSAiMC4xIGJldGEiOwotCi0gICAgc3RhdGljIGZpbmFsIFN0cmluZyByZWFkZXJDbGFzc05hbWUgPSAib3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5wbHVnaW5zLmpwZWcuSlBFR0ltYWdlUmVhZGVyIjsKLSAgICBzdGF0aWMgZmluYWwgU3RyaW5nIHdyaXRlckNsYXNzTmFtZSA9ICJvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMuanBlZy5KUEVHSW1hZ2VXcml0ZXIiOwotCi0gICAgc3RhdGljIGZpbmFsIFN0cmluZ1tdIG5hbWVzID0geyJqcGVnIiwgImpwZyIsICJKUEVHIiwgIkpQRyJ9OwotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmdbXSBzdWZmaXhlcyA9IHsianBlZyIsICJqcGcifTsKLSAgICBzdGF0aWMgZmluYWwgU3RyaW5nW10gTUlNRVR5cGVzID0geyJpbWFnZS9qcGVnIn07Ci0KLSAgICBzdGF0aWMgZmluYWwgU3RyaW5nW10gd3JpdGVyU3BpTmFtZXMgPSB7Im9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ucGx1Z2lucy5qcGVnLkpQRUdJbWFnZVdyaXRlclNwaSJ9OwotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmdbXSByZWFkZXJTcGlOYW1lcyA9IHsib3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5wbHVnaW5zLmpwZWcuSlBFR0ltYWdlUmVhZGVyU3BpIn07Ci0KLSAgICAvLy0tIFRPRE8gZmlsbCB0aGlzIHN0dWZmIHdpdGggY29ycmVjdCBkYXRhCi0gICAgc3RhdGljIGZpbmFsIGJvb2xlYW4gc3VwcG9ydHNTdGFuZGFyZFN0cmVhbU1ldGFkYXRhRm9ybWF0ID0gZmFsc2U7Ci0gICAgc3RhdGljIGZpbmFsIFN0cmluZyBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWUgPSBudWxsOwotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmcgbmF0aXZlU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWUgPSBudWxsOwotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmdbXSBleHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZXMgPSBudWxsOwotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmdbXSBleHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lcyA9IG51bGw7Ci0gICAgc3RhdGljIGZpbmFsIGJvb2xlYW4gc3VwcG9ydHNTdGFuZGFyZEltYWdlTWV0YWRhdGFGb3JtYXQgPSBmYWxzZTsKLSAgICBzdGF0aWMgZmluYWwgU3RyaW5nIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lID0KLSAgICAgICAgICAgICJvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMuanBlZy5NeUZvcm1hdE1ldGFkYXRhXzEuMCI7Ci0gICAgc3RhdGljIGZpbmFsIFN0cmluZyBuYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lID0KLSAgICAgICAgICAgICJvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMuanBlZy5NeUZvcm1hdE1ldGFkYXRhIjsKLSAgICBzdGF0aWMgZmluYWwgU3RyaW5nW10gZXh0cmFJbWFnZU1ldGFkYXRhRm9ybWF0TmFtZXMgPSBudWxsOwotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmdbXSBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzID0gbnVsbDsKLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9wbmcvUE5HSW1hZ2VSZWFkZXIuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvcG5nL1BOR0ltYWdlUmVhZGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQ4MDA0MWMuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9wbmcvUE5HSW1hZ2VSZWFkZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDEwNiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5wbHVnaW5zLnBuZzsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS5hd3QuZ2wuaW1hZ2UuRGVjb2RpbmdJbWFnZVNvdXJjZTsKLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkuYXd0LmdsLmltYWdlLk9mZnNjcmVlbkltYWdlOwotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ucGx1Z2lucy5qcGVnLklJU0RlY29kaW5nSW1hZ2VTb3VyY2U7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlUmVhZGVyOwotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VUeXBlU3BlY2lmaWVyOwotaW1wb3J0IGphdmF4LmltYWdlaW8uSW1hZ2VSZWFkUGFyYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5wbHVnaW5zLmpwZWcuSlBFR0ltYWdlUmVhZFBhcmFtOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLkltYWdlUmVhZGVyU3BpOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3RyZWFtLkltYWdlSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5tZXRhZGF0YS5JSU9NZXRhZGF0YTsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEudXRpbC5JdGVyYXRvcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5CdWZmZXJlZEltYWdlOwotCi1wdWJsaWMgY2xhc3MgUE5HSW1hZ2VSZWFkZXIgIGV4dGVuZHMgSW1hZ2VSZWFkZXIgewotICAgIEltYWdlSW5wdXRTdHJlYW0gaWlzOwotCi0gICAgcHVibGljIFBOR0ltYWdlUmVhZGVyKEltYWdlUmVhZGVyU3BpIGltYWdlUmVhZGVyU3BpKSB7Ci0gICAgICAgIHN1cGVyKGltYWdlUmVhZGVyU3BpKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldE51bUltYWdlcyhib29sZWFuIGFsbG93U2VhcmNoKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLy0tIFRPRE8gaW1sZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJub3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBnZXRXaWR0aChpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgLy8tLSBUT0RPIGltbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigibm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0SGVpZ2h0KGludCBpbWFnZUluZGV4KSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICAvLy0tIFRPRE8gaW1sZW1lbnQKLSAgICAgICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJub3QgaW1wbGVtZW50ZWQgeWV0Iik7Ci0gICAgfQotCi0gICAgcHVibGljIEl0ZXJhdG9yPEltYWdlVHlwZVNwZWNpZmllcj4gZ2V0SW1hZ2VUeXBlcyhpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgLy8tLSBUT0RPIGltbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigibm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJSU9NZXRhZGF0YSBnZXRTdHJlYW1NZXRhZGF0YSgpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIC8vLS0gVE9ETyBpbWxlbWVudAotICAgICAgICB0aHJvdyBuZXcgVW5zdXBwb3J0ZWRPcGVyYXRpb25FeGNlcHRpb24oIm5vdCBpbXBsZW1lbnRlZCB5ZXQiKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSUlPTWV0YWRhdGEgZ2V0SW1hZ2VNZXRhZGF0YShpbnQgaW1hZ2VJbmRleCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgLy8tLSBUT0RPIGltbGVtZW50Ci0gICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigibm90IGltcGxlbWVudGVkIHlldCIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBCdWZmZXJlZEltYWdlIHJlYWQoaW50IGksIEltYWdlUmVhZFBhcmFtIGltYWdlUmVhZFBhcmFtKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAoaWlzID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oImlucHV0IHN0cmVhbSA9PSBudWxsIik7Ci0gICAgICAgIH0KLQotICAgICAgICBEZWNvZGluZ0ltYWdlU291cmNlIHNvdXJjZSA9IG5ldyBJSVNEZWNvZGluZ0ltYWdlU291cmNlKGlpcyk7Ci0gICAgICAgIE9mZnNjcmVlbkltYWdlIGltYWdlID0gbmV3IE9mZnNjcmVlbkltYWdlKHNvdXJjZSk7Ci0gICAgICAgIHNvdXJjZS5hZGRDb25zdW1lcihpbWFnZSk7Ci0gICAgICAgIHNvdXJjZS5sb2FkKCk7Ci0gICAgICAgIC8vIFRoZSBpbnRlcnJ1cHRlZCBmbGFnIHNob3VsZCBiZSBjbGVhcmVkIGJlY2F1c2UgSW1hZ2VEZWNvZGVyIGludGVycnVwdHMKLSAgICAgICAgLy8gY3VycmVudCB0aHJlYWQgd2hpbGUgZGVjb2RpbmcgKGR1ZSBpdHMgYXJjaGl0ZWN0dXJlKS4KLSAgICAgICAgVGhyZWFkLmludGVycnVwdGVkKCk7Ci0gICAgICAgIHJldHVybiBpbWFnZS5nZXRCdWZmZXJlZEltYWdlKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEJ1ZmZlcmVkSW1hZ2UgcmVhZChpbnQgaSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgcmV0dXJuIHJlYWQoaSwgbnVsbCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgc2V0SW5wdXQoT2JqZWN0IGlucHV0LCBib29sZWFuIHNlZWtGb3J3YXJkT25seSwgYm9vbGVhbiBpZ25vcmVNZXRhZGF0YSkgewotICAgICAgICBzdXBlci5zZXRJbnB1dChpbnB1dCwgc2Vla0ZvcndhcmRPbmx5LCBpZ25vcmVNZXRhZGF0YSk7Ci0gICAgICAgIGlpcyA9IChJbWFnZUlucHV0U3RyZWFtKSBpbnB1dDsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSW1hZ2VSZWFkUGFyYW0gZ2V0RGVmYXVsdFJlYWRQYXJhbSgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBJbWFnZVJlYWRQYXJhbSgpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvcG5nL1BOR0ltYWdlUmVhZGVyU3BpLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL3BuZy9QTkdJbWFnZVJlYWRlclNwaS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1MGY4YjEwLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvcG5nL1BOR0ltYWdlUmVhZGVyU3BpLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4OCArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5wbHVnaW5zLnBuZzsKLQotaW1wb3J0IG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ucGx1Z2lucy5qcGVnLkpQRUdTcGlDb25zdHM7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLnNwaS5JbWFnZVJlYWRlclNwaTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLnNwaS5TZXJ2aWNlUmVnaXN0cnk7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5JbWFnZVJlYWRlcjsKLWltcG9ydCBqYXZheC5pbWFnZWlvLnN0cmVhbS5JbWFnZUlucHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLQotcHVibGljIGNsYXNzIFBOR0ltYWdlUmVhZGVyU3BpIGV4dGVuZHMgSW1hZ2VSZWFkZXJTcGkgewotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmcgUE5HX05BTUVTW10gPSBuZXcgU3RyaW5nW10geyJwbmciLCAiUE5HIn07Ci0gICAgc3RhdGljIGZpbmFsIFN0cmluZyBQTkdfU1VGRklYRVNbXSA9IG5ldyBTdHJpbmdbXSB7InBuZyJ9OwotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmcgUE5HX01JTUVfVFlQRVNbXSA9IG5ldyBTdHJpbmdbXSB7ImltYWdlL3BuZyJ9OwotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmcgUE5HX1JFQURFUl9DTEFTU19OQU1FID0gIm9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ucGx1Z2lucy5wbmcuUE5HSW1hZ2VSZWFkZXIiOwotICAgIHN0YXRpYyBmaW5hbCBTdHJpbmcgUE5HX1JFQURFUl9TUElfTkFNRVNbXSA9IHsib3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5wbHVnaW5zLnBuZy5QTkdJbWFnZVJlYWRlclNwaSJ9OwotCi0gICAgcHVibGljIFBOR0ltYWdlUmVhZGVyU3BpKCkgewotICAgICAgICBzdXBlcigKLSAgICAgICAgICAgICAgICBKUEVHU3BpQ29uc3RzLnZlbmRvck5hbWUsIEpQRUdTcGlDb25zdHMudmVyc2lvbiwKLSAgICAgICAgICAgICAgICBQTkdfTkFNRVMsIFBOR19TVUZGSVhFUywKLSAgICAgICAgICAgICAgICBQTkdfTUlNRV9UWVBFUywgUE5HX1JFQURFUl9DTEFTU19OQU1FLAotICAgICAgICAgICAgICAgIFNUQU5EQVJEX0lOUFVUX1RZUEUsIG51bGwsCi0gICAgICAgICAgICAgICAgZmFsc2UsIG51bGwsCi0gICAgICAgICAgICAgICAgbnVsbCwgbnVsbCwKLSAgICAgICAgICAgICAgICBudWxsLCBmYWxzZSwgCi0gICAgICAgICAgICAgICAgbnVsbCwgbnVsbCwKLSAgICAgICAgICAgICAgICBudWxsLCBudWxsCi0gICAgICAgICk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gY2FuRGVjb2RlSW5wdXQoT2JqZWN0IHNvdXJjZSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgSW1hZ2VJbnB1dFN0cmVhbSBtYXJrYWJsZSA9IChJbWFnZUlucHV0U3RyZWFtKSBzb3VyY2U7Ci0gICAgICAgIG1hcmthYmxlLm1hcmsoKTsKLQotICAgICAgICBieXRlW10gc2lnbmF0dXJlID0gbmV3IGJ5dGVbOF07Ci0gICAgICAgIG1hcmthYmxlLnNlZWsoMCk7Ci0KLSAgICAgICAgaW50IG5CeXRlcyA9IG1hcmthYmxlLnJlYWQoc2lnbmF0dXJlLCAwLCA4KTsKLSAgICAgICAgaWYobkJ5dGVzICE9IDgpIG1hcmthYmxlLnJlYWQoc2lnbmF0dXJlLCBuQnl0ZXMsIDgtbkJ5dGVzKTsKLSAgICAgICAgbWFya2FibGUucmVzZXQoKTsKLQotICAgICAgICAvLyBQTkcgc2lnbmF0dXJlOiAxMzcgODAgNzggNzEgMTMgMTAgMjYgMTAKLSAgICAgICAgcmV0dXJuICAoc2lnbmF0dXJlWzBdICYgMHhGRikgPT0gMTM3ICYmCi0gICAgICAgICAgICAgICAgKHNpZ25hdHVyZVsxXSAmIDB4RkYpID09IDgwICYmCi0gICAgICAgICAgICAgICAgKHNpZ25hdHVyZVsyXSAmIDB4RkYpID09IDc4ICYmCi0gICAgICAgICAgICAgICAgKHNpZ25hdHVyZVszXSAmIDB4RkYpID09IDcxICYmCi0gICAgICAgICAgICAgICAgKHNpZ25hdHVyZVs0XSAmIDB4RkYpID09IDEzICYmCi0gICAgICAgICAgICAgICAgKHNpZ25hdHVyZVs1XSAmIDB4RkYpID09IDEwICYmCi0gICAgICAgICAgICAgICAgKHNpZ25hdHVyZVs2XSAmIDB4RkYpID09IDI2ICYmCi0gICAgICAgICAgICAgICAgKHNpZ25hdHVyZVs3XSAmIDB4RkYpID09IDEwOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJbWFnZVJlYWRlciBjcmVhdGVSZWFkZXJJbnN0YW5jZShPYmplY3QgZXh0ZW5zaW9uKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gbmV3IFBOR0ltYWdlUmVhZGVyKHRoaXMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RGVzY3JpcHRpb24oTG9jYWxlIGxvY2FsZSkgewotICAgICAgICByZXR1cm4gIkRSTCBQTkcgZGVjb2RlciI7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIHZvaWQgb25SZWdpc3RyYXRpb24oU2VydmljZVJlZ2lzdHJ5IHJlZ2lzdHJ5LCBDbGFzczw/PiBjYXRlZ29yeSkgewotICAgICAgICBzdXBlci5vblJlZ2lzdHJhdGlvbihyZWdpc3RyeSwgY2F0ZWdvcnkpOwotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvcG5nL1BOR0ltYWdlV3JpdGVyLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL3BuZy9QTkdJbWFnZVdyaXRlci5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlMmE4ZDdkLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvcG5nL1BOR0ltYWdlV3JpdGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwyNDcgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBWaXNrb3YgTmlrb2xheQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ucGx1Z2lucy5wbmc7Ci0KLWltcG9ydCBjb20uYW5kcm9pZC5pbnRlcm5hbC5hd3QuSW1hZ2VPdXRwdXRTdHJlYW1XcmFwcGVyOwotCi1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuQnVmZmVyZWRJbWFnZTsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJCeXRlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJJbnQ7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuSW5kZXhDb2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlJhc3RlcjsKLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5SZW5kZXJlZEltYWdlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlNhbXBsZU1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLlNpbmdsZVBpeGVsUGFja2VkU2FtcGxlTW9kZWw7Ci1pbXBvcnQgamF2YS5hd3QuaW1hZ2UuV3JpdGFibGVSYXN0ZXI7Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uSUlPSW1hZ2U7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5JbWFnZVR5cGVTcGVjaWZpZXI7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5JbWFnZVdyaXRlUGFyYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5JbWFnZVdyaXRlcjsKLWltcG9ydCBqYXZheC5pbWFnZWlvLm1ldGFkYXRhLklJT01ldGFkYXRhOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLkltYWdlV3JpdGVyU3BpOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3RyZWFtLkltYWdlT3V0cHV0U3RyZWFtOwotCi1pbXBvcnQgb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5pbnRlcm5hbC5ubHMuTWVzc2FnZXM7Ci0KLWltcG9ydCBvcmcuYXBhY2hlLmhhcm1vbnkubHVuaS51dGlsLk5vdEltcGxlbWVudGVkRXhjZXB0aW9uOwotCi1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5CaXRtYXA7Ci1pbXBvcnQgYW5kcm9pZC5ncmFwaGljcy5CaXRtYXAuQ29tcHJlc3NGb3JtYXQ7Ci0KLXB1YmxpYyBjbGFzcyBQTkdJbWFnZVdyaXRlciBleHRlbmRzIEltYWdlV3JpdGVyIHsKLSAgICAKLSAgICAvLyAvKiA/Pz9BV1Q6IERlYnVnZ2luZwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGJvb2xlYW4gREVCVUcgPSBmYWxzZTsKLSAgICBwcml2YXRlIHN0YXRpYyBCaXRtYXAgYm07Ci0gICAgcHVibGljIHN0YXRpYyBCaXRtYXAgZ2V0Qml0bWFwKCkgewotICAgICAgICByZXR1cm4gYm07Ci0gICAgfQotICAgIC8vICovCi0gICAgCi0gICAgcHJpdmF0ZSBzdGF0aWMgaW50W11bXSBCQU5EX09GRlNFVFMgPSB7Ci0gICAgICAgICAgICB7fSwgewotICAgICAgICAgICAgICAgIDAgfSwgewotICAgICAgICAgICAgICAgICAgICAwLCAxIH0sIHsKLSAgICAgICAgICAgICAgICAgICAgMCwgMSwgMiB9LCB7Ci0gICAgICAgICAgICAgICAgICAgIDAsIDEsIDIsIDMgfSB9OwotCi0gICAgLy8gRWFjaCBwaXhlbCBpcyBhIGdyYXlzY2FsZSBzYW1wbGUuCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFBOR19DT0xPUl9UWVBFX0dSQVkgPSAwOwotICAgIC8vIEVhY2ggcGl4ZWwgaXMgYW4gUixHLEIgdHJpcGxlLgotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBQTkdfQ09MT1JfVFlQRV9SR0IgPSAyOwotICAgIC8vIEVhY2ggcGl4ZWwgaXMgYSBwYWxldHRlIGluZGV4LCBhIFBMVEUgY2h1bmsgbXVzdCBhcHBlYXIuCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFBOR19DT0xPUl9UWVBFX1BMVEUgPSAzOwotICAgIC8vIEVhY2ggcGl4ZWwgaXMgYSBncmF5c2NhbGUgc2FtcGxlLCBmb2xsb3dlZCBieSBhbiBhbHBoYSBzYW1wbGUuCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFBOR19DT0xPUl9UWVBFX0dSQVlfQUxQSEEgPSA0OwotICAgIC8vIEVhY2ggcGl4ZWwgaXMgYW4gUixHLEIgdHJpcGxlLCBmb2xsb3dlZCBieSBhbiBhbHBoYSBzYW1wbGUuCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgaW50IFBOR19DT0xPUl9UWVBFX1JHQkEgPSA2OwotICAgIAotICAgIC8vPz8/QVdUOiBwcml2YXRlIHN0YXRpYyBuYXRpdmUgdm9pZCBpbml0SURzKENsYXNzPEltYWdlT3V0cHV0U3RyZWFtPiBpb3NDbGFzcyk7Ci0KLSAgICBzdGF0aWMgewotICAgICAgICAvLz8/P0FXVAotICAgICAgICAvKgotICAgICAgICBTeXN0ZW0ubG9hZExpYnJhcnkoInBuZ2VuY29kZXIiKTsgLy8kTk9OLU5MUy0xJAotICAgICAgICBpbml0SURzKEltYWdlT3V0cHV0U3RyZWFtLmNsYXNzKTsKLSAgICAgICAgKi8KLSAgICB9Ci0gICAgCi0gICAgLyoKLSAgICBwcml2YXRlIG5hdGl2ZSBpbnQgZW5jb2RlKGJ5dGVbXSBpbnB1dCwgaW50IGJ5dGVzSW5CdWZmZXIsIGludCBieXRlUGl4ZWxTaXplLCBPYmplY3QgaW9zLCBpbnQgaW1hZ2VXaWR0aCwKLSAgICAgICAgICAgIGludCBpbWFnZUhlaWdodCwgaW50IGJpdERlcHRoLCBpbnQgY29sb3JUeXBlLCBpbnRbXSBwYWxldHRlLCBpbnQgaSwgYm9vbGVhbiBiKTsKLSAgICAqLwotICAgIAotICAgIHByb3RlY3RlZCBQTkdJbWFnZVdyaXRlcihJbWFnZVdyaXRlclNwaSBpd1NwaSkgewotICAgICAgICBzdXBlcihpd1NwaSk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIElJT01ldGFkYXRhIGNvbnZlcnRTdHJlYW1NZXRhZGF0YShJSU9NZXRhZGF0YSBhcmcwLCBJbWFnZVdyaXRlUGFyYW0gYXJnMSkgewotICAgICAgICB0aHJvdyBuZXcgTm90SW1wbGVtZW50ZWRFeGNlcHRpb24oKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSUlPTWV0YWRhdGEgY29udmVydEltYWdlTWV0YWRhdGEoSUlPTWV0YWRhdGEgYXJnMCwgSW1hZ2VUeXBlU3BlY2lmaWVyIGFyZzEsIEltYWdlV3JpdGVQYXJhbSBhcmcyKSB7Ci0gICAgICAgIHRocm93IG5ldyBOb3RJbXBsZW1lbnRlZEV4Y2VwdGlvbigpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJSU9NZXRhZGF0YSBnZXREZWZhdWx0SW1hZ2VNZXRhZGF0YShJbWFnZVR5cGVTcGVjaWZpZXIgYXJnMCwgSW1hZ2VXcml0ZVBhcmFtIGFyZzEpIHsKLSAgICAgICAgdGhyb3cgbmV3IE5vdEltcGxlbWVudGVkRXhjZXB0aW9uKCk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIElJT01ldGFkYXRhIGdldERlZmF1bHRTdHJlYW1NZXRhZGF0YShJbWFnZVdyaXRlUGFyYW0gYXJnMCkgewotICAgICAgICB0aHJvdyBuZXcgTm90SW1wbGVtZW50ZWRFeGNlcHRpb24oKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgdm9pZCB3cml0ZShJSU9NZXRhZGF0YSBzdHJlYW1NZXRhZGF0YSwgSUlPSW1hZ2UgaWlvSW1hZ2UsIEltYWdlV3JpdGVQYXJhbSBwYXJhbSkgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKG91dHB1dCA9PSBudWxsKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKCJPdXRwdXQgbm90IGJlZW4gc2V0Iik7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGlpb0ltYWdlID09IG51bGwpIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIkltYWdlIGVxdWFscyBudWxsIik7Ci0gICAgICAgIH0KLSAgICAgICAgLy8gQVdUPz8/OiBJIHRoaW5rIHRoaXMgaXMgbm90IG5lZWRlZCBhbnltb3JlCi0gICAgICAgIC8vIGlmIChpaW9JbWFnZS5oYXNSYXN0ZXIoKSAmJiAhY2FuV3JpdGVSYXN0ZXJzKCkpIHsKLSAgICAgICAgLy8gICAgdGhyb3cgbmV3IFVuc3VwcG9ydGVkT3BlcmF0aW9uRXhjZXB0aW9uKCJDYW4ndCB3cml0ZSByYXN0ZXIiKTsKLSAgICAgICAgLy99Ly8gSW1hZ2VPdXRwdXRTdHJlYW1JbXBsCi0gICAgICAgIAotICAgICAgICBSYXN0ZXIgc291cmNlUmFzdGVyOwotICAgICAgICBSZW5kZXJlZEltYWdlIGltZyA9IG51bGw7Ci0gICAgICAgIGlmICghaWlvSW1hZ2UuaGFzUmFzdGVyKCkpIHsKLSAgICAgICAgICAgIGltZyA9IGlpb0ltYWdlLmdldFJlbmRlcmVkSW1hZ2UoKTsKLSAgICAgICAgICAgIGlmIChpbWcgaW5zdGFuY2VvZiBCdWZmZXJlZEltYWdlKSB7Ci0gICAgICAgICAgICAgICAgc291cmNlUmFzdGVyID0gKChCdWZmZXJlZEltYWdlKSBpbWcpLmdldFJhc3RlcigpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBzb3VyY2VSYXN0ZXIgPSBpbWcuZ2V0RGF0YSgpOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgc291cmNlUmFzdGVyID0gaWlvSW1hZ2UuZ2V0UmFzdGVyKCk7Ci0gICAgICAgIH0KLQotICAgICAgICBTYW1wbGVNb2RlbCBtb2RlbCA9IHNvdXJjZVJhc3Rlci5nZXRTYW1wbGVNb2RlbCgpOwotICAgICAgICBpbnQgc3JjV2lkdGggPSBzb3VyY2VSYXN0ZXIuZ2V0V2lkdGgoKTsKLSAgICAgICAgaW50IHNyY0hlaWdodCA9IHNvdXJjZVJhc3Rlci5nZXRIZWlnaHQoKTsKLSAgICAgICAgaW50IG51bUJhbmRzID0gbW9kZWwuZ2V0TnVtQmFuZHMoKTsKLSAgICAgICAgCi0gICAgICAgIENvbG9yTW9kZWwgY29sb3JNb2RlbCA9IGltZy5nZXRDb2xvck1vZGVsKCk7Ci0gICAgICAgIGludCBwaXhlbFNpemUgPSBjb2xvck1vZGVsLmdldFBpeGVsU2l6ZSgpOwotICAgICAgICBpbnQgYnl0ZVBpeGVsU2l6ZSA9IHBpeGVsU2l6ZSAvIDg7Ci0gICAgICAgIGludCBiaXREZXB0aCA9IHBpeGVsU2l6ZSAvIG51bUJhbmRzOwotICAgICAgICAKLSAgICAgICAgLy8gYnl0ZSBwZXIgYmFuZAotICAgICAgICBpbnQgYnBiID0gYml0RGVwdGggPiA4ID8gMiA6IDE7Ci0gICAgICAgIAotICAgICAgICBib29sZWFuIGlzSW50ZXJsYWNlID0gdHJ1ZTsKLSAgICAgICAgaWYgKHBhcmFtIGluc3RhbmNlb2YgUE5HSW1hZ2VXcml0ZXJQYXJhbSkgewotICAgICAgICAgICAgaXNJbnRlcmxhY2UgPSAoKFBOR0ltYWdlV3JpdGVyUGFyYW0pIHBhcmFtKS5nZXRJbnRlcmxhY2UoKTsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgaW50IGNvbG9yVHlwZSA9IFBOR19DT0xPUl9UWVBFX0dSQVk7Ci0gICAgICAgIGludFtdIHBhbGV0dGUgPSBudWxsOwotICAgICAgICAKLSAgICAgICAgaWYgKGNvbG9yTW9kZWwgaW5zdGFuY2VvZiBJbmRleENvbG9yTW9kZWwpIHsKLSAgICAgICAgICAgIGlmIChiaXREZXB0aCAhPSAxICYmIGJpdERlcHRoICE9IDIgJiYgYml0RGVwdGggIT0gNCAmJiBiaXREZXB0aCAhPSA4KSB7Ci0vLyAgICAgICAgICAgICAgV3JvbmcgYml0RGVwdGgtbnVtQmFuZHMgY29tcG9zaXRpb24KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiaW1hZ2Vpby4xIikpOy8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChudW1CYW5kcyAhPSAxKSB7Ci0vLyAgICAgICAgICAgICAgV3JvbmcgYml0RGVwdGgtbnVtQmFuZHMgY29tcG9zaXRpb24KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiaW1hZ2Vpby4xIikpOy8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgSW5kZXhDb2xvck1vZGVsIGljbSA9IChJbmRleENvbG9yTW9kZWwpIGNvbG9yTW9kZWw7Ci0gICAgICAgICAgICBwYWxldHRlID0gbmV3IGludFtpY20uZ2V0TWFwU2l6ZSgpXTsKLSAgICAgICAgICAgIGljbS5nZXRSR0JzKHBhbGV0dGUpOwotICAgICAgICAgICAgY29sb3JUeXBlID0gUE5HX0NPTE9SX1RZUEVfUExURTsKLSAgICAgICAgfQotICAgICAgICBlbHNlIGlmIChudW1CYW5kcyA9PSAxKSB7Ci0gICAgICAgICAgICBpZiAoYml0RGVwdGggIT0gMSAmJiBiaXREZXB0aCAhPSAyICYmIGJpdERlcHRoICE9IDQgJiYgYml0RGVwdGggIT0gOCAmJiBiaXREZXB0aCAhPSAxNikgewotLy8gICAgICAgICAgICAgIFdyb25nIGJpdERlcHRoLW51bUJhbmRzIGNvbXBvc2l0aW9uCi0gICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbihNZXNzYWdlcy5nZXRTdHJpbmcoImltYWdlaW8uMSIpKTsvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjb2xvclR5cGUgPSBQTkdfQ09MT1JfVFlQRV9HUkFZOwotICAgICAgICB9Ci0gICAgICAgIGVsc2UgaWYgKG51bUJhbmRzID09IDIpIHsKLSAgICAgICAgICAgIGlmIChiaXREZXB0aCAhPSA4ICYmIGJpdERlcHRoICE9IDE2KSB7Ci0vLyAgICAgICAgICAgICAgV3JvbmcgYml0RGVwdGgtbnVtQmFuZHMgY29tcG9zaXRpb24KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiaW1hZ2Vpby4xIikpOy8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNvbG9yVHlwZSA9IFBOR19DT0xPUl9UWVBFX0dSQVlfQUxQSEE7Ci0gICAgICAgIH0KLSAgICAgICAgZWxzZSBpZiAobnVtQmFuZHMgPT0gMykgewotICAgICAgICAgICAgaWYgKGJpdERlcHRoICE9IDggJiYgYml0RGVwdGggIT0gMTYpIHsKLS8vICAgICAgICAgICAgICBXcm9uZyBiaXREZXB0aC1udW1CYW5kcyBjb21wb3NpdGlvbgotICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oTWVzc2FnZXMuZ2V0U3RyaW5nKCJpbWFnZWlvLjEiKSk7IC8vJE5PTi1OTFMtMSQKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNvbG9yVHlwZSA9IFBOR19DT0xPUl9UWVBFX1JHQjsKLSAgICAgICAgfQotICAgICAgICBlbHNlIGlmIChudW1CYW5kcyA9PSA0KSB7Ci0gICAgICAgICAgICBpZiAoYml0RGVwdGggIT0gOCAmJiBiaXREZXB0aCAhPSAxNikgewotICAgICAgICAgICAgICAgIC8vV3JvbmcgYml0RGVwdGgtbnVtQmFuZHMgY29tcG9zaXRpb24KLSAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKE1lc3NhZ2VzLmdldFN0cmluZygiaW1hZ2Vpby4xIikpOyAvLyROT04tTkxTLTEkCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjb2xvclR5cGUgPSBQTkdfQ09MT1JfVFlQRV9SR0JBOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICAvKiA/Pz9BV1Q6IEkgdGhpbmsgdGhpcyBpcyBub3QgbmVlZGVkIGFueW1vcmUKLSAgICAgICAgaW50IGRidWZmZXJMZW5naHQgPSBieXRlUGl4ZWxTaXplICogaW1hZ2VIZWlnaHQgKiBpbWFnZVdpZHRoOwotICAgICAgICBEYXRhQnVmZmVyQnl0ZSBkYnVmZmVyID0gbmV3IERhdGFCdWZmZXJCeXRlKGRidWZmZXJMZW5naHQpOwotCi0gICAgICAgIFdyaXRhYmxlUmFzdGVyIHNjYW5SYXN0ZXIgPSBSYXN0ZXIuY3JlYXRlSW50ZXJsZWF2ZWRSYXN0ZXIoZGJ1ZmZlciwgaW1hZ2VXaWR0aCwgaW1hZ2VIZWlnaHQsIGJwYiAqIG51bUJhbmRzCi0gICAgICAgICAgICAgICAgKiBpbWFnZVdpZHRoLCBicGIgKiBudW1CYW5kcywgQkFORF9PRkZTRVRTW251bUJhbmRzXSwgbnVsbCk7Ci0KLSAgICAgICAgc2NhblJhc3Rlci5zZXRSZWN0KCgoQnVmZmVyZWRJbWFnZSkgaW1hZ2UpLmdldFJhc3RlcigpLy8gaW1hZ2UuZ2V0RGF0YSgpCi0gICAgICAgICAgICAgICAgLmNyZWF0ZUNoaWxkKDAsIDAsIGltYWdlV2lkdGgsIGltYWdlSGVpZ2h0LCAwLCAwLCBudWxsKSk7Ci0gICAgICAgICovCi0KLSAgICAgICAgaWYgKERFQlVHKSB7Ci0gICAgICAgICAgICBTeXN0ZW0ub3V0LnByaW50bG4oIioqKiogcmFzdGVyOiIgKyBzb3VyY2VSYXN0ZXIpOyAgICAgICAgCi0gICAgICAgICAgICBTeXN0ZW0ub3V0LnByaW50bG4oIioqKiogbW9kZWw6IiArIG1vZGVsKTsKLSAgICAgICAgICAgIFN5c3RlbS5vdXQucHJpbnRsbigiKioqKiB0eXBlOiIgKyBjb2xvclR5cGUpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBpZiAobW9kZWwgaW5zdGFuY2VvZiBTaW5nbGVQaXhlbFBhY2tlZFNhbXBsZU1vZGVsKSB7Ci0gICAgICAgICAgICBEYXRhQnVmZmVySW50IGlidWYgPSAoRGF0YUJ1ZmZlckludClzb3VyY2VSYXN0ZXIuZ2V0RGF0YUJ1ZmZlcigpOwotICAgICAgICAgICAgaW50W10gcGl4ZWxzID0gaWJ1Zi5nZXREYXRhKCk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8vIENyZWF0ZSBhIGJpdG1hcCB3aXRoIHRoZSBwaXhlbAotICAgICAgICAgICAgYm0gPSBCaXRtYXAuY3JlYXRlQml0bWFwKHBpeGVscywgc3JjV2lkdGgsIHNyY0hlaWdodCwgQml0bWFwLkNvbmZpZy5BUkdCXzg4ODgpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyBVc2UgQml0bWFwLmNvbXByZXNzKCkgdG8gd3JpdGUgdGhlIGltYWdlCi0gICAgICAgICAgICBJbWFnZU91dHB1dFN0cmVhbSBpb3MgPSAoSW1hZ2VPdXRwdXRTdHJlYW0pIGdldE91dHB1dCgpOwotICAgICAgICAgICAgSW1hZ2VPdXRwdXRTdHJlYW1XcmFwcGVyIGlvc3cgPSBuZXcgSW1hZ2VPdXRwdXRTdHJlYW1XcmFwcGVyKGlvcyk7Ci0gICAgICAgICAgICBibS5jb21wcmVzcyhDb21wcmVzc0Zvcm1hdC5QTkcsIDEwMCwgaW9zdyk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyA/Pz9BV1Q6IEFkZCBzdXBwb3J0IGZvciBvdGhlciBjb2xvciBtb2RlbHMKLSAgICAgICAgICAgIHRocm93IG5ldyBSdW50aW1lRXhjZXB0aW9uKCJDb2xvciBtb2RlbCBub3Qgc3VwcG9ydGVkIHlldCIpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIEltYWdlV3JpdGVQYXJhbSBnZXREZWZhdWx0V3JpdGVQYXJhbSgpIHsKLSAgICAgICAgcmV0dXJuIG5ldyBQTkdJbWFnZVdyaXRlclBhcmFtKCk7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9wbmcvUE5HSW1hZ2VXcml0ZXJQYXJhbS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vcGx1Z2lucy9wbmcvUE5HSW1hZ2VXcml0ZXJQYXJhbS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiZjNhMDAwLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvcG5nL1BOR0ltYWdlV3JpdGVyUGFyYW0uamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDQxICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgVmlza292IE5pa29sYXkKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbiQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMucG5nOwotCi1pbXBvcnQgamF2YXguaW1hZ2Vpby5JbWFnZVdyaXRlUGFyYW07Ci0KLXB1YmxpYyBjbGFzcyBQTkdJbWFnZVdyaXRlclBhcmFtIGV4dGVuZHMgSW1hZ2VXcml0ZVBhcmFtIHsKLQotICAgIHByaXZhdGUgYm9vbGVhbiBpc0ludGVybGFjZSA9IHRydWU7Ci0KLSAgICBwdWJsaWMgUE5HSW1hZ2VXcml0ZXJQYXJhbSgpIHsKLSAgICAgICAgc3VwZXIoKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgYm9vbGVhbiBnZXRJbnRlcmxhY2UoKSB7Ci0gICAgICAgIHJldHVybiBpc0ludGVybGFjZTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBzZXRJbnRlcmxhY2UoYm9vbGVhbiBiKSB7Ci0gICAgICAgIGlzSW50ZXJsYWNlID0gYjsKLSAgICB9Ci0KLX0KZGlmZiAtLWdpdCBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvcG5nL1BOR0ltYWdlV3JpdGVyU3BpLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9wbHVnaW5zL3BuZy9QTkdJbWFnZVdyaXRlclNwaS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2ZWVkMTRkLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3BsdWdpbnMvcG5nL1BOR0ltYWdlV3JpdGVyU3BpLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMTMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBWaXNrb3YgTmlrb2xheQotICogQHZlcnNpb24gJFJldmlzaW9uJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8ucGx1Z2lucy5wbmc7Ci0KLWltcG9ydCBqYXZhLmF3dC5pbWFnZS5Db2xvck1vZGVsOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkRhdGFCdWZmZXJCeXRlOwotaW1wb3J0IGphdmEuYXd0LmltYWdlLkluZGV4Q29sb3JNb2RlbDsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci0KLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlVHlwZVNwZWNpZmllcjsKLWltcG9ydCBqYXZheC5pbWFnZWlvLkltYWdlV3JpdGVyOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLkltYWdlV3JpdGVyU3BpOwotCi1wdWJsaWMgY2xhc3MgUE5HSW1hZ2VXcml0ZXJTcGkgZXh0ZW5kcyBJbWFnZVdyaXRlclNwaSB7Ci0KLSAgICBwdWJsaWMgUE5HSW1hZ2VXcml0ZXJTcGkoKSB7Ci0gICAgICAgIHN1cGVyKCJJbnRlbCBDb3Jwb3JhdGlvbiIsLy8gdmVuZG9yTmFtZQotICAgICAgICAgICAgICAgICIxLjAiLC8vIHZlcnNpb24KLSAgICAgICAgICAgICAgICBuZXcgU3RyaW5nW10gewotICAgICAgICAgICAgICAgICAgICAgICAgInBuZyIsICJQTkciIH0sLy8gbmFtZXMKLSAgICAgICAgICAgICAgICBuZXcgU3RyaW5nW10gewotICAgICAgICAgICAgICAgICAgICAgICAgInBuZyIsICJQTkciIH0sLy8gc3VmZml4ZXMKLSAgICAgICAgICAgICAgICBuZXcgU3RyaW5nW10gewotICAgICAgICAgICAgICAgICAgICAiaW1hZ2UvcG5nIiB9LC8vIE1JTUVUeXBlcwotICAgICAgICAgICAgICAgICJvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMucG5nLlBOR0ltYWdlV3JpdGVyIiwvLyB3cml0ZXJDbGFzc05hbWUKLSAgICAgICAgICAgICAgICBTVEFOREFSRF9PVVRQVVRfVFlQRSwvLyBvdXRwdXRUeXBlcwotICAgICAgICAgICAgICAgIG5ldyBTdHJpbmdbXSB7Ci0gICAgICAgICAgICAgICAgICAgICJvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnBsdWdpbnMucG5nLlBOR0ltYWdlV3JpdGVyU3BpIiB9LC8vIHJlYWRlclNwaU5hbWVzCi0gICAgICAgICAgICAgICAgZmFsc2UsLy8gc3VwcG9ydHNTdGFuZGFyZFN0cmVhbU1ldGFkYXRhRm9ybWF0Ci0gICAgICAgICAgICAgICAgbnVsbCwvLyBuYXRpdmVTdHJlYW1NZXRhZGF0YUZvcm1hdE5hbWUKLSAgICAgICAgICAgICAgICBudWxsLC8vIG5hdGl2ZVN0cmVhbU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lCi0gICAgICAgICAgICAgICAgbnVsbCwvLyBleHRyYVN0cmVhbU1ldGFkYXRhRm9ybWF0TmFtZXMKLSAgICAgICAgICAgICAgICBudWxsLC8vIGV4dHJhU3RyZWFtTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzCi0gICAgICAgICAgICAgICAgZmFsc2UsLy8gc3VwcG9ydHNTdGFuZGFyZEltYWdlTWV0YWRhdGFGb3JtYXQKLSAgICAgICAgICAgICAgICBudWxsLC8vIG5hdGl2ZUltYWdlTWV0YWRhdGFGb3JtYXROYW1lCi0gICAgICAgICAgICAgICAgbnVsbCwvLyBuYXRpdmVJbWFnZU1ldGFkYXRhRm9ybWF0Q2xhc3NOYW1lCi0gICAgICAgICAgICAgICAgbnVsbCwvLyBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXROYW1lcwotICAgICAgICAgICAgICAgIG51bGwvLyBleHRyYUltYWdlTWV0YWRhdGFGb3JtYXRDbGFzc05hbWVzCi0gICAgICAgICk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gY2FuRW5jb2RlSW1hZ2UoSW1hZ2VUeXBlU3BlY2lmaWVyIHR5cGUpIHsKLSAgICAgICAgYm9vbGVhbiBjYW5FbmNvZGUgPSB0cnVlOwotCi0gICAgICAgIGludCBudW1CYW5kcyA9IHR5cGUuZ2V0U2FtcGxlTW9kZWwoKS5nZXROdW1CYW5kcygpOwotCi0gICAgICAgIENvbG9yTW9kZWwgY29sb3JNb2RlbCA9IHR5cGUuZ2V0Q29sb3JNb2RlbCgpOwotCi0gICAgICAgIGludCBiaXREZXB0aCA9IGNvbG9yTW9kZWwuZ2V0UGl4ZWxTaXplKCkgLyBudW1CYW5kczsKLQotICAgICAgICBpZiAoY29sb3JNb2RlbCBpbnN0YW5jZW9mIEluZGV4Q29sb3JNb2RlbCkgewotICAgICAgICAgICAgaWYgKGJpdERlcHRoICE9IDEgJiYgYml0RGVwdGggIT0gMiAmJiBiaXREZXB0aCAhPSA0ICYmIGJpdERlcHRoICE9IDgpIHsKLSAgICAgICAgICAgICAgICBjYW5FbmNvZGUgPSBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChudW1CYW5kcyAhPSAxKSB7Ci0gICAgICAgICAgICAgICAgY2FuRW5jb2RlID0gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgZWxzZSBpZiAobnVtQmFuZHMgPT0gMSkgewotICAgICAgICAgICAgaWYgKGJpdERlcHRoICE9IDEgJiYgYml0RGVwdGggIT0gMiAmJiBiaXREZXB0aCAhPSA0ICYmIGJpdERlcHRoICE9IDggJiYgYml0RGVwdGggIT0gMTYpIHsKLSAgICAgICAgICAgICAgICBjYW5FbmNvZGUgPSBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBlbHNlIGlmIChudW1CYW5kcyA9PSAyKSB7Ci0gICAgICAgICAgICBpZiAoYml0RGVwdGggIT0gOCAmJiBiaXREZXB0aCAhPSAxNikgewotICAgICAgICAgICAgICAgIGNhbkVuY29kZSA9IGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGVsc2UgaWYgKG51bUJhbmRzID09IDMpIHsKLSAgICAgICAgICAgIGlmIChiaXREZXB0aCAhPSA4ICYmIGJpdERlcHRoICE9IDE2KSB7Ci0gICAgICAgICAgICAgICAgY2FuRW5jb2RlID0gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgZWxzZSBpZiAobnVtQmFuZHMgPT0gNCkgewotICAgICAgICAgICAgaWYgKGJpdERlcHRoICE9IDggJiYgYml0RGVwdGggIT0gMTYpIHsKLSAgICAgICAgICAgICAgICBjYW5FbmNvZGUgPSBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIHJldHVybiBjYW5FbmNvZGU7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEltYWdlV3JpdGVyIGNyZWF0ZVdyaXRlckluc3RhbmNlKE9iamVjdCBhcmcwKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICByZXR1cm4gbmV3IFBOR0ltYWdlV3JpdGVyKHRoaXMpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RGVzY3JpcHRpb24oTG9jYWxlIGFyZzApIHsKLSAgICAgICAgcmV0dXJuICJEUkwgUE5HIGVuY29kZXIiOwotICAgIH0KLQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vc3BpL0ZpbGVJSVNTcGkuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3NwaS9GaWxlSUlTU3BpLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGQ0ZmRkNzYuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vc3BpL0ZpbGVJSVNTcGkuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDUzICswLDAgQEAKLS8qCi0gKiAgTGljZW5zZWQgdG8gdGhlIEFwYWNoZSBTb2Z0d2FyZSBGb3VuZGF0aW9uIChBU0YpIHVuZGVyIG9uZSBvciBtb3JlCi0gKiAgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0gKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgotICogIFRoZSBBU0YgbGljZW5zZXMgdGhpcyBmaWxlIHRvIFlvdSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wCi0gKiAgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiAgVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiAgU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0vKioKLSAqIEBhdXRob3IgUnVzdGVtIFYuIFJhZmlrb3YKLSAqIEB2ZXJzaW9uICRSZXZpc2lvbjogMS4yICQKLSAqLwotcGFja2FnZSBvcmcuYXBhY2hlLmhhcm1vbnkueC5pbWFnZWlvLnNwaTsKLQotaW1wb3J0IGphdmF4LmltYWdlaW8uc3BpLkltYWdlSW5wdXRTdHJlYW1TcGk7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uSW1hZ2VJbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLnN0cmVhbS5GaWxlSW1hZ2VPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uRmlsZUltYWdlSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS5pby5GaWxlOwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLQotcHVibGljIGNsYXNzIEZpbGVJSVNTcGkgZXh0ZW5kcyBJbWFnZUlucHV0U3RyZWFtU3BpIHsKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgdmVuZG9yID0gIkFwYWNoZSI7Ci0KLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgdmVyID0gIjAuMSI7Ci0KLSAgICBwdWJsaWMgRmlsZUlJU1NwaSgpIHsKLSAgICAgICAgc3VwZXIodmVuZG9yLCB2ZXIsIEZpbGUuY2xhc3MpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJbWFnZUlucHV0U3RyZWFtIGNyZWF0ZUlucHV0U3RyZWFtSW5zdGFuY2UoT2JqZWN0IGlucHV0LCBib29sZWFuIHVzZUNhY2hlLAotICAgICAgICAgICAgRmlsZSBjYWNoZURpcikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKEZpbGUuY2xhc3MuaXNJbnN0YW5jZShpbnB1dCkpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgRmlsZUltYWdlSW5wdXRTdHJlYW0oKEZpbGUpIGlucHV0KTsKLSAgICAgICAgfQotICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJpbnB1dCBpcyBub3QgYW4gaW5zdGFuY2Ugb2YgamF2YS5pby5GaWxlIik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyBnZXREZXNjcmlwdGlvbihMb2NhbGUgbG9jYWxlKSB7Ci0gICAgICAgIHJldHVybiAiRmlsZSBJSVMgU3BpIjsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zcGkvRmlsZUlPU1NwaS5qYXZhIGIvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vc3BpL0ZpbGVJT1NTcGkuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYWNkYTZhMS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zcGkvRmlsZUlPU1NwaS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTIgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjIgJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8uc3BpOwotCi1pbXBvcnQgamF2YXguaW1hZ2Vpby5zcGkuSW1hZ2VPdXRwdXRTdHJlYW1TcGk7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uSW1hZ2VPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uRmlsZUltYWdlT3V0cHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uRmlsZTsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEudXRpbC5Mb2NhbGU7Ci0KLXB1YmxpYyBjbGFzcyBGaWxlSU9TU3BpIGV4dGVuZHMgSW1hZ2VPdXRwdXRTdHJlYW1TcGkgewotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyB2ZW5kb3IgPSAiQXBhY2hlIjsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyB2ZXIgPSAiMC4xIjsKLQotICAgIHB1YmxpYyBGaWxlSU9TU3BpKCkgewotICAgICAgICBzdXBlcih2ZW5kb3IsIHZlciwgRmlsZS5jbGFzcyk7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIEltYWdlT3V0cHV0U3RyZWFtIGNyZWF0ZU91dHB1dFN0cmVhbUluc3RhbmNlKE9iamVjdCBvdXRwdXQsIGJvb2xlYW4gdXNlQ2FjaGUsCi0gICAgICAgICAgICBGaWxlIGNhY2hlRGlyKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAob3V0cHV0IGluc3RhbmNlb2YgRmlsZSkgewotICAgICAgICAgICAgcmV0dXJuIG5ldyBGaWxlSW1hZ2VPdXRwdXRTdHJlYW0oKEZpbGUpIG91dHB1dCk7Ci0gICAgICAgIH0KLSAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigib3V0cHV0IGlzIG5vdCBpbnN0YW5jZSBvZiBGaWxlIik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyBnZXREZXNjcmlwdGlvbihMb2NhbGUgbG9jYWxlKSB7Ci0gICAgICAgIHJldHVybiAiRmlsZSBJT1MgU3BpIjsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zcGkvSW5wdXRTdHJlYW1JSVNTcGkuamF2YSBiL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3NwaS9JbnB1dFN0cmVhbUlJU1NwaS5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlZDJmZWYwLi4wMDAwMDAwCi0tLSBhL2F3dC9vcmcvYXBhY2hlL2hhcm1vbnkveC9pbWFnZWlvL3NwaS9JbnB1dFN0cmVhbUlJU1NwaS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTkgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8uc3BpOwotCi1pbXBvcnQgamF2YXguaW1hZ2Vpby5zcGkuSW1hZ2VJbnB1dFN0cmVhbVNwaTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLnN0cmVhbS4qOwotaW1wb3J0IGphdmEuaW8uT3V0cHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uRmlsZTsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uSW5wdXRTdHJlYW07Ci1pbXBvcnQgamF2YS51dGlsLkxvY2FsZTsKLQotcHVibGljIGNsYXNzIElucHV0U3RyZWFtSUlTU3BpIGV4dGVuZHMgSW1hZ2VJbnB1dFN0cmVhbVNwaSB7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIHZlbmRvciA9ICJBcGFjaGUiOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIHZlciA9ICIwLjEiOwotCi0gICAgcHVibGljIElucHV0U3RyZWFtSUlTU3BpKCkgewotICAgICAgICBzdXBlcih2ZW5kb3IsIHZlciwgSW5wdXRTdHJlYW0uY2xhc3MpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RGVzY3JpcHRpb24oTG9jYWxlIGxvY2FsZSkgewotICAgICAgICByZXR1cm4gIk91dHB1dCBTdHJlYW0gSU9TIFNwaSI7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gY2FuVXNlQ2FjaGVGaWxlKCkgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSW1hZ2VJbnB1dFN0cmVhbSBjcmVhdGVJbnB1dFN0cmVhbUluc3RhbmNlKE9iamVjdCBpbnB1dCwgYm9vbGVhbiB1c2VDYWNoZSwgRmlsZSBjYWNoZURpcikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKGlucHV0IGluc3RhbmNlb2YgSW5wdXRTdHJlYW0pIHsKLSAgICAgICAgICAgIGlmICh1c2VDYWNoZSkgewotICAgICAgICAgICAgICAgIHJldHVybiBuZXcgRmlsZUNhY2hlSW1hZ2VJbnB1dFN0cmVhbSgoSW5wdXRTdHJlYW0pIGlucHV0LCBjYWNoZURpcik7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJldHVybiBuZXcgTWVtb3J5Q2FjaGVJbWFnZUlucHV0U3RyZWFtKChJbnB1dFN0cmVhbSkgaW5wdXQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHRocm93IG5ldyBJbGxlZ2FsQXJndW1lbnRFeGNlcHRpb24oIk91dHB1dCBpcyBub3QgYW4gaW5zdGFuY2Ugb2YgSW5wdXRTdHJlYW0iKTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zcGkvT3V0cHV0U3RyZWFtSU9TU3BpLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zcGkvT3V0cHV0U3RyZWFtSU9TU3BpLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGRkMWU4OGQuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vc3BpL091dHB1dFN0cmVhbUlPU1NwaS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNjAgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8uc3BpOwotCi1pbXBvcnQgamF2YXguaW1hZ2Vpby5zcGkuSW1hZ2VPdXRwdXRTdHJlYW1TcGk7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uSW1hZ2VPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uRmlsZUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uTWVtb3J5Q2FjaGVJbWFnZU91dHB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLkZpbGU7Ci1pbXBvcnQgamF2YS5pby5JT0V4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLk91dHB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLnV0aWwuTG9jYWxlOwotCi1wdWJsaWMgY2xhc3MgT3V0cHV0U3RyZWFtSU9TU3BpIGV4dGVuZHMgSW1hZ2VPdXRwdXRTdHJlYW1TcGkgewotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyB2ZW5kb3IgPSAiQXBhY2hlIjsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyB2ZXIgPSAiMC4xIjsKLQotICAgIHB1YmxpYyBPdXRwdXRTdHJlYW1JT1NTcGkoKSB7Ci0gICAgICAgIHN1cGVyKHZlbmRvciwgdmVyLCBPdXRwdXRTdHJlYW0uY2xhc3MpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJbWFnZU91dHB1dFN0cmVhbSBjcmVhdGVPdXRwdXRTdHJlYW1JbnN0YW5jZShPYmplY3Qgb3V0cHV0LCBib29sZWFuIHVzZUNhY2hlLCBGaWxlIGNhY2hlRGlyKSB0aHJvd3MgSU9FeGNlcHRpb24gewotICAgICAgICBpZiAob3V0cHV0IGluc3RhbmNlb2YgT3V0cHV0U3RyZWFtKSB7Ci0gICAgICAgICAgICBpZiAodXNlQ2FjaGUpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbmV3IEZpbGVDYWNoZUltYWdlT3V0cHV0U3RyZWFtKChPdXRwdXRTdHJlYW0pIG91dHB1dCwgY2FjaGVEaXIpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbmV3IE1lbW9yeUNhY2hlSW1hZ2VPdXRwdXRTdHJlYW0oKE91dHB1dFN0cmVhbSkgb3V0cHV0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJPdXRwdXQgaXMgbm90IGFuIGluc3RhbmNlIG9mIE91dHB1dFN0cmVhbSIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RGVzY3JpcHRpb24oTG9jYWxlIGxvY2FsZSkgewotICAgICAgICByZXR1cm4gIk91dHB1dCBTdHJlYW0gSU9TIFNwaSI7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIGJvb2xlYW4gY2FuVXNlQ2FjaGVGaWxlKCkgewotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zcGkvUkFGSUlTU3BpLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zcGkvUkFGSUlTU3BpLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGY5N2ViODcuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vc3BpL1JBRklJU1NwaS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTQgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjIgJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8uc3BpOwotCi1pbXBvcnQgamF2YXguaW1hZ2Vpby5zcGkuSW1hZ2VJbnB1dFN0cmVhbVNwaTsKLWltcG9ydCBqYXZheC5pbWFnZWlvLnN0cmVhbS5JbWFnZUlucHV0U3RyZWFtOwotaW1wb3J0IGphdmF4LmltYWdlaW8uc3RyZWFtLkZpbGVJbWFnZUlucHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uRmlsZTsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uUmFuZG9tQWNjZXNzRmlsZTsKLWltcG9ydCBqYXZhLnV0aWwuTG9jYWxlOwotCi1wdWJsaWMgY2xhc3MgUkFGSUlTU3BpIGV4dGVuZHMgSW1hZ2VJbnB1dFN0cmVhbVNwaSB7Ci0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIHZlbmRvciA9ICJBcGFjaGUiOwotCi0gICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIHZlciA9ICIwLjEiOwotCi0gICAgcHVibGljIFJBRklJU1NwaSgpIHsKLSAgICAgICAgc3VwZXIodmVuZG9yLCB2ZXIsIFJhbmRvbUFjY2Vzc0ZpbGUuY2xhc3MpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBJbWFnZUlucHV0U3RyZWFtIGNyZWF0ZUlucHV0U3RyZWFtSW5zdGFuY2UoT2JqZWN0IGlucHV0LCBib29sZWFuIHVzZUNhY2hlLAotICAgICAgICAgICAgRmlsZSBjYWNoZURpcikgdGhyb3dzIElPRXhjZXB0aW9uIHsKLSAgICAgICAgaWYgKFJhbmRvbUFjY2Vzc0ZpbGUuY2xhc3MuaXNJbnN0YW5jZShpbnB1dCkpIHsKLSAgICAgICAgICAgIHJldHVybiBuZXcgRmlsZUltYWdlSW5wdXRTdHJlYW0oKFJhbmRvbUFjY2Vzc0ZpbGUpIGlucHV0KTsKLSAgICAgICAgfQotICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKAotICAgICAgICAgICAgICAgICJpbnB1dCBpcyBub3QgYW4gaW5zdGFuY2Ugb2YgamF2YS5pby5SYW5kb21BY2Nlc3NGaWxlIik7Ci0gICAgfQotCi0gICAgQE92ZXJyaWRlCi0gICAgcHVibGljIFN0cmluZyBnZXREZXNjcmlwdGlvbihMb2NhbGUgbG9jYWxlKSB7Ci0gICAgICAgIHJldHVybiAiUmFuZG9tQWNjZXNzRmlsZSBJSVMgU3BpIjsKLSAgICB9Ci19CmRpZmYgLS1naXQgYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zcGkvUkFGSU9TU3BpLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zcGkvUkFGSU9TU3BpLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGE5ZDM2NDkuLjAwMDAwMDAKLS0tIGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vc3BpL1JBRklPU1NwaS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTMgKzAsMCBAQAotLyoKLSAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSAqICBjb250cmlidXRvciBsaWNlbnNlIGFncmVlbWVudHMuICBTZWUgdGhlIE5PVElDRSBmaWxlIGRpc3RyaWJ1dGVkIHdpdGgKLSAqICB0aGlzIHdvcmsgZm9yIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24gcmVnYXJkaW5nIGNvcHlyaWdodCBvd25lcnNoaXAuCi0gKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSAqICAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aAotICogIHRoZSBMaWNlbnNlLiAgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiAgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiAgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLS8qKgotICogQGF1dGhvciBSdXN0ZW0gVi4gUmFmaWtvdgotICogQHZlcnNpb24gJFJldmlzaW9uOiAxLjIgJAotICovCi1wYWNrYWdlIG9yZy5hcGFjaGUuaGFybW9ueS54LmltYWdlaW8uc3BpOwotCi1pbXBvcnQgamF2YXguaW1hZ2Vpby5zcGkuSW1hZ2VPdXRwdXRTdHJlYW1TcGk7Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uSW1hZ2VPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YXguaW1hZ2Vpby5zdHJlYW0uRmlsZUltYWdlT3V0cHV0U3RyZWFtOwotaW1wb3J0IGphdmEuaW8uRmlsZTsKLWltcG9ydCBqYXZhLmlvLklPRXhjZXB0aW9uOwotaW1wb3J0IGphdmEuaW8uUmFuZG9tQWNjZXNzRmlsZTsKLWltcG9ydCBqYXZhLnV0aWwuTG9jYWxlOwotCi1wdWJsaWMgY2xhc3MgUkFGSU9TU3BpIGV4dGVuZHMgSW1hZ2VPdXRwdXRTdHJlYW1TcGkgewotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyB2ZW5kb3IgPSAiQXBhY2hlIjsKLQotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyB2ZXIgPSAiMC4xIjsKLQotICAgIHB1YmxpYyBSQUZJT1NTcGkoKSB7Ci0gICAgICAgIHN1cGVyKHZlbmRvciwgdmVyLCBSYW5kb21BY2Nlc3NGaWxlLmNsYXNzKTsKLSAgICB9Ci0KLSAgICBAT3ZlcnJpZGUKLSAgICBwdWJsaWMgSW1hZ2VPdXRwdXRTdHJlYW0gY3JlYXRlT3V0cHV0U3RyZWFtSW5zdGFuY2UoT2JqZWN0IG91dHB1dCwgYm9vbGVhbiB1c2VDYWNoZSwKLSAgICAgICAgICAgIEZpbGUgY2FjaGVEaXIpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChvdXRwdXQgaW5zdGFuY2VvZiBSYW5kb21BY2Nlc3NGaWxlKSB7Ci0gICAgICAgICAgICByZXR1cm4gbmV3IEZpbGVJbWFnZU91dHB1dFN0cmVhbSgoUmFuZG9tQWNjZXNzRmlsZSkgb3V0cHV0KTsKLSAgICAgICAgfQotICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJvdXRwdXQgaXMgbm90IGluc3RhbmNlIG9mIGphdmEuaW8uUmFuZG9tQWNjZXNzRmlsZSIpOwotICAgIH0KLQotICAgIEBPdmVycmlkZQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RGVzY3JpcHRpb24oTG9jYWxlIGxvY2FsZSkgewotICAgICAgICByZXR1cm4gIlJhbmRvbUFjY2Vzc0ZpbGUgSU9TIFNwaSI7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvYXd0L29yZy9hcGFjaGUvaGFybW9ueS94L2ltYWdlaW8vc3RyZWFtL1JhbmRvbUFjY2Vzc01lbW9yeUNhY2hlLmphdmEgYi9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zdHJlYW0vUmFuZG9tQWNjZXNzTWVtb3J5Q2FjaGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjRmN2IyYS4uMDAwMDAwMAotLS0gYS9hd3Qvb3JnL2FwYWNoZS9oYXJtb255L3gvaW1hZ2Vpby9zdHJlYW0vUmFuZG9tQWNjZXNzTWVtb3J5Q2FjaGUuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDIyNiArMCwwIEBACi0vKgotICogIExpY2Vuc2VkIHRvIHRoZSBBcGFjaGUgU29mdHdhcmUgRm91bmRhdGlvbiAoQVNGKSB1bmRlciBvbmUgb3IgbW9yZQotICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAotICogIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSAqICBUaGUgQVNGIGxpY2Vuc2VzIHRoaXMgZmlsZSB0byBZb3UgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMAotICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCi0gKiAgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0KLXBhY2thZ2Ugb3JnLmFwYWNoZS5oYXJtb255LnguaW1hZ2Vpby5zdHJlYW07Ci0KLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlMaXN0OwotaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247Ci1pbXBvcnQgamF2YS5pby5JbnB1dFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLk91dHB1dFN0cmVhbTsKLQotcHVibGljIGZpbmFsIGNsYXNzIFJhbmRvbUFjY2Vzc01lbW9yeUNhY2hlIHsKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgQkxPQ0tfU0hJRlQgPSA5OwotICAgIHByaXZhdGUgc3RhdGljIGZpbmFsIGludCBCTE9DS19TSVpFID0gMSA8PCBCTE9DS19TSElGVDsKLSAgICBwcml2YXRlIHN0YXRpYyBmaW5hbCBpbnQgQkxPQ0tfTUFTSyA9IEJMT0NLX1NJWkUgLSAxOwotICAgIAotICAgIHByaXZhdGUgbG9uZyBsZW5ndGg7Ci0KLSAgICBwcml2YXRlIGludCBmaXJzdFVuZGlzcG9zZWQgPSAwOwotCi0gICAgcHJpdmF0ZSBBcnJheUxpc3Q8Ynl0ZVtdPiBibG9ja3MgPSBuZXcgQXJyYXlMaXN0PGJ5dGVbXT4oKTsKLQotICAgIHB1YmxpYyBSYW5kb21BY2Nlc3NNZW1vcnlDYWNoZSgpIHsKLSAgICB9Ci0KLSAgICBwdWJsaWMgbG9uZyBsZW5ndGgoKSB7Ci0gICAgICAgIHJldHVybiBsZW5ndGg7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgY2xvc2UoKSB7Ci0gICAgICAgIGJsb2Nrcy5jbGVhcigpOwotICAgICAgICBsZW5ndGggPSAwOwotICAgIH0KLQotICAgIHByaXZhdGUgdm9pZCBncm93KGxvbmcgcG9zKSB7Ci0gICAgICAgIGludCBibG9ja3NOZWVkZWQgPSAoaW50KShwb3MgPj4gQkxPQ0tfU0hJRlQpIC0gYmxvY2tzLnNpemUoKSArIDE7Ci0gICAgICAgIGZvciAoaW50IGk9MDsgaSA8IGJsb2Nrc05lZWRlZDsgaSsrKSB7Ci0gICAgICAgICAgICBibG9ja3MuYWRkKG5ldyBieXRlW0JMT0NLX1NJWkVdKTsKLSAgICAgICAgfQotCi0gICAgICAgIGxlbmd0aCA9IHBvcyArIDE7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgcHV0RGF0YShpbnQgb25lQnl0ZSwgbG9uZyBwb3MpIHsKLSAgICAgICAgaWYgKHBvcyA+PSBsZW5ndGgpIHsKLSAgICAgICAgICAgIGdyb3cocG9zKTsKLSAgICAgICAgfQotCi0gICAgICAgIGJ5dGVbXSBibG9jayA9IGJsb2Nrcy5nZXQoKGludCkocG9zID4+IEJMT0NLX1NISUZUKSk7Ci0gICAgICAgIGJsb2NrWyhpbnQpKHBvcyAmIEJMT0NLX01BU0spXSA9IChieXRlKSBvbmVCeXRlOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHB1dERhdGEoYnl0ZVtdIGJ1ZmZlciwgaW50IG9mZnNldCwgaW50IGNvdW50LCBsb25nIHBvcykgewotICAgICAgICBpZiAoY291bnQgPiBidWZmZXIubGVuZ3RoIC0gb2Zmc2V0IHx8IGNvdW50IDwgMCB8fCBvZmZzZXQgPCAwKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbigpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChjb3VudCA9PSAwKXsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGxvbmcgbGFzdFBvcyA9IHBvcyArIGNvdW50IC0gMTsKLSAgICAgICAgaWYgKGxhc3RQb3MgPj0gbGVuZ3RoKSB7Ci0gICAgICAgICAgICBncm93KGxhc3RQb3MpOwotICAgICAgICB9Ci0KLSAgICAgICAgd2hpbGUgKGNvdW50ID4gMCkgewotICAgICAgICAgICAgYnl0ZVtdIGJsb2NrID0gYmxvY2tzLmdldCgoaW50KShwb3MgPj4gQkxPQ0tfU0hJRlQpKTsKLSAgICAgICAgICAgIGludCBibG9ja09mZnNldCA9IChpbnQpKHBvcyAmIEJMT0NLX01BU0spOwotICAgICAgICAgICAgaW50IHRvQ29weSA9IE1hdGgubWluKEJMT0NLX1NJWkUgLSBibG9ja09mZnNldCwgY291bnQpOwotICAgICAgICAgICAgU3lzdGVtLmFycmF5Y29weShidWZmZXIsIG9mZnNldCwgYmxvY2ssIGJsb2NrT2Zmc2V0LCB0b0NvcHkpOwotICAgICAgICAgICAgcG9zICs9IHRvQ29weTsKLSAgICAgICAgICAgIGNvdW50IC09IHRvQ29weTsKLSAgICAgICAgICAgIG9mZnNldCArPSB0b0NvcHk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgaW50IGdldERhdGEobG9uZyBwb3MpIHsKLSAgICAgICAgaWYgKHBvcyA+PSBsZW5ndGgpIHsKLSAgICAgICAgICAgIHJldHVybiAtMTsKLSAgICAgICAgfQotCi0gICAgICAgIGJ5dGVbXSBibG9jayA9IGJsb2Nrcy5nZXQoKGludCkocG9zID4+IEJMT0NLX1NISUZUKSk7Ci0gICAgICAgIHJldHVybiBibG9ja1soaW50KShwb3MgJiBCTE9DS19NQVNLKV0gJiAweEZGOwotICAgIH0KLQotICAgIHB1YmxpYyBpbnQgZ2V0RGF0YShieXRlW10gYnVmZmVyLCBpbnQgb2Zmc2V0LCBpbnQgY291bnQsIGxvbmcgcG9zKSB7Ci0gICAgICAgIGlmIChjb3VudCA+IGJ1ZmZlci5sZW5ndGggLSBvZmZzZXQgfHwgY291bnQgPCAwIHx8IG9mZnNldCA8IDApIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKCk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGNvdW50ID09IDApIHsKLSAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICB9Ci0gICAgICAgIGlmIChwb3MgPj0gbGVuZ3RoKSB7Ci0gICAgICAgICAgICByZXR1cm4gLTE7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoY291bnQgKyBwb3MgPiBsZW5ndGgpIHsKLSAgICAgICAgICAgIGNvdW50ID0gKGludCkgKGxlbmd0aCAtIHBvcyk7Ci0gICAgICAgIH0KLQotICAgICAgICBieXRlW10gYmxvY2sgPSBibG9ja3MuZ2V0KChpbnQpKHBvcyA+PiBCTE9DS19TSElGVCkpOwotICAgICAgICBpbnQgbmJ5dGVzID0gTWF0aC5taW4oY291bnQsIEJMT0NLX1NJWkUgLSAoaW50KShwb3MgJiBCTE9DS19NQVNLKSk7Ci0gICAgICAgIFN5c3RlbS5hcnJheWNvcHkoYmxvY2ssIChpbnQpKHBvcyAmIEJMT0NLX01BU0spLCBidWZmZXIsIG9mZnNldCwgbmJ5dGVzKTsKLQotICAgICAgICByZXR1cm4gbmJ5dGVzOwotICAgIH0KLSAgICAvKgotICAgIHB1YmxpYyB2b2lkIHNlZWsobG9uZyBwb3MpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChwb3MgPCAwKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSU9FeGNlcHRpb24oInNlZWsgcG9zaXRpb24gaXMgbmVnYXRpdmUiKTsKLSAgICAgICAgfQotICAgICAgICB0aGlzLnBvcyA9IHBvczsgCi0gICAgfQotCi0gICAgcHVibGljIHZvaWQgcmVhZEZ1bGx5KGJ5dGVbXSBidWZmZXIpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIHJlYWRGdWxseShidWZmZXIsIDAsIGJ1ZmZlci5sZW5ndGgpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIHJlYWRGdWxseShieXRlW10gYnVmZmVyLCBpbnQgb2Zmc2V0LCBpbnQgY291bnQpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGlmICgwIDw9IG9mZnNldCAmJiBvZmZzZXQgPD0gYnVmZmVyLmxlbmd0aCAmJiAwIDw9IGNvdW50ICYmIGNvdW50IDw9IGJ1ZmZlci5sZW5ndGggLSBvZmZzZXQpIHsKLSAgICAgICAgICAgIHdoaWxlIChjb3VudCA+IDApIHsKLSAgICAgICAgICAgICAgICBpbnQgcmVzdWx0ID0gcmVhZChidWZmZXIsIG9mZnNldCwgY291bnQpOwotICAgICAgICAgICAgICAgIGlmIChyZXN1bHQgPj0gMCkgewotICAgICAgICAgICAgICAgICAgICBvZmZzZXQgKz0gcmVzdWx0OwotICAgICAgICAgICAgICAgICAgICBjb3VudCAtPSByZXN1bHQ7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVPRkV4Y2VwdGlvbigpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHRocm93IG5ldyBJbmRleE91dE9mQm91bmRzRXhjZXB0aW9uKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgbG9uZyBnZXRGaWxlUG9pbnRlcigpIHsKLSAgICAgICAgcmV0dXJuIHBvczsKLSAgICB9Ci0qLwotCi0gICAgcHVibGljIHZvaWQgZnJlZUJlZm9yZShsb25nIHBvcykgewotICAgICAgICBpbnQgYmxvY2tJZHggPSAoaW50KShwb3MgPj4gQkxPQ0tfU0hJRlQpOwotICAgICAgICBpZiAoYmxvY2tJZHggPD0gZmlyc3RVbmRpc3Bvc2VkKSB7IC8vIE5vdGhpbmcgdG8gZG8KLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IGkgPSBmaXJzdFVuZGlzcG9zZWQ7IGkgPCBibG9ja0lkeDsgaSsrKSB7Ci0gICAgICAgICAgICBibG9ja3Muc2V0KGksIG51bGwpOwotICAgICAgICB9Ci0KLSAgICAgICAgZmlyc3RVbmRpc3Bvc2VkID0gYmxvY2tJZHg7Ci0gICAgfQotCi0gICAgcHVibGljIGludCBhcHBlbmREYXRhKElucHV0U3RyZWFtIGlzLCBpbnQgY291bnQpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChjb3VudCA8PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gMDsKLSAgICAgICAgfQotCi0gICAgICAgIGxvbmcgc3RhcnRQb3MgPSBsZW5ndGg7Ci0gICAgICAgIGxvbmcgbGFzdFBvcyA9IGxlbmd0aCArIGNvdW50IC0gMTsKLSAgICAgICAgZ3JvdyhsYXN0UG9zKTsgLy8gQ2hhbmdlcyBsZW5ndGgKLQotICAgICAgICBpbnQgYmxvY2tJZHggPSAoaW50KShzdGFydFBvcyA+PiBCTE9DS19TSElGVCk7Ci0gICAgICAgIGludCBvZmZzZXQgPSAoaW50KSAoc3RhcnRQb3MgJiBCTE9DS19NQVNLKTsKLQotICAgICAgICBpbnQgYnl0ZXNBcHBlbmRlZCA9IDA7Ci0KLSAgICAgICAgd2hpbGUgKGNvdW50ID4gMCkgewotICAgICAgICAgICAgYnl0ZVtdIGJsb2NrID0gYmxvY2tzLmdldChibG9ja0lkeCk7Ci0gICAgICAgICAgICBpbnQgdG9Db3B5ID0gTWF0aC5taW4oQkxPQ0tfU0laRSAtIG9mZnNldCwgY291bnQpOwotICAgICAgICAgICAgY291bnQgLT0gdG9Db3B5OwotCi0gICAgICAgICAgICB3aGlsZSAodG9Db3B5ID4gMCkgewotICAgICAgICAgICAgICAgIGludCBieXRlc1JlYWQgPSBpcy5yZWFkKGJsb2NrLCBvZmZzZXQsIHRvQ29weSk7Ci0KLSAgICAgICAgICAgICAgICBpZiAoYnl0ZXNSZWFkIDwgMCkgewotICAgICAgICAgICAgICAgICAgICBsZW5ndGggLT0gKGNvdW50IC0gYnl0ZXNBcHBlbmRlZCk7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBieXRlc0FwcGVuZGVkOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIHRvQ29weSAtPSBieXRlc1JlYWQ7Ci0gICAgICAgICAgICAgICAgb2Zmc2V0ICs9IGJ5dGVzUmVhZDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgYmxvY2tJZHgrKzsKLSAgICAgICAgICAgIG9mZnNldCA9IDA7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gY291bnQ7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZ2V0RGF0YShPdXRwdXRTdHJlYW0gb3MsIGludCBjb3VudCwgbG9uZyBwb3MpIHRocm93cyBJT0V4Y2VwdGlvbiB7Ci0gICAgICAgIGlmIChwb3MgKyBjb3VudCA+IGxlbmd0aCkgewotICAgICAgICAgICAgdGhyb3cgbmV3IEluZGV4T3V0T2ZCb3VuZHNFeGNlcHRpb24oIkFyZ3VtZW50IG91dCBvZiBjYWNoZSIpOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGJsb2NrSWR4ID0gKGludCkocG9zID4+IEJMT0NLX1NISUZUKTsKLSAgICAgICAgaW50IG9mZnNldCA9IChpbnQpIChwb3MgJiBCTE9DS19NQVNLKTsKLSAgICAgICAgaWYgKGJsb2NrSWR4IDwgZmlyc3RVbmRpc3Bvc2VkKSB7Ci0gICAgICAgICAgICB0aHJvdyBuZXcgSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbigiVGhlIHJlcXVlc3RlZCBkYXRhIGFyZSBhbHJlYWR5IGRpc3Bvc2VkIik7Ci0gICAgICAgIH0KLQotICAgICAgICB3aGlsZSAoY291bnQgPiAwKSB7Ci0gICAgICAgICAgICBieXRlW10gYmxvY2sgPSBibG9ja3MuZ2V0KGJsb2NrSWR4KTsKLSAgICAgICAgICAgIGludCB0b1dyaXRlID0gTWF0aC5taW4oQkxPQ0tfU0laRSAtIG9mZnNldCwgY291bnQpOwotICAgICAgICAgICAgb3Mud3JpdGUoYmxvY2ssIG9mZnNldCwgdG9Xcml0ZSk7Ci0KLSAgICAgICAgICAgIGJsb2NrSWR4Kys7Ci0gICAgICAgICAgICBvZmZzZXQgPSAwOwotICAgICAgICAgICAgY291bnQgLT0gdG9Xcml0ZTsKLSAgICAgICAgfQotICAgIH0KLX0KZGlmZiAtLWdpdCBhL2F3dC9yZXNvdXJjZXMvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9pbnRlcm5hbC9ubHMvbWVzc2FnZXMucHJvcGVydGllcyBiL2F3dC9yZXNvdXJjZXMvb3JnL2FwYWNoZS9oYXJtb255L2F3dC9pbnRlcm5hbC9ubHMvbWVzc2FnZXMucHJvcGVydGllcwpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOWY2NDdlOS4uMDAwMDAwMAotLS0gYS9hd3QvcmVzb3VyY2VzL29yZy9hcGFjaGUvaGFybW9ueS9hd3QvaW50ZXJuYWwvbmxzL21lc3NhZ2VzLnByb3BlcnRpZXMKKysrIC9kZXYvbnVsbApAQCAtMSw0OTUgKzAsMCBAQAotIyBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSMgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0jIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSMgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSMgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSMgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSMgIAotIyAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotIyAgCi0jICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0jICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotIyAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0jICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0jICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSMgCi0KLSMgbWVzc2FnZXMgZm9yIEVOIGxvY2FsZQotYXd0LjAwPUZvbnRSZW5kZXJDb250ZXh0IGlzIG51bGwKLWF3dC4wMT0nezB9JyBwYXJhbWV0ZXIgaXMgbnVsbAotYXd0LjAyPSd7MH0nIHBhcmFtZXRlciBoYXMgemVybyBsZW5ndGgKLWF3dC4wMz0nezB9JyBpdGVyYXRvciBwYXJhbWV0ZXIgaXMgbnVsbAotYXd0LjA0PSd7MH0nIGl0ZXJhdG9yIHBhcmFtZXRlciBoYXMgemVybyBsZW5ndGgKLWF3dC4wNT1PcGVyYXRpb24gY2Fubm90IGJlIG51bGwKLWF3dC4wNj1VbmV4cGVjdGVkIHR5cGUgb2YgdGhlIGludGVybmFsIGRhdGEgYnVmZmVyCi1hd3QuMDc9VHJhbnNmZXIgZGF0YSBpcyBub3QgYXZhaWxhYmxlCi1hd3QuMDg9eGZsZCBwYXJzZSBzdHJpbmcgZXJyb3I6IHswfQotYXd0LjA5PW1pbiByYW5nZSBib3VuZCB2YWx1ZSBpcyBncmVhdGVyIHRoYW4gbWF4IHJhbmdlIGJvdW5kCi1hd3QuMEE9Q2Fubm90IHVzZSBTaW5nbGVQaXhlZFBhY2tlZFNhbXBsZU1vZGVsIGZvciBicHAgPSB7MH0KLWF3dC4wQj1Xcm9uZyBjb2xvciBtb2RlbCBjcmVhdGVkIGZvciBkcmF3YWJsZQotYXd0LjBDPVVua25vd24gdmlzdWFsIGNsYXNzCi1hd3QuMEQ9SW52YWxpZCB0cmFuc3BhcmVuY3kKLWF3dC4wRT1EaW1lbnNpb25zIG9mIHRoZSBpbWFnZSBzaG91bGQgYmUgcG9zaXRpdmUKLWF3dC4wRj1DYW5ub3Qgb3BlbiBkaXNwbGF5ICd7MH0nCi1hd3QuMTA9T25seSAzMi1iaXQgZm9ybWF0IGlzIHN1cHBvcnRlZCBmb3Igd2luZG93IHN0YXRlIG9wZXJhdGlvbnMuCi1hd3QuMTE9SW52YWxpZCBrZXkgY29kZQotYXd0LjEyPVhUZXN0IGlzIG5vdCBzdXBwb3J0ZWQgYnkgeW91ciBYIHNlcnZlclwhCi1hd3QuMTM9Q2Fubm90IGFsbG9jYXRlIGNvbG9yIG5hbWVkICd7MH0nCi1hd3QuMTQ9VHJhbnNmZXIgZGF0YSBpcyBub3QgYXZhaWxhYmxlCi1hd3QuMTU9Q2FuIG5vdCBnZXQgbW9uaXRvciBpbmZvCi1hd3QuMTY9Q2FuIG5vdCBjcmVhdGUgREMgZm9yIGRldmljZQotYXd0LjE3PVVua25vd24gQ29tcG9zaXRlIHR5cGUgOiB7MH0KLWF3dC4xOD1UcmFuc3BhcmVuY3kgaXMgbm90IHN1cHBvcnRlZAotYXd0LjE5PUlsbGVnYWwgc2l6ZSBvZiB2b2xhdGlsZSBpbWFnZQotYXd0LjFBPUZhaWxlZCB0byByZWdpc3RlciB3aW5kb3cgY2xhc3MgezB9IEdldExhc3RFcnJvciByZXR1cm5lZCB7MX0KLWF3dC4xQj1JbnZhbGlkIGtleSBjb2RlCi1hd3QuMUM9RmFpbHVyZSB0byBjcmVhdGUgSmF2YVdpbmRvdyBHZXRMYXN0RXJyb3IgcmV0dXJuZWQgezB9Ci1hd3QuMUQ9Q2Fubm90IGdldCBkYXRhIGZyb20gT0xFIGNsaXBib2FyZAotYXd0LjFFPUF0dGVtcHQgdG8gcmVwbGFjZSBXaW5kb3dQcm9jIGhhbmRsZXIKLWF3dC4xRj1XYWl0aW5nIGZvciByZXNvdXJjZSBhY2Nlc3MgdGhyZWFkIGludGVycnVwdGVkIG5vdCBmcm9tIHVubG9jayBtZXRob2QKLWF3dC4yMD1DYW4ndCB1bmxvY2sgbm90IGxvY2tlZCByZXNvdXJjZQotYXd0LjIxPU5vdCBvd25lciBjYW4ndCB1bmxvY2sgcmVzb3VyY2UKLWF3dC4yMj1Ob3Qgb3duZXIgY2FuJ3QgZnJlZSByZXNvdXJjZQotYXd0LjIzPU9uZSB0aHJlYWQgY2FuJ3Qgc3RvcmUgc3RhdGUgc2V2ZXJhbCB0aW1lcyBpbiBhIHJvdwotYXd0LjI0PU93bmVyIGNhbid0IG92ZXJ3cml0ZSByZXNvdXJjZSBzdGF0ZS4gTG9jayBvcGVyYXRpb25zIG1heSBiZSBsb3N0Ci1hd3QuMjU9Tm8gc3RhdGUgc3RvcmVkIGZvciBjdXJyZW50IHRocmVhZAotYXd0LjI2PVNodXRkb3duIHRocmVhZCB3YXMgaW50ZXJydXB0ZWQgd2hpbGUgc3RhcnRpbmcKLWF3dC4yNz1TaHV0ZG93biB0aHJlYWQgd2FzIGludGVycnVwdGVkIHdoaWxlIHN0b3BwaW5nCi1hd3QuMjg9YmFkIGluZGV4OiB7MH0KLWF3dC4yOT1JbnZhbGlkIHJhbmdlCi1hd3QuMkE9UG9zaXRpb24gbm90IHJlcHJlc2VudGVkIGJ5IHZpZXcKLWF3dC4yQj1ObyB3b3JkIGF0IHswfQotYXd0LjJDPUludmFsaWQgcG9zaXRpb246IHswfQotYXd0LjJEPUludmFsaWQgZGlyZWN0aW9uCi1hd3QuMkU9ezB9IG5vdCBpbiByYW5nZSB7MX0sezJ9Ci1hd3QuMkY9Tm8gbW9yZSB3b3JkcwotYXd0LjMwPXdyb25nIG51bWJlciBvZiBlbGVtZW50cyB0byBjb3B5OiB7MH0sIHNpemU6IHsxfQotYXd0LjMxPW5vIHJvb20gdG8gY29weTogezB9LCBzaXplOiB7MX0KLWF3dC4zMj1TdHJpbmc6ICd7MH0nIGRvZXMgbm90IGZpdAotYXd0LjMzPWluZGV4IGlzIG91dCBvZiByYW5nZQotYXd0LjM0PUluaXRpYWwgb2Zmc2V0IGluIHRoZSBkZXN0aW5hdGlvbiBhcnJheSBpcyB3cm9uZzogezB9Ci1hd3QuMzU9V3JvbmcgbnVtYmVyIG9mIGVsZW1lbnRzIHRvIGNvcHk6IHswfQotYXd0LjM2PVdyb25nIHNlZ21lbnQKLWF3dC4zNz1Vbmtub3duICBjb21wb3NpdGUgdHlwZSB7MH0KLWF3dC4zOD1Qcm9wZXJ0eSBuYW1lIGlzIG5vdCBkZWZpbmVkCi1hd3QuMzk9VGhpcyBtZXRob2QgaXMgbm90IGltcGxlbWVudGVkIGZvciBpbWFnZSBvYnRhaW5lZCBmcm9tIEltYWdlUHJvZHVjZXIKLWF3dC4zQT1Db2xvciBNb2RlbCBpcyBudWxsCi1hd3QuM0I9SW5jb3JyZWN0IEltYWdlQ29uc3VtZXIgY29tcGxldGlvbiBzdGF0dXMKLWF3dC4zQz1Vbmtub3duIFBORyBjb2xvciB0eXBlCi1hd3QuM0Q9VW5rbm93biBjb2xvcnNwYWNlCi1hd3QuM0U9Q2xvbmUgbm90IHN1cHBvcnRlZAotYXd0LjNGPUludmFsaWQgYmFzZWxpbmUgaW5kZXgKLWF3dC40MD1Xcm9uZyBudW1iZXIgb2YgbWV0cmljc1whCi1hd3QuNDE9Rm9udCByZXR1cm5lZCB1bnN1cHBvcnRlZCB0eXBlIG9mIGxpbmUgbWV0cmljcy4gVGhpcyBjYXNlIGlzIGtub3duLCBidXQgbm90IHN1cHBvcnRlZCB5ZXQuCi1hd3QuNDI9VGV4dEhpdEluZm8gb3V0IG9mIHJhbmdlCi1hd3QuNDM9Z2x5cGhJbmRleCBpcyBvdXQgb2YgdmVjdG9yJ3MgbGltaXRzCi1hd3QuNDQ9YmVnaW5HbHlwaEluZGV4IGlzIG91dCBvZiB2ZWN0b3IncyByYW5nZQotYXd0LjQ1PW51bUVudHJpZXMgaXMgb3V0IG9mIHZlY3RvcidzIHJhbmdlCi1hd3QuNDY9bGVuZ3RoIG9mIHNldFBvc2l0aW9ucyBhcnJheSBkaWZmZXJzIGZyb20gdGhlIGxlbmd0aCBvZiBwb3NpdGlvbnMgYXJyYXkKLWF3dC40Nz1GaXJzdCBhcmd1bWVudCBzaG91bGQgYmUgYnl0ZSBvciBzaG9ydCBhcnJheQotYXd0LjQ4PVRoZSBzcmNJbiByYXN0ZXIgaXMgaW5jb21wYXRpYmxlIHdpdGggc3JjIENvbG9yTW9kZWwKLWF3dC40OT1UaGUgZHN0SW4gcmFzdGVyIGlzIGluY29tcGF0aWJsZSB3aXRoIGRzdCBDb2xvck1vZGVsCi1hd3QuNEE9VGhlIGRzdE91dCByYXN0ZXIgaXMgaW5jb21wYXRpYmxlIHdpdGggZHN0IENvbG9yTW9kZWwKLWF3dC40Qj1JdGVyYXRvciBvdXQgb2YgYm91bmRzCi1hd3QuNEM9SW52YWxpZCBNdWx0aVJlY3RBcmVhIGluIG1ldGhvZCB7MH0KLWF3dC40RD1UaGUgcmFzdGVyIGlzIGluY29tcGF0aWJsZSB3aXRoIHRoaXMgQ29sb3JNb2RlbAotYXd0LjRFPVVua25vd24gbmF0aXZlIHBsYXRmb3JtLgotYXd0LjRGPURhdGEgaXMgbm90IGF2YWlsYWJsZQotYXd0LjUwPUl0ZXJhdG9yIGlzIHJlYWQtb25seQotYXd0LjUxPUNvbXBvbmVudCBleHBlY3RlZCB0byBiZSBhIHBhcmVudAotYXd0LjUyPVRpbWUgaW50ZXJ2YWwgY2FuJ3QgYmUgPD0gMAotYXd0LjUzPUhhbmRsZXIgY2FuJ3QgYmUgbnVsbAotYXd0LjU0PUtleSBldmVudCBmb3IgdW5mb2N1c2VkIGNvbXBvbmVudAotYXd0LjU1PURvdWJsZSBtb3VzZSBlbnRlciBldmVudCBmb3IgY29tcG9uZW50Ci1hd3QuNTY9RG91YmxlIG1vdXNlIGV4aXQgZXZlbnQgZm9yIGNvbXBvbmVudAotYXd0LjU3PURvdWJsZSBmb2N1cyBnYWluZWQgZXZlbnQgZm9yIGNvbXBvbmVudAotYXd0LjU4PURvdWJsZSBmb2N1cyBsb3N0IGV2ZW50IGZvciBjb21wb25lbnQKLWF3dC41OT1BcHBsaWNhdGlvbiBoYXMgcnVuIG91dCBvZiBjb250ZXh0IHRocmVhZCBncm91cAotYXd0LjVBPURlZmF1bHQgY2xhc3MgZm9yIFByaW50ZXJKb2IgaXMgbm90IGZvdW5kCi1hd3QuNUI9Tm8gYWNjZXNzIHRvIGRlZmF1bHQgY2xhc3MgZm9yIFByaW50ZXJKb2IKLWF3dC41Qz1JbnN0YW50aWF0aW9uIGV4Y2VwdGlvbiBmb3IgUHJpbnRlckpvYgotYXd0LjVEPXswfSBpcyBub3Qgc3VwcG9ydGVkCi1hd3QuNUU9cGFnZUluZGV4IGlzIG1vcmUgdGhhbiBib29rIHNpemUKLWF3dC41Rj13cm9uZyBvcmllbnRhdGlvbgotYXd0LjYwPVdpZHRoIGFuZCBIZWlnaHQgbXVzdG4ndCBiZSBlcXVhbCB6ZXJvIGJvdGgKLWF3dC42MT1VbnN1cHBvcnRlZCBkYXRhIHR5cGU6IHswfQotYXd0LjYyPVdyb25nIG1hc2sgOiB7MH0KLWF3dC42Mz1Db29yZGluYXRlcyBhcmUgbm90IGluIGJvdW5kcwotYXd0LjY0PVRoZSBudW1iZXIgb2YgdGhlIGJhbmRzIGluIHRoZSBzdWJzZXQgaXMgZ3JlYXRlciB0aGFuIHRoZSBudW1iZXIgb2YgYmFuZHMgaW4gdGhlIHNhbXBsZSBtb2RlbAotYXd0LjY1PW51bGwgYXJndW1lbnQKLWF3dC42Nj1JbnZhbGlkIGZvcm1hdAotYXd0LjY3PXN1YmNsYXNzIGlzIG5vdCBkZXJpdmVkIGZyb20gQVdUS2V5U3Ryb2tlCi1hd3QuNjg9c3ViY2xhc3MgY291bGQgbm90IGJlIGluc3RhbnRpYXRlZAotYXd0LjY5PWNvbHVtbnMgbGVzcyB0aGFuIHplcm8uCi1hd3QuNkE9cm93cyBsZXNzIHRoYW4gemVyby4KLWF3dC42Qj1RdWV1ZSBzdGFjayBpcyBlbXB0eQotYXd0LjZDPUV2ZW50IHF1ZXVlIHN0YWNrIGlzIGJyb2tlbgotYXd0LjZEPVBvaW50IGlzIG51bGwKLWF3dC42RT1Db2xvciBpcyBudWxsCi1hd3QuNkY9SW5kZXggbGVzcyB0aGFuIHplcm8KLWF3dC43MD1NZW51SXRlbSBpcyBudWxsCi1hd3QuNzE9UGFyZW50IGlzIG51bGwKLWF3dC43Mj1LZXkgZXZlbnQgZm9yIHVuZm9jdXNlZCBjb21wb25lbnQKLWF3dC43Mz1ubyBzdWNoIGl0ZW0KLWF3dC43ND1JbnB1dCBwYXJhbWV0ZXJzIGEgYW5kIGIgc2hvdWxkIG5vdCBiZSBudWxsCi1hd3QuNzU9cm93cyBhbmQgY29scyBjYW5ub3QgYm90aCBiZSB6ZXJvCi1hd3QuNzY9cm93cyBhbmQgY29scyBjYW5ub3QgYmUgbmVnYXRpdmUKLWF3dC43Nz1kZWZhdWx0IGZvY3VzIHRyYXZlcnNhbCBwb2xpY3kgY2Fubm90IGJlIG51bGwKLWF3dC43OD1pbnZhbGlkIGZvY3VzIHRyYXZlcnNhbCBrZXkgaWRlbnRpZmllcgotYXd0Ljc5PWNhbm5vdCBzZXQgbnVsbCBmb2N1cyB0cmF2ZXJzYWwga2V5Ci1hd3QuN0E9Zm9jdXMgdHJhdmVyc2FsIGtleXMgY2Fubm90IG1hcCB0byBLRVlfVFlQRUQgZXZlbnRzCi1hd3QuN0I9Zm9jdXMgdHJhdmVyc2FsIGtleXMgbXVzdCBiZSB1bmlxdWUgZm9yIGEgQ29tcG9uZW50Ci1hd3QuN0M9dGhpcyBLZXlib2FyZEZvY3VzTWFuYWdlciBpcyBub3QgaW5zdGFsbGVkIGluIHRoZSBjdXJyZW50IHRocmVhZCdzIGNvbnRleHQKLWF3dC43RD1Qcm9wZXJ0eSBuYW1lIGlzIG51bGwKLWF3dC43RT1pbnZhbGlkIGhvdFNwb3QKLWF3dC43Rj1BZGRMYXlvdXRDb21wb25lbnQ6IGF0dGVtcHQgdG8gYWRkIG51bGwgY29tcG9uZW50Ci1hd3QuODA9QWRkTGF5b3V0Q29tcG9uZW50OiBjb25zdHJhaW50IG9iamVjdCBtdXN0IGJlIEdyaWRCYWdDb25zdHJhaW50cwotYXd0LjgxPUFkZExheW91dENvbXBvbmVudDogezB9Ci1hd3QuODI9UmVtb3ZlTGF5b3V0Q29tcG9uZW50OiBhdHRlbXB0IHRvIHJlbW92ZSBudWxsIGNvbXBvbmVudAotYXd0LjgzPVNldENvbnN0cmFpbnRzOiBhdHRlbXB0IHRvIGdldCBjb25zdHJhaW50cyBvZiBudWxsIGNvbXBvbmVudAotYXd0Ljg0PVNldENvbnN0cmFpbnRzOiBhdHRlbXB0IHRvIHNldCBudWxsIGNvbnN0cmFpbnRzCi1hd3QuODU9U2V0Q29uc3RyYWludHM6IHswfQotYXd0Ljg2PU1pbmltdW1MYXlvdXRTaXplOiB7MH0KLWF3dC44Nz1QcmVmZXJyZWRMYXlvdXRTaXplOiB7MH0KLWF3dC44OD1MYXlvdXRDb250YWluZXI6IHswfQotYXd0Ljg5PUxvb2t1cENvbnN0cmFpbnRzOiBhdHRlbXB0IHRvIGdldCBjb25zdHJhaW50cyBvZiBudWxsIGNvbXBvbmVudAotYXd0LjhBPUFkanVzdEZvckdyYXZpdHk6IGF0dGVtcHQgdG8gdXNlIG51bGwgY29uc3RyYWludHMKLWF3dC44Qj1BZGp1c3RGb3JHcmF2aXR5OiBhdHRlbXB0IHRvIHVzZSBudWxsIHJlY3RhbmdsZQotYXd0LjhDPUFkanVzdEZvckdyYXZpdHk6IHswfQotYXd0LjhEPVJFTUlOREVSIGNvbXBvbmVudCBleHBlY3RlZCBhZnRlciBSRUxBVElWRSBvbmUKLWF3dC44RT1jb21wb25lbnQgaXMgb3V0IG9mIGdyaWQncyByYW5nZQotYXd0LjhGPVdlaWdodHMnIG92ZXJyaWRlcyBhcnJheSBpcyB0b28gbG9uZwotYXd0LjkwPUxlbmd0aHMnIG92ZXJyaWRlcyBhcnJheSBpcyB0b28gbG9uZwotYXd0LjkxPVVuc3VwcG9ydGVkIGNvbnN0cmFpbnRzIG9iamVjdDogezB9Ci1hd3QuOTI9Q29uc3RyYWludHMgb2JqZWN0IG11c3QgYmUgU3RyaW5nCi1hd3QuOTM9Y2Fubm90IGdldCBjb21wb25lbnQ6IGludmFsaWQgY29uc3RyYWludDogezB9Ci1hd3QuOTQ9dHJhbnNmb3JtIGNhbiBub3QgYmUgbnVsbAotYXd0Ljk1PVdyb25nIHN0YXJ0IGluZGV4OiB7MH0KLWF3dC45Nj1Xcm9uZyBmaW5pc2ggaW5kZXg6IHswfQotYXd0Ljk3PVdyb25nIHJhbmdlIGxlbmd0aDogezB9Ci1hd3QuOTg9V3JvbmcgY291bnQgdmFsdWUsIGNhbiBub3QgYmUgbmVnYXRpdmU6IHswfQotYXd0Ljk5PVdyb25nIFtzdGFydCArIGNvdW50XSBpcyBvdXQgb2YgcmFuZ2U6IHswfQotYXd0LjlBPVVuc3VwcG9ydGVkIGZvbnQgZm9ybWF0Ci1hd3QuOUI9Q2FuJ3QgY3JlYXRlIGZvbnQgLSBiYWQgZm9udCBkYXRhCi1hd3QuOUM9d3JvbmcgdmFsdWUgb2YgR3JpZEJhZ0NvbnN0cmFpbnRzOiB7MH0KLWF3dC45RD1yZWxhdGl2ZSBncmlkIHNpemUgcGFyYW1ldGVyIGdvZXMgYWZ0ZXIgYWJzb2x1dGUgZ3JpZCBjb29yZGluYXRlCi1hd3QuOUU9d3JvbmcgdmFsdWVzIHN1bSBvZiBHcmlkQmFnQ29uc3RyYWludHMnIGdyaWR3aWR0aCBhbmQgZ3JpZHgKLWF3dC45Rj13cm9uZyB2YWx1ZXMgc3VtIG9mIEdyaWRCYWdDb25zdHJhaW50cycgZ3JpZGhlaWdodCBhbmQgZ3JpZHkKLWF3dC4xMDA9Y29tcG9uZW50IGhhcyBSRUxBVElWRSB3aWR0aCBhbmQgaGVpZ2h0Ci1hd3QuMTAxPXBvc2l0aW9uIGxlc3MgdGhhbiB6ZXJvLgotYXd0LjEwMj1jb2x1bW5zIGxlc3MgdGhhbiB6ZXJvLgotYXd0LjEwMz1pdGVtIGlzIG51bGwKLWF3dC4xMDQ9aXRlbSBkb2Vzbid0IGV4aXN0IGluIHRoZSBjaG9pY2UgbWVudQotYXd0LjEwNT1pbmRleCBsZXNzIHRoYW4gemVybwotYXd0LjEwNj1zcGVjaWZpZWQgcG9zaXRpb24gaXMgZ3JlYXRlciB0aGFuIHRoZSBudW1iZXIgb2YgaXRlbXMKLWF3dC4xMDc9Q29sb3IgcGFyYW1ldGVyIG91dHNpZGUgb2YgZXhwZWN0ZWQgcmFuZ2U6IGNvbXBvbmVudCB7MH0KLWF3dC4xMDg9QWxwaGEgdmFsdWUgb3V0c2lkZSBvZiBleHBlY3RlZCByYW5nZQotYXd0LjEwOT1Db2xvciBwYXJhbWV0ZXIgb3V0c2lkZSBvZiBleHBlY3RlZCByYW5nZQotYXd0LjEwQT1Qcmlvcml0eSBtdXN0IGJlIGEgdmFsdWUgYmV0d2VlbiAwIGFuZCAxLCBpbmNsdXNpdmUKLWF3dC4xMEI9YUNvbnRhaW5lciBhbmQgYUNvbXBvbmVudCBjYW5ub3QgYmUgbnVsbAotYXd0LjEwQz1hQ29udGFpbmVyIGlzIG5vdCBhIGZvY3VzIGN5Y2xlIHJvb3Qgb2YgYUNvbXBvbmVudAotYXd0LjEwRD1hQ29udGFpbmVyIHNob3VsZCBiZSBmb2N1cyBjeWNsZSByb290IG9yIGZvY3VzIHRyYXZlcnNhbCBwb2xpY3kgcHJvdmlkZXIKLWF3dC4xMEU9Zm9jdXNDeWNsZVJvb3QgY2Fubm90IGJlIG51bGwKLWF3dC4xMEY9aW1wcm9wZXIgYWxpZ25tZW50OiB7MH0KLWF3dC4xMTA9SXRlcmF0b3Igb3V0IG9mIGJvdW5kcwotYXd0LjExMT1QYXJhbWV0ZXIgbnBvaW50cyBpcyBncmVhdGVyIHRoYW4gYXJyYXkgbGVuZ3RoCi1hd3QuMTEyPU5lZ2F0aXZlIG51bWJlciBvZiBwb2ludHMKLWF3dC4xMTM9aWxsZWdhbCBzY3JvbGxiYXIgb3JpZW50YXRpb24KLWF3dC4xMTQ9SW1hZ2UgaXMgbnVsbAotYXd0LjExNT1BbmNob3IgaXMgbnVsbAotYXd0LjExNj1JbnZhbGlkIHZhbHVlIGZvciBtZWRpYQotYXd0LjExNz1JbnZhbGlkIHZhbHVlIGZvciBvcmllbnRhdGlvblJlcXVlc3RlZAotYXd0LjExOD1JbnZhbGlkIHZhbHVlIGZvciBwcmludGVyUmVzb2x1dGlvbgotYXd0LjExOT1JbnZhbGlkIHZhbHVlIGZvciBvcmlnaW4KLWF3dC4xMUE9SW52YWxpZCB2YWx1ZSBmb3IgcHJpbnRRdWFsaXR5Ci1hd3QuMTFCPUludmFsaWQgdmFsdWUgZm9yIHByaW50ZXJSZXNvbHV0aW9uW10KLWF3dC4xMUM9SW52YWxpZCB2YWx1ZSBmb3IgY29sb3IKLWF3dC4xMUQ9VW5rbm93biBydWxlCi1hd3QuMTFFPVdyb25nIGFscGhhIHZhbHVlCi1hd3QuMTFGPXBhcmVudCBpcyBub3QgYSBjb21wb25lbnQKLWF3dC4xMjA9b3JpZ2luIGlzIG5vdCBhIGRlc2NlbmRhbnQgb2YgcGFyZW50Ci1hd3QuMTIxPXBhcmVudCBtdXN0IGJlIHNob3dpbmcgb24gdGhlIHNjcmVlbgotYXd0LjEyMj1Eb2VzIG5vdCBzdXBwb3J0IGRpc3BsYXkgbW9kZSBjaGFuZ2VzCi1hd3QuMTIzPVVuc3VwcG9ydGVkIGRpc3BsYXkgbW9kZTogezB9Ci1hd3QuMTI0PUNhbm5vdCBjaGFuZ2UgdGhlIG1vZGFsaXR5IHdoaWxlIHRoZSBkaWFsb2cgaXMgdmlzaWJsZQotYXd0LjEyNT1udWxsIG93bmVyIHdpbmRvdwotYXd0LjEyNj1XaW5kb3cgaXMgc2hvd2luZwotYXd0LjEyNz1DYW5ub3QgY2hhbmdlIHRoZSBkZWNvcmF0aW9ucyB3aGlsZSB0aGUgd2luZG93IGlzIHZpc2libGUKLWF3dC4xMjg9R3JhcGhpY3MgZW52aXJvbm1lbnQgaXMgaGVhZGxlc3MKLWF3dC4xMjk9Tm90IGEgc2NyZWVuIGRldmljZQotYXd0LjEyQT1pbGxlZ2FsIGNvbXBvbmVudCBwb3NpdGlvbgotYXd0LjEyQj1hZGRpbmcgY29udGFpbmVyIHRvIGl0c2VsZgotYXd0LjEyQz1hZGRpbmcgY29udGFpbmVyJ3MgcGFyZW50IHRvIGl0c2VsZgotYXd0LjEyRD1hZGRpbmcgYSB3aW5kb3cgdG8gYSBjb250YWluZXIKLWF3dC4xMkU9VW5rbm93biBjb21wb25lbnQgZXZlbnQgaWQKLWF3dC4xMkY9QXR0ZW1wdCB0byBzdGFydCBuZXN0ZWQgbW91c2UgZ3JhYgotYXd0LjEzMD1BdHRlbXB0IHRvIGdyYWIgbW91c2UgaW4gbm90IGRpc3BsYXlhYmxlIHdpbmRvdwotYXd0LjEzMT1BZGRMYXlvdXRDb21wb25lbnQ6IGNvbnN0cmFpbnQgb2JqZWN0IG11c3QgYmUgU3RyaW5nCi1hd3QuMTMyPXdyb25nIHBhcmVudCBmb3IgQ2FyZExheW91dAotYXd0LjEzMz1OZWdhdGl2ZSB3aWR0aAotYXd0LjEzND1JbGxlZ2FsIGNhcAotYXd0LjEzNT1JbGxlZ2FsIGpvaW4KLWF3dC4xMzY9bWl0ZXJMaW1pdCBsZXNzIHRoYW4gMS4wZgotYXd0LjEzNz1OZWdhdGl2ZSBkYXNoUGhhc2UKLWF3dC4xMzg9WmVybyBkYXNoIGxlbmd0aAotYXd0LjEzOT1OZWdhdGl2ZSBkYXNoW3swfV0KLWF3dC4xM0E9QWxsIGRhc2ggbGVuZ3RocyB6ZXJvCi1hd3QuMTNCPW9mZnNldCBvZmYgaXMgb3V0IG9mIHJhbmdlCi1hd3QuMTNDPW51bWJlciBvZiBlbGVtZXRzIGxlbiBpcyBvdXQgb2YgcmFuZ2UKLWF3dC4xM0Q9UmVjdGFuZ2xlIHdpZHRoIGFuZCBoZWlnaHQgbXVzdCBiZSA+IDAKLWF3dC4xM0U9Q2Fubm90IGNhbGwgbWV0aG9kIGZyb20gdGhlIGV2ZW50IGRpc3BhdGNoZXIgdGhyZWFkCi1hd3QuMTNGPURlbGF5IG11c3QgYmUgdG8gMCB0byA2MCwwMDBtcwotYXd0LjE0MD1JbnZhbGlkIGNvbWJpbmF0aW9uIG9mIGJ1dHRvbiBmbGFncwotYXd0LjE0MT1mYWlsZWQgdG8gcGFyc2UgaG90c3BvdCBwcm9wZXJ0eSBmb3IgY3Vyc29yOiAKLWF3dC4xNDI9RXhjZXB0aW9uOiBjbGFzcyB7MH0gezF9IG9jY3VycmVkIHdoaWxlIGxvYWRpbmc6IHsyfQotYXd0LjE0Mz1pbGxlZ2FsIGN1cnNvciB0eXBlCi1hd3QuMTQ0PUNhbiBiZSBzZXQgYnkgc2Nyb2xscGFuZSBvbmx5Ci1hd3QuMTQ1PWlsbGVnYWwgZmlsZSBkaWFsb2cgbW9kZQotYXd0LjE0Nj1pbGxlZ2FsIHNjcm9sbGJhciBkaXNwbGF5IHBvbGljeQotYXd0LjE0Nz1wb3NpdGlvbiBncmVhdGVyIHRoYW4gMAotYXd0LjE0OD1jaGlsZCBpcyBudWxsCi1hd3QuMTQ5PVNjcm9sbFBhbmUgY29udHJvbHMgbGF5b3V0Ci1hd3QuMTRBPUNhbiBub3QgY3JlYXRlIFZvbGF0aWxlSW1hZ2Ugd2l0aCBzcGVjaWZpZWQgY2FwYWJpbGl0aWVzCi1hd3QuMTRCPU9ubHkgQ2FudmFzIG9yIFdpbmRvdyBpcyBhbGxvd2VkCi1hd3QuMTRDPU51bWJlciBvZiBidWZmZXJzIG11c3QgYmUgZ3JlYXRlciB0aGFuIG9uZQotYXd0LjE0RD1CdWZmZXIgY2FwYWJpbGl0aWVzIHNob3VsZCBzdXBwb3J0IGZsaXBwaW5nCi1hd3QuMTRFPUNvbXBvbmVudCBzaG91bGQgYmUgZGlzcGxheWFibGUKLWF3dC4xNEY9aW52YWxpZCBmb2N1cyB0cmF2ZXJzYWwga2V5IGlkZW50aWZpZXIKLWF3dC4xNTA9bm8gcGFyZW50Ci1hd3QuMTUxPWNvbXBvbmVudCBtdXN0IGJlIHNob3dpbmcgb24gdGhlIHNjcmVlbiB0byBkZXRlcm1pbmUgaXRzIGxvY2F0aW9uCi1hd3QuMTUyPUludmFsaWQgbnVtYmVyIG9mIGNvcGllcwotYXd0LjE1Mz1JbnZhbGlkIHZhbHVlIGZvciBtYXhQYWdlCi1hd3QuMTU0PUludmFsaWQgdmFsdWUgZm9yIG1pblBhZ2UKLWF3dC4xNTU9SW52YWxpZCB2YWx1ZSBmb3IgZnJvbVBhZ2UKLWF3dC4xNTY9SW52YWxpZCB2YWx1ZSBmb3IgdG9QYWdlCi1hd3QuMTU3PUludmFsaWQgdmFsdWUgZm9yIHBhZ2VSYW5nZXMKLWF3dC4xNTg9SW52YWxpZCB2YWx1ZSBmb3IgZGVzdGluYXRpb24KLWF3dC4xNTk9SW52YWxpZCB2YWx1ZSBmb3IgZGlhbG9nCi1hd3QuMTVBPUludmFsaWQgdmFsdWUgZm9yIGRlZmF1bHRTZWxlY3Rpb24KLWF3dC4xNUI9SW52YWxpZCB2YWx1ZSBmb3IgbXVsdGlwbGVEb2N1bWVudEhhbmRsaW5nCi1hd3QuMTVDPUludmFsaWQgdmFsdWUgZm9yIGF0dHJpYnV0ZSBzaWRlcwotYXd0LjE1RD1JbnZhbGlkIGNvbG9yc3BhY2UKLWF3dC4xNUU9VW5rbm93biBjb21wb25lbnQuIE11c3QgYmUgUkVEQ09NUE9ORU5ULCBHUkVFTkNPTVBPTkVOVCBvciBCTFVFQ09NUE9ORU5ULgotYXd0LjE1Rj1Qcm9maWxlIGNsYXNzIGRvZXMgbm90IGNvbXBseSB3aXRoIElDQyBzcGVjaWZpY2F0aW9uCi1hd3QuMTYwPUNvbG9yIHNwYWNlIGRvZXNuJ3QgY29tcGx5IHdpdGggSUNDIHNwZWNpZmljYXRpb24KLWF3dC4xNjE9VW5hYmxlIHRvIG9wZW4gZmlsZSB7MH0KLWF3dC4xNjI9SW52YWxpZCBJQ0MgUHJvZmlsZSBEYXRhCi1hd3QuMTYzPUNhbid0IG9wZW4gY29sb3IgcHJvZmlsZQotYXd0LjE2ND1Ob3QgYSBwcmVkZWZpbmVkIGNvbG9yIHNwYWNlCi1hd3QuMTY1PUNvbG9yIHNwYWNlIGRvZXNuJ3QgY29tcGx5IHdpdGggSUNDIHNwZWNpZmljYXRpb24KLWF3dC4xNjY9VFJDIGlzIG5vdCBhIHNpbXBsZSBnYW1tYSB2YWx1ZQotYXd0LjE2Nz1UUkMgaXMgYSBnYW1tYSB2YWx1ZSwgbm90IGEgdGFibGUKLWF3dC4xNjg9SW52YWxpZCBwcm9maWxlIGNsYXNzCi1hd3QuMTY5PUNvbXBvbmVudCBpbmRleCBvdXQgb2YgcmFuZ2UKLWF3dC4xNkE9SW52YWxpZCBjb21wb25lbnQgaW5kZXg6IHswfQotYXd0LjE2Qj1Ob3QgYSBwcmVkZWZpbmVkIGNvbG9yc3BhY2UKLWF3dC4xNkM9Q2FuJ3QgbG9hZCBjbGFzczogezB9Ci1hd3QuMTZEPUNhbid0IHBhcnNlIE1JTUUgdHlwZTogezB9Ci1hd3QuMTZFPVRyYW5zZmVyYWJsZSBoYXMgbnVsbCBkYXRhCi1hd3QuMTZGPUNhbid0IGNyZWF0ZSByZWFkZXIgZm9yIHRoaXMgcmVwcmVzZW50YXRpb24gY2xhc3MKLWF3dC4xNzA9Q2FuJ3QgY3JlYXRlIGRlZmF1bHQgRCZEIGN1cnNvcjogezB9Ci1hd3QuMTcxPUF0dGVtcHQgdG8gc3RhcnQgYSBkcmFnIHdoaWxlIGFuIGV4aXN0aW5nIGRyYWcgb3BlcmF0aW9uIGlzIHN0aWxsIGV4ZWN1dGluZwotYXd0LjE3Mj1EcmFnIHNvdXJjZSBpcyBudWxsCi1hd3QuMTczPU9uZSBsaXN0ZW5lciBpcyBhbHJlYWR5IGV4aXN0Ci1hd3QuMTc0PWRnbCBpcyBub3QgY3VycmVudCBsaXN0ZW5lcgotYXd0LjE3NT1MaXN0ZW5lciBtaXNtYXRjaAotYXd0LjE3Nj1Ecm9wVGFyZ2V0IGNhbm5vdCBiZSBhZGRlZCBhcyBsaXN0ZW5lciB0byBpdHNlbGYKLWF3dC4xNzc9SW52YWxpZCB1c2VyIGFjdGlvbgotYXd0LjE3OD1JbnZhbGlkIHNvdXJjZSBhY3Rpb24KLWF3dC4xNzk9Q29udGV4dCBwZWVyIGlzIG51bGwKLWF3dC4xN0E9VHJpZ2dlciBldmVudCBpcyBudWxsCi1hd3QuMTdCPUNhbid0IGluaXQgQUNUSU9OX05PTkUgZHJhZwotYXd0LjE3Qz1JbWFnZSBvZmZzZXQgaXMgbnVsbAotYXd0LjE3RD1UcmFuc2ZlcmFibGUgaXMgbnVsbAotYXd0LjE3RT1Db21wb25lbnQgYXNzb2NpYXRlZCB3aXRoIHRoZSB0cmlnZ2VyIGV2ZW50IGlzIG51bGwKLWF3dC4xN0Y9RHJhZ1NvdXJjZSBmb3IgdGhlIHRyaWdnZXIgZXZlbnQgaXMgbnVsbAotYXd0LjE4MD1Tb3VyY2UgYWN0aW9ucyBmb3IgdGhlIERyYWdHZXN0dXJlUmVjb2duaXplciBhc3NvY2lhdGVkIHdpdGggdGhlIHRyaWdnZXIgZXZlbnQgYXJlIGVxdWFsIHRvIERuRENvbnN0YW50cy5BQ1RJT05fTk9ORQotYXd0LjE4MT1BdHRlbXB0IHRvIHJlZ2lzdGVyIGNvbnRleHQgYXMgaXRzIGxpc3RlbmVyCi1hd3QuMTgyPWRzbCBpcyBub3QgY3VycmVudCBsaXN0ZW5lcgotYXd0LjE4Mz1JbnZhbGlkIHN0YXR1cwotYXd0LjE4ND1JbnZhbGlkIGFjdGlvbgotYXd0LjE4NT1Db21wb25lbnQgaXMgbnVsbAotYXd0LjE4Nj1EcmFnU291cmNlIGlzIG51bGwKLWF3dC4xODc9T3JpZ2luIGlzIG51bGwKLWF3dC4xODg9RXZlbnQgbGlzdCBpcyBudWxsCi1hd3QuMTg5PUV2ZW50IGxpc3QgaXMgZW1wdHkKLWF3dC4xOEE9Q29udGV4dCBpcyBudWxsCi1hd3QuMThCPUludmFsaWQgYnV0dG9uIHZhbHVlCi1hd3QuMThDPUNhbm5vdCBpbnZva2UgbnVsbCBydW5uYWJsZQotYXd0LjE4RD1Tb3VyY2UgaXMgbnVsbAotYXd0LjE4RT1Xcm9uZyBldmVudCBpZAotYXd0LjE4Rj1UZXh0IG11c3QgYmUgbnVsbCBmb3IgQ0FSRVRfUE9TSVRJT05fQ0hBTkdFRAotYXd0LjE5MD1Xcm9uZyBjb21taXR0ZWRDaGFyYWN0ZXJDb3VudAotYXd0LjE5MT1JbnZhbGlkIGtleUNvZGUgZm9yIEtFWV9UWVBFRCBldmVudCwgbXVzdCBiZSBWS19VTkRFRklORUQKLWF3dC4xOTI9SW52YWxpZCBrZXlDaGFyIGZvciBLRVlfVFlQRUQgZXZlbnQsIGNhbid0IGJlIENIQVJfVU5ERUZJTkVECi1hd3QuMTkzPUxpc3RlbmVyIGNhbid0IGJlIHplcm8KLWF3dC4xOTQ9VW5rbm93biBhdHRyaWJ1dGUgbmFtZQotYXd0LjE5NT1PZmZzZXQgaXMgb3V0IG9mIGJvdW5kcwotYXd0LjE5Nj1KdXN0aWZpY2F0aW9uIGltcG9zc2libGUsIGxheW91dCBhbHJlYWR5IGp1c3RpZmllZAotYXd0LjE5Nz1FbmRwb2ludHMgYXJlIG91dCBvZiByYW5nZQotYXd0LjE5OD1JbGxlZ2FsIGFsaWdubWVudCBhcmd1bWVudAotYXd0LjE5OT1JbGxlZ2FsIHJhbmdlIGFyZ3VtZW50IHZhbHVlOiB7MH0KLWF3dC4xOUE9c3RhcnQgb3IgY291bnQgYXJndW1lbnRzIGFyZSBvdXQgb2YgdGV4dCByYW5nZQotYXd0LjE5Qj1jb3VudCBhcmd1bWVudCBtdXN0IGJlIHBvc2l0aXZlCi1hd3QuMTlDPXdlaWdodCBtdXN0IGJlIGEgcG9zaXRpdmUgbnVtYmVyCi1hd3QuMTlEPWdyb3dMZWZ0TGltaXQgbXVzdCBiZSBhIHBvc2l0aXZlIG51bWJlcgotYXd0LjE5RT1ncm93UmlnaHRMaW1pdCBtdXN0IGJlIGEgcG9zaXRpdmUgbnVtYmVyCi1hd3QuMTlGPWluY29ycmVjdCB2YWx1ZSBmb3Igc2hyaW5rUHJpb3JpdHksIG1vcmUgdGhhbiBQUklPUklUWV9OT05FIG9yIGxlc3MgdGhhbiBQUklPUklUWV9LQVNISURBIHZhbHVlCi1hd3QuMjAwPWluY29ycmVjdCB2YWx1ZSBmb3IgZ3Jvd1ByaW9yaXR5LCBtb3JlIHRoYW4gUFJJT1JJVFlfTk9ORSBvciBsZXNzIHRoYW4gUFJJT1JJVFlfS0FTSElEQSB2YWx1ZQotYXd0LjIwMT1zaHJpbmtMZWZ0TGltaXQgbXVzdCBiZSBhIHBvc2l0aXZlIG51bWJlcgotYXd0LjIwMj1zaHJpbmtSaWdodExpbWl0IG11c3QgYmUgYSBwb3NpdGl2ZSBudW1iZXIKLWF3dC4yMDM9T2Zmc2V0IGxpbWl0IHNob3VsZCBiZSBncmVhdGVyIHRoYW4gY3VycmVudCBwb3NpdGlvbgotYXd0LjIwND1EZXRlcm1pbmFudCBpcyB6ZXJvCi1hd3QuMjA1PUludmFsaWQgdHlwZSBvZiBBcmM6IHswfQotYXd0LjIwNj1GbGF0bmVzcyBpcyBsZXNzIHRoZW4gemVybwotYXd0LjIwNz1MaW1pdCBpcyBsZXNzIHRoZW4gemVybwotYXd0LjIwOD1QYXRoIGlzIG51bGwKLWF3dC4yMDk9SW52YWxpZCB3aW5kaW5nIHJ1bGUgdmFsdWUKLWF3dC4yMEE9Rmlyc3Qgc2VnbWVudCBzaG91bGQgYmUgU0VHX01PVkVUTyB0eXBlCi1hd3QuMjBCPXVua25vd24gaW5wdXQgbWV0aG9kIGhpZ2hsaWdodCBzdGF0ZQotYXd0LjIwQz1OdW1iZXIgb2YgQml0cyBlcXVhbHMgdG8gemVybwotYXd0LjIwRD1UaGUgbnVtYmVyIG9mIGJpdHMgcGVyIHBpeGVsIGlzIG5vdCBhIHBvd2VyIG9mIDIgb3IgcGl4ZWxzIHNwYW4gZGF0YSBlbGVtZW50IGJvdW5kYXJpZXMKLWF3dC4yMEU9RGF0YSBCaXQgb2Zmc2V0IGlzIG5vdCBhIG11bHRpcGxlIG9mIHBpeGVsIGJpdCBzdHJpZGUKLWF3dC4yMEY9TnVtYmVyIG9mIGJhbmRzIG11c3QgYmUgb25seSAxCi1hd3QuMjEwPVRoZSBjb21wb25lbnQgdmFsdWUgZm9yIHRoaXMgQ29sb3JNb2RlbCBpcyBzaWduZWQKLWF3dC4yMTE9UGl4ZWwgdmFsdWVzIGZvciB0aGlzIENvbG9yTW9kZWwgYXJlIG5vdCBjb252ZW5pZW50bHkgcmVwcmVzZW50YWJsZSBhcyBhIHNpbmdsZSBpbnQKLWF3dC4yMTI9VGhlcmUgaXMgbW9yZSB0aGFuIG9uZSBjb21wb25lbnQgaW4gdGhpcyBDb2xvck1vZGVsCi1hd3QuMjEzPVRoaXMgQ29tcG9uZW50Q29sb3JNb2RlbCBkb2VzIG5vdCBzdXBwb3J0IHRoZSB1bm5vcm1hbGl6ZWQgZm9ybQotYXd0LjIxND1UaGlzIENvbG9yIE1vZGVsIGRvZXNuJ3Qgc3VwcG9ydCB0aGlzIHRyYW5zZmVyVHlwZQotYXd0LjIxNT10cmFuc2ZlclR5cGUgaXMgbm90IG9uZSBvZiBEYXRhQnVmZmVyLlRZUEVfQllURSwgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVCwgRGF0YUJ1ZmZlci5UWVBFX0lOVCwgRGF0YUJ1ZmZlci5UWVBFX1NIT1JULCBEYXRhQnVmZmVyLlRZUEVfRkxPQVQsIG9yIERhdGFCdWZmZXIuVFlQRV9ET1VCTEUKLWF3dC4yMTY9VGhlIGNvbXBvbmVudHMgYXJyYXkgaXMgbm90IGxhcmdlIGVub3VnaCB0byBob2xkIGFsbCB0aGUgY29sb3IgYW5kIGFscGhhIGNvbXBvbmVudHMKLWF3dC4yMTc9VGhlIHRyYW5zZmVyIHR5cGUgb2YgdGhpcyBDb21wb25lbnRDb2xvck1vZGVsIGlzIG5vdCBvbmUgb2YgdGhlIGZvbGxvd2luZyB0cmFuc2ZlciB0eXBlczogRGF0YUJ1ZmZlci5UWVBFX0JZVEUsIERhdGFCdWZmZXIuVFlQRV9VU0hPUlQsIG9yIERhdGFCdWZmZXIuVFlQRV9JTlQKLWF3dC4yMTg9VGhlIGNvbXBvbmVudHMgYXJyYXkgaXMgbm90IGxhcmdlIGVub3VnaCB0byBob2xkIGFsbCB0aGUgY29sb3IgYW5kIGFscGhhIGNvbXBvbmVudHMKLWF3dC4yMTk9VGhpcyB0cmFuc2ZlclR5cGUgaXMgbm90IHN1cHBvcnRlZCBieSB0aGlzIGNvbG9yIG1vZGVsCi1hd3QuMjFBPVRoaXMgQ29tcG9uZW50Q29sb3JNb2RlbCBkb2VzIG5vdCBzdXBwb3J0IHRoaXMgdHJhbnNmZXJUeXBlCi1hd3QuMjFCPVRoZSBsZW5ndGggb2Ygbm9ybUNvbXBvbmVudHMgbWludXMgbm9ybU9mZnNldCBpcyBsZXNzIHRoYW4gbnVtQ29tcG9uZW50cwotYXd0LjIxQz1UaGUgbnVtYmVyIG9mIHNjYWxlIGZhY3RvcnMgc2hvdWxkIG5vdCBiZSB6ZXJvCi1hd3QuMjFEPU51bWJlciBvZiBzcmMgYmFuZHMgKHswfSkgZG9lcyBub3QgbWF0Y2ggbnVtYmVyIG9mIGRzdCBiYW5kcyAoezF9KQotYXd0LjIxRT1OdW1iZXIgb2Ygc2NhbGluZyBjb25zdGFudHMgaXMgbm90IGVxdWFsIHRvIHRoZSBudW1iZXIgb2YgYmFuZHMKLWF3dC4yMUY9VW5hYmxlIHRvIHRyYW5zZm9ybSBzb3VyY2UKLWF3dC4yMjA9U291cmNlIHNob3VsZCBub3QgaGF2ZSBJbmRleENvbG9yTW9kZWwKLWF3dC4yMjE9VGhlIGltYWdlVHlwZSBpcyBUWVBFX0JZVEVfQklOQVJZIGFuZCB0aGUgY29sb3IgbWFwIGhhcyBtb3JlIHRoYW4gMTYgZW50cmllcwotYXd0LjIyMj1UaGUgaW1hZ2VUeXBlIGlzIG5vdCBUWVBFX0JZVEVfQklOQVJZIG9yIFRZUEVfQllURV9JTkRFWEVECi1hd3QuMjIzPVRoZSBpbWFnZVR5cGUgaXMgbm90IGNvbXBhdGlibGUgd2l0aCBDb2xvck1vZGVsCi1hd3QuMjI0PVVua25vd24gaW1hZ2UgdHlwZQotYXd0LjIyNT1Qcm9wZXJ0eSBuYW1lIGlzIG51bGwKLWF3dC4yMjY9Qm90aCB0aWxlWCBhbmQgdGlsZVkgYXJlIG5vdCBlcXVhbCB0byAwCi1hd3QuMjI3PVRoaXMgaW1hZ2UgdHlwZSBjYW4ndCBoYXZlIGFscGhhCi1hd3QuMjI4PW1pblggb3IgbWluWSBvZiB0aGlzIHJhc3RlciBub3QgZXF1YWwgdG8gemVybwotYXd0LjIyOT1OdW1iZXIgb2YgY29tcG9uZW50cyBpbiB0aGUgTFVUIGRvZXMgbm90IG1hdGNoIHRoZSBudW1iZXIgb2YgYmFuZHMKLWF3dC4yMkE9V3JvbmcgdHlwZSBvZiBwaXhlbHMgYXJyYXkKLWF3dC4yMkI9TGVuZ3RoIG9mIGRhdGEgc2hvdWxkIG5vdCBiZSBsZXNzIHRoYW4gd2lkdGgqaGVpZ2h0Ci1hd3QuMjJDPVVua25vd24gZGF0YSB0eXBlIHswfQotYXd0LjIyRD1UaGlzIHRyYW5zZmVyVHlwZSAoIHswfSApIGlzIG5vdCBzdXBwb3J0ZWQgYnkgdGhpcyBjb2xvciBtb2RlbAotYXd0LjIyRT13IG9yIGggaXMgbGVzcyB0aGFuIG9yIGVxdWFsIHRvIHplcm8KLWF3dC4yMkY9VGhlIHByb2R1Y3Qgb2YgdyBhbmQgaCBpcyBncmVhdGVyIHRoYW4gSW50ZWdlci5NQVhfVkFMVUUKLWF3dC4yMzA9ZGF0YVR5cGUgaXMgbm90IG9uZSBvZiB0aGUgc3VwcG9ydGVkIGRhdGEgdHlwZXMKLWF3dC4yMzE9TnVtYmVyIG9mIGJhbmRzIG11c3QgYmUgbW9yZSB0aGVuIDAKLWF3dC4yMzI9T2Zmc2V0IHNob3VsZCBiZSBub3QgbGVzcyB0aGFuIHplcm8KLWF3dC4yMzM9TnVtYmVyIG9mIGNvbXBvbmVudHMgc2hvdWxkIGJlIHBvc2l0aXZlCi1hd3QuMjM0PVdpZHRoIG9yIEhlaWdodCBlcXVhbHMgemVybwotYXd0LjIzNT1Xcm9uZyBEYXRhIEJ1ZmZlciB0eXBlIDogezB9Ci1hd3QuMjM2PVRoZSBiaXRzIGlzIGxlc3MgdGhhbiAxIG9yIGdyZWF0ZXIgdGhhbiAzMgotYXd0LjIzNz1Tb3VyY2UgYW5kIGRlc3RpbmF0aW9ucyByYXN0ZXJzIGRvIG5vdCBoYXZlIHRoZSBzYW1lIG51bWJlciBvZiBiYW5kcwotYXd0LjIzOD1UaGUgbnVtYmVyIG9mIGFycmF5cyBpbiB0aGUgTG9va3VwVGFibGUgZG9lcyBub3QgbWVldCB0aGUgcmVzdHJpY3Rpb25zCi1hd3QuMjM5PVRoZSBzcGFjZSBpcyBub3QgYSBUWVBFX1JHQiBzcGFjZQotYXd0LjIzQT1UaGUgbWluL21heCBub3JtYWxpemVkIGNvbXBvbmVudCB2YWx1ZXMgYXJlIG5vdCAwLjAvMS4wCi1hd3QuMjNCPVRoZSBtYXNrIG9mIHRoZSB7MH0gY29tcG9uZW50IGlzIG5vdCBjb250aWd1b3VzCi1hd3QuMjNDPVRoZSBtYXNrIG9mIHRoZSBhbHBoYSBjb21wb25lbnQgaXMgbm90IGNvbnRpZ3VvdXMKLWF3dC4yM0Q9VGhlIG1hc2sgb2YgdGhlIHJlZCBjb21wb25lbnQgaXMgbm90IGNvbnRpZ3VvdXMKLWF3dC4yM0U9VGhlIG1hc2sgb2YgdGhlIGdyZWVuIGNvbXBvbmVudCBpcyBub3QgY29udGlndW91cwotYXd0LjIzRj1UaGUgbWFzayBvZiB0aGUgYmx1ZSBjb21wb25lbnQgaXMgbm90IGNvbnRpZ3VvdXMKLWF3dC4yNDA9VGhlIHRyYW5zZmVyVHlwZSBub3QgaXMgb25lIG9mIERhdGFCdWZmZXIuVFlQRV9CWVRFLCBEYXRhQnVmZmVyLlRZUEVfVVNIT1JUIG9yIERhdGFCdWZmZXIuVFlQRV9JTlQKLWF3dC4yNDE9QW55IG9mZnNldCBiZXR3ZWVuIGJhbmRzIGlzIGdyZWF0ZXIgdGhhbiB0aGUgU2NhbmxpbmUgc3RyaWRlCi1hd3QuMjQyPVBpeGVsIHN0cmlkZSBpcyBsZXNzIHRoYW4gYW55IG9mZnNldCBiZXR3ZWVuIGJhbmRzCi1hd3QuMjQzPVByb2R1Y3Qgb2YgUGl4ZWwgc3RyaWRlIGFuZCB3IGlzIGdyZWF0ZXIgdGhhbiBTY2FubGluZSBzdHJpZGUKLWF3dC4yNDQ9V2lkdGggb3IgSGVpZ2h0IG9mIGNoaWxkIFJhc3RlciBpcyBsZXNzIHRoYW4gb3IgZXF1YWwgdG8gemVybwotYXd0LjI0NT1wYXJlbnRYIGRpc3Bvc2VzIG91dHNpZGUgUmFzdGVyCi1hd3QuMjQ2PXBhcmVudFkgZGlzcG9zZXMgb3V0c2lkZSBSYXN0ZXIKLWF3dC4yNDc9cGFyZW50WCArIHcgcmVzdWx0cyBpbiBpbnRlZ2VyIG92ZXJmbG93Ci1hd3QuMjQ4PXBhcmVudFkgKyBoIHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotYXd0LjI0OT1jaGlsZE1pblggKyB3IHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotYXd0LjI0QT1jaGlsZE1pblkgKyBoIHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotYXd0LjI0Qj1QaXhlbCBzdHJpZGUgbXVzdCBiZSA+PSAwCi1hd3QuMjRDPVNjYW5saW5lIHN0cmlkZSBtdXN0IGJlID49IDAKLWF3dC4yNEQ9QmFuayBJbmRpY2VzIGxlbmd0aCBtdXN0IGJlIGVxdWFsIEJhbmsgT2Zmc2V0cyBsZW5ndGgKLWF3dC4yNEU9SW5kZXggb2YgezB9IGJhbmsgbXVzdCBiZSA+PSAwCi1hd3QuMjRGPVVuYWJsZSB0byBpbnZlcnQgdHJhbnNmb3JtIHswfQotYXd0LjI1MD1Vbmtub3duIGludGVycG9sYXRpb24gdHlwZTogezB9Ci1hd3QuMjUxPVRyYW5zZm9ybWVkIHdpZHRoICh7MH0pIGFuZCBoZWlnaHQgKHsxfSkgc2hvdWxkIGJlIGdyZWF0ZXIgdGhhbiAwCi1hd3QuMjUyPVNvdXJjZSBjYW4ndCBiZSBzYW1lIGFzIHRoZSBkZXN0aW5hdGlvbgotYXd0LjI1Mz1EaWZmZXJlbnQgbnVtYmVyIG9mIGJhbmRzIGluIHNvdXJjZSBhbmQgZGVzdGluYXRpb24KLWF3dC4yNTQ9TnVtYmVyIG9mIGJhbmRzIGluIHRoZSBzb3VyY2UgcmFzdGVyICh7MH0pIGlzIGluY29tcGF0aWJsZSB3aXRoIHRoZSBtYXRyaXggW3sxfXh7Mn1dCi1hd3QuMjU1PU51bWJlciBvZiBiYW5kcyBpbiB0aGUgZGVzdGluYXRpb24gcmFzdGVyICh7MH0pIGlzIGluY29tcGF0aWJsZSB3aXRoIHRoZSBtYXRyaXggW3sxfXh7Mn1dCi1hd3QuMjU2PVNvdXJjZSByYXN0ZXIgaXMgbnVsbAotYXd0LjI1Nz1Tb3VyY2UgcmFzdGVyIGlzIGVxdWFsIHRvIGRlc3RpbmF0aW9uCi1hd3QuMjU4PU51bWJlciBvZiBzb3VyY2UgYmFuZHMgKHswfSkgaXMgbm90IGVxdWFsIHRvIG51bWJlciBvZiBkZXN0aW5hdGlvbiBiYW5kcyAoezF9KQotYXd0LjI1OT1Tb3VyY2UgaW1hZ2UgaXMgbnVsbAotYXd0LjI1QT1Tb3VyY2UgZXF1YWxzIHRvIGRlc3RpbmF0aW9uCi1hd3QuMjVCPU51bGwgQ29sb3JTcGFjZSBwYXNzZWQgYXMgYSBwYXJhbWV0ZXIKLWF3dC4yNUM9TnVsbCBwcm9maWxlcyBwYXNzZWQgYXMgYSBwYXJhbWV0ZXIKLWF3dC4yNUQ9U291cmNlIG9yIGRlc3RpbmF0aW9uIGNvbG9yIHNwYWNlIGlzIG5vdCBkZWZpbmVkCi1hd3QuMjVFPUluY29ycmVjdCBudW1iZXIgb2Ygc291cmNlIHJhc3RlciBiYW5kcy4gU2hvdWxkIGJlIGVxdWFsIHRvIHRoZSBudW1iZXIgb2YgY29sb3IgY29tcG9uZW50cyBvZiBzb3VyY2UgY29sb3JzcGFjZS4KLWF3dC4yNUY9SW5jb3JyZWN0IG51bWJlciBvZiBkZXN0aW5hdGlvbiByYXN0ZXIgYmFuZHMuIFNob3VsZCBiZSBlcXVhbCB0byB0aGUgbnVtYmVyIG9mIGNvbG9yIGNvbXBvbmVudHMgb2YgZGVzdGluYXRpb24gY29sb3JzcGFjZS4KLWF3dC4yNjA9SW5jb21wYXRpYmxlIHJhc3RlcnMgLSB3aWR0aCBvciBoZWlnaHQgZGlmZmVycwotYXd0LjI2MT1EZXN0aW5hdGlvbiBjb2xvciBzcGFjZSBpcyB1bmRlZmluZWQKLWF3dC4yNjI9RGVzdGlvbmF0aW9uIGNvbG9yIHNwYWNlIHNob3VsZCBiZSBkZWZpbmVkCi1hd3QuMjYzPUluY29tcGF0aWJsZSBpbWFnZXMgLSB3aWR0aCBvciBoZWlnaHQgZGlmZmVycwotYXd0LjI2ND1TaXplIG9mIHRoZSBjb2xvciBtYXAgaXMgbGVzcyB0aGFuIDEKLWF3dC4yNjU9VGhlIHJhc3RlciBhcmd1bWVudCBpcyBub3QgY29tcGF0aWJsZSB3aXRoIHRoaXMgSW5kZXhDb2xvck1vZGVsCi1hd3QuMjY2PVRoZSBudW1iZXIgb2YgYml0cyBpbiBhIHBpeGVsIGlzIGdyZWF0ZXIgdGhhbiAxNgotYXd0LjI2Nz1UaGUgdHJhbnNmZXJUeXBlIGlzIGludmFsaWQKLWF3dC4yNjg9VGhlIHBpeGVsIGlzIG5vdCBhIHByaW1pdGl2ZSBhcnJheSBvZiB0eXBlIHRyYW5zZmVyVHlwZQotYXd0LjI2OT1UaGUgdHJhbnNmZXJUeXBlIGlzIG5vdCBvbmUgb2YgRGF0YUJ1ZmZlci5UWVBFX0JZVEUgb3IgRGF0YUJ1ZmZlci5UWVBFX1VTSE9SVAotYXd0LjI2QT1JbmNvcnJlY3QgSW1hZ2VDb25zdW1lciBjb21wbGV0aW9uIHN0YXR1cwotYXd0LjI2Qj1UaGUgbnVtYmVyIG9mIGJpdHMgaW4gdGhlIHBpeGVsIHZhbHVlcyBpcyBsZXNzIHRoYW4gMQotYXd0LjI2Qz1iaXRzIGlzIG51bGwKLWF3dC4yNkQ9VGhlIGVsZW1lbnRzIGluIGJpdHMgaXMgbGVzcyB0aGFuIDAKLWF3dC4yNkU9VGhlIHN1bSBvZiB0aGUgbnVtYmVyIG9mIGJpdHMgaW4gYml0cyBpcyBsZXNzIHRoYW4gMQotYXd0LjI2Rj1UaGUgY3NwYWNlIGlzIG51bGwKLWF3dC4yNzA9VGhlIHRyYW5zcGFyZW5jeSBpcyBub3QgYSB2YWxpZCB2YWx1ZQotYXd0LjI3MT1UaGUgbnVtYmVyIG9mIGJpdHMgaW4gYml0cyBpcyBsZXNzIHRoYW4gMQotYXd0LjI3Mj1UaGUgbGVuZ3RoIG9mIGNvbXBvbmVudHMgbWludXMgb2Zmc2V0IGlzIGxlc3MgdGhhbiBudW1Db21wb25lbnRzCi1hd3QuMjczPVRoZSBsZW5ndGggb2Ygbm9ybUNvbXBvbmVudHMgbWludXMgbm9ybU9mZnNldCBpcyBsZXNzIHRoYW4gbnVtQ29tcG9uZW50cwotYXd0LjI3ND1jb21wb25lbnRJZHggaXMgZ3JlYXRlciB0aGFuIHRoZSBudW1iZXIgb2YgY29tcG9uZW50cyBvciBsZXNzIHRoYW4gemVybwotYXd0LjI3NT1UaGlzIHBpeGVsIHJlcHJlc2VudGF0aW9uIGlzIG5vdCBzdXVwb3J0ZWQgYnkgdGlzIENvbG9yIE1vZGVsCi1hd3QuMjc2PWxvY2F0aW9uLnggKyB3IG9yIGxvY2F0aW9uLnkgKyBoIHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotYXd0LjI3Nz1iYW5rSW5kaWNlcyBvciBiYW5kT2Zmc2V0cyBpcyBudWxsCi1hd3QuMjc4PWRhdGFCdWZmZXIgaXMgbnVsbAotYXd0LjI3OT1iYW5kcyBpcyBsZXNzIHRoYW4gMQotYXd0LjI3QT1kYXRhQnVmZmVyIGhhcyBtb3JlIHRoYW4gb25lIGJhbmsKLWF3dC4yN0I9YmFuZE9mZnNldHMgaXMgbnVsbAotYXd0LjI3Qz1iYW5kTWFza3MgaXMgbnVsbAotYXd0LjI3RD1iaXRzUGVyQmFuZCBvciBiYW5kcyBpcyBub3QgZ3JlYXRlciB0aGFuIHplcm8KLWF3dC4yN0U9VGhlIHByb2R1Y3Qgb2YgYml0c1BlckJhbmQgYW5kIGJhbmRzIGlzIGdyZWF0ZXIgdGhhbiB0aGUgbnVtYmVyIG9mIGJpdHMgaGVsZCBieSBkYXRhVHlwZQotYXd0LjI3Rj1TYW1wbGVNb2RlbCBvciBEYXRhQnVmZmVyIGlzIG51bGwKLWF3dC4yODA9U2FtcGxlTW9kZWwgaXMgbnVsbAotYXd0LjI4MT1zYW1wbGVNb2RlbCwgZGF0YUJ1ZmZlciwgYVJlZ2lvbiBvciBzYW1wbGVNb2RlbFRyYW5zbGF0ZSBpcyBudWxsCi1hd3QuMjgyPWFSZWdpb24gaGFzIHdpZHRoIG9yIGhlaWdodCBsZXNzIHRoYW4gb3IgZXF1YWwgdG8gemVybwotYXd0LjI4Mz1PdmVyZmxvdyBYIGNvb3JkaW5hdGUgb2YgUmFzdGVyCi1hd3QuMjg0PU92ZXJmbG93IFkgY29vcmRpbmF0ZSBvZiBSYXN0ZXIKLWF3dC4yODU9V2lkdGggb3IgSGVpZ2h0IG9mIGNoaWxkIFJhc3RlciBpcyBsZXNzIHRoYW4gb3IgZXF1YWwgdG8gemVybwotYXd0LjI4Nj1wYXJlbnRYIGRpc3Bvc2VzIG91dHNpZGUgUmFzdGVyCi1hd3QuMjg3PXBhcmVudFkgZGlzcG9zZXMgb3V0c2lkZSBSYXN0ZXIKLWF3dC4yODg9cGFyZW50WCArIHdpZHRoIHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotYXd0LjI4OT1wYXJlbnRZICsgaGVpZ2h0IHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotYXd0LjI4QT1jaGlsZE1pblggKyB3aWR0aCByZXN1bHRzIGluIGludGVnZXIgb3ZlcmZsb3cKLWF3dC4yOEI9Y2hpbGRNaW5ZICsgaGVpZ2h0IHJlc3VsdHMgaW4gaW50ZWdlciBvdmVyZmxvdwotYXd0LjI4Qz1SZWN0IGlzIG51bGwKLWF3dC4yOEQ9TGVuZ3RoIG9mIGRhdGFBcnJheVt7MH1dIGlzIGxlc3MgdGhhbiBzaXplICsgb2Zmc2V0W3sxfV0KLWF3dC4yOEU9TGVuZ3RoIG9mIGRhdGFBcnJheSBpcyBsZXNzIHRoYW4gc2l6ZSArIG9mZnNldAotYXd0LjI4Rj1Tb3VyY2UgYW5kIGRlc3RpbmF0aW9uIHJhc3RlcnMgZG8gbm90IGhhdmUgdGhlIHNhbWUgd2lkdGghCi1hd3QuMjkwPVNvdXJjZSBhbmQgZGVzdGluYXRpb24gcmFzdGVycyBkbyBub3QgaGF2ZSB0aGUgc2FtZSBoZWlnaHQhCi1hd3QuMjkxPVNvdXJjZSBhbmQgZGVzdGluYXRpb24gaW1hZ2VzIGRvIG5vdCBoYXZlIHRoZSBzYW1lIHdpZHRoIQotYXd0LjI5Mj1Tb3VyY2UgYW5kIGRlc3RpbmF0aW9uIGltYWdlcyBkbyBub3QgaGF2ZSB0aGUgc2FtZSBoZWlnaHQhCi1hd3QuMjk0PXBpeGVsIGlzIG51bGwKLWF3dC4yOTU9ZGF0YSBpcyBudWxsCi1hd3QuMjk2PWNhbid0IGFsbG9jYXRlIG1lbW9yeSBvbiB2aWRlbyBjYXJkIHRvIGNyZWF0ZSBuZXcgZGlzcGxheSBsaXN0Ci1hd3QuMjk3PUludmFsaWQga2V5TG9jYXRpb24KLWF3dC4yOTg9ZGF0YUJ1ZmZlciBpcyB0b28gc21hbGwKLQotYXd0LmVyci4wMD1maWxlIGRpYWxvZyB7MH0gZXJyb3IhCi1hd3QuZXJyLjAxPWVycm9yOiB7MH0KLWF3dC5lcnIuMDI9R0RJUGx1cyBEcmF3RHJpdmVyU3RyaW5nIGVycm9yIHN0YXR1cyA9IHswfQotYXd0LmVyci4wMz1nZGlwRHJhd0NvbXBvc2l0ZUdseXBoVmVjdG9yOiBHRElQbHVzIERyYXdEcml2ZXJTdHJpbmcgZXJyb3Igc3RhdHVzID0gezB9Ci1hd3QuZXJyLjA0PWdkaXBEcmF3Q29tcG9zaXRlR2x5cGhWZWN0b3I6IEdESVBsdXMgRHJhd0RyaXZlclN0cmluZyBlcnJvciBzdGF0dXMgPSB7MH0KZGlmZiAtLWdpdCBhL2F3dC9yZXNvdXJjZXMvb3JnL2FwYWNoZS9oYXJtb255L2JlYW5zL2ludGVybmFscy9ubHMvbWVzc2FnZXMucHJvcGVydGllcyBiL2F3dC9yZXNvdXJjZXMvb3JnL2FwYWNoZS9oYXJtb255L2JlYW5zL2ludGVybmFscy9ubHMvbWVzc2FnZXMucHJvcGVydGllcwpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzJiMWM4Yy4uMDAwMDAwMAotLS0gYS9hd3QvcmVzb3VyY2VzL29yZy9hcGFjaGUvaGFybW9ueS9iZWFucy9pbnRlcm5hbHMvbmxzL21lc3NhZ2VzLnByb3BlcnRpZXMKKysrIC9kZXYvbnVsbApAQCAtMSwxMDMgKzAsMCBAQAotIyBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKLSMgY29udHJpYnV0b3IgbGljZW5zZSBhZ3JlZW1lbnRzLiAgU2VlIHRoZSBOT1RJQ0UgZmlsZSBkaXN0cmlidXRlZCB3aXRoCi0jIHRoaXMgd29yayBmb3IgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiByZWdhcmRpbmcgY29weXJpZ2h0IG93bmVyc2hpcC4KLSMgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKLSMgKHRoZSAiTGljZW5zZSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGgKLSMgdGhlIExpY2Vuc2UuICBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSMgIAotIyAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotIyAgCi0jICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0jICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotIyAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0jICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0jICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSMgCi0KLSMgbWVzc2FnZXMgZm9yIEVOIGxvY2FsZQotYmVhbnMuMDA9bm8gZ2V0dGVyIGZvciB7MH0gcHJvcGVydHkKLWJlYW5zLjAxPW5vIHByb3BlcnR5IGZvciBuYW1lIHswfSBpcyBmb3VuZAotYmVhbnMuMDI9aW4gRGVmYXVsdFBlcnNpc3RlbmNlRGVsZWdhdGUubXV0YXRlc1RvKCkgezB9IDogezF9Ci1iZWFucy4wMz1UYXJnZXQgQmVhbiBjbGFzcyBpcyBudWxsCi1iZWFucy4wND1iYWQgcHJvcGVydHkgbmFtZQotYmVhbnMuMDU9TW9kaWZpZXIgZm9yIHNldHRlciBtZXRob2Qgc2hvdWxkIGJlIHB1YmxpYy4KLWJlYW5zLjA2PU51bWJlciBvZiBwYXJhbWV0ZXJzIGluIHNldHRlciBtZXRob2QgaXMgbm90IGVxdWFsIHRvIDEuCi1iZWFucy4wNz1QYXJhbWV0ZXIgdHlwZSBpbiBzZXR0ZXIgbWV0aG9kIGRvZXMgbm90IGNvcnJlc3BvbmRzIHRvIHByZWRlZmluZWQuCi1iZWFucy4wOD1OdW1iZXIgb2YgcGFyYW1ldGVycyBpbiBnZXR0ZXIgbWV0aG9kIGlzIG5vdCBlcXVhbCB0byAwLgotYmVhbnMuMDk9UGFyYW1ldGVyIHR5cGUgaW4gZ2V0dGVyIG1ldGhvZCBkb2VzIG5vdCBjb3JyZXNwb25kcyB0byBwcmVkZWZpbmVkLgotYmVhbnMuMEE9TW9kaWZpZXIgZm9yIGdldHRlciBtZXRob2Qgc2hvdWxkIGJlIHB1YmxpYy4KLWJlYW5zLjBCPUV4Y2VwdGlvbiBpbiBjb21tYW5kIGV4ZWN1dGlvbgotYmVhbnMuMEM9c291cmNlIGlzIG51bGwKLWJlYW5zLjBEPUVycm9yIGluIGV4cHJlc3Npb246IHswfQotYmVhbnMuMEU9Q2hhbmdlcyBhcmUgbnVsbAotYmVhbnMuMEY9VGhlIG5ldyBCZWFuQ29udGV4dCBjYW4gbm90IGJlIHNldAotYmVhbnMuMTA9bm8gbm9kZSBpcyBmb3VuZCBmb3Igc3RhdGVtZW50IHdpdGggdGFyZ2V0ID0gezB9Ci1iZWFucy4xMT1ubyBnZXR0ZXIgZm9yIHByb3BlcnR5IHswfSBmb3VuZAotYmVhbnMuMTI9Y2Fubm90IGFjY2VzcyBwcm9wZXJ0eSB7MH0gZ2V0dGVyCi1iZWFucy4xMz1ubyBzZXR0ZXIgZm9yIHByb3BlcnR5IHswfSBmb3VuZAotYmVhbnMuMTQ9RXhjZXB0aW9uIHdoaWxlIGZpbmRpbmcgcHJvcGVydHkgZGVzY3JpcHRvcgotYmVhbnMuMTU9VGhlIGxpc3RlbmVyIGlzIG51bGwKLWJlYW5zLjE2PVRoZSBwcm92aWRlciBpcyBudWxsCi1iZWFucy4xNz1UaGUgY2hpbGQgaXMgbnVsbAotYmVhbnMuMTg9VGhlIHJlcXVlc3RvciBpcyBudWxsCi1iZWFucy4xOT1UaGUgc2VydmljZSBjbGFzcyBpcyBudWxsCi1iZWFucy4xQT1UaGUgc2VydmljZSBzZWxlY3RvciBpcyBudWxsCi1iZWFucy4xQj1UaGUgc2VydmljZSBpcyBudWxsCi1iZWFucy4xQz1UaGUgZXZlbnQgaXMgbnVsbAotYmVhbnMuMUQ9YmVhbiBpcyBudWxsCi1iZWFucy4xRT1JbGxlZ2FsIGNsYXNzIG5hbWU6IHswfQotYmVhbnMuMUY9TWV0aG9kIG5vdCBmb3VuZDogZ2V0ezB9Ci1iZWFucy4yMD1NZXRob2Qgbm90IGZvdW5kOiBzZXR7MH0KLWJlYW5zLjIxPU1vZGlmaWVyIGZvciBpbmRleGVkIGdldHRlciBtZXRob2Qgc2hvdWxkIGJlIHB1YmxpYy4KLWJlYW5zLjIyPU51bWJlciBvZiBwYXJhbWV0ZXJzIGluIGdldHRlciBtZXRob2QgaXMgbm90IGVxdWFsIHRvIDEuCi1iZWFucy4yMz1QYXJhbWV0ZXIgaW4gaW5kZXhlZCBnZXR0ZXIgbWV0aG9kIGlzIG5vdCBvZiBpbnRlZ2VyIHR5cGUuCi1iZWFucy4yND1QYXJhbWV0ZXIgdHlwZSBpbiBpbmRleGVkIGdldHRlciBtZXRob2QgZG9lcyBub3QgY29ycmVzcG9uZCB0byBwcmVkZWZpbmVkLgotYmVhbnMuMjU9TW9kaWZpZXIgZm9yIGluZGV4ZWQgc2V0dGVyIG1ldGhvZCBzaG91bGQgYmUgcHVibGljLgotYmVhbnMuMjY9TnVtYmVyIG9mIHBhcmFtZXRlcnMgaW4gaW5kZXhlZCBzZXR0ZXIgbWV0aG9kIGlzIG5vdCBlcXVhbCB0byAyLgotYmVhbnMuMjc9Rmlyc3QgcGFyYW1ldGVyIHR5cGUgaW4gaW5kZXhlZCBzZXR0ZXIgbWV0aG9kIHNob3VsZCBiZSBpbnQuCi1iZWFucy4yOD1TZWNvbmQgcGFyYW1ldGVyIHR5cGUgaW4gaW5kZXhlZCBzZXR0ZXIgbWV0aG9kIGRvZXMgbm90IGNvcnJlc3BvbmRzIHRvIHByZWRlZmluZWQuCi1iZWFucy4yOT1NZW1iZXJzaGlwIGxpc3RlbmVyIGlzIG51bGwKLWJlYW5zLjJBPVRhcmdldCBjaGlsZCBjYW4gbm90IGJlIG51bGwKLWJlYW5zLjJCPVJlc291cmNlIG5hbWUgY2FuIG5vdCBiZSBudWxsCi1iZWFucy4yQz1UaGUgY2hpbGQgY2FuIG5vdCBiZSBudWxsCi1iZWFucy4yRD1JbnZhbGlkIHJlc291cmNlCi1iZWFucy4yRT1Qcm9wZXJ0eVZldG9FeGNlcHRpb24gd2FzIHRocm93biB3aGlsZSByZW1vdmluZyBhIGNoaWxkOiB7MH07IE9yaWdpbmFsIGVycm9yIG1lc3NhZ2U6ezF9Ci1iZWFucy4yRj1UYXJnZXQgY2hpbGQgaXMgbnVsbAotYmVhbnMuMzA9UHJvcGVydHlWZXRvRXhjZXB0aW9uIHdhcyB0aHJvd24gd2hpbGUgYWRkaW5nIGEgY2hpbGQ6IHswfTsgT3JpZ2luYWwgZXJyb3IgbWVzc2FnZTp7MX0KLWJlYW5zLjMxPU5vIHZhbGlkIG1ldGhvZCB7MH0gZm9yIHsxfSBmb3VuZC4KLWJlYW5zLjMyPUNhbm5vdCBhY3F1aXJlIGV2ZW50IHR5cGUgZnJvbSB7MH0gbGlzdGVuZXIuCi1iZWFucy4zMz17MH0gZG9lcyBub3QgcmV0dXJuIDx2b2lkPgotYmVhbnMuMzQ9ezB9IHNob3VsZCBoYXZlIGEgc2luZ2xlIGlucHV0IHBhcmFtZXRlcgotYmVhbnMuMzU9U2luZ2xlIHBhcmFtZXRlciBkb2VzIG5vdCBtYXRjaCB0byB7MH0gY2xhc3MKLWJlYW5zLjM2PU5vIGlucHV0IHBhcmFtcyBhcmUgYWxsb3dlZCBmb3IgZ2V0TGlzdGVuZXJNZXRob2QKLWJlYW5zLjM3PVJldHVybiB0eXBlIG9mIGdldExpc3RlbmVyTWV0aG9kIGlzIG5vdCBhbiBhcnJheSBvZiBsaXN0ZW5lcnMKLWJlYW5zLjM4PUFkZCBhbmQgcmVtb3ZlIG1ldGhvZHMgYXJlIG5vdCBhdmFpbGFibGUKLWJlYW5zLjM5PUNhbm5vdCBnZW5lcmF0ZSBldmVudCBzZXQgZGVzY3JpcHRvciBmb3IgbmFtZSB7MH0uCi1iZWFucy4zQT1FdmVudCB0eXBlIHdpdGggbmFtZSB7MH0gaXMgbm90IGZvdW5kLgotYmVhbnMuM0I9c2tpcHBpbmcgZXhwcmVzc2lvbiB7MH0uLi4KLWJlYW5zLjNDPVVua25vd24gbWV0aG9kIG5hbWUgZm9yIGFycmF5Ci1iZWFucy4zRD1GaXJzdCBwYXJhbWV0ZXIgaW4gYXJyYXkgZ2V0dGVyKHNldHRlcikgaXMgbm90IG9mIEludGVnZXIgdHlwZQotYmVhbnMuM0U9SWxsZWdhbCBudW1iZXIgb2YgYXJndW1lbnRzIGluIGFycmF5IGdldHRlcgotYmVhbnMuM0Y9SWxsZWdhbCBudW1iZXIgb2YgYXJndW1lbnRzIGluIGFycmF5IHNldHRlcgotYmVhbnMuNDA9Tm8gY29uc3RydWN0b3IgZm9yIGNsYXNzIHswfSBmb3VuZAotYmVhbnMuNDE9Tm8gbWV0aG9kIHdpdGggbmFtZSB7MH0gaXMgZm91bmQKLWJlYW5zLjQyPXRhcmdldCBpcyBub3QgZ2VuZXJhdGVkOiBjbGFzc25hbWUgezB9IGlzIG5vdCBmb3VuZAotYmVhbnMuNDM9Q2Fubm90IGNvbnZlcnQgezB9IHRvIGNoYXIKLWJlYW5zLjQ0PWZvciBwcm9wZXJ0eSB7MH0gbm8gZ2V0dGVyKHNldHRlcikgaXMgZm91bmQKLWJlYW5zLjQ1PW1ldGhvZCBuYW1lIGlzIG5vdCBnZW5lcmF0ZWQ6IGVycm9yIGluIGdldE1ldGhvZE5hbWUoKQotYmVhbnMuNDY9Tm90IGEgdmFsaWQgY2hpbGQKLWJlYW5zLjQ3PVVuYWJsZSB0byBpbnN0YW50aWF0ZSBwcm9wZXJ0eSBlZGl0b3IKLWJlYW5zLjQ4PVByb3BlcnR5IGVkaXRvciBpcyBub3QgYXNzaWduYWJsZSBmcm9tIHRoZSBQcm9wZXJ0eUVkaXRvciBpbnRlcmZhY2UKLWJlYW5zLjQ5PUNoaWxkIGNhbm5vdCBpbXBsZW1lbnQgYm90aCBCZWFuQ29udGV4dENoaWxkIGFuZCBCZWFuQ29udGV4dFByb3h5Ci1iZWFucy40QT1uZXdJbnN0YW5jZSBpcyBudWxsCi1iZWFucy40Qj10eXBlIGlzIG51bGwKLWJlYW5zLjRDPWVuY29kZXIgaXMgbnVsbAotYmVhbnMuNEQ9SW52YWxpZCBtZXRob2QgY2FsbAotYmVhbnMuNEU9c3RvcENsYXNzIGlzIG5vdCBhbmNlc3RvciBvZiBiZWFuQ2xhc3MKLWJlYW5zLjRGPXNlYXJjaCBwYXRoIGlzIG51bGwKLWJlYW5zLjUwPW5vdCBhbiBpbmRleGVkIHByb3BlcnR5Ci1iZWFucy41MT1MaXN0ZW5lciBtZXRob2QgezB9IHNob3VsZCBoYXZlIHBhcmFtZXRlciBvZiB0eXBlIHsxfQotYmVhbnMuNTI9bGlzdGVuZXJNZXRob2ROYW1lKHMpIGlzIG51bGwKLWJlYW5zLjUzPWV2ZW50U2V0TmFtZSBpcyBudWxsCi1iZWFucy41ND1saXN0ZW5lclR5cGUgaXMgbnVsbAotYmVhbnMuNTU9TWV0aG9kIGlzIG51bGwKZGlmZiAtLWdpdCBhL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0FuZHJvaWQubWsgYi9jYW1lcmEvbGliY2FtZXJhc2VydmljZS9BbmRyb2lkLm1rCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyZGZlNjU5Li4wMDAwMDAwCi0tLSBhL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0FuZHJvaWQubWsKKysrIC9kZXYvbnVsbApAQCAtMSw1OSArMCwwIEBACi1MT0NBTF9QQVRIOj0gJChjYWxsIG15LWRpcikKLQotIwotIyBTZXQgVVNFX0NBTUVSQV9TVFVCIGZvciBub24tZW11bGF0b3IgYW5kIG5vbi1zaW11bGF0b3IgYnVpbGRzLCBpZiB5b3Ugd2FudAotIyB0aGUgY2FtZXJhIHNlcnZpY2UgdG8gdXNlIHRoZSBmYWtlIGNhbWVyYS4gIEZvciBlbXVsYXRvciBvciBzaW11bGF0b3IgYnVpbGRzLAotIyB3ZSBhbHdheXMgdXNlIHRoZSBmYWtlIGNhbWVyYS4KLQotaWZlcSAoJChVU0VfQ0FNRVJBX1NUVUIpLCkKLVVTRV9DQU1FUkFfU1RVQjo9ZmFsc2UKLWlmbmVxICgkKGZpbHRlciBzb29uZXIgZ2VuZXJpYyBzaW0sJChUQVJHRVRfREVWSUNFKSksKQotVVNFX0NBTUVSQV9TVFVCOj10cnVlCi1lbmRpZiAjbGliY2FtZXJhc3R1YgotZW5kaWYKLQotaWZlcSAoJChVU0VfQ0FNRVJBX1NUVUIpLHRydWUpCi0jCi0jIGxpYmNhbWVyYXN0dWIKLSMKLQotaW5jbHVkZSAkKENMRUFSX1ZBUlMpCi0KLUxPQ0FMX1NSQ19GSUxFUzo9ICAgICAgICAgICAgICAgXAotICAgIENhbWVyYUhhcmR3YXJlU3R1Yi5jcHAgICAgICBcCi0gICAgRmFrZUNhbWVyYS5jcHAKLQotTE9DQUxfTU9EVUxFOj0gbGliY2FtZXJhc3R1YgotCi1MT0NBTF9TSEFSRURfTElCUkFSSUVTOj0gbGlidWkKLQotaW5jbHVkZSAkKEJVSUxEX1NUQVRJQ19MSUJSQVJZKQotZW5kaWYgIyBVU0VfQ0FNRVJBX1NUVUIKLQotIwotIyBsaWJjYW1lcmFzZXJ2aWNlCi0jCi0KLWluY2x1ZGUgJChDTEVBUl9WQVJTKQotCi1MT0NBTF9TUkNfRklMRVM6PSAgICAgICAgICAgICAgIFwKLSAgICBDYW1lcmFTZXJ2aWNlLmNwcAotCi1MT0NBTF9TSEFSRURfTElCUkFSSUVTOj0gXAotICAgIGxpYnVpIFwKLSAgICBsaWJ1dGlscyBcCi0gICAgbGliY3V0aWxzCi0KLUxPQ0FMX01PRFVMRTo9IGxpYmNhbWVyYXNlcnZpY2UKLQotTE9DQUxfQ0ZMQUdTKz0tRExPR19UQUc9XCJDYW1lcmFTZXJ2aWNlXCIKLQotaWZlcSAoJChVU0VfQ0FNRVJBX1NUVUIpLCB0cnVlKQotTE9DQUxfU1RBVElDX0xJQlJBUklFUyArPSBsaWJjYW1lcmFzdHViCi1MT0NBTF9DRkxBR1MgKz0gLWluY2x1ZGUgQ2FtZXJhSGFyZHdhcmVTdHViLmgKLWVsc2UKLUxPQ0FMX1NIQVJFRF9MSUJSQVJJRVMgKz0gbGliY2FtZXJhIAotZW5kaWYKLQotaW5jbHVkZSAkKEJVSUxEX1NIQVJFRF9MSUJSQVJZKQotCmRpZmYgLS1naXQgYS9jYW1lcmEvbGliY2FtZXJhc2VydmljZS9DYW1lcmFIYXJkd2FyZVN0dWIuY3BwIGIvY2FtZXJhL2xpYmNhbWVyYXNlcnZpY2UvQ2FtZXJhSGFyZHdhcmVTdHViLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOWE0NzcwNS4uMDAwMDAwMAotLS0gYS9jYW1lcmEvbGliY2FtZXJhc2VydmljZS9DYW1lcmFIYXJkd2FyZVN0dWIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMzc5ICswLDAgQEAKLS8qCi0qKgotKiogQ29weXJpZ2h0IDIwMDgsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNkZWZpbmUgTE9HX1RBRyAiQ2FtZXJhSGFyZHdhcmVTdHViIgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSAiQ2FtZXJhSGFyZHdhcmVTdHViLmgiCi0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotI2luY2x1ZGUgPGZjbnRsLmg+Ci0jaW5jbHVkZSA8c3lzL21tYW4uaD4KLQotI2luY2x1ZGUgIkNhbm5lZEpwZWcuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1DYW1lcmFIYXJkd2FyZVN0dWI6OkNhbWVyYUhhcmR3YXJlU3R1YigpCi0gICAgICAgICAgICAgICAgICA6IG1QYXJhbWV0ZXJzKCksCi0gICAgICAgICAgICAgICAgICAgIG1IZWFwKDApLAotICAgICAgICAgICAgICAgICAgICBtRmFrZUNhbWVyYSgwKSwKLSAgICAgICAgICAgICAgICAgICAgbVByZXZpZXdGcmFtZVNpemUoMCksCi0gICAgICAgICAgICAgICAgICAgIG1SYXdQaWN0dXJlQ2FsbGJhY2soMCksCi0gICAgICAgICAgICAgICAgICAgIG1KcGVnUGljdHVyZUNhbGxiYWNrKDApLAotICAgICAgICAgICAgICAgICAgICBtUGljdHVyZUNhbGxiYWNrQ29va2llKDApLAotICAgICAgICAgICAgICAgICAgICBtUHJldmlld0NhbGxiYWNrKDApLAotICAgICAgICAgICAgICAgICAgICBtUHJldmlld0NhbGxiYWNrQ29va2llKDApLAotICAgICAgICAgICAgICAgICAgICBtQXV0b0ZvY3VzQ2FsbGJhY2soMCksCi0gICAgICAgICAgICAgICAgICAgIG1BdXRvRm9jdXNDYWxsYmFja0Nvb2tpZSgwKSwKLSAgICAgICAgICAgICAgICAgICAgbUN1cnJlbnRQcmV2aWV3RnJhbWUoMCkKLXsKLSAgICBpbml0RGVmYXVsdFBhcmFtZXRlcnMoKTsKLX0KLQotdm9pZCBDYW1lcmFIYXJkd2FyZVN0dWI6OmluaXREZWZhdWx0UGFyYW1ldGVycygpCi17Ci0gICAgQ2FtZXJhUGFyYW1ldGVycyBwOwotCi0gICAgcC5zZXRQcmV2aWV3U2l6ZSgxNzYsIDE0NCk7Ci0gICAgcC5zZXRQcmV2aWV3RnJhbWVSYXRlKDE1KTsKLSAgICBwLnNldFByZXZpZXdGb3JtYXQoInl1djQyMnNwIik7Ci0KLSAgICBwLnNldFBpY3R1cmVTaXplKGtDYW5uZWRKcGVnV2lkdGgsIGtDYW5uZWRKcGVnSGVpZ2h0KTsKLSAgICBwLnNldFBpY3R1cmVGb3JtYXQoImpwZWciKTsKLQotICAgIGlmIChzZXRQYXJhbWV0ZXJzKHApICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgIExPR0UoIkZhaWxlZCB0byBzZXQgZGVmYXVsdCBwYXJhbWV0ZXJzPyEiKTsKLSAgICB9IAotfQotCi12b2lkIENhbWVyYUhhcmR3YXJlU3R1Yjo6aW5pdEhlYXBMb2NrZWQoKQotewotICAgIGludCB3aWR0aCwgaGVpZ2h0OwotICAgIG1QYXJhbWV0ZXJzLmdldFByZXZpZXdTaXplKCZ3aWR0aCwgJmhlaWdodCk7Ci0KLSAgICBMT0dEKCJpbml0SGVhcExvY2tlZDogcHJldmlldyBzaXplPSVkeCVkIiwgd2lkdGgsIGhlaWdodCk7Ci0KLSAgICAvLyBOb3RlIHRoYXQgd2UgZW5mb3JjZSB5dXY0MjIgaW4gc2V0UGFyYW1ldGVycygpLgotICAgIGludCBob3dfYmlnID0gd2lkdGggKiBoZWlnaHQgKiAyOwotCi0gICAgLy8gSWYgd2UgYXJlIGJlaW5nIHJlaW5pdGlhbGl6ZWQgdG8gdGhlIHNhbWUgc2l6ZSBhcyBiZWZvcmUsIG5vCi0gICAgLy8gd29yayBuZWVkcyB0byBiZSBkb25lLgotICAgIGlmIChob3dfYmlnID09IG1QcmV2aWV3RnJhbWVTaXplKQotICAgICAgICByZXR1cm47Ci0KLSAgICBtUHJldmlld0ZyYW1lU2l6ZSA9IGhvd19iaWc7Ci0KLSAgICAvLyBNYWtlIGEgbmV3IG1tYXAnZWQgaGVhcCB0aGF0IGNhbiBiZSBzaGFyZWQgYWNyb3NzIHByb2Nlc3Nlcy4gCi0gICAgLy8gdXNlIGNvZGUgYmVsb3cgdG8gdGVzdCB3aXRoIHBtZW0KLSAgICBtSGVhcCA9IG5ldyBNZW1vcnlIZWFwQmFzZShtUHJldmlld0ZyYW1lU2l6ZSAqIGtCdWZmZXJDb3VudCk7Ci0gICAgLy8gTWFrZSBhbiBJTWVtb3J5IGZvciBlYWNoIGZyYW1lIHNvIHRoYXQgd2UgY2FuIHJldXNlIHRoZW0gaW4gY2FsbGJhY2tzLgotICAgIGZvciAoaW50IGkgPSAwOyBpIDwga0J1ZmZlckNvdW50OyBpKyspIHsKLSAgICAgICAgbUJ1ZmZlcnNbaV0gPSBuZXcgTWVtb3J5QmFzZShtSGVhcCwgaSAqIG1QcmV2aWV3RnJhbWVTaXplLCBtUHJldmlld0ZyYW1lU2l6ZSk7Ci0gICAgfQotCi0gICAgLy8gUmVjcmVhdGUgdGhlIGZha2UgY2FtZXJhIHRvIHJlZmxlY3QgdGhlIGN1cnJlbnQgc2l6ZS4KLSAgICBkZWxldGUgbUZha2VDYW1lcmE7Ci0gICAgbUZha2VDYW1lcmEgPSBuZXcgRmFrZUNhbWVyYSh3aWR0aCwgaGVpZ2h0KTsKLX0KLQotQ2FtZXJhSGFyZHdhcmVTdHViOjp+Q2FtZXJhSGFyZHdhcmVTdHViKCkKLXsKLSAgICBkZWxldGUgbUZha2VDYW1lcmE7Ci0gICAgbUZha2VDYW1lcmEgPSAwOyAvLyBwYXJhbm9pYQotICAgIHNpbmdsZXRvbi5jbGVhcigpOwotfQotCi1zcDxJTWVtb3J5SGVhcD4gQ2FtZXJhSGFyZHdhcmVTdHViOjpnZXRQcmV2aWV3SGVhcCgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1IZWFwOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotaW50IENhbWVyYUhhcmR3YXJlU3R1Yjo6cHJldmlld1RocmVhZCgpCi17Ci0gICAgbUxvY2subG9jaygpOwotICAgICAgICAvLyB0aGUgYXR0cmlidXRlcyBiZWxvdyBjYW4gY2hhbmdlIHVuZGVyIG91ciBmZWV0Li4uCi0KLSAgICAgICAgaW50IHByZXZpZXdGcmFtZVJhdGUgPSBtUGFyYW1ldGVycy5nZXRQcmV2aWV3RnJhbWVSYXRlKCk7Ci0KLSAgICAgICAgLy8gRmluZCB0aGUgb2Zmc2V0IHdpdGhpbiB0aGUgaGVhcCBvZiB0aGUgY3VycmVudCBidWZmZXIuCi0gICAgICAgIHNzaXplX3Qgb2Zmc2V0ID0gbUN1cnJlbnRQcmV2aWV3RnJhbWUgKiBtUHJldmlld0ZyYW1lU2l6ZTsKLQotICAgICAgICBzcDxNZW1vcnlIZWFwQmFzZT4gaGVhcCA9IG1IZWFwOwotICAgIAotICAgICAgICAvLyB0aGlzIGFzc3VtZXMgdGhlIGludGVybmFsIHN0YXRlIG9mIGZha2UgY2FtZXJhIGRvZXNuJ3QgY2hhbmdlCi0gICAgICAgIC8vIChvciBpcyB0aHJlYWQgc2FmZSkKLSAgICAgICAgRmFrZUNhbWVyYSogZmFrZUNhbWVyYSA9IG1GYWtlQ2FtZXJhOwotICAgICAgICAKLSAgICAgICAgc3A8TWVtb3J5QmFzZT4gYnVmZmVyID0gbUJ1ZmZlcnNbbUN1cnJlbnRQcmV2aWV3RnJhbWVdOwotICAgICAgICAKLSAgICBtTG9jay51bmxvY2soKTsKLQotICAgIC8vIFRPRE86IGhlcmUgY2hlY2sgYWxsIHRoZSBjb25kaXRpb25zIHRoYXQgY291bGQgZ28gd3JvbmcKLSAgICBpZiAoYnVmZmVyICE9IDApIHsKLSAgICAgICAgLy8gQ2FsY3VsYXRlIGhvdyBsb25nIHRvIHdhaXQgYmV0d2VlbiBmcmFtZXMuCi0gICAgICAgIGludCBkZWxheSA9IChpbnQpKDEwMDAwMDAuMGYgLyBmbG9hdChwcmV2aWV3RnJhbWVSYXRlKSk7Ci0gICAgCi0gICAgICAgIC8vIFRoaXMgaXMgYWx3YXlzIHZhbGlkLCBldmVuIGlmIHRoZSBjbGllbnQgZGllZCAtLSB0aGUgbWVtb3J5Ci0gICAgICAgIC8vIGlzIHN0aWxsIG1hcHBlZCBpbiBvdXIgcHJvY2Vzcy4KLSAgICAgICAgdm9pZCAqYmFzZSA9IGhlYXAtPmJhc2UoKTsKLSAgICAKLSAgICAgICAgLy8gRmlsbCB0aGUgY3VycmVudCBmcmFtZSB3aXRoIHRoZSBmYWtlIGNhbWVyYS4KLSAgICAgICAgdWludDhfdCAqZnJhbWUgPSAoKHVpbnQ4X3QgKiliYXNlKSArIG9mZnNldDsKLSAgICAgICAgZmFrZUNhbWVyYS0+Z2V0TmV4dEZyYW1lQXNZdXY0MjIoZnJhbWUpOwotICAgIAotICAgICAgICAvL0xPR1YoInByZXZpZXdUaHJlYWQ6IGdlbmVyYXRlZCBmcmFtZSB0byBidWZmZXIgJWQiLCBtQ3VycmVudFByZXZpZXdGcmFtZSk7Ci0gICAgICAgIAotICAgICAgICAvLyBOb3RpZnkgdGhlIGNsaWVudCBvZiBhIG5ldyBmcmFtZS4KLSAgICAgICAgbVByZXZpZXdDYWxsYmFjayhidWZmZXIsIG1QcmV2aWV3Q2FsbGJhY2tDb29raWUpOwotICAgIAotICAgICAgICAvLyBBZHZhbmNlIHRoZSBidWZmZXIgcG9pbnRlci4KLSAgICAgICAgbUN1cnJlbnRQcmV2aWV3RnJhbWUgPSAobUN1cnJlbnRQcmV2aWV3RnJhbWUgKyAxKSAlIGtCdWZmZXJDb3VudDsKLQotICAgICAgICAvLyBXYWl0IGZvciBpdC4uLgotICAgICAgICB1c2xlZXAoZGVsYXkpOwotICAgIH0KLQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQ2FtZXJhSGFyZHdhcmVTdHViOjpzdGFydFByZXZpZXcocHJldmlld19jYWxsYmFjayBjYiwgdm9pZCogdXNlcikKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgbG9jayhtTG9jayk7Ci0gICAgaWYgKG1QcmV2aWV3VGhyZWFkICE9IDApIHsKLSAgICAgICAgLy8gYWxyZWFkeSBydW5uaW5nCi0gICAgICAgIHJldHVybiBJTlZBTElEX09QRVJBVElPTjsKLSAgICB9Ci0gICAgbVByZXZpZXdDYWxsYmFjayA9IGNiOwotICAgIG1QcmV2aWV3Q2FsbGJhY2tDb29raWUgPSB1c2VyOwotICAgIG1QcmV2aWV3VGhyZWFkID0gbmV3IFByZXZpZXdUaHJlYWQodGhpcyk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi12b2lkIENhbWVyYUhhcmR3YXJlU3R1Yjo6c3RvcFByZXZpZXcoKQotewotICAgIHNwPFByZXZpZXdUaHJlYWQ+IHByZXZpZXdUaHJlYWQ7Ci0gICAgCi0gICAgeyAvLyBzY29wZSBmb3IgdGhlIGxvY2sKLSAgICAgICAgTXV0ZXg6OkF1dG9sb2NrIGxvY2sobUxvY2spOwotICAgICAgICBwcmV2aWV3VGhyZWFkID0gbVByZXZpZXdUaHJlYWQ7Ci0gICAgfQotCi0gICAgLy8gZG9uJ3QgaG9sZCB0aGUgbG9jayB3aGlsZSB3YWl0aW5nIGZvciB0aGUgdGhyZWFkIHRvIHF1aXQKLSAgICBpZiAocHJldmlld1RocmVhZCAhPSAwKSB7Ci0gICAgICAgIHByZXZpZXdUaHJlYWQtPnJlcXVlc3RFeGl0QW5kV2FpdCgpOwotICAgIH0KLQotICAgIE11dGV4OjpBdXRvbG9jayBsb2NrKG1Mb2NrKTsKLSAgICBtUHJldmlld1RocmVhZC5jbGVhcigpOwotfQotCi1ib29sIENhbWVyYUhhcmR3YXJlU3R1Yjo6cHJldmlld0VuYWJsZWQoKSB7Ci0gICAgcmV0dXJuIG1QcmV2aWV3VGhyZWFkICE9IDA7Ci19Ci0KLXN0YXR1c190IENhbWVyYUhhcmR3YXJlU3R1Yjo6c3RhcnRSZWNvcmRpbmcocmVjb3JkaW5nX2NhbGxiYWNrIGNiLCB2b2lkKiB1c2VyKQotewotICAgIHJldHVybiBVTktOT1dOX0VSUk9SOwotfQotCi12b2lkIENhbWVyYUhhcmR3YXJlU3R1Yjo6c3RvcFJlY29yZGluZygpCi17Ci19Ci0KLWJvb2wgQ2FtZXJhSGFyZHdhcmVTdHViOjpyZWNvcmRpbmdFbmFibGVkKCkKLXsKLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0KLXZvaWQgQ2FtZXJhSGFyZHdhcmVTdHViOjpyZWxlYXNlUmVjb3JkaW5nRnJhbWUoY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSkKLXsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWludCBDYW1lcmFIYXJkd2FyZVN0dWI6OmJlZ2luQXV0b0ZvY3VzVGhyZWFkKHZvaWQgKmNvb2tpZSkKLXsKLSAgICBDYW1lcmFIYXJkd2FyZVN0dWIgKmMgPSAoQ2FtZXJhSGFyZHdhcmVTdHViICopY29va2llOwotICAgIHJldHVybiBjLT5hdXRvRm9jdXNUaHJlYWQoKTsKLX0KLQotaW50IENhbWVyYUhhcmR3YXJlU3R1Yjo6YXV0b0ZvY3VzVGhyZWFkKCkKLXsKLSAgICBpZiAobUF1dG9Gb2N1c0NhbGxiYWNrICE9IE5VTEwpIHsKLSAgICAgICAgbUF1dG9Gb2N1c0NhbGxiYWNrKHRydWUsIG1BdXRvRm9jdXNDYWxsYmFja0Nvb2tpZSk7Ci0gICAgICAgIG1BdXRvRm9jdXNDYWxsYmFjayA9IE5VTEw7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci19Ci0KLXN0YXR1c190IENhbWVyYUhhcmR3YXJlU3R1Yjo6YXV0b0ZvY3VzKGF1dG9mb2N1c19jYWxsYmFjayBhZl9jYiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQgKnVzZXIpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIGxvY2sobUxvY2spOwotCi0gICAgaWYgKG1BdXRvRm9jdXNDYWxsYmFjayAhPSBOVUxMKSB7Ci0gICAgICAgIHJldHVybiBtQXV0b0ZvY3VzQ2FsbGJhY2sgPT0gYWZfY2IgPyBOT19FUlJPUiA6IElOVkFMSURfT1BFUkFUSU9OOwotICAgIH0KLQotICAgIG1BdXRvRm9jdXNDYWxsYmFjayA9IGFmX2NiOwotICAgIG1BdXRvRm9jdXNDYWxsYmFja0Nvb2tpZSA9IHVzZXI7Ci0gICAgaWYgKGNyZWF0ZVRocmVhZChiZWdpbkF1dG9Gb2N1c1RocmVhZCwgdGhpcykgPT0gZmFsc2UpCi0gICAgICAgIHJldHVybiBVTktOT1dOX0VSUk9SOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLypzdGF0aWMqLyBpbnQgQ2FtZXJhSGFyZHdhcmVTdHViOjpiZWdpblBpY3R1cmVUaHJlYWQodm9pZCAqY29va2llKQotewotICAgIENhbWVyYUhhcmR3YXJlU3R1YiAqYyA9IChDYW1lcmFIYXJkd2FyZVN0dWIgKiljb29raWU7Ci0gICAgcmV0dXJuIGMtPnBpY3R1cmVUaHJlYWQoKTsKLX0KLQotaW50IENhbWVyYUhhcmR3YXJlU3R1Yjo6cGljdHVyZVRocmVhZCgpCi17Ci0gICAgaWYgKG1TaHV0dGVyQ2FsbGJhY2spCi0gICAgICAgIG1TaHV0dGVyQ2FsbGJhY2sobVBpY3R1cmVDYWxsYmFja0Nvb2tpZSk7Ci0KLSAgICBpZiAobVJhd1BpY3R1cmVDYWxsYmFjaykgewotICAgICAgICAvL0ZJWE1FOiB1c2UgYSBjYW5uZWQgWVVWIGltYWdlIQotICAgICAgICAvLyBJbiB0aGUgbWVhbnRpbWUganVzdCBtYWtlIGFub3RoZXIgZmFrZSBjYW1lcmEgcGljdHVyZS4KLSAgICAgICAgaW50IHcsIGg7Ci0gICAgICAgIG1QYXJhbWV0ZXJzLmdldFBpY3R1cmVTaXplKCZ3LCAmaCk7Ci0gICAgICAgIHNwPE1lbW9yeUhlYXBCYXNlPiBoZWFwID0gbmV3IE1lbW9yeUhlYXBCYXNlKHcgKiAyICogaCk7Ci0gICAgICAgIHNwPE1lbW9yeUJhc2U+IG1lbSA9IG5ldyBNZW1vcnlCYXNlKGhlYXAsIDAsIHcgKiAyICogaCk7Ci0gICAgICAgIEZha2VDYW1lcmEgY2FtKHcsIGgpOwotICAgICAgICBjYW0uZ2V0TmV4dEZyYW1lQXNZdXY0MjIoKHVpbnQ4X3QgKiloZWFwLT5iYXNlKCkpOwotICAgICAgICBpZiAobVJhd1BpY3R1cmVDYWxsYmFjaykKLSAgICAgICAgICAgIG1SYXdQaWN0dXJlQ2FsbGJhY2sobWVtLCBtUGljdHVyZUNhbGxiYWNrQ29va2llKTsKLSAgICB9Ci0KLSAgICBpZiAobUpwZWdQaWN0dXJlQ2FsbGJhY2spIHsKLSAgICAgICAgc3A8TWVtb3J5SGVhcEJhc2U+IGhlYXAgPSBuZXcgTWVtb3J5SGVhcEJhc2Uoa0Nhbm5lZEpwZWdTaXplKTsKLSAgICAgICAgc3A8TWVtb3J5QmFzZT4gbWVtID0gbmV3IE1lbW9yeUJhc2UoaGVhcCwgMCwga0Nhbm5lZEpwZWdTaXplKTsKLSAgICAgICAgbWVtY3B5KGhlYXAtPmJhc2UoKSwga0Nhbm5lZEpwZWcsIGtDYW5uZWRKcGVnU2l6ZSk7Ci0gICAgICAgIGlmIChtSnBlZ1BpY3R1cmVDYWxsYmFjaykKLSAgICAgICAgICAgIG1KcGVnUGljdHVyZUNhbGxiYWNrKG1lbSwgbVBpY3R1cmVDYWxsYmFja0Nvb2tpZSk7Ci0gICAgfQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQ2FtZXJhSGFyZHdhcmVTdHViOjp0YWtlUGljdHVyZShzaHV0dGVyX2NhbGxiYWNrIHNodXR0ZXJfY2IsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhd19jYWxsYmFjayByYXdfY2IsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGpwZWdfY2FsbGJhY2sganBlZ19jYiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCogdXNlcikKLXsKLSAgICBzdG9wUHJldmlldygpOwotICAgIG1TaHV0dGVyQ2FsbGJhY2sgPSBzaHV0dGVyX2NiOwotICAgIG1SYXdQaWN0dXJlQ2FsbGJhY2sgPSByYXdfY2I7Ci0gICAgbUpwZWdQaWN0dXJlQ2FsbGJhY2sgPSBqcGVnX2NiOwotICAgIG1QaWN0dXJlQ2FsbGJhY2tDb29raWUgPSB1c2VyOwotICAgIGlmIChjcmVhdGVUaHJlYWQoYmVnaW5QaWN0dXJlVGhyZWFkLCB0aGlzKSA9PSBmYWxzZSkKLSAgICAgICAgcmV0dXJuIC0xOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQ2FtZXJhSGFyZHdhcmVTdHViOjpjYW5jZWxQaWN0dXJlKGJvb2wgY2FuY2VsX3NodXR0ZXIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYm9vbCBjYW5jZWxfcmF3LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2wgY2FuY2VsX2pwZWcpCi17Ci0gICAgaWYgKGNhbmNlbF9zaHV0dGVyKSBtU2h1dHRlckNhbGxiYWNrID0gTlVMTDsKLSAgICBpZiAoY2FuY2VsX3JhdykgbVJhd1BpY3R1cmVDYWxsYmFjayA9IE5VTEw7Ci0gICAgaWYgKGNhbmNlbF9qcGVnKSBtSnBlZ1BpY3R1cmVDYWxsYmFjayA9IE5VTEw7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBDYW1lcmFIYXJkd2FyZVN0dWI6OmR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKSBjb25zdAotewotICAgIGNvbnN0IHNpemVfdCBTSVpFID0gMjU2OwotICAgIGNoYXIgYnVmZmVyW1NJWkVdOwotICAgIFN0cmluZzggcmVzdWx0OwotICAgIEF1dG9NdXRleCBsb2NrKCZtTG9jayk7Ci0gICAgaWYgKG1GYWtlQ2FtZXJhICE9IDApIHsKLSAgICAgICAgbUZha2VDYW1lcmEtPmR1bXAoZmQsIGFyZ3MpOwotICAgICAgICBtUGFyYW1ldGVycy5kdW1wKGZkLCBhcmdzKTsKLSAgICAgICAgc25wcmludGYoYnVmZmVyLCAyNTUsICIgcHJldmlldyBmcmFtZSglZCksIHNpemUgKCVkKSwgcnVubmluZyglcylcbiIsIG1DdXJyZW50UHJldmlld0ZyYW1lLCBtUHJldmlld0ZyYW1lU2l6ZSwgbVByZXZpZXdSdW5uaW5nPyJ0cnVlIjogImZhbHNlIik7Ci0gICAgICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICB9IGVsc2UgewotICAgICAgICByZXN1bHQuYXBwZW5kKCJObyBjYW1lcmEgY2xpZW50IHlldC5cbiIpOwotICAgIH0KLSAgICB3cml0ZShmZCwgcmVzdWx0LnN0cmluZygpLCByZXN1bHQuc2l6ZSgpKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IENhbWVyYUhhcmR3YXJlU3R1Yjo6c2V0UGFyYW1ldGVycyhjb25zdCBDYW1lcmFQYXJhbWV0ZXJzJiBwYXJhbXMpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIGxvY2sobUxvY2spOwotICAgIC8vIFhYWCB2ZXJpZnkgcGFyYW1zCi0KLSAgICBpZiAoc3RyY21wKHBhcmFtcy5nZXRQcmV2aWV3Rm9ybWF0KCksICJ5dXY0MjJzcCIpICE9IDApIHsKLSAgICAgICAgTE9HRSgiT25seSB5dXY0MjJzcCBwcmV2aWV3IGlzIHN1cHBvcnRlZCIpOwotICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotCi0gICAgaWYgKHN0cmNtcChwYXJhbXMuZ2V0UGljdHVyZUZvcm1hdCgpLCAianBlZyIpICE9IDApIHsKLSAgICAgICAgTE9HRSgiT25seSBqcGVnIHN0aWxsIHBpY3R1cmVzIGFyZSBzdXBwb3J0ZWQiKTsKLSAgICAgICAgcmV0dXJuIC0xOwotICAgIH0KLQotICAgIGludCB3LCBoOwotICAgIHBhcmFtcy5nZXRQaWN0dXJlU2l6ZSgmdywgJmgpOwotICAgIGlmICh3ICE9IGtDYW5uZWRKcGVnV2lkdGggJiYgaCAhPSBrQ2FubmVkSnBlZ0hlaWdodCkgewotICAgICAgICBMT0dFKCJTdGlsbCBwaWN0dXJlIHNpemUgbXVzdCBiZSBzaXplIG9mIGNhbm5lZCBKUEVHICglZHglZCkiLAotICAgICAgICAgICAgIGtDYW5uZWRKcGVnV2lkdGgsIGtDYW5uZWRKcGVnSGVpZ2h0KTsKLSAgICAgICAgcmV0dXJuIC0xOwotICAgIH0KLQotICAgIG1QYXJhbWV0ZXJzID0gcGFyYW1zOwotCi0gICAgaW5pdEhlYXBMb2NrZWQoKTsKLQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotQ2FtZXJhUGFyYW1ldGVycyBDYW1lcmFIYXJkd2FyZVN0dWI6OmdldFBhcmFtZXRlcnMoKSBjb25zdAotewotICAgIE11dGV4OjpBdXRvbG9jayBsb2NrKG1Mb2NrKTsKLSAgICByZXR1cm4gbVBhcmFtZXRlcnM7Ci19Ci0KLXZvaWQgQ2FtZXJhSGFyZHdhcmVTdHViOjpyZWxlYXNlKCkKLXsKLX0KLQotd3A8Q2FtZXJhSGFyZHdhcmVJbnRlcmZhY2U+IENhbWVyYUhhcmR3YXJlU3R1Yjo6c2luZ2xldG9uOwotCi1zcDxDYW1lcmFIYXJkd2FyZUludGVyZmFjZT4gQ2FtZXJhSGFyZHdhcmVTdHViOjpjcmVhdGVJbnN0YW5jZSgpCi17Ci0gICAgaWYgKHNpbmdsZXRvbiAhPSAwKSB7Ci0gICAgICAgIHNwPENhbWVyYUhhcmR3YXJlSW50ZXJmYWNlPiBoYXJkd2FyZSA9IHNpbmdsZXRvbi5wcm9tb3RlKCk7Ci0gICAgICAgIGlmIChoYXJkd2FyZSAhPSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gaGFyZHdhcmU7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgc3A8Q2FtZXJhSGFyZHdhcmVJbnRlcmZhY2U+IGhhcmR3YXJlKG5ldyBDYW1lcmFIYXJkd2FyZVN0dWIoKSk7Ci0gICAgc2luZ2xldG9uID0gaGFyZHdhcmU7Ci0gICAgcmV0dXJuIGhhcmR3YXJlOwotfQotCi1leHRlcm4gIkMiIHNwPENhbWVyYUhhcmR3YXJlSW50ZXJmYWNlPiBvcGVuQ2FtZXJhSGFyZHdhcmUoKQotewotICAgIHJldHVybiBDYW1lcmFIYXJkd2FyZVN0dWI6OmNyZWF0ZUluc3RhbmNlKCk7Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9jYW1lcmEvbGliY2FtZXJhc2VydmljZS9DYW1lcmFIYXJkd2FyZVN0dWIuaCBiL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0NhbWVyYUhhcmR3YXJlU3R1Yi5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjZGQ2MDExLi4wMDAwMDAwCi0tLSBhL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0NhbWVyYUhhcmR3YXJlU3R1Yi5oCisrKyAvZGV2L251bGwKQEAgLTEsMTIyICswLDAgQEAKLS8qCi0qKgotKiogQ29weXJpZ2h0IDIwMDgsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9IQVJEV0FSRV9DQU1FUkFfSEFSRFdBUkVfU1RVQl9ICi0jZGVmaW5lIEFORFJPSURfSEFSRFdBUkVfQ0FNRVJBX0hBUkRXQVJFX1NUVUJfSAotCi0jaW5jbHVkZSAiRmFrZUNhbWVyYS5oIgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLSNpbmNsdWRlIDx1aS9DYW1lcmFIYXJkd2FyZUludGVyZmFjZS5oPgotI2luY2x1ZGUgPHV0aWxzL01lbW9yeUJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy9NZW1vcnlIZWFwQmFzZS5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBDYW1lcmFIYXJkd2FyZVN0dWIgOiBwdWJsaWMgQ2FtZXJhSGFyZHdhcmVJbnRlcmZhY2UgewotcHVibGljOgotICAgIHZpcnR1YWwgc3A8SU1lbW9yeUhlYXA+IGdldFByZXZpZXdIZWFwKCkgY29uc3Q7Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHN0YXJ0UHJldmlldyhwcmV2aWV3X2NhbGxiYWNrIGNiLCB2b2lkKiB1c2VyKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIHN0b3BQcmV2aWV3KCk7Ci0gICAgdmlydHVhbCBib29sICAgICAgICBwcmV2aWV3RW5hYmxlZCgpOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzdGFydFJlY29yZGluZyhyZWNvcmRpbmdfY2FsbGJhY2sgY2IsIHZvaWQqIHVzZXIpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgc3RvcFJlY29yZGluZygpOwotICAgIHZpcnR1YWwgYm9vbCAgICAgICAgcmVjb3JkaW5nRW5hYmxlZCgpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgcmVsZWFzZVJlY29yZGluZ0ZyYW1lKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0pOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBhdXRvRm9jdXMoYXV0b2ZvY3VzX2NhbGxiYWNrLCB2b2lkICp1c2VyKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHRha2VQaWN0dXJlKHNodXR0ZXJfY2FsbGJhY2ssCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXdfY2FsbGJhY2ssCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBqcGVnX2NhbGxiYWNrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCogdXNlcik7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBjYW5jZWxQaWN0dXJlKGJvb2wgY2FuY2VsX3NodXR0ZXIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2wgY2FuY2VsX3JhdywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYm9vbCBjYW5jZWxfanBlZyk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykgY29uc3Q7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRQYXJhbWV0ZXJzKGNvbnN0IENhbWVyYVBhcmFtZXRlcnMmIHBhcmFtcyk7Ci0gICAgdmlydHVhbCBDYW1lcmFQYXJhbWV0ZXJzICBnZXRQYXJhbWV0ZXJzKCkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkIHJlbGVhc2UoKTsKLQotICAgIHN0YXRpYyBzcDxDYW1lcmFIYXJkd2FyZUludGVyZmFjZT4gY3JlYXRlSW5zdGFuY2UoKTsKLQotcHJpdmF0ZToKLSAgICAgICAgICAgICAgICAgICAgICAgIENhbWVyYUhhcmR3YXJlU3R1YigpOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgfkNhbWVyYUhhcmR3YXJlU3R1YigpOwotCi0gICAgc3RhdGljIHdwPENhbWVyYUhhcmR3YXJlSW50ZXJmYWNlPiBzaW5nbGV0b247Ci0KLSAgICBzdGF0aWMgY29uc3QgaW50IGtCdWZmZXJDb3VudCA9IDQ7Ci0KLSAgICBjbGFzcyBQcmV2aWV3VGhyZWFkIDogcHVibGljIFRocmVhZCB7Ci0gICAgICAgIENhbWVyYUhhcmR3YXJlU3R1YiogbUhhcmR3YXJlOwotICAgIHB1YmxpYzoKLSAgICAgICAgUHJldmlld1RocmVhZChDYW1lcmFIYXJkd2FyZVN0dWIqIGh3KQotICAgICAgICAgICAgOiBUaHJlYWQoZmFsc2UpLCBtSGFyZHdhcmUoaHcpIHsgfQotICAgICAgICB2aXJ0dWFsIHZvaWQgb25GaXJzdFJlZigpIHsKLSAgICAgICAgICAgIHJ1bigiQ2FtZXJhUHJldmlld1RocmVhZCIsIFBSSU9SSVRZX1VSR0VOVF9ESVNQTEFZKTsKLSAgICAgICAgfQotICAgICAgICB2aXJ0dWFsIGJvb2wgdGhyZWFkTG9vcCgpIHsKLSAgICAgICAgICAgIG1IYXJkd2FyZS0+cHJldmlld1RocmVhZCgpOwotICAgICAgICAgICAgLy8gbG9vcCB1bnRpbCB3ZSBuZWVkIHRvIHF1aXQKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgfTsKLQotICAgIHZvaWQgaW5pdERlZmF1bHRQYXJhbWV0ZXJzKCk7Ci0gICAgdm9pZCBpbml0SGVhcExvY2tlZCgpOwotCi0gICAgaW50IHByZXZpZXdUaHJlYWQoKTsKLQotICAgIHN0YXRpYyBpbnQgYmVnaW5BdXRvRm9jdXNUaHJlYWQodm9pZCAqY29va2llKTsKLSAgICBpbnQgYXV0b0ZvY3VzVGhyZWFkKCk7Ci0KLSAgICBzdGF0aWMgaW50IGJlZ2luUGljdHVyZVRocmVhZCh2b2lkICpjb29raWUpOwotICAgIGludCBwaWN0dXJlVGhyZWFkKCk7Ci0KLSAgICBtdXRhYmxlIE11dGV4ICAgICAgIG1Mb2NrOwotCi0gICAgQ2FtZXJhUGFyYW1ldGVycyAgICBtUGFyYW1ldGVyczsKLQotICAgIHNwPE1lbW9yeUhlYXBCYXNlPiAgbUhlYXA7Ci0gICAgc3A8TWVtb3J5QmFzZT4gICAgICBtQnVmZmVyc1trQnVmZmVyQ291bnRdOwotCi0gICAgRmFrZUNhbWVyYSAgICAgICAgICAqbUZha2VDYW1lcmE7Ci0gICAgYm9vbCAgICAgICAgICAgICAgICBtUHJldmlld1J1bm5pbmc7Ci0gICAgaW50ICAgICAgICAgICAgICAgICBtUHJldmlld0ZyYW1lU2l6ZTsKLQotICAgIHNodXR0ZXJfY2FsbGJhY2sgICAgbVNodXR0ZXJDYWxsYmFjazsKLSAgICByYXdfY2FsbGJhY2sgICAgICAgIG1SYXdQaWN0dXJlQ2FsbGJhY2s7Ci0gICAganBlZ19jYWxsYmFjayAgICAgICBtSnBlZ1BpY3R1cmVDYWxsYmFjazsKLSAgICB2b2lkICAgICAgICAgICAgICAgICptUGljdHVyZUNhbGxiYWNrQ29va2llOwotCi0gICAgLy8gcHJvdGVjdGVkIGJ5IG1Mb2NrCi0gICAgc3A8UHJldmlld1RocmVhZD4gICBtUHJldmlld1RocmVhZDsKLSAgICBwcmV2aWV3X2NhbGxiYWNrICAgIG1QcmV2aWV3Q2FsbGJhY2s7Ci0gICAgdm9pZCAgICAgICAgICAgICAgICAqbVByZXZpZXdDYWxsYmFja0Nvb2tpZTsKLQotICAgIGF1dG9mb2N1c19jYWxsYmFjayAgbUF1dG9Gb2N1c0NhbGxiYWNrOwotICAgIHZvaWQgICAgICAgICAgICAgICAgKm1BdXRvRm9jdXNDYWxsYmFja0Nvb2tpZTsKLQotICAgIC8vIG9ubHkgdXNlZCBmcm9tIFByZXZpZXdUaHJlYWQKLSAgICBpbnQgICAgICAgICAgICAgICAgIG1DdXJyZW50UHJldmlld0ZyYW1lOwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmCmRpZmYgLS1naXQgYS9jYW1lcmEvbGliY2FtZXJhc2VydmljZS9DYW1lcmFTZXJ2aWNlLmNwcCBiL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0NhbWVyYVNlcnZpY2UuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlNWQ0MjIwLi4wMDAwMDAwCi0tLSBhL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0NhbWVyYVNlcnZpY2UuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTA3MCArMCwwIEBACi0vKgotKioKLSoqIENvcHlyaWdodCAoQykgMjAwOCwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKiogQ29weXJpZ2h0IChDKSAyMDA4IEhUQyBJbmMuCi0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0vLyNkZWZpbmUgTE9HX05ERUJVRyAwCi0jZGVmaW5lIExPR19UQUcgIkNhbWVyYVNlcnZpY2UiCi0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9JU2VydmljZU1hbmFnZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzE2Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5QmFzZS5oPgotI2luY2x1ZGUgPHV0aWxzL01lbW9yeUhlYXBCYXNlLmg+Ci0jaW5jbHVkZSA8dWkvSUNhbWVyYVNlcnZpY2UuaD4KLQotI2luY2x1ZGUgIkNhbWVyYVNlcnZpY2UuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1leHRlcm4gIkMiIHsKLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHN5cy9zdGF0Lmg+Ci0jaW5jbHVkZSA8ZmNudGwuaD4KLSNpbmNsdWRlIDxwdGhyZWFkLmg+Ci19Ci0KLS8vIFdoZW4geW91IGVuYWJsZSB0aGlzLCBhcyB3ZWxsIGFzIERFQlVHX1JFRlM9MSBhbmQKLS8vIERFQlVHX1JFRlNfRU5BQkxFRF9CWV9ERUZBVUxUPTAgaW4gbGlidXRpbHMvUmVmQmFzZS5jcHAsIHRoaXMgd2lsbCB0cmFjayBhbGwKLS8vIHJlZmVyZW5jZXMgdG8gdGhlIENhbWVyYVNlcnZpY2U6OkNsaWVudCBpbiBvcmRlciB0byBjYXRjaCB0aGUgY2FzZSB3aGVyZSB0aGUKLS8vIGNsaWVudCBpcyBiZWluZyBkZXN0cm95ZWQgd2hpbGUgYSBjYWxsYmFjayBmcm9tIHRoZSBDYW1lcmFIYXJkd2FyZUludGVyZmFjZQotLy8gaXMgb3V0c3RhbmRpbmcuICBUaGlzIGlzIGEgc2VyaW91cyBidWcgYmVjYXVzZSBpZiB3ZSBtYWtlIGFub3RoZXIgY2FsbCBpbnRvCi0vLyBDYW1lcmFIYXJkd3JlSW50ZXJmYWNlIHRoYXQgaXRzZWxmIHRyaWdnZXJzIGEgY2FsbGJhY2ssIHdlIHdpbGwgZGVhZGxvY2suCi0KLSNkZWZpbmUgREVCVUdfQ0xJRU5UX1JFRkVSRU5DRVMgMAotCi0jZGVmaW5lIFBJQ1RVUkVfVElNRU9VVCBzZWNvbmRzKDUpCi0KLSNkZWZpbmUgREVCVUdfRFVNUF9QUkVWSUVXX0ZSQU1FX1RPX0ZJTEUgMCAvKiBuLXRoIGZyYW1lIHRvIHdyaXRlICovCi0jZGVmaW5lIERFQlVHX0RVTVBfSlBFR19TTkFQU0hPVF9UT19GSUxFIDAKLSNkZWZpbmUgREVCVUdfRFVNUF9ZVVZfU05BUFNIT1RfVE9fRklMRSAwCi0KLSNpZiBERUJVR19EVU1QX1BSRVZJRVdfRlJBTUVfVE9fRklMRQotc3RhdGljIGludCBkZWJ1Z19mcmFtZV9jbnQ7Ci0jZW5kaWYKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIENhbWVyYVNlcnZpY2U6Omluc3RhbnRpYXRlKCkgewotICAgIGRlZmF1bHRTZXJ2aWNlTWFuYWdlcigpLT5hZGRTZXJ2aWNlKAotICAgICAgICAgICAgU3RyaW5nMTYoIm1lZGlhLmNhbWVyYSIpLCBuZXcgQ2FtZXJhU2VydmljZSgpKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1DYW1lcmFTZXJ2aWNlOjpDYW1lcmFTZXJ2aWNlKCkgOgotICAgIEJuQ2FtZXJhU2VydmljZSgpCi17Ci0gICAgTE9HSSgiQ2FtZXJhU2VydmljZSBzdGFydGVkOiBwaWQ9JWQiLCBnZXRwaWQoKSk7Ci19Ci0KLUNhbWVyYVNlcnZpY2U6On5DYW1lcmFTZXJ2aWNlKCkKLXsKLSAgICBpZiAobUNsaWVudCAhPSAwKSB7Ci0gICAgICAgIExPR0UoIm1DbGllbnQgd2FzIHN0aWxsIGNvbm5lY3RlZCBpbiBkZXN0cnVjdG9yISIpOwotICAgIH0KLX0KLQotc3A8SUNhbWVyYT4gQ2FtZXJhU2VydmljZTo6Y29ubmVjdChjb25zdCBzcDxJQ2FtZXJhQ2xpZW50PiYgY2FtZXJhQ2xpZW50KQotewotICAgIExPR0QoIkNvbm5lY3QgRSBmcm9tIElDYW1lcmFDbGllbnQgJXAiLCBjYW1lcmFDbGllbnQtPmFzQmluZGVyKCkuZ2V0KCkpOwotCi0gICAgTXV0ZXg6OkF1dG9sb2NrIGxvY2sobUxvY2spOwotICAgIHNwPENsaWVudD4gY2xpZW50OwotICAgIGlmIChtQ2xpZW50ICE9IDApIHsKLSAgICAgICAgc3A8Q2xpZW50PiBjdXJyZW50Q2xpZW50ID0gbUNsaWVudC5wcm9tb3RlKCk7Ci0gICAgICAgIGlmIChjdXJyZW50Q2xpZW50ICE9IDApIHsKLSAgICAgICAgICAgIHNwPElDYW1lcmFDbGllbnQ+IGN1cnJlbnRDYW1lcmFDbGllbnQoY3VycmVudENsaWVudC0+Z2V0Q2FtZXJhQ2xpZW50KCkpOwotICAgICAgICAgICAgaWYgKGNhbWVyYUNsaWVudC0+YXNCaW5kZXIoKSA9PSBjdXJyZW50Q2FtZXJhQ2xpZW50LT5hc0JpbmRlcigpKSB7Ci0gICAgICAgICAgICAgICAgLy8gdGhpcyBpcyB0aGUgc2FtZSBjbGllbnQgcmVjb25uZWN0aW5nLi4uCi0gICAgICAgICAgICAgICAgTE9HRCgiQ29ubmVjdCBYIHNhbWUgY2xpZW50ICglcCkgaXMgcmVjb25uZWN0aW5nLi4uIiwgY2FtZXJhQ2xpZW50LT5hc0JpbmRlcigpLmdldCgpKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gY3VycmVudENsaWVudDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgLy8gaXQncyBhbm90aGVyIGNsaWVudC4uLiByZWplY3QgaXQKLSAgICAgICAgICAgICAgICBMT0dEKCJuZXcgY2xpZW50ICglcCkgYXR0ZW1wdGluZyB0byBjb25uZWN0IC0gcmVqZWN0ZWQiLCBjYW1lcmFDbGllbnQtPmFzQmluZGVyKCkuZ2V0KCkpOwotICAgICAgICAgICAgICAgIHJldHVybiBjbGllbnQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBjYW4ndCBwcm9tb3RlLCB0aGUgcHJldmlvdXMgY2xpZW50IGhhcyBkaWVkLi4uCi0gICAgICAgICAgICBMT0dEKCJuZXcgY2xpZW50IGNvbm5lY3RpbmcsIG9sZCByZWZlcmVuY2Ugd2FzIGRhbmdsaW5nLi4uIik7Ci0gICAgICAgICAgICBtQ2xpZW50LmNsZWFyKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyBjcmVhdGUgYSBuZXcgQ2xpZW50IG9iamVjdAotICAgIGNsaWVudCA9IG5ldyBDbGllbnQodGhpcywgY2FtZXJhQ2xpZW50LCBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5nZXRDYWxsaW5nUGlkKCkpOwotICAgIG1DbGllbnQgPSBjbGllbnQ7Ci0jaWYgREVCVUdfQ0xJRU5UX1JFRkVSRU5DRVMKLSAgICAvLyBFbmFibGUgdHJhY2tpbmcgZm9yIHRoaXMgb2JqZWN0LCBhbmQgdHJhY2sgaW5jcmVtZW50cyBhbmQgZGVjcmVtZW50cyBvZgotICAgIC8vIHRoZSByZWZjb3VudC4KLSAgICBjbGllbnQtPnRyYWNrTWUodHJ1ZSwgdHJ1ZSk7Ci0jZW5kaWYKLSAgICBMT0dEKCJDb25uZWN0IFgiKTsKLSAgICByZXR1cm4gY2xpZW50OwotfQotCi12b2lkIENhbWVyYVNlcnZpY2U6OnJlbW92ZUNsaWVudChjb25zdCBzcDxJQ2FtZXJhQ2xpZW50PiYgY2FtZXJhQ2xpZW50KQotewotICAgIC8vIGRlY2xhciB0aGlzIG91dHNpZGUgdGhlIGxvY2sgdG8gbWFrZSBhYnNvbHV0ZWx5IHN1cmUgdGhlCi0gICAgLy8gZGVzdHJ1Y3RvciB3b24ndCBiZSBjYWxsZWQgd2l0aCB0aGUgbG9jayBoZWxkLgotICAgIHNwPENsaWVudD4gY2xpZW50OwotCi0gICAgTXV0ZXg6OkF1dG9sb2NrIGxvY2sobUxvY2spOwotCi0gICAgaWYgKG1DbGllbnQgPT0gMCkgewotICAgICAgICAvLyBUaGlzIGhhcHBlbnMgd2hlbiB3ZSBoYXZlIGFscmVhZHkgZGlzY29ubmVjdGVkLgotICAgICAgICBMT0dWKCJtQ2xpZW50IGlzIG51bGwuIik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICAvLyBQcm9tb3RlIG1DbGllbnQuIEl0IHNob3VsZCBuZXZlciBmYWlsIGJlY2F1c2Ugd2UncmUgY2FsbGVkIGZyb20KLSAgICAvLyBhIGJpbmRlciBjYWxsLCBzbyBzb21lb25lIGhhcyB0byBoYXZlIGEgc3Ryb25nIHJlZmVyZW5jZS4KLSAgICBjbGllbnQgPSBtQ2xpZW50LnByb21vdGUoKTsKLSAgICBpZiAoY2xpZW50ID09IDApIHsKLSAgICAgICAgTE9HVygiY2FuJ3QgZ2V0IGEgc3Ryb25nIHJlZmVyZW5jZSBvbiBtQ2xpZW50ISIpOwotICAgICAgICBtQ2xpZW50LmNsZWFyKCk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBpZiAoY2FtZXJhQ2xpZW50LT5hc0JpbmRlcigpICE9IGNsaWVudC0+Z2V0Q2FtZXJhQ2xpZW50KCktPmFzQmluZGVyKCkpIHsKLSAgICAgICAgLy8gdWdoISB0aGF0J3Mgbm90IG91ciBjbGllbnQhIQotICAgICAgICBMT0dXKCJyZW1vdmVDbGllbnQoKSBjYWxsZWQsIGJ1dCBtQ2xpZW50IGRvZXNuJ3QgbWF0Y2ghIik7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgLy8gb2theSwgZ29vZCwgZm9yZ2V0IGFib3V0IG1DbGllbnQKLSAgICAgICAgbUNsaWVudC5jbGVhcigpOwotICAgIH0KLX0KLQotQ2FtZXJhU2VydmljZTo6Q2xpZW50OjpDbGllbnQoY29uc3Qgc3A8Q2FtZXJhU2VydmljZT4mIGNhbWVyYVNlcnZpY2UsCi0gICAgICAgIGNvbnN0IHNwPElDYW1lcmFDbGllbnQ+JiBjYW1lcmFDbGllbnQsIHBpZF90IGNsaWVudFBpZCkgCi17Ci0gICAgTE9HRCgiQ2xpZW50IEUgY29uc3RydWN0b3IiKTsKLSAgICBtQ2FtZXJhU2VydmljZSA9IGNhbWVyYVNlcnZpY2U7Ci0gICAgbUNhbWVyYUNsaWVudCA9IGNhbWVyYUNsaWVudDsKLSAgICBtQ2xpZW50UGlkID0gY2xpZW50UGlkOwotICAgIG1IYXJkd2FyZSA9IG9wZW5DYW1lcmFIYXJkd2FyZSgpOwotICAgIG1Vc2VPdmVybGF5ID0gbUhhcmR3YXJlLT51c2VPdmVybGF5KCk7Ci0KLSAgICAvLyBDYWxsYmFjayBpcyBkaXNhYmxlZCBieSBkZWZhdWx0Ci0gICAgbVByZXZpZXdDYWxsYmFja0ZsYWcgPSBGUkFNRV9DQUxMQkFDS19GTEFHX05PT1A7Ci0gICAgTE9HRCgiQ2xpZW50IFggY29uc3RydWN0b3IiKTsKLX0KLQotc3RhdHVzX3QgQ2FtZXJhU2VydmljZTo6Q2xpZW50OjpjaGVja1BpZCgpCi17Ci0gICAgaWYgKG1DbGllbnRQaWQgPT0gSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpKSByZXR1cm4gTk9fRVJST1I7Ci0gICAgTE9HVygiQXR0ZW1wdCB0byB1c2UgbG9ja2VkIGNhbWVyYSAoJXApIGZyb20gZGlmZmVyZW50IHByb2Nlc3MiLCBnZXRDYW1lcmFDbGllbnQoKS0+YXNCaW5kZXIoKS5nZXQoKSk7Ci0gICAgcmV0dXJuIC1FQlVTWTsKLX0KLQotc3RhdHVzX3QgQ2FtZXJhU2VydmljZTo6Q2xpZW50Ojpsb2NrKCkKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIC8vIGxvY2sgY2FtZXJhIHRvIHRoaXMgY2xpZW50IGlmIHRoZSB0aGUgY2FtZXJhIGlzIHVubG9ja2VkCi0gICAgaWYgKG1DbGllbnRQaWQgPT0gMCkgewotICAgICAgICBtQ2xpZW50UGlkID0gSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpOwotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotICAgIC8vIHJldHVybnMgTk9fRVJST1IgaWYgdGhlIGNsaWVudCBhbHJlYWR5IG93bnMgdGhlIGNhbWVyYSwgLUVCVVNZIG90aGVyd2lzZQotICAgIHJldHVybiBjaGVja1BpZCgpOwotfQotCi1zdGF0dXNfdCBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OnVubG9jaygpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICAvLyBhbGxvdyBhbnlvbmUgdG8gdXNlIGNhbWVyYQotICAgIExPR1YoInVubG9jayAoJXApIiwgZ2V0Q2FtZXJhQ2xpZW50KCktPmFzQmluZGVyKCkuZ2V0KCkpOwotICAgIHN0YXR1c190IHJlc3VsdCA9IGNoZWNrUGlkKCk7Ci0gICAgaWYgKHJlc3VsdCA9PSBOT19FUlJPUikgbUNsaWVudFBpZCA9IDA7Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotc3RhdHVzX3QgQ2FtZXJhU2VydmljZTo6Q2xpZW50Ojpjb25uZWN0KGNvbnN0IHNwPElDYW1lcmFDbGllbnQ+JiBjbGllbnQpCi17Ci0gICAgLy8gY29ubmVjdCBhIG5ldyBwcm9jZXNzIHRvIHRoZSBjYW1lcmEKLSAgICBMT0dWKCJjb25uZWN0ICglcCkiLCBjbGllbnQtPmFzQmluZGVyKCkuZ2V0KCkpOwotCi0gICAgLy8gSSBoYXRlIHRoaXMgaGFjaywgYnV0IHRoaW5ncyBnZXQgcmVhbGx5IHVnbHkgd2hlbiB0aGUgbWVkaWEgcmVjb3JkZXIKLSAgICAvLyBzZXJ2aWNlIGlzIGhhbmRpbmcgYmFjayB0aGUgY2FtZXJhIHRvIHRoZSBhcHAuIFRoZSBJQ2FtZXJhQ2xpZW50Ci0gICAgLy8gZGVzdHJ1Y3RvciB3aWxsIGJlIGNhbGxlZCBkdXJpbmcgdGhlIHNhbWUgSVBDLCBtYWtpbmcgaXQgbG9vayBsaWtlCi0gICAgLy8gdGhlIHJlbW90ZSBjbGllbnQgaXMgdHJ5aW5nIHRvIGRpc2Nvbm5lY3QuIFRoaXMgaGFjayB0ZW1wb3JhcmlseQotICAgIC8vIHNldHMgdGhlIG1DbGllbnRQaWQgdG8gYW4gaW52YWxpZCBwaWQgdG8gcHJldmVudCB0aGUgaGFyZHdhcmUgZnJvbQotICAgIC8vICBiZWluZyB0b3JuIGRvd24uCi0gICAgewotCi0gICAgICAgIC8vIGhvbGQgYSByZWZlcmVuY2UgdG8gdGhlIG9sZCBjbGllbnQgb3Igd2Ugd2lsbCBkZWFkbG9jayBpZiB0aGUgY2xpZW50IGlzCi0gICAgICAgIC8vIGluIHRoZSBzYW1lIHByb2Nlc3MgYW5kIHdlIGhvbGQgdGhlIGxvY2sgd2hlbiB3ZSByZW1vdmUgdGhlIHJlZmVyZW5jZQotICAgICAgICBzcDxJQ2FtZXJhQ2xpZW50PiBvbGRDbGllbnQ7Ci0gICAgICAgIHsKLSAgICAgICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgICAgICAgICBpZiAobUNsaWVudFBpZCAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgTE9HVygiVHJpZWQgdG8gY29ubmVjdCB0byBsb2NrZWQgY2FtZXJhIik7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIC1FQlVTWTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG9sZENsaWVudCA9IG1DYW1lcmFDbGllbnQ7Ci0KLSAgICAgICAgICAgIC8vIGRpZCB0aGUgY2xpZW50IGFjdHVhbGx5IGNoYW5nZT8KLSAgICAgICAgICAgIGlmIChjbGllbnQtPmFzQmluZGVyKCkgPT0gbUNhbWVyYUNsaWVudC0+YXNCaW5kZXIoKSkgcmV0dXJuIE5PX0VSUk9SOwotCi0gICAgICAgICAgICBtQ2FtZXJhQ2xpZW50ID0gY2xpZW50OwotICAgICAgICAgICAgbUNsaWVudFBpZCA9IC0xOwotICAgICAgICAgICAgbVByZXZpZXdDYWxsYmFja0ZsYWcgPSBGUkFNRV9DQUxMQkFDS19GTEFHX05PT1A7Ci0gICAgICAgICAgICBMT0dWKCJjb25uZWN0IG5ldyBwcm9jZXNzICglZCkgdG8gZXhpc3RpbmcgY2FtZXJhIGNsaWVudCIsIG1DbGllbnRQaWQpOwotICAgICAgICB9Ci0KLSAgICB9Ci0gICAgLy8gdGhlIG9sZCBjbGllbnQgZGVzdHJ1Y3RvciBpcyBjYWxsZWQgd2hlbiBvbGRDbGllbnQgZ29lcyBvdXQgb2Ygc2NvcGUKLSAgICAvLyBub3cgd2Ugc2V0IHRoZSBuZXcgUElEIHRvIGxvY2sgdGhlIGludGVyZmFjZSBhZ2FpbgotICAgIG1DbGllbnRQaWQgPSBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5nZXRDYWxsaW5nUGlkKCk7Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLSNpZiBIQVZFX0FORFJPSURfT1MKLXN0YXRpYyB2b2lkICp1bnJlZ2lzdGVyX3N1cmZhY2Uodm9pZCAqYXJnKQotewotICAgIElTdXJmYWNlICpzdXJmYWNlID0gKElTdXJmYWNlICopYXJnOwotICAgIHN1cmZhY2UtPnVucmVnaXN0ZXJCdWZmZXJzKCk7Ci0gICAgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Zmx1c2hDb21tYW5kcygpOwotICAgIHJldHVybiBOVUxMOwotfQotI2VuZGlmCi0KLUNhbWVyYVNlcnZpY2U6OkNsaWVudDo6fkNsaWVudCgpCi17Ci0gICAgLy8gdGVhciBkb3duIGNsaWVudAotICAgIExPR0QoIkNsaWVudCAoJXApICBFIGRlc3RydWN0b3IiLCBnZXRDYW1lcmFDbGllbnQoKS0+YXNCaW5kZXIoKS5nZXQoKSk7Ci0gICAgaWYgKG1TdXJmYWNlICE9IDAgJiYgIW1Vc2VPdmVybGF5KSB7Ci0jaWYgSEFWRV9BTkRST0lEX09TCi0gICAgICAgIHB0aHJlYWRfdCB0aHI7Ci0gICAgICAgIC8vIFdlIHVucmVnaXN0ZXIgdGhlIGJ1ZmZlcnMgaW4gYSBkaWZmZXJlbnQgdGhyZWFkIGJlY2F1c2UgYmluZGVyIGRvZXMKLSAgICAgICAgLy8gbm90IGxldCB1cyBtYWtlIHN5Y2hyb25vdXMgdHJhbnNhY3Rpb25zIGluIGEgYmluZGVyIGRlc3RydWN0b3IgKHRoYXQKLSAgICAgICAgLy8gaXMsIHVwb24gb3VyIHJlYWNoaW5nIGEgcmVmY291bnQgb2YgemVyby4pCi0gICAgICAgIHB0aHJlYWRfY3JlYXRlKCZ0aHIsIE5VTEwsCi0gICAgICAgICAgICAgICAgICAgICAgIHVucmVnaXN0ZXJfc3VyZmFjZSwKLSAgICAgICAgICAgICAgICAgICAgICAgbVN1cmZhY2UuZ2V0KCkpOwotICAgICAgICBwdGhyZWFkX2pvaW4odGhyLCBOVUxMKTsKLSNlbHNlCi0gICAgICAgIG1TdXJmYWNlLT51bnJlZ2lzdGVyQnVmZmVycygpOwotI2VuZGlmCi0gICAgfQotCi0gICAgLy8gbWFrZSBzdXJlIHdlIHRlYXIgZG93biB0aGUgaGFyZHdhcmUKLSAgICBtQ2xpZW50UGlkID0gSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpOwotICAgIGRpc2Nvbm5lY3QoKTsKLSAgICBMT0dEKCJDbGllbnQgWCBkZXN0cnVjdG9yIik7Ci19Ci0KLXZvaWQgQ2FtZXJhU2VydmljZTo6Q2xpZW50OjpkaXNjb25uZWN0KCkKLXsKLSAgICBMT0dEKCJDbGllbnQgKCVwKSBFIGRpc2Nvbm5lY3QgZnJvbSAoJWQpIiwKLSAgICAgICAgICAgIGdldENhbWVyYUNsaWVudCgpLT5hc0JpbmRlcigpLmdldCgpLAotICAgICAgICAgICAgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpKTsKLSAgICBNdXRleDo6QXV0b2xvY2sgbG9jayhtTG9jayk7Ci0gICAgaWYgKG1DbGllbnRQaWQgPD0gMCkgewotICAgICAgICBMT0dWKCJjYW1lcmEgaXMgdW5sb2NrZWQsIGRvbid0IHRlYXIgZG93biBoYXJkd2FyZSIpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmIChjaGVja1BpZCgpICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgIExPR1YoIkRpZmZlcmVudCBjbGllbnQgLSBkb24ndCBkaXNjb25uZWN0Iik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBtQ2FtZXJhU2VydmljZS0+cmVtb3ZlQ2xpZW50KG1DYW1lcmFDbGllbnQpOwotICAgIGlmIChtSGFyZHdhcmUgIT0gMCkgewotICAgICAgICBMT0dWKCJoYXJkd2FyZSB0ZWFyZG93biIpOwotICAgICAgICAvLyBCZWZvcmUgZGVzdHJveWluZyBtSGFyZHdhcmUsIHdlIG11c3QgbWFrZSBzdXJlIGl0J3MgaW4gdGhlCi0gICAgICAgIC8vIGlkbGUgc3RhdGUuCi0gICAgICAgIG1IYXJkd2FyZS0+c3RvcFByZXZpZXcoKTsKLSAgICAgICAgLy8gQ2FuY2VsIGFsbCBwaWN0dXJlIGNhbGxiYWNrcy4KLSAgICAgICAgbUhhcmR3YXJlLT5jYW5jZWxQaWN0dXJlKHRydWUsIHRydWUsIHRydWUpOwotICAgICAgICAvLyBSZWxlYXNlIHRoZSBoYXJkd2FyZSByZXNvdXJjZXMuCi0gICAgICAgIG1IYXJkd2FyZS0+cmVsZWFzZSgpOwotICAgIH0KLSAgICBtSGFyZHdhcmUuY2xlYXIoKTsKLSAgICBMT0dEKCJDbGllbnQgWCBkaXNjb25uZWN0Iik7Ci19Ci0KLS8vIHBhc3MgdGhlIGJ1ZmZlcmVkIElTdXJmYWNlIHRvIHRoZSBjYW1lcmEgc2VydmljZQotc3RhdHVzX3QgQ2FtZXJhU2VydmljZTo6Q2xpZW50OjpzZXRQcmV2aWV3RGlzcGxheShjb25zdCBzcDxJU3VyZmFjZT4mIHN1cmZhY2UpCi17Ci0gICAgTE9HRCgic2V0UHJldmlld0Rpc3BsYXkoJXApIiwgc3VyZmFjZS5nZXQoKSk7Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIGxvY2sobUxvY2spOwotICAgIHN0YXR1c190IHJlc3VsdCA9IGNoZWNrUGlkKCk7Ci0gICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikgcmV0dXJuIHJlc3VsdDsKLSAgICBNdXRleDo6QXV0b2xvY2sgc3VyZmFjZUxvY2sobVN1cmZhY2VMb2NrKTsKLSAgICAvLyBhc0JpbmRlcigpIGlzIHNhZmUgb24gTlVMTCAocmV0dXJucyBOVUxMKQotICAgIGlmIChzdXJmYWNlLT5hc0JpbmRlcigpICE9IG1TdXJmYWNlLT5hc0JpbmRlcigpKSB7Ci0gICAgICAgIGlmIChtU3VyZmFjZSAhPSAwICYmICFtVXNlT3ZlcmxheSkgewotICAgICAgICAgICAgTE9HRCgiY2xlYXJpbmcgb2xkIHByZXZpZXcgc3VyZmFjZSAlcCIsIG1TdXJmYWNlLmdldCgpKTsKLSAgICAgICAgICAgIG1TdXJmYWNlLT51bnJlZ2lzdGVyQnVmZmVycygpOwotICAgICAgICB9Ci0gICAgICAgIG1TdXJmYWNlID0gc3VyZmFjZTsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vLyBzZXQgdGhlIHByZXZpZXcgY2FsbGJhY2sgZmxhZyB0byBhZmZlY3QgaG93IHRoZSByZWNlaXZlZCBmcmFtZXMgZnJvbQotLy8gcHJldmlldyBhcmUgaGFuZGxlZC4KLXZvaWQgQ2FtZXJhU2VydmljZTo6Q2xpZW50OjpzZXRQcmV2aWV3Q2FsbGJhY2tGbGFnKGludCBjYWxsYmFja19mbGFnKQotewotICAgIExPR1YoInNldFByZXZpZXdDYWxsYmFja0ZsYWciKTsKLSAgICBNdXRleDo6QXV0b2xvY2sgbG9jayhtTG9jayk7Ci0gICAgaWYgKGNoZWNrUGlkKCkgIT0gTk9fRVJST1IpIHJldHVybjsKLSAgICBtUHJldmlld0NhbGxiYWNrRmxhZyA9IGNhbGxiYWNrX2ZsYWc7Ci19Ci0KLS8vIHN0YXJ0IHByZXZpZXcgbW9kZSwgbXVzdCBjYWxsIHNldFByZXZpZXdEaXNwbGF5IGZpcnN0Ci1zdGF0dXNfdCBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OnN0YXJ0Q2FtZXJhTW9kZShjYW1lcmFfbW9kZSBtb2RlKQotewotICAgIExPR0QoInN0YXJ0Q2FtZXJhTW9kZSglZCkiLCBtb2RlKTsKLQotICAgIC8qIHdlIGNhbm5vdCBjYWxsIGludG8gbUhhcmR3YXJlIHdpdGggbUxvY2sgaGVsZCBiZWNhdXNlCi0gICAgICogbUhhcmR3YXJlIGhhcyBjYWxsYmFja3Mgb250byB1cyB3aGljaCBhY3F1aXJlIHRoaXMgbG9jawotICAgICAqLwotCi0gICAgTXV0ZXg6OkF1dG9sb2NrIGxvY2sobUxvY2spOwotICAgIHN0YXR1c190IHJlc3VsdCA9IGNoZWNrUGlkKCk7Ci0gICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikgcmV0dXJuIHJlc3VsdDsKLQotICAgIGlmIChtSGFyZHdhcmUgPT0gMCkgewotICAgICAgICBMT0dFKCJtSGFyZHdhcmUgaXMgTlVMTCwgcmV0dXJuaW5nLiIpOwotICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgfQotCi0gICAgaWYgKG1TdXJmYWNlID09IDApIHsKLSAgICAgICAgTE9HRSgic2V0UHJldmlld0Rpc3BsYXkgbXVzdCBiZSBjYWxsZWQgYmVmb3JlIHN0YXJ0Q2FtZXJhTW9kZSEiKTsKLSAgICAgICAgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOwotICAgIH0KLQotICAgIHN3aXRjaChtb2RlKSB7Ci0gICAgY2FzZSBDQU1FUkFfUkVDT1JESU5HX01PREU6Ci0gICAgICAgIHJldHVybiBzdGFydFJlY29yZGluZ01vZGUoKTsKLQotICAgIGRlZmF1bHQ6IC8vIENBTUVSQV9QUkVWSUVXX01PREUKLSAgICAgICAgcmV0dXJuIHN0YXJ0UHJldmlld01vZGUoKTsKLSAgICB9Ci19Ci0KLXN0YXR1c190IENhbWVyYVNlcnZpY2U6OkNsaWVudDo6c3RhcnRSZWNvcmRpbmdNb2RlKCkKLXsKLSAgICBMT0dWKCJzdGFydFJlY29yZGluZ01vZGUiKTsKLQotICAgIHN0YXR1c190IHJldCA9IFVOS05PV05fRVJST1I7Ci0KLSAgICAvLyBpZiBwcmV2aWV3IGhhcyBub3QgYmVlbiBzdGFydGVkLCBzdGFydCBwcmV2aWV3IGZpcnN0Ci0gICAgaWYgKCFtSGFyZHdhcmUtPnByZXZpZXdFbmFibGVkKCkpIHsKLSAgICAgICAgcmV0ID0gc3RhcnRQcmV2aWV3TW9kZSgpOwotICAgICAgICBpZiAocmV0ICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgICAgICByZXR1cm4gcmV0OwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gaWYgcmVjb3JkaW5nIGhhcyBiZWVuIGVuYWJsZWQsIG5vdGhpbmcgbmVlZHMgdG8gYmUgZG9uZQotICAgIGlmIChtSGFyZHdhcmUtPnJlY29yZGluZ0VuYWJsZWQoKSkgewotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotCi0gICAgLy8gc3RhcnQgcmVjb3JkaW5nIG1vZGUKLSAgICByZXQgPSBtSGFyZHdhcmUtPnN0YXJ0UmVjb3JkaW5nKHJlY29yZGluZ0NhbGxiYWNrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbUNhbWVyYVNlcnZpY2UuZ2V0KCkpOwotICAgIGlmIChyZXQgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgTE9HRSgibUhhcmR3YXJlLT5zdGFydFJlY29yZGluZygpIGZhaWxlZCB3aXRoIHN0YXR1cyAlZCIsIHJldCk7Ci0gICAgfQotICAgIHJldHVybiByZXQ7Ci19Ci0KLXN0YXR1c190IENhbWVyYVNlcnZpY2U6OkNsaWVudDo6c3RhcnRQcmV2aWV3TW9kZSgpCi17Ci0gICAgTE9HVigic3RhcnRQcmV2aWV3TW9kZSIpOwotCi0gICAgLy8gaWYgcHJldmlldyBoYXMgYmVlbiBlbmFibGVkLCBub3RoaW5nIG5lZWRzIHRvIGJlIGRvbmUKLSAgICBpZiAobUhhcmR3YXJlLT5wcmV2aWV3RW5hYmxlZCgpKSB7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0KLSAgICAvLyBzdGFydCBwcmV2aWV3IG1vZGUKLSNpZiBERUJVR19EVU1QX1BSRVZJRVdfRlJBTUVfVE9fRklMRQotICAgIGRlYnVnX2ZyYW1lX2NudCA9IDA7Ci0jZW5kaWYKLSAgICBzdGF0dXNfdCByZXQgPSBVTktOT1dOX0VSUk9SOwotICAgIGludCB3LCBoOwotICAgIENhbWVyYVBhcmFtZXRlcnMgcGFyYW1zKG1IYXJkd2FyZS0+Z2V0UGFyYW1ldGVycygpKTsKLSAgICBwYXJhbXMuZ2V0UHJldmlld1NpemUoJncsICZoKTsKLQotICAgIGlmIChtVXNlT3ZlcmxheSkgewotICAgICAgICBjb25zdCBjaGFyICpmb3JtYXQgPSBwYXJhbXMuZ2V0UHJldmlld0Zvcm1hdCgpOwotICAgICAgICBpbnQgZm10OwotICAgICAgICBMT0dEKCJVc2UgT3ZlcmxheXMiKTsKLSAgICAgICAgaWYgKCFzdHJjbXAoZm9ybWF0LCAieXV2NDIyaSIpKQotICAgICAgICAgICAgZm10ID0gT1ZFUkxBWV9GT1JNQVRfWUNiQ3JfNDIyX0k7Ci0gICAgICAgIGVsc2UgaWYgKCFzdHJjbXAoZm9ybWF0LCAicmdiNTY1IikpCi0gICAgICAgICAgICBmbXQgPSBPVkVSTEFZX0ZPUk1BVF9SR0JfNTY1OwotICAgICAgICBlbHNlIHsKLSAgICAgICAgICAgIExPR0UoIkludmFsaWQgcHJldmlldyBmb3JtYXQgZm9yIG92ZXJsYXlzIik7Ci0gICAgICAgICAgICByZXR1cm4gLUVJTlZBTDsKLSAgICAgICAgfQotICAgICAgICBzcDxPdmVybGF5UmVmPiByZWYgPSBtU3VyZmFjZS0+Y3JlYXRlT3ZlcmxheSh3LCBoLCBmbXQpOwotICAgICAgICByZXQgPSBtSGFyZHdhcmUtPnNldE92ZXJsYXkobmV3IE92ZXJsYXkocmVmKSk7Ci0gICAgICAgIGlmIChyZXQgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgIExPR0UoIm1IYXJkd2FyZS0+c2V0T3ZlcmxheSgpIGZhaWxlZCB3aXRoIHN0YXR1cyAlZFxuIiwgcmV0KTsKLSAgICAgICAgICAgIHJldHVybiByZXQ7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0ID0gbUhhcmR3YXJlLT5zdGFydFByZXZpZXcoTlVMTCwgbUNhbWVyYVNlcnZpY2UuZ2V0KCkpOwotICAgICAgICBpZiAocmV0ICE9IE5PX0VSUk9SKQotICAgICAgICAgICAgTE9HRSgibUhhcmR3YXJlLT5zdGFydFByZXZpZXcoKSBmYWlsZWQgd2l0aCBzdGF0dXMgJWRcbiIsIHJldCk7Ci0gCi0gICAgfSBlbHNlIHsKLSAgICAgICAgcmV0ID0gbUhhcmR3YXJlLT5zdGFydFByZXZpZXcocHJldmlld0NhbGxiYWNrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtQ2FtZXJhU2VydmljZS5nZXQoKSk7Ci0gICAgICAgIGlmIChyZXQgPT0gTk9fRVJST1IpIHsKLQotICAgICAgICAgICAgbVN1cmZhY2UtPnVucmVnaXN0ZXJCdWZmZXJzKCk7Ci0KLSAgICAgICAgICAgIHVpbnQzMl90IHRyYW5zZm9ybSA9IDA7Ci0gICAgICAgICAgICBpZiAocGFyYW1zLmdldE9yaWVudGF0aW9uKCkgPT0KLSAgICAgICAgICAgICAgICBDYW1lcmFQYXJhbWV0ZXJzOjpDQU1FUkFfT1JJRU5UQVRJT05fUE9SVFJBSVQpIHsKLSAgICAgICAgICAgICAgTE9HVigicG9ydHJhaXQgbW9kZSIpOwotICAgICAgICAgICAgICB0cmFuc2Zvcm0gPSBJU3VyZmFjZTo6QnVmZmVySGVhcDo6Uk9UXzkwOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgSVN1cmZhY2U6OkJ1ZmZlckhlYXAgYnVmZmVycyh3LCBoLCB3LCBoLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQSVhFTF9GT1JNQVRfWUNiQ3JfNDIwX1NQLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0cmFuc2Zvcm0sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1IYXJkd2FyZS0+Z2V0UHJldmlld0hlYXAoKSk7Ci0KLSAgICAgICAgICAgIG1TdXJmYWNlLT5yZWdpc3RlckJ1ZmZlcnMoYnVmZmVycyk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgTE9HRSgibUhhcmR3YXJlLT5zdGFydFByZXZpZXcoKSBmYWlsZWQgd2l0aCBzdGF0dXMgJWQiLCByZXQpOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiByZXQ7Ci19Ci0KLXN0YXR1c190IENhbWVyYVNlcnZpY2U6OkNsaWVudDo6c3RhcnRQcmV2aWV3KCkKLXsKLSAgICByZXR1cm4gc3RhcnRDYW1lcmFNb2RlKENBTUVSQV9QUkVWSUVXX01PREUpOwotfQotCi1zdGF0dXNfdCBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OnN0YXJ0UmVjb3JkaW5nKCkKLXsKLSAgICByZXR1cm4gc3RhcnRDYW1lcmFNb2RlKENBTUVSQV9SRUNPUkRJTkdfTU9ERSk7Ci19Ci0KLS8vIHN0b3AgcHJldmlldyBtb2RlCi12b2lkIENhbWVyYVNlcnZpY2U6OkNsaWVudDo6c3RvcFByZXZpZXcoKQotewotICAgIExPR0QoInN0b3BQcmV2aWV3KCkiKTsKLQotICAgIE11dGV4OjpBdXRvbG9jayBsb2NrKG1Mb2NrKTsKLSAgICBpZiAoY2hlY2tQaWQoKSAhPSBOT19FUlJPUikgcmV0dXJuOwotCi0gICAgaWYgKG1IYXJkd2FyZSA9PSAwKSB7Ci0gICAgICAgIExPR0UoIm1IYXJkd2FyZSBpcyBOVUxMLCByZXR1cm5pbmcuIik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBtSGFyZHdhcmUtPnN0b3BQcmV2aWV3KCk7Ci0gICAgTE9HRCgic3RvcFByZXZpZXcoKSwgaGFyZHdhcmUgc3RvcHBlZCBPSyIpOwotCi0gICAgaWYgKG1TdXJmYWNlICE9IDAgJiYgIW1Vc2VPdmVybGF5KSB7Ci0gICAgICAgIG1TdXJmYWNlLT51bnJlZ2lzdGVyQnVmZmVycygpOwotICAgIH0KLSAgICBtUHJldmlld0J1ZmZlci5jbGVhcigpOwotfQotCi0vLyBzdG9wIHJlY29yZGluZyBtb2RlCi12b2lkIENhbWVyYVNlcnZpY2U6OkNsaWVudDo6c3RvcFJlY29yZGluZygpCi17Ci0gICAgTE9HVigic3RvcFJlY29yZGluZygpIik7Ci0KLSAgICBNdXRleDo6QXV0b2xvY2sgbG9jayhtTG9jayk7Ci0gICAgaWYgKGNoZWNrUGlkKCkgIT0gTk9fRVJST1IpIHJldHVybjsKLQotICAgIGlmIChtSGFyZHdhcmUgPT0gMCkgewotICAgICAgICBMT0dFKCJtSGFyZHdhcmUgaXMgTlVMTCwgcmV0dXJuaW5nLiIpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgbUhhcmR3YXJlLT5zdG9wUmVjb3JkaW5nKCk7Ci0gICAgTE9HVigic3RvcFJlY29yZGluZygpLCBoYXJkd2FyZSBzdG9wcGVkIE9LIik7Ci0gICAgbVByZXZpZXdCdWZmZXIuY2xlYXIoKTsKLX0KLQotLy8gcmVsZWFzZSBhIHJlY29yZGluZyBmcmFtZQotdm9pZCBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OnJlbGVhc2VSZWNvcmRpbmdGcmFtZShjb25zdCBzcDxJTWVtb3J5PiYgbWVtKQotewotICAgIExPR1YoInJlbGVhc2VSZWNvcmRpbmdGcmFtZSgpIik7Ci0KLSAgICBNdXRleDo6QXV0b2xvY2sgbG9jayhtTG9jayk7Ci0gICAgaWYgKGNoZWNrUGlkKCkgIT0gTk9fRVJST1IpIHJldHVybjsKLQotICAgIGlmIChtSGFyZHdhcmUgPT0gMCkgewotICAgICAgICBMT0dFKCJtSGFyZHdhcmUgaXMgTlVMTCwgcmV0dXJuaW5nLiIpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgbUhhcmR3YXJlLT5yZWxlYXNlUmVjb3JkaW5nRnJhbWUobWVtKTsKLX0KLQotYm9vbCBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OnByZXZpZXdFbmFibGVkKCkKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgbG9jayhtTG9jayk7Ci0gICAgaWYgKG1IYXJkd2FyZSA9PSAwKSByZXR1cm4gZmFsc2U7Ci0gICAgcmV0dXJuIG1IYXJkd2FyZS0+cHJldmlld0VuYWJsZWQoKTsKLX0KLQotYm9vbCBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OnJlY29yZGluZ0VuYWJsZWQoKQotewotICAgIE11dGV4OjpBdXRvbG9jayBsb2NrKG1Mb2NrKTsKLSAgICBpZiAobUhhcmR3YXJlID09IDApIHJldHVybiBmYWxzZTsKLSAgICByZXR1cm4gbUhhcmR3YXJlLT5yZWNvcmRpbmdFbmFibGVkKCk7Ci19Ci0KLS8vIFNhZmVseSByZXRyaWV2ZXMgYSBzdHJvbmcgcG9pbnRlciB0byB0aGUgY2xpZW50IGR1cmluZyBhIGhhcmR3YXJlIGNhbGxiYWNrLgotc3A8Q2FtZXJhU2VydmljZTo6Q2xpZW50PiBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OmdldENsaWVudEZyb21Db29raWUodm9pZCogdXNlcikKLXsKLSAgICBzcDxDbGllbnQ+IGNsaWVudCA9IDA7Ci0gICAgQ2FtZXJhU2VydmljZSAqc2VydmljZSA9IHN0YXRpY19jYXN0PENhbWVyYVNlcnZpY2UqPih1c2VyKTsKLSAgICBpZiAoc2VydmljZSAhPSBOVUxMKSB7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBvdXJMb2NrKHNlcnZpY2UtPm1Mb2NrKTsKLSAgICAgICAgaWYgKHNlcnZpY2UtPm1DbGllbnQgIT0gMCkgewotICAgICAgICAgICAgY2xpZW50ID0gc2VydmljZS0+bUNsaWVudC5wcm9tb3RlKCk7Ci0gICAgICAgICAgICBpZiAoY2xpZW50ID09IDApIHsKLSAgICAgICAgICAgICAgICBMT0dFKCJnZXRDbGllbnRGcm9tQ29va2llOiBjbGllbnQgYXBwZWFycyB0byBoYXZlIGRpZWQiKTsKLSAgICAgICAgICAgICAgICBzZXJ2aWNlLT5tQ2xpZW50LmNsZWFyKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBMT0dFKCJnZXRDbGllbnRGcm9tQ29va2llOiBnb3QgY2FsbGJhY2sgYnV0IGNsaWVudCB3YXMgTlVMTCIpOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBjbGllbnQ7Ci19Ci0KLQotI2lmIERFQlVHX0RVTVBfSlBFR19TTkFQU0hPVF9UT19GSUxFIHx8IFwKLSAgICBERUJVR19EVU1QX1lVVl9TTkFQU0hPVF9UT19GSUxFIHx8IFwKLSAgICBERUJVR19EVU1QX1BSRVZJRVdfRlJBTUVfVE9fRklMRQotc3RhdGljIHZvaWQgZHVtcF90b19maWxlKGNvbnN0IGNoYXIgKmZuYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQ4X3QgKmJ1ZiwgdWludDMyX3Qgc2l6ZSkKLXsKLSAgICBpbnQgbncsIGNudCA9IDA7Ci0gICAgdWludDMyX3Qgd3JpdHRlbiA9IDA7Ci0KLSAgICBMT0dEKCJvcGVuaW5nIGZpbGUgWyVzXVxuIiwgZm5hbWUpOwotICAgIGludCBmZCA9IG9wZW4oZm5hbWUsIE9fUkRXUiB8IE9fQ1JFQVQpOwotICAgIGlmIChmZCA8IDApIHsKLSAgICAgICAgTE9HRSgiZmFpbGVkIHRvIGNyZWF0ZSBmaWxlIFslc106ICVzIiwgZm5hbWUsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBMT0dEKCJ3cml0aW5nICVkIGJ5dGVzIHRvIGZpbGUgWyVzXVxuIiwgc2l6ZSwgZm5hbWUpOwotICAgIHdoaWxlICh3cml0dGVuIDwgc2l6ZSkgewotICAgICAgICBudyA9IDo6d3JpdGUoZmQsCi0gICAgICAgICAgICAgICAgICAgICBidWYgKyB3cml0dGVuLAotICAgICAgICAgICAgICAgICAgICAgc2l6ZSAtIHdyaXR0ZW4pOwotICAgICAgICBpZiAobncgPCAwKSB7Ci0gICAgICAgICAgICBMT0dFKCJmYWlsZWQgdG8gd3JpdGUgdG8gZmlsZSBbJXNdOiAlcyIsCi0gICAgICAgICAgICAgICAgIGZuYW1lLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgd3JpdHRlbiArPSBudzsKLSAgICAgICAgY250Kys7Ci0gICAgfQotICAgIExPR0QoImRvbmUgd3JpdGluZyAlZCBieXRlcyB0byBmaWxlIFslc10gaW4gJWQgcGFzc2VzXG4iLAotICAgICAgICAgc2l6ZSwgZm5hbWUsIGNudCk7Ci0gICAgOjpjbG9zZShmZCk7Ci19Ci0jZW5kaWYKLQotLy8gcHJldmlldyBjYWxsYmFjayAtIGZyYW1lIGJ1ZmZlciB1cGRhdGUKLXZvaWQgQ2FtZXJhU2VydmljZTo6Q2xpZW50OjpwcmV2aWV3Q2FsbGJhY2soY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSwgdm9pZCogdXNlcikKLXsKLSAgICBMT0dWKCJwcmV2aWV3Q2FsbGJhY2soKSIpOwotICAgIHNwPENsaWVudD4gY2xpZW50ID0gZ2V0Q2xpZW50RnJvbUNvb2tpZSh1c2VyKTsKLSAgICBpZiAoY2xpZW50ID09IDApIHsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotI2lmIERFQlVHX0hFQVBfTEVBS1MgJiYgMCAvLyBkZWJ1Z2dpbmcKLSAgICBpZiAoZ1dlYWtIZWFwID09IE5VTEwpIHsKLSAgICAgICAgc3NpemVfdCBvZmZzZXQ7Ci0gICAgICAgIHNpemVfdCBzaXplOwotICAgICAgICBzcDxJTWVtb3J5SGVhcD4gaGVhcCA9IG1lbS0+Z2V0TWVtb3J5KCZvZmZzZXQsICZzaXplKTsKLSAgICAgICAgaWYgKGdXZWFrSGVhcCAhPSBoZWFwKSB7Ci0gICAgICAgICAgICBMT0dEKCJTRVRUSU5HIFBSRVZJRVcgSEVBUCIpOwotICAgICAgICAgICAgaGVhcC0+dHJhY2tNZSh0cnVlLCB0cnVlKTsKLSAgICAgICAgICAgIGdXZWFrSGVhcCA9IGhlYXA7Ci0gICAgICAgIH0KLSAgICB9Ci0jZW5kaWYKLQotI2lmIERFQlVHX0RVTVBfUFJFVklFV19GUkFNRV9UT19GSUxFCi0gICAgewotICAgICAgICBpZiAoZGVidWdfZnJhbWVfY250KysgPT0gREVCVUdfRFVNUF9QUkVWSUVXX0ZSQU1FX1RPX0ZJTEUpIHsKLSAgICAgICAgICAgIHNzaXplX3Qgb2Zmc2V0OwotICAgICAgICAgICAgc2l6ZV90IHNpemU7Ci0gICAgICAgICAgICBzcDxJTWVtb3J5SGVhcD4gaGVhcCA9IG1lbS0+Z2V0TWVtb3J5KCZvZmZzZXQsICZzaXplKTsKLSAgICAgICAgICAgIGR1bXBfdG9fZmlsZSgiL2RhdGEvcHJldmlldy55dXYiLAotICAgICAgICAgICAgICAgICAgICAgICAgICh1aW50OF90ICopaGVhcC0+YmFzZSgpICsgb2Zmc2V0LCBzaXplKTsKLSAgICAgICAgfQotICAgIH0KLSNlbmRpZgotCi0gICAgLy8gVGhlIHN0cm9uZyBwb2ludGVyIGd1YXJhbnRlZXMgdGhlIGNsaWVudCB3aWxsIGV4aXN0LCBidXQgbm8gbG9jayBpcyBoZWxkLgotICAgIGNsaWVudC0+cG9zdFByZXZpZXdGcmFtZShtZW0pOwotCi0jaWYgREVCVUdfQ0xJRU5UX1JFRkVSRU5DRVMKLSAgICAvLyoqKiogaWYgdGhlIGNsaWVudCdzIHJlZmNvdW50IGlzIDEsIHRoZW4gd2UgYXJlIGFib3V0IHRvIGRlc3Ryb3kgaXQgaGVyZSwKLSAgICAvLyB3aGljaCBpcyBiYWQtLXByaW50IGFsbCByZWZjb3VudHMuCi0gICAgaWYgKGNsaWVudC0+Z2V0U3Ryb25nQ291bnQoKSA9PSAxKSB7Ci0gICAgICAgIExPR0UoIisrKysrKysrKysrKysrKysgKFBSRVZJRVcpIFRISVMgV0lMTCBDQVVTRSBBIExPQ0tVUCEiKTsKLSAgICAgICAgY2xpZW50LT5wcmludFJlZnMoKTsKLSAgICB9Ci0jZW5kaWYKLX0KLQotLy8gcmVjb3JkaW5nIGNhbGxiYWNrCi12b2lkIENhbWVyYVNlcnZpY2U6OkNsaWVudDo6cmVjb3JkaW5nQ2FsbGJhY2soY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSwgdm9pZCogdXNlcikKLXsKLSAgICBMT0dWKCJyZWNvcmRpbmdDYWxsYmFjayIpOwotICAgIHNwPENsaWVudD4gY2xpZW50ID0gZ2V0Q2xpZW50RnJvbUNvb2tpZSh1c2VyKTsKLSAgICBpZiAoY2xpZW50ID09IDApIHsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICAvLyBUaGUgc3Ryb25nIHBvaW50ZXIgZ3VhcmFudGVlcyB0aGUgY2xpZW50IHdpbGwgZXhpc3QsIGJ1dCBubyBsb2NrIGlzIGhlbGQuCi0gICAgY2xpZW50LT5wb3N0UmVjb3JkaW5nRnJhbWUobWVtKTsKLX0KLQotLy8gdGFrZSBhIHBpY3R1cmUgLSBpbWFnZSBpcyByZXR1cm5lZCBpbiBjYWxsYmFjawotc3RhdHVzX3QgQ2FtZXJhU2VydmljZTo6Q2xpZW50OjphdXRvRm9jdXMoKQotewotICAgIExPR1YoImF1dG9Gb2N1cyIpOwotCi0gICAgTXV0ZXg6OkF1dG9sb2NrIGxvY2sobUxvY2spOwotICAgIHN0YXR1c190IHJlc3VsdCA9IGNoZWNrUGlkKCk7Ci0gICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikgcmV0dXJuIHJlc3VsdDsKLQotICAgIGlmIChtSGFyZHdhcmUgPT0gMCkgewotICAgICAgICBMT0dFKCJtSGFyZHdhcmUgaXMgTlVMTCwgcmV0dXJuaW5nLiIpOwotICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgfQotCi0gICAgcmV0dXJuIG1IYXJkd2FyZS0+YXV0b0ZvY3VzKGF1dG9Gb2N1c0NhbGxiYWNrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtQ2FtZXJhU2VydmljZS5nZXQoKSk7Ci19Ci0KLS8vIHRha2UgYSBwaWN0dXJlIC0gaW1hZ2UgaXMgcmV0dXJuZWQgaW4gY2FsbGJhY2sKLXN0YXR1c190IENhbWVyYVNlcnZpY2U6OkNsaWVudDo6dGFrZVBpY3R1cmUoKQotewotICAgIExPR0QoInRha2VQaWN0dXJlIik7Ci0KLSAgICBNdXRleDo6QXV0b2xvY2sgbG9jayhtTG9jayk7Ci0gICAgc3RhdHVzX3QgcmVzdWx0ID0gY2hlY2tQaWQoKTsKLSAgICBpZiAocmVzdWx0ICE9IE5PX0VSUk9SKSByZXR1cm4gcmVzdWx0OwotCi0gICAgaWYgKG1IYXJkd2FyZSA9PSAwKSB7Ci0gICAgICAgIExPR0UoIm1IYXJkd2FyZSBpcyBOVUxMLCByZXR1cm5pbmcuIik7Ci0gICAgICAgIHJldHVybiBJTlZBTElEX09QRVJBVElPTjsKLSAgICB9Ci0KLSAgICBpZiAobVN1cmZhY2UgIT0gTlVMTCAmJiAhbVVzZU92ZXJsYXkpCi0gICAgICAgIG1TdXJmYWNlLT51bnJlZ2lzdGVyQnVmZmVycygpOwotCi0gICAgcmV0dXJuIG1IYXJkd2FyZS0+dGFrZVBpY3R1cmUoc2h1dHRlckNhbGxiYWNrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHl1dlBpY3R1cmVDYWxsYmFjaywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBqcGVnUGljdHVyZUNhbGxiYWNrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1DYW1lcmFTZXJ2aWNlLmdldCgpKTsKLX0KLQotLy8gcGljdHVyZSBjYWxsYmFjayAtIHNuYXBzaG90IHRha2VuCi12b2lkIENhbWVyYVNlcnZpY2U6OkNsaWVudDo6c2h1dHRlckNhbGxiYWNrKHZvaWQgKnVzZXIpCi17Ci0gICAgc3A8Q2xpZW50PiBjbGllbnQgPSBnZXRDbGllbnRGcm9tQ29va2llKHVzZXIpOwotICAgIGlmIChjbGllbnQgPT0gMCkgewotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgY2xpZW50LT5wb3N0U2h1dHRlcigpOwotfQotCi0vLyBwaWN0dXJlIGNhbGxiYWNrIC0gcmF3IGltYWdlIHJlYWR5Ci12b2lkIENhbWVyYVNlcnZpY2U6OkNsaWVudDo6eXV2UGljdHVyZUNhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQgKnVzZXIpCi17Ci0gICAgc3A8Q2xpZW50PiBjbGllbnQgPSBnZXRDbGllbnRGcm9tQ29va2llKHVzZXIpOwotICAgIGlmIChjbGllbnQgPT0gMCkgewotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmIChtZW0gPT0gTlVMTCkgewotICAgICAgICBjbGllbnQtPnBvc3RSYXcoTlVMTCk7Ci0gICAgICAgIGNsaWVudC0+cG9zdEVycm9yKFVOS05PV05fRVJST1IpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgc3NpemVfdCBvZmZzZXQ7Ci0gICAgc2l6ZV90IHNpemU7Ci0gICAgc3A8SU1lbW9yeUhlYXA+IGhlYXAgPSBtZW0tPmdldE1lbW9yeSgmb2Zmc2V0LCAmc2l6ZSk7Ci0jaWYgREVCVUdfSEVBUF9MRUFLUyAmJiAwIC8vIGRlYnVnZ2luZwotICAgIGdXZWFrSGVhcCA9IGhlYXA7IC8vIGRlYnVnZ2luZwotI2VuZGlmCi0KLSAgICAvL0xPR1YoInl1dlBpY3R1cmVDYWxsYmFjayglZCwgJWQsICVwKSIsIG9mZnNldCwgc2l6ZSwgdXNlcik7Ci0jaWYgREVCVUdfRFVNUF9ZVVZfU05BUFNIT1RfVE9fRklMRSAvLyBmb3IgdGVzdGluZyBwdXJzcG9zZXMgb25seQotICAgIGR1bXBfdG9fZmlsZSgiL2RhdGEvcGhvdG8ueXV2IiwKLSAgICAgICAgICAgICAgICAgKHVpbnQ4X3QgKiloZWFwLT5iYXNlKCkgKyBvZmZzZXQsIHNpemUpOwotI2VuZGlmCi0KLSAgICAvLyBQdXQgdGhlIFlVViB2ZXJzaW9uIG9mIHRoZSBzbmFwc2hvdCBpbiB0aGUgcHJldmlldyBkaXNwbGF5LgotICAgIGludCB3LCBoOwotICAgIENhbWVyYVBhcmFtZXRlcnMgcGFyYW1zKGNsaWVudC0+bUhhcmR3YXJlLT5nZXRQYXJhbWV0ZXJzKCkpOwotICAgIHBhcmFtcy5nZXRQaWN0dXJlU2l6ZSgmdywgJmgpOwotCi0vLyAgTXV0ZXg6OkF1dG9sb2NrIGNsaWVudExvY2soY2xpZW50LT5tTG9jayk7Ci0gICAgaWYgKGNsaWVudC0+bVN1cmZhY2UgIT0gMCAmJiAhY2xpZW50LT5tVXNlT3ZlcmxheSkgewotICAgICAgICBjbGllbnQtPm1TdXJmYWNlLT51bnJlZ2lzdGVyQnVmZmVycygpOwotICAgICAgICAKLSAgICAgICAgdWludDMyX3QgdHJhbnNmb3JtID0gMDsKLSAgICAgICAgaWYgKHBhcmFtcy5nZXRPcmllbnRhdGlvbigpID09IENhbWVyYVBhcmFtZXRlcnM6OkNBTUVSQV9PUklFTlRBVElPTl9QT1JUUkFJVCkgewotICAgICAgICAgICAgTE9HVigicG9ydHJhaXQgbW9kZSIpOwotICAgICAgICAgICAgdHJhbnNmb3JtID0gSVN1cmZhY2U6OkJ1ZmZlckhlYXA6OlJPVF85MDsKLSAgICAgICAgfQotICAgICAgICBJU3VyZmFjZTo6QnVmZmVySGVhcCBidWZmZXJzKHcsIGgsIHcsIGgsCi0gICAgICAgICAgICAgICAgUElYRUxfRk9STUFUX1lDYkNyXzQyMF9TUCwgdHJhbnNmb3JtLCAwLCBoZWFwKTsKLSAgICAgICAgCi0gICAgICAgIGNsaWVudC0+bVN1cmZhY2UtPnJlZ2lzdGVyQnVmZmVycyhidWZmZXJzKTsKLSAgICAgICAgY2xpZW50LT5tU3VyZmFjZS0+cG9zdEJ1ZmZlcihvZmZzZXQpOwotICAgIH0KLQotICAgIGNsaWVudC0+cG9zdFJhdyhtZW0pOwotCi0jaWYgREVCVUdfQ0xJRU5UX1JFRkVSRU5DRVMKLSAgICAvLyoqKiogaWYgdGhlIGNsaWVudCdzIHJlZmNvdW50IGlzIDEsIHRoZW4gd2UgYXJlIGFib3V0IHRvIGRlc3Ryb3kgaXQgaGVyZSwKLSAgICAvLyB3aGljaCBpcyBiYWQtLXByaW50IGFsbCByZWZjb3VudHMuCi0gICAgaWYgKGNsaWVudC0+Z2V0U3Ryb25nQ291bnQoKSA9PSAxKSB7Ci0gICAgICAgIExPR0UoIisrKysrKysrKysrKysrKysgKFJBVykgVEhJUyBXSUxMIENBVVNFIEEgTE9DS1VQISIpOwotICAgICAgICBjbGllbnQtPnByaW50UmVmcygpOwotICAgIH0KLSNlbmRpZgotfQotCi0vLyBwaWN0dXJlIGNhbGxiYWNrIC0ganBlZyByZWFkeQotdm9pZCBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OmpwZWdQaWN0dXJlQ2FsbGJhY2soY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSwgdm9pZCAqdXNlcikKLXsKLSAgICBzcDxDbGllbnQ+IGNsaWVudCA9IGdldENsaWVudEZyb21Db29raWUodXNlcik7Ci0gICAgaWYgKGNsaWVudCA9PSAwKSB7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKG1lbSA9PSBOVUxMKSB7Ci0gICAgICAgIGNsaWVudC0+cG9zdEpwZWcoTlVMTCk7Ci0gICAgICAgIGNsaWVudC0+cG9zdEVycm9yKFVOS05PV05fRVJST1IpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgLyoqIFdlIGFic29sdXRlbHkgQ0FOTk9UIGNhbGwgaW50byB1c2VyIGNvZGUgd2l0aCBhIGxvY2sgaGVsZCAqKi8KLQotI2lmIERFQlVHX0RVTVBfSlBFR19TTkFQU0hPVF9UT19GSUxFIC8vIGZvciB0ZXN0aW5nIHB1cnNwb3NlcyBvbmx5Ci0gICAgewotICAgICAgICBzc2l6ZV90IG9mZnNldDsKLSAgICAgICAgc2l6ZV90IHNpemU7Ci0gICAgICAgIHNwPElNZW1vcnlIZWFwPiBoZWFwID0gbWVtLT5nZXRNZW1vcnkoJm9mZnNldCwgJnNpemUpOwotICAgICAgICBkdW1wX3RvX2ZpbGUoIi9kYXRhL3Bob3RvLmpwZyIsCi0gICAgICAgICAgICAgICAgICAgICAodWludDhfdCAqKWhlYXAtPmJhc2UoKSArIG9mZnNldCwgc2l6ZSk7Ci0gICAgfQotI2VuZGlmCi0KLSAgICBjbGllbnQtPnBvc3RKcGVnKG1lbSk7Ci0KLSNpZiBERUJVR19DTElFTlRfUkVGRVJFTkNFUwotICAgIC8vKioqKiBpZiB0aGUgY2xpZW50J3MgcmVmY291bnQgaXMgMSwgdGhlbiB3ZSBhcmUgYWJvdXQgdG8gZGVzdHJveSBpdCBoZXJlLAotICAgIC8vIHdoaWNoIGlzIGJhZC0tcHJpbnQgYWxsIHJlZmNvdW50cy4KLSAgICBpZiAoY2xpZW50LT5nZXRTdHJvbmdDb3VudCgpID09IDEpIHsKLSAgICAgICAgTE9HRSgiKysrKysrKysrKysrKysrKyAoSlBFRykgVEhJUyBXSUxMIENBVVNFIEEgTE9DS1VQISIpOwotICAgICAgICBjbGllbnQtPnByaW50UmVmcygpOwotICAgIH0KLSNlbmRpZgotfQotCi12b2lkIENhbWVyYVNlcnZpY2U6OkNsaWVudDo6YXV0b0ZvY3VzQ2FsbGJhY2soYm9vbCBmb2N1c2VkLCB2b2lkICp1c2VyKQotewotICAgIExPR1YoImF1dG9Gb2N1c0NhbGxiYWNrIik7Ci0KLSAgICBzcDxDbGllbnQ+IGNsaWVudCA9IGdldENsaWVudEZyb21Db29raWUodXNlcik7Ci0gICAgaWYgKGNsaWVudCA9PSAwKSB7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBjbGllbnQtPnBvc3RBdXRvRm9jdXMoZm9jdXNlZCk7Ci0KLSNpZiBERUJVR19DTElFTlRfUkVGRVJFTkNFUwotICAgIGlmIChjbGllbnQtPmdldFN0cm9uZ0NvdW50KCkgPT0gMSkgewotICAgICAgICBMT0dFKCIrKysrKysrKysrKysrKysrIChBVVRPRk9DVVMpIFRISVMgV0lMTCBDQVVTRSBBIExPQ0tVUCEiKTsKLSAgICAgICAgY2xpZW50LT5wcmludFJlZnMoKTsKLSAgICB9Ci0jZW5kaWYKLX0KLQotLy8gc2V0IHByZXZpZXcvY2FwdHVyZSBwYXJhbWV0ZXJzIC0ga2V5L3ZhbHVlIHBhaXJzCi1zdGF0dXNfdCBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OnNldFBhcmFtZXRlcnMoY29uc3QgU3RyaW5nOCYgcGFyYW1zKQotewotICAgIExPR0QoInNldFBhcmFtZXRlcnMoJXMpIiwgcGFyYW1zLnN0cmluZygpKTsKLQotICAgIE11dGV4OjpBdXRvbG9jayBsb2NrKG1Mb2NrKTsKLSAgICBzdGF0dXNfdCByZXN1bHQgPSBjaGVja1BpZCgpOwotICAgIGlmIChyZXN1bHQgIT0gTk9fRVJST1IpIHJldHVybiByZXN1bHQ7Ci0KLSAgICBpZiAobUhhcmR3YXJlID09IDApIHsKLSAgICAgICAgTE9HRSgibUhhcmR3YXJlIGlzIE5VTEwsIHJldHVybmluZy4iKTsKLSAgICAgICAgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOwotICAgIH0KLQotICAgIENhbWVyYVBhcmFtZXRlcnMgcChwYXJhbXMpOwotICAgIG1IYXJkd2FyZS0+c2V0UGFyYW1ldGVycyhwKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLS8vIGdldCBwcmV2aWV3L2NhcHR1cmUgcGFyYW1ldGVycyAtIGtleS92YWx1ZSBwYWlycwotU3RyaW5nOCBDYW1lcmFTZXJ2aWNlOjpDbGllbnQ6OmdldFBhcmFtZXRlcnMoKSBjb25zdAotewotICAgIExPR0QoImdldFBhcmFtZXRlcnMiKTsKLQotICAgIE11dGV4OjpBdXRvbG9jayBsb2NrKG1Mb2NrKTsKLQotICAgIGlmIChtSGFyZHdhcmUgPT0gMCkgewotICAgICAgICBMT0dFKCJtSGFyZHdhcmUgaXMgTlVMTCwgcmV0dXJuaW5nLiIpOwotICAgICAgICByZXR1cm4gU3RyaW5nOCgpOwotICAgIH0KLQotICAgIHJldHVybiBtSGFyZHdhcmUtPmdldFBhcmFtZXRlcnMoKS5mbGF0dGVuKCk7Ci19Ci0KLXZvaWQgQ2FtZXJhU2VydmljZTo6Q2xpZW50Ojpwb3N0QXV0b0ZvY3VzKGJvb2wgZm9jdXNlZCkKLXsKLSAgICBMT0dWKCJwb3N0QXV0b0ZvY3VzIik7Ci0gICAgbUNhbWVyYUNsaWVudC0+YXV0b0ZvY3VzQ2FsbGJhY2soZm9jdXNlZCk7Ci19Ci0KLXZvaWQgQ2FtZXJhU2VydmljZTo6Q2xpZW50Ojpwb3N0U2h1dHRlcigpCi17Ci0gICAgbUNhbWVyYUNsaWVudC0+c2h1dHRlckNhbGxiYWNrKCk7Ci19Ci0KLXZvaWQgQ2FtZXJhU2VydmljZTo6Q2xpZW50Ojpwb3N0UmF3KGNvbnN0IHNwPElNZW1vcnk+JiBtZW0pCi17Ci0gICAgTE9HRCgicG9zdFJhdyIpOwotICAgIG1DYW1lcmFDbGllbnQtPnJhd0NhbGxiYWNrKG1lbSk7Ci19Ci0KLXZvaWQgQ2FtZXJhU2VydmljZTo6Q2xpZW50Ojpwb3N0SnBlZyhjb25zdCBzcDxJTWVtb3J5PiYgbWVtKQotewotICAgIExPR0QoInBvc3RKcGVnIik7Ci0gICAgbUNhbWVyYUNsaWVudC0+anBlZ0NhbGxiYWNrKG1lbSk7Ci19Ci0KLXZvaWQgQ2FtZXJhU2VydmljZTo6Q2xpZW50Ojpjb3B5RnJhbWVBbmRQb3N0Q29waWVkRnJhbWUoc3A8SU1lbW9yeUhlYXA+IGhlYXAsIHNpemVfdCBvZmZzZXQsIHNpemVfdCBzaXplKQotewotICAgIExPR1YoImNvcHlGcmFtZUFuZFBvc3RDb3BpZWRGcmFtZSIpOwotICAgIC8vIEl0IGlzIG5lY2Vzc2FyeSB0byBjb3B5IG91dCBvZiBwbWVtIGJlZm9yZSBzZW5kaW5nIHRoaXMgdG8KLSAgICAvLyB0aGUgY2FsbGJhY2suIEZvciBlZmZpY2llbmN5LCByZXVzZSB0aGUgc2FtZSBNZW1vcnlIZWFwQmFzZQotICAgIC8vIHByb3ZpZGVkIGl0J3MgYmlnIGVub3VnaC4gRG9uJ3QgYWxsb2NhdGUgdGhlIG1lbW9yeSBvcgotICAgIC8vIHBlcmZvcm0gdGhlIGNvcHkgaWYgdGhlcmUncyBubyBjYWxsYmFjay4KLSAgICBpZiAobVByZXZpZXdCdWZmZXIgPT0gMCkgewotICAgICAgICBtUHJldmlld0J1ZmZlciA9IG5ldyBNZW1vcnlIZWFwQmFzZShzaXplLCAwLCBOVUxMKTsKLSAgICB9IGVsc2UgaWYgKHNpemUgPiBtUHJldmlld0J1ZmZlci0+dmlydHVhbFNpemUoKSkgewotICAgICAgICBtUHJldmlld0J1ZmZlci5jbGVhcigpOwotICAgICAgICBtUHJldmlld0J1ZmZlciA9IG5ldyBNZW1vcnlIZWFwQmFzZShzaXplLCAwLCBOVUxMKTsKLSAgICAgICAgaWYgKG1QcmV2aWV3QnVmZmVyID09IDApIHsKLSAgICAgICAgICAgIExPR0UoImZhaWxlZCB0byBhbGxvY2F0ZSBzcGFjZSBmb3IgcHJldmlldyBidWZmZXIiKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgIH0KLSAgICBtZW1jcHkobVByZXZpZXdCdWZmZXItPmJhc2UoKSwKLSAgICAgICAgICAgKHVpbnQ4X3QgKiloZWFwLT5iYXNlKCkgKyBvZmZzZXQsIHNpemUpOwotCi0gICAgc3A8TWVtb3J5QmFzZT4gZnJhbWUgPSBuZXcgTWVtb3J5QmFzZShtUHJldmlld0J1ZmZlciwgMCwgc2l6ZSk7Ci0gICAgaWYgKGZyYW1lID09IDApIHsKLSAgICAgICAgTE9HRSgiZmFpbGVkIHRvIGFsbG9jYXRlIHNwYWNlIGZvciBmcmFtZSBjYWxsYmFjayIpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIG1DYW1lcmFDbGllbnQtPnByZXZpZXdDYWxsYmFjayhmcmFtZSk7Ci19Ci0KLXZvaWQgQ2FtZXJhU2VydmljZTo6Q2xpZW50Ojpwb3N0UmVjb3JkaW5nRnJhbWUoY29uc3Qgc3A8SU1lbW9yeT4mIGZyYW1lKQotewotICAgIExPR1YoInBvc3RSZWNvcmRpbmdGcmFtZSIpOwotICAgIGlmIChmcmFtZSA9PSAwKSB7Ci0gICAgICAgIExPR1coImZyYW1lIGlzIGEgbnVsbCBwb2ludGVyIik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgbUNhbWVyYUNsaWVudC0+cmVjb3JkaW5nQ2FsbGJhY2soZnJhbWUpOwotfQotCi12b2lkIENhbWVyYVNlcnZpY2U6OkNsaWVudDo6cG9zdFByZXZpZXdGcmFtZShjb25zdCBzcDxJTWVtb3J5PiYgbWVtKQotewotICAgIExPR1YoInBvc3RQcmV2aWV3RnJhbWUiKTsKLSAgICBpZiAobWVtID09IDApIHsKLSAgICAgICAgTE9HVygibWVtIGlzIGEgbnVsbCBwb2ludGVyIik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBzc2l6ZV90IG9mZnNldDsKLSAgICBzaXplX3Qgc2l6ZTsKLSAgICBzcDxJTWVtb3J5SGVhcD4gaGVhcCA9IG1lbS0+Z2V0TWVtb3J5KCZvZmZzZXQsICZzaXplKTsKLSAgICB7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBzdXJmYWNlTG9jayhtU3VyZmFjZUxvY2spOwotICAgICAgICBpZiAobVN1cmZhY2UgIT0gTlVMTCkgewotICAgICAgICAgICAgbVN1cmZhY2UtPnBvc3RCdWZmZXIob2Zmc2V0KTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vIElzIHRoZSBjYWxsYmFjayBlbmFibGVkIG9yIG5vdD8KLSAgICBpZiAoIShtUHJldmlld0NhbGxiYWNrRmxhZyAmIEZSQU1FX0NBTExCQUNLX0ZMQUdfRU5BQkxFX01BU0spKSB7Ci0gICAgICAgIC8vIElmIHRoZSBlbmFibGUgYml0IGlzIG9mZiwgdGhlIGNvcHktb3V0IGFuZCBvbmUtc2hvdCBiaXRzIGFyZSBpZ25vcmVkCi0gICAgICAgIExPR1YoImZyYW1lIGNhbGxiYWNrIGlzIGRpYWJsZWQiKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIC8vIElzIHRoZSByZWNlaXZlZCBmcmFtZSBjb3BpZWQgb3V0IG9yIG5vdD8KLSAgICBpZiAobVByZXZpZXdDYWxsYmFja0ZsYWcgJiBGUkFNRV9DQUxMQkFDS19GTEFHX0NPUFlfT1VUX01BU0spIHsKLSAgICAgICAgTE9HVigiZnJhbWUgaXMgY29waWVkIG91dCIpOwotICAgICAgICBjb3B5RnJhbWVBbmRQb3N0Q29waWVkRnJhbWUoaGVhcCwgb2Zmc2V0LCBzaXplKTsKLSAgICB9IGVsc2UgewotICAgICAgICBMT0dWKCJmcmFtZSBpcyBkaXJlY3RseSBzZW50IG91dCB3aXRob3V0IGNvcHlpbmciKTsKLSAgICAgICAgbUNhbWVyYUNsaWVudC0+cHJldmlld0NhbGxiYWNrKG1lbSk7Ci0gICAgfQotCi0gICAgLy8gSXMgdGhpcyBpcyBvbmUtc2hvdCBvbmx5PwotICAgIGlmIChtUHJldmlld0NhbGxiYWNrRmxhZyAmIEZSQU1FX0NBTExCQUNLX0ZMQUdfT05FX1NIT1RfTUFTSykgewotICAgICAgICBMT0dWKCJPbmUtc2hvdCBvbmx5LCB0aHVzIGNsZWFyIHRoZSBiaXRzIGFuZCBkaXNhYmxlIGZyYW1lIGNhbGxiYWNrIik7Ci0gICAgICAgIG1QcmV2aWV3Q2FsbGJhY2tGbGFnICY9IH4oRlJBTUVfQ0FMTEJBQ0tfRkxBR19PTkVfU0hPVF9NQVNLIHwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRlJBTUVfQ0FMTEJBQ0tfRkxBR19DT1BZX09VVF9NQVNLIHwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRlJBTUVfQ0FMTEJBQ0tfRkxBR19FTkFCTEVfTUFTSyk7Ci0gICAgfQotfQotCi12b2lkIENhbWVyYVNlcnZpY2U6OkNsaWVudDo6cG9zdEVycm9yKHN0YXR1c190IGVycm9yKQotewotICAgIG1DYW1lcmFDbGllbnQtPmVycm9yQ2FsbGJhY2soZXJyb3IpOwotfQotCi1zdGF0dXNfdCBDYW1lcmFTZXJ2aWNlOjpkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBjb25zdCBzaXplX3QgU0laRSA9IDI1NjsKLSAgICBjaGFyIGJ1ZmZlcltTSVpFXTsKLSAgICBTdHJpbmc4IHJlc3VsdDsKLSAgICBpZiAoY2hlY2tDYWxsaW5nUGVybWlzc2lvbihTdHJpbmcxNigiYW5kcm9pZC5wZXJtaXNzaW9uLkRVTVAiKSkgPT0gZmFsc2UpIHsKLSAgICAgICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiUGVybWlzc2lvbiBEZW5pYWw6ICIKLSAgICAgICAgICAgICAgICAiY2FuJ3QgZHVtcCBDYW1lcmFTZXJ2aWNlIGZyb20gcGlkPSVkLCB1aWQ9JWRcbiIsCi0gICAgICAgICAgICAgICAgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpLAotICAgICAgICAgICAgICAgIElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPmdldENhbGxpbmdVaWQoKSk7Ci0gICAgICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICAgICAgd3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgQXV0b011dGV4IGxvY2soJm1Mb2NrKTsKLSAgICAgICAgaWYgKG1DbGllbnQgIT0gMCkgewotICAgICAgICAgICAgc3A8Q2xpZW50PiBjdXJyZW50Q2xpZW50ID0gbUNsaWVudC5wcm9tb3RlKCk7Ci0gICAgICAgICAgICBzcHJpbnRmKGJ1ZmZlciwgIkNsaWVudCAoJXApIFBJRDogJWQiLAotICAgICAgICAgICAgICAgICAgICBjdXJyZW50Q2xpZW50LT5nZXRDYW1lcmFDbGllbnQoKS0+YXNCaW5kZXIoKS5nZXQoKSwKLSAgICAgICAgICAgICAgICAgICAgY3VycmVudENsaWVudC0+bUNsaWVudFBpZCk7Ci0gICAgICAgICAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgICAgICAgICB3cml0ZShmZCwgcmVzdWx0LnN0cmluZygpLCByZXN1bHQuc2l6ZSgpKTsKLSAgICAgICAgICAgIGN1cnJlbnRDbGllbnQtPm1IYXJkd2FyZS0+ZHVtcChmZCwgYXJncyk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICByZXN1bHQuYXBwZW5kKCJObyBjYW1lcmEgY2xpZW50IHlldC5cbiIpOwotICAgICAgICAgICAgd3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0KLSNpZiBERUJVR19IRUFQX0xFQUtTCi0KLSNkZWZpbmUgQ0hFQ0tfSU5URVJGQUNFKGludGVyZmFjZSwgZGF0YSwgcmVwbHkpIFwKLSAgICAgICAgZG8geyBpZiAoIWRhdGEuZW5mb3JjZUludGVyZmFjZShpbnRlcmZhY2U6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSkpIHsgXAotICAgICAgICAgICAgTE9HVygiQ2FsbCBpbmNvcnJlY3RseSByb3V0ZWQgdG8gIiAjaW50ZXJmYWNlKTsgXAotICAgICAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOyBcCi0gICAgICAgIH0gfSB3aGlsZSAoMCkKLQotc3RhdHVzX3QgQ2FtZXJhU2VydmljZTo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIC8vIHBlcm1pc3Npb24gY2hlY2tzLi4uCi0gICAgc3dpdGNoIChjb2RlKSB7Ci0gICAgICAgIGNhc2UgQm5DYW1lcmFTZXJ2aWNlOjpDT05ORUNUOgotICAgICAgICAgICAgSVBDVGhyZWFkU3RhdGUqIGlwYyA9IElQQ1RocmVhZFN0YXRlOjpzZWxmKCk7Ci0gICAgICAgICAgICBjb25zdCBpbnQgcGlkID0gaXBjLT5nZXRDYWxsaW5nUGlkKCk7Ci0gICAgICAgICAgICBjb25zdCBpbnQgc2VsZl9waWQgPSBnZXRwaWQoKTsKLSAgICAgICAgICAgIGlmIChwaWQgIT0gc2VsZl9waWQpIHsKLSAgICAgICAgICAgICAgICAvLyB3ZSdyZSBjYWxsZWQgZnJvbSBhIGRpZmZlcmVudCBwcm9jZXNzLCBkbyB0aGUgcmVhbCBjaGVjawotICAgICAgICAgICAgICAgIGlmICghY2hlY2tDYWxsaW5nUGVybWlzc2lvbigKLSAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzE2KCJhbmRyb2lkLnBlcm1pc3Npb24uQ0FNRVJBIikpKQotICAgICAgICAgICAgICAgIHsKLSAgICAgICAgICAgICAgICAgICAgY29uc3QgaW50IHVpZCA9IGlwYy0+Z2V0Q2FsbGluZ1VpZCgpOwotICAgICAgICAgICAgICAgICAgICBMT0dFKCJQZXJtaXNzaW9uIERlbmlhbDogIgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICJjYW4ndCB1c2UgdGhlIGNhbWVyYSBwaWQ9JWQsIHVpZD0lZCIsIHBpZCwgdWlkKTsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGJyZWFrOwotICAgIH0KLQotICAgIHN0YXR1c190IGVyciA9IEJuQ2FtZXJhU2VydmljZTo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotCi0gICAgTE9HRCgiKysrIG9uVHJhbnNhY3QgZXJyICVkIGNvZGUgJWQiLCBlcnIsIGNvZGUpOwotCi0gICAgaWYgKGVyciA9PSBVTktOT1dOX1RSQU5TQUNUSU9OIHx8IGVyciA9PSBQRVJNSVNTSU9OX0RFTklFRCkgewotICAgICAgICAvLyB0aGUgJ3NlcnZpY2UnIGNvbW1hbmQgaW50ZXJyb2dhdGVzIHRoaXMgYmluZGVyIGZvciBpdHMgbmFtZSwgYW5kIHRoZW4gc3VwcGxpZXMgaXQKLSAgICAgICAgLy8gZXZlbiBmb3IgdGhlIGRlYnVnZ2luZyBjb21tYW5kcy4gIHRoYXQgbWVhbnMgd2UgbmVlZCB0byBjaGVjayBmb3IgaXQgaGVyZSwgdXNpbmcKLSAgICAgICAgLy8gSVN1cmZhY2VDb21wb3NlciAoc2luY2Ugd2UgZGVsZWdhdGVkIHRoZSBJTlRFUkZBQ0VfVFJBTlNBQ1RJT04gaGFuZGxpbmcgdG8KLSAgICAgICAgLy8gQm5TdXJmYWNlQ29tcG9zZXIgYmVmb3JlIGZhbGxpbmcgdGhyb3VnaCB0byB0aGlzIGNvZGUpLgotCi0gICAgICAgIExPR0QoIisrKyBvblRyYW5zYWN0IGNvZGUgJWQiLCBjb2RlKTsKLQotICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSUNhbWVyYVNlcnZpY2UsIGRhdGEsIHJlcGx5KTsKLQotICAgICAgICBzd2l0Y2goY29kZSkgewotICAgICAgICBjYXNlIDEwMDA6Ci0gICAgICAgIHsKLSAgICAgICAgICAgIGlmIChnV2Vha0hlYXAgIT0gMCkgewotICAgICAgICAgICAgICAgIHNwPElNZW1vcnlIZWFwPiBoID0gZ1dlYWtIZWFwLnByb21vdGUoKTsKLSAgICAgICAgICAgICAgICBJTWVtb3J5SGVhcCAqcCA9IGdXZWFrSGVhcC51bnNhZmVfZ2V0KCk7Ci0gICAgICAgICAgICAgICAgTE9HRCgiQ0hFQ0tJTkcgV0VBSyBSRUZFUkVOQ0UgJXAgKCVwKSIsIGguZ2V0KCksIHApOwotICAgICAgICAgICAgICAgIGlmIChoICE9IDApCi0gICAgICAgICAgICAgICAgICAgIGgtPnByaW50UmVmcygpOwotICAgICAgICAgICAgICAgIGJvb2wgYXR0ZW1wdF90b19kZWxldGUgPSBkYXRhLnJlYWRJbnQzMigpID09IDE7Ci0gICAgICAgICAgICAgICAgaWYgKGF0dGVtcHRfdG9fZGVsZXRlKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIE5PVCBTQUZFIQotICAgICAgICAgICAgICAgICAgICBMT0dEKCJERUxFVElORyBXRUFLIFJFRkVSRU5DRSAlcCAoJXApIiwgaC5nZXQoKSwgcCk7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChwKSBkZWxldGUgcDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotI2VuZGlmIC8vIERFQlVHX0hFQVBfTEVBS1MKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0NhbWVyYVNlcnZpY2UuaCBiL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0NhbWVyYVNlcnZpY2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDliNzkyNy4uMDAwMDAwMAotLS0gYS9jYW1lcmEvbGliY2FtZXJhc2VydmljZS9DYW1lcmFTZXJ2aWNlLmgKKysrIC9kZXYvbnVsbApAQCAtMSwyMDYgKzAsMCBAQAotLyoKLSoqCi0qKiBDb3B5cmlnaHQgKEMpIDIwMDgsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqIENvcHlyaWdodCAoQykgMjAwOCBIVEMgSW5jLgotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2lmbmRlZiBBTkRST0lEX1NFUlZFUlNfQ0FNRVJBX0NBTUVSQVNFUlZJQ0VfSAotI2RlZmluZSBBTkRST0lEX1NFUlZFUlNfQ0FNRVJBX0NBTUVSQVNFUlZJQ0VfSAotCi0jaW5jbHVkZSA8dWkvSUNhbWVyYVNlcnZpY2UuaD4KLSNpbmNsdWRlIDx1aS9DYW1lcmFIYXJkd2FyZUludGVyZmFjZS5oPgotI2luY2x1ZGUgPHVpL0NhbWVyYS5oPgotCi1jbGFzcyBhbmRyb2lkOjpNZW1vcnlIZWFwQmFzZTsKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNkZWZpbmUgTElLRUxZKCBleHAgKSAgICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgdHJ1ZSAgKSkKLSNkZWZpbmUgVU5MSUtFTFkoIGV4cCApICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgZmFsc2UgKSkKLQotLy8gV2hlbiBlbmFibGVkLCB0aGlzIGZlYXR1cmUgYWxsb3dzIHlvdSB0byBzZW5kIGFuIGV2ZW50IHRvIHRoZSBDYW1lcmFTZXJ2aWNlCi0vLyBzbyB0aGF0IHlvdSBjYW4gY2F1c2UgYWxsIHJlZmVyZW5jZXMgdG8gdGhlIGhlYXAgb2JqZWN0IGdXZWFrSGVhcCwgZGVmaW5lZAotLy8gYmVsb3csIHRvIGJlIHByaW50ZWQuIFlvdSB3aWxsIGFsc28gbmVlZCB0byBzZXQgREVCVUdfUkVGUz0xIGFuZAotLy8gREVCVUdfUkVGU19FTkFCTEVEX0JZX0RFRkFVTFQ9MCBpbiBsaWJ1dGlscy9SZWZCYXNlLmNwcC4gWW91IGp1c3QgaGF2ZSB0bwotLy8gc2V0IGdXZWFrSGVhcCB0byB0aGUgYXBwcm9wcmlhdGUgaGVhcCB5b3Ugd2FudCB0byB0cmFjay4KLQotI2RlZmluZSBERUJVR19IRUFQX0xFQUtTIDAKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBDYW1lcmFTZXJ2aWNlIDogcHVibGljIEJuQ2FtZXJhU2VydmljZQotewotICAgIGNsYXNzIENsaWVudDsKLQotcHVibGljOgotICAgIHN0YXRpYyB2b2lkIGluc3RhbnRpYXRlKCk7Ci0KLSAgICAvLyBJQ2FtZXJhU2VydmljZSBpbnRlcmZhY2UKLSAgICB2aXJ0dWFsIHNwPElDYW1lcmE+ICAgICBjb25uZWN0KGNvbnN0IHNwPElDYW1lcmFDbGllbnQ+JiBjYW1lcmFDbGllbnQpOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgZHVtcChpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpOwotCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgcmVtb3ZlQ2xpZW50KGNvbnN0IHNwPElDYW1lcmFDbGllbnQ+JiBjYW1lcmFDbGllbnQpOwotCi0jaWYgREVCVUdfSEVBUF9MRUFLUwotICAgIHZpcnR1YWwgc3RhdHVzX3Qgb25UcmFuc2FjdCgKLSAgICAgICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncyk7Ci0jZW5kaWYKLQotcHJpdmF0ZToKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0gICAgY2xhc3MgQ2xpZW50IDogcHVibGljIEJuQ2FtZXJhIHsKLQotICAgIHB1YmxpYzoKLSAgICAgICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgZGlzY29ubmVjdCgpOwotCi0gICAgICAgIC8vIGNvbm5lY3QgbmV3IGNsaWVudCB3aXRoIGV4aXN0aW5nIGNhbWVyYSByZW1vdGUKLSAgICAgICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgY29ubmVjdChjb25zdCBzcDxJQ2FtZXJhQ2xpZW50PiYgY2xpZW50KTsKLQotICAgICAgICAvLyBwcmV2ZW50IG90aGVyIHByb2Nlc3NlcyBmcm9tIHVzaW5nIHRoaXMgSUNhbWVyYSBpbnRlcmZhY2UKLSAgICAgICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgbG9jaygpOwotCi0gICAgICAgIC8vIGFsbG93IG90aGVyIHByb2Nlc3NlcyB0byB1c2UgdGhpcyBJQ2FtZXJhIGludGVyZmFjZQotICAgICAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICB1bmxvY2soKTsKLQotICAgICAgICAvLyBwYXNzIHRoZSBidWZmZXJlZCBJU3VyZmFjZSB0byB0aGUgY2FtZXJhIHNlcnZpY2UKLSAgICAgICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgc2V0UHJldmlld0Rpc3BsYXkoY29uc3Qgc3A8SVN1cmZhY2U+JiBzdXJmYWNlKTsKLQotICAgICAgICAvLyBzZXQgdGhlIHByZXZpZXcgY2FsbGJhY2sgZmxhZyB0byBhZmZlY3QgaG93IHRoZSByZWNlaXZlZCBmcmFtZXMgZnJvbQotICAgICAgICAvLyBwcmV2aWV3IGFyZSBoYW5kbGVkLgotICAgICAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICBzZXRQcmV2aWV3Q2FsbGJhY2tGbGFnKGludCBjYWxsYmFja19mbGFnKTsKLQotICAgICAgICAvLyBzdGFydCBwcmV2aWV3IG1vZGUsIG11c3QgY2FsbCBzZXRQcmV2aWV3RGlzcGxheSBmaXJzdAotICAgICAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICBzdGFydFByZXZpZXcoKTsKLQotICAgICAgICAvLyBzdG9wIHByZXZpZXcgbW9kZQotICAgICAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICBzdG9wUHJldmlldygpOwotCi0gICAgICAgIC8vIGdldCBwcmV2aWV3IHN0YXRlCi0gICAgICAgIHZpcnR1YWwgYm9vbCAgICAgICAgICAgIHByZXZpZXdFbmFibGVkKCk7Ci0KLSAgICAgICAgLy8gc3RhcnQgcmVjb3JkaW5nIG1vZGUKLSAgICAgICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgc3RhcnRSZWNvcmRpbmcoKTsKLQotICAgICAgICAvLyBzdG9wIHJlY29yZGluZyBtb2RlCi0gICAgICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHN0b3BSZWNvcmRpbmcoKTsKLQotICAgICAgICAvLyBnZXQgcmVjb3JkaW5nIHN0YXRlCi0gICAgICAgIHZpcnR1YWwgYm9vbCAgICAgICAgICAgIHJlY29yZGluZ0VuYWJsZWQoKTsKLQotICAgICAgICAvLyByZWxlYXNlIGEgcmVjb3JkaW5nIGZyYW1lCi0gICAgICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHJlbGVhc2VSZWNvcmRpbmdGcmFtZShjb25zdCBzcDxJTWVtb3J5PiYgbWVtKTsKLQotICAgICAgICAvLyBhdXRvIGZvY3VzCi0gICAgICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgIGF1dG9Gb2N1cygpOwotCi0gICAgICAgIC8vIHRha2UgYSBwaWN0dXJlIC0gcmV0dXJucyBhbiBJTWVtb3J5IChyZWYtY291bnRlZCBtbWFwKQotICAgICAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICB0YWtlUGljdHVyZSgpOwotCi0gICAgICAgIC8vIHNldCBwcmV2aWV3L2NhcHR1cmUgcGFyYW1ldGVycyAtIGtleS92YWx1ZSBwYWlycwotICAgICAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICBzZXRQYXJhbWV0ZXJzKGNvbnN0IFN0cmluZzgmIHBhcmFtcyk7Ci0KLSAgICAgICAgLy8gZ2V0IHByZXZpZXcvY2FwdHVyZSBwYXJhbWV0ZXJzIC0ga2V5L3ZhbHVlIHBhaXJzCi0gICAgICAgIHZpcnR1YWwgU3RyaW5nOCAgICAgICAgIGdldFBhcmFtZXRlcnMoKSBjb25zdDsKLQotICAgICAgICAvLyBvdXIgY2xpZW50Li4uCi0gICAgICAgIGNvbnN0IHNwPElDYW1lcmFDbGllbnQ+JiAgICBnZXRDYW1lcmFDbGllbnQoKSBjb25zdCB7IHJldHVybiBtQ2FtZXJhQ2xpZW50OyB9Ci0KLSAgICBwcml2YXRlOgotICAgICAgICBmcmllbmQgY2xhc3MgQ2FtZXJhU2VydmljZTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ2xpZW50KGNvbnN0IHNwPENhbWVyYVNlcnZpY2U+JiBjYW1lcmFTZXJ2aWNlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHNwPElDYW1lcmFDbGllbnQ+JiBjYW1lcmFDbGllbnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcGlkX3QgY2xpZW50UGlkKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ2xpZW50KCk7Ci0gICAgICAgIHZpcnR1YWwgICAgICAgICAgICAgICAgIH5DbGllbnQoKTsKLQotICAgICAgICAgICAgICAgICAgICBzdGF0dXNfdCAgICBjaGVja1BpZCgpOwotCi0gICAgICAgIHN0YXRpYyAgICAgIHZvaWQgICAgICAgIHJlY29yZGluZ0NhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0sIHZvaWQqIHVzZXIpOwotICAgICAgICBzdGF0aWMgICAgICB2b2lkICAgICAgICBwcmV2aWV3Q2FsbGJhY2soY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSwgdm9pZCogdXNlcik7Ci0gICAgICAgIHN0YXRpYyAgICAgIHZvaWQgICAgICAgIHNodXR0ZXJDYWxsYmFjayh2b2lkICp1c2VyKTsKLSAgICAgICAgc3RhdGljICAgICAgdm9pZCAgICAgICAgeXV2UGljdHVyZUNhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0sIHZvaWQqIHVzZXIpOwotICAgICAgICBzdGF0aWMgICAgICB2b2lkICAgICAgICBqcGVnUGljdHVyZUNhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0sIHZvaWQqIHVzZXIpOwotICAgICAgICBzdGF0aWMgICAgICB2b2lkICAgICAgICBhdXRvRm9jdXNDYWxsYmFjayhib29sIGZvY3VzZWQsIHZvaWQqIHVzZXIpOwotICAgICAgICBzdGF0aWMgICAgICBzcDxDbGllbnQ+ICBnZXRDbGllbnRGcm9tQ29va2llKHZvaWQqIHVzZXIpOwotCi0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIHBvc3RTaHV0dGVyKCk7Ci0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIHBvc3RSYXcoY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSk7Ci0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIHBvc3RKcGVnKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0pOwotICAgICAgICAgICAgICAgICAgICB2b2lkICAgICAgICBwb3N0UHJldmlld0ZyYW1lKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0pOwotICAgICAgICAgICAgICAgICAgICB2b2lkICAgICAgICBwb3N0UmVjb3JkaW5nRnJhbWUoY29uc3Qgc3A8SU1lbW9yeT4mIGZyYW1lKTsKLSAgICAgICAgICAgICAgICAgICAgdm9pZCAgICAgICAgY29weUZyYW1lQW5kUG9zdENvcGllZEZyYW1lKHNwPElNZW1vcnlIZWFwPiBoZWFwLCBzaXplX3Qgb2Zmc2V0LCBzaXplX3Qgc2l6ZSk7Ci0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIHBvc3RFcnJvcihzdGF0dXNfdCBlcnJvcik7Ci0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIHBvc3RBdXRvRm9jdXMoYm9vbCBmb2N1c2VkKTsKLQotICAgICAgICAvLyBjYW1lcmEgb3BlcmF0aW9uIG1vZGUKLSAgICAgICAgZW51bSBjYW1lcmFfbW9kZSB7Ci0gICAgICAgICAgICBDQU1FUkFfUFJFVklFV19NT0RFICAgPSAwLCAgLy8gZnJhbWUgYXV0b21hdGljYWxseSByZWxlYXNlZAotICAgICAgICAgICAgQ0FNRVJBX1JFQ09SRElOR19NT0RFID0gMSwgIC8vIGZyYW1lIGhhcyB0byBiZSBleHBsaWNpdGx5IHJlbGVhc2VkIGJ5IHJlbGVhc2VSZWNvcmRpbmdGcmFtZSgpCi0gICAgICAgIH07Ci0gICAgICAgIHN0YXR1c190ICAgICAgICAgICAgICAgIHN0YXJ0Q2FtZXJhTW9kZShjYW1lcmFfbW9kZSBtb2RlKTsKLSAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICAgICAgc3RhcnRQcmV2aWV3TW9kZSgpOwotICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgICAgICBzdGFydFJlY29yZGluZ01vZGUoKTsKLQotICAgICAgICAvLyBFbnN1cmVzIGF0b21pY2l0eSBhbW9uZyB0aGUgcHVibGljIG1ldGhvZHMKLSAgICAgICAgbXV0YWJsZSAgICAgTXV0ZXggICAgICAgICAgICAgICAgICAgICAgIG1Mb2NrOwotCi0gICAgICAgIC8vIG1TdXJmYWNlTG9jayBzeW5jaHJvbml6ZXMgYWNjZXNzIHRvIG1TdXJmYWNlIGJldHdlZW4KLSAgICAgICAgLy8gc2V0UHJldmlld1N1cmZhY2UoKSBhbmQgcG9zdFByZXZpZXdGcmFtZSgpLiAgTm90ZSB0aGF0IGFtb25nCi0gICAgICAgIC8vIHRoZSBwdWJsaWMgbWV0aG9kcywgYWxsIGFjY2Vzc2VzIHRvIG1TdXJmYWNlIGFyZQotICAgICAgICAvLyBzeW5jcmhvbml6ZWQgYnkgbUxvY2suICBIb3dldmVyLCBwb3N0UHJldmlld0ZyYW1lKCkgaXMgY2FsbGVkCi0gICAgICAgIC8vIGJ5IHRoZSBDYW1lcmFIYXJkd2FyZUludGVyZmFjZSBjYWxsYmFjaywgYW5kIG5lZWRzIHRvCi0gICAgICAgIC8vIGFjY2VzcyBtU3VyZmFjZS4gIEl0IGNhbm5vdCBob2xkIG1Mb2NrLCBob3dldmVyLCBiZWNhdXNlCi0gICAgICAgIC8vIHN0b3BQcmV2aWV3KCkgbWF5IGJlIGhvbGRpbmcgdGhhdCBsb2NrIHdoaWxlIGF0dGVtcHRpbmcKLSAgICAgICAgLy8gdG8gc3RvcCBwcmV2aWV3LCBhbmQgc3RvcFByZXZpZXcgaXRzZWxmIHdpbGwgYmxvY2sgd2FpdGluZwotICAgICAgICAvLyBmb3IgYSBjYWxsYmFjayBmcm9tIENhbWVyYUhhcmR3YXJlSW50ZXJmYWNlLiAgSWYgdGhpcwotICAgICAgICAvLyBoYXBwZW5zLCBpdCB3aWxsIGNhdXNlIGEgZGVhZGxvY2suCi0gICAgICAgIG11dGFibGUgICAgIE11dGV4ICAgICAgICAgICAgICAgICAgICAgICBtU3VyZmFjZUxvY2s7Ci0gICAgICAgIG11dGFibGUgICAgIENvbmRpdGlvbiAgICAgICAgICAgICAgICAgICBtUmVhZHk7Ci0gICAgICAgICAgICAgICAgICAgIHNwPENhbWVyYVNlcnZpY2U+ICAgICAgICAgICBtQ2FtZXJhU2VydmljZTsKLSAgICAgICAgICAgICAgICAgICAgc3A8SVN1cmZhY2U+ICAgICAgICAgICAgICAgIG1TdXJmYWNlOwotICAgICAgICAgICAgICAgICAgICBzcDxNZW1vcnlIZWFwQmFzZT4gICAgICAgICAgbVByZXZpZXdCdWZmZXI7Ci0gICAgICAgICAgICAgICAgICAgIGludCAgICAgICAgICAgICAgICAgICAgICAgICBtUHJldmlld0NhbGxiYWNrRmxhZzsKLQotICAgICAgICAgICAgICAgICAgICAvLyB0aGVzZSBhcmUgaW1tdXRhYmxlIG9uY2UgdGhlIG9iamVjdCBpcyBjcmVhdGVkLAotICAgICAgICAgICAgICAgICAgICAvLyB0aGV5IGRvbid0IG5lZWQgdG8gYmUgcHJvdGVjdGVkIGJ5IGEgbG9jawotICAgICAgICAgICAgICAgICAgICBzcDxJQ2FtZXJhQ2xpZW50PiAgICAgICAgICAgbUNhbWVyYUNsaWVudDsKLSAgICAgICAgICAgICAgICAgICAgc3A8Q2FtZXJhSGFyZHdhcmVJbnRlcmZhY2U+IG1IYXJkd2FyZTsKLSAgICAgICAgICAgICAgICAgICAgcGlkX3QgICAgICAgICAgICAgICAgICAgICAgIG1DbGllbnRQaWQ7Ci0gICAgICAgICAgICAgICAgICAgIGJvb2wgICAgICAgICAgICAgICAgICAgICAgICBtVXNlT3ZlcmxheTsKLSAgICB9OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBDYW1lcmFTZXJ2aWNlKCk7Ci0gICAgdmlydHVhbCAgICAgICAgICAgICAgICAgfkNhbWVyYVNlcnZpY2UoKTsKLQotICAgIG11dGFibGUgICAgIE11dGV4ICAgICAgICAgICAgICAgICAgICAgICBtTG9jazsKLSAgICAgICAgICAgICAgICB3cDxDbGllbnQ+ICAgICAgICAgICAgICAgICAgbUNsaWVudDsKLQotI2lmIERFQlVHX0hFQVBfTEVBS1MKLSAgICAgICAgICAgICAgICB3cDxJTWVtb3J5SGVhcD4gICAgICAgICAgICAgZ1dlYWtIZWFwOwotI2VuZGlmCi19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZgpkaWZmIC0tZ2l0IGEvY2FtZXJhL2xpYmNhbWVyYXNlcnZpY2UvQ2FubmVkSnBlZy5oIGIvY2FtZXJhL2xpYmNhbWVyYXNlcnZpY2UvQ2FubmVkSnBlZy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1MzI1NjBhLi4wMDAwMDAwCi0tLSBhL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0Nhbm5lZEpwZWcuaAorKysgL2Rldi9udWxsCkBAIC0xLDE1NDYgKzAsMCBAQAotY29uc3QgaW50IGtDYW5uZWRKcGVnV2lkdGggPSAyMTM7Ci1jb25zdCBpbnQga0Nhbm5lZEpwZWdIZWlnaHQgPSAzNTA7Ci1jb25zdCBpbnQga0Nhbm5lZEpwZWdTaXplID0gMTg0NzQ7Ci0KLWNvbnN0IGNoYXIga0Nhbm5lZEpwZWdbXSA9IHsKLSAgMHhmZiwgMHhkOCwgMHhmZiwgMHhlMCwgMHgwMCwgMHgxMCwgMHg0YSwgMHg0NiwgMHg0OSwgMHg0NiwgMHgwMCwgMHgwMSwKLSAgMHgwMSwgMHgwMSwgMHgwMCwgMHg0OCwgMHgwMCwgMHg0OCwgMHgwMCwgMHgwMCwgMHhmZiwgMHhkYiwgMHgwMCwgMHg0MywKLSAgMHgwMCwgMHgwNSwgMHgwMywgMHgwNCwgMHgwNCwgMHgwNCwgMHgwMywgMHgwNSwgMHgwNCwgMHgwNCwgMHgwNCwgMHgwNSwKLSAgMHgwNSwgMHgwNSwgMHgwNiwgMHgwNywgMHgwYywgMHgwOCwgMHgwNywgMHgwNywgMHgwNywgMHgwNywgMHgwZiwgMHgwYiwKLSAgMHgwYiwgMHgwOSwgMHgwYywgMHgxMSwgMHgwZiwgMHgxMiwgMHgxMiwgMHgxMSwgMHgwZiwgMHgxMSwgMHgxMSwgMHgxMywKLSAgMHgxNiwgMHgxYywgMHgxNywgMHgxMywgMHgxNCwgMHgxYSwgMHgxNSwgMHgxMSwgMHgxMSwgMHgxOCwgMHgyMSwgMHgxOCwKLSAgMHgxYSwgMHgxZCwgMHgxZCwgMHgxZiwgMHgxZiwgMHgxZiwgMHgxMywgMHgxNywgMHgyMiwgMHgyNCwgMHgyMiwgMHgxZSwKLSAgMHgyNCwgMHgxYywgMHgxZSwgMHgxZiwgMHgxZSwgMHhmZiwgMHhkYiwgMHgwMCwgMHg0MywgMHgwMSwgMHgwNSwgMHgwNSwKLSAgMHgwNSwgMHgwNywgMHgwNiwgMHgwNywgMHgwZSwgMHgwOCwgMHgwOCwgMHgwZSwgMHgxZSwgMHgxNCwgMHgxMSwgMHgxNCwKLSAgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwKLSAgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwKLSAgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwKLSAgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwgMHgxZSwKLSAgMHgxZSwgMHgxZSwgMHhmZiwgMHhjMCwgMHgwMCwgMHgxMSwgMHgwOCwgMHgwMSwgMHg1ZSwgMHgwMCwgMHhkNSwgMHgwMywKLSAgMHgwMSwgMHgyMiwgMHgwMCwgMHgwMiwgMHgxMSwgMHgwMSwgMHgwMywgMHgxMSwgMHgwMSwgMHhmZiwgMHhjNCwgMHgwMCwKLSAgMHgxYywgMHgwMCwgMHgwMCwgMHgwMiwgMHgwMiwgMHgwMywgMHgwMSwgMHgwMSwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwKLSAgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwNiwgMHgwNSwgMHgwNywgMHgwMSwgMHgwMywKLSAgMHgwNCwgMHgwMiwgMHgwOCwgMHhmZiwgMHhjNCwgMHgwMCwgMHg1NSwgMHgxMCwgMHgwMCwgMHgwMSwgMHgwMywgMHgwNCwKLSAgMHgwMCwgMHgwMywgMHgwMywgMHgwNSwgMHgwYSwgMHgwOSwgMHgwOCwgMHgwOSwgMHgwMiwgMHgwNiwgMHgwMywgMHgwMCwKLSAgMHgwMSwgMHgwMiwgMHgwMywgMHgwNCwgMHgwMCwgMHgwNSwgMHgwNiwgMHgxMSwgMHgwNywgMHgxMiwgMHgyMSwgMHgxMywKLSAgMHgzMSwgMHg0MSwgMHgxNCwgMHgyMiwgMHg1MSwgMHg2MSwgMHg3MSwgMHgxNSwgMHgyMywgMHgzMiwgMHgzNCwgMHgzNywKLSAgMHg3MiwgMHg3NSwgMHg4MSwgMHhiMSwgMHhiMywgMHgwOCwgMHgxNywgMHgzMywgMHg0MiwgMHg1MiwgMHg2MiwgMHg3NiwKLSAgMHg5MywgMHhiMiwgMHgxNiwgMHgyNCwgMHg0MywgMHg1MywgMHg1NiwgMHg5MSwgMHhhMSwgMHhkMiwgMHgyNSwgMHgzNiwKLSAgMHg2MywgMHg3MywgMHg4MiwgMHg5MiwgMHhhMiwgMHhjMSwgMHhkMSwgMHg2NSwgMHhmMCwgMHgyNiwgMHgyNywgMHg2NCwKLSAgMHg2NiwgMHg3NCwgMHhlMSwgMHg4MywgMHhjMiwgMHhmMSwgMHhmZiwgMHhjNCwgMHgwMCwgMHgxYiwgMHgwMSwgMHgwMSwKLSAgMHgwMCwgMHgwMywgMHgwMSwgMHgwMSwgMHgwMSwgMHgwMSwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwKLSAgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMSwgMHgwMiwgMHgwMywgMHgwNCwgMHgwNSwgMHgwNiwgMHgwNywgMHhmZiwKLSAgMHhjNCwgMHgwMCwgMHgzNSwgMHgxMSwgMHgwMCwgMHgwMiwgMHgwMSwgMHgwMywgMHgwMywgMHgwMSwgMHgwNSwgMHgwNSwKLSAgMHgwNywgMHgwNCwgMHgwMywgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMCwgMHgwMSwgMHgwMiwgMHgwMywKLSAgMHgwNCwgMHgxMSwgMHgxMiwgMHgyMSwgMHgzMSwgMHg0MSwgMHgwNSwgMHgyMiwgMHgzMiwgMHg1MSwgMHg2MSwgMHgwNiwKLSAgMHgxMywgMHg3MSwgMHg4MSwgMHhiMSwgMHgxNCwgMHgzMywgMHg0MiwgMHg5MSwgMHhhMSwgMHhjMSwgMHhkMSwgMHgxNSwKLSAgMHgyMywgMHhlMSwgMHhmMCwgMHgyNCwgMHg1MywgMHg5MiwgMHhmZiwgMHhkYSwgMHgwMCwgMHgwYywgMHgwMywgMHgwMSwKLSAgMHgwMCwgMHgwMiwgMHgxMSwgMHgwMywgMHgxMSwgMHgwMCwgMHgzZiwgMHgwMCwgMHhmYiwgMHgyZSwgMHg4YSwgMHgyYiwKLSAgMHhjYSwgMHg5NSwgMHhhZSwgMHhmZSwgMHhlYSwgMHgwMywgMHhkNSwgMHgxNSwgMHg4ZCwgMHhmYiwgMHgyOCwgMHhkZiwKLSAgMHhiMiwgMHg4MCwgMHhjZCwgMHgxNSwgMHg4ZCwgMHhmYiwgMHgyOCwgMHhkZiwgMHhiMiwgMHg4MCwgMHhjZCwgMHgxNSwKLSAgMHg4ZCwgMHhmYiwgMHgyOCwgMHhlNiwgMHhhMCwgMHgzMywgMHg0NSwgMHg2MywgMHg5YSwgMHhiMywgMHg0MCwgMHgxNCwKLSAgMHg1MSwgMHg0NSwgMHgwMCwgMHg1MSwgMHg0NSwgMHgxNCwgMHgwMSwgMHg0NSwgMHgxNCwgMHg1MCwgMHgwNSwgMHgxNCwKLSAgMHg1NiwgMHgzNywgMHg0MCwgMHg2NiwgMHg4YSwgMHgzNywgMHhkNiwgMHhiMSwgMHhiZiwgMHg2NSwgMHgwMSwgMHg5YSwKLSAgMHgyYiwgMHgxYiwgMHhmNiwgMHg1MSwgMHhiZiwgMHg2NSwgMHgwMSwgMHg5YSwgMHgyYiwgMHgxYiwgMHhmNiwgMHg1MSwKLSAgMHhiZiwgMHg2NSwgMHgwMSwgMHg5YSwgMHgyYiwgMHhjOSwgMHg1NiwgMHhiYiwgMHhlYiwgMHhkMCwgMHhhMCwgMHgwYSwKLSAgMHgyYiwgMHhjYSwgMHg5NSwgMHhhMywgMHg0NSwgMHgwMSwgMHhlYSwgMHg5MywgMHg3OCwgMHhlMSwgMHhiMSwgMHhjMSwKLSAgMHhlYywgMHhiOCwgMHg4MiwgMHg0MSwgMHhmNywgMHgxZSwgMHg0ZiwgMHg3MSwgMHhkNywgMHhmNCwgMHg2NiwgMHg5YywKLSAgMHhhOSwgMHgzNywgMHg4ZSwgMHg1ZiwgMHgyMywgMHhiOSwgMHg3NywgMHhkMCwgMHhmMiwgMHg3ZSwgMHhlYywgMHhkMCwKLSAgMHgxNSwgMHhiMywgMHg1YywgMHgzYywgMHhjMSwgMHhkNCwgMHhkMiwgMHgxNCwgMHg3MSwgMHg3YiwgMHg2OSwgMHgyNSwKLSAgMHgyMCwgMHg5ZiwgMHg3YSwgMHhhZiwgMHg1ZiwgMHg4YiwgMHhiYywgMHgxYiwgMHhmYiwgMHgyZCwgMHg2YywgMHhmZCwKLSAgMHhkNSwgMHgzMywgMHgzMSwgMHhmOSwgMHgxNiwgMHhmZSwgMHg2OCwgMHhmYiwgMHgyYiwgMHhkZCwgMHg3YiwgMHg5YSwKLSAgMHgyMywgMHhlNCwgMHg3OCwgMHg5YSwgMHhlNSwgMHhlNiwgMHgyYiwgMHg3ZSwgMHgyZSwgMHhmMCwgMHg2ZiwgMHhlYywKLSAgMHhiNSwgMHhiMywgMHhmNywgMHg1NCwgMHgxZSwgMHgxZSwgMHg2MCwgMHhjMywgMHhhZiwgMHhmMiwgMHg1YSwgMHhkOSwKLSAgMHhhZiwgMHhmNywgMHg1NCwgMHhjNywgMHgzNiwgMHg1NCwgMHg2OCwgMHg1MCwgMHhkZSwgMHg5OSwgMHgzMiwgMHg0MywKLSAgMHg1MSwgMHhhMywgMHgzMCwgMHg5MiwgMHhiNywgMHg1ZSwgMHg3NSwgMHg0MSwgMHgyOCwgMHg0MiwgMHg0MCwgMHhlYSwKLSAgMHg1NCwgMHg0ZiwgMHg0MCwgMHgyYSwgMHgyZSwgMHhjZiwgMHgxMywgMHgyOCwgMHhjZCwgMHhmOSwgMHg1YywgMHhiMywKLSAgMHgwNywgMHhiMSwgMHhjYywgMHg3ZCwgMHg2MywgMHg3ZSwgMHhlYiwgMHg0OSwgMHg2NywgMHhmOSwgMHhkNCwgMHg5NCwKLSAgMHhmYSwgMHg2MywgMHgzNCwgMHhhZiwgMHg4MCwgMHg5MywgMHhlMCwgMHhlMywgMHg4MywgMHhhZSwgMHhmNiwgMHgxMCwKLSAgMHg3YSwgMHgxYSwgMHhjZSwgMHhhNCwgMHhhOSwgMHg1MywgMHg1OSwgMHg5MiwgMHg0NiwgMHg5NCwgMHhlMywgMHg1MiwKLSAgMHhhMywgMHhjNCwgMHg1YiwgMHgxNywgMHgyZiwgMHg5OCwgMHhkZiwgMHgwYiwgMHhlYywgMHhhYSwgMHg2NSwgMHhhOSwKLSAgMHhmNiwgMHgxYiwgMHg2MCwgMHg5MiwgMHhmZiwgMHgwMCwgMHg0OCwgMHhmMSwgMHgxOSwgMHg4YywgMHhhNywgMHhhNCwKLSAgMHgzZSwgMHg3ZCwgMHgwOCwgMHg2OSwgMHgwMCwgMHhhZCwgMHg1ZiwgMHg1MCwgMHhkNywgMHhhNywgMHg1NSwgMHg5OCwKLSAgMHgxYywgMHgyOCwgMHg5MywgMHg5MSwgMHg4NCwgMHgyYSwgMHgwZSwgMHgwOSwgMHg2MiwgMHhjNSwgMHgyMCwgMHhhOCwKLSAgMHhlZiwgMHhjYSwgMHg2ZSwgMHhjYywgMHg4NywgMHhlNSwgMHg5MSwgMHhhZSwgMHhmNCwgMHhjNywgMHg0MiwgMHhiOSwKLSAgMHg1MCwgMHg3ZSwgMHg3YSwgMHhmNiwgMHgzZiwgMHg0NiwgMHhhZSwgMHg5YywgMHgzMywgMHgwYSwgMHhjNywgMHg3MSwKLSAgMHg0NiwgMHhkNiwgMHhiYiwgMHg0YywgMHgxZiwgMHhlNywgMHg2ZiwgMHgwZiwgMHhlNywgMHgzMywgMHhhNCwgMHgyYywKLSAgMHhiYiwgMHgyYSwgMHg0MSwgMHhmNCwgMHhhZCwgMHhkNSwgMHg3OSwgMHhjNywgMHhiYiwgMHhiYiwgMHhhMiwgMHg0NywKLSAgMHg4MCwgMHgxNCwgMHhjNSwgMHhhYSwgMHhmMywgMHhlYSwgMHg1YywgMHg2YSwgMHhmMCwgMHhhNCwgMHg4ZSwgMHhmYSwKLSAgMHg3NCwgMHgzNCwgMHhlZSwgMHhkYiwgMHg2NSwgMHg0NywgMHg4ZSwgMHg3ZSwgMHgwZiwgMHgxYywgMHgzNywgMHhiNiwKLSAgMHhiZSwgMHgyNiwgMHg1YywgMHg2ZCwgMHhlZiwgMHhkZiwgMHgyNiwgMHhlYiwgMHhmMiwgMHg5MywgMHgxYywgMHhlNSwKLSAgMHg2ZCwgMHgzZSwgMHg5MCwgMHg5NiwgMHg1YiwgMHhlNSwgMHg2YywgMHgwZiwgMHg2OCwgMHgyNywgMHhkNywgMHg0YywKLSAgMHhiZiwgMHg4YSwgMHgyZSwgMHgxOSwgMHhmZiwgMHgwMCwgMHg2MiwgMHhhYywgMHhmZiwgMHgwMCwgMHhiOCwgMHhhNywKLSAgMHg4YSwgMHgyYiwgMHg5YywgMHhlOCwgMHgxMiwgMHgzZiwgMHgxNCwgMHg3YywgMHgzMywgMHhmZSwgMHhjNSwgMHg1OSwKLSAgMHhmZiwgMHgwMCwgMHg3MSwgMHg0NywgMHhlMiwgMHg4ZiwgMHg4NiwgMHg3ZiwgMHhkOCwgMHhhYiwgMHgzZiwgMHhlZSwKLSAgMHgyOSwgMHhkZSwgMHg4YSwgMHgwMSwgMHgwOSwgMHhjZSwgMHgxNiwgMHg3MCwgMHhhZCwgMHhiNywgMHhkMCwgMHhjMywKLSAgMHg5OCwgMHg4ZCwgMHg4OSwgMHgwZSwgMHhiOSwgMHhmMCwgMHgxMCwgMHhhNiwgMHhjMCwgMHg1MiwgMHhiZCwgMHg4MywKLSAgMHg3ZCwgMHg2YiwgMHg3NywgMHhlMiwgMHg4ZiwgMHg4NiwgMHg3ZiwgMHhkOCwgMHhhYiwgMHgzZiwgMHhlZSwgMHgyYiwKLSAgMHhiNiwgMHhlZiwgMHhlZiwgMHg5YywgMHg0MCwgMHhiNSwgMHgyMywgMHhhZiwgMHg5OCwgMHhjYSwgMHhkNSwgMHhmNiwKLSAgMHhkMywgMHg1NSwgMHg3MiwgMHhkYiwgMHhkNywgMHg3NSwgMHg2NSwgMHg1MSwgMHg2MywgMHhjMiwgMHhmMSwgMHhmYSwKLSAgMHgyNywgMHhmYiwgMHg5MCwgMHg5ZSwgMHg0NCwgMHg4ZiwgMHhjNSwgMHgxZiwgMHgwYywgMHhmZiwgMHgwMCwgMHhiMSwKLSAgMHg1NiwgMHg3ZiwgMHhkYywgMHg1MSwgMHhmOCwgMHhhMywgMHhlMSwgMHg5ZiwgMHhmNiwgMHgyYSwgMHhjZiwgMHhmYiwKLSAgMHg4YSwgMHg3NywgMHhhMiwgMHhiYSwgMHg4OSwgMHgxMiwgMHgzZiwgMHgxNCwgMHg3YywgMHgzMywgMHhmZSwgMHhjNSwKLSAgMHg1OSwgMHhmZiwgMHgwMCwgMHg3MSwgMHg1OCwgMHg1NywgMHgwOCwgMHhiOCwgMHg2NiwgMHhhNCwgMHg5NCwgMHg5YywKLSAgMHgyYSwgMHhkMSwgMHhhMiwgMHgzNCwgMHg3NCwgMHhjZSwgMHhhOSwgMHhlMiwgMHg4YSwgMHgwMiwgMHg5MSwgMHhiZSwKLSAgMHg3ZSwgMHgwZSwgMHgzOCwgMHg3MCwgMHg3NSwgMHhkOSwgMHg1OCwgMHhhOSwgMHgxNiwgMHhhNywgMHhkNCwgMHg3NiwKLSAgMHgyMywgMHg0YywgMHg2YiwgMHhjYiwgMHhhMSwgMHhhOCwgMHhmYSwgMHgwYSwgMHgxYywgMHgzYywgMHhlOSwgMHhkZiwKLSAgMHhlYSwgMHhhZCwgMHgzZSwgMHhjYSwgMHg1OCwgMHhiOCwgMHg2MSwgMHgzNiwgMHhjYywgMHg2OCwgMHg3ZiwgMHhmMSwKLSAgMHg4ZiwgMHgwYywgMHhlMCwgMHgzNywgMHgxOSwgMHgzZiwgMHgwYSwgMHhlYiwgMHg2OCwgMHg2OSwgMHg1MiwgMHhhMiwKLSAgMHgwMSwgMHhhZSwgMHhmNSwgMHhhMywgMHg1ZCwgMHhhYiwgMHg0MywgMHhkNiwgMHg1MiwgMHhhNCwgMHg4ZiwgMHhkMiwKLSAgMHhmMSwgMHhhZiwgMHhhNSwgMHg0OCwgMHgxYiwgMHhkZCwgMHg2MywgMHg0MiwgMHhiNCwgMHg4NSwgMHg0NywgMHgwZSwKLSAgMHgwYywgMHhlNywgMHg0ZCwgMHg0ZiwgMHg5MywgMHhlNywgMHhmYiwgMHg3NiwgMHgxMSwgMHhjMywgMHhhYiwgMHg5NCwKLSAgMHgyNiwgMHhlNiwgMHhkYiwgMHhhYywgMHgzNiwgMHg1OSwgMHg5MSwgMHg5ZCwgMHgxYiwgMHg2ZCwgMHhlNiwgMHgxMiwKLSAgMHgxNiwgMHg4NSwgMHg4ZiwgMHg1MSwgMHg0OSwgMHgyMywgMHhmZiwgMHgwMCwgMHg3YSwgMHhhZSwgMHg4MSwgMHhjMywKLSAgMHhiYywgMHgxYiwgMHg1ZiwgMHhlYSwgMHhiNSwgMHhiMywgMHhmNywgMHg1NCwgMHhmMSwgMHg5NSwgMHg3MCwgMHhjZSwKLSAgMHhkZiwgMHgyYSwgMHg3MywgMHhmNywgMHg4YywgMHg1NiwgMHg1MSwgMHhjNiwgMHhlZiwgMHg0ZSwgMHg5ZSwgMHg2NywKLSAgMHgxYywgMHg2MSwgMHhiMCwgMHhhOCwgMHhiMiwgMHg5NSwgMHhmZiwgMHgwMCwgMHhkNCwgMHgzMSwgMHhkMSwgMHgyYiwKLSAgMHhkZiwgMHhlOSwgMHgwZSwgMHg1NSwgMHg4ZiwgMHhkMiwgMHhmMCwgMHhhNSwgMHg1OCwgMHhiNywgMHg4OSwgMHhkMCwKLSAgMHhlZiwgMHgyOCwgMHhjNywgMHhiMiwgMHhhYiwgMHg3YSwgMHgyZCwgMHgxNywgMHg4NSwgMHg4MiwgMHg2MywgMHg5NCwKLSAgMHhiOCwgMHg1NywgMHgxNiwgMHg3OCwgMHgxZCwgMHhlYSwgMHg4ZSwgMHhlMSwgMHgxZCwgMHg3NCwgMHgzYSwgMHg5NCwKLSAgMHgxMSwgMHhjYywgMHg5MSwgMHhlOSwgMHgxZSwgMHg3NSwgMHg3NywgMHhkMiwgMHhhZCwgMHg0ZSwgMHhhNiwgMHhjZCwKLSAgMHgyNCwgMHhjZSwgMHgwYSwgMHhiNCwgMHg2YSwgMHg1MywgMHhkZCwgMHgzNiwgMHhkMSwgMHhjMywgMHhmOCwgMHhiYiwKLSAgMHhjMSwgMHhiZiwgMHhiMiwgMHhkNiwgMHhjZiwgMHhkZCwgMHg1MSwgMHhmOCwgMHhiYiwgMHhjMSwgMHhiZiwgMHhiMiwKLSAgMHhkNiwgMHhjZiwgMHhkZCwgMHg1MywgMHg0OCwgMHhlZSwgMHhhMiwgMHhiYSwgMHg3NCwgMHg0NywgMHhjOCwgMHhlNywKLSAgMHhkNywgMHgyZiwgMHgzMSwgMHgwZSwgMHhmMSwgMHg4OCwgMHhlMywgMHgzNiwgMHg0YiwgMHhjNiwgMHgyZCwgMHgzYSwKLSAgMHhkMSwgMHg2NSwgMHg4NywgMHgwNiwgMHg1MCwgMHhjOSwgMHgyZCwgMHhlOCwgMHhlZCwgMHg1OSwgMHg0NywgMHgyYSwKLSAgMHhiOSwgMHg0YiwgMHhhMCwgMHgxMSwgMHhiZiwgMHg0MSwgMHgxNSwgMHhmNCwgMHg2MCwgMHhlZSwgMHhhYSwgMHgzZiwKLSAgMHgzMywgMHhmOCwgMHhkNiwgMHgyZCwgMHhmYiwgMHg0ZiwgMHg2ZSwgMHhmYiwgMHhlMSwgMHg1NywgMHg4MCwgMHhlZSwKLSAgMHhhZiwgMHgzNiwgMHhlZCwgMHgyNSwgMHg1MywgMHg2MywgMHhkMiwgMHhiNSwgMHg2ZCwgMHhjMywgMHg3MywgMHgwNCwKLSAgMHg2YywgMHhmNywgMHhkMSwgMHg1OSwgMHhhMiwgMHhiOSwgMHg0ZSwgMHg5MCwgMHhhNCwgMHhkZSwgMHgzOSwgMHg3YywKLSAgMHg4ZSwgMHhlNSwgMHhkZiwgMHg0MywgMHhjOSwgMHhmYiwgMHhiMywgMHg0ZSwgMHg1NCwgMHg5YiwgMHhjNywgMHgyZiwKLSAgMHg5MSwgMHhkYywgMHhiYiwgMHhlOCwgMHg3OSwgMHgzZiwgMHg3NiwgMHg2OCwgMHgwNSwgMHhlNiwgMHgzZiwgMHgyMiwKLSAgMHhkZiwgMHhjZCwgMHgxZiwgMHg2NSwgMHg2OCwgMHhiYiwgMHhkYywgMHg2MSwgMHg1YSwgMHgyZCwgMHg5MiwgMHgyZSwKLSAgMHg1NywgMHgyOSwgMHgwOCwgMHg4ZiwgMHgxMiwgMHgzYSwgMHgzOSwgMHhkZCwgMHg3MSwgMHg1YiwgMHhkMCwgMHgxZSwKLSAgMHhhZiwgMHg0OSwgMHgyNywgMHg0MCwgMHgwMSwgMHhkNCwgMHg5MiwgMHgwNSwgMHg3NCwgMHgzMSwgMHhmOSwgMHgyNiwKLSAgMHhjNywgMHhhNSwgMHgyMCwgMHgwZSwgMHg5ZSwgMHhhYSwgMHhlMCwgMHhlMSwgMHhlZCwgMHhhMSwgMHgxOSwgMHhiZSwKLSAgMHg0MiwgMHg4YywgMHhiYSwgMHg3YSwgMHg3OSwgMHhmMSwgMHhmYiwgMHg1YywgMHg4NSwgMHgyNiwgMHhjNiwgMHhjMiwKLSAgMHhiZSwgMHgwYywgMHg5OSwgMHgwOCwgMHgzYywgMHhhYiwgMHg5OCwgMHhhMSwgMHhlMiwgMHgxMiwgMHhhMCwgMHg1MiwKLSAgMHhkOCwgMHgzZCwgMHgzYSwgMHgxNSwgMHhmNywgMHg5NCwgMHg5MSwgMHhlYywgMHhkNiwgMHhhYSwgMHhhOSwgMHg0NywKLSAgMHgyNywgMHg4ZiwgMHg0NiwgMHg5MywgMHhhOSwgMHgyYywgMHgxZCwgMHgxOCwgMHg4ZSwgMHgxNywgMHgzMywgMHgyOSwKLSAgMHg5NywgMHgxZiwgMHgyMiwgMHhjZCwgMHhhMSwgMHhhZSwgMHgzYywgMHgyNiwgMHg1YywgMHg0YiwgMHhkNiwgMHhjYiwKLSAgMHgwYiwgMHhhMywgMHhhMywgMHg2NCwgMHgxZCwgMHhhNSwgMHhmOSwgMHg0MywgMHhiOSwgMHg0ZSwgMHhmOCwgMHg4NiwKLSAgMHhmYSwgMHhhNSwgMHhiZiwgMHg1YSwgMHhiNiwgMHg0NSwgMHhiMCwgMHgwNywgMHg0YSwgMHgwMCwgMHgxYSwgMHhlOSwKLSAgMHhmZSwgMHgxNSwgMHg5YSwgMHhmMiwgMHgyNSwgMHgyNywgMHgzNywgMHg5NiwgMHg3YSwgMHhkMSwgMHg4MiwgMHg4YSwKLSAgMHhjMiwgMHgwYSwgMHgyYiwgMHg5MiwgMHhlZSwgMHhmYywgMHhhOCwgMHhkMCwgMHgxYywgMHg3YSwgMHgxYywgMHg2ZiwKLSAgMHgyOSwgMHg3OSwgMHgyMywgMHhjZCwgMHg2ZiwgMHg3YSwgMHhkZCwgMHg3MSwgMHhlMywgMHhmNywgMHhmOCwgMHg3NywKLSAgMHg2NCwgMHg3MiwgMHgwMiwgMHg1OSwgMHg5MiwgMHg5ZSwgMHg4ZSwgMHgzMiwgMHhiZSwgMHg4YSwgMHgwNywgMHhmZSwKLSAgMHhmNSwgMHhjZCwgMHgyYiwgMHg5YSwgMHg3MSwgMHhhYSwgMHhhOSwgMHg0OSwgMHhlMSwgMHhiZSwgMHgzZCwgMHg0OSwKLSAgMHhjZiwgMHg0MiwgMHg1ZSwgMHg4YSwgMHhkNywgMHgyNSwgMHhmNiwgMHg2MywgMHgzMCwgMHhhNywgMHg5ZiwgMHg3MSwKLSAgMHgyZCwgMHhiNiwgMHg5MSwgMHhiMiwgMHhhNSwgMHgxZCwgMHgwYSwgMHhhZSwgMHhiMiwgMHg4YywgMHhlOSwgMHhmNywKLSAgMHhkNCwgMHhiOCwgMHhkNiwgMHg4ZiwgMHg3YSwgMHg2YiwgMHhiOCwgMHhiYywgMHg3ZSwgMHgxMiwgMHhiZCwgMHg5ZSwKLSAgMHg4YSwgMHhlNiwgMHhiZiwgMHhlZCwgMHgzYSwgMHgxNiwgMHgzMCwgMHhkNSwgMHg1NSwgMHhlZiwgMHhkMSwgMHg3NSwKLSAgMHg2NCwgMHg0YSwgMHg0OSwgMHg3MiwgMHgzZSwgMHg1YywgMHhhZSwgMHhkNiwgMHhkYiwgMHg3MiwgMHg3OSwgMHhhNiwKLSAgMHhjYywgMHg2OSwgMHg5ZiwgMHg1MSwgMHgzZCwgMHg3ZiwgMHhiOCwgMHg3NSwgMHhhNSwgMHhjOSwgMHg5YywgMHg0MCwKLSAgMHhiNCwgMHhiNCwgMHg0OCwgMHg2MSwgMHg5NywgMHhkZiwgMHhmNSwgMHhlYiwgMHg0MiwgMHhhYiwgMHgxNywgMHg5ZCwKLSAgMHg3NSwgMHhlNywgMHgwYiwgMHg4ZiwgMHgzOCwgMHhhNywgMHgxNiwgMHg3YiwgMHhkNCwgMHhhMywgMHhiMywgMHg1ZSwKLSAgMHgyYiwgMHhlMywgMHhhZSwgMHg3ZCwgMHhhYiwgMHhiOSwgMHg5YiwgMHhmZSwgMHhkNCwgMHg1NCwgMHg1NywgMHhlNiwKLSAgMHhmZiwgMHgwMCwgMHhkZiwgMHg5MSwgMHg5MywgMHhhYSwgMHhmYSwgMHgwZCwgMHhlZiwgMHg2NiwgMHgwYywgMHhiOSwKLSAgMHg5MywgMHhiNSwgMHg3NywgMHgzMCwgMHg5NywgMHhjOCwgMHhkYiwgMHgyNSwgMHhiMCwgMHg4ZSwgMHg3MSwgMHhiZSwKLSAgMHhiZSwgMHgzNCwgMHhjNywgMHgwZiwgMHg4OCwgMHgxNiwgMHg4NywgMHg1NCwgMHgwMywgMHhlZCwgMHgzZSwgMHhjNiwKLSAgMHhmYywgMHg0OCwgMHhkOCwgMHhmZiwgMHgwMCwgMHgwYSwgMHhhYiwgMHg0MCwgMHgyNCwgMHg4MCwgMHgwMSwgMHgyNCwKLSAgMHhmNywgMHgwMSwgMHg1ZCwgMHg3MywgMHg2ZCwgMHg5NywgMHgwOCwgMHg0ZCwgMHgyMSwgMHhkOSwgMHg1MSwgMHgxZCwKLSAgMHg2NSwgMHgwYiwgMHhmOCwgMHgyYSwgMHg1MiwgMHg3YSwgMHgxYSwgMHhlMywgMHhiNiwgMHhlZCwgMHhjYiwgMHhmYSwKLSAgMHg1YSwgMHhhNywgMHgxZCwgMHhkMywgMHg3OSwgMHg3YiwgMHgxMCwgMHhhNywgMHgyMiwgMHhlYSwgMHhiNywgMHg1ZCwKLSAgMHgyMCwgMHg1YywgMHg1MSwgMHhjZiwgMHgwZSwgMHg1MywgMHg2ZSwgMHg4ZiwgMHg0MCwgMHgzZCwgMHg0NywgMHhkNSwKLSAgMHg1ZCwgMHg5NSwgMHg0MSwgMHg0NiwgMHg5MCwgMHhmYywgMHg2NywgMHg0MywgMHhiMSwgMHhkZCwgMHg1YiwgMHg0YiwKLSAgMHgxZCwgMHhjYSwgMHg0OSwgMHhkMSwgMHhhYiwgMHgwNywgMHgxMCwgMHhjZCwgMHhiYiwgMHg2NSwgMHhhMiwgMHgxNSwKLSAgMHhkZCwgMHg0MSwgMHgyYiwgMHgzZCwgMHgxMCwgMHhmNywgMHg3MCwgMHgzZSwgMHhkYSwgMHhmYSwgMHg1ZSwgMHhjZCwKLSAgMHhmNiwgMHg5YSwgMHg5NSwgMHhjNCwgMHg5NSwgMHgzYSwgMHhlYiwgMHg0YiwgMHg3ZCwgMHg3YSwgMHg3ZiwgMHg4MywKLSAgMHg0OCwgMHhkNCwgMHg0ZiwgMHg5MSwgMHhmMiwgMHg4YSwgMHgwMSwgMHgwNCwgMHg2YywgMHgxZCwgMHg4MywgMHg0NSwKLSAgMHg3ZCwgMHg0MSwgMHhhMCwgMHg1MSwgMHg0NSwgMHgxNCwgMHgwNiwgMHgwYSwgMHg0MSwgMHgzYiwgMHhhOCwgMHg3YywKLSAgMHhiZiwgMHgxYiwgMHhiMywgMHhlNSwgMHg1NiwgMHg0NywgMHhhYywgMHhmNywgMHhhOCwgMHhiZSwgMHg1MSwgMHgxOSwKLSAgMHhjNSwgMHgwNSwgMHhhNCwgMHhhNCwgMHhmMiwgMHhhZCwgMHhhNywgMHgxMywgMHhkNSwgMHgyZSwgMHgyMSwgMHg0MywKLSAgMHhhYSwgMHgxNiwgMHg5MywgMHhkNCwgMHgyOCwgMHg3NSwgMHgxNSwgMHgzMywgMHg0NiwgMHhhOCwgMHgwYSwgMHg0NSwKLSAgMHgyZSwgMHg1ZCwgMHg3MSwgMHg1YiwgMHhmMywgMHg1OCwgMHhhZSwgMHg1MSwgMHgyNCwgMHg0YSwgMHgzMiwgMHgzNiwKLSAgMHgyYywgMHhmNywgMHg1MiwgMHg5ZSwgMHg0MSwgMHgzOCwgMHgyNCwgMHg2YywgMHhiNCwgMHhlMCwgMHhlZSwgMHg0YiwKLSAgMHhlOSwgMHgwMywgMHg2NywgMHg1ZCwgMHgxNiwgMHgzYywgMHhlMSwgMHhhZSwgMHhhMiwgMHhhNywgMHhhOSwgMHhkYiwKLSAgMHgzMywgMHhjNiwgMHhlZCwgMHg5OSwgMHg1NiwgMHgzZiwgMHgyYSwgMHhjYiwgMHg3NCwgMHg2ZCwgMHg2NSwgMHg4NywKLSAgMHg4MCwgMHgyOCwgMHg3MSwgMHhiNSwgMHg3MiwgMHhiYSwgMHhjYiwgMHg4OSwgMHhlYSwgMHg4NywgMHgxYiwgMHg1MCwKLSAgMHhlYSwgMHg5NSwgMHhhNCwgMHg4MCwgMHhhMCwgMHg3ZCwgMHgzNSwgMHg1NywgMHhlMywgMHgxMywgMHgyZSwgMHgzZCwKLSAgMHhhNCwgMHhlYiwgMHgwNSwgMHhmYiwgMHg5MywgMHhkZCwgMHhjYiwgMHgzYiwgMHg4MSwgMHg4OSwgMHg2YSwgMHg0MiwKLSAgMHg0MiwgMHg1MywgMHgyMSwgMHgyNCwgMHg3MywgMHgzNSwgMHgyMSwgMHgwMCwgMHg3NywgMHgyNSwgMHhjNCwgMHhlYywKLSAgMHhlYiwgMHhjMSwgMHg0MSwgMHg0MywgMHhjMywgMHg1NSwgMHhlOSwgMHg1YSwgMHhkNywgMHhkNSwgMHhkZCwgMHg5NywKLSAgMHgyNywgMHg5YiwgMHg3MywgMHg0MywgMHg0ZiwgMHg3YSwgMHgzYywgMHgxYSwgMHg3MywgMHgzZiwgMHg4ZCwgMHg2MiwKLSAgMHhkZiwgMHhiNCwgMHhmNiwgMHhlZiwgMHhiZSwgMHgxNSwgMHg3OCwgMHgwZSwgMHhlYSwgMHhhMywgMHhiMywgMHgzZiwKLSAgMHg4ZCwgMHg2MiwgMHhkZiwgMHhiNCwgMHhmNiwgMHhkZiwgMHhiZSwgMHgxNSwgMHg3OCwgMHg4ZSwgMHhlYSwgMHhlNywKLSAgMHhiYywgMHhmYiwgMHhjZiwgMHg5MSwgMHhkMSwgMHg2OSwgMHhmNywgMHg2MSwgMHg0NSwgMHgxNCwgMHg1NywgMHgyOSwKLSAgMHhkNCwgMHgxNCwgMHg5YiwgMHhjNywgMHgyZiwgMHg5MSwgMHhkYywgMHhiYiwgMHhlOCwgMHg3OSwgMHgzZiwgMHg3NiwKLSAgMHg2OSwgMHhjYSwgMHg5MywgMHgzOCwgMHhlNCwgMHg3ZiwgMHhmOSwgMHgzZCwgMHg5NywgMHg3NCwgMHhkZiwgMHhmYSwKLSAgMHgyMiwgMHg0ZiwgMHhkZCwgMHg5YSwgMHgwMSwgMHgwMywgMHgyOCwgMHgxMiwgMHhhZSwgMHg0OSwgMHhiNSwgMHg2MiwKLSAgMHg1NiwgMHhkNywgMHg5NiwgMHhjYywgMHhkYiwgMHhmYiwgMHhiZSwgMHg0YywgMHhhNywgMHg1MCwgMHg3NCwgMHhhNiwKLSAgMHgyMiwgMHhhNSwgMHgxYywgMHhkMiwgMHgxZCwgMHgxZSwgMHg4MiwgMHgxMCwgMHgzOSwgMHg0NywgMHhlYiwgMHgyZCwKLSAgMHgzNSwgMHg3NCwgMHg1YSwgMHhhMCwgMHg0NCwgMHhiNiwgMHg1YiwgMHgyMywgMHg1YiwgMHg2MCwgMHgzMCwgMHg4OCwKLSAgMHhmMSwgMHgyMiwgMHhiNCwgMHg5NiwgMHg1OCwgMHg2OSwgMHgwMywgMHhjZCwgMHg0MiwgMHgxMiwgMHgzNCwgMHg5MCwKLSAgMHgzZCwgMHg4MCwgMHgwYSwgMHhhYywgMHg3OCwgMHg1MSwgMHgxMCwgMHhkYywgMHhmOCwgMHg4NywgMHg5MCwgMHhkZiwKLSAgMHgxZCwgMHg0ZiwgMHgzMywgMHg1NiwgMHg5OCwgMHhhYywgMHg1OSwgMHhlMiwgMHgxZCwgMHhmNCwgMHgwYiwgMHg1MiwKLSAgMHg0MywgMHhmMiwgMHgwZSwgMHhiZCwgMHgyNywgMHg5OSwgMHg4MSwgMHhiZiwgMHhkNCwgMHhhYiwgMHg2MCwgMHg1NiwKLSAgMHhmNywgMHgzNSwgMHgzNSwgMHhjZCwgMHhmYSwgMHgxOCwgMHg1YiwgMHg0MywgMHg0NCwgMHgxNywgMHhhOCwgMHgwMSwKLSAgMHhhMSwgMHhhYSwgMHgyOCwgMHhhMiwgMHhiMCwgMHgzNywgMHgwYSwgMHg4MSwgMHhjOCwgMHhmMSwgMHhmOCwgMHhiMywKLSAgMHg0MSwgMHg5YSwgMHhjMywgMHg5ZSwgMHg0NywgMHgzMSwgMHhhMSwgMHhjYywgMHg5NywgMHhkMSwgMHhkMywgMHhiYiwKLSAgMHhkMywgMHg1MywgMHhkNCwgMHg5ZCwgMHhjNCwgMHhmYiwgMHhiOSwgMHg4NywgMHg2ZCwgMHg0ZCwgMHhiZCwgMHg5NSwKLSAgMHg2OSwgMHhkOSwgMHgzZiwgMHgwYiwgMHhkNCwgMHg4YSwgMHhmMywgMHhmYiwgMHg1MiwgMHg3NCwgMHg2OSwgMHhkYSwKLSAgMHhjYSwgMHg3NSwgMHg5NiwgMHg1MiwgMHhmYSwgMHhmNCwgMHhjMSwgMHg1OSwgMHg2MywgMHgxYiwgMHg4OSwgMHg1OSwKLSAgMHgwNiwgMHg0NSwgMHg3MywgMHhiOSwgMHhiNCwgMHg4OCwgMHg3MiwgMHg1ZiwgMHg0YSwgMHg5YiwgMHg2YiwgMHhjZCwKLSAgMHgyNSwgMHgxZCwgMHgwMywgMHg4NCwgMHg3OCwgMHg5YSwgMHg4NCwgMHhhZCwgMHg5MSwgMHg2MywgMHhiZCwgMHgyOSwKLSAgMHhmNCwgMHhiMywgMHgxZCwgMHhhNSwgMHgzOCwgMHhlMiwgMHhiYiwgMHg5MiwgMHg5MSwgMHhkNCwgMHhkNywgMHhiOSwKLSAgMHhkMCwgMHhhNSwgMHg0MSwgMHg3YiwgMHhiMSwgMHg5NiwgMHhjMiwgMHhkOSwgMHg1ZSwgMHhiNywgMHhhNSwgMHgwZCwKLSAgMHg1NywgMHhlNSwgMHhkNSwgMHhhYSwgMHg1NiwgMHhiOCwgMHg2ZSwgMHhhZCwgMHg0YywgMHhiZSwgMHg5OSwgMHhmZCwKLSAgMHhiMiwgMHg3MywgMHhiYywgMHhiZCwgMHhjZCwgMHgxNCwgMHg1MSwgMHg0NSwgMHg3MywgMHgxMCwgMHg3NCwgMHg1YiwKLSAgMHgyNCwgMHgwOCwgMHg5NywgMHgwNiwgMHgyNCwgMHhhOSwgMHgxYywgMHhlMSwgMHhhNSwgMHg4NSwgMHgxNCwgMHhmYSwKLSAgMHg3NSwgMHg0ZSwgMHgxOSwgMHg5ZSwgMHg1OSwgMHg2ZiwgMHhiYSwgMHhkOSwgMHg4NCwgMHgzOCwgMHg4ZCwgMHgyYywKLSAgMHhhZCwgMHg2YSwgMHgwNSwgMHg0NSwgMHg2OSwgMHhkNywgMHgyZSwgMHhhOSwgMHgxZSwgMHg4YSwgMHhlZSwgMHhiNywKLSAgMHhlZCwgMHgwYSwgMHhkNiwgMHhmNCwgMHg2NywgMHg0NiwgMHgxYywgMHg0YiwgMHg5MiwgMHg1NCwgMHg5YSwgMHg1OCwKLSAgMHgwYSwgMHgyOCwgMHg0OCwgMHgyYSwgMHgzYSwgMHgwMCwgMHg5MiwgMHg3YywgMHgwNSwgMHgzMCwgMHg1OSwgMHhiMSwKLSAgMHgzYiwgMHg5YywgMHhmNCwgMHg4NywgMHg1ZCwgMHg0OCwgMHg4YiwgMHgxZiwgMHhiYywgMHhiOCwgMHhlZiwgMHg0ZSwKLSAgMHg5ZSwgMHhjYSwgMHhjMiwgMHg4NSwgMHhiNSwgMHg1YiwgMHg4OSwgMHg2OSwgMHhhNSwgMHgxNiwgMHhkOCwgMHg0OSwKLSAgMHhiZSwgMHgwNiwgMHhhZSwgMHgxYSwgMHhlNCwgMHgwYSwgMHg5NiwgMHhkMSwgMHhiNSwgMHg0YiwgMHg1ZSwgMHhkZCwKLSAgMHg2YywgMHg2ZCwgMHhhNSwgMHgxZSwgMHhmNSwgMHgyNywgMHhkMSwgMHhlZCwgMHhhNywgMHg0NywgMHhkZCwgMHg2ZCwKLSAgMHg4NiwgMHg1NiwgMHhmMywgMHhjYiwgMHg0YSwgMHgxYiwgMHg0MCwgMHhkYSwgMHg5NCwgMHg0ZSwgMHg4MCwgMHgxNSwKLSAgMHg1ZCwgMHhjNywgMHg5MywgMHg4YywgMHhlMywgMHgwZSwgMHg4NSwgMHhiMCwgMHhhNSwgMHhjZiwgMHg5YSwgMHg5ZSwKLSAgMHg5YywgMHhlMCwgMHhmNCwgMHg0ZCwgMHg0YSwgMHg3MSwgMHgxNiwgMHhlMCwgMHg1ZiwgMHhjNCwgMHhhMywgMHhiZiwKLSAgMHgxNSwgMHg1ZSwgMHhmMywgMHgyNSwgMHg2OSwgMHhlNiwgMHgyMywgMHhkMSwgMHhhMiwgMHg3NSwgMHhmZiwgMHgwMCwKLSAgMHhiZiwgMHg0NSwgMHg3ZCwgMHhmNSwgMHg4ZiwgMHg2OCwgMHgzYiwgMHg2YiwgMHgyOSwgMHg0NiwgMHhhYywgMHg5NCwKLSAgMHhhNywgMHg0ZCwgMHg3MCwgMHg5ZSwgMHg3NiwgMHhlOSwgMHg5NiwgMHg2ZCwgMHgxOSwgMHg2MSwgMHg2ZSwgMHg3OCwKLSAgMHg5ZiwgMHhjNCwgMHgzOCwgMHg0ZCwgMHgzZSwgMHg1YiwgMHg4YiwgMHgwZCwgMHhjNywgMHhkMCwgMHgwZiwgMHhjMywKLSAgMHgyYSwgMHhlNSwgMHgwNywgMHhkOCwgMHgyYSwgMHg2ZiwgMHgxYSwgMHhjOSwgMHg2MCwgMHg1ZiwgMHgwMSwgMHg0MywKLSAgMHgwNCwgMHhiNiwgMHhmYSwgMHg0NiwgMHhkNCwgMHhkMiwgMHhmYiwgMHhmNSwgMHhlOSwgMHgxZSwgMHg5YSwgMHhhNiwKLSAgMHg2YiwgMHhiMiwgMHhjYiwgMHgzZCwgMHhjYiwgMHg2NSwgMHhkMSwgMHg4OSwgMHhhZCwgMHgxMywgMHhiNiwgMHhkNSwKLSAgMHhiMiwgMHgwNywgMHg4OCwgMHhmMSwgMHgxZiwgMHhkZCwgMHg1ZSwgMHgwZCwgMHhhZiwgMHhiNCwgMHhmNywgMHg0YSwKLSAgMHhiYSwgMHg3NSwgMHg5YSwgMHg3MSwgMHg3YywgMHhhYywgMHg3MSwgMHhmMCwgMHgyOCwgMHhhYSwgMHgzYywgMHhlZSwKLSAgMHg1ZSwgMHhiNCwgMHg1NiwgMHhhOCwgMHg4ZiwgMHhiNywgMHgyYSwgMHgyYiwgMHg1MiwgMHgxYSwgMHg1MCwgMHg1MywKLSAgMHg2ZSwgMHgyNCwgMHgyOSwgMHgyNCwgMHg3OCwgMHg4MywgMHg1YiwgMHg2YiwgMHhmNCwgMHgyNCwgMHhkNCwgMHg5NiwKLSAgMHg1MSwgMHhiOCwgMHg2YiwgMHhhZSwgMHhlYSwgMHhhZSwgMHhlMywgMHg1YywgMHgxZiwgMHg3MiwgMHgyZSwgMHgxNiwKLSAgMHg5YywgMHhmYSwgMHgzOCwgMHhlNSwgMHg0YywgMHgyNSwgMHhhNiwgMHgwNSwgMHhkZiwgMHg0NywgMHg0MSwgMHg3MCwKLSAgMHg5ZSwgMHg1OCwgMHgwMSwgMHg0NywgMHhmZCwgMHhkMywgMHhhNSwgMHgwYiwgMHgxZSwgMHg4MCwgMHg1NywgMHhlOSwKLSAgMHhhYiwgMHg0NiwgMHhhMywgMHhiMiwgMHg1YiwgMHg0YywgMHg1YiwgMHhmNSwgMHg4MiwgMHhlMSwgMHg2NCwgMHg5YSwKLSAgMHg5ZSwgMHg2OCwgMHhkMywgMHhlMywgMHgzOSwgMHgxOSwgMHhkZiwgMHg5YSwgMHhiNCwgMHg5NCwgMHg5ZCwgMHg3YSwKLSAgMHhmYSwgMHhkNSwgMHhlMywgMHgyZCwgMHgyZCwgMHgzNCwgMHg1NiwgMHg1MSwgMHg1MiwgMHg0ZCwgMHgzMiwgMHhhMCwKLSAgMHhjYywgMHhjMSwgMHgxMiwgMHhiMSwgMHg2MCwgMHg3YiwgMHhjNiwgMHg0ZiwgMHg2ZSwgMHhmYiwgMHhlMSwgMHg1NywKLSAgMHg4MCwgMHhlZSwgMHhhZiwgMHg5YiwgMHhhZCwgMHhkMywgMHhhNCwgMHhjZiwgMHhjNSwgMHg3MCwgMHg1NywgMHgyNywKLSAgMHgxMiwgMHg2NywgMHgzMSwgMHg5MiwgMHg1YiwgMHhlMiwgMHg0YiwgMHhkZiwgMHg3ZiwgMHg2ZSwgMHhjYywgMHg4ZSwKLSAgMHhjOSwgMHhjZCwgMHhmYSwgMHhjYSwgMHg5MCwgMHg0ZiwgMHhkNywgMHg1ZiwgMHg0OCwgMHg4ZSwgMHhlYSwgMHhkZSwKLSAgMHhlZCwgMHhlYSwgMHg5YSwgMHg3ZSwgMHg4NiwgMHgxNiwgMHhhYiwgMHg0YywgMHgxYSwgMHhmNSwgMHgwYSwgMHgyOCwKLSAgMHhhMiwgMHhiOSwgMHg4ZSwgMHg5MCwgMHhhNCwgMHhkZSwgMHgzNywgMHg4ZCwgMHhmMCwgMHg3ZiwgMHgyZSwgMHgxYiwKLSAgMHhkNywgMHhmYSwgMHgyMiwgMHg0NywgMHhkZCwgMHg5YSwgMHg3MiwgMHhhNCwgMHhlZSwgMHgzNywgMHg3NCwgMHhlMSwKLSAgMHgwNiwgMHg1YywgMHg3ZCwgMHgxNiwgMHg4OSwgMHgxZiwgMHg3NiwgMHg2OCwgMHgwOCwgMHg4ZiwgMHhjMSwgMHhjZCwKLSAgMHgyMSwgMHhlZSwgMHgxYiwgMHgwYiwgMHhjNywgMHg0ZSwgMHhkMiwgMHhlZiwgMHg3MywgMHg5YiwgMHgzNSwgMHg2NCwKLSAgMHg3YSwgMHhlNCwgMHgyZCwgMHgwOSwgMHhmZiwgMHgwMCwgMHhhMSwgMHgwOCwgMHgxNSwgMHg2NCwgMHg1NSwgMHg3YiwKLSAgMHhmOCwgMHgzNywgMHhhNSwgMHgyMywgMHg4MSwgMHhiOCwgMHg5OCwgMHgxZSwgMHgzMCwgMHg0MiwgMHg4ZiwgMHhiNCwKLSAgMHhhZCwgMHg0NCwgMHhmZiwgMHgwMCwgMHg4OSwgMHgzNSwgMHg2MSwgMHg1MSwgMHhlZSwgMHhmMiwgMHg0MiwgMHhkOCwKLSAgMHgyOCwgMHhhMiwgMHg4YSwgMHgxMiwgMHgxNSwgMHg0ZSwgMHhmMSwgMHgwMiwgMHg2MiwgMHhhNSwgMHhlNCwgMHhmMiwKLSAgMHg3YywgMHhlMiwgMHg1MiwgMHhjOSwgMHhlYywgMHhkMiwgMHgzZCwgMHgxYSwgMHhhYiwgMHg4OCwgMHhmNywgMHg1NSwKLSAgMHgxNywgMHg3ZSwgMHgyNSwgMHg1NywgMHhhOSwgMHhhNCwgMHhmNywgMHhmNiwgMHhlYSwgMHhmYiwgMHg2YiwgMHhlNCwKLSAgMHhmZCwgMHhhZCwgMHhhOCwgMHhlMywgMHg2ZiwgMHgwOCwgMHgyZSwgMHhhZiwgMHhlOCwgMHg4YywgMHhhYSwgMHhmMCwKLSAgMHg3NiwgMHhlMSwgMHg5NywgMHg3NiwgMHg2YywgMHhkNywgMHg5NCwgMHhjYSwgMHg5MCwgMHhkOSwgMHg1YiwgMHg2NSwKLSAgMHgyNSwgMHgyNywgMHg5NywgMHhiYywgMHg2ZiwgMHhjNiwgMHhiYiwgMHgzMywgMHhkYiwgMHhmNCwgMHg1YiwgMHhkYywKLSAgMHg5NiwgMHgzYywgMHg5MSwgMHhiNSwgMHgwNCwgMHgzNCwgMHgwZiwgMHg5ZSwgMHhhMSwgMHhhMiwgMHg3NywgMHg0YiwKLSAgMHgzNCwgMHg1NywgMHhjNywgMHhjNywgMHhiNCwgMHgyYiwgMHg0NiwgMHhkNSwgMHhkYSwgMHhhZiwgMHgwYiwgMHg3OSwKLSAgMHgzMiwgMHhkNCwgMHhmMSwgMHg4MCwgMHhhMiwgMHhiZCwgMHhiMiwgMHhkMywgMHhhZiwgMHgzOCwgMHgxYiwgMHg2NSwKLSAgMHhiNSwgMHgzOCwgMHhiMywgMHhkYywgMHgxMiwgMHgzNiwgMHg2OSwgMHg5YSwgMHhkOSwgMHg4NywgMHg0OCwgMHg1MywKLSAgMHg0MiwgMHg0ZCwgMHhkOSwgMHhmNCwgMHg0MSwgMHg4ZSwgMHgzYSwgMHg5ZSwgMHg2MywgMHhlNywgMHgxYSwgMHhjZSwKLSAgMHhkYSwgMHhjZSwgMHhiNSwgMHhjYiwgMHhjNSwgMHgyOCwgMHhlNywgMHhlOSwgMHhmMywgMHg2MSwgMHgyNiwgMHhjNSwKLSAgMHg3NCwgMHgyMSwgMHg2ZSwgMHgyOCwgMHgyNSwgMHgwOSwgMHgyYSwgMHg1MSwgMHhlZSwgMHgwMCwgMHg2ZSwgMHg5OSwKLSAgMHgyZCwgMHgxOCwgMHg3YywgMHhmOSwgMHgyZCwgMHg4OSwgMHgxMywgMHg1NCwgMHg5OCwgMHg1MSwgMHhmYiwgMHhjYSwKLSAgMHg5YywgMHgzYSwgMHgzYSwgMHhmNiwgMHg1NCwgMHg4MywgMHg5NywgMHhjYiwgMHgwNSwgMHg4OSwgMHgwNSwgMHhhYiwKLSAgMHgyYywgMHgzMSwgMHgyNSwgMHhmMSwgMHhkMCwgMHhiZSwgMHhlMCwgMHhlOSwgMHg0YiwgMHg3NywgMHg3YiwgMHhkZCwKLSAgMHhjYSwgMHhlOCwgMHhlMSwgMHg1NCwgMHhiOSwgMHgyYiwgMHg1MiwgMHg3YywgMHgxMCwgMHgwZSwgMHg5MiwgMHgzZSwKLSAgMHhhYSwgMHhlZCwgMHhmNywgMHgzNiwgMHg3NiwgMHhiZiwgMHg3YiwgMHgyZiwgMHg3OSwgMHgyZiwgMHgyOCwgMHhmMSwKLSAgMHhmMywgMHg3ZCwgMHg3ZSwgMHg0NCwgMHhlMSwgMHgyMSwgMHg5NCwgMHhkYywgMHg3MSwgMHg5YywgMHg3OCwgMHg3MiwKLSAgMHhkYiwgMHhkOCwgMHhmNywgMHg0MiwgMHg1OCwgMHhmZSwgMHg5NSwgMHg3ZiwgMHgwNCwgMHgxYSwgMHg1ZiwgMHhiYywKLSAgMHhlNCwgMHg1NywgMHg0YiwgMHhhMiwgMHg4OCwgMHg3ZSwgMHg0MSwgMHg0MywgMHg1ZSwgMHgwZCwgMHhhMywgMHhhMiwKLSAgMHg0NSwgMHg0NCwgMHhkMSwgMHg1OCwgMHg1NywgMHhlZCwgMHgyYSwgMHhkNSwgMHg2MywgMHhlZSwgMHhlMywgMHhkZCwKLSAgMHg4ZiwgMHg5MiwgMHhkOSwgMHg3YywgMHhmYywgMHhmZSwgMHg2NCwgMHgzOSwgMHgzNiwgMHgxNCwgMHhlZCwgMHg4NywKLSAgMHgzOCwgMHg4YiwgMHhkNiwgMHgzOSwgMHgzNywgMHgxZCwgMHg3ZCwgMHg0MywgMHhiNCwgMHg0YSwgMHg3YiwgMHg0OCwKLSAgMHhlNCwgMHhmOCwgMHgxZiwgMHhmZiwgMHgwMCwgMHhkZiwgMHhmYiwgMHhkMiwgMHg0ZCwgMHg3NywgMHg1OCwgMHgyZSwKLSAgMHgwZSwgMHg1YiwgMHgyZSwgMHhjYywgMHg0YywgMHg2YywgMHhmYywgMHgwNSwgMHg3OSwgMHhjMywgMHhkMiwgMHgzYywKLSAgMHg0NSwgMHg1NywgMHhiMywgMHhhZSwgMHg1NSwgMHhiZCwgMHg3NCwgMHhlNywgMHhlMSwgMHg3YiwgMHgzZiwgMHg4MywKLSAgMHhlNCwgMHg0NSwgMHhlMSwgMHg5YywgMHg2ZSwgMHhiNiwgMHhiNiwgMHg5ZCwgMHg1YiwgMHg0ZSwgMHgyNCwgMHhhNSwKLSAgMHg2OCwgMHg1MSwgMHg0YSwgMHg4MSwgMHhmMCwgMHgyMywgMHhiYywgMHg1NywgMHg5YSwgMHg2YSwgMHhlMiwgMHgzNSwKLSAgMHhiZCwgMHhiNiwgMHhhZSwgMHgwZCwgMHg1ZCwgMHg2MiwgMHhmNSwgMHg4ZCwgMHgzOSwgMHgyMSwgMHg3YiwgMHgxZSwKLSAgMHgwYSwgMHhkNywgMHhmZCwgMHhjNywgMHhmZCwgMHhlOSwgMHg1NiwgMHhiMiwgMHhiYywgMHhiNiwgMHg3NiwgMHhkNSwKLSAgMHhhNSwgMHg0OSwgMHhmNCwgMHhmZCwgMHg1NywgMHg0NywgMHhmOSwgMHgwNiwgMHhiMCwgMHhmMCwgMHg1OSwgMHg3YywKLSAgMHgyYiwgMHhiYiwgMHg3NiwgMHhmMCwgMHgxYywgMHhiNSwgMHhiYSwgMHhhZCwgMHhhZCwgMHg4ZiwgMHgzOSwgMHhiZCwKLSAgMHhmOCwgMHhhNCwgMHhmZiwgMHgwMCwgMHhlMCwgMHhmZiwgMHgwMCwgMHhkYSwgMHg5ZCwgMHhhYSwgMHg4ZSwgMHhjNywKLSAgMHgyZSwgMHg0YiwgMHhiNSwgMHg1ZSwgMHgxOCwgMHg5OCwgMHg5MiwgMHg3OSwgMHg1MiwgMHhhZCwgMHgyYywgMHg3YSwKLSAgMHg1MiwgMHg3YiwgMHhlYSwgMHhlZSwgMHg2NSwgMHhjNCwgMHgzYywgMHhjYSwgMHgxZCwgMHg2YywgMHg4NSwgMHgyMSwKLSAgMHg2MCwgMHgxMCwgMHg0NywgMHg4OCwgMHhhZiwgMHhiZSwgMHhmNiwgMHg2YSwgMHhmYiwgMHhlZCwgMHgxNiwgMHhiZSwKLSAgMHhlYSwgMHg0ZiwgMHhiZCwgMHgwZCwgMHhiZSwgMHg1ZCwgMHgzZiwgMHg4MywgMHg2YSwgMHg3MiwgMHhjYSwgMHgzZCwKLSAgMHhkNiwgMHgwOCwgMHhlYiwgMHg1OSwgMHhhMywgMHg1NSwgMHhmNCwgMHg2NiwgMHg4NywgMHhjZSwgMHgzNywgMHg1OCwKLSAgMHhjMiwgMHhkYiwgMHg5YiwgMHhjZiwgMHhiNSwgMHgwNSwgMHhmMywgMHgwNiwgMHhmOCwgMHg4YiwgMHg2ZiwgMHg5NiwKLSAgMHg5ZiwgMHg1MiwgMHg2NCwgMHg4NiwgMHg1ZSwgMHgzZiwgMHhmNSwgMHgyOSwgMHg3NSwgMHhmNCwgMHg3MCwgMHhlZSwKLSAgMHhhZiwgMHg5ZSwgMHgzMywgMHg1MSwgMHhhZSwgMHgzMywgMHhkYywgMHhmZiwgMHgwMCwgMHg2YSwgMHg3MSwgMHhkMywKLSAgMHhmNSwgMHhmMiwgMHgwMSwgMHhmZiwgMHgwMCwgMHg4YSwgMHhmYSwgMHgxYywgMHg3NywgMHg1NiwgMHg5MywgMHg3OSwKLSAgMHhjNywgMHhjMCwgMHhjZSwgMHgxYywgMHhiZiwgMHg4ZiwgMHhmMCwgMHgxNCwgMHg1MSwgMHg0NSwgMHg2NiwgMHg2OCwKLSAgMHgxNCwgMHg5YiwgMHhjNiwgMHhmMywgMHhhZSwgMHgxMCwgMHg2NSwgMHhhNywgMHhmZiwgMHgwMCwgMHg0OCwgMHg5MSwKLSAgMHhmNywgMHg2NiwgMHg5YywgMHhhOSwgMHgzMywgMHg4ZSwgMHg1ZiwgMHgyMywgMHhiOSwgMHg3NywgMHhkMCwgMHhmMiwKLSAgMHg3ZSwgMHhlYywgMHhkMCwgMHgxMSwgMHg5ZiwgMHg4MywgMHg3OSwgMHgyZCwgMHhmMCwgMHg4MiwgMHhkNSwgMHgwNSwKLSAgMHg2NywgMHhkZiwgMHg2MCwgMHgzZiwgMHgyZSwgMHgxYiwgMHhhMCwgMHhmNywgMHg4MiwgMHhkYywgMHg5NywgMHg1MywKLSAgMHhhZiwgMHhlZSwgMHgwMywgMHhmYiwgMHhlYSwgMHhjNiwgMHhhYSwgMHhiNywgMHg4MiwgMHhiMiwgMHhiYywgMHg4NywKLSAgMHgyMiwgMHhjYiwgMHgzMSwgMHhiNywgMHgzNCwgMHg5MiwgMHg2NCwgMHhiNSwgMHg3OCwgMHg4ZiwgMHhiMywgMHhmMCwKLSAgMHg5YiwgMHg5MCwgMHhkOCwgMHg0YSwgMHhmNCwgMHgzZCwgMHgwMSwgMHhkNiwgMHg5YywgMHhmZiwgMHgwMCwgMHg5OCwKLSAgMHg3YSwgMHg2YSwgMHhkMiwgMHgxZCwgMHhkNSwgMHg2OSwgMHhhZCwgMHgzMiwgMHg2OCwgMHhhYywgMHgyNSwgMHhhYSwKLSAgMHgyOSwgMHg4NSwgMHgxNCwgMHg1MSwgMHg1NSwgMHgyYywgMHgxNSwgMHg0YSwgMHhlNiwgMHg3MSwgMHg4YywgMHg0YywKLSAgMHg5YSwgMHg2YiwgMHg0NCwgMHgxMCwgMHgwYiwgMHg5YywgMHhjMywgMHhkOCwgMHg3YSwgMHhkNSwgMHhkNSwgMHg0YSwKLSAgMHhkOSwgMHg4ZCwgMHg4NiwgMHhkNywgMHgyNiwgMHg0YSwgMHg2ZSwgMHhkNywgMHgxNywgMHhkNCwgMHhjYiwgMHg0ZCwKLSAgMHgyMywgMHg5NSwgMHhjZSwgMHg1MSwgMHhmMCwgMHhiZCwgMHgxNSwgMHhlMCwgMHhmYiwgMHg0MywgMHg2MSwgMHgyYiwKLSAgMHhjYiwgMHg2NSwgMHhhMywgMHg5OCwgMHhiYywgMHhlZiwgMHhiNiwgMHhkZCwgMHg0YSwgMHg0ZSwgMHgzOSwgMHg0NSwKLSAgMHg1OSwgMHgxNiwgMHgzNCwgMHg4OSwgMHg0ZSwgMHg4NiwgMHhhMywgMHhiMiwgMHhiNywgMHg1NiwgMHg3YiwgMHg4MiwKLSAgMHg0NiwgMHhlOSwgMHhhNiwgMHgwNiwgMHgxZSwgMHgxOSwgMHg2OCwgMHg0YSwgMHhiZSwgMHhjYiwgMHg2ZSwgMHgxYiwKLSAgMHgzZCwgMHhmYywgMHg5YiwgMHhmMywgMHg4ZCwgMHg3YiwgMHg5NywgMHg5NiwgMHg0MywgMHhiNywgMHhiNCwgMHg2MywKLSAgMHg2MywgMHhkMCwgMHg1MCwgMHhjYSwgMHg3YiwgMHhiYiwgMHg2NSwgMHg4ZiwgMHgzOCwgMHhmYSwgMHhlOSwgMHg1YSwKLSAgMHg3YywgMHhlOSwgMHg3MywgMHg5ZSwgMHgyZSwgMHhjYiwgMHg5MCwgMHhiNywgMHg1NCwgMHg3ZiwgMHg0OCwgMHhkNywKLSAgMHhjNSwgMHg2MiwgMHhjYSwgMHhkNywgMHg5ZiwgMHhlZSwgMHhjYiwgMHhmMiwgMHg4ZiwgMHhmMiwgMHhjYywgMHg3NiwKLSAgMHg0MywgMHg1YiwgMHhmOSwgMHgyZCwgMHhhNiwgMHhkMCwgMHhkOSwgMHg2MywgMHgxZiwgMHg4MiwgMHg5MiwgMHhiZSwKLSAgMHhlMiwgMHhmYiwgMHg4MywgMHhhZCwgMHgyYywgMHg1YywgMHhlZSwgMHg5MywgMHhlZSwgMHg0ZSwgMHg5NywgMHgyNiwKLSAgMHg0OSwgMHg1YiwgMHg4NywgMHhkMSwgMHhiZSwgMHg4MywgMHhlYSwgMHhhZSwgMHgzYSwgMHgyYiwgMHg5YSwgMHhlNywKLSAgMHhiNCwgMHgyYiwgMHhkYywgMHgyZCwgMHgyZCwgMHhlMiwgMHgzZSwgMHg0YiwgMHg2NSwgMHhmOSwgMHgxMCwgMHhlNCwKLSAgMHhkOCwgMHg1MSwgMHg0NSwgMHgxNSwgMHhjMiwgMHg0MCwgMHg1MSwgMHg0NSwgMHgxNCwgMHgwMSwgMHg0NSwgMHgxNCwKLSAgMHg1MCwgMHgwZiwgMHgxOCwgMHhlMiwgMHg4NiwgMHg0MSwgMHg4OCwgMHhjYSwgMHhiMiwgMHgzOCwgMHg0MSwgMHg5MywKLSAgMHgxOCwgMHg3NiwgMHg5MSwgMHhjOSwgMHhmZiwgMHgwMCwgMHgwZiwgMHhmYywgMHg3ZCwgMHg3NCwgMHg4ZSwgMHhhMCwKLSAgMHg1MiwgMHhhMiwgMHg5NSwgMHgwMiwgMHgwOCwgMHgzYSwgMHgyMCwgMHhmOCwgMHg1NCwgMHg5ZSwgMHgyZiwgMHg3MywKLSAgMHg1NSwgMHhhYSwgMHhmNCwgMHhjNCwgMHhhMCwgMHg0ZiwgMHgyMCwgMHg1NywgMHgyYiwgMHg4MywgMHhkMiwgMHg5MywKLSAgMHhkZiwgMHg1MiwgMHg1YywgMHg0MiwgMHhiNiwgMHgyNiwgMHgxZCwgMHhlMCwgMHg0YywgMHg4ZSwgMHgzNywgMHgxYSwKLSAgMHg2YSwgMHg3YiwgMHg1NCwgMHgxMSwgMHhkZCwgMHhiZiwgMHgxZiwgMHhmYywgMHhmZCwgMHg3NSwgMHhlYywgMHg1NywKLSAgMHhmZiwgMHgwMCwgMHg5NSwgMHg2NywgMHgxYSwgMHhkZiwgMHg4YSwgMHg5ZiwgMHg3NSwgMHhmYywgMHgzZiwgMHgwYiwKLSAgMHhmZCwgMHhiZiwgMHgyMiwgMHhjZiwgMHg3NSwgMHg5MSwgMHg2YSwgMHhhZCwgMHgyZSwgMHgxOCwgMHg1ZCwgMHhiYywKLSAgMHhiMiwgMHhkMCwgMHg2MCwgMHgzYSwgMHhhZCwgMHhiYiwgMHgxNywgMHhhMiwgMHg3NywgMHhlMiwgMHg4MywgMHhkZCwKLSAgMHhmZCwgMHhkZCwgMHhkNSwgMHg1NiwgMHhkNCwgMHhjNiwgMHgxOSwgMHg3MSwgMHg1ZCwgMHhiNywgMHgyMSwgMHg4YywKLSAgMHhlYSwgMHg0OSwgMHhlNCwgMHg3MSwgMHg0MSwgMHhiNywgMHgwNywgMHhhNCwgMHgxMywgMHhhZiwgMHhmYywgMHgxYSwKLSAgMHhhZiwgMHg2MiwgMHhkZSwgMHhiYiwgMHgzYiwgMHhiOCwgMHhjOSwgMHhmMCwgMHhmNiwgMHg3ZiwgMHgwNywgMHhmYywKLSAgMHgwOCwgMHgzYywgMHgzMiwgMHhlOSwgMHhhYywgMHgxMywgMHhhYSwgMHhjZCwgMHg2OCwgMHg5YiwgMHgyNSwgMHg5OCwKLSAgMHg5MSwgMHg5ZSwgMHg5NSwgMHgyNSwgMHhjNCwgMHhiNCwgMHhjMywgMHgyZCwgMHhhOSwgMHhjNywgMHgxNiwgMHhhMywKLSAgMHhhMCwgMHg5NCwgMHhhNCwgMHg2YywgMHg5MywgMHhlYywgMHgxNSwgMHhmYSwgMHhhMSwgMHhkMiwgMHg3YywgMHhmOSwKLSAgMHg5MiwgMHgzYywgMHhkYywgMHhkZSwgMHgyNywgMHg1YywgMHhlZSwgMHgwYywgMHhhYiwgMHg5OSwgMHgwNywgMHgzYiwKLSAgMHhiMywgMHhjMywgMHhkZiwgMHhlYiwgMHgzMiwgMHhkYiwgMHgwMSwgMHg1ZiwgMHhmNSwgMHgyOCwgMHhkNywgMHhkMSwKLSAgMHg0MywgMHhiYSwgMHhiZSwgMHg2OSwgMHhiMywgMHhhMSwgMHhkNywgMHg3MSwgMHhmYywgMHg1NiwgMHhlZiwgMHgyNSwKLSAgMHhiMiwgMHhkYywgMHg5YiwgMHhkZSwgMHg2MywgMHgxNiwgMHhlYywgMHhmMiwgMHg3ZCwgMHgwNiwgMHg0NCwgMHhhMiwKLSAgMHhiNCwgMHg4ZiwgMHhhOSwgMHgwNSwgMHgwMywgMHhlYSwgMHhhZiwgMHhhNSwgMHg4NSwgMHg2YiwgMHg1NiwgMHgzYSwKLSAgMHg3MCwgMHhiZCwgMHgwYywgMHg2OCwgMHhjYiwgMHg1NiwgMHg1ZiwgMHhhOCwgMHg1MSwgMHg0NSwgMHgxNSwgMHg5MSwKLSAgMHhiMCwgMHg1MiwgMHg2NywgMHgxYywgMHhiZSwgMHg0NywgMHg3MiwgMHhlZiwgMHhhMSwgMHhlNCwgMHhmZCwgMHhkOSwKLSAgMHhhNywgMHgzYSwgMHg0YywgMHhlMywgMHg5MSwgMHhkNywgMHgwNywgMHg3MiwgMHhmZCwgMHhmNywgMHg3YiwgMHg5MSwKLSAgMHgyNywgMHhlZSwgMHhjZCwgMHgwMCwgMHg4MywgMHg3NCwgMHg5OCwgMHgzMSwgMHg5YywgMHg4ZSwgMHhjMywgMHg5YSwKLSAgMHg5NSwgMHg3MiwgMHg0NCwgMHg4NiwgMHgwYywgMHgxYiwgMHhhOSwgMHgxZSwgMHgxMCwgMHhkZSwgMHgyOSwgMHhmNywKLSAgMHhjNSwgMHg3OCwgMHhlOSwgMHhiNywgMHgwMywgMHg2YiwgMHhmOSwgMHhiYywgMHhmNSwgMHg3OCwgMHhhNywgMHhiOCwKLSAgMHg1NSwgMHg2NSwgMHgxYSwgMHhkMCwgMHhmNSwgMHhkZSwgMHgxMiwgMHhhMiwgMHgxOCwgMHg3ZSwgMHg1MywgMHgxZCwKLSAgMHhlNiwgMHhiYiwgMHgzNywgMHg1MCwgMHhlMCwgMHhkMiwgMHgxNiwgMHg5MiwgMHgzNCwgMHg1MiwgMHg0OSwgMHhmMCwKLSAgMHgyMCwgMHg5MSwgMHg1ZCwgMHg1YywgMHgzNiwgMHg5ZiwgMHgzNywgMHgxOSwgMHg5YywgMHg5ZSwgMHgxYywgMHhlNCwKLSAgMHg2ZiwgMHhmMywgMHhiZiwgMHgxOSwgMHg5MiwgMHhiYiwgMHgxYywgMHhjNSwgMHg5ZCwgMHhmOSwgMHg3YywgMHgzNCwKLSAgMHhmZSwgMHg2NiwgMHhmYywgMHg1ZSwgMHg2YiwgMHhhMiwgMHg1NCwgMHgzYiwgMHhjYSwgMHg3OSwgMHg1NSwgMHhkNywKLSAgMHg2YSwgMHhkNywgMHg1ZCwgMHhkYywgMHg2MiwgMHhhNSwgMHg5NCwgMHhjZSwgMHg0YiwgMHg0OSwgMHg0YiwgMHg0ZSwKLSAgMHgxYSwgMHhlMCwgMHhiMCwgMHhlYSwgMHgzNywgMHgyMywgMHhiYiwgMHhiMywgMHg2NSwgMHhiNiwgMHgyZSwgMHg2MywKLSAgMHhjMywgMHg5OCwgMHg4ZSwgMHg4OCwgMHg0NiwgMHhmNSwgMHhjYywgMHhhZiwgMHg0NSwgMHg0OSwgMHgwYSwgMHg0YSwKLSAgMHhlMiwgMHhkMywgMHgyZSwgMHhhZSwgMHhkNSwgMHgxNSwgMHhkNCwgMHg2YywgMHhhMSwgMHgwZSwgMHg5ZSwgMHg3ZCwKLSAgMHg3YSwgMHhjNywgMHg0YSwgMHhmMSwgMHhiYiwgMHg0ZSwgMHhlMiwgMHg3NiwgMHhmNiwgMHg5MywgMHhhYiwgMHg0ZiwKLSAgMHg5NCwgMHg4ZSwgMHhhOSwgMHgzYywgMHgyMSwgMHg1YSwgMHg1ZSwgMHg2OSwgMHg3ZSwgMHg3ZCwgMHhmMiwgMHhlMiwKLSAgMHgyNSwgMHgwNiwgMHg1MywgMHhiZSwgMHg4OCwgMHg0MiwgMHg0NiwgMHhhOSwgMHg4YiwgMHgxYiwgMHhjOCwgMHg3ZiwKLSAgMHg5NCwgMHg3MSwgMHg5ZSwgMHhiMiwgMHg1ZCwgMHgxMiwgMHg4ZSwgMHhkOSwgMHhjNiwgMHhjOCwgMHg0MiwgMHhjMCwKLSAgMHhkMCwgMHg1NywgMHhmZiwgMHgwMCwgMHhiYSwgMHhhZSwgMHhhYiwgMHg3ZCwgMHhiZSwgMHg1MywgMHg5MCwgMHhhNiwKLSAgMHhiMywgMHgyZCwgMHhhMywgMHhhNSwgMHhiNCwgMHhiMCwgMHhhMSwgMHg1ZiwgMHg5ZCwgMHg1YSwgMHhmNiwgMHhjNSwKLSAgMHhjZCwgMHgzYSwgMHhiYSwgMHhhYSwgMHhjZCwgMHhjYSwgMHgyZiwgMHg5NCwgMHhmNywgMHhjYSwgMHhlYSwgMHg3MywKLSAgMHhhOSwgMHhiYywgMHhlZSwgMHgxNywgMHgwOCwgMHhhZSwgMHg0MiwgMHg5YSwgMHhmNCwgMHg1NywgMHg0NiwgMHg5NiwKLSAgMHhkMiwgMHg4YSwgMHg0ZCwgMHg2OCwgMHhhNywgMHgxZSwgMHgyMCwgMHg0NSwgMHg2ZSwgMHg1YywgMHg3OCwgMHg5OSwKLSAgMHgwNCwgMHg1MCwgMHgwYiwgMHg3MiwgMHgxMCwgMHgwMywgMHg5YSwgMHhmMCwgMHg1NSwgMHgyNywgMHg1NywgMHgyNSwKLSAgMHhmNSwgMHhiNywgMHhkOSwgMHhhYiwgMHhjYSwgMHg5YSwgMHhlMywgMHg5NSwgMHhlYSwgMHg5ZiwgMHgwNCwgMHg0OSwKLSAgMHg2MSwgMHg4NSwgMHgxNCwgMHg1MSwgMHg1YywgMHg2NCwgMHgwNSwgMHgxNCwgMHg1MSwgMHg0MCwgMHgxNCwgMHg1MSwKLSAgMHg0NSwgMHgwMCwgMHg1MSwgMHg0NSwgMHgxNCwgMHgwMSwgMHg0ZiwgMHg1NiwgMHg4NCwgMHhmZiwgMHgwMCwgMHgyOSwKLSAgMHhmMCwgMHhiNywgMHgyZCwgMHhhNCwgMHg4MywgMHgzNiwgMHgxMSwgMHhlNiwgMHg2NCwgMHg5ZSwgMHhmMiwgMHgzYywKLSAgMHgwNywgMHhkYSwgMHgzZSwgMHhiMSwgMHg0OCwgMHhiNSwgMHgzMywgMHg4NiwgMHhkZCwgMHgwZCwgMHhhNiwgMHhmYSwKLSAgMHhjYiwgMHhlNCwgMHg5ZSwgMHhjOSwgMHg2NywgMHg5MSwgMHhjMSwgMHhlYSwgMHgzNSwgMHhlOSwgMHg3NiwgMHg2NSwKLSAgMHg3OCwgMHg1MiwgMHhhZCwgMHhhMiwgMHhhNywgMHg4MiwgMHg3ZCwgMHhkNywgMHhmMCwgMHg3ZCwgMHg3ZSwgMHg1YywKLSAgMHg5NiwgMHg4YiwgMHhjMywgMHgyMiwgMHhhNCwgMHhjNywgMHg3YSwgMHgzMywgMHhjYSwgMHg2NSwgMHhmNiwgMHg5NCwKLSAgMHhkYiwgMHg4OSwgMHgzYSwgMHgyOSwgMHg1MCwgMHhkNSwgMHgzMiwgMHhmMCwgMHhmZiwgMHgwMCwgMHgxZiwgMHg3ZSwKLSAgMHhlMywgMHg3NCwgMHg2ZSwgMHg2YiwgMHhjOCwgMHgyOCwgMHg4OSwgMHgxZCwgMHg0MSwgMHg0NCwgMHg5MSwgMHhmMCwKLSAgMHhkNCwgMHgzYiwgMHg5MiwgMHgzZiwgMHhlZiwgMHg1NiwgMHg4MywgMHhkMCwgMHhlMSwgMHg0YSwgMHhlNSwgMHg3MSwKLSAgMHhkOCwgMHhjYywgMHhiYiwgMHhlMiwgMHgxNCwgMHhhNCwgMHgwMywgMHg1YiwgMHg5YiwgMHg0MiwgMHgxYiwgMHg0MCwKLSAgMHg0MywgMHg2OCwgMHg0YSwgMHgxMiwgMHgzYiwgMHg4MiwgMHg0NiwgMHg4MCwgMHhhZiwgMHhhYiwgMHhiNCwgMHhmNiwKLSAgMHg1YSwgMHgxNCwgMHhhYiwgMHhhYSwgMHg5MywgMHg5ZSwgMHhhOCwgMHhhZCwgMHhkMiwgMHhjNywgMHhkNCwgMHhkMSwKLSAgMHg1MywgMHhjMywgMHgzZCwgMHg1NSwgMHg3NSwgMHhjNywgMHg3YiwgMHg4MiwgMHhkNywgMHg4ZSwgMHg0NywgMHhjNCwKLSAgMHgyMiwgMHgzYSwgMHg1MSwgMHgzNywgMHgyNSwgMHg3YiwgMHhjOCwgMHhjOSwgMHg0OSwgMHhmMywgMHg5YiwgMHg4OCwKLSAgMHgwMCwgMHg1NCwgMHhhNywgMHgzZCwgMHg4MSwgMHhiZCwgMHhhMywgMHhkYSwgMHhlMiwgMHg2OSwgMHhlYSwgMHhmMSwKLSAgMHg3MiwgMHg4MywgMHg2NywgMHhiNSwgMHhjYSwgMHhiYSwgMHg1YywgMHhhNSwgMHgzNywgMHgxNiwgMHgxNCwgMHg0NiwKLSAgMHg5NCwgMHhmMywgMHhlZiwgMHgzOCwgMHg3NCwgMHg5NCwgMHgyMSwgMHgyMywgMHg2NCwgMHg5MywgMHg1NCwgMHhlZCwKLSAgMHg5NSwgMHhjOSwgMHhiOSwgMHgwZCwgMHhmZSwgMHg2ZSwgMHg2ZiwgMHg3NSwgMHg4YywgMHhlNCwgMHg3NywgMHhhNiwKLSAgMHhiNiwgMHgyMywgMHhkYiwgMHg2MiwgMHgzYSwgMHgzNCwgMHhiOCwgMHg3MCwgMHg0MSwgMHhkYSwgMHg0MiwgMHhjNywKLSAgMHg4MywgMHg4ZSwgMHgxZiwgMHgzZCwgMHg0MywgMHhiYywgMHg2ZCwgMHgyOSwgMHhmYywgMHhkYSwgMHhmYiwgMHgzYSwKLSAgMHgxNCwgMHg5ZCwgMHg0OSwgMHhlMCwgMHg4YSwgMHhkNSwgMHg1NSwgMHgzOCwgMHhlNywgMHhhOSwgMHhhNywgMHgyZSwKLSAgMHg0MywgMHg2ZCwgMHgzZCwgMHg4YSwgMHhiNCwgMHhkMiwgMHgwMiwgMHgxYiwgMHg0ZSwgMHg0YiwgMHg2ZCwgMHg0YSwKLSAgMHg1MiwgMHgwNiwgMHg4MiwgMHg0MCwgMHg3OCwgMHg2OCwgMHgwZiwgMHg2NSwgMHg1ZSwgMHg2MywgMHhiYSwgMHhhOCwKLSAgMHhlYywgMHhjYiwgMHhlMywgMHgzOCwgMHhiNywgMHg3ZiwgMHhmYSwgMHhjZiwgMHg2ZSwgMHhmYiwgMHhlMSwgMHg1NywKLSAgMHg4OCwgMHhlZSwgMHhhZCwgMHgyZiwgMHgzZSwgMHhmMywgMHhlNCwgMHg2NywgMHg2NywgMHhmNywgMHg2MSwgMHg0NSwKLSAgMHgxNCwgMHg1NywgMHgyOSwgMHhkNCwgMHgxNCwgMHg5OSwgMHhjNywgMHgxZiwgMHg5MSwgMHhlYywgMHhiNywgMHhlOCwKLSAgMHg4OSwgMHgxZiwgMHhjMCwgMHg2OSwgMHhjZSwgMHg5MywgMHgzOCwgMHhlMywgMHhmMiwgMHgzZCwgMHg5NywgMHg3ZCwKLSAgMHgxMSwgMHgyMywgMHhmOCwgMHgwZCwgMHgwMCwgMHhkYiwgMHgxMSwgMHgwOSwgMHg0YywgMHg1NiwgMHg5MiwgMHg5MCwKLSAgMHgwMCwgMHgwOSwgMHgxZCwgMHgwMCwgMHhhOCwgMHg1YywgMHhlNywgMHgxNywgMHg4MSwgMHg5NSwgMHhkOSwgMHhjNCwKLSAgMHgwOSwgMHg2YiwgMHg3NiwgMHgzYiwgMHhlYywgMHhiOCwgMHg5OSwgMHgxMCwgMHg2NiwgMHhiMiwgMHg3NCwgMHhmNCwKLSAgMHgzOSwgMHgwOSwgMHhmOCwgMHgwZSwgMHhiNiwgMHg3ZiwgMHg0OCwgMHgxZiwgMHgwZSwgMHhlMiwgMHgzNiwgMHgwOCwKLSAgMHgyMCwgMHg5YSwgMHg5YywgMHg4ZCwgMHhmMSwgMHg3NiwgMHhmZSwgMHg2MCwgMHhmYiwgMHgyYiwgMHg2MSwgMHgxYiwKLSAgMHhhMCwgMHgxMSwgMHhmMSwgMHgzYywgMHhiNiwgMHg3NCwgMHg3YiwgMHhhYiwgMHg3OCwgMHg5NiwgMHg2YywgMHhkYiwKLSAgMHgzMCwgMHhhZiwgMHhkYSwgMHgzZSwgMHg0NywgMHgyOSwgMHgwMywgMHg5NiwgMHgzNSwgMHhkOSwgMHgwMywgMHhmMywKLSAgMHhkOSwgMHgyNywgMHhiOSwgMHhjMCwgMHgzYSwgMHhhOSwgMHhhMywgMHhlNywgMHgwZSwgMHhhNCwgMHg2ZCwgMHgzZCwKLSAgMHg2OSwgMHhiZSwgMHhlMSwgMHgxMiwgMHgzZCwgMHhjMiwgMHgxMywgMHg5MSwgMHgyNCwgMHhhMCwgMHgyZCwgMHhhNywKLSAgMHgwNiwgMHg4OCwgMHhhZSwgMHgzYywgMHg5YiwgMHgxZiwgMHhiMywgMHhlNCwgMHg5NiwgMHg4NywgMHgyZCwgMHg1NywKLSAgMHhiOCwgMHgyZCwgMHhjYywgMHg4OCwgMHhiMiwgMHgxNSwgMHhjOCwgMHhhZCwgMHg4MiwgMHg4NSwgMHgwZSwgMHhlNSwKLSAgMHhhMSwgMHg0MywgMHhhYSwgMHgxNCwgMHgzYiwgMHhjMiwgMHg5MiwgMHg0MSwgMHgwNywgMHhiYSwgMHg5MywgMHhmYiwKLSAgMHg0YywgMHhkNywgMHgwNSwgMHg0NiwgMHg5ZiwgMHg0YywgMHhiYywgMHhkMywgMHgxZSwgMHg0NywgMHg3MywgMHhhZCwKLSAgMHg4MCwgMHg2ZSwgMHhiMSwgMHgxMywgMHhkNywgMHhlMSwgMHgyNywgMHhhMiwgMHg2NCwgMHhhNCwgMHgwMCwgMHgzYSwKLSAgMHg4ZSwgMHg1NywgMHgzZCwgMHg0YiwgMHgzZCwgMHg0ZCwgMHg2NSwgMHgxNSwgMHgyNCwgMHhlMywgMHgyNSwgMHg5NCwKLSAgMHhjMSwgMHgxZCwgMHg5MSwgMHhlMSwgMHgxNywgMHgxYiwgMHg3OSwgMHg1MywgMHhmMCwgMHg0MiwgMHhhNiwgMHhjNiwKLSAgMHhlZiwgMHhmMywgMHg0NywgMHhiZSwgMHgyNywgMHhkYSwgMHgzYywgMHg3ZCwgMHhhMiwgMHg5NSwgMHgwOCwgMHgyMCwKLSAgMHg5MCwgMHg0MSwgMHgwNCwgMHg3NCwgMHgyMCwgMHhkNSwgMHhkMSwgMHg4YSwgMHhlNCwgMHhmNiwgMHgyYywgMHhhMiwKLSAgMHgwNywgMHg5NywgMHg1OCwgMHhhZSwgMHg2YywgMHg0ZCwgMHg2OSwgMHgyNywgMHg5NSwgMHhjMCwgMHg4MiwgMHg0MywKLSAgMHg4YywgMHhhYiwgMHhmNCwgMHg1YywgMHg0MSwgMHhkMiwgMHg5MCwgMHhhZiwgMHhkNSwgMHg1MCwgMHgwNywgMHhkNSwKLSAgMHg1ZSwgMHhhZiwgMHg1OCwgMHhlZCwgMHhhMiwgMHhlZiwgMHhiNSwgMHg0YSwgMHg4YSwgMHg5MCwgMHhlZiwgMHhmNSwKLSAgMHhhZCwgMHhmOSwgMHhhYiwgMHhmZSwgMHhmMSwgMHhkZiwgMHhmNSwgMHhlZSwgMHhiZSwgMHg0NiwgMHhmZiwgMHgwMCwKLSAgMHhkOSwgMHg1OCwgMHhjZCwgMHhiOSwgMHhkYSwgMHhjYiwgMHgxZSwgMHg4ZiwgMHg4ZiwgMHg5MywgMHgzMiwgMHg5NSwKLSAgMHgzZiwgMHgyMSwgMHgyNywgMHgwNiwgMHg3ZCwgMHhiYiwgMHhhNSwgMHhhMiwgMHg2NiwgMHgzYiwgMHgyNCwgMHhlZiwKLSAgMHg5OSwgMHgyNSwgMHg2YywgMHg2ZiwgMHhjMCwgMHhkMiwgMHg3YywgMHhhNiwgMHgxYywgMHg4ZCwgMHgyNSwgMHhjNiwKLSAgMHgxZCwgMHgxYSwgMHg1YiwgMHg2YSwgMHgyOSwgMHgyMiwgMHhhYywgMHgwNCwgMHg2MSwgMHgxMywgMHhhZCwgMHg3NywKLSAgMHgyNiwgMHg2NywgMHhkYSwgMHgyNiwgMHhhMSwgMHhkZSwgMHhjOSwgMHg3YiwgMHhlYywgMHhkZCwgMHgxYywgMHhhYSwKLSAgMHgyMywgMHhjNCwgMHg2YywgMHg3NCwgMHgzZiwgMHhlMSwgMHg1YywgMHg1YywgMHg0OCwgMHhiMSwgMHhjOSwgMHhmMiwKLSAgMHhlNCwgMHg1YywgMHhlMywgMHg0NSwgMHg3MSwgMHg0OSwgMHg3OSwgMHgzZSwgMHhmYywgMHgxMCwgMHg5MiwgMHhhZSwKLSAgMHg1NSwgMHg3YSwgMHhmNSwgMHg1ZSwgMHg1NSwgMHhlZiwgMHg2NywgMHg1YywgMHhmZCwgMHg4ZCwgMHgzYSwgMHhiMCwKLSAgMHg3YSwgMHhhOSwgMHhlZCwgMHhlNywgMHg5OCwgMHhiZSwgMHgzOCwgMHhmMiwgMHg3ZiwgMHhhMSwgMHg1NywgMHgxNywKLSAgMHg4ZCwgMHhjNCwgMHg4YSwgMHgyOCwgMHgyMCwgMHg4MiwgMHg0MSwgMHgwNCwgMHgxMSwgMHhlMCwgMHg2OCwgMHhhZiwKLSAgMHg5YSwgMHgzMywgMHgwYSwgMHgyOCwgMHhhMiwgMHg4MCwgMHgyOCwgMHhhMiwgMHhiZCwgMHgzNCwgMHhkMywgMHg4ZiwKLSAgMHgyYywgMHgyMSwgMHhhNiwgMHhkNiwgMHhlMiwgMHg4ZiwgMHg3MiwgMHg1MiwgMHg5ZCwgMHg5YSwgMHg5NCwgMHg5YiwKLSAgMHhkOSwgMHgwMywgMHhjZCwgMHgxNSwgMHgzOSwgMHg2ZiwgMHhjNCwgMHhhZiwgMHhmMywgMHg0OCwgMHgyOSwgMHg4MCwKLSAgMHhiNiwgMHg1MCwgMHg3ZiwgMHgzOSwgMHhmZiwgMHgwMCwgMHgzMywgMHg1ZiwgMHg1MSwgMHhlYiwgMHhmZSwgMHgxNCwKLSAgMHhkMywgMHg2OSwgMHhlMSwgMHhkYiwgMHgyOCwgMHgyMSwgMHg3NywgMHg0OSwgMHhhNSwgMHhjMywgMHhlMiwgMHhkYiwKLSAgMHgwMywgMHg0MywgMHhmZSwgMHg2MywgMHhkNCwgMHhmZiwgMHgwMCwgMHg3MCwgMHhhZiwgMHg0ZSwgMHhkYiwgMHhiMSwKLSAgMHhhZiwgMHg2ZSwgMHg1ZiwgMHg3NiwgMHg5YiwgMHg0YiwgMHhjZCwgMHhlYywgMHhiZiwgMHg1ZiwgMHhkOCwgMHhiMiwKLSAgMHg4MywgMHg2NSwgMHg3ZCwgMHgxNiwgMHgzYiwgMHhmMiwgMHg5ZiwgMHg0YiwgMHgxMSwgMHg5OSwgMHg1YiwgMHhjZSwKLSAgMHhhYiwgMHhiOSwgMHgyOCwgMHg0ZSwgMHhjOSwgMHhhNywgMHhiYywgMHg2NywgMHgwMiwgMHg1MSwgMHg1MiwgMHgyNCwKLSAgMHhkZSwgMHhkNSwgMHhjYSwgMHgwNiwgMHg4OCwgMHg4ZSwgMHg4NSwgMHg3NSwgMHgzZiwgMHgzOCwgMHg4ZiwgMHhiMCwKLSAgMHg3ZiwgMHg3ZCwgMHgzYiwgMHhkYiwgMHgyZCwgMHg5MCwgMHgyZCwgMHg4YywgMHhmNiwgMHg1MCwgMHg2MiwgMHhiNiwKLSAgMHhjMiwgMHg3YywgMHg0YSwgMHg0NywgMHg1MywgMHhlZCwgMHgzZCwgMHhlNywgMHhlYiwgMHhhZSwgMHhhMiwgMHg0OCwKLSAgMHhhZiwgMHhhZCwgMHhlYywgMHhmZiwgMHgwMCwgMHg2NiwgMHgyOCwgMHhkMSwgMHg2YSwgMHg3NywgMHgwZiwgMHg1MywKLSAgMHhmMiwgMHhlOSwgMHhmZSwgMHg3ZiwgMHhkZCwgMHg4ZCwgMHg2MywgMHg0ZCwgMHgyZSwgMHg0MSwgMHhiNCwgMHgyNSwKLSAgMHhiNiwgMHhkMiwgMHhkYSwgMHgxMiwgMHgxMiwgMHg5NCwgMHg4MCwgMHgxMiwgMHgwNywgMHg3MCwgMHgwMiwgMHhiOSwKLSAgMHg2ZSwgMHhkNywgMHgxOCwgMHgzNiwgMHg5YiwgMHg3NCwgMHg4YiwgMHg5NSwgMHhjYSwgMHg1YiwgMHgxMCwgMHhlMSwKLSAgMHg0NiwgMHg2YywgMHhiOCwgMHhmYiwgMHhlZiwgMHgyYywgMHgyNSwgMHgwZCwgMHhhNCwgMHg3NywgMHg5MiwgMHg0ZiwKLSAgMHg0MSwgMHg0YiwgMHg3OSwgMHg0NiwgMHg3ZCwgMHg2YywgMHhiNSwgMHg1YywgMHg3ZCwgMHhjNCwgMHhiNSwgMHhjNywKLSAgMHg3ZiwgMHgyMCwgMHhjOCwgMHg1NCwgMHgzYywgMHhjYiwgMHg1ZCwgMHhiZiwgMHg0YiwgMHg3MSwgMHgxZSwgMHg4NSwKLSAgMHgzYywgMHhiMiwgMHg3OSwgMHgxOCwgMHg0NywgMHg3NywgMHg5YywgMHhiMiwgMHgzZCwgMHg0MSwgMHg0NywgMHhhNSwKLSAgMHg3MCwgMHg1YiwgMHhmMCwgMHhlYiwgMHhhNSwgMHhmZSwgMHhlNiwgMHhjNSwgMHhlZiwgMHg4OCwgMHg3MiwgMHg2MiwKLSAgMHhjZCwgMHg1YywgMHg3NywgMHgwMywgMHhiMCwgMHhhYywgMHhiMSwgMHg0OSwgMHgzMCwgMHgyMSwgMHhhYywgMHgxZiwKLSAgMHgzNSwgMHg2YSwgMHhlNiwgMHgwMCwgMHhiZSwgMHhlOCwgMHhmZCwgMHgyNSwgMHgwMCwgMHg5NCwgMHg5ZiwgMHg4MiwKLSAgMHg5MCwgMHg3YSwgMHhkNywgMHhkNCwgMHhmMCwgMHg2OCwgMHg3MiwgMHg0NCwgMHg4YiwgMHgzMywgMHg4OSwgMHgzNywKLSAgMHgxOCwgMHhkNywgMHg3YiwgMHhjNCwgMHg1NywgMHhlMCwgMHhlMiwgMHgxMSwgMHgxZSwgMHg0MywgMHhmNiwgMHhlYiwKLSAgMHg3MywgMHhjOSwgMHgyOCwgMHg3NiwgMHhlOCwgMHhlYSwgMHg0ZSwgMHhkMywgMHgyMSwgMHhmNCwgMHg5ZSwgMHhhOSwKLSAgMHg2NSwgMHgyNCwgMHgwMiwgMHhkYiwgMHg2NywgMHhhYSwgMHg4OCwgMHhlNywgMHg1ZSwgMHg4NiwgMHg5MywgMHg0ZSwKLSAgMHhmNywgMHgzYiwgMHgzNCwgMHgxOSwgMHhlOSwgMHhkYiwgMHg4ZCwgMHhmMiwgMHgzOSwgMHhlMCwgMHhiNCwgMHg3NCwKLSAgMHgyMywgMHhmZiwgMHgwMCwgMHgzNSwgMHgyMiwgMHgwMCwgMHhkNSwgMHgxYSwgMHgxYSwgMHhhOSwgMHg4YywgMHg5YywKLSAgMHg1ZSwgMHg1MSwgMHg1OSwgMHg0NSwgMHg0OSwgMHg2MSwgMHg5NSwgMHgxNywgMHgxMiwgMHhlYywgMHhkMiwgMHg2ZCwKLSAgMHhkMiwgMHgzMSwgMHg2NywgMHg1NCwgMHhhNCwgMHhiOCwgMHhjOSwgMHhjYSwgMHgyZCwgMHhhMCwgMHgyYywgMHg3NCwKLSAgMHgyMywgMHhkZiwgMHg4NywgMHg3OCwgMHhhYiwgMHg3OCwgMHg3NywgMHg1MiwgMHgyNywgMHgxOSwgMHg0NywgMHhmMywKLSAgMHgyYywgMHg1ZiwgMHhmNiwgMHhhYSwgMHhkOSwgMHhmNywgMHhlMiwgMHg5ZSwgMHhlYSwgMHhkNSwgMHgyYSwgMHg0YSwKLSAgMHhhMywgMHhjYywgMHg4OCwgMHhhNywgMHg0ZCwgMHg1MywgMHg1OCwgMHg0MSwgMHg0NSwgMHg3OSwgMHgyNCwgMHg4MywKLSAgMHhhMCwgMHgwMSwgMHhmYSwgMHhlOCwgMHhhYSwgMHgxNywgMHgzZCwgMHg1MiwgMHg2NywgMHgxYywgMHg3ZSwgMHg0NywKLSAgMHhiMiwgMHhlZiwgMHhhMiwgMHgyNCwgMHg3ZiwgMHgwMSwgMHhhNywgMHgzYSwgMHg0YywgMHhlMywgMHg4ZiwgMHhjOCwKLSAgMHhmNiwgMHg1ZCwgMHhmNCwgMHg0NCwgMHg4ZiwgMHhlMCwgMHgzNCwgMHgwMywgMHg3YywgMHg2ZiwgMHg4YiwgMHhiNywKLSAgMHhmMywgMHgwNywgMHhkOSwgMHg1YiwgMHgyYiwgMHg1YywgMHg2ZiwgMHg4YiwgMHhiNywgMHhmMywgMHgwNywgMHhkOSwKLSAgMHg1YiwgMHgyOCwgMHgwMywgMHg1ZCwgMHg2OCwgMHhkNSwgMHgxNCwgMHg1MCwgMHgwYSwgMHhkOSwgMHgzZSwgMHgwNywKLSAgMHg4ZSwgMHhkZiwgMHhhNywgMHg3YiwgMHhhNiwgMHhlNCwgMHg3NywgMHhhZCwgMHhmNywgMHg3NCwgMHg4ZCwgMHgyMiwKLSAgMHhlOSwgMHg2ZSwgMHg3OCwgMHhjNiwgMHg5NiwgMHg5ZiwgMHg1NywgMHg2OCwgMHg4ZCwgMHgxNSwgMHgwZiwgMHhkNSwKLSAgMHg1ZiwgMHgzMiwgMHg3ZCwgMHg1NSwgMHgxNCwgMHgyMiwgMHhmMSwgMHgzNywgMHgxZSwgMHhmOCwgMHg5NCwgMHhmYiwKLSAgMHg1ZSwgMHg2MywgMHgwZCwgMHgwMCwgMHhlOSwgMHhhOSwgMHhjMywgMHhjOCwgMHg2NiwgMHg4MSwgMHhlMCwgMHgwMywKLSAgMHhhOCwgMHg0OSwgMHg2OSwgMHg2NywgMHhkYSwgMHg4NCwgMHg3YiwgMHg3YywgMHg2OSwgMHhmNywgMHg1ZCwgMHg3NywKLSAgMHg0NywgMHgyOCwgMHhhMCwgMHgxMSwgMHgwNywgMHgxMywgMHg2ZCwgMHhkNiwgMHhmMCwgMHg1MywgMHg5NSwgMHhkOCwKLSAgMHg3MiwgMHgxYywgMHg2NCwgMHhhMCwgMHgwMCwgMHhhNywgMHg2NiwgMHg0MCwgMHg1MywgMHhiMSwgMHhiNywgMHhlYSwKLSAgMHg3ZCwgMHg5ZSwgMHg3NiwgMHhmNSwgMHhlYiwgMHg1MSwgMHg0ZiwgMHhkNSwgMHg0YywgMHhiNiwgMHgyYywgMHg5ZiwKLSAgMHgxZCwgMHhiZiwgMHhhNCwgMHhhYSwgMHhjNywgMHg3ZCwgMHhiNiwgMHg1YywgMHhmNCwgMHgzNiwgMHg0NCwgMHg1OSwKLSAgMHg0OCwgMHg3MCwgMHg4MSwgMHhlYiwgMHgwOSwgMHgyNCwgMHg4YSwgMHg5NiwgMHhkNSwgMHgyZSwgMHg1ZiwgMHhmMCwKLSAgMHgyYywgMHgyYSwgMHhmYywgMHhiNSwgMHgzOSwgMHg3NywgMHhjNSwgMHhhZCwgMHgxMiwgMHhkZSwgMHg1NywgMHg3YiwKLSAgMHhjYSwgMHg4YSwgMHg4MCwgMHhlZiwgMHhkNCwgMHhiMCwgMHgwMiwgMHg4NywgMHhkNCwgMHg2OCwgMHgwOSwgMHhjOSwKLSAgMHgzMSwgMHgyMiwgMHg0OSwgMHgxZiwgMHhjZSwgMHg2MiwgMHgzMCwgMHhmOCwgMHhmZiwgMHgwMCwgMHg2OCwgMHhkOCwKLSAgMHg1NywgMHhkYiwgMHg1MSwgMHhlZSwgMHhlMywgMHgzNiwgMHgwNywgMHhiNywgMHhjZiwgMHg2OSwgMHg4YywgMHgzZSwKLSAgMHg2YSwgMHg3OSwgMHg3ZSwgMHhjYSwgMHg1YiwgMHg1ZiwgMHgwYiwgMHhhYywgMHg2YywgMHgyMSwgMHgyZCwgMHhkOSwKLSAgMHg2ZiwgMHg1OSwgMHg1NSwgMHg5MSwgMHgyOSwgMHhlZSwgMHg0YywgMHgyYiwgMHhlNCwgMHg4ZSwgMHg0MSwgMHhmZiwKLSAgMHgwMCwgMHgwMywgMHg4YSwgMHg1YSwgMHg3ZiwgMHhjMiwgMHhhMSwgMHgzMiwgMHhmMCwgMHgzMCwgMHhiOCwgMHg2MSwKLSAgMHhkYiwgMHg5ZiwgMHgxYywgMHhhNiwgMHhkOSwgMHg1YSwgMHhkNywgMHg5OCwgMHgyZSwgMHhjOCwgMHg4MiwgMHhlYSwKLSAgMHg5NywgMHhlYywgMHgwNSwgMHhhNCwgMHhhOSwgMHg1ZiwgMHg1NiwgMHhlYiwgMHgxYSwgMHg5NiwgMHhkNCwgMHg2YSwKLSAgMHg3OCwgMHhlMCwgMHg5ZiwgMHhjNSwgMHgyNiwgMHg0NiwgMHgxMCwgMHhlZSwgMHhhYywgMHgzNywgMHgxYiwgMHgyNywKLSAgMHhhZCwgMHhiMCwgMHg3ZCwgMHg0ZiwgMHgzOCwgMHgzZiwgMHhmZSwgMHhkNSwgMHg5NCwgMHg2MSwgMHhmOCwgMHhkYSwKLSAgMHg0ZiwgMHg0YiwgMHg2MiwgMHg0ZiwgMHhiNSwgMHhkNSwgMHg5ZiwgMHhiNSwgMHg1NSwgMHg0ZSwgMHg1YSwgMHhiOCwKLSAgMHg5OSwgMHgxNiwgMHg3NCwgMHhkMSwgMHgxNiwgMHgwZiwgMHhlMSwgMHgxMSwgMHg4YiwgMHgyOSwgMHhkMywgMHhmMCwKLSAgMHg1MywgMHg3MCwgMHhiMCwgMHgyNSwgMHg5MCwgMHhhZiwgMHg2MiwgMHg5NCwgMHhlYiwgMHg2MCwgMHhmZCwgMHg1NCwKLSAgMHhmZCwgMHgwYSwgMHhkYiwgMHhjNCwgMHhjOSwgMHhiMSwgMHhkMSwgMHgyNiwgMHgxZiwgMHgxMiwgMHgzMSwgMHhiOSwKLSAgMHgwYywgMHgyYywgMHg2ZCwgMHgwZSwgMHgzNSwgMHg4ZiwgMHg3MywgMHgyMSwgMHg0MywgMHhkMiwgMHgwOCwgMHg5MywKLSAgMHhhMywgMHg1OCwgMHg3ZiwgMHg0ZSwgMHhiMywgMHhmZiwgMHgwMCwgMHhhYSwgMHgzZiwgMHhmOSwgMHg1ZiwgMHhjMCwKLSAgMHhkMiwgMHhiYywgMHg4NywgMHgwNiwgMHgzMSwgMHhlYiwgMHgxMywgMHgzZCwgMHg1YiwgMHhiNCwgMHhjNCwgMHhkOCwKLSAgMHhmMSwgMHg1MywgMHg0MSwgMHg1ZiwgMHg2ZSwgMHhlYSwgMHg0MSwgMHg5NiwgMHg5YSwgMHg2NSwgMHgxYywgMHhhYywKLSAgMHhiNCwgMHhkYiwgMHg0OSwgMHhmNCwgMHgyMSwgMHgzYSwgMHgxNCwgMHg4OCwgMHgzMSwgMHhhZSwgMHgyNiwgMHgzOSwKLSAgMHhmOSwgMHg3ZSwgMHgyNywgMHhjNSwgMHg2ZiwgMHhmZiwgMHgwMCwgMHhjNywgMHhjNywgMHgxYSwgMHg0ZSwgMHhiZiwKLSAgMHhlNywgMHg3MSwgMHg3NSwgMHhiNSwgMHg1OCwgMHgzZSwgMHg0MSwgMHgyZCwgMHhiZSwgMHg1YiwgMHhhNywgMHgxMywKLSAgMHhiMiwgMHg5NywgMHg3ZiwgMHhmYywgMHgzNiwgMHhlMiwgMHg0NSwgMHgxZiwgMHhmNCwgMHgzMywgMHhiZiwgMHhmMSwKLSAgMHhhZSwgMHg4YSwgMHg3NCwgMHgyOSwgMHhkMywgMHhmMCwgMHg0NSwgMHgyZiwgMHg4MiwgMHgyNywgMHgwOCwgMHg3OSwKLSAgMHgyNywgMHhkNSwgMHg1MCwgMHhmOSwgMHg1NiwgMHg1MSwgMHg2MCwgMHhjNSwgMHhhMSwgMHgyMiwgMHg2NiwgMHg0MywKLSAgMHg3NSwgMHg4ZCwgMHg2ZiwgMHg2NSwgMHhjNSwgMHg3MiwgMHgzNCwgMHg1ZCwgMHg1NywgMHg5YywgMHhlMiwgMHhiNSwKLSAgMHhiZSwgMHg1NCwgMHgyNCwgMHg3NSwgMHg1MSwgMHhkNywgMHg4MCwgMHgwNCwgMHhkMiwgMHgzZCwgMHg4ZSwgMHhjYywKLSAgMHhkZCwgMHg4NywgMHg4YywgMHhmMCwgMHg2ZCwgMHg3MSwgMHgyZiwgMHg3NywgMHhlOSwgMHhlMywgMHhmOSwgMHgzZiwKLSAgMHgyNiwgMHg0YywgMHhiMSwgMHg3MSwgMHhiOSwgMHhiZCwgMHgyNywgMHg5YiwgMHg2ZiwgMHhiMCwgMHg4NiwgMHhjZiwKLSAgMHgyYSwgMHg4OSwgMHg0OCwgMHhmOCwgMHgyZSwgMHg2YiwgMHg0MCwgMHg3OCwgMHhkNCwgMHg4YywgMHgxMiwgMHgzMiwKLSAgMHg2ZSwgMHgzMSwgMHhiZiwgMHgzZCwgMHhiMCwgMHg5NSwgMHhkYiwgMHhiMSwgMHg1OCwgMHg4YiwgMHg4NCwgMHhkYiwKLSAgMHg5ZSwgMHgwNiwgMHg3YywgMHg4ZSwgMHg1NSwgMHgzYSwgMHgxMiwgMHg3YywgMHg3YiwgMHgzNiwgMHg5MiwgMHhkYSwKLSAgMHg0ZiwgMHhhMCwgMHhiYSwgMHg0NywgMHg3OCwgMHhhZCwgMHg0MSwgMHhiMSwgMHg1OSwgMHhmNSwgMHhkMiwgMHhlYSwKLSAgMHg0YSwgMHgzMSwgMHgwYywgMHgyMiwgMHhmNSwgMHg3MywgMHgwNCwgMHg4ZSwgMHg1OSwgMHg3NywgMHgwNCwgMHg3YiwKLSAgMHg5YiwgMHgxNywgMHg5NCwgMHhmZSwgMHg3NiwgMHhkZSwgMHgxZCwgMHhhYSwgMHg4NywgMHhjZCwgMHg2YywgMHhkNywKLSAgMHg5MywgMHg4YSwgMHhlNiwgMHgzOSwgMHgwMSwgMHhkZSwgMHg1ZiwgMHg5NiwgMHg5OCwgMHg5MCwgMHhkNCwgMHg3YywKLSAgMHhlYiwgMHg2NiwgMHgzZSwgMHg5NSwgMHg0NiwgMHg0MiwgMHg4NywgMHg1ZSwgMHg4YiwgMHg5MCwgMHhhZCwgMHhiYywKLSAgMHhhMCwgMHg0NywgMHg3ZiwgMHgyZiwgMHg2NywgMHhlYywgMHhhNywgMHhkMCwgMHgwMSwgMHgxZSwgMHgwNCwgMHgxZiwKLSAgMHg1NSwgMHg2NywgMHg0MiwgMHg4MCwgMHg4YSwgMHhjNiwgMHg3MSwgMHhkYiwgMHgxZSwgMHgzNywgMHg2ZCwgMHgxNiwKLSAgMHhmYiwgMHgwZCwgMHhhZSwgMHgzNSwgMHhiYSwgMHgzMCwgMHgzYywgMHhjYSwgMHg0MywgMHgyOCwgMHhkNywgMHgzYSwKLSAgMHhiYywgMHg1NCwgMHhhMywgMHhkZSwgMHhhNSwgMHgxZiwgMHgxNSwgMHgxMiwgMHg0OSwgMHhmNCwgMHhkNCwgMHhhZSwKLSAgMHhiYSwgMHg2YSwgMHg4YSwgMHgyOCwgMHgwMiwgMHg4YSwgMHgyOCwgMHhhMCwgMHgxMSwgMHgzOCwgMHhjYiwgMHhmMSwKLSAgMHgyYywgMHg1ZiwgMHhmNiwgMHhhYSwgMHhkOSwgMHhmNywgMHhlMiwgMHg5ZSwgMHhlOSwgMHgxMywgMHg4YywgMHhiZiwKLSAgMHgxMiwgMHhjNSwgMHhmZiwgMHgwMCwgMHg2YSwgMHhhZCwgMHg5ZiwgMHg3ZSwgMHgyOSwgMHhlZSwgMHg4MCwgMHhjMSwKLSAgMHhlZiwgMHhkZCwgMHgxNCwgMHgxYSwgMHgyYSwgMHg1MSwgMHg1NiwgMHhjYywgMHhkMiwgMHg2NywgMHgxYywgMHg3ZSwKLSAgMHg0NywgMHhiMiwgMHhlZiwgMHhhMiwgMHgyNCwgMHg3ZiwgMHgwMSwgMHhhNywgMHgzYSwgMHg0YywgMHhlMywgMHg4ZiwKLSAgMHhjOCwgMHhmNiwgMHg1ZCwgMHhmNCwgMHg0NCwgMHg4ZiwgMHhlMCwgMHgzNSwgMHgwNSwgMHg4NiwgMHhmOCwgMHhkZiwKLSAgMHgxNywgMHg2ZiwgMHhlNiwgMHgwZiwgMHhiMiwgMHhiNiwgMHg1NiwgMHhiOCwgMHhkZiwgMHgxNywgMHg2ZiwgMHhlNiwKLSAgMHgwZiwgMHhiMiwgMHhiNiwgMHg1MCwgMHgwNSwgMHgxNCwgMHg1NiwgMHgwOSwgMHgzYiwgMHhlZSwgMHhhMCwgMHgzMywKLSAgMHg1OCwgMHgyYSwgMHhkNywgMHg3ZCwgMHgyOCwgMHhlNywgMHgzYywgMHg0OCwgMHhjMywgMHhmMCwgMHhkNywgMHg1MSwKLSAgMHgxNiwgMHhmNSwgMHg3NywgMHg0NywgMHhiYSwgMHgyZSwgMHhmZSwgMHg0MiwgMHhkYiwgMHgxNSwgMHhiNSwgMHg0OCwKLSAgMHg5OCwgMHhmMSwgMHhmMCwgMHgwOSwgMHg2NSwgMHhiMCwgMHg1NiwgMHg3NywgMHhkMywgMHhhOSwgMHgwMCwgMHg3YSwKLSAgMHhlOSwgMHg2NCwgMHhlNCwgMHgxYywgMHg1NiwgMHhjYywgMHg1NSwgMHhjOSwgMHg4YywgMHhlMywgMHg0YywgMHhlMSwKLSAgMHgxNiwgMHhjNSwgMHg3NCwgMHhmNywgMHg0ZiwgMHgyMSwgMHg0OCwgMHg3YSwgMHg2MiwgMHg4NywgMHhmYiwgMHgzOCwKLSAgMHg4ZCwgMHhhYiwgMHg0OSwgMHgzZiwgMHhlZiwgMHg1NiwgMHgzZCwgMHg5NCwgMHgwNSwgMHg4ZCwgMHg3ZCwgMHhiZCwKLSAgMHg1YSwgMHg2YywgMHgzNiwgMHhjNywgMHg2ZSwgMHg3NywgMHhiYiwgMHg5NCwgMHg0YiwgMHg2YywgMHgyNiwgMHg0NiwKLSAgMHhkYywgMHg5MSwgMHgyOSwgMHhkNCwgMHhiNiwgMHg4NCwgMHhmZiwgMHgwMCwgMHhjNCwgMHhhMiwgMHgwNSwgMHg1NywKLSAgMHg2YiwgMHhlMiwgMHhlNCwgMHg4YiwgMHhmYSwgMHg4YiwgMHgzYywgMHgzNCwgMHhjMiwgMHhhZiwgMHgxOSwgMHg2MCwKLSAgMHgzYiwgMHhkNSwgMHhjNSwgMHhkMSwgMHhlZSwgMHg3ZCwgMHhiOCwgMHg2YiwgMHhkMCwgMHhmYiwgMHhjMywgMHg2YiwKLSAgMHhlYiwgMHhmYSwgMHgwOSwgMHg1NywgMHhiNiwgMHhiYSwgMHhhYywgMHg3YywgMHgxZCwgMHhjNywgMHhkMywgMHg3NiwKLSAgMHg2NiwgMHhmZiwgMHgwMCwgMHg5NywgMHhjZiwgMHhiOCwgMHg2NiwgMHhiNywgMHhjNiwgMHhiYSwgMHhiNywgMHgyYSwKLSAgMHhmMiwgMHhiMCwgMHhiNiwgMHg1OSwgMHgzZiwgMHhlYywgMHhhMywgMHhhNCwgMHgwNiwgMHg5YiwgMHgxZiwgMHhmMCwKLSAgMHg5MywgMHhlYiwgMHhhYiwgMHgxOSwgMHgyOSwgMHg0YSwgMHgzYywgMHhkNCwgMHg4MCwgMHgwMCwgMHgxZCwgMHgwMCwKLSAgMHgxZCwgMHgwNSwgMHgwMSwgMHg1NiwgMHgxYywgMHgzYiwgMHg4OSwgMHhiOSwgMHg2NywgMHhmYSwgMHhlYiwgMHg5ZCwKLSAgMHgzMywgMHg2MiwgMHg4MCwgMHhiMSwgMHhlNywgMHhkYiwgMHgzMSwgMHg1NiwgMHg5NCwgMHhkMiwgMHhkNCwgMHgzNSwKLSAgMHhmMCwgMHg1NywgMHgyZCwgMHhjZCwgMHhiOSwgMHhlZCwgMHhlNCwgMHg0YSwgMHgzNywgMHg0YywgMHgxOCwgMHg4ZiwKLSAgMHgwYSwgMHgzOCwgMHg3NywgMHg4YiwgMHhjOCwgMHgxMywgMHgyZCwgMHgzOCwgMHhhNSwgMHhiYywgMHg0ZSwgMHhlZiwKLSAgMHgzMywgMHg2NCwgMHhhNCwgMHhjOSwgMHg5MCwgMHg0ZiwgMHhhNCwgMHhiYSwgMHhlOSwgMHg1MiwgMHhmNywgMHhmNSwKLSAgMHhkMywgMHhhOCwgMHgwMywgMHg1NCwgMHg2ZiwgMHhjMywgMHg1NCwgMHgwNywgMHgwZCwgMHhkYSwgMHhjOSwgMHg2NywKLSAgMHhiYiwgMHhiMSwgMHhlNCwgMHhmNywgMHg2YiwgMHg1NCwgMHgxYiwgMHg4MywgMHgzYSwgMHhkNywgMHg2NywgMHgyYSwKLSAgMHgzYSwgMHgxZCwgMHg0ZSwgMHhiZCwgMHgxYSwgMHg1MCwgMHgyMiwgMHhhYSwgMHg5YywgMHhiMywgMHgwZCwgMHg3NywKLSAgMHg4NiwgMHgxMiwgMHgzZiwgMHg5NiwgMHhkYywgMHgzMSwgMHhiNywgMHgzZSwgMHhkNCwgMHgxNiwgMHgxNywgMHhjZiwKLSAgMHg3ZCwgMHhjNiwgMHhlMiwgMHgyOCwgMHhmOSwgMHgzYywgMHhkOCwgMHhmZiwgMHgwMCwgMHg5ZSwgMHhlYiwgMHgyZCwKLSAgMHgxZSwgMHg4OCwgMHg3ZCwgMHgwMywgMHhhOCwgMHhlNSwgMHhkMCwgMHg1MCwgMHgwNCwgMHg2YiwgMHg3NSwgMHg3MCwKLSAgMHhiYywgMHhmMywgMHg2YywgMHhiNCwgMHhiNywgMHg1ZCwgMHg3MSwgMHgwOCwgMHg2ZCwgMHgwMCwgMHg5NSwgMHgyOSwKLSAgMHg2YSwgMHhkMCwgMHg0OCwgMHgxZSwgMHgyNCwgMHhmOCwgMHg1NywgMHgwNSwgMHg5YSwgMHhmOSwgMHg2NSwgMHhiZCwKLSAgMHhhMSwgMHhjNywgMHhhYywgMHhkNywgMHg2OCwgMHgxNywgMHgyNCwgMHgzNCwgMHhhMCwgMHg5NywgMHgxNSwgMHgxNiwKLSAgMHg0YSwgMHgxZCwgMHg0YSwgMHgwZiwgMHhhMCwgMHhmMiwgMHg5MywgMHhhMywgMHg0MCwgMHgxOCwgMHhjZCwgMHhmMiwKLSAgMHhkNywgMHg5MSwgMHhlMywgMHhmMCwgMHg2ZiwgMHhiNiwgMHg1OSwgMHhhZCwgMHhjYywgMHhiNywgMHg0ZCwgMHg2NCwKLSAgMHgzYywgMHhjMywgMHhlOCwgMHhlZSwgMHg1MiwgMHg0ZiwgMHhkOCwgMHg3YywgMHgwOCwgMHgzZCwgMHhjNCwgMHgxYSwKLSAgMHg5MCwgMHhkOSwgMHgxYiwgMHhkOCwgMHhhYSwgMHg5MywgMHgwNywgMHhlNSwgMHhlMSwgMHhkNywgMHgxNiwgMHhhZSwKLSAgMHg1OCwgMHgzMywgMHhkZSwgMHhmMywgMHg2MCwgMHhjOSwgMHg5YywgMHg3NiwgMHhlZCwgMHg4ZiwgMHgxZSwgMHhlNCwKLSAgMHgzNSwgMHgyMywgMHhiZSwgMHg1NCwgMHg1MSwgMHhkMywgMHg0MywgMHhhZiwgMHhiZSwgMHhhNCwgMHgwZiwgMHgwNSwKLSAgMHgyYiwgMHhkMSwgMHg1NiwgMHg4NSwgMHhmMiwgMHgxYiwgMHhkNywgMHgxYiwgMHgzNCwgMHhlYiwgMHg3YywgMHg3OSwKLSAgMHg4ZSwgMHhjMiwgMHg3YSwgMHg0YywgMHg3NywgMHgxOSwgMHg0NCwgMHg5NiwgMHhjNiwgMHhkNiwgMHhjYSwgMHg5NCwKLSAgMHg5MiwgMHgwMiwgMHhjMCwgMHhmNCwgMHg4ZCwgMHhlZiwgMHhlYSwgMHhhMCwgMHgyOSwgMHhkOCwgMHg5OSwgMHgyZCwKLSAgMHhjMiwgMHhlNSwgMHg5NSwgMHhlNCwgMHgxNywgMHgxYywgMHg2YywgMHhiNywgMHgyYSwgMHhmYiwgMHg5MCwgMHgzYSwKLSAgMHg5YiwgMHg2ZCwgMHg4OCwgMHgyYywgMHgwNSwgMHhiNywgMHgwZSwgMHhkZiwgMHgxNCwgMHhhZCwgMHgwZSwgMHg0ZSwKLSAgMHg3MCwgMHg4ZSwgMHhlNiwgMHg4YiwgMHhjYSwgMHg3OCwgMHhhNCwgMHg3ZiwgMHg0OCwgMHg0MiwgMHgxMiwgMHgzNywKLSAgMHhiZCwgMHg4YiwgMHg1ZiwgMHgwZCwgMHhjNywgMHg2MCwgMHhlMywgMHgxOCwgMHhlNCwgMHg1YiwgMHgzNCwgMHgxMiwKLSAgMHhiNSwgMHhhMSwgMHg5MCwgMHg1NCwgMHhlYiwgMHhjZSwgMHgxZCwgMHhiOCwgMHhmYiwgMHhhYSwgMHgzYywgMHhjZSwKLSAgMHgzYSwgMHhiMywgMHhlMiwgMHhiNSwgMHgyOCwgMHg5NSwgMHgxMywgMHhlOSwgMHgzZSwgMHg4ZCwgMHgwYSwgMHg4OSwKLSAgMHhlMSwgMHg2ZSwgMHgwMSwgMHg2OSwgMHhjMCwgMHhhYywgMHgyOSwgMHg4MywgMHgwZCwgMHhjNywgMHgyNiwgMHg0ZCwKLSAgMHg3NSwgMHgwZCwgMHg4OSwgMHg5MywgMHhkZSwgMHhmYywgMHhhNCwgMHg4MiwgMHg4NCwgMHhmMiwgMHhhMCwgMHg3YSwKLSAgMHgxMiwgMHg4NCwgMHg4ZSwgMHg4OSwgMHg0MCwgMHhlOCwgMHgzYSwgMHg5ZSwgMHhhNCwgMHg5MiwgMHg1YiwgMHhjNywKLSAgMHg0MSwgMHg0MCwgMHgxNCwgMHg1MSwgMHg0NSwgMHgwMCwgMHg1MSwgMHg0NSwgMHgxNCwgMHgwMSwgMHg0NSwgMHgxNCwKLSAgMHg1MCwgMHgwOCwgMHg5YywgMHg2NSwgMHhmOCwgMHg5NiwgMHgyZiwgMHhmYiwgMHg1NSwgMHg2YywgMHhmYiwgMHhmMSwKLSAgMHg0ZiwgMHg3NCwgMHg4OSwgMHhjNiwgMHg1ZiwgMHg4OSwgMHg2MiwgMHhmZiwgMHgwMCwgMHhiNSwgMHg1NiwgMHhjZiwKLSAgMHhiZiwgMHgxNCwgMHhmNywgMHg0MCwgMHg3OSwgMHg1MSwgMHhlYiwgMHg0NSwgMHg2NCwgMHg4ZCwgMHg5ZSwgMHhmMywKLSAgMHg0NSwgMHgwYSwgMHhiNCwgMHhmMiwgMHg2NiwgMHg5MywgMHgzOCwgMHhlMywgMHhmMiwgMHgzZCwgMHg5NywgMHg3ZCwKLSAgMHgxMSwgMHgyMywgMHhmOCwgMHgwZCwgMHgzOSwgMHhkMiwgMHg2NywgMHgxYywgMHg3ZSwgMHg0NywgMHhiMiwgMHhlZiwKLSAgMHhhMiwgMHgyNCwgMHg3ZiwgMHgwMSwgMHhhMSwgMHg2MSwgMHhiZSwgMHgzNywgMHhjNSwgMHhkYiwgMHhmOSwgMHg4MywKLSAgMHhlYywgMHhhZiwgMHg2NCwgMHhmYiwgMHgyYSwgMHgyZSwgMHhmNSwgMHg3YiwgMHhiNSwgMHhlMywgMHhiOCwgMHhlYiwKLSAgMHhkNywgMHg5YiwgMHhkNCwgMHhlNiwgMHg2MCwgMHhkYiwgMHhlMiwgMHgzMiwgMHgxYywgMHg3ZCwgMHhmNywgMHg0ZSwKLSAgMHg5MiwgMHg4MSwgMHhkMCwgMHgwZiwgMHg1OSwgMHgyNCwgMHg5MCwgMHgwMCwgMHgxZCwgMHg0OSwgMHgyMCwgMHgwZCwKLSAgMHg5MywgMHg1NSwgMHhlYSwgMHhhNiwgMHg3MSwgMHgwMywgMHg4OSwgMHg0MCwgMHhhMiwgMHhkNiwgMHg2NCwgMHg2MCwKLSAgMHg5OCwgMHhhYSwgMHhmNSwgMHhiOSwgMHg2ZiwgMHgzNywgMHhmZSwgMHg5OCwgMHg5OCwgMHg4ZiwgMHgxZSwgMHhjZCwKLSAgMHhiMywgMHhkMiwgMHgzMiwgMHg0ZiwgMHg4MiwgMHg5NSwgMHhiNSwgMHhlYiwgMHg0NCwgMHgwNCwgMHhlZSwgMHg4MCwKLSAgMHg2MCwgMHhjZSwgMHhiOCwgMHg5ZCwgMHg4ZCwgMHg2MiwgMHg5MywgMHg1YiwgMHhiNSwgMHgyOCwgMHhjYSwgMHhiYywKLSAgMHhkZiwgMHg5ZSwgMHhkOCwgMHg2MiwgMHhjYiwgMHg2OSwgMHg2YiwgMHhjYSwgMHg2NiwgMHhiOSwgMHhlYiwgMHgyOCwKLSAgMHgwNywgMHhjYywgMHg0ZiwgMHhlYiwgMHgyYywgMHhhNCwgMHg3YSwgMHhmYSwgMHg1NCwgMHgwZiwgMHhiOSwgMHg1YywKLSAgMHg1YiwgMHhjZCwgMHgwMCwgMHg1NSwgMHhlYSwgMHhmMSwgMHgxMywgMHgwMSwgMHhiNCwgMHgzYSwgMHgzNiwgMHg2MSwKLSAgMHg1YSwgMHgwOCwgMHg5MywgMHg3MiwgMHg1MiwgMHg0ZiwgMHhlNiwgMHhiOSwgMHgyMSwgMHg2MywgMHg5MSwgMHhhNSwKLSAgMHg2YiwgMHhjNSwgMHhiNCwgMHhhOCwgMHg4ZCwgMHhmYywgMHgyYSwgMHg3MywgMHhjMSwgMHhmMCwgMHhhYywgMHg1ZiwKLSAgMHgwYywgMHhiNywgMHgxOCwgMHg5OCwgMHhlNSwgMHhhMSwgMHg4OCwgMHg3ZCwgMHhhNywgMHg1NywgMHhkZiwgMHhkNywKLSAgMHgzMywgMHhmMiwgMHgxNSwgMHhiZCwgMHg5NSwgMHgzYSwgMHhlOSwgMHhkYSwgMHhkYywgMHg1NiwgMHhjOSwgMHhlYSwKLSAgMHhhMiwgMHg2OSwgMHg4NywgMHg0MiwgMHg4MCwgMHg1NiwgMHhjMSwgMHhiMCwgMHgwYywgMHg0ZiwgMHgwYywgMHg2OSwKLSAgMHhjMywgMHg2MCwgMHhiNCwgMHgzMiwgMHhjYywgMHhiNywgMHhiNiwgMHg2NCwgMHg0ZCwgMHg3NCwgMHg5NywgMHg2NSwKLSAgMHg0OCwgMHg1MSwgMHhlYSwgMHg0YiwgMHg4ZiwgMHgyYiwgMHg2YiwgMHg1NiwgMHhjZiwgMHhhNCwgMHhlYiwgMHhkMCwKLSAgMHgwNSwgMHg3NywgMHhlNSwgMHhmOSwgMHg1NiwgMHgzYiwgMHg4OCwgMHhkYSwgMHhjZCwgMHhjZiwgMHgyNCwgMHhiYywKLSAgMHg0NCwgMHhiNiwgMHg0NSwgMHhkZiwgMHgyYSwgMHg1NiwgMHhmMiwgMHhmNCwgMHhhNywgMHgxNSwgMHhmYSwgMHgyOCwKLSAgMHg0OCwgMHhmMywgMHg5NiwgMHhhZiwgMHhkNSwgMHg0OCwgMHgyNCwgMHhmYSwgMHgyYiwgMHhiMiwgMHhmMCwgMHhkZCwKLSAgMHhjZCwgMHhkNCwgMHg0NywgMHg0ZCwgMHhhZSwgMHg1NCwgMHg3OCwgMHhlYSwgMHgxMiwgMHg1YiwgMHg1MywgMHhlNSwKLSAgMHhlNiwgMHg0YiwgMHg5YywgMHhjYywgMHg4MywgMHhlNywgMHhhNSwgMHgzYSwgMHg1MCwgMHhkMiwgMHg4OCwgMHhlZSwKLSAgMHgyNywgMHg2MCwgMHg3YSwgMHgwZCwgMHg3YiwgMHg5MywgMHg2ZiwgMHg4MSwgMHgyZSwgMHg1YywgMHg2OSwgMHg3MiwKLSAgMHhhMCwgMHhjNiwgMHg3YSwgMHg0YywgMHg1NSwgMHgxNSwgMHg0NywgMHg3NSwgMHhjNiwgMHg5MiwgMHhhNSwgMHhiMiwKLSAgMHg0ZiwgMHg0MiwgMHg1MiwgMHg0OCwgMHhkYSwgMHg3NywgMHhlYSwgMHhhMCwgMHgxMiwgMHg3MSwgMHg5YywgMHhkZiwKLSAgMHgyNywgMHhjYiwgMHgyZiwgMHhkMSwgMHg5NSwgMHg2MywgMHhjMywgMHgyNiwgMHg1YiwgMHhmMSwgMHg5NCwgMHhhOCwKLSAgMHg5OSwgMHgxNywgMHg0YiwgMHhkZSwgMHhlMywgMHgzYiwgMHgyMSwgMHgzYSwgMHhlOCwgMHgyMywgMHhjNywgMHhmOCwKLSAgMHg3ZCwgMHg0OSwgMHgwNywgMHg5OSwgMHhjZSwgMHg1MSwgMHhhZiwgMHgwYSwgMHg3MSwgMHg5YiwgMHg2OSwgMHg4OSwKLSAgMHgzMiwgMHhlZCwgMHg2ZiwgMHhiYiwgMHgzYywgMHg1ZiwgMHgxMiwgMHg2MCwgMHgwNywgMHg3YiwgMHgwMCwgMHg4NywKLSAgMHg5NiwgMHg5NCwgMHg3YiwgMHhlMiwgMHg0MiwgMHg1NCwgMHg1NCwgMHg5MCwgMHg3NCwgMHhhZSwgMHg4MywgMHhhNywKLSAgMHgzMCwgMHgzYSwgMHhlYiwgMHg1MCwgMHgzOSwgMHg3ZiwgMHgxMSwgMHg3MSwgMHgxYywgMHg1ZSwgMHg0ZiwgMHg5MCwKLSAgMHg0ZSwgMHhiOSwgMHhmOSwgMHg0ZCwgMHhkNSwgMHg2MywgMHg2ZCwgMHhkYiwgMHgyMCwgMHhiNiwgMHg2NCwgMHhjYywKLSAgMHg3MCwgMHhlYiwgMHg2MywgMHg0ZCwgMHgyMywgMHg2YSwgMHgwMywgMHhmNSwgMHg5NSwgMHhhNCwgMHhmYSwgMHhlOSwKLSAgMHgzMiwgMHhlMywgMHg5NiwgMHg3MSwgMHgxMywgMHgyNiwgMHhkYSwgMHgyZCwgMHhiMSwgMHhhMywgMHg2MSwgMHg1NiwKLSAgMHhlNSwgMHg3ZiwgMHg0ZCwgMHgyNCwgMHgyMiwgMHg2NSwgMHhjNSwgMHg0MywgMHhkNCwgMHhkOCwgMHgzZCwgMHg5MywKLSAgMHg0NywgMHhlNywgMHgxNywgMHgwOCwgMHhmNCwgMHg3OCwgMHg1NSwgMHhlMSwgMHg0ZSwgMHg1MywgMHhmMCwgMHhhMiwKLSAgMHg5MywgMHhhOSwgMHgxOCwgMHg3OCwgMHg5OSwgMHg2NywgMHg2NCwgMHg3OSwgMHgwNSwgMHg4ZiwgMHgxYSwgMHhiNSwKLSAgMHhhZSwgMHhlNSwgMHg3ZiwgMHhiYSwgMHhjMiwgMHhiNSwgMHg0MiwgMHg2ZiwgMHhiZCwgMHhlOSwgMHg2ZiwgMHhhNSwKLSAgMHhiNCwgMHg3YiwgMHgwMSwgMHgyNywgMHhhOSwgMHhmNSwgMHgwZSwgMHhiNSwgMHg0MCwgMHg2NiwgMHg5ZiwgMHg4NSwKLSAgMHgzNSwgMHhhZCwgMHg0ZCwgMHhiOCwgMHhkNiwgMHgwMSwgMHg2NiwgMHg1NSwgMHhkZiwgMHg1YywgMHhjOSwgMHhmNywKLSAgMHg0MiwgMHg2NiwgMHhkYSwgMHg4ZSwgMHgwOCwgMHgzYSwgMHgyNSwgMHgyOCwgMHgxZSwgMHg3YSwgMHg4NywgMHhiNywKLSAgMHg5MywgMHhlYiwgMHgxNSwgMHgzYSwgMHhjZSwgMHgwMywgMHg4OSwgMHhjYywgMHhmMiwgMHhhMSwgMHg5NCwgMHg0NCwKLSAgMHg5MywgMHg5MiwgMHhiYiwgMHgzYSwgMHgzYSwgMHhhMywgMHhjOSwgMHg5ZCwgMHg3NSwgMHg3OCwgMHhjOCwgMHg5MiwKLSAgMHg5NCwgMHgyYiwgMHhjNSwgMHhiNSwgMHgyYiwgMHhhMywgMHg1YSwgMHhlZiwgMHgwMSwgMHgwMSwgMHgzNSwgMHhmMiwKLSAgMHhiNywgMHgxMCwgMHgzMCwgMHhlYiwgMHhjZiwgMHhlMCwgMHhmZiwgMHgwMCwgMHhjNCwgMHg5OCwgMHhlZSwgMHhhNSwKLSAgMHg1ZSwgMHhlOSwgMHg2MiwgMHhkNywgMHgyNSwgMHgxNSwgMHg0MiwgMHg5OCwgMHhhNiwgMHhkMiwgMHhhMiwgMHhlMywKLSAgMHg0MCwgMHg4ZSwgMHg2MSwgMHhkNCwgMHg2OSwgMHgwZiwgMHhhMCwgMHgxMSwgMHhkNCwgMHg2OCwgMHhmNSwgMHgwNCwKLSAgMHg2YiwgMHg3ZCwgMHgzNCwgMHg5ZCwgMHgxOSwgMHg1MywgMHg2YiwgMHg1MSwgMHg0OCwgMHg1NiwgMHg1NSwgMHgxMywKLSAgMHhkMiwgMHgzNiwgMHhlNCwgMHg1NywgMHgyYywgMHhlYiwgMHg4OCwgMHg2ZSwgMHg4OSwgMHgxOSwgMHg5ZCwgMHhmZSwKLSAgMHg2YywgMHg4OCwgMHgwYiwgMHgyMSwgMHg0ZCwgMHhjMCwgMHgxZSwgMHhmMSwgMHgxMSwgMHg0MywgMHg3ZiwgMHg5YSwKLSAgMHhkMiwgMHgwMSwgMHgyZSwgMHhmYiwgMHg0MCwgMHg3NywgMHg1ZSwgMHgyNCwgMHg3NywgMHhkMywgMHg5ZSwgMHgyNiwKLSAgMHhmYywgMHhkYywgMHgzMiwgMHg3NCwgMHg1YywgMHhiNiwgMHhkNywgMHhkYSwgMHg0OSwgMHg1NSwgMHg5NSwgMHhiMCwKLSAgMHg2ZSwgMHg2ZCwgMHgzNCwgMHg5NCwgMHhhNCwgMHgzZCwgMHg2ZCwgMHhkMCwgMHg0YiwgMHg4OCwgMHgzYSwgMHg1MSwKLSAgMHgwYSwgMHgyOCwgMHgwMCwgMHgyZCwgMHgwMywgMHg5OSwgMHg0NCwgMHgxNCwgMHg3YywgMHgxNCwgMHg2ZSwgMHhiYSwKLSAgMHgyMCwgMHg0NSwgMHg4MSwgMHgyMSwgMHhiZiwgMHgyZCwgMHg2NiwgMHg0MiwgMHg2NiwgMHgyNSwgMHhkNCwgMHgyNSwKLSAgMHhlNSwgMHgzYywgMHgwOSwgMHgyOSwgMHgyOSwgMHg1MCwgMHgwNCwgMHgyOSwgMHg0NCwgMHhlOCwgMHg5ZSwgMHg4NCwKLSAgMHg2OSwgMHg0YiwgMHhlYywgMHhmZSwgMHg3OSwgMHhhNiwgMHgyYiwgMHgyNSwgMHhiOCwgMHhiOCwgMHhhNiwgMHhhNCwKLSAgMHgwNiwgMHgxMCwgMHg4OCwgMHhhMSwgMHhjNCwgMHhlZSwgMHg0MywgMHg5YywgMHg5YywgMHhiZCwgMHg3ZiwgMHg0NSwKLSAgMHg0YSwgMHgxYywgMHhiYywgMHhkZSwgMHhiNCwgMHgyNSwgMHg2NCwgMHhlZiwgMHhmMiwgMHg4NiwgMHhiNywgMHg4ZCwKLSAgMHgxNCwgMHhkNiwgMHgwZSwgMHg3OSwgMHg1NiwgMHg3OSwgMHhjOSwgMHg2YywgMHg3MSwgMHg1YiwgMHgxOSwgMHg2ZiwKLSAgMHgzZiwgMHhlMSwgMHhmYiwgMHg0ZSwgMHhkOCwgMHhhNiwgMHgzNiwgMHg5YiwgMHhhNCwgMHg2MiwgMHhkZCwgMHhkYiwKLSAgMHgxZiwgMHg5ZSwgMHhkYSwgMHhiNiwgMHgxYiwgMHg5MiwgMHg4MSwgMHhjZSwgMHhkMiwgMHhjMiwgMHhiZiwgMHg0MSwKLSAgMHg0MCwgMHhmMiwgMHg5ZiwgMHg0YSwgMHg1NiwgMHg2YSwgMHg1MywgMHg4NSwgMHg5OSwgMHg2YiwgMHgxOSwgMHhjZSwKLSAgMHgwNSwgMHg2YiwgMHhjOSwgMHgxYSwgMHg2YiwgMHhiMCwgMHg3MiwgMHg0YiwgMHg1YywgMHhiMiwgMHhhMywgMHg5ZSwKLSAgMHhmNiwgMHgyNCwgMHgyMCwgMHg5NCwgMHgzYSwgMHhkOSwgMHgxZiwgMHhhYSwgMHhiNCwgMHhhOCwgMHg3NSwgMHhmMCwKLSAgMHhkMSwgMHhmMSwgMHhhNCwgMHhlZSwgMHgwOSwgMHg1YywgMHgwNiwgMHgzZiwgMHg3MywgMHg5YiwgMHhjMywgMHg0OSwKLSAgMHg0ZSwgMHgyOCwgMHhiMSwgMHgxNSwgMHgwNiwgMHg3NSwgMHg4NSwgMHg2YiwgMHgzZCwgMHg1YywgMHg4NCwgMHhhNSwKLSAgMHg3OSwgMHhjYywgMHhlZiwgMHhjNCwgMHhiMiwgMHhlMSwgMHhkNywgMHg3ZiwgMHhjMCwgMHg1MiwgMHgzZCwgMHgxNSwKLSAgMHhhYSwgMHgxMiwgMHg4NywgMHgwZCwgMHg3OCwgMHhkOCwgMHhmYywgMHgyNywgMHgxMywgMHhkOSwgMHhlMywgMHgxOSwKLSAgMHhlNCwgMHg4ZSwgMHhkYSwgMHgyYSwgMHg4NywgMHhjMCwgMHg4YiwgMHg3NiwgMHgwOCwgMHhmNywgMHhjNCwgMHgxZiwKLSAgMHg0MCwgMHg3ZCwgMHgyOSwgMHhlNiwgMHgxZiwgMHhhYywgMHg5MiwgMHgzYywgMHg2YiwgMHg4ZSwgMHg1MSwgMHg3MSwKLSAgMHg3OCwgMHg2NywgMHg2NCwgMHg2NCwgMHhhNCwgMHhiMiwgMHg4YiwgMHg3YywgMHg3NCwgMHgxNCwgMHg1NiwgMHgxMiwKLSAgMHg3NiwgMHg5MCwgMHg2YiwgMHgzNSwgMHg1MiwgMHhjMSwgMHg0NSwgMHgxNCwgMHg1MCwgMHgwNSwgMHgxNCwgMHg1MSwKLSAgMHg0MCwgMHgxZCwgMHg2OCwgMHhhMiwgMHg4YSwgMHgwMSwgMHgxMywgMHg4YywgMHhiZiwgMHgxMiwgMHhjNSwgMHhmZiwKLSAgMHgwMCwgMHg2YSwgMHhhZCwgMHg5ZiwgMHg3ZSwgMHgyOSwgMHhlZSwgMHg5MSwgMHgzOCwgMHhjYiwgMHhmMSwgMHgyYywKLSAgMHg1ZiwgMHhmNiwgMHhhYSwgMHhkOSwgMHhmNywgMHhlMiwgMHg5ZSwgMHhlOCwgMHgwMiwgMHg4YSwgMHhmMiwgMHhhMiwKLSAgMHg0MSwgMHhlOCwgMHgwNywgMHhkNiwgMHg3NSwgMHg0NSwgMHgwMSwgMHhlYSwgMHg5MywgMHgzOCwgMHhlMywgMHhiZiwKLSAgMHhjNCwgMHhmZSwgMHg1YywgMHgzZiwgMHhmNCwgMHg4OSwgMHgxYSwgMHhmZiwgMHgwMCwgMHg5MCwgMHhkMywgMHg5ZCwKLSAgMHgyNiwgMHg3MSwgMHhjNywgMHhlNCwgMHg3YiwgMHgyZSwgMHhmYSwgMHgyMiwgMHg0NywgMHhmMCwgMHgxYSwgMHgwMiwKLSAgMHhiZSwgMHg5ZiwgMHg3ZiwgMHg2MywgMHgyNywgMHhlMiwgMHg1MiwgMHhmZCwgMHhkMiwgMHg2MSwgMHg0ZiwgMHhkYiwKLSAgMHhhYywgMHgzNywgMHgzNywgMHgyZCwgMHhiNiwgMHg3OCwgMHhhYiwgMHhkNywgMHg2NiwgMHg2NywgMHgzMCwgMHhjYSwKLSAgMHg1YywgMHg3ZSwgMHg2MywgMHhhOSwgMHgzZCwgMHhlNSwgMHgxYywgMHhlOCwgMHg0MywgMHg2MCwgMHhlZiwgMHg0NCwKLSAgMHg5NSwgMHgwZSwgMHhhNywgMHhhNSwgMHhkYiwgMHgwNSwgMHg5NiwgMHhkOSwgMHg4YSwgMHg4NCwgMHhhMCwgMHhhOCwKLSAgMHg4MiwgMHgzNiwgMHg1NCwgMHg3YiwgMHhkNCwgMHg0ZiwgMHg3OSwgMHgzZSwgMHhiMywgMHg1ZiwgMHgzOCwgMHg3MSwKLSAgMHgzNiwgMHgxYywgMHhjYywgMHgzZiwgMHg4ZCwgMHhiNiwgMHhmOSwgMHg2OSwgMHg5OCwgMHhkNCwgMHg3YiwgMHgzZSwKLSAgMHg1MCwgMHhmYSwgMHg2ZSwgMHgyYywgMHhhZCwgMHhmMCwgMHg3YiwgMHgzNiwgMHg2NywgMHgzMiwgMHhjZiwgMHg2NSwKLSAgMHgyNSwgMHhhZSwgMHg4MCwgMHhlYiwgMHhiNiwgMHg2MCwgMHg4MiwgMHgwZSwgMHhiYSwgMHhiOCwgMHhkYSwgMHg0NywKLSAgMHhhNiwgMHhhZiwgMHg1YywgMHgxNiwgMHhmMCwgMHhjNSwgMHhkZiwgMHgxZiwgMHg4YywgMHhmMywgMHgwZiwgMHgwNywKLSAgMHg1MSwgMHhkOSwgMHgyNSwgMHg2ZCwgMHhhYiwgMHg0NCwgMHgxNSwgMHgzNiwgMHhhMSwgMHhiNCwgMHhhYiwgMHg0NywKLSAgMHhhZiwgMHg3MSwgMHhmMSwgMHhlYiwgMHg1YSwgMHhlMywgMHgzNCwgMHhmMiwgMHhiYSwgMHgzMywgMHgyNywgMHgyYywKLSAgMHg1NCwgMHhjMywgMHhlYSwgMHg4NiwgMHgwMiwgMHg3NSwgMHhlMSwgMHhkMiwgMHhiYywgMHhiOCwgMHhlMiwgMHg1YiwKLSAgMHg2ZCwgMHg0YiwgMHg3MSwgMHg0OSwgMHg0MiwgMHg1MiwgMHgzNiwgMHhhMiwgMHg0ZSwgMHg4MCwgMHgxZSwgMHg5MywKLSAgMHg1NCwgMHgxZiwgMHgxYywgMHg3ZiwgMHgwOCwgMHg4NSwgMHg2MSwgMHhiOSwgMHgxNCwgMHg5YywgMHg0ZiwgMHgxNCwKLSAgMHhjNSwgMHg5ZiwgMHhiZSwgMHg1ZSwgMHg1OSwgMHgyMSwgMHhiNywgMHg2NCwgMHgzYywgMHg0YSwgMHgyMiwgMHhiNCwKLSAgMHhiMiwgMHg5MCwgMHhhZSwgMHg1ZiwgMHgzNywgMHg2YSwgMHg3MCwgMHg4MCwgMHg0NiwgMHhjMCwgMHhlNSwgMHgwMywKLSAgMHg3YSwgMHhkZiwgMHg3ZCwgMHg3YywgMHg3NywgMHhjNiwgMHhlZSwgMHgyOSwgMHg3MSwgMHg0NywgMHgyOSwgMHhiYiwKLSAgMHg0OCwgMHhiNCwgMHhlNiwgMHgxNywgMHhmOSwgMHgwOSwgMHg4ZSwgMHg5ZCwgMHgxZiwgMHgyMCwgMHg4YiwgMHhlZiwKLSAgMHg1MSwgMHg4YSwgMHg0OCwgMHhkOCwgMHhmMywgMHg1MiwgMHg3YywgMHhlZSwgMHg4NCwgMHg3YywgMHgyMiwgMHhhMiwKLSAgMHgzYiwgMHhhYiwgMHgzNCwgMHg4ZCwgMHgzMiwgMHg4ZiwgMHhiZSwgMHhhZiwgMHgzYywgMHg2MCwgMHhjNywgMHgzYywKLSAgMHhiNiwgMHg0NSwgMHhhZiwgMHgxMSwgMHg2MiwgMHg0NiwgMHg2MywgMHg3NSwgMHg2MCwgMHhmMiwgMHhiYywgMHhkNSwKLSAgMHhhZCwgMHg0OSwgMHgzMSwgMHhkOCwgMHg1NywgMHg4NywgMHg2YiwgMHgyMSwgMHg0NywgMHhiMywgMHg0NywgMHg3MSwKLSAgMHhlOCwgMHgwYSwgMHg5NSwgMHhlYSwgMHhhNSwgMHg2YiwgMHhhMywgMHhiOSwgMHhiNiwgMHg0NCwgMHhkYiwgMHhiMiwKLSAgMHgzMiwgMHhlYywgMHg5NSwgMHgxOCwgMHhmZCwgMHhhZiwgMHhiYywgMHhkYiwgMHhhYywgMHg4ZSwgMHg5NiwgMHg3YywKLSAgMHhjZiwgMHg0MywgMHhkMiwgMHhkNCwgMHgwMywgMHg4NywgMHhiZiwgMHg0NCwgMHgzNiwgMHgxMCwgMHgzZCwgMHhiNSwKLSAgMHhmMywgMHg3NywgMHhlMCwgMHgyOSwgMHg3NCwgMHg0YiwgMHg3OSwgMHgxNiwgMHg0YiwgMHg2NiwgMHg1YiwgMHg4OSwKLSAgMHhkYywgMHg4OCwgMHg4ZCwgMHg0OSwgMHg0MiwgMHgwOSwgMHhlOCwgMHg3YiwgMHgzNSwgMHg5NCwgMHg5ZSwgMHg5ZiwKLSAgMHhmZiwgMHgwMCwgMHgyMCwgMHhmZSwgMHhlYSwgMHhmYSwgMHgxYiwgMHg4OSwgMHgzOSwgMHgyNiwgMHgzZiwgMHg4OSwKLSAgMHhkYiwgMHg1YiwgMHhiYywgMHhkZSwgMHhkMiwgMHhlYywgMHhiNywgMHhiOSwgMHhiYiwgMHgyYiwgMHg3YywgMHgwNiwKLSAgMHg5MywgMHhkYSwgMHgzYiwgMHgyNSwgMHhlZiwgMHgwMCwgMHhkYiwgMHg3ZSwgMHgyYSwgMHhmZCwgMHg2MywgMHhkZCwKLSAgMHhlYSwgMHhmMSwgMHhlZSwgMHhhMSwgMHg0NiwgMHgxYSwgMHgzNSwgMHhiMywgMHg4YSwgMHhiZCwgMHg1OSwgMHhlYiwKLSAgMHhkMCwgMHg4OSwgMHgxYywgMHg0MiwgMHhkZCwgMHg4ZSwgMHhkYiwgMHg2MSwgMHgyOSwgMHhhYywgMHg3MiwgMHhkYywKLSAgMHhjYywgMHg0NiwgMHgxNSwgMHhkNSwgMHg2YiwgMHg2ZCwgMHg4MiwgMHg5MiwgMHhmMSwgMHhmZCwgMHgyNSwgMHgyYywKLSAgMHg4ZCwgMHhiOCwgMHg3NywgMHhlMiwgMHg0OSwgMHhmNiwgMHhkNCwgMHhkOCwgMHhlZiwgMHhkZiwgMHhkNSwgMHhlYiwKLSAgMHhmZiwgMHgwMCwgMHhkZiwgMHhmZSwgMHhmYSwgMHhkNSwgMHgyOCwgMHhkZSwgMHg0NywgMHhjNywgMHg0YiwgMHhjNSwKLSAgMHhkMSwgMHgwYiwgMHg2ZSwgMHgwZSwgMHgyZiwgMHg4YywgMHgzMiwgMHhlOCwgMHhlNywgMHg4ZiwgMHg2ZCwgMHg5OCwKLSAgMHhkYiwgMHhiMywgMHgyNiwgMHgyZCwgMHgxZSwgMHgxYywgMHhjOCwgMHg2OCwgMHgyOSwgMHg0MywgMHhmYiwgMHg5MywKLSAgMHg1NiwgMHg3NCwgMHgxYiwgMHhlMiwgMHhkOSwgMHg4NywgMHgxZSwgMHgzZCwgMHhkMSwgMHhiNywgMHg1YywgMHhiZCwKLSAgMHhhNSwgMHgwOSwgMHgxMiwgMHgyMiwgMHhjNywgMHg4NiwgMHhlYiwgMHg2YSwgMHhlNywgMHgwMCwgMHgxNSwgMHgyOCwKLSAgMHgyMSwgMHhjZCwgMHgyOSwgMHgwZCwgMHhmNSwgMHhkOCwgMHg1MiwgMHhiNCwgMHgzYywgMHgzNywgMHhiYSwgMHhlYSwKLSAgMHg4NSwgMHg1OCwgMHhiZCwgMHg4ZSwgMHg0OSwgMHhkMywgMHg5NywgMHgyYywgMHg1MiwgMHhjZSwgMHhmOCwgMHg4MywKLSAgMHg3MywgMHg0ZCwgMHhlOSwgMHg3OCwgMHhhNiwgMHgwNSwgMHg2ZiwgMHg2NiwgMHhlNywgMHg3YiwgMHg2ZCwgMHhjMCwKLSAgMHhkYywgMHhiOSwgMHg5MiwgMHgxMiwgMHg3YywgMHg5MiwgMHgwYSwgMHg5NCwgMHgzNiwgMHgxMiwgMHg3OSwgMHg3YywKLSAgMHhlNywgMHg1ZCwgMHhkNywgMHg1ZSwgMHhjZCwgMHgyMCwgMHg5MSwgMHhlMywgMHhkYywgMHg0NSwgMHg0MiwgMHg3MSwKLSAgMHgxZiwgMHgwYywgMHhjZCwgMHhlZSwgMHhkYywgMHgxMCwgMHhjOSwgMHgyMywgMHg2NiwgMHhmNywgMHgwYiwgMHhjZCwKLSAgMHhkZSwgMHg0MCwgMHg2OSwgMHgxMiwgMHhlZCwgMHhlNSwgMHg1NiwgMHhiNiwgMHgyMywgMHhjNSwgMHg4NiwgMHhmMywKLSAgMHg2NywgMHg5OCwgMHg5NCwgMHg4NCwgMHg5MiwgMHhmMCwgMHgwYSwgMHg0ZiwgMHgzMiwgMHgzNiwgMHhhMCwgMHgwNywKLSAgMHg5ZCwgMHhiMiwgMHgwNywgMHg3OCwgMHhiNSwgMHhiOCwgMHg2NywgMHg2MiwgMHhmMiwgMHgwYywgMHg0YSwgMHhmNiwKLSAgMHhlZSwgMHgyYiwgMHg2YiwgMHg4OSwgMHgwMywgMHgyMywgMHg1ZCwgMHhjNSwgMHhlNiwgMHg4MiwgMHhhNywgMHgyOCwKLSAgMHgyYywgMHg0NywgMHgwYiwgMHg1OCwgMHg1NywgMHgzYSwgMHhiNCwgMHg0ZiwgMHgzNiwgMHhkMiwgMHhiMCwgMHhiMywKLSAgMHhhMiwgMHg3OSwgMHg5NSwgMHhhMCwgMHg0ZSwgMHg4MCwgMHhkNCwgMHg4ZSwgMHgyZiwgMHg4ZCwgMHhkYywgMHg3MSwKLSAgMHhiYywgMHgyMiwgMHhmNSwgMHg3NCwgMHhjZiwgMHg2ZSwgMHhmZSwgMHhlOCwgMHhkZCwgMHg1ZCwgMHg4YywgMHhmZiwKLSAgMHgwMCwgMHg5NiwgMHg0OCwgMHgxMiwgMHhkZSwgMHg3MSwgMHg5MiwgMHhjOSwgMHg0ZiwgMHg0MSwgMHhkOSwgMHhhOCwKLSAgMHg4NCwgMHgwMywgMHhkMCwgMHg5ZiwgMHgzNSwgMHgwOSwgMHhkNywgMHgzNywgMHgyOCwgMHhmNCwgMHg5ZiwgMHgzYSwKLSAgMHhiZCwgMHg2OSwgMHg0ZCwgMHhiNCwgMHhjZiwgMHg0NiwgMHg4NSwgMHgxOCwgMHhjMSwgMHgyNiwgMHg4YSwgMHg0MywKLSAgMHhmMCwgMHg2NiwgMHg0YSwgMHhhZSwgMHgzYywgMHgyNiwgMHhiMiwgMHhjNiwgMHg5MCwgMHgxYiwgMHg2NiwgMHhlYywKLSAgMHhkMCwgMHg3MSwgMHg3MSwgMHgxNSwgMHgyOSwgMHhiNSwgMHgzYSwgMHgwNCwgMHg2MCwgMHhlOSwgMHg0OCwgMHg3NSwKLSAgMHhhNCwgMHgyOCwgMHg4NCwgMHg5MiwgMHgwZSwgMHhkMiwgMHg0OCwgMHhlZSwgMHgyOSwgMHhlYiwgMHhkNywgMHhhNSwKLSAgMHg1OSwgMHg0YywgMHhkYiwgMHgyZSwgMHgyYiwgMHg5YywgMHg1ZSwgMHg0YiwgMHg0OSwgMHgwZSwgMHgyMSwgMHg1YSwKLSAgMHgzMywgMHhhZSwgMHgwYiwgMHgxMiwgMHgxZSwgMHgyMywgMHg3ZiwgMHhkMSwgMHhiNiwgMHg5ZiwgMHgzMSwgMHgwMywKLSAgMHhiZiwgMHhkMSwgMHhmMywgMHg2OSwgMHg2MiwgMHhkZCwgMHg2NSwgMHg1ZSwgMHgxNSwgMHg4NywgMHgxOCwgMHgzMiwKLSAgMHgxNiwgMHg5NiwgMHhkZCwgMHhjNywgMHg2ZSwgMHg5MSwgMHhlZSwgMHhiMSwgMHg0NywgMHgyZiwgMHg5YywgMHhjYywKLSAgMHgxOSwgMHhlYSwgMHg0YSwgMHgxZiwgMHg2YiwgMHhlNiwgMHg4NSwgMHhhZCwgMHhjZCwgMHg4MSwgMHhkMywgMHg2ZCwKLSAgMHg4MywgMHhlMSwgMHg1NiwgMHgzYSwgMHgxOCwgMHg2ZCwgMHhiNywgMHg5ZCwgMHg3MSwgMHgzYywgMHhkYywgMHhlZSwKLSAgMHgxZiwgMHgzYywgMHg5NSwgMHgxMywgMHhkZCwgMHhkMywgMHhhMCwgMHgzZCwgMHgwMCwgMHhlOSwgMHhkYywgMHgzNCwKLSAgMHgyYiwgMHhiNiwgMHhkZCwgMHhlYiwgMHg4ZiwgMHhjMCwgMHhlMywgMHhiOCwgMHg1YSwgMHg2NSwgMHhiNywgMHg1MiwKLSAgMHgwNywgMHgzNywgMHhiNywgMHhkYywgMHg5YywgMHg4OSwgMHgwYSwgMHhmOSwgMHg2MSwgMHhkYSwgMHhhZiwgMHhmNiwKLSAgMHgyNywgMHhiYywgMHhiNiwgMHgwNywgMHg1ZCwgMHg3NiwgMHhlNCwgMHgyNywgMHg0ZSwgMHhiMCwgMHhhZiwgMHhkNSwKLSAgMHg3NSwgMHhiZSwgMHg2NCwgMHg2YiwgMHhhNiwgMHg4ZiwgMHgyOSwgMHhmMCwgMHhhNywgMHgzYywgMHg5YSwgMHhkZCwKLSAgMHg2NywgMHhlMiwgMHhmZiwgMHgwMCwgMHgwOCwgMHgzNSwgMHgwNiwgMHg0OSwgMHg0MywgMHg1NywgMHgzOCwgMHhjOCwKLSAgMHg5NywgMHg2ZSwgMHg5NywgMHhkYywgMHhlNCwgMHg0OSwgMHgyOSwgMHgzYywgMHhjZCwgMHgyZiwgMHhkMiwgMHg5NSwKLSAgMHgyMSwgMHhjNCwgMHg4MCwgMHg0NywgMHhhOSwgMHg0MiwgMHhhMywgMHg3YywgMHg3ZiwgMHhmZCwgMHhkNCwgMHg3NywKLSAgMHgwZCwgMHhhNywgMHg3ZiwgMHgyNCwgMHg3OCwgMHg4MywgMHgyZiwgMHgxNiwgMHg3OCwgMHhmMiwgMHhkOSwgMHhmMiwKLSAgMHg0NSwgMHhiOSwgMHgzZSwgMHhkNCwgMHg0ZiwgMHg0NCwgMHhiMywgMHgzMCwgMHgwZSwgMHg2OSwgMHgyYywgMHgwZiwKLSAgMHg0MCwgMHg1OCwgMHhmNywgMHhkNCwgMHg4ZSwgMHg5ZCwgMHg3YiwgMHg1YSwgMHhjYSwgMHhmMiwgMHg5NiwgMHg3YiwKLSAgMHhlOCwgMHhkMiwgMHhjZSwgMHhhNiwgMHgzYiwgMHg4YywgMHg2YiwgMHhlMCwgMHhjZSwgMHg2MiwgMHhmZSwgMHg1ZiwKLSAgMHg4NiwgMHgyMSwgMHhlYiwgMHhhMiwgMHgxMSwgMHgxZSwgMHhmZiwgMHgwMCwgMHg2ZCwgMHg3NSwgMHg3NiwgMHhmYiwKLSAgMHhkYywgMHg2ZCwgMHhlOCwgMHhiMSwgMHgzMSwgMHhhMywgMHhjYSwgMHhlNywgMHg0MSwgMHhmOSwgMHhhYSwgMHhlOCwKLSAgMHhiNCwgMHhmYSwgMHg0MiwgMHg4NSwgMHgzYiwgMHg4ZSwgMHhlYSwgMHhhNywgMHhmMywgMHgzNCwgMHhhMywgMHg4NiwKLSAgMHhkYywgMHg1ZCwgMHg4MywgMHg5ZSwgMHgzNSwgMHhlZiwgMHg1OCwgMHhmNiwgMHg1MCwgMHhiNiwgMHhhZCwgMHg1OSwKLSAgMHgxMCwgMHgxZCwgMHgxMCwgMHhjNCwgMHhhZSwgMHhlOCwgMHg5MiwgMHhjOCwgMHhkZSwgMHg4NiwgMHhmZiwgMHgwMCwKLSAgMHgyNCwgMHhhMywgMHhkYywgMHgwMSwgMHg0OSwgMHhhYiwgMHg4MCwgMHg3NywgMHg3NywgMHg2YSwgMHhiYywgMHhmMywKLSAgMHhkMCwgMHgwYSwgMHgyOCwgMHhhMiwgMHg4MCwgMHgyOCwgMHhhMiwgMHg4YSwgMHgwMCwgMHhhMiwgMHg4YSwgMHgyOCwKLSAgMHgwNCwgMHg0ZSwgMHgzMiwgMHhmYywgMHg0YiwgMHgxNywgMHhmZCwgMHhhYSwgMHhiNiwgMHg3ZCwgMHhmOCwgMHhhNywKLSAgMHhiYSwgMHg0NCwgMHhlMywgMHgyZiwgMHhjNCwgMHhiMSwgMHg3ZiwgMHhkYSwgMHhhYiwgMHg2NywgMHhkZiwgMHg4YSwKLSAgMHg3YiwgMHhhMCwgMHgzMCwgMHg0MCwgMHgzZCwgMHhlMiwgMHg4YSwgMHhjZCwgMHgxNSwgMHgwMCwgMHgyOSwgMHgzMywKLSAgMHg4ZSwgMHgzZiwgMHgyMywgMHhkOSwgMHg3NywgMHhkMSwgMHgxMiwgMHgzZiwgMHg4MCwgMHhkMywgMHg5ZCwgMHgyNiwKLSAgMHg3MSwgMHhjNywgMHhlNCwgMHg3YiwgMHgyZSwgMHhmYSwgMHgyMiwgMHg0NywgMHhmMCwgMHgxYSwgMHg5MCwgMHg0NywKLSAgMHg3MSwgMHhjZiwgMHgwNiwgMHg1NiwgMHg3ZCwgMHhjMywgMHgyNywgMHhlZCwgMHg3MCwgMHhkYywgMHgyYywgMHg1ZSwKLSAgMHgyMiwgMHhmNiwgMHg3MywgMHhhZCwgMHgxMiwgMHgxMiwgMHg3NCwgMHhhNiwgMHg2NSwgMHhiNSwgMHhkNSwgMHhiMiwKLSAgMHgwZiwgMHg4NiwgMHhmYSwgMHhhNywgMHhmZSwgMHgyYSwgMHhhZSwgMHhiOCwgMHgyMywgMHhjNCwgMHhjOSwgMHgzNywKLSAgMHg5YiwgMHhiMywgMHg2ZiwgMHhkZCwgMHg5NiwgMHhiNCwgMHhjOCwgMHhiOCwgMHhiNiwgMHhiNywgMHg1NCwgMHhkMiwKLSAgMHhjNiwgMHhiYiwgMHgwOSwgMHgwYywgMHhmMiwgMHhiNywgMHgzNiwgMHgzMSwgMHhmNCwgMHgxNCwgMHgyZiwgMHg5NSwKLSAgMHhkNCwgMHg4MywgMHhiZiwgMHgzMSwgMHhlMSwgMHhkNCwgMHhmMiwgMHg5MywgMHg1ZiwgMHg0MiwgMHhjNiwgMHgxZiwKLSAgMHhjZCwgMHhkYiwgMHhmOSwgMHg4MywgMHhlYywgMHhhZiwgMHg5NywgMHgzOCwgMHhkNSwgMHg4ZCwgMHgzYiwgMHg4NSwKLSAgMHg3MSwgMHhhMiwgMHgyZCwgMHhkYSwgMHhkYywgMHg0MywgMHgxNiwgMHhlYywgMHhiMiwgMHg1YSwgMHgyNCwgMHg0NCwKLSAgMHg3MCwgMHhhYiwgMHg5NSwgMHhiOCwgMHhkNywgMHhhNiwgMHhjMSwgMHgwMCwgMHgyYywgMHg5ZSwgMHhlNCwgMHg0OCwKLSAgMHg0MiwgMHg5NCwgMHhkYSwgMHhiYSwgMHg4ZSwgMHhhZSwgMHgxNSwgMHg3NywgMHgyMywgMHg1NSwgMHg3YSwgMHg3MiwKLSAgMHhkMiwgMHhmNywgMHhlMCwgMHhhNSwgMHg0OCwgMHg2YSwgMHg1ZSwgMHhhNiwgMHgzOCwgMHhmNywgMHg4OSwgMHg3YiwKLSAgMHg5MywgMHhjNCwgMHg3NywgMHg2ZSwgMHgwMywgMHhiNCwgMHg1YywgMHgyYywgMHg5NywgMHhkZiwgMHhlMSwgMHhiYSwKLSAgMHg1NiwgMHgwMiwgMHgxYSwgMHg5NCwgMHg4NCwgMHhmYiwgMHhmMywgMHgyYSwgMHgzYSwgMHgyNywgMHg0YSwgMHg0YSwKLSAgMHg1MiwgMHhlYSwgMHg3YiwgMHg4MSwgMHhlNSwgMHg3MCwgMHg3OCwgMHgwYSwgMHhhMCwgMHhiOCwgMHhmYiwgMHg4MywKLSAgMHhhZCwgMHg4YiwgMHgxNCwgMHgzYywgMHhhMiwgMHgzMSwgMHg3MSwgMHhkNSwgMHhiNiwgMHhiZSwgMHhjMiwgMHg1MiwKLSAgMHg5NSwgMHhlNiwgMHhmMywgMHgyMCwgMHg5ZiwgMHgzNSwgMHg2OSwgMHgwYSwgMHhlYiwgMHhjYSwgMHgxNSwgMHhlNiwKLSAgMHg5ZSwgMHhhNywgMHg1ZCwgMHgzNywgMHhhYSwgMHhmYiwgMHgzZCwgMHhmOCwgMHg3MCwgMHhiOCwgMHhhNSwgMHhjMiwKLSAgMHhkNSwgMHhkYiwgMHgxOSwgMHg3NywgMHhiMSwgMHg5MywgMHhjYSwgMHg4OSwgMHg3NiwgMHhjNywgMHg1ZCwgMHg0OCwKLSAgMHgyYiwgMHg4ZCwgMHgyMSwgMHhiNSwgMHg2ZCwgMHgwNSwgMHg0MCwgMHhlZiwgMHhhYSwgMHgxNiwgMHgwYSwgMHgxNCwKLSAgMHgzZCwgMHgxYywgMHhjMiwgMHhhYiwgMHhmYywgMHg1MiwgMHgyMCwgMHhjOSwgMHgyYywgMHg0ZiwgMHgzNSwgMHgzMiwKLSAgMHhjZSwgMHhkNiwgMHhhNCwgMHg4NywgMHgyMSwgMHg1ZSwgMHgyMiwgMHgzOCwgMHhiMCwgMHhkYiwgMHg0YywgMHgzYSwKLSAgMHg5MywgMHhjOCwgMHhmMywgMHgyYSwgMHgwMCwgMHhhOSwgMHhkNywgMHgxNCwgMHgwZiwgMHhjMSwgMHgyYSwgMHgzYSwKLSAgMHhkMSwgMHg0YSwgMHg5MywgMHhhYSwgMHhlOSwgMHg4ZCwgMHgzZCwgMHg1OSwgMHg4MywgMHgzOSwgMHhlNSwgMHg1MywKLSAgMHgxOCwgMHg5OSwgMHhmMSwgMHg3ZiwgMHgwNywgMHhiMywgMHgxNywgMHg3MCwgMHg0ZSwgMHgyMSwgMHg1YiwgMHgzMiwKLSAgMHgyNCwgMHg4NSwgMHgyOSwgMHg4NiwgMHg5NywgMHhjOSwgMHgyOSwgMHhiMSwgMHhkZSwgMHhlMywgMHgyYiwgMHgxYSwKLSAgMHg1OCwgMHgxZSwgMHhiZCwgMHgxZCwgMHg4ZiwgMHg1OCwgMHgxNSwgMHhmNywgMHg4YiwgMHg5NiwgMHhiYiwgMHg0NSwKLSAgMHhlYiwgMHgyOSwgMHhiMywgMHg1ZiwgMHg2ZCwgMHhmMiwgMHg1OCwgMHg1ZCwgMHhjZSwgMHg3NCwgMHg1MSwgMHgxNiwKLSAgMHgwNCwgMHhjNSwgMHhhOSwgMHgyYSwgMHg0YywgMHg3OCwgMHhhYSwgMHhkYiwgMHg4YiwgMHg3NSwgMHhiNCwgMHgyYiwKLSAgMHhhNywgMHgzMSwgMHgxYSwgMHgxYiwgMHhmMSwgMHhlNiwgMHg0OCwgMHgzYiwgMHgxYiwgMHgwNywgMHhlNSwgMHhmYiwKLSAgMHhjZiwgMHhlMCwgMHhkNywgMHg3MywgMHg1NywgMHgxMSwgMHhlNiwgMHhlMywgMHhmNiwgMHg5YiwgMHhmNCwgMHgzNiwKLSAgMHgyMCwgMHhmOSwgMHgxMSwgMHhiOCwgMHg0MiwgMHg3YSwgMHg1ZiwgMHgzMSwgMHg1MSwgMHg2ZiwgMHhiNCwgMHgwOCwKLSAgMHgyZCwgMHhhYiwgMHg5NCwgMHg3YywgMHgyNCwgMHg5MiwgMHgzNiwgMHg3YywgMHg0MSwgMHgwNywgMHg0MywgMHg3YSwKLSAgMHhhYiwgMHg5ZiwgMHhmMCwgMHg3OSwgMHhiNiwgMHg1YywgMHgyZCwgMHhmOCwgMHg3ZCwgMHhjNywgMHgwZCwgMHhjOCwKLSAgMHg2MywgMHgyYSwgMHgyNSwgMHhmYSwgMHhjMywgMHgyYywgMHgzMiwgMHg2NCwgMHgyMCwgMHhlZCwgMHg0YSwgMHg2OCwKLSAgMHhmYiwgMHhlNCwgMHg3NywgMHgxYiwgMHg1OSwgMHhlZiwgMHg0OCwgMHgzYiwgMHgwOSwgMHhmNiwgMHg2OCwgMHg4ZSwKLSAgMHhmYSwgMHhiZCwgMHgxOCwgMHhjZCwgMHg2NiwgMHgxOCwgMHhlNywgMHhlYSwgMHg1NiwgMHhiNCwgMHhhOSwgMHhjOSwKLSAgMHhhOSwgMHhhNywgMHhjNywgMHhkMCwgMHhiNSwgMHhhZiwgMHgzNywgMHgwYiwgMHhjNiwgMHgyZiwgMHg3ZCwgMHhiNiwKLSAgMHg2MiwgMHg5OCwgMHg1NiwgMHgyYSwgMHhmMywgMHgzMCwgMHhlNiwgMHg0OCwgMHg0YiwgMHg1MywgMHhhZiwgMHhjZiwKLSAgMHhjMywgMHg1YywgMHhiMCwgMHgxZCwgMHg1MiwgMHg3OSwgMHhiOSwgMHg5NiwgMHg5NCwgMHhhOSwgMHgyYSwgMHg1OSwKLSAgMHhmMywgMHg4MSwgMHgyZSwgMHgyZCwgMHg0MCwgMHgwZSwgMHhhMCwgMHg2ZiwgMHg1YSwgMHgxMiwgMHg3YywgMHg0MCwKLSAgMHg2MSwgMHg0YywgMHg1ZCwgMHgzMSwgMHg4OSwgMHg0ZiwgMHhiYywgMHhkMywgMHhiNywgMHgyNSwgMHg5NywgMHg2MSwKLSAgMHhjOCwgMHg1YiwgMHg0ZCwgMHg3MiwgMHgyNSwgMHhlNiwgMHhjYiwgMHg0NSwgMHhjNSwgMHgyYiwgMHg5NCwgMHg5MiwKLSAgMHg0MCwgMHgwYiwgMHg2ZCwgMHgxYSwgMHgxYiwgMHgzYSwgMHhlNywgMHhlZiwgMHhmMSwgMHhhMSwgMHg4YywgMHhiMywKLSAgMHgyOSwgMHg2MiwgMHgzOCwgMHg2MSwgMHhmYiwgMHg2ZCwgMHhhNiwgMHg3MywgMHhjMCwgMHg2OSwgMHgzMiwgMHg0NCwKLSAgMHhhNSwgMHhiMCwgMHgxNywgMHhlOCwgMHgyNSwgMHhhZSwgMHg0NSwgMHg2YiwgMHhlYSwgMHg1NywgMHhmNywgMHg3NywKLSAgMHg1NCwgMHg1OSwgMHg0YywgMHhlOSwgMHhiNywgMHgzMywgMHg3NSwgMHhiZCwgMHhjYiwgMHg2ZSwgMHg1NCwgMHhkMCwKLSAgMHhkOSwgMHg2OSwgMHhiNCwgMHhiNCwgMHg4ZSwgMHg0NiwgMHg2MywgMHgyMCwgMHg5ZCwgMHhhOSwgMHgyZCwgMHg4ZCwKLSAgMHg5MywgMHhiMiwgMHg0MCwgMHhkYSwgMHg4OSwgMHgyNCwgMHhlOCwgMHg3NywgMHgwZSwgMHg5NSwgMHg0YSwgMHg1NiwKLSAgMHhmNSwgMHgzNSwgMHhhNiwgMHhjYiwgMHg1NiwgMHhhZiwgMHg0ZiwgMHg0NiwgMHgxNiwgMHhlNiwgMHhmOCwgMHg1MSwKLSAgMHhlNywgMHgzNywgMHg3MCwgMHg1YywgMHhjYiwgMHgyZCwgMHhjNCwgMHhjMCwgMHhiOCwgMHgzYSwgMHhkYSwgMHg1MiwKLSAgMHhiMCwgMHhhNiwgMHg0MywgMHhhZCwgMHg0OCwgMHgwOSwgMHhkZSwgMHg5MiwgMHhhNCwgMHg2YywgMHgxZSwgMHg2MSwKLSAgMHhjYywgMHg3NCwgMHhhMCwgMHg0NywgMHg0ZSwgMHg4NywgMHg2MywgMHg0MCwgMHg3MiwgMHg0YiwgMHg4OSwgMHgyZSwKLSAgMHhlNiwgMHhmMywgMHg2ZiwgMHg1ZiwgMHhlZSwgMHg5MiwgMHg2ZSwgMHhhYSwgMHg2OSwgMHg0MSwgMHg2ZCwgMHhiMiwKLSAgMHhiMCwgMHgxYiwgMHg4ZSwgMHg4NSwgMHgwZSwgMHhlMiwgMHgxYSwgMHg0ZiwgMHg0MiwgMHg0NywgMHhlYiwgMHhmMywKLSAgMHgxMSwgMHg1ZCwgMHg4NSwgMHhjNCwgMHhiNCwgMHgwYiwgMHhjYiwgMHg1OCwgMHg0MiwgMHg1YiwgMHgxYywgMHhlNSwKLSAgMHg2NSwgMHg1ZCwgMHgxMiwgMHgwNywgMHg1ZCwgMHg5MywgMHhlMCwgMHgzYSwgMHg3NSwgMHhkZSwgMHhiYiwgMHhhYSwKLSAgMHgyZiwgMHgxYSwgMHhiYiwgMHhjYiwgMHhiZiwgMHhjNiwgMHg5NSwgMHg3NiwgMHg5MCwgMHg5MSwgMHhlNCwgMHhmMiwKLSAgMHgyNiwgMHgzYSwgMHg2MCwgMHhhYywgMHgzNCwgMHg1MSwgMHhkYSwgMHhjNiwgMHgwNywgMHg0OCwgMHg3MywgMHg1ZCwKLSAgMHhmZSwgMHg3NiwgMHg4OSwgMHgwNywgMHg1ZCwgMHg0NiwgMHg4ZSwgMHhiYSwgMHhkNywgMHg2NywgMHhiYiwgMHg4ZiwKLSAgMHhiYywgMHhjZSwgMHgzOSwgMHgzOCwgMHg5NSwgMHg0OSwgMHg3YiwgMHhiYywgMHg2NywgMHg4MiwgMHgwMywgMHg4ZCwKLSAgMHhmMiwgMHg1MywgMHgxMiwgMHhkMCwgMHg4OSwgMHhjYiwgMHg0YiwgMHg4YiwgMHg0ZCwgMHhkZSwgMHhjMywgMHgzYSwKLSAgMHhjOCwgMHhlNiwgMHhmYSwgMHg4ZSwgMHhkZCwgMHgxYSwgMHg3NiwgMHgzZiwgMHhkNiwgMHg3ZCwgMHhmMywgMHg1NCwKLSAgMHhlOSwgMHhhMiwgMHgzYSwgMHgxMywgMHhiMiwgMHgzYiwgMHhjZiwgMHhhNiwgMHg5NywgMHhmOCwgMHg5OCwgMHg1YSwKLSAgMHg0NiwgMHgwNCwgMHhiYiwgMHg4MywgMHhhMywgMHg5ZiwgMHhkYywgMHg0YiwgMHhiNCwgMHgwYiwgMHg5OCwgMHgwNCwKLSAgMHg2ZiwgMHhjZCwgMHgwZiwgMHgyNSwgMHhiNywgMHgzYSwgMHgxZiwgMHhkNCwgMHg1YSwgMHhhOSwgMHg4NSwgMHg0MCwKLSAgMHg4NSwgMHgyOCwgMHgxZSwgMHhmZCwgMHg5YSwgMHhhZCwgMHgxOCwgMHhlOSwgMHg5YywgMHg5MSwgMHg2YSwgMHhkMiwKLSAgMHhkNSwgMHgxOCwgMHhiMywgMHgxNSwgMHgwOSwgMHg5YSwgMHg1OSwgMHg1ZCwgMHhiZCwgMHhkOCwgMHhkNCwgMHhjNCwKLSAgMHgyOSwgMHgwMiwgMHgyZCwgMHhjZSwgMHgzMywgMHhjOCwgMHg5NywgMHg2ZCwgMHg5MiwgMHg3ZiwgMHhhMCwgMHg5NCwKLSAgMHhkMSwgMHhlNiwgMHg2ZCwgMHg3ZSwgMHhiMSwgMHhiZSwgMHg4NCwgMHg3OCwgMHhhNCwgMHhhOCwgMHg3OCwgMHhkNCwKLSAgMHhkZCwgMHgxYSwgMHhhZCwgMHhlNCwgMHg5MywgMHg0ZCwgMHgzMywgMHgxNCwgMHhkYSwgMHg2OSwgMHhhMiwgMHg0ZSwKLSAgMHhkYSwgMHhlZCwgMHhhMywgMHg4YiwgMHhmYywgMHgyMywgMHg5MSwgMHgwZSwgMHhlOSwgMHgxNCwgMHhiMiwgMHhkNSwKLSAgMHhjZSwgMHgzMywgMHg5MCwgMHhhZSwgMHg1MSwgMHg0ZiwgMHg1NSwgMHhjNCwgMHg5MiwgMHg5MiwgMHg1MiwgMHhlMiwKLSAgMHgwZiwgMHhhMSwgMHg0OCwgMHg3MCwgMHg2YywgMHgxZiwgMHg1MiwgMHg0ZCwgMHg3MiwgMHg3MCwgMHgxMywgMHgyNCwKLSAgMHhiOSwgMHg1YywgMHhiMSwgMHg5OSwgMHg1OCwgMHhiZSwgMHg0OCwgMHhlNywgMHgzZSwgMHg1MSwgMHg4YiwgMHg0OCwKLSAgMHhmNywgMHgzMiwgMHhlNiwgMHg0ZiwgMHg3YiwgMHhlMSwgMHgyMywgMHhkZSwgMHg2NCwgMHhmNSwgMHhlYSwgMHg1MiwKLSAgMHhlYiwgMHg3YywgMHhhYSwgMHhkZiwgMHhhNywgMHg5YiwgMHhkMSwgMHg0YiwgMHg3NiwgMHgxOSwgMHhhOSwgMHhjMiwKLSAgMHhmOCwgMHg5ZSwgMHhkYywgMHg5MiwgMHhhZSwgMHg0YiwgMHgxZSwgMHg1OCwgMHhlMiwgMHgyMiwgMHhjYywgMHgwNCwKLSAgMHhmOSwgMHg5MSwgMHhlZSwgMHgyMSwgMHgzYSwgMHg2NSwgMHhkZiwgMHg1MCwgMHg3NSwgMHgyOSwgMHhlYywgMHhjZiwKLSAgMHhlYiwgMHgyNSwgMHhiZiwgMHg0ZCwgMHg3NiwgMHg3MSwgMHg4MywgMHg5ZiwgMHgwMiwgMHhjZCwgMHhhYywgMHhmYywKLSAgMHg1YiwgMHg4OCwgMHg4NSwgMHgyNiwgMHhkZSwgMHg5MCwgMHg4YiwgMHg0ZSwgMHg1MCwgMHg4NCwgMHgwZSwgMHg4YSwKLSAgMHg4NiwgMHhiNSwgMHg2OSwgMHhhOSwgMHgwNywgMHhkMiwgMHg1OSwgMHg3MSwgMHg0MywgMHg2NywgMHhiZiwgMHg5NSwKLSAgMHg0NCwgMHg3NywgMHgwYSwgMHhmMSwgMHg2YSwgMHg0MSwgMHhkMywgMHg5MywgMHg4OSwgMHhlYywgMHg1MiwgMHg5ZSwKLSAgMHhiOCwgMHhhNiwgMHg1YiwgMHhlOSwgMHgyNCwgMHg4MCwgMHg0ZCwgMHg2NiwgMHhiYywgMHhiNCwgMHhiNCwgMHgzOCwKLSAgMHhkMiwgMHg1YywgMHg2ZCwgMHg0OSwgMHg1YSwgMHgxNiwgMHgwMiwgMHg5MiwgMHhhNCwgMHg5ZCwgMHg4MiwgMHgwZiwKLSAgMHg3MSwgMHgxNSwgMHhlYSwgMHhhOCwgMHg2OCwgMHgxNCwgMHg1MSwgMHg0NSwgMHgwMCwgMHg1MSwgMHg0NSwgMHgxNCwKLSAgMHgwMiwgMHgyNywgMHgxOSwgMHg3ZSwgMHgyNSwgMHg4YiwgMHhmZSwgMHhkNSwgMHg1YiwgMHgzZSwgMHhmYywgMHg1MywKLSAgMHhkZCwgMHgyMiwgMHg3MSwgMHg5NywgMHhlMiwgMHg1OCwgMHhiZiwgMHhlZCwgMHg1NSwgMHhiMywgMHhlZiwgMHhjNSwKLSAgMHgzZCwgMHhkMCwgMHgwNSwgMHgxNCwgMHg1MSwgMHg1MCwgMHgwMiwgMHg5MywgMHgzOCwgMHhlMywgMHhmMiwgMHgzZCwKLSAgMHg5NywgMHg3ZCwgMHgxMSwgMHgyMywgMHhmOCwgMHgwZCwgMHgzOSwgMHhkMiwgMHg2NywgMHgxYywgMHg3ZSwgMHg0NywKLSAgMHhiMiwgMHhlZiwgMHhhMiwgMHgyNCwgMHg3ZiwgMHgwMSwgMHhhOSwgMHgwMywgMHg3YywgMHg2ZiwgMHg4YiwgMHhiNywKLSAgMHhmMywgMHgwNywgMHhkOSwgMHg0YSwgMHg5YywgMHg1ZiwgMHhjMiwgMHgyMSwgMHg3MSwgMHgwYiwgMHgwMSwgMHhiOSwKLSAgMHhlMywgMHgxMiwgMHhjOCwgMHg2OSwgMHhkNywgMHhkYiwgMHhlNywgMHg4NywgMHgyMywgMHhmMywgMHhhMywgMHhjOCwKLSAgMHg0ZiwgMHg1NiwgMHhkYywgMHg0OSwgMHhlZiwgMHgwNCwgMHgyYiwgMHhiZiwgMHg1ZSwgMHgwNCwgMHg4ZiwgMHgxYSwKLSAgMHg2YiwgMHg4ZCwgMHhmMSwgMHg3NiwgMHhmZSwgMHg2MCwgMHhmYiwgMHgyYiwgMHg2NiwgMHhiYSwgMHhkMCwgMHgxZiwKLSAgMHgzMCwgMHg3MCwgMHg0ZiwgMHgyZSwgMHhiOCwgMHgzMSwgMHgzZCwgMHhiZiwgMHg3NSwgMHgxMiwgMHg2MywgMHhjZSwKLSAgMHg3YSwgMHg1YiwgMHhiMCwgMHhhZSwgMHhiMSwgMHhkNiwgMHg3NSwgMHhlNCwgMHhmNywgMHg4NiwgMHg0NywgMHhiZiwKLSAgMHg3YiwgMHgxMywgMHgyNSwgMHhiNCwgMHhmNiwgMHhjMywgMHhkMiwgMHhiNCwgMHgzZCwgMHhhZiwgMHgwYSwgMHg3MCwKLSAgMHhlMiwgMHg0NSwgMHhhNiwgMHgzNSwgMHg4MywgMHgzOCwgMHg4NywgMHg5NiwgMHhjNiwgMHg0OSwgMHg2NiwgMHhjZCwKLSAgMHg5MiwgMHhhZCwgMHhiOCwgMHhiMywgMHhkYywgMHg0NiwgMHg5MywgMHhlNCwgMHhiMywgMHhiNSwgMHhjYSwgMHhjNCwKLSAgMHg4ZSwgMHhiYiwgMHhlNSwgMHhlZCwgMHgxMywgMHhlZiwgMHgyYSwgMHgzZSwgMHg5ZSwgMHhjZiwgMHhkMywgMHg0YSwKLSAgMHhmZiwgMHgwMCwgMHg4NCwgMHgzNiwgMHgzYSwgMHhhYywgMHg1MywgMHg4OSwgMHg1MSwgMHhmMiwgMHhjOCwgMHg2ZiwKLSAgMHgyNiwgMHgwZCwgMHhhMywgMHgyYywgMHgyYywgMHhkYiwgMHhlZSwgMHgyZiwgMHg5ZSwgMHg4ZCwgMHhjMiwgMHhiYSwKLSAgMHgzNCwgMHg3OSwgMHhhMCwgMHhjYywgMHg1NywgMHg0ZSwgMHg4MywgMHg5OCwgMHgwNCwgMHgyOCwgMHhmZSwgMHg4MSwKLSAgMHg1ZiwgMHhlOSwgMHg1NSwgMHg4YiwgMHg4OSwgMHg0OSwgMHhiNiwgMHhmMSwgMHgxMywgMHgwMiwgMHhiOCwgMHg2MywKLSAgMHg3NywgMHg5OCwgMHhhYiwgMHg2NSwgMHgzMiwgMHhkYiwgMHg3NiwgMHgyNCwgMHhjOCwgMHg4YSwgMHg1MCwgMHhlZCwKLSAgMHgyMSwgMHhjOCwgMHg0MSwgMHhkMywgMHg4ZCwgMHhmYSwgMHg5NCwgMHg4NSwgMHg4ZSwgMHg2NCwgMHg5ZiwgMHg1MCwKLSAgMHgyMywgMHhjMiwgMHhiYSwgMHgyMSwgMHgzNywgMHhhNywgMHg1NywgMHg1OCwgMHhmZCwgMHgwZSwgMHg3YSwgMHg5MCwKLSAgMHg1YSwgMHhiMSwgMHhkMiwgMHg1ZiwgMHg1MSwgMHg1MywgMHg4OCwgMHg5NiwgMHhiOSwgMHg4YSwgMHgxMSwgMHg3MiwKLSAgMHgyYiwgMHg0YywgMHgxMSwgMHgzZSwgMHg3NSwgMHhiNCwgMHgzOCwgMHg4NywgMHhhMCwgMHg2OCwgMHgxZiwgMHgyZiwKLSAgMHg4NiwgMHhlNiwgMHhiYiwgMHg2NiwgMHgzYSwgMHhmNCwgMHgyYSwgMHhmMywgMHg1MiwgMHhiNCwgMHg2ZiwgMHhmMywKLSAgMHg5MCwgMHgwNywgMHg4ZCwgMHg2YiwgMHhjMSwgMHhlZSwgMHhkNiwgMHhlNywgMHgyNSwgMHg4OCwgMHhkNiwgMHhkOSwKLSAgMHhjZSwgMHg0YiwgMHhiNCwgMHhjZSwgMHg4YSwgMHg4OSwgMHgzNiwgMHg5NywgMHgxNiwgMHg5NSwgMHgyOSwgMHg0OCwKLSAgMHg0ZSwgMHhkNiwgMHgxYywgMHg2MCwgMHhhYywgMHhmNSwgMHhmNywgMHhiMiwgMHg5ZCwgMHhmMiwgMHgyOCwgMHhmMywKLSAgMHgyNywgMHg5OCwgMHg4ZiwgMHgwMSwgMHg1MiwgMHgzOCwgMHgzYywgMHhlYiwgMHg4YiwgMHg5MCwgMHhlNCwgMHhkOSwKLSAgMHg2ZiwgMHhhYSwgMHhkZCwgMHhmYSwgMHhjOCwgMHhmMSwgMHg4MywgMHg3MSwgMHgzZCwgMHhkZCwgMHhhOSwgMHgwMCwKLSAgMHgxNiwgMHhkZiwgMHgxZiwgMHhhYSwgMHhlYiwgMHg3YywgMHhhYiwgMHgwNywgMHhkMywgMHhjYywgMHgyYiwgMHg5MiwKLSAgMHhlMywgMHg4OCwgMHgzYywgMHg4YiwgMHg5YywgMHg5YiwgMHhhZSwgMHgyZCwgMHg3ZSwgMHg5MSwgMHg4ZiwgMHg0YiwKLSAgMHg5OCwgMHhhZSwgMHg3OSwgMHhhZCwgMHhiNywgMHgxZCwgMHgxMiwgMHgyMiwgMHhjOSwgMHg1ZiwgMHhmNSwgMHg4YSwKLSAgMHg2MSwgMHg1ZCwgMHgwMywgMHg5ZiwgMHhhYywgMHg5MiwgMHg5MiwgMHg3YywgMHg3NywgMHg1ZSwgMHg4YSwgMHhlZiwKLSAgMHg2MiwgMHg3MSwgMHgzYywgMHhlNiwgMHhmMSwgMHhkZCwgMHg5MCwgMHhkNSwgMHhjYywgMHg5MCwgMHg0YSwgMHg3ZCwKLSAgMHgwMywgMHg3YSwgMHhmNCwgMHhkNiwgMHgzNiwgMHgwOCwgMHgxZCwgMHg0MSwgMHgxZCwgMHhmYiwgMHhkZiwgMHg0YSwKLSAgMHg1MywgMHhiOCwgMHgzMSwgMHg3OSwgMHhjNywgMHg2ZCwgMHg4OSwgMHhiYSwgMHgxOSwgMHhmMywgMHhiMiwgMHgwZSwKLSAgMHhjNSwgMHgzYywgMHhkNywgMHgzNiwgMHg5NCwgMHg4NCwgMHgwNywgMHgxZSwgMHg0NiwgMHhjZSwgMHhkZCwgMHg2NSwKLSAgMHgwOSwgMHgwMCwgMHgyNSwgMHg2OSwgMHhkOCwgMHhmMywgMHgwNywgMHg0NSwgMHgyNSwgMHgyNCwgMHg3YywgMHgyZSwKLSAgMHhhNiwgMHg0ZSwgMHhjZCwgMHg3OCwgMHhiNywgMHg1ZiwgMHgyZCwgMHgyZCwgMHhkZCwgMHhhZCwgMHgxMywgMHhkOSwKLSAgMHg5YiwgMHgwOSwgMHhmNCwgMHg4NSwgMHgyMSwgMHhlNiwgMHg1NSwgMHhiNCwgMHhlOCwgMHhmNSwgMHgzYiwgMHhmMSwKLSAgMHg0OSwgMHhlZSwgMHgxYSwgMHgyMCwgMHgxYSwgMHhiZSwgMHhiZCwgMHhmMCwgMHhjYSwgMHg2OSwgMHhkOCwgMHhkNSwKLSAgMHg5YiwgMHg2NSwgMHgxMCwgMHgzMSwgMHg5YiwgMHg1OCwgMHg5NywgMHgzNSwgMHg4OSwgMHg5MiwgMHhiYiwgMHg1NSwKLSAgMHgxNiwgMHhkOSwgMHg2MiwgMHgyYywgMHg3MiwgMHhlYSwgMHhkZCwgMHg1NywgMHhhMywgMHg1ZCwgMHhjOSwgMHgxZSwKLSAgMHg5MiwgMHhhMiwgMHgwNywgMHg3NywgMHhiMiwgMHhhMCwgMHgzMSwgMHg1YiwgMHg4NiwgMHg3ZCwgMHg3ZSwgMHhiZSwKLSAgMHhjNywgMHg5ZiwgMHg3MywgMHg4NSwgMHgxNywgMHgxZCwgMHhiMCwgMHgyMCwgMHgyOSwgMHg0MiwgMHgxYiwgMHg4OCwKLSAgMHgwYiwgMHg5NSwgMHgyYiwgMHhhMSwgMHgwOSwgMHgwNCwgMHhhYiwgMHhhYSwgMHg1MiwgMHgwZSwgMHg4OSwgMHhmMywKLSAgMHg1MiwgMHg3ZCwgMHgxYiwgMHhlZiwgMHhhNywgMHgxZSwgMHg3NywgMHg1YiwgMHg0OSwgMHhmMywgMHg5NCwgMHg5NSwKLSAgMHgxMiwgMHg0MSwgMHgwMCwgMHhmOCwgMHhmZSwgMHg3NywgMHhkNSwgMHhiZSwgMHg4MywgMHg3ZSwgMHg4MywgMHg1YywKLSAgMHg3MSwgMHg2ZCwgMHhiMSwgMHg2MywgMHg0YywgMHg3ZSwgMHg0YywgMHg2NiwgMHhiOSwgMHg2NCwgMHg0OCwgMHg1NywKLSAgMHhiZSwgMHhiYywgMHg1NiwgMHg1NiwgMHhhNSwgMHgxZCwgMHg2YywgMHg5ZCwgMHg5MiwgMHg3NCwgMHg5NCwgMHg4ZSwKLSAgMHhlMCwgMHgwOCwgMHgwMywgMHhhZiwgMHhhMiwgMHhhYiwgMHgyNCwgMHhkYiwgMHhjZSwgMHg0YiwgMHg0NSwgMHhhNCwKLSAgMHhiMSwgMHg4MywgMHg0ZiwgMHgxMCwgMHhkYywgMHg2NywgMHhmMSwgMHg2YiwgMHg5YSwgMHhhNSwgMHhjMiwgMHgzOSwKLSAgMHgzZCwgMHhjMiwgMHg5MiwgMHg0YSwgMHg4OSwgMHgxYSwgMHg0YSwgMHhiNSwgMHhlNiwgMHg3ZiwgMHhkNCwgMHgwNSwKLSAgMHg3OCwgMHg5MSwgMHg5YiwgMHg2MiwgMHg1MCwgMHgxOCwgMHg0MCwgMHg5ZCwgMHg5MywgMHhkYSwgMHgxYiwgMHg3NSwKLSAgMHgyOCwgMHgxZCwgMHhhMiwgMHgwNCwgMHhhNCwgMHgyOSwgMHg0OSwgMHgzYSwgMHhlYSwgMHgwOCwgMHg0OSwgMHgyNCwKLSAgMHg1MiwgMHhkNSwgMHhkNiwgMHhlZSwgMHhkZSwgMHg0ZiwgMHgyYSwgMHgwZCwgMHhhMiwgMHgwMywgMHgyNSwgMHhkYywKLSAgMHg1NywgMHhkZCwgMHg0NiwgMHg5OSwgMHhiYSwgMHhkYywgMHg5MiwgMHg5MCwgMHg1OSwgMHg3OSwgMHhmNCwgMHg5MiwKLSAgMHhlMywgMHg1MSwgMHgxYiwgMHgyNywgMHhhMiwgMHhkMCwgMHg1YywgMHg2ZCwgMHgwMSwgMHg2YSwgMHgxZCwgMHgzNiwKLSAgMHg5NCwgMHgyNCwgMHgxZCwgMHhhOCwgMHhkNCwgMHg5ZSwgMHg0YiwgMHgxNywgMHgxMiwgMHhjNywgMHg2NywgMHgyMiwKLSAgMHhlNiwgMHhlZCwgMHhlNywgMHhmOSwgMHgzZCwgMHg3MSwgMHg5MCwgMHhiMSwgMHhkOCwgMHhmOSwgMHgxYSwgMHgwMCwKLSAgMHg5MCwgMHhmZSwgMHhjZiwgMHhjMSwgMHg0MiwgMHg1YSwgMHg0OCwgMHg3NSwgMHg2NywgMHhkMCwgMHgzYywgMHhmNCwKLSAgMHhmYSwgMHg1MCwgMHg3NywgMHg1OCwgMHg0YSwgMHg3MiwgMHg1MiwgMHg5NCwgMHhlMiwgMHhiMiwgMHg3NCwgMHg0NiwKLSAgMHgxMSwgMHg3MSwgMHg4YywgMHgyNCwgMHhmMCwgMHg2ZCwgMHgxYywgMHg0ZSwgMHhjMCwgMHhkNSwgMHhhZSwgMHg0YywKLSAgMHg4ZCwgMHg4NSwgMHhlZiwgMHhiYiwgMHg5MSwgMHg4NywgMHg5NSwgMHhmNiwgMHgyMCwgMHhkNywgMHhhMywgMHhjNCwKLSAgMHhkYywgMHgwNSwgMHgwOSwgMHgyYSwgMHg3YiwgMHgyNywgMHg4NywgMHgxZCwgMHgyMCwgMHhlYiwgMHg5YSwgMHg0MiwKLSAgMHg1NiwgMHhjOCwgMHgyNywgMHhkMCwgMHgwYSwgMHhkMiwgMHgzNiwgMHg3ZCwgMHg1ZCwgMHhmNSwgMHgyZCwgMHg2NCwKLSAgMHhmYywgMHg3MCwgMHg2NCwgMHg2YywgMHhhMSwgMHhhOCwgMHg1MiwgMHgxNywgMHg4ZiwgMHhjMCwgMHgwYSwgMHgyMCwKLSAgMHg1ZCwgMHg2ZiwgMHg1MSwgMHg4MCwgMHg5YSwgMHhlYSwgMHgzNywgMHhkMSwgMHg0OSwgMHg4NiwgMHg4MywgMHhjYSwKLSAgMHg4NSwgMHg2YiwgMHhiYywgMHhiOCwgMHg0NywgMHhhNywgMHg5MCwgMHg3NywgMHg1NCwgMHhlZiwgMHhmMiwgMHg2NywKLSAgMHgwNiwgMHhjMywgMHhlNSwgMHhjNiwgMHhiYywgMHhlNSwgMHg5NywgMHg2OSwgMHgzNywgMHhjYiwgMHhlNiwgMHhmNSwKLSAgMHgxYSwgMHg2NSwgMHhlNSwgMHhmMywgMHgyZSwgMHg0OSwgMHg1NiwgMHhjMSwgMHhlNSwgMHg4ZSwgMHhjMCwgMHgxYSwKLSAgMHg0OSwgMHhkZiwgMHg4MywgMHg0OCwgMHgwNywgMHhkYiwgMHg1OSwgMHg0YSwgMHhlZSwgMHg0YiwgMHhjOCwgMHhkMiwKLSAgMHgzNiwgMHg5MSwgMHg3ZSwgMHg2OCwgMHg0YywgMHg3NSwgMHhiYiwgMHg5NywgMHgxMiwgMHg2MywgMHgzNywgMHg2NywKLSAgMHhiMywgMHg2MywgMHhiNywgMHgwMSwgMHg2MSwgMHg5MCwgMHhmYiwgMHgyYSwgMHg5NywgMHg3OSwgMHg5YywgMHg4MywKLSAgMHgxNSwgMHhhZSwgMHhjOSwgMHgyZSwgMHgyNSwgMHg2NCwgMHhjNywgMHg0MiwgMHhiZCwgMHhmMSwgMHhjNSwgMHgxZSwKLSAgMHg1ZCwgMHgwNSwgMHgwNCwgMHg4NCwgMHg4MywgMHhhMywgMHhiYSwgMHhiYiwgMHhlZiwgMHg3NiwgMHhkOCwgMHgzNywKLSAgMHg5YiwgMHgzNCwgMHhiYiwgMHg0ZCwgMHhkMiwgMHgyMywgMHg3MiwgMHhlMSwgMHg0YywgMHg2MSwgMHg0YywgMHg0OCwKLSAgMHg2NSwgMHhjMSwgMHhiNCwgMHhiOCwgMHg4NSwgMHgwZSwgMHg1NSwgMHgyNCwgMHg4ZiwgMHg1OCwgMHgyNiwgMHg5MywKLSAgMHg2NiwgMHg2NCwgMHg5OSwgMHgzNSwgMHhlMSwgMHg2MSwgMHg5YiwgMHg0YywgMHgzNiwgMHhiMSwgMHhmOCwgMHhjYSwKLSAgMHhkZSwgMHhhNCwgMHg0ZSwgMHg2ZiwgMHhiNywgMHg5OCwgMHhiMSwgMHhhZiwgMHhlOCwgMHhlMywgMHgyNCwgMHhlOSwKLSAgMHgxZCwgMHg0NywgMHg3YiwgMHhhYSwgMHgxZSwgMHhiNCwgMHg1MCwgMHg3MSwgMHgzYiwgMHg4NCwgMHhjMCwgMHhhNywKLSAgMHgxZiwgMHhiZSwgMHhlNSwgMHg2ZSwgMHgzOCwgMHg3NiwgMHg0YiwgMHhjYiwgMHhiYiwgMHg3OSwgMHgzMSwgMHgzZSwKLSAgMHhjNiwgMHhkYSwgMHg0OCwgMHg0MCwgMHgxZSwgMHhhMiwgMHgyYiwgMHg5ZSwgMHhhMywgMHg5ZCwgMHg0NywgMHhhYSwKLSAgMHg0NywgMHg0NSwgMHgzNSwgMHgwYSwgMHg2YiwgMHg0YSwgMHgyMiwgMHhiOCwgMHgwYiwgMHg3MywgMHg5OSwgMHg2NywKLSAgMHg1NSwgMHhkYiwgMHg4NSwgMHgzNywgMHhkOSwgMHgyYSwgMHg3YSwgMHhlNywgMHg4YSwgMHhhOSwgMHgyOSwgMHg4NCwKLSAgMHhmYiwgMHhhNywgMHhjZSwgMHg5OSwgMHg2ZCwgMHg1ZSwgMHhjYywgMHg3NywgMHg3ZCwgMHg2NSwgMHgyMywgMHg2ZCwKLSAgMHhhYiwgMHg1ZCwgMHhjNSwgMHgwMywgMHhkMywgMHg1NiwgMHhiMCwgMHhlZSwgMHhhZiwgMHg5YywgMHgzOCwgMHhjYSwKLSAgMHhiYywgMHg4ZiwgMHg4NywgMHg3OSwgMHgyNiwgMHgzYiwgMHhjNCwgMHgxOSwgMHg2ZSwgMHhjOSwgMHhiYiwgMHgzNSwKLSAgMHg2OSwgMHg3NywgMHhiMCwgMHgzMywgMHg0YiwgMHg2OCwgMHgwZiwgMHhiZCwgMHgwNSwgMHhkZiwgMHhjYiwgMHg0MywKLSAgMHg5MSwgMHhjOCwgMHgwMiwgMHg1NCwgMHhiNCwgMHhlOSwgMHgyZSwgMHgzNiwgMHhiMCwgMHgwMCwgMHgyNSwgMHgyYSwKLSAgMHgwNywgMHhhOSwgMHgwNSwgMHg1ZiwgMHg0MiwgMHhkYiwgMHgyNywgMHg0NSwgMHhiOSwgMHg1YiwgMHg2MywgMHg1YywKLSAgMHg2MCwgMHhiZSwgMHhkYywgMHg4OCwgMHg5MiwgMHg5OSwgMHg0MywgMHhjYywgMHgzYSwgMHg4MywgMHhiNCwgMHhhZCwKLSAgMHgwYiwgMHgwMCwgMHhhNSwgMHg0MywgMHhkNCwgMHg0MSwgMHgwNiwgMHhiMSwgMHgzNiwgMHgzYSwgMHhhOCwgMHhhMiwKLSAgMHg4YSwgMHgwMCwgMHhhMiwgMHg4YSwgMHgyOCwgMHgwNCwgMHg0ZSwgMHgzMiwgMHhmYywgMHg0YiwgMHgxNywgMHhmZCwKLSAgMHhhYSwgMHhiNiwgMHg3ZCwgMHhmOCwgMHhhNywgMHhiYSwgMHg0NCwgMHhlMywgMHgyZiwgMHhjNCwgMHhiMSwgMHg3ZiwKLSAgMHhkYSwgMHhhYiwgMHg2NywgMHhkZiwgMHg4YSwgMHg3YiwgMHhhMCwgMHgwYSwgMHgyOCwgMHhhMiwgMHhhMCwgMHgwNSwKLSAgMHgyNiwgMHg3MSwgMHhjNywgMHhlNCwgMHg3YiwgMHgyZSwgMHhmYSwgMHgyMiwgMHg0NywgMHhmMCwgMHgxYSwgMHg3MywKLSAgMHhhNCwgMHhjZSwgMHgzOCwgMHhmYywgMHg4ZiwgMHg2NSwgMHhkZiwgMHg0NCwgMHg0OCwgMHhmZSwgMHgwMywgMHg1MiwKLSAgMHgwNiwgMHhmOCwgMHhkZiwgMHgxNywgMHg2ZiwgMHhlNiwgMHgwZiwgMHhiMiwgMHhiNiwgMHg1NiwgMHhiOCwgMHhkZiwKLSAgMHgxNywgMHg2ZiwgMHhlNiwgMHgwZiwgMHhiMiwgMHhiNiwgMHg1MCwgMHgwYiwgMHhiYywgMHg0NywgMHhjNSwgMHgyZCwKLSAgMHhkOSwgMHhjNiwgMHgxNSwgMHg3NywgMHhjNSwgMHhhZSwgMHhhOCwgMHhkYywgMHg1YiwgMHg4YywgMHg3MiwgMHhkNywKLSAgMHgzNywgMHg3OSwgMHg2ZCwgMHg3ZCwgMHhlOCwgMHg1OCwgMHhmNSwgMHhhNSwgMHg0MCwgMHgyOCwgMHg3YiwgMHgyYiwKLSAgMHhlNywgMHgwZSwgMHgxMywgMHhlNCwgMHg1NywgMHgzYiwgMHgzZSwgMHg0YSwgMHgxOCwgMHhiZCwgMHgyYiwgMHhiMCwKLSAgMHhiZCwgMHg0NiwgMHg5OCwgMHg4YywgMHg3ZiwgMHgyMiwgMHgwNCwgMHhmNCwgMHgzMywgMHg5YiwgMHg0OSwgMHhmMiwKLSAgMHgyOSwgMHg4NywgMHhkMiwgMHg5OSwgMHgwZCwgMHgyMCwgMHhiNCwgMHg0ZiwgMHg4YSwgMHg5YiwgMHg0MSwgMHgzZiwKLSAgMHgwYiwgMHhhZiwgMHhkNiwgMHgwNCwgMHg2YywgMHhlZSwgMHhiZSwgMHg3NSwgMHhmYywgMHgyOCwgMHg3MSwgMHg4OCwKLSAgMHg5NiwgMHg0YywgMHg5MiwgMHgyNywgMHgxMCwgMHg5YywgMHg0YiwgMHg4ZCwgMHhkOCwgMHhlZSwgMHg4YywgMHgyNiwKLSAgMHhjNSwgMHg5NSwgMHhhOSwgMHhhZCwgMHhlZCwgMHhhNiwgMHg1NiwgMHhhMCwgMHg2MywgMHhjZCwgMHgxYSwgMHhmYywKLSAgMHhlNiwgMHg1ZCwgMHgwOCwgMHgzYiwgMHhlZiwgMHhkMiwgMHg1MiwgMHgzYywgMHg0ZCwgMHg1YSwgMHgxMiwgMHg3MCwKLSAgMHg5MiwgMHg2OCwgMHhhYywgMHhlMiwgMHhhNywgMHgxYywgMHgzMSwgMHhhMywgMHg4YiwgMHgwZCwgMHhiNSwgMHg2OSwKLSAgMHg5NywgMHg2ZSwgMHhlMiwgMHg5YywgMHgyNCwgMHg5NCwgMHhjNCwgMHg2ZCwgMHhhNCwgMHhjMSwgMHhjOCwgMHg1MCwKLSAgMHg5MSwgMHhiZiwgMHhlNiwgMHg4NSwgMHg1ZSwgMHg2MywgMHhlNywgMHhkMiwgMHg1OSwgMHg3MCwgMHg5MiwgMHg3ZiwKLSAgMHg1MSwgMHg2YiwgMHhmNCwgMHgwYSwgMHg5MSwgMHhkOCwgMHg1MiwgMHg3NiwgMHgwYSwgMHg1NCwgMHg5ZCwgMHg3YywKLSAgMHgyMCwgMHg3NiwgMHgzNSwgMHhlOSwgMHhhZCwgMHgxYywgMHgxZCwgMHhiZiwgMHgyZSwgMHhmZCwgMHg2NSwgMHg5ZCwKLSAgMHg4ZSwgMHhlNCwgMHhhZCwgMHgzMiwgMHhlZCwgMHhjMSwgMHg4NSwgMHhiOSwgMHg2ZSwgMHhiYiwgMHgzMywgMHhhMCwKLSAgMHg1YiwgMHg3MSwgMHhkMCwgMHg4MSwgMHhiNSwgMHg4ZiwgMHgwMiwgMHhkYiwgMHhjZCwgMHhhOSwgMHgyZSwgMHgyNywKLSAgMHhjMCwgMHg4NSwgMHg3NCwgMHhhNCwgMHg3NiwgMHhlNywgMHg1ZiwgMHgzMSwgMHgwYiwgMHg4MywgMHhkYywgMHgzMiwKLSAgMHg4NSwgMHgwMSwgMHhjYiwgMHhhNSwgMHhlZSwgMHgxNywgMHg1YiwgMHg0MywgMHhmMiwgMHgwMiwgMHhiYiwgMHgwMywKLSAgMHg2ZSwgMHgyNywgMHg0ZCwgMHgzZSwgMHhmYiwgMHg4MCwgMHg3NSwgMHgwZCwgMHg3ZSwgMHg0OSwgMHg0MCwgMHgwZSwKLSAgMHg2NSwgMHgyOSwgMHgwMywgMHg1ZCwgMHhmYiwgMHgxZCwgMHhmNCwgMHgyYSwgMHgyOCwgMHgzNywgMHgxZSwgMHg4ZiwKLSAgMHg3NCwgMHg3MCwgMHg1NiwgMHhhNiwgMHhlNiwgMHhiNSwgMHg3NSwgMHg1YiwgMHgzMiwgMHg1MywgMHg4OSwgMHg3OSwKLSAgMHgxNCwgMHhhYiwgMHgzYywgMHgzOCwgMHhiNiwgMHhkYiwgMHg1MywgMHhlZCwgMHhiMSwgMHg3OCwgMHhiYSwgMHgyZCwKLSAgMHhjNCwgMHhjNywgMHg3NSwgMHg2MywgMHg5OCwgMHg0NiwgMHg2OSwgMHgwOSwgMHgyYSwgMHg3YSwgMHg0MSwgMHg0ZiwKLSAgMHhlNywgMHg3MiwgMHhhMCwgMHg2OSwgMHgyMywgMHhjNSwgMHg0NSwgMHgyMCwgMHhmNCwgMHhkZSwgMHg5NywgMHhiMSwKLSAgMHhjYywgMHg0MiwgMHhkNSwgMHg2ZiwgMHhjOSwgMHg2NywgMHg1OCwgMHgxMiwgMHhiYiwgMHg5NCwgMHgwOSwgMHgyNiwKLSAgMHgzYSwgMHg2NywgMHhkYiwgMHhhNywgMHhjNywgMHg5NCwgMHg1YSwgMHg5NCwgMHhhNiwgMHhkNCwgMHg0MCwgMHg3OSwKLSAgMHhiNywgMHgxNCwgMHg5ZiwgMHgzNSwgMHhjMiwgMHg5NywgMHg3YSwgMHhlOSwgMHg0MSwgMHg0MCwgMHgwNywgMHgwNywKLSAgMHg4MCwgMHgxNSwgMHgxZCwgMHgyNywgMHgxNCwgMHg1YywgMHg0ZSwgMHgyOCwgMHhiNywgMHgyYSwgMHg3YywgMHhiNywKLSAgMHhhZSwgMHg3MiwgMHg5ZSwgMHhiNCwgMHgzNiwgMHhiOSwgMHg5NywgMHgxNywgMHhiYywgMHhkMCwgMHg0OSwgMHg5YywKLSAgMHhjOCwgMHg1MiwgMHg1MiwgMHgzNywgMHhjYSwgMHhkYiwgMHg2OSwgMHg2ZCwgMHgwNCwgMHgwNCwgMHhmYSwgMHgwOSwKLSAgMHgyNCwgMHg5MiwgMHg0OSwgMHhhNiwgMHhiNywgMHhlZiwgMHg3MSwgMHgzMiwgMHgxYiwgMHhmYywgMHgzNywgMHg3MCwKLSAgMHhmYiwgMHg0YywgMHhlYywgMHhhYSwgMHhlNywgMHg2ZSwgMHg1MywgMHhhOCwgMHg0YiwgMHg5MCwgMHg0OCwgMHg0NCwKLSAgMHgzNiwgMHhmYiwgMHg0NCwgMHg3MiwgMHhhZCwgMHgwZSwgMHhjOSwgMHg1ZiwgMHhiZCwgMHg4MSwgMHhkMCwgMHgxZCwKLSAgMHgyNywgMHg5OSwgMHg1YiwgMHg0OCwgMHhlOSwgMHhkMiwgMHhiNSwgMHg3MiwgMHg1OSwgMHg3MiwgMHg5ZSwgMHhkOCwKLSAgMHgzMiwgMHg1MSwgMHg3OCwgMHg0YSwgMHgzYiwgMHhlNCwgMHhkYSwgMHhhYywgMHg3NywgMHgzMiwgMHg2ZCwgMHg3YywKLSAgMHg5MCwgMHhiMywgMHhjNCwgMHhhZCwgMHhhMSwgMHhhMCwgMHg4ZiwgMHgyYSwgMHhiMiwgMHgzMiwgMHhlYiwgMHg4MCwKLSAgMHgwZSwgMHhlZiwgMHgzOSwgMHgyYSwgMHg0MCwgMHgzZiwgMHhmMiwgMHhlZSwgMHg5NiwgMHhlZSwgMHhiMSwgMHg2ZCwKLSAgMHgyZiwgMHg1YywgMHg3ZCwgMHhjOCwgMHhiZSwgMHhlNCwgMHhiNywgMHhmYywgMHhlNiwgMHhlNiwgMHgwMSwgMHhmZiwKLSAgMHgwMCwgMHg0MSwgMHg1YSwgMHg5OSwgMHg0YSwgMHg1MCwgMHhhZiwgMHhmNywgMHhhZCwgMHhiMCwgMHgxMiwgMHgwMCwKLSAgMHhmNSwgMHhiYSwgMHhiMCwgMHg5ZiwgMHg0ZCwgMHg1OCwgMHg4ZCwgMHhmMCwgMHhmMiwgMHhmNywgMHg3NSwgMHg4YSwKLSAgMHhlNCwgMHhiZSwgMHgyMSwgMHhlNSwgMHg0OCwgMHg4MywgMHg2ZSwgMHgwOSwgMHhlNiwgMHg3NiwgMHhkNywgMHg2NSwKLSAgMHg3MSwgMHg1MSwgMHg5OCwgMHgwOSwgMHgxZSwgMHgwZiwgMHg0OSwgMHg1NiwgMHg5YywgMHg1OCwgMHhkNywgMHg3ZiwKLSAgMHgyZiwgMHg2NiwgMHgyYSwgMHg1MiwgMHhkNywgMHg3ZCwgMHhiMCwgMHg1OCwgMHgyZCwgMHhmMSwgMHhhYywgMHg5YywKLSAgMHgzNSwgMHhjNSwgMHgxMSwgMHgyZCwgMHg5NywgMHg1ZCwgMHgwYywgMHhiMiwgMHhiOCwgMHhlOSwgMHg0YywgMHg2OCwKLSAgMHg2YSwgMHg1NywgMHg1ZSwgMHg2NSwgMHgxNywgMHg0OCwgMHhkYiwgMHhiYSwgMHgwMCwgMHg5MiwgMHhhNCwgMHgyNSwKLSAgMHg3ZSwgMHgzYiwgMHgzYiwgMHhhZSwgMHg2YSwgMHg5NywgMHgzMSwgMHhlMiwgMHgwOCwgMHhlYSwgMHhhNywgMHg2YiwKLSAgMHgyNywgMHhiYywgMHhjOCwgMHgzYiwgMHg0ZSwgMHgwYiwgMHg5YiwgMHg1ZiwgMHhlZCwgMHhlOSwgMHg4MywgMHg3NCwKLSAgMHg3ZSwgMHgxZSwgMHgxMywgMHg2MSwgMHhkMiwgMHgxMiwgMHgyZCwgMHhkNiwgMHhjMCwgMHg4NywgMHhlNiwgMHgxNCwKLSAgMHgyMSwgMHg1YywgMHhjOSwgMHgwNSwgMHhkMiwgMHgzYiwgMHgyNiwgMHg3NCwgMHg0MCwgMHgyMywgMHhiMywgMHg0YSwKLSAgMHg4OCwgMHhmMCwgMHg1MCwgMHhhOSwgMHg5YiwgMHgyNCwgMHg2ZSwgMHgxZCwgMHg2MCwgMHhiMiwgMHhlNCwgMHhiNywKLSAgMHg4ZCwgMHhkYSwgMHg5ZiwgMHhiYywgMHg1ZiwgMHhjYSwgMHg3ZiwgMHg5YywgMHhhZSwgMHgxYSwgMHg1NSwgMHgzZSwKLSAgMHg3YiwgMHg4NywgMHhiOCwgMHhmNiwgMHhhZiwgMHgyOCwgMHg5ZSwgMHg0ZCwgMHg5ZiwgMHgwNSwgMHhhOSwgMHgyMywKLSAgMHhkNSwgMHg1YiwgMHg5ZCwgMHhiMSwgMHg1ZiwgMHg3MiwgMHg4NSwgMHgyYywgMHg1ZSwgMHhlZiwgMHgwYiwgMHg5MywKLSAgMHgxMCwgMHg5ZCwgMHgyYSwgMHgzNCwgMHg1ZSwgMHg2OCwgMHhmMCwgMHhmZCwgMHg2OSwgMHhkMCwgMHgzZCwgMHhhMywKLSAgMHhhMCwgMHgxZiwgMHgxNSwgMHgyYiwgMHg5NCwgMHhmZSwgMHg4OCwgMHhhZSwgMHhlNiwgMHhkYSwgMHhjNSwgMHg2YywKLSAgMHhjZCwgMHgxYiwgMHg3MCwgMHg5YSwgMHg0OCwgMHg2YiwgMHg2YSwgMHg1NCwgMHg2YiwgMHg3MywgMHgyYSwgMHhlNCwKLSAgMHg2YiwgMHg0MywgMHg2NCwgMHhhOSwgMHgwYywgMHhhNCwgMHhmMiwgMHhmNCwgMHhmMSwgMHgzYSwgMHhhYywgMHgyNCwKLSAgMHg5YiwgMHg3OSwgMHg5YiwgMHgzNiwgMHg4NiwgMHgyMywgMHhiNCwgMHgxMSwgMHhjOCwgMHhjNSwgMHhkMywgMHgyYiwKLSAgMHhjOSwgMHhkZCwgMHg3MiwgMHgzYiwgMHg2ZiwgMHgwYywgMHg3MCwgMHhhMCwgMHgwMiwgMHhlNCwgMHgwNiwgMHhkMiwKLSAgMHgwYywgMHhlNCwgMHhhMCwgMHhlYywgMHgwNSwgMHgyOSwgMHhjNywgMHgxMywgMHhjYSwgMHg5MCwgMHg3YSwgMHhlOCwKLSAgMHhiNiwgMHg5NSwgMHhmNSwgMHhmYywgMHhmYSwgMHg4NSwgMHg5OCwgMHhjNCwgMHgxYywgMHg1YSwgMHhmYiwgMHgzZCwKLSAgMHhjNSwgMHhhMSwgMHg4ZiwgMHgyOSwgMHg2NiwgMHgxMCwgMHg5MywgMHgyMSwgMHhkOSwgMHg0YiwgMHg3MCwgMHgzZiwKLSAgMHgzYywgMHgxMiwgMHg0MSwgMHg2ZCwgMHgxMiwgMHgxNiwgMHg3NiwgMHhhMiwgMHgzNCwgMHg4ZiwgMHgzNywgMHg2NywKLSAgMHhhYSwgMHhjMCwgMHgwOSwgMHgxNCwgMHhlOSwgMHg3NiwgMHhiNiwgMHg0OCwgMHg3YSwgMHhkNiwgMHhkNSwgMHhkMiwKLSAgMHhkMSwgMHgyYiwgMHhjYSwgMHg2NywgMHg0MywgMHg0OSwgMHg5MSwgMHg2ZCwgMHg3MywgMHg2MywgMHhkZiwgMHg1MCwKLSAgMHg0MCwgMHgyNSwgMHg5NSwgMHgxZSwgMHhlNSwgMHgyMSwgMHg2MCwgMHgwMSwgMHhlYSwgMHgzYywgMHhhNywgMHhjMCwKLSAgMHg1NCwgMHhjYywgMHgwNSwgMHg1YiwgMHgyZiwgMHg1MCwgMHhlMCwgMHg1ZCwgMHg5YSwgMHg2NSwgMHhhNywgMHhkMSwKLSAgMHhjOCwgMHgxZCwgMHg4YywgMHhiNywgMHgxMCwgMHgwYSwgMHg5YSwgMHhlNiwgMHgxZCwgMHg3NSwgMHhlMiwgMHg5NSwKLSAgMHg3OCwgMHgxZiwgMHg2NiwgMHhhYSwgMHg5YSwgMHhiNCwgMHhlYywgMHg4YiwgMHhlOSwgMHhkNSwgMHhiYiwgMHgyMywKLSAgMHhhZiwgMHhiMiwgMHgxMSwgMHg2OSwgMHhjMywgMHhlNCwgMHhkYywgMHhlZCwgMHg0YywgMHhhMSwgMHg5ZSwgMHg0NiwKLSAgMHgxMiwgMHhlOSwgMHg1MSwgMHg2OCwgMHg5MiwgMHg4NiwgMHhmYSwgMHg3MywgMHgyYywgMHhhNCwgMHhlOCwgMHg5ZSwKLSAgMHg1NCwgMHg5NSwgMHgyYiwgMHg0NywgMHhkMSwgMHg0OCwgMHgzNywgMHgzNCwgMHg0YywgMHg5MSwgMHgyYSwgMHg2YiwKLSAgMHhhOCwgMHg4ZiwgMHhlNSwgMHg1MSwgMHhhMiwgMHgyZiwgMHg0ZSwgMHg0ZiwgMHhiYSwgMHhjYSwgMHg3MSwgMHgyOSwKLSAgMHgzZSwgMHg3MSwgMHgxYywgMHhmYywgMHhmYywgMHhlOCwgMHg0MiwgMHg0NiwgMHg4NywgMHgzNywgMHhiZCwgMHgyNywKLSAgMHhhMiwgMHg0YSwgMHgwZiwgMHg1MiwgMHg0OCwgMHgxNiwgMHhlYiwgMHg4ZCwgMHhhMSwgMHhkNiwgMHhkNiwgMHhkYiwKLSAgMHg4OSwgMHgwYSwgMHg0MiwgMHg4MSwgMHgwYSwgMHg0YSwgMHhiYSwgMHg4MiwgMHgwZiwgMHg4MSwgMHgwNywgMHhlYywKLSAgMHhhNSwgMHgxNywgMHgzMSwgMHgxYywgMHgyNiwgMHhjYSwgMHg5MSwgMHg3MSwgMHg5ZSwgMHhkYiwgMHg0OCwgMHg4ZCwKLSAgMHgwYywgMHg3MywgMHgzNiwgMHg2NywgMHhjYiwgMHg1MiwgMHhkOCwgMHg4YywgMHgzZCwgMHgyMSwgMHgyYiwgMHg1NywKLSAgMHgyMiwgMHg0MCwgMHhmMCwgMHgzZSwgMHgxNSwgMHg1NSwgMHhiZiwgMHgyNCwgMHhmOCwgMHg3OCwgMHgzNCwgMHg1ZCwKLSAgMHgyYywgMHhmMiwgMHgzMiwgMHg3ZSwgMHgxNCwgMHg1YywgMHhlYywgMHg5MywgMHgxNiwgMHhhOSwgMHgyZSwgMHhiZiwKLSAgMHgxOSwgMHhlNiwgMHhhMiwgMHhjOCwgMHg3NCwgMHhlZiwgMHhiNiwgMHhlNSwgMHgyNCwgMHhjNywgMHg3OCwgMHhmYSwKLSAgMHg0OSwgMHhkMywgMHg2YSwgMHgyNywgMHhjNCwgMHhlZSwgMHg5MCwgMHg3ZiwgMHgwNiwgMHg3YiwgMHhmYSwgMHhhYywKLSAgMHhlZSwgMHgxZSwgMHgxYywgMHg1YywgMHgxNCwgMHhhNCwgMHhiMCwgMHhhOCwgMHgyZCwgMHhkZSwgMHhiMSwgMHhiNSwKLSAgMHhhYywgMHhlYywgMHgzOSwgMHgwMSwgMHhmMSwgMHhjYywgMHhhNiwgMHgwMSwgMHgzZCwgMHhlYSwgMHg2NSwgMHhjMiwKLSAgMHhhNCwgMHg3YywgMHhkZSwgMHg1ZiwgMHgwYSwgMHg3MywgMHhiOSwgMHhlNiwgMHgxNywgMHgzYywgMHg5YSwgMHgwMywKLSAgMHhmMCwgMHg3OCwgMHg3ZCwgMHg2NywgMHg5MywgMHgzMCwgMHgzYywgMHg5MiwgMHhkMiwgMHg2ZiwgMHgzMiwgMHg0NywKLSAgMHg2MSwgMHgwOSwgMHhhZCwgMHg4MiwgMHgzOSwgMHhkMCwgMHg1NSwgMHhlNywgMHhiYiwgMHhhZiwgMHhkNCwgMHg0OSwKLSAgMHgxZSwgMHhiYSwgMHg4MiwgMHhlMiwgMHhhNiwgMHgwZiwgMHgyYSwgMHhkMSwgMHhjMywgMHhmYywgMHg3NiwgMHhmMywKLSAgMHg4OCwgMHgyMywgMHhiNywgMHhjOCwgMHg3MCwgMHgyNiwgMHhkMSwgMHgyMiwgMHhkZiwgMHhlNiwgMHhmMiwgMHhhYSwKLSAgMHg2MywgMHgwZCwgMHhhMCwgMHgyNiwgMHg0MywgMHgwYSwgMHhkNywgMHhmNSwgMHg4ZCwgMHg4MywgMHhkMywgMHhmNCwKLSAgMHg4MCwgMHhmNCwgMHhkMSwgMHhhYywgMHgxMiwgMHg5ZSwgMHg0YiwgMHg4NCwgMHg1MSwgMHg1MCwgMHhmOCwgMHg2ZSwKLSAgMHg0NSwgMHg2YywgMHhjYSwgMHhmMSwgMHg2YiwgMHg2ZSwgMHg0NywgMHg2NywgMHg3YiwgMHhiNiwgMHg4MSwgMHg3MSwKLSAgMHg4ZSwgMHg5OSwgMHgwYywgMHhhYiwgMHg1ZCwgMHg3NCwgMHhhMSwgMHhkYywgMHg0NywgMHg4MiwgMHg4MSwgMHhlOCwKLSAgMHg0NywgMHg4MSwgMHgwNiwgMHhhNiwgMHgwNywgMHg3NSwgMHg0MSwgMHgyMSwgMHg0NSwgMHgxNCwgMHg1MCwgMHgwOCwKLSAgMHg5YywgMHg2NSwgMHhmOCwgMHg5NiwgMHgyZiwgMHhmYiwgMHg1NSwgMHg2YywgMHhmYiwgMHhmMSwgMHg0ZiwgMHg3NCwKLSAgMHg4OSwgMHhjNiwgMHg1ZiwgMHg4OSwgMHg2MiwgMHhmZiwgMHgwMCwgMHhiNSwgMHg1NiwgMHhjZiwgMHhiZiwgMHgxNCwKLSAgMHhmNywgMHg0MCwgMHgxZSwgMHgyNiwgMHg4YSwgMHhmMiwgMHhhNCwgMHg4MiwgMHg3YSwgMHhkMSwgMHg0YywgMHgxMSwKLSAgMHg5NCwgMHg3YSwgMHhhNCwgMHhjZSwgMHgzOCwgMHhmYywgMHg4ZiwgMHg2NSwgMHhkZiwgMHg0NCwgMHg0OCwgMHhmZSwKLSAgMHgwMywgMHg0ZSwgMHg3NCwgMHg5OSwgMHhjNywgMHgxZiwgMHg5MSwgMHhlYywgMHhiYiwgMHhlOCwgMHg4OSwgMHgxZiwKLSAgMHhjMCwgMHg2OCwgMHg0OCwgMHhkZiwgMHgxYiwgMHhlMiwgMHhlZCwgMHhmYywgMHhjMSwgMHhmNiwgMHg1NiwgMHhjYSwKLSAgMHhkNywgMHgxYiwgMHhlMiwgMHhlZCwgMHhmYywgMHhjMSwgMHhmNiwgMHg1NiwgMHhjYSwgMHgwMCwgMHhhOCwgMHhkYywKLSAgMHhhMiwgMHhjOSwgMHg2ZSwgMHhjOCwgMHhmMSwgMHhlOSwgMHhmNiwgMHgxYiwgMHhiMywgMHgwMSwgMHhmOCwgMHgzNywKLSAgMHgwNiwgMHgxNywgMHgxZCwgMHhmNiwgMHhjZiwgMHg4YSwgMHg1NCwgMHgzNCwgMHg3NSwgMHhlOCwgMHgyMywgMHhiYywKLSAgMHgxZiwgMHgwMiwgMHgwNSwgMHg0OSwgMHg1MCwgMHg0NiwgMHhlOCwgMHgwZiwgMHg5OCwgMHg3OCwgMHgzNSwgMHgxMiwKLSAgMHhlNSwgMHg2YSwgMHhiZiwgMHg1YywgMHg3MSwgMHhhOSwgMHgzMSwgMHg1ZSwgMHg5MywgMHg5NiwgMHhlMywgMHg0YiwKLSAgMHg2YSwgMHhkMywgMHgzZSwgMHg0MCwgMHg5MiwgMHgxYiwgMHg0YywgMHg4YiwgMHg1YSwgMHg1MiwgMHhiNywgMHgyMSwKLSAgMHg0ZCwgMHhlNCwgMHg1NywgMHhlNSwgMHg1NCwgMHgwMSwgMHg0YiwgMHg1YiwgMHgxZCwgMHg0MiwgMHg0YSwgMHg3YiwKLSAgMHhmNSwgMHhkMiwgMHhlYSwgMHhiZSwgMHg2MywgMHgxNiwgMHg2YywgMHhlNiwgMHgxNCwgMHgyYiwgMHg4NCwgMHg5NywKLSAgMHgyNiwgMHg0MSwgMHhiOSwgMHg0NSwgMHg0YSwgMHg5MCwgMHhjNCwgMHhlOCwgMHgyZiwgMHg3NiwgMHg2ZiwgMHhiNCwKLSAgMHgxNSwgMHhhZSwgMHg2NCwgMHg4MywgMHhkNSwgMHgyYSwgMHg0MiwgMHhiNCwgMHgwZiwgMHgyYSwgMHg4MiwgMHg4NywKLSAgMHhhYiwgMHg3NSwgMHg1YiwgMHhmZSwgMHgxNCwgMHg1NiwgMHgyNCwgMHg1OCwgMHhlNiwgMHhjMCwgMHhlMiwgMHg5YywKLSAgMHg3OCwgMHg0YiwgMHg5NSwgMHgxMiwgMHgyYiwgMHgwYSwgMHhiNCwgMHhlNSwgMHgzMSwgMHg1YSwgMHhlOCwgMHhhOSwKLSAgMHg3NiwgMHhiNywgMHhjZiwgMHgyYSwgMHg5NSwgMHhkMywgMHhiZCwgMHg0ZCwgMHgyOCwgMHhmMywgMHgyNywgMHhkMCwKLSAgMHg0OCwgMHgzZSwgMHgxNSwgMHhkNywgMHhjMywgMHhlYywgMHhlMiwgMHg0ZCwgMHg5YiwgMHgxYSwgMHhiZCwgMHg0MywKLSAgMHg5ZSwgMHhmYSwgMHgyNywgMHhkYywgMHhlZCwgMHgyYywgMHgzNywgMHhkOSwgMHhiZSwgMHgwZSwgMHhkMSwgMHgzZCwKLSAgMHgwZSwgMHgyNCwgMHgyYSwgMHgyNCwgMHg5MCwgMHg3ZiwgMHg0NSwgMHhkNCwgMHhhOSwgMHgzYywgMHhjNywgMHhjMSwKLSAgMHg0MSwgMHg2MCwgMHhmNywgMHg1NiwgMHhmMSwgMHg3YSwgMHhhMSwgMHg4ZSwgMHhhYiwgMHg3NCwgMHg2MywgMHgzNSwKLSAgMHg4OSwgMHhlNywgMHhhMywgMHhkOSwgMHg5MywgMHgzMywgMHhiOCwgMHg2NywgMHg4YSwgMHg0MiwgMHgzZSwgMHhlYiwKLSAgMHhmMSwgMHgxMywgMHgyNSwgMHg5NSwgMHg3ZSwgMHg2NSwgMHhhZSwgMHg4ZCwgMHhiNSwgMHg3MywgMHg3NSwgMHhiNiwKLSAgMHgyMiwgMHgwZCwgMHgxZCwgMHg4ZCwgMHhiMCwgMHhkOCwgMHg0YSwgMHg1ZCwgMHg1NywgMHg0MSwgMHhmMCwgMHhmOSwKLSAgMHhiNiwgMHg3YiwgMHg4NSwgMHg0YiwgMHhmZiwgMHgwMCwgMHgyOSwgMHg2ZSwgMHgwZSwgMHg0NiwgMHg2YSwgMHhkZiwKLSAgMHg4OCwgMHg2MywgMHhlZCwgMHg1YiwgMHhhMSwgMHgyNSwgMHgwMSwgMHgyYywgMHgzYiwgMHgzNSwgMHg4MiwgMHhkZSwKLSAgMHg5MywgMHhlMSwgMHhkOSwgMHhjNSwgMHg0ZSwgMHg5NSwgMHhhZCwgMHg3NywgMHg3MywgMHg5NiwgMHhmZCwgMHg4NCwKLSAgMHg1NywgMHhhNCwgMHg1YSwgMHg2MiwgMHhjMSwgMHg3MSwgMHhlOSwgMHhmNywgMHgyOSwgMHg0ZCwgMHhiZCwgMHgzNiwKLSAgMHgxYiwgMHg3YywgMHhmMywgMHhhZiwgMHg1MiwgMHg5MiwgMHgxNCwgMHhlMywgMHg2NywgMHg1YiwgMHgyOCwgMHg2NCwKLSAgMHgxMSwgMHhlZiwgMHg2OSwgMHhlYiwgMHhhMCwgMHg5NCwgMHhmNSwgMHhmNCwgMHhlYywgMHg5ZCwgMHg5ZSwgMHhhZiwKLSAgMHg3NiwgMHhjZCwgMHhhMiwgMHgzYSwgMHgxZSwgMHg4ZCwgMHg4ZiwgMHhhZCwgMHhhNiwgMHgxZCwgMHg1OCwgMHg0YiwKLSAgMHg2NiwgMHg2NCwgMHhhNCwgMHhiMywgMHgyNiwgMHg1YSwgMHhiZiwgMHg0NSwgMHhhNiwgMHhjOCwgMHgyNSwgMHg0YSwKLSAgMHhkMCwgMHhlOCwgMHg5NSwgMHgxNCwgMHgxMywgMHhkMCwgMHg1NSwgMHg1YSwgMHg0YiwgMHg3OSwgMHhlZSwgMHhjOSwKLSAgMHg1OSwgMHg3YiwgMHg0MywgMHg2NCwgMHgyOSwgMHhkZCwgMHhlMiwgMHgzOSwgMHhlZSwgMHhhYSwgMHhmZiwgMHgwMCwKLSAgMHg5NCwgMHg4OCwgMHhiOCwgMHhkZSwgMHhhNSwgMHg0NCwgMHg2MCwgMHhjYywgMHg3MSwgMHg1MiwgMHg5MiwgMHg4NCwKLSAgMHhiMCwgMHhjMywgMHg2OSwgMHgwMywgMHg2NSwgMHhhNiwgMHhjZSwgMHg5YiwgMHgyYSwgMHhlYSwgMHgzNSwgMHhhMCwKLSAgMHhhNSwgMHg2YywgMHhlOCwgMHhhOCwgMHgxYSwgMHg2NywgMHhjYSwgMHhkOSwgMHg2NiwgMHhkMiwgMHg2YywgMHgxOSwKLSAgMHgzMywgMHgyYiwgMHg3NSwgMHhkOCwgMHhkMCwgMHhhNCwgMHgyOCwgMHg0OCwgMHg1MiwgMHg5NCwgMHg0ZSwgMHhkYSwKLSAgMHg3ZCwgMHgwMSwgMHgxYywgMHhlNywgMHg3ZCwgMHhjMCwgMHgxZSwgMHhjZiwgMHhkOCwgMHgzNywgMHg0YywgMHhhMSwKLSAgMHg5YiwgMHgyZSwgMHg0NSwgMHgwYSwgMHgxNSwgMHhjMSwgMHg3MSwgMHhhMywgMHg0ZiwgMHg4ZSwgMHg0MiwgMHg1ZiwKLSAgMHg4YywgMHhiNywgMHgxYiwgMHhlNiwgMHhkNiwgMHhjNywgMHg0MiwgMHgzNywgMHhkZCwgMHhlMSwgMHhmNSwgMHg4ZiwKLSAgMHg1NSwgMHg3MywgMHhiMSwgMHg3NSwgMHhjNywgMHhiMiwgMHgzNiwgMHhhNCwgMHhkYSwgMHhkYiwgMHg5YiwgMHgxYSwKLSAgMHg2MiwgMHg1NiwgMHgxYywgMHg2MSwgMHhmOCwgMHhlYSwgMHgyNCwgMHgyOSwgMHg0MCwgMHgxMiwgMHg4NSwgMHg4ZSwKLSAgMHg1MywgMHhhMywgMHhhZSwgMHhmMSwgMHhiMSwgMHhkMywgMHhhZSwgMHhmNywgMHhkZCwgMHg1MSwgMHhlZiwgMHgxZiwKLSAgMHgxZCwgMHgwYiwgMHg3YiwgMHhiNSwgMHhmMywgMHgzNSwgMHg2NSwgMHhhZCwgMHgxNiwgMHhmMSwgMHgwNywgMHhkMywKLSAgMHgwNSwgMHhhNywgMHg1NCwgMHhjYiwgMHg2MSwgMHgwYiwgMHg1YiwgMHg0YywgMHgxMywgMHhjZSwgMHhlMywgMHgwMSwKLSAgMHg2OSwgMHg1MywgMHg4OSwgMHg0ZSwgMHg4ZSwgMHhmNiwgMHg1MCwgMHgxNCwgMHgwNiwgMHhiYiwgMHhmNywgMHhhYSwKLSAgMHg4ZiwgMHg1ZSwgMHg2MSwgMHg4YiwgMHg1YiwgMHg2MSwgMHgzNSwgMHgxNiwgMHhjZCwgMHhkOSwgMHhjYSwgMHg1YiwKLSAgMHg5ZiwgMHg5MCwgMHg4NywgMHgwMSwgMHhhZSwgMHg2NSwgMHhhZiwgMHhkOCwgMHg4NCwgMHg4ZCwgMHg4ZiwgMHg2ZSwKLSAgMHhiNSwgMHhlOSwgMHhhZSwgMHg0YywgMHg1YSwgMHhmNiwgMHhhYiwgMHgwMywgMHhlYSwgMHhjNSwgMHg2ZiwgMHhjZiwKLSAgMHhhYiwgMHhiNiwgMHg4YSwgMHg3OSwgMHgyMiwgMHhiZSwgMHhiZSwgMHg4YSwgMHg5MCwgMHhjZiwgMHhlNiwgMHgyYywKLSAgMHgwZiwgMHgxZSwgMHg5ZCwgMHgwZSwgMHhiYiwgMHg5NCwgMHgxNSwgMHhlMSwgMHg1ZSwgMHhhZSwgMHgzOSwgMHhkZSwKLSAgMHgyMywgMHg2NCwgMHg5YywgMHhiOCwgMHgxNiwgMHhlNiwgMHgwYywgMHhmYiwgMHhiMywgMHhiZSwgMHg3MSwgMHg4NSwKLSAgMHg2YywgMHg4ZCwgMHhkYSwgMHg0OCwgMHg3MCwgMHhmYSwgMHg1NCwgMHg5NCwgMHg4ZCwgMHg4ZiwgMHg2YSwgMHhiNCwKLSAgMHgzZCwgMHg3NSwgMHgzYSwgMHgxZSwgMHg3MywgMHg4YywgMHg5NCwgMHhkNiwgMHhiMSwgMHg4YywgMHhlMCwgMHg5YiwKLSAgMHhjMSwgMHgyMSwgMHg0ZiwgMHg4MywgMHg4ZiwgMHhhMSwgMHg4YiwgMHg4MywgMHg2ZCwgMHhiMCwgMHhiMiwgMHhmMywKLSAgMHhhZSwgMHgzNywgMHgxOSwgMHgyNywgMHg3ZCwgMHg4MywgMHg2YSwgMHg1OSwgMHg1MiwgMHg1YiwgMHgyNywgMHhjNywKLSAgMHg0MCwgMHhlYiwgMHhkNSwgMHhkZCwgMHhkNywgMHg1YiwgMHhhNSwgMHhiNywgMHgzMywgMHgyYywgMHg3YiwgMHgwZiwKLSAgMHhiYSwgMHhkZiwgMHgyMCwgMHg1YywgMHgyNywgMHhiNCwgMHhkYiwgMHgxZSwgMHg1NCwgMHg4OSwgMHgxMSwgMHg5YSwKLSAgMHg0OSwgMHgyYiwgMHg1YSwgMHg5NCwgMHhmMiwgMHg3NiwgMHhiNCwgMHgyMSwgMHgyMywgMHg2NSwgMHg0NywgMHg5YywKLSAgMHgyOSwgMHg1YSwgMHg0OCwgMHgyNywgMHhjZiwgMHhlZSwgMHhhMCwgMHhjMywgMHhlMiwgMHgwZSwgMHg1NywgMHhiMywKLSAgMHgzYSwgMHg0MywgMHg3OCwgMHg3ZCwgMHhhZCwgMHg0NywgMHg1ZCwgMHg4MywgMHgxYywgMHg5MiwgMHgyNywgMHgzOCwKLSAgMHg5ZiwgMHg1YSwgMHhiYSwgMHhiNiwgMHhkNywgMHg0ZCwgMHg3OCwgMHgzOCwgMHg3ZSwgMHg2ZCwgMHg2ZSwgMHhiNSwKLSAgMHhjMSwgMHhjMSwgMHhiMCwgMHhjNywgMHgxZiwgMHg1NSwgMHhhZCwgMHhhNiwgMHgyNywgMHg1ZiwgMHg1NiwgMHhiNCwKLSAgMHhiNCwgMHhlMSwgMHhmMiwgMHg4NCwgMHhjOCwgMHhiOCwgMHg0OCwgMHg1YSwgMHhiNCwgMHg5MCwgMHg5NSwgMHgyOSwKLSAgMHg2NywgMHg5OCwgMHgwZSwgMHg4MywgMHhhNiwgMHhjMiwgMHg0MCwgMHhmMCwgMHgwMCwgMHg1NSwgMHg3MCwgMHg5NywKLSAgMHgyNSwgMHhkMywgMHg2YywgMHhkNiwgMHg5YiwgMHhkNiwgMHg3ZiwgMHg5MiwgMHg5MSwgMHhlZSwgMHgyNSwgMHg4ZCwKLSAgMHhiYywgMHg3NiwgMHgwYSwgMHhiYiwgMHhhNiwgMHg1ZSwgMHgwMSwgMHhlZCwgMHg4OCwgMHhmMSwgMHhlNSwgMHg4ZSwKLSAgMHg5MywgMHhjZCwgMHhiZiwgMHg5ZSwgMHg1MSwgMHhlYywgMHhhZiwgMHg1ZiwgMHhjOCwgMHg5YiwgMHgwZCwgMHhiOSwKLSAgMHhhNSwgMHhkZiwgMHhmMywgMHg0YiwgMHg4YiwgMHhkOSwgMHgwYywgMHg5OCwgMHhhOSwgMHhlZCwgMHg5NCwgMHhmZCwKLSAgMHhjZiwgMHhjZSwgMHg2MSwgMHg5MiwgMHgzYywgMHg1YiwgMHg2MCwgMHgwZSwgMHg0MCwgMHg3YywgMHgwNywgMHg0NSwKLSAgMHgyYiwgMHg3ZCwgMHhjNCwgMHg5YSwgMHhkZCwgMHg5NCwgMHg1YywgMHhmMiwgMHhmOCwgMHg1NiwgMHhhNSwgMHg0YiwKLSAgMHg5MiwgMHhjNSwgMHhiYSwgMHgwYywgMHgyNywgMHg1NiwgMHhkYiwgMHg3MiwgMHgxZCwgMHg4ZSwgMHhlYSwgMHg5NiwKLSAgMHhmNCwgMHgwNiwgMHg5NCwgMHhiNCwgMHg4NSwgMHgzYSwgMHg0OSwgMHgwMSwgMHgyYiwgMHhlNSwgMHg0OSwgMHgzZCwKLSAgMHhjMCwgMHg2OCwgMHhmNSwgMHhlYSwgMHgwNywgMHg1ZCwgMHgxMywgMHgyYywgMHgzOCwgMHhiZCwgMHhjOSwgMHgzMywKLSAgMHhhMCwgMHg1OSwgMHhkYywgMHg4YywgMHgzMiwgMHgwYiwgMHg3YiwgMHhlOCwgMHg3MCwgMHg0OSwgMHg5MiwgMHhiNSwKLSAgMHgzOCwgMHhmNywgMHg2ZSwgMHg4ZSwgMHg1NywgMHgwMCwgMHg3NSwgMHg0NCwgMHhmMywgMHgyOSwgMHhiMywgMHhlNiwKLSAgMHhlYywgMHg3NywgMHg2OCwgMHhmNCwgMHhkNywgMHg0YSwgMHg4ZCwgMHg0ZCwgMHhmMCwgMHg0YSwgMHg4YSwgMHg1YywKLSAgMHg5ZCwgMHg5NywgMHhmYywgMHg5NiwgMHhmZCwgMHgwZSwgMHhjMSwgMHgyNiwgMHhlNSwgMHgwZiwgMHgxOCwgMHg3MSwKLSAgMHg4OCwgMHhhOCwgMHg0OCwgMHg0YSwgMHgxYywgMHg5MiwgMHhmMCwgMHgwZSwgMHhiNCwgMHgwZSwgMHg4NywgMHg2YSwKLSAgMHhhNiwgMHg0MCwgMHgzZSwgMHg2MiwgMHg3NywgMHhjYywgMHg0MSwgMHg1MCwgMHg1NiwgMHg4MSwgMHhlOCwgMHgzYiwKLSAgMHhhYiwgMHg1MSwgMHhiNSwgMHhjMCwgMHg5OCwgMHg5OSwgMHg1ZSwgMHg0NSwgMHg5MiwgMHg0YywgMHg3YSwgMHhmYiwKLSAgMHgxNSwgMHg0MCwgMHgyNiwgMHg3MywgMHhiMywgMHgwOSwgMHg0YSwgMHgxZSwgMHhkMCwgMHg1MCwgMHg0ZiwgMHg2NiwKLSAgMHg5MiwgMHgxYiwgMHhlNCwgMHgzYiwgMHgwMCwgMHhhNCwgMHgyNCwgMHg3NCwgMHg1NywgMHhhYywgMHgxYSwgMHg4ZiwKLSAgMHhiNCwgMHg2NiwgMHgzNywgMHg3OSwgMHhlZSwgMHhiZSwgMHhiNSwgMHhkYSwgMHg0NCwgMHhmYiwgMHg3YywgMHhjOCwKLSAgMHhlOCwgMHg3ZCwgMHg4OCwgMHhlMSwgMHg4NywgMHg1MiwgMHhlMiwgMHgwMiwgMHhkYiwgMHg0ZSwgMHhkYSwgMHg1OSwKLSAgMHg1YSwgMHgwMywgMHgzZCwgMHg5ZSwgMHhmOSwgMHg4ZiwgMHg2OSwgMHhjZiwgMHhkNCwgMHgxZiwgMHg4MywgMHg1MCwKLSAgMHgzNiwgMHg5YiwgMHg2ZCwgMHhiZSwgMHgzYSwgMHhhMiwgMHhhNiwgMHhkZiwgMHg3ZiwgMHg4NSwgMHg3MiwgMHhiZCwKLSAgMHhiYSwgMHhjMywgMHg3MCwgMHhkNCwgMHhkNSwgMHhhOCwgMHgzNiwgMHg5OSwgMHgwZSwgMHgwNCwgMHhhMywgMHhiOCwKLSAgMHhiZSwgMHg0OSwgMHhmNywgMHhiNCwgMHg4NCwgMHhlYiwgMHhiNSwgMHhlNSwgMHhlNywgMHgwMCwgMHgwMCwgMHgxNSwKLSAgMHhiZCwgMHgyNiwgMHhhMywgMHg2NSwgMHhjOSwgMHgzYiwgMHhiZSwgMHgwZSwgMHgxZSwgMHgwYSwgMHhkYywgMHhjNiwKLSAgMHgyMywgMHg5ZiwgMHhjOSwgMHhjMiwgMHhkNiwgMHhkNywgMHg5MywgMHg1OCwgMHhmMiwgMHg2NCwgMHhiZCwgMHg3OSwKLSAgMHhjNywgMHhkYiwgMHhkZiwgMHg5YiwgMHgxOSwgMHhmMCwgMHhhMiwgMHgyNiwgMHhjMiwgMHgxZSwgMHgxYSwgMHg0YiwKLSAgMHg5YiwgMHg3MSwgMHgzYSwgMHhkNywgMHg5YSwgMHhhZiwgMHhlZSwgMHhiZCwgMHg1MywgMHhkYywgMHgyYSwgMHg5MiwKLSAgMHhlMywgMHgxZSwgMHgzMSwgMHg5MCwgMHhiYiwgMHg4NiwgMHg1YiwgMHgxYSwgMHg4MSwgMHgxMiwgMHhkNSwgMHgwYSwKLSAgMHhlMywgMHg2OCwgMHg5NCwgMHg4OSwgMHg1OCwgMHhmMywgMHhmMCwgMHhiOSwgMHhjZiwgMHg5MiwgMHg0YywgMHg0NywKLSAgMHhlNCwgMHhkYSwgMHg3MSwgMHg0YiwgMHhlYSwgMHhiNCwgMHgzYywgMHgzNiwgMHhkOSwgMHgzZSwgMHg2ZSwgMHhkNCwKLSAgMHhiNCwgMHhlYywgMHgxZCwgMHhlZSwgMHhhYywgMHg2ZSwgMHgxNiwgMHhlNiwgMHg1MCwgMHg3MywgMHhjYywgMHgxMiwKLSAgMHhkYiwgMHg5MywgMHhjMCwgMHg0MSwgMHg2NCwgMHg0OSwgMHg2ZiwgMHg0ZiwgMHhjNywgMHg1MSwgMHhmMywgMHhlMywKLSAgMHhiZSwgMHg5MywgMHhjYSwgMHhlMywgMHg0YSwgMHgxZSwgMHgwNSwgMHgyYSwgMHgwNCwgMHg3NSwgMHhmMCwgMHhkMSwKLSAgMHhmMSwgMHhhOCwgMHgyNCwgMHg2OCwgMHhhMiwgMHg4MSwgMHhkZCwgMHg0NSwgMHgwMCwgMHg4OSwgMHhjNiwgMHg1ZiwKLSAgMHg4OSwgMHg2MiwgMHhmZiwgMHgwMCwgMHhiNSwgMHg1NiwgMHhjZiwgMHhiZiwgMHgxNCwgMHhmNywgMHg0OCwgMHg5YywKLSAgMHg2NSwgMHhmOCwgMHg5NiwgMHgyZiwgMHhmYiwgMHg1NSwgMHg2YywgMHhmYiwgMHhmMSwgMHg0ZiwgMHg3NCwgMHgwNywKLSAgMHg5NSwgMHgxMCwgMHgwZiwgMHg1MiwgMHgwNywgMHhiNiwgMHg4YSwgMHhjOSwgMHgxZCwgMHg2OCwgMHhhOCwgMHgxOCwKLSAgMHg0NiwgMHg2OSwgMHgzMywgMHg4ZSwgMHgzZiwgMHgyMywgMHhkOSwgMHg3NywgMHhkMSwgMHgxMiwgMHgzZiwgMHg4MCwKLSAgMHhkMywgMHg5ZCwgMHgyNiwgMHg3MSwgMHhjNywgMHhlNCwgMHg3YiwgMHgyZSwgMHhmYSwgMHgyMiwgMHg0NywgMHhmMCwKLSAgMHgxYSwgMHg5MCwgMHgzNywgMHhjNiwgMHhmOCwgMHhiYiwgMHg3ZiwgMHgzMCwgMHg3ZCwgMHg5NSwgMHhiMiwgMHhiNSwKLSAgMHhjNiwgMHhmOCwgMHhiYiwgMHg3ZiwgMHgzMCwgMHg3ZCwgMHg5NSwgMHhiMiwgMHg4MCwgMHgyOCwgMHhhMiwgMHg4YSwKLSAgMHgwMywgMHg5NiwgMHhlOSwgMHgwYSwgMHgyZCwgMHhjYSwgMHhkZiwgMHgyNiwgMHhkZiwgMHgzOSwgMHg4NiwgMHhlNCwKLSAgMHg0NSwgMHg5NCwgMHhkMiwgMHg5OSwgMHg3ZCwgMHhhNSwgMHg4ZCwgMHhhNSwgMHhjNCwgMHgyOCwgMHgxMCwgMHhhNCwKLSAgMHg5MSwgMHhlOCwgMHgyMCwgMHg5YSwgMHhmOCwgMHhkYSwgMHhlMywgMHgwNiwgMHhlMSwgMHg4NCwgMHhjYSwgMHhiZCwKLSAgMHhlMiwgMHg3MywgMHgyNCwgMHgyZiwgMHhiNCwgMHhjNSwgMHhkOCwgMHg1NCwgMHg1NCwgMHhiYSwgMHhiNSwgMHgxMiwKLSAgMHg1ZSwgMHhiMSwgMHg0OSwgMHg3MywgMHg5ZSwgMHgyNCwgMHg4ZCwgMHhmOCwgMHhmOSwgMHgyYywgMHg5MiwgMHgwMiwKLSAgMHhiZiwgMHg1NSwgMHhjNSwgMHg3OCwgMHgyNiwgMHhiZSwgMHhkMywgMHhkNSwgMHg1MiwgMHhmZiwgMHgwMCwgMHg4NCwKLSAgMHhlZSwgMHgzZSwgMHhmYywgMHg3YiwgMHg3YywgMHgwZSwgMHgyNiwgMHg1YSwgMHgyMCwgMHgyNiwgMHg3NCwgMHhmYywKLSAgMHg2OCwgMHgzOCwgMHg5YiwgMHg4YywgMHgzMiwgMHg5ZSwgMHg2MSwgMHg3MCwgMHhiNSwgMHgzYSwgMHgzOSwgMHg2NSwKLSAgMHgzMCwgMHhhMSwgMHhlMywgMHhlNiwgMHg5MiwgMHhhMSwgMHhiZSwgMHhlZiwgMHgzOCwgMHhkNCwgMHhhNywgMHg4NywKLSAgMHg5MiwgMHgxYSwgMHhjYSwgMHhjMCwgMHhkOSwgMHg4ZiwgMHg0YywgMHg2MywgMHgzOSwgMHhjMCwgMHgyNCwgMHhiZCwKLSAgMHgwOSwgMHhjMSwgMHgxZCwgMHhlYiwgMHg4MywgMHg0NSwgMHg0NCwgMHhmNywgMHg4NiwgMHgyNCwgMHgwMCwgMHgwMiwKLSAgMHg5MCwgMHhhZiwgMHg1YSwgMHgxYywgMHg0OSwgMHgwNCwgMHg3YSwgMHgzZCwgMHhiYiwgMHhhOCwgMHgzNSwgMHhlMywKLSAgMHhmNywgMHg5NywgMHhlZiwgMHg4MiwgMHg1YSwgMHhlYywgMHg5NywgMHgxNywgMHhlZSwgMHg0YiwgMHg1ZiwgMHg5NCwKLSAgMHgxMSwgMHgyZSwgMHg3ZSwgMHhlZCwgMHhkMSwgMHg1ZCwgMHg1NiwgMHg4YSwgMHg5NiwgMHgwMCwgMHhmMywgMHg5NywKLSAgMHhhMiwgMHg5NCwgMHhlOSwgMHgxZCwgMHg3YSwgMHgwMSwgMHhiZCwgMHg2OCwgMHg2YSwgMHhiZCwgMHhlMCwgMHg2MiwKLSAgMHhlNywgMHg0MCwgMHg5OSwgMHgyZSwgMHhkOSwgMHg4OCwgMHhkZiwgMHgxOCwgMHgyZiwgMHhjNiwgMHhlYywgMHg5NiwKLSAgMHhjMywgMHg1MywgMHhiOSwgMHg5NSwgMHgxNiwgMHhlZiwgMHg2ZiwgMHg3NSwgMHgzYywgMHhkMCwgMHhlNCwgMHgxMiwKLSAgMHg5ZiwgMHgzOSwgMHhiNywgMHg4MywgMHg3ZSwgMHhmNCwgMHhhNywgMHgxMiwgMHgwZiwgMHg5YywgMHhjZSwgMHg5NCwKLSAgMHgwOSwgMHhkNiwgMHhhZCwgMHhlNywgMHhhNCwgMHg3MSwgMHg1NiwgMHg2YSwgMHhmYiwgMHgzNiwgMHhhZCwgMHgxOCwKLSAgMHhhNSwgMHhhOSwgMHgyNCwgMHhmNSwgMHg3ZCwgMHg3NywgMHgxNywgMHhhNCwgMHg5NCwgMHg4ZiwgMHg0OCwgMHg0MCwKLSAgMHg2NSwgMHgxYywgMHhjNywgMHhkYSwgMHhhNCwgMHhlZSwgMHhiNCwgMHg5NywgMHgzOSwgMHg3ZCwgMHg0YywgMHhhMywKLSAgMHhjNiwgMHgxNywgMHg0MSwgMHhhYSwgMHhkYiwgMHgxZSwgMHgyZCwgMHhhMiwgMHhkNCwgMHhkYywgMHg2NSwgMHhiYSwKLSAgMHhkYSwgMHg1MiwgMHg5ZSwgMHg3NSwgMHgyZCwgMHg0YSwgMHgzYywgMHhhOSwgMHgyYSwgMHg1MSwgMHgyYSwgMHg1YSwKLSAgMHhiNCwgMHg0ZiwgMHg0MCwgMHg0YSwgMHg5NCwgMHg3NSwgMHhlMSwgMHhiYSwgMHhhZSwgMHg2MywgMHhkZCwgMHhmMSwKLSAgMHg5NiwgMHhhZiwgMHgzMiwgMHgxNSwgMHg4MywgMHhkOCwgMHhlNiwgMHhlNSwgMHg3NywgMHgzMCwgMHhmMSwgMHgyZCwKLSAgMHhiYSwgMHhjZiwgMHg5ZCwgMHgxNiwgMHgwYSwgMHhiOSwgMHg3NCwgMHhhNCwgMHgyNiwgMHg0MiwgMHhmNCwgMHg5NiwKLSAgMHhkMiwgMHg3NiwgMHg0OSwgMHg0YSwgMHg0OSwgMHgzYiwgMHgyNywgMHhhNywgMHg4NSwgMHg0OCwgMHhkYywgMHhiMCwKLSAgMHhmYiwgMHg1YiwgMHg0ZCwgMHgwYiwgMHg4ZiwgMHgxMCwgMHhmMiwgMHg0OSwgMHg1NywgMHhkNCwgMHg5NSwgMHg4NCwKLSAgMHg4OCwgMHhlZSwgMHgwZSwgMHhjMiwgMHgyMiwgMHhkNSwgMHhkZSwgMHgxMiwgMHgyMywgMHhiNywgMHhiMiwgMHhlOSwKLSAgMHhlZiwgMHhmMywgMHg1NCwgMHg1NywgMHhiZiwgMHgwMSwgMHg1MiwgMHgzZiwgMHhjYSwgMHg0NCwgMHhmNiwgMHhmMCwKLSAgMHhmMSwgMHhiYiwgMHgwNSwgMHhhZCwgMHhjYiwgMHg1YywgMHg5NywgMHg1MCwgMHhhNSwgMHgyMywgMHhjYiwgMHg2MCwKLSAgMHhhOSwgMHhhNiwgMHg2MywgMHhjNiwgMHg0MCwgMHhmMywgMHg5ZCwgMHgwOCwgMHgxYSwgMHgwYSwgMHgxYiwgMHgyOSwKLSAgMHg0OCwgMHg0ZSwgMHhkMiwgMHg3NiwgMHhhMSwgMHhiZCwgMHg3OCwgMHhkMywgMHhiYSwgMHhiZCwgMHg0ZCwgMHgzMSwKLSAgMHgyNiwgMHg0NSwgMHg0ZSwgMHhjMywgMHgyZSwgMHhkOSwgMHgzNCwgMHg2NSwgMHg0YSwgMHhlMiwgMHg0ZCwgMHhmNiwKLSAgMHgzYiwgMHg3NiwgMHhlNCwgMHgxMiwgMHhiMywgMHg2YywgMHhiNywgMHg5ZSwgMHhjZCwgMHg4NCwgMHgyNywgMHhkMCwKLSAgMHhlNCwgMHg4NSwgMHgwZSwgMHhkMSwgMHg1ZCwgMHgzYiwgMHhjYSwgMHg3YiwgMHgzMSwgMHhlZCwgMHhlZiwgMHgzZCwKLSAgMHhiNiwgMHg2YiwgMHhiZSwgMHgxOSwgMHg4ZiwgMHg0MywgMHg2MiwgMHhjYiwgMHg4NSwgMHhkYiwgMHg2MSwgMHhmNiwKLSAgMHhkMiwgMHg1NywgMHhkOSwgMHhjNSwgMHg4OSwgMHgxNSwgMHhhZSwgMHhjMSwgMHgzMiwgMHgwOCwgMHgwNywgMHg2YiwKLSAgMHhlZCwgMHgwYSwgMHg3NCwgMHhiNCwgMHhhNCwgMHgwMiwgMHg1NCwgMHhiMSwgMHhjYywgMHg0MCwgMHgxZSwgMHgyNCwKLSAgMHhlYSwgMHg4YywgMHg4ZSwgMHhkYiwgMHgyZSwgMHgwYywgMHhiYiwgMHg0NCwgMHhlYiwgMHhjZCwgMHhkZCwgMHhmYiwKLSAgMHhiZCwgMHg5OSwgMHgxMiwgMHhkMiwgMHgyNywgMHgzNSwgMHgyMSwgMHhhNiwgMHg5MiwgMHg4NCwgMHgyZCwgMHg2MywKLSAgMHg5MSwgMHhhNywgMHgwMCwgMHg0YSwgMHg1MywgMHhlZiwgMHg2OSwgMHg1YSwgMHg4NiwgMHhjMSwgMHgyNywgMHg1ZCwKLSAgMHgwZiwgMHg4MSwgMHhkZiwgMHgxZSwgMHg1MywgMHg5NSwgMHg0NCwgMHhiOCwgMHg0MCwgMHg3NiwgMHgzMywgMHgzMSwKLSAgMHhhNiwgMHhiMSwgMHg5MCwgMHhkYiwgMHg2NiwgMHgwMywgMHgwOSwgMHhhNSwgMHhjNywgMHg1YiwgMHg2ZSwgMHgyOSwKLSAgMHhlMCwgMHhhMiwgMHgxMSwgMHhjYSwgMHgwOCwgMHgxYywgMHhlOCwgMHg3MSwgMHgzYywgMHhjMCwgMHg5NCwgMHhlZCwKLSAgMHgzYywgMHhhYSwgMHgzZCwgMHg3YywgMHg0MywgMHg3OSwgMHgwZCwgMHhhMiwgMHg3NSwgMHhlNSwgMHgwOSwgMHhjOSwKLSAgMHg5YSwgMHg4OSwgMHgwZCwgMHhkYiwgMHhlNCwgMHhmOCwgMHg2NiwgMHhjZSwgMHhmNCwgMHg5NCwgMHgyMiwgMHhlOCwKLSAgMHg4OCwgMHgwZCwgMHhhZCwgMHhiNSwgMHhiMCwgMHhjOSwgMHhkOCwgMHgxYSwgMHg3MCwgMHhhZiwgMHg2YSwgMHg2YywKLSAgMHhhOCwgMHhhNCwgMHgyYywgMHhlOSwgMHgyNywgMHg5NywgMHg2NCwgMHg2OCwgMHg2ZSwgMHhiOSwgMHhhZSwgMHg5MywKLSAgMHg3MSwgMHg2OSwgMHhmOCwgMHg5YywgMHg4OSwgMHhkNiwgMHgwMSwgMHgxMiwgMHgxNCwgMHg4YiwgMHgxMywgMHg4ZSwKLSAgMHg0OCwgMHg4ZSwgMHg1MiwgMHhjMCwgMHg0MCwgMHg4YSwgMHhmYiwgMHg2NCwgMHg4MiwgMHg4NSwgMHg4MCwgMHgwNywKLSAgMHgyZiwgMHgzZSwgMHg4YSwgMHgwYSwgMHg0ZiwgMHg1MiwgMHgxNSwgMHhkMCwgMHg2ZiwgMHhiYiwgMHg4NiwgMHhlYSwKLSAgMHhhYywgMHg5NSwgMHg4NCwgMHg1YywgMHgyZCwgMHhkNywgMHg2OSwgMHg2NiwgMHgzNCwgMHgyYiwgMHhhMiwgMHhkZCwKLSAgMHg2OSwgMHg4NCwgMHgzZSwgMHhkYSwgMHgxZCwgMHg3ZCwgMHgyOCwgMHg3MywgMHg3YywgMHhlOCwgMHg2OSwgMHhiNiwKLSAgMHg0YSwgMHg5NCwgMHhmOSwgMHgwMSwgMHg1ZSwgMHg2YSwgMHg5NCwgMHgxMSwgMHhjYiwgMHhkMywgMHg5OCwgMHgyYSwKLSAgMHhiMywgMHgxZSwgMHgxZCwgMHhhZSwgMHg0NSwgMHhmYiwgMHhjOCwgMHhhMCwgMHg0NCwgMHg5OSwgMHg2ZCwgMHg3ZCwKLSAgMHg5NiwgMHhkMywgMHgzMSwgMHhmNywgMHhlZiwgMHgyMSwgMHg2ZSwgMHg5NiwgMHg5MywgMHhiZSwgMHg0NiwgMHhkNCwKLSAgMHhkMiwgMHgxNiwgMHhhMiwgMHg4NSwgMHgyYywgMHhlOCwgMHg4MCwgMHhiMywgMHhjZCwgMHhjOSwgMHhhMCwgMHgzNSwKLSAgMHhkNCwgMHgwYSwgMHg2MSwgMHgyZiwgMHg1MiwgMHg3NywgMHg2MSwgMHg3NSwgMHhiYywgMHg2NSwgMHgyYiwgMHg1ZCwKLSAgMHhkMywgMHhjYSwgMHg5YiwgMHg0NCwgMHgzYiwgMHgyYywgMHhiNiwgMHg4OCwgMHg2ZiwgMHhkZCwgMHg1NiwgMHg5MiwKLSAgMHhjOCwgMHg4YywgMHhiNSwgMHhhMywgMHg0NSwgMHgyYSwgMHg1ZiwgMHgzMSwgMHhlZCwgMHg1MywgMHhkZSwgMHg1MiwKLSAgMHg4NCwgMHgyNywgMHg5OCwgMHg5NSwgMHgwMCwgMHg0OCwgMHhkMSwgMHhhZCwgMHg0NiwgMHhkOSwgMHg2OSwgMHg5NywKLSAgMHgzOSwgMHg4YiwgMHg1NSwgMHhhZCwgMHhlOSwgMHg5NywgMHg1YiwgMHg5YiwgMHhkMSwgMHg0MiwgMHg1YywgMHg2ZSwKLSAgMHhlMCwgMHhjOSwgMHg4OCwgMHg4MywgMHgxZCwgMHhiNCwgMHg4MCwgMHgxNiwgMHhmMiwgMHg4MywgMHg0OSwgMHg3OSwKLSAgMHhlNiwgMHhiOSwgMHg4OCwgMHhkMiwgMHgwYSwgMHg5NCwgMHg5MiwgMHg0OSwgMHgwNCwgMHhlOCwgMHgxMywgMHg1ZCwKLSAgMHhmOSwgMHg2MywgMHhiMywgMHgzMSwgMHgwYiwgMHg5NCwgMHg0YywgMHg4ZSwgMHg1MiwgMHhlZSwgMHgzOSwgMHgyYywKLSAgMHgzNCwgMHgzMiwgMHhiNiwgMHg2MywgMHhiNiwgMHhlMiwgMHg1YSwgMHgyZSwgMHhiMCwgMHhmYSwgMHg4OCwgMHgyNSwKLSAgMHg2OSwgMHhlNSwgMHg0YSwgMHg3NiwgMHgxNCwgMHg4NCwgMHhhYiwgMHg0MywgMHg1YiwgMHhkOCwgMHhlNSwgMHgxZiwKLSAgMHgwYywgMHgwMSwgMHg5YywgMHg5ZSwgMHhlOSwgMHg2ZSwgMHhjYSwgMHhkOCwgMHhiNiwgMHgyYiwgMHgxZSwgMHg5MSwKLSAgMHgyZCwgMHgzNywgMHg0OSwgMHgwYSwgMHg1MywgMHgyOSwgMHhlYywgMHg0MiwgMHg5OCwgMHg3ZCwgMHgzMSwgMHg1YywKLSAgMHgxYywgMHhhZSwgMHhhOSwgMHg1YiwgMHg0ZSwgMHhkYiwgMHg0OCwgMHhlOCwgMHhhMCwgMHhhNSwgMHgyNywgMHhlMSwKLSAgMHhhMSwgMHgzYSwgMHgwNywgMHhiOCwgMHhjNiwgMHhmZCwgMHg0MiwgMHhjNiwgMHg3NiwgMHgzZCwgMHg2NCwgMHhiMSwKLSAgMHhlNywgMHg1YSwgMHg5NywgMHg2YiwgMHgzOSwgMHgxYywgMHg4NiwgMHg2ZSwgMHg3OCwgMHhmMiwgMHgwYSwgMHhjYSwKLSAgMHhlMywgMHg0NiwgMHg4MywgMHhjOCwgMHg5MCwgMHhmMiwgMHg1MywgMHhlZiwgMHgyZCwgMHhmMiwgMHgwMiwgMHg3OSwKLSAgMHg5YiwgMHgyNCwgMHgxMCwgMHg5MCwgMHg3ZiwgMHgzZiwgMHg5MCwgMHg3NSwgMHhkZSwgMHg4NiwgMHhjYywgMHhjMiwKLSAgMHhlMywgMHg2OCwgMHhiYiwgMHhlMSwgMHhkMCwgMHgzMiwgMHg0YiwgMHgxYywgMHg5NCwgMHhhMiwgMHg1MywgMHgwYiwKLSAgMHg0NywgMHhiOSwgMHg2ZSwgMHhiMiwgMHgwMCwgMHg1MiwgMHhjYiwgMHhiYSwgMHg0YSwgMHg5OSwgMHg2ZiwgMHhhNiwKLSAgMHg5NCwgMHg1NCwgMHg5ZCwgMHg4ZCwgMHgwZSwgMHhlMiwgMHg5MCwgMHg3ZiwgMHgzNiwgMHhiOCwgMHgxNSwgMHhlZSwKLSAgMHhkMCwgMHhiNiwgMHg0NCwgMHhjNywgMHhmMiwgMHg4OSwgMHg2ZSwgMHgyYywgMHg5NSwgMHgzNiwgMHhlYiwgMHhhZCwKLSAgMHgzNiwgMHg4MCwgMHhmNCwgMHhkOSwgMHg0OSwgMHg2ZCwgMHg2MSwgMHg0OSwgMHgwOSwgMHg2ZCwgMHg5MSwgMHhjYSwKLSAgMHhkYSwgMHg3NiwgMHg5ZCwgMHgxNSwgMHg5MiwgMHg3NywgMHhhMywgMHhkMSwgMHgzNSwgMHhkNywgMHg2MCwgMHhiNywKLSAgMHhjNCwgMHhiOSwgMHhkZSwgMHg2NCwgMHgyZSwgMHhjMCwgMHhjMywgMHhiNiwgMHg1NiwgMHhlMCwgMHgxNCwgMHhjNywKLSAgMHg3YSwgMHg0YywgMHhiNiwgMHhkNCwgMHhlYywgMHhkNCwgMHhiOCwgMHhhNCwgMHhmMywgMHgyOSwgMHhhNiwgMHhiYiwKLSAgMHg2MiwgMHhhMCwgMHhkMiwgMHg0MiwgMHg1NCwgMHgzNiwgMHg0MCwgMHgyMSwgMHg1YiwgMHhlOSwgMHhkMCwgMHg2YywKLSAgMHhiNiwgMHg1YywgMHgwNSwgMHg5NywgMHhjOSwgMHhjZiwgMHg3MCwgMHg5OSwgMHg3YiwgMHg3YiwgMHgxMSwgMHgzNiwKLSAgMHg1YywgMHg5MSwgMHhmNCwgMHhjNiwgMHgwZiwgMHgyOCwgMHgzMywgMHgyNiwgMHhlNSwgMHgzNSwgMHg4NiwgMHhlMywKLSAgMHgwMiwgMHg5MCwgMHhhMSwgMHhkMSwgMHg5NiwgMHg1MCwgMHhlMiwgMHhkNCwgMHhiNywgMHg0OCwgMHgxYiwgMHgwNCwKLSAgMHgxMCwgMHg5MCwgMHg0MSwgMHgyMywgMHhjMCwgMHg1MiwgMHhkZSwgMHgzOSwgMHg3OCwgMHhiNywgMHhlMSwgMHg3YywKLSAgMHg3NCwgMHg5MCwgMHhjNCwgMHgwOSwgMHgwNCwgMHhlMiwgMHhkOSwgMHhkMywgMHhjYSwgMHg1YSwgMHgwMiwgMHg5MCwKLSAgMHhhNCwgMHgwOCwgMHg3NywgMHg5NCwgMHgyNCwgMHgxNywgMHgxMCwgMHg1MiwgMHhhZCwgMHgyOSwgMHgzZCwgMHhiMywKLSAgMHg3YSwgMHg1ZiwgMHg1MSwgMHhkNSwgMHg0MywgMHhhNywgMHg0YSwgMHg2ZCwgMHg5YywgMHg2ZSwgMHgxOCwgMHhiZSwKLSAgMHg0NywgMHgzMiwgMHg1MywgMHhkMSwgMHg2NSwgMHhlNCwgMHhkMiwgMHg5YywgMHg4YSwgMHg4NSwgMHhkYSwgMHg4MiwKLSAgMHg4YiwgMHg2MSwgMHhmZSwgMHhjNSwgMHhiNSwgMHg3ZiwgMHgzOCwgMHg2OSwgMHgwNCwgMHgwNCwgMHg4MiwgMHhlMCwKLSAgMHgwYiwgMHhlNywgMHgxZCwgMHgwMSwgMHg1OCwgMHhkMiwgMHg3NiwgMHg0YSwgMHg3NywgMHg0YiwgMHhmYywgMHg1MSwKLSAgMHhiMSwgMHg0NiwgMHhlMiwgMHg0YywgMHg2NiwgMHhhZCwgMHhiNiwgMHhjNCwgMHg0OSwgMHhiNywgMHhkZSwgMHgyNSwKLSAgMHhjNSwgMHg1MywgMHhlYSwgMHhlZCwgMHgxMSwgMHhjYSwgMHhlYywgMHgwNywgMHg5OSwgMHhkYiwgMHg5MSwgMHgyNCwKLSAgMHgzOCwgMHgwMSwgMHhmMywgMHgxNiwgMHgxYywgMHhmMywgMHg0MCwgMHhlZiwgMHg1MiwgMHgxZCwgMHg1ZiwgMHg3OCwKLSAgMHgxZCwgMHgwZiwgMHg3ZCwgMHhjMiwgMHhjNywgMHgwNSwgMHhjYywgMHg5MywgMHhiMSwgMHhiYSwgMHhjZCwgMHgyMywKLSAgMHhmMCwgMHg1NywgMHgzNCwgMHg3MywgMHgzNiwgMHhjMSwgMHg2MywgMHhjZiwgMHg5ZSwgMHhjZiwgMHg5MiwgMHhkZSwKLSAgMHhhMSwgMHgyZCwgMHg1MCwgMHgyZiwgMHgzMSwgMHgwOCwgMHhkMiwgMHg5OCwgMHg5YSwgMHhkMSwgMHhlNSwgMHg3MSwKLSAgMHgyNCwgMHg3OCwgMHgwMiwgMHg3YywgMHhlMSwgMHhlYSwgMHg1MCwgMHhmNCwgMHg1MywgMHhjMCwgMHhhOCwgMHgyNCwKLSAgMHg0NCwgMHhlMywgMHgyZiwgMHhjNCwgMHhiMSwgMHg3ZiwgMHhkYSwgMHhhYiwgMHg2NywgMHhkZiwgMHg4YSwgMHg3YiwKLSAgMHhhNCwgMHg0ZSwgMHgzMiwgMHhmYywgMHg0YiwgMHgxNywgMHhmZCwgMHhhYSwgMHhiNiwgMHg3ZCwgMHhmOCwgMHhhNywKLSAgMHhiYSwgMHgwMCwgMHhhMiwgMHg4YSwgMHgyOCwgMHgwMiwgMHg5MywgMHgzOCwgMHhlMywgMHhmMiwgMHgzZCwgMHg5NywKLSAgMHg3ZCwgMHgxMSwgMHgyMywgMHhmOCwgMHgwZCwgMHgzOSwgMHhkMiwgMHg2NywgMHgxYywgMHg3ZSwgMHg0NywgMHhiMiwKLSAgMHhlZiwgMHhhMiwgMHgyNCwgMHg3ZiwgMHgwMSwgMHhhMCwgMHgxYiwgMHhlMywgMHg3YywgMHg1ZCwgMHhiZiwgMHg5OCwKLSAgMHgzZSwgMHhjYSwgMHhkOSwgMHg1YSwgMHhlMywgMHg3YywgMHg1ZCwgMHhiZiwgMHg5OCwgMHgzZSwgMHhjYSwgMHhkOSwKLSAgMHg0MCwgMHgxNCwgMHg1MSwgMHg0NSwgMHgwMCwgMHg1NiwgMHhiOSwgMHgwZCwgMHhiNCwgMHhmYiwgMHg0ZSwgMHgzMiwKLSAgMHhmMywgMHg2OCwgMHg3NSwgMHhiNywgMHgxMiwgMHg1MiwgMHhiNCwgMHgyOCwgMHg2YywgMHgyOCwgMHgxMSwgMHhkNCwKLSAgMHgxMSwgMHhlOCwgMHgyMiwgMHhiNiwgMHg1MCwgMHg0NiwgMHhlOCwgMHgwZiwgMHg4ZCwgMHhlNSwgMHg0MywgMHg5MywKLSAgMHhjMiwgMHg2ZSwgMHgyMSwgMHhjYiwgMHhiNCwgMHgyNywgMHhiNywgMHhlYywgMHhmMSwgMHhjNCwgMHhhZSwgMHg3NSwKLSAgMHhiMSwgMHg1ZCwgMHg0OSwgMHg5OSwgMHg4ZiwgMHhiZSwgMHhlZiwgMHgzMywgMHhlYywgMHg4ZSwgMHhlZCwgMHhhZSwKLSAgMHgyYiwgMHhhNywgMHhiNiwgMHgxZiwgMHhhYSwgMHgxZCwgMHhmMCwgMHhkNSwgMHg3ZCwgMHgyOSwgMHgyMiwgMHhmNCwKLSAgMHhlZCwgMHhkOCwgMHg1YSwgMHg2MiwgMHhjNCwgMHhiYiwgMHhhMiwgMHhkZiwgMHgxZSwgMHg2MywgMHgyYiwgMHg3MSwKLSAgMHhmOSwgMHg2ZCwgMHhmMiwgMHhhOSwgMHg2YSwgMHhkMSwgMHg2ZCwgMHgyOSwgMHg0MywgMHg2NSwgMHg1YiwgMHgwMCwKLSAgMHhhYiwgMHhiNCwgMHgwNywgMHg5OCwgMHg4MiwgMHg3NSwgMHhhZCwgMHg3NSwgMHgzYiwgMHgwYiwgMHg1ZiwgMHg4NCwKLSAgMHhjNiwgMHgyOSwgMHgyZSwgMHhlZCwgMHg4OSwgMHhjNywgMHhjYiwgMHgyYywgMHgzMSwgMHg5YiwgMHg3YiwgMHgyMywKLSAgMHhjNSwgMHg1YywgMHg1NSwgMHhjMiwgMHgyMywgMHg2YiwgMHg0MCwgMHg1MCwgMHg5NCwgMHhjNywgMHgyZSwgMHhhNCwKLSAgMHg0NiwgMHg1MCwgMHhmMSwgMHg0MywgMHg4ZCwgMHhmMywgMHgwMiwgMHg5ZiwgMHgxZCwgMHg2YiwgMHhjNiwgMHhhYiwKLSAgMHhiZSwgMHgwZiwgMHgzZiwgMHg2NSwgMHhjOCwgMHhhYywgMHhhYiwgMHhjMSwgMHg5ZSwgMHg5MCwgMHhlMiwgMHhlZCwKLSAgMHg2YSwgMHg4ZSwgMHhkZCwgMHhjNywgMHgxZSwgMHg5MiwgMHhiNSwgMHhlZCwgMHhjMSwgMHgwMSwgMHhkNSwgMHgxZSwKLSAgMHhjZCwgMHgyNSwgMHg0MywgMHg0NywgMHg5OSwgMHg4NywgMHgwMiwgMHg5OCwgMHg1ZSwgMHg4OCwgMHgyMCwgMHhhMSwKLSAgMHgyNywgMHg3ZCwgMHhjNiwgMHhiNSwgMHg4NywgMHg3YSwgMHgzYSwgMHg0YywgMHhhNywgMHhkZCwgMHg5NiwgMHhhZiwKLSAgMHhjYywgMHhiNSwgMHhmMywgMHgwOCwgMHhhNywgMHgwYiwgMHg3NiwgMHgyNiwgMHg1OSwgMHgxZCwgMHgzNywgMHgwYiwKLSAgMHhhYywgMHg3OCwgMHhhZCwgMHhhZCwgMHg4NywgMHhlMywgMHg0OCwgMHg5NiwgMHhiNywgMHg4YSwgMHgxNCwgMHhlMSwKLSAgMHg0OCwgMHg0YiwgMHhjOCwgMHgyZSwgMHgxMiwgMHg0MiwgMHhiNiwgMHgwMiwgMHgwOCwgMHgwNCwgMHgwZCwgMHgyZiwKLSAgMHg3YSwgMHgxYSwgMHgzYiwgMHhmNSwgMHg5MywgMHhkZCwgMHg2ZCwgMHgxOSwgMHg0NCwgMHg2OCwgMHgxMSwgMHhlYywKLSAgMHhhYSwgMHg3MSwgMHhmYiwgMHhhMCwgMHg3OSwgMHhiNSwgMHgzMiwgMHgxYSwgMHgwYSwgMHg0YSwgMHg5OCwgMHg2OSwKLSAgMHg3YSwgMHg0YiwgMHhhNSwgMHhjMiwgMHg5ZSwgMHhhZCwgMHhhMCwgMHhiNCwgMHhhNSwgMHg4ZSwgMHhiYSwgMHgyNCwKLSAgMHg4MSwgMHhhZCwgMHg5MCwgMHgzNSwgMHhjOSwgMHgwYSwgMHhjOSwgMHg3MiwgMHg2YSwgMHg1YSwgMHgyYywgMHhhZCwKLSAgMHhjYSwgMHg3YSwgMHhmYywgMHg2ZCwgMHhhNSwgMHgwYiwgMHg1OSwgMHg5NCwgMHhlMiwgMHg5YSwgMHg4YSwgMHhkMywKLSAgMHg5YSwgMHhlNiwgMHg0MSwgMHg3NCwgMHhhOCwgMHhhZCwgMHhjNywgMHhkYywgMHhkNiwgMHg5NCwgMHgxMywgMHhiZSwKLSAgMHg1MSwgMHhiMCwgMHg0ZSwgMHg4ZSwgMHhiNSwgMHhlZSwgMHgwYSwgMHgyZCwgMHhlNiwgMHhmNSwgMHgyMiwgMHhkMywKLSAgMHg5NiwgMHgwOCwgMHg3NiwgMHhjZSwgMHg0OSwgMHgwZCwgMHgzMSwgMHgwYSwgMHhkOSwgMHgxMSwgMHhjMiwgMHhkNCwKLSAgMHg1OSwgMHg1YywgMHhlOCwgMHgyYSwgMHgwZSwgMHhlOCwgMHgwNCwgMHg5NSwgMHg5NSwgMHgxNCwgMHgzOCwgMHg5ZSwKLSAgMHg1MywgMHhiMCwgMHg5ZSwgMHhjYywgMHg4ZSwgMHhiYiwgMHhkOSwgMHhjZiwgMHgyYiwgMHg4NCwgMHg2OSwgMHhiZiwKLSAgMHgyYywgMHhlNywgMHg3MiwgMHhkZiwgMHg3NywgMHg2NSwgMHgxMSwgMHhiMSwgMHhkYiwgMHhjMiwgMHg5YywgMHhiYSwKLSAgMHgzZSwgMHg1YiwgMHg0MiwgMHg5ZiwgMHg2MiwgMHgyMiwgMHhiYiwgMHg2NywgMHhlNSwgMHgzNCwgMHg5NSwgMHgxMiwKLSAgMHg5MCwgMHhlYiwgMHhhYiwgMHg0YiwgMHg2ZCwgMHhiMiwgMHhkOSwgMHgyOSwgMHgwMCwgMHhlOCwgMHg2ZCwgMHg1YSwKLSAgMHgzYSwgMHgzZCwgMHhmNSwgMHhkMSwgMHg2ZiwgMHg2NiwgMHgxMiwgMHhmMiwgMHgwOSwgMHg3NiwgMHhmNSwgMHg1YiwKLSAgMHg5MywgMHg4OCwgMHg4OCwgMHhlYywgMHhiNiwgMHhiOSwgMHgwOSwgMHg4ZCwgMHhkOSwgMHg4NywgMHhlNSwgMHgxNywKLSAgMHg1NiwgMHhhZSwgMHg1MCwgMHhkYiwgMHhhOSwgMHhlYSwgMHgxYSwgMHhmMywgMHgzNCwgMHg3OSwgMHg3OSwgMHg0ZiwKLSAgMHgzMSwgMHhkNiwgMHhjNiwgMHhiYSwgMHhlOSwgMHhiZiwgMHgyZCwgMHg3OCwgMHg1NCwgMHhlOSwgMHgwZSwgMHg2MywKLSAgMHgyMiwgMHgwYSwgMHgxOCwgMHhiYSwgMHgyMSwgMHgxMiwgMHg1YSwgMHg0MywgMHhjNSwgMHg3ZCwgMHg5MiwgMHgwYiwKLSAgMHgyYiwgMHg0MiwgMHg1ZCwgMHg2OCwgMHgyOSwgMHgyOSwgMHg1NywgMHg2NiwgMHg4NSwgMHgyNSwgMHhjZCwgMHhhNSwKLSAgMHg1YSwgMHgyOSwgMHg0MiwgMHg4MSwgMHhkOCwgMHgwMCwgMHhmNCwgMHgzMiwgMHgyNiwgMHgxYSwgMHhjZSwgMHgxMywKLSAgMHgwMSwgMHhlOSwgMHgyZCwgMHhiNSwgMHgwMiwgMHgyMywgMHgxYywgMHhlMSwgMHhkYiwgMHg4MywgMHhlOSwgMHg1MywKLSAgMHg0ZCwgMHhhZCwgMHg0YSwgMHhlNSwgMHgyOCwgMHg2ZCwgMHhhZCwgMHhhOSwgMHgwYSwgMHg1OSwgMHgwYiwgMHg0YSwKLSAgMHg1NSwgMHhjZSwgMHg5MiwgMHg5MSwgMHhlNiwgMHg4ZSwgMHg1MiwgMHg3NywgMHg1MywgMHhjZiwgMHgyMSwgMHg3YSwKLSAgMHgxYiwgMHg3MiwgMHgzNiwgMHg2NCwgMHhlMSwgMHg3MywgMHhkZCwgMHhiYSwgMHg1OSwgMHg0OCwgMHg5NiwgMHg5OSwKLSAgMHg5MCwgMHhjYywgMHg3OCwgMHg4OCwgMHg5ZiwgMHgzNSwgMHg0YSwgMHg0YywgMHg3NywgMHg5YiwgMHgyNSwgMHhkNSwKLSAgMHgzNiwgMHg5NywgMHgxYywgMHgzZCwgMHgwMywgMHg4OCwgMHgwYiwgMHhkNiwgMHhjZSwgMHg4MiwgMHhkMCwgMHg5MSwKLSAgMHhkMCwgMHgxZSwgMHg5YSwgMHg3MiwgMHg2NywgMHhkOSwgMHhjZiwgMHgxYSwgMHg4MywgMHgxMiwgMHgwNCwgMHgxZSwKLSAgMHg2OSwgMHg2YiwgMHg0OSwgMHg0YywgMHhhMSwgMHhkYiwgMHgxMCwgMHgxOCwgMHg4ZSwgMHhiNCwgMHgxMiwgMHhhNCwKLSAgMHgzYywgMHhiNCwgMHg2YywgMHgyMSwgMHg1YywgMHhlOSwgMHg2ZCwgMHg0OSwgMHg0OCwgMHgyNSwgMHg0MSwgMHg0OCwKLSAgMHgwYSwgMHhkNiwgMHg4NywgMHg1ZSwgMHhlYiwgMHg1YywgMHgyYiwgMHg4YywgMHg5YiwgMHg4MiwgMHg5ZSwgMHg4NywKLSAgMHgxZCwgMHhhYiwgMHhjMSwgMHg4MiwgMHhhNywgMHgxOCwgMHg0NCwgMHg4OSwgMHg0YSwgMHhmMiwgMHg1OCwgMHgyZCwKLSAgMHgyZiwgMHg1YywgMHhhZSwgMHgwNiwgMHg5YSwgMHg0OCwgMHg1MiwgMHg5YywgMHg1NywgMHg1MiwgMHg5MiwgMHhiNSwKLSAgMHg2ZiwgMHhmMywgMHg4MiwgMHg0ZiwgMHg3OCwgMHhhZCwgMHg1NiwgMHg4NSwgMHhkYSwgMHgyZSwgMHg0YiwgMHgyYywKLSAgMHhlNCwgMHhlYSwgMHg0YywgMHgxNywgMHg1NSwgMHgzMSwgMHhmOCwgMHg5MSwgMHhlYywgMHhhOCwgMHg3ZiwgMHhiMywKLSAgMHg4ZSwgMHhjMSwgMHg2YiwgMHg0NCwgMHhlYywgMHgyMywgMHg0MSwgMHhjMiwgMHg1MiwgMHg1MiwgMHhiZSwgMHg2NSwKLSAgMHg3ZSwgMHg2YSwgMHhjNiwgMHg4MCwgMHhhOCwgMHhmNCwgMHg0MywgMHgxOCwgMHhkZCwgMHg5ZSwgMHgxMSwgMHgxZSwKLSAgMHg0MywgMHg5MiwgMHgyMywgMHg1OSwgMHgyNywgMHgzNCwgMHhlNSwgMHhlMCwgMHg1YiwgMHg1NiwgMHg5NywgMHgxNSwKLSAgMHgwMiwgMHgxNywgMHhiZSwgMHg2ZCwgMHhjMCwgMHg5ZiwgMHgzNSwgMHg3MiwgMHhhNCwgMHgzYSwgMHgxMCwgMHg5MiwKLSAgMHhhZCwgMHgxZSwgMHg2MCwgMHg4MCwgMHgxMiwgMHg0OSwgMHhkMSwgMHhlYSwgMHgzYSwgMHg1NywgMHg0NSwgMHg4NSwKLSAgMHgxNiwgMHhmYiwgMHg4MywgMHhhZiwgMHhiNiwgMHhlMSwgMHg1NiwgMHgzNSwgMHgwOCwgMHg0ZiwgMHgzMCwgMHhkYSwKLSAgMHhiNSwgMHg0NSwgMHhlNCwgMHg4OCwgMHhlMywgMHhhZiwgMHgyMSwgMHgxYywgMHhjYSwgMHhlZCwgMHgxNiwgMHhkOSwKLSAgMHhkOSwgMHgyNCwgMHgxMiwgMHg1MiwgMHgxMCwgMHhhMCwgMHgzOSwgMHgwMCwgMHgzZSwgMHg3NiwgMHhmYSwgMHg0NiwKLSAgMHg0OCwgMHhiYiwgMHhjYiwgMHhjMiwgMHgyZSwgMHgwOSwgMHhiNSwgMHhkYiwgMHg1YywgMHg4MSwgMHgyMiwgMHgwMiwKLSAgMHhlNSwgMHgxOSwgMHhjOSwgMHgxMiwgMHhhNSwgMHg3NiwgMHg2YiwgMHg3YSwgMHgzYiwgMHhkYywgMHhjMSwgMHg0OSwKLSAgMHhlZCwgMHg1NywgMHhiMSwgMHhjZSwgMHhkYSwgMHhkMywgMHhkMCwgMHgyOCwgMHhmOSwgMHhlOCwgMHgyMCwgMHgwMywKLSAgMHhiNCwgMHg5YSwgMHhkMywgMHg5MiwgMHhhNiwgMHhkZCwgMHg3ZSwgMHg5YSwgMHhmNSwgMHhjNiwgMHhlOCwgMHgyMCwKLSAgMHg0MywgMHhiNywgMHg0YSwgMHg2OSwgMHhhNCwgMHhhNSwgMHhiOSwgMHhiMCwgMHg5YywgMHg3MSwgMHg3MywgMHg5ZSwKLSAgMHg2ZCwgMHg0YiwgMHhlNSwgMHg3NSwgMHg5NiwgMHg5MiwgMHhhNCwgMHhiZCwgMHhiNCwgMHhhNSwgMHg1YSwgMHgwYiwKLSAgMHgxYSwgMHgyYSwgMHhkZSwgMHg4MCwgMHgyOSwgMHhkMSwgMHhhOSwgMHhmNSwgMHg2MSwgMHgzZSwgMHg4OCwgMHg5MCwKLSAgMHg3ZSwgMHg2NSwgMHhjMywgMHgwZCwgMHhiYiwgMHhkYywgMHg1OSwgMHg4NiwgMHhjYSwgMHgyZiwgMHgwZSwgMHhjYSwKLSAgMHg3MSwgMHhiOSwgMHg2ZCwgMHgyYSwgMHg1NCwgMHhiMCwgMHg5NywgMHhkZSwgMHg4YSwgMHg5MCwgMHgxYiwgMHg1MywKLSAgMHgwOCwgMHg1YSwgMHhmNCwgMHgxNCwgMHhlMywgMHg2YSwgMHgwMSwgMHg0MCwgMHgyOCwgMHhmOSwgMHhjOSwgMHg1ZiwKLSAgMHg1MywgMHhiZSwgMHhmZSwgMHg2YiwgMHhlYywgMHgzNywgMHgzMywgMHg1YiwgMHhkYSwgMHgyNCwgMHg1OCwgMHg1YiwKLSAgMHg3ZSwgMHgzMiwgMHhkMiwgMHhjYSwgMHg1NywgMHgyZSwgMHg0MywgMHg3MiwgMHg0MywgMHg2OSwgMHg2ZCwgMHhmNiwKLSAgMHhkNiwgMHg5MiwgMHhjOCwgMHgwZSwgMHhhNSwgMHgyYiwgMHg0NywgMHg2YywgMHgwMiwgMHg5ZCwgMHg0OSwgMHhkMCwKLSAgMHg1OCwgMHgwMSwgMHg1YSwgMHgzZiwgMHg5YiwgMHhhZSwgMHhlYiwgMHg2NSwgMHhiNiwgMHhlYywgMHg1MSwgMHgyMiwKLSAgMHg2ZiwgMHhmMiwgMHg2OSwgMHgzNywgMHgyZSwgMHg2NiwgMHgxMiwgMHg4MiwgMHg2ZSwgMHhjZSwgMHhhMSwgMHg4MiwKLSAgMHhlMywgMHg2MCwgMHhmMywgMHg3NiwgMHg0YywgMHhiMCwgMHg5NCwgMHhhOCwgMHgzNiwgMHgzNiwgMHgwMSwgMHhmMywKLSAgMHhjOCwgMHgyNCwgMHhlYiwgMHg5OCwgMHg5ZCwgMHg3NCwgMHhmMywgMHg2MSwgMHg1NiwgMHgzNSwgMHg3NywgMHg2NiwKLSAgMHhkYywgMHg5YiwgMHhmNCwgMHhlNSwgMHhiZCwgMHg3MSwgMHg5YiwgMHgwZCwgMHg1MiwgMHhkOSwgMHg4OCwgMHgxZCwKLSAgMHg1YiwgMHg0YywgMHhjNCwgMHg2YywgMHgxZSwgMHg1NSwgMHgyMSwgMHhiNCwgMHgyMCwgMHg4NCwgMHhhNCwgMHhiNiwKLSAgMHg0ZiwgMHgyOSwgMHgzZCwgMHg1NSwgMHhiNCwgMHhmNywgMHhkMSwgMHg3OSwgMHgyNCwgMHgxZSwgMHhkYiwgMHhiMywKLSAgMHhjMywgMHg3MSwgMHhkZCwgMHhiOSwgMHhjZCwgMHhkNCwgMHhmYiwgMHg1YiwgMHg5OSwgMHgyZiwgMHg5MCwgMHhhNSwKLSAgMHg2YywgMHgyNSwgMHg4NiwgMHgwMiwgMHg1MywgMHgxNSwgMHg4NSwgMHgxMywgMHhhNywgMHgwOSwgMHg3NSwgMHhkMiwKLSAgMHgwYiwgMHhjZSwgMHhmNCwgMHhkNywgMHg5YSwgMHgwMCwgMHg0ZiwgMHg1MSwgMHhhNCwgMHg5YSwgMHhkZiwgMHg4OSwKLSAgMHgzNywgMHg2ZiwgMHhiYiwgMHhjNywgMHg0MiwgMHgzYywgMHhiMSwgMHhmYiwgMHgyYiwgMHg0ZiwgMHgxNywgMHg5YywKLSAgMHg2MiwgMHhjZiwgMHgwYiwgMHhmOSwgMHhhMiwgMHhkYSwgMHg0YSwgMHgxNiwgMHg1YiwgMHgyYSwgMHg3OCwgMHhhNywKLSAgMHhjZiwgMHg1MywgMHg5YiwgMHgxZSwgMHg3NywgMHg1MCwgMHg5MCwgMHg3YSwgMHgxMCwgMHg3YiwgMHhjYywgMHg2ZCwKLSAgMHhiZiwgMHgyYywgMHg5NSwgMHg4OCwgMHhjZiwgMHg2NywgMHgxOSwgMHg3YSwgMHgzMywgMHg1MiwgMHg2MCwgMHhjMiwKLSAgMHgwYiwgMHg2MCwgMHhjYiwgMHgwYiwgMHhmMywgMHhjOSwgMHgzYywgMHhhZSwgMHgzNCwgMHhlMiwgMHg4MSwgMHhlYSwKLSAgMHhiZCwgMHhhMSwgMHg3YSwgMHg1MCwgMHg0NywgMHgzMiwgMHhmOSwgMHg4MiwgMHg4ZiwgMHgyZSwgMHg4OCwgMHhhZSwKLSAgMHg5NCwgMHg2MSwgMHhmMiwgMHhiMiwgMHgwYiwgMHhmMywgMHhkOSwgMHgyYSwgMHhhMSwgMHhiMSwgMHg2YywgMHgyNSwKLSAgMHhkNCwgMHhiOSwgMHgxOSwgMHgzMiwgMHg1OSwgMHg3NCwgMHgzOSwgMHhiZSwgMHg0NCwgMHg4NywgMHgxYywgMHgyOCwKLSAgMHg2ZCwgMHhkNCwgMHg2OCwgMHgyOCwgMHhhMSwgMHgzZSwgMHhmNiwgMHhlNywgMHgzNiwgMHhmOSwgMHg0MSwgMHgyMCwKLSAgMHgxMiwgMHg0NSwgMHgzMSwgMHhkNSwgMHg4YywgMHhmNCwgMHg0MiwgMHg1NCwgMHg1OSwgMHhiMiwgMHg3OCwgMHg2YiwKLSAgMHhjNywgMHg3NCwgMHhjYywgMHg5YSwgMHhlOSwgMHg1ZCwgMHg5YiwgMHgyNywgMHg3MSwgMHg5YiwgMHg2ZCwgMHhkNSwKLSAgMHhmZSwgMHg4MSwgMHgwMiwgMHg2MSwgMHg0OSwgMHhmMiwgMHgzOSwgMHg2NywgMHhiOCwgMHgwMiwgMHhlYSwgMHgxMiwKLSAgMHhhNiwgMHg5NiwgMHg3NSwgMHhkNSwgMHg2ZCwgMHg5NSwgMHg3OCwgMHg4MSwgMHg1ZiwgMHg0MiwgMHgyNywgMHhhYSwKLSAgMHg3NywgMHg1NSwgMHhmNiwgMHg0ZCwgMHg4MiwgMHhkYSwgMHg3MiwgMHgwYiwgMHgxNSwgMHhkZiwgMHgxMSwgMHhiOSwKLSAgMHhiYSwgMHhlYywgMHhkNywgMHgyZiwgMHgxMSwgMHhiOSwgMHhlNSwgMHhjZSwgMHg3MCwgMHg4ZSwgMHhkNSwgMHhiNywKLSAgMHgxMSwgMHhhZSwgMHhjOSwgMHg2OCwgMHg0YSwgMHg0MCwgMHgwOCwgMHgwOCwgMHg1NywgMHg1NCwgMHg4MSwgMHhhZCwKLSAgMHg3YSwgMHhjZSwgMHhjOSwgMHhjZiwgMHgwMCwgMHhiMiwgMHg0YiwgMHhhZSwgMHg0MywgMHhjMywgMHhlNCwgMHgzNSwKLSAgMHg5MCwgMHg2OCwgMHhkZiwgMHg2YywgMHhiMiwgMHhkZiwgMHhiMywgMHhkZCwgMHgxNywgMHhiZCwgMHg4NywgMHgyNCwKLSAgMHg0NywgMHg1NywgMHgyMSwgMHg1ZSwgMHhmZiwgMHgwMCwgMHg1OCwgMHg3MiwgMHhhOCwgMHhmYSwgMHhjOSwgMHhhOCwKLSAgMHg2OSwgMHhhMCwgMHg5YSwgMHg2NiwgMHhlZSwgMHgzMiwgMHhmYywgMHg0YiwgMHgxNywgMHhmZCwgMHhhYSwgMHhiNiwKLSAgMHg3ZCwgMHhmOCwgMHhhNywgMHhiYSwgMHg0NCwgMHhlMywgMHgyNywgMHhjNCwgMHg3MSwgMHg3ZiwgMHhkYSwgMHhhYiwKLSAgMHg2NywgMHhkZiwgMHg4YSwgMHg3YiwgMHhhOCwgMHgyNCwgMHgyOCwgMHhhYywgMHg2ZiwgMHg0NywgMHhhOSwgMHgxNCwKLSAgMHg1MCwgMHgxOSwgMHhhNCwgMHhjZSwgMHgzOCwgMHhmYywgMHg4ZiwgMHg2NSwgMHhkZiwgMHg0NCwgMHg0OCwgMHhmZSwKLSAgMHgwMywgMHg0ZSwgMHg3NCwgMHg5OSwgMHhjNywgMHgxZiwgMHg5MSwgMHhlYywgMHhiYiwgMHhlOCwgMHg4OSwgMHgxZiwKLSAgMHhjMCwgMHg2OCwgMHgwNiwgMHhmOCwgMHhkZiwgMHgxNywgMHg2ZiwgMHhlNiwgMHgwZiwgMHhiMiwgMHhiNiwgMHg1NiwKLSAgMHhiOCwgMHhkZiwgMHgxNywgMHg2ZiwgMHhlNiwgMHgwZiwgMHhiMiwgMHhiNiwgMHg1MCwgMHgwNSwgMHgxNCwgMHg1MSwKLSAgMHg0MCwgMHgxNCwgMHg1MSwgMHg0NSwgMHgwMSwgMHhlNSwgMHg0MCwgMHg2ZiwgMHg2NCwgMHg2ZiwgMHhkMSwgMHg1ZiwKLSAgMHgyMiwgMHg2NiwgMHg1NiwgMHgyOSwgMHg5YywgMHgzNywgMHhlMiwgMHg1YywgMHhiYiwgMHg1NSwgMHhhOSwgMHg4MiwKLSAgMHg1YiwgMHg4YywgMHhiNywgMHhmMiwgMHg0YywgMHg1ZCwgMHhhMSwgMHhkMCwgMHg0OCwgMHg4YywgMHhhMSwgMHhmZSwKLSAgMHg5MywgMHhiNiwgMHhhNywgMHhmZSwgMHgxMSwgMHhkYiwgMHgyMCwgMHg3OCwgMHgxNCwgMHgwZSwgMHhmMiwgMHhhYSwKLSAgMHhmYSwgMHhmNywgMHg1ZCwgMHg3NywgMHg1NSwgMHhhNywgMHhlMSwgMHgxMSwgMHg4OCwgMHg0ZiwgMHhjOSwgMHhiMCwKLSAgMHg3NCwgMHhkYywgMHg3MSwgMHhlZiwgMHgzMywgMHgyOCwgMHhjNywgMHhhNCwgMHgyNiwgMHhlYiwgMHg2NSwgMHg3NCwKLSAgMHgwZCwgMHhhYiwgMHhiNiwgMHg2ZiwgMHhhYSwgMHg5YiwgMHhmNSwgMHg4NSwgMHhhNCwgMHgxNCwgMHhlYiwgMHhiOCwKLSAgMHg5ZCwgMHg2ZSwgMHhhNSwgMHgzYywgMHgzYywgMHhhMiwgMHgxYSwgMHhjYSwgMHhjMywgMHgzOSwgMHg1OCwgMHhiYywKLSAgMHg1YywgMHg2ZSwgMHhiNiwgMHgwNiwgMHhiMiwgMHgxYywgMHg0ZSwgMHg2MywgMHhlZiwgMHhhMiwgMHg3NywgMHg5MywKLSAgMHhjYiwgMHgwZSwgMHgzNCwgMHhjNywgMHg2YywgMHg5NSwgMHg3MiwgMHhhNCwgMHg3NSwgMHg1YSwgMHgwMSwgMHhlNiwKLSAgMHgyOCwgMHg3MSwgMHhiZCwgMHgyNCwgMHg5NCwgMHhlZCwgMHg0OSwgMHg1MiwgMHg0NywgMHg0ZSwgMHhmYSwgMHhkMywKLSAgMHg5NiwgMHg0NywgMHhiNywgMHg2NCwgMHgxMywgMHhlMywgMHg5YywgMHhhZSwgMHhlMiwgMHhkZCwgMHg5OSwgMHhiOCwKLSAgMHhjYywgMHhhOCwgMHgzMCwgMHhkYiwgMHhhYSwgMHg2ZiwgMHhjYSwgMHg5YywgMHgyNSwgMHg2ZCwgMHhhYywgMHgzOCwKLSAgMHhlMiwgMHgxMywgMHhjYywgMHgxYSwgMHg2ZCwgMHgxZCwgMHg5OCwgMHhkMSwgMHhkZSwgMHhmNiwgMHg0OSwgMHgyNSwKLSAgMHgzZCwgMHhjNSwgMHgwMywgMHhmMCwgMHg3OCwgMHhjYywgMHhhMCwgMHhjNywgMHhiOCwgMHhiNywgMHgxMiwgMHgyMiwKLSAgMHgxNSwgMHgxMiwgMHhjZCwgMHg3YSwgMHg0YiwgMHg5NywgMHgyYiwgMHg0MywgMHgyNCwgMHg3OSwgMHhiMSwgMHhjZSwKLSAgMHhmZiwgMHgwMCwgMHg5ZSwgMHhjMiwgMHhmNSwgMHgxNiwgMHg1ZSwgMHgyNSwgMHg0OSwgMHg0ZiwgMHhlOCwgMHgyZCwKLSAgMHgyNiwgMHhhZCwgMHg3YywgMHg2NiwgMHhmZCwgMHg2YSwgMHhiMiwgMHgzYiwgMHgwMiwgMHhjMywgMHgzNiwgMHhkZCwKLSAgMHgyNiwgMHgzYywgMHhjOSwgMHhjZCwgMHg2ZCwgMHhkOSwgMHg0YiwgMHg2OCwgMHhhYywgMHg0YiwgMHg5NCwgMHgxNCwKLSAgMHhhMCwgMHhlMiwgMHg1NCwgMHhiMSwgMHhmMCwgMHg5NiwgMHg0OCwgMHhlNiwgMHhkMSwgMHhlZiwgMHg0YSwgMHg4NiwKLSAgMHhiYSwgMHg3NywgMHg2YiwgMHgzNSwgMHg5YywgMHgzOCwgMHhmNSwgMHgzMiwgMHg4MywgMHg0YiwgMHgyYSwgMHg1ZCwKLSAgMHgwZiwgMHg3OCwgMHhjYywgMHg1YiwgMHhiYywgMHg4MCwgMHg2ZiwgMHg5NiwgMHg2NCwgMHg0NCwgMHg1OCwgMHg3ZCwKLSAgMHhhZSwgMHg1NiwgMHgyNywgMHg1ZCwgMHg1NCwgMHhhNywgMHgxZSwgMHg5MiwgMHhkNywgMHg3YSwgMHg0MiwgMHgxMCwKLSAgMHhkZiwgMHgyYSwgMHgxOCwgMHg2OSwgMHg0NywgMHg0NCwgMHg3MiwgMHhlZiwgMHg3ZCwgMHgwOSwgMHg0ZSwgMHhjNSwKLSAgMHg3MywgMHhlMywgMHgzMiwgMHhiMSwgMHhiOSwgMHg4ZSwgMHhjNiwgMHg1NiwgMHg0ZCwgMHhlNCwgMHhmMiwgMHgzMiwKLSAgMHgxOSwgMHhhMiwgMHg0MCwgMHg3NSwgMHhiOSwgMHhhOSwgMHgwYiwgMHgwYywgMHhhOSwgMHhhNSwgMHgwMCwgMHhlYiwKLSAgMHgwZCwgMHg4NSwgMHgwZCwgMHgyNCwgMHgyNCwgMHgyOSwgMHgzZCwgMHhjMywgMHg2YSwgMHgxYSwgMHg1MSwgMHhkZSwKLSAgMHhmNywgMHg1MSwgMHg5NiwgMHhkNSwgMHg1ZSwgMHhlMywgMHhkZSwgMHg1NSwgMHhmYywgMHg5YSwgMHg3ZSwgMHg3NCwKLSAgMHhjYiwgMHgwNCwgMHgyNywgMHg5ZCwgMHg4YiwgMHgxZCwgMHgyZCwgMHgyMSwgMHhhNSwgMHgzMiwgMHhhNiwgMHhiOSwKLSAgMHg0MiwgMHhmNCwgMHg4NywgMHgxNiwgMHhlMiwgMHg3OSwgMHgzYiwgMHgzNSwgMHhhOSwgMHg0OCwgMHhlNywgMHhkMiwKLSAgMHhkMiwgMHg1MiwgMHg5MCwgMHgwNiwgMHg5NCwgMHg5ZSwgMHhiYSwgMHhlNywgMHgyYiwgMHgxZCwgMHg5NCwgMHhkNCwKLSAgMHhhYiwgMHhjZCwgMHhmNiwgMHg2YiwgMHhkMiwgMHhhNiwgMHgyZSwgMHg2OCwgMHg3ZCwgMHg4NCwgMHg1YiwgMHgzYiwKLSAgMHg0MywgMHgxOSwgMHgyZiwgMHgwNCwgMHgyNSwgMHhiNiwgMHg5YSwgMHg0MywgMHhmYywgMHhiYywgMHg4NSwgMHhkMiwKLSAgMHgxMiwgMHg5MSwgMHhjZSwgMHgwYSwgMHg0ZSwgMHhjZSwgMHg4NiwgMHg4NywgMHg0YSwgMHhjOSwgMHg2MywgMHg5MywKLSAgMHg1ZCwgMHhmYSwgMHgxYiwgMHg5OSwgMHg5OSwgMHg3OSwgMHhjNywgMHhlZSwgMHhjNiwgMHhjZiwgMHg4YywgMHgyOSwKLSAgMHgzMiwgMHg2ZCwgMHg5NiwgMHhkNywgMHhjOCwgMHhmMiwgMHg2MSwgMHgxNSwgMHhkNywgMHg5MCwgMHg1OCwgMHg3YywKLSAgMHgwNywgMHgxMiwgMHg5ZSwgMHg2NiwgMHhjMiwgMHg4YSwgMHg1ZCwgMHg0MSwgMHhlNiwgMHhkNywgMHgzMCwgMHhlNSwKLSAgMHg1MiwgMHgxNCwgMHgwMSwgMHgyMCwgMHhmNSwgMHgzOSwgMHhiZCwgMHhjNiwgMHg2YSwgMHhlYiwgMHg3NSwgMHg5MSwKLSAgMHgyNiwgMHhlNiwgMHg4NiwgMHhkOCwgMHg5YSwgMHhlOSwgMHg4ZSwgMHhhNiwgMHgyZCwgMHg2ZCwgMHhjMiwgMHg2YSwKLSAgMHg3NCwgMHg4MiwgMHhhNiwgMHg4MiwgMHhmOSwgMHg1ZSwgMHg3MSwgMHgwNywgMHg2MSwgMHhhNSwgMHhmOSwgMHhlMywKLSAgMHg0YSwgMHhlNiwgMHgwMCwgMHgwNCwgMHhmNSwgMHgyNywgMHhjMiwgMHg1NSwgMHhmYiwgMHgwNiwgMHg0ZiwgMHgxYiwKLSAgMHgxNywgMHg5NiwgMHhjZCwgMHg5YSwgMHhlMSwgMHgwMiwgMHhkZiwgMHgyNCwgMHgzNiwgMHhhNSwgMHgyNiwgMHhkYiwKLSAgMHgwZCwgMHg4MCwgMHg1YiwgMHgyMCwgMHhlZiwgMHg2OSwgMHg1MywgMHhhYSwgMHgzYywgMHhlYSwgMHg3NSwgMHg0OSwKLSAgMHhmZSwgMHg5MywgMHhjZCwgMHhmMywgMHhiYSwgMHg5MCwgMHg2YiwgMHg3YywgMHgwYywgMHhhYiwgMHgxZSwgMHhiMiwKLSAgMHhkYiwgMHgxMCwgMHhhOCwgMHgxNiwgMHgwYiwgMHhiMywgMHg3NiwgMHhmNywgMHg1OCwgMHg0YywgMHg5NiwgMHg1YywKLSAgMHg4NSwgMHg2YywgMHg3NSwgMHhmNCwgMHgzOSwgMHhjYywgMHgzYywgMHhlZSwgMHg2NSwgMHgzNiwgMHg5MywgMHhjYSwKLSAgMHhlMCwgMHgyMCwgMHg4NSwgMHg3MywgMHg5MCwgMHg3YSwgMHg2ZiwgMHg3YSwgMHhlYiwgMHg0YywgMHg5MiwgMHg5MiwKLSAgMHgzNiwgMHg1OCwgMHhmMSwgMHhmYiwgMHhlOCwgMHg3MSwgMHhlNywgMHhkNywgMHgyYywgMHg1YSwgMHhkNywgMHgyYiwKLSAgMHg5NywgMHhiNywgMHg3OSwgMHg0YiwgMHgxMiwgMHg2NywgMHgzYSwgMHg5MCwgMHg0ZSwgMHg5MiwgMHg1NiwgMHg0MCwKLSAgMHg2OSwgMHhhMCwgMHgzNywgMHhmMCwgMHg1MCwgMHg4MiwgMHg5MSwgMHhiMywgMHhhZSwgMHhiYiwgMHgzNSwgMHgzZiwKLSAgMHg2YywgMHhiNCwgMHhkYSwgMHgyYywgMHg4NSwgMHhkNywgMHhkOSwgMHg2YywgMHg5OSwgMHgyYiwgMHg0NywgMHhiZiwKLSAgMHg0OSwgMHg3OSwgMHg2YSwgMHg3NSwgMHhmNywgMHg0MCwgMHhmZCwgMHgyNSwgMHgyYiwgMHg2YSwgMHgyMywgMHhkNSwKLSAgMHhkZCwgMHhlOCwgMHgxNCwgMHg4MiwgMHhhYywgMHhjMywgMHgzYSwgMHhjYiwgMHhkNiwgMHgyMywgMHhlMSwgMHhmNiwKLSAgMHgyZiwgMHg3MiwgMHgyMCwgMHgxMywgMHhhNSwgMHg1YywgMHg2NiwgMHgyOSwgMHgwZSwgMHhiOCwgMHg0NywgMHg0ZiwKLSAgMHg4MiwgMHgwNiwgMHhkYSwgMHg0OSwgMHhmNSwgMHhlZCwgMHhkMiwgMHgzYywgMHg1MSwgMHhiYSwgMHg5YiwgMHhjNiwKLSAgMHhiOCwgMHg3ZSwgMHg5OCwgMHhmMSwgMHg5NCwgMHg2ZiwgMHhmNywgMHgxNywgMHhlZSwgMHgyZSwgMHgzZSwgMHg0MiwKLSAgMHhhNCwgMHgzNiwgMHgxZCwgMHg1MSwgMHg0YiwgMHhlNywgMHhmZCwgMHhhYSwgMHhjZiwgMHg5ZSwgMHhlOCwgMHhmZCwKLSAgMHg1MiwgMHg0MywgMHg3ZSwgMHg4NCwgMHgwMSwgMHhhMSwgMHg1MCwgMHg0OSwgMHgyZSwgMHhhYywgMHhiYSwgMHhkOSwKLSAgMHgyMSwgMHg0NSwgMHg4YiwgMHgzMywgMHgzMiwgMHhhZiwgMHgzMiwgMHg0NywgMHg5YSwgMHg1YSwgMHg4OCwgMHhkZiwKLSAgMHg0NCwgMHgxZiwgMHhkNywgMHg1YSwgMHhiNCwgMHg5NCwgMHg3YiwgMHgxNCwgMHg3NywgMHhlYSwgMHhhOCwgMHgzNiwKLSAgMHgzMCwgMHg2OSwgMHg5NywgMHgxNSwgMHgyMiwgMHg3ZCwgMHhkNiwgMHg0NCwgMHg2OCwgMHg3MiwgMHg3YywgMHhiZCwKLSAgMHhjOSwgMHhjZCwgMHhjNywgMHg2NiwgMHgzYiwgMHg3MiwgMHgwNCwgMHg0NSwgMHgyZiwgMHg1ZiwgMHg5MywgMHg3MSwKLSAgMHhjNCwgMHhlZCwgMHgyYiwgMHgzYSwgMHhkYSwgMHhiYSwgMHgxNCwgMHhlYywgMHg5ZSwgMHg5ZSwgMHgyNSwgMHhiZCwKLSAgMHhkOSwgMHgxNiwgMHhiYiwgMHgzNCwgMHgzNCwgMHgzMCwgMHg5MCwgMHhjYywgMHg2NiwgMHg5MCwgMHgzNCwgMHhkYiwKLSAgMHgyZCwgMHgyMCwgMHgwZCwgMHg3YSwgMHg4MiwgMHg0NSwgMHgyNSwgMHg2NiwgMHhiYywgMHg0NywgMHhiNywgMHg1OSwKLSAgMHg1YiwgMHg2YywgMHg0YSwgMHg5NiwgMHg4OCwgMHgyMSwgMHhmMywgMHhjYSwgMHhjMywgMHg0MSwgMHgyNSwgMHhkOSwKLSAgMHgyZiwgMHg5ZiwgMHg0MywgMHg2ZCwgMHhhNywgMHg2YSwgMHg1MSwgMHhmNSwgMHgyNCwgMHgxMywgMHhlYywgMHhhZCwKLSAgMHg2OSwgMHhkMSwgMHg5NCwgMHhmNywgMHg1YywgMHg3OSwgMHg5OCwgMHhjZSwgMHhiNCwgMHg2MSwgMHhiNywgMHg1MSwKLSAgMHhiNiwgMHgwYywgMHg2YiwgMHg0NiwgMHgzNywgMHgwMSwgMHg0ZCwgMHhhNSwgMHhkNSwgMHgyNCwgMHhiOCwgMHhlMSwKLSAgMHg3NSwgMHhkNywgMHgxZSwgMHg3MCwgMHhiOCwgMHhmMywgMHhlZSwgMHgxMCwgMHgwMSwgMHg1YSwgMHg4OSwgMHhlYSwKLSAgMHhhNSwgMHg3NCwgMHgwMywgMHhkMCwgMHgwMCwgMHgwMCwgMHg2OCwgMHgwMCwgMHgyYSwgMHgyNiwgMHhlNSwgMHg5MiwKLSAgMHgzYywgMHhmNywgMHhiZCwgMHg0MiwgMHg0MSwgMHg2NSwgMHgwNCwgMHhlYiwgMHg5YywgMHg4MCwgMHg1NCwgMHg3ZCwKLSAgMHg5ZSwgMHgwMywgMHhlZCwgMHhhYSwgMHhiYSwgMHhmZiwgMHgwMCwgMHg5NCwgMHg1YywgMHhhMywgMHgzNCwgMHhkZCwKLSAgMHhlYiwgMHgyYywgMHg5ZiwgMHgxNywgMHgwMywgMHhjNywgMHhjZiwgMHg5YywgMHhhOSwgMHgxNywgMHg4NywgMHgwMiwKLSAgMHhlZSwgMHg3MiwgMHg4NywgMHhhMSwgMHg5OCwgMHhlOSwgMHgyNywgMHg5MywgMHg3ZCwgMHhkYiwgMHg1NywgMHgzMSwKLSAgMHhkNywgMHhlNiwgMHg4YSwgMHg4MSwgMHhmYywgMHg2ZSwgMHg2NiwgMHgxOSwgMHhiMywgMHg2ZSwgMHhjMywgMHhlMCwKLSAgMHgxNiwgMHgwNywgMHgyMiwgMHg1YiwgMHg2OCwgMHhmMywgMHgxZSwgMHhjOCwgMHhhZiwgMHg0OSwgMHg0YiwgMHg0YywKLSAgMHhhMywgMHhhZSwgMHhiZCwgMHhlZCwgMHgyYSwgMHgyMCwgMHgyOCwgMHhmOCwgMHhmNSwgMHgzYiwgMHhmZCwgMHg0YSwKLSAgMHhkNSwgMHg3YiwgMHhhYSwgMHg1YiwgMHhiZSwgMHhmMywgMHhmZCwgMHgwYywgMHhhNCwgMHhhYiwgMHg1NSwgMHhmNCwKLSAgMHg0NSwgMHhiOSwgMHg3NSwgMHhjOSwgMHgyYywgMHg3YywgMHgzYywgMHhjNSwgMHhhNywgMHg2NSwgMHg1OSwgMHg3ZCwKLSAgMHhjMSwgMHhiOCwgMHgyYywgMHg4NCwgMHg3OSwgMHg4OSwgMHg3MSwgMHg1ZSwgMHhmYSwgMHhmMSwgMHhkNywgMHg0NCwKLSAgMHgyMSwgMHgzZCwgMHhlYSwgMHg1MiwgMHg4ZSwgMHhiNCwgMHgzYywgMHgzYiwgMHhmYiwgMHhiNiwgMHg2YiwgMHg5YiwKLSAgMHhmMCwgMHg2ZiwgMHhiMywgMHg1YywgMHhlZCwgMHhiYywgMHgzNywgMHgxNywgMHg1YiwgMHhkYiwgMHg0NSwgMHg5YiwKLSAgMHhhZSwgMHg0NywgMHgzYSwgMHg0NSwgMHhmMiwgMHg2MywgMHg0NywgMHhiZCwgMHhiNSwgMHg0OSwgMHg1ZiwgMHgzYSwKLSAgMHg1MiwgMHg3ZCwgMHg2MSwgMHgxYywgMHg4MCwgMHg4ZiwgMHgwMywgMHhiMSwgMHg0YiwgMHhkYywgMHgzYiwgMHhlMCwKLSAgMHg0NCwgMHg2NiwgMHhhZSwgMHg0ZCwgMHg2NSwgMHg5YywgMHg1NCwgMHhiYSwgMHgzOSwgMHg5YywgMHg2NSwgMHg5YiwKLSAgMHhlNiwgMHg0YiwgMHg5MywgMHgwOSwgMHg1YywgMHg0OCwgMHg3ZCwgMHg3NiwgMHgxMiwgMHhjYiwgMHg0NywgMHhjZCwKLSAgMHhlOSwgMHhkMywgMHhhOSwgMHgxYSwgMHhkOCwgMHhkOCwgMHgwMiwgMHhhZSwgMHhjMCwgMHgzYSwgMHg3NywgMHg1NiwKLSAgMHgxNSwgMHgyNiwgMHhlNywgMHgyNywgMHgyNiwgMHg2ZiwgMHg0ZSwgMHgwYSwgMHgxMSwgMHhkMiwgMHg4NCwgMHg1ZSwKLSAgMHgzMiwgMHg3YywgMHg0NywgMHgxNywgMHhmZCwgMHhhYSwgMHhiNiwgMHg3ZCwgMHhmOCwgMHhhNywgMHhiYSwgMHg0NCwKLSAgMHhlMywgMHgyMCwgMHhkNCwgMHgxYywgMHg1YywgMHg3ZiwgMHhmNywgMHg1NSwgMHhiMywgMHhlZiwgMHhjNSwgMHgzZCwKLSAgMHhkNSwgMHgwYiwgMHg5ZSwgMHg1NywgMHhkZiwgMHhkMCwgMHg5MSwgMHg0NSwgMHg2NCwgMHg4ZCwgMHg5YSwgMHgyOCwKLSAgMHg0MywgMHhjOSwgMHg5YSwgMHg0YywgMHhlMywgMHg4ZiwgMHhjOCwgMHhmNiwgMHg1ZCwgMHhmNCwgMHg0NCwgMHg4ZiwKLSAgMHhlMCwgMHgzNCwgMHhlNywgMHg1MCwgMHgxYywgMHg0NSwgMHhiMiwgMHhjOCwgMHhjOSwgMHgzMCwgMHg1YiwgMHhlNiwKLSAgMHgzZiwgMHgxMSwgMHhkNiwgMHhkOSwgMHg5MSwgMHg3MCwgMHg4MiwgMHhlYywgMHg2NiwgMHg5YywgMHg3MywgMHg3YywKLSAgMHg4OSwgMHg1MiwgMHhkMiwgMHg0MCwgMHgyYSwgMHhkNywgMHg1ZCwgMHg2ZSwgMHg4NCwgMHg5MywgMHg3MSwgMHhiZSwKLSAgMHgyZSwgMHhkZiwgMHhjYywgMHgxZiwgMHg2NSwgMHg2YywgMHhhYSwgMHhmZCwgMHhiZiwgMHhjNywgMHgyYSwgMHgxYiwKLSAgMHg0YSwgMHgwMywgMHg1OCwgMHgxMSwgMHgwOSwgMHgxYSwgMHhkZiwgMHg2OSwgMHgyZiwgMHhhZiwgMHhmZCwgMHgzNSwKLSAgMHhlYiwgMHg5YiwgMHg4YywgMHhiZiwgMHhkNCwgMHhlMCwgMHgzZiwgMHhiZCwgMHg5NywgMHhmZSwgMHg1YSwgMHgwMSwKLSAgMHhmYSwgMHg4YSwgMHg0MSwgMHhlNywgMHhlMywgMHgyZiwgMHhmNSwgMHg1OCwgMHgwZiwgMHhlZiwgMHg2NSwgMHhmZiwKLSAgMHgwMCwgMHg5NiwgMHg4ZSwgMHg2ZSwgMHgzMiwgMHhmZiwgMHgwMCwgMHg1MywgMHg4MCwgMHhmZSwgMHhmNiwgMHg1ZiwKLSAgMHhmOSwgMHg2OCwgMHgwNywgMHhlYSwgMHgyOSwgMHgwNywgMHg5YiwgMHg4YywgMHhiZiwgMHhkNCwgMHhlMCwgMHgzZiwKLSAgMHhiZCwgMHg5NywgMHhmZSwgMHg1YSwgMHgzOSwgMHhmOCwgMHhjYiwgMHhmZCwgMHg1NiwgMHgwMywgMHhmYiwgMHhkOSwKLSAgMHg3ZiwgMHhlNSwgMHhhMCwgMHgxZiwgMHhhOCwgMHgyMywgMHg3NCwgMHg4MywgMHhjZiwgMHhjNiwgMHg1ZiwgMHhlYSwKLSAgMHhiMCwgMHgxZiwgMHhkZSwgMHg0YiwgMHhmZiwgMHgwMCwgMHgyZCwgMHgxYywgMHhmYywgMHg2NSwgMHhmZSwgMHhhYiwKLSAgMHgwMSwgMHhmZCwgMHhlNCwgMHhiZiwgMHhmMiwgMHhkMCwgMHgxNCwgMHg5NywgMHgxNSwgMHg3MSwgMHg0OSwgMHg1OCwKLSAgMHg4NywgMHgxMiwgMHhlNSwgMHg1YSwgMHhhYywgMHhjOCwgMHg0YiwgMHgzMSwgMHhmMiwgMHg0OSwgMHgyNiwgMHhmZCwKLSAgMHg4YiwgMHgyZiwgMHg1YSwgMHg0NCwgMHg3YiwgMHhkYiwgMHgyOSwgMHhmNywgMHhmOCwgMHhiYiwgMHhlZSwgMHg0YSwKLSAgMHgyNCwgMHhiNSwgMHhiNCwgMHhmNSwgMHhlOCwgMHg0OSwgMHgwMCwgMHg3YywgMHgxYSwgMHhiMywgMHgyYywgMHgzNywKLSAgMHg5YiwgMHgwNiwgMHg1NSwgMHg4MywgMHhjNSwgMHhjOCwgMHg1YSwgMHhiNywgMHgzMSwgMHgyOSwgMHhmOSwgMHg0YywKLSAgMHgzMywgMHhlNCwgMHhlMiwgMHg0MywgMHg4YSwgMHg0MiwgMHgzYywgMHhlNSwgMHhhNSwgMHgxYywgMHg4ZiwgMHg4MCwKLSAgMHg0NywgMHgzMCwgMHg2ZCwgMHg0NCwgMHhlYywgMHgxZiwgMHhkMSwgMHgyMywgMHhhNSwgMHg3MCwgMHg3MSwgMHg3MywKLSAgMHgwYywgMHhlMiwgMHhiNiwgMHg3NywgMHg4OCwgMHhiOSwgMHg2YiwgMHg5NCwgMHgzMCwgMHhjOCwgMHhmMiwgMHhhMywKLSAgMHgzYywgMHhkYywgMHhkYiwgMHg3YywgMHg5OCwgMHhhZSwgMHhjOSwgMHgwZiwgMHg0NywgMHg5MiwgMHhkMiwgMHhiOSwKLSAgMHg5MCwgMHhiNCwgMHgxNSwgMHgyNywgMHg1YiwgMHhlZiwgMHgxZCwgMHg3YSwgMHg3NSwgMHgzNSwgMHg1ZiwgMHhmMCwKLSAgMHhiNywgMHgyNiwgMHg4ZSwgMHhiYiwgMHg4MiwgMHgxNSwgMHgzNiwgMHgzMCwgMHg4ZCwgMHg2YiwgMHhjYiwgMHg1ZiwKLSAgMHg3NSwgMHg0NiwgMHgzYSwgMHhiYSwgMHgyMiwgMHgwNSwgMHhlOSwgMHgwMywgMHg1MywgMHhhMSwgMHgxMSwgMHhkZSwKLSAgMHg4ZSwgMHhkMCwgMHhlOSwgMHhlNCwgMHgwZSwgMHg5YiwgMHhlNiwgMHgyMCwgMHg3NSwgMHhhZCwgMHhhOSwgMHg2MiwKLSAgMHg1OSwgMHg4MywgMHhlYiwgMHhmNSwgMHgzMiwgMHhhYiwgMHg5OCwgMHhlMiwgMHg2YiwgMHhhNywgMHhkMCwgMHhiYiwKLSAgMHhhZiwgMHhkOCwgMHhjYywgMHhjNywgMHhmMSwgMHhjNywgMHg4NCwgMHg2YiwgMHhmNCwgMHhhYiwgMHg4NCwgMHg4NiwKLSAgMHgzOSwgMHg0MywgMHg5MCwgMHg5MiwgMHg1MCwgMHhkNCwgMHg1NywgMHhkMCwgMHg4NSwgMHgwMiwgMHhhOCwgMHhkZCwKLSAgMHg5MiwgMHg0NiwgMHg5MCwgMHgxNCwgMHgwMSwgMHg0NywgMHg3ZiwgMHgzMCwgMHhkOCwgMHhkOSwgMHgyMiwgMHhiNSwKLSAgMHg4YywgMHhhYSwgMHhkNywgMHg3YSwgMHhiNSwgMHhjZiwgMHg4NSwgMHg3NSwgMHhiNywgMHhhYSwgMHgxZCwgMHg5NSwKLSAgMHhjNCwgMHgyMSwgMHgzMSwgMHhlNSwgMHgxZiwgMHgzNSwgMHgwZiwgMHgzNCwgMHhiNCwgMHgwNSwgMHgyMCwgMHhiNCwKLSAgMHg0MCwgMHgzYiwgMHg3NSwgMHgyYSwgMHhmMywgMHg3OSwgMHgxMywgMHhlNywgMHgyNSwgMHg0MywgMHhiYiwgMHhiYiwKLSAgMHg3YSwgMHgyZCwgMHgzOCwgMHhlYiwgMHg4ZSwgMHg0ZiwgMHg5ZCwgMHgyZSwgMHgzNSwgMHhmMiwgMHgzNCwgMHgwNiwKLSAgMHgyNywgMHhiYywgMHgxZiwgMHg3YywgMHhiNCwgMHg1NSwgMHhkYiwgMHg3MywgMHg4NCwgMHhmMiwgMHhhOSwgMHg0OSwKLSAgMHg0MiwgMHhkNCwgMHg1YiwgMHg2ZCwgMHg2NCwgMHg2YiwgMHg5OSwgMHg0MSwgMHgyNCwgMHgxMiwgMHgzNywgMHhhMCwKLSAgMHg2OSwgMHg4NiwgMHgxYywgMHgzYywgMHgzNiwgMHhjNCwgMHg5NiwgMHg5NywgMHgwYSwgMHgwZCwgMHhiZCwgMHgwZiwKLSAgMHgzMiwgMHhkOCwgMHg2ZCwgMHgwZSwgMHgzNCwgMHhjYSwgMHg1NCwgMHhlOSwgMHgwMCwgMHg2YiwgMHhhYSwgMHhiNSwKLSAgMHhiMiwgMHg3NSwgMHhlMiwgMHg0ZCwgMHg0NiwgMHg4NiwgMHg5ZSwgMHgzMSwgMHhiOCwgMHhmNywgMHg4OSwgMHhhYywKLSAgMHhiNywgMHhiMCwgMHhhZiwgMHg4YiwgMHhkOCwgMHhmMywgMHg1NiwgMHhlNywgMHhmYiwgMHhhZSwgMHg5NywgMHgwYywKLSAgMHg2NSwgMHhkYywgMHg2MywgMHhiMiwgMHhmNCwgMHhjNiwgMHhlNSwgMHhjOCwgMHgwNSwgMHhhZSwgMHhkYywgMHgzNCwKLSAgMHg4NCwgMHgxNywgMHgwYiwgMHg0OSwgMHg2YywgMHgyYywgMHgyZiwgMHg0OSwgMHgxYiwgMHg2ZiwgMHg5YywgMHgyNywKLSAgMHg3YiwgMHgzYiwgMHgxYiwgMHhkNSwgMHgzNCwgMHhkYSwgMHhiMCwgMHg3YywgMHg3YSwgMHgyNCwgMHg2NiwgMHg1MywKLSAgMHgyZSwgMHgxMywgMHg1NywgMHgwOSwgMHgyOCwgMHhkYSwgMHhkZCwgMHg3ZCwgMHhmNCwgMHgwMywgMHhkYiwgMHgzOCwKLSAgMHg1NCwgMHg1NCwgMHhhNSwgMHhhOSwgMHgzZiwgMHgwNywgMHg3YiwgMHg1MSwgMHhkNywgMHg0ZSwgMHg5ZCwgMHgwMCwKLSAgMHhlZSwgMHhhMiwgMHg0NiwgMHg1NCwgMHg4MCwgMHg5ZCwgMHg0NSwgMHg4YSwgMHg0OCwgMHgwMCwgMHhmOSwgMHhjZSwKLSAgMHgyYiwgMHhhMCwgMHhmNiwgMHhlYiwgMHhmZiwgMHgwMCwgMHgzNSwgMHg1ZCwgMHg1ZCwgMHhiOCwgMHhiMywgMHgwNiwKLSAgMHhlNSwgMHg3MCwgMHg1NSwgMHhhYSwgMHhjMiwgMHhlYywgMHhkYywgMHhhMiwgMHhlMiwgMHgwOSwgMHgwNiwgMHgwNSwKLSAgMHg4MSwgMHhhZSwgMHhkZiwgMHg5MywgMHhjMywgMHhkZiwgMHgxYywgMHgwNCwgMHgzNiwgMHhkOCwgMHhkZiwgMHg4YSwKLSAgMHhkNiwgMHgyYiwgMHg0ZiwgMHhiMywgMHhjZiwgMHg5OSwgMHhlYywgMHg2NywgMHhmNiwgMHg4OCwgMHg3ZSwgMHgwZCwKLSAgMHhjYiwgMHg2NiwgMHg3ZCwgMHhlMiwgMHhkZCwgMHgwMSwgMHgxZCwgMHg5OSwgMHg3MCwgMHgyZCwgMHg2OSwgMHhlZSwKLSAgMHg2ZCwgMHhiZSwgMHhiYSwgMHhmZiwgMHgwMCwgMHhiMCwgMHhhNCwgMHgxYywgMHhkNywgMHg4OSwgMHg5NiwgMHhmYiwKLSAgMHg0YiwgMHhhZCwgMHhjNCwgMHg5MywgMHgzOCwgMHgzMSwgMHgyYSwgMHg0NywgMHg0OCwgMHhmMCwgMHgyMiwgMHhiNiwKLSAgMHhhNywgMHhlNiwgMHgzZSwgMHg3YywgMHgzOSwgMHg1YiwgMHg0MCwgMHgyYiwgMHgzZiwgMHhkYywgMHgwNywgMHhhZiwKLSAgMHhjNiwgMHhhOSwgMHgwZSwgMHgyMywgMHg3MSwgMHgzZCwgMHgxNiwgMHg4MiwgMHhhOCwgMHhkOSwgMHg0ZSwgMHg1NSwKLSAgMHgxZSwgMHhjOCwgMHhlOSwgMHgyNSwgMHgzZSwgMHhlMCwgMHg2MiwgMHhlZSwgMHgyMiwgMHg3NSwgMHhjZCwgMHg1ZCwKLSAgMHhmZSwgMHg2MywgMHhkMiwgMHhjZiwgMHhiZCwgMHg0NywgMHgzYiwgMHhlZiwgMHhlNCwgMHgwNSwgMHg1ZCwgMHg3YSwKLSAgMHgxYSwgMHhlMywgMHhlMSwgMHhlNSwgMHg4NywgMHg4YywgMHhiOSwgMHg4NCwgMHg1OSwgMHgzZiwgMHg4YiwgMHhmYywKLSAgMHg1ZSwgMHgxNywgMHgwYSwgMHhlYywgMHgzMiwgMHg5MiwgMHhhNSwgMHgyYSwgMHhlYiwgMHgyNCwgMHgyZCwgMHhjYiwKLSAgMHg4YywgMHhkZSwgMHhhMywgMHg1ZCwgMHhhMywgMHhlZSwgMHg2ZCwgMHhlNSwgMHg2ZiwgMHg1YiwgMHhkYSwgMHg0MiwKLSAgMHg1MywgMHg0ZCwgMHg1NCwgMHhlOSwgMHhmOCwgMHg1NiwgMHg1ZiwgMHg5YiwgMHgxYSwgMHgyYSwgMHg1NCwgMHhmMSwKLSAgMHgzYywgMHgyMSwgMHhjYiwgMHgzOSwgMHhjZSwgMHhhNSwgMHg1YiwgMHgyMywgMHgyZSwgMHg1ZSwgMHg1MywgMHg3ZCwKLSAgMHg4NywgMHhjMywgMHhkOCwgMHg0YiwgMHg0NywgMHgzYSwgMHgxYSwgMHg5MiwgMHg1MywgMHgzMiwgMHhmYiwgMHgyMSwKLSAgMHgzZCwgMHgzYSwgMHhhMiwgMHgzMiwgMHg0OSwgMHg0MywgMHgzYiwgMHhkZSwgMHhiOSwgMHg5NiwgMHg1NCwgMHg0NywKLSAgMHg4OCwgMHgxNCwgMHg4NSwgMHg4OCwgMHg1ZCwgMHhmOCwgMHg4YiwgMHg5YSwgMHhjZSwgMHg3NywgMHhmMSwgMHgyOSwKLSAgMHg4MywgMHgzOSwgMHg2MSwgMHg4ZCwgMHgyMywgMHg2OCwgMHg3ZiwgMHgzMSwgMHhiZiwgMHhhYiwgMHhiNywgMHg5YywKLSAgMHhmYSwgMHg3NywgMHhhMywgMHhhNywgMHg5YywgMHgwNSwgMHgyOSwgMHgxZiwgMHhhOCwgMHhkOCwgMHg1NiwgMHhiNSwKLSAgMHhkMywgMHg1NCwgMHhjZiwgMHg4MCwgMHhmZSwgMHgwYiwgMHhkNywgMHhjYywgMHg3NiwgMHhmMCwgMHg2ZiwgMHg3NywKLSAgMHhhNCwgMHhlMiwgMHhiOSwgMHg3NSwgMHhkNywgMHhiNSwgMHhlZCwgMHg0MywgMHhkNywgMHg3OSwgMHgxMiwgMHg5NCwKLSAgMHhkZiwgMHgzMCwgMHgzYiwgMHgwYSwgMHgyZCwgMHg4NCwgMHhlOSwgMHg2NywgMHhlNywgMHg5NSwgMHgwYSwgMHhiYywKLSAgMHg1OSwgMHg2ZiwgMHg4YywgMHgwYywgMHgzNCwgMHg4NiwgMHg5OSwgMHg4ZCwgMHhjMywgMHhmNiwgMHhkYiwgMHg2ZCwKLSAgMHgyMSwgMHgyOCwgMHg0YSwgMHg1NywgMHgyZCwgMHgyMSwgMHgyOSwgMHgxZCwgMHhjMCwgMHgwMCwgMHg5ZSwgMHg4MywKLSAgMHhhNywgMHg3NSwgMHg2NSwgMHgzYSwgMHhiMiwgMHg5ZiwgMHgyYywgMHhkYSwgMHgxNCwgMHhhMywgMHgwZSwgMHgwYSwKLSAgMHhmYiwgMHgwMiwgMHhmYywgMHgxOCwgMHhmMSwgMHhkOCwgMHhkNywgMHgwNCwgMHhlNCwgMHgxYywgMHg0YSwgMHhiYiwKLSAgMHg0ZSwgMHhjZSwgMHhlZiwgMHhlYiwgMHg1NywgMHgzYiwgMHg4YiwgMHg5YywgMHhlYSwgMHg4YywgMHg2MCwgMHhhZSwKLSAgMHg5ZCwgMHgzOSwgMHgwOSwgMHgyNSwgMHg3YSwgMHgzZCwgMHgzNCwgMHhhMiwgMHg1MywgMHhhMSwgMHhmMCwgMHg0NSwKLSAgMHg1ZiwgMHg3MCwgMHhlMSwgMHhjNSwgMHg4NywgMHgxMSwgMHhhOCwgMHg5MCwgMHhlMywgMHhiNSwgMHgxYSwgMHgzYiwKLSAgMHgyOSwgMHgwOCwgMHg2ZCwgMHhhNiwgMHg5MCwgMHgxMCwgMHg5NCwgMHgyNCwgMHg3OCwgMHgwMCwgMHgzYSwgMHgwMSwKLSAgMHhlYywgMHhhNCwgMHg3ZCwgMHhmMSwgMHg5MywgMHhmYSwgMHhhYywgMHgwNywgMHhmNywgMHg5MiwgMHhmZiwgMHgwMCwKLSAgMHhjYiwgMHg1OSwgMHgwYSwgMHhlMywgMHgyZiwgMHhmNSwgMHgzOCwgMHgwZiwgMHhlZiwgMHg2NSwgMHhmZiwgMHgwMCwKLSAgMHg5NiwgMHhiMywgMHgyZSwgMHgzZSwgMHhlOCwgMHg1NiwgMHg0NywgMHg0YSwgMHg0MSwgMHhlNywgMHhlMywgMHgyZiwKLSAgMHhmNSwgMHg1OCwgMHgwZiwgMHhlZiwgMHg2NSwgMHhmZiwgMHgwMCwgMHg5NiwgMHg4ZSwgMHg3ZSwgMHgzMiwgMHhmZiwKLSAgMHgwMCwgMHg1NSwgMHg4MCwgMHhmZSwgMHhmMiwgMHg1ZiwgMHhmOSwgMHg2OCwgMHgwZiwgMHg3YywgMHg2NSwgMHhmOCwKLSAgMHg5NiwgMHgyZiwgMHhmYiwgMHg1NSwgMHg2YywgMHhmYiwgMHhmMSwgMHg0ZiwgMHg3NSwgMHg1OCwgMHhkZCwgMHhhYywKLSAgMHg1YywgMHg0ZCwgMHhjOCwgMHg2NiwgMHg1OSwgMHg1YSwgMHhiZSwgMHhhZiwgMHgxMCwgMHg2MiwgMHgwNCwgMHgxYiwKLSAgMHhiNCwgMHg2YiwgMHg4MywgMHhhNiwgMHgxMiwgMHhhNCwgMHgxNywgMHg1NCwgMHgxOSwgMHg1OCwgMHg1NywgMHgyYSwKLSAgMHg0MiwgMHhjNiwgMHhiYSwgMHhlYSwgMHhhYywgMHhlYSwgMHgwMCwgMHhhMiwgMHhiMCwgMHg0OSwgMHhmMCwgMHhhMiwKLSAgMHg4MCwgMHhjZCwgMHgxYSwgMHhlYiwgMHg0NSwgMHgxNCwgMHgwMSwgMHg0NSwgMHgxNCwgMHg1MCwgMHgwNSwgMHgxNCwKLSAgMHg1MSwgMHg0MCwgMHgxNCwgMHg1MSwgMHg0NSwgMHgwMCwgMHg1MSwgMHg0NSwgMHgxNCwgMHgwMSwgMHg1ZiwgMHgyYiwKLSAgMHg3ZSwgMHgxMSwgMHgzOCwgMHg3NSwgMHhjZiwgMHgwYSwgMHhjOSwgMHhlZiwgMHg3OSwgMHg3YywgMHgxYywgMHg3ZCwKLSAgMHhlYiwgMHhmNiwgMHgwOSwgMHg5MSwgMHg4NiwgMHg5YywgMHhjOCwgMHhlMCwgMHg0MywgMHg1MSwgMHg0YywgMHhhOCwKLSAgMHgxMiwgMHhkYiwgMHhkZiwgMHgyNCwgMHhlNiwgMHgwOCwgMHhmOCwgMHgwYiwgMHgxZiwgMHgwYiwgMHg5OCwgMHg3NCwKLSAgMHhkZiwgMHgzMCwgMHg1NywgMHg0MiwgMHgwZCwgMHg3ZCwgMHg1MywgMHg1OCwgMHg1MiwgMHg0MiwgMHhiYSwgMHgyOCwKLSAgMHg2YywgMHgxZSwgMHg4NCwgMHg1MCwgMHgxZiwgMHgxOSwgMHhlMywgMHgzYywgMHg0NiwgMHhlMiwgMHg1YywgMHg3YiwKLSAgMHgxMiwgMHgyZSwgMHg5NiwgMHg5YiwgMHg2NCwgMHgxZSwgMHgyOCwgMHg2MywgMHhjZCwgMHg2OCwgMHg3OSwgMHg3NSwKLSAgMHhhZCwgMHg2NSwgMHhhYiwgMHg5MywgMHg0OSwgMHhmMCwgMHgxMiwgMHgyMywgMHg4ZCwgMHhhOSwgMHgwZSwgMHgwMSwKLSAgMHhkZSwgMHg0MiwgMHgwOCwgMHhkZiwgMHhlNywgMHgxMCwgMHg3NiwgMHg1OCwgMHhlYywgMHhkYywgMHg1MiwgMHhjYiwKLSAgMHhiMiwgMHgwOCwgMHhjYiwgMHg3ZSwgMHhkNSwgMHhjMywgMHgyOSwgMHg3NiwgMHg4OCwgMHg2ZCwgMHg5MCwgMHg5NywKLSAgMHhhZSwgMHg5OSwgMHgxYywgMHhlNCwgMHhkYiwgMHhlMSwgMHgzMiwgMHg3NywgMHhkNCwgMHhhOSwgMHg2YSwgMHg0ZSwKLSAgMHhkNSwgMHhlYiwgMHgwOSwgMHhkOSwgMHhmNSwgMHg1NSwgMHg4ZCwgMHhjNCwgMHgwZiwgMHhjMSwgMHhkNywgMHgwNSwKLSAgMHhjOCwgMHhlZSwgMHhjNiwgMHhmYiwgMHg2NSwgMHg3MiwgMHhlMywgMHg4NywgMHhkZSwgMHhmNywgMHhiZiwgMHgyZCwKLSAgMHhiMSwgMHhiZCwgMHhlNCwgMHhmYywgMHhjNywgMHhkMiwgMHhhNCwgMHgwMSwgMHhhZCwgMHhmOCwgMHhlYywgMHg2OCwKLSAgMHg5MywgMHhkNCwgMHg5YSwgMHhlNiwgMHhiNCwgMHg3ZSwgMHgwZCwgMHhiOCwgMHg2YSwgMHhlZSwgMHhjZCwgMHhkZCwKLSAgMHhmMywgMHg0YiwgMHhjZSwgMHg0OSwgMHg5YywgMHhjZSwgMHg2YywgMHgwZSwgMHg1MywgMHg3YywgMHhiOCwgMHgyOSwKLSAgMHhkNiwgMHhjNiwgMHhiZiwgMHg1MCwgMHg2YiwgMHg2MywgMHhhMCwgMHhlOCwgMHg0OSwgMHgxZSwgMHhhYSwgMHhkZCwKLSAgMHg1YywgMHhkNCwgMHg0YiwgMHgxOSwgMHgzMCwgMHg3NiwgMHhkNCwgMHhkYiwgMHhjYiwgMHg0NSwgMHg0ZCwgMHgxNywKLSAgMHgzZiwgMHg3MiwgMHg2ZCwgMHhjMywgMHhkYywgMHhlYiwgMHg0MiwgMHgyZSwgMHg5YywgMHg1ZSwgMHhjOSwgMHgwNCwKLSAgMHg4MCwgMHhmMCwgMHg2MiwgMHhjYywgMHhjYSwgMHhlMSwgMHg1OSwgMHhhMCwgMHgxMCwgMHg4NSwgMHgyMywgMHhiMywKLSAgMHgyYiwgMHhlZiwgMHg3OSwgMHgxZSwgMHg3OSwgMHgyNywgMHg5ZiwgMHhjZCwgMHgzYSwgMHgxZCwgMHhkNCwgMHhlOSwKLSAgMHgwMywgMHg4MywgMHhmYywgMHg1NSwgMHhjYywgMHhlZCwgMHhjZCwgMHhjMywgMHhjZCwgMHhmMiwgMHhlOCwgMHgxOCwKLSAgMHg1NiwgMHgzYywgMHg0NywgMHhmYSwgMHhiNywgMHg4YSwgMHg0NywgMHgwZCwgMHgyMywgMHg5MywgMHhhNywgMHg5YSwKLSAgMHhiNywgMHgzYiwgMHhiYywgMHgwNywgMHg3ZiwgMHgzOCwgMHhhZiwgMHhhMSwgMHg2YywgMHhmNiwgMHg5YiwgMHg1ZCwKLSAgMHg5YSwgMHgwYSwgMHg2MCwgMHg1YSwgMHgyZCwgMHhkMSwgMHgyZCwgMHhmMSwgMHgxMSwgMHhmMCwgMHg1OSwgMHg4YSwKLSAgMHhjYSwgMHg1YSwgMHg0MCwgMHhmNiwgMHgyNSwgMHgyMCwgMHgwYSwgMHhlYywgMHgwMCwgMHgwYSwgMHhjYSwgMHg1MywKLSAgMHg5NCwgMHhiOSwgMHgzNSwgMHg4YywgMHgyMywgMHgxZSwgMHgxMSwgMHg1ZiwgMHg3MCwgMHhlMywgMHg4MiwgMHhmYywKLSAgMHgzNiwgMHhjMCwgMHg0MywgMHg0ZSwgMHhlMywgMHhkOCwgMHhjYywgMHg1MSwgMHgzNSwgMHhiMSwgMHhmMSwgMHhlOSwKLSAgMHg0MywgMHhiNywgMHg5MiwgMHg0ZiwgMHhhNywgMHg5ZCwgMHg3YiwgMHhlNSwgMHgzZiwgMHgzNywgMHg0MywgMHhkNSwKLSAgMHg1NiwgMHgwZiwgMHgyOCwgMHhhYywgMHg4MSwgMHhhZSwgMHhlYSwgMHgyYSwgMHhhNSwgMHg4YywgMHg2OCwgMHg1NiwKLSAgMHg2OCwgMHhhMiwgMHg4MCwgMHgyOCwgMHhhMiwgMHg4YSwgMHgwMCwgMHhhMiwgMHg4YSwgMHgyOCwgMHgwMywgMHg0MiwKLSAgMHg4YSwgMHgyOCwgMHhhMCwgMHgzMCwgMHg1MiwgMHgwZiwgMHg3ZCwgMHgxNSwgMHg5YSwgMHgyOCwgMHgwMiwgMHg4YSwKLSAgMHgyOCwgMHhhMCwgMHgwYSwgMHgyOCwgMHhhMiwgMHg4MCwgMHgyOCwgMHhhMiwgMHg4YSwgMHgwMCwgMHhhMiwgMHg4YSwKLSAgMHgyOCwgMHgwMiwgMHg4YSwgMHgyOCwgMHhhMCwgMHgwYSwgMHgyOCwgMHhhMiwgMHg4MCwgMHgzNSwgMHg0NSwgMHgxNCwKLSAgMHg1MCwgMHgwNSwgMHgxNCwgMHg1MSwgMHg0MCwgMHgxNCwgMHg1MSwgMHg0NSwgMHgwMCwgMHg1MSwgMHg0NSwgMHgxNCwKLSAgMHgwMSwgMHg0NSwgMHgxNCwgMHg1MCwgMHgwNSwgMHgxNCwgMHg1MSwgMHg0MCwgMHgxNCwgMHg1MSwgMHg0NSwgMHgwMCwKLSAgMHg1MSwgMHg0NSwgMHgxNCwgMHgwNywgMHhmZiwgMHhkOQotfTsKZGlmZiAtLWdpdCBhL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0Zha2VDYW1lcmEuY3BwIGIvY2FtZXJhL2xpYmNhbWVyYXNlcnZpY2UvRmFrZUNhbWVyYS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDM1OTJlYWIuLjAwMDAwMDAKLS0tIGEvY2FtZXJhL2xpYmNhbWVyYXNlcnZpY2UvRmFrZUNhbWVyYS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw0MDQgKzAsMCBAQAotI2RlZmluZSBMT0dfVEFHICJGYWtlQ2FtZXJhIgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSAiRmFrZUNhbWVyYS5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLXN0YXRpYyBpbnQgdGFibGVzX2luaXRpYWxpemVkID0gMDsKLXVpbnQ4X3QgKmdZVGFibGUsICpnQ2JUYWJsZSwgKmdDclRhYmxlOwotCi1zdGF0aWMgaW50Ci1jbGFtcChpbnQgIHgpCi17Ci0gICAgaWYgKHggPiAyNTUpIHJldHVybiAyNTU7Ci0gICAgaWYgKHggPCAwKSAgIHJldHVybiAwOwotICAgIHJldHVybiB4OwotfQotCi0vKiB0aGUgZXF1YXRpb24gdXNlZCBieSB0aGUgdmlkZW8gY29kZSB0byB0cmFuc2xhdGUgWVVWIHRvIFJHQiBsb29rcyBsaWtlIHRoaXMKLSAqCi0gKiAgICBZICA9IChZMCAtIDE2KSprMAotICogICAgQ2IgPSBDYjAgLSAxMjgKLSAqICAgIENyID0gQ3IwIC0gMTI4Ci0gKgotICogICAgRyA9ICggWSAtIGsxKkNyIC0gazIqQ2IgKQotICogICAgUiA9ICggWSArIGszKkNyICkKLSAqICAgIEIgPSAoIFkgKyBrNCpDYiApCi0gKgotICovCi0KLXN0YXRpYyBjb25zdCBkb3VibGUgIGswID0gMS4xNjQ7Ci1zdGF0aWMgY29uc3QgZG91YmxlICBrMSA9IDAuODEzOwotc3RhdGljIGNvbnN0IGRvdWJsZSAgazIgPSAwLjM5MTsKLXN0YXRpYyBjb25zdCBkb3VibGUgIGszID0gMS41OTY7Ci1zdGF0aWMgY29uc3QgZG91YmxlICBrNCA9IDIuMDE4OwotCi0vKiBsZXQncyB0cnkgdG8gZXh0cmFjdCB0aGUgdmFsdWUgb2YgWQotICoKLSAqICAgRyArIGsxL2szKlIgKyBrMi9rNCpCID0gWSooIDEgKyBrMS9rMyArIGsyL2s0ICkKLSAqCi0gKiAgIFkgID0gKCBHICsgazEvazMqUiArIGsyL2s0KkIgKSAvICgxICsgazEvazMgKyBrMi9rNCkKLSAqICAgWTAgPSAoIEcwICsgazEvazMqUjAgKyBrMi9rNCpCMCApIC8gKCgxICsgazEvazMgKyBrMi9rNCkqazApICsgMTYKLSAqCi0gKiBsZXQgZGVmaW5lOgotICogICBrWXIgPSBrMS9rMwotICogICBrWWIgPSBrMi9rNAotICogICBrWXkgPSBrMCAqICggMSArIGtZciArIGtZYiApCi0gKgotICogd2UgaGF2ZToKLSAqICAgIFkgID0gKCBHICsga1lyKlIgKyBrWWIqQiApCi0gKiAgICBZMCA9IGNsYW1wWyBZL2tZeSArIDE2IF0KLSAqLwotCi1zdGF0aWMgY29uc3QgZG91YmxlIGtZciA9IGsxL2szOwotc3RhdGljIGNvbnN0IGRvdWJsZSBrWWIgPSBrMi9rNDsKLXN0YXRpYyBjb25zdCBkb3VibGUga1l5ID0gazAqKCAxLiArIGtZciArIGtZYiApOwotCi1zdGF0aWMgdm9pZAotaW5pdFl0YWIoIHZvaWQgKQotewotICAgIGNvbnN0ICBpbnQgaW1heCA9IChpbnQpKCAoa1lyICsga1liKSooMzEgPDwgMikgKyAoNjEgPDwgMykgKyAwLjEgKTsKLSAgICBpbnQgICAgaTsKLQotICAgIGdZVGFibGUgPSAodWludDhfdCAqKW1hbGxvYyhpbWF4KTsKLQotICAgIGZvcihpPTA7IGk8aW1heDsgaSsrKSB7Ci0gICAgICAgIGludCAgeCA9IChpbnQpKGkva1l5ICsgMTYuNSk7Ci0gICAgICAgIGlmICh4IDwgMTYpIHggPSAxNjsKLSAgICAgICAgZWxzZSBpZiAoeCA+IDIzNSkgeCA9IDIzNTsKLSAgICAgICAgZ1lUYWJsZVtpXSA9ICh1aW50OF90KSB4OwotICAgIH0KLX0KLQotLyoKLSAqICAgdGhlIHNvdXJjZSBpcyBSR0I1NjUsIHNvIGFkanVzdCBmb3IgOC1iaXQgcmFuZ2Ugb2YgaW5wdXQgdmFsdWVzOgotICoKLSAqICAgRyA9IChwaXhlbHMgPj4gMykgJiAweEZDOwotICogICBSID0gKHBpeGVscyA+PiA4KSAmIDB4Rjg7Ci0gKiAgIEIgPSAocGl4ZWxzICYgMHgxZikgPDwgMzsKLSAqCi0gKiAgIFIyID0gKHBpeGVscyA+PiAxMSkgICAgICBSID0gUjIqOAotICogICBCMiA9IChwaXhlbHMgJiAweDFmKSAgICAgQiA9IEIyKjgKLSAqCi0gKiAgIGtZcipSID0ga1lyMipSMiA9PiAga1lyMiA9IGtZcio4Ci0gKiAgIGtZYipCID0ga1liMipCMiA9PiAga1liMiA9IGtZYio4Ci0gKgotICogICB3ZSB3YW50IHRvIHVzZSBpbnRlZ2VyIG11bHRpcGxpY2F0aW9uczoKLSAqCi0gKiAgIFNISUZUMSA9IDkKLSAqCi0gKiAgIChBTFBIQSpSMikgPj4gU0hJRlQxID09IFIqa1lyICA9PiAgQUxQSEEgPSBrWXIqOCooMSA8PCBTSElGVDEpCi0gKgotICogICBBTFBIQSA9IGtZciooMSA8PCAoU0hJRlQxKzMpKQotICogICBCRVRBICA9IGtZYiooMSA8PCAoU0hJRlQxKzMpKQotICovCi0KLXN0YXRpYyBjb25zdCBpbnQgIFNISUZUMSAgPSA5Owotc3RhdGljIGNvbnN0IGludCAgQUxQSEEgICA9IChpbnQpKCBrWXIqKDEgPDwgKFNISUZUMSszKSkgKyAwLjUgKTsKLXN0YXRpYyBjb25zdCBpbnQgIEJFVEEgICAgPSAoaW50KSgga1liKigxIDw8IChTSElGVDErMykpICsgMC41ICk7Ci0KLS8qCi0gKiAgbm93IGxldCdzIHRyeSB0byBnZXQgdGhlIHZhbHVlcyBvZiBDYiBhbmQgQ3IKLSAqCi0gKiAgUi1CID0gKGszKkNyIC0gazQqQ2IpCi0gKgotICogICAgazMqQ3IgPSBrNCpDYiArIChSLUIpCi0gKiAgICBrNCpDYiA9IGszKkNyIC0gKFItQikKLSAqCi0gKiAgUi1HID0gKGsxK2szKSpDciArIGsyKkNiCi0gKiAgICAgID0gKGsxK2szKSpDciArIGsyL2s0KihrMypDciAtIChSLUIpL2swKQotICogICAgICA9IChrMSArIGszICsgazIqazMvazQpKkNyIC0gazIvazQqKFItQikKLSAqCi0gKiAga1JyKkNyID0gKFItRykgKyBrWWIqKFItQikKLSAqCi0gKiAgQ3IgID0gKChSLUcpICsga1liKihSLUIpKS9rUnIKLSAqICBDcjAgPSBjbGFtcChDciArIDEyOCkKLSAqLwotCi1zdGF0aWMgY29uc3QgZG91YmxlICBrUnIgPSAoazEgKyBrMyArIGsyKmszL2s0KTsKLQotc3RhdGljIHZvaWQKLWluaXRDcnRhYiggdm9pZCApCi17Ci0gICAgdWludDhfdCAqcFRhYmxlOwotICAgIGludCBpOwotCi0gICAgZ0NyVGFibGUgPSAodWludDhfdCAqKW1hbGxvYyg3NjgqMik7Ci0KLSAgICBwVGFibGUgPSBnQ3JUYWJsZSArIDM4NDsKLSAgICBmb3IoaT0tMzg0OyBpPDM4NDsgaSsrKQotICAgICAgICBwVGFibGVbaV0gPSAodWludDhfdCkgY2xhbXAoIGkva1JyICsgMTI4LjUgKTsKLX0KLQotLyoKLSAqICBCLUcgPSAoazIgKyBrNCkqQ2IgKyBrMSpDcgotICogICAgICA9IChrMiArIGs0KSpDYiArIGsxL2szKihrNCpDYiArIChSLUIpKQotICogICAgICA9IChrMiArIGs0ICsgazEqazQvazMpKkNiICsgazEvazMqKFItQikKLSAqCi0gKiAga0JiKkNiID0gKEItRykgLSBrWXIqKFItQikKLSAqCi0gKiAgQ2IgICA9ICgoQi1HKSAtIGtZciooUi1CKSkva0JiCi0gKiAgQ2IwICA9IGNsYW1wKENiICsgMTI4KQotICoKLSAqLwotCi1zdGF0aWMgY29uc3QgZG91YmxlICBrQmIgPSAoazIgKyBrNCArIGsxKms0L2szKTsKLQotc3RhdGljIHZvaWQKLWluaXRDYnRhYiggdm9pZCApCi17Ci0gICAgdWludDhfdCAqcFRhYmxlOwotICAgIGludCBpOwotCi0gICAgZ0NiVGFibGUgPSAodWludDhfdCAqKW1hbGxvYyg3NjgqMik7Ci0KLSAgICBwVGFibGUgPSBnQ2JUYWJsZSArIDM4NDsKLSAgICBmb3IoaT0tMzg0OyBpPDM4NDsgaSsrKQotICAgICAgICBwVGFibGVbaV0gPSAodWludDhfdCkgY2xhbXAoIGkva0JiICsgMTI4LjUgKTsKLX0KLQotLyoKLSAqICAgU0hJRlQyID0gMTYKLSAqCi0gKiAgIERFTFRBID0ga1liKigxIDw8IFNISUZUMikKLSAqICAgR0FNTUEgPSBrWXIqKDEgPDwgU0hJRlQyKQotICovCi0KLXN0YXRpYyBjb25zdCBpbnQgIFNISUZUMiA9IDE2Owotc3RhdGljIGNvbnN0IGludCAgREVMVEEgID0ga1liKigxIDw8IFNISUZUMik7Ci1zdGF0aWMgY29uc3QgaW50ICBHQU1NQSAgPSBrWXIqKDEgPDwgU0hJRlQyKTsKLQotaW50MzJfdCBjY3JnYjE2dG95dXZfd29fY29sb3JrZXkodWludDhfdCAqcmdiMTYsdWludDhfdCAqeXV2NDIyLHVpbnQzMl90ICpwYXJhbSx1aW50OF90ICp0YWJsZVtdKQotewotICAgIHVpbnQxNl90ICppbnB1dFJHQiA9ICh1aW50MTZfdCopcmdiMTY7Ci0gICAgdWludDhfdCAqb3V0WVVWID0gIHl1djQyMjsKLSAgICBpbnQzMl90IHdpZHRoX2RzdCA9IHBhcmFtWzBdOwotICAgIGludDMyX3QgaGVpZ2h0X2RzdCA9IHBhcmFtWzFdOwotICAgIGludDMyX3QgcGl0Y2hfZHN0ID0gcGFyYW1bMl07Ci0gICAgaW50MzJfdCBtaGVpZ2h0X2RzdCA9IHBhcmFtWzNdOwotICAgIGludDMyX3QgcGl0Y2hfc3JjID0gcGFyYW1bNF07Ci0gICAgdWludDhfdCAqeV90YWIgPSB0YWJsZVswXTsKLSAgICB1aW50OF90ICpjYl90YWIgPSB0YWJsZVsxXTsKLSAgICB1aW50OF90ICpjcl90YWIgPSB0YWJsZVsyXTsKLQotICAgIGludDMyX3Qgc2l6ZTE2ID0gcGl0Y2hfZHN0Km1oZWlnaHRfZHN0OwotICAgIGludDMyX3QgaSxqLGNvdW50OwotICAgIGludDMyX3QgaWxpbWl0LGpsaW1pdDsKLSAgICB1aW50OF90ICp0ZW1wWSwqdGVtcFUsKnRlbXBWOwotICAgIHVpbnQxNl90IHBpeGVsczsKLSAgICBpbnQgICB0bXA7Ci11aW50MzJfdCB0ZW1wOwotCi0gICAgdGVtcFkgPSBvdXRZVVY7Ci0gICAgdGVtcFUgPSBvdXRZVVYgKyAoaGVpZ2h0X2RzdCAqIHBpdGNoX2RzdCk7Ci0gICAgdGVtcFYgPSB0ZW1wVSArIDE7Ci0KLSAgICBqbGltaXQgPSBoZWlnaHRfZHN0OwotICAgIGlsaW1pdCA9IHdpZHRoX2RzdDsKLQotICAgIGZvcihqPTA7IGo8amxpbWl0OyBqKz0xKQotICAgIHsKLSAgICAgICAgZm9yIChpPTA7IGk8aWxpbWl0OyBpKz0yKQotICAgICAgICB7Ci0gICAgICAgICAgICBpbnQzMl90ICAgR19kcyA9IDAsIEJfZHMgPSAwLCBSX2RzID0gMDsKLSAgICAgICAgICAgIHVpbnQ4X3QgICB5MCwgeTEsIHUsIHY7Ci0KLSAgICAgICAgICAgIHBpeGVscyA9ICBpbnB1dFJHQltpXTsKLSAgICAgICAgICAgIHRlbXAgPSAoQUxQSEEqKHBpeGVscyAmIDB4MDAxRikgKyBCRVRBKihwaXhlbHM+PjExKSApOwotICAgICAgICAgICAgeTAgICA9IHlfdGFiWyh0ZW1wPj5TSElGVDEpICsgKChwaXhlbHM+PjMpICYgMHgwMEZDKV07Ci0KLSAgICAgICAgICAgIEdfZHMgICAgKz0gKHBpeGVscz4+MSkgJiAweDAzRTA7Ci0gICAgICAgICAgICBCX2RzICAgICs9IChwaXhlbHM8PDUpICYgMHgwM0UwOwotICAgICAgICAgICAgUl9kcyAgICArPSAocGl4ZWxzPj42KSAmIDB4MDNFMDsKLQotICAgICAgICAgICAgcGl4ZWxzID0gIGlucHV0UkdCW2krMV07Ci0gICAgICAgICAgICB0ZW1wID0gKEFMUEhBKihwaXhlbHMgJiAweDAwMUYpICsgQkVUQSoocGl4ZWxzPj4xMSkgKTsKLSAgICAgICAgICAgIHkxICAgPSB5X3RhYlsodGVtcD4+U0hJRlQxKSArICgocGl4ZWxzPj4zKSAmIDB4MDBGQyldOwotCi0gICAgICAgICAgICBHX2RzICAgICs9IChwaXhlbHM+PjEpICYgMHgwM0UwOwotICAgICAgICAgICAgQl9kcyAgICArPSAocGl4ZWxzPDw1KSAmIDB4MDNFMDsKLSAgICAgICAgICAgIFJfZHMgICAgKz0gKHBpeGVscz4+NikgJiAweDAzRTA7Ci0KLSAgICAgICAgICAgIFJfZHMgPj49IDE7Ci0gICAgICAgICAgICBCX2RzID4+PSAxOwotICAgICAgICAgICAgR19kcyA+Pj0gMTsKLQotICAgICAgICAgICAgdG1wID0gUl9kcyAtIEJfZHM7Ci0KLSAgICAgICAgICAgIHUgPSBjYl90YWJbKCgoUl9kcy1HX2RzKTw8U0hJRlQyKSArIERFTFRBKnRtcCk+PihTSElGVDIrMildOwotICAgICAgICAgICAgdiA9IGNyX3RhYlsoKChCX2RzLUdfZHMpPDxTSElGVDIpIC0gR0FNTUEqdG1wKT4+KFNISUZUMisyKV07Ci0KLSAgICAgICAgICAgIHRlbXBZWzBdID0geTA7Ci0gICAgICAgICAgICB0ZW1wWVsxXSA9IHkxOwotICAgICAgICAgICAgdGVtcFVbMF0gPSB1OwotICAgICAgICAgICAgdGVtcFZbMF0gPSB2OwotCi0gICAgICAgICAgICB0ZW1wWSArPSAyOwotICAgICAgICAgICAgdGVtcFUgKz0gMjsKLSAgICAgICAgICAgIHRlbXBWICs9IDI7Ci0gICAgICAgIH0KLQotICAgICAgICBpbnB1dFJHQiArPSBwaXRjaF9zcmM7Ci0gICAgfQotCi0gICAgcmV0dXJuIDE7Ci19Ci0KLSNkZWZpbmUgbWluKGEsYikgKChhKTwoYik/KGEpOihiKSkKLSNkZWZpbmUgbWF4KGEsYikgKChhKT4oYik/KGEpOihiKSkKLQotc3RhdGljIHZvaWQgY29udmVydF9yZ2IxNl90b195dXY0MjIodWludDhfdCAqcmdiLCB1aW50OF90ICp5dXYsIGludCB3aWR0aCwgaW50IGhlaWdodCkKLXsKLSAgICBpZiAoIXRhYmxlc19pbml0aWFsaXplZCkgewotICAgICAgICBpbml0WXRhYigpOwotICAgICAgICBpbml0Q3J0YWIoKTsKLSAgICAgICAgaW5pdENidGFiKCk7Ci0gICAgICAgIHRhYmxlc19pbml0aWFsaXplZCA9IDE7Ci0gICAgfQotCi0gICAgdWludDMyX3QgcGFyYW1bNl07Ci0gICAgcGFyYW1bMF0gPSAodWludDMyX3QpIHdpZHRoOwotICAgIHBhcmFtWzFdID0gKHVpbnQzMl90KSBoZWlnaHQ7Ci0gICAgcGFyYW1bMl0gPSAodWludDMyX3QpIHdpZHRoOwotICAgIHBhcmFtWzNdID0gKHVpbnQzMl90KSBoZWlnaHQ7Ci0gICAgcGFyYW1bNF0gPSAodWludDMyX3QpIHdpZHRoOwotICAgIHBhcmFtWzVdID0gKHVpbnQzMl90KSAwOwotCi0gICAgdWludDhfdCAqdGFibGVbM107Ci0gICAgdGFibGVbMF0gPSBnWVRhYmxlOwotICAgIHRhYmxlWzFdID0gZ0NiVGFibGUgKyAzODQ7Ci0gICAgdGFibGVbMl0gPSBnQ3JUYWJsZSArIDM4NDsKLQotICAgIGNjcmdiMTZ0b3l1dl93b19jb2xvcmtleShyZ2IsIHl1diwgcGFyYW0sIHRhYmxlKTsKLX0KLQotY29uc3QgaW50IEZha2VDYW1lcmE6OmtSZWQ7Ci1jb25zdCBpbnQgRmFrZUNhbWVyYTo6a0dyZWVuOwotY29uc3QgaW50IEZha2VDYW1lcmE6OmtCbHVlOwotCi1GYWtlQ2FtZXJhOjpGYWtlQ2FtZXJhKGludCB3aWR0aCwgaW50IGhlaWdodCkKLSAgICAgICAgICA6IG1UbXBSZ2IxNkJ1ZmZlcigwKQotewotICAgIHNldFNpemUod2lkdGgsIGhlaWdodCk7Ci19Ci0KLUZha2VDYW1lcmE6On5GYWtlQ2FtZXJhKCkKLXsKLSAgICBkZWxldGVbXSBtVG1wUmdiMTZCdWZmZXI7Ci19Ci0KLXZvaWQgRmFrZUNhbWVyYTo6c2V0U2l6ZShpbnQgd2lkdGgsIGludCBoZWlnaHQpCi17Ci0gICAgbVdpZHRoID0gd2lkdGg7Ci0gICAgbUhlaWdodCA9IGhlaWdodDsKLSAgICBtQ291bnRlciA9IDA7Ci0gICAgbUNoZWNrWCA9IDA7Ci0gICAgbUNoZWNrWSA9IDA7Ci0KLSAgICAvLyBUaGlzIHdpbGwgY2F1c2UgaXQgdG8gYmUgcmVhbGxvY2F0ZWQgb24gdGhlIG5leHQgY2FsbAotICAgIC8vIHRvIGdldE5leHRGcmFtZUFzWXV2NDIyKCkuCi0gICAgZGVsZXRlW10gbVRtcFJnYjE2QnVmZmVyOwotICAgIG1UbXBSZ2IxNkJ1ZmZlciA9IDA7Ci19Ci0KLXZvaWQgRmFrZUNhbWVyYTo6Z2V0TmV4dEZyYW1lQXNSZ2I1NjUodWludDE2X3QgKmJ1ZmZlcikKLXsKLSAgICBpbnQgc2l6ZSA9IG1XaWR0aCAvIDEwOwotCi0gICAgZHJhd0NoZWNrZXJib2FyZChidWZmZXIsIHNpemUpOwotCi0gICAgaW50IHggPSAoKG1Db3VudGVyKjMpJjI1NSk7Ci0gICAgaWYoeD4xMjgpIHggPSAyNTUgLSB4OwotICAgIGludCB5ID0gKChtQ291bnRlcio1KSYyNTUpOwotICAgIGlmKHk+MTI4KSB5ID0gMjU1IC0geTsKLQotICAgIGRyYXdTcXVhcmUoYnVmZmVyLCB4KnNpemUvMzIsIHkqc2l6ZS8zMiwgKHNpemUqNSk+PjEsIChtQ291bnRlciYweDEwMCk/a1JlZDprR3JlZW4sIGtCbHVlKTsKLQotICAgIG1Db3VudGVyKys7Ci19Ci0KLXZvaWQgRmFrZUNhbWVyYTo6Z2V0TmV4dEZyYW1lQXNZdXY0MjIodWludDhfdCAqYnVmZmVyKQotewotICAgIGlmIChtVG1wUmdiMTZCdWZmZXIgPT0gMCkKLSAgICAgICAgbVRtcFJnYjE2QnVmZmVyID0gbmV3IHVpbnQxNl90W21XaWR0aCAqIG1IZWlnaHRdOwotCi0gICAgZ2V0TmV4dEZyYW1lQXNSZ2I1NjUobVRtcFJnYjE2QnVmZmVyKTsKLSAgICBjb252ZXJ0X3JnYjE2X3RvX3l1djQyMigodWludDhfdCopbVRtcFJnYjE2QnVmZmVyLCBidWZmZXIsIG1XaWR0aCwgbUhlaWdodCk7Ci19Ci0KLXZvaWQgRmFrZUNhbWVyYTo6ZHJhd1NxdWFyZSh1aW50MTZfdCAqZHN0LCBpbnQgeCwgaW50IHksIGludCBzaXplLCBpbnQgY29sb3IsIGludCBzaGFkb3cpCi17Ci0gICAgaW50IHNxdWFyZV94c3RvcCwgc3F1YXJlX3lzdG9wLCBzaGFkb3dfeHN0b3AsIHNoYWRvd195c3RvcDsKLQotICAgIHNxdWFyZV94c3RvcCA9IG1pbihtV2lkdGgsIHgrc2l6ZSk7Ci0gICAgc3F1YXJlX3lzdG9wID0gbWluKG1IZWlnaHQsIHkrc2l6ZSk7Ci0gICAgc2hhZG93X3hzdG9wID0gbWluKG1XaWR0aCwgeCtzaXplKyhzaXplLzQpKTsKLSAgICBzaGFkb3dfeXN0b3AgPSBtaW4obUhlaWdodCwgeStzaXplKyhzaXplLzQpKTsKLQotICAgIC8vIERvIHRoZSBzaGFkb3cuCi0gICAgdWludDE2X3QgKnNoID0gJmRzdFsoeSsoc2l6ZS80KSkqbVdpZHRoXTsKLSAgICBmb3IgKGludCBqID0geSArIChzaXplLzQpOyBqIDwgc2hhZG93X3lzdG9wOyBqKyspIHsKLSAgICAgICAgZm9yIChpbnQgaSA9IHggKyAoc2l6ZS80KTsgaSA8IHNoYWRvd194c3RvcDsgaSsrKSB7Ci0gICAgICAgICAgICBzaFtpXSAmPSBzaGFkb3c7Ci0gICAgICAgIH0KLSAgICAgICAgc2ggKz0gbVdpZHRoOwotICAgIH0KLQotICAgIC8vIERyYXcgdGhlIHNxdWFyZS4KLSAgICB1aW50MTZfdCAqc3EgPSAmZHN0W3kqbVdpZHRoXTsKLSAgICBmb3IgKGludCBqID0geTsgaiA8IHNxdWFyZV95c3RvcDsgaisrKSB7Ci0gICAgICAgIGZvciAoaW50IGkgPSB4OyBpIDwgc3F1YXJlX3hzdG9wOyBpKyspIHsKLSAgICAgICAgICAgIHNxW2ldID0gY29sb3I7Ci0gICAgICAgIH0KLSAgICAgICAgc3EgKz0gbVdpZHRoOwotICAgIH0KLX0KLQotdm9pZCBGYWtlQ2FtZXJhOjpkcmF3Q2hlY2tlcmJvYXJkKHVpbnQxNl90ICpkc3QsIGludCBzaXplKQotewotICAgIGJvb2wgYmxhY2sgPSB0cnVlOwotCi0gICAgaWYoKG1DaGVja1gvc2l6ZSkmMSkKLSAgICAgICAgYmxhY2sgPSBmYWxzZTsKLSAgICBpZigobUNoZWNrWS9zaXplKSYxKQotICAgICAgICBibGFjayA9ICFibGFjazsKLQotICAgIGludCBjb3VudHkgPSBtQ2hlY2tZJXNpemU7Ci0gICAgaW50IGNoZWNreHJlbWFpbmRlciA9IG1DaGVja1glc2l6ZTsKLQotICAgIGZvcihpbnQgeT0wO3k8bUhlaWdodDt5KyspIHsKLSAgICAgICAgaW50IGNvdW50eCA9IGNoZWNreHJlbWFpbmRlcjsKLSAgICAgICAgYm9vbCBjdXJyZW50ID0gYmxhY2s7Ci0gICAgICAgIGZvcihpbnQgeD0wO3g8bVdpZHRoO3grKykgewotICAgICAgICAgICAgZHN0W3kqbVdpZHRoK3hdID0gY3VycmVudD8wOjB4ZmZmZjsKLSAgICAgICAgICAgIGlmKGNvdW50eCsrID49IHNpemUpIHsKLSAgICAgICAgICAgICAgICBjb3VudHg9MDsKLSAgICAgICAgICAgICAgICBjdXJyZW50ID0gIWN1cnJlbnQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgaWYoY291bnR5KysgPj0gc2l6ZSkgewotICAgICAgICAgICAgY291bnR5PTA7Ci0gICAgICAgICAgICBibGFjayA9ICFibGFjazsKLSAgICAgICAgfQotICAgIH0KLSAgICBtQ2hlY2tYICs9IDM7Ci0gICAgbUNoZWNrWSsrOwotfQotCi0KLXN0YXR1c190IEZha2VDYW1lcmE6OmR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKQotewotICAgIGNvbnN0IHNpemVfdCBTSVpFID0gMjU2OwotICAgIGNoYXIgYnVmZmVyW1NJWkVdOwotICAgIFN0cmluZzggcmVzdWx0OwotICAgIHNucHJpbnRmKGJ1ZmZlciwgMjU1LCAiIHdpZHRoIHggaGVpZ2h0ICglZCB4ICVkKSwgY291bnRlciAoJWQpLCBjaGVjayB4LXkgY29vcmRpbmF0ZSglZCwgJWQpXG4iLCBtV2lkdGgsIG1IZWlnaHQsIG1Db3VudGVyLCBtQ2hlY2tYLCBtQ2hlY2tZKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgOjp3cml0ZShmZCwgcmVzdWx0LnN0cmluZygpLCByZXN1bHQuc2l6ZSgpKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0Zha2VDYW1lcmEuaCBiL2NhbWVyYS9saWJjYW1lcmFzZXJ2aWNlL0Zha2VDYW1lcmEuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzdjOTk0Yy4uMDAwMDAwMAotLS0gYS9jYW1lcmEvbGliY2FtZXJhc2VydmljZS9GYWtlQ2FtZXJhLmgKKysrIC9kZXYvbnVsbApAQCAtMSw1MSArMCwwIEBACi0vKgotKioKLSoqIENvcHlyaWdodCAyMDA4LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9IQVJEV0FSRV9GQUtFQ0FNRVJBX0gKLSNkZWZpbmUgQU5EUk9JRF9IQVJEV0FSRV9GQUtFQ0FNRVJBX0gKLQotI2luY2x1ZGUgPHVpL0NhbWVyYUhhcmR3YXJlSW50ZXJmYWNlLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgRmFrZUNhbWVyYSB7Ci1wdWJsaWM6Ci0gICAgRmFrZUNhbWVyYShpbnQgd2lkdGgsIGludCBoZWlnaHQpOwotICAgIH5GYWtlQ2FtZXJhKCk7Ci0KLSAgICB2b2lkIHNldFNpemUoaW50IHdpZHRoLCBpbnQgaGVpZ2h0KTsKLSAgICB2b2lkIGdldE5leHRGcmFtZUFzUmdiNTY1KHVpbnQxNl90ICpidWZmZXIpOwotICAgIHZvaWQgZ2V0TmV4dEZyYW1lQXNZdXY0MjIodWludDhfdCAqYnVmZmVyKTsKLSAgICBzdGF0dXNfdCBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0KLXByaXZhdGU6Ci0gICAgdm9pZCBkcmF3U3F1YXJlKHVpbnQxNl90ICpidWZmZXIsIGludCB4LCBpbnQgeSwgaW50IHNpemUsIGludCBjb2xvciwgaW50IHNoYWRvdyk7Ci0gICAgdm9pZCBkcmF3Q2hlY2tlcmJvYXJkKHVpbnQxNl90ICpidWZmZXIsIGludCBzaXplKTsKLQotICAgIHN0YXRpYyBjb25zdCBpbnQga1JlZCA9IDB4ZjgwMDsKLSAgICBzdGF0aWMgY29uc3QgaW50IGtHcmVlbiA9IDB4MDdjMDsKLSAgICBzdGF0aWMgY29uc3QgaW50IGtCbHVlID0gMHgwMDNlOwotCi0gICAgaW50ICAgICAgICAgbVdpZHRoLCBtSGVpZ2h0OwotICAgIGludCAgICAgICAgIG1Db3VudGVyOwotICAgIGludCAgICAgICAgIG1DaGVja1gsIG1DaGVja1k7Ci0gICAgdWludDE2X3QgICAgKm1UbXBSZ2IxNkJ1ZmZlcjsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0hBUkRXQVJFX0ZBS0VDQU1FUkFfSApkaWZmIC0tZ2l0IGEvY21kcy9ydW50aW1lL0FuZHJvaWQubWsgYi9jbWRzL3J1bnRpbWUvQW5kcm9pZC5tawpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNTIxZWIyYi4uMDAwMDAwMAotLS0gYS9jbWRzL3J1bnRpbWUvQW5kcm9pZC5taworKysgL2Rldi9udWxsCkBAIC0xLDI5ICswLDAgQEAKLWlmZXEgKCQoVEFSR0VUX1NJTVVMQVRPUiksdHJ1ZSkKLQotTE9DQUxfUEFUSDo9ICQoY2FsbCBteS1kaXIpCi1pbmNsdWRlICQoQ0xFQVJfVkFSUykKLQotTE9DQUxfU1JDX0ZJTEVTOj0gXAotCVNlcnZpY2VNYW5hZ2VyLmNwcCBcCi0JU2lnbmFsSGFuZGxlci5jcHAgXAotCW1haW5fcnVudGltZS5jcHAgCi0KLUxPQ0FMX1NIQVJFRF9MSUJSQVJJRVMgOj0gXAotCWxpYnV0aWxzIFwKLQlsaWJhbmRyb2lkX3J1bnRpbWUgXAotCWxpYmN1dGlscyBcCi0JbGlidWkgXAotCWxpYnN5c3RlbV9zZXJ2ZXIgXAotCWxpYmhhcmR3YXJlX2xlZ2FjeQotCi1MT0NBTF9DX0lOQ0xVREVTIDo9IFwKLQkkKEpOSV9IX0lOQ0xVREUpCi0KLWlmZXEgKCQoVEFSR0VUX09TKSxsaW51eCkKLQlMT0NBTF9DRkxBR1MgKz0gLURYUF9VTklYCi1lbmRpZgotCi1MT0NBTF9NT0RVTEU6PSBydW50aW1lCi0KLWluY2x1ZGUgJChCVUlMRF9FWEVDVVRBQkxFKQotZW5kaWYKZGlmZiAtLWdpdCBhL2NtZHMvcnVudGltZS9NT0RVTEVfTElDRU5TRV9BUEFDSEUyIGIvY21kcy9ydW50aW1lL01PRFVMRV9MSUNFTlNFX0FQQUNIRTIKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGU2OWRlMjkuLjAwMDAwMDAKLS0tIGEvY21kcy9ydW50aW1lL01PRFVMRV9MSUNFTlNFX0FQQUNIRTIKKysrIC9kZXYvbnVsbApkaWZmIC0tZ2l0IGEvY21kcy9ydW50aW1lL05PVElDRSBiL2NtZHMvcnVudGltZS9OT1RJQ0UKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM1YjFlZmEuLjAwMDAwMDAKLS0tIGEvY21kcy9ydW50aW1lL05PVElDRQorKysgL2Rldi9udWxsCkBAIC0xLDE5MCArMCwwIEBACi0KLSAgIENvcHlyaWdodCAoYykgMjAwNS0yMDA4LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0KLSAgIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICAgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotCi0gICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLQotCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBBcGFjaGUgTGljZW5zZQotICAgICAgICAgICAgICAgICAgICAgICAgICAgVmVyc2lvbiAyLjAsIEphbnVhcnkgMjAwNAotICAgICAgICAgICAgICAgICAgICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzLwotCi0gICBURVJNUyBBTkQgQ09ORElUSU9OUyBGT1IgVVNFLCBSRVBST0RVQ1RJT04sIEFORCBESVNUUklCVVRJT04KLQotICAgMS4gRGVmaW5pdGlvbnMuCi0KLSAgICAgICJMaWNlbnNlIiBzaGFsbCBtZWFuIHRoZSB0ZXJtcyBhbmQgY29uZGl0aW9ucyBmb3IgdXNlLCByZXByb2R1Y3Rpb24sCi0gICAgICBhbmQgZGlzdHJpYnV0aW9uIGFzIGRlZmluZWQgYnkgU2VjdGlvbnMgMSB0aHJvdWdoIDkgb2YgdGhpcyBkb2N1bWVudC4KLQotICAgICAgIkxpY2Vuc29yIiBzaGFsbCBtZWFuIHRoZSBjb3B5cmlnaHQgb3duZXIgb3IgZW50aXR5IGF1dGhvcml6ZWQgYnkKLSAgICAgIHRoZSBjb3B5cmlnaHQgb3duZXIgdGhhdCBpcyBncmFudGluZyB0aGUgTGljZW5zZS4KLQotICAgICAgIkxlZ2FsIEVudGl0eSIgc2hhbGwgbWVhbiB0aGUgdW5pb24gb2YgdGhlIGFjdGluZyBlbnRpdHkgYW5kIGFsbAotICAgICAgb3RoZXIgZW50aXRpZXMgdGhhdCBjb250cm9sLCBhcmUgY29udHJvbGxlZCBieSwgb3IgYXJlIHVuZGVyIGNvbW1vbgotICAgICAgY29udHJvbCB3aXRoIHRoYXQgZW50aXR5LiBGb3IgdGhlIHB1cnBvc2VzIG9mIHRoaXMgZGVmaW5pdGlvbiwKLSAgICAgICJjb250cm9sIiBtZWFucyAoaSkgdGhlIHBvd2VyLCBkaXJlY3Qgb3IgaW5kaXJlY3QsIHRvIGNhdXNlIHRoZQotICAgICAgZGlyZWN0aW9uIG9yIG1hbmFnZW1lbnQgb2Ygc3VjaCBlbnRpdHksIHdoZXRoZXIgYnkgY29udHJhY3Qgb3IKLSAgICAgIG90aGVyd2lzZSwgb3IgKGlpKSBvd25lcnNoaXAgb2YgZmlmdHkgcGVyY2VudCAoNTAlKSBvciBtb3JlIG9mIHRoZQotICAgICAgb3V0c3RhbmRpbmcgc2hhcmVzLCBvciAoaWlpKSBiZW5lZmljaWFsIG93bmVyc2hpcCBvZiBzdWNoIGVudGl0eS4KLQotICAgICAgIllvdSIgKG9yICJZb3VyIikgc2hhbGwgbWVhbiBhbiBpbmRpdmlkdWFsIG9yIExlZ2FsIEVudGl0eQotICAgICAgZXhlcmNpc2luZyBwZXJtaXNzaW9ucyBncmFudGVkIGJ5IHRoaXMgTGljZW5zZS4KLQotICAgICAgIlNvdXJjZSIgZm9ybSBzaGFsbCBtZWFuIHRoZSBwcmVmZXJyZWQgZm9ybSBmb3IgbWFraW5nIG1vZGlmaWNhdGlvbnMsCi0gICAgICBpbmNsdWRpbmcgYnV0IG5vdCBsaW1pdGVkIHRvIHNvZnR3YXJlIHNvdXJjZSBjb2RlLCBkb2N1bWVudGF0aW9uCi0gICAgICBzb3VyY2UsIGFuZCBjb25maWd1cmF0aW9uIGZpbGVzLgotCi0gICAgICAiT2JqZWN0IiBmb3JtIHNoYWxsIG1lYW4gYW55IGZvcm0gcmVzdWx0aW5nIGZyb20gbWVjaGFuaWNhbAotICAgICAgdHJhbnNmb3JtYXRpb24gb3IgdHJhbnNsYXRpb24gb2YgYSBTb3VyY2UgZm9ybSwgaW5jbHVkaW5nIGJ1dAotICAgICAgbm90IGxpbWl0ZWQgdG8gY29tcGlsZWQgb2JqZWN0IGNvZGUsIGdlbmVyYXRlZCBkb2N1bWVudGF0aW9uLAotICAgICAgYW5kIGNvbnZlcnNpb25zIHRvIG90aGVyIG1lZGlhIHR5cGVzLgotCi0gICAgICAiV29yayIgc2hhbGwgbWVhbiB0aGUgd29yayBvZiBhdXRob3JzaGlwLCB3aGV0aGVyIGluIFNvdXJjZSBvcgotICAgICAgT2JqZWN0IGZvcm0sIG1hZGUgYXZhaWxhYmxlIHVuZGVyIHRoZSBMaWNlbnNlLCBhcyBpbmRpY2F0ZWQgYnkgYQotICAgICAgY29weXJpZ2h0IG5vdGljZSB0aGF0IGlzIGluY2x1ZGVkIGluIG9yIGF0dGFjaGVkIHRvIHRoZSB3b3JrCi0gICAgICAoYW4gZXhhbXBsZSBpcyBwcm92aWRlZCBpbiB0aGUgQXBwZW5kaXggYmVsb3cpLgotCi0gICAgICAiRGVyaXZhdGl2ZSBXb3JrcyIgc2hhbGwgbWVhbiBhbnkgd29yaywgd2hldGhlciBpbiBTb3VyY2Ugb3IgT2JqZWN0Ci0gICAgICBmb3JtLCB0aGF0IGlzIGJhc2VkIG9uIChvciBkZXJpdmVkIGZyb20pIHRoZSBXb3JrIGFuZCBmb3Igd2hpY2ggdGhlCi0gICAgICBlZGl0b3JpYWwgcmV2aXNpb25zLCBhbm5vdGF0aW9ucywgZWxhYm9yYXRpb25zLCBvciBvdGhlciBtb2RpZmljYXRpb25zCi0gICAgICByZXByZXNlbnQsIGFzIGEgd2hvbGUsIGFuIG9yaWdpbmFsIHdvcmsgb2YgYXV0aG9yc2hpcC4gRm9yIHRoZSBwdXJwb3NlcwotICAgICAgb2YgdGhpcyBMaWNlbnNlLCBEZXJpdmF0aXZlIFdvcmtzIHNoYWxsIG5vdCBpbmNsdWRlIHdvcmtzIHRoYXQgcmVtYWluCi0gICAgICBzZXBhcmFibGUgZnJvbSwgb3IgbWVyZWx5IGxpbmsgKG9yIGJpbmQgYnkgbmFtZSkgdG8gdGhlIGludGVyZmFjZXMgb2YsCi0gICAgICB0aGUgV29yayBhbmQgRGVyaXZhdGl2ZSBXb3JrcyB0aGVyZW9mLgotCi0gICAgICAiQ29udHJpYnV0aW9uIiBzaGFsbCBtZWFuIGFueSB3b3JrIG9mIGF1dGhvcnNoaXAsIGluY2x1ZGluZwotICAgICAgdGhlIG9yaWdpbmFsIHZlcnNpb24gb2YgdGhlIFdvcmsgYW5kIGFueSBtb2RpZmljYXRpb25zIG9yIGFkZGl0aW9ucwotICAgICAgdG8gdGhhdCBXb3JrIG9yIERlcml2YXRpdmUgV29ya3MgdGhlcmVvZiwgdGhhdCBpcyBpbnRlbnRpb25hbGx5Ci0gICAgICBzdWJtaXR0ZWQgdG8gTGljZW5zb3IgZm9yIGluY2x1c2lvbiBpbiB0aGUgV29yayBieSB0aGUgY29weXJpZ2h0IG93bmVyCi0gICAgICBvciBieSBhbiBpbmRpdmlkdWFsIG9yIExlZ2FsIEVudGl0eSBhdXRob3JpemVkIHRvIHN1Ym1pdCBvbiBiZWhhbGYgb2YKLSAgICAgIHRoZSBjb3B5cmlnaHQgb3duZXIuIEZvciB0aGUgcHVycG9zZXMgb2YgdGhpcyBkZWZpbml0aW9uLCAic3VibWl0dGVkIgotICAgICAgbWVhbnMgYW55IGZvcm0gb2YgZWxlY3Ryb25pYywgdmVyYmFsLCBvciB3cml0dGVuIGNvbW11bmljYXRpb24gc2VudAotICAgICAgdG8gdGhlIExpY2Vuc29yIG9yIGl0cyByZXByZXNlbnRhdGl2ZXMsIGluY2x1ZGluZyBidXQgbm90IGxpbWl0ZWQgdG8KLSAgICAgIGNvbW11bmljYXRpb24gb24gZWxlY3Ryb25pYyBtYWlsaW5nIGxpc3RzLCBzb3VyY2UgY29kZSBjb250cm9sIHN5c3RlbXMsCi0gICAgICBhbmQgaXNzdWUgdHJhY2tpbmcgc3lzdGVtcyB0aGF0IGFyZSBtYW5hZ2VkIGJ5LCBvciBvbiBiZWhhbGYgb2YsIHRoZQotICAgICAgTGljZW5zb3IgZm9yIHRoZSBwdXJwb3NlIG9mIGRpc2N1c3NpbmcgYW5kIGltcHJvdmluZyB0aGUgV29yaywgYnV0Ci0gICAgICBleGNsdWRpbmcgY29tbXVuaWNhdGlvbiB0aGF0IGlzIGNvbnNwaWN1b3VzbHkgbWFya2VkIG9yIG90aGVyd2lzZQotICAgICAgZGVzaWduYXRlZCBpbiB3cml0aW5nIGJ5IHRoZSBjb3B5cmlnaHQgb3duZXIgYXMgIk5vdCBhIENvbnRyaWJ1dGlvbi4iCi0KLSAgICAgICJDb250cmlidXRvciIgc2hhbGwgbWVhbiBMaWNlbnNvciBhbmQgYW55IGluZGl2aWR1YWwgb3IgTGVnYWwgRW50aXR5Ci0gICAgICBvbiBiZWhhbGYgb2Ygd2hvbSBhIENvbnRyaWJ1dGlvbiBoYXMgYmVlbiByZWNlaXZlZCBieSBMaWNlbnNvciBhbmQKLSAgICAgIHN1YnNlcXVlbnRseSBpbmNvcnBvcmF0ZWQgd2l0aGluIHRoZSBXb3JrLgotCi0gICAyLiBHcmFudCBvZiBDb3B5cmlnaHQgTGljZW5zZS4gU3ViamVjdCB0byB0aGUgdGVybXMgYW5kIGNvbmRpdGlvbnMgb2YKLSAgICAgIHRoaXMgTGljZW5zZSwgZWFjaCBDb250cmlidXRvciBoZXJlYnkgZ3JhbnRzIHRvIFlvdSBhIHBlcnBldHVhbCwKLSAgICAgIHdvcmxkd2lkZSwgbm9uLWV4Y2x1c2l2ZSwgbm8tY2hhcmdlLCByb3lhbHR5LWZyZWUsIGlycmV2b2NhYmxlCi0gICAgICBjb3B5cmlnaHQgbGljZW5zZSB0byByZXByb2R1Y2UsIHByZXBhcmUgRGVyaXZhdGl2ZSBXb3JrcyBvZiwKLSAgICAgIHB1YmxpY2x5IGRpc3BsYXksIHB1YmxpY2x5IHBlcmZvcm0sIHN1YmxpY2Vuc2UsIGFuZCBkaXN0cmlidXRlIHRoZQotICAgICAgV29yayBhbmQgc3VjaCBEZXJpdmF0aXZlIFdvcmtzIGluIFNvdXJjZSBvciBPYmplY3QgZm9ybS4KLQotICAgMy4gR3JhbnQgb2YgUGF0ZW50IExpY2Vuc2UuIFN1YmplY3QgdG8gdGhlIHRlcm1zIGFuZCBjb25kaXRpb25zIG9mCi0gICAgICB0aGlzIExpY2Vuc2UsIGVhY2ggQ29udHJpYnV0b3IgaGVyZWJ5IGdyYW50cyB0byBZb3UgYSBwZXJwZXR1YWwsCi0gICAgICB3b3JsZHdpZGUsIG5vbi1leGNsdXNpdmUsIG5vLWNoYXJnZSwgcm95YWx0eS1mcmVlLCBpcnJldm9jYWJsZQotICAgICAgKGV4Y2VwdCBhcyBzdGF0ZWQgaW4gdGhpcyBzZWN0aW9uKSBwYXRlbnQgbGljZW5zZSB0byBtYWtlLCBoYXZlIG1hZGUsCi0gICAgICB1c2UsIG9mZmVyIHRvIHNlbGwsIHNlbGwsIGltcG9ydCwgYW5kIG90aGVyd2lzZSB0cmFuc2ZlciB0aGUgV29yaywKLSAgICAgIHdoZXJlIHN1Y2ggbGljZW5zZSBhcHBsaWVzIG9ubHkgdG8gdGhvc2UgcGF0ZW50IGNsYWltcyBsaWNlbnNhYmxlCi0gICAgICBieSBzdWNoIENvbnRyaWJ1dG9yIHRoYXQgYXJlIG5lY2Vzc2FyaWx5IGluZnJpbmdlZCBieSB0aGVpcgotICAgICAgQ29udHJpYnV0aW9uKHMpIGFsb25lIG9yIGJ5IGNvbWJpbmF0aW9uIG9mIHRoZWlyIENvbnRyaWJ1dGlvbihzKQotICAgICAgd2l0aCB0aGUgV29yayB0byB3aGljaCBzdWNoIENvbnRyaWJ1dGlvbihzKSB3YXMgc3VibWl0dGVkLiBJZiBZb3UKLSAgICAgIGluc3RpdHV0ZSBwYXRlbnQgbGl0aWdhdGlvbiBhZ2FpbnN0IGFueSBlbnRpdHkgKGluY2x1ZGluZyBhCi0gICAgICBjcm9zcy1jbGFpbSBvciBjb3VudGVyY2xhaW0gaW4gYSBsYXdzdWl0KSBhbGxlZ2luZyB0aGF0IHRoZSBXb3JrCi0gICAgICBvciBhIENvbnRyaWJ1dGlvbiBpbmNvcnBvcmF0ZWQgd2l0aGluIHRoZSBXb3JrIGNvbnN0aXR1dGVzIGRpcmVjdAotICAgICAgb3IgY29udHJpYnV0b3J5IHBhdGVudCBpbmZyaW5nZW1lbnQsIHRoZW4gYW55IHBhdGVudCBsaWNlbnNlcwotICAgICAgZ3JhbnRlZCB0byBZb3UgdW5kZXIgdGhpcyBMaWNlbnNlIGZvciB0aGF0IFdvcmsgc2hhbGwgdGVybWluYXRlCi0gICAgICBhcyBvZiB0aGUgZGF0ZSBzdWNoIGxpdGlnYXRpb24gaXMgZmlsZWQuCi0KLSAgIDQuIFJlZGlzdHJpYnV0aW9uLiBZb3UgbWF5IHJlcHJvZHVjZSBhbmQgZGlzdHJpYnV0ZSBjb3BpZXMgb2YgdGhlCi0gICAgICBXb3JrIG9yIERlcml2YXRpdmUgV29ya3MgdGhlcmVvZiBpbiBhbnkgbWVkaXVtLCB3aXRoIG9yIHdpdGhvdXQKLSAgICAgIG1vZGlmaWNhdGlvbnMsIGFuZCBpbiBTb3VyY2Ugb3IgT2JqZWN0IGZvcm0sIHByb3ZpZGVkIHRoYXQgWW91Ci0gICAgICBtZWV0IHRoZSBmb2xsb3dpbmcgY29uZGl0aW9uczoKLQotICAgICAgKGEpIFlvdSBtdXN0IGdpdmUgYW55IG90aGVyIHJlY2lwaWVudHMgb2YgdGhlIFdvcmsgb3IKLSAgICAgICAgICBEZXJpdmF0aXZlIFdvcmtzIGEgY29weSBvZiB0aGlzIExpY2Vuc2U7IGFuZAotCi0gICAgICAoYikgWW91IG11c3QgY2F1c2UgYW55IG1vZGlmaWVkIGZpbGVzIHRvIGNhcnJ5IHByb21pbmVudCBub3RpY2VzCi0gICAgICAgICAgc3RhdGluZyB0aGF0IFlvdSBjaGFuZ2VkIHRoZSBmaWxlczsgYW5kCi0KLSAgICAgIChjKSBZb3UgbXVzdCByZXRhaW4sIGluIHRoZSBTb3VyY2UgZm9ybSBvZiBhbnkgRGVyaXZhdGl2ZSBXb3JrcwotICAgICAgICAgIHRoYXQgWW91IGRpc3RyaWJ1dGUsIGFsbCBjb3B5cmlnaHQsIHBhdGVudCwgdHJhZGVtYXJrLCBhbmQKLSAgICAgICAgICBhdHRyaWJ1dGlvbiBub3RpY2VzIGZyb20gdGhlIFNvdXJjZSBmb3JtIG9mIHRoZSBXb3JrLAotICAgICAgICAgIGV4Y2x1ZGluZyB0aG9zZSBub3RpY2VzIHRoYXQgZG8gbm90IHBlcnRhaW4gdG8gYW55IHBhcnQgb2YKLSAgICAgICAgICB0aGUgRGVyaXZhdGl2ZSBXb3JrczsgYW5kCi0KLSAgICAgIChkKSBJZiB0aGUgV29yayBpbmNsdWRlcyBhICJOT1RJQ0UiIHRleHQgZmlsZSBhcyBwYXJ0IG9mIGl0cwotICAgICAgICAgIGRpc3RyaWJ1dGlvbiwgdGhlbiBhbnkgRGVyaXZhdGl2ZSBXb3JrcyB0aGF0IFlvdSBkaXN0cmlidXRlIG11c3QKLSAgICAgICAgICBpbmNsdWRlIGEgcmVhZGFibGUgY29weSBvZiB0aGUgYXR0cmlidXRpb24gbm90aWNlcyBjb250YWluZWQKLSAgICAgICAgICB3aXRoaW4gc3VjaCBOT1RJQ0UgZmlsZSwgZXhjbHVkaW5nIHRob3NlIG5vdGljZXMgdGhhdCBkbyBub3QKLSAgICAgICAgICBwZXJ0YWluIHRvIGFueSBwYXJ0IG9mIHRoZSBEZXJpdmF0aXZlIFdvcmtzLCBpbiBhdCBsZWFzdCBvbmUKLSAgICAgICAgICBvZiB0aGUgZm9sbG93aW5nIHBsYWNlczogd2l0aGluIGEgTk9USUNFIHRleHQgZmlsZSBkaXN0cmlidXRlZAotICAgICAgICAgIGFzIHBhcnQgb2YgdGhlIERlcml2YXRpdmUgV29ya3M7IHdpdGhpbiB0aGUgU291cmNlIGZvcm0gb3IKLSAgICAgICAgICBkb2N1bWVudGF0aW9uLCBpZiBwcm92aWRlZCBhbG9uZyB3aXRoIHRoZSBEZXJpdmF0aXZlIFdvcmtzOyBvciwKLSAgICAgICAgICB3aXRoaW4gYSBkaXNwbGF5IGdlbmVyYXRlZCBieSB0aGUgRGVyaXZhdGl2ZSBXb3JrcywgaWYgYW5kCi0gICAgICAgICAgd2hlcmV2ZXIgc3VjaCB0aGlyZC1wYXJ0eSBub3RpY2VzIG5vcm1hbGx5IGFwcGVhci4gVGhlIGNvbnRlbnRzCi0gICAgICAgICAgb2YgdGhlIE5PVElDRSBmaWxlIGFyZSBmb3IgaW5mb3JtYXRpb25hbCBwdXJwb3NlcyBvbmx5IGFuZAotICAgICAgICAgIGRvIG5vdCBtb2RpZnkgdGhlIExpY2Vuc2UuIFlvdSBtYXkgYWRkIFlvdXIgb3duIGF0dHJpYnV0aW9uCi0gICAgICAgICAgbm90aWNlcyB3aXRoaW4gRGVyaXZhdGl2ZSBXb3JrcyB0aGF0IFlvdSBkaXN0cmlidXRlLCBhbG9uZ3NpZGUKLSAgICAgICAgICBvciBhcyBhbiBhZGRlbmR1bSB0byB0aGUgTk9USUNFIHRleHQgZnJvbSB0aGUgV29yaywgcHJvdmlkZWQKLSAgICAgICAgICB0aGF0IHN1Y2ggYWRkaXRpb25hbCBhdHRyaWJ1dGlvbiBub3RpY2VzIGNhbm5vdCBiZSBjb25zdHJ1ZWQKLSAgICAgICAgICBhcyBtb2RpZnlpbmcgdGhlIExpY2Vuc2UuCi0KLSAgICAgIFlvdSBtYXkgYWRkIFlvdXIgb3duIGNvcHlyaWdodCBzdGF0ZW1lbnQgdG8gWW91ciBtb2RpZmljYXRpb25zIGFuZAotICAgICAgbWF5IHByb3ZpZGUgYWRkaXRpb25hbCBvciBkaWZmZXJlbnQgbGljZW5zZSB0ZXJtcyBhbmQgY29uZGl0aW9ucwotICAgICAgZm9yIHVzZSwgcmVwcm9kdWN0aW9uLCBvciBkaXN0cmlidXRpb24gb2YgWW91ciBtb2RpZmljYXRpb25zLCBvcgotICAgICAgZm9yIGFueSBzdWNoIERlcml2YXRpdmUgV29ya3MgYXMgYSB3aG9sZSwgcHJvdmlkZWQgWW91ciB1c2UsCi0gICAgICByZXByb2R1Y3Rpb24sIGFuZCBkaXN0cmlidXRpb24gb2YgdGhlIFdvcmsgb3RoZXJ3aXNlIGNvbXBsaWVzIHdpdGgKLSAgICAgIHRoZSBjb25kaXRpb25zIHN0YXRlZCBpbiB0aGlzIExpY2Vuc2UuCi0KLSAgIDUuIFN1Ym1pc3Npb24gb2YgQ29udHJpYnV0aW9ucy4gVW5sZXNzIFlvdSBleHBsaWNpdGx5IHN0YXRlIG90aGVyd2lzZSwKLSAgICAgIGFueSBDb250cmlidXRpb24gaW50ZW50aW9uYWxseSBzdWJtaXR0ZWQgZm9yIGluY2x1c2lvbiBpbiB0aGUgV29yawotICAgICAgYnkgWW91IHRvIHRoZSBMaWNlbnNvciBzaGFsbCBiZSB1bmRlciB0aGUgdGVybXMgYW5kIGNvbmRpdGlvbnMgb2YKLSAgICAgIHRoaXMgTGljZW5zZSwgd2l0aG91dCBhbnkgYWRkaXRpb25hbCB0ZXJtcyBvciBjb25kaXRpb25zLgotICAgICAgTm90d2l0aHN0YW5kaW5nIHRoZSBhYm92ZSwgbm90aGluZyBoZXJlaW4gc2hhbGwgc3VwZXJzZWRlIG9yIG1vZGlmeQotICAgICAgdGhlIHRlcm1zIG9mIGFueSBzZXBhcmF0ZSBsaWNlbnNlIGFncmVlbWVudCB5b3UgbWF5IGhhdmUgZXhlY3V0ZWQKLSAgICAgIHdpdGggTGljZW5zb3IgcmVnYXJkaW5nIHN1Y2ggQ29udHJpYnV0aW9ucy4KLQotICAgNi4gVHJhZGVtYXJrcy4gVGhpcyBMaWNlbnNlIGRvZXMgbm90IGdyYW50IHBlcm1pc3Npb24gdG8gdXNlIHRoZSB0cmFkZQotICAgICAgbmFtZXMsIHRyYWRlbWFya3MsIHNlcnZpY2UgbWFya3MsIG9yIHByb2R1Y3QgbmFtZXMgb2YgdGhlIExpY2Vuc29yLAotICAgICAgZXhjZXB0IGFzIHJlcXVpcmVkIGZvciByZWFzb25hYmxlIGFuZCBjdXN0b21hcnkgdXNlIGluIGRlc2NyaWJpbmcgdGhlCi0gICAgICBvcmlnaW4gb2YgdGhlIFdvcmsgYW5kIHJlcHJvZHVjaW5nIHRoZSBjb250ZW50IG9mIHRoZSBOT1RJQ0UgZmlsZS4KLQotICAgNy4gRGlzY2xhaW1lciBvZiBXYXJyYW50eS4gVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yCi0gICAgICBhZ3JlZWQgdG8gaW4gd3JpdGluZywgTGljZW5zb3IgcHJvdmlkZXMgdGhlIFdvcmsgKGFuZCBlYWNoCi0gICAgICBDb250cmlidXRvciBwcm92aWRlcyBpdHMgQ29udHJpYnV0aW9ucykgb24gYW4gIkFTIElTIiBCQVNJUywKLSAgICAgIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvcgotICAgICAgaW1wbGllZCwgaW5jbHVkaW5nLCB3aXRob3V0IGxpbWl0YXRpb24sIGFueSB3YXJyYW50aWVzIG9yIGNvbmRpdGlvbnMKLSAgICAgIG9mIFRJVExFLCBOT04tSU5GUklOR0VNRU5ULCBNRVJDSEFOVEFCSUxJVFksIG9yIEZJVE5FU1MgRk9SIEEKLSAgICAgIFBBUlRJQ1VMQVIgUFVSUE9TRS4gWW91IGFyZSBzb2xlbHkgcmVzcG9uc2libGUgZm9yIGRldGVybWluaW5nIHRoZQotICAgICAgYXBwcm9wcmlhdGVuZXNzIG9mIHVzaW5nIG9yIHJlZGlzdHJpYnV0aW5nIHRoZSBXb3JrIGFuZCBhc3N1bWUgYW55Ci0gICAgICByaXNrcyBhc3NvY2lhdGVkIHdpdGggWW91ciBleGVyY2lzZSBvZiBwZXJtaXNzaW9ucyB1bmRlciB0aGlzIExpY2Vuc2UuCi0KLSAgIDguIExpbWl0YXRpb24gb2YgTGlhYmlsaXR5LiBJbiBubyBldmVudCBhbmQgdW5kZXIgbm8gbGVnYWwgdGhlb3J5LAotICAgICAgd2hldGhlciBpbiB0b3J0IChpbmNsdWRpbmcgbmVnbGlnZW5jZSksIGNvbnRyYWN0LCBvciBvdGhlcndpc2UsCi0gICAgICB1bmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgKHN1Y2ggYXMgZGVsaWJlcmF0ZSBhbmQgZ3Jvc3NseQotICAgICAgbmVnbGlnZW50IGFjdHMpIG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzaGFsbCBhbnkgQ29udHJpYnV0b3IgYmUKLSAgICAgIGxpYWJsZSB0byBZb3UgZm9yIGRhbWFnZXMsIGluY2x1ZGluZyBhbnkgZGlyZWN0LCBpbmRpcmVjdCwgc3BlY2lhbCwKLSAgICAgIGluY2lkZW50YWwsIG9yIGNvbnNlcXVlbnRpYWwgZGFtYWdlcyBvZiBhbnkgY2hhcmFjdGVyIGFyaXNpbmcgYXMgYQotICAgICAgcmVzdWx0IG9mIHRoaXMgTGljZW5zZSBvciBvdXQgb2YgdGhlIHVzZSBvciBpbmFiaWxpdHkgdG8gdXNlIHRoZQotICAgICAgV29yayAoaW5jbHVkaW5nIGJ1dCBub3QgbGltaXRlZCB0byBkYW1hZ2VzIGZvciBsb3NzIG9mIGdvb2R3aWxsLAotICAgICAgd29yayBzdG9wcGFnZSwgY29tcHV0ZXIgZmFpbHVyZSBvciBtYWxmdW5jdGlvbiwgb3IgYW55IGFuZCBhbGwKLSAgICAgIG90aGVyIGNvbW1lcmNpYWwgZGFtYWdlcyBvciBsb3NzZXMpLCBldmVuIGlmIHN1Y2ggQ29udHJpYnV0b3IKLSAgICAgIGhhcyBiZWVuIGFkdmlzZWQgb2YgdGhlIHBvc3NpYmlsaXR5IG9mIHN1Y2ggZGFtYWdlcy4KLQotICAgOS4gQWNjZXB0aW5nIFdhcnJhbnR5IG9yIEFkZGl0aW9uYWwgTGlhYmlsaXR5LiBXaGlsZSByZWRpc3RyaWJ1dGluZwotICAgICAgdGhlIFdvcmsgb3IgRGVyaXZhdGl2ZSBXb3JrcyB0aGVyZW9mLCBZb3UgbWF5IGNob29zZSB0byBvZmZlciwKLSAgICAgIGFuZCBjaGFyZ2UgYSBmZWUgZm9yLCBhY2NlcHRhbmNlIG9mIHN1cHBvcnQsIHdhcnJhbnR5LCBpbmRlbW5pdHksCi0gICAgICBvciBvdGhlciBsaWFiaWxpdHkgb2JsaWdhdGlvbnMgYW5kL29yIHJpZ2h0cyBjb25zaXN0ZW50IHdpdGggdGhpcwotICAgICAgTGljZW5zZS4gSG93ZXZlciwgaW4gYWNjZXB0aW5nIHN1Y2ggb2JsaWdhdGlvbnMsIFlvdSBtYXkgYWN0IG9ubHkKLSAgICAgIG9uIFlvdXIgb3duIGJlaGFsZiBhbmQgb24gWW91ciBzb2xlIHJlc3BvbnNpYmlsaXR5LCBub3Qgb24gYmVoYWxmCi0gICAgICBvZiBhbnkgb3RoZXIgQ29udHJpYnV0b3IsIGFuZCBvbmx5IGlmIFlvdSBhZ3JlZSB0byBpbmRlbW5pZnksCi0gICAgICBkZWZlbmQsIGFuZCBob2xkIGVhY2ggQ29udHJpYnV0b3IgaGFybWxlc3MgZm9yIGFueSBsaWFiaWxpdHkKLSAgICAgIGluY3VycmVkIGJ5LCBvciBjbGFpbXMgYXNzZXJ0ZWQgYWdhaW5zdCwgc3VjaCBDb250cmlidXRvciBieSByZWFzb24KLSAgICAgIG9mIHlvdXIgYWNjZXB0aW5nIGFueSBzdWNoIHdhcnJhbnR5IG9yIGFkZGl0aW9uYWwgbGlhYmlsaXR5LgotCi0gICBFTkQgT0YgVEVSTVMgQU5EIENPTkRJVElPTlMKLQpkaWZmIC0tZ2l0IGEvY21kcy9ydW50aW1lL1NlcnZpY2VNYW5hZ2VyLmNwcCBiL2NtZHMvcnVudGltZS9TZXJ2aWNlTWFuYWdlci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDc1OGE5NWMuLjAwMDAwMDAKLS0tIGEvY21kcy9ydW50aW1lL1NlcnZpY2VNYW5hZ2VyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDc0ICswLDAgQEAKLS8vCi0vLyBDb3B5cmlnaHQgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0vLwotCi0jZGVmaW5lIExPR19UQUcgIlNlcnZpY2VNYW5hZ2VyIgotCi0jaW5jbHVkZSAiU2VydmljZU1hbmFnZXIuaCIKLSNpbmNsdWRlICJTaWduYWxIYW5kbGVyLmgiCi0KLSNpbmNsdWRlIDx1dGlscy9EZWJ1Zy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLSNpbmNsdWRlIDx1dGlscy9Qcm9jZXNzU3RhdGUuaD4KLQotI2luY2x1ZGUgPHByaXZhdGUvdXRpbHMvU3RhdGljLmg+Ci0KLSNpbmNsdWRlIDxjdHlwZS5oPgotI2luY2x1ZGUgPGVycm5vLmg+Ci0jaW5jbHVkZSA8bGltaXRzLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzeXMvc3RhdC5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLUJTZXJ2aWNlTWFuYWdlcjo6QlNlcnZpY2VNYW5hZ2VyKCkKLXsKLX0KLQotc3A8SUJpbmRlcj4gQlNlcnZpY2VNYW5hZ2VyOjpnZXRTZXJ2aWNlKGNvbnN0IFN0cmluZzE2JiBuYW1lKSBjb25zdAotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgc3NpemVfdCBpID0gbVNlcnZpY2VzLmluZGV4T2ZLZXkobmFtZSk7Ci0gICAgTE9HVigiU2VydmljZU1hbmFnZXI6IGdldFNlcnZpY2UoJXMpIC0+ICVkXG4iLCBTdHJpbmc4KG5hbWUpLnN0cmluZygpLCBpKTsKLSAgICBpZiAoaSA+PSAwKSByZXR1cm4gbVNlcnZpY2VzLnZhbHVlQXQoaSk7Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLXNwPElCaW5kZXI+IEJTZXJ2aWNlTWFuYWdlcjo6Y2hlY2tTZXJ2aWNlKGNvbnN0IFN0cmluZzE2JiBuYW1lKSBjb25zdAotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgc3NpemVfdCBpID0gbVNlcnZpY2VzLmluZGV4T2ZLZXkobmFtZSk7Ci0gICAgTE9HVigiU2VydmljZU1hbmFnZXI6IGdldFNlcnZpY2UoJXMpIC0+ICVkXG4iLCBTdHJpbmc4KG5hbWUpLnN0cmluZygpLCBpKTsKLSAgICBpZiAoaSA+PSAwKSByZXR1cm4gbVNlcnZpY2VzLnZhbHVlQXQoaSk7Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLXN0YXR1c190IEJTZXJ2aWNlTWFuYWdlcjo6YWRkU2VydmljZShjb25zdCBTdHJpbmcxNiYgbmFtZSwgY29uc3Qgc3A8SUJpbmRlcj4mIHNlcnZpY2UpCi17Ci0gICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLSAgICBMT0dJKCJTZXJ2aWNlTWFuYWdlcjogYWRkU2VydmljZSglcywgJXApXG4iLCBTdHJpbmc4KG5hbWUpLnN0cmluZygpLCBzZXJ2aWNlLmdldCgpKTsKLSAgICBjb25zdCBzc2l6ZV90IHJlcyA9IG1TZXJ2aWNlcy5hZGQobmFtZSwgc2VydmljZSk7Ci0gICAgaWYgKHJlcyA+PSBOT19FUlJPUikgewotICAgICAgICBtQ2hhbmdlZC5icm9hZGNhc3QoKTsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLSAgICByZXR1cm4gcmVzOwotfQotCi1WZWN0b3I8U3RyaW5nMTY+IEJTZXJ2aWNlTWFuYWdlcjo6bGlzdFNlcnZpY2VzKCkKLXsKLSAgICBWZWN0b3I8U3RyaW5nMTY+IHJlczsKLQotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgY29uc3Qgc2l6ZV90IE4gPSBtU2VydmljZXMuc2l6ZSgpOwotICAgIGZvciAoc2l6ZV90IGk9MDsgaTxOOyBpKyspIHsKLSAgICAgICAgcmVzLmFkZChtU2VydmljZXMua2V5QXQoaSkpOwotICAgIH0KLQotICAgIHJldHVybiByZXM7Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9jbWRzL3J1bnRpbWUvU2VydmljZU1hbmFnZXIuaCBiL2NtZHMvcnVudGltZS9TZXJ2aWNlTWFuYWdlci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkMDljZWM4Li4wMDAwMDAwCi0tLSBhL2NtZHMvcnVudGltZS9TZXJ2aWNlTWFuYWdlci5oCisrKyAvZGV2L251bGwKQEAgLTEsMzggKzAsMCBAQAotLy8KLS8vIENvcHlyaWdodCAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLS8vCi0jaWZuZGVmIEFORFJPSURfU0VSVklDRV9NQU5BR0VSX0gKLSNkZWZpbmUgQU5EUk9JRF9TRVJWSUNFX01BTkFHRVJfSAotCi0jaW5jbHVkZSA8dXRpbHMvSVNlcnZpY2VNYW5hZ2VyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvS2V5ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBCU2VydmljZU1hbmFnZXIgOiBwdWJsaWMgQm5TZXJ2aWNlTWFuYWdlcgotewotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBCU2VydmljZU1hbmFnZXIoKTsKLSAgICAKLSAgICB2aXJ0dWFsIHNwPElCaW5kZXI+ICAgICAgICAgZ2V0U2VydmljZSggY29uc3QgU3RyaW5nMTYmIG5hbWUpIGNvbnN0OwotICAgIHZpcnR1YWwgc3A8SUJpbmRlcj4gICAgICAgICBjaGVja1NlcnZpY2UoIGNvbnN0IFN0cmluZzE2JiBuYW1lKSBjb25zdDsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICAgICAgYWRkU2VydmljZSggY29uc3QgU3RyaW5nMTYmIG5hbWUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHNwPElCaW5kZXI+JiBzZXJ2aWNlKTsKLSAgICB2aXJ0dWFsIFZlY3RvcjxTdHJpbmcxNj4gICAgbGlzdFNlcnZpY2VzKCk7Ci0KLSAgICAKLXByaXZhdGU6Ci0gICAgbXV0YWJsZSBNdXRleCAgICAgICAgICAgICAgIG1Mb2NrOwotICAgIG11dGFibGUgQ29uZGl0aW9uICAgICAgICAgICBtQ2hhbmdlZDsKLSAgICBzcDxJUGVybWlzc2lvbkNvbnRyb2xsZXI+ICAgbVBlcm1pc3Npb25Db250cm9sbGVyOwotICAgIEtleWVkVmVjdG9yPFN0cmluZzE2LCBzcDxJQmluZGVyPiA+IG1TZXJ2aWNlczsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfU0VSVklDRV9NQU5BR0VSX0gKZGlmZiAtLWdpdCBhL2NtZHMvcnVudGltZS9TaWduYWxIYW5kbGVyLmNwcCBiL2NtZHMvcnVudGltZS9TaWduYWxIYW5kbGVyLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggY2NjYWFiZi4uMDAwMDAwMAotLS0gYS9jbWRzL3J1bnRpbWUvU2lnbmFsSGFuZGxlci5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwyNDkgKzAsMCBAQAotLy8KLS8vIENvcHlyaWdodCAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLS8vCi0KLSNkZWZpbmUgTE9HX1RBRyAiU2lnbmFsSGFuZGxlciIKLQotI2luY2x1ZGUgIlNpZ25hbEhhbmRsZXIuaCIKLQotI2luY2x1ZGUgPHV0aWxzL0F0b21pYy5oPgotI2luY2x1ZGUgPHV0aWxzL0RlYnVnLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPHN5cy93YWl0Lmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgU2lnbmFsSGFuZGxlcjo6UHJvY2Vzc1RocmVhZCA6IHB1YmxpYyBUaHJlYWQKLXsKLXB1YmxpYzoKLSAgICBQcm9jZXNzVGhyZWFkKFNpZ25hbEhhbmRsZXImIHNoKQotICAgICAgICA6IFRocmVhZChmYWxzZSkKLSAgICAgICAgLCBtT3duZXIoc2gpCi0gICAgewotICAgIH0KLQotICAgIHZpcnR1YWwgYm9vbCB0aHJlYWRMb29wKCkKLSAgICB7Ci0gICAgICAgIGNoYXIgYnVmZmVyWzMyXTsKLSAgICAgICAgcmVhZChtT3duZXIubUF2YWlsTXNnWzBdLCBidWZmZXIsIHNpemVvZihidWZmZXIpKTsKLQotICAgICAgICBMT0dWKCJTaWduYWwgY29tbWFuZCBwcm9jZXNzaW5nIHRocmVhZCB3b2tlIHVwISIpOwotCi0gICAgICAgIGlmIChtT3duZXIubUxvc3RDb21tYW5kcykgewotICAgICAgICAgICAgTE9HRSgiTG9zdCAlZCBzaWduYWxzISIsIG1Pd25lci5tTG9zdENvbW1hbmRzKTsKLSAgICAgICAgICAgIG1Pd25lci5tTG9zdENvbW1hbmRzID0gMDsKLSAgICAgICAgfQotCi0gICAgICAgIGludCBjdXI7Ci0gICAgICAgIHdoaWxlICgoY3VyPW1Pd25lci5tQ29tbWFuZEJvdHRvbSkgIT0gbU93bmVyLm1Db21tYW5kVG9wKSB7Ci0gICAgICAgICAgICBpZiAobU93bmVyLm1Db21tYW5kc1tjdXJdLmZpbGxlZCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgTE9HVigiQ29tbWFuZCBhdCAlZCBpcyBub3QgeWV0IGZpbGxlZCIsIGN1cik7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIExPR1YoIlByb2Nlc3NpbmcgY29tbWFuZCBhdCAlZCwgdG9wIGlzICVkIiwKLSAgICAgICAgICAgICAgICAgY3VyLCBtT3duZXIubUNvbW1hbmRUb3ApOwotICAgICAgICAgICAgcHJvY2Vzc0NvbW1hbmQobU93bmVyLm1Db21tYW5kc1tjdXJdKTsKLSAgICAgICAgICAgIG1Pd25lci5tQ29tbWFuZHNbY3VyXS5maWxsZWQgPSAwOwotCi0gICAgICAgICAgICBpbnQgbmV4dCA9IG1Pd25lci5tQ29tbWFuZEJvdHRvbSsxOwotICAgICAgICAgICAgaWYgKG5leHQgPj0gQ09NTUFORF9RVUVVRV9TSVpFKSB7Ci0gICAgICAgICAgICAgICAgbmV4dCA9IDA7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIG1Pd25lci5tQ29tbWFuZEJvdHRvbSA9IG5leHQ7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICB2b2lkIHByb2Nlc3NDb21tYW5kKGNvbnN0IENvbW1hbmRFbnRyeSYgZW50cnkpCi0gICAgewotICAgICAgICBzd2l0Y2ggKGVudHJ5LnNpZ251bSkgewotICAgICAgICBjYXNlIFNJR0NITEQ6IHsKLSAgICAgICAgICAgIG1Pd25lci5tTG9jay5sb2NrKCk7Ci0gICAgICAgICAgICBzc2l6ZV90IGkgPSBtT3duZXIubUNoaWxkSGFuZGxlcnMuaW5kZXhPZktleShlbnRyeS5pbmZvLnNpX3BpZCk7Ci0gICAgICAgICAgICBDaGlsZEhhbmRsZXIgY2g7Ci0gICAgICAgICAgICBpZiAoaSA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgY2ggPSBtT3duZXIubUNoaWxkSGFuZGxlcnMudmFsdWVBdChpKTsKLSAgICAgICAgICAgICAgICBtT3duZXIubUNoaWxkSGFuZGxlcnMucmVtb3ZlSXRlbXNBdChpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1Pd25lci5tTG9jay51bmxvY2soKTsKLQotICAgICAgICAgICAgTE9HRCgiU0lHQ0hMRDogcGlkPSVkLCBoYW5kbGUgaW5kZXg9JWQiLCBlbnRyeS5pbmZvLnNpX3BpZCwgaSk7Ci0KLSAgICAgICAgICAgIGlmIChpID49IDApIHsKLSAgICAgICAgICAgICAgICBpbnQgcmVzID0gd2FpdHBpZChlbnRyeS5pbmZvLnNpX3BpZCwgTlVMTCwgV05PSEFORyk7Ci0gICAgICAgICAgICAgICAgTE9HV19JRihyZXMgPT0gMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICJSZWNlaXZlZCBTSUdDSExELCBidXQgcGlkICVkIGlzIG5vdCB5ZXQgc3RvcHBlZCIsCi0gICAgICAgICAgICAgICAgICAgICAgICBlbnRyeS5pbmZvLnNpX3BpZCk7Ci0gICAgICAgICAgICAgICAgaWYgKGNoLmhhbmRsZXIpIHsKLSAgICAgICAgICAgICAgICAgICAgY2guaGFuZGxlcihlbnRyeS5pbmZvLnNpX3BpZCwgY2gudXNlckRhdGEpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgTE9HVygiVW5oYW5kbGVkIFNJR0NITEQgZm9yIHBpZCAlZCIsIGVudHJ5LmluZm8uc2lfcGlkKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgfQotICAgIH0KLQotICAgIFNpZ25hbEhhbmRsZXImIG1Pd25lcjsKLX07Ci0KLQotTXV0ZXggU2lnbmFsSGFuZGxlcjo6bUluc3RhbmNlTG9jazsKLVNpZ25hbEhhbmRsZXIqIFNpZ25hbEhhbmRsZXI6Om1JbnN0YW5jZSA9IE5VTEw7Ci0KLXN0YXR1c190IFNpZ25hbEhhbmRsZXI6OnNldENoaWxkSGFuZGxlcihwaWRfdCBjaGlsZFBpZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgdGFnLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNoaWxkX2NhbGxiYWNrX3QgaGFuZGxlciwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiB1c2VyRGF0YSkKLXsKLSAgICBTaWduYWxIYW5kbGVyKiBjb25zdCBzZWxmID0gZ2V0SW5zdGFuY2UoKTsKLQotICAgIHNlbGYtPm1Mb2NrLmxvY2soKTsKLQotICAgIC8vIEZpcnN0IG1ha2Ugc3VyZSB0aGlzIGNoaWxkIGhhc24ndCBhbHJlYWR5IGV4aXRlZC4KLSAgICBwaWRfdCByZXMgPSB3YWl0cGlkKGNoaWxkUGlkLCBOVUxMLCBXTk9IQU5HKTsKLSAgICBpZiAocmVzICE9IDApIHsKLSAgICAgICAgaWYgKHJlcyA8IDApIHsKLSAgICAgICAgICAgIExPR1coInNldENoaWxkSGFuZGxlciB3YWl0cGlkIG9mICVkIGZhaWxlZDogJWQgKCVzKSIsCi0gICAgICAgICAgICAgICAgIGNoaWxkUGlkLCByZXMsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBMT0dXKCJzZXRDaGlsZEhhbmRsZXIgd2FpdHBpZCBvZiAlZCBzYWlkICVkIGFscmVhZHkgZGVhZCIsCi0gICAgICAgICAgICAgICAgIGNoaWxkUGlkLCByZXMpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gU29tZSBraW5kIG9mIGVycm9yLi4uICBqdXN0IGhhbmRsZSB0aGUgZXhpdCBub3cuCi0gICAgICAgIHNlbGYtPm1Mb2NrLnVubG9jaygpOwotCi0gICAgICAgIGlmIChoYW5kbGVyKSB7Ci0gICAgICAgICAgICBoYW5kbGVyKGNoaWxkUGlkLCB1c2VyRGF0YSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBSZXR1cm4gYW4gZXJyb3IgY29kZSAtLSAwIG1lYW5zIGl0IGFscmVhZHkgZXhpdGVkLgotICAgICAgICByZXR1cm4gKHN0YXR1c190KXJlczsKLSAgICB9Ci0KLSAgICBDaGlsZEhhbmRsZXIgZW50cnk7Ci0gICAgZW50cnkuY2hpbGRQaWQgPSBjaGlsZFBpZDsKLSAgICBlbnRyeS50YWcgPSB0YWc7Ci0gICAgZW50cnkuaGFuZGxlciA9IGhhbmRsZXI7Ci0gICAgZW50cnkudXNlckRhdGEgPSB1c2VyRGF0YTsKLQotICAgIC8vIE5vdGU6IHRoaXMgcmVwbGFjZXMgYW4gZXhpc3RpbmcgZW50cnkgZm9yIHRoaXMgcGlkLCBpZiB0aGVyZSBhbHJlYWR5Ci0gICAgLy8gaXMgb25lLiAgVGhpcyBpcyB0aGUgcmVxdWlyZWQgYmVoYXZpb3IuCi0gICAgTE9HRCgic2V0Q2hpbGRIYW5kbGVyIGFkZGluZyBwaWQgJWQsIHRhZyAlZCwgaGFuZGxlciAlcCwgZGF0YSAlcCIsCi0gICAgICAgICBjaGlsZFBpZCwgdGFnLCBoYW5kbGVyLCB1c2VyRGF0YSk7Ci0gICAgc2VsZi0+bUNoaWxkSGFuZGxlcnMuYWRkKGNoaWxkUGlkLCBlbnRyeSk7Ci0KLSAgICBzZWxmLT5tTG9jay51bmxvY2soKTsKLQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotdm9pZCBTaWduYWxIYW5kbGVyOjpraWxsQWxsQ2hpbGRyZW4oaW50IHRhZykKLXsKLSAgICBTaWduYWxIYW5kbGVyKiBjb25zdCBzZWxmID0gZ2V0SW5zdGFuY2UoKTsKLQotICAgIEF1dG9NdXRleCBfbCAoc2VsZi0+bUxvY2spOwotICAgIGNvbnN0IHNpemVfdCBOID0gc2VsZi0+bUNoaWxkSGFuZGxlcnMuc2l6ZSgpOwotICAgIGZvciAoc2l6ZV90IGk9MDsgaTxOOyBpKyspIHsKLSAgICAgICAgY29uc3QgQ2hpbGRIYW5kbGVyJiBjaChzZWxmLT5tQ2hpbGRIYW5kbGVycy52YWx1ZUF0KGkpKTsKLSAgICAgICAgaWYgKHRhZyA9PSAwIHx8IGNoLnRhZyA9PSB0YWcpIHsKLSAgICAgICAgICAgIGNvbnN0IHBpZF90IHBpZCA9IGNoLmNoaWxkUGlkOwotICAgICAgICAgICAgTE9HSSgiS2lsbGluZyBjaGlsZCAlZCAodGFnICVkKVxuIiwgcGlkLCBjaC50YWcpOwotICAgICAgICAgICAga2lsbChwaWQsIFNJR0tJTEwpOwotICAgICAgICB9Ci0gICAgfQotfQotCi1TaWduYWxIYW5kbGVyOjpTaWduYWxIYW5kbGVyKCkKLSAgICA6IG1Db21tYW5kVG9wKDApCi0gICAgLCBtQ29tbWFuZEJvdHRvbSgwKQotICAgICwgbUxvc3RDb21tYW5kcygwKQotewotICAgIG1lbXNldChtQ29tbWFuZHMsIDAsIHNpemVvZihtQ29tbWFuZHMpKTsKLQotICAgIGludCByZXMgPSBwaXBlKG1BdmFpbE1zZyk7Ci0gICAgTE9HRV9JRihyZXMgIT0gMCwgIlVuYWJsZSB0byBjcmVhdGUgc2lnbmFsIGhhbmRsZXIgcGlwZTogJXMiLCBzdHJlcnJvcihlcnJubykpOwotCi0gICAgbVByb2Nlc3NUaHJlYWQgPSBuZXcgUHJvY2Vzc1RocmVhZCgqdGhpcyk7Ci0gICAgbVByb2Nlc3NUaHJlYWQtPnJ1bigiU2lnbmFsSGFuZGxlciIsIFBSSU9SSVRZX0hJR0hFU1QpOwotCi0gICAgc3RydWN0IHNpZ2FjdGlvbiBzYTsKLSAgICBtZW1zZXQoJnNhLCAwLCBzaXplb2Yoc2EpKTsKLSAgICBzYS5zYV9zaWdhY3Rpb24gPSBzaWdBY3Rpb247Ci0gICAgc2Euc2FfZmxhZ3MgPSBTQV9OT0NMRFNUT1B8U0FfU0lHSU5GTzsKLSAgICBzaWdhY3Rpb24oU0lHQ0hMRCwgJnNhLCBOVUxMKTsKLX0KLQotU2lnbmFsSGFuZGxlcjo6flNpZ25hbEhhbmRsZXIoKQotewotfQotCi1TaWduYWxIYW5kbGVyKiBTaWduYWxIYW5kbGVyOjpnZXRJbnN0YW5jZSgpCi17Ci0gICAgQXV0b011dGV4IF9sKG1JbnN0YW5jZUxvY2spOwotICAgIGlmIChtSW5zdGFuY2UgPT0gTlVMTCkgewotICAgICAgICBtSW5zdGFuY2UgPSBuZXcgU2lnbmFsSGFuZGxlcigpOwotICAgIH0KLSAgICByZXR1cm4gbUluc3RhbmNlOwotfQotCi12b2lkIFNpZ25hbEhhbmRsZXI6OnNpZ0FjdGlvbihpbnQgc2lnbnVtLCBzaWdpbmZvX3QqIGluZm8sIHZvaWQqKQotewotICAgIHN0YXRpYyBjb25zdCBjaGFyIHdha2V1cE1zZ1sxXSA9IHsgMHhmZiB9OwotCi0gICAgLy8gSWYgb3VyIHNpZ25hbCBoYW5kbGVyIGlzIGJlaW5nIGNhbGxlZCwgdGhlbiB3ZSBrbm93IHdlIGhhdmUKLSAgICAvLyBhbHJlYWR5IGluaXRpYWxpemVkIHRoZSBTaWduYWxIYW5kbGVyIGNsYXNzIGFuZCB0aHVzIG1JbnN0YW5jZQotICAgIC8vIGlzIHZhbGlkLgotICAgIFNpZ25hbEhhbmRsZXIqIGNvbnN0IHNlbGYgPSBtSW5zdGFuY2U7Ci0KLSAgICAvLyBYWFggVGhpcyBpcyBub3Qgc2FmZSEKLSAgICAjaWYgMAotICAgIExPR1YoIlNpZ25hbCAlZDogc2lnbm89JWQsIGVycm5vPSVkLCBjb2RlPSVkLCBwaWQ9JWRcbiIsCi0gICAgICAgICAgIHNpZ251bSwKLSAgICAgICAgICAgaW5mby0+c2lfc2lnbm8sIGluZm8tPnNpX2Vycm5vLCBpbmZvLT5zaV9jb2RlLAotICAgICAgICAgICBpbmZvLT5zaV9waWQpOwotICAgICNlbmRpZgotCi0gICAgaW50MzJfdCBvbGRUb3AsIG5ld1RvcDsKLQotICAgIC8vIEZpbmQgdGhlIG5leHQgY29tbWFuZCBzbG90Li4uCi0gICAgZG8gewotICAgICAgICBvbGRUb3AgPSBzZWxmLT5tQ29tbWFuZFRvcDsKLQotICAgICAgICBuZXdUb3AgPSBvbGRUb3AgKyAxOwotICAgICAgICBpZiAobmV3VG9wID49IENPTU1BTkRfUVVFVUVfU0laRSkgewotICAgICAgICAgICAgbmV3VG9wID0gMDsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChuZXdUb3AgPT0gc2VsZi0+bUNvbW1hbmRCb3R0b20pIHsKLSAgICAgICAgICAgIC8vIFRoZSBidWZmZXIgaXMgZmlsbGVkIHVwISAgT3VjaCEKLSAgICAgICAgICAgIC8vIFhYWCBUaGlzIGlzIG5vdCBzYWZlIQotICAgICAgICAgICAgI2lmIDAKLSAgICAgICAgICAgIExPR0UoIkNvbW1hbmQgYnVmZmVyIG92ZXJmbG93ISAgbmV3VG9wPSVkXG4iLCBuZXdUb3ApOwotICAgICAgICAgICAgI2VuZGlmCi0gICAgICAgICAgICBhbmRyb2lkX2F0b21pY19hZGQoMSwgJnNlbGYtPm1Mb3N0Q29tbWFuZHMpOwotICAgICAgICAgICAgd3JpdGUoc2VsZi0+bUF2YWlsTXNnWzFdLCB3YWtldXBNc2csIHNpemVvZih3YWtldXBNc2cpKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgIH0gd2hpbGUoYW5kcm9pZF9hdG9taWNfY21weGNoZyhvbGRUb3AsIG5ld1RvcCwgJihzZWxmLT5tQ29tbWFuZFRvcCkpKTsKLQotICAgIC8vIEZpbGwgaW4gdGhlIGNvbW1hbmQgZGF0YS4uLgotICAgIHNlbGYtPm1Db21tYW5kc1tvbGRUb3BdLnNpZ251bSA9IHNpZ251bTsKLSAgICBzZWxmLT5tQ29tbWFuZHNbb2xkVG9wXS5pbmZvID0gKmluZm87Ci0KLSAgICAvLyBBbmQgbm93IG1ha2UgdGhpcyBjb21tYW5kIGF2YWlsYWJsZS4KLSAgICBzZWxmLT5tQ29tbWFuZHNbb2xkVG9wXS5maWxsZWQgPSAxOwotCi0gICAgLy8gV2FrZSB1cCB0aGUgcHJvY2Vzc2luZyB0aHJlYWQuCi0gICAgd3JpdGUoc2VsZi0+bUF2YWlsTXNnWzFdLCB3YWtldXBNc2csIHNpemVvZih3YWtldXBNc2cpKTsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvY21kcy9ydW50aW1lL1NpZ25hbEhhbmRsZXIuaCBiL2NtZHMvcnVudGltZS9TaWduYWxIYW5kbGVyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDdmNGVmOGUuLjAwMDAwMDAKLS0tIGEvY21kcy9ydW50aW1lL1NpZ25hbEhhbmRsZXIuaAorKysgL2Rldi9udWxsCkBAIC0xLDEzNyArMCwwIEBACi0vLwotLy8gQ29weXJpZ2h0IDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotLy8KLSNpZm5kZWYgQU5EUk9JRF9TSUdOQUxfSEFORExFUl9ICi0jZGVmaW5lIEFORFJPSURfU0lHTkFMX0hBTkRMRVJfSAotCi0jaW5jbHVkZSA8dXRpbHMvS2V5ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLSNpbmNsdWRlIDxzaWduYWwuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWVudW0gewotICAgIERFRkFVTFRfUFJPQ0VTU19UQUcgPSAxCi19OwotCi1jbGFzcyBTaWduYWxIYW5kbGVyCi17Ci1wdWJsaWM6Ci0gICAgdHlwZWRlZiB2b2lkICgqY2hpbGRfY2FsbGJhY2tfdCkocGlkX3QgY2hpbGQsIHZvaWQqIHVzZXJEYXRhKTsKLQotICAgIC8qKgotICAgICAqIFNldCBhIGhhbmRsZXIgZm9yIHdoZW4gYSBjaGlsZCBwcm9jZXNzIGV4aXRzLiAgQnkgY2FsbGluZwotICAgICAqIHRoaXMsIGEgd2FpdHBpZCgpIHdpbGwgYmUgZG9uZSB3aGVuIHRoZSBjaGlsZCBleGl0cyB0byByZW1vdmUKLSAgICAgKiBpdCBmcm9tIHRoZSB6b21iaWUgc3RhdGUuICBZb3UgY2FuIGFsc28gb3B0aW9uYWxseSBzcGVjaWZ5IGEKLSAgICAgKiBoYW5kbGVyIHRvIGJlIGNhbGxlZCB3aGVuIHRoZSBjaGlsZCBleGl0cy4KLSAgICAgKiAKLSAgICAgKiBJZiB0aGVyZSBpcyBhbHJlYWR5IGEgaGFuZGxlciBmb3IgdGhpcyBjaGlsZCBwcm9jZXNzLCBpdCBpcwotICAgICAqIHJlcGxhY2VkIGJ5IHRoaXMgbmV3IGhhbmRsZXIuICBJbiB0aGlzIGNhc2UgdGhlIG9sZCBoYW5kbGVyJ3MKLSAgICAgKiBmdW5jdGlvbiBpcyBub3QgY2FsbGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSBjaGlsZFBpZCBQcm9jZXNzIElEIG9mIGNoaWxkIHRvIHdhdGNoLgotICAgICAqIEBwYXJhbSBjaGlsZFRhZyBVc2VyLWRlZmluZWQgdGFnIGZvciB0aGlzIGNoaWxkLiAgTXVzdCBiZQotICAgICAqICAgICAgICAgICAgICAgICBncmVhdGVyIHRoYW4gemVyby4KLSAgICAgKiBAcGFyYW0gaGFuZGxlciBJZiBub24tTlVMTCwgdGhpcyB3aWxsIGJlIGNhbGxlZCB3aGVuIHRoZQotICAgICAqICAgICAgICAgICAgICAgIGNoaWxkIGV4aXRzLiAgSXQgbWF5IGJlIGNhbGxlZCBpbiBlaXRoZXIgYQotICAgICAqICAgICAgICAgICAgICAgIHNlcGFyYXRlIHNpZ25hbCBoYW5kbGluZyB0aHJlYWQsIG9yCi0gICAgICogICAgICAgICAgICAgICAgaW1tZWRpYXRlbHkgaWYgdGhlIGNoaWxkIGhhcyBhbHJlYWR5IGV4aXRlZC4KLSAgICAgKiBAcGFyYW0gdXNlckRhdGEgUHJvcGFnZXRlZCBhcy1pcyB0byBoYW5kbGVyLgotICAgICAqIAotICAgICAqIEByZXR1cm4gc3RhdHVzX3QgTk9fRVJST1IgaWYgYWxsIGlzIHdlbGwuCi0gICAgICovCi0gICAgc3RhdGljIHN0YXR1c190ICAgICAgICAgICAgIHNldENoaWxkSGFuZGxlcihwaWRfdCBjaGlsZFBpZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBjaGlsZFRhZyA9IERFRkFVTFRfUFJPQ0VTU19UQUcsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjaGlsZF9jYWxsYmFja190IGhhbmRsZXIgPSBOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCogdXNlckRhdGEgPSBOVUxMKTsKLQotICAgIC8qKgotICAgICAqIEtpbGwgYWxsIG9mIHRoZSBjaGlsZCBwcm9jZXNzZXMgZm9yIHdoaWNoIHdlIGhhdmUgYSB3YWl0aW5nCi0gICAgICogaGFuZGxlciwgd2hvc2UgdGFnIGlzIHRoZSBnaXZlbiB2YWx1ZS4gIElmIHRhZyBpcyAwLCBhbGwKLSAgICAgKiBjaGlsZHJlbiBhcmUga2lsbGVkLgotICAgICAqIAotICAgICAqIEBwYXJhbSB0YWcKLSAgICAgKi8KLSAgICBzdGF0aWMgdm9pZCAgICAgICAgICAgICAgICAga2lsbEFsbENoaWxkcmVuKGludCB0YWcgPSAwKTsKLQotcHJpdmF0ZToKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU2lnbmFsSGFuZGxlcigpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB+U2lnbmFsSGFuZGxlcigpOwotCi0gICAgc3RhdGljIFNpZ25hbEhhbmRsZXIqICAgICAgIGdldEluc3RhbmNlKCk7Ci0KLSAgICBzdGF0aWMgdm9pZCAgICAgICAgICAgICAgICAgc2lnQWN0aW9uKGludCwgc2lnaW5mb190Kiwgdm9pZCopOwotCi0gICAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSAgICAvLyBTaGFyZWQgc3RhdGUuLi4gIGFsbCBvZiB0aGlzIGlzIHByb3RlY3RlZCBieSBtTG9jay4KLSAgICAvLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0gICAgbXV0YWJsZSBNdXRleCAgICAgICAgICAgICAgICAgICAgICAgbUxvY2s7Ci0KLSAgICBzdHJ1Y3QgQ2hpbGRIYW5kbGVyCi0gICAgewotICAgICAgICBwaWRfdCBjaGlsZFBpZDsKLSAgICAgICAgaW50IHRhZzsKLSAgICAgICAgY2hpbGRfY2FsbGJhY2tfdCBoYW5kbGVyOwotICAgICAgICB2b2lkKiB1c2VyRGF0YTsKLSAgICB9OwotICAgIEtleWVkVmVjdG9yPHBpZF90LCBDaGlsZEhhbmRsZXI+ICAgIG1DaGlsZEhhbmRsZXJzOwotCi0gICAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSAgICAvLyBDb21tbWFuZCBxdWV1ZS4uLiAgZGF0YSBpcyBpbnNlcnRlZCBieSB0aGUgc2lnbmFsCi0gICAgLy8gaGFuZGxlciB1c2luZyBhdG9taWMgb3BzLCBhbmQgcmV0cmlldmVkIGJ5IHRoZQotICAgIC8vIHNpZ25hbCBwcm9jZXNzaW5nIHRocmVhZC4gIEJlY2F1c2UgdGhlc2UgYXJlIHRvdWNoZWQKLSAgICAvLyBieSB0aGUgc2lnbmFsIGhhbmRsZXIsIG5vIGxvY2sgaXMgdXNlZC4KLSAgICAvLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0gICAgZW51bSB7Ci0gICAgICAgIENPTU1BTkRfUVVFVUVfU0laRSA9IDY0Ci0gICAgfTsKLSAgICBzdHJ1Y3QgQ29tbWFuZEVudHJ5Ci0gICAgewotICAgICAgICBpbnQgZmlsbGVkOwotICAgICAgICBpbnQgc2lnbnVtOwotICAgICAgICBzaWdpbmZvX3QgaW5mbzsKLSAgICB9OwotCi0gICAgLy8gVGhlIHRvcCBvZiB0aGUgcXVldWUuICBUaGlzIGlzIGluY3JlbWVudGVkIGF0b21pY2FsbHkgYnkgdGhlCi0gICAgLy8gc2lnbmFsIGhhbmRsZXIgYmVmb3JlIHBsYWNpbmcgYSBjb21tYW5kIGluIHRoZSBxdWV1ZS4KLSAgICB2b2xhdGlsZSBpbnQzMl90ICAgICAgICAgICAgICAgICAgICBtQ29tbWFuZFRvcDsKLQotICAgIC8vIFRoZSBib3R0b20gb2YgdGhlIHF1ZXVlLiAgT25seSBtb2RpZmllZCBieSB0aGUgcHJvY2Vzc2luZwotICAgIC8vIHRocmVhZDsgdGhlIHNpZ25hbCBoYW5kbGVyIHJlYWRzIGl0IG9ubHkgdG8gZGV0ZXJtaW5lIGlmIHRoZQotICAgIC8vIHF1ZXVlIGlzIGZ1bGwuCi0gICAgaW50MzJfdCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbUNvbW1hbmRCb3R0b207Ci0KLSAgICAvLyBJbmNyZW1lbnRlZCBlYWNoIHRpbWUgd2UgcmVjZWl2ZSBhIHNpZ25hbCBhbmQgZG9uJ3QgaGF2ZSByb29tCi0gICAgLy8gZm9yIGl0IG9uIHRoZSBjb21tYW5kIHF1ZXVlLgotICAgIHZvbGF0aWxlIGludDMyX3QgICAgICAgICAgICAgICAgICAgIG1Mb3N0Q29tbWFuZHM7Ci0KLSAgICAvLyBUaGUgY29tbWFuZCBwcm9jZXNzaW5nIHRocmVhZC4KLSAgICBjbGFzcyBQcm9jZXNzVGhyZWFkOwotICAgIHNwPFRocmVhZD4gICAgICAgICAgICAgICAgICAgICAgICAgIG1Qcm9jZXNzVGhyZWFkOwotCi0gICAgLy8gUGlwZSB1c2VkIHRvIHRlbGwgY29tbWFuZCBwcm9jZXNzaW5nIHRocmVhZCB3aGVuIG5ldyBjb21tYW5kcy4KLSAgICAvLyBhcmUgYXZhaWxhYmxlLiAgVGhlIHRocmVhZCBibG9ja3Mgb24gdGhlIHJlYWQgZW5kLCB0aGUgc2lnbmFsCi0gICAgLy8gaGFuZGxlciB3cml0ZXMgd2hlbiBpdCBlbnF1ZXVlcyBuZXcgY29tbWFuZHMuCi0gICAgaW50ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbUF2YWlsTXNnWzJdOwotCi0gICAgLy8gVGhlIGNvbW1hbmRzLgotICAgIENvbW1hbmRFbnRyeSAgICAgICAgICAgICAgICAgICAgICAgIG1Db21tYW5kc1tDT01NQU5EX1FVRVVFX1NJWkVdOwotCi0gICAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSAgICAvLyBTaW5nbGV0b24uCi0gICAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotICAgIHN0YXRpYyBNdXRleCAgICAgICAgICAgICAgICAgICAgICAgIG1JbnN0YW5jZUxvY2s7Ci0gICAgc3RhdGljIFNpZ25hbEhhbmRsZXIqICAgICAgICAgICAgICAgbUluc3RhbmNlOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9TSUdOQUxfSEFORExFUl9ICmRpZmYgLS1naXQgYS9jbWRzL3J1bnRpbWUvbWFpbl9ydW50aW1lLmNwcCBiL2NtZHMvcnVudGltZS9tYWluX3J1bnRpbWUuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxNTMxYTllLi4wMDAwMDAwCi0tLSBhL2NtZHMvcnVudGltZS9tYWluX3J1bnRpbWUuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNTE0ICswLDAgQEAKLS8vCi0vLyBDb3B5cmlnaHQgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0vLwotLy8gTWFpbiBlbnRyeSBwb2ludCBmb3IgcnVudGltZS4KLS8vCi0KLSNpbmNsdWRlICJTZXJ2aWNlTWFuYWdlci5oIgotI2luY2x1ZGUgIlNpZ25hbEhhbmRsZXIuaCIKLQotI2luY2x1ZGUgPHV0aWxzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSVBDVGhyZWFkU3RhdGUuaD4KLSNpbmNsdWRlIDx1dGlscy9Qcm9jZXNzU3RhdGUuaD4KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4gIAotI2luY2x1ZGUgPGN1dGlscy96eWdvdGUuaD4KLQotI2luY2x1ZGUgPGN1dGlscy9wcm9wZXJ0aWVzLmg+Ci0KLSNpbmNsdWRlIDxwcml2YXRlL3V0aWxzL1N0YXRpYy5oPgotCi0jaW5jbHVkZSA8dWkvSVN1cmZhY2VDb21wb3Nlci5oPgotCi0jaW5jbHVkZSA8YW5kcm9pZF9ydW50aW1lL0FuZHJvaWRSdW50aW1lLmg+Ci0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8Z2V0b3B0Lmg+Ci0jaW5jbHVkZSA8c2lnbmFsLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxzeXMvc3RhdC5oPgotI2luY2x1ZGUgPGxpbnV4L2NhcGFiaWxpdHkuaD4KLSNpbmNsdWRlIDxsaW51eC9pb2N0bC5oPgotI2lmZGVmIEhBVkVfQU5EUk9JRF9PUwotIyBpbmNsdWRlIDxsaW51eC9hbmRyb2lkX2FsYXJtLmg+Ci0jZW5kaWYKLQotI3VuZGVmIExPR19UQUcKLSNkZWZpbmUgTE9HX1RBRyAicnVudGltZSIKLQotc3RhdGljIGNvbnN0IGNoYXIqIFpZR09URV9BUkdWW10gPSB7IAotICAgICItLXNldHVpZD0xMDAwIiwKLSAgICAiLS1zZXRnaWQ9MTAwMCIsCi0gICAgIi0tc2V0Z3JvdXBzPTEwMDEsMTAwMiwxMDAzLDEwMDQsMTAwNSwxMDA2LDEwMDcsMTAwOCwxMDA5LDEwMTAsMzAwMSwzMDAyLDMwMDMiLAotICAgIC8qIENBUF9TWVNfVFRZX0NPTkZJRyAmIENBUF9TWVNfUkVTT1VSQ0UgJiBDQVBfTkVUX0JST0FEQ0FTVCAmCi0gICAgICogQ0FQX05FVF9BRE1JTiAmIENBUF9ORVRfUkFXICYgQ0FQX05FVF9CSU5EX1NFUlZJQ0UgICYgQ0FQX0tJTEwgJgotICAgICAqIENBUF9TWVNfQk9PVAotICAgICAqLwotICAgICItLWNhcGFiaWxpdGllcz04ODE2MTMxMiw4ODE2MTMxMiIsCi0gICAgIi0tcnVudGltZS1pbml0IiwKLSAgICAiLS1uaWNlLW5hbWU9c3lzdGVtX3NlcnZlciIsCi0gICAgImNvbS5hbmRyb2lkLnNlcnZlci5TeXN0ZW1TZXJ2ZXIiCi19OwotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotZXh0ZXJuICJDIiBzdGF0dXNfdCBzeXN0ZW1faW5pdCgpOwotCi1lbnVtIHsKLSAgICBTWVNURU1fUFJPQ0VTU19UQUcgPSBERUZBVUxUX1BST0NFU1NfVEFHKzEKLX07Ci0KLWV4dGVybiBNdXRleCBnRXZlbnRRTXV0ZXg7Ci1leHRlcm4gQ29uZGl0aW9uIGdFdmVudFFDb25kaXRpb247Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotZXh0ZXJuIHN0YXR1c190IGFwcF9pbml0KGNvbnN0IGNoYXIqIGNsYXNzTmFtZSk7Ci1leHRlcm4gdm9pZCBzZXRfZmluaXNoX2luaXRfZnVuYyh2b2lkICgqZnVuYykoKSk7Ci0KLQotLyoqCi0gKiBUaGlzIGNsYXNzIGlzIHVzZWQgdG8ga2lsbCB0aGlzIHByb2Nlc3MgKHJ1bnRpbWUpIHdoZW4gdGhlIHN5c3RlbV9zZXJ2ZXIgZGllcy4KLSAqLwotY2xhc3MgR3JpbVJlYXBlciA6IHB1YmxpYyBJQmluZGVyOjpEZWF0aFJlY2lwaWVudCB7Ci1wdWJsaWM6IAotICAgIEdyaW1SZWFwZXIoKSB7IH0KLQotICAgIHZpcnR1YWwgdm9pZCBiaW5kZXJEaWVkKGNvbnN0IHdwPElCaW5kZXI+JiB3aG8pCi0gICAgewotICAgICAgICBMT0dJKCJHcmltIFJlYXBlciBraWxsaW5nIHJ1bnRpbWUuLi4iKTsKLSAgICAgICAga2lsbChnZXRwaWQoKSwgU0lHS0lMTCk7Ci0gICAgfQotfTsKLQotZXh0ZXJuIHZvaWQgUXVpY2tUZXN0cygpOwotCi0vKgotICogUHJpbnQgdXNhZ2UgaW5mby4KLSAqLwotc3RhdGljIHZvaWQgdXNhZ2UoY29uc3QgY2hhciogYXJndjApCi17Ci0gICAgZnByaW50ZihzdGRlcnIsCi0gICAgICAgICJVc2FnZTogcnVudGltZSBbLWcgZ2FtbWFdIFstbCBsb2dmaWxlXSBbLW5dIFstc11cbiIKLSAgICAgICAgIiAgICAgICAgICAgICAgIFstaiBhcHAtY29tcG9uZW50XSBbLXYgYXBwLXZlcmJdIFstZCBhcHAtZGF0YV1cbiIKLSAgICAgICAgIlxuIgotICAgICAgICAiLWw6IEZpbGUgdG8gc2VuZCBsb2cgbWVzc2FnZXMgdG9cbiIKLSAgICAgICAgIi1uOiBEb24ndCBwcmludCB0byBzdGRvdXQvc3RkZXJyXG4iCi0gICAgICAgICItczogRm9yY2Ugc2luZ2xlLXByb2Nlc3MgbW9kZVxuIgotICAgICAgICAiLWo6IEN1c3RvbSBob21lIGFwcCBjb21wb25lbnQgbmFtZVxuIgotICAgICAgICAiLXY6IEN1c3RvbSBob21lIGFwcCBpbnRlbnQgdmVyYlxuIgotICAgICAgICAiLWQ6IEN1c3RvbSBob21lIGFwcCBpbnRlbnQgZGF0YVxuIgotICAgICk7Ci0gICAgZXhpdCgxKTsKLX0KLQotLy8gU2VsZWN0ZWQgYXBwbGljYXRpb24gdG8gcnVuLgotc3RhdGljIGNvbnN0IGNoYXIqIGdJbml0aWFsQXBwbGljYXRpb24gPSBOVUxMOwotc3RhdGljIGNvbnN0IGNoYXIqIGdJbml0aWFsVmVyYiA9IE5VTEw7Ci1zdGF0aWMgY29uc3QgY2hhciogZ0luaXRpYWxEYXRhID0gTlVMTDsKLQotc3RhdGljIHZvaWQgd3JpdGVTdHJpbmdUb1BhcmNlbChQYXJjZWwmIHBhcmNlbCwgY29uc3QgY2hhciogc3RyKQotewotICAgIGlmIChzdHIpIHsKLSAgICAgICAgcGFyY2VsLndyaXRlU3RyaW5nMTYoU3RyaW5nMTYoc3RyKSk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgcGFyY2VsLndyaXRlU3RyaW5nMTYoTlVMTCwgMCk7Ci0gICAgfQotfQotCi0vKgotICogU3RhcnRpbmcgcG9pbnQgZm9yIHByb2dyYW0gbG9naWMuCi0gKgotICogUmV0dXJucyB3aXRoIGFuIGV4aXQgc3RhdHVzIGNvZGUgKDAgb24gc3VjY2Vzcywgbm9uemVybyBvbiBlcnJvcikuCi0gKi8KLXN0YXRpYyBpbnQgcnVuKHNwPFByb2Nlc3NTdGF0ZT4mIHByb2MpCi17Ci0gICAgLy8gVGVtcG9yYXJ5IGhhY2sgdG8gY2FsbCBzdGFydFJ1bm5pbmcoKSBvbiB0aGUgYWN0aXZpdHkgbWFuYWdlci4KLSAgICBzcDxJU2VydmljZU1hbmFnZXI+IHNtID0gZGVmYXVsdFNlcnZpY2VNYW5hZ2VyKCk7Ci0gICAgc3A8SUJpbmRlcj4gYW07Ci0gICAgd2hpbGUgKChhbSA9IHNtLT5nZXRTZXJ2aWNlKFN0cmluZzE2KCJhY3Rpdml0eSIpKSkgPT0gTlVMTCkgewotICAgICAgICBMT0dJKCJXYWl0aW5nIGZvciBhY3Rpdml0eSBtYW5hZ2VyLi4uIik7Ci0gICAgfQotICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAvLyBYWFggTmVlZCB0byBhbHNvIHN1cHBseSBhIHBhY2thZ2UgbmFtZSBmb3IgdGhpcyB0byB3b3JrIGFnYWluLgotICAgIC8vIElBY3Rpdml0eU1hbmFnZXI6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSBpcyB0aGUgdG9rZW4gZm9yIGludm9raW5nIG9uIHRoaXMgaW50ZXJmYWNlOwotICAgIC8vIGhhcmRjb2RpbmcgaXQgaGVyZSBhdm9pZHMgaGF2aW5nIHRvIGxpbmsgd2l0aCB0aGUgZnVsbCBBY3Rpdml0eSBNYW5hZ2VyIGxpYnJhcnkKLSAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oU3RyaW5nMTYoImFuZHJvaWQuYXBwLklBY3Rpdml0eU1hbmFnZXIiKSk7Ci0gICAgd3JpdGVTdHJpbmdUb1BhcmNlbChkYXRhLCBOVUxMKTsKLSAgICB3cml0ZVN0cmluZ1RvUGFyY2VsKGRhdGEsIGdJbml0aWFsQXBwbGljYXRpb24pOwotICAgIHdyaXRlU3RyaW5nVG9QYXJjZWwoZGF0YSwgZ0luaXRpYWxWZXJiKTsKLSAgICB3cml0ZVN0cmluZ1RvUGFyY2VsKGRhdGEsIGdJbml0aWFsRGF0YSk7Ci1MT0dJKCJydW4oKSBzZW5kaW5nIEZJUlNUX0NBTExfVFJBTlNBQ1RJT04gdG8gYWN0aXZpdHkgbWFuYWdlciIpOwotICAgIGFtLT50cmFuc2FjdChJQmluZGVyOjpGSVJTVF9DQUxMX1RSQU5TQUNUSU9OLCBkYXRhLCAmcmVwbHkpOwotCi0gICAgaWYgKHByb2MtPnN1cHBvcnRzUHJvY2Vzc2VzKCkpIHsKLSAgICAgICAgLy8gTm93IHdlIGxpbmsgdG8gdGhlIEFjdGl2aXR5IE1hbmFnZXIgd2FpdGluZyBmb3IgaXQgdG8gZGllLiBJZiBpdCBkb2VzIGtpbGwgb3Vyc2VsZi4KLSAgICAgICAgLy8gaW5pdGQgd2lsbCByZXN0YXJ0IHRoaXMgcHJvY2VzcyBhbmQgYnJpbmcgdGhlIHN5c3RlbSBiYWNrIHVwLgotICAgICAgICBzcDxHcmltUmVhcGVyPiBncmltID0gbmV3IEdyaW1SZWFwZXIoKTsKLSAgICAgICAgYW0tPmxpbmtUb0RlYXRoKGdyaW0sIGdyaW0uZ2V0KCksIDApOwotCi0gICAgICAgIC8vIE5vdyBqb2luIHRoZSB0aHJlYWQgcG9vbC4gTm90ZSB0aGlzIGlzIG5lZWRlZCBzbyB0aGF0IHRoZSBtZXNzYWdlIGVucXVldWVkIGluIHRoZSBkcml2ZXIKLSAgICAgICAgLy8gZm9yIHRoZSBsaW5rVG9EZWF0aCBnZXRzIHByb2Nlc3NlZC4KLSAgICAgICAgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+am9pblRocmVhZFBvb2woKTsKLSAgICB9IGVsc2UgewotICAgICAgICAvLyBLZWVwIHRoaXMgdGhyZWFkIHJ1bm5pbmcgZm9yZXZlci4uLgotICAgICAgICB3aGlsZSAoMSkgewotICAgICAgICAgICAgdXNsZWVwKDEwMDAwMCk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIDE7Ci19Ci0KLQotfTsgIC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLQotLyoKLSAqIFBvc3Qtc3lzdGVtLXByb2Nlc3MgaW5pdGlhbGl6YXRpb24uCi0gKiAKLSAqIFRoaXMgZnVuY3Rpb24gY29udGludWVzIGluaXRpYWxpemF0aW9uIGFmdGVyIHRoZSBzeXN0ZW0gcHJvY2VzcwotICogaGFzIGJlZW4gaW5pdGlhbGl6ZWQuICBJdCBuZWVkcyB0byBiZSBzZXBhcmF0ZSBiZWNhdXNlIHRoZSBzeXN0ZW0KLSAqIGluaXRpYWxpemF0aW9uIG5lZWRzIHRvIGNhcmUgb2Ygc3RhcnRpbmcgdGhlIEFuZHJvaWQgcnVudGltZSBpZiBpdCBpcyBub3QKLSAqIHJ1bm5pbmcgaW4gaXRzIG93biBwcm9jZXNzLCB3aGljaCBkb2Vzbid0IHJldHVybiB1bnRpbCB0aGUgcnVudGltZSBpcwotICogYmVpbmcgc2h1dCBkb3duLiAgU28gaXQgd2lsbCBjYWxsIGJhY2sgdG8gaGVyZSBmcm9tIGluc2lkZSBvZiBEYWx2aWssCi0gKiB0byBhbGxvdyB1cyB0byBjb250aW51ZSBib290aW5nIHVwLgotICovCi1zdGF0aWMgdm9pZCBmaW5pc2hfc3lzdGVtX2luaXQoc3A8UHJvY2Vzc1N0YXRlPiYgcHJvYykKLXsKLSAgICAvLyBJZiB3ZSBhcmUgcnVubmluZyBtdWx0aXByb2Nlc3MsIHdlIG5vdyBuZWVkIHRvIGhhdmUgdGhlCi0gICAgLy8gdGhyZWFkIHBvb2wgc3RhcnRlZCBoZXJlLiAgV2UgZG9uJ3QgZG8gdGhpcyBpbiBib290X2luaXQoKQotICAgIC8vIGJlY2F1c2Ugd2hlbiBydW5uaW5nIHNpbmdsZSBwcm9jZXNzIHdlIG5lZWQgdG8gc3RhcnQgdGhlCi0gICAgLy8gdGhyZWFkIHBvb2wgYWZ0ZXIgdGhlIEFuZHJvaWQgcnVudGltZSBoYXMgYmVlbiBzdGFydGVkIChzbwotICAgIC8vIHRoZSBwb29sIHVzZXMgRGFsdmlrIHRocmVhZHMpLgotICAgIGlmIChwcm9jLT5zdXBwb3J0c1Byb2Nlc3NlcygpKSB7Ci0gICAgICAgIHByb2MtPnN0YXJ0VGhyZWFkUG9vbCgpOwotICAgIH0KLX0KLQotCi0vLyBUaGlzIGZ1bmN0aW9uIGNhbiBiZSB1c2VkIHRvIGVuZm9yY2Ugc2VjdXJpdHkgdG8gZGlmZmVyZW50Ci0vLyByb290IGNvbnRleHRzLiAgRm9yIG5vdywgd2UganVzdCBnaXZlIGV2ZXJ5IGFjY2Vzcy4KLXN0YXRpYyBib29sIGNvbnRleHRDaGVja2VyKAotICAgIGNvbnN0IFN0cmluZzE2JiBuYW1lLCBjb25zdCBzcDxJQmluZGVyPiYgY2FsbGVyLCB2b2lkKiB1c2VyRGF0YSkKLXsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotLyoKLSAqIEluaXRpYWxpemF0aW9uIG9mIGJvb3Qgc2VydmljZXMuCi0gKgotICogVGhpcyBpcyB3aGVyZSB3ZSBwZXJmb3JtIGluaXRpYWxpemF0aW9uIG9mIGFsbCBvZiBvdXIgbG93LWxldmVsCi0gKiBib290IHNlcnZpY2VzLiAgTW9zdCBpbXBvcnRhbnRseSwgaGVyZSB3ZSBiZWNvbWUgdGhlIGNvbnRleHQKLSAqIG1hbmFnZXIgYW5kIHVzZSB0aGF0IHRvIHB1Ymxpc2ggdGhlIHNlcnZpY2UgbWFuYWdlciB0aGF0IHdpbGwgcHJvdmlkZQotICogYWNjZXNzIHRvIGFsbCBvdGhlciBzZXJ2aWNlcy4KLSAqLwotc3RhdGljIHZvaWQgYm9vdF9pbml0KCkKLXsKLSAgICBMT0dJKCJFbnRlcmVkIGJvb3RfaW5pdCgpIVxuIik7Ci0gICAgCi0gICAgc3A8UHJvY2Vzc1N0YXRlPiBwcm9jKFByb2Nlc3NTdGF0ZTo6c2VsZigpKTsKLSAgICBMT0dEKCJQcm9jZXNzU3RhdGU6ICVwXG4iLCBwcm9jLmdldCgpKTsKLSAgICBwcm9jLT5iZWNvbWVDb250ZXh0TWFuYWdlcihjb250ZXh0Q2hlY2tlciwgTlVMTCk7Ci0gICAgCi0gICAgaWYgKHByb2MtPnN1cHBvcnRzUHJvY2Vzc2VzKCkpIHsKLSAgICAgICAgTE9HSSgiQmluZGVyIGRyaXZlciBvcGVuZWQuICBNdWx0aXByb2Nlc3MgZW5hYmxlZC5cbiIpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIExPR0koIkJpbmRlciBkcml2ZXIgbm90IGZvdW5kLiAgUHJvY2Vzc2VzIG5vdCBzdXBwb3J0ZWQuXG4iKTsKLSAgICB9Ci0gICAgCi0gICAgc3A8QlNlcnZpY2VNYW5hZ2VyPiBzbSA9IG5ldyBCU2VydmljZU1hbmFnZXI7Ci0gICAgcHJvYy0+c2V0Q29udGV4dE9iamVjdChzbSk7Ci19Ci0KLS8qCi0gKiBSZWRpcmVjdCBzdGRpbi9zdGRvdXQvc3RkZXJyIHRvIC9kZXYvbnVsbC4KLSAqLwotc3RhdGljIHZvaWQgcmVkaXJlY3RTdGRGZHModm9pZCkKLXsKLSAgICBpbnQgZmQgPSBvcGVuKCIvZGV2L251bGwiLCBPX1JEV1IsIDApOwotICAgIGlmIChmZCA8IDApIHsKLSAgICAgICAgTE9HVygiVW5hYmxlIHRvIG9wZW4gL2Rldi9udWxsOiAlc1xuIiwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICB9IGVsc2UgewotICAgICAgICBkdXAyKGZkLCAwKTsKLSAgICAgICAgZHVwMihmZCwgMSk7Ci0gICAgICAgIGR1cDIoZmQsIDIpOwotICAgICAgICBjbG9zZShmZCk7Ci0gICAgfQotfQotCi1zdGF0aWMgaW50IGhhc0Rpcihjb25zdCBjaGFyKiBkaXIpCi17Ci0gICAgc3RydWN0IHN0YXQgczsKLSAgICBpbnQgcmVzID0gc3RhdChkaXIsICZzKTsKLSAgICBpZiAocmVzID09IDApIHsKLSAgICAgICAgcmV0dXJuIFNfSVNESVIocy5zdF9tb2RlKTsKLSAgICB9Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXN0YXRpYyB2b2lkIHZhbGlkYXRlVGltZSgpCi17Ci0jaWYgSEFWRV9BTkRST0lEX09TCi0gICAgaW50IGZkOwotICAgIGludCByZXM7Ci0gICAgdGltZV90IG1pbl90aW1lID0gMTE2NzY1MjgwMDsgLy8gamFuIDEgMjAwNywgdHlwZSAnZGF0ZSAtdWQgIjEvMSAxMjowMCIgKyVzJyB0byBnZXQgdmFsdWUgZm9yIGN1cnJlbnQgeWVhcgotICAgIHN0cnVjdCB0aW1lc3BlYyB0czsKLSAgICAKLSAgICBmZCA9IG9wZW4oIi9kZXYvYWxhcm0iLCBPX1JEV1IpOwotICAgIGlmKGZkIDwgMCkgewotICAgICAgICBMT0dXKCJVbmFibGUgdG8gb3BlbiBhbGFybSBkcml2ZXI6ICVzXG4iLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIHJlcyA9IGlvY3RsKGZkLCBBTkRST0lEX0FMQVJNX0dFVF9USU1FKEFORFJPSURfQUxBUk1fUlRDX1dBS0VVUCksICZ0cyk7Ci0gICAgaWYocmVzIDwgMCkgewotICAgICAgICBMT0dXKCJVbmFibGUgdG8gcmVhZCBydGMsICVzXG4iLCBzdHJlcnJvcihlcnJubykpOwotICAgIH0KLSAgICBlbHNlIGlmKHRzLnR2X3NlYyA+PSBtaW5fdGltZSkgewotICAgICAgICBnb3RvIGRvbmU7Ci0gICAgfQotICAgIExPR1coIkludmFsaWQgdGltZSBkZXRlY3RlZCwgJWxkIHNldCB0byAlbGRcbiIsIHRzLnR2X3NlYywgbWluX3RpbWUpOwotICAgIHRzLnR2X3NlYyA9IG1pbl90aW1lOwotICAgIHRzLnR2X25zZWMgPSAwOwotICAgIHJlcyA9IGlvY3RsKGZkLCBBTkRST0lEX0FMQVJNX1NFVF9SVEMsICZ0cyk7Ci0gICAgaWYocmVzIDwgMCkgewotICAgICAgICBMT0dXKCJVbmFibGUgdG8gc2V0IHJ0YyB0byAlbGQ6ICVzXG4iLCB0cy50dl9zZWMsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgfQotZG9uZToKLSAgICBjbG9zZShmZCk7Ci0jZW5kaWYKLX0KLQotI2lmbmRlZiBIQVZFX0FORFJPSURfT1MKLWNsYXNzIFF1aWNrUnVudGltZSA6IHB1YmxpYyBBbmRyb2lkUnVudGltZQotewotcHVibGljOgotICAgIFF1aWNrUnVudGltZSgpIHt9Ci0KLSAgICB2aXJ0dWFsIHZvaWQgb25TdGFydGVkKCkKLSAgICB7Ci0gICAgICAgIHByaW50ZigiUXVpY2tSdW50aW1lOiBvblN0YXJ0ZWRcbiIpOwotICAgIH0KLX07Ci0jZW5kaWYKLQotc3RhdGljIHN0YXR1c190IHN0YXJ0X3Byb2Nlc3MoY29uc3QgY2hhciogbmFtZSk7Ci0KLXN0YXRpYyB2b2lkIHJlc3RhcnRfbWUocGlkX3QgY2hpbGQsIHZvaWQqIHVzZXJEYXRhKQotewotICAgIHN0YXJ0X3Byb2Nlc3MoKGNvbnN0IGNoYXIqKXVzZXJEYXRhKTsKLX0KLQotc3RhdGljIHN0YXR1c190IHN0YXJ0X3Byb2Nlc3MoY29uc3QgY2hhciogbmFtZSkKLXsKLSAgICBTdHJpbmc4IHBhdGgobmFtZSk7Ci0gICAgVmVjdG9yPGNvbnN0IGNoYXIqPiBhcmdzOwotICAgIFN0cmluZzggbGVhZihwYXRoLmdldFBhdGhMZWFmKCkpOwotICAgIFN0cmluZzggcGFyZW50RGlyKHBhdGguZ2V0UGF0aERpcigpKTsKLSAgICBhcmdzLmluc2VydEF0KGxlYWYuc3RyaW5nKCksIDApOwotICAgIGFyZ3MuYWRkKHBhcmVudERpci5zdHJpbmcoKSk7Ci0gICAgYXJncy5hZGQoTlVMTCk7Ci0gICAgcGlkX3QgY2hpbGQgPSBmb3JrKCk7Ci0gICAgaWYgKGNoaWxkIDwgMCkgewotICAgICAgICBzdGF0dXNfdCBlcnIgPSBlcnJubzsKLSAgICAgICAgTE9HRSgiKioqIGZvcmsgb2YgY2hpbGQgJXMgZmFpbGVkOiAlcyIsIGxlYWYuc3RyaW5nKCksIHN0cmVycm9yKGVycikpOwotICAgICAgICByZXR1cm4gLWVycm5vOwotICAgIH0gZWxzZSBpZiAoY2hpbGQgPT0gMCkgewotICAgICAgICBMT0dJKCJFeGVjdXRpbmc6ICVzIiwgcGF0aC5zdHJpbmcoKSk7Ci0gICAgICAgIGV4ZWN2KHBhdGguc3RyaW5nKCksIGNvbnN0X2Nhc3Q8Y2hhcioqPihhcmdzLmFycmF5KCkpKTsKLSAgICAgICAgaW50IGVyciA9IGVycm5vOwotICAgICAgICBMT0dFKCJFeGVjIGZhaWxlZDogJXNcbiIsIHN0cmVycm9yKGVycikpOwotICAgICAgICBfZXhpdChlcnIpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIFNpZ25hbEhhbmRsZXI6OnNldENoaWxkSGFuZGxlcihjaGlsZCwgREVGQVVMVF9QUk9DRVNTX1RBRywKLSAgICAgICAgICAgICAgICByZXN0YXJ0X21lLCAodm9pZCopbmFtZSk7Ci0gICAgfQotICAgIHJldHVybiAtZXJybm87Ci19Ci0KLS8qCi0gKiBBcHBsaWNhdGlvbiBlbnRyeSBwb2ludC4KLSAqCi0gKiBQYXJzZSBhcmd1bWVudHMsIHNldCBzb21lIHZhbHVlcywgYW5kIHBhc3MgY29udHJvbCBvZmYgdG8gUnVuKCkuCi0gKgotICogVGhpcyBpcyByZWRlZmluZWQgdG8gIlNETF9tYWluIiBvbiBTREwgc2ltdWxhdG9yIGJ1aWxkcywgYW5kCi0gKiAicnVudGltZV9tYWluIiBvbiB3eFdpZGdldHMgYnVpbGRzLgotICovCi1leHRlcm4gIkMiCi1pbnQgbWFpbihpbnQgYXJnYywgY2hhciogY29uc3QgYXJndltdKQotewotICAgIGJvb2wgc2luZ2xlUHJvY2VzcyA9IGZhbHNlOwotICAgIGNvbnN0IGNoYXIqIGxvZ0ZpbGUgPSBOVUxMOwotICAgIGludCBpYzsKLSAgICBpbnQgcmVzdWx0ID0gMTsKLSAgICBwaWRfdCBzeXN0ZW1QaWQ7Ci0gICAgCi0gICAgc3A8UHJvY2Vzc1N0YXRlPiBwcm9jOwotCi0jaWZuZGVmIEhBVkVfQU5EUk9JRF9PUwotICAgIC8qIFNldCBzdGRvdXQvc3RkZXJyIHRvIHVuYnVmZmVyZWQgZm9yIE1pbkdXL01TWVMuICovCi0gICAgLy9zZXR2YnVmKHN0ZG91dCwgTlVMTCwgX0lPTkJGLCAwKTsKLSAgICAvL3NldHZidWYoc3RkZXJyLCBOVUxMLCBfSU9OQkYsIDApOwotICAgIAotICAgIExPR0koImNvbW1hbmRsaW5lIGFyZ3M6XG4iKTsKLSAgICBmb3IgKGludCBpID0gMDsgaSA8IGFyZ2M7IGkrKykKLSAgICAgICAgTE9HSSgiICAlMmQ6ICclcydcbiIsIGksIGFyZ3ZbaV0pOwotI2VuZGlmCi0KLSAgICB3aGlsZSAoMSkgewotICAgICAgICBpYyA9IGdldG9wdChhcmdjLCBhcmd2LCAiZzpqOnY6ZDpsOm5zIik7Ci0gICAgICAgIGlmIChpYyA8IDApCi0gICAgICAgICAgICBicmVhazsKLQotICAgICAgICBzd2l0Y2ggKGljKSB7Ci0gICAgICAgIGNhc2UgJ2cnOgotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgJ2onOgotICAgICAgICAgICAgZ0luaXRpYWxBcHBsaWNhdGlvbiA9IG9wdGFyZzsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlICd2JzoKLSAgICAgICAgICAgIGdJbml0aWFsVmVyYiA9IG9wdGFyZzsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlICdkJzoKLSAgICAgICAgICAgIGdJbml0aWFsRGF0YSA9IG9wdGFyZzsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlICdsJzoKLSAgICAgICAgICAgIGxvZ0ZpbGUgPSBvcHRhcmc7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSAnbic6Ci0gICAgICAgICAgICByZWRpcmVjdFN0ZEZkcygpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgJ3MnOgotICAgICAgICAgICAgc2luZ2xlUHJvY2VzcyA9IHRydWU7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSAnPyc6Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICBMT0dFKCJydW50aW1lOiB1bnJlY29nbml6ZWQgZmxhZyAtJWNcbiIsIGljKTsKLSAgICAgICAgICAgIHVzYWdlKGFyZ3ZbMF0pOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgaWYgKG9wdGluZCA8IGFyZ2MpIHsKLSAgICAgICAgTE9HRSgicnVudGltZTogZXh0cmEgc3R1ZmY6ICVzXG4iLCBhcmd2W29wdGluZF0pOwotICAgICAgICB1c2FnZShhcmd2WzBdKTsKLSAgICB9Ci0KLSAgICBpZiAoc2luZ2xlUHJvY2VzcykgewotICAgICAgICBQcm9jZXNzU3RhdGU6OnNldFNpbmdsZVByb2Nlc3ModHJ1ZSk7Ci0gICAgfQotCi0gICAgaWYgKGxvZ0ZpbGUgIT0gTlVMTCkgewotICAgICAgICBhbmRyb2lkX2xvZ1RvRmlsZShOVUxMLCBsb2dGaWxlKTsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFNldCB1cCBBTkRST0lEXyogZW52aXJvbm1lbnQgdmFyaWFibGVzLgotICAgICAqCi0gICAgICogVE9ETzogdGhlIHVzZSBvZiAkQU5EUk9JRF9QUk9EVUNUX09VVCB3aWxsIGdvIGF3YXkgc29vbi4KLSAgICAgKi8KLSAgICBzdGF0aWMgY29uc3QgY2hhcioga1N5c3RlbURpciA9ICIvc3lzdGVtIjsKLSAgICBzdGF0aWMgY29uc3QgY2hhcioga0RhdGFEaXIgPSAiL2RhdGEiOwotICAgIHN0YXRpYyBjb25zdCBjaGFyKiBrQXBwU3ViZGlyID0gIi9hcHAiOwotICAgIGNvbnN0IGNoYXIqIG91dCA9IE5VTEw7Ci0jaWZuZGVmIEhBVkVfQU5EUk9JRF9PUwotICAgIC8vb3V0ID0gZ2V0ZW52KCJBTkRST0lEX1BST0RVQ1RfT1VUIik7Ci0jZW5kaWYKLSAgICBpZiAob3V0ID09IE5VTEwpCi0gICAgICAgIG91dCA9ICIiOwotCi0gICAgY2hhciogc3lzdGVtRGlyID0gKGNoYXIqKSBtYWxsb2Moc3RybGVuKG91dCkgKyBzdHJsZW4oa1N5c3RlbURpcikgKzEpOwotICAgIGNoYXIqIGRhdGFEaXIgPSAoY2hhciopIG1hbGxvYyhzdHJsZW4ob3V0KSArIHN0cmxlbihrRGF0YURpcikgKzEpOwotCi0gICAgc3ByaW50ZihzeXN0ZW1EaXIsICIlcyVzIiwgb3V0LCBrU3lzdGVtRGlyKTsKLSAgICBzcHJpbnRmKGRhdGFEaXIsICIlcyVzIiwgb3V0LCBrRGF0YURpcik7Ci0gICAgc2V0ZW52KCJBTkRST0lEX1JPT1QiLCBzeXN0ZW1EaXIsIDEpOwotICAgIHNldGVudigiQU5EUk9JRF9EQVRBIiwgZGF0YURpciwgMSk7Ci0KLSAgICBjaGFyKiBhc3NldERpciA9IChjaGFyKikgbWFsbG9jKHN0cmxlbihzeXN0ZW1EaXIpICsgc3RybGVuKGtBcHBTdWJkaXIpICsxKTsKLSAgICBzcHJpbnRmKGFzc2V0RGlyLCAiJXMlcyIsIHN5c3RlbURpciwga0FwcFN1YmRpcik7Ci0KLSAgICBMT0dJKCJTdGFydHVwOiBzeXM9JyVzJyBhc3NldD0nJXMnIGRhdGE9JyVzJ1xuIiwKLSAgICAgICAgc3lzdGVtRGlyLCBhc3NldERpciwgZGF0YURpcik7Ci0gICAgZnJlZShzeXN0ZW1EaXIpOwotICAgIGZyZWUoZGF0YURpcik7Ci0KLSNpZmRlZiBIQVZFX0FORFJPSURfT1MKLSAgICAvKiBzZXQgdXAgYSBwcm9jZXNzIGdyb3VwIGZvciBlYXNpZXIga2lsbGluZyBvbiB0aGUgZGV2aWNlICovCi0gICAgc2V0cGdpZCgwLCBnZXRwaWQoKSk7Ci0jZW5kaWYKLQotICAgIC8vIENoYW5nZSB0byBhc3NldCBkaXIuICBUaGlzIGlzIG9ubHkgbmVjZXNzYXJ5IGlmIHdlJ3ZlIGNoYW5nZWQgdG8KLSAgICAvLyBhIGRpZmZlcmVudCBkaXJlY3RvcnksIGJ1dCB0aGVyZSdzIGxpdHRsZSBoYXJtIGluIGRvaW5nIGl0IHJlZ2FyZGxlc3MuCi0gICAgLy8KLSAgICAvLyBFeHBlY3RpbmcgYXNzZXRzIHRvIGxpdmUgaW4gdGhlIGN1cnJlbnQgZGlyIGlzIG5vdCBhIGdyZWF0IGlkZWEsCi0gICAgLy8gYmVjYXVzZSBzb21lIG9mIG91ciBjb2RlIG9yIG9uZSBvZiBvdXIgbGlicmFyaWVzIGNvdWxkIGNoYW5nZSB0aGUKLSAgICAvLyBkaXJlY3Rvcnkgb3V0IGZyb20gdW5kZXIgdXMuICBQcmVzZXJ2ZSB0aGUgYmVoYXZpb3IgZm9yIG5vdy4KLSAgICBpZiAoY2hkaXIoYXNzZXREaXIpICE9IDApIHsKLSAgICAgICAgTE9HVygiV0FSTklORzogY291bGQgbm90IGNoYW5nZSBkaXIgdG8gJyVzJzogJXNcbiIsCi0gICAgICAgICAgICAgYXNzZXREaXIsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgfQotICAgIGZyZWUoYXNzZXREaXIpOwotCi0jaWYgMAotICAgIC8vIEhhY2sgdG8ga2VlcCBsaWJjIGZyb20gYmVhdGluZyB0aGUgZmlsZXN5c3RlbSB0byBkZWF0aC4gIEl0J3MKLSAgICAvLyBoaXR0aW5nIC9ldGMvbG9jYWx0aW1lIGZyZXF1ZW50bHksIAotICAgIC8vCi0gICAgLy8gVGhpcyBzdGF0ZW1lbnQgbG9ja3MgdXMgaW50byBQYWNpZmljIHRpbWUuICBXZSBjb3VsZCBkbyBiZXR0ZXIsCi0gICAgLy8gYnV0IHRoZXJlJ3Mgbm90IG11Y2ggcG9pbnQgdW50aWwgd2UncmUgc3VyZSB0aGF0IHRoZSBsaWJyYXJ5Ci0gICAgLy8gY2FuJ3QgYmUgY2hhbmdlZCB0byBkbyBtb3JlIGFsb25nIHRoZSBsaW5lcyBvZiB3aGF0IHdlIHdhbnQuCi0jaWZuZGVmIFhQX1dJTgotICAgIHNldGVudigiVFoiLCAiUFNUKzhQRFQsTTQuMS4wLzIsTTEwLjUuMC8yIiwgdHJ1ZSk7Ci0jZW5kaWYKLSNlbmRpZgotCi0gICAgLyogdHJhY2sgb3VyIHByb2dyZXNzIHRocm91Z2ggdGhlIGJvb3Qgc2VxdWVuY2UgKi8KLSAgICBjb25zdCBpbnQgTE9HX0JPT1RfUFJPR1JFU1NfU1RBUlQgPSAzMDAwOwotICAgIExPR19FVkVOVF9MT05HKExPR19CT09UX1BST0dSRVNTX1NUQVJULCAKLSAgICAgICAgbnMybXMoc3lzdGVtVGltZShTWVNURU1fVElNRV9NT05PVE9OSUMpKSk7Ci0KLSAgICB2YWxpZGF0ZVRpbWUoKTsKLQotICAgIHByb2MgPSBQcm9jZXNzU3RhdGU6OnNlbGYoKTsKLSAgICAKLSAgICBib290X2luaXQoKTsKLSAgICAKLSAgICAvKiBJZiB3ZSBhcmUgaW4gbXVsdGlwcm9jZXNzIG1vZGUsIGhhdmUgenlnb3RlIHNwYXduIHRoZSBzeXN0ZW0KLSAgICAgKiBzZXJ2ZXIgcHJvY2VzcyBhbmQgY2FsbCBzeXN0ZW1faW5pdCgpLiBJZiB3ZSBhcmUgcnVubmluZyBpbgotICAgICAqIHNpbmdsZSBwcm9jZXNzIG1vZGUganVzdCBjYWxsIHN5c3RlbV9pbml0KCkgZGlyZWN0bHkuCi0gICAgICovCi0gICAgaWYgKHByb2MtPnN1cHBvcnRzUHJvY2Vzc2VzKCkpIHsKLSAgICAgICAgLy8gSWYgc3RkaW8gbG9nZ2luZyBpcyBvbiwgc3lzdGVtX3NlcnZlciBzaG91bGQgbm90IGluaGVyaXQgb3VyIHN0ZGlvCi0gICAgICAgIC8vIFRoZSBkYWx2aWt2bSBpbnN0YW5jZSB3aWxsIGNvcHkgc3RkaW8gdG8gdGhlIGxvZyBvbiBpdHMgb3duCi0gICAgICAgIGNoYXIgcHJvcEJ1ZltQUk9QRVJUWV9WQUxVRV9NQVhdOwotICAgICAgICBib29sIGxvZ1N0ZGlvID0gZmFsc2U7Ci0gICAgICAgIHByb3BlcnR5X2dldCgibG9nLnJlZGlyZWN0LXN0ZGlvIiwgcHJvcEJ1ZiwgIiIpOwotICAgICAgICBsb2dTdGRpbyA9IChzdHJjbXAocHJvcEJ1ZiwgInRydWUiKSA9PSAwKTsKLQotICAgICAgICB6eWdvdGVfcnVuX29uZXNob3QoKGludCkoIWxvZ1N0ZGlvKSwgCi0gICAgICAgICAgICAgICAgc2l6ZW9mKFpZR09URV9BUkdWKSAvIHNpemVvZihaWUdPVEVfQVJHVlswXSksIAotICAgICAgICAgICAgICAgIFpZR09URV9BUkdWKTsKLQotICAgICAgICAvL3N0YXJ0X3Byb2Nlc3MoIi9zeXN0ZW0vYmluL21lZGlhc2VydmVyIik7Ci0KLSAgICB9IGVsc2UgewotI2lmbmRlZiBIQVZFX0FORFJPSURfT1MKLSAgICAgICAgUXVpY2tSdW50aW1lKiBydW50ID0gbmV3IFF1aWNrUnVudGltZSgpOwotICAgICAgICBydW50LT5zdGFydCgiY29tL2FuZHJvaWQvc2VydmVyL1N5c3RlbVNlcnZlciIsIAotICAgICAgICAgICAgICAgICAgICBmYWxzZSAvKiBzcG9udGFuZW91c2x5IGZvcmsgc3lzdGVtIHNlcnZlciBmcm9tIHp5Z290ZSAqLyk7Ci0jZW5kaWYKLSAgICB9Ci0KLSAgICAvL3ByaW50ZigiKysrIHBvc3Qtenlnb3RlXG4iKTsKLQotICAgIGZpbmlzaF9zeXN0ZW1faW5pdChwcm9jKTsKLSAgICBydW4ocHJvYyk7Ci0gICAgCi1iYWlsOgotICAgIGlmIChwcm9jICE9IE5VTEwpIHsKLSAgICAgICAgcHJvYy0+c2V0Q29udGV4dE9iamVjdChOVUxMKTsKLSAgICB9Ci0gICAgCi0gICAgcmV0dXJuIDA7Ci19CmRpZmYgLS1naXQgYS9jbWRzL3N1cmZhY2VmbGluZ2VyL0FuZHJvaWQubWsgYi9jbWRzL3N1cmZhY2VmbGluZ2VyL0FuZHJvaWQubWsKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDM3YzNkOTQuLjAwMDAwMDAKLS0tIGEvY21kcy9zdXJmYWNlZmxpbmdlci9BbmRyb2lkLm1rCisrKyAvZGV2L251bGwKQEAgLTEsMTYgKzAsMCBAQAotTE9DQUxfUEFUSDo9ICQoY2FsbCBteS1kaXIpCi1pbmNsdWRlICQoQ0xFQVJfVkFSUykKLQotTE9DQUxfU1JDX0ZJTEVTOj0gXAotCW1haW5fc3VyZmFjZWZsaW5nZXIuY3BwIAotCi1MT0NBTF9TSEFSRURfTElCUkFSSUVTIDo9IFwKLQlsaWJzdXJmYWNlZmxpbmdlciBcCi0JbGlidXRpbHMKLQotTE9DQUxfQ19JTkNMVURFUyA6PSBcCi0JJChMT0NBTF9QQVRIKS8uLi8uLi9saWJzL3N1cmZhY2VmbGluZ2VyCi0KLUxPQ0FMX01PRFVMRTo9IHN1cmZhY2VmbGluZ2VyCi0KLWluY2x1ZGUgJChCVUlMRF9FWEVDVVRBQkxFKQpkaWZmIC0tZ2l0IGEvY21kcy9zdXJmYWNlZmxpbmdlci9tYWluX3N1cmZhY2VmbGluZ2VyLmNwcCBiL2NtZHMvc3VyZmFjZWZsaW5nZXIvbWFpbl9zdXJmYWNlZmxpbmdlci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDdjODk1NzguLjAwMDAwMDAKLS0tIGEvY21kcy9zdXJmYWNlZmxpbmdlci9tYWluX3N1cmZhY2VmbGluZ2VyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDE4ICswLDAgQEAKLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL1Byb2Nlc3NTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL0lTZXJ2aWNlTWFuYWdlci5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8U3VyZmFjZUZsaW5nZXIuaD4KLQotdXNpbmcgbmFtZXNwYWNlIGFuZHJvaWQ7Ci0KLWludCBtYWluKGludCBhcmdjLCBjaGFyKiogYXJndikKLXsKLSAgICBzcDxQcm9jZXNzU3RhdGU+IHByb2MoUHJvY2Vzc1N0YXRlOjpzZWxmKCkpOwotICAgIHNwPElTZXJ2aWNlTWFuYWdlcj4gc20gPSBkZWZhdWx0U2VydmljZU1hbmFnZXIoKTsKLSAgICBMT0dJKCJTZXJ2aWNlTWFuYWdlcjogJXAiLCBzbS5nZXQoKSk7Ci0gICAgU3VyZmFjZUZsaW5nZXI6Omluc3RhbnRpYXRlKCk7Ci0gICAgUHJvY2Vzc1N0YXRlOjpzZWxmKCktPnN0YXJ0VGhyZWFkUG9vbCgpOwotICAgIElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPmpvaW5UaHJlYWRQb29sKCk7Ci19CmRpZmYgLS1naXQgYS9pbS9qYXZhL2FuZHJvaWQvaW0vQnJhbmRpbmdSZXNvdXJjZUlEcy5qYXZhIGIvaW0vamF2YS9hbmRyb2lkL2ltL0JyYW5kaW5nUmVzb3VyY2VJRHMuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTk2MDcyMi4uMDAwMDAwMAotLS0gYS9pbS9qYXZhL2FuZHJvaWQvaW0vQnJhbmRpbmdSZXNvdXJjZUlEcy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsNTIgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi1wYWNrYWdlIGFuZHJvaWQuaW07Ci0KLS8qKgotICogQGhpZGUKLSAqIERlZmluZXMgdGhlIElEcyBvZiBicmFuZGluZyByZXNvdXJjZXMuCi0gKi8KLXB1YmxpYyBpbnRlcmZhY2UgQnJhbmRpbmdSZXNvdXJjZUlEcyB7Ci0gICAgLyoqCi0gICAgICogVGhlIGxvZ28gaWNvbiBvZiB0aGUgcHJvdmlkZXIgd2hpY2ggaXMgZGlzcGxheWVkIGluIHRoZSBsYW5kaW5nIHBhZ2UuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRFJBV0FCTEVfTE9HTyAgICAgICAgICAgICAgICA9IDEwMDsKLSAgICAvKioKLSAgICAgKiBUaGUgaWNvbiBvZiBvbmxpbmUgcHJlc2VuY2Ugc3RhdHVzLgotICAgICAqLwotICAgIHB1YmxpYyBzdGF0aWMgZmluYWwgaW50IERSQVdBQkxFX1BSRVNFTkNFX09OTElORSAgICAgPSAxMDI7Ci0gICAgLyoqCi0gICAgICogVGhlIGljb24gb2YgYnVzeSBwcmVzZW5jZSBzdGF0dXMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRFJBV0FCTEVfUFJFU0VOQ0VfQlVTWSAgICAgICA9IDEwMzsKLSAgICAvKioKLSAgICAgKiBUaGUgaWNvbiBvZiBhd2F5IHByZXNlbmNlIHN0YXR1cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBEUkFXQUJMRV9QUkVTRU5DRV9BV0FZICAgICAgID0gMTA0OwotICAgIC8qKgotICAgICAqIFRoZSBpY29uIG9mIGludmlzaWJsZSBwcmVzZW5jZSBzdGF0dXMuCi0gICAgICovCi0gICAgcHVibGljIHN0YXRpYyBmaW5hbCBpbnQgRFJBV0FCTEVfUFJFU0VOQ0VfSU5WSVNJQkxFICA9IDEwNTsKLSAgICAvKioKLSAgICAgKiBUaGUgaWNvbiBvZiBvZmZsaW5lIHByZXNlbmNlIHN0YXR1cy4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBEUkFXQUJMRV9QUkVTRU5DRV9PRkZMSU5FICAgID0gMTA2OwotICAgIC8qKgotICAgICAqIFRoZSBsYWJlbCBvZiB0aGUgbWVudSB0byBnbyB0byB0aGUgY29udGFjdCBsaXN0IHNjcmVlbi4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIGludCBTVFJJTkdfTUVOVV9DT05UQUNUX0xJU1QgICAgID0gMTA3OwotCi19CmRpZmYgLS1naXQgYS9pbS9qYXZhL2FuZHJvaWQvaW0vSUltUGx1Z2luLmFpZGwgYi9pbS9qYXZhL2FuZHJvaWQvaW0vSUltUGx1Z2luLmFpZGwKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDIyOWNkMGUuLjAwMDAwMDAKLS0tIGEvaW0vamF2YS9hbmRyb2lkL2ltL0lJbVBsdWdpbi5haWRsCisrKyAvZGV2L251bGwKQEAgLTEsNjkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi1wYWNrYWdlIGFuZHJvaWQuaW07Ci0KLS8qKgotICogQGhpZGUKLSAqLwotaW50ZXJmYWNlIElJbVBsdWdpbiB7Ci0gICAgLyoqCi0gICAgICogTm90aWZ5IHRoZSBwbHVnaW4gdGhlIGZyb250IGRvb3IgYWN0aXZpdHkgaXMgY3JlYXRlZC4gVGhpcyBnaXZlcyB0aGUgcGx1Z2luIGEgY2hhbmNlIHRvCi0gICAgICogc3RhcnQgaXRzIG93biBzZXJ2aWNzLCBldGMuCi0gICAgICovCi0gICAgdm9pZCBvblN0YXJ0KCk7Ci0gICAgCi0gICAgLyoqCi0gICAgICogTm90aWZ5IHRoZSBwbHVnaW4gdGhlIGZyb250IGRvb3IgYWN0aXZpdHkgaXMgc3RvcHBpbmcuCi0gICAgICovCi0gICAgdm9pZCBvblN0b3AoKTsKLQotICAgIC8qKgotICAgICAqIFNpZ24gaW4gdG8gdGhlIHNlcnZpY2UgZm9yIHRoZSBhY2NvdW50IHBhc3NlZCBpbi4KLSAgICAgKgotICAgICAqIEBwYXJhbSBhY2NvdW50IHRoZSBhY2NvdW50IGlkIGZvciB0aGUgYWNjb250IHRvIGJlIHNpZ25lZCBpbnRvLgotICAgICAqLwotICAgIHZvaWQgc2lnbkluKGxvbmcgYWNjb3VudCk7Ci0KLSAgICAvKioKLSAgICAgKiBTaWduIG91dCBvZiB0aGUgc2VydmljZSBmb3IgdGhlIGFjY291bnQgcGFzc2VkIGluLgotICAgICAqCi0gICAgICogQHBhcmFtIGFjY291bnQgdGhlIGFjY291bnQgaWQgZm9yIHRoZSBhY2NvbnQgdG8gYmUgc2lnbmVkIG91dCBvZi4KLSAgICAgKi8KLSAgICB2b2lkIHNpZ25PdXQobG9uZyBhY2NvdW50KTsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdGhlIHBhY2thZ2UgbmFtZSB1c2VkIHRvIGxvYWQgdGhlIHJlc291cmNlcyBmb3IgdGhlIGdpdmVuIHByb3ZpZGVyIG5hbWUuCi0gICAgICoKLSAgICAgKiBAcmV0dXJuIFRoZSBwYWNrYWdlIG5hbWUgdG8gbG9hZCB0aGUgcmVzb3VyY3MgZm9yIHRoZSBnaXZlbiBwcm92aWRlci4KLSAgICAgKi8KLSAgICBTdHJpbmcgZ2V0UmVzb3VyY2VQYWNrYWdlTmFtZUZvclByb3ZpZGVyKFN0cmluZyBwcm92aWRlck5hbWUpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJucyBhIG1hcCBvZiBicmFuZGluZyByZXNvdXJjZXMgZm9yIHRoZSBnaXZlbiBwcm92aWRlci4gVGhlIGtleXMgYXJlIGRlZmluZWQKLSAgICAgKiBpbiB7QGxpbmsgYW5kcm9pZC5pbS5CcmFuZGluZ1Jlc291cmNlSURzfS4gVGhlIHZhbHVlcyBhcmUgdGhlIHJlc291cmNlIGlkZW50aWZpZXJzIGdlbmVyYXRlZAotICAgICAqIGJ5IHRoZSBhYXB0IHRvb2wuCi0gICAgICoKLSAgICAgKiBAcmV0dXJuIFRoZSBtYXAgb2YgYnJhbmRpbmcgcmVzb3VyY2VzIGZvciB0aGUgZ2l2ZW4gcHJvdmlkZXIuCi0gICAgICovCi0gICAgTWFwIGdldFJlc291cmNlTWFwRm9yUHJvdmlkZXIoU3RyaW5nIHByb3ZpZGVyTmFtZSk7Ci0KLSAgICAvKgotICAgICAqIFJldHVybnMgYSBsaXN0IG9mIHN1cHBvcnRlZCBJTSBwcm92aWRlcnMuCi0gICAgICoKLSAgICAgKiBAcmV0dXJuIGEgTGlzdCBvZiBzdXBwb3J0ZWQgcHJvdmlkZXJzLgotICAgICAqLwotICAgIExpc3QgZ2V0U3VwcG9ydGVkUHJvdmlkZXJzKCk7Ci19CmRpZmYgLS1naXQgYS9pbS9qYXZhL2FuZHJvaWQvaW0vSW1QbHVnaW5Db25zdHMuamF2YSBiL2ltL2phdmEvYW5kcm9pZC9pbS9JbVBsdWdpbkNvbnN0cy5qYXZhCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0MTY0OTNmLi4wMDAwMDAwCi0tLSBhL2ltL2phdmEvYW5kcm9pZC9pbS9JbVBsdWdpbkNvbnN0cy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMjcgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLXBhY2thZ2UgYW5kcm9pZC5pbTsKLQotLyoqCi0gKiBAaGlkZQotICovCi1wdWJsaWMgY2xhc3MgSW1QbHVnaW5Db25zdHMgewotICAgIC8qKgotICAgICAqIFRoZSBpbnRlbnQgYWN0aW9uIG5hbWUgZm9yIHRoZSBwbHVnaW4gc2VydmljZS4KLSAgICAgKi8KLSAgICBwdWJsaWMgc3RhdGljIGZpbmFsIFN0cmluZyBQTFVHSU5fQUNUSU9OX05BTUUgPSAiYW5kcm9pZC5pbS5wbHVnaW4iOwotfQpcIE5vIG5ld2xpbmUgYXQgZW5kIG9mIGZpbGUKZGlmZiAtLWdpdCBhL2luY2x1ZGUvcGltL0V2ZW50UmVjdXJyZW5jZS5oIGIvaW5jbHVkZS9waW0vRXZlbnRSZWN1cnJlbmNlLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDFjZWRhNDEuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS9waW0vRXZlbnRSZWN1cnJlbmNlLmgKKysrIC9kZXYvbnVsbApAQCAtMSw4MiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA2IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLSNpZm5kZWYgX1BJTV9FVkVOVF9SRUNVUlJFTkNFX0gKLSNkZWZpbmUgX1BJTV9FVkVOVF9SRUNVUlJFTkNFX0gKLQotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzE2Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotc3RydWN0IEV2ZW50UmVjdXJyZW5jZQotewotcHVibGljOgotICAgICAgICAgICAgICAgIEV2ZW50UmVjdXJyZW5jZSgpOwotICAgICAgICAgICAgICAgIH5FdmVudFJlY3VycmVuY2UoKTsKLSAgICAKLSAgICBzdGF0dXNfdCAgICBwYXJzZShjb25zdCBTdHJpbmcxNiYpOwotCi0KLSAgICBlbnVtIGZyZXFfdCB7Ci0gICAgICAgIFNFQ09ORExZID0gMSwKLSAgICAgICAgTUlOVVRFTFkgPSAyLAotICAgICAgICBIT1VSTFkgPSAzLAotICAgICAgICBEQUlMWSA9IDQsCi0gICAgICAgIFdFRUtMWSA9IDUsCi0gICAgICAgIE1PTlRITFkgPSA2LAotICAgICAgICBZRUFSTFkgPSA3Ci0gICAgfTsKLQotICAgIGVudW0gewotICAgICAgICBTVSA9IDB4MDAwMTAwMDAsCi0gICAgICAgIE1PID0gMHgwMDAyMDAwMCwKLSAgICAgICAgVFUgPSAweDAwMDQwMDAwLAotICAgICAgICBXRSA9IDB4MDAwODAwMDAsCi0gICAgICAgIFRIID0gMHgwMDEwMDAwMCwKLSAgICAgICAgRlIgPSAweDAwMjAwMDAwLAotICAgICAgICBTQSA9IDB4MDA0MDAwMDAKLSAgICB9OwotICAgIAotICAgIGZyZXFfdCAgICBmcmVxOwotICAgIFN0cmluZzE2ICB1bnRpbDsKLSAgICBpbnQgICAgICAgY291bnQ7Ci0gICAgaW50ICAgICAgIGludGVydmFsOwotICAgIGludCogICAgICBieXNlY29uZDsKLSAgICBpbnQgICAgICAgYnlzZWNvbmRDb3VudDsKLSAgICBpbnQqICAgICAgYnltaW51dGU7Ci0gICAgaW50ICAgICAgIGJ5bWludXRlQ291bnQ7Ci0gICAgaW50KiAgICAgIGJ5aG91cjsKLSAgICBpbnQgICAgICAgYnlob3VyQ291bnQ7Ci0gICAgaW50KiAgICAgIGJ5ZGF5OwotICAgIGludCogICAgICBieWRheU51bTsKLSAgICBpbnQgICAgICAgYnlkYXlDb3VudDsgICAKLSAgICBpbnQqICAgICAgYnltb250aGRheTsKLSAgICBpbnQgICAgICAgYnltb250aGRheUNvdW50OwotICAgIGludCogICAgICBieXllYXJkYXk7Ci0gICAgaW50ICAgICAgIGJ5eWVhcmRheUNvdW50OwotICAgIGludCogICAgICBieXdlZWtubzsKLSAgICBpbnQgICAgICAgYnl3ZWVrbm9Db3VudDsKLSAgICBpbnQqICAgICAgYnltb250aDsKLSAgICBpbnQgICAgICAgYnltb250aENvdW50OwotICAgIGludCogICAgICBieXNldHBvczsKLSAgICBpbnQgICAgICAgYnlzZXRwb3NDb3VudDsKLSAgICBpbnQgICAgICAgd2tzdDsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBfUElNX0VWRU5UX1JFQ1VSUkVOQ0VfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS9wcml2YXRlL29wZW5nbGVzL2dsX2NvbnRleHQuaCBiL2luY2x1ZGUvcHJpdmF0ZS9vcGVuZ2xlcy9nbF9jb250ZXh0LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBjN2FkNDYuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS9wcml2YXRlL29wZW5nbGVzL2dsX2NvbnRleHQuaAorKysgL2Rldi9udWxsCkBAIC0xLDYzMiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA2IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX09QRU5HTEVTX0NPTlRFWFRfSAotI2RlZmluZSBBTkRST0lEX09QRU5HTEVTX0NPTlRFWFRfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3RkZGVmLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8cHRocmVhZC5oPgotI2lmZGVmIEhBVkVfQU5EUk9JRF9PUwotI2luY2x1ZGUgPGJpb25pY190bHMuaD4KLSNlbmRpZgotCi0jaW5jbHVkZSA8cHJpdmF0ZS9waXhlbGZsaW5nZXIvZ2dsX2NvbnRleHQuaD4KLQotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLSNpbmNsdWRlIDxHTEVTL2dsZXh0Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY29uc3QgdW5zaWduZWQgaW50IE9HTEVTX05VTV9DT01QUkVTU0VEX1RFWFRVUkVfRk9STUFUUyA9IDEwOwotCi1jbGFzcyBFR0xUZXh0dXJlT2JqZWN0OwotY2xhc3MgRUdMU3VyZmFjZU1hbmFnZXI7Ci1jbGFzcyBFR0xCdWZmZXJPYmplY3RNYW5hZ2VyOwotCi1uYW1lc3BhY2UgZ2wgewotIAotc3RydWN0IG9nbGVzX2NvbnRleHRfdDsKLXN0cnVjdCBtYXRyaXh4X3Q7Ci1zdHJ1Y3QgdHJhbnNmb3JtX3Q7Ci1zdHJ1Y3QgYnVmZmVyX3Q7Ci0KLW9nbGVzX2NvbnRleHRfdCogZ2V0R2xDb250ZXh0KCk7Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+Ci1zdGF0aWMgaW5saW5lIHZvaWQgc3dhcChUJiBhLCBUJiBiKSB7Ci0gICAgVCB0KGEpOyBhID0gYjsgYiA9IHQ7Ci19Ci10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotaW5saW5lIFQgbWF4KFQgYSwgVCBiKSB7Ci0gICAgcmV0dXJuIGE8YiA/IGIgOiBhOwotfQotdGVtcGxhdGU8dHlwZW5hbWUgVD4KLWlubGluZSBUIG1heChUIGEsIFQgYiwgVCBjKSB7Ci0gICAgcmV0dXJuIG1heChhLCBtYXgoYiwgYykpOwotfQotdGVtcGxhdGU8dHlwZW5hbWUgVD4KLWlubGluZSBUIG1pbihUIGEsIFQgYikgewotICAgIHJldHVybiBhPGIgPyBhIDogYjsKLX0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+Ci1pbmxpbmUgVCBtaW4oVCBhLCBUIGIsIFQgYykgewotICAgIHJldHVybiBtaW4oYSwgbWluKGIsIGMpKTsKLX0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+Ci1pbmxpbmUgVCBtaW4oVCBhLCBUIGIsIFQgYywgVCBkKSB7Ci0gICAgcmV0dXJuIG1pbihtaW4oYSxiKSwgbWluKGMsZCkpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyB2ZXJ0aWNlcwotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3QgdmVjM190IHsKLSAgICB1bmlvbiB7Ci0gICAgICAgIHN0cnVjdCB7IEdMZml4ZWQgeCwgeSwgejsgfTsKLSAgICAgICAgc3RydWN0IHsgR0xmaXhlZCByLCBnLCBiOyB9OwotICAgICAgICBzdHJ1Y3QgeyBHTGZpeGVkIFMsIFQsIFI7IH07Ci0gICAgICAgIEdMZml4ZWQgdlszXTsKLSAgICB9OwotfTsKLQotc3RydWN0IHZlYzRfdCB7Ci0gICAgdW5pb24gewotICAgICAgICBzdHJ1Y3QgeyBHTGZpeGVkIHgsIHksIHosIHc7IH07Ci0gICAgICAgIHN0cnVjdCB7IEdMZml4ZWQgciwgZywgYiwgYTsgfTsKLSAgICAgICAgc3RydWN0IHsgR0xmaXhlZCBTLCBULCBSLCBROyB9OwotICAgICAgICBHTGZpeGVkIHZbNF07Ci0gICAgfTsKLX07Ci0KLXN0cnVjdCB2ZXJ0ZXhfdCB7Ci0gICAgZW51bSB7Ci0gICAgICAgIC8vIHRoZXNlIGNvbnN0YW50IG1hdHRlciBmb3Igb3VyIGNsaXBwaW5nIAotICAgICAgICBDTElQX0wgICAgICAgICAgPSAweDAwMDEsICAgLy8gY2xpcHBpbmcgZmxhZ3MKLSAgICAgICAgQ0xJUF9SICAgICAgICAgID0gMHgwMDAyLAotICAgICAgICBDTElQX0IgICAgICAgICAgPSAweDAwMDQsCi0gICAgICAgIENMSVBfVCAgICAgICAgICA9IDB4MDAwOCwKLSAgICAgICAgQ0xJUF9OICAgICAgICAgID0gMHgwMDEwLAotICAgICAgICBDTElQX0YgICAgICAgICAgPSAweDAwMjAsCi0KLSAgICAgICAgRVlFICAgICAgICAgICAgID0gMHgwMDQwLAotICAgICAgICBSRVNFUlZFRCAgICAgICAgPSAweDAwODAsCi0gICAgICAgIAotICAgICAgICBVU0VSX0NMSVBfMCAgICAgPSAweDAxMDAsICAgLy8gdXNlciBjbGlwcGluZyBmbGFncwotICAgICAgICBVU0VSX0NMSVBfMSAgICAgPSAweDAyMDAsCi0gICAgICAgIFVTRVJfQ0xJUF8yICAgICA9IDB4MDQwMCwKLSAgICAgICAgVVNFUl9DTElQXzMgICAgID0gMHgwODAwLAotICAgICAgICBVU0VSX0NMSVBfNCAgICAgPSAweDEwMDAsCi0gICAgICAgIFVTRVJfQ0xJUF81ICAgICA9IDB4MjAwMCwKLQotICAgICAgICBMSVQgICAgICAgICAgICAgPSAweDQwMDAsICAgLy8gbGlnaHRpbmcgaGFzIGJlZW4gYXBwbGllZAotICAgICAgICBUVCAgICAgICAgICAgICAgPSAweDgwMDAsICAgLy8gdGV4dHVyZSBjb29yZHMgdHJhbnNmb3JtZWQKLQotICAgICAgICBGUlVTVFVNX0NMSVBfQUxMPSAweDAwM0YsCi0gICAgICAgIFVTRVJfQ0xJUF9BTEwgICA9IDB4M0YwMCwKLSAgICAgICAgQ0xJUF9BTEwgICAgICAgID0gMHgzRjNGLAotICAgIH07Ci0gICAgCi0gICAgLy8gdGhlIGZpZWxkcyBiZWxvdyBhcmUgYXJyYW5nZWQgdG8gbWluaW1pemUgZC1jYWNoZSB1c2FnZQotICAgIC8vIHdlIGdyb3VwIHRvZ2V0aGVyLCBieSBjYWNoZS1saW5lLCB0aGUgZmllbGRzIG1vc3QgbGlrZWx5IHRvIGJlIHVzZWQKLQotICAgIHVuaW9uIHsKLSAgICB2ZWM0X3QgICAgICAgICAgb2JqOwotICAgIHZlYzRfdCAgICAgICAgICBleWU7Ci0gICAgfTsKLSAgICB2ZWM0X3QgICAgICAgICAgY2xpcDsKLSAgICAKLSAgICB1aW50MzJfdCAgICAgICAgZmxhZ3M7Ci0gICAgc2l6ZV90ICAgICAgICAgIGluZGV4OyAgLy8gY2FjaGUgdGFnLCBhbmQgdmVydGV4IGluZGV4Ci0gICAgR0xmaXhlZCAgICAgICAgIGZvZzsKLSAgICB1aW50OF90ICAgICAgICAgbG9ja2VkOwotICAgIHVpbnQ4X3QgICAgICAgICBtcnU7Ci0gICAgdWludDhfdCAgICAgICAgIHJlc2VydmVkWzJdOwotICAgIHZlYzRfdCAgICAgICAgICB3aW5kb3c7Ci0KLSAgICB2ZWM0X3QgICAgICAgICAgY29sb3I7Ci0gICAgdmVjNF90ICAgICAgICAgIHRleHR1cmVbR0dMX1RFWFRVUkVfVU5JVF9DT1VOVF07Ci0gICAgdWludDMyX3QgICAgICAgIHJlc2VydmVkMVs0XTsKLSAgICAKLSAgICBpbmxpbmUgdm9pZCBjbGVhcigpIHsKLSAgICAgICAgZmxhZ3MgPSBpbmRleCA9IGxvY2tlZCA9IG1ydSA9IDA7Ci0gICAgfQotfTsKLQotc3RydWN0IHBvaW50X3NpemVfdCB7Ci0gICAgR0dMY29vcmQgICAgc2l6ZTsKLSAgICBHTGJvb2xlYW4gICBzbW9vdGg7Ci19OwotCi1zdHJ1Y3QgbGluZV93aWR0aF90IHsKLSAgICBHR0xjb29yZCAgICB3aWR0aDsKLSAgICBHTGJvb2xlYW4gICBzbW9vdGg7Ci19OwotCi1zdHJ1Y3QgcG9seWdvbl9vZmZzZXRfdCB7Ci0gICAgR0xmaXhlZCAgICAgZmFjdG9yOwotICAgIEdMZml4ZWQgICAgIHVuaXRzOwotICAgIEdMYm9vbGVhbiAgIGVuYWJsZTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIGFycmF5cwotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3QgYXJyYXlfdCB7Ci0gICAgdHlwZWRlZiB2b2lkICgqZmV0Y2hlcl90KShvZ2xlc19jb250ZXh0X3QqLCBHTGZpeGVkKiwgY29uc3QgR0x2b2lkKik7Ci0gICAgZmV0Y2hlcl90ICAgICAgIGZldGNoOwotICAgIEdMdm9pZCBjb25zdCogICBwaHlzaWNhbF9wb2ludGVyOwotICAgIEdMaW50ICAgICAgICAgICBzaXplOwotICAgIEdMc2l6ZWkgICAgICAgICBzdHJpZGU7Ci0gICAgR0x2b2lkIGNvbnN0KiAgIHBvaW50ZXI7Ci0gICAgYnVmZmVyX3QgY29uc3QqIGJvOwotICAgIHVpbnQxNl90ICAgICAgICB0eXBlOwotICAgIEdMYm9vbGVhbiAgICAgICBlbmFibGU7Ci0gICAgR0xib29sZWFuICAgICAgIHBhZDsKLSAgICBHTHNpemVpICAgICAgICAgYm91bmRzOwotICAgIHZvaWQgaW5pdChHTGludCwgR0xlbnVtLCBHTHNpemVpLCBjb25zdCBHTHZvaWQgKiwgY29uc3QgYnVmZmVyX3QqLCBHTHNpemVpKTsKLSAgICBpbmxpbmUgdm9pZCByZXNvbHZlKCk7Ci0gICAgaW5saW5lIGNvbnN0IEdMdWJ5dGUqIGVsZW1lbnQoR0xpbnQgaSkgY29uc3QgewotICAgICAgICByZXR1cm4gKGNvbnN0IEdMdWJ5dGUqKXBoeXNpY2FsX3BvaW50ZXIgKyBpICogc3RyaWRlOwotICAgIH0KLX07Ci0KLXN0cnVjdCBhcnJheV9tYWNoaW5lX3QgewotICAgIGFycmF5X3QgICAgICAgICB2ZXJ0ZXg7Ci0gICAgYXJyYXlfdCAgICAgICAgIG5vcm1hbDsKLSAgICBhcnJheV90ICAgICAgICAgY29sb3I7Ci0gICAgYXJyYXlfdCAgICAgICAgIHRleHR1cmVbR0dMX1RFWFRVUkVfVU5JVF9DT1VOVF07Ci0gICAgdWludDhfdCAgICAgICAgIGFjdGl2ZVRleHR1cmU7Ci0gICAgdWludDhfdCAgICAgICAgIHRtdTsKLSAgICB1aW50MTZfdCAgICAgICAgY3VsbDsKLSAgICB1aW50MzJfdCAgICAgICAgZmxhZ3M7Ci0gICAgR0xlbnVtICAgICAgICAgIGluZGljZXNUeXBlOwotICAgIGJ1ZmZlcl90IGNvbnN0KiBhcnJheV9idWZmZXI7Ci0gICAgYnVmZmVyX3QgY29uc3QqIGVsZW1lbnRfYXJyYXlfYnVmZmVyOwotICAgIAotICAgIHZvaWQgKCpjb21waWxlRWxlbWVudHMpKG9nbGVzX2NvbnRleHRfdCosIHZlcnRleF90KiwgR0xpbnQsIEdMc2l6ZWkpOwotICAgIHZvaWQgKCpjb21waWxlRWxlbWVudCkob2dsZXNfY29udGV4dF90KiwgdmVydGV4X3QqLCBHTGludCk7Ci0KLSAgICB2b2lkICgqbXZwX3RyYW5zZm9ybSkodHJhbnNmb3JtX3QgY29uc3QqLCB2ZWM0X3QqLCB2ZWM0X3QgY29uc3QqKTsKLSAgICB2b2lkICgqbXZfdHJhbnNmb3JtKSh0cmFuc2Zvcm1fdCBjb25zdCosIHZlYzRfdCosIHZlYzRfdCBjb25zdCopOwotICAgIHZvaWQgKCp0ZXhfdHJhbnNmb3JtWzJdKSh0cmFuc2Zvcm1fdCBjb25zdCosIHZlYzRfdCosIHZlYzRfdCBjb25zdCopOwotICAgIHZvaWQgKCpwZXJzcGVjdGl2ZSkob2dsZXNfY29udGV4dF90KmMsIHZlcnRleF90KiB2KTsKLSAgICB2b2lkICgqY2xpcFZlcnRleCkob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogbnYsCi0gICAgICAgICAgICBHR0xmaXhlZCB0LCBjb25zdCB2ZXJ0ZXhfdCogcywgY29uc3QgdmVydGV4X3QqIHApOwotICAgIHZvaWQgKCpjbGlwRXllKShvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiBudiwKLSAgICAgICAgICAgIEdHTGZpeGVkIHQsIGNvbnN0IHZlcnRleF90KiBzLCBjb25zdCB2ZXJ0ZXhfdCogcCk7Ci19OwotCi1zdHJ1Y3QgdmVydGV4X2NhY2hlX3QgewotICAgIGVudW0gewotICAgICAgICAvLyBtdXN0IGJlIGF0IGxlYXN0IDQKLSAgICAgICAgLy8gMyB2ZXJ0aWNlIGZvciB0cmlhbmdsZXMKLSAgICAgICAgLy8gb3IgMiArIDIgZm9yIGluZGV4ZWQgdHJpYW5nbGVzIHcvIGNhY2hlIGNvbnRlbnRpb24KLSAgICAgICAgVkVSVEVYX0JVRkZFUl9TSVpFICA9IDgsCi0gICAgICAgIC8vIG11c3QgYmUgYSBwb3dlciBvZiB0d28gYW5kIGF0IGxlYXN0IDMKLSAgICAgICAgVkVSVEVYX0NBQ0hFX1NJWkUgICA9IDY0LCAgIC8vIDggS0IKLQotICAgICAgICBJTkRFWF9CSVRTICAgICAgPSAxNiwKLSAgICAgICAgSU5ERVhfTUFTSyAgICAgID0gKCgxTFU8PElOREVYX0JJVFMpLTEpLAotICAgICAgICBJTkRFWF9TRVEgICAgICAgPSAxTFU8PElOREVYX0JJVFMsCi0gICAgfTsKLSAgICB2ZXJ0ZXhfdCogICAgICAgdkJ1ZmZlcjsKLSAgICB2ZXJ0ZXhfdCogICAgICAgdkNhY2hlOwotICAgIHVpbnQzMl90ICAgICAgICBzZXF1ZW5jZTsKLSAgICB2b2lkKiAgICAgICAgICAgYmFzZTsKLSAgICB1aW50MzJfdCAgICAgICAgdG90YWw7Ci0gICAgdWludDMyX3QgICAgICAgIG1pc3NlczsKLSAgICBpbnQ2NF90ICAgICAgICAgc3RhcnRUaW1lOwotICAgIHZvaWQgaW5pdCgpOwotICAgIHZvaWQgdW5pbml0KCk7Ci0gICAgdm9pZCBjbGVhcigpOwotICAgIHZvaWQgZHVtcF9zdGF0cyhHTGVudW0gbW9kZSk7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBmb2cKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RydWN0IGZvZ190IHsKLSAgICBHTGZpeGVkICAgICBkZW5zaXR5OwotICAgIEdMZml4ZWQgICAgIHN0YXJ0OwotICAgIEdMZml4ZWQgICAgIGVuZDsKLSAgICBHTGZpeGVkICAgICBpbnZFbmRNaW51c1N0YXJ0OwotICAgIEdMZW51bSAgICAgIG1vZGU7Ci0gICAgR0xmaXhlZCAgICAgKCpmb2cpKG9nbGVzX2NvbnRleHRfdCogYywgR0xmaXhlZCB6KTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIHVzZXIgY2xpcCBwbGFuZXMKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY29uc3QgdW5zaWduZWQgaW50IE9HTEVTX01BWF9DTElQX1BMQU5FUyA9IDY7Ci0KLXN0cnVjdCBjbGlwX3BsYW5lX3QgewotICAgIHZlYzRfdCAgICAgIGVxdWF0aW9uOwotfTsKLQotc3RydWN0IHVzZXJfY2xpcF9wbGFuZXNfdCB7Ci0gICAgY2xpcF9wbGFuZV90ICAgIHBsYW5lW09HTEVTX01BWF9DTElQX1BMQU5FU107Ci0gICAgdWludDMyX3QgICAgICAgIGVuYWJsZTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIGxpZ2h0aW5nCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNvbnN0IHVuc2lnbmVkIGludCBPR0xFU19NQVhfTElHSFRTID0gODsKLQotc3RydWN0IGxpZ2h0X3QgewotICAgIHZlYzRfdCAgICAgIGFtYmllbnQ7Ci0gICAgdmVjNF90ICAgICAgZGlmZnVzZTsKLSAgICB2ZWM0X3QgICAgICBzcGVjdWxhcjsKLSAgICB2ZWM0X3QgICAgICBpbXBsaWNpdEFtYmllbnQ7Ci0gICAgdmVjNF90ICAgICAgaW1wbGljaXREaWZmdXNlOwotICAgIHZlYzRfdCAgICAgIGltcGxpY2l0U3BlY3VsYXI7Ci0gICAgdmVjNF90ICAgICAgcG9zaXRpb247ICAgICAgIC8vIHBvc2l0aW9uIGluIGV5ZSBzcGFjZQotICAgIHZlYzRfdCAgICAgIG9ialBvc2l0aW9uOwotICAgIHZlYzRfdCAgICAgIG5vcm1hbGl6ZWRPYmpQb3NpdGlvbjsKLSAgICB2ZWM0X3QgICAgICBzcG90RGlyOwotICAgIHZlYzRfdCAgICAgIG5vcm1hbGl6ZWRTcG90RGlyOwotICAgIEdMZml4ZWQgICAgIHNwb3RFeHA7Ci0gICAgR0xmaXhlZCAgICAgc3BvdEN1dG9mZjsKLSAgICBHTGZpeGVkICAgICBzcG90Q3V0b2ZmQ29zaW5lOwotICAgIEdMZml4ZWQgICAgIGF0dGVudWF0aW9uWzNdOwotICAgIEdMZml4ZWQgICAgIHJDb25zdEF0dGVudWF0aW9uOwotICAgIEdMYm9vbGVhbiAgIGVuYWJsZTsKLX07Ci0KLXN0cnVjdCBtYXRlcmlhbF90IHsKLSAgICB2ZWM0X3QgICAgICBhbWJpZW50OwotICAgIHZlYzRfdCAgICAgIGRpZmZ1c2U7Ci0gICAgdmVjNF90ICAgICAgc3BlY3VsYXI7Ci0gICAgdmVjNF90ICAgICAgZW1pc3Npb247Ci0gICAgR0xmaXhlZCAgICAgc2hpbmluZXNzOwotfTsKLQotc3RydWN0IGxpZ2h0X21vZGVsX3QgewotICAgIHZlYzRfdCAgICAgIGFtYmllbnQ7Ci0gICAgR0xib29sZWFuICAgdHdvU2lkZTsKLX07Ci0KLXN0cnVjdCBjb2xvcl9tYXRlcmlhbF90IHsKLSAgICBHTGVudW0gICAgICBmYWNlOwotICAgIEdMZW51bSAgICAgIG1vZGU7Ci0gICAgR0xib29sZWFuICAgZW5hYmxlOwotfTsKLQotc3RydWN0IGxpZ2h0aW5nX3QgewotICAgIGxpZ2h0X3QgICAgICAgICAgICAgbGlnaHRzW09HTEVTX01BWF9MSUdIVFNdOwotICAgIG1hdGVyaWFsX3QgICAgICAgICAgZnJvbnQ7Ci0gICAgbGlnaHRfbW9kZWxfdCAgICAgICBsaWdodE1vZGVsOwotICAgIGNvbG9yX21hdGVyaWFsX3QgICAgY29sb3JNYXRlcmlhbDsKLSAgICB1aW50MzJfdCAgICAgICAgICAgIGVuYWJsZWRMaWdodHM7Ci0gICAgR0xib29sZWFuICAgICAgICAgICBlbmFibGU7Ci0gICAgdmVjNF90ICAgICAgICAgICAgICBpbXBsaWNpdFNjZW5lRW1pc3Npb25BbmRBbWJpZW50OwotICAgIEdMZW51bSAgICAgICAgICAgICAgc2hhZGVNb2RlbDsKLSAgICB0eXBlZGVmIHZvaWQgKCpsaWdodF9mY3RfdCkob2dsZXNfY29udGV4dF90KiwgdmVydGV4X3QqKTsKLSAgICB2b2lkICgqbGlnaHRWZXJ0ZXgpKG9nbGVzX2NvbnRleHRfdCogYywgdmVydGV4X3QqIHYpOwotICAgIHZvaWQgKCpsaWdodFRyaWFuZ2xlKShvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgICAgICB2ZXJ0ZXhfdCogdjAsIHZlcnRleF90KiB2MSwgdmVydGV4X3QqIHYyKTsKLX07Ci0KLXN0cnVjdCBjdWxsaW5nX3QgewotICAgIEdMZW51bSAgICAgIGN1bGxGYWNlOwotICAgIEdMZW51bSAgICAgIGZyb250RmFjZTsKLSAgICBHTGJvb2xlYW4gICBlbmFibGU7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyB0ZXh0dXJlcwotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3QgdGV4dHVyZV91bml0X3QgewotICAgIEdMdWludCAgICAgICAgICAgICAgbmFtZTsKLSAgICBFR0xUZXh0dXJlT2JqZWN0KiAgIHRleHR1cmU7Ci0gICAgdWludDhfdCAgICAgICAgICAgICBkaXJ0eTsKLX07Ci0KLXN0cnVjdCB0ZXh0dXJlX3N0YXRlX3QKLXsKLSAgICB0ZXh0dXJlX3VuaXRfdCAgICAgIHRtdVtHR0xfVEVYVFVSRV9VTklUX0NPVU5UXTsKLSAgICBpbnQgICAgICAgICAgICAgICAgIGFjdGl2ZTsgICAgIC8vIGFjdGl2ZSB0bXUKLSAgICBFR0xUZXh0dXJlT2JqZWN0KiAgIGRlZmF1bHRUZXh0dXJlOwotICAgIEdHTENvbnRleHQqICAgICAgICAgZ2dsOwotICAgIHVpbnQ4X3QgICAgICAgICAgICAgcGFja0FsaWdubWVudDsKLSAgICB1aW50OF90ICAgICAgICAgICAgIHVucGFja0FsaWdubWVudDsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIHRyYW5zZm9ybWF0aW9uIGFuZCBtYXRyaWNlcwotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3QgbWF0cml4Zl90OwotCi1zdHJ1Y3QgbWF0cml4eF90IHsKLSAgICBHTGZpeGVkIG1bMTZdOwotICAgIHZvaWQgbG9hZChjb25zdCBtYXRyaXhmX3QmIHJocyk7Ci19OwotCi1zdHJ1Y3QgbWF0cml4X3N0YWNrX3Q7Ci0KLQotc3RydWN0IG1hdHJpeGZfdCB7Ci0gICAgdm9pZCBsb2FkSWRlbnRpdHkoKTsKLSAgICB2b2lkIGxvYWQoY29uc3QgbWF0cml4Zl90JiByaHMpOwotCi0gICAgaW5saW5lIEdMZmxvYXQqIGVkaXRFbGVtZW50cygpIHsgcmV0dXJuIG07IH0KLSAgICBpbmxpbmUgR0xmbG9hdCBjb25zdCogZWxlbWVudHMoKSBjb25zdCB7IHJldHVybiBtOyB9Ci0KLSAgICB2b2lkIHNldChjb25zdCBHTGZpeGVkKiByaHMpOwotICAgIHZvaWQgc2V0KGNvbnN0IEdMZmxvYXQqIHJocyk7Ci0KLSAgICBzdGF0aWMgdm9pZCBtdWx0aXBseShtYXRyaXhmX3QmIHIsCi0gICAgICAgICAgICBjb25zdCBtYXRyaXhmX3QmIGxocywgY29uc3QgbWF0cml4Zl90JiByaHMpOwotCi0gICAgdm9pZCBkdW1wKGNvbnN0IGNoYXIqIHdoYXQpOwotCi1wcml2YXRlOgotICAgIGZyaWVuZCBzdHJ1Y3QgbWF0cml4X3N0YWNrX3Q7Ci0gICAgR0xmbG9hdCAgICAgbVsxNl07Ci0gICAgdm9pZCBsb2FkKGNvbnN0IEdMZml4ZWQqIHJocyk7Ci0gICAgdm9pZCBsb2FkKGNvbnN0IEdMZmxvYXQqIHJocyk7Ci0gICAgdm9pZCBtdWx0aXBseShjb25zdCBtYXRyaXhmX3QmIHJocyk7Ci0gICAgdm9pZCB0cmFuc2xhdGUoR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeik7Ci0gICAgdm9pZCBzY2FsZShHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KTsKLSAgICB2b2lkIHJvdGF0ZShHTGZsb2F0IGEsIEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopOwotfTsKLQotZW51bSB7Ci0gICAgT1BfSURFTlRJVFkgICAgICAgICA9IDB4MDAsCi0gICAgT1BfVFJBTlNMQVRFICAgICAgICA9IDB4MDEsCi0gICAgT1BfVU5JRk9STV9TQ0FMRSAgICA9IDB4MDIsCi0gICAgT1BfU0NBTEUgICAgICAgICAgICA9IDB4MDUsCi0gICAgT1BfUk9UQVRFICAgICAgICAgICA9IDB4MDgsCi0gICAgT1BfU0tFVyAgICAgICAgICAgICA9IDB4MTAsCi0gICAgT1BfQUxMICAgICAgICAgICAgICA9IDB4MUYKLX07Ci0KLXN0cnVjdCB0cmFuc2Zvcm1fdCB7Ci0gICAgZW51bSB7Ci0gICAgICAgIEZMQUdTXzJEX1BST0pFQ1RJT04gPSAweDEKLSAgICB9OwotICAgIG1hdHJpeHhfdCAgICAgICBtYXRyaXg7Ci0gICAgdWludDMyX3QgICAgICAgIGZsYWdzOwotICAgIHVpbnQzMl90ICAgICAgICBvcHM7Ci0gICAgCi0gICAgdW5pb24gewotICAgICAgICBzdHJ1Y3QgewotICAgICAgICAgICAgdm9pZCAoKnBvaW50MikodHJhbnNmb3JtX3QgY29uc3QqIHQsIHZlYzRfdCosIHZlYzRfdCBjb25zdCopOwotICAgICAgICAgICAgdm9pZCAoKnBvaW50MykodHJhbnNmb3JtX3QgY29uc3QqIHQsIHZlYzRfdCosIHZlYzRfdCBjb25zdCopOwotICAgICAgICAgICAgdm9pZCAoKnBvaW50NCkodHJhbnNmb3JtX3QgY29uc3QqIHQsIHZlYzRfdCosIHZlYzRfdCBjb25zdCopOwotICAgICAgICB9OwotICAgICAgICB2b2lkICgqcG9pbnR2WzNdKSh0cmFuc2Zvcm1fdCBjb25zdCogdCwgdmVjNF90KiwgdmVjNF90IGNvbnN0Kik7Ci0gICAgfTsKLQotICAgIHZvaWQgbG9hZElkZW50aXR5KCk7Ci0gICAgdm9pZCBwaWNrZXIoKTsKLSAgICB2b2lkIGR1bXAoY29uc3QgY2hhciogd2hhdCk7Ci19OwotCi1zdHJ1Y3QgbXZ1aV90cmFuc2Zvcm1fdCA6IHB1YmxpYyB0cmFuc2Zvcm1fdAotewotICAgIHZvaWQgcGlja2VyKCk7Ci19OwotCi1zdHJ1Y3QgbWF0cml4X3N0YWNrX3QgewotICAgIGVudW0gewotICAgICAgICBET19QSUNLRVIgICAgICAgICAgID0gMHgxLAotICAgICAgICBET19GTE9BVF9UT19GSVhFRCAgID0gMHgyCi0gICAgfTsKLSAgICB0cmFuc2Zvcm1fdCAgICAgdHJhbnNmb3JtOwotICAgIHVpbnQ4X3QgICAgICAgICBtYXhEZXB0aDsKLSAgICB1aW50OF90ICAgICAgICAgZGVwdGg7Ci0gICAgdWludDhfdCAgICAgICAgIGRpcnR5OwotICAgIHVpbnQ4X3QgICAgICAgICByZXNlcnZlZDsKLSAgICBtYXRyaXhmX3QgICAgICAgKnN0YWNrOwotICAgIHVpbnQ4X3QgICAgICAgICAqb3BzOwotICAgIHZvaWQgaW5pdChpbnQgZGVwdGgpOwotICAgIHZvaWQgdW5pbml0KCk7Ci0gICAgdm9pZCBsb2FkSWRlbnRpdHkoKTsKLSAgICB2b2lkIGxvYWQoY29uc3QgR0xmaXhlZCogcmhzKTsKLSAgICB2b2lkIGxvYWQoY29uc3QgR0xmbG9hdCogcmhzKTsKLSAgICB2b2lkIG11bHRpcGx5KGNvbnN0IG1hdHJpeGZfdCYgcmhzKTsKLSAgICB2b2lkIHRyYW5zbGF0ZShHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KTsKLSAgICB2b2lkIHNjYWxlKEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopOwotICAgIHZvaWQgcm90YXRlKEdMZmxvYXQgYSwgR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeik7Ci0gICAgR0xpbnQgcHVzaCgpOwotICAgIEdMaW50IHBvcCgpOwotICAgIHZvaWQgdmFsaWRhdGUoKTsKLSAgICBtYXRyaXhmX3QmIHRvcCgpIHsgcmV0dXJuIHN0YWNrW2RlcHRoXTsgfQotICAgIGNvbnN0IG1hdHJpeGZfdCYgdG9wKCkgY29uc3QgeyByZXR1cm4gc3RhY2tbZGVwdGhdOyB9Ci0gICAgY29uc3QgdWludDMyX3QgdG9wX29wcygpIGNvbnN0IHsgcmV0dXJuIG9wc1tkZXB0aF07IH0KLSAgICBpbmxpbmUgYm9vbCBpc1JpZ2lkQm9keSgpIGNvbnN0IHsKLSAgICAgICAgcmV0dXJuICEob3BzW2RlcHRoXSAmIH4oT1BfVFJBTlNMQVRFfE9QX1VOSUZPUk1fU0NBTEV8T1BfUk9UQVRFKSk7Ci0gICAgfQotfTsKLQotc3RydWN0IHZwX3RyYW5zZm9ybV90IHsKLSAgICB0cmFuc2Zvcm1fdCAgICAgdHJhbnNmb3JtOwotICAgIG1hdHJpeGZfdCAgICAgICBtYXRyaXg7Ci0gICAgR0xmbG9hdCAgICAgICAgIHpOZWFyOwotICAgIEdMZmxvYXQgICAgICAgICB6RmFyOwotICAgIHZvaWQgbG9hZElkZW50aXR5KCk7Ci19OwotCi1zdHJ1Y3QgdHJhbnNmb3JtX3N0YXRlX3QgewotICAgIGVudW0gewotICAgICAgICBNT0RFTFZJRVcgICAgICAgICAgID0gMHgwMSwKLSAgICAgICAgUFJPSkVDVElPTiAgICAgICAgICA9IDB4MDIsCi0gICAgICAgIFZJRVdQT1JUICAgICAgICAgICAgPSAweDA0LAotICAgICAgICBURVhUVVJFICAgICAgICAgICAgID0gMHgwOCwKLSAgICAgICAgTVZVSSAgICAgICAgICAgICAgICA9IDB4MTAsCi0gICAgICAgIE1WSVQgICAgICAgICAgICAgICAgPSAweDIwLAotICAgICAgICBNVlAgICAgICAgICAgICAgICAgID0gMHg0MCwKLSAgICB9OwotICAgIG1hdHJpeF9zdGFja190ICAgICAgKmN1cnJlbnQ7Ci0gICAgbWF0cml4X3N0YWNrX3QgICAgICBtb2RlbHZpZXc7Ci0gICAgbWF0cml4X3N0YWNrX3QgICAgICBwcm9qZWN0aW9uOwotICAgIG1hdHJpeF9zdGFja190ICAgICAgdGV4dHVyZVtHR0xfVEVYVFVSRV9VTklUX0NPVU5UXTsKLQotICAgIC8vIG1vZGVsdmlldyAqIHByb2plY3Rpb24KLSAgICB0cmFuc2Zvcm1fdCAgICAgICAgIG12cCAgICAgX19hdHRyaWJ1dGVfXygoYWxpZ25lZCgzMikpKTsKLSAgICAvLyB2aWV3cG9ydCB0cmFuc2Zvcm1hdGlvbgotICAgIHZwX3RyYW5zZm9ybV90ICAgICAgdnB0ICAgICBfX2F0dHJpYnV0ZV9fKChhbGlnbmVkKDMyKSkpOwotICAgIC8vIHNhbWUgZm9yIDQtRCB2ZXJ0aWNlcwotICAgIHRyYW5zZm9ybV90ICAgICAgICAgbXZwNDsKLSAgICAvLyBmdWxsIG1vZGVsdmlldyBpbnZlcnNlIHRyYW5zcG9zZQotICAgIHRyYW5zZm9ybV90ICAgICAgICAgbXZpdDQ7Ci0gICAgLy8gdXBwZXIgM3gzIG9mIG12LWludmVyc2UtdHJhbnNwb3NlIChmb3Igbm9ybWFscykKLSAgICBtdnVpX3RyYW5zZm9ybV90ICAgIG12dWk7Ci0KLSAgICBHTGVudW0gICAgICAgICAgICAgIG1hdHJpeE1vZGU7Ci0gICAgR0xlbnVtICAgICAgICAgICAgICByZXNjYWxlTm9ybWFsczsKLSAgICB1aW50MzJfdCAgICAgICAgICAgIGRpcnR5OwotICAgIHZvaWQgaW52YWxpZGF0ZSgpOwotICAgIHZvaWQgdXBkYXRlX212cCgpOwotICAgIHZvaWQgdXBkYXRlX212aXQoKTsKLSAgICB2b2lkIHVwZGF0ZV9tdnVpKCk7Ci19OwotCi1zdHJ1Y3Qgdmlld3BvcnRfdCB7Ci0gICAgR0xpbnQgICAgICAgeDsKLSAgICBHTGludCAgICAgICB5OwotICAgIEdMc2l6ZWkgICAgIHc7Ci0gICAgR0xzaXplaSAgICAgaDsgCi0gICAgc3RydWN0IHsKLSAgICAgICAgR0xpbnQgICAgICAgeDsKLSAgICAgICAgR0xpbnQgICAgICAgeTsKLSAgICB9IHN1cmZhY2Vwb3J0OyAgCi0gICAgc3RydWN0IHsKLSAgICAgICAgR0xpbnQgICAgICAgeDsKLSAgICAgICAgR0xpbnQgICAgICAgeTsKLSAgICAgICAgR0xzaXplaSAgICAgdzsKLSAgICAgICAgR0xzaXplaSAgICAgaDsgCi0gICAgfSBzY2lzc29yOyAgCi19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBMZXJwaW5nCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0cnVjdCBjb21wdXRlX2l0ZXJhdG9yc190Ci17Ci0gICAgdm9pZCBpbml0VHJpYW5nbGUoCi0gICAgICAgICAgICB2ZXJ0ZXhfdCBjb25zdCogdjAsCi0gICAgICAgICAgICB2ZXJ0ZXhfdCBjb25zdCogdjEsCi0gICAgICAgICAgICB2ZXJ0ZXhfdCBjb25zdCogdjIpOwotCi0gICAgdm9pZCBpbml0TGluZSgKLSAgICAgICAgICAgIHZlcnRleF90IGNvbnN0KiB2MCwKLSAgICAgICAgICAgIHZlcnRleF90IGNvbnN0KiB2MSk7Ci0KLSAgICBpbmxpbmUgdm9pZCBpbml0TGVycCh2ZXJ0ZXhfdCBjb25zdCogdjAsIHVpbnQzMl90IGVuYWJsZXMpOwotCi0gICAgaW50IGl0ZXJhdG9yc1NjYWxlKGludDMyX3QgaXRbM10sCi0gICAgICAgICAgICBpbnQzMl90IGMwLCBpbnQzMl90IGMxLCBpbnQzMl90IGMyKSBjb25zdDsKLQotICAgIHZvaWQgaXRlcmF0b3JzMTYxNihHR0xmaXhlZCBpdFszXSwKLSAgICAgICAgICAgIEdHTGZpeGVkIGMwLCBHR0xmaXhlZCBjMSwgR0dMZml4ZWQgYzIpIGNvbnN0OwotCi0gICAgdm9pZCBpdGVyYXRvcnMwMDMyKGludDMyX3QgaXRbM10sCi0gICAgICAgICAgICBpbnQzMl90IGMwLCBpbnQzMl90IGMxLCBpbnQzMl90IGMyKSBjb25zdDsKLQotICAgIHZvaWQgaXRlcmF0b3JzMDAzMihpbnQ2NF90IGl0WzNdLAotICAgICAgICAgICAgaW50MzJfdCBjMCwgaW50MzJfdCBjMSwgaW50MzJfdCBjMikgY29uc3Q7Ci0KLSAgICBHR0xjb29yZCBhcmVhKCkgY29uc3QgeyByZXR1cm4gbV9hcmVhOyB9Ci0KLXByaXZhdGU6Ci0gICAgLy8gZG9uJ3QgY2hhbmdlIG9yZGVyIG9mIG1lbWJlcnMgaGVyZSAtLSB1c2VkIGJ5IGl0ZXJhdG9ycy5TCi0gICAgR0dMY29vcmQgbV9keDAxLCBtX2R5MTAsIG1fZHgyMCwgbV9keTAyOwotICAgIEdHTGNvb3JkIG1feDAsIG1feTA7Ci0gICAgR0dMY29vcmQgbV9hcmVhOwotICAgIHVpbnQ4X3QgbV9zY2FsZTsKLSAgICB1aW50OF90IG1fYXJlYV9zY2FsZTsKLSAgICB1aW50OF90IG1fcmVzZXJ2ZWRbMl07Ci0KLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIHN0YXRlCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNpZmRlZiBIQVZFX0FORFJPSURfT1MKLSAgICAvLyBXZSBoYXZlIGEgZGVkaWNhdGVkIFRMUyBzbG90IGluIGJpb25pYwotICAgIGlubGluZSB2b2lkIHNldEdsVGhyZWFkU3BlY2lmaWMob2dsZXNfY29udGV4dF90ICp2YWx1ZSkgewotICAgICAgICAoKHVpbnQzMl90ICopX19nZXRfdGxzKCkpW1RMU19TTE9UX09QRU5HTF0gPSAodWludDMyX3QpdmFsdWU7Ci0gICAgfQotICAgIGlubGluZSBvZ2xlc19jb250ZXh0X3QqIGdldEdsVGhyZWFkU3BlY2lmaWMoKSB7Ci0gICAgICAgIHJldHVybiAob2dsZXNfY29udGV4dF90ICopKCgodW5zaWduZWQgKilfX2dldF90bHMoKSlbVExTX1NMT1RfT1BFTkdMXSk7Ci0gICAgfQotI2Vsc2UKLSAgICBleHRlcm4gcHRocmVhZF9rZXlfdCBnR0xLZXk7Ci0gICAgaW5saW5lIHZvaWQgc2V0R2xUaHJlYWRTcGVjaWZpYyhvZ2xlc19jb250ZXh0X3QgKnZhbHVlKSB7Ci0gICAgICAgIHB0aHJlYWRfc2V0c3BlY2lmaWMoZ0dMS2V5LCB2YWx1ZSk7Ci0gICAgfQotICAgIGlubGluZSBvZ2xlc19jb250ZXh0X3QqIGdldEdsVGhyZWFkU3BlY2lmaWMoKSB7Ci0gICAgICAgIHJldHVybiBzdGF0aWNfY2FzdDxvZ2xlc19jb250ZXh0X3QqPihwdGhyZWFkX2dldHNwZWNpZmljKGdHTEtleSkpOwotICAgIH0KLSNlbmRpZgotCi0KLXN0cnVjdCBwcmltc190IHsKLSAgICB0eXBlZGVmIG9nbGVzX2NvbnRleHRfdCogR0w7Ci0gICAgdm9pZCAoKnJlbmRlclBvaW50KShHTCwgdmVydGV4X3QqKTsKLSAgICB2b2lkICgqcmVuZGVyTGluZSkoR0wsIHZlcnRleF90KiwgdmVydGV4X3QqKTsKLSAgICB2b2lkICgqcmVuZGVyVHJpYW5nbGUpKEdMLCB2ZXJ0ZXhfdCosIHZlcnRleF90KiwgdmVydGV4X3QqKTsKLX07Ci0KLXN0cnVjdCBvZ2xlc19jb250ZXh0X3QgewotICAgIGNvbnRleHRfdCAgICAgICAgICAgICAgIHJhc3Rlcml6ZXI7Ci0gICAgYXJyYXlfbWFjaGluZV90ICAgICAgICAgYXJyYXlzICAgICAgICAgX19hdHRyaWJ1dGVfXygoYWxpZ25lZCgzMikpKTsKLSAgICB0ZXh0dXJlX3N0YXRlX3QgICAgICAgICB0ZXh0dXJlczsKLSAgICB0cmFuc2Zvcm1fc3RhdGVfdCAgICAgICB0cmFuc2Zvcm1zOwotICAgIHZlcnRleF9jYWNoZV90ICAgICAgICAgIHZjOwotICAgIHByaW1zX3QgICAgICAgICAgICAgICAgIHByaW1zOwotICAgIGN1bGxpbmdfdCAgICAgICAgICAgICAgIGN1bGw7Ci0gICAgbGlnaHRpbmdfdCAgICAgICAgICAgICAgbGlnaHRpbmc7Ci0gICAgdXNlcl9jbGlwX3BsYW5lc190ICAgICAgY2xpcFBsYW5lczsKLSAgICBjb21wdXRlX2l0ZXJhdG9yc190ICAgICBsZXJwOyAgICAgICAgICAgX19hdHRyaWJ1dGVfXygoYWxpZ25lZCgzMikpKTsKLSAgICB2ZXJ0ZXhfdCAgICAgICAgICAgICAgICBjdXJyZW50OwotICAgIHZlYzRfdCAgICAgICAgICAgICAgICAgIGN1cnJlbnRDb2xvckNsYW1wZWQ7Ci0gICAgdmVjM190ICAgICAgICAgICAgICAgICAgY3VycmVudE5vcm1hbDsKLSAgICB2aWV3cG9ydF90ICAgICAgICAgICAgICB2aWV3cG9ydDsKLSAgICBwb2ludF9zaXplX3QgICAgICAgICAgICBwb2ludDsKLSAgICBsaW5lX3dpZHRoX3QgICAgICAgICAgICBsaW5lOwotICAgIHBvbHlnb25fb2Zmc2V0X3QgICAgICAgIHBvbHlnb25PZmZzZXQ7Ci0gICAgZm9nX3QgICAgICAgICAgICAgICAgICAgZm9nOwotICAgIHVpbnQzMl90ICAgICAgICAgICAgICAgIHBlcnNwZWN0aXZlIDogMTsKLSAgICB1aW50MzJfdCAgICAgICAgICAgICAgICB0cmFuc2Zvcm1UZXh0dXJlcyA6IDE7Ci0gICAgRUdMU3VyZmFjZU1hbmFnZXIqICAgICAgc3VyZmFjZU1hbmFnZXI7Ci0gICAgRUdMQnVmZmVyT2JqZWN0TWFuYWdlciogYnVmZmVyT2JqZWN0TWFuYWdlcjsKLSAgICBHTGVudW0gICAgICAgICAgICAgICAgICBlcnJvcjsKLQotICAgIHN0YXRpYyBpbmxpbmUgb2dsZXNfY29udGV4dF90KiBnZXQoKSB7Ci0gICAgICAgIHJldHVybiBnZXRHbFRocmVhZFNwZWNpZmljKCk7Ci0gICAgfQotCi19OwotCi19OyAvLyBuYW1lc3BhY2UgZ2wKLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX09QRU5HTEVTX0NPTlRFWFRfSAotCmRpZmYgLS1naXQgYS9pbmNsdWRlL3ByaXZhdGUvdWkvTGF5ZXJTdGF0ZS5oIGIvaW5jbHVkZS9wcml2YXRlL3VpL0xheWVyU3RhdGUuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYjZmY2Q4MC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3ByaXZhdGUvdWkvTGF5ZXJTdGF0ZS5oCisrKyAvZGV2L251bGwKQEAgLTEsNzUgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9DT01QT1NFUl9MQVlFUl9TVEFURV9ICi0jZGVmaW5lIEFORFJPSURfQ09NUE9TRVJfTEFZRVJfU1RBVEVfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLQotI2luY2x1ZGUgPHVpL0lTdXJmYWNlRmxpbmdlckNsaWVudC5oPgotI2luY2x1ZGUgPHVpL1JlZ2lvbi5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS91aS9TaGFyZWRTdGF0ZS5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIFBhcmNlbDsKLQotc3RydWN0IGxheWVyX3N0YXRlX3QgewotCi0gICAgbGF5ZXJfc3RhdGVfdCgpCi0gICAgICAgIDogICBzdXJmYWNlKDApLCB3aGF0KDApLAotICAgICAgICAgICAgeCgwKSwgeSgwKSwgeigwKSwgdygwKSwgaCgwKSwKLSAgICAgICAgICAgIGFscGhhKDApLCB0aW50KDApLCBmbGFncygwKSwgbWFzaygwKSwKLSAgICAgICAgICAgIHJlc2VydmVkKDApCi0gICAgewotICAgICAgICBtYXRyaXguZHNkeCA9IG1hdHJpeC5kdGR5ID0gMS4wZjsKLSAgICAgICAgbWF0cml4LmRzZHkgPSBtYXRyaXguZHRkeCA9IDAuMGY7Ci0gICAgfQotCi0gICAgc3RhdHVzX3QgICAgd3JpdGUoUGFyY2VsJiBvdXRwdXQpIGNvbnN0OwotICAgIHN0YXR1c190ICAgIHJlYWQoY29uc3QgUGFyY2VsJiBpbnB1dCk7Ci0KLSAgICAgICAgICAgIHN0cnVjdCBtYXRyaXgyMl90IHsKLSAgICAgICAgICAgICAgICBmbG9hdCAgIGRzZHg7Ci0gICAgICAgICAgICAgICAgZmxvYXQgICBkdGR4OwotICAgICAgICAgICAgICAgIGZsb2F0ICAgZHNkeTsKLSAgICAgICAgICAgICAgICBmbG9hdCAgIGR0ZHk7Ci0gICAgICAgICAgICB9OwotICAgICAgICAgICAgU3VyZmFjZUlEICAgICAgIHN1cmZhY2U7Ci0gICAgICAgICAgICB1aW50MzJfdCAgICAgICAgd2hhdDsKLSAgICAgICAgICAgIGludDMyX3QgICAgICAgICB4OwotICAgICAgICAgICAgaW50MzJfdCAgICAgICAgIHk7Ci0gICAgICAgICAgICB1aW50MzJfdCAgICAgICAgejsKLSAgICAgICAgICAgIHVpbnQzMl90ICAgICAgICB3OwotICAgICAgICAgICAgdWludDMyX3QgICAgICAgIGg7Ci0gICAgICAgICAgICBmbG9hdCAgICAgICAgICAgYWxwaGE7Ci0gICAgICAgICAgICB1aW50MzJfdCAgICAgICAgdGludDsKLSAgICAgICAgICAgIHVpbnQ4X3QgICAgICAgICBmbGFnczsKLSAgICAgICAgICAgIHVpbnQ4X3QgICAgICAgICBtYXNrOwotICAgICAgICAgICAgdWludDhfdCAgICAgICAgIHJlc2VydmVkOwotICAgICAgICAgICAgbWF0cml4MjJfdCAgICAgIG1hdHJpeDsKLSAgICAgICAgICAgIC8vIG5vbiBQT0QgbXVzdCBiZSBsYXN0LiBzZWUgd3JpdGUvcmVhZAotICAgICAgICAgICAgUmVnaW9uICAgICAgICAgIHRyYW5zcGFyZW50UmVnaW9uOwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfQ09NUE9TRVJfTEFZRVJfU1RBVEVfSAotCmRpZmYgLS1naXQgYS9pbmNsdWRlL3ByaXZhdGUvdWkvU2hhcmVkU3RhdGUuaCBiL2luY2x1ZGUvcHJpdmF0ZS91aS9TaGFyZWRTdGF0ZS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1NDZkMGFkLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvcHJpdmF0ZS91aS9TaGFyZWRTdGF0ZS5oCisrKyAvZGV2L251bGwKQEAgLTEsMTY4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfVUlfU0hBUkVEX1NUQVRFX0gKLSNkZWZpbmUgQU5EUk9JRF9VSV9TSEFSRURfU1RBVEVfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyoKLSAqIFRoZXNlIHN0cnVjdHVyZXMgYXJlIHNoYXJlZCBiZXR3ZWVuIHRoZSBjb21wb3NlciBwcm9jZXNzIGFuZCBpdHMgY2xpZW50cwotICovCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3Qgc3VyZmFjZV9pbmZvX3QgeyAvLyA0IGxvbmdzLCAxNiBieXRlcwotICAgIGVudW0gewotICAgICAgICBlQnVmZmVyRGlydHkgICAgPSAweDAxCi0gICAgfTsKLSAgICB1aW50MTZfdCAgICB3OwotICAgIHVpbnQxNl90ICAgIGg7Ci0gICAgdWludDE2X3QgICAgc3RyaWRlOwotICAgIHVpbnQxNl90ICAgIGJwcjsKLSAgICB1aW50MTZfdCAgICByZXNlcnZlZDsKLSAgICB1aW50OF90ICAgICBmb3JtYXQ7Ci0gICAgdWludDhfdCAgICAgZmxhZ3M7Ci0gICAgc3NpemVfdCAgICAgYml0c19vZmZzZXQ7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY29uc3QgdWludDMyX3QgTlVNX0xBWUVSU19NQVggPSAzMTsKLQotZW51bSB7IC8vIGxheWVyX2NibGtfdCBzd2FwU3RhdGUKLSAgICBlSW5kZXggICAgICAgICAgICAgID0gMHgwMDAwMDAwMSwKLSAgICBlRmxpcFJlcXVlc3RlZCAgICAgID0gMHgwMDAwMDAwMiwKLSAgICAKLSAgICBlUmVzaXplQnVmZmVyMCAgICAgID0gMHgwMDAwMDAwNCwKLSAgICBlUmVzaXplQnVmZmVyMSAgICAgID0gMHgwMDAwMDAwOCwgICAgCi0gICAgZVJlc2l6ZVJlcXVlc3RlZCAgICA9IGVSZXNpemVCdWZmZXIwIHwgZVJlc2l6ZUJ1ZmZlcjEsCi0gICAgCi0gICAgZUJ1c3kgICAgICAgICAgICAgICA9IDB4MDAwMDAwMTAsCi0gICAgZUxvY2tlZCAgICAgICAgICAgICA9IDB4MDAwMDAwMjAsCi0gICAgZU5leHRGbGlwUGVuZGluZyAgICA9IDB4MDAwMDAwNDAsICAgIAotICAgIGVJbnZhbGlkU3VyZmFjZSAgICAgPSAweDAwMDAwMDgwCi19OwotCi1lbnVtIHsgLy8gbGF5ZXJfY2Jsa190IGZsYWdzCi0gICAgZUxheWVyTm90UG9zdGVkICAgICA9IDB4MDAwMDAwMDEsCi0gICAgZU5vQ29weUJhY2sgICAgICAgICA9IDB4MDAwMDAwMDIsCi0gICAgZVJlc2VydmVkICAgICAgICAgICA9IDB4MDAwMDAwN0MsCi0gICAgZUJ1ZmZlckluZGV4U2hpZnQgICA9IDcsCi0gICAgZUJ1ZmZlckluZGV4ICAgICAgICA9IDE8PGVCdWZmZXJJbmRleFNoaWZ0LAotfTsKLQotc3RydWN0IGZsYXRfcmVnaW9uX3QgICAgLy8gNDAgYnl0ZXMKLXsKLSAgICBpbnQzMl90ICAgICBjb3VudDsKLSAgICBpbnQxNl90ICAgICBsOwotICAgIGludDE2X3QgICAgIHQ7Ci0gICAgaW50MTZfdCAgICAgcjsKLSAgICBpbnQxNl90ICAgICBiOwotICAgIHVpbnQxNl90ICAgIHJ1bnNbMTRdOwotfTsKLQotc3RydWN0IGxheWVyX2NibGtfdCAgICAgLy8gKDEyOCBieXRlcykKLXsKLSAgICB2b2xhdGlsZSAgICBpbnQzMl90ICAgICAgICAgICAgIHN3YXBTdGF0ZTsgICAgICAvLyAgNAotICAgIHZvbGF0aWxlICAgIGludDMyX3QgICAgICAgICAgICAgZmxhZ3M7ICAgICAgICAgIC8vICA0Ci0gICAgdm9sYXRpbGUgICAgaW50MzJfdCAgICAgICAgICAgICBpZGVudGl0eTsgICAgICAgLy8gIDQKLSAgICAgICAgICAgICAgICBpbnQzMl90ICAgICAgICAgICAgIHJlc2VydmVkOyAgICAgICAvLyAgNAotICAgICAgICAgICAgICAgIHN1cmZhY2VfaW5mb190ICAgICAgc3VyZmFjZVsyXTsgICAgIC8vIDMyCi0gICAgICAgICAgICAgICAgZmxhdF9yZWdpb25fdCAgICAgICByZWdpb25bMl07ICAgICAgLy8gODAKLQotICAgIHN0YXRpYyBpbmxpbmUgaW50IGJhY2tCdWZmZXIodWludDMyX3Qgc3RhdGUpIHsKLSAgICAgICAgcmV0dXJuICgoc3RhdGUgJiBlSW5kZXgpIF4gKChzdGF0ZSAmIGVGbGlwUmVxdWVzdGVkKT4+MSkpOwotICAgIH0KLSAgICBzdGF0aWMgaW5saW5lIGludCBmcm9udEJ1ZmZlcih1aW50MzJfdCBzdGF0ZSkgewotICAgICAgICByZXR1cm4gMSAtIGJhY2tCdWZmZXIoc3RhdGUpOwotICAgIH0KLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3QgcGVyX2NsaWVudF9jYmxrX3QgICAvLyA0S0IgbWF4Ci17Ci0gICAgICAgICAgICAgICAgTXV0ZXggICAgICAgICAgIGxvY2s7Ci0gICAgICAgICAgICAgICAgQ29uZGl0aW9uICAgICAgIGN2OwotICAgICAgICAgICAgICAgIGxheWVyX2NibGtfdCAgICBsYXllcnNbTlVNX0xBWUVSU19NQVhdIF9fYXR0cmlidXRlX18oKGFsaWduZWQoMzIpKSk7Ci0KLSAgICBlbnVtIHsKLSAgICAgICAgQkxPQ0tJTkcgPSAweDAwMDAwMDAxLAotICAgICAgICBJTlNQRUNUICA9IDB4MDAwMDAwMDIKLSAgICB9OwotCi0gICAgcGVyX2NsaWVudF9jYmxrX3QoKTsKLQotICAgIC8vIHRoZXNlIGZ1bmN0aW9ucyBhcmUgdXNlZCBieSB0aGUgY2xpZW50cwotICAgIHN0YXR1c190IHZhbGlkYXRlKHNpemVfdCBpKSBjb25zdDsKLSAgICBpbnQzMl90IGxvY2tfbGF5ZXIoc2l6ZV90IGksIHVpbnQzMl90IGZsYWdzKTsKLSAgICB1aW50MzJfdCB1bmxvY2tfbGF5ZXJfYW5kX3Bvc3Qoc2l6ZV90IGkpOwotICAgIHZvaWQgdW5sb2NrX2xheWVyKHNpemVfdCBpKTsKLX07Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY29uc3QgdWludDMyX3QgTlVNX0RJU1BMQVlfTUFYID0gNDsKLQotc3RydWN0IGRpc3BsYXlfY2Jsa190Ci17Ci0gICAgdWludDE2X3QgICAgdzsKLSAgICB1aW50MTZfdCAgICBoOwotICAgIHVpbnQ4X3QgICAgIGZvcm1hdDsKLSAgICB1aW50OF90ICAgICBvcmllbnRhdGlvbjsKLSAgICB1aW50OF90ICAgICByZXNlcnZlZFsyXTsKLSAgICBmbG9hdCAgICAgICBmcHM7Ci0gICAgZmxvYXQgICAgICAgZGVuc2l0eTsKLSAgICBmbG9hdCAgICAgICB4ZHBpOwotICAgIGZsb2F0ICAgICAgIHlkcGk7Ci0gICAgdWludDMyX3QgICAgcGFkWzJdOwotfTsKLQotc3RydWN0IHN1cmZhY2VfZmxpbmdlcl9jYmxrX3QgICAvLyA0S0IgbWF4Ci17Ci0gICAgc3VyZmFjZV9mbGluZ2VyX2NibGtfdCgpOwotICAgIAotICAgIHVpbnQ4X3QgICAgICAgICBjb25uZWN0ZWQ7Ci0gICAgdWludDhfdCAgICAgICAgIHJlc2VydmVkWzNdOwotICAgIHVpbnQzMl90ICAgICAgICBwYWRbN107Ci0gCi0gICAgZGlzcGxheV9jYmxrX3QgIGRpc3BsYXlzW05VTV9ESVNQTEFZX01BWF07Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdGVtcGxhdGU8Ym9vbD4gc3RydWN0IENUQTsKLXRlbXBsYXRlPD4gc3RydWN0IENUQTx0cnVlPiB7IH07Ci0KLS8vIGNvbXBpbGUtdGltZSBhc3NlcnRpb25zLiBqdXN0IHRvIGF2b2lkIGNhdGFzdHJvcGhlcy4KLWlubGluZSB2b2lkIGNvbXBpbGVfdGltZV9hc3NlcnRzKCkgewotICAgIENUQTxzaXplb2YobGF5ZXJfY2Jsa190KSA9PSAxMjg+IHNpemVvZl9fbGF5ZXJfY2Jsa190X19lcV8xMjg7Ci0gICAgKHZvaWQpc2l6ZW9mX19sYXllcl9jYmxrX3RfX2VxXzEyODsgLy8gd2UgZG9uJ3Qgd2FudCBhIHdhcm5pbmcKLSAgICBDVEE8c2l6ZW9mKHBlcl9jbGllbnRfY2Jsa190KSA8PSA0MDk2PiBzaXplb2ZfX3Blcl9jbGllbnRfY2Jsa190X19sZV80MDk2OwotICAgICh2b2lkKXNpemVvZl9fcGVyX2NsaWVudF9jYmxrX3RfX2xlXzQwOTY7ICAvLyB3ZSBkb24ndCB3YW50IGEgd2FybmluZwotICAgIENUQTxzaXplb2Yoc3VyZmFjZV9mbGluZ2VyX2NibGtfdCkgPD0gNDA5Nj4gc2l6ZW9mX19zdXJmYWNlX2ZsaW5nZXJfY2Jsa190X19sZV80MDk2OwotICAgICh2b2lkKXNpemVvZl9fc3VyZmFjZV9mbGluZ2VyX2NibGtfdF9fbGVfNDA5NjsgIC8vIHdlIGRvbid0IHdhbnQgYSB3YXJuaW5nCi19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX1VJX1NIQVJFRF9TVEFURV9ICi0KZGlmZiAtLWdpdCBhL2luY2x1ZGUvcHJpdmF0ZS91aS9TdXJmYWNlRmxpbmdlclN5bmNocm8uaCBiL2luY2x1ZGUvcHJpdmF0ZS91aS9TdXJmYWNlRmxpbmdlclN5bmNocm8uaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZmY5MWI2MS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3ByaXZhdGUvdWkvU3VyZmFjZUZsaW5nZXJTeW5jaHJvLmgKKysrIC9kZXYvbnVsbApAQCAtMSw3NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotCi0jaWZuZGVmIEFORFJPSURfU1VSRkFDRV9GTElOR0VSX1NZTkNIUk9fSAotI2RlZmluZSBBTkRST0lEX1NVUkZBQ0VfRkxJTkdFUl9TWU5DSFJPX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLSNpbmNsdWRlIDx1aS9JU3VyZmFjZUNvbXBvc2VyLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgU3VyZmFjZUZsaW5nZXI7Ci0KLWNsYXNzIFN1cmZhY2VGbGluZ2VyU3luY2hybwotewotcHVibGljOgotCi0gICAgICAgICAgICAgICAgLy8gY2xpZW50IGNvbnN0cnVjdG9yCi0gICAgICAgICAgICAgICAgU3VyZmFjZUZsaW5nZXJTeW5jaHJvKGNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBmbGluZ2VyKTsKLSAgICAgICAgICAgICAgICB+U3VyZmFjZUZsaW5nZXJTeW5jaHJvKCk7Ci0gICAgCi0gICAgICAgICAgICAgICAgLy8gc2lnbmFsIHN1cmZhY2VmbGluZ2VyIGZvciBzb21lIHdvcmsKLSAgICBzdGF0dXNfdCAgICBzaWduYWwoKTsKLSAgICAKLXByaXZhdGU6Ci0gICAgY2xhc3MgQmFycmllciB7Ci0gICAgcHVibGljOgotICAgICAgICBCYXJyaWVyKCk7Ci0gICAgICAgIH5CYXJyaWVyKCk7Ci0gICAgICAgIHZvaWQgb3BlbigpOwotICAgICAgICB2b2lkIGNsb3NlKCk7Ci0gICAgICAgIHZvaWQgd2FpdEFuZENsb3NlKCk7Ci0gICAgICAgIHN0YXR1c190IHdhaXRBbmRDbG9zZShuc2Vjc190IHRpbWVvdXQpOwotICAgIHByaXZhdGU6Ci0gICAgICAgIGVudW0geyBPUEVORUQsIENMT1NFRCB9OwotICAgICAgICBtdXRhYmxlICAgICBNdXRleCAgICAgICBsb2NrOwotICAgICAgICBtdXRhYmxlICAgICBDb25kaXRpb24gICBjdjsKLSAgICAgICAgdm9sYXRpbGUgICAgaW50ICAgICAgICAgc3RhdGU7Ci0gICAgfTsKLQotICAgIGZyaWVuZCBjbGFzcyBTdXJmYWNlRmxpbmdlcjsKLQotICAgICAgICAgICAgICAgIC8vIHNlcnZlciBjb25zdHJ1Y3RvcgotICAgICAgICAgICAgICAgIFN1cmZhY2VGbGluZ2VyU3luY2hybygpOwotICAgICAgICAgICAgICAgIAotICAgIHZvaWQgICAgICAgIG9wZW4oKTsKLSAgICAKLSAgICAgICAgICAgICAgICAvLyB3YWl0IHVudGlsIHRoZXJlIGlzIHNvbWUgd29yayB0byBkbwotICAgIHN0YXR1c190ICAgIHdhaXQoKTsKLSAgICBzdGF0dXNfdCAgICB3YWl0KG5zZWNzX3QgdGltZW91dCk7Ci0gICAgCi0gICAgc3A8SVN1cmZhY2VDb21wb3Nlcj4gbVN1cmZhY2VDb21wb3NlcjsKLSAgICBCYXJyaWVyICAgICAgICAgICAgICBtQmFycmllcjsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX1NVUkZBQ0VfRkxJTkdFUl9TWU5DSFJPX0gKLQpkaWZmIC0tZ2l0IGEvaW5jbHVkZS9wcml2YXRlL3V0aWxzL1N0YXRpYy5oIGIvaW5jbHVkZS9wcml2YXRlL3V0aWxzL1N0YXRpYy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmMTQzOWI3Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvcHJpdmF0ZS91dGlscy9TdGF0aWMuaAorKysgL2Rldi9udWxsCkBAIC0xLDU4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLyBBbGwgc3RhdGljIHZhcmlhYmxlcyBnbyBoZXJlLCB0byBjb250cm9sIGluaXRpYWxpemF0aW9uIGFuZAotLy8gZGVzdHJ1Y3Rpb24gb3JkZXIgaW4gdGhlIGxpYnJhcnkuCi0KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvS2V5ZWRWZWN0b3IuaD4KLQotI2lmbmRlZiBMSUJVVElMU19OQVRJVkUKLSNpbmNsdWRlIDx1dGlscy9JQmluZGVyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSU1lbW9yeS5oPgotI2luY2x1ZGUgPHV0aWxzL1Byb2Nlc3NTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL0lQZXJtaXNzaW9uQ29udHJvbGxlci5oPgotI2luY2x1ZGUgPHV0aWxzL0lTZXJ2aWNlTWFuYWdlci5oPgotI2VuZGlmCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8vIEZvciBUZXh0U3RyZWFtLmNwcAotZXh0ZXJuIFZlY3RvcjxpbnQzMl90PiBnVGV4dEJ1ZmZlcnM7Ci0KLS8vIEZvciBTdHJpbmc4LmNwcAotZXh0ZXJuIHZvaWQgaW5pdGlhbGl6ZV9zdHJpbmc4KCk7Ci1leHRlcm4gdm9pZCB0ZXJtaW5hdGVfc3RyaW5nOCgpOwotCi0vLyBGb3IgU3RyaW5nMTYuY3BwCi1leHRlcm4gdm9pZCBpbml0aWFsaXplX3N0cmluZzE2KCk7Ci1leHRlcm4gdm9pZCB0ZXJtaW5hdGVfc3RyaW5nMTYoKTsKLQotCi0KLSNpZm5kZWYgTElCVVRJTFNfTkFUSVZFCi0KLS8vIEZvciBQcm9jZXNzU3RhdGUuY3BwCi1leHRlcm4gTXV0ZXggZ1Byb2Nlc3NNdXRleDsKLWV4dGVybiBzcDxQcm9jZXNzU3RhdGU+IGdQcm9jZXNzOwotCi0vLyBGb3IgU2VydmljZU1hbmFnZXIuY3BwCi1leHRlcm4gTXV0ZXggZ0RlZmF1bHRTZXJ2aWNlTWFuYWdlckxvY2s7Ci1leHRlcm4gc3A8SVNlcnZpY2VNYW5hZ2VyPiBnRGVmYXVsdFNlcnZpY2VNYW5hZ2VyOwotZXh0ZXJuIHNwPElQZXJtaXNzaW9uQ29udHJvbGxlcj4gZ1Blcm1pc3Npb25Db250cm9sbGVyOwotCi0jZW5kaWYKLQotfSAgIC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9pbmNsdWRlL3ByaXZhdGUvdXRpbHMvYmluZGVyX21vZHVsZS5oIGIvaW5jbHVkZS9wcml2YXRlL3V0aWxzL2JpbmRlcl9tb2R1bGUuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZmRmMzI3ZS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3ByaXZhdGUvdXRpbHMvYmluZGVyX21vZHVsZS5oCisrKyAvZGV2L251bGwKQEAgLTEsMTQ4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIF9CSU5ERVJfTU9EVUxFX0hfCi0jZGVmaW5lIF9CSU5ERVJfTU9EVUxFX0hfCi0KLSNpZmRlZiBfX2NwbHVzcGx1cwotbmFtZXNwYWNlIGFuZHJvaWQgewotI2VuZGlmCi0KLSNpZiBkZWZpbmVkKEhBVkVfQU5EUk9JRF9PUykKLQotLyogb2J0YWluIHN0cnVjdHVyZXMgYW5kIGNvbnN0YW50cyBmcm9tIHRoZSBrZXJuZWwgaGVhZGVyICovCi0KLSNpbmNsdWRlIDxzeXMvaW9jdGwuaD4KLSNpbmNsdWRlIDxsaW51eC9iaW5kZXIuaD4KLQotI2Vsc2UKLQotLyogU29tZSBwYXJ0cyBvZiB0aGUgc2ltdWxhdG9yIG5lZWQgZmFrZSB2ZXJzaW9ucyBvZiB0aGlzIAotICogc3R1ZmYgaW4gb3JkZXIgdG8gY29tcGlsZS4gIFJlYWxseSB0aGlzIHNob3VsZCBnbyBhd2F5Ci0gKiBlbnRpcmVseS4uLgotICovCi0KLSNkZWZpbmUgQklOREVSX0NVUlJFTlRfUFJPVE9DT0xfVkVSU0lPTiA3Ci0KLSNkZWZpbmUgQklOREVSX1RZUEVfQklOREVSIDEKLSNkZWZpbmUgQklOREVSX1RZUEVfV0VBS19CSU5ERVIgMgotI2RlZmluZSBCSU5ERVJfVFlQRV9IQU5ETEUgMwotI2RlZmluZSBCSU5ERVJfVFlQRV9XRUFLX0hBTkRMRSA0Ci0jZGVmaW5lIEJJTkRFUl9UWVBFX0ZEIDUKLQotc3RydWN0IGZsYXRfYmluZGVyX29iamVjdCB7Ci0gICAgdW5zaWduZWQgbG9uZyB0eXBlOwotICAgIHVuc2lnbmVkIGxvbmcgZmxhZ3M7Ci0gICAgdW5pb24gewotICAgICAgICB2b2lkICpiaW5kZXI7Ci0gICAgICAgIHNpZ25lZCBsb25nIGhhbmRsZTsKLSAgICB9OwotICAgIHZvaWQgKmNvb2tpZTsKLX07Ci0KLXN0cnVjdCBiaW5kZXJfd3JpdGVfcmVhZCB7Ci0gICAgc2lnbmVkIGxvbmcgd3JpdGVfc2l6ZTsKLSAgICBzaWduZWQgbG9uZyB3cml0ZV9jb25zdW1lZDsKLSAgICB1bnNpZ25lZCBsb25nIHdyaXRlX2J1ZmZlcjsKLSAgICBzaWduZWQgbG9uZyByZWFkX3NpemU7Ci0gICAgc2lnbmVkIGxvbmcgcmVhZF9jb25zdW1lZDsKLSAgICB1bnNpZ25lZCBsb25nIHJlYWRfYnVmZmVyOwotfTsKLQotc3RydWN0IGJpbmRlcl90cmFuc2FjdGlvbl9kYXRhIHsKLSAgICB1bmlvbiB7Ci0gICAgICAgIHNpemVfdCBoYW5kbGU7Ci0gICAgICAgIHZvaWQgKnB0cjsKLSAgICB9IHRhcmdldDsKLSAgICB2b2lkICpjb29raWU7Ci0gICAgdW5zaWduZWQgaW50IGNvZGU7Ci0gICAgCi0gICAgdW5zaWduZWQgaW50IGZsYWdzOwotICAgIHBpZF90IHNlbmRlcl9waWQ7Ci0gICAgdWlkX3Qgc2VuZGVyX2V1aWQ7Ci0gICAgc2l6ZV90IGRhdGFfc2l6ZTsKLSAgICBzaXplX3Qgb2Zmc2V0c19zaXplOwotICAgIAotICAgIHVuaW9uIHsKLSAgICAgICAgc3RydWN0IHsKLSAgICAgICAgICAgIGNvbnN0IHZvaWQgKmJ1ZmZlcjsKLSAgICAgICAgICAgIGNvbnN0IHZvaWQgKm9mZnNldHM7Ci0gICAgICAgIH0gcHRyOwotICAgICAgICB1aW50OF90IGJ1Zls4XTsKLSAgICB9IGRhdGE7Ci19OwotCi1lbnVtIHRyYW5zYWN0aW9uX2ZsYWdzIHsKLSAgICBURl9PTkVfV0FZID0gMHgwMSwKLSAgICBURl9ST09UX09CSkVDVCA9IDB4MDQsCi0gICAgVEZfU1RBVFVTX0NPREUgPSAweDA4LAotICAgIFRGX0FDQ0VQVF9GRFMgPSAweDEwLAotfTsKLQotCi1lbnVtIHsKLSAgICBGTEFUX0JJTkRFUl9GTEFHX1BSSU9SSVRZX01BU0sgPSAweGZmLAotICAgIEZMQVRfQklOREVSX0ZMQUdfQUNDRVBUU19GRFMgPSAweDEwMCwKLX07Ci0KLWVudW0gQmluZGVyRHJpdmVyUmV0dXJuUHJvdG9jb2wgewotICAgIEJSX0VSUk9SLAotICAgIEJSX09LLAotICAgIEJSX1RSQU5TQUNUSU9OLAotICAgIEJSX1JFUExZLAotICAgIEJSX0FDUVVJUkVfUkVTVUxULAotICAgIEJSX0RFQURfUkVQTFksCi0gICAgQlJfVFJBTlNBQ1RJT05fQ09NUExFVEUsCi0gICAgQlJfSU5DUkVGUywKLSAgICBCUl9BQ1FVSVJFLAotICAgIEJSX1JFTEVBU0UsCi0gICAgQlJfREVDUkVGUywKLSAgICBCUl9BVFRFTVBUX0FDUVVJUkUsCi0gICAgQlJfTk9PUCwKLSAgICBCUl9TUEFXTl9MT09QRVIsCi0gICAgQlJfRklOSVNIRUQsCi0gICAgQlJfREVBRF9CSU5ERVIsCi0gICAgQlJfQ0xFQVJfREVBVEhfTk9USUZJQ0FUSU9OX0RPTkUsCi0gICAgQlJfRkFJTEVEX1JFUExZLAotfTsKLQotZW51bSBCaW5kZXJEcml2ZXJDb21tYW5kUHJvdG9jb2wgewotICAgIEJDX1RSQU5TQUNUSU9OLAotICAgIEJDX1JFUExZLAotICAgIEJDX0FDUVVJUkVfUkVTVUxULAotICAgIEJDX0ZSRUVfQlVGRkVSLAotICAgIEJDX0lOQ1JFRlMsCi0gICAgQkNfQUNRVUlSRSwKLSAgICBCQ19SRUxFQVNFLAotICAgIEJDX0RFQ1JFRlMsCi0gICAgQkNfSU5DUkVGU19ET05FLAotICAgIEJDX0FDUVVJUkVfRE9ORSwKLSAgICBCQ19BVFRFTVBUX0FDUVVJUkUsCi0gICAgQkNfUkVHSVNURVJfTE9PUEVSLAotICAgIEJDX0VOVEVSX0xPT1BFUiwKLSAgICBCQ19FWElUX0xPT1BFUiwKLSAgICBCQ19SRVFVRVNUX0RFQVRIX05PVElGSUNBVElPTiwKLSAgICBCQ19DTEVBUl9ERUFUSF9OT1RJRklDQVRJT04sCi0gICAgQkNfREVBRF9CSU5ERVJfRE9ORSwKLX07Ci0KLSNlbmRpZgotCi0jaWZkZWYgX19jcGx1c3BsdXMKLX0gICAvLyBuYW1lc3BhY2UgYW5kcm9pZAotI2VuZGlmCi0KLSNlbmRpZiAvLyBfQklOREVSX01PRFVMRV9IXwpkaWZmIC0tZ2l0IGEvaW5jbHVkZS9wcml2YXRlL3V0aWxzL2Z1dGV4X3N5bmNocm8uaCBiL2luY2x1ZGUvcHJpdmF0ZS91dGlscy9mdXRleF9zeW5jaHJvLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGFjMmFiMTkuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS9wcml2YXRlL3V0aWxzL2Z1dGV4X3N5bmNocm8uaAorKysgL2Rldi9udWxsCkBAIC0xLDYwICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIF9GVVRFWF9TWU5DSFJPX0gKLSNkZWZpbmUgX0ZVVEVYX1NZTkNIUk9fSAotCi0jaWZuZGVmIEhBVkVfRlVURVgKLSNlcnJvciAiSEFWRV9GVVRFWCBub3QgZGVmaW5lZCIKLSNlbmRpZgotCi0jZGVmaW5lIEZVVEVYX1dBSVRfSU5GSU5JVEUgKDApCi0KLXR5cGVkZWYgc3RydWN0IGZ1dGV4X211dGV4X3QgZnV0ZXhfbXV0ZXhfdDsKLQotc3RydWN0IGZ1dGV4X211dGV4X3QgCi17Ci0gICAgdm9sYXRpbGUgaW50IHZhbHVlOwotfTsKLQotdHlwZWRlZiBzdHJ1Y3QgZnV0ZXhfY29uZF90IGZ1dGV4X2NvbmRfdDsKLQotc3RydWN0IGZ1dGV4X2NvbmRfdCAKLXsKLSAgICB2b2xhdGlsZSBpbnQgdmFsdWU7Ci19OwotCi0KLSNpZiBfX2NwbHVzcGx1cwotZXh0ZXJuICJDIiB7Ci0jZW5kaWYKLQotdm9pZCBmdXRleF9tdXRleF9pbml0KGZ1dGV4X211dGV4X3QgKm0pOwotaW50IGZ1dGV4X211dGV4X2xvY2soZnV0ZXhfbXV0ZXhfdCAqbSwgdW5zaWduZWQgbXNlYyk7Ci12b2lkIGZ1dGV4X211dGV4X3VubG9jayhmdXRleF9tdXRleF90ICptKTsKLWludCBmdXRleF9tdXRleF90cnlsb2NrKGZ1dGV4X211dGV4X3QgKm0pOwotCi12b2lkIGZ1dGV4X2NvbmRfaW5pdChmdXRleF9jb25kX3QgKmMpOwotaW50IGZ1dGV4X2NvbmRfd2FpdChmdXRleF9jb25kX3QgKmMsIGZ1dGV4X211dGV4X3QgKm0sIHVuc2lnbmVkIG1zZWMpOwotdm9pZCBmdXRleF9jb25kX3NpZ25hbChmdXRleF9jb25kX3QgKmMpOwotdm9pZCBmdXRleF9jb25kX2Jyb2FkY2FzdChmdXRleF9jb25kX3QgKmMpOwotCi0jaWYgX19jcGx1c3BsdXMKLX0gLy8gZXh0ZXJuICJDIgotI2VuZGlmCi0KLSNlbmRpZiAvLyBfRlVURVhfU1lOQ0hST19ICi0KZGlmZiAtLWdpdCBhL2luY2x1ZGUvdWkvQ2FtZXJhLmggYi9pbmNsdWRlL3VpL0NhbWVyYS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlNTkzZmVhLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdWkvQ2FtZXJhLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxOTYgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKiBDb3B5cmlnaHQgKEMpIDIwMDggSFRDIEluYy4KLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0hBUkRXQVJFX0NBTUVSQV9ICi0jZGVmaW5lIEFORFJPSURfSEFSRFdBUkVfQ0FNRVJBX0gKLQotI2luY2x1ZGUgPHVpL0lDYW1lcmFDbGllbnQuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vKgotICogQSBzZXQgb2YgYml0IG1hc2tzIGZvciBzcGVjaWZ5aW5nIGhvdyB0aGUgcmVjZWl2ZWQgcHJldmlldyBmcmFtZXMgYXJlCi0gKiBoYW5kbGVkIGJlZm9yZSB0aGUgcHJldmlld0NhbGxiYWNrKCkgY2FsbC4KLSAqCi0gKiBUaGUgbGVhc3Qgc2lnbmlmaWNhbnQgMyBiaXRzIG9mIGFuICJpbnQiIHZhbHVlIGFyZSB1c2VkIGZvciB0aGlzIHB1cnBvc2U6Ci0gKgotICogLi4uLi4gMCAwIDAKLSAqICAgICAgIF4gXiBeCi0gKiAgICAgICB8IHwgfC0tLS0tLS0tLT4gZGV0ZXJtaW5lIHdoZXRoZXIgdGhlIGNhbGxiYWNrIGlzIGVuYWJsZWQgb3Igbm90Ci0gKiAgICAgICB8IHwtLS0tLS0tLS0tLT4gZGV0ZXJtaW5lIHdoZXRoZXIgdGhlIGNhbGxiYWNrIGlzIG9uZS1zaG90IG9yIG5vdAotICogICAgICAgfC0tLS0tLS0tLS0tLS0+IGRldGVybWluZSB3aGV0aGVyIHRoZSBmcmFtZSBpcyBjb3BpZWQgb3V0IG9yIG5vdAotICoKLSAqIFdBUk5JTkc6Ci0gKiBXaGVuIGEgZnJhbWUgaXMgc2VudCBkaXJlY3RseSB3aXRob3V0IGNvcHlpbmcsIGl0IGlzIHRoZSBmcmFtZSByZWNlaXZlcidzCi0gKiByZXNwb25zaWJsaXR5IHRvIG1ha2Ugc3VyZSB0aGF0IHRoZSBmcmFtZSBkYXRhIHdvbid0IGdldCBjb3JydXB0ZWQgYnkKLSAqIHN1YnNlcXVlbnQgcHJldmlldyBmcmFtZXMgZmlsbGVkIGJ5IHRoZSBjYW1lcmEuIFRoaXMgZmxhZyBpcyByZWNvbW1lbmRlZAotICogb25seSB3aGVuIGNvcHlpbmcgb3V0IGRhdGEgYnJpbmdzIHNpZ25pZmljYW50IHBlcmZvcm1hbmNlIHByaWNlIGFuZCB0aGUKLSAqIGhhbmRsaW5nL3Byb2Nlc3Npbmcgb2YgdGhlIHJlY2VpdmVkIGZyYW1lIGRhdGEgaXMgYWx3YXlzIGZhc3RlciB0aGFuCi0gKiB0aGUgcHJldmlldyBmcmFtZSByYXRlIHNvIHRoYXQgZGF0YSBjb3JydXB0aW9uIHdvbid0IG9jY3VyLgotICoKLSAqIEZvciBpbnN0YW5jZSwKLSAqIDEuIDB4MDAgZGlzYWJsZXMgdGhlIGNhbGxiYWNrLiBJbiB0aGlzIGNhc2UsIGNvcHkgb3V0IGFuZCBvbmUgc2hvdCBiaXRzCi0gKiAgICBhcmUgaWdub3JlZC4KLSAqIDIuIDB4MDEgZW5hYmxlcyBhIGNhbGxiYWNrIHdpdGhvdXQgY29weWluZyBvdXQgdGhlIHJlY2VpdmVkIGZyYW1lcy4gQQotICogICAgdHlwaWNhbCB1c2UgY2FzZSBpcyB0aGUgQ2FtY29yZGVyIGFwcGxpY2F0aW9uIHRvIGF2b2lkIG1ha2luZyBjb3N0bHkKLSAqICAgIGZyYW1lIGNvcGllcy4KLSAqIDMuIDB4MDUgaXMgZW5hYmxpbmcgYSBjYWxsYmFjayB3aXRoIGZyYW1lIGNvcGllZCBvdXQgcmVwZWF0ZWRseS4gQSB0eXBpY2FsCi0gKiAgICB1c2UgY2FzZSBpcyB0aGUgQ2FtZXJhIGFwcGxpY2F0aW9uLgotICogNC4gMHgwNyBpcyBlbmFibGluZyBhIGNhbGxiYWNrIHdpdGggZnJhbWUgY29waWVkIG91dCBvbmx5IG9uY2UuIEEgdHlwaWNhbCB1c2UKLSAqICAgIGNhc2UgaXMgdGhlIEJhcmNvZGUgc2Nhbm5lciBhcHBsaWNhdGlvbi4KLSAqLwotI2RlZmluZSBGUkFNRV9DQUxMQkFDS19GTEFHX0VOQUJMRV9NQVNLICAgICAgICAgICAgICAweDAxCi0jZGVmaW5lIEZSQU1FX0NBTExCQUNLX0ZMQUdfT05FX1NIT1RfTUFTSyAgICAgICAgICAgIDB4MDIKLSNkZWZpbmUgRlJBTUVfQ0FMTEJBQ0tfRkxBR19DT1BZX09VVF9NQVNLICAgICAgICAgICAgMHgwNAotCi0vLyBUeXBpY2FsIHVzZSBjYXNlcwotI2RlZmluZSBGUkFNRV9DQUxMQkFDS19GTEFHX05PT1AgICAgICAgICAgICAgICAgICAgICAweDAwCi0jZGVmaW5lIEZSQU1FX0NBTExCQUNLX0ZMQUdfQ0FNQ09SREVSICAgICAgICAgICAgICAgIDB4MDEKLSNkZWZpbmUgRlJBTUVfQ0FMTEJBQ0tfRkxBR19DQU1FUkEgICAgICAgICAgICAgICAgICAgMHgwNQotI2RlZmluZSBGUkFNRV9DQUxMQkFDS19GTEFHX0JBUkNPREVfU0NBTk5FUiAgICAgICAgICAweDA3Ci0KLWNsYXNzIElDYW1lcmFTZXJ2aWNlOwotY2xhc3MgSUNhbWVyYTsKLWNsYXNzIFN1cmZhY2U7Ci1jbGFzcyBNdXRleDsKLWNsYXNzIFN0cmluZzg7Ci0KLXR5cGVkZWYgdm9pZCAoKnNodXR0ZXJfY2FsbGJhY2spKHZvaWQgKmNvb2tpZSk7Ci10eXBlZGVmIHZvaWQgKCpmcmFtZV9jYWxsYmFjaykoY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSwgdm9pZCAqY29va2llKTsKLXR5cGVkZWYgdm9pZCAoKmF1dG9mb2N1c19jYWxsYmFjaykoYm9vbCBmb2N1c2VkLCB2b2lkICpjb29raWUpOwotdHlwZWRlZiB2b2lkICgqZXJyb3JfY2FsbGJhY2spKHN0YXR1c190IGVyciwgdm9pZCAqY29va2llKTsKLQotY2xhc3MgQ2FtZXJhIDogcHVibGljIEJuQ2FtZXJhQ2xpZW50LCBwdWJsaWMgSUJpbmRlcjo6RGVhdGhSZWNpcGllbnQKLXsKLXB1YmxpYzoKLSAgICAgICAgICAgIC8vIGNvbnN0cnVjdCBhIGNhbWVyYSBjbGllbnQgZnJvbSBhbiBleGlzdGluZyByZW1vdGUKLSAgICAgICAgICAgIENhbWVyYShjb25zdCBzcDxJQ2FtZXJhPiYgY2FtZXJhKTsKLQotICAgIHN0YXRpYyAgc3A8Q2FtZXJhPiAgY29ubmVjdCgpOwotICAgICAgICAgICAgICAgICAgICAgICAgfkNhbWVyYSgpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgaW5pdCgpOwotCi0gICAgICAgICAgICBzdGF0dXNfdCAgICByZWNvbm5lY3QoKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIGRpc2Nvbm5lY3QoKTsKLSAgICAgICAgICAgIHN0YXR1c190ICAgIGxvY2soKTsKLSAgICAgICAgICAgIHN0YXR1c190ICAgIHVubG9jaygpOwotCi0gICAgICAgICAgICBzdGF0dXNfdCAgICBnZXRTdGF0dXMoKSB7IHJldHVybiBtU3RhdHVzOyB9Ci0KLSAgICAgICAgICAgIC8vIHBhc3MgdGhlIGJ1ZmZlcmVkIElTdXJmYWNlIHRvIHRoZSBjYW1lcmEgc2VydmljZQotICAgICAgICAgICAgc3RhdHVzX3QgICAgc2V0UHJldmlld0Rpc3BsYXkoY29uc3Qgc3A8U3VyZmFjZT4mIHN1cmZhY2UpOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgc2V0UHJldmlld0Rpc3BsYXkoY29uc3Qgc3A8SVN1cmZhY2U+JiBzdXJmYWNlKTsKLQotICAgICAgICAgICAgLy8gc3RhcnQgcHJldmlldyBtb2RlLCBtdXN0IGNhbGwgc2V0UHJldmlld0Rpc3BsYXkgZmlyc3QKLSAgICAgICAgICAgIHN0YXR1c190ICAgIHN0YXJ0UHJldmlldygpOwotCi0gICAgICAgICAgICAvLyBzdG9wIHByZXZpZXcgbW9kZQotICAgICAgICAgICAgdm9pZCAgICAgICAgc3RvcFByZXZpZXcoKTsKLQotICAgICAgICAgICAgLy8gZ2V0IHByZXZpZXcgc3RhdGUKLSAgICAgICAgICAgIGJvb2wgICAgICAgIHByZXZpZXdFbmFibGVkKCk7Ci0KLSAgICAgICAgICAgIC8vIHN0YXJ0IHJlY29yZGluZyBtb2RlLCBtdXN0IGNhbGwgc2V0UHJldmlld0Rpc3BsYXkgZmlyc3QKLSAgICAgICAgICAgIHN0YXR1c190ICAgIHN0YXJ0UmVjb3JkaW5nKCk7Ci0KLSAgICAgICAgICAgIC8vIHN0b3AgcmVjb3JkaW5nIG1vZGUKLSAgICAgICAgICAgIHZvaWQgICAgICAgIHN0b3BSZWNvcmRpbmcoKTsKLQotICAgICAgICAgICAgLy8gZ2V0IHJlY29yZGluZyBzdGF0ZQotICAgICAgICAgICAgYm9vbCAgICAgICAgcmVjb3JkaW5nRW5hYmxlZCgpOwotCi0gICAgICAgICAgICAvLyByZWxlYXNlIGEgcmVjb3JkaW5nIGZyYW1lCi0gICAgICAgICAgICB2b2lkICAgICAgICByZWxlYXNlUmVjb3JkaW5nRnJhbWUoY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSk7Ci0KLSAgICAgICAgICAgIC8vIGF1dG9Gb2N1cyAtIHN0YXR1cyByZXR1cm5lZCBmcm9tIGNhbGxiYWNrCi0gICAgICAgICAgICBzdGF0dXNfdCAgICBhdXRvRm9jdXMoKTsKLQotICAgICAgICAgICAgLy8gdGFrZSBhIHBpY3R1cmUgLSBwaWN0dXJlIHJldHVybmVkIGZyb20gY2FsbGJhY2sKLSAgICAgICAgICAgIHN0YXR1c190ICAgIHRha2VQaWN0dXJlKCk7Ci0KLSAgICAgICAgICAgIC8vIHNldCBwcmV2aWV3L2NhcHR1cmUgcGFyYW1ldGVycyAtIGtleS92YWx1ZSBwYWlycwotICAgICAgICAgICAgc3RhdHVzX3QgICAgc2V0UGFyYW1ldGVycyhjb25zdCBTdHJpbmc4JiBwYXJhbXMpOwotCi0gICAgICAgICAgICAvLyBnZXQgcHJldmlldy9jYXB0dXJlIHBhcmFtZXRlcnMgLSBrZXkvdmFsdWUgcGFpcnMKLSAgICAgICAgICAgIFN0cmluZzggICAgIGdldFBhcmFtZXRlcnMoKSBjb25zdDsKLQotICAgICAgICAgICAgdm9pZCAgICAgICAgc2V0U2h1dHRlckNhbGxiYWNrKHNodXR0ZXJfY2FsbGJhY2sgY2IsIHZvaWQgKmNvb2tpZSk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICBzZXRSYXdDYWxsYmFjayhmcmFtZV9jYWxsYmFjayBjYiwgdm9pZCAqY29va2llKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIHNldEpwZWdDYWxsYmFjayhmcmFtZV9jYWxsYmFjayBjYiwgdm9pZCAqY29va2llKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIHNldFJlY29yZGluZ0NhbGxiYWNrKGZyYW1lX2NhbGxiYWNrIGNiLCB2b2lkICpjb29raWUpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgc2V0UHJldmlld0NhbGxiYWNrKGZyYW1lX2NhbGxiYWNrIGNiLCB2b2lkICpjb29raWUsIGludCBwcmV2aWV3X2NhbGxiYWNrX2ZsYWcgPSBGUkFNRV9DQUxMQkFDS19GTEFHX05PT1ApOwotICAgICAgICAgICAgdm9pZCAgICAgICAgc2V0RXJyb3JDYWxsYmFjayhlcnJvcl9jYWxsYmFjayBjYiwgdm9pZCAqY29va2llKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIHNldEF1dG9Gb2N1c0NhbGxiYWNrKGF1dG9mb2N1c19jYWxsYmFjayBjYiwgdm9pZCAqY29va2llKTsKLQotICAgIC8vIElDYW1lcmFDbGllbnQgaW50ZXJmYWNlCi0gICAgdmlydHVhbCB2b2lkICAgICAgICBzaHV0dGVyQ2FsbGJhY2soKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIHJhd0NhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBwaWN0dXJlKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIGpwZWdDYWxsYmFjayhjb25zdCBzcDxJTWVtb3J5PiYgcGljdHVyZSk7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICBwcmV2aWV3Q2FsbGJhY2soY29uc3Qgc3A8SU1lbW9yeT4mIGZyYW1lKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIGVycm9yQ2FsbGJhY2soc3RhdHVzX3QgZXJyb3IpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgYXV0b0ZvY3VzQ2FsbGJhY2soYm9vbCBmb2N1c2VkKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIHJlY29yZGluZ0NhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBmcmFtZSk7Ci0KLSAgICBzcDxJQ2FtZXJhPiAgICAgICAgIHJlbW90ZSgpOwotCi1wcml2YXRlOgotICAgICAgICAgICAgICAgICAgICAgICAgQ2FtZXJhKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB2aXJ0dWFsIHZvaWQgYmluZGVyRGllZChjb25zdCB3cDxJQmluZGVyPiYgd2hvKTsKLQotICAgICAgICAgICAgY2xhc3MgRGVhdGhOb3RpZmllcjogcHVibGljIElCaW5kZXI6OkRlYXRoUmVjaXBpZW50Ci0gICAgICAgICAgICB7Ci0gICAgICAgICAgICBwdWJsaWM6Ci0gICAgICAgICAgICAgICAgRGVhdGhOb3RpZmllcigpIHsKLSAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICB2aXJ0dWFsIHZvaWQgYmluZGVyRGllZChjb25zdCB3cDxJQmluZGVyPiYgd2hvKTsKLSAgICAgICAgICAgIH07Ci0KLSAgICAgICAgICAgIHN0YXRpYyBzcDxEZWF0aE5vdGlmaWVyPiBtRGVhdGhOb3RpZmllcjsKLQotICAgICAgICAgICAgLy8gaGVscGVyIGZ1bmN0aW9uIHRvIG9idGFpbiBjYW1lcmEgc2VydmljZSBoYW5kbGUKLSAgICAgICAgICAgIHN0YXRpYyBjb25zdCBzcDxJQ2FtZXJhU2VydmljZT4mIGdldENhbWVyYVNlcnZpY2UoKTsKLQotICAgICAgICAgICAgc3A8SUNhbWVyYT4gICAgICAgICBtQ2FtZXJhOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICBtU3RhdHVzOwotCi0gICAgICAgICAgICBzaHV0dGVyX2NhbGxiYWNrICAgIG1TaHV0dGVyQ2FsbGJhY2s7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgICptU2h1dHRlckNhbGxiYWNrQ29va2llOwotICAgICAgICAgICAgZnJhbWVfY2FsbGJhY2sgICAgICBtUmF3Q2FsbGJhY2s7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgICptUmF3Q2FsbGJhY2tDb29raWU7Ci0gICAgICAgICAgICBmcmFtZV9jYWxsYmFjayAgICAgIG1KcGVnQ2FsbGJhY2s7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgICptSnBlZ0NhbGxiYWNrQ29va2llOwotICAgICAgICAgICAgZnJhbWVfY2FsbGJhY2sgICAgICBtUHJldmlld0NhbGxiYWNrOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICAqbVByZXZpZXdDYWxsYmFja0Nvb2tpZTsKLSAgICAgICAgICAgIGZyYW1lX2NhbGxiYWNrICAgICAgbVJlY29yZGluZ0NhbGxiYWNrOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICAqbVJlY29yZGluZ0NhbGxiYWNrQ29va2llOwotICAgICAgICAgICAgZXJyb3JfY2FsbGJhY2sgICAgICBtRXJyb3JDYWxsYmFjazsKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgKm1FcnJvckNhbGxiYWNrQ29va2llOwotICAgICAgICAgICAgYXV0b2ZvY3VzX2NhbGxiYWNrICBtQXV0b0ZvY3VzQ2FsbGJhY2s7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgICptQXV0b0ZvY3VzQ2FsbGJhY2tDb29raWU7Ci0KLSAgICAgICAgICAgIGZyaWVuZCBjbGFzcyBEZWF0aE5vdGlmaWVyOwotCi0gICAgICAgICAgICBzdGF0aWMgIE11dGV4ICAgICAgICAgICAgICAgbUxvY2s7Ci0gICAgICAgICAgICBzdGF0aWMgIHNwPElDYW1lcmFTZXJ2aWNlPiAgbUNhbWVyYVNlcnZpY2U7Ci0KLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZgotCmRpZmYgLS1naXQgYS9pbmNsdWRlL3VpL0NhbWVyYUhhcmR3YXJlSW50ZXJmYWNlLmggYi9pbmNsdWRlL3VpL0NhbWVyYUhhcmR3YXJlSW50ZXJmYWNlLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGIwNjhjNTIuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91aS9DYW1lcmFIYXJkd2FyZUludGVyZmFjZS5oCisrKyAvZGV2L251bGwKQEAgLTEsMTg3ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfSEFSRFdBUkVfQ0FNRVJBX0hBUkRXQVJFX0lOVEVSRkFDRV9ICi0jZGVmaW5lIEFORFJPSURfSEFSRFdBUkVfQ0FNRVJBX0hBUkRXQVJFX0lOVEVSRkFDRV9ICi0KLSNpbmNsdWRlIDx1dGlscy9JTWVtb3J5Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvUmVmQmFzZS5oPgotI2luY2x1ZGUgPHVpL0NhbWVyYVBhcmFtZXRlcnMuaD4KLSNpbmNsdWRlIDx1aS9PdmVybGF5Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyoqIENhbGxiYWNrIGZvciBzdGFydFByZXZpZXcoKSAqLwotdHlwZWRlZiB2b2lkICgqcHJldmlld19jYWxsYmFjaykoY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSwgdm9pZCogdXNlcik7Ci0KLS8qKiBDYWxsYmFjayBmb3Igc3RhcnRSZWNvcmQoKSAqLwotdHlwZWRlZiB2b2lkICgqcmVjb3JkaW5nX2NhbGxiYWNrKShjb25zdCBzcDxJTWVtb3J5PiYgbWVtLCB2b2lkKiB1c2VyKTsKLQotLyoqIENhbGxiYWNrIGZvciB0YWtlUGljdHVyZSgpICovCi10eXBlZGVmIHZvaWQgKCpzaHV0dGVyX2NhbGxiYWNrKSh2b2lkKiB1c2VyKTsKLQotLyoqIENhbGxiYWNrIGZvciB0YWtlUGljdHVyZSgpICovCi10eXBlZGVmIHZvaWQgKCpyYXdfY2FsbGJhY2spKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0sIHZvaWQqIHVzZXIpOwotCi0vKiogQ2FsbGJhY2sgZm9yIHRha2VQaWN0dXJlKCkgKi8KLXR5cGVkZWYgdm9pZCAoKmpwZWdfY2FsbGJhY2spKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0sIHZvaWQqIHVzZXIpOwotCi0vKiogQ2FsbGJhY2sgZm9yIGF1dG9Gb2N1cygpICovCi10eXBlZGVmIHZvaWQgKCphdXRvZm9jdXNfY2FsbGJhY2spKGJvb2wgZm9jdXNlZCwgdm9pZCogdXNlcik7Ci0KLS8qKgotICogQ2FtZXJhSGFyZHdhcmVJbnRlcmZhY2UuaCBkZWZpbmVzIHRoZSBpbnRlcmZhY2UgdG8gdGhlCi0gKiBjYW1lcmEgaGFyZHdhcmUgYWJzdHJhY3Rpb24gbGF5ZXIsIHVzZWQgZm9yIHNldHRpbmcgYW5kIGdldHRpbmcKLSAqIHBhcmFtZXRlcnMsIGxpdmUgcHJldmlld2luZywgYW5kIHRha2luZyBwaWN0dXJlcy4KLSAqCi0gKiBJdCBpcyBhIHJlZmVyZW5jZWQgY291bnRlZCBpbnRlcmZhY2Ugd2l0aCBSZWZCYXNlIGFzIGl0cyBiYXNlIGNsYXNzLgotICogQ2FtZXJhU2VydmljZSBjYWxscyBvcGVuQ2FtZXJhSGFyZHdhcmUoKSB0byByZXRyaWV2ZSBhIHN0cm9uZyBwb2ludGVyIHRvIHRoZQotICogaW5zdGFuY2Ugb2YgdGhpcyBpbnRlcmZhY2UgYW5kIG1heSBiZSBjYWxsZWQgbXVsdGlwbGUgdGltZXMuIFRoZQotICogZm9sbG93aW5nIHN0ZXBzIGRlc2NyaWJlIGEgdHlwaWNhbCBzZXF1ZW5jZToKLSAqCi0gKiAgIC0jIEFmdGVyIENhbWVyYVNlcnZpY2UgY2FsbHMgb3BlbkNhbWVyYUhhcmR3YXJlKCksIGdldFBhcmFtZXRlcnMoKSBhbmQKLSAqICAgICAgc2V0UGFyYW1ldGVycygpIGFyZSB1c2VkIHRvIGluaXRpYWxpemUgdGhlIGNhbWVyYSBpbnN0YW5jZS4KLSAqICAgICAgQ2FtZXJhU2VydmljZSBjYWxscyBnZXRQcmV2aWV3SGVhcCgpIHRvIGVzdGFibGlzaCBhY2Nlc3MgdG8gdGhlCi0gKiAgICAgIHByZXZpZXcgaGVhcCBzbyBpdCBjYW4gYmUgcmVnaXN0ZXJlZCB3aXRoIFN1cmZhY2VGbGluZ2VyIGZvcgotICogICAgICBlZmZpY2llbnQgZGlzcGxheSB1cGRhdGluZyB3aGlsZSBpbiBwcmV2aWV3IG1vZGUuCi0gKiAgIC0jIHN0YXJ0UHJldmlldygpIGlzIGNhbGxlZCwgd2hpY2ggaXMgcGFzc2VkIGEgcHJldmlld19jYWxsYmFjaygpCi0gKiAgICAgIGZ1bmN0aW9uIGFuZCBhIHVzZXIgcGFyYW1ldGVyLiBUaGUgY2FtZXJhIGluc3RhbmNlIHRoZW4gcGVyaW9kaWNhbGx5Ci0gKiAgICAgIGNhbGxzIHByZXZpZXdfY2FsbGJhY2soKSBlYWNoIHRpbWUgYSBuZXcgcHJldmlldyBmcmFtZSBpcyBhdmFpbGFibGUuCi0gKiAgICAgIFRoZSBjYWxsYmFjayByb3V0aW5lIGhhcyB0d28gcGFyYW1ldGVyczogdGhlIGZpcnN0IGlzIGEgcG9pbnRlciB0bwotICogICAgICB0aGUgSU1lbW9yeSBjb250YWluaW5nIHRoZSBmcmFtZSBhbmQgdGhlIHNlY29uZCBhIHVzZXIgcGFyYW1ldGVyLiBJZgotICogICAgICB0aGUgcHJldmlld19jYWxsYmFjayBjb2RlIG5lZWRzIHRvIHVzZSB0aGlzIG1lbW9yeSBhZnRlciByZXR1cm5pbmcsCi0gKiAgICAgIGl0IG11c3QgY29weSB0aGUgZGF0YS4KLSAqCi0gKiBQcmlvciB0byB0YWtpbmcgYSBwaWN0dXJlLCBDYW1lcmFTZXJ2aWNlIGNhbGxzIGF1dG9mb2N1cygpIHdpdGgKLSAqIGF1dG9mb2N1c19jYWxsYmFjaygpIGFuZCBhIHVzZXIgcGFyYW1ldGVyLiBXaGVuIGF1dG8gZm9jdXNpbmcgaGFzCi0gKiBjb21wbGV0ZWQsIHRoZSBjYW1lcmEgaW5zdGFuY2UgY2FsbHMgYXV0b2ZvY3VzX2NhbGxiYWNrKCksIHdoaWNoIGluZm9ybXMKLSAqIHRoZSBhcHBsaWNhdGlvbiB3aGV0aGVyIGZvY3VzaW5nIHdhcyBzdWNjZXNzZnVsLiBUaGUgY2FtZXJhIGluc3RhbmNlCi0gKiBvbmx5IGNhbGxzIGF1dG9mb2N1c19jYWxsYmFjaygpIG9uY2UgYW5kIGl0IGlzIHVwIHRvIHRoZSBhcHBsaWNhdGlvbiB0bwotICogY2FsbCBhdXRvRm9jdXMoKSBhZ2FpbiBpZiByZWZvY3VzaW5nIGlzIGRlc2lyZWQuCi0gKgotICogQ2FtZXJhU2VydmljZSBjYWxscyB0YWtlUGljdHVyZSgpIHRvIHJlcXVlc3QgdGhlIGNhbWVyYSBpbnN0YW5jZSB0YWtlIGEKLSAqIHBpY3R1cmUuIFRoaXMgbWV0aG9kIGhhcyB0d28gY2FsbGJhY2tzOiByYXdfY2FsbGJhY2soKSBhbmQganBlZ19jYWxsYmFjaygpLgotICogV2hlbiB0aGUgcmF3IGltYWdlIGlzIGF2YWlsYWJsZSwgcmF3X2NhbGxiYWNrKCkgaXMgY2FsbGVkIHdpdGggYSBwb2ludGVyCi0gKiB0byB0aGUgSU1lbW9yeSBjb250YWluaW5nIHRoZSByYXcgaW1hZ2UuIFdoZW4gdGhlIGpwZWcgaW1hZ2UgaXMgYXZhaWxhYmxlLAotICoganBlZ19jYWxsYmFjaygpIGlzIGNhbGxlZCB3aXRoIGEgcG9pbnRlciB0byB0aGUgSU1lbW9yeSBjb250YWluaW5nIHRoZQotICoganBlZyBpbWFnZS4gQXMgd2l0aCBwcmV2aWV3X2NhbGxiYWNrKCksIHRoZSBtZW1vcnkgbXVzdCBiZSBjb3BpZWQgaWYgaXQncwotICogbmVlZGVkIGFmdGVyIHJldHVybmluZy4KLSAqLwotY2xhc3MgQ2FtZXJhSGFyZHdhcmVJbnRlcmZhY2UgOiBwdWJsaWMgdmlydHVhbCBSZWZCYXNlIHsKLXB1YmxpYzoKLSAgICB2aXJ0dWFsIH5DYW1lcmFIYXJkd2FyZUludGVyZmFjZSgpIHsgfQotCi0gICAgLyoqIFJldHVybiB0aGUgSU1lbW9yeUhlYXAgZm9yIHRoZSBwcmV2aWV3IGltYWdlIGhlYXAgKi8KLSAgICB2aXJ0dWFsIHNwPElNZW1vcnlIZWFwPiAgICAgICAgIGdldFByZXZpZXdIZWFwKCkgY29uc3QgPSAwOwotCi0gICAgLyoqCi0gICAgICogU3RhcnQgcHJldmlldyBtb2RlLiBXaGVuIGEgcHJldmlldyBpbWFnZSBpcyBhdmFpbGFibGUKLSAgICAgKiBwcmV2aWV3X2NhbGxiYWNrIGlzIGNhbGxlZCB3aXRoIHRoZSB1c2VyIHBhcmFtZXRlci4gVGhlCi0gICAgICogY2FsbCBiYWNrIHBhcmFtZXRlciBtYXkgYmUgbnVsbC4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHN0YXJ0UHJldmlldyhwcmV2aWV3X2NhbGxiYWNrIGNiLCB2b2lkKiB1c2VyKSA9IDA7Ci0gICAgLyoqCi0gICAgICogT25seSB1c2VkIGlmIG92ZXJsYXlzIGFyZSB1c2VkIGZvciBjYW1lcmEgcHJldmlldy4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIGJvb2wgdXNlT3ZlcmxheSgpIHtyZXR1cm4gZmFsc2U7fQotICAgIHZpcnR1YWwgc3RhdHVzX3Qgc2V0T3ZlcmxheShjb25zdCBzcDxPdmVybGF5PiAmb3ZlcmxheSkge3JldHVybiBCQURfVkFMVUU7fQotCi0gICAgLyoqCi0gICAgICogU3RvcCBhIHByZXZpb3VzbHkgc3RhcnRlZCBwcmV2aWV3LgotICAgICAqLwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgc3RvcFByZXZpZXcoKSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBSZXR1cm5zIHRydWUgaWYgcHJldmlldyBpcyBlbmFibGVkLgotICAgICAqLwotICAgIHZpcnR1YWwgYm9vbCAgICAgICAgcHJldmlld0VuYWJsZWQoKSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBTdGFydCByZWNvcmQgbW9kZS4gV2hlbiBhIHJlY29yZCBpbWFnZSBpcyBhdmFpbGFibGUgcmVjb3JkaW5nX2NhbGxiYWNrKCkKLSAgICAgKiBpcyBjYWxsZWQgd2l0aCB0aGUgdXNlciBwYXJhbWV0ZXIuICBFdmVyeSByZWNvcmQgZnJhbWUgbXVzdCBiZSByZWxlYXNlZAotICAgICAqIGJ5IGNhbGxpbmcgcmVsZWFzZVJlY29yZGluZ0ZyYW1lKCkuCi0gICAgICovCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzdGFydFJlY29yZGluZyhyZWNvcmRpbmdfY2FsbGJhY2sgY2IsIHZvaWQqIHVzZXIpID0gMDsKLQotICAgIC8qKgotICAgICAqIFN0b3AgYSBwcmV2aW91c2x5IHN0YXJ0ZWQgcmVjb3JkaW5nLgotICAgICAqLwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgc3RvcFJlY29yZGluZygpID0gMDsKLQotICAgIC8qKgotICAgICAqIFJldHVybnMgdHJ1ZSBpZiByZWNvcmRpbmcgaXMgZW5hYmxlZC4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIGJvb2wgICAgICAgIHJlY29yZGluZ0VuYWJsZWQoKSA9IDA7Ci0gICAgCi0gICAgLyoqCi0gICAgICogUmVsZWFzZSBhIHJlY29yZCBmcmFtZSBwcmV2aW91c2x5IHJldHVybmVkIGJ5IHRoZSByZWNvcmRpbmdfY2FsbGJhY2soKQotICAgICAqIHBhc3NlZCB0byBzdGFydFJlY29yZCgpLgotICAgICAqLwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgcmVsZWFzZVJlY29yZGluZ0ZyYW1lKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0pID0gMDsKLQotICAgIC8qKgotICAgICAqIFN0YXJ0IGF1dG8gZm9jdXMsIHRoZSBjYWxsYmFjayByb3V0aW5lIGlzIGNhbGxlZAotICAgICAqIG9uY2Ugd2hlbiBmb2N1c2luZyBpcyBjb21wbGV0ZS4gYXV0b0ZvY3VzKCkgd2lsbAotICAgICAqIGJlIGNhbGxlZCBhZ2FpbiBpZiBhbm90aGVyIGF1dG8gZm9jdXMgaXMgbmVlZGVkLgotICAgICAqLwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgYXV0b0ZvY3VzKGF1dG9mb2N1c19jYWxsYmFjaywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiB1c2VyKSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBUYWtlIGEgcGljdHVyZS4gVGhlIHJhd19jYWxsYmFjayBpcyBjYWxsZWQgd2hlbgotICAgICAqIHRoZSB1bmNvbXByZXNzZWQgaW1hZ2UgaXMgYXZhaWxhYmxlLiBUaGUganBlZ19jYWxsYmFjawotICAgICAqIGlzIGNhbGxlZCB3aGVuIHRoZSBjb21wcmVzc2VkIGltYWdlIGlzIGF2YWlsYWJsZS4gVGhlc2UKLSAgICAgKiBjYWxsIGJhY2tzIG1heSBiZSBudWxsLiBUaGUgdXNlciBwYXJhbWV0ZXIgaXMgcGFzc2VkCi0gICAgICogdG8gZWFjaCBvZiB0aGUgY2FsbCBiYWNrIHJvdXRpbmVzLgotICAgICAqLwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgdGFrZVBpY3R1cmUoc2h1dHRlcl9jYWxsYmFjaywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhd19jYWxsYmFjaywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGpwZWdfY2FsbGJhY2ssCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiB1c2VyKSA9IDA7Ci0KLSAgICAvKioKLSAgICAgKiBDYW5jZWwgYSBwaWN0dXJlIHRoYXQgd2FzIHN0YXJ0ZWQgd2l0aCB0YWtlUGljdHVyZS4gIFlvdSBtYXkgY2FuY2VsIGFueQotICAgICAqIG9mIHRoZSBzaHV0dGVyLCByYXcsIG9yIGpwZWcgY2FsbGJhY2tzLiAgQ2FsbGluZyB0aGlzIG1ldGhvZCB3aGVuIG5vCi0gICAgICogcGljdHVyZSBpcyBiZWluZyB0YWtlbiBpcyBhIG5vLW9wLgotICAgICAqLwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgY2FuY2VsUGljdHVyZShib29sIGNhbmNlbF9zaHV0dGVyLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBib29sIGNhbmNlbF9yYXcsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2wgY2FuY2VsX2pwZWcpID0gMDsKLQotICAgIC8qKiBTZXQgdGhlIGNhbWVyYSBwYXJhbWV0ZXJzLiAqLwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0UGFyYW1ldGVycyhjb25zdCBDYW1lcmFQYXJhbWV0ZXJzJiBwYXJhbXMpID0gMDsKLQotICAgIC8qKiBSZXR1cm4gdGhlIGNhbWVyYSBwYXJhbWV0ZXJzLiAqLwotICAgIHZpcnR1YWwgQ2FtZXJhUGFyYW1ldGVycyAgZ2V0UGFyYW1ldGVycygpIGNvbnN0ID0gMDsKLQotICAgIC8qKgotICAgICAqIFJlbGVhc2UgdGhlIGhhcmR3YXJlIHJlc291cmNlcyBvd25lZCBieSB0aGlzIG9iamVjdC4gIE5vdGUgdGhhdCB0aGlzIGlzCi0gICAgICogKm5vdCogZG9uZSBpbiB0aGUgZGVzdHJ1Y3Rvci4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHZvaWQgcmVsZWFzZSgpID0gMDsKLSAgICAKLSAgICAvKioKLSAgICAgKiBEdW1wIHN0YXRlIG9mIHRoZSBjYW1lcmEgaGFyZHdhcmUKLSAgICAgKi8KLSAgICB2aXJ0dWFsIHN0YXR1c190IGR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKSBjb25zdCA9IDA7Ci19OwotCi0vKiogZmFjdG9yeSBmdW5jdGlvbiB0byBpbnN0YW50aWF0ZSBhIGNhbWVyYSBoYXJkd2FyZSBvYmplY3QgKi8KLWV4dGVybiAiQyIgc3A8Q2FtZXJhSGFyZHdhcmVJbnRlcmZhY2U+IG9wZW5DYW1lcmFIYXJkd2FyZSgpOwotCi19OyAgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmCmRpZmYgLS1naXQgYS9pbmNsdWRlL3VpL0NhbWVyYVBhcmFtZXRlcnMuaCBiL2luY2x1ZGUvdWkvQ2FtZXJhUGFyYW1ldGVycy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5Y2ExODA2Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdWkvQ2FtZXJhUGFyYW1ldGVycy5oCisrKyAvZGV2L251bGwKQEAgLTEsNzkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9IQVJEV0FSRV9DQU1FUkFfUEFSQU1FVEVSU19ICi0jZGVmaW5lIEFORFJPSURfSEFSRFdBUkVfQ0FNRVJBX1BBUkFNRVRFUlNfSAotCi0jaW5jbHVkZSA8dXRpbHMvS2V5ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgQ2FtZXJhUGFyYW1ldGVycwotewotcHVibGljOgotICAgIENhbWVyYVBhcmFtZXRlcnMoKTsKLSAgICBDYW1lcmFQYXJhbWV0ZXJzKGNvbnN0IFN0cmluZzggJnBhcmFtcykgeyB1bmZsYXR0ZW4ocGFyYW1zKTsgfQotICAgIH5DYW1lcmFQYXJhbWV0ZXJzKCk7Ci0KLSAgICBlbnVtIHsKLSAgICAgICAgQ0FNRVJBX09SSUVOVEFUSU9OX1VOS05PV04gPSAwLAotICAgICAgICBDQU1FUkFfT1JJRU5UQVRJT05fUE9SVFJBSVQgPSAxLAotICAgICAgICBDQU1FUkFfT1JJRU5UQVRJT05fTEFORFNDQVBFID0gMiwKLSAgICB9OwotCi0gICAgU3RyaW5nOCBmbGF0dGVuKCkgY29uc3Q7Ci0gICAgdm9pZCB1bmZsYXR0ZW4oY29uc3QgU3RyaW5nOCAmcGFyYW1zKTsKLQotICAgIHZvaWQgc2V0KGNvbnN0IGNoYXIgKmtleSwgY29uc3QgY2hhciAqdmFsdWUpOwotICAgIHZvaWQgc2V0KGNvbnN0IGNoYXIgKmtleSwgaW50IHZhbHVlKTsKLSAgICBjb25zdCBjaGFyICpnZXQoY29uc3QgY2hhciAqa2V5KSBjb25zdDsKLSAgICBpbnQgZ2V0SW50KGNvbnN0IGNoYXIgKmtleSkgY29uc3Q7Ci0KLSAgICAvKiBwcmV2aWV3LXNpemU9MTc2eDE0NCAqLwotICAgIHZvaWQgc2V0UHJldmlld1NpemUoaW50IHdpZHRoLCBpbnQgaGVpZ2h0KTsKLSAgICB2b2lkIGdldFByZXZpZXdTaXplKGludCAqd2lkdGgsIGludCAqaGVpZ2h0KSBjb25zdDsKLQotICAgIC8qIHByZXZpZXctZnBzPTE1ICovCi0gICAgdm9pZCBzZXRQcmV2aWV3RnJhbWVSYXRlKGludCBmcHMpOwotICAgIGludCBnZXRQcmV2aWV3RnJhbWVSYXRlKCkgY29uc3Q7Ci0KLSAgICAvKiBwcmV2aWV3LWZvcm1hdD1yZ2I1NjV8eXV2NDIyICovCi0gICAgdm9pZCBzZXRQcmV2aWV3Rm9ybWF0KGNvbnN0IGNoYXIgKmZvcm1hdCk7Ci0gICAgY29uc3QgY2hhciAqZ2V0UHJldmlld0Zvcm1hdCgpIGNvbnN0OwotCi0gICAgLyogcGljdHVyZS1zaXplPTEwMjR4NzY4ICovCi0gICAgdm9pZCBzZXRQaWN0dXJlU2l6ZShpbnQgd2lkdGgsIGludCBoZWlnaHQpOwotICAgIHZvaWQgZ2V0UGljdHVyZVNpemUoaW50ICp3aWR0aCwgaW50ICpoZWlnaHQpIGNvbnN0OwotCi0gICAgLyogcGljdHVyZS1mb3JtYXQ9eXV2NDIyfGpwZWcgKi8KLSAgICB2b2lkIHNldFBpY3R1cmVGb3JtYXQoY29uc3QgY2hhciAqZm9ybWF0KTsKLSAgICBjb25zdCBjaGFyICpnZXRQaWN0dXJlRm9ybWF0KCkgY29uc3Q7Ci0KLSAgICBpbnQgZ2V0T3JpZW50YXRpb24oKSBjb25zdDsKLSAgICB2b2lkIHNldE9yaWVudGF0aW9uKGludCBvcmllbnRhdGlvbik7Ci0KLSAgICB2b2lkIGR1bXAoKSBjb25zdDsKLSAgICBzdGF0dXNfdCBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykgY29uc3Q7Ci0KLXByaXZhdGU6Ci0gICAgRGVmYXVsdEtleWVkVmVjdG9yPFN0cmluZzgsU3RyaW5nOD4gICAgbU1hcDsKLX07Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmCmRpZmYgLS1naXQgYS9pbmNsdWRlL3VpL0Rpc3BsYXlJbmZvLmggYi9pbmNsdWRlL3VpL0Rpc3BsYXlJbmZvLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM0MTllZmUuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91aS9EaXNwbGF5SW5mby5oCisrKyAvZGV2L251bGwKQEAgLTEsNDMgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLQotI2lmbmRlZiBBTkRST0lEX1VJX0RJU1BMQVlfSU5GT19ICi0jZGVmaW5lIEFORFJPSURfVUlfRElTUExBWV9JTkZPX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dWkvUGl4ZWxGb3JtYXQuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1zdHJ1Y3QgRGlzcGxheUluZm8gewotICAgIHVpbnQzMl90ICAgICAgICAgICAgdzsKLSAgICB1aW50MzJfdCAgICAgICAgICAgIGg7Ci0gICAgUGl4ZWxGb3JtYXRJbmZvICAgICBwaXhlbEZvcm1hdEluZm87Ci0gICAgdWludDhfdCAgICAgICAgICAgICBvcmllbnRhdGlvbjsKLSAgICB1aW50OF90ICAgICAgICAgICAgIHJlc2VydmVkWzNdOwotICAgIGZsb2F0ICAgICAgICAgICAgICAgZnBzOwotICAgIGZsb2F0ICAgICAgICAgICAgICAgZGVuc2l0eTsKLSAgICBmbG9hdCAgICAgICAgICAgICAgIHhkcGk7Ci0gICAgZmxvYXQgICAgICAgICAgICAgICB5ZHBpOwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfQ09NUE9TRVJfRElTUExBWV9JTkZPX0gKLQpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91aS9FR0xEaXNwbGF5U3VyZmFjZS5oIGIvaW5jbHVkZS91aS9FR0xEaXNwbGF5U3VyZmFjZS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBhOGI1ODUzLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdWkvRUdMRGlzcGxheVN1cmZhY2UuaAorKysgL2Rldi9udWxsCkBAIC0xLDg2ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfRUdMX0RJU1BMQVlfU1VSRkFDRV9ICi0jZGVmaW5lIEFORFJPSURfRUdMX0RJU1BMQVlfU1VSRkFDRV9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL1RpbWVycy5oPgotCi0jaW5jbHVkZSA8dWkvRUdMTmF0aXZlU3VyZmFjZS5oPgotCi0jaW5jbHVkZSA8cGl4ZWxmbGluZ2VyL3BpeGVsZmxpbmdlci5oPgotI2luY2x1ZGUgPGxpbnV4L2ZiLmg+Ci0KLSNpbmNsdWRlIDxFR0wvZWdsLmg+Ci0KLXN0cnVjdCBjb3B5Yml0X2RldmljZV90Owotc3RydWN0IGNvcHliaXRfaW1hZ2VfdDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgUmVnaW9uOwotY2xhc3MgUmVjdDsKLQotY2xhc3MgRUdMRGlzcGxheVN1cmZhY2UgOiBwdWJsaWMgRUdMTmF0aXZlU3VyZmFjZTxFR0xEaXNwbGF5U3VyZmFjZT4KLXsKLXB1YmxpYzoKLSAgICBFR0xEaXNwbGF5U3VyZmFjZSgpOwotICAgIH5FR0xEaXNwbGF5U3VyZmFjZSgpOwotICAgIAotICAgIGludDMyX3QgZ2V0UGFnZUZsaXBDb3VudCgpIGNvbnN0OwotICAgIHZvaWQgICAgY29weUZyb250VG9CYWNrKGNvbnN0IFJlZ2lvbiYgY29weWJhY2spOwotICAgIHZvaWQgICAgY29weUZyb250VG9JbWFnZShjb25zdCBjb3B5Yml0X2ltYWdlX3QmIGRzdCk7Ci0gICAgdm9pZCAgICBjb3B5QmFja1RvSW1hZ2UoY29uc3QgY29weWJpdF9pbWFnZV90JiBkc3QpOwotICAgIAotICAgIHZvaWQgICAgICAgIHNldFN3YXBSZWN0YW5nbGUoaW50IGwsIGludCB0LCBpbnQgdywgaW50IGgpOwotCi1wcml2YXRlOgotICAgIHN0YXRpYyB2b2lkICAgICAgICAgaG9va19pbmNSZWYoTmF0aXZlV2luZG93VHlwZSB3aW5kb3cpOwotICAgIHN0YXRpYyB2b2lkICAgICAgICAgaG9va19kZWNSZWYoTmF0aXZlV2luZG93VHlwZSB3aW5kb3cpOwotICAgIHN0YXRpYyB1aW50MzJfdCAgICAgaG9va19zd2FwQnVmZmVycyhOYXRpdmVXaW5kb3dUeXBlIHdpbmRvdyk7Ci0gICAgIAotICAgICAgICAgICAgdWludDMyX3QgICAgc3dhcEJ1ZmZlcnMoKTsKLQotICAgICAgICAgICAgc3RhdHVzX3QgICAgbWFwRnJhbWVCdWZmZXIoKTsKLQotICAgICAgICAgICAgZW51bSB7Ci0gICAgICAgICAgICAgICAgUEFHRV9GTElQID0gMHgwMDAwMDAwMQotICAgICAgICAgICAgfTsKLSAgICBHR0xTdXJmYWNlICAgICAgICAgIG1GYlsyXTsKLSAgICBpbnQgICAgICAgICAgICAgICAgIG1JbmRleDsKLSAgICB1aW50MzJfdCAgICAgICAgICAgIG1GbGFnczsKLSAgICBzaXplX3QgICAgICAgICAgICAgIG1TaXplOwotICAgIGZiX3Zhcl9zY3JlZW5pbmZvICAgbUluZm87Ci0gICAgZmJfZml4X3NjcmVlbmluZm8gICBtRmluZm87Ci0gICAgaW50MzJfdCAgICAgICAgICAgICBtUGFnZUZsaXBDb3VudDsKLSAgICBuc2Vjc190ICAgICAgICAgICAgIG1UaW1lOwotICAgIGludDMyX3QgICAgICAgICAgICAgbVN3YXBDb3VudDsKLSAgICBuc2Vjc190ICAgICAgICAgICAgIG1TbGVlcDsKLSAgICB1aW50MzJfdCAgICAgICAgICAgIG1GZWF0dXJlRmxhZ3M7Ci0gICAgY29weWJpdF9kZXZpY2VfdCogICBtQmxpdEVuZ2luZTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLy8gQU5EUk9JRF9FR0xfRElTUExBWV9TVVJGQUNFX0gKLQpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91aS9FR0xOYXRpdmVTdXJmYWNlLmggYi9pbmNsdWRlL3VpL0VHTE5hdGl2ZVN1cmZhY2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzk2NGU3Yy4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3VpL0VHTE5hdGl2ZVN1cmZhY2UuaAorKysgL2Rldi9udWxsCkBAIC0xLDU1ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfRUdMX05BVElWRV9TVVJGQUNFX0gKLSNkZWZpbmUgQU5EUk9JRF9FR0xfTkFUSVZFX1NVUkZBQ0VfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDxjdXRpbHMvYXRvbWljLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUmVmQmFzZS5oPgotCi0jaW5jbHVkZSA8RUdML2VnbG5hdGl2ZXMuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdGVtcGxhdGUgPGNsYXNzIFRZUEU+Ci1jbGFzcyBFR0xOYXRpdmVTdXJmYWNlIDogcHVibGljIGVnbF9uYXRpdmVfd2luZG93X3QsIHB1YmxpYyBMaWdodFJlZkJhc2U8VFlQRT4KLXsKLXB1YmxpYzoKLSAgICBFR0xOYXRpdmVTdXJmYWNlKCkgeyAKLSAgICAgICAgbWVtc2V0KGVnbF9uYXRpdmVfd2luZG93X3Q6OnJlc2VydmVkLCAwLCAKLSAgICAgICAgICAgICAgICBzaXplb2YoZWdsX25hdGl2ZV93aW5kb3dfdDo6cmVzZXJ2ZWQpKTsKLSAgICAgICAgbWVtc2V0KGVnbF9uYXRpdmVfd2luZG93X3Q6OnJlc2VydmVkX3Byb2MsIDAsIAotICAgICAgICAgICAgICAgIHNpemVvZihlZ2xfbmF0aXZlX3dpbmRvd190OjpyZXNlcnZlZF9wcm9jKSk7Ci0gICAgICAgIG1lbXNldChlZ2xfbmF0aXZlX3dpbmRvd190OjpvZW0sIDAsIAotICAgICAgICAgICAgICAgIHNpemVvZihlZ2xfbmF0aXZlX3dpbmRvd190OjpvZW0pKTsKLSAgICB9Ci1wcm90ZWN0ZWQ6Ci0gICAgRUdMTmF0aXZlU3VyZmFjZSYgb3BlcmF0b3IgPSAoY29uc3QgRUdMTmF0aXZlU3VyZmFjZSYgcmhzKTsKLSAgICBFR0xOYXRpdmVTdXJmYWNlKGNvbnN0IEVHTE5hdGl2ZVN1cmZhY2UmIHJocyk7Ci0gICAgaW5saW5lIH5FR0xOYXRpdmVTdXJmYWNlKCkgeyB9OwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNlbmRpZiAvLyBBTkRST0lEX0VHTF9TVVJGQUNFX0gKLQpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91aS9FR0xOYXRpdmVXaW5kb3dTdXJmYWNlLmggYi9pbmNsdWRlL3VpL0VHTE5hdGl2ZVdpbmRvd1N1cmZhY2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzQ5NDIzNC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3VpL0VHTE5hdGl2ZVdpbmRvd1N1cmZhY2UuaAorKysgL2Rldi9udWxsCkBAIC0xLDU5ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfRUdMX05BVElWRV9XSU5ET1dfU1VSRkFDRV9ICi0jZGVmaW5lIEFORFJPSURfRUdMX05BVElWRV9XSU5ET1dfU1VSRkFDRV9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDx1aS9FR0xOYXRpdmVTdXJmYWNlLmg+Ci0jaW5jbHVkZSA8RUdML2VnbC5oPgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBTdXJmYWNlOwotCi1jbGFzcyBFR0xOYXRpdmVXaW5kb3dTdXJmYWNlIDogcHVibGljIEVHTE5hdGl2ZVN1cmZhY2U8RUdMTmF0aXZlV2luZG93U3VyZmFjZT4KLXsKLXB1YmxpYzoKLSAgICBFR0xOYXRpdmVXaW5kb3dTdXJmYWNlKGNvbnN0IHNwPFN1cmZhY2U+JiBzdXJmYWNlKTsKLSAgICB+RUdMTmF0aXZlV2luZG93U3VyZmFjZSgpOwotCi0gICAgdm9pZCAgICAgICAgc2V0U3dhcFJlY3RhbmdsZShpbnQgbCwgaW50IHQsIGludCB3LCBpbnQgaCk7Ci0KLXByaXZhdGU6Ci0gICAgc3RhdGljIHZvaWQgICAgICAgICBob29rX2luY1JlZihOYXRpdmVXaW5kb3dUeXBlIHdpbmRvdyk7Ci0gICAgc3RhdGljIHZvaWQgICAgICAgICBob29rX2RlY1JlZihOYXRpdmVXaW5kb3dUeXBlIHdpbmRvdyk7Ci0gICAgc3RhdGljIHVpbnQzMl90ICAgICBob29rX3N3YXBCdWZmZXJzKE5hdGl2ZVdpbmRvd1R5cGUgd2luZG93KTsKLSAgICBzdGF0aWMgdm9pZCAgICAgICAgIGhvb2tfY29ubmVjdChOYXRpdmVXaW5kb3dUeXBlIHdpbmRvdyk7Ci0gICAgc3RhdGljIHZvaWQgICAgICAgICBob29rX2Rpc2Nvbm5lY3QoTmF0aXZlV2luZG93VHlwZSB3aW5kb3cpOwotCi0gICAgICAgICAgICB1aW50MzJfdCAgICBzd2FwQnVmZmVycygpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgY29ubmVjdCgpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgZGlzY29ubmVjdCgpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBzcDxTdXJmYWNlPiBtU3VyZmFjZTsKLSAgICAgICAgICAgIGJvb2wgICAgICAgIG1Db25uZWN0ZWQ7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2VuZGlmIC8vIEFORFJPSURfRUdMX05BVElWRV9XSU5ET1dfU1VSRkFDRV9ICi0KZGlmZiAtLWdpdCBhL2luY2x1ZGUvdWkvRXZlbnRIdWIuaCBiL2luY2x1ZGUvdWkvRXZlbnRIdWIuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzg0OGQ4Yy4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3VpL0V2ZW50SHViLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxNDUgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0jaWZuZGVmIF9SVU5USU1FX0VWRU5UX0hVQl9ICi0jZGVmaW5lIF9SVU5USU1FX0VWRU5UX0hVQl9ICi0KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotI2luY2x1ZGUgPHV0aWxzLmg+Ci0KLSNpbmNsdWRlIDxsaW51eC9pbnB1dC5oPgotCi1zdHJ1Y3QgcG9sbGZkOwotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIEtleUxheW91dE1hcDsKLQotLyoKLSAqIEdyYW5kIENlbnRyYWwgU3RhdGlvbiBmb3IgZXZlbnRzLiAgV2l0aCBhIHNpbmdsZSBjYWxsIHRvIHdhaXRFdmVudCgpCi0gKiB5b3UgY2FuIHdhaXQgZm9yOgotICogIC0gaW5wdXQgZXZlbnRzIGZyb20gdGhlIGtleXBhZCBvZiBhIHJlYWwgZGV2aWNlCi0gKiAgLSBpbnB1dCBldmVudHMgYW5kIG1ldGEtZXZlbnRzIChlLmcuICJxdWl0IikgZnJvbSB0aGUgc2ltdWxhdG9yCi0gKiAgLSBzeW50aGV0aWMgZXZlbnRzIGZyb20gdGhlIHJ1bnRpbWUgKGUuZy4gIlVSTCBmZXRjaCBjb21wbGV0ZWQiKQotICogIC0gcmVhbCBvciBmb3JnZWQgInZzeW5jIiBldmVudHMKLSAqCi0gKiBEbyBub3QgaW5zdGFudGlhdGUgdGhpcyBjbGFzcy4gIEluc3RlYWQsIGNhbGwgc3RhcnRVcCgpLgotICovCi1jbGFzcyBFdmVudEh1YiA6IHB1YmxpYyBSZWZCYXNlCi17Ci1wdWJsaWM6Ci0gICAgRXZlbnRIdWIoKTsKLSAgICAKLSAgICBzdGF0dXNfdCBlcnJvckNoZWNrKCkgY29uc3Q7Ci0gICAgCi0gICAgLy8gYml0IGZpZWxkcyBmb3IgY2xhc3NlcyBvZiBkZXZpY2VzLgotICAgIGVudW0gewotICAgICAgICBDTEFTU19LRVlCT0FSRCAgICAgID0gMHgwMDAwMDAwMSwKLSAgICAgICAgQ0xBU1NfQUxQSEFLRVkgICAgICA9IDB4MDAwMDAwMDIsCi0gICAgICAgIENMQVNTX1RPVUNIU0NSRUVOICAgPSAweDAwMDAwMDA0LAotICAgICAgICBDTEFTU19UUkFDS0JBTEwgICAgID0gMHgwMDAwMDAwOAotICAgIH07Ci0gICAgdWludDMyX3QgZ2V0RGV2aWNlQ2xhc3NlcyhpbnQzMl90IGRldmljZUlkKSBjb25zdDsKLSAgICAKLSAgICBTdHJpbmc4IGdldERldmljZU5hbWUoaW50MzJfdCBkZXZpY2VJZCkgY29uc3Q7Ci0gICAgCi0gICAgaW50IGdldEFic29sdXRlSW5mbyhpbnQzMl90IGRldmljZUlkLCBpbnQgYXhpcywgaW50ICpvdXRNaW5WYWx1ZSwKLSAgICAgICAgICAgIGludCogb3V0TWF4VmFsdWUsIGludCogb3V0RmxhdCwgaW50KiBvdXRGdXp6KSBjb25zdDsKLSAgICAgICAgCi0gICAgaW50IGdldFN3aXRjaFN0YXRlKGludCBzdykgY29uc3Q7Ci0gICAgaW50IGdldFN3aXRjaFN0YXRlKGludDMyX3QgZGV2aWNlSWQsIGludCBzdykgY29uc3Q7Ci0gICAgCi0gICAgaW50IGdldFNjYW5jb2RlU3RhdGUoaW50IGtleSkgY29uc3Q7Ci0gICAgaW50IGdldFNjYW5jb2RlU3RhdGUoaW50MzJfdCBkZXZpY2VJZCwgaW50IGtleSkgY29uc3Q7Ci0gICAgCi0gICAgaW50IGdldEtleWNvZGVTdGF0ZShpbnQga2V5KSBjb25zdDsKLSAgICBpbnQgZ2V0S2V5Y29kZVN0YXRlKGludDMyX3QgZGV2aWNlSWQsIGludCBrZXkpIGNvbnN0OwotICAgIAotICAgIC8vIHNwZWNpYWwgdHlwZSBjb2RlcyB3aGVuIGRldmljZXMgYXJlIGFkZGVkL3JlbW92ZWQuCi0gICAgZW51bSB7Ci0gICAgICAgIERFVklDRV9BRERFRCA9IDB4MTAwMDAwMDAsCi0gICAgICAgIERFVklDRV9SRU1PVkVEID0gMHgyMDAwMDAwMAotICAgIH07Ci0gICAgCi0gICAgLy8gZXhhbWluZSBrZXkgaW5wdXQgZGV2aWNlcyBmb3Igc3BlY2lmaWMgZnJhbWV3b3JrIGtleWNvZGUgc3VwcG9ydAotICAgIGJvb2wgaGFzS2V5cyhzaXplX3QgbnVtQ29kZXMsIGludDMyX3QqIGtleUNvZGVzLCB1aW50OF90KiBvdXRGbGFncyk7Ci0KLSAgICB2aXJ0dWFsIGJvb2wgZ2V0RXZlbnQoaW50MzJfdCogb3V0RGV2aWNlSWQsIGludDMyX3QqIG91dFR5cGUsCi0gICAgICAgICAgICBpbnQzMl90KiBvdXRTY2FuY29kZSwgaW50MzJfdCogb3V0S2V5Y29kZSwgdWludDMyX3QgKm91dEZsYWdzLAotICAgICAgICAgICAgaW50MzJfdCogb3V0VmFsdWUsIG5zZWNzX3QqIG91dFdoZW4pOwotICAgIAotcHJvdGVjdGVkOgotICAgIHZpcnR1YWwgfkV2ZW50SHViKCk7Ci0gICAgdmlydHVhbCB2b2lkIG9uRmlyc3RSZWYoKTsKLSAgICAKLXByaXZhdGU6Ci0gICAgYm9vbCBvcGVuUGxhdGZvcm1JbnB1dCh2b2lkKTsKLSAgICBpbnQzMl90IGNvbnZlcnREZXZpY2VLZXlfVElfUDIoaW50IGNvZGUpOwotCi0gICAgaW50IG9wZW5fZGV2aWNlKGNvbnN0IGNoYXIgKmRldmljZSk7Ci0gICAgaW50IGNsb3NlX2RldmljZShjb25zdCBjaGFyICpkZXZpY2UpOwotICAgIGludCBzY2FuX2Rpcihjb25zdCBjaGFyICpkaXJuYW1lKTsKLSAgICBpbnQgcmVhZF9ub3RpZnkoaW50IG5mZCk7Ci0KLSAgICBzdGF0dXNfdCBtRXJyb3I7Ci0KLSAgICBzdHJ1Y3QgZGV2aWNlX3QgewotICAgICAgICBjb25zdCBpbnQzMl90ICAgaWQ7Ci0gICAgICAgIGNvbnN0IFN0cmluZzggICBwYXRoOwotICAgICAgICBTdHJpbmc4ICAgICAgICAgbmFtZTsKLSAgICAgICAgdWludDMyX3QgICAgICAgIGNsYXNzZXM7Ci0gICAgICAgIHVpbnQ4X3QqICAgICAgICBrZXlCaXRtYXNrOwotICAgICAgICBLZXlMYXlvdXRNYXAqICAgbGF5b3V0TWFwOwotICAgICAgICBTdHJpbmc4ICAgICAgICAga2V5bGF5b3V0RmlsZW5hbWU7Ci0gICAgICAgIGRldmljZV90KiAgICAgICBuZXh0OwotICAgICAgICAKLSAgICAgICAgZGV2aWNlX3QoaW50MzJfdCBfaWQsIGNvbnN0IGNoYXIqIF9wYXRoKTsKLSAgICAgICAgfmRldmljZV90KCk7Ci0gICAgfTsKLQotICAgIGRldmljZV90KiBnZXREZXZpY2UoaW50MzJfdCBkZXZpY2VJZCkgY29uc3Q7Ci0gICAgCi0gICAgLy8gUHJvdGVjdCBhbGwgaW50ZXJuYWwgc3RhdGUuCi0gICAgbXV0YWJsZSBNdXRleCAgIG1Mb2NrOwotICAgIAotICAgIGJvb2wgICAgICAgICAgICBtSGF2ZUZpcnN0S2V5Ym9hcmQ7Ci0gICAgaW50MzJfdCAgICAgICAgIG1GaXJzdEtleWJvYXJkSWQ7IC8vIHRoZSBBUEkgaXMgdGhhdCB0aGUgYnVpbGQgaW4ga2V5Ym9hcmQgaXMgaWQgMCwgc28gbWFwIGl0Ci0gICAgCi0gICAgc3RydWN0IGRldmljZV9lbnQgewotICAgICAgICBkZXZpY2VfdCogZGV2aWNlOwotICAgICAgICB1aW50MzJfdCBzZXE7Ci0gICAgfTsKLSAgICBkZXZpY2VfZW50ICAgICAgKm1EZXZpY2VzQnlJZDsKLSAgICBpbnQgICAgICAgICAgICAgbU51bURldmljZXNCeUlkOwotICAgIAotICAgIGRldmljZV90ICAgICAgICAqbU9wZW5pbmdEZXZpY2VzOwotICAgIGRldmljZV90ICAgICAgICAqbUNsb3NpbmdEZXZpY2VzOwotICAgIAotICAgIGRldmljZV90ICAgICAgICAqKm1EZXZpY2VzOwotICAgIHN0cnVjdCBwb2xsZmQgICAqbUZEczsKLSAgICBpbnQgICAgICAgICAgICAgbUZEQ291bnQ7Ci0gICAgCi0gICAgLy8gZGV2aWNlIGlkcyB0aGF0IHJlcG9ydCBwYXJ0aWN1bGFyIHN3aXRjaGVzLgotI2lmZGVmIEVWX1NXCi0gICAgaW50MzJfdCAgICAgICAgIG1Td2l0Y2hlc1tTV19NQVgrMV07Ci0jZW5kaWYKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBfUlVOVElNRV9FVkVOVF9IVUJfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91aS9JQ2FtZXJhLmggYi9pbmNsdWRlL3VpL0lDYW1lcmEuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjQxZmI2My4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3VpL0lDYW1lcmEuaAorKysgL2Rldi9udWxsCkBAIC0xLDEwMiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0hBUkRXQVJFX0lDQU1FUkFfSAotI2RlZmluZSBBTkRST0lEX0hBUkRXQVJFX0lDQU1FUkFfSAotCi0jaW5jbHVkZSA8dXRpbHMvUmVmQmFzZS5oPgotI2luY2x1ZGUgPHV0aWxzL0lJbnRlcmZhY2UuaD4KLSNpbmNsdWRlIDx1dGlscy9QYXJjZWwuaD4KLSNpbmNsdWRlIDx1aS9JU3VyZmFjZS5oPgotI2luY2x1ZGUgPHV0aWxzL0lNZW1vcnkuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0jaW5jbHVkZSA8dWkvQ2FtZXJhLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgSUNhbWVyYUNsaWVudDsKLQotY2xhc3MgSUNhbWVyYTogcHVibGljIElJbnRlcmZhY2UKLXsKLXB1YmxpYzoKLSAgICBERUNMQVJFX01FVEFfSU5URVJGQUNFKENhbWVyYSk7Ci0KLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICBkaXNjb25uZWN0KCkgPSAwOwotCi0gICAgLy8gY29ubmVjdCBuZXcgY2xpZW50IHdpdGggZXhpc3RpbmcgY2FtZXJhIHJlbW90ZQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgIGNvbm5lY3QoY29uc3Qgc3A8SUNhbWVyYUNsaWVudD4mIGNsaWVudCkgPSAwOwotCi0gICAgLy8gcHJldmVudCBvdGhlciBwcm9jZXNzZXMgZnJvbSB1c2luZyB0aGlzIElDYW1lcmEgaW50ZXJmYWNlCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgbG9jaygpID0gMDsKLQotICAgIC8vIGFsbG93IG90aGVyIHByb2Nlc3NlcyB0byB1c2UgdGhpcyBJQ2FtZXJhIGludGVyZmFjZQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgIHVubG9jaygpID0gMDsKLQotICAgIC8vIHBhc3MgdGhlIGJ1ZmZlcmVkIElTdXJmYWNlIHRvIHRoZSBjYW1lcmEgc2VydmljZQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgIHNldFByZXZpZXdEaXNwbGF5KGNvbnN0IHNwPElTdXJmYWNlPiYgc3VyZmFjZSkgPSAwOwotCi0gICAgLy8gc2V0IHRoZSBwcmV2aWV3IGNhbGxiYWNrIGZsYWcgdG8gYWZmZWN0IGhvdyB0aGUgcmVjZWl2ZWQgZnJhbWVzIGZyb20KLSAgICAvLyBwcmV2aWV3IGFyZSBoYW5kbGVkLgotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHNldFByZXZpZXdDYWxsYmFja0ZsYWcoaW50IGZsYWcpID0gMDsKLQotICAgIC8vIHN0YXJ0IHByZXZpZXcgbW9kZSwgbXVzdCBjYWxsIHNldFByZXZpZXdEaXNwbGF5IGZpcnN0Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgc3RhcnRQcmV2aWV3KCkgPSAwOwotCi0gICAgLy8gc3RvcCBwcmV2aWV3IG1vZGUKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICBzdG9wUHJldmlldygpID0gMDsKLQotICAgIC8vIGdldCBwcmV2aWV3IHN0YXRlCi0gICAgdmlydHVhbCBib29sICAgICAgICAgICAgcHJldmlld0VuYWJsZWQoKSA9IDA7Ci0KLSAgICAvLyBzdGFydCByZWNvcmRpbmcgbW9kZQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgIHN0YXJ0UmVjb3JkaW5nKCkgPSAwOwotCi0gICAgLy8gc3RvcCByZWNvcmRpbmcgbW9kZQotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHN0b3BSZWNvcmRpbmcoKSA9IDA7ICAgIAotCi0gICAgLy8gZ2V0IHJlY29yZGluZyBzdGF0ZQotICAgIHZpcnR1YWwgYm9vbCAgICAgICAgICAgIHJlY29yZGluZ0VuYWJsZWQoKSA9IDA7Ci0KLSAgICAvLyByZWxlYXNlIGEgcmVjb3JkaW5nIGZyYW1lCi0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgcmVsZWFzZVJlY29yZGluZ0ZyYW1lKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0pID0gMDsKLQotICAgIC8vIGF1dG8gZm9jdXMKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICBhdXRvRm9jdXMoKSA9IDA7Ci0KLSAgICAvLyB0YWtlIGEgcGljdHVyZQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgIHRha2VQaWN0dXJlKCkgPSAwOwotCi0gICAgLy8gc2V0IHByZXZpZXcvY2FwdHVyZSBwYXJhbWV0ZXJzIC0ga2V5L3ZhbHVlIHBhaXJzCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgc2V0UGFyYW1ldGVycyhjb25zdCBTdHJpbmc4JiBwYXJhbXMpID0gMDsKLQotICAgIC8vIGdldCBwcmV2aWV3L2NhcHR1cmUgcGFyYW1ldGVycyAtIGtleS92YWx1ZSBwYWlycwotICAgIHZpcnR1YWwgU3RyaW5nOCAgICAgICAgIGdldFBhcmFtZXRlcnMoKSBjb25zdCA9IDA7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIEJuQ2FtZXJhOiBwdWJsaWMgQm5JbnRlcmZhY2U8SUNhbWVyYT4KLXsKLXB1YmxpYzoKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIG9uVHJhbnNhY3QoIHVpbnQzMl90IGNvZGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBQYXJjZWwmIGRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJjZWwqIHJlcGx5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MgPSAwKTsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZgpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91aS9JQ2FtZXJhQ2xpZW50LmggYi9pbmNsdWRlL3VpL0lDYW1lcmFDbGllbnQuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzNiOTUxYy4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3VpL0lDYW1lcmFDbGllbnQuaAorKysgL2Rldi9udWxsCkBAIC0xLDU1ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfSEFSRFdBUkVfSUNBTUVSQV9BUFBfSAotI2RlZmluZSBBTkRST0lEX0hBUkRXQVJFX0lDQU1FUkFfQVBQX0gKLQotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy9JSW50ZXJmYWNlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUGFyY2VsLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSU1lbW9yeS5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIElDYW1lcmFDbGllbnQ6IHB1YmxpYyBJSW50ZXJmYWNlCi17Ci1wdWJsaWM6Ci0gICAgREVDTEFSRV9NRVRBX0lOVEVSRkFDRShDYW1lcmFDbGllbnQpOwotCi0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgc2h1dHRlckNhbGxiYWNrKCkgPSAwOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHJhd0NhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBwaWN0dXJlKSA9IDA7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAganBlZ0NhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBwaWN0dXJlKSA9IDA7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgcHJldmlld0NhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBmcmFtZSkgPSAwOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIGVycm9yQ2FsbGJhY2soc3RhdHVzX3QgZXJyb3IpID0gMDsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICBhdXRvRm9jdXNDYWxsYmFjayhib29sIGZvY3VzZWQpID0gMDsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICByZWNvcmRpbmdDYWxsYmFjayhjb25zdCBzcDxJTWVtb3J5PiYgZnJhbWUpID0gMDsKLQotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBCbkNhbWVyYUNsaWVudDogcHVibGljIEJuSW50ZXJmYWNlPElDYW1lcmFDbGllbnQ+Ci17Ci1wdWJsaWM6Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBvblRyYW5zYWN0KCB1aW50MzJfdCBjb2RlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgUGFyY2VsJiBkYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUGFyY2VsKiByZXBseSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGZsYWdzID0gMCk7Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdWkvSUNhbWVyYVNlcnZpY2UuaCBiL2luY2x1ZGUvdWkvSUNhbWVyYVNlcnZpY2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGZkODkyMy4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3VpL0lDYW1lcmFTZXJ2aWNlLmgKKysrIC9kZXYvbnVsbApAQCAtMSw1NSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0hBUkRXQVJFX0lDQU1FUkFTRVJWSUNFX0gKLSNkZWZpbmUgQU5EUk9JRF9IQVJEV0FSRV9JQ0FNRVJBU0VSVklDRV9ICi0KLSNpbmNsdWRlIDx1dGlscy9SZWZCYXNlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSUludGVyZmFjZS5oPgotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotCi0jaW5jbHVkZSA8dWkvSUNhbWVyYUNsaWVudC5oPgotI2luY2x1ZGUgPHVpL0lDYW1lcmEuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBJQ2FtZXJhU2VydmljZSA6IHB1YmxpYyBJSW50ZXJmYWNlCi17Ci1wcm90ZWN0ZWQ6Ci0gICAgZW51bSB7Ci0gICAgICAgIENPTk5FQ1QgPSBJQmluZGVyOjpGSVJTVF9DQUxMX1RSQU5TQUNUSU9OLAotICAgIH07Ci0KLXB1YmxpYzoKLSAgICBERUNMQVJFX01FVEFfSU5URVJGQUNFKENhbWVyYVNlcnZpY2UpOwotCi0gICAgdmlydHVhbCBzcDxJQ2FtZXJhPiAgICAgY29ubmVjdChjb25zdCBzcDxJQ2FtZXJhQ2xpZW50PiYgY2FtZXJhQ2xpZW50KSA9IDA7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIEJuQ2FtZXJhU2VydmljZTogcHVibGljIEJuSW50ZXJmYWNlPElDYW1lcmFTZXJ2aWNlPgotewotcHVibGljOgotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgb25UcmFuc2FjdCggdWludDMyX3QgY29kZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFBhcmNlbCYgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBhcmNlbCogcmVwbHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDApOwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmCmRpZmYgLS1naXQgYS9pbmNsdWRlL3VpL0lPdmVybGF5LmggYi9pbmNsdWRlL3VpL0lPdmVybGF5LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDY5OWIxYjAuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91aS9JT3ZlcmxheS5oCisrKyAvZGV2L251bGwKQEAgLTEsNTMgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9JT1ZFUkxBWV9ICi0jZGVmaW5lIEFORFJPSURfSU9WRVJMQVlfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy9JSW50ZXJmYWNlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUmVmQmFzZS5oPgotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgSU92ZXJsYXkgOiBwdWJsaWMgSUludGVyZmFjZQotewotcHVibGljOiAKLSAgICBERUNMQVJFX01FVEFfSU5URVJGQUNFKE92ZXJsYXkpOwotCi0gICAgdmlydHVhbCB2b2lkIGRlc3Ryb3koKSA9IDA7IC8vIG9uZS13YXkKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgQm5PdmVybGF5IDogcHVibGljIEJuSW50ZXJmYWNlPElPdmVybGF5PgotewotcHVibGljOgotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgb25UcmFuc2FjdCggdWludDMyX3QgY29kZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFBhcmNlbCYgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBhcmNlbCogcmVwbHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDApOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9JT1ZFUkxBWV9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3VpL0lTdXJmYWNlLmggYi9pbmNsdWRlL3VpL0lTdXJmYWNlLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDg3YjMyMGYuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91aS9JU3VyZmFjZS5oCisrKyAvZGV2L251bGwKQEAgLTEsMTA1ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfSVNVUkZBQ0VfSAotI2RlZmluZSBBTkRST0lEX0lTVVJGQUNFX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSUludGVyZmFjZS5oPgotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLSNpbmNsdWRlIDx1aS9QaXhlbEZvcm1hdC5oPgotCi0jaW5jbHVkZSA8aGFyZHdhcmUvaGFyZHdhcmUuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi10eXBlZGVmIGludDMyX3QgICAgU3VyZmFjZUlEOwotCi1jbGFzcyBJTWVtb3J5SGVhcDsKLWNsYXNzIE92ZXJsYXlSZWY7Ci0KLWNsYXNzIElTdXJmYWNlIDogcHVibGljIElJbnRlcmZhY2UKLXsKLXByb3RlY3RlZDoKLSAgICBlbnVtIHsKLSAgICAgICAgUkVHSVNURVJfQlVGRkVSUyA9IElCaW5kZXI6OkZJUlNUX0NBTExfVFJBTlNBQ1RJT04sCi0gICAgICAgIFVOUkVHSVNURVJfQlVGRkVSUywKLSAgICAgICAgUE9TVF9CVUZGRVIsIC8vIG9uZS13YXkgdHJhbnNhY3Rpb24KLSAgICAgICAgQ1JFQVRFX09WRVJMQVksCi0gICAgfTsKLQotcHVibGljOiAKLSAgICBERUNMQVJFX01FVEFfSU5URVJGQUNFKFN1cmZhY2UpOwotCi0gICAgCi0gICAgY2xhc3MgQnVmZmVySGVhcCB7Ci0gICAgcHVibGljOgotICAgICAgICBlbnVtIHsKLSAgICAgICAgICAgIC8qIHJvdGF0ZSBzb3VyY2UgaW1hZ2UgOTAgZGVncmVlcyAqLwotICAgICAgICAgICAgUk9UXzkwICAgID0gSEFMX1RSQU5TRk9STV9ST1RfOTAsCi0gICAgICAgIH07Ci0gICAgICAgIEJ1ZmZlckhlYXAoKTsKLSAgICAgICAgCi0gICAgICAgIEJ1ZmZlckhlYXAodWludDMyX3QgdywgdWludDMyX3QgaCwKLSAgICAgICAgICAgICAgICBpbnQzMl90IGhvcl9zdHJpZGUsIGludDMyX3QgdmVyX3N0cmlkZSwgCi0gICAgICAgICAgICAgICAgUGl4ZWxGb3JtYXQgZm9ybWF0LCBjb25zdCBzcDxJTWVtb3J5SGVhcD4mIGhlYXApOwotICAgICAgICAKLSAgICAgICAgQnVmZmVySGVhcCh1aW50MzJfdCB3LCB1aW50MzJfdCBoLAotICAgICAgICAgICAgICAgIGludDMyX3QgaG9yX3N0cmlkZSwgaW50MzJfdCB2ZXJfc3RyaWRlLCAKLSAgICAgICAgICAgICAgICBQaXhlbEZvcm1hdCBmb3JtYXQsIHVpbnQzMl90IHRyYW5zZm9ybSwgdWludDMyX3QgZmxhZ3MsCi0gICAgICAgICAgICAgICAgY29uc3Qgc3A8SU1lbW9yeUhlYXA+JiBoZWFwKTsKLSAgICAgICAgCi0gICAgICAgIH5CdWZmZXJIZWFwKCk7IAotICAgICAgICAKLSAgICAgICAgdWludDMyX3QgdzsKLSAgICAgICAgdWludDMyX3QgaDsKLSAgICAgICAgaW50MzJfdCBob3Jfc3RyaWRlOwotICAgICAgICBpbnQzMl90IHZlcl9zdHJpZGU7Ci0gICAgICAgIFBpeGVsRm9ybWF0IGZvcm1hdDsKLSAgICAgICAgdWludDMyX3QgdHJhbnNmb3JtOwotICAgICAgICB1aW50MzJfdCBmbGFnczsKLSAgICAgICAgc3A8SU1lbW9yeUhlYXA+IGhlYXA7Ci0gICAgfTsKLSAgICAKLSAgICB2aXJ0dWFsIHN0YXR1c190IHJlZ2lzdGVyQnVmZmVycyhjb25zdCBCdWZmZXJIZWFwJiBidWZmZXJzKSA9IDA7Ci0KLSAgICB2aXJ0dWFsIHZvaWQgcG9zdEJ1ZmZlcihzc2l6ZV90IG9mZnNldCkgPSAwOyAvLyBvbmUtd2F5Ci0KLSAgICB2aXJ0dWFsIHZvaWQgdW5yZWdpc3RlckJ1ZmZlcnMoKSA9IDA7Ci0gICAgCi0gICAgdmlydHVhbCBzcDxPdmVybGF5UmVmPiBjcmVhdGVPdmVybGF5KAotICAgICAgICAgICAgdWludDMyX3QgdywgdWludDMyX3QgaCwgaW50MzJfdCBmb3JtYXQpID0gMDsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgQm5TdXJmYWNlIDogcHVibGljIEJuSW50ZXJmYWNlPElTdXJmYWNlPgotewotcHVibGljOgotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgb25UcmFuc2FjdCggdWludDMyX3QgY29kZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFBhcmNlbCYgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBhcmNlbCogcmVwbHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDApOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9JU1VSRkFDRV9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3VpL0lTdXJmYWNlQ29tcG9zZXIuaCBiL2luY2x1ZGUvdWkvSVN1cmZhY2VDb21wb3Nlci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmOWVlYjMwLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdWkvSVN1cmZhY2VDb21wb3Nlci5oCisrKyAvZGV2L251bGwKQEAgLTEsMTgxICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDYgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfSVNVUkZBQ0VfQ09NUE9TRVJfSAotI2RlZmluZSBBTkRST0lEX0lTVVJGQUNFX0NPTVBPU0VSX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvUmVmQmFzZS5oPgotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL0lJbnRlcmZhY2UuaD4KLQotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0jaW5jbHVkZSA8dWkvSVN1cmZhY2VGbGluZ2VyQ2xpZW50Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBEaXNwbGF5SW5mbzsKLWNsYXNzIElHUFVDYWxsYmFjazsKLQotY2xhc3MgSVN1cmZhY2VDb21wb3NlciA6IHB1YmxpYyBJSW50ZXJmYWNlCi17Ci1wdWJsaWM6Ci0gICAgREVDTEFSRV9NRVRBX0lOVEVSRkFDRShTdXJmYWNlQ29tcG9zZXIpOwotCi0gICAgZW51bSB7IC8vIChrZWVwIGluIHN5bmMgd2l0aCBTdXJmYWNlLmphdmEpCi0gICAgICAgIGVIaWRkZW4gICAgICAgICAgICAgPSAweDAwMDAwMDA0LAotICAgICAgICBlR1BVICAgICAgICAgICAgICAgID0gMHgwMDAwMDAwOCwKLSAgICAgICAgZUhhcmR3YXJlICAgICAgICAgICA9IDB4MDAwMDAwMTAsCi0gICAgICAgIGVEZXN0cm95QmFja2J1ZmZlciAgPSAweDAwMDAwMDIwLAotICAgICAgICBlU2VjdXJlICAgICAgICAgICAgID0gMHgwMDAwMDA4MCwKLSAgICAgICAgZU5vblByZW11bHRpcGxpZWQgICA9IDB4MDAwMDAxMDAsCi0gICAgICAgIGVQdXNoQnVmZmVycyAgICAgICAgPSAweDAwMDAwMjAwLAotCi0gICAgICAgIGVGWFN1cmZhY2VOb3JtYWwgICAgPSAweDAwMDAwMDAwLAotICAgICAgICBlRlhTdXJmYWNlQmx1ciAgICAgID0gMHgwMDAxMDAwMCwKLSAgICAgICAgZUZYU3VyZmFjZURpbSAgICAgICA9IDB4MDAwMjAwMDAsCi0gICAgICAgIGVGWFN1cmZhY2VNYXNrICAgICAgPSAweDAwMEYwMDAwLAotICAgIH07Ci0KLSAgICBlbnVtIHsKLSAgICAgICAgZVBvc2l0aW9uQ2hhbmdlZCAgICAgICAgICAgID0gMHgwMDAwMDAwMSwKLSAgICAgICAgZUxheWVyQ2hhbmdlZCAgICAgICAgICAgICAgID0gMHgwMDAwMDAwMiwKLSAgICAgICAgZVNpemVDaGFuZ2VkICAgICAgICAgICAgICAgID0gMHgwMDAwMDAwNCwKLSAgICAgICAgZUFscGhhQ2hhbmdlZCAgICAgICAgICAgICAgID0gMHgwMDAwMDAwOCwKLSAgICAgICAgZU1hdHJpeENoYW5nZWQgICAgICAgICAgICAgID0gMHgwMDAwMDAxMCwKLSAgICAgICAgZVRyYW5zcGFyZW50UmVnaW9uQ2hhbmdlZCAgID0gMHgwMDAwMDAyMCwKLSAgICAgICAgZVZpc2liaWxpdHlDaGFuZ2VkICAgICAgICAgID0gMHgwMDAwMDA0MCwKLSAgICAgICAgZUZyZWV6ZVRpbnRDaGFuZ2VkICAgICAgICAgID0gMHgwMDAwMDA4MCwKLSAgICAgICAgZURlc3Ryb3llZCAgICAgICAgICAgICAgICAgID0gMHgwMDAwMDEwMAotICAgIH07Ci0KLSAgICBlbnVtIHsKLSAgICAgICAgZUxheWVySGlkZGVuICAgICAgICA9IDB4MDEsCi0gICAgICAgIGVMYXllckZyb3plbiAgICAgICAgPSAweDAyLAotICAgICAgICBlTGF5ZXJEaXRoZXIgICAgICAgID0gMHgwNCwKLSAgICAgICAgZUxheWVyRmlsdGVyICAgICAgICA9IDB4MDgsCi0gICAgICAgIGVMYXllckJsdXJGcmVlemUgICAgPSAweDEwCi0gICAgfTsKLQotICAgIGVudW0gewotICAgICAgICBlT3JpZW50YXRpb25EZWZhdWx0ICAgICA9IDAsCi0gICAgICAgIGVPcmllbnRhdGlvbjkwICAgICAgICAgID0gMSwKLSAgICAgICAgZU9yaWVudGF0aW9uMTgwICAgICAgICAgPSAyLAotICAgICAgICBlT3JpZW50YXRpb24yNzAgICAgICAgICA9IDMsCi0gICAgICAgIGVPcmllbnRhdGlvblN3YXBNYXNrICAgID0gMHgwMQotICAgIH07Ci0KLSAgICAvKiBjcmVhdGUgY29ubmVjdGlvbiB3aXRoIHN1cmZhY2UgZmxpbmdlciwgcmVxdWlyZXMKLSAgICAgKiBBQ0NFU1NfU1VSRkFDRV9GTElOR0VSIHBlcm1pc3Npb24KLSAgICAgKi8KLQotICAgIHZpcnR1YWwgc3A8SVN1cmZhY2VGbGluZ2VyQ2xpZW50PiBjcmVhdGVDb25uZWN0aW9uKCkgPSAwOwotCi0gICAgLyogcmV0cmlldmUgdGhlIGNvbnRyb2wgYmxvY2sgKi8KLSAgICB2aXJ0dWFsIHNwPElNZW1vcnk+IGdldENibGsoKSBjb25zdCA9IDA7Ci0KLSAgICAvKiBvcGVuL2Nsb3NlIHRyYW5zYWN0aW9ucy4gcmVjcXVpcmVzIEFDQ0VTU19TVVJGQUNFX0ZMSU5HRVIgcGVybWlzc2lvbiAqLwotICAgIHZpcnR1YWwgdm9pZCBvcGVuR2xvYmFsVHJhbnNhY3Rpb24oKSA9IDA7Ci0gICAgdmlydHVhbCB2b2lkIGNsb3NlR2xvYmFsVHJhbnNhY3Rpb24oKSA9IDA7Ci0KLSAgICAvKiBbdW5dZnJlZXplIGRpc3BsYXkuIHJlY3F1aXJlcyBBQ0NFU1NfU1VSRkFDRV9GTElOR0VSIHBlcm1pc3Npb24gKi8KLSAgICB2aXJ0dWFsIHN0YXR1c190IGZyZWV6ZURpc3BsYXkoRGlzcGxheUlEIGRweSwgdWludDMyX3QgZmxhZ3MpID0gMDsKLSAgICB2aXJ0dWFsIHN0YXR1c190IHVuZnJlZXplRGlzcGxheShEaXNwbGF5SUQgZHB5LCB1aW50MzJfdCBmbGFncykgPSAwOwotCi0gICAgLyogU2V0IGRpc3BsYXkgb3JpZW50YXRpb24uIHJlY3F1aXJlcyBBQ0NFU1NfU1VSRkFDRV9GTElOR0VSIHBlcm1pc3Npb24gKi8KLSAgICB2aXJ0dWFsIGludCBzZXRPcmllbnRhdGlvbihEaXNwbGF5SUQgZHB5LCBpbnQgb3JpZW50YXRpb24pID0gMDsKLQotICAgIC8qIHNpZ25hbCB0aGF0IHdlJ3JlIGRvbmUgYm9vdGluZy4KLSAgICAgKiByZWNxdWlyZXMgQUNDRVNTX1NVUkZBQ0VfRkxJTkdFUiBwZXJtaXNzaW9uCi0gICAgICovCi0gICAgdmlydHVhbCB2b2lkIGJvb3RGaW5pc2hlZCgpID0gMDsKLQotICAgIC8qIGdldCBhY2Nlc3MgdG8gdGhlIEdQVS4gQWNjZXNzIGlzIHJlbGlucXVpc2hlZCB3aGVuIHJlbGVhc2luZyByZWdzICovCi0gICAgc3RydWN0IGdwdV9pbmZvX3QgewotICAgICAgICBzdHJ1Y3QgZ3B1X3JlZ2lvbl90IHsKLSAgICAgICAgICAgIHNwPElNZW1vcnk+IHJlZ2lvbjsKLSAgICAgICAgICAgIHNpemVfdCByZXNlcnZlZDsKLSAgICAgICAgfTsKLSAgICAgICAgc3A8SU1lbW9yeT4gICAgICAgICAgICAgcmVnczsKLSAgICAgICAgc2l6ZV90ICAgICAgICAgICAgICAgICAgY291bnQ7Ci0gICAgICAgIGdwdV9yZWdpb25fdCAgICAgICAgICAgIHJlZ2lvbnNbMl07Ci0gICAgfTsKLSAgICB2aXJ0dWFsIHN0YXR1c190IHJlcXVlc3RHUFUoCi0gICAgICAgICAgICBjb25zdCBzcDxJR1BVQ2FsbGJhY2s+JiBjYWxsYmFjaywKLSAgICAgICAgICAgIGdwdV9pbmZvX3QqIGdwdSkgPSAwOwotCi0gICAgLyogdGFrZSB0aGUgZ3B1IGJhY2sgZnJvbSBhbnkgYXBwcyB1c2luZyBpdC4gVGhleSdsbCBnZXQgYQotICAgICAqIEVHTF9DT05URVhUX0xPU1QgZXJyb3IgKi8KLSAgICB2aXJ0dWFsIHN0YXR1c190IHJldm9rZUdQVSgpID0gMDsKLQotICAgIC8qIFNpZ25hbCBzdXJmYWNlZmxpbmdlciB0aGF0IHRoZXJlIG1pZ2h0IGJlIHNvbWUgd29yayB0byBkbwotICAgICAqIFRoaXMgaXMgYW4gQVNZTkNIUk9OT1VTIGNhbGwuCi0gICAgICovCi0gICAgdmlydHVhbCB2b2lkIHNpZ25hbCgpIGNvbnN0ID0gMDsKLX07Ci0KLWNsYXNzIElHUFVDYWxsYmFjayA6IHB1YmxpYyBJSW50ZXJmYWNlCi17Ci1wdWJsaWM6Ci0gICAgREVDTEFSRV9NRVRBX0lOVEVSRkFDRShHUFVDYWxsYmFjayk7Ci0gICAgdmlydHVhbCB2b2lkIGdwdUxvc3QoKSA9IDA7IC8vb25lLXdheQotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBCblN1cmZhY2VDb21wb3NlciA6IHB1YmxpYyBCbkludGVyZmFjZTxJU3VyZmFjZUNvbXBvc2VyPgotewotcHVibGljOgotICAgIGVudW0gewotICAgICAgICAvLyBOb3RlOiBCT09UX0ZJTklTSEVEIG11c3QgcmVtYWluIHRoaXMgdmFsdWUsIGl0IGlzIGNhbGxlZCBmcm9tCi0gICAgICAgIC8vIEphdmEgYnkgQWN0aXZpdHlNYW5hZ2VyU2VydmljZS4KLSAgICAgICAgQk9PVF9GSU5JU0hFRCA9IElCaW5kZXI6OkZJUlNUX0NBTExfVFJBTlNBQ1RJT04sCi0gICAgICAgIENSRUFURV9DT05ORUNUSU9OLAotICAgICAgICBHRVRfQ0JMSywKLSAgICAgICAgT1BFTl9HTE9CQUxfVFJBTlNBQ1RJT04sCi0gICAgICAgIENMT1NFX0dMT0JBTF9UUkFOU0FDVElPTiwKLSAgICAgICAgU0VUX09SSUVOVEFUSU9OLAotICAgICAgICBGUkVFWkVfRElTUExBWSwKLSAgICAgICAgVU5GUkVFWkVfRElTUExBWSwKLSAgICAgICAgUkVRVUVTVF9HUFUsCi0gICAgICAgIFJFVk9LRV9HUFUsCi0gICAgICAgIFNJR05BTAotICAgIH07Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIG9uVHJhbnNhY3QoIHVpbnQzMl90IGNvZGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBQYXJjZWwmIGRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJjZWwqIHJlcGx5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MgPSAwKTsKLX07Ci0KLWNsYXNzIEJuR1BVQ2FsbGJhY2sgOiBwdWJsaWMgQm5JbnRlcmZhY2U8SUdQVUNhbGxiYWNrPgotewotcHVibGljOgotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgb25UcmFuc2FjdCggdWludDMyX3QgY29kZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFBhcmNlbCYgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBhcmNlbCogcmVwbHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDApOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9JU1VSRkFDRV9DT01QT1NFUl9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3VpL0lTdXJmYWNlRmxpbmdlckNsaWVudC5oIGIvaW5jbHVkZS91aS9JU3VyZmFjZUZsaW5nZXJDbGllbnQuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNWI5MzYxZC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3VpL0lTdXJmYWNlRmxpbmdlckNsaWVudC5oCisrKyAvZGV2L251bGwKQEAgLTEsOTAgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9JU1VSRkFDRV9GTElOR0VSX0NMSUVOVF9ICi0jZGVmaW5lIEFORFJPSURfSVNVUkZBQ0VfRkxJTkdFUl9DTElFTlRfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy9JSW50ZXJmYWNlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUmVmQmFzZS5oPgotCi0jaW5jbHVkZSA8dWkvSVN1cmZhY2UuaD4KLQotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0gIAotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIFJlY3Q7Ci1jbGFzcyBQb2ludDsKLWNsYXNzIElNZW1vcnk7Ci1jbGFzcyBJU3VyZmFjZTsKLQotdHlwZWRlZiBpbnQzMl90ICAgIENsaWVudElEOwotdHlwZWRlZiBpbnQzMl90ICAgIERpc3BsYXlJRDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBsYXllcl9zdGF0ZV90OwotCi1jbGFzcyBJU3VyZmFjZUZsaW5nZXJDbGllbnQgOiBwdWJsaWMgSUludGVyZmFjZQotewotcHVibGljOiAKLSAgICBERUNMQVJFX01FVEFfSU5URVJGQUNFKFN1cmZhY2VGbGluZ2VyQ2xpZW50KTsKLQotICAgIHN0cnVjdCBzdXJmYWNlX2RhdGFfdCB7Ci0gICAgICAgIGludDMyX3QgICAgICAgICAgICAgdG9rZW47Ci0gICAgICAgIGludDMyX3QgICAgICAgICAgICAgaWRlbnRpdHk7Ci0gICAgICAgIHNwPElNZW1vcnlIZWFwPiAgICAgaGVhcFsyXTsKLSAgICAgICAgc3RhdHVzX3QgcmVhZEZyb21QYXJjZWwoY29uc3QgUGFyY2VsJiBwYXJjZWwpOwotICAgICAgICBzdGF0dXNfdCB3cml0ZVRvUGFyY2VsKFBhcmNlbCogcGFyY2VsKSBjb25zdDsKLSAgICB9OwotICAgIAotICAgIHZpcnR1YWwgdm9pZCBnZXRDb250cm9sQmxvY2tzKHNwPElNZW1vcnk+KiBjdGwpIGNvbnN0ID0gMDsKLQotICAgIHZpcnR1YWwgc3A8SVN1cmZhY2U+IGNyZWF0ZVN1cmZhY2UoIHN1cmZhY2VfZGF0YV90KiBkYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBwaWQsIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIERpc3BsYXlJRCBkaXNwbGF5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHcsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgaCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQaXhlbEZvcm1hdCBmb3JtYXQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MpID0gMDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgZGVzdHJveVN1cmZhY2UoU3VyZmFjZUlEIHNpZCkgPSAwOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRTdGF0ZShpbnQzMl90IGNvdW50LCBjb25zdCBsYXllcl9zdGF0ZV90KiBzdGF0ZXMpID0gMDsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgQm5TdXJmYWNlRmxpbmdlckNsaWVudCA6IHB1YmxpYyBCbkludGVyZmFjZTxJU3VyZmFjZUZsaW5nZXJDbGllbnQ+Ci17Ci1wdWJsaWM6Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBvblRyYW5zYWN0KCB1aW50MzJfdCBjb2RlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgUGFyY2VsJiBkYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUGFyY2VsKiByZXBseSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGZsYWdzID0gMCk7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0lTVVJGQUNFX0ZMSU5HRVJfQ0xJRU5UX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdWkvS2V5Q2hhcmFjdGVyTWFwLmggYi9pbmNsdWRlL3VpL0tleUNoYXJhY3Rlck1hcC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiYWQyY2Y4Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdWkvS2V5Q2hhcmFjdGVyTWFwLmgKKysrIC9kZXYvbnVsbApAQCAtMSw3MiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBfVUlfS0VZX0NIQVJBQ1RFUl9NQVBfSAotI2RlZmluZSBfVUlfS0VZX0NIQVJBQ1RFUl9NQVBfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvVmVjdG9yLmg+Ci0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi1jbGFzcyBLZXlDaGFyYWN0ZXJNYXAKLXsKLXB1YmxpYzoKLSAgICB+S2V5Q2hhcmFjdGVyTWFwKCk7Ci0KLSAgICAvLyBzZWUgdGhlIGphdmFkb2MgZm9yIGFuZHJvaWQudGV4dC5tZXRob2QuS2V5Q2hhcmFjdGVyTWFwIGZvciB3aGF0Ci0gICAgLy8gdGhlc2UgZG8KLSAgICB1bnNpZ25lZCBzaG9ydCBnZXQoaW50IGtleWNvZGUsIGludCBtZXRhKTsKLSAgICB1bnNpZ25lZCBzaG9ydCBnZXROdW1iZXIoaW50IGtleWNvZGUpOwotICAgIHVuc2lnbmVkIHNob3J0IGdldE1hdGNoKGludCBrZXljb2RlLCBjb25zdCB1bnNpZ25lZCBzaG9ydCogY2hhcnMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGNoYXJzaXplLCB1aW50MzJfdCBtb2RpZmllcnMpOwotICAgIHVuc2lnbmVkIHNob3J0IGdldERpc3BsYXlMYWJlbChpbnQga2V5Y29kZSk7Ci0gICAgYm9vbCBnZXRLZXlEYXRhKGludCBrZXljb2RlLCB1bnNpZ25lZCBzaG9ydCAqZGlzcGxheUxhYmVsLAotICAgICAgICAgICAgICAgICAgICB1bnNpZ25lZCBzaG9ydCAqbnVtYmVyLCB1bnNpZ25lZCBzaG9ydCogcmVzdWx0cyk7Ci0gICAgaW5saW5lIHVuc2lnbmVkIGludCBnZXRLZXlib2FyZFR5cGUoKSB7IHJldHVybiBtX3R5cGU7IH0KLSAgICBib29sIGdldEV2ZW50cyh1aW50MTZfdCogY2hhcnMsIHNpemVfdCBsZW4sCi0gICAgICAgICAgICAgICAgICAgVmVjdG9yPGludDMyX3Q+KiBrZXlzLCBWZWN0b3I8dWludDMyX3Q+KiBtb2RpZmllcnMpOwotCi0gICAgc3RhdGljIEtleUNoYXJhY3Rlck1hcCogbG9hZChpbnQgaWQpOwotCi0gICAgZW51bSB7Ci0gICAgICAgIE5VTUVSSUMgPSAxLAotICAgICAgICBRMTQgPSAyLAotICAgICAgICBRV0VSVFkgPSAzIC8vIG9yIEFaRVJUWSBvciB3aGF0ZXZlcgotICAgIH07Ci0KLSNkZWZpbmUgTUVUQV9NQVNLIDMKLQotcHJpdmF0ZToKLSAgICBzdHJ1Y3QgS2V5Ci0gICAgewotICAgICAgICBpbnQzMl90IGtleWNvZGU7Ci0gICAgICAgIHVpbnQxNl90IGRpc3BsYXlfbGFiZWw7Ci0gICAgICAgIHVpbnQxNl90IG51bWJlcjsKLSAgICAgICAgdWludDE2X3QgZGF0YVtNRVRBX01BU0sgKyAxXTsKLSAgICB9OwotCi0gICAgS2V5Q2hhcmFjdGVyTWFwKCk7Ci0gICAgc3RhdGljIEtleUNoYXJhY3Rlck1hcCogdHJ5X2ZpbGUoY29uc3QgY2hhciogZmlsZW5hbWUpOwotICAgIEtleSogZmluZF9rZXkoaW50IGtleWNvZGUpOwotICAgIGJvb2wgZmluZF9jaGFyKHVpbnQxNl90IGMsIHVpbnQzMl90KiBrZXksIHVpbnQzMl90KiBtb2RzKTsKLQotICAgIHVuc2lnbmVkIGludCBtX3R5cGU7Ci0gICAgdW5zaWduZWQgaW50IG1fa2V5Q291bnQ7Ci0gICAgS2V5KiBtX2tleXM7Ci19OwotCi0jZW5kaWYgLy8gX1VJX0tFWV9DSEFSQUNURVJfTUFQX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdWkvS2V5Y29kZUxhYmVscy5oIGIvaW5jbHVkZS91aS9LZXljb2RlTGFiZWxzLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGVmYTZkMmIuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91aS9LZXljb2RlTGFiZWxzLmgKKysrIC9kZXYvbnVsbApAQCAtMSwyMzYgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgX1VJX0tFWUNPREVfTEFCRUxTX0gKLSNkZWZpbmUgX1VJX0tFWUNPREVfTEFCRUxTX0gKLQotc3RydWN0IEtleWNvZGVMYWJlbCB7Ci0gICAgY29uc3QgY2hhciAqbGl0ZXJhbDsKLSAgICBpbnQgdmFsdWU7Ci19OwotCi1zdGF0aWMgY29uc3QgS2V5Y29kZUxhYmVsIEtFWUNPREVTW10gPSB7Ci0gICAgeyAiU09GVF9MRUZUIiwgMSB9LAotICAgIHsgIlNPRlRfUklHSFQiLCAyIH0sCi0gICAgeyAiSE9NRSIsIDMgfSwKLSAgICB7ICJCQUNLIiwgNCB9LAotICAgIHsgIkNBTEwiLCA1IH0sCi0gICAgeyAiRU5EQ0FMTCIsIDYgfSwKLSAgICB7ICIwIiwgNyB9LAotICAgIHsgIjEiLCA4IH0sCi0gICAgeyAiMiIsIDkgfSwKLSAgICB7ICIzIiwgMTAgfSwKLSAgICB7ICI0IiwgMTEgfSwKLSAgICB7ICI1IiwgMTIgfSwKLSAgICB7ICI2IiwgMTMgfSwKLSAgICB7ICI3IiwgMTQgfSwKLSAgICB7ICI4IiwgMTUgfSwKLSAgICB7ICI5IiwgMTYgfSwKLSAgICB7ICJTVEFSIiwgMTcgfSwKLSAgICB7ICJQT1VORCIsIDE4IH0sCi0gICAgeyAiRFBBRF9VUCIsIDE5IH0sCi0gICAgeyAiRFBBRF9ET1dOIiwgMjAgfSwKLSAgICB7ICJEUEFEX0xFRlQiLCAyMSB9LAotICAgIHsgIkRQQURfUklHSFQiLCAyMiB9LAotICAgIHsgIkRQQURfQ0VOVEVSIiwgMjMgfSwKLSAgICB7ICJWT0xVTUVfVVAiLCAyNCB9LAotICAgIHsgIlZPTFVNRV9ET1dOIiwgMjUgfSwKLSAgICB7ICJQT1dFUiIsIDI2IH0sCi0gICAgeyAiQ0FNRVJBIiwgMjcgfSwKLSAgICB7ICJDTEVBUiIsIDI4IH0sCi0gICAgeyAiQSIsIDI5IH0sCi0gICAgeyAiQiIsIDMwIH0sCi0gICAgeyAiQyIsIDMxIH0sCi0gICAgeyAiRCIsIDMyIH0sCi0gICAgeyAiRSIsIDMzIH0sCi0gICAgeyAiRiIsIDM0IH0sCi0gICAgeyAiRyIsIDM1IH0sCi0gICAgeyAiSCIsIDM2IH0sCi0gICAgeyAiSSIsIDM3IH0sCi0gICAgeyAiSiIsIDM4IH0sCi0gICAgeyAiSyIsIDM5IH0sCi0gICAgeyAiTCIsIDQwIH0sCi0gICAgeyAiTSIsIDQxIH0sCi0gICAgeyAiTiIsIDQyIH0sCi0gICAgeyAiTyIsIDQzIH0sCi0gICAgeyAiUCIsIDQ0IH0sCi0gICAgeyAiUSIsIDQ1IH0sCi0gICAgeyAiUiIsIDQ2IH0sCi0gICAgeyAiUyIsIDQ3IH0sCi0gICAgeyAiVCIsIDQ4IH0sCi0gICAgeyAiVSIsIDQ5IH0sCi0gICAgeyAiViIsIDUwIH0sCi0gICAgeyAiVyIsIDUxIH0sCi0gICAgeyAiWCIsIDUyIH0sCi0gICAgeyAiWSIsIDUzIH0sCi0gICAgeyAiWiIsIDU0IH0sCi0gICAgeyAiQ09NTUEiLCA1NSB9LAotICAgIHsgIlBFUklPRCIsIDU2IH0sCi0gICAgeyAiQUxUX0xFRlQiLCA1NyB9LAotICAgIHsgIkFMVF9SSUdIVCIsIDU4IH0sCi0gICAgeyAiU0hJRlRfTEVGVCIsIDU5IH0sCi0gICAgeyAiU0hJRlRfUklHSFQiLCA2MCB9LAotICAgIHsgIlRBQiIsIDYxIH0sCi0gICAgeyAiU1BBQ0UiLCA2MiB9LAotICAgIHsgIlNZTSIsIDYzIH0sCi0gICAgeyAiRVhQTE9SRVIiLCA2NCB9LAotICAgIHsgIkVOVkVMT1BFIiwgNjUgfSwKLSAgICB7ICJFTlRFUiIsIDY2IH0sCi0gICAgeyAiREVMIiwgNjcgfSwKLSAgICB7ICJHUkFWRSIsIDY4IH0sCi0gICAgeyAiTUlOVVMiLCA2OSB9LAotICAgIHsgIkVRVUFMUyIsIDcwIH0sCi0gICAgeyAiTEVGVF9CUkFDS0VUIiwgNzEgfSwKLSAgICB7ICJSSUdIVF9CUkFDS0VUIiwgNzIgfSwKLSAgICB7ICJCQUNLU0xBU0giLCA3MyB9LAotICAgIHsgIlNFTUlDT0xPTiIsIDc0IH0sCi0gICAgeyAiQVBPU1RST1BIRSIsIDc1IH0sCi0gICAgeyAiU0xBU0giLCA3NiB9LAotICAgIHsgIkFUIiwgNzcgfSwKLSAgICB7ICJOVU0iLCA3OCB9LAotICAgIHsgIkhFQURTRVRIT09LIiwgNzkgfSwKLSAgICB7ICJGT0NVUyIsIDgwIH0sCi0gICAgeyAiUExVUyIsIDgxIH0sCi0gICAgeyAiTUVOVSIsIDgyIH0sCi0gICAgeyAiTk9USUZJQ0FUSU9OIiwgODMgfSwKLSAgICB7ICJTRUFSQ0giLCA4NCB9LAotICAgIHsgIlBMQVlQQVVTRSIsIDg1IH0sCi0gICAgeyAiU1RPUCIsIDg2IH0sCi0gICAgeyAiTkVYVFNPTkciLCA4NyB9LAotICAgIHsgIlBSRVZJT1VTU09ORyIsIDg4IH0sCi0gICAgeyAiUkVXSU5EIiwgODkgfSwKLSAgICB7ICJGT1JXQVJEIiwgOTAgfSwKLSAgICB7ICJNVVRFIiwgOTEgfSwKLQotICAgIC8vIE5PVEU6IElmIHlvdSBhZGQgYSBuZXcga2V5Y29kZSBoZXJlIHlvdSBtdXN0IGFsc28gYWRkIGl0IHRvOgotICAgIC8vICAgKGVudW0gS2V5Q29kZSwgaW4gdGhpcyBmaWxlKQotICAgIC8vICAgZnJhbWV3b3Jrcy9iYXNlL2NvcmUvamF2YS9hbmRyb2lkL3ZpZXcvS2V5RXZlbnQuamF2YQotICAgIC8vICAgdG9vbHMvcHVwcGV0X21hc3Rlci9QdXBwZXRNYXN0ZXIubmF2X2tleXMucHkKLSAgICAvLyAgIGZyYW1ld29ya3MvYmFzZS9jb3JlL3Jlcy9yZXMvdmFsdWVzL2F0dHJzLnhtbAotCi0gICAgeyBOVUxMLCAwIH0KLX07Ci0KLS8vIFRoZXNlIGNvbnN0YW50cyBuZWVkIHRvIG1hdGNoIHRoZSBhYm92ZSBtYXBwaW5ncy4KLXR5cGVkZWYgZW51bSBLZXlDb2RlIHsKLSAgICBrS2V5Q29kZVVua25vd24gPSAwLAotCi0gICAga0tleUNvZGVTb2Z0TGVmdCA9IDEsCi0gICAga0tleUNvZGVTb2Z0UmlnaHQgPSAyLAotICAgIGtLZXlDb2RlSG9tZSA9IDMsCi0gICAga0tleUNvZGVCYWNrID0gNCwKLSAgICBrS2V5Q29kZUNhbGwgPSA1LAotICAgIGtLZXlDb2RlRW5kQ2FsbCA9IDYsCi0gICAga0tleUNvZGUwID0gNywKLSAgICBrS2V5Q29kZTEgPSA4LAotICAgIGtLZXlDb2RlMiA9IDksCi0gICAga0tleUNvZGUzID0gMTAsCi0gICAga0tleUNvZGU0ID0gMTEsCi0gICAga0tleUNvZGU1ID0gMTIsCi0gICAga0tleUNvZGU2ID0gMTMsCi0gICAga0tleUNvZGU3ID0gMTQsCi0gICAga0tleUNvZGU4ID0gMTUsCi0gICAga0tleUNvZGU5ID0gMTYsCi0gICAga0tleUNvZGVTdGFyID0gMTcsCi0gICAga0tleUNvZGVQb3VuZCA9IDE4LAotICAgIGtLZXlDb2RlRHBhZFVwID0gMTksCi0gICAga0tleUNvZGVEcGFkRG93biA9IDIwLAotICAgIGtLZXlDb2RlRHBhZExlZnQgPSAyMSwKLSAgICBrS2V5Q29kZURwYWRSaWdodCA9IDIyLAotICAgIGtLZXlDb2RlRHBhZENlbnRlciA9IDIzLAotICAgIGtLZXlDb2RlVm9sdW1lVXAgPSAyNCwKLSAgICBrS2V5Q29kZVZvbHVtZURvd24gPSAyNSwKLSAgICBrS2V5Q29kZVBvd2VyID0gMjYsCi0gICAga0tleUNvZGVDYW1lcmEgPSAyNywKLSAgICBrS2V5Q29kZUNsZWFyID0gMjgsCi0gICAga0tleUNvZGVBID0gMjksCi0gICAga0tleUNvZGVCID0gMzAsCi0gICAga0tleUNvZGVDID0gMzEsCi0gICAga0tleUNvZGVEID0gMzIsCi0gICAga0tleUNvZGVFID0gMzMsCi0gICAga0tleUNvZGVGID0gMzQsCi0gICAga0tleUNvZGVHID0gMzUsCi0gICAga0tleUNvZGVIID0gMzYsCi0gICAga0tleUNvZGVJID0gMzcsCi0gICAga0tleUNvZGVKID0gMzgsCi0gICAga0tleUNvZGVLID0gMzksCi0gICAga0tleUNvZGVMID0gNDAsCi0gICAga0tleUNvZGVNID0gNDEsCi0gICAga0tleUNvZGVOID0gNDIsCi0gICAga0tleUNvZGVPID0gNDMsCi0gICAga0tleUNvZGVQID0gNDQsCi0gICAga0tleUNvZGVRID0gNDUsCi0gICAga0tleUNvZGVSID0gNDYsCi0gICAga0tleUNvZGVTID0gNDcsCi0gICAga0tleUNvZGVUID0gNDgsCi0gICAga0tleUNvZGVVID0gNDksCi0gICAga0tleUNvZGVWID0gNTAsCi0gICAga0tleUNvZGVXID0gNTEsCi0gICAga0tleUNvZGVYID0gNTIsCi0gICAga0tleUNvZGVZID0gNTMsCi0gICAga0tleUNvZGVaID0gNTQsCi0gICAga0tleUNvZGVDb21tYSA9IDU1LAotICAgIGtLZXlDb2RlUGVyaW9kID0gNTYsCi0gICAga0tleUNvZGVBbHRMZWZ0ID0gNTcsCi0gICAga0tleUNvZGVBbHRSaWdodCA9IDU4LAotICAgIGtLZXlDb2RlU2hpZnRMZWZ0ID0gNTksCi0gICAga0tleUNvZGVTaGlmdFJpZ2h0ID0gNjAsCi0gICAga0tleUNvZGVUYWIgPSA2MSwKLSAgICBrS2V5Q29kZVNwYWNlID0gNjIsCi0gICAga0tleUNvZGVTeW0gPSA2MywKLSAgICBrS2V5Q29kZUV4cGxvcmVyID0gNjQsCi0gICAga0tleUNvZGVFbnZlbG9wZSA9IDY1LAotICAgIGtLZXlDb2RlTmV3bGluZSA9IDY2LAotICAgIGtLZXlDb2RlRGVsID0gNjcsCi0gICAga0tleUNvZGVHcmF2ZSA9IDY4LAotICAgIGtLZXlDb2RlTWludXMgPSA2OSwKLSAgICBrS2V5Q29kZUVxdWFscyA9IDcwLAotICAgIGtLZXlDb2RlTGVmdEJyYWNrZXQgPSA3MSwKLSAgICBrS2V5Q29kZVJpZ2h0QnJhY2tldCA9IDcyLAotICAgIGtLZXlDb2RlQmFja3NsYXNoID0gNzMsCi0gICAga0tleUNvZGVTZW1pY29sb24gPSA3NCwKLSAgICBrS2V5Q29kZUFwb3N0cm9waGUgPSA3NSwKLSAgICBrS2V5Q29kZVNsYXNoID0gNzYsCi0gICAga0tleUNvZGVBdCA9IDc3LAotICAgIGtLZXlDb2RlTnVtID0gNzgsCi0gICAga0tleUNvZGVIZWFkU2V0SG9vayA9IDc5LAotICAgIGtLZXlDb2RlRm9jdXMgPSA4MCwKLSAgICBrS2V5Q29kZVBsdXMgPSA4MSwKLSAgICBrS2V5Q29kZU1lbnUgPSA4MiwKLSAgICBrS2V5Q29kZU5vdGlmaWNhdGlvbiA9IDgzLAotICAgIGtLZXlDb2RlU2VhcmNoID0gODQsCi0gICAga0tleUNvZGVQbGF5UGF1c2UgPSA4NSwKLSAgICBrS2V5Q29kZVN0b3AgPSA4NiwKLSAgICBrS2V5Q29kZU5leHRTb25nID0gODcsCi0gICAga0tleUNvZGVQcmV2aW91c1NvbmcgPSA4OCwKLSAgICBrS2V5Q29kZVJld2luZCA9IDg5LAotICAgIGtLZXlDb2RlRm9yd2FyZCA9IDkwLAotICAgIGtLZXlDb2RlTXV0ZSA9IDkxCi19IEtleUNvZGU7Ci0KLXN0YXRpYyBjb25zdCBLZXljb2RlTGFiZWwgRkxBR1NbXSA9IHsKLSAgICB7ICJXQUtFIiwgMHgwMDAwMDAwMSB9LAotICAgIHsgIldBS0VfRFJPUFBFRCIsIDB4MDAwMDAwMDIgfSwKLSAgICB7ICJTSElGVCIsIDB4MDAwMDAwMDQgfSwKLSAgICB7ICJDQVBTX0xPQ0siLCAweDAwMDAwMDA4IH0sCi0gICAgeyAiQUxUIiwgMHgwMDAwMDAxMCB9LAotICAgIHsgIkFMVF9HUiIsIDB4MDAwMDAwMjAgfSwKLSAgICB7ICJNRU5VIiwgMHgwMDAwMDA0MCB9LAotICAgIHsgIkxBVU5DSEVSIiwgMHgwMDAwMDA4MCB9LAotICAgIHsgTlVMTCwgMCB9Ci19OwotCi0jZW5kaWYgLy8gX1VJX0tFWUNPREVfTEFCRUxTX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdWkvT3ZlcmxheS5oIGIvaW5jbHVkZS91aS9PdmVybGF5LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDY2NTE0YjQuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91aS9PdmVybGF5LmgKKysrIC9kZXYvbnVsbApAQCAtMSwxMDkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9PVkVSTEFZX0gKLSNkZWZpbmUgQU5EUk9JRF9PVkVSTEFZX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSUludGVyZmFjZS5oPgotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLSNpbmNsdWRlIDx1aS9QaXhlbEZvcm1hdC5oPgotI2luY2x1ZGUgPHVpL0lPdmVybGF5Lmg+Ci0KLSNpbmNsdWRlIDxoYXJkd2FyZS9vdmVybGF5Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgSU1lbW9yeTsKLWNsYXNzIElNZW1vcnlIZWFwOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIE92ZXJsYXlSZWYgOiBwdWJsaWMgTGlnaHRSZWZCYXNlPE92ZXJsYXlSZWY+Ci17Ci1wdWJsaWM6Ci0gICAgT3ZlcmxheVJlZihvdmVybGF5X2hhbmRsZV90LCBjb25zdCBzcDxJT3ZlcmxheT4mLAotICAgICAgICAgICAgdWludDMyX3QgdywgdWludDMyX3QgaCwgaW50MzJfdCBmLCB1aW50MzJfdCB3cywgdWludDMyX3QgaHMpOwotCi0gICAgc3RhdGljIHNwPE92ZXJsYXlSZWY+IHJlYWRGcm9tUGFyY2VsKGNvbnN0IFBhcmNlbCYgZGF0YSk7Ci0gICAgc3RhdGljIHN0YXR1c190IHdyaXRlVG9QYXJjZWwoUGFyY2VsKiByZXBseSwgY29uc3Qgc3A8T3ZlcmxheVJlZj4mIG8pOyAgICAKLQotcHJpdmF0ZToKLSAgICBmcmllbmQgY2xhc3MgTGlnaHRSZWZCYXNlPE92ZXJsYXlSZWY+OwotICAgIGZyaWVuZCBjbGFzcyBPdmVybGF5OwotCi0gICAgT3ZlcmxheVJlZigpOwotICAgIHZpcnR1YWwgfk92ZXJsYXlSZWYoKTsKLQotICAgIG92ZXJsYXlfaGFuZGxlX3QgbU92ZXJsYXlIYW5kbGU7Ci0gICAgc3A8SU92ZXJsYXk+IG1PdmVybGF5Q2hhbm5lbDsKLSAgICB1aW50MzJfdCBtV2lkdGg7Ci0gICAgdWludDMyX3QgbUhlaWdodDsKLSAgICBpbnQzMl90ICBtRm9ybWF0OwotICAgIGludDMyX3QgIG1XaWR0aFN0cmlkZTsKLSAgICBpbnQzMl90ICBtSGVpZ2h0U3RyaWRlOwotICAgIGJvb2wgbU93bkhhbmRsZTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgT3ZlcmxheSA6IHB1YmxpYyB2aXJ0dWFsIFJlZkJhc2UKLXsKLXB1YmxpYzoKLSAgICBPdmVybGF5KGNvbnN0IHNwPE92ZXJsYXlSZWY+JiBvdmVybGF5UmVmKTsKLQotICAgIC8qIGRlc3Ryb3lzIHRoaXMgb3ZlcmxheSAqLwotICAgIHZvaWQgZGVzdHJveSgpOwotICAgIAotICAgIC8qIGdldCB0aGUgSEFMIGhhbmRsZSBmb3IgdGhpcyBvdmVybGF5ICovCi0gICAgb3ZlcmxheV9oYW5kbGVfdCBnZXRIYW5kbGVSZWYoKSBjb25zdDsKLQotICAgIC8qIGJsb2NrcyB1bnRpbCBhbiBvdmVybGF5IGJ1ZmZlciBpcyBhdmFpbGFibGUgYW5kIHJldHVybiB0aGF0IGJ1ZmZlci4gKi8KLSAgICBzdGF0dXNfdCBkZXF1ZXVlQnVmZmVyKG92ZXJsYXlfYnVmZmVyX3QqIGJ1ZmZlcik7Ci0KLSAgICAvKiByZWxlYXNlIHRoZSBvdmVybGF5IGJ1ZmZlciBhbmQgcG9zdCBpdCAqLwotICAgIHN0YXR1c190IHF1ZXVlQnVmZmVyKG92ZXJsYXlfYnVmZmVyX3QgYnVmZmVyKTsKLQotICAgIC8qIHJldHVybnMgdGhlIGFkZHJlc3Mgb2YgYSBnaXZlbiBidWZmZXIgaWYgc3VwcG9ydGVkLCBOVUxMIG90aGVyd2lzZS4gKi8KLSAgICB2b2lkKiBnZXRCdWZmZXJBZGRyZXNzKG92ZXJsYXlfYnVmZmVyX3QgYnVmZmVyKTsKLQotICAgIC8qIGdldCBwaHlzaWNhbCBpbmZvcm1hdGlvbnMgYWJvdXQgdGhlIG92ZXJsYXkgKi8KLSAgICB1aW50MzJfdCBnZXRXaWR0aCgpIGNvbnN0OwotICAgIHVpbnQzMl90IGdldEhlaWdodCgpIGNvbnN0OwotICAgIGludDMyX3QgZ2V0Rm9ybWF0KCkgY29uc3Q7Ci0gICAgaW50MzJfdCBnZXRXaWR0aFN0cmlkZSgpIGNvbnN0OwotICAgIGludDMyX3QgZ2V0SGVpZ2h0U3RyaWRlKCkgY29uc3Q7Ci0gICAgaW50MzJfdCBnZXRCdWZmZXJDb3VudCgpIGNvbnN0OwotICAgIHN0YXR1c190IGdldFN0YXR1cygpIGNvbnN0OwotICAgIAotcHJpdmF0ZToKLSAgICB2aXJ0dWFsIH5PdmVybGF5KCk7Ci0KLSAgICBzcDxPdmVybGF5UmVmPiBtT3ZlcmxheVJlZjsKLSAgICBvdmVybGF5X2RhdGFfZGV2aWNlX3QgKm1PdmVybGF5RGF0YTsKLSAgICBzdGF0dXNfdCBtU3RhdHVzOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9PVkVSTEFZX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdWkvUGl4ZWxGb3JtYXQuaCBiL2luY2x1ZGUvdWkvUGl4ZWxGb3JtYXQuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTRhZjgyMy4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3VpL1BpeGVsRm9ybWF0LmgKKysrIC9kZXYvbnVsbApAQCAtMSwxMjUgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0KLS8vIFBpeGVsIGZvcm1hdHMgdXNlZCBhY3Jvc3MgdGhlIHN5c3RlbS4KLS8vIFRoZXNlIGZvcm1hdHMgbWlnaHQgbm90IGFsbCBiZSBzdXBwb3J0ZWQgYnkgYWxsIHJlbmRlcmVycywgZm9yIGluc3RhbmNlCi0vLyBza2lhIG9yIFN1cmZhY2VGbGluZ2VyIGFyZSBub3QgcmVxdWlyZWQgdG8gc3VwcG9ydCBhbGwgb2YgdGhlc2UgZm9ybWF0cwotLy8gKGVpdGhlciBhcyBzb3VyY2Ugb3IgZGVzdGluYXRpb24pCi0KLS8vIFhYWDogd2Ugc2hvdWxkIGNvbnNvbGlkYXRlIHRoZXNlIGZvcm1hdHMgYW5kIHNraWEncwotCi0jaWZuZGVmIFVJX1BJWEVMRk9STUFUX0gKLSNkZWZpbmUgVUlfUElYRUxGT1JNQVRfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8cGl4ZWxmbGluZ2VyL2Zvcm1hdC5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWVudW0gewotICAgIC8vCi0gICAgLy8gdGhlc2UgY29uc3RhbnRzIG5lZWQgdG8gbWF0Y2ggdGhvc2UKLSAgICAvLyBpbiBncmFwaGljcy9QaXhlbEZvcm1hdC5qYXZhICYgcGl4ZWxmbGluZ2VyL2Zvcm1hdC5oCi0gICAgLy8KLSAgICBQSVhFTF9GT1JNQVRfVU5LTk9XTiAgICA9ICAgMCwKLSAgICBQSVhFTF9GT1JNQVRfTk9ORSAgICAgICA9ICAgMCwKLQotICAgIC8vIGxvZ2ljYWwgcGl4ZWwgZm9ybWF0cyB1c2VkIGJ5IHRoZSBTdXJmYWNlRmxpbmdlciAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICAgIFBJWEVMX0ZPUk1BVF9DVVNUT00gICAgICAgICA9IC00LAotICAgICAgICAvLyBDdXN0b20gcGl4ZWwtZm9ybWF0IGRlc2NyaWJlZCBieSBhIFBpeGVsRm9ybWF0SW5mbyBzdHJ1Y3R1cmUKLQotICAgIFBJWEVMX0ZPUk1BVF9UUkFOU0xVQ0VOVCAgICA9IC0zLAotICAgICAgICAvLyBTeXN0ZW0gY2hvb3NlcyBhIGZvcm1hdCB0aGF0IHN1cHBvcnRzIHRyYW5zbHVjZW5jeSAobWFueSBhbHBoYSBiaXRzKQotCi0gICAgUElYRUxfRk9STUFUX1RSQU5TUEFSRU5UICAgID0gLTIsCi0gICAgICAgIC8vIFN5c3RlbSBjaG9vc2VzIGEgZm9ybWF0IHRoYXQgc3VwcG9ydHMgdHJhbnNwYXJlbmN5Ci0gICAgICAgIC8vIChhdCBsZWFzdCAxIGFscGhhIGJpdCkKLQotICAgIFBJWEVMX0ZPUk1BVF9PUEFRVUUgICAgICAgICA9IC0xLAotICAgICAgICAvLyBTeXN0ZW0gY2hvb3NlcyBhbiBvcGFxdWUgZm9ybWF0IChubyBhbHBoYSBiaXRzIHJlcXVpcmVkKQotICAgIAotICAgIC8vIHJlYWwgcGl4ZWwgZm9ybWF0cyBzdXBwb3J0ZWQgZm9yIHJlbmRlcmluZyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0gICAgUElYRUxfRk9STUFUX1JHQkFfODg4OCAgID0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzg4ODgsICAvLyA0eDgtYml0IFJHQkEKLSAgICBQSVhFTF9GT1JNQVRfUkdCWF84ODg4ICAgPSBHR0xfUElYRUxfRk9STUFUX1JHQlhfODg4OCwgIC8vIDR4OC1iaXQgUkdCMAotICAgIFBJWEVMX0ZPUk1BVF9SR0JfODg4ICAgICA9IEdHTF9QSVhFTF9GT1JNQVRfUkdCXzg4OCwgICAgLy8gM3g4LWJpdCBSR0IKLSAgICBQSVhFTF9GT1JNQVRfUkdCXzU2NSAgICAgPSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjUsICAgIC8vIDE2LWJpdCBSR0IKLSAgICBQSVhFTF9GT1JNQVRfQkdSQV84ODg4ICAgPSBHR0xfUElYRUxfRk9STUFUX0JHUkFfODg4OCwgIC8vIDR4OC1iaXQgQkdSQQotICAgIFBJWEVMX0ZPUk1BVF9SR0JBXzU1NTEgICA9IEdHTF9QSVhFTF9GT1JNQVRfUkdCQV81NTUxLCAgLy8gMTYtYml0IEFSR0IKLSAgICBQSVhFTF9GT1JNQVRfUkdCQV80NDQ0ICAgPSBHR0xfUElYRUxfRk9STUFUX1JHQkFfNDQ0NCwgIC8vIDE2LWJpdCBBUkdCCi0gICAgUElYRUxfRk9STUFUX0FfOCAgICAgICAgID0gR0dMX1BJWEVMX0ZPUk1BVF9BXzgsICAgICAgICAvLyA4LWJpdCBBCi0gICAgUElYRUxfRk9STUFUX0xfOCAgICAgICAgID0gR0dMX1BJWEVMX0ZPUk1BVF9MXzgsICAgICAgICAvLyA4LWJpdCBMIChSPUc9Qj1MKQotICAgIFBJWEVMX0ZPUk1BVF9MQV84OCAgICAgICA9IEdHTF9QSVhFTF9GT1JNQVRfTEFfODgsICAgICAgLy8gMTYtYml0IExBCi0gICAgUElYRUxfRk9STUFUX1JHQl8zMzIgICAgID0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JfMzMyLCAgICAvLyA4LWJpdCBSR0IKLQotICAgIFBJWEVMX0ZPUk1BVF9ZQ2JDcl80MjJfU1A9IEdHTF9QSVhFTF9GT1JNQVRfWUNiQ3JfNDIyX1NQLAotICAgIFBJWEVMX0ZPUk1BVF9ZQ2JDcl80MjBfU1A9IEdHTF9QSVhFTF9GT1JNQVRfWUNiQ3JfNDIwX1NQLAotICAgIFBJWEVMX0ZPUk1BVF9ZQ2JDcl80MjJfUCA9IEdHTF9QSVhFTF9GT1JNQVRfWUNiQ3JfNDIyX1AsCi0gICAgUElYRUxfRk9STUFUX1lDYkNyXzQyMF9QID0gR0dMX1BJWEVMX0ZPUk1BVF9ZQ2JDcl80MjBfUCwKLSAgICBQSVhFTF9GT1JNQVRfWUNiQ3JfNDIyX0kgPSBHR0xfUElYRUxfRk9STUFUX1lDYkNyXzQyMl9JLAotICAgIFBJWEVMX0ZPUk1BVF9ZQ2JDcl80MjBfSSA9IEdHTF9QSVhFTF9GT1JNQVRfWUNiQ3JfNDIwX0ksCi0KLSAgICAvLyBOZXcgZm9ybWF0cyBjYW4gYmUgYWRkZWQgaWYgdGhleSdyZSBhbHNvIGRlZmluZWQgaW4KLSAgICAvLyBwaXhlbGZsaW5nZXIvZm9ybWF0LmgKLX07Ci0KLXR5cGVkZWYgaW50MzJfdCBQaXhlbEZvcm1hdDsKLQotc3RydWN0IFBpeGVsRm9ybWF0SW5mbwotewotICAgIGVudW0geyAvLyBjb21wb25lbnRzCi0gICAgICAgIEFMUEhBICAgICAgICAgICAgICAgPSAxLAotICAgICAgICBSR0IgICAgICAgICAgICAgICAgID0gMiwKLSAgICAgICAgUkdCQSAgICAgICAgICAgICAgICA9IDMsCi0gICAgICAgIExVTUlOQU5DRSAgICAgICAgICAgPSA0LAotICAgICAgICBMVU1JTkFOQ0VfQUxQSEEgICAgID0gNSwKLSAgICAgICAgWV9DQl9DUl9TUCAgICAgICAgICA9IDYsCi0gICAgICAgIFlfQ0JfQ1JfUCAgICAgICAgICAgPSA3LAotICAgICAgICBZX0NCX0NSX0kgICAgICAgICAgID0gOCwKLSAgICB9OwotCi0gICAgaW5saW5lIFBpeGVsRm9ybWF0SW5mbygpIDogdmVyc2lvbihzaXplb2YoUGl4ZWxGb3JtYXRJbmZvKSkgeyB9Ci0gICAgc2l6ZV90IGdldFNjYW5saW5lU2l6ZSh1bnNpZ25lZCBpbnQgd2lkdGgpIGNvbnN0OwotICAgIHNpemVfdCAgICAgIHZlcnNpb247Ci0gICAgUGl4ZWxGb3JtYXQgZm9ybWF0OwotICAgIHNpemVfdCAgICAgIGJ5dGVzUGVyUGl4ZWw7Ci0gICAgc2l6ZV90ICAgICAgYml0c1BlclBpeGVsOwotICAgIHVpbnQ4X3QgICAgIGhfYWxwaGE7Ci0gICAgdWludDhfdCAgICAgbF9hbHBoYTsKLSAgICB1aW50OF90ICAgICBoX3JlZDsKLSAgICB1aW50OF90ICAgICBsX3JlZDsKLSAgICB1aW50OF90ICAgICBoX2dyZWVuOwotICAgIHVpbnQ4X3QgICAgIGxfZ3JlZW47Ci0gICAgdWludDhfdCAgICAgaF9ibHVlOwotICAgIHVpbnQ4X3QgICAgIGxfYmx1ZTsKLSAgICB1aW50OF90ICAgICBjb21wb25lbnRzOwotICAgIHVpbnQ4X3QgICAgIHJlc2VydmVkMFszXTsKLSAgICB1aW50MzJfdCAgICByZXNlcnZlZDE7Ci19OwotCi0vLyBDb25zaWRlciBjYWNoaW5nIHRoZSByZXN1bHRzIG9mIHRoZXNlIGZ1bmN0aW9ucyBhcmUgdGhleSdyZSBub3QKLS8vIGd1YXJhbnRlZWQgdG8gYmUgZmFzdC4KLXNzaXplX3QgICAgIGJ5dGVzUGVyUGl4ZWwoUGl4ZWxGb3JtYXQgZm9ybWF0KTsKLXNzaXplX3QgICAgIGJpdHNQZXJQaXhlbChQaXhlbEZvcm1hdCBmb3JtYXQpOwotc3RhdHVzX3QgICAgZ2V0UGl4ZWxGb3JtYXRJbmZvKFBpeGVsRm9ybWF0IGZvcm1hdCwgUGl4ZWxGb3JtYXRJbmZvKiBpbmZvKTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIFVJX1BJWEVMRk9STUFUX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdWkvUG9pbnQuaCBiL2luY2x1ZGUvdWkvUG9pbnQuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGJiYWQxZS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3VpL1BvaW50LmgKKysrIC9kZXYvbnVsbApAQCAtMSw4OCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA2IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1VJX1BPSU5UCi0jZGVmaW5lIEFORFJPSURfVUlfUE9JTlQKLQotI2luY2x1ZGUgPHV0aWxzL1R5cGVIZWxwZXJzLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgUG9pbnQKLXsKLXB1YmxpYzoKLSAgICBpbnQgeDsKLSAgICBpbnQgeTsKLQotICAgIC8vIHdlIGRvbid0IHByb3ZpZGUgY29weS1jdG9yIGFuZCBvcGVyYXRvcj0gb24gcHVycG9zZQotICAgIC8vIGJlY2F1c2Ugd2Ugd2FudCB0aGUgY29tcGlsZXIgZ2VuZXJhdGVkIHZlcnNpb25zCi0KLSAgICAvLyBEZWZhdWx0IGNvbnN0cnVjdG9yIGRvZXNuJ3QgaW5pdGlhbGl6ZSB0aGUgUG9pbnQKLSAgICBpbmxpbmUgUG9pbnQoKQotICAgIHsKLSAgICB9Ci0KLSAgICBpbmxpbmUgUG9pbnQoaW50IF94LCBpbnQgX3kpIDogeChfeCksIHkoX3kpCi0gICAgewotICAgIH0KLQotICAgIGlubGluZSBib29sIG9wZXJhdG9yID09IChjb25zdCBQb2ludCYgcmhzKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiAoeCA9PSByaHMueCkgJiYgKHkgPT0gcmhzLnkpOwotICAgIH0KLSAgICBpbmxpbmUgYm9vbCBvcGVyYXRvciAhPSAoY29uc3QgUG9pbnQmIHJocykgY29uc3QgewotICAgICAgICByZXR1cm4gIW9wZXJhdG9yID09IChyaHMpOwotICAgIH0KLQotICAgIGlubGluZSBib29sIGlzT3JpZ2luKCkgY29uc3QgewotICAgICAgICByZXR1cm4gISh4fHkpOwotICAgIH0KLQotICAgIC8vIG9wZXJhdG9yIDwgZGVmaW5lcyBhbiBvcmRlciB3aGljaCBhbGxvd3MgdG8gdXNlIHBvaW50cyBpbiBzb3J0ZWQKLSAgICAvLyB2ZWN0b3JzLgotICAgIGJvb2wgb3BlcmF0b3IgPCAoY29uc3QgUG9pbnQmIHJocykgY29uc3QgewotICAgICAgICByZXR1cm4geTxyaHMueSB8fCAoeT09cmhzLnkgJiYgeDxyaHMueCk7Ci0gICAgfQotCi0gICAgaW5saW5lIFBvaW50JiBvcGVyYXRvciAtICgpIHsKLSAgICAgICAgeD0teDsKLSAgICAgICAgeT0teTsKLSAgICAgICAgcmV0dXJuICp0aGlzOwotICAgIH0KLSAgICAKLSAgICBpbmxpbmUgUG9pbnQmIG9wZXJhdG9yICs9IChjb25zdCBQb2ludCYgcmhzKSB7Ci0gICAgICAgIHggKz0gcmhzLng7Ci0gICAgICAgIHkgKz0gcmhzLnk7Ci0gICAgICAgIHJldHVybiAqdGhpczsKLSAgICB9Ci0gICAgaW5saW5lIFBvaW50JiBvcGVyYXRvciAtPSAoY29uc3QgUG9pbnQmIHJocykgewotICAgICAgICB4IC09IHJocy54OwotICAgICAgICB5IC09IHJocy55OwotICAgICAgICByZXR1cm4gKnRoaXM7Ci0gICAgfQotICAgIAotICAgIFBvaW50IG9wZXJhdG9yICsgKGNvbnN0IFBvaW50JiByaHMpIGNvbnN0IHsKLSAgICAgICAgcmV0dXJuIFBvaW50KHgrcmhzLngsIHkrcmhzLnkpOwotICAgIH0KLSAgICBQb2ludCBvcGVyYXRvciAtIChjb25zdCBQb2ludCYgcmhzKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiBQb2ludCh4LXJocy54LCB5LXJocy55KTsKLSAgICB9ICAgIAotfTsKLQotQU5EUk9JRF9CQVNJQ19UWVBFU19UUkFJVFMoUG9pbnQpCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX1VJX1BPSU5UCmRpZmYgLS1naXQgYS9pbmNsdWRlL3VpL1JlY3QuaCBiL2luY2x1ZGUvdWkvUmVjdC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkMjMyODQ3Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdWkvUmVjdC5oCisrKyAvZGV2L251bGwKQEAgLTEsMTUyICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDYgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfVUlfUkVDVAotI2RlZmluZSBBTkRST0lEX1VJX1JFQ1QKLQotI2luY2x1ZGUgPHV0aWxzL1R5cGVIZWxwZXJzLmg+Ci0jaW5jbHVkZSA8dWkvUG9pbnQuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBSZWN0Ci17Ci1wdWJsaWM6Ci0gICAgaW50IGxlZnQ7Ci0gICAgaW50IHRvcDsKLSAgICBpbnQgcmlnaHQ7Ci0gICAgaW50IGJvdHRvbTsKLQotICAgIC8vIHdlIGRvbid0IHByb3ZpZGUgY29weS1jdG9yIGFuZCBvcGVyYXRvcj0gb24gcHVycG9zZQotICAgIC8vIGJlY2F1c2Ugd2Ugd2FudCB0aGUgY29tcGlsZXIgZ2VuZXJhdGVkIHZlcnNpb25zCi0KLSAgICBpbmxpbmUgUmVjdCgpCi0gICAgewotICAgIH0KLQotICAgIGlubGluZSBSZWN0KGludCB3LCBpbnQgaCkKLSAgICAgICAgOiBsZWZ0KDApLCB0b3AoMCksIHJpZ2h0KHcpLCBib3R0b20oaCkKLSAgICB7Ci0gICAgfQotCi0gICAgaW5saW5lIFJlY3QoaW50IGwsIGludCB0LCBpbnQgciwgaW50IGIpCi0gICAgICAgIDogbGVmdChsKSwgdG9wKHQpLCByaWdodChyKSwgYm90dG9tKGIpCi0gICAgewotICAgIH0KLQotICAgIGlubGluZSBSZWN0KGNvbnN0IFBvaW50JiBsdCwgY29uc3QgUG9pbnQmIHJiKSAKLSAgICAgICAgOiBsZWZ0KGx0LngpLCB0b3AobHQueSksIHJpZ2h0KHJiLngpLCBib3R0b20ocmIueSkKLSAgICB7Ci0gICAgfQotCi0gICAgdm9pZCBtYWtlSW52YWxpZCgpOwotICAgIAotICAgIC8vIGEgdmFsaWQgcmVjdGFuZ2xlIGhhcyBhIG5vbiBuZWdhdGl2ZSB3aWR0aCBhbmQgaGVpZ2h0Ci0gICAgaW5saW5lIGJvb2wgaXNWYWxpZCgpIGNvbnN0IHsKLSAgICAgICAgcmV0dXJuICh3aWR0aCgpPj0wKSAmJiAoaGVpZ2h0KCk+PTApOwotICAgIH0KLQotICAgIC8vIGFuIGVtcHR5IHJlY3QgaGFzIGEgemVybyB3aWR0aCBvciBoZWlnaHQsIG9yIGlzIGludmFsaWQKLSAgICBpbmxpbmUgYm9vbCBpc0VtcHR5KCkgY29uc3QgewotICAgICAgICByZXR1cm4gKHdpZHRoKCk8PTApIHx8IChoZWlnaHQoKTw9MCk7Ci0gICAgfQotCi0gICAgaW5saW5lIHZvaWQgc2V0KGNvbnN0IFJlY3QmIHJocykgewotICAgICAgICBvcGVyYXRvciA9IChyaHMpOwotICAgIH0KLQotICAgIC8vIHJlY3RhbmdsZSdzIHdpZHRoCi0gICAgaW5saW5lIGludCB3aWR0aCgpIGNvbnN0IHsKLSAgICAgICAgcmV0dXJuIHJpZ2h0LWxlZnQ7Ci0gICAgfQotICAgIAotICAgIC8vIHJlY3RhbmdsZSdzIGhlaWdodAotICAgIGlubGluZSBpbnQgaGVpZ2h0KCkgY29uc3QgewotICAgICAgICByZXR1cm4gYm90dG9tLXRvcDsKLSAgICB9Ci0KLSAgICAvLyByZXR1cm5zIGxlZnQtdG9wIFBvaW50IG5vbi1jb25zdCByZWZlcmVuY2UsIGNhbiBiZSBhc3NpZ25lZAotICAgIGlubGluZSBQb2ludCYgbGVmdFRvcCgpIHsKLSAgICAgICAgcmV0dXJuIHJlaW50ZXJwcmV0X2Nhc3Q8UG9pbnQmPihsZWZ0KTsKLSAgICB9Ci0gICAgLy8gcmV0dXJucyByaWdodCBib3R0b20gbm9uLWNvbnN0IHJlZmVyZW5jZSwgY2FuIGJlIGFzc2lnbmVkCi0gICAgaW5saW5lIFBvaW50JiByaWdodEJvdHRvbSgpIHsKLSAgICAgICAgcmV0dXJuIHJlaW50ZXJwcmV0X2Nhc3Q8UG9pbnQmPihyaWdodCk7Ci0gICAgfQotICAgIAotICAgIC8vIHRoZSBmb2xsb3dpbmcgNCBmdW5jdGlvbnMgcmV0dXJuIHRoZSA0IGNvcm5lcnMgb2YgdGhlIHJlY3QgYXMgUG9pbnQKLSAgICBpbmxpbmUgY29uc3QgUG9pbnQmIGxlZnRUb3AoKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiByZWludGVycHJldF9jYXN0PGNvbnN0IFBvaW50Jj4obGVmdCk7Ci0gICAgfQotICAgIGlubGluZSBjb25zdCBQb2ludCYgcmlnaHRCb3R0b20oKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiByZWludGVycHJldF9jYXN0PGNvbnN0IFBvaW50Jj4ocmlnaHQpOwotICAgIH0KLSAgICBQb2ludCByaWdodFRvcCgpIGNvbnN0IHsKLSAgICAgICAgcmV0dXJuIFBvaW50KHJpZ2h0LCB0b3ApOwotICAgIH0KLSAgICBQb2ludCBsZWZ0Qm90dG9tKCkgY29uc3QgewotICAgICAgICByZXR1cm4gUG9pbnQobGVmdCwgYm90dG9tKTsKLSAgICB9Ci0KLSAgICAvLyBjb21wYXJpc29ucwotICAgIGlubGluZSBib29sIG9wZXJhdG9yID09IChjb25zdCBSZWN0JiByaHMpIGNvbnN0IHsKLSAgICAgICAgcmV0dXJuIChsZWZ0ID09IHJocy5sZWZ0KSAmJiAodG9wID09IHJocy50b3ApICYmCi0gICAgICAgICAgICAgICAocmlnaHQgPT0gcmhzLnJpZ2h0KSAmJiAoYm90dG9tID09IHJocy5ib3R0b20pOwotICAgIH0KLQotICAgIGlubGluZSBib29sIG9wZXJhdG9yICE9IChjb25zdCBSZWN0JiByaHMpIGNvbnN0IHsKLSAgICAgICAgcmV0dXJuICFvcGVyYXRvciA9PSAocmhzKTsKLSAgICB9Ci0KLSAgICAvLyBvcGVyYXRvciA8IGRlZmluZXMgYW4gb3JkZXIgd2hpY2ggYWxsb3dzIHRvIHVzZSByZWN0YW5nbGVzIGluIHNvcnRlZAotICAgIC8vIHZlY3RvcnMuCi0gICAgYm9vbCBvcGVyYXRvciA8IChjb25zdCBSZWN0JiByaHMpIGNvbnN0OwotCi0gICAgUmVjdCYgb2Zmc2V0VG9PcmlnaW4oKSB7Ci0gICAgICAgIHJpZ2h0IC09IGxlZnQ7Ci0gICAgICAgIGJvdHRvbSAtPSB0b3A7Ci0gICAgICAgIGxlZnQgPSB0b3AgPSAwOwotICAgICAgICByZXR1cm4gKnRoaXM7Ci0gICAgfQotICAgIFJlY3QmIG9mZnNldFRvKGNvbnN0IFBvaW50JiBwKSB7Ci0gICAgICAgIHJldHVybiBvZmZzZXRUbyhwLngsIHAueSk7Ci0gICAgfQotICAgIFJlY3QmIG9mZnNldEJ5KGNvbnN0IFBvaW50JiBkcCkgewotICAgICAgICByZXR1cm4gb2Zmc2V0QnkoZHAueCwgZHAueSk7Ci0gICAgfQotICAgIFJlY3QmIG9wZXJhdG9yICs9IChjb25zdCBQb2ludCYgcmhzKSB7Ci0gICAgICAgIHJldHVybiBvZmZzZXRCeShyaHMueCwgcmhzLnkpOwotICAgIH0KLSAgICBSZWN0JiBvcGVyYXRvciAtPSAoY29uc3QgUG9pbnQmIHJocykgewotICAgICAgICByZXR1cm4gb2Zmc2V0QnkoLXJocy54LCAtcmhzLnkpOwotICAgIH0KLSAgICBSZWN0IG9wZXJhdG9yICsgKGNvbnN0IFBvaW50JiByaHMpIGNvbnN0OwotICAgIFJlY3Qgb3BlcmF0b3IgLSAoY29uc3QgUG9pbnQmIHJocykgY29uc3Q7Ci0KLSAgICB2b2lkIHRyYW5zbGF0ZShpbnQgZHgsIGludCBkeSkgeyAvLyBsZWdhY3ksIGRvbid0IHVzZS4KLSAgICAgICAgb2Zmc2V0QnkoZHgsIGR5KTsKLSAgICB9Ci0gCi0gICAgUmVjdCYgICBvZmZzZXRUbyhpbnQgeCwgaW50IHkpOwotICAgIFJlY3QmICAgb2Zmc2V0QnkoaW50IHgsIGludCB5KTsKLSAgICBib29sICAgIGludGVyc2VjdChjb25zdCBSZWN0JiB3aXRoLCBSZWN0KiByZXN1bHQpIGNvbnN0OwotfTsKLQotQU5EUk9JRF9CQVNJQ19UWVBFU19UUkFJVFMoUmVjdCkKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfVUlfUkVDVApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91aS9SZWdpb24uaCBiL2luY2x1ZGUvdWkvUmVnaW9uLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDc2ODk2NzMuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91aS9SZWdpb24uaAorKysgL2Rldi9udWxsCkBAIC0xLDE3NCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1VJX1JFR0lPTl9ICi0jZGVmaW5lIEFORFJPSURfVUlfUkVHSU9OX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvVmVjdG9yLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUGFyY2VsLmg+Ci0KLSNpbmNsdWRlIDx1aS9SZWN0Lmg+Ci0KLSNpbmNsdWRlIDxoYXJkd2FyZS9jb3B5Yml0Lmg+Ci0KLSNpbmNsdWRlIDxjb3JlL1NrUmVnaW9uLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBTdHJpbmc4OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLWNsYXNzIFJlZ2lvbgotewotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgUmVnaW9uKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBSZWdpb24oY29uc3QgUmVnaW9uJiByaHMpOwotICAgIGV4cGxpY2l0ICAgICAgICAgICAgUmVnaW9uKGNvbnN0IFNrUmVnaW9uJiByaHMpOwotICAgIGV4cGxpY2l0ICAgICAgICAgICAgUmVnaW9uKGNvbnN0IFJlY3QmIHJocyk7Ci0gICAgZXhwbGljaXQgICAgICAgICAgICBSZWdpb24oY29uc3QgUGFyY2VsJiBwYXJjZWwpOwotICAgIGV4cGxpY2l0ICAgICAgICAgICAgUmVnaW9uKGNvbnN0IHZvaWQqIGJ1ZmZlcik7Ci0gICAgICAgICAgICAgICAgICAgICAgICB+UmVnaW9uKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgUmVnaW9uJiBvcGVyYXRvciA9IChjb25zdCBSZWdpb24mIHJocyk7Ci0KLSAgICBpbmxpbmUgIGJvb2wgICAgICAgIGlzRW1wdHkoKSBjb25zdCAgICAgeyByZXR1cm4gbVJlZ2lvbi5pc0VtcHR5KCk7IH0KLSAgICBpbmxpbmUgIGJvb2wgICAgICAgIGlzUmVjdCgpIGNvbnN0ICAgICAgeyByZXR1cm4gbVJlZ2lvbi5pc1JlY3QoKTsgfQotCi0gICAgICAgICAgICBSZWN0ICAgICAgICBib3VuZHMoKSBjb25zdDsKLQotICAgICAgICAgICAgY29uc3QgU2tSZWdpb24mIHRvU2tSZWdpb24oKSBjb25zdDsKLQotICAgICAgICAgICAgdm9pZCAgICAgICAgY2xlYXIoKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIHNldChjb25zdCBSZWN0JiByKTsKLSAgICAgICAgCi0gICAgICAgICAgICBSZWdpb24mICAgICBvclNlbGYoY29uc3QgUmVjdCYgcmhzKTsKLSAgICAgICAgICAgIFJlZ2lvbiYgICAgIGFuZFNlbGYoY29uc3QgUmVjdCYgcmhzKTsKLQotICAgICAgICAgICAgLy8gYm9vbGVhbiBvcGVyYXRvcnMsIGFwcGxpZWQgb24gdGhpcwotICAgICAgICAgICAgUmVnaW9uJiAgICAgb3JTZWxmKGNvbnN0IFJlZ2lvbiYgcmhzKTsKLSAgICAgICAgICAgIFJlZ2lvbiYgICAgIGFuZFNlbGYoY29uc3QgUmVnaW9uJiByaHMpOwotICAgICAgICAgICAgUmVnaW9uJiAgICAgc3VidHJhY3RTZWxmKGNvbnN0IFJlZ2lvbiYgcmhzKTsKLQotICAgICAgICAgICAgLy8gdGhlc2UgdHJhbnNsYXRlIHJocyBmaXJzdAotICAgICAgICAgICAgUmVnaW9uJiAgICAgdHJhbnNsYXRlU2VsZihpbnQgZHgsIGludCBkeSk7Ci0gICAgICAgICAgICBSZWdpb24mICAgICBvclNlbGYoY29uc3QgUmVnaW9uJiByaHMsIGludCBkeCwgaW50IGR5KTsKLSAgICAgICAgICAgIFJlZ2lvbiYgICAgIGFuZFNlbGYoY29uc3QgUmVnaW9uJiByaHMsIGludCBkeCwgaW50IGR5KTsKLSAgICAgICAgICAgIFJlZ2lvbiYgICAgIHN1YnRyYWN0U2VsZihjb25zdCBSZWdpb24mIHJocywgaW50IGR4LCBpbnQgZHkpOwotCi0gICAgICAgICAgICAvLyBib29sZWFuIG9wZXJhdG9ycwotICAgICAgICAgICAgUmVnaW9uICAgICAgbWVyZ2UoY29uc3QgUmVnaW9uJiByaHMpIGNvbnN0OwotICAgICAgICAgICAgUmVnaW9uICAgICAgaW50ZXJzZWN0KGNvbnN0IFJlZ2lvbiYgcmhzKSBjb25zdDsKLSAgICAgICAgICAgIFJlZ2lvbiAgICAgIHN1YnRyYWN0KGNvbnN0IFJlZ2lvbiYgcmhzKSBjb25zdDsKLQotICAgICAgICAgICAgLy8gdGhlc2UgdHJhbnNsYXRlIHJocyBmaXJzdAotICAgICAgICAgICAgUmVnaW9uICAgICAgdHJhbnNsYXRlKGludCBkeCwgaW50IGR5KSBjb25zdDsKLSAgICAgICAgICAgIFJlZ2lvbiAgICAgIG1lcmdlKGNvbnN0IFJlZ2lvbiYgcmhzLCBpbnQgZHgsIGludCBkeSkgY29uc3Q7Ci0gICAgICAgICAgICBSZWdpb24gICAgICBpbnRlcnNlY3QoY29uc3QgUmVnaW9uJiByaHMsIGludCBkeCwgaW50IGR5KSBjb25zdDsKLSAgICAgICAgICAgIFJlZ2lvbiAgICAgIHN1YnRyYWN0KGNvbnN0IFJlZ2lvbiYgcmhzLCBpbnQgZHgsIGludCBkeSkgY29uc3Q7Ci0KLSAgICAvLyBjb252ZW5pZW5jZSBvcGVyYXRvcnMgb3ZlcmxvYWRzCi0gICAgaW5saW5lICBSZWdpb24gICAgICBvcGVyYXRvciB8IChjb25zdCBSZWdpb24mIHJocykgY29uc3Q7Ci0gICAgaW5saW5lICBSZWdpb24gICAgICBvcGVyYXRvciAmIChjb25zdCBSZWdpb24mIHJocykgY29uc3Q7Ci0gICAgaW5saW5lICBSZWdpb24gICAgICBvcGVyYXRvciAtIChjb25zdCBSZWdpb24mIHJocykgY29uc3Q7Ci0gICAgaW5saW5lICBSZWdpb24gICAgICBvcGVyYXRvciArIChjb25zdCBQb2ludCYgcHQpIGNvbnN0OwotCi0gICAgaW5saW5lICBSZWdpb24mICAgICBvcGVyYXRvciB8PSAoY29uc3QgUmVnaW9uJiByaHMpOwotICAgIGlubGluZSAgUmVnaW9uJiAgICAgb3BlcmF0b3IgJj0gKGNvbnN0IFJlZ2lvbiYgcmhzKTsKLSAgICBpbmxpbmUgIFJlZ2lvbiYgICAgIG9wZXJhdG9yIC09IChjb25zdCBSZWdpb24mIHJocyk7Ci0gICAgaW5saW5lICBSZWdpb24mICAgICBvcGVyYXRvciArPSAoY29uc3QgUG9pbnQmIHB0KTsKLQotICAgIGNsYXNzIGl0ZXJhdG9yIHsKLSAgICAgICAgU2tSZWdpb246Okl0ZXJhdG9yICBtSXQ7Ci0gICAgcHVibGljOgotICAgICAgICBpdGVyYXRvcihjb25zdCBSZWdpb24mIHIpOwotICAgICAgICBpbmxpbmUgb3BlcmF0b3IgYm9vbCAoKSBjb25zdCB7IHJldHVybiAhZG9uZSgpOyB9Ci0gICAgICAgIGludCBpdGVyYXRlKFJlY3QqIHJlY3QpOwotICAgIHByaXZhdGU6Ci0gICAgICAgIGlubGluZSBib29sIGRvbmUoKSBjb25zdCB7Ci0gICAgICAgICAgICByZXR1cm4gY29uc3RfY2FzdDxTa1JlZ2lvbjo6SXRlcmF0b3ImPihtSXQpLmRvbmUoKTsKLSAgICAgICAgfQotICAgIH07Ci0KLSAgICAgICAgICAgIHNpemVfdCAgICAgIHJlY3RzKFZlY3RvcjxSZWN0PiYgcmVjdExpc3QpIGNvbnN0OwotCi0gICAgICAgICAgICAvLyBmbGF0dGVuL3VuZmxhdHRlbiBhIHJlZ2lvbiB0by9mcm9tIGEgUGFyY2VsCi0gICAgICAgICAgICBzdGF0dXNfdCAgICB3cml0ZShQYXJjZWwmIHBhcmNlbCkgY29uc3Q7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICByZWFkKGNvbnN0IFBhcmNlbCYgcGFyY2VsKTsKLQotICAgICAgICAgICAgLy8gZmxhdHRlbi91bmZsYXR0ZW4gYSByZWdpb24gdG8vZnJvbSBhIHJhdyBidWZmZXIKLSAgICAgICAgICAgIHNzaXplX3QgICAgIHdyaXRlKHZvaWQqIGJ1ZmZlciwgc2l6ZV90IHNpemUpIGNvbnN0OwotICAgIHN0YXRpYyAgc3NpemVfdCAgICAgd3JpdGVFbXB0eSh2b2lkKiBidWZmZXIsIHNpemVfdCBzaXplKTsKLQotICAgICAgICAgICAgc3NpemVfdCAgICAgcmVhZChjb25zdCB2b2lkKiBidWZmZXIpOwotICAgIHN0YXRpYyAgYm9vbCAgICAgICAgaXNFbXB0eSh2b2lkKiBidWZmZXIpOwotCi0gICAgdm9pZCAgICAgICAgZHVtcChTdHJpbmc4JiBvdXQsIGNvbnN0IGNoYXIqIHdoYXQsIHVpbnQzMl90IGZsYWdzPTApIGNvbnN0OwotICAgIHZvaWQgICAgICAgIGR1bXAoY29uc3QgY2hhciogd2hhdCwgdWludDMyX3QgZmxhZ3M9MCkgY29uc3Q7Ci0KLXByaXZhdGU6Ci0gICAgU2tSZWdpb24gICAgbVJlZ2lvbjsKLX07Ci0KLQotUmVnaW9uIFJlZ2lvbjo6b3BlcmF0b3IgfCAoY29uc3QgUmVnaW9uJiByaHMpIGNvbnN0IHsKLSAgICByZXR1cm4gbWVyZ2UocmhzKTsKLX0KLVJlZ2lvbiBSZWdpb246Om9wZXJhdG9yICYgKGNvbnN0IFJlZ2lvbiYgcmhzKSBjb25zdCB7Ci0gICAgcmV0dXJuIGludGVyc2VjdChyaHMpOwotfQotUmVnaW9uIFJlZ2lvbjo6b3BlcmF0b3IgLSAoY29uc3QgUmVnaW9uJiByaHMpIGNvbnN0IHsKLSAgICByZXR1cm4gc3VidHJhY3QocmhzKTsKLX0KLVJlZ2lvbiBSZWdpb246Om9wZXJhdG9yICsgKGNvbnN0IFBvaW50JiBwdCkgY29uc3QgewotICAgIHJldHVybiB0cmFuc2xhdGUocHQueCwgcHQueSk7Ci19Ci0KLQotUmVnaW9uJiBSZWdpb246Om9wZXJhdG9yIHw9IChjb25zdCBSZWdpb24mIHJocykgewotICAgIHJldHVybiBvclNlbGYocmhzKTsKLX0KLVJlZ2lvbiYgUmVnaW9uOjpvcGVyYXRvciAmPSAoY29uc3QgUmVnaW9uJiByaHMpIHsKLSAgICByZXR1cm4gYW5kU2VsZihyaHMpOwotfQotUmVnaW9uJiBSZWdpb246Om9wZXJhdG9yIC09IChjb25zdCBSZWdpb24mIHJocykgewotICAgIHJldHVybiBzdWJ0cmFjdFNlbGYocmhzKTsKLX0KLVJlZ2lvbiYgUmVnaW9uOjpvcGVyYXRvciArPSAoY29uc3QgUG9pbnQmIHB0KSB7Ci0gICAgcmV0dXJuIHRyYW5zbGF0ZVNlbGYocHQueCwgcHQueSk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3QgcmVnaW9uX2l0ZXJhdG9yIDogcHVibGljIGNvcHliaXRfcmVnaW9uX3QgewotICAgIHJlZ2lvbl9pdGVyYXRvcihjb25zdCBSZWdpb24mIHJlZ2lvbikgOiBpKHJlZ2lvbikgewotICAgICAgICB0aGlzLT5uZXh0ID0gaXRlcmF0ZTsKLSAgICB9Ci1wcml2YXRlOgotICAgIHN0YXRpYyBpbnQgaXRlcmF0ZShjb3B5Yml0X3JlZ2lvbl90IGNvbnN0ICogc2VsZiwgY29weWJpdF9yZWN0X3QqIHJlY3QpIHsKLSAgICAgICAgcmV0dXJuIHN0YXRpY19jYXN0PGNvbnN0IHJlZ2lvbl9pdGVyYXRvcio+KHNlbGYpCi0gICAgICAgIC0+aS5pdGVyYXRlKHJlaW50ZXJwcmV0X2Nhc3Q8UmVjdCo+KHJlY3QpKTsKLSAgICB9Ci0gICAgbXV0YWJsZSBSZWdpb246Oml0ZXJhdG9yIGk7Ci19OwotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9VSV9SRUdJT05fSAotCmRpZmYgLS1naXQgYS9pbmNsdWRlL3VpL1N1cmZhY2UuaCBiL2luY2x1ZGUvdWkvU3VyZmFjZS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzMzk1M2E5Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdWkvU3VyZmFjZS5oCisrKyAvZGV2L251bGwKQEAgLTEsMTM3ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfVUlfU1VSRkFDRV9ICi0jZGVmaW5lIEFORFJPSURfVUlfU1VSRkFDRV9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLSNpbmNsdWRlIDx1aS9JU3VyZmFjZS5oPgotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0jaW5jbHVkZSA8dWkvUmVnaW9uLmg+Ci0jaW5jbHVkZSA8dWkvSVN1cmZhY2VGbGluZ2VyQ2xpZW50Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIFJlY3Q7Ci1jbGFzcyBTdXJmYWNlQ29tcG9zZXJDbGllbnQ7Ci0KLWNsYXNzIFN1cmZhY2UgOiBwdWJsaWMgUmVmQmFzZQotewotCi1wdWJsaWM6Ci0gICAgc3RydWN0IFN1cmZhY2VJbmZvIHsKLSAgICAgICAgdWludDMyX3QgICAgdzsKLSAgICAgICAgdWludDMyX3QgICAgaDsKLSAgICAgICAgdWludDMyX3QgICAgYnByOwotICAgICAgICBQaXhlbEZvcm1hdCBmb3JtYXQ7Ci0gICAgICAgIHZvaWQqICAgICAgIGJpdHM7Ci0gICAgICAgIHZvaWQqICAgICAgIGJhc2U7Ci0gICAgICAgIHVpbnQzMl90ICAgIHJlc2VydmVkWzJdOwotICAgIH07Ci0KLSAgICBib29sICAgICAgICBpc1ZhbGlkKCkgY29uc3QgeyByZXR1cm4gdGhpcyAmJiBtVG9rZW4+PTAgJiYgbUNsaWVudCE9MDsgfQotICAgIFN1cmZhY2VJRCAgIElEKCkgY29uc3QgICAgICB7IHJldHVybiBtVG9rZW47IH0KLQotICAgIHN0YXR1c190ICAgIGxvY2soU3VyZmFjZUluZm8qIGluZm8sIGJvb2wgYmxvY2tpbmcgPSB0cnVlKTsKLSAgICBzdGF0dXNfdCAgICBsb2NrKFN1cmZhY2VJbmZvKiBpbmZvLCBSZWdpb24qIGRpcnR5LCBib29sIGJsb2NraW5nID0gdHJ1ZSk7Ci0gICAgc3RhdHVzX3QgICAgdW5sb2NrQW5kUG9zdCgpOwotICAgIHN0YXR1c190ICAgIHVubG9jaygpOwotCi0gICAgdm9pZCogICAgICAgaGVhcEJhc2UoaW50IGkpIGNvbnN0OwotICAgIHVpbnQzMl90ICAgIGdldEZsYWdzKCkgY29uc3QgeyByZXR1cm4gbUZsYWdzOyB9Ci0KLSAgICAvLyBzZXRTd2FwUmVjdGFuZ2xlKCkgaXMgbWFpbmx5IHVzZWQgYnkgRUdMCi0gICAgdm9pZCAgICAgICAgc2V0U3dhcFJlY3RhbmdsZShjb25zdCBSZWN0JiByKTsKLSAgICBjb25zdCBSZWN0JiBzd2FwUmVjdGFuZ2xlKCkgY29uc3Q7Ci0gICAgc3RhdHVzX3QgICAgbmV4dEJ1ZmZlcihTdXJmYWNlSW5mbyogaW5mbyk7Ci0KLSAgICBzcDxTdXJmYWNlPiAgICAgICAgIGR1cCgpIGNvbnN0OwotICAgIHN0YXRpYyBzcDxTdXJmYWNlPiAgcmVhZEZyb21QYXJjZWwoUGFyY2VsKiBwYXJjZWwpOwotICAgIHN0YXRpYyBzdGF0dXNfdCAgICAgd3JpdGVUb1BhcmNlbChjb25zdCBzcDxTdXJmYWNlPiYgc3VyZmFjZSwgUGFyY2VsKiBwYXJjZWwpOwotICAgIHN0YXRpYyBib29sICAgICAgICAgaXNTYW1lU3VyZmFjZShjb25zdCBzcDxTdXJmYWNlPiYgbGhzLCBjb25zdCBzcDxTdXJmYWNlPiYgcmhzKTsKLQotICAgIHN0YXR1c190ICAgIHNldExheWVyKGludDMyX3QgbGF5ZXIpOwotICAgIHN0YXR1c190ICAgIHNldFBvc2l0aW9uKGludDMyX3QgeCwgaW50MzJfdCB5KTsKLSAgICBzdGF0dXNfdCAgICBzZXRTaXplKHVpbnQzMl90IHcsIHVpbnQzMl90IGgpOwotICAgIHN0YXR1c190ICAgIGhpZGUoKTsKLSAgICBzdGF0dXNfdCAgICBzaG93KGludDMyX3QgbGF5ZXIgPSAtMSk7Ci0gICAgc3RhdHVzX3QgICAgZnJlZXplKCk7Ci0gICAgc3RhdHVzX3QgICAgdW5mcmVlemUoKTsKLSAgICBzdGF0dXNfdCAgICBzZXRGbGFncyh1aW50MzJfdCBmbGFncywgdWludDMyX3QgbWFzayk7Ci0gICAgc3RhdHVzX3QgICAgc2V0VHJhbnNwYXJlbnRSZWdpb25IaW50KGNvbnN0IFJlZ2lvbiYgdHJhbnNwYXJlbnQpOwotICAgIHN0YXR1c190ICAgIHNldEFscGhhKGZsb2F0IGFscGhhPTEuMGYpOwotICAgIHN0YXR1c190ICAgIHNldE1hdHJpeChmbG9hdCBkc2R4LCBmbG9hdCBkdGR4LCBmbG9hdCBkc2R5LCBmbG9hdCBkdGR5KTsKLSAgICBzdGF0dXNfdCAgICBzZXRGcmVlemVUaW50KHVpbnQzMl90IHRpbnQpOwotCi0gICAgdWludDMyX3QgICAgZ2V0SWRlbnRpdHkoKSBjb25zdCB7IHJldHVybiBtSWRlbnRpdHk7IH0KLXByaXZhdGU6Ci0gICAgZnJpZW5kIGNsYXNzIFN1cmZhY2VDb21wb3NlckNsaWVudDsKLQotICAgIC8vIGNhbWVyYSBhbmQgY2FtY29yZGVyIG5lZWQgYWNjZXNzIHRvIHRoZSBJU3VyZmFjZSBiaW5kZXIgaW50ZXJmYWNlIGZvciBwcmV2aWV3Ci0gICAgZnJpZW5kIGNsYXNzIENhbWVyYTsKLSAgICBmcmllbmQgY2xhc3MgTWVkaWFSZWNvcmRlcjsKLSAgICAvLyBtZWRpYXBsYXllciBuZWVkcyBhY2Nlc3MgdG8gSVN1cmZhY2UgZm9yIGRpc3BsYXkKLSAgICBmcmllbmQgY2xhc3MgTWVkaWFQbGF5ZXI7Ci0gICAgZnJpZW5kIGNsYXNzIFRlc3Q7Ci0gICAgY29uc3Qgc3A8SVN1cmZhY2U+JiBnZXRJU3VyZmFjZSgpIGNvbnN0IHsgcmV0dXJuIG1TdXJmYWNlOyB9Ci0KLSAgICAvLyBjYW4ndCBiZSBjb3BpZWQKLSAgICBTdXJmYWNlJiBvcGVyYXRvciA9IChTdXJmYWNlJiByaHMpOwotICAgIFN1cmZhY2UoY29uc3QgU3VyZmFjZSYgcmhzKTsKLQotICAgIFN1cmZhY2UoY29uc3Qgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiYgY2xpZW50LAotICAgICAgICAgICAgY29uc3Qgc3A8SVN1cmZhY2U+JiBzdXJmYWNlLAotICAgICAgICAgICAgY29uc3QgSVN1cmZhY2VGbGluZ2VyQ2xpZW50OjpzdXJmYWNlX2RhdGFfdCYgZGF0YSwKLSAgICAgICAgICAgIHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIFBpeGVsRm9ybWF0IGZvcm1hdCwgdWludDMyX3QgZmxhZ3MsCi0gICAgICAgICAgICBib29sIG93bmVyID0gdHJ1ZSk7Ci0KLSAgICBTdXJmYWNlKFN1cmZhY2UgY29uc3QqIHJocyk7Ci0KLSAgICB+U3VyZmFjZSgpOwotCi0gICAgUmVnaW9uIGRpcnR5UmVnaW9uKCkgY29uc3Q7Ci0gICAgdm9pZCBzZXREaXJ0eVJlZ2lvbihjb25zdCBSZWdpb24mIHJlZ2lvbikgY29uc3Q7Ci0KLSAgICAvLyB0aGlzIGxvY2tzIHByb3RlY3RzIGNhbGxzIHRvIGxvY2tTdXJmYWNlKCkgLyB1bmxvY2tTdXJmYWNlKCkKLSAgICAvLyBhbmQgaXMgY2FsbGVkIGJ5IFN1cmZhY2VDb21wb3NlckNsaWVudC4KLSAgICBNdXRleCYgZ2V0TG9jaygpIGNvbnN0IHsgcmV0dXJuIG1TdXJmYWNlTG9jazsgfQotCi0gICAgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiAgIG1DbGllbnQ7Ci0gICAgc3A8SVN1cmZhY2U+ICAgICAgICAgICAgICAgIG1TdXJmYWNlOwotICAgIHNwPElNZW1vcnlIZWFwPiAgICAgICAgICAgICBtSGVhcFsyXTsKLSAgICBTdXJmYWNlSUQgICAgICAgICAgICAgICAgICAgbVRva2VuOwotICAgIHVpbnQzMl90ICAgICAgICAgICAgICAgICAgICBtSWRlbnRpdHk7Ci0gICAgUGl4ZWxGb3JtYXQgICAgICAgICAgICAgICAgIG1Gb3JtYXQ7Ci0gICAgdWludDMyX3QgICAgICAgICAgICAgICAgICAgIG1GbGFnczsKLSAgICBjb25zdCBib29sICAgICAgICAgICAgICAgICAgbU93bmVyOwotICAgIG11dGFibGUgdm9pZCogICAgICAgICAgICAgICBtU3VyZmFjZUhlYXBCYXNlWzJdOwotICAgIG11dGFibGUgUmVnaW9uICAgICAgICAgICAgICBtRGlydHlSZWdpb247Ci0gICAgbXV0YWJsZSBSZWN0ICAgICAgICAgICAgICAgIG1Td2FwUmVjdGFuZ2xlOwotICAgIG11dGFibGUgdWludDhfdCAgICAgICAgICAgICBtQmFja2J1ZmZlckluZGV4OwotICAgIG11dGFibGUgTXV0ZXggICAgICAgICAgICAgICBtU3VyZmFjZUxvY2s7Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9VSV9TVVJGQUNFX0gKLQpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91aS9TdXJmYWNlQ29tcG9zZXJDbGllbnQuaCBiL2luY2x1ZGUvdWkvU3VyZmFjZUNvbXBvc2VyQ2xpZW50LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDVkOTIyMmQuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91aS9TdXJmYWNlQ29tcG9zZXJDbGllbnQuaAorKysgL2Rldi9udWxsCkBAIC0xLDE3OSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1NVUkZBQ0VfQ09NUE9TRVJfQ0xJRU5UX0gKLSNkZWZpbmUgQU5EUk9JRF9TVVJGQUNFX0NPTVBPU0VSX0NMSUVOVF9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL1NvcnRlZFZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL0tleWVkVmVjdG9yLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUmVmQmFzZS5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLQotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0jaW5jbHVkZSA8dWkvSVN1cmZhY2VDb21wb3Nlci5oPgotI2luY2x1ZGUgPHVpL1JlZ2lvbi5oPgotI2luY2x1ZGUgPHVpL1N1cmZhY2UuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgUmVnaW9uOwotY2xhc3MgU3VyZmFjZUZsaW5nZXJTeW5jaHJvOwotc3RydWN0IHBlcl9jbGllbnRfY2Jsa190Owotc3RydWN0IGxheWVyX2NibGtfdDsKLQotY2xhc3MgU3VyZmFjZUNvbXBvc2VyQ2xpZW50IDogdmlydHVhbCBwdWJsaWMgUmVmQmFzZQotewotcHVibGljOiAgICAKLSAgICAgICAgICAgICAgICBTdXJmYWNlQ29tcG9zZXJDbGllbnQoKTsKLSAgICB2aXJ0dWFsICAgICB+U3VyZmFjZUNvbXBvc2VyQ2xpZW50KCk7Ci0KLSAgICAvLyBBbHdheXMgbWFrZSBzdXJlIHdlIGNvdWxkIGluaXRpYWxpemUKLSAgICBzdGF0dXNfdCAgICBpbml0Q2hlY2soKSBjb25zdDsKLQotICAgIC8vIFJldHVybiB0aGUgY29ubmVjdGlvbiBvZiB0aGlzIGNsaWVudAotICAgIHNwPElCaW5kZXI+IGNvbm5lY3Rpb24oKSBjb25zdDsKLSAgICAKLSAgICAvLyBSZXRyaWV2ZSBhIGNsaWVudCBmb3IgYW4gZXhpc3RpbmcgY29ubmVjdGlvbi4KLSAgICBzdGF0aWMgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PgotICAgICAgICAgICAgICAgIGNsaWVudEZvckNvbm5lY3Rpb24oY29uc3Qgc3A8SUJpbmRlcj4mIGNvbm4pOwotCi0gICAgLy8gRm9yY2libHkgcmVtb3ZlIGNvbm5lY3Rpb24gYmVmb3JlIGFsbCByZWZlcmVuY2VzIGhhdmUgZ29uZSBhd2F5LgotICAgIHZvaWQgICAgICAgIGRpc3Bvc2UoKTsKLQotICAgIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICAgIC8vIHN1cmZhY2UgY3JlYXRpb24gLyBkZXN0cnVjdGlvbgotCi0gICAgLy8hIENyZWF0ZSBhIHN1cmZhY2UKLSAgICBzcDxTdXJmYWNlPiAgIGNyZWF0ZVN1cmZhY2UoCi0gICAgICAgICAgICBpbnQgcGlkLCAgICAgICAgICAgIC8vITwgcGlkIG9mIHRoZSBwcm9jZXNzIHRoZSBzdXJmYWNlYyBpcyBmb3IKLSAgICAgICAgICAgIERpc3BsYXlJRCBkaXNwbGF5LCAgLy8hPCBEaXNwbGF5IHRvIGNyZWF0ZSB0aGlzIHN1cmZhY2Ugb24KLSAgICAgICAgICAgIHVpbnQzMl90IHcsICAgICAgICAgLy8hPCB3aWR0aCBpbiBwaXhlbAotICAgICAgICAgICAgdWludDMyX3QgaCwgICAgICAgICAvLyE8IGhlaWdodCBpbiBwaXhlbAotICAgICAgICAgICAgUGl4ZWxGb3JtYXQgZm9ybWF0LCAvLyE8IHBpeGVsLWZvcm1hdCBkZXNpcmVkCi0gICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDAgIC8vITwgdXNhZ2UgZmxhZ3MKLSAgICApOwotCi0gICAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0gICAgLy8gQ29tcG9zZXIgcGFyYW1ldGVycwotICAgIC8vIEFsbCBjb21wb3NlciBwYXJhbWV0ZXJzIG11c3QgYmUgY2hhbmdlZCB3aXRoaW4gYSB0cmFuc2FjdGlvbgotICAgIC8vIHNldmVyYWwgc3VyZmFjZXMgY2FuIGJlIHVwZGF0ZWQgaW4gb25lIHRyYW5zYWN0aW9uLCBhbGwgY2hhbmdlcyBhcmUKLSAgICAvLyBjb21taXR0ZWQgYXQgb25jZSB3aGVuIHRoZSB0cmFuc2FjdGlvbiBpcyBjbG9zZWQuCi0gICAgLy8gQ2xvc2VUcmFuc2FjdGlvbigpIHVzdWFsbHkgcmVxdWlyZXMgYW4gSVBDIHdpdGggdGhlIHNlcnZlci4KLSAgICAKLSAgICAvLyEgT3BlbiBhIGNvbXBvc2VyIHRyYW5zYWN0aW9uCi0gICAgc3RhdHVzX3QgICAgb3BlblRyYW5zYWN0aW9uKCk7Ci0KLSAgICAvLyEgY29tbWl0IHRoZSB0cmFuc2FjdGlvbgotICAgIHN0YXR1c190ICAgIGNsb3NlVHJhbnNhY3Rpb24oKTsKLQotICAgIC8vISBPcGVuIGEgY29tcG9zZXIgdHJhbnNhY3Rpb24gb24gYWxsIGFjdGl2ZSBTdXJmYWNlQ29tcG9zZXJDbGllbnRzLgotICAgIHN0YXRpYyB2b2lkIG9wZW5HbG9iYWxUcmFuc2FjdGlvbigpOwotICAgICAgICAKLSAgICAvLyEgQ2xvc2UgYSBjb21wb3NlciB0cmFuc2FjdGlvbiBvbiBhbGwgYWN0aXZlIFN1cmZhY2VDb21wb3NlckNsaWVudHMuCi0gICAgc3RhdGljIHZvaWQgY2xvc2VHbG9iYWxUcmFuc2FjdGlvbigpOwotICAgIAotICAgIC8vISBGcmVlemUgdGhlIHNwZWNpZmllZCBkaXNwbGF5IGJ1dCBub3QgdHJhbnNhY3Rpb25zLgotICAgIHN0YXRpYyBzdGF0dXNfdCBmcmVlemVEaXNwbGF5KERpc3BsYXlJRCBkcHksIHVpbnQzMl90IGZsYWdzID0gMCk7Ci0gICAgICAgIAotICAgIC8vISBSZXN1bWUgdXBkYXRlcyBvbiB0aGUgc3BlY2lmaWVkIGRpc3BsYXkuCi0gICAgc3RhdGljIHN0YXR1c190IHVuZnJlZXplRGlzcGxheShEaXNwbGF5SUQgZHB5LCB1aW50MzJfdCBmbGFncyA9IDApOwotCi0gICAgLy8hIFNldCB0aGUgb3JpZW50YXRpb24gb2YgdGhlIGdpdmVuIGRpc3BsYXkKLSAgICBzdGF0aWMgaW50IHNldE9yaWVudGF0aW9uKERpc3BsYXlJRCBkcHksIGludCBvcmllbnRhdGlvbik7Ci0KLSAgICAvLyBRdWVyeSB0aGUgbnVtYmVyIG9mIGRpc3BsYXlzCi0gICAgc3RhdGljIHNzaXplX3QgZ2V0TnVtYmVyT2ZEaXNwbGF5cygpOwotCi0gICAgLy8gR2V0IGluZm9ybWF0aW9uIGFib3V0IGEgZGlzcGxheQotICAgIHN0YXRpYyBzdGF0dXNfdCBnZXREaXNwbGF5SW5mbyhEaXNwbGF5SUQgZHB5LCBEaXNwbGF5SW5mbyogaW5mbyk7Ci0gICAgc3RhdGljIHNzaXplX3QgZ2V0RGlzcGxheVdpZHRoKERpc3BsYXlJRCBkcHkpOwotICAgIHN0YXRpYyBzc2l6ZV90IGdldERpc3BsYXlIZWlnaHQoRGlzcGxheUlEIGRweSk7Ci0gICAgc3RhdGljIHNzaXplX3QgZ2V0RGlzcGxheU9yaWVudGF0aW9uKERpc3BsYXlJRCBkcHkpOwotCi0KLXByaXZhdGU6Ci0gICAgZnJpZW5kIGNsYXNzIFN1cmZhY2U7Ci0gICAgCi0gICAgU3VyZmFjZUNvbXBvc2VyQ2xpZW50KGNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBzbSwgCi0gICAgICAgICAgICBjb25zdCBzcDxJQmluZGVyPiYgY29ubik7Ci0KLSAgICBzdGF0dXNfdCAgICBoaWRlKFN1cmZhY2UqIHN1cmZhY2UpOwotICAgIHN0YXR1c190ICAgIHNob3coU3VyZmFjZSogc3VyZmFjZSwgaW50MzJfdCBsYXllciA9IC0xKTsKLSAgICBzdGF0dXNfdCAgICBmcmVlemUoU3VyZmFjZSogc3VyZmFjZSk7Ci0gICAgc3RhdHVzX3QgICAgdW5mcmVlemUoU3VyZmFjZSogc3VyZmFjZSk7Ci0gICAgc3RhdHVzX3QgICAgc2V0RmxhZ3MoU3VyZmFjZSogc3VyZmFjZSwgdWludDMyX3QgZmxhZ3MsIHVpbnQzMl90IG1hc2spOwotICAgIHN0YXR1c190ICAgIHNldFRyYW5zcGFyZW50UmVnaW9uSGludChTdXJmYWNlKiBzdXJmYWNlLCBjb25zdCBSZWdpb24mIHRyYW5zcGFyZW50KTsKLSAgICBzdGF0dXNfdCAgICBzZXRMYXllcihTdXJmYWNlKiBzdXJmYWNlLCBpbnQzMl90IGxheWVyKTsKLSAgICBzdGF0dXNfdCAgICBzZXRBbHBoYShTdXJmYWNlKiBzdXJmYWNlLCBmbG9hdCBhbHBoYT0xLjBmKTsKLSAgICBzdGF0dXNfdCAgICBzZXRGcmVlemVUaW50KFN1cmZhY2UqIHN1cmZhY2UsIHVpbnQzMl90IHRpbnQpOwotICAgIHN0YXR1c190ICAgIHNldE1hdHJpeChTdXJmYWNlKiBzdXJmYWNlLCBmbG9hdCBkc2R4LCBmbG9hdCBkdGR4LCBmbG9hdCBkc2R5LCBmbG9hdCBkdGR5KTsKLSAgICBzdGF0dXNfdCAgICBzZXRQb3NpdGlvbihTdXJmYWNlKiBzdXJmYWNlLCBpbnQzMl90IHgsIGludDMyX3QgeSk7Ci0gICAgc3RhdHVzX3QgICAgc2V0U2l6ZShTdXJmYWNlKiBzdXJmYWNlLCB1aW50MzJfdCB3LCB1aW50MzJfdCBoKTsKLSAgICAKLSAgICAvLyEgVW5sb2NrIHRoZSBzdXJmYWNlLCBhbmQgc3BlY2lmeSB0aGUgZGlydHkgcmVnaW9uIGlmIGFueQotICAgIHN0YXR1c190ICAgIHVubG9ja0FuZFBvc3RTdXJmYWNlKFN1cmZhY2UqIHN1cmZhY2UpOwotICAgIHN0YXR1c190ICAgIHVubG9ja1N1cmZhY2UoU3VyZmFjZSogc3VyZmFjZSk7Ci0KLSAgICBzdGF0dXNfdCAgICBsb2NrU3VyZmFjZShTdXJmYWNlKiBzdXJmYWNlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN1cmZhY2U6OlN1cmZhY2VJbmZvKiBpbmZvLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIFJlZ2lvbiogZGlydHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYm9vbCBibG9ja2luZyA9IHRydWUpOwotCi0gICAgc3RhdHVzX3QgICAgbmV4dEJ1ZmZlcihTdXJmYWNlKiBzdXJmYWNlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN1cmZhY2U6OlN1cmZhY2VJbmZvKiBpbmZvKTsKLQotICAgIHN0YXR1c190ICAgIGRlc3Ryb3lTdXJmYWNlKFN1cmZhY2VJRCBzaWQpOwotCi0gICAgdm9pZCAgICAgICAgX2luaXQoY29uc3Qgc3A8SVN1cmZhY2VDb21wb3Nlcj4mIHNtLAotICAgICAgICAgICAgICAgICAgICBjb25zdCBzcDxJU3VyZmFjZUZsaW5nZXJDbGllbnQ+JiBjb25uKTsKLSAgICB2b2lkICAgICAgICBfc2lnbmFsX3NlcnZlcigpOwotICAgIHN0YXRpYyB2b2lkIF9zZW5kX2RpcnR5X3JlZ2lvbihsYXllcl9jYmxrX3QqIGxjYmxrLCBjb25zdCBSZWdpb24mIGRpcnR5KTsKLQotICAgIGlubGluZSBsYXllcl9zdGF0ZV90KiAgIF9nZXRfc3RhdGVfbChjb25zdCBzcDxTdXJmYWNlPiYgc3VyZmFjZSk7Ci0gICAgbGF5ZXJfc3RhdGVfdCogICAgICAgICAgX2xvY2tMYXllclN0YXRlKGNvbnN0IHNwPFN1cmZhY2U+JiBzdXJmYWNlKTsKLSAgICBpbmxpbmUgdm9pZCAgICAgICAgICAgICBfdW5sb2NrTGF5ZXJTdGF0ZSgpOwotCi0gICAgc3RhdHVzX3QgdmFsaWRhdGVTdXJmYWNlKAotICAgICAgICAgICAgcGVyX2NsaWVudF9jYmxrX3QgY29uc3QqIGNibGssIFN1cmZhY2UgY29uc3QgKiBzdXJmYWNlKTsKLQotICAgIHZvaWQgcGluSGVhcChjb25zdCBzcDxJTWVtb3J5SGVhcD4mIGhlYXApOwotCi0gICAgbXV0YWJsZSAgICAgTXV0ZXggICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbUxvY2s7Ci0gICAgICAgICAgICAgICAgbGF5ZXJfc3RhdGVfdCogICAgICAgICAgICAgICAgICAgICAgbVByZWJ1aWx0TGF5ZXJTdGF0ZTsKLSAgICAgICAgICAgICAgICBTb3J0ZWRWZWN0b3I8bGF5ZXJfc3RhdGVfdD4gICAgICAgICBtU3RhdGVzOwotICAgICAgICAgICAgICAgIGludDMyX3QgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1UcmFuc2FjdGlvbk9wZW47Ci0KLSAgICAgICAgICAgICAgICAvLyB0aGVzZSBkb24ndCBuZWVkIHRvIGJlIHByb3RlY3RlZCBiZWNhdXNlIHRoZXkgbmV2ZXIgY2hhbmdlCi0gICAgICAgICAgICAgICAgLy8gYWZ0ZXIgYXNzaWdubWVudAotICAgICAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgICAgICAgICBtU3RhdHVzOwotICAgICAgICAgICAgICAgIHBlcl9jbGllbnRfY2Jsa190KiAgICAgICAgICBtQ29udHJvbDsKLSAgICAgICAgICAgICAgICBzcDxJTWVtb3J5PiAgICAgICAgICAgICAgICAgbUNvbnRyb2xNZW1vcnk7Ci0gICAgICAgICAgICAgICAgc3A8SVN1cmZhY2VGbGluZ2VyQ2xpZW50PiAgIG1DbGllbnQ7Ci0gICAgICAgICAgICAgICAgc3A8SU1lbW9yeUhlYXA+ICAgICAgICAgICAgIG1TdXJmYWNlSGVhcDsKLSAgICAgICAgICAgICAgICB1aW50OF90KiAgICAgICAgICAgICAgICAgICAgbVN1cmZhY2VIZWFwQmFzZTsKLSAgICAgICAgICAgICAgICB2b2lkKiAgICAgICAgICAgICAgICAgICAgICAgbUdMOwotICAgICAgICAgICAgICAgIFN1cmZhY2VGbGluZ2VyU3luY2hybyogICAgICBtU2lnbmFsU2VydmVyOwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfU1VSRkFDRV9DT01QT1NFUl9DTElFTlRfSAotCmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzLmggYi9pbmNsdWRlL3V0aWxzLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDMwNjQ4YjEuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy5oCisrKyAvZGV2L251bGwKQEAgLTEsMzMgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBIYW5keSB1dGlsaXR5IGZ1bmN0aW9ucyBhbmQgcG9ydGFiaWxpdHkgY29kZS4gIFRoaXMgZmlsZSBpbmNsdWRlcyBhbGwKLS8vIG9mIHRoZSBnZW5lcmFsbHktdXNlZnVsIGhlYWRlcnMgaW4gdGhlICJ1dGlscyIgZGlyZWN0b3J5LgotLy8KLSNpZm5kZWYgX0xJQlNfVVRJTFNfSAotI2RlZmluZSBfTElCU19VVElMU19ICi0KLSNpbmNsdWRlIDx1dGlscy9wb3J0ZWQuaD4KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvVGltZXJzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTGlzdC5oPgotI2luY2x1ZGUgPHV0aWxzL3N0cmluZ19hcnJheS5oPgotI2luY2x1ZGUgPHV0aWxzL21pc2MuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLQotI2VuZGlmIC8vIF9MSUJTX1VUSUxTX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvQW5kcm9pZFVuaWNvZGUuaCBiL2luY2x1ZGUvdXRpbHMvQW5kcm9pZFVuaWNvZGUuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNTYzZmNkMC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0FuZHJvaWRVbmljb2RlLmgKKysrIC9kZXYvbnVsbApAQCAtMSwyNTUgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0KLSNpZm5kZWYgQU5EUk9JRF9VTklDT0RFX0gKLSNkZWZpbmUgQU5EUk9JRF9VTklDT0RFX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jZGVmaW5lIFJFUExBQ0VNRU5UX0NIQVIgKDB4RkZGRCkKLQotLy8gdGhpcyBwYXJ0IG9mIGNvZGUgaXMgY29waWVkIGZyb20gdW1hY2hpbmUuaCB1bmRlciBJQ1UKLS8qKgotICogRGVmaW5lIFVDaGFyMzIgYXMgYSB0eXBlIGZvciBzaW5nbGUgVW5pY29kZSBjb2RlIHBvaW50cy4KLSAqIFVDaGFyMzIgaXMgYSBzaWduZWQgMzItYml0IGludGVnZXIgKHNhbWUgYXMgaW50MzJfdCkuCi0gKgotICogVGhlIFVuaWNvZGUgY29kZSBwb2ludCByYW5nZSBpcyAwLi4weDEwZmZmZi4KLSAqIEFsbCBvdGhlciB2YWx1ZXMgKG5lZ2F0aXZlIG9yID49MHgxMTAwMDApIGFyZSBpbGxlZ2FsIGFzIFVuaWNvZGUgY29kZSBwb2ludHMuCi0gKiBUaGV5IG1heSBiZSB1c2VkIGFzIHNlbnRpbmVsIHZhbHVlcyB0byBpbmRpY2F0ZSAiZG9uZSIsICJlcnJvciIKLSAqIG9yIHNpbWlsYXIgbm9uLWNvZGUgcG9pbnQgY29uZGl0aW9ucy4KLSAqCi0gKiBAc3RhYmxlIElDVSAyLjQKLSAqLwotdHlwZWRlZiBpbnQzMl90IFVDaGFyMzI7Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotICAgIGNsYXNzIEVuY29kaW5nOwotICAgIC8qKgotICAgICAqIFxjbGFzcyBVbmljb2RlCi0gICAgICoKLSAgICAgKiBIZWxwZXIgY2xhc3MgZm9yIGdldHRpbmcgcHJvcGVydGllcyBvZiBVbmljb2RlIGNoYXJhY3RlcnMuIENoYXJhY3RlcnMKLSAgICAgKiBjYW4gaGF2ZSBvbmUgb2YgdGhlIHR5cGVzIGxpc3RlZCBpbiBDaGFyVHlwZSBhbmQgZWFjaCBjaGFyYWN0ZXIgY2FuIGhhdmUgdGhlCi0gICAgICogZGlyZWN0aW9uYWxpdHkgb2YgRGlyZWN0aW9uLgotICAgICAqLwotICAgIGNsYXNzIFVuaWNvZGUKLSAgICB7Ci0gICAgcHVibGljOgotICAgICAgICAvKioKLSAgICAgICAgICogRGlyZWN0aW9ucyBzcGVjaWZpZWQgaW4gdGhlIFVuaWNvZGUgc3RhbmRhcmQuIFRoZXNlIGRpcmVjdGlvbnMgbWFwIGRpcmVjdGx5Ci0gICAgICAgICAqIHRvIGphdmEubGFuZy5DaGFyYWN0ZXIuCi0gICAgICAgICAqLwotICAgICAgICBlbnVtIERpcmVjdGlvbiB7Ci0gICAgICAgICAgICBESVJFQ1RJT05BTElUWV9VTkRFRklORUQgPSAtMSwKLSAgICAgICAgICAgIERJUkVDVElPTkFMSVRZX0xFRlRfVE9fUklHSFQsCi0gICAgICAgICAgICBESVJFQ1RJT05BTElUWV9SSUdIVF9UT19MRUZULAotICAgICAgICAgICAgRElSRUNUSU9OQUxJVFlfUklHSFRfVE9fTEVGVF9BUkFCSUMsCi0gICAgICAgICAgICBESVJFQ1RJT05BTElUWV9FVVJPUEVBTl9OVU1CRVIsCi0gICAgICAgICAgICBESVJFQ1RJT05BTElUWV9FVVJPUEVBTl9OVU1CRVJfU0VQQVJBVE9SLAotICAgICAgICAgICAgRElSRUNUSU9OQUxJVFlfRVVST1BFQU5fTlVNQkVSX1RFUk1JTkFUT1IsCi0gICAgICAgICAgICBESVJFQ1RJT05BTElUWV9BUkFCSUNfTlVNQkVSLAotICAgICAgICAgICAgRElSRUNUSU9OQUxJVFlfQ09NTU9OX05VTUJFUl9TRVBBUkFUT1IsCi0gICAgICAgICAgICBESVJFQ1RJT05BTElUWV9OT05TUEFDSU5HX01BUkssCi0gICAgICAgICAgICBESVJFQ1RJT05BTElUWV9CT1VOREFSWV9ORVVUUkFMLAotICAgICAgICAgICAgRElSRUNUSU9OQUxJVFlfUEFSQUdSQVBIX1NFUEFSQVRPUiwKLSAgICAgICAgICAgIERJUkVDVElPTkFMSVRZX1NFR01FTlRfU0VQQVJBVE9SLAotICAgICAgICAgICAgRElSRUNUSU9OQUxJVFlfV0hJVEVTUEFDRSwKLSAgICAgICAgICAgIERJUkVDVElPTkFMSVRZX09USEVSX05FVVRSQUxTLAotICAgICAgICAgICAgRElSRUNUSU9OQUxJVFlfTEVGVF9UT19SSUdIVF9FTUJFRERJTkcsCi0gICAgICAgICAgICBESVJFQ1RJT05BTElUWV9MRUZUX1RPX1JJR0hUX09WRVJSSURFLAotICAgICAgICAgICAgRElSRUNUSU9OQUxJVFlfUklHSFRfVE9fTEVGVF9FTUJFRERJTkcsCi0gICAgICAgICAgICBESVJFQ1RJT05BTElUWV9SSUdIVF9UT19MRUZUX09WRVJSSURFLAotICAgICAgICAgICAgRElSRUNUSU9OQUxJVFlfUE9QX0RJUkVDVElPTkFMX0ZPUk1BVAotICAgICAgICB9OwotCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDaGFyYWN0ZXIgdHlwZXMgYXMgc3BlY2lmaWVkIGluIHRoZSBVbmljb2RlIHN0YW5kYXJkLiBUaGVzZSBtYXAgZGlyZWN0bHkgdG8KLSAgICAgICAgICogamF2YS5sYW5nLkNoYXJhY3Rlci4KLSAgICAgICAgICovCi0gICAgICAgIGVudW0gQ2hhclR5cGUgewotICAgICAgICAgICAgQ0hBUlRZUEVfVU5BU1NJR05FRCA9IDAsCi0gICAgICAgICAgICBDSEFSVFlQRV9VUFBFUkNBU0VfTEVUVEVSLAotICAgICAgICAgICAgQ0hBUlRZUEVfTE9XRVJDQVNFX0xFVFRFUiwKLSAgICAgICAgICAgIENIQVJUWVBFX1RJVExFQ0FTRV9MRVRURVIsCi0gICAgICAgICAgICBDSEFSVFlQRV9NT0RJRklFUl9MRVRURVIsCi0gICAgICAgICAgICBDSEFSVFlQRV9PVEhFUl9MRVRURVIsCi0gICAgICAgICAgICBDSEFSVFlQRV9OT05fU1BBQ0lOR19NQVJLLAotICAgICAgICAgICAgQ0hBUlRZUEVfRU5DTE9TSU5HX01BUkssCi0gICAgICAgICAgICBDSEFSVFlQRV9DT01CSU5JTkdfU1BBQ0lOR19NQVJLLAotICAgICAgICAgICAgQ0hBUlRZUEVfREVDSU1BTF9ESUdJVF9OVU1CRVIsCi0gICAgICAgICAgICBDSEFSVFlQRV9MRVRURVJfTlVNQkVSLAotICAgICAgICAgICAgQ0hBUlRZUEVfT1RIRVJfTlVNQkVSLAotICAgICAgICAgICAgQ0hBUlRZUEVfU1BBQ0VfU0VQQVJBVE9SLAotICAgICAgICAgICAgQ0hBUlRZUEVfTElORV9TRVBBUkFUT1IsCi0gICAgICAgICAgICBDSEFSVFlQRV9QQVJBR1JBUEhfU0VQQVJBVE9SLAotICAgICAgICAgICAgQ0hBUlRZUEVfQ09OVFJPTCwKLSAgICAgICAgICAgIENIQVJUWVBFX0ZPUk1BVCwKLSAgICAgICAgICAgIENIQVJUWVBFX01JU1NJTkdfVkFMVUVfRk9SX0pBVkEsICAgIC8qIFRoaXMgaXMgdGhlIG15c3RlcmlvdXMgbWlzc2luZyAxNyB2YWx1ZSBmcm9tIHRoZSBqYXZhIGNvbnN0YW50cyAqLwotICAgICAgICAgICAgQ0hBUlRZUEVfUFJJVkFURV9VU0UsCi0gICAgICAgICAgICBDSEFSVFlQRV9TVVJST0dBVEUsCi0gICAgICAgICAgICBDSEFSVFlQRV9EQVNIX1BVTkNUVUFUSU9OLAotICAgICAgICAgICAgQ0hBUlRZUEVfU1RBUlRfUFVOQ1RVQVRJT04sCi0gICAgICAgICAgICBDSEFSVFlQRV9FTkRfUFVOQ1RVQVRJT04sCi0gICAgICAgICAgICBDSEFSVFlQRV9DT05ORUNUT1JfUFVOQ1RVQVRJT04sCi0gICAgICAgICAgICBDSEFSVFlQRV9PVEhFUl9QVU5DVFVBVElPTiwKLSAgICAgICAgICAgIENIQVJUWVBFX01BVEhfU1lNQk9MLAotICAgICAgICAgICAgQ0hBUlRZUEVfQ1VSUkVOQ1lfU1lNQk9MLAotICAgICAgICAgICAgQ0hBUlRZUEVfTU9ESUZJRVJfU1lNQk9MLAotICAgICAgICAgICAgQ0hBUlRZUEVfT1RIRVJfU1lNQk9MLAotICAgICAgICAgICAgQ0hBUlRZUEVfSU5JVElBTF9RVU9URV9QVU5DVFVBVElPTiwKLSAgICAgICAgICAgIENIQVJUWVBFX0ZJTkFMX1FVT1RFX1BVTkNUVUFUSU9OCi0gICAgICAgIH07Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIERlY29tcG9zaXRpb24gdHlwZXMgYXMgZGVzY3JpYmVkIGJ5IHRoZSB1bmljb2RlIHN0YW5kYXJkLiBUaGVzZSB2YWx1ZXMgbWFwIHRvCi0gICAgICAgICAqIHRoZSBzYW1lIHZhbHVlcyBpbiB1Y2hhci5oIGluIElDVS4KLSAgICAgICAgICovCi0gICAgICAgIGVudW0gRGVjb21wb3NpdGlvblR5cGUgewotICAgICAgICAgICAgREVDT01QT1NJVElPTl9OT05FID0gMCwKLSAgICAgICAgICAgIERFQ09NUE9TSVRJT05fQ0FOT05JQ0FMLAotICAgICAgICAgICAgREVDT01QT1NJVElPTl9DT01QQVQsCi0gICAgICAgICAgICBERUNPTVBPU0lUSU9OX0NJUkNMRSwKLSAgICAgICAgICAgIERFQ09NUE9TSVRJT05fRklOQUwsCi0gICAgICAgICAgICBERUNPTVBPU0lUSU9OX0ZPTlQsCi0gICAgICAgICAgICBERUNPTVBPU0lUSU9OX0ZSQUNUSU9OLAotICAgICAgICAgICAgREVDT01QT1NJVElPTl9JTklUSUFMLAotICAgICAgICAgICAgREVDT01QT1NJVElPTl9JU09MQVRFRCwKLSAgICAgICAgICAgIERFQ09NUE9TSVRJT05fTUVESUFMLAotICAgICAgICAgICAgREVDT01QT1NJVElPTl9OQVJST1csCi0gICAgICAgICAgICBERUNPTVBPU0lUSU9OX05PQlJFQUssCi0gICAgICAgICAgICBERUNPTVBPU0lUSU9OX1NNQUxMLAotICAgICAgICAgICAgREVDT01QT1NJVElPTl9TUVVBUkUsCi0gICAgICAgICAgICBERUNPTVBPU0lUSU9OX1NVQiwKLSAgICAgICAgICAgIERFQ09NUE9TSVRJT05fU1VQRVIsCi0gICAgICAgICAgICBERUNPTVBPU0lUSU9OX1ZFUlRJQ0FMLAotICAgICAgICAgICAgREVDT01QT1NJVElPTl9XSURFCi0gICAgICAgIH07Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFJldHVybnMgdGhlIHBhY2tlZCBkYXRhIGZvciBqYXZhIGNhbGxzCi0gICAgICAgICAqIEBwYXJhbSBjIFRoZSB1bmljb2RlIGNoYXJhY3Rlci4KLSAgICAgICAgICogQHJldHVybiBUaGUgcGFja2VkIGRhdGEgZm9yIHRoZSBjaGFyYWN0ZXIuCi0gICAgICAgICAqCi0gICAgICAgICAqIENvcGllZCBmcm9tIGphdmEubGFuZy5DaGFyYWN0ZXIgaW1wbGVtZW50YXRpb246Ci0gICAgICAgICAqIDEgMSAxIDEgMSAxIDEgMSAxIDEgMSAxIDEgMSAxIDEKLSAgICAgICAgICogRiBFIEQgQyBCIEEgOSA4IDcgNiA1IDQgMyAyIDEgMCBGIEUgRCBDIEIgQSA5IDggNyA2IDUgNCAzIDIgMSAwCi0gICAgICAgICAqIAotICAgICAgICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDMxIHR5cGVzICAgICAgICAgICAgICAgICAtLS0tLS0tLS0KLSAgICAgICAgICogICAgICAgICAgICAgICAgICAgMTggZGlyZWN0aW9uYWxpdGllcyAgICAgICAtLS0tLS0tLS0KLSAgICAgICAgICogICAgICAgICAgICAgICAgICAgMiBtaXJyb3JlZHMgICAgICAgICAgICAgLQotICAgICAgICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAtLS0tLS0tLS0tLSAgICAgIDU2ICB0b3VwcGVyIGRpZmZzCi0gICAgICAgICAqICAgICAgICAgICAgICAgICAgIC0tLS0tLS0tLS0tICAgICAgICAgICAgICAgICAgNDggIHRvbG93ZXIgZGlmZnMKLSAgICAgICAgICogICAgICAgICAgICAgICAtLS0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA0IHRvdGl0bGVjYXNlIGRpZmZzCi0gICAgICAgICAqIC0tLS0tLS0tLS0tLS0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA4NCBudW1lcmljIHZhbHVlcwotICAgICAgICAgKiAgICAgLS0tLS0tLS0tICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMjQgbWlycm9yIGNoYXIgZGlmZnMKLSAgICAgICAgICovCi0gICAgICAgIHN0YXRpYyB1aW50MzJfdCBnZXRQYWNrZWREYXRhKFVDaGFyMzIgYyk7Ci0gICAgICAgIAotICAgICAgICAvKioKLSAgICAgICAgICogR2V0IHRoZSBDaGFyYWN0ZXIgdHlwZS4KLSAgICAgICAgICogQHBhcmFtIGMgVGhlIHVuaWNvZGUgY2hhcmFjdGVyLgotICAgICAgICAgKiBAcmV0dXJuIFRoZSBjaGFyYWN0ZXIncyB0eXBlIG9yIENIQVJUWVBFX1VOQVNTSUdORUQgaWYgdGhlIGNoYXJhY3RlciBpcyBpbnZhbGlkCi0gICAgICAgICAqICAgICAgICAgb3IgaGFzIGFuIHVuYXNzaWduZWQgY2xhc3MuCi0gICAgICAgICAqLwotICAgICAgICBzdGF0aWMgQ2hhclR5cGUgZ2V0VHlwZShVQ2hhcjMyIGMpOyAgICAKLQotICAgICAgICAvKioKLSAgICAgICAgICogR2V0IHRoZSBDaGFyYWN0ZXIncyBkZWNvbXBvc2l0aW9uIHR5cGUuCi0gICAgICAgICAqIEBwYXJhbSBjIFRoZSB1bmljb2RlIGNoYXJhY3Rlci4KLSAgICAgICAgICogQHJldHVybiBUaGUgY2hhcmFjdGVyJ3MgZGVjb21wb3NpdGlvbiB0eXBlIG9yIERFQ09NUE9TSVRJT05fTk9ORSBpcyB0aGVyZSAKLSAgICAgICAgICogICAgICAgICBpcyBubyBkZWNvbXBvc2l0aW9uLgotICAgICAgICAgKi8KLSAgICAgICAgc3RhdGljIERlY29tcG9zaXRpb25UeXBlIGdldERlY29tcG9zaXRpb25UeXBlKFVDaGFyMzIgYyk7Ci0gICAgICAgIAotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJucyB0aGUgZGlnaXQgdmFsdWUgb2YgYSBjaGFyYWN0ZXIgb3IgLTEgaWYgdGhlIGNoYXJhY3RlcgotICAgICAgICAgKiBpcyBub3Qgd2l0aGluIHRoZSBzcGVjaWZpZWQgcmFkaXguCi0gICAgICAgICAqCi0gICAgICAgICAqIFRoZSBkaWdpdCB2YWx1ZSBpcyBjb21wdXRlZCBmb3IgaW50ZWdlciBjaGFyYWN0ZXJzIGFuZCBsZXR0ZXJzCi0gICAgICAgICAqIHdpdGhpbiB0aGUgZ2l2ZW4gcmFkaXguIFRoaXMgZnVuY3Rpb24gZG9lcyBub3QgaGFuZGxlIFJvbWFuIE51bWVyYWxzLAotICAgICAgICAgKiBmcmFjdGlvbnMsIG9yIGFueSBvdGhlciBjaGFyYWN0ZXJzIHRoYXQgbWF5IHJlcHJlc2VudCBudW1iZXJzLgotICAgICAgICAgKiAKLSAgICAgICAgICogQHBhcmFtIGMgVGhlIHVuaWNvZGUgY2hhcmFjdGVyCi0gICAgICAgICAqIEBwYXJhbSByYWRpeCBUaGUgaW50ZW5kZWQgcmFkaXguCi0gICAgICAgICAqIEByZXR1cm4gVGhlIGRpZ2l0IHZhbHVlIG9yIC0xIGlmIHRoZXJlIGlzIG5vIGRpZ2l0IHZhbHVlIG9yIGlmIHRoZSB2YWx1ZSBpcyBvdXRzaWRlIHRoZSByYWRpeC4KLSAgICAgICAgICovCi0gICAgICAgIHN0YXRpYyBpbnQgZ2V0RGlnaXRWYWx1ZShVQ2hhcjMyIGMsIGludCByYWRpeCA9IDEwKTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogUmV0dXJuIHRoZSBudW1lcmljIHZhbHVlIG9mIGEgY2hhcmFjdGVyCi0gICAgICAgICAqCi0gICAgICAgICAqIEBwYXJhbSBjIFRoZSB1bmljb2RlIGNoYXJhY3Rlci4KLSAgICAgICAgICogQHJldHVybiBUaGUgbnVtZXJpYyB2YWx1ZSBvZiB0aGUgY2hhcmFjdGVyLiAtMSBpZiB0aGUgY2hhcmFjdGVyIGhhcyBubyBudW1lcmljIHZhbHVlLCAKLSAgICAgICAgICogICAgICAgICAtMiBpZiB0aGUgY2hhcmFjdGVyIGhhcyBhIG51bWVyaWMgdmFsdWUgdGhhdCBpcyBub3QgcmVwcmVzZW50YWJsZSBieSBhbiBpbnRlZ2VyLgotICAgICAgICAgKi8KLSAgICAgICAgc3RhdGljIGludCBnZXROdW1lcmljVmFsdWUoVUNoYXIzMiBjKTsKLQotICAgICAgICAvKioKLSAgICAgICAgICogQ29udmVydCB0aGUgY2hhcmFjdGVyIHRvIGxvd2VyY2FzZQotICAgICAgICAgKiBAcGFyYW0gYyBUaGUgdW5pY29kZSBjaGFyYWN0ZXIuCi0gICAgICAgICAqIEByZXR1cm4gVGhlIGxvd2VyY2FzZSBjaGFyYWN0ZXIgZXF1aXZhbGVudCBvZiBjLiBJZiBjIGRvZXMgbm90IGhhdmUgYSBsb3dlcmNhc2UgZXF1aXZhbGVudCwKLSAgICAgICAgICogICAgICAgICB0aGUgb3JpZ2luYWwgY2hhcmFjdGVyIGlzIHJldHVybmVkLgotICAgICAgICAgKi8KLSAgICAgICAgc3RhdGljIFVDaGFyMzIgdG9Mb3dlcihVQ2hhcjMyIGMpOwotICAgICAgICAgICAgCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDb252ZXJ0IHRoZSBjaGFyYWN0ZXIgdG8gdXBwZXJjYXNlCi0gICAgICAgICAqIEBwYXJhbSBjIFRoZSB1bmljb2RlIGNoYXJhY3Rlci4KLSAgICAgICAgICogQHJldHVybiBUaGUgdXBwZXJjYXNlIGNoYXJhY3RlciBlcXVpdmFsZW50IG9mIGMuIElmIGMgZG9lcyBub3QgaGF2ZSBhbiB1cHBlcmNhc2UgZXF1aXZhbGVudCwKLSAgICAgICAgICogICAgICAgICB0aGUgb3JpZ2luYWwgY2hhcmFjdGVyIGlzIHJldHVybmVkLgotICAgICAgICAgKi8KLSAgICAgICAgc3RhdGljIFVDaGFyMzIgdG9VcHBlcihVQ2hhcjMyIGMpOwotICAgIAotICAgICAgICAvKioKLSAgICAgICAgICogR2V0IHRoZSBkaXJlY3Rpb25hbGl0eSBvZiB0aGUgY2hhcmFjdGVyLgotICAgICAgICAgKiBAcGFyYW0gYyBUaGUgdW5pY29kZSBjaGFyYWN0ZXIuCi0gICAgICAgICAqIEByZXR1cm4gVGhlIGRpcmVjdGlvbiBvZiB0aGUgY2hhcmFjdGVyIG9yIERJUkVDVElPTkFMSVRZX1VOREVGSU5FRC4KLSAgICAgICAgICovCi0gICAgICAgIHN0YXRpYyBEaXJlY3Rpb24gZ2V0RGlyZWN0aW9uYWxpdHkoVUNoYXIzMiBjKTsKLSAgICAgICAgICAgIAotICAgICAgICAvKioKLSAgICAgICAgICogQ2hlY2sgaWYgdGhlIGNoYXJhY3RlciBpcyBhIG1pcnJvcmVkIGNoYXJhY3Rlci4gVGhpcyBtZWFucyB0aGF0IHRoZSBjaGFyYWN0ZXIKLSAgICAgICAgICogaGFzIGFuIGVxdWl2YWxlbnQgY2hhcmFjdGVyIHRoYXQgaXMgdGhlIG1pcnJvciBpbWFnZSBvZiBpdHNlbGYuCi0gICAgICAgICAqIEBwYXJhbSBjIFRoZSB1bmljb2RlIGNoYXJhY3Rlci4KLSAgICAgICAgICogQHJldHVybiBUcnVlIGlmZiBjIGhhcyBhIG1pcnJvciBlcXVpdmFsZW50LgotICAgICAgICAgKi8KLSAgICAgICAgc3RhdGljIGJvb2wgaXNNaXJyb3JlZChVQ2hhcjMyIGMpOwotICAgICAgICAgCi0gICAgICAgIC8qKgotICAgICAgICAgKiBSZXR1cm4gdGhlIG1pcnJvciBvZiB0aGUgZ2l2ZW4gY2hhcmFjdGVyLgotICAgICAgICAgKiBAcGFyYW0gYyBUaGUgdW5pY29kZSBjaGFyYWN0ZXIuCi0gICAgICAgICAqIEByZXR1cm4gVGhlIG1pcnJvciBlcXVpdmFsZW50IG9mIGMuIElmIGMgZG9lcyBub3QgaGF2ZSBhIG1pcnJvciBlcXVpdmFsZW50LAotICAgICAgICAgKiAgICAgICAgIHRoZSBvcmlnaW5hbCBjaGFyYWN0ZXIgaXMgcmV0dXJuZWQuCi0gICAgICAgICAqIEBzZWUgaXNNaXJyb3JlZAotICAgICAgICAgKi8KLSAgICAgICAgc3RhdGljIFVDaGFyMzIgdG9NaXJyb3IoVUNoYXIzMiBjKTsKLSAgICAgICAgCi0gICAgICAgIC8qKgotICAgICAgICAgKiBDb252ZXJ0IHRoZSBjaGFyYWN0ZXIgdG8gdGl0bGUgY2FzZS4KLSAgICAgICAgICogQHBhcmFtIGMgVGhlIHVuaWNvZGUgY2hhcmFjdGVyLgotICAgICAgICAgKiBAcmV0dXJuIFRoZSB0aXRsZWNhc2UgZXF1aXZhbGVudCBvZiBjLiBJZiBjIGRvZXMgbm90IGhhdmUgYSB0aXRsZWNhc2UgZXF1aXZhbGVudCwKLSAgICAgICAgICogICAgICAgICB0aGUgb3JpZ2luYWwgY2hhcmFjdGVyIGlzIHJldHVybmVkLgotICAgICAgICAgKi8KLSAgICAgICAgc3RhdGljIFVDaGFyMzIgdG9UaXRsZShVQ2hhcjMyIGMpOwotCi0gICB9OwotCi19Ci0KLSNlbmRpZgpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9Bc3NldC5oIGIvaW5jbHVkZS91dGlscy9Bc3NldC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0NTNhMjA0Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvQXNzZXQuaAorKysgL2Rldi9udWxsCkBAIC0xLDMxNSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA2IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIENsYXNzIHByb3ZpZGluZyBhY2Nlc3MgdG8gYSByZWFkLW9ubHkgYXNzZXQuICBBc3NldCBvYmplY3RzIGFyZSBOT1QKLS8vIHRocmVhZC1zYWZlLCBhbmQgc2hvdWxkIG5vdCBiZSBzaGFyZWQgYWNyb3NzIHRocmVhZHMuCi0vLwotI2lmbmRlZiBfX0xJQlNfQVNTRVRfSAotI2RlZmluZSBfX0xJQlNfQVNTRVRfSAotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlICJGaWxlTWFwLmgiCi0jaW5jbHVkZSAiU3RyaW5nOC5oIgotI2luY2x1ZGUgIkVycm9ycy5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8qCi0gKiBJbnN0YW5jZXMgb2YgdGhpcyBjbGFzcyBwcm92aWRlIHJlYWQtb25seSBvcGVyYXRpb25zIG9uIGEgYnl0ZSBzdHJlYW0uCi0gKgotICogQWNjZXNzIG1heSBiZSBvcHRpbWl6ZWQgZm9yIHN0cmVhbWluZywgcmFuZG9tLCBvciB3aG9sZSBidWZmZXIgbW9kZXMuICBBbGwKLSAqIG9wZXJhdGlvbnMgYXJlIHN1cHBvcnRlZCByZWdhcmRsZXNzIG9mIGhvdyB0aGUgZmlsZSB3YXMgb3BlbmVkLCBidXQgc29tZQotICogdGhpbmdzIHdpbGwgYmUgbGVzcyBlZmZpY2llbnQuICBbcGFzcyB0aGF0IGluPz9dCi0gKgotICogIkFzc2V0IiBpcyB0aGUgYmFzZSBjbGFzcyBmb3IgYWxsIHR5cGVzIG9mIGFzc2V0cy4gIFRoZSBjbGFzc2VzIGJlbG93Ci0gKiBwcm92aWRlIG1vc3Qgb2YgdGhlIGltcGxlbWVudGF0aW9uLiAgVGhlIEFzc2V0TWFuYWdlciB1c2VzIG9uZSBvZiB0aGUKLSAqIHN0YXRpYyAiY3JlYXRlIiBmdW5jdGlvbnMgZGVmaW5lZCBoZXJlIHRvIGNyZWF0ZSBhIG5ldyBpbnN0YW5jZS4KLSAqLwotY2xhc3MgQXNzZXQgewotcHVibGljOgotICAgIHZpcnR1YWwgfkFzc2V0KHZvaWQpOwotCi0gICAgc3RhdGljIGludDMyX3QgZ2V0R2xvYmFsQ291bnQoKTsKLSAgICAKLSAgICAvKiB1c2VkIHdoZW4gb3BlbmluZyBhbiBhc3NldCAqLwotICAgIHR5cGVkZWYgZW51bSBBY2Nlc3NNb2RlIHsKLSAgICAgICAgQUNDRVNTX1VOS05PV04gPSAwLAotCi0gICAgICAgIC8qIHJlYWQgY2h1bmtzLCBhbmQgc2VlayBmb3J3YXJkIGFuZCBiYWNrd2FyZCAqLwotICAgICAgICBBQ0NFU1NfUkFORE9NLAotCi0gICAgICAgIC8qIHJlYWQgc2VxdWVudGlhbGx5LCB3aXRoIGFuIG9jY2FzaW9uYWwgZm9yd2FyZCBzZWVrICovCi0gICAgICAgIEFDQ0VTU19TVFJFQU1JTkcsCi0KLSAgICAgICAgLyogY2FsbGVyIHBsYW5zIHRvIGFzayBmb3IgYSByZWFkLW9ubHkgYnVmZmVyIHdpdGggYWxsIGRhdGEgKi8KLSAgICAgICAgQUNDRVNTX0JVRkZFUiwKLSAgICB9IEFjY2Vzc01vZGU7Ci0KLSAgICBlbnVtIHsKLSAgICAgICAgLyogZGF0YSBsYXJnZXIgdGhhbiB0aGlzIGRvZXMgbm90IGdldCB1bmNvbXByZXNzZWQgaW50byBhIGJ1ZmZlciAqLwotI2lmZGVmIEhBVkVfQU5EUk9JRF9PUwotICAgICAgICBVTkNPTVBSRVNTX0RBVEFfTUFYID0gMSAqIDEwMjQgKiAxMDI0Ci0jZWxzZQotICAgICAgICBVTkNPTVBSRVNTX0RBVEFfTUFYID0gMiAqIDEwMjQgKiAxMDI0Ci0jZW5kaWYKLSAgICB9OwotCi0gICAgLyoKLSAgICAgKiBSZWFkIGRhdGEgZnJvbSB0aGUgY3VycmVudCBvZmZzZXQuICBSZXR1cm5zIHRoZSBhY3R1YWwgbnVtYmVyIG9mCi0gICAgICogYnl0ZXMgcmVhZCwgMCBvbiBFT0YsIG9yIC0xIG9uIGVycm9yLgotICAgICAqLwotICAgIHZpcnR1YWwgc3NpemVfdCByZWFkKHZvaWQqIGJ1Ziwgc2l6ZV90IGNvdW50KSA9IDA7Ci0KLSAgICAvKgotICAgICAqIFNlZWsgdG8gdGhlIHNwZWNpZmllZCBvZmZzZXQuICAid2hlbmNlIiB1c2VzIHRoZSBzYW1lIHZhbHVlcyBhcwotICAgICAqIGxzZWVrL2ZzZWVrLiAgUmV0dXJucyB0aGUgbmV3IHBvc2l0aW9uIG9uIHN1Y2Nlc3MsIG9yIChvZmZfdCkgLTEKLSAgICAgKiBvbiBmYWlsdXJlLgotICAgICAqLwotICAgIHZpcnR1YWwgb2ZmX3Qgc2VlayhvZmZfdCBvZmZzZXQsIGludCB3aGVuY2UpID0gMDsKLQotICAgIC8qCi0gICAgICogQ2xvc2UgdGhlIGFzc2V0LCBmcmVlaW5nIGFsbCBhc3NvY2lhdGVkIHJlc291cmNlcy4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHZvaWQgY2xvc2Uodm9pZCkgPSAwOwotCi0gICAgLyoKLSAgICAgKiBHZXQgYSBwb2ludGVyIHRvIGEgYnVmZmVyIHdpdGggdGhlIGVudGlyZSBjb250ZW50cyBvZiB0aGUgZmlsZS4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIGNvbnN0IHZvaWQqIGdldEJ1ZmZlcihib29sIHdvcmRBbGlnbmVkKSA9IDA7Ci0KLSAgICAvKgotICAgICAqIEdldCB0aGUgdG90YWwgYW1vdW50IG9mIGRhdGEgdGhhdCBjYW4gYmUgcmVhZC4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIG9mZl90IGdldExlbmd0aCh2b2lkKSBjb25zdCA9IDA7Ci0KLSAgICAvKgotICAgICAqIEdldCB0aGUgdG90YWwgYW1vdW50IG9mIGRhdGEgdGhhdCBjYW4gYmUgcmVhZCBmcm9tIHRoZSBjdXJyZW50IHBvc2l0aW9uLgotICAgICAqLwotICAgIHZpcnR1YWwgb2ZmX3QgZ2V0UmVtYWluaW5nTGVuZ3RoKHZvaWQpIGNvbnN0ID0gMDsKLQotICAgIC8qCi0gICAgICogT3BlbiBhIG5ldyBmaWxlIGRlc2NyaXB0b3IgdGhhdCBjYW4gYmUgdXNlZCB0byByZWFkIHRoaXMgYXNzZXQuCi0gICAgICogUmV0dXJucyAtMSBpZiB5b3UgY2FuIG5vdCB1c2UgdGhlIGZpbGUgZGVzY3JpcHRvciAoZm9yIGV4YW1wbGUgaWYgdGhlCi0gICAgICogYXNzZXQgaXMgY29tcHJlc3NlZCkuCi0gICAgICovCi0gICAgdmlydHVhbCBpbnQgb3BlbkZpbGVEZXNjcmlwdG9yKG9mZl90KiBvdXRTdGFydCwgb2ZmX3QqIG91dExlbmd0aCkgY29uc3QgPSAwOwotICAgIAotICAgIC8qCi0gICAgICogR2V0IGEgc3RyaW5nIGlkZW50aWZ5aW5nIHRoZSBhc3NldCdzIHNvdXJjZS4gIFRoaXMgbWlnaHQgYmUgYSBmdWxsCi0gICAgICogcGF0aCwgaXQgbWlnaHQgYmUgYSBjb2xvbi1zZXBhcmF0ZWQgbGlzdCBvZiBpZGVudGlmaWVycy4KLSAgICAgKgotICAgICAqIFRoaXMgaXMgTk9UIGludGVuZGVkIHRvIGJlIHVzZWQgZm9yIGFueXRoaW5nIGV4Y2VwdCBkZWJ1ZyBvdXRwdXQuCi0gICAgICogRE8gTk9UIHRyeSB0byBwYXJzZSB0aGlzIG9yIHVzZSBpdCB0byBvcGVuIGEgZmlsZS4KLSAgICAgKi8KLSAgICBjb25zdCBjaGFyKiBnZXRBc3NldFNvdXJjZSh2b2lkKSBjb25zdCB7IHJldHVybiBtQXNzZXRTb3VyY2Uuc3RyaW5nKCk7IH0KLQotcHJvdGVjdGVkOgotICAgIEFzc2V0KHZvaWQpOyAgICAgICAgLy8gY29uc3RydWN0b3I7IG9ubHkgaW52b2tlZCBpbmRpcmVjdGx5Ci0KLSAgICAvKiBoYW5kbGUgY29tbW9uIHNlZWsoKSBob3VzZWtlZXBpbmcgKi8KLSAgICBvZmZfdCBoYW5kbGVTZWVrKG9mZl90IG9mZnNldCwgaW50IHdoZW5jZSwgb2ZmX3QgY3VyUG9zbiwgb2ZmX3QgbWF4UG9zbik7Ci0KLSAgICAvKiBzZXQgdGhlIGFzc2V0IHNvdXJjZSBzdHJpbmcgKi8KLSAgICB2b2lkIHNldEFzc2V0U291cmNlKGNvbnN0IFN0cmluZzgmIHBhdGgpIHsgbUFzc2V0U291cmNlID0gcGF0aDsgfQotCi0gICAgQWNjZXNzTW9kZSBnZXRBY2Nlc3NNb2RlKHZvaWQpIGNvbnN0IHsgcmV0dXJuIG1BY2Nlc3NNb2RlOyB9Ci0KLXByaXZhdGU6Ci0gICAgLyogdGhlc2Ugb3BlcmF0aW9ucyBhcmUgbm90IGltcGxlbWVudGVkICovCi0gICAgQXNzZXQoY29uc3QgQXNzZXQmIHNyYyk7Ci0gICAgQXNzZXQmIG9wZXJhdG9yPShjb25zdCBBc3NldCYgc3JjKTsKLQotICAgIC8qIEFzc2V0TWFuYWdlciBuZWVkcyBhY2Nlc3MgdG8gb3VyICJjcmVhdGUiIGZ1bmN0aW9ucyAqLwotICAgIGZyaWVuZCBjbGFzcyBBc3NldE1hbmFnZXI7Ci0KLSAgICAvKgotICAgICAqIENyZWF0ZSB0aGUgYXNzZXQgZnJvbSBhIG5hbWVkIGZpbGUgb24gZGlzay4KLSAgICAgKi8KLSAgICBzdGF0aWMgQXNzZXQqIGNyZWF0ZUZyb21GaWxlKGNvbnN0IGNoYXIqIGZpbGVOYW1lLCBBY2Nlc3NNb2RlIG1vZGUpOwotCi0gICAgLyoKLSAgICAgKiBDcmVhdGUgdGhlIGFzc2V0IGZyb20gYSBuYW1lZCwgY29tcHJlc3NlZCBmaWxlIG9uIGRpc2sgKGUuZy4gIi5neiIpLgotICAgICAqLwotICAgIHN0YXRpYyBBc3NldCogY3JlYXRlRnJvbUNvbXByZXNzZWRGaWxlKGNvbnN0IGNoYXIqIGZpbGVOYW1lLAotICAgICAgICBBY2Nlc3NNb2RlIG1vZGUpOwotCi0jaWYgMAotICAgIC8qCi0gICAgICogQ3JlYXRlIHRoZSBhc3NldCBmcm9tIGEgc2VnbWVudCBvZiBhbiBvcGVuIGZpbGUuICBUaGlzIHdpbGwgZmFpbAotICAgICAqIGlmICJvZmZzZXQiIGFuZCAibGVuZ3RoIiBkb24ndCBmaXQgd2l0aGluIHRoZSBib3VuZHMgb2YgdGhlIGZpbGUuCi0gICAgICoKLSAgICAgKiBUaGUgYXNzZXQgdGFrZXMgb3duZXJzaGlwIG9mIHRoZSBmaWxlIGRlc2NyaXB0b3IuCi0gICAgICovCi0gICAgc3RhdGljIEFzc2V0KiBjcmVhdGVGcm9tRmlsZVNlZ21lbnQoaW50IGZkLCBvZmZfdCBvZmZzZXQsIHNpemVfdCBsZW5ndGgsCi0gICAgICAgIEFjY2Vzc01vZGUgbW9kZSk7Ci0KLSAgICAvKgotICAgICAqIENyZWF0ZSBmcm9tIGNvbXByZXNzZWQgZGF0YS4gICJmZCIgc2hvdWxkIGJlIHNlZWtlZCB0byB0aGUgc3RhcnQgb2YKLSAgICAgKiB0aGUgY29tcHJlc3NlZCBkYXRhLiAgVGhpcyBjb3VsZCBiZSBpbnNpZGUgYSBnemlwIGZpbGUgb3IgcGFydCBvZiBhCi0gICAgICogWmlwIGFyY2hpdmUuCi0gICAgICoKLSAgICAgKiBUaGUgYXNzZXQgdGFrZXMgb3duZXJzaGlwIG9mIHRoZSBmaWxlIGRlc2NyaXB0b3IuCi0gICAgICoKLSAgICAgKiBUaGlzIG1heSBub3QgdmVyaWZ5IHRoZSB2YWxpZGl0eSBvZiB0aGUgY29tcHJlc3NlZCBkYXRhIHVudGlsIGZpcnN0Ci0gICAgICogdXNlLgotICAgICAqLwotICAgIHN0YXRpYyBBc3NldCogY3JlYXRlRnJvbUNvbXByZXNzZWREYXRhKGludCBmZCwgb2ZmX3Qgb2Zmc2V0LAotICAgICAgICBpbnQgY29tcHJlc3Npb25NZXRob2QsIHNpemVfdCBjb21wcmVzc2VkTGVuZ3RoLAotICAgICAgICBzaXplX3QgdW5jb21wcmVzc2VkTGVuZ3RoLCBBY2Nlc3NNb2RlIG1vZGUpOwotI2VuZGlmCi0KLSAgICAvKgotICAgICAqIENyZWF0ZSB0aGUgYXNzZXQgZnJvbSBhIG1lbW9yeS1tYXBwZWQgZmlsZSBzZWdtZW50LgotICAgICAqCi0gICAgICogVGhlIGFzc2V0IHRha2VzIG93bmVyc2hpcCBvZiB0aGUgRmlsZU1hcC4KLSAgICAgKi8KLSAgICBzdGF0aWMgQXNzZXQqIGNyZWF0ZUZyb21VbmNvbXByZXNzZWRNYXAoRmlsZU1hcCogZGF0YU1hcCwgQWNjZXNzTW9kZSBtb2RlKTsKLQotICAgIC8qCi0gICAgICogQ3JlYXRlIHRoZSBhc3NldCBmcm9tIGEgbWVtb3J5LW1hcHBlZCBmaWxlIHNlZ21lbnQgd2l0aCBjb21wcmVzc2VkCi0gICAgICogZGF0YS4gICJtZXRob2QiIGlzIGEgWmlwIGFyY2hpdmUgY29tcHJlc3Npb24gbWV0aG9kIGNvbnN0YW50LgotICAgICAqCi0gICAgICogVGhlIGFzc2V0IHRha2VzIG93bmVyc2hpcCBvZiB0aGUgRmlsZU1hcC4KLSAgICAgKi8KLSAgICBzdGF0aWMgQXNzZXQqIGNyZWF0ZUZyb21Db21wcmVzc2VkTWFwKEZpbGVNYXAqIGRhdGFNYXAsIGludCBtZXRob2QsCi0gICAgICAgIHNpemVfdCB1bmNvbXByZXNzZWRMZW4sIEFjY2Vzc01vZGUgbW9kZSk7Ci0KLQotICAgIC8qCi0gICAgICogQ3JlYXRlIGZyb20gYSByZWZlcmVuY2UtY291bnRlZCBjaHVuayBvZiBzaGFyZWQgbWVtb3J5LgotICAgICAqLwotICAgIC8vIFRPRE8KLQotICAgIEFjY2Vzc01vZGUgIG1BY2Nlc3NNb2RlOyAgICAgICAgLy8gaG93IHRoZSBhc3NldCB3YXMgb3BlbmVkCi0gICAgU3RyaW5nOCAgICBtQXNzZXRTb3VyY2U7ICAgICAgIC8vIGRlYnVnIHN0cmluZwotfTsKLQotCi0vKgotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKgotICogSW5uYXJkcyBmb2xsb3cuICBEbyBub3QgdXNlIHRoZXNlIGNsYXNzZXMgZGlyZWN0bHkuCi0gKi8KLQotLyoKLSAqIEFuIGFzc2V0IGJhc2VkIG9uIGFuIHVuY29tcHJlc3NlZCBmaWxlIG9uIGRpc2suICBJdCBtYXkgZW5jb21wYXNzIHRoZQotICogZW50aXJlIGZpbGUgb3IganVzdCBhIHBpZWNlIG9mIGl0LiAgQWNjZXNzIGlzIHRocm91Z2ggZnJlYWQvZnNlZWsuCi0gKi8KLWNsYXNzIF9GaWxlQXNzZXQgOiBwdWJsaWMgQXNzZXQgewotcHVibGljOgotICAgIF9GaWxlQXNzZXQodm9pZCk7Ci0gICAgdmlydHVhbCB+X0ZpbGVBc3NldCh2b2lkKTsKLQotICAgIC8qCi0gICAgICogVXNlIGEgcGllY2Ugb2YgYW4gYWxyZWFkeS1vcGVuIGZpbGUuCi0gICAgICoKLSAgICAgKiBPbiBzdWNjZXNzLCB0aGUgb2JqZWN0IHRha2VzIG93bmVyc2hpcCBvZiAiZmQiLgotICAgICAqLwotICAgIHN0YXR1c190IG9wZW5DaHVuayhjb25zdCBjaGFyKiBmaWxlTmFtZSwgaW50IGZkLCBvZmZfdCBvZmZzZXQsIHNpemVfdCBsZW5ndGgpOwotCi0gICAgLyoKLSAgICAgKiBVc2UgYSBtZW1vcnktbWFwcGVkIHJlZ2lvbi4KLSAgICAgKgotICAgICAqIE9uIHN1Y2Nlc3MsIHRoZSBvYmplY3QgdGFrZXMgb3duZXJzaGlwIG9mICJkYXRhTWFwIi4KLSAgICAgKi8KLSAgICBzdGF0dXNfdCBvcGVuQ2h1bmsoRmlsZU1hcCogZGF0YU1hcCk7Ci0KLSAgICAvKgotICAgICAqIFN0YW5kYXJkIEFzc2V0IGludGVyZmFjZXMuCi0gICAgICovCi0gICAgdmlydHVhbCBzc2l6ZV90IHJlYWQodm9pZCogYnVmLCBzaXplX3QgY291bnQpOwotICAgIHZpcnR1YWwgb2ZmX3Qgc2VlayhvZmZfdCBvZmZzZXQsIGludCB3aGVuY2UpOwotICAgIHZpcnR1YWwgdm9pZCBjbG9zZSh2b2lkKTsKLSAgICB2aXJ0dWFsIGNvbnN0IHZvaWQqIGdldEJ1ZmZlcihib29sIHdvcmRBbGlnbmVkKTsKLSAgICB2aXJ0dWFsIG9mZl90IGdldExlbmd0aCh2b2lkKSBjb25zdCB7IHJldHVybiBtTGVuZ3RoOyB9Ci0gICAgdmlydHVhbCBvZmZfdCBnZXRSZW1haW5pbmdMZW5ndGgodm9pZCkgY29uc3QgeyByZXR1cm4gbUxlbmd0aC1tT2Zmc2V0OyB9Ci0gICAgdmlydHVhbCBpbnQgb3BlbkZpbGVEZXNjcmlwdG9yKG9mZl90KiBvdXRTdGFydCwgb2ZmX3QqIG91dExlbmd0aCkgY29uc3Q7Ci0KLXByaXZhdGU6Ci0gICAgb2ZmX3QgICAgICAgbVN0YXJ0OyAgICAgICAgIC8vIGFic29sdXRlIGZpbGUgb2Zmc2V0IG9mIHN0YXJ0IG9mIGNodW5rCi0gICAgb2ZmX3QgICAgICAgbUxlbmd0aDsgICAgICAgIC8vIGxlbmd0aCBvZiB0aGUgY2h1bmsKLSAgICBvZmZfdCAgICAgICBtT2Zmc2V0OyAgICAgICAgLy8gY3VycmVudCBsb2NhbCBvZmZzZXQsIDAgPT0gbVN0YXJ0Ci0gICAgRklMRSogICAgICAgbUZwOyAgICAgICAgICAgIC8vIGZvciByZWFkL3NlZWsKLSAgICBjaGFyKiAgICAgICBtRmlsZU5hbWU7ICAgICAgLy8gZm9yIG9wZW5pbmcKLQotICAgIC8qCi0gICAgICogVG8gc3VwcG9ydCBnZXRCdWZmZXIoKSB3ZSBlaXRoZXIgbmVlZCB0byByZWFkIHRoZSBlbnRpcmUgdGhpbmcgaW50bwotICAgICAqIGEgYnVmZmVyIG9yIG1lbW9yeS1tYXAgaXQuICBGb3Igc21hbGwgZmlsZXMgaXQncyBwcm9iYWJseSBiZXN0IHRvCi0gICAgICoganVzdCByZWFkIHRoZW0gaW4uCi0gICAgICovCi0gICAgZW51bSB7IGtSZWFkVnNNYXBUaHJlc2hvbGQgPSA0MDk2IH07Ci0KLSAgICBGaWxlTWFwKiAgICBtTWFwOyAgICAgICAgICAgLy8gZm9yIG1lbW9yeSBtYXAKLSAgICB1bnNpZ25lZCBjaGFyKiBtQnVmOyAgICAgICAgLy8gZm9yIHJlYWQKLSAgICAKLSAgICBjb25zdCB2b2lkKiBlbnN1cmVBbGlnbm1lbnQoRmlsZU1hcCogbWFwKTsKLX07Ci0KLQotLyoKLSAqIEFuIGFzc2V0IGJhc2VkIG9uIGNvbXByZXNzZWQgZGF0YSBpbiBhIGZpbGUuCi0gKi8KLWNsYXNzIF9Db21wcmVzc2VkQXNzZXQgOiBwdWJsaWMgQXNzZXQgewotcHVibGljOgotICAgIF9Db21wcmVzc2VkQXNzZXQodm9pZCk7Ci0gICAgdmlydHVhbCB+X0NvbXByZXNzZWRBc3NldCh2b2lkKTsKLQotICAgIC8qCi0gICAgICogVXNlIGEgcGllY2Ugb2YgYW4gYWxyZWFkeS1vcGVuIGZpbGUuCi0gICAgICoKLSAgICAgKiBPbiBzdWNjZXNzLCB0aGUgb2JqZWN0IHRha2VzIG93bmVyc2hpcCBvZiAiZmQiLgotICAgICAqLwotICAgIHN0YXR1c190IG9wZW5DaHVuayhpbnQgZmQsIG9mZl90IG9mZnNldCwgaW50IGNvbXByZXNzaW9uTWV0aG9kLAotICAgICAgICBzaXplX3QgdW5jb21wcmVzc2VkTGVuLCBzaXplX3QgY29tcHJlc3NlZExlbik7Ci0KLSAgICAvKgotICAgICAqIFVzZSBhIG1lbW9yeS1tYXBwZWQgcmVnaW9uLgotICAgICAqCi0gICAgICogT24gc3VjY2VzcywgdGhlIG9iamVjdCB0YWtlcyBvd25lcnNoaXAgb2YgImZkIi4KLSAgICAgKi8KLSAgICBzdGF0dXNfdCBvcGVuQ2h1bmsoRmlsZU1hcCogZGF0YU1hcCwgaW50IGNvbXByZXNzaW9uTWV0aG9kLAotICAgICAgICBzaXplX3QgdW5jb21wcmVzc2VkTGVuKTsKLQotICAgIC8qCi0gICAgICogU3RhbmRhcmQgQXNzZXQgaW50ZXJmYWNlcy4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHNzaXplX3QgcmVhZCh2b2lkKiBidWYsIHNpemVfdCBjb3VudCk7Ci0gICAgdmlydHVhbCBvZmZfdCBzZWVrKG9mZl90IG9mZnNldCwgaW50IHdoZW5jZSk7Ci0gICAgdmlydHVhbCB2b2lkIGNsb3NlKHZvaWQpOwotICAgIHZpcnR1YWwgY29uc3Qgdm9pZCogZ2V0QnVmZmVyKGJvb2wgd29yZEFsaWduZWQpOwotICAgIHZpcnR1YWwgb2ZmX3QgZ2V0TGVuZ3RoKHZvaWQpIGNvbnN0IHsgcmV0dXJuIG1VbmNvbXByZXNzZWRMZW47IH0KLSAgICB2aXJ0dWFsIG9mZl90IGdldFJlbWFpbmluZ0xlbmd0aCh2b2lkKSBjb25zdCB7IHJldHVybiBtVW5jb21wcmVzc2VkTGVuLW1PZmZzZXQ7IH0KLSAgICB2aXJ0dWFsIGludCBvcGVuRmlsZURlc2NyaXB0b3Iob2ZmX3QqIG91dFN0YXJ0LCBvZmZfdCogb3V0TGVuZ3RoKSBjb25zdCB7IHJldHVybiAtMTsgfQotCi1wcml2YXRlOgotICAgIG9mZl90ICAgICAgIG1TdGFydDsgICAgICAgICAvLyBvZmZzZXQgdG8gc3RhcnQgb2YgY29tcHJlc3NlZCBkYXRhCi0gICAgb2ZmX3QgICAgICAgbUNvbXByZXNzZWRMZW47IC8vIGxlbmd0aCBvZiB0aGUgY29tcHJlc3NlZCBkYXRhCi0gICAgb2ZmX3QgICAgICAgbVVuY29tcHJlc3NlZExlbjsgLy8gbGVuZ3RoIG9mIHRoZSB1bmNvbXByZXNzZWQgZGF0YQotICAgIG9mZl90ICAgICAgIG1PZmZzZXQ7ICAgICAgICAvLyBjdXJyZW50IG9mZnNldCwgMCA9PSBzdGFydCBvZiB1bmNvbXAgZGF0YQotCi0gICAgRmlsZU1hcCogICAgbU1hcDsgICAgICAgICAgIC8vIGZvciBtZW1vcnktbWFwcGVkIGlucHV0Ci0gICAgaW50ICAgICAgICAgbUZkOyAgICAgICAgICAgIC8vIGZvciBmaWxlIGlucHV0Ci0KLSAgICB1bnNpZ25lZCBjaGFyKiAgbUJ1ZjsgICAgICAgLy8gZm9yIGdldEJ1ZmZlcigpCi19OwotCi0vLyBuZWVkOiBzaGFyZWQgbW1hcCB2ZXJzaW9uPwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gX19MSUJTX0FTU0VUX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvQXNzZXREaXIuaCBiL2luY2x1ZGUvdXRpbHMvQXNzZXREaXIuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYWJmOGEzNS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0Fzc2V0RGlyLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxNDUgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBBY2Nlc3MgYSBjaHVuayBvZiB0aGUgYXNzZXQgaGllcmFyY2h5IGFzIGlmIGl0IHdlcmUgYSBzaW5nbGUgZGlyZWN0b3J5LgotLy8KLSNpZm5kZWYgX19MSUJTX0FTU0VURElSX0gKLSNkZWZpbmUgX19MSUJTX0FTU0VURElSX0gKLQotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLSNpbmNsdWRlIDx1dGlscy9WZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9Tb3J0ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9taXNjLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyoKLSAqIFRoaXMgcHJvdmlkZXMgdmVjdG9yLXN0eWxlIGFjY2VzcyB0byBhIGRpcmVjdG9yeS4gIFdlIGRvIHRoaXMgcmF0aGVyCi0gKiB0aGFuIG1vZGVsaW5nIG9wZW5kaXIvcmVhZGRpciBhY2Nlc3MgYmVjYXVzZSBpdCdzIHNpbXBsZXIgYW5kIHRoZQotICogbmF0dXJlIG9mIHRoZSBvcGVyYXRpb24gcmVxdWlyZXMgdXMgdG8gaGF2ZSBhbGwgZGF0YSBvbiBoYW5kIGFueXdheS4KLSAqCi0gKiBUaGUgbGlzdCBvZiBmaWxlcyB3aWxsIGJlIHNvcnRlZCBpbiBhc2NlbmRpbmcgb3JkZXIgYnkgQVNDSUkgdmFsdWUuCi0gKgotICogVGhlIGNvbnRlbnRzIGFyZSBwb3B1bGF0ZWQgYnkgb3VyIGZyaWVuZCwgdGhlIEFzc2V0TWFuYWdlci4KLSAqLwotY2xhc3MgQXNzZXREaXIgewotcHVibGljOgotICAgIEFzc2V0RGlyKHZvaWQpCi0gICAgICAgIDogbUZpbGVJbmZvKE5VTEwpCi0gICAgICAgIHt9Ci0gICAgdmlydHVhbCB+QXNzZXREaXIodm9pZCkgewotICAgICAgICBkZWxldGUgbUZpbGVJbmZvOwotICAgIH0KLQotICAgIC8qCi0gICAgICogVmVjdG9yLXN0eWxlIGFjY2Vzcy4KLSAgICAgKi8KLSAgICBzaXplX3QgZ2V0RmlsZUNvdW50KHZvaWQpIHsgcmV0dXJuIG1GaWxlSW5mby0+c2l6ZSgpOyB9Ci0gICAgY29uc3QgU3RyaW5nOCYgZ2V0RmlsZU5hbWUoaW50IGlkeCkgewotICAgICAgICByZXR1cm4gbUZpbGVJbmZvLT5pdGVtQXQoaWR4KS5nZXRGaWxlTmFtZSgpOwotICAgIH0KLSAgICBjb25zdCBTdHJpbmc4JiBnZXRTb3VyY2VOYW1lKGludCBpZHgpIHsKLSAgICAgICAgcmV0dXJuIG1GaWxlSW5mby0+aXRlbUF0KGlkeCkuZ2V0U291cmNlTmFtZSgpOwotICAgIH0KLQotICAgIC8qCi0gICAgICogR2V0IHRoZSB0eXBlIG9mIGEgZmlsZSAodXN1YWxseSByZWd1bGFyIG9yIGRpcmVjdG9yeSkuCi0gICAgICovCi0gICAgRmlsZVR5cGUgZ2V0RmlsZVR5cGUoaW50IGlkeCkgewotICAgICAgICByZXR1cm4gbUZpbGVJbmZvLT5pdGVtQXQoaWR4KS5nZXRGaWxlVHlwZSgpOwotICAgIH0KLQotcHJpdmF0ZToKLSAgICAvKiB0aGVzZSBvcGVyYXRpb25zIGFyZSBub3QgaW1wbGVtZW50ZWQgKi8KLSAgICBBc3NldERpcihjb25zdCBBc3NldERpciYgc3JjKTsKLSAgICBjb25zdCBBc3NldERpciYgb3BlcmF0b3I9KGNvbnN0IEFzc2V0RGlyJiBzcmMpOwotCi0gICAgZnJpZW5kIGNsYXNzIEFzc2V0TWFuYWdlcjsKLQotICAgIC8qCi0gICAgICogVGhpcyBob2xkcyBpbmZvcm1hdGlvbiBhYm91dCBmaWxlcyBpbiB0aGUgYXNzZXQgaGllcmFyY2h5LgotICAgICAqLwotICAgIGNsYXNzIEZpbGVJbmZvIHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIEZpbGVJbmZvKHZvaWQpIHt9Ci0gICAgICAgIEZpbGVJbmZvKGNvbnN0IFN0cmluZzgmIHBhdGgpICAgICAgLy8gdXNlZnVsIGZvciBlLmcuIHN2ZWN0LmluZGV4T2YKLSAgICAgICAgICAgIDogbUZpbGVOYW1lKHBhdGgpLCBtRmlsZVR5cGUoa0ZpbGVUeXBlVW5rbm93bikKLSAgICAgICAgICAgIHt9Ci0gICAgICAgIH5GaWxlSW5mbyh2b2lkKSB7fQotICAgICAgICBGaWxlSW5mbyhjb25zdCBGaWxlSW5mbyYgc3JjKSB7Ci0gICAgICAgICAgICBjb3B5TWVtYmVycyhzcmMpOwotICAgICAgICB9Ci0gICAgICAgIGNvbnN0IEZpbGVJbmZvJiBvcGVyYXRvcj0gKGNvbnN0IEZpbGVJbmZvJiBzcmMpIHsKLSAgICAgICAgICAgIGlmICh0aGlzICE9ICZzcmMpCi0gICAgICAgICAgICAgICAgY29weU1lbWJlcnMoc3JjKTsKLSAgICAgICAgICAgIHJldHVybiAqdGhpczsKLSAgICAgICAgfQotCi0gICAgICAgIHZvaWQgY29weU1lbWJlcnMoY29uc3QgRmlsZUluZm8mIHNyYykgewotICAgICAgICAgICAgbUZpbGVOYW1lID0gc3JjLm1GaWxlTmFtZTsKLSAgICAgICAgICAgIG1GaWxlVHlwZSA9IHNyYy5tRmlsZVR5cGU7Ci0gICAgICAgICAgICBtU291cmNlTmFtZSA9IHNyYy5tU291cmNlTmFtZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qIG5lZWQgdGhpcyBmb3IgU29ydGVkVmVjdG9yOyBtdXN0IGNvbXBhcmUgb25seSBvbiBmaWxlIG5hbWUgKi8KLSAgICAgICAgYm9vbCBvcGVyYXRvcjwgKGNvbnN0IEZpbGVJbmZvJiByaHMpIGNvbnN0IHsKLSAgICAgICAgICAgIHJldHVybiBtRmlsZU5hbWUgPCByaHMubUZpbGVOYW1lOwotICAgICAgICB9Ci0KLSAgICAgICAgLyogdXNlZCBieSBBc3NldE1hbmFnZXIgKi8KLSAgICAgICAgYm9vbCBvcGVyYXRvcj09IChjb25zdCBGaWxlSW5mbyYgcmhzKSBjb25zdCB7Ci0gICAgICAgICAgICByZXR1cm4gbUZpbGVOYW1lID09IHJocy5tRmlsZU5hbWU7Ci0gICAgICAgIH0KLQotICAgICAgICB2b2lkIHNldChjb25zdCBTdHJpbmc4JiBwYXRoLCBGaWxlVHlwZSB0eXBlKSB7Ci0gICAgICAgICAgICBtRmlsZU5hbWUgPSBwYXRoOwotICAgICAgICAgICAgbUZpbGVUeXBlID0gdHlwZTsKLSAgICAgICAgfQotCi0gICAgICAgIGNvbnN0IFN0cmluZzgmIGdldEZpbGVOYW1lKHZvaWQpIGNvbnN0IHsgcmV0dXJuIG1GaWxlTmFtZTsgfQotICAgICAgICB2b2lkIHNldEZpbGVOYW1lKGNvbnN0IFN0cmluZzgmIHBhdGgpIHsgbUZpbGVOYW1lID0gcGF0aDsgfQotCi0gICAgICAgIEZpbGVUeXBlIGdldEZpbGVUeXBlKHZvaWQpIGNvbnN0IHsgcmV0dXJuIG1GaWxlVHlwZTsgfQotICAgICAgICB2b2lkIHNldEZpbGVUeXBlKEZpbGVUeXBlIHR5cGUpIHsgbUZpbGVUeXBlID0gdHlwZTsgfQotCi0gICAgICAgIGNvbnN0IFN0cmluZzgmIGdldFNvdXJjZU5hbWUodm9pZCkgY29uc3QgeyByZXR1cm4gbVNvdXJjZU5hbWU7IH0KLSAgICAgICAgdm9pZCBzZXRTb3VyY2VOYW1lKGNvbnN0IFN0cmluZzgmIHBhdGgpIHsgbVNvdXJjZU5hbWUgPSBwYXRoOyB9Ci0KLSAgICAgICAgLyoKLSAgICAgICAgICogSGFuZHkgdXRpbGl0eSBmb3IgZmluZGluZyBhbiBlbnRyeSBpbiBhIHNvcnRlZCB2ZWN0b3Igb2YgRmlsZUluZm8uCi0gICAgICAgICAqIFJldHVybnMgdGhlIGluZGV4IG9mIHRoZSBtYXRjaGluZyBlbnRyeSwgb3IgLTEgaWYgbm9uZSBmb3VuZC4KLSAgICAgICAgICovCi0gICAgICAgIHN0YXRpYyBpbnQgZmluZEVudHJ5KGNvbnN0IFNvcnRlZFZlY3RvcjxGaWxlSW5mbz4qIHBWZWN0b3IsCi0gICAgICAgICAgICBjb25zdCBTdHJpbmc4JiBmaWxlTmFtZSk7Ci0KLSAgICBwcml2YXRlOgotICAgICAgICBTdHJpbmc4ICAgIG1GaWxlTmFtZTsgICAgICAvLyBmaWxlbmFtZSBvbmx5Ci0gICAgICAgIEZpbGVUeXBlICAgIG1GaWxlVHlwZTsgICAgICAvLyByZWd1bGFyLCBkaXJlY3RvcnksIGV0YwotCi0gICAgICAgIFN0cmluZzggICAgbVNvdXJjZU5hbWU7ICAgIC8vIGN1cnJlbnRseSBkZWJ1Zy1vbmx5Ci0gICAgfTsKLQotICAgIC8qIEFzc2V0TWFuYWdlciB1c2VzIHRoaXMgdG8gaW5pdGlhbGl6ZSB1cyAqLwotICAgIHZvaWQgc2V0RmlsZUxpc3QoU29ydGVkVmVjdG9yPEZpbGVJbmZvPiogbGlzdCkgeyBtRmlsZUluZm8gPSBsaXN0OyB9Ci0KLSAgICBTb3J0ZWRWZWN0b3I8RmlsZUluZm8+KiBtRmlsZUluZm87Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gX19MSUJTX0FTU0VURElSX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvQXNzZXRNYW5hZ2VyLmggYi9pbmNsdWRlL3V0aWxzL0Fzc2V0TWFuYWdlci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlOTRjMGU4Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvQXNzZXRNYW5hZ2VyLmgKKysrIC9kZXYvbnVsbApAQCAtMSwzMjMgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBBc3NldCBtYW5hZ2VtZW50IGNsYXNzLiAgQXNzZXRNYW5hZ2VyIG9iamVjdHMgYXJlIHRocmVhZC1zYWZlLgotLy8KLSNpZm5kZWYgX19MSUJTX0FTU0VUTUFOQUdFUl9ICi0jZGVmaW5lIF9fTElCU19BU1NFVE1BTkFHRVJfSAotCi0jaW5jbHVkZSA8dXRpbHMvQXNzZXQuaD4KLSNpbmNsdWRlIDx1dGlscy9Bc3NldERpci5oPgotI2luY2x1ZGUgPHV0aWxzL0tleWVkVmVjdG9yLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RyaW5nOC5oPgotI2luY2x1ZGUgPHV0aWxzL1ZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzE2Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvWmlwRmlsZVJPLmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIEFzc2V0OyAgICAgICAgLy8gZndkIGRlY2wgZm9yIHRoaW5ncyB0aGF0IGluY2x1ZGUgQXNzZXQuaCBmaXJzdAotY2xhc3MgUmVzVGFibGU7Ci1zdHJ1Y3QgUmVzVGFibGVfY29uZmlnOwotCi0vKgotICogRXZlcnkgYXBwbGljYXRpb24gdGhhdCB1c2VzIGFzc2V0cyBuZWVkcyBvbmUgaW5zdGFuY2Ugb2YgdGhpcy4gIEEKLSAqIHNpbmdsZSBpbnN0YW5jZSBtYXkgYmUgc2hhcmVkIGFjcm9zcyBtdWx0aXBsZSB0aHJlYWRzLCBhbmQgYSBzaW5nbGUKLSAqIHRocmVhZCBtYXkgaGF2ZSBtb3JlIHRoYW4gb25lIGluc3RhbmNlICh0aGUgbGF0dGVyIGlzIGRpc2NvdXJhZ2VkKS4KLSAqCi0gKiBUaGUgcHVycG9zZSBvZiB0aGUgQXNzZXRNYW5hZ2VyIGlzIHRvIGNyZWF0ZSBBc3NldCBvYmplY3RzLiAgVG8gZG8KLSAqIHRoaXMgZWZmaWNpZW50bHkgaXQgbWF5IGNhY2hlIGluZm9ybWF0aW9uIGFib3V0IHRoZSBsb2NhdGlvbnMgb2YKLSAqIGZpbGVzIGl0IGhhcyBzZWVuLiAgVGhpcyBjYW4gYmUgY29udHJvbGxlZCB3aXRoIHRoZSAiY2FjaGVNb2RlIgotICogYXJndW1lbnQuCi0gKgotICogVGhlIGFzc2V0IGhpZXJhcmNoeSBtYXkgYmUgZXhhbWluZWQgbGlrZSBhIGZpbGVzeXN0ZW0sIHVzaW5nCi0gKiBBc3NldERpciBvYmplY3RzIHRvIHBlcnVzZSBhIHNpbmdsZSBkaXJlY3RvcnkuCi0gKi8KLWNsYXNzIEFzc2V0TWFuYWdlciB7Ci1wdWJsaWM6Ci0gICAgdHlwZWRlZiBlbnVtIENhY2hlTW9kZSB7Ci0gICAgICAgIENBQ0hFX1VOS05PV04gPSAwLAotICAgICAgICBDQUNIRV9PRkYsICAgICAgICAgIC8vIGRvbid0IHRyeSB0byBjYWNoZSBmaWxlIGxvY2F0aW9ucwotICAgICAgICBDQUNIRV9ERUZFUiwgICAgICAgIC8vIGNvbnN0cnVjdCBjYWNoZSBhcyBwaWVjZXMgYXJlIG5lZWRlZAotICAgICAgICAvL0NBQ0hFX1NDQU4sICAgICAgICAgLy8gc2NhbiBmdWxsKCEpIGFzc2V0IGhpZXJhcmNoeSBhdCBpbml0KCkgdGltZQotICAgIH0gQ2FjaGVNb2RlOwotCi0gICAgQXNzZXRNYW5hZ2VyKENhY2hlTW9kZSBjYWNoZU1vZGUgPSBDQUNIRV9PRkYpOwotICAgIHZpcnR1YWwgfkFzc2V0TWFuYWdlcih2b2lkKTsKLQotICAgIHN0YXRpYyBpbnQzMl90IGdldEdsb2JhbENvdW50KCk7Ci0gICAgCi0gICAgLyogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgICAqIEFkZCBhIG5ldyBzb3VyY2UgZm9yIGFzc2V0cy4gIFRoaXMgY2FuIGJlIGNhbGxlZCBtdWx0aXBsZSB0aW1lcyB0bwotICAgICAqIGxvb2sgaW4gbXVsdGlwbGUgcGxhY2VzIGZvciBhc3NldHMuICBJdCBjYW4gYmUgZWl0aGVyIGEgZGlyZWN0b3J5IChmb3IKLSAgICAgKiBmaW5kaW5nIGFzc2V0cyBhcyByYXcgZmlsZXMgb24gdGhlIGRpc2spIG9yIGEgWklQIGZpbGUuICBUaGlzIG5ld2x5Ci0gICAgICogYWRkZWQgYXNzZXQgcGF0aCB3aWxsIGJlIGV4YW1pbmVkIGZpcnN0IHdoZW4gc2VhcmNoaW5nIGZvciBhc3NldHMsCi0gICAgICogYmVmb3JlIGFueSB0aGF0IHdlcmUgcHJldmlvdXNseSBhZGRlZC4KLSAgICAgKgotICAgICAqIFJldHVybnMgInRydWUiIG9uIHN1Y2Nlc3MsICJmYWxzZSIgb24gZmFpbHVyZS4gIElmICdjb29raWUnIGlzIG5vbi1OVUxMLAotICAgICAqIHRoZW4gb24gc3VjY2VzcywgKmNvb2tpZSBpcyBzZXQgdG8gdGhlIHZhbHVlIGNvcnJlc3BvbmRpbmcgdG8gdGhlCi0gICAgICogbmV3bHktYWRkZWQgYXNzZXQgc291cmNlLgotICAgICAqLwotICAgIGJvb2wgYWRkQXNzZXRQYXRoKGNvbnN0IFN0cmluZzgmIHBhdGgsIHZvaWQqKiBjb29raWUpOwotCi0gICAgLyogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgICAqIENvbnZlbmllbmNlIGZvciBhZGRpbmcgdGhlIHN0YW5kYXJkIHN5c3RlbSBhc3NldHMuICBVc2VzIHRoZQotICAgICAqIEFORFJPSURfUk9PVCBlbnZpcm9ubWVudCB2YXJpYWJsZSB0byBmaW5kIHRoZW0uCi0gICAgICovCi0gICAgYm9vbCBhZGREZWZhdWx0QXNzZXRzKCk7Ci0KLSAgICAvKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgICogSXRlcmF0ZSBvdmVyIHRoZSBhc3NldCBwYXRocyBpbiB0aGlzIG1hbmFnZXIuICAoUHJldmlvdXNseQotICAgICAqIGFkZGVkIHZpYSBhZGRBc3NldFBhdGgoKSBhbmQgYWRkRGVmYXVsdEFzc2V0cygpLikgIE9uIGZpcnN0IGNhbGwsCi0gICAgICogJ2Nvb2tpZScgbXVzdCBiZSBOVUxMLCByZXN1bHRpbmcgaW4gdGhlIGZpcnN0IGNvb2tpZSBiZWluZyByZXR1cm5lZC4KLSAgICAgKiBFYWNoIG5leHQgY29va2llIHdpbGwgYmUgcmV0dXJuZWQgdGhlcmUtYWZ0ZXIsIHVudGlsIE5VTEwgaW5kaWNhdGluZwotICAgICAqIHRoZSBlbmQgaGFzIGJlZW4gcmVhY2hlZC4KLSAgICAgKi8KLSAgICB2b2lkKiBuZXh0QXNzZXRQYXRoKHZvaWQqIGNvb2tpZSkgY29uc3Q7Ci0KLSAgICAvKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgICogUmV0dXJuIGFuIGFzc2V0IHBhdGggaW4gdGhlIG1hbmFnZXIuICAnd2hpY2gnIG11c3QgYmUgYmV0d2VlbiAwIGFuZAotICAgICAqIGNvdW50QXNzZXRQYXRocygpLgotICAgICAqLwotICAgIFN0cmluZzggZ2V0QXNzZXRQYXRoKHZvaWQqIGNvb2tpZSkgY29uc3Q7Ci0KLSAgICAvKgotICAgICAqIFNldCB0aGUgY3VycmVudCBsb2NhbGUgYW5kIHZlbmRvci4gIFRoZSBsb2NhbGUgY2FuIGNoYW5nZSBkdXJpbmcKLSAgICAgKiB0aGUgbGlmZXRpbWUgb2YgYW4gQXNzZXRNYW5hZ2VyIGlmIHRoZSB1c2VyIHVwZGF0ZXMgdGhlIGRldmljZSdzCi0gICAgICogbGFuZ3VhZ2Ugc2V0dGluZy4gIFRoZSB2ZW5kb3IgaXMgbGVzcyBsaWtlbHkgdG8gY2hhbmdlLgotICAgICAqCi0gICAgICogUGFzcyBpbiBOVUxMIHRvIGluZGljYXRlIG5vIHByZWZlcmVuY2UuCi0gICAgICovCi0gICAgdm9pZCBzZXRMb2NhbGUoY29uc3QgY2hhciogbG9jYWxlKTsKLSAgICB2b2lkIHNldFZlbmRvcihjb25zdCBjaGFyKiB2ZW5kb3IpOwotCi0gICAgLyoKLSAgICAgKiBDaG9vc2Ugc2NyZWVuIG9yaWVudGF0aW9uIGZvciByZXNvdXJjZXMgdmFsdWVzIHJldHVybmVkLgotICAgICAqLwotICAgIHZvaWQgc2V0Q29uZmlndXJhdGlvbihjb25zdCBSZXNUYWJsZV9jb25maWcmIGNvbmZpZywgY29uc3QgY2hhciogbG9jYWxlID0gTlVMTCk7Ci0KLSAgICB0eXBlZGVmIEFzc2V0OjpBY2Nlc3NNb2RlIEFjY2Vzc01vZGU7ICAgICAgIC8vIHR5cGluZyBzaG9ydGN1dAotCi0gICAgLyoKLSAgICAgKiBPcGVuIGFuIGFzc2V0LgotICAgICAqCi0gICAgICogVGhpcyB3aWxsIHNlYXJjaCB0aHJvdWdoIGxvY2FsZS1zcGVjaWZpYyBhbmQgdmVuZG9yLXNwZWNpZmljCi0gICAgICogZGlyZWN0b3JpZXMgYW5kIHBhY2thZ2VzIHRvIGZpbmQgdGhlIGZpbGUuCi0gICAgICoKLSAgICAgKiBUaGUgb2JqZWN0IHJldHVybmVkIGRvZXMgbm90IGRlcGVuZCBvbiB0aGUgQXNzZXRNYW5hZ2VyLiAgSXQgc2hvdWxkCi0gICAgICogYmUgZnJlZWQgYnkgY2FsbGluZyBBc3NldDo6Y2xvc2UoKS4KLSAgICAgKi8KLSAgICBBc3NldCogb3Blbihjb25zdCBjaGFyKiBmaWxlTmFtZSwgQWNjZXNzTW9kZSBtb2RlKTsKLQotICAgIC8qCi0gICAgICogT3BlbiBhIG5vbi1hc3NldCBmaWxlIGFzIGFuIGFzc2V0LgotICAgICAqCi0gICAgICogVGhpcyBpcyBmb3Igb3BlbmluZyBmaWxlcyB0aGF0IGFyZSBpbmNsdWRlZCBpbiBhbiBhc3NldCBwYWNrYWdlCi0gICAgICogYnV0IGFyZW4ndCBhc3NldHMuICBUaGVzZSBzaXQgb3V0c2lkZSB0aGUgdXN1YWwgImxvY2FsZS92ZW5kb3IiCi0gICAgICogcGF0aCBoaWVyYXJjaHksIGFuZCB3aWxsIG5vdCBiZSBzZWVuIGJ5ICJBc3NldERpciIgb3IgaW5jbHVkZWQKLSAgICAgKiBpbiBvdXIgZmlsZW5hbWUgY2FjaGUuCi0gICAgICovCi0gICAgQXNzZXQqIG9wZW5Ob25Bc3NldChjb25zdCBjaGFyKiBmaWxlTmFtZSwgQWNjZXNzTW9kZSBtb2RlKTsKLQotICAgIC8qCi0gICAgICogRXhwbGljaXQgbm9uLWFzc2V0IGZpbGUuICBUaGUgZmlsZSBleHBsaWNpdGx5IG5hbWVkIGJ5IHRoZSBjb29raWUgKHRoZQotICAgICAqIHJlc291cmNlIHNldCB0byBsb29rIGluKSBhbmQgZmlsZU5hbWUgd2lsbCBiZSBvcGVuZWQgYW5kIHJldHVybmVkLgotICAgICAqLwotICAgIEFzc2V0KiBvcGVuTm9uQXNzZXQodm9pZCogY29va2llLCBjb25zdCBjaGFyKiBmaWxlTmFtZSwgQWNjZXNzTW9kZSBtb2RlKTsKLQotICAgIC8qCi0gICAgICogT3BlbiBhIGRpcmVjdG9yeSB3aXRoaW4gdGhlIGFzc2V0IGhpZXJhcmNoeS4KLSAgICAgKgotICAgICAqIFRoZSBjb250ZW50cyBvZiB0aGUgZGlyZWN0b3J5IGFyZSBhbiBhbWFsZ2FtIG9mIHZlbmRvci1zcGVjaWZpYywKLSAgICAgKiBsb2NhbGUtc3BlY2lmaWMsIGFuZCBnZW5lcmljIGFzc2V0cyBzdG9yZWQgbG9vc2VseSBvciBpbiBhc3NldAotICAgICAqIHBhY2thZ2VzLiAgRGVwZW5kaW5nIG9uIHRoZSBjYWNoZSBzZXR0aW5nIGFuZCBwcmV2aW91cyBhY2Nlc3NlcywKLSAgICAgKiB0aGlzIGNhbGwgbWF5IGluY3VyIHNpZ25pZmljYW50IGRpc2sgb3ZlcmhlYWQuCi0gICAgICoKLSAgICAgKiBUbyBvcGVuIHRoZSB0b3AtbGV2ZWwgZGlyZWN0b3J5LCBwYXNzIGluICIiLgotICAgICAqLwotICAgIEFzc2V0RGlyKiBvcGVuRGlyKGNvbnN0IGNoYXIqIGRpck5hbWUpOwotCi0gICAgLyoKLSAgICAgKiBHZXQgdGhlIHR5cGUgb2YgYSBmaWxlIGluIHRoZSBhc3NldCBoaWVyYXJjaHkuICBUaGV5IHdpbGwgZWl0aGVyCi0gICAgICogYmUgInJlZ3VsYXIiIG9yICJkaXJlY3RvcnkiLiAgW0N1cnJlbnRseSBvbmx5IHdvcmtzIGZvciAicmVndWxhciIuXQotICAgICAqCi0gICAgICogQ2FuIGFsc28gYmUgdXNlZCBhcyBhIHF1aWNrIHRlc3QgZm9yIGV4aXN0ZW5jZSBvZiBhIGZpbGUuCi0gICAgICovCi0gICAgRmlsZVR5cGUgZ2V0RmlsZVR5cGUoY29uc3QgY2hhciogZmlsZU5hbWUpOwotCi0gICAgLyogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgICAqIFJldHVybiB0aGUgY29tcGxldGUgcmVzb3VyY2UgdGFibGUgdG8gZmluZCB0aGluZ3MgaW4gdGhlIHBhY2thZ2UuCi0gICAgICovCi0gICAgY29uc3QgUmVzVGFibGUmIGdldFJlc291cmNlcyhib29sIHJlcXVpcmVkID0gdHJ1ZSkgY29uc3Q7Ci0KLSAgICAvKgotICAgICAqIERpc2NhcmQgY2FjaGVkIGZpbGVuYW1lIGluZm9ybWF0aW9uLiAgVGhpcyBvbmx5IG5lZWRzIHRvIGJlIGNhbGxlZAotICAgICAqIGlmIHNvbWVib2R5IGhhcyB1cGRhdGVkIHRoZSBzZXQgb2YgImxvb3NlIiBmaWxlcywgYW5kIHdlIHdhbnQgdG8KLSAgICAgKiBkaXNjYXJkIG91ciBjYWNoZWQgbm90aW9uIG9mIHdoYXQncyB3aGVyZS4KLSAgICAgKi8KLSAgICB2b2lkIHB1cmdlKHZvaWQpIHsgcHVyZ2VGaWxlTmFtZUNhY2hlTG9ja2VkKCk7IH0KLQotICAgIC8qCi0gICAgICogUmV0dXJuIHRydWUgaWYgdGhlIGZpbGVzIHRoaXMgQXNzZXRNYW5hZ2VyIHJlZmVyZW5jZXMgYXJlIGFsbAotICAgICAqIHVwLXRvLWRhdGUgKGhhdmUgbm90IGJlZW4gY2hhbmdlZCBzaW5jZSBpdCB3YXMgY3JlYXRlZCkuICBJZiBmYWxzZQotICAgICAqIGlzIHJldHVybmVkLCB5b3Ugd2lsbCBuZWVkIHRvIGNyZWF0ZSBhIG5ldyBBc3NldE1hbmFnZXIgdG8gZ2V0Ci0gICAgICogdGhlIGN1cnJlbnQgZGF0YS4KLSAgICAgKi8KLSAgICBib29sIGlzVXBUb0RhdGUoKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBHZXQgdGhlIGtub3duIGxvY2FsZXMgZm9yIHRoaXMgYXNzZXQgbWFuYWdlciBvYmplY3QuCi0gICAgICovCi0gICAgdm9pZCBnZXRMb2NhbGVzKFZlY3RvcjxTdHJpbmc4PiogbG9jYWxlcykgY29uc3Q7Ci0KLXByaXZhdGU6Ci0gICAgc3RydWN0IGFzc2V0X3BhdGgKLSAgICB7Ci0gICAgICAgIFN0cmluZzggcGF0aDsKLSAgICAgICAgRmlsZVR5cGUgdHlwZTsKLSAgICB9OwotCi0gICAgQXNzZXQqIG9wZW5JblBhdGhMb2NrZWQoY29uc3QgY2hhciogZmlsZU5hbWUsIEFjY2Vzc01vZGUgbW9kZSwKLSAgICAgICAgY29uc3QgYXNzZXRfcGF0aCYgcGF0aCk7Ci0gICAgQXNzZXQqIG9wZW5Ob25Bc3NldEluUGF0aExvY2tlZChjb25zdCBjaGFyKiBmaWxlTmFtZSwgQWNjZXNzTW9kZSBtb2RlLAotICAgICAgICBjb25zdCBhc3NldF9wYXRoJiBwYXRoKTsKLSAgICBBc3NldCogb3BlbkluTG9jYWxlVmVuZG9yTG9ja2VkKGNvbnN0IGNoYXIqIGZpbGVOYW1lLCBBY2Nlc3NNb2RlIG1vZGUsCi0gICAgICAgIGNvbnN0IGFzc2V0X3BhdGgmIHBhdGgsIGNvbnN0IGNoYXIqIGxvY2FsZSwgY29uc3QgY2hhciogdmVuZG9yKTsKLSAgICBTdHJpbmc4IGNyZWF0ZVBhdGhOYW1lTG9ja2VkKGNvbnN0IGFzc2V0X3BhdGgmIHBhdGgsIGNvbnN0IGNoYXIqIGxvY2FsZSwKLSAgICAgICAgY29uc3QgY2hhciogdmVuZG9yKTsKLSAgICBTdHJpbmc4IGNyZWF0ZVBhdGhOYW1lTG9ja2VkKGNvbnN0IGFzc2V0X3BhdGgmIHBhdGgsIGNvbnN0IGNoYXIqIHJvb3REaXIpOwotICAgIFN0cmluZzggY3JlYXRlWmlwU291cmNlTmFtZUxvY2tlZChjb25zdCBTdHJpbmc4JiB6aXBGaWxlTmFtZSwKLSAgICAgICAgY29uc3QgU3RyaW5nOCYgZGlyTmFtZSwgY29uc3QgU3RyaW5nOCYgZmlsZU5hbWUpOwotCi0gICAgWmlwRmlsZVJPKiBnZXRaaXBGaWxlTG9ja2VkKGNvbnN0IGFzc2V0X3BhdGgmIHBhdGgpOwotICAgIEFzc2V0KiBvcGVuQXNzZXRGcm9tRmlsZUxvY2tlZChjb25zdCBTdHJpbmc4JiBmaWxlTmFtZSwgQWNjZXNzTW9kZSBtb2RlKTsKLSAgICBBc3NldCogb3BlbkFzc2V0RnJvbVppcExvY2tlZChjb25zdCBaaXBGaWxlUk8qIHBaaXBGaWxlLAotICAgICAgICBjb25zdCBaaXBFbnRyeVJPIGVudHJ5LCBBY2Nlc3NNb2RlIG1vZGUsIGNvbnN0IFN0cmluZzgmIGVudHJ5TmFtZSk7Ci0KLSAgICBib29sIHNjYW5BbmRNZXJnZURpckxvY2tlZChTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPiogcE1lcmdlZEluZm8sCi0gICAgICAgIGNvbnN0IGFzc2V0X3BhdGgmIHBhdGgsIGNvbnN0IGNoYXIqIHJvb3REaXIsIGNvbnN0IGNoYXIqIGRpck5hbWUpOwotICAgIFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+KiBzY2FuRGlyTG9ja2VkKGNvbnN0IFN0cmluZzgmIHBhdGgpOwotICAgIGJvb2wgc2NhbkFuZE1lcmdlWmlwTG9ja2VkKFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+KiBwTWVyZ2VkSW5mbywKLSAgICAgICAgY29uc3QgYXNzZXRfcGF0aCYgcGF0aCwgY29uc3QgY2hhciogcm9vdERpciwgY29uc3QgY2hhciogZGlyTmFtZSk7Ci0gICAgdm9pZCBtZXJnZUluZm9Mb2NrZWQoU29ydGVkVmVjdG9yPEFzc2V0RGlyOjpGaWxlSW5mbz4qIHBNZXJnZWRJbmZvLAotICAgICAgICBjb25zdCBTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPiogcENvbnRlbnRzKTsKLQotICAgIHZvaWQgbG9hZEZpbGVOYW1lQ2FjaGVMb2NrZWQodm9pZCk7Ci0gICAgdm9pZCBmbmNTY2FuTG9ja2VkKFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+KiBwTWVyZ2VkSW5mbywKLSAgICAgICAgY29uc3QgY2hhciogZGlyTmFtZSk7Ci0gICAgYm9vbCBmbmNTY2FuQW5kTWVyZ2VEaXJMb2NrZWQoCi0gICAgICAgIFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+KiBwTWVyZ2VkSW5mbywKLSAgICAgICAgY29uc3QgYXNzZXRfcGF0aCYgcGF0aCwgY29uc3QgY2hhciogbG9jYWxlLCBjb25zdCBjaGFyKiB2ZW5kb3IsCi0gICAgICAgIGNvbnN0IGNoYXIqIGRpck5hbWUpOwotICAgIHZvaWQgcHVyZ2VGaWxlTmFtZUNhY2hlTG9ja2VkKHZvaWQpOwotCi0gICAgY29uc3QgUmVzVGFibGUqIGdldFJlc1RhYmxlKGJvb2wgcmVxdWlyZWQgPSB0cnVlKSBjb25zdDsKLSAgICB2b2lkIHNldExvY2FsZUxvY2tlZChjb25zdCBjaGFyKiBsb2NhbGUpOwotICAgIHZvaWQgdXBkYXRlUmVzb3VyY2VQYXJhbXNMb2NrZWQoKSBjb25zdDsKLQotICAgIGNsYXNzIFNoYXJlZFppcCA6IHB1YmxpYyBSZWZCYXNlIHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIHN0YXRpYyBzcDxTaGFyZWRaaXA+IGdldChjb25zdCBTdHJpbmc4JiBwYXRoKTsKLQotICAgICAgICBaaXBGaWxlUk8qIGdldFppcCgpOwotCi0gICAgICAgIEFzc2V0KiBnZXRSZXNvdXJjZVRhYmxlQXNzZXQoKTsKLSAgICAgICAgQXNzZXQqIHNldFJlc291cmNlVGFibGVBc3NldChBc3NldCogYXNzZXQpOwotCi0gICAgICAgIGJvb2wgaXNVcFRvRGF0ZSgpOwotICAgICAgICAKLSAgICBwcm90ZWN0ZWQ6Ci0gICAgICAgIH5TaGFyZWRaaXAoKTsKLQotICAgIHByaXZhdGU6Ci0gICAgICAgIFNoYXJlZFppcChjb25zdCBTdHJpbmc4JiBwYXRoLCB0aW1lX3QgbW9kV2hlbik7Ci0gICAgICAgIFNoYXJlZFppcCgpOyAvLyA8LS0gbm90IGltcGxlbWVudGVkCi0KLSAgICAgICAgU3RyaW5nOCBtUGF0aDsKLSAgICAgICAgWmlwRmlsZVJPKiBtWmlwRmlsZTsKLSAgICAgICAgdGltZV90IG1Nb2RXaGVuOwotCi0gICAgICAgIEFzc2V0KiBtUmVzb3VyY2VUYWJsZUFzc2V0OwotCi0gICAgICAgIHN0YXRpYyBNdXRleCBnTG9jazsKLSAgICAgICAgc3RhdGljIERlZmF1bHRLZXllZFZlY3RvcjxTdHJpbmc4LCB3cDxTaGFyZWRaaXA+ID4gZ09wZW47Ci0gICAgfTsKLQotICAgIC8qCi0gICAgICogTWFuYWdlIGEgc2V0IG9mIFppcCBmaWxlcy4gIEZvciBlYWNoIGZpbGUgd2UgbmVlZCBhIHBvaW50ZXIgdG8gdGhlCi0gICAgICogWmlwRmlsZSBhbmQgYSB0aW1lX3Qgd2l0aCB0aGUgZmlsZSdzIG1vZGlmaWNhdGlvbiBkYXRlLgotICAgICAqCi0gICAgICogV2UgY3VycmVudGx5IG9ubHkgaGF2ZSB0d28gemlwIGZpbGVzIChjdXJyZW50IGFwcCwgImNvbW1vbiIgYXBwKS4KLSAgICAgKiAoVGhpcyB3YXMgb3JpZ2luYWxseSB3cml0dGVuIGZvciA4LCBiYXNlZCBvbiBhcHAvbG9jYWxlL3ZlbmRvci4pCi0gICAgICovCi0gICAgY2xhc3MgWmlwU2V0IHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIFppcFNldCh2b2lkKTsKLSAgICAgICAgflppcFNldCh2b2lkKTsKLQotICAgICAgICAvKgotICAgICAgICAgKiBSZXR1cm4gYSBaaXBGaWxlUk8gc3RydWN0dXJlIGZvciBhIFppcEZpbGVSTyB3aXRoIHRoZSBzcGVjaWZpZWQKLSAgICAgICAgICogcGFyYW1ldGVycy4KLSAgICAgICAgICovCi0gICAgICAgIFppcEZpbGVSTyogZ2V0WmlwKGNvbnN0IFN0cmluZzgmIHBhdGgpOwotCi0gICAgICAgIEFzc2V0KiBnZXRaaXBSZXNvdXJjZVRhYmxlKGNvbnN0IFN0cmluZzgmIHBhdGgpOwotICAgICAgICBBc3NldCogc2V0WmlwUmVzb3VyY2VUYWJsZShjb25zdCBTdHJpbmc4JiBwYXRoLCBBc3NldCogYXNzZXQpOwotCi0gICAgICAgIC8vIGdlbmVyYXRlIHBhdGgsIGUuZy4gImNvbW1vbi9lbi1VUy1ub29nbGUuemlwIgotICAgICAgICBzdGF0aWMgU3RyaW5nOCBnZXRQYXRoTmFtZShjb25zdCBjaGFyKiBwYXRoKTsKLQotICAgICAgICBib29sIGlzVXBUb0RhdGUoKTsKLSAgICAgICAgCi0gICAgcHJpdmF0ZToKLSAgICAgICAgdm9pZCBjbG9zZVppcChpbnQgaWR4KTsKLQotICAgICAgICBpbnQgZ2V0SW5kZXgoY29uc3QgU3RyaW5nOCYgemlwKSBjb25zdDsKLSAgICAgICAgbXV0YWJsZSBWZWN0b3I8U3RyaW5nOD4gbVppcFBhdGg7Ci0gICAgICAgIG11dGFibGUgVmVjdG9yPHNwPFNoYXJlZFppcD4gPiBtWmlwRmlsZTsKLSAgICB9OwotCi0gICAgLy8gUHJvdGVjdCBhbGwgaW50ZXJuYWwgc3RhdGUuCi0gICAgbXV0YWJsZSBNdXRleCAgIG1Mb2NrOwotCi0gICAgWmlwU2V0ICAgICAgICAgIG1aaXBTZXQ7Ci0KLSAgICBWZWN0b3I8YXNzZXRfcGF0aD4gbUFzc2V0UGF0aHM7Ci0gICAgY2hhciogICAgICAgICAgIG1Mb2NhbGU7Ci0gICAgY2hhciogICAgICAgICAgIG1WZW5kb3I7Ci0KLSAgICBtdXRhYmxlIFJlc1RhYmxlKiBtUmVzb3VyY2VzOwotICAgIFJlc1RhYmxlX2NvbmZpZyogbUNvbmZpZzsKLQotICAgIC8qCi0gICAgICogQ2FjaGVkIGRhdGEgZm9yICJsb29zZSIgZmlsZXMuICBUaGlzIGxldHMgdXMgYXZvaWQgcG9raW5nIGF0IHRoZQotICAgICAqIGZpbGVzeXN0ZW0gd2hlbiBzZWFyY2hpbmcgZm9yIGxvb3NlIGFzc2V0cy4gIEVhY2ggZW50cnkgaXMgdGhlCi0gICAgICogImV4dGVuZGVkIHBhcnRpYWwiIHBhdGgsIGUuZy4gImRlZmF1bHQvZGVmYXVsdC9mb28vYmFyLnR4dCIuICBUaGUKLSAgICAgKiBmdWxsIHNldCBvZiBmaWxlcyBpcyBwcmVzZW50LCBpbmNsdWRpbmcgIi5FWENMVURFIiBlbnRyaWVzLgotICAgICAqCi0gICAgICogV2UgZG8gbm90IGNhY2hlIGRpcmVjdG9yeSBuYW1lcy4gIFdlIGRvbid0IHJldGFpbiB0aGUgIi5neiIsCi0gICAgICogYmVjYXVzZSB0byBvdXIgY2xpZW50cyAiZm9vIiBhbmQgImZvby5neiIgYm90aCBsb29rIGxpa2UgImZvbyIuCi0gICAgICovCi0gICAgQ2FjaGVNb2RlICAgICAgIG1DYWNoZU1vZGU7ICAgICAgICAgLy8gaXMgdGhlIGNhY2hlIGVuYWJsZWQ/Ci0gICAgYm9vbCAgICAgICAgICAgIG1DYWNoZVZhbGlkOyAgICAgICAgLy8gY2xlYXIgd2hlbiBsb2NhbGUgb3IgdmVuZG9yIGNoYW5nZXMKLSAgICBTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPiBtQ2FjaGU7Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gX19MSUJTX0FTU0VUTUFOQUdFUl9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL0F0b21pYy5oIGIvaW5jbHVkZS91dGlscy9BdG9taWMuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggN2ViNDc2Yy4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0F0b21pYy5oCisrKyAvZGV2L251bGwKQEAgLTEsMjIgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9VVElMU19BVE9NSUNfSAotI2RlZmluZSBBTkRST0lEX1VUSUxTX0FUT01JQ19ICi0KLSNpbmNsdWRlIDxjdXRpbHMvYXRvbWljLmg+Ci0KLSNlbmRpZiAvLyBBTkRST0lEX1VUSUxTX0FUT01JQ19ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL0JpbmRlci5oIGIvaW5jbHVkZS91dGlscy9CaW5kZXIuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYjViOGQ5OC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0JpbmRlci5oCisrKyAvZGV2L251bGwKQEAgLTEsMTAzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfQklOREVSX0gKLSNkZWZpbmUgQU5EUk9JRF9CSU5ERVJfSAotCi0jaW5jbHVkZSA8dXRpbHMvSUJpbmRlci5oPgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgQkJpbmRlciA6IHB1YmxpYyBJQmluZGVyCi17Ci1wdWJsaWM6Ci0gICAgICAgICAgICAgICAgICAgICAgICBCQmluZGVyKCk7Ci0KLSAgICB2aXJ0dWFsIFN0cmluZzE2ICAgIGdldEludGVyZmFjZURlc2NyaXB0b3IoKSBjb25zdDsKLSAgICB2aXJ0dWFsIGJvb2wgICAgICAgIGlzQmluZGVyQWxpdmUoKSBjb25zdDsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHBpbmdCaW5kZXIoKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIGR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKTsKLQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgdHJhbnNhY3QoICAgdWludDMyX3QgY29kZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFBhcmNlbCYgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBhcmNlbCogcmVwbHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDApOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBsaW5rVG9EZWF0aChjb25zdCBzcDxEZWF0aFJlY2lwaWVudD4mIHJlY2lwaWVudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQqIGNvb2tpZSA9IE5VTEwsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDApOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICB1bmxpbmtUb0RlYXRoKCAgY29uc3Qgd3A8RGVhdGhSZWNpcGllbnQ+JiByZWNpcGllbnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCogY29va2llID0gTlVMTCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgd3A8RGVhdGhSZWNpcGllbnQ+KiBvdXRSZWNpcGllbnQgPSBOVUxMKTsKLQotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgYXR0YWNoT2JqZWN0KCAgIGNvbnN0IHZvaWQqIG9iamVjdElELAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQqIG9iamVjdCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiBjbGVhbnVwQ29va2llLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9iamVjdF9jbGVhbnVwX2Z1bmMgZnVuYyk7Ci0gICAgdmlydHVhbCB2b2lkKiAgICAgICBmaW5kT2JqZWN0KGNvbnN0IHZvaWQqIG9iamVjdElEKSBjb25zdDsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIGRldGFjaE9iamVjdChjb25zdCB2b2lkKiBvYmplY3RJRCk7Ci0KLSAgICB2aXJ0dWFsIEJCaW5kZXIqICAgIGxvY2FsQmluZGVyKCk7Ci0KLXByb3RlY3RlZDoKLSAgICB2aXJ0dWFsICAgICAgICAgICAgIH5CQmluZGVyKCk7Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIG9uVHJhbnNhY3QoIHVpbnQzMl90IGNvZGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBQYXJjZWwmIGRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJjZWwqIHJlcGx5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MgPSAwKTsKLQotcHJpdmF0ZToKLSAgICAgICAgICAgICAgICAgICAgICAgIEJCaW5kZXIoY29uc3QgQkJpbmRlciYgbyk7Ci0gICAgICAgICAgICBCQmluZGVyJiAgICBvcGVyYXRvcj0oY29uc3QgQkJpbmRlciYgbyk7Ci0KLSAgICBjbGFzcyBFeHRyYXM7Ci0KLSAgICAgICAgICAgIEV4dHJhcyogICAgIG1FeHRyYXM7Ci0gICAgICAgICAgICB2b2lkKiAgICAgICBtUmVzZXJ2ZWQwOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIEJwUmVmQmFzZSA6IHB1YmxpYyB2aXJ0dWFsIFJlZkJhc2UKLXsKLXByb3RlY3RlZDoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBCcFJlZkJhc2UoY29uc3Qgc3A8SUJpbmRlcj4mIG8pOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgICAgIH5CcFJlZkJhc2UoKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICBvbkZpcnN0UmVmKCk7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgb25MYXN0U3Ryb25nUmVmKGNvbnN0IHZvaWQqIGlkKTsKLSAgICB2aXJ0dWFsIGJvb2wgICAgICAgICAgICBvbkluY1N0cm9uZ0F0dGVtcHRlZCh1aW50MzJfdCBmbGFncywgY29uc3Qgdm9pZCogaWQpOwotCi0gICAgaW5saW5lICBJQmluZGVyKiAgICAgICAgcmVtb3RlKCkgICAgICAgICAgICAgICAgeyByZXR1cm4gbVJlbW90ZTsgfQotICAgIGlubGluZSAgSUJpbmRlciogICAgICAgIHJlbW90ZSgpIGNvbnN0ICAgICAgICAgIHsgcmV0dXJuIG1SZW1vdGU7IH0KLQotcHJpdmF0ZToKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBCcFJlZkJhc2UoY29uc3QgQnBSZWZCYXNlJiBvKTsKLSAgICBCcFJlZkJhc2UmICAgICAgICAgICAgICBvcGVyYXRvcj0oY29uc3QgQnBSZWZCYXNlJiBvKTsKLQotICAgIElCaW5kZXIqIGNvbnN0ICAgICAgICAgIG1SZW1vdGU7Ci0gICAgUmVmQmFzZTo6d2Vha3JlZl90eXBlKiAgbVJlZnM7Ci0gICAgdm9sYXRpbGUgaW50MzJfdCAgICAgICAgbVN0YXRlOwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNlbmRpZiAvLyBBTkRST0lEX0JJTkRFUl9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL0JwQmluZGVyLmggYi9pbmNsdWRlL3V0aWxzL0JwQmluZGVyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDdiOTZlMjkuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9CcEJpbmRlci5oCisrKyAvZGV2L251bGwKQEAgLTEsMTIyICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfQlBCSU5ERVJfSAotI2RlZmluZSBBTkRST0lEX0JQQklOREVSX0gKLQotI2luY2x1ZGUgPHV0aWxzL0lCaW5kZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9LZXllZFZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIEJwQmluZGVyIDogcHVibGljIElCaW5kZXIKLXsKLXB1YmxpYzoKLSAgICAgICAgICAgICAgICAgICAgICAgIEJwQmluZGVyKGludDMyX3QgaGFuZGxlKTsKLQotICAgIGlubGluZSAgaW50MzJfdCAgICAgaGFuZGxlKCkgY29uc3QgeyByZXR1cm4gbUhhbmRsZTsgfQotCi0gICAgdmlydHVhbCBTdHJpbmcxNiAgICBnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkgY29uc3Q7Ci0gICAgdmlydHVhbCBib29sICAgICAgICBpc0JpbmRlckFsaXZlKCkgY29uc3Q7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBwaW5nQmluZGVyKCk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHRyYW5zYWN0KCAgIHVpbnQzMl90IGNvZGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBQYXJjZWwmIGRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJjZWwqIHJlcGx5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MgPSAwKTsKLQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgbGlua1RvRGVhdGgoY29uc3Qgc3A8RGVhdGhSZWNpcGllbnQ+JiByZWNpcGllbnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiBjb29raWUgPSBOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MgPSAwKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHVubGlua1RvRGVhdGgoICBjb25zdCB3cDxEZWF0aFJlY2lwaWVudD4mIHJlY2lwaWVudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiBjb29raWUgPSBOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGZsYWdzID0gMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB3cDxEZWF0aFJlY2lwaWVudD4qIG91dFJlY2lwaWVudCA9IE5VTEwpOwotCi0gICAgdmlydHVhbCB2b2lkICAgICAgICBhdHRhY2hPYmplY3QoICAgY29uc3Qgdm9pZCogb2JqZWN0SUQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCogb2JqZWN0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQqIGNsZWFudXBDb29raWUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgb2JqZWN0X2NsZWFudXBfZnVuYyBmdW5jKTsKLSAgICB2aXJ0dWFsIHZvaWQqICAgICAgIGZpbmRPYmplY3QoY29uc3Qgdm9pZCogb2JqZWN0SUQpIGNvbnN0OwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgZGV0YWNoT2JqZWN0KGNvbnN0IHZvaWQqIG9iamVjdElEKTsKLQotICAgIHZpcnR1YWwgQnBCaW5kZXIqICAgcmVtb3RlQmluZGVyKCk7Ci0KLSAgICAgICAgICAgIHN0YXR1c190ICAgIHNldENvbnN0YW50RGF0YShjb25zdCB2b2lkKiBkYXRhLCBzaXplX3Qgc2l6ZSk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICBzZW5kT2JpdHVhcnkoKTsKLQotICAgIGNsYXNzIE9iamVjdE1hbmFnZXIKLSAgICB7Ci0gICAgcHVibGljOgotICAgICAgICAgICAgICAgICAgICBPYmplY3RNYW5hZ2VyKCk7Ci0gICAgICAgICAgICAgICAgICAgIH5PYmplY3RNYW5hZ2VyKCk7Ci0KLSAgICAgICAgdm9pZCAgICAgICAgYXR0YWNoKCBjb25zdCB2b2lkKiBvYmplY3RJRCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiBvYmplY3QsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCogY2xlYW51cENvb2tpZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBJQmluZGVyOjpvYmplY3RfY2xlYW51cF9mdW5jIGZ1bmMpOwotICAgICAgICB2b2lkKiAgICAgICBmaW5kKGNvbnN0IHZvaWQqIG9iamVjdElEKSBjb25zdDsKLSAgICAgICAgdm9pZCAgICAgICAgZGV0YWNoKGNvbnN0IHZvaWQqIG9iamVjdElEKTsKLQotICAgICAgICB2b2lkICAgICAgICBraWxsKCk7Ci0KLSAgICBwcml2YXRlOgotICAgICAgICAgICAgICAgICAgICBPYmplY3RNYW5hZ2VyKGNvbnN0IE9iamVjdE1hbmFnZXImKTsKLSAgICAgICAgT2JqZWN0TWFuYWdlciYgb3BlcmF0b3I9KGNvbnN0IE9iamVjdE1hbmFnZXImKTsKLQotICAgICAgICBzdHJ1Y3QgZW50cnlfdAotICAgICAgICB7Ci0gICAgICAgICAgICB2b2lkKiBvYmplY3Q7Ci0gICAgICAgICAgICB2b2lkKiBjbGVhbnVwQ29va2llOwotICAgICAgICAgICAgSUJpbmRlcjo6b2JqZWN0X2NsZWFudXBfZnVuYyBmdW5jOwotICAgICAgICB9OwotCi0gICAgICAgIEtleWVkVmVjdG9yPGNvbnN0IHZvaWQqLCBlbnRyeV90PiBtT2JqZWN0czsKLSAgICB9OwotCi1wcm90ZWN0ZWQ6Ci0gICAgdmlydHVhbCAgICAgICAgICAgICB+QnBCaW5kZXIoKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIG9uRmlyc3RSZWYoKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIG9uTGFzdFN0cm9uZ1JlZihjb25zdCB2b2lkKiBpZCk7Ci0gICAgdmlydHVhbCBib29sICAgICAgICBvbkluY1N0cm9uZ0F0dGVtcHRlZCh1aW50MzJfdCBmbGFncywgY29uc3Qgdm9pZCogaWQpOwotCi1wcml2YXRlOgotICAgIGNvbnN0ICAgaW50MzJfdCAgICAgICAgICAgICBtSGFuZGxlOwotCi0gICAgc3RydWN0IE9iaXR1YXJ5IHsKLSAgICAgICAgd3A8RGVhdGhSZWNpcGllbnQ+IHJlY2lwaWVudDsKLSAgICAgICAgdm9pZCogY29va2llOwotICAgICAgICB1aW50MzJfdCBmbGFnczsKLSAgICB9OwotCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHJlcG9ydE9uZURlYXRoKGNvbnN0IE9iaXR1YXJ5JiBvYml0KTsKLQotICAgIG11dGFibGUgTXV0ZXggICAgICAgICAgICAgICBtTG9jazsKLSAgICAgICAgICAgIHZvbGF0aWxlIGludDMyX3QgICAgbUFsaXZlOwotICAgICAgICAgICAgdm9sYXRpbGUgaW50MzJfdCAgICBtT2JpdHNTZW50OwotICAgICAgICAgICAgVmVjdG9yPE9iaXR1YXJ5PiogICBtT2JpdHVhcmllczsKLSAgICAgICAgICAgIE9iamVjdE1hbmFnZXIgICAgICAgbU9iamVjdHM7Ci0gICAgICAgICAgICBQYXJjZWwqICAgICAgICAgICAgIG1Db25zdGFudERhdGE7Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2VuZGlmIC8vIEFORFJPSURfQlBCSU5ERVJfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9CdWZmZXIuaCBiL2luY2x1ZGUvdXRpbHMvQnVmZmVyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDhlMjJiMGYuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9CdWZmZXIuaAorKysgL2Rldi9udWxsCkBAIC0xLDEwNyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBfX1VUSUxTX0JVRkZFUl9IX18KLSNkZWZpbmUgX19VVElMU19CVUZGRVJfSF9fIDEKLQotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgQnVmZmVyCi17Ci1wcml2YXRlOgotICAgIGNoYXIgKmJ1ZjsKLSAgICBpbnQgYnVmc2l6OwotICAgIGludCB1c2VkOwotICAgIHZvaWQgZW5zdXJlQ2FwYWNpdHkoaW50IGxlbik7Ci0KLSAgICB2b2lkCi0gICAgbWFrZVJvb21Gb3IoaW50IGxlbikKLSAgICB7Ci0gICAgICAgIGlmIChsZW4gKyB1c2VkID49IGJ1ZnNpeikgewotICAgICAgICAgICAgYnVmc2l6ID0gKGxlbiArIHVzZWQpICogMy8yICsgMjsKLSAgICAgICAgICAgIGNoYXIgKmJsYWggPSBuZXcgY2hhcltidWZzaXpdOwotCi0gICAgICAgICAgICBtZW1jcHkoYmxhaCwgYnVmLCB1c2VkKTsKLSAgICAgICAgICAgIGRlbGV0ZVtdIGJ1ZjsKLSAgICAgICAgICAgIGJ1ZiA9IGJsYWg7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi1wdWJsaWM6Ci0gICAgQnVmZmVyKCkKLSAgICB7Ci0gICAgICAgIGJ1ZnNpeiA9IDE2OwotICAgICAgICBidWYgPSBuZXcgY2hhcltidWZzaXpdOwotICAgICAgICBjbGVhcigpOwotICAgIH0KLQotICAgIH5CdWZmZXIoKQotICAgIHsKLSAgICAgICBkZWxldGVbXSBidWY7Ci0gICAgfQotCi0gICAgdm9pZAotICAgIGNsZWFyKCkKLSAgICB7Ci0gICAgICAgIGJ1ZlswXSA9ICdcMCc7Ci0gICAgICAgIHVzZWQgPSAwOwotICAgIH0KLQotICAgIGludAotICAgIGxlbmd0aCgpCi0gICAgewotICAgICAgICByZXR1cm4gdXNlZDsKLSAgICB9Ci0KLSAgICB2b2lkCi0gICAgYXBwZW5kKGNvbnN0IGNoYXIgYykKLSAgICB7Ci0gICAgICAgIG1ha2VSb29tRm9yKDEpOwotICAgICAgICBidWZbdXNlZF0gPSBjOwotICAgICAgICB1c2VkKys7Ci0gICAgICAgIGJ1Zlt1c2VkXSA9ICdcMCc7Ci0gICAgfQotCi0gICAgdm9pZAotICAgIGFwcGVuZChjb25zdCBjaGFyICpzLCBpbnQgbGVuKQotICAgIHsKLSAgICAgICAgbWFrZVJvb21Gb3IobGVuKTsKLQotICAgICAgICBtZW1jcHkoYnVmICsgdXNlZCwgcywgbGVuKTsKLSAgICAgICAgdXNlZCArPSBsZW47Ci0gICAgICAgIGJ1Zlt1c2VkXSA9ICdcMCc7Ci0gICAgfQotCi0gICAgdm9pZAotICAgIGFwcGVuZChjb25zdCBjaGFyICpzKQotICAgIHsKLSAgICAgICAgYXBwZW5kKHMsIHN0cmxlbihzKSk7Ci0gICAgfQotCi0gICAgY2hhciAqCi0gICAgZ2V0Qnl0ZXMoKQotICAgIHsKLSAgICAgICAgcmV0dXJuIGJ1ZjsKLSAgICB9Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvQnVmZmVyZWRUZXh0T3V0cHV0LmggYi9pbmNsdWRlL3V0aWxzL0J1ZmZlcmVkVGV4dE91dHB1dC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2OWM2MjQwLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvQnVmZmVyZWRUZXh0T3V0cHV0LmgKKysrIC9kZXYvbnVsbApAQCAtMSw2NyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA2IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0JVRkZFUkVEVEVYVE9VVFBVVF9ICi0jZGVmaW5lIEFORFJPSURfQlVGRkVSRURURVhUT1VUUFVUX0gKLQotI2luY2x1ZGUgPHV0aWxzL1RleHRPdXRwdXQuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0jaW5jbHVkZSA8Y3V0aWxzL3Vpby5oPgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgQnVmZmVyZWRUZXh0T3V0cHV0IDogcHVibGljIFRleHRPdXRwdXQKLXsKLXB1YmxpYzoKLSAgICAvLyoqIEZsYWdzIGZvciBjb25zdHJ1Y3RvciAqLwotICAgIGVudW0gewotICAgICAgICBNVUxUSVRIUkVBREVEID0gMHgwMDAxCi0gICAgfTsKLSAgICAKLSAgICAgICAgICAgICAgICAgICAgICAgIEJ1ZmZlcmVkVGV4dE91dHB1dCh1aW50MzJfdCBmbGFncyA9IDApOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgfkJ1ZmZlcmVkVGV4dE91dHB1dCgpOwotICAgIAotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgcHJpbnQoY29uc3QgY2hhciogdHh0LCBzaXplX3QgbGVuKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIG1vdmVJbmRlbnQoaW50IGRlbHRhKTsKLSAgICAKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIHB1c2hCdW5kbGUoKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIHBvcEJ1bmRsZSgpOwotICAgIAotcHJvdGVjdGVkOgotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgd3JpdGVMaW5lcyhjb25zdCBzdHJ1Y3QgaW92ZWMmIHZlYywgc2l6ZV90IE4pID0gMDsKLQotcHJpdmF0ZToKLSAgICBzdHJ1Y3QgQnVmZmVyU3RhdGU7Ci0gICAgc3RydWN0IFRocmVhZFN0YXRlOwotICAgIAotICAgIHN0YXRpYyAgVGhyZWFkU3RhdGUqZ2V0VGhyZWFkU3RhdGUoKTsKLSAgICBzdGF0aWMgIHZvaWQgICAgICAgIHRocmVhZERlc3RydWN0b3Iodm9pZCAqc3QpOwotICAgIAotICAgICAgICAgICAgQnVmZmVyU3RhdGUqZ2V0QnVmZmVyKCkgY29uc3Q7Ci0gICAgICAgICAgICAKLSAgICB1aW50MzJfdCAgICAgICAgICAgIG1GbGFnczsKLSAgICBjb25zdCBpbnQzMl90ICAgICAgIG1TZXE7Ci0gICAgY29uc3QgaW50MzJfdCAgICAgICBtSW5kZXg7Ci0gICAgCi0gICAgTXV0ZXggICAgICAgICAgICAgICBtTG9jazsKLSAgICBCdWZmZXJTdGF0ZSogICAgICAgIG1HbG9iYWxTdGF0ZTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfQlVGRkVSRURURVhUT1VUUFVUX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvQnl0ZU9yZGVyLmggYi9pbmNsdWRlL3V0aWxzL0J5dGVPcmRlci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0YzA2MDY3Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvQnl0ZU9yZGVyLmgKKysrIC9kZXYvbnVsbApAQCAtMSw2OSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA2IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLQotI2lmbmRlZiBfTElCU19VVElMU19CWVRFX09SREVSX0gKLSNkZWZpbmUgX0xJQlNfVVRJTFNfQllURV9PUkRFUl9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpZmRlZiBIQVZFX1dJTlNPQ0sKLSNpbmNsdWRlIDx3aW5zb2NrMi5oPgotI2Vsc2UKLSNpbmNsdWRlIDxuZXRpbmV0L2luLmg+Ci0jZW5kaWYKLQotLyoKLSAqIFRoZXNlIG1hY3JvcyBhcmUgbGlrZSB0aGUgaHRvbi9udG9oIGJ5dGUgc3dhcHBpbmcgbWFjcm9zLAotICogZXhjZXB0IHRoZXkgYWxsb3cgeW91IHRvIHN3YXAgdG8gYW5kIGZyb20gdGhlICJkZXZpY2UiIGJ5dGUKLSAqIG9yZGVyLiAgVGhlIGRldmljZSBieXRlIG9yZGVyIGlzIHRoZSBlbmRpYW5uZXNzIG9mIHRoZSB0YXJnZXQKLSAqIGRldmljZSAtLSBmb3IgdGhlIEFSTSBDUFVzIHdlIHVzZSB0b2RheSwgdGhpcyBpcyBsaXR0bGUgZW5kaWFuLgotICoKLSAqIE5vdGUgdGhhdCB0aGUgYnl0ZSBzd2FwcGluZyBmdW5jdGlvbnMgaGF2ZSBub3QgYmVlbiBvcHRpbWl6ZWQKLSAqIG11Y2g7IHBlcmZvcm1hbmNlIGlzIGN1cnJlbnRseSBub3QgYW4gaXNzdWUgZm9yIHRoZW0gc2luY2UgdGhlCi0gKiBpbnRlbnQgaXMgdG8gYWxsb3cgdXMgdG8gYXZvaWQgYnl0ZSBzd2FwcGluZyBvbiB0aGUgZGV2aWNlLgotICovCi0KLSNkZWZpbmUgREVWSUNFX0JZVEVfT1JERVIgTElUVExFX0VORElBTgotCi0jaWYgQllURV9PUkRFUiA9PSBERVZJQ0VfQllURV9PUkRFUgotCi0jZGVmaW5lCWR0b2hsKHgpCSh4KQotI2RlZmluZQlkdG9ocyh4KQkoeCkKLSNkZWZpbmUJaHRvZGwoeCkJKHgpCi0jZGVmaW5lCWh0b2RzKHgpCSh4KQotCi0jZWxzZQotCi1zdGF0aWMgaW5saW5lIHVpbnQzMl90IGFuZHJvaWRfc3dhcF9sb25nKHVpbnQzMl90IHYpCi17Ci0gICAgcmV0dXJuICh2PDwyNCkgfCAoKHY8PDgpJjB4MDBGRjAwMDApIHwgKCh2Pj44KSYweDAwMDBGRjAwKSB8ICh2Pj4yNCk7Ci19Ci0KLXN0YXRpYyBpbmxpbmUgdWludDE2X3QgYW5kcm9pZF9zd2FwX3Nob3J0KHVpbnQxNl90IHYpCi17Ci0gICAgcmV0dXJuICh2PDw4KSB8ICh2Pj44KTsKLX0KLQotI2RlZmluZQlkdG9obCh4KQkoYW5kcm9pZF9zd2FwX2xvbmcoeCkpCi0jZGVmaW5lCWR0b2hzKHgpCShhbmRyb2lkX3N3YXBfc2hvcnQoeCkpCi0jZGVmaW5lCWh0b2RsKHgpCShhbmRyb2lkX3N3YXBfbG9uZyh4KSkKLSNkZWZpbmUJaHRvZHMoeCkJKGFuZHJvaWRfc3dhcF9zaG9ydCh4KSkKLQotI2VuZGlmCi0KLSNlbmRpZiAvLyBfTElCU19VVElMU19CWVRFX09SREVSX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvQ2FsbFN0YWNrLmggYi9pbmNsdWRlL3V0aWxzL0NhbGxTdGFjay5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjMmM4Y2U1Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvQ2FsbFN0YWNrLmgKKysrIC9kZXYvbnVsbApAQCAtMSw3NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0NBTExTVEFDS19ICi0jZGVmaW5lIEFORFJPSURfQ0FMTFNUQUNLX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvU3RyaW5nOC5oPgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBDYWxsU3RhY2sKLXsKLXB1YmxpYzoKLSAgICBlbnVtIHsKLSAgICAgICAgTUFYX0RFUFRIID0gMzEKLSAgICB9OwotCi0gICAgQ2FsbFN0YWNrKCk7Ci0gICAgQ2FsbFN0YWNrKGNvbnN0IENhbGxTdGFjayYgcmhzKTsKLSAgICB+Q2FsbFN0YWNrKCk7Ci0KLSAgICBDYWxsU3RhY2smIG9wZXJhdG9yID0gKGNvbnN0IENhbGxTdGFjayYgcmhzKTsKLSAgICAKLSAgICBib29sIG9wZXJhdG9yID09IChjb25zdCBDYWxsU3RhY2smIHJocykgY29uc3Q7Ci0gICAgYm9vbCBvcGVyYXRvciAhPSAoY29uc3QgQ2FsbFN0YWNrJiByaHMpIGNvbnN0OwotICAgIGJvb2wgb3BlcmF0b3IgPCAoY29uc3QgQ2FsbFN0YWNrJiByaHMpIGNvbnN0OwotICAgIGJvb2wgb3BlcmF0b3IgPj0gKGNvbnN0IENhbGxTdGFjayYgcmhzKSBjb25zdDsKLSAgICBib29sIG9wZXJhdG9yID4gKGNvbnN0IENhbGxTdGFjayYgcmhzKSBjb25zdDsKLSAgICBib29sIG9wZXJhdG9yIDw9IChjb25zdCBDYWxsU3RhY2smIHJocykgY29uc3Q7Ci0gICAgCi0gICAgY29uc3Qgdm9pZCogb3BlcmF0b3IgW10gKGludCBpbmRleCkgY29uc3Q7Ci0gICAgCi0gICAgdm9pZCBjbGVhcigpOwotCi0gICAgdm9pZCB1cGRhdGUoaW50MzJfdCBpZ25vcmVEZXB0aD0wLCBpbnQzMl90IG1heERlcHRoPU1BWF9ERVBUSCk7Ci0KLSAgICAvLyBEdW1wIGEgc3RhY2sgdHJhY2UgdG8gdGhlIGxvZwotICAgIHZvaWQgZHVtcChjb25zdCBjaGFyKiBwcmVmaXggPSAwKSBjb25zdDsKLQotICAgIC8vIFJldHVybiBhIHN0cmluZyAocG9zc2libHkgdmVyeSBsb25nKSBjb250YWluaW5nIHRoZSBjb21wbGV0ZSBzdGFjayB0cmFjZQotICAgIFN0cmluZzggdG9TdHJpbmcoY29uc3QgY2hhciogcHJlZml4ID0gMCkgY29uc3Q7Ci0gICAgCi0gICAgc2l6ZV90IHNpemUoKSBjb25zdCB7IHJldHVybiBtQ291bnQ7IH0KLQotcHJpdmF0ZToKLSAgICAvLyBJbnRlcm5hbCBoZWxwZXIgZnVuY3Rpb24KLSAgICBTdHJpbmc4IHRvU3RyaW5nU2luZ2xlTGV2ZWwoY29uc3QgY2hhciogcHJlZml4LCBpbnQzMl90IGxldmVsKSBjb25zdDsKLQotICAgIHNpemVfdCAgICAgIG1Db3VudDsKLSAgICBjb25zdCB2b2lkKiBtU3RhY2tbTUFYX0RFUFRIXTsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNlbmRpZiAvLyBBTkRST0lEX0NBTExTVEFDS19ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL0RlYnVnLmggYi9pbmNsdWRlL3V0aWxzL0RlYnVnLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGE2NjJiOWMuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9EZWJ1Zy5oCisrKyAvZGV2L251bGwKQEAgLTEsNDUgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBEZWJ1Z2dpbmcgdG9vbHMuICBUaGVzZSBzaG91bGQgYmUgYWJsZSB0byBiZSBzdHJpcHBlZAotLy8gaW4gcmVsZWFzZSBidWlsZHMuCi0vLwotI2lmbmRlZiBBTkRST0lEX0RFQlVHX0gKLSNkZWZpbmUgQU5EUk9JRF9ERUJVR19ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi10ZW1wbGF0ZTxib29sPiBzdHJ1Y3QgQ29tcGlsZVRpbWVBc3NlcnQ7Ci10ZW1wbGF0ZTw+IHN0cnVjdCBDb21waWxlVGltZUFzc2VydDx0cnVlPiB7fTsKLQotY29uc3QgY2hhciogc3RyaW5nRm9ySW5kZW50KGludDMyX3QgaW5kZW50TGV2ZWwpOwotCi10eXBlZGVmIHZvaWQgKCpkZWJ1Z1ByaW50RnVuYykodm9pZCogY29va2llLCBjb25zdCBjaGFyKiB0eHQpOwotCi12b2lkIHByaW50VHlwZUNvZGUodWludDMyX3QgdHlwZUNvZGUsCi0gICAgZGVidWdQcmludEZ1bmMgZnVuYyA9IDAsIHZvaWQqIGNvb2tpZSA9IDApOwotdm9pZCBwcmludEhleERhdGEoaW50MzJfdCBpbmRlbnQsIGNvbnN0IHZvaWQgKmJ1Ziwgc2l6ZV90IGxlbmd0aCwKLSAgICBzaXplX3QgYnl0ZXNQZXJMaW5lPTE2LCBpbnQzMl90IHNpbmdsZUxpbmVCeXRlc0N1dG9mZj0xNiwKLSAgICBzaXplX3QgYWxpZ25tZW50PTAsIGJvb2wgY0FycmF5U3R5bGU9ZmFsc2UsCi0gICAgZGVidWdQcmludEZ1bmMgZnVuYyA9IDAsIHZvaWQqIGNvb2tpZSA9IDApOwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9ERUJVR19ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL0VuZGlhbi5oIGIvaW5jbHVkZS91dGlscy9FbmRpYW4uaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTlmMjUwNC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0VuZGlhbi5oCisrKyAvZGV2L251bGwKQEAgLTEsNDAgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBBbmRyb2lkIGVuZGlhbi1uZXNzIGRlZmluZXMuCi0vLwotI2lmbmRlZiBfTElCU19VVElMU19FTkRJQU5fSAotI2RlZmluZSBfTElCU19VVElMU19FTkRJQU5fSAotCi0jaWYgZGVmaW5lZChIQVZFX0VORElBTl9IKQotCi0jaW5jbHVkZSA8ZW5kaWFuLmg+Ci0KLSNlbHNlIC8qbm90IEhBVkVfRU5ESUFOX0gqLwotCi0jZGVmaW5lIF9fQklHX0VORElBTiAweDEwMDAKLSNkZWZpbmUgX19MSVRUTEVfRU5ESUFOIDB4MDAwMQotCi0jaWYgZGVmaW5lZChIQVZFX0xJVFRMRV9FTkRJQU4pCi0jIGRlZmluZSBfX0JZVEVfT1JERVIgX19MSVRUTEVfRU5ESUFOCi0jZWxzZQotIyBkZWZpbmUgX19CWVRFX09SREVSIF9fQklHX0VORElBTgotI2VuZGlmCi0KLSNlbmRpZiAvKm5vdCBIQVZFX0VORElBTl9IKi8KLQotI2VuZGlmIC8qX0xJQlNfVVRJTFNfRU5ESUFOX0gqLwpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9FcnJvcnMuaCBiL2luY2x1ZGUvdXRpbHMvRXJyb3JzLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDFiZjllNmYuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9FcnJvcnMuaAorKysgL2Rldi9udWxsCkBAIC0xLDg3ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfRVJST1JTX0gKLSNkZWZpbmUgQU5EUk9JRF9FUlJPUlNfSAotCi0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyB1c2UgdGhpcyB0eXBlIHRvIHJldHVybiBlcnJvciBjb2RlcwotI2lmZGVmIEhBVkVfTVNfQ19SVU5USU1FCi10eXBlZGVmIGludCAgICAgICAgIHN0YXR1c190OwotI2Vsc2UKLXR5cGVkZWYgaW50MzJfdCAgICAgc3RhdHVzX3Q7Ci0jZW5kaWYKLQotLyogdGhlIE1TIEMgcnVudGltZSBsYWNrcyBhIGZldyBlcnJvciBjb2RlcyAqLwotCi0vKgotICogRXJyb3IgY29kZXMuIAotICogQWxsIGVycm9yIGNvZGVzIGFyZSBuZWdhdGl2ZSB2YWx1ZXMuCi0gKi8KLQotLy8gV2luMzIgI2RlZmluZXMgTk9fRVJST1IgYXMgd2VsbC4gIEl0IGhhcyB0aGUgc2FtZSB2YWx1ZSwgc28gdGhlcmUncyBubwotLy8gcmVhbCBjb25mbGljdCwgdGhvdWdoIGl0J3MgYSBiaXQgYXdrd2FyZC4KLSNpZmRlZiBfV0lOMzIKLSMgdW5kZWYgTk9fRVJST1IKLSNlbmRpZgotIAotZW51bSB7Ci0gICAgT0sgICAgICAgICAgICAgICAgPSAwLCAgICAvLyBFdmVyeXRoaW5nJ3Mgc3dlbGwuCi0gICAgTk9fRVJST1IgICAgICAgICAgPSAwLCAgICAvLyBObyBlcnJvcnMuCi0gICAgCi0gICAgVU5LTk9XTl9FUlJPUiAgICAgICA9IDB4ODAwMDAwMDAsCi0KLSAgICBOT19NRU1PUlkgICAgICAgICAgID0gLUVOT01FTSwKLSAgICBJTlZBTElEX09QRVJBVElPTiAgID0gLUVOT1NZUywKLSAgICBCQURfVkFMVUUgICAgICAgICAgID0gLUVJTlZBTCwKLSAgICBCQURfVFlQRSAgICAgICAgICAgID0gMHg4MDAwMDAwMSwKLSAgICBOQU1FX05PVF9GT1VORCAgICAgID0gLUVOT0VOVCwKLSAgICBQRVJNSVNTSU9OX0RFTklFRCAgID0gLUVQRVJNLAotICAgIE5PX0lOSVQgICAgICAgICAgICAgPSAtRU5PREVWLAotICAgIEFMUkVBRFlfRVhJU1RTICAgICAgPSAtRUVYSVNULAotICAgIERFQURfT0JKRUNUICAgICAgICAgPSAtRVBJUEUsCi0gICAgRkFJTEVEX1RSQU5TQUNUSU9OICA9IDB4ODAwMDAwMDIsCi0gICAgSlBBUktTX0JST0tFX0lUICAgICA9IC1FUElQRSwKLSNpZiAhZGVmaW5lZChIQVZFX01TX0NfUlVOVElNRSkKLSAgICBCQURfSU5ERVggICAgICAgICAgID0gLUVPVkVSRkxPVywKLSAgICBOT1RfRU5PVUdIX0RBVEEgICAgID0gLUVOT0RBVEEsCi0gICAgV09VTERfQkxPQ0sgICAgICAgICA9IC1FV09VTERCTE9DSywgCi0gICAgVElNRURfT1VUICAgICAgICAgICA9IC1FVElNRSwKLSAgICBVTktOT1dOX1RSQU5TQUNUSU9OID0gLUVCQURNU0csCi0jZWxzZSAgICAKLSAgICBCQURfSU5ERVggICAgICAgICAgID0gLUUyQklHLAotICAgIE5PVF9FTk9VR0hfREFUQSAgICAgPSAweDgwMDAwMDAzLAotICAgIFdPVUxEX0JMT0NLICAgICAgICAgPSAweDgwMDAwMDA0LAotICAgIFRJTUVEX09VVCAgICAgICAgICAgPSAweDgwMDAwMDA1LAotICAgIFVOS05PV05fVFJBTlNBQ1RJT04gPSAweDgwMDAwMDA2LAotI2VuZGlmICAgIAotfTsKLQotLy8gUmVzdG9yZSBkZWZpbmU7IGVudW1lcmF0aW9uIGlzIGluICJhbmRyb2lkIiBuYW1lc3BhY2UsIHNvIHRoZSB2YWx1ZSBkZWZpbmVkCi0vLyB0aGVyZSB3b24ndCB3b3JrIGZvciBXaW4zMiBjb2RlIGluIGEgZGlmZmVyZW50IG5hbWVzcGFjZS4KLSNpZmRlZiBfV0lOMzIKLSMgZGVmaW5lIE5PX0VSUk9SIDBMCi0jZW5kaWYKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLSAgICAKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICAgIAotI2VuZGlmIC8vIEFORFJPSURfRVJST1JTX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvRmlsZU1hcC5oIGIvaW5jbHVkZS91dGlscy9GaWxlTWFwLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDhkZmQzYmUuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9GaWxlTWFwLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxMzQgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBFbmNhcHN1bGF0ZSBhIHNoYXJlZCBmaWxlIG1hcHBpbmcuCi0vLwotI2lmbmRlZiBfX0xJQlNfRklMRV9NQVBfSAotI2RlZmluZSBfX0xJQlNfRklMRV9NQVBfSAotCi0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpZmRlZiBIQVZFX1dJTjMyX0ZJTEVNQVAKLSNpbmNsdWRlIDx3aW5kb3dzLmg+Ci0jZW5kaWYKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vKgotICogVGhpcyByZXByZXNlbnRzIGEgbWVtb3J5LW1hcHBlZCBmaWxlLiAgSXQgbWlnaHQgYmUgdGhlIGVudGlyZSBmaWxlIG9yCi0gKiBvbmx5IHBhcnQgb2YgaXQuICBUaGlzIHJlcXVpcmVzIGEgbGl0dGxlIGJvb2trZWVwaW5nIGJlY2F1c2UgdGhlIG1hcHBpbmcKLSAqIG5lZWRzIHRvIGJlIGFsaWduZWQgb24gcGFnZSBib3VuZGFyaWVzLCBhbmQgaW4gc29tZSBjYXNlcyB3ZSdkIGxpa2UgdG8KLSAqIGhhdmUgbXVsdGlwbGUgcmVmZXJlbmNlcyB0byB0aGUgbWFwcGVkIGFyZWEgd2l0aG91dCBjcmVhdGluZyBhZGRpdGlvbmFsCi0gKiBtYXBzLgotICoKLSAqIFRoaXMgYWx3YXlzIHVzZXMgTUFQX1NIQVJFRC4KLSAqCi0gKiBUT0RPOiB3ZSBzaG91bGQgYmUgYWJsZSB0byBjcmVhdGUgYSBuZXcgRmlsZU1hcCB0aGF0IGlzIGEgc3Vic2V0IG9mCi0gKiBhbiBleGlzdGluZyBGaWxlTWFwIGFuZCBzaGFyZXMgdGhlIHVuZGVybHlpbmcgbWFwcGVkIHBhZ2VzLiAgUmVxdWlyZXMKLSAqIGNvbXBsZXRpbmcgdGhlIHJlZmNvdW50aW5nIHN0dWZmIGFuZCBwb3NzaWJseSBpbnRyb2R1Y2luZyB0aGUgbm90aW9uCi0gKiBvZiBhIEZpbGVNYXAgaGllcmFyY2h5LgotICovCi1jbGFzcyBGaWxlTWFwIHsKLXB1YmxpYzoKLSAgICBGaWxlTWFwKHZvaWQpOwotCi0gICAgLyoKLSAgICAgKiBDcmVhdGUgYSBuZXcgbWFwcGluZyBvbiBhbiBvcGVuIGZpbGUuCi0gICAgICoKLSAgICAgKiBDbG9zaW5nIHRoZSBmaWxlIGRlc2NyaXB0b3IgZG9lcyBub3QgdW5tYXAgdGhlIHBhZ2VzLCBzbyB3ZSBkb24ndAotICAgICAqIGNsYWltIG93bmVyc2hpcCBvZiB0aGUgZmQuCi0gICAgICoKLSAgICAgKiBSZXR1cm5zICJmYWxzZSIgb24gZmFpbHVyZS4KLSAgICAgKi8KLSAgICBib29sIGNyZWF0ZShjb25zdCBjaGFyKiBvcmlnRmlsZU5hbWUsIGludCBmZCwKLSAgICAgICAgICAgICAgICBvZmZfdCBvZmZzZXQsIHNpemVfdCBsZW5ndGgsIGJvb2wgcmVhZE9ubHkpOwotCi0gICAgLyoKLSAgICAgKiBSZXR1cm4gdGhlIG5hbWUgb2YgdGhlIGZpbGUgdGhpcyBtYXAgY2FtZSBmcm9tLCBpZiBrbm93bi4KLSAgICAgKi8KLSAgICBjb25zdCBjaGFyKiBnZXRGaWxlTmFtZSh2b2lkKSBjb25zdCB7IHJldHVybiBtRmlsZU5hbWU7IH0KLSAgICAKLSAgICAvKgotICAgICAqIEdldCBhIHBvaW50ZXIgdG8gdGhlIHBpZWNlIG9mIHRoZSBmaWxlIHdlIHJlcXVlc3RlZC4KLSAgICAgKi8KLSAgICB2b2lkKiBnZXREYXRhUHRyKHZvaWQpIGNvbnN0IHsgcmV0dXJuIG1EYXRhUHRyOyB9Ci0KLSAgICAvKgotICAgICAqIEdldCB0aGUgbGVuZ3RoIHdlIHJlcXVlc3RlZC4KLSAgICAgKi8KLSAgICBzaXplX3QgZ2V0RGF0YUxlbmd0aCh2b2lkKSBjb25zdCB7IHJldHVybiBtRGF0YUxlbmd0aDsgfQotCi0gICAgLyoKLSAgICAgKiBHZXQgdGhlIGRhdGEgb2Zmc2V0IHVzZWQgdG8gY3JlYXRlIHRoaXMgbWFwLgotICAgICAqLwotICAgIG9mZl90IGdldERhdGFPZmZzZXQodm9pZCkgY29uc3QgeyByZXR1cm4gbURhdGFPZmZzZXQ7IH0KLQotICAgIC8qCi0gICAgICogR2V0IGEgImNvcHkiIG9mIHRoZSBvYmplY3QuCi0gICAgICovCi0gICAgRmlsZU1hcCogYWNxdWlyZSh2b2lkKSB7IG1SZWZDb3VudCsrOyByZXR1cm4gdGhpczsgfQotCi0gICAgLyoKLSAgICAgKiBDYWxsIHRoaXMgd2hlbiBtYXBwaW5nIGlzIG5vIGxvbmdlciBuZWVkZWQuCi0gICAgICovCi0gICAgdm9pZCByZWxlYXNlKHZvaWQpIHsKLSAgICAgICAgaWYgKC0tbVJlZkNvdW50IDw9IDApCi0gICAgICAgICAgICBkZWxldGUgdGhpczsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFRoaXMgbWFwcyBkaXJlY3RseSB0byBtYWR2aXNlKCkgdmFsdWVzLCBidXQgYWxsb3dzIHVzIHRvIGF2b2lkCi0gICAgICogaW5jbHVkaW5nIDxzeXMvbW1hbi5oPiBldmVyeXdoZXJlLgotICAgICAqLwotICAgIGVudW0gTWFwQWR2aWNlIHsKLSAgICAgICAgTk9STUFMLCBSQU5ET00sIFNFUVVFTlRJQUwsIFdJTExORUVELCBET05UTkVFRAotICAgIH07Ci0KLSAgICAvKgotICAgICAqIEFwcGx5IGFuIG1hZHZpc2UoKSBjYWxsIHRvIHRoZSBlbnRpcmUgZmlsZS4KLSAgICAgKgotICAgICAqIFJldHVybnMgMCBvbiBzdWNjZXNzLCAtMSBvbiBmYWlsdXJlLgotICAgICAqLwotICAgIGludCBhZHZpc2UoTWFwQWR2aWNlIGFkdmljZSk7Ci0KLXByb3RlY3RlZDoKLSAgICAvLyBkb24ndCBkZWxldGUgb2JqZWN0czsgY2FsbCByZWxlYXNlKCkKLSAgICB+RmlsZU1hcCh2b2lkKTsKLQotcHJpdmF0ZToKLSAgICAvLyB0aGVzZSBhcmUgbm90IGltcGxlbWVudGVkCi0gICAgRmlsZU1hcChjb25zdCBGaWxlTWFwJiBzcmMpOwotICAgIGNvbnN0IEZpbGVNYXAmIG9wZXJhdG9yPShjb25zdCBGaWxlTWFwJiBzcmMpOwotCi0gICAgaW50ICAgICAgICAgbVJlZkNvdW50OyAgICAgIC8vIHJlZmVyZW5jZSBjb3VudAotICAgIGNoYXIqICAgICAgIG1GaWxlTmFtZTsgICAgICAvLyBvcmlnaW5hbCBmaWxlIG5hbWUsIGlmIGtub3duCi0gICAgdm9pZCogICAgICAgbUJhc2VQdHI7ICAgICAgIC8vIGJhc2Ugb2YgbW1hcCBhcmVhOyBwYWdlIGFsaWduZWQKLSAgICBzaXplX3QgICAgICBtQmFzZUxlbmd0aDsgICAgLy8gbGVuZ3RoLCBtZWFzdXJlZCBmcm9tICJtQmFzZVB0ciIKLSAgICBvZmZfdCAgICAgICBtRGF0YU9mZnNldDsgICAgLy8gb2Zmc2V0IHVzZWQgd2hlbiBtYXAgd2FzIGNyZWF0ZWQKLSAgICB2b2lkKiAgICAgICBtRGF0YVB0cjsgICAgICAgLy8gc3RhcnQgb2YgcmVxdWVzdGVkIGRhdGEsIG9mZnNldCBmcm9tIGJhc2UKLSAgICBzaXplX3QgICAgICBtRGF0YUxlbmd0aDsgICAgLy8gbGVuZ3RoLCBtZWFzdXJlZCBmcm9tICJtRGF0YVB0ciIKLSNpZmRlZiBIQVZFX1dJTjMyX0ZJTEVNQVAKLSAgICBIQU5ETEUgICAgICBtRmlsZUhhbmRsZTsgICAgLy8gV2luMzIgZmlsZSBoYW5kbGUKLSAgICBIQU5ETEUgICAgICBtRmlsZU1hcHBpbmc7ICAgLy8gV2luMzIgZmlsZSBtYXBwaW5nIGhhbmRsZQotI2VuZGlmCi0KLSAgICBzdGF0aWMgbG9uZyBtUGFnZVNpemU7Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gX19MSUJTX0ZJTEVfTUFQX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvSUJpbmRlci5oIGIvaW5jbHVkZS91dGlscy9JQmluZGVyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDczNzAzMzAuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9JQmluZGVyLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxNTkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9JQklOREVSX0gKLSNkZWZpbmUgQU5EUk9JRF9JQklOREVSX0gKLQotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmcxNi5oPgotI2luY2x1ZGUgPHV0aWxzL1ZlY3Rvci5oPgotCi0KLSNkZWZpbmUgQl9QQUNLX0NIQVJTKGMxLCBjMiwgYzMsIGM0KSBcCi0gICAgKCgoKGMxKTw8MjQpKSB8ICgoKGMyKTw8MTYpKSB8ICgoKGMzKTw8OCkpIHwgKGM0KSkKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIEJCaW5kZXI7Ci1jbGFzcyBCcEJpbmRlcjsKLWNsYXNzIElJbnRlcmZhY2U7Ci1jbGFzcyBQYXJjZWw7Ci0KLS8qKgotICogQmFzZSBjbGFzcyBhbmQgbG93LWxldmVsIHByb3RvY29sIGZvciBhIHJlbW90YWJsZSBvYmplY3QuCi0gKiBZb3UgY2FuIGRlcml2ZSBmcm9tIHRoaXMgY2xhc3MgdG8gY3JlYXRlIGFuIG9iamVjdCBmb3Igd2hpY2ggb3RoZXIKLSAqIHByb2Nlc3NlcyBjYW4gaG9sZCByZWZlcmVuY2VzIHRvIGl0LiAgQ29tbXVuaWNhdGlvbiBiZXR3ZWVuIHByb2Nlc3NlcwotICogKG1ldGhvZCBjYWxscywgcHJvcGVydHkgZ2V0IGFuZCBzZXQpIGlzIGRvd24gdGhyb3VnaCBhIGxvdy1sZXZlbAotICogcHJvdG9jb2wgaW1wbGVtZW50ZWQgb24gdG9wIG9mIHRoZSB0cmFuc2FjdCgpIEFQSS4KLSAqLwotY2xhc3MgSUJpbmRlciA6IHB1YmxpYyB2aXJ0dWFsIFJlZkJhc2UKLXsKLXB1YmxpYzoKLSAgICBlbnVtIHsKLSAgICAgICAgRklSU1RfQ0FMTF9UUkFOU0FDVElPTiAgPSAweDAwMDAwMDAxLAotICAgICAgICBMQVNUX0NBTExfVFJBTlNBQ1RJT04gICA9IDB4MDBmZmZmZmYsCi0KLSAgICAgICAgUElOR19UUkFOU0FDVElPTiAgICAgICAgPSBCX1BBQ0tfQ0hBUlMoJ18nLCdQJywnTicsJ0cnKSwKLSAgICAgICAgRFVNUF9UUkFOU0FDVElPTiAgICAgICAgPSBCX1BBQ0tfQ0hBUlMoJ18nLCdEJywnTScsJ1AnKSwKLSAgICAgICAgSU5URVJGQUNFX1RSQU5TQUNUSU9OICAgPSBCX1BBQ0tfQ0hBUlMoJ18nLCAnTicsICdUJywgJ0YnKSwKLQotICAgICAgICAvLyBDb3JyZXNwb25kcyB0byB0Zk9uZVdheSAtLSBhbiBhc3luY2hyb25vdXMgY2FsbC4KLSAgICAgICAgRkxBR19PTkVXQVkgICAgICAgICAgICAgPSAweDAwMDAwMDAxCi0gICAgfTsKLQotICAgIGlubGluZSAgICAgICAgICAgICAgICAgIElCaW5kZXIoKSB7IH0KLQotICAgIC8qKgotICAgICAqIENoZWNrIGlmIHRoaXMgSUJpbmRlciBpbXBsZW1lbnRzIHRoZSBpbnRlcmZhY2UgbmFtZWQgYnkKLSAgICAgKiBAYSBkZXNjcmlwdG9yLiAgSWYgaXQgZG9lcywgdGhlIGJhc2UgcG9pbnRlciB0byBpdCBpcyByZXR1cm5lZCwKLSAgICAgKiB3aGljaCB5b3UgY2FuIHNhZmVseSBzdGF0aWNfY2FzdDw+IHRvIHRoZSBjb25jcmV0ZSBDKysgaW50ZXJmYWNlLgotICAgICAqLwotICAgIHZpcnR1YWwgc3A8SUludGVyZmFjZT4gIHF1ZXJ5TG9jYWxJbnRlcmZhY2UoY29uc3QgU3RyaW5nMTYmIGRlc2NyaXB0b3IpOwotCi0gICAgLyoqCi0gICAgICogUmV0dXJuIHRoZSBjYW5vbmljYWwgbmFtZSBvZiB0aGUgaW50ZXJmYWNlIHByb3ZpZGVkIGJ5IHRoaXMgSUJpbmRlcgotICAgICAqIG9iamVjdC4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIFN0cmluZzE2ICAgICAgICBnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkgY29uc3QgPSAwOwotCi0gICAgdmlydHVhbCBib29sICAgICAgICAgICAgaXNCaW5kZXJBbGl2ZSgpIGNvbnN0ID0gMDsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICBwaW5nQmluZGVyKCkgPSAwOwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgIGR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKSA9IDA7Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICB0cmFuc2FjdCggICB1aW50MzJfdCBjb2RlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFBhcmNlbCYgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJjZWwqIHJlcGx5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGZsYWdzID0gMCkgPSAwOwotCi0gICAgLyoqCi0gICAgICogVGhpcyBtZXRob2QgYWxsb3dzIHlvdSB0byBhZGQgZGF0YSB0aGF0IGlzIHRyYW5zcG9ydGVkIHRocm91Z2gKLSAgICAgKiBJUEMgYWxvbmcgd2l0aCB5b3VyIElCaW5kZXIgcG9pbnRlci4gIFdoZW4gaW1wbGVtZW50aW5nIGEgQmluZGVyCi0gICAgICogb2JqZWN0LCBvdmVycmlkZSBpdCB0byB3cml0ZSB5b3VyIGRlc2lyZWQgZGF0YSBpbiB0byBAYSBvdXREYXRhLgotICAgICAqIFlvdSBjYW4gdGhlbiBjYWxsIGdldENvbnN0YW50RGF0YSgpIG9uIHlvdXIgSUJpbmRlciB0byByZXRyaWV2ZQotICAgICAqIHRoYXQgZGF0YSwgZnJvbSBhbnkgcHJvY2Vzcy4gIFlvdSBNVVNUIHJldHVybiB0aGUgbnVtYmVyIG9mIGJ5dGVzCi0gICAgICogd3JpdHRlbiBpbiB0byB0aGUgcGFyY2VsIChpbmNsdWRpbmcgcGFkZGluZykuCi0gICAgICovCi0gICAgY2xhc3MgRGVhdGhSZWNpcGllbnQgOiBwdWJsaWMgdmlydHVhbCBSZWZCYXNlCi0gICAgewotICAgIHB1YmxpYzoKLSAgICAgICAgdmlydHVhbCB2b2lkIGJpbmRlckRpZWQoY29uc3Qgd3A8SUJpbmRlcj4mIHdobykgPSAwOwotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBSZWdpc3RlciB0aGUgQGEgcmVjaXBpZW50IGZvciBhIG5vdGlmaWNhdGlvbiBpZiB0aGlzIGJpbmRlcgotICAgICAqIGdvZXMgYXdheS4gIElmIHRoaXMgYmluZGVyIG9iamVjdCB1bmV4cGVjdGVkbHkgZ29lcyBhd2F5Ci0gICAgICogKHR5cGljYWxseSBiZWNhdXNlIGl0cyBob3N0aW5nIHByb2Nlc3MgaGFzIGJlZW4ga2lsbGVkKSwKLSAgICAgKiB0aGVuIERlYXRoUmVjaXBpZW50OjpiaW5kZXJEaWVkKCkgd2lsbCBiZSBjYWxsZWQgd2l0aCBhIHJlZmVyZW5lCi0gICAgICogdG8gdGhpcy4KLSAgICAgKgotICAgICAqIFRoZSBAYSBjb29raWUgaXMgb3B0aW9uYWwgLS0gaWYgbm9uLU5VTEwsIGl0IHNob3VsZCBiZSBhCi0gICAgICogbWVtb3J5IGFkZHJlc3MgdGhhdCB5b3Ugb3duICh0aGF0IGlzLCB5b3Uga25vdyBpdCBpcyB1bmlxdWUpLgotICAgICAqCi0gICAgICogQG5vdGUgWW91IHdpbGwgb25seSByZWNlaXZlIGRlYXRoIG5vdGlmaWNhdGlvbnMgZm9yIHJlbW90ZSBiaW5kZXJzLAotICAgICAqIGFzIGxvY2FsIGJpbmRlcnMgYnkgZGVmaW5pdGlvbiBjYW4ndCBkaWUgd2l0aG91dCB5b3UgZHlpbmcgYXMgd2VsbC4KLSAgICAgKiBUcnlpbmcgdG8gdXNlIHRoaXMgZnVuY3Rpb24gb24gYSBsb2NhbCBiaW5kZXIgd2lsbCByZXN1bHQgaW4gYW4KLSAgICAgKiBJTlZBTElEX09QRVJBVElPTiBjb2RlIGJlaW5nIHJldHVybmVkIGFuZCBub3RoaW5nIGhhcHBlbmluZy4KLSAgICAgKgotICAgICAqIEBub3RlIFRoaXMgbGluayBhbHdheXMgaG9sZHMgYSB3ZWFrIHJlZmVyZW5jZSB0byBpdHMgcmVjaXBpZW50LgotICAgICAqCi0gICAgICogQG5vdGUgWW91IHdpbGwgb25seSByZWNlaXZlIGEgd2VhayByZWZlcmVuY2UgdG8gdGhlIGRlYWQKLSAgICAgKiBiaW5kZXIuICBZb3Ugc2hvdWxkIG5vdCB0cnkgdG8gcHJvbW90ZSB0aGlzIHRvIGEgc3Ryb25nIHJlZmVyZW5jZS4KLSAgICAgKiAoTm9yIHNob3VsZCB5b3UgbmVlZCB0bywgYXMgdGhlcmUgaXMgbm90aGluZyB1c2VmdWwgeW91IGNhbgotICAgICAqIGRpcmVjdGx5IGRvIHdpdGggaXQgbm93IHRoYXQgaXQgaGFzIHBhc3NlZCBvbi4pCi0gICAgICovCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgbGlua1RvRGVhdGgoY29uc3Qgc3A8RGVhdGhSZWNpcGllbnQ+JiByZWNpcGllbnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCogY29va2llID0gTlVMTCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDApID0gMDsKLQotICAgIC8qKgotICAgICAqIFJlbW92ZSBhIHByZXZpb3VzbHkgcmVnaXN0ZXJlZCBkZWF0aCBub3RpZmljYXRpb24uCi0gICAgICogVGhlIEBhIHJlY2lwaWVudCB3aWxsIG5vIGxvbmdlciBiZSBjYWxsZWQgaWYgdGhpcyBvYmplY3QKLSAgICAgKiBkaWVzLiAgVGhlIEBhIGNvb2tpZSBpcyBvcHRpb25hbC4gIElmIG5vbi1OVUxMLCB5b3UgY2FuCi0gICAgICogc3VwcGx5IGEgTlVMTCBAYSByZWNpcGllbnQsIGFuZCB0aGUgcmVjaXBpZW50IHByZXZpb3VzbHkKLSAgICAgKiBhZGRlZCB3aXRoIHRoYXQgY29va2llIHdpbGwgYmUgdW5saW5rZWQuCi0gICAgICovCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICAgICAgdW5saW5rVG9EZWF0aCggIGNvbnN0IHdwPERlYXRoUmVjaXBpZW50PiYgcmVjaXBpZW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiBjb29raWUgPSBOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHdwPERlYXRoUmVjaXBpZW50Piogb3V0UmVjaXBpZW50ID0gTlVMTCkgPSAwOwotCi0gICAgdmlydHVhbCBib29sICAgICAgICAgICAgY2hlY2tTdWJjbGFzcyhjb25zdCB2b2lkKiBzdWJjbGFzc0lEKSBjb25zdDsKLQotICAgIHR5cGVkZWYgdm9pZCAoKm9iamVjdF9jbGVhbnVwX2Z1bmMpKGNvbnN0IHZvaWQqIGlkLCB2b2lkKiBvYmosIHZvaWQqIGNsZWFudXBDb29raWUpOwotCi0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgYXR0YWNoT2JqZWN0KCAgIGNvbnN0IHZvaWQqIG9iamVjdElELAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiBvYmplY3QsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQqIGNsZWFudXBDb29raWUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9iamVjdF9jbGVhbnVwX2Z1bmMgZnVuYykgPSAwOwotICAgIHZpcnR1YWwgdm9pZCogICAgICAgICAgIGZpbmRPYmplY3QoY29uc3Qgdm9pZCogb2JqZWN0SUQpIGNvbnN0ID0gMDsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICBkZXRhY2hPYmplY3QoY29uc3Qgdm9pZCogb2JqZWN0SUQpID0gMDsKLQotICAgIHZpcnR1YWwgQkJpbmRlciogICAgICAgIGxvY2FsQmluZGVyKCk7Ci0gICAgdmlydHVhbCBCcEJpbmRlciogICAgICAgcmVtb3RlQmluZGVyKCk7Ci0KLXByb3RlY3RlZDoKLSAgICBpbmxpbmUgdmlydHVhbCAgICAgICAgICB+SUJpbmRlcigpIHsgfQotCi1wcml2YXRlOgotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNlbmRpZiAvLyBBTkRST0lEX0lCSU5ERVJfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9JSW50ZXJmYWNlLmggYi9pbmNsdWRlL3V0aWxzL0lJbnRlcmZhY2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTU5NzIyYS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0lJbnRlcmZhY2UuaAorKysgL2Rldi9udWxsCkBAIC0xLDEzNSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLSNpZm5kZWYgQU5EUk9JRF9JSU5URVJGQUNFX0gKLSNkZWZpbmUgQU5EUk9JRF9JSU5URVJGQUNFX0gKLQotI2luY2x1ZGUgPHV0aWxzL0JpbmRlci5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgSUludGVyZmFjZSA6IHB1YmxpYyB2aXJ0dWFsIFJlZkJhc2UKLXsKLXB1YmxpYzoKLSAgICAgICAgICAgIHNwPElCaW5kZXI+ICAgICAgICAgYXNCaW5kZXIoKTsKLSAgICAgICAgICAgIHNwPGNvbnN0IElCaW5kZXI+ICAgYXNCaW5kZXIoKSBjb25zdDsKLQotcHJvdGVjdGVkOgotICAgIHZpcnR1YWwgSUJpbmRlciogICAgICAgICAgICBvbkFzQmluZGVyKCkgPSAwOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBJTlRFUkZBQ0U+Ci1pbmxpbmUgc3A8SU5URVJGQUNFPiBpbnRlcmZhY2VfY2FzdChjb25zdCBzcDxJQmluZGVyPiYgb2JqKQotewotICAgIHJldHVybiBJTlRFUkZBQ0U6OmFzSW50ZXJmYWNlKG9iaik7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdGVtcGxhdGU8dHlwZW5hbWUgSU5URVJGQUNFPgotY2xhc3MgQm5JbnRlcmZhY2UgOiBwdWJsaWMgSU5URVJGQUNFLCBwdWJsaWMgQkJpbmRlcgotewotcHVibGljOgotICAgIHZpcnR1YWwgc3A8SUludGVyZmFjZT4gICAgICBxdWVyeUxvY2FsSW50ZXJmYWNlKGNvbnN0IFN0cmluZzE2JiBfZGVzY3JpcHRvcik7Ci0gICAgdmlydHVhbCBTdHJpbmcxNiAgICAgICAgICAgIGdldEludGVyZmFjZURlc2NyaXB0b3IoKSBjb25zdDsKLQotcHJvdGVjdGVkOgotICAgIHZpcnR1YWwgSUJpbmRlciogICAgICAgICAgICBvbkFzQmluZGVyKCk7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXRlbXBsYXRlPHR5cGVuYW1lIElOVEVSRkFDRT4KLWNsYXNzIEJwSW50ZXJmYWNlIDogcHVibGljIElOVEVSRkFDRSwgcHVibGljIEJwUmVmQmFzZQotewotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBCcEludGVyZmFjZShjb25zdCBzcDxJQmluZGVyPiYgcmVtb3RlKTsKLQotcHJvdGVjdGVkOgotICAgIHZpcnR1YWwgSUJpbmRlciogICAgICAgICAgICBvbkFzQmluZGVyKCk7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNkZWZpbmUgREVDTEFSRV9NRVRBX0lOVEVSRkFDRShJTlRFUkZBQ0UpICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLSAgICBzdGF0aWMgY29uc3QgU3RyaW5nMTYgZGVzY3JpcHRvcjsgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLSAgICBzdGF0aWMgc3A8SSMjSU5URVJGQUNFPiBhc0ludGVyZmFjZShjb25zdCBzcDxJQmluZGVyPiYgb2JqKTsgICAgICAgIFwKLSAgICB2aXJ0dWFsIFN0cmluZzE2IGdldEludGVyZmFjZURlc2NyaXB0b3IoKSBjb25zdDsgICAgICAgICAgICAgICAgICAgIFwKLQotI2RlZmluZSBJTVBMRU1FTlRfTUVUQV9JTlRFUkZBQ0UoSU5URVJGQUNFLCBOQU1FKSAgICAgICAgICAgICAgICAgICAgICAgXAotICAgIGNvbnN0IFN0cmluZzE2IEkjI0lOVEVSRkFDRTo6ZGVzY3JpcHRvcihOQU1FKTsgICAgICAgICAgICAgICAgICAgICAgXAotICAgIFN0cmluZzE2IEkjI0lOVEVSRkFDRTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpIGNvbnN0IHsgICAgICAgICAgICAgXAotICAgICAgICByZXR1cm4gSSMjSU5URVJGQUNFOjpkZXNjcmlwdG9yOyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgIH0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgIHNwPEkjI0lOVEVSRkFDRT4gSSMjSU5URVJGQUNFOjphc0ludGVyZmFjZShjb25zdCBzcDxJQmluZGVyPiYgb2JqKSAgXAotICAgIHsgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICBzcDxJIyNJTlRFUkZBQ0U+IGludHI7ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICBpZiAob2JqICE9IE5VTEwpIHsgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgaW50ciA9IHN0YXRpY19jYXN0PEkjI0lOVEVSRkFDRSo+KCAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgICAgIG9iai0+cXVlcnlMb2NhbEludGVyZmFjZSggICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgICAgICAgICAgICAgSSMjSU5URVJGQUNFOjpkZXNjcmlwdG9yKS5nZXQoKSk7ICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgaWYgKGludHIgPT0gTlVMTCkgeyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgICAgIGludHIgPSBuZXcgQnAjI0lOVEVSRkFDRShvYmopOyAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICB9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICByZXR1cm4gaW50cjsgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgIH0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBObyB1c2VyLXNlcnZpY2FibGUgcGFydHMgYWZ0ZXIgdGhpcy4uLgotCi10ZW1wbGF0ZTx0eXBlbmFtZSBJTlRFUkZBQ0U+Ci1pbmxpbmUgc3A8SUludGVyZmFjZT4gQm5JbnRlcmZhY2U8SU5URVJGQUNFPjo6cXVlcnlMb2NhbEludGVyZmFjZSgKLSAgICAgICAgY29uc3QgU3RyaW5nMTYmIF9kZXNjcmlwdG9yKQotewotICAgIGlmIChfZGVzY3JpcHRvciA9PSBJTlRFUkZBQ0U6OmRlc2NyaXB0b3IpIHJldHVybiB0aGlzOwotICAgIHJldHVybiBOVUxMOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBJTlRFUkZBQ0U+Ci1pbmxpbmUgU3RyaW5nMTYgQm5JbnRlcmZhY2U8SU5URVJGQUNFPjo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIElOVEVSRkFDRTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBJTlRFUkZBQ0U+Ci1JQmluZGVyKiBCbkludGVyZmFjZTxJTlRFUkZBQ0U+OjpvbkFzQmluZGVyKCkKLXsKLSAgICByZXR1cm4gdGhpczsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgSU5URVJGQUNFPgotaW5saW5lIEJwSW50ZXJmYWNlPElOVEVSRkFDRT46OkJwSW50ZXJmYWNlKGNvbnN0IHNwPElCaW5kZXI+JiByZW1vdGUpCi0gICAgOiBCcFJlZkJhc2UocmVtb3RlKQotewotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBJTlRFUkZBQ0U+Ci1pbmxpbmUgSUJpbmRlciogQnBJbnRlcmZhY2U8SU5URVJGQUNFPjo6b25Bc0JpbmRlcigpCi17Ci0gICAgcmV0dXJuIHJlbW90ZSgpOwotfQotICAgIAotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9JSU5URVJGQUNFX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvSU1lbW9yeS5oIGIvaW5jbHVkZS91dGlscy9JTWVtb3J5LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDM1YTNmZDcuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9JTWVtb3J5LmgKKysrIC9kZXYvbnVsbApAQCAtMSw5NCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0lNRU1PUllfSAotI2RlZmluZSBBTkRST0lEX0lNRU1PUllfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8c3lzL21tYW4uaD4KLQotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy9JSW50ZXJmYWNlLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBJTWVtb3J5SGVhcCA6IHB1YmxpYyBJSW50ZXJmYWNlCi17Ci1wdWJsaWM6Ci0gICAgREVDTEFSRV9NRVRBX0lOVEVSRkFDRShNZW1vcnlIZWFwKTsKLQotICAgIC8vIGZsYWdzIHJldHVybmVkIGJ5IGdldEZsYWdzKCkKLSAgICBlbnVtIHsKLSAgICAgICAgUkVBRF9PTkxZICAgPSAweDAwMDAwMDAxLAotICAgICAgICBNQVBfT05DRSAgICA9IDB4MDAwMDAwMDIKLSAgICB9OwotCi0gICAgdmlydHVhbCBpbnQgICAgICAgICBnZXRIZWFwSUQoKSBjb25zdCA9IDA7Ci0gICAgdmlydHVhbCB2b2lkKiAgICAgICBnZXRCYXNlKCkgY29uc3QgPSAwOwotICAgIHZpcnR1YWwgc2l6ZV90ICAgICAgZ2V0U2l6ZSgpIGNvbnN0ID0gMDsKLSAgICB2aXJ0dWFsIHVpbnQzMl90ICAgIGdldEZsYWdzKCkgY29uc3QgPSAwOwotCi0gICAgLy8gdGhlc2UgYXJlIHRoZXJlIGp1c3QgZm9yIGJhY2t3YXJkIHNvdXJjZSBjb21wYXRpYmlsaXR5Ci0gICAgaW50MzJfdCBoZWFwSUQoKSBjb25zdCB7IHJldHVybiBnZXRIZWFwSUQoKTsgfQotICAgIHZvaWQqICAgYmFzZSgpIGNvbnN0ICB7IHJldHVybiBnZXRCYXNlKCk7IH0KLSAgICBzaXplX3QgIHZpcnR1YWxTaXplKCkgY29uc3QgeyByZXR1cm4gZ2V0U2l6ZSgpOyB9Ci19OwotCi1jbGFzcyBCbk1lbW9yeUhlYXAgOiBwdWJsaWMgQm5JbnRlcmZhY2U8SU1lbW9yeUhlYXA+Ci17Ci1wdWJsaWM6Ci0gICAgdmlydHVhbCBzdGF0dXNfdCBvblRyYW5zYWN0KCAKLSAgICAgICAgICAgIHVpbnQzMl90IGNvZGUsCi0gICAgICAgICAgICBjb25zdCBQYXJjZWwmIGRhdGEsCi0gICAgICAgICAgICBQYXJjZWwqIHJlcGx5LAotICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MgPSAwKTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgSU1lbW9yeSA6IHB1YmxpYyBJSW50ZXJmYWNlCi17Ci1wdWJsaWM6Ci0gICAgREVDTEFSRV9NRVRBX0lOVEVSRkFDRShNZW1vcnkpOwotCi0gICAgdmlydHVhbCBzcDxJTWVtb3J5SGVhcD4gZ2V0TWVtb3J5KHNzaXplX3QqIG9mZnNldD0wLCBzaXplX3QqIHNpemU9MCkgY29uc3QgPSAwOwotCi0gICAgLy8gaGVscGVycwotICAgIHZvaWQqIGZhc3RQb2ludGVyKGNvbnN0IHNwPElCaW5kZXI+JiBoZWFwLCBzc2l6ZV90IG9mZnNldCkgY29uc3Q7Ci0gICAgdm9pZCogcG9pbnRlcigpIGNvbnN0OwotICAgIHNpemVfdCBzaXplKCkgY29uc3Q7Ci0gICAgc3NpemVfdCBvZmZzZXQoKSBjb25zdDsKLX07Ci0KLWNsYXNzIEJuTWVtb3J5IDogcHVibGljIEJuSW50ZXJmYWNlPElNZW1vcnk+Ci17Ci1wdWJsaWM6Ci0gICAgdmlydHVhbCBzdGF0dXNfdCBvblRyYW5zYWN0KAotICAgICAgICAgICAgdWludDMyX3QgY29kZSwKLSAgICAgICAgICAgIGNvbnN0IFBhcmNlbCYgZGF0YSwKLSAgICAgICAgICAgIFBhcmNlbCogcmVwbHksCi0gICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDApOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9JTUVNT1JZX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvSVBDVGhyZWFkU3RhdGUuaCBiL2luY2x1ZGUvdXRpbHMvSVBDVGhyZWFkU3RhdGUuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMDQ5MGZkMy4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0lQQ1RocmVhZFN0YXRlLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxMTAgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9JUENfVEhSRUFEX1NUQVRFX0gKLSNkZWZpbmUgQU5EUk9JRF9JUENfVEhSRUFEX1NUQVRFX0gKLQotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotI2luY2x1ZGUgPHV0aWxzL1Byb2Nlc3NTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL1ZlY3Rvci5oPgotCi0jaWZkZWYgSEFWRV9XSU4zMl9QUk9DCi10eXBlZGVmICBpbnQgIHVpZF90OwotI2VuZGlmCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBJUENUaHJlYWRTdGF0ZQotewotcHVibGljOgotICAgIHN0YXRpYyAgSVBDVGhyZWFkU3RhdGUqICAgICBzZWxmKCk7Ci0gICAgCi0gICAgICAgICAgICBzcDxQcm9jZXNzU3RhdGU+ICAgIHByb2Nlc3MoKTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICBjbGVhckxhc3RFcnJvcigpOwotCi0gICAgICAgICAgICBpbnQgICAgICAgICAgICAgICAgIGdldENhbGxpbmdQaWQoKTsKLSAgICAgICAgICAgIGludCAgICAgICAgICAgICAgICAgZ2V0Q2FsbGluZ1VpZCgpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBpbnQ2NF90ICAgICAgICAgICAgIGNsZWFyQ2FsbGluZ0lkZW50aXR5KCk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHJlc3RvcmVDYWxsaW5nSWRlbnRpdHkoaW50NjRfdCB0b2tlbik7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgZmx1c2hDb21tYW5kcygpOwotCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIGpvaW5UaHJlYWRQb29sKGJvb2wgaXNNYWluID0gdHJ1ZSk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8vIFN0b3AgdGhlIGxvY2FsIHByb2Nlc3MuCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHN0b3BQcm9jZXNzKGJvb2wgaW1tZWRpYXRlID0gdHJ1ZSk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgdHJhbnNhY3QoaW50MzJfdCBoYW5kbGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGNvZGUsIGNvbnN0IFBhcmNlbCYgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUGFyY2VsKiByZXBseSwgdWludDMyX3QgZmxhZ3MpOwotCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIGluY1N0cm9uZ0hhbmRsZShpbnQzMl90IGhhbmRsZSk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIGRlY1N0cm9uZ0hhbmRsZShpbnQzMl90IGhhbmRsZSk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIGluY1dlYWtIYW5kbGUoaW50MzJfdCBoYW5kbGUpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICBkZWNXZWFrSGFuZGxlKGludDMyX3QgaGFuZGxlKTsKLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgYXR0ZW1wdEluY1N0cm9uZ0hhbmRsZShpbnQzMl90IGhhbmRsZSk7Ci0gICAgc3RhdGljICB2b2lkICAgICAgICAgICAgICAgIGV4cHVuZ2VIYW5kbGUoaW50MzJfdCBoYW5kbGUsIElCaW5kZXIqIGJpbmRlcik7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHJlcXVlc3REZWF0aE5vdGlmaWNhdGlvbiggICBpbnQzMl90IGhhbmRsZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEJwQmluZGVyKiBwcm94eSk7IAotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICBjbGVhckRlYXRoTm90aWZpY2F0aW9uKCBpbnQzMl90IGhhbmRsZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQnBCaW5kZXIqIHByb3h5KTsgCi0KLSAgICBzdGF0aWMgIHZvaWQgICAgICAgICAgICAgICAgc2h1dGRvd24oKTsKLSAgICAKLXByaXZhdGU6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIElQQ1RocmVhZFN0YXRlKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH5JUENUaHJlYWRTdGF0ZSgpOwotCi0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHNlbmRSZXBseShjb25zdCBQYXJjZWwmIHJlcGx5LCB1aW50MzJfdCBmbGFncyk7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHdhaXRGb3JSZXNwb25zZShQYXJjZWwgKnJlcGx5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3RhdHVzX3QgKmFjcXVpcmVSZXN1bHQ9TlVMTCk7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHRhbGtXaXRoRHJpdmVyKGJvb2wgZG9SZWNlaXZlPXRydWUpOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICB3cml0ZVRyYW5zYWN0aW9uRGF0YShpbnQzMl90IGNtZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgYmluZGVyRmxhZ3MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgaGFuZGxlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBjb2RlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBQYXJjZWwmIGRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXR1c190KiBzdGF0dXNCdWZmZXIpOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICBleGVjdXRlQ29tbWFuZChpbnQzMl90IGNvbW1hbmQpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIGNsZWFyQ2FsbGVyKCk7Ci0gICAgICAgICAgICAKLSAgICBzdGF0aWMgIHZvaWQgICAgICAgICAgICAgICAgdGhyZWFkRGVzdHJ1Y3Rvcih2b2lkICpzdCk7Ci0gICAgc3RhdGljICB2b2lkICAgICAgICAgICAgICAgIGZyZWVCdWZmZXIoUGFyY2VsKiBwYXJjZWwsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgdWludDhfdCogZGF0YSwgc2l6ZV90IGRhdGFTaXplLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHNpemVfdCogb2JqZWN0cywgc2l6ZV90IG9iamVjdHNTaXplLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQqIGNvb2tpZSk7Ci0gICAgCi0gICAgY29uc3QgICBzcDxQcm9jZXNzU3RhdGU+ICAgIG1Qcm9jZXNzOwotICAgICAgICAgICAgVmVjdG9yPEJCaW5kZXIqPiAgICBtUGVuZGluZ1N0cm9uZ0RlcmVmczsKLSAgICAgICAgICAgIFZlY3RvcjxSZWZCYXNlOjp3ZWFrcmVmX3R5cGUqPiBtUGVuZGluZ1dlYWtEZXJlZnM7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgUGFyY2VsICAgICAgICAgICAgICBtSW47Ci0gICAgICAgICAgICBQYXJjZWwgICAgICAgICAgICAgIG1PdXQ7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIG1MYXN0RXJyb3I7Ci0gICAgICAgICAgICBwaWRfdCAgICAgICAgICAgICAgIG1DYWxsaW5nUGlkOwotICAgICAgICAgICAgdWlkX3QgICAgICAgICAgICAgICBtQ2FsbGluZ1VpZDsKLX07Ci0gICAgCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2VuZGlmIC8vIEFORFJPSURfSVBDX1RIUkVBRF9TVEFURV9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL0lQZXJtaXNzaW9uQ29udHJvbGxlci5oIGIvaW5jbHVkZS91dGlscy9JUGVybWlzc2lvbkNvbnRyb2xsZXIuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggY2IxZGQzNC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0lQZXJtaXNzaW9uQ29udHJvbGxlci5oCisrKyAvZGV2L251bGwKQEAgLTEsNTYgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0jaWZuZGVmIEFORFJPSURfSVBFUk1JU1NJT05fQ09OVFJPTExFUl9ICi0jZGVmaW5lIEFORFJPSURfSVBFUk1JU1NJT05fQ09OVFJPTExFUl9ICi0KLSNpbmNsdWRlIDx1dGlscy9JSW50ZXJmYWNlLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBJUGVybWlzc2lvbkNvbnRyb2xsZXIgOiBwdWJsaWMgSUludGVyZmFjZQotewotcHVibGljOgotICAgIERFQ0xBUkVfTUVUQV9JTlRFUkZBQ0UoUGVybWlzc2lvbkNvbnRyb2xsZXIpOwotCi0gICAgdmlydHVhbCBib29sICAgICAgICAgICAgICAgIGNoZWNrUGVybWlzc2lvbihjb25zdCBTdHJpbmcxNiYgcGVybWlzc2lvbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgcGlkLCBpbnQzMl90IHVpZCkgPSAwOwotICAgIAotICAgIGVudW0gewotICAgICAgICBDSEVDS19QRVJNSVNTSU9OX1RSQU5TQUNUSU9OID0gSUJpbmRlcjo6RklSU1RfQ0FMTF9UUkFOU0FDVElPTgotICAgIH07Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIEJuUGVybWlzc2lvbkNvbnRyb2xsZXIgOiBwdWJsaWMgQm5JbnRlcmZhY2U8SVBlcm1pc3Npb25Db250cm9sbGVyPgotewotcHVibGljOgotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgb25UcmFuc2FjdCggdWludDMyX3QgY29kZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFBhcmNlbCYgZGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBhcmNlbCogcmVwbHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IDApOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9JUEVSTUlTU0lPTl9DT05UUk9MTEVSX0gKLQpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9JU2VydmljZU1hbmFnZXIuaCBiL2luY2x1ZGUvdXRpbHMvSVNlcnZpY2VNYW5hZ2VyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGUzZDk5ZmUuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9JU2VydmljZU1hbmFnZXIuaAorKysgL2Rldi9udWxsCkBAIC0xLDk4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotI2lmbmRlZiBBTkRST0lEX0lTRVJWSUNFX01BTkFHRVJfSAotI2RlZmluZSBBTkRST0lEX0lTRVJWSUNFX01BTkFHRVJfSAotCi0jaW5jbHVkZSA8dXRpbHMvSUludGVyZmFjZS5oPgotI2luY2x1ZGUgPHV0aWxzL0lQZXJtaXNzaW9uQ29udHJvbGxlci5oPgotI2luY2x1ZGUgPHV0aWxzL1ZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzE2Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBJU2VydmljZU1hbmFnZXIgOiBwdWJsaWMgSUludGVyZmFjZQotewotcHVibGljOgotICAgIERFQ0xBUkVfTUVUQV9JTlRFUkZBQ0UoU2VydmljZU1hbmFnZXIpOwotCi0gICAgLyoqCi0gICAgICogUmV0cmlldmUgYW4gZXhpc3Rpbmcgc2VydmljZSwgYmxvY2tpbmcgZm9yIGEgZmV3IHNlY29uZHMKLSAgICAgKiBpZiBpdCBkb2Vzbid0IHlldCBleGlzdC4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHNwPElCaW5kZXI+ICAgICAgICAgZ2V0U2VydmljZSggY29uc3QgU3RyaW5nMTYmIG5hbWUpIGNvbnN0ID0gMDsKLQotICAgIC8qKgotICAgICAqIFJldHJpZXZlIGFuIGV4aXN0aW5nIHNlcnZpY2UsIG5vbi1ibG9ja2luZy4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHNwPElCaW5kZXI+ICAgICAgICAgY2hlY2tTZXJ2aWNlKCBjb25zdCBTdHJpbmcxNiYgbmFtZSkgY29uc3QgPSAwOwotCi0gICAgLyoqCi0gICAgICogUmVnaXN0ZXIgYSBzZXJ2aWNlLgotICAgICAqLwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgICAgICBhZGRTZXJ2aWNlKCBjb25zdCBTdHJpbmcxNiYgbmFtZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3Qgc3A8SUJpbmRlcj4mIHNlcnZpY2UpID0gMDsKLQotICAgIC8qKgotICAgICAqIFJldHVybiBsaXN0IG9mIGFsbCBleGlzdGluZyBzZXJ2aWNlcy4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIFZlY3RvcjxTdHJpbmcxNj4gICAgbGlzdFNlcnZpY2VzKCkgPSAwOwotCi0gICAgZW51bSB7Ci0gICAgICAgIEdFVF9TRVJWSUNFX1RSQU5TQUNUSU9OID0gSUJpbmRlcjo6RklSU1RfQ0FMTF9UUkFOU0FDVElPTiwKLSAgICAgICAgQ0hFQ0tfU0VSVklDRV9UUkFOU0FDVElPTiwKLSAgICAgICAgQUREX1NFUlZJQ0VfVFJBTlNBQ1RJT04sCi0gICAgICAgIExJU1RfU0VSVklDRVNfVFJBTlNBQ1RJT04sCi0gICAgfTsKLX07Ci0KLXNwPElTZXJ2aWNlTWFuYWdlcj4gZGVmYXVsdFNlcnZpY2VNYW5hZ2VyKCk7Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIElOVEVSRkFDRT4KLXN0YXR1c190IGdldFNlcnZpY2UoY29uc3QgU3RyaW5nMTYmIG5hbWUsIHNwPElOVEVSRkFDRT4qIG91dFNlcnZpY2UpCi17Ci0gICAgY29uc3Qgc3A8SVNlcnZpY2VNYW5hZ2VyPiBzbSA9IGRlZmF1bHRTZXJ2aWNlTWFuYWdlcigpOwotICAgIGlmIChzbSAhPSBOVUxMKSB7Ci0gICAgICAgICpvdXRTZXJ2aWNlID0gaW50ZXJmYWNlX2Nhc3Q8SU5URVJGQUNFPihzbS0+Z2V0U2VydmljZShuYW1lKSk7Ci0gICAgICAgIGlmICgoKm91dFNlcnZpY2UpICE9IE5VTEwpIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIE5BTUVfTk9UX0ZPVU5EOwotfQotCi1ib29sIGNoZWNrQ2FsbGluZ1Blcm1pc3Npb24oY29uc3QgU3RyaW5nMTYmIHBlcm1pc3Npb24pOwotYm9vbCBjaGVja0NhbGxpbmdQZXJtaXNzaW9uKGNvbnN0IFN0cmluZzE2JiBwZXJtaXNzaW9uLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QqIG91dFBpZCwgaW50MzJfdCogb3V0VWlkKTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBCblNlcnZpY2VNYW5hZ2VyIDogcHVibGljIEJuSW50ZXJmYWNlPElTZXJ2aWNlTWFuYWdlcj4KLXsKLXB1YmxpYzoKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIG9uVHJhbnNhY3QoIHVpbnQzMl90IGNvZGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBQYXJjZWwmIGRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJjZWwqIHJlcGx5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MgPSAwKTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfSVNFUlZJQ0VfTUFOQUdFUl9ICi0KZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvS2V5ZWRWZWN0b3IuaCBiL2luY2x1ZGUvdXRpbHMvS2V5ZWRWZWN0b3IuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZjQ1MTNlZS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0tleWVkVmVjdG9yLmgKKysrIC9kZXYvbnVsbApAQCAtMSwyMDEgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9LRVlFRF9WRUNUT1JfSAotI2RlZmluZSBBTkRST0lEX0tFWUVEX1ZFQ1RPUl9ICi0KLSNpbmNsdWRlIDxhc3NlcnQuaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL1NvcnRlZFZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL1R5cGVIZWxwZXJzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLXRlbXBsYXRlIDx0eXBlbmFtZSBLRVksIHR5cGVuYW1lIFZBTFVFPgotY2xhc3MgS2V5ZWRWZWN0b3IKLXsKLXB1YmxpYzoKLSAgICB0eXBlZGVmIEtFWSAgICBrZXlfdHlwZTsKLSAgICB0eXBlZGVmIFZBTFVFICB2YWx1ZV90eXBlOwotCi0gICAgaW5saW5lICAgICAgICAgICAgICAgICAgS2V5ZWRWZWN0b3IoKTsKLQotICAgIC8qCi0gICAgICogZW1wdHkgdGhlIHZlY3RvcgotICAgICAqLwotCi0gICAgaW5saW5lICB2b2lkICAgICAgICAgICAgY2xlYXIoKSAgICAgICAgICAgICAgICAgICAgIHsgbVZlY3Rvci5jbGVhcigpOyB9Ci0KLSAgICAvKiEgCi0gICAgICogdmVjdG9yIHN0YXRzCi0gICAgICovCi0KLSAgICAvLyEgcmV0dXJucyBudW1iZXIgb2YgaXRlbXMgaW4gdGhlIHZlY3RvcgotICAgIGlubGluZSAgc2l6ZV90ICAgICAgICAgIHNpemUoKSBjb25zdCAgICAgICAgICAgICAgICB7IHJldHVybiBtVmVjdG9yLnNpemUoKTsgfQotICAgIC8vISByZXR1cm5zIHdldGhlciBvciBub3QgdGhlIHZlY3RvciBpcyBlbXB0eQotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgIGlzRW1wdHkoKSBjb25zdCAgICAgICAgICAgICB7IHJldHVybiBtVmVjdG9yLmlzRW1wdHkoKTsgfQotICAgIC8vISByZXR1cm5zIGhvdyBtYW55IGl0ZW1zIGNhbiBiZSBzdG9yZWQgd2l0aG91dCByZWFsbG9jYXRpbmcgdGhlIGJhY2tpbmcgc3RvcmUKLSAgICBpbmxpbmUgIHNpemVfdCAgICAgICAgICBjYXBhY2l0eSgpIGNvbnN0ICAgICAgICAgICAgeyByZXR1cm4gbVZlY3Rvci5jYXBhY2l0eSgpOyB9Ci0gICAgLy8hIHNldHN0IHRoZSBjYXBhY2l0eS4gY2FwYWNpdHkgY2FuIG5ldmVyIGJlIHJlZHVjZWQgbGVzcyB0aGFuIHNpemUoKQotICAgIGlubGluZSBzc2l6ZV90ICAgICAgICAgIHNldENhcGFjaXR5KHNpemVfdCBzaXplKSAgICB7IHJldHVybiBtVmVjdG9yLnNldENhcGFjaXR5KHNpemUpOyB9Ci0gICAgCi0gICAgLyohIAotICAgICAqIGFjY2Vzc29ycwotICAgICAqLwotICAgICAgICAgICAgY29uc3QgVkFMVUUmICAgIHZhbHVlRm9yKGNvbnN0IEtFWSYga2V5KSBjb25zdDsKLSAgICAgICAgICAgIGNvbnN0IFZBTFVFJiAgICB2YWx1ZUF0KHNpemVfdCBpbmRleCkgY29uc3Q7Ci0gICAgICAgICAgICBjb25zdCBLRVkmICAgICAga2V5QXQoc2l6ZV90IGluZGV4KSBjb25zdDsKLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICBpbmRleE9mS2V5KGNvbnN0IEtFWSYga2V5KSBjb25zdDsKLQotICAgIC8qIQotICAgICAqIG1vZGlmaW5nIHRoZSBhcnJheQotICAgICAqLwotCi0gICAgICAgICAgICBWQUxVRSYgICAgICAgICAgZWRpdFZhbHVlRm9yKGNvbnN0IEtFWSYga2V5KTsKLSAgICAgICAgICAgIFZBTFVFJiAgICAgICAgICBlZGl0VmFsdWVBdChzaXplX3QgaW5kZXgpOwotCi0gICAgICAgICAgICAvKiEgCi0gICAgICAgICAgICAgKiBhZGQvaW5zZXJ0L3JlcGxhY2UgaXRlbXMKLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgIAotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIGFkZChjb25zdCBLRVkmIGtleSwgY29uc3QgVkFMVUUmIGl0ZW0pOwotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIHJlcGxhY2VWYWx1ZUZvcihjb25zdCBLRVkmIGtleSwgY29uc3QgVkFMVUUmIGl0ZW0pOwotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIHJlcGxhY2VWYWx1ZUF0KHNpemVfdCBpbmRleCwgY29uc3QgVkFMVUUmIGl0ZW0pOwotCi0gICAgLyohCi0gICAgICogcmVtb3ZlIGl0ZW1zCi0gICAgICovCi0KLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICByZW1vdmVJdGVtKGNvbnN0IEtFWSYga2V5KTsKLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICByZW1vdmVJdGVtc0F0KHNpemVfdCBpbmRleCwgc2l6ZV90IGNvdW50ID0gMSk7Ci0gICAgICAgICAgICAKLXByaXZhdGU6Ci0gICAgICAgICAgICBTb3J0ZWRWZWN0b3I8IGtleV92YWx1ZV9wYWlyX3Q8S0VZLCBWQUxVRT4gPiAgICBtVmVjdG9yOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLS8qKgotICogVmFyaWF0aW9uIG9mIEtleWVkVmVjdG9yIHRoYXQgaG9sZHMgYSBkZWZhdWx0IHZhbHVlIHRvIHJldHVybiB3aGVuCi0gKiB2YWx1ZUZvcigpIGlzIGNhbGxlZCB3aXRoIGEga2V5IHRoYXQgZG9lc24ndCBleGlzdC4KLSAqLwotdGVtcGxhdGUgPHR5cGVuYW1lIEtFWSwgdHlwZW5hbWUgVkFMVUU+Ci1jbGFzcyBEZWZhdWx0S2V5ZWRWZWN0b3IgOiBwdWJsaWMgS2V5ZWRWZWN0b3I8S0VZLCBWQUxVRT4KLXsKLXB1YmxpYzoKLSAgICBpbmxpbmUgICAgICAgICAgICAgICAgICBEZWZhdWx0S2V5ZWRWZWN0b3IoY29uc3QgVkFMVUUmIGRlZlZhbHVlID0gVkFMVUUoKSk7Ci0gICAgICAgICAgICBjb25zdCBWQUxVRSYgICAgdmFsdWVGb3IoY29uc3QgS0VZJiBrZXkpIGNvbnN0OwotCi1wcml2YXRlOgotICAgICAgICAgICAgVkFMVUUgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbURlZmF1bHQ7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdGVtcGxhdGU8dHlwZW5hbWUgS0VZLCB0eXBlbmFtZSBWQUxVRT4gaW5saW5lCi1LZXllZFZlY3RvcjxLRVksVkFMVUU+OjpLZXllZFZlY3RvcigpCi17Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIEtFWSwgdHlwZW5hbWUgVkFMVUU+IGlubGluZQotc3NpemVfdCBLZXllZFZlY3RvcjxLRVksVkFMVUU+OjppbmRleE9mS2V5KGNvbnN0IEtFWSYga2V5KSBjb25zdCB7Ci0gICAgcmV0dXJuIG1WZWN0b3IuaW5kZXhPZigga2V5X3ZhbHVlX3BhaXJfdDxLRVksVkFMVUU+KGtleSkgKTsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgS0VZLCB0eXBlbmFtZSBWQUxVRT4gaW5saW5lCi1jb25zdCBWQUxVRSYgS2V5ZWRWZWN0b3I8S0VZLFZBTFVFPjo6dmFsdWVGb3IoY29uc3QgS0VZJiBrZXkpIGNvbnN0IHsKLSAgICBzc2l6ZV90IGkgPSBpbmRleE9mS2V5KGtleSk7Ci0gICAgYXNzZXJ0KGk+PTApOwotICAgIHJldHVybiBtVmVjdG9yLml0ZW1BdChpKS52YWx1ZTsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgS0VZLCB0eXBlbmFtZSBWQUxVRT4gaW5saW5lCi1jb25zdCBWQUxVRSYgS2V5ZWRWZWN0b3I8S0VZLFZBTFVFPjo6dmFsdWVBdChzaXplX3QgaW5kZXgpIGNvbnN0IHsKLSAgICByZXR1cm4gbVZlY3Rvci5pdGVtQXQoaW5kZXgpLnZhbHVlOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBLRVksIHR5cGVuYW1lIFZBTFVFPiBpbmxpbmUKLWNvbnN0IEtFWSYgS2V5ZWRWZWN0b3I8S0VZLFZBTFVFPjo6a2V5QXQoc2l6ZV90IGluZGV4KSBjb25zdCB7Ci0gICAgcmV0dXJuIG1WZWN0b3IuaXRlbUF0KGluZGV4KS5rZXk7Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIEtFWSwgdHlwZW5hbWUgVkFMVUU+IGlubGluZQotVkFMVUUmIEtleWVkVmVjdG9yPEtFWSxWQUxVRT46OmVkaXRWYWx1ZUZvcihjb25zdCBLRVkmIGtleSkgewotICAgIHNzaXplX3QgaSA9IGluZGV4T2ZLZXkoa2V5KTsKLSAgICBhc3NlcnQoaT49MCk7Ci0gICAgcmV0dXJuIG1WZWN0b3IuZWRpdEl0ZW1BdChpKS52YWx1ZTsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgS0VZLCB0eXBlbmFtZSBWQUxVRT4gaW5saW5lCi1WQUxVRSYgS2V5ZWRWZWN0b3I8S0VZLFZBTFVFPjo6ZWRpdFZhbHVlQXQoc2l6ZV90IGluZGV4KSB7Ci0gICAgcmV0dXJuIG1WZWN0b3IuZWRpdEl0ZW1BdChpbmRleCkudmFsdWU7Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIEtFWSwgdHlwZW5hbWUgVkFMVUU+IGlubGluZQotc3NpemVfdCBLZXllZFZlY3RvcjxLRVksVkFMVUU+OjphZGQoY29uc3QgS0VZJiBrZXksIGNvbnN0IFZBTFVFJiB2YWx1ZSkgewotICAgIHJldHVybiBtVmVjdG9yLmFkZCgga2V5X3ZhbHVlX3BhaXJfdDxLRVksVkFMVUU+KGtleSwgdmFsdWUpICk7Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIEtFWSwgdHlwZW5hbWUgVkFMVUU+IGlubGluZQotc3NpemVfdCBLZXllZFZlY3RvcjxLRVksVkFMVUU+OjpyZXBsYWNlVmFsdWVGb3IoY29uc3QgS0VZJiBrZXksIGNvbnN0IFZBTFVFJiB2YWx1ZSkgewotICAgIGtleV92YWx1ZV9wYWlyX3Q8S0VZLFZBTFVFPiBwYWlyKGtleSwgdmFsdWUpOwotICAgIG1WZWN0b3IucmVtb3ZlKHBhaXIpOwotICAgIHJldHVybiBtVmVjdG9yLmFkZChwYWlyKTsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgS0VZLCB0eXBlbmFtZSBWQUxVRT4gaW5saW5lCi1zc2l6ZV90IEtleWVkVmVjdG9yPEtFWSxWQUxVRT46OnJlcGxhY2VWYWx1ZUF0KHNpemVfdCBpbmRleCwgY29uc3QgVkFMVUUmIGl0ZW0pIHsKLSAgICBpZiAoaW5kZXg8c2l6ZSgpKSB7Ci0gICAgICAgIG1WZWN0b3IuZWRpdFZhbHVlQXQoaW5kZXgpLnZhbHVlID0gaXRlbTsKLSAgICAgICAgcmV0dXJuIGluZGV4OwotICAgIH0KLSAgICByZXR1cm4gQkFEX0lOREVYOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBLRVksIHR5cGVuYW1lIFZBTFVFPiBpbmxpbmUKLXNzaXplX3QgS2V5ZWRWZWN0b3I8S0VZLFZBTFVFPjo6cmVtb3ZlSXRlbShjb25zdCBLRVkmIGtleSkgewotICAgIHJldHVybiBtVmVjdG9yLnJlbW92ZShrZXlfdmFsdWVfcGFpcl90PEtFWSxWQUxVRT4oa2V5KSk7Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIEtFWSwgdHlwZW5hbWUgVkFMVUU+IGlubGluZQotc3NpemVfdCBLZXllZFZlY3RvcjxLRVksIFZBTFVFPjo6cmVtb3ZlSXRlbXNBdChzaXplX3QgaW5kZXgsIHNpemVfdCBjb3VudCkgewotICAgIHJldHVybiBtVmVjdG9yLnJlbW92ZUl0ZW1zQXQoaW5kZXgsIGNvdW50KTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXRlbXBsYXRlPHR5cGVuYW1lIEtFWSwgdHlwZW5hbWUgVkFMVUU+IGlubGluZQotRGVmYXVsdEtleWVkVmVjdG9yPEtFWSxWQUxVRT46OkRlZmF1bHRLZXllZFZlY3Rvcihjb25zdCBWQUxVRSYgZGVmVmFsdWUpCi0gICAgOiBtRGVmYXVsdChkZWZWYWx1ZSkKLXsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgS0VZLCB0eXBlbmFtZSBWQUxVRT4gaW5saW5lCi1jb25zdCBWQUxVRSYgRGVmYXVsdEtleWVkVmVjdG9yPEtFWSxWQUxVRT46OnZhbHVlRm9yKGNvbnN0IEtFWSYga2V5KSBjb25zdCB7Ci0gICAgc3NpemVfdCBpID0gaW5kZXhPZktleShrZXkpOwotICAgIHJldHVybiBpID49IDAgPyBLZXllZFZlY3RvcjxLRVksVkFMVUU+Ojp2YWx1ZUF0KGkpIDogbURlZmF1bHQ7Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLy8gQU5EUk9JRF9LRVlFRF9WRUNUT1JfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9MaXN0LmggYi9pbmNsdWRlL3V0aWxzL0xpc3QuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMWE2YmU5YS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL0xpc3QuaAorKysgL2Rldi9udWxsCkBAIC0xLDI4MCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIFRlbXBsYXRlZCBsaXN0IGNsYXNzLiAgTm9ybWFsbHkgd2UnZCB1c2UgU1RMLCBidXQgd2UgZG9uJ3QgaGF2ZSB0aGF0LgotLy8gVGhpcyBjbGFzcyBtaW1pY3MgU1RMJ3MgaW50ZXJmYWNlcy4KLS8vCi0vLyBPYmplY3RzIGFyZSBjb3BpZWQgaW50byB0aGUgbGlzdCB3aXRoIHRoZSAnPScgb3BlcmF0b3Igb3Igd2l0aCBjb3B5LQotLy8gY29uc3RydWN0aW9uLCBzbyBpZiB0aGUgY29tcGlsZXIncyBhdXRvLWdlbmVyYXRlZCB2ZXJzaW9ucyB3b24ndCB3b3JrIGZvcgotLy8geW91LCBkZWZpbmUgeW91ciBvd24uCi0vLwotLy8gVGhlIG9ubHkgY2xhc3MgeW91IHdhbnQgdG8gdXNlIGZyb20gaGVyZSBpcyAiTGlzdCIuICBEbyBub3QgdXNlIGNsYXNzZXMKLS8vIHN0YXJ0aW5nIHdpdGggIl8iIGRpcmVjdGx5LgotLy8KLSNpZm5kZWYgX0xJQlNfVVRJTFNfTElTVF9ICi0jZGVmaW5lIF9MSUJTX1VUSUxTX0xJU1RfSAotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8qCi0gKiBPbmUgZWxlbWVudCBpbiB0aGUgbGlzdC4KLSAqLwotdGVtcGxhdGU8Y2xhc3MgVD4gY2xhc3MgX0xpc3ROb2RlIHsKLXB1YmxpYzoKLSAgICB0eXBlZGVmIF9MaXN0Tm9kZTxUPiBfTm9kZTsKLQotICAgIF9MaXN0Tm9kZShjb25zdCBUJiB2YWwpIDogbVZhbCh2YWwpIHt9Ci0gICAgfl9MaXN0Tm9kZSh2b2lkKSB7fQotCi0gICAgVCYgZ2V0UmVmKHZvaWQpIHsgcmV0dXJuIG1WYWw7IH0KLSAgICB2b2lkIHNldFZhbChjb25zdCBUJiB2YWwpIHsgbVZhbCA9IHZhbDsgfQotCi0gICAgX05vZGUqIGdldFByZXYodm9pZCkgY29uc3QgeyByZXR1cm4gbXBQcmV2OyB9Ci0gICAgdm9pZCBzZXRQcmV2KF9Ob2RlKiBwdHIpIHsgbXBQcmV2ID0gcHRyOyB9Ci0gICAgX05vZGUqIGdldE5leHQodm9pZCkgY29uc3QgeyByZXR1cm4gbXBOZXh0OyB9Ci0gICAgdm9pZCBzZXROZXh0KF9Ob2RlKiBwdHIpIHsgbXBOZXh0ID0gcHRyOyB9Ci0KLXByaXZhdGU6Ci0gICAgVCAgICAgICAgICAgbVZhbDsKLSAgICBfTm9kZSogICAgICBtcFByZXY7Ci0gICAgX05vZGUqICAgICAgbXBOZXh0OwotfTsKLQotLyoKLSAqIEl0ZXJhdG9yIGZvciB3YWxraW5nIHRocm91Z2ggdGhlIGxpc3QuCi0gKi8KLXRlbXBsYXRlPGNsYXNzIFQsIGNsYXNzIFRyZWY+IGNsYXNzIF9MaXN0SXRlcmF0b3IgewotcHVibGljOgotICAgIHR5cGVkZWYgX0xpc3RJdGVyYXRvcjxULFRyZWY+IF9JdGVyOwotICAgIHR5cGVkZWYgX0xpc3ROb2RlPFQ+IF9Ob2RlOwotCi0gICAgX0xpc3RJdGVyYXRvcih2b2lkKSB7fQotICAgIF9MaXN0SXRlcmF0b3IoX05vZGUqIHB0cikgOiBtcE5vZGUocHRyKSB7fQotICAgIH5fTGlzdEl0ZXJhdG9yKHZvaWQpIHt9Ci0KLSAgICAvKgotICAgICAqIERlcmVmZXJlbmNlIG9wZXJhdG9yLiAgVXNlZCB0byBnZXQgYXQgdGhlIGp1aWN5IGluc2lkZXMuCi0gICAgICovCi0gICAgVHJlZiBvcGVyYXRvciooKSBjb25zdCB7IHJldHVybiBtcE5vZGUtPmdldFJlZigpOyB9Ci0KLSAgICAvKgotICAgICAqIEl0ZXJhdG9yIGNvbXBhcmlzb24uCi0gICAgICovCi0gICAgYm9vbCBvcGVyYXRvcj09KGNvbnN0IF9JdGVyJiByaWdodCkgY29uc3QgeyByZXR1cm4gbXBOb2RlID09IHJpZ2h0Lm1wTm9kZTsgfQotICAgIGJvb2wgb3BlcmF0b3IhPShjb25zdCBfSXRlciYgcmlnaHQpIGNvbnN0IHsgcmV0dXJuIG1wTm9kZSAhPSByaWdodC5tcE5vZGU7IH0KLQotICAgIC8qCi0gICAgICogSW5jci9kZWNyLCB1c2VkIHRvIG1vdmUgdGhyb3VnaCB0aGUgbGlzdC4KLSAgICAgKi8KLSAgICBfSXRlciYgb3BlcmF0b3IrKyh2b2lkKSB7ICAgICAgICAvLyBwcmUtaW5jcmVtZW50Ci0gICAgICAgIG1wTm9kZSA9IG1wTm9kZS0+Z2V0TmV4dCgpOwotICAgICAgICByZXR1cm4gKnRoaXM7Ci0gICAgfQotICAgIF9JdGVyIG9wZXJhdG9yKysoaW50KSB7ICAgICAgICAgIC8vIHBvc3QtaW5jcmVtZW50Ci0gICAgICAgIF9JdGVyIHRtcCA9ICp0aGlzOwotICAgICAgICArKyp0aGlzOwotICAgICAgICByZXR1cm4gdG1wOwotICAgIH0KLSAgICBfSXRlciYgb3BlcmF0b3ItLSh2b2lkKSB7ICAgICAgICAvLyBwcmUtaW5jcmVtZW50Ci0gICAgICAgIG1wTm9kZSA9IG1wTm9kZS0+Z2V0UHJldigpOwotICAgICAgICByZXR1cm4gKnRoaXM7Ci0gICAgfQotICAgIF9JdGVyIG9wZXJhdG9yLS0oaW50KSB7ICAgICAgICAgIC8vIHBvc3QtaW5jcmVtZW50Ci0gICAgICAgIF9JdGVyIHRtcCA9ICp0aGlzOwotICAgICAgICAtLSp0aGlzOwotICAgICAgICByZXR1cm4gdG1wOwotICAgIH0KLQotICAgIF9Ob2RlKiBnZXROb2RlKHZvaWQpIGNvbnN0IHsgcmV0dXJuIG1wTm9kZTsgfQotCi1wcml2YXRlOgotICAgIF9Ob2RlKiAgICAgIG1wTm9kZTsKLX07Ci0KLQotLyoKLSAqIERvdWJseS1saW5rZWQgbGlzdC4gIEluc3RhbnRpYXRlIHdpdGggIkxpc3Q8TXlDbGFzcz4gbXlMaXN0Ii4KLSAqCi0gKiBPYmplY3RzIGFkZGVkIHRvIHRoZSBsaXN0IGFyZSBjb3BpZWQgdXNpbmcgdGhlIGFzc2lnbm1lbnQgb3BlcmF0b3IsCi0gKiBzbyB0aGlzIG11c3QgYmUgZGVmaW5lZC4KLSAqLwotdGVtcGxhdGU8Y2xhc3MgVD4gY2xhc3MgTGlzdCB7Ci1wdWJsaWM6Ci0gICAgdHlwZWRlZiBfTGlzdE5vZGU8VD4gX05vZGU7Ci0KLSAgICBMaXN0KHZvaWQpIHsKLSAgICAgICAgcHJlcCgpOwotICAgIH0KLSAgICBMaXN0KGNvbnN0IExpc3Q8VD4mIHNyYykgeyAgICAgIC8vIGNvcHktY29uc3RydWN0b3IKLSAgICAgICAgcHJlcCgpOwotICAgICAgICBpbnNlcnQoYmVnaW4oKSwgc3JjLmJlZ2luKCksIHNyYy5lbmQoKSk7Ci0gICAgfQotICAgIHZpcnR1YWwgfkxpc3Qodm9pZCkgewotICAgICAgICBjbGVhcigpOwotICAgICAgICBkZWxldGVbXSAodW5zaWduZWQgY2hhciopIG1wTWlkZGxlOwotICAgIH0KLQotICAgIHR5cGVkZWYgX0xpc3RJdGVyYXRvcjxULFQmPiBpdGVyYXRvcjsKLSAgICB0eXBlZGVmIF9MaXN0SXRlcmF0b3I8VCwgY29uc3QgVCY+IGNvbnN0X2l0ZXJhdG9yOwotCi0gICAgTGlzdDxUPiYgb3BlcmF0b3I9KGNvbnN0IExpc3Q8VD4mIHJpZ2h0KTsKLQotICAgIC8qIHJldHVybnMgdHJ1ZSBpZiB0aGUgbGlzdCBpcyBlbXB0eSAqLwotICAgIGJvb2wgZW1wdHkodm9pZCkgY29uc3QgeyByZXR1cm4gbXBNaWRkbGUtPmdldE5leHQoKSA9PSBtcE1pZGRsZTsgfQotCi0gICAgLyogcmV0dXJuICNvZiBlbGVtZW50cyBpbiBsaXN0ICovCi0gICAgdW5zaWduZWQgaW50IHNpemUodm9pZCkgY29uc3QgewotICAgICAgICByZXR1cm4gZGlzdGFuY2UoYmVnaW4oKSwgZW5kKCkpOwotICAgIH0KLQotICAgIC8qCi0gICAgICogUmV0dXJuIHRoZSBmaXJzdCBlbGVtZW50IG9yIG9uZSBwYXN0IHRoZSBsYXN0IGVsZW1lbnQuICBUaGUKLSAgICAgKiBfTGlzdE5vZGUqIHdlJ3JlIHJldHVybmluZyBpcyBjb252ZXJ0ZWQgdG8gYW4gIml0ZXJhdG9yIiBieSBhCi0gICAgICogY29uc3RydWN0b3IgaW4gX0xpc3RJdGVyYXRvci4KLSAgICAgKi8KLSAgICBpdGVyYXRvciBiZWdpbigpICAgICAgICAgICAgICAgIHsgcmV0dXJuIG1wTWlkZGxlLT5nZXROZXh0KCk7IH0KLSAgICBjb25zdF9pdGVyYXRvciBiZWdpbigpIGNvbnN0ICAgIHsgcmV0dXJuIG1wTWlkZGxlLT5nZXROZXh0KCk7IH0KLSAgICBpdGVyYXRvciBlbmQoKSAgICAgICAgICAgICAgICAgIHsgcmV0dXJuIG1wTWlkZGxlOyB9Ci0gICAgY29uc3RfaXRlcmF0b3IgZW5kKCkgY29uc3QgICAgICB7IHJldHVybiBtcE1pZGRsZTsgfQotCi0gICAgLyogYWRkIHRoZSBvYmplY3QgdG8gdGhlIGhlYWQgb3IgdGFpbCBvZiB0aGUgbGlzdCAqLwotICAgIHZvaWQgcHVzaF9mcm9udChjb25zdCBUJiB2YWwpIHsgaW5zZXJ0KGJlZ2luKCksIHZhbCk7IH0KLSAgICB2b2lkIHB1c2hfYmFjayhjb25zdCBUJiB2YWwpIHsgaW5zZXJ0KGVuZCgpLCB2YWwpOyB9Ci0KLSAgICAvKiBpbnNlcnQgYmVmb3JlIHRoZSBjdXJyZW50IG5vZGU7IHJldHVybnMgaXRlcmF0b3IgYXQgbmV3IG5vZGUgKi8KLSAgICBpdGVyYXRvciBpbnNlcnQoaXRlcmF0b3IgcG9zbiwgY29uc3QgVCYgdmFsKSB7Ci0gICAgICAgIF9Ob2RlKiBuZXdOb2RlID0gbmV3IF9Ob2RlKHZhbCk7ICAgICAgICAvLyBhbGxvYyAmIGNvcHktY29uc3RydWN0Ci0gICAgICAgIG5ld05vZGUtPnNldE5leHQocG9zbi5nZXROb2RlKCkpOwotICAgICAgICBuZXdOb2RlLT5zZXRQcmV2KHBvc24uZ2V0Tm9kZSgpLT5nZXRQcmV2KCkpOwotICAgICAgICBwb3NuLmdldE5vZGUoKS0+Z2V0UHJldigpLT5zZXROZXh0KG5ld05vZGUpOwotICAgICAgICBwb3NuLmdldE5vZGUoKS0+c2V0UHJldihuZXdOb2RlKTsKLSAgICAgICAgcmV0dXJuIG5ld05vZGU7Ci0gICAgfQotCi0gICAgLyogaW5zZXJ0IGEgcmFuZ2Ugb2YgZWxlbWVudHMgYmVmb3JlIHRoZSBjdXJyZW50IG5vZGUgKi8KLSAgICB2b2lkIGluc2VydChpdGVyYXRvciBwb3NuLCBjb25zdF9pdGVyYXRvciBmaXJzdCwgY29uc3RfaXRlcmF0b3IgbGFzdCkgewotICAgICAgICBmb3IgKCA7IGZpcnN0ICE9IGxhc3Q7ICsrZmlyc3QpCi0gICAgICAgICAgICBpbnNlcnQocG9zbiwgKmZpcnN0KTsKLSAgICB9Ci0KLSAgICAvKiByZW1vdmUgb25lIGVudHJ5OyByZXR1cm5zIGl0ZXJhdG9yIGF0IG5leHQgbm9kZSAqLwotICAgIGl0ZXJhdG9yIGVyYXNlKGl0ZXJhdG9yIHBvc24pIHsKLSAgICAgICAgX05vZGUqIHBOZXh0ID0gcG9zbi5nZXROb2RlKCktPmdldE5leHQoKTsKLSAgICAgICAgX05vZGUqIHBQcmV2ID0gcG9zbi5nZXROb2RlKCktPmdldFByZXYoKTsKLSAgICAgICAgcFByZXYtPnNldE5leHQocE5leHQpOwotICAgICAgICBwTmV4dC0+c2V0UHJldihwUHJldik7Ci0gICAgICAgIGRlbGV0ZSBwb3NuLmdldE5vZGUoKTsKLSAgICAgICAgcmV0dXJuIHBOZXh0OwotICAgIH0KLQotICAgIC8qIHJlbW92ZSBhIHJhbmdlIG9mIGVsZW1lbnRzICovCi0gICAgaXRlcmF0b3IgZXJhc2UoaXRlcmF0b3IgZmlyc3QsIGl0ZXJhdG9yIGxhc3QpIHsKLSAgICAgICAgd2hpbGUgKGZpcnN0ICE9IGxhc3QpCi0gICAgICAgICAgICBlcmFzZShmaXJzdCsrKTsgICAgIC8vIGRvbid0IGVyYXNlIHRoYW4gaW5jciBsYXRlciEKLSAgICAgICAgcmV0dXJuIGxhc3Q7Ci0gICAgfQotCi0gICAgLyogcmVtb3ZlIGFsbCBjb250ZW50cyBvZiB0aGUgbGlzdCAqLwotICAgIHZvaWQgY2xlYXIodm9pZCkgewotICAgICAgICBfTm9kZSogcEN1cnJlbnQgPSBtcE1pZGRsZS0+Z2V0TmV4dCgpOwotICAgICAgICBfTm9kZSogcE5leHQ7Ci0KLSAgICAgICAgd2hpbGUgKHBDdXJyZW50ICE9IG1wTWlkZGxlKSB7Ci0gICAgICAgICAgICBwTmV4dCA9IHBDdXJyZW50LT5nZXROZXh0KCk7Ci0gICAgICAgICAgICBkZWxldGUgcEN1cnJlbnQ7Ci0gICAgICAgICAgICBwQ3VycmVudCA9IHBOZXh0OwotICAgICAgICB9Ci0gICAgICAgIG1wTWlkZGxlLT5zZXRQcmV2KG1wTWlkZGxlKTsKLSAgICAgICAgbXBNaWRkbGUtPnNldE5leHQobXBNaWRkbGUpOwotICAgIH0KLQotICAgIC8qCi0gICAgICogTWVhc3VyZSB0aGUgZGlzdGFuY2UgYmV0d2VlbiB0d28gaXRlcmF0b3JzLiAgT24gZXhpc3QsICJmaXJzdCIKLSAgICAgKiB3aWxsIGJlIGVxdWFsIHRvICJsYXN0Ii4gIFRoZSBpdGVyYXRvcnMgbXVzdCByZWZlciB0byB0aGUgc2FtZQotICAgICAqIGxpc3QuCi0gICAgICoKLSAgICAgKiAoVGhpcyBpcyBhY3R1YWxseSBhIGdlbmVyaWMgaXRlcmF0b3IgZnVuY3Rpb24uICBJdCBzaG91bGQgYmUgcGFydAotICAgICAqIG9mIHNvbWUgb3RoZXIgY2xhc3MsIHBvc3NpYmx5IGFuIGl0ZXJhdG9yIGJhc2UgY2xhc3MuICBJdCBuZWVkcyB0bwotICAgICAqIGtub3cgdGhlIGRpZmZlcmVuY2UgYmV0d2VlbiBhIGxpc3QsIHdoaWNoIGhhcyB0byBtYXJjaCB0aHJvdWdoLAotICAgICAqIGFuZCBhIHZlY3Rvciwgd2hpY2ggY2FuIGp1c3QgZG8gcG9pbnRlciBtYXRoLikKLSAgICAgKi8KLSAgICB1bnNpZ25lZCBpbnQgZGlzdGFuY2UoaXRlcmF0b3IgZmlyc3QsIGl0ZXJhdG9yIGxhc3QpIHsKLSAgICAgICAgdW5zaWduZWQgaW50IGNvdW50ID0gMDsKLSAgICAgICAgd2hpbGUgKGZpcnN0ICE9IGxhc3QpIHsKLSAgICAgICAgICAgICsrZmlyc3Q7Ci0gICAgICAgICAgICArK2NvdW50OwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBjb3VudDsKLSAgICB9Ci0gICAgdW5zaWduZWQgaW50IGRpc3RhbmNlKGNvbnN0X2l0ZXJhdG9yIGZpcnN0LCBjb25zdF9pdGVyYXRvciBsYXN0KSBjb25zdCB7Ci0gICAgICAgIHVuc2lnbmVkIGludCBjb3VudCA9IDA7Ci0gICAgICAgIHdoaWxlIChmaXJzdCAhPSBsYXN0KSB7Ci0gICAgICAgICAgICArK2ZpcnN0OwotICAgICAgICAgICAgKytjb3VudDsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gY291bnQ7Ci0gICAgfQotCi1wcml2YXRlOgotICAgIC8qCi0gICAgICogSSB3YW50IGEgX0xpc3ROb2RlIGJ1dCBkb24ndCBuZWVkIGl0IHRvIGhvbGQgdmFsaWQgZGF0YS4gIE1vcmUKLSAgICAgKiB0byB0aGUgcG9pbnQsIEkgZG9uJ3Qgd2FudCBUJ3MgY29uc3RydWN0b3IgdG8gZmlyZSwgc2luY2UgaXQKLSAgICAgKiBtaWdodCBoYXZlIHNpZGUtZWZmZWN0cyBvciByZXF1aXJlIGFyZ3VtZW50cy4gIFNvLCB3ZSBkbyB0aGlzCi0gICAgICogc2xpZ2h0bHkgdW5jb3V0aCBzdG9yYWdlIGFsbG9jLgotICAgICAqLwotICAgIHZvaWQgcHJlcCh2b2lkKSB7Ci0gICAgICAgIG1wTWlkZGxlID0gKF9Ob2RlKikgbmV3IHVuc2lnbmVkIGNoYXJbc2l6ZW9mKF9Ob2RlKV07Ci0gICAgICAgIG1wTWlkZGxlLT5zZXRQcmV2KG1wTWlkZGxlKTsKLSAgICAgICAgbXBNaWRkbGUtPnNldE5leHQobXBNaWRkbGUpOwotICAgIH0KLQotICAgIC8qCi0gICAgICogVGhpcyBub2RlIHBsYXlzIHRoZSByb2xlIG9mICJwb2ludGVyIHRvIGhlYWQiIGFuZCAicG9pbnRlciB0byB0YWlsIi4KLSAgICAgKiBJdCBzaXRzIGluIHRoZSBtaWRkbGUgb2YgYSBjaXJjdWxhciBsaXN0IG9mIG5vZGVzLiAgVGhlIGl0ZXJhdG9yCi0gICAgICogcnVucyBhcm91bmQgdGhlIGNpcmNsZSB1bnRpbCBpdCBlbmNvdW50ZXJzIHRoaXMgb25lLgotICAgICAqLwotICAgIF9Ob2RlKiAgICAgIG1wTWlkZGxlOwotfTsKLQotLyoKLSAqIEFzc2lnbm1lbnQgb3BlcmF0b3IuCi0gKgotICogVGhlIHNpbXBsZXN0IHdheSB0byBkbyB0aGlzIHdvdWxkIGJlIHRvIGNsZWFyIG91dCB0aGUgdGFyZ2V0IGxpc3QgYW5kCi0gKiBmaWxsIGl0IHdpdGggdGhlIHNvdXJjZS4gIEhvd2V2ZXIsIHdlIGNhbiBzcGVlZCB0aGluZ3MgYWxvbmcgYnkKLSAqIHJlLXVzaW5nIGV4aXN0aW5nIGVsZW1lbnRzLgotICovCi10ZW1wbGF0ZTxjbGFzcyBUPgotTGlzdDxUPiYgTGlzdDxUPjo6b3BlcmF0b3I9KGNvbnN0IExpc3Q8VD4mIHJpZ2h0KQotewotICAgIGlmICh0aGlzID09ICZyaWdodCkKLSAgICAgICAgcmV0dXJuICp0aGlzOyAgICAgICAvLyBzZWxmLWFzc2lnbm1lbnQKLSAgICBpdGVyYXRvciBmaXJzdERzdCA9IGJlZ2luKCk7Ci0gICAgaXRlcmF0b3IgbGFzdERzdCA9IGVuZCgpOwotICAgIGNvbnN0X2l0ZXJhdG9yIGZpcnN0U3JjID0gcmlnaHQuYmVnaW4oKTsKLSAgICBjb25zdF9pdGVyYXRvciBsYXN0U3JjID0gcmlnaHQuZW5kKCk7Ci0gICAgd2hpbGUgKGZpcnN0U3JjICE9IGxhc3RTcmMgJiYgZmlyc3REc3QgIT0gbGFzdERzdCkKLSAgICAgICAgKmZpcnN0RHN0KysgPSAqZmlyc3RTcmMrKzsKLSAgICBpZiAoZmlyc3RTcmMgPT0gbGFzdFNyYykgICAgICAgIC8vIHJhbiBvdXQgb2YgZWxlbWVudHMgaW4gc291cmNlPwotICAgICAgICBlcmFzZShmaXJzdERzdCwgbGFzdERzdCk7ICAgLy8geWVzLCBlcmFzZSBhbnkgZXh0cmFzCi0gICAgZWxzZQotICAgICAgICBpbnNlcnQobGFzdERzdCwgZmlyc3RTcmMsIGxhc3RTcmMpOyAgICAgLy8gY29weSByZW1haW5pbmcgb3ZlcgotICAgIHJldHVybiAqdGhpczsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIF9MSUJTX1VUSUxTX0xJU1RfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9Mb2cuaCBiL2luY2x1ZGUvdXRpbHMvTG9nLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDNjNmNjOGIuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9Mb2cuaAorKysgL2Rldi9udWxsCkBAIC0xLDMzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gQy9DKysgbG9nZ2luZyBmdW5jdGlvbnMuICBTZWUgdGhlIGxvZ2dpbmcgZG9jdW1lbnRhdGlvbiBmb3IgQVBJIGRldGFpbHMuCi0vLwotLy8gV2UnZCBsaWtlIHRoZXNlIHRvIGJlIGF2YWlsYWJsZSBmcm9tIEMgY29kZSAoaW4gY2FzZSB3ZSBpbXBvcnQgc29tZSBmcm9tCi0vLyBzb21ld2hlcmUpLCBzbyB0aGlzIGhhcyBhIEMgaW50ZXJmYWNlLgotLy8KLS8vIFRoZSBvdXRwdXQgd2lsbCBiZSBjb3JyZWN0IHdoZW4gdGhlIGxvZyBmaWxlIGlzIHNoYXJlZCBiZXR3ZWVuIG11bHRpcGxlCi0vLyB0aHJlYWRzIGFuZC9vciBtdWx0aXBsZSBwcm9jZXNzZXMgc28gbG9uZyBhcyB0aGUgb3BlcmF0aW5nIHN5c3RlbQotLy8gc3VwcG9ydHMgT19BUFBFTkQuICBUaGVzZSBjYWxscyBoYXZlIG11dGV4LXByb3RlY3RlZCBkYXRhIHN0cnVjdHVyZXMKLS8vIGFuZCBzbyBhcmUgTk9UIHJlZW50cmFudC4gIERvIG5vdCB1c2UgTE9HIGluIGEgc2lnbmFsIGhhbmRsZXIuCi0vLwotI2lmbmRlZiBfTElCU19VVElMU19MT0dfSAotI2RlZmluZSBfTElCU19VVElMU19MT0dfSAotCi0jaW5jbHVkZSA8Y3V0aWxzL2xvZy5oPgotCi0jZW5kaWYgLy8gX0xJQlNfVVRJTFNfTE9HX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvTG9nU29ja2V0LmggYi9pbmNsdWRlL3V0aWxzL0xvZ1NvY2tldC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwMWZiZmI1Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvTG9nU29ja2V0LmgKKysrIC9kZXYvbnVsbApAQCAtMSwyMCArMCwwIEBACi0vKiB1dGlscy9Mb2dTb2NrZXQuaAotKiogCi0qKiBDb3B5cmlnaHQgMjAwOCwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIFRoaXMgZmlsZSBpcyBkdWFsIGxpY2Vuc2VkLiAgSXQgbWF5IGJlIHJlZGlzdHJpYnV0ZWQgYW5kL29yIG1vZGlmaWVkCi0qKiB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEFwYWNoZSAyLjAgTGljZW5zZSBPUiB2ZXJzaW9uIDIgb2YgdGhlIEdOVQotKiogR2VuZXJhbCBQdWJsaWMgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgX1VUSUxTX0xPR1NPQ0tFVF9ICi0jZGVmaW5lIF9VVElMU19MT0dTT0NLRVRfSAotCi0jZGVmaW5lIFNPQ0tFVF9DTE9TRV9MT0NBTCAwCi0KLXZvaWQgYWRkX3NlbmRfc3RhdHMoaW50IGZkLCBpbnQgc2VuZCk7Ci12b2lkIGFkZF9yZWN2X3N0YXRzKGludCBmZCwgaW50IHJlY3YpOwotdm9pZCBsb2dfc29ja2V0X2Nsb3NlKGludCBmZCwgc2hvcnQgcmVhc29uKTsKLXZvaWQgbG9nX3NvY2tldF9jb25uZWN0KGludCBmZCwgdW5zaWduZWQgaW50IGlwLCB1bnNpZ25lZCBzaG9ydCBwb3J0KTsKLQotI2VuZGlmIC8qIF9VVElMU19MT0dTT0NLRVRfSCAqLwpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9NZW1vcnlCYXNlLmggYi9pbmNsdWRlL3V0aWxzL01lbW9yeUJhc2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZWI1YTlkMi4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL01lbW9yeUJhc2UuaAorKysgL2Rldi9udWxsCkBAIC0xLDUxICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfTUVNT1JZX0JBU0VfSAotI2RlZmluZSBBTkRST0lEX01FTU9SWV9CQVNFX0gKLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGludC5oPgotCi0jaW5jbHVkZSA8dXRpbHMvSU1lbW9yeS5oPgotCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIE1lbW9yeUJhc2UgOiBwdWJsaWMgQm5NZW1vcnkgCi17Ci1wdWJsaWM6Ci0gICAgTWVtb3J5QmFzZShjb25zdCBzcDxJTWVtb3J5SGVhcD4mIGhlYXAsIHNzaXplX3Qgb2Zmc2V0LCBzaXplX3Qgc2l6ZSk7Ci0gICAgdmlydHVhbCB+TWVtb3J5QmFzZSgpOwotICAgIHZpcnR1YWwgc3A8SU1lbW9yeUhlYXA+IGdldE1lbW9yeShzc2l6ZV90KiBvZmZzZXQsIHNpemVfdCogc2l6ZSkgY29uc3Q7Ci0KLXByb3RlY3RlZDoKLSAgICBzaXplX3QgZ2V0U2l6ZSgpIGNvbnN0IHsgcmV0dXJuIG1TaXplOyB9Ci0gICAgc3NpemVfdCBnZXRPZmZzZXQoKSBjb25zdCB7IHJldHVybiBtT2Zmc2V0OyB9Ci0gICAgY29uc3Qgc3A8SU1lbW9yeUhlYXA+JiBnZXRIZWFwKCkgY29uc3QgeyByZXR1cm4gbUhlYXA7IH0KLQotcHJpdmF0ZToKLSAgICBzaXplX3QgICAgICAgICAgbVNpemU7Ci0gICAgc3NpemVfdCAgICAgICAgIG1PZmZzZXQ7Ci0gICAgc3A8SU1lbW9yeUhlYXA+IG1IZWFwOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9NRU1PUllfQkFTRV9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL01lbW9yeURlYWxlci5oIGIvaW5jbHVkZS91dGlscy9NZW1vcnlEZWFsZXIuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDU0YjYyNy4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL01lbW9yeURlYWxlci5oCisrKyAvZGV2L251bGwKQEAgLTEsMjM4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfTUVNT1JZX0RFQUxFUl9ICi0jZGVmaW5lIEFORFJPSURfTUVNT1JZX0RFQUxFUl9ICi0KLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvSU1lbW9yeS5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLSNpbmNsdWRlIDx1dGlscy9NZW1vcnlIZWFwQmFzZS5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi1jbGFzcyBTdHJpbmc4OwotCi0vKgotICogaW50ZXJmYWNlIGZvciBpbXBsZW1lbnRpbmcgYSAiaGVhcCIuIEEgaGVhcCBiYXNpY2FsbHkgcHJvdmlkZXMKLSAqIHRoZSBJTWVtb3J5SGVhcCBpbnRlcmZhY2UgZm9yIGNyb3NzLXByb2Nlc3Mgc2hhcmluZyBhbmQgdGhlCi0gKiBhYmlsaXR5IHRvIG1hcC91bm1hcCBwYWdlcyB3aXRoaW4gdGhlIGhlYXAuCi0gKi8KLWNsYXNzIEhlYXBJbnRlcmZhY2UgOiBwdWJsaWMgdmlydHVhbCBCbk1lbW9yeUhlYXAKLXsKLXB1YmxpYzoKLSAgICAvLyBhbGwgdmFsdWVzIG11c3QgYmUgcGFnZS1hbGlnbmVkCi0gICAgdmlydHVhbCBzcDxJTWVtb3J5PiBtYXBNZW1vcnkoc2l6ZV90IG9mZnNldCwgc2l6ZV90IHNpemUpID0gMDsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLyoKLSAqIGludGVyZmFjZSBmb3IgaW1wbGVtZW50aW5nIGFuIGFsbG9jYXRvci4gQW4gYWxsb2NhdG9yIHByb3ZpZGVzCi0gKiBtZXRob2RzIGZvciBhbGxvY2F0aW5nIGFuZCBmcmVlaW5nIG1lbW9yeSBibG9ja3MgYW5kIGR1bXBpbmcKLSAqIGl0cyBzdGF0ZS4KLSAqLwotY2xhc3MgQWxsb2NhdG9ySW50ZXJmYWNlIDogcHVibGljIFJlZkJhc2UKLXsKLXB1YmxpYzoKLSAgICBlbnVtIHsKLSAgICAgICAgUEFHRV9BTElHTkVEID0gMHgwMDAwMDAwMQotICAgIH07Ci0KLSAgICB2aXJ0dWFsIHNpemVfdCAgICAgIGFsbG9jYXRlKHNpemVfdCBzaXplLCB1aW50MzJfdCBmbGFncyA9IDApID0gMDsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIGRlYWxsb2NhdGUoc2l6ZV90IG9mZnNldCkgPSAwOwotICAgIHZpcnR1YWwgc2l6ZV90ICAgICAgc2l6ZSgpIGNvbnN0ID0gMDsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIGR1bXAoY29uc3QgY2hhciogd2hhdCwgdWludDMyX3QgZmxhZ3MgPSAwKSBjb25zdCA9IDA7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICBkdW1wKFN0cmluZzgmIHJlcywKLSAgICAgICAgICAgIGNvbnN0IGNoYXIqIHdoYXQsIHVpbnQzMl90IGZsYWdzID0gMCkgY29uc3QgPSAwOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0vKgotICogY29uY3JldGUgaW1wbGVtZW50YXRpb24gb2YgSGVhcEludGVyZmFjZSBvbiB0b3Agb2YgbW1hcCgpIAotICovCi1jbGFzcyBTaGFyZWRIZWFwIDogcHVibGljIEhlYXBJbnRlcmZhY2UsIHB1YmxpYyBNZW1vcnlIZWFwQmFzZQotewotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgU2hhcmVkSGVhcChzaXplX3Qgc2l6ZSwgdWludDMyX3QgZmxhZ3MgPSAwLCBjaGFyIGNvbnN0ICogbmFtZSA9IE5VTEwpOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgflNoYXJlZEhlYXAoKTsKLSAgICB2aXJ0dWFsIHNwPElNZW1vcnk+IG1hcE1lbW9yeShzaXplX3Qgb2Zmc2V0LCBzaXplX3Qgc2l6ZSk7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLS8qCi0gKiBBIHNpbXBsZSB0ZW1wbGF0aXplZCBkb3VibHkgbGlua2VkLWxpc3QgaW1wbGVtZW50YXRpb24KLSAqLwotCi10ZW1wbGF0ZSA8dHlwZW5hbWUgTk9ERT4KLWNsYXNzIExpbmtlZExpc3QKLXsKLSAgICBOT0RFKiAgbUZpcnN0OwotICAgIE5PREUqICBtTGFzdDsKLQotcHVibGljOgotICAgICAgICAgICAgICAgIExpbmtlZExpc3QoKSA6IG1GaXJzdCgwKSwgbUxhc3QoMCkgeyB9Ci0gICAgYm9vbCAgICAgICAgaXNFbXB0eSgpIGNvbnN0IHsgcmV0dXJuIG1GaXJzdCA9PSAwOyB9Ci0gICAgTk9ERSBjb25zdCogaGVhZCgpIGNvbnN0IHsgcmV0dXJuIG1GaXJzdDsgfQotICAgIE5PREUqICAgICAgIGhlYWQoKSB7IHJldHVybiBtRmlyc3Q7IH0KLSAgICBOT0RFIGNvbnN0KiB0YWlsKCkgY29uc3QgeyByZXR1cm4gbUxhc3Q7IH0KLSAgICBOT0RFKiAgICAgICB0YWlsKCkgeyByZXR1cm4gbUxhc3Q7IH0KLQotICAgIHZvaWQgaW5zZXJ0QWZ0ZXIoTk9ERSogbm9kZSwgTk9ERSogbmV3Tm9kZSkgewotICAgICAgICBuZXdOb2RlLT5wcmV2ID0gbm9kZTsKLSAgICAgICAgbmV3Tm9kZS0+bmV4dCA9IG5vZGUtPm5leHQ7Ci0gICAgICAgIGlmIChub2RlLT5uZXh0ID09IDApIG1MYXN0ID0gbmV3Tm9kZTsKLSAgICAgICAgZWxzZSAgICAgICAgICAgICAgICAgbm9kZS0+bmV4dC0+cHJldiA9IG5ld05vZGU7Ci0gICAgICAgIG5vZGUtPm5leHQgPSBuZXdOb2RlOwotICAgIH0KLQotICAgIHZvaWQgaW5zZXJ0QmVmb3JlKE5PREUqIG5vZGUsIE5PREUqIG5ld05vZGUpIHsKLSAgICAgICAgIG5ld05vZGUtPnByZXYgPSBub2RlLT5wcmV2OwotICAgICAgICAgbmV3Tm9kZS0+bmV4dCA9IG5vZGU7Ci0gICAgICAgICBpZiAobm9kZS0+cHJldiA9PSAwKSAgIG1GaXJzdCA9IG5ld05vZGU7Ci0gICAgICAgICBlbHNlICAgICAgICAgICAgICAgICAgIG5vZGUtPnByZXYtPm5leHQgPSBuZXdOb2RlOwotICAgICAgICAgbm9kZS0+cHJldiA9IG5ld05vZGU7Ci0gICAgfQotCi0gICAgdm9pZCBpbnNlcnRIZWFkKE5PREUqIG5ld05vZGUpIHsKLSAgICAgICAgaWYgKG1GaXJzdCA9PSAwKSB7Ci0gICAgICAgICAgICBtRmlyc3QgPSBtTGFzdCA9IG5ld05vZGU7Ci0gICAgICAgICAgICBuZXdOb2RlLT5wcmV2ID0gbmV3Tm9kZS0+bmV4dCA9IDA7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpbnNlcnRCZWZvcmUobUZpcnN0LCBuZXdOb2RlKTsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICB2b2lkIGluc2VydFRhaWwoTk9ERSogbmV3Tm9kZSkgewotICAgICAgICBpZiAobUxhc3QgPT0gMCkgaW5zZXJ0QmVnaW5uaW5nKG5ld05vZGUpOwotICAgICAgICBlbHNlICAgICAgICAgICAgaW5zZXJ0QWZ0ZXIobUxhc3QsIG5ld05vZGUpOwotICAgIH0KLQotICAgIE5PREUqIHJlbW92ZShOT0RFKiBub2RlKSB7Ci0gICAgICAgIGlmIChub2RlLT5wcmV2ID09IDApICAgIG1GaXJzdCA9IG5vZGUtPm5leHQ7Ci0gICAgICAgIGVsc2UgICAgICAgICAgICAgICAgICAgIG5vZGUtPnByZXYtPm5leHQgPSBub2RlLT5uZXh0OwotICAgICAgICBpZiAobm9kZS0+bmV4dCA9PSAwKSAgICBtTGFzdCA9IG5vZGUtPnByZXY7Ci0gICAgICAgIGVsc2UgICAgICAgICAgICAgICAgICAgIG5vZGUtPm5leHQtPnByZXYgPSBub2RlLT5wcmV2OwotICAgICAgICByZXR1cm4gbm9kZTsKLSAgICB9Ci19OwotCi0KLS8qCi0gKiBjb25jcmV0ZSBpbXBsZW1lbnRhdGlvbiBvZiBBbGxvY2F0b3JJbnRlcmZhY2UgdXNpbmcgYSBzaW1wbGUKLSAqIGJlc3QtZml0IGFsbG9jYXRpb24gc2NoZW1lCi0gKi8KLWNsYXNzIFNpbXBsZUJlc3RGaXRBbGxvY2F0b3IgOiBwdWJsaWMgQWxsb2NhdG9ySW50ZXJmYWNlCi17Ci1wdWJsaWM6Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIFNpbXBsZUJlc3RGaXRBbGxvY2F0b3Ioc2l6ZV90IHNpemUpOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgflNpbXBsZUJlc3RGaXRBbGxvY2F0b3IoKTsKLQotICAgIHZpcnR1YWwgc2l6ZV90ICAgICAgYWxsb2NhdGUoc2l6ZV90IHNpemUsIHVpbnQzMl90IGZsYWdzID0gMCk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBkZWFsbG9jYXRlKHNpemVfdCBvZmZzZXQpOwotICAgIHZpcnR1YWwgc2l6ZV90ICAgICAgc2l6ZSgpIGNvbnN0OwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgZHVtcChjb25zdCBjaGFyKiB3aGF0LCB1aW50MzJfdCBmbGFncyA9IDApIGNvbnN0OwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgZHVtcChTdHJpbmc4JiByZXMsCi0gICAgICAgICAgICBjb25zdCBjaGFyKiB3aGF0LCB1aW50MzJfdCBmbGFncyA9IDApIGNvbnN0OwotCi1wcml2YXRlOgotCi0gICAgc3RydWN0IGNodW5rX3QgewotICAgICAgICBjaHVua190KHNpemVfdCBzdGFydCwgc2l6ZV90IHNpemUpIAotICAgICAgICAgICAgOiBzdGFydChzdGFydCksIHNpemUoc2l6ZSksIGZyZWUoMSksIHByZXYoMCksIG5leHQoMCkgewotICAgICAgICB9Ci0gICAgICAgIHNpemVfdCAgICAgICAgICAgICAgc3RhcnQ7Ci0gICAgICAgIHNpemVfdCAgICAgICAgICAgICAgc2l6ZSA6IDI4OwotICAgICAgICBpbnQgICAgICAgICAgICAgICAgIGZyZWUgOiA0OwotICAgICAgICBtdXRhYmxlIGNodW5rX3QqICAgIHByZXY7Ci0gICAgICAgIG11dGFibGUgY2h1bmtfdCogICAgbmV4dDsKLSAgICB9OwotCi0gICAgc3NpemVfdCAgYWxsb2Moc2l6ZV90IHNpemUsIHVpbnQzMl90IGZsYWdzKTsKLSAgICBjaHVua190KiBkZWFsbG9jKHNpemVfdCBzdGFydCk7Ci0gICAgdm9pZCAgICAgZHVtcF9sKGNvbnN0IGNoYXIqIHdoYXQsIHVpbnQzMl90IGZsYWdzID0gMCkgY29uc3Q7Ci0gICAgdm9pZCAgICAgZHVtcF9sKFN0cmluZzgmIHJlcywgY29uc3QgY2hhciogd2hhdCwgdWludDMyX3QgZmxhZ3MgPSAwKSBjb25zdDsKLQotICAgIHN0YXRpYyBjb25zdCBpbnQgICAga01lbW9yeUFsaWduOwotICAgIG11dGFibGUgTXV0ZXggICAgICAgbUxvY2s7Ci0gICAgTGlua2VkTGlzdDxjaHVua190PiBtTGlzdDsKLSAgICBzaXplX3QgICAgICAgICAgICAgIG1IZWFwU2l6ZTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgTWVtb3J5RGVhbGVyIDogcHVibGljIFJlZkJhc2UKLXsKLXB1YmxpYzoKLQotICAgIGVudW0gewotICAgICAgICBSRUFEX09OTFkgPSBNZW1vcnlIZWFwQmFzZTo6UkVBRF9PTkxZLAotICAgICAgICBQQUdFX0FMSUdORUQgPSBBbGxvY2F0b3JJbnRlcmZhY2U6OlBBR0VfQUxJR05FRAotICAgIH07Ci0KLSAgICAvLyBjcmVhdGVzIGEgbWVtb3J5IGRlYWxlciB3aXRoIHRoZSBTaGFyZWRIZWFwIGFuZCBTaW1wbGVCZXN0Rml0QWxsb2NhdG9yCi0gICAgTWVtb3J5RGVhbGVyKHNpemVfdCBzaXplLCB1aW50MzJfdCBmbGFncyA9IDAsIGNvbnN0IGNoYXIqIG5hbWUgPSAwKTsKLQotICAgIC8vIHByb3ZpZGUgYSBjdXN0b20gaGVhcCBidXQgdXNlIHRoZSBTaW1wbGVCZXN0Rml0QWxsb2NhdG9yCi0gICAgTWVtb3J5RGVhbGVyKGNvbnN0IHNwPEhlYXBJbnRlcmZhY2U+JiBoZWFwKTsKLQotICAgIC8vIHByb3ZpZGUgYm90aCBjdXN0b20gaGVhcCBhbmQgYWxsb2NvdGFyCi0gICAgTWVtb3J5RGVhbGVyKAotICAgICAgICAgICAgY29uc3Qgc3A8SGVhcEludGVyZmFjZT4mIGhlYXAsCi0gICAgICAgICAgICBjb25zdCBzcDxBbGxvY2F0b3JJbnRlcmZhY2U+JiBhbGxvY2F0b3IpOwotCi0gICAgdmlydHVhbCB+TWVtb3J5RGVhbGVyKCk7Ci0KLSAgICB2aXJ0dWFsIHNwPElNZW1vcnk+IGFsbG9jYXRlKHNpemVfdCBzaXplLCB1aW50MzJfdCBmbGFncyA9IDApOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgZGVhbGxvY2F0ZShzaXplX3Qgb2Zmc2V0KTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIGR1bXAoY29uc3QgY2hhciogd2hhdCwgdWludDMyX3QgZmxhZ3MgPSAwKSBjb25zdDsKLQotCi0gICAgc3A8SU1lbW9yeUhlYXA+IGdldE1lbW9yeUhlYXAoKSBjb25zdCB7IHJldHVybiBoZWFwKCk7IH0KLSAgICBzcDxBbGxvY2F0b3JJbnRlcmZhY2U+IGdldEFsbG9jYXRvcigpIGNvbnN0IHsgcmV0dXJuIGFsbG9jYXRvcigpOyB9Ci0KLXByaXZhdGU6ICAgIAotICAgIGNvbnN0IHNwPEhlYXBJbnRlcmZhY2U+JiAgICAgICAgaGVhcCgpIGNvbnN0OwotICAgIGNvbnN0IHNwPEFsbG9jYXRvckludGVyZmFjZT4mICAgYWxsb2NhdG9yKCkgY29uc3Q7Ci0KLSAgICBjbGFzcyBBbGxvY2F0aW9uIDogcHVibGljIEJuTWVtb3J5IHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIEFsbG9jYXRpb24oY29uc3Qgc3A8TWVtb3J5RGVhbGVyPiYgZGVhbGVyLAotICAgICAgICAgICAgICAgIHNzaXplX3Qgb2Zmc2V0LCBzaXplX3Qgc2l6ZSwgY29uc3Qgc3A8SU1lbW9yeT4mIG1lbW9yeSk7Ci0gICAgICAgIHZpcnR1YWwgfkFsbG9jYXRpb24oKTsKLSAgICAgICAgdmlydHVhbCBzcDxJTWVtb3J5SGVhcD4gZ2V0TWVtb3J5KHNzaXplX3QqIG9mZnNldCwgc2l6ZV90KiBzaXplKSBjb25zdDsKLSAgICBwcml2YXRlOgotICAgICAgICBzcDxNZW1vcnlEZWFsZXI+ICAgICAgICBtRGVhbGVyOwotICAgICAgICBzc2l6ZV90ICAgICAgICAgICAgICAgICBtT2Zmc2V0OwotICAgICAgICBzaXplX3QgICAgICAgICAgICAgICAgICBtU2l6ZTsKLSAgICAgICAgc3A8SU1lbW9yeT4gICAgICAgICAgICAgbU1lbW9yeTsKLSAgICB9OwotCi0gICAgc3A8SGVhcEludGVyZmFjZT4gICAgICAgICAgIG1IZWFwOwotICAgIHNwPEFsbG9jYXRvckludGVyZmFjZT4gICAgICBtQWxsb2NhdG9yOwotfTsKLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9NRU1PUllfREVBTEVSX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvTWVtb3J5SGVhcEJhc2UuaCBiL2luY2x1ZGUvdXRpbHMvTWVtb3J5SGVhcEJhc2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNTc0YWNmNC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL01lbW9yeUhlYXBCYXNlLmgKKysrIC9kZXYvbnVsbApAQCAtMSw5OCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX01FTU9SWV9IRUFQX0JBU0VfSAotI2RlZmluZSBBTkRST0lEX01FTU9SWV9IRUFQX0JBU0VfSAotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9JTWVtb3J5Lmg+Ci0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgTWVtb3J5SGVhcEJhc2UgOiBwdWJsaWMgdmlydHVhbCBCbk1lbW9yeUhlYXAgCi17Ci1wdWJsaWM6Ci0gICAgZW51bSB7Ci0gICAgICAgIFJFQURfT05MWSA9IElNZW1vcnlIZWFwOjpSRUFEX09OTFksCi0gICAgICAgIE1BUF9PTkNFID0gSU1lbW9yeUhlYXA6Ok1BUF9PTkNFLAotICAgICAgICAvLyBtZW1vcnkgd29uJ3QgYmUgbWFwcGVkIGxvY2FsbHksIGJ1dCB3aWxsIGJlIG1hcHBlZCBpbiB0aGUgcmVtb3RlCi0gICAgICAgIC8vIHByb2Nlc3MuCi0gICAgICAgIERPTlRfTUFQX0xPQ0FMTFkgPSAweDAwMDAwMTAwCi0gICAgfTsKLQotICAgIC8qIAotICAgICAqIG1hcHMgdGhlIG1lbW9yeSByZWZlcmVuY2VkIGJ5IGZkLiBidXQgRE9FU04nVCB0YWtlIG93bmVyc2hpcAotICAgICAqIG9mIHRoZSBmaWxlZGVzY3JpcHRvciAoaXQgbWFrZXMgYSBjb3B5IHdpdGggZHVwKCkKLSAgICAgKi8KLSAgICBNZW1vcnlIZWFwQmFzZShpbnQgZmQsIHNpemVfdCBzaXplLCB1aW50MzJfdCBmbGFncyA9IDApOwotICAgIAotICAgIC8qCi0gICAgICogbWFwcyBtZW1vcnkgZnJvbSB0aGUgZ2l2ZW4gZGV2aWNlCi0gICAgICovCi0gICAgTWVtb3J5SGVhcEJhc2UoY29uc3QgY2hhciogZGV2aWNlLCBzaXplX3Qgc2l6ZSA9IDAsIHVpbnQzMl90IGZsYWdzID0gMCk7Ci0KLSAgICAvKgotICAgICAqIG1hcHMgbWVtb3J5IGZyb20gYXNobWVtLCB3aXRoIHRoZSBnaXZlbiBuYW1lIGZvciBkZWJ1Z2dpbmcKLSAgICAgKi8KLSAgICBNZW1vcnlIZWFwQmFzZShzaXplX3Qgc2l6ZSwgdWludDMyX3QgZmxhZ3MgPSAwLCBjaGFyIGNvbnN0KiBuYW1lID0gTlVMTCk7Ci0KLSAgICB2aXJ0dWFsIH5NZW1vcnlIZWFwQmFzZSgpOwotCi0gICAgLyogaW1wbGVtZW50IElNZW1vcnlIZWFwIGludGVyZmFjZSAqLwotICAgIHZpcnR1YWwgaW50ICAgICAgICAgZ2V0SGVhcElEKCkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkKiAgICAgICBnZXRCYXNlKCkgY29uc3Q7Ci0gICAgdmlydHVhbCBzaXplX3QgICAgICBnZXRTaXplKCkgY29uc3Q7Ci0gICAgdmlydHVhbCB1aW50MzJfdCAgICBnZXRGbGFncygpIGNvbnN0OwotCi0gICAgY29uc3QgY2hhciogICAgICAgICBnZXREZXZpY2UoKSBjb25zdDsKLSAgICAKLSAgICAvKiB0aGlzIGNsb3NlcyB0aGlzIGhlYXAgLS0gdXNlIGNhcmVmdWxseSAqLwotICAgIHZvaWQgZGlzcG9zZSgpOwotCi0gICAgLyogdGhpcyBpcyBvbmx5IG5lZWRlZCBhcyBhIHdvcmthcm91bmQsIHVzZSBvbmx5IGlmIHlvdSBrbm93Ci0gICAgICogd2hhdCB5b3UgYXJlIGRvaW5nICovCi0gICAgc3RhdHVzX3Qgc2V0RGV2aWNlKGNvbnN0IGNoYXIqIGRldmljZSkgewotICAgICAgICBpZiAobURldmljZSA9PSAwKQotICAgICAgICAgICAgbURldmljZSA9IGRldmljZTsKLSAgICAgICAgcmV0dXJuIG1EZXZpY2UgPyBOT19FUlJPUiA6IEFMUkVBRFlfRVhJU1RTOwotICAgIH0KLSAgICAKLXByb3RlY3RlZDoKLSAgICAgICAgICAgIE1lbW9yeUhlYXBCYXNlKCk7Ci0gICAgLy8gaW5pdCgpIHRha2VzIG93bmVyc2hpcCBvZiBmZAotICAgIHN0YXR1c190IGluaXQoaW50IGZkLCB2b2lkICpiYXNlLCBpbnQgc2l6ZSwKLSAgICAgICAgICAgIGludCBmbGFncyA9IDAsIGNvbnN0IGNoYXIqIGRldmljZSA9IE5VTEwpOyAgICAKLQotcHJpdmF0ZToKLSAgICBzdGF0dXNfdCBtYXBmZChpbnQgZmQsIHNpemVfdCBzaXplKTsKLQotICAgIGludCAgICAgICAgIG1GRDsKLSAgICBzaXplX3QgICAgICBtU2l6ZTsKLSAgICB2b2lkKiAgICAgICBtQmFzZTsKLSAgICB1aW50MzJfdCAgICBtRmxhZ3M7Ci0gICAgY29uc3QgY2hhciogbURldmljZTsKLSAgICBib29sICAgICAgICBtTmVlZFVubWFwOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9NRU1PUllfSEVBUF9CQVNFX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvTWVtb3J5SGVhcFBtZW0uaCBiL2luY2x1ZGUvdXRpbHMvTWVtb3J5SGVhcFBtZW0uaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjAzMzVhZC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL01lbW9yeUhlYXBQbWVtLmgKKysrIC9kZXYvbnVsbApAQCAtMSw4MCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX01FTU9SWV9IRUFQX1BNRU1fSAotI2RlZmluZSBBTkRST0lEX01FTU9SWV9IRUFQX1BNRU1fSAotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9NZW1vcnlEZWFsZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9NZW1vcnlIZWFwQmFzZS5oPgotI2luY2x1ZGUgPHV0aWxzL0lNZW1vcnkuaD4KLSNpbmNsdWRlIDx1dGlscy9Tb3J0ZWRWZWN0b3IuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBNZW1vcnlIZWFwQmFzZTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIE1lbW9yeUhlYXBQbWVtIDogcHVibGljIEhlYXBJbnRlcmZhY2UsIHB1YmxpYyBNZW1vcnlIZWFwQmFzZQotewotcHVibGljOgotICAgIGNsYXNzIE1lbW9yeVBtZW0gOiBwdWJsaWMgQm5NZW1vcnkgewotICAgIHB1YmxpYzoKLSAgICAgICAgTWVtb3J5UG1lbShjb25zdCBzcDxNZW1vcnlIZWFwUG1lbT4mIGhlYXApOwotICAgICAgICB+TWVtb3J5UG1lbSgpOwotICAgIHByb3RlY3RlZDoKLSAgICAgICAgY29uc3Qgc3A8TWVtb3J5SGVhcFBtZW0+JiAgZ2V0SGVhcCgpIGNvbnN0IHsgcmV0dXJuIG1DbGllbnRIZWFwOyB9Ci0gICAgcHJpdmF0ZToKLSAgICAgICAgZnJpZW5kIGNsYXNzIE1lbW9yeUhlYXBQbWVtOwotICAgICAgICB2aXJ0dWFsIHZvaWQgcmV2b2tlKCkgPSAwOwotICAgICAgICBzcDxNZW1vcnlIZWFwUG1lbT4gIG1DbGllbnRIZWFwOwotICAgIH07Ci0gICAgCi0gICAgTWVtb3J5SGVhcFBtZW0oY29uc3Qgc3A8TWVtb3J5SGVhcEJhc2U+JiBwbWVtSGVhcCwKLSAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IElNZW1vcnlIZWFwOjpNQVBfT05DRSk7Ci0gICAgfk1lbW9yeUhlYXBQbWVtKCk7Ci0KLSAgICAvKiBIZWFwSW50ZXJmYWNlIGFkZGl0aW9ucyAqLwotICAgIHZpcnR1YWwgc3A8SU1lbW9yeT4gbWFwTWVtb3J5KHNpemVfdCBvZmZzZXQsIHNpemVfdCBzaXplKTsKLQotICAgIC8qIG1ha2UgdGhlIHdob2xlIGhlYXAgdmlzaWJsZSAoeW91IGtub3cgd2hvIHlvdSBhcmUpICovCi0gICAgdmlydHVhbCBzdGF0dXNfdCBzbGFwKCk7Ci0gICAgCi0gICAgLyogaGlkZSAocmV2b2tlKSB0aGUgd2hvbGUgaGVhcCAodGhlIGNsaWVudCB3aWxsIHNlZSB0aGUgZ2FyYmFnZSBwYWdlKSAqLwotICAgIHZpcnR1YWwgc3RhdHVzX3QgdW5zbGFwKCk7Ci0gICAgCi0gICAgLyogcmV2b2tlIGFsbCBhbGxvY2F0aW9ucyBtYWRlIGJ5IHRoaXMgaGVhcCAqLwotICAgIHZpcnR1YWwgdm9pZCByZXZva2UoKTsKLQotcHJpdmF0ZToKLSAgICAvKiB1c2UgdGhpcyB0byBjcmVhdGUgeW91ciBvd24gSU1lbW9yeSBmb3IgbWFwTWVtb3J5ICovCi0gICAgdmlydHVhbCBzcDxNZW1vcnlQbWVtPiBjcmVhdGVNZW1vcnkoc2l6ZV90IG9mZnNldCwgc2l6ZV90IHNpemUpOwotICAgIHZvaWQgcmVtb3ZlKGNvbnN0IHdwPE1lbW9yeVBtZW0+JiBtZW1vcnkpOwotCi1wcml2YXRlOgotICAgIHNwPE1lbW9yeUhlYXBCYXNlPiAgICAgICAgICAgICAgbVBhcmVudEhlYXA7Ci0gICAgbXV0YWJsZSBNdXRleCAgICAgICAgICAgICAgICAgICBtTG9jazsKLSAgICBTb3J0ZWRWZWN0b3I8IHdwPE1lbW9yeVBtZW0+ID4gIG1BbGxvY2F0aW9uczsKLX07Ci0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9NRU1PUllfSEVBUF9QTUVNX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvUGFyY2VsLmggYi9pbmNsdWRlL3V0aWxzL1BhcmNlbC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5MDg3YzQ0Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvUGFyY2VsLmgKKysrIC9kZXYvbnVsbApAQCAtMSwyMDkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9QQVJDRUxfSAotI2RlZmluZSBBTkRST0lEX1BBUkNFTF9ICi0KLSNpbmNsdWRlIDxjdXRpbHMvbmF0aXZlX2hhbmRsZS5oPgotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmcxNi5oPgotI2luY2x1ZGUgPHV0aWxzL1ZlY3Rvci5oPgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgSUJpbmRlcjsKLWNsYXNzIFByb2Nlc3NTdGF0ZTsKLWNsYXNzIFN0cmluZzg7Ci1jbGFzcyBUZXh0T3V0cHV0OwotCi1zdHJ1Y3QgZmxhdF9iaW5kZXJfb2JqZWN0OyAgLy8gZGVmaW5lZCBpbiBzdXBwb3J0X3AvYmluZGVyX21vZHVsZS5oCi0KLWNsYXNzIFBhcmNlbAotewotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgUGFyY2VsKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB+UGFyY2VsKCk7Ci0gICAgCi0gICAgY29uc3QgdWludDhfdCogICAgICBkYXRhKCkgY29uc3Q7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICBkYXRhU2l6ZSgpIGNvbnN0OwotICAgIHNpemVfdCAgICAgICAgICAgICAgZGF0YUF2YWlsKCkgY29uc3Q7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICBkYXRhUG9zaXRpb24oKSBjb25zdDsKLSAgICBzaXplX3QgICAgICAgICAgICAgIGRhdGFDYXBhY2l0eSgpIGNvbnN0OwotICAgIAotICAgIHN0YXR1c190ICAgICAgICAgICAgc2V0RGF0YVNpemUoc2l6ZV90IHNpemUpOwotICAgIHZvaWQgICAgICAgICAgICAgICAgc2V0RGF0YVBvc2l0aW9uKHNpemVfdCBwb3MpIGNvbnN0OwotICAgIHN0YXR1c190ICAgICAgICAgICAgc2V0RGF0YUNhcGFjaXR5KHNpemVfdCBzaXplKTsKLSAgICAKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHNldERhdGEoY29uc3QgdWludDhfdCogYnVmZmVyLCBzaXplX3QgbGVuKTsKLQotICAgIHN0YXR1c190ICAgICAgICAgICAgYXBwZW5kRnJvbShQYXJjZWwgKnBhcmNlbCwgc2l6ZV90IHN0YXJ0LCBzaXplX3QgbGVuKTsKLQotICAgIGJvb2wgICAgICAgICAgICAgICAgaGFzRmlsZURlc2NyaXB0b3JzKCkgY29uc3Q7Ci0KLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHdyaXRlSW50ZXJmYWNlVG9rZW4oY29uc3QgU3RyaW5nMTYmIGludGVyZmFjZSk7Ci0gICAgYm9vbCAgICAgICAgICAgICAgICBlbmZvcmNlSW50ZXJmYWNlKGNvbnN0IFN0cmluZzE2JiBpbnRlcmZhY2UpIGNvbnN0OwotICAgICAgICAgICAgCi0gICAgdm9pZCAgICAgICAgICAgICAgICBmcmVlRGF0YSgpOwotCi0gICAgY29uc3Qgc2l6ZV90KiAgICAgICBvYmplY3RzKCkgY29uc3Q7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICBvYmplY3RzQ291bnQoKSBjb25zdDsKLSAgICAKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIGVycm9yQ2hlY2soKSBjb25zdDsKLSAgICB2b2lkICAgICAgICAgICAgICAgIHNldEVycm9yKHN0YXR1c190IGVycik7Ci0gICAgCi0gICAgc3RhdHVzX3QgICAgICAgICAgICB3cml0ZShjb25zdCB2b2lkKiBkYXRhLCBzaXplX3QgbGVuKTsKLSAgICB2b2lkKiAgICAgICAgICAgICAgIHdyaXRlSW5wbGFjZShzaXplX3QgbGVuKTsKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHdyaXRlVW5wYWRkZWQoY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IGxlbik7Ci0gICAgc3RhdHVzX3QgICAgICAgICAgICB3cml0ZUludDMyKGludDMyX3QgdmFsKTsKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHdyaXRlSW50NjQoaW50NjRfdCB2YWwpOwotICAgIHN0YXR1c190ICAgICAgICAgICAgd3JpdGVGbG9hdChmbG9hdCB2YWwpOwotICAgIHN0YXR1c190ICAgICAgICAgICAgd3JpdGVEb3VibGUoZG91YmxlIHZhbCk7Ci0gICAgc3RhdHVzX3QgICAgICAgICAgICB3cml0ZUNTdHJpbmcoY29uc3QgY2hhciogc3RyKTsKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHdyaXRlU3RyaW5nOChjb25zdCBTdHJpbmc4JiBzdHIpOwotICAgIHN0YXR1c190ICAgICAgICAgICAgd3JpdGVTdHJpbmcxNihjb25zdCBTdHJpbmcxNiYgc3RyKTsKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHdyaXRlU3RyaW5nMTYoY29uc3QgY2hhcjE2X3QqIHN0ciwgc2l6ZV90IGxlbik7Ci0gICAgc3RhdHVzX3QgICAgICAgICAgICB3cml0ZVN0cm9uZ0JpbmRlcihjb25zdCBzcDxJQmluZGVyPiYgdmFsKTsKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHdyaXRlV2Vha0JpbmRlcihjb25zdCB3cDxJQmluZGVyPiYgdmFsKTsKLQotICAgIC8vIGRvZXNuJ3QgdGFrZSBvd25lcnNoaXAgb2YgdGhlIG5hdGl2ZV9oYW5kbGUKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHdyaXRlTmF0aXZlSGFuZGxlKGNvbnN0IG5hdGl2ZV9oYW5kbGUmIGhhbmRsZSk7Ci0gICAgCi0gICAgLy8gUGxhY2UgYSBmaWxlIGRlc2NyaXB0b3IgaW50byB0aGUgcGFyY2VsLiAgVGhlIGdpdmVuIGZkIG11c3QgcmVtYWluCi0gICAgLy8gdmFsaWQgZm9yIHRoZSBsaWZldGltZSBvZiB0aGUgcGFyY2VsLgotICAgIHN0YXR1c190ICAgICAgICAgICAgd3JpdGVGaWxlRGVzY3JpcHRvcihpbnQgZmQpOwotICAgIAotICAgIC8vIFBsYWNlIGEgZmlsZSBkZXNjcmlwdG9yIGludG8gdGhlIHBhcmNlbC4gIEEgZHVwIG9mIHRoZSBmZCBpcyBtYWRlLCB3aGljaAotICAgIC8vIHdpbGwgYmUgY2xvc2VkIG9uY2UgdGhlIHBhcmNlbCBpcyBkZXN0cm95ZWQuCi0gICAgc3RhdHVzX3QgICAgICAgICAgICB3cml0ZUR1cEZpbGVEZXNjcmlwdG9yKGludCBmZCk7Ci0gICAgCi0gICAgc3RhdHVzX3QgICAgICAgICAgICB3cml0ZU9iamVjdChjb25zdCBmbGF0X2JpbmRlcl9vYmplY3QmIHZhbCwgYm9vbCBudWxsTWV0YURhdGEpOwotCi0gICAgdm9pZCAgICAgICAgICAgICAgICByZW1vdmUoc2l6ZV90IHN0YXJ0LCBzaXplX3QgYW10KTsKLSAgICAKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHJlYWQodm9pZCogb3V0RGF0YSwgc2l6ZV90IGxlbikgY29uc3Q7Ci0gICAgY29uc3Qgdm9pZCogICAgICAgICByZWFkSW5wbGFjZShzaXplX3QgbGVuKSBjb25zdDsKLSAgICBpbnQzMl90ICAgICAgICAgICAgIHJlYWRJbnQzMigpIGNvbnN0OwotICAgIHN0YXR1c190ICAgICAgICAgICAgcmVhZEludDMyKGludDMyX3QgKnBBcmcpIGNvbnN0OwotICAgIGludDY0X3QgICAgICAgICAgICAgcmVhZEludDY0KCkgY29uc3Q7Ci0gICAgc3RhdHVzX3QgICAgICAgICAgICByZWFkSW50NjQoaW50NjRfdCAqcEFyZykgY29uc3Q7Ci0gICAgZmxvYXQgICAgICAgICAgICAgICByZWFkRmxvYXQoKSBjb25zdDsKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHJlYWRGbG9hdChmbG9hdCAqcEFyZykgY29uc3Q7Ci0gICAgZG91YmxlICAgICAgICAgICAgICByZWFkRG91YmxlKCkgY29uc3Q7Ci0gICAgc3RhdHVzX3QgICAgICAgICAgICByZWFkRG91YmxlKGRvdWJsZSAqcEFyZykgY29uc3Q7Ci0KLSAgICBjb25zdCBjaGFyKiAgICAgICAgIHJlYWRDU3RyaW5nKCkgY29uc3Q7Ci0gICAgU3RyaW5nOCAgICAgICAgICAgICByZWFkU3RyaW5nOCgpIGNvbnN0OwotICAgIFN0cmluZzE2ICAgICAgICAgICAgcmVhZFN0cmluZzE2KCkgY29uc3Q7Ci0gICAgY29uc3QgY2hhcjE2X3QqICAgICByZWFkU3RyaW5nMTZJbnBsYWNlKHNpemVfdCogb3V0TGVuKSBjb25zdDsKLSAgICBzcDxJQmluZGVyPiAgICAgICAgIHJlYWRTdHJvbmdCaW5kZXIoKSBjb25zdDsKLSAgICB3cDxJQmluZGVyPiAgICAgICAgIHJlYWRXZWFrQmluZGVyKCkgY29uc3Q7Ci0KLSAgICAKLSAgICAvLyBpZiBhbGxvYyBpcyBOVUxMLCBuYXRpdmVfaGFuZGxlIGlzIGFsbG9jYXRlZCB3aXRoIG1hbGxvYygpLCBvdGhlcndpc2UKLSAgICAvLyBhbGxvYyBpcyB1c2VkLiBJZiB0aGUgZnVuY3Rpb24gZmFpbHMsIHRoZSBlZmZlY3RzIG9mIGFsbG9jKCkgbXVzdCBiZQotICAgIC8vIHJldmVydGVkIGJ5IHRoZSBjYWxsZXIuCi0gICAgbmF0aXZlX2hhbmRsZSogICAgIHJlYWROYXRpdmVIYW5kbGUoCi0gICAgICAgICAgICBuYXRpdmVfaGFuZGxlKiAoKmFsbG9jKSh2b2lkKiBjb29raWUsIGludCBudW1GZHMsIGludCBpbnRzKSwKLSAgICAgICAgICAgIHZvaWQqIGNvb2tpZSkgY29uc3Q7Ci0KLSAgICAKLSAgICAvLyBSZXRyaWV2ZSBhIGZpbGUgZGVzY3JpcHRvciBmcm9tIHRoZSBwYXJjZWwuICBUaGlzIHJldHVybnMgdGhlIHJhdyBmZAotICAgIC8vIGluIHRoZSBwYXJjZWwsIHdoaWNoIHlvdSBkbyBub3Qgb3duIC0tIHVzZSBkdXAoKSB0byBnZXQgeW91ciBvd24gY29weS4KLSAgICBpbnQgICAgICAgICAgICAgICAgIHJlYWRGaWxlRGVzY3JpcHRvcigpIGNvbnN0OwotICAgIAotICAgIGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCogcmVhZE9iamVjdChib29sIG51bGxNZXRhRGF0YSkgY29uc3Q7Ci0KLSAgICAvLyBFeHBsaWNpdGx5IGNsb3NlIGFsbCBmaWxlIGRlc2NyaXB0b3JzIGluIHRoZSBwYXJjZWwuCi0gICAgdm9pZCAgICAgICAgICAgICAgICBjbG9zZUZpbGVEZXNjcmlwdG9ycygpOwotICAgIAotICAgIHR5cGVkZWYgdm9pZCAgICAgICAgKCpyZWxlYXNlX2Z1bmMpKFBhcmNlbCogcGFyY2VsLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHVpbnQ4X3QqIGRhdGEsIHNpemVfdCBkYXRhU2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBzaXplX3QqIG9iamVjdHMsIHNpemVfdCBvYmplY3RzU2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiBjb29raWUpOwotICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgY29uc3QgdWludDhfdCogICAgICBpcGNEYXRhKCkgY29uc3Q7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICBpcGNEYXRhU2l6ZSgpIGNvbnN0OwotICAgIGNvbnN0IHNpemVfdCogICAgICAgaXBjT2JqZWN0cygpIGNvbnN0OwotICAgIHNpemVfdCAgICAgICAgICAgICAgaXBjT2JqZWN0c0NvdW50KCkgY29uc3Q7Ci0gICAgdm9pZCAgICAgICAgICAgICAgICBpcGNTZXREYXRhUmVmZXJlbmNlKGNvbnN0IHVpbnQ4X3QqIGRhdGEsIHNpemVfdCBkYXRhU2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3Qgc2l6ZV90KiBvYmplY3RzLCBzaXplX3Qgb2JqZWN0c0NvdW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZWxlYXNlX2Z1bmMgcmVsRnVuYywgdm9pZCogcmVsQ29va2llKTsKLSAgICAKLSAgICB2b2lkICAgICAgICAgICAgICAgIHByaW50KFRleHRPdXRwdXQmIHRvLCB1aW50MzJfdCBmbGFncyA9IDApIGNvbnN0OwotICAgIAotcHJpdmF0ZToKLSAgICAgICAgICAgICAgICAgICAgICAgIFBhcmNlbChjb25zdCBQYXJjZWwmIG8pOwotICAgIFBhcmNlbCYgICAgICAgICAgICAgb3BlcmF0b3I9KGNvbnN0IFBhcmNlbCYgbyk7Ci0gICAgCi0gICAgc3RhdHVzX3QgICAgICAgICAgICBmaW5pc2hXcml0ZShzaXplX3QgbGVuKTsKLSAgICB2b2lkICAgICAgICAgICAgICAgIHJlbGVhc2VPYmplY3RzKCk7Ci0gICAgdm9pZCAgICAgICAgICAgICAgICBhY3F1aXJlT2JqZWN0cygpOwotICAgIHN0YXR1c190ICAgICAgICAgICAgZ3Jvd0RhdGEoc2l6ZV90IGxlbik7Ci0gICAgc3RhdHVzX3QgICAgICAgICAgICByZXN0YXJ0V3JpdGUoc2l6ZV90IGRlc2lyZWQpOwotICAgIHN0YXR1c190ICAgICAgICAgICAgY29udGludWVXcml0ZShzaXplX3QgZGVzaXJlZCk7Ci0gICAgdm9pZCAgICAgICAgICAgICAgICBmcmVlRGF0YU5vSW5pdCgpOwotICAgIHZvaWQgICAgICAgICAgICAgICAgaW5pdFN0YXRlKCk7Ci0gICAgdm9pZCAgICAgICAgICAgICAgICBzY2FuRm9yRmRzKCkgY29uc3Q7Ci0gICAgICAgICAgICAgICAgICAgICAgICAKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIG1FcnJvcjsKLSAgICB1aW50OF90KiAgICAgICAgICAgIG1EYXRhOwotICAgIHNpemVfdCAgICAgICAgICAgICAgbURhdGFTaXplOwotICAgIHNpemVfdCAgICAgICAgICAgICAgbURhdGFDYXBhY2l0eTsKLSAgICBtdXRhYmxlIHNpemVfdCAgICAgIG1EYXRhUG9zOwotICAgIHNpemVfdCogICAgICAgICAgICAgbU9iamVjdHM7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICBtT2JqZWN0c1NpemU7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICBtT2JqZWN0c0NhcGFjaXR5OwotICAgIG11dGFibGUgc2l6ZV90ICAgICAgbU5leHRPYmplY3RIaW50OwotCi0gICAgbXV0YWJsZSBib29sICAgICAgICBtRmRzS25vd247Ci0gICAgbXV0YWJsZSBib29sICAgICAgICBtSGFzRmRzOwotICAgIAotICAgIHJlbGVhc2VfZnVuYyAgICAgICAgbU93bmVyOwotICAgIHZvaWQqICAgICAgICAgICAgICAgbU93bmVyQ29va2llOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWlubGluZSBUZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCBjb25zdCBQYXJjZWwmIHBhcmNlbCkKLXsKLSAgICBwYXJjZWwucHJpbnQodG8pOwotICAgIHJldHVybiB0bzsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLS8vIEdlbmVyaWMgYWNxdWlyZSBhbmQgcmVsZWFzZSBvZiBvYmplY3RzLgotdm9pZCBhY3F1aXJlX29iamVjdChjb25zdCBzcDxQcm9jZXNzU3RhdGU+JiBwcm9jLAotICAgICAgICAgICAgICAgICAgICBjb25zdCBmbGF0X2JpbmRlcl9vYmplY3QmIG9iaiwgY29uc3Qgdm9pZCogd2hvKTsKLXZvaWQgcmVsZWFzZV9vYmplY3QoY29uc3Qgc3A8UHJvY2Vzc1N0YXRlPiYgcHJvYywKLSAgICAgICAgICAgICAgICAgICAgY29uc3QgZmxhdF9iaW5kZXJfb2JqZWN0JiBvYmosIGNvbnN0IHZvaWQqIHdobyk7Ci0KLXZvaWQgZmxhdHRlbl9iaW5kZXIoY29uc3Qgc3A8UHJvY2Vzc1N0YXRlPiYgcHJvYywKLSAgICAgICAgICAgICAgICAgICAgY29uc3Qgc3A8SUJpbmRlcj4mIGJpbmRlciwgZmxhdF9iaW5kZXJfb2JqZWN0KiBvdXQpOwotdm9pZCBmbGF0dGVuX2JpbmRlcihjb25zdCBzcDxQcm9jZXNzU3RhdGU+JiBwcm9jLAotICAgICAgICAgICAgICAgICAgICBjb25zdCB3cDxJQmluZGVyPiYgYmluZGVyLCBmbGF0X2JpbmRlcl9vYmplY3QqIG91dCk7Ci1zdGF0dXNfdCB1bmZsYXR0ZW5fYmluZGVyKGNvbnN0IHNwPFByb2Nlc3NTdGF0ZT4mIHByb2MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCYgZmxhdCwgc3A8SUJpbmRlcj4qIG91dCk7Ci1zdGF0dXNfdCB1bmZsYXR0ZW5fYmluZGVyKGNvbnN0IHNwPFByb2Nlc3NTdGF0ZT4mIHByb2MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCYgZmxhdCwgd3A8SUJpbmRlcj4qIG91dCk7Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLy8gQU5EUk9JRF9QQVJDRUxfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9QaXBlLmggYi9pbmNsdWRlL3V0aWxzL1BpcGUuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjQwNDE2OC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL1BpcGUuaAorKysgL2Rldi9udWxsCkBAIC0xLDEwOCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIEZJRk8gSS9PLgotLy8KLSNpZm5kZWYgX0xJQlNfVVRJTFNfUElQRV9ICi0jZGVmaW5lIF9MSUJTX1VUSUxTX1BJUEVfSAotCi0jaWZkZWYgSEFWRV9BTkRST0lEX09TCi0jZXJyb3IgRE8gTk9UIFVTRSBUSElTIEZJTEUgSU4gVEhFIERFVklDRSBCVUlMRAotI2VuZGlmCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyoKLSAqIFNpbXBsZSBhbm9ueW1vdXMgdW5pZGlyZWN0aW9uYWwgcGlwZS4KLSAqCi0gKiBUaGUgcHJpbWFyeSBnb2FsIGlzIHRvIGNyZWF0ZSBhbiBpbXBsZW1lbnRhdGlvbiB3aXRoIG1pbmltYWwgb3ZlcmhlYWQKLSAqIHVuZGVyIExpbnV4LiAgTWFraW5nIFdpbmRvd3MsIE1hYyBPUyBYLCBhbmQgTGludXggYWxsIHdvcmsgdGhlIHNhbWUgd2F5Ci0gKiBpcyBhIHNlY29uZGFyeSBnb2FsLiAgUGFydCBvZiB0aGlzIGdvYWwgaXMgdG8gaGF2ZSBzb21ldGhpbmcgdGhhdCBjYW4KLSAqIGJlIGZlZCB0byBhIHNlbGVjdCgpIGNhbGwsIHNvIHRoYXQgdGhlIGFwcGxpY2F0aW9uIGNhbiBzbGVlcCBpbiB0aGUKLSAqIGtlcm5lbCB1bnRpbCBzb21ldGhpbmcgaW50ZXJlc3RpbmcgaGFwcGVucy4KLSAqLwotY2xhc3MgUGlwZSB7Ci1wdWJsaWM6Ci0gICAgUGlwZSh2b2lkKTsKLSAgICB2aXJ0dWFsIH5QaXBlKHZvaWQpOwotCi0gICAgLyogQ3JlYXRlIHRoZSBwaXBlICovCi0gICAgYm9vbCBjcmVhdGUodm9pZCk7Ci0KLSAgICAvKiBDcmVhdGUgYSByZWFkLW9ubHkgcGlwZSwgdXNpbmcgdGhlIHN1cHBsaWVkIGhhbmRsZSBhcyByZWFkIGhhbmRsZSAqLwotICAgIGJvb2wgY3JlYXRlUmVhZGVyKHVuc2lnbmVkIGxvbmcgaGFuZGxlKTsKLSAgICAvKiBDcmVhdGUgYSB3cml0ZS1vbmx5IHBpcGUsIHVzaW5nIHRoZSBzdXBwbGllZCBoYW5kbGUgYXMgd3JpdGUgaGFuZGxlICovCi0gICAgYm9vbCBjcmVhdGVXcml0ZXIodW5zaWduZWQgbG9uZyBoYW5kbGUpOwotCi0gICAgLyogSXMgdGhpcyBvYmplY3QgcmVhZHkgdG8gZ28/ICovCi0gICAgYm9vbCBpc0NyZWF0ZWQodm9pZCk7Ci0KLSAgICAvKgotICAgICAqIFJlYWQgImNvdW50IiBieXRlcyBmcm9tIHRoZSBwaXBlLiAgUmV0dXJucyB0aGUgYW1vdW50IG9mIGRhdGEgcmVhZCwKLSAgICAgKiBvciAwIGlmIG5vIGRhdGEgYXZhaWxhYmxlIGFuZCB3ZSdyZSBub24tYmxvY2tpbmcuCi0gICAgICogUmV0dXJucyAtMSBvbiBlcnJvci4KLSAgICAgKi8KLSAgICBpbnQgcmVhZCh2b2lkKiBidWYsIGludCBjb3VudCk7Ci0KLSAgICAvKgotICAgICAqIFdyaXRlICJjb3VudCIgYnl0ZXMgaW50byB0aGUgcGlwZS4gIFJldHVybnMgbnVtYmVyIG9mIGJ5dGVzIHdyaXR0ZW4sCi0gICAgICogb3IgMCBpZiB0aGVyZSdzIG5vIHJvb20gZm9yIG1vcmUgZGF0YSBhbmQgd2UncmUgbm9uLWJsb2NraW5nLgotICAgICAqIFJldHVybnMgLTEgb24gZXJyb3IuCi0gICAgICovCi0gICAgaW50IHdyaXRlKGNvbnN0IHZvaWQqIGJ1ZiwgaW50IGNvdW50KTsKLQotICAgIC8qIFJldHVybnMgInRydWUiIGlmIGRhdGEgaXMgYXZhaWxhYmxlIHRvIHJlYWQgKi8KLSAgICBib29sIHJlYWRSZWFkeSh2b2lkKTsKLQotICAgIC8qIEVuYWJsZSBvciBkaXNhYmxlIG5vbi1ibG9ja2luZyBJL08gZm9yIHJlYWRzICovCi0gICAgYm9vbCBzZXRSZWFkTm9uQmxvY2tpbmcoYm9vbCB2YWwpOwotICAgIC8qIEVuYWJsZSBvciBkaXNhYmxlIG5vbi1ibG9ja2luZyBJL08gZm9yIHdyaXRlcy4gIE9ubHkgd29ya3Mgb24gTGludXguICovCi0gICAgYm9vbCBzZXRXcml0ZU5vbkJsb2NraW5nKGJvb2wgdmFsKTsKLQotICAgIC8qCi0gICAgICogR2V0IHRoZSBoYW5kbGUuICBPbmx5IHVzZWZ1bCBpbiBzb21lIHBsYXRmb3JtLXNwZWNpZmljIHNpdHVhdGlvbnMuCi0gICAgICovCi0gICAgdW5zaWduZWQgbG9uZyBnZXRSZWFkSGFuZGxlKHZvaWQpOwotICAgIHVuc2lnbmVkIGxvbmcgZ2V0V3JpdGVIYW5kbGUodm9pZCk7Ci0KLSAgICAvKgotICAgICAqIE1vZGlmeSBpbmhlcml0YW5jZSwgaS5lLiB3aGV0aGVyIG9yIG5vdCBhIGNoaWxkIHByb2Nlc3Mgd2lsbCBnZXQKLSAgICAgKiBjb3BpZXMgb2YgdGhlIGRlc2NyaXB0b3JzLiAgU3lzdGVtcyB3aXRoIGZvcmsrZXhlYyBhbGxvdyB1cyB0byBjbG9zZQotICAgICAqIHRoZSBkZXNjcmlwdG9ycyBiZWZvcmUgbGF1bmNoaW5nIHRoZSBjaGlsZCBwcm9jZXNzLCBidXQgV2luMzIKLSAgICAgKiBkb2Vzbid0IGFsbG93IGl0LgotICAgICAqLwotICAgIGJvb2wgZGlzYWxsb3dSZWFkSW5oZXJpdCh2b2lkKTsKLSAgICBib29sIGRpc2FsbG93V3JpdGVJbmhlcml0KHZvaWQpOwotCi0gICAgLyoKLSAgICAgKiBDbG9zZSBvbmUgc2lkZSBvciB0aGUgb3RoZXIuICBVc2VmdWwgaW4gdGhlIHBhcmVudCBhZnRlciBsYXVuY2hpbmcKLSAgICAgKiBhIGNoaWxkIHByb2Nlc3MuCi0gICAgICovCi0gICAgYm9vbCBjbG9zZVJlYWQodm9pZCk7Ci0gICAgYm9vbCBjbG9zZVdyaXRlKHZvaWQpOwotCi1wcml2YXRlOgotICAgIGJvb2wgICAgbVJlYWROb25CbG9ja2luZzsKLSAgICBib29sICAgIG1Xcml0ZU5vbkJsb2NraW5nOwotCi0gICAgdW5zaWduZWQgbG9uZyBtUmVhZEhhbmRsZTsKLSAgICB1bnNpZ25lZCBsb25nIG1Xcml0ZUhhbmRsZTsKLX07Ci0KLX07IC8vIGFuZHJvaWQKLQotI2VuZGlmIC8vIF9MSUJTX1VUSUxTX1BJUEVfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9Qcm9jZXNzU3RhdGUuaCBiL2luY2x1ZGUvdXRpbHMvUHJvY2Vzc1N0YXRlLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDM5NTg0ZjQuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9Qcm9jZXNzU3RhdGUuaAorKysgL2Rldi9udWxsCkBAIC0xLDExNyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1BST0NFU1NfU1RBVEVfSAotI2RlZmluZSBBTkRST0lEX1BST0NFU1NfU1RBVEVfSAotCi0jaW5jbHVkZSA8dXRpbHMvSUJpbmRlci5oPgotI2luY2x1ZGUgPHV0aWxzL0tleWVkVmVjdG9yLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RyaW5nOC5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzE2Lmg+Ci0KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyBHbG9iYWwgdmFyaWFibGVzCi1leHRlcm4gaW50ICAgICAgICAgICAgICAgICBtQXJnQzsKLWV4dGVybiBjb25zdCBjaGFyKiBjb25zdCogIG1BcmdWOwotZXh0ZXJuIGludCAgICAgICAgICAgICAgICAgbUFyZ0xlbjsKLQotY2xhc3MgSVBDVGhyZWFkU3RhdGU7Ci0KLWNsYXNzIFByb2Nlc3NTdGF0ZSA6IHB1YmxpYyB2aXJ0dWFsIFJlZkJhc2UKLXsKLXB1YmxpYzoKLSAgICBzdGF0aWMgIHNwPFByb2Nlc3NTdGF0ZT4gICAgc2VsZigpOwotCi0gICAgc3RhdGljICB2b2lkICAgICAgICAgICAgICAgIHNldFNpbmdsZVByb2Nlc3MoYm9vbCBzaW5nbGVQcm9jZXNzKTsKLQotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICBzZXRDb250ZXh0T2JqZWN0KGNvbnN0IHNwPElCaW5kZXI+JiBvYmplY3QpOwotICAgICAgICAgICAgc3A8SUJpbmRlcj4gICAgICAgICBnZXRDb250ZXh0T2JqZWN0KGNvbnN0IHNwPElCaW5kZXI+JiBjYWxsZXIpOwotICAgICAgICAKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgc2V0Q29udGV4dE9iamVjdChjb25zdCBzcDxJQmluZGVyPiYgb2JqZWN0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFN0cmluZzE2JiBuYW1lKTsKLSAgICAgICAgICAgIHNwPElCaW5kZXI+ICAgICAgICAgZ2V0Q29udGV4dE9iamVjdChjb25zdCBTdHJpbmcxNiYgbmFtZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBzcDxJQmluZGVyPiYgY2FsbGVyKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgIGJvb2wgICAgICAgICAgICAgICAgc3VwcG9ydHNQcm9jZXNzZXMoKSBjb25zdDsKLQotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICBzdGFydFRocmVhZFBvb2woKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIAotICAgIHR5cGVkZWYgYm9vbCAoKmNvbnRleHRfY2hlY2tfZnVuYykoY29uc3QgU3RyaW5nMTYmIG5hbWUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBzcDxJQmluZGVyPiYgY2FsbGVyLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCogdXNlckRhdGEpOwotICAgICAgICAKLSAgICAgICAgICAgIGJvb2wgICAgICAgICAgICAgICAgaXNDb250ZXh0TWFuYWdlcih2b2lkKSBjb25zdDsKLSAgICAgICAgICAgIGJvb2wgICAgICAgICAgICAgICAgYmVjb21lQ29udGV4dE1hbmFnZXIoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb250ZXh0X2NoZWNrX2Z1bmMgY2hlY2tGdW5jLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCogdXNlckRhdGEpOwotCi0gICAgICAgICAgICBzcDxJQmluZGVyPiAgICAgICAgIGdldFN0cm9uZ1Byb3h5Rm9ySGFuZGxlKGludDMyX3QgaGFuZGxlKTsKLSAgICAgICAgICAgIHdwPElCaW5kZXI+ICAgICAgICAgZ2V0V2Vha1Byb3h5Rm9ySGFuZGxlKGludDMyX3QgaGFuZGxlKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgZXhwdW5nZUhhbmRsZShpbnQzMl90IGhhbmRsZSwgSUJpbmRlciogYmluZGVyKTsKLQotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICBzZXRBcmdzKGludCBhcmdjLCBjb25zdCBjaGFyKiBjb25zdCBhcmd2W10pOwotICAgICAgICAgICAgaW50ICAgICAgICAgICAgICAgICBnZXRBcmdDKCkgY29uc3Q7Ci0gICAgICAgICAgICBjb25zdCBjaGFyKiBjb25zdCogIGdldEFyZ1YoKSBjb25zdDsKLQotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICBzZXRBcmdWMChjb25zdCBjaGFyKiB0eHQpOwotCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHNwYXduUG9vbGVkVGhyZWFkKGJvb2wgaXNNYWluKTsKLSAgICAgICAgICAgIAotcHJpdmF0ZToKLSAgICBmcmllbmQgY2xhc3MgSVBDVGhyZWFkU3RhdGU7Ci0gICAgCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFByb2Nlc3NTdGF0ZSgpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB+UHJvY2Vzc1N0YXRlKCk7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUHJvY2Vzc1N0YXRlKGNvbnN0IFByb2Nlc3NTdGF0ZSYgbyk7Ci0gICAgICAgICAgICBQcm9jZXNzU3RhdGUmICAgICAgIG9wZXJhdG9yPShjb25zdCBQcm9jZXNzU3RhdGUmIG8pOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBzdHJ1Y3QgaGFuZGxlX2VudHJ5IHsKLSAgICAgICAgICAgICAgICBJQmluZGVyKiBiaW5kZXI7Ci0gICAgICAgICAgICAgICAgUmVmQmFzZTo6d2Vha3JlZl90eXBlKiByZWZzOwotICAgICAgICAgICAgfTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgaGFuZGxlX2VudHJ5KiAgICAgICBsb29rdXBIYW5kbGVMb2NrZWQoaW50MzJfdCBoYW5kbGUpOwotCi0gICAgICAgICAgICBpbnQgICAgICAgICAgICAgICAgIG1Ecml2ZXJGRDsKLSAgICAgICAgICAgIHZvaWQqICAgICAgICAgICAgICAgbVZNU3RhcnQ7Ci0gICAgICAgICAgICAKLSAgICBtdXRhYmxlIE11dGV4ICAgICAgICAgICAgICAgbUxvY2s7ICAvLyBwcm90ZWN0cyBldmVyeXRoaW5nIGJlbG93LgotICAgICAgICAgICAgCi0gICAgICAgICAgICBWZWN0b3I8aGFuZGxlX2VudHJ5Pm1IYW5kbGVUb09iamVjdDsKLQotICAgICAgICAgICAgYm9vbCAgICAgICAgICAgICAgICBtTWFuYWdlc0NvbnRleHRzOwotICAgICAgICAgICAgY29udGV4dF9jaGVja19mdW5jICBtQmluZGVyQ29udGV4dENoZWNrRnVuYzsKLSAgICAgICAgICAgIHZvaWQqICAgICAgICAgICAgICAgbUJpbmRlckNvbnRleHRVc2VyRGF0YTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgS2V5ZWRWZWN0b3I8U3RyaW5nMTYsIHNwPElCaW5kZXI+ID4KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbUNvbnRleHRzOwotCi0KLSAgICAgICAgICAgIFN0cmluZzggICAgICAgICAgICAgbVJvb3REaXI7Ci0gICAgICAgICAgICBib29sICAgICAgICAgICAgICAgIG1UaHJlYWRQb29sU3RhcnRlZDsKLSAgICB2b2xhdGlsZSBpbnQzMl90ICAgICAgICAgICAgbVRocmVhZFBvb2xTZXE7Ci19OwotICAgIAotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNlbmRpZiAvLyBBTkRST0lEX1BST0NFU1NfU1RBVEVfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9SZWZCYXNlLmggYi9pbmNsdWRlL3V0aWxzL1JlZkJhc2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggY2JkYTBmZC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL1JlZkJhc2UuaAorKysgL2Rldi9udWxsCkBAIC0xLDU1MCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1JFRl9CQVNFX0gKLSNkZWZpbmUgQU5EUk9JRF9SRUZfQkFTRV9ICi0KLSNpbmNsdWRlIDxjdXRpbHMvYXRvbWljLmg+Ci0jaW5jbHVkZSA8dXRpbHMvVGV4dE91dHB1dC5oPgotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPiBjbGFzcyB3cDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNkZWZpbmUgQ09NUEFSRShfb3BfKSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi1pbmxpbmUgYm9vbCBvcGVyYXRvciBfb3BfIChjb25zdCBzcDxUPiYgbykgY29uc3QgeyAgICAgICAgICAgICAgXAotICAgIHJldHVybiBtX3B0ciBfb3BfIG8ubV9wdHI7ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLX0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi1pbmxpbmUgYm9vbCBvcGVyYXRvciBfb3BfIChjb25zdCB3cDxUPiYgbykgY29uc3QgeyAgICAgICAgICAgICAgXAotICAgIHJldHVybiBtX3B0ciBfb3BfIG8ubV9wdHI7ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLX0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi1pbmxpbmUgYm9vbCBvcGVyYXRvciBfb3BfIChjb25zdCBUKiBvKSBjb25zdCB7ICAgICAgICAgICAgICAgICAgXAotICAgIHJldHVybiBtX3B0ciBfb3BfIG87ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLX0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi10ZW1wbGF0ZTx0eXBlbmFtZSBVPiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotaW5saW5lIGJvb2wgb3BlcmF0b3IgX29wXyAoY29uc3Qgc3A8VT4mIG8pIGNvbnN0IHsgICAgICAgICAgICAgIFwKLSAgICByZXR1cm4gbV9wdHIgX29wXyBvLm1fcHRyOyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi19ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotdGVtcGxhdGU8dHlwZW5hbWUgVT4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLWlubGluZSBib29sIG9wZXJhdG9yIF9vcF8gKGNvbnN0IHdwPFU+JiBvKSBjb25zdCB7ICAgICAgICAgICAgICBcCi0gICAgcmV0dXJuIG1fcHRyIF9vcF8gby5tX3B0cjsgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLXRlbXBsYXRlPHR5cGVuYW1lIFU+ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi1pbmxpbmUgYm9vbCBvcGVyYXRvciBfb3BfIChjb25zdCBVKiBvKSBjb25zdCB7ICAgICAgICAgICAgICAgICAgXAotICAgIHJldHVybiBtX3B0ciBfb3BfIG87ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIFJlZkJhc2UKLXsKLXB1YmxpYzoKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICBpbmNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0OwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgIGRlY1N0cm9uZyhjb25zdCB2b2lkKiBpZCkgY29uc3Q7Ci0gICAgCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgZm9yY2VJbmNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0OwotCi0gICAgICAgICAgICAvLyEgREVCVUdHSU5HIE9OTFk6IEdldCBjdXJyZW50IHN0cm9uZyByZWYgY291bnQuCi0gICAgICAgICAgICBpbnQzMl90ICAgICAgICAgZ2V0U3Ryb25nQ291bnQoKSBjb25zdDsKLQotICAgIGNsYXNzIHdlYWtyZWZfdHlwZQotICAgIHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIFJlZkJhc2UqICAgICAgICAgICAgcmVmQmFzZSgpIGNvbnN0OwotICAgICAgICAKLSAgICAgICAgdm9pZCAgICAgICAgICAgICAgICBpbmNXZWFrKGNvbnN0IHZvaWQqIGlkKTsKLSAgICAgICAgdm9pZCAgICAgICAgICAgICAgICBkZWNXZWFrKGNvbnN0IHZvaWQqIGlkKTsKLSAgICAgICAgCi0gICAgICAgIGJvb2wgICAgICAgICAgICAgICAgYXR0ZW1wdEluY1N0cm9uZyhjb25zdCB2b2lkKiBpZCk7Ci0gICAgICAgIAotICAgICAgICAvLyEgVGhpcyBpcyBvbmx5IHNhZmUgaWYgeW91IGhhdmUgc2V0IE9CSkVDVF9MSUZFVElNRV9GT1JFVkVSLgotICAgICAgICBib29sICAgICAgICAgICAgICAgIGF0dGVtcHRJbmNXZWFrKGNvbnN0IHZvaWQqIGlkKTsKLQotICAgICAgICAvLyEgREVCVUdHSU5HIE9OTFk6IEdldCBjdXJyZW50IHdlYWsgcmVmIGNvdW50LgotICAgICAgICBpbnQzMl90ICAgICAgICAgICAgIGdldFdlYWtDb3VudCgpIGNvbnN0OwotCi0gICAgICAgIC8vISBERUJVR0dJTkcgT05MWTogUHJpbnQgcmVmZXJlbmNlcyBoZWxkIG9uIG9iamVjdC4KLSAgICAgICAgdm9pZCAgICAgICAgICAgICAgICBwcmludFJlZnMoKSBjb25zdDsKLQotICAgICAgICAvLyEgREVCVUdHSU5HIE9OTFk6IEVuYWJsZSB0cmFja2luZyBmb3IgdGhpcyBvYmplY3QuCi0gICAgICAgIC8vIGVuYWJsZSAtLSBlbmFibGUvZGlzYWJsZSB0cmFja2luZwotICAgICAgICAvLyByZXRhaW4gLS0gd2hlbiB0cmFja2luZyBpcyBlbmFibGUsIGlmIHRydWUsIHRoZW4gd2Ugc2F2ZSBhIHN0YWNrIHRyYWNlCi0gICAgICAgIC8vICAgICAgICAgICBmb3IgZWFjaCByZWZlcmVuY2UgYW5kIGRlcmVmZXJlbmNlOyB3aGVuIHJldGFpbiA9PSBmYWxzZSwgd2UKLSAgICAgICAgLy8gICAgICAgICAgIG1hdGNoIHVwIHJlZmVyZW5jZXMgYW5kIGRlcmVmZXJlbmNlcyBhbmQga2VlcCBvbmx5IHRoZSAKLSAgICAgICAgLy8gICAgICAgICAgIG91dHN0YW5kaW5nIG9uZXMuCi0gICAgICAgIAotICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHRyYWNrTWUoYm9vbCBlbmFibGUsIGJvb2wgcmV0YWluKTsKLSAgICB9OwotICAgIAotICAgICAgICAgICAgd2Vha3JlZl90eXBlKiAgIGNyZWF0ZVdlYWsoY29uc3Qgdm9pZCogaWQpIGNvbnN0OwotICAgICAgICAgICAgCi0gICAgICAgICAgICB3ZWFrcmVmX3R5cGUqICAgZ2V0V2Vha1JlZnMoKSBjb25zdDsKLQotICAgICAgICAgICAgLy8hIERFQlVHR0lORyBPTkxZOiBQcmludCByZWZlcmVuY2VzIGhlbGQgb24gb2JqZWN0LgotICAgIGlubGluZSAgdm9pZCAgICAgICAgICAgIHByaW50UmVmcygpIGNvbnN0IHsgZ2V0V2Vha1JlZnMoKS0+cHJpbnRSZWZzKCk7IH0KLQotICAgICAgICAgICAgLy8hIERFQlVHR0lORyBPTkxZOiBFbmFibGUgdHJhY2tpbmcgb2Ygb2JqZWN0LgotICAgIGlubGluZSAgdm9pZCAgICAgICAgICAgIHRyYWNrTWUoYm9vbCBlbmFibGUsIGJvb2wgcmV0YWluKQotICAgIHsgCi0gICAgICAgIGdldFdlYWtSZWZzKCktPnRyYWNrTWUoZW5hYmxlLCByZXRhaW4pOyAKLSAgICB9Ci0KLXByb3RlY3RlZDoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBSZWZCYXNlKCk7Ci0gICAgdmlydHVhbCAgICAgICAgICAgICAgICAgflJlZkJhc2UoKTsKLSAgICAKLSAgICAvLyEgRmxhZ3MgZm9yIGV4dGVuZE9iamVjdExpZmV0aW1lKCkKLSAgICBlbnVtIHsKLSAgICAgICAgT0JKRUNUX0xJRkVUSU1FX1dFQUsgICAgPSAweDAwMDEsCi0gICAgICAgIE9CSkVDVF9MSUZFVElNRV9GT1JFVkVSID0gMHgwMDAzCi0gICAgfTsKLSAgICAKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICBleHRlbmRPYmplY3RMaWZldGltZShpbnQzMl90IG1vZGUpOwotICAgICAgICAgICAgCi0gICAgLy8hIEZsYWdzIGZvciBvbkluY1N0cm9uZ0F0dGVtcHRlZCgpCi0gICAgZW51bSB7Ci0gICAgICAgIEZJUlNUX0lOQ19TVFJPTkcgPSAweDAwMDEKLSAgICB9OwotICAgIAotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIG9uRmlyc3RSZWYoKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICBvbkxhc3RTdHJvbmdSZWYoY29uc3Qgdm9pZCogaWQpOwotICAgIHZpcnR1YWwgYm9vbCAgICAgICAgICAgIG9uSW5jU3Ryb25nQXR0ZW1wdGVkKHVpbnQzMl90IGZsYWdzLCBjb25zdCB2b2lkKiBpZCk7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgb25MYXN0V2Vha1JlZihjb25zdCB2b2lkKiBpZCk7Ci0KLXByaXZhdGU6Ci0gICAgZnJpZW5kIGNsYXNzIHdlYWtyZWZfdHlwZTsKLSAgICBjbGFzcyB3ZWFrcmVmX2ltcGw7Ci0gICAgCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgUmVmQmFzZShjb25zdCBSZWZCYXNlJiBvKTsKLSAgICAgICAgICAgIFJlZkJhc2UmICAgICAgICBvcGVyYXRvcj0oY29uc3QgUmVmQmFzZSYgbyk7Ci0gICAgICAgICAgICAKLSAgICAgICAgd2Vha3JlZl9pbXBsKiBjb25zdCBtUmVmczsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi10ZW1wbGF0ZSA8Y2xhc3MgVD4KLWNsYXNzIExpZ2h0UmVmQmFzZQotewotcHVibGljOgotICAgIGlubGluZSBMaWdodFJlZkJhc2UoKSA6IG1Db3VudCgwKSB7IH0KLSAgICBpbmxpbmUgdm9pZCBpbmNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0IHsKLSAgICAgICAgYW5kcm9pZF9hdG9taWNfaW5jKCZtQ291bnQpOwotICAgIH0KLSAgICBpbmxpbmUgdm9pZCBkZWNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0IHsKLSAgICAgICAgaWYgKGFuZHJvaWRfYXRvbWljX2RlYygmbUNvdW50KSA9PSAxKSB7Ci0gICAgICAgICAgICBkZWxldGUgc3RhdGljX2Nhc3Q8Y29uc3QgVCo+KHRoaXMpOwotICAgICAgICB9Ci0gICAgfQotICAgIAotcHJvdGVjdGVkOgotICAgIGlubGluZSB+TGlnaHRSZWZCYXNlKCkgeyB9Ci0gICAgCi1wcml2YXRlOgotICAgIG11dGFibGUgdm9sYXRpbGUgaW50MzJfdCBtQ291bnQ7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdGVtcGxhdGUgPHR5cGVuYW1lIFQ+Ci1jbGFzcyBzcAotewotcHVibGljOgotICAgIHR5cGVkZWYgdHlwZW5hbWUgUmVmQmFzZTo6d2Vha3JlZl90eXBlIHdlYWtyZWZfdHlwZTsKLSAgICAKLSAgICBpbmxpbmUgc3AoKSA6IG1fcHRyKDApIHsgfQotCi0gICAgc3AoVCogb3RoZXIpOwotICAgIHNwKGNvbnN0IHNwPFQ+JiBvdGhlcik7Ci0gICAgdGVtcGxhdGU8dHlwZW5hbWUgVT4gc3AoVSogb3RoZXIpOwotICAgIHRlbXBsYXRlPHR5cGVuYW1lIFU+IHNwKGNvbnN0IHNwPFU+JiBvdGhlcik7Ci0KLSAgICB+c3AoKTsKLSAgICAKLSAgICAvLyBBc3NpZ25tZW50Ci0KLSAgICBzcCYgb3BlcmF0b3IgPSAoVCogb3RoZXIpOwotICAgIHNwJiBvcGVyYXRvciA9IChjb25zdCBzcDxUPiYgb3RoZXIpOwotICAgIAotICAgIHRlbXBsYXRlPHR5cGVuYW1lIFU+IHNwJiBvcGVyYXRvciA9IChjb25zdCBzcDxVPiYgb3RoZXIpOwotICAgIHRlbXBsYXRlPHR5cGVuYW1lIFU+IHNwJiBvcGVyYXRvciA9IChVKiBvdGhlcik7Ci0gICAgCi0gICAgLy8hIFNwZWNpYWwgb3B0aW1pemF0aW9uIGZvciB1c2UgYnkgUHJvY2Vzc1N0YXRlIChhbmQgbm9ib2R5IGVsc2UpLgotICAgIHZvaWQgZm9yY2Vfc2V0KFQqIG90aGVyKTsKLSAgICAKLSAgICAvLyBSZXNldAotICAgIAotICAgIHZvaWQgY2xlYXIoKTsKLSAgICAKLSAgICAvLyBBY2Nlc3NvcnMKLQotICAgIGlubGluZSAgVCYgICAgICBvcGVyYXRvciogKCkgY29uc3QgIHsgcmV0dXJuICptX3B0cjsgfQotICAgIGlubGluZSAgVCogICAgICBvcGVyYXRvci0+ICgpIGNvbnN0IHsgcmV0dXJuIG1fcHRyOyAgfQotICAgIGlubGluZSAgVCogICAgICBnZXQoKSBjb25zdCAgICAgICAgIHsgcmV0dXJuIG1fcHRyOyB9Ci0KLSAgICAvLyBPcGVyYXRvcnMKLSAgICAgICAgCi0gICAgQ09NUEFSRSg9PSkKLSAgICBDT01QQVJFKCE9KQotICAgIENPTVBBUkUoPikKLSAgICBDT01QQVJFKDwpCi0gICAgQ09NUEFSRSg8PSkKLSAgICBDT01QQVJFKD49KQotCi1wcml2YXRlOiAgICAKLSAgICB0ZW1wbGF0ZTx0eXBlbmFtZSBZPiBmcmllbmQgY2xhc3Mgc3A7Ci0gICAgdGVtcGxhdGU8dHlwZW5hbWUgWT4gZnJpZW5kIGNsYXNzIHdwOwotCi0gICAgLy8gT3B0aW1pemF0aW9uIGZvciB3cDo6cHJvbW90ZSgpLgotICAgIHNwKFQqIHAsIHdlYWtyZWZfdHlwZSogcmVmcyk7Ci0gICAgCi0gICAgVCogICAgICAgICAgICAgIG1fcHRyOwotfTsKLQotdGVtcGxhdGUgPHR5cGVuYW1lIFQ+Ci1UZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCBjb25zdCBzcDxUPiYgdmFsKTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXRlbXBsYXRlIDx0eXBlbmFtZSBUPgotY2xhc3Mgd3AKLXsKLXB1YmxpYzoKLSAgICB0eXBlZGVmIHR5cGVuYW1lIFJlZkJhc2U6OndlYWtyZWZfdHlwZSB3ZWFrcmVmX3R5cGU7Ci0gICAgCi0gICAgaW5saW5lIHdwKCkgOiBtX3B0cigwKSB7IH0KLQotICAgIHdwKFQqIG90aGVyKTsKLSAgICB3cChjb25zdCB3cDxUPiYgb3RoZXIpOwotICAgIHdwKGNvbnN0IHNwPFQ+JiBvdGhlcik7Ci0gICAgdGVtcGxhdGU8dHlwZW5hbWUgVT4gd3AoVSogb3RoZXIpOwotICAgIHRlbXBsYXRlPHR5cGVuYW1lIFU+IHdwKGNvbnN0IHNwPFU+JiBvdGhlcik7Ci0gICAgdGVtcGxhdGU8dHlwZW5hbWUgVT4gd3AoY29uc3Qgd3A8VT4mIG90aGVyKTsKLQotICAgIH53cCgpOwotICAgIAotICAgIC8vIEFzc2lnbm1lbnQKLQotICAgIHdwJiBvcGVyYXRvciA9IChUKiBvdGhlcik7Ci0gICAgd3AmIG9wZXJhdG9yID0gKGNvbnN0IHdwPFQ+JiBvdGhlcik7Ci0gICAgd3AmIG9wZXJhdG9yID0gKGNvbnN0IHNwPFQ+JiBvdGhlcik7Ci0gICAgCi0gICAgdGVtcGxhdGU8dHlwZW5hbWUgVT4gd3AmIG9wZXJhdG9yID0gKFUqIG90aGVyKTsKLSAgICB0ZW1wbGF0ZTx0eXBlbmFtZSBVPiB3cCYgb3BlcmF0b3IgPSAoY29uc3Qgd3A8VT4mIG90aGVyKTsKLSAgICB0ZW1wbGF0ZTx0eXBlbmFtZSBVPiB3cCYgb3BlcmF0b3IgPSAoY29uc3Qgc3A8VT4mIG90aGVyKTsKLSAgICAKLSAgICB2b2lkIHNldF9vYmplY3RfYW5kX3JlZnMoVCogb3RoZXIsIHdlYWtyZWZfdHlwZSogcmVmcyk7Ci0KLSAgICAvLyBwcm9tb3Rpb24gdG8gc3AKLSAgICAKLSAgICBzcDxUPiBwcm9tb3RlKCkgY29uc3Q7Ci0KLSAgICAvLyBSZXNldAotICAgIAotICAgIHZvaWQgY2xlYXIoKTsKLQotICAgIC8vIEFjY2Vzc29ycwotICAgIAotICAgIGlubGluZSAgd2Vha3JlZl90eXBlKiBnZXRfcmVmcygpIGNvbnN0IHsgcmV0dXJuIG1fcmVmczsgfQotICAgIAotICAgIGlubGluZSAgVCogdW5zYWZlX2dldCgpIGNvbnN0IHsgcmV0dXJuIG1fcHRyOyB9Ci0KLSAgICAvLyBPcGVyYXRvcnMKLSAgICAgICAgCi0gICAgQ09NUEFSRSg9PSkKLSAgICBDT01QQVJFKCE9KQotICAgIENPTVBBUkUoPikKLSAgICBDT01QQVJFKDwpCi0gICAgQ09NUEFSRSg8PSkKLSAgICBDT01QQVJFKD49KQotCi1wcml2YXRlOgotICAgIHRlbXBsYXRlPHR5cGVuYW1lIFk+IGZyaWVuZCBjbGFzcyBzcDsKLSAgICB0ZW1wbGF0ZTx0eXBlbmFtZSBZPiBmcmllbmQgY2xhc3Mgd3A7Ci0KLSAgICBUKiAgICAgICAgICAgICAgbV9wdHI7Ci0gICAgd2Vha3JlZl90eXBlKiAgIG1fcmVmczsKLX07Ci0KLXRlbXBsYXRlIDx0eXBlbmFtZSBUPgotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgY29uc3Qgd3A8VD4mIHZhbCk7Ci0KLSN1bmRlZiBDT01QQVJFCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gTm8gdXNlciBzZXJ2aWNlYWJsZSBwYXJ0cyBiZWxvdyBoZXJlLgotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotc3A8VD46OnNwKFQqIG90aGVyKQotICAgIDogbV9wdHIob3RoZXIpCi17Ci0gICAgaWYgKG90aGVyKSBvdGhlci0+aW5jU3Ryb25nKHRoaXMpOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotc3A8VD46OnNwKGNvbnN0IHNwPFQ+JiBvdGhlcikKLSAgICA6IG1fcHRyKG90aGVyLm1fcHRyKQotewotICAgIGlmIChtX3B0cikgbV9wdHItPmluY1N0cm9uZyh0aGlzKTsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgVD4gdGVtcGxhdGU8dHlwZW5hbWUgVT4KLXNwPFQ+OjpzcChVKiBvdGhlcikgOiBtX3B0cihvdGhlcikKLXsKLSAgICBpZiAob3RoZXIpIG90aGVyLT5pbmNTdHJvbmcodGhpcyk7Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+IHRlbXBsYXRlPHR5cGVuYW1lIFU+Ci1zcDxUPjo6c3AoY29uc3Qgc3A8VT4mIG90aGVyKQotICAgIDogbV9wdHIob3RoZXIubV9wdHIpCi17Ci0gICAgaWYgKG1fcHRyKSBtX3B0ci0+aW5jU3Ryb25nKHRoaXMpOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotc3A8VD46On5zcCgpCi17Ci0gICAgaWYgKG1fcHRyKSBtX3B0ci0+ZGVjU3Ryb25nKHRoaXMpOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotc3A8VD4mIHNwPFQ+OjpvcGVyYXRvciA9IChjb25zdCBzcDxUPiYgb3RoZXIpIHsKLSAgICBpZiAob3RoZXIubV9wdHIpIG90aGVyLm1fcHRyLT5pbmNTdHJvbmcodGhpcyk7Ci0gICAgaWYgKG1fcHRyKSBtX3B0ci0+ZGVjU3Ryb25nKHRoaXMpOwotICAgIG1fcHRyID0gb3RoZXIubV9wdHI7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotc3A8VD4mIHNwPFQ+OjpvcGVyYXRvciA9IChUKiBvdGhlcikKLXsKLSAgICBpZiAob3RoZXIpIG90aGVyLT5pbmNTdHJvbmcodGhpcyk7Ci0gICAgaWYgKG1fcHRyKSBtX3B0ci0+ZGVjU3Ryb25nKHRoaXMpOwotICAgIG1fcHRyID0gb3RoZXI7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPiB0ZW1wbGF0ZTx0eXBlbmFtZSBVPgotc3A8VD4mIHNwPFQ+OjpvcGVyYXRvciA9IChjb25zdCBzcDxVPiYgb3RoZXIpCi17Ci0gICAgaWYgKG90aGVyLm1fcHRyKSBvdGhlci5tX3B0ci0+aW5jU3Ryb25nKHRoaXMpOwotICAgIGlmIChtX3B0cikgbV9wdHItPmRlY1N0cm9uZyh0aGlzKTsKLSAgICBtX3B0ciA9IG90aGVyLm1fcHRyOwotICAgIHJldHVybiAqdGhpczsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgVD4gdGVtcGxhdGU8dHlwZW5hbWUgVT4KLXNwPFQ+JiBzcDxUPjo6b3BlcmF0b3IgPSAoVSogb3RoZXIpCi17Ci0gICAgaWYgKG90aGVyKSBvdGhlci0+aW5jU3Ryb25nKHRoaXMpOwotICAgIGlmIChtX3B0cikgbV9wdHItPmRlY1N0cm9uZyh0aGlzKTsKLSAgICBtX3B0ciA9IG90aGVyOwotICAgIHJldHVybiAqdGhpczsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgVD4gICAgCi12b2lkIHNwPFQ+Ojpmb3JjZV9zZXQoVCogb3RoZXIpCi17Ci0gICAgb3RoZXItPmZvcmNlSW5jU3Ryb25nKHRoaXMpOwotICAgIG1fcHRyID0gb3RoZXI7Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+Ci12b2lkIHNwPFQ+OjpjbGVhcigpCi17Ci0gICAgaWYgKG1fcHRyKSB7Ci0gICAgICAgIG1fcHRyLT5kZWNTdHJvbmcodGhpcyk7Ci0gICAgICAgIG1fcHRyID0gMDsKLSAgICB9Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+Ci1zcDxUPjo6c3AoVCogcCwgd2Vha3JlZl90eXBlKiByZWZzKQotICAgIDogbV9wdHIoKHAgJiYgcmVmcy0+YXR0ZW1wdEluY1N0cm9uZyh0aGlzKSkgPyBwIDogMCkKLXsKLX0KLQotdGVtcGxhdGUgPHR5cGVuYW1lIFQ+Ci1pbmxpbmUgVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgY29uc3Qgc3A8VD4mIHZhbCkKLXsKLSAgICB0byA8PCAic3A8PigiIDw8IHZhbC5nZXQoKSA8PCAiKSI7Ci0gICAgcmV0dXJuIHRvOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdGVtcGxhdGU8dHlwZW5hbWUgVD4KLXdwPFQ+Ojp3cChUKiBvdGhlcikKLSAgICA6IG1fcHRyKG90aGVyKQotewotICAgIGlmIChvdGhlcikgbV9yZWZzID0gb3RoZXItPmNyZWF0ZVdlYWsodGhpcyk7Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+Ci13cDxUPjo6d3AoY29uc3Qgd3A8VD4mIG90aGVyKQotICAgIDogbV9wdHIob3RoZXIubV9wdHIpLCBtX3JlZnMob3RoZXIubV9yZWZzKQotewotICAgIGlmIChtX3B0cikgbV9yZWZzLT5pbmNXZWFrKHRoaXMpOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotd3A8VD46OndwKGNvbnN0IHNwPFQ+JiBvdGhlcikKLSAgICA6IG1fcHRyKG90aGVyLm1fcHRyKQotewotICAgIGlmIChtX3B0cikgewotICAgICAgICBtX3JlZnMgPSBtX3B0ci0+Y3JlYXRlV2Vhayh0aGlzKTsKLSAgICB9Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+IHRlbXBsYXRlPHR5cGVuYW1lIFU+Ci13cDxUPjo6d3AoVSogb3RoZXIpCi0gICAgOiBtX3B0cihvdGhlcikKLXsKLSAgICBpZiAob3RoZXIpIG1fcmVmcyA9IG90aGVyLT5jcmVhdGVXZWFrKHRoaXMpOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPiB0ZW1wbGF0ZTx0eXBlbmFtZSBVPgotd3A8VD46OndwKGNvbnN0IHdwPFU+JiBvdGhlcikKLSAgICA6IG1fcHRyKG90aGVyLm1fcHRyKQotewotICAgIGlmIChtX3B0cikgewotICAgICAgICBtX3JlZnMgPSBvdGhlci5tX3JlZnM7Ci0gICAgICAgIG1fcmVmcy0+aW5jV2Vhayh0aGlzKTsKLSAgICB9Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+IHRlbXBsYXRlPHR5cGVuYW1lIFU+Ci13cDxUPjo6d3AoY29uc3Qgc3A8VT4mIG90aGVyKQotICAgIDogbV9wdHIob3RoZXIubV9wdHIpCi17Ci0gICAgaWYgKG1fcHRyKSB7Ci0gICAgICAgIG1fcmVmcyA9IG1fcHRyLT5jcmVhdGVXZWFrKHRoaXMpOwotICAgIH0KLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgVD4KLXdwPFQ+Ojp+d3AoKQotewotICAgIGlmIChtX3B0cikgbV9yZWZzLT5kZWNXZWFrKHRoaXMpOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotd3A8VD4mIHdwPFQ+OjpvcGVyYXRvciA9IChUKiBvdGhlcikKLXsKLSAgICB3ZWFrcmVmX3R5cGUqIG5ld1JlZnMgPQotICAgICAgICBvdGhlciA/IG90aGVyLT5jcmVhdGVXZWFrKHRoaXMpIDogMDsKLSAgICBpZiAobV9wdHIpIG1fcmVmcy0+ZGVjV2Vhayh0aGlzKTsKLSAgICBtX3B0ciA9IG90aGVyOwotICAgIG1fcmVmcyA9IG5ld1JlZnM7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotd3A8VD4mIHdwPFQ+OjpvcGVyYXRvciA9IChjb25zdCB3cDxUPiYgb3RoZXIpCi17Ci0gICAgaWYgKG90aGVyLm1fcHRyKSBvdGhlci5tX3JlZnMtPmluY1dlYWsodGhpcyk7Ci0gICAgaWYgKG1fcHRyKSBtX3JlZnMtPmRlY1dlYWsodGhpcyk7Ci0gICAgbV9wdHIgPSBvdGhlci5tX3B0cjsKLSAgICBtX3JlZnMgPSBvdGhlci5tX3JlZnM7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotd3A8VD4mIHdwPFQ+OjpvcGVyYXRvciA9IChjb25zdCBzcDxUPiYgb3RoZXIpCi17Ci0gICAgd2Vha3JlZl90eXBlKiBuZXdSZWZzID0KLSAgICAgICAgb3RoZXIgIT0gTlVMTCA/IG90aGVyLT5jcmVhdGVXZWFrKHRoaXMpIDogMDsKLSAgICBpZiAobV9wdHIpIG1fcmVmcy0+ZGVjV2Vhayh0aGlzKTsKLSAgICBtX3B0ciA9IG90aGVyLmdldCgpOwotICAgIG1fcmVmcyA9IG5ld1JlZnM7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPiB0ZW1wbGF0ZTx0eXBlbmFtZSBVPgotd3A8VD4mIHdwPFQ+OjpvcGVyYXRvciA9IChVKiBvdGhlcikKLXsKLSAgICB3ZWFrcmVmX3R5cGUqIG5ld1JlZnMgPQotICAgICAgICBvdGhlciA/IG90aGVyLT5jcmVhdGVXZWFrKHRoaXMpIDogMDsKLSAgICBpZiAobV9wdHIpIG1fcmVmcy0+ZGVjV2Vhayh0aGlzKTsKLSAgICBtX3B0ciA9IG90aGVyOwotICAgIG1fcmVmcyA9IG5ld1JlZnM7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPiB0ZW1wbGF0ZTx0eXBlbmFtZSBVPgotd3A8VD4mIHdwPFQ+OjpvcGVyYXRvciA9IChjb25zdCB3cDxVPiYgb3RoZXIpCi17Ci0gICAgaWYgKG90aGVyLm1fcHRyKSBvdGhlci5tX3JlZnMtPmluY1dlYWsodGhpcyk7Ci0gICAgaWYgKG1fcHRyKSBtX3JlZnMtPmRlY1dlYWsodGhpcyk7Ci0gICAgbV9wdHIgPSBvdGhlci5tX3B0cjsKLSAgICBtX3JlZnMgPSBvdGhlci5tX3JlZnM7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPiB0ZW1wbGF0ZTx0eXBlbmFtZSBVPgotd3A8VD4mIHdwPFQ+OjpvcGVyYXRvciA9IChjb25zdCBzcDxVPiYgb3RoZXIpCi17Ci0gICAgd2Vha3JlZl90eXBlKiBuZXdSZWZzID0KLSAgICAgICAgb3RoZXIgIT0gTlVMTCA/IG90aGVyLT5jcmVhdGVXZWFrKHRoaXMpIDogMDsKLSAgICBpZiAobV9wdHIpIG1fcmVmcy0+ZGVjV2Vhayh0aGlzKTsKLSAgICBtX3B0ciA9IG90aGVyLmdldCgpOwotICAgIG1fcmVmcyA9IG5ld1JlZnM7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotdm9pZCB3cDxUPjo6c2V0X29iamVjdF9hbmRfcmVmcyhUKiBvdGhlciwgd2Vha3JlZl90eXBlKiByZWZzKQotewotICAgIGlmIChvdGhlcikgcmVmcy0+aW5jV2Vhayh0aGlzKTsKLSAgICBpZiAobV9wdHIpIG1fcmVmcy0+ZGVjV2Vhayh0aGlzKTsKLSAgICBtX3B0ciA9IG90aGVyOwotICAgIG1fcmVmcyA9IHJlZnM7Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+Ci1zcDxUPiB3cDxUPjo6cHJvbW90ZSgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHNwPFQ+KG1fcHRyLCBtX3JlZnMpOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotdm9pZCB3cDxUPjo6Y2xlYXIoKQotewotICAgIGlmIChtX3B0cikgewotICAgICAgICBtX3JlZnMtPmRlY1dlYWsodGhpcyk7Ci0gICAgICAgIG1fcHRyID0gMDsKLSAgICB9Ci19Ci0KLXRlbXBsYXRlIDx0eXBlbmFtZSBUPgotaW5saW5lIFRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGNvbnN0IHdwPFQ+JiB2YWwpCi17Ci0gICAgdG8gPDwgIndwPD4oIiA8PCB2YWwudW5zYWZlX2dldCgpIDw8ICIpIjsKLSAgICByZXR1cm4gdG87Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLy8gQU5EUk9JRF9SRUZfQkFTRV9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL1Jlc291cmNlVHlwZXMuaCBiL2luY2x1ZGUvdXRpbHMvUmVzb3VyY2VUeXBlcy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkODNhMzNjLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvUmVzb3VyY2VUeXBlcy5oCisrKyAvZGV2L251bGwKQEAgLTEsMTcxNCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIERlZmluaXRpb25zIG9mIHJlc291cmNlIGRhdGEgc3RydWN0dXJlcy4KLS8vCi0jaWZuZGVmIF9MSUJTX1VUSUxTX1JFU09VUkNFX1RZUEVTX0gKLSNkZWZpbmUgX0xJQlNfVVRJTFNfUkVTT1VSQ0VfVFlQRVNfSAotCi0jaW5jbHVkZSA8dXRpbHMvQXNzZXQuaD4KLSNpbmNsdWRlIDx1dGlscy9CeXRlT3JkZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmcxNi5oPgotI2luY2x1ZGUgPHV0aWxzL1ZlY3Rvci5oPgotCi0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyoqICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCi0gKiAgUE5HIEV4dGVuc2lvbnMKLSAqCi0gKiAgTmV3IHByaXZhdGUgY2h1bmtzIHRoYXQgbWF5IGJlIHBsYWNlZCBpbiBQTkcgaW1hZ2VzLgotICoKLSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiAqLwotCi0vKioKLSAqIFRoaXMgY2h1bmsgc3BlY2lmaWVzIGhvdyB0byBzcGxpdCBhbiBpbWFnZSBpbnRvIHNlZ21lbnRzIGZvcgotICogc2NhbGluZy4KLSAqCi0gKiBUaGVyZSBhcmUgSiBob3Jpem9udGFsIGFuZCBLIHZlcnRpY2FsIHNlZ21lbnRzLiAgVGhlc2Ugc2VnbWVudHMgZGl2aWRlCi0gKiB0aGUgaW1hZ2UgaW50byBKKksgcmVnaW9ucyBhcyBmb2xsb3dzICh3aGVyZSBKPTQgYW5kIEs9Myk6Ci0gKgotICogICAgICBGMCAgIFMwICAgIEYxICAgICBTMQotICogICArLS0tLS0rLS0tLSstLS0tLS0rLS0tLS0tLSsKLSAqIFMyfCAgMCAgfCAgMSB8ICAyICAgfCAgIDMgICB8Ci0gKiAgICstLS0tLSstLS0tKy0tLS0tLSstLS0tLS0tKwotICogICB8ICAgICB8ICAgIHwgICAgICB8ICAgICAgIHwKLSAqICAgfCAgICAgfCAgICB8ICAgICAgfCAgICAgICB8Ci0gKiBGMnwgIDQgIHwgIDUgfCAgNiAgIHwgICA3ICAgfAotICogICB8ICAgICB8ICAgIHwgICAgICB8ICAgICAgIHwKLSAqICAgfCAgICAgfCAgICB8ICAgICAgfCAgICAgICB8Ci0gKiAgICstLS0tLSstLS0tKy0tLS0tLSstLS0tLS0tKwotICogUzN8ICA4ICB8ICA5IHwgIDEwICB8ICAgMTEgIHwKLSAqICAgKy0tLS0tKy0tLS0rLS0tLS0tKy0tLS0tLS0rCi0gKgotICogRWFjaCBob3Jpem9udGFsIGFuZCB2ZXJ0aWNhbCBzZWdtZW50IGlzIGNvbnNpZGVyZWQgdG8gYnkgZWl0aGVyCi0gKiBzdHJldGNoYWJsZSAobWFya2VkIGJ5IHRoZSBTeCBsYWJlbHMpIG9yIGZpeGVkIChtYXJrZWQgYnkgdGhlIEZ5Ci0gKiBsYWJlbHMpLCBpbiB0aGUgaG9yaXpvbnRhbCBvciB2ZXJ0aWNhbCBheGlzLCByZXNwZWN0aXZlbHkuIEluIHRoZQotICogYWJvdmUgZXhhbXBsZSwgdGhlIGZpcnN0IGlzIGhvcml6b250YWwgc2VnbWVudCAoRjApIGlzIGZpeGVkLCB0aGUKLSAqIG5leHQgaXMgc3RyZXRjaGFibGUgYW5kIHRoZW4gdGhleSBjb250aW51ZSB0byBhbHRlcm5hdGUuIE5vdGUgdGhhdAotICogdGhlIHNlZ21lbnQgbGlzdCBmb3IgZWFjaCBheGlzIGNhbiBiZWdpbiBvciBlbmQgd2l0aCBhIHN0cmV0Y2hhYmxlCi0gKiBvciBmaXhlZCBzZWdtZW50LgotICoKLSAqIFRoZSByZWxhdGl2ZSBzaXplcyBvZiB0aGUgc3RyZXRjaHkgc2VnbWVudHMgaW5kaWNhdGVzIHRoZSByZWxhdGl2ZQotICogYW1vdW50IG9mIHN0cmV0Y2hpbmVzcyBvZiB0aGUgcmVnaW9ucyBib3JkZXJlZCBieSB0aGUgc2VnbWVudHMuICBGb3IKLSAqIGV4YW1wbGUsIHJlZ2lvbnMgMywgNyBhbmQgMTEgYWJvdmUgd2lsbCB0YWtlIHVwIG1vcmUgaG9yaXpvbnRhbCBzcGFjZQotICogdGhhbiByZWdpb25zIDEsIDUgYW5kIDkgc2luY2UgdGhlIGhvcml6b25hbCBzZWdtZW50IGFzc29jaWF0ZWQgd2l0aAotICogdGhlIGZpcnN0IHNldCBvZiByZWdpb25zIGlzIGxhcmdlciB0aGFuIHRoZSBvdGhlciBzZXQgb2YgcmVnaW9ucy4gIFRoZQotICogcmF0aW9zIG9mIHRoZSBhbW91bnQgb2YgaG9yaXpvbnRhbCAob3IgdmVydGljYWwpIHNwYWNlIHRha2VuIGJ5IGFueQotICogdHdvIHN0cmV0Y2hhYmxlIHNsaWNlcyBpcyBleGFjdGx5IHRoZSByYXRpbyBvZiB0aGVpciBjb3JyZXNwb25kaW5nCi0gKiBzZWdtZW50IGxlbmd0aHMuCi0gKgotICogeERpdnMgYW5kIHlEaXZzIHBvaW50IHRvIGFycmF5cyBvZiBob3Jpem9udGFsIGFuZCB2ZXJ0aWNhbCBwaXhlbAotICogaW5kaWNlcy4gIFRoZSBmaXJzdCBwYWlyIG9mIERpdnMgKGluIGVpdGhlciBhcnJheSkgaW5kaWNhdGUgdGhlCi0gKiBzdGFydGluZyBhbmQgZW5kaW5nIHBvaW50cyBvZiB0aGUgZmlyc3Qgc3RyZXRjaGFibGUgc2VnbWVudCBpbiB0aGF0Ci0gKiBheGlzLiBUaGUgbmV4dCBwYWlyIHNwZWNpZmllcyB0aGUgbmV4dCBzdHJldGNoYWJsZSBzZWdtZW50LCBldGMuIFNvCi0gKiBpbiB0aGUgYWJvdmUgZXhhbXBsZSB4RGl2WzBdIGFuZCB4RGl2WzFdIHNwZWNpZnkgdGhlIGhvcml6b250YWwKLSAqIGNvb3JkaW5hdGVzIGZvciB0aGUgcmVnaW9ucyBsYWJlbGVkIDEsIDUgYW5kIDkuICB4RGl2WzJdIGFuZAotICogeERpdlszXSBzcGVjaWZ5IHRoZSBjb29yZGluYXRlcyBmb3IgcmVnaW9ucyAzLCA3IGFuZCAxMS4gTm90ZSB0aGF0Ci0gKiB0aGUgbGVmdG1vc3Qgc2xpY2VzIGFsd2F5cyBzdGFydCBhdCB4PTAgYW5kIHRoZSByaWdodG1vc3Qgc2xpY2VzCi0gKiBhbHdheXMgZW5kIGF0IHRoZSBlbmQgb2YgdGhlIGltYWdlLiBTbywgZm9yIGV4YW1wbGUsIHRoZSByZWdpb25zIDAsCi0gKiA0IGFuZCA4ICh3aGljaCBhcmUgZml4ZWQgYWxvbmcgdGhlIFggYXhpcykgc3RhcnQgYXQgeCB2YWx1ZSAwIGFuZAotICogZ28gdG8geERpdlswXSBhbWQgc2xpY2VzIDIsIDYgYW5kIDEwIHN0YXJ0IGF0IHhEaXZbMV0gYW5kIGVuZCBhdAotICogeERpdlsyXS4KLSAqCi0gKiBUaGUgYXJyYXkgcG9pbnRlZCB0byBieSB0aGUgY29sb3JzIGZpZWxkIGxpc3RzIGNvbnRhaW5zIGhpbnRzIGZvcgotICogZWFjaCBvZiB0aGUgcmVnaW9ucy4gIFRoZXkgYXJlIG9yZGVyZWQgYWNjb3JkaW5nIGxlZnQtdG8tcmlnaHQgYW5kCi0gKiB0b3AtdG8tYm90dG9tIGFzIGluZGljYXRlZCBhYm92ZS4gRm9yIGVhY2ggc2VnbWVudCB0aGF0IGlzIGEgc29saWQKLSAqIGNvbG9yIHRoZSBhcnJheSBlbnRyeSB3aWxsIGNvbnRhaW4gdGhhdCBjb2xvciB2YWx1ZTsgb3RoZXJ3aXNlIGl0Ci0gKiB3aWxsIGNvbnRhaW4gTk9fQ09MT1IuICBTZWdtZW50cyB0aGF0IGFyZSBjb21wbGV0ZWx5IHRyYW5zcGFyZW50Ci0gKiB3aWxsIGFsd2F5cyBoYXZlIHRoZSB2YWx1ZSBUUkFOU1BBUkVOVF9DT0xPUi4KLSAqCi0gKiBUaGUgUE5HIGNodW5rIHR5cGUgaXMgIm5wVGMiLgotICovCi1zdHJ1Y3QgUmVzX3BuZ185cGF0Y2gKLXsKLSAgICBSZXNfcG5nXzlwYXRjaCgpIDogd2FzRGVzZXJpYWxpemVkKGZhbHNlKSwgeERpdnMoTlVMTCksCi0gICAgICAgICAgICAgICAgICAgICAgIHlEaXZzKE5VTEwpLCBjb2xvcnMoTlVMTCkgeyB9Ci0KLSAgICBpbnQ4X3Qgd2FzRGVzZXJpYWxpemVkOwotICAgIGludDhfdCBudW1YRGl2czsKLSAgICBpbnQ4X3QgbnVtWURpdnM7Ci0gICAgaW50OF90IG51bUNvbG9yczsKLQotICAgIC8vIFRoZXNlIHRlbGwgd2hlcmUgdGhlIG5leHQgc2VjdGlvbiBvZiBhIHBhdGNoIHN0YXJ0cy4KLSAgICAvLyBGb3IgZXhhbXBsZSwgdGhlIGZpcnN0IHBhdGNoIGluY2x1ZGVzIHRoZSBwaXhlbHMgZnJvbQotICAgIC8vIDAgdG8geERpdnNbMF0tMSBhbmQgdGhlIHNlY29uZCBwYXRjaCBpbmNsdWRlcyB0aGUgcGl4ZWxzCi0gICAgLy8gZnJvbSB4RGl2c1swXSB0byB4RGl2c1sxXS0xLgotICAgIC8vIE5vdGU6IGFsbG9jYXRpb24vZnJlZSBvZiB0aGVzZSBwb2ludGVycyBpcyBsZWZ0IHRvIHRoZSBjYWxsZXIuCi0gICAgaW50MzJfdCogeERpdnM7Ci0gICAgaW50MzJfdCogeURpdnM7Ci0KLSAgICBpbnQzMl90IHBhZGRpbmdMZWZ0LCBwYWRkaW5nUmlnaHQ7Ci0gICAgaW50MzJfdCBwYWRkaW5nVG9wLCBwYWRkaW5nQm90dG9tOwotCi0gICAgZW51bSB7Ci0gICAgICAgIC8vIFRoZSA5IHBhdGNoIHNlZ21lbnQgaXMgbm90IGEgc29saWQgY29sb3IuCi0gICAgICAgIE5PX0NPTE9SID0gMHgwMDAwMDAwMSwKLQotICAgICAgICAvLyBUaGUgOSBwYXRjaCBzZWdtZW50IGlzIGNvbXBsZXRlbHkgdHJhbnNwYXJlbnQuCi0gICAgICAgIFRSQU5TUEFSRU5UX0NPTE9SID0gMHgwMDAwMDAwMAotICAgIH07Ci0gICAgLy8gTm90ZTogYWxsb2NhdGlvbi9mcmVlIG9mIHRoaXMgcG9pbnRlciBpcyBsZWZ0IHRvIHRoZSBjYWxsZXIuCi0gICAgdWludDMyX3QqIGNvbG9yczsKLQotICAgIC8vIENvbnZlcnQgZGF0YSBmcm9tIGRldmljZSByZXByZXNlbnRhdGlvbiB0byBQTkcgZmlsZSByZXByZXNlbnRhdGlvbi4KLSAgICB2b2lkIGRldmljZVRvRmlsZSgpOwotICAgIC8vIENvbnZlcnQgZGF0YSBmcm9tIFBORyBmaWxlIHJlcHJlc2VudGF0aW9uIHRvIGRldmljZSByZXByZXNlbnRhdGlvbi4KLSAgICB2b2lkIGZpbGVUb0RldmljZSgpOwotICAgIC8vIFNlcmlhbGl6ZS9NYXJzaGFsbCB0aGUgcGF0Y2ggZGF0YSBpbnRvIGEgbmV3bHkgbWFsbG9jLWVkIGJsb2NrCi0gICAgdm9pZCogc2VyaWFsaXplKCk7Ci0gICAgLy8gU2VyaWFsaXplL01hcnNoYWxsIHRoZSBwYXRjaCBkYXRhCi0gICAgdm9pZCBzZXJpYWxpemUodm9pZCogb3V0RGF0YSk7Ci0gICAgLy8gRGVzZXJpYWxpemUvVW5tYXJzaGFsbCB0aGUgcGF0Y2ggZGF0YQotICAgIHN0YXRpYyBSZXNfcG5nXzlwYXRjaCogZGVzZXJpYWxpemUoY29uc3Qgdm9pZCogZGF0YSk7Ci0gICAgLy8gQ29tcHV0ZSB0aGUgc2l6ZSBvZiB0aGUgc2VyaWFsaXplZCBkYXRhIHN0cnVjdHVyZQotICAgIHNpemVfdCBzZXJpYWxpemVkU2l6ZSgpOwotfTsKLQotLyoqICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCi0gKiAgQmFzZSBUeXBlcwotICoKLSAqICBUaGVzZSBhcmUgc3RhbmRhcmQgdHlwZXMgdGhhdCBhcmUgc2hhcmVkIGJldHdlZW4gbXVsdGlwbGUgc3BlY2lmaWMKLSAqICByZXNvdXJjZSB0eXBlcy4KLSAqCi0gKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiogKi8KLQotLyoqCi0gKiBIZWFkZXIgdGhhdCBhcHBlYXJzIGF0IHRoZSBmcm9udCBvZiBldmVyeSBkYXRhIGNodW5rIGluIGEgcmVzb3VyY2UuCi0gKi8KLXN0cnVjdCBSZXNDaHVua19oZWFkZXIKLXsKLSAgICAvLyBUeXBlIGlkZW50aWZpZXIgZm9yIHRoaXMgY2h1bmsuICBUaGUgbWVhbmluZyBvZiB0aGlzIHZhbHVlIGRlcGVuZHMKLSAgICAvLyBvbiB0aGUgY29udGFpbmluZyBjaHVuay4KLSAgICB1aW50MTZfdCB0eXBlOwotCi0gICAgLy8gU2l6ZSBvZiB0aGUgY2h1bmsgaGVhZGVyIChpbiBieXRlcykuICBBZGRpbmcgdGhpcyB2YWx1ZSB0bwotICAgIC8vIHRoZSBhZGRyZXNzIG9mIHRoZSBjaHVuayBhbGxvd3MgeW91IHRvIGZpbmQgaXRzIGFzc29jaWF0ZWQgZGF0YQotICAgIC8vIChpZiBhbnkpLgotICAgIHVpbnQxNl90IGhlYWRlclNpemU7Ci0KLSAgICAvLyBUb3RhbCBzaXplIG9mIHRoaXMgY2h1bmsgKGluIGJ5dGVzKS4gIFRoaXMgaXMgdGhlIGNodW5rU2l6ZSBwbHVzCi0gICAgLy8gdGhlIHNpemUgb2YgYW55IGRhdGEgYXNzb2NpYXRlZCB3aXRoIHRoZSBjaHVuay4gIEFkZGluZyB0aGlzIHZhbHVlCi0gICAgLy8gdG8gdGhlIGNodW5rIGFsbG93cyB5b3UgdG8gY29tcGxldGVseSBza2lwIGl0cyBjb250ZW50cyAoaW5jbHVkaW5nCi0gICAgLy8gYW55IGNoaWxkIGNodW5rcykuICBJZiB0aGlzIHZhbHVlIGlzIHRoZSBzYW1lIGFzIGNodW5rU2l6ZSwgdGhlcmUgaXMKLSAgICAvLyBubyBkYXRhIGFzc29jaWF0ZWQgd2l0aCB0aGUgY2h1bmsuCi0gICAgdWludDMyX3Qgc2l6ZTsKLX07Ci0KLWVudW0gewotICAgIFJFU19OVUxMX1RZUEUgICAgICAgICAgICAgICA9IDB4MDAwMCwKLSAgICBSRVNfU1RSSU5HX1BPT0xfVFlQRSAgICAgICAgPSAweDAwMDEsCi0gICAgUkVTX1RBQkxFX1RZUEUgICAgICAgICAgICAgID0gMHgwMDAyLAotICAgIFJFU19YTUxfVFlQRSAgICAgICAgICAgICAgICA9IDB4MDAwMywKLQotICAgIC8vIENodW5rIHR5cGVzIGluIFJFU19YTUxfVFlQRQotICAgIFJFU19YTUxfRklSU1RfQ0hVTktfVFlQRSAgICA9IDB4MDEwMCwKLSAgICBSRVNfWE1MX1NUQVJUX05BTUVTUEFDRV9UWVBFPSAweDAxMDAsCi0gICAgUkVTX1hNTF9FTkRfTkFNRVNQQUNFX1RZUEUgID0gMHgwMTAxLAotICAgIFJFU19YTUxfU1RBUlRfRUxFTUVOVF9UWVBFICA9IDB4MDEwMiwKLSAgICBSRVNfWE1MX0VORF9FTEVNRU5UX1RZUEUgICAgPSAweDAxMDMsCi0gICAgUkVTX1hNTF9DREFUQV9UWVBFICAgICAgICAgID0gMHgwMTA0LAotICAgIFJFU19YTUxfTEFTVF9DSFVOS19UWVBFICAgICA9IDB4MDE3ZiwKLSAgICAvLyBUaGlzIGNvbnRhaW5zIGEgdWludDMyX3QgYXJyYXkgbWFwcGluZyBzdHJpbmdzIGluIHRoZSBzdHJpbmcKLSAgICAvLyBwb29sIGJhY2sgdG8gcmVzb3VyY2UgaWRlbnRpZmllcnMuICBJdCBpcyBvcHRpb25hbC4KLSAgICBSRVNfWE1MX1JFU09VUkNFX01BUF9UWVBFICAgPSAweDAxODAsCi0KLSAgICAvLyBDaHVuayB0eXBlcyBpbiBSRVNfVEFCTEVfVFlQRQotICAgIFJFU19UQUJMRV9QQUNLQUdFX1RZUEUgICAgICA9IDB4MDIwMCwKLSAgICBSRVNfVEFCTEVfVFlQRV9UWVBFICAgICAgICAgPSAweDAyMDEsCi0gICAgUkVTX1RBQkxFX1RZUEVfU1BFQ19UWVBFICAgID0gMHgwMjAyCi19OwotCi0vKioKLSAqIE1hY3JvcyBmb3IgYnVpbGRpbmcvc3BsaXR0aW5nIHJlc291cmNlIGlkZW50aWZpZXJzLgotICovCi0jZGVmaW5lIFJlc19WQUxJRElEKHJlc2lkKSAocmVzaWQgIT0gMCkKLSNkZWZpbmUgUmVzX0NIRUNLSUQocmVzaWQpICgocmVzaWQmMHhGRkZGMDAwMCkgIT0gMCkKLSNkZWZpbmUgUmVzX01BS0VJRChwYWNrYWdlLCB0eXBlLCBlbnRyeSkgXAotICAgICgoKHBhY2thZ2UrMSk8PDI0KSB8ICgoKHR5cGUrMSkmMHhGRik8PDE2KSB8IChlbnRyeSYweEZGRkYpKQotI2RlZmluZSBSZXNfR0VUUEFDS0FHRShpZCkgKChpZD4+MjQpLTEpCi0jZGVmaW5lIFJlc19HRVRUWVBFKGlkKSAoKChpZD4+MTYpJjB4RkYpLTEpCi0jZGVmaW5lIFJlc19HRVRFTlRSWShpZCkgKGlkJjB4RkZGRikKLQotI2RlZmluZSBSZXNfSU5URVJOQUxJRChyZXNpZCkgKChyZXNpZCYweEZGRkYwMDAwKSAhPSAwICYmIChyZXNpZCYweEZGMDAwMCkgPT0gMCkKLSNkZWZpbmUgUmVzX01BS0VJTlRFUk5BTChlbnRyeSkgKDB4MDEwMDAwMDAgfCAoZW50cnkmMHhGRkZGKSkKLSNkZWZpbmUgUmVzX01BS0VBUlJBWShlbnRyeSkgKDB4MDIwMDAwMDAgfCAoZW50cnkmMHhGRkZGKSkKLQotI2RlZmluZSBSZXNfTUFYUEFDS0FHRSAyNTUKLQotLyoqCi0gKiBSZXByZXNlbnRhdGlvbiBvZiBhIHZhbHVlIGluIGEgcmVzb3VyY2UsIHN1cHBseWluZyB0eXBlCi0gKiBpbmZvcm1hdGlvbi4KLSAqLwotc3RydWN0IFJlc192YWx1ZQotewotICAgIC8vIE51bWJlciBvZiBieXRlcyBpbiB0aGlzIHN0cnVjdHVyZS4KLSAgICB1aW50MTZfdCBzaXplOwotCi0gICAgLy8gQWx3YXlzIHNldCB0byAwLgotICAgIHVpbnQ4X3QgcmVzMDsKLSAgICAgICAgCi0gICAgLy8gVHlwZSBvZiB0aGUgZGF0YSB2YWx1ZS4KLSAgICBlbnVtIHsKLSAgICAgICAgLy8gQ29udGFpbnMgbm8gZGF0YS4KLSAgICAgICAgVFlQRV9OVUxMID0gMHgwMCwKLSAgICAgICAgLy8gVGhlICdkYXRhJyBob2xkcyBhIFJlc1RhYmxlX3JlZiwgYSByZWZlcmVuY2UgdG8gYW5vdGhlciByZXNvdXJjZQotICAgICAgICAvLyB0YWJsZSBlbnRyeS4KLSAgICAgICAgVFlQRV9SRUZFUkVOQ0UgPSAweDAxLAotICAgICAgICAvLyBUaGUgJ2RhdGEnIGhvbGRzIGFuIGF0dHJpYnV0ZSByZXNvdXJjZSBpZGVudGlmaWVyLgotICAgICAgICBUWVBFX0FUVFJJQlVURSA9IDB4MDIsCi0gICAgICAgIC8vIFRoZSAnZGF0YScgaG9sZHMgYW4gaW5kZXggaW50byB0aGUgY29udGFpbmluZyByZXNvdXJjZSB0YWJsZSdzCi0gICAgICAgIC8vIGdsb2JhbCB2YWx1ZSBzdHJpbmcgcG9vbC4KLSAgICAgICAgVFlQRV9TVFJJTkcgPSAweDAzLAotICAgICAgICAvLyBUaGUgJ2RhdGEnIGhvbGRzIGEgc2luZ2xlLXByZWNpc2lvbiBmbG9hdGluZyBwb2ludCBudW1iZXIuCi0gICAgICAgIFRZUEVfRkxPQVQgPSAweDA0LAotICAgICAgICAvLyBUaGUgJ2RhdGEnIGhvbGRzIGEgY29tcGxleCBudW1iZXIgZW5jb2RpbmcgYSBkaW1lbnNpb24gdmFsdWUsCi0gICAgICAgIC8vIHN1Y2ggYXMgIjEwMGluIi4KLSAgICAgICAgVFlQRV9ESU1FTlNJT04gPSAweDA1LAotICAgICAgICAvLyBUaGUgJ2RhdGEnIGhvbGRzIGEgY29tcGxleCBudW1iZXIgZW5jb2RpbmcgYSBmcmFjdGlvbiBvZiBhCi0gICAgICAgIC8vIGNvbnRhaW5lci4KLSAgICAgICAgVFlQRV9GUkFDVElPTiA9IDB4MDYsCi0KLSAgICAgICAgLy8gQmVnaW5uaW5nIG9mIGludGVnZXIgZmxhdm9ycy4uLgotICAgICAgICBUWVBFX0ZJUlNUX0lOVCA9IDB4MTAsCi0KLSAgICAgICAgLy8gVGhlICdkYXRhJyBpcyBhIHJhdyBpbnRlZ2VyIHZhbHVlIG9mIHRoZSBmb3JtIG4uLm4uCi0gICAgICAgIFRZUEVfSU5UX0RFQyA9IDB4MTAsCi0gICAgICAgIC8vIFRoZSAnZGF0YScgaXMgYSByYXcgaW50ZWdlciB2YWx1ZSBvZiB0aGUgZm9ybSAweG4uLm4uCi0gICAgICAgIFRZUEVfSU5UX0hFWCA9IDB4MTEsCi0gICAgICAgIC8vIFRoZSAnZGF0YScgaXMgZWl0aGVyIDAgb3IgMSwgZm9yIGlucHV0ICJmYWxzZSIgb3IgInRydWUiIHJlc3BlY3RpdmVseS4KLSAgICAgICAgVFlQRV9JTlRfQk9PTEVBTiA9IDB4MTIsCi0KLSAgICAgICAgLy8gQmVnaW5uaW5nIG9mIGNvbG9yIGludGVnZXIgZmxhdm9ycy4uLgotICAgICAgICBUWVBFX0ZJUlNUX0NPTE9SX0lOVCA9IDB4MWMsCi0KLSAgICAgICAgLy8gVGhlICdkYXRhJyBpcyBhIHJhdyBpbnRlZ2VyIHZhbHVlIG9mIHRoZSBmb3JtICNhYXJyZ2diYi4KLSAgICAgICAgVFlQRV9JTlRfQ09MT1JfQVJHQjggPSAweDFjLAotICAgICAgICAvLyBUaGUgJ2RhdGEnIGlzIGEgcmF3IGludGVnZXIgdmFsdWUgb2YgdGhlIGZvcm0gI3JyZ2diYi4KLSAgICAgICAgVFlQRV9JTlRfQ09MT1JfUkdCOCA9IDB4MWQsCi0gICAgICAgIC8vIFRoZSAnZGF0YScgaXMgYSByYXcgaW50ZWdlciB2YWx1ZSBvZiB0aGUgZm9ybSAjYXJnYi4KLSAgICAgICAgVFlQRV9JTlRfQ09MT1JfQVJHQjQgPSAweDFlLAotICAgICAgICAvLyBUaGUgJ2RhdGEnIGlzIGEgcmF3IGludGVnZXIgdmFsdWUgb2YgdGhlIGZvcm0gI3JnYi4KLSAgICAgICAgVFlQRV9JTlRfQ09MT1JfUkdCNCA9IDB4MWYsCi0KLSAgICAgICAgLy8gLi4uZW5kIG9mIGludGVnZXIgZmxhdm9ycy4KLSAgICAgICAgVFlQRV9MQVNUX0NPTE9SX0lOVCA9IDB4MWYsCi0KLSAgICAgICAgLy8gLi4uZW5kIG9mIGludGVnZXIgZmxhdm9ycy4KLSAgICAgICAgVFlQRV9MQVNUX0lOVCA9IDB4MWYKLSAgICB9OwotICAgIHVpbnQ4X3QgZGF0YVR5cGU7Ci0KLSAgICAvLyBTdHJ1Y3R1cmUgb2YgY29tcGxleCBkYXRhIHZhbHVlcyAoVFlQRV9VTklUIGFuZCBUWVBFX0ZSQUNUSU9OKQotICAgIGVudW0gewotICAgICAgICAvLyBXaGVyZSB0aGUgdW5pdCB0eXBlIGluZm9ybWF0aW9uIGlzLiAgVGhpcyBnaXZlcyB1cyAxNiBwb3NzaWJsZQotICAgICAgICAvLyB0eXBlcywgYXMgZGVmaW5lZCBiZWxvdy4KLSAgICAgICAgQ09NUExFWF9VTklUX1NISUZUID0gMCwKLSAgICAgICAgQ09NUExFWF9VTklUX01BU0sgPSAweGYsCi0KLSAgICAgICAgLy8gVFlQRV9ESU1FTlNJT046IFZhbHVlIGlzIHJhdyBwaXhlbHMuCi0gICAgICAgIENPTVBMRVhfVU5JVF9QWCA9IDAsCi0gICAgICAgIC8vIFRZUEVfRElNRU5TSU9OOiBWYWx1ZSBpcyBEZXZpY2UgSW5kZXBlbmRlbnQgUGl4ZWxzLgotICAgICAgICBDT01QTEVYX1VOSVRfRElQID0gMSwKLSAgICAgICAgLy8gVFlQRV9ESU1FTlNJT046IFZhbHVlIGlzIGEgU2NhbGVkIGRldmljZSBpbmRlcGVuZGVudCBQaXhlbHMuCi0gICAgICAgIENPTVBMRVhfVU5JVF9TUCA9IDIsCi0gICAgICAgIC8vIFRZUEVfRElNRU5TSU9OOiBWYWx1ZSBpcyBpbiBwb2ludHMuCi0gICAgICAgIENPTVBMRVhfVU5JVF9QVCA9IDMsCi0gICAgICAgIC8vIFRZUEVfRElNRU5TSU9OOiBWYWx1ZSBpcyBpbiBpbmNoZXMuCi0gICAgICAgIENPTVBMRVhfVU5JVF9JTiA9IDQsCi0gICAgICAgIC8vIFRZUEVfRElNRU5TSU9OOiBWYWx1ZSBpcyBpbiBtaWxsaW1ldGVycy4KLSAgICAgICAgQ09NUExFWF9VTklUX01NID0gNSwKLQotICAgICAgICAvLyBUWVBFX0ZSQUNUSU9OOiBBIGJhc2ljIGZyYWN0aW9uIG9mIHRoZSBvdmVyYWxsIHNpemUuCi0gICAgICAgIENPTVBMRVhfVU5JVF9GUkFDVElPTiA9IDAsCi0gICAgICAgIC8vIFRZUEVfRlJBQ1RJT046IEEgZnJhY3Rpb24gb2YgdGhlIHBhcmVudCBzaXplLgotICAgICAgICBDT01QTEVYX1VOSVRfRlJBQ1RJT05fUEFSRU5UID0gMSwKLQotICAgICAgICAvLyBXaGVyZSB0aGUgcmFkaXggaW5mb3JtYXRpb24gaXMsIHRlbGxpbmcgd2hlcmUgdGhlIGRlY2ltYWwgcGxhY2UKLSAgICAgICAgLy8gYXBwZWFycyBpbiB0aGUgbWFudGlzc2EuICBUaGlzIGdpdmUgdXMgNCBwb3NzaWJsZSBmaXhlZCBwb2ludAotICAgICAgICAvLyByZXByZXNlbnRhdGlvbnMgYXMgZGVmaW5lZCBiZWxvdy4KLSAgICAgICAgQ09NUExFWF9SQURJWF9TSElGVCA9IDQsCi0gICAgICAgIENPTVBMRVhfUkFESVhfTUFTSyA9IDB4MywKLQotICAgICAgICAvLyBUaGUgbWFudGlzc2EgaXMgYW4gaW50ZWdyYWwgbnVtYmVyIC0tIGkuZS4sIDB4bm5ubm5uLjAKLSAgICAgICAgQ09NUExFWF9SQURJWF8yM3AwID0gMCwKLSAgICAgICAgLy8gVGhlIG1hbnRpc3NhIG1hZ25pdHVkZSBpcyAxNiBiaXRzIC0tIGkuZSwgMHhubm5uLm5uCi0gICAgICAgIENPTVBMRVhfUkFESVhfMTZwNyA9IDEsCi0gICAgICAgIC8vIFRoZSBtYW50aXNzYSBtYWduaXR1ZGUgaXMgOCBiaXRzIC0tIGkuZSwgMHhubi5ubm5uCi0gICAgICAgIENPTVBMRVhfUkFESVhfOHAxNSA9IDIsCi0gICAgICAgIC8vIFRoZSBtYW50aXNzYSBtYWduaXR1ZGUgaXMgMCBiaXRzIC0tIGkuZSwgMHgwLm5ubm5ubgotICAgICAgICBDT01QTEVYX1JBRElYXzBwMjMgPSAzLAotCi0gICAgICAgIC8vIFdoZXJlIHRoZSBhY3R1YWwgdmFsdWUgaXMuICBUaGlzIGdpdmVzIHVzIDIzIGJpdHMgb2YKLSAgICAgICAgLy8gcHJlY2lzaW9uLiAgVGhlIHRvcCBiaXQgaXMgdGhlIHNpZ24uCi0gICAgICAgIENPTVBMRVhfTUFOVElTU0FfU0hJRlQgPSA4LAotICAgICAgICBDT01QTEVYX01BTlRJU1NBX01BU0sgPSAweGZmZmZmZgotICAgIH07Ci0KLSAgICAvLyBUaGUgZGF0YSBmb3IgdGhpcyBpdGVtLCBhcyBpbnRlcnByZXRlZCBhY2NvcmRpbmcgdG8gZGF0YVR5cGUuCi0gICAgdWludDMyX3QgZGF0YTsKLQotICAgIHZvaWQgY29weUZyb21fZHRvaChjb25zdCBSZXNfdmFsdWUmIHNyYyk7Ci19OwotCi0vKioKLSAqICBUaGlzIGlzIGEgcmVmZXJlbmNlIHRvIGEgdW5pcXVlIGVudHJ5IChhIFJlc1RhYmxlX2VudHJ5IHN0cnVjdHVyZSkKLSAqICBpbiBhIHJlc291cmNlIHRhYmxlLiAgVGhlIHZhbHVlIGlzIHN0cnVjdHVyZWQgYXM6IDB4cHB0dGVlZWUsCi0gKiAgd2hlcmUgcHAgaXMgdGhlIHBhY2thZ2UgaW5kZXgsIHR0IGlzIHRoZSB0eXBlIGluZGV4IGluIHRoYXQKLSAqICBwYWNrYWdlLCBhbmQgZWVlZSBpcyB0aGUgZW50cnkgaW5kZXggaW4gdGhhdCB0eXBlLiAgVGhlIHBhY2thZ2UKLSAqICBhbmQgdHlwZSB2YWx1ZXMgc3RhcnQgYXQgMSBmb3IgdGhlIGZpcnN0IGl0ZW0sIHRvIGhlbHAgY2F0Y2ggY2FzZXMKLSAqICB3aGVyZSB0aGV5IGhhdmUgbm90IGJlZW4gc3VwcGxpZWQuCi0gKi8KLXN0cnVjdCBSZXNUYWJsZV9yZWYKLXsKLSAgICB1aW50MzJfdCBpZGVudDsKLX07Ci0KLS8qKgotICogUmVmZXJlbmNlIHRvIGEgc3RyaW5nIGluIGEgc3RyaW5nIHBvb2wuCi0gKi8KLXN0cnVjdCBSZXNTdHJpbmdQb29sX3JlZgotewotICAgIC8vIEluZGV4IGludG8gdGhlIHN0cmluZyBwb29sIHRhYmxlICh1aW50MzJfdC1vZmZzZXQgZnJvbSB0aGUgaW5kaWNlcwotICAgIC8vIGltbWVkaWF0ZWx5IGFmdGVyIFJlc1N0cmluZ1Bvb2xfaGVhZGVyKSBhdCB3aGljaCB0byBmaW5kIHRoZSBsb2NhdGlvbgotICAgIC8vIG9mIHRoZSBzdHJpbmcgZGF0YSBpbiB0aGUgcG9vbC4KLSAgICB1aW50MzJfdCBpbmRleDsKLX07Ci0KLS8qKiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgotICogIFN0cmluZyBQb29sCi0gKgotICogIEEgc2V0IG9mIHN0cmluZ3MgdGhhdCBjYW4gYmUgcmVmZXJlbmNlcyBieSBvdGhlcnMgdGhyb3VnaCBhCi0gKiAgUmVzU3RyaW5nUG9vbF9yZWYuCi0gKgotICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqICovCi0KLS8qKgotICogRGVmaW5pdGlvbiBmb3IgYSBwb29sIG9mIHN0cmluZ3MuICBUaGUgZGF0YSBvZiB0aGlzIGNodW5rIGlzIGFuCi0gKiBhcnJheSBvZiB1aW50MzJfdCBwcm92aWRpbmcgaW5kaWNlcyBpbnRvIHRoZSBwb29sLCByZWxhdGl2ZSB0bwotICogc3RyaW5nc1N0YXJ0LiAgQXQgc3RyaW5nc1N0YXJ0IGFyZSBhbGwgb2YgdGhlIFVURi0xNiBzdHJpbmdzCi0gKiBjb25jYXRlbmF0ZWQgdG9nZXRoZXI7IGVhY2ggc3RhcnRzIHdpdGggYSB1aW50MTZfdCBvZiB0aGUgc3RyaW5nJ3MKLSAqIGxlbmd0aCBhbmQgZWFjaCBlbmRzIHdpdGggYSAweDAwMDAgdGVybWluYXRvci4gIElmIGEgc3RyaW5nIGlzID4KLSAqIDMyNzY3IGNoYXJhY3RlcnMsIHRoZSBoaWdoIGJpdCBvZiB0aGUgbGVuZ3RoIGlzIHNldCBtZWFuaW5nIHRvIHRha2UKLSAqIHRob3NlIDE1IGJpdHMgYXMgYSBoaWdoIHdvcmQgYW5kIGl0IHdpbGwgYmUgZm9sbG93ZWQgYnkgYW5vdGhlcgotICogdWludDE2X3QgY29udGFpbmluZyB0aGUgbG93IHdvcmQuCi0gKgotICogSWYgc3R5bGVDb3VudCBpcyBub3QgemVybywgdGhlbiBpbW1lZGlhdGVseSBmb2xsb3dpbmcgdGhlIGFycmF5IG9mCi0gKiB1aW50MzJfdCBpbmRpY2VzIGludG8gdGhlIHN0cmluZyB0YWJsZSBpcyBhbm90aGVyIGFycmF5IG9mIGluZGljZXMKLSAqIGludG8gYSBzdHlsZSB0YWJsZSBzdGFydGluZyBhdCBzdHlsZXNTdGFydC4gIEVhY2ggZW50cnkgaW4gdGhlCi0gKiBzdHlsZSB0YWJsZSBpcyBhbiBhcnJheSBvZiBSZXNTdHJpbmdQb29sX3NwYW4gc3RydWN0dXJlcy4KLSAqLwotc3RydWN0IFJlc1N0cmluZ1Bvb2xfaGVhZGVyCi17Ci0gICAgc3RydWN0IFJlc0NodW5rX2hlYWRlciBoZWFkZXI7Ci0KLSAgICAvLyBOdW1iZXIgb2Ygc3RyaW5ncyBpbiB0aGlzIHBvb2wgKG51bWJlciBvZiB1aW50MzJfdCBpbmRpY2VzIHRoYXQgZm9sbG93Ci0gICAgLy8gaW4gdGhlIGRhdGEpLgotICAgIHVpbnQzMl90IHN0cmluZ0NvdW50OwotCi0gICAgLy8gTnVtYmVyIG9mIHN0eWxlIHNwYW4gYXJyYXlzIGluIHRoZSBwb29sIChudW1iZXIgb2YgdWludDMyX3QgaW5kaWNlcwotICAgIC8vIGZvbGxvdyB0aGUgc3RyaW5nIGluZGljZXMpLgotICAgIHVpbnQzMl90IHN0eWxlQ291bnQ7Ci0KLSAgICAvLyBGbGFncy4KLSAgICBlbnVtIHsKLSAgICAgICAgLy8gSWYgc2V0LCB0aGUgc3RyaW5nIGluZGV4IGlzIHNvcnRlZCBieSB0aGUgc3RyaW5nIHZhbHVlcyAoYmFzZWQKLSAgICAgICAgLy8gb24gc3RyY21wMTYoKSkuCi0gICAgICAgIFNPUlRFRF9GTEFHID0gMTw8MAotICAgIH07Ci0gICAgdWludDMyX3QgZmxhZ3M7Ci0KLSAgICAvLyBJbmRleCBmcm9tIGhlYWRlciBvZiB0aGUgc3RyaW5nIGRhdGEuCi0gICAgdWludDMyX3Qgc3RyaW5nc1N0YXJ0OwotCi0gICAgLy8gSW5kZXggZnJvbSBoZWFkZXIgb2YgdGhlIHN0eWxlIGRhdGEuCi0gICAgdWludDMyX3Qgc3R5bGVzU3RhcnQ7Ci19OwotCi0vKioKLSAqIFRoaXMgc3RydWN0dXJlIGRlZmluZXMgYSBzcGFuIG9mIHN0eWxlIGluZm9ybWF0aW9uIGFzc29jaWF0ZWQgd2l0aAotICogYSBzdHJpbmcgaW4gdGhlIHBvb2wuCi0gKi8KLXN0cnVjdCBSZXNTdHJpbmdQb29sX3NwYW4KLXsKLSAgICBlbnVtIHsKLSAgICAgICAgRU5EID0gMHhGRkZGRkZGRgotICAgIH07Ci0KLSAgICAvLyBUaGlzIGlzIHRoZSBuYW1lIG9mIHRoZSBzcGFuIC0tIHRoYXQgaXMsIHRoZSBuYW1lIG9mIHRoZSBYTUwKLSAgICAvLyB0YWcgdGhhdCBkZWZpbmVkIGl0LiAgVGhlIHNwZWNpYWwgdmFsdWUgRU5EICgweEZGRkZGRkZGKSBpbmRpY2F0ZXMKLSAgICAvLyB0aGUgZW5kIG9mIGFuIGFycmF5IG9mIHNwYW5zLgotICAgIFJlc1N0cmluZ1Bvb2xfcmVmIG5hbWU7Ci0KLSAgICAvLyBUaGUgcmFuZ2Ugb2YgY2hhcmFjdGVycyBpbiB0aGUgc3RyaW5nIHRoYXQgdGhpcyBzcGFuIGFwcGxpZXMgdG8uCi0gICAgdWludDMyX3QgZmlyc3RDaGFyLCBsYXN0Q2hhcjsKLX07Ci0KLS8qKgotICogQ29udmVuaWVuY2UgY2xhc3MgZm9yIGFjY2Vzc2luZyBkYXRhIGluIGEgUmVzU3RyaW5nUG9vbCByZXNvdXJjZS4KLSAqLwotY2xhc3MgUmVzU3RyaW5nUG9vbAotewotcHVibGljOgotICAgIFJlc1N0cmluZ1Bvb2woKTsKLSAgICBSZXNTdHJpbmdQb29sKGNvbnN0IHZvaWQqIGRhdGEsIHNpemVfdCBzaXplLCBib29sIGNvcHlEYXRhPWZhbHNlKTsKLSAgICB+UmVzU3RyaW5nUG9vbCgpOwotCi0gICAgc3RhdHVzX3Qgc2V0VG8oY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsIGJvb2wgY29weURhdGE9ZmFsc2UpOwotCi0gICAgc3RhdHVzX3QgZ2V0RXJyb3IoKSBjb25zdDsKLQotICAgIHZvaWQgdW5pbml0KCk7Ci0KLSAgICBpbmxpbmUgY29uc3QgY2hhcjE2X3QqIHN0cmluZ0F0KGNvbnN0IFJlc1N0cmluZ1Bvb2xfcmVmJiByZWYsIHNpemVfdCogb3V0TGVuKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiBzdHJpbmdBdChyZWYuaW5kZXgsIG91dExlbik7Ci0gICAgfQotICAgIGNvbnN0IGNoYXIxNl90KiBzdHJpbmdBdChzaXplX3QgaWR4LCBzaXplX3QqIG91dExlbikgY29uc3Q7Ci0KLSAgICBjb25zdCBSZXNTdHJpbmdQb29sX3NwYW4qIHN0eWxlQXQoY29uc3QgUmVzU3RyaW5nUG9vbF9yZWYmIHJlZikgY29uc3Q7Ci0gICAgY29uc3QgUmVzU3RyaW5nUG9vbF9zcGFuKiBzdHlsZUF0KHNpemVfdCBpZHgpIGNvbnN0OwotCi0gICAgc3NpemVfdCBpbmRleE9mU3RyaW5nKGNvbnN0IGNoYXIxNl90KiBzdHIsIHNpemVfdCBzdHJMZW4pIGNvbnN0OwotCi0gICAgc2l6ZV90IHNpemUoKSBjb25zdDsKLQotcHJpdmF0ZToKLSAgICBzdGF0dXNfdCAgICAgICAgICAgICAgICAgICAgbUVycm9yOwotICAgIHZvaWQqICAgICAgICAgICAgICAgICAgICAgICBtT3duZWREYXRhOwotICAgIGNvbnN0IFJlc1N0cmluZ1Bvb2xfaGVhZGVyKiBtSGVhZGVyOwotICAgIHNpemVfdCAgICAgICAgICAgICAgICAgICAgICBtU2l6ZTsKLSAgICBjb25zdCB1aW50MzJfdCogICAgICAgICAgICAgbUVudHJpZXM7Ci0gICAgY29uc3QgdWludDMyX3QqICAgICAgICAgICAgIG1FbnRyeVN0eWxlczsKLSAgICBjb25zdCBjaGFyMTZfdCogICAgICAgICAgICAgbVN0cmluZ3M7Ci0gICAgdWludDMyX3QgICAgICAgICAgICAgICAgICAgIG1TdHJpbmdQb29sU2l6ZTsgICAgLy8gbnVtYmVyIG9mIHVpbnQxNl90Ci0gICAgY29uc3QgdWludDMyX3QqICAgICAgICAgICAgIG1TdHlsZXM7Ci0gICAgdWludDMyX3QgICAgICAgICAgICAgICAgICAgIG1TdHlsZVBvb2xTaXplOyAgICAvLyBudW1iZXIgb2YgdWludDMyX3QKLX07Ci0KLS8qKiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgotICogIFhNTCBUcmVlCi0gKgotICogIEJpbmFyeSByZXByZXNlbnRhdGlvbiBvZiBhbiBYTUwgZG9jdW1lbnQuICBUaGlzIGlzIGRlc2lnbmVkIHRvCi0gKiAgZXhwcmVzcyBldmVyeXRoaW5nIGluIGFuIFhNTCBkb2N1bWVudCwgaW4gYSBmb3JtIHRoYXQgaXMgbXVjaAotICogIGVhc2llciB0byBwYXJzZSBvbiB0aGUgZGV2aWNlLgotICoKLSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiAqLwotCi0vKioKLSAqIFhNTCB0cmVlIGhlYWRlci4gIFRoaXMgYXBwZWFycyBhdCB0aGUgZnJvbnQgb2YgYW4gWE1MIHRyZWUsCi0gKiBkZXNjcmliaW5nIGl0cyBjb250ZW50LiAgSXQgaXMgZm9sbG93ZWQgYnkgYSBmbGF0IGFycmF5IG9mCi0gKiBSZXNYTUxUcmVlX25vZGUgc3RydWN0dXJlczsgdGhlIGhpZXJhcmNoeSBvZiB0aGUgWE1MIGRvY3VtZW50Ci0gKiBpcyBkZXNjcmliZWQgYnkgdGhlIG9jY3VycmFuY2Ugb2YgUkVTX1hNTF9TVEFSVF9FTEVNRU5UX1RZUEUKLSAqIGFuZCBjb3JyZXNwb25kaW5nIFJFU19YTUxfRU5EX0VMRU1FTlRfVFlQRSBub2RlcyBpbiB0aGUgYXJyYXkuCi0gKi8KLXN0cnVjdCBSZXNYTUxUcmVlX2hlYWRlcgotewotICAgIHN0cnVjdCBSZXNDaHVua19oZWFkZXIgaGVhZGVyOwotfTsKLQotLyoqCi0gKiBCYXNpYyBYTUwgdHJlZSBub2RlLiAgQSBzaW5nbGUgaXRlbSBpbiB0aGUgWE1MIGRvY3VtZW50LiAgRXh0ZW5kZWQgaW5mbwotICogYWJvdXQgdGhlIG5vZGUgY2FuIGJlIGZvdW5kIGFmdGVyIGhlYWRlci5oZWFkZXJTaXplLgotICovCi1zdHJ1Y3QgUmVzWE1MVHJlZV9ub2RlCi17Ci0gICAgc3RydWN0IFJlc0NodW5rX2hlYWRlciBoZWFkZXI7Ci0KLSAgICAvLyBMaW5lIG51bWJlciBpbiBvcmlnaW5hbCBzb3VyY2UgZmlsZSBhdCB3aGljaCB0aGlzIGVsZW1lbnQgYXBwZWFyZWQuCi0gICAgdWludDMyX3QgbGluZU51bWJlcjsKLQotICAgIC8vIE9wdGlvbmFsIFhNTCBjb21tZW50IHRoYXQgd2FzIGFzc29jaWF0ZWQgd2l0aCB0aGlzIGVsZW1lbnQ7IC0xIGlmIG5vbmUuCi0gICAgc3RydWN0IFJlc1N0cmluZ1Bvb2xfcmVmIGNvbW1lbnQ7Ci19OwotCi0vKioKLSAqIEV4dGVuZGVkIFhNTCB0cmVlIG5vZGUgZm9yIENEQVRBIHRhZ3MgLS0gaW5jbHVkZXMgdGhlIENEQVRBIHN0cmluZy4KLSAqIEFwcGVhcnMgaGVhZGVyLmhlYWRlclNpemUgYnl0ZXMgYWZ0ZXIgYSBSZXNYTUxUcmVlX25vZGUuCi0gKi8KLXN0cnVjdCBSZXNYTUxUcmVlX2NkYXRhRXh0Ci17Ci0gICAgLy8gVGhlIHJhdyBDREFUQSBjaGFyYWN0ZXIgZGF0YS4KLSAgICBzdHJ1Y3QgUmVzU3RyaW5nUG9vbF9yZWYgZGF0YTsKLSAgICAKLSAgICAvLyBUaGUgdHlwZWQgdmFsdWUgb2YgdGhlIGNoYXJhY3RlciBkYXRhIGlmIHRoaXMgaXMgYSBDREFUQSBub2RlLgotICAgIHN0cnVjdCBSZXNfdmFsdWUgdHlwZWREYXRhOwotfTsKLQotLyoqCi0gKiBFeHRlbmRlZCBYTUwgdHJlZSBub2RlIGZvciBuYW1lc3BhY2Ugc3RhcnQvZW5kIG5vZGVzLgotICogQXBwZWFycyBoZWFkZXIuaGVhZGVyU2l6ZSBieXRlcyBhZnRlciBhIFJlc1hNTFRyZWVfbm9kZS4KLSAqLwotc3RydWN0IFJlc1hNTFRyZWVfbmFtZXNwYWNlRXh0Ci17Ci0gICAgLy8gVGhlIHByZWZpeCBvZiB0aGUgbmFtZXNwYWNlLgotICAgIHN0cnVjdCBSZXNTdHJpbmdQb29sX3JlZiBwcmVmaXg7Ci0gICAgCi0gICAgLy8gVGhlIFVSSSBvZiB0aGUgbmFtZXNwYWNlLgotICAgIHN0cnVjdCBSZXNTdHJpbmdQb29sX3JlZiB1cmk7Ci19OwotCi0vKioKLSAqIEV4dGVuZGVkIFhNTCB0cmVlIG5vZGUgZm9yIGVsZW1lbnQgc3RhcnQvZW5kIG5vZGVzLgotICogQXBwZWFycyBoZWFkZXIuaGVhZGVyU2l6ZSBieXRlcyBhZnRlciBhIFJlc1hNTFRyZWVfbm9kZS4KLSAqLwotc3RydWN0IFJlc1hNTFRyZWVfZW5kRWxlbWVudEV4dAotewotICAgIC8vIFN0cmluZyBvZiB0aGUgZnVsbCBuYW1lc3BhY2Ugb2YgdGhpcyBlbGVtZW50LgotICAgIHN0cnVjdCBSZXNTdHJpbmdQb29sX3JlZiBuczsKLSAgICAKLSAgICAvLyBTdHJpbmcgbmFtZSBvZiB0aGlzIG5vZGUgaWYgaXQgaXMgYW4gRUxFTUVOVDsgdGhlIHJhdwotICAgIC8vIGNoYXJhY3RlciBkYXRhIGlmIHRoaXMgaXMgYSBDREFUQSBub2RlLgotICAgIHN0cnVjdCBSZXNTdHJpbmdQb29sX3JlZiBuYW1lOwotfTsKLQotLyoqCi0gKiBFeHRlbmRlZCBYTUwgdHJlZSBub2RlIGZvciBzdGFydCB0YWdzIC0tIGluY2x1ZGVzIGF0dHJpYnV0ZQotICogaW5mb3JtYXRpb24uCi0gKiBBcHBlYXJzIGhlYWRlci5oZWFkZXJTaXplIGJ5dGVzIGFmdGVyIGEgUmVzWE1MVHJlZV9ub2RlLgotICovCi1zdHJ1Y3QgUmVzWE1MVHJlZV9hdHRyRXh0Ci17Ci0gICAgLy8gU3RyaW5nIG9mIHRoZSBmdWxsIG5hbWVzcGFjZSBvZiB0aGlzIGVsZW1lbnQuCi0gICAgc3RydWN0IFJlc1N0cmluZ1Bvb2xfcmVmIG5zOwotICAgIAotICAgIC8vIFN0cmluZyBuYW1lIG9mIHRoaXMgbm9kZSBpZiBpdCBpcyBhbiBFTEVNRU5UOyB0aGUgcmF3Ci0gICAgLy8gY2hhcmFjdGVyIGRhdGEgaWYgdGhpcyBpcyBhIENEQVRBIG5vZGUuCi0gICAgc3RydWN0IFJlc1N0cmluZ1Bvb2xfcmVmIG5hbWU7Ci0gICAgCi0gICAgLy8gQnl0ZSBvZmZzZXQgZnJvbSB0aGUgc3RhcnQgb2YgdGhpcyBzdHJ1Y3R1cmUgd2hlcmUgdGhlIGF0dHJpYnV0ZXMgc3RhcnQuCi0gICAgdWludDE2X3QgYXR0cmlidXRlU3RhcnQ7Ci0gICAgCi0gICAgLy8gU2l6ZSBvZiB0aGUgUmVzWE1MVHJlZV9hdHRyaWJ1dGUgc3RydWN0dXJlcyB0aGF0IGZvbGxvdy4KLSAgICB1aW50MTZfdCBhdHRyaWJ1dGVTaXplOwotICAgIAotICAgIC8vIE51bWJlciBvZiBhdHRyaWJ1dGVzIGFzc29jaWF0ZWQgd2l0aCBhbiBFTEVNRU5ULiAgVGhlc2UgYXJlCi0gICAgLy8gYXZhaWxhYmxlIGFzIGFuIGFycmF5IG9mIFJlc1hNTFRyZWVfYXR0cmlidXRlIHN0cnVjdHVyZXMKLSAgICAvLyBpbW1lZGlhdGVseSBmb2xsb3dpbmcgdGhpcyBub2RlLgotICAgIHVpbnQxNl90IGF0dHJpYnV0ZUNvdW50OwotICAgIAotICAgIC8vIEluZGV4ICgxLWJhc2VkKSBvZiB0aGUgImlkIiBhdHRyaWJ1dGUuIDAgaWYgbm9uZS4KLSAgICB1aW50MTZfdCBpZEluZGV4OwotICAgIAotICAgIC8vIEluZGV4ICgxLWJhc2VkKSBvZiB0aGUgImNsYXNzIiBhdHRyaWJ1dGUuIDAgaWYgbm9uZS4KLSAgICB1aW50MTZfdCBjbGFzc0luZGV4OwotICAgIAotICAgIC8vIEluZGV4ICgxLWJhc2VkKSBvZiB0aGUgInN0eWxlIiBhdHRyaWJ1dGUuIDAgaWYgbm9uZS4KLSAgICB1aW50MTZfdCBzdHlsZUluZGV4OwotfTsKLQotc3RydWN0IFJlc1hNTFRyZWVfYXR0cmlidXRlCi17Ci0gICAgLy8gTmFtZXNwYWNlIG9mIHRoaXMgYXR0cmlidXRlLgotICAgIHN0cnVjdCBSZXNTdHJpbmdQb29sX3JlZiBuczsKLSAgICAKLSAgICAvLyBOYW1lIG9mIHRoaXMgYXR0cmlidXRlLgotICAgIHN0cnVjdCBSZXNTdHJpbmdQb29sX3JlZiBuYW1lOwotCi0gICAgLy8gVGhlIG9yaWdpbmFsIHJhdyBzdHJpbmcgdmFsdWUgb2YgdGhpcyBhdHRyaWJ1dGUuCi0gICAgc3RydWN0IFJlc1N0cmluZ1Bvb2xfcmVmIHJhd1ZhbHVlOwotICAgIAotICAgIC8vIFByb2Nlc3Nlc2QgdHlwZWQgdmFsdWUgb2YgdGhpcyBhdHRyaWJ1dGUuCi0gICAgc3RydWN0IFJlc192YWx1ZSB0eXBlZFZhbHVlOwotfTsKLQotY2xhc3MgUmVzWE1MVHJlZTsKLQotY2xhc3MgUmVzWE1MUGFyc2VyCi17Ci1wdWJsaWM6Ci0gICAgUmVzWE1MUGFyc2VyKGNvbnN0IFJlc1hNTFRyZWUmIHRyZWUpOwotCi0gICAgZW51bSBldmVudF9jb2RlX3QgewotICAgICAgICBCQURfRE9DVU1FTlQgPSAtMSwKLSAgICAgICAgU1RBUlRfRE9DVU1FTlQgPSAwLAotICAgICAgICBFTkRfRE9DVU1FTlQgPSAxLAotICAgICAgICAKLSAgICAgICAgRklSU1RfQ0hVTktfQ09ERSA9IFJFU19YTUxfRklSU1RfQ0hVTktfVFlQRSwgCi0gICAgICAgIAotICAgICAgICBTVEFSVF9OQU1FU1BBQ0UgPSBSRVNfWE1MX1NUQVJUX05BTUVTUEFDRV9UWVBFLAotICAgICAgICBFTkRfTkFNRVNQQUNFID0gUkVTX1hNTF9FTkRfTkFNRVNQQUNFX1RZUEUsCi0gICAgICAgIFNUQVJUX1RBRyA9IFJFU19YTUxfU1RBUlRfRUxFTUVOVF9UWVBFLAotICAgICAgICBFTkRfVEFHID0gUkVTX1hNTF9FTkRfRUxFTUVOVF9UWVBFLAotICAgICAgICBURVhUID0gUkVTX1hNTF9DREFUQV9UWVBFCi0gICAgfTsKLQotICAgIHN0cnVjdCBSZXNYTUxQb3NpdGlvbgotICAgIHsKLSAgICAgICAgZXZlbnRfY29kZV90ICAgICAgICAgICAgICAgIGV2ZW50Q29kZTsKLSAgICAgICAgY29uc3QgUmVzWE1MVHJlZV9ub2RlKiAgICAgIGN1ck5vZGU7Ci0gICAgICAgIGNvbnN0IHZvaWQqICAgICAgICAgICAgICAgICBjdXJFeHQ7Ci0gICAgfTsKLQotICAgIHZvaWQgcmVzdGFydCgpOwotCi0gICAgZXZlbnRfY29kZV90IGdldEV2ZW50VHlwZSgpIGNvbnN0OwotICAgIC8vIE5vdGUsIHVubGlrZSBYbWxQdWxsUGFyc2VyLCB0aGUgZmlyc3QgY2FsbCB0byBuZXh0KCkgd2lsbCByZXR1cm4KLSAgICAvLyBTVEFSVF9UQUcgb2YgdGhlIGZpcnN0IGVsZW1lbnQuCi0gICAgZXZlbnRfY29kZV90IG5leHQoKTsKLQotICAgIC8vIFRoZXNlIGFyZSBhdmFpbGFibGUgZm9yIGFsbCBub2RlczoKLSAgICBjb25zdCBpbnQzMl90IGdldENvbW1lbnRJRCgpIGNvbnN0OwotICAgIGNvbnN0IHVpbnQxNl90KiBnZXRDb21tZW50KHNpemVfdCogb3V0TGVuKSBjb25zdDsKLSAgICBjb25zdCB1aW50MzJfdCBnZXRMaW5lTnVtYmVyKCkgY29uc3Q7Ci0gICAgCi0gICAgLy8gVGhpcyBpcyBhdmFpbGFibGUgZm9yIFRFWFQ6Ci0gICAgY29uc3QgaW50MzJfdCBnZXRUZXh0SUQoKSBjb25zdDsKLSAgICBjb25zdCB1aW50MTZfdCogZ2V0VGV4dChzaXplX3QqIG91dExlbikgY29uc3Q7Ci0gICAgc3NpemVfdCBnZXRUZXh0VmFsdWUoUmVzX3ZhbHVlKiBvdXRWYWx1ZSkgY29uc3Q7Ci0gICAgCi0gICAgLy8gVGhlc2UgYXJlIGF2YWlsYWJsZSBmb3IgU1RBUlRfTkFNRVNQQUNFIGFuZCBFTkRfTkFNRVNQQUNFOgotICAgIGNvbnN0IGludDMyX3QgZ2V0TmFtZXNwYWNlUHJlZml4SUQoKSBjb25zdDsKLSAgICBjb25zdCB1aW50MTZfdCogZ2V0TmFtZXNwYWNlUHJlZml4KHNpemVfdCogb3V0TGVuKSBjb25zdDsKLSAgICBjb25zdCBpbnQzMl90IGdldE5hbWVzcGFjZVVyaUlEKCkgY29uc3Q7Ci0gICAgY29uc3QgdWludDE2X3QqIGdldE5hbWVzcGFjZVVyaShzaXplX3QqIG91dExlbikgY29uc3Q7Ci0gICAgCi0gICAgLy8gVGhlc2UgYXJlIGF2YWlsYWJsZSBmb3IgU1RBUlRfVEFHIGFuZCBFTkRfVEFHOgotICAgIGNvbnN0IGludDMyX3QgZ2V0RWxlbWVudE5hbWVzcGFjZUlEKCkgY29uc3Q7Ci0gICAgY29uc3QgdWludDE2X3QqIGdldEVsZW1lbnROYW1lc3BhY2Uoc2l6ZV90KiBvdXRMZW4pIGNvbnN0OwotICAgIGNvbnN0IGludDMyX3QgZ2V0RWxlbWVudE5hbWVJRCgpIGNvbnN0OwotICAgIGNvbnN0IHVpbnQxNl90KiBnZXRFbGVtZW50TmFtZShzaXplX3QqIG91dExlbikgY29uc3Q7Ci0gICAgCi0gICAgLy8gUmVtYWluaW5nIG1ldGhvZHMgYXJlIGZvciByZXRyaWV2aW5nIGluZm9ybWF0aW9uIGFib3V0IGF0dHJpYnV0ZXMKLSAgICAvLyBhc3NvY2lhdGVkIHdpdGggYSBTVEFSVF9UQUc6Ci0gICAgCi0gICAgc2l6ZV90IGdldEF0dHJpYnV0ZUNvdW50KCkgY29uc3Q7Ci0gICAgCi0gICAgLy8gUmV0dXJucyAtMSBpZiBubyBuYW1lc3BhY2UsIC0yIGlmIGlkeCBvdXQgb2YgcmFuZ2UuCi0gICAgY29uc3QgaW50MzJfdCBnZXRBdHRyaWJ1dGVOYW1lc3BhY2VJRChzaXplX3QgaWR4KSBjb25zdDsKLSAgICBjb25zdCB1aW50MTZfdCogZ2V0QXR0cmlidXRlTmFtZXNwYWNlKHNpemVfdCBpZHgsIHNpemVfdCogb3V0TGVuKSBjb25zdDsKLSAgICAKLSAgICBjb25zdCBpbnQzMl90IGdldEF0dHJpYnV0ZU5hbWVJRChzaXplX3QgaWR4KSBjb25zdDsKLSAgICBjb25zdCB1aW50MTZfdCogZ2V0QXR0cmlidXRlTmFtZShzaXplX3QgaWR4LCBzaXplX3QqIG91dExlbikgY29uc3Q7Ci0gICAgY29uc3QgdWludDMyX3QgZ2V0QXR0cmlidXRlTmFtZVJlc0lEKHNpemVfdCBpZHgpIGNvbnN0OwotICAgIAotICAgIGNvbnN0IGludDMyX3QgZ2V0QXR0cmlidXRlVmFsdWVTdHJpbmdJRChzaXplX3QgaWR4KSBjb25zdDsKLSAgICBjb25zdCB1aW50MTZfdCogZ2V0QXR0cmlidXRlU3RyaW5nVmFsdWUoc2l6ZV90IGlkeCwgc2l6ZV90KiBvdXRMZW4pIGNvbnN0OwotICAgIAotICAgIGludDMyX3QgZ2V0QXR0cmlidXRlRGF0YVR5cGUoc2l6ZV90IGlkeCkgY29uc3Q7Ci0gICAgaW50MzJfdCBnZXRBdHRyaWJ1dGVEYXRhKHNpemVfdCBpZHgpIGNvbnN0OwotICAgIHNzaXplX3QgZ2V0QXR0cmlidXRlVmFsdWUoc2l6ZV90IGlkeCwgUmVzX3ZhbHVlKiBvdXRWYWx1ZSkgY29uc3Q7Ci0KLSAgICBzc2l6ZV90IGluZGV4T2ZBdHRyaWJ1dGUoY29uc3QgY2hhciogbnMsIGNvbnN0IGNoYXIqIGF0dHIpIGNvbnN0OwotICAgIHNzaXplX3QgaW5kZXhPZkF0dHJpYnV0ZShjb25zdCBjaGFyMTZfdCogbnMsIHNpemVfdCBuc0xlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY2hhcjE2X3QqIGF0dHIsIHNpemVfdCBhdHRyTGVuKSBjb25zdDsKLQotICAgIHNzaXplX3QgaW5kZXhPZklEKCkgY29uc3Q7Ci0gICAgc3NpemVfdCBpbmRleE9mQ2xhc3MoKSBjb25zdDsKLSAgICBzc2l6ZV90IGluZGV4T2ZTdHlsZSgpIGNvbnN0OwotCi0gICAgdm9pZCBnZXRQb3NpdGlvbihSZXNYTUxQb3NpdGlvbiogcG9zKSBjb25zdDsKLSAgICB2b2lkIHNldFBvc2l0aW9uKGNvbnN0IFJlc1hNTFBvc2l0aW9uJiBwb3MpOwotCi1wcml2YXRlOgotICAgIGZyaWVuZCBjbGFzcyBSZXNYTUxUcmVlOwotICAgIAotICAgIGV2ZW50X2NvZGVfdCBuZXh0Tm9kZSgpOwotCi0gICAgY29uc3QgUmVzWE1MVHJlZSYgICAgICAgICAgIG1UcmVlOwotICAgIGV2ZW50X2NvZGVfdCAgICAgICAgICAgICAgICBtRXZlbnRDb2RlOwotICAgIGNvbnN0IFJlc1hNTFRyZWVfbm9kZSogICAgICBtQ3VyTm9kZTsKLSAgICBjb25zdCB2b2lkKiAgICAgICAgICAgICAgICAgbUN1ckV4dDsKLX07Ci0KLS8qKgotICogQ29udmVuaWVuY2UgY2xhc3MgZm9yIGFjY2Vzc2luZyBkYXRhIGluIGEgUmVzWE1MVHJlZSByZXNvdXJjZS4KLSAqLwotY2xhc3MgUmVzWE1MVHJlZSA6IHB1YmxpYyBSZXNYTUxQYXJzZXIKLXsKLXB1YmxpYzoKLSAgICBSZXNYTUxUcmVlKCk7Ci0gICAgUmVzWE1MVHJlZShjb25zdCB2b2lkKiBkYXRhLCBzaXplX3Qgc2l6ZSwgYm9vbCBjb3B5RGF0YT1mYWxzZSk7Ci0gICAgflJlc1hNTFRyZWUoKTsKLQotICAgIHN0YXR1c190IHNldFRvKGNvbnN0IHZvaWQqIGRhdGEsIHNpemVfdCBzaXplLCBib29sIGNvcHlEYXRhPWZhbHNlKTsKLQotICAgIHN0YXR1c190IGdldEVycm9yKCkgY29uc3Q7Ci0KLSAgICB2b2lkIHVuaW5pdCgpOwotCi0gICAgY29uc3QgUmVzU3RyaW5nUG9vbCYgZ2V0U3RyaW5ncygpIGNvbnN0OwotCi1wcml2YXRlOgotICAgIGZyaWVuZCBjbGFzcyBSZXNYTUxQYXJzZXI7Ci0KLSAgICBzdGF0dXNfdCB2YWxpZGF0ZU5vZGUoY29uc3QgUmVzWE1MVHJlZV9ub2RlKiBub2RlKSBjb25zdDsKLQotICAgIHN0YXR1c190ICAgICAgICAgICAgICAgICAgICBtRXJyb3I7Ci0gICAgdm9pZCogICAgICAgICAgICAgICAgICAgICAgIG1Pd25lZERhdGE7Ci0gICAgY29uc3QgUmVzWE1MVHJlZV9oZWFkZXIqICAgIG1IZWFkZXI7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICAgICAgICAgIG1TaXplOwotICAgIGNvbnN0IHVpbnQ4X3QqICAgICAgICAgICAgICBtRGF0YUVuZDsKLSAgICBSZXNTdHJpbmdQb29sICAgICAgICAgICAgICAgbVN0cmluZ3M7Ci0gICAgY29uc3QgdWludDMyX3QqICAgICAgICAgICAgIG1SZXNJZHM7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICAgICAgICAgIG1OdW1SZXNJZHM7Ci0gICAgY29uc3QgUmVzWE1MVHJlZV9ub2RlKiAgICAgIG1Sb290Tm9kZTsKLSAgICBjb25zdCB2b2lkKiAgICAgICAgICAgICAgICAgbVJvb3RFeHQ7Ci0gICAgZXZlbnRfY29kZV90ICAgICAgICAgICAgICAgIG1Sb290Q29kZTsKLX07Ci0KLS8qKiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgotICogIFJFU09VUkNFIFRBQkxFCi0gKgotICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqICovCi0KLS8qKgotICogSGVhZGVyIGZvciBhIHJlc291cmNlIHRhYmxlLiAgSXRzIGRhdGEgY29udGFpbnMgYSBzZXJpZXMgb2YKLSAqIGFkZGl0aW9uYWwgY2h1bmtzOgotICogICAqIEEgUmVzU3RyaW5nUG9vbF9oZWFkZXIgY29udGFpbmluZyBhbGwgdGFibGUgdmFsdWVzLgotICogICAqIE9uZSBvciBtb3JlIFJlc1RhYmxlX3BhY2thZ2UgY2h1bmtzLgotICoKLSAqIFNwZWNpZmljIGVudHJpZXMgd2l0aGluIGEgcmVzb3VyY2UgdGFibGUgY2FuIGJlIHVuaXF1ZWx5IGlkZW50aWZpZWQKLSAqIHdpdGggYSBzaW5nbGUgaW50ZWdlciBhcyBkZWZpbmVkIGJ5IHRoZSBSZXNUYWJsZV9yZWYgc3RydWN0dXJlLgotICovCi1zdHJ1Y3QgUmVzVGFibGVfaGVhZGVyCi17Ci0gICAgc3RydWN0IFJlc0NodW5rX2hlYWRlciBoZWFkZXI7Ci0KLSAgICAvLyBUaGUgbnVtYmVyIG9mIFJlc1RhYmxlX3BhY2thZ2Ugc3RydWN0dXJlcy4KLSAgICB1aW50MzJfdCBwYWNrYWdlQ291bnQ7Ci19OwotCi0vKioKLSAqIEEgY29sbGVjdGlvbiBvZiByZXNvdXJjZSBkYXRhIHR5cGVzIHdpdGhpbiBhIHBhY2thZ2UuICBGb2xsb3dlZCBieQotICogb25lIG9yIG1vcmUgUmVzVGFibGVfdHlwZSBhbmQgUmVzVGFibGVfdHlwZVNwZWMgc3RydWN0dXJlcyBjb250YWluaW5nIHRoZQotICogZW50cnkgdmFsdWVzIGZvciBlYWNoIHJlc291cmNlIHR5cGUuCi0gKi8KLXN0cnVjdCBSZXNUYWJsZV9wYWNrYWdlCi17Ci0gICAgc3RydWN0IFJlc0NodW5rX2hlYWRlciBoZWFkZXI7Ci0KLSAgICAvLyBJZiB0aGlzIGlzIGEgYmFzZSBwYWNrYWdlLCBpdHMgSUQuICBQYWNrYWdlIElEcyBzdGFydAotICAgIC8vIGF0IDEgKGNvcnJlc3BvbmRpbmcgdG8gdGhlIHZhbHVlIG9mIHRoZSBwYWNrYWdlIGJpdHMgaW4gYQotICAgIC8vIHJlc291cmNlIGlkZW50aWZpZXIpLiAgMCBtZWFucyB0aGlzIGlzIG5vdCBhIGJhc2UgcGFja2FnZS4KLSAgICB1aW50MzJfdCBpZDsKLQotICAgIC8vIEFjdHVhbCBuYW1lIG9mIHRoaXMgcGFja2FnZSwgXDAtdGVybWluYXRlZC4KLSAgICBjaGFyMTZfdCBuYW1lWzEyOF07Ci0KLSAgICAvLyBPZmZzZXQgdG8gYSBSZXNTdHJpbmdQb29sX2hlYWRlciBkZWZpbmluZyB0aGUgcmVzb3VyY2UKLSAgICAvLyB0eXBlIHN5bWJvbCB0YWJsZS4gIElmIHplcm8sIHRoaXMgcGFja2FnZSBpcyBpbmhlcml0aW5nIGZyb20KLSAgICAvLyBhbm90aGVyIGJhc2UgcGFja2FnZSAob3ZlcnJpZGluZyBzcGVjaWZpYyB2YWx1ZXMgaW4gaXQpLgotICAgIHVpbnQzMl90IHR5cGVTdHJpbmdzOwotCi0gICAgLy8gTGFzdCBpbmRleCBpbnRvIHR5cGVTdHJpbmdzIHRoYXQgaXMgZm9yIHB1YmxpYyB1c2UgYnkgb3RoZXJzLgotICAgIHVpbnQzMl90IGxhc3RQdWJsaWNUeXBlOwotCi0gICAgLy8gT2Zmc2V0IHRvIGEgUmVzU3RyaW5nUG9vbF9oZWFkZXIgZGVmaW5pbmcgdGhlIHJlc291cmNlCi0gICAgLy8ga2V5IHN5bWJvbCB0YWJsZS4gIElmIHplcm8sIHRoaXMgcGFja2FnZSBpcyBpbmhlcml0aW5nIGZyb20KLSAgICAvLyBhbm90aGVyIGJhc2UgcGFja2FnZSAob3ZlcnJpZGluZyBzcGVjaWZpYyB2YWx1ZXMgaW4gaXQpLgotICAgIHVpbnQzMl90IGtleVN0cmluZ3M7Ci0KLSAgICAvLyBMYXN0IGluZGV4IGludG8ga2V5U3RyaW5ncyB0aGF0IGlzIGZvciBwdWJsaWMgdXNlIGJ5IG90aGVycy4KLSAgICB1aW50MzJfdCBsYXN0UHVibGljS2V5OwotfTsKLQotLyoqCi0gKiBEZXNjcmliZXMgYSBwYXJ0aWN1bGFyIHJlc291cmNlIGNvbmZpZ3VyYXRpb24uCi0gKi8KLXN0cnVjdCBSZXNUYWJsZV9jb25maWcKLXsKLSAgICAvLyBOdW1iZXIgb2YgYnl0ZXMgaW4gdGhpcyBzdHJ1Y3R1cmUuCi0gICAgdWludDMyX3Qgc2l6ZTsKLSAgICAKLSAgICB1bmlvbiB7Ci0gICAgICAgIHN0cnVjdCB7Ci0gICAgICAgICAgICAvLyBNb2JpbGUgY291bnRyeSBjb2RlIChmcm9tIFNJTSkuICAwIG1lYW5zICJhbnkiLgotICAgICAgICAgICAgdWludDE2X3QgbWNjOwotICAgICAgICAgICAgLy8gTW9iaWxlIG5ldHdvcmsgY29kZSAoZnJvbSBTSU0pLiAgMCBtZWFucyAiYW55Ii4KLSAgICAgICAgICAgIHVpbnQxNl90IG1uYzsKLSAgICAgICAgfTsKLSAgICAgICAgdWludDMyX3QgaW1zaTsKLSAgICB9OwotICAgIAotICAgIHVuaW9uIHsKLSAgICAgICAgc3RydWN0IHsKLSAgICAgICAgICAgIC8vIFwwXDAgbWVhbnMgImFueSIuICBPdGhlcndpc2UsIGVuLCBmciwgZXRjLgotICAgICAgICAgICAgY2hhciBsYW5ndWFnZVsyXTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8gXDBcMCBtZWFucyAiYW55Ii4gIE90aGVyd2lzZSwgVVMsIENBLCBldGMuCi0gICAgICAgICAgICBjaGFyIGNvdW50cnlbMl07Ci0gICAgICAgIH07Ci0gICAgICAgIHVpbnQzMl90IGxvY2FsZTsKLSAgICB9OwotICAgIAotICAgIGVudW0gewotICAgICAgICBPUklFTlRBVElPTl9BTlkgID0gMHgwMDAwLAotICAgICAgICBPUklFTlRBVElPTl9QT1JUID0gMHgwMDAxLAotICAgICAgICBPUklFTlRBVElPTl9MQU5EID0gMHgwMDAyLAotICAgICAgICBPUklFTlRBVElPTl9TUVVBUkUgPSAweDAwMDIsCi0gICAgfTsKLSAgICAKLSAgICBlbnVtIHsKLSAgICAgICAgVE9VQ0hTQ1JFRU5fQU5ZICA9IDB4MDAwMCwKLSAgICAgICAgVE9VQ0hTQ1JFRU5fTk9UT1VDSCAgPSAweDAwMDEsCi0gICAgICAgIFRPVUNIU0NSRUVOX1NUWUxVUyAgPSAweDAwMDIsCi0gICAgICAgIFRPVUNIU0NSRUVOX0ZJTkdFUiAgPSAweDAwMDMsCi0gICAgfTsKLSAgICAKLSAgICBlbnVtIHsKLSAgICAgICAgREVOU0lUWV9BTlkgPSAwCi0gICAgfTsKLSAgICAKLSAgICB1bmlvbiB7Ci0gICAgICAgIHN0cnVjdCB7Ci0gICAgICAgICAgICB1aW50OF90IG9yaWVudGF0aW9uOwotICAgICAgICAgICAgdWludDhfdCB0b3VjaHNjcmVlbjsKLSAgICAgICAgICAgIHVpbnQxNl90IGRlbnNpdHk7Ci0gICAgICAgIH07Ci0gICAgICAgIHVpbnQzMl90IHNjcmVlblR5cGU7Ci0gICAgfTsKLSAgICAKLSAgICBlbnVtIHsKLSAgICAgICAgS0VZQk9BUkRfQU5ZICA9IDB4MDAwMCwKLSAgICAgICAgS0VZQk9BUkRfTk9LRVlTICA9IDB4MDAwMSwKLSAgICAgICAgS0VZQk9BUkRfUVdFUlRZICA9IDB4MDAwMiwKLSAgICAgICAgS0VZQk9BUkRfMTJLRVkgID0gMHgwMDAzLAotICAgIH07Ci0gICAgCi0gICAgZW51bSB7Ci0gICAgICAgIE5BVklHQVRJT05fQU5ZICA9IDB4MDAwMCwKLSAgICAgICAgTkFWSUdBVElPTl9OT05BViAgPSAweDAwMDEsCi0gICAgICAgIE5BVklHQVRJT05fRFBBRCAgPSAweDAwMDIsCi0gICAgICAgIE5BVklHQVRJT05fVFJBQ0tCQUxMICA9IDB4MDAwMywKLSAgICAgICAgTkFWSUdBVElPTl9XSEVFTCAgPSAweDAwMDQsCi0gICAgfTsKLSAgICAKLSAgICBlbnVtIHsKLSAgICAgICAgTUFTS19LRVlTSElEREVOID0gMHgwMDAzLAotICAgICAgICBTSElGVF9LRVlTSElEREVOID0gMCwKLSAgICAgICAgS0VZU0hJRERFTl9BTlkgPSAweDAwMDAsCi0gICAgICAgIEtFWVNISURERU5fTk8gPSAweDAwMDEsCi0gICAgICAgIEtFWVNISURERU5fWUVTID0gMHgwMDAyLAotICAgICAgICBLRVlTSElEREVOX1NPRlQgPSAweDAwMDMsCi0gICAgfTsKLSAgICAKLSAgICB1bmlvbiB7Ci0gICAgICAgIHN0cnVjdCB7Ci0gICAgICAgICAgICB1aW50OF90IGtleWJvYXJkOwotICAgICAgICAgICAgdWludDhfdCBuYXZpZ2F0aW9uOwotICAgICAgICAgICAgdWludDhfdCBpbnB1dEZsYWdzOwotICAgICAgICAgICAgdWludDhfdCBwYWQwOwotICAgICAgICB9OwotICAgICAgICB1aW50MzJfdCBpbnB1dDsKLSAgICB9OwotICAgIAotICAgIGVudW0gewotICAgICAgICBTQ1JFRU5XSURUSF9BTlkgPSAwCi0gICAgfTsKLSAgICAKLSAgICBlbnVtIHsKLSAgICAgICAgU0NSRUVOSEVJR0hUX0FOWSA9IDAKLSAgICB9OwotICAgIAotICAgIHVuaW9uIHsKLSAgICAgICAgc3RydWN0IHsKLSAgICAgICAgICAgIHVpbnQxNl90IHNjcmVlbldpZHRoOwotICAgICAgICAgICAgdWludDE2X3Qgc2NyZWVuSGVpZ2h0OwotICAgICAgICB9OwotICAgICAgICB1aW50MzJfdCBzY3JlZW5TaXplOwotICAgIH07Ci0gICAgCi0gICAgZW51bSB7Ci0gICAgICAgIFNES1ZFUlNJT05fQU5ZID0gMAotICAgIH07Ci0gICAgCi0gICAgZW51bSB7Ci0gICAgICAgIE1JTk9SVkVSU0lPTl9BTlkgPSAwCi0gICAgfTsKLSAgICAKLSAgICB1bmlvbiB7Ci0gICAgICAgIHN0cnVjdCB7Ci0gICAgICAgICAgICB1aW50MTZfdCBzZGtWZXJzaW9uOwotICAgICAgICAgICAgLy8gRm9yIG5vdyBtaW5vclZlcnNpb24gbXVzdCBhbHdheXMgYmUgMCEhISAgSXRzIG1lYW5pbmcKLSAgICAgICAgICAgIC8vIGlzIGN1cnJlbnRseSB1bmRlZmluZWQuCi0gICAgICAgICAgICB1aW50MTZfdCBtaW5vclZlcnNpb247Ci0gICAgICAgIH07Ci0gICAgICAgIHVpbnQzMl90IHZlcnNpb247Ci0gICAgfTsKLSAgICAKLSAgICBpbmxpbmUgdm9pZCBjb3B5RnJvbURldmljZU5vU3dhcChjb25zdCBSZXNUYWJsZV9jb25maWcmIG8pIHsKLSAgICAgICAgY29uc3Qgc2l6ZV90IHNpemUgPSBkdG9obChvLnNpemUpOwotICAgICAgICBpZiAoc2l6ZSA+PSBzaXplb2YoUmVzVGFibGVfY29uZmlnKSkgewotICAgICAgICAgICAgKnRoaXMgPSBvOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbWVtY3B5KHRoaXMsICZvLCBzaXplKTsKLSAgICAgICAgICAgIG1lbXNldCgoKHVpbnQ4X3QqKXRoaXMpK3NpemUsIDAsIHNpemVvZihSZXNUYWJsZV9jb25maWcpLXNpemUpOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIGlubGluZSB2b2lkIGNvcHlGcm9tRHRvSChjb25zdCBSZXNUYWJsZV9jb25maWcmIG8pIHsKLSAgICAgICAgY29weUZyb21EZXZpY2VOb1N3YXAobyk7Ci0gICAgICAgIHNpemUgPSBzaXplb2YoUmVzVGFibGVfY29uZmlnKTsKLSAgICAgICAgbWNjID0gZHRvaHMobWNjKTsKLSAgICAgICAgbW5jID0gZHRvaHMobW5jKTsKLSAgICAgICAgZGVuc2l0eSA9IGR0b2hzKGRlbnNpdHkpOwotICAgICAgICBzY3JlZW5XaWR0aCA9IGR0b2hzKHNjcmVlbldpZHRoKTsKLSAgICAgICAgc2NyZWVuSGVpZ2h0ID0gZHRvaHMoc2NyZWVuSGVpZ2h0KTsKLSAgICAgICAgc2RrVmVyc2lvbiA9IGR0b2hzKHNka1ZlcnNpb24pOwotICAgICAgICBtaW5vclZlcnNpb24gPSBkdG9ocyhtaW5vclZlcnNpb24pOwotICAgIH0KLSAgICAKLSAgICBpbmxpbmUgdm9pZCBzd2FwSHRvRCgpIHsKLSAgICAgICAgc2l6ZSA9IGh0b2RsKHNpemUpOwotICAgICAgICBtY2MgPSBodG9kcyhtY2MpOwotICAgICAgICBtbmMgPSBodG9kcyhtbmMpOwotICAgICAgICBkZW5zaXR5ID0gaHRvZHMoZGVuc2l0eSk7Ci0gICAgICAgIHNjcmVlbldpZHRoID0gaHRvZHMoc2NyZWVuV2lkdGgpOwotICAgICAgICBzY3JlZW5IZWlnaHQgPSBodG9kcyhzY3JlZW5IZWlnaHQpOwotICAgICAgICBzZGtWZXJzaW9uID0gaHRvZHMoc2RrVmVyc2lvbik7Ci0gICAgICAgIG1pbm9yVmVyc2lvbiA9IGh0b2RzKG1pbm9yVmVyc2lvbik7Ci0gICAgfQotICAgIAotICAgIGlubGluZSBpbnQgY29tcGFyZShjb25zdCBSZXNUYWJsZV9jb25maWcmIG8pIGNvbnN0IHsKLSAgICAgICAgaW50MzJfdCBkaWZmID0gKGludDMyX3QpKGltc2kgLSBvLmltc2kpOwotICAgICAgICBpZiAoZGlmZiAhPSAwKSByZXR1cm4gZGlmZjsKLSAgICAgICAgZGlmZiA9IChpbnQzMl90KShsb2NhbGUgLSBvLmxvY2FsZSk7Ci0gICAgICAgIGlmIChkaWZmICE9IDApIHJldHVybiBkaWZmOwotICAgICAgICBkaWZmID0gKGludDMyX3QpKHNjcmVlblR5cGUgLSBvLnNjcmVlblR5cGUpOwotICAgICAgICBpZiAoZGlmZiAhPSAwKSByZXR1cm4gZGlmZjsKLSAgICAgICAgZGlmZiA9IChpbnQzMl90KShpbnB1dCAtIG8uaW5wdXQpOwotICAgICAgICBpZiAoZGlmZiAhPSAwKSByZXR1cm4gZGlmZjsKLSAgICAgICAgZGlmZiA9IChpbnQzMl90KShzY3JlZW5TaXplIC0gby5zY3JlZW5TaXplKTsKLSAgICAgICAgaWYgKGRpZmYgIT0gMCkgcmV0dXJuIGRpZmY7Ci0gICAgICAgIGRpZmYgPSAoaW50MzJfdCkodmVyc2lvbiAtIG8udmVyc2lvbik7Ci0gICAgICAgIHJldHVybiAoaW50KWRpZmY7Ci0gICAgfQotICAgIAotICAgIC8vIEZsYWdzIGluZGljYXRpbmcgYSBzZXQgb2YgY29uZmlnIHZhbHVlcy4gIFRoZXNlIGZsYWcgY29uc3RhbnRzIG11c3QKLSAgICAvLyBtYXRjaCB0aGUgY29ycmVzcG9uZGluZyBvbmVzIGluIGFuZHJvaWQuY29udGVudC5wbS5BY3Rpdml0eUluZm8gYW5kCi0gICAgLy8gYXR0cnNfbWFuaWZlc3QueG1sLgotICAgIGVudW0gewotICAgICAgICBDT05GSUdfTUNDID0gMHgwMDAxLAotICAgICAgICBDT05GSUdfTU5DID0gMHgwMDAyLAotICAgICAgICBDT05GSUdfTE9DQUxFID0gMHgwMDA0LAotICAgICAgICBDT05GSUdfVE9VQ0hTQ1JFRU4gPSAweDAwMDgsCi0gICAgICAgIENPTkZJR19LRVlCT0FSRCA9IDB4MDAxMCwKLSAgICAgICAgQ09ORklHX0tFWUJPQVJEX0hJRERFTiA9IDB4MDAyMCwKLSAgICAgICAgQ09ORklHX05BVklHQVRJT04gPSAweDAwNDAsCi0gICAgICAgIENPTkZJR19PUklFTlRBVElPTiA9IDB4MDA4MCwKLSAgICAgICAgQ09ORklHX0RFTlNJVFkgPSAweDAxMDAsCi0gICAgICAgIENPTkZJR19TQ1JFRU5fU0laRSA9IDB4MDIwMCwKLSAgICAgICAgQ09ORklHX1ZFUlNJT04gPSAweDA0MDAKLSAgICB9OwotICAgIAotICAgIC8vIENvbXBhcmUgdHdvIGNvbmZpZ3VyYXRpb24sIHJldHVybmluZyBDT05GSUdfKiBmbGFncyBzZXQgZm9yIGVhY2ggdmFsdWUKLSAgICAvLyB0aGF0IGlzIGRpZmZlcmVudC4KLSAgICBpbmxpbmUgaW50IGRpZmYoY29uc3QgUmVzVGFibGVfY29uZmlnJiBvKSBjb25zdCB7Ci0gICAgICAgIGludCBkaWZmcyA9IDA7Ci0gICAgICAgIGlmIChtY2MgIT0gby5tY2MpIGRpZmZzIHw9IENPTkZJR19NQ0M7Ci0gICAgICAgIGlmIChtbmMgIT0gby5tbmMpIGRpZmZzIHw9IENPTkZJR19NTkM7Ci0gICAgICAgIGlmIChsb2NhbGUgIT0gby5sb2NhbGUpIGRpZmZzIHw9IENPTkZJR19MT0NBTEU7Ci0gICAgICAgIGlmIChvcmllbnRhdGlvbiAhPSBvLm9yaWVudGF0aW9uKSBkaWZmcyB8PSBDT05GSUdfT1JJRU5UQVRJT047Ci0gICAgICAgIGlmIChkZW5zaXR5ICE9IG8uZGVuc2l0eSkgZGlmZnMgfD0gQ09ORklHX0RFTlNJVFk7Ci0gICAgICAgIGlmICh0b3VjaHNjcmVlbiAhPSBvLnRvdWNoc2NyZWVuKSBkaWZmcyB8PSBDT05GSUdfVE9VQ0hTQ1JFRU47Ci0gICAgICAgIGlmICgoKGlucHV0RmxhZ3Neby5pbnB1dEZsYWdzKSZNQVNLX0tFWVNISURERU4pICE9IDApIGRpZmZzIHw9IENPTkZJR19LRVlCT0FSRF9ISURERU47Ci0gICAgICAgIGlmIChrZXlib2FyZCAhPSBvLmtleWJvYXJkKSBkaWZmcyB8PSBDT05GSUdfS0VZQk9BUkQ7Ci0gICAgICAgIGlmIChuYXZpZ2F0aW9uICE9IG8ubmF2aWdhdGlvbikgZGlmZnMgfD0gQ09ORklHX05BVklHQVRJT047Ci0gICAgICAgIGlmIChzY3JlZW5TaXplICE9IG8uc2NyZWVuU2l6ZSkgZGlmZnMgfD0gQ09ORklHX1NDUkVFTl9TSVpFOwotICAgICAgICBpZiAodmVyc2lvbiAhPSBvLnZlcnNpb24pIGRpZmZzIHw9IENPTkZJR19WRVJTSU9OOwotICAgICAgICByZXR1cm4gZGlmZnM7Ci0gICAgfQotICAgIAotICAgIC8vIFJldHVybiB0cnVlIGlmICd0aGlzJyBpcyBtb3JlIHNwZWNpZmljIHRoYW4gJ28nLiAgT3B0aW9uYWxseSwgaWYKLSAgICAvLyAncmVxdWVzdGVkJyBpcyBudWxsLCB0aGVuIHRoZXkgd2lsbCBhbHNvIGJlIGNvbXBhcmVkIGFnYWluc3QgdGhlCi0gICAgLy8gcmVxdWVzdGVkIGNvbmZpZ3VyYXRpb24gYW5kIHRydWUgd2lsbCBvbmx5IGJlIHJldHVybmVkIGlmICd0aGlzJwotICAgIC8vIGlzIGEgYmV0dGVyIGNhbmRpZGF0ZSB0aGFuICdvJyBmb3IgdGhlIGNvbmZpZ3VyYXRpb24uICBUaGlzIGFzc3VtZXMgdGhhdAotICAgIC8vIG1hdGNoKCkgaGFzIGFscmVhZHkgYmVlbiB1c2VkIHRvIHJlbW92ZSBhbnkgY29uZmlndXJhdGlvbnMgdGhhdCBkb24ndAotICAgIC8vIG1hdGNoIHRoZSByZXF1ZXN0ZWQgY29uZmlndXJhdGlvbiBhdCBhbGw7IGlmIHRoZXkgYXJlIG5vdCBmaXJzdCBmaWx0ZXJlZCwKLSAgICAvLyBub24tbWF0Y2hpbmcgcmVzdWx0cyBjYW4gYmUgY29uc2lkZXJlZCBiZXR0ZXIgdGhhbiBtYXRjaGluZyBvbmVzLgotICAgIGlubGluZSBib29sCi0gICAgaXNCZXR0ZXJUaGFuKGNvbnN0IFJlc1RhYmxlX2NvbmZpZyYgbywgY29uc3QgUmVzVGFibGVfY29uZmlnKiByZXF1ZXN0ZWQgPSBOVUxMKSBjb25zdCB7Ci0gICAgICAgIC8vIFRoZSBvcmRlciBvZiB0aGUgZm9sbG93aW5nIHRlc3RzIGRlZmluZXMgdGhlIGltcG9ydGFuY2Ugb2Ygb25lCi0gICAgICAgIC8vIGNvbmZpZ3VyYXRpb24gcGFyYW1ldGVyIG92ZXIgYW5vdGhlci4gIFRob3NlIHRlc3RzIGZpcnN0IGFyZSBtb3JlCi0gICAgICAgIC8vIGltcG9ydGFudCwgdHJ1bXBpbmcgYW55IHZhbHVlcyBpbiB0aG9zZSBmb2xsb3dpbmcgdGhlbS4KLSAgICAgICAgaWYgKGltc2kgIT0gMCAmJiAoIXJlcXVlc3RlZCB8fCByZXF1ZXN0ZWQtPmltc2kgIT0gMCkpIHsKLSAgICAgICAgICAgIGlmIChtY2MgIT0gMCAmJiAoIXJlcXVlc3RlZCB8fCByZXF1ZXN0ZWQtPm1jYyAhPSAwKSkgewotICAgICAgICAgICAgICAgIGlmIChvLm1jYyA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChtbmMgIT0gMCAmJiAoIXJlcXVlc3RlZCB8fCByZXF1ZXN0ZWQtPm1uYyAhPSAwKSkgewotICAgICAgICAgICAgICAgIGlmIChvLm1uYyA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBpZiAobG9jYWxlICE9IDAgJiYgKCFyZXF1ZXN0ZWQgfHwgcmVxdWVzdGVkLT5sb2NhbGUgIT0gMCkpIHsKLSAgICAgICAgICAgIGlmIChsYW5ndWFnZVswXSAhPSAwICYmICghcmVxdWVzdGVkIHx8IHJlcXVlc3RlZC0+bGFuZ3VhZ2VbMF0gIT0gMCkpIHsKLSAgICAgICAgICAgICAgICBpZiAoby5sYW5ndWFnZVswXSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChjb3VudHJ5WzBdICE9IDAgJiYgKCFyZXF1ZXN0ZWQgfHwgcmVxdWVzdGVkLT5jb3VudHJ5WzBdICE9IDApKSB7Ci0gICAgICAgICAgICAgICAgaWYgKG8uY291bnRyeVswXSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBpZiAoc2NyZWVuVHlwZSAhPSAwICYmICghcmVxdWVzdGVkIHx8IHJlcXVlc3RlZC0+c2NyZWVuVHlwZSAhPSAwKSkgewotICAgICAgICAgICAgaWYgKG9yaWVudGF0aW9uICE9IDAgJiYgKCFyZXF1ZXN0ZWQgfHwgcmVxdWVzdGVkLT5vcmllbnRhdGlvbiAhPSAwKSkgewotICAgICAgICAgICAgICAgIGlmIChvLm9yaWVudGF0aW9uID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGRlbnNpdHkgIT0gMCAmJiAoIXJlcXVlc3RlZCB8fCByZXF1ZXN0ZWQtPmRlbnNpdHkgIT0gMCkpIHsKLSAgICAgICAgICAgICAgICBpZiAoby5kZW5zaXR5ID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHRvdWNoc2NyZWVuICE9IDAgJiYgKCFyZXF1ZXN0ZWQgfHwgcmVxdWVzdGVkLT50b3VjaHNjcmVlbiAhPSAwKSkgewotICAgICAgICAgICAgICAgIGlmIChvLnRvdWNoc2NyZWVuID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGlmIChpbnB1dCAhPSAwICYmICghcmVxdWVzdGVkIHx8IHJlcXVlc3RlZC0+aW5wdXQgIT0gMCkpIHsKLSAgICAgICAgICAgIGNvbnN0IGludCBrZXlzSGlkZGVuID0gaW5wdXRGbGFncyZNQVNLX0tFWVNISURERU47Ci0gICAgICAgICAgICBjb25zdCBpbnQgcmVxS2V5c0hpZGRlbiA9IHJlcXVlc3RlZAotICAgICAgICAgICAgICAgICAgICA/IHJlcXVlc3RlZC0+aW5wdXRGbGFncyZNQVNLX0tFWVNISURERU4gOiAwOwotICAgICAgICAgICAgaWYgKGtleXNIaWRkZW4gIT0gMCAmJiByZXFLZXlzSGlkZGVuICE9IDApIHsKLSAgICAgICAgICAgICAgICBjb25zdCBpbnQgb0tleXNIaWRkZW4gPSBvLmlucHV0RmxhZ3MmTUFTS19LRVlTSElEREVOOwotICAgICAgICAgICAgICAgIC8vTE9HSSgiaXNCZXR0ZXJUaGFuIGtleXNIaWRkZW46IGN1cj0lZCwgZ2l2ZW49JWQsIGNvbmZpZz0lZFxuIiwKLSAgICAgICAgICAgICAgICAvLyAgICAgICAga2V5c0hpZGRlbiwgb0tleXNIaWRkZW4sIHJlcUtleXNIaWRkZW4pOwotICAgICAgICAgICAgICAgIGlmIChvS2V5c0hpZGRlbiA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vTE9HSSgiQmV0dGVyIGJlY2F1c2UgMCEiKTsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIC8vIEZvciBjb21wYXRpYmlsaXR5LCB3ZSBjb3VudCBLRVlTSElEREVOX05PIGFzIGJlaW5nCi0gICAgICAgICAgICAgICAgLy8gdGhlIHNhbWUgYXMgS0VZU0hJRERFTl9TT0ZULiAgSGVyZSB3ZSBkaXNhbWJpZ3VhdGUgdGhlc2UKLSAgICAgICAgICAgICAgICAvLyBtYXkgbWFraW5nIGFuIGV4YWN0IG1hdGNoIG1vcmUgc3BlY2lmaWMuCi0gICAgICAgICAgICAgICAgaWYgKGtleXNIaWRkZW4gPT0gcmVxS2V5c0hpZGRlbiAmJiBvS2V5c0hpZGRlbiAhPSByZXFLZXlzSGlkZGVuKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIFRoZSBjdXJyZW50IGNvbmZpZ3VyYXRpb24gaXMgYW4gZXhhY3QgbWF0Y2gsIGFuZAotICAgICAgICAgICAgICAgICAgICAvLyB0aGUgZ2l2ZW4gb25lIGlzIG5vdCwgc28gdGhlIGN1cnJlbnQgb25lIGlzIGJldHRlci4KLSAgICAgICAgICAgICAgICAgICAgLy9MT0dJKCJCZXR0ZXIgYmVjYXVzZSBvdGhlciBub3Qgc2FtZSEiKTsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGtleWJvYXJkICE9IDAgJiYgKCFyZXF1ZXN0ZWQgfHwgcmVxdWVzdGVkLT5rZXlib2FyZCAhPSAwKSkgewotICAgICAgICAgICAgICAgIGlmIChvLmtleWJvYXJkID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKG5hdmlnYXRpb24gIT0gMCAmJiAoIXJlcXVlc3RlZCB8fCByZXF1ZXN0ZWQtPm5hdmlnYXRpb24gIT0gMCkpIHsKLSAgICAgICAgICAgICAgICBpZiAoby5uYXZpZ2F0aW9uID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGlmIChzY3JlZW5TaXplICE9IDAgJiYgKCFyZXF1ZXN0ZWQgfHwgcmVxdWVzdGVkLT5zY3JlZW5TaXplICE9IDApKSB7Ci0gICAgICAgICAgICBpZiAoc2NyZWVuV2lkdGggIT0gMCAmJiAoIXJlcXVlc3RlZCB8fCByZXF1ZXN0ZWQtPnNjcmVlbldpZHRoICE9IDApKSB7Ci0gICAgICAgICAgICAgICAgaWYgKG8uc2NyZWVuV2lkdGggPT0gMCkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoc2NyZWVuSGVpZ2h0ICE9IDAgJiYgKCFyZXF1ZXN0ZWQgfHwgcmVxdWVzdGVkLT5zY3JlZW5IZWlnaHQgIT0gMCkpIHsKLSAgICAgICAgICAgICAgICBpZiAoby5zY3JlZW5IZWlnaHQgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHZlcnNpb24gIT0gMCAmJiAoIXJlcXVlc3RlZCB8fCByZXF1ZXN0ZWQtPnZlcnNpb24gIT0gMCkpIHsKLSAgICAgICAgICAgIGlmIChzZGtWZXJzaW9uICE9IDAgJiYgKCFyZXF1ZXN0ZWQgfHwgcmVxdWVzdGVkLT5zZGtWZXJzaW9uICE9IDApKSB7Ci0gICAgICAgICAgICAgICAgaWYgKG8uc2RrVmVyc2lvbiA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChtaW5vclZlcnNpb24gIT0gMCAmJiAoIXJlcXVlc3RlZCB8fCByZXF1ZXN0ZWQtPm1pbm9yVmVyc2lvbiAhPSAwKSkgewotICAgICAgICAgICAgICAgIGlmIChvLm1pbm9yVmVyc2lvbiA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotICAgIAotICAgIC8vIFJldHVybiB0cnVlIGlmICd0aGlzJyBjYW4gYmUgY29uc2lkZXJlZCBhIG1hdGNoIGZvciB0aGUgcGFyYW1ldGVycyBpbgotICAgIC8vICdzZXR0aW5ncycuCi0gICAgaW5saW5lIGJvb2wgbWF0Y2goY29uc3QgUmVzVGFibGVfY29uZmlnJiBzZXR0aW5ncykgY29uc3QgewotICAgICAgICBpZiAoaW1zaSAhPSAwKSB7Ci0gICAgICAgICAgICBpZiAoc2V0dGluZ3MubWNjICE9IDAgJiYgbWNjICE9IDAKLSAgICAgICAgICAgICAgICAmJiBtY2MgIT0gc2V0dGluZ3MubWNjKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHNldHRpbmdzLm1uYyAhPSAwICYmIG1uYyAhPSAwCi0gICAgICAgICAgICAgICAgJiYgbW5jICE9IHNldHRpbmdzLm1uYykgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBpZiAobG9jYWxlICE9IDApIHsKLSAgICAgICAgICAgIGlmIChzZXR0aW5ncy5sYW5ndWFnZVswXSAhPSAwICYmIGxhbmd1YWdlWzBdICE9IDAKLSAgICAgICAgICAgICAgICAmJiAobGFuZ3VhZ2VbMF0gIT0gc2V0dGluZ3MubGFuZ3VhZ2VbMF0KLSAgICAgICAgICAgICAgICAgICAgfHwgbGFuZ3VhZ2VbMV0gIT0gc2V0dGluZ3MubGFuZ3VhZ2VbMV0pKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHNldHRpbmdzLmNvdW50cnlbMF0gIT0gMCAmJiBjb3VudHJ5WzBdICE9IDAKLSAgICAgICAgICAgICAgICAmJiAoY291bnRyeVswXSAhPSBzZXR0aW5ncy5jb3VudHJ5WzBdCi0gICAgICAgICAgICAgICAgICAgIHx8IGNvdW50cnlbMV0gIT0gc2V0dGluZ3MuY291bnRyeVsxXSkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHNjcmVlblR5cGUgIT0gMCkgewotICAgICAgICAgICAgaWYgKHNldHRpbmdzLm9yaWVudGF0aW9uICE9IDAgJiYgb3JpZW50YXRpb24gIT0gMAotICAgICAgICAgICAgICAgICYmIG9yaWVudGF0aW9uICE9IHNldHRpbmdzLm9yaWVudGF0aW9uKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgLy8gRGVuc2l0eSBub3QgdGFrZW4gaW50byBhY2NvdW50LCBhbHdheXMgbWF0Y2gsIG5vIG1hdHRlciB3aGF0Ci0gICAgICAgICAgICAvLyBkZW5zaXR5IGlzIHNwZWNpZmllZCBmb3IgdGhlIHJlc291cmNlCi0gICAgICAgICAgICBpZiAoc2V0dGluZ3MudG91Y2hzY3JlZW4gIT0gMCAmJiB0b3VjaHNjcmVlbiAhPSAwCi0gICAgICAgICAgICAgICAgJiYgdG91Y2hzY3JlZW4gIT0gc2V0dGluZ3MudG91Y2hzY3JlZW4pIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGlucHV0ICE9IDApIHsKLSAgICAgICAgICAgIGNvbnN0IGludCBrZXlzSGlkZGVuID0gaW5wdXRGbGFncyZNQVNLX0tFWVNISURERU47Ci0gICAgICAgICAgICBjb25zdCBpbnQgc2V0S2V5c0hpZGRlbiA9IHNldHRpbmdzLmlucHV0RmxhZ3MmTUFTS19LRVlTSElEREVOOwotICAgICAgICAgICAgaWYgKHNldEtleXNIaWRkZW4gIT0gMCAmJiBrZXlzSGlkZGVuICE9IDAKLSAgICAgICAgICAgICAgICAmJiBrZXlzSGlkZGVuICE9IHNldEtleXNIaWRkZW4pIHsKLSAgICAgICAgICAgICAgICAvLyBGb3IgY29tcGF0aWJpbGl0eSwgd2UgY291bnQgYSByZXF1ZXN0IGZvciBLRVlTSElEREVOX05PIGFzIGFsc28KLSAgICAgICAgICAgICAgICAvLyBtYXRjaGluZyB0aGUgbW9yZSByZWNlbnQgS0VZU0hJRERFTl9TT0ZULiAgQmFzaWNhbGx5Ci0gICAgICAgICAgICAgICAgLy8gS0VZU0hJRERFTl9OTyBtZWFucyB0aGVyZSBpcyBzb21lIGtpbmQgb2Yga2V5Ym9hcmQgYXZhaWxhYmxlLgotICAgICAgICAgICAgICAgIC8vTE9HSSgiTWF0Y2hpbmcga2V5c0hpZGRlbjogaGF2ZT0lZCwgY29uZmlnPSVkXG4iLCBrZXlzSGlkZGVuLCBzZXRLZXlzSGlkZGVuKTsKLSAgICAgICAgICAgICAgICBpZiAoa2V5c0hpZGRlbiAhPSBLRVlTSElEREVOX05PIHx8IHNldEtleXNIaWRkZW4gIT0gS0VZU0hJRERFTl9TT0ZUKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vTE9HSSgiTm8gbWF0Y2ghIik7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoc2V0dGluZ3Mua2V5Ym9hcmQgIT0gMCAmJiBrZXlib2FyZCAhPSAwCi0gICAgICAgICAgICAgICAgJiYga2V5Ym9hcmQgIT0gc2V0dGluZ3Mua2V5Ym9hcmQpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoc2V0dGluZ3MubmF2aWdhdGlvbiAhPSAwICYmIG5hdmlnYXRpb24gIT0gMAotICAgICAgICAgICAgICAgICYmIG5hdmlnYXRpb24gIT0gc2V0dGluZ3MubmF2aWdhdGlvbikgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBpZiAoc2NyZWVuU2l6ZSAhPSAwKSB7Ci0gICAgICAgICAgICBpZiAoc2V0dGluZ3Muc2NyZWVuV2lkdGggIT0gMCAmJiBzY3JlZW5XaWR0aCAhPSAwCi0gICAgICAgICAgICAgICAgJiYgc2NyZWVuV2lkdGggIT0gc2V0dGluZ3Muc2NyZWVuV2lkdGgpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoc2V0dGluZ3Muc2NyZWVuSGVpZ2h0ICE9IDAgJiYgc2NyZWVuSGVpZ2h0ICE9IDAKLSAgICAgICAgICAgICAgICAmJiBzY3JlZW5IZWlnaHQgIT0gc2V0dGluZ3Muc2NyZWVuSGVpZ2h0KSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGlmICh2ZXJzaW9uICE9IDApIHsKLSAgICAgICAgICAgIGlmIChzZXR0aW5ncy5zZGtWZXJzaW9uICE9IDAgJiYgc2RrVmVyc2lvbiAhPSAwCi0gICAgICAgICAgICAgICAgJiYgc2RrVmVyc2lvbiAhPSBzZXR0aW5ncy5zZGtWZXJzaW9uKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHNldHRpbmdzLm1pbm9yVmVyc2lvbiAhPSAwICYmIG1pbm9yVmVyc2lvbiAhPSAwCi0gICAgICAgICAgICAgICAgJiYgbWlub3JWZXJzaW9uICE9IHNldHRpbmdzLm1pbm9yVmVyc2lvbikgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0KLSAgICB2b2lkIGdldExvY2FsZShjaGFyIHN0cls2XSkgY29uc3QgewotICAgICAgICBtZW1zZXQoc3RyLCAwLCA2KTsKLSAgICAgICAgaWYgKGxhbmd1YWdlWzBdKSB7Ci0gICAgICAgICAgICBzdHJbMF0gPSBsYW5ndWFnZVswXTsKLSAgICAgICAgICAgIHN0clsxXSA9IGxhbmd1YWdlWzFdOwotICAgICAgICAgICAgaWYgKGNvdW50cnlbMF0pIHsKLSAgICAgICAgICAgICAgICBzdHJbMl0gPSAnXyc7Ci0gICAgICAgICAgICAgICAgc3RyWzNdID0gY291bnRyeVswXTsKLSAgICAgICAgICAgICAgICBzdHJbNF0gPSBjb3VudHJ5WzFdOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgU3RyaW5nOCB0b1N0cmluZygpIGNvbnN0IHsKLSAgICAgICAgY2hhciBidWZbMjAwXTsKLSAgICAgICAgc3ByaW50ZihidWYsICJpbXNpPSVkLyVkIGxhbmc9JWMlYyByZWc9JWMlYyBvcmllbnQ9MHglMDJ4IHRvdWNoPTB4JTAyeCBkZW5zPTB4JTAyeCAiCi0gICAgICAgICAgICAgICAgImtiZD0weCUwMnggbmF2PTB4JTAyeCBpbnB1dD0weCUwMnggc2NyZWVuVz0weCUwNHggc2NyZWVuSD0weCUwNHggdmVycz0lZC4lZCIsCi0gICAgICAgICAgICAgICAgbWNjLCBtbmMsCi0gICAgICAgICAgICAgICAgbGFuZ3VhZ2VbMF0gPyBsYW5ndWFnZVswXSA6ICctJywgbGFuZ3VhZ2VbMV0gPyBsYW5ndWFnZVsxXSA6ICctJywKLSAgICAgICAgICAgICAgICBjb3VudHJ5WzBdID8gY291bnRyeVswXSA6ICctJywgY291bnRyeVsxXSA/IGNvdW50cnlbMV0gOiAnLScsCi0gICAgICAgICAgICAgICAgb3JpZW50YXRpb24sIHRvdWNoc2NyZWVuLCBkZW5zaXR5LCBrZXlib2FyZCwgbmF2aWdhdGlvbiwgaW5wdXRGbGFncywKLSAgICAgICAgICAgICAgICBzY3JlZW5XaWR0aCwgc2NyZWVuSGVpZ2h0LCBzZGtWZXJzaW9uLCBtaW5vclZlcnNpb24pOwotICAgICAgICByZXR1cm4gU3RyaW5nOChidWYpOwotICAgIH0KLX07Ci0KLS8qKgotICogQSBzcGVjaWZpY2F0aW9uIG9mIHRoZSByZXNvdXJjZXMgZGVmaW5lZCBieSBhIHBhcnRpY3VsYXIgdHlwZS4KLSAqCi0gKiBUaGVyZSBzaG91bGQgYmUgb25lIG9mIHRoZXNlIGNodW5rcyBmb3IgZWFjaCByZXNvdXJjZSB0eXBlLgotICoKLSAqIFRoaXMgc3RydWN0dXJlIGlzIGZvbGxvd2VkIGJ5IGFuIGFycmF5IG9mIGludGVnZXJzIHByb3ZpZGluZyB0aGUgc2V0IG9mCi0gKiBjb25maWd1YXRpb24gY2hhbmdlIGZsYWdzIChSZXNUYWJsZV9jb25maWc6OkNPTkZJR18qKSB0aGF0IGhhdmUgbXVsdGlwbGUKLSAqIHJlc291cmNlcyBmb3IgdGhhdCBjb25maWd1cmF0aW9uLiAgSW4gYWRkaXRpb24sIHRoZSBoaWdoIGJpdCBpcyBzZXQgaWYgdGhhdAotICogcmVzb3VyY2UgaGFzIGJlZW4gbWFkZSBwdWJsaWMuCi0gKi8KLXN0cnVjdCBSZXNUYWJsZV90eXBlU3BlYwotewotICAgIHN0cnVjdCBSZXNDaHVua19oZWFkZXIgaGVhZGVyOwotCi0gICAgLy8gVGhlIHR5cGUgaWRlbnRpZmllciB0aGlzIGNodW5rIGlzIGhvbGRpbmcuICBUeXBlIElEcyBzdGFydAotICAgIC8vIGF0IDEgKGNvcnJlc3BvbmRpbmcgdG8gdGhlIHZhbHVlIG9mIHRoZSB0eXBlIGJpdHMgaW4gYQotICAgIC8vIHJlc291cmNlIGlkZW50aWZpZXIpLiAgMCBpcyBpbnZhbGlkLgotICAgIHVpbnQ4X3QgaWQ7Ci0gICAgCi0gICAgLy8gTXVzdCBiZSAwLgotICAgIHVpbnQ4X3QgcmVzMDsKLSAgICAvLyBNdXN0IGJlIDAuCi0gICAgdWludDE2X3QgcmVzMTsKLSAgICAKLSAgICAvLyBOdW1iZXIgb2YgdWludDMyX3QgZW50cnkgY29uZmlndXJhdGlvbiBtYXNrcyB0aGF0IGZvbGxvdy4KLSAgICB1aW50MzJfdCBlbnRyeUNvdW50OwotCi0gICAgZW51bSB7Ci0gICAgICAgIC8vIEFkZGl0aW9uYWwgZmxhZyBpbmRpY2F0aW5nIGFuIGVudHJ5IGlzIHB1YmxpYy4KLSAgICAgICAgU1BFQ19QVUJMSUMgPSAweDQwMDAwMDAwCi0gICAgfTsKLX07Ci0KLS8qKgotICogQSBjb2xsZWN0aW9uIG9mIHJlc291cmNlIGVudHJpZXMgZm9yIGEgcGFydGljdWxhciByZXNvdXJjZSBkYXRhCi0gKiB0eXBlLiBGb2xsb3dlZCBieSBhbiBhcnJheSBvZiB1aW50MzJfdCBkZWZpbmluZyB0aGUgcmVzb3VyY2UKLSAqIHZhbHVlcywgY29ycmVzcG9uZGluZyB0byB0aGUgYXJyYXkgb2YgdHlwZSBzdHJpbmdzIGluIHRoZQotICogUmVzVGFibGVfcGFja2FnZTo6dHlwZVN0cmluZ3Mgc3RyaW5nIGJsb2NrLiBFYWNoIG9mIHRoZXNlIGhvbGQgYW4KLSAqIGluZGV4IGZyb20gZW50cmllc1N0YXJ0OyBhIHZhbHVlIG9mIE5PX0VOVFJZIG1lYW5zIHRoYXQgZW50cnkgaXMKLSAqIG5vdCBkZWZpbmVkLgotICoKLSAqIFRoZXJlIG1heSBiZSBtdWx0aXBsZSBvZiB0aGVzZSBjaHVua3MgZm9yIGEgcGFydGljdWxhciByZXNvdXJjZSB0eXBlLAotICogc3VwcGx5IGRpZmZlcmVudCBjb25maWd1cmF0aW9uIHZhcmlhdGlvbnMgZm9yIHRoZSByZXNvdXJjZSB2YWx1ZXMgb2YKLSAqIHRoYXQgdHlwZS4KLSAqCi0gKiBJdCB3b3VsZCBiZSBuaWNlIHRvIGhhdmUgYW4gYWRkaXRpb25hbCBvcmRlcmVkIGluZGV4IG9mIGVudHJpZXMsIHNvCi0gKiB3ZSBjYW4gZG8gYSBiaW5hcnkgc2VhcmNoIGlmIHRyeWluZyB0byBmaW5kIGEgcmVzb3VyY2UgYnkgc3RyaW5nIG5hbWUuCi0gKi8KLXN0cnVjdCBSZXNUYWJsZV90eXBlCi17Ci0gICAgc3RydWN0IFJlc0NodW5rX2hlYWRlciBoZWFkZXI7Ci0KLSAgICBlbnVtIHsKLSAgICAgICAgTk9fRU5UUlkgPSAweEZGRkZGRkZGCi0gICAgfTsKLSAgICAKLSAgICAvLyBUaGUgdHlwZSBpZGVudGlmaWVyIHRoaXMgY2h1bmsgaXMgaG9sZGluZy4gIFR5cGUgSURzIHN0YXJ0Ci0gICAgLy8gYXQgMSAoY29ycmVzcG9uZGluZyB0byB0aGUgdmFsdWUgb2YgdGhlIHR5cGUgYml0cyBpbiBhCi0gICAgLy8gcmVzb3VyY2UgaWRlbnRpZmllcikuICAwIGlzIGludmFsaWQuCi0gICAgdWludDhfdCBpZDsKLSAgICAKLSAgICAvLyBNdXN0IGJlIDAuCi0gICAgdWludDhfdCByZXMwOwotICAgIC8vIE11c3QgYmUgMC4KLSAgICB1aW50MTZfdCByZXMxOwotICAgIAotICAgIC8vIE51bWJlciBvZiB1aW50MzJfdCBlbnRyeSBpbmRpY2VzIHRoYXQgZm9sbG93LgotICAgIHVpbnQzMl90IGVudHJ5Q291bnQ7Ci0KLSAgICAvLyBPZmZzZXQgZnJvbSBoZWFkZXIgd2hlcmUgUmVzVGFibGVfZW50cnkgZGF0YSBzdGFydHMuCi0gICAgdWludDMyX3QgZW50cmllc1N0YXJ0OwotICAgIAotICAgIC8vIENvbmZpZ3VyYXRpb24gdGhpcyBjb2xsZWN0aW9uIG9mIGVudHJpZXMgaXMgZGVzaWduZWQgZm9yLgotICAgIFJlc1RhYmxlX2NvbmZpZyBjb25maWc7Ci19OwotCi0vKioKLSAqIFRoaXMgaXMgdGhlIGJlZ2lubmluZyBvZiBpbmZvcm1hdGlvbiBhYm91dCBhbiBlbnRyeSBpbiB0aGUgcmVzb3VyY2UKLSAqIHRhYmxlLiAgSXQgaG9sZHMgdGhlIHJlZmVyZW5jZSB0byB0aGUgbmFtZSBvZiB0aGlzIGVudHJ5LCBhbmQgaXMKLSAqIGltbWVkaWF0ZWx5IGZvbGxvd2VkIGJ5IG9uZSBvZjoKLSAqICAgKiBBIFJlc192YWx1ZSBzdHJ1Y3R1cmVzLCBpZiBGTEFHX0NPTVBMRVggaXMgLW5vdC0gc2V0LgotICogICAqIEFuIGFycmF5IG9mIFJlc1RhYmxlX21hcCBzdHJ1Y3R1cmVzLCBpZiBGTEFHX0NPTVBMRVggaXMgc2V0LgotICogICAgIFRoZXNlIHN1cHBseSBhIHNldCBvZiBuYW1lL3ZhbHVlIG1hcHBpbmdzIG9mIGRhdGEuCi0gKi8KLXN0cnVjdCBSZXNUYWJsZV9lbnRyeQotewotICAgIC8vIE51bWJlciBvZiBieXRlcyBpbiB0aGlzIHN0cnVjdHVyZS4KLSAgICB1aW50MTZfdCBzaXplOwotCi0gICAgZW51bSB7Ci0gICAgICAgIC8vIElmIHNldCwgdGhpcyBpcyBhIGNvbXBsZXggZW50cnksIGhvbGRpbmcgYSBzZXQgb2YgbmFtZS92YWx1ZQotICAgICAgICAvLyBtYXBwaW5ncy4gIEl0IGlzIGZvbGxvd2VkIGJ5IGFuIGFycmF5IG9mIFJlc1RhYmxlX21hcCBzdHJ1Y3R1cmVzLgotICAgICAgICBGTEFHX0NPTVBMRVggPSAweDAwMDEsCi0gICAgICAgIC8vIElmIHNldCwgdGhpcyByZXNvdXJjZSBoYXMgYmVlbiBkZWNsYXJlZCBwdWJsaWMsIHNvIGxpYnJhcmllcwotICAgICAgICAvLyBhcmUgYWxsb3dlZCB0byByZWZlcmVuY2UgaXQuCi0gICAgICAgIEZMQUdfUFVCTElDID0gMHgwMDAyCi0gICAgfTsKLSAgICB1aW50MTZfdCBmbGFnczsKLSAgICAKLSAgICAvLyBSZWZlcmVuY2UgaW50byBSZXNUYWJsZV9wYWNrYWdlOjprZXlTdHJpbmdzIGlkZW50aWZ5aW5nIHRoaXMgZW50cnkuCi0gICAgc3RydWN0IFJlc1N0cmluZ1Bvb2xfcmVmIGtleTsKLX07Ci0KLS8qKgotICogRXh0ZW5kZWQgZm9ybSBvZiBhIFJlc1RhYmxlX2VudHJ5IGZvciBtYXAgZW50cmllcywgZGVmaW5pbmcgYSBwYXJlbnQgbWFwCi0gKiByZXNvdXJjZSBmcm9tIHdoaWNoIHRvIGluaGVyaXQgdmFsdWVzLgotICovCi1zdHJ1Y3QgUmVzVGFibGVfbWFwX2VudHJ5IDogcHVibGljIFJlc1RhYmxlX2VudHJ5Ci17Ci0gICAgLy8gUmVzb3VyY2UgaWRlbnRpZmllciBvZiB0aGUgcGFyZW50IG1hcHBpbmcsIG9yIDAgaWYgdGhlcmUgaXMgbm9uZS4KLSAgICBSZXNUYWJsZV9yZWYgcGFyZW50OwotICAgIC8vIE51bWJlciBvZiBuYW1lL3ZhbHVlIHBhaXJzIHRoYXQgZm9sbG93IGZvciBGTEFHX0NPTVBMRVguCi0gICAgdWludDMyX3QgY291bnQ7Ci19OwotCi0vKioKLSAqIEEgc2luZ2xlIG5hbWUvdmFsdWUgbWFwcGluZyB0aGF0IGlzIHBhcnQgb2YgYSBjb21wbGV4IHJlc291cmNlCi0gKiBlbnRyeS4KLSAqLwotc3RydWN0IFJlc1RhYmxlX21hcAotewotICAgIC8vIFRoZSByZXNvdXJjZSBpZGVudGlmaWVyIGRlZmluaW5nIHRoaXMgbWFwcGluZydzIG5hbWUuICBGb3IgYXR0cmlidXRlCi0gICAgLy8gcmVzb3VyY2VzLCAnbmFtZScgY2FuIGJlIG9uZSBvZiB0aGUgZm9sbG93aW5nIHNwZWNpYWwgcmVzb3VyY2UgdHlwZXMKLSAgICAvLyB0byBzdXBwbHkgbWV0YS1kYXRhIGFib3V0IHRoZSBhdHRyaWJ1dGU7IGZvciBhbGwgb3RoZXIgcmVzb3VyY2UgdHlwZXMKLSAgICAvLyBpdCBtdXN0IGJlIGFuIGF0dHJpYnV0ZSByZXNvdXJjZS4KLSAgICBSZXNUYWJsZV9yZWYgbmFtZTsKLQotICAgIC8vIFNwZWNpYWwgdmFsdWVzIGZvciAnbmFtZScgd2hlbiBkZWZpbmluZyBhdHRyaWJ1dGUgcmVzb3VyY2VzLgotICAgIGVudW0gewotICAgICAgICAvLyBUaGlzIGVudHJ5IGhvbGRzIHRoZSBhdHRyaWJ1dGUncyB0eXBlIGNvZGUuCi0gICAgICAgIEFUVFJfVFlQRSA9IFJlc19NQUtFSU5URVJOQUwoMCksCi0KLSAgICAgICAgLy8gRm9yIGludGVncmFsIGF0dHJpYnV0ZXMsIHRoaXMgaXMgdGhlIG1pbmltdW0gdmFsdWUgaXQgY2FuIGhvbGQuCi0gICAgICAgIEFUVFJfTUlOID0gUmVzX01BS0VJTlRFUk5BTCgxKSwKLQotICAgICAgICAvLyBGb3IgaW50ZWdyYWwgYXR0cmlidXRlcywgdGhpcyBpcyB0aGUgbWF4aW11bSB2YWx1ZSBpdCBjYW4gaG9sZC4KLSAgICAgICAgQVRUUl9NQVggPSBSZXNfTUFLRUlOVEVSTkFMKDIpLAotCi0gICAgICAgIC8vIExvY2FsaXphdGlvbiBvZiB0aGlzIHJlc291cmNlIGlzIGNhbiBiZSBlbmNvdXJhZ2VkIG9yIHJlcXVpcmVkIHdpdGgKLSAgICAgICAgLy8gYW4gYWFwdCBmbGFnIGlmIHRoaXMgaXMgc2V0Ci0gICAgICAgIEFUVFJfTDEwTiA9IFJlc19NQUtFSU5URVJOQUwoMyksCi0KLSAgICAgICAgLy8gZm9yIHBsdXJhbCBzdXBwb3J0LCBzZWUgYW5kcm9pZC5jb250ZW50LnJlcy5QbHVyYWxSdWxlcyNhdHRyRm9yUXVhbnRpdHkoaW50KQotICAgICAgICBBVFRSX09USEVSID0gUmVzX01BS0VJTlRFUk5BTCg0KSwKLSAgICAgICAgQVRUUl9aRVJPID0gUmVzX01BS0VJTlRFUk5BTCg1KSwKLSAgICAgICAgQVRUUl9PTkUgPSBSZXNfTUFLRUlOVEVSTkFMKDYpLAotICAgICAgICBBVFRSX1RXTyA9IFJlc19NQUtFSU5URVJOQUwoNyksCi0gICAgICAgIEFUVFJfRkVXID0gUmVzX01BS0VJTlRFUk5BTCg4KSwKLSAgICAgICAgQVRUUl9NQU5ZID0gUmVzX01BS0VJTlRFUk5BTCg5KQotICAgICAgICAKLSAgICB9OwotCi0gICAgLy8gQml0IG1hc2sgb2YgYWxsb3dlZCB0eXBlcywgZm9yIHVzZSB3aXRoIEFUVFJfVFlQRS4KLSAgICBlbnVtIHsKLSAgICAgICAgLy8gTm8gdHlwZSBoYXMgYmVlbiBkZWZpbmVkIGZvciB0aGlzIGF0dHJpYnV0ZSwgdXNlIGdlbmVyaWMKLSAgICAgICAgLy8gdHlwZSBoYW5kbGluZy4gIFRoZSBsb3cgMTYgYml0cyBhcmUgZm9yIHR5cGVzIHRoYXQgY2FuIGJlCi0gICAgICAgIC8vIGhhbmRsZWQgZ2VuZXJpY2FsbHk7IHRoZSB1cHBlciAxNiByZXF1aXJlIGFkZGl0aW9uYWwgaW5mb3JtYXRpb24KLSAgICAgICAgLy8gaW4gdGhlIGJhZyBzbyBjYW4gbm90IGJlIGhhbmRsZWQgZ2VuZXJpY2FsbHkgZm9yIFRZUEVfQU5ZLgotICAgICAgICBUWVBFX0FOWSA9IDB4MDAwMEZGRkYsCi0KLSAgICAgICAgLy8gQXR0cmlidXRlIGhvbGRzIGEgcmVmZXJlbmNlcyB0byBhbm90aGVyIHJlc291cmNlLgotICAgICAgICBUWVBFX1JFRkVSRU5DRSA9IDE8PDAsCi0KLSAgICAgICAgLy8gQXR0cmlidXRlIGhvbGRzIGEgZ2VuZXJpYyBzdHJpbmcuCi0gICAgICAgIFRZUEVfU1RSSU5HID0gMTw8MSwKLQotICAgICAgICAvLyBBdHRyaWJ1dGUgaG9sZHMgYW4gaW50ZWdlciB2YWx1ZS4gIEFUVFJfTUlOIGFuZCBBVFRSX01JTiBjYW4KLSAgICAgICAgLy8gb3B0aW9uYWxseSBzcGVjaWZ5IGEgY29uc3RyYWluZWQgcmFuZ2Ugb2YgcG9zc2libGUgaW50ZWdlciB2YWx1ZXMuCi0gICAgICAgIFRZUEVfSU5URUdFUiA9IDE8PDIsCi0KLSAgICAgICAgLy8gQXR0cmlidXRlIGhvbGRzIGEgYm9vbGVhbiBpbnRlZ2VyLgotICAgICAgICBUWVBFX0JPT0xFQU4gPSAxPDwzLAotCi0gICAgICAgIC8vIEF0dHJpYnV0ZSBob2xkcyBhIGNvbG9yIHZhbHVlLgotICAgICAgICBUWVBFX0NPTE9SID0gMTw8NCwKLQotICAgICAgICAvLyBBdHRyaWJ1dGUgaG9sZHMgYSBmbG9hdGluZyBwb2ludCB2YWx1ZS4KLSAgICAgICAgVFlQRV9GTE9BVCA9IDE8PDUsCi0KLSAgICAgICAgLy8gQXR0cmlidXRlIGhvbGRzIGEgZGltZW5zaW9uIHZhbHVlLCBzdWNoIGFzICIyMHB4Ii4KLSAgICAgICAgVFlQRV9ESU1FTlNJT04gPSAxPDw2LAotCi0gICAgICAgIC8vIEF0dHJpYnV0ZSBob2xkcyBhIGZyYWN0aW9uIHZhbHVlLCBzdWNoIGFzICIyMCUiLgotICAgICAgICBUWVBFX0ZSQUNUSU9OID0gMTw8NywKLQotICAgICAgICAvLyBBdHRyaWJ1dGUgaG9sZHMgYW4gZW51bWVyYXRpb24uICBUaGUgZW51bWVyYXRpb24gdmFsdWVzIGFyZQotICAgICAgICAvLyBzdXBwbGllZCBhcyBhZGRpdGlvbmFsIGVudHJpZXMgaW4gdGhlIG1hcC4KLSAgICAgICAgVFlQRV9FTlVNID0gMTw8MTYsCi0KLSAgICAgICAgLy8gQXR0cmlidXRlIGhvbGRzIGEgYml0bWFrcyBvZiBmbGFncy4gIFRoZSBmbGFnIGJpdCB2YWx1ZXMgYXJlCi0gICAgICAgIC8vIHN1cHBsaWVkIGFzIGFkZGl0aW9uYWwgZW50cmllcyBpbiB0aGUgbWFwLgotICAgICAgICBUWVBFX0ZMQUdTID0gMTw8MTcKLSAgICB9OwotCi0gICAgLy8gRW51bSBvZiBsb2NhbGl6YXRpb24gbW9kZXMsIGZvciB1c2Ugd2l0aCBBVFRSX0wxME4uCi0gICAgZW51bSB7Ci0gICAgICAgIEwxME5fTk9UX1JFUVVJUkVEID0gMCwKLSAgICAgICAgTDEwTl9TVUdHRVNURUQgICAgPSAxCi0gICAgfTsKLSAgICAKLSAgICAvLyBUaGlzIG1hcHBpbmcncyB2YWx1ZS4KLSAgICBSZXNfdmFsdWUgdmFsdWU7Ci19OwotCi0vKioKLSAqIENvbnZlbmllbmNlIGNsYXNzIGZvciBhY2Nlc3NpbmcgZGF0YSBpbiBhIFJlc1RhYmxlIHJlc291cmNlLgotICovCi1jbGFzcyBSZXNUYWJsZQotewotcHVibGljOgotICAgIFJlc1RhYmxlKCk7Ci0gICAgUmVzVGFibGUoY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsIHZvaWQqIGNvb2tpZSwKLSAgICAgICAgICAgICBib29sIGNvcHlEYXRhPWZhbHNlKTsKLSAgICB+UmVzVGFibGUoKTsKLQotICAgIHN0YXR1c190IGFkZChjb25zdCB2b2lkKiBkYXRhLCBzaXplX3Qgc2l6ZSwgdm9pZCogY29va2llLAotICAgICAgICAgICAgICAgICBib29sIGNvcHlEYXRhPWZhbHNlKTsKLSAgICBzdGF0dXNfdCBhZGQoQXNzZXQqIGFzc2V0LCB2b2lkKiBjb29raWUsCi0gICAgICAgICAgICAgICAgIGJvb2wgY29weURhdGE9ZmFsc2UpOwotCi0gICAgc3RhdHVzX3QgZ2V0RXJyb3IoKSBjb25zdDsKLQotICAgIHZvaWQgdW5pbml0KCk7Ci0KLSAgICBzdHJ1Y3QgcmVzb3VyY2VfbmFtZQotICAgIHsKLSAgICAgICAgY29uc3QgY2hhcjE2X3QqIHBhY2thZ2U7Ci0gICAgICAgIHNpemVfdCBwYWNrYWdlTGVuOwotICAgICAgICBjb25zdCBjaGFyMTZfdCogdHlwZTsKLSAgICAgICAgc2l6ZV90IHR5cGVMZW47Ci0gICAgICAgIGNvbnN0IGNoYXIxNl90KiBuYW1lOwotICAgICAgICBzaXplX3QgbmFtZUxlbjsKLSAgICB9OwotCi0gICAgYm9vbCBnZXRSZXNvdXJjZU5hbWUodWludDMyX3QgcmVzSUQsIHJlc291cmNlX25hbWUqIG91dE5hbWUpIGNvbnN0OwotCi0gICAgLyoqCi0gICAgICogUmV0cmlldmUgdGhlIHZhbHVlIG9mIGEgcmVzb3VyY2UuICBJZiB0aGUgcmVzb3VyY2UgaXMgZm91bmQsIHJldHVybnMgYQotICAgICAqIHZhbHVlID49IDAgaW5kaWNhdGluZyB0aGUgdGFibGUgaXQgaXMgaW4gKGZvciB1c2Ugd2l0aAotICAgICAqIGdldFRhYmxlU3RyaW5nQmxvY2soKSBhbmQgZ2V0VGFibGVDb29raWUoKSkgYW5kIGZpbGxzIGluICdvdXRWYWx1ZScuICBJZgotICAgICAqIG5vdCBmb3VuZCwgcmV0dXJucyBhIG5lZ2F0aXZlIGVycm9yIGNvZGUuCi0gICAgICoKLSAgICAgKiBOb3RlIHRoYXQgdGhpcyBmdW5jdGlvbiBkb2VzIG5vdCBkbyByZWZlcmVuY2UgdHJhdmVyc2FsLiAgSWYgeW91IHdhbnQKLSAgICAgKiB0byBmb2xsb3cgcmVmZXJlbmNlcyB0byBvdGhlciByZXNvdXJjZXMgdG8gZ2V0IHRoZSAicmVhbCIgdmFsdWUgdG8KLSAgICAgKiB1c2UsIHlvdSBuZWVkIHRvIGNhbGwgcmVzb2x2ZVJlZmVyZW5jZSgpIGFmdGVyIHRoaXMgZnVuY3Rpb24uCi0gICAgICoKLSAgICAgKiBAcGFyYW0gcmVzSUQgVGhlIGRlc2lyZWQgcmVzb3J1Y2UgaWRlbnRpZmllci4KLSAgICAgKiBAcGFyYW0gb3V0VmFsdWUgRmlsbGVkIGluIHdpdGggdGhlIHJlc291cmNlIGRhdGEgdGhhdCB3YXMgZm91bmQuCi0gICAgICoKLSAgICAgKiBAcmV0dXJuIHNzaXplX3QgRWl0aGVyIGEgPj0gMCB0YWJsZSBpbmRleCBvciBhIG5lZ2F0aXZlIGVycm9yIGNvZGUuCi0gICAgICovCi0gICAgc3NpemVfdCBnZXRSZXNvdXJjZSh1aW50MzJfdCByZXNJRCwgUmVzX3ZhbHVlKiBvdXRWYWx1ZSwgYm9vbCBtYXlCZUJhZz1mYWxzZSwKLSAgICAgICAgICAgIHVpbnQzMl90KiBvdXRTcGVjRmxhZ3M9TlVMTCwgUmVzVGFibGVfY29uZmlnKiBvdXRDb25maWc9TlVMTCkgY29uc3Q7Ci0KLSAgICBpbmxpbmUgc3NpemVfdCBnZXRSZXNvdXJjZShjb25zdCBSZXNUYWJsZV9yZWYmIHJlcywgUmVzX3ZhbHVlKiBvdXRWYWx1ZSwKLSAgICAgICAgICAgIHVpbnQzMl90KiBvdXRTcGVjRmxhZ3M9TlVMTCkgY29uc3QgewotICAgICAgICByZXR1cm4gZ2V0UmVzb3VyY2UocmVzLmlkZW50LCBvdXRWYWx1ZSwgZmFsc2UsIG91dFNwZWNGbGFncywgTlVMTCk7Ci0gICAgfQotCi0gICAgc3NpemVfdCByZXNvbHZlUmVmZXJlbmNlKFJlc192YWx1ZSogaW5PdXRWYWx1ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3NpemVfdCBibG9ja0luZGV4LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCogb3V0TGFzdFJlZiA9IE5VTEwsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90KiBpbm91dFR5cGVTcGVjRmxhZ3MgPSBOVUxMKSBjb25zdDsKLQotICAgIGVudW0gewotICAgICAgICBUTVBfQlVGRkVSX1NJWkUgPSAxNgotICAgIH07Ci0gICAgY29uc3QgY2hhcjE2X3QqIHZhbHVlVG9TdHJpbmcoY29uc3QgUmVzX3ZhbHVlKiB2YWx1ZSwgc2l6ZV90IHN0cmluZ0Jsb2NrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNoYXIxNl90IHRtcEJ1ZmZlcltUTVBfQlVGRkVSX1NJWkVdLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpemVfdCogb3V0TGVuKTsKLQotICAgIHN0cnVjdCBiYWdfZW50cnkgewotICAgICAgICBzc2l6ZV90IHN0cmluZ0Jsb2NrOwotICAgICAgICBSZXNUYWJsZV9tYXAgbWFwOwotICAgIH07Ci0KLSAgICAvKioKLSAgICAgKiBSZXRyaWV2ZSB0aGUgYmFnIG9mIGEgcmVzb3VyY2UuICBJZiB0aGUgcmVzb3J1Y2UgaXMgZm91bmQsIHJldHVybnMgdGhlCi0gICAgICogbnVtYmVyIG9mIGJhZ3MgaXQgY29udGFpbnMgYW5kICdvdXRCYWcnIHBvaW50cyB0byBhbiBhcnJheSBvZiB0aGVpcgotICAgICAqIHZhbHVlcy4gIElmIG5vdCBmb3VuZCwgYSBuZWdhdGl2ZSBlcnJvciBjb2RlIGlzIHJldHVybmVkLgotICAgICAqCi0gICAgICogTm90ZSB0aGF0IHRoaXMgZnVuY3Rpb24gLWRvZXMtIGRvIHJlZmVyZW5jZSB0cmF2ZXJzYWwgb2YgdGhlIGJhZyBkYXRhLgotICAgICAqCi0gICAgICogQHBhcmFtIHJlc0lEIFRoZSBkZXNpcmVkIHJlc291cmNlIGlkZW50aWZpZXIuCi0gICAgICogQHBhcmFtIG91dEJhZyBGaWxsZWQgaW5tIHdpdGggYSBwb2ludGVyIHRvIHRoZSBiYWcgbWFwcGluZ3MuCi0gICAgICoKLSAgICAgKiBAcmV0dXJuIHNzaXplX3QgRWl0aGVyIGEgPj0gMCBiYWcgY291bnQgb2YgbmVnYXRpdmUgZXJyb3IgY29kZS4KLSAgICAgKi8KLSAgICBzc2l6ZV90IGxvY2tCYWcodWludDMyX3QgcmVzSUQsIGNvbnN0IGJhZ19lbnRyeSoqIG91dEJhZykgY29uc3Q7Ci0KLSAgICB2b2lkIHVubG9ja0JhZyhjb25zdCBiYWdfZW50cnkqIGJhZykgY29uc3Q7Ci0KLSAgICB2b2lkIGxvY2soKSBjb25zdDsKLQotICAgIHNzaXplX3QgZ2V0QmFnTG9ja2VkKHVpbnQzMl90IHJlc0lELCBjb25zdCBiYWdfZW50cnkqKiBvdXRCYWcsCi0gICAgICAgICAgICB1aW50MzJfdCogb3V0VHlwZVNwZWNGbGFncz1OVUxMKSBjb25zdDsKLQotICAgIHZvaWQgdW5sb2NrKCkgY29uc3Q7Ci0KLSAgICBjbGFzcyBUaGVtZSB7Ci0gICAgcHVibGljOgotICAgICAgICBUaGVtZShjb25zdCBSZXNUYWJsZSYgdGFibGUpOwotICAgICAgICB+VGhlbWUoKTsKLQotICAgICAgICBpbmxpbmUgY29uc3QgUmVzVGFibGUmIGdldFJlc1RhYmxlKCkgY29uc3QgeyByZXR1cm4gbVRhYmxlOyB9Ci0KLSAgICAgICAgc3RhdHVzX3QgYXBwbHlTdHlsZSh1aW50MzJfdCByZXNJRCwgYm9vbCBmb3JjZT1mYWxzZSk7Ci0gICAgICAgIHN0YXR1c190IHNldFRvKGNvbnN0IFRoZW1lJiBvdGhlcik7Ci0KLSAgICAgICAgLyoqCi0gICAgICAgICAqIFJldHJpZXZlIGEgdmFsdWUgaW4gdGhlIHRoZW1lLiAgSWYgdGhlIHRoZW1lIGRlZmluZXMgdGhpcwotICAgICAgICAgKiB2YWx1ZSwgcmV0dXJucyBhIHZhbHVlID49IDAgaW5kaWNhdGluZyB0aGUgdGFibGUgaXQgaXMgaW4KLSAgICAgICAgICogKGZvciB1c2Ugd2l0aCBnZXRUYWJsZVN0cmluZ0Jsb2NrKCkgYW5kIGdldFRhYmxlQ29va2llKSBhbmQKLSAgICAgICAgICogZmlsbHMgaW4gJ291dFZhbHVlJy4gIElmIG5vdCBmb3VuZCwgcmV0dXJucyBhIG5lZ2F0aXZlIGVycm9yCi0gICAgICAgICAqIGNvZGUuCi0gICAgICAgICAqCi0gICAgICAgICAqIE5vdGUgdGhhdCB0aGlzIGZ1bmN0aW9uIGRvZXMgbm90IGRvIHJlZmVyZW5jZSB0cmF2ZXJzYWwuICBJZiB5b3Ugd2FudAotICAgICAgICAgKiB0byBmb2xsb3cgcmVmZXJlbmNlcyB0byBvdGhlciByZXNvdXJjZXMgdG8gZ2V0IHRoZSAicmVhbCIgdmFsdWUgdG8KLSAgICAgICAgICogdXNlLCB5b3UgbmVlZCB0byBjYWxsIHJlc29sdmVSZWZlcmVuY2UoKSBhZnRlciB0aGlzIGZ1bmN0aW9uLgotICAgICAgICAgKgotICAgICAgICAgKiBAcGFyYW0gcmVzSUQgQSByZXNvdXJjZSBpZGVudGlmaWVyIG5hbWluZyB0aGUgZGVzaXJlZCB0aGVtZQotICAgICAgICAgKiAgICAgICAgICAgICAgYXR0cmlidXRlLgotICAgICAgICAgKiBAcGFyYW0gb3V0VmFsdWUgRmlsbGVkIGluIHdpdGggdGhlIHRoZW1lIHZhbHVlIHRoYXQgd2FzCi0gICAgICAgICAqICAgICAgICAgICAgICAgICBmb3VuZC4KLSAgICAgICAgICoKLSAgICAgICAgICogQHJldHVybiBzc2l6ZV90IEVpdGhlciBhID49IDAgdGFibGUgaW5kZXggb3IgYSBuZWdhdGl2ZSBlcnJvciBjb2RlLgotICAgICAgICAgKi8KLSAgICAgICAgc3NpemVfdCBnZXRBdHRyaWJ1dGUodWludDMyX3QgcmVzSUQsIFJlc192YWx1ZSogb3V0VmFsdWUsCi0gICAgICAgICAgICAgICAgdWludDMyX3QqIG91dFR5cGVTcGVjRmxhZ3MgPSBOVUxMKSBjb25zdDsKLQotICAgICAgICAvKioKLSAgICAgICAgICogVGhpcyBpcyBsaWtlIFJlc1RhYmxlOjpyZXNvbHZlUmVmZXJlbmNlKCksIGJ1dCBhbHNvIHRha2VzCi0gICAgICAgICAqIGNhcmUgb2YgcmVzb2x2aW5nIGF0dHJpYnV0ZSByZWZlcmVuY2VzIHRvIHRoZSB0aGVtZS4KLSAgICAgICAgICovCi0gICAgICAgIHNzaXplX3QgcmVzb2x2ZUF0dHJpYnV0ZVJlZmVyZW5jZShSZXNfdmFsdWUqIGluT3V0VmFsdWUsCi0gICAgICAgICAgICAgICAgc3NpemVfdCBibG9ja0luZGV4LCB1aW50MzJfdCogb3V0TGFzdFJlZiA9IE5VTEwsCi0gICAgICAgICAgICAgICAgdWludDMyX3QqIGlub3V0VHlwZVNwZWNGbGFncyA9IE5VTEwpIGNvbnN0OwotCi0gICAgICAgIHZvaWQgZHVtcFRvTG9nKCkgY29uc3Q7Ci0gICAgICAgIAotICAgIHByaXZhdGU6Ci0gICAgICAgIFRoZW1lKGNvbnN0IFRoZW1lJik7Ci0gICAgICAgIFRoZW1lJiBvcGVyYXRvcj0oY29uc3QgVGhlbWUmKTsKLQotICAgICAgICBzdHJ1Y3QgdGhlbWVfZW50cnkgewotICAgICAgICAgICAgc3NpemVfdCBzdHJpbmdCbG9jazsKLSAgICAgICAgICAgIHVpbnQzMl90IHR5cGVTcGVjRmxhZ3M7Ci0gICAgICAgICAgICBSZXNfdmFsdWUgdmFsdWU7Ci0gICAgICAgIH07Ci0gICAgICAgIHN0cnVjdCB0eXBlX2luZm8gewotICAgICAgICAgICAgc2l6ZV90IG51bUVudHJpZXM7Ci0gICAgICAgICAgICB0aGVtZV9lbnRyeSogZW50cmllczsKLSAgICAgICAgfTsKLSAgICAgICAgc3RydWN0IHBhY2thZ2VfaW5mbyB7Ci0gICAgICAgICAgICBzaXplX3QgbnVtVHlwZXM7Ci0gICAgICAgICAgICB0eXBlX2luZm8gdHlwZXNbXTsKLSAgICAgICAgfTsKLQotICAgICAgICB2b2lkIGZyZWVfcGFja2FnZShwYWNrYWdlX2luZm8qIHBpKTsKLSAgICAgICAgcGFja2FnZV9pbmZvKiBjb3B5X3BhY2thZ2UocGFja2FnZV9pbmZvKiBwaSk7Ci0KLSAgICAgICAgY29uc3QgUmVzVGFibGUmIG1UYWJsZTsKLSAgICAgICAgcGFja2FnZV9pbmZvKiAgIG1QYWNrYWdlc1tSZXNfTUFYUEFDS0FHRV07Ci0gICAgfTsKLQotICAgIHZvaWQgc2V0UGFyYW1ldGVycyhjb25zdCBSZXNUYWJsZV9jb25maWcqIHBhcmFtcyk7Ci0gICAgdm9pZCBnZXRQYXJhbWV0ZXJzKFJlc1RhYmxlX2NvbmZpZyogcGFyYW1zKSBjb25zdDsKLQotICAgIC8vIFJldHJpZXZlIGFuIGlkZW50aWZpZXIgKHdoaWNoIGNhbiBiZSBwYXNzZWQgdG8gZ2V0UmVzb3VyY2UpCi0gICAgLy8gZm9yIGEgZ2l2ZW4gcmVzb3VyY2UgbmFtZS4gIFRoZSAnbmFtZScgY2FuIGJlIGZ1bGx5IHF1YWxpZmllZAotICAgIC8vICg8cGFja2FnZT46PHR5cGU+LjxiYXNlbmFtZT4pIG9yIHRoZSBwYWNrYWdlIG9yIHR5cGUgY29tcG9uZW50cwotICAgIC8vIGNhbiBiZSBkcm9wcGVkIGlmIGRlZmF1bHQgdmFsdWVzIGFyZSBzdXBwbGllZCBoZXJlLgotICAgIC8vCi0gICAgLy8gUmV0dXJucyAwIGlmIG5vIHN1Y2ggcmVzb3VyY2Ugd2FzIGZvdW5kLCBlbHNlIGEgdmFsaWQgcmVzb3VyY2UgSUQuCi0gICAgdWludDMyX3QgaWRlbnRpZmllckZvck5hbWUoY29uc3QgY2hhcjE2X3QqIG5hbWUsIHNpemVfdCBuYW1lTGVuLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNoYXIxNl90KiB0eXBlID0gMCwgc2l6ZV90IHR5cGVMZW4gPSAwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNoYXIxNl90KiBkZWZQYWNrYWdlID0gMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplX3QgZGVmUGFja2FnZUxlbiA9IDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QqIG91dFR5cGVTcGVjRmxhZ3MgPSBOVUxMKSBjb25zdDsKLQotICAgIHN0YXRpYyBib29sIGV4cGFuZFJlc291cmNlUmVmKGNvbnN0IHVpbnQxNl90KiByZWZTdHIsIHNpemVfdCByZWZMZW4sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nMTYqIG91dFBhY2thZ2UsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nMTYqIG91dFR5cGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nMTYqIG91dE5hbWUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgU3RyaW5nMTYqIGRlZlR5cGUgPSBOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFN0cmluZzE2KiBkZWZQYWNrYWdlID0gTlVMTCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyKiogb3V0RXJyb3JNc2cgPSBOVUxMKTsKLQotICAgIHN0YXRpYyBib29sIHN0cmluZ1RvSW50KGNvbnN0IGNoYXIxNl90KiBzLCBzaXplX3QgbGVuLCBSZXNfdmFsdWUqIG91dFZhbHVlKTsKLSAgICBzdGF0aWMgYm9vbCBzdHJpbmdUb0Zsb2F0KGNvbnN0IGNoYXIxNl90KiBzLCBzaXplX3QgbGVuLCBSZXNfdmFsdWUqIG91dFZhbHVlKTsKLQotICAgIC8vIFVzZWQgd2l0aCBzdHJpbmdUb1ZhbHVlLgotICAgIGNsYXNzIEFjY2Vzc29yCi0gICAgewotICAgIHB1YmxpYzoKLSAgICAgICAgaW5saW5lIHZpcnR1YWwgfkFjY2Vzc29yKCkgeyB9Ci0KLSAgICAgICAgdmlydHVhbCB1aW50MzJfdCBnZXRDdXN0b21SZXNvdXJjZShjb25zdCBTdHJpbmcxNiYgcGFja2FnZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBTdHJpbmcxNiYgdHlwZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBTdHJpbmcxNiYgbmFtZSkgY29uc3QgPSAwOwotICAgICAgICB2aXJ0dWFsIHVpbnQzMl90IGdldEN1c3RvbVJlc291cmNlV2l0aENyZWF0aW9uKGNvbnN0IFN0cmluZzE2JiBwYWNrYWdlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFN0cmluZzE2JiB0eXBlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFN0cmluZzE2JiBuYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGJvb2wgY3JlYXRlSWZOZWVkZWQgPSBmYWxzZSkgPSAwOwotICAgICAgICB2aXJ0dWFsIHVpbnQzMl90IGdldFJlbWFwcGVkUGFja2FnZSh1aW50MzJfdCBvcmlnUGFja2FnZSkgY29uc3QgPSAwOwotICAgICAgICB2aXJ0dWFsIGJvb2wgZ2V0QXR0cmlidXRlVHlwZSh1aW50MzJfdCBhdHRySUQsIHVpbnQzMl90KiBvdXRUeXBlKSA9IDA7Ci0gICAgICAgIHZpcnR1YWwgYm9vbCBnZXRBdHRyaWJ1dGVNaW4odWludDMyX3QgYXR0cklELCB1aW50MzJfdCogb3V0TWluKSA9IDA7Ci0gICAgICAgIHZpcnR1YWwgYm9vbCBnZXRBdHRyaWJ1dGVNYXgodWludDMyX3QgYXR0cklELCB1aW50MzJfdCogb3V0TWF4KSA9IDA7Ci0gICAgICAgIHZpcnR1YWwgYm9vbCBnZXRBdHRyaWJ1dGVFbnVtKHVpbnQzMl90IGF0dHJJRCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY2hhcjE2X3QqIG5hbWUsIHNpemVfdCBuYW1lTGVuLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBSZXNfdmFsdWUqIG91dFZhbHVlKSA9IDA7Ci0gICAgICAgIHZpcnR1YWwgYm9vbCBnZXRBdHRyaWJ1dGVGbGFncyh1aW50MzJfdCBhdHRySUQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogbmFtZSwgc2l6ZV90IG5hbWVMZW4sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBSZXNfdmFsdWUqIG91dFZhbHVlKSA9IDA7Ci0gICAgICAgIHZpcnR1YWwgdWludDMyX3QgZ2V0QXR0cmlidXRlTDEwTih1aW50MzJfdCBhdHRySUQpID0gMDsKLSAgICAgICAgdmlydHVhbCBib29sIGdldExvY2FsaXphdGlvblNldHRpbmcoKSA9IDA7Ci0gICAgICAgIHZpcnR1YWwgdm9pZCByZXBvcnRFcnJvcih2b2lkKiBhY2Nlc3NvckNvb2tpZSwgY29uc3QgY2hhciogZm10LCAuLi4pID0gMDsKLSAgICB9OwotCi0gICAgLy8gQ29udmVydCBhIHN0cmluZyB0byBhIHJlc291cmNlIHZhbHVlLiAgSGFuZGxlcyBzdGFuZGFyZCAiQHJlcyIsCi0gICAgLy8gIiNjb2xvciIsICIxMjMiLCBhbmQgIjB4MWJkIiB0eXBlczsgcGVyZm9ybXMgZXNjYXBpbmcgb2Ygc3RyaW5ncy4KLSAgICAvLyBUaGUgcmVzdWx0aW5nIHZhbHVlIGlzIHBsYWNlZCBpbiAnb3V0VmFsdWUnOyBpZiBpdCBpcyBhIHN0cmluZyB0eXBlLAotICAgIC8vICdvdXRTdHJpbmcnIHJlY2VpdmVzIHRoZSBzdHJpbmcuICBJZiAnYXR0cklEJyBpcyBzdXBwbGllZCwgdGhlIHZhbHVlIGlzCi0gICAgLy8gdHlwZSBjaGVja2VkIGFnYWluc3QgdGhpcyBhdHRyaWJ1dGUgYW5kIGl0IGlzIHVzZWQgdG8gcGVyZm9ybSBlbnVtCi0gICAgLy8gZXZhbHVhdGlvbi4gIElmICdhY2NjZXNzb3InIGlzIHN1cHBsaWVkLCBpdCB3aWxsIGJlIHVzZWQgdG8gYXR0ZW1wdCB0bwotICAgIC8vIHJlc29sdmUgcmVzb3VyY2VzIHRoYXQgZG8gbm90IGV4aXN0IGluIHRoaXMgUmVzVGFibGUuICBJZiAnYXR0clR5cGUnIGlzCi0gICAgLy8gc3VwcGxpZWQsIHRoZSB2YWx1ZSB3aWxsIGJlIHR5cGUgY2hlY2tlZCBmb3IgdGhpcyBmb3JtYXQgaWYgJ2F0dHJJRCcKLSAgICAvLyBpcyBub3Qgc3VwcGxpZWQgb3IgZm91bmQuCi0gICAgYm9vbCBzdHJpbmdUb1ZhbHVlKFJlc192YWx1ZSogb3V0VmFsdWUsIFN0cmluZzE2KiBvdXRTdHJpbmcsCi0gICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNoYXIxNl90KiBzLCBzaXplX3QgbGVuLAotICAgICAgICAgICAgICAgICAgICAgICBib29sIHByZXNlcnZlU3BhY2VzLCBib29sIGNvZXJjZVR5cGUsCi0gICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGF0dHJJRCA9IDAsCi0gICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFN0cmluZzE2KiBkZWZUeXBlID0gTlVMTCwKLSAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgU3RyaW5nMTYqIGRlZlBhY2thZ2UgPSBOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICBBY2Nlc3NvciogYWNjZXNzb3IgPSBOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiBhY2Nlc3NvckNvb2tpZSA9IE5VTEwsCi0gICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGF0dHJUeXBlID0gUmVzVGFibGVfbWFwOjpUWVBFX0FOWSwKLSAgICAgICAgICAgICAgICAgICAgICAgYm9vbCBlbmZvcmNlUHJpdmF0ZSA9IHRydWUpIGNvbnN0OwotCi0gICAgLy8gUGVyZm9ybSBwcm9jZXNzaW5nIG9mIGVzY2FwZXMgYW5kIHF1b3RlcyBpbiBhIHN0cmluZy4KLSAgICBzdGF0aWMgYm9vbCBjb2xsZWN0U3RyaW5nKFN0cmluZzE2KiBvdXRTdHJpbmcsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogcywgc2l6ZV90IGxlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2wgcHJlc2VydmVTcGFjZXMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyKiogb3V0RXJyb3JNc2cgPSBOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYm9vbCBhcHBlbmQgPSBmYWxzZSk7Ci0KLSAgICBzaXplX3QgZ2V0QmFzZVBhY2thZ2VDb3VudCgpIGNvbnN0OwotICAgIGNvbnN0IGNoYXIxNl90KiBnZXRCYXNlUGFja2FnZU5hbWUoc2l6ZV90IGlkeCkgY29uc3Q7Ci0gICAgdWludDMyX3QgZ2V0QmFzZVBhY2thZ2VJZChzaXplX3QgaWR4KSBjb25zdDsKLQotICAgIHNpemVfdCBnZXRUYWJsZUNvdW50KCkgY29uc3Q7Ci0gICAgY29uc3QgUmVzU3RyaW5nUG9vbCogZ2V0VGFibGVTdHJpbmdCbG9jayhzaXplX3QgaW5kZXgpIGNvbnN0OwotICAgIHZvaWQqIGdldFRhYmxlQ29va2llKHNpemVfdCBpbmRleCkgY29uc3Q7Ci0KLSAgICAvLyBSZXR1cm4gdGhlIGNvbmZpZ3VyYXRpb25zIChSZXNUYWJsZV9jb25maWcpIHRoYXQgd2Uga25vdyBhYm91dAotICAgIHZvaWQgZ2V0Q29uZmlndXJhdGlvbnMoVmVjdG9yPFJlc1RhYmxlX2NvbmZpZz4qIGNvbmZpZ3MpIGNvbnN0OwotCi0gICAgdm9pZCBnZXRMb2NhbGVzKFZlY3RvcjxTdHJpbmc4PiogbG9jYWxlcykgY29uc3Q7Ci0KLSNpZm5kZWYgSEFWRV9BTkRST0lEX09TCi0gICAgdm9pZCBwcmludCgpIGNvbnN0OwotI2VuZGlmCi0KLXByaXZhdGU6Ci0gICAgc3RydWN0IEhlYWRlcjsKLSAgICBzdHJ1Y3QgVHlwZTsKLSAgICBzdHJ1Y3QgUGFja2FnZTsKLSAgICBzdHJ1Y3QgUGFja2FnZUdyb3VwOwotICAgIHN0cnVjdCBiYWdfc2V0OwotCi0gICAgc3RhdHVzX3QgYWRkKGNvbnN0IHZvaWQqIGRhdGEsIHNpemVfdCBzaXplLCB2b2lkKiBjb29raWUsCi0gICAgICAgICAgICAgICAgIEFzc2V0KiBhc3NldCwgYm9vbCBjb3B5RGF0YSk7Ci0KLSAgICBzc2l6ZV90IGdldFJlc291cmNlUGFja2FnZUluZGV4KHVpbnQzMl90IHJlc0lEKSBjb25zdDsKLSAgICBzc2l6ZV90IGdldEVudHJ5KAotICAgICAgICBjb25zdCBQYWNrYWdlKiBwYWNrYWdlLCBpbnQgdHlwZUluZGV4LCBpbnQgZW50cnlJbmRleCwKLSAgICAgICAgY29uc3QgUmVzVGFibGVfY29uZmlnKiBjb25maWcsCi0gICAgICAgIGNvbnN0IFJlc1RhYmxlX3R5cGUqKiBvdXRUeXBlLCBjb25zdCBSZXNUYWJsZV9lbnRyeSoqIG91dEVudHJ5LAotICAgICAgICBjb25zdCBUeXBlKiogb3V0VHlwZUNsYXNzKSBjb25zdDsKLSAgICBzdGF0dXNfdCBwYXJzZVBhY2thZ2UoCi0gICAgICAgIGNvbnN0IFJlc1RhYmxlX3BhY2thZ2UqIGNvbnN0IHBrZywgY29uc3QgSGVhZGVyKiBjb25zdCBoZWFkZXIpOwotCi0gICAgbXV0YWJsZSBNdXRleCAgICAgICAgICAgICAgIG1Mb2NrOwotCi0gICAgc3RhdHVzX3QgICAgICAgICAgICAgICAgICAgIG1FcnJvcjsKLQotICAgIFJlc1RhYmxlX2NvbmZpZyAgICAgICAgICAgICBtUGFyYW1zOwotCi0gICAgLy8gQXJyYXkgb2YgYWxsIHJlc291cmNlIHRhYmxlcy4KLSAgICBWZWN0b3I8SGVhZGVyKj4gICAgICAgICAgICAgbUhlYWRlcnM7Ci0KLSAgICAvLyBBcnJheSBvZiBwYWNrYWdlcyBpbiBhbGwgcmVzb3VyY2UgdGFibGVzLgotICAgIFZlY3RvcjxQYWNrYWdlR3JvdXAqPiAgICAgICBtUGFja2FnZUdyb3VwczsKLQotICAgIC8vIE1hcHBpbmcgZnJvbSByZXNvdXJjZSBwYWNrYWdlIElEcyB0byBpbmRpY2VzIGludG8gdGhlIGludGVybmFsCi0gICAgLy8gcGFja2FnZSBhcnJheS4KLSAgICB1aW50OF90ICAgICAgICAgICAgICAgICAgICAgbVBhY2thZ2VNYXBbMjU2XTsKLX07Ci0KLX0gICAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gX0xJQlNfVVRJTFNfUkVTT1VSQ0VfVFlQRVNfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9TaGFyZWRCdWZmZXIuaCBiL2luY2x1ZGUvdXRpbHMvU2hhcmVkQnVmZmVyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI0NTA4YjAuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9TaGFyZWRCdWZmZXIuaAorKysgL2Rldi9udWxsCkBAIC0xLDE0NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1NIQVJFRF9CVUZGRVJfSAotI2RlZmluZSBBTkRST0lEX1NIQVJFRF9CVUZGRVJfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIFNoYXJlZEJ1ZmZlcgotewotcHVibGljOgotCi0gICAgLyogZmxhZ3MgdG8gdXNlIHdpdGggcmVsZWFzZSgpICovCi0gICAgZW51bSB7Ci0gICAgICAgIGVLZWVwU3RvcmFnZSA9IDB4MDAwMDAwMDEKLSAgICB9OwotCi0gICAgLyohIGFsbG9jYXRlIGEgYnVmZmVyIG9mIHNpemUgJ3NpemUnIGFuZCBhY3F1aXJlKCkgaXQuCi0gICAgICogIGNhbGwgcmVsZWFzZSgpIHRvIGZyZWUgaXQuCi0gICAgICovCi0gICAgc3RhdGljICAgICAgICAgIFNoYXJlZEJ1ZmZlciogICAgICAgICAgIGFsbG9jKHNpemVfdCBzaXplKTsKLSAgICAKLSAgICAvKiEgZnJlZSB0aGUgbWVtb3J5IGFzc29jaWF0ZWQgd2l0aCB0aGUgU2hhcmVkQnVmZmVyLgotICAgICAqIEZhaWxzIGlmIHRoZXJlIGFyZSBhbnkgdXNlcnMgYXNzb2NpYXRlZCB3aXRoIHRoaXMgU2hhcmVkQnVmZmVyLgotICAgICAqIEluIG90aGVyIHdvcmRzLCB0aGUgYnVmZmVyIG11c3QgaGF2ZSBiZWVuIHJlbGVhc2UgYnkgYWxsIGl0cwotICAgICAqIHVzZXJzLgotICAgICAqLwotICAgIHN0YXRpYyAgICAgICAgICBzc2l6ZV90ICAgICAgICAgICAgICAgICBkZWFsbG9jKGNvbnN0IFNoYXJlZEJ1ZmZlciogcmVsZWFzZWQpOwotICAgIAotICAgIC8vISBnZXQgdGhlIFNoYXJlZEJ1ZmZlciBmcm9tIHRoZSBkYXRhIHBvaW50ZXIKLSAgICBzdGF0aWMgIGlubGluZSAgY29uc3QgU2hhcmVkQnVmZmVyKiAgICAgc2hhcmVkQnVmZmVyKGNvbnN0IHZvaWQqIGRhdGEpOwotCi0gICAgLy8hIGFjY2VzcyB0aGUgZGF0YSBmb3IgcmVhZAotICAgIGlubGluZSAgICAgICAgICBjb25zdCB2b2lkKiAgICAgICAgICAgICBkYXRhKCkgY29uc3Q7Ci0gICAgCi0gICAgLy8hIGFjY2VzcyB0aGUgZGF0YSBmb3IgcmVhZC93cml0ZQotICAgIGlubGluZSAgICAgICAgICB2b2lkKiAgICAgICAgICAgICAgICAgICBkYXRhKCk7Ci0KLSAgICAvLyEgZ2V0IHNpemUgb2YgdGhlIGJ1ZmZlcgotICAgIGlubGluZSAgICAgICAgICBzaXplX3QgICAgICAgICAgICAgICAgICBzaXplKCkgY29uc3Q7Ci0gCi0gICAgLy8hIGdldCBiYWNrIGEgU2hhcmVkQnVmZmVyIG9iamVjdCBmcm9tIGl0cyBkYXRhCi0gICAgc3RhdGljICBpbmxpbmUgIFNoYXJlZEJ1ZmZlciogICAgICAgICAgIGJ1ZmZlckZyb21EYXRhKHZvaWQqIGRhdGEpOwotICAgIAotICAgIC8vISBnZXQgYmFjayBhIFNoYXJlZEJ1ZmZlciBvYmplY3QgZnJvbSBpdHMgZGF0YQotICAgIHN0YXRpYyAgaW5saW5lICBjb25zdCBTaGFyZWRCdWZmZXIqICAgICBidWZmZXJGcm9tRGF0YShjb25zdCB2b2lkKiBkYXRhKTsKLQotICAgIC8vISBnZXQgdGhlIHNpemUgb2YgYSBTaGFyZWRCdWZmZXIgb2JqZWN0IGZyb20gaXRzIGRhdGEKLSAgICBzdGF0aWMgIGlubGluZSAgc2l6ZV90ICAgICAgICAgICAgICAgICAgc2l6ZUZyb21EYXRhKGNvbnN0IHZvaWQqIGRhdGEpOwotICAgIAotICAgIC8vISBlZGl0IHRoZSBidWZmZXIgKGdldCBhIHdyaXR0YWJsZSwgb3Igbm9uLWNvbnN0LCB2ZXJzaW9uIG9mIGl0KQotICAgICAgICAgICAgICAgICAgICBTaGFyZWRCdWZmZXIqICAgICAgICAgICBlZGl0KCkgY29uc3Q7Ci0KLSAgICAvLyEgZWRpdCB0aGUgYnVmZmVyLCByZXNpemluZyBpZiBuZWVkZWQKLSAgICAgICAgICAgICAgICAgICAgU2hhcmVkQnVmZmVyKiAgICAgICAgICAgZWRpdFJlc2l6ZShzaXplX3Qgc2l6ZSkgY29uc3Q7Ci0KLSAgICAvLyEgbGlrZSBlZGl0KCkgYnV0IGZhaWxzIGlmIGEgY29weSBpcyByZXF1aXJlZAotICAgICAgICAgICAgICAgICAgICBTaGFyZWRCdWZmZXIqICAgICAgICAgICBhdHRlbXB0RWRpdCgpIGNvbnN0OwotICAgIAotICAgIC8vISByZXNpemUgYW5kIGVkaXQgdGhlIGJ1ZmZlciwgbG9vc2UgaXQncyBjb250ZW50LgotICAgICAgICAgICAgICAgICAgICBTaGFyZWRCdWZmZXIqICAgICAgICAgICByZXNldChzaXplX3Qgc2l6ZSkgY29uc3Q7Ci0KLSAgICAvLyEgYWNxdWlyZS9yZWxlYXNlIGEgcmVmZXJlbmNlIG9uIHRoaXMgYnVmZmVyCi0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgICAgIGFjcXVpcmUoKSBjb25zdDsKLSAgICAgICAgICAgICAgICAgICAgCi0gICAgLyohIHJlbGVhc2UgYSByZWZlcmVuY2Ugb24gdGhpcyBidWZmZXIsIHdpdGggdGhlIG9wdGlvbiBvZiBub3QKLSAgICAgKiBmcmVlaW5nIHRoZSBtZW1vcnkgYXNzb2NpYXRlZCB3aXRoIGl0IGlmIGl0IHdhcyB0aGUgbGFzdCByZWZlcmVuY2UKLSAgICAgKiByZXR1cm5zIHRoZSBwcmV2aW91cyByZWZlcmVuY2UgY291bnQKLSAgICAgKi8gICAgIAotICAgICAgICAgICAgICAgICAgICBpbnQzMl90ICAgICAgICAgICAgICAgICByZWxlYXNlKHVpbnQzMl90IGZsYWdzID0gMCkgY29uc3Q7Ci0gICAgCi0gICAgLy8hIHJldHVybnMgd2V0aGVyIG9yIG5vdCB3ZSdyZSB0aGUgb25seSBvd25lcgotICAgIGlubGluZSAgICAgICAgICBib29sICAgICAgICAgICAgICAgICAgICBvbmx5T3duZXIoKSBjb25zdDsKLSAgICAKLQotcHJpdmF0ZToKLSAgICAgICAgaW5saW5lIFNoYXJlZEJ1ZmZlcigpIHsgfQotICAgICAgICBpbmxpbmUgflNoYXJlZEJ1ZmZlcigpIHsgfQotICAgICAgICBpbmxpbmUgU2hhcmVkQnVmZmVyKGNvbnN0IFNoYXJlZEJ1ZmZlciYpOwotIAotICAgICAgICAvLyAxNiBieXRlcy4gbXVzdCBiZSBzaXplZCB0byBwcmVzZXJ2ZSBjb3JyZWN0IGFsaW5nbWVudC4KLSAgICAgICAgbXV0YWJsZSBpbnQzMl90ICAgICAgICBtUmVmczsKLSAgICAgICAgICAgICAgICBzaXplX3QgICAgICAgICBtU2l6ZTsKLSAgICAgICAgICAgICAgICB1aW50MzJfdCAgICAgICBtUmVzZXJ2ZWRbMl07Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY29uc3QgU2hhcmVkQnVmZmVyKiBTaGFyZWRCdWZmZXI6OnNoYXJlZEJ1ZmZlcihjb25zdCB2b2lkKiBkYXRhKSB7Ci0gICAgcmV0dXJuIGRhdGEgPyByZWludGVycHJldF9jYXN0PGNvbnN0IFNoYXJlZEJ1ZmZlciAqPihkYXRhKS0xIDogMDsKLX0KLQotY29uc3Qgdm9pZCogU2hhcmVkQnVmZmVyOjpkYXRhKCkgY29uc3QgewotICAgIHJldHVybiB0aGlzICsgMTsKLX0KLQotdm9pZCogU2hhcmVkQnVmZmVyOjpkYXRhKCkgewotICAgIHJldHVybiB0aGlzICsgMTsKLX0KLQotc2l6ZV90IFNoYXJlZEJ1ZmZlcjo6c2l6ZSgpIGNvbnN0IHsKLSAgICByZXR1cm4gbVNpemU7Ci19Ci0KLVNoYXJlZEJ1ZmZlciogU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YSh2b2lkKiBkYXRhKQotewotICAgIHJldHVybiAoKFNoYXJlZEJ1ZmZlciopZGF0YSktMTsKLX0KLSAgICAKLWNvbnN0IFNoYXJlZEJ1ZmZlciogU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShjb25zdCB2b2lkKiBkYXRhKQotewotICAgIHJldHVybiAoKGNvbnN0IFNoYXJlZEJ1ZmZlciopZGF0YSktMTsKLX0KLQotc2l6ZV90IFNoYXJlZEJ1ZmZlcjo6c2l6ZUZyb21EYXRhKGNvbnN0IHZvaWQqIGRhdGEpCi17Ci0gICAgcmV0dXJuICgoKGNvbnN0IFNoYXJlZEJ1ZmZlciopZGF0YSktMSktPm1TaXplOwotfQotCi1ib29sIFNoYXJlZEJ1ZmZlcjo6b25seU93bmVyKCkgY29uc3QgewotICAgIHJldHVybiAobVJlZnMgPT0gMSk7Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLy8gQU5EUk9JRF9WRUNUT1JfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9Tb2NrZXQuaCBiL2luY2x1ZGUvdXRpbHMvU29ja2V0LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDhiN2Y0MDYuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9Tb2NrZXQuaAorKysgL2Rldi9udWxsCkBAIC0xLDgwICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gU29ja2V0IGNsYXNzLiAgTW9kZWxlZCBhZnRlciBKYXZhIGNsYXNzZXMuCi0vLwotI2lmbmRlZiBfUlVOVElNRV9TT0NLRVRfSAotI2RlZmluZSBfUlVOVElNRV9TT0NLRVRfSAotCi0jaW5jbHVkZSA8dXRpbHMvaW5ldF9hZGRyZXNzLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyoKLSAqIEJhc2ljIHNvY2tldCBjbGFzcywgbmVlZGVkIHRvIGFic3RyYWN0IGF3YXkgdGhlIGRpZmZlcmVuY2VzIGJldHdlZW4KLSAqIEJTRCBzb2NrZXRzIGFuZCBXaW5Tb2NrLiAgVGhpcyBlc3RhYmxpc2hlcyBhIHN0cmVhbWluZyBuZXR3b3JrCi0gKiBjb25uZWN0aW9uIChUQ1AvSVApIHRvIHNvbWVib2R5LgotICovCi1jbGFzcyBTb2NrZXQgewotcHVibGljOgotICAgIFNvY2tldCh2b2lkKTsKLSAgICB+U29ja2V0KHZvaWQpOwotCi0gICAgLy8gQ3JlYXRlIGEgY29ubmVjdGlvbiB0byBzb21ld2hlcmUuCi0gICAgLy8gUmV0dXJuIDAgb24gc3VjY2Vzcy4KLSAgICBpbnQgY29ubmVjdChjb25zdCBjaGFyKiBob3N0LCBpbnQgcG9ydCk7Ci0gICAgaW50IGNvbm5lY3QoY29uc3QgSW5ldEFkZHJlc3MqIGFkZHIsIGludCBwb3J0KTsKLQotCi0gICAgLy8gQ2xvc2UgdGhlIHNvY2tldC4gIERvbid0IHRyeSB0byB1c2UgdGhpcyBvYmplY3QgYWdhaW4gYWZ0ZXIKLSAgICAvLyBjYWxsaW5nIHRoaXMuICBSZXR1cm5zIGZhbHNlIG9uIGZhaWx1cmUuCi0gICAgYm9vbCBjbG9zZSh2b2lkKTsKLQotICAgIC8vIElmIHdlIGNyZWF0ZWQgdGhlIHNvY2tldCB3aXRob3V0IGFuIGFkZHJlc3MsIHdlIGNhbiB1c2UgdGhlc2UKLSAgICAvLyB0byBmaW5pc2ggdGhlIGNvbm5lY3Rpb24uICBSZXR1cm5zIDAgb24gc3VjY2Vzcy4KLSAgICBpbnQgYmluZChjb25zdCBTb2NrZXRBZGRyZXNzJiBiaW5kUG9pbnQpOwotICAgIGludCBjb25uZWN0KGNvbnN0IFNvY2tldEFkZHJlc3MmIGVuZFBvaW50KTsKLQotICAgIC8vIEhlcmUgd2UgZGV2aWF0ZSBmcm9tIHRoZSB0cmFkaXRpb25hbCBvYmplY3Qtb3JpZW50ZWQgZmFuY2luZXNzCi0gICAgLy8gYW5kIGp1c3QgcHJvdmlkZSByZWFkL3dyaXRlIG9wZXJhdG9ycyBpbnN0ZWFkIG9mIGdldHRlcnMgZm9yCi0gICAgLy8gb2JqZWN0cyB0aGF0IGFic3RyYWN0IGEgc3RyZWFtLgotICAgIC8vCi0gICAgLy8gU3RhbmRhcmQgcmVhZC93cml0ZSBzZW1hbnRpY3MuCi0gICAgaW50IHJlYWQodm9pZCogYnVmLCBzc2l6ZV90IGxlbikgY29uc3Q7Ci0gICAgaW50IHdyaXRlKGNvbnN0IHZvaWQqIGJ1Ziwgc3NpemVfdCBsZW4pIGNvbnN0OwotCi0gICAgLy8gVGhpcyBtdXN0IGJlIGNhbGxlZCBvbmNlLCBhdCBwcm9ncmFtIHN0YXJ0dXAuCi0gICAgc3RhdGljIGJvb2wgYm9vdEluaXQodm9pZCk7Ci0gICAgc3RhdGljIHZvaWQgZmluYWxTaHV0ZG93bih2b2lkKTsKLQotcHJpdmF0ZToKLSAgICAvLyBJbnRlcm5hbCBmdW5jdGlvbiB0aGF0IGVzdGFibGlzaGVzIGEgY29ubmVjdGlvbi4KLSAgICBpbnQgZG9Db25uZWN0KGNvbnN0IEluZXRTb2NrZXRBZGRyZXNzJiBhZGRyKTsKLQotICAgIHVuc2lnbmVkIGxvbmcgICBtU29jazsgICAgICAvLyBob2xkcyBTT0NLRVQgb3IgaW50Ci0KLSAgICBzdGF0aWMgYm9vbCAgICAgbUJvb3RJbml0aWFsaXplZDsKLX07Ci0KLQotLy8gZGVidWcgLS0gdW5pdCB0ZXN0cwotdm9pZCBUZXN0U29ja2V0cyh2b2lkKTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIF9SVU5USU1FX1NPQ0tFVF9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL1NvcnRlZFZlY3Rvci5oIGIvaW5jbHVkZS91dGlscy9Tb3J0ZWRWZWN0b3IuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzhhNjE1My4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL1NvcnRlZFZlY3Rvci5oCisrKyAvZGV2L251bGwKQEAgLTEsMjgyICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfU09SVEVEX1ZFQ1RPUl9ICi0jZGVmaW5lIEFORFJPSURfU09SVEVEX1ZFQ1RPUl9ICi0KLSNpbmNsdWRlIDxhc3NlcnQuaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL1ZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL1ZlY3RvckltcGwuaD4KLSNpbmNsdWRlIDx1dGlscy9UeXBlSGVscGVycy5oPgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi10ZW1wbGF0ZSA8Y2xhc3MgVFlQRT4KLWNsYXNzIFNvcnRlZFZlY3RvciA6IHByaXZhdGUgU29ydGVkVmVjdG9ySW1wbAotewotcHVibGljOgotICAgICAgICAgICAgdHlwZWRlZiBUWVBFICAgIHZhbHVlX3R5cGU7Ci0gICAgCi0gICAgLyohIAotICAgICAqIENvbnN0cnVjdG9ycyBhbmQgZGVzdHJ1Y3RvcnMKLSAgICAgKi8KLSAgICAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBTb3J0ZWRWZWN0b3IoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBTb3J0ZWRWZWN0b3IoY29uc3QgU29ydGVkVmVjdG9yPFRZUEU+JiByaHMpOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgICAgIH5Tb3J0ZWRWZWN0b3IoKTsKLQotICAgIC8qISBjb3B5IG9wZXJhdG9yICovCi0gICAgY29uc3QgU29ydGVkVmVjdG9yPFRZUEU+JiAgIG9wZXJhdG9yID0gKGNvbnN0IFNvcnRlZFZlY3RvcjxUWVBFPiYgcmhzKSBjb25zdDsgICAgCi0gICAgU29ydGVkVmVjdG9yPFRZUEU+JiAgICAgICAgIG9wZXJhdG9yID0gKGNvbnN0IFNvcnRlZFZlY3RvcjxUWVBFPiYgcmhzKTsgICAgCi0KLSAgICAvKgotICAgICAqIGVtcHR5IHRoZSB2ZWN0b3IKLSAgICAgKi8KLQotICAgIGlubGluZSAgdm9pZCAgICAgICAgICAgIGNsZWFyKCkgICAgICAgICAgICAgeyBWZWN0b3JJbXBsOjpjbGVhcigpOyB9Ci0KLSAgICAvKiEgCi0gICAgICogdmVjdG9yIHN0YXRzCi0gICAgICovCi0KLSAgICAvLyEgcmV0dXJucyBudW1iZXIgb2YgaXRlbXMgaW4gdGhlIHZlY3RvcgotICAgIGlubGluZSAgc2l6ZV90ICAgICAgICAgIHNpemUoKSBjb25zdCAgICAgICAgICAgICAgICB7IHJldHVybiBWZWN0b3JJbXBsOjpzaXplKCk7IH0KLSAgICAvLyEgcmV0dXJucyB3ZXRoZXIgb3Igbm90IHRoZSB2ZWN0b3IgaXMgZW1wdHkKLSAgICBpbmxpbmUgIGJvb2wgICAgICAgICAgICBpc0VtcHR5KCkgY29uc3QgICAgICAgICAgICAgeyByZXR1cm4gVmVjdG9ySW1wbDo6aXNFbXB0eSgpOyB9Ci0gICAgLy8hIHJldHVybnMgaG93IG1hbnkgaXRlbXMgY2FuIGJlIHN0b3JlZCB3aXRob3V0IHJlYWxsb2NhdGluZyB0aGUgYmFja2luZyBzdG9yZQotICAgIGlubGluZSAgc2l6ZV90ICAgICAgICAgIGNhcGFjaXR5KCkgY29uc3QgICAgICAgICAgICB7IHJldHVybiBWZWN0b3JJbXBsOjpjYXBhY2l0eSgpOyB9Ci0gICAgLy8hIHNldHN0IHRoZSBjYXBhY2l0eS4gY2FwYWNpdHkgY2FuIG5ldmVyIGJlIHJlZHVjZWQgbGVzcyB0aGFuIHNpemUoKQotICAgIGlubGluZSAgc3NpemVfdCAgICAgICAgIHNldENhcGFjaXR5KHNpemVfdCBzaXplKSAgICB7IHJldHVybiBWZWN0b3JJbXBsOjpzZXRDYXBhY2l0eShzaXplKTsgfQotCi0gICAgLyohIAotICAgICAqIEMtc3R5bGUgYXJyYXkgYWNjZXNzCi0gICAgICovCi0gICAgIAotICAgIC8vISByZWFkLW9ubHkgQy1zdHlsZSBhY2Nlc3MgCi0gICAgaW5saW5lICBjb25zdCBUWVBFKiAgICAgYXJyYXkoKSBjb25zdDsKLQotICAgIC8vISByZWFkLXdyaXRlIEMtc3R5bGUgYWNjZXNzLiBCRSBWRVJZIENBUkVGVUwgd2hlbiBtb2RpZnlpbmcgdGhlIGFycmF5Ci0gICAgLy8hIHlvdSB1c3Qga2VlcCBpdCBzb3J0ZWQhIFlvdSB1c3VhbGx5IGRvbid0IHVzZSB0aGlzIGZ1bmN0aW9uLgotICAgICAgICAgICAgVFlQRSogICAgICAgICAgIGVkaXRBcnJheSgpOwotCi0gICAgICAgICAgICAvLyEgZmluZHMgdGhlIGluZGV4IG9mIGFuIGl0ZW0KLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICBpbmRleE9mKGNvbnN0IFRZUEUmIGl0ZW0pIGNvbnN0OwotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyEgZmluZHMgd2hlcmUgdGhpcyBpdGVtIHNob3VsZCBiZSBpbnNlcnRlZAotICAgICAgICAgICAgc2l6ZV90ICAgICAgICAgIG9yZGVyT2YoY29uc3QgVFlQRSYgaXRlbSkgY29uc3Q7Ci0gICAgICAgICAgICAKLSAgICAKLSAgICAvKiEgCi0gICAgICogYWNjZXNzb3JzCi0gICAgICovCi0KLSAgICAvLyEgcmVhZC1vbmx5IGFjY2VzcyB0byBhbiBpdGVtIGF0IGEgZ2l2ZW4gaW5kZXgKLSAgICBpbmxpbmUgIGNvbnN0IFRZUEUmICAgICBvcGVyYXRvciBbXSAoc2l6ZV90IGluZGV4KSBjb25zdDsKLSAgICAvLyEgYWx0ZXJuYXRlIG5hbWUgZm9yIG9wZXJhdG9yIFtdCi0gICAgaW5saW5lICBjb25zdCBUWVBFJiAgICAgaXRlbUF0KHNpemVfdCBpbmRleCkgY29uc3Q7Ci0gICAgLy8hIHN0YWNrLXVzYWdlIG9mIHRoZSB2ZWN0b3IuIHJldHVybnMgdGhlIHRvcCBvZiB0aGUgc3RhY2sgKGxhc3QgZWxlbWVudCkKLSAgICAgICAgICAgIGNvbnN0IFRZUEUmICAgICB0b3AoKSBjb25zdDsKLSAgICAvLyEgc2FtZSBhcyBvcGVyYXRvciBbXSwgYnV0IGFsbG93cyB0byBhY2Nlc3MgdGhlIHZlY3RvciBiYWNrd2FyZCAoZnJvbSB0aGUgZW5kKSB3aXRoIGEgbmVnYXRpdmUgaW5kZXgKLSAgICAgICAgICAgIGNvbnN0IFRZUEUmICAgICBtaXJyb3JJdGVtQXQoc3NpemVfdCBpbmRleCkgY29uc3Q7Ci0KLSAgICAvKiEKLSAgICAgKiBtb2RpZmluZyB0aGUgYXJyYXkKLSAgICAgKi8KLQotICAgICAgICAgICAgLy8hIGFkZCBhbiBpdGVtIGluIHRoZSByaWdodCBwbGFjZSAoYW5kIHJlcGxhY2UgdGhlIG9uZSB0aGF0IGlzIHRoZXJlKQotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIGFkZChjb25zdCBUWVBFJiBpdGVtKTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8hIGVkaXRJdGVtQXQoKSBNVVNUIE5PVCBjaGFuZ2UgdGhlIG9yZGVyIG9mIHRoaXMgaXRlbQotICAgICAgICAgICAgVFlQRSYgICAgICAgICAgIGVkaXRJdGVtQXQoc2l6ZV90IGluZGV4KSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuICooIHN0YXRpY19jYXN0PFRZUEUgKj4oVmVjdG9ySW1wbDo6ZWRpdEl0ZW1Mb2NhdGlvbihpbmRleCkpICk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vISBtZXJnZXMgYSB2ZWN0b3IgaW50byB0aGlzIG9uZQotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIG1lcmdlKGNvbnN0IFZlY3RvcjxUWVBFPiYgdmVjdG9yKTsKLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICBtZXJnZShjb25zdCBTb3J0ZWRWZWN0b3I8VFlQRT4mIHZlY3Rvcik7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8vISByZW1vdmVzIGFuIGl0ZW0KLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICByZW1vdmUoY29uc3QgVFlQRSYpOwotCi0gICAgLy8hIHJlbW92ZSBzZXZlcmFsIGl0ZW1zCi0gICAgaW5saW5lICBzc2l6ZV90ICAgICAgICAgcmVtb3ZlSXRlbXNBdChzaXplX3QgaW5kZXgsIHNpemVfdCBjb3VudCA9IDEpOwotICAgIC8vISByZW1vdmUgb25lIGl0ZW0KLSAgICBpbmxpbmUgIHNzaXplX3QgICAgICAgICByZW1vdmVBdChzaXplX3QgaW5kZXgpICB7IHJldHVybiByZW1vdmVJdGVtc0F0KGluZGV4KTsgfQotICAgICAgICAgICAgCi1wcm90ZWN0ZWQ6Ci0gICAgdmlydHVhbCB2b2lkICAgIGRvX2NvbnN0cnVjdCh2b2lkKiBzdG9yYWdlLCBzaXplX3QgbnVtKSBjb25zdDsKLSAgICB2aXJ0dWFsIHZvaWQgICAgZG9fZGVzdHJveSh2b2lkKiBzdG9yYWdlLCBzaXplX3QgbnVtKSBjb25zdDsKLSAgICB2aXJ0dWFsIHZvaWQgICAgZG9fY29weSh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdDsKLSAgICB2aXJ0dWFsIHZvaWQgICAgZG9fc3BsYXQodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogaXRlbSwgc2l6ZV90IG51bSkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkICAgIGRvX21vdmVfZm9yd2FyZCh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdDsKLSAgICB2aXJ0dWFsIHZvaWQgICAgZG9fbW92ZV9iYWNrd2FyZCh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdDsKLSAgICB2aXJ0dWFsIGludCAgICAgZG9fY29tcGFyZShjb25zdCB2b2lkKiBsaHMsIGNvbnN0IHZvaWQqIHJocykgY29uc3Q7Ci19OwotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gTm8gdXNlciBzZXJ2aWNlYWJsZSBwYXJ0cyBmcm9tIGhlcmUuLi4KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLVNvcnRlZFZlY3RvcjxUWVBFPjo6U29ydGVkVmVjdG9yKCkKLSAgICA6IFNvcnRlZFZlY3RvckltcGwoc2l6ZW9mKFRZUEUpLAotICAgICAgICAgICAgICAgICgodHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9jdG9yICAgPyBIQVNfVFJJVklBTF9DVE9SICAgOiAwKQotICAgICAgICAgICAgICAgIHwodHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9kdG9yICAgPyBIQVNfVFJJVklBTF9EVE9SICAgOiAwKQotICAgICAgICAgICAgICAgIHwodHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9jb3B5ICAgPyBIQVNfVFJJVklBTF9DT1BZICAgOiAwKQotICAgICAgICAgICAgICAgIHwodHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9hc3NpZ24gPyBIQVNfVFJJVklBTF9BU1NJR04gOiAwKSkKLSAgICAgICAgICAgICAgICApCi17Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotU29ydGVkVmVjdG9yPFRZUEU+OjpTb3J0ZWRWZWN0b3IoY29uc3QgU29ydGVkVmVjdG9yPFRZUEU+JiByaHMpCi0gICAgOiBTb3J0ZWRWZWN0b3JJbXBsKHJocykgewotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLVNvcnRlZFZlY3RvcjxUWVBFPjo6flNvcnRlZFZlY3RvcigpIHsKLSAgICBmaW5pc2hfdmVjdG9yKCk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotU29ydGVkVmVjdG9yPFRZUEU+JiBTb3J0ZWRWZWN0b3I8VFlQRT46Om9wZXJhdG9yID0gKGNvbnN0IFNvcnRlZFZlY3RvcjxUWVBFPiYgcmhzKSB7Ci0gICAgU29ydGVkVmVjdG9ySW1wbDo6b3BlcmF0b3IgPSAocmhzKTsKLSAgICByZXR1cm4gKnRoaXM7IAotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLWNvbnN0IFNvcnRlZFZlY3RvcjxUWVBFPiYgU29ydGVkVmVjdG9yPFRZUEU+OjpvcGVyYXRvciA9IChjb25zdCBTb3J0ZWRWZWN0b3I8VFlQRT4mIHJocykgY29uc3QgewotICAgIFNvcnRlZFZlY3RvckltcGw6Om9wZXJhdG9yID0gKHJocyk7Ci0gICAgcmV0dXJuICp0aGlzOyAKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1jb25zdCBUWVBFKiBTb3J0ZWRWZWN0b3I8VFlQRT46OmFycmF5KCkgY29uc3QgewotICAgIHJldHVybiBzdGF0aWNfY2FzdDxjb25zdCBUWVBFICo+KGFycmF5SW1wbCgpKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1UWVBFKiBTb3J0ZWRWZWN0b3I8VFlQRT46OmVkaXRBcnJheSgpIHsKLSAgICByZXR1cm4gc3RhdGljX2Nhc3Q8VFlQRSAqPihlZGl0QXJyYXlJbXBsKCkpOwotfQotCi0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotY29uc3QgVFlQRSYgU29ydGVkVmVjdG9yPFRZUEU+OjpvcGVyYXRvcltdKHNpemVfdCBpbmRleCkgY29uc3QgewotICAgIGFzc2VydCggaW5kZXg8c2l6ZSgpICk7Ci0gICAgcmV0dXJuICooYXJyYXkoKSArIGluZGV4KTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1jb25zdCBUWVBFJiBTb3J0ZWRWZWN0b3I8VFlQRT46Oml0ZW1BdChzaXplX3QgaW5kZXgpIGNvbnN0IHsKLSAgICByZXR1cm4gb3BlcmF0b3JbXShpbmRleCk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotY29uc3QgVFlQRSYgU29ydGVkVmVjdG9yPFRZUEU+OjptaXJyb3JJdGVtQXQoc3NpemVfdCBpbmRleCkgY29uc3QgewotICAgIGFzc2VydCggKGluZGV4PjAgPyBpbmRleCA6IC1pbmRleCk8c2l6ZSgpICk7Ci0gICAgcmV0dXJuICooYXJyYXkoKSArICgoaW5kZXg8MCkgPyAoc2l6ZSgpLWluZGV4KSA6IGluZGV4KSk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotY29uc3QgVFlQRSYgU29ydGVkVmVjdG9yPFRZUEU+Ojp0b3AoKSBjb25zdCB7Ci0gICAgcmV0dXJuICooYXJyYXkoKSArIHNpemUoKSAtIDEpOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLXNzaXplX3QgU29ydGVkVmVjdG9yPFRZUEU+OjphZGQoY29uc3QgVFlQRSYgaXRlbSkgewotICAgIHJldHVybiBTb3J0ZWRWZWN0b3JJbXBsOjphZGQoJml0ZW0pOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLXNzaXplX3QgU29ydGVkVmVjdG9yPFRZUEU+OjppbmRleE9mKGNvbnN0IFRZUEUmIGl0ZW0pIGNvbnN0IHsKLSAgICByZXR1cm4gU29ydGVkVmVjdG9ySW1wbDo6aW5kZXhPZigmaXRlbSk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotc2l6ZV90IFNvcnRlZFZlY3RvcjxUWVBFPjo6b3JkZXJPZihjb25zdCBUWVBFJiBpdGVtKSBjb25zdCB7Ci0gICAgcmV0dXJuIFNvcnRlZFZlY3RvckltcGw6Om9yZGVyT2YoJml0ZW0pOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLXNzaXplX3QgU29ydGVkVmVjdG9yPFRZUEU+OjptZXJnZShjb25zdCBWZWN0b3I8VFlQRT4mIHZlY3RvcikgewotICAgIHJldHVybiBTb3J0ZWRWZWN0b3JJbXBsOjptZXJnZShyZWludGVycHJldF9jYXN0PGNvbnN0IFZlY3RvckltcGwmPih2ZWN0b3IpKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zc2l6ZV90IFNvcnRlZFZlY3RvcjxUWVBFPjo6bWVyZ2UoY29uc3QgU29ydGVkVmVjdG9yPFRZUEU+JiB2ZWN0b3IpIHsKLSAgICByZXR1cm4gU29ydGVkVmVjdG9ySW1wbDo6bWVyZ2UocmVpbnRlcnByZXRfY2FzdDxjb25zdCBTb3J0ZWRWZWN0b3JJbXBsJj4odmVjdG9yKSk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotc3NpemVfdCBTb3J0ZWRWZWN0b3I8VFlQRT46OnJlbW92ZShjb25zdCBUWVBFJiBpdGVtKSB7Ci0gICAgcmV0dXJuIFNvcnRlZFZlY3RvckltcGw6OnJlbW92ZSgmaXRlbSk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotc3NpemVfdCBTb3J0ZWRWZWN0b3I8VFlQRT46OnJlbW92ZUl0ZW1zQXQoc2l6ZV90IGluZGV4LCBzaXplX3QgY291bnQpIHsKLSAgICByZXR1cm4gVmVjdG9ySW1wbDo6cmVtb3ZlSXRlbXNBdChpbmRleCwgY291bnQpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4KLXZvaWQgU29ydGVkVmVjdG9yPFRZUEU+Ojpkb19jb25zdHJ1Y3Qodm9pZCogc3RvcmFnZSwgc2l6ZV90IG51bSkgY29uc3QgewotICAgIGNvbnN0cnVjdF90eXBlKCByZWludGVycHJldF9jYXN0PFRZUEUqPihzdG9yYWdlKSwgbnVtICk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+Ci12b2lkIFNvcnRlZFZlY3RvcjxUWVBFPjo6ZG9fZGVzdHJveSh2b2lkKiBzdG9yYWdlLCBzaXplX3QgbnVtKSBjb25zdCB7Ci0gICAgZGVzdHJveV90eXBlKCByZWludGVycHJldF9jYXN0PFRZUEUqPihzdG9yYWdlKSwgbnVtICk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+Ci12b2lkIFNvcnRlZFZlY3RvcjxUWVBFPjo6ZG9fY29weSh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdCB7Ci0gICAgY29weV90eXBlKCByZWludGVycHJldF9jYXN0PFRZUEUqPihkZXN0KSwgcmVpbnRlcnByZXRfY2FzdDxjb25zdCBUWVBFKj4oZnJvbSksIG51bSApOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPgotdm9pZCBTb3J0ZWRWZWN0b3I8VFlQRT46OmRvX3NwbGF0KHZvaWQqIGRlc3QsIGNvbnN0IHZvaWQqIGl0ZW0sIHNpemVfdCBudW0pIGNvbnN0IHsKLSAgICBzcGxhdF90eXBlKCByZWludGVycHJldF9jYXN0PFRZUEUqPihkZXN0KSwgcmVpbnRlcnByZXRfY2FzdDxjb25zdCBUWVBFKj4oaXRlbSksIG51bSApOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPgotdm9pZCBTb3J0ZWRWZWN0b3I8VFlQRT46OmRvX21vdmVfZm9yd2FyZCh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdCB7Ci0gICAgbW92ZV9mb3J3YXJkX3R5cGUoIHJlaW50ZXJwcmV0X2Nhc3Q8VFlQRSo+KGRlc3QpLCByZWludGVycHJldF9jYXN0PGNvbnN0IFRZUEUqPihmcm9tKSwgbnVtICk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+Ci12b2lkIFNvcnRlZFZlY3RvcjxUWVBFPjo6ZG9fbW92ZV9iYWNrd2FyZCh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdCB7Ci0gICAgbW92ZV9iYWNrd2FyZF90eXBlKCByZWludGVycHJldF9jYXN0PFRZUEUqPihkZXN0KSwgcmVpbnRlcnByZXRfY2FzdDxjb25zdCBUWVBFKj4oZnJvbSksIG51bSApOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPgotaW50IFNvcnRlZFZlY3RvcjxUWVBFPjo6ZG9fY29tcGFyZShjb25zdCB2b2lkKiBsaHMsIGNvbnN0IHZvaWQqIHJocykgY29uc3QgewotICAgIHJldHVybiBjb21wYXJlX3R5cGUoICpyZWludGVycHJldF9jYXN0PGNvbnN0IFRZUEUqPihsaHMpLCAqcmVpbnRlcnByZXRfY2FzdDxjb25zdCBUWVBFKj4ocmhzKSApOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLy8gQU5EUk9JRF9TT1JURURfVkVDVE9SX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvU3RvcFdhdGNoLmggYi9pbmNsdWRlL3V0aWxzL1N0b3BXYXRjaC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjYzBiZWJjLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvU3RvcFdhdGNoLmgKKysrIC9kZXYvbnVsbApAQCAtMSw2MiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1NUT1BXQVRDSF9ICi0jZGVmaW5lIEFORFJPSURfU1RPUFdBVENIX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvVGltZXJzLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIFN0b3BXYXRjaAotewotcHVibGljOgotICAgICAgICBTdG9wV2F0Y2goICBjb25zdCBjaGFyICpuYW1lLAotICAgICAgICAgICAgICAgICAgICBpbnQgY2xvY2sgPSBTWVNURU1fVElNRV9NT05PVE9OSUMsCi0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGZsYWdzID0gMCk7Ci0gICAgICAgIH5TdG9wV2F0Y2goKTsKLSAgICAgICAgCi0gICAgICAgIGNvbnN0IGNoYXIqIG5hbWUoKSBjb25zdDsKLSAgICAgICAgbnNlY3NfdCAgICAgbGFwKCk7Ci0gICAgICAgIG5zZWNzX3QgICAgIGVsYXBzZWRUaW1lKCkgY29uc3Q7Ci0gICAgICAgIAotcHJpdmF0ZToKLSAgICBjb25zdCBjaGFyKiAgICAgbU5hbWU7Ci0gICAgaW50ICAgICAgICAgICAgIG1DbG9jazsKLSAgICB1aW50MzJfdCAgICAgICAgbUZsYWdzOwotICAgIAotICAgIHN0cnVjdCBsYXBfdCB7Ci0gICAgICAgIG5zZWNzX3QgICAgIHNvRmFyOwotICAgICAgICBuc2Vjc190ICAgICB0aGlzTGFwOwotICAgIH07Ci0gICAgCi0gICAgbnNlY3NfdCAgICAgICAgIG1TdGFydFRpbWU7Ci0gICAgbGFwX3QgICAgICAgICAgIG1MYXBzWzhdOwotICAgIGludCAgICAgICAgICAgICBtTnVtTGFwczsKLX07Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2VuZGlmIC8vIEFORFJPSURfU1RPUFdBVENIX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvU3RyaW5nMTYuaCBiL2luY2x1ZGUvdXRpbHMvU3RyaW5nMTYuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTJkMjJlZS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL1N0cmluZzE2LmgKKysrIC9kZXYvbnVsbApAQCAtMSwyNjAgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9TVFJJTkcxNl9ICi0jZGVmaW5lIEFORFJPSURfU1RSSU5HMTZfSAotCi0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU2hhcmVkQnVmZmVyLmg+Ci0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWV4dGVybiAiQyIgewotCi10eXBlZGVmIHVpbnQxNl90IGNoYXIxNl90OwotCi0vLyBTdGFuZGFyZCBzdHJpbmcgZnVuY3Rpb25zIG9uIGNoYXIxNiBzdHJpbmdzLgotaW50IHN0cmNtcDE2KGNvbnN0IGNoYXIxNl90ICosIGNvbnN0IGNoYXIxNl90ICopOwotaW50IHN0cm5jbXAxNihjb25zdCBjaGFyMTZfdCAqczEsIGNvbnN0IGNoYXIxNl90ICpzMiwgc2l6ZV90IG4pOwotc2l6ZV90IHN0cmxlbjE2KGNvbnN0IGNoYXIxNl90ICopOwotc2l6ZV90IHN0cm5sZW4xNihjb25zdCBjaGFyMTZfdCAqLCBzaXplX3QpOwotY2hhcjE2X3QgKnN0cmNweTE2KGNoYXIxNl90ICosIGNvbnN0IGNoYXIxNl90ICopOwotY2hhcjE2X3QgKnN0cm5jcHkxNihjaGFyMTZfdCAqLCBjb25zdCBjaGFyMTZfdCAqLCBzaXplX3QpOwotCi0vLyBWZXJzaW9uIG9mIGNvbXBhcmlzb24gdGhhdCBzdXBwb3J0cyBlbWJlZGRlZCBudWxscy4KLS8vIFRoaXMgaXMgZGlmZmVyZW50IHRoYW4gc3RybmNtcCgpIGJlY2F1c2Ugd2UgZG9uJ3Qgc3RvcAotLy8gYXQgYSBudWwgY2hhcmFjdGVyIGFuZCBjb25zaWRlciB0aGUgc3RyaW5ncyB0byBiZSBkaWZmZXJlbnQKLS8vIGlmIHRoZSBsZW5ndGhzIGFyZSBkaWZmZXJlbnQgKHRodXMgd2UgbmVlZCB0byBzdXBwbHkgdGhlCi0vLyBsZW5ndGhzIG9mIGJvdGggc3RyaW5ncykuICBUaGlzIGNhbiBhbHNvIGJlIHVzZWQgd2hlbgotLy8geW91ciBzdHJpbmcgaXMgbm90IG51bC10ZXJtaW5hdGVkIGFzIGl0IHdpbGwgaGF2ZSB0aGUKLS8vIGVxdWl2YWxlbnQgcmVzdWx0IGFzIHN0cmNtcDE2ICh1bmxpa2Ugc3RybmNtcDE2KS4KLWludCBzdHJ6Y21wMTYoY29uc3QgY2hhcjE2X3QgKnMxLCBzaXplX3QgbjEsIGNvbnN0IGNoYXIxNl90ICpzMiwgc2l6ZV90IG4yKTsKLQotLy8gVmVyc2lvbiBvZiBzdHJ6Y21wMTYgZm9yIGNvbXBhcmluZyBzdHJpbmdzIGluIGRpZmZlcmVudCBlbmRpYW5uZXNzLgotaW50IHN0cnpjbXAxNl9oX24oY29uc3QgY2hhcjE2X3QgKnMxSCwgc2l6ZV90IG4xLCBjb25zdCBjaGFyMTZfdCAqczJOLCBzaXplX3QgbjIpOwotCi19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIFN0cmluZzg7Ci1jbGFzcyBUZXh0T3V0cHV0OwotCi0vLyEgVGhpcyBpcyBhIHN0cmluZyBob2xkaW5nIFVURi0xNiBjaGFyYWN0ZXJzLgotY2xhc3MgU3RyaW5nMTYKLXsKLXB1YmxpYzoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nMTYoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nMTYoY29uc3QgU3RyaW5nMTYmIG8pOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTdHJpbmcxNihjb25zdCBTdHJpbmcxNiYgbywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IGxlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IGJlZ2luPTApOwotICAgIGV4cGxpY2l0ICAgICAgICAgICAgICAgICAgICBTdHJpbmcxNihjb25zdCBjaGFyMTZfdCogbyk7Ci0gICAgZXhwbGljaXQgICAgICAgICAgICAgICAgICAgIFN0cmluZzE2KGNvbnN0IGNoYXIxNl90KiBvLCBzaXplX3QgbGVuKTsKLSAgICBleHBsaWNpdCAgICAgICAgICAgICAgICAgICAgU3RyaW5nMTYoY29uc3QgU3RyaW5nOCYgbyk7Ci0gICAgZXhwbGljaXQgICAgICAgICAgICAgICAgICAgIFN0cmluZzE2KGNvbnN0IGNoYXIqIG8pOwotICAgIGV4cGxpY2l0ICAgICAgICAgICAgICAgICAgICBTdHJpbmcxNihjb25zdCBjaGFyKiBvLCBzaXplX3QgbGVuKTsKLQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB+U3RyaW5nMTYoKTsKLSAgICAKLSAgICBpbmxpbmUgIGNvbnN0IGNoYXIxNl90KiAgICAgc3RyaW5nKCkgY29uc3Q7Ci0gICAgaW5saW5lICBzaXplX3QgICAgICAgICAgICAgIHNpemUoKSBjb25zdDsKLSAgICAKLSAgICBpbmxpbmUgIGNvbnN0IFNoYXJlZEJ1ZmZlciogc2hhcmVkQnVmZmVyKCkgY29uc3Q7Ci0gICAgCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHNldFRvKGNvbnN0IFN0cmluZzE2JiBvdGhlcik7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHNldFRvKGNvbnN0IGNoYXIxNl90KiBvdGhlcik7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHNldFRvKGNvbnN0IGNoYXIxNl90KiBvdGhlciwgc2l6ZV90IGxlbik7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHNldFRvKGNvbnN0IFN0cmluZzE2JiBvdGhlciwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IGxlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IGJlZ2luPTApOwotICAgIAotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICBhcHBlbmQoY29uc3QgU3RyaW5nMTYmIG90aGVyKTsKLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgYXBwZW5kKGNvbnN0IGNoYXIxNl90KiBvdGhlciwgc2l6ZV90IGxlbik7Ci0gICAgICAgICAgICAKLSAgICBpbmxpbmUgIFN0cmluZzE2JiAgICAgICAgICAgb3BlcmF0b3I9KGNvbnN0IFN0cmluZzE2JiBvdGhlcik7Ci0gICAgCi0gICAgaW5saW5lICBTdHJpbmcxNiYgICAgICAgICAgIG9wZXJhdG9yKz0oY29uc3QgU3RyaW5nMTYmIG90aGVyKTsKLSAgICBpbmxpbmUgIFN0cmluZzE2ICAgICAgICAgICAgb3BlcmF0b3IrKGNvbnN0IFN0cmluZzE2JiBvdGhlcikgY29uc3Q7Ci0KLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgaW5zZXJ0KHNpemVfdCBwb3MsIGNvbnN0IGNoYXIxNl90KiBjaHJzKTsKLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgaW5zZXJ0KHNpemVfdCBwb3MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogY2hycywgc2l6ZV90IGxlbik7Ci0KLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICAgICAgZmluZEZpcnN0KGNoYXIxNl90IGMpIGNvbnN0OwotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgICAgICBmaW5kTGFzdChjaGFyMTZfdCBjKSBjb25zdDsKLQotICAgICAgICAgICAgYm9vbCAgICAgICAgICAgICAgICBzdGFydHNXaXRoKGNvbnN0IFN0cmluZzE2JiBwcmVmaXgpIGNvbnN0OwotICAgICAgICAgICAgYm9vbCAgICAgICAgICAgICAgICBzdGFydHNXaXRoKGNvbnN0IGNoYXIxNl90KiBwcmVmaXgpIGNvbnN0OwotICAgICAgICAgICAgCi0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIG1ha2VMb3dlcigpOwotCi0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHJlcGxhY2VBbGwoY2hhcjE2X3QgcmVwbGFjZVRoaXMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY2hhcjE2X3Qgd2l0aFRoaXMpOwotCi0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHJlbW92ZShzaXplX3QgbGVuLCBzaXplX3QgYmVnaW49MCk7Ci0KLSAgICBpbmxpbmUgIGludCAgICAgICAgICAgICAgICAgY29tcGFyZShjb25zdCBTdHJpbmcxNiYgb3RoZXIpIGNvbnN0OwotCi0gICAgaW5saW5lICBib29sICAgICAgICAgICAgICAgIG9wZXJhdG9yPChjb25zdCBTdHJpbmcxNiYgb3RoZXIpIGNvbnN0OwotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcjw9KGNvbnN0IFN0cmluZzE2JiBvdGhlcikgY29uc3Q7Ci0gICAgaW5saW5lICBib29sICAgICAgICAgICAgICAgIG9wZXJhdG9yPT0oY29uc3QgU3RyaW5nMTYmIG90aGVyKSBjb25zdDsKLSAgICBpbmxpbmUgIGJvb2wgICAgICAgICAgICAgICAgb3BlcmF0b3IhPShjb25zdCBTdHJpbmcxNiYgb3RoZXIpIGNvbnN0OwotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcj49KGNvbnN0IFN0cmluZzE2JiBvdGhlcikgY29uc3Q7Ci0gICAgaW5saW5lICBib29sICAgICAgICAgICAgICAgIG9wZXJhdG9yPihjb25zdCBTdHJpbmcxNiYgb3RoZXIpIGNvbnN0OwotICAgIAotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcjwoY29uc3QgY2hhcjE2X3QqIG90aGVyKSBjb25zdDsKLSAgICBpbmxpbmUgIGJvb2wgICAgICAgICAgICAgICAgb3BlcmF0b3I8PShjb25zdCBjaGFyMTZfdCogb3RoZXIpIGNvbnN0OwotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcj09KGNvbnN0IGNoYXIxNl90KiBvdGhlcikgY29uc3Q7Ci0gICAgaW5saW5lICBib29sICAgICAgICAgICAgICAgIG9wZXJhdG9yIT0oY29uc3QgY2hhcjE2X3QqIG90aGVyKSBjb25zdDsKLSAgICBpbmxpbmUgIGJvb2wgICAgICAgICAgICAgICAgb3BlcmF0b3I+PShjb25zdCBjaGFyMTZfdCogb3RoZXIpIGNvbnN0OwotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcj4oY29uc3QgY2hhcjE2X3QqIG90aGVyKSBjb25zdDsKLSAgICAKLSAgICBpbmxpbmUgICAgICAgICAgICAgICAgICAgICAgb3BlcmF0b3IgY29uc3QgY2hhcjE2X3QqKCkgY29uc3Q7Ci0gICAgCi1wcml2YXRlOgotICAgICAgICAgICAgY29uc3QgY2hhcjE2X3QqICAgICBtU3RyaW5nOwotfTsKLQotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgY29uc3QgU3RyaW5nMTYmIHZhbCk7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gTm8gdXNlciBzZXJ2aWNhYmxlIHBhcnRzIGJlbG93LgotCi1pbmxpbmUgaW50IGNvbXBhcmVfdHlwZShjb25zdCBTdHJpbmcxNiYgbGhzLCBjb25zdCBTdHJpbmcxNiYgcmhzKQotewotICAgIHJldHVybiBsaHMuY29tcGFyZShyaHMpOwotfQotCi1pbmxpbmUgaW50IHN0cmljdGx5X29yZGVyX3R5cGUoY29uc3QgU3RyaW5nMTYmIGxocywgY29uc3QgU3RyaW5nMTYmIHJocykKLXsKLSAgICByZXR1cm4gY29tcGFyZV90eXBlKGxocywgcmhzKSA8IDA7Ci19Ci0KLWlubGluZSBjb25zdCBjaGFyMTZfdCogU3RyaW5nMTY6OnN0cmluZygpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1TdHJpbmc7Ci19Ci0KLWlubGluZSBzaXplX3QgU3RyaW5nMTY6OnNpemUoKSBjb25zdAotewotICAgIHJldHVybiBTaGFyZWRCdWZmZXI6OnNpemVGcm9tRGF0YShtU3RyaW5nKS9zaXplb2YoY2hhcjE2X3QpLTE7Ci19Ci0KLWlubGluZSBjb25zdCBTaGFyZWRCdWZmZXIqIFN0cmluZzE2OjpzaGFyZWRCdWZmZXIoKSBjb25zdAotewotICAgIHJldHVybiBTaGFyZWRCdWZmZXI6OmJ1ZmZlckZyb21EYXRhKG1TdHJpbmcpOwotfQotCi1pbmxpbmUgU3RyaW5nMTYmIFN0cmluZzE2OjpvcGVyYXRvcj0oY29uc3QgU3RyaW5nMTYmIG90aGVyKQotewotICAgIHNldFRvKG90aGVyKTsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLWlubGluZSBTdHJpbmcxNiYgU3RyaW5nMTY6Om9wZXJhdG9yKz0oY29uc3QgU3RyaW5nMTYmIG90aGVyKQotewotICAgIGFwcGVuZChvdGhlcik7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi1pbmxpbmUgU3RyaW5nMTYgU3RyaW5nMTY6Om9wZXJhdG9yKyhjb25zdCBTdHJpbmcxNiYgb3RoZXIpIGNvbnN0Ci17Ci0gICAgU3RyaW5nMTYgdG1wOwotICAgIHRtcCArPSBvdGhlcjsKLSAgICByZXR1cm4gdG1wOwotfQotCi1pbmxpbmUgaW50IFN0cmluZzE2Ojpjb21wYXJlKGNvbnN0IFN0cmluZzE2JiBvdGhlcikgY29uc3QKLXsKLSAgICByZXR1cm4gc3RyemNtcDE2KG1TdHJpbmcsIHNpemUoKSwgb3RoZXIubVN0cmluZywgb3RoZXIuc2l6ZSgpKTsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nMTY6Om9wZXJhdG9yPChjb25zdCBTdHJpbmcxNiYgb3RoZXIpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0cnpjbXAxNihtU3RyaW5nLCBzaXplKCksIG90aGVyLm1TdHJpbmcsIG90aGVyLnNpemUoKSkgPCAwOwotfQotCi1pbmxpbmUgYm9vbCBTdHJpbmcxNjo6b3BlcmF0b3I8PShjb25zdCBTdHJpbmcxNiYgb3RoZXIpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0cnpjbXAxNihtU3RyaW5nLCBzaXplKCksIG90aGVyLm1TdHJpbmcsIG90aGVyLnNpemUoKSkgPD0gMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nMTY6Om9wZXJhdG9yPT0oY29uc3QgU3RyaW5nMTYmIG90aGVyKSBjb25zdAotewotICAgIHJldHVybiBzdHJ6Y21wMTYobVN0cmluZywgc2l6ZSgpLCBvdGhlci5tU3RyaW5nLCBvdGhlci5zaXplKCkpID09IDA7Ci19Ci0KLWlubGluZSBib29sIFN0cmluZzE2OjpvcGVyYXRvciE9KGNvbnN0IFN0cmluZzE2JiBvdGhlcikgY29uc3QKLXsKLSAgICByZXR1cm4gc3RyemNtcDE2KG1TdHJpbmcsIHNpemUoKSwgb3RoZXIubVN0cmluZywgb3RoZXIuc2l6ZSgpKSAhPSAwOwotfQotCi1pbmxpbmUgYm9vbCBTdHJpbmcxNjo6b3BlcmF0b3I+PShjb25zdCBTdHJpbmcxNiYgb3RoZXIpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0cnpjbXAxNihtU3RyaW5nLCBzaXplKCksIG90aGVyLm1TdHJpbmcsIG90aGVyLnNpemUoKSkgPj0gMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nMTY6Om9wZXJhdG9yPihjb25zdCBTdHJpbmcxNiYgb3RoZXIpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0cnpjbXAxNihtU3RyaW5nLCBzaXplKCksIG90aGVyLm1TdHJpbmcsIG90aGVyLnNpemUoKSkgPiAwOwotfQotCi1pbmxpbmUgYm9vbCBTdHJpbmcxNjo6b3BlcmF0b3I8KGNvbnN0IGNoYXIxNl90KiBvdGhlcikgY29uc3QKLXsKLSAgICByZXR1cm4gc3RyY21wMTYobVN0cmluZywgb3RoZXIpIDwgMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nMTY6Om9wZXJhdG9yPD0oY29uc3QgY2hhcjE2X3QqIG90aGVyKSBjb25zdAotewotICAgIHJldHVybiBzdHJjbXAxNihtU3RyaW5nLCBvdGhlcikgPD0gMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nMTY6Om9wZXJhdG9yPT0oY29uc3QgY2hhcjE2X3QqIG90aGVyKSBjb25zdAotewotICAgIHJldHVybiBzdHJjbXAxNihtU3RyaW5nLCBvdGhlcikgPT0gMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nMTY6Om9wZXJhdG9yIT0oY29uc3QgY2hhcjE2X3QqIG90aGVyKSBjb25zdAotewotICAgIHJldHVybiBzdHJjbXAxNihtU3RyaW5nLCBvdGhlcikgIT0gMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nMTY6Om9wZXJhdG9yPj0oY29uc3QgY2hhcjE2X3QqIG90aGVyKSBjb25zdAotewotICAgIHJldHVybiBzdHJjbXAxNihtU3RyaW5nLCBvdGhlcikgPj0gMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nMTY6Om9wZXJhdG9yPihjb25zdCBjaGFyMTZfdCogb3RoZXIpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0cmNtcDE2KG1TdHJpbmcsIG90aGVyKSA+IDA7Ci19Ci0KLWlubGluZSBTdHJpbmcxNjo6b3BlcmF0b3IgY29uc3QgY2hhcjE2X3QqKCkgY29uc3QKLXsKLSAgICByZXR1cm4gbVN0cmluZzsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNlbmRpZiAvLyBBTkRST0lEX1NUUklORzE2X0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvU3RyaW5nOC5oIGIvaW5jbHVkZS91dGlscy9TdHJpbmc4LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM0OWZhZjYuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9TdHJpbmc4LmgKKysrIC9kZXYvbnVsbApAQCAtMSwzNTMgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9TVFJJTkc4X0gKLSNkZWZpbmUgQU5EUk9JRF9TVFJJTkc4X0gKLQotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotCi0vLyBOZWVkIHRoaXMgZm9yIHRoZSBjaGFyMTZfdCB0eXBlOyBTdHJpbmc4Lmggc2hvdWxkIG5vdAotLy8gYmUgZGVwZWRlbnQgb24gdGhlIFN0cmluZzE2IGNsYXNzLgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzE2Lmg+Ci0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgVGV4dE91dHB1dDsKLQotLy8hIFRoaXMgaXMgYSBzdHJpbmcgaG9sZGluZyBVVEYtOCBjaGFyYWN0ZXJzLgotY2xhc3MgU3RyaW5nOAotewotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTdHJpbmc4KCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgoY29uc3QgU3RyaW5nOCYgbyk7Ci0gICAgZXhwbGljaXQgICAgICAgICAgICAgICAgICAgIFN0cmluZzgoY29uc3QgY2hhciogbyk7Ci0gICAgZXhwbGljaXQgICAgICAgICAgICAgICAgICAgIFN0cmluZzgoY29uc3QgY2hhciogbywgc2l6ZV90IG51bUNoYXJzKTsKLSAgICAKLSAgICBleHBsaWNpdCAgICAgICAgICAgICAgICAgICAgU3RyaW5nOChjb25zdCBTdHJpbmcxNiYgbyk7Ci0gICAgZXhwbGljaXQgICAgICAgICAgICAgICAgICAgIFN0cmluZzgoY29uc3QgY2hhcjE2X3QqIG8pOwotICAgIGV4cGxpY2l0ICAgICAgICAgICAgICAgICAgICBTdHJpbmc4KGNvbnN0IGNoYXIxNl90KiBvLCBzaXplX3QgbnVtQ2hhcnMpOwotICAgIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB+U3RyaW5nOCgpOwotICAgIAotICAgIGlubGluZSAgY29uc3QgY2hhciogICAgICAgICBzdHJpbmcoKSBjb25zdDsKLSAgICBpbmxpbmUgIHNpemVfdCAgICAgICAgICAgICAgc2l6ZSgpIGNvbnN0OwotICAgIGlubGluZSAgc2l6ZV90ICAgICAgICAgICAgICBsZW5ndGgoKSBjb25zdDsKLSAgICBpbmxpbmUgIHNpemVfdCAgICAgICAgICAgICAgYnl0ZXMoKSBjb25zdDsKLSAgICAKLSAgICBpbmxpbmUgIGNvbnN0IFNoYXJlZEJ1ZmZlciogc2hhcmVkQnVmZmVyKCkgY29uc3Q7Ci0gICAgCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHNldFRvKGNvbnN0IFN0cmluZzgmIG90aGVyKTsKLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgc2V0VG8oY29uc3QgY2hhciogb3RoZXIpOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICBzZXRUbyhjb25zdCBjaGFyKiBvdGhlciwgc2l6ZV90IG51bUNoYXJzKTsKLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgc2V0VG8oY29uc3QgY2hhcjE2X3QqIG90aGVyLCBzaXplX3QgbnVtQ2hhcnMpOwotICAgIAotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICBhcHBlbmQoY29uc3QgU3RyaW5nOCYgb3RoZXIpOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICBhcHBlbmQoY29uc3QgY2hhciogb3RoZXIpOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICBhcHBlbmQoY29uc3QgY2hhciogb3RoZXIsIHNpemVfdCBudW1DaGFycyk7Ci0KLSAgICBpbmxpbmUgIFN0cmluZzgmICAgICAgICAgICAgb3BlcmF0b3I9KGNvbnN0IFN0cmluZzgmIG90aGVyKTsKLSAgICBpbmxpbmUgIFN0cmluZzgmICAgICAgICAgICAgb3BlcmF0b3I9KGNvbnN0IGNoYXIqIG90aGVyKTsKLSAgICAKLSAgICBpbmxpbmUgIFN0cmluZzgmICAgICAgICAgICAgb3BlcmF0b3IrPShjb25zdCBTdHJpbmc4JiBvdGhlcik7Ci0gICAgaW5saW5lICBTdHJpbmc4ICAgICAgICAgICAgIG9wZXJhdG9yKyhjb25zdCBTdHJpbmc4JiBvdGhlcikgY29uc3Q7Ci0gICAgCi0gICAgaW5saW5lICBTdHJpbmc4JiAgICAgICAgICAgIG9wZXJhdG9yKz0oY29uc3QgY2hhciogb3RoZXIpOwotICAgIGlubGluZSAgU3RyaW5nOCAgICAgICAgICAgICBvcGVyYXRvcisoY29uc3QgY2hhciogb3RoZXIpIGNvbnN0OwotCi0gICAgaW5saW5lICBpbnQgICAgICAgICAgICAgICAgIGNvbXBhcmUoY29uc3QgU3RyaW5nOCYgb3RoZXIpIGNvbnN0OwotCi0gICAgaW5saW5lICBib29sICAgICAgICAgICAgICAgIG9wZXJhdG9yPChjb25zdCBTdHJpbmc4JiBvdGhlcikgY29uc3Q7Ci0gICAgaW5saW5lICBib29sICAgICAgICAgICAgICAgIG9wZXJhdG9yPD0oY29uc3QgU3RyaW5nOCYgb3RoZXIpIGNvbnN0OwotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcj09KGNvbnN0IFN0cmluZzgmIG90aGVyKSBjb25zdDsKLSAgICBpbmxpbmUgIGJvb2wgICAgICAgICAgICAgICAgb3BlcmF0b3IhPShjb25zdCBTdHJpbmc4JiBvdGhlcikgY29uc3Q7Ci0gICAgaW5saW5lICBib29sICAgICAgICAgICAgICAgIG9wZXJhdG9yPj0oY29uc3QgU3RyaW5nOCYgb3RoZXIpIGNvbnN0OwotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcj4oY29uc3QgU3RyaW5nOCYgb3RoZXIpIGNvbnN0OwotICAgIAotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcjwoY29uc3QgY2hhciogb3RoZXIpIGNvbnN0OwotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcjw9KGNvbnN0IGNoYXIqIG90aGVyKSBjb25zdDsKLSAgICBpbmxpbmUgIGJvb2wgICAgICAgICAgICAgICAgb3BlcmF0b3I9PShjb25zdCBjaGFyKiBvdGhlcikgY29uc3Q7Ci0gICAgaW5saW5lICBib29sICAgICAgICAgICAgICAgIG9wZXJhdG9yIT0oY29uc3QgY2hhciogb3RoZXIpIGNvbnN0OwotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgICAgICBvcGVyYXRvcj49KGNvbnN0IGNoYXIqIG90aGVyKSBjb25zdDsKLSAgICBpbmxpbmUgIGJvb2wgICAgICAgICAgICAgICAgb3BlcmF0b3I+KGNvbnN0IGNoYXIqIG90aGVyKSBjb25zdDsKLSAgICAKLSAgICBpbmxpbmUgICAgICAgICAgICAgICAgICAgICAgb3BlcmF0b3IgY29uc3QgY2hhciooKSBjb25zdDsKLSAgICAKLSAgICAgICAgICAgIGNoYXIqICAgICAgICAgICAgICAgbG9ja0J1ZmZlcihzaXplX3Qgc2l6ZSk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHVubG9ja0J1ZmZlcigpOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgICAgICB1bmxvY2tCdWZmZXIoc2l6ZV90IHNpemUpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyByZXR1cm4gdGhlIGluZGV4IG9mIHRoZSBmaXJzdCBieXRlIG9mIG90aGVyIGluIHRoaXMgYXQgb3IgYWZ0ZXIKLSAgICAgICAgICAgIC8vIHN0YXJ0LCBvciAtMSBpZiBub3QgZm91bmQKLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICAgICAgZmluZChjb25zdCBjaGFyKiBvdGhlciwgc2l6ZV90IHN0YXJ0ID0gMCkgY29uc3Q7Ci0KLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgdG9Mb3dlcigpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICB0b0xvd2VyKHNpemVfdCBzdGFydCwgc2l6ZV90IG51bUNoYXJzKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgdG9VcHBlcigpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICB0b1VwcGVyKHNpemVfdCBzdGFydCwgc2l6ZV90IG51bUNoYXJzKTsKLSAgICAgICAgICAgIAotICAgIC8qCi0gICAgICogVGhlc2UgbWV0aG9kcyBvcGVyYXRlIG9uIHRoZSBzdHJpbmcgYXMgaWYgaXQgd2VyZSBhIHBhdGggbmFtZS4KLSAgICAgKi8KLQotICAgIC8qCi0gICAgICogU2V0IHRoZSBmaWxlbmFtZSBmaWVsZCB0byBhIHNwZWNpZmljIHZhbHVlLgotICAgICAqCi0gICAgICogTm9ybWFsaXplcyB0aGUgZmlsZW5hbWUsIHJlbW92aW5nIGEgdHJhaWxpbmcgJy8nIGlmIHByZXNlbnQuCi0gICAgICovCi0gICAgdm9pZCBzZXRQYXRoTmFtZShjb25zdCBjaGFyKiBuYW1lKTsKLSAgICB2b2lkIHNldFBhdGhOYW1lKGNvbnN0IGNoYXIqIG5hbWUsIHNpemVfdCBudW1DaGFycyk7Ci0KLSAgICAvKgotICAgICAqIEdldCBqdXN0IHRoZSBmaWxlbmFtZSBjb21wb25lbnQuCi0gICAgICoKLSAgICAgKiAiL3RtcC9mb28vYmFyLmMiIC0tPiAiYmFyLmMiCi0gICAgICovCi0gICAgU3RyaW5nOCBnZXRQYXRoTGVhZih2b2lkKSBjb25zdDsKLQotICAgIC8qCi0gICAgICogUmVtb3ZlIHRoZSBsYXN0IChmaWxlIG5hbWUpIGNvbXBvbmVudCwgbGVhdmluZyBqdXN0IHRoZSBkaXJlY3RvcnkKLSAgICAgKiBuYW1lLgotICAgICAqCi0gICAgICogIi90bXAvZm9vL2Jhci5jIiAtLT4gIi90bXAvZm9vIgotICAgICAqICIvdG1wIiAtLT4gIiIgLy8gPz8/Pz8gc2hvdWxkbid0IHRoaXMgYmUgIi8iID8/Pz8gWFhYCi0gICAgICogImJhci5jIiAtLT4gIiIKLSAgICAgKi8KLSAgICBTdHJpbmc4IGdldFBhdGhEaXIodm9pZCkgY29uc3Q7Ci0KLSAgICAvKgotICAgICAqIFJldHJpZXZlIHRoZSBmcm9udCAocm9vdCBkaXIpIGNvbXBvbmVudC4gIE9wdGlvbmFsbHkgYWxzbyByZXR1cm4gdGhlCi0gICAgICogcmVtYWluaW5nIGNvbXBvbmVudHMuCi0gICAgICoKLSAgICAgKiAiL3RtcC9mb28vYmFyLmMiIC0tPiAidG1wIiAocmVtYWluID0gImZvby9iYXIuYyIpCi0gICAgICogIi90bXAiIC0tPiAidG1wIiAocmVtYWluID0gIiIpCi0gICAgICogImJhci5jIiAtLT4gImJhci5jIiAocmVtYWluID0gIiIpCi0gICAgICovCi0gICAgU3RyaW5nOCB3YWxrUGF0aChTdHJpbmc4KiBvdXRSZW1haW5zID0gTlVMTCkgY29uc3Q7Ci0KLSAgICAvKgotICAgICAqIFJldHVybiB0aGUgZmlsZW5hbWUgZXh0ZW5zaW9uLiAgVGhpcyBpcyB0aGUgbGFzdCAnLicgYW5kIHVwIHRvCi0gICAgICogZm91ciBjaGFyYWN0ZXJzIHRoYXQgZm9sbG93IGl0LiAgVGhlICcuJyBpcyBpbmNsdWRlZCBpbiBjYXNlIHdlCi0gICAgICogZGVjaWRlIHRvIGV4cGFuZCBvdXIgZGVmaW5pdGlvbiBvZiB3aGF0IGNvbnN0aXR1dGVzIGFuIGV4dGVuc2lvbi4KLSAgICAgKgotICAgICAqICIvdG1wL2Zvby9iYXIuYyIgLS0+ICIuYyIKLSAgICAgKiAiL3RtcCIgLS0+ICIiCi0gICAgICogIi90bXAvZm9vLmJhci9iYXoiIC0tPiAiIgotICAgICAqICJmb28uanBlZyIgLS0+ICIuanBlZyIKLSAgICAgKiAiZm9vLiIgLS0+ICIiCi0gICAgICovCi0gICAgU3RyaW5nOCBnZXRQYXRoRXh0ZW5zaW9uKHZvaWQpIGNvbnN0OwotCi0gICAgLyoKLSAgICAgKiBSZXR1cm4gdGhlIHBhdGggd2l0aG91dCB0aGUgZXh0ZW5zaW9uLiAgUnVsZXMgZm9yIHdoYXQgY29uc3RpdHV0ZXMKLSAgICAgKiBhbiBleHRlbnNpb24gYXJlIGRlc2NyaWJlZCBpbiB0aGUgY29tbWVudCBmb3IgZ2V0UGF0aEV4dGVuc2lvbigpLgotICAgICAqCi0gICAgICogIi90bXAvZm9vL2Jhci5jIiAtLT4gIi90bXAvZm9vL2JhciIKLSAgICAgKi8KLSAgICBTdHJpbmc4IGdldEJhc2VQYXRoKHZvaWQpIGNvbnN0OwotCi0gICAgLyoKLSAgICAgKiBBZGQgYSBjb21wb25lbnQgdG8gdGhlIHBhdGhuYW1lLiAgV2UgZ3VhcmFudGVlIHRoYXQgdGhlcmUgaXMKLSAgICAgKiBleGFjdGx5IG9uZSBwYXRoIHNlcGFyYXRvciBiZXR3ZWVuIHRoZSBvbGQgcGF0aCBhbmQgdGhlIG5ldy4KLSAgICAgKiBJZiB0aGVyZSBpcyBubyBleGlzdGluZyBuYW1lLCB3ZSBqdXN0IGNvcHkgdGhlIG5ldyBuYW1lIGluLgotICAgICAqCi0gICAgICogSWYgbGVhZiBpcyBhIGZ1bGx5IHF1YWxpZmllZCBwYXRoIChpLmUuIHN0YXJ0cyB3aXRoICcvJywgaXQKLSAgICAgKiByZXBsYWNlcyB3aGF0ZXZlciB3YXMgdGhlcmUgYmVmb3JlLgotICAgICAqLwotICAgIFN0cmluZzgmIGFwcGVuZFBhdGgoY29uc3QgY2hhciogbGVhZik7Ci0gICAgU3RyaW5nOCYgYXBwZW5kUGF0aChjb25zdCBTdHJpbmc4JiBsZWFmKSAgeyByZXR1cm4gYXBwZW5kUGF0aChsZWFmLnN0cmluZygpKTsgfQotCi0gICAgLyoKLSAgICAgKiBMaWtlIGFwcGVuZFBhdGgoKSwgYnV0IGRvZXMgbm90IGFmZmVjdCB0aGlzIHN0cmluZy4gIFJldHVybnMgYSBuZXcgb25lIGluc3RlYWQuCi0gICAgICovCi0gICAgU3RyaW5nOCBhcHBlbmRQYXRoQ29weShjb25zdCBjaGFyKiBsZWFmKSBjb25zdAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgeyBTdHJpbmc4IHAoKnRoaXMpOyBwLmFwcGVuZFBhdGgobGVhZik7IHJldHVybiBwOyB9Ci0gICAgU3RyaW5nOCBhcHBlbmRQYXRoQ29weShjb25zdCBTdHJpbmc4JiBsZWFmKSBjb25zdCB7IHJldHVybiBhcHBlbmRQYXRoQ29weShsZWFmLnN0cmluZygpKTsgfQotCi0gICAgLyoKLSAgICAgKiBDb252ZXJ0cyBhbGwgc2VwYXJhdG9ycyBpbiB0aGlzIHN0cmluZyB0byAvLCB0aGUgZGVmYXVsdCBwYXRoIHNlcGFyYXRvci4KLSAgICAgKgotICAgICAqIElmIHRoZSBkZWZhdWx0IE9TIHNlcGFyYXRvciBpcyBiYWNrc2xhc2gsIHRoaXMgY29udmVydHMgYWxsCi0gICAgICogYmFja3NsYXNoZXMgdG8gc2xhc2hlcywgaW4tcGxhY2UuIE90aGVyd2lzZSBpdCBkb2VzIG5vdGhpbmcuCi0gICAgICogUmV0dXJucyBzZWxmLgotICAgICAqLwotICAgIFN0cmluZzgmIGNvbnZlcnRUb1Jlc1BhdGgoKTsKLQotcHJpdmF0ZToKLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgcmVhbF9hcHBlbmQoY29uc3QgY2hhciogb3RoZXIsIHNpemVfdCBudW1DaGFycyk7Ci0gICAgICAgICAgICBjaGFyKiAgICAgICAgICAgICAgIGZpbmRfZXh0ZW5zaW9uKHZvaWQpIGNvbnN0OwotCi0gICAgICAgICAgICBjb25zdCBjaGFyKiBtU3RyaW5nOwotfTsKLQotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgY29uc3QgU3RyaW5nMTYmIHZhbCk7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gTm8gdXNlciBzZXJ2aWNhYmxlIHBhcnRzIGJlbG93LgotCi1pbmxpbmUgaW50IGNvbXBhcmVfdHlwZShjb25zdCBTdHJpbmc4JiBsaHMsIGNvbnN0IFN0cmluZzgmIHJocykKLXsKLSAgICByZXR1cm4gbGhzLmNvbXBhcmUocmhzKTsKLX0KLQotaW5saW5lIGludCBzdHJpY3RseV9vcmRlcl90eXBlKGNvbnN0IFN0cmluZzgmIGxocywgY29uc3QgU3RyaW5nOCYgcmhzKQotewotICAgIHJldHVybiBjb21wYXJlX3R5cGUobGhzLCByaHMpIDwgMDsKLX0KLQotaW5saW5lIGNvbnN0IGNoYXIqIFN0cmluZzg6OnN0cmluZygpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1TdHJpbmc7Ci19Ci0KLWlubGluZSBzaXplX3QgU3RyaW5nODo6bGVuZ3RoKCkgY29uc3QKLXsKLSAgICByZXR1cm4gU2hhcmVkQnVmZmVyOjpzaXplRnJvbURhdGEobVN0cmluZyktMTsKLX0KLQotaW5saW5lIHNpemVfdCBTdHJpbmc4OjpzaXplKCkgY29uc3QKLXsKLSAgICByZXR1cm4gbGVuZ3RoKCk7Ci19Ci0KLWlubGluZSBzaXplX3QgU3RyaW5nODo6Ynl0ZXMoKSBjb25zdAotewotICAgIHJldHVybiBTaGFyZWRCdWZmZXI6OnNpemVGcm9tRGF0YShtU3RyaW5nKS0xOwotfQotCi1pbmxpbmUgY29uc3QgU2hhcmVkQnVmZmVyKiBTdHJpbmc4OjpzaGFyZWRCdWZmZXIoKSBjb25zdAotewotICAgIHJldHVybiBTaGFyZWRCdWZmZXI6OmJ1ZmZlckZyb21EYXRhKG1TdHJpbmcpOwotfQotCi1pbmxpbmUgU3RyaW5nOCYgU3RyaW5nODo6b3BlcmF0b3I9KGNvbnN0IFN0cmluZzgmIG90aGVyKQotewotICAgIHNldFRvKG90aGVyKTsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLWlubGluZSBTdHJpbmc4JiBTdHJpbmc4OjpvcGVyYXRvcj0oY29uc3QgY2hhciogb3RoZXIpCi17Ci0gICAgc2V0VG8ob3RoZXIpOwotICAgIHJldHVybiAqdGhpczsKLX0KLQotaW5saW5lIFN0cmluZzgmIFN0cmluZzg6Om9wZXJhdG9yKz0oY29uc3QgU3RyaW5nOCYgb3RoZXIpCi17Ci0gICAgYXBwZW5kKG90aGVyKTsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLWlubGluZSBTdHJpbmc4IFN0cmluZzg6Om9wZXJhdG9yKyhjb25zdCBTdHJpbmc4JiBvdGhlcikgY29uc3QKLXsKLSAgICBTdHJpbmc4IHRtcDsKLSAgICB0bXAgKz0gb3RoZXI7Ci0gICAgcmV0dXJuIHRtcDsKLX0KLQotaW5saW5lIFN0cmluZzgmIFN0cmluZzg6Om9wZXJhdG9yKz0oY29uc3QgY2hhciogb3RoZXIpCi17Ci0gICAgYXBwZW5kKG90aGVyKTsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLWlubGluZSBTdHJpbmc4IFN0cmluZzg6Om9wZXJhdG9yKyhjb25zdCBjaGFyKiBvdGhlcikgY29uc3QKLXsKLSAgICBTdHJpbmc4IHRtcDsKLSAgICB0bXAgKz0gb3RoZXI7Ci0gICAgcmV0dXJuIHRtcDsKLX0KLQotaW5saW5lIGludCBTdHJpbmc4Ojpjb21wYXJlKGNvbnN0IFN0cmluZzgmIG90aGVyKSBjb25zdAotewotICAgIHJldHVybiBzdHJjbXAobVN0cmluZywgb3RoZXIubVN0cmluZyk7Ci19Ci0KLWlubGluZSBib29sIFN0cmluZzg6Om9wZXJhdG9yPChjb25zdCBTdHJpbmc4JiBvdGhlcikgY29uc3QKLXsKLSAgICByZXR1cm4gc3RyY21wKG1TdHJpbmcsIG90aGVyLm1TdHJpbmcpIDwgMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nODo6b3BlcmF0b3I8PShjb25zdCBTdHJpbmc4JiBvdGhlcikgY29uc3QKLXsKLSAgICByZXR1cm4gc3RyY21wKG1TdHJpbmcsIG90aGVyLm1TdHJpbmcpIDw9IDA7Ci19Ci0KLWlubGluZSBib29sIFN0cmluZzg6Om9wZXJhdG9yPT0oY29uc3QgU3RyaW5nOCYgb3RoZXIpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0cmNtcChtU3RyaW5nLCBvdGhlci5tU3RyaW5nKSA9PSAwOwotfQotCi1pbmxpbmUgYm9vbCBTdHJpbmc4OjpvcGVyYXRvciE9KGNvbnN0IFN0cmluZzgmIG90aGVyKSBjb25zdAotewotICAgIHJldHVybiBzdHJjbXAobVN0cmluZywgb3RoZXIubVN0cmluZykgIT0gMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nODo6b3BlcmF0b3I+PShjb25zdCBTdHJpbmc4JiBvdGhlcikgY29uc3QKLXsKLSAgICByZXR1cm4gc3RyY21wKG1TdHJpbmcsIG90aGVyLm1TdHJpbmcpID49IDA7Ci19Ci0KLWlubGluZSBib29sIFN0cmluZzg6Om9wZXJhdG9yPihjb25zdCBTdHJpbmc4JiBvdGhlcikgY29uc3QKLXsKLSAgICByZXR1cm4gc3RyY21wKG1TdHJpbmcsIG90aGVyLm1TdHJpbmcpID4gMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nODo6b3BlcmF0b3I8KGNvbnN0IGNoYXIqIG90aGVyKSBjb25zdAotewotICAgIHJldHVybiBzdHJjbXAobVN0cmluZywgb3RoZXIpIDwgMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nODo6b3BlcmF0b3I8PShjb25zdCBjaGFyKiBvdGhlcikgY29uc3QKLXsKLSAgICByZXR1cm4gc3RyY21wKG1TdHJpbmcsIG90aGVyKSA8PSAwOwotfQotCi1pbmxpbmUgYm9vbCBTdHJpbmc4OjpvcGVyYXRvcj09KGNvbnN0IGNoYXIqIG90aGVyKSBjb25zdAotewotICAgIHJldHVybiBzdHJjbXAobVN0cmluZywgb3RoZXIpID09IDA7Ci19Ci0KLWlubGluZSBib29sIFN0cmluZzg6Om9wZXJhdG9yIT0oY29uc3QgY2hhciogb3RoZXIpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0cmNtcChtU3RyaW5nLCBvdGhlcikgIT0gMDsKLX0KLQotaW5saW5lIGJvb2wgU3RyaW5nODo6b3BlcmF0b3I+PShjb25zdCBjaGFyKiBvdGhlcikgY29uc3QKLXsKLSAgICByZXR1cm4gc3RyY21wKG1TdHJpbmcsIG90aGVyKSA+PSAwOwotfQotCi1pbmxpbmUgYm9vbCBTdHJpbmc4OjpvcGVyYXRvcj4oY29uc3QgY2hhciogb3RoZXIpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0cmNtcChtU3RyaW5nLCBvdGhlcikgPiAwOwotfQotCi1pbmxpbmUgU3RyaW5nODo6b3BlcmF0b3IgY29uc3QgY2hhciooKSBjb25zdAotewotICAgIHJldHVybiBtU3RyaW5nOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2VuZGlmIC8vIEFORFJPSURfU1RSSU5HOF9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL1N5c3RlbUNsb2NrLmggYi9pbmNsdWRlL3V0aWxzL1N5c3RlbUNsb2NrLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDdjMzE5YmUuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9TeXN0ZW1DbG9jay5oCisrKyAvZGV2L251bGwKQEAgLTEsMzIgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9VVElMU19TWVNURU1DTE9DS19ICi0jZGVmaW5lIEFORFJPSURfVVRJTFNfU1lTVEVNQ0xPQ0tfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotaW50IHNldEN1cnJlbnRUaW1lTWlsbGlzKGludDY0X3QgbWlsbGlzKTsKLWludDY0X3QgdXB0aW1lTWlsbGlzKCk7Ci1pbnQ2NF90IGVsYXBzZWRSZWFsdGltZSgpOwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9VVElMU19TWVNURU1DTE9DS19ICi0KZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvVGV4dE91dHB1dC5oIGIvaW5jbHVkZS91dGlscy9UZXh0T3V0cHV0LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGQ4ZDg2YmEuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9UZXh0T3V0cHV0LmgKKysrIC9kZXYvbnVsbApAQCAtMSwxOTAgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9URVhUT1VUUFVUX0gKLSNkZWZpbmUgQU5EUk9JRF9URVhUT1VUUFVUX0gKLQotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBUZXh0T3V0cHV0Ci17Ci1wdWJsaWM6Ci0gICAgICAgICAgICAgICAgICAgICAgICBUZXh0T3V0cHV0KCkgeyB9Ci0gICAgdmlydHVhbCAgICAgICAgICAgICB+VGV4dE91dHB1dCgpIHsgfQotICAgIAotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgcHJpbnQoY29uc3QgY2hhciogdHh0LCBzaXplX3QgbGVuKSA9IDA7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICBtb3ZlSW5kZW50KGludCBkZWx0YSkgPSAwOwotICAgIAotICAgIGNsYXNzIEJ1bmRsZSB7Ci0gICAgcHVibGljOgotICAgICAgICBpbmxpbmUgQnVuZGxlKFRleHRPdXRwdXQmIHRvKSA6IG1UTyh0bykgeyB0by5wdXNoQnVuZGxlKCk7IH0KLSAgICAgICAgaW5saW5lIH5CdW5kbGUoKSB7IG1UTy5wb3BCdW5kbGUoKTsgfQotICAgIHByaXZhdGU6Ci0gICAgICAgIFRleHRPdXRwdXQmICAgICBtVE87Ci0gICAgfTsKLSAgICAKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIHB1c2hCdW5kbGUoKSA9IDA7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICBwb3BCdW5kbGUoKSA9IDA7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLy8gVGV4dCBvdXRwdXQgc3RyZWFtIGZvciBwcmludGluZyB0byB0aGUgbG9nICh2aWEgdXRpbHMvTG9nLmgpLgotZXh0ZXJuIFRleHRPdXRwdXQmIGFsb2c7Ci0KLS8vIFRleHQgb3V0cHV0IHN0cmVhbSBmb3IgcHJpbnRpbmcgdG8gc3Rkb3V0LgotZXh0ZXJuIFRleHRPdXRwdXQmIGFvdXQ7Ci0KLS8vIFRleHQgb3V0cHV0IHN0cmVhbSBmb3IgcHJpbnRpbmcgdG8gc3RkZXJyLgotZXh0ZXJuIFRleHRPdXRwdXQmIGFlcnI7Ci0KLXR5cGVkZWYgVGV4dE91dHB1dCYgKCpUZXh0T3V0cHV0TWFuaXBGdW5jKShUZXh0T3V0cHV0Jik7Ci0KLVRleHRPdXRwdXQmIGVuZGwoVGV4dE91dHB1dCYgdG8pOwotVGV4dE91dHB1dCYgaW5kZW50KFRleHRPdXRwdXQmIHRvKTsKLVRleHRPdXRwdXQmIGRlZGVudChUZXh0T3V0cHV0JiB0byk7Ci0KLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGNvbnN0IGNoYXIqIHN0cik7Ci1UZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCBjaGFyKTsgICAgIC8vIHdyaXRlcyByYXcgY2hhcmFjdGVyCi1UZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCBib29sKTsKLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGludCk7Ci1UZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCBsb25nKTsKLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIHVuc2lnbmVkIGludCk7Ci1UZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCB1bnNpZ25lZCBsb25nKTsKLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGxvbmcgbG9uZyk7Ci1UZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCB1bnNpZ25lZCBsb25nIGxvbmcpOwotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgZmxvYXQpOwotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgZG91YmxlKTsKLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIFRleHRPdXRwdXRNYW5pcEZ1bmMgZnVuYyk7Ci1UZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCBjb25zdCB2b2lkKik7Ci0KLWNsYXNzIFR5cGVDb2RlIAotewotcHVibGljOgotICAgIGlubGluZSBUeXBlQ29kZSh1aW50MzJfdCBjb2RlKTsKLSAgICBpbmxpbmUgflR5cGVDb2RlKCk7Ci0KLSAgICBpbmxpbmUgdWludDMyX3QgdHlwZUNvZGUoKSBjb25zdDsKLSAgICAKLXByaXZhdGU6Ci0gICAgdWludDMyX3QgbUNvZGU7Ci19OwotCi1UZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCBjb25zdCBUeXBlQ29kZSYgdmFsKTsKLQotY2xhc3MgSGV4RHVtcAotewotcHVibGljOgotICAgIEhleER1bXAoY29uc3Qgdm9pZCAqYnVmLCBzaXplX3Qgc2l6ZSwgc2l6ZV90IGJ5dGVzUGVyTGluZT0xNik7Ci0gICAgaW5saW5lIH5IZXhEdW1wKCk7Ci0gICAgCi0gICAgaW5saW5lIEhleER1bXAmIHNldEJ5dGVzUGVyTGluZShzaXplX3QgYnl0ZXNQZXJMaW5lKTsKLSAgICBpbmxpbmUgSGV4RHVtcCYgc2V0U2luZ2xlTGluZUN1dG9mZihpbnQzMl90IGJ5dGVzKTsKLSAgICBpbmxpbmUgSGV4RHVtcCYgc2V0QWxpZ25tZW50KHNpemVfdCBhbGlnbm1lbnQpOwotICAgIGlubGluZSBIZXhEdW1wJiBzZXRDQXJyYXlTdHlsZShib29sIGVuYWJsZWQpOwotICAgIAotICAgIGlubGluZSBjb25zdCB2b2lkKiBidWZmZXIoKSBjb25zdDsKLSAgICBpbmxpbmUgc2l6ZV90IHNpemUoKSBjb25zdDsKLSAgICBpbmxpbmUgc2l6ZV90IGJ5dGVzUGVyTGluZSgpIGNvbnN0OwotICAgIGlubGluZSBpbnQzMl90IHNpbmdsZUxpbmVDdXRvZmYoKSBjb25zdDsKLSAgICBpbmxpbmUgc2l6ZV90IGFsaWdubWVudCgpIGNvbnN0OwotICAgIGlubGluZSBib29sIGNhcnJheVN0eWxlKCkgY29uc3Q7Ci0KLXByaXZhdGU6Ci0gICAgY29uc3Qgdm9pZCogbUJ1ZmZlcjsKLSAgICBzaXplX3QgbVNpemU7Ci0gICAgc2l6ZV90IG1CeXRlc1BlckxpbmU7Ci0gICAgaW50MzJfdCBtU2luZ2xlTGluZUN1dG9mZjsKLSAgICBzaXplX3QgbUFsaWdubWVudDsKLSAgICBib29sIG1DQXJyYXlTdHlsZTsKLX07Ci0KLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGNvbnN0IEhleER1bXAmIHZhbCk7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gTm8gdXNlciBzZXJ2aWNhYmxlIHBhcnRzIGJlbG93LgotCi1pbmxpbmUgVGV4dE91dHB1dCYgZW5kbChUZXh0T3V0cHV0JiB0bykKLXsKLSAgICB0by5wcmludCgiXG4iLCAxKTsKLSAgICByZXR1cm4gdG87Ci19Ci0KLWlubGluZSBUZXh0T3V0cHV0JiBpbmRlbnQoVGV4dE91dHB1dCYgdG8pCi17Ci0gICAgdG8ubW92ZUluZGVudCgxKTsKLSAgICByZXR1cm4gdG87Ci19Ci0KLWlubGluZSBUZXh0T3V0cHV0JiBkZWRlbnQoVGV4dE91dHB1dCYgdG8pCi17Ci0gICAgdG8ubW92ZUluZGVudCgtMSk7Ci0gICAgcmV0dXJuIHRvOwotfQotCi1pbmxpbmUgVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgY29uc3QgY2hhciogc3RyKQotewotICAgIHRvLnByaW50KHN0ciwgc3RybGVuKHN0cikpOwotICAgIHJldHVybiB0bzsKLX0KLQotaW5saW5lIFRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGNoYXIgYykKLXsKLSAgICB0by5wcmludCgmYywgMSk7Ci0gICAgcmV0dXJuIHRvOwotfQotCi1pbmxpbmUgVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgVGV4dE91dHB1dE1hbmlwRnVuYyBmdW5jKQotewotICAgIHJldHVybiAoKmZ1bmMpKHRvKTsKLX0KLQotaW5saW5lIFR5cGVDb2RlOjpUeXBlQ29kZSh1aW50MzJfdCBjb2RlKSA6IG1Db2RlKGNvZGUpIHsgfQotaW5saW5lIFR5cGVDb2RlOjp+VHlwZUNvZGUoKSB7IH0KLWlubGluZSB1aW50MzJfdCBUeXBlQ29kZTo6dHlwZUNvZGUoKSBjb25zdCB7IHJldHVybiBtQ29kZTsgfQotCi1pbmxpbmUgSGV4RHVtcDo6fkhleER1bXAoKSB7IH0KLQotaW5saW5lIEhleER1bXAmIEhleER1bXA6OnNldEJ5dGVzUGVyTGluZShzaXplX3QgYnl0ZXNQZXJMaW5lKSB7Ci0gICAgbUJ5dGVzUGVyTGluZSA9IGJ5dGVzUGVyTGluZTsgcmV0dXJuICp0aGlzOwotfQotaW5saW5lIEhleER1bXAmIEhleER1bXA6OnNldFNpbmdsZUxpbmVDdXRvZmYoaW50MzJfdCBieXRlcykgewotICAgIG1TaW5nbGVMaW5lQ3V0b2ZmID0gYnl0ZXM7IHJldHVybiAqdGhpczsKLX0KLWlubGluZSBIZXhEdW1wJiBIZXhEdW1wOjpzZXRBbGlnbm1lbnQoc2l6ZV90IGFsaWdubWVudCkgewotICAgIG1BbGlnbm1lbnQgPSBhbGlnbm1lbnQ7IHJldHVybiAqdGhpczsKLX0KLWlubGluZSBIZXhEdW1wJiBIZXhEdW1wOjpzZXRDQXJyYXlTdHlsZShib29sIGVuYWJsZWQpIHsKLSAgICBtQ0FycmF5U3R5bGUgPSBlbmFibGVkOyByZXR1cm4gKnRoaXM7Ci19Ci0KLWlubGluZSBjb25zdCB2b2lkKiBIZXhEdW1wOjpidWZmZXIoKSBjb25zdCB7IHJldHVybiBtQnVmZmVyOyB9Ci1pbmxpbmUgc2l6ZV90IEhleER1bXA6OnNpemUoKSBjb25zdCB7IHJldHVybiBtU2l6ZTsgfQotaW5saW5lIHNpemVfdCBIZXhEdW1wOjpieXRlc1BlckxpbmUoKSBjb25zdCB7IHJldHVybiBtQnl0ZXNQZXJMaW5lOyB9Ci1pbmxpbmUgaW50MzJfdCBIZXhEdW1wOjpzaW5nbGVMaW5lQ3V0b2ZmKCkgY29uc3QgeyByZXR1cm4gbVNpbmdsZUxpbmVDdXRvZmY7IH0KLWlubGluZSBzaXplX3QgSGV4RHVtcDo6YWxpZ25tZW50KCkgY29uc3QgeyByZXR1cm4gbUFsaWdubWVudDsgfQotaW5saW5lIGJvb2wgSGV4RHVtcDo6Y2FycmF5U3R5bGUoKSBjb25zdCB7IHJldHVybiBtQ0FycmF5U3R5bGU7IH0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9URVhUT1VUUFVUX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvVGltZVV0aWxzLmggYi9pbmNsdWRlL3V0aWxzL1RpbWVVdGlscy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiMTllMDIxLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvVGltZVV0aWxzLmgKKysrIC9kZXYvbnVsbApAQCAtMSw4OSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1RJTUVfSAotI2RlZmluZSBBTkRST0lEX1RJTUVfSAotCi0jaW5jbHVkZSA8dGltZS5oPgotI2luY2x1ZGUgPGN1dGlscy90enRpbWUuaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDxzeXMvdGltZS5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmcxNi5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8qCi0gKiBUaGlzIGNsYXNzIGlzIHRoZSBjb3JlIGltcGxlbWVudGF0aW9uIG9mIHRoZSBhbmRyb2lkLnV0aWwuVGltZSBqYXZhCi0gKiBjbGFzcy4gIEl0IGRvZXNuJ3QgaW1wbGVtZW50IHNvbWUgb2YgdGhlIG1ldGhvZHMgdGhhdCBhcmUgaW1wbGVtZW50ZWQKLSAqIGluIEphdmEuICBUaGV5IGNvdWxkIGJlIGRvbmUgaGVyZSwgYnV0IGl0J3Mgbm90IGV4cGVjdGVkIHRoYXQgdGhpcyBjbGFzcwotICogd2lsbCBiZSB1c2VkLiAgSWYgdGhhdCBhc3N1bXB0aW9uIGlzIGluY29ycmVjdCwgZmVlbCBmcmVlIHRvIHVwZGF0ZSB0aGlzCi0gKiBmaWxlLiAgVGhlIHJlYXNvbiB0byBkbyBpdCBoZXJlIGlzIHRvIG5vdCBtaXggdGhlIGltcGxlbWVudGF0aW9uIG9mIHRoaXMKLSAqIGNsYXNzIGFuZCB0aGUgam5pIGdsdWUgY29kZS4KLSAqLwotY2xhc3MgVGltZQotewotcHVibGljOgotICAgIHN0cnVjdCB0bSB0OwotCi0gICAgLy8gdGhpcyBvYmplY3QgZG9lc24ndCBvd24gdGhpcyBzdHJpbmcKLSAgICBjb25zdCBjaGFyICp0aW1lem9uZTsKLQotICAgIGVudW0gewotICAgICAgICBTRUMgPSAxLAotICAgICAgICBNSU4gPSAyLAotICAgICAgICBIT1VSID0gMywKLSAgICAgICAgTURBWSA9IDQsCi0gICAgICAgIE1PTiA9IDUsCi0gICAgICAgIFlFQVIgPSA2LAotICAgICAgICBXREFZID0gNywKLSAgICAgICAgWURBWSA9IDgKLSAgICB9OwotCi0gICAgc3RhdGljIGludCBjb21wYXJlKFRpbWUmIGEsIFRpbWUmIGIpOwotCi0gICAgVGltZSgpOwotCi0gICAgdm9pZCBzd2l0Y2hUaW1lem9uZShjb25zdCBjaGFyICp0aW1lem9uZSk7Ci0gICAgU3RyaW5nOCBmb3JtYXQoY29uc3QgY2hhciAqZm9ybWF0LCBjb25zdCBzdHJ1Y3Qgc3RyZnRpbWVfbG9jYWxlICpsb2NhbGUpIGNvbnN0OwotICAgIHZvaWQgZm9ybWF0MjQ0NShzaG9ydCogYnVmLCBib29sIGhhc1RpbWUpIGNvbnN0OwotICAgIFN0cmluZzggdG9TdHJpbmcoKSBjb25zdDsKLSAgICB2b2lkIHNldFRvTm93KCk7Ci0gICAgaW50NjRfdCB0b01pbGxpcyhib29sIGlnbm9yZURzdCk7Ci0gICAgdm9pZCBzZXQoaW50NjRfdCBtaWxsaXMpOwotCi0gICAgaW5saW5lIHZvaWQgc2V0KGludCBzZWMsIGludCBtaW4sIGludCBob3VyLCBpbnQgbWRheSwgaW50IG1vbiwgaW50IHllYXIsCi0gICAgICAgICAgICBpbnQgaXNkc3QpCi0gICAgewotICAgICAgICB0aGlzLT50LnRtX3NlYyA9IHNlYzsKLSAgICAgICAgdGhpcy0+dC50bV9taW4gPSBtaW47Ci0gICAgICAgIHRoaXMtPnQudG1faG91ciA9IGhvdXI7Ci0gICAgICAgIHRoaXMtPnQudG1fbWRheSA9IG1kYXk7Ci0gICAgICAgIHRoaXMtPnQudG1fbW9uID0gbW9uOwotICAgICAgICB0aGlzLT50LnRtX3llYXIgPSB5ZWFyOwotICAgICAgICB0aGlzLT50LnRtX2lzZHN0ID0gaXNkc3Q7Ci0jaWZkZWYgSEFWRV9UTV9HTVRPRkYKLSAgICAgICAgdGhpcy0+dC50bV9nbXRvZmYgPSAwOwotI2VuZGlmCi0gICAgICAgIHRoaXMtPnQudG1fd2RheSA9IDA7Ci0gICAgICAgIHRoaXMtPnQudG1feWRheSA9IDA7Ci0gICAgfQotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfVElNRV9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL1RpbWVyUHJvYmUuaCBiL2luY2x1ZGUvdXRpbHMvVGltZXJQcm9iZS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmMmUzMmIyLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvVGltZXJQcm9iZS5oCisrKyAvZGV2L251bGwKQEAgLTEsNzIgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9USU1FUl9QUk9CRV9ICi0jZGVmaW5lIEFORFJPSURfVElNRVJfUFJPQkVfSAotCi0jaWYgMCAmJiBkZWZpbmVkKEhBVkVfUE9TSVhfQ0xPQ0tTKQotI2RlZmluZSBFTkFCTEVfVElNRVJfUFJPQkUgMQotI2Vsc2UKLSNkZWZpbmUgRU5BQkxFX1RJTUVSX1BST0JFIDAKLSNlbmRpZgotCi0jaWYgRU5BQkxFX1RJTUVSX1BST0JFCi0KLSNpbmNsdWRlIDx0aW1lLmg+Ci0jaW5jbHVkZSA8c3lzL3RpbWUuaD4KLSNpbmNsdWRlIDx1dGlscy9WZWN0b3IuaD4KLQotI2RlZmluZSBUSU1FUl9QUk9CRSh0YWcpIFwKLSAgICBzdGF0aWMgaW50IF90aW1lcl9zbG90XzsgXAotICAgIGFuZHJvaWQ6OlRpbWVyUHJvYmUgcHJvYmUodGFnLCAmX3RpbWVyX3Nsb3RfKQotI2RlZmluZSBUSU1FUl9QUk9CRV9FTkQoKSBwcm9iZS5lbmQoKQotI2Vsc2UKLSNkZWZpbmUgVElNRVJfUFJPQkUodGFnKQotI2RlZmluZSBUSU1FUl9QUk9CRV9FTkQoKQotI2VuZGlmCi0KLSNpZiBFTkFCTEVfVElNRVJfUFJPQkUKLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgVGltZXJQcm9iZSB7Ci1wdWJsaWM6Ci0gICAgVGltZXJQcm9iZShjb25zdCBjaGFyIHRhZ1tdLCBpbnQqIHNsb3QpOwotICAgIHZvaWQgZW5kKCk7Ci0gICAgflRpbWVyUHJvYmUoKTsKLXByaXZhdGU6Ci0gICAgc3RydWN0IEJ1Y2tldCB7Ci0gICAgICAgIGludCBtU3RhcnQsIG1SZWFsLCBtUHJvY2VzcywgbVRocmVhZCwgbUNvdW50OwotICAgICAgICBjb25zdCBjaGFyKiBtVGFnOwotICAgICAgICBpbnQqIG1TbG90UHRyOwotICAgICAgICBpbnQgbUluZGVudDsKLSAgICB9OwotICAgIHN0YXRpYyBWZWN0b3I8QnVja2V0PiBnQnVja2V0czsKLSAgICBzdGF0aWMgVGltZXJQcm9iZSogZ0V4ZWN1dGVDaGFpbjsKLSAgICBzdGF0aWMgaW50IGdJbmRlbnQ7Ci0gICAgc3RhdGljIHRpbWVzcGVjIGdSZWFsQmFzZTsKLSAgICBUaW1lclByb2JlKiBtTmV4dDsKLSAgICBzdGF0aWMgdWludDMyX3QgRWxhcHNlZFRpbWUoY29uc3QgdGltZXNwZWMmIHN0YXJ0LCBjb25zdCB0aW1lc3BlYyYgZW5kKTsKLSAgICB2b2lkIHByaW50KGNvbnN0IHRpbWVzcGVjJiByLCBjb25zdCB0aW1lc3BlYyYgcCwgY29uc3QgdGltZXNwZWMmIHQpIGNvbnN0OwotICAgIHRpbWVzcGVjIG1SZWFsU3RhcnQsIG1QU3RhcnQsIG1UU3RhcnQ7Ci0gICAgY29uc3QgY2hhciogbVRhZzsKLSAgICBpbnQgbUluZGVudDsKLSAgICBpbnQgbUJ1Y2tldDsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZgotI2VuZGlmCmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL1RpbWVycy5oIGIvaW5jbHVkZS91dGlscy9UaW1lcnMuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTYxMDM5OS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL1RpbWVycy5oCisrKyAvZGV2L251bGwKQEAgLTEsMTM3ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gVGltZXIgZnVuY3Rpb25zLgotLy8KLSNpZm5kZWYgX0xJQlNfVVRJTFNfVElNRVJTX0gKLSNkZWZpbmUgX0xJQlNfVVRJTFNfVElNRVJTX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHN5cy90aW1lLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gQyBBUEkKLQotI2lmZGVmIF9fY3BsdXNwbHVzCi1leHRlcm4gIkMiIHsKLSNlbmRpZgotCi10eXBlZGVmIGludDY0X3QgbnNlY3NfdDsgICAgICAgLy8gbmFuby1zZWNvbmRzCi0KLXN0YXRpYyBpbmxpbmUgbnNlY3NfdCBzZWNvbmRzX3RvX25hbm9zZWNvbmRzKG5zZWNzX3Qgc2VjcykKLXsKLSAgICByZXR1cm4gc2VjcyoxMDAwMDAwMDAwOwotfQotCi1zdGF0aWMgaW5saW5lIG5zZWNzX3QgbWlsbGlzZWNvbmRzX3RvX25hbm9zZWNvbmRzKG5zZWNzX3Qgc2VjcykKLXsKLSAgICByZXR1cm4gc2VjcyoxMDAwMDAwOwotfQotCi1zdGF0aWMgaW5saW5lIG5zZWNzX3QgbWljcm9zZWNvbmRzX3RvX25hbm9zZWNvbmRzKG5zZWNzX3Qgc2VjcykKLXsKLSAgICByZXR1cm4gc2VjcyoxMDAwOwotfQotCi1zdGF0aWMgaW5saW5lIG5zZWNzX3QgbmFub3NlY29uZHNfdG9fc2Vjb25kcyhuc2Vjc190IHNlY3MpCi17Ci0gICAgcmV0dXJuIHNlY3MvMTAwMDAwMDAwMDsKLX0KLQotc3RhdGljIGlubGluZSBuc2Vjc190IG5hbm9zZWNvbmRzX3RvX21pbGxpc2Vjb25kcyhuc2Vjc190IHNlY3MpCi17Ci0gICAgcmV0dXJuIHNlY3MvMTAwMDAwMDsKLX0KLQotc3RhdGljIGlubGluZSBuc2Vjc190IG5hbm9zZWNvbmRzX3RvX21pY3Jvc2Vjb25kcyhuc2Vjc190IHNlY3MpCi17Ci0gICAgcmV0dXJuIHNlY3MvMTAwMDsKLX0KLQotc3RhdGljIGlubGluZSBuc2Vjc190IHMybnMobnNlY3NfdCB2KSAge3JldHVybiBzZWNvbmRzX3RvX25hbm9zZWNvbmRzKHYpO30KLXN0YXRpYyBpbmxpbmUgbnNlY3NfdCBtczJucyhuc2Vjc190IHYpIHtyZXR1cm4gbWlsbGlzZWNvbmRzX3RvX25hbm9zZWNvbmRzKHYpO30KLXN0YXRpYyBpbmxpbmUgbnNlY3NfdCB1czJucyhuc2Vjc190IHYpIHtyZXR1cm4gbWljcm9zZWNvbmRzX3RvX25hbm9zZWNvbmRzKHYpO30KLXN0YXRpYyBpbmxpbmUgbnNlY3NfdCBuczJzKG5zZWNzX3QgdikgIHtyZXR1cm4gbmFub3NlY29uZHNfdG9fc2Vjb25kcyh2KTt9Ci1zdGF0aWMgaW5saW5lIG5zZWNzX3QgbnMybXMobnNlY3NfdCB2KSB7cmV0dXJuIG5hbm9zZWNvbmRzX3RvX21pbGxpc2Vjb25kcyh2KTt9Ci1zdGF0aWMgaW5saW5lIG5zZWNzX3QgbnMydXMobnNlY3NfdCB2KSB7cmV0dXJuIG5hbm9zZWNvbmRzX3RvX21pY3Jvc2Vjb25kcyh2KTt9Ci0KLXN0YXRpYyBpbmxpbmUgbnNlY3NfdCBzZWNvbmRzKG5zZWNzX3QgdikgICAgICB7IHJldHVybiBzMm5zKHYpOyB9Ci1zdGF0aWMgaW5saW5lIG5zZWNzX3QgbWlsbGlzZWNvbmRzKG5zZWNzX3QgdikgeyByZXR1cm4gbXMybnModik7IH0KLXN0YXRpYyBpbmxpbmUgbnNlY3NfdCBtaWNyb3NlY29uZHMobnNlY3NfdCB2KSB7IHJldHVybiB1czJucyh2KTsgfQotCi1lbnVtIHsKLSAgICBTWVNURU1fVElNRV9SRUFMVElNRSA9IDAsICAvLyBzeXN0ZW0td2lkZSByZWFsdGltZSBjbG9jawotICAgIFNZU1RFTV9USU1FX01PTk9UT05JQyA9IDEsIC8vIG1vbm90b25pYyB0aW1lIHNpbmNlIHVuc3BlY2lmaWVkIHN0YXJ0aW5nIHBvaW50Ci0gICAgU1lTVEVNX1RJTUVfUFJPQ0VTUyA9IDIsICAgLy8gaGlnaC1yZXNvbHV0aW9uIHBlci1wcm9jZXNzIGNsb2NrCi0gICAgU1lTVEVNX1RJTUVfVEhSRUFEID0gMyAgICAgLy8gaGlnaC1yZXNvbHV0aW9uIHBlci10aHJlYWQgY2xvY2sKLX07Ci0gICAgCi0vLyByZXR1cm4gdGhlIHN5c3RlbS10aW1lIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkIGNsb2NrCi0jaWZkZWYgX19jcGx1c3BsdXMKLW5zZWNzX3Qgc3lzdGVtVGltZShpbnQgY2xvY2sgPSBTWVNURU1fVElNRV9NT05PVE9OSUMpOwotI2Vsc2UKLW5zZWNzX3Qgc3lzdGVtVGltZShpbnQgY2xvY2spOwotI2VuZGlmIC8vIGRlZiBfX2NwbHVzcGx1cwotCi0vLyByZXR1cm4gdGhlIHN5c3RlbS10aW1lIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkIGNsb2NrCi1pbnQgc2xlZXBGb3JJbnRlcnZhbChsb25nIGludGVydmFsLCBzdHJ1Y3QgdGltZXZhbCogcE5leHRUaWNrKTsKLQotI2lmZGVmIF9fY3BsdXNwbHVzCi19IC8vIGV4dGVybiAiQyIKLSNlbmRpZgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIEMrKyBBUEkKLQotI2lmZGVmIF9fY3BsdXNwbHVzCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8qCi0gKiBUaW1lIHRoZSBkdXJhdGlvbiBvZiBzb21ldGhpbmcuCi0gKgotICogSW5jbHVkZXMgc29tZSB0aW1ldmFsIG1hbmlwdWxhdGlvbiBmdW5jdGlvbnMuCi0gKi8KLWNsYXNzIER1cmF0aW9uVGltZXIgewotcHVibGljOgotICAgIER1cmF0aW9uVGltZXIodm9pZCkge30KLSAgICB+RHVyYXRpb25UaW1lcih2b2lkKSB7fQotCi0gICAgLy8gU3RhcnQgdGhlIHRpbWVyLgotICAgIHZvaWQgc3RhcnQodm9pZCk7Ci0gICAgLy8gU3RvcCB0aGUgdGltZXIuCi0gICAgdm9pZCBzdG9wKHZvaWQpOwotICAgIC8vIEdldCB0aGUgZHVyYXRpb24gaW4gbWljcm9zZWNvbmRzLgotICAgIGxvbmcgbG9uZyBkdXJhdGlvblVzZWNzKHZvaWQpIGNvbnN0OwotCi0gICAgLy8gU3VidHJhY3QgdHdvIHRpbWV2YWxzLiAgUmV0dXJucyB0aGUgZGlmZmVyZW5jZSAocHR2MS1wdHYyKSBpbgotICAgIC8vIG1pY3Jvc2Vjb25kcy4KLSAgICBzdGF0aWMgbG9uZyBsb25nIHN1YnRyYWN0VGltZXZhbHMoY29uc3Qgc3RydWN0IHRpbWV2YWwqIHB0djEsCi0gICAgICAgIGNvbnN0IHN0cnVjdCB0aW1ldmFsKiBwdHYyKTsKLQotICAgIC8vIEFkZCB0aGUgc3BlY2lmaWVkIGFtb3VudCBvZiB0aW1lIHRvIHRoZSB0aW1ldmFsLgotICAgIHN0YXRpYyB2b2lkIGFkZFRvVGltZXZhbChzdHJ1Y3QgdGltZXZhbCogcHR2LCBsb25nIHVzZWMpOwotCi1wcml2YXRlOgotICAgIHN0cnVjdCB0aW1ldmFsICBtU3RhcnRXaGVuOwotICAgIHN0cnVjdCB0aW1ldmFsICBtU3RvcFdoZW47Ci19OwotCi19OyAvLyBhbmRyb2lkCi0jZW5kaWYgLy8gZGVmIF9fY3BsdXNwbHVzCi0KLSNlbmRpZiAvLyBfTElCU19VVElMU19USU1FUlNfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9UeXBlSGVscGVycy5oIGIvaW5jbHVkZS91dGlscy9UeXBlSGVscGVycy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjMDRjMzdmLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvVHlwZUhlbHBlcnMuaAorKysgL2Rldi9udWxsCkBAIC0xLDI1NCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1RZUEVfSEVMUEVSU19ICi0jZGVmaW5lIEFORFJPSURfVFlQRV9IRUxQRVJTX0gKLQotI2luY2x1ZGUgPG5ldz4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyoKLSAqIFR5cGVzIHRyYWl0cwotICovCi0gICAgCi10ZW1wbGF0ZSA8dHlwZW5hbWUgVD4gc3RydWN0IHRyYWl0X3RyaXZpYWxfY3RvciAgeyBlbnVtIHsgdmFsdWUgPSBmYWxzZSB9OyB9OwotdGVtcGxhdGUgPHR5cGVuYW1lIFQ+IHN0cnVjdCB0cmFpdF90cml2aWFsX2R0b3IgIHsgZW51bSB7IHZhbHVlID0gZmFsc2UgfTsgfTsKLXRlbXBsYXRlIDx0eXBlbmFtZSBUPiBzdHJ1Y3QgdHJhaXRfdHJpdmlhbF9jb3B5ICB7IGVudW0geyB2YWx1ZSA9IGZhbHNlIH07IH07Ci10ZW1wbGF0ZSA8dHlwZW5hbWUgVD4gc3RydWN0IHRyYWl0X3RyaXZpYWxfYXNzaWdueyBlbnVtIHsgdmFsdWUgPSBmYWxzZSB9OyB9OwotCi10ZW1wbGF0ZSA8dHlwZW5hbWUgVD4gc3RydWN0IHRyYWl0X3BvaW50ZXIgICAgIHsgZW51bSB7IHZhbHVlID0gZmFsc2UgfTsgfTsgICAgCi10ZW1wbGF0ZSA8dHlwZW5hbWUgVD4gc3RydWN0IHRyYWl0X3BvaW50ZXI8VCo+IHsgZW51bSB7IHZhbHVlID0gdHJ1ZSB9OyB9OwotCi0jZGVmaW5lIEFORFJPSURfQkFTSUNfVFlQRVNfVFJBSVRTKCBUICkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi0gICAgdGVtcGxhdGU8PiBzdHJ1Y3QgdHJhaXRfdHJpdmlhbF9jdG9yPCBUID4gIHsgZW51bSB7IHZhbHVlID0gdHJ1ZSB9OyB9OyAgICBcCi0gICAgdGVtcGxhdGU8PiBzdHJ1Y3QgdHJhaXRfdHJpdmlhbF9kdG9yPCBUID4gIHsgZW51bSB7IHZhbHVlID0gdHJ1ZSB9OyB9OyAgICBcCi0gICAgdGVtcGxhdGU8PiBzdHJ1Y3QgdHJhaXRfdHJpdmlhbF9jb3B5PCBUID4gIHsgZW51bSB7IHZhbHVlID0gdHJ1ZSB9OyB9OyAgICBcCi0gICAgdGVtcGxhdGU8PiBzdHJ1Y3QgdHJhaXRfdHJpdmlhbF9hc3NpZ248IFQgPnsgZW51bSB7IHZhbHVlID0gdHJ1ZSB9OyB9OyAKLQotI2RlZmluZSBBTkRST0lEX1RZUEVfVFJBSVRTKCBULCBjdG9yLCBkdG9yLCBjb3B5LCBhc3NpZ24gKSAgICAgICAgICAgICAgICAgICAgXAotICAgIHRlbXBsYXRlPD4gc3RydWN0IHRyYWl0X3RyaXZpYWxfY3RvcjwgVCA+ICB7IGVudW0geyB2YWx1ZSA9IGN0b3IgfTsgfTsgICAgXAotICAgIHRlbXBsYXRlPD4gc3RydWN0IHRyYWl0X3RyaXZpYWxfZHRvcjwgVCA+ICB7IGVudW0geyB2YWx1ZSA9IGR0b3IgfTsgfTsgICAgXAotICAgIHRlbXBsYXRlPD4gc3RydWN0IHRyYWl0X3RyaXZpYWxfY29weTwgVCA+ICB7IGVudW0geyB2YWx1ZSA9IGNvcHkgfTsgfTsgICAgXAotICAgIHRlbXBsYXRlPD4gc3RydWN0IHRyYWl0X3RyaXZpYWxfYXNzaWduPCBUID57IGVudW0geyB2YWx1ZSA9IGFzc2lnbiB9OyB9OyAKLQotdGVtcGxhdGUgPHR5cGVuYW1lIFRZUEU+Ci1zdHJ1Y3QgdHJhaXRzIHsKLSAgICBlbnVtIHsKLSAgICAgICAgaXNfcG9pbnRlciAgICAgICAgICA9IHRyYWl0X3BvaW50ZXI8VFlQRT46OnZhbHVlLAotICAgICAgICBoYXNfdHJpdmlhbF9jdG9yICAgID0gaXNfcG9pbnRlciB8fCB0cmFpdF90cml2aWFsX2N0b3I8VFlQRT46OnZhbHVlLAotICAgICAgICBoYXNfdHJpdmlhbF9kdG9yICAgID0gaXNfcG9pbnRlciB8fCB0cmFpdF90cml2aWFsX2R0b3I8VFlQRT46OnZhbHVlLAotICAgICAgICBoYXNfdHJpdmlhbF9jb3B5ICAgID0gaXNfcG9pbnRlciB8fCB0cmFpdF90cml2aWFsX2NvcHk8VFlQRT46OnZhbHVlLAotICAgICAgICBoYXNfdHJpdmlhbF9hc3NpZ24gID0gaXNfcG9pbnRlciB8fCB0cmFpdF90cml2aWFsX2Fzc2lnbjxUWVBFPjo6dmFsdWUgICAKLSAgICB9OwotfTsKLQotdGVtcGxhdGUgPHR5cGVuYW1lIFQsIHR5cGVuYW1lIFU+Ci1zdHJ1Y3QgYWdncmVnYXRlX3RyYWl0cyB7Ci0gICAgZW51bSB7Ci0gICAgICAgIGlzX3BvaW50ZXIgICAgICAgICAgPSBmYWxzZSwKLSAgICAgICAgaGFzX3RyaXZpYWxfY3RvciAgICA9IHRyYWl0czxUPjo6aGFzX3RyaXZpYWxfY3RvciAmJiB0cmFpdHM8VT46Omhhc190cml2aWFsX2N0b3IsCi0gICAgICAgIGhhc190cml2aWFsX2R0b3IgICAgPSB0cmFpdHM8VD46Omhhc190cml2aWFsX2R0b3IgJiYgdHJhaXRzPFU+OjpoYXNfdHJpdmlhbF9kdG9yLAotICAgICAgICBoYXNfdHJpdmlhbF9jb3B5ICAgID0gdHJhaXRzPFQ+OjpoYXNfdHJpdmlhbF9jb3B5ICYmIHRyYWl0czxVPjo6aGFzX3RyaXZpYWxfY29weSwKLSAgICAgICAgaGFzX3RyaXZpYWxfYXNzaWduICA9IHRyYWl0czxUPjo6aGFzX3RyaXZpYWxfYXNzaWduICYmIHRyYWl0czxVPjo6aGFzX3RyaXZpYWxfYXNzaWduCi0gICAgfTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0vKgotICogYmFzaWMgdHlwZXMgdHJhaXRzCi0gKi8KLSAKLUFORFJPSURfQkFTSUNfVFlQRVNfVFJBSVRTKCB2b2lkICk7Ci1BTkRST0lEX0JBU0lDX1RZUEVTX1RSQUlUUyggYm9vbCApOwotQU5EUk9JRF9CQVNJQ19UWVBFU19UUkFJVFMoIGNoYXIgKTsKLUFORFJPSURfQkFTSUNfVFlQRVNfVFJBSVRTKCB1bnNpZ25lZCBjaGFyICk7Ci1BTkRST0lEX0JBU0lDX1RZUEVTX1RSQUlUUyggc2hvcnQgKTsKLUFORFJPSURfQkFTSUNfVFlQRVNfVFJBSVRTKCB1bnNpZ25lZCBzaG9ydCApOwotQU5EUk9JRF9CQVNJQ19UWVBFU19UUkFJVFMoIGludCApOwotQU5EUk9JRF9CQVNJQ19UWVBFU19UUkFJVFMoIHVuc2lnbmVkIGludCApOwotQU5EUk9JRF9CQVNJQ19UWVBFU19UUkFJVFMoIGxvbmcgKTsKLUFORFJPSURfQkFTSUNfVFlQRVNfVFJBSVRTKCB1bnNpZ25lZCBsb25nICk7Ci1BTkRST0lEX0JBU0lDX1RZUEVTX1RSQUlUUyggbG9uZyBsb25nICk7Ci1BTkRST0lEX0JBU0lDX1RZUEVTX1RSQUlUUyggdW5zaWduZWQgbG9uZyBsb25nICk7Ci1BTkRST0lEX0JBU0lDX1RZUEVTX1RSQUlUUyggZmxvYXQgKTsKLUFORFJPSURfQkFTSUNfVFlQRVNfVFJBSVRTKCBkb3VibGUgKTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSAgICAKLS8qCi0gKiBjb21wYXJlIGFuZCBvcmRlciB0eXBlcwotICovCi0KLXRlbXBsYXRlPHR5cGVuYW1lIFRZUEU+IGlubGluZQotaW50IHN0cmljdGx5X29yZGVyX3R5cGUoY29uc3QgVFlQRSYgbGhzLCBjb25zdCBUWVBFJiByaHMpIHsKLSAgICByZXR1cm4gKGxocyA8IHJocykgPyAxIDogMDsKLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgVFlQRT4gaW5saW5lCi1pbnQgY29tcGFyZV90eXBlKGNvbnN0IFRZUEUmIGxocywgY29uc3QgVFlQRSYgcmhzKSB7Ci0gICAgcmV0dXJuIHN0cmljdGx5X29yZGVyX3R5cGUocmhzLCBsaHMpIC0gc3RyaWN0bHlfb3JkZXJfdHlwZShsaHMsIHJocyk7Ci19Ci0KLS8qCi0gKiBjcmVhdGUsIGRlc3Ryb3ksIGNvcHkgYW5kIGFzc2lnbiB0eXBlcy4uLgotICovCi0gCi10ZW1wbGF0ZTx0eXBlbmFtZSBUWVBFPiBpbmxpbmUKLXZvaWQgY29uc3RydWN0X3R5cGUoVFlQRSogcCwgc2l6ZV90IG4pIHsKLSAgICBpZiAoIXRyYWl0czxUWVBFPjo6aGFzX3RyaXZpYWxfY3RvcikgewotICAgICAgICB3aGlsZSAobi0tKSB7Ci0gICAgICAgICAgICBuZXcocCsrKSBUWVBFOwotICAgICAgICB9Ci0gICAgfQotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUWVBFPiBpbmxpbmUKLXZvaWQgZGVzdHJveV90eXBlKFRZUEUqIHAsIHNpemVfdCBuKSB7Ci0gICAgaWYgKCF0cmFpdHM8VFlQRT46Omhhc190cml2aWFsX2R0b3IpIHsKLSAgICAgICAgd2hpbGUgKG4tLSkgewotICAgICAgICAgICAgcC0+flRZUEUoKTsKLSAgICAgICAgICAgIHArKzsKLSAgICAgICAgfQotICAgIH0KLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgVFlQRT4gaW5saW5lCi12b2lkIGNvcHlfdHlwZShUWVBFKiBkLCBjb25zdCBUWVBFKiBzLCBzaXplX3QgbikgewotICAgIGlmICghdHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9jb3B5KSB7Ci0gICAgICAgIHdoaWxlIChuLS0pIHsKLSAgICAgICAgICAgIG5ldyhkKSBUWVBFKCpzKTsKLSAgICAgICAgICAgIGQrKywgcysrOwotICAgICAgICB9Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgbWVtY3B5KGQscyxuKnNpemVvZihUWVBFKSk7Ci0gICAgfQotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUWVBFPiBpbmxpbmUKLXZvaWQgYXNzaWduX3R5cGUoVFlQRSogZCwgY29uc3QgVFlQRSogcywgc2l6ZV90IG4pIHsKLSAgICBpZiAoIXRyYWl0czxUWVBFPjo6aGFzX3RyaXZpYWxfYXNzaWduKSB7Ci0gICAgICAgIHdoaWxlIChuLS0pIHsKLSAgICAgICAgICAgICpkKysgPSAqcysrOwotICAgICAgICB9Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgbWVtY3B5KGQscyxuKnNpemVvZihUWVBFKSk7Ci0gICAgfQotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUWVBFPiBpbmxpbmUKLXZvaWQgc3BsYXRfdHlwZShUWVBFKiB3aGVyZSwgY29uc3QgVFlQRSogd2hhdCwgc2l6ZV90IG4pIHsKLSAgICBpZiAoIXRyYWl0czxUWVBFPjo6aGFzX3RyaXZpYWxfY29weSkgewotICAgICAgICB3aGlsZSAobi0tKSB7Ci0gICAgICAgICAgICBuZXcod2hlcmUpIFRZUEUoKndoYXQpOwotICAgICAgICAgICAgd2hlcmUrKzsKLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgICB3aGlsZSAobi0tKSB7Ci0gICAgICAgICAgICAgKndoZXJlKysgPSAqd2hhdDsKLSAgICAgICAgfQotICAgIH0KLX0KLQotdGVtcGxhdGU8dHlwZW5hbWUgVFlQRT4gaW5saW5lCi12b2lkIG1vdmVfZm9yd2FyZF90eXBlKFRZUEUqIGQsIGNvbnN0IFRZUEUqIHMsIHNpemVfdCBuID0gMSkgewotICAgIGlmICghdHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9jb3B5IHx8ICF0cmFpdHM8VFlQRT46Omhhc190cml2aWFsX2R0b3IpIHsKLSAgICAgICAgZCArPSBuOwotICAgICAgICBzICs9IG47Ci0gICAgICAgIHdoaWxlIChuLS0pIHsKLSAgICAgICAgICAgIC0tZCwgLS1zOwotICAgICAgICAgICAgaWYgKCF0cmFpdHM8VFlQRT46Omhhc190cml2aWFsX2NvcHkpIHsKLSAgICAgICAgICAgICAgICBuZXcoZCkgVFlQRSgqcyk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICpkID0gKnM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoIXRyYWl0czxUWVBFPjo6aGFzX3RyaXZpYWxfZHRvcikgewotICAgICAgICAgICAgICAgIHMtPn5UWVBFKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBtZW1tb3ZlKGQscyxuKnNpemVvZihUWVBFKSk7Ci0gICAgfQotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUWVBFPiBpbmxpbmUKLXZvaWQgbW92ZV9iYWNrd2FyZF90eXBlKFRZUEUqIGQsIGNvbnN0IFRZUEUqIHMsIHNpemVfdCBuID0gMSkgewotICAgIGlmICghdHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9jb3B5IHx8ICF0cmFpdHM8VFlQRT46Omhhc190cml2aWFsX2R0b3IpIHsKLSAgICAgICAgd2hpbGUgKG4tLSkgewotICAgICAgICAgICAgaWYgKCF0cmFpdHM8VFlQRT46Omhhc190cml2aWFsX2NvcHkpIHsKLSAgICAgICAgICAgICAgICBuZXcoZCkgVFlQRSgqcyk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICpkID0gKnM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoIXRyYWl0czxUWVBFPjo6aGFzX3RyaXZpYWxfZHRvcikgewotICAgICAgICAgICAgICAgIHMtPn5UWVBFKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBkKyssIHMrKzsKLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIG1lbW1vdmUoZCxzLG4qc2l6ZW9mKFRZUEUpKTsKLSAgICB9Ci19Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLyoKLSAqIGEga2V5L3ZhbHVlIHBhaXIKLSAqLwotCi10ZW1wbGF0ZSA8dHlwZW5hbWUgS0VZLCB0eXBlbmFtZSBWQUxVRT4KLXN0cnVjdCBrZXlfdmFsdWVfcGFpcl90IHsKLSAgICBLRVkgICAgIGtleTsKLSAgICBWQUxVRSAgIHZhbHVlOwotICAgIGtleV92YWx1ZV9wYWlyX3QoKSB7IH0KLSAgICBrZXlfdmFsdWVfcGFpcl90KGNvbnN0IGtleV92YWx1ZV9wYWlyX3QmIG8pIDoga2V5KG8ua2V5KSwgdmFsdWUoby52YWx1ZSkgeyB9Ci0gICAga2V5X3ZhbHVlX3BhaXJfdChjb25zdCBLRVkmIGssIGNvbnN0IFZBTFVFJiB2KSA6IGtleShrKSwgdmFsdWUodikgIHsgfQotICAgIGtleV92YWx1ZV9wYWlyX3QoY29uc3QgS0VZJiBrKSA6IGtleShrKSB7IH0KLSAgICBpbmxpbmUgYm9vbCBvcGVyYXRvciA8IChjb25zdCBrZXlfdmFsdWVfcGFpcl90JiBvKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiBzdHJpY3RseV9vcmRlcl90eXBlKGtleSwgby5rZXkpOwotICAgIH0KLX07Ci0KLXRlbXBsYXRlPD4KLXRlbXBsYXRlIDx0eXBlbmFtZSBLLCB0eXBlbmFtZSBWPgotc3RydWN0IHRyYWl0X3RyaXZpYWxfY3Rvcjwga2V5X3ZhbHVlX3BhaXJfdDxLLCBWPiA+Ci17IGVudW0geyB2YWx1ZSA9IGFnZ3JlZ2F0ZV90cmFpdHM8SyxWPjo6aGFzX3RyaXZpYWxfY3RvciB9OyB9OwotdGVtcGxhdGU8PiAKLXRlbXBsYXRlIDx0eXBlbmFtZSBLLCB0eXBlbmFtZSBWPgotc3RydWN0IHRyYWl0X3RyaXZpYWxfZHRvcjwga2V5X3ZhbHVlX3BhaXJfdDxLLCBWPiA+Ci17IGVudW0geyB2YWx1ZSA9IGFnZ3JlZ2F0ZV90cmFpdHM8SyxWPjo6aGFzX3RyaXZpYWxfZHRvciB9OyB9OwotdGVtcGxhdGU8PiAKLXRlbXBsYXRlIDx0eXBlbmFtZSBLLCB0eXBlbmFtZSBWPgotc3RydWN0IHRyYWl0X3RyaXZpYWxfY29weTwga2V5X3ZhbHVlX3BhaXJfdDxLLCBWPiA+Ci17IGVudW0geyB2YWx1ZSA9IGFnZ3JlZ2F0ZV90cmFpdHM8SyxWPjo6aGFzX3RyaXZpYWxfY29weSB9OyB9OwotdGVtcGxhdGU8PiAKLXRlbXBsYXRlIDx0eXBlbmFtZSBLLCB0eXBlbmFtZSBWPgotc3RydWN0IHRyYWl0X3RyaXZpYWxfYXNzaWduPCBrZXlfdmFsdWVfcGFpcl90PEssIFY+ID4KLXsgZW51bSB7IHZhbHVlID0gYWdncmVnYXRlX3RyYWl0czxLLFY+OjpoYXNfdHJpdmlhbF9hc3NpZ259O307Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2VuZGlmIC8vIEFORFJPSURfVFlQRV9IRUxQRVJTX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvVmVjdG9yLmggYi9pbmNsdWRlL3V0aWxzL1ZlY3Rvci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiZTM2NWQ4Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvVmVjdG9yLmgKKysrIC9kZXYvbnVsbApAQCAtMSwzNTkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9WRUNUT1JfSAotI2RlZmluZSBBTkRST0lEX1ZFQ1RPUl9ICi0KLSNpbmNsdWRlIDxuZXc+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLSNpbmNsdWRlIDx1dGlscy9WZWN0b3JJbXBsLmg+Ci0jaW5jbHVkZSA8dXRpbHMvVHlwZUhlbHBlcnMuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyohCi0gKiBUaGUgbWFpbiB0ZW1wbGF0ZWQgdmVjdG9yIGNsYXNzIGVuc3VyaW5nIHR5cGUgc2FmZXR5Ci0gKiB3aGlsZSBtYWtpbmcgdXNlIG9mIFZlY3RvckltcGwuCi0gKiBUaGlzIGlzIHRoZSBjbGFzcyB1c2VycyB3YW50IHRvIHVzZS4KLSAqLwotCi10ZW1wbGF0ZSA8Y2xhc3MgVFlQRT4KLWNsYXNzIFZlY3RvciA6IHByaXZhdGUgVmVjdG9ySW1wbAotewotcHVibGljOgotICAgICAgICAgICAgdHlwZWRlZiBUWVBFICAgIHZhbHVlX3R5cGU7Ci0gICAgCi0gICAgLyohIAotICAgICAqIENvbnN0cnVjdG9ycyBhbmQgZGVzdHJ1Y3RvcnMKLSAgICAgKi8KLSAgICAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBWZWN0b3IoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBWZWN0b3IoY29uc3QgVmVjdG9yPFRZUEU+JiByaHMpOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgICAgIH5WZWN0b3IoKTsKLQotICAgIC8qISBjb3B5IG9wZXJhdG9yICovCi0gICAgICAgICAgICBjb25zdCBWZWN0b3I8VFlQRT4mICAgICBvcGVyYXRvciA9IChjb25zdCBWZWN0b3I8VFlQRT4mIHJocykgY29uc3Q7Ci0gICAgICAgICAgICBWZWN0b3I8VFlQRT4mICAgICAgICAgICBvcGVyYXRvciA9IChjb25zdCBWZWN0b3I8VFlQRT4mIHJocyk7ICAgIAotCi0gICAgLyoKLSAgICAgKiBlbXB0eSB0aGUgdmVjdG9yCi0gICAgICovCi0KLSAgICBpbmxpbmUgIHZvaWQgICAgICAgICAgICBjbGVhcigpICAgICAgICAgICAgIHsgVmVjdG9ySW1wbDo6Y2xlYXIoKTsgfQotCi0gICAgLyohIAotICAgICAqIHZlY3RvciBzdGF0cwotICAgICAqLwotCi0gICAgLy8hIHJldHVybnMgbnVtYmVyIG9mIGl0ZW1zIGluIHRoZSB2ZWN0b3IKLSAgICBpbmxpbmUgIHNpemVfdCAgICAgICAgICBzaXplKCkgY29uc3QgICAgICAgICAgICAgICAgeyByZXR1cm4gVmVjdG9ySW1wbDo6c2l6ZSgpOyB9Ci0gICAgLy8hIHJldHVybnMgd2V0aGVyIG9yIG5vdCB0aGUgdmVjdG9yIGlzIGVtcHR5Ci0gICAgaW5saW5lICBib29sICAgICAgICAgICAgaXNFbXB0eSgpIGNvbnN0ICAgICAgICAgICAgIHsgcmV0dXJuIFZlY3RvckltcGw6OmlzRW1wdHkoKTsgfQotICAgIC8vISByZXR1cm5zIGhvdyBtYW55IGl0ZW1zIGNhbiBiZSBzdG9yZWQgd2l0aG91dCByZWFsbG9jYXRpbmcgdGhlIGJhY2tpbmcgc3RvcmUKLSAgICBpbmxpbmUgIHNpemVfdCAgICAgICAgICBjYXBhY2l0eSgpIGNvbnN0ICAgICAgICAgICAgeyByZXR1cm4gVmVjdG9ySW1wbDo6Y2FwYWNpdHkoKTsgfQotICAgIC8vISBzZXRzdCB0aGUgY2FwYWNpdHkuIGNhcGFjaXR5IGNhbiBuZXZlciBiZSByZWR1Y2VkIGxlc3MgdGhhbiBzaXplKCkKLSAgICBpbmxpbmUgIHNzaXplX3QgICAgICAgICBzZXRDYXBhY2l0eShzaXplX3Qgc2l6ZSkgICAgeyByZXR1cm4gVmVjdG9ySW1wbDo6c2V0Q2FwYWNpdHkoc2l6ZSk7IH0KLQotICAgIC8qISAKLSAgICAgKiBDLXN0eWxlIGFycmF5IGFjY2VzcwotICAgICAqLwotICAgICAKLSAgICAvLyEgcmVhZC1vbmx5IEMtc3R5bGUgYWNjZXNzIAotICAgIGlubGluZSAgY29uc3QgVFlQRSogICAgIGFycmF5KCkgY29uc3Q7Ci0gICAgLy8hIHJlYWQtd3JpdGUgQy1zdHlsZSBhY2Nlc3MKLSAgICAgICAgICAgIFRZUEUqICAgICAgICAgICBlZGl0QXJyYXkoKTsKLSAgICAKLSAgICAvKiEgCi0gICAgICogYWNjZXNzb3JzCi0gICAgICovCi0KLSAgICAvLyEgcmVhZC1vbmx5IGFjY2VzcyB0byBhbiBpdGVtIGF0IGEgZ2l2ZW4gaW5kZXgKLSAgICBpbmxpbmUgIGNvbnN0IFRZUEUmICAgICBvcGVyYXRvciBbXSAoc2l6ZV90IGluZGV4KSBjb25zdDsKLSAgICAvLyEgYWx0ZXJuYXRlIG5hbWUgZm9yIG9wZXJhdG9yIFtdCi0gICAgaW5saW5lICBjb25zdCBUWVBFJiAgICAgaXRlbUF0KHNpemVfdCBpbmRleCkgY29uc3Q7Ci0gICAgLy8hIHN0YWNrLXVzYWdlIG9mIHRoZSB2ZWN0b3IuIHJldHVybnMgdGhlIHRvcCBvZiB0aGUgc3RhY2sgKGxhc3QgZWxlbWVudCkKLSAgICAgICAgICAgIGNvbnN0IFRZUEUmICAgICB0b3AoKSBjb25zdDsKLSAgICAvLyEgc2FtZSBhcyBvcGVyYXRvciBbXSwgYnV0IGFsbG93cyB0byBhY2Nlc3MgdGhlIHZlY3RvciBiYWNrd2FyZCAoZnJvbSB0aGUgZW5kKSB3aXRoIGEgbmVnYXRpdmUgaW5kZXgKLSAgICAgICAgICAgIGNvbnN0IFRZUEUmICAgICBtaXJyb3JJdGVtQXQoc3NpemVfdCBpbmRleCkgY29uc3Q7Ci0KLSAgICAvKiEKLSAgICAgKiBtb2RpZmluZyB0aGUgYXJyYXkKLSAgICAgKi8KLQotICAgIC8vISBjb3B5LW9uIHdyaXRlIHN1cHBvcnQsIGdyYW50cyB3cml0ZSBhY2Nlc3MgdG8gYW4gaXRlbQotICAgICAgICAgICAgVFlQRSYgICAgICAgICAgIGVkaXRJdGVtQXQoc2l6ZV90IGluZGV4KTsKLSAgICAvLyEgZ3JhbnRzIHJpZ2h0IGFjY2VzIHRvIHRoZSB0b3Agb2YgdGhlIHN0YWNrIChsYXN0IGVsZW1lbnQpCi0gICAgICAgICAgICBUWVBFJiAgICAgICAgICAgZWRpdFRvcCgpOwotCi0gICAgICAgICAgICAvKiEgCi0gICAgICAgICAgICAgKiBhcHBlbmQvaW5zZXJ0IGFub3RoZXIgdmVjdG9yCi0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIAotICAgIC8vISBpbnNlcnQgYW5vdGhlciB2ZWN0b3IgYXQgYSBnaXZlbiBpbmRleAotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIGluc2VydFZlY3RvckF0KGNvbnN0IFZlY3RvcjxUWVBFPiYgdmVjdG9yLCBzaXplX3QgaW5kZXgpOwotCi0gICAgLy8hIGFwcGVuZCBhbm90aGVyIHZlY3RvciBhdCB0aGUgZW5kIG9mIHRoaXMgb25lCi0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgYXBwZW5kVmVjdG9yKGNvbnN0IFZlY3RvcjxUWVBFPiYgdmVjdG9yKTsKLQotCi0gICAgICAgICAgICAvKiEgCi0gICAgICAgICAgICAgKiBhZGQvaW5zZXJ0L3JlcGxhY2UgaXRlbXMKLSAgICAgICAgICAgICAqLwotICAgICAgICAgICAgIAotICAgIC8vISBpbnNlcnQgb25lIG9yIHNldmVyYWwgaXRlbXMgaW5pdGlhbGl6ZWQgd2l0aCB0aGVpciBkZWZhdWx0IGNvbnN0cnVjdG9yCi0gICAgaW5saW5lICBzc2l6ZV90ICAgICAgICAgaW5zZXJ0QXQoc2l6ZV90IGluZGV4LCBzaXplX3QgbnVtSXRlbXMgPSAxKTsKLSAgICAvLyEgaW5zZXJ0IG9uIG9uciBzZXZlcmFsIGl0ZW1zIGluaXRpYWxpemVkIGZyb20gYSBwcm90b3R5cGUgaXRlbQotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIGluc2VydEF0KGNvbnN0IFRZUEUmIHByb3RvdHlwZV9pdGVtLCBzaXplX3QgaW5kZXgsIHNpemVfdCBudW1JdGVtcyA9IDEpOwotICAgIC8vISBwb3AgdGhlIHRvcCBvZiB0aGUgc3RhY2sgKHJlbW92ZXMgdGhlIGxhc3QgZWxlbWVudCkuIE5vLW9wIGlmIHRoZSBzdGFjaydzIGVtcHR5Ci0gICAgaW5saW5lICB2b2lkICAgICAgICAgICAgcG9wKCk7Ci0gICAgLy8hIHB1c2hlcyBhbiBpdGVtIGluaXRpYWxpemVkIHdpdGggaXRzIGRlZmF1bHQgY29uc3RydWN0b3IKLSAgICBpbmxpbmUgIHZvaWQgICAgICAgICAgICBwdXNoKCk7Ci0gICAgLy8hIHB1c2hlcyBhbiBpdGVtIG9uIHRoZSB0b3Agb2YgdGhlIHN0YWNrCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgcHVzaChjb25zdCBUWVBFJiBpdGVtKTsKLSAgICAvLyEgc2FtZSBhcyBwdXNoKCkgYnV0IHJldHVybnMgdGhlIGluZGV4IHRoZSBpdGVtIHdhcyBhZGRlZCBhdCAob3IgYW4gZXJyb3IpCi0gICAgaW5saW5lICBzc2l6ZV90ICAgICAgICAgYWRkKCk7Ci0gICAgLy8hIHNhbWUgYXMgcHVzaCgpIGJ1dCByZXR1cm5zIHRoZSBpbmRleCB0aGUgaXRlbSB3YXMgYWRkZWQgYXQgKG9yIGFuIGVycm9yKQotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIGFkZChjb25zdCBUWVBFJiBpdGVtKTsgICAgICAgICAgICAKLSAgICAvLyEgcmVwbGFjZSBhbiBpdGVtIHdpdGggYSBuZXcgb25lIGluaXRpYWxpemVkIHdpdGggaXRzIGRlZmF1bHQgY29uc3RydWN0b3IKLSAgICBpbmxpbmUgIHNzaXplX3QgICAgICAgICByZXBsYWNlQXQoc2l6ZV90IGluZGV4KTsKLSAgICAvLyEgcmVwbGFjZSBhbiBpdGVtIHdpdGggYSBuZXcgb25lCi0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgcmVwbGFjZUF0KGNvbnN0IFRZUEUmIGl0ZW0sIHNpemVfdCBpbmRleCk7Ci0KLSAgICAvKiEKLSAgICAgKiByZW1vdmUgaXRlbXMKLSAgICAgKi8KLQotICAgIC8vISByZW1vdmUgc2V2ZXJhbCBpdGVtcwotICAgIGlubGluZSAgc3NpemVfdCAgICAgICAgIHJlbW92ZUl0ZW1zQXQoc2l6ZV90IGluZGV4LCBzaXplX3QgY291bnQgPSAxKTsKLSAgICAvLyEgcmVtb3ZlIG9uZSBpdGVtCi0gICAgaW5saW5lICBzc2l6ZV90ICAgICAgICAgcmVtb3ZlQXQoc2l6ZV90IGluZGV4KSAgeyByZXR1cm4gcmVtb3ZlSXRlbXNBdChpbmRleCk7IH0KLQotICAgIC8qIQotICAgICAqIHNvcnQgKHN0YWJsZSkgdGhlIGFycmF5Ci0gICAgICovCi0gICAgIAotICAgICB0eXBlZGVmIGludCAoKmNvbXBhcl90KShjb25zdCBUWVBFKiBsaHMsIGNvbnN0IFRZUEUqIHJocyk7Ci0gICAgIHR5cGVkZWYgaW50ICgqY29tcGFyX3JfdCkoY29uc3QgVFlQRSogbGhzLCBjb25zdCBUWVBFKiByaHMsIHZvaWQqIHN0YXRlKTsKLSAgICAgCi0gICAgIGlubGluZSBzdGF0dXNfdCAgICAgICAgc29ydChjb21wYXJfdCBjbXApOwotICAgICBpbmxpbmUgc3RhdHVzX3QgICAgICAgIHNvcnQoY29tcGFyX3JfdCBjbXAsIHZvaWQqIHN0YXRlKTsKLQotcHJvdGVjdGVkOgotICAgIHZpcnR1YWwgdm9pZCAgICBkb19jb25zdHJ1Y3Qodm9pZCogc3RvcmFnZSwgc2l6ZV90IG51bSkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkICAgIGRvX2Rlc3Ryb3kodm9pZCogc3RvcmFnZSwgc2l6ZV90IG51bSkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkICAgIGRvX2NvcHkodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkICAgIGRvX3NwbGF0KHZvaWQqIGRlc3QsIGNvbnN0IHZvaWQqIGl0ZW0sIHNpemVfdCBudW0pIGNvbnN0OwotICAgIHZpcnR1YWwgdm9pZCAgICBkb19tb3ZlX2ZvcndhcmQodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkICAgIGRvX21vdmVfYmFja3dhcmQodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3Q7Ci19OwotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gTm8gdXNlciBzZXJ2aWNlYWJsZSBwYXJ0cyBmcm9tIGhlcmUuLi4KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLVZlY3RvcjxUWVBFPjo6VmVjdG9yKCkKLSAgICA6IFZlY3RvckltcGwoc2l6ZW9mKFRZUEUpLAotICAgICAgICAgICAgICAgICgodHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9jdG9yICAgPyBIQVNfVFJJVklBTF9DVE9SICAgOiAwKQotICAgICAgICAgICAgICAgIHwodHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9kdG9yICAgPyBIQVNfVFJJVklBTF9EVE9SICAgOiAwKQotICAgICAgICAgICAgICAgIHwodHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9jb3B5ICAgPyBIQVNfVFJJVklBTF9DT1BZICAgOiAwKQotICAgICAgICAgICAgICAgIHwodHJhaXRzPFRZUEU+OjpoYXNfdHJpdmlhbF9hc3NpZ24gPyBIQVNfVFJJVklBTF9BU1NJR04gOiAwKSkKLSAgICAgICAgICAgICAgICApCi17Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotVmVjdG9yPFRZUEU+OjpWZWN0b3IoY29uc3QgVmVjdG9yPFRZUEU+JiByaHMpCi0gICAgOiBWZWN0b3JJbXBsKHJocykgewotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLVZlY3RvcjxUWVBFPjo6flZlY3RvcigpIHsKLSAgICBmaW5pc2hfdmVjdG9yKCk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotVmVjdG9yPFRZUEU+JiBWZWN0b3I8VFlQRT46Om9wZXJhdG9yID0gKGNvbnN0IFZlY3RvcjxUWVBFPiYgcmhzKSB7Ci0gICAgVmVjdG9ySW1wbDo6b3BlcmF0b3IgPSAocmhzKTsKLSAgICByZXR1cm4gKnRoaXM7IAotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLWNvbnN0IFZlY3RvcjxUWVBFPiYgVmVjdG9yPFRZUEU+OjpvcGVyYXRvciA9IChjb25zdCBWZWN0b3I8VFlQRT4mIHJocykgY29uc3QgewotICAgIFZlY3RvckltcGw6Om9wZXJhdG9yID0gKHJocyk7Ci0gICAgcmV0dXJuICp0aGlzOyAKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1jb25zdCBUWVBFKiBWZWN0b3I8VFlQRT46OmFycmF5KCkgY29uc3QgewotICAgIHJldHVybiBzdGF0aWNfY2FzdDxjb25zdCBUWVBFICo+KGFycmF5SW1wbCgpKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1UWVBFKiBWZWN0b3I8VFlQRT46OmVkaXRBcnJheSgpIHsKLSAgICByZXR1cm4gc3RhdGljX2Nhc3Q8VFlQRSAqPihlZGl0QXJyYXlJbXBsKCkpOwotfQotCi0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotY29uc3QgVFlQRSYgVmVjdG9yPFRZUEU+OjpvcGVyYXRvcltdKHNpemVfdCBpbmRleCkgY29uc3QgewotICAgIExPR19GQVRBTF9JRiggaW5kZXg+PXNpemUoKSwKLSAgICAgICAgICAgICAgICAgICJpdGVtQXQ6IGluZGV4ICVkIGlzIHBhc3Qgc2l6ZSAlZCIsIChpbnQpaW5kZXgsIChpbnQpc2l6ZSgpICk7Ci0gICAgcmV0dXJuICooYXJyYXkoKSArIGluZGV4KTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1jb25zdCBUWVBFJiBWZWN0b3I8VFlQRT46Oml0ZW1BdChzaXplX3QgaW5kZXgpIGNvbnN0IHsKLSAgICByZXR1cm4gb3BlcmF0b3JbXShpbmRleCk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotY29uc3QgVFlQRSYgVmVjdG9yPFRZUEU+OjptaXJyb3JJdGVtQXQoc3NpemVfdCBpbmRleCkgY29uc3QgewotICAgIExPR19GQVRBTF9JRiggKGluZGV4PjAgPyBpbmRleCA6IC1pbmRleCk+PXNpemUoKSwKLSAgICAgICAgICAgICAgICAgICJtaXJyb3JJdGVtQXQ6IGluZGV4ICVkIGlzIHBhc3Qgc2l6ZSAlZCIsCi0gICAgICAgICAgICAgICAgICAoaW50KWluZGV4LCAoaW50KXNpemUoKSApOwotICAgIHJldHVybiAqKGFycmF5KCkgKyAoKGluZGV4PDApID8gKHNpemUoKS1pbmRleCkgOiBpbmRleCkpOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLWNvbnN0IFRZUEUmIFZlY3RvcjxUWVBFPjo6dG9wKCkgY29uc3QgewotICAgIHJldHVybiAqKGFycmF5KCkgKyBzaXplKCkgLSAxKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1UWVBFJiBWZWN0b3I8VFlQRT46OmVkaXRJdGVtQXQoc2l6ZV90IGluZGV4KSB7Ci0gICAgcmV0dXJuICooIHN0YXRpY19jYXN0PFRZUEUgKj4oZWRpdEl0ZW1Mb2NhdGlvbihpbmRleCkpICk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotVFlQRSYgVmVjdG9yPFRZUEU+OjplZGl0VG9wKCkgewotICAgIHJldHVybiAqKCBzdGF0aWNfY2FzdDxUWVBFICo+KGVkaXRJdGVtTG9jYXRpb24oc2l6ZSgpLTEpKSApOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLXNzaXplX3QgVmVjdG9yPFRZUEU+OjppbnNlcnRWZWN0b3JBdChjb25zdCBWZWN0b3I8VFlQRT4mIHZlY3Rvciwgc2l6ZV90IGluZGV4KSB7Ci0gICAgcmV0dXJuIFZlY3RvckltcGw6Omluc2VydFZlY3RvckF0KHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgVmVjdG9ySW1wbCY+KHZlY3RvciksIGluZGV4KTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zc2l6ZV90IFZlY3RvcjxUWVBFPjo6YXBwZW5kVmVjdG9yKGNvbnN0IFZlY3RvcjxUWVBFPiYgdmVjdG9yKSB7Ci0gICAgcmV0dXJuIFZlY3RvckltcGw6OmFwcGVuZFZlY3RvcihyZWludGVycHJldF9jYXN0PGNvbnN0IFZlY3RvckltcGwmPih2ZWN0b3IpKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zc2l6ZV90IFZlY3RvcjxUWVBFPjo6aW5zZXJ0QXQoY29uc3QgVFlQRSYgaXRlbSwgc2l6ZV90IGluZGV4LCBzaXplX3QgbnVtSXRlbXMpIHsKLSAgICByZXR1cm4gVmVjdG9ySW1wbDo6aW5zZXJ0QXQoJml0ZW0sIGluZGV4LCBudW1JdGVtcyk7Ci19Ci0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+IGlubGluZQotdm9pZCBWZWN0b3I8VFlQRT46OnB1c2goY29uc3QgVFlQRSYgaXRlbSkgewotICAgIHJldHVybiBWZWN0b3JJbXBsOjpwdXNoKCZpdGVtKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zc2l6ZV90IFZlY3RvcjxUWVBFPjo6YWRkKGNvbnN0IFRZUEUmIGl0ZW0pIHsKLSAgICByZXR1cm4gVmVjdG9ySW1wbDo6YWRkKCZpdGVtKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zc2l6ZV90IFZlY3RvcjxUWVBFPjo6cmVwbGFjZUF0KGNvbnN0IFRZUEUmIGl0ZW0sIHNpemVfdCBpbmRleCkgewotICAgIHJldHVybiBWZWN0b3JJbXBsOjpyZXBsYWNlQXQoJml0ZW0sIGluZGV4KTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zc2l6ZV90IFZlY3RvcjxUWVBFPjo6aW5zZXJ0QXQoc2l6ZV90IGluZGV4LCBzaXplX3QgbnVtSXRlbXMpIHsKLSAgICByZXR1cm4gVmVjdG9ySW1wbDo6aW5zZXJ0QXQoaW5kZXgsIG51bUl0ZW1zKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi12b2lkIFZlY3RvcjxUWVBFPjo6cG9wKCkgewotICAgIFZlY3RvckltcGw6OnBvcCgpOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLXZvaWQgVmVjdG9yPFRZUEU+OjpwdXNoKCkgewotICAgIFZlY3RvckltcGw6OnB1c2goKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zc2l6ZV90IFZlY3RvcjxUWVBFPjo6YWRkKCkgewotICAgIHJldHVybiBWZWN0b3JJbXBsOjphZGQoKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zc2l6ZV90IFZlY3RvcjxUWVBFPjo6cmVwbGFjZUF0KHNpemVfdCBpbmRleCkgewotICAgIHJldHVybiBWZWN0b3JJbXBsOjpyZXBsYWNlQXQoaW5kZXgpOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPiBpbmxpbmUKLXNzaXplX3QgVmVjdG9yPFRZUEU+OjpyZW1vdmVJdGVtc0F0KHNpemVfdCBpbmRleCwgc2l6ZV90IGNvdW50KSB7Ci0gICAgcmV0dXJuIFZlY3RvckltcGw6OnJlbW92ZUl0ZW1zQXQoaW5kZXgsIGNvdW50KTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zdGF0dXNfdCBWZWN0b3I8VFlQRT46OnNvcnQoVmVjdG9yPFRZUEU+Ojpjb21wYXJfdCBjbXApIHsKLSAgICByZXR1cm4gVmVjdG9ySW1wbDo6c29ydCgoVmVjdG9ySW1wbDo6Y29tcGFyX3QpY21wKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4gaW5saW5lCi1zdGF0dXNfdCBWZWN0b3I8VFlQRT46OnNvcnQoVmVjdG9yPFRZUEU+Ojpjb21wYXJfcl90IGNtcCwgdm9pZCogc3RhdGUpIHsKLSAgICByZXR1cm4gVmVjdG9ySW1wbDo6c29ydCgoVmVjdG9ySW1wbDo6Y29tcGFyX3JfdCljbXAsIHN0YXRlKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXRlbXBsYXRlPGNsYXNzIFRZUEU+Ci12b2lkIFZlY3RvcjxUWVBFPjo6ZG9fY29uc3RydWN0KHZvaWQqIHN0b3JhZ2UsIHNpemVfdCBudW0pIGNvbnN0IHsKLSAgICBjb25zdHJ1Y3RfdHlwZSggcmVpbnRlcnByZXRfY2FzdDxUWVBFKj4oc3RvcmFnZSksIG51bSApOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPgotdm9pZCBWZWN0b3I8VFlQRT46OmRvX2Rlc3Ryb3kodm9pZCogc3RvcmFnZSwgc2l6ZV90IG51bSkgY29uc3QgewotICAgIGRlc3Ryb3lfdHlwZSggcmVpbnRlcnByZXRfY2FzdDxUWVBFKj4oc3RvcmFnZSksIG51bSApOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPgotdm9pZCBWZWN0b3I8VFlQRT46OmRvX2NvcHkodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3QgewotICAgIGNvcHlfdHlwZSggcmVpbnRlcnByZXRfY2FzdDxUWVBFKj4oZGVzdCksIHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgVFlQRSo+KGZyb20pLCBudW0gKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4KLXZvaWQgVmVjdG9yPFRZUEU+Ojpkb19zcGxhdCh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBpdGVtLCBzaXplX3QgbnVtKSBjb25zdCB7Ci0gICAgc3BsYXRfdHlwZSggcmVpbnRlcnByZXRfY2FzdDxUWVBFKj4oZGVzdCksIHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgVFlQRSo+KGl0ZW0pLCBudW0gKTsKLX0KLQotdGVtcGxhdGU8Y2xhc3MgVFlQRT4KLXZvaWQgVmVjdG9yPFRZUEU+Ojpkb19tb3ZlX2ZvcndhcmQodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3QgewotICAgIG1vdmVfZm9yd2FyZF90eXBlKCByZWludGVycHJldF9jYXN0PFRZUEUqPihkZXN0KSwgcmVpbnRlcnByZXRfY2FzdDxjb25zdCBUWVBFKj4oZnJvbSksIG51bSApOwotfQotCi10ZW1wbGF0ZTxjbGFzcyBUWVBFPgotdm9pZCBWZWN0b3I8VFlQRT46OmRvX21vdmVfYmFja3dhcmQodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3QgewotICAgIG1vdmVfYmFja3dhcmRfdHlwZSggcmVpbnRlcnByZXRfY2FzdDxUWVBFKj4oZGVzdCksIHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgVFlQRSo+KGZyb20pLCBudW0gKTsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2VuZGlmIC8vIEFORFJPSURfVkVDVE9SX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvVmVjdG9ySW1wbC5oIGIvaW5jbHVkZS91dGlscy9WZWN0b3JJbXBsLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI1MjUyMjkuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9WZWN0b3JJbXBsLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxOTkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9WRUNUT1JfSU1QTF9ICi0jZGVmaW5lIEFORFJPSURfVkVDVE9SX0lNUExfSAotCi0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gTm8gdXNlciBzZXJ2aWNlYWJsZSBwYXJ0cyBpbiBoZXJlLi4uCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vKiEKLSAqIEltcGxlbWVudGF0aW9uIG9mIHRoZSBndXRzIG9mIHRoZSB2ZWN0b3I8PiBjbGFzcwotICogdGhpcyBlbnN1cmVzIGJhY2t3YXJkIGJpbmFyeSBjb21wYXRpYmlsaXR5IGFuZAotICogcmVkdWNlcyBjb2RlIHNpemUuCi0gKiBGb3IgcGVyZm9ybWFuY2UgcmVhc29ucywgd2UgZXhwb3NlIG1TdG9yYWdlIGFuZCBtQ291bnQKLSAqIHNvIHRoZXNlIGZpZWxkcyBhcmUgc2V0IGluIHN0b25lLgotICoKLSAqLwotCi1jbGFzcyBWZWN0b3JJbXBsCi17Ci1wdWJsaWM6Ci0gICAgZW51bSB7IC8vIGZsYWdzIHBhc3NlZCB0byB0aGUgY3RvcgotICAgICAgICBIQVNfVFJJVklBTF9DVE9SICAgID0gMHgwMDAwMDAwMSwKLSAgICAgICAgSEFTX1RSSVZJQUxfRFRPUiAgICA9IDB4MDAwMDAwMDIsCi0gICAgICAgIEhBU19UUklWSUFMX0NPUFkgICAgPSAweDAwMDAwMDA0LAotICAgICAgICBIQVNfVFJJVklBTF9BU1NJR04gID0gMHgwMDAwMDAwOAotICAgIH07Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBWZWN0b3JJbXBsKHNpemVfdCBpdGVtU2l6ZSwgdWludDMyX3QgZmxhZ3MpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIFZlY3RvckltcGwoY29uc3QgVmVjdG9ySW1wbCYgcmhzKTsKLSAgICB2aXJ0dWFsICAgICAgICAgICAgICAgICB+VmVjdG9ySW1wbCgpOwotCi0gICAgLyohIG11c3QgYmUgY2FsbGVkIGZyb20gc3ViY2xhc3NlcyBkZXN0cnVjdG9yICovCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgZmluaXNoX3ZlY3RvcigpOwotCi0gICAgICAgICAgICBWZWN0b3JJbXBsJiAgICAgb3BlcmF0b3IgPSAoY29uc3QgVmVjdG9ySW1wbCYgcmhzKTsgICAgCi0gICAgICAgICAgICAKLSAgICAvKiEgQy1zdHlsZSBhcnJheSBhY2Nlc3MgKi8KLSAgICBpbmxpbmUgIGNvbnN0IHZvaWQqICAgICBhcnJheUltcGwoKSBjb25zdCAgICAgICB7IHJldHVybiBtU3RvcmFnZTsgfQotICAgICAgICAgICAgdm9pZCogICAgICAgICAgIGVkaXRBcnJheUltcGwoKTsKLSAgICAgICAgICAgIAotICAgIC8qISB2ZWN0b3Igc3RhdHMgKi8KLSAgICBpbmxpbmUgIHNpemVfdCAgICAgICAgICBzaXplKCkgY29uc3QgICAgICAgIHsgcmV0dXJuIG1Db3VudDsgfQotICAgIGlubGluZSAgYm9vbCAgICAgICAgICAgIGlzRW1wdHkoKSBjb25zdCAgICAgeyByZXR1cm4gbUNvdW50ID09IDA7IH0KLSAgICAgICAgICAgIHNpemVfdCAgICAgICAgICBjYXBhY2l0eSgpIGNvbnN0OwotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIHNldENhcGFjaXR5KHNpemVfdCBzaXplKTsKLQotICAgICAgICAgICAgLyohIGFwcGVuZC9pbnNlcnQgYW5vdGhlciB2ZWN0b3IgKi8KLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICBpbnNlcnRWZWN0b3JBdChjb25zdCBWZWN0b3JJbXBsJiB2ZWN0b3IsIHNpemVfdCBpbmRleCk7Ci0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgYXBwZW5kVmVjdG9yKGNvbnN0IFZlY3RvckltcGwmIHZlY3Rvcik7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8qISBhZGQvaW5zZXJ0L3JlcGxhY2UgaXRlbXMgKi8KLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICBpbnNlcnRBdChzaXplX3Qgd2hlcmUsIHNpemVfdCBudW1JdGVtcyA9IDEpOwotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIGluc2VydEF0KGNvbnN0IHZvaWQqIGl0ZW0sIHNpemVfdCB3aGVyZSwgc2l6ZV90IG51bUl0ZW1zID0gMSk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgcG9wKCk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgcHVzaCgpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgIHB1c2goY29uc3Qgdm9pZCogaXRlbSk7Ci0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgYWRkKCk7Ci0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgYWRkKGNvbnN0IHZvaWQqIGl0ZW0pOwotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIHJlcGxhY2VBdChzaXplX3QgaW5kZXgpOwotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIHJlcGxhY2VBdChjb25zdCB2b2lkKiBpdGVtLCBzaXplX3QgaW5kZXgpOwotCi0gICAgICAgICAgICAvKiEgcmVtb3ZlIGl0ZW1zICovCi0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgcmVtb3ZlSXRlbXNBdChzaXplX3QgaW5kZXgsIHNpemVfdCBjb3VudCA9IDEpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgIGNsZWFyKCk7Ci0KLSAgICAgICAgICAgIGNvbnN0IHZvaWQqICAgICBpdGVtTG9jYXRpb24oc2l6ZV90IGluZGV4KSBjb25zdDsKLSAgICAgICAgICAgIHZvaWQqICAgICAgICAgICBlZGl0SXRlbUxvY2F0aW9uKHNpemVfdCBpbmRleCk7Ci0KLSAgICAgICAgICAgIHR5cGVkZWYgaW50ICgqY29tcGFyX3QpKGNvbnN0IHZvaWQqIGxocywgY29uc3Qgdm9pZCogcmhzKTsKLSAgICAgICAgICAgIHR5cGVkZWYgaW50ICgqY29tcGFyX3JfdCkoY29uc3Qgdm9pZCogbGhzLCBjb25zdCB2b2lkKiByaHMsIHZvaWQqIHN0YXRlKTsKLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICBzb3J0KGNvbXBhcl90IGNtcCk7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgc29ydChjb21wYXJfcl90IGNtcCwgdm9pZCogc3RhdGUpOwotCi1wcm90ZWN0ZWQ6Ci0gICAgICAgICAgICBzaXplX3QgICAgICAgICAgaXRlbVNpemUoKSBjb25zdDsKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICByZWxlYXNlX3N0b3JhZ2UoKTsKLQotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIGRvX2NvbnN0cnVjdCh2b2lkKiBzdG9yYWdlLCBzaXplX3QgbnVtKSBjb25zdCA9IDA7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgZG9fZGVzdHJveSh2b2lkKiBzdG9yYWdlLCBzaXplX3QgbnVtKSBjb25zdCA9IDA7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgZG9fY29weSh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdCA9IDA7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgZG9fc3BsYXQodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogaXRlbSwgc2l6ZV90IG51bSkgY29uc3QgPSAwOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIGRvX21vdmVfZm9yd2FyZCh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdCA9IDA7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgZG9fbW92ZV9iYWNrd2FyZCh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdCA9IDA7Ci0KLSAgICAvLyB0YWtlIGNhcmUgb2YgRkJDLi4uCi0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgcmVzZXJ2ZWRWZWN0b3JJbXBsMSgpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHJlc2VydmVkVmVjdG9ySW1wbDIoKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICByZXNlcnZlZFZlY3RvckltcGwzKCk7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgcmVzZXJ2ZWRWZWN0b3JJbXBsNCgpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHJlc2VydmVkVmVjdG9ySW1wbDUoKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICByZXNlcnZlZFZlY3RvckltcGw2KCk7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgcmVzZXJ2ZWRWZWN0b3JJbXBsNygpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHJlc2VydmVkVmVjdG9ySW1wbDgoKTsKLSAgICAKLXByaXZhdGU6Ci0gICAgICAgIHZvaWQqIF9ncm93KHNpemVfdCB3aGVyZSwgc2l6ZV90IGFtb3VudCk7Ci0gICAgICAgIHZvaWQgIF9zaHJpbmsoc2l6ZV90IHdoZXJlLCBzaXplX3QgYW1vdW50KTsKLQotICAgICAgICBpbmxpbmUgdm9pZCBfZG9fY29uc3RydWN0KHZvaWQqIHN0b3JhZ2UsIHNpemVfdCBudW0pIGNvbnN0OwotICAgICAgICBpbmxpbmUgdm9pZCBfZG9fZGVzdHJveSh2b2lkKiBzdG9yYWdlLCBzaXplX3QgbnVtKSBjb25zdDsKLSAgICAgICAgaW5saW5lIHZvaWQgX2RvX2NvcHkodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3Q7Ci0gICAgICAgIGlubGluZSB2b2lkIF9kb19zcGxhdCh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBpdGVtLCBzaXplX3QgbnVtKSBjb25zdDsKLSAgICAgICAgaW5saW5lIHZvaWQgX2RvX21vdmVfZm9yd2FyZCh2b2lkKiBkZXN0LCBjb25zdCB2b2lkKiBmcm9tLCBzaXplX3QgbnVtKSBjb25zdDsKLSAgICAgICAgaW5saW5lIHZvaWQgX2RvX21vdmVfYmFja3dhcmQodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3Q7Ci0KLSAgICAgICAgICAgIC8vIFRoZXNlIDIgZmllbGRzIGFyZSBleHBvc2VkIGluIHRoZSBpbmxpbmVzIGJlbG93LAotICAgICAgICAgICAgLy8gc28gdGhleSdyZSBzZXQgaW4gc3RvbmUuCi0gICAgICAgICAgICB2b2lkICogICAgICBtU3RvcmFnZTsgICAvLyBiYXNlIGFkZHJlc3Mgb2YgdGhlIHZlY3RvcgotICAgICAgICAgICAgc2l6ZV90ICAgICAgbUNvdW50OyAgICAgLy8gbnVtYmVyIG9mIGl0ZW1zCi0KLSAgICBjb25zdCAgIHVpbnQzMl90ICAgIG1GbGFnczsKLSAgICBjb25zdCAgIHNpemVfdCAgICAgIG1JdGVtU2l6ZTsKLX07Ci0KLQotCi1jbGFzcyBTb3J0ZWRWZWN0b3JJbXBsIDogcHVibGljIFZlY3RvckltcGwKLXsKLXB1YmxpYzoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBTb3J0ZWRWZWN0b3JJbXBsKHNpemVfdCBpdGVtU2l6ZSwgdWludDMyX3QgZmxhZ3MpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIFNvcnRlZFZlY3RvckltcGwoY29uc3QgVmVjdG9ySW1wbCYgcmhzKTsKLSAgICB2aXJ0dWFsICAgICAgICAgICAgICAgICB+U29ydGVkVmVjdG9ySW1wbCgpOwotICAgIAotICAgIFNvcnRlZFZlY3RvckltcGwmICAgICBvcGVyYXRvciA9IChjb25zdCBTb3J0ZWRWZWN0b3JJbXBsJiByaHMpOyAgICAKLQotICAgIC8vISBmaW5kcyB0aGUgaW5kZXggb2YgYW4gaXRlbQotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIGluZGV4T2YoY29uc3Qgdm9pZCogaXRlbSkgY29uc3Q7Ci0KLSAgICAvLyEgZmluZHMgd2hlcmUgdGhpcyBpdGVtIHNob3VsZCBiZSBpbnNlcnRlZAotICAgICAgICAgICAgc2l6ZV90ICAgICAgICAgIG9yZGVyT2YoY29uc3Qgdm9pZCogaXRlbSkgY29uc3Q7Ci0KLSAgICAvLyEgYWRkIGFuIGl0ZW0gaW4gdGhlIHJpZ2h0IHBsYWNlIChvciByZXBsYWNlcyBpdCBpZiB0aGVyZSBpcyBvbmUpCi0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgYWRkKGNvbnN0IHZvaWQqIGl0ZW0pOwotCi0gICAgLy8hIG1lcmdlcyBhIHZlY3RvciBpbnRvIHRoaXMgb25lCi0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgbWVyZ2UoY29uc3QgVmVjdG9ySW1wbCYgdmVjdG9yKTsKLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICBtZXJnZShjb25zdCBTb3J0ZWRWZWN0b3JJbXBsJiB2ZWN0b3IpOwotICAgICAgICAgICAgIAotICAgIC8vISByZW1vdmVzIGFuIGl0ZW0KLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICByZW1vdmUoY29uc3Qgdm9pZCogaXRlbSk7Ci0gICAgICAgIAotcHJvdGVjdGVkOgotICAgIHZpcnR1YWwgaW50ICAgICAgICAgICAgIGRvX2NvbXBhcmUoY29uc3Qgdm9pZCogbGhzLCBjb25zdCB2b2lkKiByaHMpIGNvbnN0ID0gMDsKLQotICAgIC8vIHRha2UgY2FyZSBvZiBGQkMuLi4KLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICByZXNlcnZlZFNvcnRlZFZlY3RvckltcGwxKCk7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgcmVzZXJ2ZWRTb3J0ZWRWZWN0b3JJbXBsMigpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDMoKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICByZXNlcnZlZFNvcnRlZFZlY3RvckltcGw0KCk7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgcmVzZXJ2ZWRTb3J0ZWRWZWN0b3JJbXBsNSgpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgIHJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDYoKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgICAgICByZXNlcnZlZFNvcnRlZFZlY3RvckltcGw3KCk7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgcmVzZXJ2ZWRTb3J0ZWRWZWN0b3JJbXBsOCgpOwotCi1wcml2YXRlOgotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIF9pbmRleE9yZGVyT2YoY29uc3Qgdm9pZCogaXRlbSwgc2l6ZV90KiBvcmRlciA9IDApIGNvbnN0OwotCi0gICAgICAgICAgICAvLyB0aGVzZSBhcmUgbWFkZSBwcml2YXRlLCBiZWNhdXNlIHRoZXkgY2FuJ3QgYmUgdXNlZCBvbiBhIFNvcnRlZFZlY3RvcgotICAgICAgICAgICAgLy8gKHRoZXkgZG9uJ3QgaGF2ZSBhbiBpbXBsZW1lbnRhdGlvbiBlaXRoZXIpCi0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgYWRkKCk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgcG9wKCk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgcHVzaCgpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgIHB1c2goY29uc3Qgdm9pZCogaXRlbSk7Ci0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgaW5zZXJ0VmVjdG9yQXQoY29uc3QgVmVjdG9ySW1wbCYgdmVjdG9yLCBzaXplX3QgaW5kZXgpOwotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIGFwcGVuZFZlY3Rvcihjb25zdCBWZWN0b3JJbXBsJiB2ZWN0b3IpOwotICAgICAgICAgICAgc3NpemVfdCAgICAgICAgIGluc2VydEF0KHNpemVfdCB3aGVyZSwgc2l6ZV90IG51bUl0ZW1zID0gMSk7Ci0gICAgICAgICAgICBzc2l6ZV90ICAgICAgICAgaW5zZXJ0QXQoY29uc3Qgdm9pZCogaXRlbSwgc2l6ZV90IHdoZXJlLCBzaXplX3QgbnVtSXRlbXMgPSAxKTsKLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICByZXBsYWNlQXQoc2l6ZV90IGluZGV4KTsKLSAgICAgICAgICAgIHNzaXplX3QgICAgICAgICByZXBsYWNlQXQoY29uc3Qgdm9pZCogaXRlbSwgc2l6ZV90IGluZGV4KTsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNlbmRpZiAvLyBBTkRST0lEX1ZFQ1RPUl9JTVBMX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvWmlwRW50cnkuaCBiL2luY2x1ZGUvdXRpbHMvWmlwRW50cnkuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZTQ2OThkZi4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL1ppcEVudHJ5LmgKKysrIC9kZXYvbnVsbApAQCAtMSwzNDUgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBaaXAgYXJjaGl2ZSBlbnRyaWVzLgotLy8KLS8vIFRoZSBaaXBFbnRyeSBjbGFzcyBpcyB0aWdodGx5IG1lc2hlZCB3aXRoIHRoZSBaaXBGaWxlIGNsYXNzLgotLy8KLSNpZm5kZWYgX19MSUJTX1pJUEVOVFJZX0gKLSNkZWZpbmUgX19MSUJTX1pJUEVOVFJZX0gKLQotI2luY2x1ZGUgIkVycm9ycy5oIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBaaXBGaWxlOwotCi0vKgotICogWmlwRW50cnkgb2JqZWN0cyByZXByZXNlbnQgYSBzaW5nbGUgZW50cnkgaW4gYSBaaXAgYXJjaGl2ZS4KLSAqCi0gKiBZb3UgY2FuIHVzZSBvbmUgb2YgdGhlc2UgdG8gZ2V0IG9yIHNldCBpbmZvcm1hdGlvbiBhYm91dCBhbiBlbnRyeSwgYnV0Ci0gKiB0aGVyZSBhcmUgbm8gZnVuY3Rpb25zIGhlcmUgZm9yIGFjY2Vzc2luZyB0aGUgZGF0YSBpdHNlbGYuICAoV2UgY291bGQKLSAqIHR1Y2sgYSBwb2ludGVyIHRvIHRoZSBaaXBGaWxlIGluIGhlcmUgZm9yIGNvbnZlbmllbmNlLCBidXQgdGhhdCByYWlzZXMKLSAqIHRoZSBsaWtlbGlob29kIG9mIHVzaW5nIFppcEVudHJ5IG9iamVjdHMgYWZ0ZXIgZGlzY2FyZGluZyB0aGUgWmlwRmlsZS4pCi0gKgotICogRmlsZSBpbmZvcm1hdGlvbiBpcyBzdG9yZWQgaW4gdHdvIHBsYWNlczogbmV4dCB0byB0aGUgZmlsZSBkYXRhICh0aGUgTG9jYWwKLSAqIEZpbGUgSGVhZGVyLCBhbmQgcG9zc2libHkgYSBEYXRhIERlc2NyaXB0b3IpLCBhbmQgYXQgdGhlIGVuZCBvZiB0aGUgZmlsZQotICogKHRoZSBDZW50cmFsIERpcmVjdG9yeSBFbnRyeSkuICBUaGUgdHdvIG11c3QgYmUga2VwdCBpbiBzeW5jLgotICovCi1jbGFzcyBaaXBFbnRyeSB7Ci1wdWJsaWM6Ci0gICAgZnJpZW5kIGNsYXNzIFppcEZpbGU7Ci0KLSAgICBaaXBFbnRyeSh2b2lkKQotICAgICAgICA6IG1EZWxldGVkKGZhbHNlKSwgbU1hcmtlZChmYWxzZSkKLSAgICAgICAge30KLSAgICB+WmlwRW50cnkodm9pZCkge30KLQotICAgIC8qCi0gICAgICogUmV0dXJucyAidHJ1ZSIgaWYgdGhlIGRhdGEgaXMgY29tcHJlc3NlZC4KLSAgICAgKi8KLSAgICBib29sIGlzQ29tcHJlc3NlZCh2b2lkKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiBtQ0RFLm1Db21wcmVzc2lvbk1ldGhvZCAhPSBrQ29tcHJlc3NTdG9yZWQ7Ci0gICAgfQotICAgIGludCBnZXRDb21wcmVzc2lvbk1ldGhvZCh2b2lkKSBjb25zdCB7IHJldHVybiBtQ0RFLm1Db21wcmVzc2lvbk1ldGhvZDsgfQotCi0gICAgLyoKLSAgICAgKiBSZXR1cm4gdGhlIHVuY29tcHJlc3NlZCBsZW5ndGguCi0gICAgICovCi0gICAgb2ZmX3QgZ2V0VW5jb21wcmVzc2VkTGVuKHZvaWQpIGNvbnN0IHsgcmV0dXJuIG1DREUubVVuY29tcHJlc3NlZFNpemU7IH0KLQotICAgIC8qCi0gICAgICogUmV0dXJuIHRoZSBjb21wcmVzc2VkIGxlbmd0aC4gIEZvciB1bmNvbXByZXNzZWQgZGF0YSwgdGhpcyByZXR1cm5zCi0gICAgICogdGhlIHNhbWUgdGhpbmcgYXMgZ2V0VW5jb21wcmVzZXNkTGVuKCkuCi0gICAgICovCi0gICAgb2ZmX3QgZ2V0Q29tcHJlc3NlZExlbih2b2lkKSBjb25zdCB7IHJldHVybiBtQ0RFLm1Db21wcmVzc2VkU2l6ZTsgfQotCi0gICAgLyoKLSAgICAgKiBSZXR1cm4gdGhlIGFic29sdXRlIGZpbGUgb2Zmc2V0IG9mIHRoZSBzdGFydCBvZiB0aGUgY29tcHJlc3NlZCBvcgotICAgICAqIHVuY29tcHJlc3NlZCBkYXRhLgotICAgICAqLwotICAgIG9mZl90IGdldEZpbGVPZmZzZXQodm9pZCkgY29uc3QgewotICAgICAgICByZXR1cm4gbUNERS5tTG9jYWxIZWFkZXJSZWxPZmZzZXQgKwotICAgICAgICAgICAgICAgIExvY2FsRmlsZUhlYWRlcjo6a0xGSExlbiArCi0gICAgICAgICAgICAgICAgbUxGSC5tRmlsZU5hbWVMZW5ndGggKwotICAgICAgICAgICAgICAgIG1MRkgubUV4dHJhRmllbGRMZW5ndGg7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBSZXR1cm4gdGhlIGRhdGEgQ1JDLgotICAgICAqLwotICAgIHVuc2lnbmVkIGxvbmcgZ2V0Q1JDMzIodm9pZCkgY29uc3QgeyByZXR1cm4gbUNERS5tQ1JDMzI7IH0KLQotICAgIC8qCi0gICAgICogUmV0dXJuIGZpbGUgbW9kaWZpY2F0aW9uIHRpbWUgaW4gVU5JWCBzZWNvbmRzLXNpbmNlLWVwb2NoLgotICAgICAqLwotICAgIHRpbWVfdCBnZXRNb2RXaGVuKHZvaWQpIGNvbnN0OwotCi0gICAgLyoKLSAgICAgKiBSZXR1cm4gdGhlIGFyY2hpdmVkIGZpbGUgbmFtZS4KLSAgICAgKi8KLSAgICBjb25zdCBjaGFyKiBnZXRGaWxlTmFtZSh2b2lkKSBjb25zdCB7IHJldHVybiAoY29uc3QgY2hhciopIG1DREUubUZpbGVOYW1lOyB9Ci0KLSAgICAvKgotICAgICAqIEFwcGxpY2F0aW9uLWRlZmluZWQgIm1hcmsiLiAgQ2FuIGJlIHVzZWZ1bCB3aGVuIHN5bmNocm9uaXppbmcgdGhlCi0gICAgICogY29udGVudHMgb2YgYW4gYXJjaGl2ZSB3aXRoIGNvbnRlbnRzIG9uIGRpc2suCi0gICAgICovCi0gICAgYm9vbCBnZXRNYXJrZWQodm9pZCkgY29uc3QgeyByZXR1cm4gbU1hcmtlZDsgfQotICAgIHZvaWQgc2V0TWFya2VkKGJvb2wgdmFsKSB7IG1NYXJrZWQgPSB2YWw7IH0KLQotICAgIC8qCi0gICAgICogU29tZSBiYXNpYyBmdW5jdGlvbnMgZm9yIHJhdyBkYXRhIG1hbmlwdWxhdGlvbi4gICJMRSIgbWVhbnMKLSAgICAgKiBMaXR0bGUgRW5kaWFuLgotICAgICAqLwotICAgIHN0YXRpYyBpbmxpbmUgdW5zaWduZWQgc2hvcnQgZ2V0U2hvcnRMRShjb25zdCB1bnNpZ25lZCBjaGFyKiBidWYpIHsKLSAgICAgICAgcmV0dXJuIGJ1ZlswXSB8IChidWZbMV0gPDwgOCk7Ci0gICAgfQotICAgIHN0YXRpYyBpbmxpbmUgdW5zaWduZWQgbG9uZyBnZXRMb25nTEUoY29uc3QgdW5zaWduZWQgY2hhciogYnVmKSB7Ci0gICAgICAgIHJldHVybiBidWZbMF0gfCAoYnVmWzFdIDw8IDgpIHwgKGJ1ZlsyXSA8PCAxNikgfCAoYnVmWzNdIDw8IDI0KTsKLSAgICB9Ci0gICAgc3RhdGljIGlubGluZSB2b2lkIHB1dFNob3J0TEUodW5zaWduZWQgY2hhciogYnVmLCBzaG9ydCB2YWwpIHsKLSAgICAgICAgYnVmWzBdID0gKHVuc2lnbmVkIGNoYXIpIHZhbDsKLSAgICAgICAgYnVmWzFdID0gKHVuc2lnbmVkIGNoYXIpICh2YWwgPj4gOCk7Ci0gICAgfQotICAgIHN0YXRpYyBpbmxpbmUgdm9pZCBwdXRMb25nTEUodW5zaWduZWQgY2hhciogYnVmLCBsb25nIHZhbCkgewotICAgICAgICBidWZbMF0gPSAodW5zaWduZWQgY2hhcikgdmFsOwotICAgICAgICBidWZbMV0gPSAodW5zaWduZWQgY2hhcikgKHZhbCA+PiA4KTsKLSAgICAgICAgYnVmWzJdID0gKHVuc2lnbmVkIGNoYXIpICh2YWwgPj4gMTYpOwotICAgICAgICBidWZbM10gPSAodW5zaWduZWQgY2hhcikgKHZhbCA+PiAyNCk7Ci0gICAgfQotCi0gICAgLyogZGVmaW5lZCBmb3IgWmlwIGFyY2hpdmVzICovCi0gICAgZW51bSB7Ci0gICAgICAgIGtDb21wcmVzc1N0b3JlZCAgICAgPSAwLCAgICAgICAgLy8gbm8gY29tcHJlc3Npb24KLSAgICAgICAgLy8gc2hydW5rICAgICAgICAgICA9IDEsCi0gICAgICAgIC8vIHJlZHVjZWQgMSAgICAgICAgPSAyLAotICAgICAgICAvLyByZWR1Y2VkIDIgICAgICAgID0gMywKLSAgICAgICAgLy8gcmVkdWNlZCAzICAgICAgICA9IDQsCi0gICAgICAgIC8vIHJlZHVjZWQgNCAgICAgICAgPSA1LAotICAgICAgICAvLyBpbXBsb2RlZCAgICAgICAgID0gNiwKLSAgICAgICAgLy8gdG9rZW5pemVkICAgICAgICA9IDcsCi0gICAgICAgIGtDb21wcmVzc0RlZmxhdGVkICAgPSA4LCAgICAgICAgLy8gc3RhbmRhcmQgZGVmbGF0ZQotICAgICAgICAvLyBEZWZsYXRlNjQgICAgICAgID0gOSwKLSAgICAgICAgLy8gbGliIGltcGxvZGVkICAgICA9IDEwLAotICAgICAgICAvLyByZXNlcnZlZCAgICAgICAgID0gMTEsCi0gICAgICAgIC8vIGJ6aXAyICAgICAgICAgICAgPSAxMiwKLSAgICB9OwotCi0gICAgLyoKLSAgICAgKiBEZWxldGlvbiBmbGFnLiAgSWYgc2V0LCB0aGUgZW50cnkgd2lsbCBiZSByZW1vdmVkIG9uIHRoZSBuZXh0Ci0gICAgICogY2FsbCB0byAiZmx1c2giLgotICAgICAqLwotICAgIGJvb2wgZ2V0RGVsZXRlZCh2b2lkKSBjb25zdCB7IHJldHVybiBtRGVsZXRlZDsgfQotCi1wcm90ZWN0ZWQ6Ci0gICAgLyoKLSAgICAgKiBJbml0aWFsaXplIHRoZSBzdHJ1Y3R1cmUgZnJvbSB0aGUgZmlsZSwgd2hpY2ggaXMgcG9pbnRpbmcgYXQKLSAgICAgKiBvdXIgQ2VudHJhbCBEaXJlY3RvcnkgZW50cnkuCi0gICAgICovCi0gICAgc3RhdHVzX3QgaW5pdEZyb21DREUoRklMRSogZnApOwotCi0gICAgLyoKLSAgICAgKiBJbml0aWFsaXplIHRoZSBzdHJ1Y3R1cmUgZm9yIGEgbmV3IGZpbGUuICBXZSBuZWVkIHRoZSBmaWxlbmFtZQotICAgICAqIGFuZCBjb21tZW50IHNvIHRoYXQgd2UgY2FuIHByb3Blcmx5IHNpemUgdGhlIExGSCBhcmVhLiAgVGhlCi0gICAgICogZmlsZW5hbWUgaXMgbWFuZGF0b3J5LCB0aGUgY29tbWVudCBpcyBvcHRpb25hbC4KLSAgICAgKi8KLSAgICB2b2lkIGluaXROZXcoY29uc3QgY2hhciogZmlsZU5hbWUsIGNvbnN0IGNoYXIqIGNvbW1lbnQpOwotCi0gICAgLyoKLSAgICAgKiBJbml0aWFsaXplIHRoZSBzdHJ1Y3R1cmUgd2l0aCB0aGUgY29udGVudHMgb2YgYSBaaXBFbnRyeSBmcm9tCi0gICAgICogYW5vdGhlciBmaWxlLgotICAgICAqLwotICAgIHN0YXR1c190IGluaXRGcm9tRXh0ZXJuYWwoY29uc3QgWmlwRmlsZSogcFppcEZpbGUsIGNvbnN0IFppcEVudHJ5KiBwRW50cnkpOwotCi0gICAgLyoKLSAgICAgKiBBZGQgc29tZSBwYWQgYnl0ZXMgdG8gdGhlIExGSC4gIFdlIGRvIHRoaXMgYnkgYWRkaW5nIG9yIHJlc2l6aW5nCi0gICAgICogdGhlICJleHRyYSIgZmllbGQuCi0gICAgICovCi0gICAgc3RhdHVzX3QgYWRkUGFkZGluZyhpbnQgcGFkZGluZyk7Ci0KLSAgICAvKgotICAgICAqIFNldCBpbmZvcm1hdGlvbiBhYm91dCB0aGUgZGF0YSBmb3IgdGhpcyBlbnRyeS4KLSAgICAgKi8KLSAgICB2b2lkIHNldERhdGFJbmZvKGxvbmcgdW5jb21wTGVuLCBsb25nIGNvbXBMZW4sIHVuc2lnbmVkIGxvbmcgY3JjMzIsCi0gICAgICAgIGludCBjb21wcmVzc2lvbk1ldGhvZCk7Ci0KLSAgICAvKgotICAgICAqIFNldCB0aGUgbW9kaWZpY2F0aW9uIGRhdGUuCi0gICAgICovCi0gICAgdm9pZCBzZXRNb2RXaGVuKHRpbWVfdCB3aGVuKTsKLQotICAgIC8qCi0gICAgICogUmV0dXJuIHRoZSBvZmZzZXQgb2YgdGhlIGxvY2FsIGZpbGUgaGVhZGVyLgotICAgICAqLwotICAgIG9mZl90IGdldExGSE9mZnNldCh2b2lkKSBjb25zdCB7IHJldHVybiBtQ0RFLm1Mb2NhbEhlYWRlclJlbE9mZnNldDsgfQotCi0gICAgLyoKLSAgICAgKiBTZXQgdGhlIG9mZnNldCBvZiB0aGUgbG9jYWwgZmlsZSBoZWFkZXIsIHJlbGF0aXZlIHRvIHRoZSBzdGFydCBvZgotICAgICAqIHRoZSBjdXJyZW50IGZpbGUuCi0gICAgICovCi0gICAgdm9pZCBzZXRMRkhPZmZzZXQob2ZmX3Qgb2Zmc2V0KSB7Ci0gICAgICAgIG1DREUubUxvY2FsSGVhZGVyUmVsT2Zmc2V0ID0gKGxvbmcpIG9mZnNldDsKLSAgICB9Ci0KLSAgICAvKiBtYXJrIGZvciBkZWxldGlvbjsgdXNlZCBieSBaaXBGaWxlOjpyZW1vdmUoKSAqLwotICAgIHZvaWQgc2V0RGVsZXRlZCh2b2lkKSB7IG1EZWxldGVkID0gdHJ1ZTsgfQotCi1wcml2YXRlOgotICAgIC8qIHRoZXNlIGFyZSBwcml2YXRlIGFuZCBub3QgZGVmaW5lZCAqLwotICAgIFppcEVudHJ5KGNvbnN0IFppcEVudHJ5JiBzcmMpOwotICAgIFppcEVudHJ5JiBvcGVyYXRvcj0oY29uc3QgWmlwRW50cnkmIHNyYyk7Ci0KLSAgICAvKiByZXR1cm5zICJ0cnVlIiBpZiB0aGUgQ0RFIGFuZCB0aGUgTEZIIGFncmVlICovCi0gICAgYm9vbCBjb21wYXJlSGVhZGVycyh2b2lkKSBjb25zdDsKLSAgICB2b2lkIGNvcHlDREV0b0xGSCh2b2lkKTsKLQotICAgIGJvb2wgICAgICAgIG1EZWxldGVkOyAgICAgICAvLyBzZXQgaWYgZW50cnkgaXMgcGVuZGluZyBkZWxldGlvbgotICAgIGJvb2wgICAgICAgIG1NYXJrZWQ7ICAgICAgICAvLyBhcHAtZGVmaW5lZCBtYXJrZXIKLQotICAgIC8qCi0gICAgICogRXZlcnkgZW50cnkgaW4gdGhlIFppcCBhcmNoaXZlIHN0YXJ0cyBvZmYgd2l0aCBvbmUgb2YgdGhlc2UuCi0gICAgICovCi0gICAgY2xhc3MgTG9jYWxGaWxlSGVhZGVyIHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIExvY2FsRmlsZUhlYWRlcih2b2lkKSA6Ci0gICAgICAgICAgICBtVmVyc2lvblRvRXh0cmFjdCgwKSwKLSAgICAgICAgICAgIG1HUEJpdEZsYWcoMCksCi0gICAgICAgICAgICBtQ29tcHJlc3Npb25NZXRob2QoMCksCi0gICAgICAgICAgICBtTGFzdE1vZEZpbGVUaW1lKDApLAotICAgICAgICAgICAgbUxhc3RNb2RGaWxlRGF0ZSgwKSwKLSAgICAgICAgICAgIG1DUkMzMigwKSwKLSAgICAgICAgICAgIG1Db21wcmVzc2VkU2l6ZSgwKSwKLSAgICAgICAgICAgIG1VbmNvbXByZXNzZWRTaXplKDApLAotICAgICAgICAgICAgbUZpbGVOYW1lTGVuZ3RoKDApLAotICAgICAgICAgICAgbUV4dHJhRmllbGRMZW5ndGgoMCksCi0gICAgICAgICAgICBtRmlsZU5hbWUoTlVMTCksCi0gICAgICAgICAgICBtRXh0cmFGaWVsZChOVUxMKQotICAgICAgICB7fQotICAgICAgICB2aXJ0dWFsIH5Mb2NhbEZpbGVIZWFkZXIodm9pZCkgewotICAgICAgICAgICAgZGVsZXRlW10gbUZpbGVOYW1lOwotICAgICAgICAgICAgZGVsZXRlW10gbUV4dHJhRmllbGQ7Ci0gICAgICAgIH0KLQotICAgICAgICBzdGF0dXNfdCByZWFkKEZJTEUqIGZwKTsKLSAgICAgICAgc3RhdHVzX3Qgd3JpdGUoRklMRSogZnApOwotCi0gICAgICAgIC8vIHVuc2lnbmVkIGxvbmcgbVNpZ25hdHVyZTsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1WZXJzaW9uVG9FeHRyYWN0OwotICAgICAgICB1bnNpZ25lZCBzaG9ydCAgbUdQQml0RmxhZzsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1Db21wcmVzc2lvbk1ldGhvZDsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1MYXN0TW9kRmlsZVRpbWU7Ci0gICAgICAgIHVuc2lnbmVkIHNob3J0ICBtTGFzdE1vZEZpbGVEYXRlOwotICAgICAgICB1bnNpZ25lZCBsb25nICAgbUNSQzMyOwotICAgICAgICB1bnNpZ25lZCBsb25nICAgbUNvbXByZXNzZWRTaXplOwotICAgICAgICB1bnNpZ25lZCBsb25nICAgbVVuY29tcHJlc3NlZFNpemU7Ci0gICAgICAgIHVuc2lnbmVkIHNob3J0ICBtRmlsZU5hbWVMZW5ndGg7Ci0gICAgICAgIHVuc2lnbmVkIHNob3J0ICBtRXh0cmFGaWVsZExlbmd0aDsKLSAgICAgICAgdW5zaWduZWQgY2hhciogIG1GaWxlTmFtZTsKLSAgICAgICAgdW5zaWduZWQgY2hhciogIG1FeHRyYUZpZWxkOwotCi0gICAgICAgIGVudW0gewotICAgICAgICAgICAga1NpZ25hdHVyZSAgICAgID0gMHgwNDAzNGI1MCwKLSAgICAgICAgICAgIGtMRkhMZW4gICAgICAgICA9IDMwLCAgICAgICAvLyBMb2NhbEZpbGVIZHIgbGVuLCBleGNsLiB2YXIgZmllbGRzCi0gICAgICAgIH07Ci0KLSAgICAgICAgdm9pZCBkdW1wKHZvaWQpIGNvbnN0OwotICAgIH07Ci0KLSAgICAvKgotICAgICAqIEV2ZXJ5IGVudHJ5IGluIHRoZSBaaXAgYXJjaGl2ZSBoYXMgb25lIG9mIHRoZXNlIGluIHRoZSAiY2VudHJhbAotICAgICAqIGRpcmVjdG9yeSIgYXQgdGhlIGVuZCBvZiB0aGUgZmlsZS4KLSAgICAgKi8KLSAgICBjbGFzcyBDZW50cmFsRGlyRW50cnkgewotICAgIHB1YmxpYzoKLSAgICAgICAgQ2VudHJhbERpckVudHJ5KHZvaWQpIDoKLSAgICAgICAgICAgIG1WZXJzaW9uTWFkZUJ5KDApLAotICAgICAgICAgICAgbVZlcnNpb25Ub0V4dHJhY3QoMCksCi0gICAgICAgICAgICBtR1BCaXRGbGFnKDApLAotICAgICAgICAgICAgbUNvbXByZXNzaW9uTWV0aG9kKDApLAotICAgICAgICAgICAgbUxhc3RNb2RGaWxlVGltZSgwKSwKLSAgICAgICAgICAgIG1MYXN0TW9kRmlsZURhdGUoMCksCi0gICAgICAgICAgICBtQ1JDMzIoMCksCi0gICAgICAgICAgICBtQ29tcHJlc3NlZFNpemUoMCksCi0gICAgICAgICAgICBtVW5jb21wcmVzc2VkU2l6ZSgwKSwKLSAgICAgICAgICAgIG1GaWxlTmFtZUxlbmd0aCgwKSwKLSAgICAgICAgICAgIG1FeHRyYUZpZWxkTGVuZ3RoKDApLAotICAgICAgICAgICAgbUZpbGVDb21tZW50TGVuZ3RoKDApLAotICAgICAgICAgICAgbURpc2tOdW1iZXJTdGFydCgwKSwKLSAgICAgICAgICAgIG1JbnRlcm5hbEF0dHJzKDApLAotICAgICAgICAgICAgbUV4dGVybmFsQXR0cnMoMCksCi0gICAgICAgICAgICBtTG9jYWxIZWFkZXJSZWxPZmZzZXQoMCksCi0gICAgICAgICAgICBtRmlsZU5hbWUoTlVMTCksCi0gICAgICAgICAgICBtRXh0cmFGaWVsZChOVUxMKSwKLSAgICAgICAgICAgIG1GaWxlQ29tbWVudChOVUxMKQotICAgICAgICB7fQotICAgICAgICB2aXJ0dWFsIH5DZW50cmFsRGlyRW50cnkodm9pZCkgewotICAgICAgICAgICAgZGVsZXRlW10gbUZpbGVOYW1lOwotICAgICAgICAgICAgZGVsZXRlW10gbUV4dHJhRmllbGQ7Ci0gICAgICAgICAgICBkZWxldGVbXSBtRmlsZUNvbW1lbnQ7Ci0gICAgICAgIH0KLQotICAgICAgICBzdGF0dXNfdCByZWFkKEZJTEUqIGZwKTsKLSAgICAgICAgc3RhdHVzX3Qgd3JpdGUoRklMRSogZnApOwotCi0gICAgICAgIC8vIHVuc2lnbmVkIGxvbmcgbVNpZ25hdHVyZTsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1WZXJzaW9uTWFkZUJ5OwotICAgICAgICB1bnNpZ25lZCBzaG9ydCAgbVZlcnNpb25Ub0V4dHJhY3Q7Ci0gICAgICAgIHVuc2lnbmVkIHNob3J0ICBtR1BCaXRGbGFnOwotICAgICAgICB1bnNpZ25lZCBzaG9ydCAgbUNvbXByZXNzaW9uTWV0aG9kOwotICAgICAgICB1bnNpZ25lZCBzaG9ydCAgbUxhc3RNb2RGaWxlVGltZTsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1MYXN0TW9kRmlsZURhdGU7Ci0gICAgICAgIHVuc2lnbmVkIGxvbmcgICBtQ1JDMzI7Ci0gICAgICAgIHVuc2lnbmVkIGxvbmcgICBtQ29tcHJlc3NlZFNpemU7Ci0gICAgICAgIHVuc2lnbmVkIGxvbmcgICBtVW5jb21wcmVzc2VkU2l6ZTsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1GaWxlTmFtZUxlbmd0aDsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1FeHRyYUZpZWxkTGVuZ3RoOwotICAgICAgICB1bnNpZ25lZCBzaG9ydCAgbUZpbGVDb21tZW50TGVuZ3RoOwotICAgICAgICB1bnNpZ25lZCBzaG9ydCAgbURpc2tOdW1iZXJTdGFydDsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1JbnRlcm5hbEF0dHJzOwotICAgICAgICB1bnNpZ25lZCBsb25nICAgbUV4dGVybmFsQXR0cnM7Ci0gICAgICAgIHVuc2lnbmVkIGxvbmcgICBtTG9jYWxIZWFkZXJSZWxPZmZzZXQ7Ci0gICAgICAgIHVuc2lnbmVkIGNoYXIqICBtRmlsZU5hbWU7Ci0gICAgICAgIHVuc2lnbmVkIGNoYXIqICBtRXh0cmFGaWVsZDsKLSAgICAgICAgdW5zaWduZWQgY2hhciogIG1GaWxlQ29tbWVudDsKLQotICAgICAgICB2b2lkIGR1bXAodm9pZCkgY29uc3Q7Ci0KLSAgICAgICAgZW51bSB7Ci0gICAgICAgICAgICBrU2lnbmF0dXJlICAgICAgPSAweDAyMDE0YjUwLAotICAgICAgICAgICAga0NERUxlbiAgICAgICAgID0gNDYsICAgICAgIC8vIENlbnRyYWxEaXJFbnQgbGVuLCBleGNsLiB2YXIgZmllbGRzCi0gICAgICAgIH07Ci0gICAgfTsKLQotICAgIGVudW0gewotICAgICAgICAvL2tEYXRhRGVzY3JpcHRvclNpZ25hdHVyZSAgPSAweDA4MDc0YjUwLCAgIC8vIGN1cnJlbnRseSB1bnVzZWQKLSAgICAgICAga0RhdGFEZXNjcmlwdG9yTGVuICA9IDE2LCAgICAgICAgICAgLy8gZm91ciAzMi1iaXQgZmllbGRzCi0KLSAgICAgICAga0RlZmF1bHRWZXJzaW9uICAgICA9IDIwLCAgICAgICAgICAgLy8gbmVlZCBkZWZsYXRlLCBub3RoaW5nIG11Y2ggZWxzZQotICAgICAgICBrRGVmYXVsdE1hZGVCeSAgICAgID0gMHgwMzE3LCAgICAgICAvLyAwMz1VTklYLCAxNz1zcGVjIHYyLjMKLSAgICAgICAga1VzZXNEYXRhRGVzY3IgICAgICA9IDB4MDAwOCwgICAgICAgLy8gR1BCaXRGbGFnIGJpdCAzCi0gICAgfTsKLQotICAgIExvY2FsRmlsZUhlYWRlciAgICAgbUxGSDsKLSAgICBDZW50cmFsRGlyRW50cnkgICAgIG1DREU7Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gX19MSUJTX1pJUEVOVFJZX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvWmlwRmlsZS5oIGIvaW5jbHVkZS91dGlscy9aaXBGaWxlLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQ0ZGY1YmIuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9aaXBGaWxlLmgKKysrIC9kZXYvbnVsbApAQCAtMSwyNjkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBHZW5lcmFsLXB1cnBvc2UgWmlwIGFyY2hpdmUgYWNjZXNzLiAgVGhpcyBjbGFzcyBhbGxvd3MgYm90aCByZWFkaW5nIGFuZAotLy8gd3JpdGluZyB0byBaaXAgYXJjaGl2ZXMsIGluY2x1ZGluZyBkZWxldGlvbiBvZiBleGlzdGluZyBlbnRyaWVzLgotLy8KLSNpZm5kZWYgX19MSUJTX1pJUEZJTEVfSAotI2RlZmluZSBfX0xJQlNfWklQRklMRV9ICi0KLSNpbmNsdWRlICJaaXBFbnRyeS5oIgotI2luY2x1ZGUgIlZlY3Rvci5oIgotI2luY2x1ZGUgIkVycm9ycy5oIgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyoKLSAqIE1hbmlwdWxhdGUgYSBaaXAgYXJjaGl2ZS4KLSAqCi0gKiBTb21lIGNoYW5nZXMgd2lsbCBub3QgYmUgdmlzaWJsZSBpbiB0aGUgdW50aWwgdW50aWwgImZsdXNoIiBpcyBjYWxsZWQuCi0gKgotICogVGhlIGNvcnJlY3Qgd2F5IHRvIHVwZGF0ZSBhIGZpbGUgYXJjaGl2ZSBpcyB0byBtYWtlIGFsbCBjaGFuZ2VzIHRvIGEKLSAqIGNvcHkgb2YgdGhlIGFyY2hpdmUgaW4gYSB0ZW1wb3JhcnkgZmlsZSwgYW5kIHRoZW4gdW5saW5rL3JlbmFtZSBvdmVyCi0gKiB0aGUgb3JpZ2luYWwgYWZ0ZXIgZXZlcnl0aGluZyBjb21wbGV0ZXMuICBCZWNhdXNlIHdlJ3JlIG9ubHkgaW50ZXJlc3RlZAotICogaW4gdXNpbmcgdGhpcyBmb3IgcGFja2FnaW5nLCB3ZSBkb24ndCB3b3JyeSBhYm91dCBzdWNoIHRoaW5ncy4gIENyYXNoaW5nCi0gKiBhZnRlciBtYWtpbmcgY2hhbmdlcyBhbmQgYmVmb3JlIGZsdXNoKCkgY29tcGxldGVzIGNvdWxkIGxlYXZlIHVzIHdpdGgKLSAqIGFuIHVudXNhYmxlIFppcCBhcmNoaXZlLgotICovCi1jbGFzcyBaaXBGaWxlIHsKLXB1YmxpYzoKLSAgICBaaXBGaWxlKHZvaWQpCi0gICAgICA6IG1aaXBGcChOVUxMKSwgbVJlYWRPbmx5KGZhbHNlKSwgbU5lZWRDRFJld3JpdGUoZmFsc2UpCi0gICAgICB7fQotICAgIH5aaXBGaWxlKHZvaWQpIHsKLSAgICAgICAgaWYgKCFtUmVhZE9ubHkpCi0gICAgICAgICAgICBmbHVzaCgpOwotICAgICAgICBpZiAobVppcEZwICE9IE5VTEwpCi0gICAgICAgICAgICBmY2xvc2UobVppcEZwKTsKLSAgICAgICAgZGlzY2FyZEVudHJpZXMoKTsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIE9wZW4gYSBuZXcgb3IgZXhpc3RpbmcgYXJjaGl2ZS4KLSAgICAgKi8KLSAgICB0eXBlZGVmIGVudW0gewotICAgICAgICBrT3BlblJlYWRPbmx5ICAgPSAweDAxLAotICAgICAgICBrT3BlblJlYWRXcml0ZSAgPSAweDAyLAotICAgICAgICBrT3BlbkNyZWF0ZSAgICAgPSAweDA0LCAgICAgLy8gY3JlYXRlIGlmIGl0IGRvZXNuJ3QgZXhpc3QKLSAgICAgICAga09wZW5UcnVuY2F0ZSAgID0gMHgwOCwgICAgIC8vIGlmIGl0IGV4aXN0cywgZW1wdHkgaXQKLSAgICB9OwotICAgIHN0YXR1c190IG9wZW4oY29uc3QgY2hhciogemlwRmlsZU5hbWUsIGludCBmbGFncyk7Ci0KLSAgICAvKgotICAgICAqIEFkZCBhIGZpbGUgdG8gdGhlIGVuZCBvZiB0aGUgYXJjaGl2ZS4gIFNwZWNpZnkgd2hldGhlciB5b3Ugd2FudCB0aGUKLSAgICAgKiBsaWJyYXJ5IHRvIHRyeSB0byBzdG9yZSBpdCBjb21wcmVzc2VkLgotICAgICAqCi0gICAgICogSWYgInN0b3JhZ2VOYW1lIiBpcyBzcGVjaWZpZWQsIHRoZSBhcmNoaXZlIHdpbGwgdXNlIHRoYXQgaW5zdGVhZAotICAgICAqIG9mICJmaWxlTmFtZSIuCi0gICAgICoKLSAgICAgKiBJZiB0aGVyZSBpcyBhbHJlYWR5IGFuIGVudHJ5IHdpdGggdGhlIHNhbWUgbmFtZSwgdGhlIGNhbGwgZmFpbHMuCi0gICAgICogRXhpc3RpbmcgZW50cmllcyB3aXRoIHRoZSBzYW1lIG5hbWUgbXVzdCBiZSByZW1vdmVkIGZpcnN0LgotICAgICAqCi0gICAgICogSWYgInBwRW50cnkiIGlzIG5vbi1OVUxMLCBhIHBvaW50ZXIgdG8gdGhlIG5ldyBlbnRyeSB3aWxsIGJlIHJldHVybmVkLgotICAgICAqLwotICAgIHN0YXR1c190IGFkZChjb25zdCBjaGFyKiBmaWxlTmFtZSwgaW50IGNvbXByZXNzaW9uTWV0aG9kLAotICAgICAgICBaaXBFbnRyeSoqIHBwRW50cnkpCi0gICAgewotICAgICAgICByZXR1cm4gYWRkKGZpbGVOYW1lLCBmaWxlTmFtZSwgY29tcHJlc3Npb25NZXRob2QsIHBwRW50cnkpOwotICAgIH0KLSAgICBzdGF0dXNfdCBhZGQoY29uc3QgY2hhciogZmlsZU5hbWUsIGNvbnN0IGNoYXIqIHN0b3JhZ2VOYW1lLAotICAgICAgICBpbnQgY29tcHJlc3Npb25NZXRob2QsIFppcEVudHJ5KiogcHBFbnRyeSkKLSAgICB7Ci0gICAgICAgIHJldHVybiBhZGRDb21tb24oZmlsZU5hbWUsIE5VTEwsIDAsIHN0b3JhZ2VOYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgIFppcEVudHJ5OjprQ29tcHJlc3NTdG9yZWQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgY29tcHJlc3Npb25NZXRob2QsIHBwRW50cnkpOwotICAgIH0KLQotICAgIC8qCi0gICAgICogQWRkIGEgZmlsZSB0aGF0IGlzIGFscmVhZHkgY29tcHJlc3NlZCB3aXRoIGd6aXAuCi0gICAgICoKLSAgICAgKiBJZiAicHBFbnRyeSIgaXMgbm9uLU5VTEwsIGEgcG9pbnRlciB0byB0aGUgbmV3IGVudHJ5IHdpbGwgYmUgcmV0dXJuZWQuCi0gICAgICovCi0gICAgc3RhdHVzX3QgYWRkR3ppcChjb25zdCBjaGFyKiBmaWxlTmFtZSwgY29uc3QgY2hhciogc3RvcmFnZU5hbWUsCi0gICAgICAgIFppcEVudHJ5KiogcHBFbnRyeSkKLSAgICB7Ci0gICAgICAgIHJldHVybiBhZGRDb21tb24oZmlsZU5hbWUsIE5VTEwsIDAsIHN0b3JhZ2VOYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgIFppcEVudHJ5OjprQ29tcHJlc3NEZWZsYXRlZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICBaaXBFbnRyeTo6a0NvbXByZXNzRGVmbGF0ZWQsIHBwRW50cnkpOwotICAgIH0KLQotICAgIC8qCi0gICAgICogQWRkIGEgZmlsZSBmcm9tIGFuIGluLW1lbW9yeSBkYXRhIGJ1ZmZlci4KLSAgICAgKgotICAgICAqIElmICJwcEVudHJ5IiBpcyBub24tTlVMTCwgYSBwb2ludGVyIHRvIHRoZSBuZXcgZW50cnkgd2lsbCBiZSByZXR1cm5lZC4KLSAgICAgKi8KLSAgICBzdGF0dXNfdCBhZGQoY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsIGNvbnN0IGNoYXIqIHN0b3JhZ2VOYW1lLAotICAgICAgICBpbnQgY29tcHJlc3Npb25NZXRob2QsIFppcEVudHJ5KiogcHBFbnRyeSkKLSAgICB7Ci0gICAgICAgIHJldHVybiBhZGRDb21tb24oTlVMTCwgZGF0YSwgc2l6ZSwgc3RvcmFnZU5hbWUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgWmlwRW50cnk6OmtDb21wcmVzc1N0b3JlZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICBjb21wcmVzc2lvbk1ldGhvZCwgcHBFbnRyeSk7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBBZGQgYW4gZW50cnkgYnkgY29weWluZyBpdCBmcm9tIGFub3RoZXIgemlwIGZpbGUuICBJZiAicGFkZGluZyIgaXMKLSAgICAgKiBub256ZXJvLCB0aGUgc3BlY2lmaWVkIG51bWJlciBvZiBieXRlcyB3aWxsIGJlIGFkZGVkIHRvIHRoZSAiZXh0cmEiCi0gICAgICogZmllbGQgaW4gdGhlIGhlYWRlci4KLSAgICAgKgotICAgICAqIElmICJwcEVudHJ5IiBpcyBub24tTlVMTCwgYSBwb2ludGVyIHRvIHRoZSBuZXcgZW50cnkgd2lsbCBiZSByZXR1cm5lZC4KLSAgICAgKi8KLSAgICBzdGF0dXNfdCBhZGQoY29uc3QgWmlwRmlsZSogcFNvdXJjZVppcCwgY29uc3QgWmlwRW50cnkqIHBTb3VyY2VFbnRyeSwKLSAgICAgICAgaW50IHBhZGRpbmcsIFppcEVudHJ5KiogcHBFbnRyeSk7Ci0KLSAgICAvKgotICAgICAqIE1hcmsgYW4gZW50cnkgYXMgaGF2aW5nIGJlZW4gcmVtb3ZlZC4gIEl0IGlzIG5vdCBhY3R1YWxseSBkZWxldGVkCi0gICAgICogZnJvbSB0aGUgYXJjaGl2ZSBvciBvdXIgaW50ZXJuYWwgZGF0YSBzdHJ1Y3R1cmVzIHVudGlsIGZsdXNoKCkgaXMKLSAgICAgKiBjYWxsZWQuCi0gICAgICovCi0gICAgc3RhdHVzX3QgcmVtb3ZlKFppcEVudHJ5KiBwRW50cnkpOwotCi0gICAgLyoKLSAgICAgKiBGbHVzaCBjaGFuZ2VzLiAgSWYgbU5lZWRDRFJld3JpdGUgaXMgc2V0LCB0aGlzIHdyaXRlcyB0aGUgY2VudHJhbCBkaXIuCi0gICAgICovCi0gICAgc3RhdHVzX3QgZmx1c2godm9pZCk7Ci0KLSAgICAvKgotICAgICAqIEV4cGFuZCB0aGUgZGF0YSBpbnRvIHRoZSBidWZmZXIgcHJvdmlkZWQuICBUaGUgYnVmZmVyIG11c3QgaG9sZAotICAgICAqIGF0IGxlYXN0IDx1bmNvbXByZXNzZWQgbGVuPiBieXRlcy4gIFZhcmlhdGlvbiBleHBhbmRzIGRpcmVjdGx5Ci0gICAgICogdG8gYSBmaWxlLgotICAgICAqCi0gICAgICogUmV0dXJucyAiZmFsc2UiIGlmIGFuIGVycm9yIHdhcyBlbmNvdW50ZXJlZCBpbiB0aGUgY29tcHJlc3NlZCBkYXRhLgotICAgICAqLwotICAgIC8vYm9vbCB1bmNvbXByZXNzKGNvbnN0IFppcEVudHJ5KiBwRW50cnksIHZvaWQqIGJ1ZikgY29uc3Q7Ci0gICAgLy9ib29sIHVuY29tcHJlc3MoY29uc3QgWmlwRW50cnkqIHBFbnRyeSwgRklMRSogZnApIGNvbnN0OwotICAgIHZvaWQqIHVuY29tcHJlc3MoY29uc3QgWmlwRW50cnkqIHBFbnRyeSk7Ci0KLSAgICAvKgotICAgICAqIEdldCBhbiBlbnRyeSwgYnkgbmFtZS4gIFJldHVybnMgTlVMTCBpZiBub3QgZm91bmQuCi0gICAgICoKLSAgICAgKiBEb2VzIG5vdCByZXR1cm4gZW50cmllcyBwZW5kaW5nIGRlbGV0aW9uLgotICAgICAqLwotICAgIFppcEVudHJ5KiBnZXRFbnRyeUJ5TmFtZShjb25zdCBjaGFyKiBmaWxlTmFtZSkgY29uc3Q7Ci0KLSAgICAvKgotICAgICAqIEdldCB0aGUgTnRoIGVudHJ5IGluIHRoZSBhcmNoaXZlLgotICAgICAqCi0gICAgICogVGhpcyB3aWxsIHJldHVybiBhbiBlbnRyeSB0aGF0IGlzIHBlbmRpbmcgZGVsZXRpb24uCi0gICAgICovCi0gICAgaW50IGdldE51bUVudHJpZXModm9pZCkgY29uc3QgeyByZXR1cm4gbUVudHJpZXMuc2l6ZSgpOyB9Ci0gICAgWmlwRW50cnkqIGdldEVudHJ5QnlJbmRleChpbnQgaWR4KSBjb25zdDsKLQotcHJpdmF0ZToKLSAgICAvKiB0aGVzZSBhcmUgcHJpdmF0ZSBhbmQgbm90IGRlZmluZWQgKi8KLSAgICBaaXBGaWxlKGNvbnN0IFppcEZpbGUmIHNyYyk7Ci0gICAgWmlwRmlsZSYgb3BlcmF0b3I9KGNvbnN0IFppcEZpbGUmIHNyYyk7Ci0KLSAgICBjbGFzcyBFbmRPZkNlbnRyYWxEaXIgewotICAgIHB1YmxpYzoKLSAgICAgICAgRW5kT2ZDZW50cmFsRGlyKHZvaWQpIDoKLSAgICAgICAgICAgIG1EaXNrTnVtYmVyKDApLAotICAgICAgICAgICAgbURpc2tXaXRoQ2VudHJhbERpcigwKSwKLSAgICAgICAgICAgIG1OdW1FbnRyaWVzKDApLAotICAgICAgICAgICAgbVRvdGFsTnVtRW50cmllcygwKSwKLSAgICAgICAgICAgIG1DZW50cmFsRGlyU2l6ZSgwKSwKLSAgICAgICAgICAgIG1DZW50cmFsRGlyT2Zmc2V0KDApLAotICAgICAgICAgICAgbUNvbW1lbnRMZW4oMCksCi0gICAgICAgICAgICBtQ29tbWVudChOVUxMKQotICAgICAgICAgICAge30KLSAgICAgICAgdmlydHVhbCB+RW5kT2ZDZW50cmFsRGlyKHZvaWQpIHsKLSAgICAgICAgICAgIGRlbGV0ZVtdIG1Db21tZW50OwotICAgICAgICB9Ci0KLSAgICAgICAgc3RhdHVzX3QgcmVhZEJ1Zihjb25zdCB1bnNpZ25lZCBjaGFyKiBidWYsIGludCBsZW4pOwotICAgICAgICBzdGF0dXNfdCB3cml0ZShGSUxFKiBmcCk7Ci0KLSAgICAgICAgLy91bnNpZ25lZCBsb25nICAgbVNpZ25hdHVyZTsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1EaXNrTnVtYmVyOwotICAgICAgICB1bnNpZ25lZCBzaG9ydCAgbURpc2tXaXRoQ2VudHJhbERpcjsKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1OdW1FbnRyaWVzOwotICAgICAgICB1bnNpZ25lZCBzaG9ydCAgbVRvdGFsTnVtRW50cmllczsKLSAgICAgICAgdW5zaWduZWQgbG9uZyAgIG1DZW50cmFsRGlyU2l6ZTsKLSAgICAgICAgdW5zaWduZWQgbG9uZyAgIG1DZW50cmFsRGlyT2Zmc2V0OyAgICAgIC8vIG9mZnNldCBmcm9tIGZpcnN0IGRpc2sKLSAgICAgICAgdW5zaWduZWQgc2hvcnQgIG1Db21tZW50TGVuOwotICAgICAgICB1bnNpZ25lZCBjaGFyKiAgbUNvbW1lbnQ7Ci0KLSAgICAgICAgZW51bSB7Ci0gICAgICAgICAgICBrU2lnbmF0dXJlICAgICAgPSAweDA2MDU0YjUwLAotICAgICAgICAgICAga0VPQ0RMZW4gICAgICAgID0gMjIsICAgICAgIC8vIEVuZE9mQ2VudHJhbERpciBsZW4sIGV4Y2wuIGNvbW1lbnQKLQotICAgICAgICAgICAga01heENvbW1lbnRMZW4gID0gNjU1MzUsICAgIC8vIGxvbmdlc3QgcG9zc2libGUgaW4gdXNob3J0Ci0gICAgICAgICAgICBrTWF4RU9DRFNlYXJjaCAgPSBrTWF4Q29tbWVudExlbiArIEVuZE9mQ2VudHJhbERpcjo6a0VPQ0RMZW4sCi0KLSAgICAgICAgfTsKLQotICAgICAgICB2b2lkIGR1bXAodm9pZCkgY29uc3Q7Ci0gICAgfTsKLQotCi0gICAgLyogcmVhZCBhbGwgZW50cmllcyBpbiB0aGUgY2VudHJhbCBkaXIgKi8KLSAgICBzdGF0dXNfdCByZWFkQ2VudHJhbERpcih2b2lkKTsKLQotICAgIC8qIGNydW5jaCBkZWxldGVkIGVudHJpZXMgb3V0ICovCi0gICAgc3RhdHVzX3QgY3J1bmNoQXJjaGl2ZSh2b2lkKTsKLQotICAgIC8qIGNsZWFuIHVwIG1FbnRyaWVzICovCi0gICAgdm9pZCBkaXNjYXJkRW50cmllcyh2b2lkKTsKLQotICAgIC8qIGNvbW1vbiBoYW5kbGVyIGZvciBhbGwgImFkZCIgZnVuY3Rpb25zICovCi0gICAgc3RhdHVzX3QgYWRkQ29tbW9uKGNvbnN0IGNoYXIqIGZpbGVOYW1lLCBjb25zdCB2b2lkKiBkYXRhLCBzaXplX3Qgc2l6ZSwKLSAgICAgICAgY29uc3QgY2hhciogc3RvcmFnZU5hbWUsIGludCBzb3VyY2VUeXBlLCBpbnQgY29tcHJlc3Npb25NZXRob2QsCi0gICAgICAgIFppcEVudHJ5KiogcHBFbnRyeSk7Ci0KLSAgICAvKiBjb3B5IGFsbCBvZiAic3JjRnAiIGludG8gImRzdEZwIiAqLwotICAgIHN0YXR1c190IGNvcHlGcFRvRnAoRklMRSogZHN0RnAsIEZJTEUqIHNyY0ZwLCB1bnNpZ25lZCBsb25nKiBwQ1JDMzIpOwotICAgIC8qIGNvcHkgYWxsIG9mICJkYXRhIiBpbnRvICJkc3RGcCIgKi8KLSAgICBzdGF0dXNfdCBjb3B5RGF0YVRvRnAoRklMRSogZHN0RnAsCi0gICAgICAgIGNvbnN0IHZvaWQqIGRhdGEsIHNpemVfdCBzaXplLCB1bnNpZ25lZCBsb25nKiBwQ1JDMzIpOwotICAgIC8qIGNvcHkgc29tZSBvZiAic3JjRnAiIGludG8gImRzdEZwIiAqLwotICAgIHN0YXR1c190IGNvcHlQYXJ0aWFsRnBUb0ZwKEZJTEUqIGRzdEZwLCBGSUxFKiBzcmNGcCwgbG9uZyBsZW5ndGgsCi0gICAgICAgIHVuc2lnbmVkIGxvbmcqIHBDUkMzMik7Ci0gICAgLyogbGlrZSBtZW1tb3ZlKCksIGJ1dCBvbiBwYXJ0cyBvZiBhIHNpbmdsZSBmaWxlICovCi0gICAgc3RhdHVzX3QgZmlsZW1vdmUoRklMRSogZnAsIG9mZl90IGRlc3QsIG9mZl90IHNyYywgc2l6ZV90IG4pOwotICAgIC8qIGNvbXByZXNzIGFsbCBvZiAic3JjRnAiIGludG8gImRzdEZwIiwgdXNpbmcgRGVmbGF0ZSAqLwotICAgIHN0YXR1c190IGNvbXByZXNzRnBUb0ZwKEZJTEUqIGRzdEZwLCBGSUxFKiBzcmNGcCwKLSAgICAgICAgY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsIHVuc2lnbmVkIGxvbmcqIHBDUkMzMik7Ci0KLSAgICAvKiBnZXQgbW9kaWZpY2F0aW9uIGRhdGUgZnJvbSBhIGZpbGUgZGVzY3JpcHRvciAqLwotICAgIHRpbWVfdCBnZXRNb2RUaW1lKGludCBmZCk7Ci0KLSAgICAvKgotICAgICAqIFdlIHVzZSBzdGRpbyBGSUxFKiwgd2hpY2ggZ2l2ZXMgdXMgYnVmZmVyaW5nIGJ1dCBtYWtlcyBkZWFsaW5nCi0gICAgICogd2l0aCBmaWxlcyA+MkdCIGF3a3dhcmQuICBVbnRpbCB3ZSBzdXBwb3J0IFppcDY0LCB3ZSdyZSBmaW5lLgotICAgICAqLwotICAgIEZJTEUqICAgICAgICAgICBtWmlwRnA7ICAgICAgICAgICAgIC8vIFppcCBmaWxlIHBvaW50ZXIKLQotICAgIC8qIG9uZSBvZiB0aGVzZSBwZXIgZmlsZSAqLwotICAgIEVuZE9mQ2VudHJhbERpciBtRU9DRDsKLQotICAgIC8qIGRpZCB3ZSBvcGVuIHRoaXMgcmVhZC1vbmx5PyAqLwotICAgIGJvb2wgICAgICAgICAgICBtUmVhZE9ubHk7Ci0KLSAgICAvKiBzZXQgdGhpcyB3aGVuIHdlIHRyYXNoIHRoZSBjZW50cmFsIGRpciAqLwotICAgIGJvb2wgICAgICAgICAgICBtTmVlZENEUmV3cml0ZTsKLQotICAgIC8qCi0gICAgICogT25lIFppcEVudHJ5IHBlciBlbnRyeSBpbiB0aGUgemlwIGZpbGUuICBJJ20gdXNpbmcgcG9pbnRlcnMgaW5zdGVhZAotICAgICAqIG9mIG9iamVjdHMgYmVjYXVzZSBpdCdzIGVhc2llciB0aGFuIG1ha2luZyBvcGVyYXRvcj0gd29yayBmb3IgdGhlCi0gICAgICogY2xhc3NlcyBhbmQgc3ViLWNsYXNzZXMuCi0gICAgICovCi0gICAgVmVjdG9yPFppcEVudHJ5Kj4gICBtRW50cmllczsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBfX0xJQlNfWklQRklMRV9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL1ppcEZpbGVDUk8uaCBiL2luY2x1ZGUvdXRpbHMvWmlwRmlsZUNSTy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzMGUwMDM2Li4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvWmlwRmlsZUNSTy5oCisrKyAvZGV2L251bGwKQEAgLTEsNTkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBDIEFQSSBmb3IgZWFkLW9ubHkgYWNjZXNzIHRvIFppcCBhcmNoaXZlcywgd2l0aCBtaW5pbWFsIGhlYXAgYWxsb2NhdGlvbi4KLS8vCi0jaWZuZGVmIF9fTElCU19aSVBGSUxFQ1JPX0gKLSNkZWZpbmUgX19MSUJTX1pJUEZJTEVDUk9fSAotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLQotI2lmZGVmIF9fY3BsdXNwbHVzCi1leHRlcm4gIkMiIHsKLSNlbmRpZgotCi0vKgotICogVHJpdmlhbCB0eXBlZGVmIHRvIGVuc3VyZSB0aGF0IFppcEZpbGVDUk8gaXMgbm90IHRyZWF0ZWQgYXMgYSBzaW1wbGUgaW50ZWdlci4KLSAqLwotdHlwZWRlZiB2b2lkKiBaaXBGaWxlQ1JPOwotCi0vKgotICogVHJpdmlhbCB0eXBlZGVmIHRvIGVuc3VyZSB0aGF0IFppcEVudHJ5Q1JPIGlzIG5vdCB0cmVhdGVkIGFzIGEgc2ltcGxlCi0gKiBpbnRlZ2VyLiAgV2UgdXNlIE5VTEwgdG8gaW5kaWNhdGUgYW4gaW52YWxpZCB2YWx1ZS4KLSAqLwotdHlwZWRlZiB2b2lkKiBaaXBFbnRyeUNSTzsKLQotZXh0ZXJuIFppcEZpbGVDUk8gWmlwRmlsZVhST19vcGVuKGNvbnN0IGNoYXIqIHBhdGgpOwotCi1leHRlcm4gdm9pZCBaaXBGaWxlQ1JPX2Rlc3Ryb3koWmlwRmlsZUNSTyB6aXApOwotCi1leHRlcm4gWmlwRW50cnlDUk8gWmlwRmlsZUNST19maW5kRW50cnlCeU5hbWUoWmlwRmlsZUNSTyB6aXAsCi0gICAgICAgIGNvbnN0IGNoYXIqIGZpbGVOYW1lKTsKLQotZXh0ZXJuIGJvb2wgWmlwRmlsZUNST19nZXRFbnRyeUluZm8oWmlwRmlsZUNSTyB6aXAsIFppcEVudHJ5Q1JPIGVudHJ5LAotICAgICAgICBpbnQqIHBNZXRob2QsIGxvbmcqIHBVbmNvbXBMZW4sCi0gICAgICAgIGxvbmcqIHBDb21wTGVuLCBvZmZfdCogcE9mZnNldCwgbG9uZyogcE1vZFdoZW4sIGxvbmcqIHBDcmMzMik7Ci0KLWV4dGVybiBib29sIFppcEZpbGVDUk9fdW5jb21wcmVzc0VudHJ5KFppcEZpbGVDUk8gemlwLCBaaXBFbnRyeUNSTyBlbnRyeSwgaW50IGZkKTsKLQotI2lmZGVmIF9fY3BsdXNwbHVzCi19Ci0jZW5kaWYKLQotI2VuZGlmIC8qX19MSUJTX1pJUEZJTEVDUk9fSCovCmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL1ppcEZpbGVSTy5oIGIvaW5jbHVkZS91dGlscy9aaXBGaWxlUk8uaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNTFjNGYyZi4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL1ppcEZpbGVSTy5oCisrKyAvZGV2L251bGwKQEAgLTEsMjIyICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gUmVhZC1vbmx5IGFjY2VzcyB0byBaaXAgYXJjaGl2ZXMsIHdpdGggbWluaW1hbCBoZWFwIGFsbG9jYXRpb24uCi0vLwotLy8gVGhpcyBpcyBzaW1pbGFyIHRvIHRoZSBtb3JlLWNvbXBsZXRlIFppcEZpbGUgY2xhc3MsIGJ1dCBubyBhdHRlbXB0Ci0vLyBoYXMgYmVlbiBtYWRlIHRvIG1ha2UgdGhlbSBpbnRlcmNoYW5nZWFibGUuICBUaGlzIGNsYXNzIG9wZXJhdGVzIHVuZGVyCi0vLyBhIHZlcnkgZGlmZmVyZW50IHNldCBvZiBhc3N1bXB0aW9ucyBhbmQgY29uc3RyYWludHMuCi0vLwotI2lmbmRlZiBfX0xJQlNfWklQRklMRVJPX0gKLSNkZWZpbmUgX19MSUJTX1pJUEZJTEVST19ICi0KLSNpbmNsdWRlICJFcnJvcnMuaCIKLSNpbmNsdWRlICJGaWxlTWFwLmgiCi0KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHVuaXN0ZC5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8qCi0gKiBUcml2aWFsIHR5cGVkZWYgdG8gZW5zdXJlIHRoYXQgWmlwRW50cnlSTyBpcyBub3QgdHJlYXRlZCBhcyBhIHNpbXBsZQotICogaW50ZWdlci4gIFdlIHVzZSBOVUxMIHRvIGluZGljYXRlIGFuIGludmFsaWQgdmFsdWUuCi0gKi8KLXR5cGVkZWYgdm9pZCogWmlwRW50cnlSTzsKLQotLyoKLSAqIE9wZW4gYSBaaXAgYXJjaGl2ZSBmb3IgcmVhZGluZy4KLSAqCi0gKiBXZSB3YW50ICJvcGVuIiBhbmQgImZpbmQgZW50cnkgYnkgbmFtZSIgdG8gYmUgZmFzdCBvcGVyYXRpb25zLCBhbmQgd2UKLSAqIHdhbnQgdG8gdXNlIGFzIGxpdHRsZSBtZW1vcnkgYXMgcG9zc2libGUuICBXZSBtZW1vcnktbWFwIHRoZSBmaWxlLAotICogYW5kIGxvYWQgYSBoYXNoIHRhYmxlIHdpdGggcG9pbnRlcnMgdG8gdGhlIGZpbGVuYW1lcyAod2hpY2ggYXJlbid0Ci0gKiBudWxsLXRlcm1pbmF0ZWQpLiAgVGhlIG90aGVyIGZpZWxkcyBhcmUgYXQgYSBmaXhlZCBvZmZzZXQgZnJvbSB0aGUKLSAqIGZpbGVuYW1lLCBzbyB3ZSBkb24ndCBuZWVkIHRvIGV4dHJhY3QgdGhvc2UgKGJ1dCB3ZSBkbyBuZWVkIHRvIGJ5dGUtcmVhZAotICogYW5kIGVuZGlhbi1zd2FwIHRoZW0gZXZlcnkgdGltZSB3ZSB3YW50IHRoZW0pLgotICoKLSAqIFRvIHNwZWVkIGNvbXBhcmlzb25zIHdoZW4gZG9pbmcgYSBsb29rdXAgYnkgbmFtZSwgd2UgY291bGQgbWFrZSB0aGUgbWFwcGluZwotICogInByaXZhdGUiIChjb3B5LW9uLXdyaXRlKSBhbmQgbnVsbC10ZXJtaW5hdGUgdGhlIGZpbGVuYW1lcyBhZnRlciB2ZXJpZnlpbmcKLSAqIHRoZSByZWNvcmQgc3RydWN0dXJlLiAgSG93ZXZlciwgdGhpcyByZXF1aXJlcyBhIHByaXZhdGUgbWFwcGluZyBvZgotICogZXZlcnkgcGFnZSB0aGF0IHRoZSBDZW50cmFsIERpcmVjdG9yeSB0b3VjaGVzLiAgRWFzaWVyIHRvIHR1Y2sgYSBjb3B5Ci0gKiBvZiB0aGUgc3RyaW5nIGxlbmd0aCBpbnRvIHRoZSBoYXNoIHRhYmxlIGVudHJ5LgotICovCi1jbGFzcyBaaXBGaWxlUk8gewotcHVibGljOgotICAgIFppcEZpbGVSTygpCi0gICAgICAgIDogbUZkKC0xKSwgbUZpbGVNYXAoTlVMTCksIG1IYXNoVGFibGVTaXplKC0xKSwgbUhhc2hUYWJsZShOVUxMKQotICAgICAgICB7fQotICAgIH5aaXBGaWxlUk8oKSB7Ci0gICAgICAgIGZyZWUobUhhc2hUYWJsZSk7Ci0gICAgICAgIGlmIChtRmlsZU1hcCkKLSAgICAgICAgICAgIG1GaWxlTWFwLT5yZWxlYXNlKCk7Ci0gICAgICAgIGlmIChtRmQgPj0gMCkKLSAgICAgICAgICAgIGNsb3NlKG1GZCk7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBPcGVuIGFuIGFyY2hpdmUuCi0gICAgICovCi0gICAgc3RhdHVzX3Qgb3Blbihjb25zdCBjaGFyKiB6aXBGaWxlTmFtZSk7Ci0KLSAgICAvKgotICAgICAqIEZpbmQgYW4gZW50cnksIGJ5IG5hbWUuICBSZXR1cm5zIHRoZSBlbnRyeSBpZGVudGlmaWVyLCBvciBOVUxMIGlmCi0gICAgICogbm90IGZvdW5kLgotICAgICAqCi0gICAgICogSWYgdHdvIGVudHJpZXMgaGF2ZSB0aGUgc2FtZSBuYW1lLCBvbmUgd2lsbCBiZSBjaG9zZW4gYXQgc2VtaS1yYW5kb20uCi0gICAgICovCi0gICAgWmlwRW50cnlSTyBmaW5kRW50cnlCeU5hbWUoY29uc3QgY2hhciogZmlsZU5hbWUpIGNvbnN0OwotCi0gICAgLyoKLSAgICAgKiBSZXR1cm4gdGhlICNvZiBlbnRyaWVzIGluIHRoZSBaaXAgYXJjaGl2ZS4KLSAgICAgKi8KLSAgICBpbnQgZ2V0TnVtRW50cmllcyh2b2lkKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiBtTnVtRW50cmllczsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFJldHVybiB0aGUgTnRoIGVudHJ5LiAgWmlwIGZpbGUgZW50cmllcyBhcmUgbm90IHN0b3JlZCBpbiBzb3J0ZWQKLSAgICAgKiBvcmRlciwgYW5kIHVwZGF0ZWQgZW50cmllcyBtYXkgYXBwZWFyIGF0IHRoZSBlbmQsIHNvIGFueW9uZSB3YWxraW5nCi0gICAgICogdGhlIGFyY2hpdmUgbmVlZHMgdG8gYXZvaWQgbWFraW5nIG9yZGVyaW5nIGFzc3VtcHRpb25zLiAgV2UgdGFrZQotICAgICAqIHRoYXQgZnVydGhlciBieSByZXR1cm5pbmcgdGhlIE50aCBub24tZW1wdHkgZW50cnkgaW4gdGhlIGhhc2ggdGFibGUKLSAgICAgKiByYXRoZXIgdGhhbiB0aGUgTnRoIGVudHJ5IGluIHRoZSBhcmNoaXZlLgotICAgICAqCi0gICAgICogVmFsaWQgdmFsdWVzIGFyZSBbMC4ubnVtRW50cmllcykuCi0gICAgICoKLSAgICAgKiBbVGhpcyBpcyBjdXJyZW50bHkgTyhuKS4gIElmIGl0IG5lZWRzIHRvIGJlIGZhc3Qgd2UgY2FuIGFsbG9jYXRlIGFuCi0gICAgICogYWRkaXRpb25hbCBkYXRhIHN0cnVjdHVyZSBvciBwcm92aWRlIGFuIGl0ZXJhdG9yIGludGVyZmFjZS5dCi0gICAgICovCi0gICAgWmlwRW50cnlSTyBmaW5kRW50cnlCeUluZGV4KGludCBpZHgpIGNvbnN0OwotCi0gICAgLyoKLSAgICAgKiBDb3B5IHRoZSBmaWxlbmFtZSBpbnRvIHRoZSBzdXBwbGllZCBidWZmZXIuICBSZXR1cm5zIDAgb24gc3VjY2VzcywKLSAgICAgKiAtMSBpZiAiZW50cnkiIGlzIGludmFsaWQsIG9yIHRoZSBmaWxlbmFtZSBsZW5ndGggaWYgaXQgZGlkbid0IGZpdC4gIFRoZQotICAgICAqIGxlbmd0aCwgYW5kIHRoZSByZXR1cm5lZCBzdHJpbmcsIGluY2x1ZGUgdGhlIG51bGwtdGVybWluYXRpb24uCi0gICAgICovCi0gICAgaW50IGdldEVudHJ5RmlsZU5hbWUoWmlwRW50cnlSTyBlbnRyeSwgY2hhciogYnVmZmVyLCBpbnQgYnVmTGVuKSBjb25zdDsKLQotICAgIC8qCi0gICAgICogR2V0IHRoZSB2aXRhbCBzdGF0cyBmb3IgYW4gZW50cnkuICBQYXNzIGluIE5VTEwgcG9pbnRlcnMgZm9yIGFueXRoaW5nCi0gICAgICogeW91IGRvbid0IG5lZWQuCi0gICAgICoKLSAgICAgKiAiKnBPZmZzZXQiIGhvbGRzIHRoZSBaaXAgZmlsZSBvZmZzZXQgb2YgdGhlIGVudHJ5J3MgZGF0YS4KLSAgICAgKgotICAgICAqIFJldHVybnMgImZhbHNlIiBpZiAiZW50cnkiIGlzIGJvZ3VzIG9yIGlmIHRoZSBkYXRhIGluIHRoZSBaaXAgZmlsZQotICAgICAqIGFwcGVhcnMgdG8gYmUgYmFkLgotICAgICAqLwotICAgIGJvb2wgZ2V0RW50cnlJbmZvKFppcEVudHJ5Uk8gZW50cnksIGludCogcE1ldGhvZCwgbG9uZyogcFVuY29tcExlbiwKLSAgICAgICAgbG9uZyogcENvbXBMZW4sIG9mZl90KiBwT2Zmc2V0LCBsb25nKiBwTW9kV2hlbiwgbG9uZyogcENyYzMyKSBjb25zdDsKLQotICAgIC8qCi0gICAgICogQ3JlYXRlIGEgbmV3IEZpbGVNYXAgb2JqZWN0IHRoYXQgbWFwcyBhIHN1YnNldCBvZiB0aGUgYXJjaGl2ZS4gIEZvcgotICAgICAqIGFuIHVuY29tcHJlc3NlZCBlbnRyeSB0aGlzIGVmZmVjdGl2ZWx5IHByb3ZpZGVzIGEgcG9pbnRlciB0byB0aGUKLSAgICAgKiBhY3R1YWwgZGF0YSwgZm9yIGEgY29tcHJlc3NlZCBlbnRyeSB0aGlzIHByb3ZpZGVzIHRoZSBpbnB1dCBidWZmZXIKLSAgICAgKiBmb3IgaW5mbGF0ZSgpLgotICAgICAqLwotICAgIEZpbGVNYXAqIGNyZWF0ZUVudHJ5RmlsZU1hcChaaXBFbnRyeVJPIGVudHJ5KSBjb25zdDsKLQotICAgIC8qCi0gICAgICogVW5jb21wcmVzcyB0aGUgZGF0YSBpbnRvIGEgYnVmZmVyLiAgRGVwZW5kaW5nIG9uIHRoZSBjb21wcmVzc2lvbgotICAgICAqIGZvcm1hdCwgdGhpcyBpcyBlaXRoZXIgYW4gImluZmxhdGUiIG9wZXJhdGlvbiBvciBhIG1lbWNweS4KLSAgICAgKgotICAgICAqIFVzZSAidW5jb21wTGVuIiBmcm9tIGdldEVudHJ5SW5mbygpIHRvIGRldGVybWluZSB0aGUgcmVxdWlyZWQKLSAgICAgKiBidWZmZXIgc2l6ZS4KLSAgICAgKgotICAgICAqIFJldHVybnMgInRydWUiIG9uIHN1Y2Nlc3MuCi0gICAgICovCi0gICAgYm9vbCB1bmNvbXByZXNzRW50cnkoWmlwRW50cnlSTyBlbnRyeSwgdm9pZCogYnVmZmVyKSBjb25zdDsKLQotICAgIC8qCi0gICAgICogVW5jb21wcmVzcyB0aGUgZGF0YSB0byBhbiBvcGVuIGZpbGUgZGVzY3JpcHRvci4KLSAgICAgKi8KLSAgICBib29sIHVuY29tcHJlc3NFbnRyeShaaXBFbnRyeVJPIGVudHJ5LCBpbnQgZmQpIGNvbnN0OwotCi0gICAgLyogWmlwIGNvbXByZXNzaW9uIG1ldGhvZHMgd2Ugc3VwcG9ydCAqLwotICAgIGVudW0gewotICAgICAgICBrQ29tcHJlc3NTdG9yZWQgICAgID0gMCwgICAgICAgIC8vIG5vIGNvbXByZXNzaW9uCi0gICAgICAgIGtDb21wcmVzc0RlZmxhdGVkICAgPSA4LCAgICAgICAgLy8gc3RhbmRhcmQgZGVmbGF0ZQotICAgIH07Ci0KLSAgICAvKgotICAgICAqIFV0aWxpdHkgZnVuY3Rpb246IHVuY29tcHJlc3MgZGVmbGF0ZWQgZGF0YSwgYnVmZmVyIHRvIGJ1ZmZlci4KLSAgICAgKi8KLSAgICBzdGF0aWMgYm9vbCBpbmZsYXRlQnVmZmVyKHZvaWQqIG91dEJ1ZiwgY29uc3Qgdm9pZCogaW5CdWYsCi0gICAgICAgIGxvbmcgdW5jb21wTGVuLCBsb25nIGNvbXBMZW4pOwotCi0gICAgLyoKLSAgICAgKiBVdGlsaXR5IGZ1bmN0aW9uOiB1bmNvbXByZXNzIGRlZmxhdGVkIGRhdGEsIGJ1ZmZlciB0byBmZC4KLSAgICAgKi8KLSAgICBzdGF0aWMgYm9vbCBpbmZsYXRlQnVmZmVyKGludCBmZCwgY29uc3Qgdm9pZCogaW5CdWYsCi0gICAgICAgIGxvbmcgdW5jb21wTGVuLCBsb25nIGNvbXBMZW4pOwotCi0gICAgLyoKLSAgICAgKiBTb21lIGJhc2ljIGZ1bmN0aW9ucyBmb3IgcmF3IGRhdGEgbWFuaXB1bGF0aW9uLiAgIkxFIiBtZWFucwotICAgICAqIExpdHRsZSBFbmRpYW4uCi0gICAgICovCi0gICAgc3RhdGljIGlubGluZSB1bnNpZ25lZCBzaG9ydCBnZXQyTEUoY29uc3QgdW5zaWduZWQgY2hhciogYnVmKSB7Ci0gICAgICAgIHJldHVybiBidWZbMF0gfCAoYnVmWzFdIDw8IDgpOwotICAgIH0KLSAgICBzdGF0aWMgaW5saW5lIHVuc2lnbmVkIGxvbmcgZ2V0NExFKGNvbnN0IHVuc2lnbmVkIGNoYXIqIGJ1ZikgewotICAgICAgICByZXR1cm4gYnVmWzBdIHwgKGJ1ZlsxXSA8PCA4KSB8IChidWZbMl0gPDwgMTYpIHwgKGJ1ZlszXSA8PCAyNCk7Ci0gICAgfQotCi1wcml2YXRlOgotICAgIC8qIHRoZXNlIGFyZSBwcml2YXRlIGFuZCBub3QgZGVmaW5lZCAqLyAKLSAgICBaaXBGaWxlUk8oY29uc3QgWmlwRmlsZVJPJiBzcmMpOwotICAgIFppcEZpbGVSTyYgb3BlcmF0b3I9KGNvbnN0IFppcEZpbGVSTyYgc3JjKTsKLQotICAgIC8qIHBhcnNlIHRoZSBhcmNoaXZlLCBwcmVwcGluZyBpbnRlcm5hbCBzdHJ1Y3R1cmVzICovCi0gICAgYm9vbCBwYXJzZVppcEFyY2hpdmUodm9pZCk7Ci0KLSAgICAvKiBhZGQgYSBuZXcgZW50cnkgdG8gdGhlIGhhc2ggdGFibGUgKi8KLSAgICB2b2lkIGFkZFRvSGFzaChjb25zdCBjaGFyKiBzdHIsIGludCBzdHJMZW4sIHVuc2lnbmVkIGludCBoYXNoKTsKLQotICAgIC8qIGNvbXB1dGUgc3RyaW5nIGhhc2ggY29kZSAqLwotICAgIHN0YXRpYyB1bnNpZ25lZCBpbnQgY29tcHV0ZUhhc2goY29uc3QgY2hhciogc3RyLCBpbnQgbGVuKTsKLQotICAgIC8qIGNvbnZlcnQgYSBaaXBFbnRyeVJPIGJhY2sgdG8gYSBoYXNoIHRhYmxlIGluZGV4ICovCi0gICAgaW50IGVudHJ5VG9JbmRleChjb25zdCBaaXBFbnRyeVJPIGVudHJ5KSBjb25zdDsKLQotICAgIC8qCi0gICAgICogT25lIGVudHJ5IGluIHRoZSBoYXNoIHRhYmxlLgotICAgICAqLwotICAgIHR5cGVkZWYgc3RydWN0IEhhc2hFbnRyeSB7Ci0gICAgICAgIGNvbnN0IGNoYXIqICAgICBuYW1lOwotICAgICAgICB1bnNpZ25lZCBzaG9ydCAgbmFtZUxlbjsKLSAgICAgICAgLy91bnNpZ25lZCBpbnQgICAgaGFzaDsKLSAgICB9IEhhc2hFbnRyeTsKLQotICAgIC8qIG9wZW4gWmlwIGFyY2hpdmUgKi8KLSAgICBpbnQgICAgICAgICBtRmQ7Ci0KLSAgICAvKiBtYXBwZWQgZmlsZSAqLwotICAgIEZpbGVNYXAqICAgIG1GaWxlTWFwOwotCi0gICAgLyogbnVtYmVyIG9mIGVudHJpZXMgaW4gdGhlIFppcCBhcmNoaXZlICovCi0gICAgaW50ICAgICAgICAgbU51bUVudHJpZXM7Ci0KLSAgICAvKgotICAgICAqIFdlIGtub3cgaG93IG1hbnkgZW50cmllcyBhcmUgaW4gdGhlIFppcCBhcmNoaXZlLCBzbyB3ZSBoYXZlIGEKLSAgICAgKiBmaXhlZC1zaXplIGhhc2ggdGFibGUuICBXZSBwcm9iZSBmb3IgYW4gZW1wdHkgc2xvdC4KLSAgICAgKi8KLSAgICBpbnQgICAgICAgICBtSGFzaFRhYmxlU2l6ZTsKLSAgICBIYXNoRW50cnkqICBtSGFzaFRhYmxlOwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8qX19MSUJTX1pJUEZJTEVST19IKi8KZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvWmlwVXRpbHMuaCBiL2luY2x1ZGUvdXRpbHMvWmlwVXRpbHMuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDJjNDJiNi4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL1ppcFV0aWxzLmgKKysrIC9kZXYvbnVsbApAQCAtMSw2NyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIE1pc2NlbGxhbmVvdXMgemlwL2d6aXAgdXRpbGl0eSBmdW5jdGlvbnMuCi0vLwotI2lmbmRlZiBfX0xJQlNfWklQVVRJTFNfSAotI2RlZmluZSBfX0xJQlNfWklQVVRJTFNfSAotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vKgotICogQ29udGFpbmVyIGNsYXNzIGZvciB1dGlsaXR5IGZ1bmN0aW9ucywgcHJpbWFyaWx5IGZvciBuYW1lc3BhY2UgcmVhc29ucy4KLSAqLwotY2xhc3MgWmlwVXRpbHMgewotcHVibGljOgotICAgIC8qCi0gICAgICogR2VuZXJhbCB1dGlsaXR5IGZ1bmN0aW9uIGZvciB1bmNvbXByZXNzaW5nICJkZWZsYXRlIiBkYXRhIGZyb20gYSBmaWxlCi0gICAgICogdG8gYSBidWZmZXIuCi0gICAgICovCi0gICAgc3RhdGljIGJvb2wgaW5mbGF0ZVRvQnVmZmVyKGludCBmZCwgdm9pZCogYnVmLCBsb25nIHVuY29tcHJlc3NlZExlbiwKLSAgICAgICAgbG9uZyBjb21wcmVzc2VkTGVuKTsKLSAgICBzdGF0aWMgYm9vbCBpbmZsYXRlVG9CdWZmZXIoRklMRSogZnAsIHZvaWQqIGJ1ZiwgbG9uZyB1bmNvbXByZXNzZWRMZW4sCi0gICAgICAgIGxvbmcgY29tcHJlc3NlZExlbik7Ci0KLSAgICAvKgotICAgICAqIFNvbWVkYXkgd2UgbWlnaHQgd2FudCB0byBtYWtlIHRoaXMgZ2VuZXJpYyBhbmQgaGFuZGxlIGJ6aXAyICIuYnoyIgotICAgICAqIGZpbGVzIHRvby4KLSAgICAgKgotICAgICAqIFdlIGNvdWxkIGRlY2xhcmUgZ3ppcCB0byBiZSBhIHN1Yi1jbGFzcyBvZiB6aXAgdGhhdCBoYXMgZXhhY3RseQotICAgICAqIG9uZSBhbHdheXMtY29tcHJlc3NlZCBlbnRyeSwgYnV0IHdlIGN1cnJlbnRseSB3YW50IHRvIHRyZWF0IFppcAotICAgICAqIGFuZCBnemlwIGFzIGRpc3RpbmN0LCBzbyB0aGVyZSdzIG5vIHZhbHVlLgotICAgICAqCi0gICAgICogVGhlIHpsaWIgbGlicmFyeSBoYXMgc29tZSBnemlwIHV0aWxpdGllcywgYnV0IGl0IGhhcyBubyBpbnRlcmZhY2UKLSAgICAgKiBmb3IgZXh0cmFjdGluZyB0aGUgdW5jb21wcmVzc2VkIGxlbmd0aCBvZiB0aGUgZmlsZSAoeW91IGRvICpub3QqCi0gICAgICogd2FudCB0byBnenNlZWsgdG8gdGhlIGVuZCkuCi0gICAgICoKLSAgICAgKiBQYXNzIGluIGEgc2Vla2VkIGZpbGUgcG9pbnRlciBmb3IgdGhlIGd6aXAgZmlsZS4gIElmIHRoaXMgaXMgYSBnemlwCi0gICAgICogZmlsZSwgd2Ugc2V0IG91ciByZXR1cm4gdmFsdWVzIGFwcHJvcHJpYXRlbHkgYW5kIHJldHVybiAidHJ1ZSIgd2l0aAotICAgICAqIHRoZSBmaWxlIHNlZWtlZCB0byB0aGUgc3RhcnQgb2YgdGhlIGNvbXByZXNzZWQgZGF0YS4KLSAgICAgKi8KLSAgICBzdGF0aWMgYm9vbCBleGFtaW5lR3ppcChGSUxFKiBmcCwgaW50KiBwQ29tcHJlc3Npb25NZXRob2QsCi0gICAgICAgIGxvbmcqIHBVbmNvbXByZXNzZWRMZW4sIGxvbmcqIHBDb21wcmVzc2VkTGVuLCB1bnNpZ25lZCBsb25nKiBwQ1JDMzIpOwotCi1wcml2YXRlOgotICAgIFppcFV0aWxzKCkge30KLSAgICB+WmlwVXRpbHMoKSB7fQotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8qX19MSUJTX1pJUFVUSUxTX0gqLwpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9hc2htZW0uaCBiL2luY2x1ZGUvdXRpbHMvYXNobWVtLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDA4NTQ3NzUuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9hc2htZW0uaAorKysgL2Rldi9udWxsCkBAIC0xLDQxICswLDAgQEAKLS8qIHV0aWxzL2FzaG1lbS5oCi0gKioKLSAqKiBDb3B5cmlnaHQgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKioKLSAqKiBUaGlzIGZpbGUgaXMgZHVhbCBsaWNlbnNlZC4gIEl0IG1heSBiZSByZWRpc3RyaWJ1dGVkIGFuZC9vciBtb2RpZmllZAotICoqIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgQXBhY2hlIDIuMCBMaWNlbnNlIE9SIHZlcnNpb24gMiBvZiB0aGUgR05VCi0gKiogR2VuZXJhbCBQdWJsaWMgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIF9VVElMU19BU0hNRU1fSAotI2RlZmluZSBfVVRJTFNfQVNITUVNX0gKLQotI2luY2x1ZGUgPGxpbnV4L2xpbWl0cy5oPgotI2luY2x1ZGUgPGxpbnV4L2lvY3RsLmg+Ci0KLSNkZWZpbmUgQVNITUVNX05BTUVfTEVOCQkyNTYKLQotI2RlZmluZSBBU0hNRU1fTkFNRV9ERUYJCSJkZXYvYXNobWVtIgotCi0vKiBSZXR1cm4gdmFsdWVzIGZyb20gQVNITUVNX1BJTjogV2FzIHRoZSBtYXBwaW5nIHB1cmdlZCB3aGlsZSB1bnBpbm5lZD8gKi8KLSNkZWZpbmUgQVNITUVNX05PVF9SRUFQRUQJMAotI2RlZmluZSBBU0hNRU1fV0FTX1JFQVBFRAkxCi0KLS8qIFJldHVybiB2YWx1ZXMgZnJvbSBBU0hNRU1fVU5QSU46IElzIHRoZSBtYXBwaW5nIG5vdyBwaW5uZWQgb3IgdW5waW5uZWQ/ICovCi0jZGVmaW5lIEFTSE1FTV9OT1dfVU5QSU5ORUQJMAotI2RlZmluZSBBU0hNRU1fTk9XX1BJTk5FRAkxCi0KLSNkZWZpbmUgX19BU0hNRU1JT0MJCTB4NzcKLQotI2RlZmluZSBBU0hNRU1fU0VUX05BTUUJCV9JT1coX19BU0hNRU1JT0MsIDEsIGNoYXJbQVNITUVNX05BTUVfTEVOXSkKLSNkZWZpbmUgQVNITUVNX0dFVF9OQU1FCQlfSU9SKF9fQVNITUVNSU9DLCAyLCBjaGFyW0FTSE1FTV9OQU1FX0xFTl0pCi0jZGVmaW5lIEFTSE1FTV9TRVRfU0laRQkJX0lPVyhfX0FTSE1FTUlPQywgMywgc2l6ZV90KQotI2RlZmluZSBBU0hNRU1fR0VUX1NJWkUJCV9JTyhfX0FTSE1FTUlPQywgNCkKLSNkZWZpbmUgQVNITUVNX1NFVF9QUk9UX01BU0sJX0lPVyhfX0FTSE1FTUlPQywgNSwgdW5zaWduZWQgbG9uZykKLSNkZWZpbmUgQVNITUVNX0dFVF9QUk9UX01BU0sJX0lPKF9fQVNITUVNSU9DLCA2KQotI2RlZmluZSBBU0hNRU1fUElOCQlfSU8oX19BU0hNRU1JT0MsIDcpCi0jZGVmaW5lIEFTSE1FTV9VTlBJTgkJX0lPKF9fQVNITUVNSU9DLCA4KQotI2RlZmluZSBBU0hNRU1fSVNQSU5ORUQJCV9JTyhfX0FTSE1FTUlPQywgOSkKLSNkZWZpbmUgQVNITUVNX1BVUkdFX0FMTF9DQUNIRVMJX0lPKF9fQVNITUVNSU9DLCAxMCkKLQotI2VuZGlmCS8qIF9VVElMU19BU0hNRU1fSCAqLwpkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9leGVjdXRhYmxlcGF0aC5oIGIvaW5jbHVkZS91dGlscy9leGVjdXRhYmxlcGF0aC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjOTc5NDMyLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvZXhlY3V0YWJsZXBhdGguaAorKysgL2Rldi9udWxsCkBAIC0xLDI4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIF9VVElMU19FWEVDVVRBQkxFUEFUSF9ICi0jZGVmaW5lIF9VVElMU19FWEVDVVRBQkxFUEFUSF9ICi0KLSNpbmNsdWRlIDxsaW1pdHMuaD4KLQotLy8gcmV0dXJucyB0aGUgcGF0aCB0byB0aGlzIGV4ZWN1dGFibGUKLSNpZiBfX2NwbHVzcGx1cwotZXh0ZXJuICJDIgotI2VuZGlmCi12b2lkIGV4ZWN1dGFibGVwYXRoKGNoYXIgc1tQQVRIX01BWF0pOwotCi0jZW5kaWYgLy8gX1VUSUxTX0VYRUNVVEFCTEVQQVRIX0gKZGlmZiAtLWdpdCBhL2luY2x1ZGUvdXRpbHMvaW5ldF9hZGRyZXNzLmggYi9pbmNsdWRlL3V0aWxzL2luZXRfYWRkcmVzcy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkYmQ4NjcyLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvaW5ldF9hZGRyZXNzLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxMDMgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBJbnRlcm5ldCBhZGRyZXNzIGNsYXNzZXMuICBNb2RlbGVkIGFmdGVyIEphdmEgY2xhc3Nlcy4KLS8vCi0jaWZuZGVmIF9SVU5USU1FX0lORVRfQUREUkVTU19ICi0jZGVmaW5lIF9SVU5USU1FX0lORVRfQUREUkVTU19ICi0KLSNpZmRlZiBIQVZFX0FORFJPSURfT1MKLSNlcnJvciBETyBOT1QgVVNFIFRISVMgRklMRSBJTiBUSEUgREVWSUNFIEJVSUxECi0jZW5kaWYKLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8qCi0gKiBUaGlzIGNsYXNzIGhvbGRzIEludGVybmV0IGFkZHJlc3Nlcy4gIFBlcmhhcHMgbW9yZSB1c2VmdWwgaXMgaXRzCi0gKiBhYmlsaXR5IHRvIGxvb2sgdXAgYWRkcmVzc2VzIGJ5IG5hbWUuCi0gKgotICogSW52b2tlIG9uZSBvZiB0aGUgc3RhdGljIGZhY3RvcnkgbWV0aG9kcyB0byBjcmVhdGUgYSBuZXcgb2JqZWN0LgotICovCi1jbGFzcyBJbmV0QWRkcmVzcyB7Ci1wdWJsaWM6Ci0gICAgdmlydHVhbCB+SW5ldEFkZHJlc3Modm9pZCk7Ci0KLSAgICAvLyBjcmVhdGUgZnJvbSB3LngueS56IG9yIGZvby5iYXIuY29tIG5vdGF0aW9uCi0gICAgc3RhdGljIEluZXRBZGRyZXNzKiBnZXRCeU5hbWUoY29uc3QgY2hhciogaG9zdCk7Ci0KLSAgICAvLyBjb3B5LWNvbnN0cnVjdGlvbgotICAgIEluZXRBZGRyZXNzKGNvbnN0IEluZXRBZGRyZXNzJiBvcmlnKTsKLQotICAgIGNvbnN0IHZvaWQqIGdldEFkZHJlc3Modm9pZCkgY29uc3QgeyByZXR1cm4gbUFkZHJlc3M7IH0KLSAgICBpbnQgZ2V0QWRkcmVzc0xlbmd0aCh2b2lkKSBjb25zdCB7IHJldHVybiBtTGVuZ3RoOyB9Ci0gICAgY29uc3QgY2hhciogZ2V0SG9zdE5hbWUodm9pZCkgY29uc3QgeyByZXR1cm4gbU5hbWU7IH0KLQotcHJpdmF0ZToKLSAgICBJbmV0QWRkcmVzcyh2b2lkKTsKLSAgICAvLyBhc3NpZ25tZW50IChwcml2YXRlKQotICAgIEluZXRBZGRyZXNzJiBvcGVyYXRvcj0oY29uc3QgSW5ldEFkZHJlc3MmIGFkZHIpOwotCi0gICAgLy8gdXNlIGEgdm9pZCogaGVyZSBzbyB3ZSBkb24ndCBoYXZlIHRvIGV4cG9zZSBhY3R1YWwgc29ja2V0IGhlYWRlcnMKLSAgICB2b2lkKiAgICAgICBtQWRkcmVzczsgICAvLyB0aGlzIGlzIHJlYWxseSBhIHB0ciB0byBzb2NrYWRkcl9pbgotICAgIGludCAgICAgICAgIG1MZW5ndGg7Ci0gICAgY2hhciogICAgICAgbU5hbWU7Ci19OwotCi0KLS8qCi0gKiBCYXNlIGNsYXNzIGZvciBzb2NrZXQgYWRkcmVzc2VzLgotICovCi1jbGFzcyBTb2NrZXRBZGRyZXNzIHsKLXB1YmxpYzoKLSAgICBTb2NrZXRBZGRyZXNzKCkge30KLSAgICB2aXJ0dWFsIH5Tb2NrZXRBZGRyZXNzKCkge30KLX07Ci0KLQotLyoKLSAqIEludGVybmV0IGFkZHJlc3MgY2xhc3MuICBUaGlzIGNvbWJpbmVzIGFuIEluZXRBZGRyZXNzIHdpdGggYSBwb3J0LgotICovCi1jbGFzcyBJbmV0U29ja2V0QWRkcmVzcyA6IHB1YmxpYyBTb2NrZXRBZGRyZXNzIHsKLXB1YmxpYzoKLSAgICBJbmV0U29ja2V0QWRkcmVzcygpIDoKLSAgICAgICAgbUFkZHJlc3MoMCksIG1Qb3J0KC0xKQotICAgICAgICB7fQotICAgIH5JbmV0U29ja2V0QWRkcmVzcyh2b2lkKSB7Ci0gICAgICAgIGRlbGV0ZSBtQWRkcmVzczsKLSAgICB9Ci0KLSAgICAvLyBDcmVhdGUgYW4gYWRkcmVzcyB3aXRoIGEgaG9zdCB3aWxkY2FyZCAodXNlZnVsIGZvciBzZXJ2ZXJzKS4KLSAgICBib29sIGNyZWF0ZShpbnQgcG9ydCk7Ci0gICAgLy8gQ3JlYXRlIGFuIGFkZHJlc3Mgd2l0aCB0aGUgc3BlY2lmaWVkIGhvc3QgYW5kIHBvcnQuCi0gICAgYm9vbCBjcmVhdGUoY29uc3QgSW5ldEFkZHJlc3MqIGFkZHIsIGludCBwb3J0KTsKLSAgICAvLyBDcmVhdGUgYW4gYWRkcmVzcyB3aXRoIHRoZSBzcGVjaWZpZWQgaG9zdCBhbmQgcG9ydC4gIERvZXMgdGhlCi0gICAgLy8gaG9zdG5hbWUgbG9va3VwLgotICAgIGJvb2wgY3JlYXRlKGNvbnN0IGNoYXIqIGhvc3QsIGludCBwb3J0KTsKLQotICAgIGNvbnN0IEluZXRBZGRyZXNzKiBnZXRBZGRyZXNzKHZvaWQpIGNvbnN0IHsgcmV0dXJuIG1BZGRyZXNzOyB9Ci0gICAgY29uc3QgaW50IGdldFBvcnQodm9pZCkgY29uc3QgeyByZXR1cm4gbVBvcnQ7IH0KLSAgICBjb25zdCBjaGFyKiBnZXRIb3N0TmFtZSh2b2lkKSBjb25zdCB7IHJldHVybiBtQWRkcmVzcy0+Z2V0SG9zdE5hbWUoKTsgfQotCi1wcml2YXRlOgotICAgIEluZXRBZGRyZXNzKiBtQWRkcmVzczsKLSAgICBpbnQgICAgICAgICBtUG9ydDsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBfUlVOVElNRV9JTkVUX0FERFJFU1NfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9taXNjLmggYi9pbmNsdWRlL3V0aWxzL21pc2MuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjJlODRiNC4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL21pc2MuaAorKysgL2Rldi9udWxsCkBAIC0xLDkzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gSGFuZHkgdXRpbGl0eSBmdW5jdGlvbnMgYW5kIHBvcnRhYmlsaXR5IGNvZGUuCi0vLwotI2lmbmRlZiBfTElCU19VVElMU19NSVNDX0gKLSNkZWZpbmUgX0xJQlNfVVRJTFNfTUlTQ19ICi0KLSNpbmNsdWRlIDxzeXMvdGltZS5oPgotI2luY2x1ZGUgInV0aWxzL0VuZGlhbi5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8qIGdldCAjb2YgZWxlbWVudHMgaW4gYSBzdGF0aWMgYXJyYXkgKi8KLSNpZm5kZWYgTkVMRU0KLSMgZGVmaW5lIE5FTEVNKHgpICgoaW50KSAoc2l6ZW9mKHgpIC8gc2l6ZW9mKCh4KVswXSkpKQotI2VuZGlmCi0KLS8qCi0gKiBNYWtlIGEgY29weSBvZiB0aGUgc3RyaW5nLCB1c2luZyAibmV3W10iIGluc3RlYWQgb2YgIm1hbGxvYyIuICBGcmVlIHRoZQotICogc3RyaW5nIHdpdGggZGVsZXRlW10uCi0gKgotICogUmV0dXJucyBOVUxMIGlmICJzdHIiIGlzIE5VTEwuCi0gKi8KLWNoYXIqIHN0cmR1cE5ldyhjb25zdCBjaGFyKiBzdHIpOwotCi0vKgotICogQ29uY2F0ZW5hdGUgYW4gYXJndW1lbnQgdmVjdG9yIGludG8gYSBzaW5nbGUgc3RyaW5nLiAgSWYgYXJnYyBpcyA+PSAwCi0gKiBpdCB3aWxsIGJlIHVzZWQ7IGlmIGl0J3MgPCAwIHRoZW4gdGhlIGxhc3QgZWxlbWVudCBpbiB0aGUgYXJnIHZlY3RvcgotICogbXVzdCBiZSBOVUxMLgotICoKLSAqIFRoaXMgaW5zZXJ0cyBhIHNwYWNlIGJldHdlZW4gZWFjaCBhcmd1bWVudC4KLSAqCi0gKiBUaGlzIGRvZXMgbm90IGF1dG9tYXRpY2FsbHkgYWRkIGRvdWJsZSBxdW90ZXMgYXJvdW5kIGFyZ3VtZW50cyB3aXRoCi0gKiBzcGFjZXMgaW4gdGhlbS4gIFRoaXMgcHJhY3RpY2UgaXMgbmVjZXNzYXJ5IGZvciBXaW4zMiwgYmVjYXVzZSBXaW4zMidzCi0gKiBDcmVhdGVQcm9jZXNzIGNhbGwgaXMgc3R1cGlkLgotICoKLSAqIFRoZSBjYWxsZXIgc2hvdWxkIGRlbGV0ZVtdIHRoZSByZXR1cm5lZCBzdHJpbmcuCi0gKi8KLWNoYXIqIGNvbmNhdEFyZ3YoaW50IGFyZ2MsIGNvbnN0IGNoYXIqIGNvbnN0IGFyZ3ZbXSk7Ci0KLS8qCi0gKiBDb3VudCB1cCB0aGUgbnVtYmVyIG9mIGFyZ3VtZW50cyBpbiAiYXJndiIuICBUaGUgY291bnQgZG9lcyBub3QgaW5jbHVkZQotICogdGhlIGZpbmFsIE5VTEwgZW50cnkuCi0gKi8KLWludCBjb3VudEFyZ3YoY29uc3QgY2hhciogY29uc3QgYXJndltdKTsKLQotLyoKLSAqIFNvbWUgdXRpbGl0eSBmdW5jdGlvbnMgZm9yIHdvcmtpbmcgd2l0aCBmaWxlcy4gIFRoZXNlIGNvdWxkIGJlIG1hZGUKLSAqIHBhcnQgb2YgYSAiRmlsZSIgY2xhc3MuCi0gKi8KLXR5cGVkZWYgZW51bSBGaWxlVHlwZSB7Ci0gICAga0ZpbGVUeXBlVW5rbm93biA9IDAsCi0gICAga0ZpbGVUeXBlTm9uZXhpc3RlbnQsICAgICAgIC8vIGkuZS4gRU5PRU5UCi0gICAga0ZpbGVUeXBlUmVndWxhciwKLSAgICBrRmlsZVR5cGVEaXJlY3RvcnksCi0gICAga0ZpbGVUeXBlQ2hhckRldiwKLSAgICBrRmlsZVR5cGVCbG9ja0RldiwKLSAgICBrRmlsZVR5cGVGaWZvLAotICAgIGtGaWxlVHlwZVN5bWxpbmssCi0gICAga0ZpbGVUeXBlU29ja2V0LAotfSBGaWxlVHlwZTsKLS8qIGdldCB0aGUgZmlsZSdzIHR5cGU7IGZvbGxvd3Mgc3ltbGlua3MgKi8KLUZpbGVUeXBlIGdldEZpbGVUeXBlKGNvbnN0IGNoYXIqIGZpbGVOYW1lKTsKLS8qIGdldCB0aGUgZmlsZSdzIG1vZGlmaWNhdGlvbiBkYXRlOyByZXR1cm5zIC0xIHcvZXJybm8gc2V0IG9uIGZhaWx1cmUgKi8KLXRpbWVfdCBnZXRGaWxlTW9kRGF0ZShjb25zdCBjaGFyKiBmaWxlTmFtZSk7Ci0KLS8qCi0gKiBSb3VuZCB1cCB0byB0aGUgbmVhcmVzdCBwb3dlciBvZiAyLiAgSGFuZHkgZm9yIGhhc2ggdGFibGVzLgotICovCi11bnNpZ25lZCBpbnQgcm91bmRVcFBvd2VyMih1bnNpZ25lZCBpbnQgdmFsKTsKLQotdm9pZCBzdHJyZXZlcnNlKGNoYXIqIGJlZ2luLCBjaGFyKiBlbmQpOwotdm9pZCBrX2l0b2EoaW50IHZhbHVlLCBjaGFyKiBzdHIsIGludCBiYXNlKTsKLWNoYXIqIGl0b2EoaW50IHZhbCwgaW50IGJhc2UpOwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gX0xJQlNfVVRJTFNfTUlTQ19ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL3BvcnRlZC5oIGIvaW5jbHVkZS91dGlscy9wb3J0ZWQuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZWIzYmUwMS4uMDAwMDAwMAotLS0gYS9pbmNsdWRlL3V0aWxzL3BvcnRlZC5oCisrKyAvZGV2L251bGwKQEAgLTEsNTAgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBTdGFuZGFyZCBmdW5jdGlvbnMgcG9ydGVkIHRvIHRoZSBjdXJyZW50IHBsYXRmb3JtLiAgTm90ZSB0aGVzZSBhcmUgTk9UCi0vLyBpbiB0aGUgImFuZHJvaWQiIG5hbWVzcGFjZS4KLS8vCi0jaWZuZGVmIF9MSUJTX1VUSUxTX1BPUlRFRF9ICi0jZGVmaW5lIF9MSUJTX1VUSUxTX1BPUlRFRF9ICi0KLSNpbmNsdWRlIDxzeXMvdGltZS5oPiAgICAgICAvLyBmb3IgdGltZXZhbAotCi0jaWZkZWYgX19jcGx1c3BsdXMKLWV4dGVybiAiQyIgewotI2VuZGlmCi0KLS8qIGxpYnJhcnkgcmVwbGFjZW1lbnQgZnVuY3Rpb25zICovCi0jaWYgZGVmaW5lZChORUVEX0dFVFRJTUVPRkRBWSkKLWludCBnZXR0aW1lb2ZkYXkoc3RydWN0IHRpbWV2YWwqIHR2LCBzdHJ1Y3QgdGltZXpvbmUqIHR6KTsKLSNlbmRpZgotI2lmIGRlZmluZWQoTkVFRF9VU0xFRVApCi12b2lkIHVzbGVlcCh1bnNpZ25lZCBsb25nIHVzZWMpOwotI2VuZGlmCi0jaWYgZGVmaW5lZChORUVEX1BJUEUpCi1pbnQgcGlwZShpbnQgZmlsZWRlc1syXSk7Ci0jZW5kaWYKLSNpZiBkZWZpbmVkKE5FRURfU0VURU5WKQotaW50IHNldGVudihjb25zdCBjaGFyKiBuYW1lLCBjb25zdCBjaGFyKiB2YWx1ZSwgaW50IG92ZXJ3cml0ZSk7Ci12b2lkIHVuc2V0ZW52KGNvbnN0IGNoYXIqIG5hbWUpOwotY2hhciogZ2V0ZW52KGNvbnN0IGNoYXIqIG5hbWUpOwotI2VuZGlmCi0KLSNpZmRlZiBfX2NwbHVzcGx1cwotfQotI2VuZGlmCi0KLSNlbmRpZiAvLyBfTElCU19VVElMU19QT1JURURfSApkaWZmIC0tZ2l0IGEvaW5jbHVkZS91dGlscy9zdHJpbmdfYXJyYXkuaCBiL2luY2x1ZGUvdXRpbHMvc3RyaW5nX2FycmF5LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDA2NGRkYTIuLjAwMDAwMDAKLS0tIGEvaW5jbHVkZS91dGlscy9zdHJpbmdfYXJyYXkuaAorKysgL2Rldi9udWxsCkBAIC0xLDEzNSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIFNvcnRhYmxlIGFycmF5IG9mIHN0cmluZ3MuICBTVEwtaXNoLCBidXQgU1RMLWZyZWUuCi0vLyAgCi0jaWZuZGVmIF9MSUJTX1VUSUxTX1NUUklOR19BUlJBWV9ICi0jZGVmaW5lIF9MSUJTX1VUSUxTX1NUUklOR19BUlJBWV9ICi0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLwotLy8gQW4gZXhwYW5kaW5nIGFycmF5IG9mIHN0cmluZ3MuICBBZGQsIGdldCwgc29ydCwgZGVsZXRlLgotLy8KLWNsYXNzIFN0cmluZ0FycmF5IHsKLXB1YmxpYzoKLSAgICBTdHJpbmdBcnJheSgpCi0gICAgICAgIDogbU1heCgwKSwgbUN1cnJlbnQoMCksIG1BcnJheShOVUxMKQotICAgICAgICB7fQotICAgIHZpcnR1YWwgflN0cmluZ0FycmF5KCkgewotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG1DdXJyZW50OyBpKyspCi0gICAgICAgICAgICBkZWxldGVbXSBtQXJyYXlbaV07Ci0gICAgICAgIGRlbGV0ZVtdIG1BcnJheTsKLSAgICB9Ci0KLSAgICAvLwotICAgIC8vIEFkZCBhIHN0cmluZy4gIEEgY29weSBvZiB0aGUgc3RyaW5nIGlzIG1hZGUuCi0gICAgLy8KLSAgICBib29sIHB1c2hfYmFjayhjb25zdCBjaGFyKiBzdHIpIHsKLSAgICAgICAgaWYgKG1DdXJyZW50ID49IG1NYXgpIHsKLSAgICAgICAgICAgIGNoYXIqKiB0bXA7Ci0KLSAgICAgICAgICAgIGlmIChtTWF4ID09IDApCi0gICAgICAgICAgICAgICAgbU1heCA9IDE2OyAgICAgIC8vIGluaXRpYWwgc3RvcmFnZQotICAgICAgICAgICAgZWxzZQotICAgICAgICAgICAgICAgIG1NYXggKj0gMjsKLQotICAgICAgICAgICAgdG1wID0gbmV3IGNoYXIqW21NYXhdOwotICAgICAgICAgICAgaWYgKHRtcCA9PSBOVUxMKQotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLQotICAgICAgICAgICAgbWVtY3B5KHRtcCwgbUFycmF5LCBtQ3VycmVudCAqIHNpemVvZihjaGFyKikpOwotICAgICAgICAgICAgZGVsZXRlW10gbUFycmF5OwotICAgICAgICAgICAgbUFycmF5ID0gdG1wOwotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGxlbiA9IHN0cmxlbihzdHIpOwotICAgICAgICBtQXJyYXlbbUN1cnJlbnRdID0gbmV3IGNoYXJbbGVuKzFdOwotICAgICAgICBtZW1jcHkobUFycmF5W21DdXJyZW50XSwgc3RyLCBsZW4rMSk7Ci0gICAgICAgIG1DdXJyZW50Kys7Ci0KLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgLy8KLSAgICAvLyBEZWxldGUgYW4gZW50cnkuCi0gICAgLy8KLSAgICB2b2lkIGVyYXNlKGludCBpZHgpIHsKLSAgICAgICAgaWYgKGlkeCA8IDAgfHwgaWR4ID49IG1DdXJyZW50KQotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICBkZWxldGVbXSBtQXJyYXlbaWR4XTsKLSAgICAgICAgaWYgKGlkeCA8IG1DdXJyZW50LTEpIHsKLSAgICAgICAgICAgIG1lbW1vdmUoJm1BcnJheVtpZHhdLCAmbUFycmF5W2lkeCsxXSwKLSAgICAgICAgICAgICAgICAobUN1cnJlbnQtMSAtIGlkeCkgKiBzaXplb2YoY2hhciopKTsKLSAgICAgICAgfQotICAgICAgICBtQ3VycmVudC0tOwotICAgIH0KLQotICAgIC8vCi0gICAgLy8gU29ydCB0aGUgYXJyYXkuCi0gICAgLy8KLSAgICB2b2lkIHNvcnQoaW50ICgqY29tcGFyZSkoY29uc3Qgdm9pZCosIGNvbnN0IHZvaWQqKSkgewotICAgICAgICBxc29ydChtQXJyYXksIG1DdXJyZW50LCBzaXplb2YoY2hhciopLCBjb21wYXJlKTsKLSAgICB9Ci0KLSAgICAvLwotICAgIC8vIFBhc3MgdGhpcyB0byB0aGUgc29ydCByb3V0aW5lIHRvIGRvIGFuIGFzY2VuZGluZyBhbHBoYWJldGljYWwgc29ydC4KLSAgICAvLwotICAgIHN0YXRpYyBpbnQgY21wQXNjZW5kaW5nQWxwaGEoY29uc3Qgdm9pZCogcHN0cjEsIGNvbnN0IHZvaWQqIHBzdHIyKSB7Ci0gICAgICAgIHJldHVybiBzdHJjbXAoKihjb25zdCBjaGFyKiopcHN0cjEsICooY29uc3QgY2hhcioqKXBzdHIyKTsKLSAgICB9Ci0KLSAgICAvLwotICAgIC8vIEdldCB0aGUgI29mIGl0ZW1zIGluIHRoZSBhcnJheS4KLSAgICAvLwotICAgIGlubGluZSBpbnQgc2l6ZSh2b2lkKSBjb25zdCB7IHJldHVybiBtQ3VycmVudDsgfQotCi0gICAgLy8KLSAgICAvLyBSZXR1cm4gZW50cnkgTi4KLSAgICAvLyBbc2hvdWxkIHVzZSBvcGVyYXRvcltdIGhlcmVdCi0gICAgLy8KLSAgICBjb25zdCBjaGFyKiBnZXRFbnRyeShpbnQgaWR4KSBjb25zdCB7Ci0gICAgICAgIGlmIChpZHggPCAwIHx8IGlkeCA+PSBtQ3VycmVudCkKLSAgICAgICAgICAgIHJldHVybiBOVUxMOwotICAgICAgICByZXR1cm4gbUFycmF5W2lkeF07Ci0gICAgfQotCi0gICAgLy8KLSAgICAvLyBTZXQgZW50cnkgTiB0byBzcGVjaWZpZWQgc3RyaW5nLgotICAgIC8vIFtzaG91bGQgdXNlIG9wZXJhdG9yW10gaGVyZV0KLSAgICAvLwotICAgIHZvaWQgc2V0RW50cnkoaW50IGlkeCwgY29uc3QgY2hhciogc3RyKSB7Ci0gICAgICAgIGlmIChpZHggPCAwIHx8IGlkeCA+PSBtQ3VycmVudCkKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgZGVsZXRlW10gbUFycmF5W2lkeF07Ci0gICAgICAgIGludCBsZW4gPSBzdHJsZW4oc3RyKTsKLSAgICAgICAgbUFycmF5W2lkeF0gPSBuZXcgY2hhcltsZW4rMV07Ci0gICAgICAgIG1lbWNweShtQXJyYXlbaWR4XSwgc3RyLCBsZW4rMSk7Ci0gICAgfQotCi1wcml2YXRlOgotICAgIGludCAgICAgbU1heDsKLSAgICBpbnQgICAgIG1DdXJyZW50OwotICAgIGNoYXIqKiAgbUFycmF5OwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIF9MSUJTX1VUSUxTX1NUUklOR19BUlJBWV9ICmRpZmYgLS1naXQgYS9pbmNsdWRlL3V0aWxzL3RocmVhZHMuaCBiL2luY2x1ZGUvdXRpbHMvdGhyZWFkcy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3ZGNhODEwLi4wMDAwMDAwCi0tLSBhL2luY2x1ZGUvdXRpbHMvdGhyZWFkcy5oCisrKyAvZGV2L251bGwKQEAgLTEsMzQ3ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIF9MSUJTX1VUSUxTX1RIUkVBRFNfSAotI2RlZmluZSBfTElCU19VVElMU19USFJFQURTX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHRpbWUuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBDIEFQSQotCi0jaWZkZWYgX19jcGx1c3BsdXMKLWV4dGVybiAiQyIgewotI2VuZGlmCi0KLXR5cGVkZWYgdm9pZCogYW5kcm9pZF90aHJlYWRfaWRfdDsKLQotdHlwZWRlZiBpbnQgKCphbmRyb2lkX3RocmVhZF9mdW5jX3QpKHZvaWQqKTsKLQotZW51bSB7Ci0gICAgLyoKLSAgICAgKiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgotICAgICAqICoqIEtlZXAgaW4gc3luYyB3aXRoIGFuZHJvaWQub3MuUHJvY2Vzcy5qYXZhICoqCi0gICAgICogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKLSAgICAgKiAKLSAgICAgKiBUaGlzIG1hcHMgZGlyZWN0bHkgdG8gdGhlICJuaWNlIiBwcmlvcml0ZXMgd2UgdXNlIGluIEFuZHJvaWQuCi0gICAgICogQSB0aHJlYWQgcHJpb3JpdHkgc2hvdWxkIGJlIGNob3NlbiBpbnZlcnNlLXByb3BvcnRpbmFsbHkgdG8KLSAgICAgKiB0aGUgYW1vdW50IG9mIHdvcmsgdGhlIHRocmVhZCBpcyBleHBlY3RlZCB0byBkby4gVGhlIG1vcmUgd29yawotICAgICAqIGEgdGhyZWFkIHdpbGwgZG8sIHRoZSBsZXNzIGZhdm9yYWJsZSBwcmlvcml0eSBpdCBzaG91bGQgZ2V0IHNvIHRoYXQgCi0gICAgICogaXQgZG9lc24ndCBzdGFydmUgdGhlIHN5c3RlbS4gVGhyZWFkcyBub3QgYmVoYXZpbmcgcHJvcGVybHkgbWlnaHQKLSAgICAgKiBiZSAicHVuaXNoZWQiIGJ5IHRoZSBrZXJuZWwuCi0gICAgICogVXNlIHRoZSBsZXZlbHMgYmVsb3cgd2hlbiBhcHByb3ByaWF0ZS4gSW50ZXJtZWRpYXRlIHZhbHVlcyBhcmUKLSAgICAgKiBhY2NlcHRhYmxlLCBwcmVmZXJhYmx5IHVzZSB0aGUge01PUkV8TEVTU31fRkFWT1JBQkxFIGNvbnN0YW50cyBiZWxvdy4KLSAgICAgKi8KLSAgICBBTkRST0lEX1BSSU9SSVRZX0xPV0VTVCAgICAgICAgID0gIDE5LAotCi0gICAgLyogdXNlIGZvciBiYWNrZ3JvdW5kIHRhc2tzICovCi0gICAgQU5EUk9JRF9QUklPUklUWV9CQUNLR1JPVU5EICAgICA9ICAxMCwKLSAgICAKLSAgICAvKiBtb3N0IHRocmVhZHMgcnVuIGF0IG5vcm1hbCBwcmlvcml0eSAqLwotICAgIEFORFJPSURfUFJJT1JJVFlfTk9STUFMICAgICAgICAgPSAgIDAsCi0gICAgCi0gICAgLyogdGhyZWFkcyBjdXJyZW50bHkgcnVubmluZyBhIFVJIHRoYXQgdGhlIHVzZXIgaXMgaW50ZXJhY3Rpbmcgd2l0aCAqLwotICAgIEFORFJPSURfUFJJT1JJVFlfRk9SRUdST1VORCAgICAgPSAgLTIsCi0KLSAgICAvKiB0aGUgbWFpbiBVSSB0aHJlYWQgaGFzIGEgc2xpZ2h0bHkgbW9yZSBmYXZvcmFibGUgcHJpb3JpdHkgKi8KLSAgICBBTkRST0lEX1BSSU9SSVRZX0RJU1BMQVkgICAgICAgID0gIC00LAotICAgIAotICAgIC8qIHVpIHNlcnZpY2UgdHJlYWRzIG1pZ2h0IHdhbnQgdG8gcnVuIGF0IGEgdXJnZW50IGRpc3BsYXkgKHVuY29tbW9uKSAqLwotICAgIEFORFJPSURfUFJJT1JJVFlfVVJHRU5UX0RJU1BMQVkgPSAgLTgsCi0gICAgCi0gICAgLyogYWxsIG5vcm1hbCBhdWRpbyB0aHJlYWRzICovCi0gICAgQU5EUk9JRF9QUklPUklUWV9BVURJTyAgICAgICAgICA9IC0xNiwKLSAgICAKLSAgICAvKiBzZXJ2aWNlIGF1ZGlvIHRocmVhZHMgKHVuY29tbW9uKSAqLwotICAgIEFORFJPSURfUFJJT1JJVFlfVVJHRU5UX0FVRElPICAgPSAtMTksCi0KLSAgICAvKiBzaG91bGQgbmV2ZXIgYmUgdXNlZCBpbiBwcmFjdGljZS4gcmVndWxhciBwcm9jZXNzIG1pZ2h0IG5vdCAKLSAgICAgKiBiZSBhbGxvd2VkIHRvIHVzZSB0aGlzIGxldmVsICovCi0gICAgQU5EUk9JRF9QUklPUklUWV9ISUdIRVNUICAgICAgICA9IC0yMCwKLQotICAgIEFORFJPSURfUFJJT1JJVFlfREVGQVVMVCAgICAgICAgPSBBTkRST0lEX1BSSU9SSVRZX05PUk1BTCwKLSAgICBBTkRST0lEX1BSSU9SSVRZX01PUkVfRkFWT1JBQkxFID0gLTEsCi0gICAgQU5EUk9JRF9QUklPUklUWV9MRVNTX0ZBVk9SQUJMRSA9ICsxLAotfTsKLQotLy8gQ3JlYXRlIGFuZCBydW4gYSBuZXcgdGhyZWFkLgotZXh0ZXJuIGludCBhbmRyb2lkQ3JlYXRlVGhyZWFkKGFuZHJvaWRfdGhyZWFkX2Z1bmNfdCwgdm9pZCAqKTsKLQotLy8gQ3JlYXRlIHRocmVhZCB3aXRoIGxvdHMgb2YgcGFyYW1ldGVycwotZXh0ZXJuIGludCBhbmRyb2lkQ3JlYXRlVGhyZWFkRXRjKGFuZHJvaWRfdGhyZWFkX2Z1bmNfdCBlbnRyeUZ1bmN0aW9uLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQgKnVzZXJEYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNoYXIqIHRocmVhZE5hbWUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCB0aHJlYWRQcmlvcml0eSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplX3QgdGhyZWFkU3RhY2tTaXplLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFuZHJvaWRfdGhyZWFkX2lkX3QgKnRocmVhZElkKTsKLQotLy8gR2V0IHNvbWUgc29ydCBvZiB1bmlxdWUgaWRlbnRpZmllciBmb3IgdGhlIGN1cnJlbnQgdGhyZWFkLgotZXh0ZXJuIGFuZHJvaWRfdGhyZWFkX2lkX3QgYW5kcm9pZEdldFRocmVhZElkKCk7Ci0KLS8vIExvdy1sZXZlbCB0aHJlYWQgY3JlYXRpb24gLS0gbmV2ZXIgY3JlYXRlcyB0aHJlYWRzIHRoYXQgY2FuCi0vLyBpbnRlcmFjdCB3aXRoIHRoZSBKYXZhIFZNLgotZXh0ZXJuIGludCBhbmRyb2lkQ3JlYXRlUmF3VGhyZWFkRXRjKGFuZHJvaWRfdGhyZWFkX2Z1bmNfdCBlbnRyeUZ1bmN0aW9uLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQgKnVzZXJEYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNoYXIqIHRocmVhZE5hbWUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCB0aHJlYWRQcmlvcml0eSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplX3QgdGhyZWFkU3RhY2tTaXplLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFuZHJvaWRfdGhyZWFkX2lkX3QgKnRocmVhZElkKTsKLQotLy8gVXNlZCBieSB0aGUgSmF2YSBSdW50aW1lIHRvIGNvbnRyb2wgaG93IHRocmVhZHMgYXJlIGNyZWF0ZWQsIHNvIHRoYXQKLS8vIHRoZXkgY2FuIGJlIHByb3BlciBhbmQgbG92ZWx5IEphdmEgdGhyZWFkcy4KLXR5cGVkZWYgaW50ICgqYW5kcm9pZF9jcmVhdGVfdGhyZWFkX2ZuKShhbmRyb2lkX3RocmVhZF9mdW5jX3QgZW50cnlGdW5jdGlvbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkICp1c2VyRGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyKiB0aHJlYWROYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgdGhyZWFkUHJpb3JpdHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IHRocmVhZFN0YWNrU2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhbmRyb2lkX3RocmVhZF9pZF90ICp0aHJlYWRJZCk7Ci0KLWV4dGVybiB2b2lkIGFuZHJvaWRTZXRDcmVhdGVUaHJlYWRGdW5jKGFuZHJvaWRfY3JlYXRlX3RocmVhZF9mbiBmdW5jKTsKLQotI2lmZGVmIF9fY3BsdXNwbHVzCi19Ci0jZW5kaWYKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBDKysgQVBJCi0KLSNpZmRlZiBfX2NwbHVzcGx1cwotCi0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUmVmQmFzZS5oPgotI2luY2x1ZGUgPHV0aWxzL1RpbWVycy5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLXR5cGVkZWYgYW5kcm9pZF90aHJlYWRfaWRfdCB0aHJlYWRfaWRfdDsKLQotdHlwZWRlZiBhbmRyb2lkX3RocmVhZF9mdW5jX3QgdGhyZWFkX2Z1bmNfdDsKLQotZW51bSB7Ci0gICAgUFJJT1JJVFlfTE9XRVNUICAgICAgICAgPSBBTkRST0lEX1BSSU9SSVRZX0xPV0VTVCwKLSAgICBQUklPUklUWV9CQUNLR1JPVU5EICAgICA9IEFORFJPSURfUFJJT1JJVFlfQkFDS0dST1VORCwKLSAgICBQUklPUklUWV9OT1JNQUwgICAgICAgICA9IEFORFJPSURfUFJJT1JJVFlfTk9STUFMLAotICAgIFBSSU9SSVRZX0ZPUkVHUk9VTkQgICAgID0gQU5EUk9JRF9QUklPUklUWV9GT1JFR1JPVU5ELAotICAgIFBSSU9SSVRZX0RJU1BMQVkgICAgICAgID0gQU5EUk9JRF9QUklPUklUWV9ESVNQTEFZLAotICAgIFBSSU9SSVRZX1VSR0VOVF9ESVNQTEFZID0gQU5EUk9JRF9QUklPUklUWV9VUkdFTlRfRElTUExBWSwKLSAgICBQUklPUklUWV9BVURJTyAgICAgICAgICA9IEFORFJPSURfUFJJT1JJVFlfQVVESU8sCi0gICAgUFJJT1JJVFlfVVJHRU5UX0FVRElPICAgPSBBTkRST0lEX1BSSU9SSVRZX1VSR0VOVF9BVURJTywKLSAgICBQUklPUklUWV9ISUdIRVNUICAgICAgICA9IEFORFJPSURfUFJJT1JJVFlfSElHSEVTVCwKLSAgICBQUklPUklUWV9ERUZBVUxUICAgICAgICA9IEFORFJPSURfUFJJT1JJVFlfREVGQVVMVCwKLSAgICBQUklPUklUWV9NT1JFX0ZBVk9SQUJMRSA9IEFORFJPSURfUFJJT1JJVFlfTU9SRV9GQVZPUkFCTEUsCi0gICAgUFJJT1JJVFlfTEVTU19GQVZPUkFCTEUgPSBBTkRST0lEX1BSSU9SSVRZX0xFU1NfRkFWT1JBQkxFLAotfTsKLQotLy8gQ3JlYXRlIGFuZCBydW4gYSBuZXcgdGhyZWFkLgotaW5saW5lIGJvb2wgY3JlYXRlVGhyZWFkKHRocmVhZF9mdW5jX3QgZiwgdm9pZCAqYSkgewotICAgIHJldHVybiBhbmRyb2lkQ3JlYXRlVGhyZWFkKGYsIGEpID8gdHJ1ZSA6IGZhbHNlOwotfQotCi0vLyBDcmVhdGUgdGhyZWFkIHdpdGggbG90cyBvZiBwYXJhbWV0ZXJzCi1pbmxpbmUgYm9vbCBjcmVhdGVUaHJlYWRFdGModGhyZWFkX2Z1bmNfdCBlbnRyeUZ1bmN0aW9uLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQgKnVzZXJEYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNoYXIqIHRocmVhZE5hbWUgPSAiYW5kcm9pZDp1bm5hbWVkX3RocmVhZCIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCB0aHJlYWRQcmlvcml0eSA9IFBSSU9SSVRZX0RFRkFVTFQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IHRocmVhZFN0YWNrU2l6ZSA9IDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhyZWFkX2lkX3QgKnRocmVhZElkID0gMCkKLXsKLSAgICByZXR1cm4gYW5kcm9pZENyZWF0ZVRocmVhZEV0YyhlbnRyeUZ1bmN0aW9uLCB1c2VyRGF0YSwgdGhyZWFkTmFtZSwKLSAgICAgICAgdGhyZWFkUHJpb3JpdHksIHRocmVhZFN0YWNrU2l6ZSwgdGhyZWFkSWQpID8gdHJ1ZSA6IGZhbHNlOwotfQotCi0vLyBHZXQgc29tZSBzb3J0IG9mIHVuaXF1ZSBpZGVudGlmaWVyIGZvciB0aGUgY3VycmVudCB0aHJlYWQuCi1pbmxpbmUgdGhyZWFkX2lkX3QgZ2V0VGhyZWFkSWQoKSB7Ci0gICAgcmV0dXJuIGFuZHJvaWRHZXRUaHJlYWRJZCgpOwotfQotCi0vKgotICogU2ltcGxlIG11dGV4IGNsYXNzLiAgVGhlIGltcGxlbWVudGF0aW9uIGlzIHN5c3RlbS1kZXBlbmRlbnQuCi0gKgotICogVGhlIG11dGV4IG11c3QgYmUgdW5sb2NrZWQgYnkgdGhlIHRocmVhZCB0aGF0IGxvY2tlZCBpdC4gIFRoZXkgYXJlIG5vdAotICogcmVjdXJzaXZlLCBpLmUuIHRoZSBzYW1lIHRocmVhZCBjYW4ndCBsb2NrIGl0IG11bHRpcGxlIHRpbWVzLgotICovCi1jbGFzcyBNdXRleCB7Ci1wdWJsaWM6Ci0gICAgICAgICAgICAgICAgTXV0ZXgoKTsKLSAgICAgICAgICAgICAgICBNdXRleChjb25zdCBjaGFyKiBuYW1lKTsKLSAgICAgICAgICAgICAgICB+TXV0ZXgoKTsKLQotICAgIC8vIGxvY2sgb3IgdW5sb2NrIHRoZSBtdXRleAotICAgIHN0YXR1c190ICAgIGxvY2soKTsKLSAgICB2b2lkICAgICAgICB1bmxvY2soKTsKLQotICAgIC8vIGxvY2sgaWYgcG9zc2libGU7IHJldHVybnMgMCBvbiBzdWNjZXNzLCBlcnJvciBvdGhlcndpc2UKLSAgICBzdGF0dXNfdCAgICB0cnlMb2NrKCk7Ci0KLSAgICAvLyBNYW5hZ2VzIHRoZSBtdXRleCBhdXRvbWF0aWNhbGx5LiBJdCdsbCBiZSBsb2NrZWQgd2hlbiBBdXRvbG9jayBpcwotICAgIC8vIGNvbnN0cnVjdGVkIGFuZCByZWxlYXNlZCB3aGVuIEF1dG9sb2NrIGdvZXMgb3V0IG9mIHNjb3BlLgotICAgIGNsYXNzIEF1dG9sb2NrIHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIGlubGluZSBBdXRvbG9jayhNdXRleCYgbXV0ZXgpIDogbXBNdXRleCgmbXV0ZXgpIHsgbXV0ZXgubG9jaygpOyB9Ci0gICAgICAgIGlubGluZSBBdXRvbG9jayhNdXRleCogbXV0ZXgpIDogbXBNdXRleChtdXRleCkgeyBtdXRleC0+bG9jaygpOyB9Ci0gICAgICAgIGlubGluZSB+QXV0b2xvY2soKSB7IG1wTXV0ZXgtPnVubG9jaygpOyB9Ci0gICAgcHJpdmF0ZToKLSAgICAgICAgTXV0ZXgqICBtcE11dGV4OwotICAgIH07Ci0KLXByaXZhdGU6Ci0gICAgZnJpZW5kIGNsYXNzIENvbmRpdGlvbjsKLSAgICAKLSAgICAvLyBBIG11dGV4IGNhbm5vdCBiZSBjb3BpZWQKLSAgICAgICAgICAgICAgICBNdXRleChjb25zdCBNdXRleCYpOwotICAgIE11dGV4JiAgICAgIG9wZXJhdG9yID0gKGNvbnN0IE11dGV4Jik7Ci0gICAgdm9pZCAgICAgICAgX2luaXQoKTsKLSAgICAKLSAgICB2b2lkKiAgIG1TdGF0ZTsKLX07Ci0KLS8qCi0gKiBBdXRvbWF0aWMgbXV0ZXguICBEZWNsYXJlIG9uZSBvZiB0aGVzZSBhdCB0aGUgdG9wIG9mIGEgZnVuY3Rpb24uCi0gKiBXaGVuIHRoZSBmdW5jdGlvbiByZXR1cm5zLCBpdCB3aWxsIGdvIG91dCBvZiBzY29wZSwgYW5kIHJlbGVhc2UgdGhlCi0gKiBtdXRleC4KLSAqLwotIAotdHlwZWRlZiBNdXRleDo6QXV0b2xvY2sgQXV0b011dGV4OwotCi0KLS8qCi0gKiBDb25kaXRpb24gdmFyaWFibGUgY2xhc3MuICBUaGUgaW1wbGVtZW50YXRpb24gaXMgc3lzdGVtLWRlcGVuZGVudC4KLSAqCi0gKiBDb25kaXRpb24gdmFyaWFibGVzIGFyZSBwYWlyZWQgdXAgd2l0aCBtdXRleGVzLiAgTG9jayB0aGUgbXV0ZXgsCi0gKiBjYWxsIHdhaXQoKSwgdGhlbiBlaXRoZXIgcmUtd2FpdCgpIGlmIHRoaW5ncyBhcmVuJ3QgcXVpdGUgd2hhdCB5b3Ugd2FudCwKLSAqIG9yIHVubG9jayB0aGUgbXV0ZXggYW5kIGNvbnRpbnVlLiAgQWxsIHRocmVhZHMgY2FsbGluZyB3YWl0KCkgbXVzdAotICogdXNlIHRoZSBzYW1lIG11dGV4IGZvciBhIGdpdmVuIENvbmRpdGlvbi4KLSAqLwotY2xhc3MgQ29uZGl0aW9uIHsKLXB1YmxpYzoKLSAgICBDb25kaXRpb24oKTsKLSAgICB+Q29uZGl0aW9uKCk7Ci0gICAgLy8gV2FpdCBvbiB0aGUgY29uZGl0aW9uIHZhcmlhYmxlLiAgTG9jayB0aGUgbXV0ZXggYmVmb3JlIGNhbGxpbmcuCi0gICAgc3RhdHVzX3Qgd2FpdChNdXRleCYgbXV0ZXgpOwotICAgIC8vIFdhaXQgb24gdGhlIGNvbmRpdGlvbiB2YXJpYWJsZSB1bnRpbCB0aGUgZ2l2ZW4gdGltZS4gIExvY2sgdGhlIG11dGV4Ci0gICAgLy8gYmVmb3JlIGNhbGxpbmcuCi0gICAgc3RhdHVzX3Qgd2FpdChNdXRleCYgbXV0ZXgsIG5zZWNzX3QgYWJzdGltZSk7Ci0gICAgLy8gc2FtZSB3aXRoIHJlbGF0aXZlIHRpbWVvdXQKLSAgICBzdGF0dXNfdCB3YWl0UmVsYXRpdmUoTXV0ZXgmIG11dGV4LCBuc2Vjc190IHJlbHRpbWUpOwotICAgIC8vIFNpZ25hbCB0aGUgY29uZGl0aW9uIHZhcmlhYmxlLCBhbGxvd2luZyBvbmUgdGhyZWFkIHRvIGNvbnRpbnVlLgotICAgIHZvaWQgc2lnbmFsKCk7Ci0gICAgLy8gU2lnbmFsIHRoZSBjb25kaXRpb24gdmFyaWFibGUsIGFsbG93aW5nIGFsbCB0aHJlYWRzIHRvIGNvbnRpbnVlLgotICAgIHZvaWQgYnJvYWRjYXN0KCk7Ci0KLXByaXZhdGU6Ci0gICAgdm9pZCogICBtU3RhdGU7Ci19OwotCi0KLS8qCi0gKiBSZWFkL3dyaXRlIGxvY2suICBUaGUgcmVzb3VyY2UgY2FuIGhhdmUgbXVsdGlwbGUgcmVhZGVycyBvciBvbmUgd3JpdGVyLAotICogYnV0IGNhbid0IGJlIHJlYWQgYW5kIHdyaXR0ZW4gYXQgdGhlIHNhbWUgdGltZS4KLSAqCi0gKiBUaGUgc2FtZSB0aHJlYWQgc2hvdWxkIG5vdCBjYWxsIGEgbG9jayBmdW5jdGlvbiB3aGlsZSBpdCBhbHJlYWR5IGhhcwotICogYSBsb2NrLiAgKFNob3VsZCBiZSBva2F5IGZvciBtdWx0aXBsZSByZWFkZXJzLikKLSAqLwotY2xhc3MgUmVhZFdyaXRlTG9jayB7Ci1wdWJsaWM6Ci0gICAgUmVhZFdyaXRlTG9jaygpCi0gICAgICAgIDogbU51bVJlYWRlcnMoMCksIG1OdW1Xcml0ZXJzKDApCi0gICAgICAgIHt9Ci0gICAgflJlYWRXcml0ZUxvY2soKSB7fQotCi0gICAgdm9pZCBsb2NrRm9yUmVhZCgpOwotICAgIGJvb2wgdHJ5TG9ja0ZvclJlYWQoKTsKLSAgICB2b2lkIHVubG9ja0ZvclJlYWQoKTsKLQotICAgIHZvaWQgbG9ja0ZvcldyaXRlKCk7Ci0gICAgYm9vbCB0cnlMb2NrRm9yV3JpdGUoKTsKLSAgICB2b2lkIHVubG9ja0ZvcldyaXRlKCk7Ci0KLXByaXZhdGU6Ci0gICAgaW50ICAgICAgICAgbU51bVJlYWRlcnM7Ci0gICAgaW50ICAgICAgICAgbU51bVdyaXRlcnM7Ci0KLSAgICBNdXRleCAgICAgICBtTG9jazsKLSAgICBDb25kaXRpb24gICBtUmVhZFdhaXRlcjsKLSAgICBDb25kaXRpb24gICBtV3JpdGVXYWl0ZXI7Ci0jaWYgZGVmaW5lZChQUklOVF9SRU5ERVJfVElNRVMpCi0gICAgRHVyYXRpb25UaW1lciBtRGVidWdUaW1lcjsKLSNlbmRpZgotfTsKLQotCi0vKgotICogVGhpcyBpcyBvdXIgc3BpZmZ5IHRocmVhZCBvYmplY3QhCi0gKi8KLQotY2xhc3MgVGhyZWFkIDogdmlydHVhbCBwdWJsaWMgUmVmQmFzZQotewotcHVibGljOgotICAgIC8vIENyZWF0ZSBhIFRocmVhZCBvYmplY3QsIGJ1dCBkb2Vzbid0IGNyZWF0ZSBvciBzdGFydCB0aGUgYXNzb2NpYXRlZAotICAgIC8vIHRocmVhZC4gU2VlIHRoZSBydW4oKSBtZXRob2QuCi0gICAgICAgICAgICAgICAgICAgICAgICBUaHJlYWQoYm9vbCBjYW5DYWxsSmF2YSA9IHRydWUpOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgflRocmVhZCgpOwotCi0gICAgLy8gU3RhcnQgdGhlIHRocmVhZCBpbiB0aHJlYWRMb29wKCkgd2hpY2ggbmVlZHMgdG8gYmUgaW1wbGVtZW50ZWQuCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBydW4oICAgIGNvbnN0IGNoYXIqIG5hbWUgPSAwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQzMl90IHByaW9yaXR5ID0gUFJJT1JJVFlfREVGQVVMVCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IHN0YWNrID0gMCk7Ci0gICAgCi0gICAgLy8gQXNrIHRoaXMgb2JqZWN0J3MgdGhyZWFkIHRvIGV4aXQuIFRoaXMgZnVuY3Rpb24gaXMgYXN5bmNocm9ub3VzLCB3aGVuIHRoZQotICAgIC8vIGZ1bmN0aW9uIHJldHVybnMgdGhlIHRocmVhZCBtaWdodCBzdGlsbCBiZSBydW5uaW5nLiBPZiBjb3Vyc2UsIHRoaXMKLSAgICAvLyBmdW5jdGlvbiBjYW4gYmUgY2FsbGVkIGZyb20gYSBkaWZmZXJlbnQgdGhyZWFkLgotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgcmVxdWVzdEV4aXQoKTsKLQotICAgIC8vIEdvb2QgcGxhY2UgdG8gZG8gb25lLXRpbWUgaW5pdGlhbGl6YXRpb25zCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICByZWFkeVRvUnVuKCk7Ci0gICAgCi0gICAgLy8gQ2FsbCByZXF1ZXN0RXhpdCgpIGFuZCB3YWl0IHVudGlsIHRoaXMgb2JqZWN0J3MgdGhyZWFkIGV4aXRzLgotICAgIC8vIEJFIFZFUlkgQ0FSRUZVTCBvZiBkZWFkbG9ja3MuIEluIHBhcnRpY3VsYXIsIGl0IHdvdWxkIGJlIHNpbGx5IHRvIGNhbGwKLSAgICAvLyB0aGlzIGZ1bmN0aW9uIGZyb20gdGhpcyBvYmplY3QncyB0aHJlYWQuIFdpbGwgcmV0dXJuIFdPVUxEX0JMT0NLIGluCi0gICAgLy8gdGhhdCBjYXNlLgotICAgICAgICAgICAgc3RhdHVzX3QgICAgcmVxdWVzdEV4aXRBbmRXYWl0KCk7Ci0KLXByb3RlY3RlZDoKLSAgICAvLyBleGl0UGVuZGluZygpIHJldHVybnMgdHJ1ZSBpZiByZXF1ZXN0RXhpdCgpIGhhcyBiZWVuIGNhbGxlZC4KLSAgICAgICAgICAgIGJvb2wgICAgICAgIGV4aXRQZW5kaW5nKCkgY29uc3Q7Ci0gICAgCi1wcml2YXRlOgotICAgIC8vIERlcml2ZWQgY2xhc3MgbXVzdCBpbXBsZW10ZW50IHRocmVhZExvb3AoKS4gVGhlIHRocmVhZCBzdGFydHMgaXRzIGxpZmUKLSAgICAvLyBoZXJlLiBUaGVyZSBhcmUgdHdvIHdheXMgb2YgdXNpbmcgdGhlIFRocmVhZCBvYmplY3Q6Ci0gICAgLy8gMSkgbG9vcDogaWYgdGhyZWFkTG9vcCgpIHJldHVybnMgdHJ1ZSwgaXQgd2lsbCBiZSBjYWxsZWQgYWdhaW4gaWYKLSAgICAvLyAgICAgICAgICByZXF1ZXN0RXhpdCgpIHdhc24ndCBjYWxsZWQuCi0gICAgLy8gMikgb25jZTogaWYgdGhyZWFkTG9vcCgpIHJldHVybnMgZmFsc2UsIHRoZSB0aHJlYWQgd2lsbCBleGl0IHVwb24gcmV0dXJuLgotICAgIHZpcnR1YWwgYm9vbCAgICAgICAgdGhyZWFkTG9vcCgpID0gMDsKLQotcHJpdmF0ZToKLSAgICBUaHJlYWQmIG9wZXJhdG9yPShjb25zdCBUaHJlYWQmKTsKLSAgICBzdGF0aWMgIGludCAgICAgICAgICAgICBfdGhyZWFkTG9vcCh2b2lkKiB1c2VyKTsKLSAgICBjb25zdCAgIGJvb2wgICAgICAgICAgICBtQ2FuQ2FsbEphdmE7Ci0gICAgICAgICAgICB0aHJlYWRfaWRfdCAgICAgbVRocmVhZDsKLSAgICAgICAgICAgIE11dGV4ICAgICAgICAgICBtTG9jazsKLSAgICAgICAgICAgIENvbmRpdGlvbiAgICAgICBtVGhyZWFkRXhpdGVkQ29uZGl0aW9uOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgICAgIG1TdGF0dXM7Ci0gICAgdm9sYXRpbGUgYm9vbCAgICAgICAgICAgbUV4aXRQZW5kaW5nOwotICAgIHZvbGF0aWxlIGJvb2wgICAgICAgICAgIG1SdW5uaW5nOwotICAgICAgICAgICAgc3A8VGhyZWFkPiAgICAgIG1Ib2xkU2VsZjsKLX07Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmICAvLyBfX2NwbHVzcGx1cwotCi0jZW5kaWYgLy8gX0xJQlNfVVRJTFNfVEhSRUFEU19ICmRpZmYgLS1naXQgYS9saWJzL2F1ZGlvZmxpbmdlci9BMmRwQXVkaW9JbnRlcmZhY2UuY3BwIGIvbGlicy9hdWRpb2ZsaW5nZXIvQTJkcEF1ZGlvSW50ZXJmYWNlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDFiN2FmMy4uMDAwMDAwMAotLS0gYS9saWJzL2F1ZGlvZmxpbmdlci9BMmRwQXVkaW9JbnRlcmZhY2UuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMjQzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaW5jbHVkZSA8bWF0aC5oPgotCi0jZGVmaW5lIExPR19OREVCVUcgMAotI2RlZmluZSBMT0dfVEFHICJBMmRwQXVkaW9JbnRlcmZhY2UiCi0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RyaW5nOC5oPgotCi0jaW5jbHVkZSAiQTJkcEF1ZGlvSW50ZXJmYWNlLmgiCi0jaW5jbHVkZSAiYXVkaW8vbGliYTJkcC5oIgotCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1BMmRwQXVkaW9JbnRlcmZhY2U6OkEyZHBBdWRpb0ludGVyZmFjZSgpIDoKLSAgICBtT3V0cHV0KDApCi17Ci19Ci0KLUEyZHBBdWRpb0ludGVyZmFjZTo6fkEyZHBBdWRpb0ludGVyZmFjZSgpCi17Ci0gICAgZGVsZXRlIG1PdXRwdXQ7Ci19Ci0KLXN0YXR1c190IEEyZHBBdWRpb0ludGVyZmFjZTo6aW5pdENoZWNrKCkKLXsKLSAgICByZXR1cm4gMDsKLX0KLQotQXVkaW9TdHJlYW1PdXQqIEEyZHBBdWRpb0ludGVyZmFjZTo6b3Blbk91dHB1dFN0cmVhbSgKLSAgICAgICAgaW50IGZvcm1hdCwgaW50IGNoYW5uZWxDb3VudCwgdWludDMyX3Qgc2FtcGxlUmF0ZSwgc3RhdHVzX3QgKnN0YXR1cykKLXsKLSAgICBMT0dEKCJBMmRwQXVkaW9JbnRlcmZhY2U6Om9wZW5PdXRwdXRTdHJlYW0gJWQsICVkLCAlZFxuIiwgZm9ybWF0LCBjaGFubmVsQ291bnQsIHNhbXBsZVJhdGUpOwotICAgIE11dGV4OjpBdXRvbG9jayBsb2NrKG1Mb2NrKTsKLSAgICBzdGF0dXNfdCBlcnIgPSAwOwotCi0gICAgLy8gb25seSBvbmUgb3V0cHV0IHN0cmVhbSBhbGxvd2VkCi0gICAgaWYgKG1PdXRwdXQpIHsKLSAgICAgICAgaWYgKHN0YXR1cykKLSAgICAgICAgICAgICpzdGF0dXMgPSAtMTsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotCi0gICAgLy8gY3JlYXRlIG5ldyBvdXRwdXQgc3RyZWFtCi0gICAgQTJkcEF1ZGlvU3RyZWFtT3V0KiBvdXQgPSBuZXcgQTJkcEF1ZGlvU3RyZWFtT3V0KCk7Ci0gICAgaWYgKChlcnIgPSBvdXQtPnNldChmb3JtYXQsIGNoYW5uZWxDb3VudCwgc2FtcGxlUmF0ZSkpID09IE5PX0VSUk9SKSB7Ci0gICAgICAgIG1PdXRwdXQgPSBvdXQ7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgZGVsZXRlIG91dDsKLSAgICB9Ci0gICAgCi0gICAgaWYgKHN0YXR1cykKLSAgICAgICAgKnN0YXR1cyA9IGVycjsKLSAgICByZXR1cm4gbU91dHB1dDsKLX0KLQotQXVkaW9TdHJlYW1JbiogQTJkcEF1ZGlvSW50ZXJmYWNlOjpvcGVuSW5wdXRTdHJlYW0oCi0gICAgICAgIGludCBmb3JtYXQsIGludCBjaGFubmVsQ291bnQsIHVpbnQzMl90IHNhbXBsZVJhdGUsIHN0YXR1c190ICpzdGF0dXMsCi0gICAgICAgIEF1ZGlvU3lzdGVtOjphdWRpb19pbl9hY291c3RpY3MgYWNvdXN0aWNzKQotewotICAgIGlmIChzdGF0dXMpCi0gICAgICAgICpzdGF0dXMgPSAtMTsKLSAgICByZXR1cm4gTlVMTDsKLX0KLQotc3RhdHVzX3QgQTJkcEF1ZGlvSW50ZXJmYWNlOjpzZXRNaWNNdXRlKGJvb2wgc3RhdGUpCi17Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXN0YXR1c190IEEyZHBBdWRpb0ludGVyZmFjZTo6Z2V0TWljTXV0ZShib29sKiBzdGF0ZSkKLXsKLSAgICByZXR1cm4gMDsKLX0KLQotc3RhdHVzX3QgQTJkcEF1ZGlvSW50ZXJmYWNlOjpzZXRQYXJhbWV0ZXIoY29uc3QgY2hhciAqa2V5LCBjb25zdCBjaGFyICp2YWx1ZSkKLXsKLSAgICBMT0dEKCJzZXRQYXJhbWV0ZXIgJXMsJXNcbiIsIGtleSwgdmFsdWUpOwotICAgIAotICAgIGlmICgha2V5IHx8ICF2YWx1ZSkKLSAgICAgICAgcmV0dXJuIC1FSU5WQUw7Ci0gICAgCi0gICAgaWYgKHN0cmNtcChrZXksICJhMmRwX3NpbmtfYWRkcmVzcyIpID09IDApIHsgICAgICAgIAotICAgICAgICByZXR1cm4gbU91dHB1dC0+c2V0QWRkcmVzcyh2YWx1ZSk7Ci0gICAgfQotICAgIGlmIChzdHJjbXAoa2V5LCAiYmx1ZXRvb3RoX2VuYWJsZWQiKSA9PSAwICYmCi0gICAgICAgIHN0cmNtcCh2YWx1ZSwgImZhbHNlIikgPT0gMCkgewotICAgICAgICByZXR1cm4gbU91dHB1dC0+Y2xvc2UoKTsKLSAgICB9Ci0KLSAgICByZXR1cm4gMDsKLX0KLQotc3RhdHVzX3QgQTJkcEF1ZGlvSW50ZXJmYWNlOjpzZXRWb2ljZVZvbHVtZShmbG9hdCB2KQotewotICAgIHJldHVybiAwOwotfQotCi1zdGF0dXNfdCBBMmRwQXVkaW9JbnRlcmZhY2U6OnNldE1hc3RlclZvbHVtZShmbG9hdCB2KQotewotICAgIHJldHVybiAwOwotfQotCi1zdGF0dXNfdCBBMmRwQXVkaW9JbnRlcmZhY2U6OmRvUm91dGluZygpCi17Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXN0YXR1c190IEEyZHBBdWRpb0ludGVyZmFjZTo6ZHVtcChpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpCi17Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotQTJkcEF1ZGlvSW50ZXJmYWNlOjpBMmRwQXVkaW9TdHJlYW1PdXQ6OkEyZHBBdWRpb1N0cmVhbU91dCgpIDoKLSAgICBtRmQoLTEpLCBtU3RhbmRieSh0cnVlKSwgbVN0YXJ0Q291bnQoMCksIG1SZXRyeUNvdW50KDApLCBtRGF0YShOVUxMKSwKLSAgICBtSW5pdGlhbGl6ZWQoZmFsc2UpCi17Ci0gICAgLy8gdXNlIGFueSBhZGRyZXNzIGJ5IGRlZmF1bHQKLSAgICBzdHJuY3B5KG1BMmRwQWRkcmVzcywgIjAwOjAwOjAwOjAwOjAwOjAwIiwgc2l6ZW9mKG1BMmRwQWRkcmVzcykpOwotfQotCi1zdGF0dXNfdCBBMmRwQXVkaW9JbnRlcmZhY2U6OkEyZHBBdWRpb1N0cmVhbU91dDo6c2V0KAotICAgICAgICBpbnQgZm9ybWF0LCBpbnQgY2hhbm5lbHMsIHVpbnQzMl90IHJhdGUpCi17Ci0gICAgTE9HRCgiQTJkcEF1ZGlvU3RyZWFtT3V0OjpzZXQgJWQsICVkLCAlZFxuIiwgZm9ybWF0LCBjaGFubmVscywgcmF0ZSk7Ci0KLSAgICAvLyBmaXggdXAgZGVmYXVsdHMKLSAgICBpZiAoZm9ybWF0ID09IDApIGZvcm1hdCA9IEF1ZGlvU3lzdGVtOjpQQ01fMTZfQklUOwotICAgIGlmIChjaGFubmVscyA9PSAwKSBjaGFubmVscyA9IGNoYW5uZWxDb3VudCgpOwotICAgIGlmIChyYXRlID09IDApIHJhdGUgPSBzYW1wbGVSYXRlKCk7Ci0KLSAgICAvLyBjaGVjayB2YWx1ZXMKLSAgICBpZiAoKGZvcm1hdCAhPSBBdWRpb1N5c3RlbTo6UENNXzE2X0JJVCkgfHwKLSAgICAgICAgICAgIChjaGFubmVscyAhPSBjaGFubmVsQ291bnQoKSkgfHwKLSAgICAgICAgICAgIChyYXRlICE9IHNhbXBsZVJhdGUoKSkpCi0gICAgICAgIHJldHVybiBCQURfVkFMVUU7Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLUEyZHBBdWRpb0ludGVyZmFjZTo6QTJkcEF1ZGlvU3RyZWFtT3V0Ojp+QTJkcEF1ZGlvU3RyZWFtT3V0KCkKLXsKLSAgICBjbG9zZSgpOwotfQotCi1zc2l6ZV90IEEyZHBBdWRpb0ludGVyZmFjZTo6QTJkcEF1ZGlvU3RyZWFtT3V0Ojp3cml0ZShjb25zdCB2b2lkKiBidWZmZXIsIHNpemVfdCBieXRlcykKLXsgICAgCi0gICAgc3RhdHVzX3Qgc3RhdHVzID0gTk9fSU5JVDsKLSAgICBzaXplX3QgcmVtYWluaW5nID0gYnl0ZXM7Ci0KLSAgICBpZiAoIW1Jbml0aWFsaXplZCkgewotICAgICAgICBzdGF0dXMgPSBhMmRwX2luaXQobUEyZHBBZGRyZXNzLCA0NDEwMCwgMiwgJm1EYXRhKTsKLSAgICAgICAgaWYgKHN0YXR1cyA8IDApIHsKLSAgICAgICAgICAgIExPR0UoImEyZHBfaW5pdCBmYWlsZWQgZXJyOiAlZFxuIiwgc3RhdHVzKTsKLSAgICAgICAgICAgIGdvdG8gRXJyb3I7Ci0gICAgICAgIH0KLSAgICAgICAgbUluaXRpYWxpemVkID0gdHJ1ZTsKLSAgICB9Ci0gICAgCi0gICAgd2hpbGUgKHJlbWFpbmluZyA+IDApIHsKLSAgICAgICAgc3RhdHVzID0gYTJkcF93cml0ZShtRGF0YSwgYnVmZmVyLCByZW1haW5pbmcpOwotICAgICAgICBpZiAoc3RhdHVzIDw9IDApIHsKLSAgICAgICAgICAgIExPR0UoImEyZHBfd3JpdGUgZmFpbGVkIGVycjogJWRcbiIsIHN0YXR1cyk7Ci0gICAgICAgICAgICBnb3RvIEVycm9yOwotICAgICAgICB9Ci0gICAgICAgIHJlbWFpbmluZyAtPSBzdGF0dXM7Ci0gICAgICAgIGJ1ZmZlciA9ICgoY2hhciAqKWJ1ZmZlcikgKyBzdGF0dXM7Ci0gICAgfQotCi0gICAgbVN0YW5kYnkgPSBmYWxzZTsKLSAgICAKLSAgICByZXR1cm4gYnl0ZXM7Ci0KLUVycm9yOgotICAgIGNsb3NlKCk7Ci0gICAgLy8gU2ltdWxhdGUgYXVkaW8gb3V0cHV0IHRpbWluZyBpbiBjYXNlIG9mIGVycm9yCi0gICAgdXNsZWVwKGJ5dGVzICogMTAwMDAwMCAvIGZyYW1lU2l6ZSgpIC8gc2FtcGxlUmF0ZSgpKTsKLQotICAgIHJldHVybiBzdGF0dXM7Ci19Ci0KLXN0YXR1c190IEEyZHBBdWRpb0ludGVyZmFjZTo6QTJkcEF1ZGlvU3RyZWFtT3V0OjpzdGFuZGJ5KCkKLXsKLSAgICBpbnQgcmVzdWx0ID0gMDsKLQotICAgIGlmICghbVN0YW5kYnkpIHsKLSAgICAgICAgcmVzdWx0ID0gYTJkcF9zdG9wKG1EYXRhKTsKLSAgICAgICAgaWYgKHJlc3VsdCA9PSAwKQotICAgICAgICAgICAgbVN0YW5kYnkgPSB0cnVlOwotICAgIH0KLQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLXN0YXR1c190IEEyZHBBdWRpb0ludGVyZmFjZTo6QTJkcEF1ZGlvU3RyZWFtT3V0OjpzZXRBZGRyZXNzKGNvbnN0IGNoYXIqIGFkZHJlc3MpCi17Ci0gICAgaWYgKHN0cmxlbihhZGRyZXNzKSA8IHNpemVvZihtQTJkcEFkZHJlc3MpKQotICAgICAgICByZXR1cm4gLUVJTlZBTDsKLQotICAgIGlmIChzdHJjbXAoYWRkcmVzcywgbUEyZHBBZGRyZXNzKSkgewotICAgICAgICBzdHJjcHkobUEyZHBBZGRyZXNzLCBhZGRyZXNzKTsKLSAgICAgICAgY2xvc2UoKTsKLSAgICB9Ci0gICAgCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBBMmRwQXVkaW9JbnRlcmZhY2U6OkEyZHBBdWRpb1N0cmVhbU91dDo6Y2xvc2UoKQotewotICAgIGlmIChtRGF0YSkgewotICAgICAgICBhMmRwX2NsZWFudXAobURhdGEpOwotICAgICAgICBtRGF0YSA9IE5VTEw7Ci0gICAgICAgIG1Jbml0aWFsaXplZCA9IGZhbHNlOwotICAgIH0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IEEyZHBBdWRpb0ludGVyZmFjZTo6QTJkcEF1ZGlvU3RyZWFtT3V0OjpkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvYXVkaW9mbGluZ2VyL0EyZHBBdWRpb0ludGVyZmFjZS5oIGIvbGlicy9hdWRpb2ZsaW5nZXIvQTJkcEF1ZGlvSW50ZXJmYWNlLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDViZWY1ZGEuLjAwMDAwMDAKLS0tIGEvbGlicy9hdWRpb2ZsaW5nZXIvQTJkcEF1ZGlvSW50ZXJmYWNlLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxMTAgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQTJEUF9BVURJT19IQVJEV0FSRV9ICi0jZGVmaW5lIEEyRFBfQVVESU9fSEFSRFdBUkVfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLSNpbmNsdWRlIDxoYXJkd2FyZV9sZWdhY3kvQXVkaW9IYXJkd2FyZUJhc2UuaD4KLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIEEyZHBBdWRpb0ludGVyZmFjZSA6IHB1YmxpYyBBdWRpb0hhcmR3YXJlQmFzZQotewotICAgIGNsYXNzIEEyZHBBdWRpb1N0cmVhbU91dDsKLQotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgQTJkcEF1ZGlvSW50ZXJmYWNlKCk7Ci0gICAgdmlydHVhbCAgICAgICAgICAgICB+QTJkcEF1ZGlvSW50ZXJmYWNlKCk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBpbml0Q2hlY2soKTsKLQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0Vm9pY2VWb2x1bWUoZmxvYXQgdm9sdW1lKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHNldE1hc3RlclZvbHVtZShmbG9hdCB2b2x1bWUpOwotCi0gICAgLy8gbWljIG11dGUKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHNldE1pY011dGUoYm9vbCBzdGF0ZSk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBnZXRNaWNNdXRlKGJvb2wqIHN0YXRlKTsKLQotICAgIC8vIFRlbXBvcmFyeSBpbnRlcmZhY2UsIGRvIG5vdCB1c2UKLSAgICAvLyBUT0RPOiBSZXBsYWNlIHdpdGggYSBtb3JlIGdlbmVyaWMga2V5OnZhbHVlIGdldC9zZXQgbWVjaGFuaXNtCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRQYXJhbWV0ZXIoY29uc3QgY2hhciAqa2V5LCBjb25zdCBjaGFyICp2YWx1ZSk7Ci0KLSAgICAvLyBjcmVhdGUgSS9PIHN0cmVhbXMKLSAgICB2aXJ0dWFsIEF1ZGlvU3RyZWFtT3V0KiBvcGVuT3V0cHV0U3RyZWFtKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZm9ybWF0PTAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBjaGFubmVsQ291bnQ9MCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZT0wLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzdGF0dXNfdCAqc3RhdHVzPTApOwotCi0gICAgdmlydHVhbCBBdWRpb1N0cmVhbUluKiBvcGVuSW5wdXRTdHJlYW0oCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBmb3JtYXQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBjaGFubmVsQ291bnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHNhbXBsZVJhdGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXR1c190ICpzdGF0dXMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEF1ZGlvU3lzdGVtOjphdWRpb19pbl9hY291c3RpY3MgYWNvdXN0aWNzKTsKLQotcHJvdGVjdGVkOgotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgZG9Sb3V0aW5nKCk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0KLXByaXZhdGU6Ci0gICAgY2xhc3MgQTJkcEF1ZGlvU3RyZWFtT3V0IDogcHVibGljIEF1ZGlvU3RyZWFtT3V0IHsKLSAgICBwdWJsaWM6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgQTJkcEF1ZGlvU3RyZWFtT3V0KCk7Ci0gICAgICAgIHZpcnR1YWwgICAgICAgICAgICAgfkEyZHBBdWRpb1N0cmVhbU91dCgpOwotICAgICAgICAgICAgICAgIHN0YXR1c190ICAgIHNldChpbnQgZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlKTsKLSAgICAgICAgdmlydHVhbCB1aW50MzJfdCAgICBzYW1wbGVSYXRlKCkgY29uc3QgeyByZXR1cm4gNDQxMDA7IH0KLSAgICAgICAgLy8gU0JDIGNvZGVjIHdhbnRzIGEgbXVsdGlwbGUgb2YgNTEyCi0gICAgICAgIHZpcnR1YWwgc2l6ZV90ICAgICAgYnVmZmVyU2l6ZSgpIGNvbnN0IHsgcmV0dXJuIDUxMiAqIDIwOyB9Ci0gICAgICAgIHZpcnR1YWwgaW50ICAgICAgICAgY2hhbm5lbENvdW50KCkgY29uc3QgeyByZXR1cm4gMjsgfQotICAgICAgICB2aXJ0dWFsIGludCAgICAgICAgIGZvcm1hdCgpIGNvbnN0IHsgcmV0dXJuIEF1ZGlvU3lzdGVtOjpQQ01fMTZfQklUOyB9Ci0gICAgICAgIHZpcnR1YWwgdWludDMyX3QgICAgbGF0ZW5jeSgpIGNvbnN0IHsgcmV0dXJuICgoMTAwMCpidWZmZXJTaXplKCkpL2ZyYW1lU2l6ZSgpKS9zYW1wbGVSYXRlKCkgKyAyMDA7IH0KLSAgICAgICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRWb2x1bWUoZmxvYXQgdm9sdW1lKSB7IHJldHVybiBJTlZBTElEX09QRVJBVElPTjsgfQotICAgICAgICB2aXJ0dWFsIHNzaXplX3QgICAgIHdyaXRlKGNvbnN0IHZvaWQqIGJ1ZmZlciwgc2l6ZV90IGJ5dGVzKTsKLSAgICAgICAgICAgICAgICBzdGF0dXNfdCAgICBzdGFuZGJ5KCk7Ci0gICAgICAgICAgICAgICAgc3RhdHVzX3QgICAgY2xvc2UoKTsKLSAgICAgICAgdmlydHVhbCBzdGF0dXNfdCAgICBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0KLSAgICBwcml2YXRlOgotICAgICAgICBmcmllbmQgY2xhc3MgQTJkcEF1ZGlvSW50ZXJmYWNlOwotICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgIHNldEFkZHJlc3MoY29uc3QgY2hhciogYWRkcmVzcyk7Ci0KLSAgICBwcml2YXRlOgotICAgICAgICAgICAgICAgIGludCAgICAgICAgIG1GZDsKLSAgICAgICAgICAgICAgICBib29sICAgICAgICBtU3RhbmRieTsKLSAgICAgICAgICAgICAgICBpbnQgICAgICAgICBtU3RhcnRDb3VudDsKLSAgICAgICAgICAgICAgICBpbnQgICAgICAgICBtUmV0cnlDb3VudDsKLSAgICAgICAgICAgICAgICBjaGFyICAgICAgICBtQTJkcEFkZHJlc3NbMjBdOwotICAgICAgICAgICAgICAgIHZvaWQqICAgICAgIG1EYXRhOwotICAgICAgICAgICAgICAgIGJvb2wgICAgICAgIG1Jbml0aWFsaXplZDsKLSAgICB9OwotCi0gICAgTXV0ZXggICAgICAgICAgICAgICAgICAgbUxvY2s7Ci0gICAgQTJkcEF1ZGlvU3RyZWFtT3V0KiAgICAgbU91dHB1dDsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEEyRFBfQVVESU9fSEFSRFdBUkVfSApkaWZmIC0tZ2l0IGEvbGlicy9hdWRpb2ZsaW5nZXIvQW5kcm9pZC5tayBiL2xpYnMvYXVkaW9mbGluZ2VyL0FuZHJvaWQubWsKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDUwZDUxNmIuLjAwMDAwMDAKLS0tIGEvbGlicy9hdWRpb2ZsaW5nZXIvQW5kcm9pZC5taworKysgL2Rldi9udWxsCkBAIC0xLDU2ICswLDAgQEAKLUxPQ0FMX1BBVEg6PSAkKGNhbGwgbXktZGlyKQotCi1pbmNsdWRlICQoQ0xFQVJfVkFSUykKLQotTE9DQUxfU1JDX0ZJTEVTOj0gXAotICAgIEF1ZGlvSGFyZHdhcmVHZW5lcmljLmNwcCBcCi0gICAgQXVkaW9IYXJkd2FyZVN0dWIuY3BwIFwKLSAgICBBdWRpb0R1bXBJbnRlcmZhY2UuY3BwIFwKLSAgICBBdWRpb0hhcmR3YXJlSW50ZXJmYWNlLmNwcAotCi1MT0NBTF9TSEFSRURfTElCUkFSSUVTIDo9IFwKLSAgICBsaWJjdXRpbHMgXAotICAgIGxpYnV0aWxzIFwKLSAgICBsaWJtZWRpYSBcCi0gICAgbGliaGFyZHdhcmVfbGVnYWN5Ci0KLWlmZXEgKCQoc3RyaXAgJChCT0FSRF9VU0VTX0dFTkVSSUNfQVVESU8pKSx0cnVlKQotICBMT0NBTF9DRkxBR1MgKz0gLURHRU5FUklDX0FVRElPCi1lbmRpZgotCi1MT0NBTF9NT0RVTEU6PSBsaWJhdWRpb2ludGVyZmFjZQotCi1pbmNsdWRlICQoQlVJTERfU1RBVElDX0xJQlJBUlkpCi0KLWluY2x1ZGUgJChDTEVBUl9WQVJTKQotCi1MT0NBTF9TUkNfRklMRVM6PSAgICAgICAgICAgICAgIFwKLSAgICBBdWRpb0ZsaW5nZXIuY3BwICAgICAgICAgICAgXAotICAgIEF1ZGlvTWl4ZXIuY3BwLmFybSAgICAgICAgICBcCi0gICAgQXVkaW9SZXNhbXBsZXIuY3BwLmFybSAgICAgIFwKLSAgICBBdWRpb1Jlc2FtcGxlclNpbmMuY3BwLmFybSAgXAotICAgIEF1ZGlvUmVzYW1wbGVyQ3ViaWMuY3BwLmFybQotCi1MT0NBTF9TSEFSRURfTElCUkFSSUVTIDo9IFwKLSAgICBsaWJjdXRpbHMgXAotICAgIGxpYnV0aWxzIFwKLSAgICBsaWJtZWRpYSBcCi0gICAgbGliaGFyZHdhcmVfbGVnYWN5Ci0KLWlmZXEgKCQoc3RyaXAgJChCT0FSRF9VU0VTX0dFTkVSSUNfQVVESU8pKSx0cnVlKQotICBMT0NBTF9TVEFUSUNfTElCUkFSSUVTICs9IGxpYmF1ZGlvaW50ZXJmYWNlCi1lbHNlCi0gIExPQ0FMX1NIQVJFRF9MSUJSQVJJRVMgKz0gbGliYXVkaW8KLWVuZGlmCi0KLUxPQ0FMX01PRFVMRTo9IGxpYmF1ZGlvZmxpbmdlcgotCi1pZmVxICgkKEJPQVJEX0hBVkVfQkxVRVRPT1RIKSx0cnVlKQotICBMT0NBTF9TUkNfRklMRVMgKz0gQTJkcEF1ZGlvSW50ZXJmYWNlLmNwcAotICBMT0NBTF9TSEFSRURfTElCUkFSSUVTICs9IGxpYmEyZHAKLSAgTE9DQUxfQ0ZMQUdTICs9IC1EV0lUSF9CTFVFVE9PVEggLURXSVRIX0EyRFAKLSAgTE9DQUxfQ19JTkNMVURFUyArPSAkKGNhbGwgaW5jbHVkZS1wYXRoLWZvciwgYmx1ZXotbGlicykKLSAgTE9DQUxfQ19JTkNMVURFUyArPSAkKGNhbGwgaW5jbHVkZS1wYXRoLWZvciwgYmx1ZXotdXRpbHMpCi1lbmRpZgotCi1pbmNsdWRlICQoQlVJTERfU0hBUkVEX0xJQlJBUlkpCmRpZmYgLS1naXQgYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0J1ZmZlclByb3ZpZGVyLmggYi9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0J1ZmZlclByb3ZpZGVyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDFhNDY3YzcuLjAwMDAwMDAKLS0tIGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9CdWZmZXJQcm92aWRlci5oCisrKyAvZGV2L251bGwKQEAgLTEsNDcgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9BVURJT19CVUZGRVJfUFJPVklERVJfSAotI2RlZmluZSBBTkRST0lEX0FVRElPX0JVRkZFUl9QUk9WSURFUl9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBBdWRpb0J1ZmZlclByb3ZpZGVyCi17Ci1wdWJsaWM6Ci0KLSAgICBzdHJ1Y3QgQnVmZmVyIHsKLSAgICAgICAgdW5pb24gewotICAgICAgICAgICAgdm9pZCogICAgICAgcmF3OwotICAgICAgICAgICAgc2hvcnQqICAgICAgaTE2OwotICAgICAgICAgICAgaW50OF90KiAgICAgaTg7Ci0gICAgICAgIH07Ci0gICAgICAgIHNpemVfdCBmcmFtZUNvdW50OwotICAgIH07Ci0gICAgCi0gICAgdmlydHVhbCBzdGF0dXNfdCBnZXROZXh0QnVmZmVyKEJ1ZmZlciogYnVmZmVyKSA9IDA7Ci0gICAgdmlydHVhbCB2b2lkIHJlbGVhc2VCdWZmZXIoQnVmZmVyKiBidWZmZXIpID0gMDsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0FVRElPX0JVRkZFUl9QUk9WSURFUl9ICmRpZmYgLS1naXQgYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0R1bXBJbnRlcmZhY2UuY3BwIGIvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9EdW1wSW50ZXJmYWNlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYjQ5NDBjYi4uMDAwMDAwMAotLS0gYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0R1bXBJbnRlcmZhY2UuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTE3ICswLDAgQEAKLS8qIC8vZGV2aWNlL3NlcnZlcnMvQXVkaW9GbGluZ2VyL0F1ZGlvRHVtcEludGVyZmFjZS5jcHAKLSoqCi0qKiBDb3B5cmlnaHQgMjAwOCwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2RlZmluZSBMT0dfVEFHICJBdWRpb0ZsaW5nZXJEdW1wIgotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLQotI2luY2x1ZGUgIkF1ZGlvRHVtcEludGVyZmFjZS5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWJvb2wgZ0ZpcnN0ID0gdHJ1ZTsgICAgICAgLy8gdHJ1ZSBpZiBmaXJzdCB3cml0ZSBhZnRlciBhIHN0YW5kYnkKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1BdWRpb0R1bXBJbnRlcmZhY2U6OkF1ZGlvRHVtcEludGVyZmFjZShBdWRpb0hhcmR3YXJlSW50ZXJmYWNlKiBodykKLXsKLSAgICBpZihodyA9PSAwKSB7Ci0gICAgICAgIExPR0UoIkR1bXAgY29uc3RydWN0IGh3ID0gMCIpOwotICAgIH0KLSAgICBtRmluYWxJbnRlcmZhY2UgPSBodzsKLSAgICBtU3RyZWFtT3V0ID0gMDsKLX0KLQotCi1BdWRpb0R1bXBJbnRlcmZhY2U6On5BdWRpb0R1bXBJbnRlcmZhY2UoKQotewotICAgIGlmKG1GaW5hbEludGVyZmFjZSkgZGVsZXRlIG1GaW5hbEludGVyZmFjZTsKLSAgICBpZihtU3RyZWFtT3V0KSBkZWxldGUgbVN0cmVhbU91dDsKLX0KLQotCi1BdWRpb1N0cmVhbU91dCogQXVkaW9EdW1wSW50ZXJmYWNlOjpvcGVuT3V0cHV0U3RyZWFtKAotICAgICAgICBpbnQgZm9ybWF0LCBpbnQgY2hhbm5lbENvdW50LCB1aW50MzJfdCBzYW1wbGVSYXRlLCBzdGF0dXNfdCAqc3RhdHVzKQotewotICAgIEF1ZGlvU3RyZWFtT3V0KiBvdXRGaW5hbCA9IG1GaW5hbEludGVyZmFjZS0+b3Blbk91dHB1dFN0cmVhbShmb3JtYXQsIGNoYW5uZWxDb3VudCwgc2FtcGxlUmF0ZSwgc3RhdHVzKTsKLQotICAgIGlmKG91dEZpbmFsKSB7Ci0gICAgICAgIG1TdHJlYW1PdXQgPSAgbmV3IEF1ZGlvU3RyZWFtT3V0RHVtcChvdXRGaW5hbCk7Ci0gICAgICAgIHJldHVybiBtU3RyZWFtT3V0OwotICAgIH0gZWxzZSB7Ci0gICAgICAgIExPR0UoIkR1bXAgb3V0RmluYWw9MCIpOwotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotQXVkaW9TdHJlYW1PdXREdW1wOjpBdWRpb1N0cmVhbU91dER1bXAoIEF1ZGlvU3RyZWFtT3V0KiBmaW5hbFN0cmVhbSkKLXsKLSAgICBtRmluYWxTdHJlYW0gPSBmaW5hbFN0cmVhbTsKLSAgICBtT3V0RmlsZSA9IDA7Ci19Ci0KLQotQXVkaW9TdHJlYW1PdXREdW1wOjp+QXVkaW9TdHJlYW1PdXREdW1wKCkKLXsKLSAgICBDbG9zZSgpOwotICAgIGRlbGV0ZSBtRmluYWxTdHJlYW07Ci19Ci0KLXNzaXplX3QgQXVkaW9TdHJlYW1PdXREdW1wOjp3cml0ZShjb25zdCB2b2lkKiBidWZmZXIsIHNpemVfdCBieXRlcykKLXsKLSAgICBzc2l6ZV90IHJldDsKLQotICAgIHJldCA9IG1GaW5hbFN0cmVhbS0+d3JpdGUoYnVmZmVyLCBieXRlcyk7Ci0gICAgaWYoIW1PdXRGaWxlICYmIGdGaXJzdCkgewotICAgICAgICBnRmlyc3QgPSBmYWxzZTsKLSAgICAgICAgLy8gY2hlY2sgaWYgZHVtcCBmaWxlIGV4aXN0Ci0gICAgICAgIG1PdXRGaWxlID0gZm9wZW4oRkxJTkdFUl9EVU1QX05BTUUsICJyIik7Ci0gICAgICAgIGlmKG1PdXRGaWxlKSB7Ci0gICAgICAgICAgICBmY2xvc2UobU91dEZpbGUpOwotICAgICAgICAgICAgbU91dEZpbGUgPSBmb3BlbihGTElOR0VSX0RVTVBfTkFNRSwgImFiIik7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgaWYgKG1PdXRGaWxlKSB7Ci0gICAgICAgIGZ3cml0ZShidWZmZXIsIGJ5dGVzLCAxLCBtT3V0RmlsZSk7Ci0gICAgfQotICAgIHJldHVybiByZXQ7Ci19Ci0KLXN0YXR1c190IEF1ZGlvU3RyZWFtT3V0RHVtcDo6c3RhbmRieSgpCi17Ci0gICAgQ2xvc2UoKTsKLSAgICBnRmlyc3QgPSB0cnVlOwotICAgIHJldHVybiBtRmluYWxTdHJlYW0tPnN0YW5kYnkoKTsKLX0KLQotCi12b2lkIEF1ZGlvU3RyZWFtT3V0RHVtcDo6Q2xvc2Uodm9pZCkKLXsKLSAgICBpZihtT3V0RmlsZSkgewotICAgICAgICBmY2xvc2UobU91dEZpbGUpOwotICAgICAgICBtT3V0RmlsZSA9IDA7Ci0gICAgfQotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9EdW1wSW50ZXJmYWNlLmggYi9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0R1bXBJbnRlcmZhY2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOWE5NDEwMi4uMDAwMDAwMAotLS0gYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0R1bXBJbnRlcmZhY2UuaAorKysgL2Rldi9udWxsCkBAIC0xLDk3ICswLDAgQEAKLS8qIC8vZGV2aWNlL3NlcnZlcnMvQXVkaW9GbGluZ2VyL0F1ZGlvRHVtcEludGVyZmFjZS5oCi0qKgotKiogQ29weXJpZ2h0IDIwMDgsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9BVURJT19EVU1QX0lOVEVSRkFDRV9ICi0jZGVmaW5lIEFORFJPSURfQVVESU9fRFVNUF9JTlRFUkZBQ0VfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDxoYXJkd2FyZV9sZWdhY3kvQXVkaW9IYXJkd2FyZUJhc2UuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0jZGVmaW5lIEZMSU5HRVJfRFVNUF9OQU1FICIvZGF0YS9GbGluZ2VyT3V0LnBjbSIgLy8gbmFtZSBvZiBmaWxlIHVzZWQgZm9yIGR1bXAKLQotY2xhc3MgQXVkaW9TdHJlYW1PdXREdW1wIDogcHVibGljIEF1ZGlvU3RyZWFtT3V0IHsKLXB1YmxpYzoKLSAgICAgICAgICAgICAgICAgICAgICAgIEF1ZGlvU3RyZWFtT3V0RHVtcCggQXVkaW9TdHJlYW1PdXQqIEZpbmFsU3RyZWFtKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH5BdWRpb1N0cmVhbU91dER1bXAoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHZpcnR1YWwgc3NpemVfdCAgICAgd3JpdGUoY29uc3Qgdm9pZCogYnVmZmVyLCBzaXplX3QgYnl0ZXMpOwotCi0gICAgdmlydHVhbCB1aW50MzJfdCAgICBzYW1wbGVSYXRlKCkgY29uc3QgeyByZXR1cm4gbUZpbmFsU3RyZWFtLT5zYW1wbGVSYXRlKCk7IH0KLSAgICB2aXJ0dWFsIHNpemVfdCAgICAgIGJ1ZmZlclNpemUoKSBjb25zdCB7IHJldHVybiBtRmluYWxTdHJlYW0tPmJ1ZmZlclNpemUoKTsgfQotICAgIHZpcnR1YWwgaW50ICAgICAgICAgY2hhbm5lbENvdW50KCkgY29uc3QgeyByZXR1cm4gbUZpbmFsU3RyZWFtLT5jaGFubmVsQ291bnQoKTsgfQotICAgIHZpcnR1YWwgaW50ICAgICAgICAgZm9ybWF0KCkgY29uc3QgeyByZXR1cm4gbUZpbmFsU3RyZWFtLT5mb3JtYXQoKTsgfQotICAgIHZpcnR1YWwgdWludDMyX3QgICAgbGF0ZW5jeSgpIGNvbnN0IHsgcmV0dXJuIG1GaW5hbFN0cmVhbS0+bGF0ZW5jeSgpOyB9Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRWb2x1bWUoZmxvYXQgdm9sdW1lKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHsgcmV0dXJuIG1GaW5hbFN0cmVhbS0+c2V0Vm9sdW1lKHZvbHVtZSk7IH0KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHN0YW5kYnkoKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIGR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKSB7IHJldHVybiBtRmluYWxTdHJlYW0tPmR1bXAoZmQsIGFyZ3MpOyB9Ci0gICAgdm9pZCAgICAgICAgICAgICAgICBDbG9zZSh2b2lkKTsKLQotcHJpdmF0ZToKLSAgICBBdWRpb1N0cmVhbU91dCAgICAgICptRmluYWxTdHJlYW07Ci0gICAgRklMRSAgICAgICAgICAgICAgICAqbU91dEZpbGU7ICAgICAvLyBvdXRwdXQgZmlsZQotfTsKLQotCi1jbGFzcyBBdWRpb0R1bXBJbnRlcmZhY2UgOiBwdWJsaWMgQXVkaW9IYXJkd2FyZUJhc2UKLXsKLQotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgQXVkaW9EdW1wSW50ZXJmYWNlKEF1ZGlvSGFyZHdhcmVJbnRlcmZhY2UqIGh3KTsKLSAgICB2aXJ0dWFsIEF1ZGlvU3RyZWFtT3V0KiBvcGVuT3V0cHV0U3RyZWFtKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZm9ybWF0PTAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBjaGFubmVsQ291bnQ9MCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZT0wLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzdGF0dXNfdCAqc3RhdHVzPTApOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgfkF1ZGlvRHVtcEludGVyZmFjZSgpOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBpbml0Q2hlY2soKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHtyZXR1cm4gbUZpbmFsSW50ZXJmYWNlLT5pbml0Q2hlY2soKTt9Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRWb2ljZVZvbHVtZShmbG9hdCB2b2x1bWUpCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAge3JldHVybiBtRmluYWxJbnRlcmZhY2UtPnNldFZvaWNlVm9sdW1lKHZvbHVtZSk7fQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0TWFzdGVyVm9sdW1lKGZsb2F0IHZvbHVtZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB7cmV0dXJuIG1GaW5hbEludGVyZmFjZS0+c2V0TWFzdGVyVm9sdW1lKHZvbHVtZSk7fQotCi0gICAgLy8gbWljIG11dGUKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHNldE1pY011dGUoYm9vbCBzdGF0ZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB7cmV0dXJuIG1GaW5hbEludGVyZmFjZS0+c2V0TWljTXV0ZShzdGF0ZSk7fQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgZ2V0TWljTXV0ZShib29sKiBzdGF0ZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB7cmV0dXJuIG1GaW5hbEludGVyZmFjZS0+Z2V0TWljTXV0ZShzdGF0ZSk7fQotCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRQYXJhbWV0ZXIoY29uc3QgY2hhcioga2V5LCBjb25zdCBjaGFyKiB2YWx1ZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB7cmV0dXJuIG1GaW5hbEludGVyZmFjZS0+c2V0UGFyYW1ldGVyKGtleSwgdmFsdWUpO30KLQotICAgIHZpcnR1YWwgQXVkaW9TdHJlYW1Jbiogb3BlbklucHV0U3RyZWFtKCBpbnQgZm9ybWF0LCBpbnQgY2hhbm5lbENvdW50LCB1aW50MzJfdCBzYW1wbGVSYXRlLCBzdGF0dXNfdCAqc3RhdHVzLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBBdWRpb1N5c3RlbTo6YXVkaW9faW5fYWNvdXN0aWNzIGFjb3VzdGljcykKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB7cmV0dXJuIG1GaW5hbEludGVyZmFjZS0+b3BlbklucHV0U3RyZWFtKCBmb3JtYXQsIGNoYW5uZWxDb3VudCwgc2FtcGxlUmF0ZSwgc3RhdHVzLCBhY291c3RpY3MpO30KLQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgZHVtcChpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpIHsgcmV0dXJuIG1GaW5hbEludGVyZmFjZS0+ZHVtcFN0YXRlKGZkLCBhcmdzKTsgfQotCi1wcm90ZWN0ZWQ6Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBkb1JvdXRpbmcoKSB7cmV0dXJuIG1GaW5hbEludGVyZmFjZS0+c2V0Um91dGluZyhtTW9kZSwgbVJvdXRlc1ttTW9kZV0pO30KLQotICAgIEF1ZGlvSGFyZHdhcmVJbnRlcmZhY2UgICptRmluYWxJbnRlcmZhY2U7Ci0gICAgQXVkaW9TdHJlYW1PdXREdW1wICAgICAgKm1TdHJlYW1PdXQ7Ci0KLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0FVRElPX0RVTVBfSU5URVJGQUNFX0gKZGlmZiAtLWdpdCBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvRmxpbmdlci5jcHAgYi9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0ZsaW5nZXIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1NTdkOTNiLi4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvRmxpbmdlci5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwyNDcxICswLDAgQEAKLS8qIC8vZGV2aWNlL2luY2x1ZGUvc2VydmVyL0F1ZGlvRmxpbmdlci9BdWRpb0ZsaW5nZXIuY3BwCi0qKgotKiogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLQotI2RlZmluZSBMT0dfVEFHICJBdWRpb0ZsaW5nZXIiCi0vLyNkZWZpbmUgTE9HX05ERUJVRyAwCi0KLSNpbmNsdWRlIDxtYXRoLmg+Ci0jaW5jbHVkZSA8c2lnbmFsLmg+Ci0jaW5jbHVkZSA8c3lzL3RpbWUuaD4KLSNpbmNsdWRlIDxzeXMvcmVzb3VyY2UuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0lTZXJ2aWNlTWFuYWdlci5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotI2luY2x1ZGUgPHV0aWxzL0lQQ1RocmVhZFN0YXRlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RyaW5nMTYuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLSNpbmNsdWRlIDxjdXRpbHMvcHJvcGVydGllcy5oPgotCi0jaW5jbHVkZSA8bWVkaWEvQXVkaW9UcmFjay5oPgotI2luY2x1ZGUgPG1lZGlhL0F1ZGlvUmVjb3JkLmg+Ci0KLSNpbmNsdWRlIDxwcml2YXRlL21lZGlhL0F1ZGlvVHJhY2tTaGFyZWQuaD4KLQotI2luY2x1ZGUgPGhhcmR3YXJlX2xlZ2FjeS9BdWRpb0hhcmR3YXJlSW50ZXJmYWNlLmg+Ci0KLSNpbmNsdWRlICJBdWRpb01peGVyLmgiCi0jaW5jbHVkZSAiQXVkaW9GbGluZ2VyLmgiCi0KLSNpZmRlZiBXSVRIX0EyRFAKLSNpbmNsdWRlICJBMmRwQXVkaW9JbnRlcmZhY2UuaCIKLSNlbmRpZgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyB0aGUgc2ltIGJ1aWxkIGRvZXNuJ3QgaGF2ZSBnZXR0aWQKLQotI2lmbmRlZiBIQVZFX0dFVFRJRAotIyBkZWZpbmUgZ2V0dGlkIGdldHBpZAotI2VuZGlmCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vL3N0YXRpYyBjb25zdCBuc2Vjc190IGtTdGFuZGJ5VGltZUluTnNlY3MgPSBzZWNvbmRzKDMpOwotc3RhdGljIGNvbnN0IHVuc2lnbmVkIGxvbmcga0J1ZmZlclJlY292ZXJ5SW5Vc2VjcyA9IDIwMDA7Ci1zdGF0aWMgY29uc3QgdW5zaWduZWQgbG9uZyBrTWF4QnVmZmVyUmVjb3ZlcnlJblVzZWNzID0gMjAwMDA7Ci1zdGF0aWMgY29uc3QgZmxvYXQgTUFYX0dBSU4gPSA0MDk2LjBmOwotCi0vLyByZXRyeSBjb3VudHMgZm9yIGJ1ZmZlciBmaWxsIHRpbWVvdXQKLS8vIDUwICogfjIwbXNlY3MgPSAxIHNlY29uZAotc3RhdGljIGNvbnN0IGludDhfdCBrTWF4VHJhY2tSZXRyaWVzID0gNTA7Ci1zdGF0aWMgY29uc3QgaW50OF90IGtNYXhUcmFja1N0YXJ0dXBSZXRyaWVzID0gNTA7Ci0KLXN0YXRpYyBjb25zdCBpbnQga1N0YXJ0U2xlZXBUaW1lID0gMzAwMDA7Ci1zdGF0aWMgY29uc3QgaW50IGtTdG9wU2xlZXBUaW1lID0gMzAwMDA7Ci0KLS8vIE1heGltdW0gbnVtYmVyIG9mIHBlbmRpbmcgYnVmZmVycyBhbGxvY2F0ZWQgYnkgT3V0cHV0VHJhY2s6OndyaXRlKCkKLXN0YXRpYyBjb25zdCB1aW50OF90IGtNYXhPdXRwdXRUcmFja0J1ZmZlcnMgPSA1OwotCi0KLSNkZWZpbmUgQVVESU9GTElOR0VSX1NFQ1VSSVRZX0VOQUJMRUQgMQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyBib29sIHJlY29yZGluZ0FsbG93ZWQoKSB7Ci0jaWZuZGVmIEhBVkVfQU5EUk9JRF9PUwotICAgIHJldHVybiB0cnVlOwotI2VuZGlmCi0jaWYgQVVESU9GTElOR0VSX1NFQ1VSSVRZX0VOQUJMRUQKLSAgICBpZiAoZ2V0cGlkKCkgPT0gSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpKSByZXR1cm4gdHJ1ZTsKLSAgICBib29sIG9rID0gY2hlY2tDYWxsaW5nUGVybWlzc2lvbihTdHJpbmcxNigiYW5kcm9pZC5wZXJtaXNzaW9uLlJFQ09SRF9BVURJTyIpKTsKLSAgICBpZiAoIW9rKSBMT0dFKCJSZXF1ZXN0IHJlcXVpcmVzIGFuZHJvaWQucGVybWlzc2lvbi5SRUNPUkRfQVVESU8iKTsKLSAgICByZXR1cm4gb2s7Ci0jZWxzZQotICAgIGlmICghY2hlY2tDYWxsaW5nUGVybWlzc2lvbihTdHJpbmcxNigiYW5kcm9pZC5wZXJtaXNzaW9uLlJFQ09SRF9BVURJTyIpKSkKLSAgICAgICAgTE9HVygiV0FSTklORzogTmVlZCB0byBhZGQgYW5kcm9pZC5wZXJtaXNzaW9uLlJFQ09SRF9BVURJTyB0byBtYW5pZmVzdCIpOwotICAgIHJldHVybiB0cnVlOwotI2VuZGlmCi19Ci0KLXN0YXRpYyBib29sIHNldHRpbmdzQWxsb3dlZCgpIHsKLSNpZm5kZWYgSEFWRV9BTkRST0lEX09TCi0gICAgcmV0dXJuIHRydWU7Ci0jZW5kaWYKLSNpZiBBVURJT0ZMSU5HRVJfU0VDVVJJVFlfRU5BQkxFRAotICAgIGlmIChnZXRwaWQoKSA9PSBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5nZXRDYWxsaW5nUGlkKCkpIHJldHVybiB0cnVlOwotICAgIGJvb2wgb2sgPSBjaGVja0NhbGxpbmdQZXJtaXNzaW9uKFN0cmluZzE2KCJhbmRyb2lkLnBlcm1pc3Npb24uTU9ESUZZX0FVRElPX1NFVFRJTkdTIikpOwotICAgIGlmICghb2spIExPR0UoIlJlcXVlc3QgcmVxdWlyZXMgYW5kcm9pZC5wZXJtaXNzaW9uLk1PRElGWV9BVURJT19TRVRUSU5HUyIpOwotICAgIHJldHVybiBvazsKLSNlbHNlCi0gICAgaWYgKCFjaGVja0NhbGxpbmdQZXJtaXNzaW9uKFN0cmluZzE2KCJhbmRyb2lkLnBlcm1pc3Npb24uTU9ESUZZX0FVRElPX1NFVFRJTkdTIikpKQotICAgICAgICBMT0dXKCJXQVJOSU5HOiBOZWVkIHRvIGFkZCBhbmRyb2lkLnBlcm1pc3Npb24uTU9ESUZZX0FVRElPX1NFVFRJTkdTIHRvIG1hbmlmZXN0Iik7Ci0gICAgcmV0dXJuIHRydWU7Ci0jZW5kaWYKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1BdWRpb0ZsaW5nZXI6OkF1ZGlvRmxpbmdlcigpCi0gICAgOiBCbkF1ZGlvRmxpbmdlcigpLAotICAgICAgICBtQXVkaW9IYXJkd2FyZSgwKSwgbUEyZHBBdWRpb0ludGVyZmFjZSgwKSwKLSAgICAgICAgbUEyZHBFbmFibGVkKGZhbHNlKSwgbUEyZHBFbmFibGVkUmVxKGZhbHNlKSwKLSAgICAgICAgbUZvcmNlZFNwZWFrZXJDb3VudCgwKSwgbUZvcmNlZFJvdXRlKDApLCBtUm91dGVSZXN0b3JlVGltZSgwKSwgbU11c2ljTXV0ZVNhdmVkKGZhbHNlKQotewotICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX0hXX0lETEU7Ci0gICAgbUF1ZGlvSGFyZHdhcmUgPSBBdWRpb0hhcmR3YXJlSW50ZXJmYWNlOjpjcmVhdGUoKTsKLSAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19JTklUOwotICAgIGlmIChtQXVkaW9IYXJkd2FyZS0+aW5pdENoZWNrKCkgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgLy8gb3BlbiAxNi1iaXQgb3V0cHV0IHN0cmVhbSBmb3Igcy93IG1peGVyCi0gICAgICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX0hXX09VVFBVVF9PUEVOOwotICAgICAgICBzdGF0dXNfdCBzdGF0dXM7Ci0gICAgICAgIEF1ZGlvU3RyZWFtT3V0ICpod091dHB1dCA9IG1BdWRpb0hhcmR3YXJlLT5vcGVuT3V0cHV0U3RyZWFtKEF1ZGlvU3lzdGVtOjpQQ01fMTZfQklULCAwLCAwLCAmc3RhdHVzKTsKLSAgICAgICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfSURMRTsKLSAgICAgICAgaWYgKGh3T3V0cHV0KSB7Ci0gICAgICAgICAgICBtSGFyZHdhcmVNaXhlclRocmVhZCA9IG5ldyBNaXhlclRocmVhZCh0aGlzLCBod091dHB1dCwgQXVkaW9TeXN0ZW06OkFVRElPX09VVFBVVF9IQVJEV0FSRSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBMT0dFKCJGYWlsZWQgdG8gaW5pdGlhbGl6ZSBoYXJkd2FyZSBvdXRwdXQgc3RyZWFtLCBzdGF0dXM6ICVkIiwgc3RhdHVzKTsKLSAgICAgICAgfQotICAgICAgICAKLSNpZmRlZiBXSVRIX0EyRFAKLSAgICAgICAgLy8gQ3JlYXRlIEEyRFAgaW50ZXJmYWNlCi0gICAgICAgIG1BMmRwQXVkaW9JbnRlcmZhY2UgPSBuZXcgQTJkcEF1ZGlvSW50ZXJmYWNlKCk7Ci0gICAgICAgIEF1ZGlvU3RyZWFtT3V0ICphMmRwT3V0cHV0ID0gbUEyZHBBdWRpb0ludGVyZmFjZS0+b3Blbk91dHB1dFN0cmVhbShBdWRpb1N5c3RlbTo6UENNXzE2X0JJVCwgMCwgMCwgJnN0YXR1cyk7Ci0gICAgICAgIGlmIChhMmRwT3V0cHV0KSB7Ci0gICAgICAgICAgICBtQTJkcE1peGVyVGhyZWFkID0gbmV3IE1peGVyVGhyZWFkKHRoaXMsIGEyZHBPdXRwdXQsIEF1ZGlvU3lzdGVtOjpBVURJT19PVVRQVVRfQTJEUCk7Ci0gICAgICAgICAgICBpZiAoaHdPdXRwdXQpIHsgIAotICAgICAgICAgICAgICAgIHVpbnQzMl90IGZyYW1lQ291bnQgPSAoKGEyZHBPdXRwdXQtPmJ1ZmZlclNpemUoKS9hMmRwT3V0cHV0LT5mcmFtZVNpemUoKSkgKiBod091dHB1dC0+c2FtcGxlUmF0ZSgpKSAvIGEyZHBPdXRwdXQtPnNhbXBsZVJhdGUoKTsKLSAgICAgICAgICAgICAgICBNaXhlclRocmVhZDo6T3V0cHV0VHJhY2sgKmEyZHBPdXRUcmFjayA9IG5ldyBNaXhlclRocmVhZDo6T3V0cHV0VHJhY2sobUEyZHBNaXhlclRocmVhZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGh3T3V0cHV0LT5zYW1wbGVSYXRlKCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBBdWRpb1N5c3RlbTo6UENNXzE2X0JJVCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGh3T3V0cHV0LT5jaGFubmVsQ291bnQoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZyYW1lQ291bnQpOwotICAgICAgICAgICAgICAgIG1IYXJkd2FyZU1peGVyVGhyZWFkLT5zZXRPdXB1dFRyYWNrKGEyZHBPdXRUcmFjayk7ICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgTE9HRSgiRmFpbGVkIHRvIGluaXRpYWxpemUgQTJEUCBvdXRwdXQgc3RyZWFtLCBzdGF0dXM6ICVkIiwgc3RhdHVzKTsKLSAgICAgICAgfQotI2VuZGlmCi0gCi0gICAgICAgIC8vIEZJWE1FIC0gdGhpcyBzaG91bGQgY29tZSBmcm9tIHNldHRpbmdzCi0gICAgICAgIHNldFJvdXRpbmcoQXVkaW9TeXN0ZW06Ok1PREVfTk9STUFMLCBBdWRpb1N5c3RlbTo6Uk9VVEVfU1BFQUtFUiwgQXVkaW9TeXN0ZW06OlJPVVRFX0FMTCk7Ci0gICAgICAgIHNldFJvdXRpbmcoQXVkaW9TeXN0ZW06Ok1PREVfUklOR1RPTkUsIEF1ZGlvU3lzdGVtOjpST1VURV9TUEVBS0VSLCBBdWRpb1N5c3RlbTo6Uk9VVEVfQUxMKTsKLSAgICAgICAgc2V0Um91dGluZyhBdWRpb1N5c3RlbTo6TU9ERV9JTl9DQUxMLCBBdWRpb1N5c3RlbTo6Uk9VVEVfRUFSUElFQ0UsIEF1ZGlvU3lzdGVtOjpST1VURV9BTEwpOwotICAgICAgICBzZXRNb2RlKEF1ZGlvU3lzdGVtOjpNT0RFX05PUk1BTCk7Ci0KLSAgICAgICAgc2V0TWFzdGVyVm9sdW1lKDEuMGYpOwotICAgICAgICBzZXRNYXN0ZXJNdXRlKGZhbHNlKTsKLQotICAgICAgICAvLyBTdGFydCByZWNvcmQgdGhyZWFkCi0gICAgICAgIG1BdWRpb1JlY29yZFRocmVhZCA9IG5ldyBBdWRpb1JlY29yZFRocmVhZChtQXVkaW9IYXJkd2FyZSk7Ci0gICAgICAgIGlmIChtQXVkaW9SZWNvcmRUaHJlYWQgIT0gMCkgewotICAgICAgICAgICAgbUF1ZGlvUmVjb3JkVGhyZWFkLT5ydW4oIkF1ZGlvUmVjb3JkVGhyZWFkIiwgUFJJT1JJVFlfVVJHRU5UX0FVRElPKTsgICAgICAgICAgICAKLSAgICAgICAgfQotICAgICB9IGVsc2UgewotICAgICAgICBMT0dFKCJDb3VsZG4ndCBldmVuIGluaXRpYWxpemUgdGhlIHN0dWJiZWQgYXVkaW8gaGFyZHdhcmUhIik7Ci0gICAgfQotCi0gICAgY2hhciB2YWx1ZVtQUk9QRVJUWV9WQUxVRV9NQVhdOwotICAgIHByb3BlcnR5X2dldCgicm8uYXVkaW8uc2lsZW50IiwgdmFsdWUsICIwIik7Ci0gICAgaWYgKGF0b2kodmFsdWUpKSB7Ci0gICAgICAgIExPR0QoIlNpbGVuY2UgaXMgZ29sZGVuIik7Ci0gICAgICAgIHNldE1hc3Rlck11dGUodHJ1ZSk7Ci0gICAgfQotfQotCi1BdWRpb0ZsaW5nZXI6On5BdWRpb0ZsaW5nZXIoKQotewotICAgIGlmIChtQXVkaW9SZWNvcmRUaHJlYWQgIT0gMCkgewotICAgICAgICBtQXVkaW9SZWNvcmRUaHJlYWQtPmV4aXQoKTsKLSAgICAgICAgbUF1ZGlvUmVjb3JkVGhyZWFkLmNsZWFyKCk7ICAgICAgICAKLSAgICB9Ci0gICAgbUhhcmR3YXJlTWl4ZXJUaHJlYWQuY2xlYXIoKTsKLSAgICBkZWxldGUgbUF1ZGlvSGFyZHdhcmU7Ci0gICAgLy8gZGVsZXRpbmcgbUEyZHBBdWRpb0ludGVyZmFjZSBhbHNvIGRlbGV0ZXMgbUEyZHBPdXRwdXQ7Ci0jaWZkZWYgV0lUSF9BMkRQCi0gICAgbUEyZHBNaXhlclRocmVhZC5jbGVhcigpOwotICAgIGRlbGV0ZSBtQTJkcEF1ZGlvSW50ZXJmYWNlOwotI2VuZGlmCi19Ci0KLQotI2lmZGVmIFdJVEhfQTJEUAotdm9pZCBBdWRpb0ZsaW5nZXI6OnNldEEyZHBFbmFibGVkKGJvb2wgZW5hYmxlKQotewotICAgIExPR1ZfSUYoZW5hYmxlLCAic2V0IG91dHB1dCB0byBBMkRQXG4iKTsKLSAgICBMT0dWX0lGKCFlbmFibGUsICJzZXQgb3V0cHV0IHRvIGhhcmR3YXJlIGF1ZGlvXG4iKTsKLQotICAgIG1BMmRwRW5hYmxlZFJlcSA9IGVuYWJsZTsKLSAgICBtQTJkcE1peGVyVGhyZWFkLT53YWtlVXAoKTsKLX0KLSNlbmRpZiAvLyBXSVRIX0EyRFAKLQotYm9vbCBBdWRpb0ZsaW5nZXI6OnN0cmVhbUZvcmNlZFRvU3BlYWtlcihpbnQgc3RyZWFtVHlwZSkKLXsKLSAgICAvLyBOT1RFIHRoYXQgc3RyZWFtcyBsaXN0ZWQgaGVyZSBtdXN0IG5vdCBiZSByb3V0ZWQgdG8gQTJEUCBieSBkZWZhdWx0OgotICAgIC8vIEF1ZGlvU3lzdGVtOjpyb3V0ZWRUb0EyZHBPdXRwdXQoc3RyZWFtVHlwZSkgPT0gZmFsc2UKLSAgICByZXR1cm4gKHN0cmVhbVR5cGUgPT0gQXVkaW9TeXN0ZW06OlJJTkcgfHwKLSAgICAgICAgICAgIHN0cmVhbVR5cGUgPT0gQXVkaW9TeXN0ZW06OkFMQVJNIHx8Ci0gICAgICAgICAgICBzdHJlYW1UeXBlID09IEF1ZGlvU3lzdGVtOjpOT1RJRklDQVRJT04pOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6OmR1bXBDbGllbnRzKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBjb25zdCBzaXplX3QgU0laRSA9IDI1NjsKLSAgICBjaGFyIGJ1ZmZlcltTSVpFXTsKLSAgICBTdHJpbmc4IHJlc3VsdDsKLQotICAgIHJlc3VsdC5hcHBlbmQoIkNsaWVudHM6XG4iKTsKLSAgICBmb3IgKHNpemVfdCBpID0gMDsgaSA8IG1DbGllbnRzLnNpemUoKTsgKytpKSB7Ci0gICAgICAgIHdwPENsaWVudD4gd0NsaWVudCA9IG1DbGllbnRzLnZhbHVlQXQoaSk7Ci0gICAgICAgIGlmICh3Q2xpZW50ICE9IDApIHsKLSAgICAgICAgICAgIHNwPENsaWVudD4gY2xpZW50ID0gd0NsaWVudC5wcm9tb3RlKCk7Ci0gICAgICAgICAgICBpZiAoY2xpZW50ICE9IDApIHsKLSAgICAgICAgICAgICAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICIgIHBpZDogJWRcbiIsIGNsaWVudC0+cGlkKCkpOwotICAgICAgICAgICAgICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICB3cml0ZShmZCwgcmVzdWx0LnN0cmluZygpLCByZXN1bHQuc2l6ZSgpKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLQotc3RhdHVzX3QgQXVkaW9GbGluZ2VyOjpkdW1wSW50ZXJuYWxzKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBjb25zdCBzaXplX3QgU0laRSA9IDI1NjsKLSAgICBjaGFyIGJ1ZmZlcltTSVpFXTsKLSAgICBTdHJpbmc4IHJlc3VsdDsKLQotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIkhhcmR3YXJlIHN0YXR1czogJWRcbiIsIG1IYXJkd2FyZVN0YXR1cyk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHdyaXRlKGZkLCByZXN1bHQuc3RyaW5nKCksIHJlc3VsdC5zaXplKCkpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9GbGluZ2VyOjpkdW1wUGVybWlzc2lvbkRlbmlhbChpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpCi17Ci0gICAgY29uc3Qgc2l6ZV90IFNJWkUgPSAyNTY7Ci0gICAgY2hhciBidWZmZXJbU0laRV07Ci0gICAgU3RyaW5nOCByZXN1bHQ7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiUGVybWlzc2lvbiBEZW5pYWw6ICIKLSAgICAgICAgICAgICJjYW4ndCBkdW1wIEF1ZGlvRmxpbmdlciBmcm9tIHBpZD0lZCwgdWlkPSVkXG4iLAotICAgICAgICAgICAgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpLAotICAgICAgICAgICAgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1VpZCgpKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgd3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6OmR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKQotewotICAgIGlmIChjaGVja0NhbGxpbmdQZXJtaXNzaW9uKFN0cmluZzE2KCJhbmRyb2lkLnBlcm1pc3Npb24uRFVNUCIpKSA9PSBmYWxzZSkgewotICAgICAgICBkdW1wUGVybWlzc2lvbkRlbmlhbChmZCwgYXJncyk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgQXV0b011dGV4IGxvY2soJm1Mb2NrKTsKLQotICAgICAgICBkdW1wQ2xpZW50cyhmZCwgYXJncyk7Ci0gICAgICAgIGR1bXBJbnRlcm5hbHMoZmQsIGFyZ3MpOwotICAgICAgICBtSGFyZHdhcmVNaXhlclRocmVhZC0+ZHVtcChmZCwgYXJncyk7Ci0jaWZkZWYgV0lUSF9BMkRQCi0gICAgICAgIG1BMmRwTWl4ZXJUaHJlYWQtPmR1bXAoZmQsIGFyZ3MpOwotI2VuZGlmCi0KLSAgICAgICAgLy8gZHVtcCByZWNvcmQgY2xpZW50Ci0gICAgICAgIGlmIChtQXVkaW9SZWNvcmRUaHJlYWQgIT0gMCkgbUF1ZGlvUmVjb3JkVGhyZWFkLT5kdW1wKGZkLCBhcmdzKTsKLQotICAgICAgICBpZiAobUF1ZGlvSGFyZHdhcmUpIHsKLSAgICAgICAgICAgIG1BdWRpb0hhcmR3YXJlLT5kdW1wU3RhdGUoZmQsIGFyZ3MpOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLy8gSUF1ZGlvRmxpbmdlciBpbnRlcmZhY2UKLQotCi1zcDxJQXVkaW9UcmFjaz4gQXVkaW9GbGluZ2VyOjpjcmVhdGVUcmFjaygKLSAgICAgICAgcGlkX3QgcGlkLAotICAgICAgICBpbnQgc3RyZWFtVHlwZSwKLSAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgaW50IGZvcm1hdCwKLSAgICAgICAgaW50IGNoYW5uZWxDb3VudCwKLSAgICAgICAgaW50IGZyYW1lQ291bnQsCi0gICAgICAgIHVpbnQzMl90IGZsYWdzLAotICAgICAgICBjb25zdCBzcDxJTWVtb3J5PiYgc2hhcmVkQnVmZmVyLAotICAgICAgICBzdGF0dXNfdCAqc3RhdHVzKQotewotICAgIHNwPE1peGVyVGhyZWFkOjpUcmFjaz4gdHJhY2s7Ci0gICAgc3A8VHJhY2tIYW5kbGU+IHRyYWNrSGFuZGxlOwotICAgIHNwPENsaWVudD4gY2xpZW50OwotICAgIHdwPENsaWVudD4gd2NsaWVudDsKLSAgICBzdGF0dXNfdCBsU3RhdHVzOwotCi0gICAgaWYgKHN0cmVhbVR5cGUgPj0gQXVkaW9TeXN0ZW06Ok5VTV9TVFJFQU1fVFlQRVMpIHsKLSAgICAgICAgTE9HRSgiaW52YWxpZCBzdHJlYW0gdHlwZSIpOwotICAgICAgICBsU3RhdHVzID0gQkFEX1ZBTFVFOwotICAgICAgICBnb3RvIEV4aXQ7Ci0gICAgfQotCi0gICAgewotICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotCi0gICAgICAgIHdjbGllbnQgPSBtQ2xpZW50cy52YWx1ZUZvcihwaWQpOwotCi0gICAgICAgIGlmICh3Y2xpZW50ICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGNsaWVudCA9IHdjbGllbnQucHJvbW90ZSgpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgY2xpZW50ID0gbmV3IENsaWVudCh0aGlzLCBwaWQpOwotICAgICAgICAgICAgbUNsaWVudHMuYWRkKHBpZCwgY2xpZW50KTsKLSAgICAgICAgfQotI2lmZGVmIFdJVEhfQTJEUAotICAgICAgICBpZiAoaXNBMmRwRW5hYmxlZCgpICYmIEF1ZGlvU3lzdGVtOjpyb3V0ZWRUb0EyZHBPdXRwdXQoc3RyZWFtVHlwZSkpIHsKLSAgICAgICAgICAgIHRyYWNrID0gbUEyZHBNaXhlclRocmVhZC0+Y3JlYXRlVHJhY2soY2xpZW50LCBzdHJlYW1UeXBlLCBzYW1wbGVSYXRlLCBmb3JtYXQsCi0gICAgICAgICAgICAgICAgICAgIGNoYW5uZWxDb3VudCwgZnJhbWVDb3VudCwgc2hhcmVkQnVmZmVyLCAmbFN0YXR1cyk7ICAgICAgICAgICAgCi0gICAgICAgIH0gZWxzZSAKLSNlbmRpZgotICAgICAgICB7Ci0gICAgICAgICAgICB0cmFjayA9IG1IYXJkd2FyZU1peGVyVGhyZWFkLT5jcmVhdGVUcmFjayhjbGllbnQsIHN0cmVhbVR5cGUsIHNhbXBsZVJhdGUsIGZvcm1hdCwKLSAgICAgICAgICAgICAgICAgICAgY2hhbm5lbENvdW50LCBmcmFtZUNvdW50LCBzaGFyZWRCdWZmZXIsICZsU3RhdHVzKTsgICAgICAgICAgICAKLSAgICAgICAgfQotICAgICAgICBpZiAodHJhY2sgIT0gTlVMTCkgewotICAgICAgICAgICAgdHJhY2tIYW5kbGUgPSBuZXcgVHJhY2tIYW5kbGUodHJhY2spOwotICAgICAgICAgICAgbFN0YXR1cyA9IE5PX0VSUk9SOyAgICAgICAgICAgIAotICAgICAgICB9Ci0gICAgfQotCi1FeGl0OgotICAgIGlmKHN0YXR1cykgewotICAgICAgICAqc3RhdHVzID0gbFN0YXR1czsKLSAgICB9Ci0gICAgcmV0dXJuIHRyYWNrSGFuZGxlOwotfQotCi11aW50MzJfdCBBdWRpb0ZsaW5nZXI6OnNhbXBsZVJhdGUoaW50IG91dHB1dCkgY29uc3QKLXsKLSNpZmRlZiBXSVRIX0EyRFAKLSAgICAgaWYgKG91dHB1dCA9PSBBdWRpb1N5c3RlbTo6QVVESU9fT1VUUFVUX0EyRFApIHsKLSAgICAgICAgIHJldHVybiBtQTJkcE1peGVyVGhyZWFkLT5zYW1wbGVSYXRlKCk7Ci0gICAgIH0KLSNlbmRpZgotICAgICByZXR1cm4gbUhhcmR3YXJlTWl4ZXJUaHJlYWQtPnNhbXBsZVJhdGUoKTsKLX0KLQotaW50IEF1ZGlvRmxpbmdlcjo6Y2hhbm5lbENvdW50KGludCBvdXRwdXQpIGNvbnN0Ci17Ci0jaWZkZWYgV0lUSF9BMkRQCi0gICAgIGlmIChvdXRwdXQgPT0gQXVkaW9TeXN0ZW06OkFVRElPX09VVFBVVF9BMkRQKSB7Ci0gICAgICAgICByZXR1cm4gbUEyZHBNaXhlclRocmVhZC0+Y2hhbm5lbENvdW50KCk7Ci0gICAgIH0KLSNlbmRpZgotICAgICByZXR1cm4gbUhhcmR3YXJlTWl4ZXJUaHJlYWQtPmNoYW5uZWxDb3VudCgpOwotfQotCi1pbnQgQXVkaW9GbGluZ2VyOjpmb3JtYXQoaW50IG91dHB1dCkgY29uc3QKLXsKLSNpZmRlZiBXSVRIX0EyRFAKLSAgICAgaWYgKG91dHB1dCA9PSBBdWRpb1N5c3RlbTo6QVVESU9fT1VUUFVUX0EyRFApIHsKLSAgICAgICAgIHJldHVybiBtQTJkcE1peGVyVGhyZWFkLT5mb3JtYXQoKTsKLSAgICAgfQotI2VuZGlmCi0gICAgIHJldHVybiBtSGFyZHdhcmVNaXhlclRocmVhZC0+Zm9ybWF0KCk7Ci19Ci0KLXNpemVfdCBBdWRpb0ZsaW5nZXI6OmZyYW1lQ291bnQoaW50IG91dHB1dCkgY29uc3QKLXsKLSNpZmRlZiBXSVRIX0EyRFAKLSAgICAgaWYgKG91dHB1dCA9PSBBdWRpb1N5c3RlbTo6QVVESU9fT1VUUFVUX0EyRFApIHsKLSAgICAgICAgIHJldHVybiBtQTJkcE1peGVyVGhyZWFkLT5mcmFtZUNvdW50KCk7Ci0gICAgIH0KLSNlbmRpZgotICAgICByZXR1cm4gbUhhcmR3YXJlTWl4ZXJUaHJlYWQtPmZyYW1lQ291bnQoKTsKLX0KLQotdWludDMyX3QgQXVkaW9GbGluZ2VyOjpsYXRlbmN5KGludCBvdXRwdXQpIGNvbnN0Ci17Ci0jaWZkZWYgV0lUSF9BMkRQCi0gICAgIGlmIChvdXRwdXQgPT0gQXVkaW9TeXN0ZW06OkFVRElPX09VVFBVVF9BMkRQKSB7Ci0gICAgICAgICByZXR1cm4gbUEyZHBNaXhlclRocmVhZC0+bGF0ZW5jeSgpOwotICAgICB9Ci0jZW5kaWYKLSAgICAgcmV0dXJuIG1IYXJkd2FyZU1peGVyVGhyZWFkLT5sYXRlbmN5KCk7Ci19Ci0KLXN0YXR1c190IEF1ZGlvRmxpbmdlcjo6c2V0TWFzdGVyVm9sdW1lKGZsb2F0IHZhbHVlKQotewotICAgIC8vIGNoZWNrIGNhbGxpbmcgcGVybWlzc2lvbnMKLSAgICBpZiAoIXNldHRpbmdzQWxsb3dlZCgpKSB7Ci0gICAgICAgIHJldHVybiBQRVJNSVNTSU9OX0RFTklFRDsKLSAgICB9Ci0KLSAgICAvLyB3aGVuIGh3IHN1cHBvcnRzIG1hc3RlciB2b2x1bWUsIGRvbid0IHNjYWxlIGluIHN3IG1peGVyCi0gICAgQXV0b011dGV4IGxvY2sobUhhcmR3YXJlTG9jayk7Ci0gICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfU0VUX01BU1RFUl9WT0xVTUU7Ci0gICAgaWYgKG1BdWRpb0hhcmR3YXJlLT5zZXRNYXN0ZXJWb2x1bWUodmFsdWUpID09IE5PX0VSUk9SKSB7Ci0gICAgICAgIHZhbHVlID0gMS4wZjsKLSAgICB9Ci0gICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfSURMRTsKLSAgICBtSGFyZHdhcmVNaXhlclRocmVhZC0+c2V0TWFzdGVyVm9sdW1lKHZhbHVlKTsKLSNpZmRlZiBXSVRIX0EyRFAKLSAgICBtQTJkcE1peGVyVGhyZWFkLT5zZXRNYXN0ZXJWb2x1bWUodmFsdWUpOwotI2VuZGlmCi0gICAgCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6OnNldFJvdXRpbmcoaW50IG1vZGUsIHVpbnQzMl90IHJvdXRlcywgdWludDMyX3QgbWFzaykKLXsKLSAgICBzdGF0dXNfdCBlcnIgPSBOT19FUlJPUjsKLQotICAgIC8vIGNoZWNrIGNhbGxpbmcgcGVybWlzc2lvbnMKLSAgICBpZiAoIXNldHRpbmdzQWxsb3dlZCgpKSB7Ci0gICAgICAgIHJldHVybiBQRVJNSVNTSU9OX0RFTklFRDsKLSAgICB9Ci0gICAgaWYgKChtb2RlIDwgQXVkaW9TeXN0ZW06Ok1PREVfQ1VSUkVOVCkgfHwgKG1vZGUgPj0gQXVkaW9TeXN0ZW06Ok5VTV9NT0RFUykpIHsKLSAgICAgICAgTE9HVygiSWxsZWdhbCB2YWx1ZTogc2V0Um91dGluZyglZCwgJXUsICV1KSIsIG1vZGUsIHJvdXRlcywgbWFzayk7Ci0gICAgICAgIHJldHVybiBCQURfVkFMVUU7Ci0gICAgfQotCi0jaWZkZWYgV0lUSF9BMkRQCi0gICAgTE9HRCgic2V0Um91dGluZyAlZCAlZCAlZCwgdGlkICVkLCBjYWxsaW5nIHRpZCAlZFxuIiwgbW9kZSwgcm91dGVzLCBtYXNrLCBnZXR0aWQoKSwgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpKTsKLSAgICBpZiAobW9kZSA9PSBBdWRpb1N5c3RlbTo6TU9ERV9OT1JNQUwgJiYgCi0gICAgICAgICAgICAobWFzayAmIEF1ZGlvU3lzdGVtOjpST1VURV9CTFVFVE9PVEhfQTJEUCkpIHsKLSAgICAgICAgQXV0b011dGV4IGxvY2soJm1Mb2NrKTsKLQotICAgICAgICBib29sIGVuYWJsZUEyZHAgPSBmYWxzZTsKLSAgICAgICAgaWYgKHJvdXRlcyAmIEF1ZGlvU3lzdGVtOjpST1VURV9CTFVFVE9PVEhfQTJEUCkgewotICAgICAgICAgICAgZW5hYmxlQTJkcCA9IHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgc2V0QTJkcEVuYWJsZWQoZW5hYmxlQTJkcCk7Ci0gICAgICAgIExPR1YoInNldE91dHB1dCBkb25lXG4iKTsKLSAgICB9Ci0jZW5kaWYKLQotICAgIC8vIGRvIG5vdGhpbmcgaWYgb25seSBBMkRQIHJvdXRpbmcgaXMgYWZmZWN0ZWQKLSAgICBtYXNrICY9IH5BdWRpb1N5c3RlbTo6Uk9VVEVfQkxVRVRPT1RIX0EyRFA7Ci0gICAgaWYgKG1hc2spIHsKLSAgICAgICAgQXV0b011dGV4IGxvY2sobUhhcmR3YXJlTG9jayk7Ci0gICAgICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX0hXX0dFVF9ST1VUSU5HOwotICAgICAgICB1aW50MzJfdCByOwotICAgICAgICBlcnIgPSBtQXVkaW9IYXJkd2FyZS0+Z2V0Um91dGluZyhtb2RlLCAmcik7Ci0gICAgICAgIGlmIChlcnIgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgIHIgPSAociAmIH5tYXNrKSB8IChyb3V0ZXMgJiBtYXNrKTsKLSAgICAgICAgICAgIGlmIChtb2RlID09IEF1ZGlvU3lzdGVtOjpNT0RFX05PUk1BTCB8fCAKLSAgICAgICAgICAgICAgICAobW9kZSA9PSBBdWRpb1N5c3RlbTo6TU9ERV9DVVJSRU5UICYmIGdldE1vZGUoKSA9PSBBdWRpb1N5c3RlbTo6TU9ERV9OT1JNQUwpKSB7Ci0gICAgICAgICAgICAgICAgbVNhdmVkUm91dGUgPSByOwotICAgICAgICAgICAgICAgIHIgfD0gbUZvcmNlZFJvdXRlOwotICAgICAgICAgICAgICAgIExPR1YoInNldFJvdXRpbmcgbVNhdmVkUm91dGUgJTA4eCBtRm9yY2VkUm91dGUgJTA4eFxuIiwgbVNhdmVkUm91dGUsIG1Gb3JjZWRSb3V0ZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19TRVRfUk9VVElORzsKLSAgICAgICAgICAgIGVyciA9IG1BdWRpb0hhcmR3YXJlLT5zZXRSb3V0aW5nKG1vZGUsIHIpOwotICAgICAgICB9Ci0gICAgICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX0hXX0lETEU7Ci0gICAgfQotICAgIHJldHVybiBlcnI7Ci19Ci0KLXVpbnQzMl90IEF1ZGlvRmxpbmdlcjo6Z2V0Um91dGluZyhpbnQgbW9kZSkgY29uc3QKLXsKLSAgICB1aW50MzJfdCByb3V0ZXMgPSAwOwotICAgIGlmICgobW9kZSA+PSBBdWRpb1N5c3RlbTo6TU9ERV9DVVJSRU5UKSAmJiAobW9kZSA8IEF1ZGlvU3lzdGVtOjpOVU1fTU9ERVMpKSB7Ci0gICAgICAgIGlmIChtb2RlID09IEF1ZGlvU3lzdGVtOjpNT0RFX05PUk1BTCB8fCAKLSAgICAgICAgICAgIChtb2RlID09IEF1ZGlvU3lzdGVtOjpNT0RFX0NVUlJFTlQgJiYgZ2V0TW9kZSgpID09IEF1ZGlvU3lzdGVtOjpNT0RFX05PUk1BTCkpIHsKLSAgICAgICAgICAgIHJvdXRlcyA9IG1TYXZlZFJvdXRlOyAgICAgICAgICAgICAgICAKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX0hXX0dFVF9ST1VUSU5HOwotICAgICAgICAgICAgbUF1ZGlvSGFyZHdhcmUtPmdldFJvdXRpbmcobW9kZSwgJnJvdXRlcyk7Ci0gICAgICAgICAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19JRExFOwotICAgICAgICB9Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgTE9HVygiSWxsZWdhbCB2YWx1ZTogZ2V0Um91dGluZyglZCkiLCBtb2RlKTsKLSAgICB9Ci0gICAgcmV0dXJuIHJvdXRlczsKLX0KLQotc3RhdHVzX3QgQXVkaW9GbGluZ2VyOjpzZXRNb2RlKGludCBtb2RlKQotewotICAgIC8vIGNoZWNrIGNhbGxpbmcgcGVybWlzc2lvbnMKLSAgICBpZiAoIXNldHRpbmdzQWxsb3dlZCgpKSB7Ci0gICAgICAgIHJldHVybiBQRVJNSVNTSU9OX0RFTklFRDsKLSAgICB9Ci0gICAgaWYgKChtb2RlIDwgMCkgfHwgKG1vZGUgPj0gQXVkaW9TeXN0ZW06Ok5VTV9NT0RFUykpIHsKLSAgICAgICAgTE9HVygiSWxsZWdhbCB2YWx1ZTogc2V0TW9kZSglZCkiLCBtb2RlKTsKLSAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLSAgICB9Ci0KLSAgICBBdXRvTXV0ZXggbG9jayhtSGFyZHdhcmVMb2NrKTsKLSAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19TRVRfTU9ERTsKLSAgICBzdGF0dXNfdCByZXQgPSBtQXVkaW9IYXJkd2FyZS0+c2V0TW9kZShtb2RlKTsKLSAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19JRExFOwotICAgIHJldHVybiByZXQ7Ci19Ci0KLWludCBBdWRpb0ZsaW5nZXI6OmdldE1vZGUoKSBjb25zdAotewotICAgIGludCBtb2RlID0gQXVkaW9TeXN0ZW06Ok1PREVfSU5WQUxJRDsKLSAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19TRVRfTU9ERTsKLSAgICBtQXVkaW9IYXJkd2FyZS0+Z2V0TW9kZSgmbW9kZSk7Ci0gICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfSURMRTsKLSAgICByZXR1cm4gbW9kZTsKLX0KLQotc3RhdHVzX3QgQXVkaW9GbGluZ2VyOjpzZXRNaWNNdXRlKGJvb2wgc3RhdGUpCi17Ci0gICAgLy8gY2hlY2sgY2FsbGluZyBwZXJtaXNzaW9ucwotICAgIGlmICghc2V0dGluZ3NBbGxvd2VkKCkpIHsKLSAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOwotICAgIH0KLQotICAgIEF1dG9NdXRleCBsb2NrKG1IYXJkd2FyZUxvY2spOwotICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX0hXX1NFVF9NSUNfTVVURTsKLSAgICBzdGF0dXNfdCByZXQgPSBtQXVkaW9IYXJkd2FyZS0+c2V0TWljTXV0ZShzdGF0ZSk7Ci0gICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfSURMRTsKLSAgICByZXR1cm4gcmV0OwotfQotCi1ib29sIEF1ZGlvRmxpbmdlcjo6Z2V0TWljTXV0ZSgpIGNvbnN0Ci17Ci0gICAgYm9vbCBzdGF0ZSA9IEF1ZGlvU3lzdGVtOjpNT0RFX0lOVkFMSUQ7Ci0gICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfR0VUX01JQ19NVVRFOwotICAgIG1BdWRpb0hhcmR3YXJlLT5nZXRNaWNNdXRlKCZzdGF0ZSk7Ci0gICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfSURMRTsKLSAgICByZXR1cm4gc3RhdGU7Ci19Ci0KLXN0YXR1c190IEF1ZGlvRmxpbmdlcjo6c2V0TWFzdGVyTXV0ZShib29sIG11dGVkKQotewotICAgIC8vIGNoZWNrIGNhbGxpbmcgcGVybWlzc2lvbnMKLSAgICBpZiAoIXNldHRpbmdzQWxsb3dlZCgpKSB7Ci0gICAgICAgIHJldHVybiBQRVJNSVNTSU9OX0RFTklFRDsKLSAgICB9Ci0gICAgbUhhcmR3YXJlTWl4ZXJUaHJlYWQtPnNldE1hc3Rlck11dGUobXV0ZWQpOwotI2lmZGVmIFdJVEhfQTJEUAotICAgIG1BMmRwTWl4ZXJUaHJlYWQtPnNldE1hc3Rlck11dGUobXV0ZWQpOwotI2VuZGlmCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1mbG9hdCBBdWRpb0ZsaW5nZXI6Om1hc3RlclZvbHVtZSgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1IYXJkd2FyZU1peGVyVGhyZWFkLT5tYXN0ZXJWb2x1bWUoKTsKLX0KLQotYm9vbCBBdWRpb0ZsaW5nZXI6Om1hc3Rlck11dGUoKSBjb25zdAotewotICAgIHJldHVybiBtSGFyZHdhcmVNaXhlclRocmVhZC0+bWFzdGVyTXV0ZSgpOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6OnNldFN0cmVhbVZvbHVtZShpbnQgc3RyZWFtLCBmbG9hdCB2YWx1ZSkKLXsKLSAgICAvLyBjaGVjayBjYWxsaW5nIHBlcm1pc3Npb25zCi0gICAgaWYgKCFzZXR0aW5nc0FsbG93ZWQoKSkgewotICAgICAgICByZXR1cm4gUEVSTUlTU0lPTl9ERU5JRUQ7Ci0gICAgfQotCi0gICAgaWYgKHVpbnQzMl90KHN0cmVhbSkgPj0gQXVkaW9TeXN0ZW06Ok5VTV9TVFJFQU1fVFlQRVMpIHsKLSAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLSAgICB9Ci0KLSAgICBtSGFyZHdhcmVNaXhlclRocmVhZC0+c2V0U3RyZWFtVm9sdW1lKHN0cmVhbSwgdmFsdWUpOwotI2lmZGVmIFdJVEhfQTJEUAotICAgIG1BMmRwTWl4ZXJUaHJlYWQtPnNldFN0cmVhbVZvbHVtZShzdHJlYW0sIHZhbHVlKTsKLSNlbmRpZgotCi0gICAgc3RhdHVzX3QgcmV0ID0gTk9fRVJST1I7Ci0gICAgaWYgKHN0cmVhbSA9PSBBdWRpb1N5c3RlbTo6Vk9JQ0VfQ0FMTCB8fAotICAgICAgICBzdHJlYW0gPT0gQXVkaW9TeXN0ZW06OkJMVUVUT09USF9TQ08pIHsKLSAgICAgICAgCi0gICAgICAgIGlmIChzdHJlYW0gPT0gQXVkaW9TeXN0ZW06OlZPSUNFX0NBTEwpIHsKLSAgICAgICAgICAgIHZhbHVlID0gKGZsb2F0KUF1ZGlvU3lzdGVtOjpsb2dUb0xpbmVhcih2YWx1ZSkvMTAwLjBmOwotICAgICAgICB9IGVsc2UgeyAvLyAodHlwZSA9PSBBdWRpb1N5c3RlbTo6QkxVRVRPT1RIX1NDTykKLSAgICAgICAgICAgIHZhbHVlID0gMS4wZjsKLSAgICAgICAgfQotCi0gICAgICAgIEF1dG9NdXRleCBsb2NrKG1IYXJkd2FyZUxvY2spOwotICAgICAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19TRVRfVk9JQ0VfVk9MVU1FOwotICAgICAgICByZXQgPSBtQXVkaW9IYXJkd2FyZS0+c2V0Vm9pY2VWb2x1bWUodmFsdWUpOwotICAgICAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19JRExFOwotICAgIH0KLQotICAgIHJldHVybiByZXQ7Ci19Ci0KLXN0YXR1c190IEF1ZGlvRmxpbmdlcjo6c2V0U3RyZWFtTXV0ZShpbnQgc3RyZWFtLCBib29sIG11dGVkKQotewotICAgIC8vIGNoZWNrIGNhbGxpbmcgcGVybWlzc2lvbnMKLSAgICBpZiAoIXNldHRpbmdzQWxsb3dlZCgpKSB7Ci0gICAgICAgIHJldHVybiBQRVJNSVNTSU9OX0RFTklFRDsKLSAgICB9Ci0KLSAgICBpZiAodWludDMyX3Qoc3RyZWFtKSA+PSBBdWRpb1N5c3RlbTo6TlVNX1NUUkVBTV9UWVBFUykgewotICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOwotICAgIH0KLSAgICAKLSNpZmRlZiBXSVRIX0EyRFAKLSAgICBtQTJkcE1peGVyVGhyZWFkLT5zZXRTdHJlYW1NdXRlKHN0cmVhbSwgbXV0ZWQpOwotI2VuZGlmCi0gICAgaWYgKHN0cmVhbSA9PSBBdWRpb1N5c3RlbTo6TVVTSUMpIAotICAgIHsKLSAgICAgICAgQXV0b011dGV4IGxvY2soJm1IYXJkd2FyZUxvY2spOwotICAgICAgICBpZiAobUZvcmNlZFJvdXRlICE9IDApCi0gICAgICAgICAgICBtTXVzaWNNdXRlU2F2ZWQgPSBtdXRlZDsKLSAgICAgICAgZWxzZQotICAgICAgICAgICAgbUhhcmR3YXJlTWl4ZXJUaHJlYWQtPnNldFN0cmVhbU11dGUoc3RyZWFtLCBtdXRlZCk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgbUhhcmR3YXJlTWl4ZXJUaHJlYWQtPnNldFN0cmVhbU11dGUoc3RyZWFtLCBtdXRlZCk7Ci0gICAgfQotCi0gICAgCi0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLWZsb2F0IEF1ZGlvRmxpbmdlcjo6c3RyZWFtVm9sdW1lKGludCBzdHJlYW0pIGNvbnN0Ci17Ci0gICAgaWYgKHVpbnQzMl90KHN0cmVhbSkgPj0gQXVkaW9TeXN0ZW06Ok5VTV9TVFJFQU1fVFlQRVMpIHsKLSAgICAgICAgcmV0dXJuIDAuMGY7Ci0gICAgfQotICAgIHJldHVybiBtSGFyZHdhcmVNaXhlclRocmVhZC0+c3RyZWFtVm9sdW1lKHN0cmVhbSk7Ci19Ci0KLWJvb2wgQXVkaW9GbGluZ2VyOjpzdHJlYW1NdXRlKGludCBzdHJlYW0pIGNvbnN0Ci17Ci0gICAgaWYgKHVpbnQzMl90KHN0cmVhbSkgPj0gQXVkaW9TeXN0ZW06Ok5VTV9TVFJFQU1fVFlQRVMpIHsKLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotICAgIAotICAgIGlmIChzdHJlYW0gPT0gQXVkaW9TeXN0ZW06Ok1VU0lDICYmIG1Gb3JjZWRSb3V0ZSAhPSAwKSAKLSAgICB7Ci0gICAgICAgIHJldHVybiBtTXVzaWNNdXRlU2F2ZWQ7Ci0gICAgfQotICAgIHJldHVybiBtSGFyZHdhcmVNaXhlclRocmVhZC0+c3RyZWFtTXV0ZShzdHJlYW0pOwotfQotCi1ib29sIEF1ZGlvRmxpbmdlcjo6aXNNdXNpY0FjdGl2ZSgpIGNvbnN0Ci17Ci0gI2lmZGVmIFdJVEhfQTJEUAotICAgICBpZiAoaXNBMmRwRW5hYmxlZCgpKSB7Ci0gICAgICAgICByZXR1cm4gbUEyZHBNaXhlclRocmVhZC0+aXNNdXNpY0FjdGl2ZSgpOwotICAgICB9Ci0gI2VuZGlmCi0gICAgcmV0dXJuIG1IYXJkd2FyZU1peGVyVGhyZWFkLT5pc011c2ljQWN0aXZlKCk7Ci19Ci0KLXN0YXR1c190IEF1ZGlvRmxpbmdlcjo6c2V0UGFyYW1ldGVyKGNvbnN0IGNoYXIqIGtleSwgY29uc3QgY2hhciogdmFsdWUpCi17Ci0gICAgc3RhdHVzX3QgcmVzdWx0LCByZXN1bHQyOwotICAgIEF1dG9NdXRleCBsb2NrKG1IYXJkd2FyZUxvY2spOwotICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX1NFVF9QQVJBTUVURVI7Ci0gICAgCi0gICAgTE9HVigic2V0UGFyYW1ldGVyKCkga2V5ICVzLCB2YWx1ZSAlcywgdGlkICVkLCBjYWxsaW5nIHRpZCAlZCIsIGtleSwgdmFsdWUsIGdldHRpZCgpLCBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5nZXRDYWxsaW5nUGlkKCkpOwotICAgIHJlc3VsdCA9IG1BdWRpb0hhcmR3YXJlLT5zZXRQYXJhbWV0ZXIoa2V5LCB2YWx1ZSk7Ci0gICAgaWYgKG1BMmRwQXVkaW9JbnRlcmZhY2UpIHsKLSAgICAgICAgcmVzdWx0MiA9IG1BMmRwQXVkaW9JbnRlcmZhY2UtPnNldFBhcmFtZXRlcihrZXksIHZhbHVlKTsKLSAgICAgICAgaWYgKHJlc3VsdDIpCi0gICAgICAgICAgICByZXN1bHQgPSByZXN1bHQyOwotICAgIH0KLSAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19JRExFOwotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLXNpemVfdCBBdWRpb0ZsaW5nZXI6OmdldElucHV0QnVmZmVyU2l6ZSh1aW50MzJfdCBzYW1wbGVSYXRlLCBpbnQgZm9ybWF0LCBpbnQgY2hhbm5lbENvdW50KQotewotICAgIHJldHVybiBtQXVkaW9IYXJkd2FyZS0+Z2V0SW5wdXRCdWZmZXJTaXplKHNhbXBsZVJhdGUsIGZvcm1hdCwgY2hhbm5lbENvdW50KTsKLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6OnJlZ2lzdGVyQ2xpZW50KGNvbnN0IHNwPElBdWRpb0ZsaW5nZXJDbGllbnQ+JiBjbGllbnQpCi17Ci0gICAgCi0gICAgTE9HVigicmVnaXN0ZXJDbGllbnQoKSAlcCwgdGlkICVkLCBjYWxsaW5nIHRpZCAlZCIsIGNsaWVudC5nZXQoKSwgZ2V0dGlkKCksIElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPmdldENhbGxpbmdQaWQoKSk7Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLQotICAgIHNwPElCaW5kZXI+IGJpbmRlciA9IGNsaWVudC0+YXNCaW5kZXIoKTsKLSAgICBpZiAobU5vdGlmaWNhdGlvbkNsaWVudHMuaW5kZXhPZihiaW5kZXIpIDwgMCkgewotICAgICAgICBMT0dWKCJBZGRpbmcgbm90aWZpY2F0aW9uIGNsaWVudCAlcCIsIGJpbmRlci5nZXQoKSk7Ci0gICAgICAgIGJpbmRlci0+bGlua1RvRGVhdGgodGhpcyk7Ci0gICAgICAgIG1Ob3RpZmljYXRpb25DbGllbnRzLmFkZChiaW5kZXIpOwotICAgICAgICBjbGllbnQtPmEyZHBFbmFibGVkQ2hhbmdlZChpc0EyZHBFbmFibGVkKCkpOwotICAgIH0KLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6OmJpbmRlckRpZWQoY29uc3Qgd3A8SUJpbmRlcj4mIHdobykgewotICAgIAotICAgIExPR1YoImJpbmRlckRpZWQoKSAlcCwgdGlkICVkLCBjYWxsaW5nIHRpZCAlZCIsIHdoby51bnNhZmVfZ2V0KCksIGdldHRpZCgpLCBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5nZXRDYWxsaW5nUGlkKCkpOwotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0KLSAgICBJQmluZGVyICpiaW5kZXIgPSB3aG8udW5zYWZlX2dldCgpOwotCi0gICAgaWYgKGJpbmRlciAhPSBOVUxMKSB7Ci0gICAgICAgIGludCBpbmRleCA9IG1Ob3RpZmljYXRpb25DbGllbnRzLmluZGV4T2YoYmluZGVyKTsKLSAgICAgICAgaWYgKGluZGV4ID49IDApIHsKLSAgICAgICAgICAgIExPR1YoIlJlbW92aW5nIG5vdGlmaWNhdGlvbiBjbGllbnQgJXAiLCBiaW5kZXIpOwotICAgICAgICAgICAgbU5vdGlmaWNhdGlvbkNsaWVudHMucmVtb3ZlQXQoaW5kZXgpOwotICAgICAgICB9Ci0gICAgfQotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6aGFuZGxlT3V0cHV0U3dpdGNoKCkKLXsKLSAgICBpZiAobUEyZHBFbmFibGVkICE9IG1BMmRwRW5hYmxlZFJlcSkKLSAgICB7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0KLSAgICAgICAgaWYgKG1BMmRwRW5hYmxlZCAhPSBtQTJkcEVuYWJsZWRSZXEpCi0gICAgICAgIHsKLSAgICAgICAgICAgIG1BMmRwRW5hYmxlZCA9IG1BMmRwRW5hYmxlZFJlcTsKLSAgICAgICAgICAgIFNvcnRlZFZlY3RvciA8IHNwPE1peGVyVGhyZWFkOjpUcmFjaz4gPiB0cmFja3M7Ci0gICAgICAgICAgICBTb3J0ZWRWZWN0b3IgPCB3cDxNaXhlclRocmVhZDo6VHJhY2s+ID4gYWN0aXZlVHJhY2tzOwotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyBXZSBob2xkIG1BMmRwTWl4ZXJUaHJlYWQgbUxvY2sgYWxyZWFkeSAKLSAgICAgICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChtSGFyZHdhcmVNaXhlclRocmVhZC0+bUxvY2spOwotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyBUcmFuc2ZlciB0cmFja3MgcGxheWluZyBvbiBNVVNJQyBzdHJlYW0gZnJvbSBvbmUgbWl4ZXIgdG8gdGhlIG90aGVyCi0gICAgICAgICAgICBpZiAobUEyZHBFbmFibGVkKSB7Ci0gICAgICAgICAgICAgICAgbUhhcmR3YXJlTWl4ZXJUaHJlYWQtPmdldFRyYWNrcyh0cmFja3MsIGFjdGl2ZVRyYWNrcyk7Ci0gICAgICAgICAgICAgICAgbUEyZHBNaXhlclRocmVhZC0+cHV0VHJhY2tzKHRyYWNrcywgYWN0aXZlVHJhY2tzKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgbUEyZHBNaXhlclRocmVhZC0+Z2V0VHJhY2tzKHRyYWNrcywgYWN0aXZlVHJhY2tzKTsKLSAgICAgICAgICAgICAgICBtSGFyZHdhcmVNaXhlclRocmVhZC0+cHV0VHJhY2tzKHRyYWNrcywgYWN0aXZlVHJhY2tzKTsKLSAgICAgICAgICAgIH0gICAgICAgICAgICAKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8gTm90aWZ5IEF1ZGlvU3lzdGVtIG9mIHRoZSBBMkRQIGFjdGl2YXRpb24vZGVhY3RpdmF0aW9uCi0gICAgICAgICAgICBzaXplX3Qgc2l6ZSA9IG1Ob3RpZmljYXRpb25DbGllbnRzLnNpemUoKTsKLSAgICAgICAgICAgIGZvciAoc2l6ZV90IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgc3A8SUJpbmRlcj4gYmluZGVyID0gbU5vdGlmaWNhdGlvbkNsaWVudHMuaXRlbUF0KGkpLnByb21vdGUoKTsKLSAgICAgICAgICAgICAgICBpZiAoYmluZGVyICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgTE9HVigiTm90aWZ5aW5nIG91dHB1dCBjaGFuZ2UgdG8gY2xpZW50ICVwIiwgYmluZGVyLmdldCgpKTsKLSAgICAgICAgICAgICAgICAgICAgc3A8SUF1ZGlvRmxpbmdlckNsaWVudD4gY2xpZW50ID0gaW50ZXJmYWNlX2Nhc3Q8SUF1ZGlvRmxpbmdlckNsaWVudD4gKGJpbmRlcik7Ci0gICAgICAgICAgICAgICAgICAgIGNsaWVudC0+YTJkcEVuYWJsZWRDaGFuZ2VkKG1BMmRwRW5hYmxlZCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBtSGFyZHdhcmVNaXhlclRocmVhZC0+d2FrZVVwKCk7Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpyZW1vdmVDbGllbnQocGlkX3QgcGlkKQotewotICAgIExPR1YoInJlbW92ZUNsaWVudCgpIHBpZCAlZCwgdGlkICVkLCBjYWxsaW5nIHRpZCAlZCIsIHBpZCwgZ2V0dGlkKCksIElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPmdldENhbGxpbmdQaWQoKSk7Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICBtQ2xpZW50cy5yZW1vdmVJdGVtKHBpZCk7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjp3YWtlVXAoKQotewotICAgIG1IYXJkd2FyZU1peGVyVGhyZWFkLT53YWtlVXAoKTsKLSNpZmRlZiBXSVRIX0EyRFAKLSAgICBtQTJkcE1peGVyVGhyZWFkLT53YWtlVXAoKTsKLSNlbmRpZiAvLyBXSVRIX0EyRFAKLX0KLQotYm9vbCBBdWRpb0ZsaW5nZXI6OmlzQTJkcEVuYWJsZWQoKSBjb25zdAotewotICAgIHJldHVybiBtQTJkcEVuYWJsZWRSZXE7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpoYW5kbGVGb3JjZWRTcGVha2VyUm91dGUoaW50IGNvbW1hbmQpCi17Ci0gICAgc3dpdGNoKGNvbW1hbmQpIHsKLSAgICBjYXNlIEFDVElWRV9UUkFDS19BRERFRDoKLSAgICAgICAgewotICAgICAgICAgICAgQXV0b011dGV4IGxvY2sobUhhcmR3YXJlTG9jayk7Ci0gICAgICAgICAgICBpZiAobUZvcmNlZFNwZWFrZXJDb3VudCsrID09IDApIHsKLSAgICAgICAgICAgICAgICBtUm91dGVSZXN0b3JlVGltZSA9IDA7Ci0gICAgICAgICAgICAgICAgbU11c2ljTXV0ZVNhdmVkID0gbUhhcmR3YXJlTWl4ZXJUaHJlYWQtPnN0cmVhbU11dGUoQXVkaW9TeXN0ZW06Ok1VU0lDKTsKLSAgICAgICAgICAgICAgICBpZiAobUZvcmNlZFJvdXRlID09IDAgJiYgIShtU2F2ZWRSb3V0ZSAmIEF1ZGlvU3lzdGVtOjpST1VURV9TUEVBS0VSKSkgewotICAgICAgICAgICAgICAgICAgICBMT0dWKCJSb3V0ZSBmb3JjZWQgdG8gU3BlYWtlciBPTiAlMDh4IiwgbVNhdmVkUm91dGUgfCBBdWRpb1N5c3RlbTo6Uk9VVEVfU1BFQUtFUik7Ci0gICAgICAgICAgICAgICAgICAgIG1IYXJkd2FyZU1peGVyVGhyZWFkLT5zZXRTdHJlYW1NdXRlKEF1ZGlvU3lzdGVtOjpNVVNJQywgdHJ1ZSk7Ci0gICAgICAgICAgICAgICAgICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX0hXX1NFVF9NQVNURVJfVk9MVU1FOwotICAgICAgICAgICAgICAgICAgICBtQXVkaW9IYXJkd2FyZS0+c2V0TWFzdGVyVm9sdW1lKDApOwotICAgICAgICAgICAgICAgICAgICB1c2xlZXAobUhhcmR3YXJlTWl4ZXJUaHJlYWQtPmxhdGVuY3koKSoxMDAwKTsKLSAgICAgICAgICAgICAgICAgICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfU0VUX1JPVVRJTkc7Ci0gICAgICAgICAgICAgICAgICAgIG1BdWRpb0hhcmR3YXJlLT5zZXRSb3V0aW5nKEF1ZGlvU3lzdGVtOjpNT0RFX05PUk1BTCwgbVNhdmVkUm91dGUgfCBBdWRpb1N5c3RlbTo6Uk9VVEVfU1BFQUtFUik7Ci0gICAgICAgICAgICAgICAgICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX0hXX0lETEU7Ci0gICAgICAgICAgICAgICAgICAgIC8vIGRlbGF5IHRyYWNrIHN0YXJ0IHNvIHRoYXQgYXVkaW8gaGFyZHdhcmUgaGFzIHRpbWUgdG8gc2l3dGNoIHJvdXRlcwotICAgICAgICAgICAgICAgICAgICB1c2xlZXAoa1N0YXJ0U2xlZXBUaW1lKTsKLSAgICAgICAgICAgICAgICAgICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfU0VUX01BU1RFUl9WT0xVTUU7Ci0gICAgICAgICAgICAgICAgICAgIG1BdWRpb0hhcmR3YXJlLT5zZXRNYXN0ZXJWb2x1bWUobUhhcmR3YXJlTWl4ZXJUaHJlYWQtPm1hc3RlclZvbHVtZSgpKTsKLSAgICAgICAgICAgICAgICAgICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfSURMRTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbUZvcmNlZFJvdXRlID0gQXVkaW9TeXN0ZW06OlJPVVRFX1NQRUFLRVI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBMT0dWKCJtRm9yY2VkU3BlYWtlckNvdW50IGluY3JlbWVudGVkIHRvICVkIiwgbUZvcmNlZFNwZWFrZXJDb3VudCk7Ci0gICAgICAgIH0KLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBBQ1RJVkVfVFJBQ0tfUkVNT1ZFRDoKLSAgICAgICAgewotICAgICAgICAgICAgQXV0b011dGV4IGxvY2sobUhhcmR3YXJlTG9jayk7Ci0gICAgICAgICAgICBpZiAobUZvcmNlZFNwZWFrZXJDb3VudCA+IDApewotICAgICAgICAgICAgICAgIGlmICgtLW1Gb3JjZWRTcGVha2VyQ291bnQgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICBtUm91dGVSZXN0b3JlVGltZSA9IHN5c3RlbVRpbWUoKSArIG1pbGxpc2Vjb25kcyhrU3RvcFNsZWVwVGltZS8xMDAwKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgTE9HVigibUZvcmNlZFNwZWFrZXJDb3VudCBkZWNyZW1lbnRlZCB0byAlZCIsIG1Gb3JjZWRTcGVha2VyQ291bnQpOyAgICAgICAgICAgIAotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBMT0dFKCJtRm9yY2VkU3BlYWtlckNvdW50IGlzIGFscmVhZHkgemVybyIpOyAgICAgICAgICAgIAotICAgICAgICAgICAgfSAgICAgICAgICAgIAotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgQ0hFQ0tfUk9VVEVfUkVTVE9SRV9USU1FOgotICAgIGNhc2UgRk9SQ0VfUk9VVEVfUkVTVE9SRToKLSAgICAgICAgaWYgKG1Sb3V0ZVJlc3RvcmVUaW1lKSB7Ci0gICAgICAgICAgICBBdXRvTXV0ZXggbG9jayhtSGFyZHdhcmVMb2NrKTsKLSAgICAgICAgICAgIGlmIChtUm91dGVSZXN0b3JlVGltZSAmJiAKLSAgICAgICAgICAgICAgIChzeXN0ZW1UaW1lKCkgPiBtUm91dGVSZXN0b3JlVGltZSB8fCBjb21tYW5kID09IEZPUkNFX1JPVVRFX1JFU1RPUkUpKSB7Ci0gICAgICAgICAgICAgICAgbUhhcmR3YXJlTWl4ZXJUaHJlYWQtPnNldFN0cmVhbU11dGUoQXVkaW9TeXN0ZW06Ok1VU0lDLCBtTXVzaWNNdXRlU2F2ZWQpOwotICAgICAgICAgICAgICAgIG1Gb3JjZWRSb3V0ZSA9IDA7Ci0gICAgICAgICAgICAgICAgaWYgKCEobVNhdmVkUm91dGUgJiBBdWRpb1N5c3RlbTo6Uk9VVEVfU1BFQUtFUikpIHsKLSAgICAgICAgICAgICAgICAgICAgbUhhcmR3YXJlU3RhdHVzID0gQVVESU9fSFdfU0VUX1JPVVRJTkc7Ci0gICAgICAgICAgICAgICAgICAgIG1BdWRpb0hhcmR3YXJlLT5zZXRSb3V0aW5nKEF1ZGlvU3lzdGVtOjpNT0RFX05PUk1BTCwgbVNhdmVkUm91dGUpOwotICAgICAgICAgICAgICAgICAgICBtSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19JRExFOwotICAgICAgICAgICAgICAgICAgICBMT0dWKCJSb3V0ZSBmb3JjZWQgdG8gU3BlYWtlciBPRkYgJTA4eCIsIG1TYXZlZFJvdXRlKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbVJvdXRlUmVzdG9yZVRpbWUgPSAwOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgIH0KLX0KLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6Ok1peGVyVGhyZWFkKGNvbnN0IHNwPEF1ZGlvRmxpbmdlcj4mIGF1ZGlvRmxpbmdlciwgQXVkaW9TdHJlYW1PdXQqIG91dHB1dCwgaW50IG91dHB1dFR5cGUpCi0gICAgOiAgIFRocmVhZChmYWxzZSksCi0gICAgICAgIG1BdWRpb0ZsaW5nZXIoYXVkaW9GbGluZ2VyKSwgbUF1ZGlvTWl4ZXIoMCksIG1PdXRwdXQob3V0cHV0KSwgbU91dHB1dFR5cGUob3V0cHV0VHlwZSksIAotICAgICAgICBtU2FtcGxlUmF0ZSgwKSwgbUZyYW1lQ291bnQoMCksIG1DaGFubmVsQ291bnQoMCksIG1Gb3JtYXQoMCksIG1NaXhCdWZmZXIoMCksCi0gICAgICAgIG1MYXN0V3JpdGVUaW1lKDApLCBtTnVtV3JpdGVzKDApLCBtTnVtRGVsYXllZFdyaXRlcygwKSwgbVN0YW5kYnkoZmFsc2UpLAotICAgICAgICBtSW5Xcml0ZShmYWxzZSkKLXsKLSAgICBtU2FtcGxlUmF0ZSA9IG91dHB1dC0+c2FtcGxlUmF0ZSgpOwotICAgIG1DaGFubmVsQ291bnQgPSBvdXRwdXQtPmNoYW5uZWxDb3VudCgpOwotCi0gICAgLy8gRklYTUUgLSBDdXJyZW50IG1peGVyIGltcGxlbWVudGF0aW9uIG9ubHkgc3VwcG9ydHMgc3RlcmVvIG91dHB1dAotICAgIGlmIChtQ2hhbm5lbENvdW50ID09IDEpIHsKLSAgICAgICAgTE9HRSgiSW52YWxpZCBhdWRpbyBoYXJkd2FyZSBjaGFubmVsIGNvdW50Iik7Ci0gICAgfQotCi0gICAgbUZvcm1hdCA9IG91dHB1dC0+Zm9ybWF0KCk7Ci0gICAgbUZyYW1lQ291bnQgPSBvdXRwdXQtPmJ1ZmZlclNpemUoKSAvIG91dHB1dC0+Y2hhbm5lbENvdW50KCkgLyBzaXplb2YoaW50MTZfdCk7Ci0gICAgbUF1ZGlvTWl4ZXIgPSBuZXcgQXVkaW9NaXhlcihtRnJhbWVDb3VudCwgb3V0cHV0LT5zYW1wbGVSYXRlKCkpOwotCi0gICAgLy8gRklYTUUgLSBDdXJyZW50IG1peGVyIGltcGxlbWVudGF0aW9uIG9ubHkgc3VwcG9ydHMgc3RlcmVvIG91dHB1dDogQWx3YXlzCi0gICAgLy8gQWxsb2NhdGUgYSBzdGVyZW8gYnVmZmVyIGV2ZW4gaWYgSFcgb3V0cHV0IGlzIG1vbm8uCi0gICAgbU1peEJ1ZmZlciA9IG5ldyBpbnQxNl90W21GcmFtZUNvdW50ICogMl07Ci0gICAgbWVtc2V0KG1NaXhCdWZmZXIsIDAsIG1GcmFtZUNvdW50ICogMiAqIHNpemVvZihpbnQxNl90KSk7Ci19Ci0KLUF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6On5NaXhlclRocmVhZCgpCi17Ci0gICAgZGVsZXRlIFtdIG1NaXhCdWZmZXI7Ci0gICAgZGVsZXRlIG1BdWRpb01peGVyOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBkdW1wSW50ZXJuYWxzKGZkLCBhcmdzKTsKLSAgICBkdW1wVHJhY2tzKGZkLCBhcmdzKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OmR1bXBUcmFja3MoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKQotewotICAgIGNvbnN0IHNpemVfdCBTSVpFID0gMjU2OwotICAgIGNoYXIgYnVmZmVyW1NJWkVdOwotICAgIFN0cmluZzggcmVzdWx0OwotCi0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiT3V0cHV0ICVkIG1peGVyIHRocmVhZCB0cmFja3NcbiIsIG1PdXRwdXRUeXBlKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgcmVzdWx0LmFwcGVuZCgiICAgTmFtZSBDbGllbiBUeXAgRm10IENobiBCdWYgUyBNIEYgU1JhdGUgTGVmdFYgUmlnaFYgU2VydiBVc2VyXG4iKTsKLSAgICBmb3IgKHNpemVfdCBpID0gMDsgaSA8IG1UcmFja3Muc2l6ZSgpOyArK2kpIHsKLSAgICAgICAgd3A8VHJhY2s+IHdUcmFjayA9IG1UcmFja3NbaV07Ci0gICAgICAgIGlmICh3VHJhY2sgIT0gMCkgewotICAgICAgICAgICAgc3A8VHJhY2s+IHRyYWNrID0gd1RyYWNrLnByb21vdGUoKTsKLSAgICAgICAgICAgIGlmICh0cmFjayAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgdHJhY2stPmR1bXAoYnVmZmVyLCBTSVpFKTsKLSAgICAgICAgICAgICAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJPdXRwdXQgJWQgbWl4ZXIgdGhyZWFkIGFjdGl2ZSB0cmFja3NcbiIsIG1PdXRwdXRUeXBlKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgcmVzdWx0LmFwcGVuZCgiICAgTmFtZSBDbGllbiBUeXAgRm10IENobiBCdWYgUyBNIEYgU1JhdGUgTGVmdFYgUmlnaFYgU2VydiBVc2VyXG4iKTsKLSAgICBmb3IgKHNpemVfdCBpID0gMDsgaSA8IG1BY3RpdmVUcmFja3Muc2l6ZSgpOyArK2kpIHsKLSAgICAgICAgd3A8VHJhY2s+IHdUcmFjayA9IG1UcmFja3NbaV07Ci0gICAgICAgIGlmICh3VHJhY2sgIT0gMCkgewotICAgICAgICAgICAgc3A8VHJhY2s+IHRyYWNrID0gd1RyYWNrLnByb21vdGUoKTsKLSAgICAgICAgICAgIGlmICh0cmFjayAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgdHJhY2stPmR1bXAoYnVmZmVyLCBTSVpFKTsKLSAgICAgICAgICAgICAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgd3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpkdW1wSW50ZXJuYWxzKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBjb25zdCBzaXplX3QgU0laRSA9IDI1NjsKLSAgICBjaGFyIGJ1ZmZlcltTSVpFXTsKLSAgICBTdHJpbmc4IHJlc3VsdDsKLQotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIk91dHB1dCAlZCBtaXhlciB0aHJlYWQgaW50ZXJuYWxzXG4iLCBtT3V0cHV0VHlwZSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIkF1ZGlvTWl4ZXIgdHJhY2tzOiAlMDh4XG4iLCBtQXVkaW9NaXhlci0+dHJhY2tOYW1lcygpKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAibGFzdCB3cml0ZSBvY2N1cnJlZCAobXNlY3MpOiAlbGx1XG4iLCBuczJtcyhzeXN0ZW1UaW1lKCkgLSBtTGFzdFdyaXRlVGltZSkpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJ0b3RhbCB3cml0ZXM6ICVkXG4iLCBtTnVtV3JpdGVzKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiZGVsYXllZCB3cml0ZXM6ICVkXG4iLCBtTnVtRGVsYXllZFdyaXRlcyk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgImJsb2NrZWQgaW4gd3JpdGU6ICVkXG4iLCBtSW5Xcml0ZSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgInN0YW5kYnk6ICVkXG4iLCBtU3RhbmRieSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHdyaXRlKGZkLCByZXN1bHQuc3RyaW5nKCksIHJlc3VsdC5zaXplKCkpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLy8gVGhyZWFkIHZpcnR1YWxzCi1ib29sIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OnRocmVhZExvb3AoKQotewotICAgIHVuc2lnbmVkIGxvbmcgc2xlZXBUaW1lID0ga0J1ZmZlclJlY292ZXJ5SW5Vc2VjczsKLSAgICBpbnQxNl90KiBjdXJCdWYgPSBtTWl4QnVmZmVyOwotICAgIFZlY3Rvcjwgc3A8VHJhY2s+ID4gdHJhY2tzVG9SZW1vdmU7Ci0gICAgc2l6ZV90IGVuYWJsZWRUcmFja3MgPSAwOwotICAgIG5zZWNzX3Qgc3RhbmRieVRpbWUgPSBzeXN0ZW1UaW1lKCk7ICAgCi0gICAgc2l6ZV90IG1peEJ1ZmZlclNpemUgPSBtRnJhbWVDb3VudCptQ2hhbm5lbENvdW50KnNpemVvZihpbnQxNl90KTsKLSAgICBuc2Vjc190IG1heFBlcmlvZCA9IHNlY29uZHMobUZyYW1lQ291bnQpIC8gbVNhbXBsZVJhdGUgKiAyOwotCi0jaWZkZWYgV0lUSF9BMkRQCi0gICAgYm9vbCBvdXRwdXRUcmFja0FjdGl2ZSA9IGZhbHNlOwotI2VuZGlmCi0KLSAgICBkbyB7Ci0gICAgICAgIGVuYWJsZWRUcmFja3MgPSAwOwotICAgICAgICB7IC8vIHNjb3BlIGZvciB0aGUgbUxvY2sKLSAgICAgICAgCi0gICAgICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotCi0jaWZkZWYgV0lUSF9BMkRQCi0gICAgICAgICAgICBpZiAobU91dHB1dFR5cGUgPT0gQXVkaW9TeXN0ZW06OkFVRElPX09VVFBVVF9BMkRQKSB7Ci0gICAgICAgICAgICAgICAgbUF1ZGlvRmxpbmdlci0+aGFuZGxlT3V0cHV0U3dpdGNoKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAobU91dHB1dFRyYWNrICE9IE5VTEwgJiYgIW1BdWRpb0ZsaW5nZXItPmlzQTJkcEVuYWJsZWQoKSkgewotICAgICAgICAgICAgICAgIGlmIChvdXRwdXRUcmFja0FjdGl2ZSkgewotICAgICAgICAgICAgICAgICAgICBtT3V0cHV0VHJhY2stPnN0b3AoKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0cHV0VHJhY2tBY3RpdmUgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0jZW5kaWYKLQotICAgICAgICAgICAgY29uc3QgU29ydGVkVmVjdG9yPCB3cDxUcmFjaz4gPiYgYWN0aXZlVHJhY2tzID0gbUFjdGl2ZVRyYWNrczsKLQotICAgICAgICAgICAgLy8gcHV0IGF1ZGlvIGhhcmR3YXJlIGludG8gc3RhbmRieSBhZnRlciBzaG9ydCBkZWxheQotICAgICAgICAgICAgaWYgVU5MSUtFTFkoIWFjdGl2ZVRyYWNrcy5zaXplKCkgJiYgc3lzdGVtVGltZSgpID4gc3RhbmRieVRpbWUpIHsKLSAgICAgICAgICAgICAgICAvLyB3YWl0IHVudGlsIHdlIGhhdmUgc29tZXRoaW5nIHRvIGRvLi4uCi0gICAgICAgICAgICAgICAgTE9HVigiQXVkaW8gaGFyZHdhcmUgZW50ZXJpbmcgc3RhbmRieSwgb3V0cHV0ICVkXG4iLCBtT3V0cHV0VHlwZSk7Ci0vLyAgICAgICAgICAgICAgICBtQXVkaW9GbGluZ2VyLT5tSGFyZHdhcmVTdGF0dXMgPSBBVURJT19IV19TVEFOREJZOwotICAgICAgICAgICAgICAgIGlmICghbVN0YW5kYnkpIHsKLSAgICAgICAgICAgICAgICAgICAgbU91dHB1dC0+c3RhbmRieSgpOwotICAgICAgICAgICAgICAgICAgICBtU3RhbmRieSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIAotI2lmZGVmIFdJVEhfQTJEUAotICAgICAgICAgICAgICAgIGlmIChvdXRwdXRUcmFja0FjdGl2ZSkgewotICAgICAgICAgICAgICAgICAgICBtT3V0cHV0VHJhY2stPnN0b3AoKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0cHV0VHJhY2tBY3RpdmUgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICB9Ci0jZW5kaWYKLSAgICAgICAgICAgICAgICBpZiAobU91dHB1dFR5cGUgPT0gQXVkaW9TeXN0ZW06OkFVRElPX09VVFBVVF9IQVJEV0FSRSkgewotICAgICAgICAgICAgICAgICAgICBtQXVkaW9GbGluZ2VyLT5oYW5kbGVGb3JjZWRTcGVha2VyUm91dGUoRk9SQ0VfUk9VVEVfUkVTVE9SRSk7Ci0gICAgICAgICAgICAgICAgfSAgICAgICAgICAgICAgICAKLS8vICAgICAgICAgICAgICAgIG1IYXJkd2FyZVN0YXR1cyA9IEFVRElPX0hXX0lETEU7Ci0gICAgICAgICAgICAgICAgLy8gd2UncmUgYWJvdXQgdG8gd2FpdCwgZmx1c2ggdGhlIGJpbmRlciBjb21tYW5kIGJ1ZmZlcgotICAgICAgICAgICAgICAgIElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPmZsdXNoQ29tbWFuZHMoKTsKLSAgICAgICAgICAgICAgICBtV2FpdFdvcmtDVi53YWl0KG1Mb2NrKTsKLSAgICAgICAgICAgICAgICBMT0dWKCJBdWRpbyBoYXJkd2FyZSBleGl0aW5nIHN0YW5kYnksIG91dHB1dCAlZFxuIiwgbU91dHB1dFR5cGUpOwotICAgICAgICAgICAgICAgIHN0YW5kYnlUaW1lID0gc3lzdGVtVGltZSgpICsga1N0YW5kYnlUaW1lSW5Oc2VjczsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gRm9yY2VkIHJvdXRlIHRvIHNwZWFrZXIgaXMgaGFuZGxlZCBieSBoYXJkd2FyZSBtaXhlciB0aHJlYWQKLSAgICAgICAgICAgIGlmIChtT3V0cHV0VHlwZSA9PSBBdWRpb1N5c3RlbTo6QVVESU9fT1VUUFVUX0hBUkRXQVJFKSB7Ci0gICAgICAgICAgICAgICAgbUF1ZGlvRmxpbmdlci0+aGFuZGxlRm9yY2VkU3BlYWtlclJvdXRlKENIRUNLX1JPVVRFX1JFU1RPUkVfVElNRSk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIGZpbmQgb3V0IHdoaWNoIHRyYWNrcyBuZWVkIHRvIGJlIHByb2Nlc3NlZAotICAgICAgICAgICAgc2l6ZV90IGNvdW50ID0gYWN0aXZlVHJhY2tzLnNpemUoKTsKLSAgICAgICAgICAgIGZvciAoc2l6ZV90IGk9MCA7IGk8Y291bnQgOyBpKyspIHsKLSAgICAgICAgICAgICAgICBzcDxUcmFjaz4gdCA9IGFjdGl2ZVRyYWNrc1tpXS5wcm9tb3RlKCk7Ci0gICAgICAgICAgICAgICAgaWYgKHQgPT0gMCkgY29udGludWU7Ci0KLSAgICAgICAgICAgICAgICBUcmFjayogY29uc3QgdHJhY2sgPSB0LmdldCgpOwotICAgICAgICAgICAgICAgIGF1ZGlvX3RyYWNrX2NibGtfdCogY2JsayA9IHRyYWNrLT5jYmxrKCk7Ci0KLSAgICAgICAgICAgICAgICAvLyBUaGUgZmlyc3QgdGltZSBhIHRyYWNrIGlzIGFkZGVkIHdlIHdhaXQKLSAgICAgICAgICAgICAgICAvLyBmb3IgYWxsIGl0cyBidWZmZXJzIHRvIGJlIGZpbGxlZCBiZWZvcmUgcHJvY2Vzc2luZyBpdAotICAgICAgICAgICAgICAgIG1BdWRpb01peGVyLT5zZXRBY3RpdmVUcmFjayh0cmFjay0+bmFtZSgpKTsKLSAgICAgICAgICAgICAgICBpZiAoY2Jsay0+ZnJhbWVzUmVhZHkoKSAmJiAodHJhY2stPmlzUmVhZHkoKSB8fCB0cmFjay0+aXNTdG9wcGVkKCkpICYmCi0gICAgICAgICAgICAgICAgICAgICAgICAhdHJhY2stPmlzUGF1c2VkKCkpCi0gICAgICAgICAgICAgICAgewotICAgICAgICAgICAgICAgICAgICAvL0xPR1YoInRyYWNrICVkIHU9JTA4eCwgcz0lMDh4IFtPS10iLCB0cmFjay0+bmFtZSgpLCBjYmxrLT51c2VyLCBjYmxrLT5zZXJ2ZXIpOwotCi0gICAgICAgICAgICAgICAgICAgIC8vIGNvbXB1dGUgdm9sdW1lIGZvciB0aGlzIHRyYWNrCi0gICAgICAgICAgICAgICAgICAgIGludDE2X3QgbGVmdCwgcmlnaHQ7Ci0gICAgICAgICAgICAgICAgICAgIGlmICh0cmFjay0+aXNNdXRlZCgpIHx8IG1NYXN0ZXJNdXRlIHx8IHRyYWNrLT5pc1BhdXNpbmcoKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgbGVmdCA9IHJpZ2h0ID0gMDsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmICh0cmFjay0+aXNQYXVzaW5nKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBMT0dWKCJwYXVzZWQoJWQpIiwgdHJhY2stPm5hbWUoKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdHJhY2stPnNldFBhdXNlZCgpOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgZmxvYXQgdHlwZVZvbHVtZSA9IG1TdHJlYW1UeXBlc1t0cmFjay0+dHlwZSgpXS52b2x1bWU7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmbG9hdCB2ID0gbU1hc3RlclZvbHVtZSAqIHR5cGVWb2x1bWU7Ci0gICAgICAgICAgICAgICAgICAgICAgICBmbG9hdCB2X2NsYW1wZWQgPSB2ICogY2Jsay0+dm9sdW1lWzBdOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHZfY2xhbXBlZCA+IE1BWF9HQUlOKSB2X2NsYW1wZWQgPSBNQVhfR0FJTjsKLSAgICAgICAgICAgICAgICAgICAgICAgIGxlZnQgPSBpbnQxNl90KHZfY2xhbXBlZCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB2X2NsYW1wZWQgPSB2ICogY2Jsay0+dm9sdW1lWzFdOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHZfY2xhbXBlZCA+IE1BWF9HQUlOKSB2X2NsYW1wZWQgPSBNQVhfR0FJTjsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJpZ2h0ID0gaW50MTZfdCh2X2NsYW1wZWQpOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgLy8gWFhYOiB0aGVzZSB0aGluZ3MgRE9OJ1QgbmVlZCB0byBiZSBkb25lIGVhY2ggdGltZQotICAgICAgICAgICAgICAgICAgICBtQXVkaW9NaXhlci0+c2V0QnVmZmVyUHJvdmlkZXIodHJhY2spOwotICAgICAgICAgICAgICAgICAgICBtQXVkaW9NaXhlci0+ZW5hYmxlKEF1ZGlvTWl4ZXI6Ok1JWElORyk7Ci0KLSAgICAgICAgICAgICAgICAgICAgaW50IHBhcmFtOwotICAgICAgICAgICAgICAgICAgICBpZiAoIHRyYWNrLT5tRmlsbGluZ1VwU3RhdHVzID09IFRyYWNrOjpGU19GSUxMRUQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIG5vIHJhbXAgZm9yIHRoZSBmaXJzdCB2b2x1bWUgc2V0dGluZwotICAgICAgICAgICAgICAgICAgICAgICAgdHJhY2stPm1GaWxsaW5nVXBTdGF0dXMgPSBUcmFjazo6RlNfQUNUSVZFOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHRyYWNrLT5tU3RhdGUgPT0gVHJhY2tCYXNlOjpSRVNVTUlORykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRyYWNrLT5tU3RhdGUgPSBUcmFja0Jhc2U6OkFDVElWRTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBwYXJhbSA9IEF1ZGlvTWl4ZXI6OlJBTVBfVk9MVU1FOwotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBwYXJhbSA9IEF1ZGlvTWl4ZXI6OlZPTFVNRTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHBhcmFtID0gQXVkaW9NaXhlcjo6UkFNUF9WT0xVTUU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgbUF1ZGlvTWl4ZXItPnNldFBhcmFtZXRlcihwYXJhbSwgQXVkaW9NaXhlcjo6Vk9MVU1FMCwgbGVmdCk7Ci0gICAgICAgICAgICAgICAgICAgIG1BdWRpb01peGVyLT5zZXRQYXJhbWV0ZXIocGFyYW0sIEF1ZGlvTWl4ZXI6OlZPTFVNRTEsIHJpZ2h0KTsKLSAgICAgICAgICAgICAgICAgICAgbUF1ZGlvTWl4ZXItPnNldFBhcmFtZXRlcigKLSAgICAgICAgICAgICAgICAgICAgICAgIEF1ZGlvTWl4ZXI6OlRSQUNLLAotICAgICAgICAgICAgICAgICAgICAgICAgQXVkaW9NaXhlcjo6Rk9STUFULCB0cmFjay0+Zm9ybWF0KCkpOwotICAgICAgICAgICAgICAgICAgICBtQXVkaW9NaXhlci0+c2V0UGFyYW1ldGVyKAotICAgICAgICAgICAgICAgICAgICAgICAgQXVkaW9NaXhlcjo6VFJBQ0ssCi0gICAgICAgICAgICAgICAgICAgICAgICBBdWRpb01peGVyOjpDSEFOTkVMX0NPVU5ULCB0cmFjay0+Y2hhbm5lbENvdW50KCkpOwotICAgICAgICAgICAgICAgICAgICBtQXVkaW9NaXhlci0+c2V0UGFyYW1ldGVyKAotICAgICAgICAgICAgICAgICAgICAgICAgQXVkaW9NaXhlcjo6UkVTQU1QTEUsCi0gICAgICAgICAgICAgICAgICAgICAgICBBdWRpb01peGVyOjpTQU1QTEVfUkFURSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGludChjYmxrLT5zYW1wbGVSYXRlKSk7Ci0KLSAgICAgICAgICAgICAgICAgICAgLy8gcmVzZXQgcmV0cnkgY291bnQKLSAgICAgICAgICAgICAgICAgICAgdHJhY2stPm1SZXRyeUNvdW50ID0ga01heFRyYWNrUmV0cmllczsKLSAgICAgICAgICAgICAgICAgICAgZW5hYmxlZFRyYWNrcysrOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vTE9HVigidHJhY2sgJWQgdT0lMDh4LCBzPSUwOHggW05PVCBSRUFEWV0iLCB0cmFjay0+bmFtZSgpLCBjYmxrLT51c2VyLCBjYmxrLT5zZXJ2ZXIpOwotICAgICAgICAgICAgICAgICAgICBpZiAodHJhY2stPmlzU3RvcHBlZCgpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB0cmFjay0+cmVzZXQoKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpZiAodHJhY2stPmlzVGVybWluYXRlZCgpIHx8IHRyYWNrLT5pc1N0b3BwZWQoKSB8fCB0cmFjay0+aXNQYXVzZWQoKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gV2UgaGF2ZSBjb25zdW1lZCBhbGwgdGhlIGJ1ZmZlcnMgb2YgdGhpcyB0cmFjay4KLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIFJlbW92ZSBpdCBmcm9tIHRoZSBsaXN0IG9mIGFjdGl2ZSB0cmFja3MuCi0gICAgICAgICAgICAgICAgICAgICAgICBMT0dWKCJyZW1vdmUoJWQpIGZyb20gYWN0aXZlIGxpc3QiLCB0cmFjay0+bmFtZSgpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHRyYWNrc1RvUmVtb3ZlLmFkZCh0cmFjayk7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyBObyBidWZmZXJzIGZvciB0aGlzIHRyYWNrLiBHaXZlIGl0IGEgZmV3IGNoYW5jZXMgdG8KLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIGZpbGwgYSBidWZmZXIsIHRoZW4gcmVtb3ZlIGl0IGZyb20gYWN0aXZlIGxpc3QuCi0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoLS0odHJhY2stPm1SZXRyeUNvdW50KSA8PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgTE9HVigiQlVGRkVSIFRJTUVPVVQ6IHJlbW92ZSglZCkgZnJvbSBhY3RpdmUgbGlzdCIsIHRyYWNrLT5uYW1lKCkpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRyYWNrc1RvUmVtb3ZlLmFkZCh0cmFjayk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgLy8gTE9HVigiZGlzYWJsZSglZCkiLCB0cmFjay0+bmFtZSgpKTsKLSAgICAgICAgICAgICAgICAgICAgbUF1ZGlvTWl4ZXItPmRpc2FibGUoQXVkaW9NaXhlcjo6TUlYSU5HKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIHJlbW92ZSBhbGwgdGhlIHRyYWNrcyB0aGF0IG5lZWQgdG8gYmUuLi4KLSAgICAgICAgICAgIGNvdW50ID0gdHJhY2tzVG9SZW1vdmUuc2l6ZSgpOwotICAgICAgICAgICAgaWYgKFVOTElLRUxZKGNvdW50KSkgewotICAgICAgICAgICAgICAgIGZvciAoc2l6ZV90IGk9MCA7IGk8Y291bnQgOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgY29uc3Qgc3A8VHJhY2s+JiB0cmFjayA9IHRyYWNrc1RvUmVtb3ZlW2ldOwotICAgICAgICAgICAgICAgICAgICByZW1vdmVBY3RpdmVUcmFjayh0cmFjayk7Ci0gICAgICAgICAgICAgICAgICAgIGlmICh0cmFjay0+aXNUZXJtaW5hdGVkKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIG1UcmFja3MucmVtb3ZlKHRyYWNrKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGRlbGV0ZVRyYWNrTmFtZSh0cmFjay0+bU5hbWUpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSAgCi0gICAgICAgfQotICAgICAgICAKLSAgICAgICAgaWYgKExJS0VMWShlbmFibGVkVHJhY2tzKSkgewotICAgICAgICAgICAgLy8gbWl4IGJ1ZmZlcnMuLi4KLSAgICAgICAgICAgIG1BdWRpb01peGVyLT5wcm9jZXNzKGN1ckJ1Zik7Ci0KLSNpZmRlZiBXSVRIX0EyRFAKLSAgICAgICAgICAgIGlmIChtT3V0cHV0VHJhY2sgIT0gTlVMTCAmJiBtQXVkaW9GbGluZ2VyLT5pc0EyZHBFbmFibGVkKCkpIHsKLSAgICAgICAgICAgICAgICBpZiAoIW91dHB1dFRyYWNrQWN0aXZlKSB7Ci0gICAgICAgICAgICAgICAgICAgIExPR1YoInN0YXJ0aW5nIG91dHB1dCB0cmFjayBpbiBtaXhlciBmb3Igb3V0cHV0ICVkIiwgbU91dHB1dFR5cGUpOwotICAgICAgICAgICAgICAgICAgICBtT3V0cHV0VHJhY2stPnN0YXJ0KCk7Ci0gICAgICAgICAgICAgICAgICAgIG91dHB1dFRyYWNrQWN0aXZlID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbU91dHB1dFRyYWNrLT53cml0ZShjdXJCdWYsIG1GcmFtZUNvdW50KTsKLSAgICAgICAgICAgIH0KLSNlbmRpZgotCi0gICAgICAgICAgICAvLyBvdXRwdXQgYXVkaW8gdG8gaGFyZHdhcmUKLSAgICAgICAgICAgIG1MYXN0V3JpdGVUaW1lID0gc3lzdGVtVGltZSgpOwotICAgICAgICAgICAgbUluV3JpdGUgPSB0cnVlOwotICAgICAgICAgICAgbU91dHB1dC0+d3JpdGUoY3VyQnVmLCBtaXhCdWZmZXJTaXplKTsKLSAgICAgICAgICAgIG1OdW1Xcml0ZXMrKzsKLSAgICAgICAgICAgIG1JbldyaXRlID0gZmFsc2U7Ci0gICAgICAgICAgICBtU3RhbmRieSA9IGZhbHNlOwotICAgICAgICAgICAgbnNlY3NfdCB0ZW1wID0gc3lzdGVtVGltZSgpOwotICAgICAgICAgICAgc3RhbmRieVRpbWUgPSB0ZW1wICsga1N0YW5kYnlUaW1lSW5Oc2VjczsKLSAgICAgICAgICAgIG5zZWNzX3QgZGVsdGEgPSB0ZW1wIC0gbUxhc3RXcml0ZVRpbWU7Ci0gICAgICAgICAgICBpZiAoZGVsdGEgPiBtYXhQZXJpb2QpIHsKLSAgICAgICAgICAgICAgICBMT0dXKCJ3cml0ZSBibG9ja2VkIGZvciAlbGx1IG1zZWNzIiwgbnMybXMoZGVsdGEpKTsKLSAgICAgICAgICAgICAgICBtTnVtRGVsYXllZFdyaXRlcysrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgc2xlZXBUaW1lID0ga0J1ZmZlclJlY292ZXJ5SW5Vc2VjczsKLSAgICAgICAgfSBlbHNlIHsgICAgICAgICAKLSNpZmRlZiBXSVRIX0EyRFAKLSAgICAgICAgICAgIGlmIChtT3V0cHV0VHJhY2sgIT0gTlVMTCAmJiBtQXVkaW9GbGluZ2VyLT5pc0EyZHBFbmFibGVkKCkpIHsKLSAgICAgICAgICAgICAgICBpZiAob3V0cHV0VHJhY2tBY3RpdmUpIHsKLSAgICAgICAgICAgICAgICAgICAgbU91dHB1dFRyYWNrLT53cml0ZShjdXJCdWYsIDApOwotICAgICAgICAgICAgICAgICAgICBpZiAobU91dHB1dFRyYWNrLT5idWZmZXJRdWV1ZUVtcHR5KCkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIG1PdXRwdXRUcmFjay0+c3RvcCgpOwotICAgICAgICAgICAgICAgICAgICAgICAgb3V0cHV0VHJhY2tBY3RpdmUgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHN0YW5kYnlUaW1lID0gc3lzdGVtVGltZSgpICsga1N0YW5kYnlUaW1lSW5Oc2VjczsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSNlbmRpZgotICAgICAgICAgICAgLy8gVGhlcmUgd2FzIG5vdGhpbmcgdG8gbWl4IHRoaXMgcm91bmQsIHdoaWNoIG1lYW5zIGFsbAotICAgICAgICAgICAgLy8gYWN0aXZlIHRyYWNrcyB3ZXJlIGxhdGUuIFNsZWVwIGEgbGl0dGxlIGJpdCB0byBnaXZlCi0gICAgICAgICAgICAvLyB0aGVtIGFub3RoZXIgY2hhbmNlLiBJZiB3ZSdyZSB0b28gbGF0ZSwgdGhlIGF1ZGlvCi0gICAgICAgICAgICAvLyBoYXJkd2FyZSB3aWxsIHplcm8tZmlsbCBmb3IgdXMuCi0gICAgICAgICAgICAvL0xPR1YoIm5vIGJ1ZmZlcnMgLSB1c2xlZXAoJWx1KSIsIHNsZWVwVGltZSk7Ci0gICAgICAgICAgICB1c2xlZXAoc2xlZXBUaW1lKTsKLSAgICAgICAgICAgIGlmIChzbGVlcFRpbWUgPCBrTWF4QnVmZmVyUmVjb3ZlcnlJblVzZWNzKSB7Ci0gICAgICAgICAgICAgICAgc2xlZXBUaW1lICs9IGtCdWZmZXJSZWNvdmVyeUluVXNlY3M7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBmaW5hbGx5IGxldCBnbyBvZiBhbGwgb3VyIHRyYWNrcywgd2l0aG91dCB0aGUgbG9jayBoZWxkCi0gICAgICAgIC8vIHNpbmNlIHdlIGNhbid0IGd1YXJhbnRlZSB0aGUgZGVzdHJ1Y3RvcnMgd29uJ3QgYWNxdWlyZSB0aGF0Ci0gICAgICAgIC8vIHNhbWUgbG9jay4KLSAgICAgICAgdHJhY2tzVG9SZW1vdmUuY2xlYXIoKTsKLSAgICB9IHdoaWxlICh0cnVlKTsKLQotICAgIHJldHVybiBmYWxzZTsKLX0KLQotc3RhdHVzX3QgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6cmVhZHlUb1J1bigpCi17Ci0gICAgaWYgKG1TYW1wbGVSYXRlID09IDApIHsKLSAgICAgICAgTE9HRSgiTm8gd29ya2luZyBhdWRpbyBkcml2ZXIgZm91bmQuIik7Ci0gICAgICAgIHJldHVybiBOT19JTklUOwotICAgIH0KLSAgICBMT0dJKCJBdWRpb0ZsaW5nZXIncyB0aHJlYWQgcmVhZHkgdG8gcnVuIGZvciBvdXRwdXQgJWQiLCBtT3V0cHV0VHlwZSk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6Om9uRmlyc3RSZWYoKQotewotICAgIGNvbnN0IHNpemVfdCBTSVpFID0gMjU2OwotICAgIGNoYXIgYnVmZmVyW1NJWkVdOwotCi0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiTWl4ZXIgVGhyZWFkIGZvciBvdXRwdXQgJWQiLCBtT3V0cHV0VHlwZSk7Ci0KLSAgICBydW4oYnVmZmVyLCBBTkRST0lEX1BSSU9SSVRZX1VSR0VOVF9BVURJTyk7Ci19Ci0KLQotc3A8QXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6VHJhY2s+ICBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpjcmVhdGVUcmFjaygKLSAgICAgICAgY29uc3Qgc3A8QXVkaW9GbGluZ2VyOjpDbGllbnQ+JiBjbGllbnQsCi0gICAgICAgIGludCBzdHJlYW1UeXBlLAotICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlLAotICAgICAgICBpbnQgZm9ybWF0LAotICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICBpbnQgZnJhbWVDb3VudCwKLSAgICAgICAgY29uc3Qgc3A8SU1lbW9yeT4mIHNoYXJlZEJ1ZmZlciwKLSAgICAgICAgc3RhdHVzX3QgKnN0YXR1cykKLXsKLSAgICBzcDxUcmFjaz4gdHJhY2s7Ci0gICAgc3RhdHVzX3QgbFN0YXR1czsKLSAgICAKLSAgICAvLyBSZXNhbXBsZXIgaW1wbGVtZW50YXRpb24gbGltaXRzIGlucHV0IHNhbXBsaW5nIHJhdGUgdG8gMiB4IG91dHB1dCBzYW1wbGluZyByYXRlLgotICAgIGlmIChzYW1wbGVSYXRlID4gTUFYX1NBTVBMRV9SQVRFIHx8IHNhbXBsZVJhdGUgPiBtU2FtcGxlUmF0ZSoyKSB7Ci0gICAgICAgIExPR0UoIlNhbXBsZSByYXRlIG91dCBvZiByYW5nZTogJWQgbVNhbXBsZVJhdGUgJWQiLCBzYW1wbGVSYXRlLCBtU2FtcGxlUmF0ZSk7Ci0gICAgICAgIGxTdGF0dXMgPSBCQURfVkFMVUU7Ci0gICAgICAgIGdvdG8gRXhpdDsKLSAgICB9Ci0KLSAgICB7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0KLSAgICAgICAgaWYgKG1TYW1wbGVSYXRlID09IDApIHsKLSAgICAgICAgICAgIExPR0UoIkF1ZGlvIGRyaXZlciBub3QgaW5pdGlhbGl6ZWQuIik7Ci0gICAgICAgICAgICBsU3RhdHVzID0gTk9fSU5JVDsKLSAgICAgICAgICAgIGdvdG8gRXhpdDsKLSAgICAgICAgfQotCi0gICAgICAgIHRyYWNrID0gbmV3IFRyYWNrKHRoaXMsIGNsaWVudCwgc3RyZWFtVHlwZSwgc2FtcGxlUmF0ZSwgZm9ybWF0LAotICAgICAgICAgICAgICAgIGNoYW5uZWxDb3VudCwgZnJhbWVDb3VudCwgc2hhcmVkQnVmZmVyKTsKLSAgICAgICAgaWYgKHRyYWNrLT5nZXRDYmxrKCkgPT0gTlVMTCkgewotICAgICAgICAgICAgdHJhY2suY2xlYXIoKTsKLSAgICAgICAgICAgIGxTdGF0dXMgPSBOT19NRU1PUlk7Ci0gICAgICAgICAgICBnb3RvIEV4aXQ7Ci0gICAgICAgIH0KLSAgICAgICAgbVRyYWNrcy5hZGQodHJhY2spOwotICAgICAgICBsU3RhdHVzID0gTk9fRVJST1I7Ci0gICAgfQotCi1FeGl0OgotICAgIGlmKHN0YXR1cykgewotICAgICAgICAqc3RhdHVzID0gbFN0YXR1czsKLSAgICB9Ci0gICAgcmV0dXJuIHRyYWNrOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OmdldFRyYWNrcygKLSAgICAgICAgU29ydGVkVmVjdG9yIDwgc3A8VHJhY2s+ID4mIHRyYWNrcywKLSAgICAgICAgU29ydGVkVmVjdG9yIDwgd3A8VHJhY2s+ID4mIGFjdGl2ZVRyYWNrcykKLXsKLSAgICBzaXplX3Qgc2l6ZSA9IG1UcmFja3Muc2l6ZSgpOwotICAgIExPR1YgKCJNaXhlclRocmVhZDo6Z2V0VHJhY2tzKCkgZm9yIG91dHB1dCAlZCwgbVRyYWNrcy5zaXplICVkLCBtQWN0aXZlVHJhY2tzLnNpemUgJWQiLCBtT3V0cHV0VHlwZSwgIG1UcmFja3Muc2l6ZSgpLCBtQWN0aXZlVHJhY2tzLnNpemUoKSk7Ci0gICAgZm9yIChzaXplX3QgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKLSAgICAgICAgc3A8VHJhY2s+IHQgPSBtVHJhY2tzW2ldOwotICAgICAgICBpZiAoQXVkaW9TeXN0ZW06OnJvdXRlZFRvQTJkcE91dHB1dCh0LT5tU3RyZWFtVHlwZSkpIHsKLSAgICAgICAgICAgIHRyYWNrcy5hZGQodCk7Ci0gICAgICAgICAgICBpbnQgaiA9IG1BY3RpdmVUcmFja3MuaW5kZXhPZih0KTsKLSAgICAgICAgICAgIGlmIChqID49IDApIHsKLSAgICAgICAgICAgICAgICB0ID0gbUFjdGl2ZVRyYWNrc1tqXS5wcm9tb3RlKCk7Ci0gICAgICAgICAgICAgICAgaWYgKHQgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICBhY3RpdmVUcmFja3MuYWRkKHQpOyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIH0gICAgICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBzaXplID0gYWN0aXZlVHJhY2tzLnNpemUoKTsKLSAgICBmb3IgKHNpemVfdCBpID0gMDsgaSA8IHNpemU7IGkrKykgewotICAgICAgICByZW1vdmVBY3RpdmVUcmFjayhhY3RpdmVUcmFja3NbaV0pOwotICAgIH0KLSAgICAKLSAgICBzaXplID0gdHJhY2tzLnNpemUoKTsKLSAgICBmb3IgKHNpemVfdCBpID0gMDsgaSA8IHNpemU7IGkrKykgewotICAgICAgICBzcDxUcmFjaz4gdCA9IHRyYWNrc1tpXTsKLSAgICAgICAgbVRyYWNrcy5yZW1vdmUodCk7Ci0gICAgICAgIGRlbGV0ZVRyYWNrTmFtZSh0LT5uYW1lKCkpOwotICAgIH0KLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpwdXRUcmFja3MoCi0gICAgICAgIFNvcnRlZFZlY3RvciA8IHNwPFRyYWNrPiA+JiB0cmFja3MsCi0gICAgICAgIFNvcnRlZFZlY3RvciA8IHdwPFRyYWNrPiA+JiBhY3RpdmVUcmFja3MpCi17Ci0KLSAgICBMT0dWICgiTWl4ZXJUaHJlYWQ6OnB1dFRyYWNrcygpIGZvciBvdXRwdXQgJWQsIHRyYWNrcy5zaXplICVkLCBhY3RpdmVUcmFja3Muc2l6ZSAlZCIsIG1PdXRwdXRUeXBlLCAgdHJhY2tzLnNpemUoKSwgYWN0aXZlVHJhY2tzLnNpemUoKSk7Ci0KLSAgICBzaXplX3Qgc2l6ZSA9IHRyYWNrcy5zaXplKCk7Ci0gICAgZm9yIChzaXplX3QgaSA9IDA7IGkgPCBzaXplIDsgaSsrKSB7Ci0gICAgICAgIHNwPFRyYWNrPiB0ID0gdHJhY2tzW2ldOwotICAgICAgICBpbnQgbmFtZSA9IGdldFRyYWNrTmFtZSgpOwotCi0gICAgICAgIGlmIChuYW1lIDwgMCkgcmV0dXJuOwotICAgICAgICAKLSAgICAgICAgdC0+bU5hbWUgPSBuYW1lOwotICAgICAgICB0LT5tTWl4ZXJUaHJlYWQgPSB0aGlzOwotICAgICAgICBtVHJhY2tzLmFkZCh0KTsKLQotICAgICAgICBpbnQgaiA9IGFjdGl2ZVRyYWNrcy5pbmRleE9mKHQpOwotICAgICAgICBpZiAoaiA+PSAwKSB7Ci0gICAgICAgICAgICBhZGRBY3RpdmVUcmFjayh0KTsKLSAgICAgICAgfSAgICAgICAgICAgIAotICAgIH0KLX0KLQotdWludDMyX3QgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6c2FtcGxlUmF0ZSgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1TYW1wbGVSYXRlOwotfQotCi1pbnQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6Y2hhbm5lbENvdW50KCkgY29uc3QKLXsKLSAgICByZXR1cm4gbUNoYW5uZWxDb3VudDsKLX0KLQotaW50IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OmZvcm1hdCgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1Gb3JtYXQ7Ci19Ci0KLXNpemVfdCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpmcmFtZUNvdW50KCkgY29uc3QKLXsKLSAgICByZXR1cm4gbUZyYW1lQ291bnQ7Ci19Ci0KLXVpbnQzMl90IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OmxhdGVuY3koKSBjb25zdAotewotICAgIGlmIChtT3V0cHV0KSB7Ci0gICAgICAgIHJldHVybiBtT3V0cHV0LT5sYXRlbmN5KCk7Ci0gICAgfQotICAgIGVsc2UgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci19Ci0KLXN0YXR1c190IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OnNldE1hc3RlclZvbHVtZShmbG9hdCB2YWx1ZSkKLXsKLSAgICBtTWFzdGVyVm9sdW1lID0gdmFsdWU7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpzZXRNYXN0ZXJNdXRlKGJvb2wgbXV0ZWQpCi17Ci0gICAgbU1hc3Rlck11dGUgPSBtdXRlZDsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLWZsb2F0IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6Om1hc3RlclZvbHVtZSgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1NYXN0ZXJWb2x1bWU7Ci19Ci0KLWJvb2wgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6bWFzdGVyTXV0ZSgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1NYXN0ZXJNdXRlOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpzZXRTdHJlYW1Wb2x1bWUoaW50IHN0cmVhbSwgZmxvYXQgdmFsdWUpCi17Ci0gICAgbVN0cmVhbVR5cGVzW3N0cmVhbV0udm9sdW1lID0gdmFsdWU7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpzZXRTdHJlYW1NdXRlKGludCBzdHJlYW0sIGJvb2wgbXV0ZWQpCi17Ci0gICAgbVN0cmVhbVR5cGVzW3N0cmVhbV0ubXV0ZSA9IG11dGVkOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotZmxvYXQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6c3RyZWFtVm9sdW1lKGludCBzdHJlYW0pIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1TdHJlYW1UeXBlc1tzdHJlYW1dLnZvbHVtZTsKLX0KLQotYm9vbCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpzdHJlYW1NdXRlKGludCBzdHJlYW0pIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1TdHJlYW1UeXBlc1tzdHJlYW1dLm11dGU7Ci19Ci0KLWJvb2wgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6aXNNdXNpY0FjdGl2ZSgpIGNvbnN0Ci17Ci0gICAgc2l6ZV90IGNvdW50ID0gbUFjdGl2ZVRyYWNrcy5zaXplKCk7Ci0gICAgZm9yIChzaXplX3QgaSA9IDAgOyBpIDwgY291bnQgOyArK2kpIHsKLSAgICAgICAgc3A8VHJhY2s+IHQgPSBtQWN0aXZlVHJhY2tzW2ldLnByb21vdGUoKTsKLSAgICAgICAgaWYgKHQgPT0gMCkgY29udGludWU7Ci0gICAgICAgIFRyYWNrKiBjb25zdCB0cmFjayA9IHQuZ2V0KCk7Ci0gICAgICAgIGlmICh0LT5tU3RyZWFtVHlwZSA9PSBBdWRpb1N5c3RlbTo6TVVTSUMpCi0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0gICAgcmV0dXJuIGZhbHNlOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjphZGRUcmFjayhjb25zdCBzcDxUcmFjaz4mIHRyYWNrKQotewotICAgIHN0YXR1c190IHN0YXR1cyA9IEFMUkVBRFlfRVhJU1RTOwotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0KLSAgICAvLyBoZXJlIHRoZSB0cmFjayBjb3VsZCBiZSBlaXRoZXIgbmV3LCBvciByZXN0YXJ0ZWQKLSAgICAvLyBpbiBib3RoIGNhc2VzICJ1bnN0b3AiIHRoZSB0cmFjawotICAgIGlmICh0cmFjay0+aXNQYXVzZWQoKSkgewotICAgICAgICB0cmFjay0+bVN0YXRlID0gVHJhY2tCYXNlOjpSRVNVTUlORzsKLSAgICAgICAgTE9HVigiUEFVU0VEID0+IFJFU1VNSU5HICglZCkiLCB0cmFjay0+bmFtZSgpKTsKLSAgICB9IGVsc2UgewotICAgICAgICB0cmFjay0+bVN0YXRlID0gVHJhY2tCYXNlOjpBQ1RJVkU7Ci0gICAgICAgIExPR1YoIj8gPT4gQUNUSVZFICglZCkiLCB0cmFjay0+bmFtZSgpKTsKLSAgICB9Ci0gICAgLy8gc2V0IHJldHJ5IGNvdW50IGZvciBidWZmZXIgZmlsbAotICAgIHRyYWNrLT5tUmV0cnlDb3VudCA9IGtNYXhUcmFja1N0YXJ0dXBSZXRyaWVzOwotICAgIGlmIChtQWN0aXZlVHJhY2tzLmluZGV4T2YodHJhY2spIDwgMCkgewotICAgICAgICAvLyB0aGUgdHJhY2sgaXMgbmV3bHkgYWRkZWQsIG1ha2Ugc3VyZSBpdCBmaWxscyB1cCBhbGwgaXRzCi0gICAgICAgIC8vIGJ1ZmZlcnMgYmVmb3JlIHBsYXlpbmcuIFRoaXMgaXMgdG8gZW5zdXJlIHRoZSBjbGllbnQgd2lsbAotICAgICAgICAvLyBlZmZlY3RpdmVseSBnZXQgdGhlIGxhdGVuY3kgaXQgcmVxdWVzdGVkLgotICAgICAgICB0cmFjay0+bUZpbGxpbmdVcFN0YXR1cyA9IFRyYWNrOjpGU19GSUxMSU5HOwotICAgICAgICB0cmFjay0+bVJlc2V0RG9uZSA9IGZhbHNlOwotICAgICAgICBhZGRBY3RpdmVUcmFjayh0cmFjayk7Ci0gICAgICAgIHN0YXR1cyA9IE5PX0VSUk9SOwotICAgIH0KLSAgICAKLSAgICBMT0dWKCJtV2FpdFdvcmtDVi5icm9hZGNhc3QiKTsKLSAgICBtV2FpdFdvcmtDVi5icm9hZGNhc3QoKTsKLQotICAgIHJldHVybiBzdGF0dXM7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6cmVtb3ZlVHJhY2sod3A8VHJhY2s+IHRyYWNrLCBpbnQgbmFtZSkKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIHNwPFRyYWNrPiB0ID0gdHJhY2sucHJvbW90ZSgpOwotICAgIGlmICh0IT1OVUxMICYmICh0LT5tU3RhdGUgPD0gVHJhY2tCYXNlOjpTVE9QUEVEKSkgewotICAgICAgICByZW1vdmVfdHJhY2tfbCh0cmFjaywgbmFtZSk7Ci0gICAgfQotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OnJlbW92ZV90cmFja19sKHdwPFRyYWNrPiB0cmFjaywgaW50IG5hbWUpCi17Ci0gICAgc3A8VHJhY2s+IHQgPSB0cmFjay5wcm9tb3RlKCk7Ci0gICAgaWYgKHQhPU5VTEwpIHsKLSAgICAgICAgdC0+cmVzZXQoKTsKLSAgICB9Ci0gICAgZGVsZXRlVHJhY2tOYW1lKG5hbWUpOwotICAgIHJlbW92ZUFjdGl2ZVRyYWNrKHRyYWNrKTsKLSAgICBtV2FpdFdvcmtDVi5icm9hZGNhc3QoKTsKLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpkZXN0cm95VHJhY2soY29uc3Qgc3A8VHJhY2s+JiB0cmFjaykKLXsKLSAgICAvLyBOT1RFOiBXZSdyZSBhY3F1aXJpbmcgYSBzdHJvbmcgcmVmZXJlbmNlIG9uIHRoZSB0cmFjayBiZWZvcmUKLSAgICAvLyBhY3F1aXJpbmcgdGhlIGxvY2ssIHRoaXMgaXMgdG8gbWFrZSBzdXJlIHJlbW92aW5nIGl0IGZyb20KLSAgICAvLyBtVHJhY2tzIHdvbid0IGNhdXNlIHRoZSBkZXN0cnVjdG9yIHRvIGJlIGNhbGxlZCB3aGlsZSB0aGUgbG9jayBpcwotICAgIC8vIGhlbGQgKG5vdGUgdGhhdCB0ZWNobmljYWxseSwgJ3RyYWNrJyBjb3VsZCBiZSBhIHJlZmVyZW5jZSB0byBhbiBpdGVtCi0gICAgLy8gaW4gbVRyYWNrcywgd2hpY2ggaXMgd2h5IHdlIG5lZWQgdG8gZG8gdGhpcykuCi0gICAgc3A8VHJhY2s+IGtlZXAodHJhY2spOwotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgdHJhY2stPm1TdGF0ZSA9IFRyYWNrQmFzZTo6VEVSTUlOQVRFRDsKLSAgICBpZiAobUFjdGl2ZVRyYWNrcy5pbmRleE9mKHRyYWNrKSA8IDApIHsKLSAgICAgICAgTE9HVigicmVtb3ZlIHRyYWNrICglZCkgYW5kIGRlbGV0ZSBmcm9tIG1peGVyIiwgdHJhY2stPm5hbWUoKSk7Ci0gICAgICAgIG1UcmFja3MucmVtb3ZlKHRyYWNrKTsKLSAgICAgICAgZGVsZXRlVHJhY2tOYW1lKGtlZXAtPm5hbWUoKSk7Ci0gICAgfQotfQotCi0KLXZvaWQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6YWRkQWN0aXZlVHJhY2soY29uc3Qgd3A8VHJhY2s+JiB0KQotewotICAgIG1BY3RpdmVUcmFja3MuYWRkKHQpOwotCi0gICAgLy8gRm9yY2Ugcm91dGluZyB0byBzcGVha2VyIGZvciBjZXJ0YWluIHN0cmVhbSB0eXBlcwotICAgIC8vIFRoZSBmb3JjZWQgcm91dGluZyB0byBzcGVha2VyIGlzIG1hbmFnZWQgYnkgaGFyZHdhcmUgbWl4ZXIKLSAgICBpZiAobU91dHB1dFR5cGUgPT0gQXVkaW9TeXN0ZW06OkFVRElPX09VVFBVVF9IQVJEV0FSRSkgewotICAgICAgICBzcDxUcmFjaz4gdHJhY2sgPSB0LnByb21vdGUoKTsKLSAgICAgICAgaWYgKHRyYWNrID09IE5VTEwpIHJldHVybjsKLSAgIAotICAgICAgICBpZiAoc3RyZWFtRm9yY2VkVG9TcGVha2VyKHRyYWNrLT50eXBlKCkpKSB7Ci0gICAgICAgICAgICBtQXVkaW9GbGluZ2VyLT5oYW5kbGVGb3JjZWRTcGVha2VyUm91dGUoQUNUSVZFX1RSQUNLX0FEREVEKTsKLSAgICAgICAgfSAgICAgICAgCi0gICAgfQotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OnJlbW92ZUFjdGl2ZVRyYWNrKGNvbnN0IHdwPFRyYWNrPiYgdCkKLXsKLSAgICBtQWN0aXZlVHJhY2tzLnJlbW92ZSh0KTsKLQotICAgIC8vIEZvcmNlIHJvdXRpbmcgdG8gc3BlYWtlciBmb3IgY2VydGFpbiBzdHJlYW0gdHlwZXMKLSAgICAvLyBUaGUgZm9yY2VkIHJvdXRpbmcgdG8gc3BlYWtlciBpcyBtYW5hZ2VkIGJ5IGhhcmR3YXJlIG1peGVyCi0gICAgaWYgKG1PdXRwdXRUeXBlID09IEF1ZGlvU3lzdGVtOjpBVURJT19PVVRQVVRfSEFSRFdBUkUpIHsKLSAgICAgICAgc3A8VHJhY2s+IHRyYWNrID0gdC5wcm9tb3RlKCk7Ci0gICAgICAgIGlmICh0cmFjayA9PSBOVUxMKSByZXR1cm47Ci0KLSAgICAgICAgaWYgKHN0cmVhbUZvcmNlZFRvU3BlYWtlcih0cmFjay0+dHlwZSgpKSkgewotICAgICAgICAgICAgbUF1ZGlvRmxpbmdlci0+aGFuZGxlRm9yY2VkU3BlYWtlclJvdXRlKEFDVElWRV9UUkFDS19SRU1PVkVEKTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotaW50IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OmdldFRyYWNrTmFtZSgpCi17Ci0gICAgcmV0dXJuIG1BdWRpb01peGVyLT5nZXRUcmFja05hbWUoKTsKLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpkZWxldGVUcmFja05hbWUoaW50IG5hbWUpCi17Ci0gICAgbUF1ZGlvTWl4ZXItPmRlbGV0ZVRyYWNrTmFtZShuYW1lKTsKLX0KLQotc2l6ZV90IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OmdldE91dHB1dEZyYW1lQ291bnQoKSAKLXsKLSAgICByZXR1cm4gbU91dHB1dC0+YnVmZmVyU2l6ZSgpIC8gbU91dHB1dC0+Y2hhbm5lbENvdW50KCkgLyBzaXplb2YoaW50MTZfdCk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6VHJhY2tCYXNlOjpUcmFja0Jhc2UoCi0gICAgICAgICAgICBjb25zdCBzcDxNaXhlclRocmVhZD4mIG1peGVyVGhyZWFkLAotICAgICAgICAgICAgY29uc3Qgc3A8Q2xpZW50PiYgY2xpZW50LAotICAgICAgICAgICAgaW50IHN0cmVhbVR5cGUsCi0gICAgICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlLAotICAgICAgICAgICAgaW50IGZvcm1hdCwKLSAgICAgICAgICAgIGludCBjaGFubmVsQ291bnQsCi0gICAgICAgICAgICBpbnQgZnJhbWVDb3VudCwKLSAgICAgICAgICAgIHVpbnQzMl90IGZsYWdzLAotICAgICAgICAgICAgY29uc3Qgc3A8SU1lbW9yeT4mIHNoYXJlZEJ1ZmZlcikKLSAgICA6ICAgUmVmQmFzZSgpLAotICAgICAgICBtTWl4ZXJUaHJlYWQobWl4ZXJUaHJlYWQpLAotICAgICAgICBtQ2xpZW50KGNsaWVudCksCi0gICAgICAgIG1TdHJlYW1UeXBlKHN0cmVhbVR5cGUpLAotICAgICAgICBtRnJhbWVDb3VudCgwKSwKLSAgICAgICAgbVN0YXRlKElETEUpLAotICAgICAgICBtQ2xpZW50VGlkKC0xKSwKLSAgICAgICAgbUZvcm1hdChmb3JtYXQpLAotICAgICAgICBtRmxhZ3MoZmxhZ3MgJiB+U1lTVEVNX0ZMQUdTX01BU0spCi17Ci0gICAgbU5hbWUgPSBtaXhlclRocmVhZC0+Z2V0VHJhY2tOYW1lKCk7Ci0gICAgTE9HVigiVHJhY2tCYXNlIGNvbnRydWN0b3IgbmFtZSAlZCwgY2FsbGluZyB0aHJlYWQgJWQiLCBtTmFtZSwgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpKTsKLSAgICBpZiAobU5hbWUgPCAwKSB7Ci0gICAgICAgIExPR0UoIm5vIG1vcmUgdHJhY2sgbmFtZXMgYXZhaWxsYWJsZSIpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgTE9HVl9JRihzaGFyZWRCdWZmZXIgIT0gMCwgInNoYXJlZEJ1ZmZlcjogJXAsIHNpemU6ICVkIiwgc2hhcmVkQnVmZmVyLT5wb2ludGVyKCksIHNoYXJlZEJ1ZmZlci0+c2l6ZSgpKTsKLQotICAgIC8vIExPR0QoIkNyZWF0aW5nIHRyYWNrIHdpdGggJWQgYnVmZmVycyBAICVkIGJ5dGVzIiwgYnVmZmVyQ291bnQsIGJ1ZmZlclNpemUpOwotICAgc2l6ZV90IHNpemUgPSBzaXplb2YoYXVkaW9fdHJhY2tfY2Jsa190KTsKLSAgIHNpemVfdCBidWZmZXJTaXplID0gZnJhbWVDb3VudCpjaGFubmVsQ291bnQqc2l6ZW9mKGludDE2X3QpOwotICAgaWYgKHNoYXJlZEJ1ZmZlciA9PSAwKSB7Ci0gICAgICAgc2l6ZSArPSBidWZmZXJTaXplOwotICAgfQotCi0gICBpZiAoY2xpZW50ICE9IE5VTEwpIHsKLSAgICAgICAgbUNibGtNZW1vcnkgPSBjbGllbnQtPmhlYXAoKS0+YWxsb2NhdGUoc2l6ZSk7Ci0gICAgICAgIGlmIChtQ2Jsa01lbW9yeSAhPSAwKSB7Ci0gICAgICAgICAgICBtQ2JsayA9IHN0YXRpY19jYXN0PGF1ZGlvX3RyYWNrX2NibGtfdCAqPihtQ2Jsa01lbW9yeS0+cG9pbnRlcigpKTsKLSAgICAgICAgICAgIGlmIChtQ2JsaykgeyAvLyBjb25zdHJ1Y3QgdGhlIHNoYXJlZCBzdHJ1Y3R1cmUgaW4tcGxhY2UuCi0gICAgICAgICAgICAgICAgbmV3KG1DYmxrKSBhdWRpb190cmFja19jYmxrX3QoKTsKLSAgICAgICAgICAgICAgICAvLyBjbGVhciBhbGwgYnVmZmVycwotICAgICAgICAgICAgICAgIG1DYmxrLT5mcmFtZUNvdW50ID0gZnJhbWVDb3VudDsKLSAgICAgICAgICAgICAgICBtQ2Jsay0+c2FtcGxlUmF0ZSA9IHNhbXBsZVJhdGU7Ci0gICAgICAgICAgICAgICAgbUNibGstPmNoYW5uZWxzID0gY2hhbm5lbENvdW50OwotICAgICAgICAgICAgICAgIGlmIChzaGFyZWRCdWZmZXIgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICBtQnVmZmVyID0gKGNoYXIqKW1DYmxrICsgc2l6ZW9mKGF1ZGlvX3RyYWNrX2NibGtfdCk7Ci0gICAgICAgICAgICAgICAgICAgIG1lbXNldChtQnVmZmVyLCAwLCBmcmFtZUNvdW50KmNoYW5uZWxDb3VudCpzaXplb2YoaW50MTZfdCkpOwotICAgICAgICAgICAgICAgICAgICAvLyBGb3JjZSB1bmRlcnJ1biBjb25kaXRpb24gdG8gYXZvaWQgZmFsc2UgdW5kZXJydW4gY2FsbGJhY2sgdW50aWwgZmlyc3QgZGF0YSBpcwotICAgICAgICAgICAgICAgICAgICAvLyB3cml0dGVuIHRvIGJ1ZmZlcgotICAgICAgICAgICAgICAgICAgICBtQ2Jsay0+Zmxvd0NvbnRyb2xGbGFnID0gMTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBtQnVmZmVyID0gc2hhcmVkQnVmZmVyLT5wb2ludGVyKCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIG1CdWZmZXJFbmQgPSAodWludDhfdCAqKW1CdWZmZXIgKyBidWZmZXJTaXplOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgTE9HRSgibm90IGVub3VnaCBtZW1vcnkgZm9yIEF1ZGlvVHJhY2sgc2l6ZT0ldSIsIHNpemUpOwotICAgICAgICAgICAgY2xpZW50LT5oZWFwKCktPmR1bXAoIkF1ZGlvVHJhY2siKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgfSBlbHNlIHsKLSAgICAgICBtQ2JsayA9IChhdWRpb190cmFja19jYmxrX3QgKikobmV3IHVpbnQ4X3Rbc2l6ZV0pOwotICAgICAgIGlmIChtQ2JsaykgeyAvLyBjb25zdHJ1Y3QgdGhlIHNoYXJlZCBzdHJ1Y3R1cmUgaW4tcGxhY2UuCi0gICAgICAgICAgIG5ldyhtQ2JsaykgYXVkaW9fdHJhY2tfY2Jsa190KCk7Ci0gICAgICAgICAgIC8vIGNsZWFyIGFsbCBidWZmZXJzCi0gICAgICAgICAgIG1DYmxrLT5mcmFtZUNvdW50ID0gZnJhbWVDb3VudDsKLSAgICAgICAgICAgbUNibGstPnNhbXBsZVJhdGUgPSBzYW1wbGVSYXRlOwotICAgICAgICAgICBtQ2Jsay0+Y2hhbm5lbHMgPSBjaGFubmVsQ291bnQ7Ci0gICAgICAgICAgIG1CdWZmZXIgPSAoY2hhciopbUNibGsgKyBzaXplb2YoYXVkaW9fdHJhY2tfY2Jsa190KTsKLSAgICAgICAgICAgbWVtc2V0KG1CdWZmZXIsIDAsIGZyYW1lQ291bnQqY2hhbm5lbENvdW50KnNpemVvZihpbnQxNl90KSk7Ci0gICAgICAgICAgIC8vIEZvcmNlIHVuZGVycnVuIGNvbmRpdGlvbiB0byBhdm9pZCBmYWxzZSB1bmRlcnJ1biBjYWxsYmFjayB1bnRpbCBmaXJzdCBkYXRhIGlzCi0gICAgICAgICAgIC8vIHdyaXR0ZW4gdG8gYnVmZmVyCi0gICAgICAgICAgIG1DYmxrLT5mbG93Q29udHJvbEZsYWcgPSAxOwotICAgICAgICAgICBtQnVmZmVyRW5kID0gKHVpbnQ4X3QgKiltQnVmZmVyICsgYnVmZmVyU2l6ZTsKLSAgICAgICB9Ci0gICB9Ci19Ci0KLUF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlRyYWNrQmFzZTo6flRyYWNrQmFzZSgpCi17Ci0gICAgaWYgKG1DYmxrKSB7Ci0gICAgICAgIG1DYmxrLT5+YXVkaW9fdHJhY2tfY2Jsa190KCk7ICAgLy8gZGVzdHJveSBvdXIgc2hhcmVkLXN0cnVjdHVyZS4gICAgICAgIAotICAgIH0KLSAgICBtQ2Jsa01lbW9yeS5jbGVhcigpOyAgICAgICAgICAgIC8vIGFuZCBmcmVlIHRoZSBzaGFyZWQgbWVtb3J5Ci0gICAgbUNsaWVudC5jbGVhcigpOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlRyYWNrQmFzZTo6cmVsZWFzZUJ1ZmZlcihBdWRpb0J1ZmZlclByb3ZpZGVyOjpCdWZmZXIqIGJ1ZmZlcikKLXsKLSAgICBidWZmZXItPnJhdyA9IDA7Ci0gICAgbUZyYW1lQ291bnQgPSBidWZmZXItPmZyYW1lQ291bnQ7Ci0gICAgc3RlcCgpOwotICAgIGJ1ZmZlci0+ZnJhbWVDb3VudCA9IDA7Ci19Ci0KLWJvb2wgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6VHJhY2tCYXNlOjpzdGVwKCkgewotICAgIGJvb2wgcmVzdWx0OwotICAgIGF1ZGlvX3RyYWNrX2NibGtfdCogY2JsayA9IHRoaXMtPmNibGsoKTsKLQotICAgIHJlc3VsdCA9IGNibGstPnN0ZXBTZXJ2ZXIobUZyYW1lQ291bnQpOwotICAgIGlmICghcmVzdWx0KSB7Ci0gICAgICAgIExPR1YoInN0ZXBTZXJ2ZXIgZmFpbGVkIGFjcXVpcmluZyBjYmxrIG11dGV4Iik7Ci0gICAgICAgIG1GbGFncyB8PSBTVEVQU0VSVkVSX0ZBSUxFRDsKLSAgICB9Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpUcmFja0Jhc2U6OnJlc2V0KCkgewotICAgIGF1ZGlvX3RyYWNrX2NibGtfdCogY2JsayA9IHRoaXMtPmNibGsoKTsKLQotICAgIGNibGstPnVzZXIgPSAwOwotICAgIGNibGstPnNlcnZlciA9IDA7Ci0gICAgY2Jsay0+dXNlckJhc2UgPSAwOwotICAgIGNibGstPnNlcnZlckJhc2UgPSAwOwotICAgIG1GbGFncyAmPSAodWludDMyX3QpKH5TWVNURU1fRkxBR1NfTUFTSyk7Ci0gICAgTE9HVigiVHJhY2tCYXNlOjpyZXNldCIpOwotfQotCi1zcDxJTWVtb3J5PiBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpUcmFja0Jhc2U6OmdldENibGsoKSBjb25zdAotewotICAgIHJldHVybiBtQ2Jsa01lbW9yeTsKLX0KLQotaW50IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlRyYWNrQmFzZTo6c2FtcGxlUmF0ZSgpIGNvbnN0IHsKLSAgICByZXR1cm4gbUNibGstPnNhbXBsZVJhdGU7Ci19Ci0KLWludCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpUcmFja0Jhc2U6OmNoYW5uZWxDb3VudCgpIGNvbnN0IHsKLSAgICByZXR1cm4gbUNibGstPmNoYW5uZWxzOwotfQotCi12b2lkKiBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpUcmFja0Jhc2U6OmdldEJ1ZmZlcih1aW50MzJfdCBvZmZzZXQsIHVpbnQzMl90IGZyYW1lcykgY29uc3QgewotICAgIGF1ZGlvX3RyYWNrX2NibGtfdCogY2JsayA9IHRoaXMtPmNibGsoKTsKLSAgICBpbnQxNl90ICpidWZmZXJTdGFydCA9IChpbnQxNl90ICopbUJ1ZmZlciArIChvZmZzZXQtY2Jsay0+c2VydmVyQmFzZSkqY2Jsay0+Y2hhbm5lbHM7Ci0gICAgaW50MTZfdCAqYnVmZmVyRW5kID0gYnVmZmVyU3RhcnQgKyBmcmFtZXMgKiBjYmxrLT5jaGFubmVsczsKLQotICAgIC8vIENoZWNrIHZhbGlkaXR5IG9mIHJldHVybmVkIHBvaW50ZXIgaW4gY2FzZSB0aGUgdHJhY2sgY29udHJvbCBibG9jayB3b3VsZCBoYXZlIGJlZW4gY29ycnVwdGVkLgotICAgIGlmIChidWZmZXJTdGFydCA8IG1CdWZmZXIgfHwgYnVmZmVyU3RhcnQgPiBidWZmZXJFbmQgfHwgYnVmZmVyRW5kID4gbUJ1ZmZlckVuZCkgewotICAgICAgICBMT0dXKCJUcmFja0Jhc2U6OmdldEJ1ZmZlciBidWZmZXIgb3V0IG9mIHJhbmdlOlxuICAgIHN0YXJ0OiAlcCwgZW5kICVwICwgbUJ1ZmZlciAlcCBtQnVmZmVyRW5kICVwXG4gICAgXAotICAgICAgICAgICAgICAgIHNlcnZlciAlZCwgc2VydmVyQmFzZSAlZCwgdXNlciAlZCwgdXNlckJhc2UgJWQiLAotICAgICAgICAgICAgICAgIGJ1ZmZlclN0YXJ0LCBidWZmZXJFbmQsIG1CdWZmZXIsIG1CdWZmZXJFbmQsCi0gICAgICAgICAgICAgICAgY2Jsay0+c2VydmVyLCBjYmxrLT5zZXJ2ZXJCYXNlLCBjYmxrLT51c2VyLCBjYmxrLT51c2VyQmFzZSk7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIHJldHVybiBidWZmZXJTdGFydDsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1BdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpUcmFjazo6VHJhY2soCi0gICAgICAgICAgICBjb25zdCBzcDxNaXhlclRocmVhZD4mIG1peGVyVGhyZWFkLAotICAgICAgICAgICAgY29uc3Qgc3A8Q2xpZW50PiYgY2xpZW50LAotICAgICAgICAgICAgaW50IHN0cmVhbVR5cGUsCi0gICAgICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlLAotICAgICAgICAgICAgaW50IGZvcm1hdCwKLSAgICAgICAgICAgIGludCBjaGFubmVsQ291bnQsCi0gICAgICAgICAgICBpbnQgZnJhbWVDb3VudCwKLSAgICAgICAgICAgIGNvbnN0IHNwPElNZW1vcnk+JiBzaGFyZWRCdWZmZXIpCi0gICAgOiAgIFRyYWNrQmFzZShtaXhlclRocmVhZCwgY2xpZW50LCBzdHJlYW1UeXBlLCBzYW1wbGVSYXRlLCBmb3JtYXQsIGNoYW5uZWxDb3VudCwgZnJhbWVDb3VudCwgMCwgc2hhcmVkQnVmZmVyKQotewotICAgIG1Wb2x1bWVbMF0gPSAxLjBmOwotICAgIG1Wb2x1bWVbMV0gPSAxLjBmOwotICAgIG1NdXRlID0gZmFsc2U7Ci0gICAgbVNoYXJlZEJ1ZmZlciA9IHNoYXJlZEJ1ZmZlcjsKLX0KLQotQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6VHJhY2s6On5UcmFjaygpCi17Ci0gICAgd3A8VHJhY2s+IHdlYWsodGhpcyk7IC8vIG5ldmVyIGNyZWF0ZSBhIHN0cm9uZyByZWYgZnJvbSB0aGUgZHRvcgotICAgIG1TdGF0ZSA9IFRFUk1JTkFURUQ7Ci0gICAgbU1peGVyVGhyZWFkLT5yZW1vdmVUcmFjayh3ZWFrLCBtTmFtZSk7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6VHJhY2s6OmRlc3Ryb3koKQotewotICAgIG1NaXhlclRocmVhZC0+ZGVzdHJveVRyYWNrKHRoaXMpOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlRyYWNrOjpkdW1wKGNoYXIqIGJ1ZmZlciwgc2l6ZV90IHNpemUpCi17Ci0gICAgc25wcmludGYoYnVmZmVyLCBzaXplLCAiICAlNWQgJTVkICUzdSAlM3UgJTN1ICUzdSAlMWQgJTFkICUxZCAlNXUgJTV1ICU1dSAlMDR4ICUwNHhcbiIsCi0gICAgICAgICAgICBtTmFtZSAtIEF1ZGlvTWl4ZXI6OlRSQUNLMCwKLSAgICAgICAgICAgIChtQ2xpZW50ID09IE5VTEwpID8gZ2V0cGlkKCkgOiBtQ2xpZW50LT5waWQoKSwKLSAgICAgICAgICAgIG1TdHJlYW1UeXBlLAotICAgICAgICAgICAgbUZvcm1hdCwKLSAgICAgICAgICAgIG1DYmxrLT5jaGFubmVscywKLSAgICAgICAgICAgIG1GcmFtZUNvdW50LAotICAgICAgICAgICAgbVN0YXRlLAotICAgICAgICAgICAgbU11dGUsCi0gICAgICAgICAgICBtRmlsbGluZ1VwU3RhdHVzLAotICAgICAgICAgICAgbUNibGstPnNhbXBsZVJhdGUsCi0gICAgICAgICAgICBtQ2Jsay0+dm9sdW1lWzBdLAotICAgICAgICAgICAgbUNibGstPnZvbHVtZVsxXSwKLSAgICAgICAgICAgIG1DYmxrLT5zZXJ2ZXIsCi0gICAgICAgICAgICBtQ2Jsay0+dXNlcik7Ci19Ci0KLXN0YXR1c190IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlRyYWNrOjpnZXROZXh0QnVmZmVyKEF1ZGlvQnVmZmVyUHJvdmlkZXI6OkJ1ZmZlciogYnVmZmVyKQotewotICAgICBhdWRpb190cmFja19jYmxrX3QqIGNibGsgPSB0aGlzLT5jYmxrKCk7Ci0gICAgIHVpbnQzMl90IGZyYW1lc1JlYWR5OwotICAgICB1aW50MzJfdCBmcmFtZXNSZXEgPSBidWZmZXItPmZyYW1lQ291bnQ7Ci0KLSAgICAgLy8gQ2hlY2sgaWYgbGFzdCBzdGVwU2VydmVyIGZhaWxlZCwgdHJ5IHRvIHN0ZXAgbm93Ci0gICAgIGlmIChtRmxhZ3MgJiBUcmFja0Jhc2U6OlNURVBTRVJWRVJfRkFJTEVEKSB7Ci0gICAgICAgICBpZiAoIXN0ZXAoKSkgIGdvdG8gZ2V0TmV4dEJ1ZmZlcl9leGl0OwotICAgICAgICAgTE9HVigic3RlcFNlcnZlciByZWNvdmVyZWQiKTsKLSAgICAgICAgIG1GbGFncyAmPSB+VHJhY2tCYXNlOjpTVEVQU0VSVkVSX0ZBSUxFRDsKLSAgICAgfQotCi0gICAgIGZyYW1lc1JlYWR5ID0gY2Jsay0+ZnJhbWVzUmVhZHkoKTsKLQotICAgICBpZiAoTElLRUxZKGZyYW1lc1JlYWR5KSkgewotICAgICAgICB1aW50MzJfdCBzID0gY2Jsay0+c2VydmVyOwotICAgICAgICB1aW50MzJfdCBidWZmZXJFbmQgPSBjYmxrLT5zZXJ2ZXJCYXNlICsgY2Jsay0+ZnJhbWVDb3VudDsKLQotICAgICAgICBidWZmZXJFbmQgPSAoY2Jsay0+bG9vcEVuZCA8IGJ1ZmZlckVuZCkgPyBjYmxrLT5sb29wRW5kIDogYnVmZmVyRW5kOwotICAgICAgICBpZiAoZnJhbWVzUmVxID4gZnJhbWVzUmVhZHkpIHsKLSAgICAgICAgICAgIGZyYW1lc1JlcSA9IGZyYW1lc1JlYWR5OwotICAgICAgICB9Ci0gICAgICAgIGlmIChzICsgZnJhbWVzUmVxID4gYnVmZmVyRW5kKSB7Ci0gICAgICAgICAgICBmcmFtZXNSZXEgPSBidWZmZXJFbmQgLSBzOwotICAgICAgICB9Ci0KLSAgICAgICAgIGJ1ZmZlci0+cmF3ID0gZ2V0QnVmZmVyKHMsIGZyYW1lc1JlcSk7Ci0gICAgICAgICBpZiAoYnVmZmVyLT5yYXcgPT0gMCkgZ290byBnZXROZXh0QnVmZmVyX2V4aXQ7Ci0KLSAgICAgICAgIGJ1ZmZlci0+ZnJhbWVDb3VudCA9IGZyYW1lc1JlcTsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICB9Ci0KLWdldE5leHRCdWZmZXJfZXhpdDoKLSAgICAgYnVmZmVyLT5yYXcgPSAwOwotICAgICBidWZmZXItPmZyYW1lQ291bnQgPSAwOwotICAgICByZXR1cm4gTk9UX0VOT1VHSF9EQVRBOwotfQotCi1ib29sIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlRyYWNrOjppc1JlYWR5KCkgY29uc3QgewotICAgIGlmIChtRmlsbGluZ1VwU3RhdHVzICE9IEZTX0ZJTExJTkcpIHJldHVybiB0cnVlOwotCi0gICAgaWYgKG1DYmxrLT5mcmFtZXNSZWFkeSgpID49IG1DYmxrLT5mcmFtZUNvdW50IHx8Ci0gICAgICAgIG1DYmxrLT5mb3JjZVJlYWR5KSB7Ci0gICAgICAgIG1GaWxsaW5nVXBTdGF0dXMgPSBGU19GSUxMRUQ7Ci0gICAgICAgIG1DYmxrLT5mb3JjZVJlYWR5ID0gMDsKLSAgICAgICAgTE9HVigiVHJhY2s6OmlzUmVhZHkoKSB0cmFjayAlZCBmb3Igb3V0cHV0ICVkIiwgbU5hbWUsIG1NaXhlclRocmVhZC0+bU91dHB1dFR5cGUpOwotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0gICAgcmV0dXJuIGZhbHNlOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpUcmFjazo6c3RhcnQoKQotewotICAgIExPR1YoInN0YXJ0KCVkKSwgY2FsbGluZyB0aHJlYWQgJWQgZm9yIG91dHB1dCAlZCIsIG1OYW1lLCBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5nZXRDYWxsaW5nUGlkKCksIG1NaXhlclRocmVhZC0+bU91dHB1dFR5cGUpOwotICAgIG1NaXhlclRocmVhZC0+YWRkVHJhY2sodGhpcyk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlRyYWNrOjpzdG9wKCkKLXsKLSAgICBMT0dWKCJzdG9wKCVkKSwgY2FsbGluZyB0aHJlYWQgJWQgZm9yIG91dHB1dCAlZCIsIG1OYW1lLCBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5nZXRDYWxsaW5nUGlkKCksIG1NaXhlclRocmVhZC0+bU91dHB1dFR5cGUpOwotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTWl4ZXJUaHJlYWQtPm1Mb2NrKTsKLSAgICBpZiAobVN0YXRlID4gU1RPUFBFRCkgewotICAgICAgICBtU3RhdGUgPSBTVE9QUEVEOwotICAgICAgICAvLyBJZiB0aGUgdHJhY2sgaXMgbm90IGFjdGl2ZSAoUEFVU0VEIGFuZCBidWZmZXJzIGZ1bGwpLCBmbHVzaCBidWZmZXJzCi0gICAgICAgIGlmIChtTWl4ZXJUaHJlYWQtPm1BY3RpdmVUcmFja3MuaW5kZXhPZih0aGlzKSA8IDApIHsKLSAgICAgICAgICAgIHJlc2V0KCk7Ci0gICAgICAgIH0KLSAgICAgICAgTE9HVigiKD4gU1RPUFBFRCkgPT4gU1RPUFBFRCAoJWQpIiwgbU5hbWUpOwotICAgIH0KLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpUcmFjazo6cGF1c2UoKQotewotICAgIExPR1YoInBhdXNlKCVkKSwgY2FsbGluZyB0aHJlYWQgJWQiLCBtTmFtZSwgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpKTsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobU1peGVyVGhyZWFkLT5tTG9jayk7Ci0gICAgaWYgKG1TdGF0ZSA9PSBBQ1RJVkUgfHwgbVN0YXRlID09IFJFU1VNSU5HKSB7Ci0gICAgICAgIG1TdGF0ZSA9IFBBVVNJTkc7Ci0gICAgICAgIExPR1YoIkFDVElWRS9SRVNVTUlORyA9PiBQQVVTSU5HICglZCkiLCBtTmFtZSk7Ci0gICAgfQotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlRyYWNrOjpmbHVzaCgpCi17Ci0gICAgTE9HVigiZmx1c2goJWQpIiwgbU5hbWUpOwotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTWl4ZXJUaHJlYWQtPm1Mb2NrKTsKLSAgICBpZiAobVN0YXRlICE9IFNUT1BQRUQgJiYgbVN0YXRlICE9IFBBVVNFRCAmJiBtU3RhdGUgIT0gUEFVU0lORykgewotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIC8vIE5vIHBvaW50IHJlbWFpbmluZyBpbiBQQVVTRUQgc3RhdGUgYWZ0ZXIgYSBmbHVzaCA9PiBnbyB0bwotICAgIC8vIFNUT1BQRUQgc3RhdGUKLSAgICBtU3RhdGUgPSBTVE9QUEVEOwotCi0gICAgLy8gTk9URTogcmVzZXQoKSB3aWxsIHJlc2V0IGNibGstPnVzZXIgYW5kIGNibGstPnNlcnZlciB3aXRoCi0gICAgLy8gdGhlIHJpc2sgdGhhdCBhdCB0aGUgc2FtZSB0aW1lLCB0aGUgQXVkaW9NaXhlciBpcyB0cnlpbmcgdG8gcmVhZAotICAgIC8vIGRhdGEuIEluIHRoaXMgY2FzZSwgZ2V0TmV4dEJ1ZmZlcigpIHdvdWxkIHJldHVybiBhIE5VTEwgcG9pbnRlcgotICAgIC8vIGFzIGF1ZGlvIGJ1ZmZlciA9PiB0aGUgQXVkaW9NaXhlciBjb2RlIE1VU1QgYWx3YXlzIHRlc3QgdGhhdCBwb2ludGVyCi0gICAgLy8gcmV0dXJuZWQgYnkgZ2V0TmV4dEJ1ZmZlcigpIGlzIG5vdCBOVUxMIQotICAgIHJlc2V0KCk7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6VHJhY2s6OnJlc2V0KCkKLXsKLSAgICAvLyBEbyBub3QgcmVzZXQgdHdpY2UgdG8gYXZvaWQgZGlzY2FyZGluZyBkYXRhIHdyaXR0ZW4ganVzdCBhZnRlciBhIGZsdXNoIGFuZCBiZWZvcmUKLSAgICAvLyB0aGUgYXVkaW9mbGluZ2VyIHRocmVhZCBkZXRlY3RzIHRoZSB0cmFjayBpcyBzdG9wcGVkLgotICAgIGlmICghbVJlc2V0RG9uZSkgewotICAgICAgICBUcmFja0Jhc2U6OnJlc2V0KCk7Ci0gICAgICAgIC8vIEZvcmNlIHVuZGVycnVuIGNvbmRpdGlvbiB0byBhdm9pZCBmYWxzZSB1bmRlcnJ1biBjYWxsYmFjayB1bnRpbCBmaXJzdCBkYXRhIGlzCi0gICAgICAgIC8vIHdyaXR0ZW4gdG8gYnVmZmVyCi0gICAgICAgIG1DYmxrLT5mbG93Q29udHJvbEZsYWcgPSAxOwotICAgICAgICBtQ2Jsay0+Zm9yY2VSZWFkeSA9IDA7Ci0gICAgICAgIG1GaWxsaW5nVXBTdGF0dXMgPSBGU19GSUxMSU5HOyAgICAgICAgCi0gICAgICAgIG1SZXNldERvbmUgPSB0cnVlOwotICAgIH0KLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpUcmFjazo6bXV0ZShib29sIG11dGVkKQotewotICAgIG1NdXRlID0gbXV0ZWQ7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6VHJhY2s6OnNldFZvbHVtZShmbG9hdCBsZWZ0LCBmbG9hdCByaWdodCkKLXsKLSAgICBtVm9sdW1lWzBdID0gbGVmdDsKLSAgICBtVm9sdW1lWzFdID0gcmlnaHQ7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6UmVjb3JkVHJhY2s6OlJlY29yZFRyYWNrKAotICAgICAgICAgICAgY29uc3Qgc3A8TWl4ZXJUaHJlYWQ+JiBtaXhlclRocmVhZCwKLSAgICAgICAgICAgIGNvbnN0IHNwPENsaWVudD4mIGNsaWVudCwKLSAgICAgICAgICAgIGludCBzdHJlYW1UeXBlLAotICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgICAgIGludCBmb3JtYXQsCi0gICAgICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICAgICAgaW50IGZyYW1lQ291bnQsCi0gICAgICAgICAgICB1aW50MzJfdCBmbGFncykKLSAgICA6ICAgVHJhY2tCYXNlKG1peGVyVGhyZWFkLCBjbGllbnQsIHN0cmVhbVR5cGUsIHNhbXBsZVJhdGUsIGZvcm1hdCwKLSAgICAgICAgICAgICAgICAgIGNoYW5uZWxDb3VudCwgZnJhbWVDb3VudCwgZmxhZ3MsIDApLAotICAgICAgICBtT3ZlcmZsb3coZmFsc2UpCi17Ci19Ci0KLUF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlJlY29yZFRyYWNrOjp+UmVjb3JkVHJhY2soKQotewotICAgIG1NaXhlclRocmVhZC0+ZGVsZXRlVHJhY2tOYW1lKG1OYW1lKTsKLX0KLQotc3RhdHVzX3QgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6UmVjb3JkVHJhY2s6OmdldE5leHRCdWZmZXIoQXVkaW9CdWZmZXJQcm92aWRlcjo6QnVmZmVyKiBidWZmZXIpCi17Ci0gICAgYXVkaW9fdHJhY2tfY2Jsa190KiBjYmxrID0gdGhpcy0+Y2JsaygpOwotICAgIHVpbnQzMl90IGZyYW1lc0F2YWlsOwotICAgIHVpbnQzMl90IGZyYW1lc1JlcSA9IGJ1ZmZlci0+ZnJhbWVDb3VudDsKLQotICAgICAvLyBDaGVjayBpZiBsYXN0IHN0ZXBTZXJ2ZXIgZmFpbGVkLCB0cnkgdG8gc3RlcCBub3cKLSAgICBpZiAobUZsYWdzICYgVHJhY2tCYXNlOjpTVEVQU0VSVkVSX0ZBSUxFRCkgewotICAgICAgICBpZiAoIXN0ZXAoKSkgZ290byBnZXROZXh0QnVmZmVyX2V4aXQ7Ci0gICAgICAgIExPR1YoInN0ZXBTZXJ2ZXIgcmVjb3ZlcmVkIik7Ci0gICAgICAgIG1GbGFncyAmPSB+VHJhY2tCYXNlOjpTVEVQU0VSVkVSX0ZBSUxFRDsKLSAgICB9Ci0KLSAgICBmcmFtZXNBdmFpbCA9IGNibGstPmZyYW1lc0F2YWlsYWJsZV9sKCk7Ci0KLSAgICBpZiAoTElLRUxZKGZyYW1lc0F2YWlsKSkgewotICAgICAgICB1aW50MzJfdCBzID0gY2Jsay0+c2VydmVyOwotICAgICAgICB1aW50MzJfdCBidWZmZXJFbmQgPSBjYmxrLT5zZXJ2ZXJCYXNlICsgY2Jsay0+ZnJhbWVDb3VudDsKLQotICAgICAgICBpZiAoZnJhbWVzUmVxID4gZnJhbWVzQXZhaWwpIHsKLSAgICAgICAgICAgIGZyYW1lc1JlcSA9IGZyYW1lc0F2YWlsOwotICAgICAgICB9Ci0gICAgICAgIGlmIChzICsgZnJhbWVzUmVxID4gYnVmZmVyRW5kKSB7Ci0gICAgICAgICAgICBmcmFtZXNSZXEgPSBidWZmZXJFbmQgLSBzOwotICAgICAgICB9Ci0KLSAgICAgICAgYnVmZmVyLT5yYXcgPSBnZXRCdWZmZXIocywgZnJhbWVzUmVxKTsKLSAgICAgICAgaWYgKGJ1ZmZlci0+cmF3ID09IDApIGdvdG8gZ2V0TmV4dEJ1ZmZlcl9leGl0OwotCi0gICAgICAgIGJ1ZmZlci0+ZnJhbWVDb3VudCA9IGZyYW1lc1JlcTsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLQotZ2V0TmV4dEJ1ZmZlcl9leGl0OgotICAgIGJ1ZmZlci0+cmF3ID0gMDsKLSAgICBidWZmZXItPmZyYW1lQ291bnQgPSAwOwotICAgIHJldHVybiBOT1RfRU5PVUdIX0RBVEE7Ci19Ci0KLXN0YXR1c190IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlJlY29yZFRyYWNrOjpzdGFydCgpCi17Ci0gICAgcmV0dXJuIG1NaXhlclRocmVhZC0+bUF1ZGlvRmxpbmdlci0+c3RhcnRSZWNvcmQodGhpcyk7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6UmVjb3JkVHJhY2s6OnN0b3AoKQotewotICAgIG1NaXhlclRocmVhZC0+bUF1ZGlvRmxpbmdlci0+c3RvcFJlY29yZCh0aGlzKTsKLSAgICBUcmFja0Jhc2U6OnJlc2V0KCk7Ci0gICAgLy8gRm9yY2Ugb3ZlcmVycnVuIGNvbmRpdGlvbiB0byBhdm9pZCBmYWxzZSBvdmVycnVuIGNhbGxiYWNrIHVudGlsIGZpcnN0IGRhdGEgaXMKLSAgICAvLyByZWFkIGZyb20gYnVmZmVyCi0gICAgbUNibGstPmZsb3dDb250cm9sRmxhZyA9IDE7Ci19Ci0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1BdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpPdXRwdXRUcmFjazo6T3V0cHV0VHJhY2soCi0gICAgICAgICAgICBjb25zdCBzcDxNaXhlclRocmVhZD4mIG1peGVyVGhyZWFkLAotICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgICAgIGludCBmb3JtYXQsCi0gICAgICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICAgICAgaW50IGZyYW1lQ291bnQpCi0gICAgOiAgIFRyYWNrKG1peGVyVGhyZWFkLCBOVUxMLCBBdWRpb1N5c3RlbTo6U1lTVEVNLCBzYW1wbGVSYXRlLCBmb3JtYXQsIGNoYW5uZWxDb3VudCwgZnJhbWVDb3VudCwgTlVMTCksCi0gICAgbU91dHB1dE1peGVyVGhyZWFkKG1peGVyVGhyZWFkKQotewotICAgICAgICAgICAgICAgIAotICAgIG1DYmxrLT5vdXQgPSAxOwotICAgIG1DYmxrLT5idWZmZXJzID0gKGNoYXIqKW1DYmxrICsgc2l6ZW9mKGF1ZGlvX3RyYWNrX2NibGtfdCk7Ci0gICAgbUNibGstPnZvbHVtZVswXSA9IG1DYmxrLT52b2x1bWVbMV0gPSAweDEwMDA7Ci0gICAgbU91dEJ1ZmZlci5mcmFtZUNvdW50ID0gMDsKLSAgICBtQ2Jsay0+YnVmZmVyVGltZW91dE1zID0gMTA7Ci0gICAgCi0gICAgTE9HVigiT3V0cHV0VHJhY2sgY29uc3RydWN0b3IgbUNibGsgJXAsIG1CdWZmZXIgJXAsIG1DYmxrLT5idWZmZXJzICVwLCBtQ2Jsay0+ZnJhbWVDb3VudCAlZCwgbUNibGstPnNhbXBsZVJhdGUgJWQsIG1DYmxrLT5jaGFubmVscyAlZCBtQnVmZmVyRW5kICVwIiwgCi0gICAgICAgICAgICBtQ2JsaywgbUJ1ZmZlciwgbUNibGstPmJ1ZmZlcnMsIG1DYmxrLT5mcmFtZUNvdW50LCBtQ2Jsay0+c2FtcGxlUmF0ZSwgbUNibGstPmNoYW5uZWxzLCBtQnVmZmVyRW5kKTsKLSAgICAKLX0KLQotQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6T3V0cHV0VHJhY2s6On5PdXRwdXRUcmFjaygpCi17Ci0gICAgc3RvcCgpOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpPdXRwdXRUcmFjazo6c3RhcnQoKQotewotICAgIHN0YXR1c190IHN0YXR1cyA9IFRyYWNrOjpzdGFydCgpOwotICAgIAotICAgIG1SZXRyeUNvdW50ID0gMTI3OwotICAgIHJldHVybiBzdGF0dXM7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6T3V0cHV0VHJhY2s6OnN0b3AoKQotewotICAgIFRyYWNrOjpzdG9wKCk7Ci0gICAgY2xlYXJCdWZmZXJRdWV1ZSgpOwotICAgIG1PdXRCdWZmZXIuZnJhbWVDb3VudCA9IDA7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6T3V0cHV0VHJhY2s6OndyaXRlKGludDE2X3QqIGRhdGEsIHVpbnQzMl90IGZyYW1lcykKLXsKLSAgICBCdWZmZXIgKnBJbkJ1ZmZlcjsKLSAgICBCdWZmZXIgaW5CdWZmZXI7Ci0gICAgdWludDMyX3QgY2hhbm5lbHMgPSBtQ2Jsay0+Y2hhbm5lbHM7Ci0gICAgICAgIAotICAgIGluQnVmZmVyLmZyYW1lQ291bnQgPSBmcmFtZXM7Ci0gICAgaW5CdWZmZXIuaTE2ID0gZGF0YTsKLSAgICAKLSAgICBpZiAobUNibGstPnVzZXIgPT0gMCkgewotICAgICAgICBpZiAobU91dHB1dE1peGVyVGhyZWFkLT5pc011c2ljQWN0aXZlKCkpIHsKLSAgICAgICAgICAgIG1DYmxrLT5mb3JjZVJlYWR5ID0gMTsKLSAgICAgICAgICAgIExPR1YoIk91dHB1dFRyYWNrOjpzdGFydCgpIGZvcmNlIHJlYWR5Iik7Ci0gICAgICAgIH0gZWxzZSBpZiAobUNibGstPmZyYW1lQ291bnQgPiBmcmFtZXMpewotICAgICAgICAgICAgaWYgKG1CdWZmZXJRdWV1ZS5zaXplKCkgPCBrTWF4T3V0cHV0VHJhY2tCdWZmZXJzKSB7Ci0gICAgICAgICAgICAgICAgdWludDMyX3Qgc3RhcnRGcmFtZXMgPSAobUNibGstPmZyYW1lQ291bnQgLSBmcmFtZXMpOwotICAgICAgICAgICAgICAgIExPR1YoIk91dHB1dFRyYWNrOjpzdGFydCgpIHdyaXRlICVkIGZyYW1lcyIsIHN0YXJ0RnJhbWVzKTsKLSAgICAgICAgICAgICAgICBwSW5CdWZmZXIgPSBuZXcgQnVmZmVyOwotICAgICAgICAgICAgICAgIHBJbkJ1ZmZlci0+bUJ1ZmZlciA9IG5ldyBpbnQxNl90W3N0YXJ0RnJhbWVzICogY2hhbm5lbHNdOwotICAgICAgICAgICAgICAgIHBJbkJ1ZmZlci0+ZnJhbWVDb3VudCA9IHN0YXJ0RnJhbWVzOwotICAgICAgICAgICAgICAgIHBJbkJ1ZmZlci0+aTE2ID0gcEluQnVmZmVyLT5tQnVmZmVyOwotICAgICAgICAgICAgICAgIG1lbXNldChwSW5CdWZmZXItPnJhdywgMCwgc3RhcnRGcmFtZXMgKiBjaGFubmVscyAqIHNpemVvZihpbnQxNl90KSk7Ci0gICAgICAgICAgICAgICAgbUJ1ZmZlclF1ZXVlLmFkZChwSW5CdWZmZXIpOyAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgTE9HVyAoIk91dHB1dFRyYWNrOjp3cml0ZSgpIG5vIG1vcmUgYnVmZmVycyIpOwotICAgICAgICAgICAgfQotICAgICAgICB9ICAgICAgICAKLSAgICB9Ci0KLSAgICB3aGlsZSAoMSkgeyAKLSAgICAgICAgLy8gRmlyc3Qgd3JpdGUgcGVuZGluZyBidWZmZXJzLCB0aGVuIG5ldyBkYXRhCi0gICAgICAgIGlmIChtQnVmZmVyUXVldWUuc2l6ZSgpKSB7Ci0gICAgICAgICAgICBwSW5CdWZmZXIgPSBtQnVmZmVyUXVldWUuaXRlbUF0KDApOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcEluQnVmZmVyID0gJmluQnVmZmVyOwotICAgICAgICB9Ci0gCi0gICAgICAgIGlmIChwSW5CdWZmZXItPmZyYW1lQ291bnQgPT0gMCkgewotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIGlmIChtT3V0QnVmZmVyLmZyYW1lQ291bnQgPT0gMCkgewotICAgICAgICAgICAgbU91dEJ1ZmZlci5mcmFtZUNvdW50ID0gcEluQnVmZmVyLT5mcmFtZUNvdW50OwotICAgICAgICAgICAgaWYgKG9idGFpbkJ1ZmZlcigmbU91dEJ1ZmZlcikgPT0gKHN0YXR1c190KUF1ZGlvVHJhY2s6Ok5PX01PUkVfQlVGRkVSUykgewotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgICAgICAKLSAgICAgICAgdWludDMyX3Qgb3V0RnJhbWVzID0gcEluQnVmZmVyLT5mcmFtZUNvdW50ID4gbU91dEJ1ZmZlci5mcmFtZUNvdW50ID8gbU91dEJ1ZmZlci5mcmFtZUNvdW50IDogcEluQnVmZmVyLT5mcmFtZUNvdW50OwotICAgICAgICBtZW1jcHkobU91dEJ1ZmZlci5yYXcsIHBJbkJ1ZmZlci0+cmF3LCBvdXRGcmFtZXMgKiBjaGFubmVscyAqIHNpemVvZihpbnQxNl90KSk7Ci0gICAgICAgIG1DYmxrLT5zdGVwVXNlcihvdXRGcmFtZXMpOwotICAgICAgICBwSW5CdWZmZXItPmZyYW1lQ291bnQgLT0gb3V0RnJhbWVzOwotICAgICAgICBwSW5CdWZmZXItPmkxNiArPSBvdXRGcmFtZXMgKiBjaGFubmVsczsKLSAgICAgICAgbU91dEJ1ZmZlci5mcmFtZUNvdW50IC09IG91dEZyYW1lczsKLSAgICAgICAgbU91dEJ1ZmZlci5pMTYgKz0gb3V0RnJhbWVzICogY2hhbm5lbHM7ICAgICAgICAgICAgCi0gICAgICAgIAotICAgICAgICBpZiAocEluQnVmZmVyLT5mcmFtZUNvdW50ID09IDApIHsKLSAgICAgICAgICAgIGlmIChtQnVmZmVyUXVldWUuc2l6ZSgpKSB7Ci0gICAgICAgICAgICAgICAgbUJ1ZmZlclF1ZXVlLnJlbW92ZUF0KDApOwotICAgICAgICAgICAgICAgIGRlbGV0ZSBbXSBwSW5CdWZmZXItPm1CdWZmZXI7Ci0gICAgICAgICAgICAgICAgZGVsZXRlIHBJbkJ1ZmZlcjsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gCi0gICAgLy8gSWYgd2UgY291bGQgbm90IHdyaXRlIGFsbCBmcmFtZXMsIGFsbG9jYXRlIGEgYnVmZmVyIGFuZCBxdWV1ZSBpdCBmb3IgbmV4dCB0aW1lLgotICAgIGlmIChpbkJ1ZmZlci5mcmFtZUNvdW50KSB7Ci0gICAgICAgIGlmIChtQnVmZmVyUXVldWUuc2l6ZSgpIDwga01heE91dHB1dFRyYWNrQnVmZmVycykgewotICAgICAgICAgICAgcEluQnVmZmVyID0gbmV3IEJ1ZmZlcjsKLSAgICAgICAgICAgIHBJbkJ1ZmZlci0+bUJ1ZmZlciA9IG5ldyBpbnQxNl90W2luQnVmZmVyLmZyYW1lQ291bnQgKiBjaGFubmVsc107Ci0gICAgICAgICAgICBwSW5CdWZmZXItPmZyYW1lQ291bnQgPSBpbkJ1ZmZlci5mcmFtZUNvdW50OwotICAgICAgICAgICAgcEluQnVmZmVyLT5pMTYgPSBwSW5CdWZmZXItPm1CdWZmZXI7Ci0gICAgICAgICAgICBtZW1jcHkocEluQnVmZmVyLT5yYXcsIGluQnVmZmVyLnJhdywgaW5CdWZmZXIuZnJhbWVDb3VudCAqIGNoYW5uZWxzICogc2l6ZW9mKGludDE2X3QpKTsKLSAgICAgICAgICAgIG1CdWZmZXJRdWV1ZS5hZGQocEluQnVmZmVyKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIExPR1coIk91dHB1dFRyYWNrOjp3cml0ZSgpIG5vIG1vcmUgYnVmZmVycyIpOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIC8vIENhbGxpbmcgd3JpdGUoKSB3aXRoIGEgMCBsZW5ndGggYnVmZmVyLCBtZWFucyB0aGF0IG5vIG1vcmUgZGF0YSB3aWxsIGJlIHdyaXR0ZW46Ci0gICAgLy8gSWYgbm8gbW9yZSBidWZmZXJzIGFyZSBwZW5kaW5nLCBmaWxsIG91dHB1dCB0cmFjayBidWZmZXIgdG8gbWFrZSBzdXJlIGl0IGlzIHN0YXJ0ZWQgCi0gICAgLy8gYnkgb3V0cHV0IG1peGVyLgotICAgIGlmIChmcmFtZXMgPT0gMCAmJiBtQnVmZmVyUXVldWUuc2l6ZSgpID09IDAgJiYgbUNibGstPnVzZXIgPCBtQ2Jsay0+ZnJhbWVDb3VudCkgewotICAgICAgICBmcmFtZXMgPSBtQ2Jsay0+ZnJhbWVDb3VudCAtIG1DYmxrLT51c2VyOwotICAgICAgICBwSW5CdWZmZXIgPSBuZXcgQnVmZmVyOwotICAgICAgICBwSW5CdWZmZXItPm1CdWZmZXIgPSBuZXcgaW50MTZfdFtmcmFtZXMgKiBjaGFubmVsc107Ci0gICAgICAgIHBJbkJ1ZmZlci0+ZnJhbWVDb3VudCA9IGZyYW1lczsKLSAgICAgICAgcEluQnVmZmVyLT5pMTYgPSBwSW5CdWZmZXItPm1CdWZmZXI7Ci0gICAgICAgIG1lbXNldChwSW5CdWZmZXItPnJhdywgMCwgZnJhbWVzICogY2hhbm5lbHMgKiBzaXplb2YoaW50MTZfdCkpOwotICAgICAgICBtQnVmZmVyUXVldWUuYWRkKHBJbkJ1ZmZlcik7Ci0gICAgfQotCi19Ci0KLXN0YXR1c190IEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6Ok91dHB1dFRyYWNrOjpvYnRhaW5CdWZmZXIoQXVkaW9CdWZmZXJQcm92aWRlcjo6QnVmZmVyKiBidWZmZXIpCi17Ci0gICAgaW50IGFjdGl2ZTsKLSAgICBpbnQgdGltZW91dCA9IDA7Ci0gICAgc3RhdHVzX3QgcmVzdWx0OwotICAgIGF1ZGlvX3RyYWNrX2NibGtfdCogY2JsayA9IG1DYmxrOwotICAgIHVpbnQzMl90IGZyYW1lc1JlcSA9IGJ1ZmZlci0+ZnJhbWVDb3VudDsKLQotICAgIExPR1YoIk91dHB1dFRyYWNrOjpvYnRhaW5CdWZmZXIgdXNlciAlZCwgc2VydmVyICVkIiwgY2Jsay0+dXNlciwgY2Jsay0+c2VydmVyKTsKLSAgICBidWZmZXItPmZyYW1lQ291bnQgID0gMDsKLSAgICAKLSAgICB1aW50MzJfdCBmcmFtZXNBdmFpbCA9IGNibGstPmZyYW1lc0F2YWlsYWJsZSgpOwotCi0gICAgaWYgKGZyYW1lc0F2YWlsID09IDApIHsKLSAgICAgICAgcmV0dXJuIEF1ZGlvVHJhY2s6Ok5PX01PUkVfQlVGRkVSUzsKLSAgICB9Ci0KLSAgICBpZiAoZnJhbWVzUmVxID4gZnJhbWVzQXZhaWwpIHsKLSAgICAgICAgZnJhbWVzUmVxID0gZnJhbWVzQXZhaWw7Ci0gICAgfQotCi0gICAgdWludDMyX3QgdSA9IGNibGstPnVzZXI7Ci0gICAgdWludDMyX3QgYnVmZmVyRW5kID0gY2Jsay0+dXNlckJhc2UgKyBjYmxrLT5mcmFtZUNvdW50OwotCi0gICAgaWYgKHUgKyBmcmFtZXNSZXEgPiBidWZmZXJFbmQpIHsKLSAgICAgICAgZnJhbWVzUmVxID0gYnVmZmVyRW5kIC0gdTsKLSAgICB9Ci0KLSAgICBidWZmZXItPmZyYW1lQ291bnQgID0gZnJhbWVzUmVxOwotICAgIGJ1ZmZlci0+cmF3ICAgICAgICAgPSAodm9pZCAqKWNibGstPmJ1ZmZlcih1KTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6Ok1peGVyVGhyZWFkOjpPdXRwdXRUcmFjazo6Y2xlYXJCdWZmZXJRdWV1ZSgpCi17Ci0gICAgc2l6ZV90IHNpemUgPSBtQnVmZmVyUXVldWUuc2l6ZSgpOwotICAgIEJ1ZmZlciAqcEJ1ZmZlcjsKLSAgICAKLSAgICBmb3IgKHNpemVfdCBpID0gMDsgaSA8IHNpemU7IGkrKykgewotICAgICAgICBwQnVmZmVyID0gbUJ1ZmZlclF1ZXVlLml0ZW1BdChpKTsKLSAgICAgICAgZGVsZXRlIFtdIHBCdWZmZXItPm1CdWZmZXI7Ci0gICAgICAgIGRlbGV0ZSBwQnVmZmVyOwotICAgIH0KLSAgICBtQnVmZmVyUXVldWUuY2xlYXIoKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1BdWRpb0ZsaW5nZXI6OkNsaWVudDo6Q2xpZW50KGNvbnN0IHNwPEF1ZGlvRmxpbmdlcj4mIGF1ZGlvRmxpbmdlciwgcGlkX3QgcGlkKQotICAgIDogICBSZWZCYXNlKCksCi0gICAgICAgIG1BdWRpb0ZsaW5nZXIoYXVkaW9GbGluZ2VyKSwKLSAgICAgICAgbU1lbW9yeURlYWxlcihuZXcgTWVtb3J5RGVhbGVyKDEwMjQqMTAyNCkpLAotICAgICAgICBtUGlkKHBpZCkKLXsKLSAgICAvLyAxIE1CIG9mIGFkZHJlc3Mgc3BhY2UgaXMgZ29vZCBmb3IgMzIgdHJhY2tzLCA4IGJ1ZmZlcnMgZWFjaCwgNCBLQi9idWZmZXIKLX0KLQotQXVkaW9GbGluZ2VyOjpDbGllbnQ6On5DbGllbnQoKQotewotICAgIG1BdWRpb0ZsaW5nZXItPnJlbW92ZUNsaWVudChtUGlkKTsKLX0KLQotY29uc3Qgc3A8TWVtb3J5RGVhbGVyPiYgQXVkaW9GbGluZ2VyOjpDbGllbnQ6OmhlYXAoKSBjb25zdAotewotICAgIHJldHVybiBtTWVtb3J5RGVhbGVyOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUF1ZGlvRmxpbmdlcjo6VHJhY2tIYW5kbGU6OlRyYWNrSGFuZGxlKGNvbnN0IHNwPEF1ZGlvRmxpbmdlcjo6TWl4ZXJUaHJlYWQ6OlRyYWNrPiYgdHJhY2spCi0gICAgOiBCbkF1ZGlvVHJhY2soKSwKLSAgICAgIG1UcmFjayh0cmFjaykKLXsKLX0KLQotQXVkaW9GbGluZ2VyOjpUcmFja0hhbmRsZTo6flRyYWNrSGFuZGxlKCkgewotICAgIC8vIGp1c3Qgc3RvcCB0aGUgdHJhY2sgb24gZGVsZXRpb24sIGFzc29jaWF0ZWQgcmVzb3VyY2VzCi0gICAgLy8gd2lsbCBiZSBmcmVlZCBmcm9tIHRoZSBtYWluIHRocmVhZCBvbmNlIGFsbCBwZW5kaW5nIGJ1ZmZlcnMgaGF2ZQotICAgIC8vIGJlZW4gcGxheWVkLiBVbmxlc3MgaXQncyBub3QgaW4gdGhlIGFjdGl2ZSB0cmFjayBsaXN0LCBpbiB3aGljaAotICAgIC8vIGNhc2Ugd2UgZnJlZSBldmVyeXRoaW5nIG5vdy4uLgotICAgIG1UcmFjay0+ZGVzdHJveSgpOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6OlRyYWNrSGFuZGxlOjpzdGFydCgpIHsKLSAgICByZXR1cm4gbVRyYWNrLT5zdGFydCgpOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6VHJhY2tIYW5kbGU6OnN0b3AoKSB7Ci0gICAgbVRyYWNrLT5zdG9wKCk7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpUcmFja0hhbmRsZTo6Zmx1c2goKSB7Ci0gICAgbVRyYWNrLT5mbHVzaCgpOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6VHJhY2tIYW5kbGU6Om11dGUoYm9vbCBlKSB7Ci0gICAgbVRyYWNrLT5tdXRlKGUpOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6VHJhY2tIYW5kbGU6OnBhdXNlKCkgewotICAgIG1UcmFjay0+cGF1c2UoKTsKLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6OlRyYWNrSGFuZGxlOjpzZXRWb2x1bWUoZmxvYXQgbGVmdCwgZmxvYXQgcmlnaHQpIHsKLSAgICBtVHJhY2stPnNldFZvbHVtZShsZWZ0LCByaWdodCk7Ci19Ci0KLXNwPElNZW1vcnk+IEF1ZGlvRmxpbmdlcjo6VHJhY2tIYW5kbGU6OmdldENibGsoKSBjb25zdCB7Ci0gICAgcmV0dXJuIG1UcmFjay0+Z2V0Q2JsaygpOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6OlRyYWNrSGFuZGxlOjpvblRyYW5zYWN0KAotICAgIHVpbnQzMl90IGNvZGUsIGNvbnN0IFBhcmNlbCYgZGF0YSwgUGFyY2VsKiByZXBseSwgdWludDMyX3QgZmxhZ3MpCi17Ci0gICAgcmV0dXJuIEJuQXVkaW9UcmFjazo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXNwPElBdWRpb1JlY29yZD4gQXVkaW9GbGluZ2VyOjpvcGVuUmVjb3JkKAotICAgICAgICBwaWRfdCBwaWQsCi0gICAgICAgIGludCBzdHJlYW1UeXBlLAotICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlLAotICAgICAgICBpbnQgZm9ybWF0LAotICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICBpbnQgZnJhbWVDb3VudCwKLSAgICAgICAgdWludDMyX3QgZmxhZ3MsCi0gICAgICAgIHN0YXR1c190ICpzdGF0dXMpCi17Ci0gICAgc3A8QXVkaW9SZWNvcmRUaHJlYWQ+IHRocmVhZDsKLSAgICBzcDxNaXhlclRocmVhZDo6UmVjb3JkVHJhY2s+IHJlY29yZFRyYWNrOwotICAgIHNwPFJlY29yZEhhbmRsZT4gcmVjb3JkSGFuZGxlOwotICAgIHNwPENsaWVudD4gY2xpZW50OwotICAgIHdwPENsaWVudD4gd2NsaWVudDsKLSAgICBBdWRpb1N0cmVhbUluKiBpbnB1dCA9IDA7Ci0gICAgaW50IGluRnJhbWVDb3VudDsKLSAgICBzaXplX3QgaW5wdXRCdWZmZXJTaXplOwotICAgIHN0YXR1c190IGxTdGF0dXM7Ci0KLSAgICAvLyBjaGVjayBjYWxsaW5nIHBlcm1pc3Npb25zCi0gICAgaWYgKCFyZWNvcmRpbmdBbGxvd2VkKCkpIHsKLSAgICAgICAgbFN0YXR1cyA9IFBFUk1JU1NJT05fREVOSUVEOwotICAgICAgICBnb3RvIEV4aXQ7Ci0gICAgfQotCi0gICAgaWYgKHVpbnQzMl90KHN0cmVhbVR5cGUpID49IEF1ZGlvUmVjb3JkOjpOVU1fU1RSRUFNX1RZUEVTKSB7Ci0gICAgICAgIExPR0UoImludmFsaWQgc3RyZWFtIHR5cGUiKTsKLSAgICAgICAgbFN0YXR1cyA9IEJBRF9WQUxVRTsKLSAgICAgICAgZ290byBFeGl0OwotICAgIH0KLQotICAgIGlmIChzYW1wbGVSYXRlID4gTUFYX1NBTVBMRV9SQVRFKSB7Ci0gICAgICAgIExPR0UoIlNhbXBsZSByYXRlIG91dCBvZiByYW5nZSIpOwotICAgICAgICBsU3RhdHVzID0gQkFEX1ZBTFVFOwotICAgICAgICBnb3RvIEV4aXQ7Ci0gICAgfQotCi0gICAgaWYgKG1BdWRpb1JlY29yZFRocmVhZCA9PSAwKSB7Ci0gICAgICAgIExPR0UoIkF1ZGlvIHJlY29yZCB0aHJlYWQgbm90IHN0YXJ0ZWQiKTsKLSAgICAgICAgbFN0YXR1cyA9IE5PX0lOSVQ7Ci0gICAgICAgIGdvdG8gRXhpdDsKLSAgICB9Ci0KLQotICAgIC8vIENoZWNrIHRoYXQgYXVkaW8gaW5wdXQgc3RyZWFtIGFjY2VwdHMgcmVxdWVzdGVkIGF1ZGlvIHBhcmFtZXRlcnMgCi0gICAgaW5wdXRCdWZmZXJTaXplID0gbUF1ZGlvSGFyZHdhcmUtPmdldElucHV0QnVmZmVyU2l6ZShzYW1wbGVSYXRlLCBmb3JtYXQsIGNoYW5uZWxDb3VudCk7Ci0gICAgaWYgKGlucHV0QnVmZmVyU2l6ZSA9PSAwKSB7Ci0gICAgICAgIGxTdGF0dXMgPSBCQURfVkFMVUU7Ci0gICAgICAgIExPR0UoIkJhZCBhdWRpbyBpbnB1dCBwYXJhbWV0ZXJzOiBzYW1wbGluZyByYXRlICV1LCBmb3JtYXQgJWQsIGNoYW5uZWxzICVkIiwgIHNhbXBsZVJhdGUsIGZvcm1hdCwgY2hhbm5lbENvdW50KTsKLSAgICAgICAgZ290byBFeGl0OwotICAgIH0KLQotICAgIC8vIGFkZCBjbGllbnQgdG8gbGlzdAotICAgIHsKLSAgICAgICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICAgICAgd2NsaWVudCA9IG1DbGllbnRzLnZhbHVlRm9yKHBpZCk7Ci0gICAgICAgIGlmICh3Y2xpZW50ICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGNsaWVudCA9IHdjbGllbnQucHJvbW90ZSgpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgY2xpZW50ID0gbmV3IENsaWVudCh0aGlzLCBwaWQpOwotICAgICAgICAgICAgbUNsaWVudHMuYWRkKHBpZCwgY2xpZW50KTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vIGZyYW1lQ291bnQgbXVzdCBiZSBhIG11bHRpcGxlIG9mIGlucHV0IGJ1ZmZlciBzaXplCi0gICAgaW5GcmFtZUNvdW50ID0gaW5wdXRCdWZmZXJTaXplL2NoYW5uZWxDb3VudC9zaXplb2Yoc2hvcnQpOwotICAgIGZyYW1lQ291bnQgPSAoKGZyYW1lQ291bnQgLSAxKS9pbkZyYW1lQ291bnQgKyAxKSAqIGluRnJhbWVDb3VudDsKLQotICAgIC8vIGNyZWF0ZSBuZXcgcmVjb3JkIHRyYWNrIGFuZCBwYXNzIHRvIHJlY29yZCB0aHJlYWQKLSAgICByZWNvcmRUcmFjayA9IG5ldyBNaXhlclRocmVhZDo6UmVjb3JkVHJhY2sobUhhcmR3YXJlTWl4ZXJUaHJlYWQsIGNsaWVudCwgc3RyZWFtVHlwZSwgc2FtcGxlUmF0ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9ybWF0LCBjaGFubmVsQ291bnQsIGZyYW1lQ291bnQsIGZsYWdzKTsKLSAgICBpZiAocmVjb3JkVHJhY2stPmdldENibGsoKSA9PSBOVUxMKSB7Ci0gICAgICAgIHJlY29yZFRyYWNrLmNsZWFyKCk7Ci0gICAgICAgIGxTdGF0dXMgPSBOT19NRU1PUlk7Ci0gICAgICAgIGdvdG8gRXhpdDsKLSAgICB9Ci0KLSAgICAvLyByZXR1cm4gdG8gaGFuZGxlIHRvIGNsaWVudAotICAgIHJlY29yZEhhbmRsZSA9IG5ldyBSZWNvcmRIYW5kbGUocmVjb3JkVHJhY2spOwotICAgIGxTdGF0dXMgPSBOT19FUlJPUjsKLQotRXhpdDoKLSAgICBpZiAoc3RhdHVzKSB7Ci0gICAgICAgICpzdGF0dXMgPSBsU3RhdHVzOwotICAgIH0KLSAgICByZXR1cm4gcmVjb3JkSGFuZGxlOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6OnN0YXJ0UmVjb3JkKE1peGVyVGhyZWFkOjpSZWNvcmRUcmFjayogcmVjb3JkVHJhY2spIHsKLSAgICBpZiAobUF1ZGlvUmVjb3JkVGhyZWFkICE9IDApIHsKLSAgICAgICAgcmV0dXJuIG1BdWRpb1JlY29yZFRocmVhZC0+c3RhcnQocmVjb3JkVHJhY2spOyAgICAgICAgCi0gICAgfQotICAgIHJldHVybiBOT19JTklUOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6c3RvcFJlY29yZChNaXhlclRocmVhZDo6UmVjb3JkVHJhY2sqIHJlY29yZFRyYWNrKSB7Ci0gICAgaWYgKG1BdWRpb1JlY29yZFRocmVhZCAhPSAwKSB7Ci0gICAgICAgIG1BdWRpb1JlY29yZFRocmVhZC0+c3RvcChyZWNvcmRUcmFjayk7Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUF1ZGlvRmxpbmdlcjo6UmVjb3JkSGFuZGxlOjpSZWNvcmRIYW5kbGUoY29uc3Qgc3A8QXVkaW9GbGluZ2VyOjpNaXhlclRocmVhZDo6UmVjb3JkVHJhY2s+JiByZWNvcmRUcmFjaykKLSAgICA6IEJuQXVkaW9SZWNvcmQoKSwKLSAgICBtUmVjb3JkVHJhY2socmVjb3JkVHJhY2spCi17Ci19Ci0KLUF1ZGlvRmxpbmdlcjo6UmVjb3JkSGFuZGxlOjp+UmVjb3JkSGFuZGxlKCkgewotICAgIHN0b3AoKTsKLX0KLQotc3RhdHVzX3QgQXVkaW9GbGluZ2VyOjpSZWNvcmRIYW5kbGU6OnN0YXJ0KCkgewotICAgIExPR1YoIlJlY29yZEhhbmRsZTo6c3RhcnQoKSIpOwotICAgIHJldHVybiBtUmVjb3JkVHJhY2stPnN0YXJ0KCk7Ci19Ci0KLXZvaWQgQXVkaW9GbGluZ2VyOjpSZWNvcmRIYW5kbGU6OnN0b3AoKSB7Ci0gICAgTE9HVigiUmVjb3JkSGFuZGxlOjpzdG9wKCkiKTsKLSAgICBtUmVjb3JkVHJhY2stPnN0b3AoKTsKLX0KLQotc3A8SU1lbW9yeT4gQXVkaW9GbGluZ2VyOjpSZWNvcmRIYW5kbGU6OmdldENibGsoKSBjb25zdCB7Ci0gICAgcmV0dXJuIG1SZWNvcmRUcmFjay0+Z2V0Q2JsaygpOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6OlJlY29yZEhhbmRsZTo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIHJldHVybiBCbkF1ZGlvUmVjb3JkOjpvblRyYW5zYWN0KGNvZGUsIGRhdGEsIHJlcGx5LCBmbGFncyk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotQXVkaW9GbGluZ2VyOjpBdWRpb1JlY29yZFRocmVhZDo6QXVkaW9SZWNvcmRUaHJlYWQoQXVkaW9IYXJkd2FyZUludGVyZmFjZSogYXVkaW9IYXJkd2FyZSkgOgotICAgIG1BdWRpb0hhcmR3YXJlKGF1ZGlvSGFyZHdhcmUpLAotICAgIG1BY3RpdmUoZmFsc2UpCi17Ci19Ci0KLUF1ZGlvRmxpbmdlcjo6QXVkaW9SZWNvcmRUaHJlYWQ6On5BdWRpb1JlY29yZFRocmVhZCgpCi17Ci19Ci0KLWJvb2wgQXVkaW9GbGluZ2VyOjpBdWRpb1JlY29yZFRocmVhZDo6dGhyZWFkTG9vcCgpCi17Ci0gICAgTE9HVigiQXVkaW9SZWNvcmRUaHJlYWQ6IHN0YXJ0IHJlY29yZCBsb29wIik7Ci0gICAgQXVkaW9CdWZmZXJQcm92aWRlcjo6QnVmZmVyIGJ1ZmZlcjsKLSAgICBpbnQgaW5CdWZmZXJTaXplID0gMDsKLSAgICBpbnQgaW5GcmFtZUNvdW50ID0gMDsKLSAgICBBdWRpb1N0cmVhbUluKiBpbnB1dCA9IDA7Ci0KLSAgICBtQWN0aXZlID0gMDsKLSAgICAKLSAgICAvLyBzdGFydCByZWNvcmRpbmcKLSAgICB3aGlsZSAoIWV4aXRQZW5kaW5nKCkpIHsKLSAgICAgICAgaWYgKCFtQWN0aXZlKSB7Ci0gICAgICAgICAgICBtTG9jay5sb2NrKCk7Ci0gICAgICAgICAgICBpZiAoIW1BY3RpdmUgJiYgIWV4aXRQZW5kaW5nKCkpIHsKLSAgICAgICAgICAgICAgICBMT0dWKCJBdWRpb1JlY29yZFRocmVhZDogbG9vcCBzdG9wcGluZyIpOwotICAgICAgICAgICAgICAgIGlmIChpbnB1dCkgewotICAgICAgICAgICAgICAgICAgICBkZWxldGUgaW5wdXQ7Ci0gICAgICAgICAgICAgICAgICAgIGlucHV0ID0gMDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbVJlY29yZFRyYWNrLmNsZWFyKCk7Ci0gICAgICAgICAgICAgICAgbVN0b3BwZWQuc2lnbmFsKCk7Ci0KLSAgICAgICAgICAgICAgICBtV2FpdFdvcmtDVi53YWl0KG1Mb2NrKTsKLSAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIExPR1YoIkF1ZGlvUmVjb3JkVGhyZWFkOiBsb29wIHN0YXJ0aW5nIik7Ci0gICAgICAgICAgICAgICAgaWYgKG1SZWNvcmRUcmFjayAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlucHV0ID0gbUF1ZGlvSGFyZHdhcmUtPm9wZW5JbnB1dFN0cmVhbShtUmVjb3JkVHJhY2stPmZvcm1hdCgpLCAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1SZWNvcmRUcmFjay0+Y2hhbm5lbENvdW50KCksIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbVJlY29yZFRyYWNrLT5zYW1wbGVSYXRlKCksIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJm1TdGFydFN0YXR1cywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChBdWRpb1N5c3RlbTo6YXVkaW9faW5fYWNvdXN0aWNzKShtUmVjb3JkVHJhY2stPm1GbGFncyA+PiAxNikpOwotICAgICAgICAgICAgICAgICAgICBpZiAoaW5wdXQgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgaW5CdWZmZXJTaXplID0gaW5wdXQtPmJ1ZmZlclNpemUoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGluRnJhbWVDb3VudCA9IGluQnVmZmVyU2l6ZS9pbnB1dC0+ZnJhbWVTaXplKCk7ICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBtU3RhcnRTdGF0dXMgPSBOT19JTklUOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAobVN0YXJ0U3RhdHVzICE9Tk9fRVJST1IpIHsKLSAgICAgICAgICAgICAgICAgICAgTE9HVygicmVjb3JkIHN0YXJ0IGZhaWxlZCwgc3RhdHVzICVkIiwgbVN0YXJ0U3RhdHVzKTsKLSAgICAgICAgICAgICAgICAgICAgbUFjdGl2ZSA9IGZhbHNlOwotICAgICAgICAgICAgICAgICAgICBtUmVjb3JkVHJhY2suY2xlYXIoKTsgICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBtV2FpdFdvcmtDVi5zaWduYWwoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1Mb2NrLnVubG9jaygpOwotICAgICAgICB9IGVsc2UgaWYgKG1SZWNvcmRUcmFjayAhPSAwKSB7Ci0KLSAgICAgICAgICAgIGJ1ZmZlci5mcmFtZUNvdW50ID0gaW5GcmFtZUNvdW50OwotICAgICAgICAgICAgaWYgKExJS0VMWShtUmVjb3JkVHJhY2stPmdldE5leHRCdWZmZXIoJmJ1ZmZlcikgPT0gTk9fRVJST1IpKSB7Ci0gICAgICAgICAgICAgICAgTE9HVigiQXVkaW9SZWNvcmRUaHJlYWQgcmVhZDogJWQgZnJhbWVzIiwgYnVmZmVyLmZyYW1lQ291bnQpOwotICAgICAgICAgICAgICAgIHNzaXplX3QgYnl0ZXNSZWFkID0gaW5wdXQtPnJlYWQoYnVmZmVyLnJhdywgaW5CdWZmZXJTaXplKTsKLSAgICAgICAgICAgICAgICBpZiAoYnl0ZXNSZWFkIDwgMCkgewotICAgICAgICAgICAgICAgICAgICBMT0dFKCJFcnJvciByZWFkaW5nIGF1ZGlvIGlucHV0Iik7Ci0gICAgICAgICAgICAgICAgICAgIHNsZWVwKDEpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBtUmVjb3JkVHJhY2stPnJlbGVhc2VCdWZmZXIoJmJ1ZmZlcik7Ci0gICAgICAgICAgICAgICAgbVJlY29yZFRyYWNrLT5vdmVyZmxvdygpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBjbGllbnQgaXNuJ3QgcmV0cmlldmluZyBidWZmZXJzIGZhc3QgZW5vdWdoCi0gICAgICAgICAgICBlbHNlIHsKLSAgICAgICAgICAgICAgICBpZiAoIW1SZWNvcmRUcmFjay0+c2V0T3ZlcmZsb3coKSkKLSAgICAgICAgICAgICAgICAgICAgTE9HVygiQXVkaW9SZWNvcmRUaHJlYWQ6IGJ1ZmZlciBvdmVyZmxvdyIpOwotICAgICAgICAgICAgICAgIC8vIFJlbGVhc2UgdGhlIHByb2Nlc3NvciBmb3IgYSB3aGlsZSBiZWZvcmUgYXNraW5nIGZvciBhIG5ldyBidWZmZXIuCi0gICAgICAgICAgICAgICAgLy8gVGhpcyB3aWxsIGdpdmUgdGhlIGFwcGxpY2F0aW9uIG1vcmUgY2hhbmNlIHRvIHJlYWQgZnJvbSB0aGUgYnVmZmVyIGFuZAotICAgICAgICAgICAgICAgIC8vIGNsZWFyIHRoZSBvdmVyZmxvdy4KLSAgICAgICAgICAgICAgICB1c2xlZXAoNTAwMCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLQotICAgIGlmIChpbnB1dCkgewotICAgICAgICBkZWxldGUgaW5wdXQ7Ci0gICAgfQotICAgIG1SZWNvcmRUcmFjay5jbGVhcigpOwotICAgIAotICAgIHJldHVybiBmYWxzZTsKLX0KLQotc3RhdHVzX3QgQXVkaW9GbGluZ2VyOjpBdWRpb1JlY29yZFRocmVhZDo6c3RhcnQoTWl4ZXJUaHJlYWQ6OlJlY29yZFRyYWNrKiByZWNvcmRUcmFjaykKLXsKLSAgICBMT0dWKCJBdWRpb1JlY29yZFRocmVhZDo6c3RhcnQiKTsKLSAgICBBdXRvTXV0ZXggbG9jaygmbUxvY2spOwotICAgIG1BY3RpdmUgPSB0cnVlOwotICAgIC8vIElmIHN0YXJ0aW5nIHRoZSBhY3RpdmUgdHJhY2ssIGp1c3QgcmVzZXQgbUFjdGl2ZSBpbiBjYXNlIGEgc3RvcAotICAgIC8vIHdhcyBwZW5kaW5nIGFuZCBleGl0Ci0gICAgaWYgKHJlY29yZFRyYWNrID09IG1SZWNvcmRUcmFjay5nZXQoKSkgcmV0dXJuIE5PX0VSUk9SOwotCi0gICAgaWYgKG1SZWNvcmRUcmFjayAhPSAwKSByZXR1cm4gLUVCVVNZOwotCi0gICAgbVJlY29yZFRyYWNrID0gcmVjb3JkVHJhY2s7Ci0KLSAgICAvLyBzaWduYWwgdGhyZWFkIHRvIHN0YXJ0Ci0gICAgTE9HVigiU2lnbmFsIHJlY29yZCB0aHJlYWQiKTsKLSAgICBtV2FpdFdvcmtDVi5zaWduYWwoKTsKLSAgICBtV2FpdFdvcmtDVi53YWl0KG1Mb2NrKTsKLSAgICBMT0dWKCJSZWNvcmQgc3RhcnRlZCwgc3RhdHVzICVkIiwgbVN0YXJ0U3RhdHVzKTsKLSAgICByZXR1cm4gbVN0YXJ0U3RhdHVzOwotfQotCi12b2lkIEF1ZGlvRmxpbmdlcjo6QXVkaW9SZWNvcmRUaHJlYWQ6OnN0b3AoTWl4ZXJUaHJlYWQ6OlJlY29yZFRyYWNrKiByZWNvcmRUcmFjaykgewotICAgIExPR1YoIkF1ZGlvUmVjb3JkVGhyZWFkOjpzdG9wIik7Ci0gICAgQXV0b011dGV4IGxvY2soJm1Mb2NrKTsKLSAgICBpZiAobUFjdGl2ZSAmJiAocmVjb3JkVHJhY2sgPT0gbVJlY29yZFRyYWNrLmdldCgpKSkgewotICAgICAgICBtQWN0aXZlID0gZmFsc2U7Ci0gICAgICAgIG1TdG9wcGVkLndhaXQobUxvY2spOwotICAgIH0KLX0KLQotdm9pZCBBdWRpb0ZsaW5nZXI6OkF1ZGlvUmVjb3JkVGhyZWFkOjpleGl0KCkKLXsKLSAgICBMT0dWKCJBdWRpb1JlY29yZFRocmVhZDo6ZXhpdCIpOwotICAgIHsKLSAgICAgICAgQXV0b011dGV4IGxvY2soJm1Mb2NrKTsKLSAgICAgICAgcmVxdWVzdEV4aXQoKTsKLSAgICAgICAgbVdhaXRXb3JrQ1Yuc2lnbmFsKCk7Ci0gICAgfQotICAgIHJlcXVlc3RFeGl0QW5kV2FpdCgpOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6OkF1ZGlvUmVjb3JkVGhyZWFkOjpkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBjb25zdCBzaXplX3QgU0laRSA9IDI1NjsKLSAgICBjaGFyIGJ1ZmZlcltTSVpFXTsKLSAgICBTdHJpbmc4IHJlc3VsdDsKLSAgICBwaWRfdCBwaWQgPSAwOwotCi0gICAgaWYgKG1SZWNvcmRUcmFjayAhPSAwICYmIG1SZWNvcmRUcmFjay0+bUNsaWVudCAhPSAwKSB7Ci0gICAgICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIlJlY29yZCBjbGllbnQgcGlkOiAlZFxuIiwgbVJlY29yZFRyYWNrLT5tQ2xpZW50LT5waWQoKSk7Ci0gICAgICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICB9IGVsc2UgewotICAgICAgICByZXN1bHQuYXBwZW5kKCJObyByZWNvcmQgY2xpZW50XG4iKTsKLSAgICB9Ci0gICAgd3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBBdWRpb0ZsaW5nZXI6Om9uVHJhbnNhY3QoCi0gICAgICAgIHVpbnQzMl90IGNvZGUsIGNvbnN0IFBhcmNlbCYgZGF0YSwgUGFyY2VsKiByZXBseSwgdWludDMyX3QgZmxhZ3MpCi17Ci0gICAgcmV0dXJuIEJuQXVkaW9GbGluZ2VyOjpvblRyYW5zYWN0KGNvZGUsIGRhdGEsIHJlcGx5LCBmbGFncyk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLXZvaWQgQXVkaW9GbGluZ2VyOjppbnN0YW50aWF0ZSgpIHsKLSAgICBkZWZhdWx0U2VydmljZU1hbmFnZXIoKS0+YWRkU2VydmljZSgKLSAgICAgICAgICAgIFN0cmluZzE2KCJtZWRpYS5hdWRpb19mbGluZ2VyIiksIG5ldyBBdWRpb0ZsaW5nZXIoKSk7Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0ZsaW5nZXIuaCBiL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvRmxpbmdlci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkZmJiMWU5Li4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvRmxpbmdlci5oCisrKyAvZGV2L251bGwKQEAgLTEsNjQwICswLDAgQEAKLS8qIC8vZGV2aWNlL2luY2x1ZGUvc2VydmVyL0F1ZGlvRmxpbmdlci9BdWRpb0ZsaW5nZXIuaAotKioKLSoqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jaWZuZGVmIEFORFJPSURfQVVESU9fRkxJTkdFUl9ICi0jZGVmaW5lIEFORFJPSURfQVVESU9fRkxJTkdFUl9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPG1lZGlhL0lBdWRpb0ZsaW5nZXIuaD4KLSNpbmNsdWRlIDxtZWRpYS9JQXVkaW9GbGluZ2VyQ2xpZW50Lmg+Ci0jaW5jbHVkZSA8bWVkaWEvSUF1ZGlvVHJhY2suaD4KLSNpbmNsdWRlIDxtZWRpYS9JQXVkaW9SZWNvcmQuaD4KLSNpbmNsdWRlIDxtZWRpYS9BdWRpb1RyYWNrLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5RGVhbGVyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvS2V5ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9Tb3J0ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9WZWN0b3IuaD4KLQotI2luY2x1ZGUgPGhhcmR3YXJlX2xlZ2FjeS9BdWRpb0hhcmR3YXJlSW50ZXJmYWNlLmg+Ci0KLSNpbmNsdWRlICJBdWRpb0J1ZmZlclByb3ZpZGVyLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgYXVkaW9fdHJhY2tfY2Jsa190OwotY2xhc3MgQXVkaW9NaXhlcjsKLWNsYXNzIEF1ZGlvQnVmZmVyOwotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2RlZmluZSBMSUtFTFkoIGV4cCApICAgICAgIChfX2J1aWx0aW5fZXhwZWN0KCAoZXhwKSAhPSAwLCB0cnVlICApKQotI2RlZmluZSBVTkxJS0VMWSggZXhwICkgICAgIChfX2J1aWx0aW5fZXhwZWN0KCAoZXhwKSAhPSAwLCBmYWxzZSApKQotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RhdGljIGNvbnN0IG5zZWNzX3Qga1N0YW5kYnlUaW1lSW5Oc2VjcyA9IHNlY29uZHMoMyk7Ci0KLWNsYXNzIEF1ZGlvRmxpbmdlciA6IHB1YmxpYyBCbkF1ZGlvRmxpbmdlciwgcHVibGljIElCaW5kZXI6OkRlYXRoUmVjaXBpZW50IAotewotcHVibGljOgotICAgIHN0YXRpYyB2b2lkIGluc3RhbnRpYXRlKCk7Ci0KLSAgICB2aXJ0dWFsICAgICBzdGF0dXNfdCAgICBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0KLSAgICAvLyBJQXVkaW9GbGluZ2VyIGludGVyZmFjZQotICAgIHZpcnR1YWwgc3A8SUF1ZGlvVHJhY2s+IGNyZWF0ZVRyYWNrKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwaWRfdCBwaWQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBzdHJlYW1UeXBlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZnJhbWVDb3VudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHNwPElNZW1vcnk+JiBzaGFyZWRCdWZmZXIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXR1c190ICpzdGF0dXMpOwotCi0gICAgdmlydHVhbCAgICAgdWludDMyX3QgICAgc2FtcGxlUmF0ZShpbnQgb3V0cHV0KSBjb25zdDsKLSAgICB2aXJ0dWFsICAgICBpbnQgICAgICAgICBjaGFubmVsQ291bnQoaW50IG91dHB1dCkgY29uc3Q7Ci0gICAgdmlydHVhbCAgICAgaW50ICAgICAgICAgZm9ybWF0KGludCBvdXRwdXQpIGNvbnN0OwotICAgIHZpcnR1YWwgICAgIHNpemVfdCAgICAgIGZyYW1lQ291bnQoaW50IG91dHB1dCkgY29uc3Q7Ci0gICAgdmlydHVhbCAgICAgdWludDMyX3QgICAgbGF0ZW5jeShpbnQgb3V0cHV0KSBjb25zdDsKLQotICAgIHZpcnR1YWwgICAgIHN0YXR1c190ICAgIHNldE1hc3RlclZvbHVtZShmbG9hdCB2YWx1ZSk7Ci0gICAgdmlydHVhbCAgICAgc3RhdHVzX3QgICAgc2V0TWFzdGVyTXV0ZShib29sIG11dGVkKTsKLQotICAgIHZpcnR1YWwgICAgIGZsb2F0ICAgICAgIG1hc3RlclZvbHVtZSgpIGNvbnN0OwotICAgIHZpcnR1YWwgICAgIGJvb2wgICAgICAgIG1hc3Rlck11dGUoKSBjb25zdDsKLQotICAgIHZpcnR1YWwgICAgIHN0YXR1c190ICAgIHNldFN0cmVhbVZvbHVtZShpbnQgc3RyZWFtLCBmbG9hdCB2YWx1ZSk7Ci0gICAgdmlydHVhbCAgICAgc3RhdHVzX3QgICAgc2V0U3RyZWFtTXV0ZShpbnQgc3RyZWFtLCBib29sIG11dGVkKTsKLQotICAgIHZpcnR1YWwgICAgIGZsb2F0ICAgICAgIHN0cmVhbVZvbHVtZShpbnQgc3RyZWFtKSBjb25zdDsKLSAgICB2aXJ0dWFsICAgICBib29sICAgICAgICBzdHJlYW1NdXRlKGludCBzdHJlYW0pIGNvbnN0OwotCi0gICAgdmlydHVhbCAgICAgc3RhdHVzX3QgICAgc2V0Um91dGluZyhpbnQgbW9kZSwgdWludDMyX3Qgcm91dGVzLCB1aW50MzJfdCBtYXNrKTsKLSAgICB2aXJ0dWFsICAgICB1aW50MzJfdCAgICBnZXRSb3V0aW5nKGludCBtb2RlKSBjb25zdDsKLQotICAgIHZpcnR1YWwgICAgIHN0YXR1c190ICAgIHNldE1vZGUoaW50IG1vZGUpOwotICAgIHZpcnR1YWwgICAgIGludCAgICAgICAgIGdldE1vZGUoKSBjb25zdDsKLQotICAgIHZpcnR1YWwgICAgIHN0YXR1c190ICAgIHNldE1pY011dGUoYm9vbCBzdGF0ZSk7Ci0gICAgdmlydHVhbCAgICAgYm9vbCAgICAgICAgZ2V0TWljTXV0ZSgpIGNvbnN0OwotCi0gICAgdmlydHVhbCAgICAgYm9vbCAgICAgICAgaXNNdXNpY0FjdGl2ZSgpIGNvbnN0OwotCi0gICAgdmlydHVhbCAgICAgYm9vbCAgICAgICAgaXNBMmRwRW5hYmxlZCgpIGNvbnN0OwotCi0gICAgdmlydHVhbCAgICAgc3RhdHVzX3QgICAgc2V0UGFyYW1ldGVyKGNvbnN0IGNoYXIqIGtleSwgY29uc3QgY2hhciogdmFsdWUpOwotCi0gICAgdmlydHVhbCAgICAgdm9pZCAgICAgICAgcmVnaXN0ZXJDbGllbnQoY29uc3Qgc3A8SUF1ZGlvRmxpbmdlckNsaWVudD4mIGNsaWVudCk7Ci0gICAgCi0gICAgdmlydHVhbCAgICAgc2l6ZV90ICAgICAgZ2V0SW5wdXRCdWZmZXJTaXplKHVpbnQzMl90IHNhbXBsZVJhdGUsIGludCBmb3JtYXQsIGludCBjaGFubmVsQ291bnQpOwotICAgIAotICAgIHZpcnR1YWwgICAgIHZvaWQgICAgICAgIHdha2VVcCgpOwotICAgIAotICAgIC8vIElCaW5kZXI6OkRlYXRoUmVjaXBpZW50Ci0gICAgdmlydHVhbCAgICAgdm9pZCAgICAgICAgYmluZGVyRGllZChjb25zdCB3cDxJQmluZGVyPiYgd2hvKTsKLQotICAgIGVudW0gaGFyZHdhcmVfY2FsbF9zdGF0ZSB7Ci0gICAgICAgIEFVRElPX0hXX0lETEUgPSAwLAotICAgICAgICBBVURJT19IV19JTklULAotICAgICAgICBBVURJT19IV19PVVRQVVRfT1BFTiwKLSAgICAgICAgQVVESU9fSFdfT1VUUFVUX0NMT1NFLAotICAgICAgICBBVURJT19IV19JTlBVVF9PUEVOLAotICAgICAgICBBVURJT19IV19JTlBVVF9DTE9TRSwKLSAgICAgICAgQVVESU9fSFdfU1RBTkRCWSwKLSAgICAgICAgQVVESU9fSFdfU0VUX01BU1RFUl9WT0xVTUUsCi0gICAgICAgIEFVRElPX0hXX0dFVF9ST1VUSU5HLAotICAgICAgICBBVURJT19IV19TRVRfUk9VVElORywKLSAgICAgICAgQVVESU9fSFdfR0VUX01PREUsCi0gICAgICAgIEFVRElPX0hXX1NFVF9NT0RFLAotICAgICAgICBBVURJT19IV19HRVRfTUlDX01VVEUsCi0gICAgICAgIEFVRElPX0hXX1NFVF9NSUNfTVVURSwKLSAgICAgICAgQVVESU9fU0VUX1ZPSUNFX1ZPTFVNRSwKLSAgICAgICAgQVVESU9fU0VUX1BBUkFNRVRFUiwKLSAgICB9OwotCi0gICAgLy8gcmVjb3JkIGludGVyZmFjZQotICAgIHZpcnR1YWwgc3A8SUF1ZGlvUmVjb3JkPiBvcGVuUmVjb3JkKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwaWRfdCBwaWQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBzdHJlYW1UeXBlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZnJhbWVDb3VudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXR1c190ICpzdGF0dXMpOwotCi0gICAgdmlydHVhbCAgICAgc3RhdHVzX3QgICAgb25UcmFuc2FjdCgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgY29kZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgUGFyY2VsJiBkYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJjZWwqIHJlcGx5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncyk7Ci0KLXByaXZhdGU6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgQXVkaW9GbGluZ2VyKCk7Ci0gICAgdmlydHVhbCAgICAgICAgICAgICAgICAgfkF1ZGlvRmxpbmdlcigpOwotICAgIAotICAgIHZvaWQgICAgICAgICAgICAgICAgICAgIHNldE91dHB1dChpbnQgb3V0cHV0VHlwZSk7Ci0gICAgdm9pZCAgICAgICAgICAgICAgICAgICAgZG9TZXRPdXRwdXQoaW50IG91dHB1dFR5cGUpOwotCi0jaWZkZWYgV0lUSF9BMkRQCi0gICAgdm9pZCAgICAgICAgICAgICAgICAgICAgc2V0QTJkcEVuYWJsZWQoYm9vbCBlbmFibGUpOwotI2VuZGlmCi0gICAgc3RhdGljIGJvb2wgICAgICAgICAgICAgc3RyZWFtRm9yY2VkVG9TcGVha2VyKGludCBzdHJlYW1UeXBlKTsKLSAgICAKLSAgICAvLyBNYW5hZ2VtZW50IG9mIGZvcmNlZCByb3V0ZSB0byBzcGVha2VyIGZvciBjZXJ0YWluIHRyYWNrIHR5cGVzLgotICAgIGVudW0gZm9yY2Vfc3BlYWtlcl9jb21tYW5kIHsKLSAgICAgICAgQUNUSVZFX1RSQUNLX0FEREVEID0gMCwKLSAgICAgICAgQUNUSVZFX1RSQUNLX1JFTU9WRUQsCi0gICAgICAgIENIRUNLX1JPVVRFX1JFU1RPUkVfVElNRSwKLSAgICAgICAgRk9SQ0VfUk9VVEVfUkVTVE9SRQotICAgIH07Ci0gICAgdm9pZCAgICAgICAgICAgICAgICAgICAgaGFuZGxlRm9yY2VkU3BlYWtlclJvdXRlKGludCBjb21tYW5kKTsKLQotICAgIC8vIEludGVybmFsIGR1bXAgdXRpbGl0ZXMuCi0gICAgc3RhdHVzX3QgZHVtcFBlcm1pc3Npb25EZW5pYWwoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKTsKLSAgICBzdGF0dXNfdCBkdW1wQ2xpZW50cyhpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpOwotICAgIHN0YXR1c190IGR1bXBJbnRlcm5hbHMoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKTsKLQotICAgIC8vIC0tLSBDbGllbnQgLS0tCi0gICAgY2xhc3MgQ2xpZW50IDogcHVibGljIFJlZkJhc2UgewotICAgIHB1YmxpYzoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBDbGllbnQoY29uc3Qgc3A8QXVkaW9GbGluZ2VyPiYgYXVkaW9GbGluZ2VyLCBwaWRfdCBwaWQpOwotICAgICAgICB2aXJ0dWFsICAgICAgICAgICAgIH5DbGllbnQoKTsKLSAgICAgICAgY29uc3Qgc3A8TWVtb3J5RGVhbGVyPiYgICAgIGhlYXAoKSBjb25zdDsKLSAgICAgICAgcGlkX3QgICAgICAgICAgICAgICBwaWQoKSBjb25zdCB7IHJldHVybiBtUGlkOyB9Ci0gICAgcHJpdmF0ZToKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBDbGllbnQoY29uc3QgQ2xpZW50Jik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgQ2xpZW50JiBvcGVyYXRvciA9IChjb25zdCBDbGllbnQmKTsKLSAgICAgICAgc3A8QXVkaW9GbGluZ2VyPiAgICBtQXVkaW9GbGluZ2VyOwotICAgICAgICBzcDxNZW1vcnlEZWFsZXI+ICAgIG1NZW1vcnlEZWFsZXI7Ci0gICAgICAgIHBpZF90ICAgICAgICAgICAgICAgbVBpZDsKLSAgICB9OwotCi0KLSAgICBjbGFzcyBUcmFja0hhbmRsZTsKLSAgICBjbGFzcyBSZWNvcmRIYW5kbGU7Ci0gICAgY2xhc3MgQXVkaW9SZWNvcmRUaHJlYWQ7Ci0KLSAgICAKLSAgICAvLyAtLS0gTWl4ZXJUaHJlYWQgLS0tCi0gICAgY2xhc3MgTWl4ZXJUaHJlYWQgOiBwdWJsaWMgVGhyZWFkIHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIAotICAgICAgICAvLyAtLS0gVHJhY2sgLS0tCi0KLSAgICAgICAgLy8gYmFzZSBmb3IgcmVjb3JkIGFuZCBwbGF5YmFjawotICAgICAgICBjbGFzcyBUcmFja0Jhc2UgOiBwdWJsaWMgQXVkaW9CdWZmZXJQcm92aWRlciwgcHVibGljIFJlZkJhc2UgewotCi0gICAgICAgIHB1YmxpYzoKLSAgICAgICAgICAgIGVudW0gdHJhY2tfc3RhdGUgewotICAgICAgICAgICAgICAgIElETEUsCi0gICAgICAgICAgICAgICAgVEVSTUlOQVRFRCwKLSAgICAgICAgICAgICAgICBTVE9QUEVELAotICAgICAgICAgICAgICAgIFJFU1VNSU5HLAotICAgICAgICAgICAgICAgIEFDVElWRSwKLSAgICAgICAgICAgICAgICBQQVVTSU5HLAotICAgICAgICAgICAgICAgIFBBVVNFRAotICAgICAgICAgICAgfTsKLQotICAgICAgICAgICAgZW51bSB0cmFja19mbGFncyB7Ci0gICAgICAgICAgICAgICAgU1RFUFNFUlZFUl9GQUlMRUQgPSAweDAxLCAvLyAgU3RlcFNlcnZlciBjb3VsZCBub3QgYWNxdWlyZSBjYmxrLT5sb2NrIG11dGV4Ci0gICAgICAgICAgICAgICAgU1lTVEVNX0ZMQUdTX01BU0sgPSAweDAwMDBmZmZmVUwsCi0KLSAgICAgICAgICAgICAgICBBVURJT19JTl9BR0NfRU5BQkxFID0gQXVkaW9TeXN0ZW06OkFHQ19FTkFCTEUgPDwgMTYsCi0gICAgICAgICAgICAgICAgQVVESU9fSU5fTlNfRU5BQkxFICA9IEF1ZGlvU3lzdGVtOjpOU19FTkFCTEUgPDwgMTYsCi0gICAgICAgICAgICAgICAgQVVESU9fSU5fSUlSX0VOQUJMRSA9IEF1ZGlvU3lzdGVtOjpUWF9JSVJfRU5BQkxFIDw8IDE2Ci0gICAgICAgICAgICB9OwotCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRyYWNrQmFzZShjb25zdCBzcDxNaXhlclRocmVhZD4mIG1peGVyVGhyZWFkLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHNwPENsaWVudD4mIGNsaWVudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgc3RyZWFtVHlwZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBmb3JtYXQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGNoYW5uZWxDb3VudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZnJhbWVDb3VudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBmbGFncywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBzcDxJTWVtb3J5PiYgc2hhcmVkQnVmZmVyKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgflRyYWNrQmFzZSgpOwotCi0gICAgICAgICAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHN0YXJ0KCkgPSAwOwotICAgICAgICAgICAgdmlydHVhbCB2b2lkICAgICAgICBzdG9wKCkgPSAwOwotICAgICAgICAgICAgICAgICAgICBzcDxJTWVtb3J5PiBnZXRDYmxrKCkgY29uc3Q7Ci0KLSAgICAgICAgcHJvdGVjdGVkOgotICAgICAgICAgICAgZnJpZW5kIGNsYXNzIE1peGVyVGhyZWFkOwotICAgICAgICAgICAgZnJpZW5kIGNsYXNzIFJlY29yZEhhbmRsZTsKLSAgICAgICAgICAgIGZyaWVuZCBjbGFzcyBBdWRpb1JlY29yZFRocmVhZDsKLQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBUcmFja0Jhc2UoY29uc3QgVHJhY2tCYXNlJik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRyYWNrQmFzZSYgb3BlcmF0b3IgPSAoY29uc3QgVHJhY2tCYXNlJik7Ci0KLSAgICAgICAgICAgIHZpcnR1YWwgc3RhdHVzX3QgZ2V0TmV4dEJ1ZmZlcihBdWRpb0J1ZmZlclByb3ZpZGVyOjpCdWZmZXIqIGJ1ZmZlcikgPSAwOwotICAgICAgICAgICAgdmlydHVhbCB2b2lkIHJlbGVhc2VCdWZmZXIoQXVkaW9CdWZmZXJQcm92aWRlcjo6QnVmZmVyKiBidWZmZXIpOwotCi0gICAgICAgICAgICBhdWRpb190cmFja19jYmxrX3QqIGNibGsoKSBjb25zdCB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG1DYmxrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpbnQgdHlwZSgpIGNvbnN0IHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbVN0cmVhbVR5cGU7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGludCBmb3JtYXQoKSBjb25zdCB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG1Gb3JtYXQ7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGludCBjaGFubmVsQ291bnQoKSBjb25zdCA7Ci0KLSAgICAgICAgICAgIGludCBzYW1wbGVSYXRlKCkgY29uc3Q7Ci0KLSAgICAgICAgICAgIHZvaWQqIGdldEJ1ZmZlcih1aW50MzJfdCBvZmZzZXQsIHVpbnQzMl90IGZyYW1lcykgY29uc3Q7Ci0KLSAgICAgICAgICAgIGludCBuYW1lKCkgY29uc3QgewotICAgICAgICAgICAgICAgIHJldHVybiBtTmFtZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgYm9vbCBpc1N0b3BwZWQoKSBjb25zdCB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG1TdGF0ZSA9PSBTVE9QUEVEOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBib29sIGlzVGVybWluYXRlZCgpIGNvbnN0IHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbVN0YXRlID09IFRFUk1JTkFURUQ7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGJvb2wgc3RlcCgpOwotICAgICAgICAgICAgdm9pZCByZXNldCgpOwotCi0gICAgICAgICAgICBzcDxNaXhlclRocmVhZD4gICAgIG1NaXhlclRocmVhZDsKLSAgICAgICAgICAgIHNwPENsaWVudD4gICAgICAgICAgbUNsaWVudDsKLSAgICAgICAgICAgIHNwPElNZW1vcnk+ICAgICAgICAgbUNibGtNZW1vcnk7Ci0gICAgICAgICAgICBhdWRpb190cmFja19jYmxrX3QqIG1DYmxrOwotICAgICAgICAgICAgaW50ICAgICAgICAgICAgICAgICBtU3RyZWFtVHlwZTsKLSAgICAgICAgICAgIHZvaWQqICAgICAgICAgICAgICAgbUJ1ZmZlcjsKLSAgICAgICAgICAgIHZvaWQqICAgICAgICAgICAgICAgbUJ1ZmZlckVuZDsKLSAgICAgICAgICAgIHVpbnQzMl90ICAgICAgICAgICAgbUZyYW1lQ291bnQ7Ci0gICAgICAgICAgICBpbnQgICAgICAgICAgICAgICAgIG1OYW1lOwotICAgICAgICAgICAgLy8gd2UgZG9uJ3QgcmVhbGx5IG5lZWQgYSBsb2NrIGZvciB0aGVzZQotICAgICAgICAgICAgaW50ICAgICAgICAgICAgICAgICBtU3RhdGU7Ci0gICAgICAgICAgICBpbnQgICAgICAgICAgICAgICAgIG1DbGllbnRUaWQ7Ci0gICAgICAgICAgICB1aW50OF90ICAgICAgICAgICAgIG1Gb3JtYXQ7Ci0gICAgICAgICAgICB1aW50MzJfdCAgICAgICAgICAgIG1GbGFnczsKLSAgICAgICAgfTsKLQotICAgICAgICAvLyBwbGF5YmFjayB0cmFjawotICAgICAgICBjbGFzcyBUcmFjayA6IHB1YmxpYyBUcmFja0Jhc2UgewotICAgICAgICBwdWJsaWM6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRyYWNrKCAgY29uc3Qgc3A8TWl4ZXJUaHJlYWQ+JiBtaXhlclRocmVhZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBzcDxDbGllbnQ+JiBjbGllbnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IHN0cmVhbVR5cGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBjaGFubmVsQ291bnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGZyYW1lQ291bnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3Qgc3A8SU1lbW9yeT4mIHNoYXJlZEJ1ZmZlcik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH5UcmFjaygpOwotCi0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIGR1bXAoY2hhciogYnVmZmVyLCBzaXplX3Qgc2l6ZSk7Ci0gICAgICAgICAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHN0YXJ0KCk7Ci0gICAgICAgICAgICB2aXJ0dWFsIHZvaWQgICAgICAgIHN0b3AoKTsKLSAgICAgICAgICAgICAgICAgICAgdm9pZCAgICAgICAgcGF1c2UoKTsKLQotICAgICAgICAgICAgICAgICAgICB2b2lkICAgICAgICBmbHVzaCgpOwotICAgICAgICAgICAgICAgICAgICB2b2lkICAgICAgICBkZXN0cm95KCk7Ci0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIG11dGUoYm9vbCk7Ci0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIHNldFZvbHVtZShmbG9hdCBsZWZ0LCBmbG9hdCByaWdodCk7Ci0KLSAgICAgICAgcHJvdGVjdGVkOgotICAgICAgICAgICAgZnJpZW5kIGNsYXNzIE1peGVyVGhyZWFkOwotICAgICAgICAgICAgZnJpZW5kIGNsYXNzIEF1ZGlvRmxpbmdlcjsKLSAgICAgICAgICAgIGZyaWVuZCBjbGFzcyBBdWRpb0ZsaW5nZXI6OlRyYWNrSGFuZGxlOwotCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRyYWNrKGNvbnN0IFRyYWNrJik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRyYWNrJiBvcGVyYXRvciA9IChjb25zdCBUcmFjayYpOwotCi0gICAgICAgICAgICB2aXJ0dWFsIHN0YXR1c190IGdldE5leHRCdWZmZXIoQXVkaW9CdWZmZXJQcm92aWRlcjo6QnVmZmVyKiBidWZmZXIpOwotCi0gICAgICAgICAgICBib29sIGlzTXV0ZWQoKSBjb25zdCB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIChtTXV0ZSB8fCBtTWl4ZXJUaHJlYWQtPm1TdHJlYW1UeXBlc1ttU3RyZWFtVHlwZV0ubXV0ZSk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGJvb2wgaXNQYXVzaW5nKCkgY29uc3QgewotICAgICAgICAgICAgICAgIHJldHVybiBtU3RhdGUgPT0gUEFVU0lORzsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgYm9vbCBpc1BhdXNlZCgpIGNvbnN0IHsKLSAgICAgICAgICAgICAgICByZXR1cm4gbVN0YXRlID09IFBBVVNFRDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgYm9vbCBpc1JlYWR5KCkgY29uc3Q7Ci0KLSAgICAgICAgICAgIHZvaWQgc2V0UGF1c2VkKCkgeyBtU3RhdGUgPSBQQVVTRUQ7IH0KLSAgICAgICAgICAgIHZvaWQgcmVzZXQoKTsKLQotICAgICAgICAgICAgLy8gd2UgZG9uJ3QgcmVhbGx5IG5lZWQgYSBsb2NrIGZvciB0aGVzZQotICAgICAgICAgICAgZmxvYXQgICAgICAgICAgICAgICBtVm9sdW1lWzJdOwotICAgICAgICAgICAgdm9sYXRpbGUgYm9vbCAgICAgICBtTXV0ZTsKLSAgICAgICAgICAgIC8vIEZJTExFRCBzdGF0ZSBpcyB1c2VkIGZvciBzdXBwcmVzc2luZyB2b2x1bWUgcmFtcCBhdCBiZWdpbiBvZiBwbGF5aW5nCi0gICAgICAgICAgICBlbnVtIHtGU19GSUxMSU5HLCBGU19GSUxMRUQsIEZTX0FDVElWRX07Ci0gICAgICAgICAgICBtdXRhYmxlIHVpbnQ4X3QgICAgIG1GaWxsaW5nVXBTdGF0dXM7Ci0gICAgICAgICAgICBpbnQ4X3QgICAgICAgICAgICAgIG1SZXRyeUNvdW50OwotICAgICAgICAgICAgc3A8SU1lbW9yeT4gICAgICAgICBtU2hhcmVkQnVmZmVyOwotICAgICAgICAgICAgYm9vbCAgICAgICAgICAgICAgICBtUmVzZXREb25lOwotICAgICAgICB9OyAgLy8gZW5kIG9mIFRyYWNrCi0KLSAgICAgICAgLy8gcmVjb3JkIHRyYWNrCi0gICAgICAgIGNsYXNzIFJlY29yZFRyYWNrIDogcHVibGljIFRyYWNrQmFzZSB7Ci0gICAgICAgIHB1YmxpYzoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUmVjb3JkVHJhY2soY29uc3Qgc3A8TWl4ZXJUaHJlYWQ+JiBtaXhlclRocmVhZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBzcDxDbGllbnQ+JiBjbGllbnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IHN0cmVhbVR5cGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBjaGFubmVsQ291bnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGZyYW1lQ291bnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB+UmVjb3JkVHJhY2soKTsKLQotICAgICAgICAgICAgdmlydHVhbCBzdGF0dXNfdCAgICBzdGFydCgpOwotICAgICAgICAgICAgdmlydHVhbCB2b2lkICAgICAgICBzdG9wKCk7Ci0KLSAgICAgICAgICAgICAgICAgICAgYm9vbCAgICAgICAgb3ZlcmZsb3coKSB7IGJvb2wgdG1wID0gbU92ZXJmbG93OyBtT3ZlcmZsb3cgPSBmYWxzZTsgcmV0dXJuIHRtcDsgfQotICAgICAgICAgICAgICAgICAgICBib29sICAgICAgICBzZXRPdmVyZmxvdygpIHsgYm9vbCB0bXAgPSBtT3ZlcmZsb3c7IG1PdmVyZmxvdyA9IHRydWU7IHJldHVybiB0bXA7IH0KLQotICAgICAgICBwcml2YXRlOgotICAgICAgICAgICAgZnJpZW5kIGNsYXNzIEF1ZGlvRmxpbmdlcjsKLSAgICAgICAgICAgIGZyaWVuZCBjbGFzcyBBdWRpb0ZsaW5nZXI6OlJlY29yZEhhbmRsZTsKLSAgICAgICAgICAgIGZyaWVuZCBjbGFzcyBBdWRpb0ZsaW5nZXI6OkF1ZGlvUmVjb3JkVGhyZWFkOwotICAgICAgICAgICAgZnJpZW5kIGNsYXNzIE1peGVyVGhyZWFkOwotCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFJlY29yZFRyYWNrKGNvbnN0IFRyYWNrJik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFJlY29yZFRyYWNrJiBvcGVyYXRvciA9IChjb25zdCBUcmFjayYpOwotCi0gICAgICAgICAgICB2aXJ0dWFsIHN0YXR1c190IGdldE5leHRCdWZmZXIoQXVkaW9CdWZmZXJQcm92aWRlcjo6QnVmZmVyKiBidWZmZXIpOwotCi0gICAgICAgICAgICBib29sICAgICAgICAgICAgICAgIG1PdmVyZmxvdzsKLSAgICAgICAgfTsKLQotICAgICAgICAvLyBwbGF5YmFjayB0cmFjawotICAgICAgICBjbGFzcyBPdXRwdXRUcmFjayA6IHB1YmxpYyBUcmFjayB7Ci0gICAgICAgIHB1YmxpYzoKLSAgICAgICAgICAgIAotICAgICAgICAgICAgY2xhc3MgQnVmZmVyOiBwdWJsaWMgQXVkaW9CdWZmZXJQcm92aWRlcjo6QnVmZmVyIHsKLSAgICAgICAgICAgIHB1YmxpYzoKLSAgICAgICAgICAgICAgICBpbnQxNl90ICptQnVmZmVyOwotICAgICAgICAgICAgfTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBPdXRwdXRUcmFjayggIGNvbnN0IHNwPE1peGVyVGhyZWFkPiYgbWl4ZXJUaHJlYWQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBjaGFubmVsQ291bnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGZyYW1lQ291bnQpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB+T3V0cHV0VHJhY2soKTsKLQotICAgICAgICAgICAgdmlydHVhbCBzdGF0dXNfdCAgICBzdGFydCgpOwotICAgICAgICAgICAgdmlydHVhbCB2b2lkICAgICAgICBzdG9wKCk7Ci0gICAgICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIHdyaXRlKGludDE2X3QqIGRhdGEsIHVpbnQzMl90IGZyYW1lcyk7Ci0gICAgICAgICAgICAgICAgICAgIGJvb2wgICAgICAgIGJ1ZmZlclF1ZXVlRW1wdHkoKSB7IHJldHVybiAobUJ1ZmZlclF1ZXVlLnNpemUoKSA9PSAwKSA/IHRydWUgOiBmYWxzZTsgfQotCi0gICAgICAgIHByaXZhdGU6Ci0KLSAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgb2J0YWluQnVmZmVyKEF1ZGlvQnVmZmVyUHJvdmlkZXI6OkJ1ZmZlciogYnVmZmVyKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgY2xlYXJCdWZmZXJRdWV1ZSgpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBzcDxNaXhlclRocmVhZD4gICAgICAgICAgICAgbU91dHB1dE1peGVyVGhyZWFkOwotICAgICAgICAgICAgVmVjdG9yIDwgQnVmZmVyKiA+ICAgICAgICAgIG1CdWZmZXJRdWV1ZTsKLSAgICAgICAgICAgIEF1ZGlvQnVmZmVyUHJvdmlkZXI6OkJ1ZmZlciBtT3V0QnVmZmVyOwotICAgICAgICAgICAgdWludDMyX3QgICAgICAgICAgICAgICAgICAgIG1GcmFtZXNXcml0dGVuOwotICAgICAgICAgICAgCi0gICAgICAgICB9OyAgLy8gZW5kIG9mIE91dHB1dFRyYWNrCi0KLSAgICAgICAgTWl4ZXJUaHJlYWQgKGNvbnN0IHNwPEF1ZGlvRmxpbmdlcj4mIGF1ZGlvRmxpbmdlciwgQXVkaW9TdHJlYW1PdXQqIG91dHB1dCwgaW50IG91dHB1dFR5cGUpOwotICAgICAgICB2aXJ0dWFsICAgICAgICAgICAgIH5NaXhlclRocmVhZCgpOwotCi0gICAgICAgIHZpcnR1YWwgICAgIHN0YXR1c190ICAgIGR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKTsKLQotICAgICAgICAvLyBUaHJlYWQgdmlydHVhbHMKLSAgICAgICAgdmlydHVhbCAgICAgYm9vbCAgICAgICAgdGhyZWFkTG9vcCgpOwotICAgICAgICB2aXJ0dWFsICAgICBzdGF0dXNfdCAgICByZWFkeVRvUnVuKCk7Ci0gICAgICAgIHZpcnR1YWwgICAgIHZvaWQgICAgICAgIG9uRmlyc3RSZWYoKTsKLQotICAgICAgICB2aXJ0dWFsICAgICB1aW50MzJfdCAgICBzYW1wbGVSYXRlKCkgY29uc3Q7Ci0gICAgICAgIHZpcnR1YWwgICAgIGludCAgICAgICAgIGNoYW5uZWxDb3VudCgpIGNvbnN0OwotICAgICAgICB2aXJ0dWFsICAgICBpbnQgICAgICAgICBmb3JtYXQoKSBjb25zdDsKLSAgICAgICAgdmlydHVhbCAgICAgc2l6ZV90ICAgICAgZnJhbWVDb3VudCgpIGNvbnN0OwotICAgICAgICB2aXJ0dWFsICAgICB1aW50MzJfdCAgICBsYXRlbmN5KCkgY29uc3Q7Ci0KLSAgICAgICAgdmlydHVhbCAgICAgc3RhdHVzX3QgICAgc2V0TWFzdGVyVm9sdW1lKGZsb2F0IHZhbHVlKTsKLSAgICAgICAgdmlydHVhbCAgICAgc3RhdHVzX3QgICAgc2V0TWFzdGVyTXV0ZShib29sIG11dGVkKTsKLQotICAgICAgICB2aXJ0dWFsICAgICBmbG9hdCAgICAgICBtYXN0ZXJWb2x1bWUoKSBjb25zdDsKLSAgICAgICAgdmlydHVhbCAgICAgYm9vbCAgICAgICAgbWFzdGVyTXV0ZSgpIGNvbnN0OwotCi0gICAgICAgIHZpcnR1YWwgICAgIHN0YXR1c190ICAgIHNldFN0cmVhbVZvbHVtZShpbnQgc3RyZWFtLCBmbG9hdCB2YWx1ZSk7Ci0gICAgICAgIHZpcnR1YWwgICAgIHN0YXR1c190ICAgIHNldFN0cmVhbU11dGUoaW50IHN0cmVhbSwgYm9vbCBtdXRlZCk7Ci0KLSAgICAgICAgdmlydHVhbCAgICAgZmxvYXQgICAgICAgc3RyZWFtVm9sdW1lKGludCBzdHJlYW0pIGNvbnN0OwotICAgICAgICB2aXJ0dWFsICAgICBib29sICAgICAgICBzdHJlYW1NdXRlKGludCBzdHJlYW0pIGNvbnN0OwotCi0gICAgICAgICAgICAgICAgICAgIGJvb2wgICAgICAgIGlzTXVzaWNBY3RpdmUoKSBjb25zdDsKLSAgICAgICAgCi0gICAgICAgICAgICAgICAgICAgIAotICAgICAgICBzcDxUcmFjaz4gY3JlYXRlVHJhY2soCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBzcDxBdWRpb0ZsaW5nZXI6OkNsaWVudD4mIGNsaWVudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBzdHJlYW1UeXBlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBmb3JtYXQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGZyYW1lQ291bnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBzcDxJTWVtb3J5PiYgc2hhcmVkQnVmZmVyLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3RhdHVzX3QgKnN0YXR1cyk7Ci0KLSAgICAgICAgICAgICAgICAgICAgdm9pZCAgICAgICAgd2FrZVVwKCkgeyBtV2FpdFdvcmtDVi5icm9hZGNhc3QoKTsgfQotICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgdm9pZCAgICAgICAgZ2V0VHJhY2tzKFNvcnRlZFZlY3RvciA8IHNwPFRyYWNrPiA+JiB0cmFja3MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTb3J0ZWRWZWN0b3IgPCB3cDxUcmFjaz4gPiYgYWN0aXZlVHJhY2tzKTsKLSAgICAgICAgICAgICAgICAgICAgdm9pZCAgICAgICAgcHV0VHJhY2tzKFNvcnRlZFZlY3RvciA8IHNwPFRyYWNrPiA+JiB0cmFja3MsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTb3J0ZWRWZWN0b3IgPCB3cDxUcmFjaz4gPiYgYWN0aXZlVHJhY2tzKTsKLSAgICAgICAgICAgICAgICAgICAgdm9pZCAgICAgICAgc2V0T3VwdXRUcmFjayhPdXRwdXRUcmFjayAqdHJhY2spIHsgbU91dHB1dFRyYWNrID0gdHJhY2s7IH0KLSAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgIHN0cnVjdCAgc3RyZWFtX3R5cGVfdCB7Ci0gICAgICAgICAgICBzdHJlYW1fdHlwZV90KCkKLSAgICAgICAgICAgICAgICA6ICAgdm9sdW1lKDEuMGYpLAotICAgICAgICAgICAgICAgICAgICBtdXRlKGZhbHNlKQotICAgICAgICAgICAgewotICAgICAgICAgICAgfQotICAgICAgICAgICAgZmxvYXQgICAgICAgdm9sdW1lOwotICAgICAgICAgICAgYm9vbCAgICAgICAgbXV0ZTsKLSAgICAgICAgfTsKLQotICAgIHByaXZhdGU6Ci0KLQotICAgICAgICBmcmllbmQgY2xhc3MgQXVkaW9GbGluZ2VyOwotICAgICAgICBmcmllbmQgY2xhc3MgVHJhY2s7Ci0gICAgICAgIGZyaWVuZCBjbGFzcyBUcmFja0Jhc2U7Ci0gICAgICAgIGZyaWVuZCBjbGFzcyBSZWNvcmRUcmFjazsKLSAgICAgICAgCi0gICAgICAgIE1peGVyVGhyZWFkKGNvbnN0IENsaWVudCYpOwotICAgICAgICBNaXhlclRocmVhZCYgb3BlcmF0b3IgPSAoY29uc3QgTWl4ZXJUaHJlYWQmKTsKLSAgCi0gICAgICAgIHN0YXR1c190ICAgIGFkZFRyYWNrKGNvbnN0IHNwPFRyYWNrPiYgdHJhY2spOwotICAgICAgICB2b2lkICAgICAgICByZW1vdmVUcmFjayh3cDxUcmFjaz4gdHJhY2ssIGludCBuYW1lKTsKLSAgICAgICAgdm9pZCAgICAgICAgcmVtb3ZlX3RyYWNrX2wod3A8VHJhY2s+IHRyYWNrLCBpbnQgbmFtZSk7Ci0gICAgICAgIHZvaWQgICAgICAgIGRlc3Ryb3lUcmFjayhjb25zdCBzcDxUcmFjaz4mIHRyYWNrKTsKLSAgICAgICAgaW50ICAgICAgICAgZ2V0VHJhY2tOYW1lKCk7Ci0gICAgICAgIHZvaWQgICAgICAgIGRlbGV0ZVRyYWNrTmFtZShpbnQgbmFtZSk7Ci0gICAgICAgIHZvaWQgICAgICAgIGFkZEFjdGl2ZVRyYWNrKGNvbnN0IHdwPFRyYWNrPiYgdCk7Ci0gICAgICAgIHZvaWQgICAgICAgIHJlbW92ZUFjdGl2ZVRyYWNrKGNvbnN0IHdwPFRyYWNrPiYgdCk7Ci0gICAgICAgIHNpemVfdCAgICAgIGdldE91dHB1dEZyYW1lQ291bnQoKTsKLQotICAgICAgICBzdGF0dXNfdCAgICBkdW1wSW50ZXJuYWxzKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0gICAgICAgIHN0YXR1c190ICAgIGR1bXBUcmFja3MoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKTsKLSAgICAgICAgCi0gICAgICAgIHNwPEF1ZGlvRmxpbmdlcj4gICAgICAgICAgICAgICAgbUF1ZGlvRmxpbmdlcjsgICAgICAgCi0gICAgICAgIG11dGFibGUgICAgIE11dGV4ICAgICAgICAgICAgICAgbUxvY2s7Ci0gICAgICAgIG11dGFibGUgICAgIENvbmRpdGlvbiAgICAgICAgICAgbVdhaXRXb3JrQ1Y7Ci0gICAgICAgIFNvcnRlZFZlY3Rvcjwgd3A8VHJhY2s+ID4gICAgICAgbUFjdGl2ZVRyYWNrczsKLSAgICAgICAgU29ydGVkVmVjdG9yPCBzcDxUcmFjaz4gPiAgICAgICBtVHJhY2tzOwotICAgICAgICBzdHJlYW1fdHlwZV90ICAgICAgICAgICAgICAgICAgIG1TdHJlYW1UeXBlc1tBdWRpb1N5c3RlbTo6TlVNX1NUUkVBTV9UWVBFU107Ci0gICAgICAgIEF1ZGlvTWl4ZXIqICAgICAgICAgICAgICAgICAgICAgbUF1ZGlvTWl4ZXI7Ci0gICAgICAgIEF1ZGlvU3RyZWFtT3V0KiAgICAgICAgICAgICAgICAgbU91dHB1dDsKLSAgICAgICAgaW50ICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtT3V0cHV0VHlwZTsKLSAgICAgICAgdWludDMyX3QgICAgICAgICAgICAgICAgICAgICAgICBtU2FtcGxlUmF0ZTsKLSAgICAgICAgc2l6ZV90ICAgICAgICAgICAgICAgICAgICAgICAgICBtRnJhbWVDb3VudDsKLSAgICAgICAgaW50ICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtQ2hhbm5lbENvdW50OwotICAgICAgICBpbnQgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1Gb3JtYXQ7Ci0gICAgICAgIGludDE2X3QqICAgICAgICAgICAgICAgICAgICAgICAgbU1peEJ1ZmZlcjsKLSAgICAgICAgZmxvYXQgICAgICAgICAgICAgICAgICAgICAgICAgICBtTWFzdGVyVm9sdW1lOwotICAgICAgICBib29sICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1NYXN0ZXJNdXRlOwotICAgICAgICBuc2Vjc190ICAgICAgICAgICAgICAgICAgICAgICAgIG1MYXN0V3JpdGVUaW1lOwotICAgICAgICBpbnQgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1OdW1Xcml0ZXM7Ci0gICAgICAgIGludCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbU51bURlbGF5ZWRXcml0ZXM7Ci0gICAgICAgIGJvb2wgICAgICAgICAgICAgICAgICAgICAgICAgICAgbVN0YW5kYnk7Ci0gICAgICAgIGJvb2wgICAgICAgICAgICAgICAgICAgICAgICAgICAgbUluV3JpdGU7Ci0gICAgICAgIHNwIDxPdXRwdXRUcmFjaz4gICAgICAgICAgICAgICAgbU91dHB1dFRyYWNrOwotICAgIH07Ci0KLSAgICAKLSAgICBmcmllbmQgY2xhc3MgQXVkaW9CdWZmZXI7Ci0KLSAgICBjbGFzcyBUcmFja0hhbmRsZSA6IHB1YmxpYyBhbmRyb2lkOjpCbkF1ZGlvVHJhY2sgewotICAgIHB1YmxpYzoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBUcmFja0hhbmRsZShjb25zdCBzcDxNaXhlclRocmVhZDo6VHJhY2s+JiB0cmFjayk7Ci0gICAgICAgIHZpcnR1YWwgICAgICAgICAgICAgflRyYWNrSGFuZGxlKCk7Ci0gICAgICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc3RhcnQoKTsKLSAgICAgICAgdmlydHVhbCB2b2lkICAgICAgICBzdG9wKCk7Ci0gICAgICAgIHZpcnR1YWwgdm9pZCAgICAgICAgZmx1c2goKTsKLSAgICAgICAgdmlydHVhbCB2b2lkICAgICAgICBtdXRlKGJvb2wpOwotICAgICAgICB2aXJ0dWFsIHZvaWQgICAgICAgIHBhdXNlKCk7Ci0gICAgICAgIHZpcnR1YWwgdm9pZCAgICAgICAgc2V0Vm9sdW1lKGZsb2F0IGxlZnQsIGZsb2F0IHJpZ2h0KTsKLSAgICAgICAgdmlydHVhbCBzcDxJTWVtb3J5PiBnZXRDYmxrKCkgY29uc3Q7Ci0gICAgICAgIHZpcnR1YWwgc3RhdHVzX3Qgb25UcmFuc2FjdCgKLSAgICAgICAgICAgIHVpbnQzMl90IGNvZGUsIGNvbnN0IFBhcmNlbCYgZGF0YSwgUGFyY2VsKiByZXBseSwgdWludDMyX3QgZmxhZ3MpOwotICAgIHByaXZhdGU6Ci0gICAgICAgIHNwPE1peGVyVGhyZWFkOjpUcmFjaz4gbVRyYWNrOwotICAgIH07Ci0KLSAgICBmcmllbmQgY2xhc3MgQ2xpZW50OwotICAgIGZyaWVuZCBjbGFzcyBNaXhlclRocmVhZDo6VHJhY2s7Ci0KLQotICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIHJlbW92ZUNsaWVudChwaWRfdCBwaWQpOwotCi0KLQotICAgIGNsYXNzIFJlY29yZEhhbmRsZSA6IHB1YmxpYyBhbmRyb2lkOjpCbkF1ZGlvUmVjb3JkIHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIFJlY29yZEhhbmRsZShjb25zdCBzcDxNaXhlclRocmVhZDo6UmVjb3JkVHJhY2s+JiByZWNvcmRUcmFjayk7Ci0gICAgICAgIHZpcnR1YWwgICAgICAgICAgICAgflJlY29yZEhhbmRsZSgpOwotICAgICAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHN0YXJ0KCk7Ci0gICAgICAgIHZpcnR1YWwgdm9pZCAgICAgICAgc3RvcCgpOwotICAgICAgICB2aXJ0dWFsIHNwPElNZW1vcnk+IGdldENibGsoKSBjb25zdDsKLSAgICAgICAgdmlydHVhbCBzdGF0dXNfdCBvblRyYW5zYWN0KAotICAgICAgICAgICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncyk7Ci0gICAgcHJpdmF0ZToKLSAgICAgICAgc3A8TWl4ZXJUaHJlYWQ6OlJlY29yZFRyYWNrPiBtUmVjb3JkVHJhY2s7Ci0gICAgfTsKLQotICAgIC8vIHJlY29yZCB0aHJlYWQKLSAgICBjbGFzcyBBdWRpb1JlY29yZFRocmVhZCA6IHB1YmxpYyBUaHJlYWQKLSAgICB7Ci0gICAgcHVibGljOgotICAgICAgICBBdWRpb1JlY29yZFRocmVhZChBdWRpb0hhcmR3YXJlSW50ZXJmYWNlKiBhdWRpb0hhcmR3YXJlKTsKLSAgICAgICAgdmlydHVhbCAgICAgICAgICAgICB+QXVkaW9SZWNvcmRUaHJlYWQoKTsKLSAgICAgICAgdmlydHVhbCBib29sICAgICAgICB0aHJlYWRMb29wKCk7Ci0gICAgICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgcmVhZHlUb1J1bigpIHsgcmV0dXJuIE5PX0VSUk9SOyB9Ci0gICAgICAgIHZpcnR1YWwgdm9pZCAgICAgICAgb25GaXJzdFJlZigpIHt9Ci0KLSAgICAgICAgICAgICAgICBzdGF0dXNfdCAgICBzdGFydChNaXhlclRocmVhZDo6UmVjb3JkVHJhY2sqIHJlY29yZFRyYWNrKTsKLSAgICAgICAgICAgICAgICB2b2lkICAgICAgICBzdG9wKE1peGVyVGhyZWFkOjpSZWNvcmRUcmFjayogcmVjb3JkVHJhY2spOwotICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIGV4aXQoKTsKLSAgICAgICAgICAgICAgICBzdGF0dXNfdCAgICBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0KLSAgICBwcml2YXRlOgotICAgICAgICAgICAgICAgIEF1ZGlvUmVjb3JkVGhyZWFkKCk7Ci0gICAgICAgICAgICAgICAgQXVkaW9IYXJkd2FyZUludGVyZmFjZSAgICAgICAgICAgICAgKm1BdWRpb0hhcmR3YXJlOwotICAgICAgICAgICAgICAgIHNwPE1peGVyVGhyZWFkOjpSZWNvcmRUcmFjaz4gICAgICAgIG1SZWNvcmRUcmFjazsKLSAgICAgICAgICAgICAgICBNdXRleCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtTG9jazsKLSAgICAgICAgICAgICAgICBDb25kaXRpb24gICAgICAgICAgICAgICAgICAgICAgICAgICBtV2FpdFdvcmtDVjsKLSAgICAgICAgICAgICAgICBDb25kaXRpb24gICAgICAgICAgICAgICAgICAgICAgICAgICBtU3RvcHBlZDsKLSAgICAgICAgICAgICAgICB2b2xhdGlsZSBib29sICAgICAgICAgICAgICAgICAgICAgICBtQWN0aXZlOwotICAgICAgICAgICAgICAgIHN0YXR1c190ICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1TdGFydFN0YXR1czsKLSAgICB9OwotCi0gICAgZnJpZW5kIGNsYXNzIEF1ZGlvUmVjb3JkVGhyZWFkOwotICAgIGZyaWVuZCBjbGFzcyBNaXhlclRocmVhZDsKLQotICAgICAgICAgICAgICAgIHN0YXR1c190ICAgIHN0YXJ0UmVjb3JkKE1peGVyVGhyZWFkOjpSZWNvcmRUcmFjayogcmVjb3JkVHJhY2spOwotICAgICAgICAgICAgICAgIHZvaWQgICAgICAgIHN0b3BSZWNvcmQoTWl4ZXJUaHJlYWQ6OlJlY29yZFRyYWNrKiByZWNvcmRUcmFjayk7Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgdm9pZCAgICAgICAgaGFuZGxlT3V0cHV0U3dpdGNoKCk7Ci0KLSAgICBtdXRhYmxlICAgICBNdXRleCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1IYXJkd2FyZUxvY2s7Ci0gICAgbXV0YWJsZSAgICAgTXV0ZXggICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtTG9jazsKLSAgICAgICAgICAgICAgICBEZWZhdWx0S2V5ZWRWZWN0b3I8IHBpZF90LCB3cDxDbGllbnQ+ID4gICAgIG1DbGllbnRzOwotCi0gICAgICAgICAgICAgICAgc3A8TWl4ZXJUaHJlYWQ+ICAgICAgICAgICAgICAgICAgICAgbUEyZHBNaXhlclRocmVhZDsKLSAgICAgICAgICAgICAgICBzcDxNaXhlclRocmVhZD4gICAgICAgICAgICAgICAgICAgICBtSGFyZHdhcmVNaXhlclRocmVhZDsKLSAgICAgICAgICAgICAgICBBdWRpb0hhcmR3YXJlSW50ZXJmYWNlKiAgICAgICAgICAgICBtQXVkaW9IYXJkd2FyZTsKLSAgICAgICAgICAgICAgICBBdWRpb0hhcmR3YXJlSW50ZXJmYWNlKiAgICAgICAgICAgICBtQTJkcEF1ZGlvSW50ZXJmYWNlOwotICAgICAgICAgICAgICAgIHNwPEF1ZGlvUmVjb3JkVGhyZWFkPiAgICAgICAgICAgICAgIG1BdWRpb1JlY29yZFRocmVhZDsKLSAgICAgICAgICAgICAgICBib29sICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtQTJkcEVuYWJsZWQ7Ci0gICAgICAgICAgICAgICAgYm9vbCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbUEyZHBFbmFibGVkUmVxOwotICAgIG11dGFibGUgICAgIGludCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1IYXJkd2FyZVN0YXR1czsKLSAgICAgICAgICAgICAgICBTb3J0ZWRWZWN0b3I8IHdwPElCaW5kZXI+ID4gICAgICAgICBtTm90aWZpY2F0aW9uQ2xpZW50czsKLSAgICAgICAgICAgICAgICBpbnQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtRm9yY2VkU3BlYWtlckNvdW50OwotICAgICAgICAgICAgICAgIHVpbnQzMl90ICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1TYXZlZFJvdXRlOwotICAgICAgICAgICAgICAgIHVpbnQzMl90ICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1Gb3JjZWRSb3V0ZTsKLSAgICAgICAgICAgICAgICBuc2Vjc190ICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtUm91dGVSZXN0b3JlVGltZTsKLSAgICAgICAgICAgICAgICBib29sICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtTXVzaWNNdXRlU2F2ZWQ7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0FVRElPX0ZMSU5HRVJfSApkaWZmIC0tZ2l0IGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9IYXJkd2FyZUdlbmVyaWMuY3BwIGIvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9IYXJkd2FyZUdlbmVyaWMuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2MmJlYWRhLi4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvSGFyZHdhcmVHZW5lcmljLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDMxMyArMCwwIEBACi0vKgotKioKLSoqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHVuaXN0ZC5oPgotI2luY2x1ZGUgPHNjaGVkLmg+Ci0jaW5jbHVkZSA8ZmNudGwuaD4KLSNpbmNsdWRlIDxzeXMvaW9jdGwuaD4KLQotI2RlZmluZSBMT0dfVEFHICJBdWRpb0hhcmR3YXJlIgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLQotI2luY2x1ZGUgIkF1ZGlvSGFyZHdhcmVHZW5lcmljLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgY2hhciBjb25zdCAqIGNvbnN0IGtBdWRpb0RldmljZU5hbWUgPSAiL2Rldi9lYWMiOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUF1ZGlvSGFyZHdhcmVHZW5lcmljOjpBdWRpb0hhcmR3YXJlR2VuZXJpYygpCi0gICAgOiBtT3V0cHV0KDApLCBtSW5wdXQoMCksICBtRmQoLTEpLCBtTWljTXV0ZShmYWxzZSkKLXsKLSAgICBtRmQgPSA6Om9wZW4oa0F1ZGlvRGV2aWNlTmFtZSwgT19SRFdSKTsKLX0KLQotQXVkaW9IYXJkd2FyZUdlbmVyaWM6On5BdWRpb0hhcmR3YXJlR2VuZXJpYygpCi17Ci0gICAgaWYgKG1GZCA+PSAwKSA6OmNsb3NlKG1GZCk7Ci0gICAgZGVsZXRlIG1PdXRwdXQ7Ci0gICAgZGVsZXRlIG1JbnB1dDsKLX0KLQotc3RhdHVzX3QgQXVkaW9IYXJkd2FyZUdlbmVyaWM6OmluaXRDaGVjaygpCi17Ci0gICAgaWYgKG1GZCA+PSAwKSB7Ci0gICAgICAgIGlmICg6OmFjY2VzcyhrQXVkaW9EZXZpY2VOYW1lLCBPX1JEV1IpID09IE5PX0VSUk9SKQotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLSAgICByZXR1cm4gTk9fSU5JVDsKLX0KLQotQXVkaW9TdHJlYW1PdXQqIEF1ZGlvSGFyZHdhcmVHZW5lcmljOjpvcGVuT3V0cHV0U3RyZWFtKAotICAgICAgICBpbnQgZm9ybWF0LCBpbnQgY2hhbm5lbENvdW50LCB1aW50MzJfdCBzYW1wbGVSYXRlLCBzdGF0dXNfdCAqc3RhdHVzKQotewotICAgIEF1dG9NdXRleCBsb2NrKG1Mb2NrKTsKLQotICAgIC8vIG9ubHkgb25lIG91dHB1dCBzdHJlYW0gYWxsb3dlZAotICAgIGlmIChtT3V0cHV0KSB7Ci0gICAgICAgIGlmIChzdGF0dXMpIHsKLSAgICAgICAgICAgICpzdGF0dXMgPSBJTlZBTElEX09QRVJBVElPTjsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvLyBjcmVhdGUgbmV3IG91dHB1dCBzdHJlYW0KLSAgICBBdWRpb1N0cmVhbU91dEdlbmVyaWMqIG91dCA9IG5ldyBBdWRpb1N0cmVhbU91dEdlbmVyaWMoKTsKLSAgICBzdGF0dXNfdCBsU3RhdHVzID0gb3V0LT5zZXQodGhpcywgbUZkLCBmb3JtYXQsIGNoYW5uZWxDb3VudCwgc2FtcGxlUmF0ZSk7Ci0gICAgaWYgKHN0YXR1cykgewotICAgICAgICAqc3RhdHVzID0gbFN0YXR1czsKLSAgICB9Ci0gICAgaWYgKGxTdGF0dXMgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgbU91dHB1dCA9IG91dDsKLSAgICB9IGVsc2UgewotICAgICAgICBkZWxldGUgb3V0OwotICAgIH0KLSAgICByZXR1cm4gbU91dHB1dDsKLX0KLQotdm9pZCBBdWRpb0hhcmR3YXJlR2VuZXJpYzo6Y2xvc2VPdXRwdXRTdHJlYW0oQXVkaW9TdHJlYW1PdXRHZW5lcmljKiBvdXQpIHsKLSAgICBpZiAob3V0ID09IG1PdXRwdXQpIG1PdXRwdXQgPSAwOwotfQotCi1BdWRpb1N0cmVhbUluKiBBdWRpb0hhcmR3YXJlR2VuZXJpYzo6b3BlbklucHV0U3RyZWFtKAotICAgICAgICBpbnQgZm9ybWF0LCBpbnQgY2hhbm5lbENvdW50LCB1aW50MzJfdCBzYW1wbGVSYXRlLCBzdGF0dXNfdCAqc3RhdHVzLAotICAgICAgICBBdWRpb1N5c3RlbTo6YXVkaW9faW5fYWNvdXN0aWNzIGFjb3VzdGljcykKLXsKLSAgICBBdXRvTXV0ZXggbG9jayhtTG9jayk7Ci0KLSAgICAvLyBvbmx5IG9uZSBpbnB1dCBzdHJlYW0gYWxsb3dlZAotICAgIGlmIChtSW5wdXQpIHsKLSAgICAgICAgaWYgKHN0YXR1cykgewotICAgICAgICAgICAgKnN0YXR1cyA9IElOVkFMSURfT1BFUkFUSU9OOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8vIGNyZWF0ZSBuZXcgb3V0cHV0IHN0cmVhbQotICAgIEF1ZGlvU3RyZWFtSW5HZW5lcmljKiBpbiA9IG5ldyBBdWRpb1N0cmVhbUluR2VuZXJpYygpOwotICAgIHN0YXR1c190IGxTdGF0dXMgPSBpbi0+c2V0KHRoaXMsIG1GZCwgZm9ybWF0LCBjaGFubmVsQ291bnQsIHNhbXBsZVJhdGUsIGFjb3VzdGljcyk7Ci0gICAgaWYgKHN0YXR1cykgewotICAgICAgICAqc3RhdHVzID0gbFN0YXR1czsKLSAgICB9Ci0gICAgaWYgKGxTdGF0dXMgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgbUlucHV0ID0gaW47Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgZGVsZXRlIGluOwotICAgIH0KLSAgICByZXR1cm4gbUlucHV0OwotfQotCi12b2lkIEF1ZGlvSGFyZHdhcmVHZW5lcmljOjpjbG9zZUlucHV0U3RyZWFtKEF1ZGlvU3RyZWFtSW5HZW5lcmljKiBpbikgewotICAgIGlmIChpbiA9PSBtSW5wdXQpIG1JbnB1dCA9IDA7Ci19Ci0KLXN0YXR1c190IEF1ZGlvSGFyZHdhcmVHZW5lcmljOjpzZXRWb2ljZVZvbHVtZShmbG9hdCB2KQotewotICAgIC8vIEltcGxlbWVudDogc2V0IHZvaWNlIHZvbHVtZQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9IYXJkd2FyZUdlbmVyaWM6OnNldE1hc3RlclZvbHVtZShmbG9hdCB2KQotewotICAgIC8vIEltcGxlbWVudDogc2V0IG1hc3RlciB2b2x1bWUKLSAgICAvLyByZXR1cm4gZXJyb3IgLSBzb2Z0d2FyZSBtaXhlciB3aWxsIGhhbmRsZSBpdAotICAgIHJldHVybiBJTlZBTElEX09QRVJBVElPTjsKLX0KLQotc3RhdHVzX3QgQXVkaW9IYXJkd2FyZUdlbmVyaWM6OnNldE1pY011dGUoYm9vbCBzdGF0ZSkKLXsKLSAgICBtTWljTXV0ZSA9IHN0YXRlOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9IYXJkd2FyZUdlbmVyaWM6OmdldE1pY011dGUoYm9vbCogc3RhdGUpCi17Ci0gICAgKnN0YXRlID0gbU1pY011dGU7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBBdWRpb0hhcmR3YXJlR2VuZXJpYzo6ZHVtcEludGVybmFscyhpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpCi17Ci0gICAgY29uc3Qgc2l6ZV90IFNJWkUgPSAyNTY7Ci0gICAgY2hhciBidWZmZXJbU0laRV07Ci0gICAgU3RyaW5nOCByZXN1bHQ7Ci0gICAgcmVzdWx0LmFwcGVuZCgiQXVkaW9IYXJkd2FyZUdlbmVyaWM6OmR1bXBJbnRlcm5hbHNcbiIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIlx0bUZkOiAlZCBtTWljTXV0ZTogJXNcbiIsICBtRmQsIG1NaWNNdXRlPyAidHJ1ZSI6ICJmYWxzZSIpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICA6OndyaXRlKGZkLCByZXN1bHQuc3RyaW5nKCksIHJlc3VsdC5zaXplKCkpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9IYXJkd2FyZUdlbmVyaWM6OmR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKQotewotICAgIGR1bXBJbnRlcm5hbHMoZmQsIGFyZ3MpOwotICAgIGlmIChtSW5wdXQpIHsKLSAgICAgICAgbUlucHV0LT5kdW1wKGZkLCBhcmdzKTsKLSAgICB9Ci0gICAgaWYgKG1PdXRwdXQpIHsKLSAgICAgICAgbU91dHB1dC0+ZHVtcChmZCwgYXJncyk7Ci0gICAgfQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0dXNfdCBBdWRpb1N0cmVhbU91dEdlbmVyaWM6OnNldCgKLSAgICAgICAgQXVkaW9IYXJkd2FyZUdlbmVyaWMgKmh3LAotICAgICAgICBpbnQgZmQsCi0gICAgICAgIGludCBmb3JtYXQsCi0gICAgICAgIGludCBjaGFubmVscywKLSAgICAgICAgdWludDMyX3QgcmF0ZSkKLXsKLSAgICAvLyBmaXggdXAgZGVmYXVsdHMKLSAgICBpZiAoZm9ybWF0ID09IDApIGZvcm1hdCA9IEF1ZGlvU3lzdGVtOjpQQ01fMTZfQklUOwotICAgIGlmIChjaGFubmVscyA9PSAwKSBjaGFubmVscyA9IGNoYW5uZWxDb3VudCgpOwotICAgIGlmIChyYXRlID09IDApIHJhdGUgPSBzYW1wbGVSYXRlKCk7Ci0KLSAgICAvLyBjaGVjayB2YWx1ZXMKLSAgICBpZiAoKGZvcm1hdCAhPSBBdWRpb1N5c3RlbTo6UENNXzE2X0JJVCkgfHwKLSAgICAgICAgICAgIChjaGFubmVscyAhPSBjaGFubmVsQ291bnQoKSkgfHwKLSAgICAgICAgICAgIChyYXRlICE9IHNhbXBsZVJhdGUoKSkpCi0gICAgICAgIHJldHVybiBCQURfVkFMVUU7Ci0KLSAgICBtQXVkaW9IYXJkd2FyZSA9IGh3OwotICAgIG1GZCA9IGZkOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotQXVkaW9TdHJlYW1PdXRHZW5lcmljOjp+QXVkaW9TdHJlYW1PdXRHZW5lcmljKCkKLXsKLSAgICBpZiAobUF1ZGlvSGFyZHdhcmUpCi0gICAgICAgIG1BdWRpb0hhcmR3YXJlLT5jbG9zZU91dHB1dFN0cmVhbSh0aGlzKTsKLX0KLQotc3NpemVfdCBBdWRpb1N0cmVhbU91dEdlbmVyaWM6OndyaXRlKGNvbnN0IHZvaWQqIGJ1ZmZlciwgc2l6ZV90IGJ5dGVzKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgcmV0dXJuIHNzaXplX3QoOjp3cml0ZShtRmQsIGJ1ZmZlciwgYnl0ZXMpKTsKLX0KLQotc3RhdHVzX3QgQXVkaW9TdHJlYW1PdXRHZW5lcmljOjpzdGFuZGJ5KCkKLXsKLSAgICAvLyBJbXBsZW1lbnQ6IGF1ZGlvIGhhcmR3YXJlIHRvIHN0YW5kYnkgbW9kZQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9TdHJlYW1PdXRHZW5lcmljOjpkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBjb25zdCBzaXplX3QgU0laRSA9IDI1NjsKLSAgICBjaGFyIGJ1ZmZlcltTSVpFXTsKLSAgICBTdHJpbmc4IHJlc3VsdDsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJBdWRpb1N0cmVhbU91dEdlbmVyaWM6OmR1bXBcbiIpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJcdHNhbXBsZSByYXRlOiAlZFxuIiwgc2FtcGxlUmF0ZSgpKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiXHRidWZmZXIgc2l6ZTogJWRcbiIsIGJ1ZmZlclNpemUoKSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIlx0Y2hhbm5lbCBjb3VudDogJWRcbiIsIGNoYW5uZWxDb3VudCgpKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiXHRmb3JtYXQ6ICVkXG4iLCBmb3JtYXQoKSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIlx0bUF1ZGlvSGFyZHdhcmU6ICVwXG4iLCBtQXVkaW9IYXJkd2FyZSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIlx0bUZkOiAlZFxuIiwgbUZkKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgOjp3cml0ZShmZCwgcmVzdWx0LnN0cmluZygpLCByZXN1bHQuc2l6ZSgpKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLy8gcmVjb3JkIGZ1bmN0aW9ucwotc3RhdHVzX3QgQXVkaW9TdHJlYW1JbkdlbmVyaWM6OnNldCgKLSAgICAgICAgQXVkaW9IYXJkd2FyZUdlbmVyaWMgKmh3LAotICAgICAgICBpbnQgZmQsCi0gICAgICAgIGludCBmb3JtYXQsCi0gICAgICAgIGludCBjaGFubmVscywKLSAgICAgICAgdWludDMyX3QgcmF0ZSwKLSAgICAgICAgQXVkaW9TeXN0ZW06OmF1ZGlvX2luX2Fjb3VzdGljcyBhY291c3RpY3MpCi17Ci0gICAgLy8gRklYTUU6IHJlbW92ZSBsb2dnaW5nCi0gICAgTE9HRCgiQXVkaW9TdHJlYW1JbkdlbmVyaWM6OnNldCglcCwgJWQsICVkLCAlZCwgJXUpIiwgaHcsIGZkLCBmb3JtYXQsIGNoYW5uZWxzLCByYXRlKTsKLSAgICAvLyBjaGVjayB2YWx1ZXMKLSAgICBpZiAoKGZvcm1hdCAhPSBBdWRpb1N5c3RlbTo6UENNXzE2X0JJVCkgfHwKLSAgICAgICAgICAgIChjaGFubmVscyAhPSBjaGFubmVsQ291bnQoKSkgfHwKLSAgICAgICAgICAgIChyYXRlICE9IHNhbXBsZVJhdGUoKSkpIHsKLSAgICAgICAgTE9HRSgiRXJyb3Igb3BlbmluZyBpbnB1dCBjaGFubmVsIik7Ci0gICAgICAgIHJldHVybiBCQURfVkFMVUU7Ci0gICAgfQotCi0gICAgbUF1ZGlvSGFyZHdhcmUgPSBodzsKLSAgICBtRmQgPSBmZDsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLUF1ZGlvU3RyZWFtSW5HZW5lcmljOjp+QXVkaW9TdHJlYW1JbkdlbmVyaWMoKQotewotICAgIC8vIEZJWE1FOiByZW1vdmUgbG9nZ2luZwotICAgIExPR0QoIkF1ZGlvU3RyZWFtSW5HZW5lcmljIGRlc3RydWN0b3IiKTsKLSAgICBpZiAobUF1ZGlvSGFyZHdhcmUpCi0gICAgICAgIG1BdWRpb0hhcmR3YXJlLT5jbG9zZUlucHV0U3RyZWFtKHRoaXMpOwotfQotCi1zc2l6ZV90IEF1ZGlvU3RyZWFtSW5HZW5lcmljOjpyZWFkKHZvaWQqIGJ1ZmZlciwgc3NpemVfdCBieXRlcykKLXsKLSAgICAvLyBGSVhNRTogcmVtb3ZlIGxvZ2dpbmcKLSAgICBMT0dEKCJBdWRpb1N0cmVhbUluR2VuZXJpYzo6cmVhZCglcCwgJWQpIGZyb20gZmQgJWQiLCBidWZmZXIsIGJ5dGVzLCBtRmQpOwotICAgIEF1dG9NdXRleCBsb2NrKG1Mb2NrKTsKLSAgICBpZiAobUZkIDwgMCkgewotICAgICAgICBMT0dFKCJBdHRlbXB0IHRvIHJlYWQgZnJvbSB1bm9wZW5lZCBkZXZpY2UiKTsKLSAgICAgICAgcmV0dXJuIE5PX0lOSVQ7Ci0gICAgfQotICAgIHJldHVybiA6OnJlYWQobUZkLCBidWZmZXIsIGJ5dGVzKTsKLX0KLQotc3RhdHVzX3QgQXVkaW9TdHJlYW1JbkdlbmVyaWM6OmR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKQotewotICAgIGNvbnN0IHNpemVfdCBTSVpFID0gMjU2OwotICAgIGNoYXIgYnVmZmVyW1NJWkVdOwotICAgIFN0cmluZzggcmVzdWx0OwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIkF1ZGlvU3RyZWFtSW5HZW5lcmljOjpkdW1wXG4iKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiXHRzYW1wbGUgcmF0ZTogJWRcbiIsIHNhbXBsZVJhdGUoKSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIlx0YnVmZmVyIHNpemU6ICVkXG4iLCBidWZmZXJTaXplKCkpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJcdGNoYW5uZWwgY291bnQ6ICVkXG4iLCBjaGFubmVsQ291bnQoKSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIlx0Zm9ybWF0OiAlZFxuIiwgZm9ybWF0KCkpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJcdG1BdWRpb0hhcmR3YXJlOiAlcFxuIiwgbUF1ZGlvSGFyZHdhcmUpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJcdG1GZDogJWRcbiIsIG1GZCk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIDo6d3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0hhcmR3YXJlR2VuZXJpYy5oIGIvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9IYXJkd2FyZUdlbmVyaWMuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMWQ1ODM4OS4uMDAwMDAwMAotLS0gYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0hhcmR3YXJlR2VuZXJpYy5oCisrKyAvZGV2L251bGwKQEAgLTEsMTQxICswLDAgQEAKLS8qCi0qKgotKiogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9BVURJT19IQVJEV0FSRV9HRU5FUklDX0gKLSNkZWZpbmUgQU5EUk9JRF9BVURJT19IQVJEV0FSRV9HRU5FUklDX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotCi0jaW5jbHVkZSA8aGFyZHdhcmVfbGVnYWN5L0F1ZGlvSGFyZHdhcmVCYXNlLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBBdWRpb0hhcmR3YXJlR2VuZXJpYzsKLQotY2xhc3MgQXVkaW9TdHJlYW1PdXRHZW5lcmljIDogcHVibGljIEF1ZGlvU3RyZWFtT3V0IHsKLXB1YmxpYzoKLSAgICAgICAgICAgICAgICAgICAgICAgIEF1ZGlvU3RyZWFtT3V0R2VuZXJpYygpIDogbUF1ZGlvSGFyZHdhcmUoMCksIG1GZCgtMSkge30KLSAgICB2aXJ0dWFsICAgICAgICAgICAgIH5BdWRpb1N0cmVhbU91dEdlbmVyaWMoKTsKLQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0KAotICAgICAgICAgICAgQXVkaW9IYXJkd2FyZUdlbmVyaWMgKmh3LAotICAgICAgICAgICAgaW50IG1GZCwKLSAgICAgICAgICAgIGludCBmb3JtYXQsCi0gICAgICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSk7Ci0KLSAgICB2aXJ0dWFsIHVpbnQzMl90ICAgIHNhbXBsZVJhdGUoKSBjb25zdCB7IHJldHVybiA0NDEwMDsgfQotICAgIHZpcnR1YWwgc2l6ZV90ICAgICAgYnVmZmVyU2l6ZSgpIGNvbnN0IHsgcmV0dXJuIDQwOTY7IH0KLSAgICB2aXJ0dWFsIGludCAgICAgICAgIGNoYW5uZWxDb3VudCgpIGNvbnN0IHsgcmV0dXJuIDI7IH0KLSAgICB2aXJ0dWFsIGludCAgICAgICAgIGZvcm1hdCgpIGNvbnN0IHsgcmV0dXJuIEF1ZGlvU3lzdGVtOjpQQ01fMTZfQklUOyB9Ci0gICAgdmlydHVhbCB1aW50MzJfdCAgICBsYXRlbmN5KCkgY29uc3QgeyByZXR1cm4gMDsgfQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0Vm9sdW1lKGZsb2F0IHZvbHVtZSkgeyByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047IH0KLSAgICB2aXJ0dWFsIHNzaXplX3QgICAgIHdyaXRlKGNvbnN0IHZvaWQqIGJ1ZmZlciwgc2l6ZV90IGJ5dGVzKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHN0YW5kYnkoKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIGR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKTsKLQotcHJpdmF0ZToKLSAgICBBdWRpb0hhcmR3YXJlR2VuZXJpYyAqbUF1ZGlvSGFyZHdhcmU7Ci0gICAgTXV0ZXggICBtTG9jazsKLSAgICBpbnQgICAgIG1GZDsKLX07Ci0KLWNsYXNzIEF1ZGlvU3RyZWFtSW5HZW5lcmljIDogcHVibGljIEF1ZGlvU3RyZWFtSW4gewotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgQXVkaW9TdHJlYW1JbkdlbmVyaWMoKSA6IG1BdWRpb0hhcmR3YXJlKDApLCBtRmQoLTEpIHt9Ci0gICAgdmlydHVhbCAgICAgICAgICAgICB+QXVkaW9TdHJlYW1JbkdlbmVyaWMoKTsKLQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0KAotICAgICAgICAgICAgQXVkaW9IYXJkd2FyZUdlbmVyaWMgKmh3LAotICAgICAgICAgICAgaW50IG1GZCwKLSAgICAgICAgICAgIGludCBmb3JtYXQsCi0gICAgICAgICAgICBpbnQgY2hhbm5lbENvdW50LAotICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgICAgIEF1ZGlvU3lzdGVtOjphdWRpb19pbl9hY291c3RpY3MgYWNvdXN0aWNzKTsKLQotICAgIHVpbnQzMl90ICAgIHNhbXBsZVJhdGUoKSBjb25zdCB7IHJldHVybiA4MDAwOyB9Ci0gICAgdmlydHVhbCBzaXplX3QgICAgICBidWZmZXJTaXplKCkgY29uc3QgeyByZXR1cm4gMzIwOyB9Ci0gICAgdmlydHVhbCBpbnQgICAgICAgICBjaGFubmVsQ291bnQoKSBjb25zdCB7IHJldHVybiAxOyB9Ci0gICAgdmlydHVhbCBpbnQgICAgICAgICBmb3JtYXQoKSBjb25zdCB7IHJldHVybiBBdWRpb1N5c3RlbTo6UENNXzE2X0JJVDsgfQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0R2FpbihmbG9hdCBnYWluKSB7IHJldHVybiBJTlZBTElEX09QRVJBVElPTjsgfQotICAgIHZpcnR1YWwgc3NpemVfdCAgICAgcmVhZCh2b2lkKiBidWZmZXIsIHNzaXplX3QgYnl0ZXMpOwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgZHVtcChpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpOwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc3RhbmRieSgpIHsgcmV0dXJuIE5PX0VSUk9SOyB9Ci0KLXByaXZhdGU6Ci0gICAgQXVkaW9IYXJkd2FyZUdlbmVyaWMgKm1BdWRpb0hhcmR3YXJlOwotICAgIE11dGV4ICAgbUxvY2s7Ci0gICAgaW50ICAgICBtRmQ7Ci19OwotCi0KLWNsYXNzIEF1ZGlvSGFyZHdhcmVHZW5lcmljIDogcHVibGljIEF1ZGlvSGFyZHdhcmVCYXNlCi17Ci1wdWJsaWM6Ci0gICAgICAgICAgICAgICAgICAgICAgICBBdWRpb0hhcmR3YXJlR2VuZXJpYygpOwotICAgIHZpcnR1YWwgICAgICAgICAgICAgfkF1ZGlvSGFyZHdhcmVHZW5lcmljKCk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBpbml0Q2hlY2soKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHNldFZvaWNlVm9sdW1lKGZsb2F0IHZvbHVtZSk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRNYXN0ZXJWb2x1bWUoZmxvYXQgdm9sdW1lKTsKLQotICAgIC8vIG1pYyBtdXRlCi0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRNaWNNdXRlKGJvb2wgc3RhdGUpOwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgZ2V0TWljTXV0ZShib29sKiBzdGF0ZSk7Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHNldFBhcmFtZXRlcihjb25zdCBjaGFyKiBrZXksIGNvbnN0IGNoYXIqIHZhbHVlKQotICAgICAgICAgICAgeyByZXR1cm4gTk9fRVJST1I7IH0KLQotICAgIC8vIGNyZWF0ZSBJL08gc3RyZWFtcwotICAgIHZpcnR1YWwgQXVkaW9TdHJlYW1PdXQqIG9wZW5PdXRwdXRTdHJlYW0oCi0gICAgICAgICAgICBpbnQgZm9ybWF0PTAsCi0gICAgICAgICAgICBpbnQgY2hhbm5lbENvdW50PTAsCi0gICAgICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlPTAsCi0gICAgICAgICAgICBzdGF0dXNfdCAqc3RhdHVzPTApOwotCi0gICAgdmlydHVhbCBBdWRpb1N0cmVhbUluKiBvcGVuSW5wdXRTdHJlYW0oCi0gICAgICAgICAgICBpbnQgZm9ybWF0LAotICAgICAgICAgICAgaW50IGNoYW5uZWxDb3VudCwKLSAgICAgICAgICAgIHVpbnQzMl90IHNhbXBsZVJhdGUsCi0gICAgICAgICAgICBzdGF0dXNfdCAqc3RhdHVzLAotICAgICAgICAgICAgQXVkaW9TeXN0ZW06OmF1ZGlvX2luX2Fjb3VzdGljcyBhY291c3RpY3MpOwotCi0gICAgICAgICAgICB2b2lkICAgICAgICAgICAgY2xvc2VPdXRwdXRTdHJlYW0oQXVkaW9TdHJlYW1PdXRHZW5lcmljKiBvdXQpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgIGNsb3NlSW5wdXRTdHJlYW0oQXVkaW9TdHJlYW1JbkdlbmVyaWMqIGluKTsKLXByb3RlY3RlZDoKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICBkb1JvdXRpbmcoKSB7IHJldHVybiBOT19FUlJPUjsgfQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgIGR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKTsKLQotcHJpdmF0ZToKLSAgICBzdGF0dXNfdCAgICAgICAgICAgICAgICBkdW1wSW50ZXJuYWxzKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0KLSAgICBNdXRleCAgICAgICAgICAgICAgICAgICBtTG9jazsKLSAgICBBdWRpb1N0cmVhbU91dEdlbmVyaWMgICAqbU91dHB1dDsKLSAgICBBdWRpb1N0cmVhbUluR2VuZXJpYyAgICAqbUlucHV0OwotICAgIGludCAgICAgICAgICAgICAgICAgICAgIG1GZDsKLSAgICBib29sICAgICAgICAgICAgICAgICAgICBtTWljTXV0ZTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfQVVESU9fSEFSRFdBUkVfR0VORVJJQ19ICmRpZmYgLS1naXQgYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0hhcmR3YXJlSW50ZXJmYWNlLmNwcCBiL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvSGFyZHdhcmVJbnRlcmZhY2UuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBhYzc2YTE5Li4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvSGFyZHdhcmVJbnRlcmZhY2UuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMjQ3ICswLDAgQEAKLS8qCi0qKgotKiogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2luY2x1ZGUgPGN1dGlscy9wcm9wZXJ0aWVzLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0KLSNkZWZpbmUgTE9HX1RBRyAiQXVkaW9IYXJkd2FyZUludGVyZmFjZSIKLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0KLSNpbmNsdWRlICJBdWRpb0hhcmR3YXJlU3R1Yi5oIgotI2luY2x1ZGUgIkF1ZGlvSGFyZHdhcmVHZW5lcmljLmgiCi0KLS8vI2RlZmluZSBEVU1QX0ZMSU5HRVJfT1VUICAgICAgICAvLyBpZiBkZWZpbmVkIGFsbG93cyByZWNvcmRpbmcgc2FtcGxlcyBpbiBhIGZpbGUKLSNpZmRlZiBEVU1QX0ZMSU5HRVJfT1VUCi0jaW5jbHVkZSAiQXVkaW9EdW1wSW50ZXJmYWNlLmgiCi0jZW5kaWYKLQotCi0vLyBjaGFuZ2UgdG8gMSB0byBsb2cgcm91dGluZyBjYWxscwotI2RlZmluZSBMT0dfUk9VVElOR19DQUxMUyAwCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotI2lmIExPR19ST1VUSU5HX0NBTExTCi1zdGF0aWMgY29uc3QgY2hhciogcm91dGluZ01vZGVTdHJpbmdzW10gPQotewotICAgICJPVVQgT0YgUkFOR0UiLAotICAgICJJTlZBTElEIiwKLSAgICAiQ1VSUkVOVCIsCi0gICAgIk5PUk1BTCIsCi0gICAgIlJJTkdUT05FIiwKLSAgICAiSU5fQ0FMTCIKLX07Ci0KLXN0YXRpYyBjb25zdCBjaGFyKiByb3V0ZVN0cmluZ3NbXSA9Ci17Ci0gICAgIkVBUlBJRUNFICIsCi0gICAgIlNQRUFLRVIgIiwKLSAgICAiQkxVRVRPT1RIICIsCi0gICAgIkhFQURTRVQgIgotICAgICJCTFVFVE9PVEhfQTJEUCAiCi19Owotc3RhdGljIGNvbnN0IGNoYXIqIHJvdXRlTm9uZSA9ICJOT05FIjsKLQotc3RhdGljIGNvbnN0IGNoYXIqIGRpc3BsYXlNb2RlKGludCBtb2RlKQotewotICAgIGlmICgobW9kZSA8IC0yKSB8fCAobW9kZSA+IDIpKQotICAgICAgICByZXR1cm4gcm91dGluZ01vZGVTdHJpbmdzWzBdOwotICAgIHJldHVybiByb3V0aW5nTW9kZVN0cmluZ3NbbW9kZSszXTsKLX0KLQotc3RhdGljIGNvbnN0IGNoYXIqIGRpc3BsYXlSb3V0ZXModWludDMyX3Qgcm91dGVzKQotewotICAgIHN0YXRpYyBjaGFyIHJvdXRlU3RyWzgwXTsKLSAgICBpZiAocm91dGVzID09IDApCi0gICAgICAgIHJldHVybiByb3V0ZU5vbmU7Ci0gICAgcm91dGVTdHJbMF0gPSAwOwotICAgIGludCBiaXRNYXNrID0gMTsKLSAgICBmb3IgKGludCBpID0gMDsgaSA8IDQ7ICsraSwgYml0TWFzayA8PD0gMSkgewotICAgICAgICBpZiAocm91dGVzICYgYml0TWFzaykgewotICAgICAgICAgICAgc3RyY2F0KHJvdXRlU3RyLCByb3V0ZVN0cmluZ3NbaV0pOwotICAgICAgICB9Ci0gICAgfQotICAgIHJvdXRlU3RyW3N0cmxlbihyb3V0ZVN0ciktMV0gPSAwOwotICAgIHJldHVybiByb3V0ZVN0cjsKLX0KLSNlbmRpZgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUF1ZGlvSGFyZHdhcmVJbnRlcmZhY2UqIEF1ZGlvSGFyZHdhcmVJbnRlcmZhY2U6OmNyZWF0ZSgpCi17Ci0gICAgLyoKLSAgICAgKiBGSVhNRTogVGhpcyBjb2RlIG5lZWRzIHRvIGluc3RhbnRpYXRlIHRoZSBjb3JyZWN0IGF1ZGlvIGRldmljZQotICAgICAqIGludGVyZmFjZS4gRm9yIG5vdyAtIHdlIHVzZSBjb21waWxlLXRpbWUgc3dpdGNoZXMuCi0gICAgICovCi0gICAgQXVkaW9IYXJkd2FyZUludGVyZmFjZSogaHcgPSAwOwotICAgIGNoYXIgdmFsdWVbUFJPUEVSVFlfVkFMVUVfTUFYXTsKLQotI2lmZGVmIEdFTkVSSUNfQVVESU8KLSAgICBodyA9IG5ldyBBdWRpb0hhcmR3YXJlR2VuZXJpYygpOwotI2Vsc2UKLSAgICAvLyBpZiBydW5uaW5nIGluIGVtdWxhdGlvbiAtIHVzZSB0aGUgZW11bGF0b3IgZHJpdmVyCi0gICAgaWYgKHByb3BlcnR5X2dldCgicm8ua2VybmVsLnFlbXUiLCB2YWx1ZSwgMCkpIHsKLSAgICAgICAgTE9HRCgiUnVubmluZyBpbiBlbXVsYXRpb24gLSB1c2luZyBnZW5lcmljIGF1ZGlvIGRyaXZlciIpOwotICAgICAgICBodyA9IG5ldyBBdWRpb0hhcmR3YXJlR2VuZXJpYygpOwotICAgIH0KLSAgICBlbHNlIHsKLSAgICAgICAgTE9HVigiQ3JlYXRpbmcgVmVuZG9yIFNwZWNpZmljIEF1ZGlvSGFyZHdhcmUiKTsKLSAgICAgICAgaHcgPSBjcmVhdGVBdWRpb0hhcmR3YXJlKCk7Ci0gICAgfQotI2VuZGlmCi0gICAgaWYgKGh3LT5pbml0Q2hlY2soKSAhPSBOT19FUlJPUikgewotICAgICAgICBMT0dXKCJVc2luZyBzdHViYmVkIGF1ZGlvIGhhcmR3YXJlLiBObyBzb3VuZCB3aWxsIGJlIHByb2R1Y2VkLiIpOwotICAgICAgICBkZWxldGUgaHc7Ci0gICAgICAgIGh3ID0gbmV3IEF1ZGlvSGFyZHdhcmVTdHViKCk7Ci0gICAgfQotICAgIAotI2lmZGVmIERVTVBfRkxJTkdFUl9PVVQKLSAgICAvLyBUaGlzIGNvZGUgYWRkcyBhIHJlY29yZCBvZiBidWZmZXJzIGluIGEgZmlsZSB0byB3cml0ZSBjYWxscyBtYWRlIGJ5IEF1ZGlvRmxpbmdlci4KLSAgICAvLyBJdCByZXBsYWNlcyB0aGUgY3VycmVudCBBdWRpb0hhcmR3YXJlSW50ZXJmYWNlIG9iamVjdCBieSBhbiBpbnRlcm1lZGlhdGUgb25lIHdoaWNoCi0gICAgLy8gd2lsbCByZWNvcmQgYnVmZmVycyBpbiBhIGZpbGUgKGFmdGVyIHNlbmRpbmcgdGhlbSB0byBoYXJkd2FyZSkgZm9yIHRlc3RpbmcgcHVycG9zZS4KLSAgICAvLyBUaGlzIGZlYXR1cmUgaXMgZW5hYmxlZCBieSBkZWZpbmluZyBzeW1ib2wgRFVNUF9GTElOR0VSX09VVC4KLSAgICAvLyBUaGUgb3V0cHV0IGZpbGUgaXMgRkxJTkdFUl9EVU1QX05BTUUuIFBhdXNlIGFyZSBub3QgcmVjb3JkZWQgaW4gdGhlIGZpbGUuCi0gICAgCi0gICAgaHcgPSBuZXcgQXVkaW9EdW1wSW50ZXJmYWNlKGh3KTsgICAgLy8gcmVwbGFjZSBpbnRlcmZhY2UKLSNlbmRpZgotICAgIHJldHVybiBodzsKLX0KLQotQXVkaW9TdHJlYW1PdXQ6On5BdWRpb1N0cmVhbU91dCgpCi17Ci19Ci0KLUF1ZGlvU3RyZWFtSW46On5BdWRpb1N0cmVhbUluKCkge30KLQotQXVkaW9IYXJkd2FyZUJhc2U6OkF1ZGlvSGFyZHdhcmVCYXNlKCkKLXsKLSAgICAvLyBmb3JjZSBhIHJvdXRpbmcgdXBkYXRlIG9uIGluaXRpYWxpemF0aW9uCi0gICAgbWVtc2V0KCZtUm91dGVzLCAwLCBzaXplb2YobVJvdXRlcykpOwotICAgIG1Nb2RlID0gMDsKLX0KLQotLy8gZ2VuZXJpY3MgZm9yIGF1ZGlvIHJvdXRpbmcgLSB0aGUgcmVhbCB3b3JrIGlzIGRvbmUgaW4gZG9Sb3V0aW5nCi1zdGF0dXNfdCBBdWRpb0hhcmR3YXJlQmFzZTo6c2V0Um91dGluZyhpbnQgbW9kZSwgdWludDMyX3Qgcm91dGVzKQotewotI2lmIExPR19ST1VUSU5HX0NBTExTCi0gICAgTE9HRCgic2V0Um91dGluZzogbW9kZT0lcywgcm91dGVzPVslc10iLCBkaXNwbGF5TW9kZShtb2RlKSwgZGlzcGxheVJvdXRlcyhyb3V0ZXMpKTsKLSNlbmRpZgotICAgIGlmIChtb2RlID09IEF1ZGlvU3lzdGVtOjpNT0RFX0NVUlJFTlQpCi0gICAgICAgIG1vZGUgPSBtTW9kZTsKLSAgICBpZiAoKG1vZGUgPCAwKSB8fCAobW9kZSA+PSBBdWRpb1N5c3RlbTo6TlVNX01PREVTKSkKLSAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLSAgICB1aW50MzJfdCBvbGQgPSBtUm91dGVzW21vZGVdOwotICAgIG1Sb3V0ZXNbbW9kZV0gPSByb3V0ZXM7Ci0gICAgaWYgKChtb2RlICE9IG1Nb2RlKSB8fCAob2xkID09IHJvdXRlcykpCi0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSNpZiBMT0dfUk9VVElOR19DQUxMUwotICAgIGNvbnN0IGNoYXIqIG9sZFJvdXRlU3RyID0gc3RyZHVwKGRpc3BsYXlSb3V0ZXMob2xkKSk7Ci0gICAgTE9HRCgiZG9Sb3V0aW5nOiBtb2RlPSVzLCBvbGQgcm91dGU9WyVzXSwgbmV3IHJvdXRlPVslc10iLAotICAgICAgICAgICBkaXNwbGF5TW9kZShtb2RlKSwgb2xkUm91dGVTdHIsIGRpc3BsYXlSb3V0ZXMocm91dGVzKSk7Ci0gICAgZGVsZXRlIG9sZFJvdXRlU3RyOwotI2VuZGlmCi0gICAgcmV0dXJuIGRvUm91dGluZygpOwotfQotCi1zdGF0dXNfdCBBdWRpb0hhcmR3YXJlQmFzZTo6Z2V0Um91dGluZyhpbnQgbW9kZSwgdWludDMyX3QqIHJvdXRlcykKLXsKLSAgICBpZiAobW9kZSA9PSBBdWRpb1N5c3RlbTo6TU9ERV9DVVJSRU5UKQotICAgICAgICBtb2RlID0gbU1vZGU7Ci0gICAgaWYgKChtb2RlIDwgMCkgfHwgKG1vZGUgPj0gQXVkaW9TeXN0ZW06Ok5VTV9NT0RFUykpCi0gICAgICAgIHJldHVybiBCQURfVkFMVUU7Ci0gICAgKnJvdXRlcyA9IG1Sb3V0ZXNbbW9kZV07Ci0jaWYgTE9HX1JPVVRJTkdfQ0FMTFMKLSAgICBMT0dEKCJnZXRSb3V0aW5nOiBtb2RlPSVzLCByb3V0ZXM9WyVzXSIsCi0gICAgICAgICAgIGRpc3BsYXlNb2RlKG1vZGUpLCBkaXNwbGF5Um91dGVzKCpyb3V0ZXMpKTsKLSNlbmRpZgotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9IYXJkd2FyZUJhc2U6OnNldE1vZGUoaW50IG1vZGUpCi17Ci0jaWYgTE9HX1JPVVRJTkdfQ0FMTFMKLSAgICBMT0dEKCJzZXRNb2RlKCVzKSIsIGRpc3BsYXlNb2RlKG1vZGUpKTsKLSNlbmRpZgotICAgIGlmICgobW9kZSA8IDApIHx8IChtb2RlID49IEF1ZGlvU3lzdGVtOjpOVU1fTU9ERVMpKQotICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOwotICAgIGlmIChtTW9kZSA9PSBtb2RlKQotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0jaWYgTE9HX1JPVVRJTkdfQ0FMTFMKLSAgICBMT0dEKCJkb1JvdXRpbmc6IG9sZCBtb2RlPSVzLCBuZXcgbW9kZT0lcyByb3V0ZT1bJXNdIiwKLSAgICAgICAgICAgIGRpc3BsYXlNb2RlKG1Nb2RlKSwgZGlzcGxheU1vZGUobW9kZSksIGRpc3BsYXlSb3V0ZXMobVJvdXRlc1ttb2RlXSkpOwotI2VuZGlmCi0gICAgbU1vZGUgPSBtb2RlOwotICAgIHJldHVybiBkb1JvdXRpbmcoKTsKLX0KLQotc3RhdHVzX3QgQXVkaW9IYXJkd2FyZUJhc2U6OmdldE1vZGUoaW50KiBtb2RlKQotewotICAgIC8vIEltcGxlbWVudDogc2V0IGF1ZGlvIHJvdXRpbmcKLSAgICAqbW9kZSA9IG1Nb2RlOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9IYXJkd2FyZUJhc2U6OnNldFBhcmFtZXRlcihjb25zdCBjaGFyKiBrZXksIGNvbnN0IGNoYXIqIHZhbHVlKQotewotICAgIC8vIGRlZmF1bHQgaW1wbGVtZW50YXRpb24gaXMgdG8gaWdub3JlCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0KLS8vIGRlZmF1bHQgaW1wbGVtZW50YXRpb24KLXNpemVfdCBBdWRpb0hhcmR3YXJlQmFzZTo6Z2V0SW5wdXRCdWZmZXJTaXplKHVpbnQzMl90IHNhbXBsZVJhdGUsIGludCBmb3JtYXQsIGludCBjaGFubmVsQ291bnQpCi17Ci0gICAgaWYgKHNhbXBsZVJhdGUgIT0gODAwMCkgewotICAgICAgICBMT0dXKCJnZXRJbnB1dEJ1ZmZlclNpemUgYmFkIHNhbXBsaW5nIHJhdGU6ICVkIiwgc2FtcGxlUmF0ZSk7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLSAgICBpZiAoZm9ybWF0ICE9IEF1ZGlvU3lzdGVtOjpQQ01fMTZfQklUKSB7Ci0gICAgICAgIExPR1coImdldElucHV0QnVmZmVyU2l6ZSBiYWQgZm9ybWF0OiAlZCIsIGZvcm1hdCk7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLSAgICBpZiAoY2hhbm5lbENvdW50ICE9IDEpIHsKLSAgICAgICAgTE9HVygiZ2V0SW5wdXRCdWZmZXJTaXplIGJhZCBjaGFubmVsIGNvdW50OiAlZCIsIGNoYW5uZWxDb3VudCk7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIHJldHVybiAzMjA7Ci19Ci0KLXN0YXR1c190IEF1ZGlvSGFyZHdhcmVCYXNlOjpkdW1wU3RhdGUoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKQotewotICAgIGNvbnN0IHNpemVfdCBTSVpFID0gMjU2OwotICAgIGNoYXIgYnVmZmVyW1NJWkVdOwotICAgIFN0cmluZzggcmVzdWx0OwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIkF1ZGlvSGFyZHdhcmVCYXNlOjpkdW1wU3RhdGVcbiIpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJcdG1Nb2RlOiAlZFxuIiwgbU1vZGUpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICBmb3IgKGludCBpID0gMCwgbiA9IEF1ZGlvU3lzdGVtOjpOVU1fTU9ERVM7IGkgPCBuOyArK2kpIHsKLSAgICAgICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiXHRtUm91dGVzWyVkXTogJWRcbiIsIGksIG1Sb3V0ZXNbaV0pOwotICAgICAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgfQotICAgIDo6d3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgZHVtcChmZCwgYXJncyk7ICAvLyBEdW1wIHRoZSBzdGF0ZSBvZiB0aGUgY29uY3JldGUgY2hpbGQuCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0hhcmR3YXJlU3R1Yi5jcHAgYi9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb0hhcmR3YXJlU3R1Yi5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGIxM2NiMWMuLjAwMDAwMDAKLS0tIGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9IYXJkd2FyZVN0dWIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTg1ICswLDAgQEAKLS8qIC8vZGV2aWNlL3NlcnZlcnMvQXVkaW9GbGluZ2VyL0F1ZGlvSGFyZHdhcmVTdHViLmNwcAotKioKLSoqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0KLSNpbmNsdWRlICJBdWRpb0hhcmR3YXJlU3R1Yi5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotQXVkaW9IYXJkd2FyZVN0dWI6OkF1ZGlvSGFyZHdhcmVTdHViKCkgOiBtTWljTXV0ZShmYWxzZSkKLXsKLX0KLQotQXVkaW9IYXJkd2FyZVN0dWI6On5BdWRpb0hhcmR3YXJlU3R1YigpCi17Ci19Ci0KLXN0YXR1c190IEF1ZGlvSGFyZHdhcmVTdHViOjppbml0Q2hlY2soKQotewotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotQXVkaW9TdHJlYW1PdXQqIEF1ZGlvSGFyZHdhcmVTdHViOjpvcGVuT3V0cHV0U3RyZWFtKAotICAgICAgICBpbnQgZm9ybWF0LCBpbnQgY2hhbm5lbENvdW50LCB1aW50MzJfdCBzYW1wbGVSYXRlLCBzdGF0dXNfdCAqc3RhdHVzKQotewotICAgIEF1ZGlvU3RyZWFtT3V0U3R1Yiogb3V0ID0gbmV3IEF1ZGlvU3RyZWFtT3V0U3R1YigpOwotICAgIHN0YXR1c190IGxTdGF0dXMgPSBvdXQtPnNldChmb3JtYXQsIGNoYW5uZWxDb3VudCwgc2FtcGxlUmF0ZSk7Ci0gICAgaWYgKHN0YXR1cykgewotICAgICAgICAqc3RhdHVzID0gbFN0YXR1czsKLSAgICB9Ci0gICAgaWYgKGxTdGF0dXMgPT0gTk9fRVJST1IpCi0gICAgICAgIHJldHVybiBvdXQ7Ci0gICAgZGVsZXRlIG91dDsKLSAgICByZXR1cm4gMDsKLX0KLQotQXVkaW9TdHJlYW1JbiogQXVkaW9IYXJkd2FyZVN0dWI6Om9wZW5JbnB1dFN0cmVhbSgKLSAgICAgICAgaW50IGZvcm1hdCwgaW50IGNoYW5uZWxDb3VudCwgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgc3RhdHVzX3QgKnN0YXR1cywgQXVkaW9TeXN0ZW06OmF1ZGlvX2luX2Fjb3VzdGljcyBhY291c3RpY3MpCi17Ci0gICAgQXVkaW9TdHJlYW1JblN0dWIqIGluID0gbmV3IEF1ZGlvU3RyZWFtSW5TdHViKCk7Ci0gICAgc3RhdHVzX3QgbFN0YXR1cyA9IGluLT5zZXQoZm9ybWF0LCBjaGFubmVsQ291bnQsIHNhbXBsZVJhdGUsIGFjb3VzdGljcyk7Ci0gICAgaWYgKHN0YXR1cykgewotICAgICAgICAqc3RhdHVzID0gbFN0YXR1czsKLSAgICB9Ci0gICAgaWYgKGxTdGF0dXMgPT0gTk9fRVJST1IpCi0gICAgICAgIHJldHVybiBpbjsKLSAgICBkZWxldGUgaW47Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXN0YXR1c190IEF1ZGlvSGFyZHdhcmVTdHViOjpzZXRWb2ljZVZvbHVtZShmbG9hdCB2b2x1bWUpCi17Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBBdWRpb0hhcmR3YXJlU3R1Yjo6c2V0TWFzdGVyVm9sdW1lKGZsb2F0IHZvbHVtZSkKLXsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IEF1ZGlvSGFyZHdhcmVTdHViOjpkdW1wSW50ZXJuYWxzKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBjb25zdCBzaXplX3QgU0laRSA9IDI1NjsKLSAgICBjaGFyIGJ1ZmZlcltTSVpFXTsKLSAgICBTdHJpbmc4IHJlc3VsdDsKLSAgICByZXN1bHQuYXBwZW5kKCJBdWRpb0hhcmR3YXJlU3R1Yjo6ZHVtcEludGVybmFsc1xuIik7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiXHRtTWljTXV0ZTogJXNcbiIsIG1NaWNNdXRlPyAidHJ1ZSI6ICJmYWxzZSIpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICA6OndyaXRlKGZkLCByZXN1bHQuc3RyaW5nKCksIHJlc3VsdC5zaXplKCkpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9IYXJkd2FyZVN0dWI6OmR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKQotewotICAgIGR1bXBJbnRlcm5hbHMoZmQsIGFyZ3MpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0dXNfdCBBdWRpb1N0cmVhbU91dFN0dWI6OnNldChpbnQgZm9ybWF0LCBpbnQgY2hhbm5lbHMsIHVpbnQzMl90IHJhdGUpCi17Ci0gICAgLy8gZml4IHVwIGRlZmF1bHRzCi0gICAgaWYgKGZvcm1hdCA9PSAwKSBmb3JtYXQgPSBBdWRpb1N5c3RlbTo6UENNXzE2X0JJVDsKLSAgICBpZiAoY2hhbm5lbHMgPT0gMCkgY2hhbm5lbHMgPSBjaGFubmVsQ291bnQoKTsKLSAgICBpZiAocmF0ZSA9PSAwKSByYXRlID0gc2FtcGxlUmF0ZSgpOwotCi0gICAgaWYgKChmb3JtYXQgPT0gQXVkaW9TeXN0ZW06OlBDTV8xNl9CSVQpICYmCi0gICAgICAgICAgICAoY2hhbm5lbHMgPT0gY2hhbm5lbENvdW50KCkpICYmCi0gICAgICAgICAgICAocmF0ZSA9PSBzYW1wbGVSYXRlKCkpKQotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgcmV0dXJuIEJBRF9WQUxVRTsKLX0KLQotc3NpemVfdCBBdWRpb1N0cmVhbU91dFN0dWI6OndyaXRlKGNvbnN0IHZvaWQqIGJ1ZmZlciwgc2l6ZV90IGJ5dGVzKQotewotICAgIC8vIGZha2UgdGltaW5nIGZvciBhdWRpbyBvdXRwdXQKLSAgICB1c2xlZXAoYnl0ZXMgKiAxMDAwMDAwIC8gc2l6ZW9mKGludDE2X3QpIC8gY2hhbm5lbENvdW50KCkgLyBzYW1wbGVSYXRlKCkpOwotICAgIHJldHVybiBieXRlczsKLX0KLQotc3RhdHVzX3QgQXVkaW9TdHJlYW1PdXRTdHViOjpzdGFuZGJ5KCkKLXsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IEF1ZGlvU3RyZWFtT3V0U3R1Yjo6ZHVtcChpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpCi17Ci0gICAgY29uc3Qgc2l6ZV90IFNJWkUgPSAyNTY7Ci0gICAgY2hhciBidWZmZXJbU0laRV07Ci0gICAgU3RyaW5nOCByZXN1bHQ7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiQXVkaW9TdHJlYW1PdXRTdHViOjpkdW1wXG4iKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJcdHNhbXBsZSByYXRlOiAlZFxuIiwgc2FtcGxlUmF0ZSgpKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJcdGJ1ZmZlciBzaXplOiAlZFxuIiwgYnVmZmVyU2l6ZSgpKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJcdGNoYW5uZWwgY291bnQ6ICVkXG4iLCBjaGFubmVsQ291bnQoKSk7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiXHRmb3JtYXQ6ICVkXG4iLCBmb3JtYXQoKSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIDo6d3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXR1c190IEF1ZGlvU3RyZWFtSW5TdHViOjpzZXQoaW50IGZvcm1hdCwgaW50IGNoYW5uZWxzLCB1aW50MzJfdCByYXRlLAotCQkJCUF1ZGlvU3lzdGVtOjphdWRpb19pbl9hY291c3RpY3MgYWNvdXN0aWNzKQotewotICAgIGlmICgoZm9ybWF0ID09IEF1ZGlvU3lzdGVtOjpQQ01fMTZfQklUKSAmJgotICAgICAgICAgICAgKGNoYW5uZWxzID09IGNoYW5uZWxDb3VudCgpKSAmJgotICAgICAgICAgICAgKHJhdGUgPT0gc2FtcGxlUmF0ZSgpKSkKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIHJldHVybiBCQURfVkFMVUU7Ci19Ci0KLXNzaXplX3QgQXVkaW9TdHJlYW1JblN0dWI6OnJlYWQodm9pZCogYnVmZmVyLCBzc2l6ZV90IGJ5dGVzKQotewotICAgIC8vIGZha2UgdGltaW5nIGZvciBhdWRpbyBpbnB1dAotICAgIHVzbGVlcChieXRlcyAqIDEwMDAwMDAgLyBzaXplb2YoaW50MTZfdCkgLyBjaGFubmVsQ291bnQoKSAvIHNhbXBsZVJhdGUoKSk7Ci0gICAgbWVtc2V0KGJ1ZmZlciwgMCwgYnl0ZXMpOwotICAgIHJldHVybiBieXRlczsKLX0KLQotc3RhdHVzX3QgQXVkaW9TdHJlYW1JblN0dWI6OmR1bXAoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKQotewotICAgIGNvbnN0IHNpemVfdCBTSVpFID0gMjU2OwotICAgIGNoYXIgYnVmZmVyW1NJWkVdOwotICAgIFN0cmluZzggcmVzdWx0OwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIkF1ZGlvU3RyZWFtSW5TdHViOjpkdW1wXG4iKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiXHRzYW1wbGUgcmF0ZTogJWRcbiIsIHNhbXBsZVJhdGUoKSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIlx0YnVmZmVyIHNpemU6ICVkXG4iLCBidWZmZXJTaXplKCkpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJcdGNoYW5uZWwgY291bnQ6ICVkXG4iLCBjaGFubmVsQ291bnQoKSk7Ci0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIlx0Zm9ybWF0OiAlZFxuIiwgZm9ybWF0KCkpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICA6OndyaXRlKGZkLCByZXN1bHQuc3RyaW5nKCksIHJlc3VsdC5zaXplKCkpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9IYXJkd2FyZVN0dWIuaCBiL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvSGFyZHdhcmVTdHViLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGQ0MDY0MjQuLjAwMDAwMDAKLS0tIGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9IYXJkd2FyZVN0dWIuaAorKysgL2Rldi9udWxsCkBAIC0xLDEwMCArMCwwIEBACi0vKiAvL2RldmljZS9zZXJ2ZXJzL0F1ZGlvRmxpbmdlci9BdWRpb0hhcmR3YXJlU3R1Yi5oCi0qKgotKiogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9BVURJT19IQVJEV0FSRV9TVFVCX0gKLSNkZWZpbmUgQU5EUk9JRF9BVURJT19IQVJEV0FSRV9TVFVCX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8aGFyZHdhcmVfbGVnYWN5L0F1ZGlvSGFyZHdhcmVCYXNlLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBBdWRpb1N0cmVhbU91dFN0dWIgOiBwdWJsaWMgQXVkaW9TdHJlYW1PdXQgewotcHVibGljOgotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0KGludCBmb3JtYXQsIGludCBjaGFubmVsQ291bnQsIHVpbnQzMl90IHNhbXBsZVJhdGUpOwotICAgIHZpcnR1YWwgdWludDMyX3QgICAgc2FtcGxlUmF0ZSgpIGNvbnN0IHsgcmV0dXJuIDQ0MTAwOyB9Ci0gICAgdmlydHVhbCBzaXplX3QgICAgICBidWZmZXJTaXplKCkgY29uc3QgeyByZXR1cm4gNDA5NjsgfQotICAgIHZpcnR1YWwgaW50ICAgICAgICAgY2hhbm5lbENvdW50KCkgY29uc3QgeyByZXR1cm4gMjsgfQotICAgIHZpcnR1YWwgaW50ICAgICAgICAgZm9ybWF0KCkgY29uc3QgeyByZXR1cm4gQXVkaW9TeXN0ZW06OlBDTV8xNl9CSVQ7IH0KLSAgICB2aXJ0dWFsIHVpbnQzMl90ICAgIGxhdGVuY3koKSBjb25zdCB7IHJldHVybiAwOyB9Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRWb2x1bWUoZmxvYXQgdm9sdW1lKSB7IHJldHVybiBOT19FUlJPUjsgfQotICAgIHZpcnR1YWwgc3NpemVfdCAgICAgd3JpdGUoY29uc3Qgdm9pZCogYnVmZmVyLCBzaXplX3QgYnl0ZXMpOwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc3RhbmRieSgpOwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgZHVtcChpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpOwotfTsKLQotY2xhc3MgQXVkaW9TdHJlYW1JblN0dWIgOiBwdWJsaWMgQXVkaW9TdHJlYW1JbiB7Ci1wdWJsaWM6Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXQoaW50IGZvcm1hdCwgaW50IGNoYW5uZWxDb3VudCwgdWludDMyX3Qgc2FtcGxlUmF0ZSwgQXVkaW9TeXN0ZW06OmF1ZGlvX2luX2Fjb3VzdGljcyBhY291c3RpY3MpOwotICAgIHZpcnR1YWwgdWludDMyX3QgICAgc2FtcGxlUmF0ZSgpIGNvbnN0IHsgcmV0dXJuIDgwMDA7IH0KLSAgICB2aXJ0dWFsIHNpemVfdCAgICAgIGJ1ZmZlclNpemUoKSBjb25zdCB7IHJldHVybiAzMjA7IH0KLSAgICB2aXJ0dWFsIGludCAgICAgICAgIGNoYW5uZWxDb3VudCgpIGNvbnN0IHsgcmV0dXJuIDE7IH0KLSAgICB2aXJ0dWFsIGludCAgICAgICAgIGZvcm1hdCgpIGNvbnN0IHsgcmV0dXJuIEF1ZGlvU3lzdGVtOjpQQ01fMTZfQklUOyB9Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRHYWluKGZsb2F0IGdhaW4pIHsgcmV0dXJuIE5PX0VSUk9SOyB9Ci0gICAgdmlydHVhbCBzc2l6ZV90ICAgICByZWFkKHZvaWQqIGJ1ZmZlciwgc3NpemVfdCBieXRlcyk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzdGFuZGJ5KCkgeyByZXR1cm4gTk9fRVJST1I7IH0KLX07Ci0KLWNsYXNzIEF1ZGlvSGFyZHdhcmVTdHViIDogcHVibGljICBBdWRpb0hhcmR3YXJlQmFzZQotewotcHVibGljOgotICAgICAgICAgICAgICAgICAgICAgICAgQXVkaW9IYXJkd2FyZVN0dWIoKTsKLSAgICB2aXJ0dWFsICAgICAgICAgICAgIH5BdWRpb0hhcmR3YXJlU3R1YigpOwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgaW5pdENoZWNrKCk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBzZXRWb2ljZVZvbHVtZShmbG9hdCB2b2x1bWUpOwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0TWFzdGVyVm9sdW1lKGZsb2F0IHZvbHVtZSk7Ci0KLSAgICAvLyBtaWMgbXV0ZQotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgc2V0TWljTXV0ZShib29sIHN0YXRlKSB7IG1NaWNNdXRlID0gc3RhdGU7ICByZXR1cm4gIE5PX0VSUk9SOyB9Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBnZXRNaWNNdXRlKGJvb2wqIHN0YXRlKSB7ICpzdGF0ZSA9IG1NaWNNdXRlIDsgcmV0dXJuIE5PX0VSUk9SOyB9Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHNldFBhcmFtZXRlcihjb25zdCBjaGFyKiBrZXksIGNvbnN0IGNoYXIqIHZhbHVlKQotICAgICAgICAgICAgeyByZXR1cm4gTk9fRVJST1I7IH0KLQotICAgIC8vIGNyZWF0ZSBJL08gc3RyZWFtcwotICAgIHZpcnR1YWwgQXVkaW9TdHJlYW1PdXQqIG9wZW5PdXRwdXRTdHJlYW0oCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBmb3JtYXQ9MCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGNoYW5uZWxDb3VudD0wLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBzYW1wbGVSYXRlPTAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXR1c190ICpzdGF0dXM9MCk7Ci0KLSAgICB2aXJ0dWFsIEF1ZGlvU3RyZWFtSW4qIG9wZW5JbnB1dFN0cmVhbSgKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGZvcm1hdCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGNoYW5uZWxDb3VudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3Qgc2FtcGxlUmF0ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3RhdHVzX3QgKnN0YXR1cywKLQkJCQlBdWRpb1N5c3RlbTo6YXVkaW9faW5fYWNvdXN0aWNzIGFjb3VzdGljcyk7Ci0KLXByb3RlY3RlZDoKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIGRvUm91dGluZygpIHsgcmV0dXJuIE5PX0VSUk9SOyB9Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICBkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncyk7Ci0KLSAgICAgICAgICAgIGJvb2wgICAgICAgIG1NaWNNdXRlOwotcHJpdmF0ZToKLSAgICBzdGF0dXNfdCAgICAgICAgICAgIGR1bXBJbnRlcm5hbHMoaW50IGZkLCBjb25zdCBWZWN0b3I8U3RyaW5nMTY+JiBhcmdzKTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfQVVESU9fSEFSRFdBUkVfU1RVQl9ICmRpZmYgLS1naXQgYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb01peGVyLmNwcCBiL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvTWl4ZXIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiMDM0NjdmLi4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvTWl4ZXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsOTEzICswLDAgQEAKLS8qIC8vZGV2aWNlL2luY2x1ZGUvc2VydmVyL0F1ZGlvRmxpbmdlci9BdWRpb01peGVyLmNwcAotKioKLSoqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jZGVmaW5lIExPR19UQUcgIkF1ZGlvTWl4ZXIiCi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSAiQXVkaW9NaXhlci5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyBpbmxpbmUgaW50MTZfdCBjbGFtcDE2KGludDMyX3Qgc2FtcGxlKQotewotICAgIGlmICgoc2FtcGxlPj4xNSkgXiAoc2FtcGxlPj4zMSkpCi0gICAgICAgIHNhbXBsZSA9IDB4N0ZGRiBeIChzYW1wbGU+PjMxKTsKLSAgICByZXR1cm4gc2FtcGxlOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUF1ZGlvTWl4ZXI6OkF1ZGlvTWl4ZXIoc2l6ZV90IGZyYW1lQ291bnQsIHVpbnQzMl90IHNhbXBsZVJhdGUpCi0gICAgOiAgIG1BY3RpdmVUcmFjaygwKSwgbVRyYWNrTmFtZXMoMCksIG1TYW1wbGVSYXRlKHNhbXBsZVJhdGUpCi17Ci0gICAgbVN0YXRlLmVuYWJsZWRUcmFja3M9IDA7Ci0gICAgbVN0YXRlLm5lZWRzQ2hhbmdlZCA9IDA7Ci0gICAgbVN0YXRlLmZyYW1lQ291bnQgICA9IGZyYW1lQ291bnQ7Ci0gICAgbVN0YXRlLm91dHB1dFRlbXAgICA9IDA7Ci0gICAgbVN0YXRlLnJlc2FtcGxlVGVtcCA9IDA7Ci0gICAgbVN0YXRlLmhvb2sgICAgICAgICA9IHByb2Nlc3NfX25vcDsKLSAgICB0cmFja190KiB0ID0gbVN0YXRlLnRyYWNrczsKLSAgICBmb3IgKGludCBpPTAgOyBpPDMyIDsgaSsrKSB7Ci0gICAgICAgIHQtPm5lZWRzID0gMDsKLSAgICAgICAgdC0+dm9sdW1lWzBdID0gVU5JVFlfR0FJTjsKLSAgICAgICAgdC0+dm9sdW1lWzFdID0gVU5JVFlfR0FJTjsKLSAgICAgICAgdC0+dm9sdW1lSW5jWzBdID0gMDsKLSAgICAgICAgdC0+dm9sdW1lSW5jWzFdID0gMDsKLSAgICAgICAgdC0+Y2hhbm5lbENvdW50ID0gMjsKLSAgICAgICAgdC0+ZW5hYmxlZCA9IDA7Ci0gICAgICAgIHQtPmZvcm1hdCA9IDE2OwotICAgICAgICB0LT5idWZmZXIucmF3ID0gMDsKLSAgICAgICAgdC0+YnVmZmVyUHJvdmlkZXIgPSAwOwotICAgICAgICB0LT5ob29rID0gMDsKLSAgICAgICAgdC0+cmVzYW1wbGVyID0gMDsKLSAgICAgICAgdC0+c2FtcGxlUmF0ZSA9IG1TYW1wbGVSYXRlOwotICAgICAgICB0LT5pbiA9IDA7Ci0gICAgICAgIHQrKzsKLSAgICB9Ci19Ci0KLSBBdWRpb01peGVyOjp+QXVkaW9NaXhlcigpCi0gewotICAgICB0cmFja190KiB0ID0gbVN0YXRlLnRyYWNrczsKLSAgICAgZm9yIChpbnQgaT0wIDsgaTwzMiA7IGkrKykgewotICAgICAgICAgZGVsZXRlIHQtPnJlc2FtcGxlcjsKLSAgICAgICAgIHQrKzsKLSAgICAgfQotICAgICBkZWxldGUgW10gbVN0YXRlLm91dHB1dFRlbXA7Ci0gICAgIGRlbGV0ZSBbXSBtU3RhdGUucmVzYW1wbGVUZW1wOwotIH0KLQotIGludCBBdWRpb01peGVyOjpnZXRUcmFja05hbWUoKQotIHsKLSAgICB1aW50MzJfdCBuYW1lcyA9IG1UcmFja05hbWVzOwotICAgIHVpbnQzMl90IG1hc2sgPSAxOwotICAgIGludCBuID0gMDsKLSAgICB3aGlsZSAobmFtZXMgJiBtYXNrKSB7Ci0gICAgICAgIG1hc2sgPDw9IDE7Ci0gICAgICAgIG4rKzsKLSAgICB9Ci0gICAgaWYgKG1hc2spIHsKLSAgICAgICAgTE9HVigiYWRkIHRyYWNrICglZCkiLCBuKTsKLSAgICAgICAgbVRyYWNrTmFtZXMgfD0gbWFzazsKLSAgICAgICAgcmV0dXJuIFRSQUNLMCArIG47Ci0gICAgfQotICAgIHJldHVybiAtMTsKLSB9Ci0KLSB2b2lkIEF1ZGlvTWl4ZXI6OmludmFsaWRhdGVTdGF0ZSh1aW50MzJfdCBtYXNrKQotIHsKLSAgICBpZiAobWFzaykgewotICAgICAgICBtU3RhdGUubmVlZHNDaGFuZ2VkIHw9IG1hc2s7Ci0gICAgICAgIG1TdGF0ZS5ob29rID0gcHJvY2Vzc19fdmFsaWRhdGU7Ci0gICAgfQotIH0KLQotIHZvaWQgQXVkaW9NaXhlcjo6ZGVsZXRlVHJhY2tOYW1lKGludCBuYW1lKQotIHsKLSAgICBuYW1lIC09IFRSQUNLMDsKLSAgICBpZiAodWludDMyX3QobmFtZSkgPCBNQVhfTlVNX1RSQUNLUykgewotICAgICAgICBMT0dWKCJkZWxldGVUcmFja05hbWUoJWQpIiwgbmFtZSk7Ci0gICAgICAgIHRyYWNrX3QmIHRyYWNrKG1TdGF0ZS50cmFja3NbIG5hbWUgXSk7Ci0gICAgICAgIGlmICh0cmFjay5lbmFibGVkICE9IDApIHsKLSAgICAgICAgICAgIHRyYWNrLmVuYWJsZWQgPSAwOwotICAgICAgICAgICAgaW52YWxpZGF0ZVN0YXRlKDE8PG5hbWUpOwotICAgICAgICB9Ci0gICAgICAgIGlmICh0cmFjay5yZXNhbXBsZXIpIHsKLSAgICAgICAgICAgIC8vIGRlbGV0ZSAgdGhlIHJlc2FtcGxlcgotICAgICAgICAgICAgZGVsZXRlIHRyYWNrLnJlc2FtcGxlcjsKLSAgICAgICAgICAgIHRyYWNrLnJlc2FtcGxlciA9IDA7Ci0gICAgICAgICAgICB0cmFjay5zYW1wbGVSYXRlID0gbVNhbXBsZVJhdGU7Ci0gICAgICAgICAgICBpbnZhbGlkYXRlU3RhdGUoMTw8bmFtZSk7Ci0gICAgICAgIH0KLSAgICAgICAgdHJhY2sudm9sdW1lSW5jWzBdID0gMDsKLSAgICAgICAgdHJhY2sudm9sdW1lSW5jWzFdID0gMDsKLSAgICAgICAgbVRyYWNrTmFtZXMgJj0gfigxPDxuYW1lKTsKLSAgICB9Ci0gfQotCi1zdGF0dXNfdCBBdWRpb01peGVyOjplbmFibGUoaW50IG5hbWUpCi17Ci0gICAgc3dpdGNoIChuYW1lKSB7Ci0gICAgICAgIGNhc2UgTUlYSU5HOiB7Ci0gICAgICAgICAgICBpZiAobVN0YXRlLnRyYWNrc1sgbUFjdGl2ZVRyYWNrIF0uZW5hYmxlZCAhPSAxKSB7Ci0gICAgICAgICAgICAgICAgbVN0YXRlLnRyYWNrc1sgbUFjdGl2ZVRyYWNrIF0uZW5hYmxlZCA9IDE7Ci0gICAgICAgICAgICAgICAgTE9HVigiZW5hYmxlKCVkKSIsIG1BY3RpdmVUcmFjayk7Ci0gICAgICAgICAgICAgICAgaW52YWxpZGF0ZVN0YXRlKDE8PG1BY3RpdmVUcmFjayk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gTkFNRV9OT1RfRk9VTkQ7Ci0gICAgfQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9NaXhlcjo6ZGlzYWJsZShpbnQgbmFtZSkKLXsKLSAgICBzd2l0Y2ggKG5hbWUpIHsKLSAgICAgICAgY2FzZSBNSVhJTkc6IHsKLSAgICAgICAgICAgIGlmIChtU3RhdGUudHJhY2tzWyBtQWN0aXZlVHJhY2sgXS5lbmFibGVkICE9IDApIHsKLSAgICAgICAgICAgICAgICBtU3RhdGUudHJhY2tzWyBtQWN0aXZlVHJhY2sgXS5lbmFibGVkID0gMDsKLSAgICAgICAgICAgICAgICBMT0dWKCJkaXNhYmxlKCVkKSIsIG1BY3RpdmVUcmFjayk7Ci0gICAgICAgICAgICAgICAgaW52YWxpZGF0ZVN0YXRlKDE8PG1BY3RpdmVUcmFjayk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gTkFNRV9OT1RfRk9VTkQ7Ci0gICAgfQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9NaXhlcjo6c2V0QWN0aXZlVHJhY2soaW50IHRyYWNrKQotewotICAgIGlmICh1aW50MzJfdCh0cmFjay1UUkFDSzApID49IE1BWF9OVU1fVFJBQ0tTKSB7Ci0gICAgICAgIHJldHVybiBCQURfVkFMVUU7Ci0gICAgfQotICAgIG1BY3RpdmVUcmFjayA9IHRyYWNrIC0gVFJBQ0swOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQXVkaW9NaXhlcjo6c2V0UGFyYW1ldGVyKGludCB0YXJnZXQsIGludCBuYW1lLCBpbnQgdmFsdWUpCi17Ci0gICAgc3dpdGNoICh0YXJnZXQpIHsKLSAgICBjYXNlIFRSQUNLOgotICAgICAgICBpZiAobmFtZSA9PSBDSEFOTkVMX0NPVU5UKSB7Ci0gICAgICAgICAgICBpZiAoKHVpbnQzMl90KHZhbHVlKSA8PSBNQVhfTlVNX0NIQU5ORUxTKSAmJiAodmFsdWUpKSB7Ci0gICAgICAgICAgICAgICAgaWYgKG1TdGF0ZS50cmFja3NbIG1BY3RpdmVUcmFjayBdLmNoYW5uZWxDb3VudCAhPSB2YWx1ZSkgewotICAgICAgICAgICAgICAgICAgICBtU3RhdGUudHJhY2tzWyBtQWN0aXZlVHJhY2sgXS5jaGFubmVsQ291bnQgPSB2YWx1ZTsKLSAgICAgICAgICAgICAgICAgICAgTE9HVigic2V0UGFyYW1ldGVyKFRSQUNLLCBDSEFOTkVMX0NPVU5ULCAlZCkiLCB2YWx1ZSk7Ci0gICAgICAgICAgICAgICAgICAgIGludmFsaWRhdGVTdGF0ZSgxPDxtQWN0aXZlVHJhY2spOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBSRVNBTVBMRToKLSAgICAgICAgaWYgKG5hbWUgPT0gU0FNUExFX1JBVEUpIHsKLSAgICAgICAgICAgIGlmICh2YWx1ZSA+IDApIHsKLSAgICAgICAgICAgICAgICB0cmFja190JiB0cmFjayA9IG1TdGF0ZS50cmFja3NbIG1BY3RpdmVUcmFjayBdOwotICAgICAgICAgICAgICAgIGlmICh0cmFjay5zZXRSZXNhbXBsZXIodWludDMyX3QodmFsdWUpLCBtU2FtcGxlUmF0ZSkpIHsKLSAgICAgICAgICAgICAgICAgICAgTE9HVigic2V0UGFyYW1ldGVyKFJFU0FNUExFLCBTQU1QTEVfUkFURSwgJXUpIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCh2YWx1ZSkpOwotICAgICAgICAgICAgICAgICAgICBpbnZhbGlkYXRlU3RhdGUoMTw8bUFjdGl2ZVRyYWNrKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgUkFNUF9WT0xVTUU6Ci0gICAgY2FzZSBWT0xVTUU6Ci0gICAgICAgIGlmICgodWludDMyX3QobmFtZS1WT0xVTUUwKSA8IE1BWF9OVU1fQ0hBTk5FTFMpKSB7Ci0gICAgICAgICAgICB0cmFja190JiB0cmFjayA9IG1TdGF0ZS50cmFja3NbIG1BY3RpdmVUcmFjayBdOwotICAgICAgICAgICAgaWYgKHRyYWNrLnZvbHVtZVtuYW1lLVZPTFVNRTBdICE9IHZhbHVlKSB7Ci0gICAgICAgICAgICAgICAgdHJhY2sucHJldlZvbHVtZVtuYW1lLVZPTFVNRTBdID0gdHJhY2sudm9sdW1lW25hbWUtVk9MVU1FMF0gPDwgMTY7Ci0gICAgICAgICAgICAgICAgdHJhY2sudm9sdW1lW25hbWUtVk9MVU1FMF0gPSB2YWx1ZTsKLSAgICAgICAgICAgICAgICBpZiAodGFyZ2V0ID09IFZPTFVNRSkgewotICAgICAgICAgICAgICAgICAgICB0cmFjay5wcmV2Vm9sdW1lW25hbWUtVk9MVU1FMF0gPSB2YWx1ZSA8PCAxNjsKLSAgICAgICAgICAgICAgICAgICAgdHJhY2sudm9sdW1lSW5jW25hbWUtVk9MVU1FMF0gPSAwOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGludDMyX3QgZCA9ICh2YWx1ZTw8MTYpIC0gdHJhY2sucHJldlZvbHVtZVtuYW1lLVZPTFVNRTBdOwotICAgICAgICAgICAgICAgICAgICBpbnQzMl90IHZvbEluYyA9IGQgLyBpbnQzMl90KG1TdGF0ZS5mcmFtZUNvdW50KTsKLSAgICAgICAgICAgICAgICAgICAgdHJhY2sudm9sdW1lSW5jW25hbWUtVk9MVU1FMF0gPSB2b2xJbmM7Ci0gICAgICAgICAgICAgICAgICAgIGlmICh2b2xJbmMgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgdHJhY2sucHJldlZvbHVtZVtuYW1lLVZPTFVNRTBdID0gdmFsdWUgPDwgMTY7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaW52YWxpZGF0ZVN0YXRlKDE8PG1BY3RpdmVUcmFjayk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0KLSAgICAgICAgYnJlYWs7Ci0gICAgfQotICAgIHJldHVybiBCQURfVkFMVUU7Ci19Ci0KLWJvb2wgQXVkaW9NaXhlcjo6dHJhY2tfdDo6c2V0UmVzYW1wbGVyKHVpbnQzMl90IHZhbHVlLCB1aW50MzJfdCBkZXZTYW1wbGVSYXRlKQotewotICAgIGlmICh2YWx1ZSE9ZGV2U2FtcGxlUmF0ZSB8fCByZXNhbXBsZXIpIHsKLSAgICAgICAgaWYgKHNhbXBsZVJhdGUgIT0gdmFsdWUpIHsKLSAgICAgICAgICAgIHNhbXBsZVJhdGUgPSB2YWx1ZTsKLSAgICAgICAgICAgIGlmIChyZXNhbXBsZXIgPT0gMCkgewotICAgICAgICAgICAgICAgIHJlc2FtcGxlciA9IEF1ZGlvUmVzYW1wbGVyOjpjcmVhdGUoCi0gICAgICAgICAgICAgICAgICAgICAgICBmb3JtYXQsIGNoYW5uZWxDb3VudCwgZGV2U2FtcGxlUmF0ZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0KLWJvb2wgQXVkaW9NaXhlcjo6dHJhY2tfdDo6ZG9lc1Jlc2FtcGxlKCkgY29uc3QKLXsKLSAgICByZXR1cm4gcmVzYW1wbGVyICE9IDA7Ci19Ci0KLWlubGluZQotdm9pZCBBdWRpb01peGVyOjp0cmFja190OjphZGp1c3RWb2x1bWVSYW1wKCkKLXsKLSAgICBmb3IgKGludCBpPTAgOyBpPDIgOyBpKyspIHsKLSAgICAgICAgaWYgKCgodm9sdW1lSW5jW2ldPjApICYmICgoKHByZXZWb2x1bWVbaV0rdm9sdW1lSW5jW2ldKT4+MTYpID49IHZvbHVtZVtpXSkpIHx8Ci0gICAgICAgICAgICAoKHZvbHVtZUluY1tpXTwwKSAmJiAoKChwcmV2Vm9sdW1lW2ldK3ZvbHVtZUluY1tpXSk+PjE2KSA8PSB2b2x1bWVbaV0pKSkgewotICAgICAgICAgICAgdm9sdW1lSW5jW2ldID0gMDsKLSAgICAgICAgICAgIHByZXZWb2x1bWVbaV0gPSB2b2x1bWVbaV08PDE2OwotICAgICAgICB9Ci0gICAgfQotfQotCi0KLXN0YXR1c190IEF1ZGlvTWl4ZXI6OnNldEJ1ZmZlclByb3ZpZGVyKEF1ZGlvQnVmZmVyUHJvdmlkZXIqIGJ1ZmZlcikKLXsKLSAgICBtU3RhdGUudHJhY2tzWyBtQWN0aXZlVHJhY2sgXS5idWZmZXJQcm92aWRlciA9IGJ1ZmZlcjsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLQotCi12b2lkIEF1ZGlvTWl4ZXI6OnByb2Nlc3Modm9pZCogb3V0cHV0KQotewotICAgIG1TdGF0ZS5ob29rKCZtU3RhdGUsIG91dHB1dCk7Ci19Ci0KLQotdm9pZCBBdWRpb01peGVyOjpwcm9jZXNzX192YWxpZGF0ZShzdGF0ZV90KiBzdGF0ZSwgdm9pZCogb3V0cHV0KQotewotICAgIExPR1dfSUYoIXN0YXRlLT5uZWVkc0NoYW5nZWQsCi0gICAgICAgICJpbiBwcm9jZXNzX192YWxpZGF0ZSgpIGJ1dCBub3RoaW5nJ3MgaW52YWxpZCIpOwotCi0gICAgdWludDMyX3QgY2hhbmdlZCA9IHN0YXRlLT5uZWVkc0NoYW5nZWQ7Ci0gICAgc3RhdGUtPm5lZWRzQ2hhbmdlZCA9IDA7IC8vIGNsZWFyIHRoZSB2YWxpZGF0aW9uIGZsYWcKLQotICAgIC8vIHJlY29tcHV0ZSB3aGljaCB0cmFja3MgYXJlIGVuYWJsZWQgLyBkaXNhYmxlZAotICAgIHVpbnQzMl90IGVuYWJsZWQgPSAwOwotICAgIHVpbnQzMl90IGRpc2FibGVkID0gMDsKLSAgICB3aGlsZSAoY2hhbmdlZCkgewotICAgICAgICBjb25zdCBpbnQgaSA9IDMxIC0gX19idWlsdGluX2NseihjaGFuZ2VkKTsKLSAgICAgICAgY29uc3QgdWludDMyX3QgbWFzayA9IDE8PGk7Ci0gICAgICAgIGNoYW5nZWQgJj0gfm1hc2s7Ci0gICAgICAgIHRyYWNrX3QmIHQgPSBzdGF0ZS0+dHJhY2tzW2ldOwotICAgICAgICAodC5lbmFibGVkID8gZW5hYmxlZCA6IGRpc2FibGVkKSB8PSBtYXNrOwotICAgIH0KLSAgICBzdGF0ZS0+ZW5hYmxlZFRyYWNrcyAmPSB+ZGlzYWJsZWQ7Ci0gICAgc3RhdGUtPmVuYWJsZWRUcmFja3MgfD0gIGVuYWJsZWQ7Ci0KLSAgICAvLyBjb21wdXRlIGV2ZXJ5dGhpbmcgd2UgbmVlZC4uLgotICAgIGludCBjb3VudEFjdGl2ZVRyYWNrcyA9IDA7Ci0gICAgaW50IGFsbDE2Qml0c1N0ZXJlb05vUmVzYW1wbGUgPSAxOwotICAgIGludCByZXNhbXBsaW5nID0gMDsKLSAgICBpbnQgdm9sdW1lUmFtcCA9IDA7Ci0gICAgdWludDMyX3QgZW4gPSBzdGF0ZS0+ZW5hYmxlZFRyYWNrczsKLSAgICB3aGlsZSAoZW4pIHsKLSAgICAgICAgY29uc3QgaW50IGkgPSAzMSAtIF9fYnVpbHRpbl9jbHooZW4pOwotICAgICAgICBlbiAmPSB+KDE8PGkpOwotCi0gICAgICAgIGNvdW50QWN0aXZlVHJhY2tzKys7Ci0gICAgICAgIHRyYWNrX3QmIHQgPSBzdGF0ZS0+dHJhY2tzW2ldOwotICAgICAgICB1aW50MzJfdCBuID0gMDsKLSAgICAgICAgbiB8PSBORUVEU19DSEFOTkVMXzEgKyB0LmNoYW5uZWxDb3VudCAtIDE7Ci0gICAgICAgIG4gfD0gTkVFRFNfRk9STUFUXzE2OwotICAgICAgICBuIHw9IHQuZG9lc1Jlc2FtcGxlKCkgPyBORUVEU19SRVNBTVBMRV9FTkFCTEVEIDogTkVFRFNfUkVTQU1QTEVfRElTQUJMRUQ7Ci0gICAgICAgCi0gICAgICAgIGlmICh0LnZvbHVtZUluY1swXXx0LnZvbHVtZUluY1sxXSkgewotICAgICAgICAgICAgdm9sdW1lUmFtcCA9IDE7Ci0gICAgICAgIH0gZWxzZSBpZiAoIXQuZG9lc1Jlc2FtcGxlKCkgJiYgdC52b2x1bWVSTCA9PSAwKSB7Ci0gICAgICAgICAgICBuIHw9IE5FRURTX01VVEVfRU5BQkxFRDsKLSAgICAgICAgfQotICAgICAgICB0Lm5lZWRzID0gbjsKLQotICAgICAgICBpZiAoKG4gJiBORUVEU19NVVRFX19NQVNLKSA9PSBORUVEU19NVVRFX0VOQUJMRUQpIHsKLSAgICAgICAgICAgIHQuaG9vayA9IHRyYWNrX19ub3A7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpZiAoKG4gJiBORUVEU19SRVNBTVBMRV9fTUFTSykgPT0gTkVFRFNfUkVTQU1QTEVfRU5BQkxFRCkgewotICAgICAgICAgICAgICAgIGFsbDE2Qml0c1N0ZXJlb05vUmVzYW1wbGUgPSAwOwotICAgICAgICAgICAgICAgIHJlc2FtcGxpbmcgPSAxOwotICAgICAgICAgICAgICAgIHQuaG9vayA9IHRyYWNrX19nZW5lcmljUmVzYW1wbGU7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGlmICgobiAmIE5FRURTX0NIQU5ORUxfQ09VTlRfX01BU0spID09IE5FRURTX0NIQU5ORUxfMSl7Ci0gICAgICAgICAgICAgICAgICAgIHQuaG9vayA9IHRyYWNrX18xNkJpdHNNb25vOwotICAgICAgICAgICAgICAgICAgICBhbGwxNkJpdHNTdGVyZW9Ob1Jlc2FtcGxlID0gMDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKChuICYgTkVFRFNfQ0hBTk5FTF9DT1VOVF9fTUFTSykgPT0gTkVFRFNfQ0hBTk5FTF8yKXsKLSAgICAgICAgICAgICAgICAgICAgdC5ob29rID0gdHJhY2tfXzE2Qml0c1N0ZXJlbzsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyBzZWxlY3QgdGhlIHByb2Nlc3NpbmcgaG9va3MKLSAgICBzdGF0ZS0+aG9vayA9IHByb2Nlc3NfX25vcDsKLSAgICBpZiAoY291bnRBY3RpdmVUcmFja3MpIHsKLSAgICAgICAgaWYgKHJlc2FtcGxpbmcpIHsKLSAgICAgICAgICAgIGlmICghc3RhdGUtPm91dHB1dFRlbXApIHsKLSAgICAgICAgICAgICAgICBzdGF0ZS0+b3V0cHV0VGVtcCA9IG5ldyBpbnQzMl90W01BWF9OVU1fQ0hBTk5FTFMgKiBzdGF0ZS0+ZnJhbWVDb3VudF07Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoIXN0YXRlLT5yZXNhbXBsZVRlbXApIHsKLSAgICAgICAgICAgICAgICBzdGF0ZS0+cmVzYW1wbGVUZW1wID0gbmV3IGludDMyX3RbTUFYX05VTV9DSEFOTkVMUyAqIHN0YXRlLT5mcmFtZUNvdW50XTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHN0YXRlLT5ob29rID0gcHJvY2Vzc19fZ2VuZXJpY1Jlc2FtcGxpbmc7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpZiAoc3RhdGUtPm91dHB1dFRlbXApIHsKLSAgICAgICAgICAgICAgICBkZWxldGUgW10gc3RhdGUtPm91dHB1dFRlbXA7Ci0gICAgICAgICAgICAgICAgc3RhdGUtPm91dHB1dFRlbXAgPSAwOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHN0YXRlLT5yZXNhbXBsZVRlbXApIHsKLSAgICAgICAgICAgICAgICBkZWxldGUgW10gc3RhdGUtPnJlc2FtcGxlVGVtcDsKLSAgICAgICAgICAgICAgICBzdGF0ZS0+cmVzYW1wbGVUZW1wID0gMDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHN0YXRlLT5ob29rID0gcHJvY2Vzc19fZ2VuZXJpY05vUmVzYW1wbGluZzsKLSAgICAgICAgICAgIGlmIChhbGwxNkJpdHNTdGVyZW9Ob1Jlc2FtcGxlICYmICF2b2x1bWVSYW1wKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGNvdW50QWN0aXZlVHJhY2tzID09IDEpIHsKLSAgICAgICAgICAgICAgICAgICAgc3RhdGUtPmhvb2sgPSBwcm9jZXNzX19PbmVUcmFjazE2Qml0c1N0ZXJlb05vUmVzYW1wbGluZzsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBMT0dWKCJtaXhlciBjb25maWd1cmF0aW9uIGNoYW5nZTogJWQgYWN0aXZlVHJhY2tzICglMDh4KSAiCi0gICAgICAgICJhbGwxNkJpdHNTdGVyZW9Ob1Jlc2FtcGxlPSVkLCByZXNhbXBsaW5nPSVkLCB2b2x1bWVSYW1wPSVkIiwKLSAgICAgICAgY291bnRBY3RpdmVUcmFja3MsIHN0YXRlLT5lbmFibGVkVHJhY2tzLAotICAgICAgICBhbGwxNkJpdHNTdGVyZW9Ob1Jlc2FtcGxlLCByZXNhbXBsaW5nLCB2b2x1bWVSYW1wKTsKLQotICAgc3RhdGUtPmhvb2soc3RhdGUsIG91dHB1dCk7Ci0KLSAgIC8vIE5vdyB0aGF0IHRoZSB2b2x1bWUgcmFtcCBoYXMgYmVlbiBkb25lLCBzZXQgb3B0aW1hbCBzdGF0ZSBhbmQKLSAgIC8vIHRyYWNrIGhvb2tzIGZvciBzdWJzZXF1ZW50IG1peGVyIHByb2Nlc3MKLSAgIGlmIChjb3VudEFjdGl2ZVRyYWNrcykgewotICAgICAgIGludCBhbGxNdXRlZCA9IDE7Ci0gICAgICAgdWludDMyX3QgZW4gPSBzdGF0ZS0+ZW5hYmxlZFRyYWNrczsKLSAgICAgICB3aGlsZSAoZW4pIHsKLSAgICAgICAgICAgY29uc3QgaW50IGkgPSAzMSAtIF9fYnVpbHRpbl9jbHooZW4pOwotICAgICAgICAgICBlbiAmPSB+KDE8PGkpOwotICAgICAgICAgICB0cmFja190JiB0ID0gc3RhdGUtPnRyYWNrc1tpXTsKLSAgICAgICAgICAgaWYgKCF0LmRvZXNSZXNhbXBsZSgpICYmIHQudm9sdW1lUkwgPT0gMCkKLSAgICAgICAgICAgewotICAgICAgICAgICAgICAgdC5uZWVkcyB8PSBORUVEU19NVVRFX0VOQUJMRUQ7Ci0gICAgICAgICAgICAgICB0Lmhvb2sgPSB0cmFja19fbm9wOwotICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgYWxsTXV0ZWQgPSAwOwotICAgICAgICAgICB9Ci0gICAgICAgfQotICAgICAgIGlmIChhbGxNdXRlZCkgewotICAgICAgICAgICBzdGF0ZS0+aG9vayA9IHByb2Nlc3NfX25vcDsKLSAgICAgICB9IGVsc2UgaWYgKCFyZXNhbXBsaW5nICYmIGFsbDE2Qml0c1N0ZXJlb05vUmVzYW1wbGUpIHsKLSAgICAgICAgICAgaWYgKGNvdW50QWN0aXZlVHJhY2tzID09IDEpIHsKLSAgICAgICAgICAgICAgc3RhdGUtPmhvb2sgPSBwcm9jZXNzX19PbmVUcmFjazE2Qml0c1N0ZXJlb05vUmVzYW1wbGluZzsKLSAgICAgICAgICAgfQotICAgICAgIH0KLSAgIH0KLX0KLQotc3RhdGljIGlubGluZQotaW50MzJfdCBtdWxBZGQoaW50MTZfdCBpbiwgaW50MTZfdCB2LCBpbnQzMl90IGEpCi17Ci0jaWYgZGVmaW5lZChfX2FybV9fKSAmJiAhZGVmaW5lZChfX3RodW1iX18pCi0gICAgaW50MzJfdCBvdXQ7Ci0gICAgYXNtKCAic21sYWJiICVbb3V0XSwgJVtpbl0sICVbdl0sICVbYV0gXG4iCi0gICAgICAgICA6IFtvdXRdIj1yIihvdXQpCi0gICAgICAgICA6IFtpbl0iJXIiKGluKSwgW3ZdInIiKHYpLCBbYV0iciIoYSkKLSAgICAgICAgIDogKTsKLSAgICByZXR1cm4gb3V0OwotI2Vsc2UKLSAgICByZXR1cm4gYSArIGluICogaW50MzJfdCh2KTsKLSNlbmRpZgotfQotCi1zdGF0aWMgaW5saW5lCi1pbnQzMl90IG11bChpbnQxNl90IGluLCBpbnQxNl90IHYpCi17Ci0jaWYgZGVmaW5lZChfX2FybV9fKSAmJiAhZGVmaW5lZChfX3RodW1iX18pCi0gICAgaW50MzJfdCBvdXQ7Ci0gICAgYXNtKCAic211bGJiICVbb3V0XSwgJVtpbl0sICVbdl0gXG4iCi0gICAgICAgICA6IFtvdXRdIj1yIihvdXQpCi0gICAgICAgICA6IFtpbl0iJXIiKGluKSwgW3ZdInIiKHYpCi0gICAgICAgICA6ICk7Ci0gICAgcmV0dXJuIG91dDsKLSNlbHNlCi0gICAgcmV0dXJuIGluICogaW50MzJfdCh2KTsKLSNlbmRpZgotfQotCi1zdGF0aWMgaW5saW5lCi1pbnQzMl90IG11bEFkZFJMKGludCBsZWZ0LCB1aW50MzJfdCBpblJMLCB1aW50MzJfdCB2UkwsIGludDMyX3QgYSkKLXsKLSNpZiBkZWZpbmVkKF9fYXJtX18pICYmICFkZWZpbmVkKF9fdGh1bWJfXykKLSAgICBpbnQzMl90IG91dDsKLSAgICBpZiAobGVmdCkgewotICAgICAgICBhc20oICJzbWxhYmIgJVtvdXRdLCAlW2luUkxdLCAlW3ZSTF0sICVbYV0gXG4iCi0gICAgICAgICAgICAgOiBbb3V0XSI9ciIob3V0KQotICAgICAgICAgICAgIDogW2luUkxdIiVyIihpblJMKSwgW3ZSTF0iciIodlJMKSwgW2FdInIiKGEpCi0gICAgICAgICAgICAgOiApOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIGFzbSggInNtbGF0dCAlW291dF0sICVbaW5STF0sICVbdlJMXSwgJVthXSBcbiIKLSAgICAgICAgICAgICA6IFtvdXRdIj1yIihvdXQpCi0gICAgICAgICAgICAgOiBbaW5STF0iJXIiKGluUkwpLCBbdlJMXSJyIih2UkwpLCBbYV0iciIoYSkKLSAgICAgICAgICAgICA6ICk7Ci0gICAgfQotICAgIHJldHVybiBvdXQ7Ci0jZWxzZQotICAgIGlmIChsZWZ0KSB7Ci0gICAgICAgIHJldHVybiBhICsgaW50MTZfdChpblJMJjB4RkZGRikgKiBpbnQxNl90KHZSTCYweEZGRkYpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHJldHVybiBhICsgaW50MTZfdChpblJMPj4xNikgKiBpbnQxNl90KHZSTD4+MTYpOwotICAgIH0KLSNlbmRpZgotfQotCi1zdGF0aWMgaW5saW5lCi1pbnQzMl90IG11bFJMKGludCBsZWZ0LCB1aW50MzJfdCBpblJMLCB1aW50MzJfdCB2UkwpCi17Ci0jaWYgZGVmaW5lZChfX2FybV9fKSAmJiAhZGVmaW5lZChfX3RodW1iX18pCi0gICAgaW50MzJfdCBvdXQ7Ci0gICAgaWYgKGxlZnQpIHsKLSAgICAgICAgYXNtKCAic211bGJiICVbb3V0XSwgJVtpblJMXSwgJVt2UkxdIFxuIgotICAgICAgICAgICAgIDogW291dF0iPXIiKG91dCkKLSAgICAgICAgICAgICA6IFtpblJMXSIlciIoaW5STCksIFt2UkxdInIiKHZSTCkKLSAgICAgICAgICAgICA6ICk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgYXNtKCAic211bHR0ICVbb3V0XSwgJVtpblJMXSwgJVt2UkxdIFxuIgotICAgICAgICAgICAgIDogW291dF0iPXIiKG91dCkKLSAgICAgICAgICAgICA6IFtpblJMXSIlciIoaW5STCksIFt2UkxdInIiKHZSTCkKLSAgICAgICAgICAgICA6ICk7Ci0gICAgfQotICAgIHJldHVybiBvdXQ7Ci0jZWxzZQotICAgIGlmIChsZWZ0KSB7Ci0gICAgICAgIHJldHVybiBpbnQxNl90KGluUkwmMHhGRkZGKSAqIGludDE2X3QodlJMJjB4RkZGRik7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgcmV0dXJuIGludDE2X3QoaW5STD4+MTYpICogaW50MTZfdCh2Ukw+PjE2KTsKLSAgICB9Ci0jZW5kaWYKLX0KLQotCi12b2lkIEF1ZGlvTWl4ZXI6OnRyYWNrX19nZW5lcmljUmVzYW1wbGUodHJhY2tfdCogdCwgaW50MzJfdCogb3V0LCBzaXplX3Qgb3V0RnJhbWVDb3VudCwgaW50MzJfdCogdGVtcCkKLXsKLSAgICB0LT5yZXNhbXBsZXItPnNldFNhbXBsZVJhdGUodC0+c2FtcGxlUmF0ZSk7Ci0KLSAgICAvLyByYW1wIGdhaW4gLSByZXNhbXBsZSB0byB0ZW1wIGJ1ZmZlciBhbmQgc2NhbGUvbWl4IGluIDJuZCBzdGVwCi0gICAgaWYgVU5MSUtFTFkodC0+dm9sdW1lSW5jWzBdfHQtPnZvbHVtZUluY1sxXSkgewotICAgICAgICB0LT5yZXNhbXBsZXItPnNldFZvbHVtZShVTklUWV9HQUlOLCBVTklUWV9HQUlOKTsKLSAgICAgICAgbWVtc2V0KHRlbXAsIDAsIG91dEZyYW1lQ291bnQgKiBNQVhfTlVNX0NIQU5ORUxTICogc2l6ZW9mKGludDMyX3QpKTsKLSAgICAgICAgdC0+cmVzYW1wbGVyLT5yZXNhbXBsZSh0ZW1wLCBvdXRGcmFtZUNvdW50LCB0LT5idWZmZXJQcm92aWRlcik7Ci0gICAgICAgIHZvbHVtZVJhbXBTdGVyZW8odCwgb3V0LCBvdXRGcmFtZUNvdW50LCB0ZW1wKTsKLSAgICB9Ci0KLSAgICAvLyBjb25zdGFudCBnYWluCi0gICAgZWxzZSB7Ci0gICAgICAgIHQtPnJlc2FtcGxlci0+c2V0Vm9sdW1lKHQtPnZvbHVtZVswXSwgdC0+dm9sdW1lWzFdKTsKLSAgICAgICAgdC0+cmVzYW1wbGVyLT5yZXNhbXBsZShvdXQsIG91dEZyYW1lQ291bnQsIHQtPmJ1ZmZlclByb3ZpZGVyKTsKLSAgICB9Ci19Ci0KLXZvaWQgQXVkaW9NaXhlcjo6dHJhY2tfX25vcCh0cmFja190KiB0LCBpbnQzMl90KiBvdXQsIHNpemVfdCBvdXRGcmFtZUNvdW50LCBpbnQzMl90KiB0ZW1wKQotewotfQotCi12b2lkIEF1ZGlvTWl4ZXI6OnZvbHVtZVJhbXBTdGVyZW8odHJhY2tfdCogdCwgaW50MzJfdCogb3V0LCBzaXplX3QgZnJhbWVDb3VudCwgaW50MzJfdCogdGVtcCkKLXsKLSAgICBpbnQzMl90IHZsID0gdC0+cHJldlZvbHVtZVswXTsKLSAgICBpbnQzMl90IHZyID0gdC0+cHJldlZvbHVtZVsxXTsKLSAgICBjb25zdCBpbnQzMl90IHZsSW5jID0gdC0+dm9sdW1lSW5jWzBdOwotICAgIGNvbnN0IGludDMyX3QgdnJJbmMgPSB0LT52b2x1bWVJbmNbMV07Ci0KLSAgICAvL0xPR0QoIlswXSAlcDogaW5jPSVmLCB2MD0lZiwgdjE9JWQsIGZpbmFsPSVmLCBjb3VudD0lZCIsCi0gICAgLy8gICAgICAgIHQsIHZsSW5jLzY1NTM2LjBmLCB2bC82NTUzNi4wZiwgdC0+dm9sdW1lWzBdLAotICAgIC8vICAgICAgICh2bCArIHZsSW5jKmZyYW1lQ291bnQpLzY1NTM2LjBmLCBmcmFtZUNvdW50KTsKLSAgIAotICAgIC8vIHJhbXAgdm9sdW1lCi0gICAgZG8gewotICAgICAgICAqb3V0KysgKz0gKHZsID4+IDE2KSAqICgqdGVtcCsrID4+IDEyKTsKLSAgICAgICAgKm91dCsrICs9ICh2ciA+PiAxNikgKiAoKnRlbXArKyA+PiAxMik7Ci0gICAgICAgIHZsICs9IHZsSW5jOwotICAgICAgICB2ciArPSB2ckluYzsKLSAgICB9IHdoaWxlICgtLWZyYW1lQ291bnQpOwotCi0gICAgdC0+cHJldlZvbHVtZVswXSA9IHZsOwotICAgIHQtPnByZXZWb2x1bWVbMV0gPSB2cjsKLSAgICB0LT5hZGp1c3RWb2x1bWVSYW1wKCk7Ci19Ci0KLXZvaWQgQXVkaW9NaXhlcjo6dHJhY2tfXzE2Qml0c1N0ZXJlbyh0cmFja190KiB0LCBpbnQzMl90KiBvdXQsIHNpemVfdCBmcmFtZUNvdW50LCBpbnQzMl90KiB0ZW1wKQotewotICAgIGludDE2X3QgY29uc3QgKmluID0gc3RhdGljX2Nhc3Q8aW50MTZfdCBjb25zdCAqPih0LT5pbik7Ci0KLSAgICAvLyByYW1wIGdhaW4KLSAgICBpZiBVTkxJS0VMWSh0LT52b2x1bWVJbmNbMF18dC0+dm9sdW1lSW5jWzFdKSB7Ci0gICAgICAgIGludDMyX3QgdmwgPSB0LT5wcmV2Vm9sdW1lWzBdOwotICAgICAgICBpbnQzMl90IHZyID0gdC0+cHJldlZvbHVtZVsxXTsKLSAgICAgICAgY29uc3QgaW50MzJfdCB2bEluYyA9IHQtPnZvbHVtZUluY1swXTsKLSAgICAgICAgY29uc3QgaW50MzJfdCB2ckluYyA9IHQtPnZvbHVtZUluY1sxXTsKLQotICAgICAgICAvLyBMT0dEKCJbMV0gJXA6IGluYz0lZiwgdjA9JWYsIHYxPSVkLCBmaW5hbD0lZiwgY291bnQ9JWQiLAotICAgICAgICAvLyAgICAgICAgdCwgdmxJbmMvNjU1MzYuMGYsIHZsLzY1NTM2LjBmLCB0LT52b2x1bWVbMF0sCi0gICAgICAgIC8vICAgICAgICAodmwgKyB2bEluYypmcmFtZUNvdW50KS82NTUzNi4wZiwgZnJhbWVDb3VudCk7Ci0KLSAgICAgICAgZG8gewotICAgICAgICAgICAgKm91dCsrICs9ICh2bCA+PiAxNikgKiAoaW50MzJfdCkgKmluKys7Ci0gICAgICAgICAgICAqb3V0KysgKz0gKHZyID4+IDE2KSAqIChpbnQzMl90KSAqaW4rKzsKLSAgICAgICAgICAgIHZsICs9IHZsSW5jOwotICAgICAgICAgICAgdnIgKz0gdnJJbmM7Ci0gICAgICAgIH0gd2hpbGUgKC0tZnJhbWVDb3VudCk7Ci0gICAgICAgCi0gICAgICAgIHQtPnByZXZWb2x1bWVbMF0gPSB2bDsKLSAgICAgICAgdC0+cHJldlZvbHVtZVsxXSA9IHZyOwotICAgICAgICB0LT5hZGp1c3RWb2x1bWVSYW1wKCk7Ci0gICAgfQotCi0gICAgLy8gY29uc3RhbnQgZ2FpbgotICAgIGVsc2UgewotICAgICAgICBjb25zdCB1aW50MzJfdCB2cmwgPSB0LT52b2x1bWVSTDsKLSAgICAgICAgZG8gewotICAgICAgICAgICAgdWludDMyX3QgcmwgPSAqcmVpbnRlcnByZXRfY2FzdDx1aW50MzJfdCBjb25zdCAqPihpbik7Ci0gICAgICAgICAgICBpbiArPSAyOwotICAgICAgICAgICAgb3V0WzBdID0gbXVsQWRkUkwoMSwgcmwsIHZybCwgb3V0WzBdKTsKLSAgICAgICAgICAgIG91dFsxXSA9IG11bEFkZFJMKDAsIHJsLCB2cmwsIG91dFsxXSk7Ci0gICAgICAgICAgICBvdXQgKz0gMjsKLSAgICAgICAgfSB3aGlsZSAoLS1mcmFtZUNvdW50KTsKLSAgICB9Ci0gICAgdC0+aW4gPSBpbjsKLX0KLQotdm9pZCBBdWRpb01peGVyOjp0cmFja19fMTZCaXRzTW9ubyh0cmFja190KiB0LCBpbnQzMl90KiBvdXQsIHNpemVfdCBmcmFtZUNvdW50LCBpbnQzMl90KiB0ZW1wKQotewotICAgIGludDE2X3QgY29uc3QgKmluID0gc3RhdGljX2Nhc3Q8aW50MTZfdCBjb25zdCAqPih0LT5pbik7Ci0KLSAgICAvLyByYW1wIGdhaW4KLSAgICBpZiBVTkxJS0VMWSh0LT52b2x1bWVJbmNbMF18dC0+dm9sdW1lSW5jWzFdKSB7Ci0gICAgICAgIGludDMyX3QgdmwgPSB0LT5wcmV2Vm9sdW1lWzBdOwotICAgICAgICBpbnQzMl90IHZyID0gdC0+cHJldlZvbHVtZVsxXTsKLSAgICAgICAgY29uc3QgaW50MzJfdCB2bEluYyA9IHQtPnZvbHVtZUluY1swXTsKLSAgICAgICAgY29uc3QgaW50MzJfdCB2ckluYyA9IHQtPnZvbHVtZUluY1sxXTsKLQotICAgICAgICAvLyBMT0dEKCJbMl0gJXA6IGluYz0lZiwgdjA9JWYsIHYxPSVkLCBmaW5hbD0lZiwgY291bnQ9JWQiLAotICAgICAgICAvLyAgICAgICAgIHQsIHZsSW5jLzY1NTM2LjBmLCB2bC82NTUzNi4wZiwgdC0+dm9sdW1lWzBdLAotICAgICAgICAvLyAgICAgICAgICh2bCArIHZsSW5jKmZyYW1lQ291bnQpLzY1NTM2LjBmLCBmcmFtZUNvdW50KTsKLQotICAgICAgICBkbyB7Ci0gICAgICAgICAgICBpbnQzMl90IGwgPSAqaW4rKzsKLSAgICAgICAgICAgICpvdXQrKyArPSAodmwgPj4gMTYpICogbDsKLSAgICAgICAgICAgICpvdXQrKyArPSAodnIgPj4gMTYpICogbDsKLSAgICAgICAgICAgIHZsICs9IHZsSW5jOwotICAgICAgICAgICAgdnIgKz0gdnJJbmM7Ci0gICAgICAgIH0gd2hpbGUgKC0tZnJhbWVDb3VudCk7Ci0gICAgICAgCi0gICAgICAgIHQtPnByZXZWb2x1bWVbMF0gPSB2bDsKLSAgICAgICAgdC0+cHJldlZvbHVtZVsxXSA9IHZyOwotICAgICAgICB0LT5hZGp1c3RWb2x1bWVSYW1wKCk7Ci0gICAgfQotICAgIC8vIGNvbnN0YW50IGdhaW4KLSAgICBlbHNlIHsKLSAgICAgICAgY29uc3QgaW50MTZfdCB2bCA9IHQtPnZvbHVtZVswXTsKLSAgICAgICAgY29uc3QgaW50MTZfdCB2ciA9IHQtPnZvbHVtZVsxXTsKLSAgICAgICAgZG8gewotICAgICAgICAgICAgaW50MTZfdCBsID0gKmluKys7Ci0gICAgICAgICAgICBvdXRbMF0gPSBtdWxBZGQobCwgdmwsIG91dFswXSk7Ci0gICAgICAgICAgICBvdXRbMV0gPSBtdWxBZGQobCwgdnIsIG91dFsxXSk7Ci0gICAgICAgICAgICBvdXQgKz0gMjsKLSAgICAgICAgfSB3aGlsZSAoLS1mcmFtZUNvdW50KTsKLSAgICB9Ci0gICAgdC0+aW4gPSBpbjsKLX0KLQotaW5saW5lCi12b2lkIEF1ZGlvTWl4ZXI6OmRpdGhlckFuZENsYW1wKGludDMyX3QqIG91dCwgaW50MzJfdCBjb25zdCAqc3Vtcywgc2l6ZV90IGMpCi17Ci0gICAgZm9yIChzaXplX3QgaT0wIDsgaTxjIDsgaSsrKSB7Ci0gICAgICAgIGludDMyX3QgbCA9ICpzdW1zKys7Ci0gICAgICAgIGludDMyX3QgciA9ICpzdW1zKys7Ci0gICAgICAgIGludDMyX3QgbmwgPSBsID4+IDEyOwotICAgICAgICBpbnQzMl90IG5yID0gciA+PiAxMjsKLSAgICAgICAgbCA9IGNsYW1wMTYobmwpOwotICAgICAgICByID0gY2xhbXAxNihucik7Ci0gICAgICAgICpvdXQrKyA9IChyPDwxNikgfCAobCAmIDB4RkZGRik7Ci0gICAgfQotfQotCi0vLyBuby1vcCBjYXNlCi12b2lkIEF1ZGlvTWl4ZXI6OnByb2Nlc3NfX25vcChzdGF0ZV90KiBzdGF0ZSwgdm9pZCogb3V0cHV0KQotewotICAgIC8vIHRoaXMgYXNzdW1lcyBvdXRwdXQgMTYgYml0cyBzdGVyZW8sIG5vIHJlc2FtcGxpbmcKLSAgICBtZW1zZXQob3V0cHV0LCAwLCBzdGF0ZS0+ZnJhbWVDb3VudCo0KTsKLSAgICB1aW50MzJfdCBlbiA9IHN0YXRlLT5lbmFibGVkVHJhY2tzOwotICAgIHdoaWxlIChlbikgewotICAgICAgICBjb25zdCBpbnQgaSA9IDMxIC0gX19idWlsdGluX2Nseihlbik7Ci0gICAgICAgIGVuICY9IH4oMTw8aSk7Ci0gICAgICAgIHRyYWNrX3QmIHQgPSBzdGF0ZS0+dHJhY2tzW2ldOwotICAgICAgICBzaXplX3Qgb3V0RnJhbWVzID0gc3RhdGUtPmZyYW1lQ291bnQ7Ci0gICAgICAgIHdoaWxlIChvdXRGcmFtZXMpIHsKLSAgICAgICAgICAgIHQuYnVmZmVyLmZyYW1lQ291bnQgPSBvdXRGcmFtZXM7Ci0gICAgICAgICAgICB0LmJ1ZmZlclByb3ZpZGVyLT5nZXROZXh0QnVmZmVyKCZ0LmJ1ZmZlcik7Ci0gICAgICAgICAgICBpZiAoIXQuYnVmZmVyLnJhdykgYnJlYWs7Ci0gICAgICAgICAgICBvdXRGcmFtZXMgLT0gdC5idWZmZXIuZnJhbWVDb3VudDsKLSAgICAgICAgICAgIHQuYnVmZmVyUHJvdmlkZXItPnJlbGVhc2VCdWZmZXIoJnQuYnVmZmVyKTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotLy8gZ2VuZXJpYyBjb2RlIHdpdGhvdXQgcmVzYW1wbGluZwotdm9pZCBBdWRpb01peGVyOjpwcm9jZXNzX19nZW5lcmljTm9SZXNhbXBsaW5nKHN0YXRlX3QqIHN0YXRlLCB2b2lkKiBvdXRwdXQpCi17Ci0gICAgaW50MzJfdCBvdXRUZW1wW0JMT0NLU0laRSAqIE1BWF9OVU1fQ0hBTk5FTFNdIF9fYXR0cmlidXRlX18oKGFsaWduZWQoMzIpKSk7Ci0KLSAgICAvLyBhY3F1aXJlIGVhY2ggdHJhY2sncyBidWZmZXIKLSAgICB1aW50MzJfdCBlbmFibGVkVHJhY2tzID0gc3RhdGUtPmVuYWJsZWRUcmFja3M7Ci0gICAgdWludDMyX3QgZW4gPSBlbmFibGVkVHJhY2tzOwotICAgIHdoaWxlIChlbikgewotICAgICAgICBjb25zdCBpbnQgaSA9IDMxIC0gX19idWlsdGluX2Nseihlbik7Ci0gICAgICAgIGVuICY9IH4oMTw8aSk7Ci0gICAgICAgIHRyYWNrX3QmIHQgPSBzdGF0ZS0+dHJhY2tzW2ldOwotICAgICAgICB0LmJ1ZmZlci5mcmFtZUNvdW50ID0gc3RhdGUtPmZyYW1lQ291bnQ7Ci0gICAgICAgIHQuYnVmZmVyUHJvdmlkZXItPmdldE5leHRCdWZmZXIoJnQuYnVmZmVyKTsKLSAgICAgICAgdC5mcmFtZUNvdW50ID0gdC5idWZmZXIuZnJhbWVDb3VudDsKLSAgICAgICAgdC5pbiA9IHQuYnVmZmVyLnJhdzsKLSAgICAgICAgLy8gdC5pbiA9PSBOVUxMIGNhbiBoYXBwZW4gaWYgdGhlIHRyYWNrIHdhcyBmbHVzaGVkIGp1c3QgYWZ0ZXIgaGF2aW5nCi0gICAgICAgIC8vIGJlZW4gZW5hYmxlZCBmb3IgbWl4aW5nLgotICAgICAgICBpZiAodC5pbiA9PSBOVUxMKQotICAgICAgICAgICAgZW5hYmxlZFRyYWNrcyAmPSB+KDE8PGkpOwotICAgIH0KLQotICAgIC8vIHRoaXMgYXNzdW1lcyBvdXRwdXQgMTYgYml0cyBzdGVyZW8sIG5vIHJlc2FtcGxpbmcKLSAgICBpbnQzMl90KiBvdXQgPSBzdGF0aWNfY2FzdDxpbnQzMl90Kj4ob3V0cHV0KTsKLSAgICBzaXplX3QgbnVtRnJhbWVzID0gc3RhdGUtPmZyYW1lQ291bnQ7Ci0gICAgZG8gewotICAgICAgICBtZW1zZXQob3V0VGVtcCwgMCwgc2l6ZW9mKG91dFRlbXApKTsKLQotICAgICAgICBlbiA9IGVuYWJsZWRUcmFja3M7Ci0gICAgICAgIHdoaWxlIChlbikgewotICAgICAgICAgICAgY29uc3QgaW50IGkgPSAzMSAtIF9fYnVpbHRpbl9jbHooZW4pOwotICAgICAgICAgICAgZW4gJj0gfigxPDxpKTsKLSAgICAgICAgICAgIHRyYWNrX3QmIHQgPSBzdGF0ZS0+dHJhY2tzW2ldOwotICAgICAgICAgICAgc2l6ZV90IG91dEZyYW1lcyA9IEJMT0NLU0laRTsKLSAgICAgICAgICAgCi0gICAgICAgICAgICB3aGlsZSAob3V0RnJhbWVzKSB7Ci0gICAgICAgICAgICAgICAgc2l6ZV90IGluRnJhbWVzID0gKHQuZnJhbWVDb3VudCA+IG91dEZyYW1lcyk/b3V0RnJhbWVzOnQuZnJhbWVDb3VudDsKLSAgICAgICAgICAgICAgICBpZiAoaW5GcmFtZXMpIHsKLSAgICAgICAgICAgICAgICAgICAgKHQuaG9vaykoJnQsIG91dFRlbXAgKyAoQkxPQ0tTSVpFLW91dEZyYW1lcykqTUFYX05VTV9DSEFOTkVMUywgaW5GcmFtZXMsIHN0YXRlLT5yZXNhbXBsZVRlbXApOwotICAgICAgICAgICAgICAgICAgICB0LmZyYW1lQ291bnQgLT0gaW5GcmFtZXM7Ci0gICAgICAgICAgICAgICAgICAgIG91dEZyYW1lcyAtPSBpbkZyYW1lczsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKHQuZnJhbWVDb3VudCA9PSAwICYmIG91dEZyYW1lcykgewotICAgICAgICAgICAgICAgICAgICB0LmJ1ZmZlclByb3ZpZGVyLT5yZWxlYXNlQnVmZmVyKCZ0LmJ1ZmZlcik7Ci0gICAgICAgICAgICAgICAgICAgIHQuYnVmZmVyLmZyYW1lQ291bnQgPSBudW1GcmFtZXMgLSAoQkxPQ0tTSVpFIC0gb3V0RnJhbWVzKTsKLSAgICAgICAgICAgICAgICAgICAgdC5idWZmZXJQcm92aWRlci0+Z2V0TmV4dEJ1ZmZlcigmdC5idWZmZXIpOwotICAgICAgICAgICAgICAgICAgICB0LmluID0gdC5idWZmZXIucmF3OwotICAgICAgICAgICAgICAgICAgICBpZiAodC5pbiA9PSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBlbmFibGVkVHJhY2tzICY9IH4oMTw8aSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB0LmZyYW1lQ291bnQgPSB0LmJ1ZmZlci5mcmFtZUNvdW50OwotICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBkaXRoZXJBbmRDbGFtcChvdXQsIG91dFRlbXAsIEJMT0NLU0laRSk7Ci0gICAgICAgIG91dCArPSBCTE9DS1NJWkU7Ci0gICAgICAgIG51bUZyYW1lcyAtPSBCTE9DS1NJWkU7Ci0gICAgfSB3aGlsZSAobnVtRnJhbWVzKTsKLQotCi0gICAgLy8gcmVsZWFzZSBlYWNoIHRyYWNrJ3MgYnVmZmVyCi0gICAgZW4gPSBlbmFibGVkVHJhY2tzOwotICAgIHdoaWxlIChlbikgewotICAgICAgICBjb25zdCBpbnQgaSA9IDMxIC0gX19idWlsdGluX2Nseihlbik7Ci0gICAgICAgIGVuICY9IH4oMTw8aSk7Ci0gICAgICAgIHRyYWNrX3QmIHQgPSBzdGF0ZS0+dHJhY2tzW2ldOwotICAgICAgICB0LmJ1ZmZlclByb3ZpZGVyLT5yZWxlYXNlQnVmZmVyKCZ0LmJ1ZmZlcik7Ci0gICAgfQotfQotCi0vLyBnZW5lcmljIGNvZGUgd2l0aCByZXNhbXBsaW5nCi12b2lkIEF1ZGlvTWl4ZXI6OnByb2Nlc3NfX2dlbmVyaWNSZXNhbXBsaW5nKHN0YXRlX3QqIHN0YXRlLCB2b2lkKiBvdXRwdXQpCi17Ci0gICAgaW50MzJfdCogY29uc3Qgb3V0VGVtcCA9IHN0YXRlLT5vdXRwdXRUZW1wOwotICAgIGNvbnN0IHNpemVfdCBzaXplID0gc2l6ZW9mKGludDMyX3QpICogTUFYX05VTV9DSEFOTkVMUyAqIHN0YXRlLT5mcmFtZUNvdW50OwotICAgIG1lbXNldChvdXRUZW1wLCAwLCBzaXplKTsKLQotICAgIGludDMyX3QqIG91dCA9IHN0YXRpY19jYXN0PGludDMyX3QqPihvdXRwdXQpOwotICAgIHNpemVfdCBudW1GcmFtZXMgPSBzdGF0ZS0+ZnJhbWVDb3VudDsKLQotICAgIHVpbnQzMl90IGVuID0gc3RhdGUtPmVuYWJsZWRUcmFja3M7Ci0gICAgd2hpbGUgKGVuKSB7Ci0gICAgICAgIGNvbnN0IGludCBpID0gMzEgLSBfX2J1aWx0aW5fY2x6KGVuKTsKLSAgICAgICAgZW4gJj0gfigxPDxpKTsKLSAgICAgICAgdHJhY2tfdCYgdCA9IHN0YXRlLT50cmFja3NbaV07Ci0KLSAgICAgICAgLy8gdGhpcyBpcyBhIGxpdHRsZSBnb29meSwgb24gdGhlIHJlc2FtcGxpbmcgY2FzZSB3ZSBkb24ndAotICAgICAgICAvLyBhY3F1aXJlL3JlbGVhc2UgdGhlIGJ1ZmZlcnMgYmVjYXVzZSBpdCdzIGRvbmUgYnkKLSAgICAgICAgLy8gdGhlIHJlc2FtcGxlci4KLSAgICAgICAgaWYgKCh0Lm5lZWRzICYgTkVFRFNfUkVTQU1QTEVfX01BU0spID09IE5FRURTX1JFU0FNUExFX0VOQUJMRUQpIHsKLSAgICAgICAgICAgICh0Lmhvb2spKCZ0LCBvdXRUZW1wLCBudW1GcmFtZXMsIHN0YXRlLT5yZXNhbXBsZVRlbXApOwotICAgICAgICB9IGVsc2UgewotCi0gICAgICAgICAgICBzaXplX3Qgb3V0RnJhbWVzID0gbnVtRnJhbWVzOwotICAgICAgICAgICAKLSAgICAgICAgICAgIHdoaWxlIChvdXRGcmFtZXMpIHsKLSAgICAgICAgICAgICAgICB0LmJ1ZmZlci5mcmFtZUNvdW50ID0gb3V0RnJhbWVzOwotICAgICAgICAgICAgICAgIHQuYnVmZmVyUHJvdmlkZXItPmdldE5leHRCdWZmZXIoJnQuYnVmZmVyKTsKLSAgICAgICAgICAgICAgICB0LmluID0gdC5idWZmZXIucmF3OwotICAgICAgICAgICAgICAgIC8vIHQuaW4gPT0gTlVMTCBjYW4gaGFwcGVuIGlmIHRoZSB0cmFjayB3YXMgZmx1c2hlZCBqdXN0IGFmdGVyIGhhdmluZwotICAgICAgICAgICAgICAgIC8vIGJlZW4gZW5hYmxlZCBmb3IgbWl4aW5nLgotICAgICAgICAgICAgICAgIGlmICh0LmluID09IE5VTEwpIGJyZWFrOwotCi0gICAgICAgICAgICAgICAgKHQuaG9vaykoJnQsIG91dFRlbXAgKyAobnVtRnJhbWVzLW91dEZyYW1lcykqTUFYX05VTV9DSEFOTkVMUywgdC5idWZmZXIuZnJhbWVDb3VudCwgc3RhdGUtPnJlc2FtcGxlVGVtcCk7Ci0gICAgICAgICAgICAgICAgb3V0RnJhbWVzIC09IHQuYnVmZmVyLmZyYW1lQ291bnQ7Ci0gICAgICAgICAgICAgICAgdC5idWZmZXJQcm92aWRlci0+cmVsZWFzZUJ1ZmZlcigmdC5idWZmZXIpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgZGl0aGVyQW5kQ2xhbXAob3V0LCBvdXRUZW1wLCBudW1GcmFtZXMpOwotfQotCi0vLyBvbmUgdHJhY2ssIDE2IGJpdHMgc3RlcmVvIHdpdGhvdXQgcmVzYW1wbGluZyBpcyB0aGUgbW9zdCBjb21tb24gY2FzZQotdm9pZCBBdWRpb01peGVyOjpwcm9jZXNzX19PbmVUcmFjazE2Qml0c1N0ZXJlb05vUmVzYW1wbGluZyhzdGF0ZV90KiBzdGF0ZSwgdm9pZCogb3V0cHV0KQotewotICAgIGNvbnN0IGludCBpID0gMzEgLSBfX2J1aWx0aW5fY2x6KHN0YXRlLT5lbmFibGVkVHJhY2tzKTsKLSAgICBjb25zdCB0cmFja190JiB0ID0gc3RhdGUtPnRyYWNrc1tpXTsKLQotICAgIEF1ZGlvQnVmZmVyUHJvdmlkZXI6OkJ1ZmZlciYgYih0LmJ1ZmZlcik7Ci0gICAKLSAgICBpbnQzMl90KiBvdXQgPSBzdGF0aWNfY2FzdDxpbnQzMl90Kj4ob3V0cHV0KTsKLSAgICBzaXplX3QgbnVtRnJhbWVzID0gc3RhdGUtPmZyYW1lQ291bnQ7Ci0gIAotICAgIGNvbnN0IGludDE2X3QgdmwgPSB0LnZvbHVtZVswXTsKLSAgICBjb25zdCBpbnQxNl90IHZyID0gdC52b2x1bWVbMV07Ci0gICAgY29uc3QgdWludDMyX3QgdnJsID0gdC52b2x1bWVSTDsKLSAgICB3aGlsZSAobnVtRnJhbWVzKSB7Ci0gICAgICAgIGIuZnJhbWVDb3VudCA9IG51bUZyYW1lczsKLSAgICAgICAgdC5idWZmZXJQcm92aWRlci0+Z2V0TmV4dEJ1ZmZlcigmYik7Ci0gICAgICAgIGludDE2X3QgY29uc3QgKmluID0gYi5pMTY7Ci0KLSAgICAgICAgLy8gaW4gPT0gTlVMTCBjYW4gaGFwcGVuIGlmIHRoZSB0cmFjayB3YXMgZmx1c2hlZCBqdXN0IGFmdGVyIGhhdmluZwotICAgICAgICAvLyBiZWVuIGVuYWJsZWQgZm9yIG1peGluZy4KLSAgICAgICAgaWYgKGluID09IE5VTEwpIHsKLSAgICAgICAgICAgIG1lbXNldChvdXQsIDAsIG51bUZyYW1lcypNQVhfTlVNX0NIQU5ORUxTKnNpemVvZihpbnQxNl90KSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgc2l6ZV90IG91dEZyYW1lcyA9IGIuZnJhbWVDb3VudDsKLSAgICAgICAKLSAgICAgICAgaWYgKFVOTElLRUxZKHVpbnQzMl90KHZsKSA+IFVOSVRZX0dBSU4gfHwgdWludDMyX3QodnIpID4gVU5JVFlfR0FJTikpIHsKLSAgICAgICAgICAgIC8vIHZvbHVtZSBpcyBib29zdGVkLCBzbyB3ZSBtaWdodCBuZWVkIHRvIGNsYW1wIGV2ZW4gdGhvdWdoCi0gICAgICAgICAgICAvLyB3ZSBwcm9jZXNzIG9ubHkgb25lIHRyYWNrLgotICAgICAgICAgICAgZG8gewotICAgICAgICAgICAgICAgIHVpbnQzMl90IHJsID0gKnJlaW50ZXJwcmV0X2Nhc3Q8dWludDMyX3QgY29uc3QgKj4oaW4pOwotICAgICAgICAgICAgICAgIGluICs9IDI7Ci0gICAgICAgICAgICAgICAgaW50MzJfdCBsID0gbXVsUkwoMSwgcmwsIHZybCkgPj4gMTI7Ci0gICAgICAgICAgICAgICAgaW50MzJfdCByID0gbXVsUkwoMCwgcmwsIHZybCkgPj4gMTI7Ci0gICAgICAgICAgICAgICAgLy8gY2xhbXBpbmcuLi4KLSAgICAgICAgICAgICAgICBsID0gY2xhbXAxNihsKTsKLSAgICAgICAgICAgICAgICByID0gY2xhbXAxNihyKTsKLSAgICAgICAgICAgICAgICAqb3V0KysgPSAocjw8MTYpIHwgKGwgJiAweEZGRkYpOwotICAgICAgICAgICAgfSB3aGlsZSAoLS1vdXRGcmFtZXMpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZG8gewotICAgICAgICAgICAgICAgIHVpbnQzMl90IHJsID0gKnJlaW50ZXJwcmV0X2Nhc3Q8dWludDMyX3QgY29uc3QgKj4oaW4pOwotICAgICAgICAgICAgICAgIGluICs9IDI7Ci0gICAgICAgICAgICAgICAgaW50MzJfdCBsID0gbXVsUkwoMSwgcmwsIHZybCkgPj4gMTI7Ci0gICAgICAgICAgICAgICAgaW50MzJfdCByID0gbXVsUkwoMCwgcmwsIHZybCkgPj4gMTI7Ci0gICAgICAgICAgICAgICAgKm91dCsrID0gKHI8PDE2KSB8IChsICYgMHhGRkZGKTsKLSAgICAgICAgICAgIH0gd2hpbGUgKC0tb3V0RnJhbWVzKTsKLSAgICAgICAgfQotICAgICAgICBudW1GcmFtZXMgLT0gYi5mcmFtZUNvdW50OwotICAgICAgICB0LmJ1ZmZlclByb3ZpZGVyLT5yZWxlYXNlQnVmZmVyKCZiKTsKLSAgICB9Ci19Ci0KLS8vIDIgdHJhY2tzIGlzIGFsc28gYSBjb21tb24gY2FzZQotdm9pZCBBdWRpb01peGVyOjpwcm9jZXNzX19Ud29UcmFja3MxNkJpdHNTdGVyZW9Ob1Jlc2FtcGxpbmcoc3RhdGVfdCogc3RhdGUsIHZvaWQqIG91dHB1dCkKLXsKLSAgICBpbnQgaTsKLSAgICB1aW50MzJfdCBlbiA9IHN0YXRlLT5lbmFibGVkVHJhY2tzOwotCi0gICAgaSA9IDMxIC0gX19idWlsdGluX2Nseihlbik7Ci0gICAgY29uc3QgdHJhY2tfdCYgdDAgPSBzdGF0ZS0+dHJhY2tzW2ldOwotICAgIEF1ZGlvQnVmZmVyUHJvdmlkZXI6OkJ1ZmZlciYgYjAodDAuYnVmZmVyKTsKLQotICAgIGVuICY9IH4oMTw8aSk7Ci0gICAgaSA9IDMxIC0gX19idWlsdGluX2Nseihlbik7Ci0gICAgY29uc3QgdHJhY2tfdCYgdDEgPSBzdGF0ZS0+dHJhY2tzW2ldOwotICAgIEF1ZGlvQnVmZmVyUHJvdmlkZXI6OkJ1ZmZlciYgYjEodDEuYnVmZmVyKTsKLSAgIAotICAgIGludDE2X3QgY29uc3QgKmluMDsKLSAgICBjb25zdCBpbnQxNl90IHZsMCA9IHQwLnZvbHVtZVswXTsKLSAgICBjb25zdCBpbnQxNl90IHZyMCA9IHQwLnZvbHVtZVsxXTsKLSAgICBzaXplX3QgZnJhbWVDb3VudDAgPSAwOwotICAKLSAgICBpbnQxNl90IGNvbnN0ICppbjE7Ci0gICAgY29uc3QgaW50MTZfdCB2bDEgPSB0MS52b2x1bWVbMF07Ci0gICAgY29uc3QgaW50MTZfdCB2cjEgPSB0MS52b2x1bWVbMV07Ci0gICAgc2l6ZV90IGZyYW1lQ291bnQxID0gMDsKLSAgIAotICAgIGludDMyX3QqIG91dCA9IHN0YXRpY19jYXN0PGludDMyX3QqPihvdXRwdXQpOwotICAgIHNpemVfdCBudW1GcmFtZXMgPSBzdGF0ZS0+ZnJhbWVDb3VudDsKLSAgICBpbnQxNl90IGNvbnN0ICpidWZmID0gTlVMTDsKLQotICAKLSAgICB3aGlsZSAobnVtRnJhbWVzKSB7Ci0gICAKLSAgICAgICAgaWYgKGZyYW1lQ291bnQwID09IDApIHsKLSAgICAgICAgICAgIGIwLmZyYW1lQ291bnQgPSBudW1GcmFtZXM7Ci0gICAgICAgICAgICB0MC5idWZmZXJQcm92aWRlci0+Z2V0TmV4dEJ1ZmZlcigmYjApOwotICAgICAgICAgICAgaWYgKGIwLmkxNiA9PSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGJ1ZmYgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICBidWZmID0gbmV3IGludDE2X3RbTUFYX05VTV9DSEFOTkVMUyAqIHN0YXRlLT5mcmFtZUNvdW50XTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaW4wID0gYnVmZjsKLSAgICAgICAgICAgICAgICBiMC5mcmFtZUNvdW50ID0gbnVtRnJhbWVzOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBpbjAgPSBiMC5pMTY7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBmcmFtZUNvdW50MCA9IGIwLmZyYW1lQ291bnQ7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGZyYW1lQ291bnQxID09IDApIHsKLSAgICAgICAgICAgIGIxLmZyYW1lQ291bnQgPSBudW1GcmFtZXM7Ci0gICAgICAgICAgICB0MS5idWZmZXJQcm92aWRlci0+Z2V0TmV4dEJ1ZmZlcigmYjEpOwotICAgICAgICAgICAgaWYgKGIxLmkxNiA9PSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGJ1ZmYgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICBidWZmID0gbmV3IGludDE2X3RbTUFYX05VTV9DSEFOTkVMUyAqIHN0YXRlLT5mcmFtZUNvdW50XTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaW4xID0gYnVmZjsKLSAgICAgICAgICAgICAgICBiMS5mcmFtZUNvdW50ID0gbnVtRnJhbWVzOwotICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBpbjEgPSBiMS5pMTY7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBmcmFtZUNvdW50MSA9IGIxLmZyYW1lQ291bnQ7Ci0gICAgICAgIH0KLSAgICAgICAKLSAgICAgICAgc2l6ZV90IG91dEZyYW1lcyA9IGZyYW1lQ291bnQwIDwgZnJhbWVDb3VudDE/ZnJhbWVDb3VudDA6ZnJhbWVDb3VudDE7Ci0KLSAgICAgICAgbnVtRnJhbWVzIC09IG91dEZyYW1lczsKLSAgICAgICAgZnJhbWVDb3VudDAgLT0gb3V0RnJhbWVzOwotICAgICAgICBmcmFtZUNvdW50MSAtPSBvdXRGcmFtZXM7Ci0gICAgICAgCi0gICAgICAgIGRvIHsKLSAgICAgICAgICAgIGludDMyX3QgbDAgPSAqaW4wKys7Ci0gICAgICAgICAgICBpbnQzMl90IHIwID0gKmluMCsrOwotICAgICAgICAgICAgbDAgPSBtdWwobDAsIHZsMCk7Ci0gICAgICAgICAgICByMCA9IG11bChyMCwgdnIwKTsKLSAgICAgICAgICAgIGludDMyX3QgbCA9ICppbjErKzsKLSAgICAgICAgICAgIGludDMyX3QgciA9ICppbjErKzsKLSAgICAgICAgICAgIGwgPSBtdWxBZGQobCwgdmwxLCBsMCkgPj4gMTI7Ci0gICAgICAgICAgICByID0gbXVsQWRkKHIsIHZyMSwgcjApID4+IDEyOwotICAgICAgICAgICAgLy8gY2xhbXBpbmcuLi4KLSAgICAgICAgICAgIGwgPSBjbGFtcDE2KGwpOwotICAgICAgICAgICAgciA9IGNsYW1wMTYocik7Ci0gICAgICAgICAgICAqb3V0KysgPSAocjw8MTYpIHwgKGwgJiAweEZGRkYpOwotICAgICAgICB9IHdoaWxlICgtLW91dEZyYW1lcyk7Ci0gICAgICAgCi0gICAgICAgIGlmIChmcmFtZUNvdW50MCA9PSAwKSB7Ci0gICAgICAgICAgICB0MC5idWZmZXJQcm92aWRlci0+cmVsZWFzZUJ1ZmZlcigmYjApOwotICAgICAgICB9Ci0gICAgICAgIGlmIChmcmFtZUNvdW50MSA9PSAwKSB7Ci0gICAgICAgICAgICB0MS5idWZmZXJQcm92aWRlci0+cmVsZWFzZUJ1ZmZlcigmYjEpOwotICAgICAgICB9Ci0gICAgfSAgIAotICAgICAgIAotICAgIGlmIChidWZmICE9IE5VTEwpIHsKLSAgICAgICAgZGVsZXRlIFtdIGJ1ZmY7ICAgICAgIAotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9NaXhlci5oIGIvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9NaXhlci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3MmNhMjhhLi4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvTWl4ZXIuaAorKysgL2Rldi9udWxsCkBAIC0xLDE5MiArMCwwIEBACi0vKiAvL2RldmljZS9pbmNsdWRlL3NlcnZlci9BdWRpb0ZsaW5nZXIvQXVkaW9NaXhlci5oCi0qKgotKiogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9BVURJT19NSVhFUl9ICi0jZGVmaW5lIEFORFJPSURfQVVESU9fTUlYRVJfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlICJBdWRpb0J1ZmZlclByb3ZpZGVyLmgiCi0jaW5jbHVkZSAiQXVkaW9SZXNhbXBsZXIuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNkZWZpbmUgTElLRUxZKCBleHAgKSAgICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgdHJ1ZSAgKSkKLSNkZWZpbmUgVU5MSUtFTFkoIGV4cCApICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgZmFsc2UgKSkKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBBdWRpb01peGVyCi17Ci1wdWJsaWM6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgQXVkaW9NaXhlcihzaXplX3QgZnJhbWVDb3VudCwgdWludDMyX3Qgc2FtcGxlUmF0ZSk7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB+QXVkaW9NaXhlcigpOwotCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IE1BWF9OVU1fVFJBQ0tTID0gMzI7Ci0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IE1BWF9OVU1fQ0hBTk5FTFMgPSAyOwotCi0gICAgc3RhdGljIGNvbnN0IHVpbnQxNl90IFVOSVRZX0dBSU4gPSAweDEwMDA7Ci0KLSAgICBlbnVtIHsgLy8gbmFtZXMKLQotICAgICAgICAvLyB0cmFjayB1bml0cyAoMzIgdW5pdHMpCi0gICAgICAgIFRSQUNLMCAgICAgICAgICA9IDB4MTAwMCwKLQotICAgICAgICAvLyBlbmFibGUvZGlzYWJsZQotICAgICAgICBNSVhJTkcgICAgICAgICAgPSAweDIwMDAsCi0KLSAgICAgICAgLy8gc2V0UGFyYW1ldGVyIHRhcmdldHMKLSAgICAgICAgVFJBQ0sgICAgICAgICAgID0gMHgzMDAwLAotICAgICAgICBSRVNBTVBMRSAgICAgICAgPSAweDMwMDEsCi0gICAgICAgIFJBTVBfVk9MVU1FICAgICA9IDB4MzAwMiwgLy8gcmFtcCB0byBuZXcgdm9sdW1lCi0gICAgICAgIFZPTFVNRSAgICAgICAgICA9IDB4MzAwMywgLy8gZG9uJ3QgcmFtcAotCi0gICAgICAgIC8vIHNldCBQYXJhbWV0ZXIgbmFtZXMKLSAgICAgICAgLy8gZm9yIHRhcmdldCBUUkFDSwotICAgICAgICBDSEFOTkVMX0NPVU5UICAgPSAweDQwMDAsCi0gICAgICAgIEZPUk1BVCAgICAgICAgICA9IDB4NDAwMSwKLSAgICAgICAgLy8gZm9yIFRBUkdFVCBSRVNBTVBMRQotICAgICAgICBTQU1QTEVfUkFURSAgICAgPSAweDQxMDAsCi0gICAgICAgIC8vIGZvciBUQVJHRVQgVk9MVU1FICg4IGNoYW5uZWxzIG1heCkKLSAgICAgICAgVk9MVU1FMCAgICAgICAgID0gMHg0MjAwLAotICAgICAgICBWT0xVTUUxICAgICAgICAgPSAweDQyMDEsCi0gICAgfTsKLQotCi0gICAgaW50ICAgICAgICAgZ2V0VHJhY2tOYW1lKCk7Ci0gICAgdm9pZCAgICAgICAgZGVsZXRlVHJhY2tOYW1lKGludCBuYW1lKTsKLQotICAgIHN0YXR1c190ICAgIGVuYWJsZShpbnQgbmFtZSk7Ci0gICAgc3RhdHVzX3QgICAgZGlzYWJsZShpbnQgbmFtZSk7Ci0KLSAgICBzdGF0dXNfdCAgICBzZXRBY3RpdmVUcmFjayhpbnQgdHJhY2spOwotICAgIHN0YXR1c190ICAgIHNldFBhcmFtZXRlcihpbnQgdGFyZ2V0LCBpbnQgbmFtZSwgaW50IHZhbHVlKTsKLQotICAgIHN0YXR1c190ICAgIHNldEJ1ZmZlclByb3ZpZGVyKEF1ZGlvQnVmZmVyUHJvdmlkZXIqIGJ1ZmZlclByb3ZpZGVyKTsKLSAgICB2b2lkICAgICAgICBwcm9jZXNzKHZvaWQqIG91dHB1dCk7Ci0KLSAgICB1aW50MzJfdCAgICB0cmFja05hbWVzKCkgY29uc3QgeyByZXR1cm4gbVRyYWNrTmFtZXM7IH0KLQotcHJpdmF0ZToKLQotICAgIGVudW0gewotICAgICAgICBORUVEU19DSEFOTkVMX0NPVU5UX19NQVNLICAgPSAweDAwMDAwMDAzLAotICAgICAgICBORUVEU19GT1JNQVRfX01BU0sgICAgICAgICAgPSAweDAwMDAwMEYwLAotICAgICAgICBORUVEU19NVVRFX19NQVNLICAgICAgICAgICAgPSAweDAwMDAwMTAwLAotICAgICAgICBORUVEU19SRVNBTVBMRV9fTUFTSyAgICAgICAgPSAweDAwMDAxMDAwLAotICAgIH07Ci0KLSAgICBlbnVtIHsKLSAgICAgICAgTkVFRFNfQ0hBTk5FTF8xICAgICAgICAgICAgID0gMHgwMDAwMDAwMCwKLSAgICAgICAgTkVFRFNfQ0hBTk5FTF8yICAgICAgICAgICAgID0gMHgwMDAwMDAwMSwKLQotICAgICAgICBORUVEU19GT1JNQVRfMTYgICAgICAgICAgICAgPSAweDAwMDAwMDEwLAotCi0gICAgICAgIE5FRURTX01VVEVfRElTQUJMRUQgICAgICAgICA9IDB4MDAwMDAwMDAsCi0gICAgICAgIE5FRURTX01VVEVfRU5BQkxFRCAgICAgICAgICA9IDB4MDAwMDAxMDAsCi0KLSAgICAgICAgTkVFRFNfUkVTQU1QTEVfRElTQUJMRUQgICAgID0gMHgwMDAwMDAwMCwKLSAgICAgICAgTkVFRFNfUkVTQU1QTEVfRU5BQkxFRCAgICAgID0gMHgwMDAwMTAwMCwKLSAgICB9OwotCi0gICAgc3RhdGljIGlubGluZSBpbnQzMl90IGFwcGx5Vm9sdW1lKGludDMyX3QgaW4sIGludDMyX3QgdikgewotICAgICAgICByZXR1cm4gaW4gKiB2OwotICAgIH0KLQotCi0gICAgc3RydWN0IHN0YXRlX3Q7Ci0KLSAgICB0eXBlZGVmIHZvaWQgKCptaXhfdCkoc3RhdGVfdCogc3RhdGUsIHZvaWQqIG91dHB1dCk7Ci0KLSAgICBzdGF0aWMgY29uc3QgaW50IEJMT0NLU0laRSA9IDE2OyAvLyA0IGNhY2hlIGxpbmVzCi0KLSAgICBzdHJ1Y3QgdHJhY2tfdCB7Ci0gICAgICAgIHVpbnQzMl90ICAgIG5lZWRzOwotCi0gICAgICAgIHVuaW9uIHsKLSAgICAgICAgaW50MTZfdCAgICAgdm9sdW1lWzJdOyAgICAgIC8vIFswXTMuMTIgZml4ZWQgcG9pbnQKLSAgICAgICAgaW50MzJfdCAgICAgdm9sdW1lUkw7Ci0gICAgICAgIH07Ci0KLSAgICAgICAgaW50MzJfdCAgICAgcHJldlZvbHVtZVsyXTsKLQotICAgICAgICBpbnQzMl90ICAgICB2b2x1bWVJbmNbMl07Ci0KLSAgICAgICAgdWludDE2X3QgICAgZnJhbWVDb3VudDsKLQotICAgICAgICB1aW50OF90ICAgICBjaGFubmVsQ291bnQgOiA0OwotICAgICAgICB1aW50OF90ICAgICBlbmFibGVkICAgICAgOiAxOwotICAgICAgICB1aW50OF90ICAgICByZXNlcnZlZDAgICAgOiAzOwotICAgICAgICB1aW50OF90ICAgICBmb3JtYXQ7Ci0KLSAgICAgICAgQXVkaW9CdWZmZXJQcm92aWRlciogICAgICAgICAgICAgICAgYnVmZmVyUHJvdmlkZXI7Ci0gICAgICAgIG11dGFibGUgQXVkaW9CdWZmZXJQcm92aWRlcjo6QnVmZmVyIGJ1ZmZlcjsKLQotICAgICAgICB2b2lkICgqaG9vaykodHJhY2tfdCogdCwgaW50MzJfdCogb3V0cHV0LCBzaXplX3QgbnVtT3V0RnJhbWVzLCBpbnQzMl90KiB0ZW1wKTsKLSAgICAgICAgdm9pZCBjb25zdCogaW47ICAgICAgICAgICAgIC8vIGN1cnJlbnQgbG9jYXRpb24gaW4gYnVmZmVyCi0KLSAgICAgICAgQXVkaW9SZXNhbXBsZXIqICAgICByZXNhbXBsZXI7Ci0gICAgICAgIHVpbnQzMl90ICAgICAgICAgICAgc2FtcGxlUmF0ZTsKLQotICAgICAgICBib29sICAgICAgICBzZXRSZXNhbXBsZXIodWludDMyX3Qgc2FtcGxlUmF0ZSwgdWludDMyX3QgZGV2U2FtcGxlUmF0ZSk7Ci0gICAgICAgIGJvb2wgICAgICAgIGRvZXNSZXNhbXBsZSgpIGNvbnN0OwotICAgICAgICB2b2lkICAgICAgICBhZGp1c3RWb2x1bWVSYW1wKCk7Ci0gICAgfTsKLQotICAgIC8vIHBhZCB0byAzMi1ieXRlcyB0byBmaWxsIGNhY2hlIGxpbmUKLSAgICBzdHJ1Y3Qgc3RhdGVfdCB7Ci0gICAgICAgIHVpbnQzMl90ICAgICAgICBlbmFibGVkVHJhY2tzOwotICAgICAgICB1aW50MzJfdCAgICAgICAgbmVlZHNDaGFuZ2VkOwotICAgICAgICBzaXplX3QgICAgICAgICAgZnJhbWVDb3VudDsKLSAgICAgICAgbWl4X3QgICAgICAgICAgIGhvb2s7Ci0gICAgICAgIGludDMyX3QgICAgICAgICAqb3V0cHV0VGVtcDsKLSAgICAgICAgaW50MzJfdCAgICAgICAgICpyZXNhbXBsZVRlbXA7Ci0gICAgICAgIGludDMyX3QgICAgICAgICByZXNlcnZlZFsyXTsKLSAgICAgICAgdHJhY2tfdCAgICAgICAgIHRyYWNrc1szMl07IF9fYXR0cmlidXRlX18oKGFsaWduZWQoMzIpKSk7Ci0gICAgfTsKLQotICAgIGludCAgICAgICAgICAgICBtQWN0aXZlVHJhY2s7Ci0gICAgdWludDMyX3QgICAgICAgIG1UcmFja05hbWVzOwotICAgIGNvbnN0IHVpbnQzMl90ICBtU2FtcGxlUmF0ZTsKLQotICAgIHN0YXRlX3QgICAgICAgICBtU3RhdGUgX19hdHRyaWJ1dGVfXygoYWxpZ25lZCgzMikpKTsKLQotICAgIHZvaWQgaW52YWxpZGF0ZVN0YXRlKHVpbnQzMl90IG1hc2spOwotCi0gICAgc3RhdGljIHZvaWQgdHJhY2tfX2dlbmVyaWNSZXNhbXBsZSh0cmFja190KiB0LCBpbnQzMl90KiBvdXQsIHNpemVfdCBudW1GcmFtZXMsIGludDMyX3QqIHRlbXApOwotICAgIHN0YXRpYyB2b2lkIHRyYWNrX19ub3AodHJhY2tfdCogdCwgaW50MzJfdCogb3V0LCBzaXplX3QgbnVtRnJhbWVzLCBpbnQzMl90KiB0ZW1wKTsKLSAgICBzdGF0aWMgdm9pZCB2b2x1bWVSYW1wU3RlcmVvKHRyYWNrX3QqIHQsIGludDMyX3QqIG91dCwgc2l6ZV90IGZyYW1lQ291bnQsIGludDMyX3QqIHRlbXApOwotICAgIHN0YXRpYyB2b2lkIHRyYWNrX18xNkJpdHNTdGVyZW8odHJhY2tfdCogdCwgaW50MzJfdCogb3V0LCBzaXplX3QgbnVtRnJhbWVzLCBpbnQzMl90KiB0ZW1wKTsKLSAgICBzdGF0aWMgdm9pZCB0cmFja19fMTZCaXRzTW9ubyh0cmFja190KiB0LCBpbnQzMl90KiBvdXQsIHNpemVfdCBudW1GcmFtZXMsIGludDMyX3QqIHRlbXApOwotICAgIHN0YXRpYyB2b2lkIGRpdGhlckFuZENsYW1wKGludDMyX3QqIG91dCwgaW50MzJfdCBjb25zdCAqc3Vtcywgc2l6ZV90IGMpOwotCi0gICAgc3RhdGljIHZvaWQgcHJvY2Vzc19fdmFsaWRhdGUoc3RhdGVfdCogc3RhdGUsIHZvaWQqIG91dHB1dCk7Ci0gICAgc3RhdGljIHZvaWQgcHJvY2Vzc19fbm9wKHN0YXRlX3QqIHN0YXRlLCB2b2lkKiBvdXRwdXQpOwotICAgIHN0YXRpYyB2b2lkIHByb2Nlc3NfX2dlbmVyaWNOb1Jlc2FtcGxpbmcoc3RhdGVfdCogc3RhdGUsIHZvaWQqIG91dHB1dCk7Ci0gICAgc3RhdGljIHZvaWQgcHJvY2Vzc19fZ2VuZXJpY1Jlc2FtcGxpbmcoc3RhdGVfdCogc3RhdGUsIHZvaWQqIG91dHB1dCk7Ci0gICAgc3RhdGljIHZvaWQgcHJvY2Vzc19fT25lVHJhY2sxNkJpdHNTdGVyZW9Ob1Jlc2FtcGxpbmcoc3RhdGVfdCogc3RhdGUsIHZvaWQqIG91dHB1dCk7Ci0gICAgc3RhdGljIHZvaWQgcHJvY2Vzc19fVHdvVHJhY2tzMTZCaXRzU3RlcmVvTm9SZXNhbXBsaW5nKHN0YXRlX3QqIHN0YXRlLCB2b2lkKiBvdXRwdXQpOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfQVVESU9fTUlYRVJfSApkaWZmIC0tZ2l0IGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9SZXNhbXBsZXIuY3BwIGIvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9SZXNhbXBsZXIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1ZGFiYWNiLi4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvUmVzYW1wbGVyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDU5NSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJBdWRpb1Jlc2FtcGxlciIKLS8vI2RlZmluZSBMT0dfTkRFQlVHIDAKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPGN1dGlscy9sb2cuaD4KLSNpbmNsdWRlIDxjdXRpbHMvcHJvcGVydGllcy5oPgotI2luY2x1ZGUgIkF1ZGlvUmVzYW1wbGVyLmgiCi0jaW5jbHVkZSAiQXVkaW9SZXNhbXBsZXJTaW5jLmgiCi0jaW5jbHVkZSAiQXVkaW9SZXNhbXBsZXJDdWJpYy5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLSNpZmRlZiBfX0FSTV9BUkNIXzVFX18gIC8vIG9wdGltaXplZCBhc20gb3B0aW9uCi0gICAgI2RlZmluZSBBU01fQVJNX1JFU0FNUDEgLy8gZW5hYmxlIGFzbSBvcHRpbWlzYXRpb24gZm9yIFJlc2FtcGxlck9yZGVyMQotI2VuZGlmIC8vIF9fQVJNX0FSQ0hfNUVfXwotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBBdWRpb1Jlc2FtcGxlck9yZGVyMSA6IHB1YmxpYyBBdWRpb1Jlc2FtcGxlciB7Ci1wdWJsaWM6Ci0gICAgQXVkaW9SZXNhbXBsZXJPcmRlcjEoaW50IGJpdERlcHRoLCBpbnQgaW5DaGFubmVsQ291bnQsIGludDMyX3Qgc2FtcGxlUmF0ZSkgOgotICAgICAgICBBdWRpb1Jlc2FtcGxlcihiaXREZXB0aCwgaW5DaGFubmVsQ291bnQsIHNhbXBsZVJhdGUpLCBtWDBMKDApLCBtWDBSKDApIHsKLSAgICB9Ci0gICAgdmlydHVhbCB2b2lkIHJlc2FtcGxlKGludDMyX3QqIG91dCwgc2l6ZV90IG91dEZyYW1lQ291bnQsCi0gICAgICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcik7Ci1wcml2YXRlOgotICAgIC8vIG51bWJlciBvZiBiaXRzIHVzZWQgaW4gaW50ZXJwb2xhdGlvbiBtdWx0aXBseSAtIDE1IGJpdHMgYXZvaWRzIG92ZXJmbG93Ci0gICAgc3RhdGljIGNvbnN0IGludCBrTnVtSW50ZXJwQml0cyA9IDE1OwotCi0gICAgLy8gYml0cyB0byBzaGlmdCB0aGUgcGhhc2UgZnJhY3Rpb24gZG93biB0byBhdm9pZCBvdmVyZmxvdwotICAgIHN0YXRpYyBjb25zdCBpbnQga1ByZUludGVycFNoaWZ0ID0ga051bVBoYXNlQml0cyAtIGtOdW1JbnRlcnBCaXRzOwotCi0gICAgdm9pZCBpbml0KCkge30KLSAgICB2b2lkIHJlc2FtcGxlTW9ubzE2KGludDMyX3QqIG91dCwgc2l6ZV90IG91dEZyYW1lQ291bnQsCi0gICAgICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcik7Ci0gICAgdm9pZCByZXNhbXBsZVN0ZXJlbzE2KGludDMyX3QqIG91dCwgc2l6ZV90IG91dEZyYW1lQ291bnQsCi0gICAgICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcik7Ci0jaWZkZWYgQVNNX0FSTV9SRVNBTVAxICAvLyBhc20gb3B0aW1pc2F0aW9uIGZvciBSZXNhbXBsZXJPcmRlcjEKLSAgICB2b2lkIEFzbU1vbm8xNkxvb3AoaW50MTZfdCAqaW4sIGludDMyX3QqIG1heE91dFB0LCBpbnQzMl90IG1heEluSWR4LAotICAgICAgICAgICAgc2l6ZV90ICZvdXRwdXRJbmRleCwgaW50MzJfdCogb3V0LCBzaXplX3QgJmlucHV0SW5kZXgsIGludDMyX3QgdmwsIGludDMyX3QgdnIsCi0gICAgICAgICAgICB1aW50MzJfdCAmcGhhc2VGcmFjdGlvbiwgdWludDMyX3QgcGhhc2VJbmNyZW1lbnQpOwotICAgIHZvaWQgQXNtU3RlcmVvMTZMb29wKGludDE2X3QgKmluLCBpbnQzMl90KiBtYXhPdXRQdCwgaW50MzJfdCBtYXhJbklkeCwKLSAgICAgICAgICAgIHNpemVfdCAmb3V0cHV0SW5kZXgsIGludDMyX3QqIG91dCwgc2l6ZV90ICZpbnB1dEluZGV4LCBpbnQzMl90IHZsLCBpbnQzMl90IHZyLAotICAgICAgICAgICAgdWludDMyX3QgJnBoYXNlRnJhY3Rpb24sIHVpbnQzMl90IHBoYXNlSW5jcmVtZW50KTsKLSNlbmRpZiAgLy8gQVNNX0FSTV9SRVNBTVAxCi0KLSAgICBzdGF0aWMgaW5saW5lIGludDMyX3QgSW50ZXJwKGludDMyX3QgeDAsIGludDMyX3QgeDEsIHVpbnQzMl90IGYpIHsKLSAgICAgICAgcmV0dXJuIHgwICsgKCgoeDEgLSB4MCkgKiAoaW50MzJfdCkoZiA+PiBrUHJlSW50ZXJwU2hpZnQpKSA+PiBrTnVtSW50ZXJwQml0cyk7Ci0gICAgfQotICAgIHN0YXRpYyBpbmxpbmUgdm9pZCBBZHZhbmNlKHNpemVfdCogaW5kZXgsIHVpbnQzMl90KiBmcmFjLCB1aW50MzJfdCBpbmMpIHsKLSAgICAgICAgKmZyYWMgKz0gaW5jOwotICAgICAgICAqaW5kZXggKz0gKHNpemVfdCkoKmZyYWMgPj4ga051bVBoYXNlQml0cyk7Ci0gICAgICAgICpmcmFjICY9IGtQaGFzZU1hc2s7Ci0gICAgfQotICAgIGludCBtWDBMOwotICAgIGludCBtWDBSOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotQXVkaW9SZXNhbXBsZXIqIEF1ZGlvUmVzYW1wbGVyOjpjcmVhdGUoaW50IGJpdERlcHRoLCBpbnQgaW5DaGFubmVsQ291bnQsCi0gICAgICAgIGludDMyX3Qgc2FtcGxlUmF0ZSwgaW50IHF1YWxpdHkpIHsKLQotICAgIC8vIGNhbiBvbmx5IGNyZWF0ZSBsb3cgcXVhbGl0eSByZXNhbXBsZSBub3cKLSAgICBBdWRpb1Jlc2FtcGxlciogcmVzYW1wbGVyOwotCi0gICAgY2hhciB2YWx1ZVtQUk9QRVJUWV9WQUxVRV9NQVhdOwotICAgIGlmIChwcm9wZXJ0eV9nZXQoImFmLnJlc2FtcGxlci5xdWFsaXR5IiwgdmFsdWUsIDApKSB7Ci0gICAgICAgIHF1YWxpdHkgPSBhdG9pKHZhbHVlKTsKLSAgICAgICAgTE9HRCgiZm9yY2luZyBBdWRpb1Jlc2FtcGxlciBxdWFsaXR5IHRvICVkIiwgcXVhbGl0eSk7Ci0gICAgfQotCi0gICAgaWYgKHF1YWxpdHkgPT0gREVGQVVMVCkKLSAgICAgICAgcXVhbGl0eSA9IExPV19RVUFMSVRZOwotCi0gICAgc3dpdGNoIChxdWFsaXR5KSB7Ci0gICAgZGVmYXVsdDoKLSAgICBjYXNlIExPV19RVUFMSVRZOgotICAgICAgICBMT0dWKCJDcmVhdGUgbGluZWFyIFJlc2FtcGxlciIpOwotICAgICAgICByZXNhbXBsZXIgPSBuZXcgQXVkaW9SZXNhbXBsZXJPcmRlcjEoYml0RGVwdGgsIGluQ2hhbm5lbENvdW50LCBzYW1wbGVSYXRlKTsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBNRURfUVVBTElUWToKLSAgICAgICAgTE9HVigiQ3JlYXRlIGN1YmljIFJlc2FtcGxlciIpOwotICAgICAgICByZXNhbXBsZXIgPSBuZXcgQXVkaW9SZXNhbXBsZXJDdWJpYyhiaXREZXB0aCwgaW5DaGFubmVsQ291bnQsIHNhbXBsZVJhdGUpOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEhJR0hfUVVBTElUWToKLSAgICAgICAgTE9HVigiQ3JlYXRlIHNpbmMgUmVzYW1wbGVyIik7Ci0gICAgICAgIHJlc2FtcGxlciA9IG5ldyBBdWRpb1Jlc2FtcGxlclNpbmMoYml0RGVwdGgsIGluQ2hhbm5lbENvdW50LCBzYW1wbGVSYXRlKTsKLSAgICAgICAgYnJlYWs7Ci0gICAgfQotCi0gICAgLy8gaW5pdGlhbGl6ZSByZXNhbXBsZXIKLSAgICByZXNhbXBsZXItPmluaXQoKTsKLSAgICByZXR1cm4gcmVzYW1wbGVyOwotfQotCi1BdWRpb1Jlc2FtcGxlcjo6QXVkaW9SZXNhbXBsZXIoaW50IGJpdERlcHRoLCBpbnQgaW5DaGFubmVsQ291bnQsCi0gICAgICAgIGludDMyX3Qgc2FtcGxlUmF0ZSkgOgotICAgIG1CaXREZXB0aChiaXREZXB0aCksIG1DaGFubmVsQ291bnQoaW5DaGFubmVsQ291bnQpLAotICAgICAgICAgICAgbVNhbXBsZVJhdGUoc2FtcGxlUmF0ZSksIG1JblNhbXBsZVJhdGUoc2FtcGxlUmF0ZSksIG1JbnB1dEluZGV4KDApLAotICAgICAgICAgICAgbVBoYXNlRnJhY3Rpb24oMCkgewotICAgIC8vIHNhbml0eSBjaGVjayBvbiBmb3JtYXQKLSAgICBpZiAoKGJpdERlcHRoICE9IDE2KSB8fChpbkNoYW5uZWxDb3VudCA8IDEpIHx8IChpbkNoYW5uZWxDb3VudCA+IDIpKSB7Ci0gICAgICAgIExPR0UoIlVuc3VwcG9ydGVkIHNhbXBsZSBmb3JtYXQsICVkIGJpdHMsICVkIGNoYW5uZWxzIiwgYml0RGVwdGgsCi0gICAgICAgICAgICAgICAgaW5DaGFubmVsQ291bnQpOwotICAgICAgICAvLyBMT0dfQVNTRVJUKDApOwotICAgIH0KLQotICAgIC8vIGluaXRpYWxpemUgY29tbW9uIG1lbWJlcnMKLSAgICBtVm9sdW1lWzBdID0gbVZvbHVtZVsxXSA9IDA7Ci0gICAgbUJ1ZmZlci5mcmFtZUNvdW50ID0gMDsKLQotICAgIC8vIHNhdmUgZm9ybWF0IGZvciBxdWljayBsb29rdXAKLSAgICBpZiAoaW5DaGFubmVsQ291bnQgPT0gMSkgewotICAgICAgICBtRm9ybWF0ID0gTU9OT18xNl9CSVQ7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgbUZvcm1hdCA9IFNURVJFT18xNl9CSVQ7Ci0gICAgfQotfQotCi1BdWRpb1Jlc2FtcGxlcjo6fkF1ZGlvUmVzYW1wbGVyKCkgewotfQotCi12b2lkIEF1ZGlvUmVzYW1wbGVyOjpzZXRTYW1wbGVSYXRlKGludDMyX3QgaW5TYW1wbGVSYXRlKSB7Ci0gICAgbUluU2FtcGxlUmF0ZSA9IGluU2FtcGxlUmF0ZTsKLSAgICBtUGhhc2VJbmNyZW1lbnQgPSAodWludDMyX3QpKChrUGhhc2VNdWx0aXBsaWVyICogaW5TYW1wbGVSYXRlKSAvIG1TYW1wbGVSYXRlKTsKLX0KLQotdm9pZCBBdWRpb1Jlc2FtcGxlcjo6c2V0Vm9sdW1lKGludDE2X3QgbGVmdCwgaW50MTZfdCByaWdodCkgewotICAgIC8vIFRPRE86IEltcGxlbWVudCBhbnRpLXppcHBlciBmaWx0ZXIKLSAgICBtVm9sdW1lWzBdID0gbGVmdDsKLSAgICBtVm9sdW1lWzFdID0gcmlnaHQ7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdm9pZCBBdWRpb1Jlc2FtcGxlck9yZGVyMTo6cmVzYW1wbGUoaW50MzJfdCogb3V0LCBzaXplX3Qgb3V0RnJhbWVDb3VudCwKLSAgICAgICAgQXVkaW9CdWZmZXJQcm92aWRlciogcHJvdmlkZXIpIHsKLQotICAgIC8vIHNob3VsZCBuZXZlciBoYXBwZW4sIGJ1dCB3ZSBvdmVyZmxvdyBpZiBpdCBkb2VzCi0gICAgLy8gTE9HX0FTU0VSVChvdXRGcmFtZUNvdW50IDwgMzI3NjcpOwotCi0gICAgLy8gc2VsZWN0IHRoZSBhcHByb3ByaWF0ZSByZXNhbXBsZXIKLSAgICBzd2l0Y2ggKG1DaGFubmVsQ291bnQpIHsKLSAgICBjYXNlIDE6Ci0gICAgICAgIHJlc2FtcGxlTW9ubzE2KG91dCwgb3V0RnJhbWVDb3VudCwgcHJvdmlkZXIpOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIDI6Ci0gICAgICAgIHJlc2FtcGxlU3RlcmVvMTYob3V0LCBvdXRGcmFtZUNvdW50LCBwcm92aWRlcik7Ci0gICAgICAgIGJyZWFrOwotICAgIH0KLX0KLQotdm9pZCBBdWRpb1Jlc2FtcGxlck9yZGVyMTo6cmVzYW1wbGVTdGVyZW8xNihpbnQzMl90KiBvdXQsIHNpemVfdCBvdXRGcmFtZUNvdW50LAotICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcikgewotCi0gICAgaW50MzJfdCB2bCA9IG1Wb2x1bWVbMF07Ci0gICAgaW50MzJfdCB2ciA9IG1Wb2x1bWVbMV07Ci0KLSAgICBzaXplX3QgaW5wdXRJbmRleCA9IG1JbnB1dEluZGV4OwotICAgIHVpbnQzMl90IHBoYXNlRnJhY3Rpb24gPSBtUGhhc2VGcmFjdGlvbjsKLSAgICB1aW50MzJfdCBwaGFzZUluY3JlbWVudCA9IG1QaGFzZUluY3JlbWVudDsKLSAgICBzaXplX3Qgb3V0cHV0SW5kZXggPSAwOwotICAgIHNpemVfdCBvdXRwdXRTYW1wbGVDb3VudCA9IG91dEZyYW1lQ291bnQgKiAyOwotICAgIHNpemVfdCBpbkZyYW1lQ291bnQgPSAob3V0RnJhbWVDb3VudCptSW5TYW1wbGVSYXRlKS9tU2FtcGxlUmF0ZTsKLQotICAgIC8vIExPR0UoInN0YXJ0aW5nIHJlc2FtcGxlICVkIGZyYW1lcywgaW5wdXRJbmRleD0lZCwgcGhhc2VGcmFjdGlvbj0lZCwgcGhhc2VJbmNyZW1lbnQ9JWRcbiIsCi0gICAgLy8gICAgICBvdXRGcmFtZUNvdW50LCBpbnB1dEluZGV4LCBwaGFzZUZyYWN0aW9uLCBwaGFzZUluY3JlbWVudCk7Ci0KLSAgICB3aGlsZSAob3V0cHV0SW5kZXggPCBvdXRwdXRTYW1wbGVDb3VudCkgewotCi0gICAgICAgIC8vIGJ1ZmZlciBpcyBlbXB0eSwgZmV0Y2ggYSBuZXcgb25lCi0gICAgICAgIHdoaWxlIChtQnVmZmVyLmZyYW1lQ291bnQgPT0gMCkgewotICAgICAgICAgICAgbUJ1ZmZlci5mcmFtZUNvdW50ID0gaW5GcmFtZUNvdW50OwotICAgICAgICAgICAgcHJvdmlkZXItPmdldE5leHRCdWZmZXIoJm1CdWZmZXIpOwotICAgICAgICAgICAgaWYgKG1CdWZmZXIucmF3ID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICBnb3RvIHJlc2FtcGxlU3RlcmVvMTZfZXhpdDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gTE9HRSgiTmV3IGJ1ZmZlciBmZXRjaGVkOiAlZCBmcmFtZXNcbiIsIG1CdWZmZXIuZnJhbWVDb3VudCk7Ci0gICAgICAgICAgICBpZiAobUJ1ZmZlci5mcmFtZUNvdW50ID4gaW5wdXRJbmRleCkgYnJlYWs7Ci0KLSAgICAgICAgICAgIGlucHV0SW5kZXggLT0gbUJ1ZmZlci5mcmFtZUNvdW50OwotICAgICAgICAgICAgbVgwTCA9IG1CdWZmZXIuaTE2W21CdWZmZXIuZnJhbWVDb3VudCoyLTJdOwotICAgICAgICAgICAgbVgwUiA9IG1CdWZmZXIuaTE2W21CdWZmZXIuZnJhbWVDb3VudCoyLTFdOwotICAgICAgICAgICAgcHJvdmlkZXItPnJlbGVhc2VCdWZmZXIoJm1CdWZmZXIpOwotICAgICAgICAgICAgIC8vIG1CdWZmZXIuZnJhbWVDb3VudCA9PSAwIG5vdyBzbyB3ZSByZWxvYWQgYSBuZXcgYnVmZmVyCi0gICAgICAgIH0KLQotICAgICAgICBpbnQxNl90ICppbiA9IG1CdWZmZXIuaTE2OwotCi0gICAgICAgIC8vIGhhbmRsZSBib3VuZGFyeSBjYXNlCi0gICAgICAgIHdoaWxlIChpbnB1dEluZGV4ID09IDApIHsKLSAgICAgICAgICAgIC8vIExPR0UoImJvdW5kYXJ5IGNhc2VcbiIpOwotICAgICAgICAgICAgb3V0W291dHB1dEluZGV4KytdICs9IHZsICogSW50ZXJwKG1YMEwsIGluWzBdLCBwaGFzZUZyYWN0aW9uKTsKLSAgICAgICAgICAgIG91dFtvdXRwdXRJbmRleCsrXSArPSB2ciAqIEludGVycChtWDBSLCBpblsxXSwgcGhhc2VGcmFjdGlvbik7Ci0gICAgICAgICAgICBBZHZhbmNlKCZpbnB1dEluZGV4LCAmcGhhc2VGcmFjdGlvbiwgcGhhc2VJbmNyZW1lbnQpOwotICAgICAgICAgICAgaWYgKG91dHB1dEluZGV4ID09IG91dHB1dFNhbXBsZUNvdW50KQotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gcHJvY2VzcyBpbnB1dCBzYW1wbGVzCi0gICAgICAgIC8vIExPR0UoImdlbmVyYWwgY2FzZVxuIik7Ci0KLSNpZmRlZiBBU01fQVJNX1JFU0FNUDEgIC8vIGFzbSBvcHRpbWlzYXRpb24gZm9yIFJlc2FtcGxlck9yZGVyMQotICAgICAgICBpZiAoaW5wdXRJbmRleCArIDIgPCBtQnVmZmVyLmZyYW1lQ291bnQpIHsKLSAgICAgICAgICAgIGludDMyX3QqIG1heE91dFB0OwotICAgICAgICAgICAgaW50MzJfdCBtYXhJbklkeDsKLQotICAgICAgICAgICAgbWF4T3V0UHQgPSBvdXQgKyAob3V0cHV0U2FtcGxlQ291bnQgLSAyKTsgICAvLyAyIGJlY2F1c2UgMiBmcmFtZXMgcGVyIGxvb3AKLSAgICAgICAgICAgIG1heEluSWR4ID0gbUJ1ZmZlci5mcmFtZUNvdW50IC0gMjsKLSAgICAgICAgICAgIEFzbVN0ZXJlbzE2TG9vcChpbiwgbWF4T3V0UHQsIG1heEluSWR4LCBvdXRwdXRJbmRleCwgb3V0LCBpbnB1dEluZGV4LCB2bCwgdnIsCi0gICAgICAgICAgICAgICAgICAgIHBoYXNlRnJhY3Rpb24sIHBoYXNlSW5jcmVtZW50KTsKLSAgICAgICAgfQotI2VuZGlmICAvLyBBU01fQVJNX1JFU0FNUDEKLQotICAgICAgICB3aGlsZSAob3V0cHV0SW5kZXggPCBvdXRwdXRTYW1wbGVDb3VudCAmJiBpbnB1dEluZGV4IDwgbUJ1ZmZlci5mcmFtZUNvdW50KSB7Ci0gICAgICAgICAgICBvdXRbb3V0cHV0SW5kZXgrK10gKz0gdmwgKiBJbnRlcnAoaW5baW5wdXRJbmRleCoyLTJdLAotICAgICAgICAgICAgICAgICAgICBpbltpbnB1dEluZGV4KjJdLCBwaGFzZUZyYWN0aW9uKTsKLSAgICAgICAgICAgIG91dFtvdXRwdXRJbmRleCsrXSArPSB2ciAqIEludGVycChpbltpbnB1dEluZGV4KjItMV0sCi0gICAgICAgICAgICAgICAgICAgIGluW2lucHV0SW5kZXgqMisxXSwgcGhhc2VGcmFjdGlvbik7Ci0gICAgICAgICAgICBBZHZhbmNlKCZpbnB1dEluZGV4LCAmcGhhc2VGcmFjdGlvbiwgcGhhc2VJbmNyZW1lbnQpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gTE9HRSgibG9vcCBkb25lIC0gb3V0cHV0SW5kZXg9JWQsIGlucHV0SW5kZXg9JWRcbiIsIG91dHB1dEluZGV4LCBpbnB1dEluZGV4KTsKLQotICAgICAgICAvLyBpZiBkb25lIHdpdGggYnVmZmVyLCBzYXZlIHNhbXBsZXMKLSAgICAgICAgaWYgKGlucHV0SW5kZXggPj0gbUJ1ZmZlci5mcmFtZUNvdW50KSB7Ci0gICAgICAgICAgICBpbnB1dEluZGV4IC09IG1CdWZmZXIuZnJhbWVDb3VudDsKLQotICAgICAgICAgICAgLy8gTE9HRSgiYnVmZmVyIGRvbmUsIG5ldyBpbnB1dCBpbmRleCAlZCIsIGlucHV0SW5kZXgpOwotCi0gICAgICAgICAgICBtWDBMID0gbUJ1ZmZlci5pMTZbbUJ1ZmZlci5mcmFtZUNvdW50KjItMl07Ci0gICAgICAgICAgICBtWDBSID0gbUJ1ZmZlci5pMTZbbUJ1ZmZlci5mcmFtZUNvdW50KjItMV07Ci0gICAgICAgICAgICBwcm92aWRlci0+cmVsZWFzZUJ1ZmZlcigmbUJ1ZmZlcik7Ci0KLSAgICAgICAgICAgIC8vIHZlcmlmeSB0aGF0IHRoZSByZWxlYXNlQnVmZmVyIHJlc2V0cyB0aGUgYnVmZmVyIGZyYW1lQ291bnQKLSAgICAgICAgICAgIC8vIExPR19BU1NFUlQobUJ1ZmZlci5mcmFtZUNvdW50ID09IDApOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gTE9HRSgib3V0cHV0IGJ1ZmZlciBmdWxsIC0gb3V0cHV0SW5kZXg9JWQsIGlucHV0SW5kZXg9JWRcbiIsIG91dHB1dEluZGV4LCBpbnB1dEluZGV4KTsKLQotcmVzYW1wbGVTdGVyZW8xNl9leGl0OgotICAgIC8vIHNhdmUgc3RhdGUKLSAgICBtSW5wdXRJbmRleCA9IGlucHV0SW5kZXg7Ci0gICAgbVBoYXNlRnJhY3Rpb24gPSBwaGFzZUZyYWN0aW9uOwotfQotCi12b2lkIEF1ZGlvUmVzYW1wbGVyT3JkZXIxOjpyZXNhbXBsZU1vbm8xNihpbnQzMl90KiBvdXQsIHNpemVfdCBvdXRGcmFtZUNvdW50LAotICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcikgewotCi0gICAgaW50MzJfdCB2bCA9IG1Wb2x1bWVbMF07Ci0gICAgaW50MzJfdCB2ciA9IG1Wb2x1bWVbMV07Ci0KLSAgICBzaXplX3QgaW5wdXRJbmRleCA9IG1JbnB1dEluZGV4OwotICAgIHVpbnQzMl90IHBoYXNlRnJhY3Rpb24gPSBtUGhhc2VGcmFjdGlvbjsKLSAgICB1aW50MzJfdCBwaGFzZUluY3JlbWVudCA9IG1QaGFzZUluY3JlbWVudDsKLSAgICBzaXplX3Qgb3V0cHV0SW5kZXggPSAwOwotICAgIHNpemVfdCBvdXRwdXRTYW1wbGVDb3VudCA9IG91dEZyYW1lQ291bnQgKiAyOwotICAgIHNpemVfdCBpbkZyYW1lQ291bnQgPSAob3V0RnJhbWVDb3VudCptSW5TYW1wbGVSYXRlKS9tU2FtcGxlUmF0ZTsKLQotICAgIC8vIExPR0UoInN0YXJ0aW5nIHJlc2FtcGxlICVkIGZyYW1lcywgaW5wdXRJbmRleD0lZCwgcGhhc2VGcmFjdGlvbj0lZCwgcGhhc2VJbmNyZW1lbnQ9JWRcbiIsCi0gICAgLy8gICAgICBvdXRGcmFtZUNvdW50LCBpbnB1dEluZGV4LCBwaGFzZUZyYWN0aW9uLCBwaGFzZUluY3JlbWVudCk7Ci0gICAgd2hpbGUgKG91dHB1dEluZGV4IDwgb3V0cHV0U2FtcGxlQ291bnQpIHsKLSAgICAgICAgLy8gYnVmZmVyIGlzIGVtcHR5LCBmZXRjaCBhIG5ldyBvbmUKLSAgICAgICAgd2hpbGUgKG1CdWZmZXIuZnJhbWVDb3VudCA9PSAwKSB7Ci0gICAgICAgICAgICBtQnVmZmVyLmZyYW1lQ291bnQgPSBpbkZyYW1lQ291bnQ7Ci0gICAgICAgICAgICBwcm92aWRlci0+Z2V0TmV4dEJ1ZmZlcigmbUJ1ZmZlcik7Ci0gICAgICAgICAgICBpZiAobUJ1ZmZlci5yYXcgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgIG1JbnB1dEluZGV4ID0gaW5wdXRJbmRleDsKLSAgICAgICAgICAgICAgICBtUGhhc2VGcmFjdGlvbiA9IHBoYXNlRnJhY3Rpb247Ci0gICAgICAgICAgICAgICAgZ290byByZXNhbXBsZU1vbm8xNl9leGl0OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgLy8gTE9HRSgiTmV3IGJ1ZmZlciBmZXRjaGVkOiAlZCBmcmFtZXNcbiIsIG1CdWZmZXIuZnJhbWVDb3VudCk7Ci0gICAgICAgICAgICBpZiAobUJ1ZmZlci5mcmFtZUNvdW50ID4gIGlucHV0SW5kZXgpIGJyZWFrOwotCi0gICAgICAgICAgICBpbnB1dEluZGV4IC09IG1CdWZmZXIuZnJhbWVDb3VudDsKLSAgICAgICAgICAgIG1YMEwgPSBtQnVmZmVyLmkxNlttQnVmZmVyLmZyYW1lQ291bnQtMV07Ci0gICAgICAgICAgICBwcm92aWRlci0+cmVsZWFzZUJ1ZmZlcigmbUJ1ZmZlcik7Ci0gICAgICAgICAgICAvLyBtQnVmZmVyLmZyYW1lQ291bnQgPT0gMCBub3cgc28gd2UgcmVsb2FkIGEgbmV3IGJ1ZmZlcgotICAgICAgICB9Ci0gICAgICAgIGludDE2X3QgKmluID0gbUJ1ZmZlci5pMTY7Ci0KLSAgICAgICAgLy8gaGFuZGxlIGJvdW5kYXJ5IGNhc2UKLSAgICAgICAgd2hpbGUgKGlucHV0SW5kZXggPT0gMCkgewotICAgICAgICAgICAgLy8gTE9HRSgiYm91bmRhcnkgY2FzZVxuIik7Ci0gICAgICAgICAgICBpbnQzMl90IHNhbXBsZSA9IEludGVycChtWDBMLCBpblswXSwgcGhhc2VGcmFjdGlvbik7Ci0gICAgICAgICAgICBvdXRbb3V0cHV0SW5kZXgrK10gKz0gdmwgKiBzYW1wbGU7Ci0gICAgICAgICAgICBvdXRbb3V0cHV0SW5kZXgrK10gKz0gdnIgKiBzYW1wbGU7Ci0gICAgICAgICAgICBBZHZhbmNlKCZpbnB1dEluZGV4LCAmcGhhc2VGcmFjdGlvbiwgcGhhc2VJbmNyZW1lbnQpOwotICAgICAgICAgICAgaWYgKG91dHB1dEluZGV4ID09IG91dHB1dFNhbXBsZUNvdW50KQotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gcHJvY2VzcyBpbnB1dCBzYW1wbGVzCi0gICAgICAgIC8vIExPR0UoImdlbmVyYWwgY2FzZVxuIik7Ci0KLSNpZmRlZiBBU01fQVJNX1JFU0FNUDEgIC8vIGFzbSBvcHRpbWlzYXRpb24gZm9yIFJlc2FtcGxlck9yZGVyMQotICAgICAgICBpZiAoaW5wdXRJbmRleCArIDIgPCBtQnVmZmVyLmZyYW1lQ291bnQpIHsKLSAgICAgICAgICAgIGludDMyX3QqIG1heE91dFB0OwotICAgICAgICAgICAgaW50MzJfdCBtYXhJbklkeDsKLQotICAgICAgICAgICAgbWF4T3V0UHQgPSBvdXQgKyAob3V0cHV0U2FtcGxlQ291bnQgLSAyKTsKLSAgICAgICAgICAgIG1heEluSWR4ID0gKGludDMyX3QpbUJ1ZmZlci5mcmFtZUNvdW50IC0gMjsKLSAgICAgICAgICAgICAgICBBc21Nb25vMTZMb29wKGluLCBtYXhPdXRQdCwgbWF4SW5JZHgsIG91dHB1dEluZGV4LCBvdXQsIGlucHV0SW5kZXgsIHZsLCB2ciwKLSAgICAgICAgICAgICAgICAgICAgICAgIHBoYXNlRnJhY3Rpb24sIHBoYXNlSW5jcmVtZW50KTsKLSAgICAgICAgfQotI2VuZGlmICAvLyBBU01fQVJNX1JFU0FNUDEKLQotICAgICAgICB3aGlsZSAob3V0cHV0SW5kZXggPCBvdXRwdXRTYW1wbGVDb3VudCAmJiBpbnB1dEluZGV4IDwgbUJ1ZmZlci5mcmFtZUNvdW50KSB7Ci0gICAgICAgICAgICBpbnQzMl90IHNhbXBsZSA9IEludGVycChpbltpbnB1dEluZGV4LTFdLCBpbltpbnB1dEluZGV4XSwKLSAgICAgICAgICAgICAgICAgICAgcGhhc2VGcmFjdGlvbik7Ci0gICAgICAgICAgICBvdXRbb3V0cHV0SW5kZXgrK10gKz0gdmwgKiBzYW1wbGU7Ci0gICAgICAgICAgICBvdXRbb3V0cHV0SW5kZXgrK10gKz0gdnIgKiBzYW1wbGU7Ci0gICAgICAgICAgICBBZHZhbmNlKCZpbnB1dEluZGV4LCAmcGhhc2VGcmFjdGlvbiwgcGhhc2VJbmNyZW1lbnQpOwotICAgICAgICB9Ci0KLQotICAgICAgICAvLyBMT0dFKCJsb29wIGRvbmUgLSBvdXRwdXRJbmRleD0lZCwgaW5wdXRJbmRleD0lZFxuIiwgb3V0cHV0SW5kZXgsIGlucHV0SW5kZXgpOwotCi0gICAgICAgIC8vIGlmIGRvbmUgd2l0aCBidWZmZXIsIHNhdmUgc2FtcGxlcwotICAgICAgICBpZiAoaW5wdXRJbmRleCA+PSBtQnVmZmVyLmZyYW1lQ291bnQpIHsKLSAgICAgICAgICAgIGlucHV0SW5kZXggLT0gbUJ1ZmZlci5mcmFtZUNvdW50OwotCi0gICAgICAgICAgICAvLyBMT0dFKCJidWZmZXIgZG9uZSwgbmV3IGlucHV0IGluZGV4ICVkIiwgaW5wdXRJbmRleCk7Ci0KLSAgICAgICAgICAgIG1YMEwgPSBtQnVmZmVyLmkxNlttQnVmZmVyLmZyYW1lQ291bnQtMV07Ci0gICAgICAgICAgICBwcm92aWRlci0+cmVsZWFzZUJ1ZmZlcigmbUJ1ZmZlcik7Ci0KLSAgICAgICAgICAgIC8vIHZlcmlmeSB0aGF0IHRoZSByZWxlYXNlQnVmZmVyIHJlc2V0cyB0aGUgYnVmZmVyIGZyYW1lQ291bnQKLSAgICAgICAgICAgIC8vIExPR19BU1NFUlQobUJ1ZmZlci5mcmFtZUNvdW50ID09IDApOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gTE9HRSgib3V0cHV0IGJ1ZmZlciBmdWxsIC0gb3V0cHV0SW5kZXg9JWQsIGlucHV0SW5kZXg9JWRcbiIsIG91dHB1dEluZGV4LCBpbnB1dEluZGV4KTsKLQotcmVzYW1wbGVNb25vMTZfZXhpdDoKLSAgICAvLyBzYXZlIHN0YXRlCi0gICAgbUlucHV0SW5kZXggPSBpbnB1dEluZGV4OwotICAgIG1QaGFzZUZyYWN0aW9uID0gcGhhc2VGcmFjdGlvbjsKLX0KLQotI2lmZGVmIEFTTV9BUk1fUkVTQU1QMSAgLy8gYXNtIG9wdGltaXNhdGlvbiBmb3IgUmVzYW1wbGVyT3JkZXIxCi0KLS8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCi0qCi0qICAgQXNtTW9ubzE2TG9vcAotKiAgIGFzbSBvcHRpbWl6ZWQgbW9ub3RvbmljIGxvb3AgdmVyc2lvbjsgb25lIGxvb3AgaXMgMiBmcmFtZXMKLSogICBJbnB1dDoKLSogICAgICAgaW4gOiBwb2ludGVyIG9uIGlucHV0IHNhbXBsZXMKLSogICAgICAgbWF4T3V0UHQgOiBwb2ludGVyIG9uIGZpcnN0IG5vdCBmaWxsZWQKLSogICAgICAgbWF4SW5JZHggOiBpbmRleCBvbiBmaXJzdCBub3QgdXNlZAotKiAgICAgICBvdXRwdXRJbmRleCA6IHBvaW50ZXIgb24gY3VycmVudCBvdXRwdXQgaW5kZXgKLSogICAgICAgb3V0IDogcG9pbnRlciBvbiBvdXRwdXQgYnVmZmVyCi0qICAgICAgIGlucHV0SW5kZXggOiBwb2ludGVyIG9uIGN1cnJlbnQgaW5wdXQgaW5kZXgKLSogICAgICAgdmwsIHZyIDogbGVmdCBhbmQgcmlnaHQgZ2FpbgotKiAgICAgICBwaGFzZUZyYWN0aW9uIDogcG9pbnRlciBvbiBjdXJyZW50IHBoYXNlIGZyYWN0aW9uCi0qICAgICAgIHBoYXNlSW5jcmVtZW50Ci0qICAgT3VwdXQ6Ci0qICAgICAgIG91dHB1dEluZGV4IDoKLSogICAgICAgb3V0IDogdXBkYXRlZCBidWZmZXIKLSogICAgICAgaW5wdXRJbmRleCA6IGluZGV4IG9mIG5leHQgdG8gdXNlCi0qICAgICAgIHBoYXNlRnJhY3Rpb24gOiBwaGFzZSBmcmFjdGlvbiBmb3IgbmV4dCBpbnRlcnBvbGF0aW9uCi0qCi0qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotdm9pZCBBdWRpb1Jlc2FtcGxlck9yZGVyMTo6QXNtTW9ubzE2TG9vcChpbnQxNl90ICppbiwgaW50MzJfdCogbWF4T3V0UHQsIGludDMyX3QgbWF4SW5JZHgsCi0gICAgICAgICAgICBzaXplX3QgJm91dHB1dEluZGV4LCBpbnQzMl90KiBvdXQsIHNpemVfdCAmaW5wdXRJbmRleCwgaW50MzJfdCB2bCwgaW50MzJfdCB2ciwKLSAgICAgICAgICAgIHVpbnQzMl90ICZwaGFzZUZyYWN0aW9uLCB1aW50MzJfdCBwaGFzZUluY3JlbWVudCkKLXsKLSNkZWZpbmUgTU9fUEFSQU01ICAgIjM2IiAgICAgICAgLy8gb2Zmc2V0IG9mIHBhcmFtZXRlciA1IChvdXRwdXRJbmRleCkKLQotICAgIGFzbSgKLSAgICAgICAgInN0bWZkICBzcCEsIHtyNCwgcjUsIHI2LCByNywgcjgsIHI5LCByMTAsIHIxMSwgbHJ9XG4iCi0gICAgICAgIC8vIGdldCBwYXJhbWV0ZXJzCi0gICAgICAgICIgICBsZHIgcjYsIFtzcCwgIyIgTU9fUEFSQU01ICIgKyAyMF1cbiIgICAgLy8gJnBoYXNlRnJhY3Rpb24KLSAgICAgICAgIiAgIGxkciByNiwgW3I2XVxuIiAgICAgICAgICAgICAgICAgICAgICAgICAvLyBwaGFzZUZyYWN0aW9uCi0gICAgICAgICIgICBsZHIgcjcsIFtzcCwgIyIgTU9fUEFSQU01ICIgKyA4XVxuIiAgICAgLy8gJmlucHV0SW5kZXgKLSAgICAgICAgIiAgIGxkciByNywgW3I3XVxuIiAgICAgICAgICAgICAgICAgICAgICAgICAvLyBpbnB1dEluZGV4Ci0gICAgICAgICIgICBsZHIgcjgsIFtzcCwgIyIgTU9fUEFSQU01ICIgKyA0XVxuIiAgICAgLy8gb3V0Ci0gICAgICAgICIgICBsZHIgcjAsIFtzcCwgIyIgTU9fUEFSQU01ICIgKyAwXVxuIiAgICAgLy8gJm91dHB1dEluZGV4Ci0gICAgICAgICIgICBsZHIgcjAsIFtyMF1cbiIgICAgICAgICAgICAgICAgICAgICAgICAgLy8gb3V0cHV0SW5kZXgKLSAgICAgICAgIiAgIGFkZCByOCwgcjAsIGFzbCAjMlxuIiAgICAgICAgICAgICAgICAgICAvLyBjdXJPdXQKLSAgICAgICAgIiAgIGxkciByOSwgW3NwLCAjIiBNT19QQVJBTTUgIiArIDI0XVxuIiAgICAvLyBwaGFzZUluY3JlbWVudAotICAgICAgICAiICAgbGRyIHIxMCwgW3NwLCAjIiBNT19QQVJBTTUgIiArIDEyXVxuIiAgIC8vIHZsCi0gICAgICAgICIgICBsZHIgcjExLCBbc3AsICMiIE1PX1BBUkFNNSAiICsgMTZdXG4iICAgLy8gdnIKLQotICAgICAgICAvLyByMCBwaW4sIHgwLCBTYW1wCi0KLSAgICAgICAgLy8gcjEgaW4KLSAgICAgICAgLy8gcjIgbWF4T3V0UHQKLSAgICAgICAgLy8gcjMgbWF4SW5JZHgKLQotICAgICAgICAvLyByNCB4MSwgaTEsIGkzLCBPdXQxCi0gICAgICAgIC8vIHI1IG91dDAKLQotICAgICAgICAvLyByNiBmcmFjCi0gICAgICAgIC8vIHI3IGlucHV0SW5kZXgKLSAgICAgICAgLy8gcjggY3VyT3V0Ci0KLSAgICAgICAgLy8gcjkgaW5jCi0gICAgICAgIC8vIHIxMCB2bAotICAgICAgICAvLyByMTEgdnIKLQotICAgICAgICAvLyByMTIKLSAgICAgICAgLy8gcjEzIHNwCi0gICAgICAgIC8vIHIxNAotCi0gICAgICAgIC8vIHRoZSBmb2xsb3dpbmcgbG9vcCB3b3JrcyBvbiAyIGZyYW1lcwotCi0gICAgICAgICIuWTRMMDE6XG4iCi0gICAgICAgICIgICBjbXAgcjgsIHIyXG4iICAgICAgICAgICAgICAgICAgIC8vIGN1ck91dCAtIG1heEN1ck91dAotICAgICAgICAiICAgYmNzIC5ZNEwwMlxuIgotCi0jZGVmaW5lIE1PX09ORV9GUkFNRSBcCi0gICAgIiAgIGFkZCByMCwgcjEsIHI3LCBhc2wgIzFcbiIgICAgICAgLyogaW4gKyBpbnB1dEluZGV4ICovXAotICAgICIgICBsZHJzaCByNCwgW3IwXVxuIiAgICAgICAgICAgICAgIC8qIGluW2lucHV0SW5kZXhdICovXAotICAgICIgICBsZHIgcjUsIFtyOF1cbiIgICAgICAgICAgICAgICAgIC8qIG91dFtvdXRwdXRJbmRleF0gKi9cCi0gICAgIiAgIGxkcnNoIHIwLCBbcjAsICMtMl1cbiIgICAgICAgICAgLyogaW5baW5wdXRJbmRleC0xXSAqL1wKLSAgICAiICAgYmljIHI2LCByNiwgIzB4QzAwMDAwMDBcbiIgICAgICAvKiBwaGFzZUZyYWN0aW9uICYgLi4uICovXAotICAgICIgICBzdWIgcjQsIHI0LCByMFxuIiAgICAgICAgICAgICAgIC8qIGluW2lucHV0SW5kZXhdIC0gaW5baW5wdXRJbmRleC0xXSAqL1wKLSAgICAiICAgbW92IHI0LCByNCwgbHNsICMyXG4iICAgICAgICAgICAvKiA8PDIgKi9cCi0gICAgIiAgIHNtdWx3dCByNCwgcjQsIHI2XG4iICAgICAgICAgICAgLyogKHgxLXgwKSouLiAqL1wKLSAgICAiICAgYWRkIHI2LCByNiwgcjlcbiIgICAgICAgICAgICAgICAvKiBwaGFzZUZyYWN0aW9uICsgcGhhc2VJbmNyZW1lbnQgKi9cCi0gICAgIiAgIGFkZCByMCwgcjAsIHI0XG4iICAgICAgICAgICAgICAgLyogeDAgLSAoLi4pICovXAotICAgICIgICBtbGEgcjUsIHIwLCByMTAsIHI1XG4iICAgICAgICAgIC8qIHZsKmludGVycCArIG91dFtdICovXAotICAgICIgICBsZHIgcjQsIFtyOCwgIzRdXG4iICAgICAgICAgICAgIC8qIG91dFtvdXRwdXRJbmRleCsxXSAqL1wKLSAgICAiICAgc3RyIHI1LCBbcjhdLCAjNFxuIiAgICAgICAgICAgICAvKiBvdXRbb3V0cHV0SW5kZXgrK10gPSAuLi4gKi9cCi0gICAgIiAgIG1sYSByNCwgcjAsIHIxMSwgcjRcbiIgICAgICAgICAgLyogdnIqaW50ZXJwICsgb3V0W10gKi9cCi0gICAgIiAgIGFkZCByNywgcjcsIHI2LCBsc3IgIzMwXG4iICAgICAgLyogaW5wdXRJbmRleCArIHBoYXNlRnJhY3Rpb24+PjMwICovXAotICAgICIgICBzdHIgcjQsIFtyOF0sICM0XG4iICAgICAgICAgICAgIC8qIG91dFtvdXRwdXRJbmRleCsrXSA9IC4uLiAqLwotCi0gICAgICAgIE1PX09ORV9GUkFNRSAgICAvLyBmcmFtZSAxCi0gICAgICAgIE1PX09ORV9GUkFNRSAgICAvLyBmcmFtZSAyCi0KLSAgICAgICAgIiAgIGNtcCByNywgcjNcbiIgICAgICAgICAgICAgICAgICAgLy8gaW5wdXRJbmRleCAtIG1heEluSWR4Ci0gICAgICAgICIgICBiY2MgLlk0TDAxXG4iCi0gICAgICAgICIuWTRMMDI6XG4iCi0KLSAgICAgICAgIiAgIGJpYyByNiwgcjYsICMweEMwMDAwMDAwXG4iICAgICAgICAgICAgIC8vIHBoYXNlRnJhY3Rpb24gJiAuLi4KLSAgICAgICAgLy8gc2F2ZSBtb2RpZmllZCB2YWx1ZXMKLSAgICAgICAgIiAgIGxkciByMCwgW3NwLCAjIiBNT19QQVJBTTUgIiArIDIwXVxuIiAgICAvLyAmcGhhc2VGcmFjdGlvbgotICAgICAgICAiICAgc3RyIHI2LCBbcjBdXG4iICAgICAgICAgICAgICAgICAgICAgICAgIC8vIHBoYXNlRnJhY3Rpb24KLSAgICAgICAgIiAgIGxkciByMCwgW3NwLCAjIiBNT19QQVJBTTUgIiArIDhdXG4iICAgICAvLyAmaW5wdXRJbmRleAotICAgICAgICAiICAgc3RyIHI3LCBbcjBdXG4iICAgICAgICAgICAgICAgICAgICAgICAgIC8vIGlucHV0SW5kZXgKLSAgICAgICAgIiAgIGxkciByMCwgW3NwLCAjIiBNT19QQVJBTTUgIiArIDRdXG4iICAgICAvLyBvdXQKLSAgICAgICAgIiAgIHN1YiByOCwgcjBcbiIgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBjdXJPdXQgLSBvdXQKLSAgICAgICAgIiAgIGFzciByOCwgIzJcbiIgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBuZXcgb3V0cHV0SW5kZXgKLSAgICAgICAgIiAgIGxkciByMCwgW3NwLCAjIiBNT19QQVJBTTUgIiArIDBdXG4iICAgICAvLyAmb3V0cHV0SW5kZXgKLSAgICAgICAgIiAgIHN0ciByOCwgW3IwXVxuIiAgICAgICAgICAgICAgICAgICAgICAgICAvLyBzYXZlIG91dHB1dEluZGV4Ci0KLSAgICAgICAgIiAgIGxkbWZkICAgc3AhLCB7cjQsIHI1LCByNiwgcjcsIHI4LCByOSwgcjEwLCByMTEsIHBjfVxuIgotICAgICk7Ci19Ci0KLS8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCi0qCi0qICAgQXNtU3RlcmVvMTZMb29wCi0qICAgYXNtIG9wdGltaXplZCBzdGVyZW8gbG9vcCB2ZXJzaW9uOyBvbmUgbG9vcCBpcyAyIGZyYW1lcwotKiAgIElucHV0OgotKiAgICAgICBpbiA6IHBvaW50ZXIgb24gaW5wdXQgc2FtcGxlcwotKiAgICAgICBtYXhPdXRQdCA6IHBvaW50ZXIgb24gZmlyc3Qgbm90IGZpbGxlZAotKiAgICAgICBtYXhJbklkeCA6IGluZGV4IG9uIGZpcnN0IG5vdCB1c2VkCi0qICAgICAgIG91dHB1dEluZGV4IDogcG9pbnRlciBvbiBjdXJyZW50IG91dHB1dCBpbmRleAotKiAgICAgICBvdXQgOiBwb2ludGVyIG9uIG91dHB1dCBidWZmZXIKLSogICAgICAgaW5wdXRJbmRleCA6IHBvaW50ZXIgb24gY3VycmVudCBpbnB1dCBpbmRleAotKiAgICAgICB2bCwgdnIgOiBsZWZ0IGFuZCByaWdodCBnYWluCi0qICAgICAgIHBoYXNlRnJhY3Rpb24gOiBwb2ludGVyIG9uIGN1cnJlbnQgcGhhc2UgZnJhY3Rpb24KLSogICAgICAgcGhhc2VJbmNyZW1lbnQKLSogICBPdXB1dDoKLSogICAgICAgb3V0cHV0SW5kZXggOgotKiAgICAgICBvdXQgOiB1cGRhdGVkIGJ1ZmZlcgotKiAgICAgICBpbnB1dEluZGV4IDogaW5kZXggb2YgbmV4dCB0byB1c2UKLSogICAgICAgcGhhc2VGcmFjdGlvbiA6IHBoYXNlIGZyYWN0aW9uIGZvciBuZXh0IGludGVycG9sYXRpb24KLSoKLSoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi12b2lkIEF1ZGlvUmVzYW1wbGVyT3JkZXIxOjpBc21TdGVyZW8xNkxvb3AoaW50MTZfdCAqaW4sIGludDMyX3QqIG1heE91dFB0LCBpbnQzMl90IG1heEluSWR4LAotICAgICAgICAgICAgc2l6ZV90ICZvdXRwdXRJbmRleCwgaW50MzJfdCogb3V0LCBzaXplX3QgJmlucHV0SW5kZXgsIGludDMyX3QgdmwsIGludDMyX3QgdnIsCi0gICAgICAgICAgICB1aW50MzJfdCAmcGhhc2VGcmFjdGlvbiwgdWludDMyX3QgcGhhc2VJbmNyZW1lbnQpCi17Ci0jZGVmaW5lIFNUX1BBUkFNNSAgICAiNDAiICAgICAvLyBvZmZzZXQgb2YgcGFyYW1ldGVyIDUgKG91dHB1dEluZGV4KQotICAgIGFzbSgKLSAgICAgICAgInN0bWZkICBzcCEsIHtyNCwgcjUsIHI2LCByNywgcjgsIHI5LCByMTAsIHIxMSwgcjEyLCBscn1cbiIKLSAgICAgICAgLy8gZ2V0IHBhcmFtZXRlcnMKLSAgICAgICAgIiAgIGxkciByNiwgW3NwLCAjIiBTVF9QQVJBTTUgIiArIDIwXVxuIiAgICAvLyAmcGhhc2VGcmFjdGlvbgotICAgICAgICAiICAgbGRyIHI2LCBbcjZdXG4iICAgICAgICAgICAgICAgICAgICAgICAgIC8vIHBoYXNlRnJhY3Rpb24KLSAgICAgICAgIiAgIGxkciByNywgW3NwLCAjIiBTVF9QQVJBTTUgIiArIDhdXG4iICAgICAvLyAmaW5wdXRJbmRleAotICAgICAgICAiICAgbGRyIHI3LCBbcjddXG4iICAgICAgICAgICAgICAgICAgICAgICAgIC8vIGlucHV0SW5kZXgKLSAgICAgICAgIiAgIGxkciByOCwgW3NwLCAjIiBTVF9QQVJBTTUgIiArIDRdXG4iICAgICAvLyBvdXQKLSAgICAgICAgIiAgIGxkciByMCwgW3NwLCAjIiBTVF9QQVJBTTUgIiArIDBdXG4iICAgICAvLyAmb3V0cHV0SW5kZXgKLSAgICAgICAgIiAgIGxkciByMCwgW3IwXVxuIiAgICAgICAgICAgICAgICAgICAgICAgICAvLyBvdXRwdXRJbmRleAotICAgICAgICAiICAgYWRkIHI4LCByMCwgYXNsICMyXG4iICAgICAgICAgICAgICAgICAgIC8vIGN1ck91dAotICAgICAgICAiICAgbGRyIHI5LCBbc3AsICMiIFNUX1BBUkFNNSAiICsgMjRdXG4iICAgIC8vIHBoYXNlSW5jcmVtZW50Ci0gICAgICAgICIgICBsZHIgcjEwLCBbc3AsICMiIFNUX1BBUkFNNSAiICsgMTJdXG4iICAgLy8gdmwKLSAgICAgICAgIiAgIGxkciByMTEsIFtzcCwgIyIgU1RfUEFSQU01ICIgKyAxNl1cbiIgICAvLyB2cgotCi0gICAgICAgIC8vIHIwIHBpbiwgeDAsIFNhbXAKLQotICAgICAgICAvLyByMSBpbgotICAgICAgICAvLyByMiBtYXhPdXRQdAotICAgICAgICAvLyByMyBtYXhJbklkeAotCi0gICAgICAgIC8vIHI0IHgxLCBpMSwgaTMsIG91dDEKLSAgICAgICAgLy8gcjUgb3V0MAotCi0gICAgICAgIC8vIHI2IGZyYWMKLSAgICAgICAgLy8gcjcgaW5wdXRJbmRleAotICAgICAgICAvLyByOCBjdXJPdXQKLQotICAgICAgICAvLyByOSBpbmMKLSAgICAgICAgLy8gcjEwIHZsCi0gICAgICAgIC8vIHIxMSB2cgotCi0gICAgICAgIC8vIHIxMiB0ZW1wb3JhcnkKLSAgICAgICAgLy8gcjEzIHNwCi0gICAgICAgIC8vIHIxNAotCi0gICAgICAgICIuWTVMMDE6XG4iCi0gICAgICAgICIgICBjbXAgcjgsIHIyXG4iICAgICAgICAgICAgICAgICAgIC8vIGN1ck91dCAtIG1heEN1ck91dAotICAgICAgICAiICAgYmNzIC5ZNUwwMlxuIgotCi0jZGVmaW5lIFNUX09ORV9GUkFNRSBcCi0gICAgIiAgIGJpYyByNiwgcjYsICMweEMwMDAwMDAwXG4iICAgICAgLyogcGhhc2VGcmFjdGlvbiAmIC4uLiAqL1wKLVwKLSAgICAiICAgYWRkIHIwLCByMSwgcjcsIGFzbCAjMlxuIiAgICAgICAvKiBpbiArIDIqaW5wdXRJbmRleCAqL1wKLVwKLSAgICAiICAgbGRyc2ggcjQsIFtyMF1cbiIgICAgICAgICAgICAgICAvKiBpblsyKmlucHV0SW5kZXhdICovXAotICAgICIgICBsZHIgcjUsIFtyOF1cbiIgICAgICAgICAgICAgICAgIC8qIG91dFtvdXRwdXRJbmRleF0gKi9cCi0gICAgIiAgIGxkcnNoIHIxMiwgW3IwLCAjLTRdXG4iICAgICAgICAgLyogaW5bMippbnB1dEluZGV4LTJdICovXAotICAgICIgICBzdWIgcjQsIHI0LCByMTJcbiIgICAgICAgICAgICAgIC8qIGluWzIqSW5wdXRJbmRleF0gLSBpblsyKklucHV0SW5kZXgtMl0gKi9cCi0gICAgIiAgIG1vdiByNCwgcjQsIGxzbCAjMlxuIiAgICAgICAgICAgLyogPDwyICovXAotICAgICIgICBzbXVsd3QgcjQsIHI0LCByNlxuIiAgICAgICAgICAgIC8qICh4MS14MCkqLi4gKi9cCi0gICAgIiAgIGFkZCByMTIsIHIxMiwgcjRcbiIgICAgICAgICAgICAgLyogeDAgLSAoLi4pICovXAotICAgICIgICBtbGEgcjUsIHIxMiwgcjEwLCByNVxuIiAgICAgICAgIC8qIHZsKmludGVycCArIG91dFtdICovXAotICAgICIgICBsZHIgcjQsIFtyOCwgIzRdXG4iICAgICAgICAgICAgIC8qIG91dFtvdXRwdXRJbmRleCsxXSAqL1wKLSAgICAiICAgc3RyIHI1LCBbcjhdLCAjNFxuIiAgICAgICAgICAgICAvKiBvdXRbb3V0cHV0SW5kZXgrK10gPSAuLi4gKi9cCi1cCi0gICAgIiAgIGxkcnNoIHIxMiwgW3IwLCAjKzJdXG4iICAgICAgICAgLyogaW5bMippbnB1dEluZGV4KzFdICovXAotICAgICIgICBsZHJzaCByMCwgW3IwLCAjLTJdXG4iICAgICAgICAgIC8qIGluWzIqaW5wdXRJbmRleC0xXSAqL1wKLSAgICAiICAgc3ViIHIxMiwgcjEyLCByMFxuIiAgICAgICAgICAgICAvKiBpblsyKklucHV0SW5kZXhdIC0gaW5bMipJbnB1dEluZGV4LTJdICovXAotICAgICIgICBtb3YgcjEyLCByMTIsIGxzbCAjMlxuIiAgICAgICAgIC8qIDw8MiAqL1wKLSAgICAiICAgc211bHd0IHIxMiwgcjEyLCByNlxuIiAgICAgICAgICAvKiAoeDEteDApKi4uICovXAotICAgICIgICBhZGQgcjEyLCByMCwgcjEyXG4iICAgICAgICAgICAgIC8qIHgwIC0gKC4uKSAqL1wKLSAgICAiICAgbWxhIHI0LCByMTIsIHIxMSwgcjRcbiIgICAgICAgICAvKiB2cippbnRlcnAgKyBvdXRbXSAqL1wKLSAgICAiICAgc3RyIHI0LCBbcjhdLCAjNFxuIiAgICAgICAgICAgICAvKiBvdXRbb3V0cHV0SW5kZXgrK10gPSAuLi4gKi9cCi1cCi0gICAgIiAgIGFkZCByNiwgcjYsIHI5XG4iICAgICAgICAgICAgICAgLyogcGhhc2VGcmFjdGlvbiArIHBoYXNlSW5jcmVtZW50ICovXAotICAgICIgICBhZGQgcjcsIHI3LCByNiwgbHNyICMzMFxuIiAgICAgIC8qIGlucHV0SW5kZXggKyBwaGFzZUZyYWN0aW9uPj4zMCAqLwotCi0gICAgU1RfT05FX0ZSQU1FICAgIC8vIGZyYW1lIDEKLSAgICBTVF9PTkVfRlJBTUUgICAgLy8gZnJhbWUgMQotCi0gICAgICAgICIgICBjbXAgcjcsIHIzXG4iICAgICAgICAgICAgICAgICAgICAgICAvLyBpbnB1dEluZGV4IC0gbWF4SW5JZHgKLSAgICAgICAgIiAgIGJjYyAuWTVMMDFcbiIKLSAgICAgICAgIi5ZNUwwMjpcbiIKLQotICAgICAgICAiICAgYmljIHI2LCByNiwgIzB4QzAwMDAwMDBcbiIgICAgICAgICAgICAgIC8vIHBoYXNlRnJhY3Rpb24gJiAuLi4KLSAgICAgICAgLy8gc2F2ZSBtb2RpZmllZCB2YWx1ZXMKLSAgICAgICAgIiAgIGxkciByMCwgW3NwLCAjIiBTVF9QQVJBTTUgIiArIDIwXVxuIiAgICAvLyAmcGhhc2VGcmFjdGlvbgotICAgICAgICAiICAgc3RyIHI2LCBbcjBdXG4iICAgICAgICAgICAgICAgICAgICAgICAgIC8vIHBoYXNlRnJhY3Rpb24KLSAgICAgICAgIiAgIGxkciByMCwgW3NwLCAjIiBTVF9QQVJBTTUgIiArIDhdXG4iICAgICAvLyAmaW5wdXRJbmRleAotICAgICAgICAiICAgc3RyIHI3LCBbcjBdXG4iICAgICAgICAgICAgICAgICAgICAgICAgIC8vIGlucHV0SW5kZXgKLSAgICAgICAgIiAgIGxkciByMCwgW3NwLCAjIiBTVF9QQVJBTTUgIiArIDRdXG4iICAgICAvLyBvdXQKLSAgICAgICAgIiAgIHN1YiByOCwgcjBcbiIgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBjdXJPdXQgLSBvdXQKLSAgICAgICAgIiAgIGFzciByOCwgIzJcbiIgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBuZXcgb3V0cHV0SW5kZXgKLSAgICAgICAgIiAgIGxkciByMCwgW3NwLCAjIiBTVF9QQVJBTTUgIiArIDBdXG4iICAgICAvLyAmb3V0cHV0SW5kZXgKLSAgICAgICAgIiAgIHN0ciByOCwgW3IwXVxuIiAgICAgICAgICAgICAgICAgICAgICAgICAvLyBzYXZlIG91dHB1dEluZGV4Ci0KLSAgICAgICAgIiAgIGxkbWZkICAgc3AhLCB7cjQsIHI1LCByNiwgcjcsIHI4LCByOSwgcjEwLCByMTEsIHIxMiwgcGN9XG4iCi0gICAgKTsKLX0KLQotI2VuZGlmICAvLyBBU01fQVJNX1JFU0FNUDEKLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19Ci07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KZGlmZiAtLWdpdCBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvUmVzYW1wbGVyLmggYi9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb1Jlc2FtcGxlci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzOTY1NmMwLi4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvUmVzYW1wbGVyLmgKKysrIC9kZXYvbnVsbApAQCAtMSw5MyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0FVRElPX1JFU0FNUExFUl9ICi0jZGVmaW5lIEFORFJPSURfQVVESU9fUkVTQU1QTEVSX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSAiQXVkaW9CdWZmZXJQcm92aWRlci5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIEF1ZGlvUmVzYW1wbGVyIHsKLXB1YmxpYzoKLSAgICAvLyBEZXRlcm1pbmVzIHF1YWxpdHkgb2YgU1JDLgotICAgIC8vICBMT1dfUVVBTElUWTogbGluZWFyIGludGVycG9sYXRvciAoMXN0IG9yZGVyKQotICAgIC8vICBNRURfUVVBTElUWTogY3ViaWMgaW50ZXJwb2xhdG9yICgzcmQgb3JkZXIpCi0gICAgLy8gIEhJR0hfUVVBTElUWTogZml4ZWQgbXVsdGktdGFwIEZJUiAoZS5nLiA0OEtIei0+NDQuMUtIeikKLSAgICAvLyBOT1RFOiBoaWdoIHF1YWxpdHkgU1JDIHdpbGwgb25seSBiZSBzdXBwb3J0ZWQgZm9yCi0gICAgLy8gY2VydGFpbiBmaXhlZCByYXRlIGNvbnZlcnNpb25zLiBTYW1wbGUgcmF0ZSBjYW5ub3QgYmUKLSAgICAvLyBjaGFuZ2VkIGR5bmFtaWNhbGx5LiAKLSAgICBlbnVtIHNyY19xdWFsaXR5IHsKLSAgICAgICAgREVGQVVMVD0wLAotICAgICAgICBMT1dfUVVBTElUWT0xLAotICAgICAgICBNRURfUVVBTElUWT0yLAotICAgICAgICBISUdIX1FVQUxJVFk9MwotICAgIH07Ci0KLSAgICBzdGF0aWMgQXVkaW9SZXNhbXBsZXIqIGNyZWF0ZShpbnQgYml0RGVwdGgsIGludCBpbkNoYW5uZWxDb3VudCwKLSAgICAgICAgICAgIGludDMyX3Qgc2FtcGxlUmF0ZSwgaW50IHF1YWxpdHk9REVGQVVMVCk7Ci0KLSAgICB2aXJ0dWFsIH5BdWRpb1Jlc2FtcGxlcigpOwotCi0gICAgdmlydHVhbCB2b2lkIGluaXQoKSA9IDA7Ci0gICAgdmlydHVhbCB2b2lkIHNldFNhbXBsZVJhdGUoaW50MzJfdCBpblNhbXBsZVJhdGUpOwotICAgIHZpcnR1YWwgdm9pZCBzZXRWb2x1bWUoaW50MTZfdCBsZWZ0LCBpbnQxNl90IHJpZ2h0KTsKLQotICAgIHZpcnR1YWwgdm9pZCByZXNhbXBsZShpbnQzMl90KiBvdXQsIHNpemVfdCBvdXRGcmFtZUNvdW50LAotICAgICAgICAgICAgQXVkaW9CdWZmZXJQcm92aWRlciogcHJvdmlkZXIpID0gMDsKLQotcHJvdGVjdGVkOgotICAgIC8vIG51bWJlciBvZiBiaXRzIGZvciBwaGFzZSBmcmFjdGlvbiAtIDMwIGJpdHMgYWxsb3dzIG5lYXJseSAyeCBkb3duc2FtcGxpbmcKLSAgICBzdGF0aWMgY29uc3QgaW50IGtOdW1QaGFzZUJpdHMgPSAzMDsKLQotICAgIC8vIHBoYXNlIG1hc2sgZm9yIGZyYWN0aW9uCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IGtQaGFzZU1hc2sgPSAoMUxVPDxrTnVtUGhhc2VCaXRzKS0xOwotCi0gICAgLy8gbXVsdGlwbGllciB0byBjYWxjdWxhdGUgZml4ZWQgcG9pbnQgcGhhc2UgaW5jcmVtZW50Ci0gICAgc3RhdGljIGNvbnN0IGRvdWJsZSBrUGhhc2VNdWx0aXBsaWVyID0gMUwgPDwga051bVBoYXNlQml0czsKLQotICAgIGVudW0gZm9ybWF0IHtNT05PXzE2X0JJVCwgU1RFUkVPXzE2X0JJVH07Ci0gICAgQXVkaW9SZXNhbXBsZXIoaW50IGJpdERlcHRoLCBpbnQgaW5DaGFubmVsQ291bnQsIGludDMyX3Qgc2FtcGxlUmF0ZSk7Ci0KLSAgICAvLyBwcmV2ZW50IGNvcHlpbmcKLSAgICBBdWRpb1Jlc2FtcGxlcihjb25zdCBBdWRpb1Jlc2FtcGxlciYpOwotICAgIEF1ZGlvUmVzYW1wbGVyJiBvcGVyYXRvcj0oY29uc3QgQXVkaW9SZXNhbXBsZXImKTsKLQotICAgIGludDMyX3QgbUJpdERlcHRoOwotICAgIGludDMyX3QgbUNoYW5uZWxDb3VudDsKLSAgICBpbnQzMl90IG1TYW1wbGVSYXRlOwotICAgIGludDMyX3QgbUluU2FtcGxlUmF0ZTsKLSAgICBBdWRpb0J1ZmZlclByb3ZpZGVyOjpCdWZmZXIgbUJ1ZmZlcjsKLSAgICB1bmlvbiB7Ci0gICAgCWludDE2X3QgbVZvbHVtZVsyXTsKLSAgICAJdWludDMyX3QgbVZvbHVtZVJMOwotICAgIH07Ci0gICAgaW50MTZfdCBtVGFyZ2V0Vm9sdW1lWzJdOwotICAgIGZvcm1hdCBtRm9ybWF0OwotICAgIHNpemVfdCBtSW5wdXRJbmRleDsKLSAgICBpbnQzMl90IG1QaGFzZUluY3JlbWVudDsKLSAgICB1aW50MzJfdCBtUGhhc2VGcmFjdGlvbjsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX0KLTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfQVVESU9fUkVTQU1QTEVSX0gKZGlmZiAtLWdpdCBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvUmVzYW1wbGVyQ3ViaWMuY3BwIGIvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9SZXNhbXBsZXJDdWJpYy5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDFkMjQ3YmQuLjAwMDAwMDAKLS0tIGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9SZXNhbXBsZXJDdWJpYy5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxODQgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDxjdXRpbHMvbG9nLmg+Ci0KLSNpbmNsdWRlICJBdWRpb1Jlc2FtcGxlci5oIgotI2luY2x1ZGUgIkF1ZGlvUmVzYW1wbGVyQ3ViaWMuaCIKLQotI2RlZmluZSBMT0dfVEFHICJBdWRpb1NSQyIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIEF1ZGlvUmVzYW1wbGVyQ3ViaWM6OmluaXQoKSB7Ci0gICAgbWVtc2V0KCZsZWZ0LCAwLCBzaXplb2Yoc3RhdGUpKTsKLSAgICBtZW1zZXQoJnJpZ2h0LCAwLCBzaXplb2Yoc3RhdGUpKTsKLX0KLQotdm9pZCBBdWRpb1Jlc2FtcGxlckN1YmljOjpyZXNhbXBsZShpbnQzMl90KiBvdXQsIHNpemVfdCBvdXRGcmFtZUNvdW50LAotICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcikgewotCi0gICAgLy8gc2hvdWxkIG5ldmVyIGhhcHBlbiwgYnV0IHdlIG92ZXJmbG93IGlmIGl0IGRvZXMKLSAgICAvLyBMT0dfQVNTRVJUKG91dEZyYW1lQ291bnQgPCAzMjc2Nyk7Ci0KLSAgICAvLyBzZWxlY3QgdGhlIGFwcHJvcHJpYXRlIHJlc2FtcGxlcgotICAgIHN3aXRjaCAobUNoYW5uZWxDb3VudCkgewotICAgIGNhc2UgMToKLSAgICAgICAgcmVzYW1wbGVNb25vMTYob3V0LCBvdXRGcmFtZUNvdW50LCBwcm92aWRlcik7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgMjoKLSAgICAgICAgcmVzYW1wbGVTdGVyZW8xNihvdXQsIG91dEZyYW1lQ291bnQsIHByb3ZpZGVyKTsKLSAgICAgICAgYnJlYWs7Ci0gICAgfQotfQotCi12b2lkIEF1ZGlvUmVzYW1wbGVyQ3ViaWM6OnJlc2FtcGxlU3RlcmVvMTYoaW50MzJfdCogb3V0LCBzaXplX3Qgb3V0RnJhbWVDb3VudCwKLSAgICAgICAgQXVkaW9CdWZmZXJQcm92aWRlciogcHJvdmlkZXIpIHsKLQotICAgIGludDMyX3QgdmwgPSBtVm9sdW1lWzBdOwotICAgIGludDMyX3QgdnIgPSBtVm9sdW1lWzFdOwotCi0gICAgc2l6ZV90IGlucHV0SW5kZXggPSBtSW5wdXRJbmRleDsKLSAgICB1aW50MzJfdCBwaGFzZUZyYWN0aW9uID0gbVBoYXNlRnJhY3Rpb247Ci0gICAgdWludDMyX3QgcGhhc2VJbmNyZW1lbnQgPSBtUGhhc2VJbmNyZW1lbnQ7Ci0gICAgc2l6ZV90IG91dHB1dEluZGV4ID0gMDsKLSAgICBzaXplX3Qgb3V0cHV0U2FtcGxlQ291bnQgPSBvdXRGcmFtZUNvdW50ICogMjsKLSAgICBzaXplX3QgaW5GcmFtZUNvdW50ID0gKG91dEZyYW1lQ291bnQqbUluU2FtcGxlUmF0ZSkvbVNhbXBsZVJhdGU7Ci0KLSAgICAvLyBmZXRjaCBmaXJzdCBidWZmZXIKLSAgICBpZiAobUJ1ZmZlci5mcmFtZUNvdW50ID09IDApIHsKLSAgICAgICAgbUJ1ZmZlci5mcmFtZUNvdW50ID0gaW5GcmFtZUNvdW50OwotICAgICAgICBwcm92aWRlci0+Z2V0TmV4dEJ1ZmZlcigmbUJ1ZmZlcik7Ci0gICAgICAgIGlmIChtQnVmZmVyLnJhdyA9PSBOVUxMKQotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAvLyBMT0dXKCJOZXcgYnVmZmVyOiBvZmZzZXQ9JXAsIGZyYW1lcz0lZG4iLCBtQnVmZmVyLnJhdywgbUJ1ZmZlci5mcmFtZUNvdW50KTsKLSAgICB9Ci0gICAgaW50MTZfdCAqaW4gPSBtQnVmZmVyLmkxNjsKLQotICAgIHdoaWxlIChvdXRwdXRJbmRleCA8IG91dHB1dFNhbXBsZUNvdW50KSB7Ci0gICAgICAgIGludDMyX3Qgc2FtcGxlOwotICAgICAgICBpbnQzMl90IHg7Ci0KLSAgICAgICAgLy8gY2FsY3VsYXRlIG91dHB1dCBzYW1wbGUKLSAgICAgICAgeCA9IHBoYXNlRnJhY3Rpb24gPj4ga1ByZUludGVycFNoaWZ0OwotICAgICAgICBvdXRbb3V0cHV0SW5kZXgrK10gKz0gdmwgKiBpbnRlcnAoJmxlZnQsIHgpOwotICAgICAgICBvdXRbb3V0cHV0SW5kZXgrK10gKz0gdnIgKiBpbnRlcnAoJnJpZ2h0LCB4KTsKLSAgICAgICAgLy8gb3V0W291dHB1dEluZGV4KytdICs9IHZyICogaW5baW5wdXRJbmRleCoyXTsKLQotICAgICAgICAvLyBpbmNyZW1lbnQgcGhhc2UKLSAgICAgICAgcGhhc2VGcmFjdGlvbiArPSBwaGFzZUluY3JlbWVudDsKLSAgICAgICAgdWludDMyX3QgaW5kZXhJbmNyZW1lbnQgPSAocGhhc2VGcmFjdGlvbiA+PiBrTnVtUGhhc2VCaXRzKTsKLSAgICAgICAgcGhhc2VGcmFjdGlvbiAmPSBrUGhhc2VNYXNrOwotCi0gICAgICAgIC8vIHRpbWUgdG8gZmV0Y2ggYW5vdGhlciBzYW1wbGUKLSAgICAgICAgd2hpbGUgKGluZGV4SW5jcmVtZW50LS0pIHsKLQotICAgICAgICAgICAgaW5wdXRJbmRleCsrOwotICAgICAgICAgICAgaWYgKGlucHV0SW5kZXggPT0gbUJ1ZmZlci5mcmFtZUNvdW50KSB7Ci0gICAgICAgICAgICAgICAgaW5wdXRJbmRleCA9IDA7Ci0gICAgICAgICAgICAgICAgcHJvdmlkZXItPnJlbGVhc2VCdWZmZXIoJm1CdWZmZXIpOwotICAgICAgICAgICAgICAgIG1CdWZmZXIuZnJhbWVDb3VudCA9IGluRnJhbWVDb3VudDsKLSAgICAgICAgICAgICAgICBwcm92aWRlci0+Z2V0TmV4dEJ1ZmZlcigmbUJ1ZmZlcik7Ci0gICAgICAgICAgICAgICAgaWYgKG1CdWZmZXIucmF3ID09IE5VTEwpCi0gICAgICAgICAgICAgICAgICAgIGdvdG8gc2F2ZV9zdGF0ZTsgIC8vIHVnbHksIGJ1dCBlZmZpY2llbnQKLSAgICAgICAgICAgICAgICBpbiA9IG1CdWZmZXIuaTE2OwotICAgICAgICAgICAgICAgIC8vIExPR1coIk5ldyBidWZmZXI6IG9mZnNldD0lcCwgZnJhbWVzPSVkXG4iLCBtQnVmZmVyLnJhdywgbUJ1ZmZlci5mcmFtZUNvdW50KTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gYWR2YW5jZSBzYW1wbGUgc3RhdGUKLSAgICAgICAgICAgIGFkdmFuY2UoJmxlZnQsIGluW2lucHV0SW5kZXgqMl0pOwotICAgICAgICAgICAgYWR2YW5jZSgmcmlnaHQsIGluW2lucHV0SW5kZXgqMisxXSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLXNhdmVfc3RhdGU6Ci0gICAgLy8gTE9HVygiRG9uZTogaW5kZXg9JWQsIGZyYWN0aW9uPSV1IiwgaW5wdXRJbmRleCwgcGhhc2VGcmFjdGlvbik7Ci0gICAgbUlucHV0SW5kZXggPSBpbnB1dEluZGV4OwotICAgIG1QaGFzZUZyYWN0aW9uID0gcGhhc2VGcmFjdGlvbjsKLX0KLQotdm9pZCBBdWRpb1Jlc2FtcGxlckN1YmljOjpyZXNhbXBsZU1vbm8xNihpbnQzMl90KiBvdXQsIHNpemVfdCBvdXRGcmFtZUNvdW50LAotICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcikgewotCi0gICAgaW50MzJfdCB2bCA9IG1Wb2x1bWVbMF07Ci0gICAgaW50MzJfdCB2ciA9IG1Wb2x1bWVbMV07Ci0KLSAgICBzaXplX3QgaW5wdXRJbmRleCA9IG1JbnB1dEluZGV4OwotICAgIHVpbnQzMl90IHBoYXNlRnJhY3Rpb24gPSBtUGhhc2VGcmFjdGlvbjsKLSAgICB1aW50MzJfdCBwaGFzZUluY3JlbWVudCA9IG1QaGFzZUluY3JlbWVudDsKLSAgICBzaXplX3Qgb3V0cHV0SW5kZXggPSAwOwotICAgIHNpemVfdCBvdXRwdXRTYW1wbGVDb3VudCA9IG91dEZyYW1lQ291bnQgKiAyOwotICAgIHNpemVfdCBpbkZyYW1lQ291bnQgPSAob3V0RnJhbWVDb3VudCptSW5TYW1wbGVSYXRlKS9tU2FtcGxlUmF0ZTsKLQotICAgIC8vIGZldGNoIGZpcnN0IGJ1ZmZlcgotICAgIGlmIChtQnVmZmVyLmZyYW1lQ291bnQgPT0gMCkgewotICAgICAgICBtQnVmZmVyLmZyYW1lQ291bnQgPSBpbkZyYW1lQ291bnQ7Ci0gICAgICAgIHByb3ZpZGVyLT5nZXROZXh0QnVmZmVyKCZtQnVmZmVyKTsKLSAgICAgICAgaWYgKG1CdWZmZXIucmF3ID09IE5VTEwpCi0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIC8vIExPR1coIk5ldyBidWZmZXI6IG9mZnNldD0lcCwgZnJhbWVzPSVkXG4iLCBtQnVmZmVyLnJhdywgbUJ1ZmZlci5mcmFtZUNvdW50KTsKLSAgICB9Ci0gICAgaW50MTZfdCAqaW4gPSBtQnVmZmVyLmkxNjsKLQotICAgIHdoaWxlIChvdXRwdXRJbmRleCA8IG91dHB1dFNhbXBsZUNvdW50KSB7Ci0gICAgICAgIGludDMyX3Qgc2FtcGxlOwotICAgICAgICBpbnQzMl90IHg7Ci0KLSAgICAgICAgLy8gY2FsY3VsYXRlIG91dHB1dCBzYW1wbGUKLSAgICAgICAgeCA9IHBoYXNlRnJhY3Rpb24gPj4ga1ByZUludGVycFNoaWZ0OwotICAgICAgICBzYW1wbGUgPSBpbnRlcnAoJmxlZnQsIHgpOwotICAgICAgICBvdXRbb3V0cHV0SW5kZXgrK10gKz0gdmwgKiBzYW1wbGU7Ci0gICAgICAgIG91dFtvdXRwdXRJbmRleCsrXSArPSB2ciAqIHNhbXBsZTsKLQotICAgICAgICAvLyBpbmNyZW1lbnQgcGhhc2UKLSAgICAgICAgcGhhc2VGcmFjdGlvbiArPSBwaGFzZUluY3JlbWVudDsKLSAgICAgICAgdWludDMyX3QgaW5kZXhJbmNyZW1lbnQgPSAocGhhc2VGcmFjdGlvbiA+PiBrTnVtUGhhc2VCaXRzKTsKLSAgICAgICAgcGhhc2VGcmFjdGlvbiAmPSBrUGhhc2VNYXNrOwotCi0gICAgICAgIC8vIHRpbWUgdG8gZmV0Y2ggYW5vdGhlciBzYW1wbGUKLSAgICAgICAgd2hpbGUgKGluZGV4SW5jcmVtZW50LS0pIHsKLQotICAgICAgICAgICAgaW5wdXRJbmRleCsrOwotICAgICAgICAgICAgaWYgKGlucHV0SW5kZXggPT0gbUJ1ZmZlci5mcmFtZUNvdW50KSB7Ci0gICAgICAgICAgICAgICAgaW5wdXRJbmRleCA9IDA7Ci0gICAgICAgICAgICAgICAgcHJvdmlkZXItPnJlbGVhc2VCdWZmZXIoJm1CdWZmZXIpOwotICAgICAgICAgICAgICAgIG1CdWZmZXIuZnJhbWVDb3VudCA9IGluRnJhbWVDb3VudDsKLSAgICAgICAgICAgICAgICBwcm92aWRlci0+Z2V0TmV4dEJ1ZmZlcigmbUJ1ZmZlcik7Ci0gICAgICAgICAgICAgICAgaWYgKG1CdWZmZXIucmF3ID09IE5VTEwpCi0gICAgICAgICAgICAgICAgICAgIGdvdG8gc2F2ZV9zdGF0ZTsgIC8vIHVnbHksIGJ1dCBlZmZpY2llbnQKLSAgICAgICAgICAgICAgICAvLyBMT0dXKCJOZXcgYnVmZmVyOiBvZmZzZXQ9JXAsIGZyYW1lcz0lZG4iLCBtQnVmZmVyLnJhdywgbUJ1ZmZlci5mcmFtZUNvdW50KTsKLSAgICAgICAgICAgICAgICBpbiA9IG1CdWZmZXIuaTE2OwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBhZHZhbmNlIHNhbXBsZSBzdGF0ZQotICAgICAgICAgICAgYWR2YW5jZSgmbGVmdCwgaW5baW5wdXRJbmRleF0pOwotICAgICAgICB9Ci0gICAgfQotCi1zYXZlX3N0YXRlOgotICAgIC8vIExPR1coIkRvbmU6IGluZGV4PSVkLCBmcmFjdGlvbj0ldSIsIGlucHV0SW5kZXgsIHBoYXNlRnJhY3Rpb24pOwotICAgIG1JbnB1dEluZGV4ID0gaW5wdXRJbmRleDsKLSAgICBtUGhhc2VGcmFjdGlvbiA9IHBoYXNlRnJhY3Rpb247Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX0KLTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9SZXNhbXBsZXJDdWJpYy5oIGIvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9SZXNhbXBsZXJDdWJpYy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiNzJiNjJhLi4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvUmVzYW1wbGVyQ3ViaWMuaAorKysgL2Rldi9udWxsCkBAIC0xLDY4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfQVVESU9fUkVTQU1QTEVSX0NVQklDX0gKLSNkZWZpbmUgQU5EUk9JRF9BVURJT19SRVNBTVBMRVJfQ1VCSUNfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8Y3V0aWxzL2xvZy5oPgotCi0jaW5jbHVkZSAiQXVkaW9SZXNhbXBsZXIuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBBdWRpb1Jlc2FtcGxlckN1YmljIDogcHVibGljIEF1ZGlvUmVzYW1wbGVyIHsKLXB1YmxpYzoKLSAgICBBdWRpb1Jlc2FtcGxlckN1YmljKGludCBiaXREZXB0aCwgaW50IGluQ2hhbm5lbENvdW50LCBpbnQzMl90IHNhbXBsZVJhdGUpIDoKLSAgICAgICAgQXVkaW9SZXNhbXBsZXIoYml0RGVwdGgsIGluQ2hhbm5lbENvdW50LCBzYW1wbGVSYXRlKSB7Ci0gICAgfQotICAgIHZpcnR1YWwgdm9pZCByZXNhbXBsZShpbnQzMl90KiBvdXQsIHNpemVfdCBvdXRGcmFtZUNvdW50LAotICAgICAgICAgICAgQXVkaW9CdWZmZXJQcm92aWRlciogcHJvdmlkZXIpOwotcHJpdmF0ZToKLSAgICAvLyBudW1iZXIgb2YgYml0cyB1c2VkIGluIGludGVycG9sYXRpb24gbXVsdGlwbHkgLSAxNCBiaXRzIGF2b2lkcyBvdmVyZmxvdwotICAgIHN0YXRpYyBjb25zdCBpbnQga051bUludGVycEJpdHMgPSAxNDsKLQotICAgIC8vIGJpdHMgdG8gc2hpZnQgdGhlIHBoYXNlIGZyYWN0aW9uIGRvd24gdG8gYXZvaWQgb3ZlcmZsb3cKLSAgICBzdGF0aWMgY29uc3QgaW50IGtQcmVJbnRlcnBTaGlmdCA9IGtOdW1QaGFzZUJpdHMgLSBrTnVtSW50ZXJwQml0czsKLSAgICB0eXBlZGVmIHN0cnVjdCB7Ci0gICAgICAgIGludDMyX3QgYSwgYiwgYywgeTAsIHkxLCB5MiwgeTM7Ci0gICAgfSBzdGF0ZTsKLSAgICB2b2lkIGluaXQoKTsKLSAgICB2b2lkIHJlc2FtcGxlTW9ubzE2KGludDMyX3QqIG91dCwgc2l6ZV90IG91dEZyYW1lQ291bnQsCi0gICAgICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcik7Ci0gICAgdm9pZCByZXNhbXBsZVN0ZXJlbzE2KGludDMyX3QqIG91dCwgc2l6ZV90IG91dEZyYW1lQ291bnQsCi0gICAgICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcik7Ci0gICAgc3RhdGljIGlubGluZSBpbnQzMl90IGludGVycChzdGF0ZSogcCwgaW50MzJfdCB4KSB7Ci0gICAgICAgIHJldHVybiAoKCgoKHAtPmEgKiB4ID4+IDE0KSArIHAtPmIpICogeCA+PiAxNCkgKyBwLT5jKSAqIHggPj4gMTQpICsgcC0+eTE7Ci0gICAgfQotICAgIHN0YXRpYyBpbmxpbmUgdm9pZCBhZHZhbmNlKHN0YXRlKiBwLCBpbnQxNl90IGluKSB7Ci0gICAgICAgIHAtPnkwID0gcC0+eTE7Ci0gICAgICAgIHAtPnkxID0gcC0+eTI7Ci0gICAgICAgIHAtPnkyID0gcC0+eTM7Ci0gICAgICAgIHAtPnkzID0gaW47Ci0gICAgICAgIHAtPmEgPSAoMyAqIChwLT55MSAtIHAtPnkyKSAtIHAtPnkwICsgcC0+eTMpID4+IDE7ICAgICAgICAgICAgCi0gICAgICAgIHAtPmIgPSAocC0+eTIgPDwgMSkgKyBwLT55MCAtICgoKDUgKiBwLT55MSArIHAtPnkzKSkgPj4gMSk7Ci0gICAgICAgIHAtPmMgPSAocC0+eTIgLSBwLT55MCkgPj4gMTsKLSAgICB9Ci0gICAgc3RhdGUgbGVmdCwgcmlnaHQ7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLypBTkRST0lEX0FVRElPX1JFU0FNUExFUl9DVUJJQ19IKi8KZGlmZiAtLWdpdCBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvUmVzYW1wbGVyU2luYy5jcHAgYi9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb1Jlc2FtcGxlclNpbmMuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5ZTVlMjU0Li4wMDAwMDAwCi0tLSBhL2xpYnMvYXVkaW9mbGluZ2VyL0F1ZGlvUmVzYW1wbGVyU2luYy5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwzNTggKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlICJBdWRpb1Jlc2FtcGxlclNpbmMuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0KLS8qCi0gKiBUaGVzZSBjb2VmaWNpZW50cyBhcmUgY29tcHV0ZWQgd2l0aCB0aGUgImZpciIgdXRpbGl0eSBmb3VuZCBpbgotICogdG9vbHMvcmVzYW1wbGVyX3Rvb2xzCi0gKiBUT0RPOiBBIGdvb2Qgb3B0aW1pemF0aW9uIHdvdWxkIGJlIHRvIHRyYW5zcG9zZSB0aGlzIG1hdHJpeCwgdG8gdGFrZQotICogYmV0dGVyIGFkdmFudGFnZSBvZiB0aGUgZGF0YS1jYWNoZS4KLSAqLwotY29uc3QgaW50MzJfdCBBdWRpb1Jlc2FtcGxlclNpbmM6Om1GaXJDb2Vmc1VwW10gPSB7Ci0gICAgICAgIDB4N2ZmZmZmZmYsIDB4N2YxNWQwNzgsIDB4N2M1ZTBkYTYsIDB4NzdlY2Q4NjcsIDB4NzFlMmUyNTEsIDB4NmE2YzMwNGEsIDB4NjFiZTcyNjksIDB4NTgxNzA0MTIsIDB4NGRiOGFiMDUsIDB4NDJlOTJlYTYsIDB4MzdlZWUyMTQsIDB4MmQwZTNiYjEsIDB4MjI4NzkzNjYsIDB4MTg5NTFlOTUsIDB4MGY2OTNkMGQsIDB4MDcyZDI2MjEsCi0gICAgICAgIDB4MDAwMDAwMDAsIDB4ZjlmNjY2NTUsIDB4ZjUxYTVmZDcsIDB4ZjE2YmJkODQsIDB4ZWVlMGQ5YWMsIDB4ZWQ2N2E5MjIsIDB4ZWNlNzBkZTYsIDB4ZWQ0MDU4OTcsIDB4ZWU1MGU1MDUsIDB4ZWZmM2JlMzAsIDB4ZjIwMzM3MGYsIDB4ZjQ1YTY3NDEsIDB4ZjZkNjdkNTMsIDB4Zjk1N2RiNjYsIDB4ZmJjMmY2NDcsIDB4ZmUwMGYyYjksCi0gICAgICAgIDB4MDAwMDAwMDAsIDB4MDFiMzcyMTgsIDB4MDMxM2EwYzYsIDB4MDQxZDkzMGQsIDB4MDRkMjgwNTcsIDB4MDUzNzMxYjAsIDB4MDU1MzRkZmYsIDB4MDUzMDliZmQsIDB4MDRkYTQ0MGQsIDB4MDQ1YzFhZWUsIDB4MDNjMWZjZGQsIDB4MDMxNzNlZjUsIDB4MDI2NjNhZTgsIDB4MDFiN2Y3MzYsIDB4MDExM2VjNzksIDB4MDA3ZmU2YTksCi0gICAgICAgIDB4MDAwMDAwMDAsIDB4ZmY5NmIyMjksIDB4ZmY0NGY5OWYsIDB4ZmYwYTg2YmUsIDB4ZmVlNWY4MDMsIDB4ZmVkNTE4ZmQsIDB4ZmVkNTIxZmQsIDB4ZmVlMmY0ZmQsIDB4ZmVmYjU0ZjgsIDB4ZmYxYjE1OWIsIDB4ZmYzZjQyMDMsIDB4ZmY2NTM5ZTAsIDB4ZmY4YWM1MDIsIDB4ZmZhZTFkZGQsIDB4ZmZjZGYzZjksIDB4ZmZlOTY3OTgsCi0gICAgICAgIDB4MDAwMDAwMDAsIDB4MDAxMTlkZTYsIDB4MDAxZTZiN2UsIDB4MDAyNmNiN2EsIDB4MDAyYjQ4MzAsIDB4MDAyYzgzZDYsIDB4MDAyYjJhODIsIDB4MDAyN2U2N2EsIDB4MDAyMzU2ZjksIDB4MDAxZTA5OGUsIDB4MDAxODc1ZTQsIDB4MDAxMmZiYmUsIDB4MDAwZGUyZDEsIDB4MDAwOTVjMTAsIDB4MDAwNTg0MTQsIDB4MDAwMjY2MzYsCi0gICAgICAgIDB4MDAwMDAwMDAsIDB4ZmZmZTQ0YTksIDB4ZmZmZDIwNmQsIDB4ZmZmYzdiN2YsIDB4ZmZmYzNjOGYsIDB4ZmZmYzRhYzIsIDB4ZmZmYzhmMmIsIDB4ZmZmY2Y1YzQsIDB4ZmZmZDZkZjMsIDB4ZmZmZGVhYjIsIDB4ZmZmZTYyNzUsIDB4ZmZmZWNlY2YsIDB4ZmZmZjJjMDcsIDB4ZmZmZjc4OGMsIDB4ZmZmZmI0NzEsIDB4ZmZmZmUwZjIsCi0gICAgICAgIDB4MDAwMDAwMDAsIDB4MDAwMDEzZTYsIDB4MDAwMDFmMDMsIDB4MDAwMDIzOTYsIDB4MDAwMDIzOTksIDB4MDAwMDIwYjYsIDB4MDAwMDFjM2MsIDB4MDAwMDE3MjIsIDB4MDAwMDEyMTYsIDB4MDAwMDBkODEsIDB4MDAwMDA5OWMsIDB4MDAwMDA2N2MsIDB4MDAwMDA0MTksIDB4MDAwMDAyNWYsIDB4MDAwMDAxMzEsIDB4MDAwMDAwNzAsCi0gICAgICAgIDB4MDAwMDAwMDAsIDB4ZmZmZmZmYzcsIDB4ZmZmZmZmYjMsIDB4ZmZmZmZmYjMsIDB4ZmZmZmZmYmUsIDB4ZmZmZmZmY2QsIDB4ZmZmZmZmZGIsIDB4ZmZmZmZmZTcsIDB4ZmZmZmZmZjAsIDB4ZmZmZmZmZjcsIDB4ZmZmZmZmZmIsIDB4ZmZmZmZmZmUsIDB4ZmZmZmZmZmYsIDB4MDAwMDAwMDAsIDB4MDAwMDAwMDAsIDB4MDAwMDAwMDAsCi0gICAgICAgIDB4MDAwMDAwMDAgLy8gdGhpcyBvbmUgaXMgbmVlZGVkIGZvciBsZXJwaW5nIHRoZSBsYXN0IGNvZWZmaWNpZW50Ci19OwotCi0vKgotICogVGhlc2UgY29lZmZpY2llbnRzIGFyZSBvcHRpbWl6ZWQgZm9yIDQ4S0h6IC0+IDQ0LjFLSHogKHN0b3AtYmFuZCBhdCAyMi4wNTBLSHopCi0gKiBJdCdzIHBvc3NpYmxlIHRvIHVzZSB0aGUgYWJvdmUgY29lZmZpY2llbnQgZm9yIGFueSBkb3duLXNhbXBsaW5nCi0gKiBhdCB0aGUgZXhwZW5zZSBvZiBhIHNsb3dlciBwcm9jZXNzaW5nIGxvb3AgKHdlIGNhbiBpbnRlcnBvbGF0ZQotICogdGhlc2UgY29lZmZpY2llbnQgZnJvbSB0aGUgYWJvdmUgYnkgIlN0cmV0Y2hpbmciIHRoZW0gaW4gdGltZSkuCi0gKi8KLWNvbnN0IGludDMyX3QgQXVkaW9SZXNhbXBsZXJTaW5jOjptRmlyQ29lZnNEb3duW10gPSB7Ci0gICAgICAgIDB4N2ZmZmZmZmYsIDB4N2Y1NWU0NmQsIDB4N2Q1YjRjNjAsIDB4N2ExYjRiOTgsIDB4NzVhN2ZiMTQsIDB4NzAxOWYwYmQsIDB4Njk4Zjg3NWEsIDB4NjIyYmZkNTksIDB4NWExNjcyNTYsIDB4NTE3OGNjNTQsIDB4NDg3ZThlNmMsIDB4M2Y1M2FhZTgsIDB4MzYyMzVhZDQsIDB4MmQxNzA0N2IsIDB4MjQ1NTM5YWIsIDB4MWMwMGQ1NDAsCi0gICAgICAgIDB4MTQzODNlNTcsIDB4MGQxNGQ1Y2EsIDB4MDZhYTkxMGIsIDB4MDEwN2MzOGIsIDB4ZmMzNTE2NTQsIDB4ZjgzNWFiYWUsIDB4ZjUwNzZiNDUsIDB4ZjJhMzcyMDIsIDB4ZjBmZTlmYWEsIDB4ZjAwYTNiYmQsIDB4ZWZiNGFhODEsIDB4ZWZlYTJiMDUsIDB4ZjA5NTk3MTYsIDB4ZjFhMTFlODMsIDB4ZjJmNmY3YTAsIDB4ZjQ4MWZmZjQsCi0gICAgICAgIDB4ZjYyZTQ4Y2UsIDB4ZjdlOThjYTUsIDB4ZjlhMzhiNGMsIDB4ZmI0ZTRiZmEsIDB4ZmNkZTQ1NmYsIDB4ZmU0YTZkMzAsIDB4ZmY4YzJmZGYsIDB4MDA5ZjU1NTUsIDB4MDE4MWQzOTMsIDB4MDIzMzk0MGYsIDB4MDJiNjJmMDYsIDB4MDMwY2EwN2QsIDB4MDMzYWZhNjIsIDB4MDM0NjE3MjUsIDB4MDMzMzRmODMsIDB4MDMwODM1ZmEsCi0gICAgICAgIDB4MDJjYTU5Y2MsIDB4MDI3ZjEyZDEsIDB4MDIyYjU3MGQsIDB4MDFkMzlhNDksIDB4MDE3YmI3OGYsIDB4MDEyNmU0MTQsIDB4MDBkN2FhYWYsIDB4MDA4ZmVlYzcsIDB4MDA1MGY1ODQsIDB4MDAxYjczZTMsIDB4ZmZlZmEwNjMsIDB4ZmZjZDQ2ZWQsIDB4ZmZiM2RkY2QsIDB4ZmZhMjlhYWEsIDB4ZmY5ODg2OTEsIDB4ZmY5NDkwNjYsCi0gICAgICAgIDB4ZmY5NTlkMjQsIDB4ZmY5YTk1OWUsIDB4ZmZhMjcxOTUsIDB4ZmZhYzQwMTEsIDB4ZmZiNzJkMmIsIDB4ZmZjMjg1NjksIDB4ZmZjZGI3MDYsIDB4ZmZkODUxNzEsIDB4ZmZlMjAzNjQsIDB4ZmZlYTk3ZTksIDB4ZmZmMWYyYjIsIDB4ZmZmODBjMDYsIDB4ZmZmY2VjOTIsIDB4MDAwMGE5NTUsIDB4MDAwMzVmZDgsIDB4MDAwNTMyY2YsCi0gICAgICAgIDB4MDAwNjQ3MzUsIDB4MDAwNmMxZjksIDB4MDAwNmM2MmQsIDB4MDAwNjczYmEsIDB4MDAwNWU2OGYsIDB4MDAwNTM2MzAsIDB4MDAwNDc1YTMsIDB4MDAwM2IzOTcsIDB4MDAwMmZhYzEsIDB4MDAwMjUyNTcsIDB4MDAwMWJlOWUsIDB4MDAwMTQxN2EsIDB4MDAwMGRhZmQsIDB4MDAwMDg5ZWIsIDB4MDAwMDRjMjgsIDB4MDAwMDFmMWQsCi0gICAgICAgIDB4MDAwMDAwMDAsIDB4ZmZmZmVjMTAsIDB4ZmZmZmUwYmUsIDB4ZmZmZmRiYzUsIDB4ZmZmZmRiMzksIDB4ZmZmZmRkOGIsIDB4ZmZmZmUxODIsIDB4ZmZmZmU2MzgsIDB4ZmZmZmViMGEsIDB4ZmZmZmVmOGYsIDB4ZmZmZmYzOGIsIDB4ZmZmZmY2ZTMsIDB4ZmZmZmY5OTMsIDB4ZmZmZmZiYTYsIDB4ZmZmZmZkMzAsIDB4ZmZmZmZlNGEsCi0gICAgICAgIDB4ZmZmZmZmMDksIDB4ZmZmZmZmODUsIDB4ZmZmZmZmZDEsIDB4ZmZmZmZmZmIsIDB4MDAwMDAwMGYsIDB4MDAwMDAwMTYsIDB4MDAwMDAwMTUsIDB4MDAwMDAwMTIsIDB4MDAwMDAwMGQsIDB4MDAwMDAwMDksIDB4MDAwMDAwMDYsIDB4MDAwMDAwMDMsIDB4MDAwMDAwMDIsIDB4MDAwMDAwMDEsIDB4MDAwMDAwMDAsIDB4MDAwMDAwMDAsCi0gICAgICAgIDB4MDAwMDAwMDAgLy8gdGhpcyBvbmUgaXMgbmVlZGVkIGZvciBsZXJwaW5nIHRoZSBsYXN0IGNvZWZmaWNpZW50Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyBpbmxpbmUKLWludDMyX3QgbXVsUkwoaW50IGxlZnQsIGludDMyX3QgaW4sIHVpbnQzMl90IHZSTCkKLXsKLSNpZiBkZWZpbmVkKF9fYXJtX18pICYmICFkZWZpbmVkKF9fdGh1bWJfXykKLSAgICBpbnQzMl90IG91dDsKLSAgICBpZiAobGVmdCkgewotICAgICAgICBhc20oICJzbXVsdGIgJVtvdXRdLCAlW2luXSwgJVt2UkxdIFxuIgotICAgICAgICAgICAgIDogW291dF0iPXIiKG91dCkKLSAgICAgICAgICAgICA6IFtpbl0iJXIiKGluKSwgW3ZSTF0iciIodlJMKQotICAgICAgICAgICAgIDogKTsKLSAgICB9IGVsc2UgewotICAgICAgICBhc20oICJzbXVsdHQgJVtvdXRdLCAlW2luXSwgJVt2UkxdIFxuIgotICAgICAgICAgICAgIDogW291dF0iPXIiKG91dCkKLSAgICAgICAgICAgICA6IFtpbl0iJXIiKGluKSwgW3ZSTF0iciIodlJMKQotICAgICAgICAgICAgIDogKTsKLSAgICB9Ci0gICAgcmV0dXJuIG91dDsKLSNlbHNlCi0gICAgaWYgKGxlZnQpIHsKLSAgICAgICAgcmV0dXJuIGludDE2X3QoaW4+PjE2KSAqIGludDE2X3QodlJMJjB4RkZGRik7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgcmV0dXJuIGludDE2X3QoaW4+PjE2KSAqIGludDE2X3QodlJMPj4xNik7Ci0gICAgfQotI2VuZGlmCi19Ci0KLXN0YXRpYyBpbmxpbmUKLWludDMyX3QgbXVsQWRkKGludDE2X3QgaW4sIGludDMyX3QgdiwgaW50MzJfdCBhKQotewotI2lmIGRlZmluZWQoX19hcm1fXykgJiYgIWRlZmluZWQoX190aHVtYl9fKQotICAgIGludDMyX3Qgb3V0OwotICAgIGFzbSggInNtbGF3YiAlW291dF0sICVbdl0sICVbaW5dLCAlW2FdIFxuIgotICAgICAgICAgOiBbb3V0XSI9ciIob3V0KQotICAgICAgICAgOiBbaW5dIiVyIihpbiksIFt2XSJyIih2KSwgW2FdInIiKGEpCi0gICAgICAgICA6ICk7Ci0gICAgcmV0dXJuIG91dDsKLSNlbHNlCi0gICAgcmV0dXJuIGEgKyBpbiAqICh2Pj4xNik7Ci0gICAgLy8gaW1wcm92ZWQgcHJlY2lzaW9uCi0gICAgLy8gcmV0dXJuIGEgKyBpbiAqICh2Pj4xNikgKyAoKGluICogKHYgJiAweGZmZmYpKSA+PiAxNik7Ci0jZW5kaWYKLX0KLQotc3RhdGljIGlubGluZQotaW50MzJfdCBtdWxBZGRSTChpbnQgbGVmdCwgdWludDMyX3QgaW5STCwgaW50MzJfdCB2LCBpbnQzMl90IGEpCi17Ci0jaWYgZGVmaW5lZChfX2FybV9fKSAmJiAhZGVmaW5lZChfX3RodW1iX18pCi0gICAgaW50MzJfdCBvdXQ7Ci0gICAgaWYgKGxlZnQpIHsKLSAgICAgICAgYXNtKCAic21sYXdiICVbb3V0XSwgJVt2XSwgJVtpblJMXSwgJVthXSBcbiIKLSAgICAgICAgICAgICA6IFtvdXRdIj1yIihvdXQpCi0gICAgICAgICAgICAgOiBbaW5STF0iJXIiKGluUkwpLCBbdl0iciIodiksIFthXSJyIihhKQotICAgICAgICAgICAgIDogKTsKLSAgICB9IGVsc2UgewotICAgICAgICBhc20oICJzbWxhd3QgJVtvdXRdLCAlW3ZdLCAlW2luUkxdLCAlW2FdIFxuIgotICAgICAgICAgICAgIDogW291dF0iPXIiKG91dCkKLSAgICAgICAgICAgICA6IFtpblJMXSIlciIoaW5STCksIFt2XSJyIih2KSwgW2FdInIiKGEpCi0gICAgICAgICAgICAgOiApOwotICAgIH0KLSAgICByZXR1cm4gb3V0OwotI2Vsc2UKLSAgICBpZiAobGVmdCkgewotICAgICAgICByZXR1cm4gYSArIChpbnQxNl90KGluUkwmMHhGRkZGKSAqICh2Pj4xNikpOwotICAgICAgICAvL2ltcHJvdmVkIHByZWNpc2lvbgotICAgICAgICAvLyByZXR1cm4gYSArIChpbnQxNl90KGluUkwmMHhGRkZGKSAqICh2Pj4xNikpICsgKChpbnQxNl90KGluUkwmMHhGRkZGKSAqICh2ICYgMHhmZmZmKSkgPj4gMTYpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHJldHVybiBhICsgKGludDE2X3QoaW5STD4+MTYpICogKHY+PjE2KSk7Ci0gICAgfQotI2VuZGlmCi19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotQXVkaW9SZXNhbXBsZXJTaW5jOjpBdWRpb1Jlc2FtcGxlclNpbmMoaW50IGJpdERlcHRoLAotICAgICAgICBpbnQgaW5DaGFubmVsQ291bnQsIGludDMyX3Qgc2FtcGxlUmF0ZSkKLSAgICA6IEF1ZGlvUmVzYW1wbGVyKGJpdERlcHRoLCBpbkNoYW5uZWxDb3VudCwgc2FtcGxlUmF0ZSksCi0gICAgbVN0YXRlKDApCi17Ci0gICAgLyoKLSAgICAgKiBMYXlvdXQgb2YgdGhlIHN0YXRlIGJ1ZmZlciBmb3IgMzIgdGFwOgotICAgICAqCi0gICAgICogInByZXNlbnQiIHNhbXBsZSAgICAgICAgICAgIGJlZ2lubmluZyBvZiAybmQgYnVmZmVyCi0gICAgICogICAgICAgICAgICAgICAgIHYgICAgICAgICAgICAgICAgdgotICAgICAqICAwICAgICAgICAgICAgICAwMSAgICAgICAgICAgICAgIDIgICAgICAgICAgICAgIDIzICAgICAgICAgICAgICAzCi0gICAgICogIDAgICAgICAgICAgICAgIEYwICAgICAgICAgICAgICAgMCAgICAgICAgICAgICAgRjAgICAgICAgICAgICAgIEYKLSAgICAgKiBbcHBwcHBwcHBwcHBwcHBwSW5ubm5ubm5ubm5ubm5ubm5wcHBwcHBwcHBwcHBwcHBJbm5ubm5ubm5ubm5ubm5ubl0KLSAgICAgKiAgICAgICAgICAgICAgICAgXiAgICAgICAgICAgICAgIF4gaGVhZAotICAgICAqCi0gICAgICogcCA9IHBhc3Qgc2FtcGxlcywgY29udm9sdXRlZCB3aXRoIHRoZSAocClvc2l0aXZlIHNpZGUgb2Ygc2luYygpCi0gICAgICogbiA9IGZ1dHVyZSBzYW1wbGVzLCBjb252b2x1dGVkIHdpdGggdGhlIChuKWVnYXRpdmUgc2lkZSBvZiBzaW5jKCkKLSAgICAgKiByID0gZXh0cmEgc3BhY2UgZm9yIGltcGxlbWVudGluZyB0aGUgcmluZyBidWZmZXIKLSAgICAgKgotICAgICAqLwotCi0gICAgY29uc3Qgc2l6ZV90IG51bUNvZWZzID0gMipoYWxmTnVtQ29lZnM7Ci0gICAgY29uc3Qgc2l6ZV90IHN0YXRlU2l6ZSA9IG51bUNvZWZzICogaW5DaGFubmVsQ291bnQgKiAyOwotICAgIG1TdGF0ZSA9IG5ldyBpbnQxNl90W3N0YXRlU2l6ZV07Ci0gICAgbWVtc2V0KG1TdGF0ZSwgMCwgc2l6ZW9mKGludDE2X3QpKnN0YXRlU2l6ZSk7Ci0gICAgbUltcHVsc2UgPSBtU3RhdGUgKyAoaGFsZk51bUNvZWZzLTEpKmluQ2hhbm5lbENvdW50OwotICAgIG1SaW5nRnVsbCA9IG1JbXB1bHNlICsgKG51bUNvZWZzKzEpKmluQ2hhbm5lbENvdW50OwotfQotCi1BdWRpb1Jlc2FtcGxlclNpbmM6On5BdWRpb1Jlc2FtcGxlclNpbmMoKQotewotICAgIGRlbGV0ZSBbXSBtU3RhdGU7Ci19Ci0KLXZvaWQgQXVkaW9SZXNhbXBsZXJTaW5jOjppbml0KCkgewotfQotCi12b2lkIEF1ZGlvUmVzYW1wbGVyU2luYzo6cmVzYW1wbGUoaW50MzJfdCogb3V0LCBzaXplX3Qgb3V0RnJhbWVDb3VudCwKLSAgICAgICAgICAgIEF1ZGlvQnVmZmVyUHJvdmlkZXIqIHByb3ZpZGVyKQotewotICAgIG1GaXJDb2VmcyA9IChtSW5TYW1wbGVSYXRlIDw9IG1TYW1wbGVSYXRlKSA/IG1GaXJDb2Vmc1VwIDogbUZpckNvZWZzRG93bjsKLQotICAgIC8vIHNlbGVjdCB0aGUgYXBwcm9wcmlhdGUgcmVzYW1wbGVyCi0gICAgc3dpdGNoIChtQ2hhbm5lbENvdW50KSB7Ci0gICAgY2FzZSAxOgotICAgICAgICByZXNhbXBsZTwxPihvdXQsIG91dEZyYW1lQ291bnQsIHByb3ZpZGVyKTsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSAyOgotICAgICAgICByZXNhbXBsZTwyPihvdXQsIG91dEZyYW1lQ291bnQsIHByb3ZpZGVyKTsKLSAgICAgICAgYnJlYWs7Ci0gICAgfQotfQotCi0KLXRlbXBsYXRlPGludCBDSEFOTkVMUz4KLXZvaWQgQXVkaW9SZXNhbXBsZXJTaW5jOjpyZXNhbXBsZShpbnQzMl90KiBvdXQsIHNpemVfdCBvdXRGcmFtZUNvdW50LAotICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcikKLXsKLSAgICBpbnQxNl90KiBpbXB1bHNlID0gbUltcHVsc2U7Ci0gICAgdWludDMyX3QgdlJMID0gbVZvbHVtZVJMOwotICAgIHNpemVfdCBpbnB1dEluZGV4ID0gbUlucHV0SW5kZXg7Ci0gICAgdWludDMyX3QgcGhhc2VGcmFjdGlvbiA9IG1QaGFzZUZyYWN0aW9uOwotICAgIHVpbnQzMl90IHBoYXNlSW5jcmVtZW50ID0gbVBoYXNlSW5jcmVtZW50OwotICAgIHNpemVfdCBvdXRwdXRJbmRleCA9IDA7Ci0gICAgc2l6ZV90IG91dHB1dFNhbXBsZUNvdW50ID0gb3V0RnJhbWVDb3VudCAqIDI7Ci0gICAgc2l6ZV90IGluRnJhbWVDb3VudCA9IChvdXRGcmFtZUNvdW50Km1JblNhbXBsZVJhdGUpL21TYW1wbGVSYXRlOwotCi0gICAgQXVkaW9CdWZmZXJQcm92aWRlcjo6QnVmZmVyJiBidWZmZXIobUJ1ZmZlcik7Ci0gICAgd2hpbGUgKG91dHB1dEluZGV4IDwgb3V0cHV0U2FtcGxlQ291bnQpIHsKLSAgICAgICAgLy8gYnVmZmVyIGlzIGVtcHR5LCBmZXRjaCBhIG5ldyBvbmUKLSAgICAgICAgd2hpbGUgKGJ1ZmZlci5mcmFtZUNvdW50ID09IDApIHsKLSAgICAgICAgICAgIGJ1ZmZlci5mcmFtZUNvdW50ID0gaW5GcmFtZUNvdW50OwotICAgICAgICAgICAgcHJvdmlkZXItPmdldE5leHRCdWZmZXIoJmJ1ZmZlcik7Ci0gICAgICAgICAgICBpZiAoYnVmZmVyLnJhdyA9PSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgZ290byByZXNhbXBsZV9leGl0OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY29uc3QgdWludDMyX3QgcGhhc2VJbmRleCA9IHBoYXNlRnJhY3Rpb24gPj4ga051bVBoYXNlQml0czsKLSAgICAgICAgICAgIGlmIChwaGFzZUluZGV4ID09IDEpIHsKLSAgICAgICAgICAgICAgICAvLyByZWFkIG9uZSBmcmFtZQotICAgICAgICAgICAgICAgIHJlYWQ8Q0hBTk5FTFM+KGltcHVsc2UsIHBoYXNlRnJhY3Rpb24sIGJ1ZmZlci5pMTYsIGlucHV0SW5kZXgpOwotICAgICAgICAgICAgfSBlbHNlIGlmIChwaGFzZUluZGV4ID09IDIpIHsKLSAgICAgICAgICAgICAgICAvLyByZWFkIDIgZnJhbWVzCi0gICAgICAgICAgICAgICAgcmVhZDxDSEFOTkVMUz4oaW1wdWxzZSwgcGhhc2VGcmFjdGlvbiwgYnVmZmVyLmkxNiwgaW5wdXRJbmRleCk7Ci0gICAgICAgICAgICAgICAgaW5wdXRJbmRleCsrOwotICAgICAgICAgICAgICAgIGlmIChpbnB1dEluZGV4ID49IG1CdWZmZXIuZnJhbWVDb3VudCkgewotICAgICAgICAgICAgICAgICAgICBpbnB1dEluZGV4IC09IG1CdWZmZXIuZnJhbWVDb3VudDsKLSAgICAgICAgICAgICAgICAgICAgcHJvdmlkZXItPnJlbGVhc2VCdWZmZXIoJmJ1ZmZlcik7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgcmVhZDxDSEFOTkVMUz4oaW1wdWxzZSwgcGhhc2VGcmFjdGlvbiwgYnVmZmVyLmkxNiwgaW5wdXRJbmRleCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgaW50MTZfdCAqaW4gPSBidWZmZXIuaTE2OwotICAgICAgICBjb25zdCBzaXplX3QgZnJhbWVDb3VudCA9IGJ1ZmZlci5mcmFtZUNvdW50OwotCi0gICAgICAgIC8vIEFsd2F5cyByZWFkLWluIHRoZSBmaXJzdCBzYW1wbGVzIGZyb20gdGhlIGlucHV0IGJ1ZmZlcgotICAgICAgICBpbnQxNl90KiBoZWFkID0gaW1wdWxzZSArIGhhbGZOdW1Db2VmcypDSEFOTkVMUzsKLSAgICAgICAgaGVhZFswXSA9IGluW2lucHV0SW5kZXgqQ0hBTk5FTFMgKyAwXTsKLSAgICAgICAgaWYgKENIQU5ORUxTID09IDIpCi0gICAgICAgICAgICBoZWFkWzFdID0gaW5baW5wdXRJbmRleCpDSEFOTkVMUyArIDFdOwotCi0gICAgICAgIC8vIGhhbmRsZSBib3VuZGFyeSBjYXNlCi0gICAgICAgIGludDMyX3QgbCwgcjsKLSAgICAgICAgd2hpbGUgKG91dHB1dEluZGV4IDwgb3V0cHV0U2FtcGxlQ291bnQpIHsKLSAgICAgICAgICAgIGZpbHRlckNvZWZmaWNpZW50PENIQU5ORUxTPihsLCByLCBwaGFzZUZyYWN0aW9uLCBpbXB1bHNlKTsKLSAgICAgICAgICAgIG91dFtvdXRwdXRJbmRleCsrXSArPSAyICogbXVsUkwoMSwgbCwgdlJMKTsKLSAgICAgICAgICAgIG91dFtvdXRwdXRJbmRleCsrXSArPSAyICogbXVsUkwoMCwgciwgdlJMKTsKLQotICAgICAgICAgICAgcGhhc2VGcmFjdGlvbiArPSBwaGFzZUluY3JlbWVudDsKLSAgICAgICAgICAgIGNvbnN0IHVpbnQzMl90IHBoYXNlSW5kZXggPSBwaGFzZUZyYWN0aW9uID4+IGtOdW1QaGFzZUJpdHM7Ci0gICAgICAgICAgICBpZiAocGhhc2VJbmRleCA9PSAxKSB7Ci0gICAgICAgICAgICAgICAgaW5wdXRJbmRleCsrOwotICAgICAgICAgICAgICAgIGlmIChpbnB1dEluZGV4ID49IGZyYW1lQ291bnQpCi0gICAgICAgICAgICAgICAgICAgIGJyZWFrOyAgLy8gbmVlZCBhIG5ldyBidWZmZXIKLSAgICAgICAgICAgICAgICByZWFkPENIQU5ORUxTPihpbXB1bHNlLCBwaGFzZUZyYWN0aW9uLCBpbiwgaW5wdXRJbmRleCk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYocGhhc2VJbmRleCA9PSAyKSB7ICAgIC8vIG1heGltdW0gdmFsdWUKLSAgICAgICAgICAgICAgICBpbnB1dEluZGV4Kys7Ci0gICAgICAgICAgICAgICAgaWYgKGlucHV0SW5kZXggPj0gZnJhbWVDb3VudCkKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7ICAvLyAwIGZyYW1lIGF2YWlsYWJsZSwgMiBmcmFtZXMgbmVlZGVkCi0gICAgICAgICAgICAgICAgLy8gcmVhZCBmaXJzdCBmcmFtZQotICAgICAgICAgICAgICAgIHJlYWQ8Q0hBTk5FTFM+KGltcHVsc2UsIHBoYXNlRnJhY3Rpb24sIGluLCBpbnB1dEluZGV4KTsKLSAgICAgICAgICAgICAgICBpbnB1dEluZGV4Kys7Ci0gICAgICAgICAgICAgICAgaWYgKGlucHV0SW5kZXggPj0gZnJhbWVDb3VudCkKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7ICAvLyAwIGZyYW1lIGF2YWlsYWJsZSwgMSBmcmFtZSBuZWVkZWQKLSAgICAgICAgICAgICAgICAvLyByZWFkIHNlY29uZCBmcmFtZQotICAgICAgICAgICAgICAgIHJlYWQ8Q0hBTk5FTFM+KGltcHVsc2UsIHBoYXNlRnJhY3Rpb24sIGluLCBpbnB1dEluZGV4KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIGlmIGRvbmUgd2l0aCBidWZmZXIsIHNhdmUgc2FtcGxlcwotICAgICAgICBpZiAoaW5wdXRJbmRleCA+PSBmcmFtZUNvdW50KSB7Ci0gICAgICAgICAgICBpbnB1dEluZGV4IC09IGZyYW1lQ291bnQ7Ci0gICAgICAgICAgICBwcm92aWRlci0+cmVsZWFzZUJ1ZmZlcigmYnVmZmVyKTsKLSAgICAgICAgfQotICAgIH0KLQotcmVzYW1wbGVfZXhpdDoKLSAgICBtSW1wdWxzZSA9IGltcHVsc2U7Ci0gICAgbUlucHV0SW5kZXggPSBpbnB1dEluZGV4OwotICAgIG1QaGFzZUZyYWN0aW9uID0gcGhhc2VGcmFjdGlvbjsKLX0KLQotdGVtcGxhdGU8aW50IENIQU5ORUxTPgotLyoqKgotKiByZWFkKCkKLSoKLSogVGhpcyBmdW5jdGlvbiByZWFkcyBvbmx5IG9uZSBmcmFtZSBmcm9tIGlucHV0IGJ1ZmZlciBhbmQgd3JpdGVzIGl0IGluCi0qIHN0YXRlIGJ1ZmZlcgotKgotKiovCi12b2lkIEF1ZGlvUmVzYW1wbGVyU2luYzo6cmVhZCgKLSAgICAgICAgaW50MTZfdComIGltcHVsc2UsIHVpbnQzMl90JiBwaGFzZUZyYWN0aW9uLAotICAgICAgICBpbnQxNl90IGNvbnN0KiBpbiwgc2l6ZV90IGlucHV0SW5kZXgpCi17Ci0gICAgY29uc3QgdWludDMyX3QgcGhhc2VJbmRleCA9IHBoYXNlRnJhY3Rpb24gPj4ga051bVBoYXNlQml0czsKLSAgICBpbXB1bHNlICs9IENIQU5ORUxTOwotICAgIHBoYXNlRnJhY3Rpb24gLT0gMUxVPDxrTnVtUGhhc2VCaXRzOwotICAgIGlmIChpbXB1bHNlID49IG1SaW5nRnVsbCkgewotICAgICAgICBjb25zdCBzaXplX3Qgc3RhdGVTaXplID0gKGhhbGZOdW1Db2VmcyoyKSpDSEFOTkVMUzsKLSAgICAgICAgbWVtY3B5KG1TdGF0ZSwgbVN0YXRlK3N0YXRlU2l6ZSwgc2l6ZW9mKGludDE2X3QpKnN0YXRlU2l6ZSk7Ci0gICAgICAgIGltcHVsc2UgLT0gc3RhdGVTaXplOwotICAgIH0KLSAgICBpbnQxNl90KiBoZWFkID0gaW1wdWxzZSArIGhhbGZOdW1Db2VmcypDSEFOTkVMUzsKLSAgICBoZWFkWzBdID0gaW5baW5wdXRJbmRleCpDSEFOTkVMUyArIDBdOwotICAgIGlmIChDSEFOTkVMUyA9PSAyKQotICAgICAgICBoZWFkWzFdID0gaW5baW5wdXRJbmRleCpDSEFOTkVMUyArIDFdOwotfQotCi10ZW1wbGF0ZTxpbnQgQ0hBTk5FTFM+Ci12b2lkIEF1ZGlvUmVzYW1wbGVyU2luYzo6ZmlsdGVyQ29lZmZpY2llbnQoCi0gICAgICAgIGludDMyX3QmIGwsIGludDMyX3QmIHIsIHVpbnQzMl90IHBoYXNlLCBpbnQxNl90IGNvbnN0ICpzYW1wbGVzKQotewotICAgIC8vIGNvbXB1dGUgdGhlIGluZGV4IG9mIHRoZSBjb2VmZmljaWVudCBvbiB0aGUgcG9zaXRpdmUgc2lkZSBhbmQKLSAgICAvLyBuZWdhdGl2ZSBzaWRlCi0gICAgdWludDMyX3QgaW5kZXhQID0gKHBoYXNlICYgY01hc2spID4+IGNTaGlmdDsKLSAgICB1aW50MTZfdCBsZXJwUCAgPSAocGhhc2UgJiBwTWFzaykgPj4gcFNoaWZ0OwotICAgIHVpbnQzMl90IGluZGV4TiA9ICgtcGhhc2UgJiBjTWFzaykgPj4gY1NoaWZ0OwotICAgIHVpbnQxNl90IGxlcnBOICA9ICgtcGhhc2UgJiBwTWFzaykgPj4gcFNoaWZ0OwotICAgIGlmICgoaW5kZXhQID09IDApICYmIChsZXJwUCA9PSAwKSkgewotICAgICAgICBpbmRleE4gPSBjTWFzayA+PiBjU2hpZnQ7Ci0gICAgICAgIGxlcnBOID0gcE1hc2sgPj4gcFNoaWZ0OwotICAgIH0KLQotICAgIGwgPSAwOwotICAgIHIgPSAwOwotICAgIGludDMyX3QgY29uc3QqIGNvZWZzID0gbUZpckNvZWZzOwotICAgIGludDE2X3QgY29uc3QgKnNQID0gc2FtcGxlczsKLSAgICBpbnQxNl90IGNvbnN0ICpzTiA9IHNhbXBsZXMrQ0hBTk5FTFM7Ci0gICAgZm9yICh1bnNpZ25lZCBpbnQgaT0wIDsgaTxoYWxmTnVtQ29lZnMvNCA7IGkrKykgewotICAgICAgICBpbnRlcnBvbGF0ZTxDSEFOTkVMUz4obCwgciwgY29lZnMraW5kZXhQLCBsZXJwUCwgc1ApOwotICAgICAgICBpbnRlcnBvbGF0ZTxDSEFOTkVMUz4obCwgciwgY29lZnMraW5kZXhOLCBsZXJwTiwgc04pOwotICAgICAgICBzUCAtPSBDSEFOTkVMUzsgc04gKz0gQ0hBTk5FTFM7IGNvZWZzICs9IDE8PGNvZWZzQml0czsKLSAgICAgICAgaW50ZXJwb2xhdGU8Q0hBTk5FTFM+KGwsIHIsIGNvZWZzK2luZGV4UCwgbGVycFAsIHNQKTsKLSAgICAgICAgaW50ZXJwb2xhdGU8Q0hBTk5FTFM+KGwsIHIsIGNvZWZzK2luZGV4TiwgbGVycE4sIHNOKTsKLSAgICAgICAgc1AgLT0gQ0hBTk5FTFM7IHNOICs9IENIQU5ORUxTOyBjb2VmcyArPSAxPDxjb2Vmc0JpdHM7Ci0gICAgICAgIGludGVycG9sYXRlPENIQU5ORUxTPihsLCByLCBjb2VmcytpbmRleFAsIGxlcnBQLCBzUCk7Ci0gICAgICAgIGludGVycG9sYXRlPENIQU5ORUxTPihsLCByLCBjb2VmcytpbmRleE4sIGxlcnBOLCBzTik7Ci0gICAgICAgIHNQIC09IENIQU5ORUxTOyBzTiArPSBDSEFOTkVMUzsgY29lZnMgKz0gMTw8Y29lZnNCaXRzOwotICAgICAgICBpbnRlcnBvbGF0ZTxDSEFOTkVMUz4obCwgciwgY29lZnMraW5kZXhQLCBsZXJwUCwgc1ApOwotICAgICAgICBpbnRlcnBvbGF0ZTxDSEFOTkVMUz4obCwgciwgY29lZnMraW5kZXhOLCBsZXJwTiwgc04pOwotICAgICAgICBzUCAtPSBDSEFOTkVMUzsgc04gKz0gQ0hBTk5FTFM7IGNvZWZzICs9IDE8PGNvZWZzQml0czsKLSAgICB9Ci19Ci0KLXRlbXBsYXRlPGludCBDSEFOTkVMUz4KLXZvaWQgQXVkaW9SZXNhbXBsZXJTaW5jOjppbnRlcnBvbGF0ZSgKLSAgICAgICAgaW50MzJfdCYgbCwgaW50MzJfdCYgciwKLSAgICAgICAgaW50MzJfdCBjb25zdCogY29lZnMsIGludDE2X3QgbGVycCwgaW50MTZfdCBjb25zdCogc2FtcGxlcykKLXsKLSAgICBpbnQzMl90IGMwID0gY29lZnNbMF07Ci0gICAgaW50MzJfdCBjMSA9IGNvZWZzWzFdOwotICAgIGludDMyX3Qgc2luYyA9IG11bEFkZChsZXJwLCAoYzEtYzApPDwxLCBjMCk7Ci0gICAgaWYgKENIQU5ORUxTID09IDIpIHsKLSAgICAgICAgdWludDMyX3QgcmwgPSAqcmVpbnRlcnByZXRfY2FzdDx1aW50MzJfdCBjb25zdCo+KHNhbXBsZXMpOwotICAgICAgICBsID0gbXVsQWRkUkwoMSwgcmwsIHNpbmMsIGwpOwotICAgICAgICByID0gbXVsQWRkUkwoMCwgcmwsIHNpbmMsIHIpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHIgPSBsID0gbXVsQWRkKHNhbXBsZXNbMF0sIHNpbmMsIGwpOwotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy9hdWRpb2ZsaW5nZXIvQXVkaW9SZXNhbXBsZXJTaW5jLmggYi9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb1Jlc2FtcGxlclNpbmMuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZTZjYjkwYi4uMDAwMDAwMAotLS0gYS9saWJzL2F1ZGlvZmxpbmdlci9BdWRpb1Jlc2FtcGxlclNpbmMuaAorKysgL2Rldi9udWxsCkBAIC0xLDg4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfQVVESU9fUkVTQU1QTEVSX1NJTkNfSAotI2RlZmluZSBBTkRST0lEX0FVRElPX1JFU0FNUExFUl9TSU5DX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPGN1dGlscy9sb2cuaD4KLQotI2luY2x1ZGUgIkF1ZGlvUmVzYW1wbGVyLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBBdWRpb1Jlc2FtcGxlclNpbmMgOiBwdWJsaWMgQXVkaW9SZXNhbXBsZXIgewotcHVibGljOgotICAgIEF1ZGlvUmVzYW1wbGVyU2luYyhpbnQgYml0RGVwdGgsIGludCBpbkNoYW5uZWxDb3VudCwgaW50MzJfdCBzYW1wbGVSYXRlKTsKLQotICAgIH5BdWRpb1Jlc2FtcGxlclNpbmMoKTsKLQotICAgIHZpcnR1YWwgdm9pZCByZXNhbXBsZShpbnQzMl90KiBvdXQsIHNpemVfdCBvdXRGcmFtZUNvdW50LAotICAgICAgICAgICAgQXVkaW9CdWZmZXJQcm92aWRlciogcHJvdmlkZXIpOwotcHJpdmF0ZToKLSAgICB2b2lkIGluaXQoKTsKLQotICAgIHRlbXBsYXRlPGludCBDSEFOTkVMUz4KLSAgICB2b2lkIHJlc2FtcGxlKGludDMyX3QqIG91dCwgc2l6ZV90IG91dEZyYW1lQ291bnQsCi0gICAgICAgICAgICBBdWRpb0J1ZmZlclByb3ZpZGVyKiBwcm92aWRlcik7Ci0KLSAgICB0ZW1wbGF0ZTxpbnQgQ0hBTk5FTFM+Ci0gICAgaW5saW5lIHZvaWQgZmlsdGVyQ29lZmZpY2llbnQoCi0gICAgICAgICAgICBpbnQzMl90JiBsLCBpbnQzMl90JiByLCB1aW50MzJfdCBwaGFzZSwgaW50MTZfdCBjb25zdCAqc2FtcGxlcyk7Ci0KLSAgICB0ZW1wbGF0ZTxpbnQgQ0hBTk5FTFM+Ci0gICAgaW5saW5lIHZvaWQgaW50ZXJwb2xhdGUoCi0gICAgICAgICAgICBpbnQzMl90JiBsLCBpbnQzMl90JiByLAotICAgICAgICAgICAgaW50MzJfdCBjb25zdCogY29lZnMsIGludDE2X3QgbGVycCwgaW50MTZfdCBjb25zdCogc2FtcGxlcyk7Ci0KLSAgICB0ZW1wbGF0ZTxpbnQgQ0hBTk5FTFM+Ci0gICAgaW5saW5lIHZvaWQgcmVhZChpbnQxNl90KiYgaW1wdWxzZSwgdWludDMyX3QmIHBoYXNlRnJhY3Rpb24sCi0gICAgICAgICAgICBpbnQxNl90IGNvbnN0KiBpbiwgc2l6ZV90IGlucHV0SW5kZXgpOwotCi0gICAgaW50MTZfdCAqbVN0YXRlOwotICAgIGludDE2X3QgKm1JbXB1bHNlOwotICAgIGludDE2X3QgKm1SaW5nRnVsbDsKLQotICAgIGludDMyX3QgY29uc3QgKiBtRmlyQ29lZnM7Ci0gICAgc3RhdGljIGNvbnN0IGludDMyX3QgbUZpckNvZWZzRG93bltdOwotICAgIHN0YXRpYyBjb25zdCBpbnQzMl90IG1GaXJDb2Vmc1VwW107Ci0KLSAgICAvLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0gICAgc3RhdGljIGNvbnN0IGludDMyX3QgUkVTQU1QTEVfRklSX05VTV9DT0VGICAgICAgID0gODsKLSAgICBzdGF0aWMgY29uc3QgaW50MzJfdCBSRVNBTVBMRV9GSVJfTEVSUF9JTlRfQklUUyAgPSA0OwotCi0gICAgLy8gd2UgaGF2ZSAxNiBjb2VmcyBzYW1wbGVzIHBlciB6ZXJvLWNyb3NzaW5nCi0gICAgc3RhdGljIGNvbnN0IGludCBjb2Vmc0JpdHMgPSBSRVNBTVBMRV9GSVJfTEVSUF9JTlRfQklUUzsgICAgICAgIC8vIDQKLSAgICBzdGF0aWMgY29uc3QgaW50IGNTaGlmdCA9IGtOdW1QaGFzZUJpdHMgLSBjb2Vmc0JpdHM7ICAgICAgICAgICAgLy8gMjYKLSAgICBzdGF0aWMgY29uc3QgdWludDMyX3QgY01hc2sgID0gKCgxPDxjb2Vmc0JpdHMpLTEpIDw8IGNTaGlmdDsgICAgLy8gMHhmPDwyNiA9IDNjMDAgMDAwMAotCi0gICAgLy8gYW5kIHdlIHVzZSAxNSBiaXRzIHRvIGludGVycG9sYXRlIGJldHdlZW4gdGhlc2Ugc2FtcGxlcwotICAgIC8vIHRoaXMgY2Fubm90IGNoYW5nZSBiZWNhdXNlIHRoZSBtdWwgYmVsb3cgcmVseSBvbiBpdC4KLSAgICBzdGF0aWMgY29uc3QgaW50IHBMZXJwQml0cyA9IDE1OwotICAgIHN0YXRpYyBjb25zdCBpbnQgcFNoaWZ0ID0ga051bVBoYXNlQml0cyAtIGNvZWZzQml0cyAtIHBMZXJwQml0czsgICAgLy8gMTEKLSAgICBzdGF0aWMgY29uc3QgdWludDMyX3QgcE1hc2sgID0gKCgxPDxwTGVycEJpdHMpLTEpIDw8IHBTaGlmdDsgICAgLy8gMHg3ZmZmIDw8IDExCi0KLSAgICAvLyBudW1iZXIgb2YgemVyby1jcm9zc2luZyBvbiBlYWNoIHNpZGUKLSAgICBzdGF0aWMgY29uc3QgdW5zaWduZWQgaW50IGhhbGZOdW1Db2VmcyA9IFJFU0FNUExFX0ZJUl9OVU1fQ09FRjsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvKkFORFJPSURfQVVESU9fUkVTQU1QTEVSX1NJTkNfSCovCmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL0FuZHJvaWQubWsgYi9saWJzL3N1cmZhY2VmbGluZ2VyL0FuZHJvaWQubWsKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQ5NmUyNzEuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9BbmRyb2lkLm1rCisrKyAvZGV2L251bGwKQEAgLTEsNDkgKzAsMCBAQAotTE9DQUxfUEFUSDo9ICQoY2FsbCBteS1kaXIpCi1pbmNsdWRlICQoQ0xFQVJfVkFSUykKLQotTE9DQUxfU1JDX0ZJTEVTOj0gXAotICAgIGNsei5jcHAuYXJtIFwKLSAgICBEaXNwbGF5SGFyZHdhcmUvRGlzcGxheUhhcmR3YXJlLmNwcCBcCi0gICAgRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZUJhc2UuY3BwIFwKLSAgICBHUFVIYXJkd2FyZS9HUFVIYXJkd2FyZS5jcHAgXAotICAgIEJvb3RBbmltYXRpb24uY3BwIFwKLSAgICBCbHVyRmlsdGVyLmNwcC5hcm0gXAotICAgIENQVUdhdWdlLmNwcCBcCi0gICAgTGF5ZXIuY3BwIFwKLSAgICBMYXllckJhc2UuY3BwIFwKLSAgICBMYXllckJ1ZmZlci5jcHAgXAotICAgIExheWVyQmx1ci5jcHAgXAotICAgIExheWVyQml0bWFwLmNwcCBcCi0gICAgTGF5ZXJEaW0uY3BwIFwKLSAgICBMYXllck9yaWVudGF0aW9uQW5pbS5jcHAgXAotICAgIE9yaWVudGF0aW9uQW5pbWF0aW9uLmNwcCBcCi0gICAgU3VyZmFjZUZsaW5nZXIuY3BwIFwKLSAgICBUb2tlbml6ZXIuY3BwIFwKLSAgICBUcmFuc2Zvcm0uY3BwIFwKLSAgICBWUmFtSGVhcC5jcHAKLQotCi0jIG5lZWQgIi1scnQiIG9uIExpbnV4IHNpbXVsYXRvciB0byBwaWNrIHVwIGNsb2NrX2dldHRpbWUKLWlmZXEgKCQoVEFSR0VUX1NJTVVMQVRPUiksdHJ1ZSkKLQlpZmVxICgkKEhPU1RfT1MpLGxpbnV4KQotCQlMT0NBTF9MRExJQlMgKz0gLWxydAotCWVuZGlmCi1lbmRpZgotCi1MT0NBTF9TSEFSRURfTElCUkFSSUVTIDo9IFwKLQlsaWJoYXJkd2FyZSBcCi0JbGlidXRpbHMgXAotCWxpYmN1dGlscyBcCi0JbGlidWkgXAotCWxpYmNvcmVjZyBcCi0JbGlic2dsIFwKLQlsaWJwaXhlbGZsaW5nZXIgXAotCWxpYkVHTCBcCi0JbGliR0xFU3YxX0NNCi0KLUxPQ0FMX0NfSU5DTFVERVMgOj0gXAotCSQoY2FsbCBpbmNsdWRlLXBhdGgtZm9yLCBjb3JlY2cgZ3JhcGhpY3MpCi0KLUxPQ0FMX01PRFVMRTo9IGxpYnN1cmZhY2VmbGluZ2VyCi0KLWluY2x1ZGUgJChCVUlMRF9TSEFSRURfTElCUkFSWSkKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvQmFycmllci5oIGIvbGlicy9zdXJmYWNlZmxpbmdlci9CYXJyaWVyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGUyYmNmNmEuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9CYXJyaWVyLmgKKysrIC9kZXYvbnVsbApAQCAtMSw1OSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0JBUlJJRVJfSAotI2RlZmluZSBBTkRST0lEX0JBUlJJRVJfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIEJhcnJpZXIKLXsKLXB1YmxpYzoKLSAgICBpbmxpbmUgQmFycmllcigpIDogc3RhdGUoQ0xPU0VEKSB7IH0KLSAgICBpbmxpbmUgfkJhcnJpZXIoKSB7IH0KLSAgICB2b2lkIG9wZW4oKSB7Ci0gICAgICAgIC8vIGdjYyBtZW1vcnkgYmFycmllciwgdGhpcyBtYWtlcyBzdXJlIGFsbCBtZW1vcnkgd3JpdGVzCi0gICAgICAgIC8vIGhhdmUgYmVlbiBpc3N1ZWQgYnkgZ2NjLiBPbiBhbiBTTVAgc3lzdGVtIHdlJ2QgbmVlZCBhIHJlYWwKLSAgICAgICAgLy8gaC93IGJhcnJpZXIuCi0gICAgICAgIGFzbSB2b2xhdGlsZSAoIiI6OjoibWVtb3J5Iik7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChsb2NrKTsKLSAgICAgICAgc3RhdGUgPSBPUEVORUQ7Ci0gICAgICAgIGN2LmJyb2FkY2FzdCgpOwotICAgIH0KLSAgICB2b2lkIGNsb3NlKCkgewotICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2wobG9jayk7Ci0gICAgICAgIHN0YXRlID0gQ0xPU0VEOwotICAgIH0KLSAgICB2b2lkIHdhaXQoKSBjb25zdCB7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChsb2NrKTsKLSAgICAgICAgd2hpbGUgKHN0YXRlID09IENMT1NFRCkgewotICAgICAgICAgICAgY3Yud2FpdChsb2NrKTsKLSAgICAgICAgfQotICAgIH0KLXByaXZhdGU6Ci0gICAgZW51bSB7IE9QRU5FRCwgQ0xPU0VEIH07Ci0gICAgbXV0YWJsZSAgICAgTXV0ZXggICAgICAgbG9jazsKLSAgICBtdXRhYmxlICAgICBDb25kaXRpb24gICBjdjsKLSAgICB2b2xhdGlsZSAgICBpbnQgICAgICAgICBzdGF0ZTsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0JBUlJJRVJfSApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9CbHVyRmlsdGVyLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvQmx1ckZpbHRlci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDVkYzBiYTAuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9CbHVyRmlsdGVyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDMyNiArMCwwIEBACi0vKiAKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0cmluZy5oPgotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotCi0jaW5jbHVkZSA8cGl4ZWxmbGluZ2VyL3BpeGVsZmxpbmdlci5oPgotCi0jaW5jbHVkZSAiY2x6LmgiCi0KLSNkZWZpbmUgTElLRUxZKCBleHAgKSAgICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgdHJ1ZSAgKSkKLSNkZWZpbmUgVU5MSUtFTFkoIGV4cCApICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgZmFsc2UgKSkKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0jaWYgQllURV9PUkRFUiA9PSBMSVRUTEVfRU5ESUFOCi1pbmxpbmUgdWludDMyX3QgQkxVUl9SR0JBX1RPX0hPU1QodWludDMyX3QgdikgewotICAgIHJldHVybiB2OwotfQotaW5saW5lIHVpbnQzMl90IEJMVVJfSE9TVF9UT19SR0JBKHVpbnQzMl90IHYpIHsKLSAgICByZXR1cm4gdjsKLX0KLSNlbHNlCi1pbmxpbmUgdWludDMyX3QgQkxVUl9SR0JBX1RPX0hPU1QodWludDMyX3QgdikgewotICAgIHJldHVybiAodjw8MjQpIHwgKHY+PjI0KSB8ICgodjw8OCkmMHhmZjAwMDApIHwgKCh2Pj44KSYweGZmMDApOwotfQotaW5saW5lIHVpbnQzMl90IEJMVVJfSE9TVF9UT19SR0JBKHVpbnQzMl90IHYpIHsKLSAgICByZXR1cm4gKHY8PDI0KSB8ICh2Pj4yNCkgfCAoKHY8PDgpJjB4ZmYwMDAwKSB8ICgodj4+OCkmMHhmZjAwKTsKLX0KLSNlbmRpZgotCi1jb25zdCBpbnQgQkxVUl9ESVRIRVJfQklUUyA9IDY7ICAvLyBkaXRoZXIgd2VpZ2h0cyBzdG9yZWQgb24gNiBiaXRzCi1jb25zdCBpbnQgQkxVUl9ESVRIRVJfT1JERVJfU0hJRlQ9IDM7Ci1jb25zdCBpbnQgQkxVUl9ESVRIRVJfT1JERVIgICAgICA9ICgxPDxCTFVSX0RJVEhFUl9PUkRFUl9TSElGVCk7Ci1jb25zdCBpbnQgQkxVUl9ESVRIRVJfU0laRSAgICAgICA9IEJMVVJfRElUSEVSX09SREVSICogQkxVUl9ESVRIRVJfT1JERVI7Ci1jb25zdCBpbnQgQkxVUl9ESVRIRVJfTUFTSyAgICAgICA9IEJMVVJfRElUSEVSX09SREVSLTE7Ci0KLXN0YXRpYyBjb25zdCB1aW50OF90IGdEaXRoZXJNYXRyaXhbQkxVUl9ESVRIRVJfU0laRV0gPSB7Ci0gICAgIDAsIDMyLCAgOCwgNDAsICAyLCAzNCwgMTAsIDQyLAotICAgIDQ4LCAxNiwgNTYsIDI0LCA1MCwgMTgsIDU4LCAyNiwKLSAgICAxMiwgNDQsICA0LCAzNiwgMTQsIDQ2LCAgNiwgMzgsCi0gICAgNjAsIDI4LCA1MiwgMjAsIDYyLCAzMCwgNTQsIDIyLAotICAgICAzLCAzNSwgMTEsIDQzLCAgMSwgMzMsICA5LCA0MSwKLSAgICA1MSwgMTksIDU5LCAyNywgNDksIDE3LCA1NywgMjUsCi0gICAgMTUsIDQ3LCAgNywgMzksIDEzLCA0NSwgIDUsIDM3LAotICAgIDYzLCAzMSwgNTUsIDIzLCA2MSwgMjksIDUzLCAyMQotfTsKLQotCi10ZW1wbGF0ZSA8aW50IEZBQ1RPUiA9IDA+Ci1zdHJ1Y3QgQmx1ckNvbG9yNTY1Ci17Ci0gICAgdHlwZWRlZiB1aW50MTZfdCB0eXBlOwotICAgIGludCByLCBnLCBiOyAgICAKLSAgICBpbmxpbmUgQmx1ckNvbG9yNTY1KCkgeyB9Ci0gICAgaW5saW5lIEJsdXJDb2xvcjU2NSh1aW50MTZfdCB2KSB7Ci0gICAgICAgIHIgPSB2ID4+IDExOwotICAgICAgICBnID0gKHYgPj4gNSkgJiAweDNFOwotICAgICAgICBiID0gdiAmIDB4MUY7Ci0gICAgfQotICAgIGlubGluZSB2b2lkIGNsZWFyKCkgeyByPWc9Yj0wOyB9Ci0gICAgaW5saW5lIHVpbnQxNl90IHRvKGludCBzaGlmdCwgaW50IGxhc3QsIGludCBkaXRoZXIpIGNvbnN0IHsKLSAgICAgICAgaW50IFIgPSByOwotICAgICAgICBpbnQgRyA9IGc7Ci0gICAgICAgIGludCBCID0gYjsKLSAgICAgICAgaWYgIChVTkxJS0VMWShsYXN0KSkgewotICAgICAgICAgICAgaWYgKEZBQ1RPUj4wKSB7Ci0gICAgICAgICAgICAgICAgaW50IEwgPSAoUitHK0IpPj4xOwotICAgICAgICAgICAgICAgIFIgKz0gKCgoTD4+MSkgLSBSKSAqIEZBQ1RPUikgPj4gODsKLSAgICAgICAgICAgICAgICBHICs9ICgoKEwgICApIC0gRykgKiBGQUNUT1IpID4+IDg7Ci0gICAgICAgICAgICAgICAgQiArPSAoKChMPj4xKSAtIEIpICogRkFDVE9SKSA+PiA4OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgUiArPSAoZGl0aGVyIDw8IHNoaWZ0KSA+PiBCTFVSX0RJVEhFUl9CSVRTOwotICAgICAgICAgICAgRyArPSAoZGl0aGVyIDw8IHNoaWZ0KSA+PiBCTFVSX0RJVEhFUl9CSVRTOwotICAgICAgICAgICAgQiArPSAoZGl0aGVyIDw8IHNoaWZ0KSA+PiBCTFVSX0RJVEhFUl9CSVRTOwotICAgICAgICB9Ci0gICAgICAgIFIgPj49IHNoaWZ0OwotICAgICAgICBHID4+PSBzaGlmdDsKLSAgICAgICAgQiA+Pj0gc2hpZnQ7Ci0gICAgICAgIHJldHVybiAoUjw8MTEpIHwgKEc8PDUpIHwgQjsKLSAgICB9ICAgIAotICAgIGlubGluZSBCbHVyQ29sb3I1NjUmIG9wZXJhdG9yICs9IChjb25zdCBCbHVyQ29sb3I1NjUmIHJocykgewotICAgICAgICByICs9IHJocy5yOwotICAgICAgICBnICs9IHJocy5nOwotICAgICAgICBiICs9IHJocy5iOwotICAgICAgICByZXR1cm4gKnRoaXM7Ci0gICAgfQotICAgIGlubGluZSBCbHVyQ29sb3I1NjUmIG9wZXJhdG9yIC09IChjb25zdCBCbHVyQ29sb3I1NjUmIHJocykgewotICAgICAgICByIC09IHJocy5yOwotICAgICAgICBnIC09IHJocy5nOwotICAgICAgICBiIC09IHJocy5iOwotICAgICAgICByZXR1cm4gKnRoaXM7Ci0gICAgfQotfTsKLQotc3RydWN0IEJsdXJHcmF5NTY1Ci17Ci0gICAgdHlwZWRlZiB1aW50MTZfdCB0eXBlOwotICAgIGludCBsOyAgICAKLSAgICBpbmxpbmUgQmx1ckdyYXk1NjUoKSB7IH0KLSAgICBpbmxpbmUgQmx1ckdyYXk1NjUodWludDE2X3QgdikgewotICAgICAgICBpbnQgciA9IHYgPj4gMTE7Ci0gICAgICAgIGludCBnID0gKHYgPj4gNSkgJiAweDNGOwotICAgICAgICBpbnQgYiA9IHYgJiAweDFGOwotICAgICAgICBsID0gKHIgKyBnICsgYiArIDEpPj4xOwotICAgIH0KLSAgICBpbmxpbmUgdm9pZCBjbGVhcigpIHsgbD0wOyB9Ci0gICAgaW5saW5lIHVpbnQxNl90IHRvKGludCBzaGlmdCwgaW50IGxhc3QsIGludCBkaXRoZXIpIGNvbnN0IHsKLSAgICAgICAgaW50IEwgPSBsOwotICAgICAgICBpZiAgKFVOTElLRUxZKGxhc3QpKSB7Ci0gICAgICAgICAgICBMICs9IChkaXRoZXIgPDwgc2hpZnQpID4+IEJMVVJfRElUSEVSX0JJVFM7Ci0gICAgICAgIH0KLSAgICAgICAgTCA+Pj0gc2hpZnQ7Ci0gICAgICAgIHJldHVybiAoKEw+PjEpPDwxMSkgfCAoTDw8NSkgfCAoTD4+MSk7Ci0gICAgfQotICAgIGlubGluZSBCbHVyR3JheTU2NSYgb3BlcmF0b3IgKz0gKGNvbnN0IEJsdXJHcmF5NTY1JiByaHMpIHsKLSAgICAgICAgbCArPSByaHMubDsKLSAgICAgICAgcmV0dXJuICp0aGlzOwotICAgIH0KLSAgICBpbmxpbmUgQmx1ckdyYXk1NjUmIG9wZXJhdG9yIC09IChjb25zdCBCbHVyR3JheTU2NSYgcmhzKSB7Ci0gICAgICAgIGwgLT0gcmhzLmw7Ci0gICAgICAgIHJldHVybiAqdGhpczsKLSAgICB9Ci19OwotCi1zdHJ1Y3QgQmx1ckdyYXk4ODg4Ci17Ci0gICAgdHlwZWRlZiB1aW50MzJfdCB0eXBlOwotICAgIGludCBsLCBhOyAgICAKLSAgICBpbmxpbmUgQmx1ckdyYXk4ODg4KCkgeyB9Ci0gICAgaW5saW5lIEJsdXJHcmF5ODg4OCh1aW50MzJfdCB2KSB7Ci0gICAgICAgIHYgPSBCTFVSX1JHQkFfVE9fSE9TVCh2KTsKLSAgICAgICAgaW50IHIgPSB2ICYgMHhGRjsKLSAgICAgICAgaW50IGcgPSAodiA+PiAgOCkgJiAweEZGOwotICAgICAgICBpbnQgYiA9ICh2ID4+IDE2KSAmIDB4RkY7Ci0gICAgICAgIGEgPSB2ID4+IDI0OwotICAgICAgICBsID0gciArIGcgKyBnICsgYjsKLSAgICB9ICAgIAotICAgIGlubGluZSB2b2lkIGNsZWFyKCkgeyBsPWE9MDsgfQotICAgIGlubGluZSB1aW50MzJfdCB0byhpbnQgc2hpZnQsIGludCBsYXN0LCBpbnQgZGl0aGVyKSBjb25zdCB7Ci0gICAgICAgIGludCBMID0gbDsKLSAgICAgICAgaW50IEEgPSBhOwotICAgICAgICBpZiAgKFVOTElLRUxZKGxhc3QpKSB7Ci0gICAgICAgICAgICBMICs9IChkaXRoZXIgPDwgKHNoaWZ0KzIpKSA+PiBCTFVSX0RJVEhFUl9CSVRTOwotICAgICAgICAgICAgQSArPSAoZGl0aGVyIDw8IHNoaWZ0KSA+PiBCTFVSX0RJVEhFUl9CSVRTOwotICAgICAgICB9Ci0gICAgICAgIEwgPj49IChzaGlmdCsyKTsKLSAgICAgICAgQSA+Pj0gc2hpZnQ7Ci0gICAgICAgIHJldHVybiBCTFVSX0hPU1RfVE9fUkdCQSgoQTw8MjQpIHwgKEw8PDE2KSB8IChMPDw4KSB8IEwpOwotICAgIH0KLSAgICBpbmxpbmUgQmx1ckdyYXk4ODg4JiBvcGVyYXRvciArPSAoY29uc3QgQmx1ckdyYXk4ODg4JiByaHMpIHsKLSAgICAgICAgbCArPSByaHMubDsKLSAgICAgICAgYSArPSByaHMuYTsKLSAgICAgICAgcmV0dXJuICp0aGlzOwotICAgIH0KLSAgICBpbmxpbmUgQmx1ckdyYXk4ODg4JiBvcGVyYXRvciAtPSAoY29uc3QgQmx1ckdyYXk4ODg4JiByaHMpIHsKLSAgICAgICAgbCAtPSByaHMubDsKLSAgICAgICAgYSAtPSByaHMuYTsKLSAgICAgICAgcmV0dXJuICp0aGlzOwotICAgIH0KLX07Ci0KLQotdGVtcGxhdGU8dHlwZW5hbWUgUElYRUw+Ci1zdGF0aWMgc3RhdHVzX3QgYmx1ckZpbHRlcigKLSAgICAgICAgR0dMU3VyZmFjZSBjb25zdCogZHN0LAotICAgICAgICBHR0xTdXJmYWNlIGNvbnN0KiBzcmMsCi0gICAgICAgIGludCBrZXJuZWxTaXplVXNlciwKLSAgICAgICAgaW50IHJlcGVhdCkKLXsKLSAgICB0eXBlZGVmIHR5cGVuYW1lIFBJWEVMOjp0eXBlIFRZUEU7Ci0KLSAgICBjb25zdCBpbnQgc2hpZnQgICAgICAgICAgICAgPSAzMSAtIGNseihrZXJuZWxTaXplVXNlcik7Ci0gICAgY29uc3QgaW50IGFyZWFTaGlmdCAgICAgICAgID0gc2hpZnQqMjsKLSAgICBjb25zdCBpbnQga2VybmVsU2l6ZSAgICAgICAgPSAxPDxzaGlmdDsKLSAgICBjb25zdCBpbnQga2VybmVsSGFsZlNpemUgICAgPSBrZXJuZWxTaXplLzI7Ci0gICAgY29uc3QgaW50IG1hc2sgICAgICAgICAgICAgID0ga2VybmVsU2l6ZS0xOwotICAgIGNvbnN0IGludCB3ICAgICAgICAgICAgICAgICA9IHNyYy0+d2lkdGg7Ci0gICAgY29uc3QgaW50IGggICAgICAgICAgICAgICAgID0gc3JjLT5oZWlnaHQ7Ci0gICAgY29uc3QgdWludDhfdCogZGl0aGVyTWF0cml4ID0gZ0RpdGhlck1hdHJpeDsKLQotICAgIC8vIHdlIG5lZWQgYSB0ZW1wb3JhcnkgYnVmZmVyIHRvIHN0b3JlIG9uZSBsaW5lIG9mIGJsdXJyZWQgY29sdW1ucwotICAgIC8vIGFzIHdlbGwgYXMga2VybmVsU2l6ZSBsaW5lcyBvZiBzb3VyY2UgcGl4ZWxzIG9yZ2FuaXplZCBhcyBhIHJpbmcgYnVmZmVyLgotICAgIHZvaWQqIGNvbnN0IHRlbXBvcmFyeV9idWZmZXIgPSBtYWxsb2MoCi0gICAgICAgICAgICAodyArIGtlcm5lbFNpemUpICogc2l6ZW9mKFBJWEVMKSArCi0gICAgICAgICAgICAoc3JjLT5zdHJpZGUgKiBrZXJuZWxTaXplKSAqIHNpemVvZihUWVBFKSk7Ci0gICAgaWYgKCF0ZW1wb3JhcnlfYnVmZmVyKQotICAgICAgICByZXR1cm4gTk9fTUVNT1JZOwotCi0gICAgUElYRUwqIGNvbnN0IHN1bXMgPSAoUElYRUwqKXRlbXBvcmFyeV9idWZmZXI7Ci0gICAgVFlQRSogY29uc3Qgc2NyYXRjaCA9IChUWVBFKikoc3VtcyArIHcgKyBrZXJuZWxTaXplKTsKLQotICAgIC8vIEFwcGx5IHRoZSBibHVyICdyZXBlYXQnIHRpbWVzLCB0aGlzIGlzIHVzZWQgdG8gYXBwcm94aW1hdGUKLSAgICAvLyBnYXVzc2lhbiBibHVycy4gMyB0aW1lcyBnaXZlcyBnb29kIHJlc3VsdHMuCi0gICAgZm9yIChpbnQgaz0wIDsgazxyZXBlYXQgOyBrKyspIHsKLQotICAgICAgICAvLyBDbGVhciB0aGUgY29sdW1ucyBzdW1zIGZvciB0aGlzIHJvdW5kCi0gICAgICAgIG1lbXNldChzdW1zLCAwLCAodyArIGtlcm5lbFNpemUpICogc2l6ZW9mKFBJWEVMKSk7Ci0gICAgICAgIFRZUEUqIGhlYWQ7Ci0gICAgICAgIFRZUEUgcGl4ZWw7Ci0gICAgICAgIFBJWEVMIGN1cnJlbnQ7Ci0KLSAgICAgICAgLy8gU2luY2Ugd2UncmUgZ29pbmcgdG8gb3ZlcnJpZGUgdGhlIHNvdXJjZSBkYXRhIHdlIG5lZWQKLSAgICAgICAgLy8gdG8gY29weSBpdCBpbiBhIHRlbXBvcmFyeSBidWZmZXIuIE9ubHkga2VybmVsU2l6ZSBsaW5lcyBhcmUKLSAgICAgICAgLy8gcmVxdWlyZWQuIEJ1dCBzaW5jZSB3ZSBzdGFydCBpbiB0aGUgY2VudGVyIG9mIHRoZSBrZXJuZWwsCi0gICAgICAgIC8vIHdlIG9ubHkgY29weSBoYWxmIG9mIHRoZSBkYXRhLCBhbmQgZmlsbCB0aGUgcmVzdCB3aXRoIHplcm9zCi0gICAgICAgIC8vIChhc3N1bWluZyBibGFjay90cmFuc3BhcmVudCBwaXhlbHMpLgotICAgICAgICBtZW1jcHkoIHNjcmF0Y2ggKyBzcmMtPnN0cmlkZSprZXJuZWxIYWxmU2l6ZSwKLSAgICAgICAgICAgICAgICBzcmMtPmRhdGEsCi0gICAgICAgICAgICAgICAgc3JjLT5zdHJpZGUqa2VybmVsSGFsZlNpemUqc2l6ZW9mKFRZUEUpKTsKLQotICAgICAgICAvLyBzdW0gaGFsZiBvZiBlYWNoIGNvbHVtbiwgYmVjYXVzZSB3ZSBhc3N1bWUgdGhlIGZpcnN0IGhhbGYgaXMKLSAgICAgICAgLy8gemVyb3MgKGJsYWNrL3RyYW5zcGFyZW50KS4KLSAgICAgICAgZm9yIChpbnQgeT0wIDsgeTxrZXJuZWxIYWxmU2l6ZSA7IHkrKykgewotICAgICAgICAgICAgaGVhZCA9IChUWVBFKilzcmMtPmRhdGEgKyB5KnNyYy0+c3RyaWRlOwotICAgICAgICAgICAgZm9yIChpbnQgeD0wIDsgeDx3IDsgeCsrKQotICAgICAgICAgICAgICAgIHN1bXNbeF0gKz0gUElYRUwoICpoZWFkKysgKTsKLSAgICAgICAgfQotCi0gICAgICAgIGZvciAoaW50IHk9MCA7IHk8aCA7IHkrKykgewotICAgICAgICAgICAgVFlQRSogZmIgPSAoVFlQRSopZHN0LT5kYXRhICsgeSpkc3QtPnN0cmlkZTsKLQotICAgICAgICAgICAgLy8gY29tcHV0ZSB0aGUgZGl0aGVyIG1hdHJpeCBsaW5lCi0gICAgICAgICAgICB1aW50OF90IGNvbnN0ICogZGl0aGVyWSA9IGRpdGhlck1hdHJpeAotICAgICAgICAgICAgICAgICAgICArICh5ICYgQkxVUl9ESVRIRVJfTUFTSykqQkxVUl9ESVRIRVJfT1JERVI7Ci0KLSAgICAgICAgICAgIC8vIEhvcml6b250YWwgYmx1ciBwYXNzIG9uIHRoZSBjb2x1bW5zIHN1bXMKLSAgICAgICAgICAgIGludCBjb3VudCwgZGl0aGVyLCB4PTA7Ci0gICAgICAgICAgICBQSVhFTCBjb25zdCAqIG91dD0gc3VtczsKLSAgICAgICAgICAgIFBJWEVMIGNvbnN0ICogaW4gPSBzdW1zOwotICAgICAgICAgICAgY3VycmVudC5jbGVhcigpOwotCi0gICAgICAgICAgICBjb3VudCA9IGtlcm5lbEhhbGZTaXplOwotICAgICAgICAgICAgZG8gewotICAgICAgICAgICAgICAgIGN1cnJlbnQgKz0gKmluOwotICAgICAgICAgICAgICAgIGluKys7Ci0gICAgICAgICAgICB9IHdoaWxlICgtLWNvdW50KTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgY291bnQgPSBrZXJuZWxIYWxmU2l6ZTsKLSAgICAgICAgICAgIGRvIHsKLSAgICAgICAgICAgICAgICBjdXJyZW50ICs9ICppbjsKLSAgICAgICAgICAgICAgICBkaXRoZXIgPSAqKGRpdGhlclkgKyAoKHgrKykmQkxVUl9ESVRIRVJfTUFTSykpOwotICAgICAgICAgICAgICAgICpmYisrID0gY3VycmVudC50byhhcmVhU2hpZnQsIGs9PXJlcGVhdC0xLCBkaXRoZXIpOwotICAgICAgICAgICAgICAgIGluKys7Ci0gICAgICAgICAgICB9IHdoaWxlICgtLWNvdW50KTsKLQotICAgICAgICAgICAgY291bnQgPSB3LWtlcm5lbFNpemU7Ci0gICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgY3VycmVudCArPSAqaW47Ci0gICAgICAgICAgICAgICAgY3VycmVudCAtPSAqb3V0OwotICAgICAgICAgICAgICAgIGRpdGhlciA9ICooZGl0aGVyWSArICgoeCsrKSZCTFVSX0RJVEhFUl9NQVNLKSk7Ci0gICAgICAgICAgICAgICAgKmZiKysgPSBjdXJyZW50LnRvKGFyZWFTaGlmdCwgaz09cmVwZWF0LTEsIGRpdGhlcik7Ci0gICAgICAgICAgICAgICAgaW4rKywgb3V0Kys7Ci0gICAgICAgICAgICB9IHdoaWxlICgtLWNvdW50KTsKLQotICAgICAgICAgICAgY291bnQgPSBrZXJuZWxIYWxmU2l6ZTsKLSAgICAgICAgICAgIGRvIHsKLSAgICAgICAgICAgICAgICBjdXJyZW50IC09ICpvdXQ7Ci0gICAgICAgICAgICAgICAgZGl0aGVyID0gKihkaXRoZXJZICsgKCh4KyspJkJMVVJfRElUSEVSX01BU0spKTsKLSAgICAgICAgICAgICAgICAqZmIrKyA9IGN1cnJlbnQudG8oYXJlYVNoaWZ0LCBrPT1yZXBlYXQtMSwgZGl0aGVyKTsKLSAgICAgICAgICAgICAgICBvdXQrKzsKLSAgICAgICAgICAgIH0gd2hpbGUgKC0tY291bnQpOwotCi0gICAgICAgICAgICAvLyB2ZXJ0aWNhbCBibHVyIHBhc3MsIHN1YnRyYWN0IHRoZSBvbGRlc3QgbGluZSBmcm9tIGVhY2ggY29sdW1ucwotICAgICAgICAgICAgLy8gYW5kIGFkZCBhIG5ldyBsaW5lLiBTdWJ0cmFjdCBvciBhZGQgemVyb3MgYXQgdGhlIHRvcAotICAgICAgICAgICAgLy8gYW5kIGJvdHRvbSBlZGdlcy4KLSAgICAgICAgICAgIFRZUEUqIGNvbnN0IHRhaWwgPSBzY3JhdGNoICsgKHkgJiBtYXNrKSAqIHNyYy0+c3RyaWRlOwotICAgICAgICAgICAgaWYgKHkgPj0ga2VybmVsSGFsZlNpemUpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCB4PTAgOyB4PHcgOyB4KyspCi0gICAgICAgICAgICAgICAgICAgIHN1bXNbeF0gLT0gUElYRUwoIHRhaWxbeF0gKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh5IDwgaC1rZXJuZWxTaXplKSB7Ci0gICAgICAgICAgICAgICAgbWVtY3B5KCB0YWlsLAotICAgICAgICAgICAgICAgICAgICAgICAgKFRZUEUqKXNyYy0+ZGF0YSArICh5K2tlcm5lbEhhbGZTaXplKSpzcmMtPnN0cmlkZSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHNyYy0+c3RyaWRlKnNpemVvZihUWVBFKSk7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgeD0wIDsgeDx3IDsgeCsrKQotICAgICAgICAgICAgICAgICAgICBzdW1zW3hdICs9IFBJWEVMKCB0YWlsW3hdICk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBUaGUgc3Vic2VxdWVudCBwYXNzZXMgYXJlIGFsd2F5cyBkb25lIGluLXBsYWNlLgotICAgICAgICBzcmMgPSBkc3Q7Ci0gICAgfQotICAgIAotICAgIGZyZWUodGVtcG9yYXJ5X2J1ZmZlcik7Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXRlbXBsYXRlIHN0YXR1c190IGJsdXJGaWx0ZXI8IEJsdXJDb2xvcjU2NTwweDgwPiA+KAotICAgICAgICBHR0xTdXJmYWNlIGNvbnN0KiBkc3QsCi0gICAgICAgIEdHTFN1cmZhY2UgY29uc3QqIHNyYywKLSAgICAgICAgaW50IGtlcm5lbFNpemVVc2VyLAotICAgICAgICBpbnQgcmVwZWF0KTsKLQotc3RhdHVzX3QgYmx1ckZpbHRlcigKLSAgICAgICAgR0dMU3VyZmFjZSBjb25zdCogaW1hZ2UsCi0gICAgICAgIGludCBrZXJuZWxTaXplVXNlciwKLSAgICAgICAgaW50IHJlcGVhdCkKLXsKLSAgICByZXR1cm4gYmx1ckZpbHRlcjwgQmx1ckNvbG9yNTY1PDB4ODA+ID4oaW1hZ2UsIGltYWdlLCBrZXJuZWxTaXplVXNlciwgcmVwZWF0KTsKLX0KLQotfSAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0vL2VyciA9IGJsdXI8IEJsdXJDb2xvcjU2NTwweDgwPiA+KGRzdCwgc3JjLCBrZXJuZWxTaXplVXNlciwgcmVwZWF0KTsKLS8vZXJyID0gYmx1cjxCbHVyR3JheTU2NT4oZHN0LCBzcmMsIGtlcm5lbFNpemVVc2VyLCByZXBlYXQpOwotLy9lcnIgPSBibHVyPEJsdXJHcmF5ODg4OD4oZHN0LCBzcmMsIGtlcm5lbFNpemVVc2VyLCByZXBlYXQpOwpkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9CbHVyRmlsdGVyLmggYi9saWJzL3N1cmZhY2VmbGluZ2VyL0JsdXJGaWx0ZXIuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjk0ZGI0My4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0JsdXJGaWx0ZXIuaAorKysgL2Rldi9udWxsCkBAIC0xLDM1ICswLDAgQEAKLS8qIAotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9CTFVSX0ZJTFRFUl9ICi0jZGVmaW5lIEFORFJPSURfQkxVUl9GSUxURVJfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0KLSNpbmNsdWRlIDxwaXhlbGZsaW5nZXIvcGl4ZWxmbGluZ2VyLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotc3RhdHVzX3QgYmx1ckZpbHRlcigKLSAgICAgICAgR0dMU3VyZmFjZSBjb25zdCogaW1hZ2UsCi0gICAgICAgIGludCBrZXJuZWxTaXplVXNlciwKLSAgICAgICAgaW50IHJlcGVhdCk7Ci0KLX0gLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfQkxVUl9GSUxURVJfSApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9Cb290QW5pbWF0aW9uLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvQm9vdEFuaW1hdGlvbi5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJiMzAzMzYuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9Cb290QW5pbWF0aW9uLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDQwMyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJCb290QW5pbWF0aW9uIgotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8bWF0aC5oPgotI2luY2x1ZGUgPGZjbnRsLmg+Ci0jaW5jbHVkZSA8dXRpbHMvbWlzYy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotI2luY2x1ZGUgPHV0aWxzL0F0b21pYy5oPgotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL0Fzc2V0TWFuYWdlci5oPgotCi0jaW5jbHVkZSA8dWkvUGl4ZWxGb3JtYXQuaD4KLSNpbmNsdWRlIDx1aS9SZWN0Lmg+Ci0jaW5jbHVkZSA8dWkvUmVnaW9uLmg+Ci0jaW5jbHVkZSA8dWkvRGlzcGxheUluZm8uaD4KLSNpbmNsdWRlIDx1aS9JU3VyZmFjZUNvbXBvc2VyLmg+Ci0jaW5jbHVkZSA8dWkvSVN1cmZhY2VGbGluZ2VyQ2xpZW50Lmg+Ci0jaW5jbHVkZSA8dWkvRUdMTmF0aXZlV2luZG93U3VyZmFjZS5oPgotCi0jaW5jbHVkZSA8Y29yZS9Ta0JpdG1hcC5oPgotI2luY2x1ZGUgPGltYWdlcy9Ta0ltYWdlRGVjb2Rlci5oPgotCi0jaW5jbHVkZSA8R0xFUy9nbC5oPgotI2luY2x1ZGUgPEdMRVMvZ2xleHQuaD4KLSNpbmNsdWRlIDxFR0wvZWdsZXh0Lmg+Ci0KLSNpbmNsdWRlICJCb290QW5pbWF0aW9uLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUJvb3RBbmltYXRpb246OkJvb3RBbmltYXRpb24oY29uc3Qgc3A8SVN1cmZhY2VDb21wb3Nlcj4mIGNvbXBvc2VyKSA6Ci0gICAgVGhyZWFkKGZhbHNlKSB7Ci0gICAgbVNlc3Npb24gPSBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OmNsaWVudEZvckNvbm5lY3Rpb24oCi0gICAgICAgICAgICBjb21wb3Nlci0+Y3JlYXRlQ29ubmVjdGlvbigpLT5hc0JpbmRlcigpKTsKLX0KLQotQm9vdEFuaW1hdGlvbjo6fkJvb3RBbmltYXRpb24oKSB7Ci19Ci0KLXZvaWQgQm9vdEFuaW1hdGlvbjo6b25GaXJzdFJlZigpIHsKLSAgICBydW4oIkJvb3RBbmltYXRpb24iLCBQUklPUklUWV9ESVNQTEFZKTsKLX0KLQotY29uc3Qgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiYgQm9vdEFuaW1hdGlvbjo6c2Vzc2lvbigpIGNvbnN0IHsKLSAgICByZXR1cm4gbVNlc3Npb247Ci19Ci0KLXN0YXR1c190IEJvb3RBbmltYXRpb246OmluaXRUZXh0dXJlKFRleHR1cmUqIHRleHR1cmUsIEFzc2V0TWFuYWdlciYgYXNzZXRzLAotICAgICAgICBjb25zdCBjaGFyKiBuYW1lKSB7Ci0gICAgQXNzZXQqIGFzc2V0ID0gYXNzZXRzLm9wZW4obmFtZSwgQXNzZXQ6OkFDQ0VTU19CVUZGRVIpOwotICAgIGlmICghYXNzZXQpCi0gICAgICAgIHJldHVybiBOT19JTklUOwotICAgIFNrQml0bWFwIGJpdG1hcDsKLSAgICBTa0ltYWdlRGVjb2Rlcjo6RGVjb2RlTWVtb3J5KGFzc2V0LT5nZXRCdWZmZXIoZmFsc2UpLCBhc3NldC0+Z2V0TGVuZ3RoKCksCi0gICAgICAgICAgICAmYml0bWFwLCBTa0JpdG1hcDo6a05vX0NvbmZpZywgU2tJbWFnZURlY29kZXI6OmtEZWNvZGVQaXhlbHNfTW9kZSk7Ci0gICAgYXNzZXQtPmNsb3NlKCk7Ci0gICAgZGVsZXRlIGFzc2V0OwotCi0gICAgLy8gZW5zdXJlIHdlIGNhbiBjYWxsIGdldFBpeGVscygpLiBObyBuZWVkIHRvIGNhbGwgdW5sb2NrLCBzaW5jZSB0aGUKLSAgICAvLyBiaXRtYXAgd2lsbCBnbyBvdXQgb2Ygc2NvcGUgd2hlbiB3ZSByZXR1cm4gZnJvbSB0aGlzIG1ldGhvZC4KLSAgICBiaXRtYXAubG9ja1BpeGVscygpOwotCi0gICAgY29uc3QgaW50IHcgPSBiaXRtYXAud2lkdGgoKTsKLSAgICBjb25zdCBpbnQgaCA9IGJpdG1hcC5oZWlnaHQoKTsKLSAgICBjb25zdCB2b2lkKiBwID0gYml0bWFwLmdldFBpeGVscygpOwotCi0gICAgR0xpbnQgY3JvcFs0XSA9IHsgMCwgaCwgdywgLWggfTsKLSAgICB0ZXh0dXJlLT53ID0gdzsKLSAgICB0ZXh0dXJlLT5oID0gaDsKLQotICAgIGdsR2VuVGV4dHVyZXMoMSwgJnRleHR1cmUtPm5hbWUpOwotICAgIGdsQmluZFRleHR1cmUoR0xfVEVYVFVSRV8yRCwgdGV4dHVyZS0+bmFtZSk7Ci0KLSAgICBzd2l0Y2ggKGJpdG1hcC5nZXRDb25maWcoKSkgewotICAgICAgICBjYXNlIFNrQml0bWFwOjprQThfQ29uZmlnOgotICAgICAgICAgICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsIEdMX0FMUEhBLCB3LCBoLCAwLCBHTF9BTFBIQSwKLSAgICAgICAgICAgICAgICAgICAgR0xfVU5TSUdORURfQllURSwgcCk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBTa0JpdG1hcDo6a0FSR0JfNDQ0NF9Db25maWc6Ci0gICAgICAgICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwgR0xfUkdCQSwgdywgaCwgMCwgR0xfUkdCQSwKLSAgICAgICAgICAgICAgICAgICAgR0xfVU5TSUdORURfU0hPUlRfNF80XzRfNCwgcCk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBTa0JpdG1hcDo6a0FSR0JfODg4OF9Db25maWc6Ci0gICAgICAgICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwgR0xfUkdCQSwgdywgaCwgMCwgR0xfUkdCQSwKLSAgICAgICAgICAgICAgICAgICAgR0xfVU5TSUdORURfQllURSwgcCk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBTa0JpdG1hcDo6a1JHQl81NjVfQ29uZmlnOgotICAgICAgICAgICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsIEdMX1JHQiwgdywgaCwgMCwgR0xfUkdCLAotICAgICAgICAgICAgICAgICAgICBHTF9VTlNJR05FRF9TSE9SVF81XzZfNSwgcCk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIGJyZWFrOwotICAgIH0KLQotICAgIGdsVGV4UGFyYW1ldGVyaXYoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9DUk9QX1JFQ1RfT0VTLCBjcm9wKTsKLSAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NSU5fRklMVEVSLCBHTF9ORUFSRVNUKTsKLSAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NQUdfRklMVEVSLCBHTF9ORUFSRVNUKTsKLSAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9XUkFQX1MsIEdMX1JFUEVBVCk7Ci0gICAgZ2xUZXhQYXJhbWV0ZXJ4KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfV1JBUF9ULCBHTF9SRVBFQVQpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgQm9vdEFuaW1hdGlvbjo6cmVhZHlUb1J1bigpIHsKLSAgICBtQXNzZXRzLmFkZERlZmF1bHRBc3NldHMoKTsKLQotICAgIERpc3BsYXlJbmZvIGRpbmZvOwotICAgIHN0YXR1c190IHN0YXR1cyA9IHNlc3Npb24oKS0+Z2V0RGlzcGxheUluZm8oMCwgJmRpbmZvKTsKLSAgICBpZiAoc3RhdHVzKQotICAgICAgICByZXR1cm4gLTE7Ci0KLSAgICAvLyBjcmVhdGUgdGhlIG5hdGl2ZSBzdXJmYWNlCi0gICAgc3A8U3VyZmFjZT4gcyA9IHNlc3Npb24oKS0+Y3JlYXRlU3VyZmFjZShnZXRwaWQoKSwgMCwgZGluZm8udywgZGluZm8uaCwKLSAgICAgICAgICAgIFBJWEVMX0ZPUk1BVF9SR0JfNTY1KTsKLSAgICBzZXNzaW9uKCktPm9wZW5UcmFuc2FjdGlvbigpOwotICAgIHMtPnNldExheWVyKDB4NDAwMDAwMDApOwotICAgIHNlc3Npb24oKS0+Y2xvc2VUcmFuc2FjdGlvbigpOwotCi0gICAgLy8gaW5pdGlhbGl6ZSBvcGVuZ2wgYW5kIGVnbAotICAgIGNvbnN0IEVHTGludCBhdHRyaWJzW10gPSB7IEVHTF9SRURfU0laRSwgNSwgRUdMX0dSRUVOX1NJWkUsIDYsCi0gICAgICAgICAgICBFR0xfQkxVRV9TSVpFLCA1LCBFR0xfREVQVEhfU0laRSwgMCwgRUdMX05PTkUgfTsKLSAgICBFR0xpbnQgdywgaCwgZHVtbXk7Ci0gICAgRUdMaW50IG51bUNvbmZpZ3M7Ci0gICAgRUdMQ29uZmlnIGNvbmZpZzsKLSAgICBFR0xTdXJmYWNlIHN1cmZhY2U7Ci0gICAgRUdMQ29udGV4dCBjb250ZXh0OwotICAgIEVHTERpc3BsYXkgZGlzcGxheSA9IGVnbEdldERpc3BsYXkoRUdMX0RFRkFVTFRfRElTUExBWSk7Ci0gICAgZWdsQ2hvb3NlQ29uZmlnKGRpc3BsYXksIGF0dHJpYnMsICZjb25maWcsIDEsICZudW1Db25maWdzKTsKLQotICAgIG1OYXRpdmVXaW5kb3dTdXJmYWNlID0gbmV3IEVHTE5hdGl2ZVdpbmRvd1N1cmZhY2Uocyk7Ci0gICAgc3VyZmFjZSA9IGVnbENyZWF0ZVdpbmRvd1N1cmZhY2UoZGlzcGxheSwgY29uZmlnLCAKLSAgICAgICAgICAgIG1OYXRpdmVXaW5kb3dTdXJmYWNlLmdldCgpLCBOVUxMKTsKLQotICAgIGNvbnRleHQgPSBlZ2xDcmVhdGVDb250ZXh0KGRpc3BsYXksIGNvbmZpZywgTlVMTCwgTlVMTCk7Ci0gICAgZWdsUXVlcnlTdXJmYWNlKGRpc3BsYXksIHN1cmZhY2UsIEVHTF9XSURUSCwgJncpOwotICAgIGVnbFF1ZXJ5U3VyZmFjZShkaXNwbGF5LCBzdXJmYWNlLCBFR0xfSEVJR0hULCAmaCk7Ci0gICAgZWdsTWFrZUN1cnJlbnQoZGlzcGxheSwgc3VyZmFjZSwgc3VyZmFjZSwgY29udGV4dCk7Ci0gICAgbURpc3BsYXkgPSBkaXNwbGF5OwotICAgIG1Db250ZXh0ID0gY29udGV4dDsKLSAgICBtU3VyZmFjZSA9IHN1cmZhY2U7Ci0gICAgbVdpZHRoID0gdzsKLSAgICBtSGVpZ2h0ID0gaDsKLSAgICBtRmxpbmdlclN1cmZhY2UgPSBzOwotCi0gICAgLy8gaW5pdGlhbGl6ZSBHTAotICAgIGdsU2hhZGVNb2RlbChHTF9GTEFUKTsKLSAgICBnbEVuYWJsZShHTF9ESVRIRVIpOwotICAgIGdsRW5hYmxlKEdMX1RFWFRVUkVfMkQpOwotICAgIGdsRW5hYmxlKEdMX1NDSVNTT1JfVEVTVCk7Ci0gICAgZ2xUZXhFbnZ4KEdMX1RFWFRVUkVfRU5WLCBHTF9URVhUVVJFX0VOVl9NT0RFLCBHTF9SRVBMQUNFKTsKLQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotdm9pZCBCb290QW5pbWF0aW9uOjpyZXF1ZXN0RXhpdCgpIHsKLSAgICBtQmFycmllci5vcGVuKCk7Ci0gICAgVGhyZWFkOjpyZXF1ZXN0RXhpdCgpOwotfQotCi1ib29sIEJvb3RBbmltYXRpb246OnRocmVhZExvb3AoKSB7Ci0gICAgYm9vbCByID0gYW5kcm9pZCgpOwotICAgIGVnbE1ha2VDdXJyZW50KG1EaXNwbGF5LCBFR0xfTk9fU1VSRkFDRSwgRUdMX05PX1NVUkZBQ0UsIEVHTF9OT19DT05URVhUKTsKLSAgICBlZ2xEZXN0cm95Q29udGV4dChtRGlzcGxheSwgbUNvbnRleHQpOwotICAgIGVnbERlc3Ryb3lTdXJmYWNlKG1EaXNwbGF5LCBtU3VyZmFjZSk7Ci0gICAgbU5hdGl2ZVdpbmRvd1N1cmZhY2UuY2xlYXIoKTsKLSAgICByZXR1cm4gcjsKLX0KLQotYm9vbCBCb290QW5pbWF0aW9uOjphbmRyb2lkKCkgewotICAgIGluaXRUZXh0dXJlKCZtQW5kcm9pZFswXSwgbUFzc2V0cywgImltYWdlcy9hbmRyb2lkXzMyMHg0ODAucG5nIik7Ci0gICAgaW5pdFRleHR1cmUoJm1BbmRyb2lkWzFdLCBtQXNzZXRzLCAiaW1hZ2VzL2Jvb3Rfcm9ib3QucG5nIik7Ci0gICAgaW5pdFRleHR1cmUoJm1BbmRyb2lkWzJdLCBtQXNzZXRzLCAiaW1hZ2VzL2Jvb3Rfcm9ib3RfZ2xvdy5wbmciKTsKLQotICAgIC8vIGVyYXNlIHNjcmVlbgotICAgIGdsRGlzYWJsZShHTF9TQ0lTU09SX1RFU1QpOwotICAgIGdsQmluZFRleHR1cmUoR0xfVEVYVFVSRV8yRCwgbUFuZHJvaWRbMF0ubmFtZSk7Ci0KLSAgICAvLyBjbGVhciBzY3JlZW4KLSAgICBnbENsZWFyKEdMX0NPTE9SX0JVRkZFUl9CSVQpOwotICAgIGVnbFN3YXBCdWZmZXJzKG1EaXNwbGF5LCBtU3VyZmFjZSk7Ci0KLSAgICAvLyB3YWl0IH4xcwotICAgIHVzbGVlcCg4MDAwMDApOwotCi0gICAgLy8gZmFkZSBpbgotICAgIGdsRW5hYmxlKEdMX0JMRU5EKTsKLSAgICBnbEJsZW5kRnVuYyhHTF9TUkNfQUxQSEEsIEdMX09ORV9NSU5VU19TUkNfQUxQSEEpOwotICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfTU9EVUxBVEUpOwotICAgIGNvbnN0IGludCBzdGVwcyA9IDg7Ci0gICAgZm9yIChpbnQgaSA9IDE7IGkgPCBzdGVwczsgaSsrKSB7Ci0gICAgICAgIGZsb2F0IGZhZGUgPSBpIC8gZmxvYXQoc3RlcHMpOwotICAgICAgICBnbENvbG9yNGYoMSwgMSwgMSwgZmFkZSAqIGZhZGUpOwotICAgICAgICBnbENsZWFyKEdMX0NPTE9SX0JVRkZFUl9CSVQpOwotICAgICAgICBnbERyYXdUZXhpT0VTKDAsIDAsIDAsIG1BbmRyb2lkWzBdLncsIG1BbmRyb2lkWzBdLmgpOwotICAgICAgICBlZ2xTd2FwQnVmZmVycyhtRGlzcGxheSwgbVN1cmZhY2UpOwotICAgIH0KLQotICAgIC8vIGRyYXcgbGFzdCBmcmFtZQotICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfUkVQTEFDRSk7Ci0gICAgZ2xEaXNhYmxlKEdMX0JMRU5EKTsKLSAgICBnbERyYXdUZXhpT0VTKDAsIDAsIDAsIG1BbmRyb2lkWzBdLncsIG1BbmRyb2lkWzBdLmgpOwotICAgIGVnbFN3YXBCdWZmZXJzKG1EaXNwbGF5LCBtU3VyZmFjZSk7Ci0KLSAgICAvLyB1cGRhdGUgcmVjdCBmb3IgdGhlIHJvYm90Ci0gICAgY29uc3QgaW50IHggPSBtV2lkdGggLSBtQW5kcm9pZFsxXS53IC0gMzM7Ci0gICAgY29uc3QgaW50IHkgPSAobUhlaWdodCAtIG1BbmRyb2lkWzFdLmgpIC8gMiAtIDE7Ci0gICAgY29uc3QgUmVjdCB1cGRhdGVSZWN0KHgsIHksIHggKyBtQW5kcm9pZFsxXS53LCB5ICsgbUFuZHJvaWRbMV0uaCk7Ci0KLSAgICAvLyBkcmF3IGFuZCB1cGRhdGUgb25seSB3aGF0IHdlIG5lZWQKLSAgICBtTmF0aXZlV2luZG93U3VyZmFjZS0+c2V0U3dhcFJlY3RhbmdsZSh1cGRhdGVSZWN0LmxlZnQsCi0gICAgICAgICAgICB1cGRhdGVSZWN0LnRvcCwgdXBkYXRlUmVjdC53aWR0aCgpLCB1cGRhdGVSZWN0LmhlaWdodCgpKTsKLQotICAgIGdsRW5hYmxlKEdMX1NDSVNTT1JfVEVTVCk7Ci0gICAgZ2xTY2lzc29yKHVwZGF0ZVJlY3QubGVmdCwgbUhlaWdodCAtIHVwZGF0ZVJlY3QuYm90dG9tLCB1cGRhdGVSZWN0LndpZHRoKCksCi0gICAgICAgICAgICB1cGRhdGVSZWN0LmhlaWdodCgpKTsKLQotICAgIGNvbnN0IG5zZWNzX3Qgc3RhcnRUaW1lID0gc3lzdGVtVGltZSgpOwotICAgIGRvIHsKLSAgICAgICAgLy8gZ2xvdyBzcGVlZCBhbmQgc2hhcGUKLSAgICAgICAgbnNlY3NfdCB0aW1lID0gc3lzdGVtVGltZSgpIC0gc3RhcnRUaW1lOwotICAgICAgICBmbG9hdCB0ID0gKCg0LjBmIC8gKDM2MC4wZiAqIHVzMm5zKDE2NjY3KSkpICogdGltZSk7Ci0gICAgICAgIHQgPSB0IC0gZmxvb3JmKHQpOwotICAgICAgICBjb25zdCBmbG9hdCBmYWRlID0gMC41ZiArIDAuNWYgKiBzaW5mKHQgKiAyICogTV9QSSk7Ci0KLSAgICAgICAgLy8gZmFkZSB0aGUgZ2xvdyBpbiBhbmQgb3V0Ci0gICAgICAgIGdsRGlzYWJsZShHTF9CTEVORCk7Ci0gICAgICAgIGdsQmluZFRleHR1cmUoR0xfVEVYVFVSRV8yRCwgbUFuZHJvaWRbMl0ubmFtZSk7Ci0gICAgICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfTU9EVUxBVEUpOwotICAgICAgICBnbENvbG9yNGYoZmFkZSwgZmFkZSwgZmFkZSwgZmFkZSk7Ci0gICAgICAgIGdsRHJhd1RleGlPRVModXBkYXRlUmVjdC5sZWZ0LCBtSGVpZ2h0IC0gdXBkYXRlUmVjdC5ib3R0b20sIDAsCi0gICAgICAgICAgICAgICAgdXBkYXRlUmVjdC53aWR0aCgpLCB1cGRhdGVSZWN0LmhlaWdodCgpKTsKLQotICAgICAgICAvLyBkcmF3IHRoZSByb2JvdAotICAgICAgICBnbEVuYWJsZShHTF9CTEVORCk7Ci0gICAgICAgIGdsQmluZFRleHR1cmUoR0xfVEVYVFVSRV8yRCwgbUFuZHJvaWRbMV0ubmFtZSk7Ci0gICAgICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfUkVQTEFDRSk7Ci0gICAgICAgIGdsRHJhd1RleGlPRVModXBkYXRlUmVjdC5sZWZ0LCBtSGVpZ2h0IC0gdXBkYXRlUmVjdC5ib3R0b20sIDAsCi0gICAgICAgICAgICAgICAgdXBkYXRlUmVjdC53aWR0aCgpLCB1cGRhdGVSZWN0LmhlaWdodCgpKTsKLQotICAgICAgICAvLyBtYWtlIHN1cmUgc2xlZXAgYSBsb3QgdG8gbm90IHRha2UgdG9vIG11Y2ggQ1BVIGF3YXkgZnJvbSAKLSAgICAgICAgLy8gdGhlIGJvb3QgcHJvY2Vzcy4gV2l0aCB0aGlzICJnbG93IiBhbmltYXRpb24gdGhlcmUgaXMgbm8KLSAgICAgICAgLy8gdmlzaWJsZSBkaWZmZXJlbmNlLiAKLSAgICAgICAgdXNsZWVwKDE2NjY3ICogNCk7Ci0KLSAgICAgICAgZWdsU3dhcEJ1ZmZlcnMobURpc3BsYXksIG1TdXJmYWNlKTsKLSAgICB9IHdoaWxlICghZXhpdFBlbmRpbmcoKSk7Ci0KLSAgICBnbERlbGV0ZVRleHR1cmVzKDEsICZtQW5kcm9pZFswXS5uYW1lKTsKLSAgICBnbERlbGV0ZVRleHR1cmVzKDEsICZtQW5kcm9pZFsxXS5uYW1lKTsKLSAgICBnbERlbGV0ZVRleHR1cmVzKDEsICZtQW5kcm9pZFsyXS5uYW1lKTsKLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0KLWJvb2wgQm9vdEFuaW1hdGlvbjo6Y3lsb24oKSB7Ci0gICAgLy8gaW5pdGlhbGl6ZSB0aGUgdGV4dHVyZXMuLi4KLSAgICBpbml0VGV4dHVyZSgmbUxlZnRUcmFpbCwgbUFzc2V0cywgImltYWdlcy9jeWxvbl9sZWZ0LnBuZyIpOwotICAgIGluaXRUZXh0dXJlKCZtUmlnaHRUcmFpbCwgbUFzc2V0cywgImltYWdlcy9jeWxvbl9yaWdodC5wbmciKTsKLSAgICBpbml0VGV4dHVyZSgmbUJyaWdodFNwb3QsIG1Bc3NldHMsICJpbWFnZXMvY3lsb25fZG90LnBuZyIpOwotCi0gICAgaW50IHcgPSBtV2lkdGg7Ci0gICAgaW50IGggPSBtSGVpZ2h0OwotCi0gICAgY29uc3QgUG9pbnQgYyh3IC8gMiwgaCAvIDIpOwotICAgIGNvbnN0IEdMaW50IGFtcGxpdHVkZSA9IDYwOwotICAgIGNvbnN0IGludCBzY3ggPSBjLnggLSBhbXBsaXR1ZGUgLSBtQnJpZ2h0U3BvdC53IC8gMjsKLSAgICBjb25zdCBpbnQgc2N5ID0gYy55IC0gbUJyaWdodFNwb3QuaCAvIDI7Ci0gICAgY29uc3QgaW50IHNjdyA9IGFtcGxpdHVkZSAqIDIgKyBtQnJpZ2h0U3BvdC53OwotICAgIGNvbnN0IGludCBzY2ggPSBtQnJpZ2h0U3BvdC5oOwotICAgIGNvbnN0IFJlY3QgdXBkYXRlUmVjdChzY3gsIGggLSBzY3kgLSBzY2gsIHNjeCArIHNjdywgaCAtIHNjeSk7Ci0KLSAgICAvLyBlcmFzZSBzY3JlZW4KLSAgICBnbERpc2FibGUoR0xfU0NJU1NPUl9URVNUKTsKLSAgICBnbENsZWFyKEdMX0NPTE9SX0JVRkZFUl9CSVQpOwotCi0gICAgZWdsU3dhcEJ1ZmZlcnMobURpc3BsYXksIG1TdXJmYWNlKTsKLQotICAgIGdsQ2xlYXIoR0xfQ09MT1JfQlVGRkVSX0JJVCk7Ci0KLSAgICBtTmF0aXZlV2luZG93U3VyZmFjZS0+c2V0U3dhcFJlY3RhbmdsZSh1cGRhdGVSZWN0LmxlZnQsCi0gICAgICAgICAgICB1cGRhdGVSZWN0LnRvcCwgdXBkYXRlUmVjdC53aWR0aCgpLCB1cGRhdGVSZWN0LmhlaWdodCgpKTsKLQotICAgIGdsRW5hYmxlKEdMX1NDSVNTT1JfVEVTVCk7Ci0gICAgZ2xFbmFibGUoR0xfQkxFTkQpOwotICAgIGdsQmxlbmRGdW5jKEdMX09ORSwgR0xfT05FX01JTlVTX1NSQ19BTFBIQSk7Ci0KLSAgICAvLyBjbGVhciB0aGUgc2NyZWVuIHRvIHdoaXRlCi0gICAgUG9pbnQgcDsKLSAgICBmbG9hdCB0ID0gMDsKLSAgICBmbG9hdCBhbHBoYSA9IDEuMGY7Ci0gICAgY29uc3QgbnNlY3NfdCBzdGFydFRpbWUgPSBzeXN0ZW1UaW1lKCk7Ci0gICAgbnNlY3NfdCBmYWRlVGltZSA9IDA7Ci0KLSAgICBkbyB7Ci0gICAgICAgIC8vIFNldCBzY2lzc29yIGluIGludGVyZXN0aW5nIGFyZWEKLSAgICAgICAgZ2xTY2lzc29yKHNjeCwgc2N5LCBzY3csIHNjaCk7Ci0KLSAgICAgICAgLy8gZXJhc2Ugc2NyZWVuCi0gICAgICAgIGdsQ2xlYXIoR0xfQ09MT1JfQlVGRkVSX0JJVCk7Ci0KLSAgICAgICAgLy8gY29tcHV0ZSB3YXZlCi0gICAgICAgIGNvbnN0IGZsb2F0IGEgPSAodCAqIDIgKiBNX1BJKSAtIE1fUEkgLyAyOwotICAgICAgICBjb25zdCBmbG9hdCBzbiA9IHNpbmYoYSk7Ci0gICAgICAgIGNvbnN0IGZsb2F0IGNzID0gY29zZihhKTsKLSAgICAgICAgR0xpbnQgeCA9IEdMaW50KGFtcGxpdHVkZSAqIHNuKTsKLSAgICAgICAgZmxvYXQgZGVyaXZhdGl2ZSA9IGNzOwotCi0gICAgICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfTU9EVUxBVEUpOwotCi0gICAgICAgIGlmIChkZXJpdmF0aXZlID4gMCkgewotICAgICAgICAgICAgLy8gdmFuaXNoaW5nIHRyYWlsLi4uCi0gICAgICAgICAgICBwLnggPSAoLWFtcGxpdHVkZSArIGMueCkgLSBtQnJpZ2h0U3BvdC53IC8gMjsKLSAgICAgICAgICAgIHAueSA9IGMueSAtIG1MZWZ0VHJhaWwuaCAvIDI7Ci0gICAgICAgICAgICBmbG9hdCBmYWRlID0gMi4wZiAqICgwLjVmIC0gdCk7Ci0gICAgICAgICAgICAvL2ZhZGUgKj0gZmFkZTsKLSAgICAgICAgICAgIGdsQ29sb3I0ZihmYWRlLCBmYWRlLCBmYWRlLCBmYWRlKTsKLSAgICAgICAgICAgIGdsQmluZFRleHR1cmUoR0xfVEVYVFVSRV8yRCwgbUxlZnRUcmFpbC5uYW1lKTsKLSAgICAgICAgICAgIGdsRHJhd1RleGlPRVMocC54LCBwLnksIDAsIG1MZWZ0VHJhaWwudywgbUxlZnRUcmFpbC5oKTsKLQotICAgICAgICAgICAgLy8gdHJhaWwuLi4KLSAgICAgICAgICAgIHAueCA9ICh4ICsgYy54KSAtIChtUmlnaHRUcmFpbC53ICsgbUJyaWdodFNwb3QudyAvIDIpICsgMTY7Ci0gICAgICAgICAgICBwLnkgPSBjLnkgLSBtUmlnaHRUcmFpbC5oIC8gMjsKLSAgICAgICAgICAgIGZhZGUgPSB0IDwgMC4yNWYgPyB0ICogNC4wZiA6IDEuMGY7Ci0gICAgICAgICAgICBmYWRlICo9IGZhZGU7Ci0gICAgICAgICAgICBnbENvbG9yNGYoZmFkZSwgZmFkZSwgZmFkZSwgZmFkZSk7Ci0gICAgICAgICAgICBnbEJpbmRUZXh0dXJlKEdMX1RFWFRVUkVfMkQsIG1SaWdodFRyYWlsLm5hbWUpOwotICAgICAgICAgICAgZ2xEcmF3VGV4aU9FUyhwLngsIHAueSwgMCwgbVJpZ2h0VHJhaWwudywgbVJpZ2h0VHJhaWwuaCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyB2YW5pc2hpbmcgdHJhaWwuLgotICAgICAgICAgICAgcC54ID0gKGFtcGxpdHVkZSArIGMueCkgLSAobVJpZ2h0VHJhaWwudyArIG1CcmlnaHRTcG90LncgLyAyKSArIDE2OwotICAgICAgICAgICAgcC55ID0gYy55IC0gbVJpZ2h0VHJhaWwuaCAvIDI7Ci0gICAgICAgICAgICBmbG9hdCBmYWRlID0gMi4wZiAqICgwLjVmIC0gKHQgLSAwLjVmKSk7Ci0gICAgICAgICAgICAvL2ZhZGUgKj0gZmFkZTsKLSAgICAgICAgICAgIGdsQ29sb3I0ZihmYWRlLCBmYWRlLCBmYWRlLCBmYWRlKTsKLSAgICAgICAgICAgIGdsQmluZFRleHR1cmUoR0xfVEVYVFVSRV8yRCwgbVJpZ2h0VHJhaWwubmFtZSk7Ci0gICAgICAgICAgICBnbERyYXdUZXhpT0VTKHAueCwgcC55LCAwLCBtUmlnaHRUcmFpbC53LCBtUmlnaHRUcmFpbC5oKTsKLQotICAgICAgICAgICAgLy8gdHJhaWwuLi4KLSAgICAgICAgICAgIHAueCA9ICh4ICsgYy54KSAtIG1CcmlnaHRTcG90LncgLyAyOwotICAgICAgICAgICAgcC55ID0gYy55IC0gbUxlZnRUcmFpbC5oIC8gMjsKLSAgICAgICAgICAgIGZhZGUgPSB0IDwgMC41ZiArIDAuMjVmID8gKHQgLSAwLjVmKSAqIDQuMGYgOiAxLjBmOwotICAgICAgICAgICAgZmFkZSAqPSBmYWRlOwotICAgICAgICAgICAgZ2xDb2xvcjRmKGZhZGUsIGZhZGUsIGZhZGUsIGZhZGUpOwotICAgICAgICAgICAgZ2xCaW5kVGV4dHVyZShHTF9URVhUVVJFXzJELCBtTGVmdFRyYWlsLm5hbWUpOwotICAgICAgICAgICAgZ2xEcmF3VGV4aU9FUyhwLngsIHAueSwgMCwgbUxlZnRUcmFpbC53LCBtTGVmdFRyYWlsLmgpOwotICAgICAgICB9Ci0KLSAgICAgICAgY29uc3QgUG9pbnQgcCh4ICsgYy54IC0gbUJyaWdodFNwb3QudyAvIDIsIGMueSAtIG1CcmlnaHRTcG90LmggLyAyKTsKLSAgICAgICAgZ2xCaW5kVGV4dHVyZShHTF9URVhUVVJFXzJELCBtQnJpZ2h0U3BvdC5uYW1lKTsKLSAgICAgICAgZ2xDb2xvcjRmKDEsIDAuNSwgMC41LCAxKTsKLSAgICAgICAgZ2xEcmF3VGV4aU9FUyhwLngsIHAueSwgMCwgbUJyaWdodFNwb3QudywgbUJyaWdodFNwb3QuaCk7Ci0KLSAgICAgICAgLy8gdXBkYXRlIGFuaW1hdGlvbgotICAgICAgICBuc2Vjc190IHRpbWUgPSBzeXN0ZW1UaW1lKCkgLSBzdGFydFRpbWU7Ci0gICAgICAgIHQgPSAoKDQuMGYgLyAoMzYwLjBmICogdXMybnMoMTY2NjcpKSkgKiB0aW1lKTsKLSAgICAgICAgdCA9IHQgLSBmbG9vcmYodCk7Ci0KLSAgICAgICAgZWdsU3dhcEJ1ZmZlcnMobURpc3BsYXksIG1TdXJmYWNlKTsKLQotICAgICAgICBpZiAoZXhpdFBlbmRpbmcoKSkgewotICAgICAgICAgICAgaWYgKGZhZGVUaW1lID09IDApIHsKLSAgICAgICAgICAgICAgICBmYWRlVGltZSA9IHRpbWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB0aW1lIC09IGZhZGVUaW1lOwotICAgICAgICAgICAgYWxwaGEgPSAxLjBmIC0gKChmbG9hdCh0aW1lKSAqIDYuMGYpIC8gZmxvYXQoczJucygxKSkpOwotCi0gICAgICAgICAgICBzZXNzaW9uKCktPm9wZW5UcmFuc2FjdGlvbigpOwotICAgICAgICAgICAgbUZsaW5nZXJTdXJmYWNlLT5zZXRBbHBoYShhbHBoYSAqIGFscGhhKTsKLSAgICAgICAgICAgIHNlc3Npb24oKS0+Y2xvc2VUcmFuc2FjdGlvbigpOwotICAgICAgICB9Ci0gICAgfSB3aGlsZSAoYWxwaGEgPiAwKTsKLQotICAgIC8vIGNsZWFudXAKLSAgICBnbEZpbmlzaCgpOwotICAgIGdsRGVsZXRlVGV4dHVyZXMoMSwgJm1MZWZ0VHJhaWwubmFtZSk7Ci0gICAgZ2xEZWxldGVUZXh0dXJlcygxLCAmbVJpZ2h0VHJhaWwubmFtZSk7Ci0gICAgZ2xEZWxldGVUZXh0dXJlcygxLCAmbUJyaWdodFNwb3QubmFtZSk7Ci0gICAgcmV0dXJuIGZhbHNlOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfQotOyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9Cb290QW5pbWF0aW9uLmggYi9saWJzL3N1cmZhY2VmbGluZ2VyL0Jvb3RBbmltYXRpb24uaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYjIwY2VhMC4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0Jvb3RBbmltYXRpb24uaAorKysgL2Rldi9udWxsCkBAIC0xLDg3ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfQk9PVEFOSU1BVElPTl9ICi0jZGVmaW5lIEFORFJPSURfQk9PVEFOSU1BVElPTl9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLSNpbmNsdWRlIDx1dGlscy9Bc3NldE1hbmFnZXIuaD4KLQotI2luY2x1ZGUgPHVpL0lTdXJmYWNlQ29tcG9zZXIuaD4KLSNpbmNsdWRlIDx1aS9TdXJmYWNlQ29tcG9zZXJDbGllbnQuaD4KLQotI2luY2x1ZGUgPEVHTC9lZ2wuaD4KLSNpbmNsdWRlIDxHTEVTL2dsLmg+Ci0KLSNpbmNsdWRlICJCYXJyaWVyLmgiCi0KLWNsYXNzIFNrQml0bWFwOwotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIEFzc2V0TWFuYWdlcjsKLWNsYXNzIEVHTE5hdGl2ZVdpbmRvd1N1cmZhY2U7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBCb290QW5pbWF0aW9uIDogcHVibGljIFRocmVhZAotewotcHVibGljOgotICAgICAgICAgICAgICAgIEJvb3RBbmltYXRpb24oY29uc3Qgc3A8SVN1cmZhY2VDb21wb3Nlcj4mIGNvbXBvc2VyKTsKLSAgICB2aXJ0dWFsICAgICB+Qm9vdEFuaW1hdGlvbigpOwotCi0gICAgY29uc3Qgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiYgc2Vzc2lvbigpIGNvbnN0OwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgcmVxdWVzdEV4aXQoKTsKLQotcHJpdmF0ZToKLSAgICB2aXJ0dWFsIGJvb2wgICAgICAgIHRocmVhZExvb3AoKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHJlYWR5VG9SdW4oKTsKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIG9uRmlyc3RSZWYoKTsKLQotICAgIHN0cnVjdCBUZXh0dXJlIHsKLSAgICAgICAgR0xpbnQgICB3OwotICAgICAgICBHTGludCAgIGg7Ci0gICAgICAgIEdMdWludCAgbmFtZTsKLSAgICB9OwotCi0gICAgc3RhdHVzX3QgaW5pdFRleHR1cmUoVGV4dHVyZSogdGV4dHVyZSwgQXNzZXRNYW5hZ2VyJiBhc3NldCwgY29uc3QgY2hhciogbmFtZSk7Ci0gICAgYm9vbCBhbmRyb2lkKCk7Ci0gICAgYm9vbCBjeWxvbigpOwotCi0gICAgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiAgICAgICBtU2Vzc2lvbjsKLSAgICBBc3NldE1hbmFnZXIgbUFzc2V0czsKLSAgICBUZXh0dXJlIG1MZWZ0VHJhaWw7Ci0gICAgVGV4dHVyZSBtUmlnaHRUcmFpbDsKLSAgICBUZXh0dXJlIG1CcmlnaHRTcG90OwotICAgIFRleHR1cmUgbUFuZHJvaWRbM107Ci0gICAgaW50ICAgICBtV2lkdGg7Ci0gICAgaW50ICAgICBtSGVpZ2h0OwotICAgIEVHTERpc3BsYXkgIG1EaXNwbGF5OwotICAgIEVHTERpc3BsYXkgIG1Db250ZXh0OwotICAgIEVHTERpc3BsYXkgIG1TdXJmYWNlOwotICAgIHNwPFN1cmZhY2U+IG1GbGluZ2VyU3VyZmFjZTsKLSAgICBzcDxFR0xOYXRpdmVXaW5kb3dTdXJmYWNlPiBtTmF0aXZlV2luZG93U3VyZmFjZTsKLSAgICBCYXJyaWVyIG1CYXJyaWVyOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0JPT1RBTklNQVRJT05fSApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9DUFVHYXVnZS5jcHAgYi9saWJzL3N1cmZhY2VmbGluZ2VyL0NQVUdhdWdlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzRhOTI3MC4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0NQVUdhdWdlLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDE3MSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJDUFVHYXVnZSIKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPGxpbWl0cy5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPG1hdGguaD4KLQotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLQotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0jaW5jbHVkZSA8dWkvUmVjdC5oPgotI2luY2x1ZGUgPHVpL1JlZ2lvbi5oPgotI2luY2x1ZGUgPHVpL0Rpc3BsYXlJbmZvLmg+Ci0jaW5jbHVkZSA8dWkvSVN1cmZhY2VDb21wb3Nlci5oPgotI2luY2x1ZGUgPHVpL0lTdXJmYWNlRmxpbmdlckNsaWVudC5oPgotCi0jaW5jbHVkZSA8cGl4ZWxmbGluZ2VyL3BpeGVsZmxpbmdlci5oPgotCi0jaW5jbHVkZSAiQ1BVR2F1Z2UuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1DUFVHYXVnZTo6Q1BVR2F1Z2UoIGNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBjb21wb3NlciwKLSAgICAgICAgICAgICAgICAgICAgbnNlY3NfdCBpbnRlcnZhbCwKLSAgICAgICAgICAgICAgICAgICAgaW50IGNsb2NrLAotICAgICAgICAgICAgICAgICAgICBpbnQgcmVmY2xvY2spCi0gICAgOiAgIFRocmVhZChmYWxzZSksIAotICAgICAgICBtSW50ZXJ2YWwoaW50ZXJ2YWwpLCBtQ2xvY2soY2xvY2spLCBtUmVmQ2xvY2socmVmY2xvY2spLAotICAgICAgICBtUmVmZXJlbmNlVGltZSgwKSwKLSAgICAgICAgbVJlZmVyZW5jZVdvcmtpbmdUaW1lKDApLCBtQ3B1VXNhZ2UoMCksCi0gICAgICAgIG1SZWZJZGxlVGltZSgwKSwgbUlkbGVUaW1lKDApCi17Ci0gICAgbUZkID0gZm9wZW4oIi9wcm9jL3N0YXQiLCAiciIpOwotICAgIHNldHZidWYobUZkLCBOVUxMLCBfSU9OQkYsIDApOwotCi0gICAgbVNlc3Npb24gPSBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OmNsaWVudEZvckNvbm5lY3Rpb24oCi0gICAgICAgIGNvbXBvc2VyLT5jcmVhdGVDb25uZWN0aW9uKCktPmFzQmluZGVyKCkpOwotfQotCi1DUFVHYXVnZTo6fkNQVUdhdWdlKCkKLXsKLSAgICBmY2xvc2UobUZkKTsKLX0KLQotY29uc3Qgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiYgQ1BVR2F1Z2U6OnNlc3Npb24oKSBjb25zdCAKLXsKLSAgICByZXR1cm4gbVNlc3Npb247Ci19Ci0KLXZvaWQgQ1BVR2F1Z2U6Om9uRmlyc3RSZWYoKQotewotICAgIHJ1bigiQ1BVIEdhdWdlIik7Ci19Ci0KLXN0YXR1c190IENQVUdhdWdlOjpyZWFkeVRvUnVuKCkKLXsKLSAgICBMT0dJKCJTdGFydGluZyBDUFUgZ2F1Z2UuLi4iKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLWJvb2wgQ1BVR2F1Z2U6OnRocmVhZExvb3AoKQotewotICAgIERpc3BsYXlJbmZvIGRpbmZvOwotICAgIHNlc3Npb24oKS0+Z2V0RGlzcGxheUluZm8oMCwgJmRpbmZvKTsKLSAgICBzcDxTdXJmYWNlPiBzKHNlc3Npb24oKS0+Y3JlYXRlU3VyZmFjZShnZXRwaWQoKSwgMCwgZGluZm8udywgNCwgUElYRUxfRk9STUFUX09QQVFVRSkpOwotICAgIHNlc3Npb24oKS0+b3BlblRyYW5zYWN0aW9uKCk7Ci0gICAgcy0+c2V0TGF5ZXIoSU5UX01BWCk7Ci0gICAgc2Vzc2lvbigpLT5jbG9zZVRyYW5zYWN0aW9uKCk7Ci0gICAgCi0gICAgc3RhdGljIGNvbnN0IEdHTGZpeGVkIGNvbG9yc1s0XVs0XSA9IHsKLSAgICAgICAgICAgIHsgMHgwMDAwMCwgMHgxMDAwMCwgMHgwMDAwMCwgMHgxMDAwMCB9LAotICAgICAgICAgICAgeyAweDEwMDAwLCAweDEwMDAwLCAweDAwMDAwLCAweDEwMDAwIH0sCi0gICAgICAgICAgICB7IDB4MTAwMDAsIDB4MDAwMDAsIDB4MDAwMDAsIDB4MTAwMDAgfSwKLSAgICAgICAgICAgIHsgMHgwMDAwMCwgMHgwMDAwMCwgMHgwMDAwMCwgMHgxMDAwMCB9LAotICAgICAgICB9OwotCi0gICAgR0dMQ29udGV4dCogZ2w7Ci0gICAgZ2dsSW5pdCgmZ2wpOwotICAgIGdsLT5hY3RpdmVUZXh0dXJlKGdsLCAwKTsKLSAgICBnbC0+ZGlzYWJsZShnbCwgR0dMX1RFWFRVUkVfMkQpOwotICAgIGdsLT5kaXNhYmxlKGdsLCBHR0xfQkxFTkQpOwotCi0gICAgY29uc3QgaW50IHcgPSBkaW5mby53OwotCi0gICAgd2hpbGUoIWV4aXRQZW5kaW5nKCkpCi0gICAgewotICAgICAgICBtTG9jay5sb2NrKCk7Ci0gICAgICAgICAgICBjb25zdCBmbG9hdCBjcHVVc2FnZSA9IHRoaXMtPmNwdVVzYWdlKCk7Ci0gICAgICAgICAgICBjb25zdCBmbG9hdCB0b3RhbENwdVVzYWdlID0gMS4wZiAtIGlkbGUoKTsKLSAgICAgICAgbUxvY2sudW5sb2NrKCk7Ci0KLSAgICAgICAgU3VyZmFjZTo6U3VyZmFjZUluZm8gaW5mbzsKLSAgICAgICAgcy0+bG9jaygmaW5mbyk7Ci0gICAgICAgICAgICBHR0xTdXJmYWNlIGZiOwotICAgICAgICAgICAgICAgIGZiLnZlcnNpb24gPSBzaXplb2YoR0dMU3VyZmFjZSk7Ci0gICAgICAgICAgICAgICAgZmIud2lkdGggICA9IGluZm8udzsKLSAgICAgICAgICAgICAgICBmYi5oZWlnaHQgID0gaW5mby5oOwotICAgICAgICAgICAgICAgIGZiLnN0cmlkZSAgPSBpbmZvLnc7Ci0gICAgICAgICAgICAgICAgZmIuZm9ybWF0ICA9IGluZm8uZm9ybWF0OwotICAgICAgICAgICAgICAgIGZiLmRhdGEgPSAoR0dMdWJ5dGUqKWluZm8uYml0czsKLQotICAgICAgICAgICAgZ2wtPmNvbG9yQnVmZmVyKGdsLCAmZmIpOwotICAgICAgICAgICAgZ2wtPmNvbG9yNHh2KGdsLCBjb2xvcnNbM10pOwotICAgICAgICAgICAgZ2wtPnJlY3RpKGdsLCAwLCAwLCB3LCA0KTsKLSAgICAgICAgICAgIGdsLT5jb2xvcjR4dihnbCwgY29sb3JzWzJdKTsgLy8gcmVkCi0gICAgICAgICAgICBnbC0+cmVjdGkoZ2wsIDAsIDAsIGludCh0b3RhbENwdVVzYWdlKncpLCAyKTsKLSAgICAgICAgICAgIGdsLT5jb2xvcjR4dihnbCwgY29sb3JzWzBdKTsgLy8gZ3JlZW4KLSAgICAgICAgICAgIGdsLT5yZWN0aShnbCwgMCwgMiwgaW50KGNwdVVzYWdlKncpLCA0KTsKLSAgICAgICAgCi0gICAgICAgIHMtPnVubG9ja0FuZFBvc3QoKTsgCi0KLSAgICAgICAgdXNsZWVwKG5zMnVzKG1JbnRlcnZhbCkpOwotICAgIH0KLQotICAgIGdnbFVuaW5pdChnbCk7Ci0gICAgcmV0dXJuIGZhbHNlOwotfQotCi12b2lkIENQVUdhdWdlOjpzYW1wbGUoKQotewotICAgIGlmIChtTG9jay50cnlMb2NrKCkgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgY29uc3QgbnNlY3NfdCBub3cgPSBzeXN0ZW1UaW1lKG1SZWZDbG9jayk7Ci0gICAgICAgIGNvbnN0IG5zZWNzX3QgcmVmZXJlbmNlVGltZSA9IG5vdy1tUmVmZXJlbmNlVGltZTsKLSAgICAgICAgaWYgKHJlZmVyZW5jZVRpbWUgPj0gbUludGVydmFsKSB7Ci0gICAgICAgICAgICBjb25zdCBmbG9hdCByZWZ0aW1lID0gMS4wZiAvIHJlZmVyZW5jZVRpbWU7Ci0gICAgICAgICAgICBjb25zdCBuc2Vjc190IG5vd1dvcmtpbmdUaW1lID0gc3lzdGVtVGltZShtQ2xvY2spOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBjaGFyIGJ1ZlsyNTZdOwotICAgICAgICAgICAgZmdldHMoYnVmLCAyNTYsIG1GZCk7Ci0gICAgICAgICAgICByZXdpbmQobUZkKTsKLSAgICAgICAgICAgIGNoYXIgKnN0ciA9IGJ1Zis1OwotICAgICAgICAgICAgY2hhciBjb25zdCAqIGNvbnN0IHVzZXJtb2RlID0gc3Ryc2VwKCZzdHIsICIgIik7ICAodm9pZCl1c2VybW9kZTsKLSAgICAgICAgICAgIGNoYXIgY29uc3QgKiBjb25zdCB1c2VybmljZSA9IHN0cnNlcCgmc3RyLCAiICIpOyAgKHZvaWQpdXNlcm5pY2U7Ci0gICAgICAgICAgICBjaGFyIGNvbnN0ICogY29uc3Qgc3lzdGVtbW9kZSA9IHN0cnNlcCgmc3RyLCAiICIpOyh2b2lkKXN5c3RlbW1vZGU7Ci0gICAgICAgICAgICBjaGFyIGNvbnN0ICogY29uc3QgaWRsZSA9IHN0cnNlcCgmc3RyLCAiICIpOwotICAgICAgICAgICAgY29uc3QgbnNlY3NfdCBub3dJZGxlVGltZSA9IGF0b2koaWRsZSkgKiAxMDAwMDAwMExMOwotICAgICAgICAgICAgbUlkbGVUaW1lID0gZmxvYXQobm93SWRsZVRpbWUgLSBtUmVmSWRsZVRpbWUpICogcmVmdGltZTsKLSAgICAgICAgICAgIG1SZWZJZGxlVGltZSA9IG5vd0lkbGVUaW1lOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBjb25zdCBuc2Vjc190IHdvcmtpbmdUaW1lID0gbm93V29ya2luZ1RpbWUgLSBtUmVmZXJlbmNlV29ya2luZ1RpbWU7Ci0gICAgICAgICAgICBjb25zdCBmbG9hdCBuZXdDcHVVc2FnZSA9IGZsb2F0KHdvcmtpbmdUaW1lKSAqIHJlZnRpbWU7Ci0gICAgICAgICAgICBpZiAobUNwdVVzYWdlICE9IG5ld0NwdVVzYWdlKSB7ICAgICAgICAKLSAgICAgICAgICAgICAgICBtQ3B1VXNhZ2UgPSBuZXdDcHVVc2FnZTsKLSAgICAgICAgICAgICAgICBtUmVmZXJlbmNlV29ya2luZ1RpbWUgPSBub3dXb3JraW5nVGltZTsKLSAgICAgICAgICAgICAgICBtUmVmZXJlbmNlVGltZSA9IG5vdzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBtTG9jay51bmxvY2soKTsKLSAgICB9Ci19Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvQ1BVR2F1Z2UuaCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvQ1BVR2F1Z2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNWJiNTNjMC4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0NQVUdhdWdlLmgKKysrIC9kZXYvbnVsbApAQCAtMSw3NCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0NQVUdBVUdFX0gKLSNkZWZpbmUgQU5EUk9JRF9DUFVHQVVHRV9ICi0KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0cmluZy5oPgotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9UaW1lcnMuaD4KLQotI2luY2x1ZGUgPHVpL1N1cmZhY2VDb21wb3NlckNsaWVudC5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIENQVUdhdWdlIDogcHVibGljIFRocmVhZAotewotcHVibGljOgotICAgIENQVUdhdWdlKCAgIGNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBjb21wb3NlciwKLSAgICAgICAgICAgICAgICBuc2Vjc190IGludGVydmFsPXMybnMoMSksCi0gICAgICAgICAgICAgICAgaW50IGNsb2NrPVNZU1RFTV9USU1FX1RIUkVBRCwKLSAgICAgICAgICAgICAgICBpbnQgcmVmY2xvY2s9U1lTVEVNX1RJTUVfTU9OT1RPTklDKTsKLSAgICAgICAgICAgICAgICAKLSAgICB+Q1BVR2F1Z2UoKTsKLQotICAgIGNvbnN0IHNwPFN1cmZhY2VDb21wb3NlckNsaWVudD4mIHNlc3Npb24oKSBjb25zdDsKLQotICAgIHZvaWQgc2FtcGxlKCk7Ci0gCi0gICAgaW5saW5lIGZsb2F0IGNwdVVzYWdlKCkgY29uc3QgeyByZXR1cm4gbUNwdVVzYWdlOyB9Ci0gICAgaW5saW5lIGZsb2F0IGlkbGUoKSBjb25zdCB7IHJldHVybiBtSWRsZVRpbWU7IH0KLQotcHJpdmF0ZToKLSAgICB2aXJ0dWFsIHZvaWQgICAgICAgIG9uRmlyc3RSZWYoKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgIHJlYWR5VG9SdW4oKTsKLSAgICB2aXJ0dWFsIGJvb2wgICAgICAgIHRocmVhZExvb3AoKTsKLQotICAgIE11dGV4IG1Mb2NrOwotCi0gICAgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiBtU2Vzc2lvbjsKLQotICAgIGNvbnN0IG5zZWNzX3QgbUludGVydmFsOwotICAgIGNvbnN0IGludCBtQ2xvY2s7Ci0gICAgY29uc3QgaW50IG1SZWZDbG9jazsKLQotICAgIG5zZWNzX3QgbVJlZmVyZW5jZVRpbWU7Ci0gICAgbnNlY3NfdCBtUmVmZXJlbmNlV29ya2luZ1RpbWU7Ci0gICAgZmxvYXQgbUNwdVVzYWdlOwotICAgIG5zZWNzX3QgbVJlZklkbGVUaW1lOwotICAgIGZsb2F0IG1JZGxlVGltZTsKLSAgICBGSUxFKiAgIG1GZDsKLX07Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfQ1BVR0FVR0VfSApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9EaXNwbGF5SGFyZHdhcmUvRGlzcGxheUhhcmR3YXJlLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGYxNGQ3ZTkuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9EaXNwbGF5SGFyZHdhcmUvRGlzcGxheUhhcmR3YXJlLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDM1MyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJTdXJmYWNlRmxpbmdlciIKLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8bWF0aC5oPgotCi0jaW5jbHVkZSA8Y3V0aWxzL3Byb3BlcnRpZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8dWkvRUdMRGlzcGxheVN1cmZhY2UuaD4KLQotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLSNpbmNsdWRlIDxFR0wvZWdsZXh0Lmg+Ci0KLQotI2luY2x1ZGUgIkRpc3BsYXlIYXJkd2FyZS9EaXNwbGF5SGFyZHdhcmUuaCIKLQotI2luY2x1ZGUgPGhhcmR3YXJlL2NvcHliaXQuaD4KLSNpbmNsdWRlIDxoYXJkd2FyZS9vdmVybGF5Lmg+Ci0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi1zdGF0aWMgX19hdHRyaWJ1dGVfXygobm9pbmxpbmUpKQotY29uc3QgY2hhciAqZWdsX3N0cmVycm9yKEVHTGludCBlcnIpCi17Ci0gICAgc3dpdGNoIChlcnIpewotICAgICAgICBjYXNlIEVHTF9TVUNDRVNTOiAgICAgICAgICAgcmV0dXJuICJFR0xfU1VDQ0VTUyI7Ci0gICAgICAgIGNhc2UgRUdMX05PVF9JTklUSUFMSVpFRDogICByZXR1cm4gIkVHTF9OT1RfSU5JVElBTElaRUQiOwotICAgICAgICBjYXNlIEVHTF9CQURfQUNDRVNTOiAgICAgICAgcmV0dXJuICJFR0xfQkFEX0FDQ0VTUyI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9BTExPQzogICAgICAgICByZXR1cm4gIkVHTF9CQURfQUxMT0MiOwotICAgICAgICBjYXNlIEVHTF9CQURfQVRUUklCVVRFOiAgICAgcmV0dXJuICJFR0xfQkFEX0FUVFJJQlVURSI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9DT05GSUc6ICAgICAgICByZXR1cm4gIkVHTF9CQURfQ09ORklHIjsKLSAgICAgICAgY2FzZSBFR0xfQkFEX0NPTlRFWFQ6ICAgICAgIHJldHVybiAiRUdMX0JBRF9DT05URVhUIjsKLSAgICAgICAgY2FzZSBFR0xfQkFEX0NVUlJFTlRfU1VSRkFDRTogcmV0dXJuICJFR0xfQkFEX0NVUlJFTlRfU1VSRkFDRSI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9ESVNQTEFZOiAgICAgICByZXR1cm4gIkVHTF9CQURfRElTUExBWSI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9NQVRDSDogICAgICAgICByZXR1cm4gIkVHTF9CQURfTUFUQ0giOwotICAgICAgICBjYXNlIEVHTF9CQURfTkFUSVZFX1BJWE1BUDogcmV0dXJuICJFR0xfQkFEX05BVElWRV9QSVhNQVAiOwotICAgICAgICBjYXNlIEVHTF9CQURfTkFUSVZFX1dJTkRPVzogcmV0dXJuICJFR0xfQkFEX05BVElWRV9XSU5ET1ciOwotICAgICAgICBjYXNlIEVHTF9CQURfUEFSQU1FVEVSOiAgICAgcmV0dXJuICJFR0xfQkFEX1BBUkFNRVRFUiI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9TVVJGQUNFOiAgICAgICByZXR1cm4gIkVHTF9CQURfU1VSRkFDRSI7Ci0gICAgICAgIGNhc2UgRUdMX0NPTlRFWFRfTE9TVDogICAgICByZXR1cm4gIkVHTF9DT05URVhUX0xPU1QiOwotICAgICAgICBkZWZhdWx0OiByZXR1cm4gIlVOS05PV04iOwotICAgIH0KLX0KLQotc3RhdGljIF9fYXR0cmlidXRlX18oKG5vaW5saW5lKSkKLXZvaWQgY2hlY2tHTEVycm9ycygpCi17Ci0gICAgR0xlbnVtIGVycm9yID0gZ2xHZXRFcnJvcigpOwotICAgIGlmIChlcnJvciAhPSBHTF9OT19FUlJPUikKLSAgICAgICAgTE9HRSgiR0wgZXJyb3IgMHglMDR4IiwgaW50KGVycm9yKSk7Ci19Ci0KLXN0YXRpYyBfX2F0dHJpYnV0ZV9fKChub2lubGluZSkpCi12b2lkIGNoZWNrRUdMRXJyb3JzKGNvbnN0IGNoYXIqIHRva2VuKQotewotICAgIEVHTGludCBlcnJvciA9IGVnbEdldEVycm9yKCk7Ci0gICAgLy8gR0xFU29uR0wgc2VlbXMgdG8gYmUgcmV0dXJuaW5nIDAgd2hlbiB0aGVyZSBpcyBubyBlcnJvcnM/Ci0gICAgaWYgKGVycm9yICYmIGVycm9yICE9IEVHTF9TVUNDRVNTKQotICAgICAgICBMT0dFKCIlcyBlcnJvciAweCUwNHggKCVzKSIsCi0gICAgICAgICAgICAgICAgdG9rZW4sIGludChlcnJvciksIGVnbF9zdHJlcnJvcihlcnJvcikpOwotfQotCi0KLS8qCi0gKiBJbml0aWFsaXplIHRoZSBkaXNwbGF5IHRvIHRoZSBzcGVjaWZpZWQgdmFsdWVzLgotICoKLSAqLwotCi1EaXNwbGF5SGFyZHdhcmU6OkRpc3BsYXlIYXJkd2FyZSgKLSAgICAgICAgY29uc3Qgc3A8U3VyZmFjZUZsaW5nZXI+JiBmbGluZ2VyLAotICAgICAgICB1aW50MzJfdCBkcHkpCi0gICAgOiBEaXNwbGF5SGFyZHdhcmVCYXNlKGZsaW5nZXIsIGRweSkKLXsKLSAgICBpbml0KGRweSk7Ci19Ci0KLURpc3BsYXlIYXJkd2FyZTo6fkRpc3BsYXlIYXJkd2FyZSgpCi17Ci0gICAgZmluaSgpOwotfQotCi1mbG9hdCBEaXNwbGF5SGFyZHdhcmU6OmdldERwaVgoKSBjb25zdCAgICAgICAgICB7IHJldHVybiBtRHBpWDsgfQotZmxvYXQgRGlzcGxheUhhcmR3YXJlOjpnZXREcGlZKCkgY29uc3QgICAgICAgICAgeyByZXR1cm4gbURwaVk7IH0KLWZsb2F0IERpc3BsYXlIYXJkd2FyZTo6Z2V0RGVuc2l0eSgpIGNvbnN0ICAgICAgIHsgcmV0dXJuIG1EZW5zaXR5OyB9Ci1mbG9hdCBEaXNwbGF5SGFyZHdhcmU6OmdldFJlZnJlc2hSYXRlKCkgY29uc3QgICB7IHJldHVybiBtUmVmcmVzaFJhdGU7IH0KLWludCBEaXNwbGF5SGFyZHdhcmU6OmdldFdpZHRoKCkgY29uc3QgICAgICAgICAgIHsgcmV0dXJuIG1XaWR0aDsgfQotaW50IERpc3BsYXlIYXJkd2FyZTo6Z2V0SGVpZ2h0KCkgY29uc3QgICAgICAgICAgeyByZXR1cm4gbUhlaWdodDsgfQotUGl4ZWxGb3JtYXQgRGlzcGxheUhhcmR3YXJlOjpnZXRGb3JtYXQoKSBjb25zdCAgeyByZXR1cm4gbUZvcm1hdDsgfQotCi12b2lkIERpc3BsYXlIYXJkd2FyZTo6aW5pdCh1aW50MzJfdCBkcHkpCi17Ci0gICAgLy8gaW5pdGlhbGl6ZSBFR0wKLSAgICBjb25zdCBFR0xpbnQgYXR0cmlic1tdID0gewotICAgICAgICAgICAgRUdMX1JFRF9TSVpFLCAgICAgICA1LAotICAgICAgICAgICAgRUdMX0dSRUVOX1NJWkUsICAgICA2LAotICAgICAgICAgICAgRUdMX0JMVUVfU0laRSwgICAgICA1LAotICAgICAgICAgICAgRUdMX0RFUFRIX1NJWkUsICAgICAwLAotICAgICAgICAgICAgRUdMX05PTkUKLSAgICB9OwotICAgIEVHTGludCB3LCBoLCBkdW1teTsKLSAgICBFR0xpbnQgbnVtQ29uZmlncywgbjsKLSAgICBFR0xDb25maWcgY29uZmlnOwotICAgIEVHTFN1cmZhY2Ugc3VyZmFjZTsKLSAgICBFR0xDb250ZXh0IGNvbnRleHQ7Ci0gICAgbUZsYWdzID0gMDsKLQotICAgIC8vIFRPRE86IGFsbCB0aGUgZXh0ZW5zaW9ucyBiZWxvdyBzaG91bGQgYmUgcXVlcmllZCB0aHJvdWdoCi0gICAgLy8gZWdsR2V0UHJvY0FkZHJlc3MoKS4KLQotICAgIEVHTERpc3BsYXkgZGlzcGxheSA9IGVnbEdldERpc3BsYXkoRUdMX0RFRkFVTFRfRElTUExBWSk7Ci0gICAgZWdsSW5pdGlhbGl6ZShkaXNwbGF5LCBOVUxMLCBOVUxMKTsKLSAgICBlZ2xHZXRDb25maWdzKGRpc3BsYXksIE5VTEwsIDAsICZudW1Db25maWdzKTsKLSAgICBlZ2xDaG9vc2VDb25maWcoZGlzcGxheSwgYXR0cmlicywgJmNvbmZpZywgMSwgJm4pOwotCi0gICAgLyoKLSAgICAgKiBHYXRoZXIgRUdMIGV4dGVuc2lvbnMKLSAgICAgKi8KLQotICAgIGNvbnN0IGNoYXIqIGNvbnN0IGVnbF9leHRlbnNpb25zID0gZWdsUXVlcnlTdHJpbmcoCi0gICAgICAgICAgICBkaXNwbGF5LCBFR0xfRVhURU5TSU9OUyk7Ci0gICAgCi0gICAgTE9HSSgiRUdMIGluZm9ybWF0aW9uczoiKTsKLSAgICBMT0dJKCIjIG9mIGNvbmZpZ3MgOiAlZCIsIG51bUNvbmZpZ3MpOwotICAgIExPR0koInZlbmRvciAgICA6ICVzIiwgZWdsUXVlcnlTdHJpbmcoZGlzcGxheSwgRUdMX1ZFTkRPUikpOwotICAgIExPR0koInZlcnNpb24gICA6ICVzIiwgZWdsUXVlcnlTdHJpbmcoZGlzcGxheSwgRUdMX1ZFUlNJT04pKTsKLSAgICBMT0dJKCJleHRlbnNpb25zOiAlcyIsIGVnbF9leHRlbnNpb25zKTsKLSAgICBMT0dJKCJDbGllbnQgQVBJOiAlcyIsIGVnbFF1ZXJ5U3RyaW5nKGRpc3BsYXksIEVHTF9DTElFTlRfQVBJUyk/OiJOb3QgU3VwcG9ydGVkIik7Ci0KLSAgICAvLyBUT0RPOiBnZXQgdGhpcyBmcm9tIHRoZSBkZXZmYiBkcml2ZXIgKHByb2JhYmx5IHNob3VsZCBiZSBIQUwgbW9kdWxlKQotICAgIG1GbGFncyB8PSBTV0FQX1JFQ1RBTkdMRV9FWFRFTlNJT047Ci0gICAgCi0gICAgLy8gVE9ETzogZ2V0IHRoZSByZWFsICJ1cGRhdGVfb25fZGVtYW5kIiBiZWhhdmlvciAocHJvYmFibHkgc2hvdWxkIGJlIEhBTCBtb2R1bGUpCi0gICAgbUZsYWdzIHw9IFVQREFURV9PTl9ERU1BTkQ7Ci0KLSAgICBpZiAoZWdsR2V0Q29uZmlnQXR0cmliKGRpc3BsYXksIGNvbmZpZywgRUdMX0NPTkZJR19DQVZFQVQsICZkdW1teSkgPT0gRUdMX1RSVUUpIHsKLSAgICAgICAgaWYgKGR1bW15ID09IEVHTF9TTE9XX0NPTkZJRykKLSAgICAgICAgICAgIG1GbGFncyB8PSBTTE9XX0NPTkZJRzsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIENyZWF0ZSBvdXIgbWFpbiBzdXJmYWNlCi0gICAgICovCi0KLSAgICBtRGlzcGxheVN1cmZhY2UgPSBuZXcgRUdMRGlzcGxheVN1cmZhY2UoKTsKLQotICAgIHN1cmZhY2UgPSBlZ2xDcmVhdGVXaW5kb3dTdXJmYWNlKGRpc3BsYXksIGNvbmZpZywgbURpc3BsYXlTdXJmYWNlLmdldCgpLCBOVUxMKTsKLSAgICAvL2NoZWNrRUdMRXJyb3JzKCJlZ2xDcmVhdGVEaXNwbGF5U3VyZmFjZUFORFJPSUQiKTsKLQotICAgIGlmIChlZ2xRdWVyeVN1cmZhY2UoZGlzcGxheSwgc3VyZmFjZSwgRUdMX1NXQVBfQkVIQVZJT1IsICZkdW1teSkgPT0gRUdMX1RSVUUpIHsKLSAgICAgICAgaWYgKGR1bW15ID09IEVHTF9CVUZGRVJfUFJFU0VSVkVEKSB7Ci0gICAgICAgICAgICBtRmxhZ3MgfD0gQlVGRkVSX1BSRVNFUlZFRDsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBHTGludCB2YWx1ZSA9IEVHTF9VTktOT1dOOwotICAgIGVnbFF1ZXJ5U3VyZmFjZShkaXNwbGF5LCBzdXJmYWNlLCBFR0xfSE9SSVpPTlRBTF9SRVNPTFVUSU9OLCAmdmFsdWUpOwotICAgIGlmICh2YWx1ZSA9PSBFR0xfVU5LTk9XTikgewotICAgICAgICBtRHBpWCA9IDE2MC4wZjsKLSAgICB9IGVsc2UgewotICAgICAgICBtRHBpWCA9IDI1LjRmICogZmxvYXQodmFsdWUpL0VHTF9ESVNQTEFZX1NDQUxJTkc7Ci0gICAgfQotICAgIHZhbHVlID0gRUdMX1VOS05PV047Ci0gICAgZWdsUXVlcnlTdXJmYWNlKGRpc3BsYXksIHN1cmZhY2UsIEVHTF9WRVJUSUNBTF9SRVNPTFVUSU9OLCAmdmFsdWUpOwotICAgIGlmICh2YWx1ZSA9PSBFR0xfVU5LTk9XTikgewotICAgICAgICBtRHBpWSA9IDE2MC4wZjsKLSAgICB9IGVsc2UgewotICAgICAgICBtRHBpWSA9IDI1LjRmICogZmxvYXQodmFsdWUpL0VHTF9ESVNQTEFZX1NDQUxJTkc7Ci0gICAgfQotICAgIG1SZWZyZXNoUmF0ZSA9IDYwLmY7ICAgIC8vIFRPRE86IGdldCB0aGUgcmVhbCByZWZyZXNoIHJhdGUgCi0gICAgCi0gICAgCi0gICAgY2hhciBwcm9wZXJ0eVtQUk9QRVJUWV9WQUxVRV9NQVhdOwotICAgIGlmIChwcm9wZXJ0eV9nZXQoInJvLnNmLmxjZF9kZW5zaXR5IiwgcHJvcGVydHksIE5VTEwpIDw9IDApIHsKLSAgICAgICAgTE9HVygicm8uc2YubGNkX2RlbnNpdHkgbm90IGRlZmluZWQsIHVzaW5nIDE2MCBkcGkgYnkgZGVmYXVsdC4iKTsKLSAgICAgICAgc3RyY3B5KHByb3BlcnR5LCAiMTYwIik7Ci0gICAgfQotICAgIG1EZW5zaXR5ID0gYXRvaShwcm9wZXJ0eSkgKiAoMS4wZi8xNjAuMGYpOwotCi0KLSAgICAvKgotICAgICAqIENyZWF0ZSBvdXIgT3BlbkdMIEVTIGNvbnRleHQKLSAgICAgKi8KLSAgICAKLSAgICBjb250ZXh0ID0gZWdsQ3JlYXRlQ29udGV4dChkaXNwbGF5LCBjb25maWcsIE5VTEwsIE5VTEwpOwotICAgIC8vY2hlY2tFR0xFcnJvcnMoImVnbENyZWF0ZUNvbnRleHQiKTsKLSAgICAKLSAgICBlZ2xRdWVyeVN1cmZhY2UoZGlzcGxheSwgc3VyZmFjZSwgRUdMX1dJRFRILCAmbVdpZHRoKTsKLSAgICBlZ2xRdWVyeVN1cmZhY2UoZGlzcGxheSwgc3VyZmFjZSwgRUdMX0hFSUdIVCwgJm1IZWlnaHQpOwotICAgIAotICAgIAotICAgIC8qCi0gICAgICogR2F0aGVyIE9wZW5HTCBFUyBleHRlbnNpb25zCi0gICAgICovCi0KLSAgICBlZ2xNYWtlQ3VycmVudChkaXNwbGF5LCBzdXJmYWNlLCBzdXJmYWNlLCBjb250ZXh0KTsKLSAgICBjb25zdCBjaGFyKiBjb25zdCAgZ2xfZXh0ZW5zaW9ucyA9IChjb25zdCBjaGFyKilnbEdldFN0cmluZyhHTF9FWFRFTlNJT05TKTsKLSAgICBMT0dJKCJPcGVuR0wgaW5mb3JtYXRpb25zOiIpOwotICAgIExPR0koInZlbmRvciAgICA6ICVzIiwgZ2xHZXRTdHJpbmcoR0xfVkVORE9SKSk7Ci0gICAgTE9HSSgicmVuZGVyZXIgIDogJXMiLCBnbEdldFN0cmluZyhHTF9SRU5ERVJFUikpOwotICAgIExPR0koInZlcnNpb24gICA6ICVzIiwgZ2xHZXRTdHJpbmcoR0xfVkVSU0lPTikpOwotICAgIExPR0koImV4dGVuc2lvbnM6ICVzIiwgZ2xfZXh0ZW5zaW9ucyk7Ci0KLSAgICBpZiAoc3Ryc3RyKGdsX2V4dGVuc2lvbnMsICJHTF9BUkJfdGV4dHVyZV9ub25fcG93ZXJfb2ZfdHdvIikpIHsKLSAgICAgICAgbUZsYWdzIHw9IE5QT1RfRVhURU5TSU9OOwotICAgIH0KLSAgICBpZiAoc3Ryc3RyKGdsX2V4dGVuc2lvbnMsICJHTF9PRVNfZHJhd190ZXh0dXJlIikpIHsKLSAgICAgICAgbUZsYWdzIHw9IERSQVdfVEVYVFVSRV9FWFRFTlNJT047Ci0gICAgfQotICAgIGlmIChzdHJzdHIoZ2xfZXh0ZW5zaW9ucywgIkdMX0FORFJPSURfZGlyZWN0X3RleHR1cmUiKSkgewotICAgICAgICBtRmxhZ3MgfD0gRElSRUNUX1RFWFRVUkU7Ci0gICAgfQotCi0gICAgLy8gVW5iaW5kIHRoZSBjb250ZXh0IGZyb20gdGhpcyB0aHJlYWQKLSAgICBlZ2xNYWtlQ3VycmVudChkaXNwbGF5LCBFR0xfTk9fU1VSRkFDRSwgRUdMX05PX1NVUkZBQ0UsIEVHTF9OT19DT05URVhUKTsKLQotICAgIG1EaXNwbGF5ID0gZGlzcGxheTsKLSAgICBtQ29uZmlnICA9IGNvbmZpZzsKLSAgICBtU3VyZmFjZSA9IHN1cmZhY2U7Ci0gICAgbUNvbnRleHQgPSBjb250ZXh0OwotICAgIG1Gb3JtYXQgID0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JfNTY1OwotICAgIAotICAgIGh3X21vZHVsZV90IGNvbnN0KiBtb2R1bGU7Ci0KLSAgICBtQmxpdEVuZ2luZSA9IE5VTEw7Ci0gICAgaWYgKGh3X2dldF9tb2R1bGUoQ09QWUJJVF9IQVJEV0FSRV9NT0RVTEVfSUQsICZtb2R1bGUpID09IDApIHsKLSAgICAgICAgY29weWJpdF9vcGVuKG1vZHVsZSwgJm1CbGl0RW5naW5lKTsKLSAgICB9Ci0KLSAgICBtT3ZlcmxheUVuZ2luZSA9IE5VTEw7Ci0gICAgaWYgKGh3X2dldF9tb2R1bGUoT1ZFUkxBWV9IQVJEV0FSRV9NT0RVTEVfSUQsICZtb2R1bGUpID09IDApIHsKLSAgICAgICAgb3ZlcmxheV9jb250cm9sX29wZW4obW9kdWxlLCAmbU92ZXJsYXlFbmdpbmUpOwotICAgIH0KLX0KLQotLyoKLSAqIENsZWFuIHVwLiAgVGhyb3cgb3V0IG91ciBsb2NhbCBzdGF0ZS4KLSAqCi0gKiAoSXQncyBlbnRpcmVseSBwb3NzaWJsZSB3ZSdsbCBuZXZlciBnZXQgaGVyZSwgc2luY2UgdGhpcyBpcyBtZWFudAotICogZm9yIHJlYWwgaGFyZHdhcmUsIHdoaWNoIGRvZXNuJ3QgcmVzdGFydC4pCi0gKi8KLQotdm9pZCBEaXNwbGF5SGFyZHdhcmU6OmZpbmkoKQotewotICAgIGVnbE1ha2VDdXJyZW50KG1EaXNwbGF5LCBFR0xfTk9fU1VSRkFDRSwgRUdMX05PX1NVUkZBQ0UsIEVHTF9OT19DT05URVhUKTsKLSAgICBlZ2xUZXJtaW5hdGUobURpc3BsYXkpOwotICAgIGNvcHliaXRfY2xvc2UobUJsaXRFbmdpbmUpOwotICAgIG92ZXJsYXlfY29udHJvbF9jbG9zZShtT3ZlcmxheUVuZ2luZSk7Ci19Ci0KLXZvaWQgRGlzcGxheUhhcmR3YXJlOjpyZWxlYXNlU2NyZWVuKCkgY29uc3QKLXsKLSAgICBEaXNwbGF5SGFyZHdhcmVCYXNlOjpyZWxlYXNlU2NyZWVuKCk7Ci19Ci0KLXZvaWQgRGlzcGxheUhhcmR3YXJlOjphY3F1aXJlU2NyZWVuKCkgY29uc3QKLXsKLSAgICBEaXNwbGF5SGFyZHdhcmVCYXNlOjphY3F1aXJlU2NyZWVuKCk7Ci19Ci0KLXZvaWQgRGlzcGxheUhhcmR3YXJlOjpnZXREaXNwbGF5U3VyZmFjZShjb3B5Yml0X2ltYWdlX3QqIGltZykgY29uc3QKLXsKLSAgICBpbWctPncgICAgICA9IG1EaXNwbGF5U3VyZmFjZS0+c3RyaWRlOwotICAgIGltZy0+aCAgICAgID0gbURpc3BsYXlTdXJmYWNlLT5oZWlnaHQ7Ci0gICAgaW1nLT5mb3JtYXQgPSBtRGlzcGxheVN1cmZhY2UtPmZvcm1hdDsKLSAgICBpbWctPm9mZnNldCA9IG1EaXNwbGF5U3VyZmFjZS0+b2Zmc2V0OwotICAgIGltZy0+YmFzZSAgID0gKHZvaWQqKW1EaXNwbGF5U3VyZmFjZS0+YmFzZTsKLSAgICBpbWctPmZkICAgICA9IG1EaXNwbGF5U3VyZmFjZS0+ZmQ7Ci19Ci0KLXZvaWQgRGlzcGxheUhhcmR3YXJlOjpnZXREaXNwbGF5U3VyZmFjZShHR0xTdXJmYWNlKiBmYikgY29uc3QKLXsKLSAgICBmYi0+dmVyc2lvbj0gc2l6ZW9mKEdHTFN1cmZhY2UpOwotICAgIGZiLT53aWR0aCAgPSBtRGlzcGxheVN1cmZhY2UtPndpZHRoOwotICAgIGZiLT5oZWlnaHQgPSBtRGlzcGxheVN1cmZhY2UtPmhlaWdodDsKLSAgICBmYi0+c3RyaWRlID0gbURpc3BsYXlTdXJmYWNlLT5zdHJpZGU7Ci0gICAgZmItPmZvcm1hdCA9IG1EaXNwbGF5U3VyZmFjZS0+Zm9ybWF0OwotICAgIGZiLT5kYXRhICAgPSAoR0dMdWJ5dGUqKW1EaXNwbGF5U3VyZmFjZS0+YmFzZSArIG1EaXNwbGF5U3VyZmFjZS0+b2Zmc2V0OwotfQotCi11aW50MzJfdCBEaXNwbGF5SGFyZHdhcmU6OmdldFBhZ2VGbGlwQ291bnQoKSBjb25zdCB7Ci0gICAgcmV0dXJuIG1EaXNwbGF5U3VyZmFjZS0+Z2V0UGFnZUZsaXBDb3VudCgpOwotfQotCi0vKgotICogIkZsaXAiIHRoZSBmcm9udCBhbmQgYmFjayBidWZmZXJzLgotICovCi0KLXZvaWQgRGlzcGxheUhhcmR3YXJlOjpmbGlwKGNvbnN0IFJlZ2lvbiYgZGlydHkpIGNvbnN0Ci17Ci0gICAgY2hlY2tHTEVycm9ycygpOwotCi0gICAgRUdMRGlzcGxheSBkcHkgPSBtRGlzcGxheTsKLSAgICBFR0xTdXJmYWNlIHN1cmZhY2UgPSBtU3VyZmFjZTsKLQotICAgIFJlZ2lvbiBuZXdEaXJ0eShkaXJ0eSk7Ci0gICAgbmV3RGlydHkuYW5kU2VsZihSZWN0KG1XaWR0aCwgbUhlaWdodCkpOwotCi0gICAgaWYgKG1GbGFncyAmIEJVRkZFUl9QUkVTRVJWRUQpIHsKLSAgICAgICAgY29uc3QgUmVnaW9uIGNvcHliYWNrKG1EaXJ0eS5zdWJ0cmFjdChuZXdEaXJ0eSkpOwotICAgICAgICBtRGlydHkgPSBuZXdEaXJ0eTsKLSAgICAgICAgbURpc3BsYXlTdXJmYWNlLT5jb3B5RnJvbnRUb0JhY2soY29weWJhY2spOwotICAgIH0gCi0KLSAgICBpZiAobUZsYWdzICYgU1dBUF9SRUNUQU5HTEVfRVhURU5TSU9OKSB7Ci0gICAgICAgIGNvbnN0IFJlY3QmIGIobmV3RGlydHkuYm91bmRzKCkpOwotICAgICAgICBtRGlzcGxheVN1cmZhY2UtPnNldFN3YXBSZWN0YW5nbGUoCi0gICAgICAgICAgICAgICAgYi5sZWZ0LCBiLnRvcCwgYi53aWR0aCgpLCBiLmhlaWdodCgpKTsKLSAgICB9Ci0KLSAgICBlZ2xTd2FwQnVmZmVycyhkcHksIHN1cmZhY2UpOwotICAgIGNoZWNrRUdMRXJyb3JzKCJlZ2xTd2FwQnVmZmVycyIpOwotCi0gICAgLy8gZm9yIGRlYnVnZ2luZwotICAgIC8vZ2xDbGVhckNvbG9yKDEsMCwwLDApOwotICAgIC8vZ2xDbGVhcihHTF9DT0xPUl9CVUZGRVJfQklUKTsKLX0KLQotdWludDMyX3QgRGlzcGxheUhhcmR3YXJlOjpnZXRGbGFncygpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1GbGFnczsKLX0KLQotdm9pZCBEaXNwbGF5SGFyZHdhcmU6Om1ha2VDdXJyZW50KCkgY29uc3QKLXsKLSAgICBlZ2xNYWtlQ3VycmVudChtRGlzcGxheSwgbVN1cmZhY2UsIG1TdXJmYWNlLCBtQ29udGV4dCk7Ci19Ci0KLXZvaWQgRGlzcGxheUhhcmR3YXJlOjpjb3B5RnJvbnRUb0ltYWdlKGNvbnN0IGNvcHliaXRfaW1hZ2VfdCYgZnJvbnQpIGNvbnN0IHsKLSAgICBtRGlzcGxheVN1cmZhY2UtPmNvcHlGcm9udFRvSW1hZ2UoZnJvbnQpOwotfQotCi12b2lkIERpc3BsYXlIYXJkd2FyZTo6Y29weUJhY2tUb0ltYWdlKGNvbnN0IGNvcHliaXRfaW1hZ2VfdCYgZnJvbnQpIGNvbnN0IHsKLSAgICBtRGlzcGxheVN1cmZhY2UtPmNvcHlCYWNrVG9JbWFnZShmcm9udCk7Ci19CmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL0Rpc3BsYXlIYXJkd2FyZS9EaXNwbGF5SGFyZHdhcmUuaCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1NTBhNGQxLi4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZS5oCisrKyAvZGV2L251bGwKQEAgLTEsMTEzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfRElTUExBWV9IQVJEV0FSRV9ICi0jZGVmaW5lIEFORFJPSURfRElTUExBWV9IQVJEV0FSRV9ICi0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLQotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0jaW5jbHVkZSA8dWkvUmVnaW9uLmg+Ci0KLSNpbmNsdWRlIDxFR0wvZWdsLmg+Ci0KLSNpbmNsdWRlICJEaXNwbGF5SGFyZHdhcmUvRGlzcGxheUhhcmR3YXJlQmFzZS5oIgotCi1zdHJ1Y3Qgb3ZlcmxheV9jb250cm9sX2RldmljZV90Owotc3RydWN0IGNvcHliaXRfZGV2aWNlX3Q7Ci1zdHJ1Y3QgY29weWJpdF9pbWFnZV90Owotc3RydWN0IGNvcHliaXRfdDsKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBFR0xEaXNwbGF5U3VyZmFjZTsKLQotY2xhc3MgRGlzcGxheUhhcmR3YXJlIDogcHVibGljIERpc3BsYXlIYXJkd2FyZUJhc2UKLXsKLXB1YmxpYzoKLSAgICBlbnVtIHsKLSAgICAgICAgRElSRUNUX1RFWFRVUkUgICAgICAgICAgPSAweDAwMDAwMDAyLAotICAgICAgICBTV0FQX1JFQ1RBTkdMRV9FWFRFTlNJT049IDB4MDAwMDAwMDQsCi0gICAgICAgIENPUFlfQklUU19FWFRFTlNJT04gICAgID0gMHgwMDAwMDAwOCwKLSAgICAgICAgTlBPVF9FWFRFTlNJT04gICAgICAgICAgPSAweDAwMDAwMTAwLAotICAgICAgICBEUkFXX1RFWFRVUkVfRVhURU5TSU9OICA9IDB4MDAwMDAyMDAsCi0gICAgICAgIEJVRkZFUl9QUkVTRVJWRUQgICAgICAgID0gMHgwMDAxMDAwMCwKLSAgICAgICAgVVBEQVRFX09OX0RFTUFORCAgICAgICAgPSAweDAwMDIwMDAwLCAgIC8vIHZpZGVvIGRyaXZlciBmZWF0dXJlCi0gICAgICAgIFNMT1dfQ09ORklHICAgICAgICAgICAgID0gMHgwMDA0MDAwMCwgICAvLyBzb2Z0d2FyZQotICAgIH07Ci0KLSAgICBEaXNwbGF5SGFyZHdhcmUoCi0gICAgICAgICAgICBjb25zdCBzcDxTdXJmYWNlRmxpbmdlcj4mIGZsaW5nZXIsCi0gICAgICAgICAgICB1aW50MzJfdCBkaXNwbGF5SW5kZXgpOwotCi0gICAgfkRpc3BsYXlIYXJkd2FyZSgpOwotCi0gICAgdm9pZCByZWxlYXNlU2NyZWVuKCkgY29uc3Q7Ci0gICAgdm9pZCBhY3F1aXJlU2NyZWVuKCkgY29uc3Q7Ci0KLSAgICAvLyBGbGlwIHRoZSBmcm9udCBhbmQgYmFjayBidWZmZXJzIGlmIHRoZSBiYWNrIGJ1ZmZlciBpcyAiZGlydHkiLiAgTWlnaHQKLSAgICAvLyBiZSBpbnN0YW50YW5lb3VzLCBtaWdodCBpbnZvbHZlIGNvcHlpbmcgdGhlIGZyYW1lIGJ1ZmZlciBhcm91bmQuCi0gICAgdm9pZCBmbGlwKGNvbnN0IFJlZ2lvbiYgZGlydHkpIGNvbnN0OwotCi0gICAgZmxvYXQgICAgICAgZ2V0RHBpWCgpIGNvbnN0OwotICAgIGZsb2F0ICAgICAgIGdldERwaVkoKSBjb25zdDsKLSAgICBmbG9hdCAgICAgICBnZXRSZWZyZXNoUmF0ZSgpIGNvbnN0OwotICAgIGZsb2F0ICAgICAgIGdldERlbnNpdHkoKSBjb25zdDsKLSAgICBpbnQgICAgICAgICBnZXRXaWR0aCgpIGNvbnN0OwotICAgIGludCAgICAgICAgIGdldEhlaWdodCgpIGNvbnN0OwotICAgIFBpeGVsRm9ybWF0IGdldEZvcm1hdCgpIGNvbnN0OwotICAgIHVpbnQzMl90ICAgIGdldEZsYWdzKCkgY29uc3Q7Ci0gICAgdm9pZCAgICAgICAgbWFrZUN1cnJlbnQoKSBjb25zdDsKLQotICAgIHVpbnQzMl90IGdldFBhZ2VGbGlwQ291bnQoKSBjb25zdDsKLSAgICB2b2lkIGdldERpc3BsYXlTdXJmYWNlKGNvcHliaXRfaW1hZ2VfdCogaW1nKSBjb25zdDsKLSAgICB2b2lkIGdldERpc3BsYXlTdXJmYWNlKEdHTFN1cmZhY2UqIGZiKSBjb25zdDsKLSAgICBFR0xEaXNwbGF5IGdldEVHTERpc3BsYXkoKSBjb25zdCB7IHJldHVybiBtRGlzcGxheTsgfQotICAgIGNvcHliaXRfZGV2aWNlX3QqIGdldEJsaXRFbmdpbmUoKSBjb25zdCB7IHJldHVybiBtQmxpdEVuZ2luZTsgfQotICAgIG92ZXJsYXlfY29udHJvbF9kZXZpY2VfdCogZ2V0T3ZlcmxheUVuZ2luZSgpIGNvbnN0IHsgcmV0dXJuIG1PdmVybGF5RW5naW5lOyB9Ci0gICAgCi0gICAgdm9pZCBjb3B5RnJvbnRUb0ltYWdlKGNvbnN0IGNvcHliaXRfaW1hZ2VfdCYgZnJvbnQpIGNvbnN0OwotICAgIHZvaWQgY29weUJhY2tUb0ltYWdlKGNvbnN0IGNvcHliaXRfaW1hZ2VfdCYgZnJvbnQpIGNvbnN0OwotICAgICAgIAotICAgIFJlY3QgYm91bmRzKCkgY29uc3QgewotICAgICAgICByZXR1cm4gUmVjdChtV2lkdGgsIG1IZWlnaHQpOwotICAgIH0KLQotcHJpdmF0ZToKLSAgICB2b2lkIGluaXQodWludDMyX3QgZGlzcGxheUluZGV4KSBfX2F0dHJpYnV0ZV9fKChub2lubGluZSkpOwotICAgIHZvaWQgZmluaSgpIF9fYXR0cmlidXRlX18oKG5vaW5saW5lKSk7Ci0KLSAgICBFR0xEaXNwbGF5ICAgICAgbURpc3BsYXk7Ci0gICAgRUdMU3VyZmFjZSAgICAgIG1TdXJmYWNlOwotICAgIEVHTENvbnRleHQgICAgICBtQ29udGV4dDsKLSAgICBFR0xDb25maWcgICAgICAgbUNvbmZpZzsKLSAgICBmbG9hdCAgICAgICAgICAgbURwaVg7Ci0gICAgZmxvYXQgICAgICAgICAgIG1EcGlZOwotICAgIGZsb2F0ICAgICAgICAgICBtUmVmcmVzaFJhdGU7Ci0gICAgZmxvYXQgICAgICAgICAgIG1EZW5zaXR5OwotICAgIGludCAgICAgICAgICAgICBtV2lkdGg7Ci0gICAgaW50ICAgICAgICAgICAgIG1IZWlnaHQ7Ci0gICAgUGl4ZWxGb3JtYXQgICAgIG1Gb3JtYXQ7Ci0gICAgdWludDMyX3QgICAgICAgIG1GbGFnczsKLSAgICBtdXRhYmxlIFJlZ2lvbiAgbURpcnR5OwotICAgIHNwPEVHTERpc3BsYXlTdXJmYWNlPiBtRGlzcGxheVN1cmZhY2U7Ci0gICAgY29weWJpdF9kZXZpY2VfdCogICAgIG1CbGl0RW5naW5lOwotICAgIG92ZXJsYXlfY29udHJvbF9kZXZpY2VfdCogbU92ZXJsYXlFbmdpbmU7Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9ESVNQTEFZX0hBUkRXQVJFX0gKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZUJhc2UuY3BwIGIvbGlicy9zdXJmYWNlZmxpbmdlci9EaXNwbGF5SGFyZHdhcmUvRGlzcGxheUhhcmR3YXJlQmFzZS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGY3NWU1YzIuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9EaXNwbGF5SGFyZHdhcmUvRGlzcGxheUhhcmR3YXJlQmFzZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw0MDMgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNkZWZpbmUgTE9HX1RBRyAiU3VyZmFjZUZsaW5nZXIiCi0KLSNpbmNsdWRlIDxhc3NlcnQuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPHNpZ25hbC5oPgotI2luY2x1ZGUgPHRlcm1pb3MuaD4KLSNpbmNsdWRlIDxzeXMvaW9jdGwuaD4KLSNpbmNsdWRlIDxzeXMvbW1hbi5oPgotI2luY2x1ZGUgPHN5cy90aW1lLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8c3lzL3Jlc291cmNlLmg+Ci0KLSNpbmNsdWRlIDxsaW51eC91bmlzdGQuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSAiRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZUJhc2UuaCIKLSNpbmNsdWRlICJTdXJmYWNlRmxpbmdlci5oIgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyB0aGUgc2ltIGJ1aWxkIGRvZXNuJ3QgaGF2ZSBnZXR0aWQKLQotI2lmbmRlZiBIQVZFX0dFVFRJRAotIyBkZWZpbmUgZ2V0dGlkIGdldHBpZAotI2VuZGlmCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotc3RhdGljIGNoYXIgY29uc3QgKiBrU2xlZXBGaWxlTmFtZSA9ICIvc3lzL3Bvd2VyL3dhaXRfZm9yX2ZiX3NsZWVwIjsKLXN0YXRpYyBjaGFyIGNvbnN0ICoga1dha2VGaWxlTmFtZSA9ICIvc3lzL3Bvd2VyL3dhaXRfZm9yX2ZiX3dha2UiOwotc3RhdGljIGNoYXIgY29uc3QgKiBjb25zdCBrT2xkU2xlZXBGaWxlTmFtZSA9ICIvc3lzL2FuZHJvaWRfcG93ZXIvd2FpdF9mb3JfZmJfc2xlZXAiOwotc3RhdGljIGNoYXIgY29uc3QgKiBjb25zdCBrT2xkV2FrZUZpbGVOYW1lID0gIi9zeXMvYW5kcm9pZF9wb3dlci93YWl0X2Zvcl9mYl93YWtlIjsKLQotLy8gVGhpcyBkaXIgZXhpc3RzIGlmIHRoZSBmcmFtZWJ1ZmZlciBjb25zb2xlIGlzIHByZXNlbnQsIGVpdGhlciBidWlsdCBpbnRvCi0vLyB0aGUga2VybmVsIG9yIGxvYWRlZCBhcyBhIG1vZHVsZS4KLXN0YXRpYyBjaGFyIGNvbnN0ICogY29uc3Qga0ZiY29uU3lzRGlyID0gIi9zeXMvY2xhc3MvZ3JhcGhpY3MvZmJjb24iOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLURpc3BsYXlIYXJkd2FyZUJhc2U6OkRpc3BsYXlFdmVudFRocmVhZEJhc2U6OkRpc3BsYXlFdmVudFRocmVhZEJhc2UoCi0gICAgICAgIGNvbnN0IHNwPFN1cmZhY2VGbGluZ2VyPiYgZmxpbmdlcikKLSAgICA6IFRocmVhZChmYWxzZSksIG1GbGluZ2VyKGZsaW5nZXIpIHsKLX0KLQotRGlzcGxheUhhcmR3YXJlQmFzZTo6RGlzcGxheUV2ZW50VGhyZWFkQmFzZTo6fkRpc3BsYXlFdmVudFRocmVhZEJhc2UoKSB7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotRGlzcGxheUhhcmR3YXJlQmFzZTo6RGlzcGxheUV2ZW50VGhyZWFkOjpEaXNwbGF5RXZlbnRUaHJlYWQoCi0gICAgICAgIGNvbnN0IHNwPFN1cmZhY2VGbGluZ2VyPiYgZmxpbmdlcikKLSAgICA6IERpc3BsYXlFdmVudFRocmVhZEJhc2UoZmxpbmdlcikKLXsKLX0KLQotRGlzcGxheUhhcmR3YXJlQmFzZTo6RGlzcGxheUV2ZW50VGhyZWFkOjp+RGlzcGxheUV2ZW50VGhyZWFkKCkKLXsKLX0KLQotYm9vbCBEaXNwbGF5SGFyZHdhcmVCYXNlOjpEaXNwbGF5RXZlbnRUaHJlYWQ6OnRocmVhZExvb3AoKQotewotICAgIGludCBlcnIgPSAwOwotICAgIGNoYXIgYnVmOwotICAgIGludCBmZDsKLQotICAgIGZkID0gb3BlbihrU2xlZXBGaWxlTmFtZSwgT19SRE9OTFksIDApOwotICAgIGRvIHsKLSAgICAgIGVyciA9IHJlYWQoZmQsICZidWYsIDEpOwotICAgIH0gd2hpbGUgKGVyciA8IDAgJiYgZXJybm8gPT0gRUlOVFIpOwotICAgIGNsb3NlKGZkKTsKLSAgICBMT0dXX0lGKGVycjwwLCAiQU5EUk9JRF9XQUlUX0ZPUl9GQl9TTEVFUCBmYWlsZWQgKCVzKSIsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgaWYgKGVyciA+PSAwKSB7Ci0gICAgICAgIHNwPFN1cmZhY2VGbGluZ2VyPiBmbGluZ2VyID0gbUZsaW5nZXIucHJvbW90ZSgpOwotICAgICAgICBMT0dEKCJBYm91dCB0byBnaXZlLXVwIHNjcmVlbiwgZmxpbmdlciA9ICVwIiwgZmxpbmdlci5nZXQoKSk7Ci0gICAgICAgIGlmIChmbGluZ2VyICE9IDApIHsKLSAgICAgICAgICAgIG1CYXJyaWVyLmNsb3NlKCk7Ci0gICAgICAgICAgICBmbGluZ2VyLT5zY3JlZW5SZWxlYXNlZCgwKTsKLSAgICAgICAgICAgIG1CYXJyaWVyLndhaXQoKTsKLSAgICAgICAgfQotICAgIH0KLSAgICBmZCA9IG9wZW4oa1dha2VGaWxlTmFtZSwgT19SRE9OTFksIDApOwotICAgIGRvIHsKLSAgICAgIGVyciA9IHJlYWQoZmQsICZidWYsIDEpOwotICAgIH0gd2hpbGUgKGVyciA8IDAgJiYgZXJybm8gPT0gRUlOVFIpOwotICAgIGNsb3NlKGZkKTsKLSAgICBMT0dXX0lGKGVycjwwLCAiQU5EUk9JRF9XQUlUX0ZPUl9GQl9XQUtFIGZhaWxlZCAoJXMpIiwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICBpZiAoZXJyID49IDApIHsKLSAgICAgICAgc3A8U3VyZmFjZUZsaW5nZXI+IGZsaW5nZXIgPSBtRmxpbmdlci5wcm9tb3RlKCk7Ci0gICAgICAgIExPR0QoIlNjcmVlbiBhYm91dCB0byByZXR1cm4sIGZsaW5nZXIgPSAlcCIsIGZsaW5nZXIuZ2V0KCkpOwotICAgICAgICBpZiAoZmxpbmdlciAhPSAwKQotICAgICAgICAgICAgZmxpbmdlci0+c2NyZWVuQWNxdWlyZWQoMCk7Ci0gICAgfQotICAgIHJldHVybiB0cnVlOwotfQotCi1zdGF0dXNfdCBEaXNwbGF5SGFyZHdhcmVCYXNlOjpEaXNwbGF5RXZlbnRUaHJlYWQ6OnJlbGVhc2VTY3JlZW4oKSBjb25zdAotewotICAgIG1CYXJyaWVyLm9wZW4oKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IERpc3BsYXlIYXJkd2FyZUJhc2U6OkRpc3BsYXlFdmVudFRocmVhZDo6cmVhZHlUb1J1bigpCi17Ci0gICAgaWYgKGFjY2VzcyhrU2xlZXBGaWxlTmFtZSwgUl9PSykgfHwgYWNjZXNzKGtXYWtlRmlsZU5hbWUsIFJfT0spKSB7Ci0gICAgICAgIGlmIChhY2Nlc3Moa09sZFNsZWVwRmlsZU5hbWUsIFJfT0spIHx8IGFjY2VzcyhrT2xkV2FrZUZpbGVOYW1lLCBSX09LKSkgewotICAgICAgICAgICAgTE9HRSgiQ291bGRuJ3Qgb3BlbiAlcyBvciAlcyIsIGtTbGVlcEZpbGVOYW1lLCBrV2FrZUZpbGVOYW1lKTsKLSAgICAgICAgICAgIHJldHVybiBOT19JTklUOwotICAgICAgICB9Ci0gICAgICAgIGtTbGVlcEZpbGVOYW1lID0ga09sZFNsZWVwRmlsZU5hbWU7Ci0gICAgICAgIGtXYWtlRmlsZU5hbWUgPSBrT2xkV2FrZUZpbGVOYW1lOwotICAgIH0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IERpc3BsYXlIYXJkd2FyZUJhc2U6OkRpc3BsYXlFdmVudFRocmVhZDo6aW5pdENoZWNrKCkgY29uc3QKLXsKLSAgICByZXR1cm4gKCgoYWNjZXNzKGtTbGVlcEZpbGVOYW1lLCBSX09LKSA9PSAwICYmCi0gICAgICAgICAgICBhY2Nlc3Moa1dha2VGaWxlTmFtZSwgUl9PSykgPT0gMCkgfHwKLSAgICAgICAgICAgIChhY2Nlc3Moa09sZFNsZWVwRmlsZU5hbWUsIFJfT0spID09IDAgJiYKLSAgICAgICAgICAgIGFjY2VzcyhrT2xkV2FrZUZpbGVOYW1lLCBSX09LKSA9PSAwKSkgJiYKLSAgICAgICAgICAgIGFjY2VzcyhrRmJjb25TeXNEaXIsIEZfT0spICE9IDApID8gTk9fRVJST1IgOiBOT19JTklUOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXBpZF90IERpc3BsYXlIYXJkd2FyZUJhc2U6OkNvbnNvbGVNYW5hZ2VyVGhyZWFkOjpzU2lnbmFsQ2F0Y2hlclBpZCA9IDA7Ci0KLURpc3BsYXlIYXJkd2FyZUJhc2U6OkNvbnNvbGVNYW5hZ2VyVGhyZWFkOjpDb25zb2xlTWFuYWdlclRocmVhZCgKLSAgICAgICAgY29uc3Qgc3A8U3VyZmFjZUZsaW5nZXI+JiBmbGluZ2VyKQotICAgIDogRGlzcGxheUV2ZW50VGhyZWFkQmFzZShmbGluZ2VyKSwgY29uc29sZUZkKC0xKQoteyAgIAotICAgIHNTaWduYWxDYXRjaGVyUGlkID0gMDsKLQotICAgIC8vIGNyZWF0ZSBhIG5ldyBjb25zb2xlCi0gICAgY2hhciBjb25zdCAqIGNvbnN0IHR0eWRldiA9ICIvZGV2L3R0eTAiOwotICAgIGludCBmZCA9IG9wZW4odHR5ZGV2LCBPX1JEV1IgfCBPX1NZTkMpOwotICAgIGlmIChmZDwwKSB7Ci0gICAgICAgIExPR0UoIkNhbid0IG9wZW4gJXMiLCB0dHlkZXYpOwotICAgICAgICB0aGlzLT5jb25zb2xlRmQgPSAtZXJybm87Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICAvLyB0byBtYWtlIHN1cmUgdGhhdCB3ZSBhcmUgaW4gdGV4dCBtb2RlCi0gICAgaW50IHJlcyA9IGlvY3RsKGZkLCBLRFNFVE1PREUsICh2b2lkKikgS0RfVEVYVCk7Ci0gICAgaWYgKHJlczwwKSB7Ci0gICAgICAgIExPR0UoImlvY3RsKCVkLCBLRFNFVE1PREUsIC4uLikgZmFpbGVkLCByZXMgJWQgKCVzKSIsCi0gICAgICAgICAgICAgICAgZmQsIHJlcywgc3RyZXJyb3IoZXJybm8pKTsKLSAgICB9Ci0gICAgCi0gICAgLy8gZ2V0IHRoZSBjdXJyZW50IGNvbnNvbGUKLSAgICBzdHJ1Y3QgdnRfc3RhdCB2czsKLSAgICByZXMgPSBpb2N0bChmZCwgVlRfR0VUU1RBVEUsICZ2cyk7Ci0gICAgaWYgKHJlczwwKSB7Ci0gICAgICAgIExPR0UoImlvY3RsKCVkLCBWVF9HRVRTVEFURSwgLi4uKSBmYWlsZWQsIHJlcyAlZCAoJXMpIiwKLSAgICAgICAgICAgICAgICBmZCwgcmVzLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICB0aGlzLT5jb25zb2xlRmQgPSAtZXJybm87Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICAvLyBzd2l0Y2ggdG8gY29uc29sZSA3ICh3aGljaCBpcyB3aGF0IFggbm9ybWFseSB1c2VzKQotICAgIGludCB2dG51bSA9IDc7Ci0gICAgZG8gewotICAgICAgICByZXMgPSBpb2N0bChmZCwgVlRfQUNUSVZBVEUsICh2b2lkKil2dG51bSk7Ci0gICAgfSB3aGlsZShyZXMgPCAwICYmIGVycm5vID09IEVJTlRSKTsKLSAgICBpZiAocmVzPDApIHsKLSAgICAgICAgTE9HRSgiaW9jdGwoJWQsIFZUX0FDVElWQVRFLCAuLi4pIGZhaWxlZCwgJWQgKCVzKSBmb3IgJWQiLAotICAgICAgICAgICAgICAgIGZkLCBlcnJubywgc3RyZXJyb3IoZXJybm8pLCB2dG51bSk7Ci0gICAgICAgIHRoaXMtPmNvbnNvbGVGZCA9IC1lcnJubzsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIGRvIHsKLSAgICAgICAgcmVzID0gaW9jdGwoZmQsIFZUX1dBSVRBQ1RJVkUsICh2b2lkKil2dG51bSk7Ci0gICAgfSB3aGlsZShyZXMgPCAwICYmIGVycm5vID09IEVJTlRSKTsKLSAgICBpZiAocmVzPDApIHsKLSAgICAgICAgTE9HRSgiaW9jdGwoJWQsIFZUX1dBSVRBQ1RJVkUsIC4uLikgZmFpbGVkLCAlZCAlZCAlcyBmb3IgJWQiLAotICAgICAgICAgICAgICAgIGZkLCByZXMsIGVycm5vLCBzdHJlcnJvcihlcnJubyksIHZ0bnVtKTsKLSAgICAgICAgdGhpcy0+Y29uc29sZUZkID0gLWVycm5vOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgLy8gb3BlbiB0aGUgbmV3IGNvbnNvbGUKLSAgICBjbG9zZShmZCk7Ci0gICAgZmQgPSBvcGVuKHR0eWRldiwgT19SRFdSIHwgT19TWU5DKTsKLSAgICBpZiAoZmQ8MCkgewotICAgICAgICBMT0dFKCJDYW4ndCBvcGVuIG5ldyBjb25zb2xlICVzIiwgdHR5ZGV2KTsKLSAgICAgICAgdGhpcy0+Y29uc29sZUZkID0gLWVycm5vOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgLyogZGlzYWJsZSBjb25zb2xlIGxpbmUgYnVmZmVyLCBlY2hvLCAuLi4gKi8KLSAgICBzdHJ1Y3QgdGVybWlvcyB0dHlhcmc7Ci0gICAgaW9jdGwoZmQsIFRDR0VUUyAsICZ0dHlhcmcpOwotICAgIHR0eWFyZy5jX2lmbGFnID0gMDsKLSAgICB0dHlhcmcuY19sZmxhZyA9IDA7Ci0gICAgaW9jdGwoZmQsIFRDU0VUUyAsICZ0dHlhcmcpOwotCi0gICAgLy8gc2V0IHVwIHNpZ25hbHMgc28gd2UncmUgbm90aWZpZWQgd2hlbiB0aGUgY29uc29sZSBjaGFuZ2VzCi0gICAgLy8gd2UgY2FuJ3QgdXNlIFNJR1VTUjEgYmVjYXVzZSBpdCdzIHVzZWQgYnkgdGhlIGphdmEtdm0KLSAgICB2bS5tb2RlID0gVlRfUFJPQ0VTUzsKLSAgICB2bS53YWl0diA9IDA7Ci0gICAgdm0ucmVsc2lnID0gU0lHVVNSMjsKLSAgICB2bS5hY3FzaWcgPSBTSUdVTlVTRUQ7Ci0gICAgdm0uZnJzaWcgPSAwOwotCi0gICAgc3RydWN0IHNpZ2FjdGlvbiBhY3Q7Ci0gICAgc2lnZW1wdHlzZXQoJmFjdC5zYV9tYXNrKTsKLSAgICBhY3Quc2FfaGFuZGxlciA9IHNpZ0hhbmRsZXI7Ci0gICAgYWN0LnNhX2ZsYWdzID0gMDsKLSAgICBzaWdhY3Rpb24odm0ucmVsc2lnLCAmYWN0LCBOVUxMKTsKLQotICAgIHNpZ2VtcHR5c2V0KCZhY3Quc2FfbWFzayk7Ci0gICAgYWN0LnNhX2hhbmRsZXIgPSBzaWdIYW5kbGVyOwotICAgIGFjdC5zYV9mbGFncyA9IDA7Ci0gICAgc2lnYWN0aW9uKHZtLmFjcXNpZywgJmFjdCwgTlVMTCk7Ci0KLSAgICBzaWdzZXRfdCBtYXNrOwotICAgIHNpZ2VtcHR5c2V0KCZtYXNrKTsKLSAgICBzaWdhZGRzZXQoJm1hc2ssIHZtLnJlbHNpZyk7Ci0gICAgc2lnYWRkc2V0KCZtYXNrLCB2bS5hY3FzaWcpOwotICAgIHNpZ3Byb2NtYXNrKFNJR19CTE9DSywgJm1hc2ssIE5VTEwpOwotCi0gICAgLy8gc3dpdGNoIHRvIGdyYXBoaWMgbW9kZQotICAgIHJlcyA9IGlvY3RsKGZkLCBLRFNFVE1PREUsICh2b2lkKilLRF9HUkFQSElDUyk7Ci0gICAgTE9HV19JRihyZXM8MCwKLSAgICAgICAgICAgICJpb2N0bCglZCwgS0RTRVRNT0RFLCBLRF9HUkFQSElDUykgZmFpbGVkLCByZXMgJWQiLCBmZCwgcmVzKTsKLQotICAgIHRoaXMtPnByZXZfdnRfbnVtID0gdnMudl9hY3RpdmU7Ci0gICAgdGhpcy0+dnRfbnVtID0gdnRudW07Ci0gICAgdGhpcy0+Y29uc29sZUZkID0gZmQ7Ci19Ci0KLURpc3BsYXlIYXJkd2FyZUJhc2U6OkNvbnNvbGVNYW5hZ2VyVGhyZWFkOjp+Q29uc29sZU1hbmFnZXJUaHJlYWQoKQoteyAgIAotICAgIGlmICh0aGlzLT5jb25zb2xlRmQgPj0gMCkgewotICAgICAgICBpbnQgZmQgPSB0aGlzLT5jb25zb2xlRmQ7Ci0gICAgICAgIGludCBwcmV2X3Z0X251bSA9IHRoaXMtPnByZXZfdnRfbnVtOwotICAgICAgICBpbnQgcmVzOwotICAgICAgICBpb2N0bChmZCwgS0RTRVRNT0RFLCAodm9pZCopS0RfVEVYVCk7Ci0gICAgICAgIGRvIHsKLSAgICAgICAgICAgIHJlcyA9IGlvY3RsKGZkLCBWVF9BQ1RJVkFURSwgKHZvaWQqKXByZXZfdnRfbnVtKTsKLSAgICAgICAgfSB3aGlsZShyZXMgPCAwICYmIGVycm5vID09IEVJTlRSKTsKLSAgICAgICAgZG8gewotICAgICAgICAgICAgcmVzID0gaW9jdGwoZmQsIFZUX1dBSVRBQ1RJVkUsICh2b2lkKilwcmV2X3Z0X251bSk7Ci0gICAgICAgIH0gd2hpbGUocmVzIDwgMCAmJiBlcnJubyA9PSBFSU5UUik7Ci0gICAgICAgIGNsb3NlKGZkKTsgICAgCi0gICAgICAgIGNoYXIgY29uc3QgKiBjb25zdCB0dHlkZXYgPSAiL2Rldi90dHkwIjsKLSAgICAgICAgZmQgPSBvcGVuKHR0eWRldiwgT19SRFdSIHwgT19TWU5DKTsKLSAgICAgICAgaW9jdGwoZmQsIFZUX0RJU0FMTE9DQVRFLCAwKTsKLSAgICAgICAgY2xvc2UoZmQpOwotICAgIH0KLX0KLQotc3RhdHVzX3QgRGlzcGxheUhhcmR3YXJlQmFzZTo6Q29uc29sZU1hbmFnZXJUaHJlYWQ6OnJlYWR5VG9SdW4oKQotewotICAgIGlmICh0aGlzLT5jb25zb2xlRmQgPj0gMCkgewotICAgICAgICBzU2lnbmFsQ2F0Y2hlclBpZCA9IGdldHRpZCgpOwotICAgICAgICAKLSAgICAgICAgc2lnc2V0X3QgbWFzazsKLSAgICAgICAgc2lnZW1wdHlzZXQoJm1hc2spOwotICAgICAgICBzaWdhZGRzZXQoJm1hc2ssIHZtLnJlbHNpZyk7Ci0gICAgICAgIHNpZ2FkZHNldCgmbWFzaywgdm0uYWNxc2lnKTsKLSAgICAgICAgc2lncHJvY21hc2soU0lHX0JMT0NLLCAmbWFzaywgTlVMTCk7Ci0KLSAgICAgICAgaW50IHJlcyA9IGlvY3RsKHRoaXMtPmNvbnNvbGVGZCwgVlRfU0VUTU9ERSwgJnZtKTsKLSAgICAgICAgaWYgKHJlczwwKSB7Ci0gICAgICAgICAgICBMT0dFKCJpb2N0bCglZCwgVlRfU0VUTU9ERSwgLi4uKSBmYWlsZWQsICVkICglcykiLAotICAgICAgICAgICAgICAgICAgICB0aGlzLT5jb25zb2xlRmQsIGVycm5vLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIHRoaXMtPmNvbnNvbGVGZDsKLX0KLQotdm9pZCBEaXNwbGF5SGFyZHdhcmVCYXNlOjpDb25zb2xlTWFuYWdlclRocmVhZDo6cmVxdWVzdEV4aXQoKQotewotICAgIFRocmVhZDo6cmVxdWVzdEV4aXQoKTsKLSAgICBpZiAoc1NpZ25hbENhdGNoZXJQaWQgIT0gMCkgewotICAgICAgICAvLyB3YWtlIHRoZSB0aHJlYWQgdXAKLSAgICAgICAga2lsbChzU2lnbmFsQ2F0Y2hlclBpZCwgU0lHSU5UKTsKLSAgICAgICAgLy8gd2FpdCBmb3IgaXQuLi4KLSAgICB9Ci19Ci0KLXZvaWQgRGlzcGxheUhhcmR3YXJlQmFzZTo6Q29uc29sZU1hbmFnZXJUaHJlYWQ6OnNpZ0hhbmRsZXIoaW50IHNpZykKLXsKLSAgICAvLyByZXNlbmQgdGhlIHNpZ25hbCB0byBvdXIgc2lnbmFsIGNhdGNoZXIgdGhyZWFkCi0gICAgTE9HVygicmVjZWl2ZWQgc2lnbmFsICVkIGluIHRocmVhZCAlZCwgcmVzZW5kaW5nIHRvICVkIiwKLSAgICAgICAgICAgIHNpZywgZ2V0dGlkKCksIHNTaWduYWxDYXRjaGVyUGlkKTsKLQotICAgIC8vIHdlIGFic29sdXRlbHkgbmVlZCB0aGUgZGVsYXlzIGJlbG93IGJlY2F1c2Ugd2l0aG91dCB0aGVtCi0gICAgLy8gb3VyIG1haW4gdGhyZWFkIG5ldmVyIGdldHMgYSBjaGFuY2UgdG8gaGFuZGxlIHRoZSBzaWduYWwuCi0gICAgdXNsZWVwKDEwMDAwKTsKLSAgICBraWxsKHNTaWduYWxDYXRjaGVyUGlkLCBzaWcpOwotICAgIHVzbGVlcCgxMDAwMCk7Ci19Ci0KLXN0YXR1c190IERpc3BsYXlIYXJkd2FyZUJhc2U6OkNvbnNvbGVNYW5hZ2VyVGhyZWFkOjpyZWxlYXNlU2NyZWVuKCkgY29uc3QKLXsKLSAgICBpbnQgZmQgPSB0aGlzLT5jb25zb2xlRmQ7Ci0gICAgaW50IGVyciA9IGlvY3RsKGZkLCBWVF9SRUxESVNQLCAodm9pZCopMSk7Ci0gICAgTE9HRV9JRihlcnI8MCwgImlvY3RsKCVkLCBWVF9SRUxESVNQLCAxKSBmYWlsZWQgJWQgKCVzKSIsCi0gICAgICAgIGZkLCBlcnJubywgc3RyZXJyb3IoZXJybm8pKTsKLSAgICByZXR1cm4gKGVycjwwKSA/ICgtZXJybm8pIDogc3RhdHVzX3QoTk9fRVJST1IpOwotfQotCi1ib29sIERpc3BsYXlIYXJkd2FyZUJhc2U6OkNvbnNvbGVNYW5hZ2VyVGhyZWFkOjp0aHJlYWRMb29wKCkKLXsKLSAgICBzaWdzZXRfdCBtYXNrOwotICAgIHNpZ2VtcHR5c2V0KCZtYXNrKTsKLSAgICBzaWdhZGRzZXQoJm1hc2ssIHZtLnJlbHNpZyk7Ci0gICAgc2lnYWRkc2V0KCZtYXNrLCB2bS5hY3FzaWcpOwotCi0gICAgaW50IHNpZyA9IDA7Ci0gICAgc2lnd2FpdCgmbWFzaywgJnNpZyk7Ci0KLSAgICBpZiAoc2lnID09IHZtLnJlbHNpZykgewotICAgICAgICBzcDxTdXJmYWNlRmxpbmdlcj4gZmxpbmdlciA9IG1GbGluZ2VyLnByb21vdGUoKTsKLSAgICAgICAgLy9MT0dEKCJBYm91dCB0byBnaXZlLXVwIHNjcmVlbiwgZmxpbmdlciA9ICVwIiwgZmxpbmdlci5nZXQoKSk7Ci0gICAgICAgIGlmIChmbGluZ2VyICE9IDApCi0gICAgICAgICAgICBmbGluZ2VyLT5zY3JlZW5SZWxlYXNlZCgwKTsKLSAgICB9IGVsc2UgaWYgKHNpZyA9PSB2bS5hY3FzaWcpIHsKLSAgICAgICAgc3A8U3VyZmFjZUZsaW5nZXI+IGZsaW5nZXIgPSBtRmxpbmdlci5wcm9tb3RlKCk7Ci0gICAgICAgIC8vTE9HRCgiU2NyZWVuIGFib3V0IHRvIHJldHVybiwgZmxpbmdlciA9ICVwIiwgZmxpbmdlci5nZXQoKSk7Ci0gICAgICAgIGlmIChmbGluZ2VyICE9IDApIAotICAgICAgICAgICAgZmxpbmdlci0+c2NyZWVuQWNxdWlyZWQoMCk7Ci0gICAgfQotICAgIAotICAgIHJldHVybiB0cnVlOwotfQotCi1zdGF0dXNfdCBEaXNwbGF5SGFyZHdhcmVCYXNlOjpDb25zb2xlTWFuYWdlclRocmVhZDo6aW5pdENoZWNrKCkgY29uc3QKLXsKLSAgICByZXR1cm4gY29uc29sZUZkID49IDAgPyBOT19FUlJPUiA6IE5PX0lOSVQ7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotRGlzcGxheUhhcmR3YXJlQmFzZTo6RGlzcGxheUhhcmR3YXJlQmFzZShjb25zdCBzcDxTdXJmYWNlRmxpbmdlcj4mIGZsaW5nZXIsCi0gICAgICAgIHVpbnQzMl90IGRpc3BsYXlJbmRleCkgCi0gICAgOiBtQ2FuRHJhdyh0cnVlKQotewotICAgIG1EaXNwbGF5RXZlbnRUaHJlYWQgPSBuZXcgRGlzcGxheUV2ZW50VGhyZWFkKGZsaW5nZXIpOwotICAgIGlmIChtRGlzcGxheUV2ZW50VGhyZWFkLT5pbml0Q2hlY2soKSAhPSBOT19FUlJPUikgewotICAgICAgICAvLyBmYWxsLWJhY2sgb24gdGhlIGNvbnNvbGUKLSAgICAgICAgbURpc3BsYXlFdmVudFRocmVhZCA9IG5ldyBDb25zb2xlTWFuYWdlclRocmVhZChmbGluZ2VyKTsKLSAgICB9Ci19Ci0KLURpc3BsYXlIYXJkd2FyZUJhc2U6On5EaXNwbGF5SGFyZHdhcmVCYXNlKCkKLXsKLSAgICAvLyByZXF1ZXN0IGV4aXQKLSAgICBtRGlzcGxheUV2ZW50VGhyZWFkLT5yZXF1ZXN0RXhpdEFuZFdhaXQoKTsKLX0KLQotCi1ib29sIERpc3BsYXlIYXJkd2FyZUJhc2U6OmNhbkRyYXcoKSBjb25zdAotewotICAgIHJldHVybiBtQ2FuRHJhdzsKLX0KLQotdm9pZCBEaXNwbGF5SGFyZHdhcmVCYXNlOjpyZWxlYXNlU2NyZWVuKCkgY29uc3QKLXsKLSAgICBzdGF0dXNfdCBlcnIgPSBtRGlzcGxheUV2ZW50VGhyZWFkLT5yZWxlYXNlU2NyZWVuKCk7Ci0gICAgaWYgKGVyciA+PSAwKSB7Ci0gICAgICAgIC8vTE9HRCgic2NyZWVuIGdpdmVuLXVwIik7Ci0gICAgICAgIG1DYW5EcmF3ID0gZmFsc2U7Ci0gICAgfQotfQotCi12b2lkIERpc3BsYXlIYXJkd2FyZUJhc2U6OmFjcXVpcmVTY3JlZW4oKSBjb25zdAotewotICAgIHN0YXR1c190IGVyciA9IG1EaXNwbGF5RXZlbnRUaHJlYWQtPmFjcXVpcmVTY3JlZW4oKTsKLSAgICBpZiAoZXJyID49IDApIHsKLSAgICAgICAgLy9MT0dEKCJzY3JlZW4gcmV0dXJuZWQiKTsKLSAgICAgICAgbUNhbkRyYXcgPSB0cnVlOwotICAgIH0KLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZUJhc2UuaCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZUJhc2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggODM2OWJiOC4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0Rpc3BsYXlIYXJkd2FyZS9EaXNwbGF5SGFyZHdhcmVCYXNlLmgKKysrIC9kZXYvbnVsbApAQCAtMSw5NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0RJU1BMQVlfSEFSRFdBUkVfQkFTRV9ICi0jZGVmaW5lIEFORFJPSURfRElTUExBWV9IQVJEV0FSRV9CQVNFX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0jaW5jbHVkZSA8bGludXgva2QuaD4KLSNpbmNsdWRlIDxsaW51eC92dC5oPgotI2luY2x1ZGUgIkJhcnJpZXIuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBTdXJmYWNlRmxpbmdlcjsgCi0KLWNsYXNzIERpc3BsYXlIYXJkd2FyZUJhc2UKLXsKLXB1YmxpYzoKLSAgICAgICAgICAgICAgICBEaXNwbGF5SGFyZHdhcmVCYXNlKAotICAgICAgICAgICAgICAgICAgICAgICAgY29uc3Qgc3A8U3VyZmFjZUZsaW5nZXI+JiBmbGluZ2VyLAotICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZGlzcGxheUluZGV4KTsKLQotICAgICAgICAgICAgICAgIH5EaXNwbGF5SGFyZHdhcmVCYXNlKCk7Ci0KLSAgICAvLyBjb25zb2xlIG1hbmFnbWVudAotICAgIHZvaWQgcmVsZWFzZVNjcmVlbigpIGNvbnN0OwotICAgIHZvaWQgYWNxdWlyZVNjcmVlbigpIGNvbnN0OwotICAgIGJvb2wgY2FuRHJhdygpIGNvbnN0OwotCi1wcml2YXRlOgotICAgIGNsYXNzIERpc3BsYXlFdmVudFRocmVhZEJhc2UgOiBwdWJsaWMgVGhyZWFkIHsKLSAgICBwcm90ZWN0ZWQ6Ci0gICAgICAgIHdwPFN1cmZhY2VGbGluZ2VyPiBtRmxpbmdlcjsKLSAgICBwdWJsaWM6Ci0gICAgICAgIERpc3BsYXlFdmVudFRocmVhZEJhc2UoY29uc3Qgc3A8U3VyZmFjZUZsaW5nZXI+JiBmbGluZ2VyKTsKLSAgICAgICAgdmlydHVhbCB+RGlzcGxheUV2ZW50VGhyZWFkQmFzZSgpOwotICAgICAgICB2aXJ0dWFsIHZvaWQgb25GaXJzdFJlZigpIHsKLSAgICAgICAgICAgIHJ1bigiRGlzcGxheUV2ZW50VGhyZWFkIiwgUFJJT1JJVFlfVVJHRU5UX0RJU1BMQVkpOwotICAgICAgICB9Ci0gICAgICAgIHZpcnR1YWwgc3RhdHVzX3QgYWNxdWlyZVNjcmVlbigpIGNvbnN0IHsgcmV0dXJuIE5PX0VSUk9SOyB9OwotICAgICAgICB2aXJ0dWFsIHN0YXR1c190IHJlbGVhc2VTY3JlZW4oKSBjb25zdCB7IHJldHVybiBOT19FUlJPUjsgfTsKLSAgICAgICAgdmlydHVhbCBzdGF0dXNfdCBpbml0Q2hlY2soKSBjb25zdCA9IDA7Ci0gICAgfTsKLQotICAgIGNsYXNzIERpc3BsYXlFdmVudFRocmVhZCA6IHB1YmxpYyBEaXNwbGF5RXZlbnRUaHJlYWRCYXNlIAotICAgIHsKLSAgICAgICAgbXV0YWJsZSBCYXJyaWVyIG1CYXJyaWVyOwotICAgIHB1YmxpYzoKLSAgICAgICAgICAgICAgICBEaXNwbGF5RXZlbnRUaHJlYWQoY29uc3Qgc3A8U3VyZmFjZUZsaW5nZXI+JiBmbGluZ2VyKTsKLSAgICAgICAgdmlydHVhbCB+RGlzcGxheUV2ZW50VGhyZWFkKCk7Ci0gICAgICAgIHZpcnR1YWwgYm9vbCB0aHJlYWRMb29wKCk7Ci0gICAgICAgIHZpcnR1YWwgc3RhdHVzX3QgcmVhZHlUb1J1bigpOwotICAgICAgICB2aXJ0dWFsIHN0YXR1c190IHJlbGVhc2VTY3JlZW4oKSBjb25zdDsKLSAgICAgICAgdmlydHVhbCBzdGF0dXNfdCBpbml0Q2hlY2soKSBjb25zdDsKLSAgICB9OwotCi0gICAgY2xhc3MgQ29uc29sZU1hbmFnZXJUaHJlYWQgOiBwdWJsaWMgRGlzcGxheUV2ZW50VGhyZWFkQmFzZSAKLSAgICB7Ci0gICAgICAgIGludCBjb25zb2xlRmQ7Ci0gICAgICAgIGludCB2dF9udW07Ci0gICAgICAgIGludCBwcmV2X3Z0X251bTsKLSAgICAgICAgdnRfbW9kZSB2bTsKLSAgICAgICAgc3RhdGljIHZvaWQgc2lnSGFuZGxlcihpbnQgc2lnKTsKLSAgICAgICAgc3RhdGljIHBpZF90IHNTaWduYWxDYXRjaGVyUGlkOwotICAgIHB1YmxpYzoKLSAgICAgICAgICAgICAgICBDb25zb2xlTWFuYWdlclRocmVhZChjb25zdCBzcDxTdXJmYWNlRmxpbmdlcj4mIGZsaW5nZXIpOwotICAgICAgICB2aXJ0dWFsIH5Db25zb2xlTWFuYWdlclRocmVhZCgpOwotICAgICAgICB2aXJ0dWFsIGJvb2wgdGhyZWFkTG9vcCgpOwotICAgICAgICB2aXJ0dWFsIHN0YXR1c190IHJlYWR5VG9SdW4oKTsKLSAgICAgICAgdmlydHVhbCB2b2lkIHJlcXVlc3RFeGl0KCk7Ci0gICAgICAgIHZpcnR1YWwgc3RhdHVzX3QgcmVsZWFzZVNjcmVlbigpIGNvbnN0OwotICAgICAgICB2aXJ0dWFsIHN0YXR1c190IGluaXRDaGVjaygpIGNvbnN0OwotICAgIH07Ci0KLSAgICBzcDxEaXNwbGF5RXZlbnRUaHJlYWRCYXNlPiAgbURpc3BsYXlFdmVudFRocmVhZDsKLSAgICBtdXRhYmxlIGludCAgICAgICAgICAgICAgICAgbUNhbkRyYXc7Ci19OwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9ESVNQTEFZX0hBUkRXQVJFX0JBU0VfSApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9HUFVIYXJkd2FyZS9HUFVIYXJkd2FyZS5jcHAgYi9saWJzL3N1cmZhY2VmbGluZ2VyL0dQVUhhcmR3YXJlL0dQVUhhcmR3YXJlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZWI3NWY5OS4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0dQVUhhcmR3YXJlL0dQVUhhcmR3YXJlLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDU4MSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJTdXJmYWNlRmxpbmdlciIKLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8ZmNudGwuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPG1hdGguaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDxzeXMvc3RhdC5oPgotI2luY2x1ZGUgPHN5cy9pb2N0bC5oPgotCi0jaW5jbHVkZSA8Y3V0aWxzL2xvZy5oPgotI2luY2x1ZGUgPGN1dGlscy9wcm9wZXJ0aWVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9JQmluZGVyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5RGVhbGVyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5QmFzZS5oPgotI2luY2x1ZGUgPHV0aWxzL01lbW9yeUhlYXBQbWVtLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5SGVhcEJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL1N0b3BXYXRjaC5oPgotCi0jaW5jbHVkZSA8dWkvSVN1cmZhY2VDb21wb3Nlci5oPgotCi0jaW5jbHVkZSAiVlJhbUhlYXAuaCIKLSNpbmNsdWRlICJHUFVIYXJkd2FyZS5oIgotCi0jaWYgSEFWRV9BTkRST0lEX09TCi0jaW5jbHVkZSA8bGludXgvYW5kcm9pZF9wbWVtLmg+Ci0jZW5kaWYKLQotI2luY2x1ZGUgIkdQVUhhcmR3YXJlL0dQVUhhcmR3YXJlLmgiCi0KLQotLyogCi0gKiBNYW5hZ2UgdGhlIEdQVS4gVGhpcyBpbXBsZW1lbnRhdGlvbiBpcyB2ZXJ5IHNwZWNpZmljIHRvIHRoZSBHMS4KLSAqIFRoZXJlIGFyZSBubyBhYnN0cmFjdGlvbiBoZXJlLiAKLSAqIAotICogQWxsIHRoaXMgY29kZSB3aWxsIHNvb24gZ28tYXdheSBhbmQgYmUgcmVwbGFjZWQgYnkgYSBuZXcgYXJjaGl0ZWN0dXJlCi0gKiBmb3IgbWFuYWdpbmcgZ3JhcGhpY3MgYWNjZWxlcmF0b3JzLgotICogCi0gKiBJbiB0aGUgbWVhbnRpbWUsIGl0IGlzIGNvbmNlcHR1YWxseSBwb3NzaWJsZSB0byBpbnN0YW50aWF0ZSBhCi0gKiBHUFVIYXJkd2FyZUludGVyZmFjZSBmb3IgYW5vdGhlciBHUFUgKHNlZSBHUFVGYWN0b3J5IGF0IHRoZSBib3R0b20KLSAqIG9mIHRoaXMgZmlsZSk7IHByYWN0aWNhbGx5Li4uIGRvdWJ0ZnVsLgotICogCi0gKi8KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgR1BVQ2xpZW50SGVhcDsKLWNsYXNzIEdQVUFyZWFIZWFwOwotCi1jbGFzcyBHUFVIYXJkd2FyZSA6IHB1YmxpYyBHUFVIYXJkd2FyZUludGVyZmFjZSwgcHVibGljIElCaW5kZXI6OkRlYXRoUmVjaXBpZW50Ci17Ci1wdWJsaWM6Ci0gICAgc3RhdGljIGNvbnN0IGludCBHUFVfUkVTRVJWRURfU0laRTsKLSAgICBzdGF0aWMgY29uc3QgaW50IEdQVVJfU0laRTsKLQotICAgICAgICAgICAgR1BVSGFyZHdhcmUoKTsKLSAgICB2aXJ0dWFsIH5HUFVIYXJkd2FyZSgpOwotICAgIAotICAgIHZpcnR1YWwgdm9pZCByZXZva2UoaW50IHBpZCk7Ci0gICAgdmlydHVhbCBzcDxNZW1vcnlEZWFsZXI+IHJlcXVlc3QoaW50IHBpZCk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCByZXF1ZXN0KGludCBwaWQsIAotICAgICAgICAgICAgY29uc3Qgc3A8SUdQVUNhbGxiYWNrPiYgY2FsbGJhY2ssCi0gICAgICAgICAgICBJU3VyZmFjZUNvbXBvc2VyOjpncHVfaW5mb190KiBncHUpOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCBmcmllbmRseVJldm9rZSgpOwotICAgIHZpcnR1YWwgdm9pZCB1bmNvbmRpdGlvbmFsUmV2b2tlKCk7Ci0gICAgCi0gICAgdmlydHVhbCBwaWRfdCBnZXRPd25lcigpIGNvbnN0IHsgcmV0dXJuIG1Pd25lcjsgfQotCi0gICAgLy8gdXNlZCBmb3IgZGVidWdnaW5nIG9ubHkuLi4KLSAgICB2aXJ0dWFsIHNwPFNpbXBsZUJlc3RGaXRBbGxvY2F0b3I+IGdldEFsbG9jYXRvcigpIGNvbnN0OwotCi1wcml2YXRlOgotICAgIAotICAgIAotICAgIGVudW0gewotICAgICAgICBOT19PV05FUiA9IC0xLAotICAgIH07Ci0gICAgICAgIAotICAgIHN0cnVjdCBHUFVBcmVhIHsKLSAgICAgICAgc3A8R1BVQXJlYUhlYXA+ICAgICBoZWFwOwotICAgICAgICBzcDxNZW1vcnlIZWFwUG1lbT4gIGNsaWVudEhlYXA7Ci0gICAgICAgIHNwPElNZW1vcnk+IG1hcCgpOwotICAgIH07Ci0gICAgCi0gICAgc3RydWN0IENsaWVudCB7Ci0gICAgICAgIHBpZF90ICAgICAgIHBpZDsKLSAgICAgICAgR1BVQXJlYSAgICAgc21pOwotICAgICAgICBHUFVBcmVhICAgICBlYmk7Ci0gICAgICAgIEdQVUFyZWEgICAgIHJlZzsKLSAgICAgICAgdm9pZCBjcmVhdGVDbGllbnRIZWFwcygpOwotICAgICAgICB2b2lkIHJldm9rZUFsbEhlYXBzKCk7Ci0gICAgfTsKLSAgICAKLSAgICBDbGllbnQmIGdldENsaWVudExvY2tlZChwaWRfdCBwaWQpOwotICAgIHN0YXR1c190IHJlcXVlc3RMb2NrZWQoaW50IHBpZCk7Ci0gICAgdm9pZCByZWxlYXNlTG9ja2VkKCk7Ci0gICAgdm9pZCB0YWtlQmFja0dQVUxvY2tlZCgpOwotICAgIHZvaWQgcmVnaXN0ZXJDYWxsYmFja0xvY2tlZChjb25zdCBzcDxJR1BVQ2FsbGJhY2s+JiBjYWxsYmFjaywKLSAgICAgICAgICAgIENsaWVudCYgY2xpZW50KTsKLQotICAgIHZpcnR1YWwgdm9pZCBiaW5kZXJEaWVkKGNvbnN0IHdwPElCaW5kZXI+JiB3aG8pOwotCi0gICAgbXV0YWJsZSBNdXRleCAgICAgICAgICAgbUxvY2s7Ci0gICAgc3A8R1BVQXJlYUhlYXA+ICAgICAgICAgbVNNSUhlYXA7Ci0gICAgc3A8R1BVQXJlYUhlYXA+ICAgICAgICAgbUVCSUhlYXA7Ci0gICAgc3A8R1BVQXJlYUhlYXA+ICAgICAgICAgbVJFR0hlYXA7Ci0KLSAgICBLZXllZFZlY3RvcjxwaWRfdCwgQ2xpZW50PiBtQ2xpZW50czsKLSAgICBEZWZhdWx0S2V5ZWRWZWN0b3I8IHdwPElCaW5kZXI+LCBwaWRfdCA+IG1SZWdpc3RlcmVkQ2xpZW50czsKLSAgICAKLSAgICBwaWRfdCAgICAgICAgICAgICAgICAgICBtT3duZXI7Ci0KLSAgICBzcDxNZW1vcnlEZWFsZXI+ICAgICAgICBtQ3VycmVudEFsbG9jYXRvcjsKLSAgICBzcDxJR1BVQ2FsbGJhY2s+ICAgICAgICBtQ2FsbGJhY2s7Ci0gICAgCi0gICAgc3A8U2ltcGxlQmVzdEZpdEFsbG9jYXRvcj4gIG1BbGxvY2F0b3I7Ci0KLSAgICBDb25kaXRpb24gICAgICAgICAgICAgICBtQ29uZGl0aW9uOwotfTsKLQotLy8gc2l6ZSByZXNlcnZlZCBmb3IgR1BVIHN1cmZhY2VzCi0vLyAxMjAwIEtCIGZpdHMgZXhhY3RseToKLS8vICAtIHR3byAzMjAqNDgwIDE2LWJpdHMgZG91YmxlLWJ1ZmZlcmVkIHN1cmZhY2VzCi0vLyAgLSBvbmUgMzIwKjQ4MCAzMi1iaXRzIGRvdWJsZS1idWZmZXJlZCBzdXJmYWNlCi0vLyAgLSBvbmUgMzIwKjI0MCAxNi1iaXRzIGRvdWJsZS1idWZmZXJlZCwgNHggYW50aS1hbGlhc2VkIHN1cmZhY2UKLWNvbnN0IGludCBHUFVIYXJkd2FyZTo6R1BVX1JFU0VSVkVEX1NJWkUgID0gMTIwMCAqIDEwMjQ7Ci1jb25zdCBpbnQgR1BVSGFyZHdhcmU6OkdQVVJfU0laRSAgICAgICAgICA9IDEgKiAxMDI0ICogMTAyNDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLS8qIAotICogR1BVSGFuZGxlIGlzIGEgc3BlY2lhbCBJTWVtb3J5IGdpdmVuIHRvIHRoZSBjbGllbnQuIEl0IHJlcHJlc2VudHMgdGhlaXIKLSAqIGhhbmRsZSB0byB0aGUgR1BVLiBPbmNlIHRoZXkgZ2l2ZSBpdCB1cCwgdGhleSBsb29zZSBHUFUgYWNjZXNzLCBvciBpZgotICogdGhleSBleHBsaWNpdGx5IHJldm9rZSB0aGVpciBhY2Nlc3MgdGhyb3VnaCB0aGUgYmluZGVyIGNvZGUgMTAwMC4KLSAqIEluIGJvdGggY2FzZXMsIHRoaXMgdHJpZ2dlcnMgYSBjYWxsYmFjayB0byByZXZva2UoKQotICogZmlyc3QsIGFuZCB0aGVuIGFjdHVhbGx5IHBvd2VycyBkb3duIHRoZSBjaGlwLgotICogCi0gKiBJbiB0aGUgY2FzZSBvZiBhIG1pc2JlaGF2aW5nIGFwcCwgR1BVSGFyZHdhcmUgY2FuIGFzayBmb3IgYW4gaW1tZWRpYXRlCi0gKiByZWxlYXNlIG9mIHRoZSBHUFUgdG8gdGhlIHRhcmdldCBwcm9jZXNzIHdoaWNoIHNob3VsZCBhbnN3ZXIgYnkgY2FsbGluZwotICogY29kZSAxMDAwIG9uIEdQVUhhbmRsZS4gSWYgaXQgZG9lc24ndCBpbiBhIHRpbWVseSBtYW5uZXIsIHRoZSBHUFUgd2lsbAotICogYmUgcmV2b2tlZCBmcm9tIHVuZGVyIHRoZWlyIGZlZXQuCi0gKiAKLSAqIFdlIHNob3VsZCBuZXZlciBob2xkIGEgc3Ryb25nIHJlZmVyZW5jZSBvbiBHUFVIYW5kbGUuIEluIHByYWN0aWNlIHRoaXMKLSAqIHNob3VsZG4ndCBiZSBhIGJpZyBpc3N1ZSB0aG91Z2ggYmVjYXVzZSBjbGllbnRzIHNob3VsZCB1c2UgY29kZSAxMDAwIGFuZAotICogbm90IHJlbHkgb24gdGhlIGR0b3IgYmVpbmcgY2FsbGVkLgotICogCi0gKi8KLQotY2xhc3MgR1BVQ2xpZW50SGVhcCA6IHB1YmxpYyBNZW1vcnlIZWFwUG1lbQotewotcHVibGljOgotICAgIEdQVUNsaWVudEhlYXAoY29uc3Qgd3A8R1BVSGFyZHdhcmU+JiBncHUsIAotICAgICAgICAgICAgY29uc3Qgc3A8TWVtb3J5SGVhcEJhc2U+JiBoZWFwKQotICAgICAgICA6ICBNZW1vcnlIZWFwUG1lbShoZWFwKSwgbUdQVShncHUpIHsgfQotcHJvdGVjdGVkOgotICAgIHdwPEdQVUhhcmR3YXJlPiBtR1BVOwotfTsKLQotY2xhc3MgR1BVQXJlYUhlYXAgOiBwdWJsaWMgTWVtb3J5SGVhcEJhc2UKLXsKLXB1YmxpYzoKLSAgICBHUFVBcmVhSGVhcChjb25zdCB3cDxHUFVIYXJkd2FyZT4mIGdwdSwKLSAgICAgICAgICAgIGNvbnN0IGNoYXIqIGNvbnN0IHZyYW0sIHNpemVfdCBzaXplPTAsIHNpemVfdCByZXNlcnZlZD0wKQotICAgIDogTWVtb3J5SGVhcEJhc2UodnJhbSwgc2l6ZSksIG1HUFUoZ3B1KSB7IAotICAgICAgICBpZiAoYmFzZSgpICE9IE1BUF9GQUlMRUQpIHsKLSAgICAgICAgICAgIGlmIChyZXNlcnZlZCA9PSAwKQotICAgICAgICAgICAgICAgIHJlc2VydmVkID0gdmlydHVhbFNpemUoKTsKLSAgICAgICAgICAgIG1BbGxvY2F0b3IgPSBuZXcgU2ltcGxlQmVzdEZpdEFsbG9jYXRvcihyZXNlcnZlZCk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgdmlydHVhbCBzcDxNZW1vcnlIZWFwUG1lbT4gY3JlYXRlQ2xpZW50SGVhcCgpIHsKLSAgICAgICAgc3A8TWVtb3J5SGVhcEJhc2U+IHBhcmVudEhlYXAodGhpcyk7Ci0gICAgICAgIHJldHVybiBuZXcgR1BVQ2xpZW50SGVhcChtR1BVLCBwYXJlbnRIZWFwKTsKLSAgICB9Ci0gICAgdmlydHVhbCBjb25zdCBzcDxTaW1wbGVCZXN0Rml0QWxsb2NhdG9yPiYgZ2V0QWxsb2NhdG9yKCkgY29uc3QgewotICAgICAgICByZXR1cm4gbUFsbG9jYXRvcjsgCi0gICAgfQotcHJpdmF0ZToKLSAgICBzcDxTaW1wbGVCZXN0Rml0QWxsb2NhdG9yPiAgbUFsbG9jYXRvcjsKLXByb3RlY3RlZDoKLSAgICB3cDxHUFVIYXJkd2FyZT4gbUdQVTsKLX07Ci0KLWNsYXNzIEdQVVJlZ2lzdGVySGVhcCA6IHB1YmxpYyBHUFVBcmVhSGVhcAotewotcHVibGljOgotICAgIEdQVVJlZ2lzdGVySGVhcChjb25zdCBzcDxHUFVIYXJkd2FyZT4mIGdwdSkKLSAgICAgICAgOiBHUFVBcmVhSGVhcChncHUsICIvZGV2L2h3M2QiLCBHUFVIYXJkd2FyZTo6R1BVUl9TSVpFKSB7IH0KLSAgICB2aXJ0dWFsIHNwPE1lbW9yeUhlYXBQbWVtPiBjcmVhdGVDbGllbnRIZWFwKCkgewotICAgICAgICBzcDxNZW1vcnlIZWFwQmFzZT4gcGFyZW50SGVhcCh0aGlzKTsKLSAgICAgICAgcmV0dXJuIG5ldyBNZW1vcnlIZWFwUmVncyhtR1BVLCBwYXJlbnRIZWFwKTsKLSAgICB9Ci1wcml2YXRlOgotICAgIGNsYXNzIE1lbW9yeUhlYXBSZWdzIDogcHVibGljIEdQVUNsaWVudEhlYXAgIHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIE1lbW9yeUhlYXBSZWdzKGNvbnN0IHdwPEdQVUhhcmR3YXJlPiYgZ3B1LCAKLSAgICAgICAgICAgICBjb25zdCBzcDxNZW1vcnlIZWFwQmFzZT4mIGhlYXApCi0gICAgICAgICAgICA6IEdQVUNsaWVudEhlYXAoZ3B1LCBoZWFwKSB7IH0KLSAgICAgICAgc3A8TWVtb3J5SGVhcFBtZW06Ok1lbW9yeVBtZW0+IGNyZWF0ZU1lbW9yeShzaXplX3Qgb2Zmc2V0LCBzaXplX3Qgc2l6ZSk7Ci0gICAgICAgIHZpcnR1YWwgdm9pZCByZXZva2UoKTsKLSAgICBwcml2YXRlOgotICAgICAgICBjbGFzcyBHUFVIYW5kbGUgOiBwdWJsaWMgTWVtb3J5SGVhcFBtZW06Ok1lbW9yeVBtZW0gewotICAgICAgICBwdWJsaWM6Ci0gICAgICAgICAgICBHUFVIYW5kbGUoY29uc3Qgc3A8R1BVSGFyZHdhcmU+JiBncHUsCi0gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNwPE1lbW9yeUhlYXBQbWVtPiYgaGVhcCkKLSAgICAgICAgICAgICAgICA6IE1lbW9yeUhlYXBQbWVtOjpNZW1vcnlQbWVtKGhlYXApLCAKLSAgICAgICAgICAgICAgICAgIG1HUFUoZ3B1KSwgbU93bmVyKGdwdS0+Z2V0T3duZXIoKSkgeyB9Ci0gICAgICAgICAgICB2aXJ0dWFsIH5HUFVIYW5kbGUoKTsKLSAgICAgICAgICAgIHZpcnR1YWwgc3A8SU1lbW9yeUhlYXA+IGdldE1lbW9yeSgKLSAgICAgICAgICAgICAgICAgICAgc3NpemVfdCogb2Zmc2V0LCBzaXplX3QqIHNpemUpIGNvbnN0OwotICAgICAgICAgICAgdmlydHVhbCB2b2lkIHJldm9rZSgpIHsgfTsKLSAgICAgICAgICAgIHZpcnR1YWwgc3RhdHVzX3Qgb25UcmFuc2FjdCgKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCAKLSAgICAgICAgICAgICAgICAgICAgUGFyY2VsKiByZXBseSwgdWludDMyX3QgZmxhZ3MpOwotICAgICAgICBwcml2YXRlOgotICAgICAgICAgICAgdm9pZCByZXZva2VOb3RpZmljYXRpb24oKTsKLSAgICAgICAgICAgIHdwPEdQVUhhcmR3YXJlPiBtR1BVOwotICAgICAgICAgICAgcGlkX3QgbU93bmVyOwotICAgICAgICB9OwotICAgIH07Ci19OwotCi1HUFVSZWdpc3RlckhlYXA6Ok1lbW9yeUhlYXBSZWdzOjpHUFVIYW5kbGU6On5HUFVIYW5kbGUoKSB7IAotICAgIC8vTE9HRCgiR1BVSGFuZGxlICVwIHJlbGVhc2VkLCByZXZva2luZyBHUFUiLCB0aGlzKTsKLSAgICByZXZva2VOb3RpZmljYXRpb24oKTsgCi19Ci12b2lkIEdQVVJlZ2lzdGVySGVhcDo6TWVtb3J5SGVhcFJlZ3M6OkdQVUhhbmRsZTo6cmV2b2tlTm90aWZpY2F0aW9uKCkgIHsKLSAgICBzcDxHUFVIYXJkd2FyZT4gaHcobUdQVS5wcm9tb3RlKCkpOwotICAgIGlmIChodyAhPSAwKSB7Ci0gICAgICAgIGh3LT5yZXZva2UobU93bmVyKTsKLSAgICB9Ci19Ci1zcDxJTWVtb3J5SGVhcD4gR1BVUmVnaXN0ZXJIZWFwOjpNZW1vcnlIZWFwUmVnczo6R1BVSGFuZGxlOjpnZXRNZW1vcnkoCi0gICAgICAgIHNzaXplX3QqIG9mZnNldCwgc2l6ZV90KiBzaXplKSBjb25zdAotewotICAgIHNwPE1lbW9yeUhlYXBQbWVtPiBoZWFwID0gZ2V0SGVhcCgpOwotICAgIGlmIChvZmZzZXQpICpvZmZzZXQgPSAwOwotICAgIGlmIChzaXplKSAgICpzaXplID0gaGVhcCAhPTAgPyBoZWFwLT52aXJ0dWFsU2l6ZSgpIDogMDsKLSAgICByZXR1cm4gaGVhcDsKLX0KLXN0YXR1c190IEdQVVJlZ2lzdGVySGVhcDo6TWVtb3J5SGVhcFJlZ3M6OkdQVUhhbmRsZTo6b25UcmFuc2FjdCgKLSAgICAgICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBzdGF0dXNfdCBlcnIgPSBCbk1lbW9yeTo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotICAgIGlmIChlcnIgPT0gVU5LTk9XTl9UUkFOU0FDVElPTiAmJiBjb2RlID09IDEwMDApIHsKLSAgICAgICAgaW50IGNhbGxpbmdQaWQgPSBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5nZXRDYWxsaW5nUGlkKCk7Ci0gICAgICAgIC8vTE9HRCgicGlkICVkIHZvbHVudGFyaWx5IHJldm9raW5nIGdwdSIsIGNhbGxpbmdQaWQpOwotICAgICAgICBpZiAoY2FsbGluZ1BpZCA9PSBtT3duZXIpIHsKLSAgICAgICAgICAgIHJldm9rZU5vdGlmaWNhdGlvbigpOwotICAgICAgICAgICAgLy8gd2UndmUgcmV2b2tlZCB0aGUgR1BVLCBkb24ndCBkbyBpdCBhZ2FpbiBsYXRlciB3aGVuIHdlCi0gICAgICAgICAgICAvLyBhcmUgZGVzdHJveWVkLgotICAgICAgICAgICAgbUdQVS5jbGVhcigpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgTE9HVygiJWQgcmV2b2tpbmcgc29tZW9uZSBlbHNlJ3MgZ3B1PyAob3duZXI9JWQpIiwKLSAgICAgICAgICAgICAgICAgICAgY2FsbGluZ1BpZCwgbU93bmVyKTsgICAgICAgICAgICAKLSAgICAgICAgfQotICAgICAgICBlcnIgPSBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLQotc3A8TWVtb3J5SGVhcFBtZW06Ok1lbW9yeVBtZW0+IEdQVVJlZ2lzdGVySGVhcDo6TWVtb3J5SGVhcFJlZ3M6OmNyZWF0ZU1lbW9yeSgKLSAgICAgICAgc2l6ZV90IG9mZnNldCwgc2l6ZV90IHNpemUpCi17Ci0gICAgc3A8R1BVSGFuZGxlPiBtZW1vcnk7Ci0gICAgc3A8R1BVSGFyZHdhcmU+IGdwdSA9IG1HUFUucHJvbW90ZSgpOwotICAgIGlmIChoZWFwSUQoKT4wICYmIGdwdSE9MCkgewotI2lmIEhBVkVfQU5EUk9JRF9PUwotICAgICAgICAvKiB0aGlzIGlzIHdoZXJlIHRoZSBHUFUgaXMgcG93ZXJlZCBvbiBhbmQgdGhlIHJlZ2lzdGVycyBhcmUgbWFwcGVkCi0gICAgICAgICAqIGluIHRoZSBjbGllbnQgKi8KLSAgICAgICAgLy9MT0dEKCJpb2N0bChIVzNEX0dSQU5UX0dQVSkiKTsKLSAgICAgICAgaW50IGVyciA9IGlvY3RsKGhlYXBJRCgpLCBIVzNEX0dSQU5UX0dQVSwgYmFzZSgpKTsKLSAgICAgICAgaWYgKGVycikgewotICAgICAgICAgICAgLy8gaXQgY2FuIGhhcHBlbiBpZiB0aGUgbWFzdGVyIGhlYXAgaGFzIGJlZW4gY2xvc2VkIGFscmVhZHkKLSAgICAgICAgICAgIC8vIGluIHdoaWNoIGNhc2UgdGhlIEdQVSBhbHJlYWR5IGlzIHJldm9rZWQgKGFwcCBjcmFzaCBmb3IKLSAgICAgICAgICAgIC8vIGluc3RhbmNlKS4KLSAgICAgICAgICAgIExPR1coIkhXM0RfR1JBTlRfR1BVIGZhaWxlZCAoJXMpLCBtRkQ9JWQsIGJhc2U9JXAiLAotICAgICAgICAgICAgICAgICAgICBzdHJlcnJvcihlcnJubyksIGhlYXBJRCgpLCBiYXNlKCkpOwotICAgICAgICB9Ci0gICAgICAgIG1lbW9yeSA9IG5ldyBHUFVIYW5kbGUoZ3B1LCB0aGlzKTsKLSNlbmRpZgotICAgIH0KLSAgICByZXR1cm4gbWVtb3J5OwotfQotCi12b2lkIEdQVVJlZ2lzdGVySGVhcDo6TWVtb3J5SGVhcFJlZ3M6OnJldm9rZSgpIAotewotICAgIE1lbW9yeUhlYXBQbWVtOjpyZXZva2UoKTsKLSNpZiBIQVZFX0FORFJPSURfT1MKLSAgICBpZiAoaGVhcElEKCkgPiAwKSB7Ci0gICAgICAgIC8vTE9HRCgiaW9jdGwoSFczRF9SRVZPS0VfR1BVKSIpOwotICAgICAgICBpbnQgZXJyID0gaW9jdGwoaGVhcElEKCksIEhXM0RfUkVWT0tFX0dQVSwgYmFzZSgpKTsKLSAgICAgICAgTE9HRV9JRihlcnIsICJIVzNEX1JFVk9LRV9HUFUgZmFpbGVkICglcyksIG1GRD0lZCwgYmFzZT0lcCIsCi0gICAgICAgICAgICAgICAgc3RyZXJyb3IoZXJybm8pLCBoZWFwSUQoKSwgYmFzZSgpKTsKLSAgICB9Ci0jZW5kaWYKLX0KLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi1HUFVIYXJkd2FyZTo6R1BVSGFyZHdhcmUoKQotICAgIDogbU93bmVyKE5PX09XTkVSKQotewotfQotCi1HUFVIYXJkd2FyZTo6fkdQVUhhcmR3YXJlKCkKLXsKLX0KLQotc3RhdHVzX3QgR1BVSGFyZHdhcmU6OnJlcXVlc3RMb2NrZWQoaW50IHBpZCkKLXsKLSAgICBjb25zdCBpbnQgc2VsZl9waWQgPSBnZXRwaWQoKTsKLSAgICBpZiAocGlkID09IHNlbGZfcGlkKSB7Ci0gICAgICAgIC8vIGNhbid0IHVzZSBHUFUgZnJvbSBzdXJmYWNlZmxpbmdlcidzIHByb2Nlc3MKLSAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOwotICAgIH0KLQotICAgIGlmIChtT3duZXIgIT0gcGlkKSB7Ci0gICAgICAgIGlmIChtUkVHSGVhcCAhPSAwKSB7Ci0gICAgICAgICAgICBpZiAobU93bmVyICE9IE5PX09XTkVSKSB7Ci0gICAgICAgICAgICAgICAgLy8gc29tZW9uZSBhbHJlYWR5IGhhcyB0aGUgZ3B1LgotICAgICAgICAgICAgICAgIHRha2VCYWNrR1BVTG9ja2VkKCk7Ci0gICAgICAgICAgICAgICAgcmVsZWFzZUxvY2tlZCgpOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLy8gZmlyc3QgdGltZSwgaW5pdGlhbGl6ZSB0aGUgc3R1ZmYuCi0gICAgICAgICAgICBpZiAobVNNSUhlYXAgPT0gMCkKLSAgICAgICAgICAgICAgICBtU01JSGVhcCA9IG5ldyBHUFVBcmVhSGVhcCh0aGlzLCAiL2Rldi9wbWVtX2dwdTAiKTsKLSAgICAgICAgICAgIGlmIChtRUJJSGVhcCA9PSAwKQotICAgICAgICAgICAgICAgIG1FQklIZWFwID0gbmV3IEdQVUFyZWFIZWFwKHRoaXMsIAotICAgICAgICAgICAgICAgICAgICAgICAgIi9kZXYvcG1lbV9ncHUxIiwgMCwgR1BVX1JFU0VSVkVEX1NJWkUpOwotICAgICAgICAgICAgbVJFR0hlYXAgPSBuZXcgR1BVUmVnaXN0ZXJIZWFwKHRoaXMpOwotICAgICAgICAgICAgbUFsbG9jYXRvciA9IG1FQklIZWFwLT5nZXRBbGxvY2F0b3IoKTsKLSAgICAgICAgICAgIGlmIChtQWxsb2NhdG9yID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAvLyBzb21ldGhpbmcgd2VudCB0ZXJyaWJseSB3cm9uZy4KLSAgICAgICAgICAgICAgICBtU01JSGVhcC5jbGVhcigpOwotICAgICAgICAgICAgICAgIG1FQklIZWFwLmNsZWFyKCk7Ci0gICAgICAgICAgICAgICAgbVJFR0hlYXAuY2xlYXIoKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgQ2xpZW50JiBjbGllbnQgPSBnZXRDbGllbnRMb2NrZWQocGlkKTsKLSAgICAgICAgbUN1cnJlbnRBbGxvY2F0b3IgPSBuZXcgTWVtb3J5RGVhbGVyKGNsaWVudC5lYmkuY2xpZW50SGVhcCwgbUFsbG9jYXRvcik7Ci0gICAgICAgIG1Pd25lciA9IHBpZDsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zcDxNZW1vcnlEZWFsZXI+IEdQVUhhcmR3YXJlOjpyZXF1ZXN0KGludCBwaWQpCi17Ci0gICAgc3A8TWVtb3J5RGVhbGVyPiBkZWFsZXI7Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICBDbGllbnQqIGNsaWVudDsKLSAgICBMT0dEKCJwaWQgJWQgcmVxdWVzdGluZyBncHUgc3VyZmFjZSAoY3VycmVudCBvd25lciA9ICVkKSIsIHBpZCwgbU93bmVyKTsKLSAgICBpZiAocmVxdWVzdExvY2tlZChwaWQpID09IE5PX0VSUk9SKSB7Ci0gICAgICAgIGRlYWxlciA9IG1DdXJyZW50QWxsb2NhdG9yOwotICAgICAgICBMT0dEX0lGKGRlYWxlciE9MCwgImdwdSBzdXJmYWNlIGdyYW50ZWQgdG8gcGlkICVkIiwgbU93bmVyKTsKLSAgICB9Ci0gICAgcmV0dXJuIGRlYWxlcjsKLX0KLQotc3RhdHVzX3QgR1BVSGFyZHdhcmU6OnJlcXVlc3QoaW50IHBpZCwgY29uc3Qgc3A8SUdQVUNhbGxiYWNrPiYgY2FsbGJhY2ssCi0gICAgICAgIElTdXJmYWNlQ29tcG9zZXI6OmdwdV9pbmZvX3QqIGdwdSkKLXsKLSAgICBpZiAoY2FsbGJhY2sgPT0gMCkKLSAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLQotICAgIHNwPElNZW1vcnk+IGdwdUhhbmRsZTsKLSAgICBMT0dEKCJwaWQgJWQgcmVxdWVzdGluZyBncHUgY29yZSAob3duZXIgPSAlZCkiLCBwaWQsIG1Pd25lcik7Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICBzdGF0dXNfdCBlcnIgPSByZXF1ZXN0TG9ja2VkKHBpZCk7Ci0gICAgaWYgKGVyciA9PSBOT19FUlJPUikgewotICAgICAgICAvLyBpdCdzIGd1YXJhbnRlZWQgdG8gYmUgdGhlcmUsIGJlIGNvbnN0cnVjdGlvbgotICAgICAgICBDbGllbnQmIGNsaWVudCA9IG1DbGllbnRzLmVkaXRWYWx1ZUZvcihwaWQpOwotICAgICAgICByZWdpc3RlckNhbGxiYWNrTG9ja2VkKGNhbGxiYWNrLCBjbGllbnQpOwotICAgICAgICBncHUtPmNvdW50ID0gMjsKLSAgICAgICAgZ3B1LT5yZWdpb25zWzBdLnJlZ2lvbiA9IGNsaWVudC5zbWkubWFwKCk7Ci0gICAgICAgIGdwdS0+cmVnaW9uc1sxXS5yZWdpb24gPSBjbGllbnQuZWJpLm1hcCgpOwotICAgICAgICBncHUtPnJlZ3MgICAgICAgICAgICAgID0gY2xpZW50LnJlZy5tYXAoKTsKLSAgICAgICAgZ3B1LT5yZWdpb25zWzBdLnJlc2VydmVkID0gMDsKLSAgICAgICAgZ3B1LT5yZWdpb25zWzFdLnJlc2VydmVkID0gR1BVX1JFU0VSVkVEX1NJWkU7Ci0gICAgICAgIGlmIChncHUtPnJlZ3MgIT0gMCkgewotICAgICAgICAgICAgLy9MT0dEKCJncHUgY29yZSBncmFudGVkIHRvIHBpZCAlZCwgaGFuZGxlIGJhc2U9JXAiLAotICAgICAgICAgICAgLy8gICAgICAgIG1Pd25lciwgZ3B1LT5yZWdzLT5wb2ludGVyKCkpOwotICAgICAgICB9Ci0gICAgICAgIG1DYWxsYmFjayA9IGNhbGxiYWNrOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIExPR1coImNvdWxkbid0IGdyYW50IGdwdSBjb3JlIHRvIHBpZCAlZCIsIHBpZCk7Ci0gICAgfQotICAgIHJldHVybiBlcnI7Ci19Ci0KLXZvaWQgR1BVSGFyZHdhcmU6OnJldm9rZShpbnQgcGlkKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgaWYgKG1Pd25lciA+IDApIHsKLSAgICAgICAgaWYgKHBpZCAhPSBtT3duZXIpIHsKLSAgICAgICAgICAgIExPR1coIkdQVSBvd25lZCBieSAlZCwgcmV2b2tlIGZyb20gJWQiLCBtT3duZXIsIHBpZCk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgLy9MT0dEKCJyZXZva2UgcGlkPSVkLCBvd25lcj0lZCIsIHBpZCwgbU93bmVyKTsKLSAgICAgICAgLy8gbU93bmVyIGNvdWxkIGJlIDwwIGlmIHRoZSBzYW1lIHByb2Nlc3MgYWNxdWlyZWQgdGhlIEdQVQotICAgICAgICAvLyBzZXZlcmFsIHRpbWVzIHdpdGhvdXQgcmVsZWFzaW5nIGl0IGZpcnN0LgotICAgICAgICBtQ29uZGl0aW9uLnNpZ25hbCgpOwotICAgICAgICByZWxlYXNlTG9ja2VkKCk7Ci0gICAgfQotfQotCi1zdGF0dXNfdCBHUFVIYXJkd2FyZTo6ZnJpZW5kbHlSZXZva2UoKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgLy9MT0dEKCJmcmllbmRseVJldm9rZSBvd25lcj0lZCIsIG1Pd25lcik7Ci0gICAgdGFrZUJhY2tHUFVMb2NrZWQoKTsKLSAgICByZWxlYXNlTG9ja2VkKCk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi12b2lkIEdQVUhhcmR3YXJlOjp0YWtlQmFja0dQVUxvY2tlZCgpCi17Ci0gICAgc3A8SUdQVUNhbGxiYWNrPiBjYWxsYmFjayA9IG1DYWxsYmFjazsKLSAgICBtQ2FsbGJhY2suY2xlYXIoKTsKLSAgICBpZiAoY2FsbGJhY2sgIT0gMCkgewotICAgICAgICBjYWxsYmFjay0+Z3B1TG9zdCgpOyAvLyBvbmUtd2F5Ci0gICAgICAgIG1Db25kaXRpb24ud2FpdFJlbGF0aXZlKG1Mb2NrLCBtczJucygyNTApKTsKLSAgICB9Ci19Ci0KLXZvaWQgR1BVSGFyZHdhcmU6OnJlbGVhc2VMb2NrZWQoKQotewotICAgIC8vTE9HRCgicmV2b2tpbmcgZ3B1IGZyb20gcGlkICVkIiwgbU93bmVyKTsKLSAgICBpZiAobU93bmVyICE9IE5PX09XTkVSKSB7Ci0gICAgICAgIC8vIHRoaXMgbWF5IGZhaWwgYmVjYXVzZSB0aGUgY2xpZW50IG1pZ2h0IGhhdmUgZGllZCwgYW5kIGhhdmUKLSAgICAgICAgLy8gYmVlbiByZW1vdmVkIGZyb20gdGhlIGxpc3QuCi0gICAgICAgIHNzaXplX3QgaW5kZXggPSBtQ2xpZW50cy5pbmRleE9mS2V5KG1Pd25lcik7Ci0gICAgICAgIGlmIChpbmRleCA+PSAwKSB7Ci0gICAgICAgICAgICBDbGllbnQmIGNsaWVudChtQ2xpZW50cy5lZGl0VmFsdWVBdChpbmRleCkpOwotICAgICAgICAgICAgY2xpZW50LnJldm9rZUFsbEhlYXBzKCk7Ci0gICAgICAgIH0KLSAgICAgICAgbU93bmVyID0gTk9fT1dORVI7Ci0gICAgICAgIG1DdXJyZW50QWxsb2NhdG9yLmNsZWFyKCk7Ci0gICAgICAgIG1DYWxsYmFjay5jbGVhcigpOwotICAgIH0KLX0KLQotR1BVSGFyZHdhcmU6OkNsaWVudCYgR1BVSGFyZHdhcmU6OmdldENsaWVudExvY2tlZChwaWRfdCBwaWQpCi17Ci0gICAgc3NpemVfdCBpbmRleCA9IG1DbGllbnRzLmluZGV4T2ZLZXkocGlkKTsKLSAgICBpZiAoaW5kZXggPCAwKSB7Ci0gICAgICAgIENsaWVudCBjbGllbnQ7Ci0gICAgICAgIGNsaWVudC5waWQgPSBwaWQ7Ci0gICAgICAgIGNsaWVudC5zbWkuaGVhcCA9IG1TTUlIZWFwOwotICAgICAgICBjbGllbnQuZWJpLmhlYXAgPSBtRUJJSGVhcDsKLSAgICAgICAgY2xpZW50LnJlZy5oZWFwID0gbVJFR0hlYXA7Ci0gICAgICAgIGluZGV4ID0gbUNsaWVudHMuYWRkKHBpZCwgY2xpZW50KTsKLSAgICB9Ci0gICAgQ2xpZW50JiBjbGllbnQobUNsaWVudHMuZWRpdFZhbHVlQXQoaW5kZXgpKTsKLSAgICBjbGllbnQuY3JlYXRlQ2xpZW50SGVhcHMoKTsKLSAgICByZXR1cm4gY2xpZW50OwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBmb3IgZGVidWdnaW5nIC8gdGVzdGluZyAuLi4KLQotc3A8U2ltcGxlQmVzdEZpdEFsbG9jYXRvcj4gR1BVSGFyZHdhcmU6OmdldEFsbG9jYXRvcigpIGNvbnN0IHsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIHJldHVybiBtQWxsb2NhdG9yOwotfQotCi12b2lkIEdQVUhhcmR3YXJlOjp1bmNvbmRpdGlvbmFsUmV2b2tlKCkKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIHJlbGVhc2VMb2NrZWQoKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXNwPElNZW1vcnk+IEdQVUhhcmR3YXJlOjpHUFVBcmVhOjptYXAoKSB7Ci0gICAgc3A8SU1lbW9yeT4gbWVtb3J5OwotICAgIGlmIChjbGllbnRIZWFwICE9IDAgJiYgaGVhcCAhPSAwKSB7Ci0gICAgICAgIG1lbW9yeSA9IGNsaWVudEhlYXAtPm1hcE1lbW9yeSgwLCBoZWFwLT52aXJ0dWFsU2l6ZSgpKTsKLSAgICB9Ci0gICAgcmV0dXJuIG1lbW9yeTsKLX0KLQotdm9pZCBHUFVIYXJkd2FyZTo6Q2xpZW50OjpjcmVhdGVDbGllbnRIZWFwcygpIAotewotICAgIGlmIChzbWkuY2xpZW50SGVhcCA9PSAwKQotICAgICAgICBzbWkuY2xpZW50SGVhcCA9IHNtaS5oZWFwLT5jcmVhdGVDbGllbnRIZWFwKCk7Ci0gICAgaWYgKGViaS5jbGllbnRIZWFwID09IDApCi0gICAgICAgIGViaS5jbGllbnRIZWFwID0gZWJpLmhlYXAtPmNyZWF0ZUNsaWVudEhlYXAoKTsKLSAgICBpZiAocmVnLmNsaWVudEhlYXAgPT0gMCkKLSAgICAgICAgcmVnLmNsaWVudEhlYXAgPSByZWcuaGVhcC0+Y3JlYXRlQ2xpZW50SGVhcCgpOwotfQotCi12b2lkIEdQVUhhcmR3YXJlOjpDbGllbnQ6OnJldm9rZUFsbEhlYXBzKCkgCi17Ci0gICAgaWYgKHNtaS5jbGllbnRIZWFwICE9IDApCi0gICAgICAgIHNtaS5jbGllbnRIZWFwLT5yZXZva2UoKTsKLSAgICBpZiAoZWJpLmNsaWVudEhlYXAgIT0gMCkKLSAgICAgICAgZWJpLmNsaWVudEhlYXAtPnJldm9rZSgpOwotICAgIGlmIChyZWcuY2xpZW50SGVhcCAhPSAwKQotICAgICAgICByZWcuY2xpZW50SGVhcC0+cmV2b2tlKCk7Ci19Ci0KLXZvaWQgR1BVSGFyZHdhcmU6OnJlZ2lzdGVyQ2FsbGJhY2tMb2NrZWQoY29uc3Qgc3A8SUdQVUNhbGxiYWNrPiYgY2FsbGJhY2ssCi0gICAgICAgIENsaWVudCYgY2xpZW50KQotewotICAgIHNwPElCaW5kZXI+IGJpbmRlciA9IGNhbGxiYWNrLT5hc0JpbmRlcigpOwotICAgIGlmIChtUmVnaXN0ZXJlZENsaWVudHMuYWRkKGJpbmRlciwgY2xpZW50LnBpZCkgPj0gMCkgewotICAgICAgICBiaW5kZXItPmxpbmtUb0RlYXRoKHRoaXMpOwotICAgIH0KLX0KLQotdm9pZCBHUFVIYXJkd2FyZTo6YmluZGVyRGllZChjb25zdCB3cDxJQmluZGVyPiYgd2hvKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgcGlkX3QgcGlkID0gbVJlZ2lzdGVyZWRDbGllbnRzLnZhbHVlRm9yKHdobyk7Ci0gICAgaWYgKHBpZCAhPSAwKSB7Ci0gICAgICAgIHNzaXplX3QgaW5kZXggPSBtQ2xpZW50cy5pbmRleE9mS2V5KHBpZCk7Ci0gICAgICAgIGlmIChpbmRleCA+PSAwKSB7Ci0gICAgICAgICAgICAvL0xPR0QoIioqKiByZW1vdmluZyBjbGllbnQgYXQgJWQiLCBpbmRleCk7Ci0gICAgICAgICAgICBDbGllbnQmIGNsaWVudChtQ2xpZW50cy5lZGl0VmFsdWVBdChpbmRleCkpOwotICAgICAgICAgICAgY2xpZW50LnJldm9rZUFsbEhlYXBzKCk7IC8vIG5vdCByZWFsbHkgbmVlZGVkIGluIHRoZW9yeQotICAgICAgICAgICAgbUNsaWVudHMucmVtb3ZlSXRlbXNBdChpbmRleCk7Ci0gICAgICAgICAgICBpZiAobUNsaWVudHMuc2l6ZSgpID09IDApIHsKLSAgICAgICAgICAgICAgICAvL0xPR0QoIioqKiB3YXMgbGFzdCBjbGllbnQgY2xvc2luZyBldmVyeXRoaW5nIik7Ci0gICAgICAgICAgICAgICAgbUNhbGxiYWNrLmNsZWFyKCk7Ci0gICAgICAgICAgICAgICAgbUFsbG9jYXRvci5jbGVhcigpOwotICAgICAgICAgICAgICAgIG1DdXJyZW50QWxsb2NhdG9yLmNsZWFyKCk7Ci0gICAgICAgICAgICAgICAgbVNNSUhlYXAuY2xlYXIoKTsKLSAgICAgICAgICAgICAgICBtUkVHSGVhcC5jbGVhcigpOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIC8vIE5PVEU6IHdlIGNhbm5vdCBjbGVhciB0aGUgRUJJIGhlYXAgYmVjYXVzZSBzdXJmYWNlZmxpbmdlcgotICAgICAgICAgICAgICAgIC8vIGl0c2VsZiBtYXkgYmUgdXNpbmcgaXQsIHNpbmNlIHRoaXMgaXMgd2hlcmUgc3VyZmFjZXMKLSAgICAgICAgICAgICAgICAvLyBhcmUgYWxsb2NhdGVkLiBpZiB3ZSdyZSBpbiB0aGUgbWlkZGxlIG9mIGNvbXBvc2l0aW5nIAotICAgICAgICAgICAgICAgIC8vIGEgc3VyZmFjZSAoZXZlbiBpZiBpdHMgcHJvY2VzcyBqdXN0IGRpZWQpLCB3ZSBjYW5ub3QKLSAgICAgICAgICAgICAgICAvLyByaXAgdGhlIGhlYXAgdW5kZXIgb3VyIGZlZXQuCi0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgbU93bmVyID0gTk9fT1dORVI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zcDxHUFVIYXJkd2FyZUludGVyZmFjZT4gR1BVRmFjdG9yeTo6Z2V0R1BVKCkKLXsKLSAgICByZXR1cm4gbmV3IEdQVUhhcmR3YXJlKCk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9HUFVIYXJkd2FyZS9HUFVIYXJkd2FyZS5oIGIvbGlicy9zdXJmYWNlZmxpbmdlci9HUFVIYXJkd2FyZS9HUFVIYXJkd2FyZS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzMzU0NTI4Li4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvR1BVSGFyZHdhcmUvR1BVSGFyZHdhcmUuaAorKysgL2Rldi9udWxsCkBAIC0xLDYzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfR1BVX0hBUkRXQVJFX0gKLSNkZWZpbmUgQU5EUk9JRF9HUFVfSEFSRFdBUkVfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9SZWZCYXNlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotI2luY2x1ZGUgPHV0aWxzL0tleWVkVmVjdG9yLmg+Ci0KLSNpbmNsdWRlIDx1aS9JU3VyZmFjZUNvbXBvc2VyLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIElHUFVDYWxsYmFjazsKLQotY2xhc3MgR1BVSGFyZHdhcmVJbnRlcmZhY2UgOiBwdWJsaWMgdmlydHVhbCBSZWZCYXNlCi17Ci1wdWJsaWM6Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgICAgIHJldm9rZShpbnQgcGlkKSA9IDA7Ci0gICAgdmlydHVhbCBzcDxNZW1vcnlEZWFsZXI+ICAgIHJlcXVlc3QoaW50IHBpZCkgPSAwOwotICAgIHZpcnR1YWwgc3RhdHVzX3QgICAgICAgICAgICByZXF1ZXN0KGludCBwaWQsIGNvbnN0IHNwPElHUFVDYWxsYmFjaz4mIGNhbGxiYWNrLAotICAgICAgICAgICAgSVN1cmZhY2VDb21wb3Nlcjo6Z3B1X2luZm9fdCogZ3B1KSA9IDA7Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICAgICAgZnJpZW5kbHlSZXZva2UoKSA9IDA7Ci0gICAgCi0gICAgLy8gdXNlZCBmb3IgZGVidWdnaW5nIG9ubHkuLi4KLSAgICB2aXJ0dWFsIHNwPFNpbXBsZUJlc3RGaXRBbGxvY2F0b3I+IGdldEFsbG9jYXRvcigpIGNvbnN0ICA9IDA7Ci0gICAgdmlydHVhbCBwaWRfdCBnZXRPd25lcigpIGNvbnN0ID0gMDsKLSAgICB2aXJ0dWFsIHZvaWQgdW5jb25kaXRpb25hbFJldm9rZSgpID0gMDsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBHUFVGYWN0b3J5Ci17ICAgIAotcHVibGljOgotICAgIC8vIHRoZSBncHUgZmFjdG9yeQotICAgIHN0YXRpYyBzcDxHUFVIYXJkd2FyZUludGVyZmFjZT4gZ2V0R1BVKCk7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0dQVV9IQVJEV0FSRV9ICmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmNjVkNjY5Li4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNTY4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIlN1cmZhY2VGbGluZ2VyIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDxjdXRpbHMvcHJvcGVydGllcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RvcFdhdGNoLmg+Ci0KLSNpbmNsdWRlIDx1aS9QaXhlbEZvcm1hdC5oPgotI2luY2x1ZGUgPHVpL0VHTERpc3BsYXlTdXJmYWNlLmg+Ci0KLSNpbmNsdWRlICJjbHouaCIKLSNpbmNsdWRlICJMYXllci5oIgotI2luY2x1ZGUgIkxheWVyQml0bWFwLmgiCi0jaW5jbHVkZSAiU3VyZmFjZUZsaW5nZXIuaCIKLSNpbmNsdWRlICJWUmFtSGVhcC5oIgotI2luY2x1ZGUgIkRpc3BsYXlIYXJkd2FyZS9EaXNwbGF5SGFyZHdhcmUuaCIKLQotCi0jZGVmaW5lIERFQlVHX1JFU0laRSAgICAwCi0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY29uc3QgdWludDMyX3QgTGF5ZXI6OnR5cGVJbmZvID0gTGF5ZXJCYXNlQ2xpZW50Ojp0eXBlSW5mbyB8IDQ7Ci1jb25zdCBjaGFyKiBjb25zdCBMYXllcjo6dHlwZUlEID0gIkxheWVyIjsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUxheWVyOjpMYXllcihTdXJmYWNlRmxpbmdlciogZmxpbmdlciwgRGlzcGxheUlEIGRpc3BsYXksIENsaWVudCogYywgaW50MzJfdCBpKQotICAgIDogICBMYXllckJhc2VDbGllbnQoZmxpbmdlciwgZGlzcGxheSwgYywgaSksCi0gICAgICAgIG1TZWN1cmUoZmFsc2UpLAotICAgICAgICBtRnJvbnRCdWZmZXJJbmRleCgxKSwKLSAgICAgICAgbU5lZWRzQmxlbmRpbmcodHJ1ZSksCi0gICAgICAgIG1SZXNpemVUcmFuc2FjdGlvbkRvbmUoZmFsc2UpLAotICAgICAgICBtVGV4dHVyZU5hbWUoLTFVKSwgbVRleHR1cmVXaWR0aCgwKSwgbVRleHR1cmVIZWlnaHQoMCkKLXsKLSAgICAvLyBubyBPcGVuR0wgb3BlcmF0aW9uIGlzIHBvc3NpYmxlIGhlcmUsIHNpbmNlIHdlIG1pZ2h0IG5vdCBiZQotICAgIC8vIGluIHRoZSBPcGVuR0wgdGhyZWFkLgotfQotCi1MYXllcjo6fkxheWVyKCkKLXsKLSAgICBjbGllbnQtPmZyZWUoY2xpZW50SW5kZXgoKSk7Ci0gICAgLy8gdGhpcyBzaG91bGQgYWx3YXlzIGJlIGNhbGxlZCBmcm9tIHRoZSBPcGVuR0wgdGhyZWFkCi0gICAgaWYgKG1UZXh0dXJlTmFtZSAhPSAtMVUpIHsKLSAgICAgICAgLy9nbERlbGV0ZVRleHR1cmVzKDEsICZtVGV4dHVyZU5hbWUpOwotICAgICAgICBkZWxldGVkVGV4dHVyZXMuYWRkKG1UZXh0dXJlTmFtZSk7Ci0gICAgfQotfQotCi12b2lkIExheWVyOjppbml0U3RhdGVzKHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIHVpbnQzMl90IGZsYWdzKQotewotICAgIExheWVyQmFzZTo6aW5pdFN0YXRlcyh3LGgsZmxhZ3MpOwotCi0gICAgaWYgKGZsYWdzICYgSVN1cmZhY2VDb21wb3Nlcjo6ZURlc3Ryb3lCYWNrYnVmZmVyKQotICAgICAgICBsY2Jsay0+ZmxhZ3MgfD0gZU5vQ29weUJhY2s7Ci19Ci0KLXNwPExheWVyQmFzZUNsaWVudDo6U3VyZmFjZT4gTGF5ZXI6OmdldFN1cmZhY2UoKSBjb25zdAotewotICAgIHJldHVybiBtU3VyZmFjZTsKLX0KLQotc3RhdHVzX3QgTGF5ZXI6OnNldEJ1ZmZlcnMoIENsaWVudCogY2xpZW50LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHcsIHVpbnQzMl90IGgsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgUGl4ZWxGb3JtYXQgZm9ybWF0LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBQaXhlbEZvcm1hdEluZm8gaW5mbzsKLSAgICBzdGF0dXNfdCBlcnIgPSBnZXRQaXhlbEZvcm1hdEluZm8oZm9ybWF0LCAmaW5mbyk7Ci0gICAgaWYgKGVycikgcmV0dXJuIGVycjsKLQotICAgIC8vIFRPRE86IGlmIGVIYXJkd2FyZSBpcyBleHBsaWNpdGx5IHJlcXVlc3RlZCwgd2Ugc2hvdWxkIGZhaWwKLSAgICAvLyBvbiBzeXN0ZW1zIHdoZXJlIHdlIGNhbid0IGFsbG9jYXRlIG1lbW9yeSB0aGF0IGNhbiBiZSB1c2VkIHdpdGgKLSAgICAvLyBETUEgZW5naW5lcyBmb3IgaW5zdGFuY2UuCi0gICAgCi0gICAgLy8gRklYTUU6IHdlIGFsd2F5cyBhc2sgZm9yIGhhcmR3YXJlIGZvciBub3cgKHRoaXMgc2hvdWxkIGNvbWUgZnJvbSBjb3B5Yml0KQotICAgIGZsYWdzIHw9IElTdXJmYWNlQ29tcG9zZXI6OmVIYXJkd2FyZTsKLQotICAgIGNvbnN0IHVpbnQzMl90IG1lbW9yeV9mbGFncyA9IGZsYWdzICYgCi0gICAgICAgICAgICAoSVN1cmZhY2VDb21wb3Nlcjo6ZUdQVSB8IAotICAgICAgICAgICAgIElTdXJmYWNlQ29tcG9zZXI6OmVIYXJkd2FyZSB8IAotICAgICAgICAgICAgIElTdXJmYWNlQ29tcG9zZXI6OmVTZWN1cmUpOwotICAgIAotICAgIC8vIHBpeGVsLWFsaWdubWVudC4gdGhlIGZpbmFsIGFsaWdubWVudCBtYXkgYmUgYmlnZ2VyIGJlY2F1c2UKLSAgICAvLyB3ZSBhbHdheXMgZm9yY2UgYSA0LWJ5dGUgYWxpZ25lZCBicHIuCi0gICAgdWludDMyX3QgYWxpZ25tZW50ID0gMTsKLQotICAgIGlmIChmbGFncyAmIElTdXJmYWNlQ29tcG9zZXI6OmVHUFUpIHsKLSAgICAgICAgLy8gRklYTUU6IHRoaXMgdmFsdWUgc2hvdWxkIGNvbWUgZnJvbSB0aGUgaC93Ci0gICAgICAgIGFsaWdubWVudCA9IDg7IAotICAgICAgICAvLyBGSVhNRTogdGhpcyBpcyBtc203MjAxQSBzcGVjaWZpYywgYXMgaXRzIEdQVSBvbmx5IHN1cHBvcnRzCi0gICAgICAgIC8vIEJHUkFfODg4OC4KLSAgICAgICAgaWYgKGZvcm1hdCA9PSBQSVhFTF9GT1JNQVRfUkdCQV84ODg4KSB7Ci0gICAgICAgICAgICBmb3JtYXQgPSBQSVhFTF9GT1JNQVRfQkdSQV84ODg4OwotICAgICAgICB9Ci0gICAgfQotCi0gICAgbVNlY3VyZSA9IChmbGFncyAmIElTdXJmYWNlQ29tcG9zZXI6OmVTZWN1cmUpID8gdHJ1ZSA6IGZhbHNlOwotICAgIG1OZWVkc0JsZW5kaW5nID0gKGluZm8uaF9hbHBoYSAtIGluZm8ubF9hbHBoYSkgPiAwOwotICAgIHNwPE1lbW9yeURlYWxlcj4gYWxsb2NhdG9yc1syXTsKLSAgICBmb3IgKGludCBpPTAgOyBpPDIgOyBpKyspIHsKLSAgICAgICAgYWxsb2NhdG9yc1tpXSA9IGNsaWVudC0+Y3JlYXRlQWxsb2NhdG9yKG1lbW9yeV9mbGFncyk7Ci0gICAgICAgIGlmIChhbGxvY2F0b3JzW2ldID09IDApCi0gICAgICAgICAgICByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICBtQnVmZmVyc1tpXS5pbml0KGFsbG9jYXRvcnNbaV0pOwotICAgICAgICBpbnQgZXJyID0gbUJ1ZmZlcnNbaV0uc2V0Qml0cyh3LCBoLCBhbGlnbm1lbnQsIGZvcm1hdCwgTGF5ZXJCaXRtYXA6OlNFQ1VSRV9CSVRTKTsKLSAgICAgICAgaWYgKGVyciAhPSBOT19FUlJPUikKLSAgICAgICAgICAgIHJldHVybiBlcnI7Ci0gICAgICAgIG1CdWZmZXJzW2ldLmNsZWFyKCk7IC8vIGNsZWFyIHRoZSBiaXRzIGZvciBzZWN1cml0eQotICAgICAgICBtQnVmZmVyc1tpXS5nZXRJbmZvKGxjYmxrLT5zdXJmYWNlICsgaSk7Ci0gICAgfQotCi0gICAgbVN1cmZhY2UgPSBuZXcgU3VyZmFjZShjbGllbnRJbmRleCgpLAotICAgICAgICAgICAgYWxsb2NhdG9yc1swXS0+Z2V0TWVtb3J5SGVhcCgpLAotICAgICAgICAgICAgYWxsb2NhdG9yc1sxXS0+Z2V0TWVtb3J5SGVhcCgpLAotICAgICAgICAgICAgbUlkZW50aXR5KTsKLQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotdm9pZCBMYXllcjo6cmVsb2FkVGV4dHVyZShjb25zdCBSZWdpb24mIGRpcnR5KQotewotICAgIGlmIChVTkxJS0VMWShtVGV4dHVyZU5hbWUgPT0gLTFVKSkgewotICAgICAgICAvLyBjcmVhdGUgdGhlIHRleHR1cmUgbmFtZSB0aGUgZmlyc3QgdGltZQotICAgICAgICAvLyBjYW4ndCBkbyB0aGF0IGluIHRoZSBjdG9yLCBiZWNhdXNlIGl0IHJ1bnMgaW4gYW5vdGhlciB0aHJlYWQuCi0gICAgICAgIG1UZXh0dXJlTmFtZSA9IGNyZWF0ZVRleHR1cmUoKTsKLSAgICB9Ci0gICAgY29uc3QgR0dMU3VyZmFjZSYgdChmcm9udEJ1ZmZlcigpLnN1cmZhY2UoKSk7Ci0gICAgbG9hZFRleHR1cmUoZGlydHksIG1UZXh0dXJlTmFtZSwgdCwgbVRleHR1cmVXaWR0aCwgbVRleHR1cmVIZWlnaHQpOwotfQotCi0KLXZvaWQgTGF5ZXI6Om9uRHJhdyhjb25zdCBSZWdpb24mIGNsaXApIGNvbnN0Ci17Ci0gICAgaWYgKFVOTElLRUxZKG1UZXh0dXJlTmFtZSA9PSAtMUxVKSkgewotICAgICAgICAvL0xPR1coIkxheWVyICVwIGRvZXNuJ3QgaGF2ZSBhIHRleHR1cmUiLCB0aGlzKTsKLSAgICAgICAgLy8gdGhlIHRleHR1cmUgaGFzIG5vdCBiZWVuIGNyZWF0ZWQgeWV0LCB0aGlzIExheWVyIGhhcwotICAgICAgICAvLyBpbiBmYWN0IG5ldmVyIGJlZW4gZHJhd24gaW50by4gdGhpcyBoYXBwZW5zIGZyZXF1ZW50bHkgd2l0aAotICAgICAgICAvLyBTdXJmYWNlVmlldy4KLSAgICAgICAgY2xlYXJXaXRoT3BlbkdMKGNsaXApOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiBodyhncmFwaGljUGxhbmUoMCkuZGlzcGxheUhhcmR3YXJlKCkpOwotICAgIGNvbnN0IExheWVyQml0bWFwJiBmcm9udChmcm9udEJ1ZmZlcigpKTsKLSAgICBjb25zdCBHR0xTdXJmYWNlJiB0KGZyb250LnN1cmZhY2UoKSk7Ci0KLSAgICBzdGF0dXNfdCBlcnIgPSBOT19FUlJPUjsKLSAgICBjb25zdCBpbnQgY2FuX3VzZV9jb3B5Yml0ID0gY2FuVXNlQ29weWJpdCgpOwotICAgIGlmIChjYW5fdXNlX2NvcHliaXQpICB7Ci0gICAgICAgIC8vIFN0b3BXYXRjaCB3YXRjaCgiY29weWJpdCIpOwotICAgICAgICBjb25zdCBTdGF0ZSYgcyhkcmF3aW5nU3RhdGUoKSk7Ci0KLSAgICAgICAgY29weWJpdF9pbWFnZV90IGRzdDsKLSAgICAgICAgaHcuZ2V0RGlzcGxheVN1cmZhY2UoJmRzdCk7Ci0gICAgICAgIGNvbnN0IGNvcHliaXRfcmVjdF90JiBkcmVjdAotICAgICAgICAgICAgPSByZWludGVycHJldF9jYXN0PGNvbnN0IGNvcHliaXRfcmVjdF90Jj4obVRyYW5zZm9ybWVkQm91bmRzKTsKLQotICAgICAgICBjb3B5Yml0X2ltYWdlX3Qgc3JjOwotICAgICAgICBmcm9udC5nZXRCaXRtYXBTdXJmYWNlKCZzcmMpOwotICAgICAgICBjb3B5Yml0X3JlY3RfdCBzcmVjdCA9IHsgMCwgMCwgdC53aWR0aCwgdC5oZWlnaHQgfTsKLQotICAgICAgICBjb3B5Yml0X2RldmljZV90KiBjb3B5Yml0ID0gbUZsaW5nZXItPmdldEJsaXRFbmdpbmUoKTsKLSAgICAgICAgY29weWJpdC0+c2V0X3BhcmFtZXRlcihjb3B5Yml0LCBDT1BZQklUX1RSQU5TRk9STSwgZ2V0T3JpZW50YXRpb24oKSk7Ci0gICAgICAgIGNvcHliaXQtPnNldF9wYXJhbWV0ZXIoY29weWJpdCwgQ09QWUJJVF9QTEFORV9BTFBIQSwgcy5hbHBoYSk7Ci0gICAgICAgIGNvcHliaXQtPnNldF9wYXJhbWV0ZXIoY29weWJpdCwgQ09QWUJJVF9ESVRIRVIsCi0gICAgICAgICAgICAgICAgcy5mbGFncyAmIElTdXJmYWNlQ29tcG9zZXI6OmVMYXllckRpdGhlciA/Ci0gICAgICAgICAgICAgICAgICAgICAgICBDT1BZQklUX0VOQUJMRSA6IENPUFlCSVRfRElTQUJMRSk7Ci0KLSAgICAgICAgcmVnaW9uX2l0ZXJhdG9yIGl0KGNsaXApOwotICAgICAgICBlcnIgPSBjb3B5Yml0LT5zdHJldGNoKGNvcHliaXQsICZkc3QsICZzcmMsICZkcmVjdCwgJnNyZWN0LCAmaXQpOwotICAgIH0KLQotICAgIGlmICghY2FuX3VzZV9jb3B5Yml0IHx8IGVycikgewotICAgICAgICBkcmF3V2l0aE9wZW5HTChjbGlwLCBtVGV4dHVyZU5hbWUsIHQpOwotICAgIH0KLX0KLQotc3RhdHVzX3QgTGF5ZXI6OnJlYWxsb2NhdGVCdWZmZXIoaW50MzJfdCBpbmRleCwgdWludDMyX3QgdywgdWludDMyX3QgaCkKLXsKLSAgICBMT0dEX0lGKERFQlVHX1JFU0laRSwKLSAgICAgICAgICAgICAgICAicmVhbGxvY2F0ZUJ1ZmZlciAobGF5ZXI9JXApLCAiCi0gICAgICAgICAgICAgICAgInJlcXVlc3RlZCAoJWR4JWQpLCAiCi0gICAgICAgICAgICAgICAgImluZGV4PSVkLCAoJWR4JWQpLCAoJWR4JWQpIiwKLSAgICAgICAgICAgICAgICB0aGlzLAotICAgICAgICAgICAgICAgIGludCh3KSwgaW50KGgpLAotICAgICAgICAgICAgICAgIGludChpbmRleCksCi0gICAgICAgICAgICAgICAgaW50KG1CdWZmZXJzWzBdLndpZHRoKCkpLCBpbnQobUJ1ZmZlcnNbMF0uaGVpZ2h0KCkpLAotICAgICAgICAgICAgICAgIGludChtQnVmZmVyc1sxXS53aWR0aCgpKSwgaW50KG1CdWZmZXJzWzFdLmhlaWdodCgpKSk7Ci0KLSAgICBzdGF0dXNfdCBlcnIgPSBtQnVmZmVyc1tpbmRleF0ucmVzaXplKHcsIGgpOwotICAgIGlmIChlcnIgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgbUJ1ZmZlcnNbaW5kZXhdLmdldEluZm8obGNibGstPnN1cmZhY2UgKyBpbmRleCk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgTE9HRSgicmVzaXppbmcgYnVmZmVyICVkIHRvICgldSwldSkgZmFpbGVkIFslMDh4XSAlcyIsCi0gICAgICAgICAgICBpbmRleCwgdywgaCwgZXJyLCBzdHJlcnJvcihlcnIpKTsKLSAgICAgICAgLy8gWFhYOiB3aGF0IHRvIGRvLCB3aGF0IHRvIGRvPyBXZSBjb3VsZCB0cnkgdG8gZnJlZSBzb21lCi0gICAgICAgIC8vIGhpZGRlbiBzdXJmYWNlcywgaW5zdGVhZCBvZiBraWxsaW5nIHRoaXMgb25lPwotICAgIH0KLSAgICByZXR1cm4gZXJyOwotfQotCi11aW50MzJfdCBMYXllcjo6ZG9UcmFuc2FjdGlvbih1aW50MzJfdCBmbGFncykKLXsKLSAgICBjb25zdCBMYXllcjo6U3RhdGUmIGZyb250KGRyYXdpbmdTdGF0ZSgpKTsKLSAgICBjb25zdCBMYXllcjo6U3RhdGUmIHRlbXAoY3VycmVudFN0YXRlKCkpOwotCi0gICAgLy8gdGhlIHRlc3QgZnJvbnQue3d8aH0gIT0gdGVtcC57d3xofSBpcyBub3QgZW5vdWdoIGJlY2F1c2UgaXQgaXMgcG9zc2libGUKLSAgICAvLyB0aGF0IHRoZSBzaXplIGNoYW5nZWQgYmFjayB0byBpdHMgcHJldmlvdXMgdmFsdWUgYmVmb3JlIHRoZSBidWZmZXIKLSAgICAvLyB3YXMgcmVzaXplZCAoaW4gdGhlIGVMb2NrZWQgY2FzZSBiZWxvdyksIGluIHdoaWNoIGNhc2UsIHdlIHN0aWxsCi0gICAgLy8gbmVlZCB0byBleGVjdXRlIHRoZSBjb2RlIGJlbG93IHNvIHRoZSBjbGllbnRzIGhhdmUgYSBjaGFuY2UgdG8gYmUKLSAgICAvLyByZWxlYXNlLiByZXN6ZSgpIGRlYWxzIHdpdGggdGhlIGZhY3QgdGhhdCB0aGUgc2l6ZSBjYW4gYmUgdGhlIHNhbWUuCi0KLSAgICAvKgotICAgICAqICBWYXJpb3VzIHN0YXRlcyB3ZSBjb3VsZCBiZSBpbi4uLgotCi0gICAgICAgICByZXNpemUgPSBzdGF0ZSAmIGVSZXNpemVSZXF1ZXN0ZWQ7Ci0gICAgICAgICBpZiAoYmFja2J1ZmZlckNoYW5nZWQpIHsKLSAgICAgICAgICAgICBpZiAocmVzaXplID09IDApIHsKLSAgICAgICAgICAgICAgICAgLy8gRVJST1IsIHRoZSByZXNpemVkIGJ1ZmZlciBkb2Vzbid0IGhhdmUgaXRzIHJlc2l6ZSBmbGFnIHNldAotICAgICAgICAgICAgIH0gZWxzZSBpZiAocmVzaXplID09IG1hc2spIHsKLSAgICAgICAgICAgICAgICAgLy8gRVJST1Igb25lIG9mIHRoZSBidWZmZXIgaGFzIGFscmVhZHkgYmVlbiByZXNpemVkCi0gICAgICAgICAgICAgfSBlbHNlIGlmIChyZXNpemUgPT0gbWFzayBeIGVSZXNpemVSZXF1ZXN0ZWQpIHsKLSAgICAgICAgICAgICAgICAgLy8gRVJST1IsIHRoZSByZXNpemVkIGJ1ZmZlciBkb2Vzbid0IGhhdmUgaXRzIHJlc2l6ZSBmbGFnIHNldAotICAgICAgICAgICAgIH0gZWxzZSBpZiAocmVzaXplID09IGVSZXNpemVSZXF1ZXN0ZWQpIHsKLSAgICAgICAgICAgICAgICAgLy8gT0ssIE5vcm1hbCBjYXNlLCBwcm9jZWVkIHdpdGggcmVzaXplCi0gICAgICAgICAgICAgfQotICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICBpZiAocmVzaXplID09IDApIHsKLSAgICAgICAgICAgICAgICAgLy8gT0ssIG5vdGhpbmcgc3BlY2lhbCwgZG8gbm90aGluZwotICAgICAgICAgICAgIH0gZWxzZSBpZiAocmVzaXplID09IG1hc2spIHsKLSAgICAgICAgICAgICAgICAgLy8gcmVzdGFydGVkIHRyYW5zYWN0aW9uLCBkbyBub3RoaW5nCi0gICAgICAgICAgICAgfSBlbHNlIGlmIChyZXNpemUgPT0gbWFzayBeIGVSZXNpemVSZXF1ZXN0ZWQpIHsKLSAgICAgICAgICAgICAgICAgLy8gcmVzdGFydGVkIHRyYW5zYWN0aW9uLCBkbyBub3RoaW5nCi0gICAgICAgICAgICAgfSBlbHNlIGlmIChyZXNpemUgPT0gZVJlc2l6ZVJlcXVlc3RlZCkgewotICAgICAgICAgICAgICAgICAvLyBPSywgc2l6ZSByZXNldCB0byBwcmV2aW91cyB2YWx1ZSwgcHJvY2VlZCB3aXRoIHJlc2l6ZQotICAgICAgICAgICAgIH0KLSAgICAgICAgIH0KLSAgICAgKi8KLQotICAgIC8vIEluZGV4IG9mIHRoZSBiYWNrIGJ1ZmZlcgotICAgIGNvbnN0IGJvb2wgYmFja2J1ZmZlckNoYW5nZWQgPSAoZnJvbnQudyAhPSB0ZW1wLncpIHx8IChmcm9udC5oICE9IHRlbXAuaCk7Ci0gICAgY29uc3QgdWludDMyX3Qgc3RhdGUgPSBsY2Jsay0+c3dhcFN0YXRlOwotICAgIGNvbnN0IGludDMyX3QgY2xpZW50QmFja0J1ZmZlckluZGV4ID0gbGF5ZXJfY2Jsa190OjpiYWNrQnVmZmVyKHN0YXRlKTsKLSAgICBjb25zdCB1aW50MzJfdCBtYXNrID0gY2xpZW50QmFja0J1ZmZlckluZGV4ID8gZVJlc2l6ZUJ1ZmZlcjEgOiBlUmVzaXplQnVmZmVyMDsKLSAgICB1aW50MzJfdCByZXNpemVGbGFncyA9IHN0YXRlICYgZVJlc2l6ZVJlcXVlc3RlZDsKLQotICAgIGlmIChVTkxJS0VMWShiYWNrYnVmZmVyQ2hhbmdlZCAmJiAocmVzaXplRmxhZ3MgIT0gZVJlc2l6ZVJlcXVlc3RlZCkpKSB7Ci0gICAgICAgIExPR0UoICAgImJhY2tidWZmZXIgc2l6ZSBjaGFuZ2VkLCBidXQgYm90aCByZXNpemUgZmxhZ3MgYXJlIG5vdCBzZXQhICIKLSAgICAgICAgICAgICAgICAiKGxheWVyPSVwKSwgc3RhdGU9JTA4eCwgcmVxdWVzdGVkICglZHglZCksIGRyYXdpbmcgKCVkLCVkKSwgIgotICAgICAgICAgICAgICAgICJpbmRleD0lZCwgKCVkeCVkKSwgKCVkeCVkKSIsCi0gICAgICAgICAgICAgICAgdGhpcywgIHN0YXRlLAotICAgICAgICAgICAgICAgIGludCh0ZW1wLncpLCBpbnQodGVtcC5oKSwKLSAgICAgICAgICAgICAgICBpbnQoZHJhd2luZ1N0YXRlKCkudyksIGludChkcmF3aW5nU3RhdGUoKS5oKSwKLSAgICAgICAgICAgICAgICBpbnQoY2xpZW50QmFja0J1ZmZlckluZGV4KSwKLSAgICAgICAgICAgICAgICBpbnQobUJ1ZmZlcnNbMF0ud2lkdGgoKSksIGludChtQnVmZmVyc1swXS5oZWlnaHQoKSksCi0gICAgICAgICAgICAgICAgaW50KG1CdWZmZXJzWzFdLndpZHRoKCkpLCBpbnQobUJ1ZmZlcnNbMV0uaGVpZ2h0KCkpKTsKLSAgICAgICAgLy8gaWYgd2UgZ2V0IHRoZXJlIHdlJ3JlIHByZXR0eSBzY3Jld2VkLiB0aGUgb25seSByZWFzb25hYmxlCi0gICAgICAgIC8vIHRoaW5nIHRvIGRvIGlzIHRvIHByZXRlbmQgd2Ugc2hvdWxkIGRvIHRoZSByZXNpemUgc2luY2UKLSAgICAgICAgLy8gYmFja2J1ZmZlckNoYW5nZWQgaXMgc2V0ICh0aGlzIGFsc28gd2lsbCBnaXZlIGEgY2hhbmNlIHRvCi0gICAgICAgIC8vIGNsaWVudCB0byBnZXQgdW5ibG9ja2VkKQotICAgICAgICByZXNpemVGbGFncyA9IGVSZXNpemVSZXF1ZXN0ZWQ7Ci0gICAgfQotCi0gICAgaWYgKHJlc2l6ZUZsYWdzID09IGVSZXNpemVSZXF1ZXN0ZWQpICB7Ci0gICAgICAgIC8vIE5PVEU6IGFzc2VydGluZyB0aGF0IGNsaWVudEJhY2tCdWZmZXJJbmRleCE9bUZyb250QnVmZmVySW5kZXgKLSAgICAgICAgLy8gaGVyZSwgd291bGQgYmUgd3JvbmcgYW5kIG1pc2xlYWRpbmcgYmVjYXVzZSBieSB0aGlzIHBvaW50Ci0gICAgICAgIC8vIG1Gcm9udEJ1ZmZlckluZGV4IGhhcyBub3QgYmVlbiB1cGRhdGVkIHlldC4KLQotICAgICAgICBMT0dEX0lGKERFQlVHX1JFU0laRSwKLSAgICAgICAgICAgICAgICAgICAgInJlc2l6ZSAobGF5ZXI9JXApLCBzdGF0ZT0lMDh4LCAiCi0gICAgICAgICAgICAgICAgICAgICJyZXF1ZXN0ZWQgKCVkeCVkKSwgIgotICAgICAgICAgICAgICAgICAgICAiZHJhd2luZyAoJWQsJWQpLCAiCi0gICAgICAgICAgICAgICAgICAgICJpbmRleD0lZCwgKCVkeCVkKSwgKCVkeCVkKSIsCi0gICAgICAgICAgICAgICAgICAgIHRoaXMsICBzdGF0ZSwKLSAgICAgICAgICAgICAgICAgICAgaW50KHRlbXAudyksIGludCh0ZW1wLmgpLAotICAgICAgICAgICAgICAgICAgICBpbnQoZHJhd2luZ1N0YXRlKCkudyksIGludChkcmF3aW5nU3RhdGUoKS5oKSwKLSAgICAgICAgICAgICAgICAgICAgaW50KGNsaWVudEJhY2tCdWZmZXJJbmRleCksCi0gICAgICAgICAgICAgICAgICAgIGludChtQnVmZmVyc1swXS53aWR0aCgpKSwgaW50KG1CdWZmZXJzWzBdLmhlaWdodCgpKSwKLSAgICAgICAgICAgICAgICAgICAgaW50KG1CdWZmZXJzWzFdLndpZHRoKCkpLCBpbnQobUJ1ZmZlcnNbMV0uaGVpZ2h0KCkpKTsKLQotICAgICAgICBpZiAoc3RhdGUgJiBlTG9ja2VkKSB7Ci0gICAgICAgICAgICAvLyBpZiB0aGUgYnVmZmVyIGlzIGxvY2tlZCwgd2UgY2FuJ3QgcmVzaXplIGFueXRoaW5nIGJlY2F1c2UKLSAgICAgICAgICAgIC8vIC0gdGhlIGJhY2tidWZmZXIgaXMgY3VycmVudGx5IGluIHVzZSBieSB0aGUgdXNlcgotICAgICAgICAgICAgLy8gLSB0aGUgZnJvbnQgYnVmZmVyIGlzIGJlaW5nIHNob3duCi0gICAgICAgICAgICAvLyBXZSBqdXN0IGFjdCBhcyBpZiB0aGUgdHJhbnNhY3Rpb24gZGlkbid0IGhhcHBlbiBhbmQgd2UKLSAgICAgICAgICAgIC8vIHJlc2NoZWR1bGUgaXQgbGF0ZXIuLi4KLSAgICAgICAgICAgIGZsYWdzIHw9IGVSZXN0YXJ0VHJhbnNhY3Rpb247Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBUaGlzIGJ1ZmZlciBuZWVkcyB0byBiZSByZXNpemVkCi0gICAgICAgICAgICBzdGF0dXNfdCBlcnIgPQotICAgICAgICAgICAgICAgIHJlc2l6ZShjbGllbnRCYWNrQnVmZmVySW5kZXgsIHRlbXAudywgdGVtcC5oLCAidHJhbnNhY3Rpb24iKTsKLSAgICAgICAgICAgIGlmIChlcnIgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgICAgICBjb25zdCB1aW50MzJfdCBtYXNrID0gY2xpZW50QmFja0J1ZmZlckluZGV4ID8gZVJlc2l6ZUJ1ZmZlcjEgOiBlUmVzaXplQnVmZmVyMDsKLSAgICAgICAgICAgICAgICBhbmRyb2lkX2F0b21pY19hbmQofm1hc2ssICYobGNibGstPnN3YXBTdGF0ZSkpOwotICAgICAgICAgICAgICAgIC8vIHNpbmNlIGEgYnVmZmVyIGJlY2FtZSBhdmFpbGFibGUsIHdlIGNhbiBsZXQgdGhlIGNsaWVudCBnby4uLgotICAgICAgICAgICAgICAgIG1GbGluZ2VyLT5zY2hlZHVsZUJyb2FkY2FzdChjbGllbnQpOwotICAgICAgICAgICAgICAgIG1SZXNpemVUcmFuc2FjdGlvbkRvbmUgPSB0cnVlOwotCi0gICAgICAgICAgICAgICAgLy8gd2UncmUgYmVpbmcgcmVzaXplZCBhbmQgdGhlcmUgaXMgYSBmcmVlemUgZGlzcGxheSByZXF1ZXN0LAotICAgICAgICAgICAgICAgIC8vIGFjcXVpcmUgYSBmcmVlemUgbG9jaywgc28gdGhhdCB0aGUgc2NyZWVuIHN0YXlzIHB1dAotICAgICAgICAgICAgICAgIC8vIHVudGlsIHdlJ3ZlIHJlZHJhd24gYXQgdGhlIG5ldyBzaXplOyB0aGlzIGlzIHRvIGF2b2lkCi0gICAgICAgICAgICAgICAgLy8gZ2xpdGNoZXMgdXBvbiBvcmllbnRhdGlvbiBjaGFuZ2VzLgotICAgICAgICAgICAgICAgIGlmIChtRmxpbmdlci0+aGFzRnJlZXplUmVxdWVzdCgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIGlmIHRoZSBzdXJmYWNlIGlzIGhpZGRlbiwgZG9uJ3QgdHJ5IHRvIGFjcXVpcmUgdGhlCi0gICAgICAgICAgICAgICAgICAgIC8vIGZyZWV6ZSBsb2NrLCBzaW5jZSBoaWRkZW4gc3VyZmFjZXMgbWF5IG5ldmVyIHJlZHJhdwotICAgICAgICAgICAgICAgICAgICBpZiAoIShmcm9udC5mbGFncyAmIElTdXJmYWNlQ29tcG9zZXI6OmVMYXllckhpZGRlbikpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIG1GcmVlemVMb2NrID0gbUZsaW5nZXItPmdldEZyZWV6ZUxvY2soKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBpZiAodGVtcC5zZXF1ZW5jZSAhPSBmcm9udC5zZXF1ZW5jZSkgewotICAgICAgICBpZiAodGVtcC5mbGFncyAmIElTdXJmYWNlQ29tcG9zZXI6OmVMYXllckhpZGRlbiB8fCB0ZW1wLmFscGhhID09IDApIHsKLSAgICAgICAgICAgIC8vIHRoaXMgc3VyZmFjZSBpcyBub3cgaGlkZGVuLCBzbyBpdCBzaG91bGRuJ3QgaG9sZCBhIGZyZWV6ZSBsb2NrCi0gICAgICAgICAgICAvLyAoaXQgbWF5IG5ldmVyIHJlZHJhdywgd2hpY2ggaXMgZmluZSBpZiBpdCBpcyBoaWRkZW4pCi0gICAgICAgICAgICBtRnJlZXplTG9jay5jbGVhcigpOwotICAgICAgICB9Ci0gICAgfQotICAgICAgICAKLSAgICByZXR1cm4gTGF5ZXJCYXNlOjpkb1RyYW5zYWN0aW9uKGZsYWdzKTsKLX0KLQotc3RhdHVzX3QgTGF5ZXI6OnJlc2l6ZSgKLSAgICAgICAgaW50MzJfdCBjbGllbnRCYWNrQnVmZmVySW5kZXgsCi0gICAgICAgIHVpbnQzMl90IHdpZHRoLCB1aW50MzJfdCBoZWlnaHQsCi0gICAgICAgIGNvbnN0IGNoYXIqIHdoYXQpCi17Ci0gICAgLyoKLSAgICAgKiBoYW5kbGUgcmVzaXplIChiYWNrYnVmZmVyIGFuZCBmcm9udGJ1ZmZlciByZWFsbG9jYXRpb24pCi0gICAgICovCi0KLSAgICBjb25zdCBMYXllckJpdG1hcCYgY2xpZW50QmFja0J1ZmZlcihtQnVmZmVyc1tjbGllbnRCYWNrQnVmZmVySW5kZXhdKTsKLQotICAgIC8vIGlmIHRoZSBuZXcgKHRyYW5zYWN0aW9uKSBzaXplIGlzICE9IGZyb20gdGhlIHRoZSBiYWNrYnVmZmVyCi0gICAgLy8gdGhlbiB3ZSBuZWVkIHRvIHJlYWxsb2NhdGUgdGhlIGJhY2tidWZmZXIKLSAgICBib29sIGJhY2tidWZmZXJDaGFuZ2VkID0gKGNsaWVudEJhY2tCdWZmZXIud2lkdGgoKSAgIT0gd2lkdGgpIHx8Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIChjbGllbnRCYWNrQnVmZmVyLmhlaWdodCgpICE9IGhlaWdodCk7Ci0KLSAgICBMT0dEX0lGKCFiYWNrYnVmZmVyQ2hhbmdlZCwKLSAgICAgICAgICAgICIoJXMpIGVSZXNpemVSZXF1ZXN0ZWQgKGxheWVyPSVwKSwgYnV0IHNpemUgbm90IGNoYW5nZWQ6ICIKLSAgICAgICAgICAgICJyZXF1ZXN0ZWQgKCVkeCVkKSwgZHJhd2luZyAoJWQsJWQpLCBjdXJyZW50ICglZCwlZCksIgotICAgICAgICAgICAgInN0YXRlPSUwOGx4LCBpbmRleD0lZCwgKCVkeCVkKSwgKCVkeCVkKSIsCi0gICAgICAgICAgICB3aGF0LCB0aGlzLAotICAgICAgICAgICAgaW50KHdpZHRoKSwgaW50KGhlaWdodCksCi0gICAgICAgICAgICBpbnQoZHJhd2luZ1N0YXRlKCkudyksIGludChkcmF3aW5nU3RhdGUoKS5oKSwKLSAgICAgICAgICAgIGludChjdXJyZW50U3RhdGUoKS53KSwgaW50KGN1cnJlbnRTdGF0ZSgpLmgpLAotICAgICAgICAgICAgbG9uZyhsY2Jsay0+c3dhcFN0YXRlKSwKLSAgICAgICAgICAgIGludChjbGllbnRCYWNrQnVmZmVySW5kZXgpLAotICAgICAgICAgICAgaW50KG1CdWZmZXJzWzBdLndpZHRoKCkpLCBpbnQobUJ1ZmZlcnNbMF0uaGVpZ2h0KCkpLAotICAgICAgICAgICAgaW50KG1CdWZmZXJzWzFdLndpZHRoKCkpLCBpbnQobUJ1ZmZlcnNbMV0uaGVpZ2h0KCkpKTsKLQotICAgIC8vIHRoaXMgY2FuIGhhcHBlbiB3aGVuIGNoYW5naW5nIHRoZSBzaXplIGJhY2sgYW5kIGZvcnRoIHF1aWNrbHkKLSAgICBzdGF0dXNfdCBlcnIgPSBOT19FUlJPUjsKLSAgICBpZiAoYmFja2J1ZmZlckNoYW5nZWQpIHsKLSAgICAgICAgZXJyID0gcmVhbGxvY2F0ZUJ1ZmZlcihjbGllbnRCYWNrQnVmZmVySW5kZXgsIHdpZHRoLCBoZWlnaHQpOwotICAgIH0KLSAgICBpZiAoVU5MSUtFTFkoZXJyICE9IE5PX0VSUk9SKSkgewotICAgICAgICAvLyBjb3VsZG4ndCByZWFsbG9jYXRlIHRoZSBzdXJmYWNlCi0gICAgICAgIGFuZHJvaWRfYXRvbWljX3dyaXRlKGVJbnZhbGlkU3VyZmFjZSwgJmxjYmxrLT5zd2FwU3RhdGUpOwotICAgICAgICBtZW1zZXQobGNibGstPnN1cmZhY2UrY2xpZW50QmFja0J1ZmZlckluZGV4LCAwLCBzaXplb2Yoc3VyZmFjZV9pbmZvX3QpKTsKLSAgICB9Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotdm9pZCBMYXllcjo6c2V0U2l6ZUNoYW5nZWQodWludDMyX3QgdywgdWludDMyX3QgaCkKLXsKLSAgICBMT0dEX0lGKERFQlVHX1JFU0laRSwKLSAgICAgICAgICAgICJzZXRTaXplQ2hhbmdlZCB3PSVkLCBoPSVkIChvbGQ6IHc9JWQsIGg9JWQpIiwKLSAgICAgICAgICAgIHcsIGgsIG1DdXJyZW50U3RhdGUudywgbUN1cnJlbnRTdGF0ZS5oKTsKLSAgICBhbmRyb2lkX2F0b21pY19vcihlUmVzaXplUmVxdWVzdGVkLCAmKGxjYmxrLT5zd2FwU3RhdGUpKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gcGFnZWZsaXAgaGFuZGxpbmcuLi4KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdm9pZCBMYXllcjo6bG9ja1BhZ2VGbGlwKGJvb2wmIHJlY29tcHV0ZVZpc2libGVSZWdpb25zKQotewotICAgIHVpbnQzMl90IHN0YXRlID0gYW5kcm9pZF9hdG9taWNfb3IoZUJ1c3ksICYobGNibGstPnN3YXBTdGF0ZSkpOwotICAgIC8vIHByZWVtcHRpdmVseSBibG9jayB0aGUgY2xpZW50LCBiZWNhdXNlIGhlIG1pZ2h0IHNldAotICAgIC8vIGVGbGlwUmVxdWVzdGVkIGF0IGFueSB0aW1lIGFuZCB3YW50IHRvIHVzZSB0aGlzIGJ1ZmZlcgotICAgIC8vIGZvciB0aGUgbmV4dCBmcmFtZS4gVGhpcyB3aWxsIGJlIHVuc2V0IGJlbG93IGlmIGl0Ci0gICAgLy8gdHVybnMgb3V0IHdlIGRpZG4ndCBuZWVkIGl0LgotCi0gICAgdWludDMyX3QgbWFzayA9IGVJbnZhbGlkU3VyZmFjZSB8IGVGbGlwUmVxdWVzdGVkIHwgZVJlc2l6ZVJlcXVlc3RlZDsKLSAgICBpZiAoIShzdGF0ZSAmIG1hc2spKQotICAgICAgICByZXR1cm47Ci0KLSAgICBpZiAoVU5MSUtFTFkoc3RhdGUgJiBlSW52YWxpZFN1cmZhY2UpKSB7Ci0gICAgICAgIC8vIGlmIGVJbnZhbGlkU3VyZmFjZSBpcyBzZXQsIHRoaXMgbWVhbnMgdGhlIHN1cmZhY2UKLSAgICAgICAgLy8gYmVjYW1lIGludmFsaWQgZHVyaW5nIGEgdHJhbnNhY3Rpb24gKE5PX01FTU9SWSBmb3IgaW5zdGFuY2UpCi0gICAgICAgIG1GbGluZ2VyLT5zY2hlZHVsZUJyb2FkY2FzdChjbGllbnQpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgaWYgKFVOTElLRUxZKHN0YXRlICYgZUZsaXBSZXF1ZXN0ZWQpKSB7Ci0gICAgICAgIHVpbnQzMl90IG9sZFN0YXRlOwotICAgICAgICBtUG9zdGVkRGlydHlSZWdpb24gPSBwb3N0KCZvbGRTdGF0ZSwgcmVjb21wdXRlVmlzaWJsZVJlZ2lvbnMpOwotICAgICAgICBpZiAob2xkU3RhdGUgJiBlTmV4dEZsaXBQZW5kaW5nKSB7Ci0gICAgICAgICAgICAvLyBQcm9jZXNzIGFub3RoZXIgcm91bmQgKHdlIGtub3cgYXQgbGVhc3QgYSBidWZmZXIKLSAgICAgICAgICAgIC8vIGlzIHJlYWR5IGZvciB0aGF0IGNsaWVudCkuCi0gICAgICAgICAgICBtRmxpbmdlci0+c2lnbmFsRXZlbnQoKTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotUmVnaW9uIExheWVyOjpwb3N0KHVpbnQzMl90KiBwcmV2aW91c1NhdGUsIGJvb2wmIHJlY29tcHV0ZVZpc2libGVSZWdpb25zKQotewotICAgIC8vIGF0b21pY2FsbHkgc3dhcCBidWZmZXJzIGFuZCAocmUpc2V0IGVGbGlwUmVxdWVzdGVkCi0gICAgaW50MzJfdCBvbGRWYWx1ZSwgbmV3VmFsdWU7Ci0gICAgbGF5ZXJfY2Jsa190ICogY29uc3QgbGNibGsgPSB0aGlzLT5sY2JsazsKLSAgICBkbyB7Ci0gICAgICAgIG9sZFZhbHVlID0gbGNibGstPnN3YXBTdGF0ZTsKLSAgICAgICAgICAgIC8vIGdldCB0aGUgY3VycmVudCB2YWx1ZQotCi0gICAgICAgIExPR19BU1NFUlQob2xkVmFsdWUmZUZsaXBSZXF1ZXN0ZWQsCi0gICAgICAgICAgICAiZUZsaXBSZXF1ZXN0ZWQgbm90IHNldCwgeWV0IHdlJ3JlIGZsaXBwaW5nISAoc3RhdGU9MHglMDhseCkiLAotICAgICAgICAgICAgbG9uZyhvbGRWYWx1ZSkpOwotCi0gICAgICAgIG5ld1ZhbHVlID0gKG9sZFZhbHVlIF4gZUluZGV4KTsKLSAgICAgICAgICAgIC8vIHN3YXAgYnVmZmVycwotCi0gICAgICAgIG5ld1ZhbHVlICY9IH4oZUZsaXBSZXF1ZXN0ZWQgfCBlTmV4dEZsaXBQZW5kaW5nKTsKLSAgICAgICAgICAgIC8vIGNsZWFyIGVGbGlwUmVxdWVzdGVkIGFuZCBlTmV4dEZsaXBQZW5kaW5nCi0KLSAgICAgICAgaWYgKG9sZFZhbHVlICYgZU5leHRGbGlwUGVuZGluZykKLSAgICAgICAgICAgIG5ld1ZhbHVlIHw9IGVGbGlwUmVxdWVzdGVkOwotICAgICAgICAgICAgLy8gaWYgZU5leHRGbGlwUGVuZGluZyBpcyBzZXQgKHNlY29uZCBidWZmZXIgYWxyZWFkeSBoYXMgc29tZXRoaW5nCi0gICAgICAgICAgICAvLyBpbiBpdCkgd2UgbmVlZCB0byByZXNldCBlRmxpcFJlcXVlc3RlZCBiZWNhdXNlIHRoZSBjbGllbnQKLSAgICAgICAgICAgIC8vIG1pZ2h0IG5ldmVyIGRvIGl0Ci0KLSAgICB9IHdoaWxlKGFuZHJvaWRfYXRvbWljX2NtcHhjaGcob2xkVmFsdWUsIG5ld1ZhbHVlLCAmKGxjYmxrLT5zd2FwU3RhdGUpKSk7Ci0gICAgKnByZXZpb3VzU2F0ZSA9IG9sZFZhbHVlOwotCi0gICAgY29uc3QgaW50MzJfdCBpbmRleCA9IChuZXdWYWx1ZSAmIGVJbmRleCkgXiAxOwotICAgIG1Gcm9udEJ1ZmZlckluZGV4ID0gaW5kZXg7Ci0KLSAgICAvLyAuLi4gcG9zdCB0aGUgbmV3IGZyb250LWJ1ZmZlcgotICAgIFJlZ2lvbiBkaXJ0eShsY2Jsay0+cmVnaW9uICsgaW5kZXgpOwotICAgIGRpcnR5LmFuZFNlbGYoZnJvbnRCdWZmZXIoKS5ib3VuZHMoKSk7Ci0KLSAgICAvL0xPR0koIkRpZCBwb3N0IG9sZFZhbHVlPSUwOGx4LCBuZXdWYWx1ZT0lMDhseCwgbUZyb250QnVmZmVySW5kZXg9JXVcbiIsCi0gICAgLy8gICAgb2xkVmFsdWUsIG5ld1ZhbHVlLCBtRnJvbnRCdWZmZXJJbmRleCk7Ci0gICAgLy9kaXJ0eS5kdW1wKCJkaXJ0eSIpOwotCi0gICAgaWYgKFVOTElLRUxZKG9sZFZhbHVlICYgZVJlc2l6ZVJlcXVlc3RlZCkpIHsKLQotICAgICAgICBMT0dEX0lGKERFQlVHX1JFU0laRSwKLSAgICAgICAgICAgICAgICAgICAgICJwb3N0IChsYXllcj0lcCksIHN0YXRlPSUwOHgsICIKLSAgICAgICAgICAgICAgICAgICAgICJpbmRleD0lZCwgKCVkeCVkKSwgKCVkeCVkKSIsCi0gICAgICAgICAgICAgICAgICAgICB0aGlzLCAgbmV3VmFsdWUsCi0gICAgICAgICAgICAgICAgICAgICBpbnQoMS1pbmRleCksCi0gICAgICAgICAgICAgICAgICAgICBpbnQobUJ1ZmZlcnNbMF0ud2lkdGgoKSksIGludChtQnVmZmVyc1swXS5oZWlnaHQoKSksCi0gICAgICAgICAgICAgICAgICAgICBpbnQobUJ1ZmZlcnNbMV0ud2lkdGgoKSksIGludChtQnVmZmVyc1sxXS5oZWlnaHQoKSkpOwotCi0gICAgICAgIC8vIGhlcmUsIHdlIGp1c3QgcG9zdGVkIHRoZSBzdXJmYWNlIGFuZCB3ZSBoYXZlIHJlc29sdmVkCi0gICAgICAgIC8vIHRoZSBmcm9udC9iYWNrIGJ1ZmZlciBpbmRpY2VzLiBUaGUgY2xpZW50IGlzIGJsb2NrZWQsIHNvCi0gICAgICAgIC8vIGl0IGNhbm5vdCBzdGFydCB1c2luZyB0aGUgbmV3IGJhY2tidWZmZXIuCi0KLSAgICAgICAgLy8gSWYgdGhlIGJhY2tidWZmZXIgd2FzIHJlc2l6ZWQgaW4gVEhJUyByb3VuZCwgd2UgYWN0dWFsbHkgY2Fubm90Ci0gICAgICAgIC8vIHJlc2l6ZSB0aGUgZnJvbnRidWZmZXIgYmVjYXVzZSBpdCBoYXMgKmp1c3QqIGJlZW4gZHJhd24gKGFuZCB3ZQotICAgICAgICAvLyB3b3VsZCBoYXZlIG5vdGhpbmcgdG8gZHJhdykuIEluIHRoaXMgY2FzZSB3ZSBqdXN0IHNraXAgdGhlIHJlc2l6ZQotICAgICAgICAvLyBpdCdsbCBoYXBwZW4gYWZ0ZXIgdGhlIG5leHQgcGFnZSBmbGlwIG9yIGR1cmluZyB0aGUgbmV4dAotICAgICAgICAvLyB0cmFuc2FjdGlvbi4KLQotICAgICAgICBjb25zdCB1aW50MzJfdCBtYXNrID0gKDEtaW5kZXgpID8gZVJlc2l6ZUJ1ZmZlcjEgOiBlUmVzaXplQnVmZmVyMDsKLSAgICAgICAgaWYgKG1SZXNpemVUcmFuc2FjdGlvbkRvbmUgJiYgKG5ld1ZhbHVlICYgbWFzaykpIHsKLSAgICAgICAgICAgIC8vIFJlc2l6ZSB0aGUgbGF5ZXIncyBzZWNvbmQgYnVmZmVyIG9ubHkgaWYgdGhlIHRyYW5zYWN0aW9uCi0gICAgICAgICAgICAvLyBoYXBwZW5lZC4gSXQgbWF5IG5vdCBoYXZlIGhhcHBlbmVkIHlldCBpZiBlUmVzaXplUmVxdWVzdGVkCi0gICAgICAgICAgICAvLyB3YXMgc2V0IGltbWVkaWF0ZWx5IGFmdGVyIHRoZSAidHJhbnNhY3Rpb25SZXF1ZXN0ZWQiIHRlc3QsCi0gICAgICAgICAgICAvLyBpbiB3aGljaCBjYXNlIHRoZSBkcmF3aW5nIHN0YXRlJ3Mgc2l6ZSB3b3VsZCBiZSB3cm9uZy4KLSAgICAgICAgICAgIG1GcmVlemVMb2NrLmNsZWFyKCk7Ci0gICAgICAgICAgICBjb25zdCBMYXllcjo6U3RhdGUmIHMoZHJhd2luZ1N0YXRlKCkpOwotICAgICAgICAgICAgaWYgKHJlc2l6ZSgxLWluZGV4LCBzLncsIHMuaCwgInBvc3QiKSA9PSBOT19FUlJPUikgewotICAgICAgICAgICAgICAgIGRvIHsKLSAgICAgICAgICAgICAgICAgICAgb2xkVmFsdWUgPSBsY2Jsay0+c3dhcFN0YXRlOwotICAgICAgICAgICAgICAgICAgICBpZiAoKG9sZFZhbHVlICYgZVJlc2l6ZVJlcXVlc3RlZCkgPT0gZVJlc2l6ZVJlcXVlc3RlZCkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gdWdoLCBhbm90aGVyIHJlc2l6ZSB3YXMgcmVxdWVzdGVkIHNpbmNlIHdlIHByb2Nlc3NlZAotICAgICAgICAgICAgICAgICAgICAgICAgLy8gdGhlIGZpcnN0IGJ1ZmZlciwgZG9uJ3QgZnJlZSB0aGUgY2xpZW50LCBhbmQgbGV0Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyB0aGUgbmV4dCB0cmFuc2FjdGlvbiBoYW5kbGUgZXZlcnl0aGluZy4KLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIG5ld1ZhbHVlID0gb2xkVmFsdWUgJiB+bWFzazsKLSAgICAgICAgICAgICAgICB9IHdoaWxlKGFuZHJvaWRfYXRvbWljX2NtcHhjaGcob2xkVmFsdWUsIG5ld1ZhbHVlLCAmKGxjYmxrLT5zd2FwU3RhdGUpKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBtUmVzaXplVHJhbnNhY3Rpb25Eb25lID0gZmFsc2U7Ci0gICAgICAgICAgICByZWNvbXB1dGVWaXNpYmxlUmVnaW9ucyA9IHRydWU7Ci0gICAgICAgICAgICB0aGlzLT5jb250ZW50RGlydHkgPSB0cnVlOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcmVsb2FkVGV4dHVyZShkaXJ0eSk7Ci0KLSAgICByZXR1cm4gZGlydHk7Ci19Ci0KLVBvaW50IExheWVyOjpnZXRQaHlzaWNhbFNpemUoKSBjb25zdAotewotICAgIGNvbnN0IExheWVyQml0bWFwJiBmcm9udChmcm9udEJ1ZmZlcigpKTsKLSAgICByZXR1cm4gUG9pbnQoZnJvbnQud2lkdGgoKSwgZnJvbnQuaGVpZ2h0KCkpOwotfQotCi12b2lkIExheWVyOjp1bmxvY2tQYWdlRmxpcCgKLSAgICAgICAgY29uc3QgVHJhbnNmb3JtJiBwbGFuZVRyYW5zZm9ybSwgUmVnaW9uJiBvdXREaXJ0eVJlZ2lvbikKLXsKLSAgICBSZWdpb24gZGlydHlSZWdpb24obVBvc3RlZERpcnR5UmVnaW9uKTsKLSAgICBpZiAoIWRpcnR5UmVnaW9uLmlzRW1wdHkoKSkgewotICAgICAgICBtUG9zdGVkRGlydHlSZWdpb24uY2xlYXIoKTsKLSAgICAgICAgLy8gVGhlIGRpcnR5IHJlZ2lvbiBpcyBnaXZlbiBpbiB0aGUgbGF5ZXIncyBjb29yZGluYXRlIHNwYWNlCi0gICAgICAgIC8vIHRyYW5zZm9ybSB0aGUgZGlydHkgcmVnaW9uIGJ5IHRoZSBzdXJmYWNlJ3MgdHJhbnNmb3JtYXRpb24KLSAgICAgICAgLy8gYW5kIHRoZSBnbG9iYWwgdHJhbnNmb3JtYXRpb24uCi0gICAgICAgIGNvbnN0IExheWVyOjpTdGF0ZSYgcyhkcmF3aW5nU3RhdGUoKSk7Ci0gICAgICAgIGNvbnN0IFRyYW5zZm9ybSB0cihwbGFuZVRyYW5zZm9ybSAqIHMudHJhbnNmb3JtKTsKLSAgICAgICAgZGlydHlSZWdpb24gPSB0ci50cmFuc2Zvcm0oZGlydHlSZWdpb24pOwotCi0gICAgICAgIC8vIEF0IHRoaXMgcG9pbnQsIHRoZSBkaXJ0eSByZWdpb24gaXMgaW4gc2NyZWVuIHNwYWNlLgotICAgICAgICAvLyBNYWtlIHN1cmUgaXQncyBjb25zdHJhaW5lZCBieSB0aGUgdmlzaWJsZSByZWdpb24gKHdoaWNoCi0gICAgICAgIC8vIGlzIGluIHNjcmVlbiBzcGFjZSBhcyB3ZWxsKS4KLSAgICAgICAgZGlydHlSZWdpb24uYW5kU2VsZih2aXNpYmxlUmVnaW9uU2NyZWVuKTsKLSAgICAgICAgb3V0RGlydHlSZWdpb24ub3JTZWxmKGRpcnR5UmVnaW9uKTsKLQotICAgICAgICAvLyBjbGllbnQgY291bGQgYmUgYmxvY2tlZCwgc28gc2lnbmFsIHRoZW0gc28gdGhleSBnZXQgYQotICAgICAgICAvLyBjaGFuY2UgdG8gcmVldmFsdWF0ZSB0aGVpciBjb25kaXRpb24uCi0gICAgICAgIG1GbGluZ2VyLT5zY2hlZHVsZUJyb2FkY2FzdChjbGllbnQpOwotICAgIH0KLX0KLQotdm9pZCBMYXllcjo6ZmluaXNoUGFnZUZsaXAoKQotewotICAgIGlmIChMSUtFTFkoIShsY2Jsay0+c3dhcFN0YXRlICYgZUludmFsaWRTdXJmYWNlKSkpIHsKLSAgICAgICAgTE9HRV9JRighKGxjYmxrLT5zd2FwU3RhdGUgJiBlQnVzeSksCi0gICAgICAgICAgICAgICAgImxheWVyICVwIHdhc24ndCBsb2NrZWQhIiwgdGhpcyk7Ci0gICAgICAgIGFuZHJvaWRfYXRvbWljX2FuZCh+ZUJ1c3ksICYobGNibGstPnN3YXBTdGF0ZSkpOwotICAgIH0KLSAgICBtRmxpbmdlci0+c2NoZWR1bGVCcm9hZGNhc3QoY2xpZW50KTsKLX0KLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllci5oIGIvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyODY3ZjJiLi4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXIuaAorKysgL2Rldi9udWxsCkBAIC0xLDEyMCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0xBWUVSX0gKLSNkZWZpbmUgQU5EUk9JRF9MQVlFUl9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0KLSNpbmNsdWRlIDxwcml2YXRlL3VpL1NoYXJlZFN0YXRlLmg+Ci0jaW5jbHVkZSA8cHJpdmF0ZS91aS9MYXllclN0YXRlLmg+Ci0KLSNpbmNsdWRlIDxwaXhlbGZsaW5nZXIvcGl4ZWxmbGluZ2VyLmg+Ci0KLSNpbmNsdWRlICJMYXllckJpdG1hcC5oIgotI2luY2x1ZGUgIkxheWVyQmFzZS5oIgotI2luY2x1ZGUgIlRyYW5zZm9ybS5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBDbGllbnQ7Ci1jbGFzcyBMYXllckJpdG1hcDsKLWNsYXNzIE1lbW9yeURlYWxlcjsKLWNsYXNzIEZyZWV6ZUxvY2s7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBMYXllciA6IHB1YmxpYyBMYXllckJhc2VDbGllbnQKLXsKLXB1YmxpYzogICAgCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IHR5cGVJbmZvOwotICAgIHN0YXRpYyBjb25zdCBjaGFyKiBjb25zdCB0eXBlSUQ7Ci0gICAgdmlydHVhbCBjaGFyIGNvbnN0KiBnZXRUeXBlSUQoKSBjb25zdCB7IHJldHVybiB0eXBlSUQ7IH0KLSAgICB2aXJ0dWFsIHVpbnQzMl90IGdldFR5cGVJbmZvKCkgY29uc3QgeyByZXR1cm4gdHlwZUluZm87IH0KLQotICAgICAgICAgICAgICAgICBMYXllcihTdXJmYWNlRmxpbmdlciogZmxpbmdlciwgRGlzcGxheUlEIGRpc3BsYXksCi0gICAgICAgICAgICAgICAgICAgICAgICAgQ2xpZW50KiBjLCBpbnQzMl90IGkpOwotCi0gICAgICAgIHZpcnR1YWwgfkxheWVyKCk7Ci0KLSAgICBpbmxpbmUgUGl4ZWxGb3JtYXQgcGl4ZWxGb3JtYXQoKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiBmcm9udEJ1ZmZlcigpLnBpeGVsRm9ybWF0KCk7Ci0gICAgfQotCi0gICAgc3RhdHVzX3Qgc2V0QnVmZmVycyggICAgQ2xpZW50KiBjbGllbnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgdywgdWludDMyX3QgaCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBQaXhlbEZvcm1hdCBmb3JtYXQsIHVpbnQzMl90IGZsYWdzPTApOwotCi0gICAgdmlydHVhbCB2b2lkIG9uRHJhdyhjb25zdCBSZWdpb24mIGNsaXApIGNvbnN0OwotICAgIHZpcnR1YWwgdm9pZCBpbml0U3RhdGVzKHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIHVpbnQzMl90IGZsYWdzKTsKLSAgICB2aXJ0dWFsIHZvaWQgc2V0U2l6ZUNoYW5nZWQodWludDMyX3QgdywgdWludDMyX3QgaCk7Ci0gICAgdmlydHVhbCB1aW50MzJfdCBkb1RyYW5zYWN0aW9uKHVpbnQzMl90IHRyYW5zYWN0aW9uRmxhZ3MpOwotICAgIHZpcnR1YWwgUG9pbnQgZ2V0UGh5c2ljYWxTaXplKCkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkIGxvY2tQYWdlRmxpcChib29sJiByZWNvbXB1dGVWaXNpYmxlUmVnaW9ucyk7Ci0gICAgdmlydHVhbCB2b2lkIHVubG9ja1BhZ2VGbGlwKGNvbnN0IFRyYW5zZm9ybSYgcGxhbmVUcmFuc2Zvcm0sIFJlZ2lvbiYgb3V0RGlydHlSZWdpb24pOwotICAgIHZpcnR1YWwgdm9pZCBmaW5pc2hQYWdlRmxpcCgpOwotICAgIHZpcnR1YWwgYm9vbCBuZWVkc0JsZW5kaW5nKCkgY29uc3QgICAgICB7IHJldHVybiBtTmVlZHNCbGVuZGluZzsgfQotICAgIHZpcnR1YWwgYm9vbCBpc1NlY3VyZSgpIGNvbnN0ICAgICAgICAgICB7IHJldHVybiBtU2VjdXJlOyB9Ci0gICAgdmlydHVhbCBHTHVpbnQgZ2V0VGV4dHVyZU5hbWUoKSBjb25zdCAgIHsgcmV0dXJuIG1UZXh0dXJlTmFtZTsgfQotICAgIHZpcnR1YWwgc3A8U3VyZmFjZT4gZ2V0U3VyZmFjZSgpIGNvbnN0OwotCi0gICAgY29uc3QgTGF5ZXJCaXRtYXAmIGdldEJ1ZmZlcihpbnQgaSkgY29uc3QgeyByZXR1cm4gbUJ1ZmZlcnNbaV07IH0KLSAgICAgICAgICBMYXllckJpdG1hcCYgZ2V0QnVmZmVyKGludCBpKSAgICAgICB7IHJldHVybiBtQnVmZmVyc1tpXTsgfQotCi0gICAgLy8gb25seSBmb3IgZGVidWdnaW5nCi0gICAgY29uc3Qgc3A8RnJlZXplTG9jaz4mICBnZXRGcmVlemVMb2NrKCkgY29uc3QgeyByZXR1cm4gbUZyZWV6ZUxvY2s7IH0KLQotcHJpdmF0ZToKLSAgICBpbmxpbmUgY29uc3QgTGF5ZXJCaXRtYXAmCi0gICAgICAgICAgICBmcm9udEJ1ZmZlcigpIGNvbnN0IHsgcmV0dXJuIGdldEJ1ZmZlcihtRnJvbnRCdWZmZXJJbmRleCk7IH0KLSAgICBpbmxpbmUgTGF5ZXJCaXRtYXAmCi0gICAgICAgICAgICBmcm9udEJ1ZmZlcigpICAgICAgIHsgcmV0dXJuIGdldEJ1ZmZlcihtRnJvbnRCdWZmZXJJbmRleCk7IH0KLSAgICBpbmxpbmUgY29uc3QgTGF5ZXJCaXRtYXAmCi0gICAgICAgICAgICBiYWNrQnVmZmVyKCkgY29uc3QgIHsgcmV0dXJuIGdldEJ1ZmZlcigxLW1Gcm9udEJ1ZmZlckluZGV4KTsgfQotICAgIGlubGluZSBMYXllckJpdG1hcCYKLSAgICAgICAgICAgIGJhY2tCdWZmZXIoKSAgICAgICAgeyByZXR1cm4gZ2V0QnVmZmVyKDEtbUZyb250QnVmZmVySW5kZXgpOyB9Ci0KLSAgICB2b2lkIHJlbG9hZFRleHR1cmUoY29uc3QgUmVnaW9uJiBkaXJ0eSk7Ci0KLSAgICBzdGF0dXNfdCByZXNpemUoaW50MzJfdCBpbmRleCwgdWludDMyX3QgdywgdWludDMyX3QgaCwgY29uc3QgY2hhciogd2hhdCk7Ci0gICAgUmVnaW9uIHBvc3QodWludDMyX3QqIG9sZFN0YXRlLCBib29sJiByZWNvbXB1dGVWaXNpYmxlUmVnaW9ucyk7Ci0gICAgc3RhdHVzX3QgcmVhbGxvY2F0ZUJ1ZmZlcihpbnQzMl90IGluZGV4LCB1aW50MzJfdCB3LCB1aW50MzJfdCBoKTsKLQotICAgIHNwPFN1cmZhY2U+ICAgICAgICAgICAgIG1TdXJmYWNlOwotCi0gICAgICAgICAgICBib29sICAgICAgICAgICAgbVNlY3VyZTsKLSAgICAgICAgICAgIExheWVyQml0bWFwICAgICBtQnVmZmVyc1syXTsKLSAgICAgICAgICAgIGludDMyX3QgICAgICAgICBtRnJvbnRCdWZmZXJJbmRleDsKLSAgICAgICAgICAgIGJvb2wgICAgICAgICAgICBtTmVlZHNCbGVuZGluZzsKLSAgICAgICAgICAgIGJvb2wgICAgICAgICAgICBtUmVzaXplVHJhbnNhY3Rpb25Eb25lOwotICAgICAgICAgICAgUmVnaW9uICAgICAgICAgIG1Qb3N0ZWREaXJ0eVJlZ2lvbjsKLSAgICAgICAgICAgIHNwPEZyZWV6ZUxvY2s+ICBtRnJlZXplTG9jazsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgR0x1aW50ICAgICAgICAgIG1UZXh0dXJlTmFtZTsKLSAgICAgICAgICAgIEdMdWludCAgICAgICAgICBtVGV4dHVyZVdpZHRoOwotICAgICAgICAgICAgR0x1aW50ICAgICAgICAgIG1UZXh0dXJlSGVpZ2h0OwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0xBWUVSX0gKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCYXNlLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCYXNlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMGNmNTNmNy4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyQmFzZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw3NDAgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNkZWZpbmUgTE9HX1RBRyAiU3VyZmFjZUZsaW5nZXIiCi0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8R0xFUy9nbC5oPgotI2luY2x1ZGUgPEdMRVMvZ2xleHQuaD4KLQotI2luY2x1ZGUgPGhhcmR3YXJlL2hhcmR3YXJlLmg+Ci0KLSNpbmNsdWRlICJjbHouaCIKLSNpbmNsdWRlICJMYXllckJhc2UuaCIKLSNpbmNsdWRlICJMYXllckJsdXIuaCIKLSNpbmNsdWRlICJTdXJmYWNlRmxpbmdlci5oIgotI2luY2x1ZGUgIkRpc3BsYXlIYXJkd2FyZS9EaXNwbGF5SGFyZHdhcmUuaCIKLQotCi0vLyBXZSBkb24ndCBob25vciB0aGUgcHJlbXVsdGlwbGllZCBhbHBoYSBmbGFncywgd2hpY2ggbWVhbnMgdGhhdAotLy8gcHJlbXVsdGlwbGllZCBzdXJmYWNlIG1heSBiZSBjb21wb3NlZCB1c2luZyBhIG5vbi1wcmVtdWx0aXBsaWVkCi0vLyBlcXVhdGlvbi4gV2UgZG8gdGhpcyBiZWNhdXNlIGl0IG1heSBiZSBhIGxvdCBmYXN0ZXIgb24gc29tZSBoYXJkd2FyZQotLy8gVGhlIGNvcnJlY3QgdmFsdWUgaXMgSE9OT1JfUFJFTVVMVElQTElFRF9BTFBIQSA9IDEKLSNkZWZpbmUgSE9OT1JfUFJFTVVMVElQTElFRF9BTFBIQSAgIDAKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY29uc3QgdWludDMyX3QgTGF5ZXJCYXNlOjp0eXBlSW5mbyA9IDE7Ci1jb25zdCBjaGFyKiBjb25zdCBMYXllckJhc2U6OnR5cGVJRCA9ICJMYXllckJhc2UiOwotCi1jb25zdCB1aW50MzJfdCBMYXllckJhc2VDbGllbnQ6OnR5cGVJbmZvID0gTGF5ZXJCYXNlOjp0eXBlSW5mbyB8IDI7Ci1jb25zdCBjaGFyKiBjb25zdCBMYXllckJhc2VDbGllbnQ6OnR5cGVJRCA9ICJMYXllckJhc2VDbGllbnQiOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotVmVjdG9yPEdMdWludD4gTGF5ZXJCYXNlOjpkZWxldGVkVGV4dHVyZXM7IAotCi1pbnQzMl90IExheWVyQmFzZTo6c0lkZW50aXR5ID0gMDsKLQotTGF5ZXJCYXNlOjpMYXllckJhc2UoU3VyZmFjZUZsaW5nZXIqIGZsaW5nZXIsIERpc3BsYXlJRCBkaXNwbGF5KQotICAgIDogZHB5KGRpc3BsYXkpLCBjb250ZW50RGlydHkoZmFsc2UpLAotICAgICAgbUZsaW5nZXIoZmxpbmdlciksCi0gICAgICBtVHJhbnNmb3JtZWQoZmFsc2UpLAotICAgICAgbU9yaWVudGF0aW9uKDApLAotICAgICAgbUNhblVzZUNvcHlCaXQoZmFsc2UpLAotICAgICAgbVRyYW5zYWN0aW9uRmxhZ3MoMCksCi0gICAgICBtUHJlbXVsdGlwbGllZEFscGhhKHRydWUpLAotICAgICAgbUlkZW50aXR5KHVpbnQzMl90KGFuZHJvaWRfYXRvbWljX2luYygmc0lkZW50aXR5KSkpLAotICAgICAgbUludmFsaWRhdGUoMCkKLXsKLSAgICBjb25zdCBEaXNwbGF5SGFyZHdhcmUmIGh3KGZsaW5nZXItPmdyYXBoaWNQbGFuZSgwKS5kaXNwbGF5SGFyZHdhcmUoKSk7Ci0gICAgbUZsYWdzID0gaHcuZ2V0RmxhZ3MoKTsKLX0KLQotTGF5ZXJCYXNlOjp+TGF5ZXJCYXNlKCkKLXsKLX0KLQotY29uc3QgR3JhcGhpY1BsYW5lJiBMYXllckJhc2U6OmdyYXBoaWNQbGFuZShpbnQgZHB5KSBjb25zdAoteyAKLSAgICByZXR1cm4gbUZsaW5nZXItPmdyYXBoaWNQbGFuZShkcHkpOwotfQotCi1HcmFwaGljUGxhbmUmIExheWVyQmFzZTo6Z3JhcGhpY1BsYW5lKGludCBkcHkpCi17Ci0gICAgcmV0dXJuIG1GbGluZ2VyLT5ncmFwaGljUGxhbmUoZHB5KTsgCi19Ci0KLXZvaWQgTGF5ZXJCYXNlOjppbml0U3RhdGVzKHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIHVpbnQzMl90IGZsYWdzKQotewotICAgIHVpbnQzMl90IGxheWVyRmxhZ3MgPSAwOwotICAgIGlmIChmbGFncyAmIElTdXJmYWNlQ29tcG9zZXI6OmVIaWRkZW4pCi0gICAgICAgIGxheWVyRmxhZ3MgPSBJU3VyZmFjZUNvbXBvc2VyOjplTGF5ZXJIaWRkZW47Ci0KLSAgICBpZiAoZmxhZ3MgJiBJU3VyZmFjZUNvbXBvc2VyOjplTm9uUHJlbXVsdGlwbGllZCkKLSAgICAgICAgbVByZW11bHRpcGxpZWRBbHBoYSA9IGZhbHNlOwotCi0gICAgbUN1cnJlbnRTdGF0ZS56ICAgICAgICAgPSAwOwotICAgIG1DdXJyZW50U3RhdGUudyAgICAgICAgID0gdzsKLSAgICBtQ3VycmVudFN0YXRlLmggICAgICAgICA9IGg7Ci0gICAgbUN1cnJlbnRTdGF0ZS5hbHBoYSAgICAgPSAweEZGOwotICAgIG1DdXJyZW50U3RhdGUuZmxhZ3MgICAgID0gbGF5ZXJGbGFnczsKLSAgICBtQ3VycmVudFN0YXRlLnNlcXVlbmNlICA9IDA7Ci0gICAgbUN1cnJlbnRTdGF0ZS50cmFuc2Zvcm0uc2V0KDAsIDApOwotCi0gICAgLy8gZHJhd2luZyBzdGF0ZSAmIGN1cnJlbnQgc3RhdGUgYXJlIGlkZW50aWNhbAotICAgIG1EcmF3aW5nU3RhdGUgPSBtQ3VycmVudFN0YXRlOwotfQotCi12b2lkIExheWVyQmFzZTo6Y29tbWl0VHJhbnNhY3Rpb24oYm9vbCBza2lwU2l6ZSkgewotICAgIGNvbnN0IHVpbnQzMl90IHcgPSBtRHJhd2luZ1N0YXRlLnc7Ci0gICAgY29uc3QgdWludDMyX3QgaCA9IG1EcmF3aW5nU3RhdGUuaDsKLSAgICBtRHJhd2luZ1N0YXRlID0gbUN1cnJlbnRTdGF0ZTsKLSAgICBpZiAoc2tpcFNpemUpIHsKLSAgICAgICAgbURyYXdpbmdTdGF0ZS53ID0gdzsKLSAgICAgICAgbURyYXdpbmdTdGF0ZS5oID0gaDsKLSAgICB9Ci19Ci12b2lkIExheWVyQmFzZTo6Zm9yY2VWaXNpYmlsaXR5VHJhbnNhY3Rpb24oKSB7Ci0gICAgLy8gdGhpcyBjYW4gYmUgY2FsbGVkIHdpdGhvdXQgU3VyZmFjZUZsaW5nZXIubVN0YXRlTG9jaywgYnV0IGlmIHdlCi0gICAgLy8gY2FuIGF0b21pY2FsbHkgaW5jcmVtZW50IHRoZSBzZXF1ZW5jZSBudW1iZXIsIGl0IGRvZXNuJ3QgbWF0dGVyLgotICAgIGFuZHJvaWRfYXRvbWljX2luYygmbUN1cnJlbnRTdGF0ZS5zZXF1ZW5jZSk7Ci0gICAgcmVxdWVzdFRyYW5zYWN0aW9uKCk7Ci19Ci1ib29sIExheWVyQmFzZTo6cmVxdWVzdFRyYW5zYWN0aW9uKCkgewotICAgIGludDMyX3Qgb2xkID0gc2V0VHJhbnNhY3Rpb25GbGFncyhlVHJhbnNhY3Rpb25OZWVkZWQpOwotICAgIHJldHVybiAoKG9sZCAmIGVUcmFuc2FjdGlvbk5lZWRlZCkgPT0gMCk7Ci19Ci11aW50MzJfdCBMYXllckJhc2U6OmdldFRyYW5zYWN0aW9uRmxhZ3ModWludDMyX3QgZmxhZ3MpIHsKLSAgICByZXR1cm4gYW5kcm9pZF9hdG9taWNfYW5kKH5mbGFncywgJm1UcmFuc2FjdGlvbkZsYWdzKSAmIGZsYWdzOwotfQotdWludDMyX3QgTGF5ZXJCYXNlOjpzZXRUcmFuc2FjdGlvbkZsYWdzKHVpbnQzMl90IGZsYWdzKSB7Ci0gICAgcmV0dXJuIGFuZHJvaWRfYXRvbWljX29yKGZsYWdzLCAmbVRyYW5zYWN0aW9uRmxhZ3MpOwotfQotCi12b2lkIExheWVyQmFzZTo6c2V0U2l6ZUNoYW5nZWQodWludDMyX3QgdywgdWludDMyX3QgaCkgewotfQotCi1ib29sIExheWVyQmFzZTo6c2V0UG9zaXRpb24oaW50MzJfdCB4LCBpbnQzMl90IHkpIHsKLSAgICBpZiAobUN1cnJlbnRTdGF0ZS50cmFuc2Zvcm0udHgoKSA9PSB4ICYmIG1DdXJyZW50U3RhdGUudHJhbnNmb3JtLnR5KCkgPT0geSkKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIG1DdXJyZW50U3RhdGUuc2VxdWVuY2UrKzsKLSAgICBtQ3VycmVudFN0YXRlLnRyYW5zZm9ybS5zZXQoeCwgeSk7Ci0gICAgcmVxdWVzdFRyYW5zYWN0aW9uKCk7Ci0gICAgcmV0dXJuIHRydWU7Ci19Ci1ib29sIExheWVyQmFzZTo6c2V0TGF5ZXIodWludDMyX3QgeikgewotICAgIGlmIChtQ3VycmVudFN0YXRlLnogPT0geikKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIG1DdXJyZW50U3RhdGUuc2VxdWVuY2UrKzsKLSAgICBtQ3VycmVudFN0YXRlLnogPSB6OwotICAgIHJlcXVlc3RUcmFuc2FjdGlvbigpOwotICAgIHJldHVybiB0cnVlOwotfQotYm9vbCBMYXllckJhc2U6OnNldFNpemUodWludDMyX3QgdywgdWludDMyX3QgaCkgewotICAgIGlmIChtQ3VycmVudFN0YXRlLncgPT0gdyAmJiBtQ3VycmVudFN0YXRlLmggPT0gaCkKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIHNldFNpemVDaGFuZ2VkKHcsIGgpOwotICAgIG1DdXJyZW50U3RhdGUudyA9IHc7Ci0gICAgbUN1cnJlbnRTdGF0ZS5oID0gaDsKLSAgICByZXF1ZXN0VHJhbnNhY3Rpb24oKTsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLWJvb2wgTGF5ZXJCYXNlOjpzZXRBbHBoYSh1aW50OF90IGFscGhhKSB7Ci0gICAgaWYgKG1DdXJyZW50U3RhdGUuYWxwaGEgPT0gYWxwaGEpCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICBtQ3VycmVudFN0YXRlLnNlcXVlbmNlKys7Ci0gICAgbUN1cnJlbnRTdGF0ZS5hbHBoYSA9IGFscGhhOwotICAgIHJlcXVlc3RUcmFuc2FjdGlvbigpOwotICAgIHJldHVybiB0cnVlOwotfQotYm9vbCBMYXllckJhc2U6OnNldE1hdHJpeChjb25zdCBsYXllcl9zdGF0ZV90OjptYXRyaXgyMl90JiBtYXRyaXgpIHsKLSAgICAvLyBUT0RPOiBjaGVjayB0aGUgbWF0cml4IGhhcyBjaGFuZ2VkCi0gICAgbUN1cnJlbnRTdGF0ZS5zZXF1ZW5jZSsrOwotICAgIG1DdXJyZW50U3RhdGUudHJhbnNmb3JtLnNldCgKLSAgICAgICAgICAgIG1hdHJpeC5kc2R4LCBtYXRyaXguZHNkeSwgbWF0cml4LmR0ZHgsIG1hdHJpeC5kdGR5KTsKLSAgICByZXF1ZXN0VHJhbnNhY3Rpb24oKTsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLWJvb2wgTGF5ZXJCYXNlOjpzZXRUcmFuc3BhcmVudFJlZ2lvbkhpbnQoY29uc3QgUmVnaW9uJiB0cmFuc3BhcmVudCkgewotICAgIC8vIFRPRE86IGNoZWNrIHRoZSByZWdpb24gaGFzIGNoYW5nZWQKLSAgICBtQ3VycmVudFN0YXRlLnNlcXVlbmNlKys7Ci0gICAgbUN1cnJlbnRTdGF0ZS50cmFuc3BhcmVudFJlZ2lvbiA9IHRyYW5zcGFyZW50OwotICAgIHJlcXVlc3RUcmFuc2FjdGlvbigpOwotICAgIHJldHVybiB0cnVlOwotfQotYm9vbCBMYXllckJhc2U6OnNldEZsYWdzKHVpbnQ4X3QgZmxhZ3MsIHVpbnQ4X3QgbWFzaykgewotICAgIGNvbnN0IHVpbnQzMl90IG5ld0ZsYWdzID0gKG1DdXJyZW50U3RhdGUuZmxhZ3MgJiB+bWFzaykgfCAoZmxhZ3MgJiBtYXNrKTsKLSAgICBpZiAobUN1cnJlbnRTdGF0ZS5mbGFncyA9PSBuZXdGbGFncykKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIG1DdXJyZW50U3RhdGUuc2VxdWVuY2UrKzsKLSAgICBtQ3VycmVudFN0YXRlLmZsYWdzID0gbmV3RmxhZ3M7Ci0gICAgcmVxdWVzdFRyYW5zYWN0aW9uKCk7Ci0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLVJlY3QgTGF5ZXJCYXNlOjp2aXNpYmxlQm91bmRzKCkgY29uc3QKLXsKLSAgICByZXR1cm4gbVRyYW5zZm9ybWVkQm91bmRzOwotfSAgICAgIAotCi12b2lkIExheWVyQmFzZTo6c2V0VmlzaWJsZVJlZ2lvbihjb25zdCBSZWdpb24mIHZpc2libGVSZWdpb24pIHsKLSAgICAvLyBhbHdheXMgY2FsbGVkIGZyb20gbWFpbiB0aHJlYWQKLSAgICB2aXNpYmxlUmVnaW9uU2NyZWVuID0gdmlzaWJsZVJlZ2lvbjsKLX0KLQotdm9pZCBMYXllckJhc2U6OnNldENvdmVyZWRSZWdpb24oY29uc3QgUmVnaW9uJiBjb3ZlcmVkUmVnaW9uKSB7Ci0gICAgLy8gYWx3YXlzIGNhbGxlZCBmcm9tIG1haW4gdGhyZWFkCi0gICAgY292ZXJlZFJlZ2lvblNjcmVlbiA9IGNvdmVyZWRSZWdpb247Ci19Ci0KLXVpbnQzMl90IExheWVyQmFzZTo6ZG9UcmFuc2FjdGlvbih1aW50MzJfdCBmbGFncykKLXsKLSAgICBjb25zdCBMYXllcjo6U3RhdGUmIGZyb250KGRyYXdpbmdTdGF0ZSgpKTsKLSAgICBjb25zdCBMYXllcjo6U3RhdGUmIHRlbXAoY3VycmVudFN0YXRlKCkpOwotCi0gICAgaWYgKHRlbXAuc2VxdWVuY2UgIT0gZnJvbnQuc2VxdWVuY2UpIHsKLSAgICAgICAgLy8gaW52YWxpZGF0ZSBhbmQgcmVjb21wdXRlIHRoZSB2aXNpYmxlIHJlZ2lvbnMgaWYgbmVlZGVkCi0gICAgICAgIGZsYWdzIHw9IGVWaXNpYmxlUmVnaW9uOwotICAgICAgICB0aGlzLT5jb250ZW50RGlydHkgPSB0cnVlOwotICAgIH0KLSAgICAKLSAgICAvLyBDb21taXQgdGhlIHRyYW5zYWN0aW9uCi0gICAgY29tbWl0VHJhbnNhY3Rpb24oZmxhZ3MgJiBlUmVzdGFydFRyYW5zYWN0aW9uKTsKLSAgICByZXR1cm4gZmxhZ3M7Ci19Ci0KLVBvaW50IExheWVyQmFzZTo6Z2V0UGh5c2ljYWxTaXplKCkgY29uc3QKLXsKLSAgICBjb25zdCBMYXllcjo6U3RhdGUmIGZyb250KGRyYXdpbmdTdGF0ZSgpKTsKLSAgICByZXR1cm4gUG9pbnQoZnJvbnQudywgZnJvbnQuaCk7Ci19Ci0KLXZvaWQgTGF5ZXJCYXNlOjp2YWxpZGF0ZVZpc2liaWxpdHkoY29uc3QgVHJhbnNmb3JtJiBwbGFuZVRyYW5zZm9ybSkKLXsKLSAgICBjb25zdCBMYXllcjo6U3RhdGUmIHMoZHJhd2luZ1N0YXRlKCkpOwotICAgIGNvbnN0IFRyYW5zZm9ybSB0cihwbGFuZVRyYW5zZm9ybSAqIHMudHJhbnNmb3JtKTsKLSAgICBjb25zdCBib29sIHRyYW5zZm9ybWVkID0gdHIudHJhbnNmb3JtZWQoKTsKLSAgIAotICAgIGNvbnN0IFBvaW50IHNpemUoZ2V0UGh5c2ljYWxTaXplKCkpOwotICAgIHVpbnQzMl90IHcgPSBzaXplLng7Ci0gICAgdWludDMyX3QgaCA9IHNpemUueTsgICAgCi0gICAgdHIudHJhbnNmb3JtKG1WZXJ0aWNlc1swXSwgMCwgMCk7Ci0gICAgdHIudHJhbnNmb3JtKG1WZXJ0aWNlc1sxXSwgMCwgaCk7Ci0gICAgdHIudHJhbnNmb3JtKG1WZXJ0aWNlc1syXSwgdywgaCk7Ci0gICAgdHIudHJhbnNmb3JtKG1WZXJ0aWNlc1szXSwgdywgMCk7Ci0gICAgaWYgKFVOTElLRUxZKHRyYW5zZm9ybWVkKSkgewotICAgICAgICAvLyBOT1RFOiBoZXJlIHdlIGNvdWxkIGFsc28gcHVudCBpZiB3ZSBoYXZlIHRvbyBtYW55IHJlY3RhbmdsZXMKLSAgICAgICAgLy8gaW4gdGhlIHRyYW5zcGFyZW50IHJlZ2lvbgotICAgICAgICBpZiAodHIucHJlc2VydmVSZWN0cygpKSB7Ci0gICAgICAgICAgICAvLyB0cmFuc2Zvcm0gdGhlIHRyYW5zcGFyZW50IHJlZ2lvbgotICAgICAgICAgICAgdHJhbnNwYXJlbnRSZWdpb25TY3JlZW4gPSB0ci50cmFuc2Zvcm0ocy50cmFuc3BhcmVudFJlZ2lvbik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyB0cmFuc2Zvcm1hdGlvbiB0b28gY29tcGxleCwgY2FuJ3QgZG8gdGhlIHRyYW5zcGFyZW50IHJlZ2lvbgotICAgICAgICAgICAgLy8gb3B0aW1pemF0aW9uLgotICAgICAgICAgICAgdHJhbnNwYXJlbnRSZWdpb25TY3JlZW4uY2xlYXIoKTsKLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIHRyYW5zcGFyZW50UmVnaW9uU2NyZWVuID0gcy50cmFuc3BhcmVudFJlZ2lvbjsKLSAgICB9Ci0KLSAgICAvLyBjYWNoZSBhIGZldyB0aGluZ3MuLi4KLSAgICBtT3JpZW50YXRpb24gPSB0ci5nZXRPcmllbnRhdGlvbigpOwotICAgIG1UcmFuc2Zvcm1lZEJvdW5kcyA9IHRyLm1ha2VCb3VuZHModywgaCk7Ci0gICAgbVRyYW5zZm9ybWVkID0gdHJhbnNmb3JtZWQ7Ci0gICAgbUxlZnQgPSB0ci50eCgpOwotICAgIG1Ub3AgID0gdHIudHkoKTsKLQotICAgIC8vIHNlZSBpZiB3ZSBjYW4vc2hvdWxkIHVzZSAyRCBoL3cgd2l0aCB0aGUgbmV3IGNvbmZpZ3VyYXRpb24KLSAgICBtQ2FuVXNlQ29weUJpdCA9IGZhbHNlOwotICAgIGNvcHliaXRfZGV2aWNlX3QqIGNvcHliaXQgPSBtRmxpbmdlci0+Z2V0QmxpdEVuZ2luZSgpOwotICAgIGlmIChjb3B5Yml0KSB7IAotICAgICAgICBjb25zdCBpbnQgc3RlcCA9IGNvcHliaXQtPmdldChjb3B5Yml0LCBDT1BZQklUX1JPVEFUSU9OX1NURVBfREVHKTsKLSAgICAgICAgY29uc3QgaW50IHNjYWxlQml0cyA9IGNvcHliaXQtPmdldChjb3B5Yml0LCBDT1BZQklUX1NDQUxJTkdfRlJBQ19CSVRTKTsKLSAgICAgICAgbUNhblVzZUNvcHlCaXQgPSB0cnVlOwotICAgICAgICBpZiAoKG1PcmllbnRhdGlvbiA8IDApICYmIChzdGVwID4gMSkpIHsKLSAgICAgICAgICAgIC8vIGFyYml0cmFyeSBvcmllbnRhdGlvbnMgbm90IHN1cHBvcnRlZAotICAgICAgICAgICAgbUNhblVzZUNvcHlCaXQgPSBmYWxzZTsKLSAgICAgICAgfSBlbHNlIGlmICgobU9yaWVudGF0aW9uID4gMCkgJiYgKHN0ZXAgPiA5MCkpIHsKLSAgICAgICAgICAgIC8vIDkwIGRlZyByb3RhdGlvbnMgbm90IHN1cHBvcnRlZAotICAgICAgICAgICAgbUNhblVzZUNvcHlCaXQgPSBmYWxzZTsKLSAgICAgICAgfSBlbHNlIGlmICgodHIuZ2V0VHlwZSgpICYgU2tNYXRyaXg6OmtTY2FsZV9NYXNrKSAmJiAoc2NhbGVCaXRzIDwgMTIpKSB7IAotICAgICAgICAgICAgLy8gYXJiaXRyYXJ5IHNjYWxpbmcgbm90IHN1cHBvcnRlZAotICAgICAgICAgICAgbUNhblVzZUNvcHlCaXQgPSBmYWxzZTsKLSAgICAgICAgfQotI2lmIEhPTk9SX1BSRU1VTFRJUExJRURfQUxQSEEgCi0gICAgICAgIGVsc2UgaWYgKG5lZWRzQmxlbmRpbmcoKSAmJiBtUHJlbXVsdGlwbGllZEFscGhhKSB7Ci0gICAgICAgICAgICAvLyBwcmUtbXVsdGlwbGllZCBhbHBoYSBub3Qgc3VwcG9ydGVkCi0gICAgICAgICAgICBtQ2FuVXNlQ29weUJpdCA9IGZhbHNlOwotICAgICAgICB9Ci0jZW5kaWYKLSAgICAgICAgZWxzZSB7Ci0gICAgICAgICAgICAvLyBoZXJlLCB3ZSBkZXRlcm1pbmVkIHdlIGNhbiB1c2UgY29weWJpdAotICAgICAgICAgICAgaWYgKHRyLmdldFR5cGUoKSAmIFNrTWF0cml4OjprU2NhbGVfTWFzaykgewotICAgICAgICAgICAgICAgIC8vIGFuZCB3ZSBoYXZlIHNjYWxpbmcKLSAgICAgICAgICAgICAgICBpZiAoIXRyYW5zcGFyZW50UmVnaW9uU2NyZWVuLmlzUmVjdCgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIHdlIHB1bnQgYmVjYXVzZSBibGVuZGluZyBpcyBjaGVhcCAoaC93KSBhbmQgdGhlIHJlZ2lvbiBpcwotICAgICAgICAgICAgICAgICAgICAvLyBjb21wbGV4LCB3aGljaCBtYXkgY2F1c2VzIGFydGlmYWN0cyB3aGVuIGNvcHlpbmcKLSAgICAgICAgICAgICAgICAgICAgLy8gc2NhbGVkIGNvbnRlbnQKLSAgICAgICAgICAgICAgICAgICAgdHJhbnNwYXJlbnRSZWdpb25TY3JlZW4uY2xlYXIoKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLXZvaWQgTGF5ZXJCYXNlOjpsb2NrUGFnZUZsaXAoYm9vbCYgcmVjb21wdXRlVmlzaWJsZVJlZ2lvbnMpCi17Ci19Ci0KLXZvaWQgTGF5ZXJCYXNlOjp1bmxvY2tQYWdlRmxpcCgKLSAgICAgICAgY29uc3QgVHJhbnNmb3JtJiBwbGFuZVRyYW5zZm9ybSwgUmVnaW9uJiBvdXREaXJ0eVJlZ2lvbikKLXsKLSAgICBpZiAoKGFuZHJvaWRfYXRvbWljX2FuZCh+MSwgJm1JbnZhbGlkYXRlKSYxKSA9PSAxKSB7Ci0gICAgICAgIG91dERpcnR5UmVnaW9uLm9yU2VsZih2aXNpYmxlUmVnaW9uU2NyZWVuKTsKLSAgICB9Ci19Ci0KLXZvaWQgTGF5ZXJCYXNlOjpmaW5pc2hQYWdlRmxpcCgpCi17Ci19Ci0KLXZvaWQgTGF5ZXJCYXNlOjppbnZhbGlkYXRlKCkKLXsKLSAgICBpZiAoKGFuZHJvaWRfYXRvbWljX29yKDEsICZtSW52YWxpZGF0ZSkmMSkgPT0gMCkgewotICAgICAgICBtRmxpbmdlci0+c2lnbmFsRXZlbnQoKTsKLSAgICB9Ci19Ci0KLXZvaWQgTGF5ZXJCYXNlOjpkcmF3UmVnaW9uKGNvbnN0IFJlZ2lvbiYgcmVnKSBjb25zdAotewotICAgIFJlZ2lvbjo6aXRlcmF0b3IgaXRlcmF0b3IocmVnKTsKLSAgICBpZiAoaXRlcmF0b3IpIHsKLSAgICAgICAgUmVjdCByOwotICAgICAgICBjb25zdCBEaXNwbGF5SGFyZHdhcmUmIGh3KGdyYXBoaWNQbGFuZSgwKS5kaXNwbGF5SGFyZHdhcmUoKSk7Ci0gICAgICAgIGNvbnN0IGludDMyX3QgZmJXaWR0aCAgPSBody5nZXRXaWR0aCgpOwotICAgICAgICBjb25zdCBpbnQzMl90IGZiSGVpZ2h0ID0gaHcuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgIGNvbnN0IEdMc2hvcnQgdmVydGljZXNbXVsyXSA9IHsgeyAwLCAwIH0sIHsgZmJXaWR0aCwgMCB9LCAKLSAgICAgICAgICAgICAgICB7IGZiV2lkdGgsIGZiSGVpZ2h0IH0sIHsgMCwgZmJIZWlnaHQgfSAgfTsKLSAgICAgICAgZ2xWZXJ0ZXhQb2ludGVyKDIsIEdMX1NIT1JULCAwLCB2ZXJ0aWNlcyk7Ci0gICAgICAgIHdoaWxlIChpdGVyYXRvci5pdGVyYXRlKCZyKSkgewotICAgICAgICAgICAgY29uc3QgR0xpbnQgc3kgPSBmYkhlaWdodCAtIChyLnRvcCArIHIuaGVpZ2h0KCkpOwotICAgICAgICAgICAgZ2xTY2lzc29yKHIubGVmdCwgc3ksIHIud2lkdGgoKSwgci5oZWlnaHQoKSk7Ci0gICAgICAgICAgICBnbERyYXdBcnJheXMoR0xfVFJJQU5HTEVfRkFOLCAwLCA0KTsgCi0gICAgICAgIH0KLSAgICB9Ci19Ci0KLXZvaWQgTGF5ZXJCYXNlOjpkcmF3KGNvbnN0IFJlZ2lvbiYgaW5DbGlwKSBjb25zdAotewotICAgIC8vIGludmFsaWRhdGUgdGhlIHJlZ2lvbiB3ZSdsbCB1cGRhdGUKLSAgICBSZWdpb24gY2xpcChpbkNsaXApOyAgLy8gY29weS1vbi13cml0ZSwgc28gbm8tb3AgbW9zdCBvZiB0aGUgdGltZQotCi0gICAgLy8gUmVtb3ZlIHRoZSB0cmFuc3BhcmVudCBhcmVhIGZyb20gdGhlIGNsaXBwaW5nIHJlZ2lvbgotICAgIGNvbnN0IFN0YXRlJiBzID0gZHJhd2luZ1N0YXRlKCk7Ci0gICAgaWYgKExJS0VMWSghcy50cmFuc3BhcmVudFJlZ2lvbi5pc0VtcHR5KCkpKSB7Ci0gICAgICAgIGNsaXAuc3VidHJhY3QodHJhbnNwYXJlbnRSZWdpb25TY3JlZW4pOwotICAgICAgICBpZiAoY2xpcC5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgIC8vIHVzdWFsbHkgdGhpcyB3b24ndCBoYXBwZW4gYmVjYXVzZSB0aGlzIHNob3VsZCBiZSB0YWtlbiBjYXJlIG9mCi0gICAgICAgICAgICAvLyBieSBTdXJmYWNlRmxpbmdlcjo6Y29tcHV0ZVZpc2libGVSZWdpb25zKCkKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfSAgICAgICAgCi0gICAgfQotCi0gICAgLy8gcmVzZXQgR0wgc3RhdGUKLSAgICBnbEVuYWJsZShHTF9TQ0lTU09SX1RFU1QpOwotCi0gICAgb25EcmF3KGNsaXApOwotCi0gICAgLyoKLSAgICBnbERpc2FibGUoR0xfVEVYVFVSRV8yRCk7Ci0gICAgZ2xEaXNhYmxlKEdMX0RJVEhFUik7Ci0gICAgZ2xFbmFibGUoR0xfQkxFTkQpOwotICAgIGdsQmxlbmRGdW5jKEdMX09ORSwgR0xfT05FX01JTlVTX1NSQ19BTFBIQSk7Ci0gICAgZ2xDb2xvcjR4KDAsIDB4ODAwMCwgMCwgMHgxMDAwMCk7Ci0gICAgZHJhd1JlZ2lvbih0cmFuc3BhcmVudFJlZ2lvblNjcmVlbik7Ci0gICAgZ2xEaXNhYmxlKEdMX0JMRU5EKTsKLSAgICAqLwotfQotCi1HTHVpbnQgTGF5ZXJCYXNlOjpjcmVhdGVUZXh0dXJlKCkgY29uc3QKLXsKLSAgICBHTHVpbnQgdGV4dHVyZU5hbWUgPSAtMTsKLSAgICBnbEdlblRleHR1cmVzKDEsICZ0ZXh0dXJlTmFtZSk7Ci0gICAgZ2xCaW5kVGV4dHVyZShHTF9URVhUVVJFXzJELCB0ZXh0dXJlTmFtZSk7Ci0gICAgZ2xUZXhQYXJhbWV0ZXJ4KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfV1JBUF9TLCBHTF9DTEFNUF9UT19FREdFKTsKLSAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9XUkFQX1QsIEdMX0NMQU1QX1RPX0VER0UpOwotICAgIGlmIChtRmxhZ3MgJiBEaXNwbGF5SGFyZHdhcmU6OlNMT1dfQ09ORklHKSB7Ci0gICAgICAgIGdsVGV4UGFyYW1ldGVyeChHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX01BR19GSUxURVIsIEdMX05FQVJFU1QpOwotICAgICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NSU5fRklMVEVSLCBHTF9ORUFSRVNUKTsKLSAgICB9IGVsc2UgewotICAgICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NQUdfRklMVEVSLCBHTF9MSU5FQVIpOwotICAgICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NSU5fRklMVEVSLCBHTF9MSU5FQVIpOwotICAgIH0KLSAgICByZXR1cm4gdGV4dHVyZU5hbWU7Ci19Ci0KLXZvaWQgTGF5ZXJCYXNlOjpjbGVhcldpdGhPcGVuR0woY29uc3QgUmVnaW9uJiBjbGlwKSBjb25zdAotewotICAgIGNvbnN0IERpc3BsYXlIYXJkd2FyZSYgaHcoZ3JhcGhpY1BsYW5lKDApLmRpc3BsYXlIYXJkd2FyZSgpKTsKLSAgICBjb25zdCB1aW50MzJfdCBmYkhlaWdodCA9IGh3LmdldEhlaWdodCgpOwotICAgIGdsQ29sb3I0eCgwLDAsMCwwKTsKLSAgICBnbERpc2FibGUoR0xfVEVYVFVSRV8yRCk7Ci0gICAgZ2xEaXNhYmxlKEdMX0JMRU5EKTsKLSAgICBnbERpc2FibGUoR0xfRElUSEVSKTsKLSAgICBSZWN0IHI7Ci0gICAgUmVnaW9uOjppdGVyYXRvciBpdGVyYXRvcihjbGlwKTsKLSAgICBpZiAoaXRlcmF0b3IpIHsKLSAgICAgICAgZ2xFbmFibGUoR0xfU0NJU1NPUl9URVNUKTsKLSAgICAgICAgZ2xWZXJ0ZXhQb2ludGVyKDIsIEdMX0ZJWEVELCAwLCBtVmVydGljZXMpOwotICAgICAgICB3aGlsZSAoaXRlcmF0b3IuaXRlcmF0ZSgmcikpIHsKLSAgICAgICAgICAgIGNvbnN0IEdMaW50IHN5ID0gZmJIZWlnaHQgLSAoci50b3AgKyByLmhlaWdodCgpKTsKLSAgICAgICAgICAgIGdsU2Npc3NvcihyLmxlZnQsIHN5LCByLndpZHRoKCksIHIuaGVpZ2h0KCkpOwotICAgICAgICAgICAgZ2xEcmF3QXJyYXlzKEdMX1RSSUFOR0xFX0ZBTiwgMCwgNCk7IAotICAgICAgICB9Ci0gICAgfQotfQotCi12b2lkIExheWVyQmFzZTo6ZHJhd1dpdGhPcGVuR0woY29uc3QgUmVnaW9uJiBjbGlwLAotICAgICAgICBHTGludCB0ZXh0dXJlTmFtZSwgY29uc3QgR0dMU3VyZmFjZSYgdCwgaW50IHRyYW5zZm9ybSkgY29uc3QKLXsKLSAgICBjb25zdCBEaXNwbGF5SGFyZHdhcmUmIGh3KGdyYXBoaWNQbGFuZSgwKS5kaXNwbGF5SGFyZHdhcmUoKSk7Ci0gICAgY29uc3QgdWludDMyX3QgZmJIZWlnaHQgPSBody5nZXRIZWlnaHQoKTsKLSAgICBjb25zdCBTdGF0ZSYgcyhkcmF3aW5nU3RhdGUoKSk7Ci0KLSAgICAvLyBiaW5kIG91ciB0ZXh0dXJlCi0gICAgdmFsaWRhdGVUZXh0dXJlKHRleHR1cmVOYW1lKTsKLSAgICBnbEVuYWJsZShHTF9URVhUVVJFXzJEKTsKLQotICAgIC8vIERpdGhlcmluZy4uLgotICAgIGlmIChzLmZsYWdzICYgSVN1cmZhY2VDb21wb3Nlcjo6ZUxheWVyRGl0aGVyKSB7Ci0gICAgICAgIGdsRW5hYmxlKEdMX0RJVEhFUik7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgZ2xEaXNhYmxlKEdMX0RJVEhFUik7Ci0gICAgfQotCi0gICAgaWYgKFVOTElLRUxZKHMuYWxwaGEgPCAweEZGKSkgewotICAgICAgICAvLyBXZSBoYXZlIGFuIGFscGhhLW1vZHVsYXRpb24uIFdlIG5lZWQgdG8gbW9kdWxhdGUgYWxsCi0gICAgICAgIC8vIHRleHR1cmUgY29tcG9uZW50cyBieSBhbHBoYSBiZWNhdXNlIHdlJ3JlIGFsd2F5cyB1c2luZyAKLSAgICAgICAgLy8gcHJlbXVsdGlwbGllZCBhbHBoYS4KLSAgICAgICAgCi0gICAgICAgIC8vIElmIHRoZSB0ZXh0dXJlIGRvZXNuJ3QgaGF2ZSBhbiBhbHBoYSBjaGFubmVsIHdlIGNhbgotICAgICAgICAvLyB1c2UgUkVQTEFDRSBhbmQgc3dpdGNoIHRvIG5vbiBwcmVtdWx0aXBsaWVkIGFscGhhCi0gICAgICAgIC8vIGJsZW5kaW5nIChTUkNBL09ORV9NSU5VU19TUkNBKS4KLSAgICAgICAgCi0gICAgICAgIEdMZW51bSBlbnYsIHNyYzsKLSAgICAgICAgaWYgKG5lZWRzQmxlbmRpbmcoKSkgewotICAgICAgICAgICAgZW52ID0gR0xfTU9EVUxBVEU7Ci0gICAgICAgICAgICBzcmMgPSBtUHJlbXVsdGlwbGllZEFscGhhID8gR0xfT05FIDogR0xfU1JDX0FMUEhBOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZW52ID0gR0xfUkVQTEFDRTsKLSAgICAgICAgICAgIHNyYyA9IEdMX1NSQ19BTFBIQTsKLSAgICAgICAgfQotICAgICAgICBjb25zdCBHR0xmaXhlZCBhbHBoYSA9IChzLmFscGhhIDw8IDE2KS8yNTU7Ci0gICAgICAgIGdsQ29sb3I0eChhbHBoYSwgYWxwaGEsIGFscGhhLCBhbHBoYSk7Ci0gICAgICAgIGdsRW5hYmxlKEdMX0JMRU5EKTsKLSAgICAgICAgZ2xCbGVuZEZ1bmMoc3JjLCBHTF9PTkVfTUlOVVNfU1JDX0FMUEhBKTsKLSAgICAgICAgZ2xUZXhFbnZ4KEdMX1RFWFRVUkVfRU5WLCBHTF9URVhUVVJFX0VOVl9NT0RFLCBlbnYpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfUkVQTEFDRSk7Ci0gICAgICAgIGdsQ29sb3I0eCgweDEwMDAwLCAweDEwMDAwLCAweDEwMDAwLCAweDEwMDAwKTsKLSAgICAgICAgaWYgKG5lZWRzQmxlbmRpbmcoKSkgewotICAgICAgICAgICAgR0xlbnVtIHNyYyA9IG1QcmVtdWx0aXBsaWVkQWxwaGEgPyBHTF9PTkUgOiBHTF9TUkNfQUxQSEE7Ci0gICAgICAgICAgICBnbEVuYWJsZShHTF9CTEVORCk7Ci0gICAgICAgICAgICBnbEJsZW5kRnVuYyhzcmMsIEdMX09ORV9NSU5VU19TUkNfQUxQSEEpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZ2xEaXNhYmxlKEdMX0JMRU5EKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIGlmIChVTkxJS0VMWSh0cmFuc2Zvcm1lZCgpCi0gICAgICAgICAgICB8fCAhKG1GbGFncyAmIERpc3BsYXlIYXJkd2FyZTo6RFJBV19URVhUVVJFX0VYVEVOU0lPTikgKSkgCi0gICAgewotICAgICAgICAvL1N0b3BXYXRjaCB3YXRjaCgiR0wgdHJhbnNmb3JtZWQiKTsKLSAgICAgICAgUmVnaW9uOjppdGVyYXRvciBpdGVyYXRvcihjbGlwKTsKLSAgICAgICAgaWYgKGl0ZXJhdG9yKSB7Ci0gICAgICAgICAgICAvLyBhbHdheXMgdXNlIGhpZ2gtcXVhbGl0eSBmaWx0ZXJpbmcgd2l0aCBmYXN0IGNvbmZpZ3VyYXRpb25zCi0gICAgICAgICAgICBib29sIGZhc3QgPSAhKG1GbGFncyAmIERpc3BsYXlIYXJkd2FyZTo6U0xPV19DT05GSUcpOwotICAgICAgICAgICAgaWYgKCFmYXN0ICYmIHMuZmxhZ3MgJiBJU3VyZmFjZUNvbXBvc2VyOjplTGF5ZXJGaWx0ZXIpIHsKLSAgICAgICAgICAgICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NQUdfRklMVEVSLCBHTF9MSU5FQVIpOwotICAgICAgICAgICAgICAgIGdsVGV4UGFyYW1ldGVyeChHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX01JTl9GSUxURVIsIEdMX0xJTkVBUik7Ci0gICAgICAgICAgICB9ICAgICAgICAgICAgCi0gICAgICAgICAgICBjb25zdCBHTGZpeGVkIHRleENvb3Jkc1s0XVsyXSA9IHsKLSAgICAgICAgICAgICAgICAgICAgeyAwLCAgICAgICAgMCB9LAotICAgICAgICAgICAgICAgICAgICB7IDAsICAgICAgICAweDEwMDAwIH0sCi0gICAgICAgICAgICAgICAgICAgIHsgMHgxMDAwMCwgIDB4MTAwMDAgfSwKLSAgICAgICAgICAgICAgICAgICAgeyAweDEwMDAwLCAgMCB9Ci0gICAgICAgICAgICB9OwotCi0gICAgICAgICAgICBnbE1hdHJpeE1vZGUoR0xfVEVYVFVSRSk7Ci0gICAgICAgICAgICBnbExvYWRJZGVudGl0eSgpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBpZiAodHJhbnNmb3JtID09IEhBTF9UUkFOU0ZPUk1fUk9UXzkwKSB7Ci0gICAgICAgICAgICAgICAgZ2xUcmFuc2xhdGVmKDAsIDEsIDApOwotICAgICAgICAgICAgICAgIGdsUm90YXRlZigtOTAsIDAsIDAsIDEpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoIShtRmxhZ3MgJiBEaXNwbGF5SGFyZHdhcmU6Ok5QT1RfRVhURU5TSU9OKSkgewotICAgICAgICAgICAgICAgIC8vIGZpbmQgdGhlIHNtYWxsZXN0IHBvd2VyLW9mLXR3byB0aGF0IHdpbGwgYWNjb21tb2RhdGUgb3VyIHN1cmZhY2UKLSAgICAgICAgICAgICAgICBHTHVpbnQgdHcgPSAxIDw8ICgzMSAtIGNseih0LndpZHRoKSk7Ci0gICAgICAgICAgICAgICAgR0x1aW50IHRoID0gMSA8PCAoMzEgLSBjbHoodC5oZWlnaHQpKTsKLSAgICAgICAgICAgICAgICBpZiAodHcgPCB0LndpZHRoKSAgdHcgPDw9IDE7Ci0gICAgICAgICAgICAgICAgaWYgKHRoIDwgdC5oZWlnaHQpIHRoIDw8PSAxOwotICAgICAgICAgICAgICAgIC8vIHRoaXMgZGl2aWRlIHNob3VsZCBiZSByZWxhdGl2ZWx5IGZhc3QgYmVjYXVzZSBpdCdzCi0gICAgICAgICAgICAgICAgLy8gYSBwb3dlci1vZi10d28gKG9wdGltaXplZCBwYXRoIGluIGxpYmdjYykKLSAgICAgICAgICAgICAgICBHTGZsb2F0IHdzID0gR0xmbG9hdCh0LndpZHRoKSAvdHc7Ci0gICAgICAgICAgICAgICAgR0xmbG9hdCBocyA9IEdMZmxvYXQodC5oZWlnaHQpL3RoOwotICAgICAgICAgICAgICAgIGdsU2NhbGVmKHdzLCBocywgMS4wZik7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGdsRW5hYmxlQ2xpZW50U3RhdGUoR0xfVEVYVFVSRV9DT09SRF9BUlJBWSk7Ci0gICAgICAgICAgICBnbFZlcnRleFBvaW50ZXIoMiwgR0xfRklYRUQsIDAsIG1WZXJ0aWNlcyk7Ci0gICAgICAgICAgICBnbFRleENvb3JkUG9pbnRlcigyLCBHTF9GSVhFRCwgMCwgdGV4Q29vcmRzKTsKLQotICAgICAgICAgICAgUmVjdCByOwotICAgICAgICAgICAgd2hpbGUgKGl0ZXJhdG9yLml0ZXJhdGUoJnIpKSB7Ci0gICAgICAgICAgICAgICAgY29uc3QgR0xpbnQgc3kgPSBmYkhlaWdodCAtIChyLnRvcCArIHIuaGVpZ2h0KCkpOwotICAgICAgICAgICAgICAgIGdsU2Npc3NvcihyLmxlZnQsIHN5LCByLndpZHRoKCksIHIuaGVpZ2h0KCkpOwotICAgICAgICAgICAgICAgIGdsRHJhd0FycmF5cyhHTF9UUklBTkdMRV9GQU4sIDAsIDQpOyAKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKCFmYXN0ICYmIHMuZmxhZ3MgJiBJU3VyZmFjZUNvbXBvc2VyOjplTGF5ZXJGaWx0ZXIpIHsKLSAgICAgICAgICAgICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NQUdfRklMVEVSLCBHTF9ORUFSRVNUKTsKLSAgICAgICAgICAgICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NSU5fRklMVEVSLCBHTF9ORUFSRVNUKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGdsRGlzYWJsZUNsaWVudFN0YXRlKEdMX1RFWFRVUkVfQ09PUkRfQVJSQVkpOwotICAgICAgICB9Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgUmVnaW9uOjppdGVyYXRvciBpdGVyYXRvcihjbGlwKTsKLSAgICAgICAgaWYgKGl0ZXJhdG9yKSB7Ci0gICAgICAgICAgICBSZWN0IHI7Ci0gICAgICAgICAgICBHTGludCBjcm9wWzRdID0geyAwLCB0LmhlaWdodCwgdC53aWR0aCwgLXQuaGVpZ2h0IH07Ci0gICAgICAgICAgICBnbFRleFBhcmFtZXRlcml2KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfQ1JPUF9SRUNUX09FUywgY3JvcCk7Ci0gICAgICAgICAgICBpbnQgeCA9IHR4KCk7Ci0gICAgICAgICAgICBpbnQgeSA9IHR5KCk7Ci0gICAgICAgICAgICB5ID0gZmJIZWlnaHQgLSAoeSArIHQuaGVpZ2h0KTsKLSAgICAgICAgICAgIHdoaWxlIChpdGVyYXRvci5pdGVyYXRlKCZyKSkgewotICAgICAgICAgICAgICAgIGNvbnN0IEdMaW50IHN5ID0gZmJIZWlnaHQgLSAoci50b3AgKyByLmhlaWdodCgpKTsKLSAgICAgICAgICAgICAgICBnbFNjaXNzb3Ioci5sZWZ0LCBzeSwgci53aWR0aCgpLCByLmhlaWdodCgpKTsKLSAgICAgICAgICAgICAgICBnbERyYXdUZXhpT0VTKHgsIHksIDAsIHQud2lkdGgsIHQuaGVpZ2h0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLX0KLQotdm9pZCBMYXllckJhc2U6OnZhbGlkYXRlVGV4dHVyZShHTGludCB0ZXh0dXJlTmFtZSkgY29uc3QKLXsKLSAgICBnbEJpbmRUZXh0dXJlKEdMX1RFWFRVUkVfMkQsIHRleHR1cmVOYW1lKTsKLSAgICAvLyBUT0RPOiByZWxvYWQgdGhlIHRleHR1cmUgaWYgbmVlZGVkCi0gICAgLy8gdGhpcyBpcyBjdXJyZW50bHkgZG9uZSBpbiBsb2FkVGV4dHVyZSgpIGJlbG93Ci19Ci0KLXZvaWQgTGF5ZXJCYXNlOjpsb2FkVGV4dHVyZShjb25zdCBSZWdpb24mIGRpcnR5LAotICAgICAgICBHTGludCB0ZXh0dXJlTmFtZSwgY29uc3QgR0dMU3VyZmFjZSYgdCwKLSAgICAgICAgR0x1aW50JiB0ZXh0dXJlV2lkdGgsIEdMdWludCYgdGV4dHVyZUhlaWdodCkgY29uc3QKLXsKLSAgICAvLyBUT0RPOiBkZWZlciB0aGUgYWN0dWFsIHRleHR1cmUgcmVsb2FkIHVudGlsIExheWVyQmFzZTo6dmFsaWRhdGVUZXh0dXJlCi0gICAgLy8gaXMgY2FsbGVkLgotCi0gICAgdWludDMyX3QgZmxhZ3MgPSBtRmxhZ3M7Ci0gICAgZ2xCaW5kVGV4dHVyZShHTF9URVhUVVJFXzJELCB0ZXh0dXJlTmFtZSk7Ci0KLSAgICBHTHVpbnQgdHcgPSB0LndpZHRoOwotICAgIEdMdWludCB0aCA9IHQuaGVpZ2h0OwotCi0gICAgLyoKLSAgICAgKiBJbiBPcGVuR0wgRVMgd2UgY2FuJ3Qgc3BlY2lmeSBhIHN0cmlkZSB3aXRoIGdsVGV4SW1hZ2UyRCAoaG93ZXZlciwKLSAgICAgKiBHTF9VTlBBQ0tfQUxJR05NRU5UIGlzIDQsIHdoaWNoIGluIGVzc2VuY2UgYWxsb3dzIGEgbGltaXRlZCBmb3JtIG9mCi0gICAgICogc3RyaWRlKS4KLSAgICAgKiBTbyBpZiB0aGUgc3RyaWRlIGhlcmUgaXNuJ3QgcmVwcmVzZW50YWJsZSB3aXRoIEdMX1VOUEFDS19BTElHTk1FTlQsIHdlCi0gICAgICogbmVlZCB0byBkbyBzb21ldGhpbmcgcmVhc29uYWJsZSAoaGVyZSBjcmVhdGluZyBhIGJpZ2dlciB0ZXh0dXJlKS4KLSAgICAgKiAKLSAgICAgKiBleHRyYSBwaXhlbHMgPSAoKChzdHJpZGUgLSB3aWR0aCkgKiBwaXhlbHNpemUpIC8gR0xfVU5QQUNLX0FMSUdOTUVOVCk7Ci0gICAgICogCi0gICAgICogVGhpcyBzaXR1YXRpb24gZG9lc24ndCBoYXBwZW4gb2Z0ZW4sIGJ1dCBzb21lIGgvdyBoYXZlIGEgbGltaXRhdGlvbgotICAgICAqIGZvciB0aGVpciBmcmFtZWJ1ZmZlciAoZWc6IG11c3QgYmUgbXVsdGlwbGUgb2YgOCBwaXhlbHMpLCBhbmQKLSAgICAgKiB3ZSBuZWVkIHRvIHRha2UgdGhhdCBpbnRvIGFjY291bnQgd2hlbiB1c2luZyB0aGVzZSBidWZmZXJzIGFzCi0gICAgICogdGV4dHVyZXMuCi0gICAgICoKLSAgICAgKiBUaGlzIHNob3VsZCBuZXZlciBiZSBhIHByb2JsZW0gd2l0aCBQT1QgdGV4dHVyZXMKLSAgICAgKi8KLQotICAgIHR3ICs9ICgoKHQuc3RyaWRlIC0gdHcpICogYnl0ZXNQZXJQaXhlbCh0LmZvcm1hdCkpIC8gNCk7Ci0KLSAgICAvKgotICAgICAqIHJvdW5kIHRvIFBPVCBpZiBuZWVkZWQgCi0gICAgICovCi0gICAgCi0gICAgR0x1aW50IHRleHR1cmVfdyA9IHR3OwotICAgIEdMdWludCB0ZXh0dXJlX2ggPSB0aDsKLSAgICBpZiAoIShmbGFncyAmIERpc3BsYXlIYXJkd2FyZTo6TlBPVF9FWFRFTlNJT04pKSB7Ci0gICAgICAgIC8vIGZpbmQgdGhlIHNtYWxsZXN0IHBvd2VyLW9mLXR3byB0aGF0IHdpbGwgYWNjb21tb2RhdGUgb3VyIHN1cmZhY2UKLSAgICAgICAgdGV4dHVyZV93ID0gMSA8PCAoMzEgLSBjbHoodC53aWR0aCkpOwotICAgICAgICB0ZXh0dXJlX2ggPSAxIDw8ICgzMSAtIGNseih0LmhlaWdodCkpOwotICAgICAgICBpZiAodGV4dHVyZV93IDwgdC53aWR0aCkgIHRleHR1cmVfdyA8PD0gMTsKLSAgICAgICAgaWYgKHRleHR1cmVfaCA8IHQuaGVpZ2h0KSB0ZXh0dXJlX2ggPDw9IDE7Ci0gICAgICAgIGlmICh0ZXh0dXJlX3cgIT0gdHcgfHwgdGV4dHVyZV9oICE9IHRoKSB7Ci0gICAgICAgICAgICAvLyB3ZSBjYW4ndCB1c2UgRElSRUNUX1RFWFRVUkUgc2luY2Ugd2UgY2hhbmdlZCB0aGUgc2l6ZQotICAgICAgICAgICAgLy8gb2YgdGhlIHRleHR1cmUKLSAgICAgICAgICAgIGZsYWdzICY9IH5EaXNwbGF5SGFyZHdhcmU6OkRJUkVDVF9URVhUVVJFOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgaWYgKGZsYWdzICYgRGlzcGxheUhhcmR3YXJlOjpESVJFQ1RfVEVYVFVSRSkgewotICAgICAgICAvLyBoZXJlIHdlJ3JlIGd1YXJhbnRlZWQgdGhhdCB0ZXh0dXJlX3t3fGh9ID09IHR7d3xofQotICAgICAgICBpZiAodC5mb3JtYXQgPT0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JfNTY1KSB7Ci0gICAgICAgICAgICBnbFRleEltYWdlMkQoR0xfRElSRUNUX1RFWFRVUkVfMkRfUVVBTENPTU0sIDAsCi0gICAgICAgICAgICAgICAgICAgIEdMX1JHQiwgdHcsIHRoLCAwLAotICAgICAgICAgICAgICAgICAgICBHTF9SR0IsIEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81LCB0LmRhdGEpOwotICAgICAgICB9IGVsc2UgaWYgKHQuZm9ybWF0ID09IEdHTF9QSVhFTF9GT1JNQVRfUkdCQV80NDQ0KSB7Ci0gICAgICAgICAgICBnbFRleEltYWdlMkQoR0xfRElSRUNUX1RFWFRVUkVfMkRfUVVBTENPTU0sIDAsCi0gICAgICAgICAgICAgICAgICAgIEdMX1JHQkEsIHR3LCB0aCwgMCwKLSAgICAgICAgICAgICAgICAgICAgR0xfUkdCQSwgR0xfVU5TSUdORURfU0hPUlRfNF80XzRfNCwgdC5kYXRhKTsKLSAgICAgICAgfSBlbHNlIGlmICh0LmZvcm1hdCA9PSBHR0xfUElYRUxfRk9STUFUX1JHQkFfODg4OCkgewotICAgICAgICAgICAgZ2xUZXhJbWFnZTJEKEdMX0RJUkVDVF9URVhUVVJFXzJEX1FVQUxDT01NLCAwLAotICAgICAgICAgICAgICAgICAgICBHTF9SR0JBLCB0dywgdGgsIDAsCi0gICAgICAgICAgICAgICAgICAgIEdMX1JHQkEsIEdMX1VOU0lHTkVEX0JZVEUsIHQuZGF0YSk7Ci0gICAgICAgIH0gZWxzZSBpZiAodC5mb3JtYXQgPT0gR0dMX1BJWEVMX0ZPUk1BVF9CR1JBXzg4ODgpIHsKLSAgICAgICAgICAgIC8vIFRPRE86IGFkZCBHTF9CR1JBIGV4dGVuc2lvbgotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLy8gb29wcywgd2UgZG9uJ3QgaGFuZGxlIHRoaXMgZm9ybWF0LCB0cnkgdGhlIHJlZ3VsYXIgcGF0aAotICAgICAgICAgICAgZ290byByZWd1bGFyOwotICAgICAgICB9Ci0gICAgICAgIHRleHR1cmVXaWR0aCA9IHR3OwotICAgICAgICB0ZXh0dXJlSGVpZ2h0ID0gdGg7Ci0gICAgfSBlbHNlIHsKLXJlZ3VsYXI6Ci0gICAgICAgIFJlY3QgYm91bmRzKGRpcnR5LmJvdW5kcygpKTsKLSAgICAgICAgR0x2b2lkKiBkYXRhID0gMDsKLSAgICAgICAgaWYgKHRleHR1cmVfdyE9dGV4dHVyZVdpZHRoIHx8IHRleHR1cmVfaCE9dGV4dHVyZUhlaWdodCkgewotICAgICAgICAgICAgLy8gdGV4dHVyZSBzaXplIGNoYW5nZWQsIHdlIG5lZWQgdG8gY3JlYXRlIGEgbmV3IG9uZQotCi0gICAgICAgICAgICBpZiAoIXRleHR1cmVXaWR0aCB8fCAhdGV4dHVyZUhlaWdodCkgewotICAgICAgICAgICAgICAgIC8vIHRoaXMgaXMgdGhlIGZpcnN0IHRpbWUsIGxvYWQgdGhlIHdob2xlIHRleHR1cmUKLSAgICAgICAgICAgICAgICBpZiAodGV4dHVyZV93PT10dyAmJiB0ZXh0dXJlX2g9PXRoKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIHdlIGNhbiBkbyBpdCBvbmUgcGFzcwotICAgICAgICAgICAgICAgICAgICBkYXRhID0gdC5kYXRhOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIHdlIGhhdmUgdG8gY3JlYXRlIHRoZSB0ZXh0dXJlIGZpcnN0IGJlY2F1c2UgaXQKLSAgICAgICAgICAgICAgICAgICAgLy8gZG9lc24ndCBtYXRjaCB0aGUgc2l6ZSBvZiB0aGUgYnVmZmVyCi0gICAgICAgICAgICAgICAgICAgIGJvdW5kcy5zZXQoUmVjdCh0dywgdGgpKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIGlmICh0LmZvcm1hdCA9PSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjUpIHsKLSAgICAgICAgICAgICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMX1JHQiwgdGV4dHVyZV93LCB0ZXh0dXJlX2gsIDAsCi0gICAgICAgICAgICAgICAgICAgICAgICBHTF9SR0IsIEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81LCBkYXRhKTsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAodC5mb3JtYXQgPT0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzQ0NDQpIHsKLSAgICAgICAgICAgICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMX1JHQkEsIHRleHR1cmVfdywgdGV4dHVyZV9oLCAwLAotICAgICAgICAgICAgICAgICAgICAgICAgR0xfUkdCQSwgR0xfVU5TSUdORURfU0hPUlRfNF80XzRfNCwgZGF0YSk7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKHQuZm9ybWF0ID09IEdHTF9QSVhFTF9GT1JNQVRfUkdCQV84ODg4KSB7Ci0gICAgICAgICAgICAgICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsCi0gICAgICAgICAgICAgICAgICAgICAgICBHTF9SR0JBLCB0ZXh0dXJlX3csIHRleHR1cmVfaCwgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMX1JHQkEsIEdMX1VOU0lHTkVEX0JZVEUsIGRhdGEpOwotICAgICAgICAgICAgfSBlbHNlIGlmICggdC5mb3JtYXQgPT0gR0dMX1BJWEVMX0ZPUk1BVF9ZQ2JDcl80MjJfU1AgfHwKLSAgICAgICAgICAgICAgICAgICAgICAgIHQuZm9ybWF0ID09IEdHTF9QSVhFTF9GT1JNQVRfWUNiQ3JfNDIwX1NQKSB7Ci0gICAgICAgICAgICAgICAgLy8ganVzdCBzaG93IHRoZSBZIHBsYW5lIG9mIFlVViBidWZmZXJzCi0gICAgICAgICAgICAgICAgZGF0YSA9IHQuZGF0YTsKLSAgICAgICAgICAgICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMX0xVTUlOQU5DRSwgdGV4dHVyZV93LCB0ZXh0dXJlX2gsIDAsCi0gICAgICAgICAgICAgICAgICAgICAgICBHTF9MVU1JTkFOQ0UsIEdMX1VOU0lHTkVEX0JZVEUsIGRhdGEpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAvLyBvb3BzLCB3ZSBkb24ndCBoYW5kbGUgdGhpcyBmb3JtYXQhCi0gICAgICAgICAgICAgICAgTE9HRSgibGF5ZXIgJXAsIHRleHR1cmU9JWQsIHVzaW5nIGZvcm1hdCAlZCwgd2hpY2ggaXMgbm90ICIKLSAgICAgICAgICAgICAgICAgICAgICJzdXBwb3J0ZWQgYnkgdGhlIEdMIiwgdGhpcywgdGV4dHVyZU5hbWUsIHQuZm9ybWF0KTsKLSAgICAgICAgICAgICAgICB0ZXh0dXJlTmFtZSA9IC0xOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgdGV4dHVyZVdpZHRoID0gdGV4dHVyZV93OwotICAgICAgICAgICAgdGV4dHVyZUhlaWdodCA9IHRleHR1cmVfaDsKLSAgICAgICAgfQotICAgICAgICBpZiAoIWRhdGEgJiYgdGV4dHVyZU5hbWU+PTApIHsKLSAgICAgICAgICAgIGlmICh0LmZvcm1hdCA9PSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjUpIHsKLSAgICAgICAgICAgICAgICBnbFRleFN1YkltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwKLSAgICAgICAgICAgICAgICAgICAgICAgIDAsIGJvdW5kcy50b3AsIHQud2lkdGgsIGJvdW5kcy5oZWlnaHQoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMX1JHQiwgR0xfVU5TSUdORURfU0hPUlRfNV82XzUsCi0gICAgICAgICAgICAgICAgICAgICAgICB0LmRhdGEgKyBib3VuZHMudG9wKnQud2lkdGgqMik7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKHQuZm9ybWF0ID09IEdHTF9QSVhFTF9GT1JNQVRfUkdCQV80NDQ0KSB7Ci0gICAgICAgICAgICAgICAgZ2xUZXhTdWJJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAwLCBib3VuZHMudG9wLCB0LndpZHRoLCBib3VuZHMuaGVpZ2h0KCksCi0gICAgICAgICAgICAgICAgICAgICAgICBHTF9SR0JBLCBHTF9VTlNJR05FRF9TSE9SVF80XzRfNF80LAotICAgICAgICAgICAgICAgICAgICAgICAgdC5kYXRhICsgYm91bmRzLnRvcCp0LndpZHRoKjIpOwotICAgICAgICAgICAgfSBlbHNlIGlmICh0LmZvcm1hdCA9PSBHR0xfUElYRUxfRk9STUFUX1JHQkFfODg4OCkgewotICAgICAgICAgICAgICAgIGdsVGV4U3ViSW1hZ2UyRChHTF9URVhUVVJFXzJELCAwLAotICAgICAgICAgICAgICAgICAgICAgICAgMCwgYm91bmRzLnRvcCwgdC53aWR0aCwgYm91bmRzLmhlaWdodCgpLAotICAgICAgICAgICAgICAgICAgICAgICAgR0xfUkdCQSwgR0xfVU5TSUdORURfQllURSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHQuZGF0YSArIGJvdW5kcy50b3AqdC53aWR0aCo0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLX0KLQotYm9vbCBMYXllckJhc2U6OmNhblVzZUNvcHliaXQoKSBjb25zdAotewotICAgIHJldHVybiBtQ2FuVXNlQ29weUJpdDsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUxheWVyQmFzZUNsaWVudDo6TGF5ZXJCYXNlQ2xpZW50KFN1cmZhY2VGbGluZ2VyKiBmbGluZ2VyLCBEaXNwbGF5SUQgZGlzcGxheSwKLSAgICAgICAgQ2xpZW50KiBjLCBpbnQzMl90IGkpCi0gICAgOiBMYXllckJhc2UoZmxpbmdlciwgZGlzcGxheSksIGNsaWVudChjKSwKLSAgICAgIGxjYmxrKCBjID8gJihjLT5jdHJsYmxrLT5sYXllcnNbaV0pIDogMCApLAotICAgICAgbUluZGV4KGkpCi17Ci0gICAgaWYgKGNsaWVudCkgewotICAgICAgICBjbGllbnQtPmJpbmRMYXllcih0aGlzLCBpKTsKLQotICAgICAgICAvLyBJbml0aWFsaXplIHRoaXMgbGF5ZXIncyBjb250cm9sIGJsb2NrCi0gICAgICAgIG1lbXNldCh0aGlzLT5sY2JsaywgMCwgc2l6ZW9mKGxheWVyX2NibGtfdCkpOwotICAgICAgICB0aGlzLT5sY2Jsay0+aWRlbnRpdHkgPSBtSWRlbnRpdHk7Ci0gICAgICAgIFJlZ2lvbjo6d3JpdGVFbXB0eSgmKHRoaXMtPmxjYmxrLT5yZWdpb25bMF0pLCBzaXplb2YoZmxhdF9yZWdpb25fdCkpOwotICAgICAgICBSZWdpb246OndyaXRlRW1wdHkoJih0aGlzLT5sY2Jsay0+cmVnaW9uWzFdKSwgc2l6ZW9mKGZsYXRfcmVnaW9uX3QpKTsKLSAgICB9Ci19Ci0KLUxheWVyQmFzZUNsaWVudDo6fkxheWVyQmFzZUNsaWVudCgpCi17Ci0gICAgaWYgKGNsaWVudCkgewotICAgICAgICBjbGllbnQtPmZyZWUobUluZGV4KTsKLSAgICB9Ci19Ci0KLWludDMyX3QgTGF5ZXJCYXNlQ2xpZW50OjpzZXJ2ZXJJbmRleCgpIGNvbnN0IHsKLSAgICBpZiAoY2xpZW50KSB7Ci0gICAgICAgIHJldHVybiAoY2xpZW50LT5jaWQ8PDE2KXxtSW5kZXg7Ci0gICAgfQotICAgIHJldHVybiAweEZGRkYwMDAwIHwgbUluZGV4OwotfQotCi1zcDxMYXllckJhc2VDbGllbnQ6OlN1cmZhY2U+IExheWVyQmFzZUNsaWVudDo6Z2V0U3VyZmFjZSgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG5ldyBTdXJmYWNlKGNsaWVudEluZGV4KCksIG1JZGVudGl0eSk7Ci19Ci0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyQmFzZS5oIGIvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllckJhc2UuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTAyMGY0NC4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyQmFzZS5oCisrKyAvZGV2L251bGwKQEAgLTEsMzU2ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfTEFZRVJfQkFTRV9ICi0jZGVmaW5lIEFORFJPSURfTEFZRVJfQkFTRV9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHByaXZhdGUvdWkvTGF5ZXJTdGF0ZS5oPgotCi0jaW5jbHVkZSA8dWkvUmVnaW9uLmg+Ci0jaW5jbHVkZSA8dWkvT3ZlcmxheS5oPgotCi0jaW5jbHVkZSA8cGl4ZWxmbGluZ2VyL3BpeGVsZmxpbmdlci5oPgotCi0jaW5jbHVkZSAiVHJhbnNmb3JtLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIFN1cmZhY2VGbGluZ2VyOwotY2xhc3MgRGlzcGxheUhhcmR3YXJlOwotY2xhc3MgR3JhcGhpY1BsYW5lOwotY2xhc3MgQ2xpZW50OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgTGF5ZXJCYXNlCi17Ci0gICAgLy8gcG9vciBtYW4ncyBkeW5hbWljX2Nhc3QgYmVsb3cKLSAgICB0ZW1wbGF0ZTx0eXBlbmFtZSBUPgotICAgIHN0cnVjdCBnZXRUeXBlSW5mb09mQW55VHlwZSB7Ci0gICAgICAgIHN0YXRpYyB1aW50MzJfdCBnZXQoKSB7IHJldHVybiBUOjp0eXBlSW5mbzsgfQotICAgIH07Ci0KLSAgICB0ZW1wbGF0ZTx0eXBlbmFtZSBUPgotICAgIHN0cnVjdCBnZXRUeXBlSW5mb09mQW55VHlwZTxUKj4gewotICAgICAgICBzdGF0aWMgdWludDMyX3QgZ2V0KCkgeyByZXR1cm4gZ2V0VHlwZUluZm9PZkFueVR5cGU8VD46OmdldCgpOyB9Ci0gICAgfTsKLQotcHVibGljOgotICAgIHN0YXRpYyBjb25zdCB1aW50MzJfdCB0eXBlSW5mbzsKLSAgICBzdGF0aWMgY29uc3QgY2hhciogY29uc3QgdHlwZUlEOwotICAgIHZpcnR1YWwgY2hhciBjb25zdCogZ2V0VHlwZUlEKCkgY29uc3QgeyByZXR1cm4gdHlwZUlEOyB9Ci0gICAgdmlydHVhbCB1aW50MzJfdCBnZXRUeXBlSW5mbygpIGNvbnN0IHsgcmV0dXJuIHR5cGVJbmZvOyB9Ci0gICAgCi0gICAgdGVtcGxhdGU8dHlwZW5hbWUgVD4KLSAgICBzdGF0aWMgVCBkeW5hbWljQ2FzdChMYXllckJhc2UqIGJhc2UpIHsKLSAgICAgICAgdWludDMyX3QgbW9zdERlcml2ZWRJbmZvID0gYmFzZS0+Z2V0VHlwZUluZm8oKTsKLSAgICAgICAgdWludDMyX3QgY2FzdFRvSW5mbyA9IGdldFR5cGVJbmZvT2ZBbnlUeXBlPFQ+OjpnZXQoKTsKLSAgICAgICAgaWYgKChtb3N0RGVyaXZlZEluZm8gJiBjYXN0VG9JbmZvKSA9PSBjYXN0VG9JbmZvKQotICAgICAgICAgICAgcmV0dXJuIHN0YXRpY19jYXN0PFQ+KGJhc2UpOwotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAKLSAgICBzdGF0aWMgVmVjdG9yPEdMdWludD4gZGVsZXRlZFRleHR1cmVzOyAKLQotICAgIExheWVyQmFzZShTdXJmYWNlRmxpbmdlciogZmxpbmdlciwgRGlzcGxheUlEIGRpc3BsYXkpOwotICAgIHZpcnR1YWwgfkxheWVyQmFzZSgpOwotICAgIAotICAgIERpc3BsYXlJRCAgICAgICAgICAgZHB5OwotICAgIG11dGFibGUgYm9vbCAgICAgICAgY29udGVudERpcnR5OwotICAgICAgICAgICAgUmVnaW9uICAgICAgdmlzaWJsZVJlZ2lvblNjcmVlbjsKLSAgICAgICAgICAgIFJlZ2lvbiAgICAgIHRyYW5zcGFyZW50UmVnaW9uU2NyZWVuOwotICAgICAgICAgICAgUmVnaW9uICAgICAgY292ZXJlZFJlZ2lvblNjcmVlbjsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgc3RydWN0IFN0YXRlIHsKLSAgICAgICAgICAgICAgICB1aW50MzJfdCAgICAgICAgdzsKLSAgICAgICAgICAgICAgICB1aW50MzJfdCAgICAgICAgaDsKLSAgICAgICAgICAgICAgICB1aW50MzJfdCAgICAgICAgejsKLSAgICAgICAgICAgICAgICB1aW50OF90ICAgICAgICAgYWxwaGE7Ci0gICAgICAgICAgICAgICAgdWludDhfdCAgICAgICAgIGZsYWdzOwotICAgICAgICAgICAgICAgIHVpbnQ4X3QgICAgICAgICByZXNlcnZlZFsyXTsKLSAgICAgICAgICAgICAgICBpbnQzMl90ICAgICAgICAgc2VxdWVuY2U7ICAgLy8gY2hhbmdlcyB3aGVuIHZpc2libGUgcmVnaW9ucyBjYW4gY2hhbmdlCi0gICAgICAgICAgICAgICAgdWludDMyX3QgICAgICAgIHRpbnQ7Ci0gICAgICAgICAgICAgICAgVHJhbnNmb3JtICAgICAgIHRyYW5zZm9ybTsKLSAgICAgICAgICAgICAgICBSZWdpb24gICAgICAgICAgdHJhbnNwYXJlbnRSZWdpb247Ci0gICAgICAgICAgICB9OwotCi0gICAgICAgICAgICAvLyBtb2RpZnkgY3VycmVudCBzdGF0ZQotICAgICAgICAgICAgYm9vbCBzZXRQb3NpdGlvbihpbnQzMl90IHgsIGludDMyX3QgeSk7Ci0gICAgICAgICAgICBib29sIHNldExheWVyKHVpbnQzMl90IHopOwotICAgICAgICAgICAgYm9vbCBzZXRTaXplKHVpbnQzMl90IHcsIHVpbnQzMl90IGgpOwotICAgICAgICAgICAgYm9vbCBzZXRBbHBoYSh1aW50OF90IGFscGhhKTsKLSAgICAgICAgICAgIGJvb2wgc2V0TWF0cml4KGNvbnN0IGxheWVyX3N0YXRlX3Q6Om1hdHJpeDIyX3QmIG1hdHJpeCk7Ci0gICAgICAgICAgICBib29sIHNldFRyYW5zcGFyZW50UmVnaW9uSGludChjb25zdCBSZWdpb24mIG9wYXF1ZSk7Ci0gICAgICAgICAgICBib29sIHNldEZsYWdzKHVpbnQ4X3QgZmxhZ3MsIHVpbnQ4X3QgbWFzayk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIHZvaWQgY29tbWl0VHJhbnNhY3Rpb24oYm9vbCBza2lwU2l6ZSk7Ci0gICAgICAgICAgICBib29sIHJlcXVlc3RUcmFuc2FjdGlvbigpOwotICAgICAgICAgICAgdm9pZCBmb3JjZVZpc2liaWxpdHlUcmFuc2FjdGlvbigpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICB1aW50MzJfdCBnZXRUcmFuc2FjdGlvbkZsYWdzKHVpbnQzMl90IGZsYWdzKTsKLSAgICAgICAgICAgIHVpbnQzMl90IHNldFRyYW5zYWN0aW9uRmxhZ3ModWludDMyX3QgZmxhZ3MpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBSZWN0IHZpc2libGVCb3VuZHMoKSBjb25zdDsKLSAgICAgICAgICAgIHZvaWQgZHJhd1JlZ2lvbihjb25zdCBSZWdpb24mIHJlZykgY29uc3Q7Ci0KLSAgICAgICAgICAgIHZvaWQgaW52YWxpZGF0ZSgpOwotICAgICAgICAgICAgCi0gICAgLyoqCi0gICAgICogZHJhdyAtIHBlcmZvcm1zIHNvbWUgZ2xvYmFsIGNsaXBwaW5nIG9wdGltaXphdGlvbnMKLSAgICAgKiBhbmQgY2FsbHMgb25EcmF3KCkuCi0gICAgICogVHlwaWNhbGx5IHRoaXMgbWV0aG9kIGlzIG5vdCBvdmVycmlkZGVuLCBpbnN0ZWFkIGltcGxlbWVudCBvbkRyYXcoKQotICAgICAqIHRvIHBlcmZvcm0gdGhlIGFjdHVhbCBkcmF3aW5nLiAgCi0gICAgICovCi0gICAgdmlydHVhbCB2b2lkIGRyYXcoY29uc3QgUmVnaW9uJiBjbGlwKSBjb25zdDsKLSAgICAKLSAgICAvKioKLSAgICAgKiBvbkRyYXcgLSBkcmF3cyB0aGUgc3VyZmFjZS4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHZvaWQgb25EcmF3KGNvbnN0IFJlZ2lvbiYgY2xpcCkgY29uc3QgPSAwOwotICAgIAotICAgIC8qKgotICAgICAqIGluaXRTdGF0ZXMgLSBjYWxsZWQganVzdCBhZnRlciBjb25zdHJ1Y3Rpb24KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHZvaWQgaW5pdFN0YXRlcyh1aW50MzJfdCB3LCB1aW50MzJfdCBoLCB1aW50MzJfdCBmbGFncyk7Ci0gICAgCi0gICAgLyoqCi0gICAgICogc2V0U2l6ZUNoYW5nZWQgLSBjYWxsZWQgd2hlbiB0aGUgKmN1cnJlbnQqIHN0YXRlJ3Mgc2l6ZSBpcyBjaGFuZ2VkLgotICAgICAqLwotICAgIHZpcnR1YWwgdm9pZCBzZXRTaXplQ2hhbmdlZCh1aW50MzJfdCB3LCB1aW50MzJfdCBoKTsKLSAgICAKLSAgICAvKioKLSAgICAgKiBkb1RyYW5zYWN0aW9uIC0gcHJvY2VzcyB0aGUgdHJhbnNhY3Rpb24uIFRoaXMgaXMgYSBnb29kIHBsYWNlIHRvIGZpZ3VyZQotICAgICAqIG91dCB3aGljaCBhdHRyaWJ1dGVzIG9mIHRoZSBzdXJmYWNlIGhhdmUgY2hhbmdlZC4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHVpbnQzMl90IGRvVHJhbnNhY3Rpb24odWludDMyX3QgdHJhbnNhY3Rpb25GbGFncyk7Ci0gICAgCi0gICAgLyoqCi0gICAgICogc2V0VmlzaWJsZVJlZ2lvbiAtIGNhbGxlZCB0byBzZXQgdGhlIG5ldyB2aXNpYmxlIHJlZ2lvbi4gVGhpcyBnaXZlcwotICAgICAqIGEgY2hhbmNlIHRvIHVwZGF0ZSB0aGUgbmV3IHZpc2libGUgcmVnaW9uIG9yIHJlY29yZCB0aGUgZmFjdCBpdCBjaGFuZ2VkLgotICAgICAqLwotICAgIHZpcnR1YWwgdm9pZCBzZXRWaXNpYmxlUmVnaW9uKGNvbnN0IFJlZ2lvbiYgdmlzaWJsZVJlZ2lvbik7Ci0gICAgCi0gICAgLyoqCi0gICAgICogc2V0Q292ZXJlZFJlZ2lvbiAtIGNhbGxlZCB3aGVuIHRoZSBjb3ZlcmVkIHJlZ2lvbiBjaGFuZ2VzLiBUaGUgY292ZXJlZAotICAgICAqIHJlZ2lvbiBjb3JyZXNwb25kIHRvIGFueSBhcmVhIG9mIHRoZSBzdXJmYWNlIHRoYXQgaXMgY292ZXJlZCAKLSAgICAgKiAodHJhbnNwYXJlbnRseSBvciBub3QpIGJ5IGFub3RoZXIgc3VyZmFjZS4KLSAgICAgKi8KLSAgICB2aXJ0dWFsIHZvaWQgc2V0Q292ZXJlZFJlZ2lvbihjb25zdCBSZWdpb24mIGNvdmVyZWRSZWdpb24pOwotICAgIAotICAgIC8qKgotICAgICAqIGdldFBoeXNpY2FsU2l6ZSAtIHJldHVybnMgdGhlIHBoeXNpY2FsIHNpemUgb2YgdGhlIGRyYXdpbmcgc3RhdGUgb2YKLSAgICAgKiB0aGUgc3VyZmFjZS4gSWYgdGhlIHN1cmZhY2UgaXMgYmFja2VkIGJ5IGEgYml0bWFwLCB0aGlzIGlzIHRoZSBzaXplIG9mCi0gICAgICogdGhlIGJpdG1hcCAoYXMgb3Bwb3NlZCB0byB0aGUgc2l6ZSBvZiB0aGUgZHJhd2luZyBzdGF0ZSkuCi0gICAgICovCi0gICAgdmlydHVhbCBQb2ludCBnZXRQaHlzaWNhbFNpemUoKSBjb25zdDsKLQotICAgIC8qKgotICAgICAqIHZhbGlkYXRlVmlzaWJpbGl0eSAtIGNhY2hlIGEgYnVuY2ggb2YgdGhpbmdzCi0gICAgICovCi0gICAgdmlydHVhbCB2b2lkIHZhbGlkYXRlVmlzaWJpbGl0eShjb25zdCBUcmFuc2Zvcm0mIGdsb2JhbFRyYW5zZm9ybSk7Ci0KLSAgICAvKioKLSAgICAgKiBsb2NrUGFnZUZsaXAgLSBjYWxsZWQgZWFjaCB0aW1lIHRoZSBzY3JlZW4gaXMgcmVkcmF3biBhbmQgcmV0dXJucyB3aGV0aGVyCi0gICAgICogdGhlIHZpc2libGUgcmVnaW9ucyBuZWVkIHRvIGJlIHJlY29tcHV0ZWQgKHRoaXMgaXMgYSBmYWlybHkgaGVhdnkKLSAgICAgKiBvcGVyYXRpb24sIHNvIHRoaXMgc2hvdWxkIGJlIHNldCBvbmx5IGlmIG5lZWRlZCkuIFR5cGljYWxseSB0aGlzIGlzIHVzZWQKLSAgICAgKiB0byBmaWd1cmUgb3V0IGlmIHRoZSBjb250ZW50IG9yIHNpemUgb2YgYSBzdXJmYWNlIGhhcyBjaGFuZ2VkLgotICAgICAqLwotICAgIHZpcnR1YWwgdm9pZCBsb2NrUGFnZUZsaXAoYm9vbCYgcmVjb21wdXRlVmlzaWJsZVJlZ2lvbnMpOwotICAgIAotICAgIC8qKgotICAgICAqIHVubG9ja1BhZ2VGbGlwIC0gY2FsbGVkIGVhY2ggdGltZSB0aGUgc2NyZWVuIGlzIHJlZHJhd24uIHVwZGF0ZXMgdGhlCi0gICAgICogZmluYWwgZGlydHkgcmVnaW9uIHdydCB0aGUgcGxhbmVUcmFuc2Zvcm0uCi0gICAgICogQXQgdGhpcyBwb2ludCwgYWxsIHZpc2libGUgcmVnaW9ucywgc3VyZmFjZSBwb3NpdGlvbiBhbmQgc2l6ZSwgZXRjLi4uIGFyZQotICAgICAqIGNvcnJlY3QuCi0gICAgICovCi0gICAgdmlydHVhbCB2b2lkIHVubG9ja1BhZ2VGbGlwKGNvbnN0IFRyYW5zZm9ybSYgcGxhbmVUcmFuc2Zvcm0sIFJlZ2lvbiYgb3V0RGlydHlSZWdpb24pOwotICAgIAotICAgIC8qKgotICAgICAqIGZpbmlzaFBhZ2VGbGlwIC0gY2FsbGVkIGFmdGVyIGFsbCBzdXJmYWNlcyBoYXZlIGRyYXduLgotICAgICAqLwotICAgIHZpcnR1YWwgdm9pZCBmaW5pc2hQYWdlRmxpcCgpOwotICAgIAotICAgIC8qKgotICAgICAqIG5lZWRzQmxlbmRpbmcgLSB0cnVlIGlmIHRoaXMgc3VyZmFjZSBuZWVkcyBibGVuZGluZwotICAgICAqLwotICAgIHZpcnR1YWwgYm9vbCBuZWVkc0JsZW5kaW5nKCkgY29uc3QgIHsgcmV0dXJuIGZhbHNlOyB9Ci0KLSAgICAvKioKLSAgICAgKiB0cmFuc2Zvcm1lZCAtLSB0cnVlIGlzIHRoaXMgc3VyZmFjZSBuZWVkcyBhIHRvIGJlIHRyYW5zZm9ybWVkCi0gICAgICovCi0gICAgdmlydHVhbCBib29sIHRyYW5zZm9ybWVkKCkgY29uc3QgICAgeyByZXR1cm4gbVRyYW5zZm9ybWVkOyB9Ci0KLSAgICAvKioKLSAgICAgKiBpc1NlY3VyZSAtIHRydWUgaWYgdGhpcyBzdXJmYWNlIGlzIHNlY3VyZSwgdGhhdCBpcyBpZiBpdCBwcmV2ZW50cwotICAgICAqIHNjcmVlbnNob3RzIG9yIHZucyBzZXJ2ZXJzLgotICAgICAqLwotICAgIHZpcnR1YWwgYm9vbCBpc1NlY3VyZSgpIGNvbnN0ICAgICAgIHsgcmV0dXJuIGZhbHNlOyB9Ci0KLSAgICAgICAgICAgIGVudW0geyAvLyBmbGFncyBmb3IgZG9UcmFuc2FjdGlvbigpCi0gICAgICAgICAgICAgICAgZVZpc2libGVSZWdpb24gICAgICA9IDB4MDAwMDAwMDIsCi0gICAgICAgICAgICAgICAgZVJlc3RhcnRUcmFuc2FjdGlvbiA9IDB4MDAwMDAwMDgKLSAgICAgICAgICAgIH07Ci0KLQotICAgIGlubGluZSAgY29uc3QgU3RhdGUmICAgIGRyYXdpbmdTdGF0ZSgpIGNvbnN0ICAgIHsgcmV0dXJuIG1EcmF3aW5nU3RhdGU7IH0KLSAgICBpbmxpbmUgIGNvbnN0IFN0YXRlJiAgICBjdXJyZW50U3RhdGUoKSBjb25zdCAgICB7IHJldHVybiBtQ3VycmVudFN0YXRlOyB9Ci0gICAgaW5saW5lICBTdGF0ZSYgICAgICAgICAgY3VycmVudFN0YXRlKCkgICAgICAgICAgeyByZXR1cm4gbUN1cnJlbnRTdGF0ZTsgfQotCi0gICAgc3RhdGljIGludCBjb21wYXJlQ3VycmVudFN0YXRlWihMYXllckJhc2UqY29uc3QqIGxheWVyQSwgTGF5ZXJCYXNlKmNvbnN0KiBsYXllckIpIHsKLSAgICAgICAgcmV0dXJuIGxheWVyQVswXS0+Y3VycmVudFN0YXRlKCkueiAtIGxheWVyQlswXS0+Y3VycmVudFN0YXRlKCkuejsKLSAgICB9Ci0KLSAgICBpbnQzMl90ICBnZXRPcmllbnRhdGlvbigpIGNvbnN0IHsgcmV0dXJuIG1PcmllbnRhdGlvbjsgfQotICAgIGludCAgdHgoKSBjb25zdCAgICAgICAgICAgICB7IHJldHVybiBtTGVmdDsgfQotICAgIGludCAgdHkoKSBjb25zdCAgICAgICAgICAgICB7IHJldHVybiBtVG9wOyB9Ci0gICAgCi1wcm90ZWN0ZWQ6Ci0gICAgY29uc3QgR3JhcGhpY1BsYW5lJiBncmFwaGljUGxhbmUoaW50IGRweSkgY29uc3Q7Ci0gICAgICAgICAgR3JhcGhpY1BsYW5lJiBncmFwaGljUGxhbmUoaW50IGRweSk7Ci0KLSAgICAgICAgICBHTHVpbnQgY3JlYXRlVGV4dHVyZSgpIGNvbnN0OwotICAgIAotICAgICAgICAgIHZvaWQgZHJhd1dpdGhPcGVuR0woY29uc3QgUmVnaW9uJiBjbGlwLAotICAgICAgICAgICAgICAgICAgR0xpbnQgdGV4dHVyZU5hbWUsCi0gICAgICAgICAgICAgICAgICBjb25zdCBHR0xTdXJmYWNlJiBzdXJmYWNlLAotICAgICAgICAgICAgICAgICAgaW50IHRyYW5zZm9ybSA9IDApIGNvbnN0OwotCi0gICAgICAgICAgdm9pZCBjbGVhcldpdGhPcGVuR0woY29uc3QgUmVnaW9uJiBjbGlwKSBjb25zdDsKLQotICAgICAgICAgIHZvaWQgbG9hZFRleHR1cmUoY29uc3QgUmVnaW9uJiBkaXJ0eSwKLSAgICAgICAgICAgICAgICAgIEdMaW50IHRleHR1cmVOYW1lLCBjb25zdCBHR0xTdXJmYWNlJiB0LAotICAgICAgICAgICAgICAgICAgR0x1aW50JiB0ZXh0dXJlV2lkdGgsIEdMdWludCYgdGV4dHVyZUhlaWdodCkgY29uc3Q7Ci0KLSAgICAgICAgICBib29sIGNhblVzZUNvcHliaXQoKSBjb25zdDsKLSAgICAgICAgICAKLSAgICAgICAgICAgICAgICBTdXJmYWNlRmxpbmdlciogbUZsaW5nZXI7Ci0gICAgICAgICAgICAgICAgdWludDMyX3QgICAgICAgIG1GbGFnczsKLQotICAgICAgICAgICAgICAgIC8vIGNhY2hlZCBkdXJpbmcgdmFsaWRhdGVWaXNpYmlsaXR5KCkKLSAgICAgICAgICAgICAgICBib29sICAgICAgICAgICAgbVRyYW5zZm9ybWVkOwotICAgICAgICAgICAgICAgIGludDMyX3QgICAgICAgICBtT3JpZW50YXRpb247Ci0gICAgICAgICAgICAgICAgR0xmaXhlZCAgICAgICAgIG1WZXJ0aWNlc1s0XVsyXTsKLSAgICAgICAgICAgICAgICBSZWN0ICAgICAgICAgICAgbVRyYW5zZm9ybWVkQm91bmRzOwotICAgICAgICAgICAgICAgIGJvb2wgICAgICAgICAgICBtQ2FuVXNlQ29weUJpdDsKLSAgICAgICAgICAgICAgICBpbnQgICAgICAgICAgICAgbUxlZnQ7Ci0gICAgICAgICAgICAgICAgaW50ICAgICAgICAgICAgIG1Ub3A7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAvLyB0aGVzZSBhcmUgcHJvdGVjdGVkIGJ5IGFuIGV4dGVybmFsIGxvY2sKLSAgICAgICAgICAgICAgICBTdGF0ZSAgICAgICAgICAgbUN1cnJlbnRTdGF0ZTsKLSAgICAgICAgICAgICAgICBTdGF0ZSAgICAgICAgICAgbURyYXdpbmdTdGF0ZTsKLSAgICB2b2xhdGlsZSAgICBpbnQzMl90ICAgICAgICAgbVRyYW5zYWN0aW9uRmxhZ3M7Ci0KLSAgICAgICAgICAgICAgICAvLyBkb24ndCBjaGFuZ2UsIGRvbid0IG5lZWQgYSBsb2NrCi0gICAgICAgICAgICAgICAgYm9vbCAgICAgICAgICAgIG1QcmVtdWx0aXBsaWVkQWxwaGE7Ci0KLSAgICAgICAgICAgICAgICAvLyBvbmx5IHJlYWQKLSAgICBjb25zdCAgICAgICB1aW50MzJfdCAgICAgICAgbUlkZW50aXR5OwotICAgICAKLSAgICAgICAgICAgICAgICAvLyBhdG9taWMKLSAgICB2b2xhdGlsZSAgICBpbnQzMl90ICAgICAgICAgbUludmFsaWRhdGU7Ci0gICAgICAgICAgICAgICAgCi0KLXByaXZhdGU6Ci0gICAgICAgICAgICAgICAgdm9pZCB2YWxpZGF0ZVRleHR1cmUoR0xpbnQgdGV4dHVyZU5hbWUpIGNvbnN0OwotICAgIHN0YXRpYyAgICAgIGludDMyX3QgICAgICAgICBzSWRlbnRpdHk7Ci19OwotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBMYXllckJhc2VDbGllbnQgOiBwdWJsaWMgTGF5ZXJCYXNlCi17Ci1wdWJsaWM6Ci0gICAgY2xhc3MgU3VyZmFjZTsKLSAgIHN0YXRpYyBjb25zdCB1aW50MzJfdCB0eXBlSW5mbzsKLSAgICBzdGF0aWMgY29uc3QgY2hhciogY29uc3QgdHlwZUlEOwotICAgIHZpcnR1YWwgY2hhciBjb25zdCogZ2V0VHlwZUlEKCkgY29uc3QgeyByZXR1cm4gdHlwZUlEOyB9Ci0gICAgdmlydHVhbCB1aW50MzJfdCBnZXRUeXBlSW5mbygpIGNvbnN0IHsgcmV0dXJuIHR5cGVJbmZvOyB9Ci0KLSAgICBMYXllckJhc2VDbGllbnQoU3VyZmFjZUZsaW5nZXIqIGZsaW5nZXIsIERpc3BsYXlJRCBkaXNwbGF5LCAKLSAgICAgICAgICAgIENsaWVudCogY2xpZW50LCBpbnQzMl90IGkpOwotICAgIHZpcnR1YWwgfkxheWVyQmFzZUNsaWVudCgpOwotCi0KLSAgICBDbGllbnQqICAgICAgICAgICAgIGNvbnN0IGNsaWVudDsKLSAgICBsYXllcl9jYmxrX3QqICAgICAgIGNvbnN0IGxjYmxrOwotCi0gICAgaW5saW5lICBpbnQzMl90ICAgICBjbGllbnRJbmRleCgpIGNvbnN0IHsgcmV0dXJuIG1JbmRleDsgfQotICAgICAgICAgICAgaW50MzJfdCAgICAgc2VydmVySW5kZXgoKSBjb25zdDsKLQotICAgIHZpcnR1YWwgc3A8U3VyZmFjZT4gZ2V0U3VyZmFjZSgpIGNvbnN0OwotICAgCi0gICAgICAgICAgICB1aW50MzJfdCAgICBnZXRJZGVudGl0eSgpIGNvbnN0IHsgcmV0dXJuIG1JZGVudGl0eTsgfQotCi0gICAgY2xhc3MgU3VyZmFjZSA6IHB1YmxpYyBCblN1cmZhY2UgCi0gICAgewotICAgIHB1YmxpYzoKLSAgICAgICAgU3VyZmFjZShTdXJmYWNlSUQgaWQsIGludCBpZGVudGl0eSkgeyAKLSAgICAgICAgICAgIG1QYXJhbXMudG9rZW4gPSBpZDsKLSAgICAgICAgICAgIG1QYXJhbXMuaWRlbnRpdHkgPSBpZGVudGl0eTsKLSAgICAgICAgfQotICAgICAgICBTdXJmYWNlKFN1cmZhY2VJRCBpZCwgCi0gICAgICAgICAgICAgICAgY29uc3Qgc3A8SU1lbW9yeUhlYXA+JiBoZWFwMCwKLSAgICAgICAgICAgICAgICBjb25zdCBzcDxJTWVtb3J5SGVhcD4mIGhlYXAxLAotICAgICAgICAgICAgICAgIGludCBpZGVudGl0eSkKLSAgICAgICAgewotICAgICAgICAgICAgbVBhcmFtcy50b2tlbiA9IGlkOwotICAgICAgICAgICAgbVBhcmFtcy5pZGVudGl0eSA9IGlkZW50aXR5OwotICAgICAgICAgICAgbVBhcmFtcy5oZWFwWzBdID0gaGVhcDA7Ci0gICAgICAgICAgICBtUGFyYW1zLmhlYXBbMV0gPSBoZWFwMTsKLSAgICAgICAgfQotICAgICAgICB2aXJ0dWFsIH5TdXJmYWNlKCkgewotICAgICAgICAgICAgLy8gVE9ETzogV2Ugbm93IGhhdmUgYSBwb2ludCBoZXJlIHdlcmUgd2UgY2FuIGNsZWFuLXVwIHRoZQotICAgICAgICAgICAgLy8gY2xpZW50J3MgbWVzcy4KLSAgICAgICAgICAgIC8vIFRoaXMgaXMgYWxzbyB3aGVyZSBzdXJmYWNlIGlkIHNob3VsZCBiZSByZWN5Y2xlZC4KLSAgICAgICAgICAgIC8vTE9HRCgiU3VyZmFjZSAlZCwgaGVhcHM9eyVwLCAlcH0gZGVzdHJveWVkIiwKLSAgICAgICAgICAgIC8vICAgICAgICBtSWQsIG1IZWFwWzBdLmdldCgpLCBtSGVhcFsxXS5nZXQoKSk7Ci0gICAgICAgIH0KLQotICAgICAgICB2aXJ0dWFsIHZvaWQgZ2V0U3VyZmFjZURhdGEoCi0gICAgICAgICAgICAgICAgSVN1cmZhY2VGbGluZ2VyQ2xpZW50OjpzdXJmYWNlX2RhdGFfdCogcGFyYW1zKSBjb25zdCB7Ci0gICAgICAgICAgICAqcGFyYW1zID0gbVBhcmFtczsKLSAgICAgICAgfQotCi0gICAgICAgIHZpcnR1YWwgc3RhdHVzX3QgcmVnaXN0ZXJCdWZmZXJzKGNvbnN0IElTdXJmYWNlOjpCdWZmZXJIZWFwJiBidWZmZXJzKSAKLSAgICAgICAgICAgICAgICB7IHJldHVybiBJTlZBTElEX09QRVJBVElPTjsgfQotICAgICAgICB2aXJ0dWFsIHZvaWQgcG9zdEJ1ZmZlcihzc2l6ZV90IG9mZnNldCkgeyB9Ci0gICAgICAgIHZpcnR1YWwgdm9pZCB1bnJlZ2lzdGVyQnVmZmVycygpIHsgfTsKLSAgICAgICAgdmlydHVhbCBzcDxPdmVybGF5UmVmPiBjcmVhdGVPdmVybGF5KAotICAgICAgICAgICAgICAgIHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIGludDMyX3QgZm9ybWF0KSB7Ci0gICAgICAgICAgICByZXR1cm4gTlVMTDsKLSAgICAgICAgfTsKLQotICAgIHByaXZhdGU6Ci0gICAgICAgIElTdXJmYWNlRmxpbmdlckNsaWVudDo6c3VyZmFjZV9kYXRhX3QgbVBhcmFtczsKLSAgICB9OwotCi1wcml2YXRlOgotICAgIGludDMyX3QgbUluZGV4OwotCi19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfTEFZRVJfQkFTRV9ICmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyQml0bWFwLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCaXRtYXAuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlODQ0MzUwLi4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCaXRtYXAuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTg1ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIlN1cmZhY2VGbGluZ2VyIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDxjdXRpbHMvbWVtb3J5Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5RGVhbGVyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSU1lbW9yeS5oPgotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0jaW5jbHVkZSA8cGl4ZWxmbGluZ2VyL3BpeGVsZmxpbmdlci5oPgotCi0jaW5jbHVkZSAiTGF5ZXJCaXRtYXAuaCIKLSNpbmNsdWRlICJTdXJmYWNlRmxpbmdlci5oIgotI2luY2x1ZGUgIlZSYW1IZWFwLmgiCi0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotTGF5ZXJCaXRtYXA6OkxheWVyQml0bWFwKCkKLSAgICA6IG1BbGxvY0ZsYWdzKDApLCBtT2Zmc2V0KDApLCBtU2l6ZSgtMVUpLCBtQWxpZ25tZW50KDIpCi17Ci0gICAgbWVtc2V0KCZtU3VyZmFjZSwgMCwgc2l6ZW9mKG1TdXJmYWNlKSk7Ci19Ci0KLUxheWVyQml0bWFwOjp+TGF5ZXJCaXRtYXAoKQotewotICAgIG1TdXJmYWNlLmRhdGEgPSAwOwotfQotCi1zdGF0dXNfdCBMYXllckJpdG1hcDo6aW5pdChjb25zdCBzcDxNZW1vcnlEZWFsZXI+JiBhbGxvY2F0b3IpCi17Ci0gICAgaWYgKG1BbGxvY2F0b3IgIT0gTlVMTCkKLSAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLSAgICBtQWxsb2NhdG9yID0gYWxsb2NhdG9yOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgTGF5ZXJCaXRtYXA6OnNldEJpdHModWludDMyX3QgdywgdWludDMyX3QgaCwgdWludDMyX3QgYWxpZ25tZW50LCAKLSAgICAgICAgUGl4ZWxGb3JtYXQgZm9ybWF0LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBjb25zdCBzcDxNZW1vcnlEZWFsZXI+JiBhbGxvY2F0b3IobUFsbG9jYXRvcik7Ci0gICAgaWYgKGFsbG9jYXRvciA9PSBOVUxMKQotICAgICAgICByZXR1cm4gTk9fSU5JVDsKLQotICAgIGlmIChVTkxJS0VMWSh3ID09IG1TdXJmYWNlLndpZHRoICYmIGggPT0gbVN1cmZhY2UuaGVpZ2h0ICYmCi0gICAgICAgICAgICBmb3JtYXQgPT0gbVN1cmZhY2UuZm9ybWF0KSkKLSAgICB7IC8vIHNhbWUgZm9ybWF0IGFuZCBzaXplLCBkbyBub3RoaW5nLgotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotCi0gICAgUGl4ZWxGb3JtYXRJbmZvIGluZm87Ci0gICAgZ2V0UGl4ZWxGb3JtYXRJbmZvKGZvcm1hdCwgJmluZm8pOwotCi0gICAgdWludDMyX3QgYWxsb2NGbGFncyA9IE1lbW9yeURlYWxlcjo6UEFHRV9BTElHTkVEOwotICAgIGNvbnN0IHVpbnQzMl90IGFsaWduID0gNDsgLy8gbXVzdCBtYXRjaCBHTF9VTlBBQ0tfQUxJR05NRU5UCi0gICAgY29uc3QgdWludDMyX3QgQnBwID0gaW5mby5ieXRlc1BlclBpeGVsOwotICAgIHVpbnQzMl90IHN0cmlkZSA9ICh3ICsgKGFsaWdubWVudC0xKSkgJiB+KGFsaWdubWVudC0xKTsKLSAgICBzdHJpZGUgPSAoKHN0cmlkZSAqIEJwcCArIChhbGlnbi0xKSkgJiB+KGFsaWduLTEpKSAvIEJwcDsKLSAgICBzaXplX3Qgc2l6ZSA9IGluZm8uZ2V0U2NhbmxpbmVTaXplKHN0cmlkZSkgKiBoOwotICAgIGlmIChhbGxvY0ZsYWdzICYgTWVtb3J5RGVhbGVyOjpQQUdFX0FMSUdORUQpIHsKLSAgICAgICAgc2l6ZV90IHBhZ2VzaXplID0gZ2V0cGFnZXNpemUoKTsKLSAgICAgICAgc2l6ZSA9IChzaXplICsgKHBhZ2VzaXplLTEpKSAmIH4ocGFnZXNpemUtMSk7Ci0gICAgfQotCi0gICAgLyogRklYTUU6IHdlIHNob3VsZCBiZSBhYmxlIHRvIGhhdmUgYSBoL3Ygc3RyaWRlIGJlY2F1c2UgdGhlIHVzZXIgb2YgdGhlCi0gICAgICogc3VyZmFjZSBtaWdodCBoYXZlIHN0cmlkZSBsaW1pdGF0aW9uIChmb3IgaW5zdGFuY2UgaC93IGNvZGVjcyBvZnRlbiBkbykKLSAgICAgKi8KLSAgICBpbnQzMl90IHZzdHJpZGUgPSAwOwotCi0gICAgbUFsaWdubWVudCA9IGFsaWdubWVudDsKLSAgICBtQWxsb2NGbGFncyA9IGFsbG9jRmxhZ3M7Ci0gICAgbU9mZnNldCA9IDA7Ci0gICAgaWYgKG1TaXplICE9IHNpemUpIHsKLSAgICAgICAgLy8gd291bGQgYmUgbmljZSB0byBoYXZlIGEgcmVhbGxvY2F0ZSgpIGFwaQotICAgICAgICBtQml0c01lbW9yeS5jbGVhcigpOyAvLyBmcmVlLW1lbW9yeQotICAgICAgICBtQml0c01lbW9yeSA9IGFsbG9jYXRvci0+YWxsb2NhdGUoc2l6ZSwgYWxsb2NGbGFncyk7Ci0gICAgICAgIG1TaXplID0gc2l6ZTsKLSAgICB9IGVsc2UgewotICAgICAgICAvLyBkb24ndCBlcmFzZSBtZW1vcnkgaWYgd2UgZGlkbid0IGhhdmUgdG8gcmVhbGxvY2F0ZQotICAgICAgICBmbGFncyAmPSB+U0VDVVJFX0JJVFM7Ci0gICAgfQotICAgIGlmIChtQml0c01lbW9yeSAhPSAwKSB7Ci0gICAgICAgIG1PZmZzZXQgPSBtQml0c01lbW9yeS0+b2Zmc2V0KCk7Ci0gICAgICAgIG1TdXJmYWNlLmRhdGEgPSBzdGF0aWNfY2FzdDxHR0x1Ynl0ZSo+KG1CaXRzTWVtb3J5LT5wb2ludGVyKCkpOwotICAgICAgICBtU3VyZmFjZS52ZXJzaW9uID0gc2l6ZW9mKEdHTFN1cmZhY2UpOwotICAgICAgICBtU3VyZmFjZS53aWR0aCAgPSB3OwotICAgICAgICBtU3VyZmFjZS5oZWlnaHQgPSBoOwotICAgICAgICBtU3VyZmFjZS5zdHJpZGUgPSBzdHJpZGU7Ci0gICAgICAgIG1TdXJmYWNlLnZzdHJpZGUgPSB2c3RyaWRlOwotICAgICAgICBtU3VyZmFjZS5mb3JtYXQgPSBmb3JtYXQ7Ci0gICAgICAgIGlmIChmbGFncyAmIFNFQ1VSRV9CSVRTKQotICAgICAgICAgICAgY2xlYXIoKTsKLSAgICB9Ci0KLSAgICBpZiAobUJpdHNNZW1vcnk9PTAgfHwgbVN1cmZhY2UuZGF0YT09MCkgewotICAgICAgICBMT0dFKCJub3QgZW5vdWdoIG1lbW9yeSBmb3IgbGF5ZXIgYml0bWFwIHNpemU9JXUiLCBzaXplKTsKLSAgICAgICAgYWxsb2NhdG9yLT5kdW1wKCJMYXllckJpdG1hcCIpOwotICAgICAgICBtU3VyZmFjZS5kYXRhID0gMDsKLSAgICAgICAgbVNpemUgPSAtMVU7Ci0gICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgfQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotdm9pZCBMYXllckJpdG1hcDo6Y2xlYXIoKQotewotICAgIC8vIE5PVEU6IHRoaXMgbWVtc2V0IHNob3VsZCBub3QgYmUgbmVjZXNzYXJ5LCBhdCBsZWFzdCBmb3IKLSAgICAvLyBvcGFxdWUgc3VyZmFjZS4gSG93ZXZlciwgZm9yIHNlY3VyaXR5IHJlYXNvbnMgaXQncyBiZXR0ZXIgdG8ga2VlcCBpdAotICAgIC8vIChpbiB0aGUgY2FzZSBvZiBwbWVtLCBpdCdzIHBvc3NpYmxlIHRoYXQgdGhlIG1lbW9yeSBjb250YWlucyBvbGQKLSAgICAvLyBkYXRhKQotICAgIGlmIChtU3VyZmFjZS5kYXRhKSB7Ci0gICAgICAgIG1lbXNldChtU3VyZmFjZS5kYXRhLCAwLCBtU2l6ZSk7Ci0gICAgICAgIC8vaWYgKGJ5dGVzUGVyUGl4ZWwobVN1cmZhY2UuZm9ybWF0KSA9PSA0KSB7Ci0gICAgICAgIC8vICAgIGFuZHJvaWRfbWVtc2V0MzIoKHVpbnQzMl90KiltU3VyZmFjZS5kYXRhLCAweEZGMDAwMEZGLCBtU2l6ZSk7Ci0gICAgICAgIC8vfSBlbHNlICB7Ci0gICAgICAgIC8vICAgIGFuZHJvaWRfbWVtc2V0MTYoKHVpbnQxNl90KiltU3VyZmFjZS5kYXRhLCAweEY4MDAsIG1TaXplKTsKLSAgICAgICAgLy99Ci0gICAgfQotfQotCi1zdGF0dXNfdCBMYXllckJpdG1hcDo6Z2V0SW5mbyhzdXJmYWNlX2luZm9fdCogaW5mbykgY29uc3QKLXsKLSAgICBpZiAobVN1cmZhY2UuZGF0YSA9PSAwKSB7Ci0gICAgICAgIG1lbXNldChpbmZvLCAwLCBzaXplb2Yoc3VyZmFjZV9pbmZvX3QpKTsKLSAgICAgICAgaW5mby0+Yml0c19vZmZzZXQgPSBOT19NRU1PUlk7Ci0gICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgfQotICAgIGluZm8tPncgICAgID0gdWludDE2X3Qod2lkdGgoKSk7Ci0gICAgaW5mby0+aCAgICAgPSB1aW50MTZfdChoZWlnaHQoKSk7Ci0gICAgaW5mby0+c3RyaWRlPSB1aW50MTZfdChzdHJpZGUoKSk7Ci0gICAgaW5mby0+YnByICAgPSB1aW50MTZfdChzdHJpZGUoKSAqIGJ5dGVzUGVyUGl4ZWwocGl4ZWxGb3JtYXQoKSkpOwotICAgIGluZm8tPmZvcm1hdD0gdWludDhfdChwaXhlbEZvcm1hdCgpKTsKLSAgICBpbmZvLT5mbGFncyA9IHN1cmZhY2VfaW5mb190OjplQnVmZmVyRGlydHk7Ci0gICAgaW5mby0+Yml0c19vZmZzZXQgPSBzc2l6ZV90KG1PZmZzZXQpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgTGF5ZXJCaXRtYXA6OnJlc2l6ZSh1aW50MzJfdCB3LCB1aW50MzJfdCBoKQotewotICAgIGludCBlcnIgPSBzZXRCaXRzKHcsIGgsIG1BbGlnbm1lbnQsIHBpeGVsRm9ybWF0KCksIFNFQ1VSRV9CSVRTKTsKLSAgICByZXR1cm4gZXJyOwotfQotCi1zaXplX3QgTGF5ZXJCaXRtYXA6OnNpemUoKSBjb25zdAotewotICAgIHJldHVybiBtU2l6ZTsKLX0KLQotdm9pZCBMYXllckJpdG1hcDo6Z2V0Qml0bWFwU3VyZmFjZShjb3B5Yml0X2ltYWdlX3QqIGltZykgY29uc3QKLXsKLSAgICBjb25zdCBzcDxJTWVtb3J5SGVhcD4mIG1oKGdldEFsbG9jYXRvcigpLT5nZXRNZW1vcnlIZWFwKCkpOwotICAgIHZvaWQqIHNiYXNlID0gbWgtPmJhc2UoKTsKLSAgICBjb25zdCBHR0xTdXJmYWNlJiB0KHN1cmZhY2UoKSk7Ci0gICAgaW1nLT53ID0gdC5zdHJpZGUgID86IHQud2lkdGg7Ci0gICAgaW1nLT5oID0gdC52c3RyaWRlID86IHQuaGVpZ2h0OwotICAgIGltZy0+Zm9ybWF0ID0gdC5mb3JtYXQ7Ci0gICAgaW1nLT5vZmZzZXQgPSBpbnRwdHJfdCh0LmRhdGEpIC0gaW50cHRyX3Qoc2Jhc2UpOwotICAgIGltZy0+YmFzZSA9IHNiYXNlOwotICAgIGltZy0+ZmQgPSBtaC0+aGVhcElEKCk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllckJpdG1hcC5oIGIvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllckJpdG1hcC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5YWQ2NGM0Li4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCaXRtYXAuaAorKysgL2Rldi9udWxsCkBAIC0xLDg0ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfTEFZRVJfQklUTUFQX0gKLSNkZWZpbmUgQU5EUk9JRF9MQVlFUl9CSVRNQVBfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1aS9QaXhlbEZvcm1hdC5oPgotI2luY2x1ZGUgPHVpL1JlY3QuaD4KLSNpbmNsdWRlIDxwcml2YXRlL3VpL1NoYXJlZFN0YXRlLmg+Ci0jaW5jbHVkZSA8cGl4ZWxmbGluZ2VyL3BpeGVsZmxpbmdlci5oPgotCi1jbGFzcyBjb3B5Yml0X2ltYWdlX3Q7Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIElNZW1vcnk7Ci1jbGFzcyBNZW1vcnlEZWFsZXI7Ci1jbGFzcyBMYXllckJpdG1hcDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIExheWVyQml0bWFwCi17Ci1wdWJsaWM6Ci0KLSAgICBlbnVtIHsKLSAgICAgICAgLy8gZXJhc2UgbWVtb3J5IHRvIGVuc3VyZSBzZWN1cml0eSB3aGVuIG5lY2Vzc2FyeQotICAgICAgICBTRUNVUkVfQklUUyA9IDB4MDAwMDAwMDEKLSAgICB9OwotCi0gICAgICAgICAgICAgICAgTGF5ZXJCaXRtYXAoKTsKLSAgICAgICAgICAgICAgICB+TGF5ZXJCaXRtYXAoKTsKLSAgICBzdGF0dXNfdCAgICBpbml0KGNvbnN0IHNwPE1lbW9yeURlYWxlcj4mIGFsbG9jYXRvcik7Ci0KLSAgICBzdGF0dXNfdCAgICBzZXRCaXRzKHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIHVpbnQzMl90IGFsaWdubWVudCwKLSAgICAgICAgICAgICAgICAgICAgICAgIFBpeGVsRm9ybWF0IGZvcm1hdCwgdWludDMyX3QgZmxhZ3MgPSAwKTsKLSAgICB2b2lkICAgICAgICBjbGVhcigpOwotCi0gICAgc3RhdHVzX3QgICAgZ2V0SW5mbyhzdXJmYWNlX2luZm9fdCogaW5mbykgY29uc3Q7Ci0gICAgc3RhdHVzX3QgICAgcmVzaXplKHVpbnQzMl90IHcsIHVpbnQzMl90IGgpOwotCi0gICAgY29uc3QgR0dMU3VyZmFjZSYgc3VyZmFjZSgpIGNvbnN0ICAgeyByZXR1cm4gbVN1cmZhY2U7IH0KLSAgICBSZWN0IGJvdW5kcygpIGNvbnN0ICAgICAgICAgICAgICAgICB7IHJldHVybiBSZWN0KHdpZHRoKCksIGhlaWdodCgpKTsgfQotICAgIHVpbnQzMl90IHdpZHRoKCkgY29uc3QgICAgICAgICAgICAgIHsgcmV0dXJuIHN1cmZhY2UoKS53aWR0aDsgfQotICAgIHVpbnQzMl90IGhlaWdodCgpIGNvbnN0ICAgICAgICAgICAgIHsgcmV0dXJuIHN1cmZhY2UoKS5oZWlnaHQ7IH0KLSAgICB1aW50MzJfdCBzdHJpZGUoKSBjb25zdCAgICAgICAgICAgICB7IHJldHVybiBzdXJmYWNlKCkuc3RyaWRlOyB9Ci0gICAgUGl4ZWxGb3JtYXQgcGl4ZWxGb3JtYXQoKSBjb25zdCAgICAgeyByZXR1cm4gc3VyZmFjZSgpLmZvcm1hdDsgfQotICAgIHZvaWQqIHNlcnZlckJpdHMoKSBjb25zdCAgICAgICAgICAgIHsgcmV0dXJuIHN1cmZhY2UoKS5kYXRhOyB9Ci0gICAgc2l6ZV90IHNpemUoKSBjb25zdDsKLSAgICBjb25zdCBzcDxNZW1vcnlEZWFsZXI+JiBnZXRBbGxvY2F0b3IoKSBjb25zdCB7IHJldHVybiBtQWxsb2NhdG9yOyB9Ci0gICAgdm9pZCBnZXRCaXRtYXBTdXJmYWNlKGNvcHliaXRfaW1hZ2VfdCogaW1nKSBjb25zdDsKLQotcHJpdmF0ZToKLSAgICBzcDxNZW1vcnlEZWFsZXI+ICAgICAgICBtQWxsb2NhdG9yOwotICAgIHNwPElNZW1vcnk+ICAgICAgICAgICAgIG1CaXRzTWVtb3J5OwotICAgIHVpbnQzMl90ICAgICAgICAgICAgICAgIG1BbGxvY0ZsYWdzOwotICAgIHNzaXplX3QgICAgICAgICAgICAgICAgIG1PZmZzZXQ7Ci0gICAgR0dMU3VyZmFjZSAgICAgICAgICAgICAgbVN1cmZhY2U7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICAgICAgbVNpemU7Ci0gICAgdWludDMyX3QgICAgICAgICAgICAgICAgbUFsaWdubWVudDsKLX07Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX0xBWUVSX0JJVE1BUF9ICmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyQmx1ci5jcHAgYi9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyQmx1ci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGQzZTQ1NmYuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllckJsdXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMjM0ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIlN1cmZhY2VGbGluZ2VyIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLQotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLSNpbmNsdWRlIDxHTEVTL2dsZXh0Lmg+Ci0KLSNpbmNsdWRlICJCbHVyRmlsdGVyLmgiCi0jaW5jbHVkZSAiTGF5ZXJCbHVyLmgiCi0jaW5jbHVkZSAiU3VyZmFjZUZsaW5nZXIuaCIKLSNpbmNsdWRlICJEaXNwbGF5SGFyZHdhcmUvRGlzcGxheUhhcmR3YXJlLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jb25zdCB1aW50MzJfdCBMYXllckJsdXI6OnR5cGVJbmZvID0gTGF5ZXJCYXNlQ2xpZW50Ojp0eXBlSW5mbyB8IDg7Ci1jb25zdCBjaGFyKiBjb25zdCBMYXllckJsdXI6OnR5cGVJRCA9ICJMYXllckJsdXIiOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotTGF5ZXJCbHVyOjpMYXllckJsdXIoU3VyZmFjZUZsaW5nZXIqIGZsaW5nZXIsIERpc3BsYXlJRCBkaXNwbGF5LAotICAgICAgICBDbGllbnQqIGNsaWVudCwgaW50MzJfdCBpKQotICAgICA6IExheWVyQmFzZUNsaWVudChmbGluZ2VyLCBkaXNwbGF5LCBjbGllbnQsIGkpLCBtQ2FjaGVEaXJ0eSh0cnVlKSwKLSAgICAgbVJlZnJlc2hDYWNoZSh0cnVlKSwgbUNhY2hlQWdlKDApLCBtVGV4dHVyZU5hbWUoLTFVKQotewotfQotCi1MYXllckJsdXI6On5MYXllckJsdXIoKQotewotICAgIGlmIChtVGV4dHVyZU5hbWUgIT0gLTFVKSB7Ci0gICAgICAgIC8vZ2xEZWxldGVUZXh0dXJlcygxLCAmbVRleHR1cmVOYW1lKTsKLSAgICAgICAgZGVsZXRlZFRleHR1cmVzLmFkZChtVGV4dHVyZU5hbWUpOwotICAgIH0KLX0KLQotdm9pZCBMYXllckJsdXI6OnNldFZpc2libGVSZWdpb24oY29uc3QgUmVnaW9uJiB2aXNpYmxlUmVnaW9uKQotewotICAgIExheWVyQmFzZUNsaWVudDo6c2V0VmlzaWJsZVJlZ2lvbih2aXNpYmxlUmVnaW9uKTsKLSAgICBpZiAodmlzaWJsZVJlZ2lvblNjcmVlbi5pc0VtcHR5KCkpIHsKLSAgICAgICAgaWYgKG1UZXh0dXJlTmFtZSAhPSAtMVUpIHsKLSAgICAgICAgICAgIC8vIFdlJ3JlIG5vdCB2aXNpYmxlLCBmcmVlIHRoZSB0ZXh0dXJlIHVwLgotICAgICAgICAgICAgZ2xCaW5kVGV4dHVyZShHTF9URVhUVVJFXzJELCAwKTsKLSAgICAgICAgICAgIGdsRGVsZXRlVGV4dHVyZXMoMSwgJm1UZXh0dXJlTmFtZSk7Ci0gICAgICAgICAgICBtVGV4dHVyZU5hbWUgPSAtMVU7Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLXVpbnQzMl90IExheWVyQmx1cjo6ZG9UcmFuc2FjdGlvbih1aW50MzJfdCBmbGFncykKLXsKLSAgICAvLyB3ZSdyZSBkb2luZyBhIHRyYW5zYWN0aW9uLCByZWZyZXNoIHRoZSBjYWNoZSEKLSAgICBpZiAoIW1GbGluZ2VyLT5pc0Zyb3plbigpKSB7Ci0gICAgICAgIG1SZWZyZXNoQ2FjaGUgPSB0cnVlOwotICAgICAgICBtQ2FjaGVEaXJ0eSA9IHRydWU7Ci0gICAgICAgIGZsYWdzIHw9IGVWaXNpYmxlUmVnaW9uOwotICAgICAgICB0aGlzLT5jb250ZW50RGlydHkgPSB0cnVlOwotICAgIH0KLSAgICByZXR1cm4gTGF5ZXJCYXNlOjpkb1RyYW5zYWN0aW9uKGZsYWdzKTsgICAgCi19Ci0KLXZvaWQgTGF5ZXJCbHVyOjp1bmxvY2tQYWdlRmxpcChjb25zdCBUcmFuc2Zvcm0mIHBsYW5lVHJhbnNmb3JtLCBSZWdpb24mIG91dERpcnR5UmVnaW9uKQotewotICAgIC8vIHRoaXMgY29kZS1wYXRoIG11c3QgYmUgYXMgdGlnaHQgYXMgcG9zc2libGUsIGl0J3MgY2FsbGVkIGVhY2ggdGltZQotICAgIC8vIHRoZSBzY3JlZW4gaXMgY29tcG9zaXRlZC4KLSAgICBpZiAoVU5MSUtFTFkoIXZpc2libGVSZWdpb25TY3JlZW4uaXNFbXB0eSgpKSkgewotICAgICAgICAvLyBpZiBhbnl0aGluZyB2aXNpYmxlIGJlbG93IHVzIGlzIGludmFsaWRhdGVkLCB0aGUgY2FjaGUgYmVjb21lcyBkaXJ0eQotICAgICAgICBpZiAoIW1DYWNoZURpcnR5ICYmIAotICAgICAgICAgICAgICAgICF2aXNpYmxlUmVnaW9uU2NyZWVuLmludGVyc2VjdChvdXREaXJ0eVJlZ2lvbikuaXNFbXB0eSgpKSB7Ci0gICAgICAgICAgICBtQ2FjaGVEaXJ0eSA9IHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG1DYWNoZURpcnR5KSB7Ci0gICAgICAgICAgICBpZiAoIW1GbGluZ2VyLT5pc0Zyb3plbigpKSB7Ci0gICAgICAgICAgICAgICAgLy8gdXBkYXRlIGV2ZXJ5dGhpbmcgYmVsb3cgdXMgdGhhdCBpcyB2aXNpYmxlCi0gICAgICAgICAgICAgICAgb3V0RGlydHlSZWdpb24ub3JTZWxmKHZpc2libGVSZWdpb25TY3JlZW4pOwotICAgICAgICAgICAgICAgIG5zZWNzX3Qgbm93ID0gc3lzdGVtVGltZSgpOwotICAgICAgICAgICAgICAgIGlmICgobm93IC0gbUNhY2hlQWdlKSA+PSBtczJucyg1MDApKSB7Ci0gICAgICAgICAgICAgICAgICAgIG1DYWNoZUFnZSA9IG5vdzsKLSAgICAgICAgICAgICAgICAgICAgbVJlZnJlc2hDYWNoZSA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIG1DYWNoZURpcnR5ID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKCFtQXV0b1JlZnJlc2hQZW5kaW5nKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBtRmxpbmdlci0+c2lnbmFsRGVsYXllZEV2ZW50KG1zMm5zKDUwMCkpOwotICAgICAgICAgICAgICAgICAgICAgICAgbUF1dG9SZWZyZXNoUGVuZGluZyA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgTGF5ZXJCYXNlOjp1bmxvY2tQYWdlRmxpcChwbGFuZVRyYW5zZm9ybSwgb3V0RGlydHlSZWdpb24pOwotfQotCi12b2lkIExheWVyQmx1cjo6b25EcmF3KGNvbnN0IFJlZ2lvbiYgY2xpcCkgY29uc3QKLXsKLSAgICBjb25zdCBEaXNwbGF5SGFyZHdhcmUmIGh3KGdyYXBoaWNQbGFuZSgwKS5kaXNwbGF5SGFyZHdhcmUoKSk7Ci0gICAgY29uc3QgdWludDMyX3QgZmJIZWlnaHQgPSBody5nZXRIZWlnaHQoKTsKLSAgICBpbnQgeCA9IG1UcmFuc2Zvcm1lZEJvdW5kcy5sZWZ0OwotICAgIGludCB5ID0gbVRyYW5zZm9ybWVkQm91bmRzLnRvcDsKLSAgICBpbnQgdyA9IG1UcmFuc2Zvcm1lZEJvdW5kcy53aWR0aCgpOwotICAgIGludCBoID0gbVRyYW5zZm9ybWVkQm91bmRzLmhlaWdodCgpOwotICAgIEdMaW50IFggPSB4OwotICAgIEdMaW50IFkgPSBmYkhlaWdodCAtICh5ICsgaCk7Ci0gICAgaWYgKFggPCAwKSB7Ci0gICAgICAgIHcgKz0gWDsKLSAgICAgICAgWCA9IDA7Ci0gICAgfQotICAgIGlmIChZIDwgMCkgewotICAgICAgICBoICs9IFk7Ci0gICAgICAgIFkgPSAwOwotICAgIH0KLSAgICBpZiAodzwwIHx8IGg8MCkgewotICAgICAgICAvLyB3ZSdyZSBvdXRzaWRlIG9mIHRoZSBmcmFtZWJ1ZmZlcgotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgaWYgKG1UZXh0dXJlTmFtZSA9PSAtMVUpIHsKLSAgICAgICAgLy8gY3JlYXRlIHRoZSB0ZXh0dXJlIG5hbWUgdGhlIGZpcnN0IHRpbWUKLSAgICAgICAgLy8gY2FuJ3QgZG8gdGhhdCBpbiB0aGUgY3RvciwgYmVjYXVzZSBpdCBydW5zIGluIGFub3RoZXIgdGhyZWFkLgotICAgICAgICBnbEdlblRleHR1cmVzKDEsICZtVGV4dHVyZU5hbWUpOwotICAgIH0KLQotICAgIFJlZ2lvbjo6aXRlcmF0b3IgaXRlcmF0b3IoY2xpcCk7Ci0gICAgaWYgKGl0ZXJhdG9yKSB7Ci0gICAgICAgIGdsRW5hYmxlKEdMX1RFWFRVUkVfMkQpOwotICAgICAgICBnbEJpbmRUZXh0dXJlKEdMX1RFWFRVUkVfMkQsIG1UZXh0dXJlTmFtZSk7Ci0gICAgCi0gICAgICAgIGlmIChtUmVmcmVzaENhY2hlKSB7Ci0gICAgICAgICAgICBtUmVmcmVzaENhY2hlID0gZmFsc2U7Ci0gICAgICAgICAgICBtQXV0b1JlZnJlc2hQZW5kaW5nID0gZmFsc2U7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8vIGFsbG9jYXRlIGVub3VnaCBtZW1vcnkgZm9yIDQtYnl0ZXMgKDIgcGl4ZWxzKSBhbGlnbmVkIGRhdGEKLSAgICAgICAgICAgIGNvbnN0IGludDMyX3QgcyA9ICh3ICsgMSkgJiB+MTsKLSAgICAgICAgICAgIHVpbnQxNl90KiBjb25zdCBwaXhlbHMgPSAodWludDE2X3QqKW1hbGxvYyhzKmgqMik7Ci0KLSAgICAgICAgICAgIC8vIFRoaXMgcmVhZHMgdGhlIGZyYW1lLWJ1ZmZlciwgc28gYSBoL3cgR0wgd291bGQgaGF2ZSB0bwotICAgICAgICAgICAgLy8gZmluaXNoKCkgaXRzIHJlbmRlcmluZyBmaXJzdC4gd2UgZG9uJ3Qgd2FudCB0byBkbyB0aGF0Ci0gICAgICAgICAgICAvLyB0b28gb2Z0ZW4uIFJlYWQgZGF0YSBpcyA0LWJ5dGVzIGFsaWduZWQuCi0gICAgICAgICAgICBnbFJlYWRQaXhlbHMoWCwgWSwgdywgaCwgR0xfUkdCLCBHTF9VTlNJR05FRF9TSE9SVF81XzZfNSwgcGl4ZWxzKTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8gYmx1ciB0aGF0IHRleHR1cmUuCi0gICAgICAgICAgICBHR0xTdXJmYWNlIGJsOwotICAgICAgICAgICAgYmwudmVyc2lvbiA9IHNpemVvZihHR0xTdXJmYWNlKTsKLSAgICAgICAgICAgIGJsLndpZHRoID0gdzsKLSAgICAgICAgICAgIGJsLmhlaWdodCA9IGg7Ci0gICAgICAgICAgICBibC5zdHJpZGUgPSBzOwotICAgICAgICAgICAgYmwuZm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JfNTY1OwotICAgICAgICAgICAgYmwuZGF0YSA9IChHR0x1Ynl0ZSopcGl4ZWxzOyAgICAgICAgICAgIAotICAgICAgICAgICAgYmx1ckZpbHRlcigmYmwsIDgsIDIpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyBOT1RFOiB0aGlzIHdvcmtzIG9ubHkgYmVjYXVzZSB3ZSBoYXZlIFBPVC4gd2UnZCBoYXZlIHRvIHJvdW5kIHRoZQotICAgICAgICAgICAgLy8gdGV4dHVyZSBzaXplIHVwLCBvdGhlcndpc2UuCi0gICAgICAgICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwgR0xfUkdCLCB3LCBoLCAwLAotICAgICAgICAgICAgICAgICAgICBHTF9SR0IsIEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81LCBwaXhlbHMpOwotCi0gICAgICAgICAgICBmcmVlKCh2b2lkKilwaXhlbHMpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBjb25zdCBTdGF0ZSYgcyA9IGRyYXdpbmdTdGF0ZSgpOwotICAgICAgICBpZiAoVU5MSUtFTFkocy5hbHBoYSA8IDB4RkYpKSB7Ci0gICAgICAgICAgICBjb25zdCBHR0xmaXhlZCBhbHBoYSA9IChzLmFscGhhIDw8IDE2KS8yNTU7Ci0gICAgICAgICAgICBnbENvbG9yNHgoMCwgMCwgMCwgYWxwaGEpOwotICAgICAgICAgICAgZ2xFbmFibGUoR0xfQkxFTkQpOwotICAgICAgICAgICAgZ2xCbGVuZEZ1bmMoR0xfU1JDX0FMUEhBLCBHTF9PTkVfTUlOVVNfU1JDX0FMUEhBKTsKLSAgICAgICAgICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfUkVQTEFDRSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBnbERpc2FibGUoR0xfQkxFTkQpOwotICAgICAgICB9Ci0KLSAgICAgICAgZ2xEaXNhYmxlKEdMX0RJVEhFUik7Ci0gICAgICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfUkVQTEFDRSk7Ci0gICAgICAgIGdsVGV4UGFyYW1ldGVyeChHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX01BR19GSUxURVIsIEdMX05FQVJFU1QpOwotICAgICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NSU5fRklMVEVSLCBHTF9ORUFSRVNUKTsKLQotICAgICAgICBpZiAoVU5MSUtFTFkodHJhbnNmb3JtZWQoKQotICAgICAgICAgICAgICAgIHx8ICEobUZsYWdzICYgRGlzcGxheUhhcmR3YXJlOjpEUkFXX1RFWFRVUkVfRVhURU5TSU9OKSApKSB7Ci0gICAgICAgICAgICAvLyBUaGlzIGlzIGEgdmVyeSByYXJlIHNjZW5hcmlvLgotICAgICAgICAgICAgZ2xNYXRyaXhNb2RlKEdMX1RFWFRVUkUpOwotICAgICAgICAgICAgZ2xMb2FkSWRlbnRpdHkoKTsKLSAgICAgICAgICAgIGdsU2NhbGVmKDEuMGYvdywgLTEuMGYvaCwgMSk7Ci0gICAgICAgICAgICBnbFRyYW5zbGF0ZWYoLXgsIC15LCAwKTsKLSAgICAgICAgICAgIGdsRW5hYmxlQ2xpZW50U3RhdGUoR0xfVEVYVFVSRV9DT09SRF9BUlJBWSk7Ci0gICAgICAgICAgICBnbFZlcnRleFBvaW50ZXIoMiwgR0xfRklYRUQsIDAsIG1WZXJ0aWNlcyk7Ci0gICAgICAgICAgICBnbFRleENvb3JkUG9pbnRlcigyLCBHTF9GSVhFRCwgMCwgbVZlcnRpY2VzKTsKLSAgICAgICAgICAgIFJlY3QgcjsKLSAgICAgICAgICAgIHdoaWxlIChpdGVyYXRvci5pdGVyYXRlKCZyKSkgewotICAgICAgICAgICAgICAgIGNvbnN0IEdMaW50IHN5ID0gZmJIZWlnaHQgLSAoci50b3AgKyByLmhlaWdodCgpKTsKLSAgICAgICAgICAgICAgICBnbFNjaXNzb3Ioci5sZWZ0LCBzeSwgci53aWR0aCgpLCByLmhlaWdodCgpKTsKLSAgICAgICAgICAgICAgICBnbERyYXdBcnJheXMoR0xfVFJJQU5HTEVfRkFOLCAwLCA0KTsgCi0gICAgICAgICAgICB9ICAgICAgIAotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgUmVnaW9uOjppdGVyYXRvciBpdGVyYXRvcihjbGlwKTsKLSAgICAgICAgICAgIGlmIChpdGVyYXRvcikgewotICAgICAgICAgICAgICAgIC8vIE5PVEU6IHRoaXMgaXMgbWFyZ2luYWxseSBmYXN0ZXIgd2l0aCB0aGUgc29mdHdhcmUgZ2wsIGJlY2F1c2UKLSAgICAgICAgICAgICAgICAvLyBnbFJlYWRQaXhlbHMoKSByZWFkcyB0aGUgZmIgYm90dG9tLXRvLXRvcCwgaG93ZXZlciB3ZSdsbAotICAgICAgICAgICAgICAgIC8vIHNraXAgYWxsIHRoZSBqYWNjb2JpYW4gY29tcHV0YXRpb25zLgotICAgICAgICAgICAgICAgIFJlY3QgcjsKLSAgICAgICAgICAgICAgICBHTGludCBjcm9wWzRdID0geyAwLCAwLCB3LCBoIH07Ci0gICAgICAgICAgICAgICAgZ2xUZXhQYXJhbWV0ZXJpdihHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX0NST1BfUkVDVF9PRVMsIGNyb3ApOwotICAgICAgICAgICAgICAgIHkgPSBmYkhlaWdodCAtICh5ICsgaCk7Ci0gICAgICAgICAgICAgICAgd2hpbGUgKGl0ZXJhdG9yLml0ZXJhdGUoJnIpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGNvbnN0IEdMaW50IHN5ID0gZmJIZWlnaHQgLSAoci50b3AgKyByLmhlaWdodCgpKTsKLSAgICAgICAgICAgICAgICAgICAgZ2xTY2lzc29yKHIubGVmdCwgc3ksIHIud2lkdGgoKSwgci5oZWlnaHQoKSk7Ci0gICAgICAgICAgICAgICAgICAgIGdsRHJhd1RleGlPRVMoeCwgeSwgMCwgdywgaCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgZ2xEaXNhYmxlQ2xpZW50U3RhdGUoR0xfVEVYVFVSRV9DT09SRF9BUlJBWSk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllckJsdXIuaCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCbHVyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI0YjExNTYuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllckJsdXIuaAorKysgL2Rldi9udWxsCkBAIC0xLDY1ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfTEFZRVJfQkxVUl9ICi0jZGVmaW5lIEFORFJPSURfTEFZRVJfQkxVUl9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHByaXZhdGUvdWkvTGF5ZXJTdGF0ZS5oPgotCi0jaW5jbHVkZSA8dWkvUmVnaW9uLmg+Ci0KLSNpbmNsdWRlICJMYXllckJhc2UuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgTGF5ZXJCbHVyIDogcHVibGljIExheWVyQmFzZUNsaWVudAotewotcHVibGljOiAgICAKLSAgICBzdGF0aWMgY29uc3QgdWludDMyX3QgdHlwZUluZm87Ci0gICAgc3RhdGljIGNvbnN0IGNoYXIqIGNvbnN0IHR5cGVJRDsKLSAgICB2aXJ0dWFsIGNoYXIgY29uc3QqIGdldFR5cGVJRCgpIGNvbnN0IHsgcmV0dXJuIHR5cGVJRDsgfQotICAgIHZpcnR1YWwgdWludDMyX3QgZ2V0VHlwZUluZm8oKSBjb25zdCB7IHJldHVybiB0eXBlSW5mbzsgfQotICAgIAotICAgICAgICAgICAgICAgIExheWVyQmx1cihTdXJmYWNlRmxpbmdlciogZmxpbmdlciwgRGlzcGxheUlEIGRpc3BsYXksCi0gICAgICAgICAgICAgICAgICAgICAgICBDbGllbnQqIGNsaWVudCwgaW50MzJfdCBpKTsKLSAgICAgICAgdmlydHVhbCB+TGF5ZXJCbHVyKCk7Ci0KLSAgICB2aXJ0dWFsIHZvaWQgb25EcmF3KGNvbnN0IFJlZ2lvbiYgY2xpcCkgY29uc3Q7Ci0gICAgdmlydHVhbCBib29sIG5lZWRzQmxlbmRpbmcoKSBjb25zdCAgeyByZXR1cm4gdHJ1ZTsgfQotICAgIHZpcnR1YWwgYm9vbCBpc1NlY3VyZSgpIGNvbnN0ICAgICAgIHsgcmV0dXJuIGZhbHNlOyB9Ci0KLSAgICB2aXJ0dWFsIHVpbnQzMl90IGRvVHJhbnNhY3Rpb24odWludDMyX3QgZmxhZ3MpOwotICAgIHZpcnR1YWwgdm9pZCBzZXRWaXNpYmxlUmVnaW9uKGNvbnN0IFJlZ2lvbiYgdmlzaWJsZVJlZ2lvbik7Ci0gICAgdmlydHVhbCB2b2lkIHVubG9ja1BhZ2VGbGlwKGNvbnN0IFRyYW5zZm9ybSYgcGxhbmVUcmFuc2Zvcm0sIFJlZ2lvbiYgb3V0RGlydHlSZWdpb24pOwotCi1wcml2YXRlOgotICAgICAgICAgICAgYm9vbCAgICBtQ2FjaGVEaXJ0eTsKLSAgICBtdXRhYmxlIGJvb2wgICAgbVJlZnJlc2hDYWNoZTsKLSAgICBtdXRhYmxlIGJvb2wgICAgbUF1dG9SZWZyZXNoUGVuZGluZzsKLSAgICAgICAgICAgIG5zZWNzX3QgbUNhY2hlQWdlOwotICAgIG11dGFibGUgR0x1aW50ICBtVGV4dHVyZU5hbWU7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfTEFZRVJfQkxVUl9ICmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyQnVmZmVyLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCdWZmZXIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwMGZhYjcwLi4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCdWZmZXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNjU1ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIlN1cmZhY2VGbGluZ2VyIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8bWF0aC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RvcFdhdGNoLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL0lTZXJ2aWNlTWFuYWdlci5oPgotCi0jaW5jbHVkZSA8dWkvUGl4ZWxGb3JtYXQuaD4KLSNpbmNsdWRlIDx1aS9FR0xEaXNwbGF5U3VyZmFjZS5oPgotCi0jaW5jbHVkZSAiTGF5ZXJCdWZmZXIuaCIKLSNpbmNsdWRlICJTdXJmYWNlRmxpbmdlci5oIgotI2luY2x1ZGUgIlZSYW1IZWFwLmgiCi0jaW5jbHVkZSAiRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZS5oIgotCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNvbnN0IHVpbnQzMl90IExheWVyQnVmZmVyOjp0eXBlSW5mbyA9IExheWVyQmFzZUNsaWVudDo6dHlwZUluZm8gfCAweDIwOwotY29uc3QgY2hhciogY29uc3QgTGF5ZXJCdWZmZXI6OnR5cGVJRCA9ICJMYXllckJ1ZmZlciI7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1MYXllckJ1ZmZlcjo6TGF5ZXJCdWZmZXIoU3VyZmFjZUZsaW5nZXIqIGZsaW5nZXIsIERpc3BsYXlJRCBkaXNwbGF5LAotICAgICAgICBDbGllbnQqIGNsaWVudCwgaW50MzJfdCBpKQotICAgIDogTGF5ZXJCYXNlQ2xpZW50KGZsaW5nZXIsIGRpc3BsYXksIGNsaWVudCwgaSksCi0gICAgICBtTmVlZHNCbGVuZGluZyhmYWxzZSkKLXsKLX0KLQotTGF5ZXJCdWZmZXI6On5MYXllckJ1ZmZlcigpCi17Ci0gICAgc3A8U3VyZmFjZUJ1ZmZlcj4gcyhnZXRDbGllbnRTdXJmYWNlKCkpOwotICAgIGlmIChzICE9IDApIHsKLSAgICAgICAgcy0+ZGlzb3duKCk7Ci0gICAgICAgIG1DbGllbnRTdXJmYWNlLmNsZWFyKCk7Ci0gICAgfQotfQotCi1zcDxMYXllckJ1ZmZlcjo6U3VyZmFjZUJ1ZmZlcj4gTGF5ZXJCdWZmZXI6OmdldENsaWVudFN1cmZhY2UoKSBjb25zdAotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgcmV0dXJuIG1DbGllbnRTdXJmYWNlLnByb21vdGUoKTsKLX0KLQotc3A8TGF5ZXJCYXNlQ2xpZW50OjpTdXJmYWNlPiBMYXllckJ1ZmZlcjo6Z2V0U3VyZmFjZSgpIGNvbnN0Ci17Ci0gICAgc3A8U3VyZmFjZUJ1ZmZlcj4gczsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIHMgPSBtQ2xpZW50U3VyZmFjZS5wcm9tb3RlKCk7Ci0gICAgaWYgKHMgPT0gMCkgewotICAgICAgICBzID0gbmV3IFN1cmZhY2VCdWZmZXIoY2xpZW50SW5kZXgoKSwKLSAgICAgICAgICAgICAgICBjb25zdF9jYXN0PExheWVyQnVmZmVyICo+KHRoaXMpKTsKLSAgICAgICAgbUNsaWVudFN1cmZhY2UgPSBzOwotICAgIH0KLSAgICByZXR1cm4gczsKLX0KLQotYm9vbCBMYXllckJ1ZmZlcjo6bmVlZHNCbGVuZGluZygpIGNvbnN0IHsKLSAgICByZXR1cm4gbU5lZWRzQmxlbmRpbmc7Ci19Ci0KLXZvaWQgTGF5ZXJCdWZmZXI6OnNldE5lZWRzQmxlbmRpbmcoYm9vbCBibGVuZGluZykgewotICAgIG1OZWVkc0JsZW5kaW5nID0gYmxlbmRpbmc7Ci19Ci0KLXZvaWQgTGF5ZXJCdWZmZXI6OnBvc3RCdWZmZXIoc3NpemVfdCBvZmZzZXQpCi17Ci0gICAgc3A8U291cmNlPiBzb3VyY2UoZ2V0U291cmNlKCkpOwotICAgIGlmIChzb3VyY2UgIT0gMCkKLSAgICAgICAgc291cmNlLT5wb3N0QnVmZmVyKG9mZnNldCk7Ci19Ci0KLXZvaWQgTGF5ZXJCdWZmZXI6OnVucmVnaXN0ZXJCdWZmZXJzKCkKLXsKLSAgICBzcDxTb3VyY2U+IHNvdXJjZShjbGVhclNvdXJjZSgpKTsKLSAgICBpZiAoc291cmNlICE9IDApCi0gICAgICAgIHNvdXJjZS0+dW5yZWdpc3RlckJ1ZmZlcnMoKTsKLX0KLQotdWludDMyX3QgTGF5ZXJCdWZmZXI6OmRvVHJhbnNhY3Rpb24odWludDMyX3QgZmxhZ3MpCi17Ci0gICAgc3A8U291cmNlPiBzb3VyY2UoZ2V0U291cmNlKCkpOwotICAgIGlmIChzb3VyY2UgIT0gMCkKLSAgICAgICAgc291cmNlLT5vblRyYW5zYWN0aW9uKGZsYWdzKTsKLSAgICByZXR1cm4gTGF5ZXJCYXNlOjpkb1RyYW5zYWN0aW9uKGZsYWdzKTsgICAgCi19Ci0KLXZvaWQgTGF5ZXJCdWZmZXI6OnVubG9ja1BhZ2VGbGlwKGNvbnN0IFRyYW5zZm9ybSYgcGxhbmVUcmFuc2Zvcm0sCi0gICAgICAgIFJlZ2lvbiYgb3V0RGlydHlSZWdpb24pCi17Ci0gICAgLy8gdGhpcyBjb2RlLXBhdGggbXVzdCBiZSBhcyB0aWdodCBhcyBwb3NzaWJsZSwgaXQncyBjYWxsZWQgZWFjaCB0aW1lCi0gICAgLy8gdGhlIHNjcmVlbiBpcyBjb21wb3NpdGVkLgotICAgIHNwPFNvdXJjZT4gc291cmNlKGdldFNvdXJjZSgpKTsKLSAgICBpZiAoc291cmNlICE9IDApCi0gICAgICAgIHNvdXJjZS0+b25WaXNpYmlsaXR5UmVzb2x2ZWQocGxhbmVUcmFuc2Zvcm0pOwotICAgIExheWVyQmFzZTo6dW5sb2NrUGFnZUZsaXAocGxhbmVUcmFuc2Zvcm0sIG91dERpcnR5UmVnaW9uKTsgICAgCi19Ci0KLXZvaWQgTGF5ZXJCdWZmZXI6Om9uRHJhdyhjb25zdCBSZWdpb24mIGNsaXApIGNvbnN0Ci17Ci0gICAgc3A8U291cmNlPiBzb3VyY2UoZ2V0U291cmNlKCkpOwotICAgIGlmIChMSUtFTFkoc291cmNlICE9IDApKSB7Ci0gICAgICAgIHNvdXJjZS0+b25EcmF3KGNsaXApOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIGNsZWFyV2l0aE9wZW5HTChjbGlwKTsKLSAgICB9Ci19Ci0KLWJvb2wgTGF5ZXJCdWZmZXI6OnRyYW5zZm9ybWVkKCkgY29uc3QKLXsKLSAgICBzcDxTb3VyY2U+IHNvdXJjZShnZXRTb3VyY2UoKSk7Ci0gICAgaWYgKExJS0VMWShzb3VyY2UgIT0gMCkpCi0gICAgICAgIHJldHVybiBzb3VyY2UtPnRyYW5zZm9ybWVkKCk7Ci0gICAgcmV0dXJuIGZhbHNlOwotfQotCi0vKioKLSAqIFRoaXMgY3JlYXRlcyBhICJidWZmZXIiIHNvdXJjZSBmb3IgdGhpcyBzdXJmYWNlCi0gKi8KLXN0YXR1c190IExheWVyQnVmZmVyOjpyZWdpc3RlckJ1ZmZlcnMoY29uc3QgSVN1cmZhY2U6OkJ1ZmZlckhlYXAmIGJ1ZmZlcnMpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICBpZiAobVNvdXJjZSAhPSAwKQotICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0KLSAgICBzcDxCdWZmZXJTb3VyY2U+IHNvdXJjZSA9IG5ldyBCdWZmZXJTb3VyY2UoKnRoaXMsIGJ1ZmZlcnMpOwotCi0gICAgc3RhdHVzX3QgcmVzdWx0ID0gc291cmNlLT5nZXRTdGF0dXMoKTsKLSAgICBpZiAocmVzdWx0ID09IE5PX0VSUk9SKSB7Ci0gICAgICAgIG1Tb3VyY2UgPSBzb3VyY2U7Ci0gICAgfQotICAgIHJldHVybiByZXN1bHQ7Ci19ICAgIAotCi0vKioKLSAqIFRoaXMgY3JlYXRlcyBhbiAib3ZlcmxheSIgc291cmNlIGZvciB0aGlzIHN1cmZhY2UKLSAqLwotc3A8T3ZlcmxheVJlZj4gTGF5ZXJCdWZmZXI6OmNyZWF0ZU92ZXJsYXkodWludDMyX3QgdywgdWludDMyX3QgaCwgaW50MzJfdCBmKQotewotICAgIHNwPE92ZXJsYXlSZWY+IHJlc3VsdDsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIGlmIChtU291cmNlICE9IDApCi0gICAgICAgIHJldHVybiByZXN1bHQ7Ci0KLSAgICBzcDxPdmVybGF5U291cmNlPiBzb3VyY2UgPSBuZXcgT3ZlcmxheVNvdXJjZSgqdGhpcywgJnJlc3VsdCwgdywgaCwgZik7Ci0gICAgaWYgKHJlc3VsdCAhPSAwKSB7Ci0gICAgICAgIG1Tb3VyY2UgPSBzb3VyY2U7Ci0gICAgfQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLXNwPExheWVyQnVmZmVyOjpTb3VyY2U+IExheWVyQnVmZmVyOjpnZXRTb3VyY2UoKSBjb25zdCB7Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICByZXR1cm4gbVNvdXJjZTsKLX0KLQotc3A8TGF5ZXJCdWZmZXI6OlNvdXJjZT4gTGF5ZXJCdWZmZXI6OmNsZWFyU291cmNlKCkgewotICAgIHNwPFNvdXJjZT4gc291cmNlOwotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgc291cmNlID0gbVNvdXJjZTsKLSAgICBtU291cmNlLmNsZWFyKCk7Ci0gICAgcmV0dXJuIHNvdXJjZTsKLX0KLQotLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotLy8gTGF5ZXJCdWZmZXI6OlN1cmZhY2VCdWZmZXIKLS8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLQotTGF5ZXJCdWZmZXI6OlN1cmZhY2VCdWZmZXI6OlN1cmZhY2VCdWZmZXIoU3VyZmFjZUlEIGlkLCBMYXllckJ1ZmZlciogb3duZXIpCi06IExheWVyQmFzZUNsaWVudDo6U3VyZmFjZShpZCwgb3duZXItPmdldElkZW50aXR5KCkpLCBtT3duZXIob3duZXIpCi17Ci19Ci0KLUxheWVyQnVmZmVyOjpTdXJmYWNlQnVmZmVyOjp+U3VyZmFjZUJ1ZmZlcigpCi17Ci0gICAgdW5yZWdpc3RlckJ1ZmZlcnMoKTsKLSAgICBtT3duZXIgPSAwOwotfQotCi1zdGF0dXNfdCBMYXllckJ1ZmZlcjo6U3VyZmFjZUJ1ZmZlcjo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIHN3aXRjaCAoY29kZSkgewotICAgICAgICBjYXNlIFJFR0lTVEVSX0JVRkZFUlM6Ci0gICAgICAgIGNhc2UgVU5SRUdJU1RFUl9CVUZGRVJTOgotICAgICAgICBjYXNlIENSRUFURV9PVkVSTEFZOgotICAgICAgICB7Ci0gICAgICAgICAgICAvLyBjb2RlcyB0aGF0IHJlcXVpcmUgcGVybWlzc2lvbiBjaGVjawotICAgICAgICAgICAgSVBDVGhyZWFkU3RhdGUqIGlwYyA9IElQQ1RocmVhZFN0YXRlOjpzZWxmKCk7Ci0gICAgICAgICAgICBjb25zdCBpbnQgcGlkID0gaXBjLT5nZXRDYWxsaW5nUGlkKCk7Ci0gICAgICAgICAgICBjb25zdCBpbnQgc2VsZl9waWQgPSBnZXRwaWQoKTsKLSAgICAgICAgICAgIGlmIChMSUtFTFkocGlkICE9IHNlbGZfcGlkKSkgewotICAgICAgICAgICAgICAgIC8vIHdlJ3JlIGNhbGxlZCBmcm9tIGEgZGlmZmVyZW50IHByb2Nlc3MsIGRvIHRoZSByZWFsIGNoZWNrCi0gICAgICAgICAgICAgICAgaWYgKCFjaGVja0NhbGxpbmdQZXJtaXNzaW9uKAotICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nMTYoImFuZHJvaWQucGVybWlzc2lvbi5BQ0NFU1NfU1VSRkFDRV9GTElOR0VSIikpKQotICAgICAgICAgICAgICAgIHsKLSAgICAgICAgICAgICAgICAgICAgY29uc3QgaW50IHVpZCA9IGlwYy0+Z2V0Q2FsbGluZ1VpZCgpOwotICAgICAgICAgICAgICAgICAgICBMT0dFKCJQZXJtaXNzaW9uIERlbmlhbDogIgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICJjYW4ndCBhY2Nlc3MgU3VyZmFjZUZsaW5nZXIgcGlkPSVkLCB1aWQ9JWQiLCBwaWQsIHVpZCk7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBQRVJNSVNTSU9OX0RFTklFRDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIExheWVyQmFzZUNsaWVudDo6U3VyZmFjZTo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotfQotCi1zdGF0dXNfdCBMYXllckJ1ZmZlcjo6U3VyZmFjZUJ1ZmZlcjo6cmVnaXN0ZXJCdWZmZXJzKGNvbnN0IElTdXJmYWNlOjpCdWZmZXJIZWFwJiBidWZmZXJzKQotewotICAgIExheWVyQnVmZmVyKiBvd25lcihnZXRPd25lcigpKTsKLSAgICBpZiAob3duZXIpCi0gICAgICAgIHJldHVybiBvd25lci0+cmVnaXN0ZXJCdWZmZXJzKGJ1ZmZlcnMpOwotICAgIHJldHVybiBOT19JTklUOwotfQotCi12b2lkIExheWVyQnVmZmVyOjpTdXJmYWNlQnVmZmVyOjpwb3N0QnVmZmVyKHNzaXplX3Qgb2Zmc2V0KQotewotICAgIExheWVyQnVmZmVyKiBvd25lcihnZXRPd25lcigpKTsKLSAgICBpZiAob3duZXIpCi0gICAgICAgIG93bmVyLT5wb3N0QnVmZmVyKG9mZnNldCk7Ci19Ci0KLXZvaWQgTGF5ZXJCdWZmZXI6OlN1cmZhY2VCdWZmZXI6OnVucmVnaXN0ZXJCdWZmZXJzKCkKLXsKLSAgICBMYXllckJ1ZmZlciogb3duZXIoZ2V0T3duZXIoKSk7Ci0gICAgaWYgKG93bmVyKQotICAgICAgICBvd25lci0+dW5yZWdpc3RlckJ1ZmZlcnMoKTsKLX0KLQotc3A8T3ZlcmxheVJlZj4gTGF5ZXJCdWZmZXI6OlN1cmZhY2VCdWZmZXI6OmNyZWF0ZU92ZXJsYXkoCi0gICAgICAgIHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIGludDMyX3QgZm9ybWF0KSB7Ci0gICAgc3A8T3ZlcmxheVJlZj4gcmVzdWx0OwotICAgIExheWVyQnVmZmVyKiBvd25lcihnZXRPd25lcigpKTsKLSAgICBpZiAob3duZXIpCi0gICAgICAgIHJlc3VsdCA9IG93bmVyLT5jcmVhdGVPdmVybGF5KHcsIGgsIGZvcm1hdCk7Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotdm9pZCBMYXllckJ1ZmZlcjo6U3VyZmFjZUJ1ZmZlcjo6ZGlzb3duKCkKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIG1Pd25lciA9IDA7Ci19Ci0KLS8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLS8vIExheWVyQnVmZmVyOjpCdWZmZXIKLS8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLQotTGF5ZXJCdWZmZXI6OkJ1ZmZlcjo6QnVmZmVyKGNvbnN0IElTdXJmYWNlOjpCdWZmZXJIZWFwJiBidWZmZXJzLCBzc2l6ZV90IG9mZnNldCkKLSAgICA6IG1CdWZmZXJIZWFwKGJ1ZmZlcnMpCi17Ci0gICAgTmF0aXZlQnVmZmVyJiBzcmMobU5hdGl2ZUJ1ZmZlcik7Ci0gICAgc3JjLmNyb3AubCA9IDA7Ci0gICAgc3JjLmNyb3AudCA9IDA7Ci0gICAgc3JjLmNyb3AuciA9IGJ1ZmZlcnMudzsKLSAgICBzcmMuY3JvcC5iID0gYnVmZmVycy5oOwotICAgIHNyYy5pbWcudyA9IGJ1ZmZlcnMuaG9yX3N0cmlkZSA/OiBidWZmZXJzLnc7Ci0gICAgc3JjLmltZy5oID0gYnVmZmVycy52ZXJfc3RyaWRlID86IGJ1ZmZlcnMuaDsKLSAgICBzcmMuaW1nLmZvcm1hdCA9IGJ1ZmZlcnMuZm9ybWF0OwotICAgIHNyYy5pbWcub2Zmc2V0ID0gb2Zmc2V0OwotICAgIHNyYy5pbWcuYmFzZSAgID0gYnVmZmVycy5oZWFwLT5iYXNlKCk7Ci0gICAgc3JjLmltZy5mZCAgICAgPSBidWZmZXJzLmhlYXAtPmhlYXBJRCgpOwotfQotCi1MYXllckJ1ZmZlcjo6QnVmZmVyOjp+QnVmZmVyKCkKLXsKLX0KLQotLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotLy8gTGF5ZXJCdWZmZXI6OlNvdXJjZQotLy8gTGF5ZXJCdWZmZXI6OkJ1ZmZlclNvdXJjZQotLy8gTGF5ZXJCdWZmZXI6Ok92ZXJsYXlTb3VyY2UKLS8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLQotTGF5ZXJCdWZmZXI6OlNvdXJjZTo6U291cmNlKExheWVyQnVmZmVyJiBsYXllcikKLSAgICA6IG1MYXllcihsYXllcikKLXsgICAgCi19Ci1MYXllckJ1ZmZlcjo6U291cmNlOjp+U291cmNlKCkgeyAgICAKLX0KLXZvaWQgTGF5ZXJCdWZmZXI6OlNvdXJjZTo6b25EcmF3KGNvbnN0IFJlZ2lvbiYgY2xpcCkgY29uc3QgewotfQotdm9pZCBMYXllckJ1ZmZlcjo6U291cmNlOjpvblRyYW5zYWN0aW9uKHVpbnQzMl90IGZsYWdzKSB7Ci19Ci12b2lkIExheWVyQnVmZmVyOjpTb3VyY2U6Om9uVmlzaWJpbGl0eVJlc29sdmVkKAotICAgICAgICBjb25zdCBUcmFuc2Zvcm0mIHBsYW5lVHJhbnNmb3JtKSB7Ci19Ci12b2lkIExheWVyQnVmZmVyOjpTb3VyY2U6OnBvc3RCdWZmZXIoc3NpemVfdCBvZmZzZXQpIHsKLX0KLXZvaWQgTGF5ZXJCdWZmZXI6OlNvdXJjZTo6dW5yZWdpc3RlckJ1ZmZlcnMoKSB7Ci19Ci1ib29sIExheWVyQnVmZmVyOjpTb3VyY2U6OnRyYW5zZm9ybWVkKCkgY29uc3QgewotICAgIHJldHVybiBtTGF5ZXIubVRyYW5zZm9ybWVkOyAKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUxheWVyQnVmZmVyOjpCdWZmZXJTb3VyY2U6OkJ1ZmZlclNvdXJjZShMYXllckJ1ZmZlciYgbGF5ZXIsCi0gICAgICAgIGNvbnN0IElTdXJmYWNlOjpCdWZmZXJIZWFwJiBidWZmZXJzKQotICAgIDogU291cmNlKGxheWVyKSwgbVN0YXR1cyhOT19FUlJPUiksIAotICAgICAgbUJ1ZmZlclNpemUoMCksIG1UZXh0dXJlTmFtZSgtMVUpCi17Ci0gICAgaWYgKGJ1ZmZlcnMuaGVhcCA9PSBOVUxMKSB7Ci0gICAgICAgIC8vIHRoaXMgaXMgYWxsb3dlZCwgYnV0IGluIHRoaXMgY2FzZSwgaXQgaXMgaWxsZWdhbCB0byByZWNlaXZlCi0gICAgICAgIC8vIHBvc3RCdWZmZXIoKS4gVGhlIHN1cmZhY2UganVzdCBlcmFzZXMgdGhlIGZyYW1lYnVmZmVyIHdpdGgKLSAgICAgICAgLy8gZnVsbHkgdHJhbnNwYXJlbnQgcGl4ZWxzLgotICAgICAgICBtQnVmZmVySGVhcCA9IGJ1ZmZlcnM7Ci0gICAgICAgIG1MYXllci5zZXROZWVkc0JsZW5kaW5nKGZhbHNlKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIHN0YXR1c190IGVyciA9IChidWZmZXJzLmhlYXAtPmhlYXBJRCgpID49IDApID8gTk9fRVJST1IgOiBOT19JTklUOwotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgTE9HRSgiTGF5ZXJCdWZmZXI6OkJ1ZmZlclNvdXJjZTogaW52YWxpZCBoZWFwICglcykiLCBzdHJlcnJvcihlcnIpKTsKLSAgICAgICAgbVN0YXR1cyA9IGVycjsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICAKLSAgICBQaXhlbEZvcm1hdEluZm8gaW5mbzsKLSAgICBlcnIgPSBnZXRQaXhlbEZvcm1hdEluZm8oYnVmZmVycy5mb3JtYXQsICZpbmZvKTsKLSAgICBpZiAoZXJyICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgIExPR0UoIkxheWVyQnVmZmVyOjpCdWZmZXJTb3VyY2U6IGludmFsaWQgZm9ybWF0ICVkICglcykiLAotICAgICAgICAgICAgICAgIGJ1ZmZlcnMuZm9ybWF0LCBzdHJlcnJvcihlcnIpKTsKLSAgICAgICAgbVN0YXR1cyA9IGVycjsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIGlmIChidWZmZXJzLmhvcl9zdHJpZGU8MCB8fCBidWZmZXJzLnZlcl9zdHJpZGU8MCkgewotICAgICAgICBMT0dFKCJMYXllckJ1ZmZlcjo6QnVmZmVyU291cmNlOiBpbnZhbGlkIHBhcmFtZXRlcnMgIgotICAgICAgICAgICAgICIodz0lZCwgaD0lZCwgeHM9JWQsIHlzPSVkKSIsIAotICAgICAgICAgICAgIGJ1ZmZlcnMudywgYnVmZmVycy5oLCBidWZmZXJzLmhvcl9zdHJpZGUsIGJ1ZmZlcnMudmVyX3N0cmlkZSk7Ci0gICAgICAgIG1TdGF0dXMgPSBCQURfVkFMVUU7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBtQnVmZmVySGVhcCA9IGJ1ZmZlcnM7Ci0gICAgbUxheWVyLnNldE5lZWRzQmxlbmRpbmcoKGluZm8uaF9hbHBoYSAtIGluZm8ubF9hbHBoYSkgPiAwKTsgICAgCi0gICAgbUJ1ZmZlclNpemUgPSBpbmZvLmdldFNjYW5saW5lU2l6ZShidWZmZXJzLmhvcl9zdHJpZGUpKmJ1ZmZlcnMudmVyX3N0cmlkZTsKLSAgICBtTGF5ZXIuZm9yY2VWaXNpYmlsaXR5VHJhbnNhY3Rpb24oKTsKLSAgICAKLX0KLQotTGF5ZXJCdWZmZXI6OkJ1ZmZlclNvdXJjZTo6fkJ1ZmZlclNvdXJjZSgpCi17ICAgIAotICAgIGlmIChtVGV4dHVyZU5hbWUgIT0gLTFVKSB7Ci0gICAgICAgIExheWVyQmFzZTo6ZGVsZXRlZFRleHR1cmVzLmFkZChtVGV4dHVyZU5hbWUpOwotICAgIH0KLX0KLQotdm9pZCBMYXllckJ1ZmZlcjo6QnVmZmVyU291cmNlOjpwb3N0QnVmZmVyKHNzaXplX3Qgb2Zmc2V0KQoteyAgICAKLSAgICBJU3VyZmFjZTo6QnVmZmVySGVhcCBidWZmZXJzOwotICAgIHsgLy8gc2NvcGUgZm9yIHRoZSBsb2NrCi0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgICAgIGJ1ZmZlcnMgPSBtQnVmZmVySGVhcDsKLSAgICAgICAgaWYgKGJ1ZmZlcnMuaGVhcCAhPSAwKSB7Ci0gICAgICAgICAgICBjb25zdCBzaXplX3QgbWVtb3J5U2l6ZSA9IGJ1ZmZlcnMuaGVhcC0+Z2V0U2l6ZSgpOwotICAgICAgICAgICAgaWYgKChzaXplX3Qob2Zmc2V0KSArIG1CdWZmZXJTaXplKSA+IG1lbW9yeVNpemUpIHsKLSAgICAgICAgICAgICAgICBMT0dFKCJMYXllckJ1ZmZlcjo6QnVmZmVyU291cmNlOjpwb3N0QnVmZmVyKCkgIgotICAgICAgICAgICAgICAgICAgICAgImludmFsaWQgYnVmZmVyIChvZmZzZXQ9JWQsIHNpemU9JWQsIGhlYXAtc2l6ZT0lZCIsCi0gICAgICAgICAgICAgICAgICAgICBpbnQob2Zmc2V0KSwgaW50KG1CdWZmZXJTaXplKSwgaW50KG1lbW9yeVNpemUpKTsKLSAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBzcDxCdWZmZXI+IGJ1ZmZlcjsKLSAgICBpZiAoYnVmZmVycy5oZWFwICE9IDApIHsKLSAgICAgICAgYnVmZmVyID0gbmV3IExheWVyQnVmZmVyOjpCdWZmZXIoYnVmZmVycywgb2Zmc2V0KTsKLSAgICAgICAgaWYgKGJ1ZmZlci0+Z2V0U3RhdHVzKCkgIT0gTk9fRVJST1IpCi0gICAgICAgICAgICBidWZmZXIuY2xlYXIoKTsKLSAgICAgICAgc2V0QnVmZmVyKGJ1ZmZlcik7Ci0gICAgICAgIG1MYXllci5pbnZhbGlkYXRlKCk7Ci0gICAgfQotfQotCi12b2lkIExheWVyQnVmZmVyOjpCdWZmZXJTb3VyY2U6OnVucmVnaXN0ZXJCdWZmZXJzKCkKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIG1CdWZmZXJIZWFwLmhlYXAuY2xlYXIoKTsKLSAgICBtQnVmZmVyLmNsZWFyKCk7Ci0gICAgbUxheWVyLmludmFsaWRhdGUoKTsKLX0KLQotc3A8TGF5ZXJCdWZmZXI6OkJ1ZmZlcj4gTGF5ZXJCdWZmZXI6OkJ1ZmZlclNvdXJjZTo6Z2V0QnVmZmVyKCkgY29uc3QKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIHJldHVybiBtQnVmZmVyOwotfQotCi12b2lkIExheWVyQnVmZmVyOjpCdWZmZXJTb3VyY2U6OnNldEJ1ZmZlcihjb25zdCBzcDxMYXllckJ1ZmZlcjo6QnVmZmVyPiYgYnVmZmVyKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgbUJ1ZmZlciA9IGJ1ZmZlcjsKLX0KLQotYm9vbCBMYXllckJ1ZmZlcjo6QnVmZmVyU291cmNlOjp0cmFuc2Zvcm1lZCgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1CdWZmZXJIZWFwLnRyYW5zZm9ybSA/IHRydWUgOiBTb3VyY2U6OnRyYW5zZm9ybWVkKCk7IAotfQotCi12b2lkIExheWVyQnVmZmVyOjpCdWZmZXJTb3VyY2U6Om9uRHJhdyhjb25zdCBSZWdpb24mIGNsaXApIGNvbnN0IAotewotICAgIHNwPEJ1ZmZlcj4gYnVmZmVyKGdldEJ1ZmZlcigpKTsKLSAgICBpZiAoVU5MSUtFTFkoYnVmZmVyID09IDApKSAgewotICAgICAgICAvLyBub3RoaW5nIHRvIGRvLCB3ZSBkb24ndCBoYXZlIGEgYnVmZmVyCi0gICAgICAgIG1MYXllci5jbGVhcldpdGhPcGVuR0woY2xpcCk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBzdGF0dXNfdCBlcnIgPSBOT19FUlJPUjsKLSAgICBOYXRpdmVCdWZmZXIgc3JjKGJ1ZmZlci0+Z2V0QnVmZmVyKCkpOwotICAgIGNvbnN0IFJlY3QmIHRyYW5zZm9ybWVkQm91bmRzID0gbUxheWVyLmdldFRyYW5zZm9ybWVkQm91bmRzKCk7Ci0gICAgY29uc3QgaW50IGNhbl91c2VfY29weWJpdCA9IG1MYXllci5jYW5Vc2VDb3B5Yml0KCk7Ci0KLSAgICBpZiAoY2FuX3VzZV9jb3B5Yml0KSAgewotICAgICAgICBjb25zdCBpbnQgc3JjX3dpZHRoICA9IHNyYy5jcm9wLnIgLSBzcmMuY3JvcC5sOwotICAgICAgICBjb25zdCBpbnQgc3JjX2hlaWdodCA9IHNyYy5jcm9wLmIgLSBzcmMuY3JvcC50OwotICAgICAgICBpbnQgVyA9IHRyYW5zZm9ybWVkQm91bmRzLndpZHRoKCk7Ci0gICAgICAgIGludCBIID0gdHJhbnNmb3JtZWRCb3VuZHMuaGVpZ2h0KCk7Ci0gICAgICAgIGlmIChtTGF5ZXIuZ2V0T3JpZW50YXRpb24oKSAmIFRyYW5zZm9ybTo6Uk9UXzkwKSB7Ci0gICAgICAgICAgICBpbnQgdChXKTsgVz1IOyBIPXQ7Ci0gICAgICAgIH0KLQotICAgICAgICAvKiBXaXRoIExheWVyQnVmZmVyLCBpdCBpcyBsaWtlbHkgdGhhdCB3ZSdsbCBoYXZlIHRvIHJlc2NhbGUgdGhlCi0gICAgICAgICAqIHN1cmZhY2UsIGJlY2F1c2UgdGhpcyBpcyBvZnRlbiB1c2VkIGZvciB2aWRlbyBwbGF5YmFjayBvcgotICAgICAgICAgKiBjYW1lcmEtcHJldmlldy4gU2luY2Ugd2Ugd2FudCB0aGVzZSBvcGVyYXRpb24gYXMgZmFzdCBhcyBwb3NzaWJsZQotICAgICAgICAgKiB3ZSBtYWtlIHN1cmUgd2UgY2FuIHVzZSB0aGUgMkQgSC9XIGV2ZW4gaWYgaXQgZG9lc24ndCBzdXBwb3J0Ci0gICAgICAgICAqIHRoZSByZXF1ZXN0ZWQgc2NhbGUgZmFjdG9yLCBpbiB3aGljaCBjYXNlIHdlIHBlcmZvcm0gdGhlIHNjYWxpbmcKLSAgICAgICAgICogaW4gc2V2ZXJhbCBwYXNzZXMuICovCi0KLSAgICAgICAgY29weWJpdF9kZXZpY2VfdCogY29weWJpdCA9IG1MYXllci5tRmxpbmdlci0+Z2V0QmxpdEVuZ2luZSgpOwotICAgICAgICBjb25zdCBmbG9hdCBtaW4gPSBjb3B5Yml0LT5nZXQoY29weWJpdCwgQ09QWUJJVF9NSU5JRklDQVRJT05fTElNSVQpOwotICAgICAgICBjb25zdCBmbG9hdCBtYWcgPSBjb3B5Yml0LT5nZXQoY29weWJpdCwgQ09QWUJJVF9NQUdOSUZJQ0FUSU9OX0xJTUlUKTsKLQotICAgICAgICBmbG9hdCB4c2NhbGUgPSAxLjBmOwotICAgICAgICBpZiAoc3JjX3dpZHRoID4gVyptaW4pICAgICAgICAgIHhzY2FsZSA9IDEuMGYgLyBtaW47Ci0gICAgICAgIGVsc2UgaWYgKHNyY193aWR0aCptYWcgPCBXKSAgICAgeHNjYWxlID0gbWFnOwotCi0gICAgICAgIGZsb2F0IHlzY2FsZSA9IDEuMGY7Ci0gICAgICAgIGlmIChzcmNfaGVpZ2h0ID4gSCptaW4pICAgICAgICAgeXNjYWxlID0gMS4wZiAvIG1pbjsKLSAgICAgICAgZWxzZSBpZiAoc3JjX2hlaWdodCptYWcgPCBIKSAgICB5c2NhbGUgPSBtYWc7Ci0KLSAgICAgICAgaWYgKFVOTElLRUxZKHhzY2FsZSE9MS4wZiB8fCB5c2NhbGUhPTEuMGYpKSB7Ci0gICAgICAgICAgICBpZiAoVU5MSUtFTFkobVRlbXBvcmFyeURlYWxlciA9PSAwKSkgewotICAgICAgICAgICAgICAgIC8vIGFsbG9jYXRlIGEgbWVtb3J5LWRlYWxlciBmb3IgdGhpcyB0aGUgZmlyc3QgdGltZQotICAgICAgICAgICAgICAgIG1UZW1wb3JhcnlEZWFsZXIgPSBtTGF5ZXIubUZsaW5nZXItPmdldFN1cmZhY2VIZWFwTWFuYWdlcigpCi0gICAgICAgICAgICAgICAgICAgIC0+Y3JlYXRlSGVhcChJU3VyZmFjZUNvbXBvc2VyOjplSGFyZHdhcmUpOwotICAgICAgICAgICAgICAgIG1UZW1wQml0bWFwLmluaXQobVRlbXBvcmFyeURlYWxlcik7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGNvbnN0IGludCB0bXBfdyA9IGZsb29yZihzcmNfd2lkdGggICogeHNjYWxlKTsKLSAgICAgICAgICAgIGNvbnN0IGludCB0bXBfaCA9IGZsb29yZihzcmNfaGVpZ2h0ICogeXNjYWxlKTsKLSAgICAgICAgICAgIGVyciA9IG1UZW1wQml0bWFwLnNldEJpdHModG1wX3csIHRtcF9oLCAxLCBzcmMuaW1nLmZvcm1hdCk7Ci0KLSAgICAgICAgICAgIGlmIChMSUtFTFkoZXJyID09IE5PX0VSUk9SKSkgewotICAgICAgICAgICAgICAgIE5hdGl2ZUJ1ZmZlciB0bXA7Ci0gICAgICAgICAgICAgICAgbVRlbXBCaXRtYXAuZ2V0Qml0bWFwU3VyZmFjZSgmdG1wLmltZyk7Ci0gICAgICAgICAgICAgICAgdG1wLmNyb3AubCA9IDA7Ci0gICAgICAgICAgICAgICAgdG1wLmNyb3AudCA9IDA7Ci0gICAgICAgICAgICAgICAgdG1wLmNyb3AuciA9IHRtcC5pbWcudzsKLSAgICAgICAgICAgICAgICB0bXAuY3JvcC5iID0gdG1wLmltZy5oOwotCi0gICAgICAgICAgICAgICAgcmVnaW9uX2l0ZXJhdG9yIHRtcF9pdChSZWdpb24oUmVjdCh0bXAuY3JvcC5yLCB0bXAuY3JvcC5iKSkpOwotICAgICAgICAgICAgICAgIGNvcHliaXQtPnNldF9wYXJhbWV0ZXIoY29weWJpdCwgQ09QWUJJVF9UUkFOU0ZPUk0sIDApOwotICAgICAgICAgICAgICAgIGNvcHliaXQtPnNldF9wYXJhbWV0ZXIoY29weWJpdCwgQ09QWUJJVF9QTEFORV9BTFBIQSwgMHhGRik7Ci0gICAgICAgICAgICAgICAgY29weWJpdC0+c2V0X3BhcmFtZXRlcihjb3B5Yml0LCBDT1BZQklUX0RJVEhFUiwgQ09QWUJJVF9ESVNBQkxFKTsKLSAgICAgICAgICAgICAgICBlcnIgPSBjb3B5Yml0LT5zdHJldGNoKGNvcHliaXQsCi0gICAgICAgICAgICAgICAgICAgICAgICAmdG1wLmltZywgJnNyYy5pbWcsICZ0bXAuY3JvcCwgJnNyYy5jcm9wLCAmdG1wX2l0KTsKLSAgICAgICAgICAgICAgICBzcmMgPSB0bXA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBjb25zdCBEaXNwbGF5SGFyZHdhcmUmIGh3KG1MYXllci5ncmFwaGljUGxhbmUoMCkuZGlzcGxheUhhcmR3YXJlKCkpOwotICAgICAgICBjb3B5Yml0X2ltYWdlX3QgZHN0OwotICAgICAgICBody5nZXREaXNwbGF5U3VyZmFjZSgmZHN0KTsKLSAgICAgICAgY29uc3QgY29weWJpdF9yZWN0X3QmIGRyZWN0Ci0gICAgICAgICAgICA9IHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgY29weWJpdF9yZWN0X3QmPih0cmFuc2Zvcm1lZEJvdW5kcyk7Ci0gICAgICAgIGNvbnN0IFN0YXRlJiBzKG1MYXllci5kcmF3aW5nU3RhdGUoKSk7Ci0gICAgICAgIHJlZ2lvbl9pdGVyYXRvciBpdChjbGlwKTsKLSAgICAgICAgCi0gICAgICAgIC8vIHBpY2sgdGhlIHJpZ2h0IG9yaWVudGF0aW9uIGZvciB0aGlzIGJ1ZmZlcgotICAgICAgICBpbnQgb3JpZW50YXRpb24gPSBtTGF5ZXIuZ2V0T3JpZW50YXRpb24oKTsKLSAgICAgICAgaWYgKFVOTElLRUxZKG1CdWZmZXJIZWFwLnRyYW5zZm9ybSkpIHsKLSAgICAgICAgICAgIFRyYW5zZm9ybSByb3Q5MDsKLSAgICAgICAgICAgIEdyYXBoaWNQbGFuZTo6b3JpZW50YXRpb25Ub1RyYW5zZnJvbSgKLSAgICAgICAgICAgICAgICAgICAgSVN1cmZhY2VDb21wb3Nlcjo6ZU9yaWVudGF0aW9uOTAsIDAsIDAsICZyb3Q5MCk7Ci0gICAgICAgICAgICBjb25zdCBUcmFuc2Zvcm0mIHBsYW5lVHJhbnNmb3JtKG1MYXllci5ncmFwaGljUGxhbmUoMCkudHJhbnNmb3JtKCkpOwotICAgICAgICAgICAgY29uc3QgTGF5ZXI6OlN0YXRlJiBzKG1MYXllci5kcmF3aW5nU3RhdGUoKSk7Ci0gICAgICAgICAgICBUcmFuc2Zvcm0gdHIocGxhbmVUcmFuc2Zvcm0gKiBzLnRyYW5zZm9ybSAqIHJvdDkwKTsKLSAgICAgICAgICAgIG9yaWVudGF0aW9uID0gdHIuZ2V0T3JpZW50YXRpb24oKTsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgY29weWJpdC0+c2V0X3BhcmFtZXRlcihjb3B5Yml0LCBDT1BZQklUX1RSQU5TRk9STSwgb3JpZW50YXRpb24pOwotICAgICAgICBjb3B5Yml0LT5zZXRfcGFyYW1ldGVyKGNvcHliaXQsIENPUFlCSVRfUExBTkVfQUxQSEEsIHMuYWxwaGEpOwotICAgICAgICBjb3B5Yml0LT5zZXRfcGFyYW1ldGVyKGNvcHliaXQsIENPUFlCSVRfRElUSEVSLCBDT1BZQklUX0VOQUJMRSk7Ci0KLSAgICAgICAgZXJyID0gY29weWJpdC0+c3RyZXRjaChjb3B5Yml0LAotICAgICAgICAgICAgICAgICZkc3QsICZzcmMuaW1nLCAmZHJlY3QsICZzcmMuY3JvcCwgJml0KTsKLSAgICAgICAgaWYgKGVyciAhPSBOT19FUlJPUikgewotICAgICAgICAgICAgTE9HRSgiY29weWJpdCBmYWlsZWQgKCVzKSIsIHN0cmVycm9yKGVycikpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgaWYgKCFjYW5fdXNlX2NvcHliaXQgfHwgZXJyKSB7Ci0gICAgICAgIGlmIChVTkxJS0VMWShtVGV4dHVyZU5hbWUgPT0gLTFMVSkpIHsKLSAgICAgICAgICAgIG1UZXh0dXJlTmFtZSA9IG1MYXllci5jcmVhdGVUZXh0dXJlKCk7Ci0gICAgICAgIH0KLSAgICAgICAgR0x1aW50IHcgPSAwOwotICAgICAgICBHTHVpbnQgaCA9IDA7Ci0gICAgICAgIEdHTFN1cmZhY2UgdDsKLSAgICAgICAgdC52ZXJzaW9uID0gc2l6ZW9mKEdHTFN1cmZhY2UpOwotICAgICAgICB0LndpZHRoICA9IHNyYy5jcm9wLnI7Ci0gICAgICAgIHQuaGVpZ2h0ID0gc3JjLmNyb3AuYjsKLSAgICAgICAgdC5zdHJpZGUgPSBzcmMuaW1nLnc7Ci0gICAgICAgIHQudnN0cmlkZT0gc3JjLmltZy5oOwotICAgICAgICB0LmZvcm1hdCA9IHNyYy5pbWcuZm9ybWF0OwotICAgICAgICB0LmRhdGEgPSAoR0dMdWJ5dGUqKShpbnRwdHJfdChzcmMuaW1nLmJhc2UpICsgc3JjLmltZy5vZmZzZXQpOwotICAgICAgICBjb25zdCBSZWdpb24gZGlydHkoUmVjdCh0LndpZHRoLCB0LmhlaWdodCkpOwotICAgICAgICBtTGF5ZXIubG9hZFRleHR1cmUoZGlydHksIG1UZXh0dXJlTmFtZSwgdCwgdywgaCk7Ci0gICAgICAgIG1MYXllci5kcmF3V2l0aE9wZW5HTChjbGlwLCBtVGV4dHVyZU5hbWUsIHQsIG1CdWZmZXJIZWFwLnRyYW5zZm9ybSk7Ci0gICAgfQotfQotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1MYXllckJ1ZmZlcjo6T3ZlcmxheVNvdXJjZTo6T3ZlcmxheVNvdXJjZShMYXllckJ1ZmZlciYgbGF5ZXIsCi0gICAgICAgIHNwPE92ZXJsYXlSZWY+KiBvdmVybGF5UmVmLCAKLSAgICAgICAgdWludDMyX3QgdywgdWludDMyX3QgaCwgaW50MzJfdCBmb3JtYXQpCi0gICAgOiBTb3VyY2UobGF5ZXIpLCBtVmlzaWJpbGl0eUNoYW5nZWQoZmFsc2UpLAotICAgIG1PdmVybGF5KDApLCBtT3ZlcmxheUhhbmRsZSgwKSwgbU92ZXJsYXlEZXZpY2UoMCkKLXsKLSAgICBvdmVybGF5X2NvbnRyb2xfZGV2aWNlX3QqIG92ZXJsYXlfZGV2ID0gbUxheWVyLm1GbGluZ2VyLT5nZXRPdmVybGF5RW5naW5lKCk7Ci0gICAgaWYgKG92ZXJsYXlfZGV2ID09IE5VTEwpIHsKLSAgICAgICAgLy8gb3ZlcmxheXMgbm90IHN1cHBvcnRlZAotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgbU92ZXJsYXlEZXZpY2UgPSBvdmVybGF5X2RldjsKLSAgICBvdmVybGF5X3QqIG92ZXJsYXkgPSBvdmVybGF5X2Rldi0+Y3JlYXRlT3ZlcmxheShvdmVybGF5X2RldiwgdywgaCwgZm9ybWF0KTsKLSAgICBpZiAob3ZlcmxheSA9PSBOVUxMKSB7Ci0gICAgICAgIC8vIGNvdWxkbid0IGNyZWF0ZSB0aGUgb3ZlcmxheSAobm8gbWVtb3J5PyBubyBtb3JlIG92ZXJsYXlzPykKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIC8vIGVuYWJsZSBkaXRoZXJpbmcuLi4KLSAgICBvdmVybGF5X2Rldi0+c2V0UGFyYW1ldGVyKG92ZXJsYXlfZGV2LCBvdmVybGF5LCAKLSAgICAgICAgICAgIE9WRVJMQVlfRElUSEVSLCBPVkVSTEFZX0VOQUJMRSk7Ci0KLSAgICBtT3ZlcmxheSA9IG92ZXJsYXk7Ci0gICAgbVdpZHRoID0gb3ZlcmxheS0+dzsKLSAgICBtSGVpZ2h0ID0gb3ZlcmxheS0+aDsKLSAgICBtRm9ybWF0ID0gb3ZlcmxheS0+Zm9ybWF0OyAKLSAgICBtV2lkdGhTdHJpZGUgPSBvdmVybGF5LT53X3N0cmlkZTsKLSAgICBtSGVpZ2h0U3RyaWRlID0gb3ZlcmxheS0+aF9zdHJpZGU7Ci0KLSAgICBtT3ZlcmxheUhhbmRsZSA9IG92ZXJsYXktPmdldEhhbmRsZVJlZihvdmVybGF5KTsKLSAgICAKLSAgICAvLyBOT1RFOiBoZXJlIGl0J3Mgb2theSB0byBhY3F1aXJlIGEgcmVmZXJlbmNlIHRvICJ0aGlzIm0gYXMgbG9uZyBhcwotICAgIC8vIHRoZSByZWZlcmVuY2UgaXMgbm90IHJlbGVhc2VkIGJlZm9yZSB3ZSBsZWF2ZSB0aGUgY3Rvci4KLSAgICBzcDxPdmVybGF5Q2hhbm5lbD4gY2hhbm5lbCA9IG5ldyBPdmVybGF5Q2hhbm5lbCh0aGlzKTsKLQotICAgICpvdmVybGF5UmVmID0gbmV3IE92ZXJsYXlSZWYobU92ZXJsYXlIYW5kbGUsIGNoYW5uZWwsCi0gICAgICAgICAgICBtV2lkdGgsIG1IZWlnaHQsIG1Gb3JtYXQsIG1XaWR0aFN0cmlkZSwgbUhlaWdodFN0cmlkZSk7Ci19Ci0KLUxheWVyQnVmZmVyOjpPdmVybGF5U291cmNlOjp+T3ZlcmxheVNvdXJjZSgpCi17Ci0gICAgaWYgKG1PdmVybGF5ICYmIG1PdmVybGF5RGV2aWNlKSB7Ci0gICAgICAgIG92ZXJsYXlfY29udHJvbF9kZXZpY2VfdCogb3ZlcmxheV9kZXYgPSBtT3ZlcmxheURldmljZTsKLSAgICAgICAgb3ZlcmxheV9kZXYtPmRlc3Ryb3lPdmVybGF5KG92ZXJsYXlfZGV2LCBtT3ZlcmxheSk7Ci0gICAgfQotfQotCi12b2lkIExheWVyQnVmZmVyOjpPdmVybGF5U291cmNlOjpvblRyYW5zYWN0aW9uKHVpbnQzMl90IGZsYWdzKQotewotICAgIGNvbnN0IExheWVyOjpTdGF0ZSYgZnJvbnQobUxheWVyLmRyYXdpbmdTdGF0ZSgpKTsKLSAgICBjb25zdCBMYXllcjo6U3RhdGUmIHRlbXAobUxheWVyLmN1cnJlbnRTdGF0ZSgpKTsKLSAgICBpZiAodGVtcC5zZXF1ZW5jZSAhPSBmcm9udC5zZXF1ZW5jZSkgewotICAgICAgICBtVmlzaWJpbGl0eUNoYW5nZWQgPSB0cnVlOwotICAgIH0KLX0KLQotdm9pZCBMYXllckJ1ZmZlcjo6T3ZlcmxheVNvdXJjZTo6b25WaXNpYmlsaXR5UmVzb2x2ZWQoCi0gICAgICAgIGNvbnN0IFRyYW5zZm9ybSYgcGxhbmVUcmFuc2Zvcm0pCi17Ci0gICAgLy8gdGhpcyBjb2RlLXBhdGggbXVzdCBiZSBhcyB0aWdodCBhcyBwb3NzaWJsZSwgaXQncyBjYWxsZWQgZWFjaCB0aW1lCi0gICAgLy8gdGhlIHNjcmVlbiBpcyBjb21wb3NpdGVkLgotICAgIGlmIChVTkxJS0VMWShtT3ZlcmxheSAhPSAwKSkgewotICAgICAgICBpZiAobVZpc2liaWxpdHlDaGFuZ2VkKSB7Ci0gICAgICAgICAgICBtVmlzaWJpbGl0eUNoYW5nZWQgPSBmYWxzZTsKLSAgICAgICAgICAgIGNvbnN0IFJlY3QmIGJvdW5kcyA9IG1MYXllci5nZXRUcmFuc2Zvcm1lZEJvdW5kcygpOwotICAgICAgICAgICAgaW50IHggPSBib3VuZHMubGVmdDsKLSAgICAgICAgICAgIGludCB5ID0gYm91bmRzLnRvcDsKLSAgICAgICAgICAgIGludCB3ID0gYm91bmRzLndpZHRoKCk7Ci0gICAgICAgICAgICBpbnQgaCA9IGJvdW5kcy5oZWlnaHQoKTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8gd2UgbmVlZCBhIGxvY2sgaGVyZSB0byBwcm90ZWN0ICJkZXN0cm95IgotICAgICAgICAgICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICAgICAgICAgIGlmIChtT3ZlcmxheSkgewotICAgICAgICAgICAgICAgIG92ZXJsYXlfY29udHJvbF9kZXZpY2VfdCogb3ZlcmxheV9kZXYgPSBtT3ZlcmxheURldmljZTsKLSAgICAgICAgICAgICAgICBvdmVybGF5X2Rldi0+c2V0UG9zaXRpb24ob3ZlcmxheV9kZXYsIG1PdmVybGF5LCB4LHksdyxoKTsKLSAgICAgICAgICAgICAgICBvdmVybGF5X2Rldi0+c2V0UGFyYW1ldGVyKG92ZXJsYXlfZGV2LCBtT3ZlcmxheSwgCi0gICAgICAgICAgICAgICAgICAgICAgICBPVkVSTEFZX1RSQU5TRk9STSwgbUxheWVyLmdldE9yaWVudGF0aW9uKCkpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQotCi12b2lkIExheWVyQnVmZmVyOjpPdmVybGF5U291cmNlOjpzZXJ2ZXJEZXN0cm95KCkgCi17Ci0gICAgbUxheWVyLmNsZWFyU291cmNlKCk7Ci0gICAgZGVzdHJveU92ZXJsYXkoKTsKLX0KLQotdm9pZCBMYXllckJ1ZmZlcjo6T3ZlcmxheVNvdXJjZTo6ZGVzdHJveU92ZXJsYXkoKSAKLXsKLSAgICAvLyB3ZSBuZWVkIGEgbG9jayBoZXJlIHRvIHByb3RlY3QgIm9uVmlzaWJpbGl0eVJlc29sdmVkIgotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgaWYgKG1PdmVybGF5KSB7Ci0gICAgICAgIG92ZXJsYXlfY29udHJvbF9kZXZpY2VfdCogb3ZlcmxheV9kZXYgPSBtT3ZlcmxheURldmljZTsKLSAgICAgICAgb3ZlcmxheV9kZXYtPmRlc3Ryb3lPdmVybGF5KG92ZXJsYXlfZGV2LCBtT3ZlcmxheSk7Ci0gICAgICAgIG1PdmVybGF5ID0gMDsKLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCdWZmZXIuaCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJCdWZmZXIuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMmRjNzdmMS4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyQnVmZmVyLmgKKysrIC9kZXYvbnVsbApAQCAtMSwyMTYgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9MQVlFUl9CVUZGRVJfSAotI2RlZmluZSBBTkRST0lEX0xBWUVSX0JVRkZFUl9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0lNZW1vcnkuaD4KLSNpbmNsdWRlIDxwcml2YXRlL3VpL0xheWVyU3RhdGUuaD4KLSNpbmNsdWRlIDxFR0wvZWdsbmF0aXZlcy5oPgotCi0jaW5jbHVkZSAiTGF5ZXJCYXNlLmgiCi0jaW5jbHVkZSAiTGF5ZXJCaXRtYXAuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgTWVtb3J5RGVhbGVyOwotY2xhc3MgUmVnaW9uOwotY2xhc3MgT3ZlcmxheVJlZjsKLQotY2xhc3MgTGF5ZXJCdWZmZXIgOiBwdWJsaWMgTGF5ZXJCYXNlQ2xpZW50Ci17Ci0gICAgY2xhc3MgU291cmNlIDogcHVibGljIExpZ2h0UmVmQmFzZTxTb3VyY2U+IHsKLSAgICBwdWJsaWM6Ci0gICAgICAgIFNvdXJjZShMYXllckJ1ZmZlciYgbGF5ZXIpOwotICAgICAgICB2aXJ0dWFsIH5Tb3VyY2UoKTsKLSAgICAgICAgdmlydHVhbCB2b2lkIG9uRHJhdyhjb25zdCBSZWdpb24mIGNsaXApIGNvbnN0OwotICAgICAgICB2aXJ0dWFsIHZvaWQgb25UcmFuc2FjdGlvbih1aW50MzJfdCBmbGFncyk7Ci0gICAgICAgIHZpcnR1YWwgdm9pZCBvblZpc2liaWxpdHlSZXNvbHZlZChjb25zdCBUcmFuc2Zvcm0mIHBsYW5lVHJhbnNmb3JtKTsKLSAgICAgICAgdmlydHVhbCB2b2lkIHBvc3RCdWZmZXIoc3NpemVfdCBvZmZzZXQpOwotICAgICAgICB2aXJ0dWFsIHZvaWQgdW5yZWdpc3RlckJ1ZmZlcnMoKTsKLSAgICAgICAgdmlydHVhbCBib29sIHRyYW5zZm9ybWVkKCkgY29uc3Q7Ci0gICAgcHJvdGVjdGVkOgotICAgICAgICBMYXllckJ1ZmZlciYgbUxheWVyOwotICAgIH07Ci0KLQotcHVibGljOgotICAgIHN0YXRpYyBjb25zdCB1aW50MzJfdCB0eXBlSW5mbzsKLSAgICBzdGF0aWMgY29uc3QgY2hhciogY29uc3QgdHlwZUlEOwotICAgIHZpcnR1YWwgY2hhciBjb25zdCogZ2V0VHlwZUlEKCkgY29uc3QgeyByZXR1cm4gdHlwZUlEOyB9Ci0gICAgdmlydHVhbCB1aW50MzJfdCBnZXRUeXBlSW5mbygpIGNvbnN0IHsgcmV0dXJuIHR5cGVJbmZvOyB9Ci0KLSAgICAgICAgICAgIExheWVyQnVmZmVyKFN1cmZhY2VGbGluZ2VyKiBmbGluZ2VyLCBEaXNwbGF5SUQgZGlzcGxheSwKLSAgICAgICAgICAgICAgICAgICAgICAgIENsaWVudCogY2xpZW50LCBpbnQzMl90IGkpOwotICAgICAgICB2aXJ0dWFsIH5MYXllckJ1ZmZlcigpOwotCi0gICAgdmlydHVhbCBib29sIG5lZWRzQmxlbmRpbmcoKSBjb25zdDsKLQotICAgIHZpcnR1YWwgc3A8TGF5ZXJCYXNlQ2xpZW50OjpTdXJmYWNlPiBnZXRTdXJmYWNlKCkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkIG9uRHJhdyhjb25zdCBSZWdpb24mIGNsaXApIGNvbnN0OwotICAgIHZpcnR1YWwgdWludDMyX3QgZG9UcmFuc2FjdGlvbih1aW50MzJfdCBmbGFncyk7Ci0gICAgdmlydHVhbCB2b2lkIHVubG9ja1BhZ2VGbGlwKGNvbnN0IFRyYW5zZm9ybSYgcGxhbmVUcmFuc2Zvcm0sIFJlZ2lvbiYgb3V0RGlydHlSZWdpb24pOwotICAgIHZpcnR1YWwgYm9vbCB0cmFuc2Zvcm1lZCgpIGNvbnN0OwotCi0gICAgc3RhdHVzX3QgcmVnaXN0ZXJCdWZmZXJzKGNvbnN0IElTdXJmYWNlOjpCdWZmZXJIZWFwJiBidWZmZXJzKTsKLSAgICB2b2lkIHBvc3RCdWZmZXIoc3NpemVfdCBvZmZzZXQpOwotICAgIHZvaWQgdW5yZWdpc3RlckJ1ZmZlcnMoKTsKLSAgICBzcDxPdmVybGF5UmVmPiBjcmVhdGVPdmVybGF5KHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIGludDMyX3QgZm9ybWF0KTsKLSAgICAKLSAgICBzcDxTb3VyY2U+IGdldFNvdXJjZSgpIGNvbnN0OwotICAgIHNwPFNvdXJjZT4gY2xlYXJTb3VyY2UoKTsKLSAgICB2b2lkIHNldE5lZWRzQmxlbmRpbmcoYm9vbCBibGVuZGluZyk7Ci0gICAgY29uc3QgUmVjdCYgZ2V0VHJhbnNmb3JtZWRCb3VuZHMoKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiBtVHJhbnNmb3JtZWRCb3VuZHM7Ci0gICAgfQotCi1wcml2YXRlOgotICAgIHN0cnVjdCBOYXRpdmVCdWZmZXIgewotICAgICAgICBjb3B5Yml0X2ltYWdlX3QgICBpbWc7Ci0gICAgICAgIGNvcHliaXRfcmVjdF90ICAgIGNyb3A7Ci0gICAgfTsKLQotICAgIGNsYXNzIEJ1ZmZlciA6IHB1YmxpYyBMaWdodFJlZkJhc2U8QnVmZmVyPiB7Ci0gICAgcHVibGljOgotICAgICAgICBCdWZmZXIoY29uc3QgSVN1cmZhY2U6OkJ1ZmZlckhlYXAmIGJ1ZmZlcnMsIHNzaXplX3Qgb2Zmc2V0KTsKLSAgICAgICAgaW5saW5lIHN0YXR1c190IGdldFN0YXR1cygpIGNvbnN0IHsKLSAgICAgICAgICAgIHJldHVybiBtQnVmZmVySGVhcC5oZWFwIT0wID8gTk9fRVJST1IgOiBOT19JTklUOwotICAgICAgICB9Ci0gICAgICAgIGlubGluZSBjb25zdCBOYXRpdmVCdWZmZXImIGdldEJ1ZmZlcigpIGNvbnN0IHsKLSAgICAgICAgICAgIHJldHVybiBtTmF0aXZlQnVmZmVyOwotICAgICAgICB9Ci0gICAgcHJvdGVjdGVkOgotICAgICAgICBmcmllbmQgY2xhc3MgTGlnaHRSZWZCYXNlPEJ1ZmZlcj47Ci0gICAgICAgIEJ1ZmZlciYgb3BlcmF0b3IgPSAoY29uc3QgQnVmZmVyJiByaHMpOwotICAgICAgICBCdWZmZXIoY29uc3QgQnVmZmVyJiByaHMpOwotICAgICAgICB+QnVmZmVyKCk7Ci0gICAgcHJpdmF0ZToKLSAgICAgICAgSVN1cmZhY2U6OkJ1ZmZlckhlYXAgICAgbUJ1ZmZlckhlYXA7Ci0gICAgICAgIE5hdGl2ZUJ1ZmZlciAgICAgICAgICAgIG1OYXRpdmVCdWZmZXI7Ci0gICAgfTsKLQotICAgIGNsYXNzIEJ1ZmZlclNvdXJjZSA6IHB1YmxpYyBTb3VyY2UgewotICAgIHB1YmxpYzoKLSAgICAgICAgQnVmZmVyU291cmNlKExheWVyQnVmZmVyJiBsYXllciwgY29uc3QgSVN1cmZhY2U6OkJ1ZmZlckhlYXAmIGJ1ZmZlcnMpOwotICAgICAgICB2aXJ0dWFsIH5CdWZmZXJTb3VyY2UoKTsKLQotICAgICAgICBzdGF0dXNfdCBnZXRTdGF0dXMoKSBjb25zdCB7IHJldHVybiBtU3RhdHVzOyB9Ci0gICAgICAgIHNwPEJ1ZmZlcj4gZ2V0QnVmZmVyKCkgY29uc3Q7Ci0gICAgICAgIHZvaWQgc2V0QnVmZmVyKGNvbnN0IHNwPEJ1ZmZlcj4mIGJ1ZmZlcik7Ci0KLSAgICAgICAgdmlydHVhbCB2b2lkIG9uRHJhdyhjb25zdCBSZWdpb24mIGNsaXApIGNvbnN0OwotICAgICAgICB2aXJ0dWFsIHZvaWQgcG9zdEJ1ZmZlcihzc2l6ZV90IG9mZnNldCk7Ci0gICAgICAgIHZpcnR1YWwgdm9pZCB1bnJlZ2lzdGVyQnVmZmVycygpOwotICAgICAgICB2aXJ0dWFsIGJvb2wgdHJhbnNmb3JtZWQoKSBjb25zdDsKLSAgICBwcml2YXRlOgotICAgICAgICBtdXRhYmxlIE11dGV4ICAgbUxvY2s7Ci0gICAgICAgIHNwPEJ1ZmZlcj4gICAgICBtQnVmZmVyOwotICAgICAgICBzdGF0dXNfdCAgICAgICAgbVN0YXR1czsKLSAgICAgICAgSVN1cmZhY2U6OkJ1ZmZlckhlYXAgbUJ1ZmZlckhlYXA7Ci0gICAgICAgIHNpemVfdCAgICAgICAgICBtQnVmZmVyU2l6ZTsKLSAgICAgICAgbXV0YWJsZSBzcDxNZW1vcnlEZWFsZXI+IG1UZW1wb3JhcnlEZWFsZXI7Ci0gICAgICAgIG11dGFibGUgTGF5ZXJCaXRtYXAgbVRlbXBCaXRtYXA7Ci0gICAgICAgIG11dGFibGUgR0x1aW50ICBtVGV4dHVyZU5hbWU7Ci0gICAgfTsKLSAgICAKLSAgICBjbGFzcyBPdmVybGF5U291cmNlIDogcHVibGljIFNvdXJjZSB7Ci0gICAgcHVibGljOgotICAgICAgICBPdmVybGF5U291cmNlKExheWVyQnVmZmVyJiBsYXllciwKLSAgICAgICAgICAgICAgICBzcDxPdmVybGF5UmVmPiogb3ZlcmxheVJlZiwgCi0gICAgICAgICAgICAgICAgdWludDMyX3QgdywgdWludDMyX3QgaCwgaW50MzJfdCBmb3JtYXQpOwotICAgICAgICB2aXJ0dWFsIH5PdmVybGF5U291cmNlKCk7Ci0gICAgICAgIHZpcnR1YWwgdm9pZCBvblRyYW5zYWN0aW9uKHVpbnQzMl90IGZsYWdzKTsKLSAgICAgICAgdmlydHVhbCB2b2lkIG9uVmlzaWJpbGl0eVJlc29sdmVkKGNvbnN0IFRyYW5zZm9ybSYgcGxhbmVUcmFuc2Zvcm0pOwotICAgIHByaXZhdGU6Ci0gICAgICAgIHZvaWQgc2VydmVyRGVzdHJveSgpOyAKLSAgICAgICAgdm9pZCBkZXN0cm95T3ZlcmxheSgpOyAKLSAgICAgICAgY2xhc3MgT3ZlcmxheUNoYW5uZWwgOiBwdWJsaWMgQm5PdmVybGF5IHsKLSAgICAgICAgICAgIG11dGFibGUgTXV0ZXggbUxvY2s7Ci0gICAgICAgICAgICBzcDxPdmVybGF5U291cmNlPiBtU291cmNlOwotICAgICAgICAgICAgdmlydHVhbCB2b2lkIGRlc3Ryb3koKSB7Ci0gICAgICAgICAgICAgICAgc3A8T3ZlcmxheVNvdXJjZT4gc291cmNlOwotICAgICAgICAgICAgICAgIHsgLy8gc2NvcGUgZm9yIHRoZSBsb2NrOwotICAgICAgICAgICAgICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgICAgICAgICAgICAgICAgICBzb3VyY2UgPSBtU291cmNlOwotICAgICAgICAgICAgICAgICAgICBtU291cmNlLmNsZWFyKCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChzb3VyY2UgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICBzb3VyY2UtPnNlcnZlckRlc3Ryb3koKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIHB1YmxpYzoKLSAgICAgICAgICAgIE92ZXJsYXlDaGFubmVsKGNvbnN0IHNwPE92ZXJsYXlTb3VyY2U+JiBzb3VyY2UpCi0gICAgICAgICAgICAgICAgOiBtU291cmNlKHNvdXJjZSkgewotICAgICAgICAgICAgfQotICAgICAgICB9OwotICAgICAgICBmcmllbmQgY2xhc3MgT3ZlcmxheUNoYW5uZWw7Ci0gICAgICAgIGJvb2wgbVZpc2liaWxpdHlDaGFuZ2VkOwotCi0gICAgICAgIG92ZXJsYXlfdCogbU92ZXJsYXk7ICAgICAgICAKLSAgICAgICAgb3ZlcmxheV9oYW5kbGVfdCBtT3ZlcmxheUhhbmRsZTsKLSAgICAgICAgb3ZlcmxheV9jb250cm9sX2RldmljZV90KiBtT3ZlcmxheURldmljZTsKLSAgICAgICAgdWludDMyX3QgbVdpZHRoOwotICAgICAgICB1aW50MzJfdCBtSGVpZ2h0OwotICAgICAgICBpbnQzMl90IG1Gb3JtYXQ7Ci0gICAgICAgIGludDMyX3QgbVdpZHRoU3RyaWRlOwotICAgICAgICBpbnQzMl90IG1IZWlnaHRTdHJpZGU7Ci0gICAgICAgIG11dGFibGUgTXV0ZXggbUxvY2s7Ci0gICAgfTsKLQotCi0gICAgY2xhc3MgU3VyZmFjZUJ1ZmZlciA6IHB1YmxpYyBMYXllckJhc2VDbGllbnQ6OlN1cmZhY2UKLSAgICB7Ci0gICAgcHVibGljOgotICAgICAgICAgICAgICAgIFN1cmZhY2VCdWZmZXIoU3VyZmFjZUlEIGlkLCBMYXllckJ1ZmZlciogb3duZXIpOwotICAgICAgICB2aXJ0dWFsIH5TdXJmYWNlQnVmZmVyKCk7Ci0gICAgICAgIHZpcnR1YWwgc3RhdHVzX3Qgb25UcmFuc2FjdCgKLSAgICAgICAgICAgIHVpbnQzMl90IGNvZGUsIGNvbnN0IFBhcmNlbCYgZGF0YSwgUGFyY2VsKiByZXBseSwgdWludDMyX3QgZmxhZ3MpOwotICAgICAgICB2aXJ0dWFsIHN0YXR1c190IHJlZ2lzdGVyQnVmZmVycyhjb25zdCBJU3VyZmFjZTo6QnVmZmVySGVhcCYgYnVmZmVycyk7Ci0gICAgICAgIHZpcnR1YWwgdm9pZCBwb3N0QnVmZmVyKHNzaXplX3Qgb2Zmc2V0KTsKLSAgICAgICAgdmlydHVhbCB2b2lkIHVucmVnaXN0ZXJCdWZmZXJzKCk7Ci0gICAgICAgIHZpcnR1YWwgc3A8T3ZlcmxheVJlZj4gY3JlYXRlT3ZlcmxheSgKLSAgICAgICAgICAgICAgICB1aW50MzJfdCB3LCB1aW50MzJfdCBoLCBpbnQzMl90IGZvcm1hdCk7Ci0gICAgICAgdm9pZCBkaXNvd24oKTsKLSAgICBwcml2YXRlOgotICAgICAgICBMYXllckJ1ZmZlciogZ2V0T3duZXIoKSBjb25zdCB7Ci0gICAgICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgICAgICAgICAgcmV0dXJuIG1Pd25lcjsKLSAgICAgICAgfQotICAgICAgICBtdXRhYmxlIE11dGV4ICAgbUxvY2s7Ci0gICAgICAgIExheWVyQnVmZmVyKiAgICBtT3duZXI7Ci0gICAgfTsKLQotICAgIGZyaWVuZCBjbGFzcyBTdXJmYWNlRmxpbmdlcjsKLSAgICBzcDxTdXJmYWNlQnVmZmVyPiAgIGdldENsaWVudFN1cmZhY2UoKSBjb25zdDsKLQotICAgIG11dGFibGUgTXV0ZXggICBtTG9jazsKLSAgICBzcDxTb3VyY2U+ICAgICAgbVNvdXJjZTsKLQotICAgIGJvb2wgICAgICAgICAgICBtSW52YWxpZGF0ZTsKLSAgICBib29sICAgICAgICAgICAgbU5lZWRzQmxlbmRpbmc7Ci0gICAgbXV0YWJsZSB3cDxTdXJmYWNlQnVmZmVyPiBtQ2xpZW50U3VyZmFjZTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9MQVlFUl9CVUZGRVJfSApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllckRpbS5jcHAgYi9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyRGltLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMGMzNDdjYy4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyRGltLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDExMyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJTdXJmYWNlRmxpbmdlciIKLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlICJMYXllckRpbS5oIgotI2luY2x1ZGUgIlN1cmZhY2VGbGluZ2VyLmgiCi0jaW5jbHVkZSAiVlJhbUhlYXAuaCIKLSNpbmNsdWRlICJEaXNwbGF5SGFyZHdhcmUvRGlzcGxheUhhcmR3YXJlLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jb25zdCB1aW50MzJfdCBMYXllckRpbTo6dHlwZUluZm8gPSBMYXllckJhc2VDbGllbnQ6OnR5cGVJbmZvIHwgMHgxMDsKLWNvbnN0IGNoYXIqIGNvbnN0IExheWVyRGltOjp0eXBlSUQgPSAiTGF5ZXJEaW0iOwotc3A8TWVtb3J5RGVhbGVyPiBMYXllckRpbTo6bURpbW1lckRlYWxlcjsKLUxheWVyQml0bWFwIExheWVyRGltOjptRGltbWVyQml0bWFwOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotTGF5ZXJEaW06OkxheWVyRGltKFN1cmZhY2VGbGluZ2VyKiBmbGluZ2VyLCBEaXNwbGF5SUQgZGlzcGxheSwKLSAgICAgICAgQ2xpZW50KiBjbGllbnQsIGludDMyX3QgaSkKLSAgICAgOiBMYXllckJhc2VDbGllbnQoZmxpbmdlciwgZGlzcGxheSwgY2xpZW50LCBpKQotewotfQotCi12b2lkIExheWVyRGltOjppbml0RGltbWVyKFN1cmZhY2VGbGluZ2VyKiBmbGluZ2VyLCB1aW50MzJfdCB3LCB1aW50MzJfdCBoKQotewotICAgIC8vIG11c3Qgb25seSBiZSBjYWxsZWQgb25jZS4KLSAgICBtRGltbWVyRGVhbGVyID0gZmxpbmdlci0+Z2V0U3VyZmFjZUhlYXBNYW5hZ2VyKCkKLSAgICAgICAgICAgIC0+Y3JlYXRlSGVhcChJU3VyZmFjZUNvbXBvc2VyOjplSGFyZHdhcmUpOwotICAgIGlmIChtRGltbWVyRGVhbGVyICE9IDApIHsKLSAgICAgICAgbURpbW1lckJpdG1hcC5pbml0KG1EaW1tZXJEZWFsZXIpOwotICAgICAgICBtRGltbWVyQml0bWFwLnNldEJpdHModywgaCwgMSwgUElYRUxfRk9STUFUX1JHQl81NjUpOwotICAgICAgICBtRGltbWVyQml0bWFwLmNsZWFyKCk7Ci0gICAgfQotfQotCi1MYXllckRpbTo6fkxheWVyRGltKCkKLXsKLX0KLQotdm9pZCBMYXllckRpbTo6b25EcmF3KGNvbnN0IFJlZ2lvbiYgY2xpcCkgY29uc3QKLXsKLSAgICBjb25zdCBTdGF0ZSYgcyhkcmF3aW5nU3RhdGUoKSk7Ci0KLSAgICBSZWdpb246Oml0ZXJhdG9yIGl0ZXJhdG9yKGNsaXApOwotICAgIGlmIChzLmFscGhhPjAgJiYgaXRlcmF0b3IpIHsKLSAgICAgICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiBodyhncmFwaGljUGxhbmUoMCkuZGlzcGxheUhhcmR3YXJlKCkpOwotCi0gICAgICAgIHN0YXR1c190IGVyciA9IE5PX0VSUk9SOwotICAgICAgICBjb25zdCBpbnQgY2FuX3VzZV9jb3B5Yml0ID0gY2FuVXNlQ29weWJpdCgpOwotICAgICAgICBpZiAoY2FuX3VzZV9jb3B5Yml0KSAgewotICAgICAgICAgICAgLy8gU3RvcFdhdGNoIHdhdGNoKCJjb3B5Yml0Iik7Ci0gICAgICAgICAgICBjb3B5Yml0X2ltYWdlX3QgZHN0OwotICAgICAgICAgICAgaHcuZ2V0RGlzcGxheVN1cmZhY2UoJmRzdCk7Ci0gICAgICAgICAgICBjb25zdCBjb3B5Yml0X3JlY3RfdCYgZHJlY3QKLSAgICAgICAgICAgICAgICA9IHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgY29weWJpdF9yZWN0X3QmPihtVHJhbnNmb3JtZWRCb3VuZHMpOwotCi0gICAgICAgICAgICBjb3B5Yml0X2ltYWdlX3Qgc3JjOwotICAgICAgICAgICAgbURpbW1lckJpdG1hcC5nZXRCaXRtYXBTdXJmYWNlKCZzcmMpOwotICAgICAgICAgICAgY29uc3QgY29weWJpdF9yZWN0X3QmIHNyZWN0KGRyZWN0KTsKLQotICAgICAgICAgICAgY29weWJpdF9kZXZpY2VfdCogY29weWJpdCA9IG1GbGluZ2VyLT5nZXRCbGl0RW5naW5lKCk7Ci0gICAgICAgICAgICBjb3B5Yml0LT5zZXRfcGFyYW1ldGVyKGNvcHliaXQsIENPUFlCSVRfVFJBTlNGT1JNLCAwKTsKLSAgICAgICAgICAgIGNvcHliaXQtPnNldF9wYXJhbWV0ZXIoY29weWJpdCwgQ09QWUJJVF9QTEFORV9BTFBIQSwgcy5hbHBoYSk7Ci0gICAgICAgICAgICBjb3B5Yml0LT5zZXRfcGFyYW1ldGVyKGNvcHliaXQsIENPUFlCSVRfRElUSEVSLCBDT1BZQklUX0VOQUJMRSk7Ci0gICAgICAgICAgICByZWdpb25faXRlcmF0b3IgaXQoY2xpcCk7Ci0gICAgICAgICAgICBlcnIgPSBjb3B5Yml0LT5zdHJldGNoKGNvcHliaXQsICZkc3QsICZzcmMsICZkcmVjdCwgJnNyZWN0LCAmaXQpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKCFjYW5fdXNlX2NvcHliaXQgfHwgZXJyKSB7Ci0gICAgICAgICAgICBjb25zdCBHR0xmaXhlZCBhbHBoYSA9IChzLmFscGhhIDw8IDE2KS8yNTU7Ci0gICAgICAgICAgICBjb25zdCB1aW50MzJfdCBmYkhlaWdodCA9IGh3LmdldEhlaWdodCgpOwotICAgICAgICAgICAgZ2xEaXNhYmxlKEdMX1RFWFRVUkVfMkQpOwotICAgICAgICAgICAgZ2xEaXNhYmxlKEdMX0RJVEhFUik7Ci0gICAgICAgICAgICBnbEVuYWJsZShHTF9CTEVORCk7Ci0gICAgICAgICAgICBnbEJsZW5kRnVuYyhHTF9PTkUsIEdMX09ORV9NSU5VU19TUkNfQUxQSEEpOwotICAgICAgICAgICAgZ2xDb2xvcjR4KDAsIDAsIDAsIGFscGhhKTsKLSAgICAgICAgICAgIGdsVmVydGV4UG9pbnRlcigyLCBHTF9GSVhFRCwgMCwgbVZlcnRpY2VzKTsKLSAgICAgICAgICAgIFJlY3QgcjsKLSAgICAgICAgICAgIHdoaWxlIChpdGVyYXRvci5pdGVyYXRlKCZyKSkgewotICAgICAgICAgICAgICAgIGNvbnN0IEdMaW50IHN5ID0gZmJIZWlnaHQgLSAoci50b3AgKyByLmhlaWdodCgpKTsKLSAgICAgICAgICAgICAgICBnbFNjaXNzb3Ioci5sZWZ0LCBzeSwgci53aWR0aCgpLCByLmhlaWdodCgpKTsKLSAgICAgICAgICAgICAgICBnbERyYXdBcnJheXMoR0xfVFJJQU5HTEVfRkFOLCAwLCA0KTsgCi0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllckRpbS5oIGIvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllckRpbS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzZTM3YTQ3Li4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJEaW0uaAorKysgL2Rldi9udWxsCkBAIC0xLDU3ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfTEFZRVJfRElNX0gKLSNkZWZpbmUgQU5EUk9JRF9MQVlFUl9ESU1fSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlICJMYXllckJhc2UuaCIKLSNpbmNsdWRlICJMYXllckJpdG1hcC5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBMYXllckRpbSA6IHB1YmxpYyBMYXllckJhc2VDbGllbnQKLXsKLXB1YmxpYzogICAgCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IHR5cGVJbmZvOwotICAgIHN0YXRpYyBjb25zdCBjaGFyKiBjb25zdCB0eXBlSUQ7Ci0gICAgdmlydHVhbCBjaGFyIGNvbnN0KiBnZXRUeXBlSUQoKSBjb25zdCB7IHJldHVybiB0eXBlSUQ7IH0KLSAgICB2aXJ0dWFsIHVpbnQzMl90IGdldFR5cGVJbmZvKCkgY29uc3QgeyByZXR1cm4gdHlwZUluZm87IH0KLSAgICAKLSAgICAgICAgICAgICAgICBMYXllckRpbShTdXJmYWNlRmxpbmdlciogZmxpbmdlciwgRGlzcGxheUlEIGRpc3BsYXksCi0gICAgICAgICAgICAgICAgICAgICAgICBDbGllbnQqIGNsaWVudCwgaW50MzJfdCBpKTsKLSAgICAgICAgdmlydHVhbCB+TGF5ZXJEaW0oKTsKLQotICAgIHZpcnR1YWwgdm9pZCBvbkRyYXcoY29uc3QgUmVnaW9uJiBjbGlwKSBjb25zdDsKLSAgICB2aXJ0dWFsIGJvb2wgbmVlZHNCbGVuZGluZygpIGNvbnN0ICB7IHJldHVybiB0cnVlOyB9Ci0gICAgdmlydHVhbCBib29sIGlzU2VjdXJlKCkgY29uc3QgICAgICAgeyByZXR1cm4gZmFsc2U7IH0KLQotICAgIHN0YXRpYyB2b2lkIGluaXREaW1tZXIoU3VyZmFjZUZsaW5nZXIqIGZsaW5nZXIsIHVpbnQzMl90IHcsIHVpbnQzMl90IGgpOwotCi1wcml2YXRlOgotICAgIHN0YXRpYyBzcDxNZW1vcnlEZWFsZXI+IG1EaW1tZXJEZWFsZXI7Ci0gICAgc3RhdGljIExheWVyQml0bWFwIG1EaW1tZXJCaXRtYXA7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfTEFZRVJfRElNX0gKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJPcmllbnRhdGlvbkFuaW0uY3BwIGIvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllck9yaWVudGF0aW9uQW5pbS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJiNzJkN2MuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9MYXllck9yaWVudGF0aW9uQW5pbS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwyODcgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNkZWZpbmUgTE9HX1RBRyAiU3VyZmFjZUZsaW5nZXIiCi0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8Y29yZS9Ta0JpdG1hcC5oPgotCi0jaW5jbHVkZSA8dWkvRUdMRGlzcGxheVN1cmZhY2UuaD4KLQotI2luY2x1ZGUgIkxheWVyQmFzZS5oIgotI2luY2x1ZGUgIkxheWVyT3JpZW50YXRpb25BbmltLmgiCi0jaW5jbHVkZSAiU3VyZmFjZUZsaW5nZXIuaCIKLSNpbmNsdWRlICJEaXNwbGF5SGFyZHdhcmUvRGlzcGxheUhhcmR3YXJlLmgiCi0jaW5jbHVkZSAiT3JpZW50YXRpb25BbmltYXRpb24uaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNvbnN0IHVpbnQzMl90IExheWVyT3JpZW50YXRpb25BbmltOjp0eXBlSW5mbyA9IExheWVyQmFzZTo6dHlwZUluZm8gfCAweDgwOwotY29uc3QgY2hhciogY29uc3QgTGF5ZXJPcmllbnRhdGlvbkFuaW06OnR5cGVJRCA9ICJMYXllck9yaWVudGF0aW9uQW5pbSI7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1MYXllck9yaWVudGF0aW9uQW5pbTo6TGF5ZXJPcmllbnRhdGlvbkFuaW0oCi0gICAgICAgIFN1cmZhY2VGbGluZ2VyKiBmbGluZ2VyLCBEaXNwbGF5SUQgZGlzcGxheSwgCi0gICAgICAgIE9yaWVudGF0aW9uQW5pbWF0aW9uKiBhbmltLCAKLSAgICAgICAgY29uc3QgTGF5ZXJCaXRtYXAmIGJpdG1hcCwKLSAgICAgICAgY29uc3QgTGF5ZXJCaXRtYXAmIGJpdG1hcEluKQotICAgIDogTGF5ZXJCYXNlKGZsaW5nZXIsIGRpc3BsYXkpLCBtQW5pbShhbmltKSwgCi0gICAgICBtQml0bWFwKGJpdG1hcCksIG1CaXRtYXBJbihiaXRtYXBJbiksIAotICAgICAgbVRleHR1cmVOYW1lKC0xKSwgbVRleHR1cmVOYW1lSW4oLTEpCi17Ci0gICAgbVN0YXJ0VGltZSA9IHN5c3RlbVRpbWUoKTsKLSAgICBtRmluaXNoVGltZSA9IDA7Ci0gICAgbU9yaWVudGF0aW9uQ29tcGxldGVkID0gZmFsc2U7Ci0gICAgbUZpcnN0UmVkcmF3ID0gZmFsc2U7Ci0gICAgbUxhc3ROb3JtYWxpemVkVGltZSA9IDA7Ci0gICAgbUxhc3RTY2FsZSA9IDA7Ci0gICAgbU5lZWRzQmxlbmRpbmcgPSBmYWxzZTsKLX0KLQotTGF5ZXJPcmllbnRhdGlvbkFuaW06On5MYXllck9yaWVudGF0aW9uQW5pbSgpCi17Ci0gICAgaWYgKG1UZXh0dXJlTmFtZSAhPSAtMVUpIHsKLSAgICAgICAgTGF5ZXJCYXNlOjpkZWxldGVkVGV4dHVyZXMuYWRkKG1UZXh0dXJlTmFtZSk7Ci0gICAgfQotICAgIGlmIChtVGV4dHVyZU5hbWVJbiAhPSAtMVUpIHsKLSAgICAgICAgTGF5ZXJCYXNlOjpkZWxldGVkVGV4dHVyZXMuYWRkKG1UZXh0dXJlTmFtZUluKTsKLSAgICB9Ci19Ci0KLWJvb2wgTGF5ZXJPcmllbnRhdGlvbkFuaW06Om5lZWRzQmxlbmRpbmcoKSBjb25zdCAKLXsKLSAgICByZXR1cm4gbU5lZWRzQmxlbmRpbmc7IAotfQotCi1Qb2ludCBMYXllck9yaWVudGF0aW9uQW5pbTo6Z2V0UGh5c2ljYWxTaXplKCkgY29uc3QKLXsKLSAgICBjb25zdCBHcmFwaGljUGxhbmUmIHBsYW5lKGdyYXBoaWNQbGFuZSgwKSk7Ci0gICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiBodyhwbGFuZS5kaXNwbGF5SGFyZHdhcmUoKSk7Ci0gICAgcmV0dXJuIFBvaW50KGh3LmdldFdpZHRoKCksIGh3LmdldEhlaWdodCgpKTsKLX0KLQotdm9pZCBMYXllck9yaWVudGF0aW9uQW5pbTo6dmFsaWRhdGVWaXNpYmlsaXR5KGNvbnN0IFRyYW5zZm9ybSYpCi17Ci0gICAgY29uc3QgTGF5ZXI6OlN0YXRlJiBzKGRyYXdpbmdTdGF0ZSgpKTsKLSAgICBjb25zdCBUcmFuc2Zvcm0gdHIocy50cmFuc2Zvcm0pOwotICAgIGNvbnN0IFBvaW50IHNpemUoZ2V0UGh5c2ljYWxTaXplKCkpOwotICAgIHVpbnQzMl90IHcgPSBzaXplLng7Ci0gICAgdWludDMyX3QgaCA9IHNpemUueTsKLSAgICBtVHJhbnNmb3JtZWRCb3VuZHMgPSB0ci5tYWtlQm91bmRzKHcsIGgpOwotICAgIG1MZWZ0ID0gdHIudHgoKTsKLSAgICBtVG9wICA9IHRyLnR5KCk7Ci0gICAgdHJhbnNwYXJlbnRSZWdpb25TY3JlZW4uY2xlYXIoKTsKLSAgICBtVHJhbnNmb3JtZWQgPSB0cnVlOwotICAgIG1DYW5Vc2VDb3B5Qml0ID0gZmFsc2U7Ci0gICAgY29weWJpdF9kZXZpY2VfdCogY29weWJpdCA9IG1GbGluZ2VyLT5nZXRCbGl0RW5naW5lKCk7Ci0gICAgaWYgKGNvcHliaXQpIHsgCi0gICAgICAgIG1DYW5Vc2VDb3B5Qml0ID0gdHJ1ZTsKLSAgICB9Ci19Ci0KLXZvaWQgTGF5ZXJPcmllbnRhdGlvbkFuaW06Om9uT3JpZW50YXRpb25Db21wbGV0ZWQoKQotewotICAgIG1GaW5pc2hUaW1lID0gc3lzdGVtVGltZSgpOwotICAgIG1PcmllbnRhdGlvbkNvbXBsZXRlZCA9IHRydWU7Ci0gICAgbUZpcnN0UmVkcmF3ID0gdHJ1ZTsKLSAgICBtTmVlZHNCbGVuZGluZyA9IHRydWU7Ci0gICAgbUZsaW5nZXItPmludmFsaWRhdGVMYXllclZpc2liaWxpdHkodGhpcyk7Ci19Ci0KLXZvaWQgTGF5ZXJPcmllbnRhdGlvbkFuaW06Om9uRHJhdyhjb25zdCBSZWdpb24mIGNsaXApIGNvbnN0Ci17Ci0gICAgLy8gQW5pbWF0aW9uLi4uCi0gICAgY29uc3QgZmxvYXQgTUlOX1NDQUxFID0gMC41ZjsKLSAgICBjb25zdCBmbG9hdCBEVVJBVElPTiA9IG1zMm5zKDIwMCk7Ci0gICAgY29uc3QgZmxvYXQgQk9VTkNFU19QRVJfU0VDT05EID0gMS42MThmOwotICAgIGNvbnN0IGZsb2F0IEJPVU5DRVNfQU1QTElUVURFID0gMS4wZi8zMi4wZjsKLQotICAgIGNvbnN0IG5zZWNzX3Qgbm93ID0gc3lzdGVtVGltZSgpOwotICAgIGZsb2F0IHNjYWxlLCBhbHBoYTsKLSAgICAKLSAgICBpZiAobU9yaWVudGF0aW9uQ29tcGxldGVkKSB7Ci0gICAgICAgIGlmIChtRmlyc3RSZWRyYXcpIHsKLSAgICAgICAgICAgIG1GaXJzdFJlZHJhdyA9IGZhbHNlOwotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyBtYWtlIGEgY29weSBvZiB3aGF0J3Mgb24gc2NyZWVuCi0gICAgICAgICAgICBjb3B5Yml0X2ltYWdlX3QgaW1hZ2U7Ci0gICAgICAgICAgICBtQml0bWFwSW4uZ2V0Qml0bWFwU3VyZmFjZSgmaW1hZ2UpOwotICAgICAgICAgICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiBodyhncmFwaGljUGxhbmUoMCkuZGlzcGxheUhhcmR3YXJlKCkpOwotICAgICAgICAgICAgaHcuY29weUJhY2tUb0ltYWdlKGltYWdlKTsKLQotICAgICAgICAgICAgLy8gYW5kIGVyYXNlIHRoZSBzY3JlZW4gZm9yIHRoaXMgcm91bmQKLSAgICAgICAgICAgIGdsRGlzYWJsZShHTF9CTEVORCk7Ci0gICAgICAgICAgICBnbERpc2FibGUoR0xfRElUSEVSKTsKLSAgICAgICAgICAgIGdsRGlzYWJsZShHTF9TQ0lTU09SX1RFU1QpOwotICAgICAgICAgICAgZ2xDbGVhckNvbG9yKDAsMCwwLDApOwotICAgICAgICAgICAgZ2xDbGVhcihHTF9DT0xPUl9CVUZGRVJfQklUKTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8gRklYTUU6IGNvZGUgYmVsb3cgaXMgZ3Jvc3MKLSAgICAgICAgICAgIG1OZWVkc0JsZW5kaW5nID0gZmFsc2U7Ci0gICAgICAgICAgICBMYXllck9yaWVudGF0aW9uQW5pbSogc2VsZihjb25zdF9jYXN0PExheWVyT3JpZW50YXRpb25BbmltKj4odGhpcykpOwotICAgICAgICAgICAgbUZsaW5nZXItPmludmFsaWRhdGVMYXllclZpc2liaWxpdHkoc2VsZik7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBtYWtlIHN1cmUgcGljay11cCB3aGVyZSB3ZSBsZWZ0IG9mZgotICAgICAgICBjb25zdCBmbG9hdCBkdXJhdGlvbiA9IERVUkFUSU9OICogbUxhc3ROb3JtYWxpemVkVGltZTsKLSAgICAgICAgY29uc3QgZmxvYXQgbm9ybWFsaXplZFRpbWUgPSAoZmxvYXQobm93IC0gbUZpbmlzaFRpbWUpIC8gZHVyYXRpb24pOwotICAgICAgICBpZiAobm9ybWFsaXplZFRpbWUgPD0gMS4wZikgewotICAgICAgICAgICAgY29uc3QgZmxvYXQgc3F1YXJlZFRpbWUgPSBub3JtYWxpemVkVGltZSpub3JtYWxpemVkVGltZTsKLSAgICAgICAgICAgIHNjYWxlID0gKDEuMGYgLSBtTGFzdFNjYWxlKSpzcXVhcmVkVGltZSArIG1MYXN0U2NhbGU7Ci0gICAgICAgICAgICBhbHBoYSA9ICgxLjBmIC0gbm9ybWFsaXplZFRpbWUpOwotICAgICAgICAgICAgYWxwaGEgKj0gYWxwaGE7Ci0gICAgICAgICAgICBhbHBoYSAqPSBhbHBoYTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG1BbmltLT5vbkFuaW1hdGlvbkZpbmlzaGVkKCk7Ci0gICAgICAgICAgICBzY2FsZSA9IDEuMGY7Ci0gICAgICAgICAgICBhbHBoYSA9IDAuMGY7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBjb25zdCBmbG9hdCBub3JtYWxpemVkVGltZSA9IGZsb2F0KG5vdyAtIG1TdGFydFRpbWUpIC8gRFVSQVRJT047Ci0gICAgICAgIGlmIChub3JtYWxpemVkVGltZSA8PSAxLjBmKSB7Ci0gICAgICAgICAgICBtTGFzdE5vcm1hbGl6ZWRUaW1lID0gbm9ybWFsaXplZFRpbWU7Ci0gICAgICAgICAgICBjb25zdCBmbG9hdCBzcXVhcmVkVGltZSA9IG5vcm1hbGl6ZWRUaW1lKm5vcm1hbGl6ZWRUaW1lOwotICAgICAgICAgICAgc2NhbGUgPSAoTUlOX1NDQUxFLTEuMGYpKnNxdWFyZWRUaW1lICsgMS4wZjsKLSAgICAgICAgICAgIGFscGhhID0gMS4wZjsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG1MYXN0Tm9ybWFsaXplZFRpbWUgPSAxLjBmOwotICAgICAgICAgICAgY29uc3QgZmxvYXQgdG9fc2Vjb25kcyA9IERVUkFUSU9OIC8gc2Vjb25kcygxKTsKLSAgICAgICAgICAgIGNvbnN0IGZsb2F0IHBoaSA9IEJPVU5DRVNfUEVSX1NFQ09ORCAqIAotICAgICAgICAgICAgICAgICAgICAoKChub3JtYWxpemVkVGltZSAtIDEuMGYpICogdG9fc2Vjb25kcykqTV9QSSoyKTsKLSAgICAgICAgICAgIHNjYWxlID0gTUlOX1NDQUxFICsgQk9VTkNFU19BTVBMSVRVREUgKiAoMS4wZiAtIGNvc2YocGhpKSk7Ci0gICAgICAgICAgICBhbHBoYSA9IDEuMGY7Ci0gICAgICAgIH0KLSAgICAgICAgbUxhc3RTY2FsZSA9IHNjYWxlOwotICAgIH0KLSAgICBkcmF3U2NhbGVkKHNjYWxlLCBhbHBoYSk7Ci19Ci0KLXZvaWQgTGF5ZXJPcmllbnRhdGlvbkFuaW06OmRyYXdTY2FsZWQoZmxvYXQgZiwgZmxvYXQgYWxwaGEpIGNvbnN0Ci17Ci0gICAgY29weWJpdF9pbWFnZV90IGRzdDsKLSAgICBjb25zdCBHcmFwaGljUGxhbmUmIHBsYW5lKGdyYXBoaWNQbGFuZSgwKSk7Ci0gICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiBodyhwbGFuZS5kaXNwbGF5SGFyZHdhcmUoKSk7Ci0gICAgaHcuZ2V0RGlzcGxheVN1cmZhY2UoJmRzdCk7Ci0KLSAgICAvLyBjbGVhciBzY3JlZW4KLSAgICAvLyBUT0RPOiB3aXRoIHVwZGF0ZSBvbiBkZW1hbmQsIHdlIG1heSBiZSBhYmxlIAotICAgIC8vIHRvIG5vdCBlcmFzZSB0aGUgc2NyZWVuIGF0IGFsbCBkdXJpbmcgdGhlIGFuaW1hdGlvbiAKLSAgICBpZiAoIW1PcmllbnRhdGlvbkNvbXBsZXRlZCkgewotICAgICAgICBnbERpc2FibGUoR0xfQkxFTkQpOwotICAgICAgICBnbERpc2FibGUoR0xfRElUSEVSKTsKLSAgICAgICAgZ2xEaXNhYmxlKEdMX1NDSVNTT1JfVEVTVCk7Ci0gICAgICAgIGdsQ2xlYXJDb2xvcigwLDAsMCwwKTsKLSAgICAgICAgZ2xDbGVhcihHTF9DT0xPUl9CVUZGRVJfQklUKTsKLSAgICB9Ci0gICAgCi0gICAgY29uc3QgaW50IHcgPSBkc3QudypmOyAKLSAgICBjb25zdCBpbnQgaCA9IGRzdC5oKmY7IAotICAgIGNvbnN0IGludCB4YyA9IHVpbnQzMl90KGRzdC53LXcpLzI7Ci0gICAgY29uc3QgaW50IHljID0gdWludDMyX3QoZHN0LmgtaCkvMjsKLSAgICBjb25zdCBjb3B5Yml0X3JlY3RfdCBkcmVjdCA9IHsgeGMsIHljLCB4Yyt3LCB5YytoIH07IAotCi0gICAgY29weWJpdF9pbWFnZV90IHNyYzsKLSAgICBtQml0bWFwLmdldEJpdG1hcFN1cmZhY2UoJnNyYyk7Ci0gICAgY29uc3QgY29weWJpdF9yZWN0X3Qgc3JlY3QgPSB7IDAsIDAsIHNyYy53LCBzcmMuaCB9OwotCi0gICAgaW50IGVyciA9IE5PX0VSUk9SOwotICAgIGNvbnN0IGludCBjYW5fdXNlX2NvcHliaXQgPSBjYW5Vc2VDb3B5Yml0KCk7Ci0gICAgaWYgKGNhbl91c2VfY29weWJpdCkgIHsKLSAgICAgICAgY29weWJpdF9kZXZpY2VfdCogY29weWJpdCA9IG1GbGluZ2VyLT5nZXRCbGl0RW5naW5lKCk7Ci0gICAgICAgIGNvcHliaXQtPnNldF9wYXJhbWV0ZXIoY29weWJpdCwgQ09QWUJJVF9UUkFOU0ZPUk0sIDApOwotICAgICAgICBjb3B5Yml0LT5zZXRfcGFyYW1ldGVyKGNvcHliaXQsIENPUFlCSVRfRElUSEVSLCBDT1BZQklUX0VOQUJMRSk7Ci0KLSAgICAgICAgaWYgKGFscGhhIDwgMS4wZikgewotICAgICAgICAgICAgY29weWJpdF9pbWFnZV90IHNyY0luOwotICAgICAgICAgICAgbUJpdG1hcEluLmdldEJpdG1hcFN1cmZhY2UoJnNyY0luKTsKLSAgICAgICAgICAgIHJlZ2lvbl9pdGVyYXRvciBpdChSZWdpb24oUmVjdCggZHJlY3QubCwgZHJlY3QudCwgZHJlY3QuciwgZHJlY3QuYiApKSk7Ci0gICAgICAgICAgICBjb3B5Yml0LT5zZXRfcGFyYW1ldGVyKGNvcHliaXQsIENPUFlCSVRfUExBTkVfQUxQSEEsIDB4RkYpOwotICAgICAgICAgICAgZXJyID0gY29weWJpdC0+c3RyZXRjaChjb3B5Yml0LCAmZHN0LCAmc3JjSW4sICZkcmVjdCwgJnNyZWN0LCAmaXQpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKCFlcnIgJiYgYWxwaGEgPiAwLjBmKSB7Ci0gICAgICAgICAgICByZWdpb25faXRlcmF0b3IgaXQoUmVnaW9uKFJlY3QoIGRyZWN0LmwsIGRyZWN0LnQsIGRyZWN0LnIsIGRyZWN0LmIgKSkpOwotICAgICAgICAgICAgY29weWJpdC0+c2V0X3BhcmFtZXRlcihjb3B5Yml0LCBDT1BZQklUX1BMQU5FX0FMUEhBLCBpbnQoYWxwaGEqMjU1KSk7Ci0gICAgICAgICAgICBlcnIgPSBjb3B5Yml0LT5zdHJldGNoKGNvcHliaXQsICZkc3QsICZzcmMsICZkcmVjdCwgJnNyZWN0LCAmaXQpOwotICAgICAgICB9Ci0gICAgICAgIExPR0VfSUYoZXJyICE9IE5PX0VSUk9SLCAiY29weWJpdCBmYWlsZWQgKCVzKSIsIHN0cmVycm9yKGVycikpOwotICAgIH0KLSAgICBpZiAoIWNhbl91c2VfY29weWJpdCB8fCBlcnIpIHsgICAKLSAgICAgICAgR0dMU3VyZmFjZSB0OwotICAgICAgICB0LnZlcnNpb24gPSBzaXplb2YoR0dMU3VyZmFjZSk7Ci0gICAgICAgIHQud2lkdGggID0gc3JjLnc7Ci0gICAgICAgIHQuaGVpZ2h0ID0gc3JjLmg7Ci0gICAgICAgIHQuc3RyaWRlID0gc3JjLnc7Ci0gICAgICAgIHQudnN0cmlkZT0gc3JjLmg7Ci0gICAgICAgIHQuZm9ybWF0ID0gc3JjLmZvcm1hdDsKLSAgICAgICAgdC5kYXRhID0gKEdHTHVieXRlKikoaW50cHRyX3Qoc3JjLmJhc2UpICsgc3JjLm9mZnNldCk7Ci0KLSAgICAgICAgVHJhbnNmb3JtIHRyOwotICAgICAgICB0ci5zZXQoZiwwLDAsZik7Ci0gICAgICAgIHRyLnNldCh4YywgeWMpOwotICAgICAgICAKLSAgICAgICAgLy8gRklYTUU6IHdlIHNob3VsZCBub3QgYWNjZXNzIG1WZXJ0aWNlcyBhbmQgbURyYXdpbmdTdGF0ZSBsaWtlIHRoYXQsCi0gICAgICAgIC8vIGJ1dCBzaW5jZSB3ZSBjb250cm9sIHRoZSBhbmltYXRpb24sIHdlIGtub3cgaXQncyBnb2luZyB0byB3b3JrIG9rYXkuCi0gICAgICAgIC8vIGV2ZW50dWFsbHkgd2UnZCBuZWVkIGEgbW9yZSBmb3JtYWwgd2F5IG9mIGRvaW5nIHRoaW5ncyBsaWtlIHRoaXMuCi0gICAgICAgIExheWVyT3JpZW50YXRpb25BbmltJiBzZWxmKGNvbnN0X2Nhc3Q8TGF5ZXJPcmllbnRhdGlvbkFuaW0mPigqdGhpcykpOwotICAgICAgICB0ci50cmFuc2Zvcm0oc2VsZi5tVmVydGljZXNbMF0sIDAsIDApOwotICAgICAgICB0ci50cmFuc2Zvcm0oc2VsZi5tVmVydGljZXNbMV0sIDAsIHNyYy5oKTsKLSAgICAgICAgdHIudHJhbnNmb3JtKHNlbGYubVZlcnRpY2VzWzJdLCBzcmMudywgc3JjLmgpOwotICAgICAgICB0ci50cmFuc2Zvcm0oc2VsZi5tVmVydGljZXNbM10sIHNyYy53LCAwKTsKLSAgICAgICAgaWYgKCEobUZsYWdzICYgRGlzcGxheUhhcmR3YXJlOjpTTE9XX0NPTkZJRykpIHsKLSAgICAgICAgICAgIC8vIFRvbyBzbG93IHRvIGRvIHRoaXMgaW4gc29mdHdhcmUKLSAgICAgICAgICAgIHNlbGYubURyYXdpbmdTdGF0ZS5mbGFncyB8PSBJU3VyZmFjZUNvbXBvc2VyOjplTGF5ZXJGaWx0ZXI7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoYWxwaGEgPCAxLjBmKSB7Ci0gICAgICAgICAgICBjb3B5Yml0X2ltYWdlX3Qgc3JjOwotICAgICAgICAgICAgbUJpdG1hcEluLmdldEJpdG1hcFN1cmZhY2UoJnNyYyk7Ci0gICAgICAgICAgICB0LmRhdGEgPSAoR0dMdWJ5dGUqKShpbnRwdHJfdChzcmMuYmFzZSkgKyBzcmMub2Zmc2V0KTsKLSAgICAgICAgICAgIGlmIChVTkxJS0VMWShtVGV4dHVyZU5hbWVJbiA9PSAtMUxVKSkgewotICAgICAgICAgICAgICAgIG1UZXh0dXJlTmFtZUluID0gY3JlYXRlVGV4dHVyZSgpOwotICAgICAgICAgICAgICAgIEdMdWludCB3PTAsIGg9MDsKLSAgICAgICAgICAgICAgICBjb25zdCBSZWdpb24gZGlydHkoUmVjdCh0LndpZHRoLCB0LmhlaWdodCkpOwotICAgICAgICAgICAgICAgIGxvYWRUZXh0dXJlKGRpcnR5LCBtVGV4dHVyZU5hbWVJbiwgdCwgdywgaCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBzZWxmLm1EcmF3aW5nU3RhdGUuYWxwaGEgPSAyNTU7Ci0gICAgICAgICAgICBjb25zdCBSZWdpb24gY2xpcChSZWN0KCBkcmVjdC5sLCBkcmVjdC50LCBkcmVjdC5yLCBkcmVjdC5iICkpOwotICAgICAgICAgICAgZHJhd1dpdGhPcGVuR0woY2xpcCwgbVRleHR1cmVOYW1lLCB0KTsKLSAgICAgICAgfQotCi0gICAgICAgIHQuZGF0YSA9IChHR0x1Ynl0ZSopKGludHB0cl90KHNyYy5iYXNlKSArIHNyYy5vZmZzZXQpOwotICAgICAgICBpZiAoVU5MSUtFTFkobVRleHR1cmVOYW1lID09IC0xTFUpKSB7Ci0gICAgICAgICAgICBtVGV4dHVyZU5hbWUgPSBjcmVhdGVUZXh0dXJlKCk7Ci0gICAgICAgICAgICBHTHVpbnQgdz0wLCBoPTA7Ci0gICAgICAgICAgICBjb25zdCBSZWdpb24gZGlydHkoUmVjdCh0LndpZHRoLCB0LmhlaWdodCkpOwotICAgICAgICAgICAgbG9hZFRleHR1cmUoZGlydHksIG1UZXh0dXJlTmFtZSwgdCwgdywgaCk7Ci0gICAgICAgIH0KLSAgICAgICAgc2VsZi5tRHJhd2luZ1N0YXRlLmFscGhhID0gaW50KGFscGhhKjI1NSk7Ci0gICAgICAgIGNvbnN0IFJlZ2lvbiBjbGlwKFJlY3QoIGRyZWN0LmwsIGRyZWN0LnQsIGRyZWN0LnIsIGRyZWN0LmIgKSk7Ci0gICAgICAgIGRyYXdXaXRoT3BlbkdMKGNsaXAsIG1UZXh0dXJlTmFtZSwgdCk7Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJPcmllbnRhdGlvbkFuaW0uaCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvTGF5ZXJPcmllbnRhdGlvbkFuaW0uaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzM2NzY4NS4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL0xheWVyT3JpZW50YXRpb25BbmltLmgKKysrIC9kZXYvbnVsbApAQCAtMSw3NSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0xBWUVSX09SSUVOVEFUSU9OX0FOSU1fSAotI2RlZmluZSBBTkRST0lEX0xBWUVSX09SSUVOVEFUSU9OX0FOSU1fSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotCi0jaW5jbHVkZSAiTGF5ZXJCYXNlLmgiCi0jaW5jbHVkZSAiTGF5ZXJCaXRtYXAuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLWNsYXNzIE9yaWVudGF0aW9uQW5pbWF0aW9uOwotCi1jbGFzcyBMYXllck9yaWVudGF0aW9uQW5pbSA6IHB1YmxpYyBMYXllckJhc2UKLXsKLXB1YmxpYzogICAgCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IHR5cGVJbmZvOwotICAgIHN0YXRpYyBjb25zdCBjaGFyKiBjb25zdCB0eXBlSUQ7Ci0gICAgdmlydHVhbCBjaGFyIGNvbnN0KiBnZXRUeXBlSUQoKSBjb25zdCB7IHJldHVybiB0eXBlSUQ7IH0KLSAgICB2aXJ0dWFsIHVpbnQzMl90IGdldFR5cGVJbmZvKCkgY29uc3QgeyByZXR1cm4gdHlwZUluZm87IH0KLSAgICAKLSAgICAgICAgICAgICAgICBMYXllck9yaWVudGF0aW9uQW5pbShTdXJmYWNlRmxpbmdlciogZmxpbmdlciwgRGlzcGxheUlEIGRpc3BsYXksCi0gICAgICAgICAgICAgICAgICAgICAgICBPcmllbnRhdGlvbkFuaW1hdGlvbiogYW5pbSwgCi0gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBMYXllckJpdG1hcCYgem9vbU91dCwKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IExheWVyQml0bWFwJiB6b29tSW4pOwotICAgICAgICB2aXJ0dWFsIH5MYXllck9yaWVudGF0aW9uQW5pbSgpOwotCi0gICAgICAgICAgICB2b2lkIG9uT3JpZW50YXRpb25Db21wbGV0ZWQoKTsKLQotICAgIHZpcnR1YWwgdm9pZCBvbkRyYXcoY29uc3QgUmVnaW9uJiBjbGlwKSBjb25zdDsKLSAgICB2aXJ0dWFsIFBvaW50IGdldFBoeXNpY2FsU2l6ZSgpIGNvbnN0OwotICAgIHZpcnR1YWwgdm9pZCB2YWxpZGF0ZVZpc2liaWxpdHkoY29uc3QgVHJhbnNmb3JtJiBnbG9iYWxUcmFuc2Zvcm0pOwotICAgIHZpcnR1YWwgYm9vbCBuZWVkc0JsZW5kaW5nKCkgY29uc3Q7Ci0gICAgdmlydHVhbCBib29sIGlzU2VjdXJlKCkgY29uc3QgICAgICAgeyByZXR1cm4gZmFsc2U7IH0KLXByaXZhdGU6Ci0gICAgdm9pZCBkcmF3U2NhbGVkKGZsb2F0IHNjYWxlLCBmbG9hdCBhbHBoYSkgY29uc3Q7Ci0KLSAgICBPcmllbnRhdGlvbkFuaW1hdGlvbiogbUFuaW07Ci0gICAgTGF5ZXJCaXRtYXAgbUJpdG1hcDsKLSAgICBMYXllckJpdG1hcCBtQml0bWFwSW47Ci0gICAgbnNlY3NfdCBtU3RhcnRUaW1lOwotICAgIG5zZWNzX3QgbUZpbmlzaFRpbWU7Ci0gICAgYm9vbCBtT3JpZW50YXRpb25Db21wbGV0ZWQ7Ci0gICAgbXV0YWJsZSBib29sIG1GaXJzdFJlZHJhdzsKLSAgICBtdXRhYmxlIGZsb2F0IG1MYXN0Tm9ybWFsaXplZFRpbWU7Ci0gICAgbXV0YWJsZSBmbG9hdCBtTGFzdFNjYWxlOwotICAgIG11dGFibGUgR0x1aW50ICBtVGV4dHVyZU5hbWU7Ci0gICAgbXV0YWJsZSBHTHVpbnQgIG1UZXh0dXJlTmFtZUluOwotICAgIG11dGFibGUgYm9vbCBtTmVlZHNCbGVuZGluZzsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9MQVlFUl9PUklFTlRBVElPTl9BTklNX0gKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvTU9EVUxFX0xJQ0VOU0VfQVBBQ0hFMiBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvTU9EVUxFX0xJQ0VOU0VfQVBBQ0hFMgpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZTY5ZGUyOS4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL01PRFVMRV9MSUNFTlNFX0FQQUNIRTIKKysrIC9kZXYvbnVsbApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9PcmllbnRhdGlvbkFuaW1hdGlvbi5jcHAgYi9saWJzL3N1cmZhY2VmbGluZ2VyL09yaWVudGF0aW9uQW5pbWF0aW9uLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZjZmMTMyNi4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL09yaWVudGF0aW9uQW5pbWF0aW9uLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDE1NSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJTdXJmYWNlRmxpbmdlciIKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPGxpbWl0cy5oPgotCi0jaW5jbHVkZSAiTGF5ZXJPcmllbnRhdGlvbkFuaW0uaCIKLSNpbmNsdWRlICJPcmllbnRhdGlvbkFuaW1hdGlvbi5oIgotI2luY2x1ZGUgIlN1cmZhY2VGbGluZ2VyLmgiCi0jaW5jbHVkZSAiVlJhbUhlYXAuaCIKLQotI2luY2x1ZGUgIkRpc3BsYXlIYXJkd2FyZS9EaXNwbGF5SGFyZHdhcmUuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotT3JpZW50YXRpb25BbmltYXRpb246Ok9yaWVudGF0aW9uQW5pbWF0aW9uKGNvbnN0IHNwPFN1cmZhY2VGbGluZ2VyPiYgZmxpbmdlcikKLSAgICA6IG1GbGluZ2VyKGZsaW5nZXIpLCBtTGF5ZXJPcmllbnRhdGlvbkFuaW0oTlVMTCksIG1TdGF0ZShET05FKQotewotICAgIC8vIGFsbG9jYXRlIGEgbWVtb3J5LWRlYWxlciBmb3IgdGhpcyB0aGUgZmlyc3QgdGltZQotICAgIG1UZW1wb3JhcnlEZWFsZXIgPSBtRmxpbmdlci0+Z2V0U3VyZmFjZUhlYXBNYW5hZ2VyKCktPmNyZWF0ZUhlYXAoCi0gICAgICAgICAgICBJU3VyZmFjZUNvbXBvc2VyOjplSGFyZHdhcmUpOwotfQotCi1PcmllbnRhdGlvbkFuaW1hdGlvbjo6fk9yaWVudGF0aW9uQW5pbWF0aW9uKCkKLXsKLX0KLQotdm9pZCBPcmllbnRhdGlvbkFuaW1hdGlvbjo6b25PcmllbnRhdGlvbkNoYW5nZWQoKQotewotICAgIGlmIChtU3RhdGUgPT0gRE9ORSkKLSAgICAgICAgbVN0YXRlID0gUFJFUEFSRTsKLX0KLQotdm9pZCBPcmllbnRhdGlvbkFuaW1hdGlvbjo6b25BbmltYXRpb25GaW5pc2hlZCgpCi17Ci0gICAgaWYgKG1TdGF0ZSAhPSBET05FKQotICAgICAgICBtU3RhdGUgPSBGSU5JU0g7Ci19Ci0KLWJvb2wgT3JpZW50YXRpb25BbmltYXRpb246OnJ1bl9pbXBsKCkKLXsKLSAgICBib29sIHNraXBfZnJhbWU7Ci0gICAgc3dpdGNoIChtU3RhdGUpIHsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgY2FzZSBET05FOgotICAgICAgICAgICAgc2tpcF9mcmFtZSA9IGRvbmUoKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFBSRVBBUkU6Ci0gICAgICAgICAgICBza2lwX2ZyYW1lID0gcHJlcGFyZSgpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgUEhBU0UxOgotICAgICAgICAgICAgc2tpcF9mcmFtZSA9IHBoYXNlMSgpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgUEhBU0UyOgotICAgICAgICAgICAgc2tpcF9mcmFtZSA9IHBoYXNlMigpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgRklOSVNIOgotICAgICAgICAgICAgc2tpcF9mcmFtZSA9IGZpbmlzaGVkKCk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICB9Ci0gICAgcmV0dXJuIHNraXBfZnJhbWU7Ci19Ci0KLWJvb2wgT3JpZW50YXRpb25BbmltYXRpb246OmRvbmUoKQotewotICAgIGlmIChtRmxpbmdlci0+aXNGcm96ZW4oKSkgewotICAgICAgICAvLyB3ZSBhcmUgbm90IGFsbG93ZWQgdG8gZHJhdywgYnV0IHBhdXNlIGEgYml0IHRvIG1ha2Ugc3VyZQotICAgICAgICAvLyBhcHBzIGRvbid0IGVuZCB1cCB1c2luZyB0aGUgd2hvbGUgQ1BVLCBpZiB0aGV5IGRlcGVuZCBvbgotICAgICAgICAvLyBzdXJmYWNlZmxpbmdlciBmb3Igc3luY2hyb25pemF0aW9uLgotICAgICAgICB1c2xlZXAoODMzMyk7IC8vIDguM21zIH4gMTIwZnBzCi0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0KLWJvb2wgT3JpZW50YXRpb25BbmltYXRpb246OnByZXBhcmUoKQotewotICAgIG1TdGF0ZSA9IFBIQVNFMTsKLSAgICAKLSAgICBjb25zdCBHcmFwaGljUGxhbmUmIHBsYW5lKG1GbGluZ2VyLT5ncmFwaGljUGxhbmUoMCkpOwotICAgIGNvbnN0IERpc3BsYXlIYXJkd2FyZSYgaHcocGxhbmUuZGlzcGxheUhhcmR3YXJlKCkpOwotICAgIGNvbnN0IHVpbnQzMl90IHcgPSBody5nZXRXaWR0aCgpOwotICAgIGNvbnN0IHVpbnQzMl90IGggPSBody5nZXRIZWlnaHQoKTsKLQotICAgIExheWVyQml0bWFwIGJpdG1hcDsKLSAgICBiaXRtYXAuaW5pdChtVGVtcG9yYXJ5RGVhbGVyKTsKLSAgICBiaXRtYXAuc2V0Qml0cyh3LCBoLCAxLCBody5nZXRGb3JtYXQoKSk7Ci0KLSAgICBMYXllckJpdG1hcCBiaXRtYXBJbjsKLSAgICBiaXRtYXBJbi5pbml0KG1UZW1wb3JhcnlEZWFsZXIpOwotICAgIGJpdG1hcEluLnNldEJpdHModywgaCwgMSwgaHcuZ2V0Rm9ybWF0KCkpOwotCi0gICAgY29weWJpdF9pbWFnZV90IGZyb250OwotICAgIGJpdG1hcC5nZXRCaXRtYXBTdXJmYWNlKCZmcm9udCk7Ci0gICAgaHcuY29weUZyb250VG9JbWFnZShmcm9udCk7Ci0KLSAgICBMYXllck9yaWVudGF0aW9uQW5pbSogbCA9IG5ldyBMYXllck9yaWVudGF0aW9uQW5pbSgKLSAgICAgICAgICAgIG1GbGluZ2VyLmdldCgpLCAwLCB0aGlzLCBiaXRtYXAsIGJpdG1hcEluKTsKLSAgICBsLT5pbml0U3RhdGVzKHcsIGgsIDApOwotICAgIGwtPnNldExheWVyKElOVF9NQVgtMSk7Ci0gICAgbUZsaW5nZXItPmFkZExheWVyKGwpOwotICAgIG1MYXllck9yaWVudGF0aW9uQW5pbSA9IGw7Ci0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLWJvb2wgT3JpZW50YXRpb25BbmltYXRpb246OnBoYXNlMSgpCi17Ci0gICAgaWYgKG1GbGluZ2VyLT5pc0Zyb3plbigpID09IGZhbHNlKSB7Ci0gICAgICAgIC8vIHN0YXJ0IHBoYXNlIDIKLSAgICAgICAgbVN0YXRlID0gUEhBU0UyOwotICAgICAgICBtTGF5ZXJPcmllbnRhdGlvbkFuaW0tPm9uT3JpZW50YXRpb25Db21wbGV0ZWQoKTsKLSAgICAgICAgbUxheWVyT3JpZW50YXRpb25BbmltLT5pbnZhbGlkYXRlKCk7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAKLSAgICB9Ci0gICAgbUxheWVyT3JpZW50YXRpb25BbmltLT5pbnZhbGlkYXRlKCk7Ci0gICAgcmV0dXJuIGZhbHNlOwotfQotCi1ib29sIE9yaWVudGF0aW9uQW5pbWF0aW9uOjpwaGFzZTIoKQotewotICAgIC8vIGRvIHRoZSAybmQgcGhhc2Ugb2YgdGhlIGFuaW1hdGlvbgotICAgIG1MYXllck9yaWVudGF0aW9uQW5pbS0+aW52YWxpZGF0ZSgpOwotICAgIHJldHVybiBmYWxzZTsKLX0KLQotYm9vbCBPcmllbnRhdGlvbkFuaW1hdGlvbjo6ZmluaXNoZWQoKQotewotICAgIG1TdGF0ZSA9IERPTkU7Ci0gICAgbUZsaW5nZXItPnJlbW92ZUxheWVyKG1MYXllck9yaWVudGF0aW9uQW5pbSk7Ci0gICAgbUxheWVyT3JpZW50YXRpb25BbmltID0gTlVMTDsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL09yaWVudGF0aW9uQW5pbWF0aW9uLmggYi9saWJzL3N1cmZhY2VmbGluZ2VyL09yaWVudGF0aW9uQW5pbWF0aW9uLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGJhMzNmY2UuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9PcmllbnRhdGlvbkFuaW1hdGlvbi5oCisrKyAvZGV2L251bGwKQEAgLTEsNzMgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9PUklFTlRBVElPTl9BTklNQVRJT05fSAotI2RlZmluZSBBTkRST0lEX09SSUVOVEFUSU9OX0FOSU1BVElPTl9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgIlN1cmZhY2VGbGluZ2VyLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIFN1cmZhY2VGbGluZ2VyOwotY2xhc3MgTWVtb3J5RGVhbGVyOwotY2xhc3MgTGF5ZXJPcmllbnRhdGlvbkFuaW07Ci0KLWNsYXNzIE9yaWVudGF0aW9uQW5pbWF0aW9uCi17Ci1wdWJsaWM6ICAgIAotICAgICAgICAgICAgICAgICBPcmllbnRhdGlvbkFuaW1hdGlvbihjb25zdCBzcDxTdXJmYWNlRmxpbmdlcj4mIGZsaW5nZXIpOwotICAgICAgICB2aXJ0dWFsIH5PcmllbnRhdGlvbkFuaW1hdGlvbigpOwotCi0gICB2b2lkIG9uT3JpZW50YXRpb25DaGFuZ2VkKCk7Ci0gICB2b2lkIG9uQW5pbWF0aW9uRmluaXNoZWQoKTsKLSAgIGlubGluZSBib29sIHJ1bigpIHsKLSAgICAgICBpZiAoTElLRUxZKG1TdGF0ZSA9PSBET05FKSkKLSAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgIHJldHVybiBydW5faW1wbCgpOwotICAgfQotCi1wcml2YXRlOgotICAgIGVudW0gewotICAgICAgICBET05FID0gMCwKLSAgICAgICAgUFJFUEFSRSwKLSAgICAgICAgUEhBU0UxLAotICAgICAgICBQSEFTRTIsCi0gICAgICAgIEZJTklTSAotICAgIH07Ci0KLSAgICBib29sIHJ1bl9pbXBsKCk7Ci0gICAgYm9vbCBkb25lKCk7Ci0gICAgYm9vbCBwcmVwYXJlKCk7Ci0gICAgYm9vbCBwaGFzZTEoKTsKLSAgICBib29sIHBoYXNlMigpOwotICAgIGJvb2wgZmluaXNoZWQoKTsKLQotICAgIHNwPFN1cmZhY2VGbGluZ2VyPiBtRmxpbmdlcjsKLSAgICBzcDxNZW1vcnlEZWFsZXI+IG1UZW1wb3JhcnlEZWFsZXI7Ci0gICAgTGF5ZXJPcmllbnRhdGlvbkFuaW0qIG1MYXllck9yaWVudGF0aW9uQW5pbTsKLSAgICBpbnQgbVN0YXRlOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX09SSUVOVEFUSU9OX0FOSU1BVElPTl9ICmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL1N1cmZhY2VGbGluZ2VyLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvU3VyZmFjZUZsaW5nZXIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5MDAyODJhLi4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvU3VyZmFjZUZsaW5nZXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTg0MCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJTdXJmYWNlRmxpbmdlciIKLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8ZmNudGwuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPG1hdGguaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDxzeXMvc3RhdC5oPgotI2luY2x1ZGUgPHN5cy9pb2N0bC5oPgotCi0jaW5jbHVkZSA8Y3V0aWxzL2xvZy5oPgotI2luY2x1ZGUgPGN1dGlscy9wcm9wZXJ0aWVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL0lTZXJ2aWNlTWFuYWdlci5oPgotI2luY2x1ZGUgPHV0aWxzL01lbW9yeURlYWxlci5oPgotI2luY2x1ZGUgPHV0aWxzL01lbW9yeUJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RyaW5nMTYuaD4KLSNpbmNsdWRlIDx1dGlscy9TdG9wV2F0Y2guaD4KLQotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0jaW5jbHVkZSA8dWkvRGlzcGxheUluZm8uaD4KLSNpbmNsdWRlIDx1aS9FR0xEaXNwbGF5U3VyZmFjZS5oPgotCi0jaW5jbHVkZSA8cGl4ZWxmbGluZ2VyL3BpeGVsZmxpbmdlci5oPgotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLQotI2luY2x1ZGUgImNsei5oIgotI2luY2x1ZGUgIkNQVUdhdWdlLmgiCi0jaW5jbHVkZSAiTGF5ZXIuaCIKLSNpbmNsdWRlICJMYXllckJsdXIuaCIKLSNpbmNsdWRlICJMYXllckJ1ZmZlci5oIgotI2luY2x1ZGUgIkxheWVyRGltLmgiCi0jaW5jbHVkZSAiTGF5ZXJCaXRtYXAuaCIKLSNpbmNsdWRlICJMYXllck9yaWVudGF0aW9uQW5pbS5oIgotI2luY2x1ZGUgIk9yaWVudGF0aW9uQW5pbWF0aW9uLmgiCi0jaW5jbHVkZSAiU3VyZmFjZUZsaW5nZXIuaCIKLSNpbmNsdWRlICJWUmFtSGVhcC5oIgotCi0jaW5jbHVkZSAiRGlzcGxheUhhcmR3YXJlL0Rpc3BsYXlIYXJkd2FyZS5oIgotI2luY2x1ZGUgIkdQVUhhcmR3YXJlL0dQVUhhcmR3YXJlLmgiCi0KLQotI2RlZmluZSBESVNQTEFZX0NPVU5UICAgICAgIDEKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6aW5zdGFudGlhdGUoKSB7Ci0gICAgZGVmYXVsdFNlcnZpY2VNYW5hZ2VyKCktPmFkZFNlcnZpY2UoCi0gICAgICAgICAgICBTdHJpbmcxNigiU3VyZmFjZUZsaW5nZXIiKSwgbmV3IFN1cmZhY2VGbGluZ2VyKCkpOwotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjpzaHV0ZG93bigpIHsKLSAgICAvLyB3ZSBzaG91bGQgdW5yZWdpc3RlciBoZXJlLCBidXQgbm90IHJlYWxseSBiZWNhdXNlCi0gICAgLy8gd2hlbiAoaWYpIHRoZSBzZXJ2aWNlIG1hbmFnZXIgZ29lcyBhd2F5LCBhbGwgdGhlIHNlcnZpY2VzCi0gICAgLy8gaXQgaGFzIGEgcmVmZXJlbmNlIHRvIHdpbGwgbGVhdmUgdG9vLgotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotU3VyZmFjZUZsaW5nZXI6OkxheWVyVmVjdG9yOjpMYXllclZlY3Rvcihjb25zdCBTdXJmYWNlRmxpbmdlcjo6TGF5ZXJWZWN0b3ImIHJocykKLSAgICA6IGxvb2t1cChyaHMubG9va3VwKSwgbGF5ZXJzKHJocy5sYXllcnMpCi17Ci19Ci0KLXNzaXplX3QgU3VyZmFjZUZsaW5nZXI6OkxheWVyVmVjdG9yOjppbmRleE9mKAotICAgICAgICBMYXllckJhc2UqIGtleSwgc2l6ZV90IGd1ZXNzKSBjb25zdAotewotICAgIGlmIChndWVzczxzaXplKCkgJiYgbG9va3VwLmtleUF0KGd1ZXNzKSA9PSBrZXkpCi0gICAgICAgIHJldHVybiBndWVzczsKLSAgICBjb25zdCBzc2l6ZV90IGkgPSBsb29rdXAuaW5kZXhPZktleShrZXkpOwotICAgIGlmIChpPj0wKSB7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBpZHggPSBsb29rdXAudmFsdWVBdChpKTsKLSAgICAgICAgTE9HX0FTU0VSVChsYXllcnNbaWR4XT09a2V5LAotICAgICAgICAgICAgIkxheWVyVmVjdG9yWyVwXTogbGF5ZXJzWyVkXT0lcCwga2V5PSVwIiwKLSAgICAgICAgICAgIHRoaXMsIGludChpZHgpLCBsYXllcnNbaWR4XSwga2V5KTsKLSAgICAgICAgcmV0dXJuIGlkeDsKLSAgICB9Ci0gICAgcmV0dXJuIGk7Ci19Ci0KLXNzaXplX3QgU3VyZmFjZUZsaW5nZXI6OkxheWVyVmVjdG9yOjphZGQoCi0gICAgICAgIExheWVyQmFzZSogbGF5ZXIsCi0gICAgICAgIFZlY3RvcjxMYXllckJhc2UqPjo6Y29tcGFyX3QgY21wKQotewotICAgIHNpemVfdCBjb3VudCA9IGxheWVycy5zaXplKCk7Ci0gICAgc3NpemVfdCBsID0gMDsKLSAgICBzc2l6ZV90IGggPSBjb3VudC0xOwotICAgIHNzaXplX3QgbWlkOwotICAgIExheWVyQmFzZSogY29uc3QqIGEgPSBsYXllcnMuYXJyYXkoKTsKLSAgICB3aGlsZSAobCA8PSBoKSB7Ci0gICAgICAgIG1pZCA9IGwgKyAoaCAtIGwpLzI7Ci0gICAgICAgIGNvbnN0IGludCBjID0gY21wKGErbWlkLCAmbGF5ZXIpOwotICAgICAgICBpZiAoYyA9PSAwKSAgICAgeyBsID0gbWlkOyBicmVhazsgfQotICAgICAgICBlbHNlIGlmIChjPDApICAgeyBsID0gbWlkKzE7IH0KLSAgICAgICAgZWxzZSAgICAgICAgICAgIHsgaCA9IG1pZC0xOyB9Ci0gICAgfQotICAgIHNpemVfdCBvcmRlciA9IGw7Ci0gICAgd2hpbGUgKG9yZGVyPGNvdW50ICYmICFjbXAoJmxheWVyLCBhK29yZGVyKSkgewotICAgICAgICBvcmRlcisrOwotICAgIH0KLSAgICBjb3VudCA9IGxvb2t1cC5zaXplKCk7Ci0gICAgZm9yIChzaXplX3QgaT0wIDsgaTxjb3VudCA7IGkrKykgewotICAgICAgICBpZiAobG9va3VwLnZhbHVlQXQoaSkgPj0gb3JkZXIpIHsKLSAgICAgICAgICAgIGxvb2t1cC5lZGl0VmFsdWVBdChpKSsrOwotICAgICAgICB9Ci0gICAgfQotICAgIGxheWVycy5pbnNlcnRBdChsYXllciwgb3JkZXIpOwotICAgIGxvb2t1cC5hZGQobGF5ZXIsIG9yZGVyKTsKLSAgICByZXR1cm4gb3JkZXI7Ci19Ci0KLXNzaXplX3QgU3VyZmFjZUZsaW5nZXI6OkxheWVyVmVjdG9yOjpyZW1vdmUoTGF5ZXJCYXNlKiBsYXllcikKLXsKLSAgICBjb25zdCBzc2l6ZV90IGtleUluZGV4ID0gbG9va3VwLmluZGV4T2ZLZXkobGF5ZXIpOwotICAgIGlmIChrZXlJbmRleCA+PSAwKSB7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBpbmRleCA9IGxvb2t1cC52YWx1ZUF0KGtleUluZGV4KTsKLSAgICAgICAgTE9HX0FTU0VSVChsYXllcnNbaW5kZXhdPT1sYXllciwKLSAgICAgICAgICAgICAgICAiTGF5ZXJWZWN0b3JbJXBdOiBsYXllcnNbJXVdPSVwLCBsYXllcj0lcCIsCi0gICAgICAgICAgICAgICAgdGhpcywgaW50KGluZGV4KSwgbGF5ZXJzW2luZGV4XSwgbGF5ZXIpOwotICAgICAgICBsYXllcnMucmVtb3ZlSXRlbXNBdChpbmRleCk7Ci0gICAgICAgIGxvb2t1cC5yZW1vdmVJdGVtc0F0KGtleUluZGV4KTsKLSAgICAgICAgY29uc3Qgc2l6ZV90IGNvdW50ID0gbG9va3VwLnNpemUoKTsKLSAgICAgICAgZm9yIChzaXplX3QgaT0wIDsgaTxjb3VudCA7IGkrKykgewotICAgICAgICAgICAgaWYgKGxvb2t1cC52YWx1ZUF0KGkpID49IHNpemVfdChpbmRleCkpIHsKLSAgICAgICAgICAgICAgICBsb29rdXAuZWRpdFZhbHVlQXQoaSktLTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gaW5kZXg7Ci0gICAgfQotICAgIHJldHVybiBOQU1FX05PVF9GT1VORDsKLX0KLQotc3NpemVfdCBTdXJmYWNlRmxpbmdlcjo6TGF5ZXJWZWN0b3I6OnJlb3JkZXIoCi0gICAgICAgIExheWVyQmFzZSogbGF5ZXIsCi0gICAgICAgIFZlY3RvcjxMYXllckJhc2UqPjo6Y29tcGFyX3QgY21wKQotewotICAgIC8vIFhYWDogaXQncyBhIGxpdHRsZSBsYW1lLiBidXQgb2ggd2VsbC4uLgotICAgIHNzaXplX3QgZXJyID0gcmVtb3ZlKGxheWVyKTsKLSAgICBpZiAoZXJyID49MCkKLSAgICAgICAgZXJyID0gYWRkKGxheWVyLCBjbXApOwotICAgIHJldHVybiBlcnI7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jZW5kaWYKLQotU3VyZmFjZUZsaW5nZXI6OlN1cmZhY2VGbGluZ2VyKCkKLSAgICA6ICAgQm5TdXJmYWNlQ29tcG9zZXIoKSwgVGhyZWFkKGZhbHNlKSwKLSAgICAgICAgbVRyYW5zYWN0aW9uRmxhZ3MoMCksCi0gICAgICAgIG1UcmFuc2FjdGlvbkNvdW50KDApLAotICAgICAgICBtQm9vdFRpbWUoc3lzdGVtVGltZSgpKSwKLSAgICAgICAgbUxhc3RTY2hlZHVsZWRCcm9hZGNhc3QoTlVMTCksCi0gICAgICAgIG1WaXNpYmxlUmVnaW9uc0RpcnR5KGZhbHNlKSwKLSAgICAgICAgbURlZmVyUmVsZWFzZUNvbnNvbGUoZmFsc2UpLAotICAgICAgICBtRnJlZXplRGlzcGxheShmYWxzZSksCi0gICAgICAgIG1GcmVlemVDb3VudCgwKSwKLSAgICAgICAgbURlYnVnUmVnaW9uKDApLAotICAgICAgICBtRGVidWdDcHUoMCksCi0gICAgICAgIG1EZWJ1Z0ZwcygwKSwKLSAgICAgICAgbURlYnVnQmFja2dyb3VuZCgwKSwKLSAgICAgICAgbURlYnVnTm9Cb290QW5pbWF0aW9uKDApLAotICAgICAgICBtU3luY09iamVjdCgpLAotICAgICAgICBtRGVwbGF5ZWRUcmFuc2FjdGlvblBlbmRpbmcoMCksCi0gICAgICAgIG1Db25zb2xlU2lnbmFscygwKSwKLSAgICAgICAgbVNlY3VyZUZyYW1lQnVmZmVyKDApCi17Ci0gICAgaW5pdCgpOwotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjppbml0KCkKLXsKLSAgICBMT0dJKCJTdXJmYWNlRmxpbmdlciBpcyBzdGFydGluZyIpOwotCi0gICAgLy8gZGVidWdnaW5nIHN0dWZmLi4uCi0gICAgY2hhciB2YWx1ZVtQUk9QRVJUWV9WQUxVRV9NQVhdOwotICAgIHByb3BlcnR5X2dldCgiZGVidWcuc2Yuc2hvd3VwZGF0ZXMiLCB2YWx1ZSwgIjAiKTsKLSAgICBtRGVidWdSZWdpb24gPSBhdG9pKHZhbHVlKTsKLSAgICBwcm9wZXJ0eV9nZXQoImRlYnVnLnNmLnNob3djcHUiLCB2YWx1ZSwgIjAiKTsKLSAgICBtRGVidWdDcHUgPSBhdG9pKHZhbHVlKTsKLSAgICBwcm9wZXJ0eV9nZXQoImRlYnVnLnNmLnNob3diYWNrZ3JvdW5kIiwgdmFsdWUsICIwIik7Ci0gICAgbURlYnVnQmFja2dyb3VuZCA9IGF0b2kodmFsdWUpOwotICAgIHByb3BlcnR5X2dldCgiZGVidWcuc2Yuc2hvd2ZwcyIsIHZhbHVlLCAiMCIpOwotICAgIG1EZWJ1Z0ZwcyA9IGF0b2kodmFsdWUpOwotICAgIHByb3BlcnR5X2dldCgiZGVidWcuc2Yubm9ib290YW5pbWF0aW9uIiwgdmFsdWUsICIwIik7Ci0gICAgbURlYnVnTm9Cb290QW5pbWF0aW9uID0gYXRvaSh2YWx1ZSk7Ci0KLSAgICBMT0dJX0lGKG1EZWJ1Z1JlZ2lvbiwgICAgICAgICAgICJzaG93dXBkYXRlcyBlbmFibGVkIik7Ci0gICAgTE9HSV9JRihtRGVidWdDcHUsICAgICAgICAgICAgICAic2hvd2NwdSBlbmFibGVkIik7Ci0gICAgTE9HSV9JRihtRGVidWdCYWNrZ3JvdW5kLCAgICAgICAic2hvd2JhY2tncm91bmQgZW5hYmxlZCIpOwotICAgIExPR0lfSUYobURlYnVnRnBzLCAgICAgICAgICAgICAgInNob3dmcHMgZW5hYmxlZCIpOwotICAgIExPR0lfSUYobURlYnVnTm9Cb290QW5pbWF0aW9uLCAgImJvb3QgYW5pbWF0aW9uIGRpc2FibGVkIik7Ci19Ci0KLVN1cmZhY2VGbGluZ2VyOjp+U3VyZmFjZUZsaW5nZXIoKQotewotICAgIGdsRGVsZXRlVGV4dHVyZXMoMSwgJm1Xb3JtaG9sZVRleE5hbWUpOwotICAgIGRlbGV0ZSBtT3JpZW50YXRpb25BbmltYXRpb247Ci19Ci0KLWNvcHliaXRfZGV2aWNlX3QqIFN1cmZhY2VGbGluZ2VyOjpnZXRCbGl0RW5naW5lKCkgY29uc3QKLXsKLSAgICByZXR1cm4gZ3JhcGhpY1BsYW5lKDApLmRpc3BsYXlIYXJkd2FyZSgpLmdldEJsaXRFbmdpbmUoKTsKLX0KLQotb3ZlcmxheV9jb250cm9sX2RldmljZV90KiBTdXJmYWNlRmxpbmdlcjo6Z2V0T3ZlcmxheUVuZ2luZSgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIGdyYXBoaWNQbGFuZSgwKS5kaXNwbGF5SGFyZHdhcmUoKS5nZXRPdmVybGF5RW5naW5lKCk7Ci19Ci0KLXNwPElNZW1vcnk+IFN1cmZhY2VGbGluZ2VyOjpnZXRDYmxrKCkgY29uc3QKLXsKLSAgICByZXR1cm4gbVNlcnZlckNibGtNZW1vcnk7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VGbGluZ2VyOjpyZXF1ZXN0R1BVKGNvbnN0IHNwPElHUFVDYWxsYmFjaz4mIGNhbGxiYWNrLAotICAgICAgICBncHVfaW5mb190KiBncHUpCi17Ci0gICAgSVBDVGhyZWFkU3RhdGUqIGlwYyA9IElQQ1RocmVhZFN0YXRlOjpzZWxmKCk7Ci0gICAgY29uc3QgaW50IHBpZCA9IGlwYy0+Z2V0Q2FsbGluZ1BpZCgpOwotICAgIHN0YXR1c190IGVyciA9IG1HUFUtPnJlcXVlc3QocGlkLCBjYWxsYmFjaywgZ3B1KTsKLSAgICByZXR1cm4gZXJyOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlRmxpbmdlcjo6cmV2b2tlR1BVKCkKLXsKLSAgICByZXR1cm4gbUdQVS0+ZnJpZW5kbHlSZXZva2UoKTsKLX0KLQotc3A8SVN1cmZhY2VGbGluZ2VyQ2xpZW50PiBTdXJmYWNlRmxpbmdlcjo6Y3JlYXRlQ29ubmVjdGlvbigpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1TdGF0ZUxvY2spOwotICAgIHVpbnQzMl90IHRva2VuID0gbVRva2Vucy5hY3F1aXJlKCk7Ci0KLSAgICBDbGllbnQqIGNsaWVudCA9IG5ldyBDbGllbnQodG9rZW4sIHRoaXMpOwotICAgIGlmICgoY2xpZW50ID09IDApIHx8IChjbGllbnQtPmN0cmxibGsgPT0gMCkpIHsKLSAgICAgICAgbVRva2Vucy5yZWxlYXNlKHRva2VuKTsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotICAgIHN0YXR1c190IGVyciA9IG1DbGllbnRzTWFwLmFkZCh0b2tlbiwgY2xpZW50KTsKLSAgICBpZiAoZXJyIDwgMCkgewotICAgICAgICBkZWxldGUgY2xpZW50OwotICAgICAgICBtVG9rZW5zLnJlbGVhc2UodG9rZW4pOwotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0gICAgc3A8QkNsaWVudD4gYmNsaWVudCA9Ci0gICAgICAgIG5ldyBCQ2xpZW50KHRoaXMsIHRva2VuLCBjbGllbnQtPmNvbnRyb2xCbG9ja01lbW9yeSgpKTsKLSAgICByZXR1cm4gYmNsaWVudDsKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6ZGVzdHJveUNvbm5lY3Rpb24oQ2xpZW50SUQgY2lkKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtU3RhdGVMb2NrKTsKLSAgICBDbGllbnQqIGNvbnN0IGNsaWVudCA9IG1DbGllbnRzTWFwLnZhbHVlRm9yKGNpZCk7Ci0gICAgaWYgKGNsaWVudCkgewotICAgICAgICAvLyBmcmVlIGFsbCB0aGUgbGF5ZXJzIHRoaXMgY2xpZW50IG93bnMKLSAgICAgICAgY29uc3QgVmVjdG9yPExheWVyQmFzZUNsaWVudCo+JiBsYXllcnMgPSBjbGllbnQtPmdldExheWVycygpOwotICAgICAgICBjb25zdCBzaXplX3QgY291bnQgPSBsYXllcnMuc2l6ZSgpOwotICAgICAgICBmb3IgKHNpemVfdCBpPTAgOyBpPGNvdW50IDsgaSsrKSB7Ci0gICAgICAgICAgICBMYXllckJhc2VDbGllbnQqIGNvbnN0IGxheWVyID0gbGF5ZXJzW2ldOwotICAgICAgICAgICAgcmVtb3ZlTGF5ZXJfbChsYXllcik7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyB0aGUgcmVzb3VyY2VzIGFzc29jaWF0ZWQgd2l0aCB0aGlzIGNsaWVudCB3aWxsIGJlIGZyZWVkCi0gICAgICAgIC8vIGR1cmluZyB0aGUgbmV4dCB0cmFuc2FjdGlvbiwgYWZ0ZXIgdGhlc2Ugc3VyZmFjZXMgaGF2ZSBiZWVuCi0gICAgICAgIC8vIHByb3Blcmx5IHJlbW92ZWQgZnJvbSB0aGUgc2NyZWVuCi0KLSAgICAgICAgLy8gcmVtb3ZlIHRoaXMgY2xpZW50IGZyb20gb3VyIENsaWVudElELT5DbGllbnQgbWFwcGluZy4KLSAgICAgICAgbUNsaWVudHNNYXAucmVtb3ZlSXRlbShjaWQpOwotCi0gICAgICAgIC8vIGFuZCBhZGQgaXQgdG8gdGhlIGxpc3Qgb2YgZGlzY29ubmVjdGVkIGNsaWVudHMKLSAgICAgICAgbURpc2Nvbm5lY3RlZENsaWVudHMuYWRkKGNsaWVudCk7Ci0KLSAgICAgICAgLy8gcmVxdWVzdCBhIHRyYW5zYWN0aW9uCi0gICAgICAgIHNldFRyYW5zYWN0aW9uRmxhZ3MoZVRyYW5zYWN0aW9uTmVlZGVkKTsKLSAgICB9Ci19Ci0KLWNvbnN0IEdyYXBoaWNQbGFuZSYgU3VyZmFjZUZsaW5nZXI6OmdyYXBoaWNQbGFuZShpbnQgZHB5KSBjb25zdAotewotICAgIExPR0VfSUYodWludDMyX3QoZHB5KSA+PSBESVNQTEFZX0NPVU5ULCAiSW52YWxpZCBEaXNwbGF5SUQgJWQiLCBkcHkpOwotICAgIGNvbnN0IEdyYXBoaWNQbGFuZSYgcGxhbmUobUdyYXBoaWNQbGFuZXNbZHB5XSk7Ci0gICAgcmV0dXJuIHBsYW5lOwotfQotCi1HcmFwaGljUGxhbmUmIFN1cmZhY2VGbGluZ2VyOjpncmFwaGljUGxhbmUoaW50IGRweSkKLXsKLSAgICByZXR1cm4gY29uc3RfY2FzdDxHcmFwaGljUGxhbmUmPigKLSAgICAgICAgY29uc3RfY2FzdDxTdXJmYWNlRmxpbmdlciBjb25zdCAqPih0aGlzKS0+Z3JhcGhpY1BsYW5lKGRweSkpOwotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjpib290RmluaXNoZWQoKQotewotICAgIGNvbnN0IG5zZWNzX3Qgbm93ID0gc3lzdGVtVGltZSgpOwotICAgIGNvbnN0IG5zZWNzX3QgZHVyYXRpb24gPSBub3cgLSBtQm9vdFRpbWU7Ci0gICAgTE9HSSgiQm9vdCBpcyBmaW5pc2hlZCAoJWxkIG1zKSIsIGxvbmcobnMybXMoZHVyYXRpb24pKSApOwotICAgIGlmIChtQm9vdEFuaW1hdGlvbiAhPSAwKSB7Ci0gICAgICAgIG1Cb290QW5pbWF0aW9uLT5yZXF1ZXN0RXhpdCgpOwotICAgICAgICBtQm9vdEFuaW1hdGlvbi5jbGVhcigpOwotICAgIH0KLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6b25GaXJzdFJlZigpCi17Ci0gICAgcnVuKCJTdXJmYWNlRmxpbmdlciIsIFBSSU9SSVRZX1VSR0VOVF9ESVNQTEFZKTsKLQotICAgIC8vIFdhaXQgZm9yIHRoZSBtYWluIHRocmVhZCB0byBiZSBkb25lIHdpdGggaXRzIGluaXRpYWxpemF0aW9uCi0gICAgbVJlYWR5VG9SdW5CYXJyaWVyLndhaXQoKTsKLX0KLQotCi1zdGF0aWMgaW5saW5lIHVpbnQxNl90IHBhY2s1NjUoaW50IHIsIGludCBnLCBpbnQgYikgewotICAgIHJldHVybiAocjw8MTEpfChnPDw1KXxiOwotfQotCi0vLyB0aGlzIGlzIGRlZmluZWQgaW4gbGliR0xFU19DTS5zbwotZXh0ZXJuIElTdXJmYWNlQ29tcG9zZXIqIEdMRVNfbG9jYWxTdXJmYWNlTWFuYWdlcjsKLQotc3RhdHVzX3QgU3VyZmFjZUZsaW5nZXI6OnJlYWR5VG9SdW4oKQotewotICAgIExPR0koICAgIlN1cmZhY2VGbGluZ2VyJ3MgbWFpbiB0aHJlYWQgcmVhZHkgdG8gcnVuLiAiCi0gICAgICAgICAgICAiSW5pdGlhbGl6aW5nIGdyYXBoaWNzIEgvVy4uLiIpOwotCi0gICAgLy8gY3JlYXRlIHRoZSBzaGFyZWQgY29udHJvbC1ibG9jawotICAgIG1TZXJ2ZXJIZWFwID0gbmV3IE1lbW9yeURlYWxlcig0MDk2LCBNZW1vcnlEZWFsZXI6OlJFQURfT05MWSk7Ci0gICAgTE9HRV9JRihtU2VydmVySGVhcD09MCwgImNhbid0IGNyZWF0ZSBzaGFyZWQgbWVtb3J5IGRlYWxlciIpOwotCi0gICAgbVNlcnZlckNibGtNZW1vcnkgPSBtU2VydmVySGVhcC0+YWxsb2NhdGUoNDA5Nik7Ci0gICAgTE9HRV9JRihtU2VydmVyQ2Jsa01lbW9yeT09MCwgImNhbid0IGNyZWF0ZSBzaGFyZWQgY29udHJvbCBibG9jayIpOwotCi0gICAgbVNlcnZlckNibGsgPSBzdGF0aWNfY2FzdDxzdXJmYWNlX2ZsaW5nZXJfY2Jsa190ICo+KG1TZXJ2ZXJDYmxrTWVtb3J5LT5wb2ludGVyKCkpOwotICAgIExPR0VfSUYobVNlcnZlckNibGs9PTAsICJjYW4ndCBnZXQgdG8gc2hhcmVkIGNvbnRyb2wgYmxvY2sncyBhZGRyZXNzIik7Ci0gICAgbmV3KG1TZXJ2ZXJDYmxrKSBzdXJmYWNlX2ZsaW5nZXJfY2Jsa190OwotCi0gICAgLy8gZ2V0IGEgcmVmZXJlbmNlIHRvIHRoZSBHUFUgaWYgd2UgaGF2ZSBvbmUKLSAgICBtR1BVID0gR1BVRmFjdG9yeTo6Z2V0R1BVKCk7Ci0KLSAgICAvLyBjcmVhdGUgdGhlIHN1cmZhY2UgSGVhcCBtYW5hZ2VyLCB3aGljaCBtYW5hZ2VzIHRoZSBoZWFwcwotICAgIC8vIChiZSBpdCBpbiBSQU0gb3IgVlJBTSkgd2hlcmUgc3VyZmFjZXMgYXJlIGFsbG9jYXRlZAotICAgIC8vIFdlIGdpdmUgOCBNQiBwZXIgY2xpZW50LgotICAgIG1TdXJmYWNlSGVhcE1hbmFnZXIgPSBuZXcgU3VyZmFjZUhlYXBNYW5hZ2VyKHRoaXMsIDggPDwgMjApOwotCi0gICAgCi0gICAgR0xFU19sb2NhbFN1cmZhY2VNYW5hZ2VyID0gc3RhdGljX2Nhc3Q8SVN1cmZhY2VDb21wb3Nlcio+KHRoaXMpOwotCi0gICAgLy8gd2Ugb25seSBzdXBwb3J0IG9uZSBkaXNwbGF5IGN1cnJlbnRseQotICAgIGludCBkcHkgPSAwOwotCi0gICAgewotICAgICAgICAvLyBpbml0aWFsaXplIHRoZSBtYWluIGRpc3BsYXkKLSAgICAgICAgR3JhcGhpY1BsYW5lJiBwbGFuZShncmFwaGljUGxhbmUoZHB5KSk7Ci0gICAgICAgIERpc3BsYXlIYXJkd2FyZSogY29uc3QgaHcgPSBuZXcgRGlzcGxheUhhcmR3YXJlKHRoaXMsIGRweSk7Ci0gICAgICAgIHBsYW5lLnNldERpc3BsYXlIYXJkd2FyZShodyk7Ci0gICAgfQotCi0gICAgLy8gaW5pdGlhbGl6ZSBwcmltYXJ5IHNjcmVlbgotICAgIC8vIChvdGhlciBkaXNwbGF5IHNob3VsZCBiZSBpbml0aWFsaXplZCBpbiB0aGUgc2FtZSBtYW5uZXIsIGJ1dAotICAgIC8vIGFzeW5jaHJvbm91c2x5LCBhcyB0aGV5IGNvdWxkIGNvbWUgYW5kIGdvLiBOb25lIG9mIHRoaXMgaXMgc3VwcG9ydGVkCi0gICAgLy8geWV0KS4KLSAgICBjb25zdCBHcmFwaGljUGxhbmUmIHBsYW5lKGdyYXBoaWNQbGFuZShkcHkpKTsKLSAgICBjb25zdCBEaXNwbGF5SGFyZHdhcmUmIGh3ID0gcGxhbmUuZGlzcGxheUhhcmR3YXJlKCk7Ci0gICAgY29uc3QgdWludDMyX3QgdyA9IGh3LmdldFdpZHRoKCk7Ci0gICAgY29uc3QgdWludDMyX3QgaCA9IGh3LmdldEhlaWdodCgpOwotICAgIGNvbnN0IHVpbnQzMl90IGYgPSBody5nZXRGb3JtYXQoKTsKLSAgICBody5tYWtlQ3VycmVudCgpOwotCi0gICAgLy8gaW5pdGlhbGl6ZSB0aGUgc2hhcmVkIGNvbnRyb2wgYmxvY2sKLSAgICBtU2VydmVyQ2Jsay0+Y29ubmVjdGVkIHw9IDE8PGRweTsKLSAgICBkaXNwbGF5X2NibGtfdCogZGNibGsgPSBtU2VydmVyQ2Jsay0+ZGlzcGxheXMgKyBkcHk7Ci0gICAgbWVtc2V0KGRjYmxrLCAwLCBzaXplb2YoZGlzcGxheV9jYmxrX3QpKTsKLSAgICBkY2Jsay0+dyAgICAgICAgICAgID0gdzsKLSAgICBkY2Jsay0+aCAgICAgICAgICAgID0gaDsKLSAgICBkY2Jsay0+Zm9ybWF0ICAgICAgID0gZjsKLSAgICBkY2Jsay0+b3JpZW50YXRpb24gID0gSVN1cmZhY2VDb21wb3Nlcjo6ZU9yaWVudGF0aW9uRGVmYXVsdDsKLSAgICBkY2Jsay0+eGRwaSAgICAgICAgID0gaHcuZ2V0RHBpWCgpOwotICAgIGRjYmxrLT55ZHBpICAgICAgICAgPSBody5nZXREcGlZKCk7Ci0gICAgZGNibGstPmZwcyAgICAgICAgICA9IGh3LmdldFJlZnJlc2hSYXRlKCk7Ci0gICAgZGNibGstPmRlbnNpdHkgICAgICA9IGh3LmdldERlbnNpdHkoKTsKLSAgICBhc20gdm9sYXRpbGUgKCIiOjo6Im1lbW9yeSIpOwotCi0gICAgLy8gSW5pdGlhbGl6ZSBPcGVuR0x8RVMKLSAgICBnbEFjdGl2ZVRleHR1cmUoR0xfVEVYVFVSRTApOwotICAgIGdsQmluZFRleHR1cmUoR0xfVEVYVFVSRV8yRCwgMCk7Ci0gICAgZ2xUZXhQYXJhbWV0ZXJ4KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfV1JBUF9TLCBHTF9DTEFNUF9UT19FREdFKTsKLSAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9XUkFQX1QsIEdMX0NMQU1QX1RPX0VER0UpOwotICAgIGdsVGV4UGFyYW1ldGVyeChHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX01BR19GSUxURVIsIEdMX05FQVJFU1QpOwotICAgIGdsVGV4UGFyYW1ldGVyeChHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX01JTl9GSUxURVIsIEdMX05FQVJFU1QpOwotICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfUkVQTEFDRSk7Ci0gICAgZ2xQaXhlbFN0b3JlaShHTF9VTlBBQ0tfQUxJR05NRU5ULCA0KTsKLSAgICBnbFBpeGVsU3RvcmVpKEdMX1BBQ0tfQUxJR05NRU5ULCA0KTsgCi0gICAgZ2xFbmFibGVDbGllbnRTdGF0ZShHTF9WRVJURVhfQVJSQVkpOwotICAgIGdsRW5hYmxlKEdMX1NDSVNTT1JfVEVTVCk7Ci0gICAgZ2xTaGFkZU1vZGVsKEdMX0ZMQVQpOwotICAgIGdsRGlzYWJsZShHTF9ESVRIRVIpOwotICAgIGdsRGlzYWJsZShHTF9DVUxMX0ZBQ0UpOwotCi0gICAgY29uc3QgdWludDE2X3QgZzAgPSBwYWNrNTY1KDB4MEYsMHgxRiwweDBGKTsKLSAgICBjb25zdCB1aW50MTZfdCBnMSA9IHBhY2s1NjUoMHgxNywweDJmLDB4MTcpOwotICAgIGNvbnN0IHVpbnQxNl90IHRleHR1cmVEYXRhWzRdID0geyBnMCwgZzEsIGcxLCBnMCB9OwotICAgIGdsR2VuVGV4dHVyZXMoMSwgJm1Xb3JtaG9sZVRleE5hbWUpOwotICAgIGdsQmluZFRleHR1cmUoR0xfVEVYVFVSRV8yRCwgbVdvcm1ob2xlVGV4TmFtZSk7Ci0gICAgZ2xUZXhQYXJhbWV0ZXJ4KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfTUFHX0ZJTFRFUiwgR0xfTkVBUkVTVCk7Ci0gICAgZ2xUZXhQYXJhbWV0ZXJ4KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfTUlOX0ZJTFRFUiwgR0xfTkVBUkVTVCk7Ci0gICAgZ2xUZXhQYXJhbWV0ZXJ4KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfV1JBUF9TLCBHTF9SRVBFQVQpOwotICAgIGdsVGV4UGFyYW1ldGVyeChHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX1dSQVBfVCwgR0xfUkVQRUFUKTsKLSAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwgR0xfUkdCLCAyLCAyLCAwLAotICAgICAgICAgICAgR0xfUkdCLCBHTF9VTlNJR05FRF9TSE9SVF81XzZfNSwgdGV4dHVyZURhdGEpOwotCi0gICAgZ2xWaWV3cG9ydCgwLCAwLCB3LCBoKTsKLSAgICBnbE1hdHJpeE1vZGUoR0xfUFJPSkVDVElPTik7Ci0gICAgZ2xMb2FkSWRlbnRpdHkoKTsKLSAgICBnbE9ydGhvZigwLCB3LCBoLCAwLCAwLCAxKTsKLQotICAgTGF5ZXJEaW06OmluaXREaW1tZXIodGhpcywgdywgaCk7Ci0KLSAgICBtUmVhZHlUb1J1bkJhcnJpZXIub3BlbigpOwotCi0gICAgLyoKLSAgICAgKiAgV2UncmUgbm93IHJlYWR5IHRvIGFjY2VwdCBjbGllbnRzLi4uCi0gICAgICovCi0KLSAgICBtT3JpZW50YXRpb25BbmltYXRpb24gPSBuZXcgT3JpZW50YXRpb25BbmltYXRpb24odGhpcyk7Ci0gICAgCi0gICAgLy8gc3RhcnQgQ1BVIGdhdWdlIGRpc3BsYXkKLSAgICBpZiAobURlYnVnQ3B1KQotICAgICAgICBtQ3B1R2F1Z2UgPSBuZXcgQ1BVR2F1Z2UodGhpcywgbXMybnMoNTAwKSk7Ci0KLSAgICAvLyB0aGUgYm9vdCBhbmltYXRpb24hCi0gICAgaWYgKG1EZWJ1Z05vQm9vdEFuaW1hdGlvbiA9PSBmYWxzZSkKLSAgICAgICAgbUJvb3RBbmltYXRpb24gPSBuZXcgQm9vdEFuaW1hdGlvbih0aGlzKTsKLQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jcHJhZ21hIG1hcmsgRXZlbnRzIEhhbmRsZXIKLSNlbmRpZgotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjp3YWl0Rm9yRXZlbnQoKQotewotICAgIC8vIHdhaXQgZm9yIHNvbWV0aGluZyB0byBkbwotICAgIGlmIChVTkxJS0VMWShpc0Zyb3plbigpKSkgewotICAgICAgICAvLyB3YWl0IDUgc2Vjb25kcwotICAgICAgICBpbnQgZXJyID0gbVN5bmNPYmplY3Qud2FpdChtczJucyg1MDAwKSk7Ci0gICAgICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgIGlmIChpc0Zyb3plbigpKSB7Ci0gICAgICAgICAgICAgICAgLy8gd2UgdGltZWQgb3V0IGFuZCBhcmUgc3RpbGwgZnJvemVuCi0gICAgICAgICAgICAgICAgTE9HVygidGltZW91dCBleHBpcmVkIG1GcmVlemVEaXNwbGF5PSVkLCBtRnJlZXplQ291bnQ9JWQiLAotICAgICAgICAgICAgICAgICAgICAgICAgbUZyZWV6ZURpc3BsYXksIG1GcmVlemVDb3VudCk7Ci0gICAgICAgICAgICAgICAgbUZyZWV6ZUNvdW50ID0gMDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIG1TeW5jT2JqZWN0LndhaXQoKTsKLSAgICB9Ci19Ci0KLXZvaWQgU3VyZmFjZUZsaW5nZXI6OnNpZ25hbEV2ZW50KCkgewotICAgIG1TeW5jT2JqZWN0Lm9wZW4oKTsKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6c2lnbmFsKCkgY29uc3QgewotICAgIG1TeW5jT2JqZWN0Lm9wZW4oKTsKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6c2lnbmFsRGVsYXllZEV2ZW50KG5zZWNzX3QgZGVsYXkpCi17Ci0gICAgaWYgKGFuZHJvaWRfYXRvbWljX29yKDEsICZtRGVwbGF5ZWRUcmFuc2FjdGlvblBlbmRpbmcpID09IDApIHsKLSAgICAgICAgc3A8RGVsYXllZFRyYW5zYWN0aW9uPiBkZWxheWVkRXZlbnQobmV3IERlbGF5ZWRUcmFuc2FjdGlvbih0aGlzLCBkZWxheSkpOwotICAgICAgICBkZWxheWVkRXZlbnQtPnJ1bigiRGVsYXllZGVFdmVudCIsIFBSSU9SSVRZX1VSR0VOVF9ESVNQTEFZKTsKLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIE1haW4gbG9vcAotI2VuZGlmCi0KLWJvb2wgU3VyZmFjZUZsaW5nZXI6OnRocmVhZExvb3AoKQotewotICAgIHdhaXRGb3JFdmVudCgpOwotCi0gICAgLy8gY2hlY2sgZm9yIHRyYW5zYWN0aW9ucwotICAgIGlmIChVTkxJS0VMWShtQ29uc29sZVNpZ25hbHMpKSB7Ci0gICAgICAgIGhhbmRsZUNvbnNvbGVFdmVudHMoKTsKLSAgICB9Ci0KLSAgICBpZiAoTElLRUxZKG1UcmFuc2FjdGlvbkNvdW50ID09IDApKSB7Ci0gICAgICAgIC8vIGlmIHdlJ3JlIGluIGEgZ2xvYmFsIHRyYW5zYWN0aW9uLCBkb24ndCBkbyBhbnl0aGluZy4KLSAgICAgICAgY29uc3QgdWludDMyX3QgbWFzayA9IGVUcmFuc2FjdGlvbk5lZWRlZCB8IGVUcmF2ZXJzYWxOZWVkZWQ7Ci0gICAgICAgIHVpbnQzMl90IHRyYW5zYWN0aW9uRmxhZ3MgPSBnZXRUcmFuc2FjdGlvbkZsYWdzKG1hc2spOwotICAgICAgICBpZiAoTElLRUxZKHRyYW5zYWN0aW9uRmxhZ3MpKSB7Ci0gICAgICAgICAgICBoYW5kbGVUcmFuc2FjdGlvbih0cmFuc2FjdGlvbkZsYWdzKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vIHBvc3Qgc3VyZmFjZXMgKGlmIG5lZWRlZCkKLSAgICBoYW5kbGVQYWdlRmxpcCgpOwotCi0gICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiBodyhncmFwaGljUGxhbmUoMCkuZGlzcGxheUhhcmR3YXJlKCkpOwotICAgIGlmIChMSUtFTFkoaHcuY2FuRHJhdygpKSkgewotICAgICAgICAvLyByZXBhaW50IHRoZSBmcmFtZWJ1ZmZlciAoaWYgbmVlZGVkKQotICAgICAgICBoYW5kbGVSZXBhaW50KCk7Ci0KLSAgICAgICAgLy8gcmVsZWFzZSB0aGUgY2xpZW50cyBiZWZvcmUgd2UgZmxpcCAoJ2NhdXNlIGZsaXAgbWlnaHQgYmxvY2spCi0gICAgICAgIHVubG9ja0NsaWVudHMoKTsKLSAgICAgICAgZXhlY3V0ZVNjaGVkdWxlZEJyb2FkY2FzdHMoKTsKLQotICAgICAgICAvLyBzYW1wbGUgdGhlIGNwdSBnYXVnZQotICAgICAgICBpZiAoVU5MSUtFTFkobURlYnVnQ3B1KSkgewotICAgICAgICAgICAgaGFuZGxlRGVidWdDcHUoKTsKLSAgICAgICAgfQotCi0gICAgICAgIHBvc3RGcmFtZWJ1ZmZlcigpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIC8vIHByZXRlbmQgd2UgZGlkIHRoZSBwb3N0Ci0gICAgICAgIHVubG9ja0NsaWVudHMoKTsKLSAgICAgICAgZXhlY3V0ZVNjaGVkdWxlZEJyb2FkY2FzdHMoKTsKLSAgICAgICAgdXNsZWVwKDE2NjY3KTsgLy8gNjAgZnBzIHBlcmlvZAotICAgIH0KLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6cG9zdEZyYW1lYnVmZmVyKCkKLXsKLSAgICBjb25zdCBib29sIHNraXAgPSBtT3JpZW50YXRpb25BbmltYXRpb24tPnJ1bigpOwotICAgIGlmIChVTkxJS0VMWShza2lwKSkgewotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgaWYgKCFtSW52YWxpZFJlZ2lvbi5pc0VtcHR5KCkpIHsKLSAgICAgICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiBodyhncmFwaGljUGxhbmUoMCkuZGlzcGxheUhhcmR3YXJlKCkpOwotCi0gICAgICAgIGlmIChVTkxJS0VMWShtRGVidWdGcHMpKSB7Ci0gICAgICAgICAgICBkZWJ1Z1Nob3dGUFMoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGh3LmZsaXAobUludmFsaWRSZWdpb24pOwotCi0gICAgICAgIG1JbnZhbGlkUmVnaW9uLmNsZWFyKCk7Ci0KLSAgICAgICAgaWYgKExheWVyOjpkZWxldGVkVGV4dHVyZXMuc2l6ZSgpKSB7Ci0gICAgICAgICAgICBnbERlbGV0ZVRleHR1cmVzKAotICAgICAgICAgICAgICAgICAgICBMYXllcjo6ZGVsZXRlZFRleHR1cmVzLnNpemUoKSwKLSAgICAgICAgICAgICAgICAgICAgTGF5ZXI6OmRlbGV0ZWRUZXh0dXJlcy5hcnJheSgpKTsKLSAgICAgICAgICAgIExheWVyOjpkZWxldGVkVGV4dHVyZXMuY2xlYXIoKTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6aGFuZGxlQ29uc29sZUV2ZW50cygpCi17Ci0gICAgLy8gc29tZXRoaW5nIHRvIGRvIHdpdGggdGhlIGNvbnNvbGUKLSAgICBjb25zdCBEaXNwbGF5SGFyZHdhcmUmIGh3ID0gZ3JhcGhpY1BsYW5lKDApLmRpc3BsYXlIYXJkd2FyZSgpOwotCi0gICAgaW50IHdoYXQgPSBhbmRyb2lkX2F0b21pY19hbmQoMCwgJm1Db25zb2xlU2lnbmFscyk7Ci0gICAgaWYgKHdoYXQgJiBlQ29uc29sZUFjcXVpcmVkKSB7Ci0gICAgICAgIGh3LmFjcXVpcmVTY3JlZW4oKTsKLSAgICB9Ci0KLSAgICBpZiAobURlZmVyUmVsZWFzZUNvbnNvbGUgJiYgaHcuY2FuRHJhdygpKSB7Ci0gICAgICAgIC8vIFdlIGdvdCB0aGUgcmVsZWFzZSBzaWduYWwgYmVmb3JlIHRoZSBhcXVpcmUgc2lnbmFsCi0gICAgICAgIG1EZWZlclJlbGVhc2VDb25zb2xlID0gZmFsc2U7Ci0gICAgICAgIHJldm9rZUdQVSgpOwotICAgICAgICBody5yZWxlYXNlU2NyZWVuKCk7Ci0gICAgfQotCi0gICAgaWYgKHdoYXQgJiBlQ29uc29sZVJlbGVhc2VkKSB7Ci0gICAgICAgIGlmIChody5jYW5EcmF3KCkpIHsKLSAgICAgICAgICAgIHJldm9rZUdQVSgpOwotICAgICAgICAgICAgaHcucmVsZWFzZVNjcmVlbigpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbURlZmVyUmVsZWFzZUNvbnNvbGUgPSB0cnVlOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgbURpcnR5UmVnaW9uLnNldChody5ib3VuZHMoKSk7Ci19Ci0KLXZvaWQgU3VyZmFjZUZsaW5nZXI6OmhhbmRsZVRyYW5zYWN0aW9uKHVpbnQzMl90IHRyYW5zYWN0aW9uRmxhZ3MpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1TdGF0ZUxvY2spOwotCi0gICAgY29uc3QgTGF5ZXJWZWN0b3ImIGN1cnJlbnRMYXllcnMgPSBtQ3VycmVudFN0YXRlLmxheWVyc1NvcnRlZEJ5WjsKLSAgICBjb25zdCBzaXplX3QgY291bnQgPSBjdXJyZW50TGF5ZXJzLnNpemUoKTsKLQotICAgIC8qCi0gICAgICogVHJhdmVyc2FsIG9mIHRoZSBjaGlsZHJlbgotICAgICAqIChwZXJmb3JtIHRoZSB0cmFuc2FjdGlvbiBmb3IgZWFjaCBvZiB0aGVtIGlmIG5lZWRlZCkKLSAgICAgKi8KLQotICAgIGNvbnN0IGJvb2wgbGF5ZXJzTmVlZFRyYW5zYWN0aW9uID0gdHJhbnNhY3Rpb25GbGFncyAmIGVUcmF2ZXJzYWxOZWVkZWQ7Ci0gICAgaWYgKGxheWVyc05lZWRUcmFuc2FjdGlvbikgewotICAgICAgICBmb3IgKHNpemVfdCBpPTAgOyBpPGNvdW50IDsgaSsrKSB7Ci0gICAgICAgICAgICBMYXllckJhc2UqIGNvbnN0IGxheWVyID0gY3VycmVudExheWVyc1tpXTsKLSAgICAgICAgICAgIHVpbnQzMl90IHRyRmxhZ3MgPSBsYXllci0+Z2V0VHJhbnNhY3Rpb25GbGFncyhlVHJhbnNhY3Rpb25OZWVkZWQpOwotICAgICAgICAgICAgaWYgKCF0ckZsYWdzKSBjb250aW51ZTsKLQotICAgICAgICAgICAgY29uc3QgdWludDMyX3QgZmxhZ3MgPSBsYXllci0+ZG9UcmFuc2FjdGlvbigwKTsKLSAgICAgICAgICAgIGlmIChmbGFncyAmIExheWVyOjplVmlzaWJsZVJlZ2lvbikKLSAgICAgICAgICAgICAgICBtVmlzaWJsZVJlZ2lvbnNEaXJ0eSA9IHRydWU7Ci0KLSAgICAgICAgICAgIGlmIChmbGFncyAmIExheWVyOjplUmVzdGFydFRyYW5zYWN0aW9uKSB7Ci0gICAgICAgICAgICAgICAgLy8gcmVzdGFydCB0aGUgdHJhbnNhY3Rpb24sIGJ1dCBiYWNrLW9mZiBhIGxpdHRsZQotICAgICAgICAgICAgICAgIGxheWVyLT5zZXRUcmFuc2FjdGlvbkZsYWdzKGVUcmFuc2FjdGlvbk5lZWRlZCk7Ci0gICAgICAgICAgICAgICAgc2V0VHJhbnNhY3Rpb25GbGFncyhlVHJhdmVyc2FsTmVlZGVkLCBtczJucyg4KSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFBlcmZvcm0gb3VyIG93biB0cmFuc2FjdGlvbiBpZiBuZWVkZWQKLSAgICAgKi8KLQotICAgIGlmICh0cmFuc2FjdGlvbkZsYWdzICYgZVRyYW5zYWN0aW9uTmVlZGVkKSB7Ci0gICAgICAgIGlmIChtQ3VycmVudFN0YXRlLm9yaWVudGF0aW9uICE9IG1EcmF3aW5nU3RhdGUub3JpZW50YXRpb24pIHsKLSAgICAgICAgICAgIC8vIHRoZSBvcmllbnRhdGlvbiBoYXMgY2hhbmdlZCwgcmVjb21wdXRlIGFsbCB2aXNpYmxlIHJlZ2lvbnMKLSAgICAgICAgICAgIC8vIGFuZCBpbnZhbGlkYXRlIGV2ZXJ5dGhpbmcuCi0KLSAgICAgICAgICAgIGNvbnN0IGludCBkcHkgPSAwOwotICAgICAgICAgICAgY29uc3QgaW50IG9yaWVudGF0aW9uID0gbUN1cnJlbnRTdGF0ZS5vcmllbnRhdGlvbjsKLSAgICAgICAgICAgIEdyYXBoaWNQbGFuZSYgcGxhbmUoZ3JhcGhpY1BsYW5lKGRweSkpOwotICAgICAgICAgICAgcGxhbmUuc2V0T3JpZW50YXRpb24ob3JpZW50YXRpb24pOwotCi0gICAgICAgICAgICAvLyB1cGRhdGUgdGhlIHNoYXJlZCBjb250cm9sIGJsb2NrCi0gICAgICAgICAgICBjb25zdCBEaXNwbGF5SGFyZHdhcmUmIGh3KHBsYW5lLmRpc3BsYXlIYXJkd2FyZSgpKTsKLSAgICAgICAgICAgIHZvbGF0aWxlIGRpc3BsYXlfY2Jsa190KiBkY2JsayA9IG1TZXJ2ZXJDYmxrLT5kaXNwbGF5cyArIGRweTsKLSAgICAgICAgICAgIGRjYmxrLT5vcmllbnRhdGlvbiA9IG9yaWVudGF0aW9uOwotICAgICAgICAgICAgaWYgKG9yaWVudGF0aW9uICYgZU9yaWVudGF0aW9uU3dhcE1hc2spIHsKLSAgICAgICAgICAgICAgICAvLyA5MCBvciAyNzAgZGVncmVlcyBvcmllbnRhdGlvbgotICAgICAgICAgICAgICAgIGRjYmxrLT53ID0gaHcuZ2V0SGVpZ2h0KCk7Ci0gICAgICAgICAgICAgICAgZGNibGstPmggPSBody5nZXRXaWR0aCgpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBkY2Jsay0+dyA9IGh3LmdldFdpZHRoKCk7Ci0gICAgICAgICAgICAgICAgZGNibGstPmggPSBody5nZXRIZWlnaHQoKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgbVZpc2libGVSZWdpb25zRGlydHkgPSB0cnVlOwotICAgICAgICAgICAgbURpcnR5UmVnaW9uLnNldChody5ib3VuZHMoKSk7Ci0KLSAgICAgICAgICAgIG1PcmllbnRhdGlvbkFuaW1hdGlvbi0+b25PcmllbnRhdGlvbkNoYW5nZWQoKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChtQ3VycmVudFN0YXRlLmZyZWV6ZURpc3BsYXkgIT0gbURyYXdpbmdTdGF0ZS5mcmVlemVEaXNwbGF5KSB7Ci0gICAgICAgICAgICAvLyBmcmVlemluZyBvciB1bmZyZWV6aW5nIHRoZSBkaXNwbGF5IC0+IHRyaWdnZXIgYW5pbWF0aW9uIGlmIG5lZWRlZAotICAgICAgICAgICAgbUZyZWV6ZURpc3BsYXkgPSBtQ3VycmVudFN0YXRlLmZyZWV6ZURpc3BsYXk7Ci0gICAgICAgICAgICBjb25zdCBuc2Vjc190IG5vdyA9IHN5c3RlbVRpbWUoKTsKLSAgICAgICAgICAgIGlmIChtRnJlZXplRGlzcGxheSkgewotICAgICAgICAgICAgICAgIG1GcmVlemVEaXNwbGF5VGltZSA9IG5vdzsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgLy9MT0dEKCJTY3JlZW4gd2FzIGZyb3plbiBmb3IgJWxsdSB1cyIsCi0gICAgICAgICAgICAgICAgLy8gICAgICAgIG5zMnVzKG5vdy1tRnJlZXplRGlzcGxheVRpbWUpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIHNvbWUgbGF5ZXJzIG1pZ2h0IGhhdmUgYmVlbiByZW1vdmVkLCBzbwotICAgICAgICAvLyB3ZSBuZWVkIHRvIHVwZGF0ZSB0aGUgcmVnaW9ucyB0aGV5J3JlIGV4cG9zaW5nLgotICAgICAgICBzaXplX3QgYyA9IG1SZW1vdmVkTGF5ZXJzLnNpemUoKTsKLSAgICAgICAgaWYgKGMpIHsKLSAgICAgICAgICAgIG1WaXNpYmxlUmVnaW9uc0RpcnR5ID0gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGNvbnN0IExheWVyVmVjdG9yJiBjdXJyZW50TGF5ZXJzID0gbUN1cnJlbnRTdGF0ZS5sYXllcnNTb3J0ZWRCeVo7Ci0gICAgICAgIGlmIChjdXJyZW50TGF5ZXJzLnNpemUoKSA+IG1EcmF3aW5nU3RhdGUubGF5ZXJzU29ydGVkQnlaLnNpemUoKSkgewotICAgICAgICAgICAgLy8gbGF5ZXJzIGhhdmUgYmVlbiBhZGRlZAotICAgICAgICAgICAgbVZpc2libGVSZWdpb25zRGlydHkgPSB0cnVlOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gZ2V0IHJpZCBvZiBhbGwgcmVzb3VyY2VzIHdlIGRvbid0IG5lZWQgYW55bW9yZQotICAgICAgICAvLyAobGF5ZXJzIGFuZCBjbGllbnRzKQotICAgICAgICBmcmVlX3Jlc291cmNlc19sKCk7Ci0gICAgfQotCi0gICAgY29tbWl0VHJhbnNhY3Rpb24oKTsKLX0KLQotc3A8RnJlZXplTG9jaz4gU3VyZmFjZUZsaW5nZXI6OmdldEZyZWV6ZUxvY2soKSBjb25zdAotewotICAgIHJldHVybiBuZXcgRnJlZXplTG9jayhjb25zdF9jYXN0PFN1cmZhY2VGbGluZ2VyICo+KHRoaXMpKTsKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6Y29tcHV0ZVZpc2libGVSZWdpb25zKAotICAgIExheWVyVmVjdG9yJiBjdXJyZW50TGF5ZXJzLCBSZWdpb24mIGRpcnR5UmVnaW9uLCBSZWdpb24mIG9wYXF1ZVJlZ2lvbikKLXsKLSAgICBjb25zdCBHcmFwaGljUGxhbmUmIHBsYW5lKGdyYXBoaWNQbGFuZSgwKSk7Ci0gICAgY29uc3QgVHJhbnNmb3JtJiBwbGFuZVRyYW5zZm9ybShwbGFuZS50cmFuc2Zvcm0oKSk7Ci0KLSAgICBSZWdpb24gYWJvdmVPcGFxdWVMYXllcnM7Ci0gICAgUmVnaW9uIGFib3ZlQ292ZXJlZExheWVyczsKLSAgICBSZWdpb24gZGlydHk7Ci0KLSAgICBib29sIHNlY3VyZUZyYW1lQnVmZmVyID0gZmFsc2U7Ci0KLSAgICBzaXplX3QgaSA9IGN1cnJlbnRMYXllcnMuc2l6ZSgpOwotICAgIHdoaWxlIChpLS0pIHsKLSAgICAgICAgTGF5ZXJCYXNlKiBjb25zdCBsYXllciA9IGN1cnJlbnRMYXllcnNbaV07Ci0gICAgICAgIGxheWVyLT52YWxpZGF0ZVZpc2liaWxpdHkocGxhbmVUcmFuc2Zvcm0pOwotCi0gICAgICAgIC8vIHN0YXJ0IHdpdGggdGhlIHdob2xlIHN1cmZhY2UgYXQgaXRzIGN1cnJlbnQgbG9jYXRpb24KLSAgICAgICAgY29uc3QgTGF5ZXI6OlN0YXRlJiBzID0gbGF5ZXItPmRyYXdpbmdTdGF0ZSgpOwotICAgICAgICBjb25zdCBSZWN0IGJvdW5kcyhsYXllci0+dmlzaWJsZUJvdW5kcygpKTsKLQotICAgICAgICAvLyBoYW5kbGUgaGlkZGVuIHN1cmZhY2VzIGJ5IHNldHRpbmcgdGhlIHZpc2libGUgcmVnaW9uIHRvIGVtcHR5Ci0gICAgICAgIFJlZ2lvbiBvcGFxdWVSZWdpb247Ci0gICAgICAgIFJlZ2lvbiB2aXNpYmxlUmVnaW9uOwotICAgICAgICBSZWdpb24gY292ZXJlZFJlZ2lvbjsKLSAgICAgICAgaWYgKFVOTElLRUxZKChzLmZsYWdzICYgSVN1cmZhY2VDb21wb3Nlcjo6ZUxheWVySGlkZGVuKSB8fCAhcy5hbHBoYSkpIHsKLSAgICAgICAgICAgIHZpc2libGVSZWdpb24uY2xlYXIoKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGNvbnN0IGJvb2wgdHJhbnNsdWNlbnQgPSBsYXllci0+bmVlZHNCbGVuZGluZygpOwotICAgICAgICAgICAgdmlzaWJsZVJlZ2lvbi5zZXQoYm91bmRzKTsKLSAgICAgICAgICAgIGNvdmVyZWRSZWdpb24gPSB2aXNpYmxlUmVnaW9uOwotCi0gICAgICAgICAgICAvLyBSZW1vdmUgdGhlIHRyYW5zcGFyZW50IGFyZWEgZnJvbSB0aGUgdmlzaWJsZSByZWdpb24KLSAgICAgICAgICAgIGlmICh0cmFuc2x1Y2VudCkgewotICAgICAgICAgICAgICAgIHZpc2libGVSZWdpb24uc3VidHJhY3RTZWxmKGxheWVyLT50cmFuc3BhcmVudFJlZ2lvblNjcmVlbik7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIGNvbXB1dGUgdGhlIG9wYXF1ZSByZWdpb24KLSAgICAgICAgICAgIGlmIChzLmFscGhhPT0yNTUgJiYgIXRyYW5zbHVjZW50ICYmIGxheWVyLT5nZXRPcmllbnRhdGlvbigpPj0wKSB7Ci0gICAgICAgICAgICAgICAgLy8gdGhlIG9wYXF1ZSByZWdpb24gaXMgdGhlIHZpc2libGUgcmVnaW9uCi0gICAgICAgICAgICAgICAgb3BhcXVlUmVnaW9uID0gdmlzaWJsZVJlZ2lvbjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIHN1YnRyYWN0IHRoZSBvcGFxdWUgcmVnaW9uIGNvdmVyZWQgYnkgdGhlIGxheWVycyBhYm92ZSB1cwotICAgICAgICB2aXNpYmxlUmVnaW9uLnN1YnRyYWN0U2VsZihhYm92ZU9wYXF1ZUxheWVycyk7Ci0gICAgICAgIGNvdmVyZWRSZWdpb24uYW5kU2VsZihhYm92ZUNvdmVyZWRMYXllcnMpOwotCi0gICAgICAgIC8vIGNvbXB1dGUgdGhpcyBsYXllcidzIGRpcnR5IHJlZ2lvbgotICAgICAgICBpZiAobGF5ZXItPmNvbnRlbnREaXJ0eSkgewotICAgICAgICAgICAgLy8gd2UgbmVlZCB0byBpbnZhbGlkYXRlIHRoZSB3aG9sZSByZWdpb24KLSAgICAgICAgICAgIGRpcnR5ID0gdmlzaWJsZVJlZ2lvbjsKLSAgICAgICAgICAgIC8vIGFzIHdlbGwsIGFzIHRoZSBvbGQgdmlzaWJsZSByZWdpb24KLSAgICAgICAgICAgIGRpcnR5Lm9yU2VsZihsYXllci0+dmlzaWJsZVJlZ2lvblNjcmVlbik7Ci0gICAgICAgICAgICBsYXllci0+Y29udGVudERpcnR5ID0gZmFsc2U7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBjb21wdXRlIHRoZSBleHBvc2VkIHJlZ2lvbgotICAgICAgICAgICAgLy8gZGlydHkgPSB3aGF0J3MgdmlzaWJsZSBub3cgLSB3aGF0J3Mgd2Fzbid0IGNvdmVyZWQgYmVmb3JlCi0gICAgICAgICAgICAvLyAgICAgICA9IHdoYXQncyB2aXNpYmxlIG5vdyAmIHdoYXQncyB3YXMgY292ZXJlZCBiZWZvcmUKLSAgICAgICAgICAgIGRpcnR5ID0gdmlzaWJsZVJlZ2lvbi5pbnRlcnNlY3QobGF5ZXItPmNvdmVyZWRSZWdpb25TY3JlZW4pOyAgICAgICAgICAgIAotICAgICAgICB9Ci0gICAgICAgIGRpcnR5LnN1YnRyYWN0U2VsZihhYm92ZU9wYXF1ZUxheWVycyk7Ci0KLSAgICAgICAgLy8gYWNjdW11bGF0ZSB0byB0aGUgc2NyZWVuIGRpcnR5IHJlZ2lvbgotICAgICAgICBkaXJ0eVJlZ2lvbi5vclNlbGYoZGlydHkpOwotCi0gICAgICAgIC8vIHVwZGFkZSBhYm92ZU9wYXF1ZUxheWVycy9hYm92ZUNvdmVyZWRMYXllcnMgZm9yIG5leHQgKGxvd2VyKSBsYXllcgotICAgICAgICBhYm92ZU9wYXF1ZUxheWVycy5vclNlbGYob3BhcXVlUmVnaW9uKTsKLSAgICAgICAgYWJvdmVDb3ZlcmVkTGF5ZXJzLm9yU2VsZihib3VuZHMpOwotICAgICAgICAKLSAgICAgICAgLy8gU3RvcmUgdGhlIHZpc2libGUgcmVnaW9uIGlzIHNjcmVlbiBzcGFjZQotICAgICAgICBsYXllci0+c2V0VmlzaWJsZVJlZ2lvbih2aXNpYmxlUmVnaW9uKTsKLSAgICAgICAgbGF5ZXItPnNldENvdmVyZWRSZWdpb24oY292ZXJlZFJlZ2lvbik7Ci0KLSAgICAgICAgLy8gSWYgYSBzZWN1cmUgbGF5ZXIgaXMgcGFydGlhbGx5IHZpc2libGUsIGxvY2tkb3duIHRoZSBzY3JlZW4hCi0gICAgICAgIGlmIChsYXllci0+aXNTZWN1cmUoKSAmJiAhdmlzaWJsZVJlZ2lvbi5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgIHNlY3VyZUZyYW1lQnVmZmVyID0gdHJ1ZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIG1TZWN1cmVGcmFtZUJ1ZmZlciA9IHNlY3VyZUZyYW1lQnVmZmVyOwotICAgIG9wYXF1ZVJlZ2lvbiA9IGFib3ZlT3BhcXVlTGF5ZXJzOwotfQotCi0KLXZvaWQgU3VyZmFjZUZsaW5nZXI6OmNvbW1pdFRyYW5zYWN0aW9uKCkKLXsKLSAgICBtRHJhd2luZ1N0YXRlID0gbUN1cnJlbnRTdGF0ZTsKLSAgICBtVHJhbnNhY3Rpb25DVi5zaWduYWwoKTsKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6aGFuZGxlUGFnZUZsaXAoKQotewotICAgIGJvb2wgdmlzaWJsZVJlZ2lvbnMgPSBtVmlzaWJsZVJlZ2lvbnNEaXJ0eTsKLSAgICBMYXllclZlY3RvciYgY3VycmVudExheWVycyA9IGNvbnN0X2Nhc3Q8TGF5ZXJWZWN0b3ImPihtRHJhd2luZ1N0YXRlLmxheWVyc1NvcnRlZEJ5Wik7Ci0gICAgdmlzaWJsZVJlZ2lvbnMgfD0gbG9ja1BhZ2VGbGlwKGN1cnJlbnRMYXllcnMpOwotCi0gICAgICAgIGNvbnN0IERpc3BsYXlIYXJkd2FyZSYgaHcgPSBncmFwaGljUGxhbmUoMCkuZGlzcGxheUhhcmR3YXJlKCk7Ci0gICAgICAgIGNvbnN0IFJlZ2lvbiBzY3JlZW5SZWdpb24oaHcuYm91bmRzKCkpOwotICAgICAgICBpZiAodmlzaWJsZVJlZ2lvbnMpIHsKLSAgICAgICAgICAgIFJlZ2lvbiBvcGFxdWVSZWdpb247Ci0gICAgICAgICAgICBjb21wdXRlVmlzaWJsZVJlZ2lvbnMoY3VycmVudExheWVycywgbURpcnR5UmVnaW9uLCBvcGFxdWVSZWdpb24pOwotICAgICAgICAgICAgbVdvcm1ob2xlUmVnaW9uID0gc2NyZWVuUmVnaW9uLnN1YnRyYWN0KG9wYXF1ZVJlZ2lvbik7Ci0gICAgICAgICAgICBtVmlzaWJsZVJlZ2lvbnNEaXJ0eSA9IGZhbHNlOwotICAgICAgICB9Ci0KLSAgICB1bmxvY2tQYWdlRmxpcChjdXJyZW50TGF5ZXJzKTsKLSAgICBtRGlydHlSZWdpb24uYW5kU2VsZihzY3JlZW5SZWdpb24pOwotfQotCi1ib29sIFN1cmZhY2VGbGluZ2VyOjpsb2NrUGFnZUZsaXAoY29uc3QgTGF5ZXJWZWN0b3ImIGN1cnJlbnRMYXllcnMpCi17Ci0gICAgYm9vbCByZWNvbXB1dGVWaXNpYmxlUmVnaW9ucyA9IGZhbHNlOwotICAgIHNpemVfdCBjb3VudCA9IGN1cnJlbnRMYXllcnMuc2l6ZSgpOwotICAgIExheWVyQmFzZSogY29uc3QqIGxheWVycyA9IGN1cnJlbnRMYXllcnMuYXJyYXkoKTsKLSAgICBmb3IgKHNpemVfdCBpPTAgOyBpPGNvdW50IDsgaSsrKSB7Ci0gICAgICAgIExheWVyQmFzZSogY29uc3QgbGF5ZXIgPSBsYXllcnNbaV07Ci0gICAgICAgIGxheWVyLT5sb2NrUGFnZUZsaXAocmVjb21wdXRlVmlzaWJsZVJlZ2lvbnMpOwotICAgIH0KLSAgICByZXR1cm4gcmVjb21wdXRlVmlzaWJsZVJlZ2lvbnM7Ci19Ci0KLXZvaWQgU3VyZmFjZUZsaW5nZXI6OnVubG9ja1BhZ2VGbGlwKGNvbnN0IExheWVyVmVjdG9yJiBjdXJyZW50TGF5ZXJzKQotewotICAgIGNvbnN0IEdyYXBoaWNQbGFuZSYgcGxhbmUoZ3JhcGhpY1BsYW5lKDApKTsKLSAgICBjb25zdCBUcmFuc2Zvcm0mIHBsYW5lVHJhbnNmb3JtKHBsYW5lLnRyYW5zZm9ybSgpKTsKLSAgICBzaXplX3QgY291bnQgPSBjdXJyZW50TGF5ZXJzLnNpemUoKTsKLSAgICBMYXllckJhc2UqIGNvbnN0KiBsYXllcnMgPSBjdXJyZW50TGF5ZXJzLmFycmF5KCk7Ci0gICAgZm9yIChzaXplX3QgaT0wIDsgaTxjb3VudCA7IGkrKykgewotICAgICAgICBMYXllckJhc2UqIGNvbnN0IGxheWVyID0gbGF5ZXJzW2ldOwotICAgICAgICBsYXllci0+dW5sb2NrUGFnZUZsaXAocGxhbmVUcmFuc2Zvcm0sIG1EaXJ0eVJlZ2lvbik7Ci0gICAgfQotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjpoYW5kbGVSZXBhaW50KCkKLXsKLSAgICAvLyBzZXQgdGhlIGZyYW1lIGJ1ZmZlcgotICAgIGNvbnN0IERpc3BsYXlIYXJkd2FyZSYgaHcoZ3JhcGhpY1BsYW5lKDApLmRpc3BsYXlIYXJkd2FyZSgpKTsKLSAgICBnbE1hdHJpeE1vZGUoR0xfTU9ERUxWSUVXKTsKLSAgICBnbExvYWRJZGVudGl0eSgpOwotCi0gICAgaWYgKFVOTElLRUxZKG1EZWJ1Z1JlZ2lvbikpIHsKLSAgICAgICAgZGVidWdGbGFzaFJlZ2lvbnMoKTsKLSAgICB9Ci0KLSAgICAvLyBjb21wdXRlIHRoZSBpbnZhbGlkIHJlZ2lvbgotICAgIG1JbnZhbGlkUmVnaW9uLm9yU2VsZihtRGlydHlSZWdpb24pOwotCi0gICAgdWludDMyX3QgZmxhZ3MgPSBody5nZXRGbGFncygpOwotICAgIGlmIChmbGFncyAmIERpc3BsYXlIYXJkd2FyZTo6QlVGRkVSX1BSRVNFUlZFRCkgewotICAgICAgICAvLyBoZXJlIHdlIGFzc3VtZSBEaXNwbGF5SGFyZHdhcmU6OmZsaXAoKSdzICBpbXBsZW1lbnRhdGlvbgotICAgICAgICAvLyBwZXJmb3JtcyB0aGUgY29weS1iYWNrIG9wdGltaXphdGlvbi4KLSAgICB9IGVsc2UgewotICAgICAgICBpZiAoZmxhZ3MgJiBEaXNwbGF5SGFyZHdhcmU6OlVQREFURV9PTl9ERU1BTkQpIHsKLSAgICAgICAgICAgIC8vIHdlIG5lZWQgdG8gZnVsbHkgcmVkcmF3IHRoZSBwYXJ0IHRoYXQgd2lsbCBiZSB1cGRhdGVkCi0gICAgICAgICAgICBtRGlydHlSZWdpb24uc2V0KG1JbnZhbGlkUmVnaW9uLmJvdW5kcygpKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIHdlIG5lZWQgdG8gcmVkcmF3IGV2ZXJ5dGhpbmcKLSAgICAgICAgICAgIG1EaXJ0eVJlZ2lvbi5zZXQoaHcuYm91bmRzKCkpOwotICAgICAgICAgICAgbUludmFsaWRSZWdpb24gPSBtRGlydHlSZWdpb247Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyBjb21wb3NlIGFsbCBzdXJmYWNlcwotICAgIGNvbXBvc2VTdXJmYWNlcyhtRGlydHlSZWdpb24pOwotCi0gICAgLy8gY2xlYXIgdGhlIGRpcnR5IHJlZ2lvbnMKLSAgICBtRGlydHlSZWdpb24uY2xlYXIoKTsKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6Y29tcG9zZVN1cmZhY2VzKGNvbnN0IFJlZ2lvbiYgZGlydHkpCi17Ci0gICAgaWYgKFVOTElLRUxZKCFtV29ybWhvbGVSZWdpb24uaXNFbXB0eSgpKSkgewotICAgICAgICAvLyBzaG91bGQgbmV2ZXIgaGFwcGVuIHVubGVzcyB0aGUgd2luZG93IG1hbmFnZXIgaGFzIGEgYnVnCi0gICAgICAgIC8vIGRyYXcgc29tZXRoaW5nLi4uCi0gICAgICAgIGRyYXdXb3JtaG9sZSgpOwotICAgIH0KLSAgICBjb25zdCBTdXJmYWNlRmxpbmdlciYgZmxpbmdlcigqdGhpcyk7Ci0gICAgY29uc3QgTGF5ZXJWZWN0b3ImIGRyYXdpbmdMYXllcnMobURyYXdpbmdTdGF0ZS5sYXllcnNTb3J0ZWRCeVopOwotICAgIGNvbnN0IHNpemVfdCBjb3VudCA9IGRyYXdpbmdMYXllcnMuc2l6ZSgpOwotICAgIExheWVyQmFzZSBjb25zdCogY29uc3QqIGNvbnN0IGxheWVycyA9IGRyYXdpbmdMYXllcnMuYXJyYXkoKTsKLSAgICBmb3IgKHNpemVfdCBpPTAgOyBpPGNvdW50IDsgKytpKSB7Ci0gICAgICAgIExheWVyQmFzZSBjb25zdCAqIGNvbnN0IGxheWVyID0gbGF5ZXJzW2ldOwotICAgICAgICBjb25zdCBSZWdpb24mIHZpc2libGVSZWdpb24obGF5ZXItPnZpc2libGVSZWdpb25TY3JlZW4pOwotICAgICAgICBpZiAoIXZpc2libGVSZWdpb24uaXNFbXB0eSgpKSAgewotICAgICAgICAgICAgY29uc3QgUmVnaW9uIGNsaXAoZGlydHkuaW50ZXJzZWN0KHZpc2libGVSZWdpb24pKTsKLSAgICAgICAgICAgIGlmICghY2xpcC5pc0VtcHR5KCkpIHsKLSAgICAgICAgICAgICAgICBsYXllci0+ZHJhdyhjbGlwKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6dW5sb2NrQ2xpZW50cygpCi17Ci0gICAgY29uc3QgTGF5ZXJWZWN0b3ImIGRyYXdpbmdMYXllcnMobURyYXdpbmdTdGF0ZS5sYXllcnNTb3J0ZWRCeVopOwotICAgIGNvbnN0IHNpemVfdCBjb3VudCA9IGRyYXdpbmdMYXllcnMuc2l6ZSgpOwotICAgIExheWVyQmFzZSogY29uc3QqIGNvbnN0IGxheWVycyA9IGRyYXdpbmdMYXllcnMuYXJyYXkoKTsKLSAgICBmb3IgKHNpemVfdCBpPTAgOyBpPGNvdW50IDsgKytpKSB7Ci0gICAgICAgIExheWVyQmFzZSogY29uc3QgbGF5ZXIgPSBsYXllcnNbaV07Ci0gICAgICAgIGxheWVyLT5maW5pc2hQYWdlRmxpcCgpOwotICAgIH0KLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6c2NoZWR1bGVCcm9hZGNhc3QoQ2xpZW50KiBjbGllbnQpCi17Ci0gICAgaWYgKG1MYXN0U2NoZWR1bGVkQnJvYWRjYXN0ICE9IGNsaWVudCkgewotICAgICAgICBtTGFzdFNjaGVkdWxlZEJyb2FkY2FzdCA9IGNsaWVudDsKLSAgICAgICAgbVNjaGVkdWxlZEJyb2FkY2FzdHMuYWRkKGNsaWVudCk7Ci0gICAgfQotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjpleGVjdXRlU2NoZWR1bGVkQnJvYWRjYXN0cygpCi17Ci0gICAgU29ydGVkVmVjdG9yPENsaWVudCo+JiBsaXN0ID0gbVNjaGVkdWxlZEJyb2FkY2FzdHM7Ci0gICAgc2l6ZV90IGNvdW50ID0gbGlzdC5zaXplKCk7Ci0gICAgd2hpbGUgKGNvdW50LS0pIHsKLSAgICAgICAgcGVyX2NsaWVudF9jYmxrX3QqIGNvbnN0IGNibGsgPSBsaXN0W2NvdW50XS0+Y3RybGJsazsKLSAgICAgICAgaWYgKGNibGstPmxvY2sudHJ5TG9jaygpID09IE5PX0VSUk9SKSB7Ci0gICAgICAgICAgICBjYmxrLT5jdi5icm9hZGNhc3QoKTsKLSAgICAgICAgICAgIGxpc3QucmVtb3ZlQXQoY291bnQpOwotICAgICAgICAgICAgY2Jsay0+bG9jay51bmxvY2soKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIHNjaGVkdWxlIGFub3RoZXIgcm91bmQKLSAgICAgICAgICAgIExPR1coImV4ZWN1dGVTY2hlZHVsZWRCcm9hZGNhc3RzKCkgc2tpcHBlZCwgIgotICAgICAgICAgICAgICAgICJjb250ZW50aW9uIG9uIHRoZSBjbGllbnQuIFdlJ2xsIHRyeSBhZ2FpbiBsYXRlci4uLiIpOwotICAgICAgICAgICAgc2lnbmFsRGVsYXllZEV2ZW50KG1zMm5zKDQpKTsKLSAgICAgICAgfQotICAgIH0KLSAgICBtTGFzdFNjaGVkdWxlZEJyb2FkY2FzdCA9IDA7Ci19Ci0KLXZvaWQgU3VyZmFjZUZsaW5nZXI6OmhhbmRsZURlYnVnQ3B1KCkKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobURlYnVnTG9jayk7Ci0gICAgaWYgKG1DcHVHYXVnZSAhPSAwKQotICAgICAgICBtQ3B1R2F1Z2UtPnNhbXBsZSgpOwotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjpkZWJ1Z0ZsYXNoUmVnaW9ucygpCi17Ci0gICAgaWYgKFVOTElLRUxZKCFtRGlydHlSZWdpb24uaXNSZWN0KCkpKSB7Ci0gICAgICAgIC8vIFRPRE86IGRvIHRoaXMgb25seSBpZiB3ZSBkb24ndCBoYXZlIHByZXNlcnZpbmcKLSAgICAgICAgLy8gc3dhcEJ1ZmZlci4gSWYgd2UgZG9uJ3QgaGF2ZSB1cGRhdGUtb24tZGVtYW5kLAotICAgICAgICAvLyByZWRyYXcgZXZlcnl0aGluZy4KLSAgICAgICAgY29tcG9zZVN1cmZhY2VzKFJlZ2lvbihtRGlydHlSZWdpb24uYm91bmRzKCkpKTsKLSAgICB9Ci0KLSAgICBnbERpc2FibGUoR0xfVEVYVFVSRV8yRCk7Ci0gICAgZ2xEaXNhYmxlKEdMX0JMRU5EKTsKLSAgICBnbERpc2FibGUoR0xfRElUSEVSKTsKLSAgICBnbERpc2FibGUoR0xfU0NJU1NPUl9URVNUKTsKLQotICAgIGdsQ29sb3I0eCgweDEwMDAwLCAwLCAweDEwMDAwLCAweDEwMDAwKTsKLQotICAgIFJlY3QgcjsKLSAgICBSZWdpb246Oml0ZXJhdG9yIGl0ZXJhdG9yKG1EaXJ0eVJlZ2lvbik7Ci0gICAgd2hpbGUgKGl0ZXJhdG9yLml0ZXJhdGUoJnIpKSB7Ci0gICAgICAgIEdMZmxvYXQgdmVydGljZXNbXVsyXSA9IHsKLSAgICAgICAgICAgICAgICB7IHIubGVmdCwgIHIudG9wIH0sCi0gICAgICAgICAgICAgICAgeyByLmxlZnQsICByLmJvdHRvbSB9LAotICAgICAgICAgICAgICAgIHsgci5yaWdodCwgci5ib3R0b20gfSwKLSAgICAgICAgICAgICAgICB7IHIucmlnaHQsIHIudG9wIH0KLSAgICAgICAgfTsKLSAgICAgICAgZ2xWZXJ0ZXhQb2ludGVyKDIsIEdMX0ZMT0FULCAwLCB2ZXJ0aWNlcyk7Ci0gICAgICAgIGdsRHJhd0FycmF5cyhHTF9UUklBTkdMRV9GQU4sIDAsIDQpOwotICAgIH0KLQotICAgIGNvbnN0IERpc3BsYXlIYXJkd2FyZSYgaHcoZ3JhcGhpY1BsYW5lKDApLmRpc3BsYXlIYXJkd2FyZSgpKTsKLSAgICBody5mbGlwKG1EaXJ0eVJlZ2lvbi5tZXJnZShtSW52YWxpZFJlZ2lvbikpOwotICAgIG1JbnZhbGlkUmVnaW9uLmNsZWFyKCk7Ci0KLSAgICBpZiAobURlYnVnUmVnaW9uID4gMSkKLSAgICAgICB1c2xlZXAobURlYnVnUmVnaW9uICogMTAwMCk7Ci0KLSAgICBnbEVuYWJsZShHTF9TQ0lTU09SX1RFU1QpOwotICAgIC8vbURpcnR5UmVnaW9uLmR1bXAoIm1EaXJ0eVJlZ2lvbiIpOwotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjpkcmF3V29ybWhvbGUoKSBjb25zdAotewotICAgIGNvbnN0IFJlZ2lvbiByZWdpb24obVdvcm1ob2xlUmVnaW9uLmludGVyc2VjdChtRGlydHlSZWdpb24pKTsKLSAgICBpZiAocmVnaW9uLmlzRW1wdHkoKSkKLSAgICAgICAgcmV0dXJuOwotCi0gICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiBodyhncmFwaGljUGxhbmUoMCkuZGlzcGxheUhhcmR3YXJlKCkpOwotICAgIGNvbnN0IGludDMyX3Qgd2lkdGggPSBody5nZXRXaWR0aCgpOwotICAgIGNvbnN0IGludDMyX3QgaGVpZ2h0ID0gaHcuZ2V0SGVpZ2h0KCk7Ci0KLSAgICBnbERpc2FibGUoR0xfQkxFTkQpOwotICAgIGdsRGlzYWJsZShHTF9ESVRIRVIpOwotCi0gICAgaWYgKExJS0VMWSghbURlYnVnQmFja2dyb3VuZCkpIHsKLSAgICAgICAgZ2xDbGVhckNvbG9yeCgwLDAsMCwwKTsKLSAgICAgICAgUmVjdCByOwotICAgICAgICBSZWdpb246Oml0ZXJhdG9yIGl0ZXJhdG9yKHJlZ2lvbik7Ci0gICAgICAgIHdoaWxlIChpdGVyYXRvci5pdGVyYXRlKCZyKSkgewotICAgICAgICAgICAgY29uc3QgR0xpbnQgc3kgPSBoZWlnaHQgLSAoci50b3AgKyByLmhlaWdodCgpKTsKLSAgICAgICAgICAgIGdsU2Npc3NvcihyLmxlZnQsIHN5LCByLndpZHRoKCksIHIuaGVpZ2h0KCkpOwotICAgICAgICAgICAgZ2xDbGVhcihHTF9DT0xPUl9CVUZGRVJfQklUKTsKLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIGNvbnN0IEdMc2hvcnQgdmVydGljZXNbXVsyXSA9IHsgeyAwLCAwIH0sIHsgd2lkdGgsIDAgfSwKLSAgICAgICAgICAgICAgICB7IHdpZHRoLCBoZWlnaHQgfSwgeyAwLCBoZWlnaHQgfSAgfTsKLSAgICAgICAgY29uc3QgR0xzaG9ydCB0Y29vcmRzW11bMl0gPSB7IHsgMCwgMCB9LCB7IDEsIDAgfSwgIHsgMSwgMSB9LCB7IDAsIDEgfSB9OwotICAgICAgICBnbFZlcnRleFBvaW50ZXIoMiwgR0xfU0hPUlQsIDAsIHZlcnRpY2VzKTsKLSAgICAgICAgZ2xUZXhDb29yZFBvaW50ZXIoMiwgR0xfU0hPUlQsIDAsIHRjb29yZHMpOwotICAgICAgICBnbEVuYWJsZUNsaWVudFN0YXRlKEdMX1RFWFRVUkVfQ09PUkRfQVJSQVkpOwotICAgICAgICBnbEVuYWJsZShHTF9URVhUVVJFXzJEKTsKLSAgICAgICAgZ2xCaW5kVGV4dHVyZShHTF9URVhUVVJFXzJELCBtV29ybWhvbGVUZXhOYW1lKTsKLSAgICAgICAgZ2xUZXhFbnZ4KEdMX1RFWFRVUkVfRU5WLCBHTF9URVhUVVJFX0VOVl9NT0RFLCBHTF9SRVBMQUNFKTsKLSAgICAgICAgZ2xNYXRyaXhNb2RlKEdMX1RFWFRVUkUpOwotICAgICAgICBnbExvYWRJZGVudGl0eSgpOwotICAgICAgICBnbFNjYWxlZih3aWR0aCooMS4wZi8zMi4wZiksIGhlaWdodCooMS4wZi8zMi4wZiksIDEpOwotICAgICAgICBSZWN0IHI7Ci0gICAgICAgIFJlZ2lvbjo6aXRlcmF0b3IgaXRlcmF0b3IocmVnaW9uKTsKLSAgICAgICAgd2hpbGUgKGl0ZXJhdG9yLml0ZXJhdGUoJnIpKSB7Ci0gICAgICAgICAgICBjb25zdCBHTGludCBzeSA9IGhlaWdodCAtIChyLnRvcCArIHIuaGVpZ2h0KCkpOwotICAgICAgICAgICAgZ2xTY2lzc29yKHIubGVmdCwgc3ksIHIud2lkdGgoKSwgci5oZWlnaHQoKSk7Ci0gICAgICAgICAgICBnbERyYXdBcnJheXMoR0xfVFJJQU5HTEVfRkFOLCAwLCA0KTsKLSAgICAgICAgfQotICAgICAgICBnbERpc2FibGVDbGllbnRTdGF0ZShHTF9URVhUVVJFX0NPT1JEX0FSUkFZKTsKLSAgICB9Ci19Ci0KLXZvaWQgU3VyZmFjZUZsaW5nZXI6OmRlYnVnU2hvd0ZQUygpIGNvbnN0Ci17Ci0gICAgc3RhdGljIGludCBtRnJhbWVDb3VudDsKLSAgICBzdGF0aWMgaW50IG1MYXN0RnJhbWVDb3VudCA9IDA7Ci0gICAgc3RhdGljIG5zZWNzX3QgbUxhc3RGcHNUaW1lID0gMDsKLSAgICBzdGF0aWMgZmxvYXQgbUZwcyA9IDA7Ci0gICAgbUZyYW1lQ291bnQrKzsKLSAgICBuc2Vjc190IG5vdyA9IHN5c3RlbVRpbWUoKTsKLSAgICBuc2Vjc190IGRpZmYgPSBub3cgLSBtTGFzdEZwc1RpbWU7Ci0gICAgaWYgKGRpZmYgPiBtczJucygyNTApKSB7Ci0gICAgICAgIG1GcHMgPSAgKChtRnJhbWVDb3VudCAtIG1MYXN0RnJhbWVDb3VudCkgKiBmbG9hdChzMm5zKDEpKSkgLyBkaWZmOwotICAgICAgICBtTGFzdEZwc1RpbWUgPSBub3c7Ci0gICAgICAgIG1MYXN0RnJhbWVDb3VudCA9IG1GcmFtZUNvdW50OwotICAgIH0KLSAgICAvLyBYWFg6IG1GUFMgaGFzIHRoZSB2YWx1ZSB3ZSB3YW50Ci0gfQotCi1zdGF0dXNfdCBTdXJmYWNlRmxpbmdlcjo6YWRkTGF5ZXIoTGF5ZXJCYXNlKiBsYXllcikKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobVN0YXRlTG9jayk7Ci0gICAgYWRkTGF5ZXJfbChsYXllcik7Ci0gICAgc2V0VHJhbnNhY3Rpb25GbGFncyhlVHJhbnNhY3Rpb25OZWVkZWR8ZVRyYXZlcnNhbE5lZWRlZCk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlRmxpbmdlcjo6cmVtb3ZlTGF5ZXIoTGF5ZXJCYXNlKiBsYXllcikKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobVN0YXRlTG9jayk7Ci0gICAgcmVtb3ZlTGF5ZXJfbChsYXllcik7Ci0gICAgc2V0VHJhbnNhY3Rpb25GbGFncyhlVHJhbnNhY3Rpb25OZWVkZWQpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZUZsaW5nZXI6OmludmFsaWRhdGVMYXllclZpc2liaWxpdHkoTGF5ZXJCYXNlKiBsYXllcikKLXsKLSAgICBsYXllci0+Zm9yY2VWaXNpYmlsaXR5VHJhbnNhY3Rpb24oKTsKLSAgICBzZXRUcmFuc2FjdGlvbkZsYWdzKGVUcmF2ZXJzYWxOZWVkZWQpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZUZsaW5nZXI6OmFkZExheWVyX2woTGF5ZXJCYXNlKiBsYXllcikKLXsKLSAgICBzc2l6ZV90IGkgPSBtQ3VycmVudFN0YXRlLmxheWVyc1NvcnRlZEJ5Wi5hZGQoCi0gICAgICAgICAgICAgICAgbGF5ZXIsICZMYXllckJhc2U6OmNvbXBhcmVDdXJyZW50U3RhdGVaKTsKLSAgICBMYXllckJhc2VDbGllbnQqIGxiYyA9IExheWVyQmFzZTo6ZHluYW1pY0Nhc3Q8TGF5ZXJCYXNlQ2xpZW50Kj4obGF5ZXIpOwotICAgIGlmIChsYmMpIHsKLSAgICAgICAgbUxheWVyTWFwLmFkZChsYmMtPnNlcnZlckluZGV4KCksIGxiYyk7Ci0gICAgfQotICAgIG1SZW1vdmVkTGF5ZXJzLnJlbW92ZShsYXllcik7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlRmxpbmdlcjo6cmVtb3ZlTGF5ZXJfbChMYXllckJhc2UqIGxheWVyQmFzZSkKLXsKLSAgICBzc2l6ZV90IGluZGV4ID0gbUN1cnJlbnRTdGF0ZS5sYXllcnNTb3J0ZWRCeVoucmVtb3ZlKGxheWVyQmFzZSk7Ci0gICAgaWYgKGluZGV4ID49IDApIHsKLSAgICAgICAgbVJlbW92ZWRMYXllcnMuYWRkKGxheWVyQmFzZSk7Ci0gICAgICAgIExheWVyQmFzZUNsaWVudCogbGF5ZXIgPSBMYXllckJhc2U6OmR5bmFtaWNDYXN0PExheWVyQmFzZUNsaWVudCo+KGxheWVyQmFzZSk7Ci0gICAgICAgIGlmIChsYXllcikgewotICAgICAgICAgICAgbUxheWVyTWFwLnJlbW92ZUl0ZW0obGF5ZXItPnNlcnZlckluZGV4KCkpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgLy8gaXQncyBwb3NzaWJsZSB0aGF0IHdlIGRvbid0IGZpbmQgYSBsYXllciwgYmVjYXVzZSBpdCBtaWdodAotICAgIC8vIGhhdmUgYmVlbiBkZXN0cm95ZWQgYWxyZWFkeSAtLSB0aGlzIGlzIG5vdCB0ZWNobmljYWxseSBhbiBlcnJvcgotICAgIC8vIGZyb20gdGhlIHVzZXIgYmVjYXVzZSB0aGVyZSBpcyBhIHJhY2UgYmV0d2VlbiBkZXN0cm95U3VyZmFjZSwKLSAgICAvLyBkZXN0cm95Y2xpZW50IGFuZCBkZXN0cm95U3VyZmFjZS1mcm9tLWEtdHJhbnNhY3Rpb24uCi0gICAgcmV0dXJuIChpbmRleCA9PSBOQU1FX05PVF9GT1VORCkgPyBzdGF0dXNfdChOT19FUlJPUikgOiBpbmRleDsKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6ZnJlZV9yZXNvdXJjZXNfbCgpCi17Ci0gICAgLy8gRGVzdHJveSBsYXllcnMgdGhhdCB3ZXJlIHJlbW92ZWQKLSAgICBkZXN0cm95X2FsbF9yZW1vdmVkX2xheWVyc19sKCk7Ci0KLSAgICAvLyBmcmVlIHJlc291cmNlcyBhc3NvY2lhdGVkIHdpdGggZGlzY29ubmVjdGVkIGNsaWVudHMKLSAgICBTb3J0ZWRWZWN0b3I8Q2xpZW50Kj4mIHNjaGVkdWxlZEJyb2FkY2FzdHMobVNjaGVkdWxlZEJyb2FkY2FzdHMpOwotICAgIFZlY3RvcjxDbGllbnQqPiYgZGlzY29ubmVjdGVkQ2xpZW50cyhtRGlzY29ubmVjdGVkQ2xpZW50cyk7Ci0gICAgY29uc3Qgc2l6ZV90IGNvdW50ID0gZGlzY29ubmVjdGVkQ2xpZW50cy5zaXplKCk7Ci0gICAgZm9yIChzaXplX3QgaT0wIDsgaTxjb3VudCA7IGkrKykgewotICAgICAgICBDbGllbnQqIGNsaWVudCA9IGRpc2Nvbm5lY3RlZENsaWVudHNbaV07Ci0gICAgICAgIC8vIGlmIHRoaXMgY2xpZW50IGlzIHRoZSBzY2hlZHVsZWQgYnJvYWRjYXN0IGxpc3QsCi0gICAgICAgIC8vIHJlbW92ZSBpdCBmcm9tIHRoZXJlIChhbmQgd2UgZG9uJ3QgbmVlZCB0byBzaWduYWwgaXQKLSAgICAgICAgLy8gc2luY2UgaXQgaXMgZGVhZCkuCi0gICAgICAgIGludDMyX3QgaW5kZXggPSBzY2hlZHVsZWRCcm9hZGNhc3RzLmluZGV4T2YoY2xpZW50KTsKLSAgICAgICAgaWYgKGluZGV4ID49IDApIHsKLSAgICAgICAgICAgIHNjaGVkdWxlZEJyb2FkY2FzdHMucmVtb3ZlSXRlbXNBdChpbmRleCk7Ci0gICAgICAgIH0KLSAgICAgICAgbVRva2Vucy5yZWxlYXNlKGNsaWVudC0+Y2lkKTsKLSAgICAgICAgZGVsZXRlIGNsaWVudDsKLSAgICB9Ci0gICAgZGlzY29ubmVjdGVkQ2xpZW50cy5jbGVhcigpOwotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjpkZXN0cm95X2FsbF9yZW1vdmVkX2xheWVyc19sKCkKLXsKLSAgICBzaXplX3QgYyA9IG1SZW1vdmVkTGF5ZXJzLnNpemUoKTsKLSAgICB3aGlsZSAoYy0tKSB7Ci0gICAgICAgIExheWVyQmFzZSogY29uc3QgcmVtb3ZlZF9sYXllciA9IG1SZW1vdmVkTGF5ZXJzW2NdOwotCi0gICAgICAgIExPR0VfSUYobUN1cnJlbnRTdGF0ZS5sYXllcnNTb3J0ZWRCeVouaW5kZXhPZihyZW1vdmVkX2xheWVyKSA+PSAwLAotICAgICAgICAgICAgImxheWVyICVwIHJlbW92ZWQgYnV0IHN0aWxsIGluIHRoZSBjdXJyZW50IHN0YXRlIGxpc3QiLAotICAgICAgICAgICAgcmVtb3ZlZF9sYXllcik7Ci0KLSAgICAgICAgZGVsZXRlIHJlbW92ZWRfbGF5ZXI7Ci0gICAgfQotICAgIG1SZW1vdmVkTGF5ZXJzLmNsZWFyKCk7Ci19Ci0KLQotdWludDMyX3QgU3VyZmFjZUZsaW5nZXI6OmdldFRyYW5zYWN0aW9uRmxhZ3ModWludDMyX3QgZmxhZ3MpCi17Ci0gICAgcmV0dXJuIGFuZHJvaWRfYXRvbWljX2FuZCh+ZmxhZ3MsICZtVHJhbnNhY3Rpb25GbGFncykgJiBmbGFnczsKLX0KLQotdWludDMyX3QgU3VyZmFjZUZsaW5nZXI6OnNldFRyYW5zYWN0aW9uRmxhZ3ModWludDMyX3QgZmxhZ3MsIG5zZWNzX3QgZGVsYXkpCi17Ci0gICAgdWludDMyX3Qgb2xkID0gYW5kcm9pZF9hdG9taWNfb3IoZmxhZ3MsICZtVHJhbnNhY3Rpb25GbGFncyk7Ci0gICAgaWYgKChvbGQgJiBmbGFncyk9PTApIHsgLy8gd2FrZSB0aGUgc2VydmVyIHVwCi0gICAgICAgIGlmIChkZWxheSA+IDApIHsKLSAgICAgICAgICAgIHNpZ25hbERlbGF5ZWRFdmVudChkZWxheSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBzaWduYWxFdmVudCgpOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBvbGQ7Ci19Ci0KLXZvaWQgU3VyZmFjZUZsaW5nZXI6Om9wZW5HbG9iYWxUcmFuc2FjdGlvbigpCi17Ci0gICAgYW5kcm9pZF9hdG9taWNfaW5jKCZtVHJhbnNhY3Rpb25Db3VudCk7Ci19Ci0KLXZvaWQgU3VyZmFjZUZsaW5nZXI6OmNsb3NlR2xvYmFsVHJhbnNhY3Rpb24oKQotewotICAgIGlmIChhbmRyb2lkX2F0b21pY19kZWMoJm1UcmFuc2FjdGlvbkNvdW50KSA9PSAxKSB7Ci0gICAgICAgIHNpZ25hbEV2ZW50KCk7Ci0gICAgfQotfQotCi1zdGF0dXNfdCBTdXJmYWNlRmxpbmdlcjo6ZnJlZXplRGlzcGxheShEaXNwbGF5SUQgZHB5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBpZiAoVU5MSUtFTFkodWludDMyX3QoZHB5KSA+PSBESVNQTEFZX0NPVU5UKSkKLSAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLQotICAgIE11dGV4OjpBdXRvbG9jayBfbChtU3RhdGVMb2NrKTsKLSAgICBtQ3VycmVudFN0YXRlLmZyZWV6ZURpc3BsYXkgPSAxOwotICAgIHNldFRyYW5zYWN0aW9uRmxhZ3MoZVRyYW5zYWN0aW9uTmVlZGVkKTsKLQotICAgIC8vIGZsYWdzIGlzIGludGVuZGVkIHRvIGNvbW11bmljYXRlIHNvbWUgc29ydCBvZiBhbmltYXRpb24gYmVoYXZpb3IKLSAgICAvLyAoZm9yIGluc3RhbmNlIGZhZGRpbmcpCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlRmxpbmdlcjo6dW5mcmVlemVEaXNwbGF5KERpc3BsYXlJRCBkcHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIGlmIChVTkxJS0VMWSh1aW50MzJfdChkcHkpID49IERJU1BMQVlfQ09VTlQpKQotICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOwotCi0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1TdGF0ZUxvY2spOwotICAgIG1DdXJyZW50U3RhdGUuZnJlZXplRGlzcGxheSA9IDA7Ci0gICAgc2V0VHJhbnNhY3Rpb25GbGFncyhlVHJhbnNhY3Rpb25OZWVkZWQpOwotCi0gICAgLy8gZmxhZ3MgaXMgaW50ZW5kZWQgdG8gY29tbXVuaWNhdGUgc29tZSBzb3J0IG9mIGFuaW1hdGlvbiBiZWhhdmlvcgotICAgIC8vIChmb3IgaW5zdGFuY2UgZmFkZGluZykKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLWludCBTdXJmYWNlRmxpbmdlcjo6c2V0T3JpZW50YXRpb24oRGlzcGxheUlEIGRweSwgaW50IG9yaWVudGF0aW9uKQotewotICAgIGlmIChVTkxJS0VMWSh1aW50MzJfdChkcHkpID49IERJU1BMQVlfQ09VTlQpKQotICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOwotCi0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1TdGF0ZUxvY2spOwotICAgIGlmIChtQ3VycmVudFN0YXRlLm9yaWVudGF0aW9uICE9IG9yaWVudGF0aW9uKSB7Ci0gICAgICAgIGlmICh1aW50MzJfdChvcmllbnRhdGlvbik8PWVPcmllbnRhdGlvbjI3MCB8fCBvcmllbnRhdGlvbj09NDIpIHsKLSAgICAgICAgICAgIG1DdXJyZW50U3RhdGUub3JpZW50YXRpb24gPSBvcmllbnRhdGlvbjsKLSAgICAgICAgICAgIHNldFRyYW5zYWN0aW9uRmxhZ3MoZVRyYW5zYWN0aW9uTmVlZGVkKTsKLSAgICAgICAgICAgIG1UcmFuc2FjdGlvbkNWLndhaXQobVN0YXRlTG9jayk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBvcmllbnRhdGlvbiA9IEJBRF9WQUxVRTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gb3JpZW50YXRpb247Ci19Ci0KLXNwPElTdXJmYWNlPiBTdXJmYWNlRmxpbmdlcjo6Y3JlYXRlU3VyZmFjZShDbGllbnRJRCBjbGllbnRJZCwgaW50IHBpZCwKLSAgICAgICAgSVN1cmZhY2VGbGluZ2VyQ2xpZW50OjpzdXJmYWNlX2RhdGFfdCogcGFyYW1zLAotICAgICAgICBEaXNwbGF5SUQgZCwgdWludDMyX3QgdywgdWludDMyX3QgaCwgUGl4ZWxGb3JtYXQgZm9ybWF0LAotICAgICAgICB1aW50MzJfdCBmbGFncykKLXsKLSAgICBMYXllckJhc2VDbGllbnQqIGxheWVyID0gMDsKLSAgICBzcDxMYXllckJhc2VDbGllbnQ6OlN1cmZhY2U+IHN1cmZhY2VIYW5kbGU7Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1TdGF0ZUxvY2spOwotICAgIENsaWVudCogY29uc3QgYyA9IG1DbGllbnRzTWFwLnZhbHVlRm9yKGNsaWVudElkKTsKLSAgICBpZiAoVU5MSUtFTFkoIWMpKSB7Ci0gICAgICAgIExPR0UoImNyZWF0ZVN1cmZhY2UoKSBmYWlsZWQsIGNsaWVudCBub3QgZm91bmQgKGlkPSVkKSIsIGNsaWVudElkKTsKLSAgICAgICAgcmV0dXJuIHN1cmZhY2VIYW5kbGU7Ci0gICAgfQotCi0gICAgLy9MT0dEKCJjcmVhdGVTdXJmYWNlIGZvciBwaWQgJWQgKCVkIHggJWQpIiwgcGlkLCB3LCBoKTsKLSAgICBpbnQzMl90IGlkID0gYy0+Z2VuZXJhdGVJZChwaWQpOwotICAgIGlmICh1aW50MzJfdChpZCkgPj0gTlVNX0xBWUVSU19NQVgpIHsKLSAgICAgICAgTE9HRSgiY3JlYXRlU3VyZmFjZSgpIGZhaWxlZCwgZ2VuZXJhdGVJZCA9ICVkIiwgaWQpOwotICAgICAgICByZXR1cm4gc3VyZmFjZUhhbmRsZTsKLSAgICB9Ci0KLSAgICBzd2l0Y2ggKGZsYWdzICYgZUZYU3VyZmFjZU1hc2spIHsKLSAgICAgICAgY2FzZSBlRlhTdXJmYWNlTm9ybWFsOgotICAgICAgICAgICAgaWYgKFVOTElLRUxZKGZsYWdzICYgZVB1c2hCdWZmZXJzKSkgewotICAgICAgICAgICAgICAgIGxheWVyID0gY3JlYXRlUHVzaEJ1ZmZlcnNTdXJmYWNlTG9ja2VkKGMsIGQsIGlkLCB3LCBoLCBmbGFncyk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGxheWVyID0gY3JlYXRlTm9ybWFsU3VyZmFjZUxvY2tlZChjLCBkLCBpZCwgdywgaCwgZm9ybWF0LCBmbGFncyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBlRlhTdXJmYWNlQmx1cjoKLSAgICAgICAgICAgIGxheWVyID0gY3JlYXRlQmx1clN1cmZhY2VMb2NrZWQoYywgZCwgaWQsIHcsIGgsIGZsYWdzKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIGVGWFN1cmZhY2VEaW06Ci0gICAgICAgICAgICBsYXllciA9IGNyZWF0ZURpbVN1cmZhY2VMb2NrZWQoYywgZCwgaWQsIHcsIGgsIGZsYWdzKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgIH0KLQotICAgIGlmIChsYXllcikgewotICAgICAgICBzZXRUcmFuc2FjdGlvbkZsYWdzKGVUcmFuc2FjdGlvbk5lZWRlZCk7Ci0gICAgICAgIHN1cmZhY2VIYW5kbGUgPSBsYXllci0+Z2V0U3VyZmFjZSgpOwotICAgICAgICBpZiAoc3VyZmFjZUhhbmRsZSAhPSAwKQotICAgICAgICAgICAgc3VyZmFjZUhhbmRsZS0+Z2V0U3VyZmFjZURhdGEocGFyYW1zKTsKLSAgICB9Ci0KLSAgICByZXR1cm4gc3VyZmFjZUhhbmRsZTsKLX0KLQotTGF5ZXJCYXNlQ2xpZW50KiBTdXJmYWNlRmxpbmdlcjo6Y3JlYXRlTm9ybWFsU3VyZmFjZUxvY2tlZCgKLSAgICAgICAgQ2xpZW50KiBjbGllbnQsIERpc3BsYXlJRCBkaXNwbGF5LAotICAgICAgICBpbnQzMl90IGlkLCB1aW50MzJfdCB3LCB1aW50MzJfdCBoLCBQaXhlbEZvcm1hdCBmb3JtYXQsIHVpbnQzMl90IGZsYWdzKQotewotICAgIC8vIGluaXRpYWxpemUgdGhlIHN1cmZhY2VzCi0gICAgc3dpdGNoIChmb3JtYXQpIHsgLy8gVE9ETzogdGFrZSBoL3cgaW50byBhY2NvdW50Ci0gICAgY2FzZSBQSVhFTF9GT1JNQVRfVFJBTlNQQVJFTlQ6Ci0gICAgY2FzZSBQSVhFTF9GT1JNQVRfVFJBTlNMVUNFTlQ6Ci0gICAgICAgIGZvcm1hdCA9IFBJWEVMX0ZPUk1BVF9SR0JBXzg4ODg7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgUElYRUxfRk9STUFUX09QQVFVRToKLSAgICAgICAgZm9ybWF0ID0gUElYRUxfRk9STUFUX1JHQl81NjU7Ci0gICAgICAgIGJyZWFrOwotICAgIH0KLQotICAgIExheWVyKiBsYXllciA9IG5ldyBMYXllcih0aGlzLCBkaXNwbGF5LCBjbGllbnQsIGlkKTsKLSAgICBzdGF0dXNfdCBlcnIgPSBsYXllci0+c2V0QnVmZmVycyhjbGllbnQsIHcsIGgsIGZvcm1hdCwgZmxhZ3MpOwotICAgIGlmIChMSUtFTFkoZXJyID09IE5PX0VSUk9SKSkgewotICAgICAgICBsYXllci0+aW5pdFN0YXRlcyh3LCBoLCBmbGFncyk7Ci0gICAgICAgIGFkZExheWVyX2wobGF5ZXIpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIExPR0UoImNyZWF0ZU5vcm1hbFN1cmZhY2VMb2NrZWQoKSBmYWlsZWQgKCVzKSIsIHN0cmVycm9yKC1lcnIpKTsKLSAgICAgICAgZGVsZXRlIGxheWVyOwotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0gICAgcmV0dXJuIGxheWVyOwotfQotCi1MYXllckJhc2VDbGllbnQqIFN1cmZhY2VGbGluZ2VyOjpjcmVhdGVCbHVyU3VyZmFjZUxvY2tlZCgKLSAgICAgICAgQ2xpZW50KiBjbGllbnQsIERpc3BsYXlJRCBkaXNwbGF5LAotICAgICAgICBpbnQzMl90IGlkLCB1aW50MzJfdCB3LCB1aW50MzJfdCBoLCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBMYXllckJsdXIqIGxheWVyID0gbmV3IExheWVyQmx1cih0aGlzLCBkaXNwbGF5LCBjbGllbnQsIGlkKTsKLSAgICBsYXllci0+aW5pdFN0YXRlcyh3LCBoLCBmbGFncyk7Ci0gICAgYWRkTGF5ZXJfbChsYXllcik7Ci0gICAgcmV0dXJuIGxheWVyOwotfQotCi1MYXllckJhc2VDbGllbnQqIFN1cmZhY2VGbGluZ2VyOjpjcmVhdGVEaW1TdXJmYWNlTG9ja2VkKAotICAgICAgICBDbGllbnQqIGNsaWVudCwgRGlzcGxheUlEIGRpc3BsYXksCi0gICAgICAgIGludDMyX3QgaWQsIHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIHVpbnQzMl90IGZsYWdzKQotewotICAgIExheWVyRGltKiBsYXllciA9IG5ldyBMYXllckRpbSh0aGlzLCBkaXNwbGF5LCBjbGllbnQsIGlkKTsKLSAgICBsYXllci0+aW5pdFN0YXRlcyh3LCBoLCBmbGFncyk7Ci0gICAgYWRkTGF5ZXJfbChsYXllcik7Ci0gICAgcmV0dXJuIGxheWVyOwotfQotCi1MYXllckJhc2VDbGllbnQqIFN1cmZhY2VGbGluZ2VyOjpjcmVhdGVQdXNoQnVmZmVyc1N1cmZhY2VMb2NrZWQoCi0gICAgICAgIENsaWVudCogY2xpZW50LCBEaXNwbGF5SUQgZGlzcGxheSwKLSAgICAgICAgaW50MzJfdCBpZCwgdWludDMyX3QgdywgdWludDMyX3QgaCwgdWludDMyX3QgZmxhZ3MpCi17Ci0gICAgTGF5ZXJCdWZmZXIqIGxheWVyID0gbmV3IExheWVyQnVmZmVyKHRoaXMsIGRpc3BsYXksIGNsaWVudCwgaWQpOwotICAgIGxheWVyLT5pbml0U3RhdGVzKHcsIGgsIGZsYWdzKTsKLSAgICBhZGRMYXllcl9sKGxheWVyKTsKLSAgICByZXR1cm4gbGF5ZXI7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VGbGluZ2VyOjpkZXN0cm95U3VyZmFjZShTdXJmYWNlSUQgaW5kZXgpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1TdGF0ZUxvY2spOwotICAgIExheWVyQmFzZUNsaWVudCogY29uc3QgbGF5ZXIgPSBnZXRMYXllclVzZXJfbChpbmRleCk7Ci0gICAgc3RhdHVzX3QgZXJyID0gcmVtb3ZlTGF5ZXJfbChsYXllcik7Ci0gICAgaWYgKGVyciA8IDApCi0gICAgICAgIHJldHVybiBlcnI7Ci0gICAgc2V0VHJhbnNhY3Rpb25GbGFncyhlVHJhbnNhY3Rpb25OZWVkZWQpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZUZsaW5nZXI6OnNldENsaWVudFN0YXRlKAotICAgICAgICBDbGllbnRJRCBjaWQsCi0gICAgICAgIGludDMyX3QgY291bnQsCi0gICAgICAgIGNvbnN0IGxheWVyX3N0YXRlX3QqIHN0YXRlcykKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobVN0YXRlTG9jayk7Ci0gICAgdWludDMyX3QgZmxhZ3MgPSAwOwotICAgIGNpZCA8PD0gMTY7Ci0gICAgZm9yIChpbnQgaT0wIDsgaTxjb3VudCA7IGkrKykgewotICAgICAgICBjb25zdCBsYXllcl9zdGF0ZV90JiBzID0gc3RhdGVzW2ldOwotICAgICAgICBMYXllckJhc2VDbGllbnQqIGxheWVyID0gZ2V0TGF5ZXJVc2VyX2wocy5zdXJmYWNlIHwgY2lkKTsKLSAgICAgICAgaWYgKGxheWVyKSB7Ci0gICAgICAgICAgICBjb25zdCB1aW50MzJfdCB3aGF0ID0gcy53aGF0OwotICAgICAgICAgICAgLy8gY2hlY2sgaWYgaXQgaGFzIGJlZW4gZGVzdHJveWVkIGZpcnN0Ci0gICAgICAgICAgICBpZiAod2hhdCAmIGVEZXN0cm95ZWQpIHsKLSAgICAgICAgICAgICAgICBpZiAocmVtb3ZlTGF5ZXJfbChsYXllcikgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgICAgICAgICAgZmxhZ3MgfD0gZVRyYW5zYWN0aW9uTmVlZGVkOwotICAgICAgICAgICAgICAgICAgICAvLyB3ZSBza2lwIGV2ZXJ5dGhpbmcgZWxzZS4uLiB3ZWxsLCBubywgbm90IHJlYWxseQotICAgICAgICAgICAgICAgICAgICAvLyB3ZSBza2lwIE9OTFkgdGhhdCB0cmFuc2FjdGlvbi4KLSAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHdoYXQgJiBlUG9zaXRpb25DaGFuZ2VkKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGxheWVyLT5zZXRQb3NpdGlvbihzLngsIHMueSkpCi0gICAgICAgICAgICAgICAgICAgIGZsYWdzIHw9IGVUcmF2ZXJzYWxOZWVkZWQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAod2hhdCAmIGVMYXllckNoYW5nZWQpIHsKLSAgICAgICAgICAgICAgICBpZiAobGF5ZXItPnNldExheWVyKHMueikpIHsKLSAgICAgICAgICAgICAgICAgICAgbUN1cnJlbnRTdGF0ZS5sYXllcnNTb3J0ZWRCeVoucmVvcmRlcigKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBsYXllciwgJkxheWVyOjpjb21wYXJlQ3VycmVudFN0YXRlWik7Ci0gICAgICAgICAgICAgICAgICAgIC8vIHdlIG5lZWQgdHJhdmVyc2FsIChzdGF0ZSBjaGFuZ2VkKQotICAgICAgICAgICAgICAgICAgICAvLyBBTkQgdHJhbnNhY3Rpb24gKGxpc3QgY2hhbmdlZCkKLSAgICAgICAgICAgICAgICAgICAgZmxhZ3MgfD0gZVRyYW5zYWN0aW9uTmVlZGVkfGVUcmF2ZXJzYWxOZWVkZWQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHdoYXQgJiBlU2l6ZUNoYW5nZWQpIHsKLSAgICAgICAgICAgICAgICBpZiAobGF5ZXItPnNldFNpemUocy53LCBzLmgpKQotICAgICAgICAgICAgICAgICAgICBmbGFncyB8PSBlVHJhdmVyc2FsTmVlZGVkOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHdoYXQgJiBlQWxwaGFDaGFuZ2VkKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGxheWVyLT5zZXRBbHBoYSh1aW50OF90KDI1NS4wZipzLmFscGhhKzAuNWYpKSkKLSAgICAgICAgICAgICAgICAgICAgZmxhZ3MgfD0gZVRyYXZlcnNhbE5lZWRlZDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh3aGF0ICYgZU1hdHJpeENoYW5nZWQpIHsKLSAgICAgICAgICAgICAgICBpZiAobGF5ZXItPnNldE1hdHJpeChzLm1hdHJpeCkpCi0gICAgICAgICAgICAgICAgICAgIGZsYWdzIHw9IGVUcmF2ZXJzYWxOZWVkZWQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAod2hhdCAmIGVUcmFuc3BhcmVudFJlZ2lvbkNoYW5nZWQpIHsKLSAgICAgICAgICAgICAgICBpZiAobGF5ZXItPnNldFRyYW5zcGFyZW50UmVnaW9uSGludChzLnRyYW5zcGFyZW50UmVnaW9uKSkKLSAgICAgICAgICAgICAgICAgICAgZmxhZ3MgfD0gZVRyYXZlcnNhbE5lZWRlZDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICh3aGF0ICYgZVZpc2liaWxpdHlDaGFuZ2VkKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGxheWVyLT5zZXRGbGFncyhzLmZsYWdzLCBzLm1hc2spKQotICAgICAgICAgICAgICAgICAgICBmbGFncyB8PSBlVHJhdmVyc2FsTmVlZGVkOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIGlmIChmbGFncykgewotICAgICAgICBzZXRUcmFuc2FjdGlvbkZsYWdzKGZsYWdzKTsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1MYXllckJhc2VDbGllbnQqIFN1cmZhY2VGbGluZ2VyOjpnZXRMYXllclVzZXJfbChTdXJmYWNlSUQgcykgY29uc3QKLXsKLSAgICByZXR1cm4gbUxheWVyTWFwLnZhbHVlRm9yKHMpOwotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyOjpzY3JlZW5SZWxlYXNlZChpbnQgZHB5KQotewotICAgIC8vIHRoaXMgbWF5IGJlIGNhbGxlZCBieSBhIHNpZ25hbCBoYW5kbGVyLCB3ZSBjYW4ndCBkbyB0b28gbXVjaCBpbiBoZXJlCi0gICAgYW5kcm9pZF9hdG9taWNfb3IoZUNvbnNvbGVSZWxlYXNlZCwgJm1Db25zb2xlU2lnbmFscyk7Ci0gICAgc2lnbmFsRXZlbnQoKTsKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlcjo6c2NyZWVuQWNxdWlyZWQoaW50IGRweSkKLXsKLSAgICAvLyB0aGlzIG1heSBiZSBjYWxsZWQgYnkgYSBzaWduYWwgaGFuZGxlciwgd2UgY2FuJ3QgZG8gdG9vIG11Y2ggaW4gaGVyZQotICAgIGFuZHJvaWRfYXRvbWljX29yKGVDb25zb2xlQWNxdWlyZWQsICZtQ29uc29sZVNpZ25hbHMpOwotICAgIHNpZ25hbEV2ZW50KCk7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VGbGluZ2VyOjpkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBjb25zdCBzaXplX3QgU0laRSA9IDEwMjQ7Ci0gICAgY2hhciBidWZmZXJbU0laRV07Ci0gICAgU3RyaW5nOCByZXN1bHQ7Ci0gICAgaWYgKGNoZWNrQ2FsbGluZ1Blcm1pc3Npb24oCi0gICAgICAgICAgICBTdHJpbmcxNigiYW5kcm9pZC5wZXJtaXNzaW9uLkRVTVAiKSkgPT0gZmFsc2UpCi0gICAgeyAvLyBub3QgYWxsb3dlZAotICAgICAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICJQZXJtaXNzaW9uIERlbmlhbDogIgotICAgICAgICAgICAgICAgICJjYW4ndCBkdW1wIFN1cmZhY2VGbGluZ2VyIGZyb20gcGlkPSVkLCB1aWQ9JWRcbiIsCi0gICAgICAgICAgICAgICAgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Z2V0Q2FsbGluZ1BpZCgpLAotICAgICAgICAgICAgICAgIElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPmdldENhbGxpbmdVaWQoKSk7Ci0gICAgICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICB9IGVsc2UgewotICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2wobVN0YXRlTG9jayk7Ci0gICAgICAgIHNpemVfdCBzID0gbUNsaWVudHNNYXAuc2l6ZSgpOwotICAgICAgICBjaGFyIG5hbWVbNjRdOwotICAgICAgICBmb3IgKHNpemVfdCBpPTAgOyBpPHMgOyBpKyspIHsKLSAgICAgICAgICAgIENsaWVudCogY2xpZW50ID0gbUNsaWVudHNNYXAudmFsdWVBdChpKTsKLSAgICAgICAgICAgIHNwcmludGYobmFtZSwgIiAgQ2xpZW50IChpZD0weCUwOHgpIiwgY2xpZW50LT5jaWQpOwotICAgICAgICAgICAgY2xpZW50LT5kdW1wKG5hbWUpOwotICAgICAgICB9Ci0gICAgICAgIGNvbnN0IExheWVyVmVjdG9yJiBjdXJyZW50TGF5ZXJzID0gbUN1cnJlbnRTdGF0ZS5sYXllcnNTb3J0ZWRCeVo7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBjb3VudCA9IGN1cnJlbnRMYXllcnMuc2l6ZSgpOwotICAgICAgICBmb3IgKHNpemVfdCBpPTAgOyBpPGNvdW50IDsgaSsrKSB7Ci0gICAgICAgICAgICAvKioqIExheWVyQmFzZSAqKiovCi0gICAgICAgICAgICBMYXllckJhc2UgY29uc3QgKiBjb25zdCBsYXllciA9IGN1cnJlbnRMYXllcnNbaV07Ci0gICAgICAgICAgICBjb25zdCBMYXllcjo6U3RhdGUmIHMgPSBsYXllci0+ZHJhd2luZ1N0YXRlKCk7Ci0gICAgICAgICAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsCi0gICAgICAgICAgICAgICAgICAgICIrICVzICVwXG4iCi0gICAgICAgICAgICAgICAgICAgICIgICAgICAiCi0gICAgICAgICAgICAgICAgICAgICJ6PSU5ZCwgcG9zPSglNGQsJTRkKSwgc2l6ZT0oJTRkLCU0ZCksICIKLSAgICAgICAgICAgICAgICAgICAgIm5lZWRzQmxlbmRpbmc9JTFkLCBpbnZhbGlkYXRlPSUxZCwgIgotICAgICAgICAgICAgICAgICAgICAiYWxwaGE9MHglMDJ4LCBmbGFncz0weCUwOHgsIHRyPVslLjJmLCAlLjJmXVslLjJmLCAlLjJmXVxuIiwKLSAgICAgICAgICAgICAgICAgICAgbGF5ZXItPmdldFR5cGVJRCgpLCBsYXllciwKLSAgICAgICAgICAgICAgICAgICAgcy56LCBsYXllci0+dHgoKSwgbGF5ZXItPnR5KCksIHMudywgcy5oLAotICAgICAgICAgICAgICAgICAgICBsYXllci0+bmVlZHNCbGVuZGluZygpLCBsYXllci0+Y29udGVudERpcnR5LAotICAgICAgICAgICAgICAgICAgICBzLmFscGhhLCBzLmZsYWdzLAotICAgICAgICAgICAgICAgICAgICBzLnRyYW5zZm9ybVswXSwgcy50cmFuc2Zvcm1bMV0sCi0gICAgICAgICAgICAgICAgICAgIHMudHJhbnNmb3JtWzJdLCBzLnRyYW5zZm9ybVszXSk7Ci0gICAgICAgICAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgICAgICAgICBidWZmZXJbMF0gPSAwOwotICAgICAgICAgICAgLyoqKiBMYXllckJhc2VDbGllbnQgKioqLwotICAgICAgICAgICAgTGF5ZXJCYXNlQ2xpZW50KiBjb25zdCBsYmMgPQotICAgICAgICAgICAgICAgIExheWVyQmFzZTo6ZHluYW1pY0Nhc3Q8TGF5ZXJCYXNlQ2xpZW50Kj4oKExheWVyQmFzZSopbGF5ZXIpOwotICAgICAgICAgICAgaWYgKGxiYykgewotICAgICAgICAgICAgICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwKLSAgICAgICAgICAgICAgICAgICAgICAgICIgICAgICAiCi0gICAgICAgICAgICAgICAgICAgICAgICAiaWQ9MHglMDh4LCBjbGllbnQ9MHglMDh4LCBpZGVudGl0eT0ldVxuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgIGxiYy0+Y2xpZW50SW5kZXgoKSwgbGJjLT5jbGllbnQgPyBsYmMtPmNsaWVudC0+Y2lkIDogMCwKLSAgICAgICAgICAgICAgICAgICAgICAgIGxiYy0+Z2V0SWRlbnRpdHkoKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgICAgICAgICBidWZmZXJbMF0gPSAwOwotICAgICAgICAgICAgLyoqKiBMYXllciAqKiovCi0gICAgICAgICAgICBMYXllciogY29uc3QgbCA9IExheWVyQmFzZTo6ZHluYW1pY0Nhc3Q8TGF5ZXIqPigoTGF5ZXJCYXNlKilsYXllcik7Ci0gICAgICAgICAgICBpZiAobCkgewotICAgICAgICAgICAgICAgIGNvbnN0IExheWVyQml0bWFwJiBidWYwKGwtPmdldEJ1ZmZlcigwKSk7Ci0gICAgICAgICAgICAgICAgY29uc3QgTGF5ZXJCaXRtYXAmIGJ1ZjEobC0+Z2V0QnVmZmVyKDEpKTsKLSAgICAgICAgICAgICAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsCi0gICAgICAgICAgICAgICAgICAgICAgICAiICAgICAgIgotICAgICAgICAgICAgICAgICAgICAgICAgImZvcm1hdD0lMmQsIFslM3V4JTN1OiUzdV0gWyUzdXglM3U6JTN1XSwgbVRleHR1cmVOYW1lPSVkLCIKLSAgICAgICAgICAgICAgICAgICAgICAgICIgZnJlZXplTG9jaz0lcCwgc3dhcFN0YXRlPTB4JTA4eFxuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgIGwtPnBpeGVsRm9ybWF0KCksCi0gICAgICAgICAgICAgICAgICAgICAgICBidWYwLndpZHRoKCksIGJ1ZjAuaGVpZ2h0KCksIGJ1ZjAuc3RyaWRlKCksCi0gICAgICAgICAgICAgICAgICAgICAgICBidWYxLndpZHRoKCksIGJ1ZjEuaGVpZ2h0KCksIGJ1ZjEuc3RyaWRlKCksCi0gICAgICAgICAgICAgICAgICAgICAgICBsLT5nZXRUZXh0dXJlTmFtZSgpLCBsLT5nZXRGcmVlemVMb2NrKCkuZ2V0KCksCi0gICAgICAgICAgICAgICAgICAgICAgICBsLT5sY2Jsay0+c3dhcFN0YXRlKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICAgICAgICAgIGJ1ZmZlclswXSA9IDA7Ci0gICAgICAgICAgICBzLnRyYW5zcGFyZW50UmVnaW9uLmR1bXAocmVzdWx0LCAidHJhbnNwYXJlbnRSZWdpb24iKTsKLSAgICAgICAgICAgIGxheWVyLT50cmFuc3BhcmVudFJlZ2lvblNjcmVlbi5kdW1wKHJlc3VsdCwgInRyYW5zcGFyZW50UmVnaW9uU2NyZWVuIik7Ci0gICAgICAgICAgICBsYXllci0+dmlzaWJsZVJlZ2lvblNjcmVlbi5kdW1wKHJlc3VsdCwgInZpc2libGVSZWdpb25TY3JlZW4iKTsKLSAgICAgICAgfQotICAgICAgICBtV29ybWhvbGVSZWdpb24uZHVtcChyZXN1bHQsICJXb3JtaG9sZVJlZ2lvbiIpOwotICAgICAgICBjb25zdCBEaXNwbGF5SGFyZHdhcmUmIGh3KGdyYXBoaWNQbGFuZSgwKS5kaXNwbGF5SGFyZHdhcmUoKSk7Ci0gICAgICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwKLSAgICAgICAgICAgICAgICAiICBkaXNwbGF5IGZyb3plbjogJXMsIGZyZWV6ZUNvdW50PSVkLCBvcmllbnRhdGlvbj0lZCwgY2FuRHJhdz0lZFxuIiwKLSAgICAgICAgICAgICAgICBtRnJlZXplRGlzcGxheT8ieWVzIjoibm8iLCBtRnJlZXplQ291bnQsCi0gICAgICAgICAgICAgICAgbUN1cnJlbnRTdGF0ZS5vcmllbnRhdGlvbiwgaHcuY2FuRHJhdygpKTsKLSAgICAgICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotCi0gICAgICAgIHNwPEFsbG9jYXRvckludGVyZmFjZT4gYWxsb2NhdG9yOwotICAgICAgICBpZiAobUdQVSAhPSAwKSB7Ci0gICAgICAgICAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICIgIEdQVSBvd25lcjogJWRcbiIsIG1HUFUtPmdldE93bmVyKCkpOwotICAgICAgICAgICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgICAgICAgICAgYWxsb2NhdG9yID0gbUdQVS0+Z2V0QWxsb2NhdG9yKCk7Ci0gICAgICAgICAgICBpZiAoYWxsb2NhdG9yICE9IDApIHsKLSAgICAgICAgICAgICAgICBhbGxvY2F0b3ItPmR1bXAocmVzdWx0LCAiR1BVIEFsbG9jYXRvciIpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGFsbG9jYXRvciA9IG1TdXJmYWNlSGVhcE1hbmFnZXItPmdldEFsbG9jYXRvcihOQVRJVkVfTUVNT1JZX1RZUEVfUE1FTSk7Ci0gICAgICAgIGlmIChhbGxvY2F0b3IgIT0gMCkgewotICAgICAgICAgICAgYWxsb2NhdG9yLT5kdW1wKHJlc3VsdCwgIlBNRU0gQWxsb2NhdG9yIik7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgd3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlRmxpbmdlcjo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIHN3aXRjaCAoY29kZSkgewotICAgICAgICBjYXNlIENSRUFURV9DT05ORUNUSU9OOgotICAgICAgICBjYXNlIE9QRU5fR0xPQkFMX1RSQU5TQUNUSU9OOgotICAgICAgICBjYXNlIENMT1NFX0dMT0JBTF9UUkFOU0FDVElPTjoKLSAgICAgICAgY2FzZSBTRVRfT1JJRU5UQVRJT046Ci0gICAgICAgIGNhc2UgRlJFRVpFX0RJU1BMQVk6Ci0gICAgICAgIGNhc2UgVU5GUkVFWkVfRElTUExBWToKLSAgICAgICAgY2FzZSBCT09UX0ZJTklTSEVEOgotICAgICAgICBjYXNlIFJFVk9LRV9HUFU6Ci0gICAgICAgIHsKLSAgICAgICAgICAgIC8vIGNvZGVzIHRoYXQgcmVxdWlyZSBwZXJtaXNzaW9uIGNoZWNrCi0gICAgICAgICAgICBJUENUaHJlYWRTdGF0ZSogaXBjID0gSVBDVGhyZWFkU3RhdGU6OnNlbGYoKTsKLSAgICAgICAgICAgIGNvbnN0IGludCBwaWQgPSBpcGMtPmdldENhbGxpbmdQaWQoKTsKLSAgICAgICAgICAgIGNvbnN0IGludCBzZWxmX3BpZCA9IGdldHBpZCgpOwotICAgICAgICAgICAgaWYgKFVOTElLRUxZKHBpZCAhPSBzZWxmX3BpZCkpIHsKLSAgICAgICAgICAgICAgICAvLyB3ZSdyZSBjYWxsZWQgZnJvbSBhIGRpZmZlcmVudCBwcm9jZXNzLCBkbyB0aGUgcmVhbCBjaGVjawotICAgICAgICAgICAgICAgIGlmICghY2hlY2tDYWxsaW5nUGVybWlzc2lvbigKLSAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzE2KCJhbmRyb2lkLnBlcm1pc3Npb24uQUNDRVNTX1NVUkZBQ0VfRkxJTkdFUiIpKSkKLSAgICAgICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgICAgIGNvbnN0IGludCB1aWQgPSBpcGMtPmdldENhbGxpbmdVaWQoKTsKLSAgICAgICAgICAgICAgICAgICAgTE9HRSgiUGVybWlzc2lvbiBEZW5pYWw6ICIKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAiY2FuJ3QgYWNjZXNzIFN1cmZhY2VGbGluZ2VyIHBpZD0lZCwgdWlkPSVkIiwgcGlkLCB1aWQpOwotICAgICAgICAgICAgICAgICAgICByZXR1cm4gUEVSTUlTU0lPTl9ERU5JRUQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgc3RhdHVzX3QgZXJyID0gQm5TdXJmYWNlQ29tcG9zZXI6Om9uVHJhbnNhY3QoY29kZSwgZGF0YSwgcmVwbHksIGZsYWdzKTsKLSAgICBpZiAoZXJyID09IFVOS05PV05fVFJBTlNBQ1RJT04gfHwgZXJyID09IFBFUk1JU1NJT05fREVOSUVEKSB7Ci0gICAgICAgIC8vIEhBUkRXQVJFX1RFU1Qgc3R1ZmYuLi4KLSAgICAgICAgaWYgKFVOTElLRUxZKGNoZWNrQ2FsbGluZ1Blcm1pc3Npb24oCi0gICAgICAgICAgICAgICAgU3RyaW5nMTYoImFuZHJvaWQucGVybWlzc2lvbi5IQVJEV0FSRV9URVNUIikpID09IGZhbHNlKSkKLSAgICAgICAgeyAvLyBub3QgYWxsb3dlZAotICAgICAgICAgICAgTE9HRSgiUGVybWlzc2lvbiBEZW5pYWw6IHBpZD0lZCwgdWlkPSVkXG4iLAotICAgICAgICAgICAgICAgICAgICBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5nZXRDYWxsaW5nUGlkKCksCi0gICAgICAgICAgICAgICAgICAgIElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPmdldENhbGxpbmdVaWQoKSk7Ci0gICAgICAgICAgICByZXR1cm4gUEVSTUlTU0lPTl9ERU5JRUQ7Ci0gICAgICAgIH0KLSAgICAgICAgaW50IG47Ci0gICAgICAgIHN3aXRjaCAoY29kZSkgewotICAgICAgICAgICAgY2FzZSAxMDAwOiAvLyBTSE9XX0NQVQotICAgICAgICAgICAgICAgIG4gPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgICAgIG1EZWJ1Z0NwdSA9IG4gPyAxIDogMDsKLSAgICAgICAgICAgICAgICBpZiAobURlYnVnQ3B1KSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChtQ3B1R2F1Z2UgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgbUNwdUdhdWdlID0gbmV3IENQVUdhdWdlKHRoaXMsIG1zMm5zKDUwMCkpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKG1DcHVHYXVnZSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBtQ3B1R2F1Z2UtPnJlcXVlc3RFeGl0QW5kV2FpdCgpOwotICAgICAgICAgICAgICAgICAgICAgICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1EZWJ1Z0xvY2spOwotICAgICAgICAgICAgICAgICAgICAgICAgbUNwdUdhdWdlLmNsZWFyKCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgICAgY2FzZSAxMDAxOiAgLy8gU0hPV19GUFMKLSAgICAgICAgICAgICAgICBuID0gZGF0YS5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgICAgICBtRGVidWdGcHMgPSBuID8gMSA6IDA7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgICAgY2FzZSAxMDAyOiAgLy8gU0hPV19VUERBVEVTCi0gICAgICAgICAgICAgICAgbiA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICAgICAgbURlYnVnUmVnaW9uID0gbiA/IG4gOiAobURlYnVnUmVnaW9uID8gMCA6IDEpOwotICAgICAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgICAgIGNhc2UgMTAwMzogIC8vIFNIT1dfQkFDS0dST1VORAotICAgICAgICAgICAgICAgIG4gPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgICAgIG1EZWJ1Z0JhY2tncm91bmQgPSBuID8gMSA6IDA7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgICAgY2FzZSAxMDA0OnsgLy8gcmVwYWludCBldmVyeXRoaW5nCi0gICAgICAgICAgICAgICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1TdGF0ZUxvY2spOwotICAgICAgICAgICAgICAgIGNvbnN0IERpc3BsYXlIYXJkd2FyZSYgaHcoZ3JhcGhpY1BsYW5lKDApLmRpc3BsYXlIYXJkd2FyZSgpKTsKLSAgICAgICAgICAgICAgICBtRGlydHlSZWdpb24uc2V0KGh3LmJvdW5kcygpKTsgLy8gY2FyZWZ1bCB0aGF0J3Mgbm90IHRocmVhZC1zYWZlCi0gICAgICAgICAgICAgICAgc2lnbmFsRXZlbnQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgICAgIGNhc2UgMTAwNTogLy8gYXNrIEdQVSByZXZva2UKLSAgICAgICAgICAgICAgICBtR1BVLT5mcmllbmRseVJldm9rZSgpOwotICAgICAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgICAgIGNhc2UgMTAwNjogLy8gcmV2b2tlIEdQVQotICAgICAgICAgICAgICAgIG1HUFUtPnVuY29uZGl0aW9uYWxSZXZva2UoKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgICAgICBjYXNlIDEwMDc6IC8vIHNldCBtRnJlZXplQ291bnQKLSAgICAgICAgICAgICAgICBtRnJlZXplQ291bnQgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgICAgIGNhc2UgMTAxMDogIC8vIGludGVycm9nYXRlLgotICAgICAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKG1EZWJ1Z0NwdSk7Ci0gICAgICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIoMCk7Ci0gICAgICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIobURlYnVnUmVnaW9uKTsKLSAgICAgICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihtRGVidWdCYWNrZ3JvdW5kKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgICAgICBjYXNlIDEwMTM6IHsKLSAgICAgICAgICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2wobVN0YXRlTG9jayk7Ci0gICAgICAgICAgICAgICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiBodyhncmFwaGljUGxhbmUoMCkuZGlzcGxheUhhcmR3YXJlKCkpOwotICAgICAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKGh3LmdldFBhZ2VGbGlwQ291bnQoKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNlbmRpZgotCi1DbGllbnQ6OkNsaWVudChDbGllbnRJRCBjbGllbnRJRCwgY29uc3Qgc3A8U3VyZmFjZUZsaW5nZXI+JiBmbGluZ2VyKQotICAgIDogY3RybGJsaygwKSwgY2lkKGNsaWVudElEKSwgbVBpZCgwKSwgbUJpdG1hcCgwKSwgbUZsaW5nZXIoZmxpbmdlcikKLXsKLSAgICBtU2hhcmVkSGVhcEFsbG9jYXRvciA9IGdldFN1cmZhY2VIZWFwTWFuYWdlcigpLT5jcmVhdGVIZWFwKCk7Ci0gICAgY29uc3QgaW50IHBnc2l6ZSA9IGdldHBhZ2VzaXplKCk7Ci0gICAgY29uc3QgaW50IGNibGtzaXplPSgoc2l6ZW9mKHBlcl9jbGllbnRfY2Jsa190KSsocGdzaXplLTEpKSZ+KHBnc2l6ZS0xKSk7Ci0gICAgbUNibGtIZWFwID0gbmV3IE1lbW9yeURlYWxlcihjYmxrc2l6ZSk7Ci0gICAgbUNibGtNZW1vcnkgPSBtQ2Jsa0hlYXAtPmFsbG9jYXRlKGNibGtzaXplKTsKLSAgICBpZiAobUNibGtNZW1vcnkgIT0gMCkgewotICAgICAgICBjdHJsYmxrID0gc3RhdGljX2Nhc3Q8cGVyX2NsaWVudF9jYmxrX3QgKj4obUNibGtNZW1vcnktPnBvaW50ZXIoKSk7Ci0gICAgICAgIGlmIChjdHJsYmxrKSB7IC8vIGNvbnN0cnVjdCB0aGUgc2hhcmVkIHN0cnVjdHVyZSBpbi1wbGFjZS4KLSAgICAgICAgICAgIG5ldyhjdHJsYmxrKSBwZXJfY2xpZW50X2NibGtfdDsKLSAgICAgICAgfQotICAgIH0KLX0KLQotQ2xpZW50Ojp+Q2xpZW50KCkgewotICAgIGlmIChjdHJsYmxrKSB7Ci0gICAgICAgIGNvbnN0IGludCBwZ3NpemUgPSBnZXRwYWdlc2l6ZSgpOwotICAgICAgICBjdHJsYmxrLT5+cGVyX2NsaWVudF9jYmxrX3QoKTsgIC8vIGRlc3Ryb3kgb3VyIHNoYXJlZC1zdHJ1Y3R1cmUuCi0gICAgfQotfQotCi1jb25zdCBzcDxTdXJmYWNlSGVhcE1hbmFnZXI+JiBDbGllbnQ6OmdldFN1cmZhY2VIZWFwTWFuYWdlcigpIGNvbnN0IHsKLSAgICByZXR1cm4gbUZsaW5nZXItPmdldFN1cmZhY2VIZWFwTWFuYWdlcigpOwotfQotCi1pbnQzMl90IENsaWVudDo6Z2VuZXJhdGVJZChpbnQgcGlkKQotewotICAgIGNvbnN0IHVpbnQzMl90IGkgPSBjbHooIH5tQml0bWFwICk7Ci0gICAgaWYgKGkgPj0gTlVNX0xBWUVSU19NQVgpIHsKLSAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLSAgICB9Ci0gICAgbVBpZCA9IHBpZDsKLSAgICBtSW5Vc2UuYWRkKHVpbnQ4X3QoaSkpOwotICAgIG1CaXRtYXAgfD0gMTw8KDMxLWkpOwotICAgIHJldHVybiBpOwotfQotc3RhdHVzX3QgQ2xpZW50OjpiaW5kTGF5ZXIoTGF5ZXJCYXNlQ2xpZW50KiBsYXllciwgaW50MzJfdCBpZCkKLXsKLSAgICBzc2l6ZV90IGlkeCA9IG1JblVzZS5pbmRleE9mKGlkKTsKLSAgICBpZiAoaWR4IDwgMCkKLSAgICAgICAgcmV0dXJuIE5BTUVfTk9UX0ZPVU5EOwotICAgIHJldHVybiBtTGF5ZXJzLmluc2VydEF0KGxheWVyLCBpZHgpOwotfQotdm9pZCBDbGllbnQ6OmZyZWUoaW50MzJfdCBpZCkKLXsKLSAgICBzc2l6ZV90IGlkeCA9IG1JblVzZS5yZW1vdmUodWludDhfdChpZCkpOwotICAgIGlmIChpZHggPj0gMCkgewotICAgICAgICBtQml0bWFwICY9IH4oMTw8KDMxLWlkKSk7Ci0gICAgICAgIG1MYXllcnMucmVtb3ZlSXRlbXNBdChpZHgpOwotICAgIH0KLX0KLQotc3A8TWVtb3J5RGVhbGVyPiBDbGllbnQ6OmNyZWF0ZUFsbG9jYXRvcih1aW50MzJfdCBmbGFncykKLXsKLSAgICBzcDxNZW1vcnlEZWFsZXI+IGFsbG9jYXRvcjsKLSAgICBhbGxvY2F0b3IgPSBnZXRTdXJmYWNlSGVhcE1hbmFnZXIoKS0+Y3JlYXRlSGVhcCgKLSAgICAgICAgICAgIGZsYWdzLCBnZXRDbGllbnRQaWQoKSwgbVNoYXJlZEhlYXBBbGxvY2F0b3IpOwotICAgIHJldHVybiBhbGxvY2F0b3I7Ci19Ci0KLWJvb2wgQ2xpZW50Ojppc1ZhbGlkKGludDMyX3QgaSkgY29uc3QgewotICAgIHJldHVybiAodWludDMyX3QoaSk8TlVNX0xBWUVSU19NQVgpICYmIChtQml0bWFwICYgKDE8PCgzMS1pKSkpOwotfQotY29uc3QgdWludDhfdCogQ2xpZW50OjppblVzZUFycmF5KCkgY29uc3QgewotICAgIHJldHVybiBtSW5Vc2UuYXJyYXkoKTsKLX0KLXNpemVfdCBDbGllbnQ6Om51bUFjdGl2ZUxheWVycygpIGNvbnN0IHsKLSAgICByZXR1cm4gbUluVXNlLnNpemUoKTsKLX0KLUxheWVyQmFzZUNsaWVudCogQ2xpZW50OjpnZXRMYXllclVzZXIoaW50MzJfdCBpKSBjb25zdCB7Ci0gICAgc3NpemVfdCBpZHggPSBtSW5Vc2UuaW5kZXhPZih1aW50OF90KGkpKTsKLSAgICBpZiAoaWR4PDApIHJldHVybiAwOwotICAgIHJldHVybiBtTGF5ZXJzW2lkeF07Ci19Ci0KLXZvaWQgQ2xpZW50OjpkdW1wKGNvbnN0IGNoYXIqIHdoYXQpCi17Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jZW5kaWYKLQotQkNsaWVudDo6QkNsaWVudChTdXJmYWNlRmxpbmdlciAqZmxpbmdlciwgQ2xpZW50SUQgY2lkLCBjb25zdCBzcDxJTWVtb3J5PiYgY2JsaykKLSAgICA6IG1JZChjaWQpLCBtRmxpbmdlcihmbGluZ2VyKSwgbUNibGsoY2JsaykKLXsKLX0KLQotQkNsaWVudDo6fkJDbGllbnQoKSB7Ci0gICAgLy8gZGVzdHJveSBhbGwgcmVzb3VyY2VzIGF0dGFjaGVkIHRvIHRoaXMgY2xpZW50Ci0gICAgbUZsaW5nZXItPmRlc3Ryb3lDb25uZWN0aW9uKG1JZCk7Ci19Ci0KLXZvaWQgQkNsaWVudDo6Z2V0Q29udHJvbEJsb2NrcyhzcDxJTWVtb3J5PiogY3RybCkgY29uc3QgewotICAgICpjdHJsID0gbUNibGs7Ci19Ci0KLXNwPElTdXJmYWNlPiBCQ2xpZW50OjpjcmVhdGVTdXJmYWNlKAotICAgICAgICBJU3VyZmFjZUZsaW5nZXJDbGllbnQ6OnN1cmZhY2VfZGF0YV90KiBwYXJhbXMsIGludCBwaWQsCi0gICAgICAgIERpc3BsYXlJRCBkaXNwbGF5LCB1aW50MzJfdCB3LCB1aW50MzJfdCBoLCBQaXhlbEZvcm1hdCBmb3JtYXQsCi0gICAgICAgIHVpbnQzMl90IGZsYWdzKQotewotICAgIHJldHVybiBtRmxpbmdlci0+Y3JlYXRlU3VyZmFjZShtSWQsIHBpZCwgcGFyYW1zLCBkaXNwbGF5LCB3LCBoLCBmb3JtYXQsIGZsYWdzKTsKLX0KLQotc3RhdHVzX3QgQkNsaWVudDo6ZGVzdHJveVN1cmZhY2UoU3VyZmFjZUlEIHNpZCkKLXsKLSAgICBzaWQgfD0gKG1JZCA8PCAxNik7IC8vIGFkZCB0aGUgY2xpZW50LXBhcnQgdG8gaWQKLSAgICByZXR1cm4gbUZsaW5nZXItPmRlc3Ryb3lTdXJmYWNlKHNpZCk7Ci19Ci0KLXN0YXR1c190IEJDbGllbnQ6OnNldFN0YXRlKGludDMyX3QgY291bnQsIGNvbnN0IGxheWVyX3N0YXRlX3QqIHN0YXRlcykKLXsKLSAgICByZXR1cm4gbUZsaW5nZXItPnNldENsaWVudFN0YXRlKG1JZCwgY291bnQsIHN0YXRlcyk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1HcmFwaGljUGxhbmU6OkdyYXBoaWNQbGFuZSgpCi0gICAgOiBtSHcoMCkKLXsKLX0KLQotR3JhcGhpY1BsYW5lOjp+R3JhcGhpY1BsYW5lKCkgewotICAgIGRlbGV0ZSBtSHc7Ci19Ci0KLWJvb2wgR3JhcGhpY1BsYW5lOjppbml0aWFsaXplZCgpIGNvbnN0IHsKLSAgICByZXR1cm4gbUh3ID8gdHJ1ZSA6IGZhbHNlOwotfQotCi12b2lkIEdyYXBoaWNQbGFuZTo6c2V0RGlzcGxheUhhcmR3YXJlKERpc3BsYXlIYXJkd2FyZSAqaHcpIHsKLSAgICBtSHcgPSBodzsKLX0KLQotdm9pZCBHcmFwaGljUGxhbmU6OnNldFRyYW5zZm9ybShjb25zdCBUcmFuc2Zvcm0mIHRyKSB7Ci0gICAgbVRyYW5zZm9ybSA9IHRyOwotICAgIG1HbG9iYWxUcmFuc2Zvcm0gPSBtT3JpZW50YXRpb25UcmFuc2Zvcm0gKiBtVHJhbnNmb3JtOwotfQotCi1zdGF0dXNfdCBHcmFwaGljUGxhbmU6Om9yaWVudGF0aW9uVG9UcmFuc2Zyb20oCi0gICAgICAgIGludCBvcmllbnRhdGlvbiwgaW50IHcsIGludCBoLCBUcmFuc2Zvcm0qIHRyKQoteyAgICAKLSAgICBmbG9hdCBhLCBiLCBjLCBkLCB4LCB5OwotICAgIHN3aXRjaCAob3JpZW50YXRpb24pIHsKLSAgICBjYXNlIElTdXJmYWNlQ29tcG9zZXI6OmVPcmllbnRhdGlvbkRlZmF1bHQ6Ci0gICAgICAgIGE9MTsgYj0wOyBjPTA7IGQ9MTsgeD0wOyB5PTA7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgSVN1cmZhY2VDb21wb3Nlcjo6ZU9yaWVudGF0aW9uOTA6Ci0gICAgICAgIGE9MDsgYj0tMTsgYz0xOyBkPTA7IHg9dzsgeT0wOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIElTdXJmYWNlQ29tcG9zZXI6OmVPcmllbnRhdGlvbjE4MDoKLSAgICAgICAgYT0tMTsgYj0wOyBjPTA7IGQ9LTE7IHg9dzsgeT1oOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIElTdXJmYWNlQ29tcG9zZXI6OmVPcmllbnRhdGlvbjI3MDoKLSAgICAgICAgYT0wOyBiPTE7IGM9LTE7IGQ9MDsgeD0wOyB5PWg7Ci0gICAgICAgIGJyZWFrOwotICAgIGRlZmF1bHQ6Ci0gICAgICAgIHJldHVybiBCQURfVkFMVUU7Ci0gICAgfQotICAgIHRyLT5zZXQoYSwgYiwgYywgZCk7Ci0gICAgdHItPnNldCh4LCB5KTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IEdyYXBoaWNQbGFuZTo6c2V0T3JpZW50YXRpb24oaW50IG9yaWVudGF0aW9uKQotewotICAgIGNvbnN0IERpc3BsYXlIYXJkd2FyZSYgaHcoZGlzcGxheUhhcmR3YXJlKCkpOwotICAgIGNvbnN0IGZsb2F0IHcgPSBody5nZXRXaWR0aCgpOwotICAgIGNvbnN0IGZsb2F0IGggPSBody5nZXRIZWlnaHQoKTsKLQotICAgIGlmIChvcmllbnRhdGlvbiA9PSBJU3VyZmFjZUNvbXBvc2VyOjplT3JpZW50YXRpb25EZWZhdWx0KSB7Ci0gICAgICAgIC8vIG1ha2Ugc3VyZSB0aGUgZGVmYXVsdCBvcmllbnRhdGlvbiBpcyBvcHRpbWFsCi0gICAgICAgIG1PcmllbnRhdGlvblRyYW5zZm9ybS5yZXNldCgpOwotICAgICAgICBtR2xvYmFsVHJhbnNmb3JtID0gbVRyYW5zZm9ybTsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLQotICAgIC8vIElmIHRoZSByb3RhdGlvbiBjYW4gYmUgaGFuZGxlZCBpbiBoYXJkd2FyZSwgdGhpcyBpcyB3aGVyZQotICAgIC8vIHRoZSBtYWdpYyBzaG91bGQgaGFwcGVuLgotICAgIGlmIChVTkxJS0VMWShvcmllbnRhdGlvbiA9PSA0MikpIHsKLSAgICAgICAgZmxvYXQgYSwgYiwgYywgZCwgeCwgeTsKLSAgICAgICAgY29uc3QgZmxvYXQgciA9ICgzLjE0MTU5MjY1ZiAvIDE4MC4wZikgKiA0Mi4wZjsKLSAgICAgICAgY29uc3QgZmxvYXQgc2kgPSBzaW5mKHIpOwotICAgICAgICBjb25zdCBmbG9hdCBjbyA9IGNvc2Yocik7Ci0gICAgICAgIGE9Y287IGI9LXNpOyBjPXNpOyBkPWNvOwotICAgICAgICB4ID0gc2kqKGgqMC41ZikgKyAoMS1jbykqKHcqMC41Zik7Ci0gICAgICAgIHkgPS1zaSoodyowLjVmKSArICgxLWNvKSooaCowLjVmKTsKLSAgICAgICAgbU9yaWVudGF0aW9uVHJhbnNmb3JtLnNldChhLCBiLCBjLCBkKTsKLSAgICAgICAgbU9yaWVudGF0aW9uVHJhbnNmb3JtLnNldCh4LCB5KTsKLSAgICB9IGVsc2UgewotICAgICAgICBHcmFwaGljUGxhbmU6Om9yaWVudGF0aW9uVG9UcmFuc2Zyb20ob3JpZW50YXRpb24sIHcsIGgsCi0gICAgICAgICAgICAgICAgJm1PcmllbnRhdGlvblRyYW5zZm9ybSk7Ci0gICAgfQotICAgIAotICAgIG1HbG9iYWxUcmFuc2Zvcm0gPSBtT3JpZW50YXRpb25UcmFuc2Zvcm0gKiBtVHJhbnNmb3JtOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotY29uc3QgRGlzcGxheUhhcmR3YXJlJiBHcmFwaGljUGxhbmU6OmRpc3BsYXlIYXJkd2FyZSgpIGNvbnN0IHsKLSAgICByZXR1cm4gKm1IdzsKLX0KLQotY29uc3QgVHJhbnNmb3JtJiBHcmFwaGljUGxhbmU6OnRyYW5zZm9ybSgpIGNvbnN0IHsKLSAgICByZXR1cm4gbUdsb2JhbFRyYW5zZm9ybTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL1N1cmZhY2VGbGluZ2VyLmggYi9saWJzL3N1cmZhY2VmbGluZ2VyL1N1cmZhY2VGbGluZ2VyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGY3ZDc3NjQuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9TdXJmYWNlRmxpbmdlci5oCisrKyAvZGV2L251bGwKQEAgLTEsNDM1ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfU1VSRkFDRV9GTElOR0VSX0gKLSNkZWZpbmUgQU5EUk9JRF9TVVJGQUNFX0ZMSU5HRVJfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9Tb3J0ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9LZXllZFZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy9NZW1vcnlEZWFsZXIuaD4KLQotI2luY2x1ZGUgPHVpL1BpeGVsRm9ybWF0Lmg+Ci0jaW5jbHVkZSA8dWkvSVN1cmZhY2VDb21wb3Nlci5oPgotI2luY2x1ZGUgPHVpL0lTdXJmYWNlRmxpbmdlckNsaWVudC5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS91aS9TaGFyZWRTdGF0ZS5oPgotI2luY2x1ZGUgPHByaXZhdGUvdWkvTGF5ZXJTdGF0ZS5oPgotI2luY2x1ZGUgPHByaXZhdGUvdWkvU3VyZmFjZUZsaW5nZXJTeW5jaHJvLmg+Ci0KLSNpbmNsdWRlICJCYXJyaWVyLmgiCi0jaW5jbHVkZSAiQm9vdEFuaW1hdGlvbi5oIgotI2luY2x1ZGUgIkNQVUdhdWdlLmgiCi0jaW5jbHVkZSAiTGF5ZXIuaCIKLSNpbmNsdWRlICJUb2tlbml6ZXIuaCIKLQotc3RydWN0IGNvcHliaXRfZGV2aWNlX3Q7Ci1zdHJ1Y3Qgb3ZlcmxheV9kZXZpY2VfdDsKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgQ2xpZW50OwotY2xhc3MgQkNsaWVudDsKLWNsYXNzIERpc3BsYXlIYXJkd2FyZTsKLWNsYXNzIEZyZWV6ZUxvY2s7Ci1jbGFzcyBHUFVIYXJkd2FyZUludGVyZmFjZTsKLWNsYXNzIElHUFVDYWxsYmFjazsKLWNsYXNzIExheWVyOwotY2xhc3MgTGF5ZXJCdWZmZXI7Ci1jbGFzcyBMYXllck9yaWVudGF0aW9uQW5pbTsKLWNsYXNzIE9yaWVudGF0aW9uQW5pbWF0aW9uOwotY2xhc3MgU3VyZmFjZUhlYXBNYW5hZ2VyOwotCi10eXBlZGVmIGludDMyX3QgQ2xpZW50SUQ7Ci0KLSNkZWZpbmUgTElLRUxZKCBleHAgKSAgICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgdHJ1ZSAgKSkKLSNkZWZpbmUgVU5MSUtFTFkoIGV4cCApICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgZmFsc2UgKSkKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIENsaWVudAotewotcHVibGljOgotICAgICAgICAgICAgQ2xpZW50KENsaWVudElEIGNpZCwgY29uc3Qgc3A8U3VyZmFjZUZsaW5nZXI+JiBmbGluZ2VyKTsKLSAgICAgICAgICAgIH5DbGllbnQoKTsKLQotICAgICAgICAgICAgaW50MzJfdCAgICAgICAgICAgICAgICAgZ2VuZXJhdGVJZChpbnQgcGlkKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgICAgIGZyZWUoaW50MzJfdCBpZCk7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICAgICAgICAgICAgICBiaW5kTGF5ZXIoTGF5ZXJCYXNlQ2xpZW50KiBsYXllciwgaW50MzJfdCBpZCk7Ci0gICAgICAgICAgICBzcDxNZW1vcnlEZWFsZXI+ICAgICAgICBjcmVhdGVBbGxvY2F0b3IodWludDMyX3QgbWVtb3J5X3R5cGUpOwotCi0gICAgaW5saW5lICBib29sICAgICAgICAgICAgICAgICAgICBpc1ZhbGlkKGludDMyX3QgaSkgY29uc3Q7Ci0gICAgaW5saW5lICBjb25zdCB1aW50OF90KiAgICAgICAgICBpblVzZUFycmF5KCkgY29uc3Q7Ci0gICAgaW5saW5lICBzaXplX3QgICAgICAgICAgICAgICAgICBudW1BY3RpdmVMYXllcnMoKSBjb25zdDsKLSAgICBMYXllckJhc2VDbGllbnQqICAgICAgICAgICAgICAgIGdldExheWVyVXNlcihpbnQzMl90IGkpIGNvbnN0OwotICAgIGNvbnN0IFZlY3RvcjxMYXllckJhc2VDbGllbnQqPiYgZ2V0TGF5ZXJzKCkgY29uc3QgeyByZXR1cm4gbUxheWVyczsgfQotICAgIGNvbnN0IHNwPElNZW1vcnk+JiAgICAgICAgICAgICAgY29udHJvbEJsb2NrTWVtb3J5KCkgY29uc3QgeyByZXR1cm4gbUNibGtNZW1vcnk7IH0KLSAgICB2b2lkICAgICAgICAgICAgICAgICAgICAgICAgICAgIGR1bXAoY29uc3QgY2hhciogd2hhdCk7Ci0gICAgY29uc3Qgc3A8U3VyZmFjZUhlYXBNYW5hZ2VyPiYgICBnZXRTdXJmYWNlSGVhcE1hbmFnZXIoKSBjb25zdDsKLSAgICAKLSAgICAvLyBwb2ludGVyIHRvIHRoaXMgY2xpZW50J3MgY29udHJvbCBibG9jawotICAgIHBlcl9jbGllbnRfY2Jsa190KiAgICAgIGN0cmxibGs7Ci0gICAgQ2xpZW50SUQgICAgICAgICAgICAgICAgY2lkOwotCi0gICAgCi1wcml2YXRlOgotICAgIGludCAgICAgICAgICAgICAgICAgICAgIGdldENsaWVudFBpZCgpIGNvbnN0IHsgcmV0dXJuIG1QaWQ7IH0KLSAgICAgICAgCi0gICAgaW50ICAgICAgICAgICAgICAgICAgICAgICAgIG1QaWQ7Ci0gICAgdWludDMyX3QgICAgICAgICAgICAgICAgICAgIG1CaXRtYXA7Ci0gICAgU29ydGVkVmVjdG9yPHVpbnQ4X3Q+ICAgICAgIG1JblVzZTsKLSAgICBWZWN0b3I8TGF5ZXJCYXNlQ2xpZW50Kj4gICAgbUxheWVyczsKLSAgICBzcDxNZW1vcnlEZWFsZXI+ICAgICAgICAgICAgbUNibGtIZWFwOwotICAgIHNwPFN1cmZhY2VGbGluZ2VyPiAgICAgICAgICBtRmxpbmdlcjsKLSAgICBzcDxNZW1vcnlEZWFsZXI+ICAgICAgICAgICAgbVNoYXJlZEhlYXBBbGxvY2F0b3I7Ci0gICAgc3A8TWVtb3J5RGVhbGVyPiAgICAgICAgICAgIG1QTWVtQWxsb2NhdG9yOwotICAgIHNwPElNZW1vcnk+ICAgICAgICAgICAgICAgICBtQ2Jsa01lbW9yeTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBHcmFwaGljUGxhbmUKLXsKLXB1YmxpYzoKLSAgICBzdGF0aWMgc3RhdHVzX3Qgb3JpZW50YXRpb25Ub1RyYW5zZnJvbShpbnQgb3JpZW50YXRpb24sIGludCB3LCBpbnQgaCwKLSAgICAgICAgICAgIFRyYW5zZm9ybSogdHIpOwotCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdyYXBoaWNQbGFuZSgpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB+R3JhcGhpY1BsYW5lKCk7Ci0KLSAgICAgICAgYm9vbCAgICAgICAgICAgICAgICAgICAgaW5pdGlhbGl6ZWQoKSBjb25zdDsKLQotICAgICAgICB2b2lkICAgICAgICAgICAgICAgICAgICBzZXREaXNwbGF5SGFyZHdhcmUoRGlzcGxheUhhcmR3YXJlICopOwotICAgICAgICB2b2lkICAgICAgICAgICAgICAgICAgICBzZXRUcmFuc2Zvcm0oY29uc3QgVHJhbnNmb3JtJiB0cik7Ci0gICAgICAgIHN0YXR1c190ICAgICAgICAgICAgICAgIHNldE9yaWVudGF0aW9uKGludCBvcmllbnRhdGlvbik7Ci0KLSAgICAgICAgY29uc3QgRGlzcGxheUhhcmR3YXJlJiAgZGlzcGxheUhhcmR3YXJlKCkgY29uc3Q7Ci0gICAgICAgIGNvbnN0IFRyYW5zZm9ybSYgICAgICAgIHRyYW5zZm9ybSgpIGNvbnN0OwotcHJpdmF0ZToKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR3JhcGhpY1BsYW5lKGNvbnN0IEdyYXBoaWNQbGFuZSYpOwotICAgICAgICBHcmFwaGljUGxhbmUgICAgICAgICAgICBvcGVyYXRvciA9IChjb25zdCBHcmFwaGljUGxhbmUmKTsKLQotICAgICAgICBEaXNwbGF5SGFyZHdhcmUqICAgICAgICBtSHc7Ci0gICAgICAgIFRyYW5zZm9ybSAgICAgICAgICAgICAgIG1UcmFuc2Zvcm07Ci0gICAgICAgIFRyYW5zZm9ybSAgICAgICAgICAgICAgIG1PcmllbnRhdGlvblRyYW5zZm9ybTsKLSAgICAgICAgVHJhbnNmb3JtICAgICAgICAgICAgICAgbUdsb2JhbFRyYW5zZm9ybTsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1lbnVtIHsKLSAgICBlVHJhbnNhY3Rpb25OZWVkZWQgICAgICA9IDB4MDEsCi0gICAgZVRyYXZlcnNhbE5lZWRlZCAgICAgICAgPSAweDAyCi19OwotCi1jbGFzcyBTdXJmYWNlRmxpbmdlciA6IHB1YmxpYyBCblN1cmZhY2VDb21wb3NlciwgcHJvdGVjdGVkIFRocmVhZAotewotcHVibGljOgotICAgIHN0YXRpYyB2b2lkIGluc3RhbnRpYXRlKCk7Ci0gICAgc3RhdGljIHZvaWQgc2h1dGRvd24oKTsKLQotICAgICAgICAgICAgICAgICAgICBTdXJmYWNlRmxpbmdlcigpOwotICAgIHZpcnR1YWwgICAgICAgICB+U3VyZmFjZUZsaW5nZXIoKTsKLSAgICAgICAgICAgIHZvaWQgICAgaW5pdCgpOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCBvblRyYW5zYWN0KAotICAgICAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKTsKLQotICAgIHZpcnR1YWwgc3RhdHVzX3QgZHVtcChpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpOwotCi0gICAgLy8gSVN1cmZhY2VDb21wb3NlciBpbnRlcmZhY2UKLSAgICB2aXJ0dWFsIHNwPElTdXJmYWNlRmxpbmdlckNsaWVudD4gICBjcmVhdGVDb25uZWN0aW9uKCk7Ci0gICAgdmlydHVhbCBzcDxJTWVtb3J5PiAgICAgICAgICAgICAgICAgZ2V0Q2JsaygpIGNvbnN0OwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgICAgICAgICAgICAgIGJvb3RGaW5pc2hlZCgpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgICAgICAgICAgICAgIG9wZW5HbG9iYWxUcmFuc2FjdGlvbigpOwotICAgIHZpcnR1YWwgdm9pZCAgICAgICAgICAgICAgICAgICAgICAgIGNsb3NlR2xvYmFsVHJhbnNhY3Rpb24oKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICAgICAgICAgICAgICBmcmVlemVEaXNwbGF5KERpc3BsYXlJRCBkcHksIHVpbnQzMl90IGZsYWdzKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190ICAgICAgICAgICAgICAgICAgICB1bmZyZWV6ZURpc3BsYXkoRGlzcGxheUlEIGRweSwgdWludDMyX3QgZmxhZ3MpOwotICAgIHZpcnR1YWwgaW50ICAgICAgICAgICAgICAgICAgICAgICAgIHNldE9yaWVudGF0aW9uKERpc3BsYXlJRCBkcHksIGludCBvcmllbnRhdGlvbik7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICAgICAgICAgICAgICAgICAgc2lnbmFsKCkgY29uc3Q7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCByZXF1ZXN0R1BVKGNvbnN0IHNwPElHUFVDYWxsYmFjaz4mIGNhbGxiYWNrLCAKLSAgICAgICAgICAgIGdwdV9pbmZvX3QqIGdwdSk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCByZXZva2VHUFUoKTsKLQotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICAgICAgICAgIHNjcmVlblJlbGVhc2VkKERpc3BsYXlJRCBkcHkpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgICAgICAgICAgICAgICAgIHNjcmVlbkFjcXVpcmVkKERpc3BsYXlJRCBkcHkpOwotCi0gICAgICAgICAgICBjb25zdCBzcDxTdXJmYWNlSGVhcE1hbmFnZXI+JiBnZXRTdXJmYWNlSGVhcE1hbmFnZXIoKSBjb25zdCB7IAotICAgICAgICAgICAgICAgIHJldHVybiBtU3VyZmFjZUhlYXBNYW5hZ2VyOyAKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY29uc3Qgc3A8R1BVSGFyZHdhcmVJbnRlcmZhY2U+JiBnZXRHUFUoKSBjb25zdCB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG1HUFU7IAotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBjb3B5Yml0X2RldmljZV90KiBnZXRCbGl0RW5naW5lKCkgY29uc3Q7Ci0gICAgICAgICAgICBvdmVybGF5X2NvbnRyb2xfZGV2aWNlX3QqIGdldE92ZXJsYXlFbmdpbmUoKSBjb25zdDsKLQotICAgICAgICAgICAgCi0gICAgc3RhdHVzX3QgcmVtb3ZlTGF5ZXIoTGF5ZXJCYXNlKiBsYXllcik7Ci0gICAgc3RhdHVzX3QgYWRkTGF5ZXIoTGF5ZXJCYXNlKiBsYXllcik7Ci0gICAgc3RhdHVzX3QgaW52YWxpZGF0ZUxheWVyVmlzaWJpbGl0eShMYXllckJhc2UqIGxheWVyKTsKLSAgICAKLXByaXZhdGU6Ci0gICAgZnJpZW5kIGNsYXNzIEJDbGllbnQ7Ci0gICAgZnJpZW5kIGNsYXNzIExheWVyQmFzZTsKLSAgICBmcmllbmQgY2xhc3MgTGF5ZXJCdWZmZXI7Ci0gICAgZnJpZW5kIGNsYXNzIExheWVyQmFzZUNsaWVudDsKLSAgICBmcmllbmQgY2xhc3MgTGF5ZXI7Ci0gICAgZnJpZW5kIGNsYXNzIExheWVyQmx1cjsKLQotICAgIHNwPElTdXJmYWNlPiBjcmVhdGVTdXJmYWNlKENsaWVudElEIGNsaWVudCwgaW50IHBpZCwgCi0gICAgICAgICAgICBJU3VyZmFjZUZsaW5nZXJDbGllbnQ6OnN1cmZhY2VfZGF0YV90KiBwYXJhbXMsCi0gICAgICAgICAgICBEaXNwbGF5SUQgZGlzcGxheSwgdWludDMyX3QgdywgdWludDMyX3QgaCwgUGl4ZWxGb3JtYXQgZm9ybWF0LAotICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MpOwotCi0gICAgTGF5ZXJCYXNlQ2xpZW50KiBjcmVhdGVOb3JtYWxTdXJmYWNlTG9ja2VkKENsaWVudCogY2xpZW50LCBEaXNwbGF5SUQgZGlzcGxheSwKLSAgICAgICAgICAgIGludDMyX3QgaWQsIHVpbnQzMl90IHcsIHVpbnQzMl90IGgsIFBpeGVsRm9ybWF0IGZvcm1hdCwgdWludDMyX3QgZmxhZ3MpOwotCi0gICAgTGF5ZXJCYXNlQ2xpZW50KiBjcmVhdGVCbHVyU3VyZmFjZUxvY2tlZChDbGllbnQqIGNsaWVudCwgRGlzcGxheUlEIGRpc3BsYXksCi0gICAgICAgICAgICBpbnQzMl90IGlkLCB1aW50MzJfdCB3LCB1aW50MzJfdCBoLCB1aW50MzJfdCBmbGFncyk7Ci0KLSAgICBMYXllckJhc2VDbGllbnQqIGNyZWF0ZURpbVN1cmZhY2VMb2NrZWQoQ2xpZW50KiBjbGllbnQsIERpc3BsYXlJRCBkaXNwbGF5LAotICAgICAgICAgICAgaW50MzJfdCBpZCwgdWludDMyX3QgdywgdWludDMyX3QgaCwgdWludDMyX3QgZmxhZ3MpOwotCi0gICAgTGF5ZXJCYXNlQ2xpZW50KiBjcmVhdGVQdXNoQnVmZmVyc1N1cmZhY2VMb2NrZWQoQ2xpZW50KiBjbGllbnQsIERpc3BsYXlJRCBkaXNwbGF5LAotICAgICAgICAgICAgaW50MzJfdCBpZCwgdWludDMyX3QgdywgdWludDMyX3QgaCwgdWludDMyX3QgZmxhZ3MpOwotCi0gICAgc3RhdHVzX3QgICAgZGVzdHJveVN1cmZhY2UoU3VyZmFjZUlEIHN1cmZhY2VfaWQpOwotICAgIHN0YXR1c190ICAgIHNldENsaWVudFN0YXRlKENsaWVudElEIGNpZCwgaW50MzJfdCBjb3VudCwgY29uc3QgbGF5ZXJfc3RhdGVfdCogc3RhdGVzKTsKLQotCi0gICAgY2xhc3MgTGF5ZXJWZWN0b3IgewotICAgIHB1YmxpYzoKLSAgICAgICAgaW5saW5lICAgICAgICAgICAgICBMYXllclZlY3RvcigpIHsgfQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIExheWVyVmVjdG9yKGNvbnN0IExheWVyVmVjdG9yJik7Ci0gICAgICAgIGlubGluZSBzaXplX3QgICAgICAgc2l6ZSgpIGNvbnN0IHsgcmV0dXJuIGxheWVycy5zaXplKCk7IH0KLSAgICAgICAgaW5saW5lIExheWVyQmFzZSpjb25zdCogYXJyYXkoKSBjb25zdCB7IHJldHVybiBsYXllcnMuYXJyYXkoKTsgfQotICAgICAgICBzc2l6ZV90ICAgICAgICAgICAgIGFkZChMYXllckJhc2UqLCBWZWN0b3I8TGF5ZXJCYXNlKj46OmNvbXBhcl90KTsKLSAgICAgICAgc3NpemVfdCAgICAgICAgICAgICByZW1vdmUoTGF5ZXJCYXNlKik7Ci0gICAgICAgIHNzaXplX3QgICAgICAgICAgICAgcmVvcmRlcihMYXllckJhc2UqLCBWZWN0b3I8TGF5ZXJCYXNlKj46OmNvbXBhcl90KTsKLSAgICAgICAgc3NpemVfdCAgICAgICAgICAgICBpbmRleE9mKExheWVyQmFzZSoga2V5LCBzaXplX3QgZ3Vlc3M9MCkgY29uc3Q7Ci0gICAgICAgIGlubGluZSBMYXllckJhc2UqICAgb3BlcmF0b3IgW10gKHNpemVfdCBpKSBjb25zdCB7IHJldHVybiBsYXllcnNbaV07IH0KLSAgICBwcml2YXRlOgotICAgICAgICBLZXllZFZlY3RvcjxMYXllckJhc2UqLCBzaXplX3Q+IGxvb2t1cDsKLSAgICAgICAgVmVjdG9yPExheWVyQmFzZSo+ICAgICAgICAgICAgICBsYXllcnM7Ci0gICAgfTsKLQotICAgIHN0cnVjdCBTdGF0ZSB7Ci0gICAgICAgIFN0YXRlKCkgewotICAgICAgICAgICAgb3JpZW50YXRpb24gPSBJU3VyZmFjZUNvbXBvc2VyOjplT3JpZW50YXRpb25EZWZhdWx0OwotICAgICAgICAgICAgZnJlZXplRGlzcGxheSA9IDA7Ci0gICAgICAgIH0KLSAgICAgICAgTGF5ZXJWZWN0b3IgICAgIGxheWVyc1NvcnRlZEJ5WjsKLSAgICAgICAgdWludDhfdCAgICAgICAgIG9yaWVudGF0aW9uOwotICAgICAgICB1aW50OF90ICAgICAgICAgZnJlZXplRGlzcGxheTsKLSAgICB9OwotCi0gICAgY2xhc3MgRGVsYXllZFRyYW5zYWN0aW9uIDogcHVibGljIFRocmVhZAotICAgIHsKLSAgICAgICAgZnJpZW5kIGNsYXNzIFN1cmZhY2VGbGluZ2VyOwotICAgICAgICBzcDxTdXJmYWNlRmxpbmdlcj4gIG1GbGluZ2VyOwotICAgICAgICBuc2Vjc190ICAgICAgICAgICAgIG1EZWxheTsKLSAgICBwdWJsaWM6Ci0gICAgICAgIERlbGF5ZWRUcmFuc2FjdGlvbihjb25zdCBzcDxTdXJmYWNlRmxpbmdlcj4mIGZsaW5nZXIsIG5zZWNzX3QgZGVsYXkpCi0gICAgICAgICAgICA6IFRocmVhZChmYWxzZSksIG1GbGluZ2VyKGZsaW5nZXIpLCBtRGVsYXkoZGVsYXkpIHsKLSAgICAgICAgfQotICAgICAgICB2aXJ0dWFsIGJvb2wgdGhyZWFkTG9vcCgpIHsKLSAgICAgICAgICAgIHVzbGVlcChtRGVsYXkgLyAxMDAwKTsKLSAgICAgICAgICAgIGlmIChhbmRyb2lkX2F0b21pY19hbmQofjEsCi0gICAgICAgICAgICAgICAgICAgICZtRmxpbmdlci0+bURlcGxheWVkVHJhbnNhY3Rpb25QZW5kaW5nKSA9PSAxKSB7Ci0gICAgICAgICAgICAgICAgbUZsaW5nZXItPnNpZ25hbEV2ZW50KCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICB9OwotCi0gICAgdmlydHVhbCBib29sICAgICAgICB0aHJlYWRMb29wKCk7Ci0gICAgdmlydHVhbCBzdGF0dXNfdCAgICByZWFkeVRvUnVuKCk7Ci0gICAgdmlydHVhbCB2b2lkICAgICAgICBvbkZpcnN0UmVmKCk7Ci0KLSAgICBjb25zdCBHcmFwaGljUGxhbmUmICAgICBncmFwaGljUGxhbmUoaW50IGRweSkgY29uc3Q7Ci0gICAgICAgICAgR3JhcGhpY1BsYW5lJiAgICAgZ3JhcGhpY1BsYW5lKGludCBkcHkpOwotCi0gICAgICAgICAgICB2b2lkICAgICAgICB3YWl0Rm9yRXZlbnQoKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIHNpZ25hbEV2ZW50KCk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICBzaWduYWxEZWxheWVkRXZlbnQobnNlY3NfdCBkZWxheSk7Ci0KLSAgICAgICAgICAgIHZvaWQgICAgICAgIGhhbmRsZUNvbnNvbGVFdmVudHMoKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIGhhbmRsZVRyYW5zYWN0aW9uKHVpbnQzMl90IHRyYW5zYWN0aW9uRmxhZ3MpOwotCi0gICAgICAgICAgICB2b2lkICAgICAgICBjb21wdXRlVmlzaWJsZVJlZ2lvbnMoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgTGF5ZXJWZWN0b3ImIGN1cnJlbnRMYXllcnMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgUmVnaW9uJiBkaXJ0eVJlZ2lvbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBSZWdpb24mIHdvcm1ob2xlUmVnaW9uKTsKLQotICAgICAgICAgICAgdm9pZCAgICAgICAgaGFuZGxlUGFnZUZsaXAoKTsKLSAgICAgICAgICAgIGJvb2wgICAgICAgIGxvY2tQYWdlRmxpcChjb25zdCBMYXllclZlY3RvciYgY3VycmVudExheWVycyk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICB1bmxvY2tQYWdlRmxpcChjb25zdCBMYXllclZlY3RvciYgY3VycmVudExheWVycyk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICBoYW5kbGVSZXBhaW50KCk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICBoYW5kbGVEZWJ1Z0NwdSgpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgc2NoZWR1bGVCcm9hZGNhc3QoQ2xpZW50KiBjbGllbnQpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgZXhlY3V0ZVNjaGVkdWxlZEJyb2FkY2FzdHMoKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIHBvc3RGcmFtZWJ1ZmZlcigpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgY29tcG9zZVN1cmZhY2VzKGNvbnN0IFJlZ2lvbiYgZGlydHkpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgdW5sb2NrQ2xpZW50cygpOwotCi0KLSAgICAgICAgICAgIHZvaWQgICAgICAgIGRlc3Ryb3lDb25uZWN0aW9uKENsaWVudElEIGNpZCk7Ci0gICAgICAgICAgICBMYXllckJhc2VDbGllbnQqIGdldExheWVyVXNlcl9sKFN1cmZhY2VJRCBpbmRleCkgY29uc3Q7Ci0gICAgICAgICAgICBzdGF0dXNfdCAgICBhZGRMYXllcl9sKExheWVyQmFzZSogbGF5ZXIpOwotICAgICAgICAgICAgc3RhdHVzX3QgICAgcmVtb3ZlTGF5ZXJfbChMYXllckJhc2UqIGxheWVyKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIGRlc3Ryb3lfYWxsX3JlbW92ZWRfbGF5ZXJzX2woKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIGZyZWVfcmVzb3VyY2VzX2woKTsKLQotICAgICAgICAgICAgdWludDMyX3QgICAgZ2V0VHJhbnNhY3Rpb25GbGFncyh1aW50MzJfdCBmbGFncyk7Ci0gICAgICAgICAgICB1aW50MzJfdCAgICBzZXRUcmFuc2FjdGlvbkZsYWdzKHVpbnQzMl90IGZsYWdzLCBuc2Vjc190IGRlbGF5ID0gMCk7Ci0gICAgICAgICAgICB2b2lkICAgICAgICBjb21taXRUcmFuc2FjdGlvbigpOwotCi0KLSAgICAgICAgICAgIGZyaWVuZCBjbGFzcyBGcmVlemVMb2NrOwotICAgICAgICAgICAgc3A8RnJlZXplTG9jaz4gZ2V0RnJlZXplTG9jaygpIGNvbnN0OwotICAgICAgICAgICAgaW5saW5lIHZvaWQgaW5jRnJlZXplQ291bnQoKSB7IG1GcmVlemVDb3VudCsrOyB9Ci0gICAgICAgICAgICBpbmxpbmUgdm9pZCBkZWNGcmVlemVDb3VudCgpIHsgaWYgKG1GcmVlemVDb3VudCA+IDApIG1GcmVlemVDb3VudC0tOyB9Ci0gICAgICAgICAgICBpbmxpbmUgYm9vbCBoYXNGcmVlemVSZXF1ZXN0KCkgY29uc3QgeyByZXR1cm4gbUZyZWV6ZURpc3BsYXk7IH0KLSAgICAgICAgICAgIGlubGluZSBib29sIGlzRnJvemVuKCkgY29uc3QgeyAKLSAgICAgICAgICAgICAgICByZXR1cm4gbUZyZWV6ZURpc3BsYXkgfHwgbUZyZWV6ZUNvdW50PjA7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIAotICAgICAgICAgICAgdm9pZCAgICAgICAgZGVidWdGbGFzaFJlZ2lvbnMoKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIGRlYnVnU2hvd0ZQUygpIGNvbnN0OwotICAgICAgICAgICAgdm9pZCAgICAgICAgZHJhd1dvcm1ob2xlKCkgY29uc3Q7Ci0gICAgICAgICAgIAotICAgICAgICAgICAgICAgIC8vIGFjY2VzcyBtdXN0IGJlIHByb3RlY3RlZCBieSBtU3RhdGVMb2NrCi0gICAgbXV0YWJsZSAgICAgTXV0ZXggICAgICAgICAgICAgICAgICAgbVN0YXRlTG9jazsKLSAgICAgICAgICAgICAgICBTdGF0ZSAgICAgICAgICAgICAgICAgICBtQ3VycmVudFN0YXRlOwotICAgICAgICAgICAgICAgIFN0YXRlICAgICAgICAgICAgICAgICAgIG1EcmF3aW5nU3RhdGU7Ci0gICAgdm9sYXRpbGUgICAgaW50MzJfdCAgICAgICAgICAgICAgICAgbVRyYW5zYWN0aW9uRmxhZ3M7Ci0gICAgdm9sYXRpbGUgICAgaW50MzJfdCAgICAgICAgICAgICAgICAgbVRyYW5zYWN0aW9uQ291bnQ7Ci0gICAgICAgICAgICAgICAgQ29uZGl0aW9uICAgICAgICAgICAgICAgbVRyYW5zYWN0aW9uQ1Y7Ci0KLSAgICAgICAgICAgICAgICAvLyBwcm90ZWN0ZWQgYnkgbVN0YXRlTG9jayAoYnV0IHdlIGNvdWxkIHVzZSBhbm90aGVyIGxvY2spCi0gICAgICAgICAgICAgICAgVG9rZW5pemVyICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1Ub2tlbnM7Ci0gICAgICAgICAgICAgICAgRGVmYXVsdEtleWVkVmVjdG9yPENsaWVudElELCBDbGllbnQqPiAgIG1DbGllbnRzTWFwOwotICAgICAgICAgICAgICAgIERlZmF1bHRLZXllZFZlY3RvcjxTdXJmYWNlSUQsIExheWVyQmFzZUNsaWVudCo+ICAgbUxheWVyTWFwOwotICAgICAgICAgICAgICAgIEdyYXBoaWNQbGFuZSAgICAgICAgICAgICAgICAgICAgICAgICAgICBtR3JhcGhpY1BsYW5lc1sxXTsKLSAgICAgICAgICAgICAgICBTb3J0ZWRWZWN0b3I8TGF5ZXJCYXNlKj4gICAgICAgICAgICAgICAgbVJlbW92ZWRMYXllcnM7Ci0gICAgICAgICAgICAgICAgVmVjdG9yPENsaWVudCo+ICAgICAgICAgICAgICAgICAgICAgICAgIG1EaXNjb25uZWN0ZWRDbGllbnRzOwotCi0gICAgICAgICAgICAgICAgLy8gY29uc3RhbnQgbWVtYmVycyAobm8gc3luY2hyb25pemF0aW9uIG5lZWRlZCBmb3IgYWNjZXNzKQotICAgICAgICAgICAgICAgIHNwPE1lbW9yeURlYWxlcj4gICAgICAgICAgICBtU2VydmVySGVhcDsKLSAgICAgICAgICAgICAgICBzcDxJTWVtb3J5PiAgICAgICAgICAgICAgICAgbVNlcnZlckNibGtNZW1vcnk7Ci0gICAgICAgICAgICAgICAgc3VyZmFjZV9mbGluZ2VyX2NibGtfdCogICAgIG1TZXJ2ZXJDYmxrOwotICAgICAgICAgICAgICAgIHNwPFN1cmZhY2VIZWFwTWFuYWdlcj4gICAgICBtU3VyZmFjZUhlYXBNYW5hZ2VyOwotICAgICAgICAgICAgICAgIHNwPEdQVUhhcmR3YXJlSW50ZXJmYWNlPiAgICBtR1BVOwotICAgICAgICAgICAgICAgIEdMdWludCAgICAgICAgICAgICAgICAgICAgICBtV29ybWhvbGVUZXhOYW1lOwotICAgICAgICAgICAgICAgIHNwPEJvb3RBbmltYXRpb24+ICAgICAgICAgICBtQm9vdEFuaW1hdGlvbjsKLSAgICAgICAgICAgICAgICBuc2Vjc190ICAgICAgICAgICAgICAgICAgICAgbUJvb3RUaW1lOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIC8vIENhbiBvbmx5IGFjY2Vzc2VkIGZyb20gdGhlIG1haW4gdGhyZWFkLCB0aGVzZSBtZW1iZXJzCi0gICAgICAgICAgICAgICAgLy8gZG9uJ3QgbmVlZCBzeW5jaHJvbml6YXRpb24KLSAgICAgICAgICAgICAgICBSZWdpb24gICAgICAgICAgICAgICAgICAgICAgbURpcnR5UmVnaW9uOwotICAgICAgICAgICAgICAgIFJlZ2lvbiAgICAgICAgICAgICAgICAgICAgICBtSW52YWxpZFJlZ2lvbjsKLSAgICAgICAgICAgICAgICBSZWdpb24gICAgICAgICAgICAgICAgICAgICAgbVdvcm1ob2xlUmVnaW9uOwotICAgICAgICAgICAgICAgIENsaWVudCogICAgICAgICAgICAgICAgICAgICBtTGFzdFNjaGVkdWxlZEJyb2FkY2FzdDsKLSAgICAgICAgICAgICAgICBTb3J0ZWRWZWN0b3I8Q2xpZW50Kj4gICAgICAgbVNjaGVkdWxlZEJyb2FkY2FzdHM7Ci0gICAgICAgICAgICAgICAgYm9vbCAgICAgICAgICAgICAgICAgICAgICAgIG1WaXNpYmxlUmVnaW9uc0RpcnR5OwotICAgICAgICAgICAgICAgIGJvb2wgICAgICAgICAgICAgICAgICAgICAgICBtRGVmZXJSZWxlYXNlQ29uc29sZTsKLSAgICAgICAgICAgICAgICBib29sICAgICAgICAgICAgICAgICAgICAgICAgbUZyZWV6ZURpc3BsYXk7Ci0gICAgICAgICAgICAgICAgaW50MzJfdCAgICAgICAgICAgICAgICAgICAgIG1GcmVlemVDb3VudDsKLSAgICAgICAgICAgICAgICBuc2Vjc190ICAgICAgICAgICAgICAgICAgICAgbUZyZWV6ZURpc3BsYXlUaW1lOwotICAgICAgICAgICAgICAgIGZyaWVuZCBjbGFzcyBPcmllbnRhdGlvbkFuaW1hdGlvbjsKLSAgICAgICAgICAgICAgICBPcmllbnRhdGlvbkFuaW1hdGlvbiogICAgICAgbU9yaWVudGF0aW9uQW5pbWF0aW9uOwotCi0gICAgICAgICAgICAgICAgLy8gYWNjZXNzIHByb3RlY3RlZCBieSBtRGVidWdMb2NrCi0gICAgbXV0YWJsZSAgICAgTXV0ZXggICAgICAgICAgICAgICAgICAgICAgIG1EZWJ1Z0xvY2s7Ci0gICAgICAgICAgICAgICAgc3A8Q1BVR2F1Z2U+ICAgICAgICAgICAgICAgIG1DcHVHYXVnZTsKLQotICAgICAgICAgICAgICAgIC8vIGRvbid0IHVzZSBhIGxvY2sgZm9yIHRoZXNlLCB3ZSBkb24ndCBjYXJlCi0gICAgICAgICAgICAgICAgaW50ICAgICAgICAgICAgICAgICAgICAgICAgIG1EZWJ1Z1JlZ2lvbjsKLSAgICAgICAgICAgICAgICBpbnQgICAgICAgICAgICAgICAgICAgICAgICAgbURlYnVnQ3B1OwotICAgICAgICAgICAgICAgIGludCAgICAgICAgICAgICAgICAgICAgICAgICBtRGVidWdGcHM7Ci0gICAgICAgICAgICAgICAgaW50ICAgICAgICAgICAgICAgICAgICAgICAgIG1EZWJ1Z0JhY2tncm91bmQ7Ci0gICAgICAgICAgICAgICAgaW50ICAgICAgICAgICAgICAgICAgICAgICAgIG1EZWJ1Z05vQm9vdEFuaW1hdGlvbjsKLQotICAgICAgICAgICAgICAgIC8vIHRoZXNlIGFyZSB0aHJlYWQgc2FmZQotICAgIG11dGFibGUgICAgIEJhcnJpZXIgICAgICAgICAgICAgICAgICAgICBtUmVhZHlUb1J1bkJhcnJpZXI7Ci0gICAgbXV0YWJsZSAgICAgU3VyZmFjZUZsaW5nZXJTeW5jaHJvICAgICAgIG1TeW5jT2JqZWN0OwotICAgIHZvbGF0aWxlICAgIGludDMyX3QgICAgICAgICAgICAgICAgICAgICBtRGVwbGF5ZWRUcmFuc2FjdGlvblBlbmRpbmc7Ci0KLSAgICAgICAgICAgICAgICAvLyBhdG9taWMgdmFyaWFibGVzCi0gICAgICAgICAgICAgICAgZW51bSB7Ci0gICAgICAgICAgICAgICAgICAgIGVDb25zb2xlUmVsZWFzZWQgPSAxLAotICAgICAgICAgICAgICAgICAgICBlQ29uc29sZUFjcXVpcmVkID0gMgotICAgICAgICAgICAgICAgIH07Ci0gICB2b2xhdGlsZSAgICAgaW50MzJfdCAgICAgICAgICAgICAgICAgICAgIG1Db25zb2xlU2lnbmFsczsKLQotICAgLy8gb25seSB3cml0dGVuIGluIHRoZSBtYWluIHRocmVhZCwgb25seSByZWFkIGluIG90aGVyIHRocmVhZHMKLSAgIHZvbGF0aWxlICAgICBpbnQzMl90ICAgICAgICAgICAgICAgICAgICAgbVNlY3VyZUZyYW1lQnVmZmVyOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIEZyZWV6ZUxvY2sgOiBwdWJsaWMgTGlnaHRSZWZCYXNlPEZyZWV6ZUxvY2s+IHsKLSAgICBTdXJmYWNlRmxpbmdlciogbUZsaW5nZXI7Ci1wdWJsaWM6Ci0gICAgRnJlZXplTG9jayhTdXJmYWNlRmxpbmdlciogZmxpbmdlcikKLSAgICAgICAgOiBtRmxpbmdlcihmbGluZ2VyKSB7Ci0gICAgICAgIG1GbGluZ2VyLT5pbmNGcmVlemVDb3VudCgpOwotICAgIH0KLSAgICB+RnJlZXplTG9jaygpIHsKLSAgICAgICAgbUZsaW5nZXItPmRlY0ZyZWV6ZUNvdW50KCk7Ci0gICAgfQotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIEJDbGllbnQgOiBwdWJsaWMgQm5TdXJmYWNlRmxpbmdlckNsaWVudAotewotcHVibGljOgotICAgIEJDbGllbnQoU3VyZmFjZUZsaW5nZXIgKmZsaW5nZXIsIENsaWVudElEIGNpZCwKLSAgICAgICAgICAgIGNvbnN0IHNwPElNZW1vcnk+JiBjYmxrKTsKLSAgICB+QkNsaWVudCgpOwotCi0gICAgLy8gSVN1cmZhY2VGbGluZ2VyQ2xpZW50IGludGVyZmFjZQotICAgIHZpcnR1YWwgdm9pZCBnZXRDb250cm9sQmxvY2tzKHNwPElNZW1vcnk+KiBjdHJsKSBjb25zdDsKLQotICAgIHZpcnR1YWwgc3A8SVN1cmZhY2U+IGNyZWF0ZVN1cmZhY2UoCi0gICAgICAgICAgICBzdXJmYWNlX2RhdGFfdCogcGFyYW1zLCBpbnQgcGlkLAotICAgICAgICAgICAgRGlzcGxheUlEIGRpc3BsYXksIHVpbnQzMl90IHcsIHVpbnQzMl90IGgsUGl4ZWxGb3JtYXQgZm9ybWF0LAotICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MpOwotCi0gICAgdmlydHVhbCBzdGF0dXNfdCBkZXN0cm95U3VyZmFjZShTdXJmYWNlSUQgc3VyZmFjZUlkKTsKLSAgICB2aXJ0dWFsIHN0YXR1c190IHNldFN0YXRlKGludDMyX3QgY291bnQsIGNvbnN0IGxheWVyX3N0YXRlX3QqIHN0YXRlcyk7Ci0KLXByaXZhdGU6Ci0gICAgQ2xpZW50SUQgICAgICAgICAgICBtSWQ7Ci0gICAgU3VyZmFjZUZsaW5nZXIqICAgICBtRmxpbmdlcjsKLSAgICBzcDxJTWVtb3J5PiAgICAgICAgIG1DYmxrOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9TVVJGQUNFX0ZMSU5HRVJfSApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9Ub2tlbml6ZXIuY3BwIGIvbGlicy9zdXJmYWNlZmxpbmdlci9Ub2tlbml6ZXIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlZjUxZDZhLi4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvVG9rZW5pemVyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDE3MiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0KLSNpbmNsdWRlICJUb2tlbml6ZXIuaCIKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLUFORFJPSURfQkFTSUNfVFlQRVNfVFJBSVRTKFRva2VuaXplcjo6cnVuX3QpCi0KLVRva2VuaXplcjo6VG9rZW5pemVyKCkKLXsKLX0KLQotVG9rZW5pemVyOjpUb2tlbml6ZXIoY29uc3QgVG9rZW5pemVyJiBvdGhlcikKLSAgICA6IG1SYW5nZXMob3RoZXIubVJhbmdlcykKLXsKLX0KLQotVG9rZW5pemVyOjp+VG9rZW5pemVyKCkKLXsKLX0KLQotdWludDMyX3QgVG9rZW5pemVyOjphY3F1aXJlKCkKLXsKLSAgICBpZiAoIW1SYW5nZXMuc2l6ZSgpIHx8IG1SYW5nZXNbMF0uZmlyc3QpIHsKLSAgICAgICAgX2luc2VydFRva2VuQXQoMCwwKTsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotICAgIAotICAgIC8vIGp1c3QgZXh0ZW5kIHRoZSBmaXJzdCBydW4KLSAgICBjb25zdCBydW5fdCYgcnVuID0gbVJhbmdlc1swXTsKLSAgICB1aW50MzJfdCB0b2tlbiA9IHJ1bi5maXJzdCArIHJ1bi5sZW5ndGg7Ci0gICAgX2luc2VydFRva2VuQXQodG9rZW4sIDEpOwotICAgIHJldHVybiB0b2tlbjsKLX0KLQotYm9vbCBUb2tlbml6ZXI6OmlzQWNxdWlyZWQodWludDMyX3QgdG9rZW4pIGNvbnN0Ci17Ci0gICAgcmV0dXJuIChfaW5kZXhPcmRlck9mKHRva2VuKSA+PSAwKTsKLX0KLQotc3RhdHVzX3QgVG9rZW5pemVyOjpyZXNlcnZlKHVpbnQzMl90IHRva2VuKQotewotICAgIHNpemVfdCBvOwotICAgIGNvbnN0IHNzaXplX3QgaSA9IF9pbmRleE9yZGVyT2YodG9rZW4sICZvKTsKLSAgICBpZiAoaSA+PSAwKSB7Ci0gICAgICAgIHJldHVybiBCQURfVkFMVUU7IC8vIHRoaXMgdG9rZW4gaXMgYWxyZWFkeSB0YWtlbgotICAgIH0KLSAgICBzc2l6ZV90IGVyciA9IF9pbnNlcnRUb2tlbkF0KHRva2VuLCBvKTsKLSAgICByZXR1cm4gKGVycjwwKSA/IGVyciA6IHN0YXR1c190KE5PX0VSUk9SKTsKLX0KLQotc3RhdHVzX3QgVG9rZW5pemVyOjpyZWxlYXNlKHVpbnQzMl90IHRva2VuKQotewotICAgIGNvbnN0IHNzaXplX3QgaSA9IF9pbmRleE9yZGVyT2YodG9rZW4pOwotICAgIGlmIChpID49IDApIHsKLSAgICAgICAgY29uc3QgcnVuX3QmIHJ1biA9IG1SYW5nZXNbaV07Ci0gICAgICAgIGlmICgodG9rZW4gPj0gcnVuLmZpcnN0KSAmJiAodG9rZW4gPCBydW4uZmlyc3QrcnVuLmxlbmd0aCkpIHsKLSAgICAgICAgICAgIC8vIHRva2VuIGluIHRoaXMgcmFuZ2UsIHdlIG5lZWQgdG8gc3BsaXQKLSAgICAgICAgICAgIHJ1bl90JiBydW4gPSBtUmFuZ2VzLmVkaXRJdGVtQXQoaSk7Ci0gICAgICAgICAgICBpZiAoKHRva2VuID09IHJ1bi5maXJzdCkgfHwgKHRva2VuID09IHJ1bi5maXJzdCtydW4ubGVuZ3RoLTEpKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHRva2VuID09IHJ1bi5maXJzdCkgewotICAgICAgICAgICAgICAgICAgICBydW4uZmlyc3QgKz0gMTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcnVuLmxlbmd0aCAtPSAxOwotICAgICAgICAgICAgICAgIGlmIChydW4ubGVuZ3RoID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gWFhYOiBzaG91bGQgd2Ugc3lzdGVtYXRpY2FsbHkgcmVtb3ZlIGEgcnVuIHRoYXQncyBlbXB0eT8KLSAgICAgICAgICAgICAgICAgICAgbVJhbmdlcy5yZW1vdmVJdGVtc0F0KGkpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgLy8gc3BsaXQgdGhlIHJ1bgotICAgICAgICAgICAgICAgIHJ1bl90IG5ld19ydW47Ci0gICAgICAgICAgICAgICAgbmV3X3J1bi5maXJzdCA9IHRva2VuKzE7Ci0gICAgICAgICAgICAgICAgbmV3X3J1bi5sZW5ndGggPSBydW4uZmlyc3QrcnVuLmxlbmd0aCAtIG5ld19ydW4uZmlyc3Q7Ci0gICAgICAgICAgICAgICAgcnVuLmxlbmd0aCA9IHRva2VuIC0gcnVuLmZpcnN0OwotICAgICAgICAgICAgICAgIG1SYW5nZXMuaW5zZXJ0QXQobmV3X3J1biwgaSsxKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gTkFNRV9OT1RfRk9VTkQ7Ci19Ci0KLXNzaXplX3QgVG9rZW5pemVyOjpfaW5kZXhPcmRlck9mKHVpbnQzMl90IHRva2VuLCBzaXplX3QqIG9yZGVyKSBjb25zdAotewotICAgIC8vIGJpbmFyeSBzZWFyY2gKLSAgICBzc2l6ZV90IGVyciA9IE5BTUVfTk9UX0ZPVU5EOwotICAgIHNzaXplX3QgbCA9IDA7Ci0gICAgc3NpemVfdCBoID0gbVJhbmdlcy5zaXplKCktMTsKLSAgICBzc2l6ZV90IG1pZDsKLSAgICBjb25zdCBydW5fdCogYSA9IG1SYW5nZXMuYXJyYXkoKTsKLSAgICB3aGlsZSAobCA8PSBoKSB7Ci0gICAgICAgIG1pZCA9IGwgKyAoaCAtIGwpLzI7Ci0gICAgICAgIGNvbnN0IHJ1bl90KiBjb25zdCBjdXJyID0gYSArIG1pZDsKLSAgICAgICAgaW50IGMgPSAwOwotICAgICAgICBpZiAodG9rZW4gPCBjdXJyLT5maXJzdCkgICAgICAgICAgICAgICAgICAgICAgICBjID0gMTsKLSAgICAgICAgZWxzZSBpZiAodG9rZW4gPj0gY3Vyci0+Zmlyc3QrY3Vyci0+bGVuZ3RoKSAgICAgYyA9IC0xOwotICAgICAgICBpZiAoYyA9PSAwKSB7Ci0gICAgICAgICAgICBlcnIgPSBsID0gbWlkOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0gZWxzZSBpZiAoYyA8IDApIHsKLSAgICAgICAgICAgIGwgPSBtaWQgKyAxOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaCA9IG1pZCAtIDE7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgaWYgKG9yZGVyKSAqb3JkZXIgPSBsOwotICAgIHJldHVybiBlcnI7Ci19Ci0KLXNzaXplX3QgVG9rZW5pemVyOjpfaW5zZXJ0VG9rZW5BdCh1aW50MzJfdCB0b2tlbiwgc2l6ZV90IGluZGV4KQotewotICAgIGNvbnN0IHNpemVfdCBjID0gbVJhbmdlcy5zaXplKCk7Ci0KLSAgICBpZiAoaW5kZXggPj0gMSkgewotICAgICAgICAvLyBkbyB3ZSBuZWVkIHRvIG1lcmdlIHdpdGggdGhlIHByZXZpb3VzIHJ1bj8KLSAgICAgICAgcnVuX3QmIHAgPSBtUmFuZ2VzLmVkaXRJdGVtQXQoaW5kZXgtMSk7Ci0gICAgICAgIGlmIChwLmZpcnN0K3AubGVuZ3RoID09IHRva2VuKSB7Ci0gICAgICAgICAgICBwLmxlbmd0aCArPSAxOwotICAgICAgICAgICAgaWYgKGluZGV4IDwgYykgewotICAgICAgICAgICAgICAgIGNvbnN0IHJ1bl90JiBuID0gbVJhbmdlc1tpbmRleF07Ci0gICAgICAgICAgICAgICAgaWYgKHRva2VuKzEgPT0gbi5maXJzdCkgewotICAgICAgICAgICAgICAgICAgICBwLmxlbmd0aCArPSBuLmxlbmd0aDsKLSAgICAgICAgICAgICAgICAgICAgbVJhbmdlcy5yZW1vdmVJdGVtc0F0KGluZGV4KTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gaW5kZXg7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgaWYgKGluZGV4IDwgYykgewotICAgICAgICAvLyBkbyB3ZSBuZWVkIHRvIG1lcmdlIHdpdGggdGhlIG5leHQgcnVuPwotICAgICAgICBydW5fdCYgbiA9IG1SYW5nZXMuZWRpdEl0ZW1BdChpbmRleCk7Ci0gICAgICAgIGlmICh0b2tlbisxID09IG4uZmlyc3QpIHsKLSAgICAgICAgICAgIG4uZmlyc3QgLT0gMTsKLSAgICAgICAgICAgIG4ubGVuZ3RoICs9IDE7Ci0gICAgICAgICAgICByZXR1cm4gaW5kZXg7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gbVJhbmdlcy5pbnNlcnRBdChydW5fdCh0b2tlbiwxKSwgaW5kZXgpOwotfQotCi12b2lkIFRva2VuaXplcjo6ZHVtcCgpIGNvbnN0Ci17Ci0gICAgY29uc3QgcnVuX3QqIHJhbmdlcyA9IG1SYW5nZXMuYXJyYXkoKTsKLSAgICBjb25zdCBzaXplX3QgYyA9IG1SYW5nZXMuc2l6ZSgpOwotICAgIHByaW50ZigiVG9rZW5pemVyICglcCwgc2l6ZSA9ICVsdSlcbiIsIHRoaXMsIGMpOwotICAgIGZvciAoc2l6ZV90IGk9MCA7IGk8YyA7IGkrKykgewotICAgICAgICBwcmludGYoIiVsdTogKCV1LCAldSlcbiIsIGksIHJhbmdlc1tpXS5maXJzdCwgcmFuZ2VzW2ldLmxlbmd0aCk7Ci0gICAgfQotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL1Rva2VuaXplci5oIGIvbGlicy9zdXJmYWNlZmxpbmdlci9Ub2tlbml6ZXIuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNmIzMDU3ZC4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL1Rva2VuaXplci5oCisrKyAvZGV2L251bGwKQEAgLTEsNTcgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9UT0tFTklaRVJfSAotI2RlZmluZSBBTkRST0lEX1RPS0VOSVpFUl9ICi0KLSNpbmNsdWRlIDx1dGlscy9WZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIFRva2VuaXplcgotewotcHVibGljOgotICAgICAgICAgICAgICAgIFRva2VuaXplcigpOwotICAgICAgICAgICAgICAgIFRva2VuaXplcihjb25zdCBUb2tlbml6ZXImIG90aGVyKTsKLSAgICAgICAgICAgICAgICB+VG9rZW5pemVyKCk7Ci0KLSAgICB1aW50MzJfdCAgICBhY3F1aXJlKCk7Ci0gICAgc3RhdHVzX3QgICAgcmVzZXJ2ZSh1aW50MzJfdCB0b2tlbik7Ci0gICAgc3RhdHVzX3QgICAgcmVsZWFzZSh1aW50MzJfdCB0b2tlbik7Ci0gICAgYm9vbCAgICAgICAgaXNBY3F1aXJlZCh1aW50MzJfdCB0b2tlbikgY29uc3Q7Ci0KLSAgICB2b2lkIGR1bXAoKSBjb25zdDsKLQotICAgIHN0cnVjdCBydW5fdCB7Ci0gICAgICAgIHJ1bl90KCkge307Ci0gICAgICAgIHJ1bl90KHVpbnQzMl90IGYsIHVpbnQzMl90IGwpIDogZmlyc3QoZiksIGxlbmd0aChsKSB7fQotICAgICAgICB1aW50MzJfdCAgICBmaXJzdDsKLSAgICAgICAgdWludDMyX3QgICAgbGVuZ3RoOwotICAgIH07Ci1wcml2YXRlOgotICAgIHNzaXplX3QgX2luZGV4T3JkZXJPZih1aW50MzJfdCB0b2tlbiwgc2l6ZV90KiBvcmRlcj0wKSBjb25zdDsKLSAgICBzc2l6ZV90IF9pbnNlcnRUb2tlbkF0KHVpbnQzMl90IHRva2VuLCBzaXplX3QgaW5kZXgpOwotICAgIFZlY3RvcjxydW5fdD4gICBtUmFuZ2VzOwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLy8gQU5EUk9JRF9UT0tFTklaRVJfSApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9UcmFuc2Zvcm0uY3BwIGIvbGlicy9zdXJmYWNlZmxpbmdlci9UcmFuc2Zvcm0uY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiZWM3YTY0Li4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvVHJhbnNmb3JtLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDIwNCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2luY2x1ZGUgPHVpL1JlZ2lvbi5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS9waXhlbGZsaW5nZXIvZ2dsX2ZpeGVkLmg+Ci0KLSNpbmNsdWRlICJUcmFuc2Zvcm0uaCIKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNkZWZpbmUgTElLRUxZKCBleHAgKSAgICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgdHJ1ZSAgKSkKLSNkZWZpbmUgVU5MSUtFTFkoIGV4cCApICAgICAoX19idWlsdGluX2V4cGVjdCggKGV4cCkgIT0gMCwgZmFsc2UgKSkKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLVRyYW5zZm9ybTo6VHJhbnNmb3JtKCkKLSAgICA6IG1UeXBlKDApCi17Ci0gICAgbVRyYW5zZm9ybS5yZXNldCgpOwotfQotCi1UcmFuc2Zvcm06OlRyYW5zZm9ybShjb25zdCBUcmFuc2Zvcm0mICBvdGhlcikKLSAgICA6IG1UcmFuc2Zvcm0ob3RoZXIubVRyYW5zZm9ybSksIG1UeXBlKG90aGVyLm1UeXBlKQotewotfQotCi1UcmFuc2Zvcm06On5UcmFuc2Zvcm0oKSB7Ci19Ci0KLVRyYW5zZm9ybSBUcmFuc2Zvcm06Om9wZXJhdG9yICogKGNvbnN0IFRyYW5zZm9ybSYgcmhzKSBjb25zdAotewotICAgIGlmIChMSUtFTFkobVR5cGUgPT0gMCkpCi0gICAgICAgIHJldHVybiByaHM7Ci0KLSAgICBUcmFuc2Zvcm0gcigqdGhpcyk7Ci0gICAgci5tVHJhbnNmb3JtLnByZUNvbmNhdChyaHMubVRyYW5zZm9ybSk7Ci0gICAgci5tVHlwZSB8PSByaHMubVR5cGU7Ci0gICAgcmV0dXJuIHI7Ci19Ci0KLWZsb2F0IFRyYW5zZm9ybTo6b3BlcmF0b3IgW10gKGludCBpKSBjb25zdAotewotICAgIGZsb2F0IHIgPSAwOwotICAgIHN3aXRjaChpKSB7Ci0gICAgICAgIGNhc2UgMDogciA9IFNrU2NhbGFyVG9GbG9hdCggbVRyYW5zZm9ybVtTa01hdHJpeDo6a01TY2FsZVhdICk7ICBicmVhazsKLSAgICAgICAgY2FzZSAxOiByID0gU2tTY2FsYXJUb0Zsb2F0KCBtVHJhbnNmb3JtW1NrTWF0cml4OjprTVNrZXdYXSApOyAgIGJyZWFrOwotICAgICAgICBjYXNlIDI6IHIgPSBTa1NjYWxhclRvRmxvYXQoIG1UcmFuc2Zvcm1bU2tNYXRyaXg6OmtNU2tld1ldICk7ICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgMzogciA9IFNrU2NhbGFyVG9GbG9hdCggbVRyYW5zZm9ybVtTa01hdHJpeDo6a01TY2FsZVldICk7ICBicmVhazsKLSAgICB9Ci0gICAgcmV0dXJuIHI7Ci19Ci0KLXVpbnQ4X3QgVHJhbnNmb3JtOjp0eXBlKCkgY29uc3QKLXsKLSAgICBpZiAoVU5MSUtFTFkobVR5cGUgJiAweDgwMDAwMDAwKSkgewotICAgICAgICBtVHlwZSA9IG1UcmFuc2Zvcm0uZ2V0VHlwZSgpOwotICAgIH0KLSAgICByZXR1cm4gdWludDhfdChtVHlwZSAmIDB4RkYpOwotfQotCi1ib29sIFRyYW5zZm9ybTo6dHJhbnNmb3JtZWQoKSBjb25zdCB7Ci0gICAgcmV0dXJuIHR5cGUoKSA+IFNrTWF0cml4OjprVHJhbnNsYXRlX01hc2s7Ci19Ci0KLWludCBUcmFuc2Zvcm06OnR4KCkgY29uc3QgewotICAgIHJldHVybiBTa1NjYWxhclJvdW5kKCBtVHJhbnNmb3JtW1NrTWF0cml4OjprTVRyYW5zWF0gKTsKLX0KLQotaW50IFRyYW5zZm9ybTo6dHkoKSBjb25zdCB7Ci0gICAgcmV0dXJuIFNrU2NhbGFyUm91bmQoIG1UcmFuc2Zvcm1bU2tNYXRyaXg6OmtNVHJhbnNZXSApOwotfQotCi12b2lkIFRyYW5zZm9ybTo6cmVzZXQoKSB7Ci0gICAgbVRyYW5zZm9ybS5yZXNldCgpOwotICAgIG1UeXBlID0gMDsKLX0KLQotdm9pZCBUcmFuc2Zvcm06OnNldCggZmxvYXQgeHgsIGZsb2F0IHh5LAotICAgICAgICAgICAgICAgICAgICAgZmxvYXQgeXgsIGZsb2F0IHl5KQotewotICAgIG1UcmFuc2Zvcm0uc2V0KFNrTWF0cml4OjprTVNjYWxlWCwgU2tGbG9hdFRvU2NhbGFyKHh4KSk7Ci0gICAgbVRyYW5zZm9ybS5zZXQoU2tNYXRyaXg6OmtNU2tld1gsIFNrRmxvYXRUb1NjYWxhcih4eSkpOwotICAgIG1UcmFuc2Zvcm0uc2V0KFNrTWF0cml4OjprTVNrZXdZLCBTa0Zsb2F0VG9TY2FsYXIoeXgpKTsKLSAgICBtVHJhbnNmb3JtLnNldChTa01hdHJpeDo6a01TY2FsZVksIFNrRmxvYXRUb1NjYWxhcih5eSkpOwotICAgIG1UeXBlIHw9IDB4ODAwMDAwMDA7Ci19Ci0KLXZvaWQgVHJhbnNmb3JtOjpzZXQoaW50IHR4LCBpbnQgdHkpCi17Ci0gICAgaWYgKHR4IHwgdHkpIHsKLSAgICAgICAgbVRyYW5zZm9ybS5zZXQoU2tNYXRyaXg6OmtNVHJhbnNYLCBTa0ludFRvU2NhbGFyKHR4KSk7Ci0gICAgICAgIG1UcmFuc2Zvcm0uc2V0KFNrTWF0cml4OjprTVRyYW5zWSwgU2tJbnRUb1NjYWxhcih0eSkpOwotICAgICAgICBtVHlwZSB8PSBTa01hdHJpeDo6a1RyYW5zbGF0ZV9NYXNrOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIG1UcmFuc2Zvcm0uc2V0KFNrTWF0cml4OjprTVRyYW5zWCwgMCk7Ci0gICAgICAgIG1UcmFuc2Zvcm0uc2V0KFNrTWF0cml4OjprTVRyYW5zWSwgMCk7Ci0gICAgICAgIG1UeXBlICY9IH5Ta01hdHJpeDo6a1RyYW5zbGF0ZV9NYXNrOwotICAgIH0KLX0KLQotdm9pZCBUcmFuc2Zvcm06OnRyYW5zZm9ybShHTGZpeGVkKiBwb2ludCwgaW50IHgsIGludCB5KSBjb25zdAotewotICAgIFNrUG9pbnQgczsKLSAgICBtVHJhbnNmb3JtLm1hcFhZKFNrSW50VG9TY2FsYXIoeCksIFNrSW50VG9TY2FsYXIoeSksICZzKTsKLSAgICBwb2ludFswXSA9IFNrU2NhbGFyVG9GaXhlZChzLmZYKTsKLSAgICBwb2ludFsxXSA9IFNrU2NhbGFyVG9GaXhlZChzLmZZKTsKLX0KLQotUmVjdCBUcmFuc2Zvcm06Om1ha2VCb3VuZHMoaW50IHcsIGludCBoKSBjb25zdAotewotICAgIFJlY3QgcjsKLSAgICBTa1JlY3QgZCwgczsKLSAgICBzLnNldCgwLCAwLCBTa0ludFRvU2NhbGFyKHcpLCBTa0ludFRvU2NhbGFyKGgpKTsKLSAgICBtVHJhbnNmb3JtLm1hcFJlY3QoJmQsIHMpOwotICAgIHIubGVmdCAgID0gU2tTY2FsYXJSb3VuZCggZC5mTGVmdCApOwotICAgIHIudG9wICAgID0gU2tTY2FsYXJSb3VuZCggZC5mVG9wICk7Ci0gICAgci5yaWdodCAgPSBTa1NjYWxhclJvdW5kKCBkLmZSaWdodCApOwotICAgIHIuYm90dG9tID0gU2tTY2FsYXJSb3VuZCggZC5mQm90dG9tICk7Ci0gICAgcmV0dXJuIHI7Ci19Ci0KLVJlY3QgVHJhbnNmb3JtOjp0cmFuc2Zvcm0oY29uc3QgUmVjdCYgYm91bmRzKSBjb25zdAotewotICAgIFJlY3QgcjsKLSAgICBTa1JlY3QgZCwgczsKLSAgICBzLnNldCggIFNrSW50VG9TY2FsYXIoIGJvdW5kcy5sZWZ0ICksCi0gICAgICAgICAgICBTa0ludFRvU2NhbGFyKCBib3VuZHMudG9wICksCi0gICAgICAgICAgICBTa0ludFRvU2NhbGFyKCBib3VuZHMucmlnaHQgKSwKLSAgICAgICAgICAgIFNrSW50VG9TY2FsYXIoIGJvdW5kcy5ib3R0b20gKSk7Ci0gICAgbVRyYW5zZm9ybS5tYXBSZWN0KCZkLCBzKTsKLSAgICByLmxlZnQgICA9IFNrU2NhbGFyUm91bmQoIGQuZkxlZnQgKTsKLSAgICByLnRvcCAgICA9IFNrU2NhbGFyUm91bmQoIGQuZlRvcCApOwotICAgIHIucmlnaHQgID0gU2tTY2FsYXJSb3VuZCggZC5mUmlnaHQgKTsKLSAgICByLmJvdHRvbSA9IFNrU2NhbGFyUm91bmQoIGQuZkJvdHRvbSApOwotICAgIHJldHVybiByOwotfQotCi1SZWdpb24gVHJhbnNmb3JtOjp0cmFuc2Zvcm0oY29uc3QgUmVnaW9uJiByZWcpIGNvbnN0Ci17Ci0gICAgUmVnaW9uIG91dDsKLSAgICBpZiAoVU5MSUtFTFkodHJhbnNmb3JtZWQoKSkpIHsKLSAgICAgICAgaWYgKExJS0VMWShwcmVzZXJ2ZVJlY3RzKCkpKSB7Ci0gICAgICAgICAgICBSZWN0IHI7Ci0gICAgICAgICAgICBSZWdpb246Oml0ZXJhdG9yIGl0ZXJhdG9yKHJlZyk7Ci0gICAgICAgICAgICB3aGlsZSAoaXRlcmF0b3IuaXRlcmF0ZSgmcikpIHsKLSAgICAgICAgICAgICAgICBvdXQub3JTZWxmKHRyYW5zZm9ybShyKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBvdXQuc2V0KHRyYW5zZm9ybShyZWcuYm91bmRzKCkpKTsKLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIG91dCA9IHJlZy50cmFuc2xhdGUodHgoKSwgdHkoKSk7Ci0gICAgfQotICAgIHJldHVybiBvdXQ7Ci19Ci0KLWludDMyX3QgVHJhbnNmb3JtOjpnZXRPcmllbnRhdGlvbigpIGNvbnN0Ci17Ci0gICAgdWludDMyX3QgZmxhZ3MgPSAwOwotICAgIGlmIChVTkxJS0VMWSh0cmFuc2Zvcm1lZCgpKSkgewotICAgICAgICBTa1NjYWxhciBhID0gbVRyYW5zZm9ybVtTa01hdHJpeDo6a01TY2FsZVhdOwotICAgICAgICBTa1NjYWxhciBiID0gbVRyYW5zZm9ybVtTa01hdHJpeDo6a01Ta2V3WF07Ci0gICAgICAgIFNrU2NhbGFyIGMgPSBtVHJhbnNmb3JtW1NrTWF0cml4OjprTVNrZXdZXTsKLSAgICAgICAgU2tTY2FsYXIgZCA9IG1UcmFuc2Zvcm1bU2tNYXRyaXg6OmtNU2NhbGVZXTsKLSAgICAgICAgaWYgKGI9PTAgJiYgYz09MCAmJiBhICYmIGQpIHsKLSAgICAgICAgICAgIGlmIChhPDApICAgIGZsYWdzIHw9IEZMSVBfSDsKLSAgICAgICAgICAgIGlmIChkPDApICAgIGZsYWdzIHw9IEZMSVBfVjsKLSAgICAgICAgfSBlbHNlIGlmIChiICYmIGMgJiYgYT09MCAmJiBkPT0wKSB7Ci0gICAgICAgICAgICBmbGFncyB8PSBST1RfOTA7Ci0gICAgICAgICAgICBpZiAoYj4wKSAgICBmbGFncyB8PSBGTElQX0g7Ci0gICAgICAgICAgICBpZiAoYzwwKSAgICBmbGFncyB8PSBGTElQX1Y7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBmbGFncyA9IDB4ODAwMDAwMDA7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIGZsYWdzOwotfQotCi1ib29sIFRyYW5zZm9ybTo6cHJlc2VydmVSZWN0cygpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1UcmFuc2Zvcm0ucmVjdFN0YXlzUmVjdCgpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvVHJhbnNmb3JtLmggYi9saWJzL3N1cmZhY2VmbGluZ2VyL1RyYW5zZm9ybS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwYjQ4MzVlLi4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvVHJhbnNmb3JtLmgKKysrIC9kZXYvbnVsbApAQCAtMSw4NyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX1RSQU5TRk9STV9ICi0jZGVmaW5lIEFORFJPSURfVFJBTlNGT1JNX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dWkvUG9pbnQuaD4KLSNpbmNsdWRlIDx1aS9SZWN0Lmg+Ci0KLSNpbmNsdWRlIDxHTEVTL2dsLmg+Ci0KLSNpbmNsdWRlIDxjb3JlL1NrTWF0cml4Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgUmVnaW9uOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgVHJhbnNmb3JtCi17Ci1wdWJsaWM6Ci0gICAgICAgICAgICAgICAgICAgIFRyYW5zZm9ybSgpOwotICAgICAgICAgICAgICAgICAgICBUcmFuc2Zvcm0oY29uc3QgVHJhbnNmb3JtJiAgb3RoZXIpOwotICAgICAgICAgICAgICAgICAgICB+VHJhbnNmb3JtKCk7Ci0KLSAgICAgICAgICAgIGVudW0gb3JpZW50YXRpb25fZmxhZ3MgewotICAgICAgICAgICAgICAgIFJPVF8wICAgPSAweDAwMDAwMDAwLAotICAgICAgICAgICAgICAgIEZMSVBfSCAgPSAweDAwMDAwMDAxLAotICAgICAgICAgICAgICAgIEZMSVBfViAgPSAweDAwMDAwMDAyLAotICAgICAgICAgICAgICAgIFJPVF85MCAgPSAweDAwMDAwMDA0LAotICAgICAgICAgICAgICAgIFJPVF8xODAgPSBGTElQX0h8RkxJUF9WLAotICAgICAgICAgICAgICAgIFJPVF8yNzAgPSBST1RfMTgwfFJPVF85MCwKLSAgICAgICAgICAgICAgICBST1RfSU5WQUxJRCA9IDB4ODAwMDAwMDAKLSAgICAgICAgICAgIH07Ci0KLSAgICAgICAgICAgIGJvb2wgICAgdHJhbnNmb3JtZWQoKSBjb25zdDsKLSAgICAgICAgICAgIGludDMyX3QgZ2V0T3JpZW50YXRpb24oKSBjb25zdDsKLSAgICAgICAgICAgIGJvb2wgICAgcHJlc2VydmVSZWN0cygpIGNvbnN0OwotICAgICAgICAgICAgCi0gICAgICAgICAgICBpbnQgICAgIHR4KCkgY29uc3Q7Ci0gICAgICAgICAgICBpbnQgICAgIHR5KCkgY29uc3Q7Ci0gICAgICAgIAotICAgICAgICAgICAgdm9pZCAgICByZXNldCgpOwotICAgICAgICAgICAgdm9pZCAgICBzZXQoZmxvYXQgeHgsIGZsb2F0IHh5LCBmbG9hdCB5eCwgZmxvYXQgeXkpOwotICAgICAgICAgICAgdm9pZCAgICBzZXQoaW50IHR4LCBpbnQgdHkpOwotCi0gICAgICAgICAgICBSZWN0ICAgIG1ha2VCb3VuZHMoaW50IHcsIGludCBoKSBjb25zdDsKLSAgICAgICAgICAgIHZvaWQgICAgdHJhbnNmb3JtKEdMZml4ZWQqIHBvaW50LCBpbnQgeCwgaW50IHkpIGNvbnN0OwotICAgICAgICAgICAgUmVnaW9uICB0cmFuc2Zvcm0oY29uc3QgUmVnaW9uJiByZWcpIGNvbnN0OwotICAgICAgICAgICAgUmVjdCAgICB0cmFuc2Zvcm0oY29uc3QgUmVjdCYgYm91bmRzKSBjb25zdDsKLQotICAgICAgICAgICAgVHJhbnNmb3JtIG9wZXJhdG9yICogKGNvbnN0IFRyYW5zZm9ybSYgcmhzKSBjb25zdDsKLSAgICAgICAgICAgIGZsb2F0IG9wZXJhdG9yIFtdIChpbnQgaSkgY29uc3Q7Ci0KLSAgICBpbmxpbmUgdWludDMyX3QgZ2V0VHlwZSgpIGNvbnN0IHsgcmV0dXJuIHR5cGUoKTsgfQotICAgICAgICAgICAgCi0gICAgaW5saW5lIFRyYW5zZm9ybShib29sKSA6IG1UeXBlKDB4RkYpIHsgfTsKLQotcHJpdmF0ZToKLSAgICB1aW50OF90ICAgICB0eXBlKCkgY29uc3Q7Ci0KLXByaXZhdGU6Ci0gICAgICAgICAgICBTa01hdHJpeCAgICBtVHJhbnNmb3JtOwotICAgIG11dGFibGUgdWludDMyX3QgICAgbVR5cGU7ICAgICAgCi19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvKiBBTkRST0lEX1RSQU5TRk9STV9IICovCmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL1ZSYW1IZWFwLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvVlJhbUhlYXAuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwY2NkNzFmLi4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvVlJhbUhlYXAuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTc2ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIlN1cmZhY2VGbGluZ2VyIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPGVycm5vLmg+Ci0jaW5jbHVkZSA8bWF0aC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHN5cy9zdGF0Lmg+Ci0jaW5jbHVkZSA8c3lzL2lvY3RsLmg+Ci0KLSNpbmNsdWRlIDxjdXRpbHMvbG9nLmg+Ci0jaW5jbHVkZSA8Y3V0aWxzL3Byb3BlcnRpZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL01lbW9yeURlYWxlci5oPgotI2luY2x1ZGUgPHV0aWxzL01lbW9yeUJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy9NZW1vcnlIZWFwUG1lbS5oPgotI2luY2x1ZGUgPHV0aWxzL01lbW9yeUhlYXBCYXNlLmg+Ci0KLSNpbmNsdWRlICJHUFVIYXJkd2FyZS9HUFVIYXJkd2FyZS5oIgotI2luY2x1ZGUgIlN1cmZhY2VGbGluZ2VyLmgiCi0jaW5jbHVkZSAiVlJhbUhlYXAuaCIKLQotI2lmIEhBVkVfQU5EUk9JRF9PUwotI2luY2x1ZGUgPGxpbnV4L2FuZHJvaWRfcG1lbS5oPgotI2VuZGlmCi0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLyoKLSAqIEFtb3VudCBvZiBtZW1vcnkgd2UgcmVzZXJ2ZSBmb3Igc3VyZmFjZSwgcGVyIGNsaWVudCBpbiBQTUVNCi0gKiAoUE1FTSBpcyB1c2VkIGZvciAyRCBhY2NlbGVyYXRpb24pCi0gKiA4IE1CIG9mIGFkZHJlc3Mgc3BhY2UgcGVyIGNsaWVudCBzaG91bGQgYmUgZW5vdWdoLgotICovCi1zdGF0aWMgY29uc3QgaW50IFBNRU1fU0laRSA9IGludCg4ICogMTAyNCAqIDEwMjQpOwotCi1pbnQgU3VyZmFjZUhlYXBNYW5hZ2VyOjpnbG9iYWxfcG1lbV9oZWFwID0gMDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLVN1cmZhY2VIZWFwTWFuYWdlcjo6U3VyZmFjZUhlYXBNYW5hZ2VyKGNvbnN0IHNwPFN1cmZhY2VGbGluZ2VyPiYgZmxpbmdlciwgCi0gICAgICAgIHNpemVfdCBjbGllbnRIZWFwU2l6ZSkKLSAgICA6IG1GbGluZ2VyKGZsaW5nZXIpLCBtQ2xpZW50SGVhcFNpemUoY2xpZW50SGVhcFNpemUpCi17Ci0gICAgU3VyZmFjZUhlYXBNYW5hZ2VyOjpnbG9iYWxfcG1lbV9oZWFwID0gMTsKLX0KLQotU3VyZmFjZUhlYXBNYW5hZ2VyOjp+U3VyZmFjZUhlYXBNYW5hZ2VyKCkKLXsKLX0KLQotdm9pZCBTdXJmYWNlSGVhcE1hbmFnZXI6Om9uRmlyc3RSZWYoKQotewotICAgIGlmIChnbG9iYWxfcG1lbV9oZWFwKSB7Ci0gICAgICAgIGNvbnN0IGNoYXIqIGRldmljZSA9ICIvZGV2L3BtZW0iOwotICAgICAgICBtUE1lbUhlYXAgPSBuZXcgUE1lbUhlYXAoZGV2aWNlLCBQTUVNX1NJWkUpOwotICAgICAgICBpZiAobVBNZW1IZWFwLT5iYXNlKCkgPT0gTUFQX0ZBSUxFRCkgewotICAgICAgICAgICAgbVBNZW1IZWFwLmNsZWFyKCk7Ci0gICAgICAgICAgICBnbG9iYWxfcG1lbV9oZWFwID0gMDsKLSAgICAgICAgfQotICAgIH0KLX0KLQotc3A8TWVtb3J5RGVhbGVyPiBTdXJmYWNlSGVhcE1hbmFnZXI6OmNyZWF0ZUhlYXAoCi0gICAgICAgIHVpbnQzMl90IGZsYWdzLAotICAgICAgICBwaWRfdCBjbGllbnRfcGlkLAotICAgICAgICBjb25zdCBzcDxNZW1vcnlEZWFsZXI+JiBkZWZhdWx0QWxsb2NhdG9yKQotewotICAgIHNwPE1lbW9yeURlYWxlcj4gZGVhbGVyOyAKLQotICAgIGlmIChmbGFncyAmIElTdXJmYWNlQ29tcG9zZXI6OmVHUFUpIHsKLSAgICAgICAgLy8gZG9uJ3QgZ3JhbnQgR1BVIG1lbW9yeSBpZiBHUFUgaXMgZGlzYWJsZWQKLSAgICAgICAgY2hhciB2YWx1ZVtQUk9QRVJUWV9WQUxVRV9NQVhdOwotICAgICAgICBwcm9wZXJ0eV9nZXQoImRlYnVnLmVnbC5odyIsIHZhbHVlLCAiMSIpOwotICAgICAgICBpZiAoYXRvaSh2YWx1ZSkgPT0gMCkgewotICAgICAgICAgICAgZmxhZ3MgJj0gfklTdXJmYWNlQ29tcG9zZXI6OmVHUFU7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBpZiAoZmxhZ3MgJiBJU3VyZmFjZUNvbXBvc2VyOjplR1BVKSB7Ci0gICAgICAgIC8vIEZJWE1FOiB0aGlzIGlzIG1zbTcyMDFBIHNwZWNpZmljLCB3aGVyZSBncHUgc3VyZmFjZXMgbWF5IG5vdCBiZSBzZWN1cmUKLSAgICAgICAgaWYgKCEoZmxhZ3MgJiBJU3VyZmFjZUNvbXBvc2VyOjplU2VjdXJlKSkgewotICAgICAgICAgICAgLy8gaWYgR1BVIGRvZXNuJ3Qgd29yaywgd2UgdHJ5IGVIYXJkd2FyZQotICAgICAgICAgICAgZmxhZ3MgfD0gSVN1cmZhY2VDb21wb3Nlcjo6ZUhhcmR3YXJlOwotICAgICAgICAgICAgLy8gYXNrZWQgZm9yIEdQVSBtZW1vcnksIHRyeSB0aGF0IGZpcnN0Ci0gICAgICAgICAgICBkZWFsZXIgPSBtRmxpbmdlci0+Z2V0R1BVKCktPnJlcXVlc3QoY2xpZW50X3BpZCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBpZiAoZGVhbGVyID09IE5VTEwpIHsKLSAgICAgICAgaWYgKGRlZmF1bHRBbGxvY2F0b3IgIT0gTlVMTCkKLSAgICAgICAgICAgIC8vIGlmIGEgZGVmYXVsdCBhbGxvY2F0b3IgaXMgZ2l2ZW4sIHVzZSB0aGF0Ci0gICAgICAgICAgICBkZWFsZXIgPSBkZWZhdWx0QWxsb2NhdG9yOwotICAgIH0KLSAgICAKLSAgICBpZiAoZGVhbGVyID09IE5VTEwpIHsKLSAgICAgICAgLy8gYWx3YXlzIHRyeSBoL3cgYWNjZWxlcmF0ZWQgbWVtb3J5IGZpcnN0Ci0gICAgICAgIGlmIChnbG9iYWxfcG1lbV9oZWFwKSB7Ci0gICAgICAgICAgICBjb25zdCBzcDxQTWVtSGVhcD4mIGhlYXAobVBNZW1IZWFwKTsKLSAgICAgICAgICAgIGlmIChkZWFsZXIgPT0gTlVMTCAmJiBoZWFwICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICBkZWFsZXIgPSBuZXcgTWVtb3J5RGVhbGVyKCAKLSAgICAgICAgICAgICAgICAgICAgICAgIGhlYXAtPmNyZWF0ZUNsaWVudEhlYXAoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGhlYXAtPmdldEFsbG9jYXRvcigpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIGlmIChkZWFsZXIgPT0gTlVMTCkgewotICAgICAgICAvLyByZXR1cm4gdGhlIGFzaG1lbSBhbGxvY2F0b3IgKHNvZnR3YXJlIHJlbmRlcmluZykKLSAgICAgICAgZGVhbGVyID0gbmV3IE1lbW9yeURlYWxlcihtQ2xpZW50SGVhcFNpemUsIDAsICJTRk5hdGl2ZUhlYXAiKTsKLSAgICB9Ci0gICAgcmV0dXJuIGRlYWxlcjsKLX0KLQotc3A8U2ltcGxlQmVzdEZpdEFsbG9jYXRvcj4gU3VyZmFjZUhlYXBNYW5hZ2VyOjpnZXRBbGxvY2F0b3IoaW50IHR5cGUpIGNvbnN0IAotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgc3A8U2ltcGxlQmVzdEZpdEFsbG9jYXRvcj4gYWxsb2NhdG9yOwotCi0gICAgLy8gdGhpcyBpcyBvbmx5IHVzZWQgZm9yIGRlYnVnZ2luZwotICAgIHN3aXRjaCAodHlwZSkgewotICAgICAgICBjYXNlIE5BVElWRV9NRU1PUllfVFlQRV9QTUVNOgotICAgICAgICAgICAgaWYgKG1QTWVtSGVhcCAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgYWxsb2NhdG9yID0gbVBNZW1IZWFwLT5nZXRBbGxvY2F0b3IoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGJyZWFrOwotICAgIH0KLSAgICByZXR1cm4gYWxsb2NhdG9yOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotUE1lbUhlYXA6OlBNZW1IZWFwKGNvbnN0IGNoYXIqIGNvbnN0IGRldmljZSwgc2l6ZV90IHNpemUsIHNpemVfdCByZXNlcnZlZCkKLSAgICA6IE1lbW9yeUhlYXBCYXNlKGRldmljZSwgc2l6ZSkKLXsKLSAgICAvL0xPR0QoIiVzLCAlcCwgbUZEPSVkIiwgX19QUkVUVFlfRlVOQ1RJT05fXywgdGhpcywgaGVhcElEKCkpOwotICAgIGlmIChiYXNlKCkgIT0gTUFQX0ZBSUxFRCkgewotICAgICAgICAvL0xPR0QoIiVzLCAldSBieXRlcyIsIGRldmljZSwgdmlydHVhbFNpemUoKSk7Ci0gICAgICAgIGlmIChyZXNlcnZlZCA9PSAwKQotICAgICAgICAgICAgcmVzZXJ2ZWQgPSB2aXJ0dWFsU2l6ZSgpOwotICAgICAgICBtQWxsb2NhdG9yID0gbmV3IFNpbXBsZUJlc3RGaXRBbGxvY2F0b3IocmVzZXJ2ZWQpOwotICAgIH0KLX0KLQotUE1lbUhlYXA6On5QTWVtSGVhcCgpIHsKLSAgICAvL0xPR0QoIiVzLCAlcCwgbUZEPSVkIiwgX19QUkVUVFlfRlVOQ1RJT05fXywgdGhpcywgaGVhcElEKCkpOwotfQotCi1zcDxNZW1vcnlIZWFwUG1lbT4gUE1lbUhlYXA6OmNyZWF0ZUNsaWVudEhlYXAoKSB7Ci0gICAgc3A8TWVtb3J5SGVhcEJhc2U+IHBhcmVudEhlYXAodGhpcyk7Ci0gICAgcmV0dXJuIG5ldyBNZW1vcnlIZWFwUG1lbShwYXJlbnRIZWFwKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9WUmFtSGVhcC5oIGIvbGlicy9zdXJmYWNlZmxpbmdlci9WUmFtSGVhcC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5MTQwMTY3Li4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvVlJhbUhlYXAuaAorKysgL2Rldi9udWxsCkBAIC0xLDc4ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfVlJBTV9IRUFQX0gKLSNkZWZpbmUgQU5EUk9JRF9WUkFNX0hFQVBfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5RGVhbGVyLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIFBNZW1IZWFwOwotY2xhc3MgTWVtb3J5SGVhcFBtZW07Ci1jbGFzcyBTdXJmYWNlRmxpbmdlcjsgCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBTdXJmYWNlSGVhcE1hbmFnZXIgIDogcHVibGljIFJlZkJhc2UKLXsKLXB1YmxpYzoKLSAgICBTdXJmYWNlSGVhcE1hbmFnZXIoY29uc3Qgc3A8U3VyZmFjZUZsaW5nZXI+JiBmbGluZ2VyLCBzaXplX3QgY2xpZW50SGVhcFNpemUpOwotICAgIHZpcnR1YWwgflN1cmZhY2VIZWFwTWFuYWdlcigpOwotICAgIHZpcnR1YWwgdm9pZCBvbkZpcnN0UmVmKCk7Ci0gICAgLyogdXNlIElTdXJmYWNlQ29tcG9zZXIgZmxhZ3MgZUdQVXxlSEFyZHdhcmV8ZVNlY3VyZSAqLwotICAgIHNwPE1lbW9yeURlYWxlcj4gY3JlYXRlSGVhcCh1aW50MzJfdCBmbGFncz0wLCBwaWRfdCBjbGllbnRfcGlkID0gMCwKLSAgICAgICAgICAgIGNvbnN0IHNwPE1lbW9yeURlYWxlcj4mIGRlZmF1bHRBbGxvY2F0b3IgPSAwKTsKLSAgICAKLSAgICAvLyB1c2VkIGZvciBkZWJ1Z2dpbmcgb25seS4uLgotICAgIHNwPFNpbXBsZUJlc3RGaXRBbGxvY2F0b3I+IGdldEFsbG9jYXRvcihpbnQgdHlwZSkgY29uc3Q7Ci0KLXByaXZhdGU6Ci0gICAgc3A8UE1lbUhlYXA+IGdldEhlYXAoaW50IHR5cGUpIGNvbnN0OwotCi0gICAgc3A8U3VyZmFjZUZsaW5nZXI+IG1GbGluZ2VyOwotICAgIG11dGFibGUgTXV0ZXggICBtTG9jazsKLSAgICBzaXplX3QgICAgICAgICAgbUNsaWVudEhlYXBTaXplOwotICAgIHNwPFBNZW1IZWFwPiAgICBtUE1lbUhlYXA7Ci0gICAgc3RhdGljIGludCBnbG9iYWxfcG1lbV9oZWFwOwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIFBNZW1IZWFwIDogcHVibGljIE1lbW9yeUhlYXBCYXNlCi17Ci1wdWJsaWM6Ci0gICAgICAgICAgICAgICAgUE1lbUhlYXAoY29uc3QgY2hhciogY29uc3QgdnJhbSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHNpemVfdCBzaXplPTAsIHNpemVfdCByZXNlcnZlZD0wKTsKLSAgICB2aXJ0dWFsICAgICB+UE1lbUhlYXAoKTsKLSAgICAKLSAgICB2aXJ0dWFsIGNvbnN0IHNwPFNpbXBsZUJlc3RGaXRBbGxvY2F0b3I+JiBnZXRBbGxvY2F0b3IoKSBjb25zdCB7Ci0gICAgICAgIHJldHVybiBtQWxsb2NhdG9yOyAKLSAgICB9Ci0gICAgdmlydHVhbCBzcDxNZW1vcnlIZWFwUG1lbT4gY3JlYXRlQ2xpZW50SGVhcCgpOwotICAgIAotcHJpdmF0ZToKLSAgICBzcDxTaW1wbGVCZXN0Rml0QWxsb2NhdG9yPiAgbUFsbG9jYXRvcjsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfVlJBTV9IRUFQX0gKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvY2x6LmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvY2x6LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjQ1NmI4Ni4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL2Nsei5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwzNyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2luY2x1ZGUgImNsei5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWludCBjbHpfaW1wbChpbnQzMl90IHgpCi17Ci0jaWYgZGVmaW5lZChfX2FybV9fKSAmJiAhZGVmaW5lZChfX3RodW1iX18pCi0gICAgcmV0dXJuIF9fYnVpbHRpbl9jbHooeCk7Ci0jZWxzZQotICAgIGlmICgheCkgcmV0dXJuIDMyOwotICAgIGludCBlID0gMzE7Ci0gICAgaWYgKHgmMHhGRkZGMDAwMCkgICB7IGUgLT0xNjsgeCA+Pj0xNjsgfQotICAgIGlmICh4JjB4MDAwMEZGMDApICAgeyBlIC09IDg7IHggPj49IDg7IH0KLSAgICBpZiAoeCYweDAwMDAwMEYwKSAgIHsgZSAtPSA0OyB4ID4+PSA0OyB9Ci0gICAgaWYgKHgmMHgwMDAwMDAwQykgICB7IGUgLT0gMjsgeCA+Pj0gMjsgfQotICAgIGlmICh4JjB4MDAwMDAwMDIpICAgeyBlIC09IDE7IH0KLSAgICByZXR1cm4gZTsKLSNlbmRpZgotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy9zdXJmYWNlZmxpbmdlci9jbHouaCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvY2x6LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBkZGY5ODYuLjAwMDAwMDAKLS0tIGEvbGlicy9zdXJmYWNlZmxpbmdlci9jbHouaAorKysgL2Rldi9udWxsCkBAIC0xLDM3ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfU1VSRkFDRV9GTElOR0VSX0NMWl9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1pbnQgY2x6X2ltcGwoaW50MzJfdCB4KTsKLQotaW50IGlubGluZSBjbHooaW50MzJfdCB4KQotewotI2lmIGRlZmluZWQoX19hcm1fXykgJiYgIWRlZmluZWQoX190aHVtYl9fKQotICAgIHJldHVybiBfX2J1aWx0aW5fY2x6KHgpOwotI2Vsc2UKLSAgICByZXR1cm4gY2x6X2ltcGwoeCk7Ci0jZW5kaWYKLX0KLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLyogQU5EUk9JRF9TVVJGQUNFX0ZMSU5HRVJfQ0xaX0ggKi8KZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvdGVzdHMvQW5kcm9pZC5tayBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvdGVzdHMvQW5kcm9pZC5tawpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNTA1M2U3ZC4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL3Rlc3RzL0FuZHJvaWQubWsKKysrIC9kZXYvbnVsbApAQCAtMSArMCwwIEBACi1pbmNsdWRlICQoY2FsbCBhbGwtc3ViZGlyLW1ha2VmaWxlcykKZGlmZiAtLWdpdCBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvdGVzdHMvb3ZlcmxheXMvQW5kcm9pZC5tayBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvdGVzdHMvb3ZlcmxheXMvQW5kcm9pZC5tawpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGM0N2U0NS4uMDAwMDAwMAotLS0gYS9saWJzL3N1cmZhY2VmbGluZ2VyL3Rlc3RzL292ZXJsYXlzL0FuZHJvaWQubWsKKysrIC9kZXYvbnVsbApAQCAtMSwxNiArMCwwIEBACi1MT0NBTF9QQVRIOj0gJChjYWxsIG15LWRpcikKLWluY2x1ZGUgJChDTEVBUl9WQVJTKQotCi1MT0NBTF9TUkNfRklMRVM6PSBcCi0Jb3ZlcmxheXMuY3BwCi0KLUxPQ0FMX1NIQVJFRF9MSUJSQVJJRVMgOj0gXAotCWxpYmN1dGlscyBcCi0JbGlidXRpbHMgXAotICAgIGxpYnVpCi0KLUxPQ0FMX01PRFVMRTo9IHRlc3Qtb3ZlcmxheXMKLQotTE9DQUxfTU9EVUxFX1RBR1MgOj0gdGVzdHMKLQotaW5jbHVkZSAkKEJVSUxEX0VYRUNVVEFCTEUpCmRpZmYgLS1naXQgYS9saWJzL3N1cmZhY2VmbGluZ2VyL3Rlc3RzL292ZXJsYXlzL292ZXJsYXlzLmNwcCBiL2xpYnMvc3VyZmFjZWZsaW5nZXIvdGVzdHMvb3ZlcmxheXMvb3ZlcmxheXMuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmM2MwNDZmLi4wMDAwMDAwCi0tLSBhL2xpYnMvc3VyZmFjZWZsaW5nZXIvdGVzdHMvb3ZlcmxheXMvb3ZlcmxheXMuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNTggKzAsMCBAQAotI2luY2x1ZGUgPHV0aWxzL0lQQ1RocmVhZFN0YXRlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUHJvY2Vzc1N0YXRlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSVNlcnZpY2VNYW5hZ2VyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlIDx1aS9TdXJmYWNlLmg+Ci0jaW5jbHVkZSA8dWkvSVN1cmZhY2UuaD4KLSNpbmNsdWRlIDx1aS9PdmVybGF5Lmg+Ci0jaW5jbHVkZSA8dWkvU3VyZmFjZUNvbXBvc2VyQ2xpZW50Lmg+Ci0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci1jbGFzcyBUZXN0IHsKLXB1YmxpYzoKLSAgICBzdGF0aWMgY29uc3Qgc3A8SVN1cmZhY2U+JiBnZXRJU3VyZmFjZShjb25zdCBzcDxTdXJmYWNlPiYgcykgewotICAgICAgICByZXR1cm4gcy0+Z2V0SVN1cmZhY2UoKTsKLSAgICB9Ci19OwotfTsKLQotaW50IG1haW4oaW50IGFyZ2MsIGNoYXIqKiBhcmd2KQotewotICAgIC8vIHNldCB1cCB0aGUgdGhyZWFkLXBvb2wKLSAgICBzcDxQcm9jZXNzU3RhdGU+IHByb2MoUHJvY2Vzc1N0YXRlOjpzZWxmKCkpOwotICAgIFByb2Nlc3NTdGF0ZTo6c2VsZigpLT5zdGFydFRocmVhZFBvb2woKTsKLQotICAgIC8vIGNyZWF0ZSBhIGNsaWVudCB0byBzdXJmYWNlZmxpbmdlcgotICAgIHNwPFN1cmZhY2VDb21wb3NlckNsaWVudD4gY2xpZW50ID0gbmV3IFN1cmZhY2VDb21wb3NlckNsaWVudCgpOwotICAgIAotICAgIC8vIGNyZWF0ZSBwdXNoYnVmZmVyIHN1cmZhY2UKLSAgICBzcDxTdXJmYWNlPiBzdXJmYWNlID0gY2xpZW50LT5jcmVhdGVTdXJmYWNlKGdldHBpZCgpLCAwLCAzMjAsIDI0MCwgCi0gICAgICAgICAgICBQSVhFTF9GT1JNQVRfVU5LTk9XTiwgSVN1cmZhY2VDb21wb3Nlcjo6ZVB1c2hCdWZmZXJzKTsKLQotICAgIC8vIGdldCB0byB0aGUgaXN1cmZhY2UKLSAgICBzcDxJU3VyZmFjZT4gaXN1cmZhY2UgPSBUZXN0OjpnZXRJU3VyZmFjZShzdXJmYWNlKTsKLSAgICBwcmludGYoImlzdXJmYWNlID0gJXBcbiIsIGlzdXJmYWNlLmdldCgpKTsKLSAgICAKLSAgICAvLyBub3cgcmVxdWVzdCBhbiBvdmVybGF5Ci0gICAgc3A8T3ZlcmxheVJlZj4gcmVmID0gaXN1cmZhY2UtPmNyZWF0ZU92ZXJsYXkoMzIwLCAyNDAsIFBJWEVMX0ZPUk1BVF9SR0JfNTY1KTsKLSAgICBzcDxPdmVybGF5PiBvdmVybGF5ID0gbmV3IE92ZXJsYXkocmVmKTsKLSAgICAKLQotICAgIC8qCi0gICAgICogaGVyZSB3ZSBjYW4gdXNlIHRoZSBvdmVybGF5IEFQSSAKLSAgICAgKi8KLSAgICAKLSAgICBvdmVybGF5X2J1ZmZlcl90IGJ1ZmZlcjsgCi0gICAgb3ZlcmxheS0+ZGVxdWV1ZUJ1ZmZlcigmYnVmZmVyKTsKLSAgICBwcmludGYoImJ1ZmZlciA9ICVwXG4iLCBidWZmZXIpOwotICAgIAotICAgIHZvaWQqIGFkZHJlc3MgPSBvdmVybGF5LT5nZXRCdWZmZXJBZGRyZXNzKGJ1ZmZlcik7Ci0gICAgcHJpbnRmKCJhZGRyZXNzID0gJXBcbiIsIGFkZHJlc3MpOwotCi0gICAgb3ZlcmxheS0+cXVldWVCdWZmZXIoYnVmZmVyKTsKLQotICAgIHJldHVybiAwOwotfQpkaWZmIC0tZ2l0IGEvbGlicy91aS9BbmRyb2lkLm1rIGIvbGlicy91aS9BbmRyb2lkLm1rCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmOTQ0MzU3Li4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvQW5kcm9pZC5taworKysgL2Rldi9udWxsCkBAIC0xLDQxICswLDAgQEAKLUxPQ0FMX1BBVEg6PSAkKGNhbGwgbXktZGlyKQotaW5jbHVkZSAkKENMRUFSX1ZBUlMpCi0KLUxPQ0FMX1NSQ19GSUxFUzo9IFwKLQlDYW1lcmEuY3BwIFwKLQlDYW1lcmFQYXJhbWV0ZXJzLmNwcCBcCi0JRUdMRGlzcGxheVN1cmZhY2UuY3BwIFwKLQlFR0xOYXRpdmVXaW5kb3dTdXJmYWNlLmNwcCBcCi0JRXZlbnRIdWIuY3BwIFwKLQlFdmVudFJlY3VycmVuY2UuY3BwIFwKLQlLZXlMYXlvdXRNYXAuY3BwIFwKLQlLZXlDaGFyYWN0ZXJNYXAuY3BwIFwKLQlJQ2FtZXJhLmNwcCBcCi0JSUNhbWVyYUNsaWVudC5jcHAgXAotCUlDYW1lcmFTZXJ2aWNlLmNwcCBcCi0JSU92ZXJsYXkuY3BwIFwKLQlJU3VyZmFjZUNvbXBvc2VyLmNwcCBcCi0JSVN1cmZhY2UuY3BwIFwKLQlJU3VyZmFjZUZsaW5nZXJDbGllbnQuY3BwIFwKLQlMYXllclN0YXRlLmNwcCBcCi0JT3ZlcmxheS5jcHAgXAotCVBpeGVsRm9ybWF0LmNwcCBcCi0JUG9pbnQuY3BwIFwKLQlSZWN0LmNwcCBcCi0JUmVnaW9uLmNwcCBcCi0JU3VyZmFjZS5jcHAgXAotCVN1cmZhY2VDb21wb3NlckNsaWVudC5jcHAgXAotCVN1cmZhY2VGbGluZ2VyU3luY2hyby5jcHAgXAotCVRpbWUuY3BwCi0KLUxPQ0FMX1NIQVJFRF9MSUJSQVJJRVMgOj0gXAotCWxpYmNvcmVjZyBcCi0JbGliY3V0aWxzIFwKLQlsaWJ1dGlscyBcCi0JbGlicGl4ZWxmbGluZ2VyIFwKLQlsaWJoYXJkd2FyZSBcCi0JbGliaGFyZHdhcmVfbGVnYWN5Ci0KLUxPQ0FMX01PRFVMRTo9IGxpYnVpCi0KLWluY2x1ZGUgJChCVUlMRF9TSEFSRURfTElCUkFSWSkKZGlmZiAtLWdpdCBhL2xpYnMvdWkvQ2FtZXJhLmNwcCBiL2xpYnMvdWkvQ2FtZXJhLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNmM2MGI4NS4uMDAwMDAwMAotLS0gYS9saWJzL3VpL0NhbWVyYS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw0MDYgKzAsMCBAQAotLyoKLSoqCi0qKiBDb3B5cmlnaHQgKEMpIDIwMDgsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqIENvcHlyaWdodCAoQykgMjAwOCBIVEMgSW5jLgotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotLy8jZGVmaW5lIExPR19OREVCVUcgMAotI2RlZmluZSBMT0dfVEFHICJDYW1lcmEiCi0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSVNlcnZpY2VNYW5hZ2VyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotI2luY2x1ZGUgPHV0aWxzL0lNZW1vcnkuaD4KLSNpbmNsdWRlIDx1aS9TdXJmYWNlLmg+Ci0jaW5jbHVkZSA8dWkvQ2FtZXJhLmg+Ci0jaW5jbHVkZSA8dWkvSUNhbWVyYVNlcnZpY2UuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyBjbGllbnQgc2luZ2xldG9uIGZvciBjYW1lcmEgc2VydmljZSBiaW5kZXIgaW50ZXJmYWNlCi1NdXRleCBDYW1lcmE6Om1Mb2NrOwotc3A8SUNhbWVyYVNlcnZpY2U+IENhbWVyYTo6bUNhbWVyYVNlcnZpY2U7Ci1zcDxDYW1lcmE6OkRlYXRoTm90aWZpZXI+IENhbWVyYTo6bURlYXRoTm90aWZpZXI7Ci0KLS8vIGVzdGFibGlzaCBiaW5kZXIgaW50ZXJmYWNlIHRvIGNhbWVyYSBzZXJ2aWNlCi1jb25zdCBzcDxJQ2FtZXJhU2VydmljZT4mIENhbWVyYTo6Z2V0Q2FtZXJhU2VydmljZSgpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICBpZiAobUNhbWVyYVNlcnZpY2UuZ2V0KCkgPT0gMCkgewotICAgICAgICBzcDxJU2VydmljZU1hbmFnZXI+IHNtID0gZGVmYXVsdFNlcnZpY2VNYW5hZ2VyKCk7Ci0gICAgICAgIHNwPElCaW5kZXI+IGJpbmRlcjsKLSAgICAgICAgZG8gewotICAgICAgICAgICAgYmluZGVyID0gc20tPmdldFNlcnZpY2UoU3RyaW5nMTYoIm1lZGlhLmNhbWVyYSIpKTsKLSAgICAgICAgICAgIGlmIChiaW5kZXIgIT0gMCkKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIExPR1coIkNhbWVyYVNlcnZpY2Ugbm90IHB1Ymxpc2hlZCwgd2FpdGluZy4uLiIpOwotICAgICAgICAgICAgdXNsZWVwKDUwMDAwMCk7IC8vIDAuNSBzCi0gICAgICAgIH0gd2hpbGUodHJ1ZSk7Ci0gICAgICAgIGlmIChtRGVhdGhOb3RpZmllciA9PSBOVUxMKSB7Ci0gICAgICAgICAgICBtRGVhdGhOb3RpZmllciA9IG5ldyBEZWF0aE5vdGlmaWVyKCk7Ci0gICAgICAgIH0KLSAgICAgICAgYmluZGVyLT5saW5rVG9EZWF0aChtRGVhdGhOb3RpZmllcik7Ci0gICAgICAgIG1DYW1lcmFTZXJ2aWNlID0gaW50ZXJmYWNlX2Nhc3Q8SUNhbWVyYVNlcnZpY2U+KGJpbmRlcik7Ci0gICAgfQotICAgIExPR0VfSUYobUNhbWVyYVNlcnZpY2U9PTAsICJubyBDYW1lcmFTZXJ2aWNlIT8iKTsKLSAgICByZXR1cm4gbUNhbWVyYVNlcnZpY2U7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1DYW1lcmE6OkNhbWVyYSgpCi17Ci0gICAgaW5pdCgpOwotfQotCi1DYW1lcmE6OkNhbWVyYShjb25zdCBzcDxJQ2FtZXJhPiYgY2FtZXJhKQotewotICAgIGluaXQoKTsKLSAgICAvLyBjb25uZWN0IHRoaXMgY2xpZW50IHRvIGV4aXN0aW5nIGNhbWVyYSByZW1vdGUKLSAgICBpZiAoY2FtZXJhLT5jb25uZWN0KHRoaXMpID09IE5PX0VSUk9SKSB7Ci0gICAgICAgIG1TdGF0dXMgPSBOT19FUlJPUjsKLSAgICAgICAgbUNhbWVyYSA9IGNhbWVyYTsKLSAgICAgICAgY2FtZXJhLT5hc0JpbmRlcigpLT5saW5rVG9EZWF0aCh0aGlzKTsKLSAgICB9Ci19Ci0KLXZvaWQgQ2FtZXJhOjppbml0KCkKLXsKLSAgICBtU3RhdHVzID0gVU5LTk9XTl9FUlJPUjsKLSAgICBtU2h1dHRlckNhbGxiYWNrID0gMDsKLSAgICBtU2h1dHRlckNhbGxiYWNrQ29va2llID0gMDsKLSAgICBtUmF3Q2FsbGJhY2sgPSAwOwotICAgIG1SYXdDYWxsYmFja0Nvb2tpZSA9IDA7Ci0gICAgbUpwZWdDYWxsYmFjayA9IDA7Ci0gICAgbUpwZWdDYWxsYmFja0Nvb2tpZSA9IDA7Ci0gICAgbVByZXZpZXdDYWxsYmFjayA9IDA7Ci0gICAgbVByZXZpZXdDYWxsYmFja0Nvb2tpZSA9IDA7Ci0gICAgbVJlY29yZGluZ0NhbGxiYWNrID0gMDsKLSAgICBtUmVjb3JkaW5nQ2FsbGJhY2tDb29raWUgPSAwOwotICAgIG1FcnJvckNhbGxiYWNrID0gMDsKLSAgICBtRXJyb3JDYWxsYmFja0Nvb2tpZSA9IDA7Ci0gICAgbUF1dG9Gb2N1c0NhbGxiYWNrID0gMDsKLSAgICBtQXV0b0ZvY3VzQ2FsbGJhY2tDb29raWUgPSAwOwotfQotCi1DYW1lcmE6On5DYW1lcmEoKQotewotICAgIGRpc2Nvbm5lY3QoKTsKLX0KLQotc3A8Q2FtZXJhPiBDYW1lcmE6OmNvbm5lY3QoKQotewotICAgIExPR1YoImNvbm5lY3QiKTsKLSAgICBzcDxDYW1lcmE+IGMgPSBuZXcgQ2FtZXJhKCk7Ci0gICAgY29uc3Qgc3A8SUNhbWVyYVNlcnZpY2U+JiBjcyA9IGdldENhbWVyYVNlcnZpY2UoKTsKLSAgICBpZiAoY3MgIT0gMCkgewotICAgICAgICBjLT5tQ2FtZXJhID0gY3MtPmNvbm5lY3QoYyk7Ci0gICAgfQotICAgIGlmIChjLT5tQ2FtZXJhICE9IDApIHsKLSAgICAgICAgYy0+bUNhbWVyYS0+YXNCaW5kZXIoKS0+bGlua1RvRGVhdGgoYyk7Ci0gICAgICAgIGMtPm1TdGF0dXMgPSBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIGM7Ci19Ci0KLXZvaWQgQ2FtZXJhOjpkaXNjb25uZWN0KCkKLXsKLSAgICBMT0dWKCJkaXNjb25uZWN0Iik7Ci0gICAgaWYgKG1DYW1lcmEgIT0gMCkgewotICAgICAgICBtRXJyb3JDYWxsYmFjayA9IDA7Ci0gICAgICAgIG1DYW1lcmEtPmRpc2Nvbm5lY3QoKTsKLSAgICAgICAgbUNhbWVyYSA9IDA7Ci0gICAgfQotfQotCi1zdGF0dXNfdCBDYW1lcmE6OnJlY29ubmVjdCgpCi17Ci0gICAgTE9HVigicmVjb25uZWN0Iik7Ci0gICAgc3AgPElDYW1lcmE+IGMgPSBtQ2FtZXJhOwotICAgIGlmIChjID09IDApIHJldHVybiBOT19JTklUOwotICAgIHJldHVybiBjLT5jb25uZWN0KHRoaXMpOwotfQotCi1zcDxJQ2FtZXJhPiBDYW1lcmE6OnJlbW90ZSgpCi17Ci0gICAgcmV0dXJuIG1DYW1lcmE7Ci19Ci0KLXN0YXR1c190IENhbWVyYTo6bG9jaygpCi17Ci0gICAgc3AgPElDYW1lcmE+IGMgPSBtQ2FtZXJhOwotICAgIGlmIChjID09IDApIHJldHVybiBOT19JTklUOwotICAgIHJldHVybiBjLT5sb2NrKCk7Ci19Ci0KLXN0YXR1c190IENhbWVyYTo6dW5sb2NrKCkKLXsKLSAgICBzcCA8SUNhbWVyYT4gYyA9IG1DYW1lcmE7Ci0gICAgaWYgKGMgPT0gMCkgcmV0dXJuIE5PX0lOSVQ7Ci0gICAgcmV0dXJuIGMtPnVubG9jaygpOwotfQotCi0vLyBwYXNzIHRoZSBidWZmZXJlZCBJU3VyZmFjZSB0byB0aGUgY2FtZXJhIHNlcnZpY2UKLXN0YXR1c190IENhbWVyYTo6c2V0UHJldmlld0Rpc3BsYXkoY29uc3Qgc3A8U3VyZmFjZT4mIHN1cmZhY2UpCi17Ci0gICAgTE9HVigic2V0UHJldmlld0Rpc3BsYXkiKTsKLSAgICBpZiAoc3VyZmFjZSA9PSAwKSB7Ci0gICAgICAgIExPR0UoImFwcCBwYXNzZWQgTlVMTCBzdXJmYWNlIik7Ci0gICAgICAgIHJldHVybiBOT19JTklUOwotICAgIH0KLSAgICBzcCA8SUNhbWVyYT4gYyA9IG1DYW1lcmE7Ci0gICAgaWYgKGMgPT0gMCkgcmV0dXJuIE5PX0lOSVQ7Ci0gICAgcmV0dXJuIGMtPnNldFByZXZpZXdEaXNwbGF5KHN1cmZhY2UtPmdldElTdXJmYWNlKCkpOwotfQotCi1zdGF0dXNfdCBDYW1lcmE6OnNldFByZXZpZXdEaXNwbGF5KGNvbnN0IHNwPElTdXJmYWNlPiYgc3VyZmFjZSkKLXsKLSAgICBMT0dWKCJzZXRQcmV2aWV3RGlzcGxheSIpOwotICAgIGlmIChzdXJmYWNlID09IDApIHsKLSAgICAgICAgTE9HRSgiYXBwIHBhc3NlZCBOVUxMIHN1cmZhY2UiKTsKLSAgICAgICAgcmV0dXJuIE5PX0lOSVQ7Ci0gICAgfQotICAgIHNwIDxJQ2FtZXJhPiBjID0gbUNhbWVyYTsKLSAgICBpZiAoYyA9PSAwKSByZXR1cm4gTk9fSU5JVDsKLSAgICByZXR1cm4gYy0+c2V0UHJldmlld0Rpc3BsYXkoc3VyZmFjZSk7Ci19Ci0KLQotLy8gc3RhcnQgcHJldmlldyBtb2RlLCBtdXN0IGNhbGwgc2V0UHJldmlld0Rpc3BsYXkgZmlyc3QKLXN0YXR1c190IENhbWVyYTo6c3RhcnRQcmV2aWV3KCkKLXsKLSAgICBMT0dWKCJzdGFydFByZXZpZXciKTsKLSAgICBzcCA8SUNhbWVyYT4gYyA9IG1DYW1lcmE7Ci0gICAgaWYgKGMgPT0gMCkgcmV0dXJuIE5PX0lOSVQ7Ci0gICAgcmV0dXJuIGMtPnN0YXJ0UHJldmlldygpOwotfQotCi0vLyBzdGFydCByZWNvcmRpbmcgbW9kZSwgbXVzdCBjYWxsIHNldFByZXZpZXdEaXNwbGF5IGZpcnN0Ci1zdGF0dXNfdCBDYW1lcmE6OnN0YXJ0UmVjb3JkaW5nKCkKLXsKLSAgICBMT0dWKCJzdGFydFJlY29yZGluZyIpOwotICAgIHNwIDxJQ2FtZXJhPiBjID0gbUNhbWVyYTsKLSAgICBpZiAoYyA9PSAwKSByZXR1cm4gTk9fSU5JVDsKLSAgICByZXR1cm4gYy0+c3RhcnRSZWNvcmRpbmcoKTsKLX0KLQotLy8gc3RvcCBwcmV2aWV3IG1vZGUKLXZvaWQgQ2FtZXJhOjpzdG9wUHJldmlldygpCi17Ci0gICAgTE9HVigic3RvcFByZXZpZXciKTsKLSAgICBzcCA8SUNhbWVyYT4gYyA9IG1DYW1lcmE7Ci0gICAgaWYgKGMgPT0gMCkgcmV0dXJuOwotICAgIGMtPnN0b3BQcmV2aWV3KCk7Ci19Ci0KLS8vIHN0b3AgcmVjb3JkaW5nIG1vZGUKLXZvaWQgQ2FtZXJhOjpzdG9wUmVjb3JkaW5nKCkKLXsKLSAgICBMT0dWKCJzdG9wUmVjb3JkaW5nIik7Ci0gICAgc3AgPElDYW1lcmE+IGMgPSBtQ2FtZXJhOwotICAgIGlmIChjID09IDApIHJldHVybjsKLSAgICBjLT5zdG9wUmVjb3JkaW5nKCk7Ci19Ci0KLS8vIHJlbGVhc2UgYSByZWNvcmRpbmcgZnJhbWUKLXZvaWQgQ2FtZXJhOjpyZWxlYXNlUmVjb3JkaW5nRnJhbWUoY29uc3Qgc3A8SU1lbW9yeT4mIG1lbSkKLXsKLSAgICBMT0dWKCJyZWxlYXNlUmVjb3JkaW5nRnJhbWUiKTsKLSAgICBzcCA8SUNhbWVyYT4gYyA9IG1DYW1lcmE7Ci0gICAgaWYgKGMgPT0gMCkgcmV0dXJuOwotICAgIGMtPnJlbGVhc2VSZWNvcmRpbmdGcmFtZShtZW0pOwotfQotCi0vLyBnZXQgcHJldmlldyBzdGF0ZQotYm9vbCBDYW1lcmE6OnByZXZpZXdFbmFibGVkKCkKLXsKLSAgICBMT0dWKCJwcmV2aWV3RW5hYmxlZCIpOwotICAgIHNwIDxJQ2FtZXJhPiBjID0gbUNhbWVyYTsKLSAgICBpZiAoYyA9PSAwKSByZXR1cm4gZmFsc2U7Ci0gICAgcmV0dXJuIGMtPnByZXZpZXdFbmFibGVkKCk7Ci19Ci0KLS8vIGdldCByZWNvcmRpbmcgc3RhdGUKLWJvb2wgQ2FtZXJhOjpyZWNvcmRpbmdFbmFibGVkKCkKLXsKLSAgICBMT0dWKCJyZWNvcmRpbmdFbmFibGVkIik7Ci0gICAgc3AgPElDYW1lcmE+IGMgPSBtQ2FtZXJhOwotICAgIGlmIChjID09IDApIHJldHVybiBmYWxzZTsKLSAgICByZXR1cm4gYy0+cmVjb3JkaW5nRW5hYmxlZCgpOwotfQotCi1zdGF0dXNfdCBDYW1lcmE6OmF1dG9Gb2N1cygpCi17Ci0gICAgTE9HVigiYXV0b0ZvY3VzIik7Ci0gICAgc3AgPElDYW1lcmE+IGMgPSBtQ2FtZXJhOwotICAgIGlmIChjID09IDApIHJldHVybiBOT19JTklUOwotICAgIHJldHVybiBjLT5hdXRvRm9jdXMoKTsKLX0KLQotLy8gdGFrZSBhIHBpY3R1cmUKLXN0YXR1c190IENhbWVyYTo6dGFrZVBpY3R1cmUoKQotewotICAgIExPR1YoInRha2VQaWN0dXJlIik7Ci0gICAgc3AgPElDYW1lcmE+IGMgPSBtQ2FtZXJhOwotICAgIGlmIChjID09IDApIHJldHVybiBOT19JTklUOwotICAgIHJldHVybiBjLT50YWtlUGljdHVyZSgpOwotfQotCi0vLyBzZXQgcHJldmlldy9jYXB0dXJlIHBhcmFtZXRlcnMgLSBrZXkvdmFsdWUgcGFpcnMKLXN0YXR1c190IENhbWVyYTo6c2V0UGFyYW1ldGVycyhjb25zdCBTdHJpbmc4JiBwYXJhbXMpCi17Ci0gICAgTE9HVigic2V0UGFyYW1ldGVycyIpOwotICAgIHNwIDxJQ2FtZXJhPiBjID0gbUNhbWVyYTsKLSAgICBpZiAoYyA9PSAwKSByZXR1cm4gTk9fSU5JVDsKLSAgICByZXR1cm4gYy0+c2V0UGFyYW1ldGVycyhwYXJhbXMpOwotfQotCi0vLyBnZXQgcHJldmlldy9jYXB0dXJlIHBhcmFtZXRlcnMgLSBrZXkvdmFsdWUgcGFpcnMKLVN0cmluZzggQ2FtZXJhOjpnZXRQYXJhbWV0ZXJzKCkgY29uc3QKLXsKLSAgICBMT0dWKCJnZXRQYXJhbWV0ZXJzIik7Ci0gICAgU3RyaW5nOCBwYXJhbXM7Ci0gICAgc3AgPElDYW1lcmE+IGMgPSBtQ2FtZXJhOwotICAgIGlmIChjICE9IDApIHBhcmFtcyA9IG1DYW1lcmEtPmdldFBhcmFtZXRlcnMoKTsKLSAgICByZXR1cm4gcGFyYW1zOwotfQotCi12b2lkIENhbWVyYTo6c2V0QXV0b0ZvY3VzQ2FsbGJhY2soYXV0b2ZvY3VzX2NhbGxiYWNrIGNiLCB2b2lkICpjb29raWUpCi17Ci0gICAgTE9HVigic2V0QXV0b0ZvY3VzQ2FsbGJhY2siKTsKLSAgICBtQXV0b0ZvY3VzQ2FsbGJhY2sgPSBjYjsKLSAgICBtQXV0b0ZvY3VzQ2FsbGJhY2tDb29raWUgPSBjb29raWU7Ci19Ci0KLXZvaWQgQ2FtZXJhOjpzZXRTaHV0dGVyQ2FsbGJhY2soc2h1dHRlcl9jYWxsYmFjayBjYiwgdm9pZCAqY29va2llKQotewotICAgIExPR1YoInNldFNodXR0ZXJDYWxsYmFjayIpOwotICAgIG1TaHV0dGVyQ2FsbGJhY2sgPSBjYjsKLSAgICBtU2h1dHRlckNhbGxiYWNrQ29va2llID0gY29va2llOwotfQotCi12b2lkIENhbWVyYTo6c2V0UmF3Q2FsbGJhY2soZnJhbWVfY2FsbGJhY2sgY2IsIHZvaWQgKmNvb2tpZSkKLXsKLSAgICBMT0dWKCJzZXRSYXdDYWxsYmFjayIpOwotICAgIG1SYXdDYWxsYmFjayA9IGNiOwotICAgIG1SYXdDYWxsYmFja0Nvb2tpZSA9IGNvb2tpZTsKLX0KLQotdm9pZCBDYW1lcmE6OnNldEpwZWdDYWxsYmFjayhmcmFtZV9jYWxsYmFjayBjYiwgdm9pZCAqY29va2llKQotewotICAgIExPR1YoInNldEpwZWdDYWxsYmFjayIpOwotICAgIG1KcGVnQ2FsbGJhY2sgPSBjYjsKLSAgICBtSnBlZ0NhbGxiYWNrQ29va2llID0gY29va2llOwotfQotCi12b2lkIENhbWVyYTo6c2V0UHJldmlld0NhbGxiYWNrKGZyYW1lX2NhbGxiYWNrIGNiLCB2b2lkICpjb29raWUsIGludCBmbGFnKQotewotICAgIExPR1YoInNldFByZXZpZXdDYWxsYmFjayIpOwotICAgIG1QcmV2aWV3Q2FsbGJhY2sgPSBjYjsKLSAgICBtUHJldmlld0NhbGxiYWNrQ29va2llID0gY29va2llOwotICAgIHNwIDxJQ2FtZXJhPiBjID0gbUNhbWVyYTsKLSAgICBpZiAoYyA9PSAwKSByZXR1cm47Ci0gICAgbUNhbWVyYS0+c2V0UHJldmlld0NhbGxiYWNrRmxhZyhmbGFnKTsKLX0KLQotdm9pZCBDYW1lcmE6OnNldFJlY29yZGluZ0NhbGxiYWNrKGZyYW1lX2NhbGxiYWNrIGNiLCB2b2lkICpjb29raWUpCi17Ci0gICAgTE9HVigic2V0UmVjb3JkaW5nQ2FsbGJhY2siKTsKLSAgICBtUmVjb3JkaW5nQ2FsbGJhY2sgPSBjYjsKLSAgICBtUmVjb3JkaW5nQ2FsbGJhY2tDb29raWUgPSBjb29raWU7Ci19Ci0KLXZvaWQgQ2FtZXJhOjpzZXRFcnJvckNhbGxiYWNrKGVycm9yX2NhbGxiYWNrIGNiLCB2b2lkICpjb29raWUpCi17Ci0gICAgTE9HVigic2V0RXJyb3JDYWxsYmFjayIpOwotICAgIG1FcnJvckNhbGxiYWNrID0gY2I7Ci0gICAgbUVycm9yQ2FsbGJhY2tDb29raWUgPSBjb29raWU7Ci19Ci0KLXZvaWQgQ2FtZXJhOjphdXRvRm9jdXNDYWxsYmFjayhib29sIGZvY3VzZWQpCi17Ci0gICAgTE9HVigiYXV0b0ZvY3VzQ2FsbGJhY2siKTsKLSAgICBpZiAobUF1dG9Gb2N1c0NhbGxiYWNrKSB7Ci0gICAgICAgIG1BdXRvRm9jdXNDYWxsYmFjayhmb2N1c2VkLCBtQXV0b0ZvY3VzQ2FsbGJhY2tDb29raWUpOwotICAgIH0KLX0KLQotdm9pZCBDYW1lcmE6OnNodXR0ZXJDYWxsYmFjaygpCi17Ci0gICAgTE9HVigic2h1dHRlckNhbGxiYWNrIik7Ci0gICAgaWYgKG1TaHV0dGVyQ2FsbGJhY2spIHsKLSAgICAgICAgbVNodXR0ZXJDYWxsYmFjayhtU2h1dHRlckNhbGxiYWNrQ29va2llKTsKLSAgICB9Ci19Ci0KLXZvaWQgQ2FtZXJhOjpyYXdDYWxsYmFjayhjb25zdCBzcDxJTWVtb3J5PiYgcGljdHVyZSkKLXsKLSAgICBMT0dWKCJyYXdDYWxsYmFjayIpOwotICAgIGlmIChtUmF3Q2FsbGJhY2spIHsKLSAgICAgICAgbVJhd0NhbGxiYWNrKHBpY3R1cmUsIG1SYXdDYWxsYmFja0Nvb2tpZSk7Ci0gICAgfQotfQotCi0vLyBjYWxsYmFjayBmcm9tIGNhbWVyYSBzZXJ2aWNlIHdoZW4gaW1hZ2UgaXMgcmVhZHkKLXZvaWQgQ2FtZXJhOjpqcGVnQ2FsbGJhY2soY29uc3Qgc3A8SU1lbW9yeT4mIHBpY3R1cmUpCi17Ci0gICAgTE9HVigianBlZ0NhbGxiYWNrIik7Ci0gICAgaWYgKG1KcGVnQ2FsbGJhY2spIHsKLSAgICAgICAgbUpwZWdDYWxsYmFjayhwaWN0dXJlLCBtSnBlZ0NhbGxiYWNrQ29va2llKTsKLSAgICB9Ci19Ci0KLS8vIGNhbGxiYWNrIGZyb20gY2FtZXJhIHNlcnZpY2Ugd2hlbiBwcmV2aWV3IGZyYW1lIGlzIHJlYWR5Ci12b2lkIENhbWVyYTo6cHJldmlld0NhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBmcmFtZSkKLXsKLSAgICBMT0dWKCJmcmFtZUNhbGxiYWNrIik7Ci0gICAgaWYgKG1QcmV2aWV3Q2FsbGJhY2spIHsKLSAgICAgICAgbVByZXZpZXdDYWxsYmFjayhmcmFtZSwgbVByZXZpZXdDYWxsYmFja0Nvb2tpZSk7Ci0gICAgfQotfQotCi0vLyBjYWxsYmFjayBmcm9tIGNhbWVyYSBzZXJ2aWNlIHdoZW4gYSByZWNvcmRpbmcgZnJhbWUgaXMgcmVhZHkKLXZvaWQgQ2FtZXJhOjpyZWNvcmRpbmdDYWxsYmFjayhjb25zdCBzcDxJTWVtb3J5PiYgZnJhbWUpCi17Ci0gICAgTE9HVigicmVjb3JkaW5nQ2FsbGJhY2siKTsKLSAgICBpZiAobVJlY29yZGluZ0NhbGxiYWNrKSB7Ci0gICAgICAgIG1SZWNvcmRpbmdDYWxsYmFjayhmcmFtZSwgbVJlY29yZGluZ0NhbGxiYWNrQ29va2llKTsKLSAgICB9Ci19Ci0KLS8vIGNhbGxiYWNrIGZyb20gY2FtZXJhIHNlcnZpY2Ugd2hlbiBhbiBlcnJvciBvY2N1cnMgaW4gcHJldmlldyBvciB0YWtlUGljdHVyZQotdm9pZCBDYW1lcmE6OmVycm9yQ2FsbGJhY2soc3RhdHVzX3QgZXJyb3IpCi17Ci0gICAgTE9HVigiZXJyb3JDYWxsYmFjayIpOwotICAgIGlmIChtRXJyb3JDYWxsYmFjaykgewotICAgICAgICBtRXJyb3JDYWxsYmFjayhlcnJvciwgbUVycm9yQ2FsbGJhY2tDb29raWUpOwotICAgIH0KLX0KLQotdm9pZCBDYW1lcmE6OmJpbmRlckRpZWQoY29uc3Qgd3A8SUJpbmRlcj4mIHdobykgewotICAgIExPR1coIklDYW1lcmEgZGllZCIpOwotICAgIGlmIChtRXJyb3JDYWxsYmFjaykgewotICAgICAgICBtRXJyb3JDYWxsYmFjayhERUFEX09CSkVDVCwgbUVycm9yQ2FsbGJhY2tDb29raWUpOwotICAgIH0KLX0KLQotdm9pZCBDYW1lcmE6OkRlYXRoTm90aWZpZXI6OmJpbmRlckRpZWQoY29uc3Qgd3A8SUJpbmRlcj4mIHdobykgewotICAgIExPR1YoImJpbmRlckRpZWQiKTsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2woQ2FtZXJhOjptTG9jayk7Ci0gICAgQ2FtZXJhOjptQ2FtZXJhU2VydmljZS5jbGVhcigpOwotICAgIExPR1coIkNhbWVyYSBzZXJ2ZXIgZGllZCEiKTsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy91aS9DYW1lcmFQYXJhbWV0ZXJzLmNwcCBiL2xpYnMvdWkvQ2FtZXJhUGFyYW1ldGVycy5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDZjMjU4MzYuLjAwMDAwMDAKLS0tIGEvbGlicy91aS9DYW1lcmFQYXJhbWV0ZXJzLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDI3MyArMCwwIEBACi0vKgotKioKLSoqIENvcHlyaWdodCAyMDA4LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNkZWZpbmUgTE9HX1RBRyAiQ2FtZXJhUGFyYW1zIgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8dWkvQ2FtZXJhUGFyYW1ldGVycy5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLXN0YXRpYyBjb25zdCBjaGFyKiBwb3J0cmFpdCA9ICJwb3J0cmFpdCI7Ci1zdGF0aWMgY29uc3QgY2hhciogbGFuZHNjYXBlID0gImxhbmRzY2FwZSI7Ci0KLUNhbWVyYVBhcmFtZXRlcnM6OkNhbWVyYVBhcmFtZXRlcnMoKQotICAgICAgICAgICAgICAgIDogbU1hcCgpCi17Ci19Ci0KLUNhbWVyYVBhcmFtZXRlcnM6On5DYW1lcmFQYXJhbWV0ZXJzKCkKLXsKLX0KLQotU3RyaW5nOCBDYW1lcmFQYXJhbWV0ZXJzOjpmbGF0dGVuKCkgY29uc3QKLXsKLSAgICBTdHJpbmc4IGZsYXR0ZW5lZCgiIik7Ci0gICAgc2l6ZV90IHNpemUgPSBtTWFwLnNpemUoKTsKLQotICAgIGZvciAoc2l6ZV90IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKSB7Ci0gICAgICAgIFN0cmluZzggaywgdjsKLSAgICAgICAgayA9IG1NYXAua2V5QXQoaSk7Ci0gICAgICAgIHYgPSBtTWFwLnZhbHVlQXQoaSk7Ci0KLSAgICAgICAgZmxhdHRlbmVkICs9IGs7Ci0gICAgICAgIGZsYXR0ZW5lZCArPSAiPSI7Ci0gICAgICAgIGZsYXR0ZW5lZCArPSB2OwotICAgICAgICBpZiAoaSAhPSBzaXplLTEpCi0gICAgICAgICAgICBmbGF0dGVuZWQgKz0gIjsiOwotICAgIH0KLQotICAgIHJldHVybiBmbGF0dGVuZWQ7Ci19Ci0KLXZvaWQgQ2FtZXJhUGFyYW1ldGVyczo6dW5mbGF0dGVuKGNvbnN0IFN0cmluZzggJnBhcmFtcykKLXsKLSAgICBjb25zdCBjaGFyICphID0gcGFyYW1zLnN0cmluZygpOwotICAgIGNvbnN0IGNoYXIgKmI7Ci0KLSAgICBtTWFwLmNsZWFyKCk7Ci0KLSAgICBmb3IgKDs7KSB7Ci0gICAgICAgIC8vIEZpbmQgdGhlIGJvdW5kcyBvZiB0aGUga2V5IG5hbWUuCi0gICAgICAgIGIgPSBzdHJjaHIoYSwgJz0nKTsKLSAgICAgICAgaWYgKGIgPT0gMCkKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIC8vIENyZWF0ZSB0aGUga2V5IHN0cmluZy4KLSAgICAgICAgU3RyaW5nOCBrKGEsIChzaXplX3QpKGItYSkpOwotCi0gICAgICAgIC8vIEZpbmQgdGhlIHZhbHVlLgotICAgICAgICBhID0gYisxOwotICAgICAgICBiID0gc3RyY2hyKGEsICc7Jyk7Ci0gICAgICAgIGlmIChiID09IDApIHsKLSAgICAgICAgICAgIC8vIElmIHRoZXJlJ3Mgbm8gc2VtaWNvbG9uLCB0aGlzIGlzIHRoZSBsYXN0IGl0ZW0uCi0gICAgICAgICAgICBTdHJpbmc4IHYoYSk7Ci0gICAgICAgICAgICBtTWFwLmFkZChrLCB2KTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0KLSAgICAgICAgU3RyaW5nOCB2KGEsIChzaXplX3QpKGItYSkpOwotICAgICAgICBtTWFwLmFkZChrLCB2KTsKLSAgICAgICAgYSA9IGIrMTsKLSAgICB9Ci19Ci0KLQotdm9pZCBDYW1lcmFQYXJhbWV0ZXJzOjpzZXQoY29uc3QgY2hhciAqa2V5LCBjb25zdCBjaGFyICp2YWx1ZSkKLXsKLSAgICAvLyBYWFggaSB0aGluayBpIGNhbiBkbyB0aGlzIHdpdGggc3Ryc3BuKCkgCi0gICAgaWYgKHN0cmNocihrZXksICc9JykgfHwgc3RyY2hyKGtleSwgJzsnKSkgewotICAgICAgICAvL1hYWCBMT0dFKCJLZXkgXCIlc1wiY29udGFpbnMgaW52YWxpZCBjaGFyYWN0ZXIgKD0gb3IgOykiLCBrZXkpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgaWYgKHN0cmNocih2YWx1ZSwgJz0nKSB8fCBzdHJjaHIoa2V5LCAnOycpKSB7Ci0gICAgICAgIC8vWFhYIExPR0UoIlZhbHVlIFwiJXNcImNvbnRhaW5zIGludmFsaWQgY2hhcmFjdGVyICg9IG9yIDspIiwgdmFsdWUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgbU1hcC5yZXBsYWNlVmFsdWVGb3IoU3RyaW5nOChrZXkpLCBTdHJpbmc4KHZhbHVlKSk7Ci19Ci0KLXZvaWQgQ2FtZXJhUGFyYW1ldGVyczo6c2V0KGNvbnN0IGNoYXIgKmtleSwgaW50IHZhbHVlKQotewotICAgIGNoYXIgc3RyWzE2XTsKLSAgICBzcHJpbnRmKHN0ciwgIiVkIiwgdmFsdWUpOwotICAgIHNldChrZXksIHN0cik7Ci19Ci0KLWNvbnN0IGNoYXIgKkNhbWVyYVBhcmFtZXRlcnM6OmdldChjb25zdCBjaGFyICprZXkpIGNvbnN0Ci17Ci0gICAgU3RyaW5nOCB2ID0gbU1hcC52YWx1ZUZvcihTdHJpbmc4KGtleSkpOwotICAgIGlmICh2Lmxlbmd0aCgpID09IDApCi0gICAgICAgIHJldHVybiAwOwotICAgIHJldHVybiB2LnN0cmluZygpOwotfQotCi1pbnQgQ2FtZXJhUGFyYW1ldGVyczo6Z2V0SW50KGNvbnN0IGNoYXIgKmtleSkgY29uc3QKLXsKLSAgICBjb25zdCBjaGFyICp2ID0gZ2V0KGtleSk7Ci0gICAgaWYgKHYgPT0gMCkKLSAgICAgICAgcmV0dXJuIC0xOwotICAgIHJldHVybiBzdHJ0b2wodiwgMCwgMCk7Ci19Ci0KLXN0YXRpYyBpbnQgcGFyc2Vfc2l6ZShjb25zdCBjaGFyICpzdHIsIGludCAmd2lkdGgsIGludCAmaGVpZ2h0KQotewotICAgIC8vIEZpbmQgdGhlIHdpZHRoLgotICAgIGNoYXIgKmVuZDsKLSAgICBpbnQgdyA9IChpbnQpc3RydG9sKHN0ciwgJmVuZCwgMTApOwotICAgIC8vIElmIGFuICd4JyBkb2VzIG5vdCBpbW1lZGlhdGVseSBmb2xsb3csIGdpdmUgdXAuCi0gICAgaWYgKCplbmQgIT0gJ3gnKQotICAgICAgICByZXR1cm4gLTE7Ci0KLSAgICAvLyBGaW5kIHRoZSBoZWlnaHQsIGltbWVkaWF0ZWx5IGFmdGVyIHRoZSAneCcuCi0gICAgaW50IGggPSAoaW50KXN0cnRvbChlbmQrMSwgMCwgMTApOwotCi0gICAgd2lkdGggPSB3OwotICAgIGhlaWdodCA9IGg7Ci0KLSAgICByZXR1cm4gMDsKLX0KLQotdm9pZCBDYW1lcmFQYXJhbWV0ZXJzOjpzZXRQcmV2aWV3U2l6ZShpbnQgd2lkdGgsIGludCBoZWlnaHQpCi17Ci0gICAgY2hhciBzdHJbMzJdOwotICAgIHNwcmludGYoc3RyLCAiJWR4JWQiLCB3aWR0aCwgaGVpZ2h0KTsKLSAgICBzZXQoInByZXZpZXctc2l6ZSIsIHN0cik7Ci19Ci0KLXZvaWQgQ2FtZXJhUGFyYW1ldGVyczo6Z2V0UHJldmlld1NpemUoaW50ICp3aWR0aCwgaW50ICpoZWlnaHQpIGNvbnN0Ci17Ci0gICAgKndpZHRoID0gLTE7Ci0gICAgKmhlaWdodCA9IC0xOwotCi0gICAgLy8gR2V0IHRoZSBjdXJyZW50IHN0cmluZywgaWYgaXQgZG9lc24ndCBleGlzdCwgbGVhdmUgdGhlIC0xeC0xCi0gICAgY29uc3QgY2hhciAqcCA9IGdldCgicHJldmlldy1zaXplIik7Ci0gICAgaWYgKHAgPT0gMCkKLSAgICAgICAgcmV0dXJuOwotCi0gICAgaW50IHcsIGg7Ci0gICAgaWYgKHBhcnNlX3NpemUocCwgdywgaCkgPT0gMCkgewotICAgICAgICAqd2lkdGggPSB3OwotICAgICAgICAqaGVpZ2h0ID0gaDsKLSAgICB9Ci19Ci0KLXZvaWQgQ2FtZXJhUGFyYW1ldGVyczo6c2V0UHJldmlld0ZyYW1lUmF0ZShpbnQgZnBzKQotewotICAgIHNldCgicHJldmlldy1mcmFtZS1yYXRlIiwgZnBzKTsKLX0KLQotaW50IENhbWVyYVBhcmFtZXRlcnM6OmdldFByZXZpZXdGcmFtZVJhdGUoKSBjb25zdAotewotICAgIHJldHVybiBnZXRJbnQoInByZXZpZXctZnJhbWUtcmF0ZSIpOwotfQotCi12b2lkIENhbWVyYVBhcmFtZXRlcnM6OnNldFByZXZpZXdGb3JtYXQoY29uc3QgY2hhciAqZm9ybWF0KQotewotICAgIHNldCgicHJldmlldy1mb3JtYXQiLCBmb3JtYXQpOwotfQotCi1pbnQgQ2FtZXJhUGFyYW1ldGVyczo6Z2V0T3JpZW50YXRpb24oKSBjb25zdAotewotICAgIGNvbnN0IGNoYXIqIG9yaWVudGF0aW9uID0gZ2V0KCJvcmllbnRhdGlvbiIpOwotICAgIGlmIChvcmllbnRhdGlvbiAmJiAhc3RyY21wKG9yaWVudGF0aW9uLCBwb3J0cmFpdCkpCi0gICAgICAgIHJldHVybiBDQU1FUkFfT1JJRU5UQVRJT05fUE9SVFJBSVQ7Ci0gICAgcmV0dXJuIENBTUVSQV9PUklFTlRBVElPTl9MQU5EU0NBUEU7Ci19Ci0KLXZvaWQgQ2FtZXJhUGFyYW1ldGVyczo6c2V0T3JpZW50YXRpb24oaW50IG9yaWVudGF0aW9uKQotewotICAgIGlmIChvcmllbnRhdGlvbiA9PSBDQU1FUkFfT1JJRU5UQVRJT05fUE9SVFJBSVQpIHsKLSAgICAgICAgc2V0KCJwcmV2aWV3LWZvcm1hdCIsIHBvcnRyYWl0KTsKLSAgICB9IGVsc2UgewotICAgICAgICBzZXQoInByZXZpZXctZm9ybWF0IiwgbGFuZHNjYXBlKTsKLSAgICB9Ci19Ci0KLWNvbnN0IGNoYXIgKkNhbWVyYVBhcmFtZXRlcnM6OmdldFByZXZpZXdGb3JtYXQoKSBjb25zdAotewotICAgIHJldHVybiBnZXQoInByZXZpZXctZm9ybWF0Iik7Ci19Ci0KLXZvaWQgQ2FtZXJhUGFyYW1ldGVyczo6c2V0UGljdHVyZVNpemUoaW50IHdpZHRoLCBpbnQgaGVpZ2h0KQotewotICAgIGNoYXIgc3RyWzMyXTsKLSAgICBzcHJpbnRmKHN0ciwgIiVkeCVkIiwgd2lkdGgsIGhlaWdodCk7Ci0gICAgc2V0KCJwaWN0dXJlLXNpemUiLCBzdHIpOwotfQotCi12b2lkIENhbWVyYVBhcmFtZXRlcnM6OmdldFBpY3R1cmVTaXplKGludCAqd2lkdGgsIGludCAqaGVpZ2h0KSBjb25zdAotewotICAgICp3aWR0aCA9IC0xOwotICAgICpoZWlnaHQgPSAtMTsKLQotICAgIC8vIEdldCB0aGUgY3VycmVudCBzdHJpbmcsIGlmIGl0IGRvZXNuJ3QgZXhpc3QsIGxlYXZlIHRoZSAtMXgtMQotICAgIGNvbnN0IGNoYXIgKnAgPSBnZXQoInBpY3R1cmUtc2l6ZSIpOwotICAgIGlmIChwID09IDApCi0gICAgICAgIHJldHVybjsKLQotICAgIGludCB3LCBoOwotICAgIGlmIChwYXJzZV9zaXplKHAsIHcsIGgpID09IDApIHsKLSAgICAgICAgKndpZHRoID0gdzsKLSAgICAgICAgKmhlaWdodCA9IGg7Ci0gICAgfQotfQotCi12b2lkIENhbWVyYVBhcmFtZXRlcnM6OnNldFBpY3R1cmVGb3JtYXQoY29uc3QgY2hhciAqZm9ybWF0KQotewotICAgIHNldCgicGljdHVyZS1mb3JtYXQiLCBmb3JtYXQpOwotfQotCi1jb25zdCBjaGFyICpDYW1lcmFQYXJhbWV0ZXJzOjpnZXRQaWN0dXJlRm9ybWF0KCkgY29uc3QKLXsKLSAgICByZXR1cm4gZ2V0KCJwaWN0dXJlLWZvcm1hdCIpOwotfQotCi12b2lkIENhbWVyYVBhcmFtZXRlcnM6OmR1bXAoKSBjb25zdAotewotICAgIExPR0QoImR1bXA6IG1NYXAuc2l6ZSA9ICVkIiwgbU1hcC5zaXplKCkpOwotICAgIGZvciAoc2l6ZV90IGkgPSAwOyBpIDwgbU1hcC5zaXplKCk7IGkrKykgewotICAgICAgICBTdHJpbmc4IGssIHY7Ci0gICAgICAgIGsgPSBtTWFwLmtleUF0KGkpOwotICAgICAgICB2ID0gbU1hcC52YWx1ZUF0KGkpOwotICAgICAgICBMT0dEKCIlczogJXNcbiIsIGsuc3RyaW5nKCksIHYuc3RyaW5nKCkpOwotICAgIH0KLX0KLQotc3RhdHVzX3QgQ2FtZXJhUGFyYW1ldGVyczo6ZHVtcChpbnQgZmQsIGNvbnN0IFZlY3RvcjxTdHJpbmcxNj4mIGFyZ3MpIGNvbnN0Ci17Ci0gICAgY29uc3Qgc2l6ZV90IFNJWkUgPSAyNTY7Ci0gICAgY2hhciBidWZmZXJbU0laRV07Ci0gICAgU3RyaW5nOCByZXN1bHQ7Ci0gICAgc25wcmludGYoYnVmZmVyLCAyNTUsICJDYW1lcmFQYXJhbWV0ZXJzOjpkdW1wOiBtTWFwLnNpemUgPSAlZFxuIiwgbU1hcC5zaXplKCkpOwotICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICBmb3IgKHNpemVfdCBpID0gMDsgaSA8IG1NYXAuc2l6ZSgpOyBpKyspIHsKLSAgICAgICAgU3RyaW5nOCBrLCB2OwotICAgICAgICBrID0gbU1hcC5rZXlBdChpKTsKLSAgICAgICAgdiA9IG1NYXAudmFsdWVBdChpKTsKLSAgICAgICAgc25wcmludGYoYnVmZmVyLCAyNTUsICJcdCVzOiAlc1xuIiwgay5zdHJpbmcoKSwgdi5zdHJpbmcoKSk7Ci0gICAgICAgIHJlc3VsdC5hcHBlbmQoYnVmZmVyKTsKLSAgICB9Ci0gICAgd3JpdGUoZmQsIHJlc3VsdC5zdHJpbmcoKSwgcmVzdWx0LnNpemUoKSk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91aS9FR0xEaXNwbGF5U3VyZmFjZS5jcHAgYi9saWJzL3VpL0VHTERpc3BsYXlTdXJmYWNlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDA2Yzk4Yi4uMDAwMDAwMAotLS0gYS9saWJzL3VpL0VHTERpc3BsYXlTdXJmYWNlLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDUxOSArMCwwIEBACi0vKgotICoqCi0gKiogQ29weXJpZ2h0IDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoqCi0gKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlIFZlcnNpb24gMi4wKHRoZSAiTGljZW5zZSIpOwotICoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqKgotICoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqKgotICoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZyBzb2Z0d2FyZQotICoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMKLSAqKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIkVHTERpc3BsYXlTdXJmYWNlIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPHN5cy9pb2N0bC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHN5cy9tbWFuLmg+Ci0KLSNpbmNsdWRlIDxjdXRpbHMvbG9nLmg+Ci0jaW5jbHVkZSA8Y3V0aWxzL2F0b21pYy5oPgotI2luY2x1ZGUgPGN1dGlscy9wcm9wZXJ0aWVzLmg+Ci0KLSNpbmNsdWRlIDxoYXJkd2FyZS9jb3B5Yml0Lmg+Ci0KLSNpbmNsdWRlIDx1aS9TdXJmYWNlQ29tcG9zZXJDbGllbnQuaD4KLSNpbmNsdWRlIDx1aS9EaXNwbGF5SW5mby5oPgotI2luY2x1ZGUgPHVpL1JlY3QuaD4KLSNpbmNsdWRlIDx1aS9SZWdpb24uaD4KLSNpbmNsdWRlIDx1aS9FR0xEaXNwbGF5U3VyZmFjZS5oPgotCi0jaWYgSEFWRV9BTkRST0lEX09TCi0jaW5jbHVkZSA8bGludXgvbXNtX21kcC5oPgotI2VuZGlmCi0KLSNpbmNsdWRlIDxFR0wvZWdsLmg+Ci0KLSNpbmNsdWRlIDxwaXhlbGZsaW5nZXIvZm9ybWF0Lmg+Ci0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1lZ2xfbmF0aXZlX3dpbmRvd190KiBhbmRyb2lkX2NyZWF0ZURpc3BsYXlTdXJmYWNlKCkKLXsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190KiBzID0gbmV3IGFuZHJvaWQ6OkVHTERpc3BsYXlTdXJmYWNlKCk7Ci0gICAgcy0+bWVtb3J5X3R5cGUgPSBOQVRJVkVfTUVNT1JZX1RZUEVfR1BVOwotICAgIHJldHVybiBzOwotfQotCi0jZGVmaW5lIExJS0VMWSggZXhwICkgICAgICAgKF9fYnVpbHRpbl9leHBlY3QoIChleHApICE9IDAsIHRydWUgICkpCi0jZGVmaW5lIFVOTElLRUxZKCBleHAgKSAgICAgKF9fYnVpbHRpbl9leHBlY3QoIChleHApICE9IDAsIGZhbHNlICkpCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotRUdMRGlzcGxheVN1cmZhY2U6OkVHTERpc3BsYXlTdXJmYWNlKCkKLSAgICA6IEVHTE5hdGl2ZVN1cmZhY2U8RUdMRGlzcGxheVN1cmZhY2U+KCkKLXsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190Ojp2ZXJzaW9uID0gc2l6ZW9mKGVnbF9uYXRpdmVfd2luZG93X3QpOwotICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmlkZW50ID0gMDsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjppbmNSZWYgPSAmRUdMRGlzcGxheVN1cmZhY2U6Omhvb2tfaW5jUmVmOwotICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmRlY1JlZiA9ICZFR0xEaXNwbGF5U3VyZmFjZTo6aG9va19kZWNSZWY7Ci0gICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6c3dhcEJ1ZmZlcnMgPSAmRUdMRGlzcGxheVN1cmZhY2U6Omhvb2tfc3dhcEJ1ZmZlcnM7Ci0gICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6Y29ubmVjdCA9IDA7Ci0gICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6ZGlzY29ubmVjdCA9IDA7Ci0KLSAgICBtRmJbMF0uZGF0YSA9IDA7Ci0gICAgbUZiWzFdLmRhdGEgPSAwOwotICAgIG1CbGl0RW5naW5lID0gMDsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpmZCA9IG1hcEZyYW1lQnVmZmVyKCk7Ci0gICAgaWYgKGVnbF9uYXRpdmVfd2luZG93X3Q6OmZkID49IDApIHsKLSAgICAgICAgCi0gICAgICAgIGh3X21vZHVsZV90IGNvbnN0KiBtb2R1bGU7Ci0gICAgICAgIGlmIChod19nZXRfbW9kdWxlKENPUFlCSVRfSEFSRFdBUkVfTU9EVUxFX0lELCAmbW9kdWxlKSA9PSAwKSB7Ci0gICAgICAgICAgICBjb3B5Yml0X29wZW4obW9kdWxlLCAmbUJsaXRFbmdpbmUpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBjb25zdCBmbG9hdCBpbjJtbSA9IDI1LjRmOwotICAgICAgICBmbG9hdCByZWZyZXNoUmF0ZSA9IDEwMDAwMDAwMDAwMDAwMDBMTFUgLyAoCi0gICAgICAgICAgICAgICAgZmxvYXQoIG1JbmZvLnVwcGVyX21hcmdpbiArIG1JbmZvLmxvd2VyX21hcmdpbiArIG1JbmZvLnlyZXMgKQotICAgICAgICAgICAgICAgICogKCBtSW5mby5sZWZ0X21hcmdpbiAgKyBtSW5mby5yaWdodF9tYXJnaW4gKyBtSW5mby54cmVzICkKLSAgICAgICAgICAgICAgICAqIG1JbmZvLnBpeGNsb2NrKTsKLQotICAgICAgICBjb25zdCBHR0xTdXJmYWNlJiBidWZmZXIgPSBtRmJbMSAtIG1JbmRleF07Ci0gICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OndpZHRoICA9IGJ1ZmZlci53aWR0aDsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6aGVpZ2h0ID0gYnVmZmVyLmhlaWdodDsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6c3RyaWRlID0gYnVmZmVyLnN0cmlkZTsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6Zm9ybWF0ID0gYnVmZmVyLmZvcm1hdDsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6YmFzZSAgID0gaW50cHRyX3QobUZiWzBdLmRhdGEpOwotICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpvZmZzZXQgPQotICAgICAgICAgICAgaW50cHRyX3QoYnVmZmVyLmRhdGEpIC0gZWdsX25hdGl2ZV93aW5kb3dfdDo6YmFzZTsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6ZmxhZ3MgID0gMDsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6eGRwaSA9IChtSW5mby54cmVzICogaW4ybW0pIC8gbUluZm8ud2lkdGg7Ci0gICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OnlkcGkgPSAobUluZm8ueXJlcyAqIGluMm1tKSAvIG1JbmZvLmhlaWdodDsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6ZnBzICA9IHJlZnJlc2hSYXRlOwotICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjptZW1vcnlfdHlwZSA9IE5BVElWRV9NRU1PUllfVFlQRV9GQjsKLSAgICAgICAgLy8gbm8gZXJyb3IsIHNldCB0aGUgbWFnaWMgd29yZAotICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjptYWdpYyA9IDB4NjAwOTEzOwotICAgIH0KLSAgICBtU3dhcENvdW50ID0gLTE7Ci0gICAgbVBhZ2VGbGlwQ291bnQgPSAwOwotfQotCi1FR0xEaXNwbGF5U3VyZmFjZTo6fkVHTERpc3BsYXlTdXJmYWNlKCkKLXsKLSAgICBtYWdpYyA9IDA7Ci0gICAgY29weWJpdF9jbG9zZShtQmxpdEVuZ2luZSk7Ci0gICAgbUJsaXRFbmdpbmUgPSAwOwotICAgIGNsb3NlKGVnbF9uYXRpdmVfd2luZG93X3Q6OmZkKTsKLSAgICBtdW5tYXAobUZiWzBdLmRhdGEsIG1TaXplKTsKLSAgICBpZiAoIShtRmxhZ3MgJiBQQUdFX0ZMSVApKQotICAgICAgICBmcmVlKCh2b2lkKiltRmJbMV0uZGF0YSk7Ci19Ci0KLXZvaWQgRUdMRGlzcGxheVN1cmZhY2U6Omhvb2tfaW5jUmVmKE5hdGl2ZVdpbmRvd1R5cGUgd2luZG93KSB7Ci0gICAgRUdMRGlzcGxheVN1cmZhY2UqIHRoYXQgPSBzdGF0aWNfY2FzdDxFR0xEaXNwbGF5U3VyZmFjZSo+KHdpbmRvdyk7Ci0gICAgdGhhdC0+aW5jU3Ryb25nKHRoYXQpOwotfQotdm9pZCBFR0xEaXNwbGF5U3VyZmFjZTo6aG9va19kZWNSZWYoTmF0aXZlV2luZG93VHlwZSB3aW5kb3cpIHsKLSAgICBFR0xEaXNwbGF5U3VyZmFjZSogdGhhdCA9IHN0YXRpY19jYXN0PEVHTERpc3BsYXlTdXJmYWNlKj4od2luZG93KTsKLSAgICB0aGF0LT5kZWNTdHJvbmcodGhhdCk7Ci19Ci11aW50MzJfdCBFR0xEaXNwbGF5U3VyZmFjZTo6aG9va19zd2FwQnVmZmVycyhOYXRpdmVXaW5kb3dUeXBlIHdpbmRvdykgewotICAgIEVHTERpc3BsYXlTdXJmYWNlKiB0aGF0ID0gc3RhdGljX2Nhc3Q8RUdMRGlzcGxheVN1cmZhY2UqPih3aW5kb3cpOwotICAgIHJldHVybiB0aGF0LT5zd2FwQnVmZmVycygpOwotfQotCi12b2lkIEVHTERpc3BsYXlTdXJmYWNlOjpzZXRTd2FwUmVjdGFuZ2xlKGludCBsLCBpbnQgdCwgaW50IHcsIGludCBoKQotewotICAgIG1JbmZvLnJlc2VydmVkWzBdID0gMHg1NDQ0NTA1NTsgLy8gIlVQRFQiOwotICAgIG1JbmZvLnJlc2VydmVkWzFdID0gKHVpbnQxNl90KWwgfCAoKHVpbnQzMl90KXQgPDwgMTYpOwotICAgIG1JbmZvLnJlc2VydmVkWzJdID0gKHVpbnQxNl90KShsK3cpIHwgKCh1aW50MzJfdCkodCtoKSA8PCAxNik7Ci19Ci0KLXVpbnQzMl90IEVHTERpc3BsYXlTdXJmYWNlOjpzd2FwQnVmZmVycygpCi17Ci0jZGVmaW5lIFNIT1dfRlBTIDAKLSNpZiBTSE9XX0ZQUwotICAgIG5zZWNzX3Qgbm93ID0gc3lzdGVtVGltZSgpOwotICAgIGlmIChtU3dhcENvdW50ID09IC0xKSB7Ci0gICAgICAgIG1UaW1lID0gbm93OwotICAgICAgICBtU3dhcENvdW50ID0gMDsKLSAgICAgICAgbVNsZWVwID0gMDsKLSAgICB9IGVsc2UgewotICAgICAgICBuc2Vjc190IGQgPSBub3ctbVRpbWU7Ci0gICAgICAgIGlmIChkID49IHNlY29uZHMoMSkpIHsKLSAgICAgICAgICAgIGRvdWJsZSBmcHMgPSAobVN3YXBDb3VudCAqIGRvdWJsZShzZWNvbmRzKDEpKSkgLyBkb3VibGUoZCk7Ci0gICAgICAgICAgICBMT0dEKCIlZiBmcHMsIHNsZWVwPSVkIC8gZnJhbWUiLAotICAgICAgICAgICAgICAgICAgICBmcHMsIChpbnQpbnMydXMobVNsZWVwIC8gbVN3YXBDb3VudCkpOwotICAgICAgICAgICAgbVN3YXBDb3VudCA9IDA7Ci0gICAgICAgICAgICBtVGltZSA9IG5vdzsKLSAgICAgICAgICAgIG1TbGVlcCA9IDA7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBtU3dhcENvdW50Kys7Ci0gICAgICAgIH0KLSAgICB9Ci0jZW5kaWYKLSAgICAvKiBJZiB3ZSBjYW4ndCBkbyB0aGUgcGFnZV9mbGlwLCBqdXN0IGNvcHkgdGhlIGJhY2sgYnVmZmVyIHRvIHRoZSBmcm9udCAqLwotICAgIGlmICghKG1GbGFncyAmIFBBR0VfRkxJUCkpIHsKLSAgICAgICAgbWVtY3B5KG1GYlswXS5kYXRhLCBtRmJbMV0uZGF0YSwgbUluZm8ueHJlcyptSW5mby55cmVzKjIpOwotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvLyBkbyB0aGUgYWN0dWFsIGZsaXAKLSAgICBtSW5kZXggPSAxIC0gbUluZGV4OwotICAgIG1JbmZvLmFjdGl2YXRlID0gRkJfQUNUSVZBVEVfVkJMOwotICAgIG1JbmZvLnlvZmZzZXQgPSBtSW5kZXggPyBtSW5mby55cmVzIDogMDsKLSAgICBpZiAoaW9jdGwoZWdsX25hdGl2ZV93aW5kb3dfdDo6ZmQsIEZCSU9QVVRfVlNDUkVFTklORk8sICZtSW5mbykgPT0gLTEpIHsKLSAgICAgICAgTE9HRSgiRkJJT1BVVF9WU0NSRUVOSU5GTyBmYWlsZWQiKTsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiB0aGlzIGlzIGEgbW9uc3Ryb3VzIGhhY2s6IEJlY2F1c2UgdGhlIGgvdyBhY2NlbGVyYXRvciBpcyBub3QgYWJsZQotICAgICAqIHRvIHJlbmRlciBkaXJlY3RseSBpbnRvIHRoZSBmcmFtZWJ1ZmZlciwgd2UgbmVlZCB0byBjb3B5IGl0cwotICAgICAqIGludGVybmFsIGZyYW1lYnVmZmVyIG91dCB0byB0aGUgZmIuCi0gICAgICogb2VtWzBdIGlzIHVzZWQgdG8gYWNjZXNzIHRoZSBmZCBvZiBpbnRlcm5hbCBmYi4KLSAgICAgKiBBbGwgdGhpcyBpcyBuZWVkZWQgb25seSBpbiBzdGFuZGFsb25lIG1vZGUsIGluIFN1cmZhY2VGbGluZ2VyIG1vZGUKLSAgICAgKiB3ZSBjb250cm9sIHdoZXJlIHRoZSBHUFUgcmVuZGVycy4KLSAgICAgKiBXZSBkbyB0aGlzIG9ubHkgaWYgd2UgaGF2ZSBjb3B5Yml0LCBzaW5jZSB0aGlzIGhhY2sgaXMgbmVlZGVkIG9ubHkKLSAgICAgKiB3aXRoIG1zbTdrLgotICAgICAqLwotICAgIGlmIChlZ2xfbmF0aXZlX3dpbmRvd190OjptZW1vcnlfdHlwZSA9PSBOQVRJVkVfTUVNT1JZX1RZUEVfR1BVICYmIG9lbVswXSAmJiBtQmxpdEVuZ2luZSkgewotICAgICAgICBjb3B5Yml0X2RldmljZV90ICpjb3B5Yml0ID0gbUJsaXRFbmdpbmU7Ci0gICAgICAgIGNvcHliaXRfcmVjdF90IHNkcmVjdCA9IHsgMCwgMCwKLSAgICAgICAgICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190Ojp3aWR0aCwgZWdsX25hdGl2ZV93aW5kb3dfdDo6aGVpZ2h0IH07Ci0gICAgICAgIGNvcHliaXRfaW1hZ2VfdCBkc3QgPSB7Ci0gICAgICAgICAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6d2lkdGgsCi0gICAgICAgICAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6aGVpZ2h0LAotICAgICAgICAgICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmZvcm1hdCwKLSAgICAgICAgICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpvZmZzZXQsCi0gICAgICAgICAgICAgICAgKHZvaWQqKWVnbF9uYXRpdmVfd2luZG93X3Q6OmJhc2UsCi0gICAgICAgICAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6ZmQKLSAgICAgICAgfTsKLSAgICAgICAgY29weWJpdF9pbWFnZV90IHNyYyA9IHsKLSAgICAgICAgICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190Ojp3aWR0aCwKLSAgICAgICAgICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpoZWlnaHQsCi0gICAgICAgICAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6Zm9ybWF0LCAvLyBYWFg6IHVzZSBwcm9wZXIgZm9ybWF0Ci0gICAgICAgICAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6b2Zmc2V0LAotICAgICAgICAgICAgICAgICh2b2lkKillZ2xfbmF0aXZlX3dpbmRvd190OjpiYXNlLCAgLy8gWFhYOiB1c2UgcHJvcGVyIGJhc2UKLSAgICAgICAgICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpvZW1bMF0KLSAgICAgICAgfTsKLSAgICAgICAgcmVnaW9uX2l0ZXJhdG9yIGl0KFJlZ2lvbihSZWN0KAotICAgICAgICAgICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OndpZHRoLCBlZ2xfbmF0aXZlX3dpbmRvd190OjpoZWlnaHQpKSk7Ci0gICAgICAgIGNvcHliaXQtPnNldF9wYXJhbWV0ZXIoY29weWJpdCwgQ09QWUJJVF9UUkFOU0ZPUk0sIDApOwotICAgICAgICBjb3B5Yml0LT5zZXRfcGFyYW1ldGVyKGNvcHliaXQsIENPUFlCSVRfUExBTkVfQUxQSEEsIDB4RkYpOwotICAgICAgICBjb3B5Yml0LT5zZXRfcGFyYW1ldGVyKGNvcHliaXQsIENPUFlCSVRfRElUSEVSLCBDT1BZQklUX0RJU0FCTEUpOwotICAgICAgICBjb3B5Yml0LT5zdHJldGNoKGNvcHliaXQsICZkc3QsICZzcmMsICZzZHJlY3QsICZzZHJlY3QsICZpdCk7Ci0gICAgfQotCi0gICAgLy8gdXBkYXRlIHRoZSBhZGRyZXNzIG9mIHRoZSBidWZmZXIgdG8gZHJhdyB0byBuZXh0Ci0gICAgY29uc3QgR0dMU3VyZmFjZSYgYnVmZmVyID0gbUZiWzEgLSBtSW5kZXhdOwotICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6Om9mZnNldCA9Ci0gICAgICAgIGludHB0cl90KGJ1ZmZlci5kYXRhKSAtIGVnbF9uYXRpdmVfd2luZG93X3Q6OmJhc2U7Ci0KLSNpZiBTSE9XX0ZQUwotICAgIG1TbGVlcCArPSBzeXN0ZW1UaW1lKCktbm93OwotI2VuZGlmCi0KLSAgICBtUGFnZUZsaXBDb3VudCsrOwotCi0gICAgLy8gV2UgZG9uJ3Qgc3VwcG9ydCBzY3JlZW4tc2l6ZSBjaGFuZ2VzIGZvciBub3cKLSAgICByZXR1cm4gMDsKLX0KLQotaW50MzJfdCBFR0xEaXNwbGF5U3VyZmFjZTo6Z2V0UGFnZUZsaXBDb3VudCgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1QYWdlRmxpcENvdW50OwotfQotCi12b2lkIEVHTERpc3BsYXlTdXJmYWNlOjpjb3B5RnJvbnRUb0JhY2soY29uc3QgUmVnaW9uJiBjb3B5YmFjaykKLXsKLSNpZiBIQVZFX0FORFJPSURfT1MKLSAgICBpZiAobUJsaXRFbmdpbmUpIHsKLSAgICAgICAgY29weWJpdF9pbWFnZV90IGRzdCA9IHsKLSAgICAgICAgICAgICAgICB3OiAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OnN0cmlkZSwKLSAgICAgICAgICAgICAgICBoOiAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmhlaWdodCwKLSAgICAgICAgICAgICAgICBmb3JtYXQ6IGVnbF9uYXRpdmVfd2luZG93X3Q6OmZvcm1hdCwKLSAgICAgICAgICAgICAgICBvZmZzZXQ6IG1GYlsxLW1JbmRleF0uZGF0YSAtIG1GYlswXS5kYXRhLAotICAgICAgICAgICAgICAgIGJhc2U6ICAgKHZvaWQqKWVnbF9uYXRpdmVfd2luZG93X3Q6OmJhc2UsCi0gICAgICAgICAgICAgICAgZmQ6ICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpmZAotICAgICAgICB9OwotICAgICAgICBjb3B5Yml0X2ltYWdlX3Qgc3JjID0gewotICAgICAgICAgICAgICAgIHc6ICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6c3RyaWRlLAotICAgICAgICAgICAgICAgIGg6ICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6aGVpZ2h0LAotICAgICAgICAgICAgICAgIGZvcm1hdDogZWdsX25hdGl2ZV93aW5kb3dfdDo6Zm9ybWF0LAotICAgICAgICAgICAgICAgIG9mZnNldDogbUZiW21JbmRleF0uZGF0YSAtIG1GYlswXS5kYXRhLAotICAgICAgICAgICAgICAgIGJhc2U6ICAgKHZvaWQqKWVnbF9uYXRpdmVfd2luZG93X3Q6OmJhc2UsCi0gICAgICAgICAgICAgICAgZmQ6ICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpmZAotICAgICAgICB9OwotICAgICAgICByZWdpb25faXRlcmF0b3IgaXQoY29weWJhY2spOwotICAgICAgICBtQmxpdEVuZ2luZS0+YmxpdChtQmxpdEVuZ2luZSwgJmRzdCwgJnNyYywgJml0KTsKLSAgICB9IGVsc2UKLSNlbmRpZgotICAgIHsKLSAgICAgICAgLyogbm8gZXh0cmEgY29weSBuZWVkZWQgc2luY2Ugd2UgY29waWVkIGJhY2sgdG8gZnJvbnQgaW5zdGVhZCBvZgotICAgICAgICAgKiBmbGlwcGluZyAqLwotICAgICAgICBpZiAoIShtRmxhZ3MgJiBQQUdFX0ZMSVApKSB7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLQotICAgICAgICBSZWdpb246Oml0ZXJhdG9yIGl0ZXJhdG9yKGNvcHliYWNrKTsKLSAgICAgICAgaWYgKGl0ZXJhdG9yKSB7Ci0gICAgICAgICAgICBSZWN0IHI7Ci0gICAgICAgICAgICB1aW50OF90KiBjb25zdCBzY3JlZW5fc3JjID0gbUZiWyAgbUluZGV4XS5kYXRhOwotICAgICAgICAgICAgdWludDhfdCogY29uc3Qgc2NyZWVuX2RzdCA9IG1GYlsxLW1JbmRleF0uZGF0YTsKLSAgICAgICAgICAgIGNvbnN0IHNpemVfdCBicHAgPSBieXRlc1BlclBpeGVsKGVnbF9uYXRpdmVfd2luZG93X3Q6OmZvcm1hdCk7Ci0gICAgICAgICAgICBjb25zdCBzaXplX3QgYnByID0gZWdsX25hdGl2ZV93aW5kb3dfdDo6c3RyaWRlICogYnBwOwotICAgICAgICAgICAgd2hpbGUgKGl0ZXJhdG9yLml0ZXJhdGUoJnIpKSB7Ci0gICAgICAgICAgICAgICAgc3NpemVfdCBoID0gci5ib3R0b20gLSByLnRvcDsKLSAgICAgICAgICAgICAgICBpZiAoaCkgewotICAgICAgICAgICAgICAgICAgICBzaXplX3Qgc2l6ZSA9IChyLnJpZ2h0IC0gci5sZWZ0KSAqIGJwcDsKLSAgICAgICAgICAgICAgICAgICAgc2l6ZV90IG8gPSAoci5sZWZ0ICsgZWdsX25hdGl2ZV93aW5kb3dfdDo6c3RyaWRlICogci50b3ApICogYnBwOwotICAgICAgICAgICAgICAgICAgICB1aW50OF90KiBzID0gc2NyZWVuX3NyYyArIG87Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQ4X3QqIGQgPSBzY3JlZW5fZHN0ICsgbzsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHNpemUgPT0gYnByKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzaXplICo9IGg7Ci0gICAgICAgICAgICAgICAgICAgICAgICBoID0gMTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBtZW1jcHkoZCwgcywgc2l6ZSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBkICs9IGJwcjsKLSAgICAgICAgICAgICAgICAgICAgICAgIHMgKz0gYnByOwotICAgICAgICAgICAgICAgICAgICB9IHdoaWxlICgtLWggPiAwKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLXZvaWQgRUdMRGlzcGxheVN1cmZhY2U6OmNvcHlGcm9udFRvSW1hZ2UoY29uc3QgY29weWJpdF9pbWFnZV90JiBkc3QpCi17Ci0jaWYgSEFWRV9BTkRST0lEX09TCi0gICAgaWYgKG1CbGl0RW5naW5lKSB7Ci0gICAgICAgIGNvcHliaXRfaW1hZ2VfdCBzcmMgPSB7Ci0gICAgICAgICAgICAgICAgdzogICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpzdHJpZGUsCi0gICAgICAgICAgICAgICAgaDogICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpoZWlnaHQsCi0gICAgICAgICAgICAgICAgZm9ybWF0OiBlZ2xfbmF0aXZlX3dpbmRvd190Ojpmb3JtYXQsCi0gICAgICAgICAgICAgICAgb2Zmc2V0OiBtRmJbbUluZGV4XS5kYXRhIC0gbUZiWzBdLmRhdGEsCi0gICAgICAgICAgICAgICAgYmFzZTogICAodm9pZCopZWdsX25hdGl2ZV93aW5kb3dfdDo6YmFzZSwKLSAgICAgICAgICAgICAgICBmZDogICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmZkCi0gICAgICAgIH07Ci0gICAgICAgIHJlZ2lvbl9pdGVyYXRvciBpdChSZWdpb24oUmVjdCgKLSAgICAgICAgICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190Ojp3aWR0aCwgZWdsX25hdGl2ZV93aW5kb3dfdDo6aGVpZ2h0KSkpOwotICAgICAgICBtQmxpdEVuZ2luZS0+YmxpdChtQmxpdEVuZ2luZSwgJmRzdCwgJnNyYywgJml0KTsKLSAgICB9IGVsc2UKLSNlbmRpZgotICAgIHsKLSAgICAgICAgdWludDhfdCogY29uc3Qgc2NyZWVuX3NyYyA9IG1GYlsgIG1JbmRleF0uZGF0YTsKLSAgICAgICAgY29uc3Qgc2l6ZV90IGJwcCA9IGJ5dGVzUGVyUGl4ZWwoZWdsX25hdGl2ZV93aW5kb3dfdDo6Zm9ybWF0KTsKLSAgICAgICAgY29uc3Qgc2l6ZV90IGJwciA9IGVnbF9uYXRpdmVfd2luZG93X3Q6OnN0cmlkZSAqIGJwcDsKLSAgICAgICAgbWVtY3B5KChjaGFyKilkc3QuYmFzZSArIGRzdC5vZmZzZXQsIHNjcmVlbl9zcmMsCi0gICAgICAgICAgICAgICAgYnByKmVnbF9uYXRpdmVfd2luZG93X3Q6OmhlaWdodCk7Ci0gICAgfQotfQotCi12b2lkIEVHTERpc3BsYXlTdXJmYWNlOjpjb3B5QmFja1RvSW1hZ2UoY29uc3QgY29weWJpdF9pbWFnZV90JiBkc3QpCi17Ci0jaWYgSEFWRV9BTkRST0lEX09TCi0gICAgaWYgKG1CbGl0RW5naW5lKSB7Ci0gICAgICAgIGNvcHliaXRfaW1hZ2VfdCBzcmMgPSB7Ci0gICAgICAgICAgICAgICAgdzogICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpzdHJpZGUsCi0gICAgICAgICAgICAgICAgaDogICAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpoZWlnaHQsCi0gICAgICAgICAgICAgICAgZm9ybWF0OiBlZ2xfbmF0aXZlX3dpbmRvd190Ojpmb3JtYXQsCi0gICAgICAgICAgICAgICAgb2Zmc2V0OiBtRmJbMS1tSW5kZXhdLmRhdGEgLSBtRmJbMF0uZGF0YSwKLSAgICAgICAgICAgICAgICBiYXNlOiAgICh2b2lkKillZ2xfbmF0aXZlX3dpbmRvd190OjpiYXNlLAotICAgICAgICAgICAgICAgIGZkOiAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6ZmQKLSAgICAgICAgfTsKLSAgICAgICAgcmVnaW9uX2l0ZXJhdG9yIGl0KFJlZ2lvbihSZWN0KAotICAgICAgICAgICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OndpZHRoLCBlZ2xfbmF0aXZlX3dpbmRvd190OjpoZWlnaHQpKSk7Ci0gICAgICAgIG1CbGl0RW5naW5lLT5ibGl0KG1CbGl0RW5naW5lLCAmZHN0LCAmc3JjLCAmaXQpOwotICAgIH0gZWxzZQotI2VuZGlmCi0gICAgewotICAgICAgICB1aW50OF90KiBjb25zdCBzY3JlZW5fc3JjID0gbUZiWzEtbUluZGV4XS5kYXRhOwotICAgICAgICBjb25zdCBzaXplX3QgYnBwID0gYnl0ZXNQZXJQaXhlbChlZ2xfbmF0aXZlX3dpbmRvd190Ojpmb3JtYXQpOwotICAgICAgICBjb25zdCBzaXplX3QgYnByID0gZWdsX25hdGl2ZV93aW5kb3dfdDo6c3RyaWRlICogYnBwOwotICAgICAgICBtZW1jcHkoKGNoYXIqKWRzdC5iYXNlICsgZHN0Lm9mZnNldCwgc2NyZWVuX3NyYywKLSAgICAgICAgICAgICAgICBicHIqZWdsX25hdGl2ZV93aW5kb3dfdDo6aGVpZ2h0KTsKLSAgICB9Ci19Ci0KLQotc3RhdHVzX3QgRUdMRGlzcGxheVN1cmZhY2U6Om1hcEZyYW1lQnVmZmVyKCkKLXsKLSAgICBjaGFyIGNvbnN0ICogY29uc3QgZGV2aWNlX3RlbXBsYXRlW10gPSB7Ci0gICAgICAgICAgICAiL2Rldi9ncmFwaGljcy9mYiV1IiwKLSAgICAgICAgICAgICIvZGV2L2ZiJXUiLAotICAgICAgICAgICAgMCB9OwotICAgIGludCBmZCA9IC0xOwotICAgIGludCBpPTA7Ci0gICAgY2hhciBuYW1lWzY0XTsKLSAgICB3aGlsZSAoKGZkPT0tMSkgJiYgZGV2aWNlX3RlbXBsYXRlW2ldKSB7Ci0gICAgICAgIHNucHJpbnRmKG5hbWUsIDY0LCBkZXZpY2VfdGVtcGxhdGVbaV0sIDApOwotICAgICAgICBmZCA9IG9wZW4obmFtZSwgT19SRFdSLCAwKTsKLSAgICAgICAgaSsrOwotICAgIH0KLSAgICBpZiAoZmQgPCAwKQotICAgICAgICByZXR1cm4gLWVycm5vOwotCi0gICAgc3RydWN0IGZiX2ZpeF9zY3JlZW5pbmZvIGZpbmZvOwotICAgIGlmIChpb2N0bChmZCwgRkJJT0dFVF9GU0NSRUVOSU5GTywgJmZpbmZvKSA9PSAtMSkKLSAgICAgICAgcmV0dXJuIC1lcnJubzsKLQotICAgIHN0cnVjdCBmYl92YXJfc2NyZWVuaW5mbyBpbmZvOwotICAgIGlmIChpb2N0bChmZCwgRkJJT0dFVF9WU0NSRUVOSU5GTywgJmluZm8pID09IC0xKQotICAgICAgICByZXR1cm4gLWVycm5vOwotCi0gICAgaW5mby5yZXNlcnZlZFswXSA9IDA7Ci0gICAgaW5mby5yZXNlcnZlZFsxXSA9IDA7Ci0gICAgaW5mby5yZXNlcnZlZFsyXSA9IDA7Ci0gICAgaW5mby54b2Zmc2V0ID0gMDsKLSAgICBpbmZvLnlvZmZzZXQgPSAwOwotICAgIGluZm8ueXJlc192aXJ0dWFsID0gaW5mby55cmVzICogMjsKLSAgICBpbmZvLmJpdHNfcGVyX3BpeGVsID0gMTY7Ci0gICAgLyogRXhwbGljaXRseSByZXF1ZXN0IDUvNi81ICovCi0gICAgaW5mby5yZWQub2Zmc2V0ID0gMTE7Ci0gICAgaW5mby5yZWQubGVuZ3RoID0gNTsKLSAgICBpbmZvLmdyZWVuLm9mZnNldCA9IDU7Ci0gICAgaW5mby5ncmVlbi5sZW5ndGggPSA2OwotICAgIGluZm8uYmx1ZS5vZmZzZXQgPSAwOwotICAgIGluZm8uYmx1ZS5sZW5ndGggPSA1OwotICAgIGluZm8udHJhbnNwLm9mZnNldCA9IDA7Ci0gICAgaW5mby50cmFuc3AubGVuZ3RoID0gMDsKLSAgICBpbmZvLmFjdGl2YXRlID0gRkJfQUNUSVZBVEVfTk9XOwotCi0gICAgdWludDMyX3QgZmxhZ3MgPSBQQUdFX0ZMSVA7Ci0gICAgaWYgKGlvY3RsKGZkLCBGQklPUFVUX1ZTQ1JFRU5JTkZPLCAmaW5mbykgPT0gLTEpIHsKLSAgICAgICAgaW5mby55cmVzX3ZpcnR1YWwgPSBpbmZvLnlyZXM7Ci0gICAgICAgIGZsYWdzICY9IH5QQUdFX0ZMSVA7Ci0gICAgICAgIExPR1coIkZCSU9QVVRfVlNDUkVFTklORk8gZmFpbGVkLCBwYWdlIGZsaXBwaW5nIG5vdCBzdXBwb3J0ZWQiKTsKLSAgICB9Ci0KLSAgICBpZiAoaW5mby55cmVzX3ZpcnR1YWwgPCBpbmZvLnlyZXMgKiAyKSB7Ci0gICAgICAgIGluZm8ueXJlc192aXJ0dWFsID0gaW5mby55cmVzOwotICAgICAgICBmbGFncyAmPSB+UEFHRV9GTElQOwotICAgICAgICBMT0dXKCJwYWdlIGZsaXBwaW5nIG5vdCBzdXBwb3J0ZWQgKHlyZXNfdmlydHVhbD0lZCwgcmVxdWVzdGVkPSVkKSIsCi0gICAgICAgICAgICAgICAgaW5mby55cmVzX3ZpcnR1YWwsIGluZm8ueXJlcyoyKTsKLSAgICB9Ci0KLSAgICBpZiAoaW9jdGwoZmQsIEZCSU9HRVRfVlNDUkVFTklORk8sICZpbmZvKSA9PSAtMSkKLSAgICAgICAgcmV0dXJuIC1lcnJubzsKLQotICAgIGludCByZWZyZXNoUmF0ZSA9IDEwMDAwMDAwMDAwMDAwMDBMTFUgLwotICAgICgKLSAgICAgICAgICAgIHVpbnQ2NF90KCBpbmZvLnVwcGVyX21hcmdpbiArIGluZm8ubG93ZXJfbWFyZ2luICsgaW5mby55cmVzICkKLSAgICAgICAgICAgICogKCBpbmZvLmxlZnRfbWFyZ2luICArIGluZm8ucmlnaHRfbWFyZ2luICsgaW5mby54cmVzICkKLSAgICAgICAgICAgICogaW5mby5waXhjbG9jawotICAgICk7Ci0KLSAgICBpZiAocmVmcmVzaFJhdGUgPT0gMCkgewotICAgICAgICAvLyBibGVhZ2gsIGJhZCBpbmZvIGZyb20gdGhlIGRyaXZlcgotICAgICAgICByZWZyZXNoUmF0ZSA9IDYwKjEwMDA7ICAvLyA2MCBIegotICAgIH0KLSAgICBpZiAoaW50KGluZm8ud2lkdGgpIDw9IDAgfHwgaW50KGluZm8uaGVpZ2h0KSA8PSAwKSB7Ci0gICAgICAgIC8vIHRoZSBkcml2ZXIgZG9lc24ndCByZXR1cm4gdGhhdCBpbmZvcm1hdGlvbgotICAgICAgICAvLyBkZWZhdWx0IHRvIDE2MCBkcGkKLSAgICAgICAgaW5mby53aWR0aCAgPSAoKGluZm8ueHJlcyAqIDI1LjRmKS8xNjAuMGYgKyAwLjVmKTsKLSAgICAgICAgaW5mby5oZWlnaHQgPSAoKGluZm8ueXJlcyAqIDI1LjRmKS8xNjAuMGYgKyAwLjVmKTsKLSAgICB9Ci0KLSAgICBmbG9hdCB4ZHBpID0gKGluZm8ueHJlcyAqIDI1LjRmKSAvIGluZm8ud2lkdGg7Ci0gICAgZmxvYXQgeWRwaSA9IChpbmZvLnlyZXMgKiAyNS40ZikgLyBpbmZvLmhlaWdodDsKLSAgICBmbG9hdCBmcHMgID0gcmVmcmVzaFJhdGUgLyAxMDAwLjBmOwotCi0gICAgTE9HSSggICAidXNpbmcgKGZkPSVkKVxuIgotICAgICAgICAgICAgImlkICAgICAgICAgICA9ICVzXG4iCi0gICAgICAgICAgICAieHJlcyAgICAgICAgID0gJWQgcHhcbiIKLSAgICAgICAgICAgICJ5cmVzICAgICAgICAgPSAlZCBweFxuIgotICAgICAgICAgICAgInhyZXNfdmlydHVhbCA9ICVkIHB4XG4iCi0gICAgICAgICAgICAieXJlc192aXJ0dWFsID0gJWQgcHhcbiIKLSAgICAgICAgICAgICJicHAgICAgICAgICAgPSAlZFxuIgotICAgICAgICAgICAgInIgICAgICAgICAgICA9ICUydToldVxuIgotICAgICAgICAgICAgImcgICAgICAgICAgICA9ICUydToldVxuIgotICAgICAgICAgICAgImIgICAgICAgICAgICA9ICUydToldVxuIiwKLSAgICAgICAgICAgIGZkLAotICAgICAgICAgICAgZmluZm8uaWQsCi0gICAgICAgICAgICBpbmZvLnhyZXMsCi0gICAgICAgICAgICBpbmZvLnlyZXMsCi0gICAgICAgICAgICBpbmZvLnhyZXNfdmlydHVhbCwKLSAgICAgICAgICAgIGluZm8ueXJlc192aXJ0dWFsLAotICAgICAgICAgICAgaW5mby5iaXRzX3Blcl9waXhlbCwKLSAgICAgICAgICAgIGluZm8ucmVkLm9mZnNldCwgaW5mby5yZWQubGVuZ3RoLAotICAgICAgICAgICAgaW5mby5ncmVlbi5vZmZzZXQsIGluZm8uZ3JlZW4ubGVuZ3RoLAotICAgICAgICAgICAgaW5mby5ibHVlLm9mZnNldCwgaW5mby5ibHVlLmxlbmd0aAotICAgICk7Ci0KLSAgICBMT0dJKCAgICJ3aWR0aCAgICAgICAgPSAlZCBtbSAoJWYgZHBpKVxuIgotICAgICAgICAgICAgImhlaWdodCAgICAgICA9ICVkIG1tICglZiBkcGkpXG4iCi0gICAgICAgICAgICAicmVmcmVzaCByYXRlID0gJS4yZiBIelxuIiwKLSAgICAgICAgICAgIGluZm8ud2lkdGgsICB4ZHBpLAotICAgICAgICAgICAgaW5mby5oZWlnaHQsIHlkcGksCi0gICAgICAgICAgICBmcHMKLSAgICApOwotCi0KLSAgICBpZiAoaW9jdGwoZmQsIEZCSU9HRVRfRlNDUkVFTklORk8sICZmaW5mbykgPT0gLTEpCi0gICAgICAgIHJldHVybiAtZXJybm87Ci0KLSAgICBpZiAoZmluZm8uc21lbV9sZW4gPD0gMCkKLSAgICAgICAgcmV0dXJuIC1lcnJubzsKLQotICAgIC8qCi0gICAgICogT3BlbiBhbmQgbWFwIHRoZSBkaXNwbGF5LgotICAgICAqLwotCi0gICAgdm9pZCogYnVmZmVyICA9ICh1aW50MTZfdCopIG1tYXAoCi0gICAgICAgICAgICAwLCBmaW5mby5zbWVtX2xlbiwKLSAgICAgICAgICAgIFBST1RfUkVBRCB8IFBST1RfV1JJVEUsCi0gICAgICAgICAgICBNQVBfU0hBUkVELAotICAgICAgICAgICAgZmQsIDApOwotCi0gICAgaWYgKGJ1ZmZlciA9PSBNQVBfRkFJTEVEKQotICAgICAgICByZXR1cm4gLWVycm5vOwotCi0gICAgLy8gYXQgbGVhc3QgZm9yIG5vdywgYWx3YXlzIGNsZWFyIHRoZSBmYgotICAgIG1lbXNldChidWZmZXIsIDAsIGZpbmZvLnNtZW1fbGVuKTsKLQotICAgIHVpbnQ4X3QqIG9mZnNjcmVlblsyXTsKLSAgICBvZmZzY3JlZW5bMF0gPSAodWludDhfdCopYnVmZmVyOwotICAgIGlmIChmbGFncyAmIFBBR0VfRkxJUCkgewotICAgICAgICBvZmZzY3JlZW5bMV0gPSAodWludDhfdCopYnVmZmVyICsgZmluZm8ubGluZV9sZW5ndGgqaW5mby55cmVzOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIG9mZnNjcmVlblsxXSA9ICh1aW50OF90KiltYWxsb2MoZmluZm8uc21lbV9sZW4pOwotICAgICAgICBpZiAob2Zmc2NyZWVuWzFdID09IDApIHsKLSAgICAgICAgICAgIG11bm1hcChidWZmZXIsIGZpbmZvLnNtZW1fbGVuKTsKLSAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBtRmxhZ3MgPSBmbGFnczsKLSAgICBtSW5mbyA9IGluZm87Ci0gICAgbUZpbmZvID0gZmluZm87Ci0gICAgbVNpemUgPSBmaW5mby5zbWVtX2xlbjsKLSAgICBtSW5kZXggPSAwOwotICAgIGZvciAoaW50IGk9MCA7IGk8MiA7IGkrKykgewotICAgICAgICBtRmJbaV0udmVyc2lvbiA9IHNpemVvZihHR0xTdXJmYWNlKTsKLSAgICAgICAgbUZiW2ldLndpZHRoICAgPSBpbmZvLnhyZXM7Ci0gICAgICAgIG1GYltpXS5oZWlnaHQgID0gaW5mby55cmVzOwotICAgICAgICBtRmJbaV0uc3RyaWRlICA9IGZpbmZvLmxpbmVfbGVuZ3RoIC8gKGluZm8uYml0c19wZXJfcGl4ZWwgPj4gMyk7Ci0gICAgICAgIG1GYltpXS5kYXRhICAgID0gKEdHTHVieXRlKikob2Zmc2NyZWVuW2ldKTsKLSAgICAgICAgbUZiW2ldLmZvcm1hdCAgPSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjU7Ci0gICAgfQotICAgIHJldHVybiBmZDsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KZGlmZiAtLWdpdCBhL2xpYnMvdWkvRUdMTmF0aXZlV2luZG93U3VyZmFjZS5jcHAgYi9saWJzL3VpL0VHTE5hdGl2ZVdpbmRvd1N1cmZhY2UuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmMTA3MWNmLi4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvRUdMTmF0aXZlV2luZG93U3VyZmFjZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxNjEgKzAsMCBAQAotLyogCi0qKgotKiogQ29weXJpZ2h0IDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSBWZXJzaW9uIDIuMCh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUyAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5EIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2RlZmluZSBMT0dfVEFHICJFR0xOYXRpdmVXaW5kb3dTdXJmYWNlIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLQotI2luY2x1ZGUgPGN1dGlscy9sb2cuaD4KLSNpbmNsdWRlIDxjdXRpbHMvYXRvbWljLmg+Ci0KLSNpbmNsdWRlIDx1aS9TdXJmYWNlQ29tcG9zZXJDbGllbnQuaD4KLSNpbmNsdWRlIDx1aS9EaXNwbGF5SW5mby5oPgotI2luY2x1ZGUgPHVpL1JlY3QuaD4KLQotI2luY2x1ZGUgPEVHTC9lZ2wuaD4KLQotI2luY2x1ZGUgPHBpeGVsZmxpbmdlci9mb3JtYXQuaD4KLQotI2luY2x1ZGUgPHVpL0VHTE5hdGl2ZVdpbmRvd1N1cmZhY2UuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotbmFtZXNwYWNlIGFuZHJvaWQgewotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1FR0xOYXRpdmVXaW5kb3dTdXJmYWNlOjpFR0xOYXRpdmVXaW5kb3dTdXJmYWNlKGNvbnN0IHNwPFN1cmZhY2U+JiBzdXJmYWNlKQotICAgIDogRUdMTmF0aXZlU3VyZmFjZTxFR0xOYXRpdmVXaW5kb3dTdXJmYWNlPigpLAotICAgIG1TdXJmYWNlKHN1cmZhY2UpLCBtQ29ubmVjdGVkKGZhbHNlKQotewotICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6Om1hZ2ljID0gMHg2MDA5MTM7Ci0gICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6dmVyc2lvbiA9IHNpemVvZihlZ2xfbmF0aXZlX3dpbmRvd190KTsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjppZGVudCA9IDA7Ci0gICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6aW5jUmVmID0gJkVHTE5hdGl2ZVdpbmRvd1N1cmZhY2U6Omhvb2tfaW5jUmVmOwotICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmRlY1JlZiA9ICZFR0xOYXRpdmVXaW5kb3dTdXJmYWNlOjpob29rX2RlY1JlZjsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190Ojpzd2FwQnVmZmVycyA9ICZFR0xOYXRpdmVXaW5kb3dTdXJmYWNlOjpob29rX3N3YXBCdWZmZXJzOwotICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmNvbm5lY3QgPSAmRUdMTmF0aXZlV2luZG93U3VyZmFjZTo6aG9va19jb25uZWN0OwotICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmRpc2Nvbm5lY3QgPSAmRUdMTmF0aXZlV2luZG93U3VyZmFjZTo6aG9va19kaXNjb25uZWN0OwotICAgIAotICAgIERpc3BsYXlJbmZvIGRpbmZvOwotICAgIFN1cmZhY2VDb21wb3NlckNsaWVudDo6Z2V0RGlzcGxheUluZm8oMCwgJmRpbmZvKTsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190Ojp4ZHBpID0gZGluZm8ueGRwaTsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190Ojp5ZHBpID0gZGluZm8ueWRwaTsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpmcHMgID0gZGluZm8uZnBzOwotICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmZsYWdzPSBFR0xfTkFUSVZFU19GTEFHX0RFU1RST1lfQkFDS0JVRkZFUjsKLX0KLQotRUdMTmF0aXZlV2luZG93U3VyZmFjZTo6fkVHTE5hdGl2ZVdpbmRvd1N1cmZhY2UoKQotewotICAgIGRpc2Nvbm5lY3QoKTsKLSAgICBtU3VyZmFjZS5jbGVhcigpOwotICAgIG1hZ2ljID0gMDsKLX0KLQotdm9pZCBFR0xOYXRpdmVXaW5kb3dTdXJmYWNlOjpob29rX2luY1JlZihOYXRpdmVXaW5kb3dUeXBlIHdpbmRvdykKLXsKLSAgICBFR0xOYXRpdmVXaW5kb3dTdXJmYWNlKiB0aGF0ID0gc3RhdGljX2Nhc3Q8RUdMTmF0aXZlV2luZG93U3VyZmFjZSo+KHdpbmRvdyk7Ci0gICAgdGhhdC0+aW5jU3Ryb25nKHRoYXQpOwotfQotCi12b2lkIEVHTE5hdGl2ZVdpbmRvd1N1cmZhY2U6Omhvb2tfZGVjUmVmKE5hdGl2ZVdpbmRvd1R5cGUgd2luZG93KQotewotICAgIEVHTE5hdGl2ZVdpbmRvd1N1cmZhY2UqIHRoYXQgPSBzdGF0aWNfY2FzdDxFR0xOYXRpdmVXaW5kb3dTdXJmYWNlKj4od2luZG93KTsKLSAgICB0aGF0LT5kZWNTdHJvbmcodGhhdCk7Ci19Ci0KLXZvaWQgRUdMTmF0aXZlV2luZG93U3VyZmFjZTo6aG9va19jb25uZWN0KE5hdGl2ZVdpbmRvd1R5cGUgd2luZG93KQotewotICAgIEVHTE5hdGl2ZVdpbmRvd1N1cmZhY2UqIHRoYXQgPSBzdGF0aWNfY2FzdDxFR0xOYXRpdmVXaW5kb3dTdXJmYWNlKj4od2luZG93KTsKLSAgICB0aGF0LT5jb25uZWN0KCk7Ci19Ci0KLXZvaWQgRUdMTmF0aXZlV2luZG93U3VyZmFjZTo6aG9va19kaXNjb25uZWN0KE5hdGl2ZVdpbmRvd1R5cGUgd2luZG93KQotewotICAgIEVHTE5hdGl2ZVdpbmRvd1N1cmZhY2UqIHRoYXQgPSBzdGF0aWNfY2FzdDxFR0xOYXRpdmVXaW5kb3dTdXJmYWNlKj4od2luZG93KTsKLSAgICB0aGF0LT5kaXNjb25uZWN0KCk7Ci19Ci0KLXVpbnQzMl90IEVHTE5hdGl2ZVdpbmRvd1N1cmZhY2U6Omhvb2tfc3dhcEJ1ZmZlcnMoTmF0aXZlV2luZG93VHlwZSB3aW5kb3cpCi17Ci0gICAgRUdMTmF0aXZlV2luZG93U3VyZmFjZSogdGhhdCA9IHN0YXRpY19jYXN0PEVHTE5hdGl2ZVdpbmRvd1N1cmZhY2UqPih3aW5kb3cpOwotICAgIHJldHVybiB0aGF0LT5zd2FwQnVmZmVycygpOwotfQotCi12b2lkIEVHTE5hdGl2ZVdpbmRvd1N1cmZhY2U6OnNldFN3YXBSZWN0YW5nbGUoaW50IGwsIGludCB0LCBpbnQgdywgaW50IGgpCi17Ci0gICAgbVN1cmZhY2UtPnNldFN3YXBSZWN0YW5nbGUoUmVjdChsLCB0LCBsK3csIHQraCkpOwotfQotCi11aW50MzJfdCBFR0xOYXRpdmVXaW5kb3dTdXJmYWNlOjpzd2FwQnVmZmVycygpCi17Ci0gICAgY29uc3QgaW50IHcgPSBlZ2xfbmF0aXZlX3dpbmRvd190Ojp3aWR0aDsKLSAgICBjb25zdCBpbnQgaCA9IGVnbF9uYXRpdmVfd2luZG93X3Q6OmhlaWdodDsKLSAgICBjb25zdCBzcDxTdXJmYWNlPiYgc3VyZmFjZShtU3VyZmFjZSk7Ci0gICAgU3VyZmFjZTo6U3VyZmFjZUluZm8gaW5mbzsKLSAgICBzdXJmYWNlLT51bmxvY2tBbmRQb3N0KCk7Ci0gICAgc3VyZmFjZS0+bG9jaygmaW5mbyk7Ci0gICAgLy8gdXBkYXRlIHRoZSBhZGRyZXNzIG9mIHRoZSBidWZmZXIgdG8gZHJhdyB0byBuZXh0Ci0gICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6YmFzZSAgID0gaW50cHRyX3QoaW5mby5iYXNlKTsKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190OjpvZmZzZXQgPSBpbnRwdHJfdChpbmZvLmJpdHMpIC0gaW50cHRyX3QoaW5mby5iYXNlKTsKLSAgICAKLSAgICAvLyB1cGRhdGUgc2l6ZSBpZiBpdCBjaGFuZ2VkCi0gICAgaWYgKHcgIT0gaW50KGluZm8udykgfHwgaCAhPSBpbnQoaW5mby5oKSkgewotICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190Ojp3aWR0aCAgPSBpbmZvLnc7Ci0gICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmhlaWdodCA9IGluZm8uaDsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6c3RyaWRlID0gaW5mby5icHIgLyBieXRlc1BlclBpeGVsKGluZm8uZm9ybWF0KTsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6Zm9ybWF0ID0gaW5mby5mb3JtYXQ7Ci0gICAgICAgIHJldHVybiBFR0xfTkFUSVZFU19GTEFHX1NJWkVfQ0hBTkdFRDsKLSAgICB9Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXZvaWQgRUdMTmF0aXZlV2luZG93U3VyZmFjZTo6Y29ubmVjdCgpCi17ICAgCi0gICAgaWYgKCFtQ29ubmVjdGVkKSB7Ci0gICAgICAgIFN1cmZhY2U6OlN1cmZhY2VJbmZvIGluZm87Ci0gICAgICAgIG1TdXJmYWNlLT5sb2NrKCZpbmZvKTsKLSAgICAgICAgbVN1cmZhY2UtPnNldFN3YXBSZWN0YW5nbGUoUmVjdChpbmZvLncsIGluZm8uaCkpOwotICAgICAgICBtQ29ubmVjdGVkID0gdHJ1ZTsKLQotICAgICAgICBlZ2xfbmF0aXZlX3dpbmRvd190Ojp3aWR0aCAgPSBpbmZvLnc7Ci0gICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmhlaWdodCA9IGluZm8uaDsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6c3RyaWRlID0gaW5mby5icHIgLyBieXRlc1BlclBpeGVsKGluZm8uZm9ybWF0KTsKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6Zm9ybWF0ID0gaW5mby5mb3JtYXQ7Ci0gICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6OmJhc2UgICA9IGludHB0cl90KGluZm8uYmFzZSk7Ci0gICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3Q6Om9mZnNldCA9IGludHB0cl90KGluZm8uYml0cykgLSBpbnRwdHJfdChpbmZvLmJhc2UpOwotICAgICAgICAvLyBGSVhNRTogZWdsX25hdGl2ZV93aW5kb3dfdDo6bWVtb3J5X3R5cGUgdXNlZCB0byBiZSBzZXQgZnJvbQotICAgICAgICAvLyBtU3VyZmFjZSwgYnV0IHdlIHdhbnRlZCB0byBicmVhayB0aGlzIGRlcGVuZGVuY3kuIFdlIHNldCBpdCB0bwotICAgICAgICAvLyBHUFUgYmVjYXVzZSB0aGUgc29mdHdhcmUgcmVuZGVyZWQgZG9lc24ndCBjYXJlLCBidXQgdGhlIGgvdwotICAgICAgICAvLyBhY2NlbGVyYXRvciBuZWVkcyBpdC4gRXZlbnR1YWxseSwgdGhpcyB2YWx1ZSBzaG91bGQgZ28gYXdheQotICAgICAgICAvLyBjb21wbGV0ZWx5LCBzaW5jZSBtZW1vcnkgd2lsbCBiZSBtYW5hZ2VkIGJ5IE9wZW5HTC4KLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6bWVtb3J5X3R5cGUgPSBOQVRJVkVfTUVNT1JZX1RZUEVfR1BVOyAKLSAgICAgICAgZWdsX25hdGl2ZV93aW5kb3dfdDo6ZmQgPSAwOwotICAgIH0KLX0KLQotdm9pZCBFR0xOYXRpdmVXaW5kb3dTdXJmYWNlOjpkaXNjb25uZWN0KCkKLXsKLSAgICBpZiAobUNvbm5lY3RlZCkgewotICAgICAgICBtU3VyZmFjZS0+dW5sb2NrKCk7Ci0gICAgICAgIG1Db25uZWN0ZWQgPSBmYWxzZTsKLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCmRpZmYgLS1naXQgYS9saWJzL3VpL0V2ZW50SHViLmNwcCBiL2xpYnMvdWkvRXZlbnRIdWIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzYjI5YjA5Li4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvRXZlbnRIdWIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNzkzICswLDAgQEAKLS8vCi0vLyBDb3B5cmlnaHQgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0vLwotLy8gSGFuZGxlIGV2ZW50cywgbGlrZSBrZXkgaW5wdXQgYW5kIHZzeW5jLgotLy8KLS8vIFRoZSBnb2FsIGlzIHRvIHByb3ZpZGUgYW4gb3B0aW1pemVkIHNvbHV0aW9uIGZvciBMaW51eCwgbm90IGFuCi0vLyBpbXBsZW1lbnRhdGlvbiB0aGF0IHdvcmtzIHdlbGwgYWNyb3NzIGFsbCBwbGF0Zm9ybXMuICBXZSBleHBlY3QKLS8vIGV2ZW50cyB0byBhcnJpdmUgb24gZmlsZSBkZXNjcmlwdG9ycywgc28gdGhhdCB3ZSBjYW4gdXNlIGEgc2VsZWN0KCkKLS8vIHNlbGVjdCgpIGNhbGwgdG8gc2xlZXAuCi0vLwotLy8gV2UgY2FuJ3Qgc2VsZWN0KCkgb24gYW55dGhpbmcgYnV0IG5ldHdvcmsgc29ja2V0cyBpbiBXaW5kb3dzLCBzbyB3ZQotLy8gcHJvdmlkZSBhbiBhbHRlcm5hdGl2ZSBpbXBsZW1lbnRhdGlvbiBvZiB3YWl0RXZlbnQgZm9yIHRoYXQgcGxhdGZvcm0uCi0vLwotI2RlZmluZSBMT0dfVEFHICJFdmVudEh1YiIKLQotLy8jZGVmaW5lIExPR19OREVCVUcgMAotCi0jaW5jbHVkZSA8dWkvRXZlbnRIdWIuaD4KLSNpbmNsdWRlIDxoYXJkd2FyZV9sZWdhY3kvcG93ZXIuaD4KLQotI2luY2x1ZGUgPGN1dGlscy9wcm9wZXJ0aWVzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSVNlcnZpY2VNYW5hZ2VyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvVGltZXJzLmg+Ci0jaW5jbHVkZSA8dXRpbHMuaD4KLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8ZmNudGwuaD4KLSNpbmNsdWRlIDxtZW1vcnkuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPGFzc2VydC5oPgotCi0jaW5jbHVkZSAiS2V5TGF5b3V0TWFwLmgiCi0KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxkaXJlbnQuaD4KLSNpZmRlZiBIQVZFX0lOT1RJRlkKLSMgaW5jbHVkZSA8c3lzL2lub3RpZnkuaD4KLSNlbmRpZgotI2lmZGVmIEhBVkVfQU5EUk9JRF9PUwotIyBpbmNsdWRlIDxzeXMvbGltaXRzLmg+ICAgICAgICAvKiBub3QgcGFydCBvZiBMaW51eCAqLwotI2VuZGlmCi0jaW5jbHVkZSA8c3lzL3BvbGwuaD4KLSNpbmNsdWRlIDxzeXMvaW9jdGwuaD4KLQotLyogdGhpcyBtYWNybyBpcyB1c2VkIHRvIHRlbGwgaWYgImJpdCIgaXMgc2V0IGluICJhcnJheSIKLSAqIGl0IHNlbGVjdHMgYSBieXRlIGZyb20gdGhlIGFycmF5LCBhbmQgZG9lcyBhIGJvb2xlYW4gQU5ECi0gKiBvcGVyYXRpb24gd2l0aCBhIGJ5dGUgdGhhdCBvbmx5IGhhcyB0aGUgcmVsZXZhbnQgYml0IHNldC4KLSAqIGVnLiB0byBjaGVjayBmb3IgdGhlIDEydGggYml0LCB3ZSBkbyAoYXJyYXlbMV0gJiAxPDw0KQotICovCi0jZGVmaW5lIHRlc3RfYml0KGJpdCwgYXJyYXkpICAgIChhcnJheVtiaXQvOF0gJiAoMTw8KGJpdCU4KSkpCi0KLSNkZWZpbmUgSURfTUFTSyAgMHgwMDAwZmZmZgotI2RlZmluZSBTRVFfTUFTSyAweDdmZmYwMDAwCi0jZGVmaW5lIFNFUV9TSElGVCAxNgotI2RlZmluZSBpZF90b19pbmRleChpZCkgICAgICAgICAoKGlkJklEX01BU0spKzEpCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotc3RhdGljIGNvbnN0IGNoYXIgKldBS0VfTE9DS19JRCA9ICJLZXlFdmVudHMiOwotc3RhdGljIGNvbnN0IGNoYXIgKmRldmljZV9wYXRoID0gIi9kZXYvaW5wdXQiOwotCi0vKiByZXR1cm4gdGhlIGxhcmdlciBpbnRlZ2VyICovCi1zdGF0aWMgaW5saW5lIGludCBtYXgoaW50IHYxLCBpbnQgdjIpCi17Ci0gICAgcmV0dXJuICh2MSA+IHYyKSA/IHYxIDogdjI7Ci19Ci0KLUV2ZW50SHViOjpkZXZpY2VfdDo6ZGV2aWNlX3QoaW50MzJfdCBfaWQsIGNvbnN0IGNoYXIqIF9wYXRoKQotICAgIDogaWQoX2lkKSwgcGF0aChfcGF0aCksIGNsYXNzZXMoMCkKLSAgICAsIGtleUJpdG1hc2soTlVMTCksIGxheW91dE1hcChuZXcgS2V5TGF5b3V0TWFwKCkpLCBuZXh0KE5VTEwpIHsKLX0KLQotRXZlbnRIdWI6OmRldmljZV90Ojp+ZGV2aWNlX3QoKSB7Ci0gICAgZGVsZXRlIFtdIGtleUJpdG1hc2s7Ci0gICAgZGVsZXRlIGxheW91dE1hcDsKLX0KLQotRXZlbnRIdWI6OkV2ZW50SHViKHZvaWQpCi0gICAgOiBtRXJyb3IoTk9fSU5JVCksIG1IYXZlRmlyc3RLZXlib2FyZChmYWxzZSksIG1GaXJzdEtleWJvYXJkSWQoMCkKLSAgICAsIG1EZXZpY2VzQnlJZCgwKSwgbU51bURldmljZXNCeUlkKDApCi0gICAgLCBtT3BlbmluZ0RldmljZXMoMCksIG1DbG9zaW5nRGV2aWNlcygwKQotICAgICwgbURldmljZXMoMCksIG1GRHMoMCksIG1GRENvdW50KDApCi17Ci0gICAgYWNxdWlyZV93YWtlX2xvY2soUEFSVElBTF9XQUtFX0xPQ0ssIFdBS0VfTE9DS19JRCk7Ci0jaWZkZWYgRVZfU1cKLSAgICBtZW1zZXQobVN3aXRjaGVzLCAwLCBzaXplb2YobVN3aXRjaGVzKSk7Ci0jZW5kaWYKLX0KLQotLyoKLSAqIENsZWFuIHVwLgotICovCi1FdmVudEh1Yjo6fkV2ZW50SHViKHZvaWQpCi17Ci0gICAgcmVsZWFzZV93YWtlX2xvY2soV0FLRV9MT0NLX0lEKTsKLSAgICAvLyB3ZSBzaG91bGQgZnJlZSBzdHVmZiBoZXJlLi4uCi19Ci0KLXZvaWQgRXZlbnRIdWI6Om9uRmlyc3RSZWYoKQotewotICAgIG1FcnJvciA9IG9wZW5QbGF0Zm9ybUlucHV0KCkgPyBOT19FUlJPUiA6IFVOS05PV05fRVJST1I7Ci19Ci0KLXN0YXR1c190IEV2ZW50SHViOjplcnJvckNoZWNrKCkgY29uc3QKLXsKLSAgICByZXR1cm4gbUVycm9yOwotfQotCi1TdHJpbmc4IEV2ZW50SHViOjpnZXREZXZpY2VOYW1lKGludDMyX3QgZGV2aWNlSWQpIGNvbnN0Ci17Ci0gICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLSAgICBkZXZpY2VfdCogZGV2aWNlID0gZ2V0RGV2aWNlKGRldmljZUlkKTsKLSAgICBpZiAoZGV2aWNlID09IE5VTEwpIHJldHVybiBTdHJpbmc4KCk7Ci0gICAgcmV0dXJuIGRldmljZS0+bmFtZTsKLX0KLQotdWludDMyX3QgRXZlbnRIdWI6OmdldERldmljZUNsYXNzZXMoaW50MzJfdCBkZXZpY2VJZCkgY29uc3QKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgIGRldmljZV90KiBkZXZpY2UgPSBnZXREZXZpY2UoZGV2aWNlSWQpOwotICAgIGlmIChkZXZpY2UgPT0gTlVMTCkgcmV0dXJuIDA7Ci0gICAgcmV0dXJuIGRldmljZS0+Y2xhc3NlczsKLX0KLQotaW50IEV2ZW50SHViOjpnZXRBYnNvbHV0ZUluZm8oaW50MzJfdCBkZXZpY2VJZCwgaW50IGF4aXMsIGludCAqb3V0TWluVmFsdWUsCi0gICAgICAgIGludCogb3V0TWF4VmFsdWUsIGludCogb3V0RmxhdCwgaW50KiBvdXRGdXp6KSBjb25zdAotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgZGV2aWNlX3QqIGRldmljZSA9IGdldERldmljZShkZXZpY2VJZCk7Ci0gICAgaWYgKGRldmljZSA9PSBOVUxMKSByZXR1cm4gLTE7Ci0KLSAgICBzdHJ1Y3QgaW5wdXRfYWJzaW5mbyBpbmZvOwotCi0gICAgaWYoaW9jdGwobUZEc1tpZF90b19pbmRleChkZXZpY2UtPmlkKV0uZmQsIEVWSU9DR0FCUyhheGlzKSwgJmluZm8pKSB7Ci0gICAgICAgIExPR0UoIkVycm9yIHJlYWRpbmcgYWJzb2x1dGUgY29udHJvbGxlciAlZCBmb3IgZGV2aWNlICVzIGZkICVkXG4iLAotICAgICAgICAgICAgIGF4aXMsIGRldmljZS0+bmFtZS5zdHJpbmcoKSwgbUZEc1tpZF90b19pbmRleChkZXZpY2UtPmlkKV0uZmQpOwotICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotICAgICpvdXRNaW5WYWx1ZSA9IGluZm8ubWluaW11bTsKLSAgICAqb3V0TWF4VmFsdWUgPSBpbmZvLm1heGltdW07Ci0gICAgKm91dEZsYXQgPSBpbmZvLmZsYXQ7Ci0gICAgKm91dEZ1enogPSBpbmZvLmZ1eno7Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLWludCBFdmVudEh1Yjo6Z2V0U3dpdGNoU3RhdGUoaW50IHN3KSBjb25zdAotewotI2lmZGVmIEVWX1NXCi0gICAgaWYgKHN3ID49IDAgJiYgc3cgPD0gU1dfTUFYKSB7Ci0gICAgICAgIGludDMyX3QgZGV2aWQgPSBtU3dpdGNoZXNbc3ddOwotICAgICAgICBpZiAoZGV2aWQgIT0gMCkgewotICAgICAgICAgICAgcmV0dXJuIGdldFN3aXRjaFN0YXRlKGRldmlkLCBzdyk7Ci0gICAgICAgIH0KLSAgICB9Ci0jZW5kaWYKLSAgICByZXR1cm4gLTE7Ci19Ci0KLWludCBFdmVudEh1Yjo6Z2V0U3dpdGNoU3RhdGUoaW50MzJfdCBkZXZpY2VJZCwgaW50IHN3KSBjb25zdAotewotI2lmZGVmIEVWX1NXCi0gICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLSAgICBkZXZpY2VfdCogZGV2aWNlID0gZ2V0RGV2aWNlKGRldmljZUlkKTsKLSAgICBpZiAoZGV2aWNlID09IE5VTEwpIHJldHVybiAtMTsKLSAgICAKLSAgICBpZiAoc3cgPj0gMCAmJiBzdyA8PSBTV19NQVgpIHsKLSAgICAgICAgdWludDhfdCBzd19iaXRtYXNrWyhTV19NQVgrMSkvOF07Ci0gICAgICAgIG1lbXNldChzd19iaXRtYXNrLCAwLCBzaXplb2Yoc3dfYml0bWFzaykpOwotICAgICAgICBpZiAoaW9jdGwobUZEc1tpZF90b19pbmRleChkZXZpY2UtPmlkKV0uZmQsCi0gICAgICAgICAgICAgICAgICAgRVZJT0NHU1coc2l6ZW9mKHN3X2JpdG1hc2spKSwgc3dfYml0bWFzaykgPj0gMCkgewotICAgICAgICAgICAgcmV0dXJuIHRlc3RfYml0KHN3LCBzd19iaXRtYXNrKSA/IDEgOiAwOwotICAgICAgICB9Ci0gICAgfQotI2VuZGlmCi0gICAgCi0gICAgcmV0dXJuIC0xOwotfQotCi1pbnQgRXZlbnRIdWI6OmdldFNjYW5jb2RlU3RhdGUoaW50IGNvZGUpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIGdldFNjYW5jb2RlU3RhdGUobUZpcnN0S2V5Ym9hcmRJZCwgY29kZSk7Ci19Ci0KLWludCBFdmVudEh1Yjo6Z2V0U2NhbmNvZGVTdGF0ZShpbnQzMl90IGRldmljZUlkLCBpbnQgY29kZSkgY29uc3QKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgIGRldmljZV90KiBkZXZpY2UgPSBnZXREZXZpY2UoZGV2aWNlSWQpOwotICAgIGlmIChkZXZpY2UgPT0gTlVMTCkgcmV0dXJuIC0xOwotICAgIAotICAgIGlmIChjb2RlID49IDAgJiYgY29kZSA8PSBLRVlfTUFYKSB7Ci0gICAgICAgIHVpbnQ4X3Qga2V5X2JpdG1hc2tbKEtFWV9NQVgrMSkvOF07Ci0gICAgICAgIG1lbXNldChrZXlfYml0bWFzaywgMCwgc2l6ZW9mKGtleV9iaXRtYXNrKSk7Ci0gICAgICAgIGlmIChpb2N0bChtRkRzW2lkX3RvX2luZGV4KGRldmljZS0+aWQpXS5mZCwKLSAgICAgICAgICAgICAgICAgICBFVklPQ0dLRVkoc2l6ZW9mKGtleV9iaXRtYXNrKSksIGtleV9iaXRtYXNrKSA+PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gdGVzdF9iaXQoY29kZSwga2V5X2JpdG1hc2spID8gMSA6IDA7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgcmV0dXJuIC0xOwotfQotCi1pbnQgRXZlbnRIdWI6OmdldEtleWNvZGVTdGF0ZShpbnQgY29kZSkgY29uc3QKLXsKLSAgICByZXR1cm4gZ2V0S2V5Y29kZVN0YXRlKG1GaXJzdEtleWJvYXJkSWQsIGNvZGUpOwotfQotCi1pbnQgRXZlbnRIdWI6OmdldEtleWNvZGVTdGF0ZShpbnQzMl90IGRldmljZUlkLCBpbnQgY29kZSkgY29uc3QKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgIGRldmljZV90KiBkZXZpY2UgPSBnZXREZXZpY2UoZGV2aWNlSWQpOwotICAgIGlmIChkZXZpY2UgPT0gTlVMTCB8fCBkZXZpY2UtPmxheW91dE1hcCA9PSBOVUxMKSByZXR1cm4gLTE7Ci0gICAgCi0gICAgVmVjdG9yPGludDMyX3Q+IHNjYW5Db2RlczsKLSAgICBkZXZpY2UtPmxheW91dE1hcC0+ZmluZFNjYW5jb2Rlcyhjb2RlLCAmc2NhbkNvZGVzKTsKLSAgICAKLSAgICB1aW50OF90IGtleV9iaXRtYXNrWyhLRVlfTUFYKzEpLzhdOwotICAgIG1lbXNldChrZXlfYml0bWFzaywgMCwgc2l6ZW9mKGtleV9iaXRtYXNrKSk7Ci0gICAgaWYgKGlvY3RsKG1GRHNbaWRfdG9faW5kZXgoZGV2aWNlLT5pZCldLmZkLAotICAgICAgICAgICAgICAgRVZJT0NHS0VZKHNpemVvZihrZXlfYml0bWFzaykpLCBrZXlfYml0bWFzaykgPj0gMCkgewotICAgICAgICAjaWYgMAotICAgICAgICBmb3IgKHNpemVfdCBpPTA7IGk8PUtFWV9NQVg7IGkrKykgewotICAgICAgICAgICAgTE9HSSgiKFNjYW4gY29kZSAlZDogZG93bj0lZCkiLCBpLCB0ZXN0X2JpdChpLCBrZXlfYml0bWFzaykpOwotICAgICAgICB9Ci0gICAgICAgICNlbmRpZgotICAgICAgICBjb25zdCBzaXplX3QgTiA9IHNjYW5Db2Rlcy5zaXplKCk7Ci0gICAgICAgIGZvciAoc2l6ZV90IGk9MDsgaTxOICYmIGk8PUtFWV9NQVg7IGkrKykgewotICAgICAgICAgICAgaW50MzJfdCBzYyA9IHNjYW5Db2Rlcy5pdGVtQXQoaSk7Ci0gICAgICAgICAgICAvL0xPR0koIkNvZGUgJWQ6IGRvd249JWQiLCBzYywgdGVzdF9iaXQoc2MsIGtleV9iaXRtYXNrKSk7Ci0gICAgICAgICAgICBpZiAoc2MgPj0gMCAmJiBzYyA8PSBLRVlfTUFYICYmIHRlc3RfYml0KHNjLCBrZXlfYml0bWFzaykpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gMTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICByZXR1cm4gMDsKLX0KLQotRXZlbnRIdWI6OmRldmljZV90KiBFdmVudEh1Yjo6Z2V0RGV2aWNlKGludDMyX3QgZGV2aWNlSWQpIGNvbnN0Ci17Ci0gICAgaWYgKGRldmljZUlkID09IDApIGRldmljZUlkID0gbUZpcnN0S2V5Ym9hcmRJZDsKLSAgICBpbnQzMl90IGlkID0gZGV2aWNlSWQgJiBJRF9NQVNLOwotICAgIGlmIChpZCA+PSBtTnVtRGV2aWNlc0J5SWQgfHwgaWQgPCAwKSByZXR1cm4gTlVMTDsKLSAgICBkZXZpY2VfdCogZGV2ID0gbURldmljZXNCeUlkW2lkXS5kZXZpY2U7Ci0gICAgaWYgKGRldi0+aWQgPT0gZGV2aWNlSWQpIHsKLSAgICAgICAgcmV0dXJuIGRldjsKLSAgICB9Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLWJvb2wgRXZlbnRIdWI6OmdldEV2ZW50KGludDMyX3QqIG91dERldmljZUlkLCBpbnQzMl90KiBvdXRUeXBlLAotICAgICAgICBpbnQzMl90KiBvdXRTY2FuY29kZSwgaW50MzJfdCogb3V0S2V5Y29kZSwgdWludDMyX3QgKm91dEZsYWdzLAotICAgICAgICBpbnQzMl90KiBvdXRWYWx1ZSwgbnNlY3NfdCogb3V0V2hlbikKLXsKLSAgICAqb3V0RGV2aWNlSWQgPSAwOwotICAgICpvdXRUeXBlID0gMDsKLSAgICAqb3V0U2NhbmNvZGUgPSAwOwotICAgICpvdXRLZXljb2RlID0gMDsKLSAgICAqb3V0RmxhZ3MgPSAwOwotICAgICpvdXRWYWx1ZSA9IDA7Ci0gICAgKm91dFdoZW4gPSAwOwotCi0gICAgc3RhdHVzX3QgZXJyOwotCi0gICAgZmRfc2V0IHJlYWRmZHM7Ci0gICAgaW50IG1heEZkID0gLTE7Ci0gICAgaW50IGNjOwotICAgIGludCBpOwotICAgIGludCByZXM7Ci0gICAgaW50IHBvbGxyZXM7Ci0gICAgc3RydWN0IGlucHV0X2V2ZW50IGlldjsKLQotICAgIC8vIE5vdGUgdGhhdCB3ZSBvbmx5IGFsbG93IG9uZSBjYWxsZXIgdG8gZ2V0RXZlbnQoKSwgc28gZG9uJ3QgbmVlZAotICAgIC8vIHRvIGRvIGxvY2tpbmcgaGVyZS4uLiAgb25seSB3aGVuIGFkZGluZy9yZW1vdmluZyBkZXZpY2VzLgotICAgIAotICAgIHdoaWxlKDEpIHsKLQotICAgICAgICAvLyBGaXJzdCwgcmVwb3J0IGFueSBkZXZpY2VzIHRoYXQgaGFkIGxhc3QgYmVlbiBhZGRlZC9yZW1vdmVkLgotICAgICAgICBpZiAobUNsb3NpbmdEZXZpY2VzICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGRldmljZV90KiBkZXZpY2UgPSBtQ2xvc2luZ0RldmljZXM7Ci0gICAgICAgICAgICBMT0dWKCJSZXBvcnRpbmcgZGV2aWNlIGNsb3NlZDogaWQ9MHgleCwgbmFtZT0lc1xuIiwKLSAgICAgICAgICAgICAgICAgZGV2aWNlLT5pZCwgZGV2aWNlLT5wYXRoLnN0cmluZygpKTsKLSAgICAgICAgICAgIG1DbG9zaW5nRGV2aWNlcyA9IGRldmljZS0+bmV4dDsKLSAgICAgICAgICAgICpvdXREZXZpY2VJZCA9IGRldmljZS0+aWQ7Ci0gICAgICAgICAgICBpZiAoKm91dERldmljZUlkID09IG1GaXJzdEtleWJvYXJkSWQpICpvdXREZXZpY2VJZCA9IDA7Ci0gICAgICAgICAgICAqb3V0VHlwZSA9IERFVklDRV9SRU1PVkVEOwotICAgICAgICAgICAgZGVsZXRlIGRldmljZTsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChtT3BlbmluZ0RldmljZXMgIT0gTlVMTCkgewotICAgICAgICAgICAgZGV2aWNlX3QqIGRldmljZSA9IG1PcGVuaW5nRGV2aWNlczsKLSAgICAgICAgICAgIExPR1YoIlJlcG9ydGluZyBkZXZpY2Ugb3BlbmVkOiBpZD0weCV4LCBuYW1lPSVzXG4iLAotICAgICAgICAgICAgICAgICBkZXZpY2UtPmlkLCBkZXZpY2UtPnBhdGguc3RyaW5nKCkpOwotICAgICAgICAgICAgbU9wZW5pbmdEZXZpY2VzID0gZGV2aWNlLT5uZXh0OwotICAgICAgICAgICAgKm91dERldmljZUlkID0gZGV2aWNlLT5pZDsKLSAgICAgICAgICAgIGlmICgqb3V0RGV2aWNlSWQgPT0gbUZpcnN0S2V5Ym9hcmRJZCkgKm91dERldmljZUlkID0gMDsKLSAgICAgICAgICAgICpvdXRUeXBlID0gREVWSUNFX0FEREVEOwotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICByZWxlYXNlX3dha2VfbG9jayhXQUtFX0xPQ0tfSUQpOwotCi0gICAgICAgIHBvbGxyZXMgPSBwb2xsKG1GRHMsIG1GRENvdW50LCAtMSk7Ci0KLSAgICAgICAgYWNxdWlyZV93YWtlX2xvY2soUEFSVElBTF9XQUtFX0xPQ0ssIFdBS0VfTE9DS19JRCk7Ci0KLSAgICAgICAgaWYgKHBvbGxyZXMgPD0gMCkgewotICAgICAgICAgICAgaWYgKGVycm5vICE9IEVJTlRSKSB7Ci0gICAgICAgICAgICAgICAgTE9HVygic2VsZWN0IGZhaWxlZCAoZXJybm89JWQpXG4iLCBlcnJubyk7Ci0gICAgICAgICAgICAgICAgdXNsZWVwKDEwMDAwMCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vcHJpbnRmKCJwb2xsICVkLCByZXR1cm5lZCAlZFxuIiwgbUZEQ291bnQsIHBvbGxyZXMpOwotCi0gICAgICAgIC8vIG1GRHNbMF0gaXMgdXNlZCBmb3IgaW5vdGlmeSwgc28gcHJvY2VzcyByZWd1bGFyIGV2ZW50cyBzdGFydGluZyBhdCBtRkRzWzFdCi0gICAgICAgIGZvcihpID0gMTsgaSA8IG1GRENvdW50OyBpKyspIHsKLSAgICAgICAgICAgIGlmKG1GRHNbaV0ucmV2ZW50cykgewotICAgICAgICAgICAgICAgIExPR1YoInJldmVudHMgZm9yICVkID0gMHglMDh4IiwgaSwgbUZEc1tpXS5yZXZlbnRzKTsKLSAgICAgICAgICAgICAgICBpZihtRkRzW2ldLnJldmVudHMgJiBQT0xMSU4pIHsKLSAgICAgICAgICAgICAgICAgICAgcmVzID0gcmVhZChtRkRzW2ldLmZkLCAmaWV2LCBzaXplb2YoaWV2KSk7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChyZXMgPT0gc2l6ZW9mKGlldikpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIExPR1YoIiVzIGdvdDogdDA9JWQsIHQxPSVkLCB0eXBlPSVkLCBjb2RlPSVkLCB2PSVkIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbURldmljZXNbaV0tPnBhdGguc3RyaW5nKCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIChpbnQpIGlldi50aW1lLnR2X3NlYywgKGludCkgaWV2LnRpbWUudHZfdXNlYywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWV2LnR5cGUsIGlldi5jb2RlLCBpZXYudmFsdWUpOwotICAgICAgICAgICAgICAgICAgICAgICAgKm91dERldmljZUlkID0gbURldmljZXNbaV0tPmlkOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCpvdXREZXZpY2VJZCA9PSBtRmlyc3RLZXlib2FyZElkKSAqb3V0RGV2aWNlSWQgPSAwOwotICAgICAgICAgICAgICAgICAgICAgICAgKm91dFR5cGUgPSBpZXYudHlwZTsKLSAgICAgICAgICAgICAgICAgICAgICAgICpvdXRTY2FuY29kZSA9IGlldi5jb2RlOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGlldi50eXBlID09IEVWX0tFWSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGVyciA9IG1EZXZpY2VzW2ldLT5sYXlvdXRNYXAtPm1hcChpZXYuY29kZSwgb3V0S2V5Y29kZSwgb3V0RmxhZ3MpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIExPR1YoImlldi5jb2RlPSVkIG91dEtleWNvZGU9JWQgb3V0RmxhZ3M9MHglMDh4IGVycj0lZFxuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWV2LmNvZGUsICpvdXRLZXljb2RlLCAqb3V0RmxhZ3MsIGVycik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGVyciAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpvdXRLZXljb2RlID0gMDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKm91dEZsYWdzID0gMDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICpvdXRLZXljb2RlID0gaWV2LmNvZGU7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAqb3V0VmFsdWUgPSBpZXYudmFsdWU7Ci0gICAgICAgICAgICAgICAgICAgICAgICAqb3V0V2hlbiA9IHMybnMoaWV2LnRpbWUudHZfc2VjKSArIHVzMm5zKGlldi50aW1lLnR2X3VzZWMpOwotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAocmVzPDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBMT0dXKCJjb3VsZCBub3QgZ2V0IGV2ZW50IChlcnJubz0lZCkiLCBlcnJubyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIExPR0UoImNvdWxkIG5vdCBnZXQgZXZlbnQgKHdyb25nIHNpemU6ICVkKSIsIHJlcyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgLy8gcmVhZF9ub3RpZnkoKSB3aWxsIG1vZGlmeSBtRkRzIGFuZCBtRkRDb3VudCwgc28gdGhpcyBtdXN0IGJlIGRvbmUgYWZ0ZXIKLSAgICAgICAgLy8gcHJvY2Vzc2luZyBhbGwgb3RoZXIgZXZlbnRzLgotICAgICAgICBpZihtRkRzWzBdLnJldmVudHMgJiBQT0xMSU4pIHsKLSAgICAgICAgICAgIHJlYWRfbm90aWZ5KG1GRHNbMF0uZmQpOwotICAgICAgICB9Ci0gICAgfQotfQotCi0vKgotICogT3BlbiB0aGUgcGxhdGZvcm0tc3BlY2lmaWMgaW5wdXQgZGV2aWNlLgotICovCi1ib29sIEV2ZW50SHViOjpvcGVuUGxhdGZvcm1JbnB1dCh2b2lkKQotewotICAgIC8qCi0gICAgICogT3BlbiBwbGF0Zm9ybS1zcGVjaWZpYyBpbnB1dCBkZXZpY2UocykuCi0gICAgICovCi0gICAgaW50IHJlczsKLQotICAgIG1GRENvdW50ID0gMTsKLSAgICBtRkRzID0gKHBvbGxmZCAqKWNhbGxvYygxLCBzaXplb2YobUZEc1swXSkpOwotICAgIG1EZXZpY2VzID0gKGRldmljZV90ICoqKWNhbGxvYygxLCBzaXplb2YobURldmljZXNbMF0pKTsKLSAgICBtRkRzWzBdLmV2ZW50cyA9IFBPTExJTjsKLSAgICBtRGV2aWNlc1swXSA9IE5VTEw7Ci0jaWZkZWYgSEFWRV9JTk9USUZZCi0gICAgbUZEc1swXS5mZCA9IGlub3RpZnlfaW5pdCgpOwotICAgIHJlcyA9IGlub3RpZnlfYWRkX3dhdGNoKG1GRHNbMF0uZmQsIGRldmljZV9wYXRoLCBJTl9ERUxFVEUgfCBJTl9DUkVBVEUpOwotICAgIGlmKHJlcyA8IDApIHsKLSAgICAgICAgTE9HRSgiY291bGQgbm90IGFkZCB3YXRjaCBmb3IgJXMsICVzXG4iLCBkZXZpY2VfcGF0aCwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICB9Ci0jZWxzZQotICAgIC8qCi0gICAgICogVGhlIGNvZGUgaW4gRXZlbnRIdWI6OmdldEV2ZW50IGFzc3VtZXMgdGhhdCBtRkRzWzBdIGlzIGFuIGlub3RpZnkgZmQuCi0gICAgICogV2UgYWxsb2NhdGUgc3BhY2UgZm9yIGl0IGFuZCBzZXQgaXQgdG8gc29tZXRoaW5nIGludmFsaWQuCi0gICAgICovCi0gICAgbUZEc1swXS5mZCA9IC0xOwotI2VuZGlmCi0KLSAgICByZXMgPSBzY2FuX2RpcihkZXZpY2VfcGF0aCk7Ci0gICAgaWYocmVzIDwgMCkgewotICAgICAgICBMT0dFKCJzY2FuIGRpciBmYWlsZWQgZm9yICVzXG4iLCBkZXZpY2VfcGF0aCk7Ci0gICAgICAgIC8vb3Blbl9kZXZpY2UoIi9kZXYvaW5wdXQvZXZlbnQwIik7Ci0gICAgfQotCi0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLS8qCi0gKiBJbnNwZWN0IHRoZSBrbm93biBkZXZpY2VzIHRvIGRldGVybWluZSB3aGV0aGVyIHBoeXNpY2FsIGtleXMgZXhpc3QgZm9yIHRoZSBnaXZlbgotICogZnJhbWV3b3JrLWRvbWFpbiBrZXkgY29kZXMuCi0gKi8KLWJvb2wgRXZlbnRIdWI6Omhhc0tleXMoc2l6ZV90IG51bUNvZGVzLCBpbnQzMl90KiBrZXlDb2RlcywgdWludDhfdCogb3V0RmxhZ3MpIHsKLSAgICBmb3IgKHNpemVfdCBjb2RlSW5kZXggPSAwOyBjb2RlSW5kZXggPCBudW1Db2RlczsgY29kZUluZGV4KyspIHsKLSAgICAgICAgb3V0RmxhZ3NbY29kZUluZGV4XSA9IDA7Ci0KLSAgICAgICAgLy8gY2hlY2sgZWFjaCBhdmFpbGFibGUgaGFyZHdhcmUgZGV2aWNlIGZvciBzdXBwb3J0IGZvciB0aGlzIGtleWNvZGUKLSAgICAgICAgVmVjdG9yPGludDMyX3Q+IHNjYW5Db2RlczsKLSAgICAgICAgZm9yIChpbnQgbiA9IDA7IChuIDwgbUZEQ291bnQpICYmIChvdXRGbGFnc1tjb2RlSW5kZXhdID09IDApOyBuKyspIHsKLSAgICAgICAgICAgIGlmIChtRGV2aWNlc1tuXSkgewotICAgICAgICAgICAgICAgIHN0YXR1c190IGVyciA9IG1EZXZpY2VzW25dLT5sYXlvdXRNYXAtPmZpbmRTY2FuY29kZXMoa2V5Q29kZXNbY29kZUluZGV4XSwgJnNjYW5Db2Rlcyk7Ci0gICAgICAgICAgICAgICAgaWYgKCFlcnIpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gY2hlY2sgdGhlIHBvc3NpYmxlIHNjYW4gY29kZXMgaWRlbnRpZmllZCBieSB0aGUgbGF5b3V0IG1hcCBhZ2FpbnN0IHRoZQotICAgICAgICAgICAgICAgICAgICAvLyBtYXAgb2YgY29kZXMgYWN0dWFsbHkgZW1pdHRlZCBieSB0aGUgZHJpdmVyCi0gICAgICAgICAgICAgICAgICAgIGZvciAoc2l6ZV90IHNjID0gMDsgc2MgPCBzY2FuQ29kZXMuc2l6ZSgpOyBzYysrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAodGVzdF9iaXQoc2NhbkNvZGVzW3NjXSwgbURldmljZXNbbl0tPmtleUJpdG1hc2spKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgb3V0RmxhZ3NbY29kZUluZGV4XSA9IDE7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1pbnQgRXZlbnRIdWI6Om9wZW5fZGV2aWNlKGNvbnN0IGNoYXIgKmRldmljZU5hbWUpCi17Ci0gICAgaW50IHZlcnNpb247Ci0gICAgaW50IGZkOwotICAgIHN0cnVjdCBwb2xsZmQgKm5ld19tRkRzOwotICAgIGRldmljZV90ICoqbmV3X2RldmljZXM7Ci0gICAgY2hhciAqKm5ld19kZXZpY2VfbmFtZXM7Ci0gICAgY2hhciBuYW1lWzgwXTsKLSAgICBjaGFyIGxvY2F0aW9uWzgwXTsKLSAgICBjaGFyIGlkc3RyWzgwXTsKLSAgICBzdHJ1Y3QgaW5wdXRfaWQgaWQ7Ci0KLSAgICBMT0dWKCJPcGVuaW5nIGRldmljZTogJXMiLCBkZXZpY2VOYW1lKTsKLQotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgCi0gICAgZmQgPSBvcGVuKGRldmljZU5hbWUsIE9fUkRXUik7Ci0gICAgaWYoZmQgPCAwKSB7Ci0gICAgICAgIExPR0UoImNvdWxkIG5vdCBvcGVuICVzLCAlc1xuIiwgZGV2aWNlTmFtZSwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgcmV0dXJuIC0xOwotICAgIH0KLQotICAgIGlmKGlvY3RsKGZkLCBFVklPQ0dWRVJTSU9OLCAmdmVyc2lvbikpIHsKLSAgICAgICAgTE9HRSgiY291bGQgbm90IGdldCBkcml2ZXIgdmVyc2lvbiBmb3IgJXMsICVzXG4iLCBkZXZpY2VOYW1lLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotICAgIGlmKGlvY3RsKGZkLCBFVklPQ0dJRCwgJmlkKSkgewotICAgICAgICBMT0dFKCJjb3VsZCBub3QgZ2V0IGRyaXZlciBpZCBmb3IgJXMsICVzXG4iLCBkZXZpY2VOYW1lLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotICAgIG5hbWVbc2l6ZW9mKG5hbWUpIC0gMV0gPSAnXDAnOwotICAgIGxvY2F0aW9uW3NpemVvZihsb2NhdGlvbikgLSAxXSA9ICdcMCc7Ci0gICAgaWRzdHJbc2l6ZW9mKGlkc3RyKSAtIDFdID0gJ1wwJzsKLSAgICBpZihpb2N0bChmZCwgRVZJT0NHTkFNRShzaXplb2YobmFtZSkgLSAxKSwgJm5hbWUpIDwgMSkgewotICAgICAgICAvL2ZwcmludGYoc3RkZXJyLCAiY291bGQgbm90IGdldCBkZXZpY2UgbmFtZSBmb3IgJXMsICVzXG4iLCBkZXZpY2VOYW1lLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICBuYW1lWzBdID0gJ1wwJzsKLSAgICB9Ci0gICAgaWYoaW9jdGwoZmQsIEVWSU9DR1BIWVMoc2l6ZW9mKGxvY2F0aW9uKSAtIDEpLCAmbG9jYXRpb24pIDwgMSkgewotICAgICAgICAvL2ZwcmludGYoc3RkZXJyLCAiY291bGQgbm90IGdldCBsb2NhdGlvbiBmb3IgJXMsICVzXG4iLCBkZXZpY2VOYW1lLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICBsb2NhdGlvblswXSA9ICdcMCc7Ci0gICAgfQotICAgIGlmKGlvY3RsKGZkLCBFVklPQ0dVTklRKHNpemVvZihpZHN0cikgLSAxKSwgJmlkc3RyKSA8IDEpIHsKLSAgICAgICAgLy9mcHJpbnRmKHN0ZGVyciwgImNvdWxkIG5vdCBnZXQgaWRzdHJpbmcgZm9yICVzLCAlc1xuIiwgZGV2aWNlTmFtZSwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgaWRzdHJbMF0gPSAnXDAnOwotICAgIH0KLQotICAgIGludCBkZXZpZCA9IDA7Ci0gICAgd2hpbGUgKGRldmlkIDwgbU51bURldmljZXNCeUlkKSB7Ci0gICAgICAgIGlmIChtRGV2aWNlc0J5SWRbZGV2aWRdLmRldmljZSA9PSBOVUxMKSB7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgICAgICBkZXZpZCsrOwotICAgIH0KLSAgICBpZiAoZGV2aWQgPj0gbU51bURldmljZXNCeUlkKSB7Ci0gICAgICAgIGRldmljZV9lbnQqIG5ld19kZXZpZHMgPSAoZGV2aWNlX2VudCopcmVhbGxvYyhtRGV2aWNlc0J5SWQsCi0gICAgICAgICAgICAgICAgc2l6ZW9mKG1EZXZpY2VzQnlJZFswXSkgKiAoZGV2aWQgKyAxKSk7Ci0gICAgICAgIGlmIChuZXdfZGV2aWRzID09IE5VTEwpIHsKLSAgICAgICAgICAgIExPR0UoIm91dCBvZiBtZW1vcnkiKTsKLSAgICAgICAgICAgIHJldHVybiAtMTsKLSAgICAgICAgfQotICAgICAgICBtRGV2aWNlc0J5SWQgPSBuZXdfZGV2aWRzOwotICAgICAgICBtTnVtRGV2aWNlc0J5SWQgPSBkZXZpZCsxOwotICAgICAgICBtRGV2aWNlc0J5SWRbZGV2aWRdLmRldmljZSA9IE5VTEw7Ci0gICAgICAgIG1EZXZpY2VzQnlJZFtkZXZpZF0uc2VxID0gMDsKLSAgICB9Ci0KLSAgICBtRGV2aWNlc0J5SWRbZGV2aWRdLnNlcSA9IChtRGV2aWNlc0J5SWRbZGV2aWRdLnNlcSsoMTw8U0VRX1NISUZUKSkmU0VRX01BU0s7Ci0gICAgaWYgKG1EZXZpY2VzQnlJZFtkZXZpZF0uc2VxID09IDApIHsKLSAgICAgICAgbURldmljZXNCeUlkW2RldmlkXS5zZXEgPSAxPDxTRVFfU0hJRlQ7Ci0gICAgfQotCi0gICAgbmV3X21GRHMgPSAocG9sbGZkKilyZWFsbG9jKG1GRHMsIHNpemVvZihtRkRzWzBdKSAqIChtRkRDb3VudCArIDEpKTsKLSAgICBuZXdfZGV2aWNlcyA9IChkZXZpY2VfdCoqKXJlYWxsb2MobURldmljZXMsIHNpemVvZihtRGV2aWNlc1swXSkgKiAobUZEQ291bnQgKyAxKSk7Ci0gICAgaWYgKG5ld19tRkRzID09IE5VTEwgfHwgbmV3X2RldmljZXMgPT0gTlVMTCkgewotICAgICAgICBMT0dFKCJvdXQgb2YgbWVtb3J5Iik7Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0gICAgbUZEcyA9IG5ld19tRkRzOwotICAgIG1EZXZpY2VzID0gbmV3X2RldmljZXM7Ci0KLSNpZiAwCi0gICAgTE9HSSgiYWRkIGRldmljZSAlZDogJXNcbiIsIG1GRENvdW50LCBkZXZpY2VOYW1lKTsKLSAgICBMT0dJKCIgIGJ1czogICAgICAlMDR4XG4iCi0gICAgICAgICAiICB2ZW5kb3IgICAgJTA0eFxuIgotICAgICAgICAgIiAgcHJvZHVjdCAgICUwNHhcbiIKLSAgICAgICAgICIgIHZlcnNpb24gICAlMDR4XG4iLAotICAgICAgICBpZC5idXN0eXBlLCBpZC52ZW5kb3IsIGlkLnByb2R1Y3QsIGlkLnZlcnNpb24pOwotICAgIExPR0koIiAgbmFtZTogICAgIFwiJXNcIlxuIiwgbmFtZSk7Ci0gICAgTE9HSSgiICBsb2NhdGlvbjogXCIlc1wiXG4iCi0gICAgICAgICAiICBpZDogICAgICAgXCIlc1wiXG4iLCBsb2NhdGlvbiwgaWRzdHIpOwotICAgIExPR0koIiAgdmVyc2lvbjogICVkLiVkLiVkXG4iLAotICAgICAgICB2ZXJzaW9uID4+IDE2LCAodmVyc2lvbiA+PiA4KSAmIDB4ZmYsIHZlcnNpb24gJiAweGZmKTsKLSNlbmRpZgotCi0gICAgZGV2aWNlX3QqIGRldmljZSA9IG5ldyBkZXZpY2VfdChkZXZpZHxtRGV2aWNlc0J5SWRbZGV2aWRdLnNlcSwgZGV2aWNlTmFtZSk7Ci0gICAgaWYgKGRldmljZSA9PSBOVUxMKSB7Ci0gICAgICAgIExPR0UoIm91dCBvZiBtZW1vcnkiKTsKLSAgICAgICAgcmV0dXJuIC0xOwotICAgIH0KLQotICAgIG1GRHNbbUZEQ291bnRdLmZkID0gZmQ7Ci0gICAgbUZEc1ttRkRDb3VudF0uZXZlbnRzID0gUE9MTElOOwotCi0gICAgLy8gZmlndXJlIG91dCB0aGUga2luZHMgb2YgZXZlbnRzIHRoZSBkZXZpY2UgcmVwb3J0cwotICAgIHVpbnQ4X3Qga2V5X2JpdG1hc2tbKEtFWV9NQVgrMSkvOF07Ci0gICAgbWVtc2V0KGtleV9iaXRtYXNrLCAwLCBzaXplb2Yoa2V5X2JpdG1hc2spKTsKLSAgICBMT0dWKCJHZXR0aW5nIGtleXMuLi4iKTsKLSAgICBpZiAoaW9jdGwoZmQsIEVWSU9DR0JJVChFVl9LRVksIHNpemVvZihrZXlfYml0bWFzaykpLCBrZXlfYml0bWFzaykgPj0gMCkgewotICAgICAgICAvL0xPR0koIk1BUFxuIik7Ci0gICAgICAgIC8vZm9yIChpbnQgaT0wOyBpPCgoS0VZX01BWCsxKS84KTsgaSsrKSB7Ci0gICAgICAgIC8vICAgIExPR0koIiVkOiAweCUwMnhcbiIsIGksIGtleV9iaXRtYXNrW2ldKTsKLSAgICAgICAgLy99Ci0gICAgICAgIGZvciAoaW50IGk9MDsgaTwoKEJUTl9NSVNDKzcpLzgpOyBpKyspIHsKLSAgICAgICAgICAgIGlmIChrZXlfYml0bWFza1tpXSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgZGV2aWNlLT5jbGFzc2VzIHw9IENMQVNTX0tFWUJPQVJEOwotICAgICAgICAgICAgICAgIC8vICdRJyBrZXkgc3VwcG9ydCA9IGNoZWFwIHRlc3Qgb2Ygd2hldGhlciB0aGlzIGlzIGFuIGFscGhhLWNhcGFibGUga2JkCi0gICAgICAgICAgICAgICAgaWYgKHRlc3RfYml0KEtFWV9RLCBrZXlfYml0bWFzaykpIHsKLSAgICAgICAgICAgICAgICAgICAgZGV2aWNlLT5jbGFzc2VzIHw9IENMQVNTX0FMUEhBS0VZOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBpZiAoKGRldmljZS0+Y2xhc3NlcyAmIENMQVNTX0tFWUJPQVJEKSAhPSAwKSB7Ci0gICAgICAgICAgICBkZXZpY2UtPmtleUJpdG1hc2sgPSBuZXcgdWludDhfdFsoS0VZX01BWCsxKS84XTsKLSAgICAgICAgICAgIGlmIChkZXZpY2UtPmtleUJpdG1hc2sgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgIG1lbWNweShkZXZpY2UtPmtleUJpdG1hc2ssIGtleV9iaXRtYXNrLCBzaXplb2Yoa2V5X2JpdG1hc2spKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZGVsZXRlIGRldmljZTsKLSAgICAgICAgICAgICAgICBMT0dFKCJvdXQgb2YgbWVtb3J5IGFsbG9jYXRpbmcga2V5IGJpdG1hc2siKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gLTE7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgaWYgKHRlc3RfYml0KEJUTl9NT1VTRSwga2V5X2JpdG1hc2spKSB7Ci0gICAgICAgIHVpbnQ4X3QgcmVsX2JpdG1hc2tbKFJFTF9NQVgrMSkvOF07Ci0gICAgICAgIG1lbXNldChyZWxfYml0bWFzaywgMCwgc2l6ZW9mKHJlbF9iaXRtYXNrKSk7Ci0gICAgICAgIExPR1YoIkdldHRpbmcgcmVsYXRpdmUgY29udHJvbGxlcnMuLi4iKTsKLSAgICAgICAgaWYgKGlvY3RsKGZkLCBFVklPQ0dCSVQoRVZfUkVMLCBzaXplb2YocmVsX2JpdG1hc2spKSwgcmVsX2JpdG1hc2spID49IDApCi0gICAgICAgIHsKLSAgICAgICAgICAgIGlmICh0ZXN0X2JpdChSRUxfWCwgcmVsX2JpdG1hc2spICYmIHRlc3RfYml0KFJFTF9ZLCByZWxfYml0bWFzaykpIHsKLSAgICAgICAgICAgICAgICBkZXZpY2UtPmNsYXNzZXMgfD0gQ0xBU1NfVFJBQ0tCQUxMOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIGlmICh0ZXN0X2JpdChCVE5fVE9VQ0gsIGtleV9iaXRtYXNrKSkgewotICAgICAgICB1aW50OF90IGFic19iaXRtYXNrWyhBQlNfTUFYKzEpLzhdOwotICAgICAgICBtZW1zZXQoYWJzX2JpdG1hc2ssIDAsIHNpemVvZihhYnNfYml0bWFzaykpOwotICAgICAgICBMT0dWKCJHZXR0aW5nIGFic29sdXRlIGNvbnRyb2xsZXJzLi4uIik7Ci0gICAgICAgIGlmIChpb2N0bChmZCwgRVZJT0NHQklUKEVWX0FCUywgc2l6ZW9mKGFic19iaXRtYXNrKSksIGFic19iaXRtYXNrKSA+PSAwKQotICAgICAgICB7Ci0gICAgICAgICAgICBpZiAodGVzdF9iaXQoQUJTX1gsIGFic19iaXRtYXNrKSAmJiB0ZXN0X2JpdChBQlNfWSwgYWJzX2JpdG1hc2spKSB7Ci0gICAgICAgICAgICAgICAgZGV2aWNlLT5jbGFzc2VzIHw9IENMQVNTX1RPVUNIU0NSRUVOOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0jaWZkZWYgRVZfU1cKLSAgICAvLyBmaWd1cmUgb3V0IHRoZSBzd2l0Y2hlcyB0aGlzIGRldmljZSByZXBvcnRzCi0gICAgdWludDhfdCBzd19iaXRtYXNrWyhTV19NQVgrMSkvOF07Ci0gICAgbWVtc2V0KHN3X2JpdG1hc2ssIDAsIHNpemVvZihzd19iaXRtYXNrKSk7Ci0gICAgaWYgKGlvY3RsKGZkLCBFVklPQ0dCSVQoRVZfU1csIHNpemVvZihzd19iaXRtYXNrKSksIHN3X2JpdG1hc2spID49IDApIHsKLSAgICAgICAgZm9yIChpbnQgaT0wOyBpPEVWX1NXOyBpKyspIHsKLSAgICAgICAgICAgIC8vTE9HSSgiRGV2aWNlIDB4JXggc3cgJWQ6IGhhcz0lZCIsIGRldmljZS0+aWQsIGksIHRlc3RfYml0KGksIHN3X2JpdG1hc2spKTsKLSAgICAgICAgICAgIGlmICh0ZXN0X2JpdChpLCBzd19iaXRtYXNrKSkgewotICAgICAgICAgICAgICAgIGlmIChtU3dpdGNoZXNbaV0gPT0gMCkgewotICAgICAgICAgICAgICAgICAgICBtU3dpdGNoZXNbaV0gPSBkZXZpY2UtPmlkOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSNlbmRpZgotCi0gICAgTE9HSSgiTmV3IGRldmljZTogcGF0aD0lcyBuYW1lPSVzIGlkPTB4JXggKG9mIDB4JXgpIGluZGV4PSVkIGZkPSVkIGNsYXNzZXM9MHgleFxuIiwKLSAgICAgICAgIGRldmljZU5hbWUsIG5hbWUsIGRldmljZS0+aWQsIG1OdW1EZXZpY2VzQnlJZCwgbUZEQ291bnQsIGZkLCBkZXZpY2UtPmNsYXNzZXMpOwotCi0gICAgaWYgKChkZXZpY2UtPmNsYXNzZXMmQ0xBU1NfS0VZQk9BUkQpICE9IDApIHsKLSAgICAgICAgY2hhciBkZXZuYW1lWzEwMV07Ci0gICAgICAgIGNoYXIgdG1wZm5bMTAxXTsKLSAgICAgICAgY2hhciBrZXlsYXlvdXRGaWxlbmFtZVszMDBdOwotCi0gICAgICAgIC8vIGEgbW9yZSBkZXNjcmlwdGl2ZSBuYW1lCi0gICAgICAgIGlvY3RsKG1GRHNbbUZEQ291bnRdLmZkLCBFVklPQ0dOQU1FKHNpemVvZihkZXZuYW1lKS0xKSwgZGV2bmFtZSk7Ci0gICAgICAgIGRldm5hbWVbc2l6ZW9mKGRldm5hbWUpLTFdID0gMDsKLSAgICAgICAgZGV2aWNlLT5uYW1lID0gZGV2bmFtZTsKLQotICAgICAgICAvLyByZXBsYWNlIGFsbCB0aGUgc3BhY2VzIHdpdGggdW5kZXJzY29yZXMKLSAgICAgICAgc3RyY3B5KHRtcGZuLCBkZXZuYW1lKTsKLSAgICAgICAgZm9yIChjaGFyICpwID0gc3RyY2hyKHRtcGZuLCAnICcpOyBwICYmICpwOyBwID0gc3RyY2hyKHRtcGZuLCAnICcpKQotICAgICAgICAgICAgKnAgPSAnXyc7Ci0KLSAgICAgICAgLy8gZmluZCB0aGUgLmtsIGZpbGUgd2UgbmVlZCBmb3IgdGhpcyBkZXZpY2UKLSAgICAgICAgY29uc3QgY2hhciogcm9vdCA9IGdldGVudigiQU5EUk9JRF9ST09UIik7Ci0gICAgICAgIHNucHJpbnRmKGtleWxheW91dEZpbGVuYW1lLCBzaXplb2Yoa2V5bGF5b3V0RmlsZW5hbWUpLAotICAgICAgICAgICAgICAgICAiJXMvdXNyL2tleWxheW91dC8lcy5rbCIsIHJvb3QsIHRtcGZuKTsKLSAgICAgICAgYm9vbCBkZWZhdWx0S2V5bWFwID0gZmFsc2U7Ci0gICAgICAgIGlmIChhY2Nlc3Moa2V5bGF5b3V0RmlsZW5hbWUsIFJfT0spKSB7Ci0gICAgICAgICAgICBzbnByaW50ZihrZXlsYXlvdXRGaWxlbmFtZSwgc2l6ZW9mKGtleWxheW91dEZpbGVuYW1lKSwKLSAgICAgICAgICAgICAgICAgICAgICIlcy91c3Iva2V5bGF5b3V0LyVzIiwgcm9vdCwgInF3ZXJ0eS5rbCIpOwotICAgICAgICAgICAgZGVmYXVsdEtleW1hcCA9IHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgZGV2aWNlLT5sYXlvdXRNYXAtPmxvYWQoa2V5bGF5b3V0RmlsZW5hbWUpOwotCi0gICAgICAgIC8vIHRlbGwgdGhlIHdvcmxkIGFib3V0IHRoZSBkZXZuYW1lICh0aGUgZGVzY3JpcHRpdmUgbmFtZSkKLSAgICAgICAgaW50MzJfdCBwdWJsaWNJRDsKLSAgICAgICAgaWYgKCFtSGF2ZUZpcnN0S2V5Ym9hcmQgJiYgIWRlZmF1bHRLZXltYXApIHsKLSAgICAgICAgICAgIHB1YmxpY0lEID0gMDsKLSAgICAgICAgICAgIC8vIHRoZSBidWlsdC1pbiBrZXlib2FyZCBoYXMgYSB3ZWxsLWtub3duIGRldmljZSBJRCBvZiAwLAotICAgICAgICAgICAgLy8gdGhpcyBkZXZpY2UgYmV0dGVyIG5vdCBnbyBhd2F5LgotICAgICAgICAgICAgbUhhdmVGaXJzdEtleWJvYXJkID0gdHJ1ZTsKLSAgICAgICAgICAgIG1GaXJzdEtleWJvYXJkSWQgPSBkZXZpY2UtPmlkOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcHVibGljSUQgPSBkZXZpY2UtPmlkOwotICAgICAgICAgICAgLy8gZW5zdXJlIG1GaXJzdEtleWJvYXJkSWQgaXMgc2V0IHRvIC1zb21ldGhpbmctLgotICAgICAgICAgICAgaWYgKG1GaXJzdEtleWJvYXJkSWQgPT0gMCkgewotICAgICAgICAgICAgICAgIG1GaXJzdEtleWJvYXJkSWQgPSBkZXZpY2UtPmlkOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGNoYXIgcHJvcE5hbWVbMTAwXTsKLSAgICAgICAgc3ByaW50Zihwcm9wTmFtZSwgImh3LmtleWJvYXJkcy4ldS5kZXZuYW1lIiwgcHVibGljSUQpOwotICAgICAgICBwcm9wZXJ0eV9zZXQocHJvcE5hbWUsIGRldm5hbWUpOwotCi0gICAgICAgIExPR0koIk5ldyBrZXlib2FyZDogcHVibGljSUQ9JWQgZGV2aWNlLT5pZD0lZCBkZXZuYW1lPSclcycgcHJvcE5hbWU9JyVzJyBrZXlsYXlvdXQ9JyVzJ1xuIiwKLSAgICAgICAgICAgICAgICBwdWJsaWNJRCwgZGV2aWNlLT5pZCwgZGV2bmFtZSwgcHJvcE5hbWUsIGtleWxheW91dEZpbGVuYW1lKTsKLSAgICB9Ci0KLSAgICBMT0dWKCJBZGRpbmcgZGV2aWNlICVzICVwIGF0ICVkLCBpZCA9ICVkLCBjbGFzc2VzID0gMHgleFxuIiwKLSAgICAgICAgIGRldmljZU5hbWUsIGRldmljZSwgbUZEQ291bnQsIGRldmlkLCBkZXZpY2UtPmNsYXNzZXMpOwotCi0gICAgbURldmljZXNCeUlkW2RldmlkXS5kZXZpY2UgPSBkZXZpY2U7Ci0gICAgZGV2aWNlLT5uZXh0ID0gbU9wZW5pbmdEZXZpY2VzOwotICAgIG1PcGVuaW5nRGV2aWNlcyA9IGRldmljZTsKLSAgICBtRGV2aWNlc1ttRkRDb3VudF0gPSBkZXZpY2U7Ci0KLSAgICBtRkRDb3VudCsrOwotICAgIHJldHVybiAwOwotfQotCi1pbnQgRXZlbnRIdWI6OmNsb3NlX2RldmljZShjb25zdCBjaGFyICpkZXZpY2VOYW1lKQotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgCi0gICAgaW50IGk7Ci0gICAgZm9yKGkgPSAxOyBpIDwgbUZEQ291bnQ7IGkrKykgewotICAgICAgICBpZihzdHJjbXAobURldmljZXNbaV0tPnBhdGguc3RyaW5nKCksIGRldmljZU5hbWUpID09IDApIHsKLSAgICAgICAgICAgIC8vTE9HRCgicmVtb3ZlIGRldmljZSAlZDogJXNcbiIsIGksIGRldmljZU5hbWUpOwotICAgICAgICAgICAgZGV2aWNlX3QqIGRldmljZSA9IG1EZXZpY2VzW2ldOwotICAgICAgICAgICAgaW50IGNvdW50ID0gbUZEQ291bnQgLSBpIC0gMTsKLSAgICAgICAgICAgIGludCBpbmRleCA9IChkZXZpY2UtPmlkJklEX01BU0spOwotICAgICAgICAgICAgbURldmljZXNCeUlkW2luZGV4XS5kZXZpY2UgPSBOVUxMOwotICAgICAgICAgICAgbWVtbW92ZShtRGV2aWNlcyArIGksIG1EZXZpY2VzICsgaSArIDEsIHNpemVvZihtRGV2aWNlc1swXSkgKiBjb3VudCk7Ci0gICAgICAgICAgICBtZW1tb3ZlKG1GRHMgKyBpLCBtRkRzICsgaSArIDEsIHNpemVvZihtRkRzWzBdKSAqIGNvdW50KTsKLQotI2lmZGVmIEVWX1NXCi0gICAgICAgICAgICBmb3IgKGludCBqPTA7IGo8RVZfU1c7IGorKykgewotICAgICAgICAgICAgICAgIGlmIChtU3dpdGNoZXNbal0gPT0gZGV2aWNlLT5pZCkgewotICAgICAgICAgICAgICAgICAgICBtU3dpdGNoZXNbal0gPSAwOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSNlbmRpZgotICAgICAgICAgICAgCi0gICAgICAgICAgICBkZXZpY2UtPm5leHQgPSBtQ2xvc2luZ0RldmljZXM7Ci0gICAgICAgICAgICBtQ2xvc2luZ0RldmljZXMgPSBkZXZpY2U7Ci0KLSAgICAgICAgICAgIG1GRENvdW50LS07Ci0KLSAgICAgICAgICAgIHVpbnQzMl90IHB1YmxpY0lEOwotICAgICAgICAgICAgaWYgKGRldmljZS0+aWQgPT0gbUZpcnN0S2V5Ym9hcmRJZCkgewotICAgICAgICAgICAgICAgIExPR1coImJ1aWx0LWluIGtleWJvYXJkIGRldmljZSAlcyAoaWQ9JWQpIGlzIGNsb3NpbmchIHRoZSBhcHBzIHdpbGwgbm90IGxpa2UgdGhpcyIsCi0gICAgICAgICAgICAgICAgICAgICAgICBkZXZpY2UtPnBhdGguc3RyaW5nKCksIG1GaXJzdEtleWJvYXJkSWQpOwotICAgICAgICAgICAgICAgIG1GaXJzdEtleWJvYXJkSWQgPSAwOwotICAgICAgICAgICAgICAgIHB1YmxpY0lEID0gMDsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgcHVibGljSUQgPSBkZXZpY2UtPmlkOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgLy8gY2xlYXIgdGhlIHByb3BlcnR5Ci0gICAgICAgICAgICBjaGFyIHByb3BOYW1lWzEwMF07Ci0gICAgICAgICAgICBzcHJpbnRmKHByb3BOYW1lLCAiaHcua2V5Ym9hcmRzLiV1LmRldm5hbWUiLCBwdWJsaWNJRCk7Ci0gICAgICAgICAgICBwcm9wZXJ0eV9zZXQocHJvcE5hbWUsIE5VTEwpOwotICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgTE9HRSgicmVtb3RlIGRldmljZTogJXMgbm90IGZvdW5kXG4iLCBkZXZpY2VOYW1lKTsKLSAgICByZXR1cm4gLTE7Ci19Ci0KLWludCBFdmVudEh1Yjo6cmVhZF9ub3RpZnkoaW50IG5mZCkKLXsKLSNpZmRlZiBIQVZFX0lOT1RJRlkKLSAgICBpbnQgcmVzOwotICAgIGNoYXIgZGV2bmFtZVtQQVRIX01BWF07Ci0gICAgY2hhciAqZmlsZW5hbWU7Ci0gICAgY2hhciBldmVudF9idWZbNTEyXTsKLSAgICBpbnQgZXZlbnRfc2l6ZTsKLSAgICBpbnQgZXZlbnRfcG9zID0gMDsKLSAgICBzdHJ1Y3QgaW5vdGlmeV9ldmVudCAqZXZlbnQ7Ci0KLSAgICByZXMgPSByZWFkKG5mZCwgZXZlbnRfYnVmLCBzaXplb2YoZXZlbnRfYnVmKSk7Ci0gICAgaWYocmVzIDwgKGludClzaXplb2YoKmV2ZW50KSkgewotICAgICAgICBpZihlcnJubyA9PSBFSU5UUikKLSAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICBMT0dXKCJjb3VsZCBub3QgZ2V0IGV2ZW50LCAlc1xuIiwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgcmV0dXJuIDE7Ci0gICAgfQotICAgIC8vcHJpbnRmKCJnb3QgJWQgYnl0ZXMgb2YgZXZlbnQgaW5mb3JtYXRpb25cbiIsIHJlcyk7Ci0KLSAgICBzdHJjcHkoZGV2bmFtZSwgZGV2aWNlX3BhdGgpOwotICAgIGZpbGVuYW1lID0gZGV2bmFtZSArIHN0cmxlbihkZXZuYW1lKTsKLSAgICAqZmlsZW5hbWUrKyA9ICcvJzsKLQotICAgIHdoaWxlKHJlcyA+PSAoaW50KXNpemVvZigqZXZlbnQpKSB7Ci0gICAgICAgIGV2ZW50ID0gKHN0cnVjdCBpbm90aWZ5X2V2ZW50ICopKGV2ZW50X2J1ZiArIGV2ZW50X3Bvcyk7Ci0gICAgICAgIC8vcHJpbnRmKCIlZDogJTA4eCBcIiVzXCJcbiIsIGV2ZW50LT53ZCwgZXZlbnQtPm1hc2ssIGV2ZW50LT5sZW4gPyBldmVudC0+bmFtZSA6ICIiKTsKLSAgICAgICAgaWYoZXZlbnQtPmxlbikgewotICAgICAgICAgICAgc3RyY3B5KGZpbGVuYW1lLCBldmVudC0+bmFtZSk7Ci0gICAgICAgICAgICBpZihldmVudC0+bWFzayAmIElOX0NSRUFURSkgewotICAgICAgICAgICAgICAgIG9wZW5fZGV2aWNlKGRldm5hbWUpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZWxzZSB7Ci0gICAgICAgICAgICAgICAgY2xvc2VfZGV2aWNlKGRldm5hbWUpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGV2ZW50X3NpemUgPSBzaXplb2YoKmV2ZW50KSArIGV2ZW50LT5sZW47Ci0gICAgICAgIHJlcyAtPSBldmVudF9zaXplOwotICAgICAgICBldmVudF9wb3MgKz0gZXZlbnRfc2l6ZTsKLSAgICB9Ci0jZW5kaWYKLSAgICByZXR1cm4gMDsKLX0KLQotCi1pbnQgRXZlbnRIdWI6OnNjYW5fZGlyKGNvbnN0IGNoYXIgKmRpcm5hbWUpCi17Ci0gICAgY2hhciBkZXZuYW1lW1BBVEhfTUFYXTsKLSAgICBjaGFyICpmaWxlbmFtZTsKLSAgICBESVIgKmRpcjsKLSAgICBzdHJ1Y3QgZGlyZW50ICpkZTsKLSAgICBkaXIgPSBvcGVuZGlyKGRpcm5hbWUpOwotICAgIGlmKGRpciA9PSBOVUxMKQotICAgICAgICByZXR1cm4gLTE7Ci0gICAgc3RyY3B5KGRldm5hbWUsIGRpcm5hbWUpOwotICAgIGZpbGVuYW1lID0gZGV2bmFtZSArIHN0cmxlbihkZXZuYW1lKTsKLSAgICAqZmlsZW5hbWUrKyA9ICcvJzsKLSAgICB3aGlsZSgoZGUgPSByZWFkZGlyKGRpcikpKSB7Ci0gICAgICAgIGlmKGRlLT5kX25hbWVbMF0gPT0gJy4nICYmCi0gICAgICAgICAgIChkZS0+ZF9uYW1lWzFdID09ICdcMCcgfHwKLSAgICAgICAgICAgIChkZS0+ZF9uYW1lWzFdID09ICcuJyAmJiBkZS0+ZF9uYW1lWzJdID09ICdcMCcpKSkKLSAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICBzdHJjcHkoZmlsZW5hbWUsIGRlLT5kX25hbWUpOwotICAgICAgICBvcGVuX2RldmljZShkZXZuYW1lKTsKLSAgICB9Ci0gICAgY2xvc2VkaXIoZGlyKTsKLSAgICByZXR1cm4gMDsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvdWkvRXZlbnRSZWN1cnJlbmNlLmNwcCBiL2xpYnMvdWkvRXZlbnRSZWN1cnJlbmNlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYjQzNmI1MC4uMDAwMDAwMAotLS0gYS9saWJzL3VpL0V2ZW50UmVjdXJyZW5jZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw0ODQgKzAsMCBAQAotLyoKLSAqICBDb3B5cmlnaHQgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKi8KLQotI2luY2x1ZGUgPHBpbS9FdmVudFJlY3VycmVuY2UuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxsaW1pdHMuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0jZGVmaW5lIEZBSUxfSEVSRSgpIGRvIHsgXAotICAgICAgICAgICAgcHJpbnRmKCJQYXJzaW5nIGZhaWxlZCBhdCBsaW5lICVkXG4iLCBfX0xJTkVfXyk7IFwKLSAgICAgICAgICAgIHJldHVybiBVTktOT1dOX0VSUk9SOyBcCi0gICAgICAgIH0gd2hpbGUoMCkKLQotRXZlbnRSZWN1cnJlbmNlOjpFdmVudFJlY3VycmVuY2UoKQotICAgIDpmcmVxKChmcmVxX3QpMCksCi0gICAgIHVudGlsKCksCi0gICAgIGNvdW50KDApLAotICAgICBpbnRlcnZhbCgwKSwKLSAgICAgYnlzZWNvbmQoMCksCi0gICAgIGJ5c2Vjb25kQ291bnQoMCksCi0gICAgIGJ5bWludXRlKDApLAotICAgICBieW1pbnV0ZUNvdW50KDApLAotICAgICBieWhvdXIoMCksCi0gICAgIGJ5aG91ckNvdW50KDApLAotICAgICBieWRheSgwKSwKLSAgICAgYnlkYXlOdW0oMCksCi0gICAgIGJ5ZGF5Q291bnQoMCksCi0gICAgIGJ5bW9udGhkYXkoMCksCi0gICAgIGJ5bW9udGhkYXlDb3VudCgwKSwKLSAgICAgYnl5ZWFyZGF5KDApLAotICAgICBieXllYXJkYXlDb3VudCgwKSwKLSAgICAgYnl3ZWVrbm8oMCksCi0gICAgIGJ5d2Vla25vQ291bnQoMCksCi0gICAgIGJ5bW9udGgoMCksCi0gICAgIGJ5bW9udGhDb3VudCgwKSwKLSAgICAgYnlzZXRwb3MoMCksCi0gICAgIGJ5c2V0cG9zQ291bnQoMCksCi0gICAgIHdrc3QoMCkKLXsKLX0KLQotRXZlbnRSZWN1cnJlbmNlOjp+RXZlbnRSZWN1cnJlbmNlKCkKLXsKLSAgICBkZWxldGVbXSBieXNlY29uZDsKLSAgICBkZWxldGVbXSBieW1pbnV0ZTsKLSAgICBkZWxldGVbXSBieWhvdXI7Ci0gICAgZGVsZXRlW10gYnlkYXk7Ci0gICAgZGVsZXRlW10gYnlkYXlOdW07Ci0gICAgZGVsZXRlW10gYnl5ZWFyZGF5OwotICAgIGRlbGV0ZVtdIGJ5bW9udGhkYXk7Ci0gICAgZGVsZXRlW10gYnl3ZWVrbm87Ci0gICAgZGVsZXRlW10gYnltb250aDsKLSAgICBkZWxldGVbXSBieXNldHBvczsKLX0KLQotZW51bSBMSFMgewotICAgIE5PTkVfTEhTID0gMCwKLSAgICBGUkVRLAotICAgIFVOVElMLAotICAgIENPVU5ULAotICAgIElOVEVSVkFMLAotICAgIEJZU0VDT05ELAotICAgIEJZTUlOVVRFLAotICAgIEJZSE9VUiwKLSAgICBCWURBWSwKLSAgICBCWU1PTlRIREFZLAotICAgIEJZWUVBUkRBWSwKLSAgICBCWVdFRUtOTywKLSAgICBCWU1PTlRILAotICAgIEJZU0VUUE9TLAotICAgIFdLU1QKLX07Ci0KLXN0cnVjdCBMSFNQcm9jCi17Ci0gICAgY29uc3QgY2hhcjE2X3QqIHRleHQ7Ci0gICAgc2l6ZV90IHRleHRTaXplOwotICAgIHVpbnQzMl90IHZhbHVlOwotfTsKLQotY29uc3QgY2hhcjE2X3QgRlJFUV90ZXh0W10gPSB7ICdGJywgJ1InLCAnRScsICdRJyB9OwotY29uc3QgY2hhcjE2X3QgVU5USUxfdGV4dFtdID0geyAnVScsICdOJywgJ1QnLCAnSScsICdMJyB9OwotY29uc3QgY2hhcjE2X3QgQ09VTlRfdGV4dFtdID0geyAnQycsICdPJywgJ1UnLCAnTicsICdUJyB9OwotY29uc3QgY2hhcjE2X3QgSU5URVJWQUxfdGV4dFtdID0geyAnSScsICdOJywgJ1QnLCAnRScsICdSJywgJ1YnLCAnQScsICdMJ307Ci1jb25zdCBjaGFyMTZfdCBCWVNFQ09ORF90ZXh0W10gPSB7ICdCJywgJ1knLCAnUycsICdFJywgJ0MnLCAnTycsICdOJywgJ0QnIH07Ci1jb25zdCBjaGFyMTZfdCBCWU1JTlVURV90ZXh0W10gPSB7ICdCJywgJ1knLCAnTScsICdJJywgJ04nLCAnVScsICdUJywgJ0UnIH07Ci1jb25zdCBjaGFyMTZfdCBCWUhPVVJfdGV4dFtdID0geyAnQicsICdZJywgJ0gnLCAnTycsICdVJywgJ1InIH07Ci1jb25zdCBjaGFyMTZfdCBCWURBWV90ZXh0W10gPSB7ICdCJywgJ1knLCAnRCcsICdBJywgJ1knIH07Ci1jb25zdCBjaGFyMTZfdCBCWU1PTlRIREFZX3RleHRbXSA9IHsgJ0InLCdZJywnTScsJ08nLCdOJywnVCcsJ0gnLCdEJywnQScsJ1knIH07Ci1jb25zdCBjaGFyMTZfdCBCWVlFQVJEQVlfdGV4dFtdID0geyAnQicsJ1knLCdZJywnRScsJ0EnLCdSJywnRCcsJ0EnLCdZJyB9OwotY29uc3QgY2hhcjE2X3QgQllXRUVLTk9fdGV4dFtdID0geyAnQicsICdZJywgJ1cnLCAnRScsICdFJywgJ0snLCAnTicsICdPJyB9OwotY29uc3QgY2hhcjE2X3QgQllNT05USF90ZXh0W10gPSB7ICdCJywgJ1knLCAnTScsICdPJywgJ04nLCAnVCcsICdIJyB9OwotY29uc3QgY2hhcjE2X3QgQllTRVRQT1NfdGV4dFtdID0geyAnQicsICdZJywgJ1MnLCAnRScsICdUJywgJ1AnLCAnTycsICdTJyB9OwotY29uc3QgY2hhcjE2X3QgV0tTVF90ZXh0W10gPSB7ICdXJywgJ0snLCAnUycsICdUJyB9OwotCi0jZGVmaW5lIFNJWih4KSAoc2l6ZW9mKHgpL3NpemVvZih4WzBdKSkKLQotY29uc3QgTEhTUHJvYyBMSFNQUk9DW10gPSB7Ci0gICAgeyBGUkVRX3RleHQsIFNJWihGUkVRX3RleHQpLCBGUkVRIH0sCi0gICAgeyBVTlRJTF90ZXh0LCBTSVooVU5USUxfdGV4dCksIFVOVElMIH0sCi0gICAgeyBDT1VOVF90ZXh0LCBTSVooQ09VTlRfdGV4dCksIENPVU5UIH0sCi0gICAgeyBJTlRFUlZBTF90ZXh0LCBTSVooSU5URVJWQUxfdGV4dCksIElOVEVSVkFMIH0sCi0gICAgeyBCWVNFQ09ORF90ZXh0LCBTSVooQllTRUNPTkRfdGV4dCksIEJZU0VDT05EIH0sCi0gICAgeyBCWU1JTlVURV90ZXh0LCBTSVooQllNSU5VVEVfdGV4dCksIEJZTUlOVVRFIH0sCi0gICAgeyBCWUhPVVJfdGV4dCwgU0laKEJZSE9VUl90ZXh0KSwgQllIT1VSIH0sCi0gICAgeyBCWURBWV90ZXh0LCBTSVooQllEQVlfdGV4dCksIEJZREFZIH0sCi0gICAgeyBCWU1PTlRIREFZX3RleHQsIFNJWihCWU1PTlRIREFZX3RleHQpLCBCWU1PTlRIREFZIH0sCi0gICAgeyBCWVlFQVJEQVlfdGV4dCwgU0laKEJZWUVBUkRBWV90ZXh0KSwgQllZRUFSREFZIH0sCi0gICAgeyBCWVdFRUtOT190ZXh0LCBTSVooQllXRUVLTk9fdGV4dCksIEJZV0VFS05PIH0sCi0gICAgeyBCWU1PTlRIX3RleHQsIFNJWihCWU1PTlRIX3RleHQpLCBCWU1PTlRIIH0sCi0gICAgeyBCWVNFVFBPU190ZXh0LCBTSVooQllTRVRQT1NfdGV4dCksIEJZU0VUUE9TIH0sCi0gICAgeyBXS1NUX3RleHQsIFNJWihXS1NUX3RleHQpLCBXS1NUIH0sCi0gICAgeyBOVUxMLCAwLCBOT05FX0xIUyB9LAotfTsKLQotY29uc3QgY2hhcjE2X3QgU0VDT05ETFlfdGV4dFtdID0geyAnUycsJ0UnLCdDJywnTycsJ04nLCdEJywnTCcsJ1knIH07Ci1jb25zdCBjaGFyMTZfdCBNSU5VVEVMWV90ZXh0W10gPSB7ICdNJywnSScsJ04nLCdVJywnVCcsJ0UnLCdMJywnWScgfTsKLWNvbnN0IGNoYXIxNl90IEhPVVJMWV90ZXh0W10gPSB7ICdIJywnTycsJ1UnLCdSJywnTCcsJ1knIH07Ci1jb25zdCBjaGFyMTZfdCBEQUlMWV90ZXh0W10gPSB7ICdEJywnQScsJ0knLCdMJywnWScgfTsKLWNvbnN0IGNoYXIxNl90IFdFRUtMWV90ZXh0W10gPSB7ICdXJywnRScsJ0UnLCdLJywnTCcsJ1knIH07Ci1jb25zdCBjaGFyMTZfdCBNT05USExZX3RleHRbXSA9IHsgJ00nLCdPJywnTicsJ1QnLCdIJywnTCcsJ1knIH07Ci1jb25zdCBjaGFyMTZfdCBZRUFSTFlfdGV4dFtdID0geyAnWScsJ0UnLCdBJywnUicsJ0wnLCdZJyB9OwotCi10eXBlZGVmIExIU1Byb2MgRnJlcVByb2M7Ci0KLWNvbnN0IEZyZXFQcm9jIEZSRVFQUk9DW10gPSB7Ci0gICAgeyBTRUNPTkRMWV90ZXh0LCBTSVooU0VDT05ETFlfdGV4dCksIEV2ZW50UmVjdXJyZW5jZTo6U0VDT05ETFkgfSwKLSAgICB7IE1JTlVURUxZX3RleHQsIFNJWihNSU5VVEVMWV90ZXh0KSwgRXZlbnRSZWN1cnJlbmNlOjpNSU5VVEVMWSB9LAotICAgIHsgSE9VUkxZX3RleHQsIFNJWihIT1VSTFlfdGV4dCksIEV2ZW50UmVjdXJyZW5jZTo6SE9VUkxZIH0sCi0gICAgeyBEQUlMWV90ZXh0LCBTSVooREFJTFlfdGV4dCksIEV2ZW50UmVjdXJyZW5jZTo6REFJTFkgfSwKLSAgICB7IFdFRUtMWV90ZXh0LCBTSVooV0VFS0xZX3RleHQpLCBFdmVudFJlY3VycmVuY2U6OldFRUtMWSB9LAotICAgIHsgTU9OVEhMWV90ZXh0LCBTSVooTU9OVEhMWV90ZXh0KSwgRXZlbnRSZWN1cnJlbmNlOjpNT05USExZIH0sCi0gICAgeyBZRUFSTFlfdGV4dCwgU0laKFlFQVJMWV90ZXh0KSwgRXZlbnRSZWN1cnJlbmNlOjpZRUFSTFkgfSwKLSAgICB7IE5VTEwsIDAsIE5PTkVfTEhTIH0sCi19OwotCi1jb25zdCBjaGFyMTZfdCBTVV90ZXh0W10gPSB7ICdTJywnVScgfTsKLWNvbnN0IGNoYXIxNl90IE1PX3RleHRbXSA9IHsgJ00nLCdPJyB9OwotY29uc3QgY2hhcjE2X3QgVFVfdGV4dFtdID0geyAnVCcsJ1UnIH07Ci1jb25zdCBjaGFyMTZfdCBXRV90ZXh0W10gPSB7ICdXJywnRScgfTsKLWNvbnN0IGNoYXIxNl90IFRIX3RleHRbXSA9IHsgJ1QnLCdIJyB9OwotY29uc3QgY2hhcjE2X3QgRlJfdGV4dFtdID0geyAnRicsJ1InIH07Ci1jb25zdCBjaGFyMTZfdCBTQV90ZXh0W10gPSB7ICdTJywnQScgfTsKLQotY29uc3QgRnJlcVByb2MgV0VFS0RBWVBST0NbXSA9IHsKLSAgICB7IFNVX3RleHQsIFNJWihTVV90ZXh0KSwgRXZlbnRSZWN1cnJlbmNlOjpTVSB9LAotICAgIHsgTU9fdGV4dCwgU0laKE1PX3RleHQpLCBFdmVudFJlY3VycmVuY2U6Ok1PIH0sCi0gICAgeyBUVV90ZXh0LCBTSVooVFVfdGV4dCksIEV2ZW50UmVjdXJyZW5jZTo6VFUgfSwKLSAgICB7IFdFX3RleHQsIFNJWihXRV90ZXh0KSwgRXZlbnRSZWN1cnJlbmNlOjpXRSB9LAotICAgIHsgVEhfdGV4dCwgU0laKFRIX3RleHQpLCBFdmVudFJlY3VycmVuY2U6OlRIIH0sCi0gICAgeyBGUl90ZXh0LCBTSVooRlJfdGV4dCksIEV2ZW50UmVjdXJyZW5jZTo6RlIgfSwKLSAgICB7IFNBX3RleHQsIFNJWihTQV90ZXh0KSwgRXZlbnRSZWN1cnJlbmNlOjpTQSB9LAotICAgIHsgTlVMTCwgMCwgTk9ORV9MSFMgfSwKLX07Ci0KLS8vIHJldHVybnMgdGhlIGluZGV4IGludG8gTEhTUFJPQyBmb3IgdGhlIG1hdGNoIG9yIC0xIGlmIG5vdCBmb3VuZAotaW5saW5lIHN0YXRpYyBpbnQKLW1hdGNoX3Byb2MoY29uc3QgTEhTUHJvYyogcCwgY29uc3QgY2hhcjE2X3QqIHN0ciwgc2l6ZV90IGxlbikKLXsKLSAgICBpbnQgaSA9IDA7Ci0gICAgd2hpbGUgKHAtPnRleHQgIT0gTlVMTCkgewotICAgICAgICBpZiAocC0+dGV4dFNpemUgPT0gbGVuKSB7Ci0gICAgICAgICAgICBpZiAoMCA9PSBtZW1jbXAocC0+dGV4dCwgc3RyLCBsZW4qc2l6ZW9mKGNoYXIxNl90KSkpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gaTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBwKys7Ci0gICAgICAgIGkrKzsKLSAgICB9Ci0gICAgcmV0dXJuIC0xOwotfQotCi0vLyByYW5nZU1pbiBhbmQgcmFuZ2VNYXggYXJlIGluY2x1c2l2ZQotc3RhdGljIHN0YXR1c190Ci1wYXJzZV9pbnQoY29uc3QgY2hhcjE2X3QqIHN0ciwgc2l6ZV90IGxlbiwgaW50KiBvdXQsCi0gICAgICAgICAgICBpbnQgcmFuZ2VNaW4sIGludCByYW5nZU1heCwgYm9vbCB6ZXJvT0spCi17Ci0gICAgY2hhcjE2X3QgYzsKLSAgICBzaXplX3QgaT0wOwotCi0gICAgaWYgKGxlbiA9PSAwKSB7Ci0gICAgICAgIEZBSUxfSEVSRSgpOwotICAgIH0KLSAgICBib29sIG5lZ2F0aXZlID0gZmFsc2U7Ci0gICAgYyA9IHN0clswXTsKLSAgICBpZiAoYyA9PSAnLScgKSB7Ci0gICAgICAgIG5lZ2F0aXZlID0gdHJ1ZTsKLSAgICAgICAgaSsrOwotICAgIH0KLSAgICBlbHNlIGlmIChjID09ICcrJykgewotICAgICAgICBpKys7Ci0gICAgfQotICAgIGludCBuID0gMDsKLSAgICBmb3IgKDsgaTxsZW47IGkrKykgewotICAgICAgICBjID0gc3RyW2ldOwotICAgICAgICBpZiAoYyA8ICcwJyB8fCBjID4gJzknKSB7Ci0gICAgICAgICAgICBGQUlMX0hFUkUoKTsKLSAgICAgICAgfQotICAgICAgICBpbnQgcHJldiA9IG47Ci0gICAgICAgIG4gKj0gMTA7Ci0gICAgICAgIC8vIHRoZSBzcGVjIGRvZXNuJ3QgYWRkcmVzcyBob3cgYmlnIHRoZXNlIG51bWJlcnMgY2FuIGJlLAotICAgICAgICAvLyBzbyB3ZSdyZSBub3QgZ29pbmcgdG8gd29ycnkgYWJvdXQgbm90IGJlaW5nIGFibGUgdG8gcmVwcmVzZW50Ci0gICAgICAgIC8vIElOVF9NSU4sIGFuZCBpZiB3ZSdyZSBnb2luZyB0byB3cmFwLCB3ZSdsbCBqdXN0IGNsYW1wIHRvCi0gICAgICAgIC8vIElOVF9NQVggaW5zdGVhZAotICAgICAgICBpZiAobiA8IHByZXYpIHsKLSAgICAgICAgICAgIG4gPSBJTlRfTUFYOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbiArPSBjIC0gJzAnOwotICAgICAgICB9Ci0gICAgfQotICAgIGlmIChuZWdhdGl2ZSkgewotICAgICAgICBuID0gLW47Ci0gICAgfQotICAgIGlmIChuIDwgcmFuZ2VNaW4gfHwgbiA+IHJhbmdlTWF4KSB7Ci0gICAgICAgIEZBSUxfSEVSRSgpOwotICAgIH0KLSAgICBpZiAoIXplcm9PSyAmJiBuID09IDApIHsKLSAgICAgICAgRkFJTF9IRVJFKCk7Ci0gICAgfQotICAgICpvdXQgPSBuOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdGljIHN0YXR1c190Ci1wYXJzZV9pbnRfbGlzdChjb25zdCBjaGFyMTZfdCogc3RyLCBzaXplX3QgbGVuLCBpbnQqIGNvdW50T3V0LCBpbnQqKiBsaXN0T3V0LAotICAgICAgICAgIGludCByYW5nZU1pbiwgaW50IHJhbmdlTWF4LCBib29sIHplcm9PSywKLSAgICAgICAgICBzdGF0dXNfdCAoKmZ1bmMpKGNvbnN0IGNoYXIxNl90KixzaXplX3QsaW50KixpbnQsaW50LGJvb2wpPXBhcnNlX2ludCkKLXsKLSAgICBzdGF0dXNfdCBlcnI7Ci0KLSAgICBpZiAobGVuID09IDApIHsKLSAgICAgICAgKmNvdW50T3V0ID0gMDsKLSAgICAgICAgKmxpc3RPdXQgPSBOVUxMOwotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotCi0gICAgLy8gbWFrZSBvbmUgcGFzcyB0aHJvdWdoIGxvb2tpbmcgZm9yIGNvbW1hcyBzbyB3ZSBrbm93IGhvdyBiaWcgdG8gbWFrZSBvdXIKLSAgICAvLyBvdXQgYXJyYXkuCi0gICAgaW50IGNvdW50ID0gMTsKLSAgICBmb3IgKHNpemVfdCBpPTA7IGk8bGVuOyBpKyspIHsKLSAgICAgICAgaWYgKHN0cltpXSA9PSAnLCcpIHsKLSAgICAgICAgICAgIGNvdW50Kys7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBpbnQqIGxpc3QgPSBuZXcgaW50W2NvdW50XTsKLSAgICBjb25zdCBjaGFyMTZfdCogcCA9IHN0cjsKLSAgICBpbnQgY29tbWFJbmRleCA9IDA7Ci0gICAgc2l6ZV90IGk7Ci0KLSAgICBmb3IgKGk9MDsgaTxsZW47IGkrKykgewotICAgICAgICBpZiAoc3RyW2ldID09ICcsJykgewotICAgICAgICAgICAgZXJyID0gZnVuYyhwLCAoc3RyK2ktcCksIGxpc3QrY29tbWFJbmRleCwgcmFuZ2VNaW4sCi0gICAgICAgICAgICAgICAgICAgIHJhbmdlTWF4LCB6ZXJvT0spOwotICAgICAgICAgICAgaWYgKGVyciAhPSBOT19FUlJPUikgewotICAgICAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNvbW1hSW5kZXgrKzsKLSAgICAgICAgICAgIHAgPSBzdHIraSsxOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgZXJyID0gZnVuYyhwLCAoc3RyK2ktcCksIGxpc3QrY29tbWFJbmRleCwgcmFuZ2VNaW4sIHJhbmdlTWF4LCB6ZXJvT0spOwotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLSAgICBjb21tYUluZGV4Kys7Ci0KLSAgICAqY291bnRPdXQgPSBjb3VudDsKLSAgICAqbGlzdE91dCA9IGxpc3Q7Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci0KLWJhaWw6Ci0gICAgZGVsZXRlW10gbGlzdDsKLSAgICBGQUlMX0hFUkUoKTsKLX0KLQotLy8gdGhlIG51bWJlcnMgaGVyZSBhcmUgc21hbGwsIHNvIHdlIHBhY2sgdGhlbSBib3RoIGludG8gb25lIHZhbHVlLCBhbmQgdGhlbgotLy8gc3BsaXQgaXQgb3V0IGxhdGVyLiAgaXQgbGV0cyB1cyByZXVzZSBhbGwgdGhlIGNvbW1hIHNlcGFyYXRlZCBsaXN0IGNvZGUuCi1zdGF0aWMgc3RhdHVzX3QKLXBhcnNlX2J5ZGF5KGNvbnN0IGNoYXIxNl90KiBzLCBzaXplX3QgbGVuLCBpbnQqIG91dCwKLSAgICAgICAgICAgIGludCByYW5nZU1pbiwgaW50IHJhbmdlTWF4LCBib29sIHplcm9PSykKLXsKLSAgICBzdGF0dXNfdCBlcnI7Ci0gICAgaW50IG4gPSAwOwotICAgIGNvbnN0IGNoYXIxNl90KiBwID0gczsKLSAgICBzaXplX3QgcGxlbiA9IGxlbjsKLQotICAgIGlmIChsZW4gPiAwKSB7Ci0gICAgICAgIGNoYXIxNl90IGMgPSBzWzBdOwotICAgICAgICBpZiAoYyA9PSAnLScgfHwgYyA9PSAnKycgfHwgKGMgPj0gJzAnICYmIGMgPD0gJzknKSkgewotICAgICAgICAgICAgaWYgKGxlbiA+IDEpIHsKLSAgICAgICAgICAgICAgICBzaXplX3QgbmxlbiA9IDA7Ci0gICAgICAgICAgICAgICAgYyA9IHNbbmxlbl07Ci0gICAgICAgICAgICAgICAgd2hpbGUgKG5sZW4gPCBsZW4KLSAgICAgICAgICAgICAgICAgICAgICAgICYmIChjID09ICctJyB8fCBjID09ICcrJyB8fCAoYyA+PSAnMCcgJiYgYyA8PSAnOScpKSkgewotICAgICAgICAgICAgICAgICAgICBjID0gc1tubGVuXTsKLSAgICAgICAgICAgICAgICAgICAgbmxlbisrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAobmxlbiA+IDApIHsKLSAgICAgICAgICAgICAgICAgICAgbmxlbi0tOwotICAgICAgICAgICAgICAgICAgICBlcnIgPSBwYXJzZV9pbnQocywgbmxlbiwgJm4sIHJhbmdlTWluLCByYW5nZU1heCwgemVyb09LKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGVyciAhPSBOT19FUlJPUikgewotICAgICAgICAgICAgICAgICAgICAgICAgRkFJTF9IRVJFKCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcCArPSBubGVuOwotICAgICAgICAgICAgICAgICAgICBwbGVuIC09IG5sZW47Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaW50IGluZGV4ID0gbWF0Y2hfcHJvYyhXRUVLREFZUFJPQywgcCwgcGxlbik7Ci0gICAgICAgIGlmIChpbmRleCA+PSAwKSB7Ci0gICAgICAgICAgICAqb3V0ID0gKDB4ZmZmZjAwMDAgJiBXRUVLREFZUFJPQ1tpbmRleF0udmFsdWUpCi0gICAgICAgICAgICAgICAgICAgIHwgKDB4MDAwMGZmZmYgJiBuKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLX0KLQotc3RhdGljIHZvaWQKLXBvc3Rwcm9jZXNzX2J5ZGF5KGludCBjb3VudCwgaW50KiBieWRheSwgaW50KiogYnlkYXlOdW0pCi17Ci0gICAgaW50KiBiZG4gPSBuZXcgaW50W2NvdW50XTsKLSAgICAqYnlkYXlOdW0gPSBiZG47Ci0gICAgZm9yIChpbnQgaT0wOyBpPGNvdW50OyBpKyspIHsKLSAgICAgICAgdWludDMyX3QgdiA9IGJ5ZGF5W2ldOwotICAgICAgICBpbnQxNl90IG51bSA9IHYgJiAweDAwMDBmZmZmOwotICAgICAgICBieWRheVtpXSA9IHYgJiAweGZmZmYwMDAwOyAgCi0gICAgICAgIC8vIHdpbGwgc2lnbiBleHRlbmQ6Ci0gICAgICAgIGJkbltpXSA9IG51bTsKLSAgICB9Ci19Ci0KLSNkZWZpbmUgUEFSU0VfSU5UX0xJU1RfQ0hFQ0tFRChuYW1lLCByYW5nZU1pbiwgcmFuZ2VNYXgsIHplcm9PSykgXAotICAgIGlmIChuYW1lIyNDb3VudCAhPSAwIHx8IE5PX0VSUk9SICE9IHBhcnNlX2ludF9saXN0KHMsIHNsZW4sIFwKLSAgICAgICAgICAgICAgICAgICAgICAgICAmbmFtZSMjQ291bnQsICZuYW1lLCByYW5nZU1pbiwgcmFuZ2VNYXgsIHplcm9PSykpIHsgXAotICAgICAgICBGQUlMX0hFUkUoKTsgXAotICAgIH0KLXN0YXR1c190Ci1FdmVudFJlY3VycmVuY2U6OnBhcnNlKGNvbnN0IFN0cmluZzE2JiBzdHIpCi17Ci0gICAgY2hhcjE2X3QgY29uc3QqIHdvcmsgPSBzdHIuc3RyaW5nKCk7Ci0gICAgc2l6ZV90IGxlbiA9IHN0ci5zaXplKCk7Ci0KLSAgICBpbnQgbGhzSW5kZXggPSBOT05FX0xIUzsKLSAgICBpbnQgaW5kZXg7Ci0gICAgCi0gICAgc2l6ZV90IHN0YXJ0ID0gMDsKLSAgICBmb3IgKHNpemVfdCBpPTA7IGk8bGVuOyBpKyspIHsKLSAgICAgICAgY2hhcjE2X3QgYyA9IHdvcmtbaV07Ci0gICAgICAgIGlmIChjICE9ICc7JyAmJiBpID09IGxlbi0xKSB7Ci0gICAgICAgICAgICBjID0gJzsnOwotICAgICAgICAgICAgaSsrOwotICAgICAgICB9Ci0gICAgICAgIGlmIChjID09ICc7JyB8fCBjID09ICc9JykgewotICAgICAgICAgICAgaWYgKGkgIT0gc3RhcnQpIHsKLSAgICAgICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogcyA9IHdvcmsrc3RhcnQ7Ci0gICAgICAgICAgICAgICAgY29uc3Qgc2l6ZV90IHNsZW4gPSBpLXN0YXJ0OwotCi0gICAgICAgICAgICAgICAgU3RyaW5nOCB0aGVzdHJpbmcoU3RyaW5nMTYocywgc2xlbikpOwotCi0gICAgICAgICAgICAgICAgc3dpdGNoIChjKQotICAgICAgICAgICAgICAgIHsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSAnPSc6Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAobGhzSW5kZXggPT0gTk9ORV9MSFMpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBsaHNJbmRleCA9IG1hdGNoX3Byb2MoTEhTUFJPQywgcywgc2xlbik7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGxoc0luZGV4ID49IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgRkFJTF9IRVJFKCk7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgJzsnOgotICAgICAgICAgICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzd2l0Y2ggKExIU1BST0NbbGhzSW5kZXhdLnZhbHVlKQotICAgICAgICAgICAgICAgICAgICAgICAgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNhc2UgRlJFUToKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHRoaXMtPmZyZXEgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRkFJTF9IRVJFKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW5kZXggPSBtYXRjaF9wcm9jKEZSRVFQUk9DLCBzLCBzbGVuKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGluZGV4ID49IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMtPmZyZXEgPSAoZnJlcV90KUZSRVFQUk9DW2luZGV4XS52YWx1ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjYXNlIFVOVElMOgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBYWFggc2hvdWxkIGNoZWNrIHRoYXQgdGhpcyBpcyBhIHZhbGlkIHRpbWUKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdW50aWwuc2V0VG8oU3RyaW5nMTYocywgc2xlbikpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjYXNlIENPVU5UOgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoY291bnQgIT0gMAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8IE5PX0VSUk9SICE9IHBhcnNlX2ludChzLCBzbGVuLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJmNvdW50LCBJTlRfTUlOLCBJTlRfTUFYLCB0cnVlKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRkFJTF9IRVJFKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2FzZSBJTlRFUlZBTDoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGludGVydmFsICE9IDAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCBOT19FUlJPUiAhPSBwYXJzZV9pbnQocywgc2xlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJmludGVydmFsLCBJTlRfTUlOLCBJTlRfTUFYLCBmYWxzZSkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEZBSUxfSEVSRSgpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNhc2UgQllTRUNPTkQ6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBBUlNFX0lOVF9MSVNUX0NIRUNLRUQoYnlzZWNvbmQsIDAsIDU5LCB0cnVlKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjYXNlIEJZTUlOVVRFOgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQQVJTRV9JTlRfTElTVF9DSEVDS0VEKGJ5bWludXRlLCAwLCA1OSwgdHJ1ZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2FzZSBCWUhPVVI6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBBUlNFX0lOVF9MSVNUX0NIRUNLRUQoYnlob3VyLCAwLCAyMywgdHJ1ZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2FzZSBCWURBWToKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGJ5ZGF5Q291bnQgIT0gMCB8fCBOT19FUlJPUiAhPSAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwYXJzZV9pbnRfbGlzdChzLCBzbGVuLCAmYnlkYXlDb3VudCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAmYnlkYXksIC01MywgNTMsIGZhbHNlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBhcnNlX2J5ZGF5KSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRkFJTF9IRVJFKCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcG9zdHByb2Nlc3NfYnlkYXkoYnlkYXlDb3VudCwgYnlkYXksICZieWRheU51bSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNhc2UgQllNT05USERBWToKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUEFSU0VfSU5UX0xJU1RfQ0hFQ0tFRChieW1vbnRoZGF5LCAtMzEsIDMxLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmYWxzZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2FzZSBCWVlFQVJEQVk6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBBUlNFX0lOVF9MSVNUX0NIRUNLRUQoYnl5ZWFyZGF5LCAtMzY2LCAzNjYsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZhbHNlKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjYXNlIEJZV0VFS05POgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQQVJTRV9JTlRfTElTVF9DSEVDS0VEKGJ5d2Vla25vLCAtNTMsIDUzLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmYWxzZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2FzZSBCWU1PTlRIOgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQQVJTRV9JTlRfTElTVF9DSEVDS0VEKGJ5bW9udGgsIDEsIDEyLCBmYWxzZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2FzZSBCWVNFVFBPUzoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUEFSU0VfSU5UX0xJU1RfQ0hFQ0tFRChieXNldHBvcywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSU5UX01JTiwgSU5UX01BWCwgdHJ1ZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2FzZSBXS1NUOgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy0+d2tzdCAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBGQUlMX0hFUkUoKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbmRleCA9IG1hdGNoX3Byb2MoV0VFS0RBWVBST0MsIHMsIHNsZW4pOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoaW5kZXggPj0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy0+d2tzdCA9IChpbnQpV0VFS0RBWVBST0NbaW5kZXhdLnZhbHVlOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEZBSUxfSEVSRSgpOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgbGhzSW5kZXggPSBOT05FX0xIUzsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgc3RhcnQgPSBpKzE7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyBlbmZvcmNlIHRoYXQgdGhlcmUgd2FzIGEgRlJFUQotICAgIGlmIChmcmVxID09IDApIHsKLSAgICAgICAgRkFJTF9IRVJFKCk7Ci0gICAgfQotCi0gICAgLy8gZGVmYXVsdCB3a3N0IHRvIE1PIGlmIGl0IHdhc24ndCBzcGVjaWZpZWQKLSAgICBpZiAod2tzdCA9PSAwKSB7Ci0gICAgICAgIHdrc3QgPSBNTzsKLSAgICB9Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotCmRpZmYgLS1naXQgYS9saWJzL3VpL0lDYW1lcmEuY3BwIGIvbGlicy91aS9JQ2FtZXJhLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYWIwZmVmMS4uMDAwMDAwMAotLS0gYS9saWJzL3VpL0lDYW1lcmEuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMzQ2ICswLDAgQEAKLS8qCi0qKgotKiogQ29weXJpZ2h0IDIwMDgsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLS8vI2RlZmluZSBMT0dfTkRFQlVHIDAKLSNkZWZpbmUgTE9HX1RBRyAiSUNhbWVyYSIKLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDx1dGlscy9QYXJjZWwuaD4KLSNpbmNsdWRlIDx1aS9JQ2FtZXJhLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotZW51bSB7Ci0gICAgRElTQ09OTkVDVCA9IElCaW5kZXI6OkZJUlNUX0NBTExfVFJBTlNBQ1RJT04sCi0gICAgU0VUX1BSRVZJRVdfRElTUExBWSwKLSAgICBTRVRfUFJFVklFV19DQUxMQkFDS19GTEFHLAotICAgIFNUQVJUX1BSRVZJRVcsCi0gICAgU1RPUF9QUkVWSUVXLAotICAgIEFVVE9fRk9DVVMsCi0gICAgVEFLRV9QSUNUVVJFLAotICAgIFNFVF9QQVJBTUVURVJTLAotICAgIEdFVF9QQVJBTUVURVJTLAotICAgIENPTk5FQ1QsCi0gICAgTE9DSywKLSAgICBVTkxPQ0ssCi0gICAgUFJFVklFV19FTkFCTEVELAotICAgIFNUQVJUX1JFQ09SRElORywKLSAgICBTVE9QX1JFQ09SRElORywKLSAgICBSRUNPUkRJTkdfRU5BQkxFRCwKLSAgICBSRUxFQVNFX1JFQ09SRElOR19GUkFNRSwKLX07Ci0KLWNsYXNzIEJwQ2FtZXJhOiBwdWJsaWMgQnBJbnRlcmZhY2U8SUNhbWVyYT4KLXsKLXB1YmxpYzoKLSAgICBCcENhbWVyYShjb25zdCBzcDxJQmluZGVyPiYgaW1wbCkKLSAgICAgICAgOiBCcEludGVyZmFjZTxJQ2FtZXJhPihpbXBsKQotICAgIHsKLSAgICB9Ci0KLSAgICAvLyBkaXNjb25uZWN0IGZyb20gY2FtZXJhIHNlcnZpY2UKLSAgICB2b2lkIGRpc2Nvbm5lY3QoKQotICAgIHsKLSAgICAgICAgTE9HVigiZGlzY29ubmVjdCIpOwotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoRElTQ09OTkVDVCwgZGF0YSwgJnJlcGx5KTsKLSAgICB9Ci0KLSAgICAvLyBwYXNzIHRoZSBidWZmZXJlZCBJU3VyZmFjZSB0byB0aGUgY2FtZXJhIHNlcnZpY2UKLSAgICBzdGF0dXNfdCBzZXRQcmV2aWV3RGlzcGxheShjb25zdCBzcDxJU3VyZmFjZT4mIHN1cmZhY2UpCi0gICAgewotICAgICAgICBMT0dWKCJzZXRQcmV2aWV3RGlzcGxheSIpOwotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICBkYXRhLndyaXRlU3Ryb25nQmluZGVyKHN1cmZhY2UtPmFzQmluZGVyKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoU0VUX1BSRVZJRVdfRElTUExBWSwgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgcmV0dXJuIHJlcGx5LnJlYWRJbnQzMigpOwotICAgIH0KLQotICAgIC8vIHNldCB0aGUgcHJldmlldyBjYWxsYmFjayBmbGFnIHRvIGFmZmVjdCBob3cgdGhlIHJlY2VpdmVkIGZyYW1lcyBmcm9tCi0gICAgLy8gcHJldmlldyBhcmUgaGFuZGxlZC4gU2VlIENhbWVyYS5oIGZvciBkZXRhaWxzLgotICAgIHZvaWQgc2V0UHJldmlld0NhbGxiYWNrRmxhZyhpbnQgZmxhZykKLSAgICB7Ci0gICAgICAgIExPR1YoInNldFByZXZpZXdDYWxsYmFja0ZsYWcoJWQpIiwgZmxhZyk7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElDYW1lcmE6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihmbGFnKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KFNFVF9QUkVWSUVXX0NBTExCQUNLX0ZMQUcsIGRhdGEsICZyZXBseSk7Ci0gICAgfQotCi0gICAgLy8gc3RhcnQgcHJldmlldyBtb2RlLCBtdXN0IGNhbGwgc2V0UHJldmlld0Rpc3BsYXkgZmlyc3QKLSAgICBzdGF0dXNfdCBzdGFydFByZXZpZXcoKQotICAgIHsKLSAgICAgICAgTE9HVigic3RhcnRQcmV2aWV3Iik7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElDYW1lcmE6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChTVEFSVF9QUkVWSUVXLCBkYXRhLCAmcmVwbHkpOwotICAgICAgICByZXR1cm4gcmVwbHkucmVhZEludDMyKCk7Ci0gICAgfQotCi0gICAgLy8gc3RhcnQgcmVjb3JkaW5nIG1vZGUsIG11c3QgY2FsbCBzZXRQcmV2aWV3RGlzcGxheSBmaXJzdAotICAgIHN0YXR1c190IHN0YXJ0UmVjb3JkaW5nKCkKLSAgICB7Ci0gICAgICAgIExPR1YoInN0YXJ0UmVjb3JkaW5nIik7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElDYW1lcmE6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChTVEFSVF9SRUNPUkRJTkcsIGRhdGEsICZyZXBseSk7Ci0gICAgICAgIHJldHVybiByZXBseS5yZWFkSW50MzIoKTsKLSAgICB9Ci0KLSAgICAvLyBzdG9wIHByZXZpZXcgbW9kZQotICAgIHZvaWQgc3RvcFByZXZpZXcoKQotICAgIHsKLSAgICAgICAgTE9HVigic3RvcFByZXZpZXciKTsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KFNUT1BfUFJFVklFVywgZGF0YSwgJnJlcGx5KTsKLSAgICB9Ci0KLSAgICAvLyBzdG9wIHJlY29yZGluZyBtb2RlCi0gICAgdm9pZCBzdG9wUmVjb3JkaW5nKCkKLSAgICB7Ci0gICAgICAgIExPR1YoInN0b3BSZWNvcmRpbmciKTsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KFNUT1BfUkVDT1JESU5HLCBkYXRhLCAmcmVwbHkpOwotICAgIH0KLQotICAgIHZvaWQgcmVsZWFzZVJlY29yZGluZ0ZyYW1lKGNvbnN0IHNwPElNZW1vcnk+JiBtZW0pCi0gICAgewotICAgICAgICBMT0dWKCJyZWxlYXNlUmVjb3JkaW5nRnJhbWUiKTsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgZGF0YS53cml0ZVN0cm9uZ0JpbmRlcihtZW0tPmFzQmluZGVyKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoUkVMRUFTRV9SRUNPUkRJTkdfRlJBTUUsIGRhdGEsICZyZXBseSk7Ci0gICAgfQotCi0gICAgLy8gY2hlY2sgcHJldmlldyBzdGF0ZQotICAgIGJvb2wgcHJldmlld0VuYWJsZWQoKQotICAgIHsKLSAgICAgICAgTE9HVigicHJldmlld0VuYWJsZWQiKTsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KFBSRVZJRVdfRU5BQkxFRCwgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgcmV0dXJuIHJlcGx5LnJlYWRJbnQzMigpOwotICAgIH0KLQotICAgIC8vIGNoZWNrIHJlY29yZGluZyBzdGF0ZQotICAgIGJvb2wgcmVjb3JkaW5nRW5hYmxlZCgpCi0gICAgewotICAgICAgICBMT0dWKCJyZWNvcmRpbmdFbmFibGVkIik7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElDYW1lcmE6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChSRUNPUkRJTkdfRU5BQkxFRCwgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgcmV0dXJuIHJlcGx5LnJlYWRJbnQzMigpOwotICAgIH0KLQotICAgIC8vIGF1dG8gZm9jdXMKLSAgICBzdGF0dXNfdCBhdXRvRm9jdXMoKQotICAgIHsKLSAgICAgICAgTE9HVigiYXV0b0ZvY3VzIik7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElDYW1lcmE6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChBVVRPX0ZPQ1VTLCBkYXRhLCAmcmVwbHkpOwotICAgICAgICBzdGF0dXNfdCByZXQgPSByZXBseS5yZWFkSW50MzIoKTsKLSAgICAgICAgcmV0dXJuIHJldDsKLSAgICB9Ci0KLSAgICAvLyB0YWtlIGEgcGljdHVyZSAtIHJldHVybnMgYW4gSU1lbW9yeSAocmVmLWNvdW50ZWQgbW1hcCkKLSAgICBzdGF0dXNfdCB0YWtlUGljdHVyZSgpCi0gICAgewotICAgICAgICBMT0dWKCJ0YWtlUGljdHVyZSIpOwotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoVEFLRV9QSUNUVVJFLCBkYXRhLCAmcmVwbHkpOwotICAgICAgICBzdGF0dXNfdCByZXQgPSByZXBseS5yZWFkSW50MzIoKTsKLSAgICAgICAgcmV0dXJuIHJldDsKLSAgICB9Ci0KLSAgICAvLyBzZXQgcHJldmlldy9jYXB0dXJlIHBhcmFtZXRlcnMgLSBrZXkvdmFsdWUgcGFpcnMKLSAgICBzdGF0dXNfdCBzZXRQYXJhbWV0ZXJzKGNvbnN0IFN0cmluZzgmIHBhcmFtcykKLSAgICB7Ci0gICAgICAgIExPR1YoInNldFBhcmFtZXRlcnMiKTsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgZGF0YS53cml0ZVN0cmluZzgocGFyYW1zKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KFNFVF9QQVJBTUVURVJTLCBkYXRhLCAmcmVwbHkpOwotICAgICAgICByZXR1cm4gcmVwbHkucmVhZEludDMyKCk7Ci0gICAgfQotCi0gICAgLy8gZ2V0IHByZXZpZXcvY2FwdHVyZSBwYXJhbWV0ZXJzIC0ga2V5L3ZhbHVlIHBhaXJzCi0gICAgU3RyaW5nOCBnZXRQYXJhbWV0ZXJzKCkgY29uc3QKLSAgICB7Ci0gICAgICAgIExPR1YoImdldFBhcmFtZXRlcnMiKTsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KEdFVF9QQVJBTUVURVJTLCBkYXRhLCAmcmVwbHkpOwotICAgICAgICByZXR1cm4gcmVwbHkucmVhZFN0cmluZzgoKTsKLSAgICB9Ci0gICAgdmlydHVhbCBzdGF0dXNfdCBjb25uZWN0KGNvbnN0IHNwPElDYW1lcmFDbGllbnQ+JiBjYW1lcmFDbGllbnQpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICBkYXRhLndyaXRlU3Ryb25nQmluZGVyKGNhbWVyYUNsaWVudC0+YXNCaW5kZXIoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChDT05ORUNULCBkYXRhLCAmcmVwbHkpOwotICAgICAgICByZXR1cm4gcmVwbHkucmVhZEludDMyKCk7Ci0gICAgfQotICAgIHZpcnR1YWwgc3RhdHVzX3QgbG9jaygpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoTE9DSywgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgcmV0dXJuIHJlcGx5LnJlYWRJbnQzMigpOwotICAgIH0KLSAgICB2aXJ0dWFsIHN0YXR1c190IHVubG9jaygpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoVU5MT0NLLCBkYXRhLCAmcmVwbHkpOwotICAgICAgICByZXR1cm4gcmVwbHkucmVhZEludDMyKCk7Ci0gICAgfQotfTsKLQotSU1QTEVNRU5UX01FVEFfSU5URVJGQUNFKENhbWVyYSwgImFuZHJvaWQuaGFyZHdhcmUuSUNhbWVyYSIpOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNkZWZpbmUgQ0hFQ0tfSU5URVJGQUNFKGludGVyZmFjZSwgZGF0YSwgcmVwbHkpIFwKLSAgICAgICAgZG8geyBpZiAoIWRhdGEuZW5mb3JjZUludGVyZmFjZShpbnRlcmZhY2U6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSkpIHsgXAotICAgICAgICAgICAgTE9HVygiQ2FsbCBpbmNvcnJlY3RseSByb3V0ZWQgdG8gIiAjaW50ZXJmYWNlKTsgXAotICAgICAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOyBcCi0gICAgICAgIH0gfSB3aGlsZSAoMCkKLQotc3RhdHVzX3QgQm5DYW1lcmE6Om9uVHJhbnNhY3QoCi0gICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBzd2l0Y2goY29kZSkgewotICAgICAgICBjYXNlIERJU0NPTk5FQ1Q6IHsKLSAgICAgICAgICAgIExPR1YoIkRJU0NPTk5FQ1QiKTsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBkaXNjb25uZWN0KCk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgU0VUX1BSRVZJRVdfRElTUExBWTogewotICAgICAgICAgICAgTE9HVigiU0VUX1BSRVZJRVdfRElTUExBWSIpOwotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElDYW1lcmEsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIHNwPElTdXJmYWNlPiBzdXJmYWNlID0gaW50ZXJmYWNlX2Nhc3Q8SVN1cmZhY2U+KGRhdGEucmVhZFN0cm9uZ0JpbmRlcigpKTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKHNldFByZXZpZXdEaXNwbGF5KHN1cmZhY2UpKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBTRVRfUFJFVklFV19DQUxMQkFDS19GTEFHOiB7Ci0gICAgICAgICAgICBMT0dWKCJTRVRfUFJFVklFV19DQUxMQkFDS19UWVBFIik7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSUNhbWVyYSwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgaW50IGNhbGxiYWNrX2ZsYWcgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgc2V0UHJldmlld0NhbGxiYWNrRmxhZyhjYWxsYmFja19mbGFnKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBTVEFSVF9QUkVWSUVXOiB7Ci0gICAgICAgICAgICBMT0dWKCJTVEFSVF9QUkVWSUVXIik7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSUNhbWVyYSwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIoc3RhcnRQcmV2aWV3KCkpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFNUQVJUX1JFQ09SRElORzogewotICAgICAgICAgICAgTE9HVigiU1RBUlRfUkVDT1JESU5HIik7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSUNhbWVyYSwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIoc3RhcnRSZWNvcmRpbmcoKSk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgU1RPUF9QUkVWSUVXOiB7Ci0gICAgICAgICAgICBMT0dWKCJTVE9QX1BSRVZJRVciKTsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBzdG9wUHJldmlldygpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFNUT1BfUkVDT1JESU5HOiB7Ci0gICAgICAgICAgICBMT0dWKCJTVE9QX1JFQ09SRElORyIpOwotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElDYW1lcmEsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIHN0b3BSZWNvcmRpbmcoKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBSRUxFQVNFX1JFQ09SRElOR19GUkFNRTogewotICAgICAgICAgICAgTE9HVigiUkVMRUFTRV9SRUNPUkRJTkdfRlJBTUUiKTsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBzcDxJTWVtb3J5PiBtZW0gPSBpbnRlcmZhY2VfY2FzdDxJTWVtb3J5PihkYXRhLnJlYWRTdHJvbmdCaW5kZXIoKSk7Ci0gICAgICAgICAgICByZWxlYXNlUmVjb3JkaW5nRnJhbWUobWVtKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBQUkVWSUVXX0VOQUJMRUQ6IHsKLSAgICAgICAgICAgIExPR1YoIlBSRVZJRVdfRU5BQkxFRCIpOwotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElDYW1lcmEsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKHByZXZpZXdFbmFibGVkKCkpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFJFQ09SRElOR19FTkFCTEVEOiB7Ci0gICAgICAgICAgICBMT0dWKCJSRUNPUkRJTkdfRU5BQkxFRCIpOwotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElDYW1lcmEsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKHJlY29yZGluZ0VuYWJsZWQoKSk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgQVVUT19GT0NVUzogewotICAgICAgICAgICAgTE9HVigiQVVUT19GT0NVUyIpOwotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElDYW1lcmEsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKGF1dG9Gb2N1cygpKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBUQUtFX1BJQ1RVUkU6IHsKLSAgICAgICAgICAgIExPR1YoIlRBS0VfUElDVFVSRSIpOwotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElDYW1lcmEsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKHRha2VQaWN0dXJlKCkpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFNFVF9QQVJBTUVURVJTOiB7Ci0gICAgICAgICAgICBMT0dWKCJTRVRfUEFSQU1FVEVSUyIpOwotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElDYW1lcmEsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIFN0cmluZzggcGFyYW1zKGRhdGEucmVhZFN0cmluZzgoKSk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihzZXRQYXJhbWV0ZXJzKHBhcmFtcykpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBHRVRfUEFSQU1FVEVSUzogewotICAgICAgICAgICAgTE9HVigiR0VUX1BBUkFNRVRFUlMiKTsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICAgcmVwbHktPndyaXRlU3RyaW5nOChnZXRQYXJhbWV0ZXJzKCkpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBDT05ORUNUOiB7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSUNhbWVyYSwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgc3A8SUNhbWVyYUNsaWVudD4gY2FtZXJhQ2xpZW50ID0gaW50ZXJmYWNlX2Nhc3Q8SUNhbWVyYUNsaWVudD4oZGF0YS5yZWFkU3Ryb25nQmluZGVyKCkpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIoY29ubmVjdChjYW1lcmFDbGllbnQpKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBMT0NLOiB7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSUNhbWVyYSwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIobG9jaygpKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBVTkxPQ0s6IHsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMih1bmxvY2soKSk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gQkJpbmRlcjo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCmRpZmYgLS1naXQgYS9saWJzL3VpL0lDYW1lcmFDbGllbnQuY3BwIGIvbGlicy91aS9JQ2FtZXJhQ2xpZW50LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNGJlYzlkMi4uMDAwMDAwMAotLS0gYS9saWJzL3VpL0lDYW1lcmFDbGllbnQuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTg1ICswLDAgQEAKLS8qCi0qKgotKiogQ29weXJpZ2h0IDIwMDgsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotLy8jZGVmaW5lIExPR19OREVCVUcgMAotI2RlZmluZSBMT0dfVEFHICJJQ2FtZXJhQ2xpZW50IgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHVpL0lDYW1lcmFDbGllbnQuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1lbnVtIHsKLSAgICBTSFVUVEVSX0NBTExCQUNLID0gSUJpbmRlcjo6RklSU1RfQ0FMTF9UUkFOU0FDVElPTiwKLSAgICBSQVdfQ0FMTEJBQ0ssCi0gICAgSlBFR19DQUxMQkFDSywKLSAgICBQUkVWSUVXX0NBTExCQUNLLAotICAgIEVSUk9SX0NBTExCQUNLLAotICAgIEFVVE9GT0NVU19DQUxMQkFDSywKLSAgICBSRUNPUkRJTkdfQ0FMTEJBQ0ssCi19OwotCi1jbGFzcyBCcENhbWVyYUNsaWVudDogcHVibGljIEJwSW50ZXJmYWNlPElDYW1lcmFDbGllbnQ+Ci17Ci1wdWJsaWM6Ci0gICAgQnBDYW1lcmFDbGllbnQoY29uc3Qgc3A8SUJpbmRlcj4mIGltcGwpCi0gICAgICAgIDogQnBJbnRlcmZhY2U8SUNhbWVyYUNsaWVudD4oaW1wbCkKLSAgICB7Ci0gICAgfQotCi0gICAgLy8gY2FsbGJhY2sgdG8gbGV0IHRoZSBhcHAga25vdyB0aGUgc2h1dHRlciBoYXMgY2xvc2VkLCBpZGVhbCBmb3IgcGxheWluZyB0aGUgc2h1dHRlciBzb3VuZAotICAgIHZvaWQgc2h1dHRlckNhbGxiYWNrKCkKLSAgICB7Ci0gICAgICAgIExPR1YoInNodXR0ZXJDYWxsYmFjayIpOwotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhQ2xpZW50OjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoU0hVVFRFUl9DQUxMQkFDSywgZGF0YSwgJnJlcGx5LCBJQmluZGVyOjpGTEFHX09ORVdBWSk7Ci0gICAgfQotCi0gICAgLy8gY2FsbGJhY2sgZnJvbSBjYW1lcmEgc2VydmljZSB0byBhcHAgd2l0aCBwaWN0dXJlIGRhdGEKLSAgICB2b2lkIHJhd0NhbGxiYWNrKGNvbnN0IHNwPElNZW1vcnk+JiBwaWN0dXJlKQotICAgIHsKLSAgICAgICAgTE9HVigicmF3Q2FsbGJhY2siKTsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYUNsaWVudDo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgZGF0YS53cml0ZVN0cm9uZ0JpbmRlcihwaWN0dXJlLT5hc0JpbmRlcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KFJBV19DQUxMQkFDSywgZGF0YSwgJnJlcGx5LCBJQmluZGVyOjpGTEFHX09ORVdBWSk7Ci0gICAgfQotCi0gICAgLy8gY2FsbGJhY2sgZnJvbSBjYW1lcmEgc2VydmljZSB0byBhcHAgd2l0aCBwaWN0dXJlIGRhdGEKLSAgICB2b2lkIGpwZWdDYWxsYmFjayhjb25zdCBzcDxJTWVtb3J5PiYgcGljdHVyZSkKLSAgICB7Ci0gICAgICAgIExPR1YoImpwZWdDYWxsYmFjayIpOwotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhQ2xpZW50OjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICBkYXRhLndyaXRlU3Ryb25nQmluZGVyKHBpY3R1cmUtPmFzQmluZGVyKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoSlBFR19DQUxMQkFDSywgZGF0YSwgJnJlcGx5LCBJQmluZGVyOjpGTEFHX09ORVdBWSk7Ci0gICAgfQotCi0gICAgLy8gY2FsbGJhY2sgZnJvbSBjYW1lcmEgc2VydmljZSB0byBhcHAgd2l0aCBwcmV2aWV3IGZyYW1lIGRhdGEKLSAgICB2b2lkIHByZXZpZXdDYWxsYmFjayhjb25zdCBzcDxJTWVtb3J5PiYgZnJhbWUpCi0gICAgewotICAgICAgICBMT0dWKCJwcmV2aWV3Q2FsbGJhY2siKTsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYUNsaWVudDo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgZGF0YS53cml0ZVN0cm9uZ0JpbmRlcihmcmFtZS0+YXNCaW5kZXIoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChQUkVWSUVXX0NBTExCQUNLLCBkYXRhLCAmcmVwbHksIElCaW5kZXI6OkZMQUdfT05FV0FZKTsKLSAgICB9Ci0KLSAgICAvLyBjYWxsYmFjayBmcm9tIGNhbWVyYSBzZXJ2aWNlIHRvIGFwcCB3aXRoIHJlY29yZGluZyBmcmFtZSBkYXRhCi0gICAgdm9pZCByZWNvcmRpbmdDYWxsYmFjayhjb25zdCBzcDxJTWVtb3J5PiYgZnJhbWUpCi0gICAgewotICAgICAgICBMT0dWKCJyZWNvcmRpbmdDYWxsYmFjayIpOwotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhQ2xpZW50OjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICBkYXRhLndyaXRlU3Ryb25nQmluZGVyKGZyYW1lLT5hc0JpbmRlcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KFJFQ09SRElOR19DQUxMQkFDSywgZGF0YSwgJnJlcGx5LCBJQmluZGVyOjpGTEFHX09ORVdBWSk7Ci0gICAgfQotCi0gICAgLy8gY2FsbGJhY2sgZnJvbSBjYW1lcmEgc2VydmljZSB0byBhcHAgdG8gcmVwb3J0IGVycm9yCi0gICAgdm9pZCBlcnJvckNhbGxiYWNrKHN0YXR1c190IGVycm9yKQotICAgIHsKLSAgICAgICAgTE9HVigiZXJyb3JDYWxsYmFjayIpOwotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJQ2FtZXJhQ2xpZW50OjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIoZXJyb3IpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoRVJST1JfQ0FMTEJBQ0ssIGRhdGEsICZyZXBseSwgSUJpbmRlcjo6RkxBR19PTkVXQVkpOwotICAgIH0KLQotICAgIC8vIGNhbGxiYWNrIGZyb20gY2FtZXJhIHNlcnZpY2UgdG8gYXBwIHRvIHJlcG9ydCBhdXRvZm9jdXMgY29tcGxldGlvbgotICAgIHZvaWQgYXV0b0ZvY3VzQ2FsbGJhY2soYm9vbCBmb2N1c2VkKQotICAgIHsKLSAgICAgICAgTE9HVigiYXV0b0ZvY3VzQ2FsbGJhY2siKTsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYUNsaWVudDo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgZGF0YS53cml0ZUludDMyKGZvY3VzZWQpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoQVVUT0ZPQ1VTX0NBTExCQUNLLCBkYXRhLCAmcmVwbHksIElCaW5kZXI6OkZMQUdfT05FV0FZKTsKLSAgICB9Ci19OwotCi1JTVBMRU1FTlRfTUVUQV9JTlRFUkZBQ0UoQ2FtZXJhQ2xpZW50LCAiYW5kcm9pZC5oYXJkd2FyZS5JQ2FtZXJhQ2xpZW50Iik7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2RlZmluZSBDSEVDS19JTlRFUkZBQ0UoaW50ZXJmYWNlLCBkYXRhLCByZXBseSkgXAotICAgICAgICBkbyB7IGlmICghZGF0YS5lbmZvcmNlSW50ZXJmYWNlKGludGVyZmFjZTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKSkgeyBcCi0gICAgICAgICAgICBMT0dXKCJDYWxsIGluY29ycmVjdGx5IHJvdXRlZCB0byAiICNpbnRlcmZhY2UpOyBcCi0gICAgICAgICAgICByZXR1cm4gUEVSTUlTU0lPTl9ERU5JRUQ7IFwKLSAgICAgICAgfSB9IHdoaWxlICgwKQotCi1zdGF0dXNfdCBCbkNhbWVyYUNsaWVudDo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIHN3aXRjaChjb2RlKSB7Ci0gICAgICAgIGNhc2UgU0hVVFRFUl9DQUxMQkFDSzogewotICAgICAgICAgICAgTE9HVigiU0hVVFRFUl9DQUxMQkFDSyIpOwotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElDYW1lcmFDbGllbnQsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIHNodXR0ZXJDYWxsYmFjaygpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFJBV19DQUxMQkFDSzogewotICAgICAgICAgICAgTE9HVigiUkFXX0NBTExCQUNLIik7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSUNhbWVyYUNsaWVudCwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgc3A8SU1lbW9yeT4gcGljdHVyZSA9IGludGVyZmFjZV9jYXN0PElNZW1vcnk+KGRhdGEucmVhZFN0cm9uZ0JpbmRlcigpKTsKLSAgICAgICAgICAgIHJhd0NhbGxiYWNrKHBpY3R1cmUpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIEpQRUdfQ0FMTEJBQ0s6IHsKLSAgICAgICAgICAgIExPR1YoIkpQRUdfQ0FMTEJBQ0siKTsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhQ2xpZW50LCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBzcDxJTWVtb3J5PiBwaWN0dXJlID0gaW50ZXJmYWNlX2Nhc3Q8SU1lbW9yeT4oZGF0YS5yZWFkU3Ryb25nQmluZGVyKCkpOwotICAgICAgICAgICAganBlZ0NhbGxiYWNrKHBpY3R1cmUpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFBSRVZJRVdfQ0FMTEJBQ0s6IHsKLSAgICAgICAgICAgIExPR1YoIlBSRVZJRVdfQ0FMTEJBQ0siKTsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhQ2xpZW50LCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBzcDxJTWVtb3J5PiBmcmFtZSA9IGludGVyZmFjZV9jYXN0PElNZW1vcnk+KGRhdGEucmVhZFN0cm9uZ0JpbmRlcigpKTsKLSAgICAgICAgICAgIHByZXZpZXdDYWxsYmFjayhmcmFtZSk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgUkVDT1JESU5HX0NBTExCQUNLOiB7Ci0gICAgICAgICAgICBMT0dWKCJSRUNPUkRJTkdfQ0FMTEJBQ0siKTsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhQ2xpZW50LCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBzcDxJTWVtb3J5PiBmcmFtZSA9IGludGVyZmFjZV9jYXN0PElNZW1vcnk+KGRhdGEucmVhZFN0cm9uZ0JpbmRlcigpKTsKLSAgICAgICAgICAgIHJlY29yZGluZ0NhbGxiYWNrKGZyYW1lKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBFUlJPUl9DQUxMQkFDSzogewotICAgICAgICAgICAgTE9HVigiRVJST1JfQ0FMTEJBQ0siKTsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhQ2xpZW50LCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBzdGF0dXNfdCBlcnJvciA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBlcnJvckNhbGxiYWNrKGVycm9yKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBBVVRPRk9DVVNfQ0FMTEJBQ0s6IHsKLSAgICAgICAgICAgIExPR1YoIkFVVE9GT0NVU19DQUxMQkFDSyIpOwotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElDYW1lcmFDbGllbnQsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIGJvb2wgZm9jdXNlZCA9IChib29sKWRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBhdXRvRm9jdXNDYWxsYmFjayhmb2N1c2VkKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHJldHVybiBCQmluZGVyOjpvblRyYW5zYWN0KGNvZGUsIGRhdGEsIHJlcGx5LCBmbGFncyk7Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KZGlmZiAtLWdpdCBhL2xpYnMvdWkvSUNhbWVyYVNlcnZpY2UuY3BwIGIvbGlicy91aS9JQ2FtZXJhU2VydmljZS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGU1Njg3ZmUuLjAwMDAwMDAKLS0tIGEvbGlicy91aS9JQ2FtZXJhU2VydmljZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw3NyArMCwwIEBACi0vKgotKioKLSoqIENvcHlyaWdodCAyMDA4LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotI2luY2x1ZGUgPHV0aWxzL0lQQ1RocmVhZFN0YXRlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSVNlcnZpY2VNYW5hZ2VyLmg+Ci0KLSNpbmNsdWRlIDx1aS9JQ2FtZXJhU2VydmljZS5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIEJwQ2FtZXJhU2VydmljZTogcHVibGljIEJwSW50ZXJmYWNlPElDYW1lcmFTZXJ2aWNlPgotewotcHVibGljOgotICAgIEJwQ2FtZXJhU2VydmljZShjb25zdCBzcDxJQmluZGVyPiYgaW1wbCkKLSAgICAgICAgOiBCcEludGVyZmFjZTxJQ2FtZXJhU2VydmljZT4oaW1wbCkKLSAgICB7Ci0gICAgfQotCi0gICAgLy8gY29ubmVjdCB0byBjYW1lcmEgc2VydmljZQotICAgIHZpcnR1YWwgc3A8SUNhbWVyYT4gY29ubmVjdChjb25zdCBzcDxJQ2FtZXJhQ2xpZW50PiYgY2FtZXJhQ2xpZW50KQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSUNhbWVyYVNlcnZpY2U6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGRhdGEud3JpdGVTdHJvbmdCaW5kZXIoY2FtZXJhQ2xpZW50LT5hc0JpbmRlcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KEJuQ2FtZXJhU2VydmljZTo6Q09OTkVDVCwgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgcmV0dXJuIGludGVyZmFjZV9jYXN0PElDYW1lcmE+KHJlcGx5LnJlYWRTdHJvbmdCaW5kZXIoKSk7Ci0gICAgfQotfTsKLQotSU1QTEVNRU5UX01FVEFfSU5URVJGQUNFKENhbWVyYVNlcnZpY2UsICJhbmRyb2lkLmhhcmR3YXJlLklDYW1lcmFTZXJ2aWNlIik7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2RlZmluZSBDSEVDS19JTlRFUkZBQ0UoaW50ZXJmYWNlLCBkYXRhLCByZXBseSkgXAotICAgICAgICBkbyB7IGlmICghZGF0YS5lbmZvcmNlSW50ZXJmYWNlKGludGVyZmFjZTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKSkgeyBcCi0gICAgICAgICAgICBMT0dXKCJDYWxsIGluY29ycmVjdGx5IHJvdXRlZCB0byAiICNpbnRlcmZhY2UpOyBcCi0gICAgICAgICAgICByZXR1cm4gUEVSTUlTU0lPTl9ERU5JRUQ7IFwKLSAgICAgICAgfSB9IHdoaWxlICgwKQotCi1zdGF0dXNfdCBCbkNhbWVyYVNlcnZpY2U6Om9uVHJhbnNhY3QoCi0gICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBzd2l0Y2goY29kZSkgewotICAgICAgICBjYXNlIENPTk5FQ1Q6IHsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJQ2FtZXJhU2VydmljZSwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgc3A8SUNhbWVyYUNsaWVudD4gY2FtZXJhQ2xpZW50ID0gaW50ZXJmYWNlX2Nhc3Q8SUNhbWVyYUNsaWVudD4oZGF0YS5yZWFkU3Ryb25nQmluZGVyKCkpOwotICAgICAgICAgICAgc3A8SUNhbWVyYT4gY2FtZXJhID0gY29ubmVjdChjYW1lcmFDbGllbnQpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlU3Ryb25nQmluZGVyKGNhbWVyYS0+YXNCaW5kZXIoKSk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gQkJpbmRlcjo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCmRpZmYgLS1naXQgYS9saWJzL3VpL0lPdmVybGF5LmNwcCBiL2xpYnMvdWkvSU92ZXJsYXkuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmZWQ0N2MyLi4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvSU92ZXJsYXkuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNzIgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvUGFyY2VsLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSUludGVyZmFjZS5oPgotCi0jaW5jbHVkZSA8dWkvSU92ZXJsYXkuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1lbnVtIHsKLSAgICBERVNUUk9ZID0gSUJpbmRlcjo6RklSU1RfQ0FMTF9UUkFOU0FDVElPTiwgLy8gb25lLXdheSB0cmFuc2FjdGlvbgotfTsKLQotY2xhc3MgQnBPdmVybGF5IDogcHVibGljIEJwSW50ZXJmYWNlPElPdmVybGF5PgotewotcHVibGljOgotICAgIEJwT3ZlcmxheShjb25zdCBzcDxJQmluZGVyPiYgaW1wbCkKLSAgICAgICAgOiBCcEludGVyZmFjZTxJT3ZlcmxheT4oaW1wbCkKLSAgICB7Ci0gICAgfQotCi0gICAgdmlydHVhbCB2b2lkIGRlc3Ryb3koKQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSU92ZXJsYXk6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChERVNUUk9ZLCBkYXRhLCAmcmVwbHksIElCaW5kZXI6OkZMQUdfT05FV0FZKTsKLSAgICB9Ci19OwotCi1JTVBMRU1FTlRfTUVUQV9JTlRFUkZBQ0UoT3ZlcmxheSwgImFuZHJvaWQudWkuSU92ZXJsYXkiKTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZGVmaW5lIENIRUNLX0lOVEVSRkFDRShpbnRlcmZhY2UsIGRhdGEsIHJlcGx5KSBcCi0gICAgICAgIGRvIHsgaWYgKCFkYXRhLmVuZm9yY2VJbnRlcmZhY2UoaW50ZXJmYWNlOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpKSB7IFwKLSAgICAgICAgICAgIExPR1coIkNhbGwgaW5jb3JyZWN0bHkgcm91dGVkIHRvICIgI2ludGVyZmFjZSk7IFwKLSAgICAgICAgICAgIHJldHVybiBQRVJNSVNTSU9OX0RFTklFRDsgXAotICAgICAgICB9IH0gd2hpbGUgKDApCi0KLXN0YXR1c190IEJuT3ZlcmxheTo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIHN3aXRjaChjb2RlKSB7Ci0gICAgICAgIGNhc2UgREVTVFJPWTogewotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElPdmVybGF5LCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBkZXN0cm95KCk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gQkJpbmRlcjo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotICAgIH0KLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvdWkvSVN1cmZhY2UuY3BwIGIvbGlicy91aS9JU3VyZmFjZS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGQ1ZTlmODEuLjAwMDAwMDAKLS0tIGEvbGlicy91aS9JU3VyZmFjZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxNjQgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvUGFyY2VsLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSU1lbW9yeS5oPgotCi0jaW5jbHVkZSA8dWkvSVN1cmZhY2UuaD4KLSNpbmNsdWRlIDx1aS9PdmVybGF5Lmg+Ci0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1JU3VyZmFjZTo6QnVmZmVySGVhcDo6QnVmZmVySGVhcCgpIAotICAgIDogdygwKSwgaCgwKSwgaG9yX3N0cmlkZSgwKSwgdmVyX3N0cmlkZSgwKSwgZm9ybWF0KDApLAotICAgIHRyYW5zZm9ybSgwKSwgZmxhZ3MoMCkgCi17ICAgICAKLX0KLQotSVN1cmZhY2U6OkJ1ZmZlckhlYXA6OkJ1ZmZlckhlYXAodWludDMyX3QgdywgdWludDMyX3QgaCwKLSAgICAgICAgaW50MzJfdCBob3Jfc3RyaWRlLCBpbnQzMl90IHZlcl9zdHJpZGUsCi0gICAgICAgIFBpeGVsRm9ybWF0IGZvcm1hdCwgY29uc3Qgc3A8SU1lbW9yeUhlYXA+JiBoZWFwKQotICAgIDogdyh3KSwgaChoKSwgaG9yX3N0cmlkZShob3Jfc3RyaWRlKSwgdmVyX3N0cmlkZSh2ZXJfc3RyaWRlKSwKLSAgICAgIGZvcm1hdChmb3JtYXQpLCB0cmFuc2Zvcm0oMCksIGZsYWdzKDApLCBoZWFwKGhlYXApIAotewotfQotCi1JU3VyZmFjZTo6QnVmZmVySGVhcDo6QnVmZmVySGVhcCh1aW50MzJfdCB3LCB1aW50MzJfdCBoLAotICAgICAgICBpbnQzMl90IGhvcl9zdHJpZGUsIGludDMyX3QgdmVyX3N0cmlkZSwKLSAgICAgICAgUGl4ZWxGb3JtYXQgZm9ybWF0LCB1aW50MzJfdCB0cmFuc2Zvcm0sIHVpbnQzMl90IGZsYWdzLAotICAgICAgICBjb25zdCBzcDxJTWVtb3J5SGVhcD4mIGhlYXApCi0gICAgICAgIDogdyh3KSwgaChoKSwgaG9yX3N0cmlkZShob3Jfc3RyaWRlKSwgdmVyX3N0cmlkZSh2ZXJfc3RyaWRlKSwKLSAgICAgICAgICBmb3JtYXQoZm9ybWF0KSwgdHJhbnNmb3JtKHRyYW5zZm9ybSksIGZsYWdzKGZsYWdzKSwgaGVhcChoZWFwKSAKLXsKLX0KLQotCi1JU3VyZmFjZTo6QnVmZmVySGVhcDo6fkJ1ZmZlckhlYXAoKSAKLXsgICAgIAotfQotCi1jbGFzcyBCcFN1cmZhY2UgOiBwdWJsaWMgQnBJbnRlcmZhY2U8SVN1cmZhY2U+Ci17Ci1wdWJsaWM6Ci0gICAgQnBTdXJmYWNlKGNvbnN0IHNwPElCaW5kZXI+JiBpbXBsKQotICAgICAgICA6IEJwSW50ZXJmYWNlPElTdXJmYWNlPihpbXBsKQotICAgIHsKLSAgICB9Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190IHJlZ2lzdGVyQnVmZmVycyhjb25zdCBCdWZmZXJIZWFwJiBidWZmZXJzKQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSVN1cmZhY2U6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihidWZmZXJzLncpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIoYnVmZmVycy5oKTsKLSAgICAgICAgZGF0YS53cml0ZUludDMyKGJ1ZmZlcnMuaG9yX3N0cmlkZSk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihidWZmZXJzLnZlcl9zdHJpZGUpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIoYnVmZmVycy5mb3JtYXQpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIoYnVmZmVycy50cmFuc2Zvcm0pOwotICAgICAgICBkYXRhLndyaXRlSW50MzIoYnVmZmVycy5mbGFncyk7Ci0gICAgICAgIGRhdGEud3JpdGVTdHJvbmdCaW5kZXIoYnVmZmVycy5oZWFwLT5hc0JpbmRlcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KFJFR0lTVEVSX0JVRkZFUlMsIGRhdGEsICZyZXBseSk7Ci0gICAgICAgIHN0YXR1c190IHJlc3VsdCA9IHJlcGx5LnJlYWRJbnQzMigpOwotICAgICAgICByZXR1cm4gcmVzdWx0OwotICAgIH0KLQotICAgIHZpcnR1YWwgdm9pZCBwb3N0QnVmZmVyKHNzaXplX3Qgb2Zmc2V0KQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSVN1cmZhY2U6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihvZmZzZXQpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoUE9TVF9CVUZGRVIsIGRhdGEsICZyZXBseSwgSUJpbmRlcjo6RkxBR19PTkVXQVkpOwotICAgIH0KLQotICAgIHZpcnR1YWwgdm9pZCB1bnJlZ2lzdGVyQnVmZmVycygpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU3VyZmFjZTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KFVOUkVHSVNURVJfQlVGRkVSUywgZGF0YSwgJnJlcGx5KTsKLSAgICB9Ci0KLSAgICB2aXJ0dWFsIHNwPE92ZXJsYXlSZWY+IGNyZWF0ZU92ZXJsYXkoCi0gICAgICAgICAgICAgdWludDMyX3QgdywgdWludDMyX3QgaCwgaW50MzJfdCBmb3JtYXQpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU3VyZmFjZTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgZGF0YS53cml0ZUludDMyKHcpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIoaCk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihmb3JtYXQpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoQ1JFQVRFX09WRVJMQVksIGRhdGEsICZyZXBseSk7Ci0gICAgICAgIHJldHVybiBPdmVybGF5UmVmOjpyZWFkRnJvbVBhcmNlbChyZXBseSk7Ci0gICAgfQotfTsKLQotSU1QTEVNRU5UX01FVEFfSU5URVJGQUNFKFN1cmZhY2UsICJhbmRyb2lkLnVpLklTdXJmYWNlIik7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2RlZmluZSBDSEVDS19JTlRFUkZBQ0UoaW50ZXJmYWNlLCBkYXRhLCByZXBseSkgXAotICAgICAgICBkbyB7IGlmICghZGF0YS5lbmZvcmNlSW50ZXJmYWNlKGludGVyZmFjZTo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKSkgeyBcCi0gICAgICAgICAgICBMT0dXKCJDYWxsIGluY29ycmVjdGx5IHJvdXRlZCB0byAiICNpbnRlcmZhY2UpOyBcCi0gICAgICAgICAgICByZXR1cm4gUEVSTUlTU0lPTl9ERU5JRUQ7IFwKLSAgICAgICAgfSB9IHdoaWxlICgwKQotCi1zdGF0dXNfdCBCblN1cmZhY2U6Om9uVHJhbnNhY3QoCi0gICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBzd2l0Y2goY29kZSkgewotICAgICAgICBjYXNlIFJFR0lTVEVSX0JVRkZFUlM6IHsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJU3VyZmFjZSwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgQnVmZmVySGVhcCBidWZmZXI7Ci0gICAgICAgICAgICBidWZmZXIudyA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBidWZmZXIuaCA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBidWZmZXIuaG9yX3N0cmlkZSA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBidWZmZXIudmVyX3N0cmlkZT0gZGF0YS5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgIGJ1ZmZlci5mb3JtYXQgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgYnVmZmVyLnRyYW5zZm9ybSA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBidWZmZXIuZmxhZ3MgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgYnVmZmVyLmhlYXAgPSBpbnRlcmZhY2VfY2FzdDxJTWVtb3J5SGVhcD4oZGF0YS5yZWFkU3Ryb25nQmluZGVyKCkpOwotICAgICAgICAgICAgc3RhdHVzX3QgZXJyID0gcmVnaXN0ZXJCdWZmZXJzKGJ1ZmZlcik7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihlcnIpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFVOUkVHSVNURVJfQlVGRkVSUzogewotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElTdXJmYWNlLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICB1bnJlZ2lzdGVyQnVmZmVycygpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFBPU1RfQlVGRkVSOiB7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSVN1cmZhY2UsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIHNzaXplX3Qgb2Zmc2V0ID0gZGF0YS5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgIHBvc3RCdWZmZXIob2Zmc2V0KTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBDUkVBVEVfT1ZFUkxBWTogewotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElTdXJmYWNlLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBpbnQgdyA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBpbnQgaCA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBpbnQgZiA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBzcDxPdmVybGF5UmVmPiBvID0gY3JlYXRlT3ZlcmxheSh3LCBoLCBmKTsKLSAgICAgICAgICAgIHJldHVybiBPdmVybGF5UmVmOjp3cml0ZVRvUGFyY2VsKHJlcGx5LCBvKTsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHJldHVybiBCQmluZGVyOjpvblRyYW5zYWN0KGNvZGUsIGRhdGEsIHJlcGx5LCBmbGFncyk7Ci0gICAgfQotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91aS9JU3VyZmFjZUNvbXBvc2VyLmNwcCBiL2xpYnMvdWkvSVN1cmZhY2VDb21wb3Nlci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBmZWE2ZjkuLjAwMDAwMDAKLS0tIGEvbGlicy91aS9JU3VyZmFjZUNvbXBvc2VyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDI3NyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8gdGFnIGFzIHN1cmZhY2VmbGluZ2VyCi0jZGVmaW5lIExPR19UQUcgIlN1cmZhY2VGbGluZ2VyIgotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9QYXJjZWwuaD4KLSNpbmNsdWRlIDx1dGlscy9JTWVtb3J5Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvSVBDVGhyZWFkU3RhdGUuaD4KLSNpbmNsdWRlIDx1dGlscy9JU2VydmljZU1hbmFnZXIuaD4KLQotI2luY2x1ZGUgPHVpL0lTdXJmYWNlQ29tcG9zZXIuaD4KLSNpbmNsdWRlIDx1aS9EaXNwbGF5SW5mby5oPgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2RlZmluZSBMSUtFTFkoIGV4cCApICAgICAgIChfX2J1aWx0aW5fZXhwZWN0KCAoZXhwKSAhPSAwLCB0cnVlICApKQotI2RlZmluZSBVTkxJS0VMWSggZXhwICkgICAgIChfX2J1aWx0aW5fZXhwZWN0KCAoZXhwKSAhPSAwLCBmYWxzZSApKQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jbGFzcyBCcFN1cmZhY2VDb21wb3NlciA6IHB1YmxpYyBCcEludGVyZmFjZTxJU3VyZmFjZUNvbXBvc2VyPgotewotcHVibGljOgotICAgIEJwU3VyZmFjZUNvbXBvc2VyKGNvbnN0IHNwPElCaW5kZXI+JiBpbXBsKQotICAgICAgICA6IEJwSW50ZXJmYWNlPElTdXJmYWNlQ29tcG9zZXI+KGltcGwpCi0gICAgewotICAgIH0KLQotICAgIHZpcnR1YWwgc3A8SVN1cmZhY2VGbGluZ2VyQ2xpZW50PiBjcmVhdGVDb25uZWN0aW9uKCkKLSAgICB7Ci0gICAgICAgIHVpbnQzMl90IG47Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElTdXJmYWNlQ29tcG9zZXI6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChCblN1cmZhY2VDb21wb3Nlcjo6Q1JFQVRFX0NPTk5FQ1RJT04sIGRhdGEsICZyZXBseSk7Ci0gICAgICAgIHJldHVybiBpbnRlcmZhY2VfY2FzdDxJU3VyZmFjZUZsaW5nZXJDbGllbnQ+KHJlcGx5LnJlYWRTdHJvbmdCaW5kZXIoKSk7Ci0gICAgfQotCi0gICAgdmlydHVhbCBzcDxJTWVtb3J5PiBnZXRDYmxrKCkgY29uc3QKLSAgICB7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElTdXJmYWNlQ29tcG9zZXI6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChCblN1cmZhY2VDb21wb3Nlcjo6R0VUX0NCTEssIGRhdGEsICZyZXBseSk7Ci0gICAgICAgIHJldHVybiBpbnRlcmZhY2VfY2FzdDxJTWVtb3J5PihyZXBseS5yZWFkU3Ryb25nQmluZGVyKCkpOwotICAgIH0KLQotICAgIHZpcnR1YWwgdm9pZCBvcGVuR2xvYmFsVHJhbnNhY3Rpb24oKQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSVN1cmZhY2VDb21wb3Nlcjo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KEJuU3VyZmFjZUNvbXBvc2VyOjpPUEVOX0dMT0JBTF9UUkFOU0FDVElPTiwgZGF0YSwgJnJlcGx5KTsKLSAgICB9Ci0KLSAgICB2aXJ0dWFsIHZvaWQgY2xvc2VHbG9iYWxUcmFuc2FjdGlvbigpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU3VyZmFjZUNvbXBvc2VyOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoQm5TdXJmYWNlQ29tcG9zZXI6OkNMT1NFX0dMT0JBTF9UUkFOU0FDVElPTiwgZGF0YSwgJnJlcGx5KTsKLSAgICB9Ci0KLSAgICB2aXJ0dWFsIHN0YXR1c190IGZyZWV6ZURpc3BsYXkoRGlzcGxheUlEIGRweSwgdWludDMyX3QgZmxhZ3MpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU3VyZmFjZUNvbXBvc2VyOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIoZHB5KTsKLSAgICAgICAgZGF0YS53cml0ZUludDMyKGZsYWdzKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KEJuU3VyZmFjZUNvbXBvc2VyOjpGUkVFWkVfRElTUExBWSwgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgcmV0dXJuIHJlcGx5LnJlYWRJbnQzMigpOwotICAgIH0KLQotICAgIHZpcnR1YWwgc3RhdHVzX3QgdW5mcmVlemVEaXNwbGF5KERpc3BsYXlJRCBkcHksIHVpbnQzMl90IGZsYWdzKQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSVN1cmZhY2VDb21wb3Nlcjo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgZGF0YS53cml0ZUludDMyKGRweSk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihmbGFncyk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChCblN1cmZhY2VDb21wb3Nlcjo6VU5GUkVFWkVfRElTUExBWSwgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgcmV0dXJuIHJlcGx5LnJlYWRJbnQzMigpOwotICAgIH0KLQotICAgIHZpcnR1YWwgaW50IHNldE9yaWVudGF0aW9uKERpc3BsYXlJRCBkcHksIGludCBvcmllbnRhdGlvbikKLSAgICB7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElTdXJmYWNlQ29tcG9zZXI6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihkcHkpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIob3JpZW50YXRpb24pOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoQm5TdXJmYWNlQ29tcG9zZXI6OlNFVF9PUklFTlRBVElPTiwgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgcmV0dXJuIHJlcGx5LnJlYWRJbnQzMigpOwotICAgIH0KLQotICAgIHZpcnR1YWwgdm9pZCBib290RmluaXNoZWQoKQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSVN1cmZhY2VDb21wb3Nlcjo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KEJuU3VyZmFjZUNvbXBvc2VyOjpCT09UX0ZJTklTSEVELCBkYXRhLCAmcmVwbHkpOwotICAgIH0KLQotICAgIHZpcnR1YWwgc3RhdHVzX3QgcmVxdWVzdEdQVSgKLSAgICAgICAgICAgIGNvbnN0IHNwPElHUFVDYWxsYmFjaz4mIGNhbGxiYWNrLCBncHVfaW5mb190KiBncHUpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU3VyZmFjZUNvbXBvc2VyOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICBkYXRhLndyaXRlU3Ryb25nQmluZGVyKGNhbGxiYWNrLT5hc0JpbmRlcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KEJuU3VyZmFjZUNvbXBvc2VyOjpSRVFVRVNUX0dQVSwgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgZ3B1LT5yZWdzID0gaW50ZXJmYWNlX2Nhc3Q8SU1lbW9yeT4ocmVwbHkucmVhZFN0cm9uZ0JpbmRlcigpKTsKLSAgICAgICAgZ3B1LT5jb3VudCA9IHJlcGx5LnJlYWRJbnQzMigpOwotCi0gICAgICAgIC8vIEZJWE1FOiBmb3Igbm93LCB3ZSBkb24ndCBkeW5hbWljYWxseSBhbGxvY2F0ZSB0aGUgcmVnaW9ucyBhcnJheQotICAgICAgICBzaXplX3QgbWF4Q291bnQgPSBzaXplb2YoZ3B1LT5yZWdpb25zKS9zaXplb2YoKmdwdS0+cmVnaW9ucyk7Ci0gICAgICAgIGlmIChncHUtPmNvdW50ID4gbWF4Q291bnQpCi0gICAgICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOwotCi0gICAgICAgIGZvciAoc2l6ZV90IGk9MCA7IGk8Z3B1LT5jb3VudCA7IGkrKykgewotICAgICAgICAgICAgZ3B1LT5yZWdpb25zW2ldLnJlZ2lvbiA9IGludGVyZmFjZV9jYXN0PElNZW1vcnk+KHJlcGx5LnJlYWRTdHJvbmdCaW5kZXIoKSk7Ci0gICAgICAgICAgICBncHUtPnJlZ2lvbnNbaV0ucmVzZXJ2ZWQgPSByZXBseS5yZWFkSW50MzIoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcmVwbHkucmVhZEludDMyKCk7Ci0gICAgfQotCi0gICAgdmlydHVhbCBzdGF0dXNfdCByZXZva2VHUFUoKQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSVN1cmZhY2VDb21wb3Nlcjo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KEJuU3VyZmFjZUNvbXBvc2VyOjpSRVZPS0VfR1BVLCBkYXRhLCAmcmVwbHkpOwotICAgICAgICByZXR1cm4gcmVwbHkucmVhZEludDMyKCk7Ci0gICAgfQotCi0gICAgdmlydHVhbCB2b2lkIHNpZ25hbCgpIGNvbnN0Ci0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU3VyZmFjZUNvbXBvc2VyOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoQm5TdXJmYWNlQ29tcG9zZXI6OlNJR05BTCwgZGF0YSwgJnJlcGx5LCBJQmluZGVyOjpGTEFHX09ORVdBWSk7Ci0gICAgfQotfTsKLQotSU1QTEVNRU5UX01FVEFfSU5URVJGQUNFKFN1cmZhY2VDb21wb3NlciwgImFuZHJvaWQudWkuSVN1cmZhY2VDb21wb3NlciIpOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNkZWZpbmUgQ0hFQ0tfSU5URVJGQUNFKGludGVyZmFjZSwgZGF0YSwgcmVwbHkpIFwKLSAgICAgICAgZG8geyBpZiAoIWRhdGEuZW5mb3JjZUludGVyZmFjZShpbnRlcmZhY2U6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSkpIHsgXAotICAgICAgICAgICAgTE9HVygiQ2FsbCBpbmNvcnJlY3RseSByb3V0ZWQgdG8gIiAjaW50ZXJmYWNlKTsgXAotICAgICAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOyBcCi0gICAgICAgIH0gfSB3aGlsZSAoMCkKLQotc3RhdHVzX3QgQm5TdXJmYWNlQ29tcG9zZXI6Om9uVHJhbnNhY3QoCi0gICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBzdGF0dXNfdCBlcnIgPSBCbkludGVyZmFjZTxJU3VyZmFjZUNvbXBvc2VyPjo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotICAgIGlmIChlcnIgPT0gTk9fRVJST1IpCi0gICAgICAgIHJldHVybiBlcnI7Ci0KLSAgICBDSEVDS19JTlRFUkZBQ0UoSVN1cmZhY2VDb21wb3NlciwgZGF0YSwgcmVwbHkpOwotCi0gICAgc3dpdGNoKGNvZGUpIHsKLSAgICAgICAgY2FzZSBDUkVBVEVfQ09OTkVDVElPTjogewotICAgICAgICAgICAgc3A8SUJpbmRlcj4gYiA9IGNyZWF0ZUNvbm5lY3Rpb24oKS0+YXNCaW5kZXIoKTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZVN0cm9uZ0JpbmRlcihiKTsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBPUEVOX0dMT0JBTF9UUkFOU0FDVElPTjogewotICAgICAgICAgICAgb3Blbkdsb2JhbFRyYW5zYWN0aW9uKCk7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgQ0xPU0VfR0xPQkFMX1RSQU5TQUNUSU9OOiB7Ci0gICAgICAgICAgICBjbG9zZUdsb2JhbFRyYW5zYWN0aW9uKCk7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgU0VUX09SSUVOVEFUSU9OOiB7Ci0gICAgICAgICAgICBEaXNwbGF5SUQgZHB5ID0gZGF0YS5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgIGludCBvcmllbnRhdGlvbiA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMiggc2V0T3JpZW50YXRpb24oZHB5LCBvcmllbnRhdGlvbikgKTsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBGUkVFWkVfRElTUExBWTogewotICAgICAgICAgICAgRGlzcGxheUlEIGRweSA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMiggZnJlZXplRGlzcGxheShkcHksIGZsYWdzKSApOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFVORlJFRVpFX0RJU1BMQVk6IHsKLSAgICAgICAgICAgIERpc3BsYXlJRCBkcHkgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgdWludDMyX3QgZmxhZ3MgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIoIHVuZnJlZXplRGlzcGxheShkcHksIGZsYWdzKSApOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIEJPT1RfRklOSVNIRUQ6IHsKLSAgICAgICAgICAgIGJvb3RGaW5pc2hlZCgpOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFJFVk9LRV9HUFU6IHsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKCByZXZva2VHUFUoKSApOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIFNJR05BTDogewotICAgICAgICAgICAgc2lnbmFsKCk7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgR0VUX0NCTEs6IHsKLSAgICAgICAgICAgIHNwPElCaW5kZXI+IGIgPSBnZXRDYmxrKCktPmFzQmluZGVyKCk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVTdHJvbmdCaW5kZXIoYik7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgUkVRVUVTVF9HUFU6IHsKLSAgICAgICAgICAgIC8vIFRPRE86IHRoaXMgc2hvdWxkIGJlIHByb3RlY3RlZCBieSBhIHBlcm1pc3Npb24KLSAgICAgICAgICAgIGdwdV9pbmZvX3QgaW5mbzsKLSAgICAgICAgICAgIHNwPElHUFVDYWxsYmFjaz4gY2FsbGJhY2sKLSAgICAgICAgICAgICAgICA9IGludGVyZmFjZV9jYXN0PElHUFVDYWxsYmFjaz4oZGF0YS5yZWFkU3Ryb25nQmluZGVyKCkpOwotICAgICAgICAgICAgc3RhdHVzX3QgcmVzID0gcmVxdWVzdEdQVShjYWxsYmFjaywgJmluZm8pOwotCi0gICAgICAgICAgICAvLyBGSVhNRTogZm9yIG5vdywgd2UgZG9uJ3QgZHluYW1pY2FsbHkgYWxsb2NhdGUgdGhlIHJlZ2lvbnMgYXJyYXkKLSAgICAgICAgICAgIHNpemVfdCBtYXhDb3VudCA9IHNpemVvZihpbmZvLnJlZ2lvbnMpL3NpemVvZigqaW5mby5yZWdpb25zKTsKLSAgICAgICAgICAgIGlmIChpbmZvLmNvdW50ID4gbWF4Q291bnQpCi0gICAgICAgICAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLQotICAgICAgICAgICAgcmVwbHktPndyaXRlU3Ryb25nQmluZGVyKGluZm8ucmVncy0+YXNCaW5kZXIoKSk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihpbmZvLmNvdW50KTsKLSAgICAgICAgICAgIGZvciAoc2l6ZV90IGk9MCA7IGk8aW5mby5jb3VudCA7IGkrKykgewotICAgICAgICAgICAgICAgIHJlcGx5LT53cml0ZVN0cm9uZ0JpbmRlcihpbmZvLnJlZ2lvbnNbaV0ucmVnaW9uLT5hc0JpbmRlcigpKTsKLSAgICAgICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihpbmZvLnJlZ2lvbnNbaV0ucmVzZXJ2ZWQpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIocmVzKTsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHJldHVybiBVTktOT1dOX1RSQU5TQUNUSU9OOwotICAgIH0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotZW51bSB7Ci0gICAgLy8gTm90ZTogQk9PVF9GSU5JU0hFRCBtdXN0IHJlbWFpbiB0aGlzIHZhbHVlLCBpdCBpcyBjYWxsZWQgYnkgQWN0aXZpdHlNYW5hZ2VyU2VydmljZS4KLSAgICBHUFVfTE9TVCA9IElCaW5kZXI6OkZJUlNUX0NBTExfVFJBTlNBQ1RJT04KLX07Ci0KLWNsYXNzIEJwR1BVQ2FsbGJhY2sgOiBwdWJsaWMgQnBJbnRlcmZhY2U8SUdQVUNhbGxiYWNrPgotewotcHVibGljOgotICAgIEJwR1BVQ2FsbGJhY2soY29uc3Qgc3A8SUJpbmRlcj4mIGltcGwpCi0gICAgICAgIDogQnBJbnRlcmZhY2U8SUdQVUNhbGxiYWNrPihpbXBsKQotICAgIHsKLSAgICB9Ci0KLSAgICB2aXJ0dWFsIHZvaWQgZ3B1TG9zdCgpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJR1BVQ2FsbGJhY2s6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChHUFVfTE9TVCwgZGF0YSwgJnJlcGx5LCBJQmluZGVyOjpGTEFHX09ORVdBWSk7Ci0gICAgfQotfTsKLQotSU1QTEVNRU5UX01FVEFfSU5URVJGQUNFKEdQVUNhbGxiYWNrLCAiYW5kcm9pZC51aS5JR1BVQ2FsbGJhY2siKTsKLQotc3RhdHVzX3QgQm5HUFVDYWxsYmFjazo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIHN3aXRjaChjb2RlKSB7Ci0gICAgICAgIGNhc2UgR1BVX0xPU1Q6IHsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJR1BVQ2FsbGJhY2ssIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIGdwdUxvc3QoKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHJldHVybiBCQmluZGVyOjpvblRyYW5zYWN0KGNvZGUsIGRhdGEsIHJlcGx5LCBmbGFncyk7Ci0gICAgfQotfQotCi19OwpkaWZmIC0tZ2l0IGEvbGlicy91aS9JU3VyZmFjZUZsaW5nZXJDbGllbnQuY3BwIGIvbGlicy91aS9JU3VyZmFjZUZsaW5nZXJDbGllbnQuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkZDZhNzk4Li4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvSVN1cmZhY2VGbGluZ2VyQ2xpZW50LmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDIwOCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8gdGFnIGFzIHN1cmZhY2VmbGluZ2VyCi0jZGVmaW5lIExPR19UQUcgIlN1cmZhY2VGbGluZ2VyIgotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotI2luY2x1ZGUgPHV0aWxzL0lNZW1vcnkuaD4KLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL0lTZXJ2aWNlTWFuYWdlci5oPgotCi0jaW5jbHVkZSA8dWkvSVN1cmZhY2UuaD4KLSNpbmNsdWRlIDx1aS9JU3VyZmFjZUZsaW5nZXJDbGllbnQuaD4KLSNpbmNsdWRlIDx1aS9Qb2ludC5oPgotI2luY2x1ZGUgPHVpL1JlY3QuaD4KLQotI2luY2x1ZGUgPHByaXZhdGUvdWkvTGF5ZXJTdGF0ZS5oPgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2RlZmluZSBMSUtFTFkoIGV4cCApICAgICAgIChfX2J1aWx0aW5fZXhwZWN0KCAoZXhwKSAhPSAwLCB0cnVlICApKQotI2RlZmluZSBVTkxJS0VMWSggZXhwICkgICAgIChfX2J1aWx0aW5fZXhwZWN0KCAoZXhwKSAhPSAwLCBmYWxzZSApKQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1lbnVtIHsKLSAgICBHRVRfQ0JMSyA9IElCaW5kZXI6OkZJUlNUX0NBTExfVFJBTlNBQ1RJT04sCi0gICAgQ1JFQVRFX1NVUkZBQ0UsCi0gICAgREVTVFJPWV9TVVJGQUNFLAotICAgIFNFVF9TVEFURQotfTsKLQotY2xhc3MgQnBTdXJmYWNlRmxpbmdlckNsaWVudCA6IHB1YmxpYyBCcEludGVyZmFjZTxJU3VyZmFjZUZsaW5nZXJDbGllbnQ+Ci17Ci1wdWJsaWM6Ci0gICAgQnBTdXJmYWNlRmxpbmdlckNsaWVudChjb25zdCBzcDxJQmluZGVyPiYgaW1wbCkKLSAgICAgICAgOiBCcEludGVyZmFjZTxJU3VyZmFjZUZsaW5nZXJDbGllbnQ+KGltcGwpCi0gICAgewotICAgIH0KLQotICAgIHZpcnR1YWwgdm9pZCBnZXRDb250cm9sQmxvY2tzKHNwPElNZW1vcnk+KiBjdGwpIGNvbnN0Ci0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU3VyZmFjZUZsaW5nZXJDbGllbnQ6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChHRVRfQ0JMSywgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgKmN0bCAgPSBpbnRlcmZhY2VfY2FzdDxJTWVtb3J5PihyZXBseS5yZWFkU3Ryb25nQmluZGVyKCkpOwotICAgIH0KLQotICAgIHZpcnR1YWwgc3A8SVN1cmZhY2U+IGNyZWF0ZVN1cmZhY2UoIHN1cmZhY2VfZGF0YV90KiBwYXJhbXMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IHBpZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEaXNwbGF5SUQgZGlzcGxheSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCB3LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGgsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUGl4ZWxGb3JtYXQgZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGZsYWdzKQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSVN1cmZhY2VGbGluZ2VyQ2xpZW50OjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIocGlkKTsKLSAgICAgICAgZGF0YS53cml0ZUludDMyKGRpc3BsYXkpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIodyk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihoKTsKLSAgICAgICAgZGF0YS53cml0ZUludDMyKGZvcm1hdCk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihmbGFncyk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChDUkVBVEVfU1VSRkFDRSwgZGF0YSwgJnJlcGx5KTsKLSAgICAgICAgcGFyYW1zLT5yZWFkRnJvbVBhcmNlbChyZXBseSk7Ci0gICAgICAgIHJldHVybiBpbnRlcmZhY2VfY2FzdDxJU3VyZmFjZT4ocmVwbHkucmVhZFN0cm9uZ0JpbmRlcigpKTsKLSAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKLSAgICB2aXJ0dWFsIHN0YXR1c190IGRlc3Ryb3lTdXJmYWNlKFN1cmZhY2VJRCBzaWQpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU3VyZmFjZUZsaW5nZXJDbGllbnQ6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihzaWQpOwotICAgICAgICByZW1vdGUoKS0+dHJhbnNhY3QoREVTVFJPWV9TVVJGQUNFLCBkYXRhLCAmcmVwbHkpOwotICAgICAgICByZXR1cm4gcmVwbHkucmVhZEludDMyKCk7Ci0gICAgfQotCi0gICAgdmlydHVhbCBzdGF0dXNfdCBzZXRTdGF0ZShpbnQzMl90IGNvdW50LCBjb25zdCBsYXllcl9zdGF0ZV90KiBzdGF0ZXMpCi0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU3VyZmFjZUZsaW5nZXJDbGllbnQ6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihjb3VudCk7Ci0gICAgICAgIGZvciAoaW50IGk9MCA7IGk8Y291bnQgOyBpKyspCi0gICAgICAgICAgICBzdGF0ZXNbaV0ud3JpdGUoZGF0YSk7Ci0gICAgICAgIHJlbW90ZSgpLT50cmFuc2FjdChTRVRfU1RBVEUsIGRhdGEsICZyZXBseSk7Ci0gICAgICAgIHJldHVybiByZXBseS5yZWFkSW50MzIoKTsKLSAgICB9Ci19OwotCi1JTVBMRU1FTlRfTUVUQV9JTlRFUkZBQ0UoU3VyZmFjZUZsaW5nZXJDbGllbnQsICJhbmRyb2lkLnVpLklTdXJmYWNlRmxpbmdlckNsaWVudCIpOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNkZWZpbmUgQ0hFQ0tfSU5URVJGQUNFKGludGVyZmFjZSwgZGF0YSwgcmVwbHkpIFwKLSAgICAgICAgZG8geyBpZiAoIWRhdGEuZW5mb3JjZUludGVyZmFjZShpbnRlcmZhY2U6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSkpIHsgXAotICAgICAgICAgICAgTE9HVygiQ2FsbCBpbmNvcnJlY3RseSByb3V0ZWQgdG8gIiAjaW50ZXJmYWNlKTsgXAotICAgICAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOyBcCi0gICAgICAgIH0gfSB3aGlsZSAoMCkKLQotc3RhdHVzX3QgQm5TdXJmYWNlRmxpbmdlckNsaWVudDo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIC8vIGNvZGVzIHRoYXQgZG9uJ3QgcmVxdWlyZSBwZXJtaXNzaW9uIGNoZWNrCi0KLSAgICBzd2l0Y2goY29kZSkgewotICAgICAgICBjYXNlIEdFVF9DQkxLOiB7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSVN1cmZhY2VGbGluZ2VyQ2xpZW50LCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBzcDxJTWVtb3J5PiBjdGw7Ci0gICAgICAgICAgICBnZXRDb250cm9sQmxvY2tzKCZjdGwpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlU3Ryb25nQmluZGVyKGN0bC0+YXNCaW5kZXIoKSk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgfQotCi0gICAgLy8gdGhlc2UgbXVzdCBiZSBjaGVja2VkCi0gICAgIAotICAgICBJUENUaHJlYWRTdGF0ZSogaXBjID0gSVBDVGhyZWFkU3RhdGU6OnNlbGYoKTsKLSAgICAgY29uc3QgaW50IHBpZCA9IGlwYy0+Z2V0Q2FsbGluZ1BpZCgpOwotICAgICBjb25zdCBpbnQgc2VsZl9waWQgICAgPSBnZXRwaWQoKTsKLSAgICAgaWYgKFVOTElLRUxZKHBpZCAhPSBzZWxmX3BpZCkpIHsKLSAgICAgICAgIC8vIHdlJ3JlIGNhbGxlZCBmcm9tIGEgZGlmZmVyZW50IHByb2Nlc3MsIGRvIHRoZSByZWFsIGNoZWNrCi0gICAgICAgICBpZiAoIWNoZWNrQ2FsbGluZ1Blcm1pc3Npb24oCi0gICAgICAgICAgICAgICAgIFN0cmluZzE2KCJhbmRyb2lkLnBlcm1pc3Npb24uQUNDRVNTX1NVUkZBQ0VfRkxJTkdFUiIpKSkKLSAgICAgICAgIHsKLSAgICAgICAgICAgICBjb25zdCBpbnQgdWlkID0gaXBjLT5nZXRDYWxsaW5nVWlkKCk7Ci0gICAgICAgICAgICAgTE9HRSgiUGVybWlzc2lvbiBEZW5pYWw6ICIKLSAgICAgICAgICAgICAgICAgICAgICJjYW4ndCBvcGVuR2xvYmFsVHJhbnNhY3Rpb24gcGlkPSVkLCB1aWQ9JWQiLCBwaWQsIHVpZCk7Ci0gICAgICAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOwotICAgICAgICAgfQotICAgICB9Ci0gICAKLSAgICAgc3dpdGNoKGNvZGUpIHsKLSAgICAgICAgY2FzZSBDUkVBVEVfU1VSRkFDRTogewotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElTdXJmYWNlRmxpbmdlckNsaWVudCwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgc3VyZmFjZV9kYXRhX3QgcGFyYW1zOwotICAgICAgICAgICAgaW50MzJfdCBwaWQgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgRGlzcGxheUlEIGRpc3BsYXkgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgdWludDMyX3QgdyA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICB1aW50MzJfdCBoID0gZGF0YS5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgIFBpeGVsRm9ybWF0IGZvcm1hdCA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICB1aW50MzJfdCBmbGFncyA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBzcDxJU3VyZmFjZT4gcyA9IGNyZWF0ZVN1cmZhY2UoJnBhcmFtcywgcGlkLCBkaXNwbGF5LCB3LCBoLCBmb3JtYXQsIGZsYWdzKTsKLSAgICAgICAgICAgIHBhcmFtcy53cml0ZVRvUGFyY2VsKHJlcGx5KTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZVN0cm9uZ0JpbmRlcihzLT5hc0JpbmRlcigpKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBERVNUUk9ZX1NVUkZBQ0U6IHsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJU3VyZmFjZUZsaW5nZXJDbGllbnQsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKCBkZXN0cm95U3VyZmFjZSggZGF0YS5yZWFkSW50MzIoKSApICk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgU0VUX1NUQVRFOiB7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSVN1cmZhY2VGbGluZ2VyQ2xpZW50LCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBpbnQzMl90IGNvdW50ID0gZGF0YS5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgIGxheWVyX3N0YXRlX3QqIHN0YXRlcyA9IG5ldyBsYXllcl9zdGF0ZV90W2NvdW50XTsKLSAgICAgICAgICAgIGZvciAoaW50IGk9MCA7IGk8Y291bnQgOyBpKyspCi0gICAgICAgICAgICAgICAgc3RhdGVzW2ldLnJlYWQoZGF0YSk7Ci0gICAgICAgICAgICBzdGF0dXNfdCBlcnIgPSBzZXRTdGF0ZShjb3VudCwgc3RhdGVzKTsKLSAgICAgICAgICAgIGRlbGV0ZSBbXSBzdGF0ZXM7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihlcnIpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgcmV0dXJuIEJCaW5kZXI6Om9uVHJhbnNhY3QoY29kZSwgZGF0YSwgcmVwbHksIGZsYWdzKTsKLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RhdHVzX3QgSVN1cmZhY2VGbGluZ2VyQ2xpZW50OjpzdXJmYWNlX2RhdGFfdDo6cmVhZEZyb21QYXJjZWwoY29uc3QgUGFyY2VsJiBwYXJjZWwpCi17Ci0gICAgdG9rZW4gPSBwYXJjZWwucmVhZEludDMyKCk7Ci0gICAgaWRlbnRpdHkgID0gcGFyY2VsLnJlYWRJbnQzMigpOwotICAgIGhlYXBbMF0gPSBpbnRlcmZhY2VfY2FzdDxJTWVtb3J5SGVhcD4ocGFyY2VsLnJlYWRTdHJvbmdCaW5kZXIoKSk7Ci0gICAgaGVhcFsxXSA9IGludGVyZmFjZV9jYXN0PElNZW1vcnlIZWFwPihwYXJjZWwucmVhZFN0cm9uZ0JpbmRlcigpKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IElTdXJmYWNlRmxpbmdlckNsaWVudDo6c3VyZmFjZV9kYXRhX3Q6OndyaXRlVG9QYXJjZWwoUGFyY2VsKiBwYXJjZWwpIGNvbnN0Ci17Ci0gICAgcGFyY2VsLT53cml0ZUludDMyKHRva2VuKTsKLSAgICBwYXJjZWwtPndyaXRlSW50MzIoaWRlbnRpdHkpOwotICAgIHBhcmNlbC0+d3JpdGVTdHJvbmdCaW5kZXIoaGVhcFswXSE9MCA/IGhlYXBbMF0tPmFzQmluZGVyKCkgOiBOVUxMKTsKLSAgICBwYXJjZWwtPndyaXRlU3Ryb25nQmluZGVyKGhlYXBbMV0hPTAgPyBoZWFwWzFdLT5hc0JpbmRlcigpIDogTlVMTCk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91aS9LZXlDaGFyYWN0ZXJNYXAuY3BwIGIvbGlicy91aS9LZXlDaGFyYWN0ZXJNYXAuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlODkxMTgxLi4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvS2V5Q2hhcmFjdGVyTWFwLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDI2MyArMCwwIEBACi0jZGVmaW5lIExPR19UQUcgIktleUNoYXJhY3Rlck1hcCIKLQotI2luY2x1ZGUgPHVpL0tleUNoYXJhY3Rlck1hcC5oPgotI2luY2x1ZGUgPGN1dGlscy9wcm9wZXJ0aWVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPGxpbWl0cy5oPgotI2luY2x1ZGUgPHN0cmluZy5oPgotCi1zdHJ1Y3QgSGVhZGVyCi17Ci0gICAgY2hhciBtYWdpY1s4XTsKLSAgICB1bnNpZ25lZCBpbnQgZW5kaWFuOwotICAgIHVuc2lnbmVkIGludCB2ZXJzaW9uOwotICAgIHVuc2lnbmVkIGludCBrZXljb3VudDsKLSAgICB1bnNpZ25lZCBjaGFyIGtiZHR5cGU7Ci0gICAgY2hhciBwYWRkaW5nWzExXTsKLX07Ci0KLUtleUNoYXJhY3Rlck1hcDo6S2V5Q2hhcmFjdGVyTWFwKCkKLXsKLX0KLQotS2V5Q2hhcmFjdGVyTWFwOjp+S2V5Q2hhcmFjdGVyTWFwKCkKLXsKLSAgICBmcmVlKG1fa2V5cyk7Ci19Ci0KLXVuc2lnbmVkIHNob3J0Ci1LZXlDaGFyYWN0ZXJNYXA6OmdldChpbnQga2V5Y29kZSwgaW50IG1ldGEpCi17Ci0gICAgS2V5KiBrID0gZmluZF9rZXkoa2V5Y29kZSk7Ci0gICAgaWYgKGsgIT0gTlVMTCkgewotICAgICAgICByZXR1cm4gay0+ZGF0YVttZXRhICYgTUVUQV9NQVNLXTsKLSAgICB9Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXVuc2lnbmVkIHNob3J0Ci1LZXlDaGFyYWN0ZXJNYXA6OmdldE51bWJlcihpbnQga2V5Y29kZSkKLXsKLSAgICBLZXkqIGsgPSBmaW5kX2tleShrZXljb2RlKTsKLSAgICBpZiAoayAhPSBOVUxMKSB7Ci0gICAgICAgIHJldHVybiBrLT5udW1iZXI7Ci0gICAgfQotICAgIHJldHVybiAwOwotfQotCi11bnNpZ25lZCBzaG9ydAotS2V5Q2hhcmFjdGVyTWFwOjpnZXRNYXRjaChpbnQga2V5Y29kZSwgY29uc3QgdW5zaWduZWQgc2hvcnQqIGNoYXJzLAotICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgY2hhcnNpemUsIHVpbnQzMl90IG1vZGlmaWVycykKLXsKLSAgICBLZXkqIGsgPSBmaW5kX2tleShrZXljb2RlKTsKLSAgICBtb2RpZmllcnMgJj0gMzsgLy8gaWdub3JlIHRoZSBTWU0ga2V5IGJlY2F1c2Ugd2UgZG9uJ3QgaGF2ZSBrZXltYXAgZW50cmllcyBmb3IgaXQKLSAgICBpZiAoayAhPSBOVUxMKSB7Ci0gICAgICAgIGNvbnN0IHVpbnQxNl90KiBkYXRhID0gay0+ZGF0YTsKLSAgICAgICAgZm9yIChpbnQgaj0wOyBqPGNoYXJzaXplOyBqKyspIHsKLSAgICAgICAgICAgIHVpbnQxNl90IGMgPSBjaGFyc1tqXTsKLSAgICAgICAgICAgIGZvciAoaW50IGk9MDsgaTwoTUVUQV9NQVNLICsgMSk7IGkrKykgewotICAgICAgICAgICAgICAgIGlmICgobW9kaWZpZXJzID09IDApIHx8ICgobW9kaWZpZXJzICYgaSkgIT0gMCkpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGMgPT0gZGF0YVtpXSkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGM7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXVuc2lnbmVkIHNob3J0Ci1LZXlDaGFyYWN0ZXJNYXA6OmdldERpc3BsYXlMYWJlbChpbnQga2V5Y29kZSkKLXsKLSAgICBLZXkqIGsgPSBmaW5kX2tleShrZXljb2RlKTsKLSAgICBpZiAoayAhPSBOVUxMKSB7Ci0gICAgICAgIHJldHVybiBrLT5kaXNwbGF5X2xhYmVsOwotICAgIH0KLSAgICByZXR1cm4gMDsKLX0KLQotYm9vbAotS2V5Q2hhcmFjdGVyTWFwOjpnZXRLZXlEYXRhKGludCBrZXljb2RlLCB1bnNpZ25lZCBzaG9ydCAqZGlzcGxheUxhYmVsLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVuc2lnbmVkIHNob3J0ICpudW1iZXIsIHVuc2lnbmVkIHNob3J0KiByZXN1bHRzKQotewotICAgIEtleSogayA9IGZpbmRfa2V5KGtleWNvZGUpOwotICAgIGlmIChrICE9IE5VTEwpIHsKLSAgICAgICAgbWVtY3B5KHJlc3VsdHMsIGstPmRhdGEsIHNpemVvZihzaG9ydCkqKE1FVEFfTUFTSyArIDEpKTsKLSAgICAgICAgKm51bWJlciA9IGstPm51bWJlcjsKLSAgICAgICAgKmRpc3BsYXlMYWJlbCA9IGstPmRpc3BsYXlfbGFiZWw7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci19Ci0KLWJvb2wKLUtleUNoYXJhY3Rlck1hcDo6ZmluZF9jaGFyKHVpbnQxNl90IGMsIHVpbnQzMl90KiBrZXksIHVpbnQzMl90KiBtb2RzKQotewotICAgIHVpbnQzMl90IE4gPSBtX2tleUNvdW50OwotICAgIGZvciAoaW50IGo9MDsgajwoTUVUQV9NQVNLICsgMSk7IGorKykgewotICAgICAgICBLZXkgY29uc3QqIGtleXMgPSBtX2tleXM7Ci0gICAgICAgIGZvciAodWludDMyX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICAgICAgaWYgKGtleXMtPmRhdGFbal0gPT0gYykgewotICAgICAgICAgICAgICAgICprZXkgPSBrZXlzLT5rZXljb2RlOwotICAgICAgICAgICAgICAgICptb2RzID0gajsKLSAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGtleXMrKzsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0KLWJvb2wKLUtleUNoYXJhY3Rlck1hcDo6Z2V0RXZlbnRzKHVpbnQxNl90KiBjaGFycywgc2l6ZV90IGxlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIFZlY3RvcjxpbnQzMl90Pioga2V5cywgVmVjdG9yPHVpbnQzMl90PiogbW9kaWZpZXJzKQotewotICAgIGZvciAoc2l6ZV90IGk9MDsgaTxsZW47IGkrKykgewotICAgICAgICB1aW50MzJfdCBrLCBtb2RzOwotICAgICAgICBpZiAoZmluZF9jaGFyKGNoYXJzW2ldLCAmaywgJm1vZHMpKSB7Ci0gICAgICAgICAgICBrZXlzLT5hZGQoayk7Ci0gICAgICAgICAgICBtb2RpZmllcnMtPmFkZChtb2RzKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotS2V5Q2hhcmFjdGVyTWFwOjpLZXkqCi1LZXlDaGFyYWN0ZXJNYXA6OmZpbmRfa2V5KGludCBrZXljb2RlKQotewotICAgIEtleSoga2V5cyA9IG1fa2V5czsKLSAgICBpbnQgbG93ID0gMDsKLSAgICBpbnQgaGlnaCA9IG1fa2V5Q291bnQgLSAxOwotICAgIGludCBtaWQ7Ci0gICAgaW50IG47Ci0gICAgd2hpbGUgKGxvdyA8PSBoaWdoKSB7Ci0gICAgICAgIG1pZCA9IChsb3cgKyBoaWdoKSAvIDI7Ci0gICAgICAgIG4gPSBrZXlzW21pZF0ua2V5Y29kZTsKLSAgICAgICAgaWYgKGtleWNvZGUgPCBuKSB7Ci0gICAgICAgICAgICBoaWdoID0gbWlkIC0gMTsKLSAgICAgICAgfSBlbHNlIGlmIChrZXljb2RlID4gbikgewotICAgICAgICAgICAgbG93ID0gbWlkICsgMTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJldHVybiBrZXlzICsgbWlkOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBOVUxMOwotfQotCi1LZXlDaGFyYWN0ZXJNYXAqCi1LZXlDaGFyYWN0ZXJNYXA6OmxvYWQoaW50IGlkKQotewotICAgIEtleUNoYXJhY3Rlck1hcCogcnYgPSBOVUxMOwotICAgIGNoYXIgcGF0aFtQQVRIX01BWF07Ci0gICAgY2hhciBwcm9wTmFtZVsxMDBdOwotICAgIGNoYXIgZGV2W1BST1BFUlRZX1ZBTFVFX01BWF07Ci0gICAgY2hhciB0bXBmbltQUk9QRVJUWV9WQUxVRV9NQVhdOwotICAgIGludCBlcnI7Ci0gICAgY29uc3QgY2hhciogcm9vdCA9IGdldGVudigiQU5EUk9JRF9ST09UIik7Ci0KLSAgICBzcHJpbnRmKHByb3BOYW1lLCAiaHcua2V5Ym9hcmRzLiV1LmRldm5hbWUiLCBpZCk7Ci0gICAgZXJyID0gcHJvcGVydHlfZ2V0KHByb3BOYW1lLCBkZXYsICIiKTsKLSAgICBpZiAoZXJyID4gMCkgewotICAgICAgICAvLyByZXBsYWNlIGFsbCB0aGUgc3BhY2VzIHdpdGggdW5kZXJzY29yZXMKLSAgICAgICAgc3RyY3B5KHRtcGZuLCBkZXYpOwotICAgICAgICBmb3IgKGNoYXIgKnAgPSBzdHJjaHIodG1wZm4sICcgJyk7IHAgJiYgKnA7IHAgPSBzdHJjaHIodG1wZm4sICcgJykpCi0gICAgICAgICAgICAqcCA9ICdfJzsKLSAgICAgICAgc25wcmludGYocGF0aCwgc2l6ZW9mKHBhdGgpLCAiJXMvdXNyL2tleWNoYXJzLyVzLmtjbS5iaW4iLCByb290LCB0bXBmbik7Ci0gICAgICAgIC8vTE9HRCgibG9hZDogZGV2PSclcycgcGF0aD0nJXMnXG4iLCBkZXYsIHBhdGgpOwotICAgICAgICBydiA9IHRyeV9maWxlKHBhdGgpOwotICAgICAgICBpZiAocnYgIT0gTlVMTCkgewotICAgICAgICAgICAgcmV0dXJuIHJ2OwotICAgICAgICB9Ci0gICAgICAgIExPR1coIkVycm9yIGxvYWRpbmcga2V5Y2hhcm1hcCBmaWxlICclcycuICVzPSclcyciLCBwYXRoLCBwcm9wTmFtZSwgZGV2KTsKLSAgICB9IGVsc2UgewotICAgICAgICBMT0dXKCJObyBrZXlib2FyZCBmb3IgaWQgJWQiLCBpZCk7Ci0gICAgfQotCi0gICAgc25wcmludGYocGF0aCwgc2l6ZW9mKHBhdGgpLCAiJXMvdXNyL2tleWNoYXJzL3F3ZXJ0eS5rY20uYmluIiwgcm9vdCk7Ci0gICAgcnYgPSB0cnlfZmlsZShwYXRoKTsKLSAgICBpZiAocnYgPT0gTlVMTCkgewotICAgICAgICBMT0dFKCJDYW4ndCBmaW5kIGFueSBrZXljaGFybWFwcyAoYWxzbyB0cmllZCAlcykiLCBwYXRoKTsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotICAgIExPR1coIlVzaW5nIGRlZmF1bHQga2V5bWFwOiAlcyIsIHBhdGgpOwotCi0gICAgcmV0dXJuIHJ2OwotfQotCi1LZXlDaGFyYWN0ZXJNYXAqCi1LZXlDaGFyYWN0ZXJNYXA6OnRyeV9maWxlKGNvbnN0IGNoYXIqIGZpbGVuYW1lKQotewotICAgIEtleUNoYXJhY3Rlck1hcCogcnYgPSBOVUxMOwotICAgIEtleSoga2V5czsKLSAgICBpbnQgZmQ7Ci0gICAgb2ZmX3QgZmlsZXNpemU7Ci0gICAgSGVhZGVyIGhlYWRlcjsKLSAgICBpbnQgZXJyOwotICAgIAotICAgIGZkID0gb3BlbihmaWxlbmFtZSwgT19SRE9OTFkpOwotICAgIGlmIChmZCA9PSAtMSkgewotICAgICAgICBMT0dXKCJDYW4ndCBvcGVuIGtleWNoYXJtYXAgZmlsZSIpOwotICAgICAgICByZXR1cm4gTlVMTDsKLSAgICB9Ci0KLSAgICBmaWxlc2l6ZSA9IGxzZWVrKGZkLCAwLCBTRUVLX0VORCk7Ci0gICAgbHNlZWsoZmQsIDAsIFNFRUtfU0VUKTsKLQotICAgIC8vIHZhbGlkYXRlIHRoZSBoZWFkZXIKLSAgICBpZiAoZmlsZXNpemUgPD0gKG9mZl90KXNpemVvZihoZWFkZXIpKSB7Ci0gICAgICAgIExPR1coIkJhZCBrZXljaGFybWFwIC0gZmlsZXNpemU9JWRcbiIsIChpbnQpZmlsZXNpemUpOwotICAgICAgICBnb3RvIGNsZWFudXAxOwotICAgIH0KLQotICAgIGVyciA9IHJlYWQoZmQsICZoZWFkZXIsIHNpemVvZihoZWFkZXIpKTsKLSAgICBpZiAoZXJyID09IC0xKSB7Ci0gICAgICAgIExPR1coIkVycm9yIHJlYWRpbmcga2V5Y2hhcm1hcCBmaWxlIik7Ci0gICAgICAgIGdvdG8gY2xlYW51cDE7Ci0gICAgfQotCi0gICAgaWYgKDAgIT0gbWVtY21wKGhlYWRlci5tYWdpYywgImtleWNoYXIiLCA4KSkgewotICAgICAgICBMT0dXKCJCYWQga2V5Y2hhcm1hcCBtYWdpYyB0b2tlbiIpOwotICAgICAgICBnb3RvIGNsZWFudXAxOwotICAgIH0KLSAgICBpZiAoaGVhZGVyLmVuZGlhbiAhPSAweDEyMzQ1Njc4KSB7Ci0gICAgICAgIExPR1coIkJhZCBrZXljaGFybWFwIGVuZGlhbnMiKTsKLSAgICAgICAgZ290byBjbGVhbnVwMTsKLSAgICB9Ci0gICAgaWYgKChoZWFkZXIudmVyc2lvbiAmIDB4ZmYpICE9IDIpIHsKLSAgICAgICAgTE9HVygiT25seSBzdXBwb3J0IGtleWNoYXJtYXAgdmVyc2lvbiAyIChnb3QgMHglMDh4KSIsIGhlYWRlci52ZXJzaW9uKTsKLSAgICAgICAgZ290byBjbGVhbnVwMTsKLSAgICB9Ci0gICAgaWYgKGZpbGVzaXplIDwgKG9mZl90KShzaXplb2YoSGVhZGVyKSsoc2l6ZW9mKEtleSkqaGVhZGVyLmtleWNvdW50KSkpIHsKLSAgICAgICAgTE9HVygiQmFkIGtleWNoYXJtYXAgZmlsZSBzaXplXG4iKTsKLSAgICAgICAgZ290byBjbGVhbnVwMTsKLSAgICB9Ci0KLSAgICAvLyByZWFkIHRoZSBrZXkgZGF0YQotICAgIGtleXMgPSAoS2V5KiltYWxsb2Moc2l6ZW9mKEtleSkqaGVhZGVyLmtleWNvdW50KTsKLSAgICBlcnIgPSByZWFkKGZkLCBrZXlzLCBzaXplb2YoS2V5KSpoZWFkZXIua2V5Y291bnQpOwotICAgIGlmIChlcnIgPT0gLTEpIHsKLSAgICAgICAgTE9HVygiRXJyb3IgcmVhZGluZyBrZXljaGFybWFwIGZpbGUiKTsKLSAgICAgICAgZnJlZShrZXlzKTsKLSAgICAgICAgZ290byBjbGVhbnVwMTsKLSAgICB9Ci0KLSAgICAvLyByZXR1cm4gdGhlIG9iamVjdAotICAgIHJ2ID0gbmV3IEtleUNoYXJhY3Rlck1hcDsKLSAgICBydi0+bV9rZXlDb3VudCA9IGhlYWRlci5rZXljb3VudDsKLSAgICBydi0+bV9rZXlzID0ga2V5czsKLSAgICBydi0+bV90eXBlID0gaGVhZGVyLmtiZHR5cGU7Ci0KLWNsZWFudXAxOgotICAgIGNsb3NlKGZkKTsKLQotICAgIHJldHVybiBydjsKLX0KZGlmZiAtLWdpdCBhL2xpYnMvdWkvS2V5TGF5b3V0TWFwLmNwcCBiL2xpYnMvdWkvS2V5TGF5b3V0TWFwLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTVhZTU0Yy4uMDAwMDAwMAotLS0gYS9saWJzL3VpL0tleUxheW91dE1hcC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwyMzUgKzAsMCBAQAotI2RlZmluZSBMT0dfVEFHICJLZXlMYXlvdXRNYXAiCi0KLSNpbmNsdWRlICJLZXlMYXlvdXRNYXAuaCIKLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDxzeXMvc3RhdC5oPgotI2luY2x1ZGUgPGZjbnRsLmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8dWkvS2V5Y29kZUxhYmVscy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLUtleUxheW91dE1hcDo6S2V5TGF5b3V0TWFwKCkKLSAgICA6bV9zdGF0dXMoTk9fSU5JVCksCi0gICAgIG1fa2V5cygpCi17Ci19Ci0KLUtleUxheW91dE1hcDo6fktleUxheW91dE1hcCgpCi17Ci19Ci0KLXN0YXRpYyBTdHJpbmc4Ci1uZXh0X3Rva2VuKGNoYXIgY29uc3QqKiBwLCBpbnQgKmxpbmUpCi17Ci0gICAgYm9vbCBiZWd1biA9IGZhbHNlOwotICAgIGNvbnN0IGNoYXIqIGJlZ2luID0gKnA7Ci0gICAgY29uc3QgY2hhciogZW5kID0gKnA7Ci0gICAgd2hpbGUgKHRydWUpIHsKLSAgICAgICAgaWYgKCplbmQgPT0gJ1xuJykgewotICAgICAgICAgICAgKCpsaW5lKSsrOwotICAgICAgICB9Ci0gICAgICAgIHN3aXRjaCAoKmVuZCkKLSAgICAgICAgewotICAgICAgICAgICAgY2FzZSAnIyc6Ci0gICAgICAgICAgICAgICAgaWYgKGJlZ3VuKSB7Ci0gICAgICAgICAgICAgICAgICAgICpwID0gZW5kOwotICAgICAgICAgICAgICAgICAgICByZXR1cm4gU3RyaW5nOChiZWdpbiwgZW5kLWJlZ2luKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBiZWdpbisrOwotICAgICAgICAgICAgICAgICAgICAgICAgZW5kKys7Ci0gICAgICAgICAgICAgICAgICAgIH0gd2hpbGUgKCpiZWdpbiAhPSAnXDAnICYmICpiZWdpbiAhPSAnXG4nKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICBjYXNlICdcMCc6Ci0gICAgICAgICAgICBjYXNlICcgJzoKLSAgICAgICAgICAgIGNhc2UgJ1xuJzoKLSAgICAgICAgICAgIGNhc2UgJ1xyJzoKLSAgICAgICAgICAgIGNhc2UgJ1x0JzoKLSAgICAgICAgICAgICAgICBpZiAoYmVndW4gfHwgKCplbmQgPT0gJ1wwJykpIHsKLSAgICAgICAgICAgICAgICAgICAgKnAgPSBlbmQ7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBTdHJpbmc4KGJlZ2luLCBlbmQtYmVnaW4pOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGJlZ2luKys7Ci0gICAgICAgICAgICAgICAgICAgIGVuZCsrOwotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgICAgIGVuZCsrOwotICAgICAgICAgICAgICAgIGJlZ3VuID0gdHJ1ZTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotc3RhdGljIGludDMyX3QKLXRva2VuX3RvX3ZhbHVlKGNvbnN0IGNoYXIgKmxpdGVyYWwsIGNvbnN0IEtleWNvZGVMYWJlbCAqbGlzdCkKLXsKLSAgICB3aGlsZSAobGlzdC0+bGl0ZXJhbCkgewotICAgICAgICBpZiAoMCA9PSBzdHJjbXAobGl0ZXJhbCwgbGlzdC0+bGl0ZXJhbCkpIHsKLSAgICAgICAgICAgIHJldHVybiBsaXN0LT52YWx1ZTsKLSAgICAgICAgfQotICAgICAgICBsaXN0Kys7Ci0gICAgfQotICAgIHJldHVybiBsaXN0LT52YWx1ZTsKLX0KLQotc3RhdHVzX3QKLUtleUxheW91dE1hcDo6bG9hZChjb25zdCBjaGFyKiBmaWxlbmFtZSkKLXsKLSAgICBpbnQgZmQgPSBvcGVuKGZpbGVuYW1lLCBPX1JET05MWSk7Ci0gICAgaWYgKGZkIDwgMCkgewotICAgICAgICBMT0dFKCJlcnJvciBvcGVuaW5nIGZpbGU9JXMgZXJyPSVzXG4iLCBmaWxlbmFtZSwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgbV9zdGF0dXMgPSBlcnJubzsKLSAgICAgICAgcmV0dXJuIGVycm5vOwotICAgIH0KLQotICAgIG9mZl90IGxlbiA9IGxzZWVrKGZkLCAwLCBTRUVLX0VORCk7Ci0gICAgb2ZmX3QgZXJybGVuID0gbHNlZWsoZmQsIDAsIFNFRUtfU0VUKTsKLSAgICBpZiAobGVuIDwgMCB8fCBlcnJsZW4gPCAwKSB7Ci0gICAgICAgIGNsb3NlKGZkKTsKLSAgICAgICAgTE9HRSgiZXJyb3Igc2Vla2luZyBmaWxlPSVzIGVycj0lc1xuIiwgZmlsZW5hbWUsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgICAgIG1fc3RhdHVzID0gZXJybm87Ci0gICAgICAgIHJldHVybiBlcnJubzsKLSAgICB9Ci0KLSAgICBjaGFyKiBidWYgPSAoY2hhciopbWFsbG9jKGxlbisxKTsKLSAgICBpZiAocmVhZChmZCwgYnVmLCBsZW4pICE9IGxlbikgewotICAgICAgICBMT0dFKCJlcnJvciByZWFkaW5nIGZpbGU9JXMgZXJyPSVzXG4iLCBmaWxlbmFtZSwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgbV9zdGF0dXMgPSBlcnJubyAhPSAwID8gZXJybm8gOiAoKGludClOT1RfRU5PVUdIX0RBVEEpOwotICAgICAgICByZXR1cm4gZXJybm8gIT0gMCA/IGVycm5vIDogKChpbnQpTk9UX0VOT1VHSF9EQVRBKTsKLSAgICB9Ci0gICAgZXJybm8gPSAwOwotICAgIGJ1ZltsZW5dID0gJ1wwJzsKLQotICAgIGludDMyX3Qgc2NhbmNvZGUgPSAtMTsKLSAgICBpbnQzMl90IGtleWNvZGUgPSAtMTsKLSAgICB1aW50MzJfdCBmbGFncyA9IDA7Ci0gICAgdWludDMyX3QgdG1wOwotICAgIGNoYXIqIGVuZDsKLSAgICBzdGF0dXNfdCBlcnIgPSBOT19FUlJPUjsKLSAgICBpbnQgbGluZSA9IDE7Ci0gICAgY2hhciBjb25zdCogcCA9IGJ1ZjsKLSAgICBlbnVtIHsgQkVHSU4sIFNDQU5DT0RFLCBLRVlDT0RFLCBGTEFHIH0gc3RhdGUgPSBCRUdJTjsKLSAgICB3aGlsZSAodHJ1ZSkgewotICAgICAgICBTdHJpbmc4IHRva2VuID0gbmV4dF90b2tlbigmcCwgJmxpbmUpOwotICAgICAgICBpZiAoKnAgPT0gJ1wwJykgewotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgc3dpdGNoIChzdGF0ZSkKLSAgICAgICAgewotICAgICAgICAgICAgY2FzZSBCRUdJTjoKLSAgICAgICAgICAgICAgICBpZiAodG9rZW4gPT0gImtleSIpIHsKLSAgICAgICAgICAgICAgICAgICAgc3RhdGUgPSBTQ0FOQ09ERTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBMT0dFKCIlczolZDogZXhwZWN0ZWQga2V5LCBnb3QgJyVzJ1xuIiwgZmlsZW5hbWUsIGxpbmUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdG9rZW4uc3RyaW5nKCkpOwotICAgICAgICAgICAgICAgICAgICBlcnIgPSBCQURfVkFMVUU7Ci0gICAgICAgICAgICAgICAgICAgIGdvdG8gZG9uZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIFNDQU5DT0RFOgotICAgICAgICAgICAgICAgIHNjYW5jb2RlID0gc3RydG9sKHRva2VuLnN0cmluZygpLCAmZW5kLCAwKTsKLSAgICAgICAgICAgICAgICBpZiAoKmVuZCAhPSAnXDAnKSB7Ci0gICAgICAgICAgICAgICAgICAgIExPR0UoIiVzOiVkOiBleHBlY3RlZCBzY2FuY29kZSAoYSBudW1iZXIpLCBnb3QgJyVzJ1xuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBmaWxlbmFtZSwgbGluZSwgdG9rZW4uc3RyaW5nKCkpOwotICAgICAgICAgICAgICAgICAgICBnb3RvIGRvbmU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIC8vTE9HSSgiJXM6JWQ6IGdvdCBzY2FuY29kZSAlZFxuIiwgZmlsZW5hbWUsIGxpbmUsIHNjYW5jb2RlICk7Ci0gICAgICAgICAgICAgICAgc3RhdGUgPSBLRVlDT0RFOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBLRVlDT0RFOgotICAgICAgICAgICAgICAgIGtleWNvZGUgPSB0b2tlbl90b192YWx1ZSh0b2tlbi5zdHJpbmcoKSwgS0VZQ09ERVMpOwotICAgICAgICAgICAgICAgIC8vTE9HSSgiJXM6JWQ6IGdvdCBrZXljb2RlICVkIGZvciAlc1xuIiwgZmlsZW5hbWUsIGxpbmUsIGtleWNvZGUsIHRva2VuLnN0cmluZygpICk7Ci0gICAgICAgICAgICAgICAgaWYgKGtleWNvZGUgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICBMT0dFKCIlczolZDogZXhwZWN0ZWQga2V5Y29kZSwgZ290ICclcydcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZmlsZW5hbWUsIGxpbmUsIHRva2VuLnN0cmluZygpKTsKLSAgICAgICAgICAgICAgICAgICAgZ290byBkb25lOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBzdGF0ZSA9IEZMQUc7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIEZMQUc6Ci0gICAgICAgICAgICAgICAgaWYgKHRva2VuID09ICJrZXkiKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChzY2FuY29kZSAhPSAtMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy9MT0dJKCJnb3Qga2V5IGRlY2wgc2NhbmNvZGU9JWQga2V5Y29kZT0lZCIKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vICAgICAgICIgZmxhZ3M9MHglMDh4XG4iLCBzY2FuY29kZSwga2V5Y29kZSwgZmxhZ3MpOwotICAgICAgICAgICAgICAgICAgICAgICAgS2V5IGsgPSB7IGtleWNvZGUsIGZsYWdzIH07Ci0gICAgICAgICAgICAgICAgICAgICAgICBtX2tleXMuYWRkKHNjYW5jb2RlLCBrKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHN0YXRlID0gU0NBTkNPREU7Ci0gICAgICAgICAgICAgICAgICAgICAgICBzY2FuY29kZSA9IC0xOwotICAgICAgICAgICAgICAgICAgICAgICAga2V5Y29kZSA9IC0xOwotICAgICAgICAgICAgICAgICAgICAgICAgZmxhZ3MgPSAwOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgdG1wID0gdG9rZW5fdG9fdmFsdWUodG9rZW4uc3RyaW5nKCksIEZMQUdTKTsKLSAgICAgICAgICAgICAgICAvL0xPR0koIiVzOiVkOiBnb3QgZmxhZ3MgJXggZm9yICVzXG4iLCBmaWxlbmFtZSwgbGluZSwgdG1wLCB0b2tlbi5zdHJpbmcoKSApOwotICAgICAgICAgICAgICAgIGlmICh0bXAgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICBMT0dFKCIlczolZDogZXhwZWN0ZWQgZmxhZywgZ290ICclcydcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZmlsZW5hbWUsIGxpbmUsIHRva2VuLnN0cmluZygpKTsKLSAgICAgICAgICAgICAgICAgICAgZ290byBkb25lOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBmbGFncyB8PSB0bXA7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgaWYgKHN0YXRlID09IEZMQUcgJiYgc2NhbmNvZGUgIT0gLTEgKSB7Ci0gICAgICAgIC8vTE9HSSgiZ290IGtleSBkZWNsIHNjYW5jb2RlPSVkIGtleWNvZGU9JWQiCi0gICAgICAgIC8vICAgICAgICIgZmxhZ3M9MHglMDh4XG4iLCBzY2FuY29kZSwga2V5Y29kZSwgZmxhZ3MpOwotICAgICAgICBLZXkgayA9IHsga2V5Y29kZSwgZmxhZ3MgfTsKLSAgICAgICAgbV9rZXlzLmFkZChzY2FuY29kZSwgayk7Ci0gICAgfQotCi1kb25lOgotICAgIGZyZWUoYnVmKTsKLSAgICBjbG9zZShmZCk7Ci0KLSAgICBtX3N0YXR1cyA9IGVycjsKLSAgICByZXR1cm4gZXJyOwotfQotCi1zdGF0dXNfdAotS2V5TGF5b3V0TWFwOjptYXAoaW50MzJfdCBzY2FuY29kZSwgaW50MzJfdCAqa2V5Y29kZSwgdWludDMyX3QgKmZsYWdzKSBjb25zdAotewotICAgIGlmIChtX3N0YXR1cyAhPSBOT19FUlJPUikgewotICAgICAgICByZXR1cm4gbV9zdGF0dXM7Ci0gICAgfQotCi0gICAgc3NpemVfdCBpbmRleCA9IG1fa2V5cy5pbmRleE9mS2V5KHNjYW5jb2RlKTsKLSAgICBpZiAoaW5kZXggPCAwKSB7Ci0gICAgICAgIC8vTE9HVygiY291bGRuJ3QgbWFwIHNjYW5jb2RlPSVkXG4iLCBzY2FuY29kZSk7Ci0gICAgICAgIHJldHVybiBOQU1FX05PVF9GT1VORDsKLSAgICB9Ci0KLSAgICBjb25zdCBLZXkmIGsgPSBtX2tleXMudmFsdWVBdChpbmRleCk7Ci0KLSAgICAqa2V5Y29kZSA9IGsua2V5Y29kZTsKLSAgICAqZmxhZ3MgPSBrLmZsYWdzOwotCi0gICAgLy9MT0dEKCJtYXBwZWQgc2NhbmNvZGU9JWQgdG8ga2V5Y29kZT0lZCBmbGFncz0weCUwOHhcbiIsIHNjYW5jb2RlLAotICAgIC8vICAgICAgICBrZXljb2RlLCBmbGFncyk7Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190Ci1LZXlMYXlvdXRNYXA6OmZpbmRTY2FuY29kZXMoaW50MzJfdCBrZXljb2RlLCBWZWN0b3I8aW50MzJfdD4qIG91dFNjYW5jb2RlcykgY29uc3QKLXsKLSAgICBpZiAobV9zdGF0dXMgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgcmV0dXJuIG1fc3RhdHVzOwotICAgIH0KLSAgICAKLSAgICBjb25zdCBzaXplX3QgTiA9IG1fa2V5cy5zaXplKCk7Ci0gICAgZm9yIChzaXplX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICBpZiAobV9rZXlzLnZhbHVlQXQoaSkua2V5Y29kZSA9PSBrZXljb2RlKSB7Ci0gICAgICAgICAgICBvdXRTY2FuY29kZXMtPmFkZChtX2tleXMua2V5QXQoaSkpOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotfTsKZGlmZiAtLWdpdCBhL2xpYnMvdWkvS2V5TGF5b3V0TWFwLmggYi9saWJzL3VpL0tleUxheW91dE1hcC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0M2Y4NGNlLi4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvS2V5TGF5b3V0TWFwLmgKKysrIC9kZXYvbnVsbApAQCAtMSwzMSArMCwwIEBACi0jaWZuZGVmIEtFWUxBWU9VVE1BUF9ICi0jZGVmaW5lIEtFWUxBWU9VVE1BUF9ICi0KLSNpbmNsdWRlIDx1dGlscy9LZXllZFZlY3Rvci5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIEtleUxheW91dE1hcAotewotcHVibGljOgotICAgIEtleUxheW91dE1hcCgpOwotICAgIH5LZXlMYXlvdXRNYXAoKTsKLQotICAgIHN0YXR1c190IGxvYWQoY29uc3QgY2hhciogZmlsZW5hbWUpOwotCi0gICAgc3RhdHVzX3QgbWFwKGludDMyX3Qgc2NhbmNvZGUsIGludDMyX3QgKmtleWNvZGUsIHVpbnQzMl90ICpmbGFncykgY29uc3Q7Ci0gICAgc3RhdHVzX3QgZmluZFNjYW5jb2RlcyhpbnQzMl90IGtleWNvZGUsIFZlY3RvcjxpbnQzMl90Piogb3V0U2NhbmNvZGVzKSBjb25zdDsKLQotcHJpdmF0ZToKLSAgICBzdHJ1Y3QgS2V5IHsKLSAgICAgICAgaW50MzJfdCBrZXljb2RlOwotICAgICAgICB1aW50MzJfdCBmbGFnczsKLSAgICB9OwotCi0gICAgc3RhdHVzX3QgbV9zdGF0dXM7Ci0gICAgS2V5ZWRWZWN0b3I8aW50MzJfdCxLZXk+IG1fa2V5czsKLX07Ci0KLX07Ci0KLSNlbmRpZiAvLyBLRVlMQVlPVVRNQVBfSApkaWZmIC0tZ2l0IGEvbGlicy91aS9MYXllclN0YXRlLmNwcCBiL2xpYnMvdWkvTGF5ZXJTdGF0ZS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDBiNjM3NGIuLjAwMDAwMDAKLS0tIGEvbGlicy91aS9MYXllclN0YXRlLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDUzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUGFyY2VsLmg+Ci0jaW5jbHVkZSA8cHJpdmF0ZS91aS9MYXllclN0YXRlLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotc3RhdHVzX3QgbGF5ZXJfc3RhdGVfdDo6d3JpdGUoUGFyY2VsJiBvdXRwdXQpIGNvbnN0Ci17Ci0gICAgc2l6ZV90IHNpemUgPSBzaXplb2YobGF5ZXJfc3RhdGVfdCk7Ci0KLSAgICAvL291dHB1dC53cml0ZVN0cm9uZ0JpbmRlcihzdXJmYWNlLT5hc0JpbmRlcigpKTsKLSAgICAvL3NpemUgLT0gc2l6ZW9mKHN1cmZhY2UpOwotCi0gICAgdHJhbnNwYXJlbnRSZWdpb24ud3JpdGUob3V0cHV0KTsKLSAgICBzaXplIC09IHNpemVvZih0cmFuc3BhcmVudFJlZ2lvbik7Ci0gICAgCi0gICAgb3V0cHV0LndyaXRlKHRoaXMsIHNpemUpOwotICAgIAotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgbGF5ZXJfc3RhdGVfdDo6cmVhZChjb25zdCBQYXJjZWwmIGlucHV0KQotewotICAgIHNpemVfdCBzaXplID0gc2l6ZW9mKGxheWVyX3N0YXRlX3QpOwotCi0gICAgLy9zdXJmYWNlID0gaW50ZXJmYWNlX2Nhc3Q8SVN1cmZhY2U+KGlucHV0LnJlYWRTdHJvbmdCaW5kZXIoKSk7Ci0gICAgLy9zaXplIC09IHNpemVvZihzdXJmYWNlKTsKLQotICAgIHRyYW5zcGFyZW50UmVnaW9uLnJlYWQoaW5wdXQpOwotICAgIHNpemUgLT0gc2l6ZW9mKHRyYW5zcGFyZW50UmVnaW9uKTsKLQotICAgIGlucHV0LnJlYWQodGhpcywgc2l6ZSk7Ci0gICAgCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91aS9NT0RVTEVfTElDRU5TRV9BUEFDSEUyIGIvbGlicy91aS9NT0RVTEVfTElDRU5TRV9BUEFDSEUyCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlNjlkZTI5Li4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvTU9EVUxFX0xJQ0VOU0VfQVBBQ0hFMgorKysgL2Rldi9udWxsCmRpZmYgLS1naXQgYS9saWJzL3VpL05PVElDRSBiL2xpYnMvdWkvTk9USUNFCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjNWIxZWZhLi4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvTk9USUNFCisrKyAvZGV2L251bGwKQEAgLTEsMTkwICswLDAgQEAKLQotICAgQ29weXJpZ2h0IChjKSAyMDA1LTIwMDgsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLQotICAgTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gICB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0KLSAgIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAgIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gICBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAgIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAgIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotCi0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEFwYWNoZSBMaWNlbnNlCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBWZXJzaW9uIDIuMCwgSmFudWFyeSAyMDA0Ci0gICAgICAgICAgICAgICAgICAgICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvCi0KLSAgIFRFUk1TIEFORCBDT05ESVRJT05TIEZPUiBVU0UsIFJFUFJPRFVDVElPTiwgQU5EIERJU1RSSUJVVElPTgotCi0gICAxLiBEZWZpbml0aW9ucy4KLQotICAgICAgIkxpY2Vuc2UiIHNoYWxsIG1lYW4gdGhlIHRlcm1zIGFuZCBjb25kaXRpb25zIGZvciB1c2UsIHJlcHJvZHVjdGlvbiwKLSAgICAgIGFuZCBkaXN0cmlidXRpb24gYXMgZGVmaW5lZCBieSBTZWN0aW9ucyAxIHRocm91Z2ggOSBvZiB0aGlzIGRvY3VtZW50LgotCi0gICAgICAiTGljZW5zb3IiIHNoYWxsIG1lYW4gdGhlIGNvcHlyaWdodCBvd25lciBvciBlbnRpdHkgYXV0aG9yaXplZCBieQotICAgICAgdGhlIGNvcHlyaWdodCBvd25lciB0aGF0IGlzIGdyYW50aW5nIHRoZSBMaWNlbnNlLgotCi0gICAgICAiTGVnYWwgRW50aXR5IiBzaGFsbCBtZWFuIHRoZSB1bmlvbiBvZiB0aGUgYWN0aW5nIGVudGl0eSBhbmQgYWxsCi0gICAgICBvdGhlciBlbnRpdGllcyB0aGF0IGNvbnRyb2wsIGFyZSBjb250cm9sbGVkIGJ5LCBvciBhcmUgdW5kZXIgY29tbW9uCi0gICAgICBjb250cm9sIHdpdGggdGhhdCBlbnRpdHkuIEZvciB0aGUgcHVycG9zZXMgb2YgdGhpcyBkZWZpbml0aW9uLAotICAgICAgImNvbnRyb2wiIG1lYW5zIChpKSB0aGUgcG93ZXIsIGRpcmVjdCBvciBpbmRpcmVjdCwgdG8gY2F1c2UgdGhlCi0gICAgICBkaXJlY3Rpb24gb3IgbWFuYWdlbWVudCBvZiBzdWNoIGVudGl0eSwgd2hldGhlciBieSBjb250cmFjdCBvcgotICAgICAgb3RoZXJ3aXNlLCBvciAoaWkpIG93bmVyc2hpcCBvZiBmaWZ0eSBwZXJjZW50ICg1MCUpIG9yIG1vcmUgb2YgdGhlCi0gICAgICBvdXRzdGFuZGluZyBzaGFyZXMsIG9yIChpaWkpIGJlbmVmaWNpYWwgb3duZXJzaGlwIG9mIHN1Y2ggZW50aXR5LgotCi0gICAgICAiWW91IiAob3IgIllvdXIiKSBzaGFsbCBtZWFuIGFuIGluZGl2aWR1YWwgb3IgTGVnYWwgRW50aXR5Ci0gICAgICBleGVyY2lzaW5nIHBlcm1pc3Npb25zIGdyYW50ZWQgYnkgdGhpcyBMaWNlbnNlLgotCi0gICAgICAiU291cmNlIiBmb3JtIHNoYWxsIG1lYW4gdGhlIHByZWZlcnJlZCBmb3JtIGZvciBtYWtpbmcgbW9kaWZpY2F0aW9ucywKLSAgICAgIGluY2x1ZGluZyBidXQgbm90IGxpbWl0ZWQgdG8gc29mdHdhcmUgc291cmNlIGNvZGUsIGRvY3VtZW50YXRpb24KLSAgICAgIHNvdXJjZSwgYW5kIGNvbmZpZ3VyYXRpb24gZmlsZXMuCi0KLSAgICAgICJPYmplY3QiIGZvcm0gc2hhbGwgbWVhbiBhbnkgZm9ybSByZXN1bHRpbmcgZnJvbSBtZWNoYW5pY2FsCi0gICAgICB0cmFuc2Zvcm1hdGlvbiBvciB0cmFuc2xhdGlvbiBvZiBhIFNvdXJjZSBmb3JtLCBpbmNsdWRpbmcgYnV0Ci0gICAgICBub3QgbGltaXRlZCB0byBjb21waWxlZCBvYmplY3QgY29kZSwgZ2VuZXJhdGVkIGRvY3VtZW50YXRpb24sCi0gICAgICBhbmQgY29udmVyc2lvbnMgdG8gb3RoZXIgbWVkaWEgdHlwZXMuCi0KLSAgICAgICJXb3JrIiBzaGFsbCBtZWFuIHRoZSB3b3JrIG9mIGF1dGhvcnNoaXAsIHdoZXRoZXIgaW4gU291cmNlIG9yCi0gICAgICBPYmplY3QgZm9ybSwgbWFkZSBhdmFpbGFibGUgdW5kZXIgdGhlIExpY2Vuc2UsIGFzIGluZGljYXRlZCBieSBhCi0gICAgICBjb3B5cmlnaHQgbm90aWNlIHRoYXQgaXMgaW5jbHVkZWQgaW4gb3IgYXR0YWNoZWQgdG8gdGhlIHdvcmsKLSAgICAgIChhbiBleGFtcGxlIGlzIHByb3ZpZGVkIGluIHRoZSBBcHBlbmRpeCBiZWxvdykuCi0KLSAgICAgICJEZXJpdmF0aXZlIFdvcmtzIiBzaGFsbCBtZWFuIGFueSB3b3JrLCB3aGV0aGVyIGluIFNvdXJjZSBvciBPYmplY3QKLSAgICAgIGZvcm0sIHRoYXQgaXMgYmFzZWQgb24gKG9yIGRlcml2ZWQgZnJvbSkgdGhlIFdvcmsgYW5kIGZvciB3aGljaCB0aGUKLSAgICAgIGVkaXRvcmlhbCByZXZpc2lvbnMsIGFubm90YXRpb25zLCBlbGFib3JhdGlvbnMsIG9yIG90aGVyIG1vZGlmaWNhdGlvbnMKLSAgICAgIHJlcHJlc2VudCwgYXMgYSB3aG9sZSwgYW4gb3JpZ2luYWwgd29yayBvZiBhdXRob3JzaGlwLiBGb3IgdGhlIHB1cnBvc2VzCi0gICAgICBvZiB0aGlzIExpY2Vuc2UsIERlcml2YXRpdmUgV29ya3Mgc2hhbGwgbm90IGluY2x1ZGUgd29ya3MgdGhhdCByZW1haW4KLSAgICAgIHNlcGFyYWJsZSBmcm9tLCBvciBtZXJlbHkgbGluayAob3IgYmluZCBieSBuYW1lKSB0byB0aGUgaW50ZXJmYWNlcyBvZiwKLSAgICAgIHRoZSBXb3JrIGFuZCBEZXJpdmF0aXZlIFdvcmtzIHRoZXJlb2YuCi0KLSAgICAgICJDb250cmlidXRpb24iIHNoYWxsIG1lYW4gYW55IHdvcmsgb2YgYXV0aG9yc2hpcCwgaW5jbHVkaW5nCi0gICAgICB0aGUgb3JpZ2luYWwgdmVyc2lvbiBvZiB0aGUgV29yayBhbmQgYW55IG1vZGlmaWNhdGlvbnMgb3IgYWRkaXRpb25zCi0gICAgICB0byB0aGF0IFdvcmsgb3IgRGVyaXZhdGl2ZSBXb3JrcyB0aGVyZW9mLCB0aGF0IGlzIGludGVudGlvbmFsbHkKLSAgICAgIHN1Ym1pdHRlZCB0byBMaWNlbnNvciBmb3IgaW5jbHVzaW9uIGluIHRoZSBXb3JrIGJ5IHRoZSBjb3B5cmlnaHQgb3duZXIKLSAgICAgIG9yIGJ5IGFuIGluZGl2aWR1YWwgb3IgTGVnYWwgRW50aXR5IGF1dGhvcml6ZWQgdG8gc3VibWl0IG9uIGJlaGFsZiBvZgotICAgICAgdGhlIGNvcHlyaWdodCBvd25lci4gRm9yIHRoZSBwdXJwb3NlcyBvZiB0aGlzIGRlZmluaXRpb24sICJzdWJtaXR0ZWQiCi0gICAgICBtZWFucyBhbnkgZm9ybSBvZiBlbGVjdHJvbmljLCB2ZXJiYWwsIG9yIHdyaXR0ZW4gY29tbXVuaWNhdGlvbiBzZW50Ci0gICAgICB0byB0aGUgTGljZW5zb3Igb3IgaXRzIHJlcHJlc2VudGF0aXZlcywgaW5jbHVkaW5nIGJ1dCBub3QgbGltaXRlZCB0bwotICAgICAgY29tbXVuaWNhdGlvbiBvbiBlbGVjdHJvbmljIG1haWxpbmcgbGlzdHMsIHNvdXJjZSBjb2RlIGNvbnRyb2wgc3lzdGVtcywKLSAgICAgIGFuZCBpc3N1ZSB0cmFja2luZyBzeXN0ZW1zIHRoYXQgYXJlIG1hbmFnZWQgYnksIG9yIG9uIGJlaGFsZiBvZiwgdGhlCi0gICAgICBMaWNlbnNvciBmb3IgdGhlIHB1cnBvc2Ugb2YgZGlzY3Vzc2luZyBhbmQgaW1wcm92aW5nIHRoZSBXb3JrLCBidXQKLSAgICAgIGV4Y2x1ZGluZyBjb21tdW5pY2F0aW9uIHRoYXQgaXMgY29uc3BpY3VvdXNseSBtYXJrZWQgb3Igb3RoZXJ3aXNlCi0gICAgICBkZXNpZ25hdGVkIGluIHdyaXRpbmcgYnkgdGhlIGNvcHlyaWdodCBvd25lciBhcyAiTm90IGEgQ29udHJpYnV0aW9uLiIKLQotICAgICAgIkNvbnRyaWJ1dG9yIiBzaGFsbCBtZWFuIExpY2Vuc29yIGFuZCBhbnkgaW5kaXZpZHVhbCBvciBMZWdhbCBFbnRpdHkKLSAgICAgIG9uIGJlaGFsZiBvZiB3aG9tIGEgQ29udHJpYnV0aW9uIGhhcyBiZWVuIHJlY2VpdmVkIGJ5IExpY2Vuc29yIGFuZAotICAgICAgc3Vic2VxdWVudGx5IGluY29ycG9yYXRlZCB3aXRoaW4gdGhlIFdvcmsuCi0KLSAgIDIuIEdyYW50IG9mIENvcHlyaWdodCBMaWNlbnNlLiBTdWJqZWN0IHRvIHRoZSB0ZXJtcyBhbmQgY29uZGl0aW9ucyBvZgotICAgICAgdGhpcyBMaWNlbnNlLCBlYWNoIENvbnRyaWJ1dG9yIGhlcmVieSBncmFudHMgdG8gWW91IGEgcGVycGV0dWFsLAotICAgICAgd29ybGR3aWRlLCBub24tZXhjbHVzaXZlLCBuby1jaGFyZ2UsIHJveWFsdHktZnJlZSwgaXJyZXZvY2FibGUKLSAgICAgIGNvcHlyaWdodCBsaWNlbnNlIHRvIHJlcHJvZHVjZSwgcHJlcGFyZSBEZXJpdmF0aXZlIFdvcmtzIG9mLAotICAgICAgcHVibGljbHkgZGlzcGxheSwgcHVibGljbHkgcGVyZm9ybSwgc3VibGljZW5zZSwgYW5kIGRpc3RyaWJ1dGUgdGhlCi0gICAgICBXb3JrIGFuZCBzdWNoIERlcml2YXRpdmUgV29ya3MgaW4gU291cmNlIG9yIE9iamVjdCBmb3JtLgotCi0gICAzLiBHcmFudCBvZiBQYXRlbnQgTGljZW5zZS4gU3ViamVjdCB0byB0aGUgdGVybXMgYW5kIGNvbmRpdGlvbnMgb2YKLSAgICAgIHRoaXMgTGljZW5zZSwgZWFjaCBDb250cmlidXRvciBoZXJlYnkgZ3JhbnRzIHRvIFlvdSBhIHBlcnBldHVhbCwKLSAgICAgIHdvcmxkd2lkZSwgbm9uLWV4Y2x1c2l2ZSwgbm8tY2hhcmdlLCByb3lhbHR5LWZyZWUsIGlycmV2b2NhYmxlCi0gICAgICAoZXhjZXB0IGFzIHN0YXRlZCBpbiB0aGlzIHNlY3Rpb24pIHBhdGVudCBsaWNlbnNlIHRvIG1ha2UsIGhhdmUgbWFkZSwKLSAgICAgIHVzZSwgb2ZmZXIgdG8gc2VsbCwgc2VsbCwgaW1wb3J0LCBhbmQgb3RoZXJ3aXNlIHRyYW5zZmVyIHRoZSBXb3JrLAotICAgICAgd2hlcmUgc3VjaCBsaWNlbnNlIGFwcGxpZXMgb25seSB0byB0aG9zZSBwYXRlbnQgY2xhaW1zIGxpY2Vuc2FibGUKLSAgICAgIGJ5IHN1Y2ggQ29udHJpYnV0b3IgdGhhdCBhcmUgbmVjZXNzYXJpbHkgaW5mcmluZ2VkIGJ5IHRoZWlyCi0gICAgICBDb250cmlidXRpb24ocykgYWxvbmUgb3IgYnkgY29tYmluYXRpb24gb2YgdGhlaXIgQ29udHJpYnV0aW9uKHMpCi0gICAgICB3aXRoIHRoZSBXb3JrIHRvIHdoaWNoIHN1Y2ggQ29udHJpYnV0aW9uKHMpIHdhcyBzdWJtaXR0ZWQuIElmIFlvdQotICAgICAgaW5zdGl0dXRlIHBhdGVudCBsaXRpZ2F0aW9uIGFnYWluc3QgYW55IGVudGl0eSAoaW5jbHVkaW5nIGEKLSAgICAgIGNyb3NzLWNsYWltIG9yIGNvdW50ZXJjbGFpbSBpbiBhIGxhd3N1aXQpIGFsbGVnaW5nIHRoYXQgdGhlIFdvcmsKLSAgICAgIG9yIGEgQ29udHJpYnV0aW9uIGluY29ycG9yYXRlZCB3aXRoaW4gdGhlIFdvcmsgY29uc3RpdHV0ZXMgZGlyZWN0Ci0gICAgICBvciBjb250cmlidXRvcnkgcGF0ZW50IGluZnJpbmdlbWVudCwgdGhlbiBhbnkgcGF0ZW50IGxpY2Vuc2VzCi0gICAgICBncmFudGVkIHRvIFlvdSB1bmRlciB0aGlzIExpY2Vuc2UgZm9yIHRoYXQgV29yayBzaGFsbCB0ZXJtaW5hdGUKLSAgICAgIGFzIG9mIHRoZSBkYXRlIHN1Y2ggbGl0aWdhdGlvbiBpcyBmaWxlZC4KLQotICAgNC4gUmVkaXN0cmlidXRpb24uIFlvdSBtYXkgcmVwcm9kdWNlIGFuZCBkaXN0cmlidXRlIGNvcGllcyBvZiB0aGUKLSAgICAgIFdvcmsgb3IgRGVyaXZhdGl2ZSBXb3JrcyB0aGVyZW9mIGluIGFueSBtZWRpdW0sIHdpdGggb3Igd2l0aG91dAotICAgICAgbW9kaWZpY2F0aW9ucywgYW5kIGluIFNvdXJjZSBvciBPYmplY3QgZm9ybSwgcHJvdmlkZWQgdGhhdCBZb3UKLSAgICAgIG1lZXQgdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOgotCi0gICAgICAoYSkgWW91IG11c3QgZ2l2ZSBhbnkgb3RoZXIgcmVjaXBpZW50cyBvZiB0aGUgV29yayBvcgotICAgICAgICAgIERlcml2YXRpdmUgV29ya3MgYSBjb3B5IG9mIHRoaXMgTGljZW5zZTsgYW5kCi0KLSAgICAgIChiKSBZb3UgbXVzdCBjYXVzZSBhbnkgbW9kaWZpZWQgZmlsZXMgdG8gY2FycnkgcHJvbWluZW50IG5vdGljZXMKLSAgICAgICAgICBzdGF0aW5nIHRoYXQgWW91IGNoYW5nZWQgdGhlIGZpbGVzOyBhbmQKLQotICAgICAgKGMpIFlvdSBtdXN0IHJldGFpbiwgaW4gdGhlIFNvdXJjZSBmb3JtIG9mIGFueSBEZXJpdmF0aXZlIFdvcmtzCi0gICAgICAgICAgdGhhdCBZb3UgZGlzdHJpYnV0ZSwgYWxsIGNvcHlyaWdodCwgcGF0ZW50LCB0cmFkZW1hcmssIGFuZAotICAgICAgICAgIGF0dHJpYnV0aW9uIG5vdGljZXMgZnJvbSB0aGUgU291cmNlIGZvcm0gb2YgdGhlIFdvcmssCi0gICAgICAgICAgZXhjbHVkaW5nIHRob3NlIG5vdGljZXMgdGhhdCBkbyBub3QgcGVydGFpbiB0byBhbnkgcGFydCBvZgotICAgICAgICAgIHRoZSBEZXJpdmF0aXZlIFdvcmtzOyBhbmQKLQotICAgICAgKGQpIElmIHRoZSBXb3JrIGluY2x1ZGVzIGEgIk5PVElDRSIgdGV4dCBmaWxlIGFzIHBhcnQgb2YgaXRzCi0gICAgICAgICAgZGlzdHJpYnV0aW9uLCB0aGVuIGFueSBEZXJpdmF0aXZlIFdvcmtzIHRoYXQgWW91IGRpc3RyaWJ1dGUgbXVzdAotICAgICAgICAgIGluY2x1ZGUgYSByZWFkYWJsZSBjb3B5IG9mIHRoZSBhdHRyaWJ1dGlvbiBub3RpY2VzIGNvbnRhaW5lZAotICAgICAgICAgIHdpdGhpbiBzdWNoIE5PVElDRSBmaWxlLCBleGNsdWRpbmcgdGhvc2Ugbm90aWNlcyB0aGF0IGRvIG5vdAotICAgICAgICAgIHBlcnRhaW4gdG8gYW55IHBhcnQgb2YgdGhlIERlcml2YXRpdmUgV29ya3MsIGluIGF0IGxlYXN0IG9uZQotICAgICAgICAgIG9mIHRoZSBmb2xsb3dpbmcgcGxhY2VzOiB3aXRoaW4gYSBOT1RJQ0UgdGV4dCBmaWxlIGRpc3RyaWJ1dGVkCi0gICAgICAgICAgYXMgcGFydCBvZiB0aGUgRGVyaXZhdGl2ZSBXb3Jrczsgd2l0aGluIHRoZSBTb3VyY2UgZm9ybSBvcgotICAgICAgICAgIGRvY3VtZW50YXRpb24sIGlmIHByb3ZpZGVkIGFsb25nIHdpdGggdGhlIERlcml2YXRpdmUgV29ya3M7IG9yLAotICAgICAgICAgIHdpdGhpbiBhIGRpc3BsYXkgZ2VuZXJhdGVkIGJ5IHRoZSBEZXJpdmF0aXZlIFdvcmtzLCBpZiBhbmQKLSAgICAgICAgICB3aGVyZXZlciBzdWNoIHRoaXJkLXBhcnR5IG5vdGljZXMgbm9ybWFsbHkgYXBwZWFyLiBUaGUgY29udGVudHMKLSAgICAgICAgICBvZiB0aGUgTk9USUNFIGZpbGUgYXJlIGZvciBpbmZvcm1hdGlvbmFsIHB1cnBvc2VzIG9ubHkgYW5kCi0gICAgICAgICAgZG8gbm90IG1vZGlmeSB0aGUgTGljZW5zZS4gWW91IG1heSBhZGQgWW91ciBvd24gYXR0cmlidXRpb24KLSAgICAgICAgICBub3RpY2VzIHdpdGhpbiBEZXJpdmF0aXZlIFdvcmtzIHRoYXQgWW91IGRpc3RyaWJ1dGUsIGFsb25nc2lkZQotICAgICAgICAgIG9yIGFzIGFuIGFkZGVuZHVtIHRvIHRoZSBOT1RJQ0UgdGV4dCBmcm9tIHRoZSBXb3JrLCBwcm92aWRlZAotICAgICAgICAgIHRoYXQgc3VjaCBhZGRpdGlvbmFsIGF0dHJpYnV0aW9uIG5vdGljZXMgY2Fubm90IGJlIGNvbnN0cnVlZAotICAgICAgICAgIGFzIG1vZGlmeWluZyB0aGUgTGljZW5zZS4KLQotICAgICAgWW91IG1heSBhZGQgWW91ciBvd24gY29weXJpZ2h0IHN0YXRlbWVudCB0byBZb3VyIG1vZGlmaWNhdGlvbnMgYW5kCi0gICAgICBtYXkgcHJvdmlkZSBhZGRpdGlvbmFsIG9yIGRpZmZlcmVudCBsaWNlbnNlIHRlcm1zIGFuZCBjb25kaXRpb25zCi0gICAgICBmb3IgdXNlLCByZXByb2R1Y3Rpb24sIG9yIGRpc3RyaWJ1dGlvbiBvZiBZb3VyIG1vZGlmaWNhdGlvbnMsIG9yCi0gICAgICBmb3IgYW55IHN1Y2ggRGVyaXZhdGl2ZSBXb3JrcyBhcyBhIHdob2xlLCBwcm92aWRlZCBZb3VyIHVzZSwKLSAgICAgIHJlcHJvZHVjdGlvbiwgYW5kIGRpc3RyaWJ1dGlvbiBvZiB0aGUgV29yayBvdGhlcndpc2UgY29tcGxpZXMgd2l0aAotICAgICAgdGhlIGNvbmRpdGlvbnMgc3RhdGVkIGluIHRoaXMgTGljZW5zZS4KLQotICAgNS4gU3VibWlzc2lvbiBvZiBDb250cmlidXRpb25zLiBVbmxlc3MgWW91IGV4cGxpY2l0bHkgc3RhdGUgb3RoZXJ3aXNlLAotICAgICAgYW55IENvbnRyaWJ1dGlvbiBpbnRlbnRpb25hbGx5IHN1Ym1pdHRlZCBmb3IgaW5jbHVzaW9uIGluIHRoZSBXb3JrCi0gICAgICBieSBZb3UgdG8gdGhlIExpY2Vuc29yIHNoYWxsIGJlIHVuZGVyIHRoZSB0ZXJtcyBhbmQgY29uZGl0aW9ucyBvZgotICAgICAgdGhpcyBMaWNlbnNlLCB3aXRob3V0IGFueSBhZGRpdGlvbmFsIHRlcm1zIG9yIGNvbmRpdGlvbnMuCi0gICAgICBOb3R3aXRoc3RhbmRpbmcgdGhlIGFib3ZlLCBub3RoaW5nIGhlcmVpbiBzaGFsbCBzdXBlcnNlZGUgb3IgbW9kaWZ5Ci0gICAgICB0aGUgdGVybXMgb2YgYW55IHNlcGFyYXRlIGxpY2Vuc2UgYWdyZWVtZW50IHlvdSBtYXkgaGF2ZSBleGVjdXRlZAotICAgICAgd2l0aCBMaWNlbnNvciByZWdhcmRpbmcgc3VjaCBDb250cmlidXRpb25zLgotCi0gICA2LiBUcmFkZW1hcmtzLiBUaGlzIExpY2Vuc2UgZG9lcyBub3QgZ3JhbnQgcGVybWlzc2lvbiB0byB1c2UgdGhlIHRyYWRlCi0gICAgICBuYW1lcywgdHJhZGVtYXJrcywgc2VydmljZSBtYXJrcywgb3IgcHJvZHVjdCBuYW1lcyBvZiB0aGUgTGljZW5zb3IsCi0gICAgICBleGNlcHQgYXMgcmVxdWlyZWQgZm9yIHJlYXNvbmFibGUgYW5kIGN1c3RvbWFyeSB1c2UgaW4gZGVzY3JpYmluZyB0aGUKLSAgICAgIG9yaWdpbiBvZiB0aGUgV29yayBhbmQgcmVwcm9kdWNpbmcgdGhlIGNvbnRlbnQgb2YgdGhlIE5PVElDRSBmaWxlLgotCi0gICA3LiBEaXNjbGFpbWVyIG9mIFdhcnJhbnR5LiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IKLSAgICAgIGFncmVlZCB0byBpbiB3cml0aW5nLCBMaWNlbnNvciBwcm92aWRlcyB0aGUgV29yayAoYW5kIGVhY2gKLSAgICAgIENvbnRyaWJ1dG9yIHByb3ZpZGVzIGl0cyBDb250cmlidXRpb25zKSBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICAgICAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yCi0gICAgICBpbXBsaWVkLCBpbmNsdWRpbmcsIHdpdGhvdXQgbGltaXRhdGlvbiwgYW55IHdhcnJhbnRpZXMgb3IgY29uZGl0aW9ucwotICAgICAgb2YgVElUTEUsIE5PTi1JTkZSSU5HRU1FTlQsIE1FUkNIQU5UQUJJTElUWSwgb3IgRklUTkVTUyBGT1IgQQotICAgICAgUEFSVElDVUxBUiBQVVJQT1NFLiBZb3UgYXJlIHNvbGVseSByZXNwb25zaWJsZSBmb3IgZGV0ZXJtaW5pbmcgdGhlCi0gICAgICBhcHByb3ByaWF0ZW5lc3Mgb2YgdXNpbmcgb3IgcmVkaXN0cmlidXRpbmcgdGhlIFdvcmsgYW5kIGFzc3VtZSBhbnkKLSAgICAgIHJpc2tzIGFzc29jaWF0ZWQgd2l0aCBZb3VyIGV4ZXJjaXNlIG9mIHBlcm1pc3Npb25zIHVuZGVyIHRoaXMgTGljZW5zZS4KLQotICAgOC4gTGltaXRhdGlvbiBvZiBMaWFiaWxpdHkuIEluIG5vIGV2ZW50IGFuZCB1bmRlciBubyBsZWdhbCB0aGVvcnksCi0gICAgICB3aGV0aGVyIGluIHRvcnQgKGluY2x1ZGluZyBuZWdsaWdlbmNlKSwgY29udHJhY3QsIG9yIG90aGVyd2lzZSwKLSAgICAgIHVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyAoc3VjaCBhcyBkZWxpYmVyYXRlIGFuZCBncm9zc2x5Ci0gICAgICBuZWdsaWdlbnQgYWN0cykgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNoYWxsIGFueSBDb250cmlidXRvciBiZQotICAgICAgbGlhYmxlIHRvIFlvdSBmb3IgZGFtYWdlcywgaW5jbHVkaW5nIGFueSBkaXJlY3QsIGluZGlyZWN0LCBzcGVjaWFsLAotICAgICAgaW5jaWRlbnRhbCwgb3IgY29uc2VxdWVudGlhbCBkYW1hZ2VzIG9mIGFueSBjaGFyYWN0ZXIgYXJpc2luZyBhcyBhCi0gICAgICByZXN1bHQgb2YgdGhpcyBMaWNlbnNlIG9yIG91dCBvZiB0aGUgdXNlIG9yIGluYWJpbGl0eSB0byB1c2UgdGhlCi0gICAgICBXb3JrIChpbmNsdWRpbmcgYnV0IG5vdCBsaW1pdGVkIHRvIGRhbWFnZXMgZm9yIGxvc3Mgb2YgZ29vZHdpbGwsCi0gICAgICB3b3JrIHN0b3BwYWdlLCBjb21wdXRlciBmYWlsdXJlIG9yIG1hbGZ1bmN0aW9uLCBvciBhbnkgYW5kIGFsbAotICAgICAgb3RoZXIgY29tbWVyY2lhbCBkYW1hZ2VzIG9yIGxvc3NlcyksIGV2ZW4gaWYgc3VjaCBDb250cmlidXRvcgotICAgICAgaGFzIGJlZW4gYWR2aXNlZCBvZiB0aGUgcG9zc2liaWxpdHkgb2Ygc3VjaCBkYW1hZ2VzLgotCi0gICA5LiBBY2NlcHRpbmcgV2FycmFudHkgb3IgQWRkaXRpb25hbCBMaWFiaWxpdHkuIFdoaWxlIHJlZGlzdHJpYnV0aW5nCi0gICAgICB0aGUgV29yayBvciBEZXJpdmF0aXZlIFdvcmtzIHRoZXJlb2YsIFlvdSBtYXkgY2hvb3NlIHRvIG9mZmVyLAotICAgICAgYW5kIGNoYXJnZSBhIGZlZSBmb3IsIGFjY2VwdGFuY2Ugb2Ygc3VwcG9ydCwgd2FycmFudHksIGluZGVtbml0eSwKLSAgICAgIG9yIG90aGVyIGxpYWJpbGl0eSBvYmxpZ2F0aW9ucyBhbmQvb3IgcmlnaHRzIGNvbnNpc3RlbnQgd2l0aCB0aGlzCi0gICAgICBMaWNlbnNlLiBIb3dldmVyLCBpbiBhY2NlcHRpbmcgc3VjaCBvYmxpZ2F0aW9ucywgWW91IG1heSBhY3Qgb25seQotICAgICAgb24gWW91ciBvd24gYmVoYWxmIGFuZCBvbiBZb3VyIHNvbGUgcmVzcG9uc2liaWxpdHksIG5vdCBvbiBiZWhhbGYKLSAgICAgIG9mIGFueSBvdGhlciBDb250cmlidXRvciwgYW5kIG9ubHkgaWYgWW91IGFncmVlIHRvIGluZGVtbmlmeSwKLSAgICAgIGRlZmVuZCwgYW5kIGhvbGQgZWFjaCBDb250cmlidXRvciBoYXJtbGVzcyBmb3IgYW55IGxpYWJpbGl0eQotICAgICAgaW5jdXJyZWQgYnksIG9yIGNsYWltcyBhc3NlcnRlZCBhZ2FpbnN0LCBzdWNoIENvbnRyaWJ1dG9yIGJ5IHJlYXNvbgotICAgICAgb2YgeW91ciBhY2NlcHRpbmcgYW55IHN1Y2ggd2FycmFudHkgb3IgYWRkaXRpb25hbCBsaWFiaWxpdHkuCi0KLSAgIEVORCBPRiBURVJNUyBBTkQgQ09ORElUSU9OUwotCmRpZmYgLS1naXQgYS9saWJzL3VpL092ZXJsYXkuY3BwIGIvbGlicy91aS9PdmVybGF5LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYjIzNmVkYy4uMDAwMDAwMAotLS0gYS9saWJzL3VpL092ZXJsYXkuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTgxICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaW5jbHVkZSA8dXRpbHMvSU1lbW9yeS5oPgotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL01lbW9yeUhlYXBCYXNlLmg+Ci0KLSNpbmNsdWRlIDx1aS9JT3ZlcmxheS5oPgotI2luY2x1ZGUgPHVpL092ZXJsYXkuaD4KLQotI2luY2x1ZGUgPGhhcmR3YXJlL292ZXJsYXkuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1PdmVybGF5OjpPdmVybGF5KGNvbnN0IHNwPE92ZXJsYXlSZWY+JiBvdmVybGF5UmVmKQotICAgIDogbU92ZXJsYXlSZWYob3ZlcmxheVJlZiksIG1PdmVybGF5RGF0YSgwKSwgbVN0YXR1cyhOT19JTklUKQotewotICAgIG1PdmVybGF5RGF0YSA9IE5VTEw7Ci0gICAgaHdfbW9kdWxlX3QgY29uc3QqIG1vZHVsZTsKLSAgICBpZiAob3ZlcmxheVJlZiAhPSAwKSB7Ci0gICAgICAgIGlmIChod19nZXRfbW9kdWxlKE9WRVJMQVlfSEFSRFdBUkVfTU9EVUxFX0lELCAmbW9kdWxlKSA9PSAwKSB7Ci0gICAgICAgICAgICBpZiAob3ZlcmxheV9kYXRhX29wZW4obW9kdWxlLCAmbU92ZXJsYXlEYXRhKSA9PSBOT19FUlJPUikgewotICAgICAgICAgICAgICAgIG1TdGF0dXMgPSBtT3ZlcmxheURhdGEtPmluaXRpYWxpemUobU92ZXJsYXlEYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgb3ZlcmxheVJlZi0+bU92ZXJsYXlIYW5kbGUpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQotCi1PdmVybGF5Ojp+T3ZlcmxheSgpIHsKLSAgICBpZiAobU92ZXJsYXlEYXRhKSB7Ci0gICAgICAgIG92ZXJsYXlfZGF0YV9jbG9zZShtT3ZlcmxheURhdGEpOwotICAgIH0KLX0KLQotc3RhdHVzX3QgT3ZlcmxheTo6ZGVxdWV1ZUJ1ZmZlcihvdmVybGF5X2J1ZmZlcl90KiBidWZmZXIpCi17Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpIHJldHVybiBtU3RhdHVzOwotICAgIHJldHVybiAgbU92ZXJsYXlEYXRhLT5kZXF1ZXVlQnVmZmVyKG1PdmVybGF5RGF0YSwgYnVmZmVyKTsKLX0KLQotc3RhdHVzX3QgT3ZlcmxheTo6cXVldWVCdWZmZXIob3ZlcmxheV9idWZmZXJfdCBidWZmZXIpCi17Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpIHJldHVybiBtU3RhdHVzOwotICAgIHJldHVybiBtT3ZlcmxheURhdGEtPnF1ZXVlQnVmZmVyKG1PdmVybGF5RGF0YSwgYnVmZmVyKTsKLX0KLQotaW50MzJfdCBPdmVybGF5OjpnZXRCdWZmZXJDb3VudCgpIGNvbnN0Ci17Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpIHJldHVybiBtU3RhdHVzOwotICAgIHJldHVybiBtT3ZlcmxheURhdGEtPmdldEJ1ZmZlckNvdW50KG1PdmVybGF5RGF0YSk7Ci19Ci0KLXZvaWQqIE92ZXJsYXk6OmdldEJ1ZmZlckFkZHJlc3Mob3ZlcmxheV9idWZmZXJfdCBidWZmZXIpCi17Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpIHJldHVybiBOVUxMOwotICAgIHJldHVybiBtT3ZlcmxheURhdGEtPmdldEJ1ZmZlckFkZHJlc3MobU92ZXJsYXlEYXRhLCBidWZmZXIpOwotfQotCi12b2lkIE92ZXJsYXk6OmRlc3Ryb3koKSB7ICAKLSAgICBpZiAobVN0YXR1cyAhPSBOT19FUlJPUikgcmV0dXJuOwotICAgIG1PdmVybGF5UmVmLT5tT3ZlcmxheUNoYW5uZWwtPmRlc3Ryb3koKTsKLX0KLQotc3RhdHVzX3QgT3ZlcmxheTo6Z2V0U3RhdHVzKCkgY29uc3QgewotICAgIHJldHVybiBtU3RhdHVzOwotfQotCi1vdmVybGF5X2hhbmRsZV90IE92ZXJsYXk6OmdldEhhbmRsZVJlZigpIGNvbnN0IHsKLSAgICBpZiAobVN0YXR1cyAhPSBOT19FUlJPUikgcmV0dXJuIE5VTEw7Ci0gICAgcmV0dXJuIG1PdmVybGF5UmVmLT5tT3ZlcmxheUhhbmRsZTsKLX0KLQotdWludDMyX3QgT3ZlcmxheTo6Z2V0V2lkdGgoKSBjb25zdCB7Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpIHJldHVybiAwOwotICAgIHJldHVybiBtT3ZlcmxheVJlZi0+bVdpZHRoOwotfQotCi11aW50MzJfdCBPdmVybGF5OjpnZXRIZWlnaHQoKSBjb25zdCB7Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpIHJldHVybiAwOwotICAgIHJldHVybiBtT3ZlcmxheVJlZi0+bUhlaWdodDsKLX0KLQotaW50MzJfdCBPdmVybGF5OjpnZXRGb3JtYXQoKSBjb25zdCB7Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpIHJldHVybiAtMTsKLSAgICByZXR1cm4gbU92ZXJsYXlSZWYtPm1Gb3JtYXQ7Ci19Ci0KLWludDMyX3QgT3ZlcmxheTo6Z2V0V2lkdGhTdHJpZGUoKSBjb25zdCB7Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpIHJldHVybiAwOwotICAgIHJldHVybiBtT3ZlcmxheVJlZi0+bVdpZHRoU3RyaWRlOwotfQotCi1pbnQzMl90IE92ZXJsYXk6OmdldEhlaWdodFN0cmlkZSgpIGNvbnN0IHsKLSAgICBpZiAobVN0YXR1cyAhPSBOT19FUlJPUikgcmV0dXJuIDA7Ci0gICAgcmV0dXJuIG1PdmVybGF5UmVmLT5tSGVpZ2h0U3RyaWRlOwotfQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1PdmVybGF5UmVmOjpPdmVybGF5UmVmKCkgCi0gOiBtT3ZlcmxheUhhbmRsZSgwKSwKLSAgICBtV2lkdGgoMCksIG1IZWlnaHQoMCksIG1Gb3JtYXQoMCksIG1XaWR0aFN0cmlkZSgwKSwgbUhlaWdodFN0cmlkZSgwKSwKLSAgICBtT3duSGFuZGxlKHRydWUpCi17ICAgIAotfQotCi1PdmVybGF5UmVmOjpPdmVybGF5UmVmKG92ZXJsYXlfaGFuZGxlX3QgaGFuZGxlLCBjb25zdCBzcDxJT3ZlcmxheT4mIGNoYW5uZWwsCi0gICAgICAgICB1aW50MzJfdCB3LCB1aW50MzJfdCBoLCBpbnQzMl90IGYsIHVpbnQzMl90IHdzLCB1aW50MzJfdCBocykKLSAgICA6IG1PdmVybGF5SGFuZGxlKGhhbmRsZSksIG1PdmVybGF5Q2hhbm5lbChjaGFubmVsKSwKLSAgICBtV2lkdGgodyksIG1IZWlnaHQoaCksIG1Gb3JtYXQoZiksIG1XaWR0aFN0cmlkZSh3cyksIG1IZWlnaHRTdHJpZGUoaHMpLAotICAgIG1Pd25IYW5kbGUoZmFsc2UpCi17Ci19Ci0KLU92ZXJsYXlSZWY6On5PdmVybGF5UmVmKCkKLXsKLSAgICBpZiAobU93bkhhbmRsZSkgewotICAgICAgICAvKiBGSVhNRTogaGFuZGxlcyBzaG91bGQgYmUgcHJvbW90ZWQgdG8gInJlYWwiIEFQSSBhbmQgYmUgaGFuZGxlZCBieSAKLSAgICAgICAgICogdGhlIGZyYW1ld29yayAqLwotICAgICAgICBmb3IgKGludCBpPTAgOyBpPG1PdmVybGF5SGFuZGxlLT5udW1GZHMgOyBpKyspIHsKLSAgICAgICAgICAgIGNsb3NlKG1PdmVybGF5SGFuZGxlLT5kYXRhW2ldKTsKLSAgICAgICAgfQotICAgICAgICBmcmVlKCh2b2lkKiltT3ZlcmxheUhhbmRsZSk7Ci0gICAgfQotfQotCi1zcDxPdmVybGF5UmVmPiBPdmVybGF5UmVmOjpyZWFkRnJvbVBhcmNlbChjb25zdCBQYXJjZWwmIGRhdGEpIHsKLSAgICBzcDxPdmVybGF5UmVmPiByZXN1bHQ7Ci0gICAgc3A8SU92ZXJsYXk+IG92ZXJsYXkgPSBJT3ZlcmxheTo6YXNJbnRlcmZhY2UoZGF0YS5yZWFkU3Ryb25nQmluZGVyKCkpOwotICAgIGlmIChvdmVybGF5ICE9IE5VTEwpIHsKLSAgICAgICAgdWludDMyX3QgdyA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgIHVpbnQzMl90IGggPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICB1aW50MzJfdCBmID0gZGF0YS5yZWFkSW50MzIoKTsKLSAgICAgICAgdWludDMyX3Qgd3MgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICB1aW50MzJfdCBocyA9IGRhdGEucmVhZEludDMyKCk7Ci0gICAgICAgIG5hdGl2ZV9oYW5kbGUqIGhhbmRsZSA9IGRhdGEucmVhZE5hdGl2ZUhhbmRsZShOVUxMLCBOVUxMKTsKLQotICAgICAgICByZXN1bHQgPSBuZXcgT3ZlcmxheVJlZigpOwotICAgICAgICByZXN1bHQtPm1PdmVybGF5SGFuZGxlID0gaGFuZGxlOwotICAgICAgICByZXN1bHQtPm1PdmVybGF5Q2hhbm5lbCA9IG92ZXJsYXk7Ci0gICAgICAgIHJlc3VsdC0+bVdpZHRoID0gdzsKLSAgICAgICAgcmVzdWx0LT5tSGVpZ2h0ID0gaDsKLSAgICAgICAgcmVzdWx0LT5tRm9ybWF0ID0gZjsKLSAgICAgICAgcmVzdWx0LT5tV2lkdGhTdHJpZGUgPSB3czsKLSAgICAgICAgcmVzdWx0LT5tSGVpZ2h0U3RyaWRlID0gaHM7Ci0gICAgfQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLXN0YXR1c190IE92ZXJsYXlSZWY6OndyaXRlVG9QYXJjZWwoUGFyY2VsKiByZXBseSwgY29uc3Qgc3A8T3ZlcmxheVJlZj4mIG8pIHsKLSAgICBpZiAobyAhPSBOVUxMKSB7Ci0gICAgICAgIHJlcGx5LT53cml0ZVN0cm9uZ0JpbmRlcihvLT5tT3ZlcmxheUNoYW5uZWwtPmFzQmluZGVyKCkpOwotICAgICAgICByZXBseS0+d3JpdGVJbnQzMihvLT5tV2lkdGgpOwotICAgICAgICByZXBseS0+d3JpdGVJbnQzMihvLT5tSGVpZ2h0KTsKLSAgICAgICAgcmVwbHktPndyaXRlSW50MzIoby0+bUZvcm1hdCk7Ci0gICAgICAgIHJlcGx5LT53cml0ZUludDMyKG8tPm1XaWR0aFN0cmlkZSk7Ci0gICAgICAgIHJlcGx5LT53cml0ZUludDMyKG8tPm1IZWlnaHRTdHJpZGUpOwotICAgICAgICByZXBseS0+d3JpdGVOYXRpdmVIYW5kbGUoKihvLT5tT3ZlcmxheUhhbmRsZSkpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHJlcGx5LT53cml0ZVN0cm9uZ0JpbmRlcihOVUxMKTsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3VpL1BpeGVsRm9ybWF0LmNwcCBiL2xpYnMvdWkvUGl4ZWxGb3JtYXQuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiNjVlZDk3Li4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvUGl4ZWxGb3JtYXQuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsOTcgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDx1aS9QaXhlbEZvcm1hdC5oPgotI2luY2x1ZGUgPHBpeGVsZmxpbmdlci9mb3JtYXQuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1zaXplX3QgUGl4ZWxGb3JtYXRJbmZvOjpnZXRTY2FubGluZVNpemUodW5zaWduZWQgaW50IHdpZHRoKSBjb25zdAotewotICAgIHNpemVfdCBzaXplOwotICAgIGlmICgoY29tcG9uZW50cyA+PSA2KSAmJiAoY29tcG9uZW50cyA8PSA4KSkgewotICAgICAgICAvLyBZQ2JDciBmb3JtYXRzIGFyZSBkaWZmZXJlbnRzLgotICAgICAgICBzaXplID0gKHdpZHRoICogYml0c1BlclBpeGVsKT4+MzsKLSAgICB9IGVsc2UgewotICAgICAgICBzaXplID0gd2lkdGggKiBieXRlc1BlclBpeGVsOwotICAgIH0KLSAgICByZXR1cm4gc2l6ZTsKLX0KLQotc3NpemVfdCBieXRlc1BlclBpeGVsKFBpeGVsRm9ybWF0IGZvcm1hdCkKLXsKLSAgICBQaXhlbEZvcm1hdEluZm8gaW5mbzsKLSAgICBzdGF0dXNfdCBlcnIgPSBnZXRQaXhlbEZvcm1hdEluZm8oZm9ybWF0LCAmaW5mbyk7Ci0gICAgcmV0dXJuIChlcnIgPCAwKSA/IGVyciA6IGluZm8uYnl0ZXNQZXJQaXhlbDsKLX0KLQotc3NpemVfdCBiaXRzUGVyUGl4ZWwoUGl4ZWxGb3JtYXQgZm9ybWF0KQotewotICAgIFBpeGVsRm9ybWF0SW5mbyBpbmZvOwotICAgIHN0YXR1c190IGVyciA9IGdldFBpeGVsRm9ybWF0SW5mbyhmb3JtYXQsICZpbmZvKTsKLSAgICByZXR1cm4gKGVyciA8IDApID8gZXJyIDogaW5mby5iaXRzUGVyUGl4ZWw7Ci19Ci0KLXN0YXR1c190IGdldFBpeGVsRm9ybWF0SW5mbyhQaXhlbEZvcm1hdCBmb3JtYXQsIFBpeGVsRm9ybWF0SW5mbyogaW5mbykKLXsKLSAgICBpZiAoZm9ybWF0IDwgMCkKLSAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLQotICAgIGlmIChpbmZvLT52ZXJzaW9uICE9IHNpemVvZihQaXhlbEZvcm1hdEluZm8pKQotICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0KLSAgICBzaXplX3QgbnVtRW50cmllczsKLSAgICBjb25zdCBHR0xGb3JtYXQgKmkgPSBnZ2xHZXRQaXhlbEZvcm1hdFRhYmxlKCZudW1FbnRyaWVzKSArIGZvcm1hdDsKLSAgICBib29sIHZhbGlkID0gdWludDMyX3QoZm9ybWF0KSA8IG51bUVudHJpZXM7Ci0gICAgaWYgKCF2YWxpZCkgewotICAgICAgICByZXR1cm4gQkFEX0lOREVYOwotICAgIH0KLSAgICAKLSAgICAjZGVmaW5lIENPTVBPTkVOVChuYW1lKSBcIAotICAgICAgICBjYXNlIEdHTF8jI25hbWU6IGluZm8tPmNvbXBvbmVudHMgPSBQaXhlbEZvcm1hdEluZm86Om5hbWU7IGJyZWFrOwotICAgIAotICAgIHN3aXRjaCAoaS0+Y29tcG9uZW50cykgewotICAgICAgICBDT01QT05FTlQoQUxQSEEpCi0gICAgICAgIENPTVBPTkVOVChSR0IpCi0gICAgICAgIENPTVBPTkVOVChSR0JBKQotICAgICAgICBDT01QT05FTlQoTFVNSU5BTkNFKQotICAgICAgICBDT01QT05FTlQoTFVNSU5BTkNFX0FMUEhBKQotICAgICAgICBDT01QT05FTlQoWV9DQl9DUl9TUCkKLSAgICAgICAgQ09NUE9ORU5UKFlfQ0JfQ1JfUCkKLSAgICAgICAgQ09NUE9ORU5UKFlfQ0JfQ1JfSSkKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHJldHVybiBCQURfSU5ERVg7Ci0gICAgfQotICAgIAotICAgICN1bmRlZiBDT01QT05FTlQKLSAgICAKLSAgICBpbmZvLT5mb3JtYXQgPSBmb3JtYXQ7Ci0gICAgaW5mby0+Ynl0ZXNQZXJQaXhlbCA9IGktPnNpemU7Ci0gICAgaW5mby0+Yml0c1BlclBpeGVsICA9IGktPmJpdHNQZXJQaXhlbDsKLSAgICBpbmZvLT5oX2FscGhhICAgICAgID0gaS0+YWg7Ci0gICAgaW5mby0+bF9hbHBoYSAgICAgICA9IGktPmFsOwotICAgIGluZm8tPmhfcmVkICAgICAgICAgPSBpLT5yaDsKLSAgICBpbmZvLT5sX3JlZCAgICAgICAgID0gaS0+cmw7Ci0gICAgaW5mby0+aF9ncmVlbiAgICAgICA9IGktPmdoOwotICAgIGluZm8tPmxfZ3JlZW4gICAgICAgPSBpLT5nbDsKLSAgICBpbmZvLT5oX2JsdWUgICAgICAgID0gaS0+Ymg7Ci0gICAgaW5mby0+bF9ibHVlICAgICAgICA9IGktPmJsOwotCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCmRpZmYgLS1naXQgYS9saWJzL3VpL1BvaW50LmNwcCBiL2xpYnMvdWkvUG9pbnQuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0MzhkNDlmLi4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvUG9pbnQuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTEgKzAsMCBAQAotLyoKLSAqICBQb2ludC5jcHAKLSAqICBBbmRyb2lkCi0gKgotICogIENyZWF0ZWQgb24gMTEvMTYvMjAwNi4KLSAqICBDb3B5cmlnaHQgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICovCi0KLSNpbmNsdWRlIDx1aS9Qb2ludC5oPgotCmRpZmYgLS1naXQgYS9saWJzL3VpL1JlY3QuY3BwIGIvbGlicy91aS9SZWN0LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTllNjhiYi4uMDAwMDAwMAotLS0gYS9saWJzL3VpL1JlY3QuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsODYgKzAsMCBAQAotLyoKLSAqICBSZWN0LmNwcAotICogIEFuZHJvaWQKLSAqCi0gKiAgQ3JlYXRlZCBvbiAxMC8xNC8wNS4KLSAqICBDb3B5cmlnaHQgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICovCi0KLSNpbmNsdWRlIDx1aS9SZWN0Lmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotaW5saW5lIGludCBtaW4oaW50IGEsIGludCBiKSB7Ci0gICAgcmV0dXJuIChhPGIpID8gYSA6IGI7Ci19Ci0KLWlubGluZSBpbnQgbWF4KGludCBhLCBpbnQgYikgewotICAgIHJldHVybiAoYT5iKSA/IGEgOiBiOwotfQotCi12b2lkIFJlY3Q6Om1ha2VJbnZhbGlkKCkgewotICAgIGxlZnQgPSAwOwotICAgIHRvcCA9IDA7Ci0gICAgcmlnaHQgPSAtMTsKLSAgICBib3R0b20gPSAtMTsKLX0KLQotYm9vbCBSZWN0OjpvcGVyYXRvciA8IChjb25zdCBSZWN0JiByaHMpIGNvbnN0Ci17Ci0gICAgaWYgKHRvcDxyaHMudG9wKSB7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0gZWxzZSBpZiAodG9wID09IHJocy50b3ApIHsKLSAgICAgICAgaWYgKGxlZnQgPCByaHMubGVmdCkgewotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0gZWxzZSBpZiAobGVmdCA9PSByaHMubGVmdCkgewotICAgICAgICAgICAgaWYgKGJvdHRvbTxyaHMuYm90dG9tKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKGJvdHRvbSA9PSByaHMuYm90dG9tKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHJpZ2h0PHJocy5yaWdodCkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIGZhbHNlOwotfQotCi1SZWN0JiBSZWN0OjpvZmZzZXRUbyhpbnQgeCwgaW50IHkpCi17Ci0gICAgcmlnaHQgLT0gbGVmdCAtIHg7Ci0gICAgYm90dG9tIC09IHRvcCAtIHk7Ci0gICAgbGVmdCA9IHg7Ci0gICAgdG9wID0geTsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLVJlY3QmIFJlY3Q6Om9mZnNldEJ5KGludCB4LCBpbnQgeSkKLXsKLSAgICBsZWZ0ICs9IHg7Ci0gICAgdG9wICArPSB5OwotICAgIHJpZ2h0Kz0geDsKLSAgICBib3R0b20rPXk7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi1SZWN0IFJlY3Q6Om9wZXJhdG9yICsgKGNvbnN0IFBvaW50JiByaHMpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIFJlY3QobGVmdCtyaHMueCwgdG9wK3Jocy55LCByaWdodCtyaHMueCwgYm90dG9tK3Jocy55KTsgCi19Ci0KLVJlY3QgUmVjdDo6b3BlcmF0b3IgLSAoY29uc3QgUG9pbnQmIHJocykgY29uc3QKLXsKLSAgICByZXR1cm4gUmVjdChsZWZ0LXJocy54LCB0b3AtcmhzLnksIHJpZ2h0LXJocy54LCBib3R0b20tcmhzLnkpOyAKLX0KLQotYm9vbCBSZWN0OjppbnRlcnNlY3QoY29uc3QgUmVjdCYgd2l0aCwgUmVjdCogcmVzdWx0KSBjb25zdAotewotICAgIHJlc3VsdC0+bGVmdCAgICA9IG1heChsZWZ0LCB3aXRoLmxlZnQpOwotICAgIHJlc3VsdC0+dG9wICAgICA9IG1heCh0b3AsIHdpdGgudG9wKTsKLSAgICByZXN1bHQtPnJpZ2h0ICAgPSBtaW4ocmlnaHQsIHdpdGgucmlnaHQpOwotICAgIHJlc3VsdC0+Ym90dG9tICA9IG1pbihib3R0b20sIHdpdGguYm90dG9tKTsKLSAgICByZXR1cm4gIShyZXN1bHQtPmlzRW1wdHkoKSk7Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3VpL1JlZ2lvbi5jcHAgYi9saWJzL3VpL1JlZ2lvbi5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI2ZTY5NGEuLjAwMDAwMDAKLS0tIGEvbGlicy91aS9SZWdpb24uY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMzEzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIlJlZ2lvbiIKLQotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8dXRpbHMvQXRvbWljLmg+Ci0jaW5jbHVkZSA8dXRpbHMvRGVidWcuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0jaW5jbHVkZSA8dWkvUmVnaW9uLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1SZWdpb246OlJlZ2lvbigpCi17Ci19Ci0KLVJlZ2lvbjo6UmVnaW9uKGNvbnN0IFJlZ2lvbiYgcmhzKQotICAgIDogbVJlZ2lvbihyaHMubVJlZ2lvbikKLXsKLX0KLQotUmVnaW9uOjpSZWdpb24oY29uc3QgU2tSZWdpb24mIHJocykKLSAgICA6IG1SZWdpb24ocmhzKQotewotfQotCi1SZWdpb246On5SZWdpb24oKQotewotfQotCi1SZWdpb246OlJlZ2lvbihjb25zdCBSZWN0JiByaHMpCi17Ci0gICAgc2V0KHJocyk7Ci19Ci0KLVJlZ2lvbjo6UmVnaW9uKGNvbnN0IFBhcmNlbCYgcGFyY2VsKQotewotICAgIHJlYWQocGFyY2VsKTsKLX0KLQotUmVnaW9uOjpSZWdpb24oY29uc3Qgdm9pZCogYnVmZmVyKQotewotICAgIHJlYWQoYnVmZmVyKTsKLX0KLQotUmVnaW9uJiBSZWdpb246Om9wZXJhdG9yID0gKGNvbnN0IFJlZ2lvbiYgcmhzKQotewotICAgIG1SZWdpb24gPSByaHMubVJlZ2lvbjsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLWNvbnN0IFNrUmVnaW9uJiBSZWdpb246OnRvU2tSZWdpb24oKSBjb25zdAotewotICAgIHJldHVybiBtUmVnaW9uOwotfQotCi1SZWN0IFJlZ2lvbjo6Ym91bmRzKCkgY29uc3QKLXsKLSAgICBjb25zdCBTa0lSZWN0JiBiKG1SZWdpb24uZ2V0Qm91bmRzKCkpOwotICAgIHJldHVybiBSZWN0KGIuZkxlZnQsIGIuZlRvcCwgYi5mUmlnaHQsIGIuZkJvdHRvbSk7Ci19Ci0KLXZvaWQgUmVnaW9uOjpjbGVhcigpCi17Ci0gICAgbVJlZ2lvbi5zZXRFbXB0eSgpOwotfQotCi12b2lkIFJlZ2lvbjo6c2V0KGNvbnN0IFJlY3QmIHIpCi17Ci0gICAgU2tJUmVjdCBpcjsKLSAgICBpci5zZXQoci5sZWZ0LCByLnRvcCwgci5yaWdodCwgci5ib3R0b20pOwotICAgIG1SZWdpb24uc2V0UmVjdChpcik7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotUmVnaW9uJiBSZWdpb246Om9yU2VsZihjb25zdCBSZWN0JiByKQotewotICAgIFNrSVJlY3QgaXI7Ci0gICAgaXIuc2V0KHIubGVmdCwgci50b3AsIHIucmlnaHQsIHIuYm90dG9tKTsKLSAgICBtUmVnaW9uLm9wKGlyLCBTa1JlZ2lvbjo6a1VuaW9uX09wKTsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLVJlZ2lvbiYgUmVnaW9uOjphbmRTZWxmKGNvbnN0IFJlY3QmIHIpCi17Ci0gICAgU2tJUmVjdCBpcjsKLSAgICBpci5zZXQoci5sZWZ0LCByLnRvcCwgci5yaWdodCwgci5ib3R0b20pOwotICAgIG1SZWdpb24ub3AoaXIsIFNrUmVnaW9uOjprSW50ZXJzZWN0X09wKTsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotUmVnaW9uJiBSZWdpb246Om9yU2VsZihjb25zdCBSZWdpb24mIHJocykgewotICAgIG1SZWdpb24ub3AocmhzLm1SZWdpb24sIFNrUmVnaW9uOjprVW5pb25fT3ApOwotICAgIHJldHVybiAqdGhpczsKLX0KLQotUmVnaW9uJiBSZWdpb246OmFuZFNlbGYoY29uc3QgUmVnaW9uJiByaHMpIHsKLSAgICBtUmVnaW9uLm9wKHJocy5tUmVnaW9uLCBTa1JlZ2lvbjo6a0ludGVyc2VjdF9PcCk7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi1SZWdpb24mIFJlZ2lvbjo6c3VidHJhY3RTZWxmKGNvbnN0IFJlZ2lvbiYgcmhzKSB7Ci0gICAgbVJlZ2lvbi5vcChyaHMubVJlZ2lvbiwgU2tSZWdpb246OmtEaWZmZXJlbmNlX09wKTsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLVJlZ2lvbiYgUmVnaW9uOjp0cmFuc2xhdGVTZWxmKGludCB4LCBpbnQgeSkgewotICAgIGlmICh4fHkpIG1SZWdpb24udHJhbnNsYXRlKHgsIHkpOwotICAgIHJldHVybiAqdGhpczsKLX0KLQotUmVnaW9uIFJlZ2lvbjo6bWVyZ2UoY29uc3QgUmVnaW9uJiByaHMpIGNvbnN0IHsKLSAgICBSZWdpb24gcmVzdWx0OwotICAgIHJlc3VsdC5tUmVnaW9uLm9wKG1SZWdpb24sIHJocy5tUmVnaW9uLCBTa1JlZ2lvbjo6a1VuaW9uX09wKTsKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi1SZWdpb24gUmVnaW9uOjppbnRlcnNlY3QoY29uc3QgUmVnaW9uJiByaHMpIGNvbnN0IHsKLSAgICBSZWdpb24gcmVzdWx0OwotICAgIHJlc3VsdC5tUmVnaW9uLm9wKG1SZWdpb24sIHJocy5tUmVnaW9uLCBTa1JlZ2lvbjo6a0ludGVyc2VjdF9PcCk7Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotUmVnaW9uIFJlZ2lvbjo6c3VidHJhY3QoY29uc3QgUmVnaW9uJiByaHMpIGNvbnN0IHsKLSAgICBSZWdpb24gcmVzdWx0OwotICAgIHJlc3VsdC5tUmVnaW9uLm9wKG1SZWdpb24sIHJocy5tUmVnaW9uLCBTa1JlZ2lvbjo6a0RpZmZlcmVuY2VfT3ApOwotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLVJlZ2lvbiBSZWdpb246OnRyYW5zbGF0ZShpbnQgeCwgaW50IHkpIGNvbnN0IHsKLSAgICBSZWdpb24gcmVzdWx0OwotICAgIG1SZWdpb24udHJhbnNsYXRlKHgsIHksICZyZXN1bHQubVJlZ2lvbik7Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1SZWdpb24mIFJlZ2lvbjo6b3JTZWxmKGNvbnN0IFJlZ2lvbiYgcmhzLCBpbnQgZHgsIGludCBkeSkgewotICAgIFNrUmVnaW9uIHIocmhzLm1SZWdpb24pOwotICAgIHIudHJhbnNsYXRlKGR4LCBkeSk7Ci0gICAgbVJlZ2lvbi5vcChyLCBTa1JlZ2lvbjo6a1VuaW9uX09wKTsKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLVJlZ2lvbiYgUmVnaW9uOjphbmRTZWxmKGNvbnN0IFJlZ2lvbiYgcmhzLCBpbnQgZHgsIGludCBkeSkgewotICAgIFNrUmVnaW9uIHIocmhzLm1SZWdpb24pOwotICAgIHIudHJhbnNsYXRlKGR4LCBkeSk7Ci0gICAgbVJlZ2lvbi5vcChyLCBTa1JlZ2lvbjo6a0ludGVyc2VjdF9PcCk7Ci0gICAgcmV0dXJuICp0aGlzOwotfQotCi1SZWdpb24mIFJlZ2lvbjo6c3VidHJhY3RTZWxmKGNvbnN0IFJlZ2lvbiYgcmhzLCBpbnQgZHgsIGludCBkeSkgewotICAgIFNrUmVnaW9uIHIocmhzLm1SZWdpb24pOwotICAgIHIudHJhbnNsYXRlKGR4LCBkeSk7Ci0gICAgbVJlZ2lvbi5vcChyLCBTa1JlZ2lvbjo6a0RpZmZlcmVuY2VfT3ApOwotICAgIHJldHVybiAqdGhpczsKLX0KLQotUmVnaW9uIFJlZ2lvbjo6bWVyZ2UoY29uc3QgUmVnaW9uJiByaHMsIGludCBkeCwgaW50IGR5KSBjb25zdCB7Ci0gICAgUmVnaW9uIHJlc3VsdDsKLSAgICBTa1JlZ2lvbiByKHJocy5tUmVnaW9uKTsKLSAgICByLnRyYW5zbGF0ZShkeCwgZHkpOwotICAgIHJlc3VsdC5tUmVnaW9uLm9wKG1SZWdpb24sIHIsIFNrUmVnaW9uOjprVW5pb25fT3ApOwotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLVJlZ2lvbiBSZWdpb246OmludGVyc2VjdChjb25zdCBSZWdpb24mIHJocywgaW50IGR4LCBpbnQgZHkpIGNvbnN0IHsKLSAgICBSZWdpb24gcmVzdWx0OwotICAgIFNrUmVnaW9uIHIocmhzLm1SZWdpb24pOwotICAgIHIudHJhbnNsYXRlKGR4LCBkeSk7Ci0gICAgcmVzdWx0Lm1SZWdpb24ub3AobVJlZ2lvbiwgciwgU2tSZWdpb246OmtJbnRlcnNlY3RfT3ApOwotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLVJlZ2lvbiBSZWdpb246OnN1YnRyYWN0KGNvbnN0IFJlZ2lvbiYgcmhzLCBpbnQgZHgsIGludCBkeSkgY29uc3QgewotICAgIFJlZ2lvbiByZXN1bHQ7Ci0gICAgU2tSZWdpb24gcihyaHMubVJlZ2lvbik7Ci0gICAgci50cmFuc2xhdGUoZHgsIGR5KTsKLSAgICByZXN1bHQubVJlZ2lvbi5vcChtUmVnaW9uLCByLCBTa1JlZ2lvbjo6a0RpZmZlcmVuY2VfT3ApOwotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotUmVnaW9uOjppdGVyYXRvcjo6aXRlcmF0b3IoY29uc3QgUmVnaW9uJiByKQotICAgIDogbUl0KHIubVJlZ2lvbikKLXsKLX0KLQotaW50IFJlZ2lvbjo6aXRlcmF0b3I6Oml0ZXJhdGUoUmVjdCogcmVjdCkKLXsKLSAgICBpZiAobUl0LmRvbmUoKSkKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgY29uc3QgU2tJUmVjdCYgcihtSXQucmVjdCgpKTsKLSAgICByZWN0LT5sZWZ0ICA9IHIuZkxlZnQ7Ci0gICAgcmVjdC0+dG9wICAgPSByLmZUb3A7Ci0gICAgcmVjdC0+cmlnaHQgPSByLmZSaWdodDsKLSAgICByZWN0LT5ib3R0b209IHIuZkJvdHRvbTsKLSAgICBtSXQubmV4dCgpOwotICAgIHJldHVybiAxOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLS8vIHdlIHdyaXRlIGEgNGJ5dGUgc2l6ZSBhaGVhZCBvZiB0aGUgYWN0dWFsIHJlZ2lvbiwgc28gd2Uga25vdyBob3cgbXVjaCB3ZSdsbCBuZWVkIGZvciByZWFkaW5nCi0KLXN0YXR1c190IFJlZ2lvbjo6d3JpdGUoUGFyY2VsJiBwYXJjZWwpIGNvbnN0Ci17Ci0gICAgaW50MzJfdCBzaXplID0gbVJlZ2lvbi5mbGF0dGVuKE5VTEwpOwotICAgIHBhcmNlbC53cml0ZUludDMyKHNpemUpOwotICAgIG1SZWdpb24uZmxhdHRlbihwYXJjZWwud3JpdGVJbnBsYWNlKHNpemUpKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IFJlZ2lvbjo6cmVhZChjb25zdCBQYXJjZWwmIHBhcmNlbCkKLXsKLSAgICBzaXplX3Qgc2l6ZSA9IHBhcmNlbC5yZWFkSW50MzIoKTsKLSAgICBtUmVnaW9uLnVuZmxhdHRlbihwYXJjZWwucmVhZElucGxhY2Uoc2l6ZSkpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3NpemVfdCBSZWdpb246OndyaXRlKHZvaWQqIGJ1ZmZlciwgc2l6ZV90IHNpemUpIGNvbnN0Ci17Ci0gICAgc2l6ZV90IHNpemVOZWVkZWQgPSBtUmVnaW9uLmZsYXR0ZW4oTlVMTCk7Ci0gICAgaWYgKHNpemVOZWVkZWQgPiBzaXplKSByZXR1cm4gTk9fTUVNT1JZOwotICAgIHJldHVybiBtUmVnaW9uLmZsYXR0ZW4oYnVmZmVyKTsKLX0KLQotc3NpemVfdCBSZWdpb246OnJlYWQoY29uc3Qgdm9pZCogYnVmZmVyKQotewotICAgIHJldHVybiBtUmVnaW9uLnVuZmxhdHRlbihidWZmZXIpOwotfQotCi1zc2l6ZV90IFJlZ2lvbjo6d3JpdGVFbXB0eSh2b2lkKiBidWZmZXIsIHNpemVfdCBzaXplKQotewotICAgIGlmIChzaXplIDwgNCkgcmV0dXJuIE5PX01FTU9SWTsKLSAgICAvLyB0aGlzIG5lZWRzIHRvIHN0YXkgaW4gc3luYyB3aXRoIFNrUmVnaW9uCi0gICAgKnN0YXRpY19jYXN0PGludDMyX3QqPihidWZmZXIpID0gLTE7Ci0gICAgcmV0dXJuIDQ7Ci19Ci0KLWJvb2wgUmVnaW9uOjppc0VtcHR5KHZvaWQqIGJ1ZmZlcikKLXsKLSAgICAvLyB0aGlzIG5lZWRzIHRvIHN0YXkgaW4gc3luYyB3aXRoIFNrUmVnaW9uCi0gICAgcmV0dXJuICpzdGF0aWNfY2FzdDxpbnQzMl90Kj4oYnVmZmVyKSA9PSAtMTsKLX0KLQotc2l6ZV90IFJlZ2lvbjo6cmVjdHMoVmVjdG9yPFJlY3Q+JiByZWN0TGlzdCkgY29uc3QKLXsKLSAgICByZWN0TGlzdC5jbGVhcigpOwotICAgIGlmICghaXNFbXB0eSgpKSB7Ci0gICAgICAgIFNrUmVnaW9uOjpJdGVyYXRvciBpdGVyYXRvcihtUmVnaW9uKTsKLSAgICAgICAgd2hpbGUoICFpdGVyYXRvci5kb25lKCkgKSB7Ci0gICAgICAgICAgICBjb25zdCBTa0lSZWN0JiBpcihpdGVyYXRvci5yZWN0KCkpOwotICAgICAgICAgICAgcmVjdExpc3QucHVzaChSZWN0KGlyLmZMZWZ0LCBpci5mVG9wLCBpci5mUmlnaHQsIGlyLmZCb3R0b20pKTsKLSAgICAgICAgICAgIGl0ZXJhdG9yLm5leHQoKTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gcmVjdExpc3Quc2l6ZSgpOwotfQotCi12b2lkIFJlZ2lvbjo6ZHVtcChTdHJpbmc4JiBvdXQsIGNvbnN0IGNoYXIqIHdoYXQsIHVpbnQzMl90IGZsYWdzKSBjb25zdAotewotICAgICh2b2lkKWZsYWdzOwotICAgIFZlY3RvcjxSZWN0PiByOwotICAgIHJlY3RzKHIpOwotICAgIAotICAgIHNpemVfdCBTSVpFID0gMjU2OwotICAgIGNoYXIgYnVmZmVyW1NJWkVdOwotICAgIAotICAgIHNucHJpbnRmKGJ1ZmZlciwgU0laRSwgIiAgUmVnaW9uICVzICh0aGlzPSVwLCBjb3VudD0lZClcbiIsIHdoYXQsIHRoaXMsIHIuc2l6ZSgpKTsKLSAgICBvdXQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgZm9yIChzaXplX3QgaT0wIDsgaTxyLnNpemUoKSA7IGkrKykgewotICAgICAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICIgICAgWyUzZCwgJTNkLCAlM2QsICUzZF1cbiIsCi0gICAgICAgICAgICByW2ldLmxlZnQsIHJbaV0udG9wLHJbaV0ucmlnaHQscltpXS5ib3R0b20pOwotICAgICAgICBvdXQuYXBwZW5kKGJ1ZmZlcik7Ci0gICAgfQotfQotCi12b2lkIFJlZ2lvbjo6ZHVtcChjb25zdCBjaGFyKiB3aGF0LCB1aW50MzJfdCBmbGFncykgY29uc3QKLXsKLSAgICAodm9pZClmbGFnczsKLSAgICBWZWN0b3I8UmVjdD4gcjsKLSAgICByZWN0cyhyKTsKLSAgICBMT0dEKCIgIFJlZ2lvbiAlcyAodGhpcz0lcCwgY291bnQ9JWQpXG4iLCB3aGF0LCB0aGlzLCByLnNpemUoKSk7Ci0gICAgZm9yIChzaXplX3QgaT0wIDsgaTxyLnNpemUoKSA7IGkrKykgewotICAgICAgICBMT0dEKCIgICAgWyUzZCwgJTNkLCAlM2QsICUzZF1cbiIsCi0gICAgICAgICAgICByW2ldLmxlZnQsIHJbaV0udG9wLHJbaV0ucmlnaHQscltpXS5ib3R0b20pOwotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91aS9TdXJmYWNlLmNwcCBiL2xpYnMvdWkvU3VyZmFjZS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDRlYTlhZTIuLjAwMDAwMDAKLS0tIGEvbGlicy91aS9TdXJmYWNlLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDI1NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJTdXJmYWNlIgotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8ZmNudGwuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHN5cy9zdGF0Lmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSVBDVGhyZWFkU3RhdGUuaD4KLSNpbmNsdWRlIDx1dGlscy9JTWVtb3J5Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlIDx1aS9JU3VyZmFjZS5oPgotI2luY2x1ZGUgPHVpL1N1cmZhY2UuaD4KLSNpbmNsdWRlIDx1aS9TdXJmYWNlQ29tcG9zZXJDbGllbnQuaD4KLSNpbmNsdWRlIDx1aS9SZWN0Lmg+Ci0KLSNpbmNsdWRlIDxwcml2YXRlL3VpL1NoYXJlZFN0YXRlLmg+Ci0jaW5jbHVkZSA8cHJpdmF0ZS91aS9MYXllclN0YXRlLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLVN1cmZhY2U6OlN1cmZhY2UoY29uc3Qgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiYgY2xpZW50LCAKLSAgICAgICAgY29uc3Qgc3A8SVN1cmZhY2U+JiBzdXJmYWNlLAotICAgICAgICBjb25zdCBJU3VyZmFjZUZsaW5nZXJDbGllbnQ6OnN1cmZhY2VfZGF0YV90JiBkYXRhLAotICAgICAgICB1aW50MzJfdCB3LCB1aW50MzJfdCBoLCBQaXhlbEZvcm1hdCBmb3JtYXQsIHVpbnQzMl90IGZsYWdzLAotICAgICAgICBib29sIG93bmVyKQotICAgIDogbUNsaWVudChjbGllbnQpLCBtU3VyZmFjZShzdXJmYWNlKSwKLSAgICAgIG1Ub2tlbihkYXRhLnRva2VuKSwgbUlkZW50aXR5KGRhdGEuaWRlbnRpdHkpLAotICAgICAgbUZvcm1hdChmb3JtYXQpLCBtRmxhZ3MoZmxhZ3MpLCBtT3duZXIob3duZXIpCi17Ci0gICAgbVN3YXBSZWN0YW5nbGUubWFrZUludmFsaWQoKTsKLSAgICBtU3VyZmFjZUhlYXBCYXNlWzBdID0gMDsKLSAgICBtU3VyZmFjZUhlYXBCYXNlWzFdID0gMDsKLSAgICBtSGVhcFswXSA9IGRhdGEuaGVhcFswXTsgCi0gICAgbUhlYXBbMV0gPSBkYXRhLmhlYXBbMV07Ci19Ci0KLVN1cmZhY2U6OlN1cmZhY2UoU3VyZmFjZSBjb25zdCogcmhzKQotICAgIDogbU93bmVyKGZhbHNlKQotewotICAgIG1Ub2tlbiAgID0gcmhzLT5tVG9rZW47Ci0gICAgbUlkZW50aXR5PSByaHMtPm1JZGVudGl0eTsKLSAgICBtQ2xpZW50ICA9IHJocy0+bUNsaWVudDsKLSAgICBtU3VyZmFjZSA9IHJocy0+bVN1cmZhY2U7Ci0gICAgbUhlYXBbMF0gPSByaHMtPm1IZWFwWzBdOwotICAgIG1IZWFwWzFdID0gcmhzLT5tSGVhcFsxXTsKLSAgICBtRm9ybWF0ICA9IHJocy0+bUZvcm1hdDsKLSAgICBtRmxhZ3MgICA9IHJocy0+bUZsYWdzOwotICAgIG1TdXJmYWNlSGVhcEJhc2VbMF0gPSByaHMtPm1TdXJmYWNlSGVhcEJhc2VbMF07Ci0gICAgbVN1cmZhY2VIZWFwQmFzZVsxXSA9IHJocy0+bVN1cmZhY2VIZWFwQmFzZVsxXTsKLSAgICBtU3dhcFJlY3RhbmdsZS5tYWtlSW52YWxpZCgpOwotfQotCi1TdXJmYWNlOjp+U3VyZmFjZSgpCi17Ci0gICAgaWYgKG1Pd25lciAmJiBtVG9rZW4+PTAgJiYgbUNsaWVudCE9MCkgewotICAgICAgICBtQ2xpZW50LT5kZXN0cm95U3VyZmFjZShtVG9rZW4pOwotICAgIH0KLSAgICBtQ2xpZW50LmNsZWFyKCk7Ci0gICAgbVN1cmZhY2UuY2xlYXIoKTsKLSAgICBtSGVhcFswXS5jbGVhcigpOwotICAgIG1IZWFwWzFdLmNsZWFyKCk7Ci0gICAgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+Zmx1c2hDb21tYW5kcygpOwotfQotCi1zcDxTdXJmYWNlPiBTdXJmYWNlOjpkdXAoKSBjb25zdAotewotICAgIFN1cmZhY2UgY29uc3QgKiByID0gdGhpczsKLSAgICBpZiAodGhpcyAmJiBtT3duZXIpIHsKLSAgICAgICAgLy8gdGhlIG9ubHkgcmVhc29uIHdlIG5lZWQgdG8gZG8gdGhpcyBpcyBiZWNhdXNlIG9mIEphdmEncyBnYXJiYWdlCi0gICAgICAgIC8vIGNvbGxlY3RvcjogYmVjYXVzZSB3ZSdyZSBjcmVhdGluZyBhIGNvcHkgb2YgdGhlIFN1cmZhY2UKLSAgICAgICAgLy8gaW5zdGVhZCBvZiBhIHJlZmVyZW5jZSwgd2UgY2FuIGdhcmFudGVlIHRoYXQgd2hlbiBvdXIgbGFzdAotICAgICAgICAvLyByZWZlcmVuY2UgZ29lcyBhd2F5LCB0aGUgcmVhbCBzdXJmYWNlIHdpbGwgYmUgZGVsZXRlZC4KLSAgICAgICAgLy8gV2l0aG91dCB0aGlzIGhhY2sgKHRoZSBjb2RlIGlzIGNvcnJlY3QgdG9vKSwgd2UnZCBoYXZlIHRvCi0gICAgICAgIC8vIHdhaXQgZm9yIGEgR0MgZm9yIHRoZSBzdXJmYWNlIHRvIGdvIGF3YXkuCi0gICAgICAgIHIgPSBuZXcgU3VyZmFjZSh0aGlzKTsgICAgICAgIAotICAgIH0KLSAgICByZXR1cm4gY29uc3RfY2FzdDxTdXJmYWNlKj4ocik7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2U6Om5leHRCdWZmZXIoU3VyZmFjZUluZm8qIGluZm8pIHsKLSAgICByZXR1cm4gbUNsaWVudC0+bmV4dEJ1ZmZlcih0aGlzLCBpbmZvKTsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZTo6bG9jayhTdXJmYWNlSW5mbyogaW5mbywgYm9vbCBibG9ja2luZykgewotICAgIHJldHVybiBTdXJmYWNlOjpsb2NrKGluZm8sIE5VTEwsIGJsb2NraW5nKTsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZTo6bG9jayhTdXJmYWNlSW5mbyogaW5mbywgUmVnaW9uKiBkaXJ0eSwgYm9vbCBibG9ja2luZykgewotICAgIGlmIChoZWFwQmFzZSgwKSA9PSAwKSByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgaWYgKGhlYXBCYXNlKDEpID09IDApIHJldHVybiBJTlZBTElEX09QRVJBVElPTjsKLSAgICByZXR1cm4gbUNsaWVudC0+bG9ja1N1cmZhY2UodGhpcywgaW5mbywgZGlydHksIGJsb2NraW5nKTsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZTo6dW5sb2NrQW5kUG9zdCgpIHsKLSAgICBpZiAoaGVhcEJhc2UoMCkgPT0gMCkgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOwotICAgIGlmIChoZWFwQmFzZSgxKSA9PSAwKSByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgcmV0dXJuIG1DbGllbnQtPnVubG9ja0FuZFBvc3RTdXJmYWNlKHRoaXMpOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlOjp1bmxvY2soKSB7Ci0gICAgaWYgKGhlYXBCYXNlKDApID09IDApIHJldHVybiBJTlZBTElEX09QRVJBVElPTjsKLSAgICBpZiAoaGVhcEJhc2UoMSkgPT0gMCkgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOwotICAgIHJldHVybiBtQ2xpZW50LT51bmxvY2tTdXJmYWNlKHRoaXMpOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlOjpzZXRMYXllcihpbnQzMl90IGxheWVyKSB7Ci0gICAgcmV0dXJuIG1DbGllbnQtPnNldExheWVyKHRoaXMsIGxheWVyKTsKLX0KLXN0YXR1c190IFN1cmZhY2U6OnNldFBvc2l0aW9uKGludDMyX3QgeCwgaW50MzJfdCB5KSB7Ci0gICAgcmV0dXJuIG1DbGllbnQtPnNldFBvc2l0aW9uKHRoaXMsIHgsIHkpOwotfQotc3RhdHVzX3QgU3VyZmFjZTo6c2V0U2l6ZSh1aW50MzJfdCB3LCB1aW50MzJfdCBoKSB7Ci0gICAgcmV0dXJuIG1DbGllbnQtPnNldFNpemUodGhpcywgdywgaCk7Ci19Ci1zdGF0dXNfdCBTdXJmYWNlOjpoaWRlKCkgewotICAgIHJldHVybiBtQ2xpZW50LT5oaWRlKHRoaXMpOwotfQotc3RhdHVzX3QgU3VyZmFjZTo6c2hvdyhpbnQzMl90IGxheWVyKSB7Ci0gICAgcmV0dXJuIG1DbGllbnQtPnNob3codGhpcywgbGF5ZXIpOwotfQotc3RhdHVzX3QgU3VyZmFjZTo6ZnJlZXplKCkgewotICAgIHJldHVybiBtQ2xpZW50LT5mcmVlemUodGhpcyk7Ci19Ci1zdGF0dXNfdCBTdXJmYWNlOjp1bmZyZWV6ZSgpIHsKLSAgICByZXR1cm4gbUNsaWVudC0+dW5mcmVlemUodGhpcyk7Ci19Ci1zdGF0dXNfdCBTdXJmYWNlOjpzZXRGbGFncyh1aW50MzJfdCBmbGFncywgdWludDMyX3QgbWFzaykgewotICAgIHJldHVybiBtQ2xpZW50LT5zZXRGbGFncyh0aGlzLCBmbGFncywgbWFzayk7Ci19Ci1zdGF0dXNfdCBTdXJmYWNlOjpzZXRUcmFuc3BhcmVudFJlZ2lvbkhpbnQoY29uc3QgUmVnaW9uJiB0cmFuc3BhcmVudCkgewotICAgIHJldHVybiBtQ2xpZW50LT5zZXRUcmFuc3BhcmVudFJlZ2lvbkhpbnQodGhpcywgdHJhbnNwYXJlbnQpOwotfQotc3RhdHVzX3QgU3VyZmFjZTo6c2V0QWxwaGEoZmxvYXQgYWxwaGEpIHsKLSAgICByZXR1cm4gbUNsaWVudC0+c2V0QWxwaGEodGhpcywgYWxwaGEpOwotfQotc3RhdHVzX3QgU3VyZmFjZTo6c2V0TWF0cml4KGZsb2F0IGRzZHgsIGZsb2F0IGR0ZHgsIGZsb2F0IGRzZHksIGZsb2F0IGR0ZHkpIHsKLSAgICByZXR1cm4gbUNsaWVudC0+c2V0TWF0cml4KHRoaXMsIGRzZHgsIGR0ZHgsIGRzZHksIGR0ZHkpOwotfQotc3RhdHVzX3QgU3VyZmFjZTo6c2V0RnJlZXplVGludCh1aW50MzJfdCB0aW50KSB7Ci0gICAgcmV0dXJuIG1DbGllbnQtPnNldEZyZWV6ZVRpbnQodGhpcywgdGludCk7Ci19Ci0KLVJlZ2lvbiBTdXJmYWNlOjpkaXJ0eVJlZ2lvbigpIGNvbnN0ICB7Ci0gICAgcmV0dXJuIG1EaXJ0eVJlZ2lvbjsgCi19Ci12b2lkIFN1cmZhY2U6OnNldERpcnR5UmVnaW9uKGNvbnN0IFJlZ2lvbiYgcmVnaW9uKSBjb25zdCB7Ci0gICAgbURpcnR5UmVnaW9uID0gcmVnaW9uOwotfQotY29uc3QgUmVjdCYgU3VyZmFjZTo6c3dhcFJlY3RhbmdsZSgpIGNvbnN0IHsKLSAgICByZXR1cm4gbVN3YXBSZWN0YW5nbGU7Ci19Ci12b2lkIFN1cmZhY2U6OnNldFN3YXBSZWN0YW5nbGUoY29uc3QgUmVjdCYgcikgewotICAgIG1Td2FwUmVjdGFuZ2xlID0gcjsKLX0KLQotc3A8U3VyZmFjZT4gU3VyZmFjZTo6cmVhZEZyb21QYXJjZWwoUGFyY2VsKiBwYXJjZWwpCi17Ci0gICAgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiBjbGllbnQ7Ci0gICAgSVN1cmZhY2VGbGluZ2VyQ2xpZW50OjpzdXJmYWNlX2RhdGFfdCBkYXRhOwotICAgIHNwPElCaW5kZXI+IGNsaWVudEJpbmRlcj0gcGFyY2VsLT5yZWFkU3Ryb25nQmluZGVyKCk7Ci0gICAgc3A8SVN1cmZhY2U+IHN1cmZhY2UgICAgPSBpbnRlcmZhY2VfY2FzdDxJU3VyZmFjZT4ocGFyY2VsLT5yZWFkU3Ryb25nQmluZGVyKCkpOwotICAgIGRhdGEuaGVhcFswXSAgICAgICAgICAgID0gaW50ZXJmYWNlX2Nhc3Q8SU1lbW9yeUhlYXA+KHBhcmNlbC0+cmVhZFN0cm9uZ0JpbmRlcigpKTsKLSAgICBkYXRhLmhlYXBbMV0gICAgICAgICAgICA9IGludGVyZmFjZV9jYXN0PElNZW1vcnlIZWFwPihwYXJjZWwtPnJlYWRTdHJvbmdCaW5kZXIoKSk7Ci0gICAgZGF0YS50b2tlbiAgICAgICAgICAgICAgPSBwYXJjZWwtPnJlYWRJbnQzMigpOwotICAgIGRhdGEuaWRlbnRpdHkgICAgICAgICAgID0gcGFyY2VsLT5yZWFkSW50MzIoKTsKLSAgICBQaXhlbEZvcm1hdCBmb3JtYXQgICAgICA9IHBhcmNlbC0+cmVhZEludDMyKCk7Ci0gICAgdWludDMyX3QgZmxhZ3MgICAgICAgICAgPSBwYXJjZWwtPnJlYWRJbnQzMigpOwotCi0gICAgaWYgKGNsaWVudEJpbmRlciAhPSBOVUxMKQotICAgICAgICBjbGllbnQgPSBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OmNsaWVudEZvckNvbm5lY3Rpb24oY2xpZW50QmluZGVyKTsKLQotICAgIHJldHVybiBuZXcgU3VyZmFjZShjbGllbnQsIHN1cmZhY2UsIGRhdGEsIDAsIDAsIGZvcm1hdCwgZmxhZ3MsIGZhbHNlKTsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZTo6d3JpdGVUb1BhcmNlbChjb25zdCBzcDxTdXJmYWNlPiYgc3VyZmFjZSwgUGFyY2VsKiBwYXJjZWwpCi17Ci0gICAgdWludDMyX3QgZmxhZ3M9MDsKLSAgICB1aW50MzJfdCBmb3JtYXQ9MDsKLSAgICBTdXJmYWNlSUQgdG9rZW4gPSAtMTsKLSAgICB1aW50MzJfdCBpZGVudGl0eSA9IDA7Ci0gICAgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiBjbGllbnQ7Ci0gICAgc3A8SVN1cmZhY2U+IHN1cjsKLSAgICBzcDxJTWVtb3J5SGVhcD4gaGVhcFsyXTsKLSAgICBpZiAoc3VyZmFjZS0+aXNWYWxpZCgpKSB7Ci0gICAgICAgIHRva2VuID0gc3VyZmFjZS0+bVRva2VuOwotICAgICAgICBpZGVudGl0eSA9IHN1cmZhY2UtPm1JZGVudGl0eTsKLSAgICAgICAgY2xpZW50ID0gc3VyZmFjZS0+bUNsaWVudDsKLSAgICAgICAgc3VyID0gc3VyZmFjZS0+bVN1cmZhY2U7Ci0gICAgICAgIGhlYXBbMF0gPSBzdXJmYWNlLT5tSGVhcFswXTsKLSAgICAgICAgaGVhcFsxXSA9IHN1cmZhY2UtPm1IZWFwWzFdOwotICAgICAgICBmb3JtYXQgPSBzdXJmYWNlLT5tRm9ybWF0OwotICAgICAgICBmbGFncyA9IHN1cmZhY2UtPm1GbGFnczsKLSAgICB9Ci0gICAgcGFyY2VsLT53cml0ZVN0cm9uZ0JpbmRlcihjbGllbnQhPTAgID8gY2xpZW50LT5jb25uZWN0aW9uKCkgOiBOVUxMKTsKLSAgICBwYXJjZWwtPndyaXRlU3Ryb25nQmluZGVyKHN1ciE9MCAgICAgPyBzdXItPmFzQmluZGVyKCkgICAgICA6IE5VTEwpOwotICAgIHBhcmNlbC0+d3JpdGVTdHJvbmdCaW5kZXIoaGVhcFswXSE9MCA/IGhlYXBbMF0tPmFzQmluZGVyKCkgIDogTlVMTCk7Ci0gICAgcGFyY2VsLT53cml0ZVN0cm9uZ0JpbmRlcihoZWFwWzFdIT0wID8gaGVhcFsxXS0+YXNCaW5kZXIoKSAgOiBOVUxMKTsKLSAgICBwYXJjZWwtPndyaXRlSW50MzIodG9rZW4pOwotICAgIHBhcmNlbC0+d3JpdGVJbnQzMihpZGVudGl0eSk7Ci0gICAgcGFyY2VsLT53cml0ZUludDMyKGZvcm1hdCk7Ci0gICAgcGFyY2VsLT53cml0ZUludDMyKGZsYWdzKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLWJvb2wgU3VyZmFjZTo6aXNTYW1lU3VyZmFjZShjb25zdCBzcDxTdXJmYWNlPiYgbGhzLCBjb25zdCBzcDxTdXJmYWNlPiYgcmhzKSAKLXsKLSAgICBpZiAobGhzID09IDAgfHwgcmhzID09IDApCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICByZXR1cm4gbGhzLT5tU3VyZmFjZS0+YXNCaW5kZXIoKSA9PSByaHMtPm1TdXJmYWNlLT5hc0JpbmRlcigpOwotfQotCi12b2lkKiBTdXJmYWNlOjpoZWFwQmFzZShpbnQgaSkgY29uc3QgCi17Ci0gICAgdm9pZCogaGVhcEJhc2UgPSBtU3VyZmFjZUhlYXBCYXNlW2ldOwotICAgIC8vIG1hcCBsYXppbHkgc28gaXQgZG9lc24ndCBnZXQgbWFwcGVkIGluIGNsaWVudHMgdGhhdCBkb24ndCBuZWVkIGl0Ci0gICAgaWYgKGhlYXBCYXNlID09IDApIHsKLSAgICAgICAgY29uc3Qgc3A8SU1lbW9yeUhlYXA+JiBoZWFwKG1IZWFwW2ldKTsKLSAgICAgICAgaWYgKGhlYXAgIT0gMCkgewotICAgICAgICAgICAgaGVhcEJhc2UgPSBzdGF0aWNfY2FzdDx1aW50OF90Kj4oaGVhcC0+YmFzZSgpKTsKLSAgICAgICAgICAgIGlmIChoZWFwQmFzZSA9PSBNQVBfRkFJTEVEKSB7Ci0gICAgICAgICAgICAgICAgaGVhcEJhc2UgPSBOVUxMOwotICAgICAgICAgICAgICAgIExPR0UoIkNvdWxkbid0IG1hcCBTdXJmYWNlJ3MgaGVhcCAoYmluZGVyPSVwLCBoZWFwPSVwKSIsCi0gICAgICAgICAgICAgICAgICAgICAgICBoZWFwLT5hc0JpbmRlcigpLmdldCgpLCBoZWFwLmdldCgpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1TdXJmYWNlSGVhcEJhc2VbaV0gPSBoZWFwQmFzZTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gaGVhcEJhc2U7Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KZGlmZiAtLWdpdCBhL2xpYnMvdWkvU3VyZmFjZUNvbXBvc2VyQ2xpZW50LmNwcCBiL2xpYnMvdWkvU3VyZmFjZUNvbXBvc2VyQ2xpZW50LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTM1NGE3YS4uMDAwMDAwMAotLS0gYS9saWJzL3VpL1N1cmZhY2VDb21wb3NlckNsaWVudC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxMDI2ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIlN1cmZhY2VDb21wb3NlckNsaWVudCIKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHVuaXN0ZC5oPgotI2luY2x1ZGUgPGZjbnRsLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDxzeXMvc3RhdC5oPgotCi0jaW5jbHVkZSA8Y3V0aWxzL21lbW9yeS5oPgotCi0jaW5jbHVkZSA8dXRpbHMvQXRvbWljLmg+Ci0jaW5jbHVkZSA8dXRpbHMvRXJyb3JzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotI2luY2x1ZGUgPHV0aWxzL0tleWVkVmVjdG9yLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSVBDVGhyZWFkU3RhdGUuaD4KLSNpbmNsdWRlIDx1dGlscy9JU2VydmljZU1hbmFnZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9JTWVtb3J5Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlIDx1aS9JU3VyZmFjZUNvbXBvc2VyLmg+Ci0jaW5jbHVkZSA8dWkvSVN1cmZhY2VGbGluZ2VyQ2xpZW50Lmg+Ci0jaW5jbHVkZSA8dWkvSVN1cmZhY2UuaD4KLSNpbmNsdWRlIDx1aS9TdXJmYWNlQ29tcG9zZXJDbGllbnQuaD4KLSNpbmNsdWRlIDx1aS9EaXNwbGF5SW5mby5oPgotI2luY2x1ZGUgPHVpL1JlY3QuaD4KLSNpbmNsdWRlIDx1aS9Qb2ludC5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS91aS9TaGFyZWRTdGF0ZS5oPgotI2luY2x1ZGUgPHByaXZhdGUvdWkvTGF5ZXJTdGF0ZS5oPgotI2luY2x1ZGUgPHByaXZhdGUvdWkvU3VyZmFjZUZsaW5nZXJTeW5jaHJvLmg+Ci0KLSNpbmNsdWRlIDxwaXhlbGZsaW5nZXIvcGl4ZWxmbGluZ2VyLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9CcEJpbmRlci5oPgotCi0jZGVmaW5lIFZFUkJPU0UoLi4uKQkoKHZvaWQpMCkKLS8vI2RlZmluZSBWRVJCT1NFCQkJTE9HRAotCi0jZGVmaW5lIExJS0VMWSggZXhwICkgICAgICAgKF9fYnVpbHRpbl9leHBlY3QoIChleHApICE9IDAsIHRydWUgICkpCi0jZGVmaW5lIFVOTElLRUxZKCBleHAgKSAgICAgKF9fYnVpbHRpbl9leHBlY3QoIChleHApICE9IDAsIGZhbHNlICkpCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLS8vIE11c3Qgbm90IGJlIGhvbGRpbmcgU3VyZmFjZUNvbXBvc2VyQ2xpZW50OjptTG9jayB3aGVuIGFjcXVpcmluZyBnTG9jayBoZXJlLgotc3RhdGljIE11dGV4ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ0xvY2s7Ci1zdGF0aWMgc3A8SVN1cmZhY2VDb21wb3Nlcj4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnU3VyZmFjZU1hbmFnZXI7Ci1zdGF0aWMgRGVmYXVsdEtleWVkVmVjdG9yPCBzcDxJQmluZGVyPiwgc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiA+IGdBY3RpdmVDb25uZWN0aW9uczsKLXN0YXRpYyBTb3J0ZWRWZWN0b3I8c3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PiA+ICAgICAgICAgICAgIGdPcGVuVHJhbnNhY3Rpb25zOwotc3RhdGljIHNwPElNZW1vcnk+ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ1NlcnZlckNibGtNZW1vcnk7Ci1zdGF0aWMgdm9sYXRpbGUgc3VyZmFjZV9mbGluZ2VyX2NibGtfdCogICAgICAgICAgICAgICAgICAgICBnU2VydmVyQ2JsazsKLQotY29uc3Qgc3A8SVN1cmZhY2VDb21wb3Nlcj4mIF9nZXRfc3VyZmFjZV9tYW5hZ2VyKCkKLXsKLSAgICBpZiAoZ1N1cmZhY2VNYW5hZ2VyICE9IDApIHsKLSAgICAgICAgcmV0dXJuIGdTdXJmYWNlTWFuYWdlcjsKLSAgICB9Ci0KLSAgICBzcDxJQmluZGVyPiBiaW5kZXI7Ci0gICAgc3A8SVNlcnZpY2VNYW5hZ2VyPiBzbSA9IGRlZmF1bHRTZXJ2aWNlTWFuYWdlcigpOwotICAgIGRvIHsKLSAgICAgICAgYmluZGVyID0gc20tPmdldFNlcnZpY2UoU3RyaW5nMTYoIlN1cmZhY2VGbGluZ2VyIikpOwotICAgICAgICBpZiAoYmluZGVyID09IDApIHsKLSAgICAgICAgICAgIExPR1coIlN1cmZhY2VGbGluZ2VyIG5vdCBwdWJsaXNoZWQsIHdhaXRpbmcuLi4iKTsKLSAgICAgICAgICAgIHVzbGVlcCg1MDAwMDApOyAvLyAwLjUgcwotICAgICAgICB9Ci0gICAgfSB3aGlsZShiaW5kZXIgPT0gMCk7Ci0gICAgc3A8SVN1cmZhY2VDb21wb3Nlcj4gc2MoaW50ZXJmYWNlX2Nhc3Q8SVN1cmZhY2VDb21wb3Nlcj4oYmluZGVyKSk7Ci0KLSAgICBNdXRleDo6QXV0b2xvY2sgX2woZ0xvY2spOwotICAgIGlmIChnU3VyZmFjZU1hbmFnZXIgPT0gMCkgewotICAgICAgICBnU3VyZmFjZU1hbmFnZXIgPSBzYzsKLSAgICB9Ci0gICAgcmV0dXJuIGdTdXJmYWNlTWFuYWdlcjsKLX0KLQotc3RhdGljIHZvbGF0aWxlIHN1cmZhY2VfZmxpbmdlcl9jYmxrX3QgY29uc3QgKiBnZXRfY2JsaygpCi17Ci0gICAgaWYgKGdTZXJ2ZXJDYmxrID09IDApIHsKLSAgICAgICAgY29uc3Qgc3A8SVN1cmZhY2VDb21wb3Nlcj4mIHNtKF9nZXRfc3VyZmFjZV9tYW5hZ2VyKCkpOwotICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2woZ0xvY2spOwotICAgICAgICBpZiAoZ1NlcnZlckNibGsgPT0gMCkgewotICAgICAgICAgICAgZ1NlcnZlckNibGtNZW1vcnkgPSBzbS0+Z2V0Q2JsaygpOwotICAgICAgICAgICAgTE9HRV9JRihnU2VydmVyQ2Jsa01lbW9yeT09MCwgIkNhbid0IGdldCBzZXJ2ZXIgY29udHJvbCBibG9jayIpOwotICAgICAgICAgICAgZ1NlcnZlckNibGsgPSAoc3VyZmFjZV9mbGluZ2VyX2NibGtfdCAqKWdTZXJ2ZXJDYmxrTWVtb3J5LT5wb2ludGVyKCk7Ci0gICAgICAgICAgICBMT0dFX0lGKGdTZXJ2ZXJDYmxrPT0wLCAiQ2FuJ3QgZ2V0IHNlcnZlciBjb250cm9sIGJsb2NrIGFkZHJlc3MiKTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gZ1NlcnZlckNibGs7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgdm9pZCBjb3B5Qmx0KGNvbnN0IEdHTFN1cmZhY2UmIGRzdCwKLSAgICAgICAgY29uc3QgR0dMU3VyZmFjZSYgc3JjLCBjb25zdCBSZWdpb24mIHJlZykKLXsKLSAgICBSZWdpb246Oml0ZXJhdG9yIGl0ZXJhdG9yKHJlZyk7Ci0gICAgaWYgKGl0ZXJhdG9yKSB7Ci0gICAgICAgIC8vIE5PVEU6IGRzdCBhbmQgc3JjIG11c3QgYmUgdGhlIHNhbWUgZm9ybWF0Ci0gICAgICAgIFJlY3QgcjsKLSAgICAgICAgY29uc3Qgc2l6ZV90IGJwcCA9IGJ5dGVzUGVyUGl4ZWwoc3JjLmZvcm1hdCk7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBkYnByID0gZHN0LnN0cmlkZSAqIGJwcDsKLSAgICAgICAgY29uc3Qgc2l6ZV90IHNicHIgPSBzcmMuc3RyaWRlICogYnBwOwotICAgICAgICB3aGlsZSAoaXRlcmF0b3IuaXRlcmF0ZSgmcikpIHsKLSAgICAgICAgICAgIHNzaXplX3QgaCA9IHIuYm90dG9tIC0gci50b3A7Ci0gICAgICAgICAgICBpZiAoaCkgewotICAgICAgICAgICAgICAgIHNpemVfdCBzaXplID0gKHIucmlnaHQgLSByLmxlZnQpICogYnBwOwotICAgICAgICAgICAgICAgIHVpbnQ4X3QqIHMgPSBzcmMuZGF0YSArIChyLmxlZnQgKyBzcmMuc3RyaWRlICogci50b3ApICogYnBwOwotICAgICAgICAgICAgICAgIHVpbnQ4X3QqIGQgPSBkc3QuZGF0YSArIChyLmxlZnQgKyBkc3Quc3RyaWRlICogci50b3ApICogYnBwOwotICAgICAgICAgICAgICAgIGlmIChkYnByPT1zYnByICYmIHNpemU9PXNicHIpIHsKLSAgICAgICAgICAgICAgICAgICAgc2l6ZSAqPSBoOwotICAgICAgICAgICAgICAgICAgICBoID0gMTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgZG8gewotICAgICAgICAgICAgICAgICAgICBtZW1jcHkoZCwgcywgc2l6ZSk7Ci0gICAgICAgICAgICAgICAgICAgIGQgKz0gZGJwcjsKLSAgICAgICAgICAgICAgICAgICAgcyArPSBzYnByOwotICAgICAgICAgICAgICAgIH0gd2hpbGUgKC0taCA+IDApOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3VyZmFjZV9mbGluZ2VyX2NibGtfdDo6c3VyZmFjZV9mbGluZ2VyX2NibGtfdCgpCi17Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1wZXJfY2xpZW50X2NibGtfdDo6cGVyX2NsaWVudF9jYmxrX3QoKQotewotfQotCi0vLyB0aGVzZSBmdW5jdGlvbnMgYXJlIHVzZWQgYnkgdGhlIGNsaWVudHMKLWlubGluZSBzdGF0dXNfdCBwZXJfY2xpZW50X2NibGtfdDo6dmFsaWRhdGUoc2l6ZV90IGkpIGNvbnN0IHsKLSAgICBpZiAodWludDMyX3QoaSkgPj0gTlVNX0xBWUVSU19NQVgpCi0gICAgICAgIHJldHVybiBCQURfSU5ERVg7Ci0gICAgaWYgKGxheWVyc1tpXS5zd2FwU3RhdGUgJiBlSW52YWxpZFN1cmZhY2UpCi0gICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1pbnQzMl90IHBlcl9jbGllbnRfY2Jsa190Ojpsb2NrX2xheWVyKHNpemVfdCBpLCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBpbnQzMl90IGluZGV4OwotICAgIHVpbnQzMl90IHN0YXRlOwotICAgIGludCB0aW1lb3V0ID0gMDsKLSAgICBzdGF0dXNfdCByZXN1bHQ7Ci0gICAgbGF5ZXJfY2Jsa190ICogY29uc3QgbGF5ZXIgPSBsYXllcnMgKyBpOwotICAgIGNvbnN0IGJvb2wgYmxvY2tpbmcgPSBmbGFncyAmIEJMT0NLSU5HOwotICAgIGNvbnN0IGJvb2wgaW5zcGVjdCAgPSBmbGFncyAmIElOU1BFQ1Q7Ci0KLSAgICBkbyB7Ci0gICAgICAgIHN0YXRlID0gbGF5ZXItPnN3YXBTdGF0ZTsKLQotICAgICAgICBpZiAoVU5MSUtFTFkoKHN0YXRlJihlRmxpcFJlcXVlc3RlZHxlTmV4dEZsaXBQZW5kaW5nKSkgPT0gZU5leHRGbGlwUGVuZGluZykpIHsKLSAgICAgICAgICAgIExPR0UoImVOZXh0RmxpcFBlbmRpbmcgc2V0IGJ1dCBlRmxpcFJlcXVlc3RlZCBub3Qgc2V0LCAiCi0gICAgICAgICAgICAgICAgICJsYXllcj0lZCAobGNibGs9JXApLCBzdGF0ZT0lMDh4IiwKLSAgICAgICAgICAgICAgICAgaW50KGkpLCBsYXllciwgaW50KHN0YXRlKSk7Ci0gICAgICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoVU5MSUtFTFkoc3RhdGUmZUxvY2tlZCkpIHsKLSAgICAgICAgICAgIExPR0UoImVMb2NrZWQgc2V0IHdoZW4gZW50ZXJpbmcgbG9ja19sYXllcigpLCAiCi0gICAgICAgICAgICAgICAgICJsYXllcj0lZCAobGNibGs9JXApLCBzdGF0ZT0lMDh4IiwKLSAgICAgICAgICAgICAgICAgaW50KGkpLCBsYXllciwgaW50KHN0YXRlKSk7Ci0gICAgICAgICAgICByZXR1cm4gV09VTERfQkxPQ0s7Ci0gICAgICAgIH0KLQotCi0JICAgIGlmIChzdGF0ZSAmIChlRmxpcFJlcXVlc3RlZCB8IGVOZXh0RmxpcFBlbmRpbmcgfCBlUmVzaXplUmVxdWVzdGVkCi0gICAgICAgICAgICAgICAgICAgICAgICB8IGVJbnZhbGlkU3VyZmFjZSkpCi0gICAgICAgIHsKLQkgICAgICAgIGludDMyX3QgcmVzaXplSW5kZXg7Ci0JICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2wobG9jayk7Ci0JICAgICAgICAgICAgLy8gbWlnaHQgYmxvY2sgZm9yIGEgdmVyeSBzaG9ydCBhbW91bnQgb2YgdGltZQotCSAgICAgICAgICAgIC8vIHdpbGwgbmV2ZXIgY2F1c2UgdGhlIHNlcnZlciB0byBibG9jayAodHJ5bG9jaygpKQotCi0JICAgICAgICBnb3RvIHN0YXJ0X2xvb3BfaGVyZTsKLQotCSAgICAgICAgLy8gV2UgYmxvY2sgdGhlIGNsaWVudCBpZjoKLQkgICAgICAgIC8vIGVOZXh0RmxpcFBlbmRpbmc6ICB3ZSd2ZSB1c2VkIGJvdGggYnVmZmVycyBhbHJlYWR5LCBzbyB3ZSBuZWVkIHRvCi0JICAgICAgICAvLyAgICAgICAgICAgICAgICAgICAgd2FpdCBmb3Igb25lIHRvIGJlY29tZSBhdmFpbGxhYmxlLgotCSAgICAgICAgLy8gZVJlc2l6ZVJlcXVlc3RlZDogIHRoZSBidWZmZXIgd2UncmUgZ29pbmcgdG8gYWNxdWlyZSBpcyBiZWluZwotCSAgICAgICAgLy8gICAgICAgICAgICAgICAgICAgIHJlc2l6ZWQuIEJsb2NrIHVudGlsIGl0IGlzIGRvbmUuCi0JICAgICAgICAvLyBlRmxpcFJlcXVlc3RlZCAmJiBlQnVzeTogdGhlIGJ1ZmZlciB3ZSdyZSBnb2luZyB0byBhY3F1aXJlIGlzCi0JICAgICAgICAvLyAgICAgICAgICAgICAgICAgICAgY3VycmVudGx5IGluIHVzZSBieSB0aGUgc2VydmVyLgotCSAgICAgICAgLy8gZUludmFsaWRTdXJmYWNlOiAgIHRoaXMgaXMgYSBzcGVjaWFsIGNhc2UsIHdlIGRvbid0IGJsb2NrIGluIHRoaXMKLQkgICAgICAgIC8vICAgICAgICAgICAgICAgICAgICBjYXNlLCB3ZSBqdXN0IHJldHVybiBhbiBlcnJvci4KLQotCSAgICAgICAgd2hpbGUoKHN0YXRlICYgKGVOZXh0RmxpcFBlbmRpbmd8ZUludmFsaWRTdXJmYWNlKSkgfHwKLQkgICAgICAgICAgICAgIChzdGF0ZSAmICgocmVzaXplSW5kZXgpID8gZVJlc2l6ZUJ1ZmZlcjEgOiBlUmVzaXplQnVmZmVyMCkpIHx8Ci0JICAgICAgICAgICAgICAoKHN0YXRlICYgKGVGbGlwUmVxdWVzdGVkfGVCdXN5KSkgPT0gKGVGbGlwUmVxdWVzdGVkfGVCdXN5KSkgKQotCSAgICAgICAgewotCSAgICAgICAgICAgIGlmIChzdGF0ZSAmIGVJbnZhbGlkU3VyZmFjZSkKLQkgICAgICAgICAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLQotCSAgICAgICAgICAgIGlmICghYmxvY2tpbmcpCi0JICAgICAgICAgICAgICAgIHJldHVybiBXT1VMRF9CTE9DSzsKLQotICAgICAgICAgICAgICAgIHRpbWVvdXQgPSAwOwotICAgICAgICAgICAgICAgIHJlc3VsdCA9IGN2LndhaXRSZWxhdGl2ZShsb2NrLCBzZWNvbmRzKDEpKTsKLQkgICAgICAgICAgICBpZiAoX19idWlsdGluX2V4cGVjdChyZXN1bHQhPU5PX0VSUk9SLCBmYWxzZSkpIHsKLSAgICAgICAgICAgICAgICAgICAgY29uc3QgaW50IG5ld1N0YXRlID0gbGF5ZXItPnN3YXBTdGF0ZTsKLSAgICAgICAgICAgICAgICAgICAgTE9HVyggICAibG9ja19sYXllciB0aW1lZCBvdXQgKGlzIHRoZSBDUFUgcGVnZ2VkPykgIgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICJsYXllcj0lZCwgbGNibGs9JXAsIHN0YXRlPSUwOHggKHdhcyAlMDh4KSIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50KGkpLCBsYXllciwgbmV3U3RhdGUsIGludChzdGF0ZSkpOwotICAgICAgICAgICAgICAgICAgICB0aW1lb3V0ID0gbmV3U3RhdGUgIT0gaW50KHN0YXRlKTsKLSAgICAgICAgICAgICAgICB9Ci0KLQkgICAgICAgIHN0YXJ0X2xvb3BfaGVyZToKLQkgICAgICAgICAgICBzdGF0ZSA9IGxheWVyLT5zd2FwU3RhdGU7Ci0JICAgICAgICAgICAgcmVzaXplSW5kZXggPSAoc3RhdGUmZUluZGV4KSBeICgoc3RhdGUmZUZsaXBSZXF1ZXN0ZWQpPj4xKTsKLQkgICAgICAgIH0KLQotICAgICAgICAgICAgTE9HV19JRih0aW1lb3V0LAotICAgICAgICAgICAgICAgICAgICAibG9ja19sYXllcigpIHRpbWVkIG91dCBidXQgZGlkbid0IGFwcGVhciB0byBuZWVkICIKLSAgICAgICAgICAgICAgICAgICAgInRvIGJlIGxvY2tlZCBhbmQgd2UgcmVjb3ZlcmVkICIKLSAgICAgICAgICAgICAgICAgICAgIihsYXllcj0lZCwgbGNibGs9JXAsIHN0YXRlPSUwOHgpIiwKLSAgICAgICAgICAgICAgICAgICAgaW50KGkpLCBsYXllciwgaW50KHN0YXRlKSk7Ci0JICAgIH0KLQotCSAgICAvLyBlRmxpcFJlcXVlc3RlZCBpcyBub3Qgc2V0IGFuZCBjYW5ub3QgYmUgc2V0IGJ5IGFub3RoZXIgdGhyZWFkOiBpdCdzCi0JICAgIC8vIHNhZmUgdG8gdXNlIHRoZSBmaXJzdCBidWZmZXIgd2l0aG91dCBzeW5jaHJvbml6YXRpb24uCi0KLSAgICAgICAgLy8gQ2hvb3NlIHRoZSBpbmRleCBkZXBlbmRpbmcgb24gZUZsaXBSZXF1ZXN0ZWQuCi0gICAgICAgIC8vIFdoZW4gaXQncyBzZXQsIGNob29zZSB0aGUgJ290aGVyJyBidWZmZXIuCi0gICAgICAgIGluZGV4ID0gKHN0YXRlJmVJbmRleCkgXiAoKHN0YXRlJmVGbGlwUmVxdWVzdGVkKT4+MSk7Ci0KLQkgICAgLy8gbWFrZSBzdXJlIHRoaXMgYnVmZmVyIGlzIHZhbGlkCi0JICAgIGlmIChsYXllci0+c3VyZmFjZVtpbmRleF0uYml0c19vZmZzZXQgPCAwKSB7Ci0JICAgICAgICByZXR1cm4gc3RhdHVzX3QobGF5ZXItPnN1cmZhY2VbaW5kZXhdLmJpdHNfb2Zmc2V0KTsKLQkgICAgfQotCi0gICAgICAgIGlmIChpbnNwZWN0KSB7Ci0gICAgICAgICAgICAvLyB3ZSBqdXN0IHdhbnQgdG8gaW5zcGVjdCB0aGlzIGxheWVyLiBkb24ndCBsb2NrIGl0LgotICAgICAgICAgICAgZ290byBkb25lOwotICAgICAgICB9Ci0KLQkgICAgLy8gbGFzdCB0aGluZyBiZWZvcmUgd2UncmUgZG9uZSwgd2UgbmVlZCB0byBhdG9taWNhbGx5IGxvY2sgdGhlIHN0YXRlCi0gICAgfSB3aGlsZSAoYW5kcm9pZF9hdG9taWNfY21weGNoZyhzdGF0ZSwgc3RhdGV8ZUxvY2tlZCwgJihsYXllci0+c3dhcFN0YXRlKSkpOwotCi0gICAgVkVSQk9TRSgibG9ja2VkIGxheWVyPSVkIChsY2Jsaz0lcCksIGJ1ZmZlcj0lZCwgc3RhdGU9MHglMDh4IiwKLSAgICAgICAgIGludChpKSwgbGF5ZXIsIGludChpbmRleCksIGludChzdGF0ZSkpOwotCi0gICAgLy8gc3RvcmUgdGhlIGluZGV4IG9mIHRoZSBsb2NrZWQgYnVmZmVyIChmb3IgY2xpZW50IHVzZSBvbmx5KQotICAgIGxheWVyLT5mbGFncyAmPSB+ZUJ1ZmZlckluZGV4OwotICAgIGxheWVyLT5mbGFncyB8PSAoKGluZGV4IDw8IGVCdWZmZXJJbmRleFNoaWZ0KSAmIGVCdWZmZXJJbmRleCk7Ci0KLWRvbmU6Ci0gICAgcmV0dXJuIGluZGV4OwotfQotCi11aW50MzJfdCBwZXJfY2xpZW50X2NibGtfdDo6dW5sb2NrX2xheWVyX2FuZF9wb3N0KHNpemVfdCBpKQotewotICAgIC8vIGF0b21pY2FsbHkgc2V0IGVGbGlwUmVxdWVzdGVkIGFuZCBjbGVhciBlTG9ja2VkIGFuZCBvcHRpb25uYWx5Ci0gICAgLy8gc2V0IGVOZXh0RmxpcFBlbmRpbmcgaWYgZUZsaXBSZXF1ZXN0ZWQgd2FzIGFscmVhZHkgc2V0Ci0KLSAgICBsYXllcl9jYmxrX3QgKiBjb25zdCBsYXllciA9IGxheWVycyArIGk7Ci0gICAgaW50MzJfdCBvbGR2YWx1ZSwgbmV3dmFsdWU7Ci0gICAgZG8gewotICAgICAgICBvbGR2YWx1ZSA9IGxheWVyLT5zd2FwU3RhdGU7Ci0gICAgICAgICAgICAvLyBnZXQgY3VycmVudCB2YWx1ZQotCi0gICAgICAgIG5ld3ZhbHVlID0gb2xkdmFsdWUgJiB+ZUxvY2tlZDsKLSAgICAgICAgICAgIC8vIGNsZWFyIGVMb2NrZWQKLQotICAgICAgICBuZXd2YWx1ZSB8PSBlRmxpcFJlcXVlc3RlZDsKLSAgICAgICAgICAgIC8vIHNldCBlRmxpcFJlcXVlc3RlZAotCi0gICAgICAgIGlmIChvbGR2YWx1ZSAmIGVGbGlwUmVxdWVzdGVkKQotICAgICAgICAgICAgbmV3dmFsdWUgfD0gZU5leHRGbGlwUGVuZGluZzsKLSAgICAgICAgICAgIC8vIGlmIGVGbGlwUmVxdWVzdGVkIHdhcyBhbHJlYWQgc2V0LCBzZXQgZU5leHRGbGlwUGVuZGluZwotCi0gICAgfSB3aGlsZSAoYW5kcm9pZF9hdG9taWNfY21weGNoZyhvbGR2YWx1ZSwgbmV3dmFsdWUsICYobGF5ZXItPnN3YXBTdGF0ZSkpKTsKLQotICAgIFZFUkJPU0UoInJlcXVlc3QgcGFnZWZsaXAgZm9yIGxheWVyPSVkLCBidWZmZXI9JWQsIHN0YXRlPTB4JTA4eCIsCi0gICAgICAgICAgICBpbnQoaSksIGludCgobGF5ZXItPmZsYWdzICYgZUJ1ZmZlckluZGV4KSA+PiBlQnVmZmVySW5kZXhTaGlmdCksCi0gICAgICAgICAgICBpbnQobmV3dmFsdWUpKTsKLQotICAgIC8vIGZyb20gdGhpcyBwb2ludCwgdGhlIHNlcnZlciBjYW4ga2ljayBpbiBhdCBhbnl0aW1lIGFuZCB1c2UgdGhlIGZpcnN0Ci0gICAgLy8gYnVmZmVyLCBzbyB3ZSBjYW5ub3QgdXNlIGl0IGFueW1vcmUsIGFuZCB3ZSBtdXN0IHVzZSB0aGUgJ290aGVyJwotICAgIC8vIGJ1ZmZlciBpbnN0ZWFkIChvciB3YWl0IGlmIGl0IGlzIG5vdCBhdmFpbGxhYmxlIHlldCwgc2VlIGxvY2tfbGF5ZXIpLgotCi0gICAgcmV0dXJuIG5ld3ZhbHVlOwotfQotCi12b2lkIHBlcl9jbGllbnRfY2Jsa190Ojp1bmxvY2tfbGF5ZXIoc2l6ZV90IGkpCi17Ci0gICAgbGF5ZXJfY2Jsa190ICogY29uc3QgbGF5ZXIgPSBsYXllcnMgKyBpOwotICAgIGFuZHJvaWRfYXRvbWljX2FuZCh+ZUxvY2tlZCwgJmxheWVyLT5zd2FwU3RhdGUpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RhdGljIGlubGluZSBpbnQgY29tcGFyZV90eXBlKCBjb25zdCBsYXllcl9zdGF0ZV90JiBsaHMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGxheWVyX3N0YXRlX3QmIHJocykgewotICAgIGlmIChsaHMuc3VyZmFjZSA8IHJocy5zdXJmYWNlKSAgcmV0dXJuIC0xOwotICAgIGlmIChsaHMuc3VyZmFjZSA+IHJocy5zdXJmYWNlKSAgcmV0dXJuIDE7Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLVN1cmZhY2VDb21wb3NlckNsaWVudDo6U3VyZmFjZUNvbXBvc2VyQ2xpZW50KCkKLXsKLSAgICBjb25zdCBzcDxJU3VyZmFjZUNvbXBvc2VyPiYgc20oX2dldF9zdXJmYWNlX21hbmFnZXIoKSk7Ci0gICAgaWYgKHNtID09IDApIHsKLSAgICAgICAgX2luaXQoMCwgMCk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBfaW5pdChzbSwgc20tPmNyZWF0ZUNvbm5lY3Rpb24oKSk7Ci0KLSAgICBpZiAobUNsaWVudCAhPSAwKSB7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChnTG9jayk7Ci0gICAgICAgIFZFUkJPU0UoIkFkZGluZyBjbGllbnQgJXAgdG8gbWFwIiwgdGhpcyk7Ci0gICAgICAgIGdBY3RpdmVDb25uZWN0aW9ucy5hZGQobUNsaWVudC0+YXNCaW5kZXIoKSwgdGhpcyk7Ci0gICAgfQotfQotCi1TdXJmYWNlQ29tcG9zZXJDbGllbnQ6OlN1cmZhY2VDb21wb3NlckNsaWVudCgKLSAgICAgICAgY29uc3Qgc3A8SVN1cmZhY2VDb21wb3Nlcj4mIHNtLCBjb25zdCBzcDxJQmluZGVyPiYgY29ubikKLXsKLSAgICBfaW5pdChzbSwgaW50ZXJmYWNlX2Nhc3Q8SVN1cmZhY2VGbGluZ2VyQ2xpZW50Pihjb25uKSk7Ci19Ci0KLXZvaWQgU3VyZmFjZUNvbXBvc2VyQ2xpZW50OjpfaW5pdCgKLSAgICAgICAgY29uc3Qgc3A8SVN1cmZhY2VDb21wb3Nlcj4mIHNtLCBjb25zdCBzcDxJU3VyZmFjZUZsaW5nZXJDbGllbnQ+JiBjb25uKQotewotICAgIFZFUkJPU0UoIkNyZWF0aW5nIGNsaWVudCAlcCwgY29ubiAlcCIsIHRoaXMsIGNvbm4uZ2V0KCkpOwotCi0gICAgbVNpZ25hbFNlcnZlciA9IDA7Ci0gICAgbVByZWJ1aWx0TGF5ZXJTdGF0ZSA9IDA7Ci0gICAgbVRyYW5zYWN0aW9uT3BlbiA9IDA7Ci0gICAgbVN0YXR1cyA9IE5PX0VSUk9SOwotICAgIG1Db250cm9sID0gMDsKLQotICAgIG1DbGllbnQgPSBjb25uOwotICAgIGlmIChtQ2xpZW50ID09IDApIHsKLSAgICAgICAgbVN0YXR1cyA9IE5PX0lOSVQ7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBtQ2xpZW50LT5nZXRDb250cm9sQmxvY2tzKCZtQ29udHJvbE1lbW9yeSk7Ci0gICAgbVNpZ25hbFNlcnZlciA9IG5ldyBTdXJmYWNlRmxpbmdlclN5bmNocm8oc20pOwotICAgIG1Db250cm9sID0gc3RhdGljX2Nhc3Q8cGVyX2NsaWVudF9jYmxrX3QgKj4obUNvbnRyb2xNZW1vcnktPnBvaW50ZXIoKSk7Ci19Ci0KLVN1cmZhY2VDb21wb3NlckNsaWVudDo6flN1cmZhY2VDb21wb3NlckNsaWVudCgpCi17Ci0gICAgVkVSQk9TRSgiRGVzdHJveWluZyBjbGllbnQgJXAsIGNvbm4gJXAiLCB0aGlzLCBtQ2xpZW50LmdldCgpKTsKLSAgICBkaXNwb3NlKCk7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VDb21wb3NlckNsaWVudDo6aW5pdENoZWNrKCkgY29uc3QKLXsKLSAgICByZXR1cm4gbVN0YXR1czsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZUNvbXBvc2VyQ2xpZW50Ojp2YWxpZGF0ZVN1cmZhY2UoCi0gICAgICAgIHBlcl9jbGllbnRfY2Jsa190IGNvbnN0KiBjYmxrLCBTdXJmYWNlIGNvbnN0ICogc3VyZmFjZSkKLXsKLSAgICBTdXJmYWNlSUQgaW5kZXggPSBzdXJmYWNlLT5JRCgpOwotICAgIGlmIChjYmxrID09IDApIHsKLSAgICAgICAgTE9HRSgiY2JsayBpcyBudWxsIChzdXJmYWNlIGlkPSVkLCBpZGVudGl0eT0ldSkiLAotICAgICAgICAgICAgICAgIGluZGV4LCBzdXJmYWNlLT5nZXRJZGVudGl0eSgpKTsKLSAgICAgICAgcmV0dXJuIE5PX0lOSVQ7Ci0gICAgfQotCi0gICAgc3RhdHVzX3QgZXJyID0gY2Jsay0+dmFsaWRhdGUoaW5kZXgpOwotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgTE9HRSgic3VyZmFjZSAoaWQ9JWQsIGlkZW50aXR5PSV1KSBpcyBpbnZhbGlkLCBlcnI9JWQgKCVzKSIsCi0gICAgICAgICAgICAgICAgaW5kZXgsIHN1cmZhY2UtPmdldElkZW50aXR5KCksIGVyciwgc3RyZXJyb3IoLWVycikpOwotICAgICAgICByZXR1cm4gZXJyOwotICAgIH0KLQotICAgIGlmIChzdXJmYWNlLT5nZXRJZGVudGl0eSgpICE9IHVpbnQzMl90KGNibGstPmxheWVyc1tpbmRleF0uaWRlbnRpdHkpKSB7Ci0gICAgICAgIExPR0UoInVzaW5nIGFuIGludmFsaWQgc3VyZmFjZSBpZD0lZCwgaWRlbnRpdHk9JXUgc2hvdWxkIGJlICVkIiwKLSAgICAgICAgICAgICAgICBpbmRleCwgc3VyZmFjZS0+Z2V0SWRlbnRpdHkoKSwgY2Jsay0+bGF5ZXJzW2luZGV4XS5pZGVudGl0eSk7Ci0gICAgICAgIHJldHVybiBOT19JTklUOwotICAgIH0KLQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3A8SUJpbmRlcj4gU3VyZmFjZUNvbXBvc2VyQ2xpZW50Ojpjb25uZWN0aW9uKCkgY29uc3QKLXsKLSAgICByZXR1cm4gKG1DbGllbnQgIT0gMCkgPyBtQ2xpZW50LT5hc0JpbmRlcigpIDogMDsKLX0KLQotc3A8U3VyZmFjZUNvbXBvc2VyQ2xpZW50PgotU3VyZmFjZUNvbXBvc2VyQ2xpZW50OjpjbGllbnRGb3JDb25uZWN0aW9uKGNvbnN0IHNwPElCaW5kZXI+JiBjb25uKQotewotICAgIHNwPFN1cmZhY2VDb21wb3NlckNsaWVudD4gY2xpZW50OwotCi0gICAgeyAvLyBzY29wZSBmb3IgbG9jawotICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2woZ0xvY2spOwotICAgICAgICBjbGllbnQgPSBnQWN0aXZlQ29ubmVjdGlvbnMudmFsdWVGb3IoY29ubik7Ci0gICAgfQotCi0gICAgaWYgKGNsaWVudCA9PSAwKSB7Ci0gICAgICAgIC8vIE5lZWQgdG8gbWFrZSBhIG5ldyBjbGllbnQuCi0gICAgICAgIGNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBzbShfZ2V0X3N1cmZhY2VfbWFuYWdlcigpKTsKLSAgICAgICAgY2xpZW50ID0gbmV3IFN1cmZhY2VDb21wb3NlckNsaWVudChzbSwgY29ubik7Ci0gICAgICAgIGlmIChjbGllbnQgIT0gMCAmJiBjbGllbnQtPmluaXRDaGVjaygpID09IE5PX0VSUk9SKSB7Ci0gICAgICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2woZ0xvY2spOwotICAgICAgICAgICAgZ0FjdGl2ZUNvbm5lY3Rpb25zLmFkZChjb25uLCBjbGllbnQpOwotICAgICAgICAgICAgLy9MT0dEKCJ3ZSBoYXZlICVkIGNvbm5lY3Rpb25zIiwgZ0FjdGl2ZUNvbm5lY3Rpb25zLnNpemUoKSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBjbGllbnQuY2xlYXIoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBjbGllbnQ7Ci19Ci0KLXZvaWQgU3VyZmFjZUNvbXBvc2VyQ2xpZW50OjpkaXNwb3NlKCkKLXsKLSAgICAvLyB0aGlzIGNhbiBiZSBjYWxsZWQgbW9yZSB0aGFuIG9uY2UuCi0KLSAgICBzcDxJTWVtb3J5PiAgICAgICAgICAgICAgICAgY29udHJvbE1lbW9yeTsKLSAgICBzcDxJU3VyZmFjZUZsaW5nZXJDbGllbnQ+ICAgY2xpZW50OwotICAgIHNwPElNZW1vcnlIZWFwPiAgICAgICAgICAgICBzdXJmYWNlSGVhcDsKLQotICAgIHsKLSAgICAgICAgTXV0ZXg6OkF1dG9sb2NrIF9sZyhnTG9jayk7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbG0obUxvY2spOwotCi0gICAgICAgIGRlbGV0ZSBtU2lnbmFsU2VydmVyOwotICAgICAgICBtU2lnbmFsU2VydmVyID0gMDsKLQotICAgICAgICBpZiAobUNsaWVudCAhPSAwKSB7Ci0gICAgICAgICAgICBjbGllbnQgPSBtQ2xpZW50OwotICAgICAgICAgICAgbUNsaWVudC5jbGVhcigpOwotCi0gICAgICAgICAgICBzc2l6ZV90IGkgPSBnQWN0aXZlQ29ubmVjdGlvbnMuaW5kZXhPZktleShjbGllbnQtPmFzQmluZGVyKCkpOwotICAgICAgICAgICAgaWYgKGkgPj0gMCAmJiBnQWN0aXZlQ29ubmVjdGlvbnMudmFsdWVBdChpKSA9PSB0aGlzKSB7Ci0gICAgICAgICAgICAgICAgVkVSQk9TRSgiUmVtb3ZpbmcgY2xpZW50ICVwIGZyb20gbWFwIGF0ICVkIiwgdGhpcywgaW50KGkpKTsKLSAgICAgICAgICAgICAgICBnQWN0aXZlQ29ubmVjdGlvbnMucmVtb3ZlSXRlbXNBdChpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGRlbGV0ZSBtUHJlYnVpbHRMYXllclN0YXRlOwotICAgICAgICBtUHJlYnVpbHRMYXllclN0YXRlID0gMDsKLSAgICAgICAgY29udHJvbE1lbW9yeSA9IG1Db250cm9sTWVtb3J5OwotICAgICAgICBzdXJmYWNlSGVhcCA9IG1TdXJmYWNlSGVhcDsKLSAgICAgICAgbUNvbnRyb2xNZW1vcnkuY2xlYXIoKTsKLSAgICAgICAgbVN1cmZhY2VIZWFwLmNsZWFyKCk7Ci0gICAgICAgIG1Db250cm9sID0gMDsKLSAgICAgICAgbVN0YXR1cyA9IE5PX0lOSVQ7Ci0gICAgfQotfQotCi1zdGF0dXNfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OmdldERpc3BsYXlJbmZvKAotICAgICAgICBEaXNwbGF5SUQgZHB5LCBEaXNwbGF5SW5mbyogaW5mbykKLXsKLSAgICBpZiAodWludDMyX3QoZHB5KT49TlVNX0RJU1BMQVlfTUFYKQotICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOwotCi0gICAgdm9sYXRpbGUgc3VyZmFjZV9mbGluZ2VyX2NibGtfdCBjb25zdCAqIGNibGsgPSBnZXRfY2JsaygpOwotICAgIHZvbGF0aWxlIGRpc3BsYXlfY2Jsa190IGNvbnN0ICogZGNibGsgPSBjYmxrLT5kaXNwbGF5cyArIGRweTsKLQotICAgIGluZm8tPncgICAgICAgICAgICAgID0gZGNibGstPnc7Ci0gICAgaW5mby0+aCAgICAgICAgICAgICAgPSBkY2Jsay0+aDsKLSAgICBpbmZvLT5vcmllbnRhdGlvbiAgICA9IGRjYmxrLT5vcmllbnRhdGlvbjsKLSAgICBpbmZvLT54ZHBpICAgICAgICAgICA9IGRjYmxrLT54ZHBpOwotICAgIGluZm8tPnlkcGkgICAgICAgICAgID0gZGNibGstPnlkcGk7Ci0gICAgaW5mby0+ZnBzICAgICAgICAgICAgPSBkY2Jsay0+ZnBzOwotICAgIGluZm8tPmRlbnNpdHkgICAgICAgID0gZGNibGstPmRlbnNpdHk7Ci0gICAgcmV0dXJuIGdldFBpeGVsRm9ybWF0SW5mbyhkY2Jsay0+Zm9ybWF0LCAmKGluZm8tPnBpeGVsRm9ybWF0SW5mbykpOwotfQotCi1zc2l6ZV90IFN1cmZhY2VDb21wb3NlckNsaWVudDo6Z2V0RGlzcGxheVdpZHRoKERpc3BsYXlJRCBkcHkpCi17Ci0gICAgaWYgKHVpbnQzMl90KGRweSk+PU5VTV9ESVNQTEFZX01BWCkKLSAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLSAgICB2b2xhdGlsZSBzdXJmYWNlX2ZsaW5nZXJfY2Jsa190IGNvbnN0ICogY2JsayA9IGdldF9jYmxrKCk7Ci0gICAgdm9sYXRpbGUgZGlzcGxheV9jYmxrX3QgY29uc3QgKiBkY2JsayA9IGNibGstPmRpc3BsYXlzICsgZHB5OwotICAgIHJldHVybiBkY2Jsay0+dzsKLX0KLQotc3NpemVfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OmdldERpc3BsYXlIZWlnaHQoRGlzcGxheUlEIGRweSkKLXsKLSAgICBpZiAodWludDMyX3QoZHB5KT49TlVNX0RJU1BMQVlfTUFYKQotICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOwotICAgIHZvbGF0aWxlIHN1cmZhY2VfZmxpbmdlcl9jYmxrX3QgY29uc3QgKiBjYmxrID0gZ2V0X2NibGsoKTsKLSAgICB2b2xhdGlsZSBkaXNwbGF5X2NibGtfdCBjb25zdCAqIGRjYmxrID0gY2Jsay0+ZGlzcGxheXMgKyBkcHk7Ci0gICAgcmV0dXJuIGRjYmxrLT5oOwotfQotCi1zc2l6ZV90IFN1cmZhY2VDb21wb3NlckNsaWVudDo6Z2V0RGlzcGxheU9yaWVudGF0aW9uKERpc3BsYXlJRCBkcHkpCi17Ci0gICAgaWYgKHVpbnQzMl90KGRweSk+PU5VTV9ESVNQTEFZX01BWCkKLSAgICAgICAgcmV0dXJuIEJBRF9WQUxVRTsKLSAgICB2b2xhdGlsZSBzdXJmYWNlX2ZsaW5nZXJfY2Jsa190IGNvbnN0ICogY2JsayA9IGdldF9jYmxrKCk7Ci0gICAgdm9sYXRpbGUgZGlzcGxheV9jYmxrX3QgY29uc3QgKiBkY2JsayA9IGNibGstPmRpc3BsYXlzICsgZHB5OwotICAgIHJldHVybiBkY2Jsay0+b3JpZW50YXRpb247Ci19Ci0KLXNzaXplX3QgU3VyZmFjZUNvbXBvc2VyQ2xpZW50OjpnZXROdW1iZXJPZkRpc3BsYXlzKCkKLXsKLSAgICB2b2xhdGlsZSBzdXJmYWNlX2ZsaW5nZXJfY2Jsa190IGNvbnN0ICogY2JsayA9IGdldF9jYmxrKCk7Ci0gICAgdWludDMyX3QgY29ubmVjdGVkID0gY2Jsay0+Y29ubmVjdGVkOwotICAgIGludCBuID0gMDsKLSAgICB3aGlsZSAoY29ubmVjdGVkKSB7Ci0gICAgICAgIGlmIChjb25uZWN0ZWQmMSkgbisrOwotICAgICAgICBjb25uZWN0ZWQgPj49IDE7Ci0gICAgfQotICAgIHJldHVybiBuOwotfQotCi1zcDxTdXJmYWNlPiBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OmNyZWF0ZVN1cmZhY2UoCi0gICAgICAgIGludCBwaWQsCi0gICAgICAgIERpc3BsYXlJRCBkaXNwbGF5LAotICAgICAgICB1aW50MzJfdCB3LAotICAgICAgICB1aW50MzJfdCBoLAotICAgICAgICBQaXhlbEZvcm1hdCBmb3JtYXQsCi0gICAgICAgIHVpbnQzMl90IGZsYWdzKQotewotICAgIHNwPFN1cmZhY2U+IHJlc3VsdDsKLSAgICBpZiAobVN0YXR1cyA9PSBOT19FUlJPUikgewotICAgICAgICBJU3VyZmFjZUZsaW5nZXJDbGllbnQ6OnN1cmZhY2VfZGF0YV90IGRhdGE7Ci0gICAgICAgIHNwPElTdXJmYWNlPiBzdXJmYWNlID0gbUNsaWVudC0+Y3JlYXRlU3VyZmFjZSgmZGF0YSwgcGlkLAotICAgICAgICAgICAgICAgIGRpc3BsYXksIHcsIGgsIGZvcm1hdCwgZmxhZ3MpOwotICAgICAgICBpZiAoc3VyZmFjZSAhPSAwKSB7Ci0gICAgICAgICAgICBpZiAodWludDMyX3QoZGF0YS50b2tlbikgPCBOVU1fTEFZRVJTX01BWCkgewotICAgICAgICAgICAgICAgIHJlc3VsdCA9IG5ldyBTdXJmYWNlKHRoaXMsIHN1cmZhY2UsIGRhdGEsIHcsIGgsIGZvcm1hdCwgZmxhZ3MpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VDb21wb3NlckNsaWVudDo6ZGVzdHJveVN1cmZhY2UoU3VyZmFjZUlEIHNpZCkKLXsKLSAgICBpZiAobVN0YXR1cyAhPSBOT19FUlJPUikKLSAgICAgICAgcmV0dXJuIG1TdGF0dXM7Ci0KLSAgICAvLyBpdCdzIG9rYXkgdG8gZGVzdHJveSBhIHN1cmZhY2Ugd2hpbGUgYSB0cmFuc2FjdGlvbiBpcyBvcGVuLAotICAgIC8vICh0cmFuc2FjdGlvbnMgcmVhbGx5IGFyZSBhIGNsaWVudC1zaWRlIGNvbmNlcHQpCi0gICAgLy8gaG93ZXZlciwgdGhpcyBpbmRpY2F0ZXMgcHJvYmFibHkgYSBtaXN1c2Ugb2YgdGhlIEFQSSBvciBhIGJ1ZwotICAgIC8vIGluIHRoZSBjbGllbnQgY29kZS4KLSAgICBMT0dXX0lGKG1UcmFuc2FjdGlvbk9wZW4sCi0gICAgICAgICAiRGVzdHJveWluZyBzdXJmYWNlIHdoaWxlIGEgdHJhbnNhY3Rpb24gaXMgb3Blbi4gIgotICAgICAgICAgIkNsaWVudCAlcDogZGVzdHJveWluZyBzdXJmYWNlICVkLCBtVHJhbnNhY3Rpb25PcGVuPSVkIiwKLSAgICAgICAgIHRoaXMsIHNpZCwgbVRyYW5zYWN0aW9uT3Blbik7Ci0KLSAgICBzdGF0dXNfdCBlcnIgPSBtQ2xpZW50LT5kZXN0cm95U3VyZmFjZShzaWQpOwotICAgIHJldHVybiBlcnI7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VDb21wb3NlckNsaWVudDo6bmV4dEJ1ZmZlcihTdXJmYWNlKiBzdXJmYWNlLAotICAgICAgICAgICAgICAgICAgICAgICAgU3VyZmFjZTo6U3VyZmFjZUluZm8qIGluZm8pCi17Ci0gICAgU3VyZmFjZUlEIGluZGV4ID0gc3VyZmFjZS0+SUQoKTsKLSAgICBwZXJfY2xpZW50X2NibGtfdCogY29uc3QgY2JsayA9IG1Db250cm9sOwotICAgIHN0YXR1c190IGVyciA9IHZhbGlkYXRlU3VyZmFjZShjYmxrLCBzdXJmYWNlKTsKLSAgICBpZiAoZXJyICE9IE5PX0VSUk9SKQotICAgICAgICByZXR1cm4gZXJyOwotCi0gICAgaW50MzJfdCBiYWNrSWR4ID0gc3VyZmFjZS0+bUJhY2tidWZmZXJJbmRleDsKLSAgICBsYXllcl9jYmxrX3QqIGNvbnN0IGxjYmxrID0gJihjYmxrLT5sYXllcnNbaW5kZXhdKTsKLSAgICBjb25zdCBzdXJmYWNlX2luZm9fdCogY29uc3QgZnJvbnQgPSBsY2Jsay0+c3VyZmFjZSArICgxLWJhY2tJZHgpOwotICAgICAgICBpbmZvLT53ICAgICAgPSBmcm9udC0+dzsKLSAgICAgICAgaW5mby0+aCAgICAgID0gZnJvbnQtPmg7Ci0gICAgICAgIGluZm8tPmZvcm1hdCA9IGZyb250LT5mb3JtYXQ7Ci0gICAgICAgIGluZm8tPmJhc2UgICA9IHN1cmZhY2UtPmhlYXBCYXNlKDEtYmFja0lkeCk7Ci0gICAgICAgIGluZm8tPmJpdHMgICA9IHJlaW50ZXJwcmV0X2Nhc3Q8dm9pZCo+KGludHB0cl90KGluZm8tPmJhc2UpICsgZnJvbnQtPmJpdHNfb2Zmc2V0KTsKLSAgICAgICAgaW5mby0+YnByICAgID0gZnJvbnQtPmJwcjsKLQotICAgIHJldHVybiAwOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OmxvY2tTdXJmYWNlKAotICAgICAgICBTdXJmYWNlKiBzdXJmYWNlLAotICAgICAgICBTdXJmYWNlOjpTdXJmYWNlSW5mbyogb3RoZXIsCi0gICAgICAgIFJlZ2lvbiogZGlydHksCi0gICAgICAgIGJvb2wgYmxvY2tpbmcpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKHN1cmZhY2UtPmdldExvY2soKSk7Ci0KLSAgICBTdXJmYWNlSUQgaW5kZXggPSBzdXJmYWNlLT5JRCgpOwotICAgIHBlcl9jbGllbnRfY2Jsa190KiBjb25zdCBjYmxrID0gbUNvbnRyb2w7Ci0gICAgc3RhdHVzX3QgZXJyID0gdmFsaWRhdGVTdXJmYWNlKGNibGssIHN1cmZhY2UpOwotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpCi0gICAgICAgIHJldHVybiBlcnI7Ci0KLSAgICBpbnQzMl90IGJhY2tJZHggPSBjYmxrLT5sb2NrX2xheWVyKHNpemVfdChpbmRleCksCi0gICAgICAgICAgICBwZXJfY2xpZW50X2NibGtfdDo6QkxPQ0tJTkcpOwotICAgIGlmIChiYWNrSWR4ID49IDApIHsKLSAgICAgICAgc3VyZmFjZS0+bUJhY2tidWZmZXJJbmRleCA9IGJhY2tJZHg7Ci0gICAgICAgIGxheWVyX2NibGtfdCogY29uc3QgbGNibGsgPSAmKGNibGstPmxheWVyc1tpbmRleF0pOwotICAgICAgICBjb25zdCBzdXJmYWNlX2luZm9fdCogY29uc3QgYmFjayA9IGxjYmxrLT5zdXJmYWNlICsgYmFja0lkeDsKLSAgICAgICAgY29uc3Qgc3VyZmFjZV9pbmZvX3QqIGNvbnN0IGZyb250ID0gbGNibGstPnN1cmZhY2UgKyAoMS1iYWNrSWR4KTsKLSAgICAgICAgICAgIG90aGVyLT53ICAgICAgPSBiYWNrLT53OwotICAgICAgICAgICAgb3RoZXItPmggICAgICA9IGJhY2stPmg7Ci0gICAgICAgICAgICBvdGhlci0+Zm9ybWF0ID0gYmFjay0+Zm9ybWF0OwotICAgICAgICAgICAgb3RoZXItPmJhc2UgICA9IHN1cmZhY2UtPmhlYXBCYXNlKGJhY2tJZHgpOwotICAgICAgICAgICAgb3RoZXItPmJpdHMgICA9IHJlaW50ZXJwcmV0X2Nhc3Q8dm9pZCo+KGludHB0cl90KG90aGVyLT5iYXNlKSArIGJhY2stPmJpdHNfb2Zmc2V0KTsKLSAgICAgICAgICAgIG90aGVyLT5icHIgICAgPSBiYWNrLT5icHI7Ci0KLSAgICAgICAgY29uc3QgUmVjdCBib3VuZHMob3RoZXItPncsIG90aGVyLT5oKTsKLSAgICAgICAgUmVnaW9uIG5ld0RpcnR5UmVnaW9uOwotCi0gICAgICAgIGlmIChiYWNrLT5mbGFncyAmIHN1cmZhY2VfaW5mb190OjplQnVmZmVyRGlydHkpIHsKLSAgICAgICAgICAgIC8qIGl0IGlzIHNhZmUgdG8gd3JpdGUgKmJhY2sgaGVyZSwgYmVjYXVzZSB3ZSdyZSBndWFyYW50ZWVkCi0gICAgICAgICAgICAgKiBTdXJmYWNlRmxpbmdlciBpcyBub3QgdG91Y2hpbmcgaXQgKHNpbmNlIGl0IGp1c3QgZ3JhbnRlZAotICAgICAgICAgICAgICogYWNjZXNzIHRvIHVzKSAqLwotICAgICAgICAgICAgY29uc3RfY2FzdDxzdXJmYWNlX2luZm9fdCo+KGJhY2spLT5mbGFncyAmPQotICAgICAgICAgICAgICAgICAgICB+c3VyZmFjZV9pbmZvX3Q6OmVCdWZmZXJEaXJ0eTsKLQotICAgICAgICAgICAgLy8gY29udGVudCBpcyBtZWFuaW5nbGVzcyBpbiB0aGlzIGNhc2UgYW5kIHRoZSB3aG9sZSBzdXJmYWNlCi0gICAgICAgICAgICAvLyBuZWVkcyB0byBiZSByZWRyYXduLgotCi0gICAgICAgICAgICBuZXdEaXJ0eVJlZ2lvbi5zZXQoYm91bmRzKTsKLSAgICAgICAgICAgIGlmIChkaXJ0eSkgewotICAgICAgICAgICAgICAgICpkaXJ0eSA9IG5ld0RpcnR5UmVnaW9uOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvL2lmIChieXRlc1BlclBpeGVsKG90aGVyLT5mb3JtYXQpID09IDQpIHsKLSAgICAgICAgICAgIC8vICAgIGFuZHJvaWRfbWVtc2V0MzIoCi0gICAgICAgICAgICAvLyAgICAgICAgKHVpbnQzMl90KilvdGhlci0+Yml0cywgMHhGRjAwRkYwMCwgb3RoZXItPmggKiBvdGhlci0+YnByKTsKLSAgICAgICAgICAgIC8vfSBlbHNlIHsKLSAgICAgICAgICAgIC8vICAgIGFuZHJvaWRfbWVtc2V0MTYoIC8vIGZpbGwgd2l0aCBncmVlbgotICAgICAgICAgICAgLy8gICAgICAgICh1aW50MTZfdCopb3RoZXItPmJpdHMsIDB4N0UwLCBvdGhlci0+aCAqIG90aGVyLT5icHIpOwotICAgICAgICAgICAgLy99Ci0gICAgICAgIH0KLSAgICAgICAgZWxzZQotICAgICAgICB7Ci0gICAgICAgICAgICBpZiAoZGlydHkpIHsKLSAgICAgICAgICAgICAgICBkaXJ0eS0+YW5kU2VsZihSZWdpb24oYm91bmRzKSk7Ci0gICAgICAgICAgICAgICAgbmV3RGlydHlSZWdpb24gPSAqZGlydHk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIG5ld0RpcnR5UmVnaW9uLnNldChib3VuZHMpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBSZWdpb24gY29weWJhY2s7Ci0gICAgICAgICAgICBpZiAoIShsY2Jsay0+ZmxhZ3MgJiBlTm9Db3B5QmFjaykpIHsKLSAgICAgICAgICAgICAgICBjb25zdCBSZWdpb24gcHJldmlvdXNEaXJ0eVJlZ2lvbihzdXJmYWNlLT5kaXJ0eVJlZ2lvbigpKTsKLSAgICAgICAgICAgICAgICBjb3B5YmFjayA9IHByZXZpb3VzRGlydHlSZWdpb24uc3VidHJhY3QobmV3RGlydHlSZWdpb24pOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoIWNvcHliYWNrLmlzRW1wdHkoKSkgewotICAgICAgICAgICAgICAgIC8vIGNvcHkgZnJvbnQgdG8gYmFjawotICAgICAgICAgICAgICAgIEdHTFN1cmZhY2UgY2I7Ci0gICAgICAgICAgICAgICAgICAgIGNiLnZlcnNpb24gPSBzaXplb2YoR0dMU3VyZmFjZSk7Ci0gICAgICAgICAgICAgICAgICAgIGNiLndpZHRoID0gYmFjay0+dzsKLSAgICAgICAgICAgICAgICAgICAgY2IuaGVpZ2h0ID0gYmFjay0+aDsKLSAgICAgICAgICAgICAgICAgICAgY2Iuc3RyaWRlID0gYmFjay0+c3RyaWRlOwotICAgICAgICAgICAgICAgICAgICBjYi5kYXRhID0gKEdHTHVieXRlKilzdXJmYWNlLT5oZWFwQmFzZShiYWNrSWR4KTsKLSAgICAgICAgICAgICAgICAgICAgY2IuZGF0YSArPSBiYWNrLT5iaXRzX29mZnNldDsKLSAgICAgICAgICAgICAgICAgICAgY2IuZm9ybWF0ID0gYmFjay0+Zm9ybWF0OwotCi0gICAgICAgICAgICAgICAgR0dMU3VyZmFjZSB0OwotICAgICAgICAgICAgICAgICAgICB0LnZlcnNpb24gPSBzaXplb2YoR0dMU3VyZmFjZSk7Ci0gICAgICAgICAgICAgICAgICAgIHQud2lkdGggPSBmcm9udC0+dzsKLSAgICAgICAgICAgICAgICAgICAgdC5oZWlnaHQgPSBmcm9udC0+aDsKLSAgICAgICAgICAgICAgICAgICAgdC5zdHJpZGUgPSBmcm9udC0+c3RyaWRlOwotICAgICAgICAgICAgICAgICAgICB0LmRhdGEgPSAoR0dMdWJ5dGUqKXN1cmZhY2UtPmhlYXBCYXNlKDEtYmFja0lkeCk7Ci0gICAgICAgICAgICAgICAgICAgIHQuZGF0YSArPSBmcm9udC0+Yml0c19vZmZzZXQ7Ci0gICAgICAgICAgICAgICAgICAgIHQuZm9ybWF0ID0gZnJvbnQtPmZvcm1hdDsKLQotICAgICAgICAgICAgICAgIC8vY29uc3QgUmVnaW9uIGNvcHliYWNrKGxjYmxrLT5yZWdpb24gKyAxLWJhY2tJZHgpOwotICAgICAgICAgICAgICAgIGNvcHlCbHQoY2IsIHQsIGNvcHliYWNrKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIHVwZGF0ZSBkaXJ0eSByZWdpb24KLSAgICAgICAgc3VyZmFjZS0+c2V0RGlydHlSZWdpb24obmV3RGlydHlSZWdpb24pOwotICAgIH0KLSAgICByZXR1cm4gKGJhY2tJZHggPCAwKSA/IHN0YXR1c190KGJhY2tJZHgpIDogc3RhdHVzX3QoTk9fRVJST1IpOwotfQotCi12b2lkIFN1cmZhY2VDb21wb3NlckNsaWVudDo6X3NpZ25hbF9zZXJ2ZXIoKQotewotICAgIG1TaWduYWxTZXJ2ZXItPnNpZ25hbCgpOwotfQotCi12b2lkIFN1cmZhY2VDb21wb3NlckNsaWVudDo6X3NlbmRfZGlydHlfcmVnaW9uKAotICAgICAgICBsYXllcl9jYmxrX3QqIGxjYmxrLCBjb25zdCBSZWdpb24mIGRpcnR5KQotewotICAgIGNvbnN0IGludDMyX3QgaW5kZXggPSAobGNibGstPmZsYWdzICYgZUJ1ZmZlckluZGV4KSA+PiBlQnVmZmVySW5kZXhTaGlmdDsKLSAgICBmbGF0X3JlZ2lvbl90KiBmbGF0X3JlZ2lvbiA9IGxjYmxrLT5yZWdpb24gKyBpbmRleDsKLSAgICBzdGF0dXNfdCBlcnIgPSBkaXJ0eS53cml0ZShmbGF0X3JlZ2lvbiwgc2l6ZW9mKGZsYXRfcmVnaW9uX3QpKTsKLSAgICBpZiAoZXJyIDwgTk9fRVJST1IpIHsKLSAgICAgICAgLy8gcmVnaW9uIGRvZXNuJ3QgZml0LCB1c2UgdGhlIGJvdW5kcwotICAgICAgICBjb25zdCBSZWdpb24gcmVnKGRpcnR5LmJvdW5kcygpKTsKLSAgICAgICAgcmVnLndyaXRlKGZsYXRfcmVnaW9uLCBzaXplb2YoZmxhdF9yZWdpb25fdCkpOwotICAgIH0KLX0KLQotc3RhdHVzX3QgU3VyZmFjZUNvbXBvc2VyQ2xpZW50Ojp1bmxvY2tBbmRQb3N0U3VyZmFjZShTdXJmYWNlKiBzdXJmYWNlKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChzdXJmYWNlLT5nZXRMb2NrKCkpOwotCi0gICAgU3VyZmFjZUlEIGluZGV4ID0gc3VyZmFjZS0+SUQoKTsKLSAgICBwZXJfY2xpZW50X2NibGtfdCogY29uc3QgY2JsayA9IG1Db250cm9sOwotICAgIHN0YXR1c190IGVyciA9IHZhbGlkYXRlU3VyZmFjZShjYmxrLCBzdXJmYWNlKTsKLSAgICBpZiAoZXJyICE9IE5PX0VSUk9SKQotICAgICAgICByZXR1cm4gZXJyOwotCi0gICAgUmVnaW9uIGRpcnR5KHN1cmZhY2UtPmRpcnR5UmVnaW9uKCkpOwotICAgIGNvbnN0IFJlY3QmIHN3YXBSZWN0KHN1cmZhY2UtPnN3YXBSZWN0YW5nbGUoKSk7Ci0gICAgaWYgKHN3YXBSZWN0LmlzVmFsaWQoKSkgewotICAgICAgICBkaXJ0eS5zZXQoc3dhcFJlY3QpOwotICAgIH0KLQotICAgIC8vIHRyYW5zbWl0IHRoZSBkaXJ0eSByZWdpb24KLSAgICBsYXllcl9jYmxrX3QqIGNvbnN0IGxjYmxrID0gJihjYmxrLT5sYXllcnNbaW5kZXhdKTsKLSAgICBfc2VuZF9kaXJ0eV9yZWdpb24obGNibGssIGRpcnR5KTsKLSAgICB1aW50MzJfdCBuZXdzdGF0ZSA9IGNibGstPnVubG9ja19sYXllcl9hbmRfcG9zdChzaXplX3QoaW5kZXgpKTsKLSAgICBpZiAoIShuZXdzdGF0ZSAmIGVOZXh0RmxpcFBlbmRpbmcpKQotICAgICAgICBfc2lnbmFsX3NlcnZlcigpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZUNvbXBvc2VyQ2xpZW50Ojp1bmxvY2tTdXJmYWNlKFN1cmZhY2UqIHN1cmZhY2UpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKHN1cmZhY2UtPmdldExvY2soKSk7Ci0KLSAgICBTdXJmYWNlSUQgaW5kZXggPSBzdXJmYWNlLT5JRCgpOwotICAgIHBlcl9jbGllbnRfY2Jsa190KiBjb25zdCBjYmxrID0gbUNvbnRyb2w7Ci0gICAgc3RhdHVzX3QgZXJyID0gdmFsaWRhdGVTdXJmYWNlKGNibGssIHN1cmZhY2UpOwotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpCi0gICAgICAgIHJldHVybiBlcnI7Ci0KLSAgICBsYXllcl9jYmxrX3QqIGNvbnN0IGxjYmxrID0gJihjYmxrLT5sYXllcnNbaW5kZXhdKTsKLSAgICBjYmxrLT51bmxvY2tfbGF5ZXIoc2l6ZV90KGluZGV4KSk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi12b2lkIFN1cmZhY2VDb21wb3NlckNsaWVudDo6b3Blbkdsb2JhbFRyYW5zYWN0aW9uKCkKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2woZ0xvY2spOwotCi0gICAgaWYgKGdPcGVuVHJhbnNhY3Rpb25zLnNpemUoKSkgewotICAgICAgICBMT0dFKCJvcGVuR2xvYmFsVHJhbnNhY3Rpb24oKSBjYWxsZWQgbW9yZSB0aGFuIG9uY2UuIHNraXBwaW5nLiIpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgY29uc3Qgc2l6ZV90IE4gPSBnQWN0aXZlQ29ubmVjdGlvbnMuc2l6ZSgpOwotICAgIFZFUkJPU0UoIm9wZW5HbG9iYWxUcmFuc2FjdGlvbiAoJWxkIGNsaWVudHMpIiwgTik7Ci0gICAgZm9yIChzaXplX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICBzcDxTdXJmYWNlQ29tcG9zZXJDbGllbnQ+IGNsaWVudChnQWN0aXZlQ29ubmVjdGlvbnMudmFsdWVBdChpKSk7Ci0gICAgICAgIGlmIChnT3BlblRyYW5zYWN0aW9ucy5pbmRleE9mKGNsaWVudCkgPCAwKSB7Ci0gICAgICAgICAgICBpZiAoY2xpZW50LT5vcGVuVHJhbnNhY3Rpb24oKSA9PSBOT19FUlJPUikgewotICAgICAgICAgICAgICAgIGlmIChnT3BlblRyYW5zYWN0aW9ucy5hZGQoY2xpZW50KSA8IDApIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gT29vcHMhCi0gICAgICAgICAgICAgICAgICAgIExPR0UoICAgIlVuYWJsZSB0byBhZGQgYSBTdXJmYWNlQ29tcG9zZXJDbGllbnQgIgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICJ0byB0aGUgZ2xvYmFsIHRyYW5zYWN0aW9uIHNldCAob3V0IG9mIG1lbW9yeT8pIik7Ci0gICAgICAgICAgICAgICAgICAgIGNsaWVudC0+Y2xvc2VUcmFuc2FjdGlvbigpOwotICAgICAgICAgICAgICAgICAgICAvLyBsZXQgaXQgZ28sIGl0J2xsIGZhaWwgbGF0ZXIgd2hlbiB0aGUgdXNlcgotICAgICAgICAgICAgICAgICAgICAvLyB0cmllcyB0byBkbyBzb21ldGhpbmcgd2l0aCB0aGUgdHJhbnNhY3Rpb24KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIExPR0UoIm9wZW5UcmFuc2FjdGlvbiBvbiBjbGllbnQgJXAgZmFpbGVkIiwgY2xpZW50LmdldCgpKTsKLSAgICAgICAgICAgICAgICAvLyBsZXQgaXQgZ28sIGl0J2xsIGZhaWwgbGF0ZXIgd2hlbiB0aGUgdXNlcgotICAgICAgICAgICAgICAgIC8vIHRyaWVzIHRvIGRvIHNvbWV0aGluZyB3aXRoIHRoZSB0cmFuc2FjdGlvbgotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQotCi12b2lkIFN1cmZhY2VDb21wb3NlckNsaWVudDo6Y2xvc2VHbG9iYWxUcmFuc2FjdGlvbigpCi17Ci0gICAgZ0xvY2subG9jaygpOwotICAgICAgICBTb3J0ZWRWZWN0b3I8IHNwPFN1cmZhY2VDb21wb3NlckNsaWVudD4gPiBjbGllbnRzKGdPcGVuVHJhbnNhY3Rpb25zKTsKLSAgICAgICAgZ09wZW5UcmFuc2FjdGlvbnMuY2xlYXIoKTsKLSAgICBnTG9jay51bmxvY2soKTsKLQotICAgIGNvbnN0IHNpemVfdCBOID0gY2xpZW50cy5zaXplKCk7Ci0gICAgVkVSQk9TRSgiY2xvc2VHbG9iYWxUcmFuc2FjdGlvbiAoJWxkIGNsaWVudHMpIiwgTik7Ci0gICAgaWYgKE4gPT0gMSkgewotICAgICAgICBjbGllbnRzWzBdLT5jbG9zZVRyYW5zYWN0aW9uKCk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgY29uc3Qgc3A8SVN1cmZhY2VDb21wb3Nlcj4mIHNtKF9nZXRfc3VyZmFjZV9tYW5hZ2VyKCkpOwotICAgICAgICBzbS0+b3Blbkdsb2JhbFRyYW5zYWN0aW9uKCk7Ci0gICAgICAgIGZvciAoc2l6ZV90IGk9MDsgaTxOOyBpKyspIHsKLSAgICAgICAgICAgIGNsaWVudHNbaV0tPmNsb3NlVHJhbnNhY3Rpb24oKTsKLSAgICAgICAgfQotICAgICAgICBzbS0+Y2xvc2VHbG9iYWxUcmFuc2FjdGlvbigpOwotICAgIH0KLX0KLQotc3RhdHVzX3QgU3VyZmFjZUNvbXBvc2VyQ2xpZW50OjpmcmVlemVEaXNwbGF5KERpc3BsYXlJRCBkcHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIGNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBzbShfZ2V0X3N1cmZhY2VfbWFuYWdlcigpKTsKLSAgICByZXR1cm4gc20tPmZyZWV6ZURpc3BsYXkoZHB5LCBmbGFncyk7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VDb21wb3NlckNsaWVudDo6dW5mcmVlemVEaXNwbGF5KERpc3BsYXlJRCBkcHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIGNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBzbShfZ2V0X3N1cmZhY2VfbWFuYWdlcigpKTsKLSAgICByZXR1cm4gc20tPnVuZnJlZXplRGlzcGxheShkcHksIGZsYWdzKTsKLX0KLQotaW50IFN1cmZhY2VDb21wb3NlckNsaWVudDo6c2V0T3JpZW50YXRpb24oRGlzcGxheUlEIGRweSwgaW50IG9yaWVudGF0aW9uKQotewotICAgIGNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBzbShfZ2V0X3N1cmZhY2VfbWFuYWdlcigpKTsKLSAgICByZXR1cm4gc20tPnNldE9yaWVudGF0aW9uKGRweSwgb3JpZW50YXRpb24pOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6Om9wZW5UcmFuc2FjdGlvbigpCi17Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpCi0gICAgICAgIHJldHVybiBtU3RhdHVzOwotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgVkVSQk9TRSggICAib3BlblRyYW5zYWN0aW9uIChjbGllbnQgJXAsIG1UcmFuc2FjdGlvbk9wZW49JWQpIiwKLSAgICAgICAgICAgIHRoaXMsIG1UcmFuc2FjdGlvbk9wZW4pOwotICAgIG1UcmFuc2FjdGlvbk9wZW4rKzsKLSAgICBpZiAobVByZWJ1aWx0TGF5ZXJTdGF0ZSA9PSAwKSB7Ci0gICAgICAgIG1QcmVidWlsdExheWVyU3RhdGUgPSBuZXcgbGF5ZXJfc3RhdGVfdDsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0KLXN0YXR1c190IFN1cmZhY2VDb21wb3NlckNsaWVudDo6Y2xvc2VUcmFuc2FjdGlvbigpCi17Ci0gICAgaWYgKG1TdGF0dXMgIT0gTk9fRVJST1IpCi0gICAgICAgIHJldHVybiBtU3RhdHVzOwotCi0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLQotICAgIFZFUkJPU0UoICAgImNsb3NlVHJhbnNhY3Rpb24gKGNsaWVudCAlcCwgbVRyYW5zYWN0aW9uT3Blbj0lZCkiLAotICAgICAgICAgICAgdGhpcywgbVRyYW5zYWN0aW9uT3Blbik7Ci0KLSAgICBpZiAobVRyYW5zYWN0aW9uT3BlbiA8PSAwKSB7Ci0gICAgICAgIExPR0UoICAgImNsb3NlVHJhbnNhY3Rpb24gKGNsaWVudCAlcCwgbVRyYW5zYWN0aW9uT3Blbj0lZCkgIgotICAgICAgICAgICAgICAgICJjYWxsZWQgbW9yZSB0aW1lcyB0aGFuIG9wZW5UcmFuc2FjdGlvbigpIiwKLSAgICAgICAgICAgICAgICB0aGlzLCBtVHJhbnNhY3Rpb25PcGVuKTsKLSAgICAgICAgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOwotICAgIH0KLQotICAgIGlmIChtVHJhbnNhY3Rpb25PcGVuID49IDIpIHsKLSAgICAgICAgbVRyYW5zYWN0aW9uT3Blbi0tOwotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotCi0gICAgbVRyYW5zYWN0aW9uT3BlbiA9IDA7Ci0gICAgY29uc3Qgc3NpemVfdCBjb3VudCA9IG1TdGF0ZXMuc2l6ZSgpOwotICAgIGlmIChjb3VudCkgewotICAgICAgICBtQ2xpZW50LT5zZXRTdGF0ZShjb3VudCwgbVN0YXRlcy5hcnJheSgpKTsKLSAgICAgICAgbVN0YXRlcy5jbGVhcigpOwotICAgIH0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLWxheWVyX3N0YXRlX3QqIFN1cmZhY2VDb21wb3NlckNsaWVudDo6X2dldF9zdGF0ZV9sKGNvbnN0IHNwPFN1cmZhY2U+JiBzdXJmYWNlKQotewotICAgIFN1cmZhY2VJRCBpbmRleCA9IHN1cmZhY2UtPklEKCk7Ci0gICAgcGVyX2NsaWVudF9jYmxrX3QqIGNvbnN0IGNibGsgPSBtQ29udHJvbDsKLSAgICBzdGF0dXNfdCBlcnIgPSB2YWxpZGF0ZVN1cmZhY2UoY2Jsaywgc3VyZmFjZS5nZXQoKSk7Ci0gICAgaWYgKGVyciAhPSBOT19FUlJPUikKLSAgICAgICAgcmV0dXJuIDA7Ci0KLSAgICAvLyBBUEkgdXNhZ2UgZXJyb3IsIGRvIG5vdGhpbmcuCi0gICAgaWYgKG1UcmFuc2FjdGlvbk9wZW48PTApIHsKLSAgICAgICAgTE9HRSgiTm90IGluIHRyYW5zYWN0aW9uIChjbGllbnQ9JXAsIFN1cmZhY2VJRD0lZCwgbVRyYW5zYWN0aW9uT3Blbj0lZCIsCi0gICAgICAgICAgICAgICAgdGhpcywgaW50KGluZGV4KSwgbVRyYW5zYWN0aW9uT3Blbik7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIC8vIHVzZSBtUHJlYnVpbHRMYXllclN0YXRlIGp1c3QgdG8gZmluZCBvdXQgaWYgd2UgYWxyZWFkeSBoYXZlIGl0Ci0gICAgbGF5ZXJfc3RhdGVfdCYgZHVtbXkgPSAqbVByZWJ1aWx0TGF5ZXJTdGF0ZTsKLSAgICBkdW1teS5zdXJmYWNlID0gaW5kZXg7Ci0gICAgc3NpemVfdCBpID0gbVN0YXRlcy5pbmRleE9mKGR1bW15KTsKLSAgICBpZiAoaSA8IDApIHsKLSAgICAgICAgLy8gd2UgZG9uJ3QgaGF2ZSBpdCwgYWRkIGFuIGluaXRpYWxpemVkIGxheWVyX3N0YXRlIHRvIG91ciBsaXN0Ci0gICAgICAgIGkgPSBtU3RhdGVzLmFkZChkdW1teSk7Ci0gICAgfQotICAgIHJldHVybiBtU3RhdGVzLmVkaXRBcnJheSgpICsgaTsKLX0KLQotbGF5ZXJfc3RhdGVfdCogU3VyZmFjZUNvbXBvc2VyQ2xpZW50OjpfbG9ja0xheWVyU3RhdGUoY29uc3Qgc3A8U3VyZmFjZT4mIHN1cmZhY2UpCi17Ci0gICAgbGF5ZXJfc3RhdGVfdCogczsKLSAgICBtTG9jay5sb2NrKCk7Ci0gICAgcyA9IF9nZXRfc3RhdGVfbChzdXJmYWNlKTsKLSAgICBpZiAoIXMpIG1Mb2NrLnVubG9jaygpOwotICAgIHJldHVybiBzOwotfQotCi12b2lkIFN1cmZhY2VDb21wb3NlckNsaWVudDo6X3VubG9ja0xheWVyU3RhdGUoKQotewotICAgIG1Mb2NrLnVubG9jaygpOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OnNldFBvc2l0aW9uKFN1cmZhY2UqIHN1cmZhY2UsIGludDMyX3QgeCwgaW50MzJfdCB5KQotewotICAgIGxheWVyX3N0YXRlX3QqIHMgPSBfbG9ja0xheWVyU3RhdGUoc3VyZmFjZSk7Ci0gICAgaWYgKCFzKSByZXR1cm4gQkFEX0lOREVYOwotICAgIHMtPndoYXQgfD0gSVN1cmZhY2VDb21wb3Nlcjo6ZVBvc2l0aW9uQ2hhbmdlZDsKLSAgICBzLT54ID0geDsKLSAgICBzLT55ID0geTsKLSAgICBfdW5sb2NrTGF5ZXJTdGF0ZSgpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZUNvbXBvc2VyQ2xpZW50OjpzZXRTaXplKFN1cmZhY2UqIHN1cmZhY2UsIHVpbnQzMl90IHcsIHVpbnQzMl90IGgpCi17Ci0gICAgbGF5ZXJfc3RhdGVfdCogcyA9IF9sb2NrTGF5ZXJTdGF0ZShzdXJmYWNlKTsKLSAgICBpZiAoIXMpIHJldHVybiBCQURfSU5ERVg7Ci0gICAgcy0+d2hhdCB8PSBJU3VyZmFjZUNvbXBvc2VyOjplU2l6ZUNoYW5nZWQ7Ci0gICAgcy0+dyA9IHc7Ci0gICAgcy0+aCA9IGg7Ci0gICAgX3VubG9ja0xheWVyU3RhdGUoKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VDb21wb3NlckNsaWVudDo6c2V0TGF5ZXIoU3VyZmFjZSogc3VyZmFjZSwgaW50MzJfdCB6KQotewotICAgIGxheWVyX3N0YXRlX3QqIHMgPSBfbG9ja0xheWVyU3RhdGUoc3VyZmFjZSk7Ci0gICAgaWYgKCFzKSByZXR1cm4gQkFEX0lOREVYOwotICAgIHMtPndoYXQgfD0gSVN1cmZhY2VDb21wb3Nlcjo6ZUxheWVyQ2hhbmdlZDsKLSAgICBzLT56ID0gejsKLSAgICBfdW5sb2NrTGF5ZXJTdGF0ZSgpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgU3VyZmFjZUNvbXBvc2VyQ2xpZW50OjpoaWRlKFN1cmZhY2UqIHN1cmZhY2UpCi17Ci0gICAgcmV0dXJuIHNldEZsYWdzKHN1cmZhY2UsIElTdXJmYWNlQ29tcG9zZXI6OmVMYXllckhpZGRlbiwKLSAgICAgICAgICAgIElTdXJmYWNlQ29tcG9zZXI6OmVMYXllckhpZGRlbik7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VDb21wb3NlckNsaWVudDo6c2hvdyhTdXJmYWNlKiBzdXJmYWNlLCBpbnQzMl90KQotewotICAgIHJldHVybiBzZXRGbGFncyhzdXJmYWNlLCAwLCBJU3VyZmFjZUNvbXBvc2VyOjplTGF5ZXJIaWRkZW4pOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OmZyZWV6ZShTdXJmYWNlKiBzdXJmYWNlKQotewotICAgIHJldHVybiBzZXRGbGFncyhzdXJmYWNlLCBJU3VyZmFjZUNvbXBvc2VyOjplTGF5ZXJGcm96ZW4sCi0gICAgICAgICAgICBJU3VyZmFjZUNvbXBvc2VyOjplTGF5ZXJGcm96ZW4pOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OnVuZnJlZXplKFN1cmZhY2UqIHN1cmZhY2UpCi17Ci0gICAgcmV0dXJuIHNldEZsYWdzKHN1cmZhY2UsIDAsIElTdXJmYWNlQ29tcG9zZXI6OmVMYXllckZyb3plbik7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VDb21wb3NlckNsaWVudDo6c2V0RmxhZ3MoU3VyZmFjZSogc3VyZmFjZSwKLSAgICAgICAgdWludDMyX3QgZmxhZ3MsIHVpbnQzMl90IG1hc2spCi17Ci0gICAgbGF5ZXJfc3RhdGVfdCogcyA9IF9sb2NrTGF5ZXJTdGF0ZShzdXJmYWNlKTsKLSAgICBpZiAoIXMpIHJldHVybiBCQURfSU5ERVg7Ci0gICAgcy0+d2hhdCB8PSBJU3VyZmFjZUNvbXBvc2VyOjplVmlzaWJpbGl0eUNoYW5nZWQ7Ci0gICAgcy0+ZmxhZ3MgJj0gfm1hc2s7Ci0gICAgcy0+ZmxhZ3MgfD0gKGZsYWdzICYgbWFzayk7Ci0gICAgcy0+bWFzayB8PSBtYXNrOwotICAgIF91bmxvY2tMYXllclN0YXRlKCk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0KLXN0YXR1c190IFN1cmZhY2VDb21wb3NlckNsaWVudDo6c2V0VHJhbnNwYXJlbnRSZWdpb25IaW50KAotICAgICAgICBTdXJmYWNlKiBzdXJmYWNlLCBjb25zdCBSZWdpb24mIHRyYW5zcGFyZW50UmVnaW9uKQotewotICAgIGxheWVyX3N0YXRlX3QqIHMgPSBfbG9ja0xheWVyU3RhdGUoc3VyZmFjZSk7Ci0gICAgaWYgKCFzKSByZXR1cm4gQkFEX0lOREVYOwotICAgIHMtPndoYXQgfD0gSVN1cmZhY2VDb21wb3Nlcjo6ZVRyYW5zcGFyZW50UmVnaW9uQ2hhbmdlZDsKLSAgICBzLT50cmFuc3BhcmVudFJlZ2lvbiA9IHRyYW5zcGFyZW50UmVnaW9uOwotICAgIF91bmxvY2tMYXllclN0YXRlKCk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OnNldEFscGhhKFN1cmZhY2UqIHN1cmZhY2UsIGZsb2F0IGFscGhhKQotewotICAgIGxheWVyX3N0YXRlX3QqIHMgPSBfbG9ja0xheWVyU3RhdGUoc3VyZmFjZSk7Ci0gICAgaWYgKCFzKSByZXR1cm4gQkFEX0lOREVYOwotICAgIHMtPndoYXQgfD0gSVN1cmZhY2VDb21wb3Nlcjo6ZUFscGhhQ2hhbmdlZDsKLSAgICBzLT5hbHBoYSA9IGFscGhhOwotICAgIF91bmxvY2tMYXllclN0YXRlKCk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OnNldE1hdHJpeCgKLSAgICAgICAgU3VyZmFjZSogc3VyZmFjZSwKLSAgICAgICAgZmxvYXQgZHNkeCwgZmxvYXQgZHRkeCwKLSAgICAgICAgZmxvYXQgZHNkeSwgZmxvYXQgZHRkeSApCi17Ci0gICAgbGF5ZXJfc3RhdGVfdCogcyA9IF9sb2NrTGF5ZXJTdGF0ZShzdXJmYWNlKTsKLSAgICBpZiAoIXMpIHJldHVybiBCQURfSU5ERVg7Ci0gICAgcy0+d2hhdCB8PSBJU3VyZmFjZUNvbXBvc2VyOjplTWF0cml4Q2hhbmdlZDsKLSAgICBsYXllcl9zdGF0ZV90OjptYXRyaXgyMl90IG1hdHJpeDsKLSAgICBtYXRyaXguZHNkeCA9IGRzZHg7Ci0gICAgbWF0cml4LmR0ZHggPSBkdGR4OwotICAgIG1hdHJpeC5kc2R5ID0gZHNkeTsKLSAgICBtYXRyaXguZHRkeSA9IGR0ZHk7Ci0gICAgcy0+bWF0cml4ID0gbWF0cml4OwotICAgIF91bmxvY2tMYXllclN0YXRlKCk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlQ29tcG9zZXJDbGllbnQ6OnNldEZyZWV6ZVRpbnQoU3VyZmFjZSogc3VyZmFjZSwgdWludDMyX3QgdGludCkKLXsKLSAgICBsYXllcl9zdGF0ZV90KiBzID0gX2xvY2tMYXllclN0YXRlKHN1cmZhY2UpOwotICAgIGlmICghcykgcmV0dXJuIEJBRF9JTkRFWDsKLSAgICBzLT53aGF0IHw9IElTdXJmYWNlQ29tcG9zZXI6OmVGcmVlemVUaW50Q2hhbmdlZDsKLSAgICBzLT50aW50ID0gdGludDsKLSAgICBfdW5sb2NrTGF5ZXJTdGF0ZSgpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy91aS9TdXJmYWNlRmxpbmdlclN5bmNocm8uY3BwIGIvbGlicy91aS9TdXJmYWNlRmxpbmdlclN5bmNocm8uY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1Y2Q5NzU1Li4wMDAwMDAwCi0tLSBhL2xpYnMvdWkvU3VyZmFjZUZsaW5nZXJTeW5jaHJvLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDEyMyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJTdXJmYWNlRmxpbmdlclN5bmNocm8iCi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPGVycm5vLmg+Ci0jaW5jbHVkZSA8bGltaXRzLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8c3lzL3N0YXQuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0lQQ1RocmVhZFN0YXRlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlIDxwcml2YXRlL3VpL1N1cmZhY2VGbGluZ2VyU3luY2hyby5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1TdXJmYWNlRmxpbmdlclN5bmNocm86OkJhcnJpZXI6OkJhcnJpZXIoKQotICAgIDogc3RhdGUoQ0xPU0VEKSB7IAotfQotCi1TdXJmYWNlRmxpbmdlclN5bmNocm86OkJhcnJpZXI6On5CYXJyaWVyKCkgeyAKLX0KLQotdm9pZCBTdXJmYWNlRmxpbmdlclN5bmNocm86OkJhcnJpZXI6Om9wZW4oKSB7Ci0gICAgYXNtIHZvbGF0aWxlICgiIjo6OiJtZW1vcnkiKTsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobG9jayk7Ci0gICAgc3RhdGUgPSBPUEVORUQ7Ci0gICAgY3YuYnJvYWRjYXN0KCk7Ci19Ci0KLXZvaWQgU3VyZmFjZUZsaW5nZXJTeW5jaHJvOjpCYXJyaWVyOjpjbG9zZSgpIHsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobG9jayk7Ci0gICAgc3RhdGUgPSBDTE9TRUQ7Ci19Ci0KLXZvaWQgU3VyZmFjZUZsaW5nZXJTeW5jaHJvOjpCYXJyaWVyOjp3YWl0QW5kQ2xvc2UoKSAKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobG9jayk7Ci0gICAgd2hpbGUgKHN0YXRlID09IENMT1NFRCkgewotICAgICAgICAvLyB3ZSdyZSBhYm91dCB0byB3YWl0LCBmbHVzaCB0aGUgYmluZGVyIGNvbW1hbmQgYnVmZmVyCi0gICAgICAgIElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPmZsdXNoQ29tbWFuZHMoKTsKLSAgICAgICAgY3Yud2FpdChsb2NrKTsKLSAgICB9Ci0gICAgc3RhdGUgPSBDTE9TRUQ7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VGbGluZ2VyU3luY2hybzo6QmFycmllcjo6d2FpdEFuZENsb3NlKG5zZWNzX3QgdGltZW91dCkgCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKGxvY2spOwotICAgIHdoaWxlIChzdGF0ZSA9PSBDTE9TRUQpIHsKLSAgICAgICAgLy8gd2UncmUgYWJvdXQgdG8gd2FpdCwgZmx1c2ggdGhlIGJpbmRlciBjb21tYW5kIGJ1ZmZlcgotICAgICAgICBJUENUaHJlYWRTdGF0ZTo6c2VsZigpLT5mbHVzaENvbW1hbmRzKCk7Ci0gICAgICAgIGludCBlcnIgPSBjdi53YWl0UmVsYXRpdmUobG9jaywgdGltZW91dCk7Ci0gICAgICAgIGlmIChlcnIgIT0gMCkKLSAgICAgICAgICAgIHJldHVybiBlcnI7Ci0gICAgfQotICAgIHN0YXRlID0gQ0xPU0VEOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLVN1cmZhY2VGbGluZ2VyU3luY2hybzo6U3VyZmFjZUZsaW5nZXJTeW5jaHJvKGNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBmbGluZ2VyKQotICAgIDogbVN1cmZhY2VDb21wb3NlcihmbGluZ2VyKQotewotfQotCi1TdXJmYWNlRmxpbmdlclN5bmNocm86OlN1cmZhY2VGbGluZ2VyU3luY2hybygpCi17Ci19Ci0KLVN1cmZhY2VGbGluZ2VyU3luY2hybzo6flN1cmZhY2VGbGluZ2VyU3luY2hybygpCi17Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VGbGluZ2VyU3luY2hybzo6c2lnbmFsKCkKLXsKLSAgICBtU3VyZmFjZUNvbXBvc2VyLT5zaWduYWwoKTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXN0YXR1c190IFN1cmZhY2VGbGluZ2VyU3luY2hybzo6d2FpdCgpCi17Ci0gICAgbUJhcnJpZXIud2FpdEFuZENsb3NlKCk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBTdXJmYWNlRmxpbmdlclN5bmNocm86OndhaXQobnNlY3NfdCB0aW1lb3V0KQotewotICAgIGlmICh0aW1lb3V0ID09IDApCi0gICAgICAgIHJldHVybiBTdXJmYWNlRmxpbmdlclN5bmNocm86OndhaXQoKTsKLSAgICByZXR1cm4gbUJhcnJpZXIud2FpdEFuZENsb3NlKHRpbWVvdXQpOwotfQotCi12b2lkIFN1cmZhY2VGbGluZ2VyU3luY2hybzo6b3BlbigpCi17Ci0gICAgbUJhcnJpZXIub3BlbigpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy91aS9UaW1lLmNwcCBiL2xpYnMvdWkvVGltZS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGI1NTM5MTMuLjAwMDAwMDAKLS0tIGEvbGlicy91aS9UaW1lLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDE5OSArMCwwIEBACi0jaW5jbHVkZSA8dXRpbHMvVGltZVV0aWxzLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxjdXRpbHMvdHp0aW1lLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotc3RhdGljIHZvaWQKLWR1bXAoY29uc3QgVGltZSYgdCkKLXsKLSAgICAjaWZkZWYgSEFWRV9UTV9HTVRPRkYKLSAgICAgICAgbG9uZyB0bV9nbXRvZmYgPSB0LnQudG1fZ210b2ZmOwotICAgICNlbHNlCi0gICAgICAgIGxvbmcgdG1fZ210b2ZmID0gMDsKLSAgICAjZW5kaWYKLSAgICBwcmludGYoIiUwNGQtJTAyZC0lMDJkICUwMmQ6JTAyZDolMDJkICglZCwlbGQsJWQsJWQpXG4iLAotICAgICAgICAgICAgdC50LnRtX3llYXIrMTkwMCwgdC50LnRtX21vbisxLCB0LnQudG1fbWRheSwKLSAgICAgICAgICAgIHQudC50bV9ob3VyLCB0LnQudG1fbWluLCB0LnQudG1fc2VjLAotICAgICAgICAgICAgdC50LnRtX2lzZHN0LCB0bV9nbXRvZmYsIHQudC50bV93ZGF5LCB0LnQudG1feWRheSk7Ci19Ci0KLVRpbWU6OlRpbWUoKQotewotICAgIHQudG1fc2VjID0gMDsKLSAgICB0LnRtX21pbiA9IDA7Ci0gICAgdC50bV9ob3VyID0gMDsKLSAgICB0LnRtX21kYXkgPSAwOwotICAgIHQudG1fbW9uID0gMDsKLSAgICB0LnRtX3llYXIgPSAwOwotICAgIHQudG1fd2RheSA9IDA7Ci0gICAgdC50bV95ZGF5ID0gMDsKLSAgICB0LnRtX2lzZHN0ID0gLTE7IC8vIHdlIGRvbid0IGtub3csIHNvIGxldCB0aGUgQyBsaWJyYXJ5IGRldGVybWluZQotICAgICNpZmRlZiBIQVZFX1RNX0dNVE9GRgotICAgICAgICB0LnRtX2dtdG9mZiA9IDA7Ci0gICAgI2VuZGlmCi19Ci0KLQotI2RlZmluZSBDT01QQVJFX0ZJRUxEKGZpZWxkKSBkbyB7IFwKLSAgICAgICAgaW50IGRpZmYgPSBhLnQuZmllbGQgLSBiLnQuZmllbGQ7IFwKLSAgICAgICAgaWYgKGRpZmYgIT0gMCkgcmV0dXJuIGRpZmY7IFwKLSAgICB9IHdoaWxlKDApCi0KLWludAotVGltZTo6Y29tcGFyZShUaW1lJiBhLCBUaW1lJiBiKQotewotICAgIGlmICgwID09IHN0cmNtcChhLnRpbWV6b25lLCBiLnRpbWV6b25lKSkgewotICAgICAgICAvLyBpZiB0aGUgdGltZXpvbmVzIGFyZSB0aGUgc2FtZSwgd2UgY2FuIGVhc2lseSBjb21wYXJlIHRoZSB0d28KLSAgICAgICAgLy8gdGltZXMuICBPdGhlcndpc2UsIGNvbnZlcnQgdG8gbWlsbGlzZWNvbmRzIGFuZCBjb21wYXJlIHRoYXQuCi0gICAgICAgIC8vIFRoaXMgcmVxdWlyZXMgdGhhdCBvYmplY3QgYmUgbm9ybWFsaXplZC4KLSAgICAgICAgQ09NUEFSRV9GSUVMRCh0bV95ZWFyKTsKLSAgICAgICAgQ09NUEFSRV9GSUVMRCh0bV9tb24pOwotICAgICAgICBDT01QQVJFX0ZJRUxEKHRtX21kYXkpOwotICAgICAgICBDT01QQVJFX0ZJRUxEKHRtX2hvdXIpOwotICAgICAgICBDT01QQVJFX0ZJRUxEKHRtX21pbik7Ci0gICAgICAgIENPTVBBUkVfRklFTEQodG1fc2VjKTsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgaW50NjRfdCBhbSA9IGEudG9NaWxsaXMoZmFsc2UgLyogdXNlIGlzRHN0ICovKTsKLSAgICAgICAgaW50NjRfdCBibSA9IGIudG9NaWxsaXMoZmFsc2UgLyogdXNlIGlzRHN0ICovKTsKLSAgICAgICAgaW50NjRfdCBkaWZmID0gYW0tYm07Ci0gICAgICAgIHJldHVybiAoZGlmZiA8IDApID8gLTEgOiAoKGRpZmYgPiAwKSA/IDEgOiAwKTsKLSAgICB9Ci19Ci0KLXN0YXRpYyBjb25zdCBpbnQgREFZU19QRVJfTU9OVEhbXSA9IHsKLSAgICAgICAgICAgICAgICAgICAgICAgIDMxLCAyOCwgMzEsIDMwLCAzMSwgMzAsIDMxLCAzMSwgMzAsIDMxLCAzMCwgMzEKLSAgICAgICAgICAgICAgICAgICAgfTsKLQotc3RhdGljIGlubGluZSBpbnQgZGF5c190aGlzX21vbnRoKGludCB5ZWFyLCBpbnQgbW9udGgpCi17Ci0gICAgaW50IG4gPSBEQVlTX1BFUl9NT05USFttb250aF07Ci0gICAgaWYgKG4gIT0gMjgpIHsKLSAgICAgICAgcmV0dXJuIG47Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgaW50IHkgPSB5ZWFyOwotICAgICAgICByZXR1cm4gKCh5JTQpPT0wJiYoKHklMTAwKSE9MHx8KHklNDAwKT09MCkpID8gMjkgOiAyODsKLSAgICB9Ci19Ci0KLXZvaWQgCi1UaW1lOjpzd2l0Y2hUaW1lem9uZShjb25zdCBjaGFyKiB0aW1lem9uZSkKLXsKLSAgICB0aW1lX3Qgc2Vjb25kcyA9IG1rdGltZV90eigmKHRoaXMtPnQpLCB0aGlzLT50aW1lem9uZSk7Ci0gICAgbG9jYWx0aW1lX3R6KCZzZWNvbmRzLCAmKHRoaXMtPnQpLCB0aW1lem9uZSk7Ci19Ci0KLVN0cmluZzggCi1UaW1lOjpmb3JtYXQoY29uc3QgY2hhciAqZm9ybWF0LCBjb25zdCBzdHJ1Y3Qgc3RyZnRpbWVfbG9jYWxlICpsb2NhbGUpIGNvbnN0Ci17Ci0gICAgY2hhciBidWZbMjU3XTsKLSAgICBpbnQgbiA9IHN0cmZ0aW1lX3R6KGJ1ZiwgMjU3LCBmb3JtYXQsICYodGhpcy0+dCksIGxvY2FsZSk7Ci0gICAgaWYgKG4gPiAwKSB7Ci0gICAgICAgIHJldHVybiBTdHJpbmc4KGJ1Zik7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgcmV0dXJuIFN0cmluZzgoKTsKLSAgICB9Ci19Ci0KLXN0YXRpYyBpbmxpbmUgc2hvcnQKLXRvY2hhcihpbnQgbikKLXsKLSAgICByZXR1cm4gKG4gPj0gMCAmJiBuIDw9IDkpID8gKCcwJytuKSA6ICcgJzsKLX0KLQotc3RhdGljIGlubGluZSBzaG9ydAotbmV4dF9jaGFyKGludCAqbSwgaW50IGspCi17Ci0gICAgaW50IG4gPSAqbSAvIGs7Ci0gICAgKm0gPSAqbSAlIGs7Ci0gICAgcmV0dXJuIHRvY2hhcihuKTsKLX0KLQotdm9pZAotVGltZTo6Zm9ybWF0MjQ0NShzaG9ydCogYnVmLCBib29sIGhhc1RpbWUpIGNvbnN0Ci17Ci0gICAgaW50IG47Ci0KLSAgICBuID0gdC50bV95ZWFyKzE5MDA7Ci0gICAgYnVmWzBdID0gbmV4dF9jaGFyKCZuLCAxMDAwKTsKLSAgICBidWZbMV0gPSBuZXh0X2NoYXIoJm4sIDEwMCk7Ci0gICAgYnVmWzJdID0gbmV4dF9jaGFyKCZuLCAxMCk7Ci0gICAgYnVmWzNdID0gdG9jaGFyKG4pOwotCi0gICAgbiA9IHQudG1fbW9uKzE7Ci0gICAgYnVmWzRdID0gbmV4dF9jaGFyKCZuLCAxMCk7Ci0gICAgYnVmWzVdID0gdG9jaGFyKG4pOwotCi0gICAgbiA9IHQudG1fbWRheTsKLSAgICBidWZbNl0gPSBuZXh0X2NoYXIoJm4sIDEwKTsKLSAgICBidWZbN10gPSB0b2NoYXIobik7Ci0KLSAgICBpZiAoaGFzVGltZSkgewotICAgICAgYnVmWzhdID0gJ1QnOwotCi0gICAgICBuID0gdC50bV9ob3VyOwotICAgICAgYnVmWzldID0gbmV4dF9jaGFyKCZuLCAxMCk7Ci0gICAgICBidWZbMTBdID0gdG9jaGFyKG4pOwotICAgICAgCi0gICAgICBuID0gdC50bV9taW47Ci0gICAgICBidWZbMTFdID0gbmV4dF9jaGFyKCZuLCAxMCk7Ci0gICAgICBidWZbMTJdID0gdG9jaGFyKG4pOwotICAgICAgCi0gICAgICBuID0gdC50bV9zZWM7Ci0gICAgICBidWZbMTNdID0gbmV4dF9jaGFyKCZuLCAxMCk7Ci0gICAgICBidWZbMTRdID0gdG9jaGFyKG4pOwotICAgICAgYm9vbCBpblV0YyA9IHN0cmNtcCgiVVRDIiwgdGltZXpvbmUpID09IDA7Ci0gICAgICBpZiAoaW5VdGMpIHsKLSAgICAgICAgICBidWZbMTVdID0gJ1onOwotICAgICAgfQotICAgIH0KLX0KLQotU3RyaW5nOCAKLVRpbWU6OnRvU3RyaW5nKCkgY29uc3QKLXsKLSAgICBTdHJpbmc4IHN0cjsKLSAgICBjaGFyKiBzID0gc3RyLmxvY2tCdWZmZXIoMTUwKTsKLSAgICAjaWZkZWYgSEFWRV9UTV9HTVRPRkYKLSAgICAgICAgbG9uZyB0bV9nbXRvZmYgPSB0LnRtX2dtdG9mZjsKLSAgICAjZWxzZQotICAgICAgICBsb25nIHRtX2dtdG9mZiA9IDA7Ci0gICAgI2VuZGlmCi0gICAgc3ByaW50ZihzLCAiJTA0ZCUwMmQlMDJkVCUwMmQlMDJkJTAyZCVzKCVkLCVkLCVsZCwlZCwlZCkiLCAKLSAgICAgICAgICAgIHQudG1feWVhcisxOTAwLCB0LnRtX21vbisxLCB0LnRtX21kYXksIHQudG1faG91ciwgdC50bV9taW4sCi0gICAgICAgICAgICB0LnRtX3NlYywgdGltZXpvbmUsIHQudG1fd2RheSwgdC50bV95ZGF5LCB0bV9nbXRvZmYsIHQudG1faXNkc3QsCi0gICAgICAgICAgICAoaW50KSgoKFRpbWUqKXRoaXMpLT50b01pbGxpcyhmYWxzZSAvKiB1c2UgaXNEc3QgKi8pLzEwMDApKTsKLSAgICBzdHIudW5sb2NrQnVmZmVyKCk7Ci0gICAgcmV0dXJuIHN0cjsKLX0KLQotdm9pZCAKLVRpbWU6OnNldFRvTm93KCkKLXsKLSAgICB0aW1lX3Qgc2Vjb25kczsKLSAgICB0aW1lKCZzZWNvbmRzKTsKLSAgICBsb2NhbHRpbWVfdHooJnNlY29uZHMsICYodGhpcy0+dCksIHRoaXMtPnRpbWV6b25lKTsKLX0KLQotaW50NjRfdCAKLVRpbWU6OnRvTWlsbGlzKGJvb2wgaWdub3JlRHN0KQotewotICAgIGlmIChpZ25vcmVEc3QpIHsKLSAgICAgICAgdGhpcy0+dC50bV9pc2RzdCA9IC0xOwotICAgIH0KLSAgICBpbnQ2NF90IHIgPSBta3RpbWVfdHooJih0aGlzLT50KSwgdGhpcy0+dGltZXpvbmUpOwotICAgIGlmIChyID09IC0xKQotICAgICAgICByZXR1cm4gLTE7Ci0gICAgcmV0dXJuIHIgKiAxMDAwOwotfQotCi12b2lkIAotVGltZTo6c2V0KGludDY0X3QgbWlsbGlzKQotewotICAgIHRpbWVfdCBzZWNvbmRzID0gbWlsbGlzIC8gMTAwMDsKLSAgICBsb2NhbHRpbWVfdHooJnNlY29uZHMsICYodGhpcy0+dCksIHRoaXMtPnRpbWV6b25lKTsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9BbmRyb2lkLm1rIGIvbGlicy91dGlscy9BbmRyb2lkLm1rCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjZGI4Y2EyLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvQW5kcm9pZC5taworKysgL2Rldi9udWxsCkBAIC0xLDE1NiArMCwwIEBACi0jIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0jCi0jIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotIyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0jIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotIwotIyAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotIwotIyBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0jIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0jIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotIyBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0jIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotCi1MT0NBTF9QQVRIOj0gJChjYWxsIG15LWRpcikKLQotIyBsaWJ1dGlscyBpcyBhIGxpdHRsZSB1bmlxdWU6IEl0J3MgYnVpbHQgdHdpY2UsIG9uY2UgZm9yIHRoZSBob3N0Ci0jIGFuZCBvbmNlIGZvciB0aGUgZGV2aWNlLgotCi1jb21tb25Tb3VyY2VzOj0gXAotCUFzc2V0LmNwcCBcCi0JQXNzZXREaXIuY3BwIFwKLQlBc3NldE1hbmFnZXIuY3BwIFwKLQlCdWZmZXJlZFRleHRPdXRwdXQuY3BwIFwKLQlDYWxsU3RhY2suY3BwIFwKLQlEZWJ1Zy5jcHAgXAotCUZpbGVNYXAuY3BwIFwKLQlSZWZCYXNlLmNwcCBcCi0JUmVzb3VyY2VUeXBlcy5jcHAgXAotCVNoYXJlZEJ1ZmZlci5jcHAgXAotCVN0YXRpYy5jcHAgXAotCVN0b3BXYXRjaC5jcHAgXAotCVN0cmluZzguY3BwIFwKLQlTdHJpbmcxNi5jcHAgXAotCVN5c3RlbUNsb2NrLmNwcCBcCi0JVGV4dE91dHB1dC5jcHAgXAotCVRocmVhZHMuY3BwIFwKLQlUaW1lclByb2JlLmNwcCBcCi0JVGltZXJzLmNwcCBcCi0JVmVjdG9ySW1wbC5jcHAgXAotICAgIFppcEZpbGVDUk8uY3BwIFwKLQlaaXBGaWxlUk8uY3BwIFwKLQlaaXBVdGlscy5jcHAgXAotCW1pc2MuY3BwIFwKLQlwb3J0ZWQuY3BwIFwKLQlMb2dTb2NrZXQuY3BwCi0KLSMKLSMgVGhlIGNwcCBmaWxlcyBsaXN0ZWQgaGVyZSBkbyBub3QgYmVsb25nIGluIHRoZSBkZXZpY2UKLSMgYnVpbGQuICBDb25zdWx0IHdpdGggdGhlIHN3ZXRsYW5kIGJlZm9yZSBldmVuIHRoaW5raW5nIGFib3V0Ci0jIHB1dHRpbmcgdGhlbSBpbiBjb21tb25Tb3VyY2VzLgotIwotIyBUaGV5J3JlIHVzZWQgYnkgdGhlIHNpbXVsYXRvciBydW50aW1lIGFuZCBieSBob3N0LXNpZGUgdG9vbHMgbGlrZQotIyBhYXB0IGFuZCB0aGUgc2ltdWxhdG9yIGZyb250LWVuZC4KLSMKLWhvc3RTb3VyY2VzOj0gXAotCUluZXRBZGRyZXNzLmNwcCBcCi0JUGlwZS5jcHAgXAotCVNvY2tldC5jcHAgXAotCVppcEVudHJ5LmNwcCBcCi0JWmlwRmlsZS5jcHAKLQotIyBGb3IgdGhlIGhvc3QKLSMgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLQotaW5jbHVkZSAkKENMRUFSX1ZBUlMpCi0KLUxPQ0FMX1NSQ19GSUxFUzo9ICQoY29tbW9uU291cmNlcykgJChob3N0U291cmNlcykKLQotaWZlcSAoJChIT1NUX09TKSxsaW51eCkKLSMgVXNlIHRoZSBmdXRleCBiYXNlZCBtdXRleCBhbmQgY29uZGl0aW9uIHZhcmlhYmxlCi0jIGltcGxlbWVudGF0aW9uIGZyb20gYW5kcm9pZC1hcm0gYmVjYXVzZSBpdCdzIHNoYXJlZCBtZW0gc2FmZQotCUxPQ0FMX1NSQ19GSUxFUyArPSBcCi0JCWZ1dGV4X3N5bmNocm8uYyBcCi0JCWV4ZWN1dGFibGVwYXRoX2xpbnV4LmNwcAotZW5kaWYKLWlmZXEgKCQoSE9TVF9PUyksZGFyd2luKQotCUxPQ0FMX1NSQ19GSUxFUyArPSBcCi0JCWV4ZWN1dGFibGVwYXRoX2Rhcndpbi5jcHAKLWVuZGlmCi0KLUxPQ0FMX01PRFVMRTo9IGxpYnV0aWxzCi0KLUxPQ0FMX0NGTEFHUyArPSAtRExJQlVUSUxTX05BVElWRT0xICQoVE9PTF9DRkxBR1MpCi1MT0NBTF9DX0lOQ0xVREVTICs9IGV4dGVybmFsL3psaWIKLQotaWZlcSAoJChIT1NUX09TKSx3aW5kb3dzKQotaWZlcSAoJChzdHJpcCAkKFVTRV9DWUdXSU4pLCksKQotIyBVbmRlciBNaW5HVywgY3R5cGUuaCBkb2Vzbid0IG5lZWQgbXVsdGktYnl0ZSBzdXBwb3J0Ci1MT0NBTF9DRkxBR1MgKz0gLURNQl9DVVJfTUFYPTEKLWVuZGlmCi1lbmRpZgotCi1pbmNsdWRlICQoQlVJTERfSE9TVF9TVEFUSUNfTElCUkFSWSkKLQotCi0KLSMgRm9yIHRoZSBkZXZpY2UKLSMgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLWluY2x1ZGUgJChDTEVBUl9WQVJTKQotCi0KLSMgd2UgaGF2ZSB0aGUgY29tbW9uIHNvdXJjZXMsIHBsdXMgc29tZSBkZXZpY2Utc3BlY2lmaWMgc3R1ZmYKLUxPQ0FMX1NSQ19GSUxFUzo9IFwKLQkkKGNvbW1vblNvdXJjZXMpIFwKLQlCaW5kZXIuY3BwIFwKLQlCcEJpbmRlci5jcHAgXAotCUlJbnRlcmZhY2UuY3BwIFwKLQlJTWVtb3J5LmNwcCBcCi0JSVBDVGhyZWFkU3RhdGUuY3BwIFwKLQlNZW1vcnlEZWFsZXIuY3BwIFwKLSAgICBNZW1vcnlCYXNlLmNwcCBcCi0gICAgTWVtb3J5SGVhcEJhc2UuY3BwIFwKLSAgICBNZW1vcnlIZWFwUG1lbS5jcHAgXAotCVBhcmNlbC5jcHAgXAotCVByb2Nlc3NTdGF0ZS5jcHAgXAotCUlQZXJtaXNzaW9uQ29udHJvbGxlci5jcHAgXAotCUlTZXJ2aWNlTWFuYWdlci5jcHAgXAotCVVuaWNvZGUuY3BwCi0KLWlmZXEgKCQoVEFSR0VUX1NJTVVMQVRPUiksdHJ1ZSkKLUxPQ0FMX1NSQ19GSUxFUyArPSAkKGhvc3RTb3VyY2VzKQotZW5kaWYKLQotaWZlcSAoJChUQVJHRVRfT1MpLGxpbnV4KQotIyBVc2UgdGhlIGZ1dGV4IGJhc2VkIG11dGV4IGFuZCBjb25kaXRpb24gdmFyaWFibGUKLSMgaW1wbGVtZW50YXRpb24gZnJvbSBhbmRyb2lkLWFybSBiZWNhdXNlIGl0J3Mgc2hhcmVkIG1lbSBzYWZlCi1MT0NBTF9TUkNfRklMRVMgKz0gZnV0ZXhfc3luY2hyby5jCi1MT0NBTF9MRExJQlMgKz0gLWxydCAtbGRsCi1lbmRpZgotCi1MT0NBTF9DX0lOQ0xVREVTICs9IFwKLQkJZXh0ZXJuYWwvemxpYiBcCi0JCWV4dGVybmFsL2ljdTRjL2NvbW1vbgotTE9DQUxfTERMSUJTICs9IC1scHRocmVhZAotCi1MT0NBTF9TSEFSRURfTElCUkFSSUVTIDo9IFwKLQlsaWJ6IFwKLQlsaWJsb2cgXAotCWxpYmN1dGlscwotCi1pZm5lcSAoJChUQVJHRVRfU0lNVUxBVE9SKSx0cnVlKQotaWZlcSAoJChUQVJHRVRfT1MpLSQoVEFSR0VUX0FSQ0gpLGxpbnV4LXg4NikKLSMgVGhpcyBpcyBuZWVkZWQgb24geDg2IHRvIGJyaW5nIGluIGRsX2l0ZXJhdGVfcGhkciBmb3IgQ2FsbFN0YWNrLmNwcAotTE9DQUxfU0hBUkVEX0xJQlJBUklFUyArPSBcCi0JbGliZGwKLWVuZGlmICMgbGludXgteDg2Ci1lbmRpZiAjIHNpbQotCi1MT0NBTF9NT0RVTEU6PSBsaWJ1dGlscwotCi0jTE9DQUxfQ0ZMQUdTKz0KLSNMT0NBTF9MREZMQUdTOj0KLQotaW5jbHVkZSAkKEJVSUxEX1NIQVJFRF9MSUJSQVJZKQotCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL0Fzc2V0LmNwcCBiL2xpYnMvdXRpbHMvQXNzZXQuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5MTIwM2RkLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvQXNzZXQuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsODEzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDYgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gUHJvdmlkZSBhY2Nlc3MgdG8gYSByZWFkLW9ubHkgYXNzZXQuCi0vLwotCi0jZGVmaW5lIExPR19UQUcgImFzc2V0IgotLy8jZGVmaW5lIE5ERUJVRyAwCi0KLSNpbmNsdWRlIDx1dGlscy9Bc3NldC5oPgotI2luY2x1ZGUgPHV0aWxzL0F0b21pYy5oPgotI2luY2x1ZGUgPHV0aWxzL0ZpbGVNYXAuaD4KLSNpbmNsdWRlIDx1dGlscy9aaXBVdGlscy5oPgotI2luY2x1ZGUgPHV0aWxzL1ppcEZpbGVSTy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8bWVtb3J5Lmg+Ci0jaW5jbHVkZSA8ZmNudGwuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPGFzc2VydC5oPgotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotI2lmbmRlZiBPX0JJTkFSWQotIyBkZWZpbmUgT19CSU5BUlkgMAotI2VuZGlmCi0KLXN0YXRpYyB2b2xhdGlsZSBpbnQzMl90IGdDb3VudCA9IDA7Ci0KLWludDMyX3QgQXNzZXQ6OmdldEdsb2JhbENvdW50KCkKLXsKLSAgICByZXR1cm4gZ0NvdW50OwotfQotCi1Bc3NldDo6QXNzZXQodm9pZCkKLSAgICA6IG1BY2Nlc3NNb2RlKEFDQ0VTU19VTktOT1dOKQotewotICAgIGludCBjb3VudCA9IGFuZHJvaWRfYXRvbWljX2luYygmZ0NvdW50KSsxOwotICAgIC8vTE9HSSgiQ3JlYXRpbmcgQXNzZXQgJXAgIyVkXG4iLCB0aGlzLCBjb3VudCk7Ci19Ci0KLUFzc2V0Ojp+QXNzZXQodm9pZCkKLXsKLSAgICBpbnQgY291bnQgPSBhbmRyb2lkX2F0b21pY19kZWMoJmdDb3VudCk7Ci0gICAgLy9MT0dJKCJEZXN0cm95aW5nIEFzc2V0IGluICVwICMlZFxuIiwgdGhpcywgY291bnQpOwotfQotCi0vKgotICogQ3JlYXRlIGEgbmV3IEFzc2V0IGZyb20gYSBmaWxlIG9uIGRpc2suICBUaGVyZSBpcyBhIGZhaXIgY2hhbmNlIHRoYXQKLSAqIHRoZSBmaWxlIGRvZXNuJ3QgYWN0dWFsbHkgZXhpc3QuCi0gKgotICogV2UgY2FuIHVzZSAibW9kZSIgdG8gZGVjaWRlIGhvdyB3ZSB3YW50IHRvIGdvIGFib3V0IGl0LgotICovCi0vKnN0YXRpYyovIEFzc2V0KiBBc3NldDo6Y3JlYXRlRnJvbUZpbGUoY29uc3QgY2hhciogZmlsZU5hbWUsIEFjY2Vzc01vZGUgbW9kZSkKLXsKLSAgICBfRmlsZUFzc2V0KiBwQXNzZXQ7Ci0gICAgc3RhdHVzX3QgcmVzdWx0OwotICAgIG9mZl90IGxlbmd0aDsKLSAgICBpbnQgZmQ7Ci0KLSAgICBmZCA9IG9wZW4oZmlsZU5hbWUsIE9fUkRPTkxZIHwgT19CSU5BUlkpOwotICAgIGlmIChmZCA8IDApCi0gICAgICAgIHJldHVybiBOVUxMOwotCi0gICAgLyoKLSAgICAgKiBVbmRlciBMaW51eCwgdGhlIGxzZWVrIGZhaWxzIGlmIHdlIGFjdHVhbGx5IG9wZW5lZCBhIGRpcmVjdG9yeS4gIFRvCi0gICAgICogYmUgY29ycmVjdCB3ZSBzaG91bGQgdGVzdCB0aGUgZmlsZSB0eXBlIGV4cGxpY2l0bHksIGJ1dCBzaW5jZSB3ZQotICAgICAqIGFsd2F5cyBvcGVuIHRoaW5ncyByZWFkLW9ubHkgaXQgZG9lc24ndCByZWFsbHkgbWF0dGVyLCBzbyB0aGVyZSdzCi0gICAgICogbm8gdmFsdWUgaW4gaW5jdXJyaW5nIHRoZSBleHRyYSBvdmVyaGVhZCBvZiBhbiBmc3RhdCgpIGNhbGwuCi0gICAgICovCi0gICAgbGVuZ3RoID0gbHNlZWsoZmQsIDAsIFNFRUtfRU5EKTsKLSAgICBpZiAobGVuZ3RoIDwgMCkgewotICAgICAgICA6OmNsb3NlKGZkKTsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotICAgICh2b2lkKSBsc2VlayhmZCwgMCwgU0VFS19TRVQpOwotCi0gICAgcEFzc2V0ID0gbmV3IF9GaWxlQXNzZXQ7Ci0gICAgcmVzdWx0ID0gcEFzc2V0LT5vcGVuQ2h1bmsoZmlsZU5hbWUsIGZkLCAwLCBsZW5ndGgpOwotICAgIGlmIChyZXN1bHQgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgZGVsZXRlIHBBc3NldDsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotCi0gICAgcEFzc2V0LT5tQWNjZXNzTW9kZSA9IG1vZGU7Ci0gICAgcmV0dXJuIHBBc3NldDsKLX0KLQotCi0vKgotICogQ3JlYXRlIGEgbmV3IEFzc2V0IGZyb20gYSBjb21wcmVzc2VkIGZpbGUgb24gZGlzay4gIFRoZXJlIGlzIGEgZmFpciBjaGFuY2UKLSAqIHRoYXQgdGhlIGZpbGUgZG9lc24ndCBhY3R1YWxseSBleGlzdC4KLSAqCi0gKiBXZSBjdXJyZW50bHkgc3VwcG9ydCBnemlwIGZpbGVzLiAgV2UgbWlnaHQgd2FudCB0byBoYW5kbGUgLmJ6MiBzb21lZGF5LgotICovCi0vKnN0YXRpYyovIEFzc2V0KiBBc3NldDo6Y3JlYXRlRnJvbUNvbXByZXNzZWRGaWxlKGNvbnN0IGNoYXIqIGZpbGVOYW1lLAotICAgIEFjY2Vzc01vZGUgbW9kZSkKLXsKLSAgICBfQ29tcHJlc3NlZEFzc2V0KiBwQXNzZXQ7Ci0gICAgc3RhdHVzX3QgcmVzdWx0OwotICAgIG9mZl90IGZpbGVMZW47Ci0gICAgYm9vbCBzY2FuUmVzdWx0OwotICAgIGxvbmcgb2Zmc2V0OwotICAgIGludCBtZXRob2Q7Ci0gICAgbG9uZyB1bmNvbXByZXNzZWRMZW4sIGNvbXByZXNzZWRMZW47Ci0gICAgaW50IGZkOwotCi0gICAgZmQgPSBvcGVuKGZpbGVOYW1lLCBPX1JET05MWSB8IE9fQklOQVJZKTsKLSAgICBpZiAoZmQgPCAwKQotICAgICAgICByZXR1cm4gTlVMTDsKLQotICAgIGZpbGVMZW4gPSBsc2VlayhmZCwgMCwgU0VFS19FTkQpOwotICAgIGlmIChmaWxlTGVuIDwgMCkgewotICAgICAgICA6OmNsb3NlKGZkKTsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotICAgICh2b2lkKSBsc2VlayhmZCwgMCwgU0VFS19TRVQpOwotCi0gICAgLyogd2FudCBidWZmZXJlZCBJL08gZm9yIHRoZSBmaWxlIHNjYW47IG11c3QgZHVwIHNvIGZjbG9zZSgpIGlzIHNhZmUgKi8KLSAgICBGSUxFKiBmcCA9IGZkb3BlbihkdXAoZmQpLCAicmIiKTsKLSAgICBpZiAoZnAgPT0gTlVMTCkgewotICAgICAgICA6OmNsb3NlKGZkKTsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotCi0gICAgdW5zaWduZWQgbG9uZyBjcmMzMjsKLSAgICBzY2FuUmVzdWx0ID0gWmlwVXRpbHM6OmV4YW1pbmVHemlwKGZwLCAmbWV0aG9kLCAmdW5jb21wcmVzc2VkTGVuLAotICAgICAgICAgICAgICAgICAgICAmY29tcHJlc3NlZExlbiwgJmNyYzMyKTsKLSAgICBvZmZzZXQgPSBmdGVsbChmcCk7Ci0gICAgZmNsb3NlKGZwKTsKLSAgICBpZiAoIXNjYW5SZXN1bHQpIHsKLSAgICAgICAgTE9HRCgiRmlsZSAnJXMnIGlzIG5vdCBpbiBnemlwIGZvcm1hdFxuIiwgZmlsZU5hbWUpOwotICAgICAgICA6OmNsb3NlKGZkKTsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotCi0gICAgcEFzc2V0ID0gbmV3IF9Db21wcmVzc2VkQXNzZXQ7Ci0gICAgcmVzdWx0ID0gcEFzc2V0LT5vcGVuQ2h1bmsoZmQsIG9mZnNldCwgbWV0aG9kLCB1bmNvbXByZXNzZWRMZW4sCi0gICAgICAgICAgICAgICAgY29tcHJlc3NlZExlbik7Ci0gICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikgewotICAgICAgICBkZWxldGUgcEFzc2V0OwotICAgICAgICByZXR1cm4gTlVMTDsKLSAgICB9Ci0KLSAgICBwQXNzZXQtPm1BY2Nlc3NNb2RlID0gbW9kZTsKLSAgICByZXR1cm4gcEFzc2V0OwotfQotCi0KLSNpZiAwCi0vKgotICogQ3JlYXRlIGEgbmV3IEFzc2V0IGZyb20gcGFydCBvZiBhbiBvcGVuIGZpbGUuCi0gKi8KLS8qc3RhdGljKi8gQXNzZXQqIEFzc2V0OjpjcmVhdGVGcm9tRmlsZVNlZ21lbnQoaW50IGZkLCBvZmZfdCBvZmZzZXQsCi0gICAgc2l6ZV90IGxlbmd0aCwgQWNjZXNzTW9kZSBtb2RlKQotewotICAgIF9GaWxlQXNzZXQqIHBBc3NldDsKLSAgICBzdGF0dXNfdCByZXN1bHQ7Ci0KLSAgICBwQXNzZXQgPSBuZXcgX0ZpbGVBc3NldDsKLSAgICByZXN1bHQgPSBwQXNzZXQtPm9wZW5DaHVuayhOVUxMLCBmZCwgb2Zmc2V0LCBsZW5ndGgpOwotICAgIGlmIChyZXN1bHQgIT0gTk9fRVJST1IpCi0gICAgICAgIHJldHVybiBOVUxMOwotCi0gICAgcEFzc2V0LT5tQWNjZXNzTW9kZSA9IG1vZGU7Ci0gICAgcmV0dXJuIHBBc3NldDsKLX0KLQotLyoKLSAqIENyZWF0ZSBhIG5ldyBBc3NldCBmcm9tIGNvbXByZXNzZWQgZGF0YSBpbiBhbiBvcGVuIGZpbGUuCi0gKi8KLS8qc3RhdGljKi8gQXNzZXQqIEFzc2V0OjpjcmVhdGVGcm9tQ29tcHJlc3NlZERhdGEoaW50IGZkLCBvZmZfdCBvZmZzZXQsCi0gICAgaW50IGNvbXByZXNzaW9uTWV0aG9kLCBzaXplX3QgdW5jb21wcmVzc2VkTGVuLCBzaXplX3QgY29tcHJlc3NlZExlbiwKLSAgICBBY2Nlc3NNb2RlIG1vZGUpCi17Ci0gICAgX0NvbXByZXNzZWRBc3NldCogcEFzc2V0OwotICAgIHN0YXR1c190IHJlc3VsdDsKLQotICAgIHBBc3NldCA9IG5ldyBfQ29tcHJlc3NlZEFzc2V0OwotICAgIHJlc3VsdCA9IHBBc3NldC0+b3BlbkNodW5rKGZkLCBvZmZzZXQsIGNvbXByZXNzaW9uTWV0aG9kLAotICAgICAgICAgICAgICAgIHVuY29tcHJlc3NlZExlbiwgY29tcHJlc3NlZExlbik7Ci0gICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0KLSAgICBwQXNzZXQtPm1BY2Nlc3NNb2RlID0gbW9kZTsKLSAgICByZXR1cm4gcEFzc2V0OwotfQotI2VuZGlmCi0KLS8qCi0gKiBDcmVhdGUgYSBuZXcgQXNzZXQgZnJvbSBhIG1lbW9yeSBtYXBwaW5nLgotICovCi0vKnN0YXRpYyovIEFzc2V0KiBBc3NldDo6Y3JlYXRlRnJvbVVuY29tcHJlc3NlZE1hcChGaWxlTWFwKiBkYXRhTWFwLAotICAgIEFjY2Vzc01vZGUgbW9kZSkKLXsKLSAgICBfRmlsZUFzc2V0KiBwQXNzZXQ7Ci0gICAgc3RhdHVzX3QgcmVzdWx0OwotCi0gICAgcEFzc2V0ID0gbmV3IF9GaWxlQXNzZXQ7Ci0gICAgcmVzdWx0ID0gcEFzc2V0LT5vcGVuQ2h1bmsoZGF0YU1hcCk7Ci0gICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0KLSAgICBwQXNzZXQtPm1BY2Nlc3NNb2RlID0gbW9kZTsKLSAgICByZXR1cm4gcEFzc2V0OwotfQotCi0vKgotICogQ3JlYXRlIGEgbmV3IEFzc2V0IGZyb20gY29tcHJlc3NlZCBkYXRhIGluIGEgbWVtb3J5IG1hcHBpbmcuCi0gKi8KLS8qc3RhdGljKi8gQXNzZXQqIEFzc2V0OjpjcmVhdGVGcm9tQ29tcHJlc3NlZE1hcChGaWxlTWFwKiBkYXRhTWFwLAotICAgIGludCBtZXRob2QsIHNpemVfdCB1bmNvbXByZXNzZWRMZW4sIEFjY2Vzc01vZGUgbW9kZSkKLXsKLSAgICBfQ29tcHJlc3NlZEFzc2V0KiBwQXNzZXQ7Ci0gICAgc3RhdHVzX3QgcmVzdWx0OwotCi0gICAgcEFzc2V0ID0gbmV3IF9Db21wcmVzc2VkQXNzZXQ7Ci0gICAgcmVzdWx0ID0gcEFzc2V0LT5vcGVuQ2h1bmsoZGF0YU1hcCwgbWV0aG9kLCB1bmNvbXByZXNzZWRMZW4pOwotICAgIGlmIChyZXN1bHQgIT0gTk9fRVJST1IpCi0gICAgICAgIHJldHVybiBOVUxMOwotCi0gICAgcEFzc2V0LT5tQWNjZXNzTW9kZSA9IG1vZGU7Ci0gICAgcmV0dXJuIHBBc3NldDsKLX0KLQotCi0vKgotICogRG8gZ2VuZXJpYyBzZWVrKCkgaG91c2VrZWVwaW5nLiAgUGFzcyBpbiB0aGUgb2Zmc2V0L3doZW5jZSB2YWx1ZXMgZnJvbQotICogdGhlIHNlZWsgcmVxdWVzdCwgYWxvbmcgd2l0aCB0aGUgY3VycmVudCBjaHVuayBvZmZzZXQgYW5kIHRoZSBjaHVuawotICogbGVuZ3RoLgotICoKLSAqIFJldHVybnMgdGhlIG5ldyBjaHVuayBvZmZzZXQsIG9yIC0xIGlmIHRoZSBzZWVrIGlzIGlsbGVnYWwuCi0gKi8KLW9mZl90IEFzc2V0OjpoYW5kbGVTZWVrKG9mZl90IG9mZnNldCwgaW50IHdoZW5jZSwgb2ZmX3QgY3VyUG9zbiwgb2ZmX3QgbWF4UG9zbikKLXsKLSAgICBvZmZfdCBuZXdPZmZzZXQ7Ci0KLSAgICBzd2l0Y2ggKHdoZW5jZSkgewotICAgIGNhc2UgU0VFS19TRVQ6Ci0gICAgICAgIG5ld09mZnNldCA9IG9mZnNldDsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBTRUVLX0NVUjoKLSAgICAgICAgbmV3T2Zmc2V0ID0gY3VyUG9zbiArIG9mZnNldDsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBTRUVLX0VORDoKLSAgICAgICAgbmV3T2Zmc2V0ID0gbWF4UG9zbiArIG9mZnNldDsKLSAgICAgICAgYnJlYWs7Ci0gICAgZGVmYXVsdDoKLSAgICAgICAgTE9HVygidW5leHBlY3RlZCB3aGVuY2UgJWRcbiIsIHdoZW5jZSk7Ci0gICAgICAgIC8vIHRoaXMgd2FzIGhhcHBlbmluZyBkdWUgdG8gYW4gb2ZmX3Qgc2l6ZSBtaXNtYXRjaAotICAgICAgICBhc3NlcnQoZmFsc2UpOwotICAgICAgICByZXR1cm4gKG9mZl90KSAtMTsKLSAgICB9Ci0KLSAgICBpZiAobmV3T2Zmc2V0IDwgMCB8fCBuZXdPZmZzZXQgPiBtYXhQb3NuKSB7Ci0gICAgICAgIExPR1coInNlZWsgb3V0IG9mIHJhbmdlOiB3YW50ICVsZCwgZW5kPSVsZFxuIiwKLSAgICAgICAgICAgIChsb25nKSBuZXdPZmZzZXQsIChsb25nKSBtYXhQb3NuKTsKLSAgICAgICAgcmV0dXJuIChvZmZfdCkgLTE7Ci0gICAgfQotCi0gICAgcmV0dXJuIG5ld09mZnNldDsKLX0KLQotCi0vKgotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKiAgICAgIF9GaWxlQXNzZXQKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICovCi0KLS8qCi0gKiBDb25zdHJ1Y3Rvci4KLSAqLwotX0ZpbGVBc3NldDo6X0ZpbGVBc3NldCh2b2lkKQotICAgIDogbVN0YXJ0KDApLCBtTGVuZ3RoKDApLCBtT2Zmc2V0KDApLCBtRnAoTlVMTCksIG1GaWxlTmFtZShOVUxMKSwgbU1hcChOVUxMKSwgbUJ1ZihOVUxMKQotewotfQotCi0vKgotICogRGVzdHJ1Y3Rvci4gIFJlbGVhc2UgcmVzb3VyY2VzLgotICovCi1fRmlsZUFzc2V0Ojp+X0ZpbGVBc3NldCh2b2lkKQotewotICAgIGNsb3NlKCk7Ci19Ci0KLS8qCi0gKiBPcGVyYXRlIG9uIGEgY2h1bmsgb2YgYW4gdW5jb21wcmVzc2VkIGZpbGUuCi0gKgotICogWmVyby1sZW5ndGggY2h1bmtzIGFyZSBhbGxvd2VkLgotICovCi1zdGF0dXNfdCBfRmlsZUFzc2V0OjpvcGVuQ2h1bmsoY29uc3QgY2hhciogZmlsZU5hbWUsIGludCBmZCwgb2ZmX3Qgb2Zmc2V0LCBzaXplX3QgbGVuZ3RoKQotewotICAgIGFzc2VydChtRnAgPT0gTlVMTCk7ICAgIC8vIG5vIHJlb3BlbgotICAgIGFzc2VydChtTWFwID09IE5VTEwpOwotICAgIGFzc2VydChmZCA+PSAwKTsKLSAgICBhc3NlcnQob2Zmc2V0ID49IDApOwotCi0gICAgLyoKLSAgICAgKiBTZWVrIHRvIGVuZCB0byBnZXQgZmlsZSBsZW5ndGguCi0gICAgICovCi0gICAgb2ZmX3QgZmlsZUxlbmd0aDsKLSAgICBmaWxlTGVuZ3RoID0gbHNlZWsoZmQsIDAsIFNFRUtfRU5EKTsKLSAgICBpZiAoZmlsZUxlbmd0aCA9PSAob2ZmX3QpIC0xKSB7Ci0gICAgICAgIC8vIHByb2JhYmx5IGEgYmFkIGZpbGUgZGVzY3JpcHRvcgotICAgICAgICBMT0dEKCJmYWlsZWQgbHNlZWsgKGVycm5vPSVkKVxuIiwgZXJybm8pOwotICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICB9Ci0KLSAgICBpZiAoKG9mZl90KSAob2Zmc2V0ICsgbGVuZ3RoKSA+IGZpbGVMZW5ndGgpIHsKLSAgICAgICAgTE9HRCgic3RhcnQgKCVsZCkgKyBsZW4gKCVsZCkgPiBlbmQgKCVsZClcbiIsCi0gICAgICAgICAgICAobG9uZykgb2Zmc2V0LCAobG9uZykgbGVuZ3RoLCAobG9uZykgZmlsZUxlbmd0aCk7Ci0gICAgICAgIHJldHVybiBCQURfSU5ERVg7Ci0gICAgfQotCi0gICAgLyogYWZ0ZXIgZmRvcGVuLCB0aGUgZmQgd2lsbCBiZSBjbG9zZWQgb24gZmNsb3NlKCkgKi8KLSAgICBtRnAgPSBmZG9wZW4oZmQsICJyYiIpOwotICAgIGlmIChtRnAgPT0gTlVMTCkKLSAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0KLSAgICBtU3RhcnQgPSBvZmZzZXQ7Ci0gICAgbUxlbmd0aCA9IGxlbmd0aDsKLSAgICBhc3NlcnQobU9mZnNldCA9PSAwKTsKLQotICAgIC8qIHNlZWsgdGhlIEZJTEUqIHRvIHRoZSBzdGFydCBvZiBjaHVuayAqLwotICAgIGlmIChmc2VlayhtRnAsIG1TdGFydCwgU0VFS19TRVQpICE9IDApIHsKLSAgICAgICAgYXNzZXJ0KGZhbHNlKTsKLSAgICB9Ci0KLSAgICBtRmlsZU5hbWUgPSBmaWxlTmFtZSAhPSBOVUxMID8gc3RyZHVwKGZpbGVOYW1lKSA6IE5VTEw7Ci0gICAgCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vKgotICogQ3JlYXRlIHRoZSBjaHVuayBmcm9tIHRoZSBtYXAuCi0gKi8KLXN0YXR1c190IF9GaWxlQXNzZXQ6Om9wZW5DaHVuayhGaWxlTWFwKiBkYXRhTWFwKQotewotICAgIGFzc2VydChtRnAgPT0gTlVMTCk7ICAgIC8vIG5vIHJlb3BlbgotICAgIGFzc2VydChtTWFwID09IE5VTEwpOwotICAgIGFzc2VydChkYXRhTWFwICE9IE5VTEwpOwotCi0gICAgbU1hcCA9IGRhdGFNYXA7Ci0gICAgbVN0YXJ0ID0gLTE7ICAgICAgICAgICAgLy8gbm90IHVzZWQKLSAgICBtTGVuZ3RoID0gZGF0YU1hcC0+Z2V0RGF0YUxlbmd0aCgpOwotICAgIGFzc2VydChtT2Zmc2V0ID09IDApOwotCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vKgotICogUmVhZCBhIGNodW5rIG9mIGRhdGEuCi0gKi8KLXNzaXplX3QgX0ZpbGVBc3NldDo6cmVhZCh2b2lkKiBidWYsIHNpemVfdCBjb3VudCkKLXsKLSAgICBzaXplX3QgbWF4TGVuOwotICAgIHNpemVfdCBhY3R1YWw7Ci0KLSAgICBhc3NlcnQobU9mZnNldCA+PSAwICYmIG1PZmZzZXQgPD0gbUxlbmd0aCk7Ci0KLSAgICBpZiAoZ2V0QWNjZXNzTW9kZSgpID09IEFDQ0VTU19CVUZGRVIpIHsKLSAgICAgICAgLyoKLSAgICAgICAgICogT24gZmlyc3QgYWNjZXNzLCByZWFkIG9yIG1hcCB0aGUgZW50aXJlIGZpbGUuICBUaGUgY2FsbGVyIGhhcwotICAgICAgICAgKiByZXF1ZXN0ZWQgYnVmZmVyIGFjY2VzcywgZWl0aGVyIGJlY2F1c2UgdGhleSdyZSBnb2luZyB0byBiZQotICAgICAgICAgKiB1c2luZyB0aGUgYnVmZmVyIG9yIGJlY2F1c2Ugd2hhdCB0aGV5J3JlIGRvaW5nIGhhcyBhcHByb3ByaWF0ZQotICAgICAgICAgKiBwZXJmb3JtYW5jZSBuZWVkcyBhbmQgYWNjZXNzIHBhdHRlcm5zLgotICAgICAgICAgKi8KLSAgICAgICAgaWYgKG1CdWYgPT0gTlVMTCkKLSAgICAgICAgICAgIGdldEJ1ZmZlcihmYWxzZSk7Ci0gICAgfQotCi0gICAgLyogYWRqdXN0IGNvdW50IGlmIHdlJ3JlIG5lYXIgRU9GICovCi0gICAgbWF4TGVuID0gbUxlbmd0aCAtIG1PZmZzZXQ7Ci0gICAgaWYgKGNvdW50ID4gbWF4TGVuKQotICAgICAgICBjb3VudCA9IG1heExlbjsKLQotICAgIGlmICghY291bnQpCi0gICAgICAgIHJldHVybiAwOwotCi0gICAgaWYgKG1NYXAgIT0gTlVMTCkgewotICAgICAgICAvKiBjb3B5IGZyb20gbWFwcGVkIGFyZWEgKi8KLSAgICAgICAgLy9wcmludGYoIm1hcCByZWFkXG4iKTsKLSAgICAgICAgbWVtY3B5KGJ1ZiwgKGNoYXIqKW1NYXAtPmdldERhdGFQdHIoKSArIG1PZmZzZXQsIGNvdW50KTsKLSAgICAgICAgYWN0dWFsID0gY291bnQ7Ci0gICAgfSBlbHNlIGlmIChtQnVmICE9IE5VTEwpIHsKLSAgICAgICAgLyogY29weSBmcm9tIGJ1ZmZlciAqLwotICAgICAgICAvL3ByaW50ZigiYnVmIHJlYWRcbiIpOwotICAgICAgICBtZW1jcHkoYnVmLCAoY2hhciopbUJ1ZiArIG1PZmZzZXQsIGNvdW50KTsKLSAgICAgICAgYWN0dWFsID0gY291bnQ7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgLyogcmVhZCBmcm9tIHRoZSBmaWxlICovCi0gICAgICAgIC8vcHJpbnRmKCJmaWxlIHJlYWRcbiIpOwotICAgICAgICBpZiAoZnRlbGwobUZwKSAhPSBtU3RhcnQgKyBtT2Zmc2V0KSB7Ci0gICAgICAgICAgICBMT0dFKCJIb3NlZDogJWxkICE9ICVsZCslbGRcbiIsCi0gICAgICAgICAgICAgICAgZnRlbGwobUZwKSwgKGxvbmcpIG1TdGFydCwgKGxvbmcpIG1PZmZzZXQpOwotICAgICAgICAgICAgYXNzZXJ0KGZhbHNlKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8qCi0gICAgICAgICAqIFRoaXMgcmV0dXJucyAwIG9uIGVycm9yIG9yIGVvZi4gIFdlIG5lZWQgdG8gdXNlIGZlcnJvcigpIG9yIGZlb2YoKQotICAgICAgICAgKiB0byB0ZWxsIHRoZSBkaWZmZXJlbmNlLCBidXQgd2UgZG9uJ3QgY3VycmVudGx5IGhhdmUgdGhvc2Ugb24gdGhlCi0gICAgICAgICAqIGRldmljZS4gIEhvd2V2ZXIsIHdlIGtub3cgaG93IG11Y2ggZGF0YSBpcyAqc3VwcG9zZWQqIHRvIGJlIGluIHRoZQotICAgICAgICAgKiBmaWxlLCBzbyBpZiB3ZSBkb24ndCByZWFkIHRoZSBmdWxsIGFtb3VudCB3ZSBrbm93IHNvbWV0aGluZyBpcwotICAgICAgICAgKiBob3NlZC4KLSAgICAgICAgICovCi0gICAgICAgIGFjdHVhbCA9IGZyZWFkKGJ1ZiwgMSwgY291bnQsIG1GcCk7Ci0gICAgICAgIGlmIChhY3R1YWwgPT0gMCkgICAgICAgIC8vIHNvbWV0aGluZyBmYWlsZWQgLS0gSS9PIGVycm9yPwotICAgICAgICAgICAgcmV0dXJuIC0xOwotCi0gICAgICAgIGFzc2VydChhY3R1YWwgPT0gY291bnQpOwotICAgIH0KLQotICAgIG1PZmZzZXQgKz0gYWN0dWFsOwotICAgIHJldHVybiBhY3R1YWw7Ci19Ci0KLS8qCi0gKiBTZWVrIHRvIGEgbmV3IHBvc2l0aW9uLgotICovCi1vZmZfdCBfRmlsZUFzc2V0OjpzZWVrKG9mZl90IG9mZnNldCwgaW50IHdoZW5jZSkKLXsKLSAgICBvZmZfdCBuZXdQb3NuOwotICAgIGxvbmcgYWN0dWFsT2Zmc2V0OwotCi0gICAgLy8gY29tcHV0ZSBuZXcgcG9zaXRpb24gd2l0aGluIGNodW5rCi0gICAgbmV3UG9zbiA9IGhhbmRsZVNlZWsob2Zmc2V0LCB3aGVuY2UsIG1PZmZzZXQsIG1MZW5ndGgpOwotICAgIGlmIChuZXdQb3NuID09IChvZmZfdCkgLTEpCi0gICAgICAgIHJldHVybiBuZXdQb3NuOwotCi0gICAgYWN0dWFsT2Zmc2V0ID0gKGxvbmcpIChtU3RhcnQgKyBuZXdQb3NuKTsKLQotICAgIGlmIChtRnAgIT0gTlVMTCkgewotICAgICAgICBpZiAoZnNlZWsobUZwLCAobG9uZykgYWN0dWFsT2Zmc2V0LCBTRUVLX1NFVCkgIT0gMCkKLSAgICAgICAgICAgIHJldHVybiAob2ZmX3QpIC0xOwotICAgIH0KLQotICAgIG1PZmZzZXQgPSBhY3R1YWxPZmZzZXQgLSBtU3RhcnQ7Ci0gICAgcmV0dXJuIG1PZmZzZXQ7Ci19Ci0KLS8qCi0gKiBDbG9zZSB0aGUgYXNzZXQuCi0gKi8KLXZvaWQgX0ZpbGVBc3NldDo6Y2xvc2Uodm9pZCkKLXsKLSAgICBpZiAobU1hcCAhPSBOVUxMKSB7Ci0gICAgICAgIG1NYXAtPnJlbGVhc2UoKTsKLSAgICAgICAgbU1hcCA9IE5VTEw7Ci0gICAgfQotICAgIGlmIChtQnVmICE9IE5VTEwpIHsKLSAgICAgICAgZGVsZXRlW10gbUJ1ZjsKLSAgICAgICAgbUJ1ZiA9IE5VTEw7Ci0gICAgfQotCi0gICAgaWYgKG1GaWxlTmFtZSAhPSBOVUxMKSB7Ci0gICAgICAgIGZyZWUobUZpbGVOYW1lKTsKLSAgICAgICAgbUZpbGVOYW1lID0gTlVMTDsKLSAgICB9Ci0gICAgCi0gICAgaWYgKG1GcCAhPSBOVUxMKSB7Ci0gICAgICAgIC8vIGNhbiBvbmx5IGJlIE5VTEwgd2hlbiBjYWxsZWQgZnJvbSBkZXN0cnVjdG9yCi0gICAgICAgIC8vIChvdGhlcndpc2Ugd2Ugd291bGQgbmV2ZXIgcmV0dXJuIHRoaXMgb2JqZWN0KQotICAgICAgICBmY2xvc2UobUZwKTsKLSAgICAgICAgbUZwID0gTlVMTDsKLSAgICB9Ci19Ci0KLS8qCi0gKiBSZXR1cm4gYSByZWFkLW9ubHkgcG9pbnRlciB0byBhIGJ1ZmZlci4KLSAqCi0gKiBXZSBjYW4gZWl0aGVyIHJlYWQgdGhlIHdob2xlIHRoaW5nIGluIG9yIG1hcCB0aGUgcmVsZXZhbnQgcGllY2Ugb2YKLSAqIHRoZSBzb3VyY2UgZmlsZS4gIElkZWFsbHkgYSBtYXAgd291bGQgYmUgZXN0YWJsaXNoZWQgYXQgYSBoaWdoZXIKLSAqIGxldmVsIGFuZCB3ZSdkIGJlIHVzaW5nIGEgZGlmZmVyZW50IG9iamVjdCwgYnV0IHdlIGRpZG4ndCwgc28gd2UKLSAqIGRlYWwgd2l0aCBpdCBoZXJlLgotICovCi1jb25zdCB2b2lkKiBfRmlsZUFzc2V0OjpnZXRCdWZmZXIoYm9vbCB3b3JkQWxpZ25lZCkKLXsKLSAgICAvKiBzdWJzZXF1ZW50IHJlcXVlc3RzIGp1c3QgdXNlIHdoYXQgd2UgZGlkIHByZXZpb3VzbHkgKi8KLSAgICBpZiAobUJ1ZiAhPSBOVUxMKQotICAgICAgICByZXR1cm4gbUJ1ZjsKLSAgICBpZiAobU1hcCAhPSBOVUxMKSB7Ci0gICAgICAgIGlmICghd29yZEFsaWduZWQpIHsKLSAgICAgICAgICAgIHJldHVybiAgbU1hcC0+Z2V0RGF0YVB0cigpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBlbnN1cmVBbGlnbm1lbnQobU1hcCk7Ci0gICAgfQotCi0gICAgYXNzZXJ0KG1GcCAhPSBOVUxMKTsKLQotICAgIGlmIChtTGVuZ3RoIDwga1JlYWRWc01hcFRocmVzaG9sZCkgewotICAgICAgICB1bnNpZ25lZCBjaGFyKiBidWY7Ci0gICAgICAgIGxvbmcgYWxsb2NMZW47Ci0KLSAgICAgICAgLyogemVyby1sZW5ndGggZmlsZXMgYXJlIGFsbG93ZWQ7IG5vdCBzdXJlIGFib3V0IHplcm8tbGVuIGFsbG9jcyAqLwotICAgICAgICAvKiAod29ya3MgZmluZSB3aXRoIGdjYyArIHg4NmxpbnV4KSAqLwotICAgICAgICBhbGxvY0xlbiA9IG1MZW5ndGg7Ci0gICAgICAgIGlmIChtTGVuZ3RoID09IDApCi0gICAgICAgICAgICBhbGxvY0xlbiA9IDE7Ci0KLSAgICAgICAgYnVmID0gbmV3IHVuc2lnbmVkIGNoYXJbYWxsb2NMZW5dOwotICAgICAgICBpZiAoYnVmID09IE5VTEwpIHsKLSAgICAgICAgICAgIExPR0UoImFsbG9jIG9mICVsZCBieXRlcyBmYWlsZWRcbiIsIChsb25nKSBhbGxvY0xlbik7Ci0gICAgICAgICAgICByZXR1cm4gTlVMTDsKLSAgICAgICAgfQotCi0gICAgICAgIExPR1YoIkFzc2V0ICVwIGFsbG9jYXRpbmcgYnVmZmVyIHNpemUgJWQgKHNtYWxsZXIgdGhhbiB0aHJlc2hvbGQpIiwgdGhpcywgKGludClhbGxvY0xlbik7Ci0gICAgICAgIGlmIChtTGVuZ3RoID4gMCkgewotICAgICAgICAgICAgbG9uZyBvbGRQb3NuID0gZnRlbGwobUZwKTsKLSAgICAgICAgICAgIGZzZWVrKG1GcCwgbVN0YXJ0LCBTRUVLX1NFVCk7Ci0gICAgICAgICAgICBpZiAoZnJlYWQoYnVmLCAxLCBtTGVuZ3RoLCBtRnApICE9IChzaXplX3QpIG1MZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBMT0dFKCJmYWlsZWQgcmVhZGluZyAlbGQgYnl0ZXNcbiIsIChsb25nKSBtTGVuZ3RoKTsKLSAgICAgICAgICAgICAgICBkZWxldGVbXSBidWY7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBmc2VlayhtRnAsIG9sZFBvc24sIFNFRUtfU0VUKTsKLSAgICAgICAgfQotCi0gICAgICAgIExPR1YoIiBnZXRCdWZmZXI6IGxvYWRlZCBpbnRvIGJ1ZmZlclxuIik7Ci0KLSAgICAgICAgbUJ1ZiA9IGJ1ZjsKLSAgICAgICAgcmV0dXJuIG1CdWY7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgRmlsZU1hcCogbWFwOwotCi0gICAgICAgIG1hcCA9IG5ldyBGaWxlTWFwOwotICAgICAgICBpZiAoIW1hcC0+Y3JlYXRlKE5VTEwsIGZpbGVubyhtRnApLCBtU3RhcnQsIG1MZW5ndGgsIHRydWUpKSB7Ci0gICAgICAgICAgICBtYXAtPnJlbGVhc2UoKTsKLSAgICAgICAgICAgIHJldHVybiBOVUxMOwotICAgICAgICB9Ci0KLSAgICAgICAgTE9HVigiIGdldEJ1ZmZlcjogbWFwcGVkXG4iKTsKLQotICAgICAgICBtTWFwID0gbWFwOwotICAgICAgICBpZiAoIXdvcmRBbGlnbmVkKSB7Ci0gICAgICAgICAgICByZXR1cm4gIG1NYXAtPmdldERhdGFQdHIoKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZW5zdXJlQWxpZ25tZW50KG1NYXApOwotICAgIH0KLX0KLQotaW50IF9GaWxlQXNzZXQ6Om9wZW5GaWxlRGVzY3JpcHRvcihvZmZfdCogb3V0U3RhcnQsIG9mZl90KiBvdXRMZW5ndGgpIGNvbnN0Ci17Ci0gICAgaWYgKG1NYXAgIT0gTlVMTCkgewotICAgICAgICBjb25zdCBjaGFyKiBmbmFtZSA9IG1NYXAtPmdldEZpbGVOYW1lKCk7Ci0gICAgICAgIGlmIChmbmFtZSA9PSBOVUxMKSB7Ci0gICAgICAgICAgICBmbmFtZSA9IG1GaWxlTmFtZTsKLSAgICAgICAgfQotICAgICAgICBpZiAoZm5hbWUgPT0gTlVMTCkgewotICAgICAgICAgICAgcmV0dXJuIC0xOwotICAgICAgICB9Ci0gICAgICAgICpvdXRTdGFydCA9IG1NYXAtPmdldERhdGFPZmZzZXQoKTsKLSAgICAgICAgKm91dExlbmd0aCA9IG1NYXAtPmdldERhdGFMZW5ndGgoKTsKLSAgICAgICAgcmV0dXJuIG9wZW4oZm5hbWUsIE9fUkRPTkxZIHwgT19CSU5BUlkpOwotICAgIH0KLSAgICBpZiAobUZpbGVOYW1lID09IE5VTEwpIHsKLSAgICAgICAgcmV0dXJuIC0xOwotICAgIH0KLSAgICAqb3V0U3RhcnQgPSBtU3RhcnQ7Ci0gICAgKm91dExlbmd0aCA9IG1MZW5ndGg7Ci0gICAgcmV0dXJuIG9wZW4obUZpbGVOYW1lLCBPX1JET05MWSB8IE9fQklOQVJZKTsKLX0KLQotY29uc3Qgdm9pZCogX0ZpbGVBc3NldDo6ZW5zdXJlQWxpZ25tZW50KEZpbGVNYXAqIG1hcCkKLXsKLSAgICB2b2lkKiBkYXRhID0gbWFwLT5nZXREYXRhUHRyKCk7Ci0gICAgaWYgKCgoKHNpemVfdClkYXRhKSYweDMpID09IDApIHsKLSAgICAgICAgLy8gV2UgY2FuIHJldHVybiB0aGlzIGRpcmVjdGx5IGlmIGl0IGlzIGFsaWduZWQgb24gYSB3b3JkCi0gICAgICAgIC8vIGJvdW5kYXJ5LgotICAgICAgICByZXR1cm4gZGF0YTsKLSAgICB9Ci0gICAgLy8gSWYgbm90IGFsaWduZWQgb24gYSB3b3JkIGJvdW5kYXJ5LCB0aGVuIHdlIG5lZWQgdG8gY29weSBpdCBpbnRvCi0gICAgLy8gb3VyIG93biBidWZmZXIuCi0gICAgTE9HVigiQ29weWluZyBGaWxlQXNzZXQgJXAgdG8gYnVmZmVyIHNpemUgJWQgdG8gbWFrZSBpdCBhbGlnbmVkLiIsIHRoaXMsIChpbnQpbUxlbmd0aCk7Ci0gICAgdW5zaWduZWQgY2hhciogYnVmID0gbmV3IHVuc2lnbmVkIGNoYXJbbUxlbmd0aF07Ci0gICAgaWYgKGJ1ZiA9PSBOVUxMKSB7Ci0gICAgICAgIExPR0UoImFsbG9jIG9mICVsZCBieXRlcyBmYWlsZWRcbiIsIChsb25nKSBtTGVuZ3RoKTsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotICAgIG1lbWNweShidWYsIGRhdGEsIG1MZW5ndGgpOwotICAgIG1CdWYgPSBidWY7Ci0gICAgcmV0dXJuIGJ1ZjsKLX0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBfQ29tcHJlc3NlZEFzc2V0Ci0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqLwotCi0vKgotICogQ29uc3RydWN0b3IuCi0gKi8KLV9Db21wcmVzc2VkQXNzZXQ6Ol9Db21wcmVzc2VkQXNzZXQodm9pZCkKLSAgICA6IG1TdGFydCgwKSwgbUNvbXByZXNzZWRMZW4oMCksIG1VbmNvbXByZXNzZWRMZW4oMCksIG1PZmZzZXQoMCksCi0gICAgICBtTWFwKE5VTEwpLCBtRmQoLTEpLCBtQnVmKE5VTEwpCi17Ci19Ci0KLS8qCi0gKiBEZXN0cnVjdG9yLiAgUmVsZWFzZSByZXNvdXJjZXMuCi0gKi8KLV9Db21wcmVzc2VkQXNzZXQ6On5fQ29tcHJlc3NlZEFzc2V0KHZvaWQpCi17Ci0gICAgY2xvc2UoKTsKLX0KLQotLyoKLSAqIE9wZW4gYSBjaHVuayBvZiBjb21wcmVzc2VkIGRhdGEgaW5zaWRlIGEgZmlsZS4KLSAqCi0gKiBUaGlzIGN1cnJlbnRseSBqdXN0IHNldHMgdXAgc29tZSB2YWx1ZXMgYW5kIHJldHVybnMuICBPbiB0aGUgZmlyc3QKLSAqIHJlYWQsIHdlIGV4cGFuZCB0aGUgZW50aXJlIGZpbGUgaW50byBhIGJ1ZmZlciBhbmQgcmV0dXJuIGRhdGEgZnJvbSBpdC4KLSAqLwotc3RhdHVzX3QgX0NvbXByZXNzZWRBc3NldDo6b3BlbkNodW5rKGludCBmZCwgb2ZmX3Qgb2Zmc2V0LAotICAgIGludCBjb21wcmVzc2lvbk1ldGhvZCwgc2l6ZV90IHVuY29tcHJlc3NlZExlbiwgc2l6ZV90IGNvbXByZXNzZWRMZW4pCi17Ci0gICAgYXNzZXJ0KG1GZCA8IDApOyAgICAgICAgLy8gbm8gcmUtb3BlbgotICAgIGFzc2VydChtTWFwID09IE5VTEwpOwotICAgIGFzc2VydChmZCA+PSAwKTsKLSAgICBhc3NlcnQob2Zmc2V0ID49IDApOwotICAgIGFzc2VydChjb21wcmVzc2VkTGVuID4gMCk7Ci0KLSAgICBpZiAoY29tcHJlc3Npb25NZXRob2QgIT0gWmlwRmlsZVJPOjprQ29tcHJlc3NEZWZsYXRlZCkgewotICAgICAgICBhc3NlcnQoZmFsc2UpOwotICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICB9Ci0KLSAgICBtU3RhcnQgPSBvZmZzZXQ7Ci0gICAgbUNvbXByZXNzZWRMZW4gPSBjb21wcmVzc2VkTGVuOwotICAgIG1VbmNvbXByZXNzZWRMZW4gPSB1bmNvbXByZXNzZWRMZW47Ci0gICAgYXNzZXJ0KG1PZmZzZXQgPT0gMCk7Ci0gICAgbUZkID0gZmQ7Ci0gICAgYXNzZXJ0KG1CdWYgPT0gTlVMTCk7Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLS8qCi0gKiBPcGVuIGEgY2h1bmsgb2YgY29tcHJlc3NlZCBkYXRhIGluIGEgbWFwcGVkIHJlZ2lvbi4KLSAqCi0gKiBOb3RoaW5nIGlzIGV4cGFuZGVkIHVudGlsIHRoZSBmaXJzdCByZWFkIGNhbGwuCi0gKi8KLXN0YXR1c190IF9Db21wcmVzc2VkQXNzZXQ6Om9wZW5DaHVuayhGaWxlTWFwKiBkYXRhTWFwLCBpbnQgY29tcHJlc3Npb25NZXRob2QsCi0gICAgc2l6ZV90IHVuY29tcHJlc3NlZExlbikKLXsKLSAgICBhc3NlcnQobUZkIDwgMCk7ICAgICAgICAvLyBubyByZS1vcGVuCi0gICAgYXNzZXJ0KG1NYXAgPT0gTlVMTCk7Ci0gICAgYXNzZXJ0KGRhdGFNYXAgIT0gTlVMTCk7Ci0KLSAgICBpZiAoY29tcHJlc3Npb25NZXRob2QgIT0gWmlwRmlsZVJPOjprQ29tcHJlc3NEZWZsYXRlZCkgewotICAgICAgICBhc3NlcnQoZmFsc2UpOwotICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICB9Ci0KLSAgICBtTWFwID0gZGF0YU1hcDsKLSAgICBtU3RhcnQgPSAtMTsgICAgICAgIC8vIG5vdCB1c2VkCi0gICAgbUNvbXByZXNzZWRMZW4gPSBkYXRhTWFwLT5nZXREYXRhTGVuZ3RoKCk7Ci0gICAgbVVuY29tcHJlc3NlZExlbiA9IHVuY29tcHJlc3NlZExlbjsKLSAgICBhc3NlcnQobU9mZnNldCA9PSAwKTsKLQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLyoKLSAqIFJlYWQgZGF0YSBmcm9tIGEgY2h1bmsgb2YgY29tcHJlc3NlZCBkYXRhLgotICoKLSAqIFtGb3Igbm93LCB0aGF0J3MganVzdCBjb3B5aW5nIGRhdGEgb3V0IG9mIGEgYnVmZmVyLl0KLSAqLwotc3NpemVfdCBfQ29tcHJlc3NlZEFzc2V0OjpyZWFkKHZvaWQqIGJ1Ziwgc2l6ZV90IGNvdW50KQotewotICAgIHNpemVfdCBtYXhMZW47Ci0gICAgc2l6ZV90IGFjdHVhbDsKLQotICAgIGFzc2VydChtT2Zmc2V0ID49IDAgJiYgbU9mZnNldCA8PSBtVW5jb21wcmVzc2VkTGVuKTsKLQotICAgIC8vIFRPRE86IGlmIG1BY2Nlc3NNb2RlID09IEFDQ0VTU19TVFJFQU1JTkcsIHVzZSB6bGliIG1vcmUgY2xldmVybHkKLQotICAgIGlmIChtQnVmID09IE5VTEwpIHsKLSAgICAgICAgaWYgKGdldEJ1ZmZlcihmYWxzZSkgPT0gTlVMTCkKLSAgICAgICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0gICAgYXNzZXJ0KG1CdWYgIT0gTlVMTCk7Ci0KLSAgICAvKiBhZGp1c3QgY291bnQgaWYgd2UncmUgbmVhciBFT0YgKi8KLSAgICBtYXhMZW4gPSBtVW5jb21wcmVzc2VkTGVuIC0gbU9mZnNldDsKLSAgICBpZiAoY291bnQgPiBtYXhMZW4pCi0gICAgICAgIGNvdW50ID0gbWF4TGVuOwotCi0gICAgaWYgKCFjb3VudCkKLSAgICAgICAgcmV0dXJuIDA7Ci0KLSAgICAvKiBjb3B5IGZyb20gYnVmZmVyICovCi0gICAgLy9wcmludGYoImNvbXAgYnVmIHJlYWRcbiIpOwotICAgIG1lbWNweShidWYsIChjaGFyKiltQnVmICsgbU9mZnNldCwgY291bnQpOwotICAgIGFjdHVhbCA9IGNvdW50OwotCi0gICAgbU9mZnNldCArPSBhY3R1YWw7Ci0gICAgcmV0dXJuIGFjdHVhbDsKLX0KLQotLyoKLSAqIEhhbmRsZSBhIHNlZWsgcmVxdWVzdC4KLSAqCi0gKiBJZiB3ZSdyZSB3b3JraW5nIGluIGEgc3RyZWFtaW5nIG1vZGUsIHRoaXMgaXMgZ29pbmcgdG8gYmUgZmFpcmx5Ci0gKiBleHBlbnNpdmUsIGJlY2F1c2UgaXQgcmVxdWlyZXMgcGxvd2luZyB0aHJvdWdoIGEgYnVuY2ggb2YgY29tcHJlc3NlZAotICogZGF0YS4KLSAqLwotb2ZmX3QgX0NvbXByZXNzZWRBc3NldDo6c2VlayhvZmZfdCBvZmZzZXQsIGludCB3aGVuY2UpCi17Ci0gICAgb2ZmX3QgbmV3UG9zbjsKLQotICAgIC8vIGNvbXB1dGUgbmV3IHBvc2l0aW9uIHdpdGhpbiBjaHVuawotICAgIG5ld1Bvc24gPSBoYW5kbGVTZWVrKG9mZnNldCwgd2hlbmNlLCBtT2Zmc2V0LCBtVW5jb21wcmVzc2VkTGVuKTsKLSAgICBpZiAobmV3UG9zbiA9PSAob2ZmX3QpIC0xKQotICAgICAgICByZXR1cm4gbmV3UG9zbjsKLQotICAgIG1PZmZzZXQgPSBuZXdQb3NuOwotICAgIHJldHVybiBtT2Zmc2V0OwotfQotCi0vKgotICogQ2xvc2UgdGhlIGFzc2V0LgotICovCi12b2lkIF9Db21wcmVzc2VkQXNzZXQ6OmNsb3NlKHZvaWQpCi17Ci0gICAgaWYgKG1NYXAgIT0gTlVMTCkgewotICAgICAgICBtTWFwLT5yZWxlYXNlKCk7Ci0gICAgICAgIG1NYXAgPSBOVUxMOwotICAgIH0KLSAgICBpZiAobUJ1ZiAhPSBOVUxMKSB7Ci0gICAgICAgIGRlbGV0ZVtdIG1CdWY7Ci0gICAgICAgIG1CdWYgPSBOVUxMOwotICAgIH0KLQotICAgIGlmIChtRmQgPiAwKSB7Ci0gICAgICAgIDo6Y2xvc2UobUZkKTsKLSAgICAgICAgbUZkID0gLTE7Ci0gICAgfQotfQotCi0vKgotICogR2V0IGEgcG9pbnRlciB0byBhIHJlYWQtb25seSBidWZmZXIgb2YgZGF0YS4KLSAqCi0gKiBUaGUgZmlyc3QgdGltZSB0aGlzIGlzIGNhbGxlZCwgd2UgZXhwYW5kIHRoZSBjb21wcmVzc2VkIGRhdGEgaW50byBhCi0gKiBidWZmZXIuCi0gKi8KLWNvbnN0IHZvaWQqIF9Db21wcmVzc2VkQXNzZXQ6OmdldEJ1ZmZlcihib29sIHdvcmRBbGlnbmVkKQotewotICAgIHVuc2lnbmVkIGNoYXIqIGJ1ZiA9IE5VTEw7Ci0KLSAgICBpZiAobUJ1ZiAhPSBOVUxMKQotICAgICAgICByZXR1cm4gbUJ1ZjsKLQotICAgIGlmIChtVW5jb21wcmVzc2VkTGVuID4gVU5DT01QUkVTU19EQVRBX01BWCkgewotICAgICAgICBMT0dEKCJEYXRhIGV4Y2VlZHMgVU5DT01QUkVTU19EQVRBX01BWCAoJWxkIHZzICVkKVxuIiwKLSAgICAgICAgICAgIChsb25nKSBtVW5jb21wcmVzc2VkTGVuLCBVTkNPTVBSRVNTX0RBVEFfTUFYKTsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIC8qCi0gICAgICogQWxsb2NhdGUgYSBidWZmZXIgYW5kIHJlYWQgdGhlIGZpbGUgaW50byBpdC4KLSAgICAgKi8KLSAgICBidWYgPSBuZXcgdW5zaWduZWQgY2hhclttVW5jb21wcmVzc2VkTGVuXTsKLSAgICBpZiAoYnVmID09IE5VTEwpIHsKLSAgICAgICAgTE9HVygiYWxsb2MgJWxkIGJ5dGVzIGZhaWxlZFxuIiwgKGxvbmcpIG1VbmNvbXByZXNzZWRMZW4pOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgaWYgKG1NYXAgIT0gTlVMTCkgewotICAgICAgICBpZiAoIVppcEZpbGVSTzo6aW5mbGF0ZUJ1ZmZlcihidWYsIG1NYXAtPmdldERhdGFQdHIoKSwKLSAgICAgICAgICAgICAgICBtVW5jb21wcmVzc2VkTGVuLCBtQ29tcHJlc3NlZExlbikpCi0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgYXNzZXJ0KG1GZCA+PSAwKTsKLQotICAgICAgICAvKgotICAgICAgICAgKiBTZWVrIHRvIHRoZSBzdGFydCBvZiB0aGUgY29tcHJlc3NlZCBkYXRhLgotICAgICAgICAgKi8KLSAgICAgICAgaWYgKGxzZWVrKG1GZCwgbVN0YXJ0LCBTRUVLX1NFVCkgIT0gbVN0YXJ0KQotICAgICAgICAgICAgZ290byBiYWlsOwotCi0gICAgICAgIC8qCi0gICAgICAgICAqIEV4cGFuZCB0aGUgZGF0YSBpbnRvIGl0LgotICAgICAgICAgKi8KLSAgICAgICAgaWYgKCFaaXBVdGlsczo6aW5mbGF0ZVRvQnVmZmVyKG1GZCwgYnVmLCBtVW5jb21wcmVzc2VkTGVuLAotICAgICAgICAgICAgICAgIG1Db21wcmVzc2VkTGVuKSkKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICB9Ci0KLSAgICAvKiBzdWNjZXNzISAqLwotICAgIG1CdWYgPSBidWY7Ci0gICAgYnVmID0gTlVMTDsKLQotYmFpbDoKLSAgICBkZWxldGVbXSBidWY7Ci0gICAgcmV0dXJuIG1CdWY7Ci19Ci0KZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvQXNzZXREaXIuY3BwIGIvbGlicy91dGlscy9Bc3NldERpci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM1ZjY2NGUuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9Bc3NldERpci5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw2NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA2IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIFByb3ZpZGUgYWNjZXNzIHRvIGEgdmlydHVhbCBkaXJlY3RvcnkgaW4gImFzc2V0IHNwYWNlIi4gIE1vc3Qgb2YgdGhlCi0vLyBpbXBsZW1lbnRhdGlvbiBpcyBpbiB0aGUgaGVhZGVyIGZpbGUgb3IgaW4gZnJpZW5kIGZ1bmN0aW9ucyBpbgotLy8gQXNzZXRNYW5hZ2VyLgotLy8KLSNpbmNsdWRlIDx1dGlscy9Bc3NldERpci5oPgotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotCi0vKgotICogRmluZCBhIG1hdGNoaW5nIGVudHJ5IGluIGEgdmVjdG9yIG9mIEZpbGVJbmZvLiAgQmVjYXVzZSBpdCdzIHNvcnRlZCwgd2UKLSAqIGNhbiB1c2UgYSBiaW5hcnkgc2VhcmNoLgotICoKLSAqIEFzc3VtZXMgdGhlIHZlY3RvciBpcyBzb3J0ZWQgaW4gYXNjZW5kaW5nIG9yZGVyLgotICovCi0vKnN0YXRpYyovIGludCBBc3NldERpcjo6RmlsZUluZm86OmZpbmRFbnRyeShjb25zdCBTb3J0ZWRWZWN0b3I8RmlsZUluZm8+KiBwVmVjdG9yLAotICAgIGNvbnN0IFN0cmluZzgmIGZpbGVOYW1lKQotewotICAgIEZpbGVJbmZvIHRtcEluZm87Ci0KLSAgICB0bXBJbmZvLnNldEZpbGVOYW1lKGZpbGVOYW1lKTsKLSAgICByZXR1cm4gcFZlY3Rvci0+aW5kZXhPZih0bXBJbmZvKTsKLQotI2lmIDAgIC8vIGRvbid0IG5lZWQgdGhpcyBhZnRlciBhbGwgKHVzZXMgMS8yIGNvbXBhcmVzIG9mIFNvcnRlZFZlY3RvciB0aG91Z2gpCi0gICAgaW50IGxvLCBoaSwgY3VyOwotCi0gICAgbG8gPSAwOwotICAgIGhpID0gcFZlY3Rvci0+c2l6ZSgpIC0xOwotICAgIHdoaWxlIChsbyA8PSBoaSkgewotICAgICAgICBpbnQgY21wOwotCi0gICAgICAgIGN1ciA9IChoaSArIGxvKSAvIDI7Ci0gICAgICAgIGNtcCA9IHN0cmNtcChwVmVjdG9yLT5pdGVtQXQoY3VyKS5nZXRGaWxlTmFtZSgpLCBmaWxlTmFtZSk7Ci0gICAgICAgIGlmIChjbXAgPT0gMCkgewotICAgICAgICAgICAgLyogbWF0Y2gsIGJhaWwgKi8KLSAgICAgICAgICAgIHJldHVybiBjdXI7Ci0gICAgICAgIH0gZWxzZSBpZiAoY21wIDwgMCkgewotICAgICAgICAgICAgLyogdG9vIGxvdyAqLwotICAgICAgICAgICAgbG8gPSBjdXIgKyAxOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLyogdG9vIGhpZ2ggKi8KLSAgICAgICAgICAgIGhpID0gY3VyIC0xOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcmV0dXJuIC0xOwotI2VuZGlmCi19Ci0KZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvQXNzZXRNYW5hZ2VyLmNwcCBiL2xpYnMvdXRpbHMvQXNzZXRNYW5hZ2VyLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDQ3YjgwMS4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL0Fzc2V0TWFuYWdlci5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxNjM3ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDYgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gUHJvdmlkZSBhY2Nlc3MgdG8gcmVhZC1vbmx5IGFzc2V0cy4KLS8vCi0KLSNkZWZpbmUgTE9HX1RBRyAiYXNzZXQiCi0vLyNkZWZpbmUgTE9HX05ERUJVRyAwCi0KLSNpbmNsdWRlIDx1dGlscy9Bc3NldE1hbmFnZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9Bc3NldERpci5oPgotI2luY2x1ZGUgPHV0aWxzL0Fzc2V0Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvQXRvbWljLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RyaW5nOC5oPgotI2luY2x1ZGUgPHV0aWxzL1Jlc291cmNlVHlwZXMuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvWmlwRmlsZVJPLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvVGltZXJzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotCi0jaW5jbHVkZSA8ZGlyZW50Lmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxhc3NlcnQuaD4KLQotdXNpbmcgbmFtZXNwYWNlIGFuZHJvaWQ7Ci0KLS8qCi0gKiBOYW1lcyBmb3IgZGVmYXVsdCBhcHAsIGxvY2FsZSwgYW5kIHZlbmRvci4gIFdlIG1pZ2h0IHdhbnQgdG8gY2hhbmdlCi0gKiB0aGVzZSB0byBiZSBhbiBhY3R1YWwgbG9jYWxlLCBlLmcuIGFsd2F5cyB1c2UgZW4tVVMgYXMgdGhlIGRlZmF1bHQuCi0gKi8KLXN0YXRpYyBjb25zdCBjaGFyKiBrRGVmYXVsdExvY2FsZSA9ICJkZWZhdWx0IjsKLXN0YXRpYyBjb25zdCBjaGFyKiBrRGVmYXVsdFZlbmRvciA9ICJkZWZhdWx0IjsKLXN0YXRpYyBjb25zdCBjaGFyKiBrQXNzZXRzUm9vdCA9ICJhc3NldHMiOwotc3RhdGljIGNvbnN0IGNoYXIqIGtBcHBaaXBOYW1lID0gTlVMTDsgLy8iY2xhc3Nlcy5qYXIiOwotc3RhdGljIGNvbnN0IGNoYXIqIGtTeXN0ZW1Bc3NldHMgPSAiZnJhbWV3b3JrL2ZyYW1ld29yay1yZXMuYXBrIjsKLQotc3RhdGljIGNvbnN0IGNoYXIqIGtFeGNsdWRlRXh0ZW5zaW9uID0gIi5FWENMVURFIjsKLQotc3RhdGljIEFzc2V0KiBjb25zdCBrRXhjbHVkZWRBc3NldCA9IChBc3NldCopIDB4ZDAwMDAwMGQ7Ci0KLXN0YXRpYyB2b2xhdGlsZSBpbnQzMl90IGdDb3VudCA9IDA7Ci0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBBc3NldE1hbmFnZXIKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICovCi0KLWludDMyX3QgQXNzZXRNYW5hZ2VyOjpnZXRHbG9iYWxDb3VudCgpCi17Ci0gICAgcmV0dXJuIGdDb3VudDsKLX0KLQotQXNzZXRNYW5hZ2VyOjpBc3NldE1hbmFnZXIoQ2FjaGVNb2RlIGNhY2hlTW9kZSkKLSAgICA6IG1Mb2NhbGUoTlVMTCksIG1WZW5kb3IoTlVMTCksCi0gICAgICBtUmVzb3VyY2VzKE5VTEwpLCBtQ29uZmlnKG5ldyBSZXNUYWJsZV9jb25maWcpLAotICAgICAgbUNhY2hlTW9kZShjYWNoZU1vZGUpLCBtQ2FjaGVWYWxpZChmYWxzZSkKLXsKLSAgICBpbnQgY291bnQgPSBhbmRyb2lkX2F0b21pY19pbmMoJmdDb3VudCkrMTsKLSAgICAvL0xPR0koIkNyZWF0aW5nIEFzc2V0TWFuYWdlciAlcCAjJWRcbiIsIHRoaXMsIGNvdW50KTsKLSAgICBtZW1zZXQobUNvbmZpZywgMCwgc2l6ZW9mKFJlc1RhYmxlX2NvbmZpZykpOwotfQotCi1Bc3NldE1hbmFnZXI6On5Bc3NldE1hbmFnZXIodm9pZCkKLXsKLSAgICBpbnQgY291bnQgPSBhbmRyb2lkX2F0b21pY19kZWMoJmdDb3VudCk7Ci0gICAgLy9MT0dJKCJEZXN0cm95aW5nIEFzc2V0TWFuYWdlciBpbiAlcCAjJWRcbiIsIHRoaXMsIGNvdW50KTsKLQotICAgIGRlbGV0ZSBtQ29uZmlnOwotICAgIGRlbGV0ZSBtUmVzb3VyY2VzOwotCi0gICAgLy8gZG9uJ3QgaGF2ZSBhIFN0cmluZyBjbGFzcyB5ZXQsIHNvIG1ha2Ugc3VyZSB3ZSBjbGVhbiB1cAotICAgIGRlbGV0ZVtdIG1Mb2NhbGU7Ci0gICAgZGVsZXRlW10gbVZlbmRvcjsKLX0KLQotYm9vbCBBc3NldE1hbmFnZXI6OmFkZEFzc2V0UGF0aChjb25zdCBTdHJpbmc4JiBwYXRoLCB2b2lkKiogY29va2llKQotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0KLSAgICBhc3NldF9wYXRoIGFwOwotCi0gICAgU3RyaW5nOCByZWFsUGF0aChwYXRoKTsKLSAgICBpZiAoa0FwcFppcE5hbWUpIHsKLSAgICAgICAgcmVhbFBhdGguYXBwZW5kUGF0aChrQXBwWmlwTmFtZSk7Ci0gICAgfQotICAgIGFwLnR5cGUgPSA6OmdldEZpbGVUeXBlKHJlYWxQYXRoLnN0cmluZygpKTsKLSAgICBpZiAoYXAudHlwZSA9PSBrRmlsZVR5cGVSZWd1bGFyKSB7Ci0gICAgICAgIGFwLnBhdGggPSByZWFsUGF0aDsKLSAgICB9IGVsc2UgewotICAgICAgICBhcC5wYXRoID0gcGF0aDsKLSAgICAgICAgYXAudHlwZSA9IDo6Z2V0RmlsZVR5cGUocGF0aC5zdHJpbmcoKSk7Ci0gICAgICAgIGlmIChhcC50eXBlICE9IGtGaWxlVHlwZURpcmVjdG9yeSAmJiBhcC50eXBlICE9IGtGaWxlVHlwZVJlZ3VsYXIpIHsKLSAgICAgICAgICAgIExPR1coIkFzc2V0IHBhdGggJXMgaXMgbmVpdGhlciBhIGRpcmVjdG9yeSBub3IgZmlsZSAodHlwZT0lZCkuIiwKLSAgICAgICAgICAgICAgICAgcGF0aC5zdHJpbmcoKSwgKGludClhcC50eXBlKTsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vIFNraXAgaWYgd2UgaGF2ZSBpdCBhbHJlYWR5LgotICAgIGZvciAoc2l6ZV90IGk9MDsgaTxtQXNzZXRQYXRocy5zaXplKCk7IGkrKykgewotICAgICAgICBpZiAobUFzc2V0UGF0aHNbaV0ucGF0aCA9PSBhcC5wYXRoKSB7Ci0gICAgICAgICAgICBpZiAoY29va2llKSB7Ci0gICAgICAgICAgICAgICAgKmNvb2tpZSA9ICh2b2lkKikoaSsxKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIExPR1YoIkluICVwIEFzc2V0ICVzIHBhdGg6ICVzIiwgdGhpcywKLSAgICAgICAgIGFwLnR5cGUgPT0ga0ZpbGVUeXBlRGlyZWN0b3J5ID8gImRpciIgOiAiemlwIiwgYXAucGF0aC5zdHJpbmcoKSk7Ci0KLSAgICBtQXNzZXRQYXRocy5hZGQoYXApOwotCi0gICAgLy8gbmV3IHBhdGhzIGFyZSBhbHdheXMgYWRkZWQgYXQgdGhlIGVuZAotICAgIGlmIChjb29raWUpIHsKLSAgICAgICAgKmNvb2tpZSA9ICh2b2lkKiltQXNzZXRQYXRocy5zaXplKCk7Ci0gICAgfQotCi0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLWJvb2wgQXNzZXRNYW5hZ2VyOjphZGREZWZhdWx0QXNzZXRzKCkKLXsKLSAgICBjb25zdCBjaGFyKiByb290ID0gZ2V0ZW52KCJBTkRST0lEX1JPT1QiKTsKLSAgICBMT0dfQUxXQVlTX0ZBVEFMX0lGKHJvb3QgPT0gTlVMTCwgIkFORFJPSURfUk9PVCBub3Qgc2V0Iik7Ci0KLSAgICBTdHJpbmc4IHBhdGgocm9vdCk7Ci0gICAgcGF0aC5hcHBlbmRQYXRoKGtTeXN0ZW1Bc3NldHMpOwotCi0gICAgcmV0dXJuIGFkZEFzc2V0UGF0aChwYXRoLCBOVUxMKTsKLX0KLQotdm9pZCogQXNzZXRNYW5hZ2VyOjpuZXh0QXNzZXRQYXRoKHZvaWQqIGNvb2tpZSkgY29uc3QKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgIHNpemVfdCBuZXh0ID0gKChzaXplX3QpY29va2llKSsxOwotICAgIHJldHVybiBuZXh0ID4gbUFzc2V0UGF0aHMuc2l6ZSgpID8gTlVMTCA6ICh2b2lkKiluZXh0OwotfQotCi1TdHJpbmc4IEFzc2V0TWFuYWdlcjo6Z2V0QXNzZXRQYXRoKHZvaWQqIGNvb2tpZSkgY29uc3QKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgIGNvbnN0IHNpemVfdCB3aGljaCA9ICgoc2l6ZV90KWNvb2tpZSktMTsKLSAgICBpZiAod2hpY2ggPCBtQXNzZXRQYXRocy5zaXplKCkpIHsKLSAgICAgICAgcmV0dXJuIG1Bc3NldFBhdGhzW3doaWNoXS5wYXRoOwotICAgIH0KLSAgICByZXR1cm4gU3RyaW5nOCgpOwotfQotCi0vKgotICogU2V0IHRoZSBjdXJyZW50IGxvY2FsZS4gIFVzZSBOVUxMIHRvIGluZGljYXRlIG5vIGxvY2FsZS4KLSAqCi0gKiBDbG9zZSBhbmQgcmVvcGVuIFppcCBhcmNoaXZlcyBhcyBhcHByb3ByaWF0ZSwgYW5kIHJlc2V0IGNhY2hlZAotICogaW5mb3JtYXRpb24gaW4gdGhlIGxvY2FsZS1zcGVjaWZpYyBzZWN0aW9ucyBvZiB0aGUgdHJlZS4KLSAqLwotdm9pZCBBc3NldE1hbmFnZXI6OnNldExvY2FsZShjb25zdCBjaGFyKiBsb2NhbGUpCi17Ci0gICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLSAgICBzZXRMb2NhbGVMb2NrZWQobG9jYWxlKTsKLX0KLQotdm9pZCBBc3NldE1hbmFnZXI6OnNldExvY2FsZUxvY2tlZChjb25zdCBjaGFyKiBsb2NhbGUpCi17Ci0gICAgaWYgKG1Mb2NhbGUgIT0gTlVMTCkgewotICAgICAgICAvKiBwcmV2aW91c2x5IHNldCwgcHVyZ2UgY2FjaGVkIGRhdGEgKi8KLSAgICAgICAgcHVyZ2VGaWxlTmFtZUNhY2hlTG9ja2VkKCk7Ci0gICAgICAgIC8vbVppcFNldC5wdXJnZUxvY2FsZSgpOwotICAgICAgICBkZWxldGVbXSBtTG9jYWxlOwotICAgIH0KLSAgICBtTG9jYWxlID0gc3RyZHVwTmV3KGxvY2FsZSk7Ci0gICAgCi0gICAgdXBkYXRlUmVzb3VyY2VQYXJhbXNMb2NrZWQoKTsKLX0KLQotLyoKLSAqIFNldCB0aGUgY3VycmVudCB2ZW5kb3IuICBVc2UgTlVMTCB0byBpbmRpY2F0ZSBubyB2ZW5kb3IuCi0gKgotICogQ2xvc2UgYW5kIHJlb3BlbiBaaXAgYXJjaGl2ZXMgYXMgYXBwcm9wcmlhdGUsIGFuZCByZXNldCBjYWNoZWQKLSAqIGluZm9ybWF0aW9uIGluIHRoZSB2ZW5kb3Itc3BlY2lmaWMgc2VjdGlvbnMgb2YgdGhlIHRyZWUuCi0gKi8KLXZvaWQgQXNzZXRNYW5hZ2VyOjpzZXRWZW5kb3IoY29uc3QgY2hhciogdmVuZG9yKQotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0KLSAgICBpZiAobVZlbmRvciAhPSBOVUxMKSB7Ci0gICAgICAgIC8qIHByZXZpb3VzbHkgc2V0LCBwdXJnZSBjYWNoZWQgZGF0YSAqLwotICAgICAgICBwdXJnZUZpbGVOYW1lQ2FjaGVMb2NrZWQoKTsKLSAgICAgICAgLy9tWmlwU2V0LnB1cmdlVmVuZG9yKCk7Ci0gICAgICAgIGRlbGV0ZVtdIG1WZW5kb3I7Ci0gICAgfQotICAgIG1WZW5kb3IgPSBzdHJkdXBOZXcodmVuZG9yKTsKLX0KLQotdm9pZCBBc3NldE1hbmFnZXI6OnNldENvbmZpZ3VyYXRpb24oY29uc3QgUmVzVGFibGVfY29uZmlnJiBjb25maWcsIGNvbnN0IGNoYXIqIGxvY2FsZSkKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgICptQ29uZmlnID0gY29uZmlnOwotICAgIGlmIChsb2NhbGUpIHsKLSAgICAgICAgc2V0TG9jYWxlTG9ja2VkKGxvY2FsZSk7Ci0gICAgfSBlbHNlIGlmIChjb25maWcubGFuZ3VhZ2VbMF0gIT0gMCkgewotICAgICAgICBjaGFyIHNwZWNbOV07Ci0gICAgICAgIHNwZWNbMF0gPSBjb25maWcubGFuZ3VhZ2VbMF07Ci0gICAgICAgIHNwZWNbMV0gPSBjb25maWcubGFuZ3VhZ2VbMV07Ci0gICAgICAgIGlmIChjb25maWcuY291bnRyeVswXSAhPSAwKSB7Ci0gICAgICAgICAgICBzcGVjWzJdID0gJ18nOwotICAgICAgICAgICAgc3BlY1szXSA9IGNvbmZpZy5jb3VudHJ5WzBdOwotICAgICAgICAgICAgc3BlY1s0XSA9IGNvbmZpZy5jb3VudHJ5WzFdOwotICAgICAgICAgICAgc3BlY1s1XSA9IDA7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBzcGVjWzNdID0gMDsKLSAgICAgICAgfQotICAgICAgICBzZXRMb2NhbGVMb2NrZWQoc3BlYyk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgdXBkYXRlUmVzb3VyY2VQYXJhbXNMb2NrZWQoKTsKLSAgICB9Ci19Ci0KLS8qCi0gKiBPcGVuIGFuIGFzc2V0LgotICoKLSAqIFRoZSBkYXRhIGNvdWxkIGJlOwotICogIC0gSW4gYSBmaWxlIG9uIGRpc2sgKGFzc2V0QmFzZSArIGZpbGVOYW1lKS4KLSAqICAtIEluIGEgY29tcHJlc3NlZCBmaWxlIG9uIGRpc2sgKGFzc2V0QmFzZSArIGZpbGVOYW1lLmd6KS4KLSAqICAtIEluIGEgWmlwIGFyY2hpdmUsIHVuY29tcHJlc3NlZCBvciBjb21wcmVzc2VkLgotICoKLSAqIEl0IGNhbiBiZSBpbiBhIG51bWJlciBvZiBkaWZmZXJlbnQgZGlyZWN0b3JpZXMgYW5kIFppcCBhcmNoaXZlcy4KLSAqIFRoZSBzZWFyY2ggb3JkZXIgaXM6Ci0gKiAgLSBbYXBwbmFtZV0KLSAqICAgIC0gbG9jYWxlICsgdmVuZG9yCi0gKiAgICAtICJkZWZhdWx0IiArIHZlbmRvcgotICogICAgLSBsb2NhbGUgKyAiZGVmYXVsdCIKLSAqICAgIC0gImRlZmF1bHQgKyAiZGVmYXVsdCIKLSAqICAtICJjb21tb24iCi0gKiAgICAtIChzYW1lIGFzIGFib3ZlKQotICoKLSAqIFRvIGZpbmQgYSBwYXJ0aWN1bGFyIGZpbGUsIHdlIGhhdmUgdG8gdHJ5IHVwIHRvIGVpZ2h0IHBhdGhzIHdpdGgKLSAqIGFsbCB0aHJlZSBmb3JtcyBvZiBkYXRhLgotICoKLSAqIFdlIHNob3VsZCBwcm9iYWJseSByZWplY3QgcmVxdWVzdHMgZm9yICJpbGxlZ2FsIiBmaWxlbmFtZXMsIGUuZy4gdGhvc2UKLSAqIHdpdGggaWxsZWdhbCBjaGFyYWN0ZXJzIG9yICIuLi8iIGJhY2t3YXJkIHJlbGF0aXZlIHBhdGhzLgotICovCi1Bc3NldCogQXNzZXRNYW5hZ2VyOjpvcGVuKGNvbnN0IGNoYXIqIGZpbGVOYW1lLCBBY2Nlc3NNb2RlIG1vZGUpCi17Ci0gICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLQotICAgIExPR19GQVRBTF9JRihtQXNzZXRQYXRocy5zaXplKCkgPT0gMCwgIk5vIGFzc2V0cyBhZGRlZCB0byBBc3NldE1hbmFnZXIiKTsKLQotCi0gICAgaWYgKG1DYWNoZU1vZGUgIT0gQ0FDSEVfT0ZGICYmICFtQ2FjaGVWYWxpZCkKLSAgICAgICAgbG9hZEZpbGVOYW1lQ2FjaGVMb2NrZWQoKTsKLQotICAgIFN0cmluZzggYXNzZXROYW1lKGtBc3NldHNSb290KTsKLSAgICBhc3NldE5hbWUuYXBwZW5kUGF0aChmaWxlTmFtZSk7Ci0KLSAgICAvKgotICAgICAqIEZvciBlYWNoIHRvcC1sZXZlbCBhc3NldCBwYXRoLCBzZWFyY2ggZm9yIHRoZSBhc3NldC4KLSAgICAgKi8KLQotICAgIHNpemVfdCBpID0gbUFzc2V0UGF0aHMuc2l6ZSgpOwotICAgIHdoaWxlIChpID4gMCkgewotICAgICAgICBpLS07Ci0gICAgICAgIExPR1YoIkxvb2tpbmcgZm9yIGFzc2V0ICclcycgaW4gJyVzJ1xuIiwKLSAgICAgICAgICAgICAgICBhc3NldE5hbWUuc3RyaW5nKCksIG1Bc3NldFBhdGhzLml0ZW1BdChpKS5wYXRoLnN0cmluZygpKTsKLSAgICAgICAgQXNzZXQqIHBBc3NldCA9IG9wZW5Ob25Bc3NldEluUGF0aExvY2tlZChhc3NldE5hbWUuc3RyaW5nKCksIG1vZGUsIG1Bc3NldFBhdGhzLml0ZW1BdChpKSk7Ci0gICAgICAgIGlmIChwQXNzZXQgIT0gTlVMTCkgewotICAgICAgICAgICAgcmV0dXJuIHBBc3NldCAhPSBrRXhjbHVkZWRBc3NldCA/IHBBc3NldCA6IE5VTEw7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gTlVMTDsKLX0KLQotLyoKLSAqIE9wZW4gYSBub24tYXNzZXQgZmlsZSBhcyBpZiBpdCB3ZXJlIGFuIGFzc2V0LgotICoKLSAqIFRoZSAiZmlsZU5hbWUiIGlzIHRoZSBwYXJ0aWFsIHBhdGggc3RhcnRpbmcgZnJvbSB0aGUgYXBwbGljYXRpb24KLSAqIG5hbWUuCi0gKi8KLUFzc2V0KiBBc3NldE1hbmFnZXI6Om9wZW5Ob25Bc3NldChjb25zdCBjaGFyKiBmaWxlTmFtZSwgQWNjZXNzTW9kZSBtb2RlKQotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0KLSAgICBMT0dfRkFUQUxfSUYobUFzc2V0UGF0aHMuc2l6ZSgpID09IDAsICJObyBhc3NldHMgYWRkZWQgdG8gQXNzZXRNYW5hZ2VyIik7Ci0KLQotICAgIGlmIChtQ2FjaGVNb2RlICE9IENBQ0hFX09GRiAmJiAhbUNhY2hlVmFsaWQpCi0gICAgICAgIGxvYWRGaWxlTmFtZUNhY2hlTG9ja2VkKCk7Ci0KLSAgICAvKgotICAgICAqIEZvciBlYWNoIHRvcC1sZXZlbCBhc3NldCBwYXRoLCBzZWFyY2ggZm9yIHRoZSBhc3NldC4KLSAgICAgKi8KLQotICAgIHNpemVfdCBpID0gbUFzc2V0UGF0aHMuc2l6ZSgpOwotICAgIHdoaWxlIChpID4gMCkgewotICAgICAgICBpLS07Ci0gICAgICAgIExPR1YoIkxvb2tpbmcgZm9yIG5vbi1hc3NldCAnJXMnIGluICclcydcbiIsIGZpbGVOYW1lLCBtQXNzZXRQYXRocy5pdGVtQXQoaSkucGF0aC5zdHJpbmcoKSk7Ci0gICAgICAgIEFzc2V0KiBwQXNzZXQgPSBvcGVuTm9uQXNzZXRJblBhdGhMb2NrZWQoCi0gICAgICAgICAgICBmaWxlTmFtZSwgbW9kZSwgbUFzc2V0UGF0aHMuaXRlbUF0KGkpKTsKLSAgICAgICAgaWYgKHBBc3NldCAhPSBOVUxMKSB7Ci0gICAgICAgICAgICByZXR1cm4gcEFzc2V0ICE9IGtFeGNsdWRlZEFzc2V0ID8gcEFzc2V0IDogTlVMTDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBOVUxMOwotfQotCi1Bc3NldCogQXNzZXRNYW5hZ2VyOjpvcGVuTm9uQXNzZXQodm9pZCogY29va2llLCBjb25zdCBjaGFyKiBmaWxlTmFtZSwgQWNjZXNzTW9kZSBtb2RlKQotewotICAgIGNvbnN0IHNpemVfdCB3aGljaCA9ICgoc2l6ZV90KWNvb2tpZSktMTsKLQotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0KLSAgICBMT0dfRkFUQUxfSUYobUFzc2V0UGF0aHMuc2l6ZSgpID09IDAsICJObyBhc3NldHMgYWRkZWQgdG8gQXNzZXRNYW5hZ2VyIik7Ci0KLQotICAgIGlmIChtQ2FjaGVNb2RlICE9IENBQ0hFX09GRiAmJiAhbUNhY2hlVmFsaWQpCi0gICAgICAgIGxvYWRGaWxlTmFtZUNhY2hlTG9ja2VkKCk7Ci0KLSAgICBpZiAod2hpY2ggPCBtQXNzZXRQYXRocy5zaXplKCkpIHsKLSAgICAgICAgTE9HVigiTG9va2luZyBmb3Igbm9uLWFzc2V0ICclcycgaW4gJyVzJ1xuIiwgZmlsZU5hbWUsCi0gICAgICAgICAgICAgICAgbUFzc2V0UGF0aHMuaXRlbUF0KHdoaWNoKS5wYXRoLnN0cmluZygpKTsKLSAgICAgICAgQXNzZXQqIHBBc3NldCA9IG9wZW5Ob25Bc3NldEluUGF0aExvY2tlZCgKLSAgICAgICAgICAgIGZpbGVOYW1lLCBtb2RlLCBtQXNzZXRQYXRocy5pdGVtQXQod2hpY2gpKTsKLSAgICAgICAgaWYgKHBBc3NldCAhPSBOVUxMKSB7Ci0gICAgICAgICAgICByZXR1cm4gcEFzc2V0ICE9IGtFeGNsdWRlZEFzc2V0ID8gcEFzc2V0IDogTlVMTDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBOVUxMOwotfQotCi0vKgotICogR2V0IHRoZSB0eXBlIG9mIGEgZmlsZSBpbiB0aGUgYXNzZXQgbmFtZXNwYWNlLgotICoKLSAqIFRoaXMgY3VycmVudGx5IG9ubHkgd29ya3MgZm9yIHJlZ3VsYXIgZmlsZXMuICBBbGwgb3RoZXJzIChpbmNsdWRpbmcKLSAqIGRpcmVjdG9yaWVzKSB3aWxsIHJldHVybiBrRmlsZVR5cGVOb25leGlzdGVudC4KLSAqLwotRmlsZVR5cGUgQXNzZXRNYW5hZ2VyOjpnZXRGaWxlVHlwZShjb25zdCBjaGFyKiBmaWxlTmFtZSkKLXsKLSAgICBBc3NldCogcEFzc2V0ID0gTlVMTDsKLQotICAgIC8qCi0gICAgICogT3BlbiB0aGUgYXNzZXQuICBUaGlzIGlzIGxlc3MgZWZmaWNpZW50IHRoYW4gc2ltcGx5IGZpbmRpbmcgdGhlCi0gICAgICogZmlsZSwgYnV0IGl0J3Mgbm90IHRvbyBiYWQgKHdlIGRvbid0IHVuY29tcHJlc3Mgb3IgbW1hcCBkYXRhIHVudGlsCi0gICAgICogdGhlIGZpcnN0IHJlYWQoKSBjYWxsKS4KLSAgICAgKi8KLSAgICBwQXNzZXQgPSBvcGVuKGZpbGVOYW1lLCBBc3NldDo6QUNDRVNTX1NUUkVBTUlORyk7Ci0gICAgZGVsZXRlIHBBc3NldDsKLQotICAgIGlmIChwQXNzZXQgPT0gTlVMTCkKLSAgICAgICAgcmV0dXJuIGtGaWxlVHlwZU5vbmV4aXN0ZW50OwotICAgIGVsc2UKLSAgICAgICAgcmV0dXJuIGtGaWxlVHlwZVJlZ3VsYXI7Ci19Ci0KLWNvbnN0IFJlc1RhYmxlKiBBc3NldE1hbmFnZXI6OmdldFJlc1RhYmxlKGJvb2wgcmVxdWlyZWQpIGNvbnN0Ci17Ci0gICAgUmVzVGFibGUqIHJ0ID0gbVJlc291cmNlczsKLSAgICBpZiAocnQpIHsKLSAgICAgICAgcmV0dXJuIHJ0OwotICAgIH0KLQotICAgIC8vIEl0ZXJhdGUgdGhyb3VnaCBhbGwgYXNzZXQgcGFja2FnZXMsIGNvbGxlY3RpbmcgcmVzb3VyY2VzIGZyb20gZWFjaC4KLQotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0KLSAgICBpZiAobVJlc291cmNlcyAhPSBOVUxMKSB7Ci0gICAgICAgIHJldHVybiBtUmVzb3VyY2VzOwotICAgIH0KLQotICAgIGlmIChyZXF1aXJlZCkgewotICAgICAgICBMT0dfRkFUQUxfSUYobUFzc2V0UGF0aHMuc2l6ZSgpID09IDAsICJObyBhc3NldHMgYWRkZWQgdG8gQXNzZXRNYW5hZ2VyIik7Ci0gICAgfQotCi0gICAgaWYgKG1DYWNoZU1vZGUgIT0gQ0FDSEVfT0ZGICYmICFtQ2FjaGVWYWxpZCkKLSAgICAgICAgY29uc3RfY2FzdDxBc3NldE1hbmFnZXIqPih0aGlzKS0+bG9hZEZpbGVOYW1lQ2FjaGVMb2NrZWQoKTsKLQotICAgIGNvbnN0IHNpemVfdCBOID0gbUFzc2V0UGF0aHMuc2l6ZSgpOwotICAgIGZvciAoc2l6ZV90IGk9MDsgaTxOOyBpKyspIHsKLSAgICAgICAgQXNzZXQqIGFzcyA9IE5VTEw7Ci0gICAgICAgIGJvb2wgc2hhcmVkID0gdHJ1ZTsKLSAgICAgICAgY29uc3QgYXNzZXRfcGF0aCYgYXAgPSBtQXNzZXRQYXRocy5pdGVtQXQoaSk7Ci0gICAgICAgIExPR1YoIkxvb2tpbmcgZm9yIHJlc291cmNlIGFzc2V0IGluICclcydcbiIsIGFwLnBhdGguc3RyaW5nKCkpOwotICAgICAgICBpZiAoYXAudHlwZSAhPSBrRmlsZVR5cGVEaXJlY3RvcnkpIHsKLSAgICAgICAgICAgIGFzcyA9IGNvbnN0X2Nhc3Q8QXNzZXRNYW5hZ2VyKj4odGhpcyktPgotICAgICAgICAgICAgICAgIG1aaXBTZXQuZ2V0WmlwUmVzb3VyY2VUYWJsZShhcC5wYXRoKTsKLSAgICAgICAgICAgIGlmIChhc3MgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgIExPR1YoImxvYWRpbmcgcmVzb3VyY2UgdGFibGUgJXNcbiIsIGFwLnBhdGguc3RyaW5nKCkpOwotICAgICAgICAgICAgICAgIGFzcyA9IGNvbnN0X2Nhc3Q8QXNzZXRNYW5hZ2VyKj4odGhpcyktPgotICAgICAgICAgICAgICAgICAgICBvcGVuTm9uQXNzZXRJblBhdGhMb2NrZWQoInJlc291cmNlcy5hcnNjIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEFzc2V0OjpBQ0NFU1NfQlVGRkVSLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYXApOwotICAgICAgICAgICAgICAgIGlmIChhc3MgIT0gTlVMTCAmJiBhc3MgIT0ga0V4Y2x1ZGVkQXNzZXQpIHsKLSAgICAgICAgICAgICAgICAgICAgYXNzID0gY29uc3RfY2FzdDxBc3NldE1hbmFnZXIqPih0aGlzKS0+Ci0gICAgICAgICAgICAgICAgICAgICAgICBtWmlwU2V0LnNldFppcFJlc291cmNlVGFibGUoYXAucGF0aCwgYXNzKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBMT0dWKCJsb2FkaW5nIHJlc291cmNlIHRhYmxlICVzXG4iLCBhcC5wYXRoLnN0cmluZygpKTsKLSAgICAgICAgICAgIEFzc2V0KiBhc3MgPSBjb25zdF9jYXN0PEFzc2V0TWFuYWdlcio+KHRoaXMpLT4KLSAgICAgICAgICAgICAgICBvcGVuTm9uQXNzZXRJblBhdGhMb2NrZWQoInJlc291cmNlcy5hcnNjIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQXNzZXQ6OkFDQ0VTU19CVUZGRVIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFwKTsKLSAgICAgICAgICAgIHNoYXJlZCA9IGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChhc3MgIT0gTlVMTCAmJiBhc3MgIT0ga0V4Y2x1ZGVkQXNzZXQpIHsKLSAgICAgICAgICAgIGlmIChydCA9PSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgbVJlc291cmNlcyA9IHJ0ID0gbmV3IFJlc1RhYmxlKCk7Ci0gICAgICAgICAgICAgICAgdXBkYXRlUmVzb3VyY2VQYXJhbXNMb2NrZWQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIExPR1YoIkluc3RhbGxpbmcgcmVzb3VyY2UgYXNzZXQgJXAgaW4gdG8gdGFibGUgJXBcbiIsIGFzcywgbVJlc291cmNlcyk7Ci0gICAgICAgICAgICBydC0+YWRkKGFzcywgKHZvaWQqKShpKzEpLCAhc2hhcmVkKTsKLQotICAgICAgICAgICAgaWYgKCFzaGFyZWQpIHsKLSAgICAgICAgICAgICAgICBkZWxldGUgYXNzOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgaWYgKHJlcXVpcmVkICYmICFydCkgTE9HVygiVW5hYmxlIHRvIGZpbmQgcmVzb3VyY2VzIGZpbGUgcmVzb3VyY2VzLmFyc2MiKTsKLSAgICBpZiAoIXJ0KSB7Ci0gICAgICAgIG1SZXNvdXJjZXMgPSBydCA9IG5ldyBSZXNUYWJsZSgpOwotICAgIH0KLSAgICByZXR1cm4gcnQ7Ci19Ci0KLXZvaWQgQXNzZXRNYW5hZ2VyOjp1cGRhdGVSZXNvdXJjZVBhcmFtc0xvY2tlZCgpIGNvbnN0Ci17Ci0gICAgUmVzVGFibGUqIHJlcyA9IG1SZXNvdXJjZXM7Ci0gICAgaWYgKCFyZXMpIHsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIHNpemVfdCBsbGVuID0gbUxvY2FsZSA/IHN0cmxlbihtTG9jYWxlKSA6IDA7Ci0gICAgbUNvbmZpZy0+bGFuZ3VhZ2VbMF0gPSAwOwotICAgIG1Db25maWctPmxhbmd1YWdlWzFdID0gMDsKLSAgICBtQ29uZmlnLT5jb3VudHJ5WzBdID0gMDsKLSAgICBtQ29uZmlnLT5jb3VudHJ5WzFdID0gMDsKLSAgICBpZiAobGxlbiA+PSAyKSB7Ci0gICAgICAgIG1Db25maWctPmxhbmd1YWdlWzBdID0gbUxvY2FsZVswXTsKLSAgICAgICAgbUNvbmZpZy0+bGFuZ3VhZ2VbMV0gPSBtTG9jYWxlWzFdOwotICAgIH0KLSAgICBpZiAobGxlbiA+PSA1KSB7Ci0gICAgICAgIG1Db25maWctPmNvdW50cnlbMF0gPSBtTG9jYWxlWzNdOwotICAgICAgICBtQ29uZmlnLT5jb3VudHJ5WzFdID0gbUxvY2FsZVs0XTsKLSAgICB9Ci0gICAgbUNvbmZpZy0+c2l6ZSA9IHNpemVvZigqbUNvbmZpZyk7Ci0KLSAgICByZXMtPnNldFBhcmFtZXRlcnMobUNvbmZpZyk7Ci19Ci0KLWNvbnN0IFJlc1RhYmxlJiBBc3NldE1hbmFnZXI6OmdldFJlc291cmNlcyhib29sIHJlcXVpcmVkKSBjb25zdAotewotICAgIGNvbnN0IFJlc1RhYmxlKiBydCA9IGdldFJlc1RhYmxlKHJlcXVpcmVkKTsKLSAgICByZXR1cm4gKnJ0OwotfQotCi1ib29sIEFzc2V0TWFuYWdlcjo6aXNVcFRvRGF0ZSgpCi17Ci0gICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLSAgICByZXR1cm4gbVppcFNldC5pc1VwVG9EYXRlKCk7Ci19Ci0KLXZvaWQgQXNzZXRNYW5hZ2VyOjpnZXRMb2NhbGVzKFZlY3RvcjxTdHJpbmc4PiogbG9jYWxlcykgY29uc3QKLXsKLSAgICBSZXNUYWJsZSogcmVzID0gbVJlc291cmNlczsKLSAgICBpZiAocmVzICE9IE5VTEwpIHsKLSAgICAgICAgcmVzLT5nZXRMb2NhbGVzKGxvY2FsZXMpOwotICAgIH0KLX0KLQotLyoKLSAqIE9wZW4gYSBub24tYXNzZXQgZmlsZSBhcyBpZiBpdCB3ZXJlIGFuIGFzc2V0LCBzZWFyY2hpbmcgZm9yIGl0IGluIHRoZQotICogc3BlY2lmaWVkIGFwcC4KLSAqCi0gKiBQYXNzIGluIGEgTlVMTCB2YWx1ZXMgZm9yICJhcHBOYW1lIiBpZiB0aGUgY29tbW9uIGFwcCBkaXJlY3Rvcnkgc2hvdWxkCi0gKiBiZSB1c2VkLgotICovCi1Bc3NldCogQXNzZXRNYW5hZ2VyOjpvcGVuTm9uQXNzZXRJblBhdGhMb2NrZWQoY29uc3QgY2hhciogZmlsZU5hbWUsIEFjY2Vzc01vZGUgbW9kZSwKLSAgICBjb25zdCBhc3NldF9wYXRoJiBhcCkKLXsKLSAgICBBc3NldCogcEFzc2V0ID0gTlVMTDsKLQotICAgIC8qIGxvb2sgYXQgdGhlIGZpbGVzeXN0ZW0gb24gZGlzayAqLwotICAgIGlmIChhcC50eXBlID09IGtGaWxlVHlwZURpcmVjdG9yeSkgewotICAgICAgICBTdHJpbmc4IHBhdGgoYXAucGF0aCk7Ci0gICAgICAgIHBhdGguYXBwZW5kUGF0aChmaWxlTmFtZSk7Ci0KLSAgICAgICAgcEFzc2V0ID0gb3BlbkFzc2V0RnJvbUZpbGVMb2NrZWQocGF0aCwgbW9kZSk7Ci0KLSAgICAgICAgaWYgKHBBc3NldCA9PSBOVUxMKSB7Ci0gICAgICAgICAgICAvKiB0cnkgYWdhaW4sIHRoaXMgdGltZSB3aXRoICIuZ3oiICovCi0gICAgICAgICAgICBwYXRoLmFwcGVuZCgiLmd6Iik7Ci0gICAgICAgICAgICBwQXNzZXQgPSBvcGVuQXNzZXRGcm9tRmlsZUxvY2tlZChwYXRoLCBtb2RlKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChwQXNzZXQgIT0gTlVMTCkgewotICAgICAgICAgICAgLy9wcmludGYoIkZPVU5EIE5BICclcycgb24gZGlza1xuIiwgZmlsZU5hbWUpOwotICAgICAgICAgICAgcEFzc2V0LT5zZXRBc3NldFNvdXJjZShwYXRoKTsKLSAgICAgICAgfQotCi0gICAgLyogbG9vayBpbnNpZGUgdGhlIHppcCBmaWxlICovCi0gICAgfSBlbHNlIHsKLSAgICAgICAgU3RyaW5nOCBwYXRoKGZpbGVOYW1lKTsKLQotICAgICAgICAvKiBjaGVjayB0aGUgYXBwcm9wcmlhdGUgWmlwIGZpbGUgKi8KLSAgICAgICAgWmlwRmlsZVJPKiBwWmlwOwotICAgICAgICBaaXBFbnRyeVJPIGVudHJ5OwotCi0gICAgICAgIHBaaXAgPSBnZXRaaXBGaWxlTG9ja2VkKGFwKTsKLSAgICAgICAgaWYgKHBaaXAgIT0gTlVMTCkgewotICAgICAgICAgICAgLy9wcmludGYoIkdPVCB6aXAsIGNoZWNraW5nIE5BICclcydcbiIsIChjb25zdCBjaGFyKikgcGF0aCk7Ci0gICAgICAgICAgICBlbnRyeSA9IHBaaXAtPmZpbmRFbnRyeUJ5TmFtZShwYXRoLnN0cmluZygpKTsKLSAgICAgICAgICAgIGlmIChlbnRyeSAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgLy9wcmludGYoIkZPVU5EIE5BIGluIFppcCBmaWxlIGZvciAlc1xuIiwgYXBwTmFtZSA/IGFwcE5hbWUgOiBrQXBwQ29tbW9uKTsKLSAgICAgICAgICAgICAgICBwQXNzZXQgPSBvcGVuQXNzZXRGcm9tWmlwTG9ja2VkKHBaaXAsIGVudHJ5LCBtb2RlLCBwYXRoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChwQXNzZXQgIT0gTlVMTCkgewotICAgICAgICAgICAgLyogY3JlYXRlIGEgInNvdXJjZSIgbmFtZSwgZm9yIGRlYnVnL2Rpc3BsYXkgKi8KLSAgICAgICAgICAgIHBBc3NldC0+c2V0QXNzZXRTb3VyY2UoCi0gICAgICAgICAgICAgICAgICAgIGNyZWF0ZVppcFNvdXJjZU5hbWVMb2NrZWQoWmlwU2V0OjpnZXRQYXRoTmFtZShhcC5wYXRoLnN0cmluZygpKSwgU3RyaW5nOCgiIiksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTdHJpbmc4KGZpbGVOYW1lKSkpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcmV0dXJuIHBBc3NldDsKLX0KLQotLyoKLSAqIE9wZW4gYW4gYXNzZXQsIHNlYXJjaGluZyBmb3IgaXQgaW4gdGhlIGRpcmVjdG9yeSBoaWVyYXJjaHkgZm9yIHRoZQotICogc3BlY2lmaWVkIGFwcC4KLSAqCi0gKiBQYXNzIGluIGEgTlVMTCB2YWx1ZXMgZm9yICJhcHBOYW1lIiBpZiB0aGUgY29tbW9uIGFwcCBkaXJlY3Rvcnkgc2hvdWxkCi0gKiBiZSB1c2VkLgotICovCi1Bc3NldCogQXNzZXRNYW5hZ2VyOjpvcGVuSW5QYXRoTG9ja2VkKGNvbnN0IGNoYXIqIGZpbGVOYW1lLCBBY2Nlc3NNb2RlIG1vZGUsCi0gICAgY29uc3QgYXNzZXRfcGF0aCYgYXApCi17Ci0gICAgQXNzZXQqIHBBc3NldCA9IE5VTEw7Ci0KLSAgICAvKgotICAgICAqIFRyeSB2YXJpb3VzIGNvbWJpbmF0aW9ucyBvZiBsb2NhbGUgYW5kIHZlbmRvci4KLSAgICAgKi8KLSAgICBpZiAobUxvY2FsZSAhPSBOVUxMICYmIG1WZW5kb3IgIT0gTlVMTCkKLSAgICAgICAgcEFzc2V0ID0gb3BlbkluTG9jYWxlVmVuZG9yTG9ja2VkKGZpbGVOYW1lLCBtb2RlLCBhcCwgbUxvY2FsZSwgbVZlbmRvcik7Ci0gICAgaWYgKHBBc3NldCA9PSBOVUxMICYmIG1WZW5kb3IgIT0gTlVMTCkKLSAgICAgICAgcEFzc2V0ID0gb3BlbkluTG9jYWxlVmVuZG9yTG9ja2VkKGZpbGVOYW1lLCBtb2RlLCBhcCwgTlVMTCwgbVZlbmRvcik7Ci0gICAgaWYgKHBBc3NldCA9PSBOVUxMICYmIG1Mb2NhbGUgIT0gTlVMTCkKLSAgICAgICAgcEFzc2V0ID0gb3BlbkluTG9jYWxlVmVuZG9yTG9ja2VkKGZpbGVOYW1lLCBtb2RlLCBhcCwgbUxvY2FsZSwgTlVMTCk7Ci0gICAgaWYgKHBBc3NldCA9PSBOVUxMKQotICAgICAgICBwQXNzZXQgPSBvcGVuSW5Mb2NhbGVWZW5kb3JMb2NrZWQoZmlsZU5hbWUsIG1vZGUsIGFwLCBOVUxMLCBOVUxMKTsKLQotICAgIHJldHVybiBwQXNzZXQ7Ci19Ci0KLS8qCi0gKiBPcGVuIGFuIGFzc2V0LCBzZWFyY2hpbmcgZm9yIGl0IGluIHRoZSBkaXJlY3RvcnkgaGllcmFyY2h5IGZvciB0aGUKLSAqIHNwZWNpZmllZCBsb2NhbGUgYW5kIHZlbmRvci4KLSAqCi0gKiBXZSBhbHNvIHNlYXJjaCBpbiAiYXBwLmphciIuCi0gKgotICogUGFzcyBpbiBOVUxMIHZhbHVlcyBmb3IgImFwcE5hbWUiLCAibG9jYWxlIiwgYW5kICJ2ZW5kb3IiIGlmIHRoZQotICogZGVmYXVsdHMgc2hvdWxkIGJlIHVzZWQuCi0gKi8KLUFzc2V0KiBBc3NldE1hbmFnZXI6Om9wZW5JbkxvY2FsZVZlbmRvckxvY2tlZChjb25zdCBjaGFyKiBmaWxlTmFtZSwgQWNjZXNzTW9kZSBtb2RlLAotICAgIGNvbnN0IGFzc2V0X3BhdGgmIGFwLCBjb25zdCBjaGFyKiBsb2NhbGUsIGNvbnN0IGNoYXIqIHZlbmRvcikKLXsKLSAgICBBc3NldCogcEFzc2V0ID0gTlVMTDsKLQotICAgIGlmIChhcC50eXBlID09IGtGaWxlVHlwZURpcmVjdG9yeSkgewotICAgICAgICBpZiAobUNhY2hlTW9kZSA9PSBDQUNIRV9PRkYpIHsKLSAgICAgICAgICAgIC8qIGxvb2sgYXQgdGhlIGZpbGVzeXN0ZW0gb24gZGlzayAqLwotICAgICAgICAgICAgU3RyaW5nOCBwYXRoKGNyZWF0ZVBhdGhOYW1lTG9ja2VkKGFwLCBsb2NhbGUsIHZlbmRvcikpOwotICAgICAgICAgICAgcGF0aC5hcHBlbmRQYXRoKGZpbGVOYW1lKTsKLSAgICAKLSAgICAgICAgICAgIFN0cmluZzggZXhjbHVkZU5hbWUocGF0aCk7Ci0gICAgICAgICAgICBleGNsdWRlTmFtZS5hcHBlbmQoa0V4Y2x1ZGVFeHRlbnNpb24pOwotICAgICAgICAgICAgaWYgKDo6Z2V0RmlsZVR5cGUoZXhjbHVkZU5hbWUuc3RyaW5nKCkpICE9IGtGaWxlVHlwZU5vbmV4aXN0ZW50KSB7Ci0gICAgICAgICAgICAgICAgLyogc2F5IG5vIG1vcmUgKi8KLSAgICAgICAgICAgICAgICAvL3ByaW50ZigiKysrIGV4Y2x1ZGluZyAnJXMnXG4iLCAoY29uc3QgY2hhciopIGV4Y2x1ZGVOYW1lKTsKLSAgICAgICAgICAgICAgICByZXR1cm4ga0V4Y2x1ZGVkQXNzZXQ7Ci0gICAgICAgICAgICB9Ci0gICAgCi0gICAgICAgICAgICBwQXNzZXQgPSBvcGVuQXNzZXRGcm9tRmlsZUxvY2tlZChwYXRoLCBtb2RlKTsKLSAgICAKLSAgICAgICAgICAgIGlmIChwQXNzZXQgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgIC8qIHRyeSBhZ2FpbiwgdGhpcyB0aW1lIHdpdGggIi5neiIgKi8KLSAgICAgICAgICAgICAgICBwYXRoLmFwcGVuZCgiLmd6Iik7Ci0gICAgICAgICAgICAgICAgcEFzc2V0ID0gb3BlbkFzc2V0RnJvbUZpbGVMb2NrZWQocGF0aCwgbW9kZSk7Ci0gICAgICAgICAgICB9Ci0gICAgCi0gICAgICAgICAgICBpZiAocEFzc2V0ICE9IE5VTEwpCi0gICAgICAgICAgICAgICAgcEFzc2V0LT5zZXRBc3NldFNvdXJjZShwYXRoKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8qIGZpbmQgaW4gY2FjaGUgKi8KLSAgICAgICAgICAgIFN0cmluZzggcGF0aChjcmVhdGVQYXRoTmFtZUxvY2tlZChhcCwgbG9jYWxlLCB2ZW5kb3IpKTsKLSAgICAgICAgICAgIHBhdGguYXBwZW5kUGF0aChmaWxlTmFtZSk7Ci0gICAgCi0gICAgICAgICAgICBBc3NldERpcjo6RmlsZUluZm8gdG1wSW5mbzsKLSAgICAgICAgICAgIGJvb2wgZm91bmQgPSBmYWxzZTsKLSAgICAKLSAgICAgICAgICAgIFN0cmluZzggZXhjbHVkZU5hbWUocGF0aCk7Ci0gICAgICAgICAgICBleGNsdWRlTmFtZS5hcHBlbmQoa0V4Y2x1ZGVFeHRlbnNpb24pOwotICAgIAotICAgICAgICAgICAgaWYgKG1DYWNoZS5pbmRleE9mKGV4Y2x1ZGVOYW1lKSAhPSBOQU1FX05PVF9GT1VORCkgewotICAgICAgICAgICAgICAgIC8qIGdvIG5vIGZhcnRoZXIgKi8KLSAgICAgICAgICAgICAgICAvL3ByaW50ZigiKysrIEV4Y2x1ZGluZyAnJXMnXG4iLCAoY29uc3QgY2hhciopIGV4Y2x1ZGVOYW1lKTsKLSAgICAgICAgICAgICAgICByZXR1cm4ga0V4Y2x1ZGVkQXNzZXQ7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8qCi0gICAgICAgICAgICAgKiBGaWxlIGNvbXByZXNzaW9uIGV4dGVuc2lvbnMgKCIuZ3oiKSBkb24ndCBnZXQgc3RvcmVkIGluIHRoZQotICAgICAgICAgICAgICogbmFtZSBjYWNoZSwgc28gd2UgaGF2ZSB0byB0cnkgYm90aCBoZXJlLgotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBpZiAobUNhY2hlLmluZGV4T2YocGF0aCkgIT0gTkFNRV9OT1RfRk9VTkQpIHsKLSAgICAgICAgICAgICAgICBmb3VuZCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgcEFzc2V0ID0gb3BlbkFzc2V0RnJvbUZpbGVMb2NrZWQocGF0aCwgbW9kZSk7Ci0gICAgICAgICAgICAgICAgaWYgKHBBc3NldCA9PSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8qIHRyeSBhZ2FpbiwgdGhpcyB0aW1lIHdpdGggIi5neiIgKi8KLSAgICAgICAgICAgICAgICAgICAgcGF0aC5hcHBlbmQoIi5neiIpOwotICAgICAgICAgICAgICAgICAgICBwQXNzZXQgPSBvcGVuQXNzZXRGcm9tRmlsZUxvY2tlZChwYXRoLCBtb2RlKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmIChwQXNzZXQgIT0gTlVMTCkKLSAgICAgICAgICAgICAgICBwQXNzZXQtPnNldEFzc2V0U291cmNlKHBhdGgpOwotCi0gICAgICAgICAgICAvKgotICAgICAgICAgICAgICogRG9uJ3QgY29udGludWUgdGhlIHNlYXJjaCBpbnRvIHRoZSBaaXAgZmlsZXMuICBPdXIgY2FjaGVkIGluZm8KLSAgICAgICAgICAgICAqIHNhaWQgaXQgd2FzIGEgZmlsZSBvbiBkaXNrOyB0byBiZSBjb25zaXN0ZW50IHdpdGggb3BlbkRpcigpCi0gICAgICAgICAgICAgKiB3ZSB3YW50IHRvIHJldHVybiB0aGUgbG9vc2UgYXNzZXQuICBJZiB0aGUgY2FjaGVkIGZpbGUgZ2V0cwotICAgICAgICAgICAgICogcmVtb3ZlZCwgd2UgZmFpbC4KLSAgICAgICAgICAgICAqCi0gICAgICAgICAgICAgKiBUaGUgYWx0ZXJuYXRpdmUgaXMgdG8gdXBkYXRlIG91ciBjYWNoZSB3aGVuIGZpbGVzIGdldCBkZWxldGVkLAotICAgICAgICAgICAgICogb3IgbWFrZSBzb21lIHNvcnQgb2YgImJlc3QgZWZmb3J0IiBwcm9taXNlLCBidXQgZm9yIG5vdyBJJ20KLSAgICAgICAgICAgICAqIHRha2luZyB0aGUgaGFyZCBsaW5lLgotICAgICAgICAgICAgICovCi0gICAgICAgICAgICBpZiAoZm91bmQpIHsKLSAgICAgICAgICAgICAgICBpZiAocEFzc2V0ID09IE5VTEwpCi0gICAgICAgICAgICAgICAgICAgIExPR0QoIkV4cGVjdGVkIGZpbGUgbm90IGZvdW5kOiAnJXMnXG4iLCBwYXRoLnN0cmluZygpKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gcEFzc2V0OwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBFaXRoZXIgaXQgd2Fzbid0IGZvdW5kIG9uIGRpc2sgb3Igb24gdGhlIGNhY2hlZCB2aWV3IG9mIHRoZSBkaXNrLgotICAgICAqIERpZyB0aHJvdWdoIHRoZSBjdXJyZW50bHktb3BlbmVkIHNldCBvZiBaaXAgZmlsZXMuICBJZiBjYWNoaW5nCi0gICAgICogaXMgZGlzYWJsZWQsIHRoZSBaaXAgZmlsZSBtYXkgZ2V0IHJlb3BlbmVkLgotICAgICAqLwotICAgIGlmIChwQXNzZXQgPT0gTlVMTCAmJiBhcC50eXBlID09IGtGaWxlVHlwZVJlZ3VsYXIpIHsKLSAgICAgICAgU3RyaW5nOCBwYXRoOwotCi0gICAgICAgIHBhdGguYXBwZW5kUGF0aCgobG9jYWxlICE9IE5VTEwpID8gbG9jYWxlIDoga0RlZmF1bHRMb2NhbGUpOwotICAgICAgICBwYXRoLmFwcGVuZFBhdGgoKHZlbmRvciAhPSBOVUxMKSA/IHZlbmRvciA6IGtEZWZhdWx0VmVuZG9yKTsKLSAgICAgICAgcGF0aC5hcHBlbmRQYXRoKGZpbGVOYW1lKTsKLQotICAgICAgICAvKiBjaGVjayB0aGUgYXBwcm9wcmlhdGUgWmlwIGZpbGUgKi8KLSAgICAgICAgWmlwRmlsZVJPKiBwWmlwOwotICAgICAgICBaaXBFbnRyeVJPIGVudHJ5OwotCi0gICAgICAgIHBaaXAgPSBnZXRaaXBGaWxlTG9ja2VkKGFwKTsKLSAgICAgICAgaWYgKHBaaXAgIT0gTlVMTCkgewotICAgICAgICAgICAgLy9wcmludGYoIkdPVCB6aXAsIGNoZWNraW5nICclcydcbiIsIChjb25zdCBjaGFyKikgcGF0aCk7Ci0gICAgICAgICAgICBlbnRyeSA9IHBaaXAtPmZpbmRFbnRyeUJ5TmFtZShwYXRoLnN0cmluZygpKTsKLSAgICAgICAgICAgIGlmIChlbnRyeSAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgLy9wcmludGYoIkZPVU5EIGluIFppcCBmaWxlIGZvciAlcy8lcy0lc1xuIiwKLSAgICAgICAgICAgICAgICAvLyAgICBhcHBOYW1lLCBsb2NhbGUsIHZlbmRvcik7Ci0gICAgICAgICAgICAgICAgcEFzc2V0ID0gb3BlbkFzc2V0RnJvbVppcExvY2tlZChwWmlwLCBlbnRyeSwgbW9kZSwgcGF0aCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAocEFzc2V0ICE9IE5VTEwpIHsKLSAgICAgICAgICAgIC8qIGNyZWF0ZSBhICJzb3VyY2UiIG5hbWUsIGZvciBkZWJ1Zy9kaXNwbGF5ICovCi0gICAgICAgICAgICBwQXNzZXQtPnNldEFzc2V0U291cmNlKGNyZWF0ZVppcFNvdXJjZU5hbWVMb2NrZWQoWmlwU2V0OjpnZXRQYXRoTmFtZShhcC5wYXRoLnN0cmluZygpKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTdHJpbmc4KCIiKSwgU3RyaW5nOChmaWxlTmFtZSkpKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBwQXNzZXQ7Ci19Ci0KLS8qCi0gKiBDcmVhdGUgYSAic291cmNlIG5hbWUiIGZvciBhIGZpbGUgZnJvbSBhIFppcCBhcmNoaXZlLgotICovCi1TdHJpbmc4IEFzc2V0TWFuYWdlcjo6Y3JlYXRlWmlwU291cmNlTmFtZUxvY2tlZChjb25zdCBTdHJpbmc4JiB6aXBGaWxlTmFtZSwKLSAgICBjb25zdCBTdHJpbmc4JiBkaXJOYW1lLCBjb25zdCBTdHJpbmc4JiBmaWxlTmFtZSkKLXsKLSAgICBTdHJpbmc4IHNvdXJjZU5hbWUoInppcDoiKTsKLSAgICBzb3VyY2VOYW1lLmFwcGVuZCh6aXBGaWxlTmFtZSk7Ci0gICAgc291cmNlTmFtZS5hcHBlbmQoIjoiKTsKLSAgICBpZiAoZGlyTmFtZS5sZW5ndGgoKSA+IDApIHsKLSAgICAgICAgc291cmNlTmFtZS5hcHBlbmRQYXRoKGRpck5hbWUpOwotICAgIH0KLSAgICBzb3VyY2VOYW1lLmFwcGVuZFBhdGgoZmlsZU5hbWUpOwotICAgIHJldHVybiBzb3VyY2VOYW1lOwotfQotCi0vKgotICogQ3JlYXRlIGEgcGF0aCB0byBhIGxvb3NlIGFzc2V0IChhc3NldC1iYXNlL2FwcC9sb2NhbGUvdmVuZG9yKS4KLSAqLwotU3RyaW5nOCBBc3NldE1hbmFnZXI6OmNyZWF0ZVBhdGhOYW1lTG9ja2VkKGNvbnN0IGFzc2V0X3BhdGgmIGFwLCBjb25zdCBjaGFyKiBsb2NhbGUsCi0gICAgY29uc3QgY2hhciogdmVuZG9yKQotewotICAgIFN0cmluZzggcGF0aChhcC5wYXRoKTsKLSAgICBwYXRoLmFwcGVuZFBhdGgoKGxvY2FsZSAhPSBOVUxMKSA/IGxvY2FsZSA6IGtEZWZhdWx0TG9jYWxlKTsKLSAgICBwYXRoLmFwcGVuZFBhdGgoKHZlbmRvciAhPSBOVUxMKSA/IHZlbmRvciA6IGtEZWZhdWx0VmVuZG9yKTsKLSAgICByZXR1cm4gcGF0aDsKLX0KLQotLyoKLSAqIENyZWF0ZSBhIHBhdGggdG8gYSBsb29zZSBhc3NldCAoYXNzZXQtYmFzZS9hcHAvcm9vdERpcikuCi0gKi8KLVN0cmluZzggQXNzZXRNYW5hZ2VyOjpjcmVhdGVQYXRoTmFtZUxvY2tlZChjb25zdCBhc3NldF9wYXRoJiBhcCwgY29uc3QgY2hhciogcm9vdERpcikKLXsKLSAgICBTdHJpbmc4IHBhdGgoYXAucGF0aCk7Ci0gICAgaWYgKHJvb3REaXIgIT0gTlVMTCkgcGF0aC5hcHBlbmRQYXRoKHJvb3REaXIpOwotICAgIHJldHVybiBwYXRoOwotfQotCi0vKgotICogUmV0dXJuIGEgcG9pbnRlciB0byBvbmUgb2Ygb3VyIG9wZW4gWmlwIGFyY2hpdmVzLiAgUmV0dXJucyBOVUxMIGlmIG5vCi0gKiBtYXRjaGluZyBaaXAgZmlsZSBleGlzdHMuCi0gKgotICogUmlnaHQgbm93IHdlIGhhdmUgMiBwb3NzaWJsZSBaaXAgZmlsZXMgKDEgZWFjaCBpbiBhcHAvImNvbW1vbiIpLgotICoKLSAqIElmIGNhY2hpbmcgaXMgc2V0IHRvIENBQ0hFX09GRiwgdG8gZ2V0IHRoZSBleHBlY3RlZCBiZWhhdmlvciB3ZQotICogbmVlZCB0byByZW9wZW4gdGhlIFppcCBmaWxlIG9uIGV2ZXJ5IHJlcXVlc3QuICBUaGF0IHdvdWxkIGJlIHNpbGx5Ci0gKiBhbmQgZXhwZW5zaXZlLCBzbyBpbnN0ZWFkIHdlIGp1c3QgY2hlY2sgdGhlIGZpbGUgbW9kaWZpY2F0aW9uIGRhdGUuCi0gKgotICogUGFzcyBpbiBOVUxMIHZhbHVlcyBmb3IgImFwcE5hbWUiLCAibG9jYWxlIiwgYW5kICJ2ZW5kb3IiIGlmIHRoZQotICogZ2VuZXJpY3Mgc2hvdWxkIGJlIHVzZWQuCi0gKi8KLVppcEZpbGVSTyogQXNzZXRNYW5hZ2VyOjpnZXRaaXBGaWxlTG9ja2VkKGNvbnN0IGFzc2V0X3BhdGgmIGFwKQotewotICAgIExPR1YoImdldFppcEZpbGVMb2NrZWQoKSBpbiAlcFxuIiwgdGhpcyk7Ci0KLSAgICByZXR1cm4gbVppcFNldC5nZXRaaXAoYXAucGF0aCk7Ci19Ci0KLS8qCi0gKiBUcnkgdG8gb3BlbiBhbiBhc3NldCBmcm9tIGEgZmlsZSBvbiBkaXNrLgotICoKLSAqIElmIHRoZSBmaWxlIGlzIGNvbXByZXNzZWQgd2l0aCBnemlwLCB3ZSBzZWVrIHRvIHRoZSBzdGFydCBvZiB0aGUKLSAqIGRlZmxhdGVkIGRhdGEgYW5kIHBhc3MgdGhhdCBpbiAoanVzdCBsaWtlIHdlIHdvdWxkIGZvciBhIFppcCBhcmNoaXZlKS4KLSAqCi0gKiBGb3IgdW5jb21wcmVzc2VkIGRhdGEsIHdlIG1heSBhbHJlYWR5IGhhdmUgYW4gbW1hcCgpZWQgdmVyc2lvbiBzaXR0aW5nCi0gKiBhcm91bmQuICBJZiBzbywgd2Ugd2FudCB0byBoYW5kIHRoYXQgdG8gdGhlIEFzc2V0IGluc3RlYWQuCi0gKgotICogVGhpcyByZXR1cm5zIE5VTEwgaWYgdGhlIGZpbGUgZG9lc24ndCBleGlzdCwgY291bGRuJ3QgYmUgb3BlbmVkLCBvcgotICogY2xhaW1zIHRvIGJlIGEgIi5neiIgYnV0IGlzbid0LgotICovCi1Bc3NldCogQXNzZXRNYW5hZ2VyOjpvcGVuQXNzZXRGcm9tRmlsZUxvY2tlZChjb25zdCBTdHJpbmc4JiBwYXRoTmFtZSwKLSAgICBBY2Nlc3NNb2RlIG1vZGUpCi17Ci0gICAgQXNzZXQqIHBBc3NldCA9IE5VTEw7Ci0KLSAgICBpZiAoc3RyY2FzZWNtcChwYXRoTmFtZS5nZXRQYXRoRXh0ZW5zaW9uKCkuc3RyaW5nKCksICIuZ3oiKSA9PSAwKSB7Ci0gICAgICAgIC8vcHJpbnRmKCJUUllJTkcgJyVzJ1xuIiwgKGNvbnN0IGNoYXIqKSBwYXRoTmFtZSk7Ci0gICAgICAgIHBBc3NldCA9IEFzc2V0OjpjcmVhdGVGcm9tQ29tcHJlc3NlZEZpbGUocGF0aE5hbWUuc3RyaW5nKCksIG1vZGUpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIC8vcHJpbnRmKCJUUllJTkcgJyVzJ1xuIiwgKGNvbnN0IGNoYXIqKSBwYXRoTmFtZSk7Ci0gICAgICAgIHBBc3NldCA9IEFzc2V0OjpjcmVhdGVGcm9tRmlsZShwYXRoTmFtZS5zdHJpbmcoKSwgbW9kZSk7Ci0gICAgfQotCi0gICAgcmV0dXJuIHBBc3NldDsKLX0KLQotLyoKLSAqIEdpdmVuIGFuIGVudHJ5IGluIGEgWmlwIGFyY2hpdmUsIGNyZWF0ZSBhIG5ldyBBc3NldCBvYmplY3QuCi0gKgotICogSWYgdGhlIGVudHJ5IGlzIHVuY29tcHJlc3NlZCwgd2UgbWF5IHdhbnQgdG8gY3JlYXRlIG9yIHNoYXJlIGEKLSAqIHNsaWNlIG9mIHNoYXJlZCBtZW1vcnkuCi0gKi8KLUFzc2V0KiBBc3NldE1hbmFnZXI6Om9wZW5Bc3NldEZyb21aaXBMb2NrZWQoY29uc3QgWmlwRmlsZVJPKiBwWmlwRmlsZSwKLSAgICBjb25zdCBaaXBFbnRyeVJPIGVudHJ5LCBBY2Nlc3NNb2RlIG1vZGUsIGNvbnN0IFN0cmluZzgmIGVudHJ5TmFtZSkKLXsKLSAgICBBc3NldCogcEFzc2V0ID0gTlVMTDsKLQotICAgIC8vIFRPRE86IGxvb2sgZm9yIHByZXZpb3VzbHktY3JlYXRlZCBzaGFyZWQgbWVtb3J5IHNsaWNlPwotICAgIGludCBtZXRob2Q7Ci0gICAgbG9uZyB1bmNvbXByZXNzZWRMZW47Ci0KLSAgICAvL3ByaW50ZigiVVNJTkcgWmlwICclcydcbiIsIHBFbnRyeS0+Z2V0RmlsZU5hbWUoKSk7Ci0KLSAgICAvL3BaaXBGaWxlLT5nZXRFbnRyeUluZm8oZW50cnksICZtZXRob2QsICZ1bmNvbXByZXNzZWRMZW4sICZjb21wcmVzc2VkTGVuLAotICAgIC8vICAgICZvZmZzZXQpOwotICAgIGlmICghcFppcEZpbGUtPmdldEVudHJ5SW5mbyhlbnRyeSwgJm1ldGhvZCwgJnVuY29tcHJlc3NlZExlbiwgTlVMTCwgTlVMTCwKLSAgICAgICAgICAgIE5VTEwsIE5VTEwpKQotICAgIHsKLSAgICAgICAgTE9HVygiZ2V0RW50cnlJbmZvIGZhaWxlZFxuIik7Ci0gICAgICAgIHJldHVybiBOVUxMOwotICAgIH0KLQotICAgIEZpbGVNYXAqIGRhdGFNYXAgPSBwWmlwRmlsZS0+Y3JlYXRlRW50cnlGaWxlTWFwKGVudHJ5KTsKLSAgICBpZiAoZGF0YU1hcCA9PSBOVUxMKSB7Ci0gICAgICAgIExPR1coImNyZWF0ZSBtYXAgZnJvbSBlbnRyeSBmYWlsZWRcbiIpOwotICAgICAgICByZXR1cm4gTlVMTDsKLSAgICB9Ci0KLSAgICBpZiAobWV0aG9kID09IFppcEZpbGVSTzo6a0NvbXByZXNzU3RvcmVkKSB7Ci0gICAgICAgIHBBc3NldCA9IEFzc2V0OjpjcmVhdGVGcm9tVW5jb21wcmVzc2VkTWFwKGRhdGFNYXAsIG1vZGUpOwotICAgICAgICBMT0dWKCJPcGVuZWQgdW5jb21wcmVzc2VkIGVudHJ5ICVzIGluIHppcCAlcyBtb2RlICVkOiAlcCIsIGVudHJ5TmFtZS5zdHJpbmcoKSwKLSAgICAgICAgICAgICAgICBkYXRhTWFwLT5nZXRGaWxlTmFtZSgpLCBtb2RlLCBwQXNzZXQpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHBBc3NldCA9IEFzc2V0OjpjcmVhdGVGcm9tQ29tcHJlc3NlZE1hcChkYXRhTWFwLCBtZXRob2QsCi0gICAgICAgICAgICB1bmNvbXByZXNzZWRMZW4sIG1vZGUpOwotICAgICAgICBMT0dWKCJPcGVuZWQgY29tcHJlc3NlZCBlbnRyeSAlcyBpbiB6aXAgJXMgbW9kZSAlZDogJXAiLCBlbnRyeU5hbWUuc3RyaW5nKCksCi0gICAgICAgICAgICAgICAgZGF0YU1hcC0+Z2V0RmlsZU5hbWUoKSwgbW9kZSwgcEFzc2V0KTsKLSAgICB9Ci0gICAgaWYgKHBBc3NldCA9PSBOVUxMKSB7Ci0gICAgICAgIC8qIHVuZXhwZWN0ZWQgKi8KLSAgICAgICAgTE9HVygiY3JlYXRlIGZyb20gc2VnbWVudCBmYWlsZWRcbiIpOwotICAgIH0KLQotICAgIHJldHVybiBwQXNzZXQ7Ci19Ci0KLQotCi0vKgotICogT3BlbiBhIGRpcmVjdG9yeSBpbiB0aGUgYXNzZXQgbmFtZXNwYWNlLgotICoKLSAqIEFuICJhc3NldCBkaXJlY3RvcnkiIGlzIHNpbXBseSB0aGUgY29tYmluYXRpb24gb2YgYWxsIGZpbGVzIGluIGFsbAotICogbG9jYXRpb25zLCB3aXRoICIuZ3oiIHN0cmlwcGVkIGZvciBsb29zZSBmaWxlcy4gIFdpdGggYXBwLCBsb2NhbGUsIGFuZAotICogdmVuZG9yIGRlZmluZWQsIHdlIGhhdmUgOCBkaXJlY3RvcmllcyBhbmQgMiBaaXAgYXJjaGl2ZXMgdG8gc2Nhbi4KLSAqCi0gKiBQYXNzIGluICIiIGZvciB0aGUgcm9vdCBkaXIuCi0gKi8KLUFzc2V0RGlyKiBBc3NldE1hbmFnZXI6Om9wZW5EaXIoY29uc3QgY2hhciogZGlyTmFtZSkKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotCi0gICAgQXNzZXREaXIqIHBEaXIgPSBOVUxMOwotICAgIFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+KiBwTWVyZ2VkSW5mbyA9IE5VTEw7Ci0KLSAgICBMT0dfRkFUQUxfSUYobUFzc2V0UGF0aHMuc2l6ZSgpID09IDAsICJObyBhc3NldHMgYWRkZWQgdG8gQXNzZXRNYW5hZ2VyIik7Ci0gICAgYXNzZXJ0KGRpck5hbWUgIT0gTlVMTCk7Ci0KLSAgICAvL3ByaW50ZigiKysrIG9wZW5EaXIoJXMpIGluICclcydcbiIsIGRpck5hbWUsIChjb25zdCBjaGFyKikgbUFzc2V0QmFzZSk7Ci0KLSAgICBpZiAobUNhY2hlTW9kZSAhPSBDQUNIRV9PRkYgJiYgIW1DYWNoZVZhbGlkKQotICAgICAgICBsb2FkRmlsZU5hbWVDYWNoZUxvY2tlZCgpOwotCi0gICAgcERpciA9IG5ldyBBc3NldERpcjsKLQotICAgIC8qCi0gICAgICogU2NhbiB0aGUgdmFyaW91cyBkaXJlY3RvcmllcywgbWVyZ2luZyB3aGF0IHdlIGZpbmQgaW50byBhIHNpbmdsZQotICAgICAqIHZlY3Rvci4gIFdlIHdhbnQgdG8gc2NhbiB0aGVtIGluIHJldmVyc2UgcHJpb3JpdHkgb3JkZXIgc28gdGhhdAotICAgICAqIHRoZSAiLkVYQ0xVREUiIHByb2Nlc3Npbmcgd29ya3MgY29ycmVjdGx5LiAgQWxzbywgaWYgd2UgZGVjaWRlIHdlCi0gICAgICogd2FudCB0byByZW1lbWJlciB3aGVyZSB0aGUgZmlsZSBpcyBjb21pbmcgZnJvbSwgd2UnbGwgZ2V0IHRoZSByaWdodAotICAgICAqIHZlcnNpb24uCi0gICAgICoKLSAgICAgKiBXZSBzdGFydCB3aXRoIFppcCBhcmNoaXZlcywgdGhlbiBkbyBsb29zZSBmaWxlcy4KLSAgICAgKi8KLSAgICBwTWVyZ2VkSW5mbyA9IG5ldyBTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPjsKLQotICAgIHNpemVfdCBpID0gbUFzc2V0UGF0aHMuc2l6ZSgpOwotICAgIHdoaWxlIChpID4gMCkgewotICAgICAgICBpLS07Ci0gICAgICAgIGNvbnN0IGFzc2V0X3BhdGgmIGFwID0gbUFzc2V0UGF0aHMuaXRlbUF0KGkpOwotICAgICAgICBpZiAoYXAudHlwZSA9PSBrRmlsZVR5cGVSZWd1bGFyKSB7Ci0gICAgICAgICAgICBMT0dWKCJBZGRpbmcgZGlyZWN0b3J5ICVzIGZyb20gemlwICVzIiwgZGlyTmFtZSwgYXAucGF0aC5zdHJpbmcoKSk7Ci0gICAgICAgICAgICBzY2FuQW5kTWVyZ2VaaXBMb2NrZWQocE1lcmdlZEluZm8sIGFwLCBrQXNzZXRzUm9vdCwgZGlyTmFtZSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBMT0dWKCJBZGRpbmcgZGlyZWN0b3J5ICVzIGZyb20gZGlyICVzIiwgZGlyTmFtZSwgYXAucGF0aC5zdHJpbmcoKSk7Ci0gICAgICAgICAgICBzY2FuQW5kTWVyZ2VEaXJMb2NrZWQocE1lcmdlZEluZm8sIGFwLCBrQXNzZXRzUm9vdCwgZGlyTmFtZSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSNpZiAwCi0gICAgcHJpbnRmKCJGSUxFIExJU1Q6XG4iKTsKLSAgICBmb3IgKGkgPSAwOyBpIDwgKHNpemVfdCkgcE1lcmdlZEluZm8tPnNpemUoKTsgaSsrKSB7Ci0gICAgICAgIHByaW50ZigiICVkOiAoJWQpICclcydcbiIsIGksCi0gICAgICAgICAgICBwTWVyZ2VkSW5mby0+aXRlbUF0KGkpLmdldEZpbGVUeXBlKCksCi0gICAgICAgICAgICAoY29uc3QgY2hhciopIHBNZXJnZWRJbmZvLT5pdGVtQXQoaSkuZ2V0RmlsZU5hbWUoKSk7Ci0gICAgfQotI2VuZGlmCi0KLSAgICBwRGlyLT5zZXRGaWxlTGlzdChwTWVyZ2VkSW5mbyk7Ci0gICAgcmV0dXJuIHBEaXI7Ci19Ci0KLS8qCi0gKiBTY2FuIHRoZSBjb250ZW50cyBvZiB0aGUgc3BlY2lmaWVkIGRpcmVjdG9yeSBhbmQgbWVyZ2UgdGhlbSBpbnRvIHRoZQotICogInBNZXJnZWRJbmZvIiB2ZWN0b3IsIHJlbW92aW5nIHByZXZpb3VzIGVudHJpZXMgaWYgd2UgZmluZCAiZXhjbHVkZSIKLSAqIGRpcmVjdGl2ZXMuCi0gKgotICogUmV0dXJucyAiZmFsc2UiIGlmIHdlIGZvdW5kIG5vdGhpbmcgdG8gY29udHJpYnV0ZS4KLSAqLwotYm9vbCBBc3NldE1hbmFnZXI6OnNjYW5BbmRNZXJnZURpckxvY2tlZChTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPiogcE1lcmdlZEluZm8sCi0gICAgY29uc3QgYXNzZXRfcGF0aCYgYXAsIGNvbnN0IGNoYXIqIHJvb3REaXIsIGNvbnN0IGNoYXIqIGRpck5hbWUpCi17Ci0gICAgU29ydGVkVmVjdG9yPEFzc2V0RGlyOjpGaWxlSW5mbz4qIHBDb250ZW50czsKLSAgICBTdHJpbmc4IHBhdGg7Ci0KLSAgICBhc3NlcnQocE1lcmdlZEluZm8gIT0gTlVMTCk7Ci0KLSAgICAvL3ByaW50Zigic2NhbkFuZE1lcmdlRGlyOiAlcyAlcyAlcyAlc1xuIiwgYXBwTmFtZSwgbG9jYWxlLCB2ZW5kb3IsZGlyTmFtZSk7Ci0KLSAgICBpZiAobUNhY2hlVmFsaWQpIHsKLSAgICAgICAgaW50IGksIHN0YXJ0LCBjb3VudDsKLQotICAgICAgICBwQ29udGVudHMgPSBuZXcgU29ydGVkVmVjdG9yPEFzc2V0RGlyOjpGaWxlSW5mbz47Ci0KLSAgICAgICAgLyoKLSAgICAgICAgICogR2V0IHRoZSBiYXNpYyBwYXJ0aWFsIHBhdGggYW5kIGZpbmQgaXQgaW4gdGhlIGNhY2hlLiAgVGhhdCdzCi0gICAgICAgICAqIHRoZSBzdGFydCBwb2ludCBmb3IgdGhlIHNlYXJjaC4KLSAgICAgICAgICovCi0gICAgICAgIHBhdGggPSBjcmVhdGVQYXRoTmFtZUxvY2tlZChhcCwgcm9vdERpcik7Ci0gICAgICAgIGlmIChkaXJOYW1lWzBdICE9ICdcMCcpCi0gICAgICAgICAgICBwYXRoLmFwcGVuZFBhdGgoZGlyTmFtZSk7Ci0KLSAgICAgICAgc3RhcnQgPSBtQ2FjaGUuaW5kZXhPZihwYXRoKTsKLSAgICAgICAgaWYgKHN0YXJ0ID09IE5BTUVfTk9UX0ZPVU5EKSB7Ci0gICAgICAgICAgICAvL3ByaW50ZigiKysrIG5vdCBmb3VuZCBpbiBjYWNoZTogZGlyICclcydcbiIsIChjb25zdCBjaGFyKikgcGF0aCk7Ci0gICAgICAgICAgICBkZWxldGUgcENvbnRlbnRzOwotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgLyoKLSAgICAgICAgICogVGhlIG1hdGNoIHN0cmluZyBsb29rcyBsaWtlICJjb21tb24vZGVmYXVsdC9kZWZhdWx0L2Zvby9iYXIvIi4KLSAgICAgICAgICogVGhlICcvJyBvbiB0aGUgZW5kIGVuc3VyZXMgdGhhdCB3ZSBkb24ndCBtYXRjaCBvbiB0aGUgZGlyZWN0b3J5Ci0gICAgICAgICAqIGl0c2VsZiBvciBvbiAiLi4uL2Zvby9iYXJmeS8iLgotICAgICAgICAgKi8KLSAgICAgICAgcGF0aC5hcHBlbmQoIi8iKTsKLQotICAgICAgICBjb3VudCA9IG1DYWNoZS5zaXplKCk7Ci0KLSAgICAgICAgLyoKLSAgICAgICAgICogUGljayBvdXQgdGhlIHN0dWZmIGluIHRoZSBjdXJyZW50IGRpciBieSBleGFtaW5pbmcgdGhlIHBhdGhuYW1lLgotICAgICAgICAgKiBJdCBuZWVkcyB0byBtYXRjaCB0aGUgcGFydGlhbCBwYXRobmFtZSBwcmVmaXgsIGFuZCBub3QgaGF2ZSBhICcvJwotICAgICAgICAgKiAoZnNzZXApIGFueXdoZXJlIGFmdGVyIHRoZSBwcmVmaXguCi0gICAgICAgICAqLwotICAgICAgICBmb3IgKGkgPSBzdGFydCsxOyBpIDwgY291bnQ7IGkrKykgewotICAgICAgICAgICAgaWYgKG1DYWNoZVtpXS5nZXRGaWxlTmFtZSgpLmxlbmd0aCgpID4gcGF0aC5sZW5ndGgoKSAmJgotICAgICAgICAgICAgICAgIHN0cm5jbXAobUNhY2hlW2ldLmdldEZpbGVOYW1lKCkuc3RyaW5nKCksIHBhdGguc3RyaW5nKCksIHBhdGgubGVuZ3RoKCkpID09IDApCi0gICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgY29uc3QgY2hhciogbmFtZSA9IG1DYWNoZVtpXS5nZXRGaWxlTmFtZSgpLnN0cmluZygpOwotICAgICAgICAgICAgICAgIC8vIFhYWCBUSElTIElTIEJST0tFTiEgIExvb2tzIGxpa2Ugd2UgbmVlZCB0byBzdG9yZSB0aGUgZnVsbAotICAgICAgICAgICAgICAgIC8vIHBhdGggcHJlZml4IHNlcGFyYXRlbHkgZnJvbSB0aGUgZmlsZSBwYXRoLgotICAgICAgICAgICAgICAgIGlmIChzdHJjaHIobmFtZSArIHBhdGgubGVuZ3RoKCksICcvJykgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICAvKiBncmFiIGl0LCByZWR1Y2luZyBwYXRoIHRvIGp1c3QgdGhlIGZpbGVuYW1lIGNvbXBvbmVudCAqLwotICAgICAgICAgICAgICAgICAgICBBc3NldERpcjo6RmlsZUluZm8gdG1wID0gbUNhY2hlW2ldOwotICAgICAgICAgICAgICAgICAgICB0bXAuc2V0RmlsZU5hbWUodG1wLmdldEZpbGVOYW1lKCkuZ2V0UGF0aExlYWYoKSk7Ci0gICAgICAgICAgICAgICAgICAgIHBDb250ZW50cy0+YWRkKHRtcCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAvKiBubyBsb25nZXIgaW4gdGhlIGRpciBvciBpdHMgc3ViZGlycyAqLwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBwYXRoID0gY3JlYXRlUGF0aE5hbWVMb2NrZWQoYXAsIHJvb3REaXIpOwotICAgICAgICBpZiAoZGlyTmFtZVswXSAhPSAnXDAnKQotICAgICAgICAgICAgcGF0aC5hcHBlbmRQYXRoKGRpck5hbWUpOwotICAgICAgICBwQ29udGVudHMgPSBzY2FuRGlyTG9ja2VkKHBhdGgpOwotICAgICAgICBpZiAocENvbnRlbnRzID09IE5VTEwpCi0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLy8gaWYgd2Ugd2FudGVkIHRvIGRvIGFuIGluY3JlbWVudGFsIGNhY2hlIGZpbGwsIHdlIHdvdWxkIGRvIGl0IGhlcmUKLQotICAgIC8qCi0gICAgICogUHJvY2VzcyAiZXhjbHVkZSIgZGlyZWN0aXZlcy4gIElmIHdlIGZpbmQgYSBmaWxlbmFtZSB0aGF0IGVuZHMgd2l0aAotICAgICAqICIuRVhDTFVERSIsIHdlIGxvb2sgZm9yIGEgbWF0Y2hpbmcgZW50cnkgaW4gdGhlICJtZXJnZWQiIHNldCwgYW5kCi0gICAgICogcmVtb3ZlIGl0IGlmIHdlIGZpbmQgaXQuICBXZSBhbHNvIGRlbGV0ZSB0aGUgImV4Y2x1ZGUiIGVudHJ5LgotICAgICAqLwotICAgIGludCBpLCBjb3VudCwgZXhjbEV4dExlbjsKLQotICAgIGNvdW50ID0gcENvbnRlbnRzLT5zaXplKCk7Ci0gICAgZXhjbEV4dExlbiA9IHN0cmxlbihrRXhjbHVkZUV4dGVuc2lvbik7Ci0gICAgZm9yIChpID0gMDsgaSA8IGNvdW50OyBpKyspIHsKLSAgICAgICAgY29uc3QgY2hhciogbmFtZTsKLSAgICAgICAgaW50IG5hbWVMZW47Ci0KLSAgICAgICAgbmFtZSA9IHBDb250ZW50cy0+aXRlbUF0KGkpLmdldEZpbGVOYW1lKCkuc3RyaW5nKCk7Ci0gICAgICAgIG5hbWVMZW4gPSBzdHJsZW4obmFtZSk7Ci0gICAgICAgIGlmIChuYW1lTGVuID4gZXhjbEV4dExlbiAmJgotICAgICAgICAgICAgc3RyY21wKG5hbWUgKyAobmFtZUxlbiAtIGV4Y2xFeHRMZW4pLCBrRXhjbHVkZUV4dGVuc2lvbikgPT0gMCkKLSAgICAgICAgewotICAgICAgICAgICAgU3RyaW5nOCBtYXRjaChuYW1lLCBuYW1lTGVuIC0gZXhjbEV4dExlbik7Ci0gICAgICAgICAgICBpbnQgbWF0Y2hJZHg7Ci0KLSAgICAgICAgICAgIG1hdGNoSWR4ID0gQXNzZXREaXI6OkZpbGVJbmZvOjpmaW5kRW50cnkocE1lcmdlZEluZm8sIG1hdGNoKTsKLSAgICAgICAgICAgIGlmIChtYXRjaElkeCA+IDApIHsKLSAgICAgICAgICAgICAgICBMT0dWKCJFeGNsdWRpbmcgJyVzJyBbJXNdXG4iLAotICAgICAgICAgICAgICAgICAgICBwTWVyZ2VkSW5mby0+aXRlbUF0KG1hdGNoSWR4KS5nZXRGaWxlTmFtZSgpLnN0cmluZygpLAotICAgICAgICAgICAgICAgICAgICBwTWVyZ2VkSW5mby0+aXRlbUF0KG1hdGNoSWR4KS5nZXRTb3VyY2VOYW1lKCkuc3RyaW5nKCkpOwotICAgICAgICAgICAgICAgIHBNZXJnZWRJbmZvLT5yZW1vdmVBdChtYXRjaElkeCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIC8vcHJpbnRmKCIrKysgbm8gbWF0Y2ggb24gJyVzJ1xuIiwgKGNvbnN0IGNoYXIqKSBtYXRjaCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIExPR0QoIkhFWTogc2l6ZT0lZCByZW1vdmluZyAlZFxuIiwgKGludClwQ29udGVudHMtPnNpemUoKSwgaSk7Ci0gICAgICAgICAgICBwQ29udGVudHMtPnJlbW92ZUF0KGkpOwotICAgICAgICAgICAgaS0tOyAgICAgICAgLy8gYWRqdXN0ICJmb3IiIGxvb3AKLSAgICAgICAgICAgIGNvdW50LS07ICAgIC8vICBhbmQgbG9vcCBsaW1pdAotICAgICAgICB9Ci0gICAgfQotCi0gICAgbWVyZ2VJbmZvTG9ja2VkKHBNZXJnZWRJbmZvLCBwQ29udGVudHMpOwotCi0gICAgZGVsZXRlIHBDb250ZW50czsKLQotICAgIHJldHVybiB0cnVlOwotfQotCi0vKgotICogU2NhbiB0aGUgY29udGVudHMgb2YgdGhlIHNwZWNpZmllZCBkaXJlY3RvcnksIGFuZCBzdHVmZiB3aGF0IHdlIGZpbmQKLSAqIGludG8gYSBuZXdseS1hbGxvY2F0ZWQgdmVjdG9yLgotICoKLSAqIEZpbGVzIGVuZGluZyBpbiAiLmd6IiB3aWxsIGhhdmUgdGhlaXIgZXh0ZW5zaW9ucyByZW1vdmVkLgotICoKLSAqIFdlIHNob3VsZCBwcm9iYWJseSB0aGluayBhYm91dCBza2lwcGluZyBmaWxlcyB3aXRoICJpbGxlZ2FsIiBuYW1lcywKLSAqIGUuZy4gaWxsZWdhbCBjaGFyYWN0ZXJzICgvXDopIG9yIGV4Y2Vzc2l2ZSBsZW5ndGguCi0gKgotICogUmV0dXJucyBOVUxMIGlmIHRoZSBzcGVjaWZpZWQgZGlyZWN0b3J5IGRvZXNuJ3QgZXhpc3QuCi0gKi8KLVNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+KiBBc3NldE1hbmFnZXI6OnNjYW5EaXJMb2NrZWQoY29uc3QgU3RyaW5nOCYgcGF0aCkKLXsKLSAgICBTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPiogcENvbnRlbnRzID0gTlVMTDsKLSAgICBESVIqIGRpcjsKLSAgICBzdHJ1Y3QgZGlyZW50KiBlbnRyeTsKLSAgICBGaWxlVHlwZSBmaWxlVHlwZTsKLQotICAgIExPR1YoIlNjYW5uaW5nIGRpciAnJXMnXG4iLCBwYXRoLnN0cmluZygpKTsKLQotICAgIGRpciA9IG9wZW5kaXIocGF0aC5zdHJpbmcoKSk7Ci0gICAgaWYgKGRpciA9PSBOVUxMKQotICAgICAgICByZXR1cm4gTlVMTDsKLQotICAgIHBDb250ZW50cyA9IG5ldyBTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPjsKLQotICAgIHdoaWxlICgxKSB7Ci0gICAgICAgIGVudHJ5ID0gcmVhZGRpcihkaXIpOwotICAgICAgICBpZiAoZW50cnkgPT0gTlVMTCkKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgIGlmIChzdHJjbXAoZW50cnktPmRfbmFtZSwgIi4iKSA9PSAwIHx8Ci0gICAgICAgICAgICBzdHJjbXAoZW50cnktPmRfbmFtZSwgIi4uIikgPT0gMCkKLSAgICAgICAgICAgIGNvbnRpbnVlOwotCi0jaWZkZWYgX0RJUkVOVF9IQVZFX0RfVFlQRQotICAgICAgICBpZiAoZW50cnktPmRfdHlwZSA9PSBEVF9SRUcpCi0gICAgICAgICAgICBmaWxlVHlwZSA9IGtGaWxlVHlwZVJlZ3VsYXI7Ci0gICAgICAgIGVsc2UgaWYgKGVudHJ5LT5kX3R5cGUgPT0gRFRfRElSKQotICAgICAgICAgICAgZmlsZVR5cGUgPSBrRmlsZVR5cGVEaXJlY3Rvcnk7Ci0gICAgICAgIGVsc2UKLSAgICAgICAgICAgIGZpbGVUeXBlID0ga0ZpbGVUeXBlVW5rbm93bjsKLSNlbHNlCi0gICAgICAgIC8vIHN0YXQgdGhlIGZpbGUKLSAgICAgICAgZmlsZVR5cGUgPSA6OmdldEZpbGVUeXBlKHBhdGguYXBwZW5kUGF0aENvcHkoZW50cnktPmRfbmFtZSkuc3RyaW5nKCkpOwotI2VuZGlmCi0KLSAgICAgICAgaWYgKGZpbGVUeXBlICE9IGtGaWxlVHlwZVJlZ3VsYXIgJiYgZmlsZVR5cGUgIT0ga0ZpbGVUeXBlRGlyZWN0b3J5KQotICAgICAgICAgICAgY29udGludWU7Ci0KLSAgICAgICAgQXNzZXREaXI6OkZpbGVJbmZvIGluZm87Ci0gICAgICAgIGluZm8uc2V0KFN0cmluZzgoZW50cnktPmRfbmFtZSksIGZpbGVUeXBlKTsKLSAgICAgICAgaWYgKHN0cmNhc2VjbXAoaW5mby5nZXRGaWxlTmFtZSgpLmdldFBhdGhFeHRlbnNpb24oKS5zdHJpbmcoKSwgIi5neiIpID09IDApCi0gICAgICAgICAgICBpbmZvLnNldEZpbGVOYW1lKGluZm8uZ2V0RmlsZU5hbWUoKS5nZXRCYXNlUGF0aCgpKTsKLSAgICAgICAgaW5mby5zZXRTb3VyY2VOYW1lKHBhdGguYXBwZW5kUGF0aENvcHkoaW5mby5nZXRGaWxlTmFtZSgpKSk7Ci0gICAgICAgIHBDb250ZW50cy0+YWRkKGluZm8pOwotICAgIH0KLQotICAgIGNsb3NlZGlyKGRpcik7Ci0gICAgcmV0dXJuIHBDb250ZW50czsKLX0KLQotLyoKLSAqIFNjYW4gdGhlIGNvbnRlbnRzIG91dCBvZiB0aGUgc3BlY2lmaWVkIFppcCBhcmNoaXZlLCBhbmQgbWVyZ2Ugd2hhdCB3ZQotICogZmluZCBpbnRvICJwTWVyZ2VkSW5mbyIuICBJZiB0aGUgWmlwIGFyY2hpdmUgaW4gcXVlc3Rpb24gZG9lc24ndCBleGlzdCwKLSAqIHdlIHJldHVybiBpbW1lZGlhdGVseS4KLSAqCi0gKiBSZXR1cm5zICJmYWxzZSIgaWYgd2UgZm91bmQgbm90aGluZyB0byBjb250cmlidXRlLgotICovCi1ib29sIEFzc2V0TWFuYWdlcjo6c2NhbkFuZE1lcmdlWmlwTG9ja2VkKFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+KiBwTWVyZ2VkSW5mbywKLSAgICBjb25zdCBhc3NldF9wYXRoJiBhcCwgY29uc3QgY2hhciogcm9vdERpciwgY29uc3QgY2hhciogYmFzZURpck5hbWUpCi17Ci0gICAgWmlwRmlsZVJPKiBwWmlwOwotICAgIFZlY3RvcjxTdHJpbmc4PiBkaXJzOwotICAgIEFzc2V0RGlyOjpGaWxlSW5mbyBpbmZvOwotICAgIFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+IGNvbnRlbnRzOwotICAgIFN0cmluZzggc291cmNlTmFtZSwgemlwTmFtZSwgZGlyTmFtZTsKLQotICAgIHBaaXAgPSBtWmlwU2V0LmdldFppcChhcC5wYXRoKTsKLSAgICBpZiAocFppcCA9PSBOVUxMKSB7Ci0gICAgICAgIExPR1coIkZhaWx1cmUgb3BlbmluZyB6aXAgJXNcbiIsIGFwLnBhdGguc3RyaW5nKCkpOwotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgemlwTmFtZSA9IFppcFNldDo6Z2V0UGF0aE5hbWUoYXAucGF0aC5zdHJpbmcoKSk7Ci0KLSAgICAvKiBjb252ZXJ0ICJzb3VuZHMiIHRvICJyb290RGlyL3NvdW5kcyIgKi8KLSAgICBpZiAocm9vdERpciAhPSBOVUxMKSBkaXJOYW1lID0gcm9vdERpcjsKLSAgICBkaXJOYW1lLmFwcGVuZFBhdGgoYmFzZURpck5hbWUpOwotCi0gICAgLyoKLSAgICAgKiBTY2FuIHRocm91Z2ggdGhlIGxpc3Qgb2YgZmlsZXMsIGxvb2tpbmcgZm9yIGEgbWF0Y2guICBUaGUgZmlsZXMgaW4KLSAgICAgKiB0aGUgWmlwIHRhYmxlIG9mIGNvbnRlbnRzIGFyZSBub3QgaW4gc29ydGVkIG9yZGVyLCBzbyB3ZSBoYXZlIHRvCi0gICAgICogcHJvY2VzcyB0aGUgZW50aXJlIGxpc3QuICBXZSdyZSBsb29raW5nIGZvciBhIHN0cmluZyB0aGF0IGJlZ2lucwotICAgICAqIHdpdGggdGhlIGNoYXJhY3RlcnMgaW4gImRpck5hbWUiLCBpcyBmb2xsb3dlZCBieSBhICcvJywgYW5kIGhhcyBubwotICAgICAqIHN1YnNlcXVlbnQgJy8nIGluIHRoZSBzdHVmZiB0aGF0IGZvbGxvd3MuCi0gICAgICoKLSAgICAgKiBXaGF0IG1ha2VzIHRoaXMgZXNwZWNpYWxseSBmdW4gaXMgdGhhdCBkaXJlY3RvcmllcyBhcmUgbm90IHN0b3JlZAotICAgICAqIGV4cGxpY2l0bHkgaW4gWmlwIGFyY2hpdmVzLCBzbyB3ZSBoYXZlIHRvIGluZmVyIHRoZW0gZnJvbSBjb250ZXh0LgotICAgICAqIFdoZW4gd2Ugc2VlICJzb3VuZHMvZm9vLndhdiIgd2UgaGF2ZSB0byBsZWF2ZSBhIG5vdGUgdG8gb3Vyc2VsdmVzCi0gICAgICogdG8gaW5zZXJ0IGEgZGlyZWN0b3J5IGNhbGxlZCAic291bmRzIiBpbnRvIHRoZSBsaXN0LiAgV2Ugc3RvcmUKLSAgICAgKiB0aGVzZSBpbiB0ZW1wb3JhcnkgdmVjdG9yIHNvIHRoYXQgd2Ugb25seSByZXR1cm4gZWFjaCBvbmUgb25jZS4KLSAgICAgKgotICAgICAqIE5hbWUgY29tcGFyaXNvbnMgYXJlIGNhc2Utc2Vuc2l0aXZlIHRvIG1hdGNoIFVOSVggZmlsZXN5c3RlbQotICAgICAqIHNlbWFudGljcy4KLSAgICAgKi8KLSAgICBpbnQgZGlyTmFtZUxlbiA9IGRpck5hbWUubGVuZ3RoKCk7Ci0gICAgZm9yIChpbnQgaSA9IDA7IGkgPCBwWmlwLT5nZXROdW1FbnRyaWVzKCk7IGkrKykgewotICAgICAgICBaaXBFbnRyeVJPIGVudHJ5OwotICAgICAgICBjaGFyIG5hbWVCdWZbMjU2XTsKLQotICAgICAgICBlbnRyeSA9IHBaaXAtPmZpbmRFbnRyeUJ5SW5kZXgoaSk7Ci0gICAgICAgIGlmIChwWmlwLT5nZXRFbnRyeUZpbGVOYW1lKGVudHJ5LCBuYW1lQnVmLCBzaXplb2YobmFtZUJ1ZikpICE9IDApIHsKLSAgICAgICAgICAgIC8vIFRPRE86IGZpeCB0aGlzIGlmIHdlIGV4cGVjdCB0byBoYXZlIGxvbmcgbmFtZXMKLSAgICAgICAgICAgIExPR0UoIkFSR0g6IG5hbWUgdG9vIGxvbmc/XG4iKTsKLSAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChkaXJOYW1lTGVuID09IDAgfHwKLSAgICAgICAgICAgIChzdHJuY21wKG5hbWVCdWYsIGRpck5hbWUuc3RyaW5nKCksIGRpck5hbWVMZW4pID09IDAgJiYKLSAgICAgICAgICAgICBuYW1lQnVmW2Rpck5hbWVMZW5dID09ICcvJykpCi0gICAgICAgIHsKLSAgICAgICAgICAgIGNvbnN0IGNoYXIqIGNwOwotICAgICAgICAgICAgY29uc3QgY2hhciogbmV4dFNsYXNoOwotCi0gICAgICAgICAgICBjcCA9IG5hbWVCdWYgKyBkaXJOYW1lTGVuOwotICAgICAgICAgICAgaWYgKGRpck5hbWVMZW4gIT0gMCkKLSAgICAgICAgICAgICAgICBjcCsrOyAgICAgICAvLyBhZHZhbmNlIHBhc3QgdGhlICcvJwotCi0gICAgICAgICAgICBuZXh0U2xhc2ggPSBzdHJjaHIoY3AsICcvJyk7Ci0vL3h4eCB0aGlzIG1heSBicmVhayBpZiB0aGVyZSBhcmUgYmFyZSBkaXJlY3RvcnkgZW50cmllcwotICAgICAgICAgICAgaWYgKG5leHRTbGFzaCA9PSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgLyogdGhpcyBpcyBhIGZpbGUgaW4gdGhlIHJlcXVlc3RlZCBkaXJlY3RvcnkgKi8KLQotICAgICAgICAgICAgICAgIGluZm8uc2V0KFN0cmluZzgobmFtZUJ1ZikuZ2V0UGF0aExlYWYoKSwga0ZpbGVUeXBlUmVndWxhcik7Ci0KLSAgICAgICAgICAgICAgICBpbmZvLnNldFNvdXJjZU5hbWUoCi0gICAgICAgICAgICAgICAgICAgIGNyZWF0ZVppcFNvdXJjZU5hbWVMb2NrZWQoemlwTmFtZSwgZGlyTmFtZSwgaW5mby5nZXRGaWxlTmFtZSgpKSk7Ci0KLSAgICAgICAgICAgICAgICBjb250ZW50cy5hZGQoaW5mbyk7Ci0gICAgICAgICAgICAgICAgLy9wcmludGYoIkZPVU5EOiBmaWxlICclcydcbiIsIChjb25zdCBjaGFyKikgaW5mby5tRmlsZU5hbWUpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAvKiB0aGlzIGlzIGEgc3ViZGlyOyBhZGQgaXQgaWYgd2UgZG9uJ3QgYWxyZWFkeSBoYXZlIGl0Ki8KLSAgICAgICAgICAgICAgICBTdHJpbmc4IHN1YmRpck5hbWUoY3AsIG5leHRTbGFzaCAtIGNwKTsKLSAgICAgICAgICAgICAgICBzaXplX3QgajsKLSAgICAgICAgICAgICAgICBzaXplX3QgTiA9IGRpcnMuc2l6ZSgpOwotCi0gICAgICAgICAgICAgICAgZm9yIChqID0gMDsgaiA8IE47IGorKykgewotICAgICAgICAgICAgICAgICAgICBpZiAoc3ViZGlyTmFtZSA9PSBkaXJzW2pdKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAoaiA9PSBOKSB7Ci0gICAgICAgICAgICAgICAgICAgIGRpcnMuYWRkKHN1YmRpck5hbWUpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIC8vcHJpbnRmKCJGT1VORDogZGlyICclcydcbiIsIChjb25zdCBjaGFyKikgc3ViZGlyTmFtZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKgotICAgICAqIEFkZCB0aGUgc2V0IG9mIHVuaXF1ZSBkaXJlY3Rvcmllcy4KLSAgICAgKi8KLSAgICBmb3IgKGludCBpID0gMDsgaSA8IChpbnQpIGRpcnMuc2l6ZSgpOyBpKyspIHsKLSAgICAgICAgaW5mby5zZXQoZGlyc1tpXSwga0ZpbGVUeXBlRGlyZWN0b3J5KTsKLSAgICAgICAgaW5mby5zZXRTb3VyY2VOYW1lKAotICAgICAgICAgICAgY3JlYXRlWmlwU291cmNlTmFtZUxvY2tlZCh6aXBOYW1lLCBkaXJOYW1lLCBpbmZvLmdldEZpbGVOYW1lKCkpKTsKLSAgICAgICAgY29udGVudHMuYWRkKGluZm8pOwotICAgIH0KLQotICAgIG1lcmdlSW5mb0xvY2tlZChwTWVyZ2VkSW5mbywgJmNvbnRlbnRzKTsKLQotICAgIHJldHVybiB0cnVlOwotfQotCi0KLS8qCi0gKiBNZXJnZSB0d28gdmVjdG9ycyBvZiBGaWxlSW5mby4KLSAqCi0gKiBUaGUgbWVyZ2VkIGNvbnRlbnRzIHdpbGwgYmUgc3R1ZmZlZCBpbnRvICpwTWVyZ2VkSW5mby4KLSAqCi0gKiBJZiBhbiBlbnRyeSBmb3IgYSBmaWxlIGV4aXN0cyBpbiBib3RoICJwTWVyZ2VkSW5mbyIgYW5kICJwQ29udGVudHMiLAotICogd2UgdXNlIHRoZSBuZXdlciAicENvbnRlbnRzIiBlbnRyeS4KLSAqLwotdm9pZCBBc3NldE1hbmFnZXI6Om1lcmdlSW5mb0xvY2tlZChTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPiogcE1lcmdlZEluZm8sCi0gICAgY29uc3QgU29ydGVkVmVjdG9yPEFzc2V0RGlyOjpGaWxlSW5mbz4qIHBDb250ZW50cykKLXsKLSAgICAvKgotICAgICAqIE1lcmdlIHdoYXQgd2UgZm91bmQgaW4gdGhpcyBkaXJlY3Rvcnkgd2l0aCB3aGF0IHdlIGZvdW5kIGluCi0gICAgICogb3RoZXIgcGxhY2VzLgotICAgICAqCi0gICAgICogVHdvIGJhc2ljIGFwcHJvYWNoZXM6Ci0gICAgICogKDEpIENyZWF0ZSBhIG5ldyBhcnJheSB0aGF0IGhvbGRzIHRoZSB1bmlxdWUgdmFsdWVzIG9mIHRoZSB0d28KLSAgICAgKiAgICAgYXJyYXlzLgotICAgICAqICgyKSBUYWtlIHRoZSBlbGVtZW50cyBmcm9tIHBDb250ZW50cyBhbmQgc2hvdmUgdGhlbSBpbnRvIHBNZXJnZWRJbmZvLgotICAgICAqCi0gICAgICogQmVjYXVzZSB0aGVzZSBhcmUgdmVjdG9ycyBvZiBjb21wbGV4IG9iamVjdHMsIG1vdmluZyBlbGVtZW50cyBhcm91bmQKLSAgICAgKiBpbnNpZGUgdGhlIHZlY3RvciByZXF1aXJlcyBjb25zdHJ1Y3RpbmcgbmV3IG9iamVjdHMgYW5kIGFsbG9jYXRpbmcKLSAgICAgKiBzdG9yYWdlIGZvciBtZW1iZXJzLiAgV2l0aCBhcHByb2FjaCAjMSwgd2UncmUgYWx3YXlzIGFkZGluZyB0byB0aGUKLSAgICAgKiBlbmQsIHdoZXJlYXMgd2l0aCAjMiB3ZSBjb3VsZCBiZSBpbnNlcnRpbmcgbXVsdGlwbGUgZWxlbWVudHMgYXQgdGhlCi0gICAgICogZnJvbnQgb2YgdGhlIHZlY3Rvci4gIEFwcHJvYWNoICMxIHJlcXVpcmVzIGEgZnVsbCBjb3B5IG9mIHRoZQotICAgICAqIGNvbnRlbnRzIG9mIHBNZXJnZWRJbmZvLCBidXQgYXBwcm9hY2ggIzIgcmVxdWlyZXMgdGhlIHNhbWUgY29weSBmb3IKLSAgICAgKiBldmVyeSBpbnNlcnRpb24gYXQgdGhlIGZyb250IG9mIHBNZXJnZWRJbmZvLgotICAgICAqCi0gICAgICogKFdlIHNob3VsZCBwcm9iYWJseSB1c2UgYSBTb3J0ZWRWZWN0b3IgaW50ZXJmYWNlIHRoYXQgYWxsb3dzIHVzIHRvCi0gICAgICoganVzdCBzdHVmZiBpdGVtcyBpbiwgdHJ1c3RpbmcgdXMgdG8gbWFpbnRhaW4gdGhlIHNvcnQgb3JkZXIuKQotICAgICAqLwotICAgIFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+KiBwTmV3U29ydGVkOwotICAgIGludCBtZXJnZU1heCwgY29udE1heDsKLSAgICBpbnQgbWVyZ2VJZHgsIGNvbnRJZHg7Ci0KLSAgICBwTmV3U29ydGVkID0gbmV3IFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+OwotICAgIG1lcmdlTWF4ID0gcE1lcmdlZEluZm8tPnNpemUoKTsKLSAgICBjb250TWF4ID0gcENvbnRlbnRzLT5zaXplKCk7Ci0gICAgbWVyZ2VJZHggPSBjb250SWR4ID0gMDsKLQotICAgIHdoaWxlIChtZXJnZUlkeCA8IG1lcmdlTWF4IHx8IGNvbnRJZHggPCBjb250TWF4KSB7Ci0gICAgICAgIGlmIChtZXJnZUlkeCA9PSBtZXJnZU1heCkgewotICAgICAgICAgICAgLyogaGl0IGVuZCBvZiAibWVyZ2UiIGxpc3QsIGNvcHkgcmVzdCBvZiAiY29udGVudHMiICovCi0gICAgICAgICAgICBwTmV3U29ydGVkLT5hZGQocENvbnRlbnRzLT5pdGVtQXQoY29udElkeCkpOwotICAgICAgICAgICAgY29udElkeCsrOwotICAgICAgICB9IGVsc2UgaWYgKGNvbnRJZHggPT0gY29udE1heCkgewotICAgICAgICAgICAgLyogaGl0IGVuZCBvZiAiY29udCIgbGlzdCwgY29weSByZXN0IG9mICJtZXJnZSIgKi8KLSAgICAgICAgICAgIHBOZXdTb3J0ZWQtPmFkZChwTWVyZ2VkSW5mby0+aXRlbUF0KG1lcmdlSWR4KSk7Ci0gICAgICAgICAgICBtZXJnZUlkeCsrOwotICAgICAgICB9IGVsc2UgaWYgKHBNZXJnZWRJbmZvLT5pdGVtQXQobWVyZ2VJZHgpID09IHBDb250ZW50cy0+aXRlbUF0KGNvbnRJZHgpKQotICAgICAgICB7Ci0gICAgICAgICAgICAvKiBpdGVtcyBhcmUgaWRlbnRpY2FsLCBhZGQgbmV3ZXIgYW5kIGFkdmFuY2UgYm90aCBpbmRpY2VzICovCi0gICAgICAgICAgICBwTmV3U29ydGVkLT5hZGQocENvbnRlbnRzLT5pdGVtQXQoY29udElkeCkpOwotICAgICAgICAgICAgbWVyZ2VJZHgrKzsKLSAgICAgICAgICAgIGNvbnRJZHgrKzsKLSAgICAgICAgfSBlbHNlIGlmIChwTWVyZ2VkSW5mby0+aXRlbUF0KG1lcmdlSWR4KSA8IHBDb250ZW50cy0+aXRlbUF0KGNvbnRJZHgpKQotICAgICAgICB7Ci0gICAgICAgICAgICAvKiAibWVyZ2UiIGlzIGxvd2VyLCBhZGQgdGhhdCBvbmUgKi8KLSAgICAgICAgICAgIHBOZXdTb3J0ZWQtPmFkZChwTWVyZ2VkSW5mby0+aXRlbUF0KG1lcmdlSWR4KSk7Ci0gICAgICAgICAgICBtZXJnZUlkeCsrOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLyogImNvbnQiIGlzIGxvd2VyLCBhZGQgdGhhdCBvbmUgKi8KLSAgICAgICAgICAgIGFzc2VydChwQ29udGVudHMtPml0ZW1BdChjb250SWR4KSA8IHBNZXJnZWRJbmZvLT5pdGVtQXQobWVyZ2VJZHgpKTsKLSAgICAgICAgICAgIHBOZXdTb3J0ZWQtPmFkZChwQ29udGVudHMtPml0ZW1BdChjb250SWR4KSk7Ci0gICAgICAgICAgICBjb250SWR4Kys7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKgotICAgICAqIE92ZXJ3cml0ZSB0aGUgIm1lcmdlZCIgbGlzdCB3aXRoIHRoZSBuZXcgc3R1ZmYuCi0gICAgICovCi0gICAgKnBNZXJnZWRJbmZvID0gKnBOZXdTb3J0ZWQ7Ci0gICAgZGVsZXRlIHBOZXdTb3J0ZWQ7Ci0KLSNpZiAwICAgICAgIC8vIGZvciBWZWN0b3IsIHJhdGhlciB0aGFuIFNvcnRlZFZlY3RvcgotICAgIGludCBpLCBqOwotICAgIGZvciAoaSA9IHBDb250ZW50cy0+c2l6ZSgpIC0xOyBpID49IDA7IGktLSkgewotICAgICAgICBib29sIGFkZCA9IHRydWU7Ci0KLSAgICAgICAgZm9yIChqID0gcE1lcmdlZEluZm8tPnNpemUoKSAtMTsgaiA+PSAwOyBqLS0pIHsKLSAgICAgICAgICAgIC8qIGNhc2Utc2Vuc2l0aXZlIGNvbXBhcmlzb25zLCB0byBiZWhhdmUgbGlrZSBVTklYIGZzICovCi0gICAgICAgICAgICBpZiAoc3RyY21wKHBDb250ZW50cy0+aXRlbUF0KGkpLm1GaWxlTmFtZSwKLSAgICAgICAgICAgICAgICAgICAgICAgcE1lcmdlZEluZm8tPml0ZW1BdChqKS5tRmlsZU5hbWUpID09IDApCi0gICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgLyogbWF0Y2gsIGRvbid0IGFkZCB0aGlzIGVudHJ5ICovCi0gICAgICAgICAgICAgICAgYWRkID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoYWRkKQotICAgICAgICAgICAgcE1lcmdlZEluZm8tPmFkZChwQ29udGVudHMtPml0ZW1BdChpKSk7Ci0gICAgfQotI2VuZGlmCi19Ci0KLQotLyoKLSAqIExvYWQgYWxsIGZpbGVzIGludG8gdGhlIGZpbGUgbmFtZSBjYWNoZS4gIFdlIHdhbnQgdG8gZG8gdGhpcyBhY3Jvc3MKLSAqIGFsbCBjb21iaW5hdGlvbnMgb2YgeyBhcHBuYW1lLCBsb2NhbGUsIHZlbmRvciB9LCBwZXJmb3JtaW5nIGEgcmVjdXJzaXZlCi0gKiBkaXJlY3RvcnkgdHJhdmVyc2FsLgotICoKLSAqIFRoaXMgaXMgbm90IHRoZSBtb3N0IGVmZmljaWVudCBkYXRhIHN0cnVjdHVyZS4gIEFsc28sIGdhdGhlcmluZyB0aGUKLSAqIGluZm9ybWF0aW9uIGFzIHdlIG5lZWRlZCBpdCAoZmlsZS1ieS1maWxlIG9yIGRpcmVjdG9yeS1ieS1kaXJlY3RvcnkpCi0gKiB3b3VsZCBiZSBmYXN0ZXIuICBIb3dldmVyLCBvbiB0aGUgYWN0dWFsIGRldmljZSwgOTklIG9mIHRoZSBmaWxlcyB3aWxsCi0gKiBsaXZlIGluIFppcCBhcmNoaXZlcywgc28gdGhpcyBsaXN0IHdpbGwgYmUgdmVyeSBzbWFsbC4gIFRoZSB0cm91YmxlCi0gKiBpcyB0aGF0IHdlIGhhdmUgdG8gY2hlY2sgdGhlICJsb29zZSIgZmlsZXMgZmlyc3QsIHNvIGl0J3MgaW1wb3J0YW50Ci0gKiB0aGF0IHdlIGRvbid0IGJlYXQgdGhlIGZpbGVzeXN0ZW0gc2lsbHkgbG9va2luZyBmb3IgZmlsZXMgdGhhdCBhcmVuJ3QKLSAqIHRoZXJlLgotICoKLSAqIE5vdGUgb24gdGhyZWFkIHNhZmV0eTogdGhpcyBpcyB0aGUgb25seSBmdW5jdGlvbiB0aGF0IGNhdXNlcyB1cGRhdGVzCi0gKiB0byBtQ2FjaGUsIGFuZCBhbnlib2R5IHdobyB0cmllcyB0byB1c2UgaXQgd2lsbCBjYWxsIGhlcmUgaWYgIW1DYWNoZVZhbGlkLAotICogc28gd2UgbmVlZCB0byBlbXBsb3kgYSBtdXRleCBoZXJlLgotICovCi12b2lkIEFzc2V0TWFuYWdlcjo6bG9hZEZpbGVOYW1lQ2FjaGVMb2NrZWQodm9pZCkKLXsKLSAgICBhc3NlcnQoIW1DYWNoZVZhbGlkKTsKLSAgICBhc3NlcnQobUNhY2hlLnNpemUoKSA9PSAwKTsKLQotI2lmZGVmIERPX1RJTUlOR1MgICAvLyBuZWVkIHRvIGxpbmsgYWdhaW5zdCAtbHJ0IGZvciB0aGlzIG5vdwotICAgIER1cmF0aW9uVGltZXIgdGltZXI7Ci0gICAgdGltZXIuc3RhcnQoKTsKLSNlbmRpZgotCi0gICAgZm5jU2NhbkxvY2tlZCgmbUNhY2hlLCAiIik7Ci0KLSNpZmRlZiBET19USU1JTkdTCi0gICAgdGltZXIuc3RvcCgpOwotICAgIExPR0QoIkNhY2hlIHNjYW4gdG9vayAlLjNmbXNcbiIsCi0gICAgICAgIHRpbWVyLmR1cmF0aW9uVXNlY3MoKSAvIDEwMDAuMCk7Ci0jZW5kaWYKLQotI2lmIDAKLSAgICBpbnQgaTsKLSAgICBwcmludGYoIkNBQ0hFRCBGSUxFIExJU1QgKCVkIGVudHJpZXMpOlxuIiwgbUNhY2hlLnNpemUoKSk7Ci0gICAgZm9yIChpID0gMDsgaSA8IChpbnQpIG1DYWNoZS5zaXplKCk7IGkrKykgewotICAgICAgICBwcmludGYoIiAlZDogKCVkKSAnJXMnXG4iLCBpLAotICAgICAgICAgICAgbUNhY2hlLml0ZW1BdChpKS5nZXRGaWxlVHlwZSgpLAotICAgICAgICAgICAgKGNvbnN0IGNoYXIqKSBtQ2FjaGUuaXRlbUF0KGkpLmdldEZpbGVOYW1lKCkpOwotICAgIH0KLSNlbmRpZgotCi0gICAgbUNhY2hlVmFsaWQgPSB0cnVlOwotfQotCi0vKgotICogU2NhbiB1cCB0byA4IHZlcnNpb25zIG9mIHRoZSBzcGVjaWZpZWQgZGlyZWN0b3J5LgotICovCi12b2lkIEFzc2V0TWFuYWdlcjo6Zm5jU2NhbkxvY2tlZChTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPiogcE1lcmdlZEluZm8sCi0gICAgY29uc3QgY2hhciogZGlyTmFtZSkKLXsKLSAgICBzaXplX3QgaSA9IG1Bc3NldFBhdGhzLnNpemUoKTsKLSAgICB3aGlsZSAoaSA+IDApIHsKLSAgICAgICAgaS0tOwotICAgICAgICBjb25zdCBhc3NldF9wYXRoJiBhcCA9IG1Bc3NldFBhdGhzLml0ZW1BdChpKTsKLSAgICAgICAgZm5jU2NhbkFuZE1lcmdlRGlyTG9ja2VkKHBNZXJnZWRJbmZvLCBhcCwgTlVMTCwgTlVMTCwgZGlyTmFtZSk7Ci0gICAgICAgIGlmIChtTG9jYWxlICE9IE5VTEwpCi0gICAgICAgICAgICBmbmNTY2FuQW5kTWVyZ2VEaXJMb2NrZWQocE1lcmdlZEluZm8sIGFwLCBtTG9jYWxlLCBOVUxMLCBkaXJOYW1lKTsKLSAgICAgICAgaWYgKG1WZW5kb3IgIT0gTlVMTCkKLSAgICAgICAgICAgIGZuY1NjYW5BbmRNZXJnZURpckxvY2tlZChwTWVyZ2VkSW5mbywgYXAsIE5VTEwsIG1WZW5kb3IsIGRpck5hbWUpOwotICAgICAgICBpZiAobUxvY2FsZSAhPSBOVUxMICYmIG1WZW5kb3IgIT0gTlVMTCkKLSAgICAgICAgICAgIGZuY1NjYW5BbmRNZXJnZURpckxvY2tlZChwTWVyZ2VkSW5mbywgYXAsIG1Mb2NhbGUsIG1WZW5kb3IsIGRpck5hbWUpOwotICAgIH0KLX0KLQotLyoKLSAqIFJlY3Vyc2l2ZWx5IHNjYW4gdGhpcyBkaXJlY3RvcnkgYW5kIGFsbCBzdWJkaXJzLgotICoKLSAqIFRoaXMgaXMgc2ltaWxhciB0byBzY2FuQW5kTWVyZ2VEaXIsIGJ1dCB3ZSBkb24ndCByZW1vdmUgdGhlIC5FWENMVURFCi0gKiBmaWxlcywgYW5kIHdlIHByZXBlbmQgdGhlIGV4dGVuZGVkIHBhcnRpYWwgcGF0aCB0byB0aGUgZmlsZW5hbWVzLgotICovCi1ib29sIEFzc2V0TWFuYWdlcjo6Zm5jU2NhbkFuZE1lcmdlRGlyTG9ja2VkKAotICAgIFNvcnRlZFZlY3RvcjxBc3NldERpcjo6RmlsZUluZm8+KiBwTWVyZ2VkSW5mbywKLSAgICBjb25zdCBhc3NldF9wYXRoJiBhcCwgY29uc3QgY2hhciogbG9jYWxlLCBjb25zdCBjaGFyKiB2ZW5kb3IsCi0gICAgY29uc3QgY2hhciogZGlyTmFtZSkKLXsKLSAgICBTb3J0ZWRWZWN0b3I8QXNzZXREaXI6OkZpbGVJbmZvPiogcENvbnRlbnRzOwotICAgIFN0cmluZzggcGFydGlhbFBhdGg7Ci0gICAgU3RyaW5nOCBmdWxsUGF0aDsKLQotICAgIC8vIFhYWCBUaGlzIGlzIGJyb2tlbiAtLSB0aGUgZmlsZW5hbWUgY2FjaGUgbmVlZHMgdG8gaG9sZCB0aGUgYmFzZQotICAgIC8vIGFzc2V0IHBhdGggc2VwYXJhdGVseSBmcm9tIGl0cyBmaWxlbmFtZS4KLSAgICAKLSAgICBwYXJ0aWFsUGF0aCA9IGNyZWF0ZVBhdGhOYW1lTG9ja2VkKGFwLCBsb2NhbGUsIHZlbmRvcik7Ci0gICAgaWYgKGRpck5hbWVbMF0gIT0gJ1wwJykgewotICAgICAgICBwYXJ0aWFsUGF0aC5hcHBlbmRQYXRoKGRpck5hbWUpOwotICAgIH0KLQotICAgIGZ1bGxQYXRoID0gcGFydGlhbFBhdGg7Ci0gICAgcENvbnRlbnRzID0gc2NhbkRpckxvY2tlZChmdWxsUGF0aCk7Ci0gICAgaWYgKHBDb250ZW50cyA9PSBOVUxMKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsgICAgICAgLy8gZGlyZWN0b3J5IGRpZCBub3QgZXhpc3QKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFNjYW4gYWxsIHN1YmRpcmVjdG9yaWVzIG9mIHRoZSBjdXJyZW50IGRpciwgbWVyZ2luZyB3aGF0IHdlIGZpbmQKLSAgICAgKiBpbnRvICJwTWVyZ2VkSW5mbyIuCi0gICAgICovCi0gICAgZm9yIChpbnQgaSA9IDA7IGkgPCAoaW50KSBwQ29udGVudHMtPnNpemUoKTsgaSsrKSB7Ci0gICAgICAgIGlmIChwQ29udGVudHMtPml0ZW1BdChpKS5nZXRGaWxlVHlwZSgpID09IGtGaWxlVHlwZURpcmVjdG9yeSkgewotICAgICAgICAgICAgU3RyaW5nOCBzdWJkaXIoZGlyTmFtZSk7Ci0gICAgICAgICAgICBzdWJkaXIuYXBwZW5kUGF0aChwQ29udGVudHMtPml0ZW1BdChpKS5nZXRGaWxlTmFtZSgpKTsKLQotICAgICAgICAgICAgZm5jU2NhbkFuZE1lcmdlRGlyTG9ja2VkKHBNZXJnZWRJbmZvLCBhcCwgbG9jYWxlLCB2ZW5kb3IsIHN1YmRpci5zdHJpbmcoKSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFRvIGJlIGNvbnNpc3RlbnQsIHdlIHdhbnQgZW50cmllcyBmb3IgdGhlIHJvb3QgZGlyZWN0b3J5LiAgSWYKLSAgICAgKiB3ZSdyZSB0aGUgcm9vdCwgYWRkIG9uZSBub3cuCi0gICAgICovCi0gICAgaWYgKGRpck5hbWVbMF0gPT0gJ1wwJykgewotICAgICAgICBBc3NldERpcjo6RmlsZUluZm8gdG1wSW5mbzsKLQotICAgICAgICB0bXBJbmZvLnNldChTdHJpbmc4KCIiKSwga0ZpbGVUeXBlRGlyZWN0b3J5KTsKLSAgICAgICAgdG1wSW5mby5zZXRTb3VyY2VOYW1lKGNyZWF0ZVBhdGhOYW1lTG9ja2VkKGFwLCBsb2NhbGUsIHZlbmRvcikpOwotICAgICAgICBwQ29udGVudHMtPmFkZCh0bXBJbmZvKTsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFdlIHdhbnQgdG8gcHJlcGVuZCB0aGUgZXh0ZW5kZWQgcGFydGlhbCBwYXRoIHRvIGV2ZXJ5IGVudHJ5IGluCi0gICAgICogInBDb250ZW50cyIuICBJdCdzIHRoZSBzYW1lIHZhbHVlIGZvciBlYWNoIGVudHJ5LCBzbyB0aGlzIHdpbGwKLSAgICAgKiBub3QgY2hhbmdlIHRoZSBzb3J0aW5nIG9yZGVyIG9mIHRoZSB2ZWN0b3IgY29udGVudHMuCi0gICAgICovCi0gICAgZm9yIChpbnQgaSA9IDA7IGkgPCAoaW50KSBwQ29udGVudHMtPnNpemUoKTsgaSsrKSB7Ci0gICAgICAgIGNvbnN0IEFzc2V0RGlyOjpGaWxlSW5mbyYgaW5mbyA9IHBDb250ZW50cy0+aXRlbUF0KGkpOwotICAgICAgICBwQ29udGVudHMtPmVkaXRJdGVtQXQoaSkuc2V0RmlsZU5hbWUocGFydGlhbFBhdGguYXBwZW5kUGF0aENvcHkoaW5mby5nZXRGaWxlTmFtZSgpKSk7Ci0gICAgfQotCi0gICAgbWVyZ2VJbmZvTG9ja2VkKHBNZXJnZWRJbmZvLCBwQ29udGVudHMpOwotICAgIHJldHVybiB0cnVlOwotfQotCi0vKgotICogVHJhc2ggdGhlIGNhY2hlLgotICovCi12b2lkIEFzc2V0TWFuYWdlcjo6cHVyZ2VGaWxlTmFtZUNhY2hlTG9ja2VkKHZvaWQpCi17Ci0gICAgbUNhY2hlVmFsaWQgPSBmYWxzZTsKLSAgICBtQ2FjaGUuY2xlYXIoKTsKLX0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBBc3NldE1hbmFnZXI6OlNoYXJlZFppcAotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKi8KLQotCi1NdXRleCBBc3NldE1hbmFnZXI6OlNoYXJlZFppcDo6Z0xvY2s7Ci1EZWZhdWx0S2V5ZWRWZWN0b3I8U3RyaW5nOCwgd3A8QXNzZXRNYW5hZ2VyOjpTaGFyZWRaaXA+ID4gQXNzZXRNYW5hZ2VyOjpTaGFyZWRaaXA6OmdPcGVuOwotCi1Bc3NldE1hbmFnZXI6OlNoYXJlZFppcDo6U2hhcmVkWmlwKGNvbnN0IFN0cmluZzgmIHBhdGgsIHRpbWVfdCBtb2RXaGVuKQotICAgIDogbVBhdGgocGF0aCksIG1aaXBGaWxlKE5VTEwpLCBtTW9kV2hlbihtb2RXaGVuKSwgbVJlc291cmNlVGFibGVBc3NldChOVUxMKQotewotICAgIC8vTE9HSSgiQ3JlYXRpbmcgU2hhcmVkWmlwICVwICVzXG4iLCB0aGlzLCAoY29uc3QgY2hhciopbVBhdGgpOwotICAgIG1aaXBGaWxlID0gbmV3IFppcEZpbGVSTzsKLSAgICBMT0dWKCIrKysgb3BlbmluZyB6aXAgJyVzJ1xuIiwgbVBhdGguc3RyaW5nKCkpOwotICAgIGlmIChtWmlwRmlsZS0+b3BlbihtUGF0aC5zdHJpbmcoKSkgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgTE9HRCgiZmFpbGVkIHRvIG9wZW4gWmlwIGFyY2hpdmUgJyVzJ1xuIiwgbVBhdGguc3RyaW5nKCkpOwotICAgICAgICBkZWxldGUgbVppcEZpbGU7Ci0gICAgICAgIG1aaXBGaWxlID0gTlVMTDsKLSAgICB9Ci19Ci0KLXNwPEFzc2V0TWFuYWdlcjo6U2hhcmVkWmlwPiBBc3NldE1hbmFnZXI6OlNoYXJlZFppcDo6Z2V0KGNvbnN0IFN0cmluZzgmIHBhdGgpCi17Ci0gICAgQXV0b011dGV4IF9sKGdMb2NrKTsKLSAgICB0aW1lX3QgbW9kV2hlbiA9IGdldEZpbGVNb2REYXRlKHBhdGgpOwotICAgIHNwPFNoYXJlZFppcD4gemlwID0gZ09wZW4udmFsdWVGb3IocGF0aCkucHJvbW90ZSgpOwotICAgIGlmICh6aXAgIT0gTlVMTCAmJiB6aXAtPm1Nb2RXaGVuID09IG1vZFdoZW4pIHsKLSAgICAgICAgcmV0dXJuIHppcDsKLSAgICB9Ci0gICAgemlwID0gbmV3IFNoYXJlZFppcChwYXRoLCBtb2RXaGVuKTsKLSAgICBnT3Blbi5hZGQocGF0aCwgemlwKTsKLSAgICByZXR1cm4gemlwOwotCi19Ci0KLVppcEZpbGVSTyogQXNzZXRNYW5hZ2VyOjpTaGFyZWRaaXA6OmdldFppcCgpCi17Ci0gICAgcmV0dXJuIG1aaXBGaWxlOwotfQotCi1Bc3NldCogQXNzZXRNYW5hZ2VyOjpTaGFyZWRaaXA6OmdldFJlc291cmNlVGFibGVBc3NldCgpCi17Ci0gICAgTE9HVigiR2V0dGluZyBmcm9tIFNoYXJlZFppcCAlcCByZXNvdXJjZSBhc3NldCAlcFxuIiwgdGhpcywgbVJlc291cmNlVGFibGVBc3NldCk7Ci0gICAgcmV0dXJuIG1SZXNvdXJjZVRhYmxlQXNzZXQ7Ci19Ci0KLUFzc2V0KiBBc3NldE1hbmFnZXI6OlNoYXJlZFppcDo6c2V0UmVzb3VyY2VUYWJsZUFzc2V0KEFzc2V0KiBhc3NldCkKLXsKLSAgICB7Ci0gICAgICAgIEF1dG9NdXRleCBfbChnTG9jayk7Ci0gICAgICAgIGlmIChtUmVzb3VyY2VUYWJsZUFzc2V0ID09IE5VTEwpIHsKLSAgICAgICAgICAgIG1SZXNvdXJjZVRhYmxlQXNzZXQgPSBhc3NldDsKLSAgICAgICAgICAgIC8vIFRoaXMgaXMgbm90IHRocmVhZCBzYWZlIHRoZSBmaXJzdCB0aW1lIGl0IGlzIGNhbGxlZCwgc28KLSAgICAgICAgICAgIC8vIGRvIGl0IGhlcmUgd2l0aCB0aGUgZ2xvYmFsIGxvY2sgaGVsZC4KLSAgICAgICAgICAgIGFzc2V0LT5nZXRCdWZmZXIodHJ1ZSk7Ci0gICAgICAgICAgICByZXR1cm4gYXNzZXQ7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgZGVsZXRlIGFzc2V0OwotICAgIHJldHVybiBtUmVzb3VyY2VUYWJsZUFzc2V0OwotfQotCi1ib29sIEFzc2V0TWFuYWdlcjo6U2hhcmVkWmlwOjppc1VwVG9EYXRlKCkKLXsKLSAgICB0aW1lX3QgbW9kV2hlbiA9IGdldEZpbGVNb2REYXRlKG1QYXRoLnN0cmluZygpKTsKLSAgICByZXR1cm4gbU1vZFdoZW4gPT0gbW9kV2hlbjsKLX0KLQotQXNzZXRNYW5hZ2VyOjpTaGFyZWRaaXA6On5TaGFyZWRaaXAoKQotewotICAgIC8vTE9HSSgiRGVzdHJveWluZyBTaGFyZWRaaXAgJXAgJXNcbiIsIHRoaXMsIChjb25zdCBjaGFyKiltUGF0aCk7Ci0gICAgaWYgKG1SZXNvdXJjZVRhYmxlQXNzZXQgIT0gTlVMTCkgewotICAgICAgICBkZWxldGUgbVJlc291cmNlVGFibGVBc3NldDsKLSAgICB9Ci0gICAgaWYgKG1aaXBGaWxlICE9IE5VTEwpIHsKLSAgICAgICAgZGVsZXRlIG1aaXBGaWxlOwotICAgICAgICBMT0dWKCJDbG9zZWQgJyVzJ1xuIiwgbVBhdGguc3RyaW5nKCkpOwotICAgIH0KLX0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBBc3NldE1hbmFnZXI6OlppcFNldAotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKi8KLQotLyoKLSAqIENvbnN0cnVjdG9yLgotICovCi1Bc3NldE1hbmFnZXI6OlppcFNldDo6WmlwU2V0KHZvaWQpCi17Ci19Ci0KLS8qCi0gKiBEZXN0cnVjdG9yLiAgQ2xvc2UgYW55IG9wZW4gYXJjaGl2ZXMuCi0gKi8KLUFzc2V0TWFuYWdlcjo6WmlwU2V0Ojp+WmlwU2V0KHZvaWQpCi17Ci0gICAgc2l6ZV90IE4gPSBtWmlwRmlsZS5zaXplKCk7Ci0gICAgZm9yIChzaXplX3QgaSA9IDA7IGkgPCBOOyBpKyspCi0gICAgICAgIGNsb3NlWmlwKGkpOwotfQotCi0vKgotICogQ2xvc2UgYSBaaXAgZmlsZSBhbmQgcmVzZXQgdGhlIGVudHJ5LgotICovCi12b2lkIEFzc2V0TWFuYWdlcjo6WmlwU2V0OjpjbG9zZVppcChpbnQgaWR4KQotewotICAgIG1aaXBGaWxlLmVkaXRJdGVtQXQoaWR4KSA9IE5VTEw7Ci19Ci0KLQotLyoKLSAqIFJldHJpZXZlIHRoZSBhcHByb3ByaWF0ZSBaaXAgZmlsZSBmcm9tIHRoZSBzZXQuCi0gKi8KLVppcEZpbGVSTyogQXNzZXRNYW5hZ2VyOjpaaXBTZXQ6OmdldFppcChjb25zdCBTdHJpbmc4JiBwYXRoKQotewotICAgIGludCBpZHggPSBnZXRJbmRleChwYXRoKTsKLSAgICBzcDxTaGFyZWRaaXA+IHppcCA9IG1aaXBGaWxlW2lkeF07Ci0gICAgaWYgKHppcCA9PSBOVUxMKSB7Ci0gICAgICAgIHppcCA9IFNoYXJlZFppcDo6Z2V0KHBhdGgpOwotICAgICAgICBtWmlwRmlsZS5lZGl0SXRlbUF0KGlkeCkgPSB6aXA7Ci0gICAgfQotICAgIHJldHVybiB6aXAtPmdldFppcCgpOwotfQotCi1Bc3NldCogQXNzZXRNYW5hZ2VyOjpaaXBTZXQ6OmdldFppcFJlc291cmNlVGFibGUoY29uc3QgU3RyaW5nOCYgcGF0aCkKLXsKLSAgICBpbnQgaWR4ID0gZ2V0SW5kZXgocGF0aCk7Ci0gICAgc3A8U2hhcmVkWmlwPiB6aXAgPSBtWmlwRmlsZVtpZHhdOwotICAgIGlmICh6aXAgPT0gTlVMTCkgewotICAgICAgICB6aXAgPSBTaGFyZWRaaXA6OmdldChwYXRoKTsKLSAgICAgICAgbVppcEZpbGUuZWRpdEl0ZW1BdChpZHgpID0gemlwOwotICAgIH0KLSAgICByZXR1cm4gemlwLT5nZXRSZXNvdXJjZVRhYmxlQXNzZXQoKTsKLX0KLQotQXNzZXQqIEFzc2V0TWFuYWdlcjo6WmlwU2V0OjpzZXRaaXBSZXNvdXJjZVRhYmxlKGNvbnN0IFN0cmluZzgmIHBhdGgsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQXNzZXQqIGFzc2V0KQotewotICAgIGludCBpZHggPSBnZXRJbmRleChwYXRoKTsKLSAgICBzcDxTaGFyZWRaaXA+IHppcCA9IG1aaXBGaWxlW2lkeF07Ci0gICAgLy8gZG9lc24ndCBtYWtlIHNlbnNlIHRvIGNhbGwgYmVmb3JlIHByZXZpb3VzbHkgYWNjZXNzaW5nLgotICAgIHJldHVybiB6aXAtPnNldFJlc291cmNlVGFibGVBc3NldChhc3NldCk7Ci19Ci0KLS8qCi0gKiBHZW5lcmF0ZSB0aGUgcGFydGlhbCBwYXRobmFtZSBmb3IgdGhlIHNwZWNpZmllZCBhcmNoaXZlLiAgVGhlIGNhbGxlcgotICogZ2V0cyB0byBwcmVwZW5kIHRoZSBhc3NldCByb290IGRpcmVjdG9yeS4KLSAqCi0gKiBSZXR1cm5zIHNvbWV0aGluZyBsaWtlICJjb21tb24vZW4tVVMtbm9vZ2xlLmphciIuCi0gKi8KLS8qc3RhdGljKi8gU3RyaW5nOCBBc3NldE1hbmFnZXI6OlppcFNldDo6Z2V0UGF0aE5hbWUoY29uc3QgY2hhciogemlwUGF0aCkKLXsKLSAgICByZXR1cm4gU3RyaW5nOCh6aXBQYXRoKTsKLX0KLQotYm9vbCBBc3NldE1hbmFnZXI6OlppcFNldDo6aXNVcFRvRGF0ZSgpCi17Ci0gICAgY29uc3Qgc2l6ZV90IE4gPSBtWmlwRmlsZS5zaXplKCk7Ci0gICAgZm9yIChzaXplX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICBpZiAobVppcEZpbGVbaV0gIT0gTlVMTCAmJiAhbVppcEZpbGVbaV0tPmlzVXBUb0RhdGUoKSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiB0cnVlOwotfQotCi0vKgotICogQ29tcHV0ZSB0aGUgemlwIGZpbGUncyBpbmRleC4KLSAqCi0gKiAiYXBwTmFtZSIsICJsb2NhbGUiLCBhbmQgInZlbmRvciIgc2hvdWxkIGJlIHNldCB0byBOVUxMIHRvIGluZGljYXRlIHRoZQotICogZGVmYXVsdCBkaXJlY3RvcnkuCi0gKi8KLWludCBBc3NldE1hbmFnZXI6OlppcFNldDo6Z2V0SW5kZXgoY29uc3QgU3RyaW5nOCYgemlwKSBjb25zdAotewotICAgIGNvbnN0IHNpemVfdCBOID0gbVppcFBhdGguc2l6ZSgpOwotICAgIGZvciAoc2l6ZV90IGk9MDsgaTxOOyBpKyspIHsKLSAgICAgICAgaWYgKG1aaXBQYXRoW2ldID09IHppcCkgewotICAgICAgICAgICAgcmV0dXJuIGk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBtWmlwUGF0aC5hZGQoemlwKTsKLSAgICBtWmlwRmlsZS5hZGQoTlVMTCk7Ci0KLSAgICByZXR1cm4gbVppcFBhdGguc2l6ZSgpLTE7Ci19Ci0KZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvQmluZGVyLmNwcCBiL2xpYnMvdXRpbHMvQmluZGVyLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzdlNDY4NS4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL0JpbmRlci5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwyNDIgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDx1dGlscy9CaW5kZXIuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0F0b21pYy5oPgotI2luY2x1ZGUgPHV0aWxzL0JwQmluZGVyLmg+Ci0jaW5jbHVkZSA8dXRpbHMvSUludGVyZmFjZS5oPgotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3A8SUludGVyZmFjZT4gIElCaW5kZXI6OnF1ZXJ5TG9jYWxJbnRlcmZhY2UoY29uc3QgU3RyaW5nMTYmIGRlc2NyaXB0b3IpCi17Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLUJCaW5kZXIqIElCaW5kZXI6OmxvY2FsQmluZGVyKCkKLXsKLSAgICByZXR1cm4gTlVMTDsKLX0KLQotQnBCaW5kZXIqIElCaW5kZXI6OnJlbW90ZUJpbmRlcigpCi17Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLWJvb2wgSUJpbmRlcjo6Y2hlY2tTdWJjbGFzcyhjb25zdCB2b2lkKiAvKnN1YmNsYXNzSUQqLykgY29uc3QKLXsKLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBCQmluZGVyOjpFeHRyYXMKLXsKLXB1YmxpYzoKLSAgICBNdXRleCBtTG9jazsKLSAgICBCcEJpbmRlcjo6T2JqZWN0TWFuYWdlciBtT2JqZWN0czsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1CQmluZGVyOjpCQmluZGVyKCkKLSAgICA6IG1FeHRyYXMoTlVMTCkKLXsKLX0KLQotYm9vbCBCQmluZGVyOjppc0JpbmRlckFsaXZlKCkgY29uc3QKLXsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotc3RhdHVzX3QgQkJpbmRlcjo6cGluZ0JpbmRlcigpCi17Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1TdHJpbmcxNiBCQmluZGVyOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkgY29uc3QKLXsKLSAgICBMT0dXKCJyZWFjaGVkIEJCaW5kZXI6OmdldEludGVyZmFjZURlc2NyaXB0b3IgKHRoaXM9JXApIiwgdGhpcyk7Ci0gICAgcmV0dXJuIFN0cmluZzE2KCk7Ci19Ci0KLXN0YXR1c190IEJCaW5kZXI6OnRyYW5zYWN0KAotICAgIHVpbnQzMl90IGNvZGUsIGNvbnN0IFBhcmNlbCYgZGF0YSwgUGFyY2VsKiByZXBseSwgdWludDMyX3QgZmxhZ3MpCi17Ci0gICAgZGF0YS5zZXREYXRhUG9zaXRpb24oMCk7Ci0KLSAgICBzdGF0dXNfdCBlcnIgPSBOT19FUlJPUjsKLSAgICBzd2l0Y2ggKGNvZGUpIHsKLSAgICAgICAgY2FzZSBQSU5HX1RSQU5TQUNUSU9OOgotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIocGluZ0JpbmRlcigpKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgZXJyID0gb25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgfQotCi0gICAgaWYgKHJlcGx5ICE9IE5VTEwpIHsKLSAgICAgICAgcmVwbHktPnNldERhdGFQb3NpdGlvbigwKTsKLSAgICB9Ci0KLSAgICByZXR1cm4gZXJyOwotfQotCi1zdGF0dXNfdCBCQmluZGVyOjpsaW5rVG9EZWF0aCgKLSAgICBjb25zdCBzcDxEZWF0aFJlY2lwaWVudD4mIHJlY2lwaWVudCwgdm9pZCogY29va2llLCB1aW50MzJfdCBmbGFncykKLXsKLSAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci19Ci0KLXN0YXR1c190IEJCaW5kZXI6OnVubGlua1RvRGVhdGgoCi0gICAgY29uc3Qgd3A8RGVhdGhSZWNpcGllbnQ+JiByZWNpcGllbnQsIHZvaWQqIGNvb2tpZSwgdWludDMyX3QgZmxhZ3MsCi0gICAgd3A8RGVhdGhSZWNpcGllbnQ+KiBvdXRSZWNpcGllbnQpCi17Ci0gICAgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOwotfQotCi1zdGF0dXNfdCBCQmluZGVyOjpkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXZvaWQgQkJpbmRlcjo6YXR0YWNoT2JqZWN0KAotICAgIGNvbnN0IHZvaWQqIG9iamVjdElELCB2b2lkKiBvYmplY3QsIHZvaWQqIGNsZWFudXBDb29raWUsCi0gICAgb2JqZWN0X2NsZWFudXBfZnVuYyBmdW5jKQotewotICAgIEV4dHJhcyogZSA9IG1FeHRyYXM7Ci0KLSAgICBpZiAoIWUpIHsKLSAgICAgICAgZSA9IG5ldyBFeHRyYXM7Ci0gICAgICAgIGlmIChhbmRyb2lkX2F0b21pY19jbXB4Y2hnKDAsIHJlaW50ZXJwcmV0X2Nhc3Q8aW50MzJfdD4oZSksCi0gICAgICAgICAgICAgICAgcmVpbnRlcnByZXRfY2FzdDx2b2xhdGlsZSBpbnQzMl90Kj4oJm1FeHRyYXMpKSAhPSAwKSB7Ci0gICAgICAgICAgICBkZWxldGUgZTsKLSAgICAgICAgICAgIGUgPSBtRXh0cmFzOwotICAgICAgICB9Ci0gICAgICAgIGlmIChlID09IDApIHJldHVybjsgLy8gb3V0IG9mIG1lbW9yeQotICAgIH0KLQotICAgIEF1dG9NdXRleCBfbChlLT5tTG9jayk7Ci0gICAgZS0+bU9iamVjdHMuYXR0YWNoKG9iamVjdElELCBvYmplY3QsIGNsZWFudXBDb29raWUsIGZ1bmMpOwotfQotCi12b2lkKiBCQmluZGVyOjpmaW5kT2JqZWN0KGNvbnN0IHZvaWQqIG9iamVjdElEKSBjb25zdAotewotICAgIEV4dHJhcyogZSA9IG1FeHRyYXM7Ci0gICAgaWYgKCFlKSByZXR1cm4gTlVMTDsKLQotICAgIEF1dG9NdXRleCBfbChlLT5tTG9jayk7Ci0gICAgcmV0dXJuIGUtPm1PYmplY3RzLmZpbmQob2JqZWN0SUQpOwotfQotCi12b2lkIEJCaW5kZXI6OmRldGFjaE9iamVjdChjb25zdCB2b2lkKiBvYmplY3RJRCkKLXsKLSAgICBFeHRyYXMqIGUgPSBtRXh0cmFzOwotICAgIGlmICghZSkgcmV0dXJuOwotCi0gICAgQXV0b011dGV4IF9sKGUtPm1Mb2NrKTsKLSAgICBlLT5tT2JqZWN0cy5kZXRhY2gob2JqZWN0SUQpOwotfQotCi1CQmluZGVyKiBCQmluZGVyOjpsb2NhbEJpbmRlcigpCi17Ci0gICAgcmV0dXJuIHRoaXM7Ci19Ci0KLUJCaW5kZXI6On5CQmluZGVyKCkKLXsKLSAgICBpZiAobUV4dHJhcykgZGVsZXRlIG1FeHRyYXM7Ci19Ci0KLQotc3RhdHVzX3QgQkJpbmRlcjo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIHN3aXRjaCAoY29kZSkgewotICAgICAgICBjYXNlIElOVEVSRkFDRV9UUkFOU0FDVElPTjoKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZVN0cmluZzE2KGdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0KLSAgICAgICAgY2FzZSBEVU1QX1RSQU5TQUNUSU9OOiB7Ci0gICAgICAgICAgICBpbnQgZmQgPSBkYXRhLnJlYWRGaWxlRGVzY3JpcHRvcigpOwotICAgICAgICAgICAgaW50IGFyZ2MgPSBkYXRhLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgVmVjdG9yPFN0cmluZzE2PiBhcmdzOwotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBhcmdjICYmIGRhdGEuZGF0YUF2YWlsKCkgPiAwOyBpKyspIHsKLSAgICAgICAgICAgICAgIGFyZ3MuYWRkKGRhdGEucmVhZFN0cmluZzE2KCkpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGR1bXAoZmQsIGFyZ3MpOwotICAgICAgICB9Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gVU5LTk9XTl9UUkFOU0FDVElPTjsKLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1lbnVtIHsKLSAgICAvLyBUaGlzIGlzIHVzZWQgdG8gdHJhbnNmZXIgb3duZXJzaGlwIG9mIHRoZSByZW1vdGUgYmluZGVyIGZyb20KLSAgICAvLyB0aGUgQnBSZWZCYXNlIG9iamVjdCBob2xkaW5nIGl0ICh3aGVuIGl0IGlzIGNvbnN0cnVjdGVkKSwgdG8gdGhlCi0gICAgLy8gb3duZXIgb2YgdGhlIEJwUmVmQmFzZSBvYmplY3Qgd2hlbiBpdCBmaXJzdCBhY3F1aXJlcyB0aGF0IEJwUmVmQmFzZS4KLSAgICBrUmVtb3RlQWNxdWlyZWQgPSAweDAwMDAwMDAxCi19OwotCi1CcFJlZkJhc2U6OkJwUmVmQmFzZShjb25zdCBzcDxJQmluZGVyPiYgbykKLSAgICA6IG1SZW1vdGUoby5nZXQoKSksIG1SZWZzKE5VTEwpLCBtU3RhdGUoMCkKLXsKLSAgICBleHRlbmRPYmplY3RMaWZldGltZShPQkpFQ1RfTElGRVRJTUVfV0VBSyk7Ci0KLSAgICBpZiAobVJlbW90ZSkgewotICAgICAgICBtUmVtb3RlLT5pbmNTdHJvbmcodGhpcyk7ICAgICAgICAgICAvLyBSZW1vdmVkIG9uIGZpcnN0IEluY1N0cm9uZygpLgotICAgICAgICBtUmVmcyA9IG1SZW1vdGUtPmNyZWF0ZVdlYWsodGhpcyk7ICAvLyBIZWxkIGZvciBvdXIgZW50aXJlIGxpZmV0aW1lLgotICAgIH0KLX0KLQotQnBSZWZCYXNlOjp+QnBSZWZCYXNlKCkKLXsKLSAgICBpZiAobVJlbW90ZSkgewotICAgICAgICBpZiAoIShtU3RhdGUma1JlbW90ZUFjcXVpcmVkKSkgewotICAgICAgICAgICAgbVJlbW90ZS0+ZGVjU3Ryb25nKHRoaXMpOwotICAgICAgICB9Ci0gICAgICAgIG1SZWZzLT5kZWNXZWFrKHRoaXMpOwotICAgIH0KLX0KLQotdm9pZCBCcFJlZkJhc2U6Om9uRmlyc3RSZWYoKQotewotICAgIGFuZHJvaWRfYXRvbWljX29yKGtSZW1vdGVBY3F1aXJlZCwgJm1TdGF0ZSk7Ci19Ci0KLXZvaWQgQnBSZWZCYXNlOjpvbkxhc3RTdHJvbmdSZWYoY29uc3Qgdm9pZCogaWQpCi17Ci0gICAgaWYgKG1SZW1vdGUpIHsKLSAgICAgICAgbVJlbW90ZS0+ZGVjU3Ryb25nKHRoaXMpOwotICAgIH0KLX0KLQotYm9vbCBCcFJlZkJhc2U6Om9uSW5jU3Ryb25nQXR0ZW1wdGVkKHVpbnQzMl90IGZsYWdzLCBjb25zdCB2b2lkKiBpZCkKLXsKLSAgICByZXR1cm4gbVJlbW90ZSA/IG1SZWZzLT5hdHRlbXB0SW5jU3Ryb25nKHRoaXMpIDogZmFsc2U7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9CcEJpbmRlci5jcHAgYi9saWJzL3V0aWxzL0JwQmluZGVyLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjlhYjE5NS4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL0JwQmluZGVyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDM0OCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJCcEJpbmRlciIKLS8vI2RlZmluZSBMT0dfTkRFQlVHIDAKLQotI2luY2x1ZGUgPHV0aWxzL0JwQmluZGVyLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLQotLy8jdW5kZWYgTE9HVgotLy8jZGVmaW5lIExPR1YoLi4uKSBmcHJpbnRmKHN0ZGVyciwgX19WQV9BUkdTX18pCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUJwQmluZGVyOjpPYmplY3RNYW5hZ2VyOjpPYmplY3RNYW5hZ2VyKCkKLXsKLX0KLQotQnBCaW5kZXI6Ok9iamVjdE1hbmFnZXI6On5PYmplY3RNYW5hZ2VyKCkKLXsKLSAgICBraWxsKCk7Ci19Ci0KLXZvaWQgQnBCaW5kZXI6Ok9iamVjdE1hbmFnZXI6OmF0dGFjaCgKLSAgICBjb25zdCB2b2lkKiBvYmplY3RJRCwgdm9pZCogb2JqZWN0LCB2b2lkKiBjbGVhbnVwQ29va2llLAotICAgIElCaW5kZXI6Om9iamVjdF9jbGVhbnVwX2Z1bmMgZnVuYykKLXsKLSAgICBlbnRyeV90IGU7Ci0gICAgZS5vYmplY3QgPSBvYmplY3Q7Ci0gICAgZS5jbGVhbnVwQ29va2llID0gY2xlYW51cENvb2tpZTsKLSAgICBlLmZ1bmMgPSBmdW5jOwotCi0gICAgaWYgKG1PYmplY3RzLmluZGV4T2ZLZXkob2JqZWN0SUQpID49IDApIHsKLSAgICAgICAgTE9HRSgiVHJ5aW5nIHRvIGF0dGFjaCBvYmplY3QgSUQgJXAgdG8gYmluZGVyIE9iamVjdE1hbmFnZXIgJXAgd2l0aCBvYmplY3QgJXAsIGJ1dCBvYmplY3QgSUQgYWxyZWFkeSBpbiB1c2UiLAotICAgICAgICAgICAgICAgIG9iamVjdElELCB0aGlzLCAgb2JqZWN0KTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIG1PYmplY3RzLmFkZChvYmplY3RJRCwgZSk7Ci19Ci0KLXZvaWQqIEJwQmluZGVyOjpPYmplY3RNYW5hZ2VyOjpmaW5kKGNvbnN0IHZvaWQqIG9iamVjdElEKSBjb25zdAotewotICAgIGNvbnN0IHNzaXplX3QgaSA9IG1PYmplY3RzLmluZGV4T2ZLZXkob2JqZWN0SUQpOwotICAgIGlmIChpIDwgMCkgcmV0dXJuIE5VTEw7Ci0gICAgcmV0dXJuIG1PYmplY3RzLnZhbHVlQXQoaSkub2JqZWN0OwotfQotCi12b2lkIEJwQmluZGVyOjpPYmplY3RNYW5hZ2VyOjpkZXRhY2goY29uc3Qgdm9pZCogb2JqZWN0SUQpCi17Ci0gICAgbU9iamVjdHMucmVtb3ZlSXRlbShvYmplY3RJRCk7Ci19Ci0KLXZvaWQgQnBCaW5kZXI6Ok9iamVjdE1hbmFnZXI6OmtpbGwoKQotewotICAgIGNvbnN0IHNpemVfdCBOID0gbU9iamVjdHMuc2l6ZSgpOwotICAgIExPR1YoIktpbGxpbmcgJWQgb2JqZWN0cyBpbiBtYW5hZ2VyICVwIiwgTiwgdGhpcyk7Ci0gICAgZm9yIChzaXplX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICBjb25zdCBlbnRyeV90JiBlID0gbU9iamVjdHMudmFsdWVBdChpKTsKLSAgICAgICAgaWYgKGUuZnVuYyAhPSBOVUxMKSB7Ci0gICAgICAgICAgICBlLmZ1bmMobU9iamVjdHMua2V5QXQoaSksIGUub2JqZWN0LCBlLmNsZWFudXBDb29raWUpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgbU9iamVjdHMuY2xlYXIoKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUJwQmluZGVyOjpCcEJpbmRlcihpbnQzMl90IGhhbmRsZSkKLSAgICA6IG1IYW5kbGUoaGFuZGxlKQotICAgICwgbUFsaXZlKDEpCi0gICAgLCBtT2JpdHNTZW50KDApCi0gICAgLCBtT2JpdHVhcmllcyhOVUxMKQotewotICAgIExPR1YoIkNyZWF0aW5nIEJwQmluZGVyICVwIGhhbmRsZSAlZFxuIiwgdGhpcywgbUhhbmRsZSk7Ci0KLSAgICBleHRlbmRPYmplY3RMaWZldGltZShPQkpFQ1RfTElGRVRJTUVfV0VBSyk7Ci0gICAgSVBDVGhyZWFkU3RhdGU6OnNlbGYoKS0+aW5jV2Vha0hhbmRsZShoYW5kbGUpOwotfQotCi1TdHJpbmcxNiBCcEJpbmRlcjo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpIGNvbnN0Ci17Ci0gICAgU3RyaW5nMTYgcmVzOwotICAgIFBhcmNlbCBzZW5kLCByZXBseTsKLSAgICBzdGF0dXNfdCBlcnIgPSBjb25zdF9jYXN0PEJwQmluZGVyKj4odGhpcyktPnRyYW5zYWN0KAotICAgICAgICAgICAgSU5URVJGQUNFX1RSQU5TQUNUSU9OLCBzZW5kLCAmcmVwbHkpOwotICAgIGlmIChlcnIgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgcmVzID0gcmVwbHkucmVhZFN0cmluZzE2KCk7Ci0gICAgfQotICAgIHJldHVybiByZXM7Ci19Ci0KLWJvb2wgQnBCaW5kZXI6OmlzQmluZGVyQWxpdmUoKSBjb25zdAotewotICAgIHJldHVybiBtQWxpdmUgIT0gMDsKLX0KLQotc3RhdHVzX3QgQnBCaW5kZXI6OnBpbmdCaW5kZXIoKQotewotICAgIFBhcmNlbCBzZW5kOwotICAgIFBhcmNlbCByZXBseTsKLSAgICBzdGF0dXNfdCBlcnIgPSB0cmFuc2FjdChQSU5HX1RSQU5TQUNUSU9OLCBzZW5kLCAmcmVwbHkpOwotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHJldHVybiBlcnI7Ci0gICAgaWYgKHJlcGx5LmRhdGFTaXplKCkgPCBzaXplb2Yoc3RhdHVzX3QpKSByZXR1cm4gTk9UX0VOT1VHSF9EQVRBOwotICAgIHJldHVybiAoc3RhdHVzX3QpcmVwbHkucmVhZEludDMyKCk7Ci19Ci0KLXN0YXR1c190IEJwQmluZGVyOjpkdW1wKGludCBmZCwgY29uc3QgVmVjdG9yPFN0cmluZzE2PiYgYXJncykKLXsKLSAgICBQYXJjZWwgc2VuZDsKLSAgICBQYXJjZWwgcmVwbHk7Ci0gICAgc2VuZC53cml0ZUZpbGVEZXNjcmlwdG9yKGZkKTsKLSAgICBjb25zdCBzaXplX3QgbnVtQXJncyA9IGFyZ3Muc2l6ZSgpOwotICAgIHNlbmQud3JpdGVJbnQzMihudW1BcmdzKTsKLSAgICBmb3IgKHNpemVfdCBpID0gMDsgaSA8IG51bUFyZ3M7IGkrKykgewotICAgICAgICBzZW5kLndyaXRlU3RyaW5nMTYoYXJnc1tpXSk7Ci0gICAgfQotICAgIHN0YXR1c190IGVyciA9IHRyYW5zYWN0KERVTVBfVFJBTlNBQ1RJT04sIHNlbmQsICZyZXBseSk7Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotc3RhdHVzX3QgQnBCaW5kZXI6OnRyYW5zYWN0KAotICAgIHVpbnQzMl90IGNvZGUsIGNvbnN0IFBhcmNlbCYgZGF0YSwgUGFyY2VsKiByZXBseSwgdWludDMyX3QgZmxhZ3MpCi17Ci0gICAgLy8gT25jZSBhIGJpbmRlciBoYXMgZGllZCwgaXQgd2lsbCBuZXZlciBjb21lIGJhY2sgdG8gbGlmZS4KLSAgICBpZiAobUFsaXZlKSB7Ci0gICAgICAgIHN0YXR1c190IHN0YXR1cyA9IElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPnRyYW5zYWN0KAotICAgICAgICAgICAgbUhhbmRsZSwgY29kZSwgZGF0YSwgcmVwbHksIGZsYWdzKTsKLSAgICAgICAgaWYgKHN0YXR1cyA9PSBERUFEX09CSkVDVCkgbUFsaXZlID0gMDsKLSAgICAgICAgcmV0dXJuIHN0YXR1czsKLSAgICB9Ci0KLSAgICByZXR1cm4gREVBRF9PQkpFQ1Q7Ci19Ci0KLXN0YXR1c190IEJwQmluZGVyOjpsaW5rVG9EZWF0aCgKLSAgICBjb25zdCBzcDxEZWF0aFJlY2lwaWVudD4mIHJlY2lwaWVudCwgdm9pZCogY29va2llLCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBPYml0dWFyeSBvYjsKLSAgICBvYi5yZWNpcGllbnQgPSByZWNpcGllbnQ7Ci0gICAgb2IuY29va2llID0gY29va2llOwotICAgIG9iLmZsYWdzID0gZmxhZ3M7Ci0KLSAgICBMT0dfQUxXQVlTX0ZBVEFMX0lGKHJlY2lwaWVudCA9PSBOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICAgImxpbmtUb0RlYXRoKCk6IHJlY2lwaWVudCBtdXN0IGJlIG5vbi1OVUxMIik7Ci0KLSAgICB7Ci0gICAgICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0KLSAgICAgICAgaWYgKCFtT2JpdHNTZW50KSB7Ci0gICAgICAgICAgICBpZiAoIW1PYml0dWFyaWVzKSB7Ci0gICAgICAgICAgICAgICAgbU9iaXR1YXJpZXMgPSBuZXcgVmVjdG9yPE9iaXR1YXJ5PjsKLSAgICAgICAgICAgICAgICBpZiAoIW1PYml0dWFyaWVzKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIExPR1YoIlJlcXVlc3RpbmcgZGVhdGggbm90aWZpY2F0aW9uOiAlcCBoYW5kbGUgJWRcbiIsIHRoaXMsIG1IYW5kbGUpOwotICAgICAgICAgICAgICAgIGdldFdlYWtSZWZzKCktPmluY1dlYWsodGhpcyk7Ci0gICAgICAgICAgICAgICAgSVBDVGhyZWFkU3RhdGUqIHNlbGYgPSBJUENUaHJlYWRTdGF0ZTo6c2VsZigpOwotICAgICAgICAgICAgICAgIHNlbGYtPnJlcXVlc3REZWF0aE5vdGlmaWNhdGlvbihtSGFuZGxlLCB0aGlzKTsKLSAgICAgICAgICAgICAgICBzZWxmLT5mbHVzaENvbW1hbmRzKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBzc2l6ZV90IHJlcyA9IG1PYml0dWFyaWVzLT5hZGQob2IpOwotICAgICAgICAgICAgcmV0dXJuIHJlcyA+PSAoc3NpemVfdClOT19FUlJPUiA/IChzdGF0dXNfdClOT19FUlJPUiA6IHJlczsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBERUFEX09CSkVDVDsKLX0KLQotc3RhdHVzX3QgQnBCaW5kZXI6OnVubGlua1RvRGVhdGgoCi0gICAgY29uc3Qgd3A8RGVhdGhSZWNpcGllbnQ+JiByZWNpcGllbnQsIHZvaWQqIGNvb2tpZSwgdWludDMyX3QgZmxhZ3MsCi0gICAgd3A8RGVhdGhSZWNpcGllbnQ+KiBvdXRSZWNpcGllbnQpCi17Ci0gICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLQotICAgIGlmIChtT2JpdHNTZW50KSB7Ci0gICAgICAgIHJldHVybiBERUFEX09CSkVDVDsKLSAgICB9Ci0KLSAgICBjb25zdCBzaXplX3QgTiA9IG1PYml0dWFyaWVzID8gbU9iaXR1YXJpZXMtPnNpemUoKSA6IDA7Ci0gICAgZm9yIChzaXplX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICBjb25zdCBPYml0dWFyeSYgb2JpdCA9IG1PYml0dWFyaWVzLT5pdGVtQXQoaSk7Ci0gICAgICAgIGlmICgob2JpdC5yZWNpcGllbnQgPT0gcmVjaXBpZW50Ci0gICAgICAgICAgICAgICAgICAgIHx8IChyZWNpcGllbnQgPT0gTlVMTCAmJiBvYml0LmNvb2tpZSA9PSBjb29raWUpKQotICAgICAgICAgICAgICAgICYmIG9iaXQuZmxhZ3MgPT0gZmxhZ3MpIHsKLSAgICAgICAgICAgIGNvbnN0IHVpbnQzMl90IGFsbEZsYWdzID0gb2JpdC5mbGFnc3xmbGFnczsKLSAgICAgICAgICAgIGlmIChvdXRSZWNpcGllbnQgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICpvdXRSZWNpcGllbnQgPSBtT2JpdHVhcmllcy0+aXRlbUF0KGkpLnJlY2lwaWVudDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1PYml0dWFyaWVzLT5yZW1vdmVBdChpKTsKLSAgICAgICAgICAgIGlmIChtT2JpdHVhcmllcy0+c2l6ZSgpID09IDApIHsKLSAgICAgICAgICAgICAgICBMT0dWKCJDbGVhcmluZyBkZWF0aCBub3RpZmljYXRpb246ICVwIGhhbmRsZSAlZFxuIiwgdGhpcywgbUhhbmRsZSk7Ci0gICAgICAgICAgICAgICAgSVBDVGhyZWFkU3RhdGUqIHNlbGYgPSBJUENUaHJlYWRTdGF0ZTo6c2VsZigpOwotICAgICAgICAgICAgICAgIHNlbGYtPmNsZWFyRGVhdGhOb3RpZmljYXRpb24obUhhbmRsZSwgdGhpcyk7Ci0gICAgICAgICAgICAgICAgc2VsZi0+Zmx1c2hDb21tYW5kcygpOwotICAgICAgICAgICAgICAgIGRlbGV0ZSBtT2JpdHVhcmllczsKLSAgICAgICAgICAgICAgICBtT2JpdHVhcmllcyA9IE5VTEw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gTkFNRV9OT1RfRk9VTkQ7Ci19Ci0KLXZvaWQgQnBCaW5kZXI6OnNlbmRPYml0dWFyeSgpCi17Ci0gICAgTE9HVigiU2VuZGluZyBvYml0dWFyeSBmb3IgcHJveHkgJXAgaGFuZGxlICVkLCBtT2JpdHNTZW50PSVzXG4iLAotICAgICAgICB0aGlzLCBtSGFuZGxlLCBtT2JpdHNTZW50ID8gInRydWUiIDogImZhbHNlIik7Ci0KLSAgICBtQWxpdmUgPSAwOwotICAgIGlmIChtT2JpdHNTZW50KSByZXR1cm47Ci0KLSAgICBtTG9jay5sb2NrKCk7Ci0gICAgVmVjdG9yPE9iaXR1YXJ5Piogb2JpdHMgPSBtT2JpdHVhcmllczsKLSAgICBpZihvYml0cyAhPSBOVUxMKSB7Ci0gICAgICAgIExPR1YoIkNsZWFyaW5nIHNlbnQgZGVhdGggbm90aWZpY2F0aW9uOiAlcCBoYW5kbGUgJWRcbiIsIHRoaXMsIG1IYW5kbGUpOwotICAgICAgICBJUENUaHJlYWRTdGF0ZSogc2VsZiA9IElQQ1RocmVhZFN0YXRlOjpzZWxmKCk7Ci0gICAgICAgIHNlbGYtPmNsZWFyRGVhdGhOb3RpZmljYXRpb24obUhhbmRsZSwgdGhpcyk7Ci0gICAgICAgIHNlbGYtPmZsdXNoQ29tbWFuZHMoKTsKLSAgICAgICAgbU9iaXR1YXJpZXMgPSBOVUxMOwotICAgIH0KLSAgICBtT2JpdHNTZW50ID0gMTsKLSAgICBtTG9jay51bmxvY2soKTsKLQotICAgIExPR1YoIlJlcG9ydGluZyBkZWF0aCBvZiBwcm94eSAlcCBmb3IgJWQgcmVjaXBpZW50c1xuIiwKLSAgICAgICAgdGhpcywgb2JpdHMgPyBvYml0cy0+c2l6ZSgpIDogMCk7Ci0KLSAgICBpZiAob2JpdHMgIT0gTlVMTCkgewotICAgICAgICBjb25zdCBzaXplX3QgTiA9IG9iaXRzLT5zaXplKCk7Ci0gICAgICAgIGZvciAoc2l6ZV90IGk9MDsgaTxOOyBpKyspIHsKLSAgICAgICAgICAgIHJlcG9ydE9uZURlYXRoKG9iaXRzLT5pdGVtQXQoaSkpOwotICAgICAgICB9Ci0KLSAgICAgICAgZGVsZXRlIG9iaXRzOwotICAgIH0KLX0KLQotdm9pZCBCcEJpbmRlcjo6cmVwb3J0T25lRGVhdGgoY29uc3QgT2JpdHVhcnkmIG9iaXQpCi17Ci0gICAgc3A8RGVhdGhSZWNpcGllbnQ+IHJlY2lwaWVudCA9IG9iaXQucmVjaXBpZW50LnByb21vdGUoKTsKLSAgICBMT0dWKCJSZXBvcnRpbmcgZGVhdGggdG8gcmVjaXBpZW50OiAlcFxuIiwgcmVjaXBpZW50LmdldCgpKTsKLSAgICBpZiAocmVjaXBpZW50ID09IE5VTEwpIHJldHVybjsKLQotICAgIHJlY2lwaWVudC0+YmluZGVyRGllZCh0aGlzKTsKLX0KLQotCi12b2lkIEJwQmluZGVyOjphdHRhY2hPYmplY3QoCi0gICAgY29uc3Qgdm9pZCogb2JqZWN0SUQsIHZvaWQqIG9iamVjdCwgdm9pZCogY2xlYW51cENvb2tpZSwKLSAgICBvYmplY3RfY2xlYW51cF9mdW5jIGZ1bmMpCi17Ci0gICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLSAgICBMT0dWKCJBdHRhY2hpbmcgb2JqZWN0ICVwIHRvIGJpbmRlciAlcCAobWFuYWdlcj0lcCkiLCBvYmplY3QsIHRoaXMsICZtT2JqZWN0cyk7Ci0gICAgbU9iamVjdHMuYXR0YWNoKG9iamVjdElELCBvYmplY3QsIGNsZWFudXBDb29raWUsIGZ1bmMpOwotfQotCi12b2lkKiBCcEJpbmRlcjo6ZmluZE9iamVjdChjb25zdCB2b2lkKiBvYmplY3RJRCkgY29uc3QKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgIHJldHVybiBtT2JqZWN0cy5maW5kKG9iamVjdElEKTsKLX0KLQotdm9pZCBCcEJpbmRlcjo6ZGV0YWNoT2JqZWN0KGNvbnN0IHZvaWQqIG9iamVjdElEKQotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgbU9iamVjdHMuZGV0YWNoKG9iamVjdElEKTsKLX0KLQotQnBCaW5kZXIqIEJwQmluZGVyOjpyZW1vdGVCaW5kZXIoKQotewotICAgIHJldHVybiB0aGlzOwotfQotCi1CcEJpbmRlcjo6fkJwQmluZGVyKCkKLXsKLSAgICBMT0dWKCJEZXN0cm95aW5nIEJwQmluZGVyICVwIGhhbmRsZSAlZFxuIiwgdGhpcywgbUhhbmRsZSk7Ci0KLSAgICBJUENUaHJlYWRTdGF0ZSogaXBjID0gSVBDVGhyZWFkU3RhdGU6OnNlbGYoKTsKLQotICAgIG1Mb2NrLmxvY2soKTsKLSAgICBWZWN0b3I8T2JpdHVhcnk+KiBvYml0cyA9IG1PYml0dWFyaWVzOwotICAgIGlmKG9iaXRzICE9IE5VTEwpIHsKLSAgICAgICAgaWYgKGlwYykgaXBjLT5jbGVhckRlYXRoTm90aWZpY2F0aW9uKG1IYW5kbGUsIHRoaXMpOwotICAgICAgICBtT2JpdHVhcmllcyA9IE5VTEw7Ci0gICAgfQotICAgIG1Mb2NrLnVubG9jaygpOwotCi0gICAgaWYgKG9iaXRzICE9IE5VTEwpIHsKLSAgICAgICAgLy8gWFhYIFNob3VsZCB3ZSB0ZWxsIGFueSByZW1haW5pbmcgRGVhdGhSZWNpcGllbnQKLSAgICAgICAgLy8gb2JqZWN0cyB0aGF0IHRoZSBsYXN0IHN0cm9uZyByZWYgaGFzIGdvbmUgYXdheSwgc28gdGhleQotICAgICAgICAvLyBhcmUgbm8gbG9uZ2VyIGxpbmtlZD8KLSAgICAgICAgZGVsZXRlIG9iaXRzOwotICAgIH0KLQotICAgIGlmIChpcGMpIHsKLSAgICAgICAgaXBjLT5leHB1bmdlSGFuZGxlKG1IYW5kbGUsIHRoaXMpOwotICAgICAgICBpcGMtPmRlY1dlYWtIYW5kbGUobUhhbmRsZSk7Ci0gICAgfQotfQotCi12b2lkIEJwQmluZGVyOjpvbkZpcnN0UmVmKCkKLXsKLSAgICBMT0dWKCJvbkZpcnN0UmVmIEJwQmluZGVyICVwIGhhbmRsZSAlZFxuIiwgdGhpcywgbUhhbmRsZSk7Ci0gICAgSVBDVGhyZWFkU3RhdGUqIGlwYyA9IElQQ1RocmVhZFN0YXRlOjpzZWxmKCk7Ci0gICAgaWYgKGlwYykgaXBjLT5pbmNTdHJvbmdIYW5kbGUobUhhbmRsZSk7Ci19Ci0KLXZvaWQgQnBCaW5kZXI6Om9uTGFzdFN0cm9uZ1JlZihjb25zdCB2b2lkKiBpZCkKLXsKLSAgICBMT0dWKCJvbkxhc3RTdHJvbmdSZWYgQnBCaW5kZXIgJXAgaGFuZGxlICVkXG4iLCB0aGlzLCBtSGFuZGxlKTsKLSAgICBJRl9MT0dWKCkgewotICAgICAgICBwcmludFJlZnMoKTsKLSAgICB9Ci0gICAgSVBDVGhyZWFkU3RhdGUqIGlwYyA9IElQQ1RocmVhZFN0YXRlOjpzZWxmKCk7Ci0gICAgaWYgKGlwYykgaXBjLT5kZWNTdHJvbmdIYW5kbGUobUhhbmRsZSk7Ci19Ci0KLWJvb2wgQnBCaW5kZXI6Om9uSW5jU3Ryb25nQXR0ZW1wdGVkKHVpbnQzMl90IGZsYWdzLCBjb25zdCB2b2lkKiBpZCkKLXsKLSAgICBMT0dWKCJvbkluY1N0cm9uZ0F0dGVtcHRlZCBCcEJpbmRlciAlcCBoYW5kbGUgJWRcbiIsIHRoaXMsIG1IYW5kbGUpOwotICAgIElQQ1RocmVhZFN0YXRlKiBpcGMgPSBJUENUaHJlYWRTdGF0ZTo6c2VsZigpOwotICAgIHJldHVybiBpcGMgPyBpcGMtPmF0dGVtcHRJbmNTdHJvbmdIYW5kbGUobUhhbmRsZSkgPT0gTk9fRVJST1IgOiBmYWxzZTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL0J1ZmZlcmVkVGV4dE91dHB1dC5jcHAgYi9saWJzL3V0aWxzL0J1ZmZlcmVkVGV4dE91dHB1dC5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDk4OTY2MmUuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9CdWZmZXJlZFRleHRPdXRwdXQuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMjc5ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDYgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaW5jbHVkZSA8dXRpbHMvQnVmZmVyZWRUZXh0T3V0cHV0Lmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9EZWJ1Zy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLSNpbmNsdWRlIDx1dGlscy9WZWN0b3IuaD4KLSNpbmNsdWRlIDxjdXRpbHMvdGhyZWFkcy5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS91dGlscy9TdGF0aWMuaD4KLQotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLXN0cnVjdCBCdWZmZXJlZFRleHRPdXRwdXQ6OkJ1ZmZlclN0YXRlIDogcHVibGljIFJlZkJhc2UKLXsKLSAgICBCdWZmZXJTdGF0ZShpbnQzMl90IF9zZXEpCi0gICAgICAgIDogc2VxKF9zZXEpCi0gICAgICAgICwgYnVmZmVyKE5VTEwpCi0gICAgICAgICwgYnVmZmVyUG9zKDApCi0gICAgICAgICwgYnVmZmVyU2l6ZSgwKQotICAgICAgICAsIGF0RnJvbnQodHJ1ZSkKLSAgICAgICAgLCBpbmRlbnQoMCkKLSAgICAgICAgLCBidW5kbGUoMCkgewotICAgIH0KLSAgICB+QnVmZmVyU3RhdGUoKSB7Ci0gICAgICAgIGZyZWUoYnVmZmVyKTsKLSAgICB9Ci0gICAgCi0gICAgc3RhdHVzX3QgYXBwZW5kKGNvbnN0IGNoYXIqIHR4dCwgc2l6ZV90IGxlbikgewotICAgICAgICBpZiAoKGxlbitidWZmZXJQb3MpID4gYnVmZmVyU2l6ZSkgewotICAgICAgICAgICAgdm9pZCogYiA9IHJlYWxsb2MoYnVmZmVyLCAoKGxlbitidWZmZXJQb3MpKjMpLzIpOwotICAgICAgICAgICAgaWYgKCFiKSByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICAgICAgYnVmZmVyID0gKGNoYXIqKWI7Ci0gICAgICAgIH0KLSAgICAgICAgbWVtY3B5KGJ1ZmZlcitidWZmZXJQb3MsIHR4dCwgbGVuKTsKLSAgICAgICAgYnVmZmVyUG9zICs9IGxlbjsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLSAgICAKLSAgICB2b2lkIHJlc3RhcnQoKSB7Ci0gICAgICAgIGJ1ZmZlclBvcyA9IDA7Ci0gICAgICAgIGF0RnJvbnQgPSB0cnVlOwotICAgICAgICBpZiAoYnVmZmVyU2l6ZSA+IDI1NikgewotICAgICAgICAgICAgdm9pZCogYiA9IHJlYWxsb2MoYnVmZmVyLCAyNTYpOwotICAgICAgICAgICAgaWYgKGIpIHsKLSAgICAgICAgICAgICAgICBidWZmZXIgPSAoY2hhciopYjsKLSAgICAgICAgICAgICAgICBidWZmZXJTaXplID0gMjU2OwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIGNvbnN0IGludDMyX3Qgc2VxOwotICAgIGNoYXIqIGJ1ZmZlcjsKLSAgICBzaXplX3QgYnVmZmVyUG9zOwotICAgIHNpemVfdCBidWZmZXJTaXplOwotICAgIGJvb2wgYXRGcm9udDsKLSAgICBpbnQzMl90IGluZGVudDsKLSAgICBpbnQzMl90IGJ1bmRsZTsKLX07Ci0KLXN0cnVjdCBCdWZmZXJlZFRleHRPdXRwdXQ6OlRocmVhZFN0YXRlCi17Ci0gICAgVmVjdG9yPHNwPEJ1ZmZlcmVkVGV4dE91dHB1dDo6QnVmZmVyU3RhdGU+ID4gc3RhdGVzOwotfTsKLQotc3RhdGljIG11dGV4X3QgICAgICAgICAgZ011dGV4OwotCi1zdGF0aWMgdGhyZWFkX3N0b3JlX3QgICB0bHM7Ci0KLUJ1ZmZlcmVkVGV4dE91dHB1dDo6VGhyZWFkU3RhdGUqIEJ1ZmZlcmVkVGV4dE91dHB1dDo6Z2V0VGhyZWFkU3RhdGUoKQotewotICAgIFRocmVhZFN0YXRlKiAgdHMgPSAoVGhyZWFkU3RhdGUqKSB0aHJlYWRfc3RvcmVfZ2V0KCAmdGxzICk7Ci0gICAgaWYgKHRzKSByZXR1cm4gdHM7Ci0gICAgdHMgPSBuZXcgVGhyZWFkU3RhdGU7Ci0gICAgdGhyZWFkX3N0b3JlX3NldCggJnRscywgdHMsIHRocmVhZERlc3RydWN0b3IgKTsKLSAgICByZXR1cm4gdHM7Ci19Ci0KLXZvaWQgQnVmZmVyZWRUZXh0T3V0cHV0Ojp0aHJlYWREZXN0cnVjdG9yKHZvaWQgKnN0KQotewotICAgIGRlbGV0ZSAoKFRocmVhZFN0YXRlKilzdCk7Ci19Ci0KLXN0YXRpYyB2b2xhdGlsZSBpbnQzMl90IGdTZXF1ZW5jZSA9IDA7Ci0KLXN0YXRpYyB2b2xhdGlsZSBpbnQzMl90IGdGcmVlQnVmZmVySW5kZXggPSAtMTsKLQotc3RhdGljIGludDMyX3QgYWxsb2NCdWZmZXJJbmRleCgpCi17Ci0gICAgaW50MzJfdCByZXMgPSAtMTsKLSAgICAKLSAgICBtdXRleF9sb2NrKCZnTXV0ZXgpOwotICAgIAotICAgIGlmIChnRnJlZUJ1ZmZlckluZGV4ID49IDApIHsKLSAgICAgICAgcmVzID0gZ0ZyZWVCdWZmZXJJbmRleDsKLSAgICAgICAgZ0ZyZWVCdWZmZXJJbmRleCA9IGdUZXh0QnVmZmVyc1tyZXNdOwotICAgICAgICBnVGV4dEJ1ZmZlcnMuZWRpdEl0ZW1BdChyZXMpID0gLTE7Ci0KLSAgICB9IGVsc2UgewotICAgICAgICByZXMgPSBnVGV4dEJ1ZmZlcnMuc2l6ZSgpOwotICAgICAgICBnVGV4dEJ1ZmZlcnMuYWRkKC0xKTsKLSAgICB9Ci0KLSAgICBtdXRleF91bmxvY2soJmdNdXRleCk7Ci0gICAgCi0gICAgcmV0dXJuIHJlczsKLX0KLQotc3RhdGljIHZvaWQgZnJlZUJ1ZmZlckluZGV4KGludDMyX3QgaWR4KQotewotICAgIG11dGV4X2xvY2soJmdNdXRleCk7Ci0gICAgZ1RleHRCdWZmZXJzLmVkaXRJdGVtQXQoaWR4KSA9IGdGcmVlQnVmZmVySW5kZXg7Ci0gICAgZ0ZyZWVCdWZmZXJJbmRleCA9IGlkeDsKLSAgICBtdXRleF91bmxvY2soJmdNdXRleCk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1CdWZmZXJlZFRleHRPdXRwdXQ6OkJ1ZmZlcmVkVGV4dE91dHB1dCh1aW50MzJfdCBmbGFncykKLSAgICA6IG1GbGFncyhmbGFncykKLSAgICAsIG1TZXEoYW5kcm9pZF9hdG9taWNfaW5jKCZnU2VxdWVuY2UpKQotICAgICwgbUluZGV4KGFsbG9jQnVmZmVySW5kZXgoKSkKLXsKLSAgICBtR2xvYmFsU3RhdGUgPSBuZXcgQnVmZmVyU3RhdGUobVNlcSk7Ci0gICAgaWYgKG1HbG9iYWxTdGF0ZSkgbUdsb2JhbFN0YXRlLT5pbmNTdHJvbmcodGhpcyk7Ci19Ci0gICAgCi1CdWZmZXJlZFRleHRPdXRwdXQ6On5CdWZmZXJlZFRleHRPdXRwdXQoKQotewotICAgIGlmIChtR2xvYmFsU3RhdGUpIG1HbG9iYWxTdGF0ZS0+ZGVjU3Ryb25nKHRoaXMpOwotICAgIGZyZWVCdWZmZXJJbmRleChtSW5kZXgpOwotfQotCi1zdGF0dXNfdCBCdWZmZXJlZFRleHRPdXRwdXQ6OnByaW50KGNvbnN0IGNoYXIqIHR4dCwgc2l6ZV90IGxlbikKLXsKLSAgICAvL3ByaW50ZigiQnVmZmVyZWRUZXh0T3V0cHV0OiBwcmludGluZyAlZFxuIiwgbGVuKTsKLSAgICAKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgIEJ1ZmZlclN0YXRlKiBiID0gZ2V0QnVmZmVyKCk7Ci0gICAgCi0gICAgY29uc3QgY2hhciogY29uc3QgZW5kID0gdHh0K2xlbjsKLSAgICAKLSAgICBzdGF0dXNfdCBlcnI7Ci0KLSAgICB3aGlsZSAodHh0IDwgZW5kKSB7Ci0gICAgICAgIC8vIEZpbmQgdGhlIG5leHQgbGluZS4KLSAgICAgICAgY29uc3QgY2hhciogZmlyc3QgPSB0eHQ7Ci0gICAgICAgIHdoaWxlICh0eHQgPCBlbmQgJiYgKnR4dCAhPSAnXG4nKSB0eHQrKzsKLSAgICAgICAgCi0gICAgICAgIC8vIEluY2x1ZGUgdGhpcyBhbmQgYWxsIGZvbGxvd2luZyBlbXB0eSBsaW5lcy4KLSAgICAgICAgd2hpbGUgKHR4dCA8IGVuZCAmJiAqdHh0ID09ICdcbicpIHR4dCsrOwotICAgICAgICAKLSAgICAgICAgLy8gU3BlY2lhbCBjYXNlcyBmb3IgZmlyc3QgZGF0YSBvbiBhIGxpbmUuCi0gICAgICAgIGlmIChiLT5hdEZyb250KSB7Ci0gICAgICAgICAgICBpZiAoYi0+aW5kZW50ID4gMCkgewotICAgICAgICAgICAgICAgIC8vIElmIHRoaXMgaXMgdGhlIHN0YXJ0IG9mIGEgbGluZSwgYWRkIHRoZSBpbmRlbnQuCi0gICAgICAgICAgICAgICAgY29uc3QgY2hhciogcHJlZml4ID0gc3RyaW5nRm9ySW5kZW50KGItPmluZGVudCk7Ci0gICAgICAgICAgICAgICAgZXJyID0gYi0+YXBwZW5kKHByZWZpeCwgc3RybGVuKHByZWZpeCkpOwotICAgICAgICAgICAgICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHJldHVybiBlcnI7Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICB9IGVsc2UgaWYgKCoodHh0LTEpID09ICdcbicgJiYgIWItPmJ1bmRsZSkgewotICAgICAgICAgICAgICAgIC8vIEZhc3QgcGF0aDogaWYgd2UgYXJlIG5vdCBpbmRlbnRpbmcgb3IgYnVuZGxpbmcsIGFuZAotICAgICAgICAgICAgICAgIC8vIGhhdmUgYmVlbiBnaXZlbiBvbmUgb3IgbW9yZSBjb21wbGV0ZSBsaW5lcywganVzdCB3cml0ZQotICAgICAgICAgICAgICAgIC8vIHRoZW0gb3V0IHdpdGhvdXQgZ29pbmcgdGhyb3VnaCB0aGUgYnVmZmVyLgotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIC8vIFNsdXJwIHVwIGFsbCBvZiB0aGUgbGluZXMuCi0gICAgICAgICAgICAgICAgY29uc3QgY2hhciogbGFzdExpbmUgPSB0eHQrMTsKLSAgICAgICAgICAgICAgICB3aGlsZSAodHh0IDwgZW5kKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmICgqdHh0KysgPT0gJ1xuJykgbGFzdExpbmUgPSB0eHQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHN0cnVjdCBpb3ZlYyB2ZWM7Ci0gICAgICAgICAgICAgICAgdmVjLmlvdl9iYXNlID0gKHZvaWQqKWZpcnN0OwotICAgICAgICAgICAgICAgIHZlYy5pb3ZfbGVuID0gbGFzdExpbmUtZmlyc3Q7Ci0gICAgICAgICAgICAgICAgLy9wcmludGYoIldyaXRpbmcgJWQgYnl0ZXMgb2YgZGF0YSFcbiIsIHZlYy5pb3ZfbGVuKTsKLSAgICAgICAgICAgICAgICB3cml0ZUxpbmVzKHZlYywgMSk7Ci0gICAgICAgICAgICAgICAgdHh0ID0gbGFzdExpbmU7Ci0gICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIC8vIEFwcGVuZCB0aGUgbmV3IHRleHQgdG8gdGhlIGJ1ZmZlci4KLSAgICAgICAgZXJyID0gYi0+YXBwZW5kKGZpcnN0LCB0eHQtZmlyc3QpOwotICAgICAgICBpZiAoZXJyICE9IE5PX0VSUk9SKSByZXR1cm4gZXJyOwotICAgICAgICBiLT5hdEZyb250ID0gKih0eHQtMSkgPT0gJ1xuJzsKLSAgICAgICAgCi0gICAgICAgIC8vIElmIHdlIGhhdmUgZmluaXNoZWQgYSBsaW5lIGFuZCBhcmUgbm90IGJ1bmRsaW5nLCB3cml0ZQotICAgICAgICAvLyBpdCBvdXQuCi0gICAgICAgIC8vcHJpbnRmKCJCdWZmZXIgaXMgbm93ICVkIGJ5dGVzXG4iLCBiLT5idWZmZXJQb3MpOwotICAgICAgICBpZiAoYi0+YXRGcm9udCAmJiAhYi0+YnVuZGxlKSB7Ci0gICAgICAgICAgICBzdHJ1Y3QgaW92ZWMgdmVjOwotICAgICAgICAgICAgdmVjLmlvdl9iYXNlID0gYi0+YnVmZmVyOwotICAgICAgICAgICAgdmVjLmlvdl9sZW4gPSBiLT5idWZmZXJQb3M7Ci0gICAgICAgICAgICAvL3ByaW50ZigiV3JpdGluZyAlZCBieXRlcyBvZiBkYXRhIVxuIiwgdmVjLmlvdl9sZW4pOwotICAgICAgICAgICAgd3JpdGVMaW5lcyh2ZWMsIDEpOwotICAgICAgICAgICAgYi0+cmVzdGFydCgpOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotdm9pZCBCdWZmZXJlZFRleHRPdXRwdXQ6Om1vdmVJbmRlbnQoaW50IGRlbHRhKQotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgQnVmZmVyU3RhdGUqIGIgPSBnZXRCdWZmZXIoKTsKLSAgICBiLT5pbmRlbnQgKz0gZGVsdGE7Ci0gICAgaWYgKGItPmluZGVudCA8IDApIGItPmluZGVudCA9IDA7Ci19Ci0KLXZvaWQgQnVmZmVyZWRUZXh0T3V0cHV0OjpwdXNoQnVuZGxlKCkKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgIEJ1ZmZlclN0YXRlKiBiID0gZ2V0QnVmZmVyKCk7Ci0gICAgYi0+YnVuZGxlKys7Ci19Ci0KLXZvaWQgQnVmZmVyZWRUZXh0T3V0cHV0Ojpwb3BCdW5kbGUoKQotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgQnVmZmVyU3RhdGUqIGIgPSBnZXRCdWZmZXIoKTsKLSAgICBiLT5idW5kbGUtLTsKLSAgICBMT0dfRkFUQUxfSUYoYi0+YnVuZGxlIDwgMCwKLSAgICAgICAgIlRleHRPdXRwdXQ6OnBvcEJ1bmRsZSgpIGNhbGxlZCBtb3JlIHRpbWVzIHRoYW4gcHVzaEJ1bmRsZSgpIik7Ci0gICAgaWYgKGItPmJ1bmRsZSA8IDApIGItPmJ1bmRsZSA9IDA7Ci0gICAgCi0gICAgaWYgKGItPmJ1bmRsZSA9PSAwKSB7Ci0gICAgICAgIC8vIExhc3QgYnVuZGxlLCB3cml0ZSBvdXQgZGF0YSBpZiBpdCBpcyBjb21wbGV0ZS4gIElmIGl0IGlzIG5vdAotICAgICAgICAvLyBjb21wbGV0ZSwgZG9uJ3Qgd3JpdGUgdW50aWwgdGhlIGxhc3QgbGluZSBpcyBkb25lLi4uIHRoaXMgbWF5Ci0gICAgICAgIC8vIG9yIG1heSBub3QgYmUgdGhlIHdyaXRlIHRoaW5nIHRvIGRvLCBidXQgaXQncyB0aGUgZWFzaWVzdC4KLSAgICAgICAgaWYgKGItPmJ1ZmZlclBvcyA+IDAgJiYgYi0+YXRGcm9udCkgewotICAgICAgICAgICAgc3RydWN0IGlvdmVjIHZlYzsKLSAgICAgICAgICAgIHZlYy5pb3ZfYmFzZSA9IGItPmJ1ZmZlcjsKLSAgICAgICAgICAgIHZlYy5pb3ZfbGVuID0gYi0+YnVmZmVyUG9zOwotICAgICAgICAgICAgd3JpdGVMaW5lcyh2ZWMsIDEpOwotICAgICAgICAgICAgYi0+cmVzdGFydCgpOwotICAgICAgICB9Ci0gICAgfQotfQotCi1CdWZmZXJlZFRleHRPdXRwdXQ6OkJ1ZmZlclN0YXRlKiBCdWZmZXJlZFRleHRPdXRwdXQ6OmdldEJ1ZmZlcigpIGNvbnN0Ci17Ci0gICAgaWYgKChtRmxhZ3MmTVVMVElUSFJFQURFRCkgIT0gMCkgewotICAgICAgICBUaHJlYWRTdGF0ZSogdHMgPSBnZXRUaHJlYWRTdGF0ZSgpOwotICAgICAgICBpZiAodHMpIHsKLSAgICAgICAgICAgIHdoaWxlICh0cy0+c3RhdGVzLnNpemUoKSA8PSAoc2l6ZV90KW1JbmRleCkgdHMtPnN0YXRlcy5hZGQoTlVMTCk7Ci0gICAgICAgICAgICBCdWZmZXJTdGF0ZSogYnMgPSB0cy0+c3RhdGVzW21JbmRleF0uZ2V0KCk7Ci0gICAgICAgICAgICBpZiAoYnMgIT0gTlVMTCAmJiBicy0+c2VxID09IG1TZXEpIHJldHVybiBiczsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgdHMtPnN0YXRlcy5lZGl0SXRlbUF0KG1JbmRleCkgPSBuZXcgQnVmZmVyU3RhdGUobUluZGV4KTsKLSAgICAgICAgICAgIGJzID0gdHMtPnN0YXRlc1ttSW5kZXhdLmdldCgpOwotICAgICAgICAgICAgaWYgKGJzICE9IE5VTEwpIHJldHVybiBiczsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICByZXR1cm4gbUdsb2JhbFN0YXRlOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9DYWxsU3RhY2suY3BwIGIvbGlicy91dGlscy9DYWxsU3RhY2suY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyNmZiMjJhLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvQ2FsbFN0YWNrLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDMzNSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJDYWxsU3RhY2siCi0KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotCi0jaWYgSEFWRV9ETEFERFIKLSNpbmNsdWRlIDxkbGZjbi5oPgotI2VuZGlmCi0KLSNpZiBIQVZFX0NYWEFCSQotI2luY2x1ZGUgPGN4eGFiaS5oPgotI2VuZGlmCi0KLSNpbmNsdWRlIDx1bndpbmQuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL0NhbGxTdGFjay5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLQotCi0vKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLQotdHlwZWRlZiBzdHJ1Y3QgewotICAgIHNpemVfdCBjb3VudDsKLSAgICBzaXplX3QgaWdub3JlOwotICAgIGNvbnN0IHZvaWQqKiBhZGRyczsKLX0gc3RhY2tfY3Jhd2xfc3RhdGVfdDsKLQotc3RhdGljCi1fVW53aW5kX1JlYXNvbl9Db2RlIHRyYWNlX2Z1bmN0aW9uKF9VbndpbmRfQ29udGV4dCAqY29udGV4dCwgdm9pZCAqYXJnKQotewotICAgIHN0YWNrX2NyYXdsX3N0YXRlX3QqIHN0YXRlID0gKHN0YWNrX2NyYXdsX3N0YXRlX3QqKWFyZzsKLSAgICBpZiAoc3RhdGUtPmNvdW50KSB7Ci0gICAgICAgIHZvaWQqIGlwID0gKHZvaWQqKV9VbndpbmRfR2V0SVAoY29udGV4dCk7Ci0gICAgICAgIGlmIChpcCkgewotICAgICAgICAgICAgaWYgKHN0YXRlLT5pZ25vcmUpIHsKLSAgICAgICAgICAgICAgICBzdGF0ZS0+aWdub3JlLS07Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHN0YXRlLT5hZGRyc1swXSA9IGlwOyAKLSAgICAgICAgICAgICAgICBzdGF0ZS0+YWRkcnMrKzsKLSAgICAgICAgICAgICAgICBzdGF0ZS0+Y291bnQtLTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gX1VSQ19OT19SRUFTT047Ci19Ci0KLXN0YXRpYwotaW50IGJhY2t0cmFjZShjb25zdCB2b2lkKiogYWRkcnMsIHNpemVfdCBpZ25vcmUsIHNpemVfdCBzaXplKQotewotICAgIHN0YWNrX2NyYXdsX3N0YXRlX3Qgc3RhdGU7Ci0gICAgc3RhdGUuY291bnQgPSBzaXplOwotICAgIHN0YXRlLmlnbm9yZSA9IGlnbm9yZTsKLSAgICBzdGF0ZS5hZGRycyA9IGFkZHJzOwotICAgIF9VbndpbmRfQmFja3RyYWNlKHRyYWNlX2Z1bmN0aW9uLCAodm9pZCopJnN0YXRlKTsKLSAgICByZXR1cm4gc2l6ZSAtIHN0YXRlLmNvdW50OwotfQotCi0vKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi0KLXN0YXRpYyAKLWNvbnN0IGNoYXIgKmxvb2t1cF9zeW1ib2woY29uc3Qgdm9pZCogYWRkciwgdm9pZCAqKm9mZnNldCwgY2hhciogbmFtZSwgc2l6ZV90IGJ1ZlNpemUpCi17Ci0jaWYgSEFWRV9ETEFERFIKLSAgICBEbF9pbmZvIGluZm87Ci0gICAgaWYgKGRsYWRkcihhZGRyLCAmaW5mbykpIHsKLSAgICAgICAgKm9mZnNldCA9IGluZm8uZGxpX3NhZGRyOwotICAgICAgICByZXR1cm4gaW5mby5kbGlfc25hbWU7Ci0gICAgfQotI2VuZGlmCi0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLXN0YXRpYyAKLWludDMyX3QgbGludXhfZ2NjX2RlbWFuZ2xlcihjb25zdCBjaGFyICptYW5nbGVkX25hbWUsIGNoYXIgKnVubWFuZ2xlZF9uYW1lLCBzaXplX3QgYnVmZmVyc2l6ZSkKLXsKLSAgICBzaXplX3Qgb3V0X2xlbiA9IDA7Ci0jaWYgSEFWRV9DWFhBQkkKLSAgICBpbnQgc3RhdHVzID0gMDsKLSAgICBjaGFyICpkZW1hbmdsZWQgPSBhYmk6Ol9fY3hhX2RlbWFuZ2xlKG1hbmdsZWRfbmFtZSwgMCwgJm91dF9sZW4sICZzdGF0dXMpOwotICAgIGlmIChzdGF0dXMgPT0gMCkgewotICAgICAgICAvLyBPSwotICAgICAgICBpZiAob3V0X2xlbiA8IGJ1ZmZlcnNpemUpIG1lbWNweSh1bm1hbmdsZWRfbmFtZSwgZGVtYW5nbGVkLCBvdXRfbGVuKTsKLSAgICAgICAgZWxzZSBvdXRfbGVuID0gMDsKLSAgICAgICAgZnJlZShkZW1hbmdsZWQpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIG91dF9sZW4gPSAwOwotICAgIH0KLSNlbmRpZgotICAgIHJldHVybiBvdXRfbGVuOwotfQotCi0vKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi0KLWNsYXNzIE1hcEluZm8gewotICAgIHN0cnVjdCBtYXBpbmZvIHsKLSAgICAgICAgc3RydWN0IG1hcGluZm8gKm5leHQ7Ci0gICAgICAgIHVpbnQ2NF90IHN0YXJ0OwotICAgICAgICB1aW50NjRfdCBlbmQ7Ci0gICAgICAgIGNoYXIgbmFtZVtdOwotICAgIH07Ci0KLSAgICBjb25zdCBjaGFyICptYXBfdG9fbmFtZSh1aW50NjRfdCBwYywgY29uc3QgY2hhciogZGVmKSB7Ci0gICAgICAgIG1hcGluZm8qIG1pID0gZ2V0TWFwSW5mb0xpc3QoKTsKLSAgICAgICAgd2hpbGUobWkpIHsKLSAgICAgICAgICAgIGlmICgocGMgPj0gbWktPnN0YXJ0KSAmJiAocGMgPCBtaS0+ZW5kKSkKLSAgICAgICAgICAgICAgICByZXR1cm4gbWktPm5hbWU7Ci0gICAgICAgICAgICBtaSA9IG1pLT5uZXh0OwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBkZWY7Ci0gICAgfQotCi0gICAgbWFwaW5mbyAqcGFyc2VfbWFwc19saW5lKGNoYXIgKmxpbmUpIHsKLSAgICAgICAgbWFwaW5mbyAqbWk7Ci0gICAgICAgIGludCBsZW4gPSBzdHJsZW4obGluZSk7Ci0gICAgICAgIGlmIChsZW4gPCAxKSByZXR1cm4gMDsKLSAgICAgICAgbGluZVstLWxlbl0gPSAwOwotICAgICAgICBpZiAobGVuIDwgNTApIHJldHVybiAwOwotICAgICAgICBpZiAobGluZVsyMF0gIT0gJ3gnKSByZXR1cm4gMDsKLSAgICAgICAgbWkgPSAobWFwaW5mbyopbWFsbG9jKHNpemVvZihtYXBpbmZvKSArIChsZW4gLSA0NykpOwotICAgICAgICBpZiAobWkgPT0gMCkgcmV0dXJuIDA7Ci0gICAgICAgIG1pLT5zdGFydCA9IHN0cnRvdWxsKGxpbmUsIDAsIDE2KTsKLSAgICAgICAgbWktPmVuZCA9IHN0cnRvdWxsKGxpbmUgKyA5LCAwLCAxNik7Ci0gICAgICAgIG1pLT5uZXh0ID0gMDsKLSAgICAgICAgc3RyY3B5KG1pLT5uYW1lLCBsaW5lICsgNDkpOwotICAgICAgICByZXR1cm4gbWk7Ci0gICAgfQotCi0gICAgbWFwaW5mbyogZ2V0TWFwSW5mb0xpc3QoKSB7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgICAgIGlmIChtaWxpc3QgPT0gMCkgewotICAgICAgICAgICAgY2hhciBkYXRhWzEwMjRdOwotICAgICAgICAgICAgRklMRSAqZnA7Ci0gICAgICAgICAgICBzcHJpbnRmKGRhdGEsICIvcHJvYy8lZC9tYXBzIiwgZ2V0cGlkKCkpOwotICAgICAgICAgICAgZnAgPSBmb3BlbihkYXRhLCAiciIpOwotICAgICAgICAgICAgaWYgKGZwKSB7Ci0gICAgICAgICAgICAgICAgd2hpbGUoZmdldHMoZGF0YSwgMTAyNCwgZnApKSB7Ci0gICAgICAgICAgICAgICAgICAgIG1hcGluZm8gKm1pID0gcGFyc2VfbWFwc19saW5lKGRhdGEpOwotICAgICAgICAgICAgICAgICAgICBpZihtaSkgewotICAgICAgICAgICAgICAgICAgICAgICAgbWktPm5leHQgPSBtaWxpc3Q7Ci0gICAgICAgICAgICAgICAgICAgICAgICBtaWxpc3QgPSBtaTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBmY2xvc2UoZnApOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBtaWxpc3Q7Ci0gICAgfQotICAgIG1hcGluZm8qICAgIG1pbGlzdDsKLSAgICBNdXRleCAgICAgICBtTG9jazsKLSAgICBzdGF0aWMgTWFwSW5mbyBzTWFwSW5mbzsKLQotcHVibGljOgotICAgIE1hcEluZm8oKQotICAgICA6IG1pbGlzdCgwKSB7Ci0gICAgfQotCi0gICAgfk1hcEluZm8oKSB7Ci0gICAgICAgIHdoaWxlIChtaWxpc3QpIHsKLSAgICAgICAgICAgIG1hcGluZm8gKm5leHQgPSBtaWxpc3QtPm5leHQ7Ci0gICAgICAgICAgICBmcmVlKG1pbGlzdCk7Ci0gICAgICAgICAgICBtaWxpc3QgPSBuZXh0OwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIHN0YXRpYyBjb25zdCBjaGFyICptYXBBZGRyZXNzVG9OYW1lKGNvbnN0IHZvaWQqIHBjLCBjb25zdCBjaGFyKiBkZWYpIHsKLSAgICAgICAgcmV0dXJuIHNNYXBJbmZvLm1hcF90b19uYW1lKCh1aW50NjRfdClwYywgZGVmKTsKLSAgICB9Ci0KLX07Ci0KLS8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotTWFwSW5mbyBNYXBJbmZvOjpzTWFwSW5mbzsKLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi1DYWxsU3RhY2s6OkNhbGxTdGFjaygpCi0gICAgOiBtQ291bnQoMCkKLXsKLX0KLQotQ2FsbFN0YWNrOjpDYWxsU3RhY2soY29uc3QgQ2FsbFN0YWNrJiByaHMpCi0gICAgOiBtQ291bnQocmhzLm1Db3VudCkKLXsKLSAgICBpZiAobUNvdW50KSB7Ci0gICAgICAgIG1lbWNweShtU3RhY2ssIHJocy5tU3RhY2ssIG1Db3VudCpzaXplb2Yodm9pZCopKTsKLSAgICB9Ci19Ci0KLUNhbGxTdGFjazo6fkNhbGxTdGFjaygpCi17Ci19Ci0KLUNhbGxTdGFjayYgQ2FsbFN0YWNrOjpvcGVyYXRvciA9IChjb25zdCBDYWxsU3RhY2smIHJocykKLXsKLSAgICBtQ291bnQgPSByaHMubUNvdW50OwotICAgIGlmIChtQ291bnQpIHsKLSAgICAgICAgbWVtY3B5KG1TdGFjaywgcmhzLm1TdGFjaywgbUNvdW50KnNpemVvZih2b2lkKikpOwotICAgIH0KLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLWJvb2wgQ2FsbFN0YWNrOjpvcGVyYXRvciA9PSAoY29uc3QgQ2FsbFN0YWNrJiByaHMpIGNvbnN0IHsKLSAgICBpZiAobUNvdW50ICE9IHJocy5tQ291bnQpCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICByZXR1cm4gIW1Db3VudCB8fCAobWVtY21wKG1TdGFjaywgcmhzLm1TdGFjaywgbUNvdW50KnNpemVvZih2b2lkKikpID09IDApOwotfQotCi1ib29sIENhbGxTdGFjazo6b3BlcmF0b3IgIT0gKGNvbnN0IENhbGxTdGFjayYgcmhzKSBjb25zdCB7Ci0gICAgcmV0dXJuICFvcGVyYXRvciA9PSAocmhzKTsKLX0KLQotYm9vbCBDYWxsU3RhY2s6Om9wZXJhdG9yIDwgKGNvbnN0IENhbGxTdGFjayYgcmhzKSBjb25zdCB7Ci0gICAgaWYgKG1Db3VudCAhPSByaHMubUNvdW50KQotICAgICAgICByZXR1cm4gbUNvdW50IDwgcmhzLm1Db3VudDsKLSAgICByZXR1cm4gbWVtY21wKG1TdGFjaywgcmhzLm1TdGFjaywgbUNvdW50KnNpemVvZih2b2lkKikpIDwgMDsKLX0KLQotYm9vbCBDYWxsU3RhY2s6Om9wZXJhdG9yID49IChjb25zdCBDYWxsU3RhY2smIHJocykgY29uc3QgewotICAgIHJldHVybiAhb3BlcmF0b3IgPCAocmhzKTsKLX0KLQotYm9vbCBDYWxsU3RhY2s6Om9wZXJhdG9yID4gKGNvbnN0IENhbGxTdGFjayYgcmhzKSBjb25zdCB7Ci0gICAgaWYgKG1Db3VudCAhPSByaHMubUNvdW50KQotICAgICAgICByZXR1cm4gbUNvdW50ID4gcmhzLm1Db3VudDsKLSAgICByZXR1cm4gbWVtY21wKG1TdGFjaywgcmhzLm1TdGFjaywgbUNvdW50KnNpemVvZih2b2lkKikpID4gMDsKLX0KLQotYm9vbCBDYWxsU3RhY2s6Om9wZXJhdG9yIDw9IChjb25zdCBDYWxsU3RhY2smIHJocykgY29uc3QgewotICAgIHJldHVybiAhb3BlcmF0b3IgPiAocmhzKTsKLX0KLQotY29uc3Qgdm9pZCogQ2FsbFN0YWNrOjpvcGVyYXRvciBbXSAoaW50IGluZGV4KSBjb25zdCB7Ci0gICAgaWYgKGluZGV4ID49IGludChtQ291bnQpKQotICAgICAgICByZXR1cm4gMDsKLSAgICByZXR1cm4gbVN0YWNrW2luZGV4XTsKLX0KLQotCi12b2lkIENhbGxTdGFjazo6Y2xlYXIoKQotewotICAgIG1Db3VudCA9IDA7Ci19Ci0KLXZvaWQgQ2FsbFN0YWNrOjp1cGRhdGUoaW50MzJfdCBpZ25vcmVEZXB0aCwgaW50MzJfdCBtYXhEZXB0aCkKLXsKLSAgICBpZiAobWF4RGVwdGggPiBNQVhfREVQVEgpCi0gICAgICAgIG1heERlcHRoID0gTUFYX0RFUFRIOwotICAgIG1Db3VudCA9IGJhY2t0cmFjZShtU3RhY2ssIGlnbm9yZURlcHRoLCBtYXhEZXB0aCk7Ci19Ci0KLS8vIFJldHVybiB0aGUgc3RhY2sgZnJhbWUgbmFtZSBvbiB0aGUgZGVzaWduYXRlZCBsZXZlbAotU3RyaW5nOCBDYWxsU3RhY2s6OnRvU3RyaW5nU2luZ2xlTGV2ZWwoY29uc3QgY2hhciogcHJlZml4LCBpbnQzMl90IGxldmVsKSBjb25zdAotewotICAgIFN0cmluZzggcmVzOwotICAgIGNoYXIgbmFtZWJ1ZlsxMDI0XTsKLSAgICBjaGFyIHRtcFsyNTZdOwotICAgIGNoYXIgdG1wMVszMl07Ci0gICAgY2hhciB0bXAyWzMyXTsKLSAgICB2b2lkICpvZmZzOwotCi0gICAgY29uc3Qgdm9pZCogaXAgPSBtU3RhY2tbbGV2ZWxdOwotICAgIGlmICghaXApIHJldHVybiByZXM7Ci0KLSAgICBpZiAocHJlZml4KSByZXMuYXBwZW5kKHByZWZpeCk7Ci0gICAgc25wcmludGYodG1wMSwgMzIsICIjJTAyZCAgIiwgbGV2ZWwpOwotICAgIHJlcy5hcHBlbmQodG1wMSk7Ci0KLSAgICBjb25zdCBjaGFyKiBuYW1lID0gbG9va3VwX3N5bWJvbChpcCwgJm9mZnMsIG5hbWVidWYsIHNpemVvZihuYW1lYnVmKSk7Ci0gICAgaWYgKG5hbWUpIHsKLSAgICAgICAgaWYgKGxpbnV4X2djY19kZW1hbmdsZXIobmFtZSwgdG1wLCAyNTYpICE9IDApCi0gICAgICAgICAgICBuYW1lID0gdG1wOwotICAgICAgICBzbnByaW50Zih0bXAxLCAzMiwgIjB4JXA6IDwiLCBpcCk7Ci0gICAgICAgIHNucHJpbnRmKHRtcDIsIDMyLCAiPisweCVwIiwgb2Zmcyk7Ci0gICAgICAgIHJlcy5hcHBlbmQodG1wMSk7Ci0gICAgICAgIHJlcy5hcHBlbmQobmFtZSk7Ci0gICAgICAgIHJlcy5hcHBlbmQodG1wMik7Ci0gICAgfSBlbHNlIHsgCi0gICAgICAgIG5hbWUgPSBNYXBJbmZvOjptYXBBZGRyZXNzVG9OYW1lKGlwLCAiPHVua25vd24+Iik7Ci0gICAgICAgIHNucHJpbnRmKHRtcCwgMjU2LCAicGMgJXAgICVzIiwgaXAsIG5hbWUpOwotICAgICAgICByZXMuYXBwZW5kKHRtcCk7Ci0gICAgfQotICAgIHJlcy5hcHBlbmQoIlxuIik7Ci0KLSAgICByZXR1cm4gcmVzOwotfQotCi0vLyBEdW1wIGEgc3RhY2sgdHJhY2UgdG8gdGhlIGxvZwotdm9pZCBDYWxsU3RhY2s6OmR1bXAoY29uc3QgY2hhciogcHJlZml4KSBjb25zdAotewotICAgIC8qIAotICAgICAqIFNlbmRpbmcgYSBzaW5nbGUgbG9uZyBsb2cgbWF5IGJlIHRydW5jYXRlZCBzaW5jZSB0aGUgc3RhY2sgbGV2ZWxzIGNhbgotICAgICAqIGdldCB2ZXJ5IGRlZXAuIFNvIHdlIHJlcXVlc3QgZnVuY3Rpb24gbmFtZXMgb2YgZWFjaCBmcmFtZSBpbmRpdmlkdWFsbHkuCi0gICAgICovCi0gICAgZm9yIChpbnQgaT0wOyBpPGludChtQ291bnQpOyBpKyspIHsKLSAgICAgICAgTE9HRCgiJXMiLCB0b1N0cmluZ1NpbmdsZUxldmVsKHByZWZpeCwgaSkuc3RyaW5nKCkpOwotICAgIH0KLX0KLQotLy8gUmV0dXJuIGEgc3RyaW5nIChwb3NzaWJseSB2ZXJ5IGxvbmcpIGNvbnRhaW5pbmcgdGhlIGNvbXBsZXRlIHN0YWNrIHRyYWNlCi1TdHJpbmc4IENhbGxTdGFjazo6dG9TdHJpbmcoY29uc3QgY2hhciogcHJlZml4KSBjb25zdAotewotICAgIFN0cmluZzggcmVzOwotCi0gICAgZm9yIChpbnQgaT0wOyBpPGludChtQ291bnQpOyBpKyspIHsKLSAgICAgICAgcmVzLmFwcGVuZCh0b1N0cmluZ1NpbmdsZUxldmVsKHByZWZpeCwgaSkuc3RyaW5nKCkpOwotICAgIH0KLQotICAgIHJldHVybiByZXM7Ci19Ci0KLS8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvRGVidWcuY3BwIGIvbGlicy91dGlscy9EZWJ1Zy5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGY3OTg4ZWMuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9EZWJ1Zy5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwzMTggKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDx1dGlscy9EZWJ1Zy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvbWlzYy5oPgotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxjdHlwZS5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgY29uc3QgY2hhciBpbmRlbnRTdHJbXSA9Ci0iICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICIKLSIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIjsKLQotY29uc3QgY2hhciogc3RyaW5nRm9ySW5kZW50KGludDMyX3QgaW5kZW50TGV2ZWwpCi17Ci0gICAgc3NpemVfdCBvZmYgPSBzaXplb2YoaW5kZW50U3RyKS0xLShpbmRlbnRMZXZlbCoyKTsKLSAgICByZXR1cm4gaW5kZW50U3RyICsgKG9mZiA8IDAgPyAwIDogb2ZmKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyB2b2lkIGRlZmF1bHRQcmludEZ1bmModm9pZCogY29va2llLCBjb25zdCBjaGFyKiB0eHQpCi17Ci0gICAgcHJpbnRmKCIlcyIsIHR4dCk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgaW5saW5lIGludCBpc2lkZW50KGludCBjKQotewotICAgIHJldHVybiBpc2FsbnVtKGMpIHx8IGMgPT0gJ18nOwotfQotCi1zdGF0aWMgaW5saW5lIGJvb2wgaXNhc2NpaXR5cGUoY2hhciBjKQotewotICAgIGlmKCBjID49ICcgJyAmJiBjIDwgMTI3ICYmIGMgIT0gJ1wnJyAmJiBjICE9ICdcXCcgKSByZXR1cm4gdHJ1ZTsKLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0KLXN0YXRpYyBpbmxpbmUgY2hhciBtYWtlaGV4ZGlnaXQodWludDMyX3QgdmFsKQotewotICAgIHJldHVybiAiMDEyMzQ1Njc4OWFiY2RlZiJbdmFsJjB4Rl07Ci19Ci0KLXN0YXRpYyBjaGFyKiBhcHBlbmRoZXhudW0odWludDMyX3QgdmFsLCBjaGFyKiBvdXQpCi17Ci0gICAgZm9yKCBpbnQzMl90IGk9Mjg7IGk+PTA7IGktPTQgKSB7Ci0gICAgICAgICpvdXQrKyA9IG1ha2VoZXhkaWdpdCggdmFsPj5pICk7Ci0gICAgfQotICAgICpvdXQgPSAwOwotICAgIHJldHVybiBvdXQ7Ci19Ci0KLXN0YXRpYyBpbmxpbmUgY2hhciBtYWtldXBwZXJoZXhkaWdpdCh1aW50MzJfdCB2YWwpCi17Ci0gICAgcmV0dXJuICIwMTIzNDU2Nzg5QUJDREVGIlt2YWwmMHhGXTsKLX0KLQotc3RhdGljIGNoYXIqIGFwcGVuZHVwcGVyaGV4bnVtKHVpbnQzMl90IHZhbCwgY2hhciogb3V0KQotewotICAgIGZvciggaW50MzJfdCBpPTI4OyBpPj0wOyBpLT00ICkgewotICAgICAgICAqb3V0KysgPSBtYWtldXBwZXJoZXhkaWdpdCggdmFsPj5pICk7Ci0gICAgfQotICAgICpvdXQgPSAwOwotICAgIHJldHVybiBvdXQ7Ci19Ci0KLXN0YXRpYyBjaGFyKiBhcHBlbmRjaGFyb3JudW0oY2hhciBjLCBjaGFyKiBvdXQsIGJvb2wgc2tpcHplcm8gPSB0cnVlKQotewotICAgIGlmIChza2lwemVybyAmJiBjID09IDApIHJldHVybiBvdXQ7Ci0KLSAgICBpZiAoaXNhc2NpaXR5cGUoYykpIHsKLSAgICAgICAgKm91dCsrID0gYzsKLSAgICAgICAgcmV0dXJuIG91dDsKLSAgICB9Ci0KLSAgICAqb3V0KysgPSAnXFwnOwotICAgICpvdXQrKyA9ICd4JzsKLSAgICAqb3V0KysgPSBtYWtlaGV4ZGlnaXQoYz4+NCk7Ci0gICAgKm91dCsrID0gbWFrZWhleGRpZ2l0KGMpOwotICAgIHJldHVybiBvdXQ7Ci19Ci0KLXN0YXRpYyBjaGFyKiB0eXBldG9zdHJpbmcodWludDMyX3QgdHlwZSwgY2hhciogb3V0LAotICAgICAgICAgICAgICAgICAgICAgICAgICBib29sIGZ1bGxDb250ZXh0ID0gdHJ1ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgYm9vbCBzdHJpY3QgPSBmYWxzZSkKLXsKLSAgICBjaGFyKiBwb3MgPSBvdXQ7Ci0gICAgY2hhciBjWzRdOwotICAgIGNbMF0gPSAoY2hhcikoKHR5cGU+PjI0KSYweEZGKTsKLSAgICBjWzFdID0gKGNoYXIpKCh0eXBlPj4xNikmMHhGRik7Ci0gICAgY1syXSA9IChjaGFyKSgodHlwZT4+OCkmMHhGRik7Ci0gICAgY1szXSA9IChjaGFyKSh0eXBlJjB4RkYpOwotICAgIGJvb2wgdmFsaWQ7Ci0gICAgaWYoICFzdHJpY3QgKSB7Ci0gICAgICAgIC8vIG5vdyBldmVuIGxlc3Mgc3RyaWN0IQotICAgICAgICAvLyB2YWxpZCA9IGlzYXNjaWl0eXBlKGNbM10pOwotICAgICAgICB2YWxpZCA9IHRydWU7Ci0gICAgICAgIGludDMyX3QgaSA9IDA7Ci0gICAgICAgIGJvb2wgemVybyA9IHRydWU7Ci0gICAgICAgIHdoaWxlICh2YWxpZCAmJiBpPDMpIHsKLSAgICAgICAgICAgIGlmIChjW2ldID09IDApIHsKLSAgICAgICAgICAgICAgICBpZiAoIXplcm8pIHZhbGlkID0gZmFsc2U7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHplcm8gPSBmYWxzZTsKLSAgICAgICAgICAgICAgICAvL2lmICghaXNhc2NpaXR5cGUoY1tpXSkpIHZhbGlkID0gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpKys7Ci0gICAgICAgIH0KLSAgICAgICAgLy8gaWYgYWxsIHplcm9zLCBub3QgYSB2YWxpZCB0eXBlIGNvZGUuCi0gICAgICAgIGlmICh6ZXJvKSB2YWxpZCA9IGZhbHNlOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHZhbGlkID0gaXNpZGVudChjWzNdKSA/IHRydWUgOiBmYWxzZTsKLSAgICAgICAgaW50MzJfdCBpID0gMDsKLSAgICAgICAgYm9vbCB6ZXJvID0gdHJ1ZTsKLSAgICAgICAgd2hpbGUgKHZhbGlkICYmIGk8MykgewotICAgICAgICAgICAgaWYgKGNbaV0gPT0gMCkgewotICAgICAgICAgICAgICAgIGlmICghemVybykgdmFsaWQgPSBmYWxzZTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgemVybyA9IGZhbHNlOwotICAgICAgICAgICAgICAgIGlmICghaXNpZGVudChjW2ldKSkgdmFsaWQgPSBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGkrKzsKLSAgICAgICAgfQotICAgIH0KLSAgICBpZiggdmFsaWQgJiYgKCFmdWxsQ29udGV4dCB8fCBjWzBdICE9ICcwJyB8fCBjWzFdICE9ICd4JykgKSB7Ci0gICAgICAgIGlmKCBmdWxsQ29udGV4dCApICpwb3MrKyA9ICdcJyc7Ci0gICAgICAgIHBvcyA9IGFwcGVuZGNoYXJvcm51bShjWzBdLCBwb3MpOwotICAgICAgICBwb3MgPSBhcHBlbmRjaGFyb3JudW0oY1sxXSwgcG9zKTsKLSAgICAgICAgcG9zID0gYXBwZW5kY2hhcm9ybnVtKGNbMl0sIHBvcyk7Ci0gICAgICAgIHBvcyA9IGFwcGVuZGNoYXJvcm51bShjWzNdLCBwb3MpOwotICAgICAgICBpZiggZnVsbENvbnRleHQgKSAqcG9zKysgPSAnXCcnOwotICAgICAgICAqcG9zID0gMDsKLSAgICAgICAgcmV0dXJuIHBvczsKLSAgICB9Ci0gICAgCi0gICAgaWYoIGZ1bGxDb250ZXh0ICkgewotICAgICAgICAqcG9zKysgPSAnMCc7Ci0gICAgICAgICpwb3MrKyA9ICd4JzsKLSAgICB9Ci0gICAgcmV0dXJuIGFwcGVuZGhleG51bSh0eXBlLCBwb3MpOwotfQotCi12b2lkIHByaW50VHlwZUNvZGUodWludDMyX3QgdHlwZUNvZGUsIGRlYnVnUHJpbnRGdW5jIGZ1bmMsIHZvaWQqIGNvb2tpZSkKLXsKLSAgICBjaGFyIGJ1ZmZlclszMl07Ci0gICAgY2hhciogZW5kID0gdHlwZXRvc3RyaW5nKHR5cGVDb2RlLCBidWZmZXIpOwotICAgICplbmQgPSAwOwotICAgIGZ1bmMgPyAoKmZ1bmMpKGNvb2tpZSwgYnVmZmVyKSA6IGRlZmF1bHRQcmludEZ1bmMoY29va2llLCBidWZmZXIpOwotfQotCi12b2lkIHByaW50SGV4RGF0YShpbnQzMl90IGluZGVudCwgY29uc3Qgdm9pZCAqYnVmLCBzaXplX3QgbGVuZ3RoLAotICAgIHNpemVfdCBieXRlc1BlckxpbmUsIGludDMyX3Qgc2luZ2xlTGluZUJ5dGVzQ3V0b2ZmLAotICAgIHNpemVfdCBhbGlnbm1lbnQsIGJvb2wgY1N0eWxlLAotICAgIGRlYnVnUHJpbnRGdW5jIGZ1bmMsIHZvaWQqIGNvb2tpZSkKLXsKLSAgICBpZiAoYWxpZ25tZW50ID09IDApIHsKLSAgICAgICAgaWYgKGJ5dGVzUGVyTGluZSA+PSAxNikgYWxpZ25tZW50ID0gNDsKLSAgICAgICAgZWxzZSBpZiAoYnl0ZXNQZXJMaW5lID49IDgpIGFsaWdubWVudCA9IDI7Ci0gICAgICAgIGVsc2UgYWxpZ25tZW50ID0gMTsKLSAgICB9Ci0gICAgaWYgKGZ1bmMgPT0gTlVMTCkgZnVuYyA9IGRlZmF1bHRQcmludEZ1bmM7Ci0KLSAgICBzaXplX3Qgb2Zmc2V0OwotICAgIAotICAgIHVuc2lnbmVkIGNoYXIgKnBvcyA9ICh1bnNpZ25lZCBjaGFyICopYnVmOwotICAgIAotICAgIGlmIChwb3MgPT0gTlVMTCkgewotICAgICAgICBpZiAoc2luZ2xlTGluZUJ5dGVzQ3V0b2ZmIDwgMCkgZnVuYyhjb29raWUsICJcbiIpOwotICAgICAgICBmdW5jKGNvb2tpZSwgIihOVUxMKSIpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIAotICAgIGlmIChsZW5ndGggPT0gMCkgewotICAgICAgICBpZiAoc2luZ2xlTGluZUJ5dGVzQ3V0b2ZmIDwgMCkgZnVuYyhjb29raWUsICJcbiIpOwotICAgICAgICBmdW5jKGNvb2tpZSwgIihlbXB0eSkiKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICAKLSAgICBpZiAoKGludDMyX3QpbGVuZ3RoIDwgMCkgewotICAgICAgICBpZiAoc2luZ2xlTGluZUJ5dGVzQ3V0b2ZmIDwgMCkgZnVuYyhjb29raWUsICJcbiIpOwotICAgICAgICBjaGFyIGJ1Zls2NF07Ci0gICAgICAgIHNwcmludGYoYnVmLCAiKGJhZCBsZW5ndGg6ICVkKSIsIGxlbmd0aCk7Ci0gICAgICAgIGZ1bmMoY29va2llLCBidWYpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIAotICAgIGNoYXIgYnVmZmVyWzI1Nl07Ci0gICAgc3RhdGljIGNvbnN0IHNpemVfdCBtYXhCeXRlc1BlckxpbmUgPSAoc2l6ZW9mKGJ1ZmZlciktMS0xMS00KS8oMysxKTsKLSAgICAKLSAgICBpZiAoYnl0ZXNQZXJMaW5lID4gbWF4Qnl0ZXNQZXJMaW5lKSBieXRlc1BlckxpbmUgPSBtYXhCeXRlc1BlckxpbmU7Ci0gICAgCi0gICAgY29uc3QgYm9vbCBvbmVMaW5lID0gKGludDMyX3QpbGVuZ3RoIDw9IHNpbmdsZUxpbmVCeXRlc0N1dG9mZjsKLSAgICBib29sIG5ld0xpbmUgPSBmYWxzZTsKLSAgICBpZiAoY1N0eWxlKSB7Ci0gICAgICAgIGluZGVudCsrOwotICAgICAgICBmdW5jKGNvb2tpZSwgIntcbiIpOwotICAgICAgICBuZXdMaW5lID0gdHJ1ZTsKLSAgICB9IGVsc2UgaWYgKCFvbmVMaW5lKSB7Ci0gICAgICAgIGZ1bmMoY29va2llLCAiXG4iKTsKLSAgICAgICAgbmV3TGluZSA9IHRydWU7Ci0gICAgfQotICAgIAotICAgIGZvciAob2Zmc2V0ID0gMDsgOyBvZmZzZXQgKz0gYnl0ZXNQZXJMaW5lLCBwb3MgKz0gYnl0ZXNQZXJMaW5lKSB7Ci0gICAgICAgIGxvbmcgcmVtYWluID0gbGVuZ3RoOwotCi0gICAgICAgIGNoYXIqIGMgPSBidWZmZXI7Ci0gICAgICAgIGlmICghb25lTGluZSAmJiAhY1N0eWxlKSB7Ci0gICAgICAgICAgICBzcHJpbnRmKGMsICIweCUwOHg6ICIsIChpbnQpb2Zmc2V0KTsKLSAgICAgICAgICAgIGMgKz0gMTI7Ci0gICAgICAgIH0KLQotICAgICAgICBzaXplX3QgaW5kZXg7Ci0gICAgICAgIHNpemVfdCB3b3JkOwotICAgICAgICAKLSAgICAgICAgZm9yICh3b3JkID0gMDsgd29yZCA8IGJ5dGVzUGVyTGluZTsgKSB7Ci0KLSNpZmRlZiBIQVZFX0xJVFRMRV9FTkRJQU4KLSAgICAgICAgICAgIGNvbnN0IHNpemVfdCBzdGFydEluZGV4ID0gd29yZCsoYWxpZ25tZW50LShhbGlnbm1lbnQ/MTowKSk7Ci0gICAgICAgICAgICBjb25zdCBzc2l6ZV90IGRpciA9IC0xOwotI2Vsc2UKLSAgICAgICAgICAgIGNvbnN0IHNpemVfdCBzdGFydEluZGV4ID0gd29yZDsKLSAgICAgICAgICAgIGNvbnN0IHNzaXplX3QgZGlyID0gMTsKLSNlbmRpZgotCi0gICAgICAgICAgICBmb3IgKGluZGV4ID0gMDsgaW5kZXggPCBhbGlnbm1lbnQgfHwgKGFsaWdubWVudCA9PSAwICYmIGluZGV4IDwgYnl0ZXNQZXJMaW5lKTsgaW5kZXgrKykgewotICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgaWYgKCFjU3R5bGUpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGluZGV4ID09IDAgJiYgd29yZCA+IDAgJiYgYWxpZ25tZW50ID4gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgKmMrKyA9ICcgJzsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICBpZiAocmVtYWluLS0gPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB1bnNpZ25lZCBjaGFyIHZhbCA9ICoocG9zK3N0YXJ0SW5kZXgrKGluZGV4KmRpcikpOwotICAgICAgICAgICAgICAgICAgICAgICAgKmMrKyA9IG1ha2VoZXhkaWdpdCh2YWw+PjQpOwotICAgICAgICAgICAgICAgICAgICAgICAgKmMrKyA9IG1ha2VoZXhkaWdpdCh2YWwpOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKCFvbmVMaW5lKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAqYysrID0gJyAnOwotICAgICAgICAgICAgICAgICAgICAgICAgKmMrKyA9ICcgJzsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChyZW1haW4gPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoaW5kZXggPT0gMCAmJiB3b3JkID4gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICpjKysgPSAnLCc7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKmMrKyA9ICcgJzsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChpbmRleCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKmMrKyA9ICcwJzsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAqYysrID0gJ3gnOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgdW5zaWduZWQgY2hhciB2YWwgPSAqKHBvcytzdGFydEluZGV4KyhpbmRleCpkaXIpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICpjKysgPSBtYWtlaGV4ZGlnaXQodmFsPj40KTsKLSAgICAgICAgICAgICAgICAgICAgICAgICpjKysgPSBtYWtlaGV4ZGlnaXQodmFsKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlbWFpbi0tOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgCi0gICAgICAgICAgICB3b3JkICs9IGluZGV4OwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKCFjU3R5bGUpIHsKLSAgICAgICAgICAgIHJlbWFpbiA9IGxlbmd0aDsKLSAgICAgICAgICAgICpjKysgPSAnICc7Ci0gICAgICAgICAgICAqYysrID0gJ1wnJzsKLSAgICAgICAgICAgIGZvciAoaW5kZXggPSAwOyBpbmRleCA8IGJ5dGVzUGVyTGluZTsgaW5kZXgrKykgewotCi0gICAgICAgICAgICAgICAgaWYgKHJlbWFpbi0tID4gMCkgewotICAgICAgICAgICAgICAgICAgICBjb25zdCB1bnNpZ25lZCBjaGFyIHZhbCA9IHBvc1tpbmRleF07Ci0gICAgICAgICAgICAgICAgICAgICpjKysgPSAodmFsID49ICcgJyAmJiB2YWwgPCAxMjcpID8gdmFsIDogJy4nOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoIW9uZUxpbmUpIHsKLSAgICAgICAgICAgICAgICAgICAgKmMrKyA9ICcgJzsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgICpjKysgPSAnXCcnOwotICAgICAgICAgICAgaWYgKGxlbmd0aCA+IGJ5dGVzUGVyTGluZSkgKmMrKyA9ICdcbic7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpZiAocmVtYWluID4gMCkgKmMrKyA9ICcsJzsKLSAgICAgICAgICAgICpjKysgPSAnXG4nOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG5ld0xpbmUgJiYgaW5kZW50KSBmdW5jKGNvb2tpZSwgc3RyaW5nRm9ySW5kZW50KGluZGVudCkpOwotICAgICAgICAqYyA9IDA7Ci0gICAgICAgIGZ1bmMoY29va2llLCBidWZmZXIpOwotICAgICAgICBuZXdMaW5lID0gdHJ1ZTsKLSAgICAgICAgCi0gICAgICAgIGlmIChsZW5ndGggPD0gYnl0ZXNQZXJMaW5lKSBicmVhazsKLSAgICAgICAgbGVuZ3RoIC09IGJ5dGVzUGVyTGluZTsKLSAgICB9Ci0KLSAgICBpZiAoY1N0eWxlKSB7Ci0gICAgICAgIGlmIChpbmRlbnQgPiAwKSBmdW5jKGNvb2tpZSwgc3RyaW5nRm9ySW5kZW50KGluZGVudC0xKSk7Ci0gICAgICAgIGZ1bmMoY29va2llLCAifTsiKTsKLSAgICB9Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvRmlsZU1hcC5jcHAgYi9saWJzL3V0aWxzL0ZpbGVNYXAuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlMWJhOWIyLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvRmlsZU1hcC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwyMjIgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBTaGFyZWQgZmlsZSBtYXBwaW5nIGNsYXNzLgotLy8KLQotI2RlZmluZSBMT0dfVEFHICJmaWxlbWFwIgotCi0jaW5jbHVkZSA8dXRpbHMvRmlsZU1hcC5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLQotI2lmZGVmIEhBVkVfUE9TSVhfRklMRU1BUAotI2luY2x1ZGUgPHN5cy9tbWFuLmg+Ci0jZW5kaWYKLQotI2luY2x1ZGUgPHN0cmluZy5oPgotI2luY2x1ZGUgPG1lbW9yeS5oPgotI2luY2x1ZGUgPGVycm5vLmg+Ci0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi0vKnN0YXRpYyovIGxvbmcgRmlsZU1hcDo6bVBhZ2VTaXplID0gLTE7Ci0KLQotLyoKLSAqIENvbnN0cnVjdG9yLiAgQ3JlYXRlIGFuIGVtcHR5IG9iamVjdC4KLSAqLwotRmlsZU1hcDo6RmlsZU1hcCh2b2lkKQotICAgIDogbVJlZkNvdW50KDEpLCBtRmlsZU5hbWUoTlVMTCksIG1CYXNlUHRyKE5VTEwpLCBtQmFzZUxlbmd0aCgwKSwKLSAgICAgIG1EYXRhUHRyKE5VTEwpLCBtRGF0YUxlbmd0aCgwKQotewotfQotCi0vKgotICogRGVzdHJ1Y3Rvci4KLSAqLwotRmlsZU1hcDo6fkZpbGVNYXAodm9pZCkKLXsKLSAgICBhc3NlcnQobVJlZkNvdW50ID09IDApOwotCi0gICAgLy9wcmludGYoIisrKyByZW1vdmluZyBGaWxlTWFwICVwICV1XG4iLCBtRGF0YVB0ciwgbURhdGFMZW5ndGgpOwotCi0gICAgbVJlZkNvdW50ID0gLTEwMDsgICAgICAgLy8gaGVscCBjYXRjaCBkb3VibGUtZnJlZQotICAgIGlmIChtRmlsZU5hbWUgIT0gTlVMTCkgewotICAgICAgICBmcmVlKG1GaWxlTmFtZSk7Ci0gICAgfQotI2lmZGVmIEhBVkVfUE9TSVhfRklMRU1BUCAgICAKLSAgICBpZiAobXVubWFwKG1CYXNlUHRyLCBtQmFzZUxlbmd0aCkgIT0gMCkgewotICAgICAgICBMT0dEKCJtdW5tYXAoJXAsICVkKSBmYWlsZWRcbiIsIG1CYXNlUHRyLCAoaW50KSBtQmFzZUxlbmd0aCk7Ci0gICAgfQotI2VuZGlmCi0jaWZkZWYgSEFWRV9XSU4zMl9GSUxFTUFQCi0gICAgaWYgKCBVbm1hcFZpZXdPZkZpbGUobUJhc2VQdHIpID09IDApIHsKLSAgICAgICAgTE9HRCgiVW5tYXBWaWV3T2ZGaWxlKCVwKSBmYWlsZWQsIGVycm9yID0gJWxkXG4iLCBtQmFzZVB0ciwgCi0gICAgICAgICAgICAgIEdldExhc3RFcnJvcigpICk7Ci0gICAgfQotICAgIENsb3NlSGFuZGxlKG1GaWxlTWFwcGluZyk7Ci0gICAgQ2xvc2VIYW5kbGUobUZpbGVIYW5kbGUpOwotI2VuZGlmCi19Ci0KLQotLyoKLSAqIENyZWF0ZSBhIG5ldyBtYXBwaW5nIG9uIGFuIG9wZW4gZmlsZS4KLSAqCi0gKiBDbG9zaW5nIHRoZSBmaWxlIGRlc2NyaXB0b3IgZG9lcyBub3QgdW5tYXAgdGhlIHBhZ2VzLCBzbyB3ZSBkb24ndAotICogY2xhaW0gb3duZXJzaGlwIG9mIHRoZSBmZC4KLSAqCi0gKiBSZXR1cm5zICJmYWxzZSIgb24gZmFpbHVyZS4KLSAqLwotYm9vbCBGaWxlTWFwOjpjcmVhdGUoY29uc3QgY2hhciogb3JpZ0ZpbGVOYW1lLCBpbnQgZmQsIG9mZl90IG9mZnNldCwgc2l6ZV90IGxlbmd0aCwgYm9vbCByZWFkT25seSkKLXsKLSNpZmRlZiBIQVZFX1dJTjMyX0ZJTEVNQVAKLSAgICBpbnQgICAgIGFkanVzdDsKLSAgICBvZmZfdCAgIGFkak9mZnNldDsKLSAgICBzaXplX3QgIGFkakxlbmd0aDsKLQotICAgIGlmIChtUGFnZVNpemUgPT0gLTEpIHsKLSAgICAgICAgU1lTVEVNX0lORk8gIHNpOwotICAgICAgICAKLSAgICAgICAgR2V0U3lzdGVtSW5mbyggJnNpICk7Ci0gICAgICAgIG1QYWdlU2l6ZSA9IHNpLmR3QWxsb2NhdGlvbkdyYW51bGFyaXR5OwotICAgIH0KLQotICAgIERXT1JEICBwcm90ZWN0ID0gcmVhZE9ubHkgPyBQQUdFX1JFQURPTkxZIDogUEFHRV9SRUFEV1JJVEU7Ci0gICAgCi0gICAgbUZpbGVIYW5kbGUgID0gKEhBTkRMRSkgX2dldF9vc2ZoYW5kbGUoZmQpOwotICAgIG1GaWxlTWFwcGluZyA9IENyZWF0ZUZpbGVNYXBwaW5nKCBtRmlsZUhhbmRsZSwgTlVMTCwgcHJvdGVjdCwgMCwgMCwgTlVMTCk7Ci0gICAgaWYgKG1GaWxlTWFwcGluZyA9PSBOVUxMKSB7Ci0gICAgICAgIExPR0UoIkNyZWF0ZUZpbGVNYXBwaW5nKCVwLCAlbHgpIGZhaWxlZCB3aXRoIGVycm9yICVsZFxuIiwKLSAgICAgICAgICAgICAgbUZpbGVIYW5kbGUsIHByb3RlY3QsIEdldExhc3RFcnJvcigpICk7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgCi0gICAgYWRqdXN0ICAgID0gb2Zmc2V0ICUgbVBhZ2VTaXplOwotICAgIGFkak9mZnNldCA9IG9mZnNldCAtIGFkanVzdDsKLSAgICBhZGpMZW5ndGggPSBsZW5ndGggKyBhZGp1c3Q7Ci0gICAgCi0gICAgbUJhc2VQdHIgPSBNYXBWaWV3T2ZGaWxlKCBtRmlsZU1hcHBpbmcsIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVhZE9ubHkgPyBGSUxFX01BUF9SRUFEIDogRklMRV9NQVBfQUxMX0FDQ0VTUywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoRFdPUkQpKGFkak9mZnNldCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhZGpMZW5ndGggKTsKLSAgICBpZiAobUJhc2VQdHIgPT0gTlVMTCkgewotICAgICAgICBMT0dFKCJNYXBWaWV3T2ZGaWxlKCVsZCwgJWxkKSBmYWlsZWQgd2l0aCBlcnJvciAlbGRcbiIsCi0gICAgICAgICAgICAgIGFkak9mZnNldCwgYWRqTGVuZ3RoLCBHZXRMYXN0RXJyb3IoKSApOwotICAgICAgICBDbG9zZUhhbmRsZShtRmlsZU1hcHBpbmcpOwotICAgICAgICBtRmlsZU1hcHBpbmcgPSBJTlZBTElEX0hBTkRMRV9WQUxVRTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSNlbmRpZgotI2lmZGVmIEhBVkVfUE9TSVhfRklMRU1BUAotICAgIGludCAgICAgcHJvdCwgZmxhZ3MsIGFkanVzdDsKLSAgICBvZmZfdCAgIGFkak9mZnNldDsKLSAgICBzaXplX3QgIGFkakxlbmd0aDsKLQotICAgIHZvaWQqIHB0cjsKLQotICAgIGFzc2VydChtUmVmQ291bnQgPT0gMSk7Ci0gICAgYXNzZXJ0KGZkID49IDApOwotICAgIGFzc2VydChvZmZzZXQgPj0gMCk7Ci0gICAgYXNzZXJ0KGxlbmd0aCA+IDApOwotCi0gICAgLyogaW5pdCBvbiBmaXJzdCB1c2UgKi8KLSAgICBpZiAobVBhZ2VTaXplID09IC0xKSB7Ci0jaWYgTk9UX1VTSU5HX0tMSUJDCi0gICAgICAgIG1QYWdlU2l6ZSA9IHN5c2NvbmYoX1NDX1BBR0VTSVpFKTsKLSAgICAgICAgaWYgKG1QYWdlU2l6ZSA9PSAtMSkgewotICAgICAgICAgICAgTE9HRSgiY291bGQgbm90IGdldCBfU0NfUEFHRVNJWkVcbiIpOwotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0jZWxzZQotICAgICAgICAvKiB0aGlzIGhvbGRzIGZvciBMaW51eCwgRGFyd2luLCBDeWd3aW4sIGFuZCBkb2Vzbid0IHBhaW4gdGhlIEFSTSAqLwotICAgICAgICBtUGFnZVNpemUgPSA0MDk2OwotI2VuZGlmCi0gICAgfQotCi0gICAgYWRqdXN0ICAgPSBvZmZzZXQgJSBtUGFnZVNpemU7Ci10cnlfYWdhaW46Ci0gICAgYWRqT2Zmc2V0ID0gb2Zmc2V0IC0gYWRqdXN0OwotICAgIGFkakxlbmd0aCA9IGxlbmd0aCArIGFkanVzdDsKLQotICAgIGZsYWdzID0gTUFQX1NIQVJFRDsKLSAgICBwcm90ID0gUFJPVF9SRUFEOwotICAgIGlmICghcmVhZE9ubHkpCi0gICAgICAgIHByb3QgfD0gUFJPVF9XUklURTsKLQotICAgIHB0ciA9IG1tYXAoTlVMTCwgYWRqTGVuZ3RoLCBwcm90LCBmbGFncywgZmQsIGFkak9mZnNldCk7Ci0gICAgaWYgKHB0ciA9PSBNQVBfRkFJTEVEKSB7Ci0gICAgCS8vIEN5Z3dpbiBkb2VzIG5vdCBzZWVtIHRvIGxpa2UgZmlsZSBtYXBwaW5nIGZpbGVzIGZyb20gYW4gb2Zmc2V0LgotICAgIAkvLyBTbyBpZiB3ZSBmYWlsLCB0cnkgYWdhaW4gd2l0aCBvZmZzZXQgemVybwotICAgIAlpZiAoYWRqT2Zmc2V0ID4gMCkgewotICAgIAkJYWRqdXN0ID0gb2Zmc2V0OwotICAgIAkJZ290byB0cnlfYWdhaW47Ci0gICAgCX0KLSAgICAKLSAgICAgICAgTE9HRSgibW1hcCglbGQsJWxkKSBmYWlsZWQ6ICVzXG4iLAotICAgICAgICAgICAgKGxvbmcpIGFkak9mZnNldCwgKGxvbmcpIGFkakxlbmd0aCwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSAgICBtQmFzZVB0ciA9IHB0cjsKLSNlbmRpZiAvKiBIQVZFX1BPU0lYX0ZJTEVNQVAgKi8KLQotICAgIG1GaWxlTmFtZSA9IG9yaWdGaWxlTmFtZSAhPSBOVUxMID8gc3RyZHVwKG9yaWdGaWxlTmFtZSkgOiBOVUxMOwotICAgIG1CYXNlTGVuZ3RoID0gYWRqTGVuZ3RoOwotICAgIG1EYXRhT2Zmc2V0ID0gb2Zmc2V0OwotICAgIG1EYXRhUHRyID0gKGNoYXIqKSBtQmFzZVB0ciArIGFkanVzdDsKLSAgICBtRGF0YUxlbmd0aCA9IGxlbmd0aDsKLQotICAgIGFzc2VydChtQmFzZVB0ciAhPSBOVUxMKTsKLQotICAgIExPR1YoIk1BUDogYmFzZSAlcC8lZCBkYXRhICVwLyVkXG4iLAotICAgICAgICBtQmFzZVB0ciwgKGludCkgbUJhc2VMZW5ndGgsIG1EYXRhUHRyLCAoaW50KSBtRGF0YUxlbmd0aCk7Ci0KLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotLyoKLSAqIFByb3ZpZGUgZ3VpZGFuY2UgdG8gdGhlIHN5c3RlbS4KLSAqLwotaW50IEZpbGVNYXA6OmFkdmlzZShNYXBBZHZpY2UgYWR2aWNlKQotewotI2lmIEhBVkVfTUFEVklTRQotICAgIGludCBjYywgc3lzQWR2aWNlOwotCi0gICAgc3dpdGNoIChhZHZpY2UpIHsKLSAgICAgICAgY2FzZSBOT1JNQUw6ICAgICAgICBzeXNBZHZpY2UgPSBNQURWX05PUk1BTDsgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIFJBTkRPTTogICAgICAgIHN5c0FkdmljZSA9IE1BRFZfUkFORE9NOyAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgU0VRVUVOVElBTDogICAgc3lzQWR2aWNlID0gTUFEVl9TRVFVRU5USUFMOyAgICBicmVhazsKLSAgICAgICAgY2FzZSBXSUxMTkVFRDogICAgICBzeXNBZHZpY2UgPSBNQURWX1dJTExORUVEOyAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIERPTlRORUVEOiAgICAgIHN5c0FkdmljZSA9IE1BRFZfRE9OVE5FRUQ7ICAgICAgYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYXNzZXJ0KGZhbHNlKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotCi0gICAgY2MgPSBtYWR2aXNlKG1CYXNlUHRyLCBtQmFzZUxlbmd0aCwgc3lzQWR2aWNlKTsKLSAgICBpZiAoY2MgIT0gMCkKLSAgICAgICAgTE9HVygibWFkdmlzZSglZCkgZmFpbGVkOiAlc1xuIiwgc3lzQWR2aWNlLCBzdHJlcnJvcihlcnJubykpOwotICAgIHJldHVybiBjYzsKLSNlbHNlCi0JcmV0dXJuIC0xOwotI2VuZGlmIC8vIEhBVkVfTUFEVklTRQotfQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9JRGF0YUNvbm5lY3Rpb24uY3BwIGIvbGlicy91dGlscy9JRGF0YUNvbm5lY3Rpb24uY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjNmQ0OWFhLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvSURhdGFDb25uZWN0aW9uLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDg5ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDYgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9QYXJjZWwuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0lEYXRhQ29ubmVjdGlvbi5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1lbnVtCi17Ci0gICAgQ09OTkVDVF9UUkFOU0FDVElPTiA9IElCaW5kZXI6OkZJUlNUX0NBTExfVFJBTlNBQ1RJT04sCi0gICAgRElTQ09OTkVDVF9UUkFOU0FDVElPTiA9IElCaW5kZXI6OkZJUlNUX0NBTExfVFJBTlNBQ1RJT04gKyAxCi19OwotCi1jbGFzcyBCcERhdGFDb25uZWN0aW9uIDogcHVibGljIEJwSW50ZXJmYWNlPElEYXRhQ29ubmVjdGlvbj4KLXsKLXB1YmxpYzoKLSAgICBCcERhdGFDb25uZWN0aW9uOjpCcERhdGFDb25uZWN0aW9uKGNvbnN0IHNwPElCaW5kZXI+JiBpbXBsKQotICAgICAgICA6IEJwSW50ZXJmYWNlPElEYXRhQ29ubmVjdGlvbj4oaW1wbCkKLSAgICB7Ci0gICAgfQotCi0JdmlydHVhbCB2b2lkIGNvbm5lY3QoKQotCXsKLQkJUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSURhdGFDb25uZWN0aW9uOjpkZXNjcmlwdG9yKCkpOwotCQlyZW1vdGUoKS0+dHJhbnNhY3QoQ09OTkVDVF9UUkFOU0FDVElPTiwgZGF0YSwgJnJlcGx5KTsKLQl9Ci0JCi0JdmlydHVhbCB2b2lkIGRpc2Nvbm5lY3QoKQotCXsKLQkJUGFyY2VsIGRhdGEsIHJlcGx5OwotCQlyZW1vdGUoKS0+dHJhbnNhY3QoRElTQ09OTkVDVF9UUkFOU0FDVElPTiwgZGF0YSwgJnJlcGx5KTsKLQl9Ci19OwotCi1JTVBMRU1FTlRfTUVUQV9JTlRFUkZBQ0UoRGF0YUNvbm5lY3Rpb24sICJhbmRyb2lkLnV0aWxzLklEYXRhQ29ubmVjdGlvbiIpOwotCi0jZGVmaW5lIENIRUNLX0lOVEVSRkFDRShpbnRlcmZhY2UsIGRhdGEsIHJlcGx5KSBcCi0gICAgICAgIGRvIHsgaWYgKCFkYXRhLmVuZm9yY2VJbnRlcmZhY2UoaW50ZXJmYWNlOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpKSB7IFwKLSAgICAgICAgICAgIExPR1coIkNhbGwgaW5jb3JyZWN0bHkgcm91dGVkIHRvICIgI2ludGVyZmFjZSk7IFwKLSAgICAgICAgICAgIHJldHVybiBQRVJNSVNTSU9OX0RFTklFRDsgXAotICAgICAgICB9IH0gd2hpbGUgKDApCi0KLXN0YXR1c190IEJuRGF0YUNvbm5lY3Rpb246Om9uVHJhbnNhY3QodWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBzd2l0Y2goY29kZSkKLSAgICB7Ci0JCWNhc2UgQ09OTkVDVF9UUkFOU0FDVElPTjoKLQkJeyAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJRGF0YUNvbm5lY3Rpb24sIGRhdGEsIHJlcGx5KTsKLQkJCWNvbm5lY3QoKTsKLQkJCXJldHVybiBOT19FUlJPUjsKLQkJfSAgICAKLQkJCi0JCWNhc2UgRElTQ09OTkVDVF9UUkFOU0FDVElPTjoKLQkJeyAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJRGF0YUNvbm5lY3Rpb24sIGRhdGEsIHJlcGx5KTsKLQkJCWRpc2Nvbm5lY3QoKTsKLQkJCXJldHVybiBOT19FUlJPUjsKLQkJfQotICAgICAgIAotCQlkZWZhdWx0OgotCQkJcmV0dXJuIEJCaW5kZXI6Om9uVHJhbnNhY3QoY29kZSwgZGF0YSwgcmVwbHksIGZsYWdzKTsKLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvSUludGVyZmFjZS5jcHAgYi9saWJzL3V0aWxzL0lJbnRlcmZhY2UuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2ZWE4MTc4Li4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvSUludGVyZmFjZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwzNSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2luY2x1ZGUgPHV0aWxzL0lJbnRlcmZhY2UuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3A8SUJpbmRlcj4gSUludGVyZmFjZTo6YXNCaW5kZXIoKQotewotICAgIHJldHVybiB0aGlzID8gb25Bc0JpbmRlcigpIDogTlVMTDsKLX0KLQotc3A8Y29uc3QgSUJpbmRlcj4gSUludGVyZmFjZTo6YXNCaW5kZXIoKSBjb25zdAotewotICAgIHJldHVybiB0aGlzID8gY29uc3RfY2FzdDxJSW50ZXJmYWNlKj4odGhpcyktPm9uQXNCaW5kZXIoKSA6IE5VTEw7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9JTWVtb3J5LmNwcCBiL2xpYnMvdXRpbHMvSU1lbW9yeS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQyOWJjMmIuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9JTWVtb3J5LmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDQ4NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJJTWVtb3J5IgotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPHVuaXN0ZC5oPgotCi0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8c3lzL21tYW4uaD4KLQotI2luY2x1ZGUgPHV0aWxzL0lNZW1vcnkuaD4KLSNpbmNsdWRlIDx1dGlscy9LZXllZFZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9QYXJjZWwuaD4KLSNpbmNsdWRlIDx1dGlscy9DYWxsU3RhY2suaD4KLQotI2RlZmluZSBWRVJCT1NFICAgMAotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgSGVhcENhY2hlIDogcHVibGljIElCaW5kZXI6OkRlYXRoUmVjaXBpZW50Ci17Ci1wdWJsaWM6Ci0gICAgSGVhcENhY2hlKCk7Ci0gICAgdmlydHVhbCB+SGVhcENhY2hlKCk7Ci0gICAgCi0gICAgdmlydHVhbCB2b2lkIGJpbmRlckRpZWQoY29uc3Qgd3A8SUJpbmRlcj4mIHdobyk7Ci0KLSAgICBzcDxJTWVtb3J5SGVhcD4gZmluZF9oZWFwKGNvbnN0IHNwPElCaW5kZXI+JiBiaW5kZXIpOyAKLSAgICB2b2lkIHBpbl9oZWFwKGNvbnN0IHNwPElCaW5kZXI+JiBiaW5kZXIpOyAKLSAgICB2b2lkIGZyZWVfaGVhcChjb25zdCBzcDxJQmluZGVyPiYgYmluZGVyKTsgCi0gICAgc3A8SU1lbW9yeUhlYXA+IGdldF9oZWFwKGNvbnN0IHNwPElCaW5kZXI+JiBiaW5kZXIpOwotICAgIHZvaWQgZHVtcF9oZWFwcygpOwotCi1wcml2YXRlOgotICAgIC8vIEZvciBJTWVtb3J5LmNwcAotICAgIHN0cnVjdCBoZWFwX2luZm9fdCB7Ci0gICAgICAgIHNwPElNZW1vcnlIZWFwPiBoZWFwOwotICAgICAgICBpbnQzMl90ICAgICAgICAgY291bnQ7Ci0gICAgfTsKLQotICAgIHZvaWQgZnJlZV9oZWFwKGNvbnN0IHdwPElCaW5kZXI+JiBiaW5kZXIpOyAKLQotICAgIE11dGV4IG1IZWFwQ2FjaGVMb2NrOwotICAgIEtleWVkVmVjdG9yPCB3cDxJQmluZGVyPiwgaGVhcF9pbmZvX3QgPiBtSGVhcENhY2hlOwotfTsKLQotc3RhdGljIHNwPEhlYXBDYWNoZT4gZ0hlYXBDYWNoZSA9IG5ldyBIZWFwQ2FjaGUoKTsKLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotZW51bSB7Ci0gICAgSEVBUF9JRCA9IElCaW5kZXI6OkZJUlNUX0NBTExfVFJBTlNBQ1RJT04KLX07Ci0KLWNsYXNzIEJwTWVtb3J5SGVhcCA6IHB1YmxpYyBCcEludGVyZmFjZTxJTWVtb3J5SGVhcD4KLXsKLXB1YmxpYzoKLSAgICBCcE1lbW9yeUhlYXAoY29uc3Qgc3A8SUJpbmRlcj4mIGltcGwpOwotICAgIHZpcnR1YWwgfkJwTWVtb3J5SGVhcCgpOwotCi0gICAgdmlydHVhbCBpbnQgZ2V0SGVhcElEKCkgY29uc3Q7Ci0gICAgdmlydHVhbCB2b2lkKiBnZXRCYXNlKCkgY29uc3Q7Ci0gICAgdmlydHVhbCBzaXplX3QgZ2V0U2l6ZSgpIGNvbnN0OwotICAgIHZpcnR1YWwgdWludDMyX3QgZ2V0RmxhZ3MoKSBjb25zdDsKLQotcHJpdmF0ZToKLSAgICBmcmllbmQgY2xhc3MgSU1lbW9yeTsKLSAgICBmcmllbmQgY2xhc3MgSGVhcENhY2hlOwotICAgIAotICAgIC8vIGZvciBkZWJ1Z2dpbmcgaW4gdGhpcyBtb2R1bGUKLSAgICBzdGF0aWMgaW5saW5lIHNwPElNZW1vcnlIZWFwPiBmaW5kX2hlYXAoY29uc3Qgc3A8SUJpbmRlcj4mIGJpbmRlcikgewotICAgICAgICByZXR1cm4gZ0hlYXBDYWNoZS0+ZmluZF9oZWFwKGJpbmRlcik7Ci0gICAgfQotICAgIHN0YXRpYyBpbmxpbmUgdm9pZCBmcmVlX2hlYXAoY29uc3Qgc3A8SUJpbmRlcj4mIGJpbmRlcikgewotICAgICAgICBnSGVhcENhY2hlLT5mcmVlX2hlYXAoYmluZGVyKTsKLSAgICB9Ci0gICAgc3RhdGljIGlubGluZSBzcDxJTWVtb3J5SGVhcD4gZ2V0X2hlYXAoY29uc3Qgc3A8SUJpbmRlcj4mIGJpbmRlcikgewotICAgICAgICByZXR1cm4gZ0hlYXBDYWNoZS0+Z2V0X2hlYXAoYmluZGVyKTsKLSAgICB9Ci0gICAgc3RhdGljIGlubGluZSB2b2lkIGR1bXBfaGVhcHMoKSB7Ci0gICAgICAgIGdIZWFwQ2FjaGUtPmR1bXBfaGVhcHMoKTsgICAgICAgCi0gICAgfQotICAgIHZvaWQgaW5saW5lIHBpbl9oZWFwKCkgY29uc3QgewotICAgICAgICBnSGVhcENhY2hlLT5waW5faGVhcChjb25zdF9jYXN0PEJwTWVtb3J5SGVhcCo+KHRoaXMpLT5hc0JpbmRlcigpKTsKLSAgICB9Ci0KLSAgICB2b2lkIGFzc2VydE1hcHBlZCgpIGNvbnN0OwotICAgIHZvaWQgYXNzZXJ0UmVhbGx5TWFwcGVkKCkgY29uc3Q7Ci0gICAgdm9pZCBwaW5IZWFwKCkgY29uc3Q7Ci0KLSAgICBtdXRhYmxlIHZvbGF0aWxlIGludDMyX3QgbUhlYXBJZDsKLSAgICBtdXRhYmxlIHZvaWQqICAgICAgIG1CYXNlOwotICAgIG11dGFibGUgc2l6ZV90ICAgICAgbVNpemU7Ci0gICAgbXV0YWJsZSB1aW50MzJfdCAgICBtRmxhZ3M7Ci0gICAgbXV0YWJsZSBib29sICAgICAgICBtUmVhbEhlYXA7Ci0gICAgbXV0YWJsZSBNdXRleCAgICAgICBtTG9jazsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotZW51bSB7Ci0gICAgR0VUX01FTU9SWSA9IElCaW5kZXI6OkZJUlNUX0NBTExfVFJBTlNBQ1RJT04KLX07Ci0KLWNsYXNzIEJwTWVtb3J5IDogcHVibGljIEJwSW50ZXJmYWNlPElNZW1vcnk+Ci17Ci1wdWJsaWM6Ci0gICAgQnBNZW1vcnkoY29uc3Qgc3A8SUJpbmRlcj4mIGltcGwpOwotICAgIHZpcnR1YWwgfkJwTWVtb3J5KCk7Ci0gICAgdmlydHVhbCBzcDxJTWVtb3J5SGVhcD4gZ2V0TWVtb3J5KHNzaXplX3QqIG9mZnNldD0wLCBzaXplX3QqIHNpemU9MCkgY29uc3Q7Ci0gICAgCi1wcml2YXRlOgotICAgIG11dGFibGUgc3A8SU1lbW9yeUhlYXA+IG1IZWFwOwotICAgIG11dGFibGUgc3NpemVfdCBtT2Zmc2V0OwotICAgIG11dGFibGUgc2l6ZV90IG1TaXplOwotfTsKLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotdm9pZCogSU1lbW9yeTo6ZmFzdFBvaW50ZXIoY29uc3Qgc3A8SUJpbmRlcj4mIGJpbmRlciwgc3NpemVfdCBvZmZzZXQpIGNvbnN0Ci17Ci0gICAgc3A8SU1lbW9yeUhlYXA+IHJlYWxIZWFwID0gQnBNZW1vcnlIZWFwOjpnZXRfaGVhcChiaW5kZXIpOwotICAgIHZvaWQqIGNvbnN0IGJhc2UgPSByZWFsSGVhcC0+YmFzZSgpOwotICAgIGlmIChiYXNlID09IE1BUF9GQUlMRUQpCi0gICAgICAgIHJldHVybiAwOwotICAgIHJldHVybiBzdGF0aWNfY2FzdDxjaGFyKj4oYmFzZSkgKyBvZmZzZXQ7Ci19Ci0KLXZvaWQqIElNZW1vcnk6OnBvaW50ZXIoKSBjb25zdCB7Ci0gICAgc3NpemVfdCBvZmZzZXQ7Ci0gICAgc3A8SU1lbW9yeUhlYXA+IGhlYXAgPSBnZXRNZW1vcnkoJm9mZnNldCk7Ci0gICAgdm9pZCogY29uc3QgYmFzZSA9IGhlYXAhPTAgPyBoZWFwLT5iYXNlKCkgOiBNQVBfRkFJTEVEOwotICAgIGlmIChiYXNlID09IE1BUF9GQUlMRUQpCi0gICAgICAgIHJldHVybiAwOwotICAgIHJldHVybiBzdGF0aWNfY2FzdDxjaGFyKj4oYmFzZSkgKyBvZmZzZXQ7Ci19Ci0KLXNpemVfdCBJTWVtb3J5OjpzaXplKCkgY29uc3QgewotICAgIHNpemVfdCBzaXplOwotICAgIGdldE1lbW9yeShOVUxMLCAmc2l6ZSk7Ci0gICAgcmV0dXJuIHNpemU7Ci19Ci0KLXNzaXplX3QgSU1lbW9yeTo6b2Zmc2V0KCkgY29uc3QgewotICAgIHNzaXplX3Qgb2Zmc2V0OwotICAgIGdldE1lbW9yeSgmb2Zmc2V0KTsKLSAgICByZXR1cm4gb2Zmc2V0OwotfQotCi0vKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi1CcE1lbW9yeTo6QnBNZW1vcnkoY29uc3Qgc3A8SUJpbmRlcj4mIGltcGwpCi0gICAgOiBCcEludGVyZmFjZTxJTWVtb3J5PihpbXBsKSwgbU9mZnNldCgwKSwgbVNpemUoMCkKLXsKLX0KLQotQnBNZW1vcnk6On5CcE1lbW9yeSgpCi17Ci19Ci0KLXNwPElNZW1vcnlIZWFwPiBCcE1lbW9yeTo6Z2V0TWVtb3J5KHNzaXplX3QqIG9mZnNldCwgc2l6ZV90KiBzaXplKSBjb25zdAotewotICAgIGlmIChtSGVhcCA9PSAwKSB7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElNZW1vcnk6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGlmIChyZW1vdGUoKS0+dHJhbnNhY3QoR0VUX01FTU9SWSwgZGF0YSwgJnJlcGx5KSA9PSBOT19FUlJPUikgewotICAgICAgICAgICAgc3A8SUJpbmRlcj4gaGVhcCA9IHJlcGx5LnJlYWRTdHJvbmdCaW5kZXIoKTsKLSAgICAgICAgICAgIHNzaXplX3QgbyA9IHJlcGx5LnJlYWRJbnQzMigpOwotICAgICAgICAgICAgc2l6ZV90IHMgPSByZXBseS5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgIGlmIChoZWFwICE9IDApIHsKLSAgICAgICAgICAgICAgICBtSGVhcCA9IGludGVyZmFjZV9jYXN0PElNZW1vcnlIZWFwPihoZWFwKTsKLSAgICAgICAgICAgICAgICBpZiAobUhlYXAgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICBtT2Zmc2V0ID0gbzsKLSAgICAgICAgICAgICAgICAgICAgbVNpemUgPSBzOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICBpZiAob2Zmc2V0KSAqb2Zmc2V0ID0gbU9mZnNldDsKLSAgICBpZiAoc2l6ZSkgKnNpemUgPSBtU2l6ZTsKLSAgICByZXR1cm4gbUhlYXA7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1JTVBMRU1FTlRfTUVUQV9JTlRFUkZBQ0UoTWVtb3J5LCAiYW5kcm9pZC51dGlscy5JTWVtb3J5Iik7Ci0KLSNkZWZpbmUgQ0hFQ0tfSU5URVJGQUNFKGludGVyZmFjZSwgZGF0YSwgcmVwbHkpIFwKLSAgICAgICAgZG8geyBpZiAoIWRhdGEuZW5mb3JjZUludGVyZmFjZShpbnRlcmZhY2U6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSkpIHsgXAotICAgICAgICAgICAgTE9HVygiQ2FsbCBpbmNvcnJlY3RseSByb3V0ZWQgdG8gIiAjaW50ZXJmYWNlKTsgXAotICAgICAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOyBcCi0gICAgICAgIH0gfSB3aGlsZSAoMCkKLQotc3RhdHVzX3QgQm5NZW1vcnk6Om9uVHJhbnNhY3QoCi0gICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBzd2l0Y2goY29kZSkgewotICAgICAgICBjYXNlIEdFVF9NRU1PUlk6IHsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJTWVtb3J5LCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBzc2l6ZV90IG9mZnNldDsKLSAgICAgICAgICAgIHNpemVfdCBzaXplOwotICAgICAgICAgICAgcmVwbHktPndyaXRlU3Ryb25nQmluZGVyKCBnZXRNZW1vcnkoJm9mZnNldCwgJnNpemUpLT5hc0JpbmRlcigpICk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihvZmZzZXQpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIoc2l6ZSk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gQkJpbmRlcjo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotICAgIH0KLX0KLQotCi0vKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi1CcE1lbW9yeUhlYXA6OkJwTWVtb3J5SGVhcChjb25zdCBzcDxJQmluZGVyPiYgaW1wbCkKLSAgICA6IEJwSW50ZXJmYWNlPElNZW1vcnlIZWFwPihpbXBsKSwKLSAgICAgICAgbUhlYXBJZCgtMSksIG1CYXNlKE1BUF9GQUlMRUQpLCBtU2l6ZSgwKSwgbUZsYWdzKDApLCBtUmVhbEhlYXAoZmFsc2UpCi17Ci19Ci0KLUJwTWVtb3J5SGVhcDo6fkJwTWVtb3J5SGVhcCgpIHsKLSAgICBpZiAobUhlYXBJZCAhPSAtMSkgewotICAgICAgICBjbG9zZShtSGVhcElkKTsKLSAgICAgICAgaWYgKG1SZWFsSGVhcCkgewotICAgICAgICAgICAgLy8gYnkgY29uc3RydWN0aW9uIHdlJ3JlIHRoZSBsYXN0IG9uZQotICAgICAgICAgICAgaWYgKG1CYXNlICE9IE1BUF9GQUlMRUQpIHsKLSAgICAgICAgICAgICAgICBzcDxJQmluZGVyPiBiaW5kZXIgPSBjb25zdF9jYXN0PEJwTWVtb3J5SGVhcCo+KHRoaXMpLT5hc0JpbmRlcigpOwotCi0gICAgICAgICAgICAgICAgaWYgKFZFUkJPU0UpIHsKLSAgICAgICAgICAgICAgICAgICAgTE9HRCgiVU5NQVBQSU5HIGJpbmRlcj0lcCwgaGVhcD0lcCwgc2l6ZT0lZCwgZmQ9JWQiLCAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBiaW5kZXIuZ2V0KCksIHRoaXMsIG1TaXplLCBtSGVhcElkKTsKLSAgICAgICAgICAgICAgICAgICAgQ2FsbFN0YWNrIHN0YWNrOwotICAgICAgICAgICAgICAgICAgICBzdGFjay51cGRhdGUoKTsKLSAgICAgICAgICAgICAgICAgICAgc3RhY2suZHVtcCgiY2FsbHN0YWNrIik7Ci0gICAgICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAgICAgbXVubWFwKG1CYXNlLCBtU2l6ZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyByZW1vdmUgZnJvbSBsaXN0IG9ubHkgaWYgaXQgd2FzIG1hcHBlZCBiZWZvcmUKLSAgICAgICAgICAgIHNwPElCaW5kZXI+IGJpbmRlciA9IGNvbnN0X2Nhc3Q8QnBNZW1vcnlIZWFwKj4odGhpcyktPmFzQmluZGVyKCk7Ci0gICAgICAgICAgICBmcmVlX2hlYXAoYmluZGVyKTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotdm9pZCBCcE1lbW9yeUhlYXA6OmFzc2VydE1hcHBlZCgpIGNvbnN0Ci17Ci0gICAgaWYgKG1IZWFwSWQgPT0gLTEpIHsKLSAgICAgICAgc3A8SUJpbmRlcj4gYmluZGVyKGNvbnN0X2Nhc3Q8QnBNZW1vcnlIZWFwKj4odGhpcyktPmFzQmluZGVyKCkpOwotICAgICAgICBzcDxCcE1lbW9yeUhlYXA+IGhlYXAoc3RhdGljX2Nhc3Q8QnBNZW1vcnlIZWFwKj4oZmluZF9oZWFwKGJpbmRlcikuZ2V0KCkpKTsKLSAgICAgICAgaGVhcC0+YXNzZXJ0UmVhbGx5TWFwcGVkKCk7Ci0gICAgICAgIGlmIChoZWFwLT5tQmFzZSAhPSBNQVBfRkFJTEVEKSB7Ci0gICAgICAgICAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgICAgICAgICAgaWYgKG1IZWFwSWQgPT0gLTEpIHsKLSAgICAgICAgICAgICAgICBtQmFzZSAgID0gaGVhcC0+bUJhc2U7Ci0gICAgICAgICAgICAgICAgbVNpemUgICA9IGhlYXAtPm1TaXplOwotICAgICAgICAgICAgICAgIGFuZHJvaWRfYXRvbWljX3dyaXRlKCBkdXAoIGhlYXAtPm1IZWFwSWQgKSwgJm1IZWFwSWQgKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIHNvbWV0aGluZyB3ZW50IHdyb25nCi0gICAgICAgICAgICBmcmVlX2hlYXAoYmluZGVyKTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotdm9pZCBCcE1lbW9yeUhlYXA6OmFzc2VydFJlYWxseU1hcHBlZCgpIGNvbnN0Ci17Ci0gICAgaWYgKG1IZWFwSWQgPT0gLTEpIHsKLQotICAgICAgICAvLyByZW1vdGUgY2FsbCB3aXRob3V0IG1Mb2NrIGhlbGQsIHdvcnNlIGNhc2Ugc2NlbmFyaW8sIHdlIGVuZCB1cAotICAgICAgICAvLyBjYWxsaW5nIHRyYW5zYWN0KCkgZnJvbSBtdWx0aXBsZSB0aHJlYWRzLCBidXQgdGhhdCdzIG5vdCBhIHByb2JsZW0sCi0gICAgICAgIC8vIG9ubHkgbW1hcCBiZWxvdyBtdXN0IGJlIGluIHRoZSBjcml0aWNhbCBzZWN0aW9uLgotICAgICAgICAKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSU1lbW9yeUhlYXA6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIHN0YXR1c190IGVyciA9IHJlbW90ZSgpLT50cmFuc2FjdChIRUFQX0lELCBkYXRhLCAmcmVwbHkpOwotICAgICAgICBpbnQgcGFyY2VsX2ZkID0gcmVwbHkucmVhZEZpbGVEZXNjcmlwdG9yKCk7Ci0gICAgICAgIHNzaXplX3Qgc2l6ZSA9IHJlcGx5LnJlYWRJbnQzMigpOwotICAgICAgICB1aW50MzJfdCBmbGFncyA9IHJlcGx5LnJlYWRJbnQzMigpOwotCi0gICAgICAgIExPR0VfSUYoZXJyLCAiYmluZGVyPSVwIHRyYW5zYWN0aW9uIGZhaWxlZCBmZD0lZCwgc2l6ZT0lZCwgZXJyPSVkICglcykiLAotICAgICAgICAgICAgICAgIGFzQmluZGVyKCkuZ2V0KCksIHBhcmNlbF9mZCwgc2l6ZSwgZXJyLCBzdHJlcnJvcigtZXJyKSk7Ci0KLSAgICAgICAgaW50IGZkID0gZHVwKCBwYXJjZWxfZmQgKTsKLSAgICAgICAgTE9HRV9JRihmZD09LTEsICJjYW5ub3QgZHVwIGZkPSVkLCBzaXplPSVkLCBlcnI9JWQgKCVzKSIsCi0gICAgICAgICAgICAgICAgcGFyY2VsX2ZkLCBzaXplLCBlcnIsIHN0cmVycm9yKGVycm5vKSk7Ci0KLSAgICAgICAgaW50IGFjY2VzcyA9IFBST1RfUkVBRDsKLSAgICAgICAgaWYgKCEoZmxhZ3MgJiBSRUFEX09OTFkpKSB7Ci0gICAgICAgICAgICBhY2Nlc3MgfD0gUFJPVF9XUklURTsKLSAgICAgICAgfQotCi0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgICAgIGlmIChtSGVhcElkID09IC0xKSB7Ci0gICAgICAgICAgICBtUmVhbEhlYXAgPSB0cnVlOwotICAgICAgICAgICAgbUJhc2UgPSBtbWFwKDAsIHNpemUsIGFjY2VzcywgTUFQX1NIQVJFRCwgZmQsIDApOwotICAgICAgICAgICAgaWYgKG1CYXNlID09IE1BUF9GQUlMRUQpIHsKLSAgICAgICAgICAgICAgICBMT0dFKCJjYW5ub3QgbWFwIEJwTWVtb3J5SGVhcCAoYmluZGVyPSVwKSwgc2l6ZT0lZCwgZmQ9JWQgKCVzKSIsCi0gICAgICAgICAgICAgICAgICAgICAgICBhc0JpbmRlcigpLmdldCgpLCBzaXplLCBmZCwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgICAgICAgICBjbG9zZShmZCk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGlmIChmbGFncyAmIE1BUF9PTkNFKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vTE9HRCgicGlubmluZyBoZWFwIChiaW5kZXI9JXAsIHNpemU9JWQsIGZkPSVkIiwKLSAgICAgICAgICAgICAgICAgICAgLy8gICAgICAgIGFzQmluZGVyKCkuZ2V0KCksIHNpemUsIGZkKTsKLSAgICAgICAgICAgICAgICAgICAgcGluX2hlYXAoKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbVNpemUgPSBzaXplOwotICAgICAgICAgICAgICAgIG1GbGFncyA9IGZsYWdzOwotICAgICAgICAgICAgICAgIGFuZHJvaWRfYXRvbWljX3dyaXRlKGZkLCAmbUhlYXBJZCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLWludCBCcE1lbW9yeUhlYXA6OmdldEhlYXBJRCgpIGNvbnN0IHsKLSAgICBhc3NlcnRNYXBwZWQoKTsKLSAgICByZXR1cm4gbUhlYXBJZDsKLX0KLQotdm9pZCogQnBNZW1vcnlIZWFwOjpnZXRCYXNlKCkgY29uc3QgewotICAgIGFzc2VydE1hcHBlZCgpOwotICAgIHJldHVybiBtQmFzZTsKLX0KLQotc2l6ZV90IEJwTWVtb3J5SGVhcDo6Z2V0U2l6ZSgpIGNvbnN0IHsKLSAgICBhc3NlcnRNYXBwZWQoKTsKLSAgICByZXR1cm4gbVNpemU7Ci19Ci0KLXVpbnQzMl90IEJwTWVtb3J5SGVhcDo6Z2V0RmxhZ3MoKSBjb25zdCB7Ci0gICAgYXNzZXJ0TWFwcGVkKCk7Ci0gICAgcmV0dXJuIG1GbGFnczsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUlNUExFTUVOVF9NRVRBX0lOVEVSRkFDRShNZW1vcnlIZWFwLCAiYW5kcm9pZC51dGlscy5JTWVtb3J5SGVhcCIpOwotCi1zdGF0dXNfdCBCbk1lbW9yeUhlYXA6Om9uVHJhbnNhY3QoCi0gICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBzd2l0Y2goY29kZSkgewotICAgICAgIGNhc2UgSEVBUF9JRDogewotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElNZW1vcnlIZWFwLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVGaWxlRGVzY3JpcHRvcihnZXRIZWFwSUQoKSk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihnZXRTaXplKCkpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIoZ2V0RmxhZ3MoKSk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gQkJpbmRlcjo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotICAgIH0KLX0KLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi1IZWFwQ2FjaGU6OkhlYXBDYWNoZSgpCi0gICAgOiBEZWF0aFJlY2lwaWVudCgpCi17Ci19Ci0KLUhlYXBDYWNoZTo6fkhlYXBDYWNoZSgpCi17Ci19Ci0KLXZvaWQgSGVhcENhY2hlOjpiaW5kZXJEaWVkKGNvbnN0IHdwPElCaW5kZXI+JiBiaW5kZXIpCi17Ci0gICAgLy9MT0dEKCJiaW5kZXJEaWVkIGJpbmRlcj0lcCIsIGJpbmRlci51bnNhZmVfZ2V0KCkpOwotICAgIGZyZWVfaGVhcChiaW5kZXIpOyAKLX0KLQotc3A8SU1lbW9yeUhlYXA+IEhlYXBDYWNoZTo6ZmluZF9oZWFwKGNvbnN0IHNwPElCaW5kZXI+JiBiaW5kZXIpIAotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtSGVhcENhY2hlTG9jayk7Ci0gICAgc3NpemVfdCBpID0gbUhlYXBDYWNoZS5pbmRleE9mS2V5KGJpbmRlcik7Ci0gICAgaWYgKGk+PTApIHsKLSAgICAgICAgaGVhcF9pbmZvX3QmIGluZm8gPSBtSGVhcENhY2hlLmVkaXRWYWx1ZUF0KGkpOwotICAgICAgICBMT0dEX0lGKFZFUkJPU0UsCi0gICAgICAgICAgICAgICAgImZvdW5kIGJpbmRlcj0lcCwgaGVhcD0lcCwgc2l6ZT0lZCwgZmQ9JWQsIGNvdW50PSVkIiwgCi0gICAgICAgICAgICAgICAgYmluZGVyLmdldCgpLCBpbmZvLmhlYXAuZ2V0KCksCi0gICAgICAgICAgICAgICAgc3RhdGljX2Nhc3Q8QnBNZW1vcnlIZWFwKj4oaW5mby5oZWFwLmdldCgpKS0+bVNpemUsCi0gICAgICAgICAgICAgICAgc3RhdGljX2Nhc3Q8QnBNZW1vcnlIZWFwKj4oaW5mby5oZWFwLmdldCgpKS0+bUhlYXBJZCwKLSAgICAgICAgICAgICAgICBpbmZvLmNvdW50KTsKLSAgICAgICAgYW5kcm9pZF9hdG9taWNfaW5jKCZpbmZvLmNvdW50KTsKLSAgICAgICAgcmV0dXJuIGluZm8uaGVhcDsKLSAgICB9IGVsc2UgewotICAgICAgICBoZWFwX2luZm9fdCBpbmZvOwotICAgICAgICBpbmZvLmhlYXAgPSBpbnRlcmZhY2VfY2FzdDxJTWVtb3J5SGVhcD4oYmluZGVyKTsKLSAgICAgICAgaW5mby5jb3VudCA9IDE7Ci0gICAgICAgIC8vTE9HRCgiYWRkaW5nIGJpbmRlcj0lcCwgaGVhcD0lcCwgY291bnQ9JWQiLAotICAgICAgICAvLyAgICAgIGJpbmRlci5nZXQoKSwgaW5mby5oZWFwLmdldCgpLCBpbmZvLmNvdW50KTsKLSAgICAgICAgbUhlYXBDYWNoZS5hZGQoYmluZGVyLCBpbmZvKTsKLSAgICAgICAgcmV0dXJuIGluZm8uaGVhcDsKLSAgICB9Ci19Ci0KLXZvaWQgSGVhcENhY2hlOjpwaW5faGVhcChjb25zdCBzcDxJQmluZGVyPiYgYmluZGVyKSAKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUhlYXBDYWNoZUxvY2spOwotICAgIHNzaXplX3QgaSA9IG1IZWFwQ2FjaGUuaW5kZXhPZktleShiaW5kZXIpOwotICAgIGlmIChpPj0wKSB7Ci0gICAgICAgIGhlYXBfaW5mb190JiBpbmZvKG1IZWFwQ2FjaGUuZWRpdFZhbHVlQXQoaSkpOwotICAgICAgICBhbmRyb2lkX2F0b21pY19pbmMoJmluZm8uY291bnQpOwotICAgICAgICBiaW5kZXItPmxpbmtUb0RlYXRoKHRoaXMpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIExPR0UoInBpbl9oZWFwIGJpbmRlcj0lcCBub3QgZm91bmQhISEiLCBiaW5kZXIuZ2V0KCkpOwotICAgIH0gICAgCi19Ci0KLXZvaWQgSGVhcENhY2hlOjpmcmVlX2hlYXAoY29uc3Qgc3A8SUJpbmRlcj4mIGJpbmRlcikgIHsKLSAgICBmcmVlX2hlYXAoIHdwPElCaW5kZXI+KGJpbmRlcikgKTsKLX0KLQotdm9pZCBIZWFwQ2FjaGU6OmZyZWVfaGVhcChjb25zdCB3cDxJQmluZGVyPiYgYmluZGVyKSAKLXsKLSAgICBzcDxJTWVtb3J5SGVhcD4gcmVsOwotICAgIHsKLSAgICAgICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1IZWFwQ2FjaGVMb2NrKTsKLSAgICAgICAgc3NpemVfdCBpID0gbUhlYXBDYWNoZS5pbmRleE9mS2V5KGJpbmRlcik7Ci0gICAgICAgIGlmIChpPj0wKSB7Ci0gICAgICAgICAgICBoZWFwX2luZm9fdCYgaW5mbyhtSGVhcENhY2hlLmVkaXRWYWx1ZUF0KGkpKTsKLSAgICAgICAgICAgIGludDMyX3QgYyA9IGFuZHJvaWRfYXRvbWljX2RlYygmaW5mby5jb3VudCk7Ci0gICAgICAgICAgICBpZiAoYyA9PSAxKSB7Ci0gICAgICAgICAgICAgICAgTE9HRF9JRihWRVJCT1NFLAotICAgICAgICAgICAgICAgICAgICAgICAgInJlbW92aW5nIGJpbmRlcj0lcCwgaGVhcD0lcCwgc2l6ZT0lZCwgZmQ9JWQsIGNvdW50PSVkIiwgCi0gICAgICAgICAgICAgICAgICAgICAgICBiaW5kZXIudW5zYWZlX2dldCgpLCBpbmZvLmhlYXAuZ2V0KCksCi0gICAgICAgICAgICAgICAgICAgICAgICBzdGF0aWNfY2FzdDxCcE1lbW9yeUhlYXAqPihpbmZvLmhlYXAuZ2V0KCkpLT5tU2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHN0YXRpY19jYXN0PEJwTWVtb3J5SGVhcCo+KGluZm8uaGVhcC5nZXQoKSktPm1IZWFwSWQsCi0gICAgICAgICAgICAgICAgICAgICAgICBpbmZvLmNvdW50KTsKLSAgICAgICAgICAgICAgICByZWwgPSBtSGVhcENhY2hlLnZhbHVlQXQoaSkuaGVhcDsKLSAgICAgICAgICAgICAgICBtSGVhcENhY2hlLnJlbW92ZUl0ZW1zQXQoaSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBMT0dFKCJmcmVlX2hlYXAgYmluZGVyPSVwIG5vdCBmb3VuZCEhISIsIGJpbmRlci51bnNhZmVfZ2V0KCkpOwotICAgICAgICB9Ci0gICAgfQotfQotCi1zcDxJTWVtb3J5SGVhcD4gSGVhcENhY2hlOjpnZXRfaGVhcChjb25zdCBzcDxJQmluZGVyPiYgYmluZGVyKQotewotICAgIHNwPElNZW1vcnlIZWFwPiByZWFsSGVhcDsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUhlYXBDYWNoZUxvY2spOwotICAgIHNzaXplX3QgaSA9IG1IZWFwQ2FjaGUuaW5kZXhPZktleShiaW5kZXIpOwotICAgIGlmIChpPj0wKSAgIHJlYWxIZWFwID0gbUhlYXBDYWNoZS52YWx1ZUF0KGkpLmhlYXA7Ci0gICAgZWxzZSAgICAgICAgcmVhbEhlYXAgPSBpbnRlcmZhY2VfY2FzdDxJTWVtb3J5SGVhcD4oYmluZGVyKTsKLSAgICByZXR1cm4gcmVhbEhlYXA7Ci19Ci0KLXZvaWQgSGVhcENhY2hlOjpkdW1wX2hlYXBzKCkgCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1IZWFwQ2FjaGVMb2NrKTsKLSAgICBpbnQgYyA9IG1IZWFwQ2FjaGUuc2l6ZSgpOwotICAgIGZvciAoaW50IGk9MCA7IGk8YyA7IGkrKykgewotICAgICAgICBjb25zdCBoZWFwX2luZm9fdCYgaW5mbyA9IG1IZWFwQ2FjaGUudmFsdWVBdChpKTsKLSAgICAgICAgQnBNZW1vcnlIZWFwIGNvbnN0KiBoKHN0YXRpY19jYXN0PEJwTWVtb3J5SGVhcCBjb25zdCAqPihpbmZvLmhlYXAuZ2V0KCkpKTsKLSAgICAgICAgTE9HRCgiaGV5PSVwLCBoZWFwPSVwLCBjb3VudD0lZCwgKGZkPSVkLCBiYXNlPSVwLCBzaXplPSVkKSIsCi0gICAgICAgICAgICAgICAgbUhlYXBDYWNoZS5rZXlBdChpKS51bnNhZmVfZ2V0KCksCi0gICAgICAgICAgICAgICAgaW5mby5oZWFwLmdldCgpLCBpbmZvLmNvdW50LCAKLSAgICAgICAgICAgICAgICBoLT5tSGVhcElkLCBoLT5tQmFzZSwgaC0+bVNpemUpOwotICAgIH0KLX0KLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL0lQQ1RocmVhZFN0YXRlLmNwcCBiL2xpYnMvdXRpbHMvSVBDVGhyZWFkU3RhdGUuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwNGFlMTQyLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvSVBDVGhyZWFkU3RhdGUuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTAzMCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2luY2x1ZGUgPHV0aWxzL0lQQ1RocmVhZFN0YXRlLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9CaW5kZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9CcEJpbmRlci5oPgotI2luY2x1ZGUgPHV0aWxzL0RlYnVnLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvVGV4dE91dHB1dC5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLQotI2luY2x1ZGUgPHByaXZhdGUvdXRpbHMvYmluZGVyX21vZHVsZS5oPgotI2luY2x1ZGUgPHByaXZhdGUvdXRpbHMvU3RhdGljLmg+Ci0KLSNpbmNsdWRlIDxzeXMvaW9jdGwuaD4KLSNpbmNsdWRlIDxzaWduYWwuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0KLSNpZmRlZiBIQVZFX1BUSFJFQURTCi0jaW5jbHVkZSA8cHRocmVhZC5oPgotI2luY2x1ZGUgPHNjaGVkLmg+Ci0jaW5jbHVkZSA8c3lzL3Jlc291cmNlLmg+Ci0jZW5kaWYKLSNpZmRlZiBIQVZFX1dJTjMyX1RIUkVBRFMKLSNpbmNsdWRlIDx3aW5kb3dzLmg+Ci0jZW5kaWYKLQotCi0jaWYgTE9HX05ERUJVRwotCi0jZGVmaW5lIElGX0xPR19UUkFOU0FDVElPTlMoKSBpZiAoZmFsc2UpCi0jZGVmaW5lIElGX0xPR19DT01NQU5EUygpIGlmIChmYWxzZSkKLSNkZWZpbmUgTE9HX1JFTU9URVJFRlMoLi4uKSAKLSNkZWZpbmUgSUZfTE9HX1JFTU9URVJFRlMoKSBpZiAoZmFsc2UpCi0jZGVmaW5lIExPR19USFJFQURQT09MKC4uLikgCi0jZGVmaW5lIExPR19PTkVXQVkoLi4uKSAKLQotI2Vsc2UKLQotI2RlZmluZSBJRl9MT0dfVFJBTlNBQ1RJT05TKCkgSUZfTE9HKExPR19WRVJCT1NFLCAidHJhbnNhY3QiKQotI2RlZmluZSBJRl9MT0dfQ09NTUFORFMoKSBJRl9MT0coTE9HX1ZFUkJPU0UsICJpcGMiKQotI2RlZmluZSBMT0dfUkVNT1RFUkVGUyguLi4pIExPRyhMT0dfREVCVUcsICJyZW1vdGVyZWZzIiwgX19WQV9BUkdTX18pCi0jZGVmaW5lIElGX0xPR19SRU1PVEVSRUZTKCkgSUZfTE9HKExPR19ERUJVRywgInJlbW90ZXJlZnMiKQotI2RlZmluZSBMT0dfVEhSRUFEUE9PTCguLi4pIExPRyhMT0dfREVCVUcsICJ0aHJlYWRwb29sIiwgX19WQV9BUkdTX18pCi0jZGVmaW5lIExPR19PTkVXQVkoLi4uKSBMT0coTE9HX0RFQlVHLCAiaXBjIiwgX19WQV9BUkdTX18pCi0KLSNlbmRpZgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1zdGF0aWMgY29uc3QgY2hhciogZ2V0UmV0dXJuU3RyaW5nKHNpemVfdCBpZHgpOwotc3RhdGljIGNvbnN0IGNoYXIqIGdldENvbW1hbmRTdHJpbmcoc2l6ZV90IGlkeCk7Ci1zdGF0aWMgY29uc3Qgdm9pZCogcHJpbnRSZXR1cm5Db21tYW5kKFRleHRPdXRwdXQmIG91dCwgY29uc3Qgdm9pZCogX2NtZCk7Ci1zdGF0aWMgY29uc3Qgdm9pZCogcHJpbnRDb21tYW5kKFRleHRPdXRwdXQmIG91dCwgY29uc3Qgdm9pZCogX2NtZCk7Ci0KLS8vIFRoaXMgd2lsbCByZXN1bHQgaW4gYSBtaXNzaW5nIHN5bWJvbCBmYWlsdXJlIGlmIHRoZSBJRl9MT0dfQ09NTUFORFMoKQotLy8gY29uZGl0aW9uYWxzIGRvbid0IGdldCBzdHJpcHBlZC4uLiAgYnV0IHRoYXQgaXMgcHJvYmFibHkgd2hhdCB3ZSB3YW50LgotI2lmICFMT0dfTkRFQlVHCi1zdGF0aWMgY29uc3QgY2hhciAqa1JldHVyblN0cmluZ3NbXSA9IHsKLSNpZiAxIC8qIFRPRE86IGVycm9yIHVwZGF0ZSBzdHJpbmdzICovCi0gICAgInVua25vd24iLAotI2Vsc2UKLSAgICAiQlJfT0siLAotICAgICJCUl9USU1FT1VUIiwKLSAgICAiQlJfV0FLRVVQIiwKLSAgICAiQlJfVFJBTlNBQ1RJT04iLAotICAgICJCUl9SRVBMWSIsCi0gICAgIkJSX0FDUVVJUkVfUkVTVUxUIiwKLSAgICAiQlJfREVBRF9SRVBMWSIsCi0gICAgIkJSX1RSQU5TQUNUSU9OX0NPTVBMRVRFIiwKLSAgICAiQlJfSU5DUkVGUyIsCi0gICAgIkJSX0FDUVVJUkUiLAotICAgICJCUl9SRUxFQVNFIiwKLSAgICAiQlJfREVDUkVGUyIsCi0gICAgIkJSX0FUVEVNUFRfQUNRVUlSRSIsCi0gICAgIkJSX0VWRU5UX09DQ1VSUkVEIiwKLSAgICAiQlJfTk9PUCIsCi0gICAgIkJSX1NQQVdOX0xPT1BFUiIsCi0gICAgIkJSX0ZJTklTSEVEIiwKLSAgICAiQlJfREVBRF9CSU5ERVIiLAotICAgICJCUl9DTEVBUl9ERUFUSF9OT1RJRklDQVRJT05fRE9ORSIKLSNlbmRpZgotfTsKLQotc3RhdGljIGNvbnN0IGNoYXIgKmtDb21tYW5kU3RyaW5nc1tdID0gewotI2lmIDEgLyogVE9ETzogZXJyb3IgdXBkYXRlIHN0cmluZ3MgKi8KLSAgICAidW5rbm93biIsCi0jZWxzZQotICAgICJCQ19OT09QIiwKLSAgICAiQkNfVFJBTlNBQ1RJT04iLAotICAgICJCQ19SRVBMWSIsCi0gICAgIkJDX0FDUVVJUkVfUkVTVUxUIiwKLSAgICAiQkNfRlJFRV9CVUZGRVIiLAotICAgICJCQ19UUkFOU0FDVElPTl9DT01QTEVURSIsCi0gICAgIkJDX0lOQ1JFRlMiLAotICAgICJCQ19BQ1FVSVJFIiwKLSAgICAiQkNfUkVMRUFTRSIsCi0gICAgIkJDX0RFQ1JFRlMiLAotICAgICJCQ19JTkNSRUZTX0RPTkUiLAotICAgICJCQ19BQ1FVSVJFX0RPTkUiLAotICAgICJCQ19BVFRFTVBUX0FDUVVJUkUiLAotICAgICJCQ19SRVRSSUVWRV9ST09UX09CSkVDVCIsCi0gICAgIkJDX1NFVF9USFJFQURfRU5UUlkiLAotICAgICJCQ19SRUdJU1RFUl9MT09QRVIiLAotICAgICJCQ19FTlRFUl9MT09QRVIiLAotICAgICJCQ19FWElUX0xPT1BFUiIsCi0gICAgIkJDX1NZTkMiLAotICAgICJCQ19TVE9QX1BST0NFU1MiLAotICAgICJCQ19TVE9QX1NFTEYiLAotICAgICJCQ19SRVFVRVNUX0RFQVRIX05PVElGSUNBVElPTiIsCi0gICAgIkJDX0NMRUFSX0RFQVRIX05PVElGSUNBVElPTiIsCi0gICAgIkJDX0RFQURfQklOREVSX0RPTkUiCi0jZW5kaWYKLX07Ci0KLXN0YXRpYyBjb25zdCBjaGFyKiBnZXRSZXR1cm5TdHJpbmcoc2l6ZV90IGlkeCkKLXsKLSAgICBpZiAoaWR4IDwgc2l6ZW9mKGtSZXR1cm5TdHJpbmdzKSAvIHNpemVvZihrUmV0dXJuU3RyaW5nc1swXSkpCi0gICAgICAgIHJldHVybiBrUmV0dXJuU3RyaW5nc1tpZHhdOwotICAgIGVsc2UKLSAgICAgICAgcmV0dXJuICJ1bmtub3duIjsKLX0KLQotc3RhdGljIGNvbnN0IGNoYXIqIGdldENvbW1hbmRTdHJpbmcoc2l6ZV90IGlkeCkKLXsKLSAgICBpZiAoaWR4IDwgc2l6ZW9mKGtDb21tYW5kU3RyaW5ncykgLyBzaXplb2Yoa0NvbW1hbmRTdHJpbmdzWzBdKSkKLSAgICAgICAgcmV0dXJuIGtDb21tYW5kU3RyaW5nc1tpZHhdOwotICAgIGVsc2UKLSAgICAgICAgcmV0dXJuICJ1bmtub3duIjsKLX0KLQotc3RhdGljIGNvbnN0IHZvaWQqIHByaW50QmluZGVyVHJhbnNhY3Rpb25EYXRhKFRleHRPdXRwdXQmIG91dCwgY29uc3Qgdm9pZCogZGF0YSkKLXsKLSAgICBjb25zdCBiaW5kZXJfdHJhbnNhY3Rpb25fZGF0YSogYnRkID0KLSAgICAgICAgKGNvbnN0IGJpbmRlcl90cmFuc2FjdGlvbl9kYXRhKilkYXRhOwotICAgIG91dCA8PCAidGFyZ2V0PSIgPDwgYnRkLT50YXJnZXQucHRyIDw8ICIgKGNvb2tpZSAiIDw8IGJ0ZC0+Y29va2llIDw8ICIpIiA8PCBlbmRsCi0gICAgICAgIDw8ICJjb2RlPSIgPDwgVHlwZUNvZGUoYnRkLT5jb2RlKSA8PCAiLCBmbGFncz0iIDw8ICh2b2lkKilidGQtPmZsYWdzIDw8IGVuZGwKLSAgICAgICAgPDwgImRhdGE9IiA8PCBidGQtPmRhdGEucHRyLmJ1ZmZlciA8PCAiICgiIDw8ICh2b2lkKilidGQtPmRhdGFfc2l6ZQotICAgICAgICA8PCAiIGJ5dGVzKSIgPDwgZW5kbAotICAgICAgICA8PCAib2Zmc2V0cz0iIDw8IGJ0ZC0+ZGF0YS5wdHIub2Zmc2V0cyA8PCAiICgiIDw8ICh2b2lkKilidGQtPm9mZnNldHNfc2l6ZQotICAgICAgICA8PCAiIGJ5dGVzKSIgPDwgZW5kbDsKLSAgICByZXR1cm4gYnRkKzE7Ci19Ci0KLXN0YXRpYyBjb25zdCB2b2lkKiBwcmludFJldHVybkNvbW1hbmQoVGV4dE91dHB1dCYgb3V0LCBjb25zdCB2b2lkKiBfY21kKQotewotICAgIHN0YXRpYyBjb25zdCBpbnQzMl90IE4gPSBzaXplb2Yoa1JldHVyblN0cmluZ3MpL3NpemVvZihrUmV0dXJuU3RyaW5nc1swXSk7Ci0gICAgCi0gICAgY29uc3QgaW50MzJfdCogY21kID0gKGNvbnN0IGludDMyX3QqKV9jbWQ7Ci0gICAgaW50MzJfdCBjb2RlID0gKmNtZCsrOwotICAgIGlmIChjb2RlID09IEJSX0VSUk9SKSB7Ci0gICAgICAgIG91dCA8PCAiQlJfRVJST1I6ICIgPDwgKHZvaWQqKSgqY21kKyspIDw8IGVuZGw7Ci0gICAgICAgIHJldHVybiBjbWQ7Ci0gICAgfSBlbHNlIGlmIChjb2RlIDwgMCB8fCBjb2RlID49IE4pIHsKLSAgICAgICAgb3V0IDw8ICJVbmtub3duIHJlcGx5OiAiIDw8IGNvZGUgPDwgZW5kbDsKLSAgICAgICAgcmV0dXJuIGNtZDsKLSAgICB9Ci0gICAgCi0gICAgb3V0IDw8IGtSZXR1cm5TdHJpbmdzW2NvZGVdOwotICAgIHN3aXRjaCAoY29kZSkgewotICAgICAgICBjYXNlIEJSX1RSQU5TQUNUSU9OOgotICAgICAgICBjYXNlIEJSX1JFUExZOiB7Ci0gICAgICAgICAgICBvdXQgPDwgIjogIiA8PCBpbmRlbnQ7Ci0gICAgICAgICAgICBjbWQgPSAoY29uc3QgaW50MzJfdCAqKXByaW50QmluZGVyVHJhbnNhY3Rpb25EYXRhKG91dCwgY21kKTsKLSAgICAgICAgICAgIG91dCA8PCBkZWRlbnQ7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIAotICAgICAgICBjYXNlIEJSX0FDUVVJUkVfUkVTVUxUOiB7Ci0gICAgICAgICAgICBjb25zdCBpbnQzMl90IHJlcyA9ICpjbWQrKzsKLSAgICAgICAgICAgIG91dCA8PCAiOiAiIDw8IHJlcyA8PCAocmVzID8gIiAoU1VDQ0VTUykiIDogIiAoRkFJTFVSRSkiKTsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgCi0gICAgICAgIGNhc2UgQlJfSU5DUkVGUzoKLSAgICAgICAgY2FzZSBCUl9BQ1FVSVJFOgotICAgICAgICBjYXNlIEJSX1JFTEVBU0U6Ci0gICAgICAgIGNhc2UgQlJfREVDUkVGUzogewotICAgICAgICAgICAgY29uc3QgaW50MzJfdCBiID0gKmNtZCsrOwotICAgICAgICAgICAgY29uc3QgaW50MzJfdCBjID0gKmNtZCsrOwotICAgICAgICAgICAgb3V0IDw8ICI6IHRhcmdldD0iIDw8ICh2b2lkKiliIDw8ICIgKGNvb2tpZSAiIDw8ICh2b2lkKiljIDw8ICIpIjsKLSAgICAgICAgfSBicmVhazsKLSAgICAKLSAgICAgICAgY2FzZSBCUl9BVFRFTVBUX0FDUVVJUkU6IHsKLSAgICAgICAgICAgIGNvbnN0IGludDMyX3QgcCA9ICpjbWQrKzsKLSAgICAgICAgICAgIGNvbnN0IGludDMyX3QgYiA9ICpjbWQrKzsKLSAgICAgICAgICAgIGNvbnN0IGludDMyX3QgYyA9ICpjbWQrKzsKLSAgICAgICAgICAgIG91dCA8PCAiOiB0YXJnZXQ9IiA8PCAodm9pZCopYiA8PCAiIChjb29raWUgIiA8PCAodm9pZCopYwotICAgICAgICAgICAgICAgIDw8ICIpLCBwcmk9IiA8PCBwOwotICAgICAgICB9IGJyZWFrOwotCi0gICAgICAgIGNhc2UgQlJfREVBRF9CSU5ERVI6Ci0gICAgICAgIGNhc2UgQlJfQ0xFQVJfREVBVEhfTk9USUZJQ0FUSU9OX0RPTkU6IHsKLSAgICAgICAgICAgIGNvbnN0IGludDMyX3QgYyA9ICpjbWQrKzsKLSAgICAgICAgICAgIG91dCA8PCAiOiBkZWF0aCBjb29raWUgIiA8PCAodm9pZCopYzsKLSAgICAgICAgfSBicmVhazsKLSAgICB9Ci0gICAgCi0gICAgb3V0IDw8IGVuZGw7Ci0gICAgcmV0dXJuIGNtZDsKLX0KLQotc3RhdGljIGNvbnN0IHZvaWQqIHByaW50Q29tbWFuZChUZXh0T3V0cHV0JiBvdXQsIGNvbnN0IHZvaWQqIF9jbWQpCi17Ci0gICAgc3RhdGljIGNvbnN0IGludDMyX3QgTiA9IHNpemVvZihrQ29tbWFuZFN0cmluZ3MpL3NpemVvZihrQ29tbWFuZFN0cmluZ3NbMF0pOwotICAgIAotICAgIGNvbnN0IGludDMyX3QqIGNtZCA9IChjb25zdCBpbnQzMl90KilfY21kOwotICAgIGludDMyX3QgY29kZSA9ICpjbWQrKzsKLSAgICBpZiAoY29kZSA8IDAgfHwgY29kZSA+PSBOKSB7Ci0gICAgICAgIG91dCA8PCAiVW5rbm93biBjb21tYW5kOiAiIDw8IGNvZGUgPDwgZW5kbDsKLSAgICAgICAgcmV0dXJuIGNtZDsKLSAgICB9Ci0gICAgCi0gICAgb3V0IDw8IGtDb21tYW5kU3RyaW5nc1tjb2RlXTsKLSAgICBzd2l0Y2ggKGNvZGUpIHsKLSAgICAgICAgY2FzZSBCQ19UUkFOU0FDVElPTjoKLSAgICAgICAgY2FzZSBCQ19SRVBMWTogewotICAgICAgICAgICAgb3V0IDw8ICI6ICIgPDwgaW5kZW50OwotICAgICAgICAgICAgY21kID0gKGNvbnN0IGludDMyX3QgKilwcmludEJpbmRlclRyYW5zYWN0aW9uRGF0YShvdXQsIGNtZCk7Ci0gICAgICAgICAgICBvdXQgPDwgZGVkZW50OwotICAgICAgICB9IGJyZWFrOwotICAgICAgICAKLSAgICAgICAgY2FzZSBCQ19BQ1FVSVJFX1JFU1VMVDogewotICAgICAgICAgICAgY29uc3QgaW50MzJfdCByZXMgPSAqY21kKys7Ci0gICAgICAgICAgICBvdXQgPDwgIjogIiA8PCByZXMgPDwgKHJlcyA/ICIgKFNVQ0NFU1MpIiA6ICIgKEZBSUxVUkUpIik7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIAotICAgICAgICBjYXNlIEJDX0ZSRUVfQlVGRkVSOiB7Ci0gICAgICAgICAgICBjb25zdCBpbnQzMl90IGJ1ZiA9ICpjbWQrKzsKLSAgICAgICAgICAgIG91dCA8PCAiOiBidWZmZXI9IiA8PCAodm9pZCopYnVmOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICAKLSAgICAgICAgY2FzZSBCQ19JTkNSRUZTOgotICAgICAgICBjYXNlIEJDX0FDUVVJUkU6Ci0gICAgICAgIGNhc2UgQkNfUkVMRUFTRToKLSAgICAgICAgY2FzZSBCQ19ERUNSRUZTOiB7Ci0gICAgICAgICAgICBjb25zdCBpbnQzMl90IGQgPSAqY21kKys7Ci0gICAgICAgICAgICBvdXQgPDwgIjogZGVzY3JpcHRvcj0iIDw8ICh2b2lkKilkOwotICAgICAgICB9IGJyZWFrOwotICAgIAotICAgICAgICBjYXNlIEJDX0lOQ1JFRlNfRE9ORToKLSAgICAgICAgY2FzZSBCQ19BQ1FVSVJFX0RPTkU6IHsKLSAgICAgICAgICAgIGNvbnN0IGludDMyX3QgYiA9ICpjbWQrKzsKLSAgICAgICAgICAgIGNvbnN0IGludDMyX3QgYyA9ICpjbWQrKzsKLSAgICAgICAgICAgIG91dCA8PCAiOiB0YXJnZXQ9IiA8PCAodm9pZCopYiA8PCAiIChjb29raWUgIiA8PCAodm9pZCopYyA8PCAiKSI7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIAotICAgICAgICBjYXNlIEJDX0FUVEVNUFRfQUNRVUlSRTogewotICAgICAgICAgICAgY29uc3QgaW50MzJfdCBwID0gKmNtZCsrOwotICAgICAgICAgICAgY29uc3QgaW50MzJfdCBkID0gKmNtZCsrOwotICAgICAgICAgICAgb3V0IDw8ICI6IGRlY3JpcHRvcj0iIDw8ICh2b2lkKilkIDw8ICIsIHByaT0iIDw8IHA7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIAotICAgICAgICBjYXNlIEJDX1JFUVVFU1RfREVBVEhfTk9USUZJQ0FUSU9OOgotICAgICAgICBjYXNlIEJDX0NMRUFSX0RFQVRIX05PVElGSUNBVElPTjogewotICAgICAgICAgICAgY29uc3QgaW50MzJfdCBoID0gKmNtZCsrOwotICAgICAgICAgICAgY29uc3QgaW50MzJfdCBjID0gKmNtZCsrOwotICAgICAgICAgICAgb3V0IDw8ICI6IGhhbmRsZT0iIDw8IGggPDwgIiAoZGVhdGggY29va2llICIgPDwgKHZvaWQqKWMgPDwgIikiOwotICAgICAgICB9IGJyZWFrOwotCi0gICAgICAgIGNhc2UgQkNfREVBRF9CSU5ERVJfRE9ORTogewotICAgICAgICAgICAgY29uc3QgaW50MzJfdCBjID0gKmNtZCsrOwotICAgICAgICAgICAgb3V0IDw8ICI6IGRlYXRoIGNvb2tpZSAiIDw8ICh2b2lkKiljOwotICAgICAgICB9IGJyZWFrOwotICAgIH0KLSAgICAKLSAgICBvdXQgPDwgZW5kbDsKLSAgICByZXR1cm4gY21kOwotfQotI2VuZGlmCi0KLXN0YXRpYyBwdGhyZWFkX211dGV4X3QgZ1RMU011dGV4ID0gUFRIUkVBRF9NVVRFWF9JTklUSUFMSVpFUjsKLXN0YXRpYyBib29sIGdIYXZlVExTID0gZmFsc2U7Ci1zdGF0aWMgcHRocmVhZF9rZXlfdCBnVExTID0gMDsKLXN0YXRpYyBib29sIGdTaHV0ZG93biA9IGZhbHNlOwotCi1JUENUaHJlYWRTdGF0ZSogSVBDVGhyZWFkU3RhdGU6OnNlbGYoKQotewotICAgIGlmIChnSGF2ZVRMUykgewotcmVzdGFydDoKLSAgICAgICAgY29uc3QgcHRocmVhZF9rZXlfdCBrID0gZ1RMUzsKLSAgICAgICAgSVBDVGhyZWFkU3RhdGUqIHN0ID0gKElQQ1RocmVhZFN0YXRlKilwdGhyZWFkX2dldHNwZWNpZmljKGspOwotICAgICAgICBpZiAoc3QpIHJldHVybiBzdDsKLSAgICAgICAgcmV0dXJuIG5ldyBJUENUaHJlYWRTdGF0ZTsKLSAgICB9Ci0gICAgCi0gICAgaWYgKGdTaHV0ZG93bikgcmV0dXJuIE5VTEw7Ci0gICAgCi0gICAgcHRocmVhZF9tdXRleF9sb2NrKCZnVExTTXV0ZXgpOwotICAgIGlmICghZ0hhdmVUTFMpIHsKLSAgICAgICAgaWYgKHB0aHJlYWRfa2V5X2NyZWF0ZSgmZ1RMUywgdGhyZWFkRGVzdHJ1Y3RvcikgIT0gMCkgewotICAgICAgICAgICAgcHRocmVhZF9tdXRleF91bmxvY2soJmdUTFNNdXRleCk7Ci0gICAgICAgICAgICByZXR1cm4gTlVMTDsKLSAgICAgICAgfQotICAgICAgICBnSGF2ZVRMUyA9IHRydWU7Ci0gICAgfQotICAgIHB0aHJlYWRfbXV0ZXhfdW5sb2NrKCZnVExTTXV0ZXgpOwotICAgIGdvdG8gcmVzdGFydDsKLX0KLQotdm9pZCBJUENUaHJlYWRTdGF0ZTo6c2h1dGRvd24oKQotewotICAgIGdTaHV0ZG93biA9IHRydWU7Ci0gICAgCi0gICAgaWYgKGdIYXZlVExTKSB7Ci0gICAgICAgIC8vIFhYWCBOZWVkIHRvIHdhaXQgZm9yIGFsbCB0aHJlYWQgcG9vbCB0aHJlYWRzIHRvIGV4aXQhCi0gICAgICAgIElQQ1RocmVhZFN0YXRlKiBzdCA9IChJUENUaHJlYWRTdGF0ZSopcHRocmVhZF9nZXRzcGVjaWZpYyhnVExTKTsKLSAgICAgICAgaWYgKHN0KSB7Ci0gICAgICAgICAgICBkZWxldGUgc3Q7Ci0gICAgICAgICAgICBwdGhyZWFkX3NldHNwZWNpZmljKGdUTFMsIE5VTEwpOwotICAgICAgICB9Ci0gICAgICAgIGdIYXZlVExTID0gZmFsc2U7Ci0gICAgfQotfQotCi1zcDxQcm9jZXNzU3RhdGU+IElQQ1RocmVhZFN0YXRlOjpwcm9jZXNzKCkKLXsKLSAgICByZXR1cm4gbVByb2Nlc3M7Ci19Ci0KLXN0YXR1c190IElQQ1RocmVhZFN0YXRlOjpjbGVhckxhc3RFcnJvcigpCi17Ci0gICAgY29uc3Qgc3RhdHVzX3QgZXJyID0gbUxhc3RFcnJvcjsKLSAgICBtTGFzdEVycm9yID0gTk9fRVJST1I7Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotaW50IElQQ1RocmVhZFN0YXRlOjpnZXRDYWxsaW5nUGlkKCkKLXsKLSAgICByZXR1cm4gbUNhbGxpbmdQaWQ7Ci19Ci0KLWludCBJUENUaHJlYWRTdGF0ZTo6Z2V0Q2FsbGluZ1VpZCgpCi17Ci0gICAgcmV0dXJuIG1DYWxsaW5nVWlkOwotfQotCi1pbnQ2NF90IElQQ1RocmVhZFN0YXRlOjpjbGVhckNhbGxpbmdJZGVudGl0eSgpCi17Ci0gICAgaW50NjRfdCB0b2tlbiA9ICgoaW50NjRfdCltQ2FsbGluZ1VpZDw8MzIpIHwgbUNhbGxpbmdQaWQ7Ci0gICAgY2xlYXJDYWxsZXIoKTsKLSAgICByZXR1cm4gdG9rZW47Ci19Ci0KLXZvaWQgSVBDVGhyZWFkU3RhdGU6OnJlc3RvcmVDYWxsaW5nSWRlbnRpdHkoaW50NjRfdCB0b2tlbikKLXsKLSAgICBtQ2FsbGluZ1VpZCA9IChpbnQpKHRva2VuPj4zMik7Ci0gICAgbUNhbGxpbmdQaWQgPSAoaW50KXRva2VuOwotfQotCi12b2lkIElQQ1RocmVhZFN0YXRlOjpjbGVhckNhbGxlcigpCi17Ci0gICAgaWYgKG1Qcm9jZXNzLT5zdXBwb3J0c1Byb2Nlc3NlcygpKSB7Ci0gICAgICAgIG1DYWxsaW5nUGlkID0gZ2V0cGlkKCk7Ci0gICAgICAgIG1DYWxsaW5nVWlkID0gZ2V0dWlkKCk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgbUNhbGxpbmdQaWQgPSAtMTsKLSAgICAgICAgbUNhbGxpbmdVaWQgPSAtMTsKLSAgICB9Ci19Ci0KLXZvaWQgSVBDVGhyZWFkU3RhdGU6OmZsdXNoQ29tbWFuZHMoKQotewotICAgIGlmIChtUHJvY2Vzcy0+bURyaXZlckZEIDw9IDApCi0gICAgICAgIHJldHVybjsKLSAgICB0YWxrV2l0aERyaXZlcihmYWxzZSk7Ci19Ci0KLXZvaWQgSVBDVGhyZWFkU3RhdGU6OmpvaW5UaHJlYWRQb29sKGJvb2wgaXNNYWluKQotewotICAgIExPR19USFJFQURQT09MKCIqKioqIFRIUkVBRCAlcCAoUElEICVkKSBJUyBKT0lOSU5HIFRIRSBUSFJFQUQgUE9PTFxuIiwgKHZvaWQqKXB0aHJlYWRfc2VsZigpLCBnZXRwaWQoKSk7Ci0KLSAgICBtT3V0LndyaXRlSW50MzIoaXNNYWluID8gQkNfRU5URVJfTE9PUEVSIDogQkNfUkVHSVNURVJfTE9PUEVSKTsKLSAgICAKLSAgICBzdGF0dXNfdCByZXN1bHQ7Ci0gICAgZG8gewotICAgICAgICBpbnQzMl90IGNtZDsKLSAgICAgICAgCi0gICAgICAgIC8vIFdoZW4gd2UndmUgY2xlYXJlZCB0aGUgaW5jb21pbmcgY29tbWFuZCBxdWV1ZSwgcHJvY2VzcyBhbnkgcGVuZGluZyBkZXJlZnMKLSAgICAgICAgaWYgKG1Jbi5kYXRhUG9zaXRpb24oKSA+PSBtSW4uZGF0YVNpemUoKSkgewotICAgICAgICAgICAgc2l6ZV90IG51bVBlbmRpbmcgPSBtUGVuZGluZ1dlYWtEZXJlZnMuc2l6ZSgpOwotICAgICAgICAgICAgaWYgKG51bVBlbmRpbmcgPiAwKSB7Ci0gICAgICAgICAgICAgICAgZm9yIChzaXplX3QgaSA9IDA7IGkgPCBudW1QZW5kaW5nOyBpKyspIHsKLSAgICAgICAgICAgICAgICAgICAgUmVmQmFzZTo6d2Vha3JlZl90eXBlKiByZWZzID0gbVBlbmRpbmdXZWFrRGVyZWZzW2ldOwotICAgICAgICAgICAgICAgICAgICByZWZzLT5kZWNXZWFrKG1Qcm9jZXNzLmdldCgpKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbVBlbmRpbmdXZWFrRGVyZWZzLmNsZWFyKCk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIG51bVBlbmRpbmcgPSBtUGVuZGluZ1N0cm9uZ0RlcmVmcy5zaXplKCk7Ci0gICAgICAgICAgICBpZiAobnVtUGVuZGluZyA+IDApIHsKLSAgICAgICAgICAgICAgICBmb3IgKHNpemVfdCBpID0gMDsgaSA8IG51bVBlbmRpbmc7IGkrKykgewotICAgICAgICAgICAgICAgICAgICBCQmluZGVyKiBvYmogPSBtUGVuZGluZ1N0cm9uZ0RlcmVmc1tpXTsKLSAgICAgICAgICAgICAgICAgICAgb2JqLT5kZWNTdHJvbmcobVByb2Nlc3MuZ2V0KCkpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBtUGVuZGluZ1N0cm9uZ0RlcmVmcy5jbGVhcigpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgLy8gbm93IGdldCB0aGUgbmV4dCBjb21tYW5kIHRvIGJlIHByb2Nlc3NlZCwgd2FpdGluZyBpZiBuZWNlc3NhcnkKLSAgICAgICAgcmVzdWx0ID0gdGFsa1dpdGhEcml2ZXIoKTsKLSAgICAgICAgaWYgKHJlc3VsdCA+PSBOT19FUlJPUikgewotICAgICAgICAgICAgc2l6ZV90IElOID0gbUluLmRhdGFBdmFpbCgpOwotICAgICAgICAgICAgaWYgKElOIDwgc2l6ZW9mKGludDMyX3QpKSBjb250aW51ZTsKLSAgICAgICAgICAgIGNtZCA9IG1Jbi5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgIElGX0xPR19DT01NQU5EUygpIHsKLSAgICAgICAgICAgICAgICBhbG9nIDw8ICJQcm9jZXNzaW5nIHRvcC1sZXZlbCBDb21tYW5kOiAiCi0gICAgICAgICAgICAgICAgICAgIDw8IGdldFJldHVyblN0cmluZyhjbWQpIDw8IGVuZGw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXN1bHQgPSBleGVjdXRlQ29tbWFuZChjbWQpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICAvLyBMZXQgdGhpcyB0aHJlYWQgZXhpdCB0aGUgdGhyZWFkIHBvb2wgaWYgaXQgaXMgbm8gbG9uZ2VyCi0gICAgICAgIC8vIG5lZWRlZCBhbmQgaXQgaXMgbm90IHRoZSBtYWluIHByb2Nlc3MgdGhyZWFkLgotICAgICAgICBpZihyZXN1bHQgPT0gVElNRURfT1VUICYmICFpc01haW4pIHsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgfSB3aGlsZSAocmVzdWx0ICE9IC1FQ09OTlJFRlVTRUQgJiYgcmVzdWx0ICE9IC1FQkFERik7Ci0KLSAgICBMT0dfVEhSRUFEUE9PTCgiKioqKiBUSFJFQUQgJXAgKFBJRCAlZCkgSVMgTEVBVklORyBUSEUgVEhSRUFEIFBPT0wgZXJyPSVwXG4iLAotICAgICAgICAodm9pZCopcHRocmVhZF9zZWxmKCksIGdldHBpZCgpLCAodm9pZCopcmVzdWx0KTsKLSAgICAKLSAgICBtT3V0LndyaXRlSW50MzIoQkNfRVhJVF9MT09QRVIpOwotICAgIHRhbGtXaXRoRHJpdmVyKGZhbHNlKTsKLX0KLQotdm9pZCBJUENUaHJlYWRTdGF0ZTo6c3RvcFByb2Nlc3MoYm9vbCBpbW1lZGlhdGUpCi17Ci0gICAgLy9MT0dJKCIqKioqIFNUT1BQSU5HIFBST0NFU1MiKTsKLSAgICBmbHVzaENvbW1hbmRzKCk7Ci0gICAgaW50IGZkID0gbVByb2Nlc3MtPm1Ecml2ZXJGRDsKLSAgICBtUHJvY2Vzcy0+bURyaXZlckZEID0gLTE7Ci0gICAgY2xvc2UoZmQpOwotICAgIC8va2lsbChnZXRwaWQoKSwgU0lHS0lMTCk7Ci19Ci0KLXN0YXR1c190IElQQ1RocmVhZFN0YXRlOjp0cmFuc2FjdChpbnQzMl90IGhhbmRsZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUGFyY2VsKiByZXBseSwgdWludDMyX3QgZmxhZ3MpCi17Ci0gICAgc3RhdHVzX3QgZXJyID0gZGF0YS5lcnJvckNoZWNrKCk7Ci0KLSAgICBmbGFncyB8PSBURl9BQ0NFUFRfRkRTOwotCi0gICAgSUZfTE9HX1RSQU5TQUNUSU9OUygpIHsKLSAgICAgICAgVGV4dE91dHB1dDo6QnVuZGxlIF9iKGFsb2cpOwotICAgICAgICBhbG9nIDw8ICJCQ19UUkFOU0FDVElPTiB0aHIgIiA8PCAodm9pZCopcHRocmVhZF9zZWxmKCkgPDwgIiAvIGhhbmQgIgotICAgICAgICAgICAgPDwgaGFuZGxlIDw8ICIgLyBjb2RlICIgPDwgVHlwZUNvZGUoY29kZSkgPDwgIjogIgotICAgICAgICAgICAgPDwgaW5kZW50IDw8IGRhdGEgPDwgZGVkZW50IDw8IGVuZGw7Ci0gICAgfQotICAgIAotICAgIGlmIChlcnIgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgTE9HX09ORVdBWSgiPj4+PiBTRU5EIGZyb20gcGlkICVkIHVpZCAlZCAlcyIsIGdldHBpZCgpLCBnZXR1aWQoKSwKLSAgICAgICAgICAgIChmbGFncyAmIFRGX09ORV9XQVkpID09IDAgPyAiUkVBRCBSRVBMWSIgOiAiT05FIFdBWSIpOwotICAgICAgICBlcnIgPSB3cml0ZVRyYW5zYWN0aW9uRGF0YShCQ19UUkFOU0FDVElPTiwgZmxhZ3MsIGhhbmRsZSwgY29kZSwgZGF0YSwgTlVMTCk7Ci0gICAgfQotICAgIAotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgaWYgKHJlcGx5KSByZXBseS0+c2V0RXJyb3IoZXJyKTsKLSAgICAgICAgcmV0dXJuIChtTGFzdEVycm9yID0gZXJyKTsKLSAgICB9Ci0gICAgCi0gICAgaWYgKChmbGFncyAmIFRGX09ORV9XQVkpID09IDApIHsKLSAgICAgICAgaWYgKHJlcGx5KSB7Ci0gICAgICAgICAgICBlcnIgPSB3YWl0Rm9yUmVzcG9uc2UocmVwbHkpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgUGFyY2VsIGZha2VSZXBseTsKLSAgICAgICAgICAgIGVyciA9IHdhaXRGb3JSZXNwb25zZSgmZmFrZVJlcGx5KTsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgSUZfTE9HX1RSQU5TQUNUSU9OUygpIHsKLSAgICAgICAgICAgIFRleHRPdXRwdXQ6OkJ1bmRsZSBfYihhbG9nKTsKLSAgICAgICAgICAgIGFsb2cgPDwgIkJSX1JFUExZIHRociAiIDw8ICh2b2lkKilwdGhyZWFkX3NlbGYoKSA8PCAiIC8gaGFuZCAiCi0gICAgICAgICAgICAgICAgPDwgaGFuZGxlIDw8ICI6ICI7Ci0gICAgICAgICAgICBpZiAocmVwbHkpIGFsb2cgPDwgaW5kZW50IDw8ICpyZXBseSA8PCBkZWRlbnQgPDwgZW5kbDsKLSAgICAgICAgICAgIGVsc2UgYWxvZyA8PCAiKG5vbmUgcmVxdWVzdGVkKSIgPDwgZW5kbDsKLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIGVyciA9IHdhaXRGb3JSZXNwb25zZShOVUxMLCBOVUxMKTsKLSAgICB9Ci0gICAgCi0gICAgcmV0dXJuIGVycjsKLX0KLQotdm9pZCBJUENUaHJlYWRTdGF0ZTo6aW5jU3Ryb25nSGFuZGxlKGludDMyX3QgaGFuZGxlKQotewotICAgIExPR19SRU1PVEVSRUZTKCJJUENUaHJlYWRTdGF0ZTo6aW5jU3Ryb25nSGFuZGxlKCVkKVxuIiwgaGFuZGxlKTsKLSAgICBtT3V0LndyaXRlSW50MzIoQkNfQUNRVUlSRSk7Ci0gICAgbU91dC53cml0ZUludDMyKGhhbmRsZSk7Ci19Ci0KLXZvaWQgSVBDVGhyZWFkU3RhdGU6OmRlY1N0cm9uZ0hhbmRsZShpbnQzMl90IGhhbmRsZSkKLXsKLSAgICBMT0dfUkVNT1RFUkVGUygiSVBDVGhyZWFkU3RhdGU6OmRlY1N0cm9uZ0hhbmRsZSglZClcbiIsIGhhbmRsZSk7Ci0gICAgbU91dC53cml0ZUludDMyKEJDX1JFTEVBU0UpOwotICAgIG1PdXQud3JpdGVJbnQzMihoYW5kbGUpOwotfQotCi12b2lkIElQQ1RocmVhZFN0YXRlOjppbmNXZWFrSGFuZGxlKGludDMyX3QgaGFuZGxlKQotewotICAgIExPR19SRU1PVEVSRUZTKCJJUENUaHJlYWRTdGF0ZTo6aW5jV2Vha0hhbmRsZSglZClcbiIsIGhhbmRsZSk7Ci0gICAgbU91dC53cml0ZUludDMyKEJDX0lOQ1JFRlMpOwotICAgIG1PdXQud3JpdGVJbnQzMihoYW5kbGUpOwotfQotCi12b2lkIElQQ1RocmVhZFN0YXRlOjpkZWNXZWFrSGFuZGxlKGludDMyX3QgaGFuZGxlKQotewotICAgIExPR19SRU1PVEVSRUZTKCJJUENUaHJlYWRTdGF0ZTo6ZGVjV2Vha0hhbmRsZSglZClcbiIsIGhhbmRsZSk7Ci0gICAgbU91dC53cml0ZUludDMyKEJDX0RFQ1JFRlMpOwotICAgIG1PdXQud3JpdGVJbnQzMihoYW5kbGUpOwotfQotCi1zdGF0dXNfdCBJUENUaHJlYWRTdGF0ZTo6YXR0ZW1wdEluY1N0cm9uZ0hhbmRsZShpbnQzMl90IGhhbmRsZSkKLXsKLSAgICBtT3V0LndyaXRlSW50MzIoQkNfQVRURU1QVF9BQ1FVSVJFKTsKLSAgICBtT3V0LndyaXRlSW50MzIoMCk7IC8vIHh4eCB3YXMgdGhyZWFkIHByaW9yaXR5Ci0gICAgbU91dC53cml0ZUludDMyKGhhbmRsZSk7Ci0gICAgc3RhdHVzX3QgcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLSAgICAKLSAgICB3YWl0Rm9yUmVzcG9uc2UoTlVMTCwgJnJlc3VsdCk7Ci0gICAgCi0jaWYgTE9HX1JFRkNPVU5UUwotICAgIHByaW50ZigiSVBDVGhyZWFkU3RhdGU6OmF0dGVtcHRJbmNTdHJvbmdIYW5kbGUoJWxkKSA9ICVzXG4iLAotICAgICAgICBoYW5kbGUsIHJlc3VsdCA9PSBOT19FUlJPUiA/ICJTVUNDRVNTIiA6ICJGQUlMVVJFIik7Ci0jZW5kaWYKLSAgICAKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi12b2lkIElQQ1RocmVhZFN0YXRlOjpleHB1bmdlSGFuZGxlKGludDMyX3QgaGFuZGxlLCBJQmluZGVyKiBiaW5kZXIpCi17Ci0jaWYgTE9HX1JFRkNPVU5UUwotICAgIHByaW50ZigiSVBDVGhyZWFkU3RhdGU6OmV4cHVuZ2VIYW5kbGUoJWxkKVxuIiwgaGFuZGxlKTsKLSNlbmRpZgotICAgIHNlbGYoKS0+bVByb2Nlc3MtPmV4cHVuZ2VIYW5kbGUoaGFuZGxlLCBiaW5kZXIpOwotfQotCi1zdGF0dXNfdCBJUENUaHJlYWRTdGF0ZTo6cmVxdWVzdERlYXRoTm90aWZpY2F0aW9uKGludDMyX3QgaGFuZGxlLCBCcEJpbmRlciogcHJveHkpCi17Ci0gICAgbU91dC53cml0ZUludDMyKEJDX1JFUVVFU1RfREVBVEhfTk9USUZJQ0FUSU9OKTsKLSAgICBtT3V0LndyaXRlSW50MzIoKGludDMyX3QpaGFuZGxlKTsKLSAgICBtT3V0LndyaXRlSW50MzIoKGludDMyX3QpcHJveHkpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgSVBDVGhyZWFkU3RhdGU6OmNsZWFyRGVhdGhOb3RpZmljYXRpb24oaW50MzJfdCBoYW5kbGUsIEJwQmluZGVyKiBwcm94eSkKLXsKLSAgICBtT3V0LndyaXRlSW50MzIoQkNfQ0xFQVJfREVBVEhfTk9USUZJQ0FUSU9OKTsKLSAgICBtT3V0LndyaXRlSW50MzIoKGludDMyX3QpaGFuZGxlKTsKLSAgICBtT3V0LndyaXRlSW50MzIoKGludDMyX3QpcHJveHkpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotSVBDVGhyZWFkU3RhdGU6OklQQ1RocmVhZFN0YXRlKCkKLSAgICA6IG1Qcm9jZXNzKFByb2Nlc3NTdGF0ZTo6c2VsZigpKQotewotICAgIHB0aHJlYWRfc2V0c3BlY2lmaWMoZ1RMUywgdGhpcyk7Ci0gICAgICAgIGNsZWFyQ2FsbGVyKCk7Ci0gICAgbUluLnNldERhdGFDYXBhY2l0eSgyNTYpOwotICAgIG1PdXQuc2V0RGF0YUNhcGFjaXR5KDI1Nik7Ci19Ci0KLUlQQ1RocmVhZFN0YXRlOjp+SVBDVGhyZWFkU3RhdGUoKQotewotfQotCi1zdGF0dXNfdCBJUENUaHJlYWRTdGF0ZTo6c2VuZFJlcGx5KGNvbnN0IFBhcmNlbCYgcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIHN0YXR1c190IGVycjsKLSAgICBzdGF0dXNfdCBzdGF0dXNCdWZmZXI7Ci0gICAgZXJyID0gd3JpdGVUcmFuc2FjdGlvbkRhdGEoQkNfUkVQTFksIGZsYWdzLCAtMSwgMCwgcmVwbHksICZzdGF0dXNCdWZmZXIpOwotICAgIGlmIChlcnIgPCBOT19FUlJPUikgcmV0dXJuIGVycjsKLSAgICAKLSAgICByZXR1cm4gd2FpdEZvclJlc3BvbnNlKE5VTEwsIE5VTEwpOwotfQotCi1zdGF0dXNfdCBJUENUaHJlYWRTdGF0ZTo6d2FpdEZvclJlc3BvbnNlKFBhcmNlbCAqcmVwbHksIHN0YXR1c190ICphY3F1aXJlUmVzdWx0KQotewotICAgIGludDMyX3QgY21kOwotICAgIGludDMyX3QgZXJyOwotCi0gICAgd2hpbGUgKDEpIHsKLSAgICAgICAgaWYgKChlcnI9dGFsa1dpdGhEcml2ZXIoKSkgPCBOT19FUlJPUikgYnJlYWs7Ci0gICAgICAgIGVyciA9IG1Jbi5lcnJvckNoZWNrKCk7Ci0gICAgICAgIGlmIChlcnIgPCBOT19FUlJPUikgYnJlYWs7Ci0gICAgICAgIGlmIChtSW4uZGF0YUF2YWlsKCkgPT0gMCkgY29udGludWU7Ci0gICAgICAgIAotICAgICAgICBjbWQgPSBtSW4ucmVhZEludDMyKCk7Ci0gICAgICAgIAotICAgICAgICBJRl9MT0dfQ09NTUFORFMoKSB7Ci0gICAgICAgICAgICBhbG9nIDw8ICJQcm9jZXNzaW5nIHdhaXRGb3JSZXNwb25zZSBDb21tYW5kOiAiCi0gICAgICAgICAgICAgICAgPDwgZ2V0UmV0dXJuU3RyaW5nKGNtZCkgPDwgZW5kbDsKLSAgICAgICAgfQotCi0gICAgICAgIHN3aXRjaCAoY21kKSB7Ci0gICAgICAgIGNhc2UgQlJfVFJBTlNBQ1RJT05fQ09NUExFVEU6Ci0gICAgICAgICAgICBpZiAoIXJlcGx5ICYmICFhY3F1aXJlUmVzdWx0KSBnb3RvIGZpbmlzaDsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAKLSAgICAgICAgY2FzZSBCUl9ERUFEX1JFUExZOgotICAgICAgICAgICAgZXJyID0gREVBRF9PQkpFQ1Q7Ci0gICAgICAgICAgICBnb3RvIGZpbmlzaDsKLQotICAgICAgICBjYXNlIEJSX0ZBSUxFRF9SRVBMWToKLSAgICAgICAgICAgIGVyciA9IEZBSUxFRF9UUkFOU0FDVElPTjsKLSAgICAgICAgICAgIGdvdG8gZmluaXNoOwotICAgICAgICAKLSAgICAgICAgY2FzZSBCUl9BQ1FVSVJFX1JFU1VMVDoKLSAgICAgICAgICAgIHsKLSAgICAgICAgICAgICAgICBMT0dfQVNTRVJUKGFjcXVpcmVSZXN1bHQgIT0gTlVMTCwgIlVuZXhwZWN0ZWQgYnJBQ1FVSVJFX1JFU1VMVCIpOwotICAgICAgICAgICAgICAgIGNvbnN0IGludDMyX3QgcmVzdWx0ID0gbUluLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgICAgIGlmICghYWNxdWlyZVJlc3VsdCkgY29udGludWU7Ci0gICAgICAgICAgICAgICAgKmFjcXVpcmVSZXN1bHQgPSByZXN1bHQgPyBOT19FUlJPUiA6IElOVkFMSURfT1BFUkFUSU9OOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZ290byBmaW5pc2g7Ci0gICAgICAgIAotICAgICAgICBjYXNlIEJSX1JFUExZOgotICAgICAgICAgICAgewotICAgICAgICAgICAgICAgIGJpbmRlcl90cmFuc2FjdGlvbl9kYXRhIHRyOwotICAgICAgICAgICAgICAgIGVyciA9IG1Jbi5yZWFkKCZ0ciwgc2l6ZW9mKHRyKSk7Ci0gICAgICAgICAgICAgICAgTE9HX0FTU0VSVChlcnIgPT0gTk9fRVJST1IsICJOb3QgZW5vdWdoIGNvbW1hbmQgZGF0YSBmb3IgYnJSRVBMWSIpOwotICAgICAgICAgICAgICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIGdvdG8gZmluaXNoOwotCi0gICAgICAgICAgICAgICAgaWYgKHJlcGx5KSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmICgodHIuZmxhZ3MgJiBURl9TVEFUVVNfQ09ERSkgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmVwbHktPmlwY1NldERhdGFSZWZlcmVuY2UoCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVpbnRlcnByZXRfY2FzdDxjb25zdCB1aW50OF90Kj4odHIuZGF0YS5wdHIuYnVmZmVyKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB0ci5kYXRhX3NpemUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVpbnRlcnByZXRfY2FzdDxjb25zdCBzaXplX3QqPih0ci5kYXRhLnB0ci5vZmZzZXRzKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB0ci5vZmZzZXRzX3NpemUvc2l6ZW9mKHNpemVfdCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZnJlZUJ1ZmZlciwgdGhpcyk7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBlcnIgPSAqc3RhdGljX2Nhc3Q8Y29uc3Qgc3RhdHVzX3QqPih0ci5kYXRhLnB0ci5idWZmZXIpOwotICAgICAgICAgICAgICAgICAgICAgICAgZnJlZUJ1ZmZlcihOVUxMLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgdWludDhfdCo+KHRyLmRhdGEucHRyLmJ1ZmZlciksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdHIuZGF0YV9zaXplLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3Qgc2l6ZV90Kj4odHIuZGF0YS5wdHIub2Zmc2V0cyksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdHIub2Zmc2V0c19zaXplL3NpemVvZihzaXplX3QpLCB0aGlzKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGZyZWVCdWZmZXIoTlVMTCwKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgdWludDhfdCo+KHRyLmRhdGEucHRyLmJ1ZmZlciksCi0gICAgICAgICAgICAgICAgICAgICAgICB0ci5kYXRhX3NpemUsCi0gICAgICAgICAgICAgICAgICAgICAgICByZWludGVycHJldF9jYXN0PGNvbnN0IHNpemVfdCo+KHRyLmRhdGEucHRyLm9mZnNldHMpLAotICAgICAgICAgICAgICAgICAgICAgICAgdHIub2Zmc2V0c19zaXplL3NpemVvZihzaXplX3QpLCB0aGlzKTsKLSAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgZ290byBmaW5pc2g7Ci0KLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIGVyciA9IGV4ZWN1dGVDb21tYW5kKGNtZCk7Ci0gICAgICAgICAgICBpZiAoZXJyICE9IE5PX0VSUk9SKSBnb3RvIGZpbmlzaDsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgfQotCi1maW5pc2g6Ci0gICAgaWYgKGVyciAhPSBOT19FUlJPUikgewotICAgICAgICBpZiAoYWNxdWlyZVJlc3VsdCkgKmFjcXVpcmVSZXN1bHQgPSBlcnI7Ci0gICAgICAgIGlmIChyZXBseSkgcmVwbHktPnNldEVycm9yKGVycik7Ci0gICAgICAgIG1MYXN0RXJyb3IgPSBlcnI7Ci0gICAgfQotICAgIAotICAgIHJldHVybiBlcnI7Ci19Ci0KLXN0YXR1c190IElQQ1RocmVhZFN0YXRlOjp0YWxrV2l0aERyaXZlcihib29sIGRvUmVjZWl2ZSkKLXsKLSAgICBMT0dfQVNTRVJUKG1Qcm9jZXNzLT5tRHJpdmVyRkQgPj0gMCwgIkJpbmRlciBkcml2ZXIgaXMgbm90IG9wZW5lZCIpOwotICAgIAotICAgIGJpbmRlcl93cml0ZV9yZWFkIGJ3cjsKLSAgICAKLSAgICAvLyBJcyB0aGUgcmVhZCBidWZmZXIgZW1wdHk/Ci0gICAgY29uc3QgYm9vbCBuZWVkUmVhZCA9IG1Jbi5kYXRhUG9zaXRpb24oKSA+PSBtSW4uZGF0YVNpemUoKTsKLSAgICAKLSAgICAvLyBXZSBkb24ndCB3YW50IHRvIHdyaXRlIGFueXRoaW5nIGlmIHdlIGFyZSBzdGlsbCByZWFkaW5nCi0gICAgLy8gZnJvbSBkYXRhIGxlZnQgaW4gdGhlIGlucHV0IGJ1ZmZlciBhbmQgdGhlIGNhbGxlcgotICAgIC8vIGhhcyByZXF1ZXN0ZWQgdG8gcmVhZCB0aGUgbmV4dCBkYXRhLgotICAgIGNvbnN0IHNpemVfdCBvdXRBdmFpbCA9ICghZG9SZWNlaXZlIHx8IG5lZWRSZWFkKSA/IG1PdXQuZGF0YVNpemUoKSA6IDA7Ci0gICAgCi0gICAgYndyLndyaXRlX3NpemUgPSBvdXRBdmFpbDsKLSAgICBid3Iud3JpdGVfYnVmZmVyID0gKGxvbmcgdW5zaWduZWQgaW50KW1PdXQuZGF0YSgpOwotCi0gICAgLy8gVGhpcyBpcyB3aGF0IHdlJ2xsIHJlYWQuCi0gICAgaWYgKGRvUmVjZWl2ZSAmJiBuZWVkUmVhZCkgewotICAgICAgICBid3IucmVhZF9zaXplID0gbUluLmRhdGFDYXBhY2l0eSgpOwotICAgICAgICBid3IucmVhZF9idWZmZXIgPSAobG9uZyB1bnNpZ25lZCBpbnQpbUluLmRhdGEoKTsKLSAgICB9IGVsc2UgewotICAgICAgICBid3IucmVhZF9zaXplID0gMDsKLSAgICB9Ci0gICAgCi0gICAgSUZfTE9HX0NPTU1BTkRTKCkgewotICAgICAgICBUZXh0T3V0cHV0OjpCdW5kbGUgX2IoYWxvZyk7Ci0gICAgICAgIGlmIChvdXRBdmFpbCAhPSAwKSB7Ci0gICAgICAgICAgICBhbG9nIDw8ICJTZW5kaW5nIGNvbW1hbmRzIHRvIGRyaXZlcjogIiA8PCBpbmRlbnQ7Ci0gICAgICAgICAgICBjb25zdCB2b2lkKiBjbWRzID0gKGNvbnN0IHZvaWQqKWJ3ci53cml0ZV9idWZmZXI7Ci0gICAgICAgICAgICBjb25zdCB2b2lkKiBlbmQgPSAoKGNvbnN0IHVpbnQ4X3QqKWNtZHMpK2J3ci53cml0ZV9zaXplOwotICAgICAgICAgICAgYWxvZyA8PCBIZXhEdW1wKGNtZHMsIGJ3ci53cml0ZV9zaXplKSA8PCBlbmRsOwotICAgICAgICAgICAgd2hpbGUgKGNtZHMgPCBlbmQpIGNtZHMgPSBwcmludENvbW1hbmQoYWxvZywgY21kcyk7Ci0gICAgICAgICAgICBhbG9nIDw8IGRlZGVudDsKLSAgICAgICAgfQotICAgICAgICBhbG9nIDw8ICJTaXplIG9mIHJlY2VpdmUgYnVmZmVyOiAiIDw8IGJ3ci5yZWFkX3NpemUKLSAgICAgICAgICAgIDw8ICIsIG5lZWRSZWFkOiAiIDw8IG5lZWRSZWFkIDw8ICIsIGRvUmVjZWl2ZTogIiA8PCBkb1JlY2VpdmUgPDwgZW5kbDsKLSAgICB9Ci0gICAgCi0gICAgLy8gUmV0dXJuIGltbWVkaWF0ZWx5IGlmIHRoZXJlIGlzIG5vdGhpbmcgdG8gZG8uCi0gICAgaWYgKChid3Iud3JpdGVfc2l6ZSA9PSAwKSAmJiAoYndyLnJlYWRfc2l6ZSA9PSAwKSkgcmV0dXJuIE5PX0VSUk9SOwotICAgIAotICAgIGJ3ci53cml0ZV9jb25zdW1lZCA9IDA7Ci0gICAgYndyLnJlYWRfY29uc3VtZWQgPSAwOwotICAgIHN0YXR1c190IGVycjsKLSAgICBkbyB7Ci0gICAgICAgIElGX0xPR19DT01NQU5EUygpIHsKLSAgICAgICAgICAgIGFsb2cgPDwgIkFib3V0IHRvIHJlYWQvd3JpdGUsIHdyaXRlIHNpemUgPSAiIDw8IG1PdXQuZGF0YVNpemUoKSA8PCBlbmRsOwotICAgICAgICB9Ci0jaWYgZGVmaW5lZChIQVZFX0FORFJPSURfT1MpCi0gICAgICAgIGlmIChpb2N0bChtUHJvY2Vzcy0+bURyaXZlckZELCBCSU5ERVJfV1JJVEVfUkVBRCwgJmJ3cikgPj0gMCkKLSAgICAgICAgICAgIGVyciA9IE5PX0VSUk9SOwotICAgICAgICBlbHNlCi0gICAgICAgICAgICBlcnIgPSAtZXJybm87Ci0jZWxzZQotICAgICAgICBlcnIgPSBJTlZBTElEX09QRVJBVElPTjsKLSNlbmRpZgotICAgICAgICBJRl9MT0dfQ09NTUFORFMoKSB7Ci0gICAgICAgICAgICBhbG9nIDw8ICJGaW5pc2hlZCByZWFkL3dyaXRlLCB3cml0ZSBzaXplID0gIiA8PCBtT3V0LmRhdGFTaXplKCkgPDwgZW5kbDsKLSAgICAgICAgfQotICAgIH0gd2hpbGUgKGVyciA9PSAtRUlOVFIpOwotICAgIAotICAgIElGX0xPR19DT01NQU5EUygpIHsKLSAgICAgICAgYWxvZyA8PCAiT3VyIGVycjogIiA8PCAodm9pZCopZXJyIDw8ICIsIHdyaXRlIGNvbnN1bWVkOiAiCi0gICAgICAgICAgICA8PCBid3Iud3JpdGVfY29uc3VtZWQgPDwgIiAob2YgIiA8PCBtT3V0LmRhdGFTaXplKCkKLQkJCTw8ICIpLCByZWFkIGNvbnN1bWVkOiAiIDw8IGJ3ci5yZWFkX2NvbnN1bWVkIDw8IGVuZGw7Ci0gICAgfQotCi0gICAgaWYgKGVyciA+PSBOT19FUlJPUikgewotICAgICAgICBpZiAoYndyLndyaXRlX2NvbnN1bWVkID4gMCkgewotICAgICAgICAgICAgaWYgKGJ3ci53cml0ZV9jb25zdW1lZCA8IChzc2l6ZV90KW1PdXQuZGF0YVNpemUoKSkKLSAgICAgICAgICAgICAgICBtT3V0LnJlbW92ZSgwLCBid3Iud3JpdGVfY29uc3VtZWQpOwotICAgICAgICAgICAgZWxzZQotICAgICAgICAgICAgICAgIG1PdXQuc2V0RGF0YVNpemUoMCk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGJ3ci5yZWFkX2NvbnN1bWVkID4gMCkgewotICAgICAgICAgICAgbUluLnNldERhdGFTaXplKGJ3ci5yZWFkX2NvbnN1bWVkKTsKLSAgICAgICAgICAgIG1Jbi5zZXREYXRhUG9zaXRpb24oMCk7Ci0gICAgICAgIH0KLSAgICAgICAgSUZfTE9HX0NPTU1BTkRTKCkgewotICAgICAgICAgICAgVGV4dE91dHB1dDo6QnVuZGxlIF9iKGFsb2cpOwotICAgICAgICAgICAgYWxvZyA8PCAiUmVtYWluaW5nIGRhdGEgc2l6ZTogIiA8PCBtT3V0LmRhdGFTaXplKCkgPDwgZW5kbDsKLSAgICAgICAgICAgIGFsb2cgPDwgIlJlY2VpdmVkIGNvbW1hbmRzIGZyb20gZHJpdmVyOiAiIDw8IGluZGVudDsKLSAgICAgICAgICAgIGNvbnN0IHZvaWQqIGNtZHMgPSBtSW4uZGF0YSgpOwotICAgICAgICAgICAgY29uc3Qgdm9pZCogZW5kID0gbUluLmRhdGEoKSArIG1Jbi5kYXRhU2l6ZSgpOwotICAgICAgICAgICAgYWxvZyA8PCBIZXhEdW1wKGNtZHMsIG1Jbi5kYXRhU2l6ZSgpKSA8PCBlbmRsOwotICAgICAgICAgICAgd2hpbGUgKGNtZHMgPCBlbmQpIGNtZHMgPSBwcmludFJldHVybkNvbW1hbmQoYWxvZywgY21kcyk7Ci0gICAgICAgICAgICBhbG9nIDw8IGRlZGVudDsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotICAgIAotICAgIHJldHVybiBlcnI7Ci19Ci0KLXN0YXR1c190IElQQ1RocmVhZFN0YXRlOjp3cml0ZVRyYW5zYWN0aW9uRGF0YShpbnQzMl90IGNtZCwgdWludDMyX3QgYmluZGVyRmxhZ3MsCi0gICAgaW50MzJfdCBoYW5kbGUsIHVpbnQzMl90IGNvZGUsIGNvbnN0IFBhcmNlbCYgZGF0YSwgc3RhdHVzX3QqIHN0YXR1c0J1ZmZlcikKLXsKLSAgICBiaW5kZXJfdHJhbnNhY3Rpb25fZGF0YSB0cjsKLQotICAgIHRyLnRhcmdldC5oYW5kbGUgPSBoYW5kbGU7Ci0gICAgdHIuY29kZSA9IGNvZGU7Ci0gICAgdHIuZmxhZ3MgPSBiaW5kZXJGbGFnczsKLSAgICAKLSAgICBjb25zdCBzdGF0dXNfdCBlcnIgPSBkYXRhLmVycm9yQ2hlY2soKTsKLSAgICBpZiAoZXJyID09IE5PX0VSUk9SKSB7Ci0gICAgICAgIHRyLmRhdGFfc2l6ZSA9IGRhdGEuaXBjRGF0YVNpemUoKTsKLSAgICAgICAgdHIuZGF0YS5wdHIuYnVmZmVyID0gZGF0YS5pcGNEYXRhKCk7Ci0gICAgICAgIHRyLm9mZnNldHNfc2l6ZSA9IGRhdGEuaXBjT2JqZWN0c0NvdW50KCkqc2l6ZW9mKHNpemVfdCk7Ci0gICAgICAgIHRyLmRhdGEucHRyLm9mZnNldHMgPSBkYXRhLmlwY09iamVjdHMoKTsKLSAgICB9IGVsc2UgaWYgKHN0YXR1c0J1ZmZlcikgewotICAgICAgICB0ci5mbGFncyB8PSBURl9TVEFUVVNfQ09ERTsKLSAgICAgICAgKnN0YXR1c0J1ZmZlciA9IGVycjsKLSAgICAgICAgdHIuZGF0YV9zaXplID0gc2l6ZW9mKHN0YXR1c190KTsKLSAgICAgICAgdHIuZGF0YS5wdHIuYnVmZmVyID0gc3RhdHVzQnVmZmVyOwotICAgICAgICB0ci5vZmZzZXRzX3NpemUgPSAwOwotICAgICAgICB0ci5kYXRhLnB0ci5vZmZzZXRzID0gTlVMTDsKLSAgICB9IGVsc2UgewotICAgICAgICByZXR1cm4gKG1MYXN0RXJyb3IgPSBlcnIpOwotICAgIH0KLSAgICAKLSAgICBtT3V0LndyaXRlSW50MzIoY21kKTsKLSAgICBtT3V0LndyaXRlKCZ0ciwgc2l6ZW9mKHRyKSk7Ci0gICAgCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zcDxCQmluZGVyPiB0aGVfY29udGV4dF9vYmplY3Q7Ci0KLXZvaWQgc2V0VGhlQ29udGV4dE9iamVjdChzcDxCQmluZGVyPiBvYmopCi17Ci0gICAgdGhlX2NvbnRleHRfb2JqZWN0ID0gb2JqOwotfQotCi1zdGF0dXNfdCBJUENUaHJlYWRTdGF0ZTo6ZXhlY3V0ZUNvbW1hbmQoaW50MzJfdCBjbWQpCi17Ci0gICAgQkJpbmRlciogb2JqOwotICAgIFJlZkJhc2U6OndlYWtyZWZfdHlwZSogcmVmczsKLSAgICBzdGF0dXNfdCByZXN1bHQgPSBOT19FUlJPUjsKLSAgICAKLSAgICBzd2l0Y2ggKGNtZCkgewotICAgIGNhc2UgQlJfRVJST1I6Ci0gICAgICAgIHJlc3VsdCA9IG1Jbi5yZWFkSW50MzIoKTsKLSAgICAgICAgYnJlYWs7Ci0gICAgICAgIAotICAgIGNhc2UgQlJfT0s6Ci0gICAgICAgIGJyZWFrOwotICAgICAgICAKLSAgICBjYXNlIEJSX0FDUVVJUkU6Ci0gICAgICAgIHJlZnMgPSAoUmVmQmFzZTo6d2Vha3JlZl90eXBlKiltSW4ucmVhZEludDMyKCk7Ci0gICAgICAgIG9iaiA9IChCQmluZGVyKiltSW4ucmVhZEludDMyKCk7Ci0gICAgICAgIExPR19BU1NFUlQocmVmcy0+cmVmQmFzZSgpID09IG9iaiwKLSAgICAgICAgICAgICAgICAgICAiQlJfQUNRVUlSRTogb2JqZWN0ICVwIGRvZXMgbm90IG1hdGNoIGNvb2tpZSAlcCAoZXhwZWN0ZWQgJXApIiwKLSAgICAgICAgICAgICAgICAgICByZWZzLCBvYmosIHJlZnMtPnJlZkJhc2UoKSk7Ci0gICAgICAgIG9iai0+aW5jU3Ryb25nKG1Qcm9jZXNzLmdldCgpKTsKLSAgICAgICAgSUZfTE9HX1JFTU9URVJFRlMoKSB7Ci0gICAgICAgICAgICBMT0dfUkVNT1RFUkVGUygiQlJfQUNRVUlSRSBmcm9tIGRyaXZlciBvbiAlcCIsIG9iaik7Ci0gICAgICAgICAgICBvYmotPnByaW50UmVmcygpOwotICAgICAgICB9Ci0gICAgICAgIG1PdXQud3JpdGVJbnQzMihCQ19BQ1FVSVJFX0RPTkUpOwotICAgICAgICBtT3V0LndyaXRlSW50MzIoKGludDMyX3QpcmVmcyk7Ci0gICAgICAgIG1PdXQud3JpdGVJbnQzMigoaW50MzJfdClvYmopOwotICAgICAgICBicmVhazsKLSAgICAgICAgCi0gICAgY2FzZSBCUl9SRUxFQVNFOgotICAgICAgICByZWZzID0gKFJlZkJhc2U6OndlYWtyZWZfdHlwZSopbUluLnJlYWRJbnQzMigpOwotICAgICAgICBvYmogPSAoQkJpbmRlciopbUluLnJlYWRJbnQzMigpOwotICAgICAgICBMT0dfQVNTRVJUKHJlZnMtPnJlZkJhc2UoKSA9PSBvYmosCi0gICAgICAgICAgICAgICAgICAgIkJSX1JFTEVBU0U6IG9iamVjdCAlcCBkb2VzIG5vdCBtYXRjaCBjb29raWUgJXAgKGV4cGVjdGVkICVwKSIsCi0gICAgICAgICAgICAgICAgICAgcmVmcywgb2JqLCByZWZzLT5yZWZCYXNlKCkpOwotICAgICAgICBJRl9MT0dfUkVNT1RFUkVGUygpIHsKLSAgICAgICAgICAgIExPR19SRU1PVEVSRUZTKCJCUl9SRUxFQVNFIGZyb20gZHJpdmVyIG9uICVwIiwgb2JqKTsKLSAgICAgICAgICAgIG9iai0+cHJpbnRSZWZzKCk7Ci0gICAgICAgIH0KLSAgICAgICAgbVBlbmRpbmdTdHJvbmdEZXJlZnMucHVzaChvYmopOwotICAgICAgICBicmVhazsKLSAgICAgICAgCi0gICAgY2FzZSBCUl9JTkNSRUZTOgotICAgICAgICByZWZzID0gKFJlZkJhc2U6OndlYWtyZWZfdHlwZSopbUluLnJlYWRJbnQzMigpOwotICAgICAgICBvYmogPSAoQkJpbmRlciopbUluLnJlYWRJbnQzMigpOwotICAgICAgICByZWZzLT5pbmNXZWFrKG1Qcm9jZXNzLmdldCgpKTsKLSAgICAgICAgbU91dC53cml0ZUludDMyKEJDX0lOQ1JFRlNfRE9ORSk7Ci0gICAgICAgIG1PdXQud3JpdGVJbnQzMigoaW50MzJfdClyZWZzKTsKLSAgICAgICAgbU91dC53cml0ZUludDMyKChpbnQzMl90KW9iaik7Ci0gICAgICAgIGJyZWFrOwotICAgICAgICAKLSAgICBjYXNlIEJSX0RFQ1JFRlM6Ci0gICAgICAgIHJlZnMgPSAoUmVmQmFzZTo6d2Vha3JlZl90eXBlKiltSW4ucmVhZEludDMyKCk7Ci0gICAgICAgIG9iaiA9IChCQmluZGVyKiltSW4ucmVhZEludDMyKCk7Ci0gICAgICAgIC8vIE5PVEU6IFRoaXMgYXNzZXJ0aW9uIGlzIG5vdCB2YWxpZCwgYmVjYXVzZSB0aGUgb2JqZWN0IG1heSBubwotICAgICAgICAvLyBsb25nZXIgZXhpc3QgKHRodXMgdGhlIChCQmluZGVyKiljYXN0IGFib3ZlIHJlc3VsdGluZyBpbiBhIGRpZmZlcmVudAotICAgICAgICAvLyBtZW1vcnkgYWRkcmVzcykuCi0gICAgICAgIC8vTE9HX0FTU0VSVChyZWZzLT5yZWZCYXNlKCkgPT0gb2JqLAotICAgICAgICAvLyAgICAgICAgICAgIkJSX0RFQ1JFRlM6IG9iamVjdCAlcCBkb2VzIG5vdCBtYXRjaCBjb29raWUgJXAgKGV4cGVjdGVkICVwKSIsCi0gICAgICAgIC8vICAgICAgICAgICByZWZzLCBvYmosIHJlZnMtPnJlZkJhc2UoKSk7Ci0gICAgICAgIG1QZW5kaW5nV2Vha0RlcmVmcy5wdXNoKHJlZnMpOwotICAgICAgICBicmVhazsKLSAgICAgICAgCi0gICAgY2FzZSBCUl9BVFRFTVBUX0FDUVVJUkU6Ci0gICAgICAgIHJlZnMgPSAoUmVmQmFzZTo6d2Vha3JlZl90eXBlKiltSW4ucmVhZEludDMyKCk7Ci0gICAgICAgIG9iaiA9IChCQmluZGVyKiltSW4ucmVhZEludDMyKCk7Ci0gICAgICAgICAKLSAgICAgICAgewotICAgICAgICAgICAgY29uc3QgYm9vbCBzdWNjZXNzID0gcmVmcy0+YXR0ZW1wdEluY1N0cm9uZyhtUHJvY2Vzcy5nZXQoKSk7Ci0gICAgICAgICAgICBMT0dfQVNTRVJUKHN1Y2Nlc3MgJiYgcmVmcy0+cmVmQmFzZSgpID09IG9iaiwKLSAgICAgICAgICAgICAgICAgICAgICAgIkJSX0FUVEVNUFRfQUNRVUlSRTogb2JqZWN0ICVwIGRvZXMgbm90IG1hdGNoIGNvb2tpZSAlcCAoZXhwZWN0ZWQgJXApIiwKLSAgICAgICAgICAgICAgICAgICAgICAgcmVmcywgb2JqLCByZWZzLT5yZWZCYXNlKCkpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBtT3V0LndyaXRlSW50MzIoQkNfQUNRVUlSRV9SRVNVTFQpOwotICAgICAgICAgICAgbU91dC53cml0ZUludDMyKChpbnQzMl90KXN1Y2Nlc3MpOwotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgIAotICAgIGNhc2UgQlJfVFJBTlNBQ1RJT046Ci0gICAgICAgIHsKLSAgICAgICAgICAgIGJpbmRlcl90cmFuc2FjdGlvbl9kYXRhIHRyOwotICAgICAgICAgICAgcmVzdWx0ID0gbUluLnJlYWQoJnRyLCBzaXplb2YodHIpKTsKLSAgICAgICAgICAgIExPR19BU1NFUlQocmVzdWx0ID09IE5PX0VSUk9SLAotICAgICAgICAgICAgICAgICJOb3QgZW5vdWdoIGNvbW1hbmQgZGF0YSBmb3IgYnJUUkFOU0FDVElPTiIpOwotICAgICAgICAgICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikgYnJlYWs7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIFBhcmNlbCBidWZmZXI7Ci0gICAgICAgICAgICBidWZmZXIuaXBjU2V0RGF0YVJlZmVyZW5jZSgKLSAgICAgICAgICAgICAgICByZWludGVycHJldF9jYXN0PGNvbnN0IHVpbnQ4X3QqPih0ci5kYXRhLnB0ci5idWZmZXIpLAotICAgICAgICAgICAgICAgIHRyLmRhdGFfc2l6ZSwKLSAgICAgICAgICAgICAgICByZWludGVycHJldF9jYXN0PGNvbnN0IHNpemVfdCo+KHRyLmRhdGEucHRyLm9mZnNldHMpLAotICAgICAgICAgICAgICAgIHRyLm9mZnNldHNfc2l6ZS9zaXplb2Yoc2l6ZV90KSwgZnJlZUJ1ZmZlciwgdGhpcyk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIGNvbnN0IHBpZF90IG9yaWdQaWQgPSBtQ2FsbGluZ1BpZDsKLSAgICAgICAgICAgIGNvbnN0IHVpZF90IG9yaWdVaWQgPSBtQ2FsbGluZ1VpZDsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgbUNhbGxpbmdQaWQgPSB0ci5zZW5kZXJfcGlkOwotICAgICAgICAgICAgbUNhbGxpbmdVaWQgPSB0ci5zZW5kZXJfZXVpZDsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy9MT0dJKCI+Pj4+IFRSQU5TQUNUIGZyb20gcGlkICVkIHVpZCAlZFxuIiwgbUNhbGxpbmdQaWQsIG1DYWxsaW5nVWlkKTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgUGFyY2VsIHJlcGx5OwotICAgICAgICAgICAgSUZfTE9HX1RSQU5TQUNUSU9OUygpIHsKLSAgICAgICAgICAgICAgICBUZXh0T3V0cHV0OjpCdW5kbGUgX2IoYWxvZyk7Ci0gICAgICAgICAgICAgICAgYWxvZyA8PCAiQlJfVFJBTlNBQ1RJT04gdGhyICIgPDwgKHZvaWQqKXB0aHJlYWRfc2VsZigpCi0gICAgICAgICAgICAgICAgICAgIDw8ICIgLyBvYmogIiA8PCB0ci50YXJnZXQucHRyIDw8ICIgLyBjb2RlICIKLSAgICAgICAgICAgICAgICAgICAgPDwgVHlwZUNvZGUodHIuY29kZSkgPDwgIjogIiA8PCBpbmRlbnQgPDwgYnVmZmVyCi0gICAgICAgICAgICAgICAgICAgIDw8IGRlZGVudCA8PCBlbmRsCi0gICAgICAgICAgICAgICAgICAgIDw8ICJEYXRhIGFkZHIgPSAiCi0gICAgICAgICAgICAgICAgICAgIDw8IHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgdWludDhfdCo+KHRyLmRhdGEucHRyLmJ1ZmZlcikKLSAgICAgICAgICAgICAgICAgICAgPDwgIiwgb2Zmc2V0cyBhZGRyPSIKLSAgICAgICAgICAgICAgICAgICAgPDwgcmVpbnRlcnByZXRfY2FzdDxjb25zdCBzaXplX3QqPih0ci5kYXRhLnB0ci5vZmZzZXRzKSA8PCBlbmRsOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHRyLnRhcmdldC5wdHIpIHsKLSAgICAgICAgICAgICAgICBzcDxCQmluZGVyPiBiKChCQmluZGVyKil0ci5jb29raWUpOwotICAgICAgICAgICAgICAgIGNvbnN0IHN0YXR1c190IGVycm9yID0gYi0+dHJhbnNhY3QodHIuY29kZSwgYnVmZmVyLCAmcmVwbHksIDApOwotICAgICAgICAgICAgICAgIGlmIChlcnJvciA8IE5PX0VSUk9SKSByZXBseS5zZXRFcnJvcihlcnJvcik7Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGNvbnN0IHN0YXR1c190IGVycm9yID0gdGhlX2NvbnRleHRfb2JqZWN0LT50cmFuc2FjdCh0ci5jb2RlLCBidWZmZXIsICZyZXBseSwgMCk7Ci0gICAgICAgICAgICAgICAgaWYgKGVycm9yIDwgTk9fRVJST1IpIHJlcGx5LnNldEVycm9yKGVycm9yKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy9MT0dJKCI8PDw8IFRSQU5TQUNUIGZyb20gcGlkICVkIHJlc3RvcmUgcGlkICVkIHVpZCAlZFxuIiwKLSAgICAgICAgICAgIC8vICAgICBtQ2FsbGluZ1BpZCwgb3JpZ1BpZCwgb3JpZ1VpZCk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIGlmICgodHIuZmxhZ3MgJiBURl9PTkVfV0FZKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgTE9HX09ORVdBWSgiU2VuZGluZyByZXBseSB0byAlZCEiLCBtQ2FsbGluZ1BpZCk7Ci0gICAgICAgICAgICAgICAgc2VuZFJlcGx5KHJlcGx5LCAwKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgTE9HX09ORVdBWSgiTk9UIHNlbmRpbmcgcmVwbHkgdG8gJWQhIiwgbUNhbGxpbmdQaWQpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgCi0gICAgICAgICAgICBtQ2FsbGluZ1BpZCA9IG9yaWdQaWQ7Ci0gICAgICAgICAgICBtQ2FsbGluZ1VpZCA9IG9yaWdVaWQ7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIElGX0xPR19UUkFOU0FDVElPTlMoKSB7Ci0gICAgICAgICAgICAgICAgVGV4dE91dHB1dDo6QnVuZGxlIF9iKGFsb2cpOwotICAgICAgICAgICAgICAgIGFsb2cgPDwgIkJDX1JFUExZIHRociAiIDw8ICh2b2lkKilwdGhyZWFkX3NlbGYoKSA8PCAiIC8gb2JqICIKLSAgICAgICAgICAgICAgICAgICAgPDwgdHIudGFyZ2V0LnB0ciA8PCAiOiAiIDw8IGluZGVudCA8PCByZXBseSA8PCBkZWRlbnQgPDwgZW5kbDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIAotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgIAotICAgIGNhc2UgQlJfREVBRF9CSU5ERVI6Ci0gICAgICAgIHsKLSAgICAgICAgICAgIEJwQmluZGVyICpwcm94eSA9IChCcEJpbmRlciopbUluLnJlYWRJbnQzMigpOwotICAgICAgICAgICAgcHJveHktPnNlbmRPYml0dWFyeSgpOwotICAgICAgICAgICAgbU91dC53cml0ZUludDMyKEJDX0RFQURfQklOREVSX0RPTkUpOwotICAgICAgICAgICAgbU91dC53cml0ZUludDMyKChpbnQzMl90KXByb3h5KTsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgCi0gICAgY2FzZSBCUl9DTEVBUl9ERUFUSF9OT1RJRklDQVRJT05fRE9ORToKLSAgICAgICAgewotICAgICAgICAgICAgQnBCaW5kZXIgKnByb3h5ID0gKEJwQmluZGVyKiltSW4ucmVhZEludDMyKCk7Ci0gICAgICAgICAgICBwcm94eS0+Z2V0V2Vha1JlZnMoKS0+ZGVjV2Vhayhwcm94eSk7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIAotICAgIGNhc2UgQlJfRklOSVNIRUQ6Ci0gICAgICAgIHJlc3VsdCA9IFRJTUVEX09VVDsKLSAgICAgICAgYnJlYWs7Ci0gICAgICAgIAotICAgIGNhc2UgQlJfTk9PUDoKLSAgICAgICAgYnJlYWs7Ci0gICAgICAgIAotICAgIGNhc2UgQlJfU1BBV05fTE9PUEVSOgotICAgICAgICBtUHJvY2Vzcy0+c3Bhd25Qb29sZWRUaHJlYWQoZmFsc2UpOwotICAgICAgICBicmVhazsKLSAgICAgICAgCi0gICAgZGVmYXVsdDoKLSAgICAgICAgcHJpbnRmKCIqKiogQkFEIENPTU1BTkQgJWQgcmVjZWl2ZWQgZnJvbSBCaW5kZXIgZHJpdmVyXG4iLCBjbWQpOwotICAgICAgICByZXN1bHQgPSBVTktOT1dOX0VSUk9SOwotICAgICAgICBicmVhazsKLSAgICB9Ci0KLSAgICBpZiAocmVzdWx0ICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgIG1MYXN0RXJyb3IgPSByZXN1bHQ7Ci0gICAgfQotICAgIAotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLXZvaWQgSVBDVGhyZWFkU3RhdGU6OnRocmVhZERlc3RydWN0b3Iodm9pZCAqc3QpCi17Ci0JSVBDVGhyZWFkU3RhdGUqIGNvbnN0IHNlbGYgPSBzdGF0aWNfY2FzdDxJUENUaHJlYWRTdGF0ZSo+KHN0KTsKLQlpZiAoc2VsZikgewotCQlzZWxmLT5mbHVzaENvbW1hbmRzKCk7Ci0jaWYgZGVmaW5lZChIQVZFX0FORFJPSURfT1MpCi0gICAgICAgIGlvY3RsKHNlbGYtPm1Qcm9jZXNzLT5tRHJpdmVyRkQsIEJJTkRFUl9USFJFQURfRVhJVCwgMCk7Ci0jZW5kaWYKLQkJZGVsZXRlIHNlbGY7Ci0JfQotfQotCi0KLXZvaWQgSVBDVGhyZWFkU3RhdGU6OmZyZWVCdWZmZXIoUGFyY2VsKiBwYXJjZWwsIGNvbnN0IHVpbnQ4X3QqIGRhdGEsIHNpemVfdCBkYXRhU2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3Qgc2l6ZV90KiBvYmplY3RzLCBzaXplX3Qgb2JqZWN0c1NpemUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQqIGNvb2tpZSkKLXsKLSAgICAvL0xPR0koIkZyZWVpbmcgcGFyY2VsICVwIiwgJnBhcmNlbCk7Ci0gICAgSUZfTE9HX0NPTU1BTkRTKCkgewotICAgICAgICBhbG9nIDw8ICJXcml0aW5nIEJDX0ZSRUVfQlVGRkVSIGZvciAiIDw8IGRhdGEgPDwgZW5kbDsKLSAgICB9Ci0gICAgTE9HX0FTU0VSVChkYXRhICE9IE5VTEwsICJDYWxsZWQgd2l0aCBOVUxMIGRhdGEiKTsKLSAgICBpZiAocGFyY2VsICE9IE5VTEwpIHBhcmNlbC0+Y2xvc2VGaWxlRGVzY3JpcHRvcnMoKTsKLSAgICBJUENUaHJlYWRTdGF0ZSogc3RhdGUgPSBzZWxmKCk7Ci0gICAgc3RhdGUtPm1PdXQud3JpdGVJbnQzMihCQ19GUkVFX0JVRkZFUik7Ci0gICAgc3RhdGUtPm1PdXQud3JpdGVJbnQzMigoaW50MzJfdClkYXRhKTsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvSVBlcm1pc3Npb25Db250cm9sbGVyLmNwcCBiL2xpYnMvdXRpbHMvSVBlcm1pc3Npb25Db250cm9sbGVyLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZjAxZDM4Zi4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL0lQZXJtaXNzaW9uQ29udHJvbGxlci5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw4NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJQZXJtaXNzaW9uQ29udHJvbGxlciIKLQotI2luY2x1ZGUgPHV0aWxzL0lQZXJtaXNzaW9uQ29udHJvbGxlci5oPgotCi0jaW5jbHVkZSA8dXRpbHMvRGVidWcuaD4KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLSNpbmNsdWRlIDx1dGlscy9QYXJjZWwuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0KLSNpbmNsdWRlIDxwcml2YXRlL3V0aWxzL1N0YXRpYy5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgQnBQZXJtaXNzaW9uQ29udHJvbGxlciA6IHB1YmxpYyBCcEludGVyZmFjZTxJUGVybWlzc2lvbkNvbnRyb2xsZXI+Ci17Ci1wdWJsaWM6Ci0gICAgQnBQZXJtaXNzaW9uQ29udHJvbGxlcihjb25zdCBzcDxJQmluZGVyPiYgaW1wbCkKLSAgICAgICAgOiBCcEludGVyZmFjZTxJUGVybWlzc2lvbkNvbnRyb2xsZXI+KGltcGwpCi0gICAgewotICAgIH0KLSAgICAgICAgCi0gICAgdmlydHVhbCBib29sIGNoZWNrUGVybWlzc2lvbihjb25zdCBTdHJpbmcxNiYgcGVybWlzc2lvbiwgaW50MzJfdCBwaWQsIGludDMyX3QgdWlkKQotICAgIHsKLSAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICBkYXRhLndyaXRlSW50ZXJmYWNlVG9rZW4oSVBlcm1pc3Npb25Db250cm9sbGVyOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpOwotICAgICAgICBkYXRhLndyaXRlU3RyaW5nMTYocGVybWlzc2lvbik7Ci0gICAgICAgIGRhdGEud3JpdGVJbnQzMihwaWQpOwotICAgICAgICBkYXRhLndyaXRlSW50MzIodWlkKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KENIRUNLX1BFUk1JU1NJT05fVFJBTlNBQ1RJT04sIGRhdGEsICZyZXBseSk7Ci0gICAgICAgIC8vIGZhaWwgb24gZXhjZXB0aW9uCi0gICAgICAgIGlmIChyZXBseS5yZWFkSW50MzIoKSAhPSAwKSByZXR1cm4gMDsKLSAgICAgICAgcmV0dXJuIHJlcGx5LnJlYWRJbnQzMigpICE9IDA7Ci0gICAgfQotfTsKLQotSU1QTEVNRU5UX01FVEFfSU5URVJGQUNFKFBlcm1pc3Npb25Db250cm9sbGVyLCAiYW5kcm9pZC5vcy5JUGVybWlzc2lvbkNvbnRyb2xsZXIiKTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZGVmaW5lIENIRUNLX0lOVEVSRkFDRShpbnRlcmZhY2UsIGRhdGEsIHJlcGx5KSBcCi0gICAgICAgIGRvIHsgaWYgKCFkYXRhLmVuZm9yY2VJbnRlcmZhY2UoaW50ZXJmYWNlOjpnZXRJbnRlcmZhY2VEZXNjcmlwdG9yKCkpKSB7IFwKLSAgICAgICAgICAgIExPR1coIkNhbGwgaW5jb3JyZWN0bHkgcm91dGVkIHRvICIgI2ludGVyZmFjZSk7IFwKLSAgICAgICAgICAgIHJldHVybiBQRVJNSVNTSU9OX0RFTklFRDsgXAotICAgICAgICB9IH0gd2hpbGUgKDApCi0KLXN0YXR1c190IEJuUGVybWlzc2lvbkNvbnRyb2xsZXI6Om9uVHJhbnNhY3QoCi0gICAgdWludDMyX3QgY29kZSwgY29uc3QgUGFyY2VsJiBkYXRhLCBQYXJjZWwqIHJlcGx5LCB1aW50MzJfdCBmbGFncykKLXsKLSAgICAvL3ByaW50ZigiUGVybWlzc2lvbkNvbnRyb2xsZXIgcmVjZWl2ZWQ6ICIpOyBkYXRhLnByaW50KCk7Ci0gICAgc3dpdGNoKGNvZGUpIHsKLSAgICAgICAgY2FzZSBDSEVDS19QRVJNSVNTSU9OX1RSQU5TQUNUSU9OOiB7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSVBlcm1pc3Npb25Db250cm9sbGVyLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBTdHJpbmcxNiBwZXJtaXNzaW9uID0gZGF0YS5yZWFkU3RyaW5nMTYoKTsKLSAgICAgICAgICAgIGludDMyX3QgcGlkID0gZGF0YS5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgIGludDMyX3QgdWlkID0gZGF0YS5yZWFkSW50MzIoKTsKLSAgICAgICAgICAgIGJvb2wgcmVzID0gY2hlY2tQZXJtaXNzaW9uKHBlcm1pc3Npb24sIHBpZCwgdWlkKTsKLSAgICAgICAgICAgIC8vIHdyaXRlIGV4Y2VwdGlvbgotICAgICAgICAgICAgcmVwbHktPndyaXRlSW50MzIoMCk7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihyZXMgPyAxIDogMCk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICByZXR1cm4gQkJpbmRlcjo6b25UcmFuc2FjdChjb2RlLCBkYXRhLCByZXBseSwgZmxhZ3MpOwotICAgIH0KLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9JU2VydmljZU1hbmFnZXIuY3BwIGIvbGlicy91dGlscy9JU2VydmljZU1hbmFnZXIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5YmVlYWRkLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvSVNlcnZpY2VNYW5hZ2VyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDIzMCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJTZXJ2aWNlTWFuYWdlciIKLQotI2luY2x1ZGUgPHV0aWxzL0lTZXJ2aWNlTWFuYWdlci5oPgotCi0jaW5jbHVkZSA8dXRpbHMvRGVidWcuaD4KLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLSNpbmNsdWRlIDx1dGlscy9TeXN0ZW1DbG9jay5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS91dGlscy9TdGF0aWMuaD4KLQotI2luY2x1ZGUgPHVuaXN0ZC5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLXNwPElTZXJ2aWNlTWFuYWdlcj4gZGVmYXVsdFNlcnZpY2VNYW5hZ2VyKCkKLXsKLSAgICBpZiAoZ0RlZmF1bHRTZXJ2aWNlTWFuYWdlciAhPSBOVUxMKSByZXR1cm4gZ0RlZmF1bHRTZXJ2aWNlTWFuYWdlcjsKLSAgICAKLSAgICB7Ci0gICAgICAgIEF1dG9NdXRleCBfbChnRGVmYXVsdFNlcnZpY2VNYW5hZ2VyTG9jayk7Ci0gICAgICAgIGlmIChnRGVmYXVsdFNlcnZpY2VNYW5hZ2VyID09IE5VTEwpIHsKLSAgICAgICAgICAgIGdEZWZhdWx0U2VydmljZU1hbmFnZXIgPSBpbnRlcmZhY2VfY2FzdDxJU2VydmljZU1hbmFnZXI+KAotICAgICAgICAgICAgICAgIFByb2Nlc3NTdGF0ZTo6c2VsZigpLT5nZXRDb250ZXh0T2JqZWN0KE5VTEwpKTsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICByZXR1cm4gZ0RlZmF1bHRTZXJ2aWNlTWFuYWdlcjsKLX0KLQotYm9vbCBjaGVja0NhbGxpbmdQZXJtaXNzaW9uKGNvbnN0IFN0cmluZzE2JiBwZXJtaXNzaW9uKQotewotICAgIHJldHVybiBjaGVja0NhbGxpbmdQZXJtaXNzaW9uKHBlcm1pc3Npb24sIE5VTEwsIE5VTEwpOwotfQotCi1zdGF0aWMgU3RyaW5nMTYgX3Blcm1pc3Npb24oInBlcm1pc3Npb24iKTsKLQotYm9vbCBjaGVja0NhbGxpbmdQZXJtaXNzaW9uKGNvbnN0IFN0cmluZzE2JiBwZXJtaXNzaW9uLCBpbnQzMl90KiBvdXRQaWQsIGludDMyX3QqIG91dFVpZCkKLXsKLSAgICBJUENUaHJlYWRTdGF0ZSogaXBjU3RhdGUgPSBJUENUaHJlYWRTdGF0ZTo6c2VsZigpOwotICAgIGludDMyX3QgcGlkID0gaXBjU3RhdGUtPmdldENhbGxpbmdQaWQoKTsKLSAgICBpbnQzMl90IHVpZCA9IGlwY1N0YXRlLT5nZXRDYWxsaW5nVWlkKCk7Ci0gICAgaWYgKG91dFBpZCkgKm91dFBpZCA9IHBpZDsKLSAgICBpZiAob3V0VWlkKSAqb3V0VWlkPSB1aWQ7Ci0gICAgCi0gICAgc3A8SVBlcm1pc3Npb25Db250cm9sbGVyPiBwYzsKLSAgICBnRGVmYXVsdFNlcnZpY2VNYW5hZ2VyTG9jay5sb2NrKCk7Ci0gICAgcGMgPSBnUGVybWlzc2lvbkNvbnRyb2xsZXI7Ci0gICAgZ0RlZmF1bHRTZXJ2aWNlTWFuYWdlckxvY2sudW5sb2NrKCk7Ci0gICAgCi0gICAgaW50NjRfdCBzdGFydFRpbWUgPSAwOwotCi0gICAgd2hpbGUgKHRydWUpIHsKLSAgICAgICAgaWYgKHBjICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGJvb2wgcmVzID0gcGMtPmNoZWNrUGVybWlzc2lvbihwZXJtaXNzaW9uLCBwaWQsIHVpZCk7Ci0gICAgICAgICAgICBpZiAocmVzKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHN0YXJ0VGltZSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIExPR0koIkNoZWNrIHBhc3NlZCBhZnRlciAlZCBzZWNvbmRzIGZvciAlcyBmcm9tIHVpZD0lZCBwaWQ9JWQiLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIChpbnQpKCh1cHRpbWVNaWxsaXMoKS1zdGFydFRpbWUpLzEwMDApLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgocGVybWlzc2lvbikuc3RyaW5nKCksIHVpZCwgcGlkKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHJlczsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8gSXMgdGhpcyBhIHBlcm1pc3Npb24gZmFpbHVyZSwgb3IgZGlkIHRoZSBjb250cm9sbGVyIGdvIGF3YXk/Ci0gICAgICAgICAgICBpZiAocGMtPmFzQmluZGVyKCktPmlzQmluZGVyQWxpdmUoKSkgewotICAgICAgICAgICAgICAgIExPR1coIlBlcm1pc3Npb24gZmFpbHVyZTogJXMgZnJvbSB1aWQ9JWQgcGlkPSVkIiwKLSAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgocGVybWlzc2lvbikuc3RyaW5nKCksIHVpZCwgcGlkKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8vIE9iamVjdCBpcyBkZWFkIQotICAgICAgICAgICAgZ0RlZmF1bHRTZXJ2aWNlTWFuYWdlckxvY2subG9jaygpOwotICAgICAgICAgICAgaWYgKGdQZXJtaXNzaW9uQ29udHJvbGxlciA9PSBwYykgewotICAgICAgICAgICAgICAgIGdQZXJtaXNzaW9uQ29udHJvbGxlciA9IE5VTEw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBnRGVmYXVsdFNlcnZpY2VNYW5hZ2VyTG9jay51bmxvY2soKTsKLSAgICAgICAgfQotICAgIAotICAgICAgICAvLyBOZWVkIHRvIHJldHJpZXZlIHRoZSBwZXJtaXNzaW9uIGNvbnRyb2xsZXIuCi0gICAgICAgIHNwPElCaW5kZXI+IGJpbmRlciA9IGRlZmF1bHRTZXJ2aWNlTWFuYWdlcigpLT5jaGVja1NlcnZpY2UoX3Blcm1pc3Npb24pOwotICAgICAgICBpZiAoYmluZGVyID09IE5VTEwpIHsKLSAgICAgICAgICAgIC8vIFdhaXQgZm9yIHRoZSBwZXJtaXNzaW9uIGNvbnRyb2xsZXIgdG8gY29tZSBiYWNrLi4uCi0gICAgICAgICAgICBpZiAoc3RhcnRUaW1lID09IDApIHsKLSAgICAgICAgICAgICAgICBzdGFydFRpbWUgPSB1cHRpbWVNaWxsaXMoKTsKLSAgICAgICAgICAgICAgICBMT0dJKCJXYWl0aW5nIHRvIGNoZWNrIHBlcm1pc3Npb24gJXMgZnJvbSB1aWQ9JWQgcGlkPSVkIiwKLSAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgocGVybWlzc2lvbikuc3RyaW5nKCksIHVpZCwgcGlkKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHNsZWVwKDEpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcGMgPSBpbnRlcmZhY2VfY2FzdDxJUGVybWlzc2lvbkNvbnRyb2xsZXI+KGJpbmRlcik7Ci0gICAgICAgICAgICAvLyBJbnN0YWxsIHRoZSBuZXcgcGVybWlzc2lvbiBjb250cm9sbGVyLCBhbmQgdHJ5IGFnYWluLiAgICAgICAgCi0gICAgICAgICAgICBnRGVmYXVsdFNlcnZpY2VNYW5hZ2VyTG9jay5sb2NrKCk7Ci0gICAgICAgICAgICBnUGVybWlzc2lvbkNvbnRyb2xsZXIgPSBwYzsKLSAgICAgICAgICAgIGdEZWZhdWx0U2VydmljZU1hbmFnZXJMb2NrLnVubG9jaygpOwotICAgICAgICB9Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWNsYXNzIEJwU2VydmljZU1hbmFnZXIgOiBwdWJsaWMgQnBJbnRlcmZhY2U8SVNlcnZpY2VNYW5hZ2VyPgotewotcHVibGljOgotICAgIEJwU2VydmljZU1hbmFnZXIoY29uc3Qgc3A8SUJpbmRlcj4mIGltcGwpCi0gICAgICAgIDogQnBJbnRlcmZhY2U8SVNlcnZpY2VNYW5hZ2VyPihpbXBsKQotICAgIHsKLSAgICB9Ci0gICAgICAgIAotICAgIHZpcnR1YWwgc3A8SUJpbmRlcj4gZ2V0U2VydmljZShjb25zdCBTdHJpbmcxNiYgbmFtZSkgY29uc3QKLSAgICB7Ci0gICAgICAgIHVuc2lnbmVkIG47Ci0gICAgICAgIGZvciAobiA9IDA7IG4gPCA1OyBuKyspewotICAgICAgICAgICAgc3A8SUJpbmRlcj4gc3ZjID0gY2hlY2tTZXJ2aWNlKG5hbWUpOwotICAgICAgICAgICAgaWYgKHN2YyAhPSBOVUxMKSByZXR1cm4gc3ZjOwotICAgICAgICAgICAgTE9HSSgiV2FpdGluZyBmb3Igc2V2aWNlICVzLi4uXG4iLCBTdHJpbmc4KG5hbWUpLnN0cmluZygpKTsKLSAgICAgICAgICAgIHNsZWVwKDEpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBOVUxMOwotICAgIH0KLSAgICAKLSAgICB2aXJ0dWFsIHNwPElCaW5kZXI+IGNoZWNrU2VydmljZSggY29uc3QgU3RyaW5nMTYmIG5hbWUpIGNvbnN0Ci0gICAgewotICAgICAgICBQYXJjZWwgZGF0YSwgcmVwbHk7Ci0gICAgICAgIGRhdGEud3JpdGVJbnRlcmZhY2VUb2tlbihJU2VydmljZU1hbmFnZXI6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGRhdGEud3JpdGVTdHJpbmcxNihuYW1lKTsKLSAgICAgICAgcmVtb3RlKCktPnRyYW5zYWN0KENIRUNLX1NFUlZJQ0VfVFJBTlNBQ1RJT04sIGRhdGEsICZyZXBseSk7Ci0gICAgICAgIHJldHVybiByZXBseS5yZWFkU3Ryb25nQmluZGVyKCk7Ci0gICAgfQotCi0gICAgdmlydHVhbCBzdGF0dXNfdCBhZGRTZXJ2aWNlKGNvbnN0IFN0cmluZzE2JiBuYW1lLCBjb25zdCBzcDxJQmluZGVyPiYgc2VydmljZSkKLSAgICB7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElTZXJ2aWNlTWFuYWdlcjo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgZGF0YS53cml0ZVN0cmluZzE2KG5hbWUpOwotICAgICAgICBkYXRhLndyaXRlU3Ryb25nQmluZGVyKHNlcnZpY2UpOwotICAgICAgICBzdGF0dXNfdCBlcnIgPSByZW1vdGUoKS0+dHJhbnNhY3QoQUREX1NFUlZJQ0VfVFJBTlNBQ1RJT04sIGRhdGEsICZyZXBseSk7Ci0gICAgICAgIHJldHVybiBlcnIgPT0gTk9fRVJST1IgPyByZXBseS5yZWFkSW50MzIoKSA6IGVycjsKLSAgICB9Ci0KLSAgICB2aXJ0dWFsIFZlY3RvcjxTdHJpbmcxNj4gbGlzdFNlcnZpY2VzKCkKLSAgICB7Ci0gICAgICAgIFZlY3RvcjxTdHJpbmcxNj4gcmVzOwotICAgICAgICBpbnQgbiA9IDA7Ci0KLSAgICAgICAgZm9yICg7OykgewotICAgICAgICAgICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgICAgICAgICAgZGF0YS53cml0ZUludGVyZmFjZVRva2VuKElTZXJ2aWNlTWFuYWdlcjo6Z2V0SW50ZXJmYWNlRGVzY3JpcHRvcigpKTsKLSAgICAgICAgICAgIGRhdGEud3JpdGVJbnQzMihuKyspOwotICAgICAgICAgICAgc3RhdHVzX3QgZXJyID0gcmVtb3RlKCktPnRyYW5zYWN0KExJU1RfU0VSVklDRVNfVFJBTlNBQ1RJT04sIGRhdGEsICZyZXBseSk7Ci0gICAgICAgICAgICBpZiAoZXJyICE9IE5PX0VSUk9SKQotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgcmVzLmFkZChyZXBseS5yZWFkU3RyaW5nMTYoKSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIHJlczsKLSAgICB9Ci19OwotCi1JTVBMRU1FTlRfTUVUQV9JTlRFUkZBQ0UoU2VydmljZU1hbmFnZXIsICJhbmRyb2lkLm9zLklTZXJ2aWNlTWFuYWdlciIpOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNkZWZpbmUgQ0hFQ0tfSU5URVJGQUNFKGludGVyZmFjZSwgZGF0YSwgcmVwbHkpIFwKLSAgICAgICAgZG8geyBpZiAoIWRhdGEuZW5mb3JjZUludGVyZmFjZShpbnRlcmZhY2U6OmdldEludGVyZmFjZURlc2NyaXB0b3IoKSkpIHsgXAotICAgICAgICAgICAgTE9HVygiQ2FsbCBpbmNvcnJlY3RseSByb3V0ZWQgdG8gIiAjaW50ZXJmYWNlKTsgXAotICAgICAgICAgICAgcmV0dXJuIFBFUk1JU1NJT05fREVOSUVEOyBcCi0gICAgICAgIH0gfSB3aGlsZSAoMCkKLQotc3RhdHVzX3QgQm5TZXJ2aWNlTWFuYWdlcjo6b25UcmFuc2FjdCgKLSAgICB1aW50MzJfdCBjb2RlLCBjb25zdCBQYXJjZWwmIGRhdGEsIFBhcmNlbCogcmVwbHksIHVpbnQzMl90IGZsYWdzKQotewotICAgIC8vcHJpbnRmKCJTZXJ2aWNlTWFuYWdlciByZWNlaXZlZDogIik7IGRhdGEucHJpbnQoKTsKLSAgICBzd2l0Y2goY29kZSkgewotICAgICAgICBjYXNlIEdFVF9TRVJWSUNFX1RSQU5TQUNUSU9OOiB7Ci0gICAgICAgICAgICBDSEVDS19JTlRFUkZBQ0UoSVNlcnZpY2VNYW5hZ2VyLCBkYXRhLCByZXBseSk7Ci0gICAgICAgICAgICBTdHJpbmcxNiB3aGljaCA9IGRhdGEucmVhZFN0cmluZzE2KCk7Ci0gICAgICAgICAgICBzcDxJQmluZGVyPiBiID0gY29uc3RfY2FzdDxCblNlcnZpY2VNYW5hZ2VyKj4odGhpcyktPmdldFNlcnZpY2Uod2hpY2gpOwotICAgICAgICAgICAgcmVwbHktPndyaXRlU3Ryb25nQmluZGVyKGIpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIENIRUNLX1NFUlZJQ0VfVFJBTlNBQ1RJT046IHsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJU2VydmljZU1hbmFnZXIsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIFN0cmluZzE2IHdoaWNoID0gZGF0YS5yZWFkU3RyaW5nMTYoKTsKLSAgICAgICAgICAgIHNwPElCaW5kZXI+IGIgPSBjb25zdF9jYXN0PEJuU2VydmljZU1hbmFnZXIqPih0aGlzKS0+Y2hlY2tTZXJ2aWNlKHdoaWNoKTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZVN0cm9uZ0JpbmRlcihiKTsKLSAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgfSBicmVhazsKLSAgICAgICAgY2FzZSBBRERfU0VSVklDRV9UUkFOU0FDVElPTjogewotICAgICAgICAgICAgQ0hFQ0tfSU5URVJGQUNFKElTZXJ2aWNlTWFuYWdlciwgZGF0YSwgcmVwbHkpOwotICAgICAgICAgICAgU3RyaW5nMTYgd2hpY2ggPSBkYXRhLnJlYWRTdHJpbmcxNigpOwotICAgICAgICAgICAgc3A8SUJpbmRlcj4gYiA9IGRhdGEucmVhZFN0cm9uZ0JpbmRlcigpOwotICAgICAgICAgICAgc3RhdHVzX3QgZXJyID0gYWRkU2VydmljZSh3aGljaCwgYik7Ci0gICAgICAgICAgICByZXBseS0+d3JpdGVJbnQzMihlcnIpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBjYXNlIExJU1RfU0VSVklDRVNfVFJBTlNBQ1RJT046IHsKLSAgICAgICAgICAgIENIRUNLX0lOVEVSRkFDRShJU2VydmljZU1hbmFnZXIsIGRhdGEsIHJlcGx5KTsKLSAgICAgICAgICAgIFZlY3RvcjxTdHJpbmcxNj4gbGlzdCA9IGxpc3RTZXJ2aWNlcygpOwotICAgICAgICAgICAgY29uc3Qgc2l6ZV90IE4gPSBsaXN0LnNpemUoKTsKLSAgICAgICAgICAgIHJlcGx5LT53cml0ZUludDMyKE4pOwotICAgICAgICAgICAgZm9yIChzaXplX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICAgICAgICAgIHJlcGx5LT53cml0ZVN0cmluZzE2KGxpc3RbaV0pOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9IGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgcmV0dXJuIEJCaW5kZXI6Om9uVHJhbnNhY3QoY29kZSwgZGF0YSwgcmVwbHksIGZsYWdzKTsKLSAgICB9Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvSW5ldEFkZHJlc3MuY3BwIGIvbGlicy91dGlscy9JbmV0QWRkcmVzcy5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDM5YTBhNjguLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9JbmV0QWRkcmVzcy5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwyMzYgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLS8vCi0vLyBJbnRlcm5ldCBhZGRyZXNzIGNsYXNzLgotLy8KLSNpZmRlZiBIQVZFX1dJTlNPQ0sKLSMgaW5jbHVkZSA8d2luc29jazIuaD4KLSNlbHNlCi0jIGluY2x1ZGUgPHN5cy90eXBlcy5oPgotIyBpbmNsdWRlIDxzeXMvc29ja2V0Lmg+Ci0jIGluY2x1ZGUgPG5ldGluZXQvaW4uaD4KLS8vIyBpbmNsdWRlIDxhcnBhL2luZXQuaD4KLSMgaW5jbHVkZSA8bmV0ZGIuaD4KLSNlbmRpZgotCi0jaW5jbHVkZSA8dXRpbHMvaW5ldF9hZGRyZXNzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0cmluZy5oPgotI2luY2x1ZGUgPGFzc2VydC5oPgotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotCi0vKgotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKiAgICAgIEluZXRBZGRyZXNzCi0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqLwotCi0vLyBsb2NrIGZvciB0aGUgbmV4dCBjb3VwbGUgb2YgZnVuY3Rpb25zOyBjb3VsZCB0dWNrIGludG8gSW5ldEFkZHJlc3MKLXN0YXRpYyBNdXRleCogICBnR0hCTkxvY2s7Ci0KLS8qCi0gKiBMb2NrL3VubG9jayBhY2Nlc3MgdG8gdGhlIGhvc3RlbnQgc3RydWN0IHJldHVybmVkIGJ5IGdldGhvc3RieW5hbWUoKS4KLSAqLwotc3RhdGljIGlubGluZSB2b2lkIGxvY2tfZ2V0aG9zdGJ5bmFtZSh2b2lkKQotewotICAgIGlmIChnR0hCTkxvY2sgPT0gTlVMTCkKLSAgICAgICAgZ0dIQk5Mb2NrID0gbmV3IE11dGV4OwotICAgIGdHSEJOTG9jay0+bG9jaygpOwotfQotc3RhdGljIGlubGluZSB2b2lkIHVubG9ja19nZXRob3N0YnluYW1lKHZvaWQpCi17Ci0gICAgYXNzZXJ0KGdHSEJOTG9jayAhPSBOVUxMKTsKLSAgICBnR0hCTkxvY2stPnVubG9jaygpOwotfQotCi0KLS8qCi0gKiBDb25zdHJ1Y3RvciAtLSBqdXN0IGluaXQgbWVtYmVycy4gIFRoaXMgaXMgcHJpdmF0ZSBzbyB0aGF0IGNhbGxlcnMKLSAqIGFyZSByZXF1aXJlZCB0byB1c2UgZ2V0QnlOYW1lKCkuCi0gKi8KLUluZXRBZGRyZXNzOjpJbmV0QWRkcmVzcyh2b2lkKQotICAgIDogbUFkZHJlc3MoTlVMTCksIG1MZW5ndGgoLTEpLCBtTmFtZShOVUxMKQotewotfQotCi0vKgotICogRGVzdHJ1Y3RvciAtLSBmcmVlIGFkZHJlc3Mgc3RvcmFnZS4KLSAqLwotSW5ldEFkZHJlc3M6On5JbmV0QWRkcmVzcyh2b2lkKQotewotICAgIGRlbGV0ZVtdIChjaGFyKikgbUFkZHJlc3M7Ci0gICAgZGVsZXRlW10gbU5hbWU7Ci19Ci0KLS8qCi0gKiBDb3B5IGNvbnN0cnVjdG9yLgotICovCi1JbmV0QWRkcmVzczo6SW5ldEFkZHJlc3MoY29uc3QgSW5ldEFkZHJlc3MmIG9yaWcpCi17Ci0gICAgKnRoaXMgPSBvcmlnOyAgIC8vIHVzZSBhc3NpZ25tZW50IGNvZGUKLX0KLQotLyoKLSAqIEFzc2lnbm1lbnQgb3BlcmF0b3IuCi0gKi8KLUluZXRBZGRyZXNzJiBJbmV0QWRkcmVzczo6b3BlcmF0b3I9KGNvbnN0IEluZXRBZGRyZXNzJiBhZGRyKQotewotICAgIC8vIGhhbmRsZSBzZWxmLWFzc2lnbm1lbnQKLSAgICBpZiAodGhpcyA9PSAmYWRkcikKLSAgICAgICAgcmV0dXJuICp0aGlzOwotICAgIC8vIGNvcHkgbUxlbmd0aCBhbmQgbUFkZHJlc3MKLSAgICBtTGVuZ3RoID0gYWRkci5tTGVuZ3RoOwotICAgIGlmIChtTGVuZ3RoID4gMCkgewotICAgICAgICBtQWRkcmVzcyA9IG5ldyBjaGFyW21MZW5ndGhdOwotICAgICAgICBtZW1jcHkobUFkZHJlc3MsIGFkZHIubUFkZHJlc3MsIG1MZW5ndGgpOwotICAgICAgICBMT0coTE9HX0RFQlVHLCAic29ja2V0IiwKLSAgICAgICAgICAgICJIRVk6IGNvcGllZCAlZCBieXRlcyBpbiBhc3NpZ25tZW50IG9wZXJhdG9yXG4iLCBtTGVuZ3RoKTsKLSAgICB9IGVsc2UgewotICAgICAgICBtQWRkcmVzcyA9IE5VTEw7Ci0gICAgfQotICAgIC8vIGNvcHkgbU5hbWUKLSAgICBtTmFtZSA9IG5ldyBjaGFyW3N0cmxlbihhZGRyLm1OYW1lKSsxXTsKLSAgICBzdHJjcHkobU5hbWUsIGFkZHIubU5hbWUpOwotCi0gICAgcmV0dXJuICp0aGlzOwotfQotCi0vKgotICogQ3JlYXRlIGEgbmV3IG9iamVjdCBmcm9tIGEgbmFtZSBvciBhIGRvdHRlZC1udW1iZXIgSVAgbm90YXRpb24uCi0gKgotICogUmV0dXJucyBOVUxMIG9uIGZhaWx1cmUuCi0gKi8KLUluZXRBZGRyZXNzKgotSW5ldEFkZHJlc3M6OmdldEJ5TmFtZShjb25zdCBjaGFyKiBob3N0KQotewotICAgIEluZXRBZGRyZXNzKiBuZXdBZGRyID0gTlVMTDsKLSAgICBzdHJ1Y3Qgc29ja2FkZHJfaW4gYWRkcjsKLSAgICBzdHJ1Y3QgaG9zdGVudCogaGU7Ci0gICAgRHVyYXRpb25UaW1lciBob3N0VGltZXIsIGxvY2tUaW1lcjsKLQotICAgIC8vIGdldGhvc3RieW5hbWUoKSBpc24ndCByZWVudHJhbnQsIHNvIHdlIG5lZWQgdG8gbG9jayB0aGluZ3MgdW50aWwKLSAgICAvLyB3ZSBjYW4gY29weSB0aGUgZGF0YSBvdXQuCi0gICAgbG9ja1RpbWVyLnN0YXJ0KCk7Ci0gICAgbG9ja19nZXRob3N0YnluYW1lKCk7Ci0gICAgaG9zdFRpbWVyLnN0YXJ0KCk7Ci0KLSAgICBoZSA9IGdldGhvc3RieW5hbWUoaG9zdCk7Ci0gICAgaWYgKGhlID09IE5VTEwpIHsKLSAgICAgICAgTE9HKExPR19XQVJOLCAic29ja2V0IiwgIldBUk5JTkc6IGNhbm5vdCByZXNvbHZlIGhvc3QgJXNcbiIsIGhvc3QpOwotICAgICAgICB1bmxvY2tfZ2V0aG9zdGJ5bmFtZSgpOwotICAgICAgICByZXR1cm4gTlVMTDsKLSAgICB9Ci0KLSAgICBtZW1jcHkoJmFkZHIuc2luX2FkZHIsIGhlLT5oX2FkZHIsIGhlLT5oX2xlbmd0aCk7Ci0gICAgYWRkci5zaW5fZmFtaWx5ID0gaGUtPmhfYWRkcnR5cGU7Ci0gICAgYWRkci5zaW5fcG9ydCA9IDA7Ci0KLSAgICAvLyBnb3QgaXQsIHVubG9jayB1cwotICAgIGhvc3RUaW1lci5zdG9wKCk7Ci0gICAgaGUgPSBOVUxMOwotICAgIHVubG9ja19nZXRob3N0YnluYW1lKCk7Ci0KLSAgICBsb2NrVGltZXIuc3RvcCgpOwotICAgIGlmICgobG9uZykgbG9ja1RpbWVyLmR1cmF0aW9uVXNlY3MoKSA+IDEwMDAwMCkgewotICAgICAgICBsb25nIGxvY2tUaW1lID0gKGxvbmcpIGxvY2tUaW1lci5kdXJhdGlvblVzZWNzKCk7Ci0gICAgICAgIGxvbmcgaG9zdFRpbWUgPSAobG9uZykgaG9zdFRpbWVyLmR1cmF0aW9uVXNlY3MoKTsKLSAgICAgICAgTE9HKExPR19ERUJVRywgInNvY2tldCIsCi0gICAgICAgICAgICAiTG9va3VwIG9mICVzIHRvb2sgJS4zZnMgKGdldGhvc3RieW5hbWU9JS4zZnMgbG9jaz0lLjNmcylcbiIsCi0gICAgICAgICAgICBob3N0LCBsb2NrVGltZSAvIDEwMDAwMDAuMCwgaG9zdFRpbWUgLyAxMDAwMDAwLjAsCi0gICAgICAgICAgICAobG9ja1RpbWUgLSBob3N0VGltZSkgLyAxMDAwMDAwLjApOwotICAgIH0KLQotICAgIC8vIEFsbG9jIHN0b3JhZ2UgYW5kIGNvcHkgaXQgb3Zlci4KLSAgICBuZXdBZGRyID0gbmV3IEluZXRBZGRyZXNzKCk7Ci0gICAgaWYgKG5ld0FkZHIgPT0gTlVMTCkKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0KLSAgICBuZXdBZGRyLT5tTGVuZ3RoID0gc2l6ZW9mKHN0cnVjdCBzb2NrYWRkcl9pbik7Ci0gICAgbmV3QWRkci0+bUFkZHJlc3MgPSBuZXcgY2hhcltzaXplb2Yoc3RydWN0IHNvY2thZGRyX2luKV07Ci0gICAgaWYgKG5ld0FkZHItPm1BZGRyZXNzID09IE5VTEwpIHsKLSAgICAgICAgZGVsZXRlIG5ld0FkZHI7Ci0gICAgICAgIHJldHVybiBOVUxMOwotICAgIH0KLSAgICBtZW1jcHkobmV3QWRkci0+bUFkZHJlc3MsICZhZGRyLCBuZXdBZGRyLT5tTGVuZ3RoKTsKLQotICAgIC8vIEtlZXAgdGhpcyBmb3IgZGVidWcgbWVzc2FnZXMuCi0gICAgbmV3QWRkci0+bU5hbWUgPSBuZXcgY2hhcltzdHJsZW4oaG9zdCkrMV07Ci0gICAgaWYgKG5ld0FkZHItPm1OYW1lID09IE5VTEwpIHsKLSAgICAgICAgZGVsZXRlIG5ld0FkZHI7Ci0gICAgICAgIHJldHVybiBOVUxMOwotICAgIH0KLSAgICBzdHJjcHkobmV3QWRkci0+bU5hbWUsIGhvc3QpOwotCi0gICAgcmV0dXJuIG5ld0FkZHI7Ci19Ci0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBJbmV0U29ja2V0QWRkcmVzcwotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKi8KLQotLyoKLSAqIENyZWF0ZSBhbiBhZGRyZXNzIHdpdGggdGhlIGhvc3Qgd2lsZGNhcmQgKElOQUREUl9BTlkpLgotICovCi1ib29sIEluZXRTb2NrZXRBZGRyZXNzOjpjcmVhdGUoaW50IHBvcnQpCi17Ci0gICAgYXNzZXJ0KG1BZGRyZXNzID09IE5VTEwpOwotCi0gICAgbUFkZHJlc3MgPSBJbmV0QWRkcmVzczo6Z2V0QnlOYW1lKCIwLjAuMC4wIik7Ci0gICAgaWYgKG1BZGRyZXNzID09IE5VTEwpCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICBtUG9ydCA9IHBvcnQ7Ci0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLS8qCi0gKiBDcmVhdGUgYWRkcmVzcyB3aXRoIGhvc3QgYW5kIHBvcnQgc3BlY2lmaWVkLgotICovCi1ib29sIEluZXRTb2NrZXRBZGRyZXNzOjpjcmVhdGUoY29uc3QgSW5ldEFkZHJlc3MqIGFkZHIsIGludCBwb3J0KQotewotICAgIGFzc2VydChtQWRkcmVzcyA9PSBOVUxMKTsKLQotICAgIG1BZGRyZXNzID0gbmV3IEluZXRBZGRyZXNzKCphZGRyKTsgIC8vIG1ha2UgYSBjb3B5Ci0gICAgaWYgKG1BZGRyZXNzID09IE5VTEwpCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICBtUG9ydCA9IHBvcnQ7Ci0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLS8qCi0gKiBDcmVhdGUgYWRkcmVzcyB3aXRoIGhvc3QgYW5kIHBvcnQgc3BlY2lmaWVkLgotICovCi1ib29sIEluZXRTb2NrZXRBZGRyZXNzOjpjcmVhdGUoY29uc3QgY2hhciogaG9zdCwgaW50IHBvcnQpCi17Ci0gICAgYXNzZXJ0KG1BZGRyZXNzID09IE5VTEwpOwotCi0gICAgbUFkZHJlc3MgPSBJbmV0QWRkcmVzczo6Z2V0QnlOYW1lKGhvc3QpOwotICAgIGlmIChtQWRkcmVzcyA9PSBOVUxMKQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgbVBvcnQgPSBwb3J0OwotICAgIHJldHVybiB0cnVlOwotfQotCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL0xvZ1NvY2tldC5jcHAgYi9saWJzL3V0aWxzL0xvZ1NvY2tldC5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDU1YzFiOTkuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9Mb2dTb2NrZXQuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTI5ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0KLSNpZm5kZWYgSEFWRV9XSU5TT0NLCi0vLyNkZWZpbmUgU09DS0VUTE9HCi0jZW5kaWYKLQotI2lmZGVmIFNPQ0tFVExPRwotCi0jZGVmaW5lIExPR19UQUcgIlNPQ0tFVExPRyIKLQotI2luY2x1ZGUgPHN0cmluZy5oPgotI2luY2x1ZGUgPGN1dGlscy9sb2cuaD4KLSNpbmNsdWRlICJ1dGlscy9Mb2dTb2NrZXQuaCIKLSNpbmNsdWRlICJ1dGlscy9sb2dnZXIuaCIKLSNpbmNsdWRlICJjdXRpbHMvaGFzaG1hcC5oIgotCi0vLyBkZWZpbmVkIGluIC8vZGV2aWNlL2RhdGEvZXRjL2V2ZW50LWxvZy10YWdzCi0jZGVmaW5lIFNPQ0tFVF9DTE9TRV9MT0cgNTEwMDAKLQotc3RhdGljIEhhc2htYXAqIHN0YXRzTWFwID0gTlVMTDsKLQotI2RlZmluZSBMT0dfTElTVF9OVU1CRVIgNQotCi10eXBlZGVmIHN0cnVjdCBTb2NrZXRTdGF0cyB7Ci0gICAgaW50IGZkOwotICAgIHVuc2lnbmVkIGludCBzZW5kOwotICAgIHVuc2lnbmVkIGludCByZWN2OwotICAgIHVuc2lnbmVkIGludCBpcDsKLSAgICB1bnNpZ25lZCBzaG9ydCBwb3J0OwotICAgIHNob3J0IHJlYXNvbjsKLX1Tb2NrZXRTdGF0czsKLQotU29ja2V0U3RhdHMgKmdldF9zb2NrZXRfc3RhdHMoaW50IGZkKSB7Ci0gICAgaWYgKHN0YXRzTWFwID09IE5VTEwpIHsKLSAgICAgICAgc3RhdHNNYXAgPSBoYXNobWFwQ3JlYXRlKDgsICZoYXNobWFwSW50SGFzaCwgJmhhc2htYXBJbnRFcXVhbHMpOwotICAgIH0KLQotICAgIFNvY2tldFN0YXRzICpzID0gKFNvY2tldFN0YXRzKikgaGFzaG1hcEdldChzdGF0c01hcCwgJmZkKTsKLSAgICBpZiAocyA9PSBOVUxMKSB7Ci0gICAgICAgIC8vIExPR0QoImNyZWF0ZSBTb2NrZXRTdGF0cyBmb3IgZmQgJWQiLCBmZCk7Ci0gICAgICAgIHMgPSAoU29ja2V0U3RhdHMqKSBtYWxsb2Moc2l6ZW9mKFNvY2tldFN0YXRzKSk7Ci0gICAgICAgIG1lbXNldChzLCAwLCBzaXplb2YoU29ja2V0U3RhdHMpKTsKLSAgICAgICAgcy0+ZmQgPSBmZDsKLSAgICAgICAgaGFzaG1hcFB1dChzdGF0c01hcCwgJnMtPmZkLCBzKTsKLSAgICB9Ci0gICAgcmV0dXJuIHM7Ci19Ci0KLXZvaWQgbG9nX3NvY2tldF9jb25uZWN0KGludCBmZCwgdW5zaWduZWQgaW50IGlwLCB1bnNpZ25lZCBzaG9ydCBwb3J0KSB7Ci0gICAgLy8gTE9HRCgibG9nX3NvY2tldF9jb25uZWN0IGZvciBmZCAlZCBpcCAlZCBwb3J0JWQiLCBmZCwgaXAsIHBvcnQpOwotICAgIFNvY2tldFN0YXRzICpzID0gZ2V0X3NvY2tldF9zdGF0cyhmZCk7Ci0gICAgcy0+aXAgPSBpcDsKLSAgICBzLT5wb3J0ID0gcG9ydDsKLX0KLQotdm9pZCBhZGRfc2VuZF9zdGF0cyhpbnQgZmQsIGludCBzZW5kKSB7Ci0gICAgaWYgKHNlbmQgPD0wKSB7Ci0gICAgICAgIExPR0UoImFkZF9zZW5kX3N0YXRzIHNlbmQgJWQiLCBzZW5kKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBTb2NrZXRTdGF0cyAqcyA9IGdldF9zb2NrZXRfc3RhdHMoZmQpOwotICAgIHMtPnNlbmQgKz0gc2VuZDsKLSAgICAvLyBMT0dEKCJhZGRfc2VuZF9zdGF0cyBmb3IgZmQgJWQgaXAgJWQgcG9ydCVkIiwgZmQsIHMtPmlwLCBzLT5wb3J0KTsKLX0KLQotdm9pZCBhZGRfcmVjdl9zdGF0cyhpbnQgZmQsIGludCByZWN2KSB7Ci0gICAgaWYgKHJlY3YgPD0wKSB7Ci0gICAgICAgIExPR0UoImFkZF9yZWN2X3N0YXRzIHJlY3YgJWQiLCByZWN2KTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBTb2NrZXRTdGF0cyAqcyA9IGdldF9zb2NrZXRfc3RhdHMoZmQpOwotICAgIHMtPnJlY3YgKz0gcmVjdjsKLSAgICAvLyBMT0dEKCJhZGRfcmVjdl9zdGF0cyBmb3IgZmQgJWQgaXAgJWQgcG9ydCVkIiwgZmQsIHMtPmlwLCBzLT5wb3J0KTsKLX0KLQotY2hhciogcHV0X2ludChjaGFyKiBidWYsIGludCB2YWx1ZSkgewotICAgICpidWYgPSBFVkVOVF9UWVBFX0lOVDsKLSAgICBidWYrKzsKLSAgICBtZW1jcHkoYnVmLCAmdmFsdWUsIHNpemVvZihpbnQpKTsKLSAgICByZXR1cm4gYnVmICsgc2l6ZW9mKGludCk7Ci19Ci0KLXZvaWQgbG9nX3NvY2tldF9jbG9zZShpbnQgZmQsIHNob3J0IHJlYXNvbikgewotICAgIGlmIChzdGF0c01hcCkgewotICAgICAgICBTb2NrZXRTdGF0cyAqcyA9IChTb2NrZXRTdGF0cyopIGhhc2htYXBHZXQoc3RhdHNNYXAsICZmZCk7Ci0gICAgICAgIGlmIChzICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGlmIChzLT5zZW5kICE9IDAgfHwgcy0+cmVjdiAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgcy0+cmVhc29uID0gcmVhc29uOwotICAgICAgICAgICAgICAgIC8vIDUgaW50ICsgbGlzdCB0eXBlIG5lZWQgMiBieXRlcwotICAgICAgICAgICAgICAgIGNoYXIgYnVmW0xPR19MSVNUX05VTUJFUiAqIDUgKyAyXTsKLSAgICAgICAgICAgICAgICBidWZbMF0gPSBFVkVOVF9UWVBFX0xJU1Q7Ci0gICAgICAgICAgICAgICAgYnVmWzFdID0gTE9HX0xJU1RfTlVNQkVSOwotICAgICAgICAgICAgICAgIGNoYXIqIHdyaXRlUG9zID0gYnVmICsgMjsKLSAgICAgICAgICAgICAgICB3cml0ZVBvcyA9IHB1dF9pbnQod3JpdGVQb3MsIHMtPnNlbmQpOwotICAgICAgICAgICAgICAgIHdyaXRlUG9zID0gcHV0X2ludCh3cml0ZVBvcywgcy0+cmVjdik7Ci0gICAgICAgICAgICAgICAgd3JpdGVQb3MgPSBwdXRfaW50KHdyaXRlUG9zLCBzLT5pcCk7Ci0gICAgICAgICAgICAgICAgd3JpdGVQb3MgPSBwdXRfaW50KHdyaXRlUG9zLCBzLT5wb3J0KTsKLSAgICAgICAgICAgICAgICB3cml0ZVBvcyA9IHB1dF9pbnQod3JpdGVQb3MsIHMtPnJlYXNvbik7Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgYW5kcm9pZF9iV3JpdGVMb2coU09DS0VUX0NMT1NFX0xPRywgYnVmLCBzaXplb2YoYnVmKSk7Ci0gICAgICAgICAgICAgICAgLy8gTE9HRCgic2VuZCAlZCByZWN2ICVkIHJlYXNvbiAlZCIsIHMtPnNlbmQsIHMtPnJlY3YsIHMtPnJlYXNvbik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBoYXNobWFwUmVtb3ZlKHN0YXRzTWFwLCAmcy0+ZmQpOwotICAgICAgICAgICAgZnJlZShzKTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotI2Vsc2UKLXZvaWQgYWRkX3NlbmRfc3RhdHMoaW50IGZkLCBpbnQgc2VuZCkge30gCi12b2lkIGFkZF9yZWN2X3N0YXRzKGludCBmZCwgaW50IHJlY3YpIHt9Ci12b2lkIGxvZ19zb2NrZXRfY2xvc2UoaW50IGZkLCBzaG9ydCByZWFzb24pIHt9Ci12b2lkIGxvZ19zb2NrZXRfY29ubmVjdChpbnQgZmQsIHVuc2lnbmVkIGludCBpcCwgdW5zaWduZWQgc2hvcnQgcG9ydCkge30KLSNlbmRpZgpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9NT0RVTEVfTElDRU5TRV9BUEFDSEUyIGIvbGlicy91dGlscy9NT0RVTEVfTElDRU5TRV9BUEFDSEUyCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlNjlkZTI5Li4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvTU9EVUxFX0xJQ0VOU0VfQVBBQ0hFMgorKysgL2Rldi9udWxsCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL01lbW9yeUJhc2UuY3BwIGIvbGlicy91dGlscy9NZW1vcnlCYXNlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZjI1ZTExYy4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL01lbW9yeUJhc2UuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNDYgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGludC5oPgotCi0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5QmFzZS5oPgotCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLU1lbW9yeUJhc2U6Ok1lbW9yeUJhc2UoY29uc3Qgc3A8SU1lbW9yeUhlYXA+JiBoZWFwLAotICAgICAgICBzc2l6ZV90IG9mZnNldCwgc2l6ZV90IHNpemUpCi0gICAgOiBtU2l6ZShzaXplKSwgbU9mZnNldChvZmZzZXQpLCBtSGVhcChoZWFwKQotewotfQotCi1zcDxJTWVtb3J5SGVhcD4gTWVtb3J5QmFzZTo6Z2V0TWVtb3J5KHNzaXplX3QqIG9mZnNldCwgc2l6ZV90KiBzaXplKSBjb25zdAotewotICAgIGlmIChvZmZzZXQpICpvZmZzZXQgPSBtT2Zmc2V0OwotICAgIGlmIChzaXplKSAgICpzaXplID0gbVNpemU7Ci0gICAgcmV0dXJuIG1IZWFwOwotfQotCi1NZW1vcnlCYXNlOjp+TWVtb3J5QmFzZSgpCi17Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvTWVtb3J5RGVhbGVyLmNwcCBiL2xpYnMvdXRpbHMvTWVtb3J5RGVhbGVyLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggY2Y4MjAxYi4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL01lbW9yeURlYWxlci5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw0MDkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNyBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNkZWZpbmUgTE9HX1RBRyAiTWVtb3J5RGVhbGVyIgotCi0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5RGVhbGVyLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL1NvcnRlZFZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLSNpbmNsdWRlIDx1dGlscy9NZW1vcnlCYXNlLmg+Ci0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPGZjbnRsLmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLQotI2luY2x1ZGUgPHN5cy9zdGF0Lmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8c3lzL21tYW4uaD4KLSNpbmNsdWRlIDxzeXMvZmlsZS5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBTaW1wbGVNZW1vcnkgOiBwdWJsaWMgTWVtb3J5QmFzZSB7Ci1wdWJsaWM6Ci0gICAgU2ltcGxlTWVtb3J5KGNvbnN0IHNwPElNZW1vcnlIZWFwPiYgaGVhcCwgc3NpemVfdCBvZmZzZXQsIHNpemVfdCBzaXplKTsKLSAgICB2aXJ0dWFsIH5TaW1wbGVNZW1vcnkoKTsKLX07Ci0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1NZW1vcnlEZWFsZXI6OkFsbG9jYXRpb246OkFsbG9jYXRpb24oCi0gICAgICAgIGNvbnN0IHNwPE1lbW9yeURlYWxlcj4mIGRlYWxlciwgc3NpemVfdCBvZmZzZXQsIHNpemVfdCBzaXplLAotICAgICAgICBjb25zdCBzcDxJTWVtb3J5PiYgbWVtb3J5KQotICAgIDogbURlYWxlcihkZWFsZXIpLCBtT2Zmc2V0KG9mZnNldCksIG1TaXplKHNpemUpLCBtTWVtb3J5KG1lbW9yeSkgCi17Ci19Ci0KLU1lbW9yeURlYWxlcjo6QWxsb2NhdGlvbjo6fkFsbG9jYXRpb24oKQotewotICAgIGlmIChtU2l6ZSkgewotICAgICAgICAvKiBOT1RFOiBpdCdzIFZFUlkgaW1wb3J0YW50IHRvIG5vdCBmcmVlIGFsbG9jYXRpb25zIG9mIHNpemUgMCBiZWNhdXNlCi0gICAgICAgICAqIHRoZXkncmUgc3BlY2lhbCBhcyB0aGV5IGRvbid0IGhhdmUgYW55IHJlY29yZCBpbiB0aGUgYWxsb2NhdG9yCi0gICAgICAgICAqIGFuZCBjb3VsZCBhbGlhcyBzb21lIHJlYWwgYWxsb2NhdGlvbiAodGhlaXIgb2Zmc2V0IGlzIHplcm8pLiAqLwotICAgICAgICBtRGVhbGVyLT5kZWFsbG9jYXRlKG1PZmZzZXQpOwotICAgIH0KLX0KLQotc3A8SU1lbW9yeUhlYXA+IE1lbW9yeURlYWxlcjo6QWxsb2NhdGlvbjo6Z2V0TWVtb3J5KAotICAgIHNzaXplX3QqIG9mZnNldCwgc2l6ZV90KiBzaXplKSBjb25zdAotewotICAgIHJldHVybiBtTWVtb3J5LT5nZXRNZW1vcnkob2Zmc2V0LCBzaXplKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1NZW1vcnlEZWFsZXI6Ok1lbW9yeURlYWxlcihzaXplX3Qgc2l6ZSwgdWludDMyX3QgZmxhZ3MsIGNvbnN0IGNoYXIqIG5hbWUpCi0gICAgOiBtSGVhcChuZXcgU2hhcmVkSGVhcChzaXplLCBmbGFncywgbmFtZSkpLAotICAgIG1BbGxvY2F0b3IobmV3IFNpbXBsZUJlc3RGaXRBbGxvY2F0b3Ioc2l6ZSkpCi17ICAgIAotfQotCi1NZW1vcnlEZWFsZXI6Ok1lbW9yeURlYWxlcihjb25zdCBzcDxIZWFwSW50ZXJmYWNlPiYgaGVhcCkKLSAgICA6IG1IZWFwKGhlYXApLAotICAgIG1BbGxvY2F0b3IobmV3IFNpbXBsZUJlc3RGaXRBbGxvY2F0b3IoaGVhcC0+dmlydHVhbFNpemUoKSkpCi17Ci19Ci0KLU1lbW9yeURlYWxlcjo6TWVtb3J5RGVhbGVyKCBjb25zdCBzcDxIZWFwSW50ZXJmYWNlPiYgaGVhcCwKLSAgICAgICAgY29uc3Qgc3A8QWxsb2NhdG9ySW50ZXJmYWNlPiYgYWxsb2NhdG9yKQotICAgIDogbUhlYXAoaGVhcCksIG1BbGxvY2F0b3IoYWxsb2NhdG9yKQotewotfQotCi1NZW1vcnlEZWFsZXI6On5NZW1vcnlEZWFsZXIoKQotewotfQotCi1zcDxJTWVtb3J5PiBNZW1vcnlEZWFsZXI6OmFsbG9jYXRlKHNpemVfdCBzaXplLCB1aW50MzJfdCBmbGFncykKLXsKLSAgICBzcDxJTWVtb3J5PiBtZW1vcnk7Ci0gICAgY29uc3Qgc3NpemVfdCBvZmZzZXQgPSBhbGxvY2F0b3IoKS0+YWxsb2NhdGUoc2l6ZSwgZmxhZ3MpOwotICAgIGlmIChvZmZzZXQgPj0gMCkgewotICAgICAgICBzcDxJTWVtb3J5PiBuZXdfbWVtb3J5ID0gaGVhcCgpLT5tYXBNZW1vcnkob2Zmc2V0LCBzaXplKTsKLSAgICAgICAgaWYgKG5ld19tZW1vcnkgIT0gMCkgewotICAgICAgICAgICAgbWVtb3J5ID0gbmV3IEFsbG9jYXRpb24odGhpcywgb2Zmc2V0LCBzaXplLCBuZXdfbWVtb3J5KTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIExPR0UoImNvdWxkbid0IG1hcCBbJTh4LCAlZF0iLCBvZmZzZXQsIHNpemUpOwotICAgICAgICAgICAgaWYgKHNpemUpIHsKLSAgICAgICAgICAgICAgICAvKiBOT1RFOiBpdCdzIFZFUlkgaW1wb3J0YW50IHRvIG5vdCBmcmVlIGFsbG9jYXRpb25zIG9mIHNpemUgMAotICAgICAgICAgICAgICAgICAqIGJlY2F1c2UgdGhleSdyZSBzcGVjaWFsIGFzIHRoZXkgZG9uJ3QgaGF2ZSBhbnkgcmVjb3JkIGluIHRoZSAKLSAgICAgICAgICAgICAgICAgKiBhbGxvY2F0b3IgYW5kIGNvdWxkIGFsaWFzIHNvbWUgcmVhbCBhbGxvY2F0aW9uIAotICAgICAgICAgICAgICAgICAqICh0aGVpciBvZmZzZXQgaXMgemVybykuICovCi0gICAgICAgICAgICAgICAgYWxsb2NhdG9yKCktPmRlYWxsb2NhdGUob2Zmc2V0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSAgICAgICAgCi0gICAgfQotICAgIHJldHVybiBtZW1vcnk7Ci19Ci0KLXZvaWQgTWVtb3J5RGVhbGVyOjpkZWFsbG9jYXRlKHNpemVfdCBvZmZzZXQpCi17Ci0gICAgYWxsb2NhdG9yKCktPmRlYWxsb2NhdGUob2Zmc2V0KTsKLX0KLQotdm9pZCBNZW1vcnlEZWFsZXI6OmR1bXAoY29uc3QgY2hhciogd2hhdCwgdWludDMyX3QgZmxhZ3MpIGNvbnN0Ci17Ci0gICAgYWxsb2NhdG9yKCktPmR1bXAod2hhdCwgZmxhZ3MpOwotfQotCi1jb25zdCBzcDxIZWFwSW50ZXJmYWNlPiYgTWVtb3J5RGVhbGVyOjpoZWFwKCkgY29uc3QgewotICAgIHJldHVybiBtSGVhcDsKLX0KLQotY29uc3Qgc3A8QWxsb2NhdG9ySW50ZXJmYWNlPiYgTWVtb3J5RGVhbGVyOjphbGxvY2F0b3IoKSBjb25zdCB7Ci0gICAgcmV0dXJuIG1BbGxvY2F0b3I7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLy8gYWxpZ24gYWxsIHRoZSBtZW1vcnkgYmxvY2tzIG9uIGEgY2FjaGUtbGluZSBib3VuZGFyeQotY29uc3QgaW50IFNpbXBsZUJlc3RGaXRBbGxvY2F0b3I6OmtNZW1vcnlBbGlnbiA9IDMyOwotCi1TaW1wbGVCZXN0Rml0QWxsb2NhdG9yOjpTaW1wbGVCZXN0Rml0QWxsb2NhdG9yKHNpemVfdCBzaXplKQotewotICAgIHNpemVfdCBwYWdlc2l6ZSA9IGdldHBhZ2VzaXplKCk7Ci0gICAgbUhlYXBTaXplID0gKChzaXplICsgcGFnZXNpemUtMSkgJiB+KHBhZ2VzaXplLTEpKTsKLQotICAgIGNodW5rX3QqIG5vZGUgPSBuZXcgY2h1bmtfdCgwLCBtSGVhcFNpemUgLyBrTWVtb3J5QWxpZ24pOwotICAgIG1MaXN0Lmluc2VydEhlYWQobm9kZSk7Ci19Ci0KLVNpbXBsZUJlc3RGaXRBbGxvY2F0b3I6On5TaW1wbGVCZXN0Rml0QWxsb2NhdG9yKCkKLXsKLSAgICB3aGlsZSghbUxpc3QuaXNFbXB0eSgpKSB7Ci0gICAgICAgIGRlbGV0ZSBtTGlzdC5yZW1vdmUobUxpc3QuaGVhZCgpKTsKLSAgICB9Ci19Ci0KLXNpemVfdCBTaW1wbGVCZXN0Rml0QWxsb2NhdG9yOjpzaXplKCkgY29uc3QKLXsKLSAgICByZXR1cm4gbUhlYXBTaXplOwotfQotCi1zaXplX3QgU2ltcGxlQmVzdEZpdEFsbG9jYXRvcjo6YWxsb2NhdGUoc2l6ZV90IHNpemUsIHVpbnQzMl90IGZsYWdzKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgc3NpemVfdCBvZmZzZXQgPSBhbGxvYyhzaXplLCBmbGFncyk7Ci0gICAgcmV0dXJuIG9mZnNldDsKLX0KLQotc3RhdHVzX3QgU2ltcGxlQmVzdEZpdEFsbG9jYXRvcjo6ZGVhbGxvY2F0ZShzaXplX3Qgb2Zmc2V0KQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgY2h1bmtfdCBjb25zdCAqIGNvbnN0IGZyZWVkID0gZGVhbGxvYyhvZmZzZXQpOwotICAgIGlmIChmcmVlZCkgewotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotICAgIHJldHVybiBOQU1FX05PVF9GT1VORDsKLX0KLQotc3NpemVfdCBTaW1wbGVCZXN0Rml0QWxsb2NhdG9yOjphbGxvYyhzaXplX3Qgc2l6ZSwgdWludDMyX3QgZmxhZ3MpCi17Ci0gICAgaWYgKHNpemUgPT0gMCkgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0gICAgc2l6ZSA9IChzaXplICsga01lbW9yeUFsaWduLTEpIC8ga01lbW9yeUFsaWduOwotICAgIGNodW5rX3QqIGZyZWVfY2h1bmsgPSAwOwotICAgIGNodW5rX3QqIGN1ciA9IG1MaXN0LmhlYWQoKTsKLQotICAgIHNpemVfdCBwYWdlc2l6ZSA9IGdldHBhZ2VzaXplKCk7Ci0gICAgd2hpbGUgKGN1cikgewotICAgICAgICBpbnQgZXh0cmEgPSAwOwotICAgICAgICBpZiAoZmxhZ3MgJiBQQUdFX0FMSUdORUQpCi0gICAgICAgICAgICBleHRyYSA9ICggLWN1ci0+c3RhcnQgJiAoKHBhZ2VzaXplL2tNZW1vcnlBbGlnbiktMSkgKSA7Ci0KLSAgICAgICAgLy8gYmVzdCBmaXQKLSAgICAgICAgaWYgKGN1ci0+ZnJlZSAmJiAoY3VyLT5zaXplID49IChzaXplK2V4dHJhKSkpIHsKLSAgICAgICAgICAgIGlmICgoIWZyZWVfY2h1bmspIHx8IChjdXItPnNpemUgPCBmcmVlX2NodW5rLT5zaXplKSkgewotICAgICAgICAgICAgICAgIGZyZWVfY2h1bmsgPSBjdXI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoY3VyLT5zaXplID09IHNpemUpIHsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBjdXIgPSBjdXItPm5leHQ7Ci0gICAgfQotCi0gICAgaWYgKGZyZWVfY2h1bmspIHsKLSAgICAgICAgY29uc3Qgc2l6ZV90IGZyZWVfc2l6ZSA9IGZyZWVfY2h1bmstPnNpemU7Ci0gICAgICAgIGZyZWVfY2h1bmstPmZyZWUgPSAwOwotICAgICAgICBmcmVlX2NodW5rLT5zaXplID0gc2l6ZTsKLSAgICAgICAgaWYgKGZyZWVfc2l6ZSA+IHNpemUpIHsKLSAgICAgICAgICAgIGludCBleHRyYSA9IDA7Ci0gICAgICAgICAgICBpZiAoZmxhZ3MgJiBQQUdFX0FMSUdORUQpCi0gICAgICAgICAgICAgICAgZXh0cmEgPSAoIC1mcmVlX2NodW5rLT5zdGFydCAmICgocGFnZXNpemUva01lbW9yeUFsaWduKS0xKSApIDsKLSAgICAgICAgICAgIGlmIChleHRyYSkgewotICAgICAgICAgICAgICAgIGNodW5rX3QqIHNwbGl0ID0gbmV3IGNodW5rX3QoZnJlZV9jaHVuay0+c3RhcnQsIGV4dHJhKTsKLSAgICAgICAgICAgICAgICBmcmVlX2NodW5rLT5zdGFydCArPSBleHRyYTsKLSAgICAgICAgICAgICAgICBtTGlzdC5pbnNlcnRCZWZvcmUoZnJlZV9jaHVuaywgc3BsaXQpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBMT0dFX0lGKChmbGFncyZQQUdFX0FMSUdORUQpICYmIAotICAgICAgICAgICAgICAgICAgICAoKGZyZWVfY2h1bmstPnN0YXJ0KmtNZW1vcnlBbGlnbikmKHBhZ2VzaXplLTEpKSwKLSAgICAgICAgICAgICAgICAgICAgIlBBR0VfQUxJR05FRCByZXF1ZXN0ZWQsIGJ1dCBwYWdlIGlzIG5vdCBhbGlnbmVkISEhIik7Ci0KLSAgICAgICAgICAgIGNvbnN0IHNzaXplX3QgdGFpbF9mcmVlID0gZnJlZV9zaXplIC0gKHNpemUrZXh0cmEpOwotICAgICAgICAgICAgaWYgKHRhaWxfZnJlZSA+IDApIHsKLSAgICAgICAgICAgICAgICBjaHVua190KiBzcGxpdCA9IG5ldyBjaHVua190KAotICAgICAgICAgICAgICAgICAgICAgICAgZnJlZV9jaHVuay0+c3RhcnQgKyBmcmVlX2NodW5rLT5zaXplLCB0YWlsX2ZyZWUpOwotICAgICAgICAgICAgICAgIG1MaXN0Lmluc2VydEFmdGVyKGZyZWVfY2h1bmssIHNwbGl0KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gKGZyZWVfY2h1bmstPnN0YXJ0KSprTWVtb3J5QWxpZ247Ci0gICAgfQotICAgIHJldHVybiBOT19NRU1PUlk7Ci19Ci0KLVNpbXBsZUJlc3RGaXRBbGxvY2F0b3I6OmNodW5rX3QqIFNpbXBsZUJlc3RGaXRBbGxvY2F0b3I6OmRlYWxsb2Moc2l6ZV90IHN0YXJ0KQotewotICAgIHN0YXJ0ID0gc3RhcnQgLyBrTWVtb3J5QWxpZ247Ci0gICAgY2h1bmtfdCogY3VyID0gbUxpc3QuaGVhZCgpOwotICAgIHdoaWxlIChjdXIpIHsKLSAgICAgICAgaWYgKGN1ci0+c3RhcnQgPT0gc3RhcnQpIHsKLSAgICAgICAgICAgIExPR19GQVRBTF9JRihjdXItPmZyZWUsCi0gICAgICAgICAgICAgICAgImJsb2NrIGF0IG9mZnNldCAweCUwOGxYIG9mIHNpemUgMHglMDhsWCBhbHJlYWR5IGZyZWVkIiwKLSAgICAgICAgICAgICAgICBjdXItPnN0YXJ0KmtNZW1vcnlBbGlnbiwgY3VyLT5zaXplKmtNZW1vcnlBbGlnbik7Ci0KLSAgICAgICAgICAgIC8vIG1lcmdlIGZyZWVkIGJsb2NrcyB0b2dldGhlcgotICAgICAgICAgICAgY2h1bmtfdCogZnJlZWQgPSBjdXI7Ci0gICAgICAgICAgICBjdXItPmZyZWUgPSAxOwotICAgICAgICAgICAgZG8gewotICAgICAgICAgICAgICAgIGNodW5rX3QqIGNvbnN0IHAgPSBjdXItPnByZXY7Ci0gICAgICAgICAgICAgICAgY2h1bmtfdCogY29uc3QgbiA9IGN1ci0+bmV4dDsKLSAgICAgICAgICAgICAgICBpZiAocCAmJiAocC0+ZnJlZSB8fCAhY3VyLT5zaXplKSkgewotICAgICAgICAgICAgICAgICAgICBmcmVlZCA9IHA7Ci0gICAgICAgICAgICAgICAgICAgIHAtPnNpemUgKz0gY3VyLT5zaXplOwotICAgICAgICAgICAgICAgICAgICBtTGlzdC5yZW1vdmUoY3VyKTsKLSAgICAgICAgICAgICAgICAgICAgZGVsZXRlIGN1cjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgY3VyID0gbjsKLSAgICAgICAgICAgIH0gd2hpbGUgKGN1ciAmJiBjdXItPmZyZWUpOwotCi0gICAgICAgICAgICAjaWZuZGVmIE5ERUJVRwotICAgICAgICAgICAgICAgIGlmICghZnJlZWQtPmZyZWUpIHsKLSAgICAgICAgICAgICAgICAgICAgZHVtcF9sKCJkZWFsbG9jICghZnJlZWQtPmZyZWUpIik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgI2VuZGlmCi0gICAgICAgICAgICBMT0dfRkFUQUxfSUYoIWZyZWVkLT5mcmVlLAotICAgICAgICAgICAgICAgICJmcmVlZCBibG9jayBhdCBvZmZzZXQgMHglMDhsWCBvZiBzaXplIDB4JTA4bFggaXMgbm90IGZyZWUhIiwKLSAgICAgICAgICAgICAgICBmcmVlZC0+c3RhcnQgKiBrTWVtb3J5QWxpZ24sIGZyZWVkLT5zaXplICoga01lbW9yeUFsaWduKTsKLQotICAgICAgICAgICAgcmV0dXJuIGZyZWVkOwotICAgICAgICB9Ci0gICAgICAgIGN1ciA9IGN1ci0+bmV4dDsKLSAgICB9Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXZvaWQgU2ltcGxlQmVzdEZpdEFsbG9jYXRvcjo6ZHVtcChjb25zdCBjaGFyKiB3aGF0LCB1aW50MzJfdCBmbGFncykgY29uc3QKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIGR1bXBfbCh3aGF0LCBmbGFncyk7Ci19Ci0KLXZvaWQgU2ltcGxlQmVzdEZpdEFsbG9jYXRvcjo6ZHVtcF9sKGNvbnN0IGNoYXIqIHdoYXQsIHVpbnQzMl90IGZsYWdzKSBjb25zdAotewotICAgIFN0cmluZzggcmVzdWx0OwotICAgIGR1bXBfbChyZXN1bHQsIHdoYXQsIGZsYWdzKTsKLSAgICBMT0dEKCIlcyIsIHJlc3VsdC5zdHJpbmcoKSk7Ci19Ci0KLXZvaWQgU2ltcGxlQmVzdEZpdEFsbG9jYXRvcjo6ZHVtcChTdHJpbmc4JiByZXN1bHQsCi0gICAgICAgIGNvbnN0IGNoYXIqIHdoYXQsIHVpbnQzMl90IGZsYWdzKSBjb25zdAotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgZHVtcF9sKHJlc3VsdCwgd2hhdCwgZmxhZ3MpOwotfQotCi12b2lkIFNpbXBsZUJlc3RGaXRBbGxvY2F0b3I6OmR1bXBfbChTdHJpbmc4JiByZXN1bHQsCi0gICAgICAgIGNvbnN0IGNoYXIqIHdoYXQsIHVpbnQzMl90IGZsYWdzKSBjb25zdAotewotICAgIHNpemVfdCBzaXplID0gMDsKLSAgICBpbnQzMl90IGkgPSAwOwotICAgIGNodW5rX3QgY29uc3QqIGN1ciA9IG1MaXN0LmhlYWQoKTsKLSAgICAKLSAgICBjb25zdCBzaXplX3QgU0laRSA9IDI1NjsKLSAgICBjaGFyIGJ1ZmZlcltTSVpFXTsKLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICIgICVzICglcCwgc2l6ZT0ldSlcbiIsCi0gICAgICAgICAgICB3aGF0LCB0aGlzLCAodW5zaWduZWQgaW50KW1IZWFwU2l6ZSk7Ci0gICAgCi0gICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotICAgICAgICAgICAgCi0gICAgd2hpbGUgKGN1cikgewotICAgICAgICBjb25zdCBjaGFyKiBlcnJzW10gPSB7IiIsICJ8IGxpbmsgYm9ndXMgTlAiLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICJ8IGxpbmsgYm9ndXMgUE4iLCAifCBsaW5rIGJvZ3VzIE5QK1BOIiB9OwotICAgICAgICBpbnQgbnAgPSAoKGN1ci0+bmV4dCkgJiYgY3VyLT5uZXh0LT5wcmV2ICE9IGN1cikgPyAxIDogMDsKLSAgICAgICAgaW50IHBuID0gKChjdXItPnByZXYpICYmIGN1ci0+cHJldi0+bmV4dCAhPSBjdXIpID8gMiA6IDA7Ci0KLSAgICAgICAgc25wcmludGYoYnVmZmVyLCBTSVpFLCAiICAlM3U6ICUwOHggfCAweCUwOFggfCAweCUwOFggfCAlcyAlc1xuIiwKLSAgICAgICAgICAgIGksIGludChjdXIpLCBpbnQoY3VyLT5zdGFydCprTWVtb3J5QWxpZ24pLAotICAgICAgICAgICAgaW50KGN1ci0+c2l6ZSprTWVtb3J5QWxpZ24pLAotICAgICAgICAgICAgICAgICAgICBpbnQoY3VyLT5mcmVlKSA/ICJGIiA6ICJBIiwKLSAgICAgICAgICAgICAgICAgICAgZXJyc1tucHxwbl0pOwotICAgICAgICAKLSAgICAgICAgcmVzdWx0LmFwcGVuZChidWZmZXIpOwotCi0gICAgICAgIGlmICghY3VyLT5mcmVlKQotICAgICAgICAgICAgc2l6ZSArPSBjdXItPnNpemUqa01lbW9yeUFsaWduOwotCi0gICAgICAgIGkrKzsKLSAgICAgICAgY3VyID0gY3VyLT5uZXh0OwotICAgIH0KLSAgICBzbnByaW50ZihidWZmZXIsIFNJWkUsICIgIHNpemUgYWxsb2NhdGVkOiAldSAoJXUgS0IpXG4iLCBpbnQoc2l6ZSksIGludChzaXplLzEwMjQpKTsKLSAgICByZXN1bHQuYXBwZW5kKGJ1ZmZlcik7Ci19Ci0gICAgICAgIAotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0KLVNoYXJlZEhlYXA6OlNoYXJlZEhlYXAoc2l6ZV90IHNpemUsIHVpbnQzMl90IGZsYWdzLCBjaGFyIGNvbnN0ICogbmFtZSkKLSAgICA6IE1lbW9yeUhlYXBCYXNlKHNpemUsIGZsYWdzLCBuYW1lKQotewotfQotCi1TaGFyZWRIZWFwOjp+U2hhcmVkSGVhcCgpCi17Ci19Ci0KLXNwPElNZW1vcnk+IFNoYXJlZEhlYXA6Om1hcE1lbW9yeShzaXplX3Qgb2Zmc2V0LCBzaXplX3Qgc2l6ZSkKLXsKLSAgICByZXR1cm4gbmV3IFNpbXBsZU1lbW9yeSh0aGlzLCBvZmZzZXQsIHNpemUpOwotfQotIAotCi1TaW1wbGVNZW1vcnk6OlNpbXBsZU1lbW9yeShjb25zdCBzcDxJTWVtb3J5SGVhcD4mIGhlYXAsCi0gICAgICAgIHNzaXplX3Qgb2Zmc2V0LCBzaXplX3Qgc2l6ZSkKLSAgICA6IE1lbW9yeUJhc2UoaGVhcCwgb2Zmc2V0LCBzaXplKQotewotI2lmbmRlZiBOREVCVUcKLSAgICB2b2lkKiBjb25zdCBzdGFydF9wdHIgPSAodm9pZCopKGludHB0cl90KGhlYXAtPmJhc2UoKSkgKyBvZmZzZXQpOwotICAgIG1lbXNldChzdGFydF9wdHIsIDB4ZGEsIHNpemUpOwotI2VuZGlmCi19Ci0KLVNpbXBsZU1lbW9yeTo6flNpbXBsZU1lbW9yeSgpCi17Ci0gICAgc2l6ZV90IGZyZWVkT2Zmc2V0ID0gZ2V0T2Zmc2V0KCk7Ci0gICAgc2l6ZV90IGZyZWVkU2l6ZSAgID0gZ2V0U2l6ZSgpOwotCi0gICAgLy8ga2VlcCB0aGUgc2l6ZSB0byB1bm1hcCBpbiBleGNlc3MKLSAgICBzaXplX3QgcGFnZXNpemUgPSBnZXRwYWdlc2l6ZSgpOwotICAgIHNpemVfdCBzdGFydCA9IGZyZWVkT2Zmc2V0OwotICAgIHNpemVfdCBlbmQgPSBzdGFydCArIGZyZWVkU2l6ZTsKLSAgICBzdGFydCAmPSB+KHBhZ2VzaXplLTEpOwotICAgIGVuZCA9IChlbmQgKyBwYWdlc2l6ZS0xKSAmIH4ocGFnZXNpemUtMSk7Ci0KLSAgICAvLyBnaXZlIGJhY2sgdG8gdGhlIGtlcm5lbCB0aGUgcGFnZXMgd2UgZG9uJ3QgbmVlZAotICAgIHNpemVfdCBmcmVlX3N0YXJ0ID0gZnJlZWRPZmZzZXQ7Ci0gICAgc2l6ZV90IGZyZWVfZW5kID0gZnJlZV9zdGFydCArIGZyZWVkU2l6ZTsKLSAgICBpZiAoc3RhcnQgPCBmcmVlX3N0YXJ0KQotICAgICAgICBzdGFydCA9IGZyZWVfc3RhcnQ7Ci0gICAgaWYgKGVuZCA+IGZyZWVfZW5kKQotICAgICAgICBlbmQgPSBmcmVlX2VuZDsKLSAgICBzdGFydCA9IChzdGFydCArIHBhZ2VzaXplLTEpICYgfihwYWdlc2l6ZS0xKTsKLSAgICBlbmQgJj0gfihwYWdlc2l6ZS0xKTsgICAgCi0KLSAgICBpZiAoc3RhcnQgPCBlbmQpIHsKLSAgICAgICAgdm9pZCogY29uc3Qgc3RhcnRfcHRyID0gKHZvaWQqKShpbnRwdHJfdChnZXRIZWFwKCktPmJhc2UoKSkgKyBzdGFydCk7Ci0gICAgICAgIHNpemVfdCBzaXplID0gZW5kLXN0YXJ0OwotCi0jaWZuZGVmIE5ERUJVRwotICAgICAgICBtZW1zZXQoc3RhcnRfcHRyLCAweGRmLCBzaXplKTsKLSNlbmRpZgotCi0gICAgICAgIC8vIE1BRFZfUkVNT1ZFIGlzIG5vdCBkZWZpbmVkIG9uIERhcHBlciBiYXNlZCBHb29idW50dSAKLSNpZmRlZiBNQURWX1JFTU9WRSAKLSAgICAgICAgaWYgKHNpemUpIHsKLSAgICAgICAgICAgIGludCBlcnIgPSBtYWR2aXNlKHN0YXJ0X3B0ciwgc2l6ZSwgTUFEVl9SRU1PVkUpOwotICAgICAgICAgICAgTE9HV19JRihlcnIsICJtYWR2aXNlKCVwLCAldSwgTUFEVl9SRU1PVkUpIHJldHVybmVkICVzIiwKLSAgICAgICAgICAgICAgICAgICAgc3RhcnRfcHRyLCBzaXplLCBlcnI8MCA/IHN0cmVycm9yKGVycm5vKSA6ICJPayIpOwotICAgICAgICB9Ci0jZW5kaWYKLSAgICB9Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL01lbW9yeUhlYXBCYXNlLmNwcCBiL2xpYnMvdXRpbHMvTWVtb3J5SGVhcEJhc2UuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4MjUxNzI4Li4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvTWVtb3J5SGVhcEJhc2UuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTgzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDggVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIk1lbW9yeUhlYXBCYXNlIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8ZmNudGwuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHN5cy9zdGF0Lmg+Ci0jaW5jbHVkZSA8c3lzL2lvY3RsLmg+Ci0KLSNpbmNsdWRlIDxjdXRpbHMvbG9nLmg+Ci0jaW5jbHVkZSA8Y3V0aWxzL2FzaG1lbS5oPgotI2luY2x1ZGUgPGN1dGlscy9hdG9taWMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL01lbW9yeUhlYXBCYXNlLmg+Ci0KLSNpZiBIQVZFX0FORFJPSURfT1MKLSNpbmNsdWRlIDxsaW51eC9hbmRyb2lkX3BtZW0uaD4KLSNlbmRpZgotCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLU1lbW9yeUhlYXBCYXNlOjpNZW1vcnlIZWFwQmFzZSgpIAotICAgIDogbUZEKC0xKSwgbVNpemUoMCksIG1CYXNlKE1BUF9GQUlMRUQpLAotICAgICAgbURldmljZShOVUxMKSwgbU5lZWRVbm1hcChmYWxzZSkgCi17Ci19Ci0KLU1lbW9yeUhlYXBCYXNlOjpNZW1vcnlIZWFwQmFzZShzaXplX3Qgc2l6ZSwgdWludDMyX3QgZmxhZ3MsIGNoYXIgY29uc3QgKiBuYW1lKQotICAgIDogbUZEKC0xKSwgbVNpemUoMCksIG1CYXNlKE1BUF9GQUlMRUQpLCBtRmxhZ3MoZmxhZ3MpLAotICAgICAgbURldmljZSgwKSwgbU5lZWRVbm1hcChmYWxzZSkKLXsKLSAgICBjb25zdCBzaXplX3QgcGFnZXNpemUgPSBnZXRwYWdlc2l6ZSgpOwotICAgIHNpemUgPSAoKHNpemUgKyBwYWdlc2l6ZS0xKSAmIH4ocGFnZXNpemUtMSkpOwotICAgIGludCBmZCA9IGFzaG1lbV9jcmVhdGVfcmVnaW9uKG5hbWUgPT0gTlVMTCA/ICJNZW1vcnlIZWFwQmFzZSIgOiBuYW1lLCBzaXplKTsKLSAgICBMT0dFX0lGKGZkPDAsICJlcnJvciBjcmVhdGluZyBhc2htZW0gcmVnaW9uOiAlcyIsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgaWYgKGZkID49IDApIHsKLSAgICAgICAgaWYgKG1hcGZkKGZkLCBzaXplKSA9PSBOT19FUlJPUikgewotICAgICAgICAgICAgaWYgKGZsYWdzICYgUkVBRF9PTkxZKSB7Ci0gICAgICAgICAgICAgICAgYXNobWVtX3NldF9wcm90X3JlZ2lvbihmZCwgUFJPVF9SRUFEKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLX0KLQotTWVtb3J5SGVhcEJhc2U6Ok1lbW9yeUhlYXBCYXNlKGNvbnN0IGNoYXIqIGRldmljZSwgc2l6ZV90IHNpemUsIHVpbnQzMl90IGZsYWdzKQotICAgIDogbUZEKC0xKSwgbVNpemUoMCksIG1CYXNlKE1BUF9GQUlMRUQpLCBtRmxhZ3MoZmxhZ3MpLAotICAgICAgbURldmljZSgwKSwgbU5lZWRVbm1hcChmYWxzZSkKLXsKLSAgICBpbnQgZmQgPSBvcGVuKGRldmljZSwgT19SRFdSKTsKLSAgICBMT0dFX0lGKGZkPDAsICJlcnJvciBvcGVuaW5nICVzOiAlcyIsIGRldmljZSwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICBpZiAoZmQgPj0gMCkgewotICAgICAgICBjb25zdCBzaXplX3QgcGFnZXNpemUgPSBnZXRwYWdlc2l6ZSgpOwotICAgICAgICBzaXplID0gKChzaXplICsgcGFnZXNpemUtMSkgJiB+KHBhZ2VzaXplLTEpKTsKLSAgICAgICAgaWYgKG1hcGZkKGZkLCBzaXplKSA9PSBOT19FUlJPUikgewotICAgICAgICAgICAgbURldmljZSA9IGRldmljZTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotTWVtb3J5SGVhcEJhc2U6Ok1lbW9yeUhlYXBCYXNlKGludCBmZCwgc2l6ZV90IHNpemUsIHVpbnQzMl90IGZsYWdzKQotICAgIDogbUZEKC0xKSwgbVNpemUoMCksIG1CYXNlKE1BUF9GQUlMRUQpLCBtRmxhZ3MoZmxhZ3MpLAotICAgICAgbURldmljZSgwKSwgbU5lZWRVbm1hcChmYWxzZSkKLXsKLSAgICBjb25zdCBzaXplX3QgcGFnZXNpemUgPSBnZXRwYWdlc2l6ZSgpOwotICAgIHNpemUgPSAoKHNpemUgKyBwYWdlc2l6ZS0xKSAmIH4ocGFnZXNpemUtMSkpOwotICAgIG1hcGZkKGR1cChmZCksIHNpemUpOwotfQotCi1zdGF0dXNfdCBNZW1vcnlIZWFwQmFzZTo6aW5pdChpbnQgZmQsIHZvaWQgKmJhc2UsIGludCBzaXplLCBpbnQgZmxhZ3MsIGNvbnN0IGNoYXIqIGRldmljZSkKLXsKLSAgICBpZiAobUZEICE9IC0xKSB7Ci0gICAgICAgIHJldHVybiBJTlZBTElEX09QRVJBVElPTjsKLSAgICB9Ci0gICAgbUZEID0gZmQ7Ci0gICAgbUJhc2UgPSBiYXNlOwotICAgIG1TaXplID0gc2l6ZTsKLSAgICBtRmxhZ3MgPSBmbGFnczsKLSAgICBtRGV2aWNlID0gZGV2aWNlOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgTWVtb3J5SGVhcEJhc2U6Om1hcGZkKGludCBmZCwgc2l6ZV90IHNpemUpCi17Ci0gICAgaWYgKHNpemUgPT0gMCkgewotICAgICAgICAvLyB0cnkgdG8gZmlndXJlIG91dCB0aGUgc2l6ZSBhdXRvbWF0aWNhbGx5Ci0jaWYgSEFWRV9BTkRST0lEX09TCi0gICAgICAgIC8vIGZpcnN0IHRyeSB0aGUgUE1FTSBpb2N0bAotICAgICAgICBwbWVtX3JlZ2lvbiByZWc7Ci0gICAgICAgIGludCBlcnIgPSBpb2N0bChmZCwgUE1FTV9HRVRfVE9UQUxfU0laRSwgJnJlZyk7Ci0gICAgICAgIGlmIChlcnIgPT0gMCkKLSAgICAgICAgICAgIHNpemUgPSByZWcubGVuOwotI2VuZGlmCi0gICAgICAgIGlmIChzaXplID09IDApIHsgLy8gdHJ5IGZzdGF0Ci0gICAgICAgICAgICBzdHJ1Y3Qgc3RhdCBzYjsKLSAgICAgICAgICAgIGlmIChmc3RhdChmZCwgJnNiKSA9PSAwKQotICAgICAgICAgICAgICAgIHNpemUgPSBzYi5zdF9zaXplOwotICAgICAgICB9Ci0gICAgICAgIC8vIGlmIGl0IGRpZG4ndCB3b3JrLCBsZXQgbW1hcCgpIGZhaWwuCi0gICAgfQotCi0gICAgaWYgKChtRmxhZ3MgJiBET05UX01BUF9MT0NBTExZKSA9PSAwKSB7Ci0gICAgICAgIHZvaWQqIGJhc2UgPSAodWludDhfdCopbW1hcCgwLCBzaXplLAotICAgICAgICAgICAgICAgIFBST1RfUkVBRHxQUk9UX1dSSVRFLCBNQVBfU0hBUkVELCBmZCwgMCk7Ci0gICAgICAgIGlmIChiYXNlID09IE1BUF9GQUlMRUQpIHsKLSAgICAgICAgICAgIExPR0UoIm1tYXAoZmQ9JWQsIHNpemU9JXUpIGZhaWxlZCAoJXMpIiwKLSAgICAgICAgICAgICAgICAgICAgZmQsIHVpbnQzMl90KHNpemUpLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICAgICAgY2xvc2UoZmQpOwotICAgICAgICAgICAgcmV0dXJuIC1lcnJubzsKLSAgICAgICAgfQotICAgICAgICAvL0xPR0QoIm1tYXAoZmQ9JWQsIGJhc2U9JXAsIHNpemU9JWx1KSIsIGZkLCBiYXNlLCBzaXplKTsKLSAgICAgICAgbUJhc2UgPSBiYXNlOwotICAgICAgICBtTmVlZFVubWFwID0gdHJ1ZTsKLSAgICB9IGVsc2UgIHsKLSAgICAgICAgbUJhc2UgPSAwOyAvLyBub3QgTUFQX0ZBSUxFRAotICAgICAgICBtTmVlZFVubWFwID0gZmFsc2U7Ci0gICAgfQotICAgIG1GRCA9IGZkOwotICAgIG1TaXplID0gc2l6ZTsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLU1lbW9yeUhlYXBCYXNlOjp+TWVtb3J5SGVhcEJhc2UoKQotewotICAgIGRpc3Bvc2UoKTsKLX0KLQotdm9pZCBNZW1vcnlIZWFwQmFzZTo6ZGlzcG9zZSgpCi17Ci0gICAgaW50IGZkID0gYW5kcm9pZF9hdG9taWNfb3IoLTEsICZtRkQpOwotICAgIGlmIChmZCA+PSAwKSB7Ci0gICAgICAgIGlmIChtTmVlZFVubWFwKSB7Ci0gICAgICAgICAgICAvL0xPR0QoIm11bm1hcChmZD0lZCwgYmFzZT0lcCwgc2l6ZT0lbHUpIiwgZmQsIG1CYXNlLCBtU2l6ZSk7Ci0gICAgICAgICAgICBtdW5tYXAobUJhc2UsIG1TaXplKTsKLSAgICAgICAgfQotICAgICAgICBtQmFzZSA9IDA7Ci0gICAgICAgIG1TaXplID0gMDsKLSAgICAgICAgY2xvc2UoZmQpOwotICAgIH0KLX0KLQotaW50IE1lbW9yeUhlYXBCYXNlOjpnZXRIZWFwSUQoKSBjb25zdCB7Ci0gICAgcmV0dXJuIG1GRDsKLX0KLQotdm9pZCogTWVtb3J5SGVhcEJhc2U6OmdldEJhc2UoKSBjb25zdCB7Ci0gICAgcmV0dXJuIG1CYXNlOwotfQotCi1zaXplX3QgTWVtb3J5SGVhcEJhc2U6OmdldFNpemUoKSBjb25zdCB7Ci0gICAgcmV0dXJuIG1TaXplOwotfQotCi11aW50MzJfdCBNZW1vcnlIZWFwQmFzZTo6Z2V0RmxhZ3MoKSBjb25zdCB7Ci0gICAgcmV0dXJuIG1GbGFnczsKLX0KLQotY29uc3QgY2hhciogTWVtb3J5SGVhcEJhc2U6OmdldERldmljZSgpIGNvbnN0IHsKLSAgICByZXR1cm4gbURldmljZTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9NZW1vcnlIZWFwUG1lbS5jcHAgYi9saWJzL3V0aWxzL01lbW9yeUhlYXBQbWVtLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZWJhMmIzMC4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL01lbW9yeUhlYXBQbWVtLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDI0OCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJNZW1vcnlIZWFwUG1lbSIKLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHVuaXN0ZC5oPgotI2luY2x1ZGUgPGZjbnRsLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLSNpbmNsdWRlIDxzeXMvc3RhdC5oPgotI2luY2x1ZGUgPHN5cy9pb2N0bC5oPgotCi0jaW5jbHVkZSA8Y3V0aWxzL2xvZy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvTWVtb3J5SGVhcFBtZW0uaD4KLSNpbmNsdWRlIDx1dGlscy9NZW1vcnlIZWFwQmFzZS5oPgotCi0jaWYgSEFWRV9BTkRST0lEX09TCi0jaW5jbHVkZSA8bGludXgvYW5kcm9pZF9wbWVtLmg+Ci0jZW5kaWYKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotTWVtb3J5SGVhcFBtZW06Ok1lbW9yeVBtZW06Ok1lbW9yeVBtZW0oY29uc3Qgc3A8TWVtb3J5SGVhcFBtZW0+JiBoZWFwKQotICAgIDogQm5NZW1vcnkoKSwgbUNsaWVudEhlYXAoaGVhcCkKLXsKLX0KLQotTWVtb3J5SGVhcFBtZW06Ok1lbW9yeVBtZW06On5NZW1vcnlQbWVtKCkgewotICAgIGlmIChtQ2xpZW50SGVhcCAhPSBOVUxMKSB7Ci0gICAgICAgIG1DbGllbnRIZWFwLT5yZW1vdmUodGhpcyk7Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgU3ViUmVnaW9uTWVtb3J5IDogcHVibGljIE1lbW9yeUhlYXBQbWVtOjpNZW1vcnlQbWVtIHsKLXB1YmxpYzoKLSAgICBTdWJSZWdpb25NZW1vcnkoY29uc3Qgc3A8TWVtb3J5SGVhcFBtZW0+JiBoZWFwLCBzc2l6ZV90IG9mZnNldCwgc2l6ZV90IHNpemUpOwotICAgIHZpcnR1YWwgflN1YlJlZ2lvbk1lbW9yeSgpOwotICAgIHZpcnR1YWwgc3A8SU1lbW9yeUhlYXA+IGdldE1lbW9yeShzc2l6ZV90KiBvZmZzZXQsIHNpemVfdCogc2l6ZSkgY29uc3Q7Ci1wcml2YXRlOgotICAgIGZyaWVuZCBjbGFzcyBNZW1vcnlIZWFwUG1lbTsKLSAgICB2b2lkIHJldm9rZSgpOwotICAgIHNpemVfdCAgICAgICAgICAgICAgbVNpemU7Ci0gICAgc3NpemVfdCAgICAgICAgICAgICBtT2Zmc2V0OwotfTsKLQotU3ViUmVnaW9uTWVtb3J5OjpTdWJSZWdpb25NZW1vcnkoY29uc3Qgc3A8TWVtb3J5SGVhcFBtZW0+JiBoZWFwLAotICAgICAgICBzc2l6ZV90IG9mZnNldCwgc2l6ZV90IHNpemUpCi0gICAgOiBNZW1vcnlIZWFwUG1lbTo6TWVtb3J5UG1lbShoZWFwKSwgbVNpemUoc2l6ZSksIG1PZmZzZXQob2Zmc2V0KQotewotI2lmbmRlZiBOREVCVUcKLSAgICB2b2lkKiBjb25zdCBzdGFydF9wdHIgPSAodm9pZCopKGludHB0cl90KGdldEhlYXAoKS0+YmFzZSgpKSArIG9mZnNldCk7Ci0gICAgbWVtc2V0KHN0YXJ0X3B0ciwgMHhkYSwgc2l6ZSk7Ci0jZW5kaWYKLQotI2lmIEhBVkVfQU5EUk9JRF9PUwotICAgIGlmIChzaXplID4gMCkgewotICAgICAgICBjb25zdCBzaXplX3QgcGFnZXNpemUgPSBnZXRwYWdlc2l6ZSgpOwotICAgICAgICBzaXplID0gKHNpemUgKyBwYWdlc2l6ZS0xKSAmIH4ocGFnZXNpemUtMSk7Ci0gICAgICAgIGludCBvdXJfZmQgPSBoZWFwLT5oZWFwSUQoKTsKLSAgICAgICAgc3RydWN0IHBtZW1fcmVnaW9uIHN1YiA9IHsgb2Zmc2V0LCBzaXplIH07Ci0gICAgICAgIGludCBlcnIgPSBpb2N0bChvdXJfZmQsIFBNRU1fTUFQLCAmc3ViKTsKLSAgICAgICAgTE9HRV9JRihlcnI8MCwgIlBNRU1fTUFQIGZhaWxlZCAoJXMpLCAiCi0gICAgICAgICAgICAgICAgIm1GRD0lZCwgc3ViLm9mZnNldD0lbHUsIHN1Yi5zaXplPSVsdSIsCi0gICAgICAgICAgICAgICAgc3RyZXJyb3IoZXJybm8pLCBvdXJfZmQsIHN1Yi5vZmZzZXQsIHN1Yi5sZW4pOwotfQotI2VuZGlmCi19Ci0KLXNwPElNZW1vcnlIZWFwPiBTdWJSZWdpb25NZW1vcnk6OmdldE1lbW9yeShzc2l6ZV90KiBvZmZzZXQsIHNpemVfdCogc2l6ZSkgY29uc3QKLXsKLSAgICBpZiAob2Zmc2V0KSAqb2Zmc2V0ID0gbU9mZnNldDsKLSAgICBpZiAoc2l6ZSkgICAqc2l6ZSA9IG1TaXplOwotICAgIHJldHVybiBnZXRIZWFwKCk7Ci19Ci0KLVN1YlJlZ2lvbk1lbW9yeTo6flN1YlJlZ2lvbk1lbW9yeSgpCi17Ci0gICAgcmV2b2tlKCk7Ci19Ci0KLQotdm9pZCBTdWJSZWdpb25NZW1vcnk6OnJldm9rZSgpCi17Ci0gICAgLy8gTk9URTogcmV2b2tlKCkgZG9lc24ndCBuZWVkIHRvIGJlIHByb3RlY3RlZCBieSBhIGxvY2sgYmVjYXVzZSBpdAotICAgIC8vIGNhbiBvbmx5IGJlIGNhbGxlZCBmcm9tIE1lbW9yeUhlYXBQbWVtOjpyZXZva2UoKSwgd2hpY2ggbWVhbnMKLSAgICAvLyB0aGF0IHdlIGNhbid0IGJlIGluIH5TdWJSZWdpb25NZW1vcnkoKSwgb3IgaW4gflN1YlJlZ2lvbk1lbW9yeSgpLAotICAgIC8vIHdoaWNoIG1lYW5zIE1lbW9yeUhlYXBQbWVtOjpyZXZva2UoKSB3b3VsZG4ndCBoYXZlIGJlZW4gYWJsZSB0byAKLSAgICAvLyBwcm9tb3RlKCkgaXQuCi0gICAgCi0jaWYgSEFWRV9BTkRST0lEX09TCi0gICAgaWYgKG1TaXplICE9IE5VTEwpIHsKLSAgICAgICAgY29uc3Qgc3A8TWVtb3J5SGVhcFBtZW0+JiBoZWFwKGdldEhlYXAoKSk7Ci0gICAgICAgIGludCBvdXJfZmQgPSBoZWFwLT5oZWFwSUQoKTsKLSAgICAgICAgc3RydWN0IHBtZW1fcmVnaW9uIHN1YjsKLSAgICAgICAgc3ViLm9mZnNldCA9IG1PZmZzZXQ7Ci0gICAgICAgIHN1Yi5sZW4gPSBtU2l6ZTsKLSAgICAgICAgaW50IGVyciA9IGlvY3RsKG91cl9mZCwgUE1FTV9VTk1BUCwgJnN1Yik7Ci0gICAgICAgIExPR0VfSUYoZXJyPDAsICJQTUVNX1VOTUFQIGZhaWxlZCAoJXMpLCAiCi0gICAgICAgICAgICAgICAgIm1GRD0lZCwgc3ViLm9mZnNldD0lbHUsIHN1Yi5zaXplPSVsdSIsCi0gICAgICAgICAgICAgICAgc3RyZXJyb3IoZXJybm8pLCBvdXJfZmQsIHN1Yi5vZmZzZXQsIHN1Yi5sZW4pOwotICAgICAgICBtU2l6ZSA9IDA7Ci0gICAgfQotI2VuZGlmCi19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1NZW1vcnlIZWFwUG1lbTo6TWVtb3J5SGVhcFBtZW0oY29uc3Qgc3A8TWVtb3J5SGVhcEJhc2U+JiBwbWVtSGVhcCwKLSAgICAgICAgdWludDMyX3QgZmxhZ3MpCi0gICAgOiBIZWFwSW50ZXJmYWNlKCksIE1lbW9yeUhlYXBCYXNlKCkKLXsKLSAgICBjaGFyIGNvbnN0ICogY29uc3QgZGV2aWNlID0gcG1lbUhlYXAtPmdldERldmljZSgpOwotI2lmIEhBVkVfQU5EUk9JRF9PUwotICAgIGlmIChkZXZpY2UpIHsKLSAgICAgICAgaW50IGZkID0gb3BlbihkZXZpY2UsIE9fUkRXUik7Ci0gICAgICAgIExPR0VfSUYoZmQ8MCwgImNvdWxkbid0IG9wZW4gJXMgKCVzKSIsIGRldmljZSwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgaWYgKGZkID49IDApIHsKLSAgICAgICAgICAgIGludCBlcnIgPSBpb2N0bChmZCwgUE1FTV9DT05ORUNULCBwbWVtSGVhcC0+aGVhcElEKCkpOwotICAgICAgICAgICAgaWYgKGVyciA8IDApIHsKLSAgICAgICAgICAgICAgICBMT0dFKCJQTUVNX0NPTk5FQ1QgZmFpbGVkICglcyksIG1GRD0lZCwgc3ViLWZkPSVkIiwKLSAgICAgICAgICAgICAgICAgICAgICAgIHN0cmVycm9yKGVycm5vKSwgZmQsIHBtZW1IZWFwLT5oZWFwSUQoKSk7Ci0gICAgICAgICAgICAgICAgY2xvc2UoZmQpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAvLyBldmVyeXRoaW5nIHdlbnQgd2VsbC4uLgotICAgICAgICAgICAgICAgIG1QYXJlbnRIZWFwID0gcG1lbUhlYXA7Ci0gICAgICAgICAgICAgICAgTWVtb3J5SGVhcEJhc2U6OmluaXQoZmQsIAotICAgICAgICAgICAgICAgICAgICAgICAgcG1lbUhlYXAtPmdldEJhc2UoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgIHBtZW1IZWFwLT5nZXRTaXplKCksCi0gICAgICAgICAgICAgICAgICAgICAgICBwbWVtSGVhcC0+Z2V0RmxhZ3MoKSB8IGZsYWdzLAotICAgICAgICAgICAgICAgICAgICAgICAgZGV2aWNlKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSNlbHNlCi0gICAgbVBhcmVudEhlYXAgPSBwbWVtSGVhcDsKLSAgICBNZW1vcnlIZWFwQmFzZTo6aW5pdCggCi0gICAgICAgICAgICBkdXAocG1lbUhlYXAtPmhlYXBJRCgpKSwKLSAgICAgICAgICAgIHBtZW1IZWFwLT5nZXRCYXNlKCksCi0gICAgICAgICAgICBwbWVtSGVhcC0+Z2V0U2l6ZSgpLAotICAgICAgICAgICAgcG1lbUhlYXAtPmdldEZsYWdzKCkgfCBmbGFncywKLSAgICAgICAgICAgIGRldmljZSk7Ci0jZW5kaWYKLX0KLQotTWVtb3J5SGVhcFBtZW06On5NZW1vcnlIZWFwUG1lbSgpCi17Ci19Ci0KLXNwPElNZW1vcnk+IE1lbW9yeUhlYXBQbWVtOjptYXBNZW1vcnkoc2l6ZV90IG9mZnNldCwgc2l6ZV90IHNpemUpCi17Ci0gICAgc3A8TWVtb3J5UG1lbT4gbWVtb3J5ID0gY3JlYXRlTWVtb3J5KG9mZnNldCwgc2l6ZSk7Ci0gICAgaWYgKG1lbW9yeSAhPSAwKSB7Ci0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgICAgIG1BbGxvY2F0aW9ucy5hZGQobWVtb3J5KTsKLSAgICB9Ci0gICAgcmV0dXJuIG1lbW9yeTsKLX0KLQotc3A8TWVtb3J5SGVhcFBtZW06Ok1lbW9yeVBtZW0+IE1lbW9yeUhlYXBQbWVtOjpjcmVhdGVNZW1vcnkoCi0gICAgICAgIHNpemVfdCBvZmZzZXQsIHNpemVfdCBzaXplKQotewotICAgIHNwPFN1YlJlZ2lvbk1lbW9yeT4gbWVtb3J5OwotICAgIGlmIChoZWFwSUQoKSA+IDApIAotICAgICAgICBtZW1vcnkgPSBuZXcgU3ViUmVnaW9uTWVtb3J5KHRoaXMsIG9mZnNldCwgc2l6ZSk7Ci0gICAgcmV0dXJuIG1lbW9yeTsKLX0KLQotc3RhdHVzX3QgTWVtb3J5SGVhcFBtZW06OnNsYXAoKQotewotI2lmIEhBVkVfQU5EUk9JRF9PUwotICAgIHNpemVfdCBzaXplID0gZ2V0U2l6ZSgpOwotICAgIGNvbnN0IHNpemVfdCBwYWdlc2l6ZSA9IGdldHBhZ2VzaXplKCk7Ci0gICAgc2l6ZSA9IChzaXplICsgcGFnZXNpemUtMSkgJiB+KHBhZ2VzaXplLTEpOwotICAgIGludCBvdXJfZmQgPSBnZXRIZWFwSUQoKTsKLSAgICBzdHJ1Y3QgcG1lbV9yZWdpb24gc3ViID0geyAwLCBzaXplIH07Ci0gICAgaW50IGVyciA9IGlvY3RsKG91cl9mZCwgUE1FTV9NQVAsICZzdWIpOwotICAgIExPR0VfSUYoZXJyPDAsICJQTUVNX01BUCBmYWlsZWQgKCVzKSwgIgotICAgICAgICAgICAgIm1GRD0lZCwgc3ViLm9mZnNldD0lbHUsIHN1Yi5zaXplPSVsdSIsCi0gICAgICAgICAgICBzdHJlcnJvcihlcnJubyksIG91cl9mZCwgc3ViLm9mZnNldCwgc3ViLmxlbik7Ci0gICAgcmV0dXJuIC1lcnJubzsKLSNlbHNlCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotI2VuZGlmCi19Ci0KLXN0YXR1c190IE1lbW9yeUhlYXBQbWVtOjp1bnNsYXAoKQotewotI2lmIEhBVkVfQU5EUk9JRF9PUwotICAgIHNpemVfdCBzaXplID0gZ2V0U2l6ZSgpOwotICAgIGNvbnN0IHNpemVfdCBwYWdlc2l6ZSA9IGdldHBhZ2VzaXplKCk7Ci0gICAgc2l6ZSA9IChzaXplICsgcGFnZXNpemUtMSkgJiB+KHBhZ2VzaXplLTEpOwotICAgIGludCBvdXJfZmQgPSBnZXRIZWFwSUQoKTsKLSAgICBzdHJ1Y3QgcG1lbV9yZWdpb24gc3ViID0geyAwLCBzaXplIH07Ci0gICAgaW50IGVyciA9IGlvY3RsKG91cl9mZCwgUE1FTV9VTk1BUCwgJnN1Yik7Ci0gICAgTE9HRV9JRihlcnI8MCwgIlBNRU1fVU5NQVAgZmFpbGVkICglcyksICIKLSAgICAgICAgICAgICJtRkQ9JWQsIHN1Yi5vZmZzZXQ9JWx1LCBzdWIuc2l6ZT0lbHUiLAotICAgICAgICAgICAgc3RyZXJyb3IoZXJybm8pLCBvdXJfZmQsIHN1Yi5vZmZzZXQsIHN1Yi5sZW4pOwotICAgIHJldHVybiAtZXJybm87Ci0jZWxzZQotICAgIHJldHVybiBOT19FUlJPUjsKLSNlbmRpZgotfQotCi12b2lkIE1lbW9yeUhlYXBQbWVtOjpyZXZva2UoKQotewotICAgIFNvcnRlZFZlY3Rvcjwgd3A8TWVtb3J5UG1lbT4gPiBhbGxvY2F0aW9uczsKLQotICAgIHsgLy8gc2NvcGUgZm9yIGxvY2sKLSAgICAgICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICAgICAgYWxsb2NhdGlvbnMgPSBtQWxsb2NhdGlvbnM7Ci0gICAgfQotICAgIAotICAgIHNzaXplX3QgY291bnQgPSBhbGxvY2F0aW9ucy5zaXplKCk7Ci0gICAgZm9yIChzc2l6ZV90IGk9MCA7IGk8Y291bnQgOyBpKyspIHsKLSAgICAgICAgc3A8TWVtb3J5UG1lbT4gbWVtb3J5KGFsbG9jYXRpb25zW2ldLnByb21vdGUoKSk7Ci0gICAgICAgIGlmIChtZW1vcnkgIT0gMCkKLSAgICAgICAgICAgIG1lbW9yeS0+cmV2b2tlKCk7Ci0gICAgfQotfQotCi12b2lkIE1lbW9yeUhlYXBQbWVtOjpyZW1vdmUoY29uc3Qgd3A8TWVtb3J5UG1lbT4mIG1lbW9yeSkKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIG1BbGxvY2F0aW9ucy5yZW1vdmUobWVtb3J5KTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9OT1RJQ0UgYi9saWJzL3V0aWxzL05PVElDRQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzViMWVmYS4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL05PVElDRQorKysgL2Rldi9udWxsCkBAIC0xLDE5MCArMCwwIEBACi0KLSAgIENvcHlyaWdodCAoYykgMjAwNS0yMDA4LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0KLSAgIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICAgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotCi0gICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLQotCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBBcGFjaGUgTGljZW5zZQotICAgICAgICAgICAgICAgICAgICAgICAgICAgVmVyc2lvbiAyLjAsIEphbnVhcnkgMjAwNAotICAgICAgICAgICAgICAgICAgICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzLwotCi0gICBURVJNUyBBTkQgQ09ORElUSU9OUyBGT1IgVVNFLCBSRVBST0RVQ1RJT04sIEFORCBESVNUUklCVVRJT04KLQotICAgMS4gRGVmaW5pdGlvbnMuCi0KLSAgICAgICJMaWNlbnNlIiBzaGFsbCBtZWFuIHRoZSB0ZXJtcyBhbmQgY29uZGl0aW9ucyBmb3IgdXNlLCByZXByb2R1Y3Rpb24sCi0gICAgICBhbmQgZGlzdHJpYnV0aW9uIGFzIGRlZmluZWQgYnkgU2VjdGlvbnMgMSB0aHJvdWdoIDkgb2YgdGhpcyBkb2N1bWVudC4KLQotICAgICAgIkxpY2Vuc29yIiBzaGFsbCBtZWFuIHRoZSBjb3B5cmlnaHQgb3duZXIgb3IgZW50aXR5IGF1dGhvcml6ZWQgYnkKLSAgICAgIHRoZSBjb3B5cmlnaHQgb3duZXIgdGhhdCBpcyBncmFudGluZyB0aGUgTGljZW5zZS4KLQotICAgICAgIkxlZ2FsIEVudGl0eSIgc2hhbGwgbWVhbiB0aGUgdW5pb24gb2YgdGhlIGFjdGluZyBlbnRpdHkgYW5kIGFsbAotICAgICAgb3RoZXIgZW50aXRpZXMgdGhhdCBjb250cm9sLCBhcmUgY29udHJvbGxlZCBieSwgb3IgYXJlIHVuZGVyIGNvbW1vbgotICAgICAgY29udHJvbCB3aXRoIHRoYXQgZW50aXR5LiBGb3IgdGhlIHB1cnBvc2VzIG9mIHRoaXMgZGVmaW5pdGlvbiwKLSAgICAgICJjb250cm9sIiBtZWFucyAoaSkgdGhlIHBvd2VyLCBkaXJlY3Qgb3IgaW5kaXJlY3QsIHRvIGNhdXNlIHRoZQotICAgICAgZGlyZWN0aW9uIG9yIG1hbmFnZW1lbnQgb2Ygc3VjaCBlbnRpdHksIHdoZXRoZXIgYnkgY29udHJhY3Qgb3IKLSAgICAgIG90aGVyd2lzZSwgb3IgKGlpKSBvd25lcnNoaXAgb2YgZmlmdHkgcGVyY2VudCAoNTAlKSBvciBtb3JlIG9mIHRoZQotICAgICAgb3V0c3RhbmRpbmcgc2hhcmVzLCBvciAoaWlpKSBiZW5lZmljaWFsIG93bmVyc2hpcCBvZiBzdWNoIGVudGl0eS4KLQotICAgICAgIllvdSIgKG9yICJZb3VyIikgc2hhbGwgbWVhbiBhbiBpbmRpdmlkdWFsIG9yIExlZ2FsIEVudGl0eQotICAgICAgZXhlcmNpc2luZyBwZXJtaXNzaW9ucyBncmFudGVkIGJ5IHRoaXMgTGljZW5zZS4KLQotICAgICAgIlNvdXJjZSIgZm9ybSBzaGFsbCBtZWFuIHRoZSBwcmVmZXJyZWQgZm9ybSBmb3IgbWFraW5nIG1vZGlmaWNhdGlvbnMsCi0gICAgICBpbmNsdWRpbmcgYnV0IG5vdCBsaW1pdGVkIHRvIHNvZnR3YXJlIHNvdXJjZSBjb2RlLCBkb2N1bWVudGF0aW9uCi0gICAgICBzb3VyY2UsIGFuZCBjb25maWd1cmF0aW9uIGZpbGVzLgotCi0gICAgICAiT2JqZWN0IiBmb3JtIHNoYWxsIG1lYW4gYW55IGZvcm0gcmVzdWx0aW5nIGZyb20gbWVjaGFuaWNhbAotICAgICAgdHJhbnNmb3JtYXRpb24gb3IgdHJhbnNsYXRpb24gb2YgYSBTb3VyY2UgZm9ybSwgaW5jbHVkaW5nIGJ1dAotICAgICAgbm90IGxpbWl0ZWQgdG8gY29tcGlsZWQgb2JqZWN0IGNvZGUsIGdlbmVyYXRlZCBkb2N1bWVudGF0aW9uLAotICAgICAgYW5kIGNvbnZlcnNpb25zIHRvIG90aGVyIG1lZGlhIHR5cGVzLgotCi0gICAgICAiV29yayIgc2hhbGwgbWVhbiB0aGUgd29yayBvZiBhdXRob3JzaGlwLCB3aGV0aGVyIGluIFNvdXJjZSBvcgotICAgICAgT2JqZWN0IGZvcm0sIG1hZGUgYXZhaWxhYmxlIHVuZGVyIHRoZSBMaWNlbnNlLCBhcyBpbmRpY2F0ZWQgYnkgYQotICAgICAgY29weXJpZ2h0IG5vdGljZSB0aGF0IGlzIGluY2x1ZGVkIGluIG9yIGF0dGFjaGVkIHRvIHRoZSB3b3JrCi0gICAgICAoYW4gZXhhbXBsZSBpcyBwcm92aWRlZCBpbiB0aGUgQXBwZW5kaXggYmVsb3cpLgotCi0gICAgICAiRGVyaXZhdGl2ZSBXb3JrcyIgc2hhbGwgbWVhbiBhbnkgd29yaywgd2hldGhlciBpbiBTb3VyY2Ugb3IgT2JqZWN0Ci0gICAgICBmb3JtLCB0aGF0IGlzIGJhc2VkIG9uIChvciBkZXJpdmVkIGZyb20pIHRoZSBXb3JrIGFuZCBmb3Igd2hpY2ggdGhlCi0gICAgICBlZGl0b3JpYWwgcmV2aXNpb25zLCBhbm5vdGF0aW9ucywgZWxhYm9yYXRpb25zLCBvciBvdGhlciBtb2RpZmljYXRpb25zCi0gICAgICByZXByZXNlbnQsIGFzIGEgd2hvbGUsIGFuIG9yaWdpbmFsIHdvcmsgb2YgYXV0aG9yc2hpcC4gRm9yIHRoZSBwdXJwb3NlcwotICAgICAgb2YgdGhpcyBMaWNlbnNlLCBEZXJpdmF0aXZlIFdvcmtzIHNoYWxsIG5vdCBpbmNsdWRlIHdvcmtzIHRoYXQgcmVtYWluCi0gICAgICBzZXBhcmFibGUgZnJvbSwgb3IgbWVyZWx5IGxpbmsgKG9yIGJpbmQgYnkgbmFtZSkgdG8gdGhlIGludGVyZmFjZXMgb2YsCi0gICAgICB0aGUgV29yayBhbmQgRGVyaXZhdGl2ZSBXb3JrcyB0aGVyZW9mLgotCi0gICAgICAiQ29udHJpYnV0aW9uIiBzaGFsbCBtZWFuIGFueSB3b3JrIG9mIGF1dGhvcnNoaXAsIGluY2x1ZGluZwotICAgICAgdGhlIG9yaWdpbmFsIHZlcnNpb24gb2YgdGhlIFdvcmsgYW5kIGFueSBtb2RpZmljYXRpb25zIG9yIGFkZGl0aW9ucwotICAgICAgdG8gdGhhdCBXb3JrIG9yIERlcml2YXRpdmUgV29ya3MgdGhlcmVvZiwgdGhhdCBpcyBpbnRlbnRpb25hbGx5Ci0gICAgICBzdWJtaXR0ZWQgdG8gTGljZW5zb3IgZm9yIGluY2x1c2lvbiBpbiB0aGUgV29yayBieSB0aGUgY29weXJpZ2h0IG93bmVyCi0gICAgICBvciBieSBhbiBpbmRpdmlkdWFsIG9yIExlZ2FsIEVudGl0eSBhdXRob3JpemVkIHRvIHN1Ym1pdCBvbiBiZWhhbGYgb2YKLSAgICAgIHRoZSBjb3B5cmlnaHQgb3duZXIuIEZvciB0aGUgcHVycG9zZXMgb2YgdGhpcyBkZWZpbml0aW9uLCAic3VibWl0dGVkIgotICAgICAgbWVhbnMgYW55IGZvcm0gb2YgZWxlY3Ryb25pYywgdmVyYmFsLCBvciB3cml0dGVuIGNvbW11bmljYXRpb24gc2VudAotICAgICAgdG8gdGhlIExpY2Vuc29yIG9yIGl0cyByZXByZXNlbnRhdGl2ZXMsIGluY2x1ZGluZyBidXQgbm90IGxpbWl0ZWQgdG8KLSAgICAgIGNvbW11bmljYXRpb24gb24gZWxlY3Ryb25pYyBtYWlsaW5nIGxpc3RzLCBzb3VyY2UgY29kZSBjb250cm9sIHN5c3RlbXMsCi0gICAgICBhbmQgaXNzdWUgdHJhY2tpbmcgc3lzdGVtcyB0aGF0IGFyZSBtYW5hZ2VkIGJ5LCBvciBvbiBiZWhhbGYgb2YsIHRoZQotICAgICAgTGljZW5zb3IgZm9yIHRoZSBwdXJwb3NlIG9mIGRpc2N1c3NpbmcgYW5kIGltcHJvdmluZyB0aGUgV29yaywgYnV0Ci0gICAgICBleGNsdWRpbmcgY29tbXVuaWNhdGlvbiB0aGF0IGlzIGNvbnNwaWN1b3VzbHkgbWFya2VkIG9yIG90aGVyd2lzZQotICAgICAgZGVzaWduYXRlZCBpbiB3cml0aW5nIGJ5IHRoZSBjb3B5cmlnaHQgb3duZXIgYXMgIk5vdCBhIENvbnRyaWJ1dGlvbi4iCi0KLSAgICAgICJDb250cmlidXRvciIgc2hhbGwgbWVhbiBMaWNlbnNvciBhbmQgYW55IGluZGl2aWR1YWwgb3IgTGVnYWwgRW50aXR5Ci0gICAgICBvbiBiZWhhbGYgb2Ygd2hvbSBhIENvbnRyaWJ1dGlvbiBoYXMgYmVlbiByZWNlaXZlZCBieSBMaWNlbnNvciBhbmQKLSAgICAgIHN1YnNlcXVlbnRseSBpbmNvcnBvcmF0ZWQgd2l0aGluIHRoZSBXb3JrLgotCi0gICAyLiBHcmFudCBvZiBDb3B5cmlnaHQgTGljZW5zZS4gU3ViamVjdCB0byB0aGUgdGVybXMgYW5kIGNvbmRpdGlvbnMgb2YKLSAgICAgIHRoaXMgTGljZW5zZSwgZWFjaCBDb250cmlidXRvciBoZXJlYnkgZ3JhbnRzIHRvIFlvdSBhIHBlcnBldHVhbCwKLSAgICAgIHdvcmxkd2lkZSwgbm9uLWV4Y2x1c2l2ZSwgbm8tY2hhcmdlLCByb3lhbHR5LWZyZWUsIGlycmV2b2NhYmxlCi0gICAgICBjb3B5cmlnaHQgbGljZW5zZSB0byByZXByb2R1Y2UsIHByZXBhcmUgRGVyaXZhdGl2ZSBXb3JrcyBvZiwKLSAgICAgIHB1YmxpY2x5IGRpc3BsYXksIHB1YmxpY2x5IHBlcmZvcm0sIHN1YmxpY2Vuc2UsIGFuZCBkaXN0cmlidXRlIHRoZQotICAgICAgV29yayBhbmQgc3VjaCBEZXJpdmF0aXZlIFdvcmtzIGluIFNvdXJjZSBvciBPYmplY3QgZm9ybS4KLQotICAgMy4gR3JhbnQgb2YgUGF0ZW50IExpY2Vuc2UuIFN1YmplY3QgdG8gdGhlIHRlcm1zIGFuZCBjb25kaXRpb25zIG9mCi0gICAgICB0aGlzIExpY2Vuc2UsIGVhY2ggQ29udHJpYnV0b3IgaGVyZWJ5IGdyYW50cyB0byBZb3UgYSBwZXJwZXR1YWwsCi0gICAgICB3b3JsZHdpZGUsIG5vbi1leGNsdXNpdmUsIG5vLWNoYXJnZSwgcm95YWx0eS1mcmVlLCBpcnJldm9jYWJsZQotICAgICAgKGV4Y2VwdCBhcyBzdGF0ZWQgaW4gdGhpcyBzZWN0aW9uKSBwYXRlbnQgbGljZW5zZSB0byBtYWtlLCBoYXZlIG1hZGUsCi0gICAgICB1c2UsIG9mZmVyIHRvIHNlbGwsIHNlbGwsIGltcG9ydCwgYW5kIG90aGVyd2lzZSB0cmFuc2ZlciB0aGUgV29yaywKLSAgICAgIHdoZXJlIHN1Y2ggbGljZW5zZSBhcHBsaWVzIG9ubHkgdG8gdGhvc2UgcGF0ZW50IGNsYWltcyBsaWNlbnNhYmxlCi0gICAgICBieSBzdWNoIENvbnRyaWJ1dG9yIHRoYXQgYXJlIG5lY2Vzc2FyaWx5IGluZnJpbmdlZCBieSB0aGVpcgotICAgICAgQ29udHJpYnV0aW9uKHMpIGFsb25lIG9yIGJ5IGNvbWJpbmF0aW9uIG9mIHRoZWlyIENvbnRyaWJ1dGlvbihzKQotICAgICAgd2l0aCB0aGUgV29yayB0byB3aGljaCBzdWNoIENvbnRyaWJ1dGlvbihzKSB3YXMgc3VibWl0dGVkLiBJZiBZb3UKLSAgICAgIGluc3RpdHV0ZSBwYXRlbnQgbGl0aWdhdGlvbiBhZ2FpbnN0IGFueSBlbnRpdHkgKGluY2x1ZGluZyBhCi0gICAgICBjcm9zcy1jbGFpbSBvciBjb3VudGVyY2xhaW0gaW4gYSBsYXdzdWl0KSBhbGxlZ2luZyB0aGF0IHRoZSBXb3JrCi0gICAgICBvciBhIENvbnRyaWJ1dGlvbiBpbmNvcnBvcmF0ZWQgd2l0aGluIHRoZSBXb3JrIGNvbnN0aXR1dGVzIGRpcmVjdAotICAgICAgb3IgY29udHJpYnV0b3J5IHBhdGVudCBpbmZyaW5nZW1lbnQsIHRoZW4gYW55IHBhdGVudCBsaWNlbnNlcwotICAgICAgZ3JhbnRlZCB0byBZb3UgdW5kZXIgdGhpcyBMaWNlbnNlIGZvciB0aGF0IFdvcmsgc2hhbGwgdGVybWluYXRlCi0gICAgICBhcyBvZiB0aGUgZGF0ZSBzdWNoIGxpdGlnYXRpb24gaXMgZmlsZWQuCi0KLSAgIDQuIFJlZGlzdHJpYnV0aW9uLiBZb3UgbWF5IHJlcHJvZHVjZSBhbmQgZGlzdHJpYnV0ZSBjb3BpZXMgb2YgdGhlCi0gICAgICBXb3JrIG9yIERlcml2YXRpdmUgV29ya3MgdGhlcmVvZiBpbiBhbnkgbWVkaXVtLCB3aXRoIG9yIHdpdGhvdXQKLSAgICAgIG1vZGlmaWNhdGlvbnMsIGFuZCBpbiBTb3VyY2Ugb3IgT2JqZWN0IGZvcm0sIHByb3ZpZGVkIHRoYXQgWW91Ci0gICAgICBtZWV0IHRoZSBmb2xsb3dpbmcgY29uZGl0aW9uczoKLQotICAgICAgKGEpIFlvdSBtdXN0IGdpdmUgYW55IG90aGVyIHJlY2lwaWVudHMgb2YgdGhlIFdvcmsgb3IKLSAgICAgICAgICBEZXJpdmF0aXZlIFdvcmtzIGEgY29weSBvZiB0aGlzIExpY2Vuc2U7IGFuZAotCi0gICAgICAoYikgWW91IG11c3QgY2F1c2UgYW55IG1vZGlmaWVkIGZpbGVzIHRvIGNhcnJ5IHByb21pbmVudCBub3RpY2VzCi0gICAgICAgICAgc3RhdGluZyB0aGF0IFlvdSBjaGFuZ2VkIHRoZSBmaWxlczsgYW5kCi0KLSAgICAgIChjKSBZb3UgbXVzdCByZXRhaW4sIGluIHRoZSBTb3VyY2UgZm9ybSBvZiBhbnkgRGVyaXZhdGl2ZSBXb3JrcwotICAgICAgICAgIHRoYXQgWW91IGRpc3RyaWJ1dGUsIGFsbCBjb3B5cmlnaHQsIHBhdGVudCwgdHJhZGVtYXJrLCBhbmQKLSAgICAgICAgICBhdHRyaWJ1dGlvbiBub3RpY2VzIGZyb20gdGhlIFNvdXJjZSBmb3JtIG9mIHRoZSBXb3JrLAotICAgICAgICAgIGV4Y2x1ZGluZyB0aG9zZSBub3RpY2VzIHRoYXQgZG8gbm90IHBlcnRhaW4gdG8gYW55IHBhcnQgb2YKLSAgICAgICAgICB0aGUgRGVyaXZhdGl2ZSBXb3JrczsgYW5kCi0KLSAgICAgIChkKSBJZiB0aGUgV29yayBpbmNsdWRlcyBhICJOT1RJQ0UiIHRleHQgZmlsZSBhcyBwYXJ0IG9mIGl0cwotICAgICAgICAgIGRpc3RyaWJ1dGlvbiwgdGhlbiBhbnkgRGVyaXZhdGl2ZSBXb3JrcyB0aGF0IFlvdSBkaXN0cmlidXRlIG11c3QKLSAgICAgICAgICBpbmNsdWRlIGEgcmVhZGFibGUgY29weSBvZiB0aGUgYXR0cmlidXRpb24gbm90aWNlcyBjb250YWluZWQKLSAgICAgICAgICB3aXRoaW4gc3VjaCBOT1RJQ0UgZmlsZSwgZXhjbHVkaW5nIHRob3NlIG5vdGljZXMgdGhhdCBkbyBub3QKLSAgICAgICAgICBwZXJ0YWluIHRvIGFueSBwYXJ0IG9mIHRoZSBEZXJpdmF0aXZlIFdvcmtzLCBpbiBhdCBsZWFzdCBvbmUKLSAgICAgICAgICBvZiB0aGUgZm9sbG93aW5nIHBsYWNlczogd2l0aGluIGEgTk9USUNFIHRleHQgZmlsZSBkaXN0cmlidXRlZAotICAgICAgICAgIGFzIHBhcnQgb2YgdGhlIERlcml2YXRpdmUgV29ya3M7IHdpdGhpbiB0aGUgU291cmNlIGZvcm0gb3IKLSAgICAgICAgICBkb2N1bWVudGF0aW9uLCBpZiBwcm92aWRlZCBhbG9uZyB3aXRoIHRoZSBEZXJpdmF0aXZlIFdvcmtzOyBvciwKLSAgICAgICAgICB3aXRoaW4gYSBkaXNwbGF5IGdlbmVyYXRlZCBieSB0aGUgRGVyaXZhdGl2ZSBXb3JrcywgaWYgYW5kCi0gICAgICAgICAgd2hlcmV2ZXIgc3VjaCB0aGlyZC1wYXJ0eSBub3RpY2VzIG5vcm1hbGx5IGFwcGVhci4gVGhlIGNvbnRlbnRzCi0gICAgICAgICAgb2YgdGhlIE5PVElDRSBmaWxlIGFyZSBmb3IgaW5mb3JtYXRpb25hbCBwdXJwb3NlcyBvbmx5IGFuZAotICAgICAgICAgIGRvIG5vdCBtb2RpZnkgdGhlIExpY2Vuc2UuIFlvdSBtYXkgYWRkIFlvdXIgb3duIGF0dHJpYnV0aW9uCi0gICAgICAgICAgbm90aWNlcyB3aXRoaW4gRGVyaXZhdGl2ZSBXb3JrcyB0aGF0IFlvdSBkaXN0cmlidXRlLCBhbG9uZ3NpZGUKLSAgICAgICAgICBvciBhcyBhbiBhZGRlbmR1bSB0byB0aGUgTk9USUNFIHRleHQgZnJvbSB0aGUgV29yaywgcHJvdmlkZWQKLSAgICAgICAgICB0aGF0IHN1Y2ggYWRkaXRpb25hbCBhdHRyaWJ1dGlvbiBub3RpY2VzIGNhbm5vdCBiZSBjb25zdHJ1ZWQKLSAgICAgICAgICBhcyBtb2RpZnlpbmcgdGhlIExpY2Vuc2UuCi0KLSAgICAgIFlvdSBtYXkgYWRkIFlvdXIgb3duIGNvcHlyaWdodCBzdGF0ZW1lbnQgdG8gWW91ciBtb2RpZmljYXRpb25zIGFuZAotICAgICAgbWF5IHByb3ZpZGUgYWRkaXRpb25hbCBvciBkaWZmZXJlbnQgbGljZW5zZSB0ZXJtcyBhbmQgY29uZGl0aW9ucwotICAgICAgZm9yIHVzZSwgcmVwcm9kdWN0aW9uLCBvciBkaXN0cmlidXRpb24gb2YgWW91ciBtb2RpZmljYXRpb25zLCBvcgotICAgICAgZm9yIGFueSBzdWNoIERlcml2YXRpdmUgV29ya3MgYXMgYSB3aG9sZSwgcHJvdmlkZWQgWW91ciB1c2UsCi0gICAgICByZXByb2R1Y3Rpb24sIGFuZCBkaXN0cmlidXRpb24gb2YgdGhlIFdvcmsgb3RoZXJ3aXNlIGNvbXBsaWVzIHdpdGgKLSAgICAgIHRoZSBjb25kaXRpb25zIHN0YXRlZCBpbiB0aGlzIExpY2Vuc2UuCi0KLSAgIDUuIFN1Ym1pc3Npb24gb2YgQ29udHJpYnV0aW9ucy4gVW5sZXNzIFlvdSBleHBsaWNpdGx5IHN0YXRlIG90aGVyd2lzZSwKLSAgICAgIGFueSBDb250cmlidXRpb24gaW50ZW50aW9uYWxseSBzdWJtaXR0ZWQgZm9yIGluY2x1c2lvbiBpbiB0aGUgV29yawotICAgICAgYnkgWW91IHRvIHRoZSBMaWNlbnNvciBzaGFsbCBiZSB1bmRlciB0aGUgdGVybXMgYW5kIGNvbmRpdGlvbnMgb2YKLSAgICAgIHRoaXMgTGljZW5zZSwgd2l0aG91dCBhbnkgYWRkaXRpb25hbCB0ZXJtcyBvciBjb25kaXRpb25zLgotICAgICAgTm90d2l0aHN0YW5kaW5nIHRoZSBhYm92ZSwgbm90aGluZyBoZXJlaW4gc2hhbGwgc3VwZXJzZWRlIG9yIG1vZGlmeQotICAgICAgdGhlIHRlcm1zIG9mIGFueSBzZXBhcmF0ZSBsaWNlbnNlIGFncmVlbWVudCB5b3UgbWF5IGhhdmUgZXhlY3V0ZWQKLSAgICAgIHdpdGggTGljZW5zb3IgcmVnYXJkaW5nIHN1Y2ggQ29udHJpYnV0aW9ucy4KLQotICAgNi4gVHJhZGVtYXJrcy4gVGhpcyBMaWNlbnNlIGRvZXMgbm90IGdyYW50IHBlcm1pc3Npb24gdG8gdXNlIHRoZSB0cmFkZQotICAgICAgbmFtZXMsIHRyYWRlbWFya3MsIHNlcnZpY2UgbWFya3MsIG9yIHByb2R1Y3QgbmFtZXMgb2YgdGhlIExpY2Vuc29yLAotICAgICAgZXhjZXB0IGFzIHJlcXVpcmVkIGZvciByZWFzb25hYmxlIGFuZCBjdXN0b21hcnkgdXNlIGluIGRlc2NyaWJpbmcgdGhlCi0gICAgICBvcmlnaW4gb2YgdGhlIFdvcmsgYW5kIHJlcHJvZHVjaW5nIHRoZSBjb250ZW50IG9mIHRoZSBOT1RJQ0UgZmlsZS4KLQotICAgNy4gRGlzY2xhaW1lciBvZiBXYXJyYW50eS4gVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yCi0gICAgICBhZ3JlZWQgdG8gaW4gd3JpdGluZywgTGljZW5zb3IgcHJvdmlkZXMgdGhlIFdvcmsgKGFuZCBlYWNoCi0gICAgICBDb250cmlidXRvciBwcm92aWRlcyBpdHMgQ29udHJpYnV0aW9ucykgb24gYW4gIkFTIElTIiBCQVNJUywKLSAgICAgIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvcgotICAgICAgaW1wbGllZCwgaW5jbHVkaW5nLCB3aXRob3V0IGxpbWl0YXRpb24sIGFueSB3YXJyYW50aWVzIG9yIGNvbmRpdGlvbnMKLSAgICAgIG9mIFRJVExFLCBOT04tSU5GUklOR0VNRU5ULCBNRVJDSEFOVEFCSUxJVFksIG9yIEZJVE5FU1MgRk9SIEEKLSAgICAgIFBBUlRJQ1VMQVIgUFVSUE9TRS4gWW91IGFyZSBzb2xlbHkgcmVzcG9uc2libGUgZm9yIGRldGVybWluaW5nIHRoZQotICAgICAgYXBwcm9wcmlhdGVuZXNzIG9mIHVzaW5nIG9yIHJlZGlzdHJpYnV0aW5nIHRoZSBXb3JrIGFuZCBhc3N1bWUgYW55Ci0gICAgICByaXNrcyBhc3NvY2lhdGVkIHdpdGggWW91ciBleGVyY2lzZSBvZiBwZXJtaXNzaW9ucyB1bmRlciB0aGlzIExpY2Vuc2UuCi0KLSAgIDguIExpbWl0YXRpb24gb2YgTGlhYmlsaXR5LiBJbiBubyBldmVudCBhbmQgdW5kZXIgbm8gbGVnYWwgdGhlb3J5LAotICAgICAgd2hldGhlciBpbiB0b3J0IChpbmNsdWRpbmcgbmVnbGlnZW5jZSksIGNvbnRyYWN0LCBvciBvdGhlcndpc2UsCi0gICAgICB1bmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgKHN1Y2ggYXMgZGVsaWJlcmF0ZSBhbmQgZ3Jvc3NseQotICAgICAgbmVnbGlnZW50IGFjdHMpIG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzaGFsbCBhbnkgQ29udHJpYnV0b3IgYmUKLSAgICAgIGxpYWJsZSB0byBZb3UgZm9yIGRhbWFnZXMsIGluY2x1ZGluZyBhbnkgZGlyZWN0LCBpbmRpcmVjdCwgc3BlY2lhbCwKLSAgICAgIGluY2lkZW50YWwsIG9yIGNvbnNlcXVlbnRpYWwgZGFtYWdlcyBvZiBhbnkgY2hhcmFjdGVyIGFyaXNpbmcgYXMgYQotICAgICAgcmVzdWx0IG9mIHRoaXMgTGljZW5zZSBvciBvdXQgb2YgdGhlIHVzZSBvciBpbmFiaWxpdHkgdG8gdXNlIHRoZQotICAgICAgV29yayAoaW5jbHVkaW5nIGJ1dCBub3QgbGltaXRlZCB0byBkYW1hZ2VzIGZvciBsb3NzIG9mIGdvb2R3aWxsLAotICAgICAgd29yayBzdG9wcGFnZSwgY29tcHV0ZXIgZmFpbHVyZSBvciBtYWxmdW5jdGlvbiwgb3IgYW55IGFuZCBhbGwKLSAgICAgIG90aGVyIGNvbW1lcmNpYWwgZGFtYWdlcyBvciBsb3NzZXMpLCBldmVuIGlmIHN1Y2ggQ29udHJpYnV0b3IKLSAgICAgIGhhcyBiZWVuIGFkdmlzZWQgb2YgdGhlIHBvc3NpYmlsaXR5IG9mIHN1Y2ggZGFtYWdlcy4KLQotICAgOS4gQWNjZXB0aW5nIFdhcnJhbnR5IG9yIEFkZGl0aW9uYWwgTGlhYmlsaXR5LiBXaGlsZSByZWRpc3RyaWJ1dGluZwotICAgICAgdGhlIFdvcmsgb3IgRGVyaXZhdGl2ZSBXb3JrcyB0aGVyZW9mLCBZb3UgbWF5IGNob29zZSB0byBvZmZlciwKLSAgICAgIGFuZCBjaGFyZ2UgYSBmZWUgZm9yLCBhY2NlcHRhbmNlIG9mIHN1cHBvcnQsIHdhcnJhbnR5LCBpbmRlbW5pdHksCi0gICAgICBvciBvdGhlciBsaWFiaWxpdHkgb2JsaWdhdGlvbnMgYW5kL29yIHJpZ2h0cyBjb25zaXN0ZW50IHdpdGggdGhpcwotICAgICAgTGljZW5zZS4gSG93ZXZlciwgaW4gYWNjZXB0aW5nIHN1Y2ggb2JsaWdhdGlvbnMsIFlvdSBtYXkgYWN0IG9ubHkKLSAgICAgIG9uIFlvdXIgb3duIGJlaGFsZiBhbmQgb24gWW91ciBzb2xlIHJlc3BvbnNpYmlsaXR5LCBub3Qgb24gYmVoYWxmCi0gICAgICBvZiBhbnkgb3RoZXIgQ29udHJpYnV0b3IsIGFuZCBvbmx5IGlmIFlvdSBhZ3JlZSB0byBpbmRlbW5pZnksCi0gICAgICBkZWZlbmQsIGFuZCBob2xkIGVhY2ggQ29udHJpYnV0b3IgaGFybWxlc3MgZm9yIGFueSBsaWFiaWxpdHkKLSAgICAgIGluY3VycmVkIGJ5LCBvciBjbGFpbXMgYXNzZXJ0ZWQgYWdhaW5zdCwgc3VjaCBDb250cmlidXRvciBieSByZWFzb24KLSAgICAgIG9mIHlvdXIgYWNjZXB0aW5nIGFueSBzdWNoIHdhcnJhbnR5IG9yIGFkZGl0aW9uYWwgbGlhYmlsaXR5LgotCi0gICBFTkQgT0YgVEVSTVMgQU5EIENPTkRJVElPTlMKLQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9QYXJjZWwuY3BwIGIvbGlicy91dGlscy9QYXJjZWwuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwZjRiNjQ3Li4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvUGFyY2VsLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDEzNzcgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNkZWZpbmUgTE9HX1RBRyAiUGFyY2VsIgotLy8jZGVmaW5lIExPR19OREVCVUcgMAotCi0jaW5jbHVkZSA8dXRpbHMvUGFyY2VsLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9CaW5kZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9CcEJpbmRlci5oPgotI2luY2x1ZGUgPHV0aWxzL0RlYnVnLmg+Ci0jaW5jbHVkZSA8dXRpbHMvUHJvY2Vzc1N0YXRlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RyaW5nOC5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzE2Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvVGV4dE91dHB1dC5oPgotI2luY2x1ZGUgPHV0aWxzL21pc2MuaD4KLQotI2luY2x1ZGUgPHByaXZhdGUvdXRpbHMvYmluZGVyX21vZHVsZS5oPgotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLQotI2lmbmRlZiBJTlQzMl9NQVgKLSNkZWZpbmUgSU5UMzJfTUFYICgoaW50MzJfdCkoMjE0NzQ4MzY0NykpCi0jZW5kaWYKLQotI2RlZmluZSBMT0dfUkVGUyguLi4pCi0vLyNkZWZpbmUgTE9HX1JFRlMoLi4uKSBMT0coTE9HX0RFQlVHLCAiUGFyY2VsIiwgX19WQV9BUkdTX18pCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZGVmaW5lIFBBRF9TSVpFKHMpICgoKHMpKzMpJn4zKQotCi0vLyBYWFggVGhpcyBjYW4gYmUgbWFkZSBwdWJsaWMgaWYgd2Ugd2FudCB0byBwcm92aWRlCi0vLyBzdXBwb3J0IGZvciB0eXBlZCBkYXRhLgotc3RydWN0IHNtYWxsX2ZsYXRfZGF0YQotewotICAgIHVpbnQzMl90IHR5cGU7Ci0gICAgdWludDMyX3QgZGF0YTsKLX07Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotdm9pZCBhY3F1aXJlX29iamVjdChjb25zdCBzcDxQcm9jZXNzU3RhdGU+JiBwcm9jLAotICAgIGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCYgb2JqLCBjb25zdCB2b2lkKiB3aG8pCi17Ci0gICAgc3dpdGNoIChvYmoudHlwZSkgewotICAgICAgICBjYXNlIEJJTkRFUl9UWVBFX0JJTkRFUjoKLSAgICAgICAgICAgIGlmIChvYmouYmluZGVyKSB7Ci0gICAgICAgICAgICAgICAgTE9HX1JFRlMoIlBhcmNlbCAlcCBhY3F1aXJpbmcgcmVmZXJlbmNlIG9uIGxvY2FsICVwIiwgd2hvLCBvYmouY29va2llKTsKLSAgICAgICAgICAgICAgICBzdGF0aWNfY2FzdDxJQmluZGVyKj4ob2JqLmNvb2tpZSktPmluY1N0cm9uZyh3aG8pOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICBjYXNlIEJJTkRFUl9UWVBFX1dFQUtfQklOREVSOgotICAgICAgICAgICAgaWYgKG9iai5iaW5kZXIpCi0gICAgICAgICAgICAgICAgc3RhdGljX2Nhc3Q8UmVmQmFzZTo6d2Vha3JlZl90eXBlKj4ob2JqLmJpbmRlciktPmluY1dlYWsod2hvKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgY2FzZSBCSU5ERVJfVFlQRV9IQU5ETEU6IHsKLSAgICAgICAgICAgIGNvbnN0IHNwPElCaW5kZXI+IGIgPSBwcm9jLT5nZXRTdHJvbmdQcm94eUZvckhhbmRsZShvYmouaGFuZGxlKTsKLSAgICAgICAgICAgIGlmIChiICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICBMT0dfUkVGUygiUGFyY2VsICVwIGFjcXVpcmluZyByZWZlcmVuY2Ugb24gcmVtb3RlICVwIiwgd2hvLCBiLmdldCgpKTsKLSAgICAgICAgICAgICAgICBiLT5pbmNTdHJvbmcod2hvKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBjYXNlIEJJTkRFUl9UWVBFX1dFQUtfSEFORExFOiB7Ci0gICAgICAgICAgICBjb25zdCB3cDxJQmluZGVyPiBiID0gcHJvYy0+Z2V0V2Vha1Byb3h5Rm9ySGFuZGxlKG9iai5oYW5kbGUpOwotICAgICAgICAgICAgaWYgKGIgIT0gTlVMTCkgYi5nZXRfcmVmcygpLT5pbmNXZWFrKHdobyk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgY2FzZSBCSU5ERVJfVFlQRV9GRDogewotICAgICAgICAgICAgLy8gaW50ZW50aW9uYWxseSBibGFuayAtLSBub3RoaW5nIHRvIGRvIHRvIGFjcXVpcmUgdGhpcywgYnV0IHdlIGRvCi0gICAgICAgICAgICAvLyByZWNvZ25pemUgaXQgYXMgYSBsZWdpdGltYXRlIG9iamVjdCB0eXBlLgotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgTE9HRCgiSW52YWxpZCBvYmplY3QgdHlwZSAweCUwOGx4Iiwgb2JqLnR5cGUpOwotfQotCi12b2lkIHJlbGVhc2Vfb2JqZWN0KGNvbnN0IHNwPFByb2Nlc3NTdGF0ZT4mIHByb2MsCi0gICAgY29uc3QgZmxhdF9iaW5kZXJfb2JqZWN0JiBvYmosIGNvbnN0IHZvaWQqIHdobykKLXsKLSAgICBzd2l0Y2ggKG9iai50eXBlKSB7Ci0gICAgICAgIGNhc2UgQklOREVSX1RZUEVfQklOREVSOgotICAgICAgICAgICAgaWYgKG9iai5iaW5kZXIpIHsKLSAgICAgICAgICAgICAgICBMT0dfUkVGUygiUGFyY2VsICVwIHJlbGVhc2luZyByZWZlcmVuY2Ugb24gbG9jYWwgJXAiLCB3aG8sIG9iai5jb29raWUpOwotICAgICAgICAgICAgICAgIHN0YXRpY19jYXN0PElCaW5kZXIqPihvYmouY29va2llKS0+ZGVjU3Ryb25nKHdobyk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIGNhc2UgQklOREVSX1RZUEVfV0VBS19CSU5ERVI6Ci0gICAgICAgICAgICBpZiAob2JqLmJpbmRlcikKLSAgICAgICAgICAgICAgICBzdGF0aWNfY2FzdDxSZWZCYXNlOjp3ZWFrcmVmX3R5cGUqPihvYmouYmluZGVyKS0+ZGVjV2Vhayh3aG8pOwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICBjYXNlIEJJTkRFUl9UWVBFX0hBTkRMRTogewotICAgICAgICAgICAgY29uc3Qgc3A8SUJpbmRlcj4gYiA9IHByb2MtPmdldFN0cm9uZ1Byb3h5Rm9ySGFuZGxlKG9iai5oYW5kbGUpOwotICAgICAgICAgICAgaWYgKGIgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgIExPR19SRUZTKCJQYXJjZWwgJXAgcmVsZWFzaW5nIHJlZmVyZW5jZSBvbiByZW1vdGUgJXAiLCB3aG8sIGIuZ2V0KCkpOwotICAgICAgICAgICAgICAgIGItPmRlY1N0cm9uZyh3aG8pOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIGNhc2UgQklOREVSX1RZUEVfV0VBS19IQU5ETEU6IHsKLSAgICAgICAgICAgIGNvbnN0IHdwPElCaW5kZXI+IGIgPSBwcm9jLT5nZXRXZWFrUHJveHlGb3JIYW5kbGUob2JqLmhhbmRsZSk7Ci0gICAgICAgICAgICBpZiAoYiAhPSBOVUxMKSBiLmdldF9yZWZzKCktPmRlY1dlYWsod2hvKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBjYXNlIEJJTkRFUl9UWVBFX0ZEOiB7Ci0gICAgICAgICAgICBpZiAob2JqLmNvb2tpZSAhPSAodm9pZCopMCkgY2xvc2Uob2JqLmhhbmRsZSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBMT0dFKCJJbnZhbGlkIG9iamVjdCB0eXBlIDB4JTA4bHgiLCBvYmoudHlwZSk7Ci19Ci0KLWlubGluZSBzdGF0aWMgc3RhdHVzX3QgZmluaXNoX2ZsYXR0ZW5fYmluZGVyKAotICAgIGNvbnN0IHNwPElCaW5kZXI+JiBiaW5kZXIsIGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCYgZmxhdCwgUGFyY2VsKiBvdXQpCi17Ci0gICAgcmV0dXJuIG91dC0+d3JpdGVPYmplY3QoZmxhdCwgZmFsc2UpOwotfQotCi1zdGF0dXNfdCBmbGF0dGVuX2JpbmRlcihjb25zdCBzcDxQcm9jZXNzU3RhdGU+JiBwcm9jLAotICAgIGNvbnN0IHNwPElCaW5kZXI+JiBiaW5kZXIsIFBhcmNlbCogb3V0KQotewotICAgIGZsYXRfYmluZGVyX29iamVjdCBvYmo7Ci0gICAgCi0gICAgb2JqLmZsYWdzID0gMHg3ZiB8IEZMQVRfQklOREVSX0ZMQUdfQUNDRVBUU19GRFM7Ci0gICAgaWYgKGJpbmRlciAhPSBOVUxMKSB7Ci0gICAgICAgIElCaW5kZXIgKmxvY2FsID0gYmluZGVyLT5sb2NhbEJpbmRlcigpOwotICAgICAgICBpZiAoIWxvY2FsKSB7Ci0gICAgICAgICAgICBCcEJpbmRlciAqcHJveHkgPSBiaW5kZXItPnJlbW90ZUJpbmRlcigpOwotICAgICAgICAgICAgaWYgKHByb3h5ID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICBMT0dFKCJudWxsIHByb3h5Iik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjb25zdCBpbnQzMl90IGhhbmRsZSA9IHByb3h5ID8gcHJveHktPmhhbmRsZSgpIDogMDsKLSAgICAgICAgICAgIG9iai50eXBlID0gQklOREVSX1RZUEVfSEFORExFOwotICAgICAgICAgICAgb2JqLmhhbmRsZSA9IGhhbmRsZTsKLSAgICAgICAgICAgIG9iai5jb29raWUgPSBOVUxMOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgb2JqLnR5cGUgPSBCSU5ERVJfVFlQRV9CSU5ERVI7Ci0gICAgICAgICAgICBvYmouYmluZGVyID0gbG9jYWwtPmdldFdlYWtSZWZzKCk7Ci0gICAgICAgICAgICBvYmouY29va2llID0gbG9jYWw7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBvYmoudHlwZSA9IEJJTkRFUl9UWVBFX0JJTkRFUjsKLSAgICAgICAgb2JqLmJpbmRlciA9IE5VTEw7Ci0gICAgICAgIG9iai5jb29raWUgPSBOVUxMOwotICAgIH0KLSAgICAKLSAgICByZXR1cm4gZmluaXNoX2ZsYXR0ZW5fYmluZGVyKGJpbmRlciwgb2JqLCBvdXQpOwotfQotCi1zdGF0dXNfdCBmbGF0dGVuX2JpbmRlcihjb25zdCBzcDxQcm9jZXNzU3RhdGU+JiBwcm9jLAotICAgIGNvbnN0IHdwPElCaW5kZXI+JiBiaW5kZXIsIFBhcmNlbCogb3V0KQotewotICAgIGZsYXRfYmluZGVyX29iamVjdCBvYmo7Ci0gICAgCi0gICAgb2JqLmZsYWdzID0gMHg3ZiB8IEZMQVRfQklOREVSX0ZMQUdfQUNDRVBUU19GRFM7Ci0gICAgaWYgKGJpbmRlciAhPSBOVUxMKSB7Ci0gICAgICAgIHNwPElCaW5kZXI+IHJlYWwgPSBiaW5kZXIucHJvbW90ZSgpOwotICAgICAgICBpZiAocmVhbCAhPSBOVUxMKSB7Ci0gICAgICAgICAgICBJQmluZGVyICpsb2NhbCA9IHJlYWwtPmxvY2FsQmluZGVyKCk7Ci0gICAgICAgICAgICBpZiAoIWxvY2FsKSB7Ci0gICAgICAgICAgICAgICAgQnBCaW5kZXIgKnByb3h5ID0gcmVhbC0+cmVtb3RlQmluZGVyKCk7Ci0gICAgICAgICAgICAgICAgaWYgKHByb3h5ID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgTE9HRSgibnVsbCBwcm94eSIpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBjb25zdCBpbnQzMl90IGhhbmRsZSA9IHByb3h5ID8gcHJveHktPmhhbmRsZSgpIDogMDsKLSAgICAgICAgICAgICAgICBvYmoudHlwZSA9IEJJTkRFUl9UWVBFX1dFQUtfSEFORExFOwotICAgICAgICAgICAgICAgIG9iai5oYW5kbGUgPSBoYW5kbGU7Ci0gICAgICAgICAgICAgICAgb2JqLmNvb2tpZSA9IE5VTEw7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIG9iai50eXBlID0gQklOREVSX1RZUEVfV0VBS19CSU5ERVI7Ci0gICAgICAgICAgICAgICAgb2JqLmJpbmRlciA9IGJpbmRlci5nZXRfcmVmcygpOwotICAgICAgICAgICAgICAgIG9iai5jb29raWUgPSBiaW5kZXIudW5zYWZlX2dldCgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGZpbmlzaF9mbGF0dGVuX2JpbmRlcihyZWFsLCBvYmosIG91dCk7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIC8vIFhYWCBIb3cgdG8gZGVhbD8gIEluIG9yZGVyIHRvIGZsYXR0ZW4gdGhlIGdpdmVuIGJpbmRlciwKLSAgICAgICAgLy8gd2UgbmVlZCB0byBwcm9iZSBpdCBmb3IgaW5mb3JtYXRpb24sIHdoaWNoIHJlcXVpcmVzIGEgcHJpbWFyeQotICAgICAgICAvLyByZWZlcmVuY2UuLi4gIGJ1dCB3ZSBkb24ndCBoYXZlIG9uZS4KLSAgICAgICAgLy8KLSAgICAgICAgLy8gVGhlIE9wZW5CaW5kZXIgaW1wbGVtZW50YXRpb24gdXNlcyBhIGR5bmFtaWNfY2FzdDw+IGhlcmUsCi0gICAgICAgIC8vIGJ1dCB3ZSBjYW4ndCBkbyB0aGF0IHdpdGggdGhlIGRpZmZlcmVudCByZWZlcmVuY2UgY291bnRpbmcKLSAgICAgICAgLy8gaW1wbGVtZW50YXRpb24gd2UgYXJlIHVzaW5nLgotICAgICAgICBMT0dFKCJVbmFibGUgdG8gdW5mbGF0dGVuIEJpbmRlciB3ZWFrIHJlZmVyZW5jZSEiKTsKLSAgICAgICAgb2JqLnR5cGUgPSBCSU5ERVJfVFlQRV9CSU5ERVI7Ci0gICAgICAgIG9iai5iaW5kZXIgPSBOVUxMOwotICAgICAgICBvYmouY29va2llID0gTlVMTDsKLSAgICAgICAgcmV0dXJuIGZpbmlzaF9mbGF0dGVuX2JpbmRlcihOVUxMLCBvYmosIG91dCk7Ci0gICAgCi0gICAgfSBlbHNlIHsKLSAgICAgICAgb2JqLnR5cGUgPSBCSU5ERVJfVFlQRV9CSU5ERVI7Ci0gICAgICAgIG9iai5iaW5kZXIgPSBOVUxMOwotICAgICAgICBvYmouY29va2llID0gTlVMTDsKLSAgICAgICAgcmV0dXJuIGZpbmlzaF9mbGF0dGVuX2JpbmRlcihOVUxMLCBvYmosIG91dCk7Ci0gICAgfQotfQotCi1pbmxpbmUgc3RhdGljIHN0YXR1c190IGZpbmlzaF91bmZsYXR0ZW5fYmluZGVyKAotICAgIEJwQmluZGVyKiBwcm94eSwgY29uc3QgZmxhdF9iaW5kZXJfb2JqZWN0JiBmbGF0LCBjb25zdCBQYXJjZWwmIGluKQotewotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLSAgICAKLXN0YXR1c190IHVuZmxhdHRlbl9iaW5kZXIoY29uc3Qgc3A8UHJvY2Vzc1N0YXRlPiYgcHJvYywKLSAgICBjb25zdCBQYXJjZWwmIGluLCBzcDxJQmluZGVyPiogb3V0KQotewotICAgIGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCogZmxhdCA9IGluLnJlYWRPYmplY3QoZmFsc2UpOwotICAgIAotICAgIGlmIChmbGF0KSB7Ci0gICAgICAgIHN3aXRjaCAoZmxhdC0+dHlwZSkgewotICAgICAgICAgICAgY2FzZSBCSU5ERVJfVFlQRV9CSU5ERVI6Ci0gICAgICAgICAgICAgICAgKm91dCA9IHN0YXRpY19jYXN0PElCaW5kZXIqPihmbGF0LT5jb29raWUpOwotICAgICAgICAgICAgICAgIHJldHVybiBmaW5pc2hfdW5mbGF0dGVuX2JpbmRlcihOVUxMLCAqZmxhdCwgaW4pOwotICAgICAgICAgICAgY2FzZSBCSU5ERVJfVFlQRV9IQU5ETEU6Ci0gICAgICAgICAgICAgICAgKm91dCA9IHByb2MtPmdldFN0cm9uZ1Byb3h5Rm9ySGFuZGxlKGZsYXQtPmhhbmRsZSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZpbmlzaF91bmZsYXR0ZW5fYmluZGVyKAotICAgICAgICAgICAgICAgICAgICBzdGF0aWNfY2FzdDxCcEJpbmRlcio+KG91dC0+Z2V0KCkpLCAqZmxhdCwgaW4pOwotICAgICAgICB9ICAgICAgICAKLSAgICB9Ci0gICAgcmV0dXJuIEJBRF9UWVBFOwotfQotCi1zdGF0dXNfdCB1bmZsYXR0ZW5fYmluZGVyKGNvbnN0IHNwPFByb2Nlc3NTdGF0ZT4mIHByb2MsCi0gICAgY29uc3QgUGFyY2VsJiBpbiwgd3A8SUJpbmRlcj4qIG91dCkKLXsKLSAgICBjb25zdCBmbGF0X2JpbmRlcl9vYmplY3QqIGZsYXQgPSBpbi5yZWFkT2JqZWN0KGZhbHNlKTsKLSAgICAKLSAgICBpZiAoZmxhdCkgewotICAgICAgICBzd2l0Y2ggKGZsYXQtPnR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgQklOREVSX1RZUEVfQklOREVSOgotICAgICAgICAgICAgICAgICpvdXQgPSBzdGF0aWNfY2FzdDxJQmluZGVyKj4oZmxhdC0+Y29va2llKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmluaXNoX3VuZmxhdHRlbl9iaW5kZXIoTlVMTCwgKmZsYXQsIGluKTsKLSAgICAgICAgICAgIGNhc2UgQklOREVSX1RZUEVfV0VBS19CSU5ERVI6Ci0gICAgICAgICAgICAgICAgaWYgKGZsYXQtPmJpbmRlciAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgICAgIG91dC0+c2V0X29iamVjdF9hbmRfcmVmcygKLSAgICAgICAgICAgICAgICAgICAgICAgIHN0YXRpY19jYXN0PElCaW5kZXIqPihmbGF0LT5jb29raWUpLAotICAgICAgICAgICAgICAgICAgICAgICAgc3RhdGljX2Nhc3Q8UmVmQmFzZTo6d2Vha3JlZl90eXBlKj4oZmxhdC0+YmluZGVyKSk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgKm91dCA9IE5VTEw7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBmaW5pc2hfdW5mbGF0dGVuX2JpbmRlcihOVUxMLCAqZmxhdCwgaW4pOwotICAgICAgICAgICAgY2FzZSBCSU5ERVJfVFlQRV9IQU5ETEU6Ci0gICAgICAgICAgICBjYXNlIEJJTkRFUl9UWVBFX1dFQUtfSEFORExFOgotICAgICAgICAgICAgICAgICpvdXQgPSBwcm9jLT5nZXRXZWFrUHJveHlGb3JIYW5kbGUoZmxhdC0+aGFuZGxlKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmluaXNoX3VuZmxhdHRlbl9iaW5kZXIoCi0gICAgICAgICAgICAgICAgICAgIHN0YXRpY19jYXN0PEJwQmluZGVyKj4ob3V0LT51bnNhZmVfZ2V0KCkpLCAqZmxhdCwgaW4pOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBCQURfVFlQRTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLVBhcmNlbDo6UGFyY2VsKCkKLXsKLSAgICBpbml0U3RhdGUoKTsKLX0KLQotUGFyY2VsOjp+UGFyY2VsKCkKLXsKLSAgICBmcmVlRGF0YU5vSW5pdCgpOwotfQotCi1jb25zdCB1aW50OF90KiBQYXJjZWw6OmRhdGEoKSBjb25zdAotewotICAgIHJldHVybiBtRGF0YTsKLX0KLQotc2l6ZV90IFBhcmNlbDo6ZGF0YVNpemUoKSBjb25zdAotewotICAgIHJldHVybiAobURhdGFTaXplID4gbURhdGFQb3MgPyBtRGF0YVNpemUgOiBtRGF0YVBvcyk7Ci19Ci0KLXNpemVfdCBQYXJjZWw6OmRhdGFBdmFpbCgpIGNvbnN0Ci17Ci0gICAgLy8gVE9ETzogZGVjaWRlIHdoYXQgdG8gZG8gYWJvdXQgdGhlIHBvc3NpYmlsaXR5IHRoYXQgdGhpcyBjYW4KLSAgICAvLyByZXBvcnQgYW4gYXZhaWxhYmxlLWRhdGEgc2l6ZSB0aGF0IGV4Y2VlZHMgYSBKYXZhIGludCdzIG1heAotICAgIC8vIHBvc2l0aXZlIHZhbHVlLCBjYXVzaW5nIGhhdm9jLiAgRm9ydHVuYXRlbHkgdGhpcyB3aWxsIG9ubHkKLSAgICAvLyBoYXBwZW4gaWYgc29tZW9uZSBjb25zdHJ1Y3RzIGEgUGFyY2VsIGNvbnRhaW5pbmcgbW9yZSB0aGFuIHR3bwotICAgIC8vIGdpZ2FieXRlcyBvZiBkYXRhLCB3aGljaCBvbiB0eXBpY2FsIHBob25lIGhhcmR3YXJlIGlzIHNpbXBseQotICAgIC8vIG5vdCBwb3NzaWJsZS4KLSAgICByZXR1cm4gZGF0YVNpemUoKSAtIGRhdGFQb3NpdGlvbigpOwotfQotCi1zaXplX3QgUGFyY2VsOjpkYXRhUG9zaXRpb24oKSBjb25zdAotewotICAgIHJldHVybiBtRGF0YVBvczsKLX0KLQotc2l6ZV90IFBhcmNlbDo6ZGF0YUNhcGFjaXR5KCkgY29uc3QKLXsKLSAgICByZXR1cm4gbURhdGFDYXBhY2l0eTsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjpzZXREYXRhU2l6ZShzaXplX3Qgc2l6ZSkKLXsKLSAgICBzdGF0dXNfdCBlcnI7Ci0gICAgZXJyID0gY29udGludWVXcml0ZShzaXplKTsKLSAgICBpZiAoZXJyID09IE5PX0VSUk9SKSB7Ci0gICAgICAgIG1EYXRhU2l6ZSA9IHNpemU7Ci0gICAgICAgIExPR1YoInNldERhdGFTaXplIFNldHRpbmcgZGF0YSBzaXplIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVNpemUpOwotICAgIH0KLSAgICByZXR1cm4gZXJyOwotfQotCi12b2lkIFBhcmNlbDo6c2V0RGF0YVBvc2l0aW9uKHNpemVfdCBwb3MpIGNvbnN0Ci17Ci0gICAgbURhdGFQb3MgPSBwb3M7Ci0gICAgbU5leHRPYmplY3RIaW50ID0gMDsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjpzZXREYXRhQ2FwYWNpdHkoc2l6ZV90IHNpemUpCi17Ci0gICAgaWYgKHNpemUgPiBtRGF0YVNpemUpIHJldHVybiBjb250aW51ZVdyaXRlKHNpemUpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjpzZXREYXRhKGNvbnN0IHVpbnQ4X3QqIGJ1ZmZlciwgc2l6ZV90IGxlbikKLXsKLSAgICBzdGF0dXNfdCBlcnIgPSByZXN0YXJ0V3JpdGUobGVuKTsKLSAgICBpZiAoZXJyID09IE5PX0VSUk9SKSB7Ci0gICAgICAgIG1lbWNweShjb25zdF9jYXN0PHVpbnQ4X3QqPihkYXRhKCkpLCBidWZmZXIsIGxlbik7Ci0gICAgICAgIG1EYXRhU2l6ZSA9IGxlbjsKLSAgICAgICAgbUZkc0tub3duID0gZmFsc2U7Ci0gICAgfQotICAgIHJldHVybiBlcnI7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6YXBwZW5kRnJvbShQYXJjZWwgKnBhcmNlbCwgc2l6ZV90IG9mZnNldCwgc2l6ZV90IGxlbikKLXsKLSAgICBjb25zdCBzcDxQcm9jZXNzU3RhdGU+IHByb2MoUHJvY2Vzc1N0YXRlOjpzZWxmKCkpOwotICAgIHN0YXR1c190IGVycjsKLSAgICB1aW50OF90ICpkYXRhID0gcGFyY2VsLT5tRGF0YTsKLSAgICBzaXplX3QgKm9iamVjdHMgPSBwYXJjZWwtPm1PYmplY3RzOwotICAgIHNpemVfdCBzaXplID0gcGFyY2VsLT5tT2JqZWN0c1NpemU7Ci0gICAgaW50IHN0YXJ0UG9zID0gbURhdGFQb3M7Ci0gICAgaW50IGZpcnN0SW5kZXggPSAtMSwgbGFzdEluZGV4ID0gLTI7Ci0KLSAgICBpZiAobGVuID09IDApIHsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLQotICAgIC8vIHJhbmdlIGNoZWNrcyBhZ2FpbnN0IHRoZSBzb3VyY2UgcGFyY2VsIHNpemUKLSAgICBpZiAoKG9mZnNldCA+IHBhcmNlbC0+bURhdGFTaXplKQotICAgICAgICAgICAgfHwgKGxlbiA+IHBhcmNlbC0+bURhdGFTaXplKQotICAgICAgICAgICAgfHwgKG9mZnNldCArIGxlbiA+IHBhcmNlbC0+bURhdGFTaXplKSkgewotICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOwotICAgIH0KLQotICAgIC8vIENvdW50IG9iamVjdHMgaW4gcmFuZ2UKLSAgICBmb3IgKGludCBpID0gMDsgaSA8IChpbnQpIHNpemU7IGkrKykgewotICAgICAgICBzaXplX3Qgb2ZmID0gb2JqZWN0c1tpXTsKLSAgICAgICAgaWYgKChvZmYgPj0gb2Zmc2V0KSAmJiAob2ZmIDwgb2Zmc2V0ICsgbGVuKSkgewotICAgICAgICAgICAgaWYgKGZpcnN0SW5kZXggPT0gLTEpIHsKLSAgICAgICAgICAgICAgICBmaXJzdEluZGV4ID0gaTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGxhc3RJbmRleCA9IGk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgaW50IG51bU9iamVjdHMgPSBsYXN0SW5kZXggLSBmaXJzdEluZGV4ICsgMTsKLQotICAgIC8vIGdyb3cgZGF0YQotICAgIGVyciA9IGdyb3dEYXRhKGxlbik7Ci0gICAgaWYgKGVyciAhPSBOT19FUlJPUikgewotICAgICAgICByZXR1cm4gZXJyOwotICAgIH0KLQotICAgIC8vIGFwcGVuZCBkYXRhCi0gICAgbWVtY3B5KG1EYXRhICsgbURhdGFQb3MsIGRhdGEgKyBvZmZzZXQsIGxlbik7Ci0gICAgbURhdGFQb3MgKz0gbGVuOwotICAgIG1EYXRhU2l6ZSArPSBsZW47Ci0KLSAgICBpZiAobnVtT2JqZWN0cyA+IDApIHsKLSAgICAgICAgLy8gZ3JvdyBvYmplY3RzCi0gICAgICAgIGlmIChtT2JqZWN0c0NhcGFjaXR5IDwgbU9iamVjdHNTaXplICsgbnVtT2JqZWN0cykgewotICAgICAgICAgICAgaW50IG5ld1NpemUgPSAoKG1PYmplY3RzU2l6ZSArIG51bU9iamVjdHMpKjMpLzI7Ci0gICAgICAgICAgICBzaXplX3QgKm9iamVjdHMgPQotICAgICAgICAgICAgICAgIChzaXplX3QqKXJlYWxsb2MobU9iamVjdHMsIG5ld1NpemUqc2l6ZW9mKHNpemVfdCkpOwotICAgICAgICAgICAgaWYgKG9iamVjdHMgPT0gKHNpemVfdCopMCkgewotICAgICAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBtT2JqZWN0cyA9IG9iamVjdHM7Ci0gICAgICAgICAgICBtT2JqZWN0c0NhcGFjaXR5ID0gbmV3U2l6ZTsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgLy8gYXBwZW5kIGFuZCBhY3F1aXJlIG9iamVjdHMKLSAgICAgICAgaW50IGlkeCA9IG1PYmplY3RzU2l6ZTsKLSAgICAgICAgZm9yIChpbnQgaSA9IGZpcnN0SW5kZXg7IGkgPD0gbGFzdEluZGV4OyBpKyspIHsKLSAgICAgICAgICAgIHNpemVfdCBvZmYgPSBvYmplY3RzW2ldIC0gb2Zmc2V0ICsgc3RhcnRQb3M7Ci0gICAgICAgICAgICBtT2JqZWN0c1tpZHgrK10gPSBvZmY7Ci0gICAgICAgICAgICBtT2JqZWN0c1NpemUrKzsKLQotICAgICAgICAgICAgY29uc3QgZmxhdF9iaW5kZXJfb2JqZWN0KiBmbGF0Ci0gICAgICAgICAgICAgICAgPSByZWludGVycHJldF9jYXN0PGZsYXRfYmluZGVyX29iamVjdCo+KG1EYXRhICsgb2ZmKTsKLSAgICAgICAgICAgIGFjcXVpcmVfb2JqZWN0KHByb2MsICpmbGF0LCB0aGlzKTsKLQotICAgICAgICAgICAgLy8gdGFrZSBub3RlIGlmIHRoZSBvYmplY3QgaXMgYSBmaWxlIGRlc2NyaXB0b3IKLSAgICAgICAgICAgIGlmIChmbGF0LT50eXBlID09IEJJTkRFUl9UWVBFX0ZEKSB7Ci0gICAgICAgICAgICAgICAgbUhhc0ZkcyA9IG1GZHNLbm93biA9IHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLWJvb2wgUGFyY2VsOjpoYXNGaWxlRGVzY3JpcHRvcnMoKSBjb25zdAotewotICAgIGlmICghbUZkc0tub3duKSB7Ci0gICAgICAgIHNjYW5Gb3JGZHMoKTsKLSAgICB9Ci0gICAgcmV0dXJuIG1IYXNGZHM7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6d3JpdGVJbnRlcmZhY2VUb2tlbihjb25zdCBTdHJpbmcxNiYgaW50ZXJmYWNlKQotewotICAgIC8vIGN1cnJlbnRseSB0aGUgaW50ZXJmYWNlIGlkZW50aWZpY2F0aW9uIHRva2VuIGlzIGp1c3QgaXRzIG5hbWUgYXMgYSBzdHJpbmcKLSAgICByZXR1cm4gd3JpdGVTdHJpbmcxNihpbnRlcmZhY2UpOwotfQotCi1ib29sIFBhcmNlbDo6ZW5mb3JjZUludGVyZmFjZShjb25zdCBTdHJpbmcxNiYgaW50ZXJmYWNlKSBjb25zdAotewotICAgIFN0cmluZzE2IHN0ciA9IHJlYWRTdHJpbmcxNigpOwotICAgIGlmIChzdHIgPT0gaW50ZXJmYWNlKSB7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIExPR1coIioqKiogZW5mb3JjZUludGVyZmFjZSgpIGV4cGVjdGVkICclcycgYnV0IHJlYWQgJyVzJ1xuIiwKLSAgICAgICAgICAgICAgICBTdHJpbmc4KGludGVyZmFjZSkuc3RyaW5nKCksIFN0cmluZzgoc3RyKS5zdHJpbmcoKSk7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci19IAotCi1jb25zdCBzaXplX3QqIFBhcmNlbDo6b2JqZWN0cygpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1PYmplY3RzOwotfQotCi1zaXplX3QgUGFyY2VsOjpvYmplY3RzQ291bnQoKSBjb25zdAotewotICAgIHJldHVybiBtT2JqZWN0c1NpemU7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6ZXJyb3JDaGVjaygpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1FcnJvcjsKLX0KLQotdm9pZCBQYXJjZWw6OnNldEVycm9yKHN0YXR1c190IGVycikKLXsKLSAgICBtRXJyb3IgPSBlcnI7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6ZmluaXNoV3JpdGUoc2l6ZV90IGxlbikKLXsKLSAgICAvL3ByaW50ZigiRmluaXNoIHdyaXRlIG9mICVkXG4iLCBsZW4pOwotICAgIG1EYXRhUG9zICs9IGxlbjsKLSAgICBMT0dWKCJmaW5pc2hXcml0ZSBTZXR0aW5nIGRhdGEgcG9zIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVBvcyk7Ci0gICAgaWYgKG1EYXRhUG9zID4gbURhdGFTaXplKSB7Ci0gICAgICAgIG1EYXRhU2l6ZSA9IG1EYXRhUG9zOwotICAgICAgICBMT0dWKCJmaW5pc2hXcml0ZSBTZXR0aW5nIGRhdGEgc2l6ZSBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFTaXplKTsKLSAgICB9Ci0gICAgLy9wcmludGYoIk5ldyBwb3M9JWQsIHNpemU9JWRcbiIsIG1EYXRhUG9zLCBtRGF0YVNpemUpOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjp3cml0ZVVucGFkZGVkKGNvbnN0IHZvaWQqIGRhdGEsIHNpemVfdCBsZW4pCi17Ci0gICAgc2l6ZV90IGVuZCA9IG1EYXRhUG9zICsgbGVuOwotICAgIGlmIChlbmQgPCBtRGF0YVBvcykgewotICAgICAgICAvLyBpbnRlZ2VyIG92ZXJmbG93Ci0gICAgICAgIHJldHVybiBCQURfVkFMVUU7Ci0gICAgfQotCi0gICAgaWYgKGVuZCA8PSBtRGF0YUNhcGFjaXR5KSB7Ci1yZXN0YXJ0X3dyaXRlOgotICAgICAgICBtZW1jcHkobURhdGErbURhdGFQb3MsIGRhdGEsIGxlbik7Ci0gICAgICAgIHJldHVybiBmaW5pc2hXcml0ZShsZW4pOwotICAgIH0KLQotICAgIHN0YXR1c190IGVyciA9IGdyb3dEYXRhKGxlbik7Ci0gICAgaWYgKGVyciA9PSBOT19FUlJPUikgZ290byByZXN0YXJ0X3dyaXRlOwotICAgIHJldHVybiBlcnI7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6d3JpdGUoY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IGxlbikKLXsKLSAgICB2b2lkKiBjb25zdCBkID0gd3JpdGVJbnBsYWNlKGxlbik7Ci0gICAgaWYgKGQpIHsKLSAgICAgICAgbWVtY3B5KGQsIGRhdGEsIGxlbik7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIG1FcnJvcjsKLX0KLQotdm9pZCogUGFyY2VsOjp3cml0ZUlucGxhY2Uoc2l6ZV90IGxlbikKLXsKLSAgICBjb25zdCBzaXplX3QgcGFkZGVkID0gUEFEX1NJWkUobGVuKTsKLQotICAgIC8vIHNhbml0eSBjaGVjayBmb3IgaW50ZWdlciBvdmVyZmxvdwotICAgIGlmIChtRGF0YVBvcytwYWRkZWQgPCBtRGF0YVBvcykgewotICAgICAgICByZXR1cm4gTlVMTDsKLSAgICB9Ci0KLSAgICBpZiAoKG1EYXRhUG9zK3BhZGRlZCkgPD0gbURhdGFDYXBhY2l0eSkgewotcmVzdGFydF93cml0ZToKLSAgICAgICAgLy9wcmludGYoIldyaXRpbmcgJWxkIGJ5dGVzLCBwYWRkZWQgdG8gJWxkXG4iLCBsZW4sIHBhZGRlZCk7Ci0gICAgICAgIHVpbnQ4X3QqIGNvbnN0IGRhdGEgPSBtRGF0YSttRGF0YVBvczsKLQotICAgICAgICAvLyBOZWVkIHRvIHBhZCBhdCBlbmQ/Ci0gICAgICAgIGlmIChwYWRkZWQgIT0gbGVuKSB7Ci0jaWYgQllURV9PUkRFUiA9PSBCSUdfRU5ESUFOCi0gICAgICAgICAgICBzdGF0aWMgY29uc3QgdWludDMyX3QgbWFza1s0XSA9IHsKLSAgICAgICAgICAgICAgICAweDAwMDAwMDAwLCAweGZmZmZmZjAwLCAweGZmZmYwMDAwLCAweGZmMDAwMDAwCi0gICAgICAgICAgICB9OwotI2VuZGlmCi0jaWYgQllURV9PUkRFUiA9PSBMSVRUTEVfRU5ESUFOCi0gICAgICAgICAgICBzdGF0aWMgY29uc3QgdWludDMyX3QgbWFza1s0XSA9IHsKLSAgICAgICAgICAgICAgICAweDAwMDAwMDAwLCAweDAwZmZmZmZmLCAweDAwMDBmZmZmLCAweDAwMDAwMGZmCi0gICAgICAgICAgICB9OwotI2VuZGlmCi0gICAgICAgICAgICAvL3ByaW50ZigiQXBwbHlpbmcgcGFkIG1hc2s6ICVwIHRvICVwXG4iLCAodm9pZCopbWFza1twYWRkZWQtbGVuXSwKLSAgICAgICAgICAgIC8vICAgICpyZWludGVycHJldF9jYXN0PHZvaWQqKj4oZGF0YStwYWRkZWQtNCkpOwotICAgICAgICAgICAgKnJlaW50ZXJwcmV0X2Nhc3Q8dWludDMyX3QqPihkYXRhK3BhZGRlZC00KSAmPSBtYXNrW3BhZGRlZC1sZW5dOwotICAgICAgICB9Ci0KLSAgICAgICAgZmluaXNoV3JpdGUocGFkZGVkKTsKLSAgICAgICAgcmV0dXJuIGRhdGE7Ci0gICAgfQotCi0gICAgc3RhdHVzX3QgZXJyID0gZ3Jvd0RhdGEocGFkZGVkKTsKLSAgICBpZiAoZXJyID09IE5PX0VSUk9SKSBnb3RvIHJlc3RhcnRfd3JpdGU7Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6d3JpdGVJbnQzMihpbnQzMl90IHZhbCkKLXsKLSAgICBpZiAoKG1EYXRhUG9zK3NpemVvZih2YWwpKSA8PSBtRGF0YUNhcGFjaXR5KSB7Ci1yZXN0YXJ0X3dyaXRlOgotICAgICAgICAqcmVpbnRlcnByZXRfY2FzdDxpbnQzMl90Kj4obURhdGErbURhdGFQb3MpID0gdmFsOwotICAgICAgICByZXR1cm4gZmluaXNoV3JpdGUoc2l6ZW9mKHZhbCkpOwotICAgIH0KLQotICAgIHN0YXR1c190IGVyciA9IGdyb3dEYXRhKHNpemVvZih2YWwpKTsKLSAgICBpZiAoZXJyID09IE5PX0VSUk9SKSBnb3RvIHJlc3RhcnRfd3JpdGU7Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjp3cml0ZUludDY0KGludDY0X3QgdmFsKQotewotICAgIGlmICgobURhdGFQb3Mrc2l6ZW9mKHZhbCkpIDw9IG1EYXRhQ2FwYWNpdHkpIHsKLXJlc3RhcnRfd3JpdGU6Ci0gICAgICAgICpyZWludGVycHJldF9jYXN0PGludDY0X3QqPihtRGF0YSttRGF0YVBvcykgPSB2YWw7Ci0gICAgICAgIHJldHVybiBmaW5pc2hXcml0ZShzaXplb2YodmFsKSk7Ci0gICAgfQotCi0gICAgc3RhdHVzX3QgZXJyID0gZ3Jvd0RhdGEoc2l6ZW9mKHZhbCkpOwotICAgIGlmIChlcnIgPT0gTk9fRVJST1IpIGdvdG8gcmVzdGFydF93cml0ZTsKLSAgICByZXR1cm4gZXJyOwotfQotCi1zdGF0dXNfdCBQYXJjZWw6OndyaXRlRmxvYXQoZmxvYXQgdmFsKQotewotICAgIGlmICgobURhdGFQb3Mrc2l6ZW9mKHZhbCkpIDw9IG1EYXRhQ2FwYWNpdHkpIHsKLXJlc3RhcnRfd3JpdGU6Ci0gICAgICAgICpyZWludGVycHJldF9jYXN0PGZsb2F0Kj4obURhdGErbURhdGFQb3MpID0gdmFsOwotICAgICAgICByZXR1cm4gZmluaXNoV3JpdGUoc2l6ZW9mKHZhbCkpOwotICAgIH0KLQotICAgIHN0YXR1c190IGVyciA9IGdyb3dEYXRhKHNpemVvZih2YWwpKTsKLSAgICBpZiAoZXJyID09IE5PX0VSUk9SKSBnb3RvIHJlc3RhcnRfd3JpdGU7Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjp3cml0ZURvdWJsZShkb3VibGUgdmFsKQotewotICAgIGlmICgobURhdGFQb3Mrc2l6ZW9mKHZhbCkpIDw9IG1EYXRhQ2FwYWNpdHkpIHsKLXJlc3RhcnRfd3JpdGU6Ci0gICAgICAgICpyZWludGVycHJldF9jYXN0PGRvdWJsZSo+KG1EYXRhK21EYXRhUG9zKSA9IHZhbDsKLSAgICAgICAgcmV0dXJuIGZpbmlzaFdyaXRlKHNpemVvZih2YWwpKTsKLSAgICB9Ci0KLSAgICBzdGF0dXNfdCBlcnIgPSBncm93RGF0YShzaXplb2YodmFsKSk7Ci0gICAgaWYgKGVyciA9PSBOT19FUlJPUikgZ290byByZXN0YXJ0X3dyaXRlOwotICAgIHJldHVybiBlcnI7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6d3JpdGVDU3RyaW5nKGNvbnN0IGNoYXIqIHN0cikKLXsKLSAgICByZXR1cm4gd3JpdGUoc3RyLCBzdHJsZW4oc3RyKSsxKTsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjp3cml0ZVN0cmluZzgoY29uc3QgU3RyaW5nOCYgc3RyKQotewotICAgIHN0YXR1c190IGVyciA9IHdyaXRlSW50MzIoc3RyLmJ5dGVzKCkpOwotICAgIGlmIChlcnIgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgZXJyID0gd3JpdGUoc3RyLnN0cmluZygpLCBzdHIuYnl0ZXMoKSsxKTsKLSAgICB9Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjp3cml0ZVN0cmluZzE2KGNvbnN0IFN0cmluZzE2JiBzdHIpCi17Ci0gICAgcmV0dXJuIHdyaXRlU3RyaW5nMTYoc3RyLnN0cmluZygpLCBzdHIuc2l6ZSgpKTsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjp3cml0ZVN0cmluZzE2KGNvbnN0IGNoYXIxNl90KiBzdHIsIHNpemVfdCBsZW4pCi17Ci0gICAgaWYgKHN0ciA9PSBOVUxMKSByZXR1cm4gd3JpdGVJbnQzMigtMSk7Ci0gICAgCi0gICAgc3RhdHVzX3QgZXJyID0gd3JpdGVJbnQzMihsZW4pOwotICAgIGlmIChlcnIgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgbGVuICo9IHNpemVvZihjaGFyMTZfdCk7Ci0gICAgICAgIHVpbnQ4X3QqIGRhdGEgPSAodWludDhfdCopd3JpdGVJbnBsYWNlKGxlbitzaXplb2YoY2hhcjE2X3QpKTsKLSAgICAgICAgaWYgKGRhdGEpIHsKLSAgICAgICAgICAgIG1lbWNweShkYXRhLCBzdHIsIGxlbik7Ci0gICAgICAgICAgICAqcmVpbnRlcnByZXRfY2FzdDxjaGFyMTZfdCo+KGRhdGErbGVuKSA9IDA7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0KLSAgICAgICAgZXJyID0gbUVycm9yOwotICAgIH0KLSAgICByZXR1cm4gZXJyOwotfQotCi1zdGF0dXNfdCBQYXJjZWw6OndyaXRlU3Ryb25nQmluZGVyKGNvbnN0IHNwPElCaW5kZXI+JiB2YWwpCi17Ci0gICAgcmV0dXJuIGZsYXR0ZW5fYmluZGVyKFByb2Nlc3NTdGF0ZTo6c2VsZigpLCB2YWwsIHRoaXMpOwotfQotCi1zdGF0dXNfdCBQYXJjZWw6OndyaXRlV2Vha0JpbmRlcihjb25zdCB3cDxJQmluZGVyPiYgdmFsKQotewotICAgIHJldHVybiBmbGF0dGVuX2JpbmRlcihQcm9jZXNzU3RhdGU6OnNlbGYoKSwgdmFsLCB0aGlzKTsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjp3cml0ZU5hdGl2ZUhhbmRsZShjb25zdCBuYXRpdmVfaGFuZGxlJiBoYW5kbGUpCi17Ci0gICAgaWYgKGhhbmRsZS52ZXJzaW9uICE9IHNpemVvZihuYXRpdmVfaGFuZGxlKSkKLSAgICAgICAgcmV0dXJuIEJBRF9UWVBFOwotCi0gICAgc3RhdHVzX3QgZXJyOwotICAgIGVyciA9IHdyaXRlSW50MzIoaGFuZGxlLm51bUZkcyk7Ci0gICAgaWYgKGVyciAhPSBOT19FUlJPUikgcmV0dXJuIGVycjsKLQotICAgIGVyciA9IHdyaXRlSW50MzIoaGFuZGxlLm51bUludHMpOwotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHJldHVybiBlcnI7Ci0KLSAgICBmb3IgKGludCBpPTAgOyBlcnI9PU5PX0VSUk9SICYmIGk8aGFuZGxlLm51bUZkcyA7IGkrKykKLSAgICAgICAgZXJyID0gd3JpdGVEdXBGaWxlRGVzY3JpcHRvcihoYW5kbGUuZGF0YVtpXSk7Ci0KLSAgICBpZiAoZXJyICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgIExPR0QoIndyaXRlIG5hdGl2ZSBoYW5kbGUsIHdyaXRlIGR1cCBmZCBmYWlsZWQiKTsKLSAgICAgICAgcmV0dXJuIGVycjsKLSAgICB9Ci0KLSAgICBlcnIgPSB3cml0ZShoYW5kbGUuZGF0YSArIGhhbmRsZS5udW1GZHMsIHNpemVvZihpbnQpKmhhbmRsZS5udW1JbnRzKTsKLQotICAgIHJldHVybiBlcnI7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6d3JpdGVGaWxlRGVzY3JpcHRvcihpbnQgZmQpCi17Ci0gICAgZmxhdF9iaW5kZXJfb2JqZWN0IG9iajsKLSAgICBvYmoudHlwZSA9IEJJTkRFUl9UWVBFX0ZEOwotICAgIG9iai5mbGFncyA9IDB4N2YgfCBGTEFUX0JJTkRFUl9GTEFHX0FDQ0VQVFNfRkRTOwotICAgIG9iai5oYW5kbGUgPSBmZDsKLSAgICBvYmouY29va2llID0gKHZvaWQqKTA7Ci0gICAgcmV0dXJuIHdyaXRlT2JqZWN0KG9iaiwgdHJ1ZSk7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6d3JpdGVEdXBGaWxlRGVzY3JpcHRvcihpbnQgZmQpCi17Ci0gICAgZmxhdF9iaW5kZXJfb2JqZWN0IG9iajsKLSAgICBvYmoudHlwZSA9IEJJTkRFUl9UWVBFX0ZEOwotICAgIG9iai5mbGFncyA9IDB4N2YgfCBGTEFUX0JJTkRFUl9GTEFHX0FDQ0VQVFNfRkRTOwotICAgIG9iai5oYW5kbGUgPSBkdXAoZmQpOwotICAgIG9iai5jb29raWUgPSAodm9pZCopMTsKLSAgICByZXR1cm4gd3JpdGVPYmplY3Qob2JqLCB0cnVlKTsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjp3cml0ZU9iamVjdChjb25zdCBmbGF0X2JpbmRlcl9vYmplY3QmIHZhbCwgYm9vbCBudWxsTWV0YURhdGEpCi17Ci0gICAgY29uc3QgYm9vbCBlbm91Z2hEYXRhID0gKG1EYXRhUG9zK3NpemVvZih2YWwpKSA8PSBtRGF0YUNhcGFjaXR5OwotICAgIGNvbnN0IGJvb2wgZW5vdWdoT2JqZWN0cyA9IG1PYmplY3RzU2l6ZSA8IG1PYmplY3RzQ2FwYWNpdHk7Ci0gICAgaWYgKGVub3VnaERhdGEgJiYgZW5vdWdoT2JqZWN0cykgewotcmVzdGFydF93cml0ZToKLSAgICAgICAgKnJlaW50ZXJwcmV0X2Nhc3Q8ZmxhdF9iaW5kZXJfb2JqZWN0Kj4obURhdGErbURhdGFQb3MpID0gdmFsOwotICAgICAgICAKLSAgICAgICAgLy8gTmVlZCB0byB3cml0ZSBtZXRhLWRhdGE/Ci0gICAgICAgIGlmIChudWxsTWV0YURhdGEgfHwgdmFsLmJpbmRlciAhPSBOVUxMKSB7Ci0gICAgICAgICAgICBtT2JqZWN0c1ttT2JqZWN0c1NpemVdID0gbURhdGFQb3M7Ci0gICAgICAgICAgICBhY3F1aXJlX29iamVjdChQcm9jZXNzU3RhdGU6OnNlbGYoKSwgdmFsLCB0aGlzKTsKLSAgICAgICAgICAgIG1PYmplY3RzU2l6ZSsrOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICAvLyByZW1lbWJlciBpZiBpdCdzIGEgZmlsZSBkZXNjcmlwdG9yCi0gICAgICAgIGlmICh2YWwudHlwZSA9PSBCSU5ERVJfVFlQRV9GRCkgewotICAgICAgICAgICAgbUhhc0ZkcyA9IG1GZHNLbm93biA9IHRydWU7Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZmluaXNoV3JpdGUoc2l6ZW9mKGZsYXRfYmluZGVyX29iamVjdCkpOwotICAgIH0KLQotICAgIGlmICghZW5vdWdoRGF0YSkgewotICAgICAgICBjb25zdCBzdGF0dXNfdCBlcnIgPSBncm93RGF0YShzaXplb2YodmFsKSk7Ci0gICAgICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHJldHVybiBlcnI7Ci0gICAgfQotICAgIGlmICghZW5vdWdoT2JqZWN0cykgewotICAgICAgICBzaXplX3QgbmV3U2l6ZSA9ICgobU9iamVjdHNTaXplKzIpKjMpLzI7Ci0gICAgICAgIHNpemVfdCogb2JqZWN0cyA9IChzaXplX3QqKXJlYWxsb2MobU9iamVjdHMsIG5ld1NpemUqc2l6ZW9mKHNpemVfdCkpOwotICAgICAgICBpZiAob2JqZWN0cyA9PSBOVUxMKSByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICBtT2JqZWN0cyA9IG9iamVjdHM7Ci0gICAgICAgIG1PYmplY3RzQ2FwYWNpdHkgPSBuZXdTaXplOwotICAgIH0KLSAgICAKLSAgICBnb3RvIHJlc3RhcnRfd3JpdGU7Ci19Ci0KLQotdm9pZCBQYXJjZWw6OnJlbW92ZShzaXplX3Qgc3RhcnQsIHNpemVfdCBhbXQpCi17Ci0gICAgTE9HX0FMV0FZU19GQVRBTCgiUGFyY2VsOjpyZW1vdmUoKSBub3QgeWV0IGltcGxlbWVudGVkISIpOwotfQotCi1zdGF0dXNfdCBQYXJjZWw6OnJlYWQodm9pZCogb3V0RGF0YSwgc2l6ZV90IGxlbikgY29uc3QKLXsKLSAgICBpZiAoKG1EYXRhUG9zK1BBRF9TSVpFKGxlbikpID49IG1EYXRhUG9zICYmIChtRGF0YVBvcytQQURfU0laRShsZW4pKSA8PSBtRGF0YVNpemUpIHsKLSAgICAgICAgbWVtY3B5KG91dERhdGEsIG1EYXRhK21EYXRhUG9zLCBsZW4pOwotICAgICAgICBtRGF0YVBvcyArPSBQQURfU0laRShsZW4pOwotICAgICAgICBMT0dWKCJyZWFkIFNldHRpbmcgZGF0YSBwb3Mgb2YgJXAgdG8gJWRcbiIsIHRoaXMsIG1EYXRhUG9zKTsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLSAgICByZXR1cm4gTk9UX0VOT1VHSF9EQVRBOwotfQotCi1jb25zdCB2b2lkKiBQYXJjZWw6OnJlYWRJbnBsYWNlKHNpemVfdCBsZW4pIGNvbnN0Ci17Ci0gICAgaWYgKChtRGF0YVBvcytQQURfU0laRShsZW4pKSA+PSBtRGF0YVBvcyAmJiAobURhdGFQb3MrUEFEX1NJWkUobGVuKSkgPD0gbURhdGFTaXplKSB7Ci0gICAgICAgIGNvbnN0IHZvaWQqIGRhdGEgPSBtRGF0YSttRGF0YVBvczsKLSAgICAgICAgbURhdGFQb3MgKz0gUEFEX1NJWkUobGVuKTsKLSAgICAgICAgTE9HVigicmVhZElucGxhY2UgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgICAgICByZXR1cm4gZGF0YTsKLSAgICB9Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6cmVhZEludDMyKGludDMyX3QgKnBBcmcpIGNvbnN0Ci17Ci0gICAgaWYgKChtRGF0YVBvcytzaXplb2YoaW50MzJfdCkpIDw9IG1EYXRhU2l6ZSkgewotICAgICAgICBjb25zdCB2b2lkKiBkYXRhID0gbURhdGErbURhdGFQb3M7Ci0gICAgICAgIG1EYXRhUG9zICs9IHNpemVvZihpbnQzMl90KTsKLSAgICAgICAgKnBBcmcgPSAgKnJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgaW50MzJfdCo+KGRhdGEpOwotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgcmV0dXJuIE5PVF9FTk9VR0hfREFUQTsKLSAgICB9Ci19Ci0KLWludDMyX3QgUGFyY2VsOjpyZWFkSW50MzIoKSBjb25zdAotewotICAgIGlmICgobURhdGFQb3Mrc2l6ZW9mKGludDMyX3QpKSA8PSBtRGF0YVNpemUpIHsKLSAgICAgICAgY29uc3Qgdm9pZCogZGF0YSA9IG1EYXRhK21EYXRhUG9zOwotICAgICAgICBtRGF0YVBvcyArPSBzaXplb2YoaW50MzJfdCk7Ci0gICAgICAgIExPR1YoInJlYWRJbnQzMiBTZXR0aW5nIGRhdGEgcG9zIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVBvcyk7Ci0gICAgICAgIHJldHVybiAqcmVpbnRlcnByZXRfY2FzdDxjb25zdCBpbnQzMl90Kj4oZGF0YSk7Ci0gICAgfQotICAgIHJldHVybiAwOwotfQotCi0KLXN0YXR1c190IFBhcmNlbDo6cmVhZEludDY0KGludDY0X3QgKnBBcmcpIGNvbnN0Ci17Ci0gICAgaWYgKChtRGF0YVBvcytzaXplb2YoaW50NjRfdCkpIDw9IG1EYXRhU2l6ZSkgewotICAgICAgICBjb25zdCB2b2lkKiBkYXRhID0gbURhdGErbURhdGFQb3M7Ci0gICAgICAgIG1EYXRhUG9zICs9IHNpemVvZihpbnQ2NF90KTsKLSAgICAgICAgKnBBcmcgPSAqcmVpbnRlcnByZXRfY2FzdDxjb25zdCBpbnQ2NF90Kj4oZGF0YSk7Ci0gICAgICAgIExPR1YoInJlYWRJbnQ2NCBTZXR0aW5nIGRhdGEgcG9zIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVBvcyk7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9IGVsc2UgewotICAgICAgICByZXR1cm4gTk9UX0VOT1VHSF9EQVRBOwotICAgIH0KLX0KLQotCi1pbnQ2NF90IFBhcmNlbDo6cmVhZEludDY0KCkgY29uc3QKLXsKLSAgICBpZiAoKG1EYXRhUG9zK3NpemVvZihpbnQ2NF90KSkgPD0gbURhdGFTaXplKSB7Ci0gICAgICAgIGNvbnN0IHZvaWQqIGRhdGEgPSBtRGF0YSttRGF0YVBvczsKLSAgICAgICAgbURhdGFQb3MgKz0gc2l6ZW9mKGludDY0X3QpOwotICAgICAgICBMT0dWKCJyZWFkSW50NjQgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgICAgICByZXR1cm4gKnJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgaW50NjRfdCo+KGRhdGEpOwotICAgIH0KLSAgICByZXR1cm4gMDsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjpyZWFkRmxvYXQoZmxvYXQgKnBBcmcpIGNvbnN0Ci17Ci0gICAgaWYgKChtRGF0YVBvcytzaXplb2YoZmxvYXQpKSA8PSBtRGF0YVNpemUpIHsKLSAgICAgICAgY29uc3Qgdm9pZCogZGF0YSA9IG1EYXRhK21EYXRhUG9zOwotICAgICAgICBtRGF0YVBvcyArPSBzaXplb2YoZmxvYXQpOwotICAgICAgICBMT0dWKCJyZWFkRmxvYXQgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgICAgICAqcEFyZyA9ICpyZWludGVycHJldF9jYXN0PGNvbnN0IGZsb2F0Kj4oZGF0YSk7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9IGVsc2UgewotICAgICAgICByZXR1cm4gTk9UX0VOT1VHSF9EQVRBOwotICAgIH0KLX0KLQotCi1mbG9hdCBQYXJjZWw6OnJlYWRGbG9hdCgpIGNvbnN0Ci17Ci0gICAgaWYgKChtRGF0YVBvcytzaXplb2YoZmxvYXQpKSA8PSBtRGF0YVNpemUpIHsKLSAgICAgICAgY29uc3Qgdm9pZCogZGF0YSA9IG1EYXRhK21EYXRhUG9zOwotICAgICAgICBtRGF0YVBvcyArPSBzaXplb2YoZmxvYXQpOwotICAgICAgICBMT0dWKCJyZWFkRmxvYXQgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgICAgICByZXR1cm4gKnJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgZmxvYXQqPihkYXRhKTsKLSAgICB9Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6cmVhZERvdWJsZShkb3VibGUgKnBBcmcpIGNvbnN0Ci17Ci0gICAgaWYgKChtRGF0YVBvcytzaXplb2YoZG91YmxlKSkgPD0gbURhdGFTaXplKSB7Ci0gICAgICAgIGNvbnN0IHZvaWQqIGRhdGEgPSBtRGF0YSttRGF0YVBvczsKLSAgICAgICAgbURhdGFQb3MgKz0gc2l6ZW9mKGRvdWJsZSk7Ci0gICAgICAgIExPR1YoInJlYWREb3VibGUgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgICAgICAqcEFyZyA9ICpyZWludGVycHJldF9jYXN0PGNvbnN0IGRvdWJsZSo+KGRhdGEpOwotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgcmV0dXJuIE5PVF9FTk9VR0hfREFUQTsKLSAgICB9Ci19Ci0KLQotZG91YmxlIFBhcmNlbDo6cmVhZERvdWJsZSgpIGNvbnN0Ci17Ci0gICAgaWYgKChtRGF0YVBvcytzaXplb2YoZG91YmxlKSkgPD0gbURhdGFTaXplKSB7Ci0gICAgICAgIGNvbnN0IHZvaWQqIGRhdGEgPSBtRGF0YSttRGF0YVBvczsKLSAgICAgICAgbURhdGFQb3MgKz0gc2l6ZW9mKGRvdWJsZSk7Ci0gICAgICAgIExPR1YoInJlYWREb3VibGUgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgICAgICByZXR1cm4gKnJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgZG91YmxlKj4oZGF0YSk7Ci0gICAgfQotICAgIHJldHVybiAwOwotfQotCi0KLWNvbnN0IGNoYXIqIFBhcmNlbDo6cmVhZENTdHJpbmcoKSBjb25zdAotewotICAgIGNvbnN0IHNpemVfdCBhdmFpbCA9IG1EYXRhU2l6ZS1tRGF0YVBvczsKLSAgICBpZiAoYXZhaWwgPiAwKSB7Ci0gICAgICAgIGNvbnN0IGNoYXIqIHN0ciA9IHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgY2hhcio+KG1EYXRhK21EYXRhUG9zKTsKLSAgICAgICAgLy8gaXMgdGhlIHN0cmluZydzIHRyYWlsaW5nIE5VTCB3aXRoaW4gdGhlIHBhcmNlbCdzIHZhbGlkIGJvdW5kcz8KLSAgICAgICAgY29uc3QgY2hhciogZW9zID0gcmVpbnRlcnByZXRfY2FzdDxjb25zdCBjaGFyKj4obWVtY2hyKHN0ciwgMCwgYXZhaWwpKTsKLSAgICAgICAgaWYgKGVvcykgewotICAgICAgICAgICAgY29uc3Qgc2l6ZV90IGxlbiA9IGVvcyAtIHN0cjsKLSAgICAgICAgICAgIG1EYXRhUG9zICs9IFBBRF9TSVpFKGxlbisxKTsKLSAgICAgICAgICAgIExPR1YoInJlYWRDU3RyaW5nIFNldHRpbmcgZGF0YSBwb3Mgb2YgJXAgdG8gJWRcbiIsIHRoaXMsIG1EYXRhUG9zKTsKLSAgICAgICAgICAgIHJldHVybiBzdHI7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLVN0cmluZzggUGFyY2VsOjpyZWFkU3RyaW5nOCgpIGNvbnN0Ci17Ci0gICAgaW50MzJfdCBzaXplID0gcmVhZEludDMyKCk7Ci0gICAgLy8gd2F0Y2ggZm9yIHBvdGVudGlhbCBpbnQgb3ZlcmZsb3cgYWRkaW5nIDEgZm9yIHRyYWlsaW5nIE5VTAotICAgIGlmIChzaXplID4gMCAmJiBzaXplIDwgSU5UMzJfTUFYKSB7Ci0gICAgICAgIGNvbnN0IGNoYXIqIHN0ciA9IChjb25zdCBjaGFyKilyZWFkSW5wbGFjZShzaXplKzEpOwotICAgICAgICBpZiAoc3RyKSByZXR1cm4gU3RyaW5nOChzdHIsIHNpemUpOwotICAgIH0KLSAgICByZXR1cm4gU3RyaW5nOCgpOwotfQotCi1TdHJpbmcxNiBQYXJjZWw6OnJlYWRTdHJpbmcxNigpIGNvbnN0Ci17Ci0gICAgc2l6ZV90IGxlbjsKLSAgICBjb25zdCBjaGFyMTZfdCogc3RyID0gcmVhZFN0cmluZzE2SW5wbGFjZSgmbGVuKTsKLSAgICBpZiAoc3RyKSByZXR1cm4gU3RyaW5nMTYoc3RyLCBsZW4pOwotICAgIExPR0UoIlJlYWRpbmcgYSBOVUxMIHN0cmluZyBub3Qgc3VwcG9ydGVkIGhlcmUuIik7Ci0gICAgcmV0dXJuIFN0cmluZzE2KCk7Ci19Ci0KLWNvbnN0IGNoYXIxNl90KiBQYXJjZWw6OnJlYWRTdHJpbmcxNklucGxhY2Uoc2l6ZV90KiBvdXRMZW4pIGNvbnN0Ci17Ci0gICAgaW50MzJfdCBzaXplID0gcmVhZEludDMyKCk7Ci0gICAgLy8gd2F0Y2ggZm9yIHBvdGVudGlhbCBpbnQgb3ZlcmZsb3cgZnJvbSBzaXplKzEKLSAgICBpZiAoc2l6ZSA+PSAwICYmIHNpemUgPCBJTlQzMl9NQVgpIHsKLSAgICAgICAgKm91dExlbiA9IHNpemU7Ci0gICAgICAgIGNvbnN0IGNoYXIxNl90KiBzdHIgPSAoY29uc3QgY2hhcjE2X3QqKXJlYWRJbnBsYWNlKChzaXplKzEpKnNpemVvZihjaGFyMTZfdCkpOwotICAgICAgICBpZiAoc3RyICE9IE5VTEwpIHsKLSAgICAgICAgICAgIHJldHVybiBzdHI7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgKm91dExlbiA9IDA7Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLXNwPElCaW5kZXI+IFBhcmNlbDo6cmVhZFN0cm9uZ0JpbmRlcigpIGNvbnN0Ci17Ci0gICAgc3A8SUJpbmRlcj4gdmFsOwotICAgIHVuZmxhdHRlbl9iaW5kZXIoUHJvY2Vzc1N0YXRlOjpzZWxmKCksICp0aGlzLCAmdmFsKTsKLSAgICByZXR1cm4gdmFsOwotfQotCi13cDxJQmluZGVyPiBQYXJjZWw6OnJlYWRXZWFrQmluZGVyKCkgY29uc3QKLXsKLSAgICB3cDxJQmluZGVyPiB2YWw7Ci0gICAgdW5mbGF0dGVuX2JpbmRlcihQcm9jZXNzU3RhdGU6OnNlbGYoKSwgKnRoaXMsICZ2YWwpOwotICAgIHJldHVybiB2YWw7Ci19Ci0KLQotbmF0aXZlX2hhbmRsZSogUGFyY2VsOjpyZWFkTmF0aXZlSGFuZGxlKG5hdGl2ZV9oYW5kbGUqICgqYWxsb2MpKHZvaWQqLCBpbnQsIGludCksIHZvaWQqIGNvb2tpZSkgY29uc3QKLXsKLSAgICBpbnQgbnVtRmRzLCBudW1JbnRzOwotICAgIHN0YXR1c190IGVycjsKLSAgICBlcnIgPSByZWFkSW50MzIoJm51bUZkcyk7Ci0gICAgaWYgKGVyciAhPSBOT19FUlJPUikgcmV0dXJuIDA7Ci0gICAgZXJyID0gcmVhZEludDMyKCZudW1JbnRzKTsKLSAgICBpZiAoZXJyICE9IE5PX0VSUk9SKSByZXR1cm4gMDsKLQotICAgIG5hdGl2ZV9oYW5kbGUqIGg7Ci0gICAgaWYgKGFsbG9jID09IDApIHsKLSAgICAgICAgc2l6ZV90IHNpemUgPSBzaXplb2YobmF0aXZlX2hhbmRsZSkgKyBzaXplb2YoaW50KSoobnVtRmRzICsgbnVtSW50cyk7Ci0gICAgICAgIGggPSAobmF0aXZlX2hhbmRsZSopbWFsbG9jKHNpemUpOyAKLSAgICAgICAgaC0+dmVyc2lvbiA9IHNpemVvZihuYXRpdmVfaGFuZGxlKTsKLSAgICAgICAgaC0+bnVtRmRzID0gbnVtRmRzOwotICAgICAgICBoLT5udW1JbnRzID0gbnVtSW50czsKLSAgICB9IGVsc2UgewotICAgICAgICBoID0gYWxsb2MoY29va2llLCBudW1GZHMsIG51bUludHMpOwotICAgICAgICBpZiAoaC0+dmVyc2lvbiAhPSBzaXplb2YobmF0aXZlX2hhbmRsZSkpIHsKLSAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIGZvciAoaW50IGk9MCA7IGVycj09Tk9fRVJST1IgJiYgaTxudW1GZHMgOyBpKyspIHsKLSAgICAgICAgaC0+ZGF0YVtpXSA9IGR1cChyZWFkRmlsZURlc2NyaXB0b3IoKSk7Ci0gICAgICAgIGlmIChoLT5kYXRhW2ldIDwgMCkgZXJyID0gQkFEX1ZBTFVFOwotICAgIH0KLSAgICAKLSAgICBlcnIgPSByZWFkKGgtPmRhdGEgKyBudW1GZHMsIHNpemVvZihpbnQpKm51bUludHMpOwotICAgIAotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgaWYgKGFsbG9jID09IDApIHsKLSAgICAgICAgICAgIGZyZWUoaCk7Ci0gICAgICAgIH0KLSAgICAgICAgaCA9IDA7Ci0gICAgfQotICAgIHJldHVybiBoOwotfQotCi0KLWludCBQYXJjZWw6OnJlYWRGaWxlRGVzY3JpcHRvcigpIGNvbnN0Ci17Ci0gICAgY29uc3QgZmxhdF9iaW5kZXJfb2JqZWN0KiBmbGF0ID0gcmVhZE9iamVjdCh0cnVlKTsKLSAgICBpZiAoZmxhdCkgewotICAgICAgICBzd2l0Y2ggKGZsYXQtPnR5cGUpIHsKLSAgICAgICAgICAgIGNhc2UgQklOREVSX1RZUEVfRkQ6Ci0gICAgICAgICAgICAgICAgLy9MT0dJKCJSZXR1cm5pbmcgZmlsZSBkZXNjcmlwdG9yICVsZCBmcm9tIHBhcmNlbCAlcFxuIiwgZmxhdC0+aGFuZGxlLCB0aGlzKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gZmxhdC0+aGFuZGxlOwotICAgICAgICB9ICAgICAgICAKLSAgICB9Ci0gICAgcmV0dXJuIEJBRF9UWVBFOwotfQotCi1jb25zdCBmbGF0X2JpbmRlcl9vYmplY3QqIFBhcmNlbDo6cmVhZE9iamVjdChib29sIG51bGxNZXRhRGF0YSkgY29uc3QKLXsKLSAgICBjb25zdCBzaXplX3QgRFBPUyA9IG1EYXRhUG9zOwotICAgIGlmICgoRFBPUytzaXplb2YoZmxhdF9iaW5kZXJfb2JqZWN0KSkgPD0gbURhdGFTaXplKSB7Ci0gICAgICAgIGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCogb2JqCi0gICAgICAgICAgICAgICAgPSByZWludGVycHJldF9jYXN0PGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCo+KG1EYXRhK0RQT1MpOwotICAgICAgICBtRGF0YVBvcyA9IERQT1MgKyBzaXplb2YoZmxhdF9iaW5kZXJfb2JqZWN0KTsKLSAgICAgICAgaWYgKCFudWxsTWV0YURhdGEgJiYgKG9iai0+Y29va2llID09IE5VTEwgJiYgb2JqLT5iaW5kZXIgPT0gTlVMTCkpIHsKLSAgICAgICAgICAgIC8vIFdoZW4gdHJhbnNmZXJyaW5nIGEgTlVMTCBvYmplY3QsIHdlIGRvbid0IHdyaXRlIGl0IGludG8KLSAgICAgICAgICAgIC8vIHRoZSBvYmplY3QgbGlzdCwgc28gd2UgZG9uJ3Qgd2FudCB0byBjaGVjayBmb3IgaXQgd2hlbgotICAgICAgICAgICAgLy8gcmVhZGluZy4KLSAgICAgICAgICAgIExPR1YoInJlYWRPYmplY3QgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgICAgICAgICAgcmV0dXJuIG9iajsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgLy8gRW5zdXJlIHRoYXQgdGhpcyBvYmplY3QgaXMgdmFsaWQuLi4KLSAgICAgICAgc2l6ZV90KiBjb25zdCBPQkpTID0gbU9iamVjdHM7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBOID0gbU9iamVjdHNTaXplOwotICAgICAgICBzaXplX3Qgb3BvcyA9IG1OZXh0T2JqZWN0SGludDsKLSAgICAgICAgCi0gICAgICAgIGlmIChOID4gMCkgewotICAgICAgICAgICAgTE9HVigiUGFyY2VsICVwIGxvb2tpbmcgZm9yIG9iaiBhdCAlZCwgaGludD0lZFxuIiwKLSAgICAgICAgICAgICAgICAgdGhpcywgRFBPUywgb3Bvcyk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8vIFN0YXJ0IGF0IHRoZSBjdXJyZW50IGhpbnQgcG9zaXRpb24sIGxvb2tpbmcgZm9yIGFuIG9iamVjdCBhdAotICAgICAgICAgICAgLy8gdGhlIGN1cnJlbnQgZGF0YSBwb3NpdGlvbi4KLSAgICAgICAgICAgIGlmIChvcG9zIDwgTikgewotICAgICAgICAgICAgICAgIHdoaWxlIChvcG9zIDwgKE4tMSkgJiYgT0JKU1tvcG9zXSA8IERQT1MpIHsKLSAgICAgICAgICAgICAgICAgICAgb3BvcysrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgb3BvcyA9IE4tMTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChPQkpTW29wb3NdID09IERQT1MpIHsKLSAgICAgICAgICAgICAgICAvLyBGb3VuZCBpdCEKLSAgICAgICAgICAgICAgICBMT0dWKCJQYXJjZWwgZm91bmQgb2JqICVkIGF0IGluZGV4ICVkIHdpdGggZm9yd2FyZCBzZWFyY2giLAotICAgICAgICAgICAgICAgICAgICAgdGhpcywgRFBPUywgb3Bvcyk7Ci0gICAgICAgICAgICAgICAgbU5leHRPYmplY3RIaW50ID0gb3BvcysxOwotICAgICAgICAgICAgICAgIExPR1YoInJlYWRPYmplY3QgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgICAgICAgICAgICAgIHJldHVybiBvYmo7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICAgICAgLy8gTG9vayBiYWNrd2FyZHMgZm9yIGl0Li4uCi0gICAgICAgICAgICB3aGlsZSAob3BvcyA+IDAgJiYgT0JKU1tvcG9zXSA+IERQT1MpIHsKLSAgICAgICAgICAgICAgICBvcG9zLS07Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoT0JKU1tvcG9zXSA9PSBEUE9TKSB7Ci0gICAgICAgICAgICAgICAgLy8gRm91bmQgaXQhCi0gICAgICAgICAgICAgICAgTE9HVigiUGFyY2VsIGZvdW5kIG9iaiAlZCBhdCBpbmRleCAlZCB3aXRoIGJhY2t3YXJkIHNlYXJjaCIsCi0gICAgICAgICAgICAgICAgICAgICB0aGlzLCBEUE9TLCBvcG9zKTsKLSAgICAgICAgICAgICAgICBtTmV4dE9iamVjdEhpbnQgPSBvcG9zKzE7Ci0gICAgICAgICAgICAgICAgTE9HVigicmVhZE9iamVjdCBTZXR0aW5nIGRhdGEgcG9zIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVBvcyk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG9iajsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBMT0dXKCJBdHRlbXB0IHRvIHJlYWQgb2JqZWN0IGZyb20gUGFyY2VsICVwIGF0IG9mZnNldCAlZCB0aGF0IGlzIG5vdCBpbiB0aGUgb2JqZWN0IGxpc3QiLAotICAgICAgICAgICAgIHRoaXMsIERQT1MpOwotICAgIH0KLSAgICByZXR1cm4gTlVMTDsKLX0KLQotdm9pZCBQYXJjZWw6OmNsb3NlRmlsZURlc2NyaXB0b3JzKCkKLXsKLSAgICBzaXplX3QgaSA9IG1PYmplY3RzU2l6ZTsKLSAgICBpZiAoaSA+IDApIHsKLSAgICAgICAgLy9MT0dJKCJDbG9zaW5nIGZpbGUgZGVzY3JpcHRvcnMgZm9yICVkIG9iamVjdHMuLi4iLCBtT2JqZWN0c1NpemUpOwotICAgIH0KLSAgICB3aGlsZSAoaSA+IDApIHsKLSAgICAgICAgaS0tOwotICAgICAgICBjb25zdCBmbGF0X2JpbmRlcl9vYmplY3QqIGZsYXQKLSAgICAgICAgICAgID0gcmVpbnRlcnByZXRfY2FzdDxmbGF0X2JpbmRlcl9vYmplY3QqPihtRGF0YSttT2JqZWN0c1tpXSk7Ci0gICAgICAgIGlmIChmbGF0LT50eXBlID09IEJJTkRFUl9UWVBFX0ZEKSB7Ci0gICAgICAgICAgICAvL0xPR0koIkNsb3NpbmcgZmQ6ICVsZFxuIiwgZmxhdC0+aGFuZGxlKTsKLSAgICAgICAgICAgIGNsb3NlKGZsYXQtPmhhbmRsZSk7Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLWNvbnN0IHVpbnQ4X3QqIFBhcmNlbDo6aXBjRGF0YSgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1EYXRhOwotfQotCi1zaXplX3QgUGFyY2VsOjppcGNEYXRhU2l6ZSgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIChtRGF0YVNpemUgPiBtRGF0YVBvcyA/IG1EYXRhU2l6ZSA6IG1EYXRhUG9zKTsKLX0KLQotY29uc3Qgc2l6ZV90KiBQYXJjZWw6OmlwY09iamVjdHMoKSBjb25zdAotewotICAgIHJldHVybiBtT2JqZWN0czsKLX0KLQotc2l6ZV90IFBhcmNlbDo6aXBjT2JqZWN0c0NvdW50KCkgY29uc3QKLXsKLSAgICByZXR1cm4gbU9iamVjdHNTaXplOwotfQotCi12b2lkIFBhcmNlbDo6aXBjU2V0RGF0YVJlZmVyZW5jZShjb25zdCB1aW50OF90KiBkYXRhLCBzaXplX3QgZGF0YVNpemUsCi0gICAgY29uc3Qgc2l6ZV90KiBvYmplY3RzLCBzaXplX3Qgb2JqZWN0c0NvdW50LCByZWxlYXNlX2Z1bmMgcmVsRnVuYywgdm9pZCogcmVsQ29va2llKQotewotICAgIGZyZWVEYXRhTm9Jbml0KCk7Ci0gICAgbUVycm9yID0gTk9fRVJST1I7Ci0gICAgbURhdGEgPSBjb25zdF9jYXN0PHVpbnQ4X3QqPihkYXRhKTsKLSAgICBtRGF0YVNpemUgPSBtRGF0YUNhcGFjaXR5ID0gZGF0YVNpemU7Ci0gICAgLy9MT0dJKCJzZXREYXRhUmVmZXJlbmNlIFNldHRpbmcgZGF0YSBzaXplIG9mICVwIHRvICVsdSAocGlkPSVkKVxuIiwgdGhpcywgbURhdGFTaXplLCBnZXRwaWQoKSk7Ci0gICAgbURhdGFQb3MgPSAwOwotICAgIExPR1YoInNldERhdGFSZWZlcmVuY2UgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgIG1PYmplY3RzID0gY29uc3RfY2FzdDxzaXplX3QqPihvYmplY3RzKTsKLSAgICBtT2JqZWN0c1NpemUgPSBtT2JqZWN0c0NhcGFjaXR5ID0gb2JqZWN0c0NvdW50OwotICAgIG1OZXh0T2JqZWN0SGludCA9IDA7Ci0gICAgbU93bmVyID0gcmVsRnVuYzsKLSAgICBtT3duZXJDb29raWUgPSByZWxDb29raWU7Ci0gICAgc2NhbkZvckZkcygpOwotfQotCi12b2lkIFBhcmNlbDo6cHJpbnQoVGV4dE91dHB1dCYgdG8sIHVpbnQzMl90IGZsYWdzKSBjb25zdAotewotICAgIHRvIDw8ICJQYXJjZWwoIjsKLSAgICAKLSAgICBpZiAoZXJyb3JDaGVjaygpICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgIGNvbnN0IHN0YXR1c190IGVyciA9IGVycm9yQ2hlY2soKTsKLSAgICAgICAgdG8gPDwgIkVycm9yOiAiIDw8ICh2b2lkKillcnIgPDwgIiBcIiIgPDwgc3RyZXJyb3IoLWVycikgPDwgIlwiIjsKLSAgICB9IGVsc2UgaWYgKGRhdGFTaXplKCkgPiAwKSB7Ci0gICAgICAgIGNvbnN0IHVpbnQ4X3QqIERBVEEgPSBkYXRhKCk7Ci0gICAgICAgIHRvIDw8IGluZGVudCA8PCBIZXhEdW1wKERBVEEsIGRhdGFTaXplKCkpIDw8IGRlZGVudDsKLSAgICAgICAgY29uc3Qgc2l6ZV90KiBPQkpTID0gb2JqZWN0cygpOwotICAgICAgICBjb25zdCBzaXplX3QgTiA9IG9iamVjdHNDb3VudCgpOwotICAgICAgICBmb3IgKHNpemVfdCBpPTA7IGk8TjsgaSsrKSB7Ci0gICAgICAgICAgICBjb25zdCBmbGF0X2JpbmRlcl9vYmplY3QqIGZsYXQKLSAgICAgICAgICAgICAgICA9IHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgZmxhdF9iaW5kZXJfb2JqZWN0Kj4oREFUQStPQkpTW2ldKTsKLSAgICAgICAgICAgIHRvIDw8IGVuZGwgPDwgIk9iamVjdCAjIiA8PCBpIDw8ICIgQCAiIDw8ICh2b2lkKilPQkpTW2ldIDw8ICI6ICIKLSAgICAgICAgICAgICAgICA8PCBUeXBlQ29kZShmbGF0LT50eXBlICYgMHg3ZjdmN2YwMCkKLSAgICAgICAgICAgICAgICA8PCAiID0gIiA8PCBmbGF0LT5iaW5kZXI7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICB0byA8PCAiTlVMTCI7Ci0gICAgfQotICAgIAotICAgIHRvIDw8ICIpIjsKLX0KLQotdm9pZCBQYXJjZWw6OnJlbGVhc2VPYmplY3RzKCkKLXsKLSAgICBjb25zdCBzcDxQcm9jZXNzU3RhdGU+IHByb2MoUHJvY2Vzc1N0YXRlOjpzZWxmKCkpOwotICAgIHNpemVfdCBpID0gbU9iamVjdHNTaXplOwotICAgIHVpbnQ4X3QqIGNvbnN0IGRhdGEgPSBtRGF0YTsKLSAgICBzaXplX3QqIGNvbnN0IG9iamVjdHMgPSBtT2JqZWN0czsKLSAgICB3aGlsZSAoaSA+IDApIHsKLSAgICAgICAgaS0tOwotICAgICAgICBjb25zdCBmbGF0X2JpbmRlcl9vYmplY3QqIGZsYXQKLSAgICAgICAgICAgID0gcmVpbnRlcnByZXRfY2FzdDxmbGF0X2JpbmRlcl9vYmplY3QqPihkYXRhK29iamVjdHNbaV0pOwotICAgICAgICByZWxlYXNlX29iamVjdChwcm9jLCAqZmxhdCwgdGhpcyk7Ci0gICAgfQotfQotCi12b2lkIFBhcmNlbDo6YWNxdWlyZU9iamVjdHMoKQotewotICAgIGNvbnN0IHNwPFByb2Nlc3NTdGF0ZT4gcHJvYyhQcm9jZXNzU3RhdGU6OnNlbGYoKSk7Ci0gICAgc2l6ZV90IGkgPSBtT2JqZWN0c1NpemU7Ci0gICAgdWludDhfdCogY29uc3QgZGF0YSA9IG1EYXRhOwotICAgIHNpemVfdCogY29uc3Qgb2JqZWN0cyA9IG1PYmplY3RzOwotICAgIHdoaWxlIChpID4gMCkgewotICAgICAgICBpLS07Ci0gICAgICAgIGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCogZmxhdAotICAgICAgICAgICAgPSByZWludGVycHJldF9jYXN0PGZsYXRfYmluZGVyX29iamVjdCo+KGRhdGErb2JqZWN0c1tpXSk7Ci0gICAgICAgIGFjcXVpcmVfb2JqZWN0KHByb2MsICpmbGF0LCB0aGlzKTsKLSAgICB9Ci19Ci0KLXZvaWQgUGFyY2VsOjpmcmVlRGF0YSgpCi17Ci0gICAgZnJlZURhdGFOb0luaXQoKTsKLSAgICBpbml0U3RhdGUoKTsKLX0KLQotdm9pZCBQYXJjZWw6OmZyZWVEYXRhTm9Jbml0KCkKLXsKLSAgICBpZiAobU93bmVyKSB7Ci0gICAgICAgIC8vTE9HSSgiRnJlZWluZyBkYXRhIHJlZiBvZiAlcCAocGlkPSVkKVxuIiwgdGhpcywgZ2V0cGlkKCkpOwotICAgICAgICBtT3duZXIodGhpcywgbURhdGEsIG1EYXRhU2l6ZSwgbU9iamVjdHMsIG1PYmplY3RzU2l6ZSwgbU93bmVyQ29va2llKTsKLSAgICB9IGVsc2UgewotICAgICAgICByZWxlYXNlT2JqZWN0cygpOwotICAgICAgICBpZiAobURhdGEpIGZyZWUobURhdGEpOwotICAgICAgICBpZiAobU9iamVjdHMpIGZyZWUobU9iamVjdHMpOwotICAgIH0KLX0KLQotc3RhdHVzX3QgUGFyY2VsOjpncm93RGF0YShzaXplX3QgbGVuKQotewotICAgIHNpemVfdCBuZXdTaXplID0gKChtRGF0YVNpemUrbGVuKSozKS8yOwotICAgIHJldHVybiAobmV3U2l6ZSA8PSBtRGF0YVNpemUpCi0gICAgICAgICAgICA/IChzdGF0dXNfdCkgTk9fTUVNT1JZCi0gICAgICAgICAgICA6IGNvbnRpbnVlV3JpdGUobmV3U2l6ZSk7Ci19Ci0KLXN0YXR1c190IFBhcmNlbDo6cmVzdGFydFdyaXRlKHNpemVfdCBkZXNpcmVkKQotewotICAgIGlmIChtT3duZXIpIHsKLSAgICAgICAgZnJlZURhdGEoKTsKLSAgICAgICAgcmV0dXJuIGNvbnRpbnVlV3JpdGUoZGVzaXJlZCk7Ci0gICAgfQotICAgIAotICAgIHVpbnQ4X3QqIGRhdGEgPSAodWludDhfdCopcmVhbGxvYyhtRGF0YSwgZGVzaXJlZCk7Ci0gICAgaWYgKCFkYXRhICYmIGRlc2lyZWQgPiBtRGF0YUNhcGFjaXR5KSB7Ci0gICAgICAgIG1FcnJvciA9IE5PX01FTU9SWTsKLSAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLSAgICB9Ci0gICAgCi0gICAgcmVsZWFzZU9iamVjdHMoKTsKLSAgICAKLSAgICBpZiAoZGF0YSkgewotICAgICAgICBtRGF0YSA9IGRhdGE7Ci0gICAgICAgIG1EYXRhQ2FwYWNpdHkgPSBkZXNpcmVkOwotICAgIH0KLSAgICAKLSAgICBtRGF0YVNpemUgPSBtRGF0YVBvcyA9IDA7Ci0gICAgTE9HVigicmVzdGFydFdyaXRlIFNldHRpbmcgZGF0YSBzaXplIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVNpemUpOwotICAgIExPR1YoInJlc3RhcnRXcml0ZSBTZXR0aW5nIGRhdGEgcG9zIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVBvcyk7Ci0gICAgICAgIAotICAgIGZyZWUobU9iamVjdHMpOwotICAgIG1PYmplY3RzID0gTlVMTDsKLSAgICBtT2JqZWN0c1NpemUgPSBtT2JqZWN0c0NhcGFjaXR5ID0gMDsKLSAgICBtTmV4dE9iamVjdEhpbnQgPSAwOwotICAgIG1IYXNGZHMgPSBmYWxzZTsKLSAgICBtRmRzS25vd24gPSB0cnVlOwotICAgIAotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgUGFyY2VsOjpjb250aW51ZVdyaXRlKHNpemVfdCBkZXNpcmVkKQotewotICAgIC8vIElmIHNocmlua2luZywgZmlyc3QgYWRqdXN0IGZvciBhbnkgb2JqZWN0cyB0aGF0IGFwcGVhcgotICAgIC8vIGFmdGVyIHRoZSBuZXcgZGF0YSBzaXplLgotICAgIHNpemVfdCBvYmplY3RzU2l6ZSA9IG1PYmplY3RzU2l6ZTsKLSAgICBpZiAoZGVzaXJlZCA8IG1EYXRhU2l6ZSkgewotICAgICAgICBpZiAoZGVzaXJlZCA9PSAwKSB7Ci0gICAgICAgICAgICBvYmplY3RzU2l6ZSA9IDA7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICB3aGlsZSAob2JqZWN0c1NpemUgPiAwKSB7Ci0gICAgICAgICAgICAgICAgaWYgKG1PYmplY3RzW29iamVjdHNTaXplLTFdIDwgZGVzaXJlZCkKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgb2JqZWN0c1NpemUtLTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBpZiAobU93bmVyKSB7Ci0gICAgICAgIC8vIElmIHRoZSBzaXplIGlzIGdvaW5nIHRvIHplcm8sIGp1c3QgcmVsZWFzZSB0aGUgb3duZXIncyBkYXRhLgotICAgICAgICBpZiAoZGVzaXJlZCA9PSAwKSB7Ci0gICAgICAgICAgICBmcmVlRGF0YSgpOwotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gSWYgdGhlcmUgaXMgYSBkaWZmZXJlbnQgb3duZXIsIHdlIG5lZWQgdG8gdGFrZQotICAgICAgICAvLyBwb3Nlc3Npb24uCi0gICAgICAgIHVpbnQ4X3QqIGRhdGEgPSAodWludDhfdCopbWFsbG9jKGRlc2lyZWQpOwotICAgICAgICBpZiAoIWRhdGEpIHsKLSAgICAgICAgICAgIG1FcnJvciA9IE5PX01FTU9SWTsKLSAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgIH0KLSAgICAgICAgc2l6ZV90KiBvYmplY3RzID0gTlVMTDsKLSAgICAgICAgCi0gICAgICAgIGlmIChvYmplY3RzU2l6ZSkgewotICAgICAgICAgICAgb2JqZWN0cyA9IChzaXplX3QqKW1hbGxvYyhvYmplY3RzU2l6ZSpzaXplb2Yoc2l6ZV90KSk7Ci0gICAgICAgICAgICBpZiAoIW9iamVjdHMpIHsKLSAgICAgICAgICAgICAgICBtRXJyb3IgPSBOT19NRU1PUlk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gTGl0dGxlIGhhY2sgdG8gb25seSBhY3F1aXJlIHJlZmVyZW5jZXMgb24gb2JqZWN0cwotICAgICAgICAgICAgLy8gd2Ugd2lsbCBiZSBrZWVwaW5nLgotICAgICAgICAgICAgc2l6ZV90IG9sZE9iamVjdHNTaXplID0gbU9iamVjdHNTaXplOwotICAgICAgICAgICAgbU9iamVjdHNTaXplID0gb2JqZWN0c1NpemU7Ci0gICAgICAgICAgICBhY3F1aXJlT2JqZWN0cygpOwotICAgICAgICAgICAgbU9iamVjdHNTaXplID0gb2xkT2JqZWN0c1NpemU7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIGlmIChtRGF0YSkgewotICAgICAgICAgICAgbWVtY3B5KGRhdGEsIG1EYXRhLCBtRGF0YVNpemUgPCBkZXNpcmVkID8gbURhdGFTaXplIDogZGVzaXJlZCk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG9iamVjdHMgJiYgbU9iamVjdHMpIHsKLSAgICAgICAgICAgIG1lbWNweShvYmplY3RzLCBtT2JqZWN0cywgb2JqZWN0c1NpemUqc2l6ZW9mKHNpemVfdCkpOwotICAgICAgICB9Ci0gICAgICAgIC8vTE9HSSgiRnJlZWluZyBkYXRhIHJlZiBvZiAlcCAocGlkPSVkKVxuIiwgdGhpcywgZ2V0cGlkKCkpOwotICAgICAgICBtT3duZXIodGhpcywgbURhdGEsIG1EYXRhU2l6ZSwgbU9iamVjdHMsIG1PYmplY3RzU2l6ZSwgbU93bmVyQ29va2llKTsKLSAgICAgICAgbU93bmVyID0gTlVMTDsKLQotICAgICAgICBtRGF0YSA9IGRhdGE7Ci0gICAgICAgIG1PYmplY3RzID0gb2JqZWN0czsKLSAgICAgICAgbURhdGFTaXplID0gKG1EYXRhU2l6ZSA8IGRlc2lyZWQpID8gbURhdGFTaXplIDogZGVzaXJlZDsKLSAgICAgICAgTE9HVigiY29udGludWVXcml0ZSBTZXR0aW5nIGRhdGEgc2l6ZSBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFTaXplKTsKLSAgICAgICAgbURhdGFDYXBhY2l0eSA9IGRlc2lyZWQ7Ci0gICAgICAgIG1PYmplY3RzU2l6ZSA9IG1PYmplY3RzQ2FwYWNpdHkgPSBvYmplY3RzU2l6ZTsKLSAgICAgICAgbU5leHRPYmplY3RIaW50ID0gMDsKLQotICAgIH0gZWxzZSBpZiAobURhdGEpIHsKLSAgICAgICAgaWYgKG9iamVjdHNTaXplIDwgbU9iamVjdHNTaXplKSB7Ci0gICAgICAgICAgICAvLyBOZWVkIHRvIHJlbGVhc2UgcmVmcyBvbiBhbnkgb2JqZWN0cyB3ZSBhcmUgZHJvcHBpbmcuCi0gICAgICAgICAgICBjb25zdCBzcDxQcm9jZXNzU3RhdGU+IHByb2MoUHJvY2Vzc1N0YXRlOjpzZWxmKCkpOwotICAgICAgICAgICAgZm9yIChzaXplX3QgaT1vYmplY3RzU2l6ZTsgaTxtT2JqZWN0c1NpemU7IGkrKykgewotICAgICAgICAgICAgICAgIGNvbnN0IGZsYXRfYmluZGVyX29iamVjdCogZmxhdAotICAgICAgICAgICAgICAgICAgICA9IHJlaW50ZXJwcmV0X2Nhc3Q8ZmxhdF9iaW5kZXJfb2JqZWN0Kj4obURhdGErbU9iamVjdHNbaV0pOwotICAgICAgICAgICAgICAgIGlmIChmbGF0LT50eXBlID09IEJJTkRFUl9UWVBFX0ZEKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIHdpbGwgbmVlZCB0byByZXNjYW4gYmVjYXVzZSB3ZSBtYXkgaGF2ZSBsb3BwZWQgb2ZmIHRoZSBvbmx5IEZEcwotICAgICAgICAgICAgICAgICAgICBtRmRzS25vd24gPSBmYWxzZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmVsZWFzZV9vYmplY3QocHJvYywgKmZsYXQsIHRoaXMpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgc2l6ZV90KiBvYmplY3RzID0KLSAgICAgICAgICAgICAgICAoc2l6ZV90KilyZWFsbG9jKG1PYmplY3RzLCBvYmplY3RzU2l6ZSpzaXplb2Yoc2l6ZV90KSk7Ci0gICAgICAgICAgICBpZiAob2JqZWN0cykgewotICAgICAgICAgICAgICAgIG1PYmplY3RzID0gb2JqZWN0czsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1PYmplY3RzU2l6ZSA9IG9iamVjdHNTaXplOwotICAgICAgICAgICAgbU5leHRPYmplY3RIaW50ID0gMDsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIFdlIG93biB0aGUgZGF0YSwgc28gd2UgY2FuIGp1c3QgZG8gYSByZWFsbG9jKCkuCi0gICAgICAgIGlmIChkZXNpcmVkID4gbURhdGFDYXBhY2l0eSkgewotICAgICAgICAgICAgdWludDhfdCogZGF0YSA9ICh1aW50OF90KilyZWFsbG9jKG1EYXRhLCBkZXNpcmVkKTsKLSAgICAgICAgICAgIGlmIChkYXRhKSB7Ci0gICAgICAgICAgICAgICAgbURhdGEgPSBkYXRhOwotICAgICAgICAgICAgICAgIG1EYXRhQ2FwYWNpdHkgPSBkZXNpcmVkOwotICAgICAgICAgICAgfSBlbHNlIGlmIChkZXNpcmVkID4gbURhdGFDYXBhY2l0eSkgewotICAgICAgICAgICAgICAgIG1FcnJvciA9IE5PX01FTU9SWTsKLSAgICAgICAgICAgICAgICByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbURhdGFTaXplID0gZGVzaXJlZDsKLSAgICAgICAgICAgIExPR1YoImNvbnRpbnVlV3JpdGUgU2V0dGluZyBkYXRhIHNpemUgb2YgJXAgdG8gJWRcbiIsIHRoaXMsIG1EYXRhU2l6ZSk7Ci0gICAgICAgICAgICBpZiAobURhdGFQb3MgPiBkZXNpcmVkKSB7Ci0gICAgICAgICAgICAgICAgbURhdGFQb3MgPSBkZXNpcmVkOwotICAgICAgICAgICAgICAgIExPR1YoImNvbnRpbnVlV3JpdGUgU2V0dGluZyBkYXRhIHBvcyBvZiAlcCB0byAlZFxuIiwgdGhpcywgbURhdGFQb3MpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIAotICAgIH0gZWxzZSB7Ci0gICAgICAgIC8vIFRoaXMgaXMgdGhlIGZpcnN0IGRhdGEuICBFYXN5IQotICAgICAgICB1aW50OF90KiBkYXRhID0gKHVpbnQ4X3QqKW1hbGxvYyhkZXNpcmVkKTsKLSAgICAgICAgaWYgKCFkYXRhKSB7Ci0gICAgICAgICAgICBtRXJyb3IgPSBOT19NRU1PUlk7Ci0gICAgICAgICAgICByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBpZighKG1EYXRhQ2FwYWNpdHkgPT0gMCAmJiBtT2JqZWN0cyA9PSBOVUxMCi0gICAgICAgICAgICAgJiYgbU9iamVjdHNDYXBhY2l0eSA9PSAwKSkgewotICAgICAgICAgICAgTE9HRSgiY29udGludWVXcml0ZTogJWQvJXAvJWQvJWQiLCBtRGF0YUNhcGFjaXR5LCBtT2JqZWN0cywgbU9iamVjdHNDYXBhY2l0eSwgZGVzaXJlZCk7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIG1EYXRhID0gZGF0YTsKLSAgICAgICAgbURhdGFTaXplID0gbURhdGFQb3MgPSAwOwotICAgICAgICBMT0dWKCJjb250aW51ZVdyaXRlIFNldHRpbmcgZGF0YSBzaXplIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVNpemUpOwotICAgICAgICBMT0dWKCJjb250aW51ZVdyaXRlIFNldHRpbmcgZGF0YSBwb3Mgb2YgJXAgdG8gJWRcbiIsIHRoaXMsIG1EYXRhUG9zKTsKLSAgICAgICAgbURhdGFDYXBhY2l0eSA9IGRlc2lyZWQ7Ci0gICAgfQotCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi12b2lkIFBhcmNlbDo6aW5pdFN0YXRlKCkKLXsKLSAgICBtRXJyb3IgPSBOT19FUlJPUjsKLSAgICBtRGF0YSA9IDA7Ci0gICAgbURhdGFTaXplID0gMDsKLSAgICBtRGF0YUNhcGFjaXR5ID0gMDsKLSAgICBtRGF0YVBvcyA9IDA7Ci0gICAgTE9HVigiaW5pdFN0YXRlIFNldHRpbmcgZGF0YSBzaXplIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVNpemUpOwotICAgIExPR1YoImluaXRTdGF0ZSBTZXR0aW5nIGRhdGEgcG9zIG9mICVwIHRvICVkXG4iLCB0aGlzLCBtRGF0YVBvcyk7Ci0gICAgbU9iamVjdHMgPSBOVUxMOwotICAgIG1PYmplY3RzU2l6ZSA9IDA7Ci0gICAgbU9iamVjdHNDYXBhY2l0eSA9IDA7Ci0gICAgbU5leHRPYmplY3RIaW50ID0gMDsKLSAgICBtSGFzRmRzID0gZmFsc2U7Ci0gICAgbUZkc0tub3duID0gdHJ1ZTsKLSAgICBtT3duZXIgPSBOVUxMOwotfQotCi12b2lkIFBhcmNlbDo6c2NhbkZvckZkcygpIGNvbnN0Ci17Ci0gICAgYm9vbCBoYXNGZHMgPSBmYWxzZTsKLSAgICBmb3IgKHNpemVfdCBpPTA7IGk8bU9iamVjdHNTaXplOyBpKyspIHsKLSAgICAgICAgY29uc3QgZmxhdF9iaW5kZXJfb2JqZWN0KiBmbGF0Ci0gICAgICAgICAgICA9IHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgZmxhdF9iaW5kZXJfb2JqZWN0Kj4obURhdGEgKyBtT2JqZWN0c1tpXSk7Ci0gICAgICAgIGlmIChmbGF0LT50eXBlID09IEJJTkRFUl9UWVBFX0ZEKSB7Ci0gICAgICAgICAgICBoYXNGZHMgPSB0cnVlOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgbUhhc0ZkcyA9IGhhc0ZkczsKLSAgICBtRmRzS25vd24gPSB0cnVlOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9QaXBlLmNwcCBiL2xpYnMvdXRpbHMvUGlwZS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDYxMzkwNmIuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9QaXBlLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDQ2NSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIFVuaWRpcmVjdGlvbmFsIHBpcGUuCi0vLwotCi0jaW5jbHVkZSA8dXRpbHMvUGlwZS5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaWYgZGVmaW5lZChIQVZFX1dJTjMyX0lQQykKLSMgaW5jbHVkZSA8d2luZG93cy5oPgotI2Vsc2UKLSMgaW5jbHVkZSA8ZmNudGwuaD4KLSMgaW5jbHVkZSA8dW5pc3RkLmg+Ci0jIGluY2x1ZGUgPGVycm5vLmg+Ci0jZW5kaWYKLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi1jb25zdCB1bnNpZ25lZCBsb25nIGtJbnZhbGlkSGFuZGxlID0gKHVuc2lnbmVkIGxvbmcpIC0xOwotCi0KLS8qCi0gKiBDb25zdHJ1Y3Rvci4gIERvIGxpdHRsZS4KLSAqLwotUGlwZTo6UGlwZSh2b2lkKQotICAgIDogbVJlYWROb25CbG9ja2luZyhmYWxzZSksIG1SZWFkSGFuZGxlKGtJbnZhbGlkSGFuZGxlKSwKLSAgICAgIG1Xcml0ZUhhbmRsZShrSW52YWxpZEhhbmRsZSkKLXsKLX0KLQotLyoKLSAqIERlc3RydWN0b3IuICBVc2UgdGhlIHN5c3RlbS1hcHByb3ByaWF0ZSBjbG9zZSBjYWxsLgotICovCi1QaXBlOjp+UGlwZSh2b2lkKQotewotI2lmIGRlZmluZWQoSEFWRV9XSU4zMl9JUEMpCi0gICAgaWYgKG1SZWFkSGFuZGxlICE9IGtJbnZhbGlkSGFuZGxlKSB7Ci0gICAgICAgIGlmICghQ2xvc2VIYW5kbGUoKEhBTkRMRSltUmVhZEhhbmRsZSkpCi0gICAgICAgICAgICBMT0coTE9HX1dBUk4sICJwaXBlIiwgImZhaWxlZCBjbG9zaW5nIHJlYWQgaGFuZGxlICglbGQpXG4iLAotICAgICAgICAgICAgICAgIG1SZWFkSGFuZGxlKTsKLSAgICB9Ci0gICAgaWYgKG1Xcml0ZUhhbmRsZSAhPSBrSW52YWxpZEhhbmRsZSkgewotICAgICAgICBGbHVzaEZpbGVCdWZmZXJzKChIQU5ETEUpbVdyaXRlSGFuZGxlKTsKLSAgICAgICAgaWYgKCFDbG9zZUhhbmRsZSgoSEFORExFKW1Xcml0ZUhhbmRsZSkpCi0gICAgICAgICAgICBMT0coTE9HX1dBUk4sICJwaXBlIiwgImZhaWxlZCBjbG9zaW5nIHdyaXRlIGhhbmRsZSAoJWxkKVxuIiwKLSAgICAgICAgICAgICAgICBtV3JpdGVIYW5kbGUpOwotICAgIH0KLSNlbHNlCi0gICAgaWYgKG1SZWFkSGFuZGxlICE9IGtJbnZhbGlkSGFuZGxlKSB7Ci0gICAgICAgIGlmIChjbG9zZSgoaW50KSBtUmVhZEhhbmRsZSkgIT0gMCkKLSAgICAgICAgICAgIExPRyhMT0dfV0FSTiwgInBpcGUiLCAiZmFpbGVkIGNsb3NpbmcgcmVhZCBmZCAoJWQpXG4iLAotICAgICAgICAgICAgICAgIChpbnQpIG1SZWFkSGFuZGxlKTsKLSAgICB9Ci0gICAgaWYgKG1Xcml0ZUhhbmRsZSAhPSBrSW52YWxpZEhhbmRsZSkgewotICAgICAgICBpZiAoY2xvc2UoKGludCkgbVdyaXRlSGFuZGxlKSAhPSAwKQotICAgICAgICAgICAgTE9HKExPR19XQVJOLCAicGlwZSIsICJmYWlsZWQgY2xvc2luZyB3cml0ZSBmZCAoJWQpXG4iLAotICAgICAgICAgICAgICAgIChpbnQpIG1Xcml0ZUhhbmRsZSk7Ci0gICAgfQotI2VuZGlmCi19Ci0KLS8qCi0gKiBDcmVhdGUgdGhlIHBpcGUuCi0gKgotICogVXNlIHRoZSBQT1NJWCBzdHVmZiBmb3IgZXZlcnl0aGluZyBidXQgV2luZG93cy4KLSAqLwotYm9vbCBQaXBlOjpjcmVhdGUodm9pZCkKLXsKLSAgICBhc3NlcnQobVJlYWRIYW5kbGUgPT0ga0ludmFsaWRIYW5kbGUpOwotICAgIGFzc2VydChtV3JpdGVIYW5kbGUgPT0ga0ludmFsaWRIYW5kbGUpOwotCi0jaWYgZGVmaW5lZChIQVZFX1dJTjMyX0lQQykKLSAgICAvKiB3ZSB1c2UgdGhpcyBhY3Jvc3MgcHJvY2Vzc2VzLCBzbyB0aGV5IG5lZWQgdG8gYmUgaW5oZXJpdGFibGUgKi8KLSAgICBIQU5ETEUgaGFuZGxlc1syXTsKLSAgICBTRUNVUklUWV9BVFRSSUJVVEVTIHNhQXR0cjsKLQotICAgIHNhQXR0ci5uTGVuZ3RoID0gc2l6ZW9mKFNFQ1VSSVRZX0FUVFJJQlVURVMpOwotICAgIHNhQXR0ci5iSW5oZXJpdEhhbmRsZSA9IFRSVUU7Ci0gICAgc2FBdHRyLmxwU2VjdXJpdHlEZXNjcmlwdG9yID0gTlVMTDsKLQotICAgIGlmICghQ3JlYXRlUGlwZSgmaGFuZGxlc1swXSwgJmhhbmRsZXNbMV0sICZzYUF0dHIsIDApKSB7Ci0gICAgICAgIExPRyhMT0dfRVJST1IsICJwaXBlIiwgInVuYWJsZSB0byBjcmVhdGUgcGlwZVxuIik7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgbVJlYWRIYW5kbGUgPSAodW5zaWduZWQgbG9uZykgaGFuZGxlc1swXTsKLSAgICBtV3JpdGVIYW5kbGUgPSAodW5zaWduZWQgbG9uZykgaGFuZGxlc1sxXTsKLSAgICByZXR1cm4gdHJ1ZTsKLSNlbHNlCi0gICAgaW50IGZkc1syXTsKLQotICAgIGlmIChwaXBlKGZkcykgIT0gMCkgewotICAgICAgICBMT0coTE9HX0VSUk9SLCAicGlwZSIsICJ1bmFibGUgdG8gY3JlYXRlIHBpcGVcbiIpOwotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotICAgIG1SZWFkSGFuZGxlID0gZmRzWzBdOwotICAgIG1Xcml0ZUhhbmRsZSA9IGZkc1sxXTsKLSAgICByZXR1cm4gdHJ1ZTsKLSNlbmRpZgotfQotCi0vKgotICogQ3JlYXRlIGEgImhhbGYgcGlwZSIuICBQbGVhc2UsIG5vIFNlZ3dheSByaWRpbmcuCi0gKi8KLWJvb2wgUGlwZTo6Y3JlYXRlUmVhZGVyKHVuc2lnbmVkIGxvbmcgaGFuZGxlKQotewotICAgIG1SZWFkSGFuZGxlID0gaGFuZGxlOwotICAgIGFzc2VydChtV3JpdGVIYW5kbGUgPT0ga0ludmFsaWRIYW5kbGUpOwotICAgIHJldHVybiB0cnVlOwotfQotCi0vKgotICogQ3JlYXRlIGEgImhhbGYgcGlwZSIgZm9yIHdyaXRpbmcuCi0gKi8KLWJvb2wgUGlwZTo6Y3JlYXRlV3JpdGVyKHVuc2lnbmVkIGxvbmcgaGFuZGxlKQotewotICAgIG1Xcml0ZUhhbmRsZSA9IGhhbmRsZTsKLSAgICBhc3NlcnQobVJlYWRIYW5kbGUgPT0ga0ludmFsaWRIYW5kbGUpOwotICAgIHJldHVybiB0cnVlOwotfQotCi0vKgotICogUmV0dXJuICJ0cnVlIiBpZiBjcmVhdGUoKSBoYXMgYmVlbiBjYWxsZWQgc3VjY2Vzc2Z1bGx5LgotICovCi1ib29sIFBpcGU6OmlzQ3JlYXRlZCh2b2lkKQotewotICAgIC8vIG9uZSBvciB0aGUgb3RoZXIgc2hvdWxkIGJlIG9wZW4KLSAgICByZXR1cm4gKG1SZWFkSGFuZGxlICE9IGtJbnZhbGlkSGFuZGxlIHx8IG1Xcml0ZUhhbmRsZSAhPSBrSW52YWxpZEhhbmRsZSk7Ci19Ci0KLQotLyoKLSAqIFJlYWQgZGF0YSBmcm9tIHRoZSBwaXBlLgotICoKLSAqIEZvciBMaW51eCBhbmQgRGFyd2luLCBqdXN0IGNhbGwgcmVhZCgpLiAgRm9yIFdpbmRvd3MsIGltcGxlbWVudAotICogbm9uLWJsb2NraW5nIHJlYWRzIGJ5IGNhbGxpbmcgUGVla05hbWVkUGlwZSBmaXJzdC4KLSAqLwotaW50IFBpcGU6OnJlYWQodm9pZCogYnVmLCBpbnQgY291bnQpCi17Ci0gICAgYXNzZXJ0KG1SZWFkSGFuZGxlICE9IGtJbnZhbGlkSGFuZGxlKTsKLQotI2lmIGRlZmluZWQoSEFWRV9XSU4zMl9JUEMpCi0gICAgRFdPUkQgdG90YWxCeXRlc0F2YWlsID0gY291bnQ7Ci0gICAgRFdPUkQgYnl0ZXNSZWFkOwotCi0gICAgaWYgKG1SZWFkTm9uQmxvY2tpbmcpIHsKLSAgICAgICAgLy8gdXNlIFBlZWtOYW1lZFBpcGUgdG8gYWRqdXN0IHJlYWQgY291bnQgZXhwZWN0YXRpb25zCi0gICAgICAgIGlmICghUGVla05hbWVkUGlwZSgoSEFORExFKSBtUmVhZEhhbmRsZSwgTlVMTCwgMCwgTlVMTCwKLSAgICAgICAgICAgICAgICAmdG90YWxCeXRlc0F2YWlsLCBOVUxMKSkKLSAgICAgICAgewotICAgICAgICAgICAgTE9HKExPR19FUlJPUiwgInBpcGUiLCAiUGVla05hbWVkUGlwZSBmYWlsZWRcbiIpOwotICAgICAgICAgICAgcmV0dXJuIC0xOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHRvdGFsQnl0ZXNBdmFpbCA9PSAwKQotICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgaWYgKCFSZWFkRmlsZSgoSEFORExFKSBtUmVhZEhhbmRsZSwgYnVmLCB0b3RhbEJ5dGVzQXZhaWwsICZieXRlc1JlYWQsCi0gICAgICAgICAgICBOVUxMKSkKLSAgICB7Ci0gICAgICAgIERXT1JEIGVyciA9IEdldExhc3RFcnJvcigpOwotICAgICAgICBpZiAoZXJyID09IEVSUk9SX0hBTkRMRV9FT0YgfHwgZXJyID09IEVSUk9SX0JST0tFTl9QSVBFKQotICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgIExPRyhMT0dfRVJST1IsICJwaXBlIiwgIlJlYWRGaWxlIGZhaWxlZCAoZXJyPSVsZClcbiIsIGVycik7Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0KLSAgICByZXR1cm4gKGludCkgYnl0ZXNSZWFkOwotI2Vsc2UKLSAgICBpbnQgY2M7Ci0gICAgY2MgPSA6OnJlYWQobVJlYWRIYW5kbGUsIGJ1ZiwgY291bnQpOwotICAgIGlmIChjYyA8IDAgJiYgZXJybm8gPT0gRUFHQUlOKQotICAgICAgICByZXR1cm4gMDsKLSAgICByZXR1cm4gY2M7Ci0jZW5kaWYKLX0KLQotLyoKLSAqIFdyaXRlIGRhdGEgdG8gdGhlIHBpcGUuCi0gKgotICogUE9TSVggc3lzdGVtcyBhcmUgdHJpdmlhbCwgV2luZG93cyB1c2VzIGEgZGlmZmVyZW50IGNhbGwgYW5kIGRvZXNuJ3QKLSAqIGhhbmRsZSBub24tYmxvY2tpbmcgd3JpdGVzLgotICoKLSAqIElmIHdlIGFkZCBub24tYmxvY2tpbmcgc3VwcG9ydCBoZXJlLCB3ZSBwcm9iYWJseSB3YW50IHRvIG1ha2UgaXQgYW4KLSAqIGFsbC1vci1ub3RoaW5nIHdyaXRlLgotICoKLSAqIERPIE5PVCB1c2UgTE9HKCkgaGVyZSwgd2UgY291bGQgYmUgd3JpdGluZyBhIGxvZyBtZXNzYWdlLgotICovCi1pbnQgUGlwZTo6d3JpdGUoY29uc3Qgdm9pZCogYnVmLCBpbnQgY291bnQpCi17Ci0gICAgYXNzZXJ0KG1Xcml0ZUhhbmRsZSAhPSBrSW52YWxpZEhhbmRsZSk7Ci0KLSNpZiBkZWZpbmVkKEhBVkVfV0lOMzJfSVBDKQotICAgIERXT1JEIGJ5dGVzV3JpdHRlbjsKLQotICAgIGlmIChtV3JpdGVOb25CbG9ja2luZykgewotICAgICAgICAvLyBCVUc6IGNhbid0IHVzZSBQZWVrTmFtZWRQaXBlKCkgdG8gZ2V0IHRoZSBhbW91bnQgb2Ygc3BhY2UKLSAgICAgICAgLy8gbGVmdC4gIExvb2tzIGxpa2Ugd2UgbmVlZCB0byB1c2UgIm92ZXJsYXBwZWQgSS9PIiBmdW5jdGlvbnMuCi0gICAgICAgIC8vIEkganVzdCBkb24ndCBjYXJlIHRoYXQgbXVjaC4KLSAgICB9Ci0KLSAgICBpZiAoIVdyaXRlRmlsZSgoSEFORExFKSBtV3JpdGVIYW5kbGUsIGJ1ZiwgY291bnQsICZieXRlc1dyaXR0ZW4sIE5VTEwpKSB7Ci0gICAgICAgIC8vIGNhbid0IExPRywgdXNlIHN0ZGVycgotICAgICAgICBmcHJpbnRmKHN0ZGVyciwgIldyaXRlRmlsZSBmYWlsZWQgKGVycj0lbGQpXG4iLCBHZXRMYXN0RXJyb3IoKSk7Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0KLSAgICByZXR1cm4gKGludCkgYnl0ZXNXcml0dGVuOwotI2Vsc2UKLSAgICBpbnQgY2M7Ci0gICAgY2MgPSA6OndyaXRlKG1Xcml0ZUhhbmRsZSwgYnVmLCBjb3VudCk7Ci0gICAgaWYgKGNjIDwgMCAmJiBlcnJubyA9PSBFQUdBSU4pCi0gICAgICAgIHJldHVybiAwOwotICAgIHJldHVybiBjYzsKLSNlbmRpZgotfQotCi0vKgotICogRmlndXJlIG91dCBpZiB0aGVyZSBpcyBkYXRhIGF2YWlsYWJsZSBvbiB0aGUgcmVhZCBmZC4KLSAqCi0gKiBXZSByZXR1cm4gInRydWUiIG9uIGVycm9yIGJlY2F1c2Ugd2Ugd2FudCB0aGUgY2FsbGVyIHRvIHRyeSB0byByZWFkCi0gKiBmcm9tIHRoZSBwaXBlLiAgVGhleSdsbCBub3RpY2UgdGhlIHJlYWQgZmFpbHVyZSBhbmQgZG8gc29tZXRoaW5nCi0gKiBhcHByb3ByaWF0ZS4KLSAqLwotYm9vbCBQaXBlOjpyZWFkUmVhZHkodm9pZCkKLXsKLSAgICBhc3NlcnQobVJlYWRIYW5kbGUgIT0ga0ludmFsaWRIYW5kbGUpOwotCi0jaWYgZGVmaW5lZChIQVZFX1dJTjMyX0lQQykKLSAgICBEV09SRCB0b3RhbEJ5dGVzQXZhaWw7Ci0KLSAgICBpZiAoIVBlZWtOYW1lZFBpcGUoKEhBTkRMRSkgbVJlYWRIYW5kbGUsIE5VTEwsIDAsIE5VTEwsCi0gICAgICAgICAgICAmdG90YWxCeXRlc0F2YWlsLCBOVUxMKSkKLSAgICB7Ci0gICAgICAgIExPRyhMT0dfRVJST1IsICJwaXBlIiwgIlBlZWtOYW1lZFBpcGUgZmFpbGVkXG4iKTsKLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgcmV0dXJuICh0b3RhbEJ5dGVzQXZhaWwgIT0gMCk7Ci0jZWxzZQotICAgIGVycm5vID0gMDsKLSAgICBmZF9zZXQgcmVhZGZkczsKLSAgICBzdHJ1Y3QgdGltZXZhbCB0diA9IHsgMCwgMCB9OwotICAgIGludCBjYzsKLQotICAgIEZEX1pFUk8oJnJlYWRmZHMpOwotICAgIEZEX1NFVChtUmVhZEhhbmRsZSwgJnJlYWRmZHMpOwotCi0gICAgY2MgPSBzZWxlY3QobVJlYWRIYW5kbGUrMSwgJnJlYWRmZHMsIE5VTEwsIE5VTEwsICZ0dik7Ci0gICAgaWYgKGNjIDwgMCkgewotICAgICAgICBMT0coTE9HX0VSUk9SLCAicGlwZSIsICJzZWxlY3QoKSBmYWlsZWRcbiIpOwotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9IGVsc2UgaWYgKGNjID09IDApIHsKLSAgICAgICAgLyogdGltZWQgb3V0LCBub3RoaW5nIGF2YWlsYWJsZSAqLwotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfSBlbHNlIGlmIChjYyA9PSAxKSB7Ci0gICAgICAgIC8qIG91ciBmZCBpcyByZWFkeSAqLwotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9IGVsc2UgewotICAgICAgICBMT0coTE9HX0VSUk9SLCAicGlwZSIsICJIVUg/IHNlbGVjdCgpIHJldHVybmVkID4gMVxuIik7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLSNlbmRpZgotfQotCi0vKgotICogRW5hYmxlIG9yIGRpc2FibGUgbm9uLWJsb2NraW5nIG1vZGUgZm9yIHRoZSByZWFkIGRlc2NyaXB0b3IuCi0gKgotICogTk9URTogdGhlIGNhbGxzIHN1Y2NlZWQgdW5kZXIgTWFjIE9TIFgsIGJ1dCB0aGUgcGlwZSBkb2Vzbid0IGFwcGVhciB0bwotICogYWN0dWFsbHkgYmUgaW4gbm9uLWJsb2NraW5nIG1vZGUuICBJZiB0aGlzIG1hdHRlcnMgLS0gaS5lLiB5b3UncmUgbm90Ci0gKiB1c2luZyBhIHNlbGVjdCgpIGNhbGwgLS0gcHV0IGEgY2FsbCB0byByZWFkUmVhZHkoKSBpbiBmcm9udCBvZiB0aGUKLSAqIDo6cmVhZCgpIGNhbGwsIHdpdGggYSBQSVBFX05PTkJMT0NLX0JST0tFTiAjaWZkZWYgaW4gdGhlIE1ha2VmaWxlIGZvcgotICogRGFyd2luLgotICovCi1ib29sIFBpcGU6OnNldFJlYWROb25CbG9ja2luZyhib29sIHZhbCkKLXsKLSAgICBhc3NlcnQobVJlYWRIYW5kbGUgIT0ga0ludmFsaWRIYW5kbGUpOwotCi0jaWYgZGVmaW5lZChIQVZFX1dJTjMyX0lQQykKLSAgICAvLyBub3RoaW5nIHRvIGRvCi0jZWxzZQotICAgIGludCBmbGFnczsKLQotICAgIGlmIChmY250bChtUmVhZEhhbmRsZSwgRl9HRVRGTCwgJmZsYWdzKSA9PSAtMSkgewotICAgICAgICBMT0coTE9HX0VSUk9SLCAicGlwZSIsICJjb3VsZG4ndCBnZXQgZmxhZ3MgZm9yIHBpcGUgcmVhZCBmZFxuIik7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgaWYgKHZhbCkKLSAgICAgICAgZmxhZ3MgfD0gT19OT05CTE9DSzsKLSAgICBlbHNlCi0gICAgICAgIGZsYWdzICY9IH4oT19OT05CTE9DSyk7Ci0gICAgaWYgKGZjbnRsKG1SZWFkSGFuZGxlLCBGX1NFVEZMLCAmZmxhZ3MpID09IC0xKSB7Ci0gICAgICAgIExPRyhMT0dfRVJST1IsICJwaXBlIiwgImNvdWxkbid0IHNldCBmbGFncyBmb3IgcGlwZSByZWFkIGZkXG4iKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSNlbmRpZgotCi0gICAgbVJlYWROb25CbG9ja2luZyA9IHZhbDsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotLyoKLSAqIEVuYWJsZSBvciBkaXNhYmxlIG5vbi1ibG9ja2luZyBtb2RlIGZvciB0aGUgd3JpdGUgZGVzY3JpcHRvci4KLSAqCi0gKiBBcyB3aXRoIHNldFJlYWROb25CbG9ja2luZygpLCB0aGlzIGRvZXMgbm90IHdvcmsgb24gdGhlIE1hYy4KLSAqLwotYm9vbCBQaXBlOjpzZXRXcml0ZU5vbkJsb2NraW5nKGJvb2wgdmFsKQotewotICAgIGFzc2VydChtV3JpdGVIYW5kbGUgIT0ga0ludmFsaWRIYW5kbGUpOwotCi0jaWYgZGVmaW5lZChIQVZFX1dJTjMyX0lQQykKLSAgICAvLyBub3RoaW5nIHRvIGRvCi0jZWxzZQotICAgIGludCBmbGFnczsKLQotICAgIGlmIChmY250bChtV3JpdGVIYW5kbGUsIEZfR0VURkwsICZmbGFncykgPT0gLTEpIHsKLSAgICAgICAgTE9HKExPR19XQVJOLCAicGlwZSIsCi0gICAgICAgICAgICAiV2FybmluZzogY291bGRuJ3QgZ2V0IGZsYWdzIGZvciBwaXBlIHdyaXRlIGZkIChlcnJubz0lZClcbiIsCi0gICAgICAgICAgICBlcnJubyk7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgaWYgKHZhbCkKLSAgICAgICAgZmxhZ3MgfD0gT19OT05CTE9DSzsKLSAgICBlbHNlCi0gICAgICAgIGZsYWdzICY9IH4oT19OT05CTE9DSyk7Ci0gICAgaWYgKGZjbnRsKG1Xcml0ZUhhbmRsZSwgRl9TRVRGTCwgJmZsYWdzKSA9PSAtMSkgewotICAgICAgICBMT0coTE9HX1dBUk4sICJwaXBlIiwKLSAgICAgICAgICAgICJXYXJuaW5nOiBjb3VsZG4ndCBzZXQgZmxhZ3MgZm9yIHBpcGUgd3JpdGUgZmQgKGVycm5vPSVkKVxuIiwKLSAgICAgICAgICAgIGVycm5vKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSNlbmRpZgotCi0gICAgbVdyaXRlTm9uQmxvY2tpbmcgPSB2YWw7Ci0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLS8qCi0gKiBTcGVjaWZ5IHdoZXRoZXIgYSBmaWxlIGRlc2NyaXB0b3IgY2FuIGJlIGluaGVyaXRlZCBieSBhIGNoaWxkIHByb2Nlc3MuCi0gKiBVbmRlciBMaW51eCB0aGlzIG1lYW5zIHNldHRpbmcgdGhlIGNsb3NlLW9uLWV4ZWMgZmxhZywgdW5kZXIgV2luZG93cwotICogdGhpcyBpcyBTZXRIYW5kbGVJbmZvcm1hdGlvbihIQU5ETEVfRkxBR19JTkhFUklUKS4KLSAqLwotYm9vbCBQaXBlOjpkaXNhbGxvd1JlYWRJbmhlcml0KHZvaWQpCi17Ci0gICAgaWYgKG1SZWFkSGFuZGxlID09IGtJbnZhbGlkSGFuZGxlKQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0KLSNpZiBkZWZpbmVkKEhBVkVfV0lOMzJfSVBDKQotICAgIGlmIChTZXRIYW5kbGVJbmZvcm1hdGlvbigoSEFORExFKSBtUmVhZEhhbmRsZSwgSEFORExFX0ZMQUdfSU5IRVJJVCwgMCkgPT0gMCkKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotI2Vsc2UKLSAgICBpZiAoZmNudGwoKGludCkgbVJlYWRIYW5kbGUsIEZfU0VURkQsIEZEX0NMT0VYRUMpICE9IDApCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSNlbmRpZgotICAgIHJldHVybiB0cnVlOwotfQotYm9vbCBQaXBlOjpkaXNhbGxvd1dyaXRlSW5oZXJpdCh2b2lkKQotewotICAgIGlmIChtV3JpdGVIYW5kbGUgPT0ga0ludmFsaWRIYW5kbGUpCi0gICAgICAgIHJldHVybiBmYWxzZTsKLQotI2lmIGRlZmluZWQoSEFWRV9XSU4zMl9JUEMpCi0gICAgaWYgKFNldEhhbmRsZUluZm9ybWF0aW9uKChIQU5ETEUpIG1Xcml0ZUhhbmRsZSwgSEFORExFX0ZMQUdfSU5IRVJJVCwgMCkgPT0gMCkKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotI2Vsc2UKLSAgICBpZiAoZmNudGwoKGludCkgbVdyaXRlSGFuZGxlLCBGX1NFVEZELCBGRF9DTE9FWEVDKSAhPSAwKQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0jZW5kaWYKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotLyoKLSAqIENsb3NlIHJlYWQgZGVzY3JpcHRvci4KLSAqLwotYm9vbCBQaXBlOjpjbG9zZVJlYWQodm9pZCkKLXsKLSAgICBpZiAobVJlYWRIYW5kbGUgPT0ga0ludmFsaWRIYW5kbGUpCi0gICAgICAgIHJldHVybiBmYWxzZTsKLQotI2lmIGRlZmluZWQoSEFWRV9XSU4zMl9JUEMpCi0gICAgaWYgKG1SZWFkSGFuZGxlICE9IGtJbnZhbGlkSGFuZGxlKSB7Ci0gICAgICAgIGlmICghQ2xvc2VIYW5kbGUoKEhBTkRMRSltUmVhZEhhbmRsZSkpIHsKLSAgICAgICAgICAgIExPRyhMT0dfV0FSTiwgInBpcGUiLCAiZmFpbGVkIGNsb3NpbmcgcmVhZCBoYW5kbGVcbiIpOwotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgfQotI2Vsc2UKLSAgICBpZiAobVJlYWRIYW5kbGUgIT0ga0ludmFsaWRIYW5kbGUpIHsKLSAgICAgICAgaWYgKGNsb3NlKChpbnQpIG1SZWFkSGFuZGxlKSAhPSAwKSB7Ci0gICAgICAgICAgICBMT0coTE9HX1dBUk4sICJwaXBlIiwgImZhaWxlZCBjbG9zaW5nIHJlYWQgZmRcbiIpOwotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgfQotI2VuZGlmCi0gICAgbVJlYWRIYW5kbGUgPSBrSW52YWxpZEhhbmRsZTsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotLyoKLSAqIENsb3NlIHdyaXRlIGRlc2NyaXB0b3IuCi0gKi8KLWJvb2wgUGlwZTo6Y2xvc2VXcml0ZSh2b2lkKQotewotICAgIGlmIChtV3JpdGVIYW5kbGUgPT0ga0ludmFsaWRIYW5kbGUpCi0gICAgICAgIHJldHVybiBmYWxzZTsKLQotI2lmIGRlZmluZWQoSEFWRV9XSU4zMl9JUEMpCi0gICAgaWYgKG1Xcml0ZUhhbmRsZSAhPSBrSW52YWxpZEhhbmRsZSkgewotICAgICAgICBpZiAoIUNsb3NlSGFuZGxlKChIQU5ETEUpbVdyaXRlSGFuZGxlKSkgewotICAgICAgICAgICAgTE9HKExPR19XQVJOLCAicGlwZSIsICJmYWlsZWQgY2xvc2luZyB3cml0ZSBoYW5kbGVcbiIpOwotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgfQotI2Vsc2UKLSAgICBpZiAobVdyaXRlSGFuZGxlICE9IGtJbnZhbGlkSGFuZGxlKSB7Ci0gICAgICAgIGlmIChjbG9zZSgoaW50KSBtV3JpdGVIYW5kbGUpICE9IDApIHsKLSAgICAgICAgICAgIExPRyhMT0dfV0FSTiwgInBpcGUiLCAiZmFpbGVkIGNsb3Npbmcgd3JpdGUgZmRcbiIpOwotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgfQotI2VuZGlmCi0gICAgbVdyaXRlSGFuZGxlID0ga0ludmFsaWRIYW5kbGU7Ci0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLS8qCi0gKiBHZXQgdGhlIHJlYWQgaGFuZGxlLgotICovCi11bnNpZ25lZCBsb25nIFBpcGU6OmdldFJlYWRIYW5kbGUodm9pZCkKLXsKLSAgICBhc3NlcnQobVJlYWRIYW5kbGUgIT0ga0ludmFsaWRIYW5kbGUpOwotCi0gICAgcmV0dXJuIG1SZWFkSGFuZGxlOwotfQotCi0vKgotICogR2V0IHRoZSB3cml0ZSBoYW5kbGUuCi0gKi8KLXVuc2lnbmVkIGxvbmcgUGlwZTo6Z2V0V3JpdGVIYW5kbGUodm9pZCkKLXsKLSAgICBhc3NlcnQobVdyaXRlSGFuZGxlICE9IGtJbnZhbGlkSGFuZGxlKTsKLQotICAgIHJldHVybiBtV3JpdGVIYW5kbGU7Ci19Ci0KZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvUHJvY2Vzc1N0YXRlLmNwcCBiL2xpYnMvdXRpbHMvUHJvY2Vzc1N0YXRlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDU2N2RmNi4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1Byb2Nlc3NTdGF0ZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwzOTggKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNkZWZpbmUgTE9HX1RBRyAiUHJvY2Vzc1N0YXRlIgotCi0jaW5jbHVkZSA8Y3V0aWxzL3Byb2Nlc3NfbmFtZS5oPgotCi0jaW5jbHVkZSA8dXRpbHMvUHJvY2Vzc1N0YXRlLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9CcEJpbmRlci5oPgotI2luY2x1ZGUgPHV0aWxzL0lQQ1RocmVhZFN0YXRlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0jaW5jbHVkZSA8dXRpbHMvU3RyaW5nOC5oPgotI2luY2x1ZGUgPHV0aWxzL0lTZXJ2aWNlTWFuYWdlci5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLSNpbmNsdWRlIDxwcml2YXRlL3V0aWxzL2JpbmRlcl9tb2R1bGUuaD4KLSNpbmNsdWRlIDxwcml2YXRlL3V0aWxzL1N0YXRpYy5oPgotCi0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8c3lzL2lvY3RsLmg+Ci0jaW5jbHVkZSA8c3lzL21tYW4uaD4KLSNpbmNsdWRlIDxzeXMvc3RhdC5oPgotCi0jZGVmaW5lIEJJTkRFUl9WTV9TSVpFICgxKjEwMjQqMTAyNCkKLQotc3RhdGljIGJvb2wgZ1NpbmdsZVByb2Nlc3MgPSBmYWxzZTsKLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotIAotLy8gR2xvYmFsIHZhcmlhYmxlcwotaW50ICAgICAgICAgICAgICAgICBtQXJnQzsKLWNvbnN0IGNoYXIqIGNvbnN0KiAgbUFyZ1Y7Ci1pbnQgICAgICAgICAgICAgICAgIG1BcmdMZW47Ci0KLWNsYXNzIFBvb2xUaHJlYWQgOiBwdWJsaWMgVGhyZWFkCi17Ci1wdWJsaWM6Ci0gICAgUG9vbFRocmVhZChib29sIGlzTWFpbikKLSAgICAgICAgOiBtSXNNYWluKGlzTWFpbikKLSAgICB7Ci0gICAgfQotICAgIAotcHJvdGVjdGVkOgotICAgIHZpcnR1YWwgYm9vbCB0aHJlYWRMb29wKCkKLSAgICB7Ci0gICAgICAgIElQQ1RocmVhZFN0YXRlOjpzZWxmKCktPmpvaW5UaHJlYWRQb29sKG1Jc01haW4pOwotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotICAgIAotICAgIGNvbnN0IGJvb2wgbUlzTWFpbjsKLX07Ci0KLXNwPFByb2Nlc3NTdGF0ZT4gUHJvY2Vzc1N0YXRlOjpzZWxmKCkKLXsKLSAgICBpZiAoZ1Byb2Nlc3MgIT0gTlVMTCkgcmV0dXJuIGdQcm9jZXNzOwotICAgIAotICAgIEF1dG9NdXRleCBfbChnUHJvY2Vzc011dGV4KTsKLSAgICBpZiAoZ1Byb2Nlc3MgPT0gTlVMTCkgZ1Byb2Nlc3MgPSBuZXcgUHJvY2Vzc1N0YXRlOwotICAgIHJldHVybiBnUHJvY2VzczsKLX0KLQotdm9pZCBQcm9jZXNzU3RhdGU6OnNldFNpbmdsZVByb2Nlc3MoYm9vbCBzaW5nbGVQcm9jZXNzKQotewotICAgIGdTaW5nbGVQcm9jZXNzID0gc2luZ2xlUHJvY2VzczsKLX0KLQotCi12b2lkIFByb2Nlc3NTdGF0ZTo6c2V0Q29udGV4dE9iamVjdChjb25zdCBzcDxJQmluZGVyPiYgb2JqZWN0KQotewotICAgIHNldENvbnRleHRPYmplY3Qob2JqZWN0LCBTdHJpbmcxNigiZGVmYXVsdCIpKTsKLX0KLQotc3A8SUJpbmRlcj4gUHJvY2Vzc1N0YXRlOjpnZXRDb250ZXh0T2JqZWN0KGNvbnN0IHNwPElCaW5kZXI+JiBjYWxsZXIpCi17Ci0gICAgaWYgKHN1cHBvcnRzUHJvY2Vzc2VzKCkpIHsKLSAgICAgICAgcmV0dXJuIGdldFN0cm9uZ1Byb3h5Rm9ySGFuZGxlKDApOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHJldHVybiBnZXRDb250ZXh0T2JqZWN0KFN0cmluZzE2KCJkZWZhdWx0IiksIGNhbGxlcik7Ci0gICAgfQotfQotCi12b2lkIFByb2Nlc3NTdGF0ZTo6c2V0Q29udGV4dE9iamVjdChjb25zdCBzcDxJQmluZGVyPiYgb2JqZWN0LCBjb25zdCBTdHJpbmcxNiYgbmFtZSkKLXsKLSAgICBBdXRvTXV0ZXggX2wobUxvY2spOwotICAgIG1Db250ZXh0cy5hZGQobmFtZSwgb2JqZWN0KTsKLX0KLQotc3A8SUJpbmRlcj4gUHJvY2Vzc1N0YXRlOjpnZXRDb250ZXh0T2JqZWN0KGNvbnN0IFN0cmluZzE2JiBuYW1lLCBjb25zdCBzcDxJQmluZGVyPiYgY2FsbGVyKQotewotICAgIG1Mb2NrLmxvY2soKTsKLSAgICBzcDxJQmluZGVyPiBvYmplY3QoCi0gICAgICAgIG1Db250ZXh0cy5pbmRleE9mS2V5KG5hbWUpID49IDAgPyBtQ29udGV4dHMudmFsdWVGb3IobmFtZSkgOiBOVUxMKTsKLSAgICBtTG9jay51bmxvY2soKTsKLSAgICAKLSAgICAvL3ByaW50ZigiR2V0dGluZyBjb250ZXh0IG9iamVjdCAlcyBmb3IgJXBcbiIsIFN0cmluZzgobmFtZSkuc3RyaW5nKCksIGNhbGxlci5nZXQoKSk7Ci0gICAgCi0gICAgaWYgKG9iamVjdCAhPSBOVUxMKSByZXR1cm4gb2JqZWN0OwotCi0gICAgLy8gRG9uJ3QgYXR0ZW1wdCB0byByZXRyaWV2ZSBjb250ZXh0cyBpZiB3ZSBtYW5hZ2UgdGhlbQotICAgIGlmIChtTWFuYWdlc0NvbnRleHRzKSB7Ci0gICAgICAgIExPR0UoImdldENvbnRleHRPYmplY3QoJXMpIGZhaWxlZCwgYnV0IHdlIG1hbmFnZSB0aGUgY29udGV4dHMhXG4iLAotICAgICAgICAgICAgU3RyaW5nOChuYW1lKS5zdHJpbmcoKSk7Ci0gICAgICAgIHJldHVybiBOVUxMOwotICAgIH0KLSAgICAKLSAgICBJUENUaHJlYWRTdGF0ZSogaXBjID0gSVBDVGhyZWFkU3RhdGU6OnNlbGYoKTsKLSAgICB7Ci0gICAgICAgIFBhcmNlbCBkYXRhLCByZXBseTsKLSAgICAgICAgLy8gbm8gaW50ZXJmYWNlIHRva2VuIG9uIHRoaXMgbWFnaWMgdHJhbnNhY3Rpb24KLSAgICAgICAgZGF0YS53cml0ZVN0cmluZzE2KG5hbWUpOwotICAgICAgICBkYXRhLndyaXRlU3Ryb25nQmluZGVyKGNhbGxlcik7Ci0gICAgICAgIHN0YXR1c190IHJlc3VsdCA9IGlwYy0+dHJhbnNhY3QoMCAvKm1hZ2ljKi8sIDAsIGRhdGEsICZyZXBseSwgMCk7Ci0gICAgICAgIGlmIChyZXN1bHQgPT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgIG9iamVjdCA9IHJlcGx5LnJlYWRTdHJvbmdCaW5kZXIoKTsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBpcGMtPmZsdXNoQ29tbWFuZHMoKTsKLSAgICAKLSAgICBpZiAob2JqZWN0ICE9IE5VTEwpIHNldENvbnRleHRPYmplY3Qob2JqZWN0LCBuYW1lKTsKLSAgICByZXR1cm4gb2JqZWN0OwotfQotCi1ib29sIFByb2Nlc3NTdGF0ZTo6c3VwcG9ydHNQcm9jZXNzZXMoKSBjb25zdAotewotICAgIHJldHVybiBtRHJpdmVyRkQgPj0gMDsKLX0KLQotdm9pZCBQcm9jZXNzU3RhdGU6OnN0YXJ0VGhyZWFkUG9vbCgpCi17Ci0gICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLSAgICBpZiAoIW1UaHJlYWRQb29sU3RhcnRlZCkgewotICAgICAgICBtVGhyZWFkUG9vbFN0YXJ0ZWQgPSB0cnVlOwotICAgICAgICBzcGF3blBvb2xlZFRocmVhZCh0cnVlKTsKLSAgICB9Ci19Ci0KLWJvb2wgUHJvY2Vzc1N0YXRlOjppc0NvbnRleHRNYW5hZ2VyKHZvaWQpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1NYW5hZ2VzQ29udGV4dHM7Ci19Ci0KLWJvb2wgUHJvY2Vzc1N0YXRlOjpiZWNvbWVDb250ZXh0TWFuYWdlcihjb250ZXh0X2NoZWNrX2Z1bmMgY2hlY2tGdW5jLCB2b2lkKiB1c2VyRGF0YSkKLXsKLSAgICBpZiAoIW1NYW5hZ2VzQ29udGV4dHMpIHsKLSAgICAgICAgQXV0b011dGV4IF9sKG1Mb2NrKTsKLSAgICAgICAgbUJpbmRlckNvbnRleHRDaGVja0Z1bmMgPSBjaGVja0Z1bmM7Ci0gICAgICAgIG1CaW5kZXJDb250ZXh0VXNlckRhdGEgPSB1c2VyRGF0YTsKLSAgICAgICAgaWYgKG1Ecml2ZXJGRCA+PSAwKSB7Ci0gICAgICAgICAgICBpbnQgZHVtbXkgPSAwOwotI2lmIGRlZmluZWQoSEFWRV9BTkRST0lEX09TKQotICAgICAgICAgICAgc3RhdHVzX3QgcmVzdWx0ID0gaW9jdGwobURyaXZlckZELCBCSU5ERVJfU0VUX0NPTlRFWFRfTUdSLCAmZHVtbXkpOwotI2Vsc2UKLSAgICAgICAgICAgIHN0YXR1c190IHJlc3VsdCA9IElOVkFMSURfT1BFUkFUSU9OOwotI2VuZGlmCi0gICAgICAgICAgICBpZiAocmVzdWx0ID09IDApIHsKLSAgICAgICAgICAgICAgICBtTWFuYWdlc0NvbnRleHRzID0gdHJ1ZTsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAocmVzdWx0ID09IC0xKSB7Ci0gICAgICAgICAgICAgICAgbUJpbmRlckNvbnRleHRDaGVja0Z1bmMgPSBOVUxMOwotICAgICAgICAgICAgICAgIG1CaW5kZXJDb250ZXh0VXNlckRhdGEgPSBOVUxMOwotICAgICAgICAgICAgICAgIExPR0UoIkJpbmRlciBpb2N0bCB0byBiZWNvbWUgY29udGV4dCBtYW5hZ2VyIGZhaWxlZDogJXNcbiIsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBJZiB0aGVyZSBpcyBubyBkcml2ZXIsIG91ciBvbmx5IHdvcmxkIGlzIHRoZSBsb2NhbAotICAgICAgICAgICAgLy8gcHJvY2VzcyBzbyB3ZSBjYW4gYWx3YXlzIGJlY29tZSB0aGUgY29udGV4dCBtYW5hZ2VyIHRoZXJlLgotICAgICAgICAgICAgbU1hbmFnZXNDb250ZXh0cyA9IHRydWU7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIG1NYW5hZ2VzQ29udGV4dHM7Ci19Ci0KLVByb2Nlc3NTdGF0ZTo6aGFuZGxlX2VudHJ5KiBQcm9jZXNzU3RhdGU6Omxvb2t1cEhhbmRsZUxvY2tlZChpbnQzMl90IGhhbmRsZSkKLXsKLSAgICBjb25zdCBzaXplX3QgTj1tSGFuZGxlVG9PYmplY3Quc2l6ZSgpOwotICAgIGlmIChOIDw9IChzaXplX3QpaGFuZGxlKSB7Ci0gICAgICAgIGhhbmRsZV9lbnRyeSBlOwotICAgICAgICBlLmJpbmRlciA9IE5VTEw7Ci0gICAgICAgIGUucmVmcyA9IE5VTEw7Ci0gICAgICAgIHN0YXR1c190IGVyciA9IG1IYW5kbGVUb09iamVjdC5pbnNlcnRBdChlLCBOLCBoYW5kbGUrMS1OKTsKLSAgICAgICAgaWYgKGVyciA8IE5PX0VSUk9SKSByZXR1cm4gTlVMTDsKLSAgICB9Ci0gICAgcmV0dXJuICZtSGFuZGxlVG9PYmplY3QuZWRpdEl0ZW1BdChoYW5kbGUpOwotfQotCi1zcDxJQmluZGVyPiBQcm9jZXNzU3RhdGU6OmdldFN0cm9uZ1Byb3h5Rm9ySGFuZGxlKGludDMyX3QgaGFuZGxlKQotewotICAgIHNwPElCaW5kZXI+IHJlc3VsdDsKLQotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0KLSAgICBoYW5kbGVfZW50cnkqIGUgPSBsb29rdXBIYW5kbGVMb2NrZWQoaGFuZGxlKTsKLQotICAgIGlmIChlICE9IE5VTEwpIHsKLSAgICAgICAgLy8gV2UgbmVlZCB0byBjcmVhdGUgYSBuZXcgQnBCaW5kZXIgaWYgdGhlcmUgaXNuJ3QgY3VycmVudGx5IG9uZSwgT1Igd2UKLSAgICAgICAgLy8gYXJlIHVuYWJsZSB0byBhY3F1aXJlIGEgd2VhayByZWZlcmVuY2Ugb24gdGhpcyBjdXJyZW50IG9uZS4gIFNlZSBjb21tZW50Ci0gICAgICAgIC8vIGluIGdldFdlYWtQcm94eUZvckhhbmRsZSgpIGZvciBtb3JlIGluZm8gYWJvdXQgdGhpcy4KLSAgICAgICAgSUJpbmRlciogYiA9IGUtPmJpbmRlcjsKLSAgICAgICAgaWYgKGIgPT0gTlVMTCB8fCAhZS0+cmVmcy0+YXR0ZW1wdEluY1dlYWsodGhpcykpIHsKLSAgICAgICAgICAgIGIgPSBuZXcgQnBCaW5kZXIoaGFuZGxlKTsgCi0gICAgICAgICAgICBlLT5iaW5kZXIgPSBiOwotICAgICAgICAgICAgaWYgKGIpIGUtPnJlZnMgPSBiLT5nZXRXZWFrUmVmcygpOwotICAgICAgICAgICAgcmVzdWx0ID0gYjsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIFRoaXMgbGl0dGxlIGJpdCBvZiBuYXN0eW5lc3MgaXMgdG8gYWxsb3cgdXMgdG8gYWRkIGEgcHJpbWFyeQotICAgICAgICAgICAgLy8gcmVmZXJlbmNlIHRvIHRoZSByZW1vdGUgcHJveHkgd2hlbiB0aGlzIHRlYW0gZG9lc24ndCBoYXZlIG9uZQotICAgICAgICAgICAgLy8gYnV0IGFub3RoZXIgdGVhbSBpcyBzZW5kaW5nIHRoZSBoYW5kbGUgdG8gdXMuCi0gICAgICAgICAgICByZXN1bHQuZm9yY2Vfc2V0KGIpOwotICAgICAgICAgICAgZS0+cmVmcy0+ZGVjV2Vhayh0aGlzKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLXdwPElCaW5kZXI+IFByb2Nlc3NTdGF0ZTo6Z2V0V2Vha1Byb3h5Rm9ySGFuZGxlKGludDMyX3QgaGFuZGxlKQotewotICAgIHdwPElCaW5kZXI+IHJlc3VsdDsKLQotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0KLSAgICBoYW5kbGVfZW50cnkqIGUgPSBsb29rdXBIYW5kbGVMb2NrZWQoaGFuZGxlKTsKLQotICAgIGlmIChlICE9IE5VTEwpIHsgICAgICAgIAotICAgICAgICAvLyBXZSBuZWVkIHRvIGNyZWF0ZSBhIG5ldyBCcEJpbmRlciBpZiB0aGVyZSBpc24ndCBjdXJyZW50bHkgb25lLCBPUiB3ZQotICAgICAgICAvLyBhcmUgdW5hYmxlIHRvIGFjcXVpcmUgYSB3ZWFrIHJlZmVyZW5jZSBvbiB0aGlzIGN1cnJlbnQgb25lLiAgVGhlCi0gICAgICAgIC8vIGF0dGVtcHRJbmNXZWFrKCkgaXMgc2FmZSBiZWNhdXNlIHdlIGtub3cgdGhlIEJwQmluZGVyIGRlc3RydWN0b3Igd2lsbCBhbHdheXMKLSAgICAgICAgLy8gY2FsbCBleHB1bmdlSGFuZGxlKCksIHdoaWNoIGFjcXVpcmVzIHRoZSBzYW1lIGxvY2sgd2UgYXJlIGhvbGRpbmcgbm93LgotICAgICAgICAvLyBXZSBuZWVkIHRvIGRvIHRoaXMgYmVjYXVzZSB0aGVyZSBpcyBhIHJhY2UgY29uZGl0aW9uIGJldHdlZW4gc29tZW9uZQotICAgICAgICAvLyByZWxlYXNpbmcgYSByZWZlcmVuY2Ugb24gdGhpcyBCcEJpbmRlciwgYW5kIGEgbmV3IHJlZmVyZW5jZSBvbiBpdHMgaGFuZGxlCi0gICAgICAgIC8vIGFycml2aW5nIGZyb20gdGhlIGRyaXZlci4KLSAgICAgICAgSUJpbmRlciogYiA9IGUtPmJpbmRlcjsKLSAgICAgICAgaWYgKGIgPT0gTlVMTCB8fCAhZS0+cmVmcy0+YXR0ZW1wdEluY1dlYWsodGhpcykpIHsKLSAgICAgICAgICAgIGIgPSBuZXcgQnBCaW5kZXIoaGFuZGxlKTsKLSAgICAgICAgICAgIHJlc3VsdCA9IGI7Ci0gICAgICAgICAgICBlLT5iaW5kZXIgPSBiOwotICAgICAgICAgICAgaWYgKGIpIGUtPnJlZnMgPSBiLT5nZXRXZWFrUmVmcygpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgcmVzdWx0ID0gYjsKLSAgICAgICAgICAgIGUtPnJlZnMtPmRlY1dlYWsodGhpcyk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi12b2lkIFByb2Nlc3NTdGF0ZTo6ZXhwdW5nZUhhbmRsZShpbnQzMl90IGhhbmRsZSwgSUJpbmRlciogYmluZGVyKQotewotICAgIEF1dG9NdXRleCBfbChtTG9jayk7Ci0gICAgCi0gICAgaGFuZGxlX2VudHJ5KiBlID0gbG9va3VwSGFuZGxlTG9ja2VkKGhhbmRsZSk7Ci0KLSAgICAvLyBUaGlzIGhhbmRsZSBtYXkgaGF2ZSBhbHJlYWR5IGJlZW4gcmVwbGFjZWQgd2l0aCBhIG5ldyBCcEJpbmRlcgotICAgIC8vIChpZiBzb21lb25lIGZhaWxlZCB0aGUgQXR0ZW1wdEluY1dlYWsoKSBhYm92ZSk7IHdlIGRvbid0IHdhbnQKLSAgICAvLyB0byBvdmVyd3JpdGUgaXQuCi0gICAgaWYgKGUgJiYgZS0+YmluZGVyID09IGJpbmRlcikgZS0+YmluZGVyID0gTlVMTDsKLX0KLQotdm9pZCBQcm9jZXNzU3RhdGU6OnNldEFyZ3MoaW50IGFyZ2MsIGNvbnN0IGNoYXIqIGNvbnN0IGFyZ3ZbXSkKLXsKLSAgICBtQXJnQyA9IGFyZ2M7Ci0gICAgbUFyZ1YgPSAoY29uc3QgY2hhciAqKilhcmd2OwotCi0gICAgbUFyZ0xlbiA9IDA7Ci0gICAgZm9yIChpbnQgaT0wOyBpPGFyZ2M7IGkrKykgewotICAgICAgICBtQXJnTGVuICs9IHN0cmxlbihhcmd2W2ldKSArIDE7Ci0gICAgfQotICAgIG1BcmdMZW4tLTsKLX0KLQotaW50IFByb2Nlc3NTdGF0ZTo6Z2V0QXJnQygpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1BcmdDOwotfQotCi1jb25zdCBjaGFyKiBjb25zdCogUHJvY2Vzc1N0YXRlOjpnZXRBcmdWKCkgY29uc3QKLXsKLSAgICByZXR1cm4gbUFyZ1Y7Ci19Ci0KLXZvaWQgUHJvY2Vzc1N0YXRlOjpzZXRBcmdWMChjb25zdCBjaGFyKiB0eHQpCi17Ci0gICAgaWYgKG1BcmdWICE9IE5VTEwpIHsKLSAgICAgICAgc3RybmNweSgoY2hhciopbUFyZ1ZbMF0sIHR4dCwgbUFyZ0xlbik7Ci0gICAgICAgIHNldF9wcm9jZXNzX25hbWUodHh0KTsKLSAgICB9Ci19Ci0KLXZvaWQgUHJvY2Vzc1N0YXRlOjpzcGF3blBvb2xlZFRocmVhZChib29sIGlzTWFpbikKLXsKLSAgICBpZiAobVRocmVhZFBvb2xTdGFydGVkKSB7Ci0gICAgICAgIGludDMyX3QgcyA9IGFuZHJvaWRfYXRvbWljX2FkZCgxLCAmbVRocmVhZFBvb2xTZXEpOwotICAgICAgICBjaGFyIGJ1ZlszMl07Ci0gICAgICAgIHNwcmludGYoYnVmLCAiQmluZGVyIFRocmVhZCAjJWQiLCBzKTsKLSAgICAgICAgTE9HVigiU3Bhd25pbmcgbmV3IHBvb2xlZCB0aHJlYWQsIG5hbWU9JXNcbiIsIGJ1Zik7Ci0gICAgICAgIHNwPFRocmVhZD4gdCA9IG5ldyBQb29sVGhyZWFkKGlzTWFpbik7Ci0gICAgICAgIHQtPnJ1bihidWYpOwotICAgIH0KLX0KLQotc3RhdGljIGludCBvcGVuX2RyaXZlcigpCi17Ci0gICAgaWYgKGdTaW5nbGVQcm9jZXNzKSB7Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0KLSAgICBpbnQgZmQgPSBvcGVuKCIvZGV2L2JpbmRlciIsIE9fUkRXUik7Ci0gICAgaWYgKGZkID49IDApIHsKLSAgICAgICAgZmNudGwoZmQsIEZfU0VURkQsIEZEX0NMT0VYRUMpOwotICAgICAgICBpbnQgdmVyczsKLSNpZiBkZWZpbmVkKEhBVkVfQU5EUk9JRF9PUykKLSAgICAgICAgc3RhdHVzX3QgcmVzdWx0ID0gaW9jdGwoZmQsIEJJTkRFUl9WRVJTSU9OLCAmdmVycyk7Ci0jZWxzZQotICAgICAgICBzdGF0dXNfdCByZXN1bHQgPSAtMTsKLSAgICAgICAgZXJybm8gPSBFUEVSTTsKLSNlbmRpZgotICAgICAgICBpZiAocmVzdWx0ID09IC0xKSB7Ci0gICAgICAgICAgICBMT0dFKCJCaW5kZXIgaW9jdGwgdG8gb2J0YWluIHZlcnNpb24gZmFpbGVkOiAlcyIsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgICAgICAgICBjbG9zZShmZCk7Ci0gICAgICAgICAgICBmZCA9IC0xOwotICAgICAgICB9Ci0gICAgICAgIGlmIChyZXN1bHQgIT0gMCB8fCB2ZXJzICE9IEJJTkRFUl9DVVJSRU5UX1BST1RPQ09MX1ZFUlNJT04pIHsKLSAgICAgICAgICAgIExPR0UoIkJpbmRlciBkcml2ZXIgcHJvdG9jb2wgZG9lcyBub3QgbWF0Y2ggdXNlciBzcGFjZSBwcm90b2NvbCEiKTsKLSAgICAgICAgICAgIGNsb3NlKGZkKTsKLSAgICAgICAgICAgIGZkID0gLTE7Ci0gICAgICAgIH0KLSNpZiBkZWZpbmVkKEhBVkVfQU5EUk9JRF9PUykKLSAgICAgICAgc2l6ZV90IG1heFRocmVhZHMgPSAxNTsKLSAgICAgICAgcmVzdWx0ID0gaW9jdGwoZmQsIEJJTkRFUl9TRVRfTUFYX1RIUkVBRFMsICZtYXhUaHJlYWRzKTsKLSAgICAgICAgaWYgKHJlc3VsdCA9PSAtMSkgewotICAgICAgICAgICAgTE9HRSgiQmluZGVyIGlvY3RsIHRvIHNldCBtYXggdGhyZWFkcyBmYWlsZWQ6ICVzIiwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgfQotI2VuZGlmCi0gICAgICAgIAotICAgIH0gZWxzZSB7Ci0gICAgICAgIExPR1coIk9wZW5pbmcgJy9kZXYvYmluZGVyJyBmYWlsZWQ6ICVzXG4iLCBzdHJlcnJvcihlcnJubykpOwotICAgIH0KLSAgICByZXR1cm4gZmQ7Ci19Ci0KLVByb2Nlc3NTdGF0ZTo6UHJvY2Vzc1N0YXRlKCkKLSAgICA6IG1Ecml2ZXJGRChvcGVuX2RyaXZlcigpKQotICAgICwgbVZNU3RhcnQoTUFQX0ZBSUxFRCkKLSAgICAsIG1NYW5hZ2VzQ29udGV4dHMoZmFsc2UpCi0gICAgLCBtQmluZGVyQ29udGV4dENoZWNrRnVuYyhOVUxMKQotICAgICwgbUJpbmRlckNvbnRleHRVc2VyRGF0YShOVUxMKQotICAgICwgbVRocmVhZFBvb2xTdGFydGVkKGZhbHNlKQotICAgICwgbVRocmVhZFBvb2xTZXEoMSkKLXsKLSAgICBpZiAobURyaXZlckZEID49IDApIHsKLSAgICAgICAgLy8gWFhYIElkZWFsbHksIHRoZXJlIHNob3VsZCBiZSBhIHNwZWNpZmljIGRlZmluZSBmb3Igd2hldGhlciB3ZQotICAgICAgICAvLyBoYXZlIG1tYXAgKG9yIHdoZXRoZXIgd2UgY291bGQgcG9zc2libHkgaGF2ZSB0aGUga2VybmVsIG1vZHVsZQotICAgICAgICAvLyBhdmFpbGFibGEpLgotI2lmICFkZWZpbmVkKEhBVkVfV0lOMzJfSVBDKQotICAgICAgICAvLyBtbWFwIHRoZSBiaW5kZXIsIHByb3ZpZGluZyBhIGNodW5rIG9mIHZpcnR1YWwgYWRkcmVzcyBzcGFjZSB0byByZWNlaXZlIHRyYW5zYWN0aW9ucy4KLSAgICAgICAgbVZNU3RhcnQgPSBtbWFwKDAsIEJJTkRFUl9WTV9TSVpFLCBQUk9UX1JFQUQsIE1BUF9QUklWQVRFIHwgTUFQX05PUkVTRVJWRSwgbURyaXZlckZELCAwKTsKLSAgICAgICAgaWYgKG1WTVN0YXJ0ID09IE1BUF9GQUlMRUQpIHsKLSAgICAgICAgICAgIC8vICpzaWdoKgotICAgICAgICAgICAgTE9HRSgiVXNpbmcgL2Rldi9iaW5kZXIgZmFpbGVkOiB1bmFibGUgdG8gbW1hcCB0cmFuc2FjdGlvbiBtZW1vcnkuXG4iKTsKLSAgICAgICAgICAgIGNsb3NlKG1Ecml2ZXJGRCk7Ci0gICAgICAgICAgICBtRHJpdmVyRkQgPSAtMTsKLSAgICAgICAgfQotI2Vsc2UKLSAgICAgICAgbURyaXZlckZEID0gLTE7Ci0jZW5kaWYKLSAgICB9Ci0gICAgaWYgKG1Ecml2ZXJGRCA8IDApIHsKLSAgICAgICAgLy8gTmVlZCB0byBydW4gd2l0aG91dCB0aGUgZHJpdmVyLCBzdGFydGluZyBvdXIgb3duIHRocmVhZCBwb29sLgotICAgIH0KLX0KLQotUHJvY2Vzc1N0YXRlOjp+UHJvY2Vzc1N0YXRlKCkKLXsKLX0KLSAgICAgICAgCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9SRUFETUUgYi9saWJzL3V0aWxzL1JFQURNRQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzZhNzA2ZC4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1JFQURNRQorKysgL2Rldi9udWxsCkBAIC0xLDE0ICswLDAgQEAKLUFuZHJvaWQgVXRpbGl0eSBGdW5jdGlvbiBMaWJyYXJ5Ci0KLUlmIHlvdSBuZWVkIGEgZmVhdHVyZSB0aGF0IGlzIG5hdGl2ZSB0byBMaW51eCBidXQgbm90IHByZXNlbnQgb24gb3RoZXIKLXBsYXRmb3JtcywgY29uc3RydWN0IGEgcGxhdGZvcm0tZGVwZW5kZW50IGltcGxlbWVudGF0aW9uIHRoYXQgc2hhcmVzCi10aGUgTGludXggaW50ZXJmYWNlLiAgVGhhdCB3YXkgdGhlIGFjdHVhbCBkZXZpY2UgcnVucyBhcyAibGlnaHQiIGFzCi1wb3NzaWJsZS4KLQotSWYgdGhhdCBpc24ndCBmZWFzaWJsZSwgY3JlYXRlIGEgc3lzdGVtLWluZGVwZW5kZW50IGludGVyZmFjZSBhbmQgaGlkZQotdGhlIGRldGFpbHMuCi0KLVRoZSB1bHRpbWF0ZSBnb2FsIGlzICpub3QqIHRvIGNyZWF0ZSBhIHN1cGVyLWR1cGVyIHBsYXRmb3JtIGFic3RyYWN0aW9uCi1sYXllci4gIFRoZSBnb2FsIGlzIHRvIHByb3ZpZGUgYW4gb3B0aW1pemVkIHNvbHV0aW9uIGZvciBMaW51eCB3aXRoCi1yZWFzb25hYmxlIGltcGxlbWVudGF0aW9ucyBmb3Igb3RoZXIgcGxhdGZvcm1zLgotCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1JlZkJhc2UuY3BwIGIvbGlicy91dGlscy9SZWZCYXNlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMGJkMWFmNC4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1JlZkJhc2UuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNTM0ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIlJlZkJhc2UiCi0KLSNpbmNsdWRlIDx1dGlscy9SZWZCYXNlLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9DYWxsU3RhY2suaD4KLSNpbmNsdWRlIDx1dGlscy9LZXllZFZlY3Rvci5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8dHlwZWluZm8+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8c3lzL3N0YXQuaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPHVuaXN0ZC5oPgotCi0vLyBjb21waWxlIHdpdGggcmVmY291bnRpbmcgZGVidWdnaW5nIGVuYWJsZWQKLSNkZWZpbmUgREVCVUdfUkVGUyAgICAgICAgICAgICAgICAgICAgICAwCi0jZGVmaW5lIERFQlVHX1JFRlNfRU5BQkxFRF9CWV9ERUZBVUxUICAgMQotI2RlZmluZSBERUJVR19SRUZTX0NBTExTVEFDS19FTkFCTEVEICAgIDEKLQotLy8gbG9nIGFsbCByZWZlcmVuY2UgY291bnRpbmcgb3BlcmF0aW9ucwotI2RlZmluZSBQUklOVF9SRUZTICAgICAgICAgICAgICAgICAgICAgIDAKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotI2RlZmluZSBJTklUSUFMX1NUUk9OR19WQUxVRSAoMTw8MjgpCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBSZWZCYXNlOjp3ZWFrcmVmX2ltcGwgOiBwdWJsaWMgUmVmQmFzZTo6d2Vha3JlZl90eXBlCi17Ci1wdWJsaWM6Ci0gICAgdm9sYXRpbGUgaW50MzJfdCAgICBtU3Ryb25nOwotICAgIHZvbGF0aWxlIGludDMyX3QgICAgbVdlYWs7Ci0gICAgUmVmQmFzZSogY29uc3QgICAgICBtQmFzZTsKLSAgICB2b2xhdGlsZSBpbnQzMl90ICAgIG1GbGFnczsKLQotCi0jaWYgIURFQlVHX1JFRlMKLQotICAgIHdlYWtyZWZfaW1wbChSZWZCYXNlKiBiYXNlKQotICAgICAgICA6IG1TdHJvbmcoSU5JVElBTF9TVFJPTkdfVkFMVUUpCi0gICAgICAgICwgbVdlYWsoMCkKLSAgICAgICAgLCBtQmFzZShiYXNlKQotICAgICAgICAsIG1GbGFncygwKQotICAgIHsKLSAgICB9Ci0KLSAgICB2b2lkIGFkZFN0cm9uZ1JlZihjb25zdCB2b2lkKiAvKmlkKi8pIHsgfQotICAgIHZvaWQgcmVtb3ZlU3Ryb25nUmVmKGNvbnN0IHZvaWQqIC8qaWQqLykgeyB9Ci0gICAgdm9pZCBhZGRXZWFrUmVmKGNvbnN0IHZvaWQqIC8qaWQqLykgeyB9Ci0gICAgdm9pZCByZW1vdmVXZWFrUmVmKGNvbnN0IHZvaWQqIC8qaWQqLykgeyB9Ci0gICAgdm9pZCBwcmludFJlZnMoKSBjb25zdCB7IH0KLSAgICB2b2lkIHRyYWNrTWUoYm9vbCwgYm9vbCkgeyB9Ci0KLSNlbHNlCi0KLSAgICB3ZWFrcmVmX2ltcGwoUmVmQmFzZSogYmFzZSkKLSAgICAgICAgOiBtU3Ryb25nKElOSVRJQUxfU1RST05HX1ZBTFVFKQotICAgICAgICAsIG1XZWFrKDApCi0gICAgICAgICwgbUJhc2UoYmFzZSkKLSAgICAgICAgLCBtRmxhZ3MoMCkKLSAgICAgICAgLCBtU3Ryb25nUmVmcyhOVUxMKQotICAgICAgICAsIG1XZWFrUmVmcyhOVUxMKQotICAgICAgICAsIG1UcmFja0VuYWJsZWQoISFERUJVR19SRUZTX0VOQUJMRURfQllfREVGQVVMVCkKLSAgICAgICAgLCBtUmV0YWluKGZhbHNlKQotICAgIHsKLSAgICAgICAgLy9MT0dJKCJORVcgd2Vha3JlZl9pbXBsICVwIGZvciBSZWZCYXNlICVwIiwgdGhpcywgYmFzZSk7Ci0gICAgfQotICAgIAotICAgIH53ZWFrcmVmX2ltcGwoKQotICAgIHsKLSAgICAgICAgTE9HX0FMV0FZU19GQVRBTF9JRighbVJldGFpbiAmJiBtU3Ryb25nUmVmcyAhPSBOVUxMLCAiU3Ryb25nIHJlZmVyZW5jZXMgcmVtYWluISIpOwotICAgICAgICBMT0dfQUxXQVlTX0ZBVEFMX0lGKCFtUmV0YWluICYmIG1XZWFrUmVmcyAhPSBOVUxMLCAiV2VhayByZWZlcmVuY2VzIHJlbWFpbiEiKTsKLSAgICB9Ci0KLSAgICB2b2lkIGFkZFN0cm9uZ1JlZihjb25zdCB2b2lkKiBpZCkKLSAgICB7Ci0gICAgICAgIGFkZFJlZigmbVN0cm9uZ1JlZnMsIGlkLCBtU3Ryb25nKTsKLSAgICB9Ci0KLSAgICB2b2lkIHJlbW92ZVN0cm9uZ1JlZihjb25zdCB2b2lkKiBpZCkKLSAgICB7Ci0gICAgICAgIGlmICghbVJldGFpbikKLSAgICAgICAgICAgIHJlbW92ZVJlZigmbVN0cm9uZ1JlZnMsIGlkKTsKLSAgICAgICAgZWxzZQotICAgICAgICAgICAgYWRkUmVmKCZtU3Ryb25nUmVmcywgaWQsIC1tU3Ryb25nKTsKLSAgICB9Ci0KLSAgICB2b2lkIGFkZFdlYWtSZWYoY29uc3Qgdm9pZCogaWQpCi0gICAgewotICAgICAgICBhZGRSZWYoJm1XZWFrUmVmcywgaWQsIG1XZWFrKTsKLSAgICB9Ci0KLSAgICB2b2lkIHJlbW92ZVdlYWtSZWYoY29uc3Qgdm9pZCogaWQpCi0gICAgewotICAgICAgICBpZiAoIW1SZXRhaW4pCi0gICAgICAgICAgICByZW1vdmVSZWYoJm1XZWFrUmVmcywgaWQpOwotICAgICAgICBlbHNlCi0gICAgICAgICAgICBhZGRSZWYoJm1XZWFrUmVmcywgaWQsIC1tV2Vhayk7Ci0gICAgfQotCi0gICAgdm9pZCB0cmFja01lKGJvb2wgdHJhY2ssIGJvb2wgcmV0YWluKQotICAgIHsgCi0gICAgICAgIG1UcmFja0VuYWJsZWQgPSB0cmFjazsKLSAgICAgICAgbVJldGFpbiA9IHJldGFpbjsKLSAgICB9Ci0KLSAgICB2b2lkIHByaW50UmVmcygpIGNvbnN0Ci0gICAgewotICAgICAgICBTdHJpbmc4IHRleHQ7Ci0KLSAgICAgICAgewotICAgICAgICAgICAgQXV0b011dGV4IF9sKGNvbnN0X2Nhc3Q8d2Vha3JlZl9pbXBsKj4odGhpcyktPm1NdXRleCk7Ci0gICAgCi0gICAgICAgICAgICBjaGFyIGJ1ZlsxMjhdOwotICAgICAgICAgICAgc3ByaW50ZihidWYsICJTdHJvbmcgcmVmZXJlbmNlcyBvbiBSZWZCYXNlICVwICh3ZWFrcmVmX3R5cGUgJXApOlxuIiwgbUJhc2UsIHRoaXMpOwotICAgICAgICAgICAgdGV4dC5hcHBlbmQoYnVmKTsKLSAgICAgICAgICAgIHByaW50UmVmc0xvY2tlZCgmdGV4dCwgbVN0cm9uZ1JlZnMpOwotICAgICAgICAgICAgc3ByaW50ZihidWYsICJXZWFrIHJlZmVyZW5jZXMgb24gUmVmQmFzZSAlcCAod2Vha3JlZl90eXBlICVwKTpcbiIsIG1CYXNlLCB0aGlzKTsKLSAgICAgICAgICAgIHRleHQuYXBwZW5kKGJ1Zik7Ci0gICAgICAgICAgICBwcmludFJlZnNMb2NrZWQoJnRleHQsIG1XZWFrUmVmcyk7Ci0gICAgICAgIH0KLQotICAgICAgICB7Ci0gICAgICAgICAgICBjaGFyIG5hbWVbMTAwXTsKLSAgICAgICAgICAgIHNucHJpbnRmKG5hbWUsIDEwMCwgIi9kYXRhLyVwLnN0YWNrIiwgdGhpcyk7Ci0gICAgICAgICAgICBpbnQgcmMgPSBvcGVuKG5hbWUsIE9fUkRXUiB8IE9fQ1JFQVQgfCBPX0FQUEVORCk7Ci0gICAgICAgICAgICBpZiAocmMgPj0gMCkgewotICAgICAgICAgICAgICAgIHdyaXRlKHJjLCB0ZXh0LnN0cmluZygpLCB0ZXh0Lmxlbmd0aCgpKTsKLSAgICAgICAgICAgICAgICBjbG9zZShyYyk7Ci0gICAgICAgICAgICAgICAgTE9HRCgiU1RBQ0sgVFJBQ0UgZm9yICVwIHNhdmVkIGluICVzIiwgdGhpcywgbmFtZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBlbHNlIExPR0UoIkZBSUxFRCBUTyBQUklOVCBTVEFDSyBUUkFDRSBmb3IgJXAgaW4gJXM6ICVzIiwgdGhpcywKLSAgICAgICAgICAgICAgICAgICAgICBuYW1lLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICB9Ci0gICAgfQotCi1wcml2YXRlOgotICAgIHN0cnVjdCByZWZfZW50cnkKLSAgICB7Ci0gICAgICAgIHJlZl9lbnRyeSogbmV4dDsKLSAgICAgICAgY29uc3Qgdm9pZCogaWQ7Ci0jaWYgREVCVUdfUkVGU19DQUxMU1RBQ0tfRU5BQkxFRAotICAgICAgICBDYWxsU3RhY2sgc3RhY2s7Ci0jZW5kaWYKLSAgICAgICAgaW50MzJfdCByZWY7Ci0gICAgfTsKLQotICAgIHZvaWQgYWRkUmVmKHJlZl9lbnRyeSoqIHJlZnMsIGNvbnN0IHZvaWQqIGlkLCBpbnQzMl90IG1SZWYpCi0gICAgewotICAgICAgICBpZiAobVRyYWNrRW5hYmxlZCkgewotICAgICAgICAgICAgQXV0b011dGV4IF9sKG1NdXRleCk7Ci0gICAgICAgICAgICByZWZfZW50cnkqIHJlZiA9IG5ldyByZWZfZW50cnk7Ci0gICAgICAgICAgICAvLyBSZWZlcmVuY2UgY291bnQgYXQgdGhlIHRpbWUgb2YgdGhlIHNuYXBzaG90LCBidXQgYmVmb3JlIHRoZQotICAgICAgICAgICAgLy8gdXBkYXRlLiAgUG9zaXRpdmUgdmFsdWUgbWVhbnMgd2UgaW5jcmVtZW50LCBuZWdhdGl2ZS0td2UKLSAgICAgICAgICAgIC8vIGRlY3JlbWVudCB0aGUgcmVmZXJlbmNlIGNvdW50LgotICAgICAgICAgICAgcmVmLT5yZWYgPSBtUmVmOwotICAgICAgICAgICAgcmVmLT5pZCA9IGlkOwotI2lmIERFQlVHX1JFRlNfQ0FMTFNUQUNLX0VOQUJMRUQKLSAgICAgICAgICAgIHJlZi0+c3RhY2sudXBkYXRlKDIpOwotI2VuZGlmCi0gICAgICAgICAgICAKLSAgICAgICAgICAgIHJlZi0+bmV4dCA9ICpyZWZzOwotICAgICAgICAgICAgKnJlZnMgPSByZWY7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICB2b2lkIHJlbW92ZVJlZihyZWZfZW50cnkqKiByZWZzLCBjb25zdCB2b2lkKiBpZCkKLSAgICB7Ci0gICAgICAgIGlmIChtVHJhY2tFbmFibGVkKSB7Ci0gICAgICAgICAgICBBdXRvTXV0ZXggX2wobU11dGV4KTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgcmVmX2VudHJ5KiByZWYgPSAqcmVmczsKLSAgICAgICAgICAgIHdoaWxlIChyZWYgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgIGlmIChyZWYtPmlkID09IGlkKSB7Ci0gICAgICAgICAgICAgICAgICAgICpyZWZzID0gcmVmLT5uZXh0OwotICAgICAgICAgICAgICAgICAgICBkZWxldGUgcmVmOwotICAgICAgICAgICAgICAgICAgICByZXR1cm47Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIHJlZnMgPSAmcmVmLT5uZXh0OwotICAgICAgICAgICAgICAgIHJlZiA9ICpyZWZzOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgCi0gICAgICAgICAgICBMT0dfQUxXQVlTX0ZBVEFMKCJSZWZCYXNlOiByZW1vdmluZyBpZCAlcCBvbiBSZWZCYXNlICVwICh3ZWFrcmVmX3R5cGUgJXApIHRoYXQgZG9lc24ndCBleGlzdCEiLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZCwgbUJhc2UsIHRoaXMpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgdm9pZCBwcmludFJlZnNMb2NrZWQoU3RyaW5nOCogb3V0LCBjb25zdCByZWZfZW50cnkqIHJlZnMpIGNvbnN0Ci0gICAgewotICAgICAgICBjaGFyIGJ1ZlsxMjhdOwotICAgICAgICB3aGlsZSAocmVmcykgewotICAgICAgICAgICAgY2hhciBpbmMgPSByZWZzLT5yZWYgPj0gMCA/ICcrJyA6ICctJzsKLSAgICAgICAgICAgIHNwcmludGYoYnVmLCAiXHQlYyBJRCAlcCAocmVmICVkKTpcbiIsIAotICAgICAgICAgICAgICAgICAgICBpbmMsIHJlZnMtPmlkLCByZWZzLT5yZWYpOwotICAgICAgICAgICAgb3V0LT5hcHBlbmQoYnVmKTsKLSNpZiBERUJVR19SRUZTX0NBTExTVEFDS19FTkFCTEVECi0gICAgICAgICAgICBvdXQtPmFwcGVuZChyZWZzLT5zdGFjay50b1N0cmluZygiXHRcdCIpKTsKLSNlbHNlCi0gICAgICAgICAgICBvdXQtPmFwcGVuZCgiXHRcdChjYWxsIHN0YWNrcyBkaXNhYmxlZCkiKTsKLSNlbmRpZgotICAgICAgICAgICAgcmVmcyA9IHJlZnMtPm5leHQ7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBNdXRleCBtTXV0ZXg7Ci0gICAgcmVmX2VudHJ5KiBtU3Ryb25nUmVmczsKLSAgICByZWZfZW50cnkqIG1XZWFrUmVmczsKLQotICAgIGJvb2wgbVRyYWNrRW5hYmxlZDsKLSAgICAvLyBDb2xsZWN0IHN0YWNrIHRyYWNlcyBvbiBhZGRyZWYgYW5kIHJlbW92ZXJlZiwgaW5zdGVhZCBvZiBkZWxldGluZyB0aGUgc3RhY2sgcmVmZXJlbmNlcwotICAgIC8vIG9uIHJlbW92ZXJlZiB0aGF0IG1hdGNoIHRoZSBhZGRyZXNzIG9uZXMuCi0gICAgYm9vbCBtUmV0YWluOwotCi0jaWYgMAotICAgIHZvaWQgYWRkUmVmKEtleWVkVmVjdG9yPGNvbnN0IHZvaWQqLCBpbnQzMl90PiogcmVmcywgY29uc3Qgdm9pZCogaWQpCi0gICAgewotICAgICAgICBBdXRvTXV0ZXggX2wobU11dGV4KTsKLSAgICAgICAgc3NpemVfdCBpID0gcmVmcy0+aW5kZXhPZktleShpZCk7Ci0gICAgICAgIGlmIChpID49IDApIHsKLSAgICAgICAgICAgICsrKHJlZnMtPmVkaXRWYWx1ZUF0KGkpKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGkgPSByZWZzLT5hZGQoaWQsIDEpOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgdm9pZCByZW1vdmVSZWYoS2V5ZWRWZWN0b3I8Y29uc3Qgdm9pZCosIGludDMyX3Q+KiByZWZzLCBjb25zdCB2b2lkKiBpZCkKLSAgICB7Ci0gICAgICAgIEF1dG9NdXRleCBfbChtTXV0ZXgpOwotICAgICAgICBzc2l6ZV90IGkgPSByZWZzLT5pbmRleE9mS2V5KGlkKTsKLSAgICAgICAgTE9HX0FMV0FZU19GQVRBTF9JRihpIDwgMCwgIlJlZkJhc2U6IHJlbW92aW5nIGlkICVwIHRoYXQgZG9lc24ndCBleGlzdCEiLCBpZCk7Ci0gICAgICAgIGlmIChpID49IDApIHsKLSAgICAgICAgICAgIGludDMyX3QgdmFsID0gLS0ocmVmcy0+ZWRpdFZhbHVlQXQoaSkpOwotICAgICAgICAgICAgaWYgKHZhbCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgcmVmcy0+cmVtb3ZlSXRlbXNBdChpKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHZvaWQgcHJpbnRSZWZzKGNvbnN0IEtleWVkVmVjdG9yPGNvbnN0IHZvaWQqLCBpbnQzMl90PiYgcmVmcykKLSAgICB7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBOPXJlZnMuc2l6ZSgpOwotICAgICAgICBmb3IgKHNpemVfdCBpPTA7IGk8TjsgaSsrKSB7Ci0gICAgICAgICAgICBwcmludGYoIlx0SUQgJXA6ICVkIHJlbWFpblxuIiwgcmVmcy5rZXlBdChpKSwgcmVmcy52YWx1ZUF0KGkpKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIG11dGFibGUgTXV0ZXggbU11dGV4OwotICAgIEtleWVkVmVjdG9yPGNvbnN0IHZvaWQqLCBpbnQzMl90PiBtU3Ryb25nUmVmczsKLSAgICBLZXllZFZlY3Rvcjxjb25zdCB2b2lkKiwgaW50MzJfdD4gbVdlYWtSZWZzOwotI2VuZGlmCi0KLSNlbmRpZgotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXZvaWQgUmVmQmFzZTo6aW5jU3Ryb25nKGNvbnN0IHZvaWQqIGlkKSBjb25zdAotewotICAgIHdlYWtyZWZfaW1wbCogY29uc3QgcmVmcyA9IG1SZWZzOwotICAgIHJlZnMtPmFkZFdlYWtSZWYoaWQpOwotICAgIHJlZnMtPmluY1dlYWsoaWQpOwotICAgIAotICAgIHJlZnMtPmFkZFN0cm9uZ1JlZihpZCk7Ci0gICAgY29uc3QgaW50MzJfdCBjID0gYW5kcm9pZF9hdG9taWNfaW5jKCZyZWZzLT5tU3Ryb25nKTsKLSAgICBMT0dfQVNTRVJUKGMgPiAwLCAiaW5jU3Ryb25nKCkgY2FsbGVkIG9uICVwIGFmdGVyIGxhc3Qgc3Ryb25nIHJlZiIsIHJlZnMpOwotI2lmIFBSSU5UX1JFRlMKLSAgICBMT0dEKCJpbmNTdHJvbmcgb2YgJXAgZnJvbSAlcDogY250PSVkXG4iLCB0aGlzLCBpZCwgYyk7Ci0jZW5kaWYKLSAgICBpZiAoYyAhPSBJTklUSUFMX1NUUk9OR19WQUxVRSkgIHsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIGFuZHJvaWRfYXRvbWljX2FkZCgtSU5JVElBTF9TVFJPTkdfVkFMVUUsICZyZWZzLT5tU3Ryb25nKTsKLSAgICBjb25zdF9jYXN0PFJlZkJhc2UqPih0aGlzKS0+b25GaXJzdFJlZigpOwotfQotCi12b2lkIFJlZkJhc2U6OmRlY1N0cm9uZyhjb25zdCB2b2lkKiBpZCkgY29uc3QKLXsKLSAgICB3ZWFrcmVmX2ltcGwqIGNvbnN0IHJlZnMgPSBtUmVmczsKLSAgICByZWZzLT5yZW1vdmVTdHJvbmdSZWYoaWQpOwotICAgIGNvbnN0IGludDMyX3QgYyA9IGFuZHJvaWRfYXRvbWljX2RlYygmcmVmcy0+bVN0cm9uZyk7Ci0jaWYgUFJJTlRfUkVGUwotICAgIExPR0QoImRlY1N0cm9uZyBvZiAlcCBmcm9tICVwOiBjbnQ9JWRcbiIsIHRoaXMsIGlkLCBjKTsKLSNlbmRpZgotICAgIExPR19BU1NFUlQoYyA+PSAxLCAiZGVjU3Ryb25nKCkgY2FsbGVkIG9uICVwIHRvbyBtYW55IHRpbWVzIiwgcmVmcyk7Ci0gICAgaWYgKGMgPT0gMSkgewotICAgICAgICBjb25zdF9jYXN0PFJlZkJhc2UqPih0aGlzKS0+b25MYXN0U3Ryb25nUmVmKGlkKTsKLSAgICAgICAgaWYgKChyZWZzLT5tRmxhZ3MmT0JKRUNUX0xJRkVUSU1FX1dFQUspICE9IE9CSkVDVF9MSUZFVElNRV9XRUFLKSB7Ci0gICAgICAgICAgICBkZWxldGUgdGhpczsKLSAgICAgICAgfQotICAgIH0KLSAgICByZWZzLT5yZW1vdmVXZWFrUmVmKGlkKTsKLSAgICByZWZzLT5kZWNXZWFrKGlkKTsKLX0KLQotdm9pZCBSZWZCYXNlOjpmb3JjZUluY1N0cm9uZyhjb25zdCB2b2lkKiBpZCkgY29uc3QKLXsKLSAgICB3ZWFrcmVmX2ltcGwqIGNvbnN0IHJlZnMgPSBtUmVmczsKLSAgICByZWZzLT5hZGRXZWFrUmVmKGlkKTsKLSAgICByZWZzLT5pbmNXZWFrKGlkKTsKLSAgICAKLSAgICByZWZzLT5hZGRTdHJvbmdSZWYoaWQpOwotICAgIGNvbnN0IGludDMyX3QgYyA9IGFuZHJvaWRfYXRvbWljX2luYygmcmVmcy0+bVN0cm9uZyk7Ci0gICAgTE9HX0FTU0VSVChjID49IDAsICJmb3JjZUluY1N0cm9uZyBjYWxsZWQgb24gJXAgYWZ0ZXIgcmVmIGNvdW50IHVuZGVyZmxvdyIsCi0gICAgICAgICAgICAgICByZWZzKTsKLSNpZiBQUklOVF9SRUZTCi0gICAgTE9HRCgiZm9yY2VJbmNTdHJvbmcgb2YgJXAgZnJvbSAlcDogY250PSVkXG4iLCB0aGlzLCBpZCwgYyk7Ci0jZW5kaWYKLQotICAgIHN3aXRjaCAoYykgewotICAgIGNhc2UgSU5JVElBTF9TVFJPTkdfVkFMVUU6Ci0gICAgICAgIGFuZHJvaWRfYXRvbWljX2FkZCgtSU5JVElBTF9TVFJPTkdfVkFMVUUsICZyZWZzLT5tU3Ryb25nKTsKLSAgICAgICAgLy8gZmFsbCB0aHJvdWdoLi4uCi0gICAgY2FzZSAwOgotICAgICAgICBjb25zdF9jYXN0PFJlZkJhc2UqPih0aGlzKS0+b25GaXJzdFJlZigpOwotICAgIH0KLX0KLQotaW50MzJfdCBSZWZCYXNlOjpnZXRTdHJvbmdDb3VudCgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1SZWZzLT5tU3Ryb25nOwotfQotCi0KLQotUmVmQmFzZSogUmVmQmFzZTo6d2Vha3JlZl90eXBlOjpyZWZCYXNlKCkgY29uc3QKLXsKLSAgICByZXR1cm4gc3RhdGljX2Nhc3Q8Y29uc3Qgd2Vha3JlZl9pbXBsKj4odGhpcyktPm1CYXNlOwotfQotCi12b2lkIFJlZkJhc2U6OndlYWtyZWZfdHlwZTo6aW5jV2Vhayhjb25zdCB2b2lkKiBpZCkKLXsKLSAgICB3ZWFrcmVmX2ltcGwqIGNvbnN0IGltcGwgPSBzdGF0aWNfY2FzdDx3ZWFrcmVmX2ltcGwqPih0aGlzKTsKLSAgICBpbXBsLT5hZGRXZWFrUmVmKGlkKTsKLSAgICBjb25zdCBpbnQzMl90IGMgPSBhbmRyb2lkX2F0b21pY19pbmMoJmltcGwtPm1XZWFrKTsKLSAgICBMT0dfQVNTRVJUKGMgPj0gMCwgImluY1dlYWsgY2FsbGVkIG9uICVwIGFmdGVyIGxhc3Qgd2VhayByZWYiLCB0aGlzKTsKLX0KLQotdm9pZCBSZWZCYXNlOjp3ZWFrcmVmX3R5cGU6OmRlY1dlYWsoY29uc3Qgdm9pZCogaWQpCi17Ci0gICAgd2Vha3JlZl9pbXBsKiBjb25zdCBpbXBsID0gc3RhdGljX2Nhc3Q8d2Vha3JlZl9pbXBsKj4odGhpcyk7Ci0gICAgaW1wbC0+cmVtb3ZlV2Vha1JlZihpZCk7Ci0gICAgY29uc3QgaW50MzJfdCBjID0gYW5kcm9pZF9hdG9taWNfZGVjKCZpbXBsLT5tV2Vhayk7Ci0gICAgTE9HX0FTU0VSVChjID49IDEsICJkZWNXZWFrIGNhbGxlZCBvbiAlcCB0b28gbWFueSB0aW1lcyIsIHRoaXMpOwotICAgIGlmIChjICE9IDEpIHJldHVybjsKLSAgICAKLSAgICBpZiAoKGltcGwtPm1GbGFncyZPQkpFQ1RfTElGRVRJTUVfV0VBSykgIT0gT0JKRUNUX0xJRkVUSU1FX1dFQUspIHsKLSAgICAgICAgaWYgKGltcGwtPm1TdHJvbmcgPT0gSU5JVElBTF9TVFJPTkdfVkFMVUUpCi0gICAgICAgICAgICBkZWxldGUgaW1wbC0+bUJhc2U7Ci0gICAgICAgIGVsc2UgewotLy8gICAgICAgICAgICBMT0dWKCJGcmVlaW5nIHJlZnMgJXAgb2Ygb2xkIFJlZkJhc2UgJXBcbiIsIHRoaXMsIGltcGwtPm1CYXNlKTsKLSAgICAgICAgICAgIGRlbGV0ZSBpbXBsOwotICAgICAgICB9Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgaW1wbC0+bUJhc2UtPm9uTGFzdFdlYWtSZWYoaWQpOwotICAgICAgICBpZiAoKGltcGwtPm1GbGFncyZPQkpFQ1RfTElGRVRJTUVfRk9SRVZFUikgIT0gT0JKRUNUX0xJRkVUSU1FX0ZPUkVWRVIpIHsKLSAgICAgICAgICAgIGRlbGV0ZSBpbXBsLT5tQmFzZTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotYm9vbCBSZWZCYXNlOjp3ZWFrcmVmX3R5cGU6OmF0dGVtcHRJbmNTdHJvbmcoY29uc3Qgdm9pZCogaWQpCi17Ci0gICAgaW5jV2VhayhpZCk7Ci0gICAgCi0gICAgd2Vha3JlZl9pbXBsKiBjb25zdCBpbXBsID0gc3RhdGljX2Nhc3Q8d2Vha3JlZl9pbXBsKj4odGhpcyk7Ci0gICAgCi0gICAgaW50MzJfdCBjdXJDb3VudCA9IGltcGwtPm1TdHJvbmc7Ci0gICAgTE9HX0FTU0VSVChjdXJDb3VudCA+PSAwLCAiYXR0ZW1wdEluY1N0cm9uZyBjYWxsZWQgb24gJXAgYWZ0ZXIgdW5kZXJmbG93IiwKLSAgICAgICAgICAgICAgIHRoaXMpOwotICAgIHdoaWxlIChjdXJDb3VudCA+IDAgJiYgY3VyQ291bnQgIT0gSU5JVElBTF9TVFJPTkdfVkFMVUUpIHsKLSAgICAgICAgaWYgKGFuZHJvaWRfYXRvbWljX2NtcHhjaGcoY3VyQ291bnQsIGN1ckNvdW50KzEsICZpbXBsLT5tU3Ryb25nKSA9PSAwKSB7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgICAgICBjdXJDb3VudCA9IGltcGwtPm1TdHJvbmc7Ci0gICAgfQotICAgIAotICAgIGlmIChjdXJDb3VudCA8PSAwIHx8IGN1ckNvdW50ID09IElOSVRJQUxfU1RST05HX1ZBTFVFKSB7Ci0gICAgICAgIGJvb2wgYWxsb3c7Ci0gICAgICAgIGlmIChjdXJDb3VudCA9PSBJTklUSUFMX1NUUk9OR19WQUxVRSkgewotICAgICAgICAgICAgLy8gQXR0ZW1wdGluZyB0byBhY3F1aXJlIGZpcnN0IHN0cm9uZyByZWZlcmVuY2UuLi4gIHRoaXMgaXMgYWxsb3dlZAotICAgICAgICAgICAgLy8gaWYgdGhlIG9iamVjdCBkb2VzIE5PVCBoYXZlIGEgbG9uZ2VyIGxpZmV0aW1lIChtZWFuaW5nIHRoZQotICAgICAgICAgICAgLy8gaW1wbGVtZW50YXRpb24gZG9lc24ndCBuZWVkIHRvIHNlZSB0aGlzKSwgb3IgaWYgdGhlIGltcGxlbWVudGF0aW9uCi0gICAgICAgICAgICAvLyBhbGxvd3MgaXQgdG8gaGFwcGVuLgotICAgICAgICAgICAgYWxsb3cgPSAoaW1wbC0+bUZsYWdzJk9CSkVDVF9MSUZFVElNRV9XRUFLKSAhPSBPQkpFQ1RfTElGRVRJTUVfV0VBSwotICAgICAgICAgICAgICAgICAgfHwgaW1wbC0+bUJhc2UtPm9uSW5jU3Ryb25nQXR0ZW1wdGVkKEZJUlNUX0lOQ19TVFJPTkcsIGlkKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIEF0dGVtcHRpbmcgdG8gcmV2aXZlIHRoZSBvYmplY3QuLi4gIHRoaXMgaXMgYWxsb3dlZAotICAgICAgICAgICAgLy8gaWYgdGhlIG9iamVjdCBET0VTIGhhdmUgYSBsb25nZXIgbGlmZXRpbWUgKHNvIHdlIGNhbiBzYWZlbHkKLSAgICAgICAgICAgIC8vIGNhbGwgdGhlIG9iamVjdCB3aXRoIG9ubHkgYSB3ZWFrIHJlZikgYW5kIHRoZSBpbXBsZW1lbnRhdGlvbgotICAgICAgICAgICAgLy8gYWxsb3dzIGl0IHRvIGhhcHBlbi4KLSAgICAgICAgICAgIGFsbG93ID0gKGltcGwtPm1GbGFncyZPQkpFQ1RfTElGRVRJTUVfV0VBSykgPT0gT0JKRUNUX0xJRkVUSU1FX1dFQUsKLSAgICAgICAgICAgICAgICAgICYmIGltcGwtPm1CYXNlLT5vbkluY1N0cm9uZ0F0dGVtcHRlZChGSVJTVF9JTkNfU1RST05HLCBpZCk7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKCFhbGxvdykgewotICAgICAgICAgICAgZGVjV2VhayhpZCk7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgY3VyQ291bnQgPSBhbmRyb2lkX2F0b21pY19pbmMoJmltcGwtPm1TdHJvbmcpOwotCi0gICAgICAgIC8vIElmIHRoZSBzdHJvbmcgcmVmZXJlbmNlIGNvdW50IGhhcyBhbHJlYWR5IGJlZW4gaW5jcmVtZW50ZWQgYnkKLSAgICAgICAgLy8gc29tZW9uZSBlbHNlLCB0aGUgaW1wbGVtZW50b3Igb2Ygb25JbmNTdHJvbmdBdHRlbXB0ZWQoKSBpcyBob2xkaW5nCi0gICAgICAgIC8vIGFuIHVubmVlZGVkIHJlZmVyZW5jZS4gIFNvIGNhbGwgb25MYXN0U3Ryb25nUmVmKCkgaGVyZSB0byByZW1vdmUgaXQuCi0gICAgICAgIC8vIChObywgdGhpcyBpcyBub3QgcHJldHR5LikgIE5vdGUgdGhhdCB3ZSBNVVNUIE5PVCBkbyB0aGlzIGlmIHdlCi0gICAgICAgIC8vIGFyZSBpbiBmYWN0IGFjcXVpcmluZyB0aGUgZmlyc3QgcmVmZXJlbmNlLgotICAgICAgICBpZiAoY3VyQ291bnQgPiAwICYmIGN1ckNvdW50IDwgSU5JVElBTF9TVFJPTkdfVkFMVUUpIHsKLSAgICAgICAgICAgIGltcGwtPm1CYXNlLT5vbkxhc3RTdHJvbmdSZWYoaWQpOwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIGltcGwtPmFkZFdlYWtSZWYoaWQpOwotICAgIGltcGwtPmFkZFN0cm9uZ1JlZihpZCk7Ci0KLSNpZiBQUklOVF9SRUZTCi0gICAgTE9HRCgiYXR0ZW1wdEluY1N0cm9uZyBvZiAlcCBmcm9tICVwOiBjbnQ9JWRcbiIsIHRoaXMsIGlkLCBjdXJDb3VudCk7Ci0jZW5kaWYKLQotICAgIGlmIChjdXJDb3VudCA9PSBJTklUSUFMX1NUUk9OR19WQUxVRSkgewotICAgICAgICBhbmRyb2lkX2F0b21pY19hZGQoLUlOSVRJQUxfU1RST05HX1ZBTFVFLCAmaW1wbC0+bVN0cm9uZyk7Ci0gICAgICAgIGltcGwtPm1CYXNlLT5vbkZpcnN0UmVmKCk7Ci0gICAgfQotICAgIAotICAgIHJldHVybiB0cnVlOwotfQotCi1ib29sIFJlZkJhc2U6OndlYWtyZWZfdHlwZTo6YXR0ZW1wdEluY1dlYWsoY29uc3Qgdm9pZCogaWQpCi17Ci0gICAgd2Vha3JlZl9pbXBsKiBjb25zdCBpbXBsID0gc3RhdGljX2Nhc3Q8d2Vha3JlZl9pbXBsKj4odGhpcyk7Ci0gICAgCi0gICAgaW50MzJfdCBjdXJDb3VudCA9IGltcGwtPm1XZWFrOwotICAgIExPR19BU1NFUlQoY3VyQ291bnQgPj0gMCwgImF0dGVtcHRJbmNXZWFrIGNhbGxlZCBvbiAlcCBhZnRlciB1bmRlcmZsb3ciLAotICAgICAgICAgICAgICAgdGhpcyk7Ci0gICAgd2hpbGUgKGN1ckNvdW50ID4gMCkgewotICAgICAgICBpZiAoYW5kcm9pZF9hdG9taWNfY21weGNoZyhjdXJDb3VudCwgY3VyQ291bnQrMSwgJmltcGwtPm1XZWFrKSA9PSAwKSB7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgICAgICBjdXJDb3VudCA9IGltcGwtPm1XZWFrOwotICAgIH0KLQotICAgIGlmIChjdXJDb3VudCA+IDApIHsKLSAgICAgICAgaW1wbC0+YWRkV2Vha1JlZihpZCk7Ci0gICAgfQotCi0gICAgcmV0dXJuIGN1ckNvdW50ID4gMDsKLX0KLQotaW50MzJfdCBSZWZCYXNlOjp3ZWFrcmVmX3R5cGU6OmdldFdlYWtDb3VudCgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0YXRpY19jYXN0PGNvbnN0IHdlYWtyZWZfaW1wbCo+KHRoaXMpLT5tV2VhazsKLX0KLQotdm9pZCBSZWZCYXNlOjp3ZWFrcmVmX3R5cGU6OnByaW50UmVmcygpIGNvbnN0Ci17Ci0gICAgc3RhdGljX2Nhc3Q8Y29uc3Qgd2Vha3JlZl9pbXBsKj4odGhpcyktPnByaW50UmVmcygpOwotfQotCi12b2lkIFJlZkJhc2U6OndlYWtyZWZfdHlwZTo6dHJhY2tNZShib29sIGVuYWJsZSwgYm9vbCByZXRhaW4pCi17Ci0gICAgc3RhdGljX2Nhc3Q8Y29uc3Qgd2Vha3JlZl9pbXBsKj4odGhpcyktPnRyYWNrTWUoZW5hYmxlLCByZXRhaW4pOwotfQotCi1SZWZCYXNlOjp3ZWFrcmVmX3R5cGUqIFJlZkJhc2U6OmNyZWF0ZVdlYWsoY29uc3Qgdm9pZCogaWQpIGNvbnN0Ci17Ci0gICAgbVJlZnMtPmluY1dlYWsoaWQpOwotICAgIHJldHVybiBtUmVmczsKLX0KLQotUmVmQmFzZTo6d2Vha3JlZl90eXBlKiBSZWZCYXNlOjpnZXRXZWFrUmVmcygpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1SZWZzOwotfQotCi1SZWZCYXNlOjpSZWZCYXNlKCkKLSAgICA6IG1SZWZzKG5ldyB3ZWFrcmVmX2ltcGwodGhpcykpCi17Ci0vLyAgICBMT0dWKCJDcmVhdGluZyByZWZzICVwIHdpdGggUmVmQmFzZSAlcFxuIiwgbVJlZnMsIHRoaXMpOwotfQotCi1SZWZCYXNlOjp+UmVmQmFzZSgpCi17Ci0vLyAgICBMT0dWKCJEZXN0cm95aW5nIFJlZkJhc2UgJXAgKHJlZnMgJXApXG4iLCB0aGlzLCBtUmVmcyk7Ci0gICAgaWYgKG1SZWZzLT5tV2VhayA9PSAwKSB7Ci0vLyAgICAgICAgTE9HVigiRnJlZWluZyByZWZzICVwIG9mIG9sZCBSZWZCYXNlICVwXG4iLCBtUmVmcywgdGhpcyk7Ci0gICAgICAgIGRlbGV0ZSBtUmVmczsKLSAgICB9Ci19Ci0KLXZvaWQgUmVmQmFzZTo6ZXh0ZW5kT2JqZWN0TGlmZXRpbWUoaW50MzJfdCBtb2RlKQotewotICAgIGFuZHJvaWRfYXRvbWljX29yKG1vZGUsICZtUmVmcy0+bUZsYWdzKTsKLX0KLQotdm9pZCBSZWZCYXNlOjpvbkZpcnN0UmVmKCkKLXsKLX0KLQotdm9pZCBSZWZCYXNlOjpvbkxhc3RTdHJvbmdSZWYoY29uc3Qgdm9pZCogLyppZCovKQotewotfQotCi1ib29sIFJlZkJhc2U6Om9uSW5jU3Ryb25nQXR0ZW1wdGVkKHVpbnQzMl90IGZsYWdzLCBjb25zdCB2b2lkKiBpZCkKLXsKLSAgICByZXR1cm4gKGZsYWdzJkZJUlNUX0lOQ19TVFJPTkcpID8gdHJ1ZSA6IGZhbHNlOwotfQotCi12b2lkIFJlZkJhc2U6Om9uTGFzdFdlYWtSZWYoY29uc3Qgdm9pZCogLyppZCovKQotewotfQotICAgICAgICAKLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1Jlc291cmNlVHlwZXMuY3BwIGIvbGlicy91dGlscy9SZXNvdXJjZVR5cGVzLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzFlN2NkNy4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1Jlc291cmNlVHlwZXMuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMzk4MyArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJSZXNvdXJjZVR5cGUiCi0vLyNkZWZpbmUgTE9HX05ERUJVRyAwCi0KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9CeXRlT3JkZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9EZWJ1Zy5oPgotI2luY2x1ZGUgPHV0aWxzL1Jlc291cmNlVHlwZXMuaD4KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmcxNi5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLSNpbmNsdWRlIDx1dGlscy9UZXh0T3V0cHV0Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxtZW1vcnkuaD4KLSNpbmNsdWRlIDxjdHlwZS5oPgotI2luY2x1ZGUgPHN0ZGludC5oPgotCi0jaWZuZGVmIElOVDMyX01BWAotI2RlZmluZSBJTlQzMl9NQVggKChpbnQzMl90KSgyMTQ3NDgzNjQ3KSkKLSNlbmRpZgotCi0jZGVmaW5lIFBPT0xfTk9JU1koeCkgLy94Ci0jZGVmaW5lIFhNTF9OT0lTWSh4KSAvL3gKLSNkZWZpbmUgVEFCTEVfTk9JU1koeCkgLy94Ci0jZGVmaW5lIFRBQkxFX0dFVEVOVFJZKHgpIC8veAotI2RlZmluZSBUQUJMRV9TVVBFUl9OT0lTWSh4KSAvL3gKLSNkZWZpbmUgTE9BRF9UQUJMRV9OT0lTWSh4KSAvL3gKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0jaWZkZWYgSEFWRV9XSU5TT0NLCi0jdW5kZWYgIG5odG9sCi0jdW5kZWYgIGh0b25sCi0KLSNpZmRlZiBIQVZFX0xJVFRMRV9FTkRJQU4KLSNkZWZpbmUgbnRvaGwoeCkgICAgKCAoKHgpIDw8IDI0KSB8ICgoKHgpID4+IDI0KSAmIDI1NSkgfCAoKCh4KSA8PCA4KSAmIDB4ZmYwMDAwKSB8ICgoKHgpID4+IDgpICYgMHhmZjAwKSApCi0jZGVmaW5lIGh0b25sKHgpICAgIG50b2hsKHgpCi0jZGVmaW5lIG50b2hzKHgpICAgICggKCgoeCkgPDwgOCkgJiAweGZmMDApIHwgKCgoeCkgPj4gOCkgJiAyNTUpICkKLSNkZWZpbmUgaHRvbnMoeCkgICAgbnRvaHMoeCkKLSNlbHNlCi0jZGVmaW5lIG50b2hsKHgpICAgICh4KQotI2RlZmluZSBodG9ubCh4KSAgICAoeCkKLSNkZWZpbmUgbnRvaHMoeCkgICAgKHgpCi0jZGVmaW5lIGh0b25zKHgpICAgICh4KQotI2VuZGlmCi0jZW5kaWYKLQotc3RhdGljIHZvaWQgcHJpbnRUb0xvZ0Z1bmModm9pZCogY29va2llLCBjb25zdCBjaGFyKiB0eHQpCi17Ci0gICAgTE9HVigiJXMiLCB0eHQpOwotfQotCi0vLyBTdGFuZGFyZCBDIGlzc3BhY2UoKSBpcyBvbmx5IHJlcXVpcmVkIHRvIGxvb2sgYXQgdGhlIGxvdyBieXRlIG9mIGl0cyBpbnB1dCwgc28KLS8vIHByb2R1Y2VzIGluY29ycmVjdCByZXN1bHRzIGZvciBVVEYtMTYgY2hhcmFjdGVycy4gIEZvciBzYWZldHkncyBzYWtlLCBhc3N1bWUgdGhhdAotLy8gYW55IGhpZ2gtYnl0ZSBVVEYtMTYgY29kZSBwb2ludCBpcyBub3Qgd2hpdGVzcGFjZS4KLWlubGluZSBpbnQgaXNzcGFjZTE2KGNoYXIxNl90IGMpIHsKLSAgICByZXR1cm4gKGMgPCAweDAwODAgJiYgaXNzcGFjZShjKSk7Ci19Ci0KLS8vIHJhbmdlIGNoZWNrZWQ7IGd1YXJhbnRlZWQgdG8gTlVMLXRlcm1pbmF0ZSB3aXRoaW4gdGhlIHN0YXRlZCBudW1iZXIgb2YgYXZhaWxhYmxlIHNsb3RzCi0vLyBOT1RFOiBpZiB0aGlzIHRydW5jYXRlcyB0aGUgZHN0IHN0cmluZyBkdWUgdG8gcnVubmluZyBvdXQgb2Ygc3BhY2UsIG5vIGF0dGVtcHQgaXMKLS8vIG1hZGUgdG8gYXZvaWQgc3BsaXR0aW5nIHN1cnJvZ2F0ZSBwYWlycy4KLXN0YXRpYyB2b2lkIHN0cmNweTE2X2R0b2godWludDE2X3QqIGRzdCwgY29uc3QgdWludDE2X3QqIHNyYywgc2l6ZV90IGF2YWlsKQotewotICAgIHVpbnQxNl90KiBsYXN0ID0gZHN0ICsgYXZhaWwgLSAxOwotICAgIHdoaWxlICgqc3JjICYmIChkc3QgPCBsYXN0KSkgewotICAgICAgICBjaGFyMTZfdCBzID0gZHRvaHMoKnNyYyk7Ci0gICAgICAgICpkc3QrKyA9IHM7Ci0gICAgICAgIHNyYysrOwotICAgIH0KLSAgICAqZHN0ID0gMDsKLX0KLQotc3RhdGljIHN0YXR1c190IHZhbGlkYXRlX2NodW5rKGNvbnN0IFJlc0NodW5rX2hlYWRlciogY2h1bmssCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IG1pblNpemUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgdWludDhfdCogZGF0YUVuZCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyKiBuYW1lKQotewotICAgIGNvbnN0IHVpbnQxNl90IGhlYWRlclNpemUgPSBkdG9ocyhjaHVuay0+aGVhZGVyU2l6ZSk7Ci0gICAgY29uc3QgdWludDMyX3Qgc2l6ZSA9IGR0b2hsKGNodW5rLT5zaXplKTsKLQotICAgIGlmIChoZWFkZXJTaXplID49IG1pblNpemUpIHsKLSAgICAgICAgaWYgKGhlYWRlclNpemUgPD0gc2l6ZSkgewotICAgICAgICAgICAgaWYgKCgoaGVhZGVyU2l6ZXxzaXplKSYweDMpID09IDApIHsKLSAgICAgICAgICAgICAgICBpZiAoKHNzaXplX3Qpc2l6ZSA8PSAoZGF0YUVuZC0oKGNvbnN0IHVpbnQ4X3QqKWNodW5rKSkpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBMT0dXKCIlcyBkYXRhIHNpemUgJXAgZXh0ZW5kcyBiZXlvbmQgcmVzb3VyY2UgZW5kICVwLiIsCi0gICAgICAgICAgICAgICAgICAgICBuYW1lLCAodm9pZCopc2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICh2b2lkKikoZGF0YUVuZC0oKGNvbnN0IHVpbnQ4X3QqKWNodW5rKSkpOwotICAgICAgICAgICAgICAgIHJldHVybiBCQURfVFlQRTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIExPR1coIiVzIHNpemUgMHgleCBvciBoZWFkZXJTaXplIDB4JXggaXMgbm90IG9uIGFuIGludGVnZXIgYm91bmRhcnkuIiwKLSAgICAgICAgICAgICAgICAgbmFtZSwgKGludClzaXplLCAoaW50KWhlYWRlclNpemUpOwotICAgICAgICAgICAgcmV0dXJuIEJBRF9UWVBFOwotICAgICAgICB9Ci0gICAgICAgIExPR1coIiVzIHNpemUgJXAgaXMgc21hbGxlciB0aGFuIGhlYWRlciBzaXplICVwLiIsCi0gICAgICAgICAgICAgbmFtZSwgKHZvaWQqKXNpemUsICh2b2lkKikoaW50KWhlYWRlclNpemUpOwotICAgICAgICByZXR1cm4gQkFEX1RZUEU7Ci0gICAgfQotICAgIExPR1coIiVzIGhlYWRlciBzaXplICVwIGlzIHRvbyBzbWFsbC4iLAotICAgICAgICAgbmFtZSwgKHZvaWQqKShpbnQpaGVhZGVyU2l6ZSk7Ci0gICAgcmV0dXJuIEJBRF9UWVBFOwotfQotCi1pbmxpbmUgdm9pZCBSZXNfdmFsdWU6OmNvcHlGcm9tX2R0b2goY29uc3QgUmVzX3ZhbHVlJiBzcmMpCi17Ci0gICAgc2l6ZSA9IGR0b2hzKHNyYy5zaXplKTsKLSAgICByZXMwID0gc3JjLnJlczA7Ci0gICAgZGF0YVR5cGUgPSBzcmMuZGF0YVR5cGU7Ci0gICAgZGF0YSA9IGR0b2hsKHNyYy5kYXRhKTsKLX0KLQotdm9pZCBSZXNfcG5nXzlwYXRjaDo6ZGV2aWNlVG9GaWxlKCkKLXsKLSAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bVhEaXZzOyBpKyspIHsKLSAgICAgICAgeERpdnNbaV0gPSBodG9ubCh4RGl2c1tpXSk7Ci0gICAgfQotICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtWURpdnM7IGkrKykgewotICAgICAgICB5RGl2c1tpXSA9IGh0b25sKHlEaXZzW2ldKTsKLSAgICB9Ci0gICAgcGFkZGluZ0xlZnQgPSBodG9ubChwYWRkaW5nTGVmdCk7Ci0gICAgcGFkZGluZ1JpZ2h0ID0gaHRvbmwocGFkZGluZ1JpZ2h0KTsKLSAgICBwYWRkaW5nVG9wID0gaHRvbmwocGFkZGluZ1RvcCk7Ci0gICAgcGFkZGluZ0JvdHRvbSA9IGh0b25sKHBhZGRpbmdCb3R0b20pOwotICAgIGZvciAoaW50IGk9MDsgaTxudW1Db2xvcnM7IGkrKykgewotICAgICAgICBjb2xvcnNbaV0gPSBodG9ubChjb2xvcnNbaV0pOwotICAgIH0KLX0KLQotdm9pZCBSZXNfcG5nXzlwYXRjaDo6ZmlsZVRvRGV2aWNlKCkKLXsKLSAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bVhEaXZzOyBpKyspIHsKLSAgICAgICAgeERpdnNbaV0gPSBudG9obCh4RGl2c1tpXSk7Ci0gICAgfQotICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtWURpdnM7IGkrKykgewotICAgICAgICB5RGl2c1tpXSA9IG50b2hsKHlEaXZzW2ldKTsKLSAgICB9Ci0gICAgcGFkZGluZ0xlZnQgPSBudG9obChwYWRkaW5nTGVmdCk7Ci0gICAgcGFkZGluZ1JpZ2h0ID0gbnRvaGwocGFkZGluZ1JpZ2h0KTsKLSAgICBwYWRkaW5nVG9wID0gbnRvaGwocGFkZGluZ1RvcCk7Ci0gICAgcGFkZGluZ0JvdHRvbSA9IG50b2hsKHBhZGRpbmdCb3R0b20pOwotICAgIGZvciAoaW50IGk9MDsgaTxudW1Db2xvcnM7IGkrKykgewotICAgICAgICBjb2xvcnNbaV0gPSBudG9obChjb2xvcnNbaV0pOwotICAgIH0KLX0KLQotc2l6ZV90IFJlc19wbmdfOXBhdGNoOjpzZXJpYWxpemVkU2l6ZSgpCi17Ci0gICAgLy8gVGhlIHNpemUgb2YgdGhpcyBzdHJ1Y3QgaXMgMzIgYnl0ZXMgb24gdGhlIDMyLWJpdCB0YXJnZXQgc3lzdGVtCi0gICAgLy8gNCAqIGludDhfdAotICAgIC8vIDQgKiBpbnQzMl90Ci0gICAgLy8gMyAqIHBvaW50ZXIKLSAgICByZXR1cm4gMzIKLSAgICAgICAgICAgICsgbnVtWERpdnMgKiBzaXplb2YoaW50MzJfdCkKLSAgICAgICAgICAgICsgbnVtWURpdnMgKiBzaXplb2YoaW50MzJfdCkKLSAgICAgICAgICAgICsgbnVtQ29sb3JzICogc2l6ZW9mKHVpbnQzMl90KTsKLX0KLQotdm9pZCogUmVzX3BuZ185cGF0Y2g6OnNlcmlhbGl6ZSgpCi17Ci0gICAgdm9pZCogbmV3RGF0YSA9IG1hbGxvYyhzZXJpYWxpemVkU2l6ZSgpKTsKLSAgICBzZXJpYWxpemUobmV3RGF0YSk7Ci0gICAgcmV0dXJuIG5ld0RhdGE7Ci19Ci0KLXZvaWQgUmVzX3BuZ185cGF0Y2g6OnNlcmlhbGl6ZSh2b2lkICogb3V0RGF0YSkKLXsKLSAgICBjaGFyKiBkYXRhID0gKGNoYXIqKSBvdXREYXRhOwotICAgIG1lbW1vdmUoZGF0YSwgJndhc0Rlc2VyaWFsaXplZCwgNCk7ICAgICAvLyBjb3B5ICB3YXNEZXNlcmlhbGl6ZWQsIG51bVhEaXZzLCBudW1ZRGl2cywgbnVtQ29sb3JzCi0gICAgbWVtbW92ZShkYXRhICsgMTIsICZwYWRkaW5nTGVmdCwgMTYpOyAgIC8vIGNvcHkgcGFkZGluZ1hYWFgKLSAgICBkYXRhICs9IDMyOwotCi0gICAgbWVtbW92ZShkYXRhLCB0aGlzLT54RGl2cywgbnVtWERpdnMgKiBzaXplb2YoaW50MzJfdCkpOwotICAgIGRhdGEgKz0gIG51bVhEaXZzICogc2l6ZW9mKGludDMyX3QpOwotICAgIG1lbW1vdmUoZGF0YSwgdGhpcy0+eURpdnMsIG51bVlEaXZzICogc2l6ZW9mKGludDMyX3QpKTsKLSAgICBkYXRhICs9ICBudW1ZRGl2cyAqIHNpemVvZihpbnQzMl90KTsKLSAgICBtZW1tb3ZlKGRhdGEsIHRoaXMtPmNvbG9ycywgbnVtQ29sb3JzICogc2l6ZW9mKHVpbnQzMl90KSk7Ci19Ci0KLXN0YXRpYyB2b2lkIGRlc2VyaWFsaXplSW50ZXJuYWwoY29uc3Qgdm9pZCogaW5EYXRhLCBSZXNfcG5nXzlwYXRjaCogb3V0RGF0YSkgewotICAgIGNoYXIqIHBhdGNoID0gKGNoYXIqKSBpbkRhdGE7Ci0gICAgaWYgKGluRGF0YSAhPSBvdXREYXRhKSB7Ci0gICAgICAgIG1lbW1vdmUoJm91dERhdGEtPndhc0Rlc2VyaWFsaXplZCwgcGF0Y2gsIDQpOyAgICAgLy8gY29weSAgd2FzRGVzZXJpYWxpemVkLCBudW1YRGl2cywgbnVtWURpdnMsIG51bUNvbG9ycwotICAgICAgICBtZW1tb3ZlKCZvdXREYXRhLT5wYWRkaW5nTGVmdCwgcGF0Y2ggKyAxMiwgNCk7ICAgICAvLyBjb3B5ICB3YXNEZXNlcmlhbGl6ZWQsIG51bVhEaXZzLCBudW1ZRGl2cywgbnVtQ29sb3JzCi0gICAgfQotICAgIG91dERhdGEtPndhc0Rlc2VyaWFsaXplZCA9IHRydWU7Ci0gICAgY2hhciogZGF0YSA9IChjaGFyKilvdXREYXRhOwotICAgIGRhdGEgKz0gIHNpemVvZihSZXNfcG5nXzlwYXRjaCk7Ci0gICAgb3V0RGF0YS0+eERpdnMgPSAoaW50MzJfdCopIGRhdGE7Ci0gICAgZGF0YSArPSAgb3V0RGF0YS0+bnVtWERpdnMgKiBzaXplb2YoaW50MzJfdCk7Ci0gICAgb3V0RGF0YS0+eURpdnMgPSAoaW50MzJfdCopIGRhdGE7Ci0gICAgZGF0YSArPSAgb3V0RGF0YS0+bnVtWURpdnMgKiBzaXplb2YoaW50MzJfdCk7Ci0gICAgb3V0RGF0YS0+Y29sb3JzID0gKHVpbnQzMl90KikgZGF0YTsKLX0KLQotUmVzX3BuZ185cGF0Y2gqIFJlc19wbmdfOXBhdGNoOjpkZXNlcmlhbGl6ZShjb25zdCB2b2lkKiBpbkRhdGEpCi17Ci0gICAgaWYgKHNpemVvZih2b2lkKikgIT0gc2l6ZW9mKGludDMyX3QpKSB7Ci0gICAgICAgIExPR0UoIkNhbm5vdCBkZXNlcmlhbGl6ZSBvbiBub24gMzItYml0IHN5c3RlbVxuIik7Ci0gICAgICAgIHJldHVybiBOVUxMOwotICAgIH0KLSAgICBkZXNlcmlhbGl6ZUludGVybmFsKGluRGF0YSwgKFJlc19wbmdfOXBhdGNoKikgaW5EYXRhKTsKLSAgICByZXR1cm4gKFJlc19wbmdfOXBhdGNoKikgaW5EYXRhOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLVJlc1N0cmluZ1Bvb2w6OlJlc1N0cmluZ1Bvb2woKQotICAgIDogbUVycm9yKE5PX0lOSVQpLCBtT3duZWREYXRhKE5VTEwpCi17Ci19Ci0KLVJlc1N0cmluZ1Bvb2w6OlJlc1N0cmluZ1Bvb2woY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsIGJvb2wgY29weURhdGEpCi0gICAgOiBtRXJyb3IoTk9fSU5JVCksIG1Pd25lZERhdGEoTlVMTCkKLXsKLSAgICBzZXRUbyhkYXRhLCBzaXplLCBjb3B5RGF0YSk7Ci19Ci0KLVJlc1N0cmluZ1Bvb2w6On5SZXNTdHJpbmdQb29sKCkKLXsKLSAgICB1bmluaXQoKTsKLX0KLQotc3RhdHVzX3QgUmVzU3RyaW5nUG9vbDo6c2V0VG8oY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsIGJvb2wgY29weURhdGEpCi17Ci0gICAgaWYgKCFkYXRhIHx8ICFzaXplKSB7Ci0gICAgICAgIHJldHVybiAobUVycm9yPUJBRF9UWVBFKTsKLSAgICB9Ci0KLSAgICB1bmluaXQoKTsKLQotICAgIGNvbnN0IGJvb2wgbm90RGV2aWNlRW5kaWFuID0gaHRvZHMoMHhmMCkgIT0gMHhmMDsKLQotICAgIGlmIChjb3B5RGF0YSB8fCBub3REZXZpY2VFbmRpYW4pIHsKLSAgICAgICAgbU93bmVkRGF0YSA9IG1hbGxvYyhzaXplKTsKLSAgICAgICAgaWYgKG1Pd25lZERhdGEgPT0gTlVMTCkgewotICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9Tk9fTUVNT1JZKTsKLSAgICAgICAgfQotICAgICAgICBtZW1jcHkobU93bmVkRGF0YSwgZGF0YSwgc2l6ZSk7Ci0gICAgICAgIGRhdGEgPSBtT3duZWREYXRhOwotICAgIH0KLQotICAgIG1IZWFkZXIgPSAoY29uc3QgUmVzU3RyaW5nUG9vbF9oZWFkZXIqKWRhdGE7Ci0KLSAgICBpZiAobm90RGV2aWNlRW5kaWFuKSB7Ci0gICAgICAgIFJlc1N0cmluZ1Bvb2xfaGVhZGVyKiBoID0gY29uc3RfY2FzdDxSZXNTdHJpbmdQb29sX2hlYWRlcio+KG1IZWFkZXIpOwotICAgICAgICBoLT5oZWFkZXIuaGVhZGVyU2l6ZSA9IGR0b2hzKG1IZWFkZXItPmhlYWRlci5oZWFkZXJTaXplKTsKLSAgICAgICAgaC0+aGVhZGVyLnR5cGUgPSBkdG9ocyhtSGVhZGVyLT5oZWFkZXIudHlwZSk7Ci0gICAgICAgIGgtPmhlYWRlci5zaXplID0gZHRvaGwobUhlYWRlci0+aGVhZGVyLnNpemUpOwotICAgICAgICBoLT5zdHJpbmdDb3VudCA9IGR0b2hsKG1IZWFkZXItPnN0cmluZ0NvdW50KTsKLSAgICAgICAgaC0+c3R5bGVDb3VudCA9IGR0b2hsKG1IZWFkZXItPnN0eWxlQ291bnQpOwotICAgICAgICBoLT5mbGFncyA9IGR0b2hsKG1IZWFkZXItPmZsYWdzKTsKLSAgICAgICAgaC0+c3RyaW5nc1N0YXJ0ID0gZHRvaGwobUhlYWRlci0+c3RyaW5nc1N0YXJ0KTsKLSAgICAgICAgaC0+c3R5bGVzU3RhcnQgPSBkdG9obChtSGVhZGVyLT5zdHlsZXNTdGFydCk7Ci0gICAgfQotCi0gICAgaWYgKG1IZWFkZXItPmhlYWRlci5oZWFkZXJTaXplID4gbUhlYWRlci0+aGVhZGVyLnNpemUKLSAgICAgICAgICAgIHx8IG1IZWFkZXItPmhlYWRlci5zaXplID4gc2l6ZSkgewotICAgICAgICBMT0dXKCJCYWQgc3RyaW5nIGJsb2NrOiBoZWFkZXIgc2l6ZSAlZCBvciB0b3RhbCBzaXplICVkIGlzIGxhcmdlciB0aGFuIGRhdGEgc2l6ZSAlZFxuIiwKLSAgICAgICAgICAgICAgICAoaW50KW1IZWFkZXItPmhlYWRlci5oZWFkZXJTaXplLCAoaW50KW1IZWFkZXItPmhlYWRlci5zaXplLCAoaW50KXNpemUpOwotICAgICAgICByZXR1cm4gKG1FcnJvcj1CQURfVFlQRSk7Ci0gICAgfQotICAgIG1TaXplID0gbUhlYWRlci0+aGVhZGVyLnNpemU7Ci0gICAgbUVudHJpZXMgPSAoY29uc3QgdWludDMyX3QqKQotICAgICAgICAoKChjb25zdCB1aW50OF90KilkYXRhKSttSGVhZGVyLT5oZWFkZXIuaGVhZGVyU2l6ZSk7Ci0KLSAgICBpZiAobUhlYWRlci0+c3RyaW5nQ291bnQgPiAwKSB7Ci0gICAgICAgIGlmICgobUhlYWRlci0+c3RyaW5nQ291bnQqc2l6ZW9mKHVpbnQzMl90KSA8IG1IZWFkZXItPnN0cmluZ0NvdW50KSAgLy8gdWludDMyIG92ZXJmbG93PwotICAgICAgICAgICAgfHwgKG1IZWFkZXItPmhlYWRlci5oZWFkZXJTaXplKyhtSGVhZGVyLT5zdHJpbmdDb3VudCpzaXplb2YodWludDMyX3QpKSkKLSAgICAgICAgICAgICAgICA+IHNpemUpIHsKLSAgICAgICAgICAgIExPR1coIkJhZCBzdHJpbmcgYmxvY2s6IGVudHJ5IG9mICVkIGl0ZW1zIGV4dGVuZHMgcGFzdCBkYXRhIHNpemUgJWRcbiIsCi0gICAgICAgICAgICAgICAgICAgIChpbnQpKG1IZWFkZXItPmhlYWRlci5oZWFkZXJTaXplKyhtSGVhZGVyLT5zdHJpbmdDb3VudCpzaXplb2YodWludDMyX3QpKSksCi0gICAgICAgICAgICAgICAgICAgIChpbnQpc2l6ZSk7Ci0gICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1CQURfVFlQRSk7Ci0gICAgICAgIH0KLSAgICAgICAgbVN0cmluZ3MgPSAoY29uc3QgY2hhcjE2X3QqKQotICAgICAgICAgICAgKCgoY29uc3QgdWludDhfdCopZGF0YSkrbUhlYWRlci0+c3RyaW5nc1N0YXJ0KTsKLSAgICAgICAgaWYgKG1IZWFkZXItPnN0cmluZ3NTdGFydCA+PSAobUhlYWRlci0+aGVhZGVyLnNpemUtc2l6ZW9mKHVpbnQxNl90KSkpIHsKLSAgICAgICAgICAgIExPR1coIkJhZCBzdHJpbmcgYmxvY2s6IHN0cmluZyBwb29sIHN0YXJ0cyBhdCAlZCwgYWZ0ZXIgdG90YWwgc2l6ZSAlZFxuIiwKLSAgICAgICAgICAgICAgICAgICAgKGludCltSGVhZGVyLT5zdHJpbmdzU3RhcnQsIChpbnQpbUhlYWRlci0+aGVhZGVyLnNpemUpOwotICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9QkFEX1RZUEUpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChtSGVhZGVyLT5zdHlsZUNvdW50ID09IDApIHsKLSAgICAgICAgICAgIG1TdHJpbmdQb29sU2l6ZSA9Ci0gICAgICAgICAgICAgICAgKG1IZWFkZXItPmhlYWRlci5zaXplLW1IZWFkZXItPnN0cmluZ3NTdGFydCkvc2l6ZW9mKHVpbnQxNl90KTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIGNoZWNrIGludmFyaWFudDogc3R5bGVzIGZvbGxvdyB0aGUgc3RyaW5ncwotICAgICAgICAgICAgaWYgKG1IZWFkZXItPnN0eWxlc1N0YXJ0IDw9IG1IZWFkZXItPnN0cmluZ3NTdGFydCkgewotICAgICAgICAgICAgICAgIExPR1coIkJhZCBzdHlsZSBibG9jazogc3R5bGUgYmxvY2sgc3RhcnRzIGF0ICVkLCBiZWZvcmUgc3RyaW5ncyBhdCAlZFxuIiwKLSAgICAgICAgICAgICAgICAgICAgKGludCltSGVhZGVyLT5zdHlsZXNTdGFydCwgKGludCltSGVhZGVyLT5zdHJpbmdzU3RhcnQpOwotICAgICAgICAgICAgICAgIHJldHVybiAobUVycm9yPUJBRF9UWVBFKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1TdHJpbmdQb29sU2l6ZSA9Ci0gICAgICAgICAgICAgICAgKG1IZWFkZXItPnN0eWxlc1N0YXJ0LW1IZWFkZXItPnN0cmluZ3NTdGFydCkvc2l6ZW9mKHVpbnQxNl90KTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIGNoZWNrIGludmFyaWFudDogc3RyaW5nQ291bnQgPiAwIHJlcXVpcmVzIGEgc3RyaW5nIHBvb2wgdG8gZXhpc3QKLSAgICAgICAgaWYgKG1TdHJpbmdQb29sU2l6ZSA9PSAwKSB7Ci0gICAgICAgICAgICBMT0dXKCJCYWQgc3RyaW5nIGJsb2NrOiBzdHJpbmdDb3VudCBpcyAlZCBidXQgcG9vbCBzaXplIGlzIDBcbiIsIChpbnQpbUhlYWRlci0+c3RyaW5nQ291bnQpOwotICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9QkFEX1RZUEUpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG5vdERldmljZUVuZGlhbikgewotICAgICAgICAgICAgc2l6ZV90IGk7Ci0gICAgICAgICAgICB1aW50MzJfdCogZSA9IGNvbnN0X2Nhc3Q8dWludDMyX3QqPihtRW50cmllcyk7Ci0gICAgICAgICAgICBmb3IgKGk9MDsgaTxtSGVhZGVyLT5zdHJpbmdDb3VudDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgZVtpXSA9IGR0b2hsKG1FbnRyaWVzW2ldKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNoYXIxNl90KiBzID0gY29uc3RfY2FzdDxjaGFyMTZfdCo+KG1TdHJpbmdzKTsKLSAgICAgICAgICAgIGZvciAoaT0wOyBpPG1TdHJpbmdQb29sU2l6ZTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgc1tpXSA9IGR0b2hzKG1TdHJpbmdzW2ldKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIGlmIChtU3RyaW5nc1ttU3RyaW5nUG9vbFNpemUtMV0gIT0gMCkgewotICAgICAgICAgICAgTE9HVygiQmFkIHN0cmluZyBibG9jazogbGFzdCBzdHJpbmcgaXMgbm90IDAtdGVybWluYXRlZFxuIik7Ci0gICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1CQURfVFlQRSk7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBtU3RyaW5ncyA9IE5VTEw7Ci0gICAgICAgIG1TdHJpbmdQb29sU2l6ZSA9IDA7Ci0gICAgfQotCi0gICAgaWYgKG1IZWFkZXItPnN0eWxlQ291bnQgPiAwKSB7Ci0gICAgICAgIG1FbnRyeVN0eWxlcyA9IG1FbnRyaWVzICsgbUhlYWRlci0+c3RyaW5nQ291bnQ7Ci0gICAgICAgIC8vIGludmFyaWFudDogaW50ZWdlciBvdmVyZmxvdyBpbiBjYWxjdWxhdGluZyBtRW50cnlTdHlsZXMKLSAgICAgICAgaWYgKG1FbnRyeVN0eWxlcyA8IG1FbnRyaWVzKSB7Ci0gICAgICAgICAgICBMT0dXKCJCYWQgc3RyaW5nIGJsb2NrOiBpbnRlZ2VyIG92ZXJmbG93IGZpbmRpbmcgc3R5bGVzXG4iKTsKLSAgICAgICAgICAgIHJldHVybiAobUVycm9yPUJBRF9UWVBFKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICgoKGNvbnN0IHVpbnQ4X3QqKW1FbnRyeVN0eWxlcy0oY29uc3QgdWludDhfdCopbUhlYWRlcikgPiAoaW50KXNpemUpIHsKLSAgICAgICAgICAgIExPR1coIkJhZCBzdHJpbmcgYmxvY2s6IGVudHJ5IG9mICVkIHN0eWxlcyBleHRlbmRzIHBhc3QgZGF0YSBzaXplICVkXG4iLAotICAgICAgICAgICAgICAgICAgICAoaW50KSgoY29uc3QgdWludDhfdCopbUVudHJ5U3R5bGVzLShjb25zdCB1aW50OF90KiltSGVhZGVyKSwKLSAgICAgICAgICAgICAgICAgICAgKGludClzaXplKTsKLSAgICAgICAgICAgIHJldHVybiAobUVycm9yPUJBRF9UWVBFKTsKLSAgICAgICAgfQotICAgICAgICBtU3R5bGVzID0gKGNvbnN0IHVpbnQzMl90KikKLSAgICAgICAgICAgICgoKGNvbnN0IHVpbnQ4X3QqKWRhdGEpK21IZWFkZXItPnN0eWxlc1N0YXJ0KTsKLSAgICAgICAgaWYgKG1IZWFkZXItPnN0eWxlc1N0YXJ0ID49IG1IZWFkZXItPmhlYWRlci5zaXplKSB7Ci0gICAgICAgICAgICBMT0dXKCJCYWQgc3RyaW5nIGJsb2NrOiBzdHlsZSBwb29sIHN0YXJ0cyAlZCwgYWZ0ZXIgdG90YWwgc2l6ZSAlZFxuIiwKLSAgICAgICAgICAgICAgICAgICAgKGludCltSGVhZGVyLT5zdHlsZXNTdGFydCwgKGludCltSGVhZGVyLT5oZWFkZXIuc2l6ZSk7Ci0gICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1CQURfVFlQRSk7Ci0gICAgICAgIH0KLSAgICAgICAgbVN0eWxlUG9vbFNpemUgPQotICAgICAgICAgICAgKG1IZWFkZXItPmhlYWRlci5zaXplLW1IZWFkZXItPnN0eWxlc1N0YXJ0KS9zaXplb2YodWludDMyX3QpOwotCi0gICAgICAgIGlmIChub3REZXZpY2VFbmRpYW4pIHsKLSAgICAgICAgICAgIHNpemVfdCBpOwotICAgICAgICAgICAgdWludDMyX3QqIGUgPSBjb25zdF9jYXN0PHVpbnQzMl90Kj4obUVudHJ5U3R5bGVzKTsKLSAgICAgICAgICAgIGZvciAoaT0wOyBpPG1IZWFkZXItPnN0eWxlQ291bnQ7IGkrKykgewotICAgICAgICAgICAgICAgIGVbaV0gPSBkdG9obChtRW50cnlTdHlsZXNbaV0pOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgdWludDMyX3QqIHMgPSBjb25zdF9jYXN0PHVpbnQzMl90Kj4obVN0eWxlcyk7Ci0gICAgICAgICAgICBmb3IgKGk9MDsgaTxtU3R5bGVQb29sU2l6ZTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgc1tpXSA9IGR0b2hsKG1TdHlsZXNbaV0pOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgY29uc3QgUmVzU3RyaW5nUG9vbF9zcGFuIGVuZFNwYW4gPSB7Ci0gICAgICAgICAgICB7IGh0b2RsKFJlc1N0cmluZ1Bvb2xfc3Bhbjo6RU5EKSB9LAotICAgICAgICAgICAgaHRvZGwoUmVzU3RyaW5nUG9vbF9zcGFuOjpFTkQpLCBodG9kbChSZXNTdHJpbmdQb29sX3NwYW46OkVORCkKLSAgICAgICAgfTsKLSAgICAgICAgaWYgKG1lbWNtcCgmbVN0eWxlc1ttU3R5bGVQb29sU2l6ZS0oc2l6ZW9mKGVuZFNwYW4pL3NpemVvZih1aW50MzJfdCkpXSwKLSAgICAgICAgICAgICAgICAgICAmZW5kU3Bhbiwgc2l6ZW9mKGVuZFNwYW4pKSAhPSAwKSB7Ci0gICAgICAgICAgICBMT0dXKCJCYWQgc3RyaW5nIGJsb2NrOiBsYXN0IHN0eWxlIGlzIG5vdCAweEZGRkZGRkZGLXRlcm1pbmF0ZWRcbiIpOwotICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9QkFEX1RZUEUpOwotICAgICAgICB9Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgbUVudHJ5U3R5bGVzID0gTlVMTDsKLSAgICAgICAgbVN0eWxlcyA9IE5VTEw7Ci0gICAgICAgIG1TdHlsZVBvb2xTaXplID0gMDsKLSAgICB9Ci0KLSAgICByZXR1cm4gKG1FcnJvcj1OT19FUlJPUik7Ci19Ci0KLXN0YXR1c190IFJlc1N0cmluZ1Bvb2w6OmdldEVycm9yKCkgY29uc3QKLXsKLSAgICByZXR1cm4gbUVycm9yOwotfQotCi12b2lkIFJlc1N0cmluZ1Bvb2w6OnVuaW5pdCgpCi17Ci0gICAgbUVycm9yID0gTk9fSU5JVDsKLSAgICBpZiAobU93bmVkRGF0YSkgewotICAgICAgICBmcmVlKG1Pd25lZERhdGEpOwotICAgICAgICBtT3duZWREYXRhID0gTlVMTDsKLSAgICB9Ci19Ci0KLWNvbnN0IHVpbnQxNl90KiBSZXNTdHJpbmdQb29sOjpzdHJpbmdBdChzaXplX3QgaWR4LCBzaXplX3QqIG91dExlbikgY29uc3QKLXsKLSAgICBpZiAobUVycm9yID09IE5PX0VSUk9SICYmIGlkeCA8IG1IZWFkZXItPnN0cmluZ0NvdW50KSB7Ci0gICAgICAgIGNvbnN0IHVpbnQzMl90IG9mZiA9IChtRW50cmllc1tpZHhdL3NpemVvZih1aW50MTZfdCkpOwotICAgICAgICBpZiAob2ZmIDwgKG1TdHJpbmdQb29sU2l6ZS0xKSkgewotICAgICAgICAgICAgY29uc3QgY2hhcjE2X3QqIHN0ciA9IG1TdHJpbmdzK29mZjsKLSAgICAgICAgICAgICpvdXRMZW4gPSAqc3RyOwotICAgICAgICAgICAgaWYgKCgqc3RyKSYweDgwMDApIHsKLSAgICAgICAgICAgICAgICBzdHIrKzsKLSAgICAgICAgICAgICAgICAqb3V0TGVuID0gKCgoKm91dExlbikmMHg3ZmZmKTw8MTYpICsgKnN0cjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICgodWludDMyX3QpKHN0cisxKypvdXRMZW4tbVN0cmluZ3MpIDwgbVN0cmluZ1Bvb2xTaXplKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHN0cisxOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBMT0dXKCJCYWQgc3RyaW5nIGJsb2NrOiBzdHJpbmcgIyVkIGV4dGVuZHMgdG8gJWQsIHBhc3QgZW5kIGF0ICVkXG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgKGludClpZHgsIChpbnQpKHN0cisxKypvdXRMZW4tbVN0cmluZ3MpLCAoaW50KW1TdHJpbmdQb29sU2l6ZSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBMT0dXKCJCYWQgc3RyaW5nIGJsb2NrOiBzdHJpbmcgIyVkIGVudHJ5IGlzIGF0ICVkLCBwYXN0IGVuZCBhdCAlZFxuIiwKLSAgICAgICAgICAgICAgICAgICAgKGludClpZHgsIChpbnQpKG9mZipzaXplb2YodWludDE2X3QpKSwKLSAgICAgICAgICAgICAgICAgICAgKGludCkobVN0cmluZ1Bvb2xTaXplKnNpemVvZih1aW50MTZfdCkpKTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gTlVMTDsKLX0KLQotY29uc3QgUmVzU3RyaW5nUG9vbF9zcGFuKiBSZXNTdHJpbmdQb29sOjpzdHlsZUF0KGNvbnN0IFJlc1N0cmluZ1Bvb2xfcmVmJiByZWYpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIHN0eWxlQXQocmVmLmluZGV4KTsKLX0KLQotY29uc3QgUmVzU3RyaW5nUG9vbF9zcGFuKiBSZXNTdHJpbmdQb29sOjpzdHlsZUF0KHNpemVfdCBpZHgpIGNvbnN0Ci17Ci0gICAgaWYgKG1FcnJvciA9PSBOT19FUlJPUiAmJiBpZHggPCBtSGVhZGVyLT5zdHlsZUNvdW50KSB7Ci0gICAgICAgIGNvbnN0IHVpbnQzMl90IG9mZiA9IChtRW50cnlTdHlsZXNbaWR4XS9zaXplb2YodWludDMyX3QpKTsKLSAgICAgICAgaWYgKG9mZiA8IG1TdHlsZVBvb2xTaXplKSB7Ci0gICAgICAgICAgICByZXR1cm4gKGNvbnN0IFJlc1N0cmluZ1Bvb2xfc3BhbiopKG1TdHlsZXMrb2ZmKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIExPR1coIkJhZCBzdHJpbmcgYmxvY2s6IHN0eWxlICMlZCBlbnRyeSBpcyBhdCAlZCwgcGFzdCBlbmQgYXQgJWRcbiIsCi0gICAgICAgICAgICAgICAgICAgIChpbnQpaWR4LCAoaW50KShvZmYqc2l6ZW9mKHVpbnQzMl90KSksCi0gICAgICAgICAgICAgICAgICAgIChpbnQpKG1TdHlsZVBvb2xTaXplKnNpemVvZih1aW50MzJfdCkpKTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gTlVMTDsKLX0KLQotc3NpemVfdCBSZXNTdHJpbmdQb29sOjppbmRleE9mU3RyaW5nKGNvbnN0IGNoYXIxNl90KiBzdHIsIHNpemVfdCBzdHJMZW4pIGNvbnN0Ci17Ci0gICAgaWYgKG1FcnJvciAhPSBOT19FUlJPUikgewotICAgICAgICByZXR1cm4gbUVycm9yOwotICAgIH0KLQotICAgIHNpemVfdCBsZW47Ci0KLSAgICBpZiAobUhlYWRlci0+ZmxhZ3MmUmVzU3RyaW5nUG9vbF9oZWFkZXI6OlNPUlRFRF9GTEFHKSB7Ci0gICAgICAgIC8vIERvIGEgYmluYXJ5IHNlYXJjaCBmb3IgdGhlIHN0cmluZy4uLgotICAgICAgICBzc2l6ZV90IGwgPSAwOwotICAgICAgICBzc2l6ZV90IGggPSBtSGVhZGVyLT5zdHJpbmdDb3VudC0xOwotCi0gICAgICAgIHNzaXplX3QgbWlkOwotICAgICAgICB3aGlsZSAobCA8PSBoKSB7Ci0gICAgICAgICAgICBtaWQgPSBsICsgKGggLSBsKS8yOwotICAgICAgICAgICAgY29uc3QgY2hhcjE2X3QqIHMgPSBzdHJpbmdBdChtaWQsICZsZW4pOwotICAgICAgICAgICAgaW50IGMgPSBzID8gc3RyemNtcDE2KHMsIGxlbiwgc3RyLCBzdHJMZW4pIDogLTE7Ci0gICAgICAgICAgICBQT09MX05PSVNZKHByaW50ZigiTG9va2luZyBmb3IgJXMsIGF0ICVzLCBjbXA9JWQsIGwvbWlkL2g9JWQvJWQvJWRcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nOChzdHIpLnN0cmluZygpLAotICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgocykuc3RyaW5nKCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgYywgKGludClsLCAoaW50KW1pZCwgKGludCloKSk7Ci0gICAgICAgICAgICBpZiAoYyA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG1pZDsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoYyA8IDApIHsKLSAgICAgICAgICAgICAgICBsID0gbWlkICsgMTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgaCA9IG1pZCAtIDE7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICAvLyBJdCBpcyB1bnVzdWFsIHRvIGdldCB0aGUgSUQgZnJvbSBhbiB1bnNvcnRlZCBzdHJpbmcgYmxvY2suLi4KLSAgICAgICAgLy8gbW9zdCBvZnRlbiB0aGlzIGhhcHBlbnMgYmVjYXVzZSB3ZSB3YW50IHRvIGdldCBJRHMgZm9yIHN0eWxlCi0gICAgICAgIC8vIHNwYW4gdGFnczsgc2luY2UgdGhvc2UgYWx3YXlzIGFwcGVhciBhdCB0aGUgZW5kIG9mIHRoZSBzdHJpbmcKLSAgICAgICAgLy8gYmxvY2ssIHN0YXJ0IHNlYXJjaGluZyBhdCB0aGUgYmFjay4KLSAgICAgICAgZm9yIChpbnQgaT1tSGVhZGVyLT5zdHJpbmdDb3VudC0xOyBpPj0wOyBpLS0pIHsKLSAgICAgICAgICAgIGNvbnN0IGNoYXIxNl90KiBzID0gc3RyaW5nQXQoaSwgJmxlbik7Ci0gICAgICAgICAgICBQT09MX05PSVNZKHByaW50ZigiTG9va2luZyBmb3IgJXMsIGF0ICVzLCBpPSVkXG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgoc3RyLCBzdHJMZW4pLnN0cmluZygpLAotICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgocykuc3RyaW5nKCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgaSkpOwotICAgICAgICAgICAgaWYgKHMgJiYgc3RyemNtcDE2KHMsIGxlbiwgc3RyLCBzdHJMZW4pID09IDApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gaTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBOQU1FX05PVF9GT1VORDsKLX0KLQotc2l6ZV90IFJlc1N0cmluZ1Bvb2w6OnNpemUoKSBjb25zdAotewotICAgIHJldHVybiAobUVycm9yID09IE5PX0VSUk9SKSA/IG1IZWFkZXItPnN0cmluZ0NvdW50IDogMDsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1SZXNYTUxQYXJzZXI6OlJlc1hNTFBhcnNlcihjb25zdCBSZXNYTUxUcmVlJiB0cmVlKQotICAgIDogbVRyZWUodHJlZSksIG1FdmVudENvZGUoQkFEX0RPQ1VNRU5UKQotewotfQotCi12b2lkIFJlc1hNTFBhcnNlcjo6cmVzdGFydCgpCi17Ci0gICAgbUN1ck5vZGUgPSBOVUxMOwotICAgIG1FdmVudENvZGUgPSBtVHJlZS5tRXJyb3IgPT0gTk9fRVJST1IgPyBTVEFSVF9ET0NVTUVOVCA6IEJBRF9ET0NVTUVOVDsKLX0KLQotUmVzWE1MUGFyc2VyOjpldmVudF9jb2RlX3QgUmVzWE1MUGFyc2VyOjpnZXRFdmVudFR5cGUoKSBjb25zdAotewotICAgIHJldHVybiBtRXZlbnRDb2RlOwotfQotCi1SZXNYTUxQYXJzZXI6OmV2ZW50X2NvZGVfdCBSZXNYTUxQYXJzZXI6Om5leHQoKQotewotICAgIGlmIChtRXZlbnRDb2RlID09IFNUQVJUX0RPQ1VNRU5UKSB7Ci0gICAgICAgIG1DdXJOb2RlID0gbVRyZWUubVJvb3ROb2RlOwotICAgICAgICBtQ3VyRXh0ID0gbVRyZWUubVJvb3RFeHQ7Ci0gICAgICAgIHJldHVybiAobUV2ZW50Q29kZT1tVHJlZS5tUm9vdENvZGUpOwotICAgIH0gZWxzZSBpZiAobUV2ZW50Q29kZSA+PSBGSVJTVF9DSFVOS19DT0RFKSB7Ci0gICAgICAgIHJldHVybiBuZXh0Tm9kZSgpOwotICAgIH0KLSAgICByZXR1cm4gbUV2ZW50Q29kZTsKLX0KLQotY29uc3QgaW50MzJfdCBSZXNYTUxQYXJzZXI6OmdldENvbW1lbnRJRCgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1DdXJOb2RlICE9IE5VTEwgPyBkdG9obChtQ3VyTm9kZS0+Y29tbWVudC5pbmRleCkgOiAtMTsKLX0KLQotY29uc3QgdWludDE2X3QqIFJlc1hNTFBhcnNlcjo6Z2V0Q29tbWVudChzaXplX3QqIG91dExlbikgY29uc3QKLXsKLSAgICBpbnQzMl90IGlkID0gZ2V0Q29tbWVudElEKCk7Ci0gICAgcmV0dXJuIGlkID49IDAgPyBtVHJlZS5tU3RyaW5ncy5zdHJpbmdBdChpZCwgb3V0TGVuKSA6IE5VTEw7Ci19Ci0KLWNvbnN0IHVpbnQzMl90IFJlc1hNTFBhcnNlcjo6Z2V0TGluZU51bWJlcigpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1DdXJOb2RlICE9IE5VTEwgPyBkdG9obChtQ3VyTm9kZS0+bGluZU51bWJlcikgOiAtMTsKLX0KLQotY29uc3QgaW50MzJfdCBSZXNYTUxQYXJzZXI6OmdldFRleHRJRCgpIGNvbnN0Ci17Ci0gICAgaWYgKG1FdmVudENvZGUgPT0gVEVYVCkgewotICAgICAgICByZXR1cm4gZHRvaGwoKChjb25zdCBSZXNYTUxUcmVlX2NkYXRhRXh0KiltQ3VyRXh0KS0+ZGF0YS5pbmRleCk7Ci0gICAgfQotICAgIHJldHVybiAtMTsKLX0KLQotY29uc3QgdWludDE2X3QqIFJlc1hNTFBhcnNlcjo6Z2V0VGV4dChzaXplX3QqIG91dExlbikgY29uc3QKLXsKLSAgICBpbnQzMl90IGlkID0gZ2V0VGV4dElEKCk7Ci0gICAgcmV0dXJuIGlkID49IDAgPyBtVHJlZS5tU3RyaW5ncy5zdHJpbmdBdChpZCwgb3V0TGVuKSA6IE5VTEw7Ci19Ci0KLXNzaXplX3QgUmVzWE1MUGFyc2VyOjpnZXRUZXh0VmFsdWUoUmVzX3ZhbHVlKiBvdXRWYWx1ZSkgY29uc3QKLXsKLSAgICBpZiAobUV2ZW50Q29kZSA9PSBURVhUKSB7Ci0gICAgICAgIG91dFZhbHVlLT5jb3B5RnJvbV9kdG9oKCgoY29uc3QgUmVzWE1MVHJlZV9jZGF0YUV4dCopbUN1ckV4dCktPnR5cGVkRGF0YSk7Ci0gICAgICAgIHJldHVybiBzaXplb2YoUmVzX3ZhbHVlKTsKLSAgICB9Ci0gICAgcmV0dXJuIEJBRF9UWVBFOwotfQotCi1jb25zdCBpbnQzMl90IFJlc1hNTFBhcnNlcjo6Z2V0TmFtZXNwYWNlUHJlZml4SUQoKSBjb25zdAotewotICAgIGlmIChtRXZlbnRDb2RlID09IFNUQVJUX05BTUVTUEFDRSB8fCBtRXZlbnRDb2RlID09IEVORF9OQU1FU1BBQ0UpIHsKLSAgICAgICAgcmV0dXJuIGR0b2hsKCgoY29uc3QgUmVzWE1MVHJlZV9uYW1lc3BhY2VFeHQqKW1DdXJFeHQpLT5wcmVmaXguaW5kZXgpOwotICAgIH0KLSAgICByZXR1cm4gLTE7Ci19Ci0KLWNvbnN0IHVpbnQxNl90KiBSZXNYTUxQYXJzZXI6OmdldE5hbWVzcGFjZVByZWZpeChzaXplX3QqIG91dExlbikgY29uc3QKLXsKLSAgICBpbnQzMl90IGlkID0gZ2V0TmFtZXNwYWNlUHJlZml4SUQoKTsKLSAgICAvL3ByaW50ZigicHJlZml4PSVkICBldmVudD0lcFxuIiwgaWQsIG1FdmVudENvZGUpOwotICAgIHJldHVybiBpZCA+PSAwID8gbVRyZWUubVN0cmluZ3Muc3RyaW5nQXQoaWQsIG91dExlbikgOiBOVUxMOwotfQotCi1jb25zdCBpbnQzMl90IFJlc1hNTFBhcnNlcjo6Z2V0TmFtZXNwYWNlVXJpSUQoKSBjb25zdAotewotICAgIGlmIChtRXZlbnRDb2RlID09IFNUQVJUX05BTUVTUEFDRSB8fCBtRXZlbnRDb2RlID09IEVORF9OQU1FU1BBQ0UpIHsKLSAgICAgICAgcmV0dXJuIGR0b2hsKCgoY29uc3QgUmVzWE1MVHJlZV9uYW1lc3BhY2VFeHQqKW1DdXJFeHQpLT51cmkuaW5kZXgpOwotICAgIH0KLSAgICByZXR1cm4gLTE7Ci19Ci0KLWNvbnN0IHVpbnQxNl90KiBSZXNYTUxQYXJzZXI6OmdldE5hbWVzcGFjZVVyaShzaXplX3QqIG91dExlbikgY29uc3QKLXsKLSAgICBpbnQzMl90IGlkID0gZ2V0TmFtZXNwYWNlVXJpSUQoKTsKLSAgICAvL3ByaW50ZigidXJpPSVkICBldmVudD0lcFxuIiwgaWQsIG1FdmVudENvZGUpOwotICAgIHJldHVybiBpZCA+PSAwID8gbVRyZWUubVN0cmluZ3Muc3RyaW5nQXQoaWQsIG91dExlbikgOiBOVUxMOwotfQotCi1jb25zdCBpbnQzMl90IFJlc1hNTFBhcnNlcjo6Z2V0RWxlbWVudE5hbWVzcGFjZUlEKCkgY29uc3QKLXsKLSAgICBpZiAobUV2ZW50Q29kZSA9PSBTVEFSVF9UQUcpIHsKLSAgICAgICAgcmV0dXJuIGR0b2hsKCgoY29uc3QgUmVzWE1MVHJlZV9hdHRyRXh0KiltQ3VyRXh0KS0+bnMuaW5kZXgpOwotICAgIH0KLSAgICBpZiAobUV2ZW50Q29kZSA9PSBFTkRfVEFHKSB7Ci0gICAgICAgIHJldHVybiBkdG9obCgoKGNvbnN0IFJlc1hNTFRyZWVfZW5kRWxlbWVudEV4dCopbUN1ckV4dCktPm5zLmluZGV4KTsKLSAgICB9Ci0gICAgcmV0dXJuIC0xOwotfQotCi1jb25zdCB1aW50MTZfdCogUmVzWE1MUGFyc2VyOjpnZXRFbGVtZW50TmFtZXNwYWNlKHNpemVfdCogb3V0TGVuKSBjb25zdAotewotICAgIGludDMyX3QgaWQgPSBnZXRFbGVtZW50TmFtZXNwYWNlSUQoKTsKLSAgICByZXR1cm4gaWQgPj0gMCA/IG1UcmVlLm1TdHJpbmdzLnN0cmluZ0F0KGlkLCBvdXRMZW4pIDogTlVMTDsKLX0KLQotY29uc3QgaW50MzJfdCBSZXNYTUxQYXJzZXI6OmdldEVsZW1lbnROYW1lSUQoKSBjb25zdAotewotICAgIGlmIChtRXZlbnRDb2RlID09IFNUQVJUX1RBRykgewotICAgICAgICByZXR1cm4gZHRvaGwoKChjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqKW1DdXJFeHQpLT5uYW1lLmluZGV4KTsKLSAgICB9Ci0gICAgaWYgKG1FdmVudENvZGUgPT0gRU5EX1RBRykgewotICAgICAgICByZXR1cm4gZHRvaGwoKChjb25zdCBSZXNYTUxUcmVlX2VuZEVsZW1lbnRFeHQqKW1DdXJFeHQpLT5uYW1lLmluZGV4KTsKLSAgICB9Ci0gICAgcmV0dXJuIC0xOwotfQotCi1jb25zdCB1aW50MTZfdCogUmVzWE1MUGFyc2VyOjpnZXRFbGVtZW50TmFtZShzaXplX3QqIG91dExlbikgY29uc3QKLXsKLSAgICBpbnQzMl90IGlkID0gZ2V0RWxlbWVudE5hbWVJRCgpOwotICAgIHJldHVybiBpZCA+PSAwID8gbVRyZWUubVN0cmluZ3Muc3RyaW5nQXQoaWQsIG91dExlbikgOiBOVUxMOwotfQotCi1zaXplX3QgUmVzWE1MUGFyc2VyOjpnZXRBdHRyaWJ1dGVDb3VudCgpIGNvbnN0Ci17Ci0gICAgaWYgKG1FdmVudENvZGUgPT0gU1RBUlRfVEFHKSB7Ci0gICAgICAgIHJldHVybiBkdG9ocygoKGNvbnN0IFJlc1hNTFRyZWVfYXR0ckV4dCopbUN1ckV4dCktPmF0dHJpYnV0ZUNvdW50KTsKLSAgICB9Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLWNvbnN0IGludDMyX3QgUmVzWE1MUGFyc2VyOjpnZXRBdHRyaWJ1dGVOYW1lc3BhY2VJRChzaXplX3QgaWR4KSBjb25zdAotewotICAgIGlmIChtRXZlbnRDb2RlID09IFNUQVJUX1RBRykgewotICAgICAgICBjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqIHRhZyA9IChjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqKW1DdXJFeHQ7Ci0gICAgICAgIGlmIChpZHggPCBkdG9ocyh0YWctPmF0dHJpYnV0ZUNvdW50KSkgewotICAgICAgICAgICAgY29uc3QgUmVzWE1MVHJlZV9hdHRyaWJ1dGUqIGF0dHIgPSAoY29uc3QgUmVzWE1MVHJlZV9hdHRyaWJ1dGUqKQotICAgICAgICAgICAgICAgICgoKGNvbnN0IHVpbnQ4X3QqKXRhZykKLSAgICAgICAgICAgICAgICAgKyBkdG9ocyh0YWctPmF0dHJpYnV0ZVN0YXJ0KQotICAgICAgICAgICAgICAgICArIChkdG9ocyh0YWctPmF0dHJpYnV0ZVNpemUpKmlkeCkpOwotICAgICAgICAgICAgcmV0dXJuIGR0b2hsKGF0dHItPm5zLmluZGV4KTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gLTI7Ci19Ci0KLWNvbnN0IHVpbnQxNl90KiBSZXNYTUxQYXJzZXI6OmdldEF0dHJpYnV0ZU5hbWVzcGFjZShzaXplX3QgaWR4LCBzaXplX3QqIG91dExlbikgY29uc3QKLXsKLSAgICBpbnQzMl90IGlkID0gZ2V0QXR0cmlidXRlTmFtZXNwYWNlSUQoaWR4KTsKLSAgICAvL3ByaW50ZigiYXR0cmlidXRlIG5hbWVzcGFjZT0lZCAgaWR4PSVkICBldmVudD0lcFxuIiwgaWQsIGlkeCwgbUV2ZW50Q29kZSk7Ci0gICAgLy9YTUxfTk9JU1kocHJpbnRmKCJnZXRBdHRyaWJ1dGVOYW1lc3BhY2UgMHgleD0weCV4XG4iLCBpZHgsIGlkKSk7Ci0gICAgcmV0dXJuIGlkID49IDAgPyBtVHJlZS5tU3RyaW5ncy5zdHJpbmdBdChpZCwgb3V0TGVuKSA6IE5VTEw7Ci19Ci0KLWNvbnN0IGludDMyX3QgUmVzWE1MUGFyc2VyOjpnZXRBdHRyaWJ1dGVOYW1lSUQoc2l6ZV90IGlkeCkgY29uc3QKLXsKLSAgICBpZiAobUV2ZW50Q29kZSA9PSBTVEFSVF9UQUcpIHsKLSAgICAgICAgY29uc3QgUmVzWE1MVHJlZV9hdHRyRXh0KiB0YWcgPSAoY29uc3QgUmVzWE1MVHJlZV9hdHRyRXh0KiltQ3VyRXh0OwotICAgICAgICBpZiAoaWR4IDwgZHRvaHModGFnLT5hdHRyaWJ1dGVDb3VudCkpIHsKLSAgICAgICAgICAgIGNvbnN0IFJlc1hNTFRyZWVfYXR0cmlidXRlKiBhdHRyID0gKGNvbnN0IFJlc1hNTFRyZWVfYXR0cmlidXRlKikKLSAgICAgICAgICAgICAgICAoKChjb25zdCB1aW50OF90Kil0YWcpCi0gICAgICAgICAgICAgICAgICsgZHRvaHModGFnLT5hdHRyaWJ1dGVTdGFydCkKLSAgICAgICAgICAgICAgICAgKyAoZHRvaHModGFnLT5hdHRyaWJ1dGVTaXplKSppZHgpKTsKLSAgICAgICAgICAgIHJldHVybiBkdG9obChhdHRyLT5uYW1lLmluZGV4KTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gLTE7Ci19Ci0KLWNvbnN0IHVpbnQxNl90KiBSZXNYTUxQYXJzZXI6OmdldEF0dHJpYnV0ZU5hbWUoc2l6ZV90IGlkeCwgc2l6ZV90KiBvdXRMZW4pIGNvbnN0Ci17Ci0gICAgaW50MzJfdCBpZCA9IGdldEF0dHJpYnV0ZU5hbWVJRChpZHgpOwotICAgIC8vcHJpbnRmKCJhdHRyaWJ1dGUgbmFtZT0lZCAgaWR4PSVkICBldmVudD0lcFxuIiwgaWQsIGlkeCwgbUV2ZW50Q29kZSk7Ci0gICAgLy9YTUxfTk9JU1kocHJpbnRmKCJnZXRBdHRyaWJ1dGVOYW1lIDB4JXg9MHgleFxuIiwgaWR4LCBpZCkpOwotICAgIHJldHVybiBpZCA+PSAwID8gbVRyZWUubVN0cmluZ3Muc3RyaW5nQXQoaWQsIG91dExlbikgOiBOVUxMOwotfQotCi1jb25zdCB1aW50MzJfdCBSZXNYTUxQYXJzZXI6OmdldEF0dHJpYnV0ZU5hbWVSZXNJRChzaXplX3QgaWR4KSBjb25zdAotewotICAgIGludDMyX3QgaWQgPSBnZXRBdHRyaWJ1dGVOYW1lSUQoaWR4KTsKLSAgICBpZiAoaWQgPj0gMCAmJiAoc2l6ZV90KWlkIDwgbVRyZWUubU51bVJlc0lkcykgewotICAgICAgICByZXR1cm4gZHRvaGwobVRyZWUubVJlc0lkc1tpZF0pOwotICAgIH0KLSAgICByZXR1cm4gMDsKLX0KLQotY29uc3QgaW50MzJfdCBSZXNYTUxQYXJzZXI6OmdldEF0dHJpYnV0ZVZhbHVlU3RyaW5nSUQoc2l6ZV90IGlkeCkgY29uc3QKLXsKLSAgICBpZiAobUV2ZW50Q29kZSA9PSBTVEFSVF9UQUcpIHsKLSAgICAgICAgY29uc3QgUmVzWE1MVHJlZV9hdHRyRXh0KiB0YWcgPSAoY29uc3QgUmVzWE1MVHJlZV9hdHRyRXh0KiltQ3VyRXh0OwotICAgICAgICBpZiAoaWR4IDwgZHRvaHModGFnLT5hdHRyaWJ1dGVDb3VudCkpIHsKLSAgICAgICAgICAgIGNvbnN0IFJlc1hNTFRyZWVfYXR0cmlidXRlKiBhdHRyID0gKGNvbnN0IFJlc1hNTFRyZWVfYXR0cmlidXRlKikKLSAgICAgICAgICAgICAgICAoKChjb25zdCB1aW50OF90Kil0YWcpCi0gICAgICAgICAgICAgICAgICsgZHRvaHModGFnLT5hdHRyaWJ1dGVTdGFydCkKLSAgICAgICAgICAgICAgICAgKyAoZHRvaHModGFnLT5hdHRyaWJ1dGVTaXplKSppZHgpKTsKLSAgICAgICAgICAgIHJldHVybiBkdG9obChhdHRyLT5yYXdWYWx1ZS5pbmRleCk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIC0xOwotfQotCi1jb25zdCB1aW50MTZfdCogUmVzWE1MUGFyc2VyOjpnZXRBdHRyaWJ1dGVTdHJpbmdWYWx1ZShzaXplX3QgaWR4LCBzaXplX3QqIG91dExlbikgY29uc3QKLXsKLSAgICBpbnQzMl90IGlkID0gZ2V0QXR0cmlidXRlVmFsdWVTdHJpbmdJRChpZHgpOwotICAgIC8vWE1MX05PSVNZKHByaW50ZigiZ2V0QXR0cmlidXRlVmFsdWUgMHgleD0weCV4XG4iLCBpZHgsIGlkKSk7Ci0gICAgcmV0dXJuIGlkID49IDAgPyBtVHJlZS5tU3RyaW5ncy5zdHJpbmdBdChpZCwgb3V0TGVuKSA6IE5VTEw7Ci19Ci0KLWludDMyX3QgUmVzWE1MUGFyc2VyOjpnZXRBdHRyaWJ1dGVEYXRhVHlwZShzaXplX3QgaWR4KSBjb25zdAotewotICAgIGlmIChtRXZlbnRDb2RlID09IFNUQVJUX1RBRykgewotICAgICAgICBjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqIHRhZyA9IChjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqKW1DdXJFeHQ7Ci0gICAgICAgIGlmIChpZHggPCBkdG9ocyh0YWctPmF0dHJpYnV0ZUNvdW50KSkgewotICAgICAgICAgICAgY29uc3QgUmVzWE1MVHJlZV9hdHRyaWJ1dGUqIGF0dHIgPSAoY29uc3QgUmVzWE1MVHJlZV9hdHRyaWJ1dGUqKQotICAgICAgICAgICAgICAgICgoKGNvbnN0IHVpbnQ4X3QqKXRhZykKLSAgICAgICAgICAgICAgICAgKyBkdG9ocyh0YWctPmF0dHJpYnV0ZVN0YXJ0KQotICAgICAgICAgICAgICAgICArIChkdG9ocyh0YWctPmF0dHJpYnV0ZVNpemUpKmlkeCkpOwotICAgICAgICAgICAgcmV0dXJuIGF0dHItPnR5cGVkVmFsdWUuZGF0YVR5cGU7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIFJlc192YWx1ZTo6VFlQRV9OVUxMOwotfQotCi1pbnQzMl90IFJlc1hNTFBhcnNlcjo6Z2V0QXR0cmlidXRlRGF0YShzaXplX3QgaWR4KSBjb25zdAotewotICAgIGlmIChtRXZlbnRDb2RlID09IFNUQVJUX1RBRykgewotICAgICAgICBjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqIHRhZyA9IChjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqKW1DdXJFeHQ7Ci0gICAgICAgIGlmIChpZHggPCBkdG9ocyh0YWctPmF0dHJpYnV0ZUNvdW50KSkgewotICAgICAgICAgICAgY29uc3QgUmVzWE1MVHJlZV9hdHRyaWJ1dGUqIGF0dHIgPSAoY29uc3QgUmVzWE1MVHJlZV9hdHRyaWJ1dGUqKQotICAgICAgICAgICAgICAgICgoKGNvbnN0IHVpbnQ4X3QqKXRhZykKLSAgICAgICAgICAgICAgICAgKyBkdG9ocyh0YWctPmF0dHJpYnV0ZVN0YXJ0KQotICAgICAgICAgICAgICAgICArIChkdG9ocyh0YWctPmF0dHJpYnV0ZVNpemUpKmlkeCkpOwotICAgICAgICAgICAgcmV0dXJuIGR0b2hsKGF0dHItPnR5cGVkVmFsdWUuZGF0YSk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXNzaXplX3QgUmVzWE1MUGFyc2VyOjpnZXRBdHRyaWJ1dGVWYWx1ZShzaXplX3QgaWR4LCBSZXNfdmFsdWUqIG91dFZhbHVlKSBjb25zdAotewotICAgIGlmIChtRXZlbnRDb2RlID09IFNUQVJUX1RBRykgewotICAgICAgICBjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqIHRhZyA9IChjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqKW1DdXJFeHQ7Ci0gICAgICAgIGlmIChpZHggPCBkdG9ocyh0YWctPmF0dHJpYnV0ZUNvdW50KSkgewotICAgICAgICAgICAgY29uc3QgUmVzWE1MVHJlZV9hdHRyaWJ1dGUqIGF0dHIgPSAoY29uc3QgUmVzWE1MVHJlZV9hdHRyaWJ1dGUqKQotICAgICAgICAgICAgICAgICgoKGNvbnN0IHVpbnQ4X3QqKXRhZykKLSAgICAgICAgICAgICAgICAgKyBkdG9ocyh0YWctPmF0dHJpYnV0ZVN0YXJ0KQotICAgICAgICAgICAgICAgICArIChkdG9ocyh0YWctPmF0dHJpYnV0ZVNpemUpKmlkeCkpOwotICAgICAgICAgICAgb3V0VmFsdWUtPmNvcHlGcm9tX2R0b2goYXR0ci0+dHlwZWRWYWx1ZSk7Ci0gICAgICAgICAgICByZXR1cm4gc2l6ZW9mKFJlc192YWx1ZSk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIEJBRF9UWVBFOwotfQotCi1zc2l6ZV90IFJlc1hNTFBhcnNlcjo6aW5kZXhPZkF0dHJpYnV0ZShjb25zdCBjaGFyKiBucywgY29uc3QgY2hhciogYXR0cikgY29uc3QKLXsKLSAgICBTdHJpbmcxNiBuc1N0cihucyAhPSBOVUxMID8gbnMgOiAiIik7Ci0gICAgU3RyaW5nMTYgYXR0clN0cihhdHRyKTsKLSAgICByZXR1cm4gaW5kZXhPZkF0dHJpYnV0ZShucyA/IG5zU3RyLnN0cmluZygpIDogTlVMTCwgbnMgPyBuc1N0ci5zaXplKCkgOiAwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGF0dHJTdHIuc3RyaW5nKCksIGF0dHJTdHIuc2l6ZSgpKTsKLX0KLQotc3NpemVfdCBSZXNYTUxQYXJzZXI6OmluZGV4T2ZBdHRyaWJ1dGUoY29uc3QgY2hhcjE2X3QqIG5zLCBzaXplX3QgbnNMZW4sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogYXR0ciwgc2l6ZV90IGF0dHJMZW4pIGNvbnN0Ci17Ci0gICAgaWYgKG1FdmVudENvZGUgPT0gU1RBUlRfVEFHKSB7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBOID0gZ2V0QXR0cmlidXRlQ291bnQoKTsKLSAgICAgICAgZm9yIChzaXplX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICAgICAgc2l6ZV90IGN1ck5zTGVuLCBjdXJBdHRyTGVuOwotICAgICAgICAgICAgY29uc3QgY2hhcjE2X3QqIGN1ck5zID0gZ2V0QXR0cmlidXRlTmFtZXNwYWNlKGksICZjdXJOc0xlbik7Ci0gICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogY3VyQXR0ciA9IGdldEF0dHJpYnV0ZU5hbWUoaSwgJmN1ckF0dHJMZW4pOwotICAgICAgICAgICAgLy9wcmludGYoIiVkOiBucz0lcCBhdHRyPSVwIGN1ck5zPSVwIGN1ckF0dHI9JXBcbiIsCi0gICAgICAgICAgICAvLyAgICAgICBpLCBucywgYXR0ciwgY3VyTnMsIGN1ckF0dHIpOwotICAgICAgICAgICAgLy9wcmludGYoIiAtLT4gYXR0cj0lcywgY3VyQXR0cj0lc1xuIiwKLSAgICAgICAgICAgIC8vICAgICAgIFN0cmluZzgoYXR0cikuc3RyaW5nKCksIFN0cmluZzgoY3VyQXR0cikuc3RyaW5nKCkpOwotICAgICAgICAgICAgaWYgKGF0dHIgJiYgY3VyQXR0ciAmJiAoc3RyemNtcDE2KGF0dHIsIGF0dHJMZW4sIGN1ckF0dHIsIGN1ckF0dHJMZW4pID09IDApKSB7Ci0gICAgICAgICAgICAgICAgaWYgKG5zID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGN1ck5zID09IE5VTEwpIHJldHVybiBpOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoY3VyTnMgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICAvL3ByaW50ZigiIC0tPiBucz0lcywgY3VyTnM9JXNcbiIsCi0gICAgICAgICAgICAgICAgICAgIC8vICAgICAgIFN0cmluZzgobnMpLnN0cmluZygpLCBTdHJpbmc4KGN1ck5zKS5zdHJpbmcoKSk7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChzdHJ6Y21wMTYobnMsIG5zTGVuLCBjdXJOcywgY3VyTnNMZW4pID09IDApIHJldHVybiBpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBOQU1FX05PVF9GT1VORDsKLX0KLQotc3NpemVfdCBSZXNYTUxQYXJzZXI6OmluZGV4T2ZJRCgpIGNvbnN0Ci17Ci0gICAgaWYgKG1FdmVudENvZGUgPT0gU1RBUlRfVEFHKSB7Ci0gICAgICAgIGNvbnN0IHNzaXplX3QgaWR4ID0gZHRvaHMoKChjb25zdCBSZXNYTUxUcmVlX2F0dHJFeHQqKW1DdXJFeHQpLT5pZEluZGV4KTsKLSAgICAgICAgaWYgKGlkeCA+IDApIHJldHVybiAoaWR4LTEpOwotICAgIH0KLSAgICByZXR1cm4gTkFNRV9OT1RfRk9VTkQ7Ci19Ci0KLXNzaXplX3QgUmVzWE1MUGFyc2VyOjppbmRleE9mQ2xhc3MoKSBjb25zdAotewotICAgIGlmIChtRXZlbnRDb2RlID09IFNUQVJUX1RBRykgewotICAgICAgICBjb25zdCBzc2l6ZV90IGlkeCA9IGR0b2hzKCgoY29uc3QgUmVzWE1MVHJlZV9hdHRyRXh0KiltQ3VyRXh0KS0+Y2xhc3NJbmRleCk7Ci0gICAgICAgIGlmIChpZHggPiAwKSByZXR1cm4gKGlkeC0xKTsKLSAgICB9Ci0gICAgcmV0dXJuIE5BTUVfTk9UX0ZPVU5EOwotfQotCi1zc2l6ZV90IFJlc1hNTFBhcnNlcjo6aW5kZXhPZlN0eWxlKCkgY29uc3QKLXsKLSAgICBpZiAobUV2ZW50Q29kZSA9PSBTVEFSVF9UQUcpIHsKLSAgICAgICAgY29uc3Qgc3NpemVfdCBpZHggPSBkdG9ocygoKGNvbnN0IFJlc1hNTFRyZWVfYXR0ckV4dCopbUN1ckV4dCktPnN0eWxlSW5kZXgpOwotICAgICAgICBpZiAoaWR4ID4gMCkgcmV0dXJuIChpZHgtMSk7Ci0gICAgfQotICAgIHJldHVybiBOQU1FX05PVF9GT1VORDsKLX0KLQotUmVzWE1MUGFyc2VyOjpldmVudF9jb2RlX3QgUmVzWE1MUGFyc2VyOjpuZXh0Tm9kZSgpCi17Ci0gICAgaWYgKG1FdmVudENvZGUgPCAwKSB7Ci0gICAgICAgIHJldHVybiBtRXZlbnRDb2RlOwotICAgIH0KLQotICAgIGRvIHsKLSAgICAgICAgY29uc3QgUmVzWE1MVHJlZV9ub2RlKiBuZXh0ID0gKGNvbnN0IFJlc1hNTFRyZWVfbm9kZSopCi0gICAgICAgICAgICAoKChjb25zdCB1aW50OF90KiltQ3VyTm9kZSkgKyBkdG9obChtQ3VyTm9kZS0+aGVhZGVyLnNpemUpKTsKLSAgICAgICAgLy9MT0dXKCJOZXh0IG5vZGU6IHByZXY9JXAsIG5leHQ9JXBcbiIsIG1DdXJOb2RlLCBuZXh0KTsKLSAgICAgICAgCi0gICAgICAgIGlmICgoKGNvbnN0IHVpbnQ4X3QqKW5leHQpID49IG1UcmVlLm1EYXRhRW5kKSB7Ci0gICAgICAgICAgICBtQ3VyTm9kZSA9IE5VTEw7Ci0gICAgICAgICAgICByZXR1cm4gKG1FdmVudENvZGU9RU5EX0RPQ1VNRU5UKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChtVHJlZS52YWxpZGF0ZU5vZGUobmV4dCkgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgIG1DdXJOb2RlID0gTlVMTDsKLSAgICAgICAgICAgIHJldHVybiAobUV2ZW50Q29kZT1CQURfRE9DVU1FTlQpOwotICAgICAgICB9Ci0KLSAgICAgICAgbUN1ck5vZGUgPSBuZXh0OwotICAgICAgICBjb25zdCB1aW50MTZfdCBoZWFkZXJTaXplID0gZHRvaHMobmV4dC0+aGVhZGVyLmhlYWRlclNpemUpOwotICAgICAgICBjb25zdCB1aW50MzJfdCB0b3RhbFNpemUgPSBkdG9obChuZXh0LT5oZWFkZXIuc2l6ZSk7Ci0gICAgICAgIG1DdXJFeHQgPSAoKGNvbnN0IHVpbnQ4X3QqKW5leHQpICsgaGVhZGVyU2l6ZTsKLSAgICAgICAgc2l6ZV90IG1pbkV4dFNpemUgPSAwOwotICAgICAgICBldmVudF9jb2RlX3QgZXZlbnRDb2RlID0gKGV2ZW50X2NvZGVfdClkdG9ocyhuZXh0LT5oZWFkZXIudHlwZSk7Ci0gICAgICAgIHN3aXRjaCAoKG1FdmVudENvZGU9ZXZlbnRDb2RlKSkgewotICAgICAgICAgICAgY2FzZSBSRVNfWE1MX1NUQVJUX05BTUVTUEFDRV9UWVBFOgotICAgICAgICAgICAgY2FzZSBSRVNfWE1MX0VORF9OQU1FU1BBQ0VfVFlQRToKLSAgICAgICAgICAgICAgICBtaW5FeHRTaXplID0gc2l6ZW9mKFJlc1hNTFRyZWVfbmFtZXNwYWNlRXh0KTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGNhc2UgUkVTX1hNTF9TVEFSVF9FTEVNRU5UX1RZUEU6Ci0gICAgICAgICAgICAgICAgbWluRXh0U2l6ZSA9IHNpemVvZihSZXNYTUxUcmVlX2F0dHJFeHQpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBSRVNfWE1MX0VORF9FTEVNRU5UX1RZUEU6Ci0gICAgICAgICAgICAgICAgbWluRXh0U2l6ZSA9IHNpemVvZihSZXNYTUxUcmVlX2VuZEVsZW1lbnRFeHQpOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgY2FzZSBSRVNfWE1MX0NEQVRBX1RZUEU6Ci0gICAgICAgICAgICAgICAgbWluRXh0U2l6ZSA9IHNpemVvZihSZXNYTUxUcmVlX2NkYXRhRXh0KTsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgTE9HVygiVW5rbm93biBYTUwgYmxvY2s6IGhlYWRlciB0eXBlICVkIGluIG5vZGUgYXQgJWRcbiIsCi0gICAgICAgICAgICAgICAgICAgICAoaW50KWR0b2hzKG5leHQtPmhlYWRlci50eXBlKSwKLSAgICAgICAgICAgICAgICAgICAgIChpbnQpKCgoY29uc3QgdWludDhfdCopbmV4dCktKChjb25zdCB1aW50OF90KiltVHJlZS5tSGVhZGVyKSkpOwotICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBpZiAoKHRvdGFsU2l6ZS1oZWFkZXJTaXplKSA8IG1pbkV4dFNpemUpIHsKLSAgICAgICAgICAgIExPR1coIkJhZCBYTUwgYmxvY2s6IGhlYWRlciB0eXBlIDB4JXggaW4gbm9kZSBhdCAweCV4IGhhcyBzaXplICVkLCBuZWVkICVkXG4iLAotICAgICAgICAgICAgICAgICAoaW50KWR0b2hzKG5leHQtPmhlYWRlci50eXBlKSwKLSAgICAgICAgICAgICAgICAgKGludCkoKChjb25zdCB1aW50OF90KiluZXh0KS0oKGNvbnN0IHVpbnQ4X3QqKW1UcmVlLm1IZWFkZXIpKSwKLSAgICAgICAgICAgICAgICAgKGludCkodG90YWxTaXplLWhlYWRlclNpemUpLCAoaW50KW1pbkV4dFNpemUpOwotICAgICAgICAgICAgcmV0dXJuIChtRXZlbnRDb2RlPUJBRF9ET0NVTUVOVCk7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIC8vcHJpbnRmKCJDdXJOb2RlPSVwLCBDdXJFeHQ9JXAsIGhlYWRlclNpemU9JWQsIG1pbkV4dFNpemU9JWRcbiIsCi0gICAgICAgIC8vICAgICAgIG1DdXJOb2RlLCBtQ3VyRXh0LCBoZWFkZXJTaXplLCBtaW5FeHRTaXplKTsKLSAgICAgICAgCi0gICAgICAgIHJldHVybiBldmVudENvZGU7Ci0gICAgfSB3aGlsZSAodHJ1ZSk7Ci19Ci0KLXZvaWQgUmVzWE1MUGFyc2VyOjpnZXRQb3NpdGlvbihSZXNYTUxQYXJzZXI6OlJlc1hNTFBvc2l0aW9uKiBwb3MpIGNvbnN0Ci17Ci0gICAgcG9zLT5ldmVudENvZGUgPSBtRXZlbnRDb2RlOwotICAgIHBvcy0+Y3VyTm9kZSA9IG1DdXJOb2RlOwotICAgIHBvcy0+Y3VyRXh0ID0gbUN1ckV4dDsKLX0KLQotdm9pZCBSZXNYTUxQYXJzZXI6OnNldFBvc2l0aW9uKGNvbnN0IFJlc1hNTFBhcnNlcjo6UmVzWE1MUG9zaXRpb24mIHBvcykKLXsKLSAgICBtRXZlbnRDb2RlID0gcG9zLmV2ZW50Q29kZTsKLSAgICBtQ3VyTm9kZSA9IHBvcy5jdXJOb2RlOwotICAgIG1DdXJFeHQgPSBwb3MuY3VyRXh0OwotfQotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyB2b2xhdGlsZSBpbnQzMl90IGdDb3VudCA9IDA7Ci0KLVJlc1hNTFRyZWU6OlJlc1hNTFRyZWUoKQotICAgIDogUmVzWE1MUGFyc2VyKCp0aGlzKQotICAgICwgbUVycm9yKE5PX0lOSVQpLCBtT3duZWREYXRhKE5VTEwpCi17Ci0gICAgLy9MT0dJKCJDcmVhdGluZyBSZXNYTUxUcmVlICVwICMlZFxuIiwgdGhpcywgYW5kcm9pZF9hdG9taWNfaW5jKCZnQ291bnQpKzEpOwotICAgIHJlc3RhcnQoKTsKLX0KLQotUmVzWE1MVHJlZTo6UmVzWE1MVHJlZShjb25zdCB2b2lkKiBkYXRhLCBzaXplX3Qgc2l6ZSwgYm9vbCBjb3B5RGF0YSkKLSAgICA6IFJlc1hNTFBhcnNlcigqdGhpcykKLSAgICAsIG1FcnJvcihOT19JTklUKSwgbU93bmVkRGF0YShOVUxMKQotewotICAgIC8vTE9HSSgiQ3JlYXRpbmcgUmVzWE1MVHJlZSAlcCAjJWRcbiIsIHRoaXMsIGFuZHJvaWRfYXRvbWljX2luYygmZ0NvdW50KSsxKTsKLSAgICBzZXRUbyhkYXRhLCBzaXplLCBjb3B5RGF0YSk7Ci19Ci0KLVJlc1hNTFRyZWU6On5SZXNYTUxUcmVlKCkKLXsKLSAgICAvL0xPR0koIkRlc3Ryb3lpbmcgUmVzWE1MVHJlZSBpbiAlcCAjJWRcbiIsIHRoaXMsIGFuZHJvaWRfYXRvbWljX2RlYygmZ0NvdW50KS0xKTsKLSAgICB1bmluaXQoKTsKLX0KLQotc3RhdHVzX3QgUmVzWE1MVHJlZTo6c2V0VG8oY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsIGJvb2wgY29weURhdGEpCi17Ci0gICAgdW5pbml0KCk7Ci0gICAgbUV2ZW50Q29kZSA9IFNUQVJUX0RPQ1VNRU5UOwotCi0gICAgaWYgKGNvcHlEYXRhKSB7Ci0gICAgICAgIG1Pd25lZERhdGEgPSBtYWxsb2Moc2l6ZSk7Ci0gICAgICAgIGlmIChtT3duZWREYXRhID09IE5VTEwpIHsKLSAgICAgICAgICAgIHJldHVybiAobUVycm9yPU5PX01FTU9SWSk7Ci0gICAgICAgIH0KLSAgICAgICAgbWVtY3B5KG1Pd25lZERhdGEsIGRhdGEsIHNpemUpOwotICAgICAgICBkYXRhID0gbU93bmVkRGF0YTsKLSAgICB9Ci0KLSAgICBtSGVhZGVyID0gKGNvbnN0IFJlc1hNTFRyZWVfaGVhZGVyKilkYXRhOwotICAgIG1TaXplID0gZHRvaGwobUhlYWRlci0+aGVhZGVyLnNpemUpOwotICAgIGlmIChkdG9ocyhtSGVhZGVyLT5oZWFkZXIuaGVhZGVyU2l6ZSkgPiBtU2l6ZSB8fCBtU2l6ZSA+IHNpemUpIHsKLSAgICAgICAgTE9HVygiQmFkIFhNTCBibG9jazogaGVhZGVyIHNpemUgJWQgb3IgdG90YWwgc2l6ZSAlZCBpcyBsYXJnZXIgdGhhbiBkYXRhIHNpemUgJWRcbiIsCi0gICAgICAgICAgICAgKGludClkdG9ocyhtSGVhZGVyLT5oZWFkZXIuaGVhZGVyU2l6ZSksCi0gICAgICAgICAgICAgKGludClkdG9obChtSGVhZGVyLT5oZWFkZXIuc2l6ZSksIChpbnQpc2l6ZSk7Ci0gICAgICAgIG1FcnJvciA9IEJBRF9UWVBFOwotICAgICAgICByZXN0YXJ0KCk7Ci0gICAgICAgIHJldHVybiBtRXJyb3I7Ci0gICAgfQotICAgIG1EYXRhRW5kID0gKChjb25zdCB1aW50OF90KiltSGVhZGVyKSArIG1TaXplOwotCi0gICAgbVN0cmluZ3MudW5pbml0KCk7Ci0gICAgbVJvb3ROb2RlID0gTlVMTDsKLSAgICBtUmVzSWRzID0gTlVMTDsKLSAgICBtTnVtUmVzSWRzID0gMDsKLQotICAgIC8vIEZpcnN0IGxvb2sgZm9yIGEgY291cGxlIGludGVyZXN0aW5nIGNodW5rczogdGhlIHN0cmluZyBibG9jawotICAgIC8vIGFuZCBmaXJzdCBYTUwgbm9kZS4KLSAgICBjb25zdCBSZXNDaHVua19oZWFkZXIqIGNodW5rID0KLSAgICAgICAgKGNvbnN0IFJlc0NodW5rX2hlYWRlciopKCgoY29uc3QgdWludDhfdCopbUhlYWRlcikgKyBkdG9ocyhtSGVhZGVyLT5oZWFkZXIuaGVhZGVyU2l6ZSkpOwotICAgIGNvbnN0IFJlc0NodW5rX2hlYWRlciogbGFzdENodW5rID0gY2h1bms7Ci0gICAgd2hpbGUgKCgoY29uc3QgdWludDhfdCopY2h1bmspIDwgKG1EYXRhRW5kLXNpemVvZihSZXNDaHVua19oZWFkZXIpKSAmJgotICAgICAgICAgICAoKGNvbnN0IHVpbnQ4X3QqKWNodW5rKSA8IChtRGF0YUVuZC1kdG9obChjaHVuay0+c2l6ZSkpKSB7Ci0gICAgICAgIHN0YXR1c190IGVyciA9IHZhbGlkYXRlX2NodW5rKGNodW5rLCBzaXplb2YoUmVzQ2h1bmtfaGVhZGVyKSwgbURhdGFFbmQsICJYTUwiKTsKLSAgICAgICAgaWYgKGVyciAhPSBOT19FUlJPUikgewotICAgICAgICAgICAgbUVycm9yID0gZXJyOwotICAgICAgICAgICAgZ290byBkb25lOwotICAgICAgICB9Ci0gICAgICAgIGNvbnN0IHVpbnQxNl90IHR5cGUgPSBkdG9ocyhjaHVuay0+dHlwZSk7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBzaXplID0gZHRvaGwoY2h1bmstPnNpemUpOwotICAgICAgICBYTUxfTk9JU1kocHJpbnRmKCJTY2FubmluZyBAICVwOiB0eXBlPTB4JXgsIHNpemU9MHgleFxuIiwKLSAgICAgICAgICAgICAgICAgICAgICh2b2lkKikoKCh1aW50MzJfdCljaHVuayktKCh1aW50MzJfdCltSGVhZGVyKSksIHR5cGUsIHNpemUpKTsKLSAgICAgICAgaWYgKHR5cGUgPT0gUkVTX1NUUklOR19QT09MX1RZUEUpIHsKLSAgICAgICAgICAgIG1TdHJpbmdzLnNldFRvKGNodW5rLCBzaXplKTsKLSAgICAgICAgfSBlbHNlIGlmICh0eXBlID09IFJFU19YTUxfUkVTT1VSQ0VfTUFQX1RZUEUpIHsKLSAgICAgICAgICAgIG1SZXNJZHMgPSAoY29uc3QgdWludDMyX3QqKQotICAgICAgICAgICAgICAgICgoKGNvbnN0IHVpbnQ4X3QqKWNodW5rKStkdG9ocyhjaHVuay0+aGVhZGVyU2l6ZSkpOwotICAgICAgICAgICAgbU51bVJlc0lkcyA9IChkdG9obChjaHVuay0+c2l6ZSktZHRvaHMoY2h1bmstPmhlYWRlclNpemUpKS9zaXplb2YodWludDMyX3QpOwotICAgICAgICB9IGVsc2UgaWYgKHR5cGUgPj0gUkVTX1hNTF9GSVJTVF9DSFVOS19UWVBFCi0gICAgICAgICAgICAgICAgICAgJiYgdHlwZSA8PSBSRVNfWE1MX0xBU1RfQ0hVTktfVFlQRSkgewotICAgICAgICAgICAgaWYgKHZhbGlkYXRlTm9kZSgoY29uc3QgUmVzWE1MVHJlZV9ub2RlKiljaHVuaykgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgICAgICBtRXJyb3IgPSBCQURfVFlQRTsKLSAgICAgICAgICAgICAgICBnb3RvIGRvbmU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBtQ3VyTm9kZSA9IChjb25zdCBSZXNYTUxUcmVlX25vZGUqKWxhc3RDaHVuazsKLSAgICAgICAgICAgIGlmIChuZXh0Tm9kZSgpID09IEJBRF9ET0NVTUVOVCkgewotICAgICAgICAgICAgICAgIG1FcnJvciA9IEJBRF9UWVBFOwotICAgICAgICAgICAgICAgIGdvdG8gZG9uZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1Sb290Tm9kZSA9IG1DdXJOb2RlOwotICAgICAgICAgICAgbVJvb3RFeHQgPSBtQ3VyRXh0OwotICAgICAgICAgICAgbVJvb3RDb2RlID0gbUV2ZW50Q29kZTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgWE1MX05PSVNZKHByaW50ZigiU2tpcHBpbmcgdW5rbm93biBjaHVuayFcbiIpKTsKLSAgICAgICAgfQotICAgICAgICBsYXN0Q2h1bmsgPSBjaHVuazsKLSAgICAgICAgY2h1bmsgPSAoY29uc3QgUmVzQ2h1bmtfaGVhZGVyKikKLSAgICAgICAgICAgICgoKGNvbnN0IHVpbnQ4X3QqKWNodW5rKSArIHNpemUpOwotICAgIH0KLQotICAgIGlmIChtUm9vdE5vZGUgPT0gTlVMTCkgewotICAgICAgICBMT0dXKCJCYWQgWE1MIGJsb2NrOiBubyByb290IGVsZW1lbnQgbm9kZSBmb3VuZFxuIik7Ci0gICAgICAgIG1FcnJvciA9IEJBRF9UWVBFOwotICAgICAgICBnb3RvIGRvbmU7Ci0gICAgfQotCi0gICAgbUVycm9yID0gbVN0cmluZ3MuZ2V0RXJyb3IoKTsKLQotZG9uZToKLSAgICByZXN0YXJ0KCk7Ci0gICAgcmV0dXJuIG1FcnJvcjsKLX0KLQotc3RhdHVzX3QgUmVzWE1MVHJlZTo6Z2V0RXJyb3IoKSBjb25zdAotewotICAgIHJldHVybiBtRXJyb3I7Ci19Ci0KLXZvaWQgUmVzWE1MVHJlZTo6dW5pbml0KCkKLXsKLSAgICBtRXJyb3IgPSBOT19JTklUOwotICAgIGlmIChtT3duZWREYXRhKSB7Ci0gICAgICAgIGZyZWUobU93bmVkRGF0YSk7Ci0gICAgICAgIG1Pd25lZERhdGEgPSBOVUxMOwotICAgIH0KLSAgICByZXN0YXJ0KCk7Ci19Ci0KLWNvbnN0IFJlc1N0cmluZ1Bvb2wmIFJlc1hNTFRyZWU6OmdldFN0cmluZ3MoKSBjb25zdAotewotICAgIHJldHVybiBtU3RyaW5nczsKLX0KLQotc3RhdHVzX3QgUmVzWE1MVHJlZTo6dmFsaWRhdGVOb2RlKGNvbnN0IFJlc1hNTFRyZWVfbm9kZSogbm9kZSkgY29uc3QKLXsKLSAgICBjb25zdCB1aW50MTZfdCBldmVudENvZGUgPSBkdG9ocyhub2RlLT5oZWFkZXIudHlwZSk7Ci0KLSAgICBzdGF0dXNfdCBlcnIgPSB2YWxpZGF0ZV9jaHVuaygKLSAgICAgICAgJm5vZGUtPmhlYWRlciwgc2l6ZW9mKFJlc1hNTFRyZWVfbm9kZSksCi0gICAgICAgIG1EYXRhRW5kLCAiUmVzWE1MVHJlZV9ub2RlIik7Ci0KLSAgICBpZiAoZXJyID49IE5PX0VSUk9SKSB7Ci0gICAgICAgIC8vIE9ubHkgcGVyZm9ybSBhZGRpdGlvbmFsIHZhbGlkYXRpb24gb24gU1RBUlQgbm9kZXMKLSAgICAgICAgaWYgKGV2ZW50Q29kZSAhPSBSRVNfWE1MX1NUQVJUX0VMRU1FTlRfVFlQRSkgewotICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICB9Ci0KLSAgICAgICAgY29uc3QgdWludDE2X3QgaGVhZGVyU2l6ZSA9IGR0b2hzKG5vZGUtPmhlYWRlci5oZWFkZXJTaXplKTsKLSAgICAgICAgY29uc3QgdWludDMyX3Qgc2l6ZSA9IGR0b2hsKG5vZGUtPmhlYWRlci5zaXplKTsKLSAgICAgICAgY29uc3QgUmVzWE1MVHJlZV9hdHRyRXh0KiBhdHRyRXh0ID0gKGNvbnN0IFJlc1hNTFRyZWVfYXR0ckV4dCopCi0gICAgICAgICAgICAoKChjb25zdCB1aW50OF90Kilub2RlKSArIGhlYWRlclNpemUpOwotICAgICAgICAvLyBjaGVjayBmb3Igc2Vuc2ljYWwgdmFsdWVzIHB1bGxlZCBvdXQgb2YgdGhlIHN0cmVhbSBzbyBmYXIuLi4KLSAgICAgICAgaWYgKChzaXplID49IGhlYWRlclNpemUgKyBzaXplb2YoUmVzWE1MVHJlZV9hdHRyRXh0KSkKLSAgICAgICAgICAgICAgICAmJiAoKHZvaWQqKWF0dHJFeHQgPiAodm9pZCopbm9kZSkpIHsKLSAgICAgICAgICAgIGNvbnN0IHNpemVfdCBhdHRyU2l6ZSA9ICgoc2l6ZV90KWR0b2hzKGF0dHJFeHQtPmF0dHJpYnV0ZVNpemUpKQotICAgICAgICAgICAgICAgICogZHRvaHMoYXR0ckV4dC0+YXR0cmlidXRlQ291bnQpOwotICAgICAgICAgICAgaWYgKChkdG9ocyhhdHRyRXh0LT5hdHRyaWJ1dGVTdGFydCkrYXR0clNpemUpIDw9IChzaXplLWhlYWRlclNpemUpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgTE9HVygiQmFkIFhNTCBibG9jazogbm9kZSBhdHRyaWJ1dGVzIHVzZSAweCV4IGJ5dGVzLCBvbmx5IGhhdmUgMHgleCBieXRlc1xuIiwKLSAgICAgICAgICAgICAgICAgICAgKHVuc2lnbmVkIGludCkoZHRvaHMoYXR0ckV4dC0+YXR0cmlidXRlU3RhcnQpK2F0dHJTaXplKSwKLSAgICAgICAgICAgICAgICAgICAgKHVuc2lnbmVkIGludCkoc2l6ZS1oZWFkZXJTaXplKSk7Ci0gICAgICAgIH0KLSAgICAgICAgZWxzZSB7Ci0gICAgICAgICAgICBMT0dXKCJCYWQgWE1MIHN0YXJ0IGJsb2NrOiBub2RlIGhlYWRlciBzaXplIDB4JXgsIHNpemUgMHgleFxuIiwKLSAgICAgICAgICAgICAgICAodW5zaWduZWQgaW50KWhlYWRlclNpemUsICh1bnNpZ25lZCBpbnQpc2l6ZSk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIEJBRF9UWVBFOwotICAgIH0KLQotICAgIHJldHVybiBlcnI7Ci0KLSNpZiAwCi0gICAgY29uc3QgYm9vbCBpc1N0YXJ0ID0gZHRvaHMobm9kZS0+aGVhZGVyLnR5cGUpID09IFJFU19YTUxfU1RBUlRfRUxFTUVOVF9UWVBFOwotCi0gICAgY29uc3QgdWludDE2X3QgaGVhZGVyU2l6ZSA9IGR0b2hzKG5vZGUtPmhlYWRlci5oZWFkZXJTaXplKTsKLSAgICBjb25zdCB1aW50MzJfdCBzaXplID0gZHRvaGwobm9kZS0+aGVhZGVyLnNpemUpOwotCi0gICAgaWYgKGhlYWRlclNpemUgPj0gKGlzU3RhcnQgPyBzaXplb2YoUmVzWE1MVHJlZV9hdHRyTm9kZSkgOiBzaXplb2YoUmVzWE1MVHJlZV9ub2RlKSkpIHsKLSAgICAgICAgaWYgKHNpemUgPj0gaGVhZGVyU2l6ZSkgewotICAgICAgICAgICAgaWYgKCgoY29uc3QgdWludDhfdCopbm9kZSkgPD0gKG1EYXRhRW5kLXNpemUpKSB7Ci0gICAgICAgICAgICAgICAgaWYgKCFpc1N0YXJ0KSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKCgoKHNpemVfdClkdG9ocyhub2RlLT5hdHRyaWJ1dGVTaXplKSkqZHRvaHMobm9kZS0+YXR0cmlidXRlQ291bnQpKQotICAgICAgICAgICAgICAgICAgICAgICAgPD0gKHNpemUtaGVhZGVyU2l6ZSkpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBMT0dXKCJCYWQgWE1MIGJsb2NrOiBub2RlIGF0dHJpYnV0ZXMgdXNlIDB4JXggYnl0ZXMsIG9ubHkgaGF2ZSAweCV4IGJ5dGVzXG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgKChpbnQpZHRvaHMobm9kZS0+YXR0cmlidXRlU2l6ZSkpKmR0b2hzKG5vZGUtPmF0dHJpYnV0ZUNvdW50KSwKLSAgICAgICAgICAgICAgICAgICAgICAgIChpbnQpKHNpemUtaGVhZGVyU2l6ZSkpOwotICAgICAgICAgICAgICAgIHJldHVybiBCQURfVFlQRTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIExPR1coIkJhZCBYTUwgYmxvY2s6IG5vZGUgYXQgMHgleCBleHRlbmRzIGJleW9uZCBkYXRhIGVuZCAweCV4XG4iLAotICAgICAgICAgICAgICAgICAgICAoaW50KSgoKGNvbnN0IHVpbnQ4X3QqKW5vZGUpLSgoY29uc3QgdWludDhfdCopbUhlYWRlcikpLCAoaW50KW1TaXplKTsKLSAgICAgICAgICAgIHJldHVybiBCQURfVFlQRTsKLSAgICAgICAgfQotICAgICAgICBMT0dXKCJCYWQgWE1MIGJsb2NrOiBub2RlIGF0IDB4JXggaGVhZGVyIHNpemUgMHgleCBzbWFsbGVyIHRoYW4gdG90YWwgc2l6ZSAweCV4XG4iLAotICAgICAgICAgICAgICAgIChpbnQpKCgoY29uc3QgdWludDhfdCopbm9kZSktKChjb25zdCB1aW50OF90KiltSGVhZGVyKSksCi0gICAgICAgICAgICAgICAgKGludCloZWFkZXJTaXplLCAoaW50KXNpemUpOwotICAgICAgICByZXR1cm4gQkFEX1RZUEU7Ci0gICAgfQotICAgIExPR1coIkJhZCBYTUwgYmxvY2s6IG5vZGUgYXQgMHgleCBoZWFkZXIgc2l6ZSAweCV4IHRvbyBzbWFsbFxuIiwKLSAgICAgICAgICAgIChpbnQpKCgoY29uc3QgdWludDhfdCopbm9kZSktKChjb25zdCB1aW50OF90KiltSGVhZGVyKSksCi0gICAgICAgICAgICAoaW50KWhlYWRlclNpemUpOwotICAgIHJldHVybiBCQURfVFlQRTsKLSNlbmRpZgotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0cnVjdCBSZXNUYWJsZTo6SGVhZGVyCi17Ci0gICAgSGVhZGVyKCkgOiBvd25lZERhdGEoTlVMTCksIGhlYWRlcihOVUxMKSB7IH0KLQotICAgIHZvaWQqICAgICAgICAgICAgICAgICAgICAgICAgICAgb3duZWREYXRhOwotICAgIGNvbnN0IFJlc1RhYmxlX2hlYWRlciogICAgICAgICAgaGVhZGVyOwotICAgIHNpemVfdCAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZTsKLSAgICBjb25zdCB1aW50OF90KiAgICAgICAgICAgICAgICAgIGRhdGFFbmQ7Ci0gICAgc2l6ZV90ICAgICAgICAgICAgICAgICAgICAgICAgICBpbmRleDsKLSAgICB2b2lkKiAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvb2tpZTsKLQotICAgIFJlc1N0cmluZ1Bvb2wgICAgICAgICAgICAgICAgICAgdmFsdWVzOwotfTsKLQotc3RydWN0IFJlc1RhYmxlOjpUeXBlCi17Ci0gICAgVHlwZShjb25zdCBIZWFkZXIqIF9oZWFkZXIsIGNvbnN0IFBhY2thZ2UqIF9wYWNrYWdlLCBzaXplX3QgY291bnQpCi0gICAgICAgIDogaGVhZGVyKF9oZWFkZXIpLCBwYWNrYWdlKF9wYWNrYWdlKSwgZW50cnlDb3VudChjb3VudCksCi0gICAgICAgICAgdHlwZVNwZWMoTlVMTCksIHR5cGVTcGVjRmxhZ3MoTlVMTCkgeyB9Ci0gICAgY29uc3QgSGVhZGVyKiBjb25zdCAgICAgICAgICAgICBoZWFkZXI7Ci0gICAgY29uc3QgUGFja2FnZSogY29uc3QgICAgICAgICAgICBwYWNrYWdlOwotICAgIGNvbnN0IHNpemVfdCAgICAgICAgICAgICAgICAgICAgZW50cnlDb3VudDsKLSAgICBjb25zdCBSZXNUYWJsZV90eXBlU3BlYyogICAgICAgIHR5cGVTcGVjOwotICAgIGNvbnN0IHVpbnQzMl90KiAgICAgICAgICAgICAgICAgdHlwZVNwZWNGbGFnczsKLSAgICBWZWN0b3I8Y29uc3QgUmVzVGFibGVfdHlwZSo+ICAgIGNvbmZpZ3M7Ci19OwotCi1zdHJ1Y3QgUmVzVGFibGU6OlBhY2thZ2UKLXsKLSAgICBQYWNrYWdlKGNvbnN0IEhlYWRlciogX2hlYWRlciwgY29uc3QgUmVzVGFibGVfcGFja2FnZSogX3BhY2thZ2UpCi0gICAgICAgIDogaGVhZGVyKF9oZWFkZXIpLCBwYWNrYWdlKF9wYWNrYWdlKSB7IH0KLSAgICB+UGFja2FnZSgpCi0gICAgewotICAgICAgICBzaXplX3QgaSA9IHR5cGVzLnNpemUoKTsKLSAgICAgICAgd2hpbGUgKGkgPiAwKSB7Ci0gICAgICAgICAgICBpLS07Ci0gICAgICAgICAgICBkZWxldGUgdHlwZXNbaV07Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgY29uc3QgSGVhZGVyKiBjb25zdCAgICAgICAgICAgICBoZWFkZXI7Ci0gICAgY29uc3QgUmVzVGFibGVfcGFja2FnZSogY29uc3QgICBwYWNrYWdlOwotICAgIFZlY3RvcjxUeXBlKj4gICAgICAgICAgICAgICAgICAgdHlwZXM7Ci0KLSAgICBjb25zdCBUeXBlKiBnZXRUeXBlKHNpemVfdCBpZHgpIGNvbnN0IHsKLSAgICAgICAgcmV0dXJuIGlkeCA8IHR5cGVzLnNpemUoKSA/IHR5cGVzW2lkeF0gOiBOVUxMOwotICAgIH0KLX07Ci0KLS8vIEEgZ3JvdXAgb2Ygb2JqZWN0cyBkZXNjcmliaW5nIGEgcGFydGljdWxhciByZXNvdXJjZSBwYWNrYWdlLgotLy8gVGhlIGZpcnN0IGluICdwYWNrYWdlJyBpcyBhbHdheXMgdGhlIHJvb3Qgb2JqZWN0IChmcm9tIHRoZSByZXNvdXJjZQotLy8gdGFibGUgdGhhdCBkZWZpbmVkIHRoZSBwYWNrYWdlKTsgdGhlIG9uZXMgYWZ0ZXIgYXJlIHNraW5zIG9uIHRvcCBvZiBpdC4KLXN0cnVjdCBSZXNUYWJsZTo6UGFja2FnZUdyb3VwCi17Ci0gICAgUGFja2FnZUdyb3VwKGNvbnN0IFN0cmluZzE2JiBfbmFtZSwgdWludDMyX3QgX2lkKQotICAgICAgICA6IG5hbWUoX25hbWUpLCBpZChfaWQpLCB0eXBlQ291bnQoMCksIGJhZ3MoTlVMTCkgeyB9Ci0gICAgflBhY2thZ2VHcm91cCgpIHsKLSAgICAgICAgY2xlYXJCYWdDYWNoZSgpOwotICAgICAgICBjb25zdCBzaXplX3QgTiA9IHBhY2thZ2VzLnNpemUoKTsKLSAgICAgICAgZm9yIChzaXplX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICAgICAgZGVsZXRlIHBhY2thZ2VzW2ldOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgdm9pZCBjbGVhckJhZ0NhY2hlKCkgewotICAgICAgICBpZiAoYmFncykgewotICAgICAgICAgICAgVEFCTEVfTk9JU1kocHJpbnRmKCJiYWdzPSVwXG4iLCBiYWdzKSk7Ci0gICAgICAgICAgICBQYWNrYWdlKiBwa2cgPSBwYWNrYWdlc1swXTsKLSAgICAgICAgICAgIFRBQkxFX05PSVNZKHByaW50ZigidHlwZUNvdW50PSV4XG4iLCB0eXBlQ291bnQpKTsKLSAgICAgICAgICAgIGZvciAoc2l6ZV90IGk9MDsgaTx0eXBlQ291bnQ7IGkrKykgewotICAgICAgICAgICAgICAgIFRBQkxFX05PSVNZKHByaW50ZigidHlwZT0lZFxuIiwgaSkpOwotICAgICAgICAgICAgICAgIGNvbnN0IFR5cGUqIHR5cGUgPSBwa2ctPmdldFR5cGUoaSk7Ci0gICAgICAgICAgICAgICAgaWYgKHR5cGUgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICBiYWdfc2V0KiogdHlwZUJhZ3MgPSBiYWdzW2ldOwotICAgICAgICAgICAgICAgICAgICBUQUJMRV9OT0lTWShwcmludGYoInR5cGVCYWdzPSVwXG4iLCB0eXBlQmFncykpOwotICAgICAgICAgICAgICAgICAgICBpZiAodHlwZUJhZ3MpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIFRBQkxFX05PSVNZKHByaW50ZigidHlwZS0+ZW50cnlDb3VudD0leFxuIiwgdHlwZS0+ZW50cnlDb3VudCkpOwotICAgICAgICAgICAgICAgICAgICAgICAgY29uc3Qgc2l6ZV90IE4gPSB0eXBlLT5lbnRyeUNvdW50OwotICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChzaXplX3Qgaj0wOyBqPE47IGorKykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmICh0eXBlQmFnc1tqXSAmJiB0eXBlQmFnc1tqXSAhPSAoYmFnX3NldCopMHhGRkZGRkZGRikKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZnJlZSh0eXBlQmFnc1tqXSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBmcmVlKHR5cGVCYWdzKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGZyZWUoYmFncyk7Ci0gICAgICAgICAgICBiYWdzID0gTlVMTDsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBTdHJpbmcxNiBjb25zdCAgICAgICAgICAgICAgICAgIG5hbWU7Ci0gICAgdWludDMyX3QgY29uc3QgICAgICAgICAgICAgICAgICBpZDsKLSAgICBWZWN0b3I8UGFja2FnZSo+ICAgICAgICAgICAgICAgIHBhY2thZ2VzOwotCi0gICAgLy8gVGFrZW4gZnJvbSB0aGUgcm9vdCBwYWNrYWdlLgotICAgIFJlc1N0cmluZ1Bvb2wgICAgICAgICAgICAgICAgICAgdHlwZVN0cmluZ3M7Ci0gICAgUmVzU3RyaW5nUG9vbCAgICAgICAgICAgICAgICAgICBrZXlTdHJpbmdzOwotICAgIHNpemVfdCAgICAgICAgICAgICAgICAgICAgICAgICAgdHlwZUNvdW50OwotCi0gICAgLy8gQ29tcHV0ZWQgYXR0cmlidXRlIGJhZ3MsIGZpcnN0IGluZGV4ZWQgYnkgdGhlIHR5cGUgYW5kIHNlY29uZAotICAgIC8vIGJ5IHRoZSBlbnRyeSBpbiB0aGF0IHR5cGUuCi0gICAgYmFnX3NldCoqKiAgICAgICAgICAgICAgICAgICAgICBiYWdzOwotfTsKLQotc3RydWN0IFJlc1RhYmxlOjpiYWdfc2V0Ci17Ci0gICAgc2l6ZV90IG51bUF0dHJzOyAgICAvLyBudW1iZXIgaW4gYXJyYXkKLSAgICBzaXplX3QgYXZhaWxBdHRyczsgIC8vIHRvdGFsIHNwYWNlIGluIGFycmF5Ci0gICAgdWludDMyX3QgdHlwZVNwZWNGbGFnczsKLSAgICAvLyBGb2xsb3dlZCBieSAnbnVtQXR0cicgYmFnX2VudHJ5IHN0cnVjdHVyZXMuCi19OwotCi1SZXNUYWJsZTo6VGhlbWU6OlRoZW1lKGNvbnN0IFJlc1RhYmxlJiB0YWJsZSkKLSAgICA6IG1UYWJsZSh0YWJsZSkKLXsKLSAgICBtZW1zZXQobVBhY2thZ2VzLCAwLCBzaXplb2YobVBhY2thZ2VzKSk7Ci19Ci0KLVJlc1RhYmxlOjpUaGVtZTo6flRoZW1lKCkKLXsKLSAgICBmb3IgKHNpemVfdCBpPTA7IGk8UmVzX01BWFBBQ0tBR0U7IGkrKykgewotICAgICAgICBwYWNrYWdlX2luZm8qIHBpID0gbVBhY2thZ2VzW2ldOwotICAgICAgICBpZiAocGkgIT0gTlVMTCkgewotICAgICAgICAgICAgZnJlZV9wYWNrYWdlKHBpKTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotdm9pZCBSZXNUYWJsZTo6VGhlbWU6OmZyZWVfcGFja2FnZShwYWNrYWdlX2luZm8qIHBpKQotewotICAgIGZvciAoc2l6ZV90IGo9MDsgajxwaS0+bnVtVHlwZXM7IGorKykgewotICAgICAgICB0aGVtZV9lbnRyeSogdGUgPSBwaS0+dHlwZXNbal0uZW50cmllczsKLSAgICAgICAgaWYgKHRlICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGZyZWUodGUpOwotICAgICAgICB9Ci0gICAgfQotICAgIGZyZWUocGkpOwotfQotCi1SZXNUYWJsZTo6VGhlbWU6OnBhY2thZ2VfaW5mbyogUmVzVGFibGU6OlRoZW1lOjpjb3B5X3BhY2thZ2UocGFja2FnZV9pbmZvKiBwaSkKLXsKLSAgICBwYWNrYWdlX2luZm8qIG5ld3BpID0gKHBhY2thZ2VfaW5mbyopbWFsbG9jKAotICAgICAgICBzaXplb2YocGFja2FnZV9pbmZvKSArIChwaS0+bnVtVHlwZXMqc2l6ZW9mKHR5cGVfaW5mbykpKTsKLSAgICBuZXdwaS0+bnVtVHlwZXMgPSBwaS0+bnVtVHlwZXM7Ci0gICAgZm9yIChzaXplX3Qgaj0wOyBqPG5ld3BpLT5udW1UeXBlczsgaisrKSB7Ci0gICAgICAgIHNpemVfdCBjbnQgPSBwaS0+dHlwZXNbal0ubnVtRW50cmllczsKLSAgICAgICAgbmV3cGktPnR5cGVzW2pdLm51bUVudHJpZXMgPSBjbnQ7Ci0gICAgICAgIHRoZW1lX2VudHJ5KiB0ZSA9IHBpLT50eXBlc1tqXS5lbnRyaWVzOwotICAgICAgICBpZiAodGUgIT0gTlVMTCkgewotICAgICAgICAgICAgdGhlbWVfZW50cnkqIG5ld3RlID0gKHRoZW1lX2VudHJ5KiltYWxsb2MoY250KnNpemVvZih0aGVtZV9lbnRyeSkpOwotICAgICAgICAgICAgbmV3cGktPnR5cGVzW2pdLmVudHJpZXMgPSBuZXd0ZTsKLSAgICAgICAgICAgIG1lbWNweShuZXd0ZSwgdGUsIGNudCpzaXplb2YodGhlbWVfZW50cnkpKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG5ld3BpLT50eXBlc1tqXS5lbnRyaWVzID0gTlVMTDsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gbmV3cGk7Ci19Ci0KLXN0YXR1c190IFJlc1RhYmxlOjpUaGVtZTo6YXBwbHlTdHlsZSh1aW50MzJfdCByZXNJRCwgYm9vbCBmb3JjZSkKLXsKLSAgICBjb25zdCBiYWdfZW50cnkqIGJhZzsKLSAgICB1aW50MzJfdCBiYWdUeXBlU3BlY0ZsYWdzID0gMDsKLSAgICBtVGFibGUubG9jaygpOwotICAgIGNvbnN0IHNzaXplX3QgTiA9IG1UYWJsZS5nZXRCYWdMb2NrZWQocmVzSUQsICZiYWcsICZiYWdUeXBlU3BlY0ZsYWdzKTsKLSAgICBUQUJMRV9OT0lTWShMT0dWKCJBcHBseWluZyBzdHlsZSAweCUwOHggdG8gdGhlbWUgJXAsIGNvdW50PSVkIiwgcmVzSUQsIHRoaXMsIE4pKTsKLSAgICBpZiAoTiA8IDApIHsKLSAgICAgICAgbVRhYmxlLnVubG9jaygpOwotICAgICAgICByZXR1cm4gTjsKLSAgICB9Ci0KLSAgICB1aW50MzJfdCBjdXJQYWNrYWdlID0gMHhmZmZmZmZmZjsKLSAgICBzc2l6ZV90IGN1clBhY2thZ2VJbmRleCA9IDA7Ci0gICAgcGFja2FnZV9pbmZvKiBjdXJQSSA9IE5VTEw7Ci0gICAgdWludDMyX3QgY3VyVHlwZSA9IDB4ZmZmZmZmZmY7Ci0gICAgc2l6ZV90IG51bUVudHJpZXMgPSAwOwotICAgIHRoZW1lX2VudHJ5KiBjdXJFbnRyaWVzID0gTlVMTDsKLQotICAgIGNvbnN0IGJhZ19lbnRyeSogZW5kID0gYmFnICsgTjsKLSAgICB3aGlsZSAoYmFnIDwgZW5kKSB7Ci0gICAgICAgIGNvbnN0IHVpbnQzMl90IGF0dHJSZXMgPSBiYWctPm1hcC5uYW1lLmlkZW50OwotICAgICAgICBjb25zdCB1aW50MzJfdCBwID0gUmVzX0dFVFBBQ0tBR0UoYXR0clJlcyk7Ci0gICAgICAgIGNvbnN0IHVpbnQzMl90IHQgPSBSZXNfR0VUVFlQRShhdHRyUmVzKTsKLSAgICAgICAgY29uc3QgdWludDMyX3QgZSA9IFJlc19HRVRFTlRSWShhdHRyUmVzKTsKLQotICAgICAgICBpZiAoY3VyUGFja2FnZSAhPSBwKSB7Ci0gICAgICAgICAgICBjb25zdCBzc2l6ZV90IHBpZHggPSBtVGFibGUuZ2V0UmVzb3VyY2VQYWNrYWdlSW5kZXgoYXR0clJlcyk7Ci0gICAgICAgICAgICBpZiAocGlkeCA8IDApIHsKLSAgICAgICAgICAgICAgICBMT0dFKCJTdHlsZSBjb250YWlucyBrZXkgd2l0aCBiYWQgcGFja2FnZTogMHglMDh4XG4iLCBhdHRyUmVzKTsKLSAgICAgICAgICAgICAgICBiYWcrKzsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGN1clBhY2thZ2UgPSBwOwotICAgICAgICAgICAgY3VyUGFja2FnZUluZGV4ID0gcGlkeDsKLSAgICAgICAgICAgIGN1clBJID0gbVBhY2thZ2VzW3BpZHhdOwotICAgICAgICAgICAgaWYgKGN1clBJID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICBQYWNrYWdlR3JvdXAqIGNvbnN0IGdycCA9IG1UYWJsZS5tUGFja2FnZUdyb3Vwc1twaWR4XTsKLSAgICAgICAgICAgICAgICBpbnQgY250ID0gZ3JwLT50eXBlQ291bnQ7Ci0gICAgICAgICAgICAgICAgY3VyUEkgPSAocGFja2FnZV9pbmZvKiltYWxsb2MoCi0gICAgICAgICAgICAgICAgICAgIHNpemVvZihwYWNrYWdlX2luZm8pICsgKGNudCpzaXplb2YodHlwZV9pbmZvKSkpOwotICAgICAgICAgICAgICAgIGN1clBJLT5udW1UeXBlcyA9IGNudDsKLSAgICAgICAgICAgICAgICBtZW1zZXQoY3VyUEktPnR5cGVzLCAwLCBjbnQqc2l6ZW9mKHR5cGVfaW5mbykpOwotICAgICAgICAgICAgICAgIG1QYWNrYWdlc1twaWR4XSA9IGN1clBJOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY3VyVHlwZSA9IDB4ZmZmZmZmZmY7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGN1clR5cGUgIT0gdCkgewotICAgICAgICAgICAgaWYgKHQgPj0gY3VyUEktPm51bVR5cGVzKSB7Ci0gICAgICAgICAgICAgICAgTE9HRSgiU3R5bGUgY29udGFpbnMga2V5IHdpdGggYmFkIHR5cGU6IDB4JTA4eFxuIiwgYXR0clJlcyk7Ci0gICAgICAgICAgICAgICAgYmFnKys7Ci0gICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjdXJUeXBlID0gdDsKLSAgICAgICAgICAgIGN1ckVudHJpZXMgPSBjdXJQSS0+dHlwZXNbdF0uZW50cmllczsKLSAgICAgICAgICAgIGlmIChjdXJFbnRyaWVzID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICBQYWNrYWdlR3JvdXAqIGNvbnN0IGdycCA9IG1UYWJsZS5tUGFja2FnZUdyb3Vwc1tjdXJQYWNrYWdlSW5kZXhdOwotICAgICAgICAgICAgICAgIGNvbnN0IFR5cGUqIHR5cGUgPSBncnAtPnBhY2thZ2VzWzBdLT5nZXRUeXBlKHQpOwotICAgICAgICAgICAgICAgIGludCBjbnQgPSB0eXBlICE9IE5VTEwgPyB0eXBlLT5lbnRyeUNvdW50IDogMDsKLSAgICAgICAgICAgICAgICBjdXJFbnRyaWVzID0gKHRoZW1lX2VudHJ5KiltYWxsb2MoY250KnNpemVvZih0aGVtZV9lbnRyeSkpOwotICAgICAgICAgICAgICAgIG1lbXNldChjdXJFbnRyaWVzLCBSZXNfdmFsdWU6OlRZUEVfTlVMTCwgY250KnNpemVvZih0aGVtZV9lbnRyeSkpOwotICAgICAgICAgICAgICAgIGN1clBJLT50eXBlc1t0XS5udW1FbnRyaWVzID0gY250OwotICAgICAgICAgICAgICAgIGN1clBJLT50eXBlc1t0XS5lbnRyaWVzID0gY3VyRW50cmllczsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG51bUVudHJpZXMgPSBjdXJQSS0+dHlwZXNbdF0ubnVtRW50cmllczsKLSAgICAgICAgfQotICAgICAgICBpZiAoZSA+PSBudW1FbnRyaWVzKSB7Ci0gICAgICAgICAgICBMT0dFKCJTdHlsZSBjb250YWlucyBrZXkgd2l0aCBiYWQgZW50cnk6IDB4JTA4eFxuIiwgYXR0clJlcyk7Ci0gICAgICAgICAgICBiYWcrKzsKLSAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICB9Ci0gICAgICAgIHRoZW1lX2VudHJ5KiBjdXJFbnRyeSA9IGN1ckVudHJpZXMgKyBlOwotICAgICAgICBUQUJMRV9OT0lTWShMT0dWKCJBdHRyIDB4JTA4eDogdHlwZT0weCV4LCBkYXRhPTB4JTA4eDsgY3VyVHlwZT0weCV4IiwKLSAgICAgICAgICAgICAgICAgICBhdHRyUmVzLCBiYWctPm1hcC52YWx1ZS5kYXRhVHlwZSwgYmFnLT5tYXAudmFsdWUuZGF0YSwKLSAgICAgICAgICAgICBjdXJFbnRyeS0+dmFsdWUuZGF0YVR5cGUpKTsKLSAgICAgICAgaWYgKGZvcmNlIHx8IGN1ckVudHJ5LT52YWx1ZS5kYXRhVHlwZSA9PSBSZXNfdmFsdWU6OlRZUEVfTlVMTCkgewotICAgICAgICAgICAgY3VyRW50cnktPnN0cmluZ0Jsb2NrID0gYmFnLT5zdHJpbmdCbG9jazsKLSAgICAgICAgICAgIGN1ckVudHJ5LT50eXBlU3BlY0ZsYWdzIHw9IGJhZ1R5cGVTcGVjRmxhZ3M7Ci0gICAgICAgICAgICBjdXJFbnRyeS0+dmFsdWUgPSBiYWctPm1hcC52YWx1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGJhZysrOwotICAgIH0KLQotICAgIG1UYWJsZS51bmxvY2soKTsKLQotICAgIC8vTE9HSSgiQXBwbHlpbmcgc3R5bGUgMHglMDh4IChmb3JjZT0lZCkgIHRoZW1lICVwLi4uXG4iLCByZXNJRCwgZm9yY2UsIHRoaXMpOwotICAgIC8vZHVtcFRvTG9nKCk7Ci0gICAgCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBSZXNUYWJsZTo6VGhlbWU6OnNldFRvKGNvbnN0IFRoZW1lJiBvdGhlcikKLXsKLSAgICAvL0xPR0koIlNldHRpbmcgdGhlbWUgJXAgZnJvbSB0aGVtZSAlcC4uLlxuIiwgdGhpcywgJm90aGVyKTsKLSAgICAvL2R1bXBUb0xvZygpOwotICAgIC8vb3RoZXIuZHVtcFRvTG9nKCk7Ci0gICAgCi0gICAgaWYgKCZtVGFibGUgPT0gJm90aGVyLm1UYWJsZSkgewotICAgICAgICBmb3IgKHNpemVfdCBpPTA7IGk8UmVzX01BWFBBQ0tBR0U7IGkrKykgewotICAgICAgICAgICAgaWYgKG1QYWNrYWdlc1tpXSAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgZnJlZV9wYWNrYWdlKG1QYWNrYWdlc1tpXSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAob3RoZXIubVBhY2thZ2VzW2ldICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICBtUGFja2FnZXNbaV0gPSBjb3B5X3BhY2thZ2Uob3RoZXIubVBhY2thZ2VzW2ldKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgbVBhY2thZ2VzW2ldID0gTlVMTDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIC8vIEB0b2RvOiBuZWVkIHRvIHJlYWxseSBpbXBsZW1lbnQgdGhpcywgbm90IGp1c3QgY29weQotICAgICAgICAvLyB0aGUgc3lzdGVtIHBhY2thZ2UgKHdoaWNoIGlzIHN0aWxsIHdyb25nIGJlY2F1c2UgaXQgaXNuJ3QKLSAgICAgICAgLy8gZml4aW5nIHVwIHJlc291cmNlIHJlZmVyZW5jZXMpLgotICAgICAgICBmb3IgKHNpemVfdCBpPTA7IGk8UmVzX01BWFBBQ0tBR0U7IGkrKykgewotICAgICAgICAgICAgaWYgKG1QYWNrYWdlc1tpXSAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgZnJlZV9wYWNrYWdlKG1QYWNrYWdlc1tpXSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoaSA9PSAwICYmIG90aGVyLm1QYWNrYWdlc1tpXSAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgbVBhY2thZ2VzW2ldID0gY29weV9wYWNrYWdlKG90aGVyLm1QYWNrYWdlc1tpXSk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIG1QYWNrYWdlc1tpXSA9IE5VTEw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvL0xPR0koIkZpbmFsIHRoZW1lOiIpOwotICAgIC8vZHVtcFRvTG9nKCk7Ci0gICAgCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zc2l6ZV90IFJlc1RhYmxlOjpUaGVtZTo6Z2V0QXR0cmlidXRlKHVpbnQzMl90IHJlc0lELCBSZXNfdmFsdWUqIG91dFZhbHVlLAotICAgICAgICB1aW50MzJfdCogb3V0VHlwZVNwZWNGbGFncykgY29uc3QKLXsKLSAgICBpbnQgY250ID0gMjA7Ci0KLSAgICBpZiAob3V0VHlwZVNwZWNGbGFncyAhPSBOVUxMKSAqb3V0VHlwZVNwZWNGbGFncyA9IDA7Ci0gICAgCi0gICAgZG8gewotICAgICAgICBjb25zdCBzc2l6ZV90IHAgPSBtVGFibGUuZ2V0UmVzb3VyY2VQYWNrYWdlSW5kZXgocmVzSUQpOwotICAgICAgICBjb25zdCB1aW50MzJfdCB0ID0gUmVzX0dFVFRZUEUocmVzSUQpOwotICAgICAgICBjb25zdCB1aW50MzJfdCBlID0gUmVzX0dFVEVOVFJZKHJlc0lEKTsKLQotICAgICAgICBUQUJMRV9OT0lTWShMT0dWKCJMb29raW5nIHVwIGF0dHIgMHglMDh4IGluIHRoZW1lICVwIiwgcmVzSUQsIHRoaXMpKTsKLQotICAgICAgICBpZiAocCA+PSAwKSB7Ci0gICAgICAgICAgICBjb25zdCBwYWNrYWdlX2luZm8qIGNvbnN0IHBpID0gbVBhY2thZ2VzW3BdOwotICAgICAgICAgICAgaWYgKHBpICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICBpZiAodCA8IHBpLT5udW1UeXBlcykgewotICAgICAgICAgICAgICAgICAgICBjb25zdCB0eXBlX2luZm8mIHRpID0gcGktPnR5cGVzW3RdOwotICAgICAgICAgICAgICAgICAgICBpZiAoZSA8IHRpLm51bUVudHJpZXMpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHRoZW1lX2VudHJ5JiB0ZSA9IHRpLmVudHJpZXNbZV07Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKG91dFR5cGVTcGVjRmxhZ3MgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqb3V0VHlwZVNwZWNGbGFncyB8PSB0ZS50eXBlU3BlY0ZsYWdzOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHVpbnQ4X3QgdHlwZSA9IHRlLnZhbHVlLmRhdGFUeXBlOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHR5cGUgPT0gUmVzX3ZhbHVlOjpUWVBFX0FUVFJJQlVURSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjbnQgPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNudC0tOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXNJRCA9IHRlLnZhbHVlLmRhdGE7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBMT0dXKCJUb28gbWFueSBhdHRyaWJ1dGUgcmVmZXJlbmNlcywgc3RvcHBlZCBhdDogMHglMDh4XG4iLCByZXNJRCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJBRF9JTkRFWDsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAodHlwZSAhPSBSZXNfdmFsdWU6OlRZUEVfTlVMTCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICpvdXRWYWx1ZSA9IHRlLnZhbHVlOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiB0ZS5zdHJpbmdCbG9jazsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBCQURfSU5ERVg7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgYnJlYWs7Ci0KLSAgICB9IHdoaWxlICh0cnVlKTsKLQotICAgIHJldHVybiBCQURfSU5ERVg7Ci19Ci0KLXNzaXplX3QgUmVzVGFibGU6OlRoZW1lOjpyZXNvbHZlQXR0cmlidXRlUmVmZXJlbmNlKFJlc192YWx1ZSogaW5PdXRWYWx1ZSwKLSAgICAgICAgc3NpemVfdCBibG9ja0luZGV4LCB1aW50MzJfdCogb3V0TGFzdFJlZiwKLSAgICAgICAgdWludDMyX3QqIGlub3V0VHlwZVNwZWNGbGFncykgY29uc3QKLXsKLSAgICAvL3ByaW50ZigiUmVzb2x2aW5nIHR5cGU9MHgleFxuIiwgaW5PdXRWYWx1ZS0+ZGF0YVR5cGUpOwotICAgIGlmIChpbk91dFZhbHVlLT5kYXRhVHlwZSA9PSBSZXNfdmFsdWU6OlRZUEVfQVRUUklCVVRFKSB7Ci0gICAgICAgIHVpbnQzMl90IG5ld1R5cGVTcGVjRmxhZ3M7Ci0gICAgICAgIGJsb2NrSW5kZXggPSBnZXRBdHRyaWJ1dGUoaW5PdXRWYWx1ZS0+ZGF0YSwgaW5PdXRWYWx1ZSwgJm5ld1R5cGVTcGVjRmxhZ3MpOwotICAgICAgICBpZiAoaW5vdXRUeXBlU3BlY0ZsYWdzICE9IE5VTEwpICppbm91dFR5cGVTcGVjRmxhZ3MgfD0gbmV3VHlwZVNwZWNGbGFnczsKLSAgICAgICAgLy9wcmludGYoIlJldHJpZXZlZCBhdHRyaWJ1dGUgbmV3IHR5cGU9MHgleFxuIiwgaW5PdXRWYWx1ZS0+ZGF0YVR5cGUpOwotICAgICAgICBpZiAoYmxvY2tJbmRleCA8IDApIHsKLSAgICAgICAgICAgIHJldHVybiBibG9ja0luZGV4OwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBtVGFibGUucmVzb2x2ZVJlZmVyZW5jZShpbk91dFZhbHVlLCBibG9ja0luZGV4LCBvdXRMYXN0UmVmKTsKLX0KLQotdm9pZCBSZXNUYWJsZTo6VGhlbWU6OmR1bXBUb0xvZygpIGNvbnN0Ci17Ci0gICAgTE9HSSgiVGhlbWUgJXA6XG4iLCB0aGlzKTsKLSAgICBmb3IgKHNpemVfdCBpPTA7IGk8UmVzX01BWFBBQ0tBR0U7IGkrKykgewotICAgICAgICBwYWNrYWdlX2luZm8qIHBpID0gbVBhY2thZ2VzW2ldOwotICAgICAgICBpZiAocGkgPT0gTlVMTCkgY29udGludWU7Ci0gICAgICAgIAotICAgICAgICBMT0dJKCIgIFBhY2thZ2UgIzB4JTAyeDpcbiIsIChpbnQpKGkrMSkpOwotICAgICAgICBmb3IgKHNpemVfdCBqPTA7IGo8cGktPm51bVR5cGVzOyBqKyspIHsKLSAgICAgICAgICAgIHR5cGVfaW5mbyYgdGkgPSBwaS0+dHlwZXNbal07Ci0gICAgICAgICAgICBpZiAodGkubnVtRW50cmllcyA9PSAwKSBjb250aW51ZTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgTE9HSSgiICAgIFR5cGUgIzB4JTAyeDpcbiIsIChpbnQpKGorMSkpOwotICAgICAgICAgICAgZm9yIChzaXplX3Qgaz0wOyBrPHRpLm51bUVudHJpZXM7IGsrKykgewotICAgICAgICAgICAgICAgIHRoZW1lX2VudHJ5JiB0ZSA9IHRpLmVudHJpZXNba107Ci0gICAgICAgICAgICAgICAgaWYgKHRlLnZhbHVlLmRhdGFUeXBlID09IFJlc192YWx1ZTo6VFlQRV9OVUxMKSBjb250aW51ZTsKLSAgICAgICAgICAgICAgICBMT0dJKCIgICAgICAweCUwOHg6IHQ9MHgleCwgZD0weCUwOHggKGJsb2NrPSVkKVxuIiwKLSAgICAgICAgICAgICAgICAgICAgIChpbnQpUmVzX01BS0VJRChpLCBqLCBrKSwKLSAgICAgICAgICAgICAgICAgICAgIHRlLnZhbHVlLmRhdGFUeXBlLCAoaW50KXRlLnZhbHVlLmRhdGEsIChpbnQpdGUuc3RyaW5nQmxvY2spOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQotCi1SZXNUYWJsZTo6UmVzVGFibGUoKQotICAgIDogbUVycm9yKE5PX0lOSVQpCi17Ci0gICAgbWVtc2V0KCZtUGFyYW1zLCAwLCBzaXplb2YobVBhcmFtcykpOwotICAgIG1lbXNldChtUGFja2FnZU1hcCwgMCwgc2l6ZW9mKG1QYWNrYWdlTWFwKSk7Ci0gICAgLy9MT0dJKCJDcmVhdGluZyBSZXNUYWJsZSAlcFxuIiwgdGhpcyk7Ci19Ci0KLVJlc1RhYmxlOjpSZXNUYWJsZShjb25zdCB2b2lkKiBkYXRhLCBzaXplX3Qgc2l6ZSwgdm9pZCogY29va2llLCBib29sIGNvcHlEYXRhKQotICAgIDogbUVycm9yKE5PX0lOSVQpCi17Ci0gICAgbWVtc2V0KCZtUGFyYW1zLCAwLCBzaXplb2YobVBhcmFtcykpOwotICAgIG1lbXNldChtUGFja2FnZU1hcCwgMCwgc2l6ZW9mKG1QYWNrYWdlTWFwKSk7Ci0gICAgYWRkKGRhdGEsIHNpemUsIGNvb2tpZSwgY29weURhdGEpOwotICAgIExPR19GQVRBTF9JRihtRXJyb3IgIT0gTk9fRVJST1IsICJFcnJvciBwYXJzaW5nIHJlc291cmNlIHRhYmxlIik7Ci0gICAgLy9MT0dJKCJDcmVhdGluZyBSZXNUYWJsZSAlcFxuIiwgdGhpcyk7Ci19Ci0KLVJlc1RhYmxlOjp+UmVzVGFibGUoKQotewotICAgIC8vTE9HSSgiRGVzdHJveWluZyBSZXNUYWJsZSBpbiAlcFxuIiwgdGhpcyk7Ci0gICAgdW5pbml0KCk7Ci19Ci0KLWlubGluZSBzc2l6ZV90IFJlc1RhYmxlOjpnZXRSZXNvdXJjZVBhY2thZ2VJbmRleCh1aW50MzJfdCByZXNJRCkgY29uc3QKLXsKLSAgICByZXR1cm4gKChzc2l6ZV90KW1QYWNrYWdlTWFwW1Jlc19HRVRQQUNLQUdFKHJlc0lEKSsxXSktMTsKLX0KLQotc3RhdHVzX3QgUmVzVGFibGU6OmFkZChjb25zdCB2b2lkKiBkYXRhLCBzaXplX3Qgc2l6ZSwgdm9pZCogY29va2llLCBib29sIGNvcHlEYXRhKQotewotICAgIHJldHVybiBhZGQoZGF0YSwgc2l6ZSwgY29va2llLCBOVUxMLCBjb3B5RGF0YSk7Ci19Ci0KLXN0YXR1c190IFJlc1RhYmxlOjphZGQoQXNzZXQqIGFzc2V0LCB2b2lkKiBjb29raWUsIGJvb2wgY29weURhdGEpCi17Ci0gICAgY29uc3Qgdm9pZCogZGF0YSA9IGFzc2V0LT5nZXRCdWZmZXIodHJ1ZSk7Ci0gICAgaWYgKGRhdGEgPT0gTlVMTCkgewotICAgICAgICBMT0dXKCJVbmFibGUgdG8gZ2V0IGJ1ZmZlciBvZiByZXNvdXJjZSBhc3NldCBmaWxlIik7Ci0gICAgICAgIHJldHVybiBVTktOT1dOX0VSUk9SOwotICAgIH0KLSAgICBzaXplX3Qgc2l6ZSA9IChzaXplX3QpYXNzZXQtPmdldExlbmd0aCgpOwotICAgIHJldHVybiBhZGQoZGF0YSwgc2l6ZSwgY29va2llLCBhc3NldCwgY29weURhdGEpOwotfQotCi1zdGF0dXNfdCBSZXNUYWJsZTo6YWRkKGNvbnN0IHZvaWQqIGRhdGEsIHNpemVfdCBzaXplLCB2b2lkKiBjb29raWUsCi0gICAgICAgICAgICAgICAgICAgICAgIEFzc2V0KiBhc3NldCwgYm9vbCBjb3B5RGF0YSkKLXsKLSAgICBpZiAoIWRhdGEpIHJldHVybiBOT19FUlJPUjsKLSAgICBIZWFkZXIqIGhlYWRlciA9IG5ldyBIZWFkZXI7Ci0gICAgaGVhZGVyLT5pbmRleCA9IG1IZWFkZXJzLnNpemUoKTsKLSAgICBoZWFkZXItPmNvb2tpZSA9IGNvb2tpZTsKLSAgICBtSGVhZGVycy5hZGQoaGVhZGVyKTsKLQotICAgIGNvbnN0IGJvb2wgbm90RGV2aWNlRW5kaWFuID0gaHRvZHMoMHhmMCkgIT0gMHhmMDsKLQotICAgIExPQURfVEFCTEVfTk9JU1koCi0gICAgICAgIExPR1YoIkFkZGluZyByZXNvdXJjZXMgdG8gUmVzVGFibGU6IGRhdGE9JXAsIHNpemU9MHgleCwgY29va2llPSVwLCBhc3NldD0lcCwgY29weT0lZFxuIiwKLSAgICAgICAgICAgICBkYXRhLCBzaXplLCBjb29raWUsIGFzc2V0LCBjb3B5RGF0YSkpOwotICAgIAotICAgIGlmIChjb3B5RGF0YSB8fCBub3REZXZpY2VFbmRpYW4pIHsKLSAgICAgICAgaGVhZGVyLT5vd25lZERhdGEgPSBtYWxsb2Moc2l6ZSk7Ci0gICAgICAgIGlmIChoZWFkZXItPm93bmVkRGF0YSA9PSBOVUxMKSB7Ci0gICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1OT19NRU1PUlkpOwotICAgICAgICB9Ci0gICAgICAgIG1lbWNweShoZWFkZXItPm93bmVkRGF0YSwgZGF0YSwgc2l6ZSk7Ci0gICAgICAgIGRhdGEgPSBoZWFkZXItPm93bmVkRGF0YTsKLSAgICB9Ci0KLSAgICBoZWFkZXItPmhlYWRlciA9IChjb25zdCBSZXNUYWJsZV9oZWFkZXIqKWRhdGE7Ci0gICAgaGVhZGVyLT5zaXplID0gZHRvaGwoaGVhZGVyLT5oZWFkZXItPmhlYWRlci5zaXplKTsKLSAgICAvL0xPR0koIkdvdCBzaXplIDB4JXgsIGFnYWluIHNpemUgMHgleCwgcmF3IHNpemUgMHgleFxuIiwgaGVhZGVyLT5zaXplLAotICAgIC8vICAgICBkdG9obChoZWFkZXItPmhlYWRlci0+aGVhZGVyLnNpemUpLCBoZWFkZXItPmhlYWRlci0+aGVhZGVyLnNpemUpOwotICAgIExPQURfVEFCTEVfTk9JU1koTE9HVigiTG9hZGluZyBSZXNUYWJsZSBAJXA6XG4iLCBoZWFkZXItPmhlYWRlcikpOwotICAgIExPQURfVEFCTEVfTk9JU1kocHJpbnRIZXhEYXRhKDIsIGhlYWRlci0+aGVhZGVyLCBoZWFkZXItPnNpemUgPCAyNTYgPyBoZWFkZXItPnNpemUgOiAyNTYsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMTYsIDE2LCAwLCBmYWxzZSwgcHJpbnRUb0xvZ0Z1bmMpKTsKLSAgICBpZiAoZHRvaHMoaGVhZGVyLT5oZWFkZXItPmhlYWRlci5oZWFkZXJTaXplKSA+IGhlYWRlci0+c2l6ZQotICAgICAgICAgICAgfHwgaGVhZGVyLT5zaXplID4gc2l6ZSkgewotICAgICAgICBMT0dXKCJCYWQgcmVzb3VyY2UgdGFibGU6IGhlYWRlciBzaXplIDB4JXggb3IgdG90YWwgc2l6ZSAweCV4IGlzIGxhcmdlciB0aGFuIGRhdGEgc2l6ZSAweCV4XG4iLAotICAgICAgICAgICAgIChpbnQpZHRvaHMoaGVhZGVyLT5oZWFkZXItPmhlYWRlci5oZWFkZXJTaXplKSwKLSAgICAgICAgICAgICAoaW50KWhlYWRlci0+c2l6ZSwgKGludClzaXplKTsKLSAgICAgICAgcmV0dXJuIChtRXJyb3I9QkFEX1RZUEUpOwotICAgIH0KLSAgICBpZiAoKChkdG9ocyhoZWFkZXItPmhlYWRlci0+aGVhZGVyLmhlYWRlclNpemUpfGhlYWRlci0+c2l6ZSkmMHgzKSAhPSAwKSB7Ci0gICAgICAgIExPR1coIkJhZCByZXNvdXJjZSB0YWJsZTogaGVhZGVyIHNpemUgMHgleCBvciB0b3RhbCBzaXplIDB4JXggaXMgbm90IG9uIGFuIGludGVnZXIgYm91bmRhcnlcbiIsCi0gICAgICAgICAgICAgKGludClkdG9ocyhoZWFkZXItPmhlYWRlci0+aGVhZGVyLmhlYWRlclNpemUpLAotICAgICAgICAgICAgIChpbnQpaGVhZGVyLT5zaXplKTsKLSAgICAgICAgcmV0dXJuIChtRXJyb3I9QkFEX1RZUEUpOwotICAgIH0KLSAgICBoZWFkZXItPmRhdGFFbmQgPSAoKGNvbnN0IHVpbnQ4X3QqKWhlYWRlci0+aGVhZGVyKSArIGhlYWRlci0+c2l6ZTsKLQotICAgIC8vIEl0ZXJhdGUgdGhyb3VnaCBhbGwgY2h1bmtzLgotICAgIHNpemVfdCBjdXJQYWNrYWdlID0gMDsKLQotICAgIGNvbnN0IFJlc0NodW5rX2hlYWRlciogY2h1bmsgPQotICAgICAgICAoY29uc3QgUmVzQ2h1bmtfaGVhZGVyKikoKChjb25zdCB1aW50OF90KiloZWFkZXItPmhlYWRlcikKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICsgZHRvaHMoaGVhZGVyLT5oZWFkZXItPmhlYWRlci5oZWFkZXJTaXplKSk7Ci0gICAgd2hpbGUgKCgoY29uc3QgdWludDhfdCopY2h1bmspIDw9IChoZWFkZXItPmRhdGFFbmQtc2l6ZW9mKFJlc0NodW5rX2hlYWRlcikpICYmCi0gICAgICAgICAgICgoY29uc3QgdWludDhfdCopY2h1bmspIDw9IChoZWFkZXItPmRhdGFFbmQtZHRvaGwoY2h1bmstPnNpemUpKSkgewotICAgICAgICBzdGF0dXNfdCBlcnIgPSB2YWxpZGF0ZV9jaHVuayhjaHVuaywgc2l6ZW9mKFJlc0NodW5rX2hlYWRlciksIGhlYWRlci0+ZGF0YUVuZCwgIlJlc1RhYmxlIik7Ci0gICAgICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgIHJldHVybiAobUVycm9yPWVycik7Ci0gICAgICAgIH0KLSAgICAgICAgVEFCTEVfTk9JU1koTE9HVigiQ2h1bms6IHR5cGU9MHgleCwgaGVhZGVyU2l6ZT0weCV4LCBzaXplPTB4JXgsIHBvcz0lcFxuIiwKLSAgICAgICAgICAgICAgICAgICAgIGR0b2hzKGNodW5rLT50eXBlKSwgZHRvaHMoY2h1bmstPmhlYWRlclNpemUpLCBkdG9obChjaHVuay0+c2l6ZSksCi0gICAgICAgICAgICAgICAgICAgICAodm9pZCopKCgoY29uc3QgdWludDhfdCopY2h1bmspIC0gKChjb25zdCB1aW50OF90KiloZWFkZXItPmhlYWRlcikpKSk7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBjc2l6ZSA9IGR0b2hsKGNodW5rLT5zaXplKTsKLSAgICAgICAgY29uc3QgdWludDE2X3QgY3R5cGUgPSBkdG9ocyhjaHVuay0+dHlwZSk7Ci0gICAgICAgIGlmIChjdHlwZSA9PSBSRVNfU1RSSU5HX1BPT0xfVFlQRSkgewotICAgICAgICAgICAgaWYgKGhlYWRlci0+dmFsdWVzLmdldEVycm9yKCkgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgICAgICAvLyBPbmx5IHVzZSB0aGUgZmlyc3Qgc3RyaW5nIGNodW5rOyBpZ25vcmUgYW55IG90aGVycyB0aGF0Ci0gICAgICAgICAgICAgICAgLy8gbWF5IGFwcGVhci4KLSAgICAgICAgICAgICAgICBzdGF0dXNfdCBlcnIgPSBoZWFkZXItPnZhbHVlcy5zZXRUbyhjaHVuaywgY3NpemUpOwotICAgICAgICAgICAgICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9ZXJyKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIExPR1coIk11bHRpcGxlIHN0cmluZyBjaHVua3MgZm91bmQgaW4gcmVzb3VyY2UgdGFibGUuIik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSBpZiAoY3R5cGUgPT0gUkVTX1RBQkxFX1BBQ0tBR0VfVFlQRSkgewotICAgICAgICAgICAgaWYgKGN1clBhY2thZ2UgPj0gZHRvaGwoaGVhZGVyLT5oZWFkZXItPnBhY2thZ2VDb3VudCkpIHsKLSAgICAgICAgICAgICAgICBMT0dXKCJNb3JlIHBhY2thZ2UgY2h1bmtzIHdlcmUgZm91bmQgdGhhbiB0aGUgJWQgZGVjbGFyZWQgaW4gdGhlIGhlYWRlci4iLAotICAgICAgICAgICAgICAgICAgICAgZHRvaGwoaGVhZGVyLT5oZWFkZXItPnBhY2thZ2VDb3VudCkpOwotICAgICAgICAgICAgICAgIHJldHVybiAobUVycm9yPUJBRF9UWVBFKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChwYXJzZVBhY2thZ2UoKFJlc1RhYmxlX3BhY2thZ2UqKWNodW5rLCBoZWFkZXIpICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG1FcnJvcjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGN1clBhY2thZ2UrKzsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIExPR1coIlVua25vd24gY2h1bmsgdHlwZSAlcCBpbiB0YWJsZSBhdCAlcC5cbiIsCi0gICAgICAgICAgICAgICAgICh2b2lkKikoaW50KShjdHlwZSksCi0gICAgICAgICAgICAgICAgICh2b2lkKikoKChjb25zdCB1aW50OF90KiljaHVuaykgLSAoKGNvbnN0IHVpbnQ4X3QqKWhlYWRlci0+aGVhZGVyKSkpOwotICAgICAgICB9Ci0gICAgICAgIGNodW5rID0gKGNvbnN0IFJlc0NodW5rX2hlYWRlciopCi0gICAgICAgICAgICAoKChjb25zdCB1aW50OF90KiljaHVuaykgKyBjc2l6ZSk7Ci0gICAgfQotCi0gICAgaWYgKGN1clBhY2thZ2UgPCBkdG9obChoZWFkZXItPmhlYWRlci0+cGFja2FnZUNvdW50KSkgewotICAgICAgICBMT0dXKCJGZXdlciBwYWNrYWdlIGNodW5rcyAoJWQpIHdlcmUgZm91bmQgdGhhbiB0aGUgJWQgZGVjbGFyZWQgaW4gdGhlIGhlYWRlci4iLAotICAgICAgICAgICAgIChpbnQpY3VyUGFja2FnZSwgZHRvaGwoaGVhZGVyLT5oZWFkZXItPnBhY2thZ2VDb3VudCkpOwotICAgICAgICByZXR1cm4gKG1FcnJvcj1CQURfVFlQRSk7Ci0gICAgfQotICAgIG1FcnJvciA9IGhlYWRlci0+dmFsdWVzLmdldEVycm9yKCk7Ci0gICAgaWYgKG1FcnJvciAhPSBOT19FUlJPUikgewotICAgICAgICBMT0dXKCJObyBzdHJpbmcgdmFsdWVzIGZvdW5kIGluIHJlc291cmNlIHRhYmxlISIpOwotICAgIH0KLSAgICBUQUJMRV9OT0lTWShMT0dWKCJSZXR1cm5pbmcgZnJvbSBhZGQgd2l0aCBtRXJyb3I9JWRcbiIsIG1FcnJvcikpOwotICAgIHJldHVybiBtRXJyb3I7Ci19Ci0KLXN0YXR1c190IFJlc1RhYmxlOjpnZXRFcnJvcigpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1FcnJvcjsKLX0KLQotdm9pZCBSZXNUYWJsZTo6dW5pbml0KCkKLXsKLSAgICBtRXJyb3IgPSBOT19JTklUOwotICAgIHNpemVfdCBOID0gbVBhY2thZ2VHcm91cHMuc2l6ZSgpOwotICAgIGZvciAoc2l6ZV90IGk9MDsgaTxOOyBpKyspIHsKLSAgICAgICAgUGFja2FnZUdyb3VwKiBnID0gbVBhY2thZ2VHcm91cHNbaV07Ci0gICAgICAgIGRlbGV0ZSBnOwotICAgIH0KLSAgICBOID0gbUhlYWRlcnMuc2l6ZSgpOwotICAgIGZvciAoc2l6ZV90IGk9MDsgaTxOOyBpKyspIHsKLSAgICAgICAgSGVhZGVyKiBoZWFkZXIgPSBtSGVhZGVyc1tpXTsKLSAgICAgICAgaWYgKGhlYWRlci0+b3duZWREYXRhKSB7Ci0gICAgICAgICAgICBmcmVlKGhlYWRlci0+b3duZWREYXRhKTsKLSAgICAgICAgfQotICAgICAgICBkZWxldGUgaGVhZGVyOwotICAgIH0KLQotICAgIG1QYWNrYWdlR3JvdXBzLmNsZWFyKCk7Ci0gICAgbUhlYWRlcnMuY2xlYXIoKTsKLX0KLQotYm9vbCBSZXNUYWJsZTo6Z2V0UmVzb3VyY2VOYW1lKHVpbnQzMl90IHJlc0lELCByZXNvdXJjZV9uYW1lKiBvdXROYW1lKSBjb25zdAotewotICAgIGlmIChtRXJyb3IgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIGNvbnN0IHNzaXplX3QgcCA9IGdldFJlc291cmNlUGFja2FnZUluZGV4KHJlc0lEKTsKLSAgICBjb25zdCBpbnQgdCA9IFJlc19HRVRUWVBFKHJlc0lEKTsKLSAgICBjb25zdCBpbnQgZSA9IFJlc19HRVRFTlRSWShyZXNJRCk7Ci0KLSAgICBpZiAocCA8IDApIHsKLSAgICAgICAgTE9HVygiTm8gcGFja2FnZSBpZGVudGlmaWVyIHdoZW4gZ2V0dGluZyBuYW1lIGZvciByZXNvdXJjZSBudW1iZXIgMHglMDh4IiwgcmVzSUQpOwotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotICAgIGlmICh0IDwgMCkgewotICAgICAgICBMT0dXKCJObyB0eXBlIGlkZW50aWZpZXIgd2hlbiBnZXR0aW5nIG5hbWUgZm9yIHJlc291cmNlIG51bWJlciAweCUwOHgiLCByZXNJRCk7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBjb25zdCBQYWNrYWdlR3JvdXAqIGNvbnN0IGdycCA9IG1QYWNrYWdlR3JvdXBzW3BdOwotICAgIGlmIChncnAgPT0gTlVMTCkgewotICAgICAgICBMT0dXKCJCYWQgaWRlbnRpZmllciB3aGVuIGdldHRpbmcgbmFtZSBmb3IgcmVzb3VyY2UgbnVtYmVyIDB4JTA4eCIsIHJlc0lEKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSAgICBpZiAoZ3JwLT5wYWNrYWdlcy5zaXplKCkgPiAwKSB7Ci0gICAgICAgIGNvbnN0IFBhY2thZ2UqIGNvbnN0IHBhY2thZ2UgPSBncnAtPnBhY2thZ2VzWzBdOwotCi0gICAgICAgIGNvbnN0IFJlc1RhYmxlX3R5cGUqIHR5cGU7Ci0gICAgICAgIGNvbnN0IFJlc1RhYmxlX2VudHJ5KiBlbnRyeTsKLSAgICAgICAgc3NpemVfdCBvZmZzZXQgPSBnZXRFbnRyeShwYWNrYWdlLCB0LCBlLCBOVUxMLCAmdHlwZSwgJmVudHJ5LCBOVUxMKTsKLSAgICAgICAgaWYgKG9mZnNldCA8PSAwKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBvdXROYW1lLT5wYWNrYWdlID0gZ3JwLT5uYW1lLnN0cmluZygpOwotICAgICAgICBvdXROYW1lLT5wYWNrYWdlTGVuID0gZ3JwLT5uYW1lLnNpemUoKTsKLSAgICAgICAgb3V0TmFtZS0+dHlwZSA9IGdycC0+dHlwZVN0cmluZ3Muc3RyaW5nQXQodCwgJm91dE5hbWUtPnR5cGVMZW4pOwotICAgICAgICBvdXROYW1lLT5uYW1lID0gZ3JwLT5rZXlTdHJpbmdzLnN0cmluZ0F0KAotICAgICAgICAgICAgZHRvaGwoZW50cnktPmtleS5pbmRleCksICZvdXROYW1lLT5uYW1lTGVuKTsKLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgfQotCi0gICAgcmV0dXJuIGZhbHNlOwotfQotCi1zc2l6ZV90IFJlc1RhYmxlOjpnZXRSZXNvdXJjZSh1aW50MzJfdCByZXNJRCwgUmVzX3ZhbHVlKiBvdXRWYWx1ZSwgYm9vbCBtYXlCZUJhZywKLSAgICAgICAgdWludDMyX3QqIG91dFNwZWNGbGFncywgUmVzVGFibGVfY29uZmlnKiBvdXRDb25maWcpIGNvbnN0Ci17Ci0gICAgaWYgKG1FcnJvciAhPSBOT19FUlJPUikgewotICAgICAgICByZXR1cm4gbUVycm9yOwotICAgIH0KLQotICAgIGNvbnN0IHNzaXplX3QgcCA9IGdldFJlc291cmNlUGFja2FnZUluZGV4KHJlc0lEKTsKLSAgICBjb25zdCBpbnQgdCA9IFJlc19HRVRUWVBFKHJlc0lEKTsKLSAgICBjb25zdCBpbnQgZSA9IFJlc19HRVRFTlRSWShyZXNJRCk7Ci0KLSAgICBpZiAocCA8IDApIHsKLSAgICAgICAgTE9HVygiTm8gcGFja2FnZSBpZGVudGlmaWVyIHdoZW4gZ2V0dGluZyB2YWx1ZSBmb3IgcmVzb3VyY2UgbnVtYmVyIDB4JTA4eCIsIHJlc0lEKTsKLSAgICAgICAgcmV0dXJuIEJBRF9JTkRFWDsKLSAgICB9Ci0gICAgaWYgKHQgPCAwKSB7Ci0gICAgICAgIExPR1coIk5vIHR5cGUgaWRlbnRpZmllciB3aGVuIGdldHRpbmcgdmFsdWUgZm9yIHJlc291cmNlIG51bWJlciAweCUwOHgiLCByZXNJRCk7Ci0gICAgICAgIHJldHVybiBCQURfSU5ERVg7Ci0gICAgfQotCi0gICAgY29uc3QgUmVzX3ZhbHVlKiBiZXN0VmFsdWUgPSBOVUxMOwotICAgIGNvbnN0IFBhY2thZ2UqIGJlc3RQYWNrYWdlID0gTlVMTDsKLSAgICBSZXNUYWJsZV9jb25maWcgYmVzdEl0ZW07Ci0gICAgbWVtc2V0KCZiZXN0SXRlbSwgMCwgc2l6ZW9mKGJlc3RJdGVtKSk7IC8vIG1ha2UgdGhlIGNvbXBpbGVyIHNodXQgdXAKLQotICAgIGlmIChvdXRTcGVjRmxhZ3MgIT0gTlVMTCkgKm91dFNwZWNGbGFncyA9IDA7Ci0gICAgCi0gICAgLy8gTG9vayB0aHJvdWdoIGFsbCByZXNvdXJjZSBwYWNrYWdlcywgc3RhcnRpbmcgd2l0aCB0aGUgbW9zdAotICAgIC8vIHJlY2VudGx5IGFkZGVkLgotICAgIGNvbnN0IFBhY2thZ2VHcm91cCogY29uc3QgZ3JwID0gbVBhY2thZ2VHcm91cHNbcF07Ci0gICAgaWYgKGdycCA9PSBOVUxMKSB7Ci0gICAgICAgIExPR1coIkJhZCBpZGVudGlmaWVyIHdoZW4gZ2V0dGluZyB2YWx1ZSBmb3IgcmVzb3VyY2UgbnVtYmVyIDB4JTA4eCIsIHJlc0lEKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSAgICBzaXplX3QgaXAgPSBncnAtPnBhY2thZ2VzLnNpemUoKTsKLSAgICB3aGlsZSAoaXAgPiAwKSB7Ci0gICAgICAgIGlwLS07Ci0KLSAgICAgICAgY29uc3QgUGFja2FnZSogY29uc3QgcGFja2FnZSA9IGdycC0+cGFja2FnZXNbaXBdOwotCi0gICAgICAgIGNvbnN0IFJlc1RhYmxlX3R5cGUqIHR5cGU7Ci0gICAgICAgIGNvbnN0IFJlc1RhYmxlX2VudHJ5KiBlbnRyeTsKLSAgICAgICAgY29uc3QgVHlwZSogdHlwZUNsYXNzOwotICAgICAgICBzc2l6ZV90IG9mZnNldCA9IGdldEVudHJ5KHBhY2thZ2UsIHQsIGUsICZtUGFyYW1zLCAmdHlwZSwgJmVudHJ5LCAmdHlwZUNsYXNzKTsKLSAgICAgICAgaWYgKG9mZnNldCA8PSAwKSB7Ci0gICAgICAgICAgICBpZiAob2Zmc2V0IDwgMCkgewotICAgICAgICAgICAgICAgIExPR1coIkZhaWx1cmUgZ2V0dGluZyBlbnRyeSBmb3IgMHglMDh4ICh0PSVkIGU9JWQpIGluIHBhY2thZ2UgJWQ6IDB4JTA4eFxuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlc0lELCB0LCBlLCAoaW50KWlwLCAoaW50KW9mZnNldCk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG9mZnNldDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChkdG9ocyhlbnRyeS0+ZmxhZ3MpJmVudHJ5LT5GTEFHX0NPTVBMRVgpICE9IDApIHsKLSAgICAgICAgICAgIGlmICghbWF5QmVCYWcpIHsKLSAgICAgICAgICAgICAgICBMT0dXKCJSZXF1ZXN0aW5nIHJlc291cmNlICVwIGZhaWxlZCBiZWNhdXNlIGl0IGlzIGNvbXBsZXhcbiIsCi0gICAgICAgICAgICAgICAgICAgICAodm9pZCopcmVzSUQpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgIH0KLQotICAgICAgICBUQUJMRV9OT0lTWShhb3V0IDw8ICJSZXNvdXJjZSB0eXBlIGRhdGE6ICIKLSAgICAgICAgICAgICAgPDwgSGV4RHVtcCh0eXBlLCBkdG9obCh0eXBlLT5oZWFkZXIuc2l6ZSkpIDw8IGVuZGwpOwotICAgICAgICAKLSAgICAgICAgaWYgKChzaXplX3Qpb2Zmc2V0ID4gKGR0b2hsKHR5cGUtPmhlYWRlci5zaXplKS1zaXplb2YoUmVzX3ZhbHVlKSkpIHsKLSAgICAgICAgICAgIExPR1coIlJlc1RhYmxlX2l0ZW0gYXQgJWQgaXMgYmV5b25kIHR5cGUgY2h1bmsgZGF0YSAlZCIsCi0gICAgICAgICAgICAgICAgIChpbnQpb2Zmc2V0LCBkdG9obCh0eXBlLT5oZWFkZXIuc2l6ZSkpOwotICAgICAgICAgICAgcmV0dXJuIEJBRF9UWVBFOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBjb25zdCBSZXNfdmFsdWUqIGl0ZW0gPQotICAgICAgICAgICAgKGNvbnN0IFJlc192YWx1ZSopKCgoY29uc3QgdWludDhfdCopdHlwZSkgKyBvZmZzZXQpOwotICAgICAgICBSZXNUYWJsZV9jb25maWcgdGhpc0NvbmZpZzsKLSAgICAgICAgdGhpc0NvbmZpZy5jb3B5RnJvbUR0b0godHlwZS0+Y29uZmlnKTsKLQotICAgICAgICBpZiAob3V0U3BlY0ZsYWdzICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGlmICh0eXBlQ2xhc3MtPnR5cGVTcGVjRmxhZ3MgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICpvdXRTcGVjRmxhZ3MgfD0gZHRvaGwodHlwZUNsYXNzLT50eXBlU3BlY0ZsYWdzW2VdKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgKm91dFNwZWNGbGFncyA9IC0xOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBpZiAoYmVzdFBhY2thZ2UgIT0gTlVMTCAmJiBiZXN0SXRlbS5pc0JldHRlclRoYW4odGhpc0NvbmZpZykpIHsKLSAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBiZXN0SXRlbSA9IHRoaXNDb25maWc7Ci0gICAgICAgIGJlc3RWYWx1ZSA9IGl0ZW07Ci0gICAgICAgIGJlc3RQYWNrYWdlID0gcGFja2FnZTsKLSAgICB9Ci0KLSAgICBUQUJMRV9OT0lTWShwcmludGYoIkZvdW5kIHJlc3VsdDogcGFja2FnZSAlcFxuIiwgYmVzdFBhY2thZ2UpKTsKLQotICAgIGlmIChiZXN0VmFsdWUpIHsKLSAgICAgICAgb3V0VmFsdWUtPnNpemUgPSBkdG9ocyhiZXN0VmFsdWUtPnNpemUpOwotICAgICAgICBvdXRWYWx1ZS0+cmVzMCA9IGJlc3RWYWx1ZS0+cmVzMDsKLSAgICAgICAgb3V0VmFsdWUtPmRhdGFUeXBlID0gYmVzdFZhbHVlLT5kYXRhVHlwZTsKLSAgICAgICAgb3V0VmFsdWUtPmRhdGEgPSBkdG9obChiZXN0VmFsdWUtPmRhdGEpOwotICAgICAgICBpZiAob3V0Q29uZmlnICE9IE5VTEwpIHsKLSAgICAgICAgICAgICpvdXRDb25maWcgPSBiZXN0SXRlbTsKLSAgICAgICAgfQotICAgICAgICBUQUJMRV9OT0lTWShzaXplX3QgbGVuOwotICAgICAgICAgICAgICBwcmludGYoIkZvdW5kIHZhbHVlOiBwa2c9JWQsIHR5cGU9JWQsIHN0cj0lcywgaW50PSVkXG4iLAotICAgICAgICAgICAgICAgICAgICAgYmVzdFBhY2thZ2UtPmhlYWRlci0+aW5kZXgsCi0gICAgICAgICAgICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUsCi0gICAgICAgICAgICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUgPT0gYmVzdFZhbHVlLT5UWVBFX1NUUklORwotICAgICAgICAgICAgICAgICAgICAgPyBTdHJpbmc4KGJlc3RQYWNrYWdlLT5oZWFkZXItPnZhbHVlcy5zdHJpbmdBdCgKLSAgICAgICAgICAgICAgICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YSwgJmxlbikpLnN0cmluZygpCi0gICAgICAgICAgICAgICAgICAgICA6ICIiLAotICAgICAgICAgICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEpKTsKLSAgICAgICAgcmV0dXJuIGJlc3RQYWNrYWdlLT5oZWFkZXItPmluZGV4OwotICAgIH0KLQotICAgIHJldHVybiBCQURfSU5ERVg7Ci19Ci0KLXNzaXplX3QgUmVzVGFibGU6OnJlc29sdmVSZWZlcmVuY2UoUmVzX3ZhbHVlKiB2YWx1ZSwgc3NpemVfdCBibG9ja0luZGV4LAotICAgICAgICB1aW50MzJfdCogb3V0TGFzdFJlZiwgdWludDMyX3QqIGlub3V0VHlwZVNwZWNGbGFncykgY29uc3QKLXsKLSAgICBpbnQgY291bnQ9MDsKLSAgICB3aGlsZSAoYmxvY2tJbmRleCA+PSAwICYmIHZhbHVlLT5kYXRhVHlwZSA9PSB2YWx1ZS0+VFlQRV9SRUZFUkVOQ0UKLSAgICAgICAgICAgJiYgdmFsdWUtPmRhdGEgIT0gMCAmJiBjb3VudCA8IDIwKSB7Ci0gICAgICAgIGlmIChvdXRMYXN0UmVmKSAqb3V0TGFzdFJlZiA9IHZhbHVlLT5kYXRhOwotICAgICAgICB1aW50MzJfdCBsYXN0UmVmID0gdmFsdWUtPmRhdGE7Ci0gICAgICAgIHVpbnQzMl90IG5ld0ZsYWdzID0gMDsKLSAgICAgICAgY29uc3Qgc3NpemVfdCBuZXdJbmRleCA9IGdldFJlc291cmNlKHZhbHVlLT5kYXRhLCB2YWx1ZSwgdHJ1ZSwgJm5ld0ZsYWdzKTsKLSAgICAgICAgLy9MT0dJKCJSZXNvbHZpbmcgcmVmZXJlbmNlIGQ9JXA6IG5ld0luZGV4PSVkLCB0PTB4JTAyeCwgZD0lcFxuIiwKLSAgICAgICAgLy8gICAgICh2b2lkKilsYXN0UmVmLCAoaW50KW5ld0luZGV4LCAoaW50KXZhbHVlLT5kYXRhVHlwZSwgKHZvaWQqKXZhbHVlLT5kYXRhKTsKLSAgICAgICAgLy9wcmludGYoIkdldHRpbmcgcmVmZXJlbmNlIDB4JTA4eDogbmV3SW5kZXg9JWRcbiIsIHZhbHVlLT5kYXRhLCBuZXdJbmRleCk7Ci0gICAgICAgIGlmIChpbm91dFR5cGVTcGVjRmxhZ3MgIT0gTlVMTCkgKmlub3V0VHlwZVNwZWNGbGFncyB8PSBuZXdGbGFnczsKLSAgICAgICAgaWYgKG5ld0luZGV4IDwgMCkgewotICAgICAgICAgICAgLy8gVGhpcyBjYW4gZmFpbCBpZiB0aGUgcmVzb3VyY2UgYmVpbmcgcmVmZXJlbmNlZCBpcyBhIHN0eWxlLi4uCi0gICAgICAgICAgICAvLyBpbiB0aGlzIGNhc2UsIGp1c3QgcmV0dXJuIHRoZSByZWZlcmVuY2UsIGFuZCBleHBlY3QgdGhlCi0gICAgICAgICAgICAvLyBjYWxsZXIgdG8gZGVhbCB3aXRoLgotICAgICAgICAgICAgcmV0dXJuIGJsb2NrSW5kZXg7Ci0gICAgICAgIH0KLSAgICAgICAgYmxvY2tJbmRleCA9IG5ld0luZGV4OwotICAgICAgICBjb3VudCsrOwotICAgIH0KLSAgICByZXR1cm4gYmxvY2tJbmRleDsKLX0KLQotY29uc3QgY2hhcjE2X3QqIFJlc1RhYmxlOjp2YWx1ZVRvU3RyaW5nKAotICAgIGNvbnN0IFJlc192YWx1ZSogdmFsdWUsIHNpemVfdCBzdHJpbmdCbG9jaywKLSAgICBjaGFyMTZfdCB0bXBCdWZmZXJbVE1QX0JVRkZFUl9TSVpFXSwgc2l6ZV90KiBvdXRMZW4pCi17Ci0gICAgaWYgKCF2YWx1ZSkgewotICAgICAgICByZXR1cm4gTlVMTDsKLSAgICB9Ci0gICAgaWYgKHZhbHVlLT5kYXRhVHlwZSA9PSB2YWx1ZS0+VFlQRV9TVFJJTkcpIHsKLSAgICAgICAgcmV0dXJuIGdldFRhYmxlU3RyaW5nQmxvY2soc3RyaW5nQmxvY2spLT5zdHJpbmdBdCh2YWx1ZS0+ZGF0YSwgb3V0TGVuKTsKLSAgICB9Ci0gICAgLy8gWFhYIGRvIGludCB0byBzdHJpbmcgY29udmVyc2lvbnMuCi0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLXNzaXplX3QgUmVzVGFibGU6OmxvY2tCYWcodWludDMyX3QgcmVzSUQsIGNvbnN0IGJhZ19lbnRyeSoqIG91dEJhZykgY29uc3QKLXsKLSAgICBtTG9jay5sb2NrKCk7Ci0gICAgc3NpemVfdCBlcnIgPSBnZXRCYWdMb2NrZWQocmVzSUQsIG91dEJhZyk7Ci0gICAgaWYgKGVyciA8IE5PX0VSUk9SKSB7Ci0gICAgICAgIC8vcHJpbnRmKCIqKiogZ2V0IGZhaWxlZCEgIHVubG9ja2luZ1xuIik7Ci0gICAgICAgIG1Mb2NrLnVubG9jaygpOwotICAgIH0KLSAgICByZXR1cm4gZXJyOwotfQotCi12b2lkIFJlc1RhYmxlOjp1bmxvY2tCYWcoY29uc3QgYmFnX2VudHJ5KiBiYWcpIGNvbnN0Ci17Ci0gICAgLy9wcmludGYoIjw8PCB1bmxvY2tCYWcgJXBcbiIsIHRoaXMpOwotICAgIG1Mb2NrLnVubG9jaygpOwotfQotCi12b2lkIFJlc1RhYmxlOjpsb2NrKCkgY29uc3QKLXsKLSAgICBtTG9jay5sb2NrKCk7Ci19Ci0KLXZvaWQgUmVzVGFibGU6OnVubG9jaygpIGNvbnN0Ci17Ci0gICAgbUxvY2sudW5sb2NrKCk7Ci19Ci0KLXNzaXplX3QgUmVzVGFibGU6OmdldEJhZ0xvY2tlZCh1aW50MzJfdCByZXNJRCwgY29uc3QgYmFnX2VudHJ5Kiogb3V0QmFnLAotICAgICAgICB1aW50MzJfdCogb3V0VHlwZVNwZWNGbGFncykgY29uc3QKLXsKLSAgICBpZiAobUVycm9yICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgIHJldHVybiBtRXJyb3I7Ci0gICAgfQotCi0gICAgY29uc3Qgc3NpemVfdCBwID0gZ2V0UmVzb3VyY2VQYWNrYWdlSW5kZXgocmVzSUQpOwotICAgIGNvbnN0IGludCB0ID0gUmVzX0dFVFRZUEUocmVzSUQpOwotICAgIGNvbnN0IGludCBlID0gUmVzX0dFVEVOVFJZKHJlc0lEKTsKLQotICAgIGlmIChwIDwgMCkgewotICAgICAgICBMT0dXKCJJbnZhbGlkIHBhY2thZ2UgaWRlbnRpZmllciB3aGVuIGdldHRpbmcgYmFnIGZvciByZXNvdXJjZSBudW1iZXIgMHglMDh4IiwgcmVzSUQpOwotICAgICAgICByZXR1cm4gQkFEX0lOREVYOwotICAgIH0KLSAgICBpZiAodCA8IDApIHsKLSAgICAgICAgTE9HVygiTm8gdHlwZSBpZGVudGlmaWVyIHdoZW4gZ2V0dGluZyBiYWcgZm9yIHJlc291cmNlIG51bWJlciAweCUwOHgiLCByZXNJRCk7Ci0gICAgICAgIHJldHVybiBCQURfSU5ERVg7Ci0gICAgfQotCi0gICAgLy9wcmludGYoIkdldCBiYWc6IGlkPTB4JTA4eCwgcD0lZCwgdD0lZFxuIiwgcmVzSUQsIHAsIHQpOwotICAgIFBhY2thZ2VHcm91cCogY29uc3QgZ3JwID0gbVBhY2thZ2VHcm91cHNbcF07Ci0gICAgaWYgKGdycCA9PSBOVUxMKSB7Ci0gICAgICAgIExPR1coIkJhZCBpZGVudGlmaWVyIHdoZW4gZ2V0dGluZyBiYWcgZm9yIHJlc291cmNlIG51bWJlciAweCUwOHgiLCByZXNJRCk7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBpZiAodCA+PSAoaW50KWdycC0+dHlwZUNvdW50KSB7Ci0gICAgICAgIExPR1coIlR5cGUgaWRlbnRpZmllciAweCV4IGlzIGxhcmdlciB0aGFuIHR5cGUgY291bnQgMHgleCIsCi0gICAgICAgICAgICAgdCsxLCAoaW50KWdycC0+dHlwZUNvdW50KTsKLSAgICAgICAgcmV0dXJuIEJBRF9JTkRFWDsKLSAgICB9Ci0KLSAgICBjb25zdCBQYWNrYWdlKiBjb25zdCBiYXNlUGFja2FnZSA9IGdycC0+cGFja2FnZXNbMF07Ci0KLSAgICBjb25zdCBUeXBlKiBjb25zdCB0eXBlQ29uZmlncyA9IGJhc2VQYWNrYWdlLT5nZXRUeXBlKHQpOwotCi0gICAgY29uc3Qgc2l6ZV90IE5FTlRSWSA9IHR5cGVDb25maWdzLT5lbnRyeUNvdW50OwotICAgIGlmIChlID49IChpbnQpTkVOVFJZKSB7Ci0gICAgICAgIExPR1coIkVudHJ5IGlkZW50aWZpZXIgMHgleCBpcyBsYXJnZXIgdGhhbiBlbnRyeSBjb3VudCAweCV4IiwKLSAgICAgICAgICAgICBlLCAoaW50KXR5cGVDb25maWdzLT5lbnRyeUNvdW50KTsKLSAgICAgICAgcmV0dXJuIEJBRF9JTkRFWDsKLSAgICB9Ci0KLSAgICAvLyBGaXJzdCBzZWUgaWYgd2UndmUgYWxyZWFkeSBjb21wdXRlZCB0aGlzIGJhZy4uLgotICAgIGlmIChncnAtPmJhZ3MpIHsKLSAgICAgICAgYmFnX3NldCoqIHR5cGVTZXQgPSBncnAtPmJhZ3NbdF07Ci0gICAgICAgIGlmICh0eXBlU2V0KSB7Ci0gICAgICAgICAgICBiYWdfc2V0KiBzZXQgPSB0eXBlU2V0W2VdOwotICAgICAgICAgICAgaWYgKHNldCkgewotICAgICAgICAgICAgICAgIGlmIChzZXQgIT0gKGJhZ19zZXQqKTB4RkZGRkZGRkYpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKG91dFR5cGVTcGVjRmxhZ3MgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICAgICAgKm91dFR5cGVTcGVjRmxhZ3MgPSBzZXQtPnR5cGVTcGVjRmxhZ3M7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgKm91dEJhZyA9IChiYWdfZW50cnkqKShzZXQrMSk7Ci0gICAgICAgICAgICAgICAgICAgIC8vTE9HSSgiRm91bmQgZXhpc3RpbmcgYmFnIGZvcjogJXBcbiIsICh2b2lkKilyZXNJRCk7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBzZXQtPm51bUF0dHJzOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBMT0dXKCJBdHRlbXB0IHRvIHJldHJpZXZlIGJhZyAweCUwOHggd2hpY2ggaXMgaW52YWxpZCBvciBpbiBhIGN5Y2xlLiIsCi0gICAgICAgICAgICAgICAgICAgICByZXNJRCk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIEJBRF9JTkRFWDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vIEJhZyBub3QgZm91bmQsIHdlIG5lZWQgdG8gY29tcHV0ZSBpdCEKLSAgICBpZiAoIWdycC0+YmFncykgewotICAgICAgICBncnAtPmJhZ3MgPSAoYmFnX3NldCoqKiltYWxsb2Moc2l6ZW9mKGJhZ19zZXQqKSpncnAtPnR5cGVDb3VudCk7Ci0gICAgICAgIGlmICghZ3JwLT5iYWdzKSByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICBtZW1zZXQoZ3JwLT5iYWdzLCAwLCBzaXplb2YoYmFnX3NldCopKmdycC0+dHlwZUNvdW50KTsKLSAgICB9Ci0KLSAgICBiYWdfc2V0KiogdHlwZVNldCA9IGdycC0+YmFnc1t0XTsKLSAgICBpZiAoIXR5cGVTZXQpIHsKLSAgICAgICAgdHlwZVNldCA9IChiYWdfc2V0KiopbWFsbG9jKHNpemVvZihiYWdfc2V0KikqTkVOVFJZKTsKLSAgICAgICAgaWYgKCF0eXBlU2V0KSByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICBtZW1zZXQodHlwZVNldCwgMCwgc2l6ZW9mKGJhZ19zZXQqKSpORU5UUlkpOwotICAgICAgICBncnAtPmJhZ3NbdF0gPSB0eXBlU2V0OwotICAgIH0KLQotICAgIC8vIE1hcmsgdGhhdCB3ZSBhcmUgY3VycmVudGx5IHdvcmtpbmcgb24gdGhpcyBvbmUuCi0gICAgdHlwZVNldFtlXSA9IChiYWdfc2V0KikweEZGRkZGRkZGOwotCi0gICAgLy8gVGhpcyBpcyB3aGF0IHdlIGFyZSBidWlsZGluZy4KLSAgICBiYWdfc2V0KiBzZXQgPSBOVUxMOwotCi0gICAgVEFCTEVfTk9JU1koTE9HSSgiQnVpbGRpbmcgYmFnOiAlcFxuIiwgKHZvaWQqKXJlc0lEKSk7Ci0gICAgCi0gICAgLy8gTm93IGNvbGxlY3QgYWxsIGJhZyBhdHRyaWJ1dGVzIGZyb20gYWxsIHBhY2thZ2VzLgotICAgIHNpemVfdCBpcCA9IGdycC0+cGFja2FnZXMuc2l6ZSgpOwotICAgIHdoaWxlIChpcCA+IDApIHsKLSAgICAgICAgaXAtLTsKLQotICAgICAgICBjb25zdCBQYWNrYWdlKiBjb25zdCBwYWNrYWdlID0gZ3JwLT5wYWNrYWdlc1tpcF07Ci0KLSAgICAgICAgY29uc3QgUmVzVGFibGVfdHlwZSogdHlwZTsKLSAgICAgICAgY29uc3QgUmVzVGFibGVfZW50cnkqIGVudHJ5OwotICAgICAgICBjb25zdCBUeXBlKiB0eXBlQ2xhc3M7Ci0gICAgICAgIExPR1YoIkdldHRpbmcgZW50cnkgcGtnPSVwLCB0PSVkLCBlPSVkXG4iLCBwYWNrYWdlLCB0LCBlKTsKLSAgICAgICAgc3NpemVfdCBvZmZzZXQgPSBnZXRFbnRyeShwYWNrYWdlLCB0LCBlLCAmbVBhcmFtcywgJnR5cGUsICZlbnRyeSwgJnR5cGVDbGFzcyk7Ci0gICAgICAgIExPR1YoIlJlc3VsdGluZyBvZmZzZXQ9JWRcbiIsIG9mZnNldCk7Ci0gICAgICAgIGlmIChvZmZzZXQgPD0gMCkgewotICAgICAgICAgICAgaWYgKG9mZnNldCA8IDApIHsKLSAgICAgICAgICAgICAgICBpZiAoc2V0KSBmcmVlKHNldCk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIG9mZnNldDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKChkdG9ocyhlbnRyeS0+ZmxhZ3MpJmVudHJ5LT5GTEFHX0NPTVBMRVgpID09IDApIHsKLSAgICAgICAgICAgIExPR1coIlNraXBwaW5nIGVudHJ5ICVwIGluIHBhY2thZ2UgdGFibGUgJWQgYmVjYXVzZSBpdCBpcyBub3QgY29tcGxleCFcbiIsCi0gICAgICAgICAgICAgICAgICh2b2lkKilyZXNJRCwgKGludClpcCk7Ci0gICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGNvbnN0IHVpbnQxNl90IGVudHJ5U2l6ZSA9IGR0b2hzKGVudHJ5LT5zaXplKTsKLSAgICAgICAgY29uc3QgdWludDMyX3QgcGFyZW50ID0gZW50cnlTaXplID49IHNpemVvZihSZXNUYWJsZV9tYXBfZW50cnkpCi0gICAgICAgICAgICA/IGR0b2hsKCgoY29uc3QgUmVzVGFibGVfbWFwX2VudHJ5KillbnRyeSktPnBhcmVudC5pZGVudCkgOiAwOwotICAgICAgICBjb25zdCB1aW50MzJfdCBjb3VudCA9IGVudHJ5U2l6ZSA+PSBzaXplb2YoUmVzVGFibGVfbWFwX2VudHJ5KQotICAgICAgICAgICAgPyBkdG9obCgoKGNvbnN0IFJlc1RhYmxlX21hcF9lbnRyeSopZW50cnkpLT5jb3VudCkgOiAwOwotICAgICAgICAKLSAgICAgICAgc2l6ZV90IE4gPSBjb3VudDsKLQotICAgICAgICBUQUJMRV9OT0lTWShMT0dJKCJGb3VuZCBtYXA6IHNpemU9JXAgcGFyZW50PSVwIGNvdW50PSVkXG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgIGVudHJ5U2l6ZSwgcGFyZW50LCBjb3VudCkpOwotCi0gICAgICAgIGlmIChzZXQgPT0gTlVMTCkgewotICAgICAgICAgICAgLy8gSWYgdGhpcyBtYXAgaW5oZXJpdHMgZnJvbSBhbm90aGVyLCB3ZSBuZWVkIHRvIHN0YXJ0Ci0gICAgICAgICAgICAvLyB3aXRoIGl0cyBwYXJlbnQncyB2YWx1ZXMuICBPdGhlcndpc2Ugc3RhcnQgb3V0IGVtcHR5LgotICAgICAgICAgICAgVEFCTEVfTk9JU1kocHJpbnRmKCJDcmVhdGluZyBuZXcgYmFnLCBlbnRyeVNpemU9MHglMDh4LCBwYXJlbnQ9MHglMDh4XG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgIGVudHJ5U2l6ZSwgcGFyZW50KSk7Ci0gICAgICAgICAgICBpZiAocGFyZW50KSB7Ci0gICAgICAgICAgICAgICAgY29uc3QgYmFnX2VudHJ5KiBwYXJlbnRCYWc7Ci0gICAgICAgICAgICAgICAgdWludDMyX3QgcGFyZW50VHlwZVNwZWNGbGFncyA9IDA7Ci0gICAgICAgICAgICAgICAgY29uc3Qgc3NpemVfdCBOUCA9IGdldEJhZ0xvY2tlZChwYXJlbnQsICZwYXJlbnRCYWcsICZwYXJlbnRUeXBlU3BlY0ZsYWdzKTsKLSAgICAgICAgICAgICAgICBjb25zdCBzaXplX3QgTlQgPSAoKE5QID49IDApID8gTlAgOiAwKSArIE47Ci0gICAgICAgICAgICAgICAgc2V0ID0gKGJhZ19zZXQqKW1hbGxvYyhzaXplb2YoYmFnX3NldCkrc2l6ZW9mKGJhZ19lbnRyeSkqTlQpOwotICAgICAgICAgICAgICAgIGlmIChzZXQgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAoTlAgPiAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIG1lbWNweShzZXQrMSwgcGFyZW50QmFnLCBOUCpzaXplb2YoYmFnX2VudHJ5KSk7Ci0gICAgICAgICAgICAgICAgICAgIHNldC0+bnVtQXR0cnMgPSBOUDsKLSAgICAgICAgICAgICAgICAgICAgVEFCTEVfTk9JU1koTE9HSSgiSW5pdGlhbGl6ZWQgbmV3IGJhZyB3aXRoICVkIGluaGVyaXRlZCBhdHRyaWJ1dGVzLlxuIiwgTlApKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBUQUJMRV9OT0lTWShMT0dJKCJJbml0aWFsaXplZCBuZXcgYmFnIHdpdGggbm8gaW5oZXJpdGVkIGF0dHJpYnV0ZXMuXG4iKSk7Ci0gICAgICAgICAgICAgICAgICAgIHNldC0+bnVtQXR0cnMgPSAwOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBzZXQtPmF2YWlsQXR0cnMgPSBOVDsKLSAgICAgICAgICAgICAgICBzZXQtPnR5cGVTcGVjRmxhZ3MgPSBwYXJlbnRUeXBlU3BlY0ZsYWdzOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBzZXQgPSAoYmFnX3NldCopbWFsbG9jKHNpemVvZihiYWdfc2V0KStzaXplb2YoYmFnX2VudHJ5KSpOKTsKLSAgICAgICAgICAgICAgICBpZiAoc2V0ID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgc2V0LT5udW1BdHRycyA9IDA7Ci0gICAgICAgICAgICAgICAgc2V0LT5hdmFpbEF0dHJzID0gTjsKLSAgICAgICAgICAgICAgICBzZXQtPnR5cGVTcGVjRmxhZ3MgPSAwOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKHR5cGVDbGFzcy0+dHlwZVNwZWNGbGFncyAhPSBOVUxMKSB7Ci0gICAgICAgICAgICBzZXQtPnR5cGVTcGVjRmxhZ3MgfD0gZHRvaGwodHlwZUNsYXNzLT50eXBlU3BlY0ZsYWdzW2VdKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHNldC0+dHlwZVNwZWNGbGFncyA9IC0xOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICAvLyBOb3cgbWVyZ2UgaW4gdGhlIG5ldyBhdHRyaWJ1dGVzLi4uCi0gICAgICAgIHNzaXplX3QgY3VyT2ZmID0gb2Zmc2V0OwotICAgICAgICBjb25zdCBSZXNUYWJsZV9tYXAqIG1hcDsKLSAgICAgICAgYmFnX2VudHJ5KiBlbnRyaWVzID0gKGJhZ19lbnRyeSopKHNldCsxKTsKLSAgICAgICAgc2l6ZV90IGN1ckVudHJ5ID0gMDsKLSAgICAgICAgdWludDMyX3QgcG9zID0gMDsKLSAgICAgICAgVEFCTEVfTk9JU1koTE9HSSgiU3RhcnRpbmcgd2l0aCBzZXQgJXAsIGVudHJpZXM9JXAsIGF2YWlsPSVkXG4iLAotICAgICAgICAgICAgICAgICAgICAgc2V0LCBlbnRyaWVzLCBzZXQtPmF2YWlsQXR0cnMpKTsKLSAgICAgICAgd2hpbGUgKHBvcyA8IGNvdW50KSB7Ci0gICAgICAgICAgICBUQUJMRV9OT0lTWShwcmludGYoIk5vdyBhdCAlcFxuIiwgKHZvaWQqKWN1ck9mZikpOwotCi0gICAgICAgICAgICBpZiAoKHNpemVfdCljdXJPZmYgPiAoZHRvaGwodHlwZS0+aGVhZGVyLnNpemUpLXNpemVvZihSZXNUYWJsZV9tYXApKSkgewotICAgICAgICAgICAgICAgIExPR1coIlJlc1RhYmxlX21hcCBhdCAlZCBpcyBiZXlvbmQgdHlwZSBjaHVuayBkYXRhICVkIiwKLSAgICAgICAgICAgICAgICAgICAgIChpbnQpY3VyT2ZmLCBkdG9obCh0eXBlLT5oZWFkZXIuc2l6ZSkpOwotICAgICAgICAgICAgICAgIHJldHVybiBCQURfVFlQRTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG1hcCA9IChjb25zdCBSZXNUYWJsZV9tYXAqKSgoKGNvbnN0IHVpbnQ4X3QqKXR5cGUpICsgY3VyT2ZmKTsKLSAgICAgICAgICAgIE4rKzsKLQotICAgICAgICAgICAgY29uc3QgdWludDMyX3QgbmV3TmFtZSA9IGh0b2RsKG1hcC0+bmFtZS5pZGVudCk7Ci0gICAgICAgICAgICBib29sIGlzSW5zaWRlOwotICAgICAgICAgICAgdWludDMyX3Qgb2xkTmFtZSA9IDA7Ci0gICAgICAgICAgICB3aGlsZSAoKGlzSW5zaWRlPShjdXJFbnRyeSA8IHNldC0+bnVtQXR0cnMpKQotICAgICAgICAgICAgICAgICAgICAmJiAob2xkTmFtZT1lbnRyaWVzW2N1ckVudHJ5XS5tYXAubmFtZS5pZGVudCkgPCBuZXdOYW1lKSB7Ci0gICAgICAgICAgICAgICAgVEFCTEVfTk9JU1kocHJpbnRmKCIjJWQ6IEtlZXBpbmcgZXhpc3RpbmcgYXR0cmlidXRlOiAweCUwOHhcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGN1ckVudHJ5LCBlbnRyaWVzW2N1ckVudHJ5XS5tYXAubmFtZS5pZGVudCkpOwotICAgICAgICAgICAgICAgIGN1ckVudHJ5Kys7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIGlmICgoIWlzSW5zaWRlKSB8fCBvbGROYW1lICE9IG5ld05hbWUpIHsKLSAgICAgICAgICAgICAgICAvLyBUaGlzIGlzIGEgbmV3IGF0dHJpYnV0ZS4uLiAgZmlndXJlIG91dCB3aGF0IHRvIGRvIHdpdGggaXQuCi0gICAgICAgICAgICAgICAgaWYgKHNldC0+bnVtQXR0cnMgPj0gc2V0LT5hdmFpbEF0dHJzKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIE5lZWQgdG8gYWxsb2MgbW9yZSBtZW1vcnkuLi4KLSAgICAgICAgICAgICAgICAgICAgY29uc3Qgc2l6ZV90IG5ld0F2YWlsID0gc2V0LT5hdmFpbEF0dHJzK047Ci0gICAgICAgICAgICAgICAgICAgIHNldCA9IChiYWdfc2V0KilyZWFsbG9jKHNldCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGJhZ19zZXQpCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICsgc2l6ZW9mKGJhZ19lbnRyeSkqbmV3QXZhaWwpOwotICAgICAgICAgICAgICAgICAgICBpZiAoc2V0ID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgc2V0LT5hdmFpbEF0dHJzID0gbmV3QXZhaWw7Ci0gICAgICAgICAgICAgICAgICAgIGVudHJpZXMgPSAoYmFnX2VudHJ5Kikoc2V0KzEpOwotICAgICAgICAgICAgICAgICAgICBUQUJMRV9OT0lTWShwcmludGYoIlJlYWxsb2NhdGVkIHNldCAlcCwgZW50cmllcz0lcCwgYXZhaWw9JWRcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzZXQsIGVudHJpZXMsIHNldC0+YXZhaWxBdHRycykpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAoaXNJbnNpZGUpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gR29pbmcgaW4gdGhlIG1pZGRsZSwgbmVlZCB0byBtYWtlIHNwYWNlLgotICAgICAgICAgICAgICAgICAgICBtZW1tb3ZlKGVudHJpZXMrY3VyRW50cnkrMSwgZW50cmllcytjdXJFbnRyeSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YoYmFnX2VudHJ5KSooc2V0LT5udW1BdHRycy1jdXJFbnRyeSkpOwotICAgICAgICAgICAgICAgICAgICBzZXQtPm51bUF0dHJzKys7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIFRBQkxFX05PSVNZKHByaW50ZigiIyVkOiBJbnNlcnRpbmcgbmV3IGF0dHJpYnV0ZTogMHglMDh4XG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjdXJFbnRyeSwgbmV3TmFtZSkpOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBUQUJMRV9OT0lTWShwcmludGYoIiMlZDogUmVwbGFjaW5nIGV4aXN0aW5nIGF0dHJpYnV0ZTogMHglMDh4XG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjdXJFbnRyeSwgb2xkTmFtZSkpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBiYWdfZW50cnkqIGN1ciA9IGVudHJpZXMrY3VyRW50cnk7Ci0KLSAgICAgICAgICAgIGN1ci0+c3RyaW5nQmxvY2sgPSBwYWNrYWdlLT5oZWFkZXItPmluZGV4OwotICAgICAgICAgICAgY3VyLT5tYXAubmFtZS5pZGVudCA9IG5ld05hbWU7Ci0gICAgICAgICAgICBjdXItPm1hcC52YWx1ZS5jb3B5RnJvbV9kdG9oKG1hcC0+dmFsdWUpOwotICAgICAgICAgICAgVEFCTEVfTk9JU1kocHJpbnRmKCJTZXR0aW5nIGVudHJ5ICMlZCAlcDogYmxvY2s9JWQsIG5hbWU9MHglMDh4LCB0eXBlPSVkLCBkYXRhPTB4JTA4eFxuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICBjdXJFbnRyeSwgY3VyLCBjdXItPnN0cmluZ0Jsb2NrLCBjdXItPm1hcC5uYW1lLmlkZW50LAotICAgICAgICAgICAgICAgICAgICAgICAgIGN1ci0+bWFwLnZhbHVlLmRhdGFUeXBlLCBjdXItPm1hcC52YWx1ZS5kYXRhKSk7Ci0KLSAgICAgICAgICAgIC8vIE9uIHRvIHRoZSBuZXh0IQotICAgICAgICAgICAgY3VyRW50cnkrKzsKLSAgICAgICAgICAgIHBvcysrOwotICAgICAgICAgICAgY29uc3Qgc2l6ZV90IHNpemUgPSBkdG9ocyhtYXAtPnZhbHVlLnNpemUpOwotICAgICAgICAgICAgY3VyT2ZmICs9IHNpemUgKyBzaXplb2YoKm1hcCktc2l6ZW9mKG1hcC0+dmFsdWUpOwotICAgICAgICB9OwotICAgICAgICBpZiAoY3VyRW50cnkgPiBzZXQtPm51bUF0dHJzKSB7Ci0gICAgICAgICAgICBzZXQtPm51bUF0dHJzID0gY3VyRW50cnk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyBBbmQgdGhpcyBpcyBpdC4uLgotICAgIHR5cGVTZXRbZV0gPSBzZXQ7Ci0gICAgaWYgKHNldCkgewotICAgICAgICBpZiAob3V0VHlwZVNwZWNGbGFncyAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAqb3V0VHlwZVNwZWNGbGFncyA9IHNldC0+dHlwZVNwZWNGbGFnczsKLSAgICAgICAgfQotICAgICAgICAqb3V0QmFnID0gKGJhZ19lbnRyeSopKHNldCsxKTsKLSAgICAgICAgVEFCTEVfTk9JU1koTE9HSSgiUmV0dXJuaW5nICVkIGF0dHJzXG4iLCBzZXQtPm51bUF0dHJzKSk7Ci0gICAgICAgIHJldHVybiBzZXQtPm51bUF0dHJzOwotICAgIH0KLSAgICByZXR1cm4gQkFEX0lOREVYOwotfQotCi12b2lkIFJlc1RhYmxlOjpzZXRQYXJhbWV0ZXJzKGNvbnN0IFJlc1RhYmxlX2NvbmZpZyogcGFyYW1zKQotewotICAgIG1Mb2NrLmxvY2soKTsKLSAgICBUQUJMRV9HRVRFTlRSWShMT0dJKCJTZXR0aW5nIHBhcmFtZXRlcnM6IGltc2k6JWQvJWQgbGFuZzolYyVjIGNudDolYyVjICIKLSAgICAgICAgICAgICAgICAgICAgICAgICJvcmllbjolZCB0b3VjaDolZCBkZW5zaXR5OiVkIGtleTolZCBpbnA6JWQgbmF2OiVkIHc6JWQgaDolZFxuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgcGFyYW1zLT5tY2MsIHBhcmFtcy0+bW5jLAotICAgICAgICAgICAgICAgICAgICAgICBwYXJhbXMtPmxhbmd1YWdlWzBdID8gcGFyYW1zLT5sYW5ndWFnZVswXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICAgcGFyYW1zLT5sYW5ndWFnZVsxXSA/IHBhcmFtcy0+bGFuZ3VhZ2VbMV0gOiAnLScsCi0gICAgICAgICAgICAgICAgICAgICAgIHBhcmFtcy0+Y291bnRyeVswXSA/IHBhcmFtcy0+Y291bnRyeVswXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICAgcGFyYW1zLT5jb3VudHJ5WzFdID8gcGFyYW1zLT5jb3VudHJ5WzFdIDogJy0nLAotICAgICAgICAgICAgICAgICAgICAgICBwYXJhbXMtPm9yaWVudGF0aW9uLAotICAgICAgICAgICAgICAgICAgICAgICBwYXJhbXMtPnRvdWNoc2NyZWVuLAotICAgICAgICAgICAgICAgICAgICAgICBwYXJhbXMtPmRlbnNpdHksCi0gICAgICAgICAgICAgICAgICAgICAgIHBhcmFtcy0+a2V5Ym9hcmQsCi0gICAgICAgICAgICAgICAgICAgICAgIHBhcmFtcy0+aW5wdXRGbGFncywKLSAgICAgICAgICAgICAgICAgICAgICAgcGFyYW1zLT5uYXZpZ2F0aW9uLAotICAgICAgICAgICAgICAgICAgICAgICBwYXJhbXMtPnNjcmVlbldpZHRoLAotICAgICAgICAgICAgICAgICAgICAgICBwYXJhbXMtPnNjcmVlbkhlaWdodCkpOwotICAgIG1QYXJhbXMgPSAqcGFyYW1zOwotICAgIGZvciAoc2l6ZV90IGk9MDsgaTxtUGFja2FnZUdyb3Vwcy5zaXplKCk7IGkrKykgewotICAgICAgICBUQUJMRV9OT0lTWShMT0dJKCJDTEVBUklORyBCQUdTIEZPUiBHUk9VUCAlZCEiLCBpKSk7Ci0gICAgICAgIG1QYWNrYWdlR3JvdXBzW2ldLT5jbGVhckJhZ0NhY2hlKCk7Ci0gICAgfQotICAgIG1Mb2NrLnVubG9jaygpOwotfQotCi12b2lkIFJlc1RhYmxlOjpnZXRQYXJhbWV0ZXJzKFJlc1RhYmxlX2NvbmZpZyogcGFyYW1zKSBjb25zdAotewotICAgIG1Mb2NrLmxvY2soKTsKLSAgICAqcGFyYW1zID0gbVBhcmFtczsKLSAgICBtTG9jay51bmxvY2soKTsKLX0KLQotc3RydWN0IGlkX25hbWVfbWFwIHsKLSAgICB1aW50MzJfdCBpZDsKLSAgICBzaXplX3QgbGVuOwotICAgIGNoYXIxNl90IG5hbWVbNl07Ci19OwotCi1jb25zdCBzdGF0aWMgaWRfbmFtZV9tYXAgSURfTkFNRVNbXSA9IHsKLSAgICB7IFJlc1RhYmxlX21hcDo6QVRUUl9UWVBFLCAgNSwgeyAnXicsICd0JywgJ3knLCAncCcsICdlJyB9IH0sCi0gICAgeyBSZXNUYWJsZV9tYXA6OkFUVFJfTDEwTiwgIDUsIHsgJ14nLCAnbCcsICcxJywgJzAnLCAnbicgfSB9LAotICAgIHsgUmVzVGFibGVfbWFwOjpBVFRSX01JTiwgICA0LCB7ICdeJywgJ20nLCAnaScsICduJyB9IH0sCi0gICAgeyBSZXNUYWJsZV9tYXA6OkFUVFJfTUFYLCAgIDQsIHsgJ14nLCAnbScsICdhJywgJ3gnIH0gfSwKLSAgICB7IFJlc1RhYmxlX21hcDo6QVRUUl9PVEhFUiwgNiwgeyAnXicsICdvJywgJ3QnLCAnaCcsICdlJywgJ3InIH0gfSwKLSAgICB7IFJlc1RhYmxlX21hcDo6QVRUUl9aRVJPLCAgNSwgeyAnXicsICd6JywgJ2UnLCAncicsICdvJyB9IH0sCi0gICAgeyBSZXNUYWJsZV9tYXA6OkFUVFJfT05FLCAgIDQsIHsgJ14nLCAnbycsICduJywgJ2UnIH0gfSwKLSAgICB7IFJlc1RhYmxlX21hcDo6QVRUUl9UV08sICAgNCwgeyAnXicsICd0JywgJ3cnLCAnbycgfSB9LAotICAgIHsgUmVzVGFibGVfbWFwOjpBVFRSX0ZFVywgICA0LCB7ICdeJywgJ2YnLCAnZScsICd3JyB9IH0sCi0gICAgeyBSZXNUYWJsZV9tYXA6OkFUVFJfTUFOWSwgIDUsIHsgJ14nLCAnbScsICdhJywgJ24nLCAneScgfSB9LAotfTsKLQotdWludDMyX3QgUmVzVGFibGU6OmlkZW50aWZpZXJGb3JOYW1lKGNvbnN0IGNoYXIxNl90KiBuYW1lLCBzaXplX3QgbmFtZUxlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogdHlwZSwgc2l6ZV90IHR5cGVMZW4sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY2hhcjE2X3QqIHBhY2thZ2UsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IHBhY2thZ2VMZW4sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QqIG91dFR5cGVTcGVjRmxhZ3MpIGNvbnN0Ci17Ci0gICAgVEFCTEVfU1VQRVJfTk9JU1kocHJpbnRmKCJJZGVudGlmaWVyIGZvciBuYW1lOiBlcnJvcj0lZFxuIiwgbUVycm9yKSk7Ci0KLSAgICAvLyBDaGVjayBmb3IgaW50ZXJuYWwgcmVzb3VyY2UgaWRlbnRpZmllciBhcyB0aGUgdmVyeSBmaXJzdCB0aGluZywgc28KLSAgICAvLyB0aGF0IHdlIHdpbGwgYWx3YXlzIGZpbmQgdGhlbSBldmVuIHdoZW4gdGhlcmUgYXJlIG5vIHJlc291cmNlcy4KLSAgICBpZiAobmFtZVswXSA9PSAnXicpIHsKLSAgICAgICAgY29uc3QgaW50IE4gPSAoc2l6ZW9mKElEX05BTUVTKS9zaXplb2YoSURfTkFNRVNbMF0pKTsKLSAgICAgICAgc2l6ZV90IGxlbjsKLSAgICAgICAgZm9yIChpbnQgaT0wOyBpPE47IGkrKykgewotICAgICAgICAgICAgY29uc3QgaWRfbmFtZV9tYXAqIG0gPSBJRF9OQU1FUyArIGk7Ci0gICAgICAgICAgICBsZW4gPSBtLT5sZW47Ci0gICAgICAgICAgICBpZiAobGVuICE9IG5hbWVMZW4pIHsKLSAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGZvciAoc2l6ZV90IGo9MTsgajxsZW47IGorKykgewotICAgICAgICAgICAgICAgIGlmIChtLT5uYW1lW2pdICE9IG5hbWVbal0pIHsKLSAgICAgICAgICAgICAgICAgICAgZ290byBub3BlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBtLT5pZDsKLW5vcGU6Ci0gICAgICAgICAgICA7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKG5hbWVMZW4gPiA3KSB7Ci0gICAgICAgICAgICBpZiAobmFtZVsxXSA9PSAnaScgJiYgbmFtZVsyXSA9PSAnbicKLSAgICAgICAgICAgICAgICAmJiBuYW1lWzNdID09ICdkJyAmJiBuYW1lWzRdID09ICdlJyAmJiBuYW1lWzVdID09ICd4JwotICAgICAgICAgICAgICAgICYmIG5hbWVbNl0gPT0gJ18nKSB7Ci0gICAgICAgICAgICAgICAgaW50IGluZGV4ID0gYXRvaShTdHJpbmc4KG5hbWUgKyA3LCBuYW1lTGVuIC0gNykuc3RyaW5nKCkpOwotICAgICAgICAgICAgICAgIGlmIChSZXNfQ0hFQ0tJRChpbmRleCkpIHsKLSAgICAgICAgICAgICAgICAgICAgTE9HVygiQXJyYXkgcmVzb3VyY2UgaW5kZXg6ICVkIGlzIHRvbyBsYXJnZS4iLAotICAgICAgICAgICAgICAgICAgICAgICAgIGluZGV4KTsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiAgUmVzX01BS0VBUlJBWShpbmRleCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgaWYgKG1FcnJvciAhPSBOT19FUlJPUikgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICAvLyBGaWd1cmUgb3V0IHRoZSBwYWNrYWdlIGFuZCB0eXBlIHdlIGFyZSBsb29raW5nIGluLi4uCi0KLSAgICBjb25zdCBjaGFyMTZfdCogcGFja2FnZUVuZCA9IE5VTEw7Ci0gICAgY29uc3QgY2hhcjE2X3QqIHR5cGVFbmQgPSBOVUxMOwotICAgIGNvbnN0IGNoYXIxNl90KiBjb25zdCBuYW1lRW5kID0gbmFtZStuYW1lTGVuOwotICAgIGNvbnN0IGNoYXIxNl90KiBwID0gbmFtZTsKLSAgICB3aGlsZSAocCA8IG5hbWVFbmQpIHsKLSAgICAgICAgaWYgKCpwID09ICc6JykgcGFja2FnZUVuZCA9IHA7Ci0gICAgICAgIGVsc2UgaWYgKCpwID09ICcvJykgdHlwZUVuZCA9IHA7Ci0gICAgICAgIHArKzsKLSAgICB9Ci0gICAgaWYgKCpuYW1lID09ICdAJykgbmFtZSsrOwotICAgIGlmIChuYW1lID49IG5hbWVFbmQpIHsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgaWYgKHBhY2thZ2VFbmQpIHsKLSAgICAgICAgcGFja2FnZSA9IG5hbWU7Ci0gICAgICAgIHBhY2thZ2VMZW4gPSBwYWNrYWdlRW5kLW5hbWU7Ci0gICAgICAgIG5hbWUgPSBwYWNrYWdlRW5kKzE7Ci0gICAgfSBlbHNlIGlmICghcGFja2FnZSkgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICBpZiAodHlwZUVuZCkgewotICAgICAgICB0eXBlID0gbmFtZTsKLSAgICAgICAgdHlwZUxlbiA9IHR5cGVFbmQtbmFtZTsKLSAgICAgICAgbmFtZSA9IHR5cGVFbmQrMTsKLSAgICB9IGVsc2UgaWYgKCF0eXBlKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIGlmIChuYW1lID49IG5hbWVFbmQpIHsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotICAgIG5hbWVMZW4gPSBuYW1lRW5kLW5hbWU7Ci0KLSAgICBUQUJMRV9OT0lTWShwcmludGYoIkxvb2tpbmcgZm9yIGlkZW50aWZpZXI6IHR5cGU9JXMsIG5hbWU9JXMsIHBhY2thZ2U9JXNcbiIsCi0gICAgICAgICAgICAgICAgIFN0cmluZzgodHlwZSwgdHlwZUxlbikuc3RyaW5nKCksCi0gICAgICAgICAgICAgICAgIFN0cmluZzgobmFtZSwgbmFtZUxlbikuc3RyaW5nKCksCi0gICAgICAgICAgICAgICAgIFN0cmluZzgocGFja2FnZSwgcGFja2FnZUxlbikuc3RyaW5nKCkpKTsKLQotICAgIGNvbnN0IHNpemVfdCBORyA9IG1QYWNrYWdlR3JvdXBzLnNpemUoKTsKLSAgICBmb3IgKHNpemVfdCBpZz0wOyBpZzxORzsgaWcrKykgewotICAgICAgICBjb25zdCBQYWNrYWdlR3JvdXAqIGdyb3VwID0gbVBhY2thZ2VHcm91cHNbaWddOwotCi0gICAgICAgIGlmIChzdHJ6Y21wMTYocGFja2FnZSwgcGFja2FnZUxlbiwKLSAgICAgICAgICAgICAgICAgICAgICBncm91cC0+bmFtZS5zdHJpbmcoKSwgZ3JvdXAtPm5hbWUuc2l6ZSgpKSkgewotICAgICAgICAgICAgVEFCTEVfTk9JU1kocHJpbnRmKCJTa2lwcGluZyBwYWNrYWdlIGdyb3VwOiAlc1xuIiwgU3RyaW5nOChncm91cC0+bmFtZSkuc3RyaW5nKCkpKTsKLSAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICB9Ci0KLSAgICAgICAgY29uc3Qgc3NpemVfdCB0aSA9IGdyb3VwLT50eXBlU3RyaW5ncy5pbmRleE9mU3RyaW5nKHR5cGUsIHR5cGVMZW4pOwotICAgICAgICBpZiAodGkgPCAwKSB7Ci0gICAgICAgICAgICBUQUJMRV9OT0lTWShwcmludGYoIlR5cGUgbm90IGZvdW5kIGluIHBhY2thZ2UgJXNcbiIsIFN0cmluZzgoZ3JvdXAtPm5hbWUpLnN0cmluZygpKSk7Ci0gICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGNvbnN0IHNzaXplX3QgZWkgPSBncm91cC0+a2V5U3RyaW5ncy5pbmRleE9mU3RyaW5nKG5hbWUsIG5hbWVMZW4pOwotICAgICAgICBpZiAoZWkgPCAwKSB7Ci0gICAgICAgICAgICBUQUJMRV9OT0lTWShwcmludGYoIk5hbWUgbm90IGZvdW5kIGluIHBhY2thZ2UgJXNcbiIsIFN0cmluZzgoZ3JvdXAtPm5hbWUpLnN0cmluZygpKSk7Ci0gICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgfQotCi0gICAgICAgIFRBQkxFX05PSVNZKHByaW50ZigiU2VhcmNoIGluZGljZXM6IHR5cGU9JWQsIG5hbWU9JWRcbiIsIHRpLCBlaSkpOwotCi0gICAgICAgIGNvbnN0IFR5cGUqIGNvbnN0IHR5cGVDb25maWdzID0gZ3JvdXAtPnBhY2thZ2VzWzBdLT5nZXRUeXBlKHRpKTsKLSAgICAgICAgaWYgKHR5cGVDb25maWdzID09IE5VTEwgfHwgdHlwZUNvbmZpZ3MtPmNvbmZpZ3Muc2l6ZSgpIDw9IDApIHsKLSAgICAgICAgICAgIFRBQkxFX05PSVNZKHByaW50ZigiRXhwZWN0ZWQgdHlwZSBzdHJ1Y3R1cmUgbm90IGZvdW5kIGluIHBhY2thZ2UgJXMgZm9yIGlkbmV4ICVkXG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgoZ3JvdXAtPm5hbWUpLnN0cmluZygpLCB0aSkpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBzaXplX3QgTlRDID0gdHlwZUNvbmZpZ3MtPmNvbmZpZ3Muc2l6ZSgpOwotICAgICAgICBmb3IgKHNpemVfdCB0Y2k9MDsgdGNpPE5UQzsgdGNpKyspIHsKLSAgICAgICAgICAgIGNvbnN0IFJlc1RhYmxlX3R5cGUqIGNvbnN0IHR5ID0gdHlwZUNvbmZpZ3MtPmNvbmZpZ3NbdGNpXTsKLSAgICAgICAgICAgIGNvbnN0IHVpbnQzMl90IHR5cGVPZmZzZXQgPSBkdG9obCh0eS0+ZW50cmllc1N0YXJ0KTsKLQotICAgICAgICAgICAgY29uc3QgdWludDhfdCogY29uc3QgZW5kID0gKChjb25zdCB1aW50OF90Kil0eSkgKyBkdG9obCh0eS0+aGVhZGVyLnNpemUpOwotICAgICAgICAgICAgY29uc3QgdWludDMyX3QqIGNvbnN0IGVpbmRleCA9IChjb25zdCB1aW50MzJfdCopCi0gICAgICAgICAgICAgICAgKCgoY29uc3QgdWludDhfdCopdHkpICsgZHRvaHModHktPmhlYWRlci5oZWFkZXJTaXplKSk7Ci0KLSAgICAgICAgICAgIGNvbnN0IHNpemVfdCBORSA9IGR0b2hsKHR5LT5lbnRyeUNvdW50KTsKLSAgICAgICAgICAgIGZvciAoc2l6ZV90IGk9MDsgaTxORTsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgdWludDMyX3Qgb2Zmc2V0ID0gZHRvaGwoZWluZGV4W2ldKTsKLSAgICAgICAgICAgICAgICBpZiAob2Zmc2V0ID09IFJlc1RhYmxlX3R5cGU6Ok5PX0VOVFJZKSB7Ci0gICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICBvZmZzZXQgKz0gdHlwZU9mZnNldDsKLSAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICBpZiAob2Zmc2V0ID4gKGR0b2hsKHR5LT5oZWFkZXIuc2l6ZSktc2l6ZW9mKFJlc1RhYmxlX2VudHJ5KSkpIHsKLSAgICAgICAgICAgICAgICAgICAgTE9HVygiUmVzVGFibGVfZW50cnkgYXQgJWQgaXMgYmV5b25kIHR5cGUgY2h1bmsgZGF0YSAlZCIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0LCBkdG9obCh0eS0+aGVhZGVyLnNpemUpKTsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmICgob2Zmc2V0JjB4MykgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICBMT0dXKCJSZXNUYWJsZV9lbnRyeSBhdCAlZCAocGtnPSVkIHR5cGU9JWQgZW50PSVkKSBpcyBub3Qgb24gYW4gaW50ZWdlciBib3VuZGFyeSB3aGVuIGxvb2tpbmcgZm9yICVzOiVzLyVzIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAoaW50KW9mZnNldCwgKGludClncm91cC0+aWQsIChpbnQpdGkrMSwgKGludClpLAotICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgocGFja2FnZSwgcGFja2FnZUxlbikuc3RyaW5nKCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nOCh0eXBlLCB0eXBlTGVuKS5zdHJpbmcoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICBTdHJpbmc4KG5hbWUsIG5hbWVMZW4pLnN0cmluZygpKTsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIGNvbnN0IFJlc1RhYmxlX2VudHJ5KiBjb25zdCBlbnRyeSA9IChjb25zdCBSZXNUYWJsZV9lbnRyeSopCi0gICAgICAgICAgICAgICAgICAgICgoKGNvbnN0IHVpbnQ4X3QqKXR5KSArIG9mZnNldCk7Ci0gICAgICAgICAgICAgICAgaWYgKGR0b2hzKGVudHJ5LT5zaXplKSA8IHNpemVvZigqZW50cnkpKSB7Ci0gICAgICAgICAgICAgICAgICAgIExPR1coIlJlc1RhYmxlX2VudHJ5IHNpemUgJWQgaXMgdG9vIHNtYWxsIiwgZHRvaHMoZW50cnktPnNpemUpKTsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIEJBRF9UWVBFOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIFRBQkxFX1NVUEVSX05PSVNZKHByaW50ZigiTG9va2luZyBhdCBlbnRyeSAjJWQ6IHdhbnQgc3RyICVkLCBoYXZlICVkXG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpLCBlaSwgZHRvaGwoZW50cnktPmtleS5pbmRleCkpKTsKLSAgICAgICAgICAgICAgICBpZiAoZHRvaGwoZW50cnktPmtleS5pbmRleCkgPT0gKHNpemVfdCllaSkgewotICAgICAgICAgICAgICAgICAgICBpZiAob3V0VHlwZVNwZWNGbGFncykgewotICAgICAgICAgICAgICAgICAgICAgICAgKm91dFR5cGVTcGVjRmxhZ3MgPSB0eXBlQ29uZmlncy0+dHlwZVNwZWNGbGFnc1tpXTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICByZXR1cm4gUmVzX01BS0VJRChncm91cC0+aWQtMSwgdGksIGkpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiAwOwotfQotCi1ib29sIFJlc1RhYmxlOjpleHBhbmRSZXNvdXJjZVJlZihjb25zdCB1aW50MTZfdCogcmVmU3RyLCBzaXplX3QgcmVmTGVuLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nMTYqIG91dFBhY2thZ2UsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTdHJpbmcxNiogb3V0VHlwZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzE2KiBvdXROYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgU3RyaW5nMTYqIGRlZlR5cGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBTdHJpbmcxNiogZGVmUGFja2FnZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNoYXIqKiBvdXRFcnJvck1zZykKLXsKLSAgICBjb25zdCBjaGFyMTZfdCogcGFja2FnZUVuZCA9IE5VTEw7Ci0gICAgY29uc3QgY2hhcjE2X3QqIHR5cGVFbmQgPSBOVUxMOwotICAgIGNvbnN0IGNoYXIxNl90KiBwID0gcmVmU3RyOwotICAgIGNvbnN0IGNoYXIxNl90KiBjb25zdCBlbmQgPSBwICsgcmVmTGVuOwotICAgIHdoaWxlIChwIDwgZW5kKSB7Ci0gICAgICAgIGlmICgqcCA9PSAnOicpIHBhY2thZ2VFbmQgPSBwOwotICAgICAgICBlbHNlIGlmICgqcCA9PSAnLycpIHsKLSAgICAgICAgICAgIHR5cGVFbmQgPSBwOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgcCsrOwotICAgIH0KLSAgICBwID0gcmVmU3RyOwotICAgIGlmICgqcCA9PSAnQCcpIHArKzsKLQotICAgIGlmIChwYWNrYWdlRW5kKSB7Ci0gICAgICAgICpvdXRQYWNrYWdlID0gU3RyaW5nMTYocCwgcGFja2FnZUVuZC1wKTsKLSAgICAgICAgcCA9IHBhY2thZ2VFbmQrMTsKLSAgICB9IGVsc2UgewotICAgICAgICBpZiAoIWRlZlBhY2thZ2UpIHsKLSAgICAgICAgICAgIGlmIChvdXRFcnJvck1zZykgewotICAgICAgICAgICAgICAgICpvdXRFcnJvck1zZyA9ICJObyByZXNvdXJjZSBwYWNrYWdlIHNwZWNpZmllZCI7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICAgICAgKm91dFBhY2thZ2UgPSAqZGVmUGFja2FnZTsKLSAgICB9Ci0gICAgaWYgKHR5cGVFbmQpIHsKLSAgICAgICAgKm91dFR5cGUgPSBTdHJpbmcxNihwLCB0eXBlRW5kLXApOwotICAgICAgICBwID0gdHlwZUVuZCsxOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIGlmICghZGVmVHlwZSkgewotICAgICAgICAgICAgaWYgKG91dEVycm9yTXNnKSB7Ci0gICAgICAgICAgICAgICAgKm91dEVycm9yTXNnID0gIk5vIHJlc291cmNlIHR5cGUgc3BlY2lmaWVkIjsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgICAgICAqb3V0VHlwZSA9ICpkZWZUeXBlOwotICAgIH0KLSAgICAqb3V0TmFtZSA9IFN0cmluZzE2KHAsIGVuZC1wKTsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotc3RhdGljIHVpbnQzMl90IGdldF9oZXgoY2hhciBjLCBib29sKiBvdXRFcnJvcikKLXsKLSAgICBpZiAoYyA+PSAnMCcgJiYgYyA8PSAnOScpIHsKLSAgICAgICAgcmV0dXJuIGMgLSAnMCc7Ci0gICAgfSBlbHNlIGlmIChjID49ICdhJyAmJiBjIDw9ICdmJykgewotICAgICAgICByZXR1cm4gYyAtICdhJyArIDB4YTsKLSAgICB9IGVsc2UgaWYgKGMgPj0gJ0EnICYmIGMgPD0gJ0YnKSB7Ci0gICAgICAgIHJldHVybiBjIC0gJ0EnICsgMHhhOwotICAgIH0KLSAgICAqb3V0RXJyb3IgPSB0cnVlOwotICAgIHJldHVybiAwOwotfQotCi1zdHJ1Y3QgdW5pdF9lbnRyeQotewotICAgIGNvbnN0IGNoYXIqIG5hbWU7Ci0gICAgc2l6ZV90IGxlbjsKLSAgICB1aW50OF90IHR5cGU7Ci0gICAgdWludDMyX3QgdW5pdDsKLSAgICBmbG9hdCBzY2FsZTsKLX07Ci0KLXN0YXRpYyBjb25zdCB1bml0X2VudHJ5IHVuaXROYW1lc1tdID0gewotICAgIHsgInB4Iiwgc3RybGVuKCJweCIpLCBSZXNfdmFsdWU6OlRZUEVfRElNRU5TSU9OLCBSZXNfdmFsdWU6OkNPTVBMRVhfVU5JVF9QWCwgMS4wZiB9LAotICAgIHsgImRpcCIsIHN0cmxlbigiZGlwIiksIFJlc192YWx1ZTo6VFlQRV9ESU1FTlNJT04sIFJlc192YWx1ZTo6Q09NUExFWF9VTklUX0RJUCwgMS4wZiB9LAotICAgIHsgImRwIiwgc3RybGVuKCJkcCIpLCBSZXNfdmFsdWU6OlRZUEVfRElNRU5TSU9OLCBSZXNfdmFsdWU6OkNPTVBMRVhfVU5JVF9ESVAsIDEuMGYgfSwKLSAgICB7ICJzcCIsIHN0cmxlbigic3AiKSwgUmVzX3ZhbHVlOjpUWVBFX0RJTUVOU0lPTiwgUmVzX3ZhbHVlOjpDT01QTEVYX1VOSVRfU1AsIDEuMGYgfSwKLSAgICB7ICJwdCIsIHN0cmxlbigicHQiKSwgUmVzX3ZhbHVlOjpUWVBFX0RJTUVOU0lPTiwgUmVzX3ZhbHVlOjpDT01QTEVYX1VOSVRfUFQsIDEuMGYgfSwKLSAgICB7ICJpbiIsIHN0cmxlbigiaW4iKSwgUmVzX3ZhbHVlOjpUWVBFX0RJTUVOU0lPTiwgUmVzX3ZhbHVlOjpDT01QTEVYX1VOSVRfSU4sIDEuMGYgfSwKLSAgICB7ICJtbSIsIHN0cmxlbigibW0iKSwgUmVzX3ZhbHVlOjpUWVBFX0RJTUVOU0lPTiwgUmVzX3ZhbHVlOjpDT01QTEVYX1VOSVRfTU0sIDEuMGYgfSwKLSAgICB7ICIlIiwgc3RybGVuKCIlIiksIFJlc192YWx1ZTo6VFlQRV9GUkFDVElPTiwgUmVzX3ZhbHVlOjpDT01QTEVYX1VOSVRfRlJBQ1RJT04sIDEuMGYvMTAwIH0sCi0gICAgeyAiJXAiLCBzdHJsZW4oIiVwIiksIFJlc192YWx1ZTo6VFlQRV9GUkFDVElPTiwgUmVzX3ZhbHVlOjpDT01QTEVYX1VOSVRfRlJBQ1RJT05fUEFSRU5ULCAxLjBmLzEwMCB9LAotICAgIHsgTlVMTCwgMCwgMCwgMCwgMCB9Ci19OwotCi1zdGF0aWMgYm9vbCBwYXJzZV91bml0KGNvbnN0IGNoYXIqIHN0ciwgUmVzX3ZhbHVlKiBvdXRWYWx1ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgZmxvYXQqIG91dFNjYWxlLCBjb25zdCBjaGFyKiogb3V0RW5kKQotewotICAgIGNvbnN0IGNoYXIqIGVuZCA9IHN0cjsKLSAgICB3aGlsZSAoKmVuZCAhPSAwICYmICFpc3NwYWNlKCh1bnNpZ25lZCBjaGFyKSplbmQpKSB7Ci0gICAgICAgIGVuZCsrOwotICAgIH0KLSAgICBjb25zdCBzaXplX3QgbGVuID0gZW5kLXN0cjsKLQotICAgIGNvbnN0IGNoYXIqIHJlYWxFbmQgPSBlbmQ7Ci0gICAgd2hpbGUgKCpyZWFsRW5kICE9IDAgJiYgaXNzcGFjZSgodW5zaWduZWQgY2hhcikqcmVhbEVuZCkpIHsKLSAgICAgICAgcmVhbEVuZCsrOwotICAgIH0KLSAgICBpZiAoKnJlYWxFbmQgIT0gMCkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotICAgIAotICAgIGNvbnN0IHVuaXRfZW50cnkqIGN1ciA9IHVuaXROYW1lczsKLSAgICB3aGlsZSAoY3VyLT5uYW1lKSB7Ci0gICAgICAgIGlmIChsZW4gPT0gY3VyLT5sZW4gJiYgc3RybmNtcChjdXItPm5hbWUsIHN0ciwgbGVuKSA9PSAwKSB7Ci0gICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUgPSBjdXItPnR5cGU7Ci0gICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YSA9IGN1ci0+dW5pdCA8PCBSZXNfdmFsdWU6OkNPTVBMRVhfVU5JVF9TSElGVDsKLSAgICAgICAgICAgICpvdXRTY2FsZSA9IGN1ci0+c2NhbGU7Ci0gICAgICAgICAgICAqb3V0RW5kID0gZW5kOwotICAgICAgICAgICAgLy9wcmludGYoIkZvdW5kIHVuaXQgJXMgZm9yICVzXG4iLCBjdXItPm5hbWUsIHN0cik7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgICAgICBjdXIrKzsKLSAgICB9Ci0KLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0KLQotYm9vbCBSZXNUYWJsZTo6c3RyaW5nVG9JbnQoY29uc3QgY2hhcjE2X3QqIHMsIHNpemVfdCBsZW4sIFJlc192YWx1ZSogb3V0VmFsdWUpCi17Ci0gICAgd2hpbGUgKGxlbiA+IDAgJiYgaXNzcGFjZTE2KCpzKSkgewotICAgICAgICBzKys7Ci0gICAgICAgIGxlbi0tOwotICAgIH0KLQotICAgIGlmIChsZW4gPD0gMCkgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgc2l6ZV90IGkgPSAwOwotICAgIGludDMyX3QgdmFsID0gMDsKLSAgICBib29sIG5lZyA9IGZhbHNlOwotCi0gICAgaWYgKCpzID09ICctJykgewotICAgICAgICBuZWcgPSB0cnVlOwotICAgICAgICBpKys7Ci0gICAgfQotCi0gICAgaWYgKHNbaV0gPCAnMCcgfHwgc1tpXSA+ICc5JykgewotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLy8gRGVjaW1hbCBvciBoZXg/Ci0gICAgaWYgKHNbaV0gPT0gJzAnICYmIHNbaSsxXSA9PSAneCcpIHsKLSAgICAgICAgaWYgKG91dFZhbHVlKQotICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGFUeXBlID0gb3V0VmFsdWUtPlRZUEVfSU5UX0hFWDsKLSAgICAgICAgaSArPSAyOwotICAgICAgICBib29sIGVycm9yID0gZmFsc2U7Ci0gICAgICAgIHdoaWxlIChpIDwgbGVuICYmICFlcnJvcikgewotICAgICAgICAgICAgdmFsID0gKHZhbCoxNikgKyBnZXRfaGV4KHNbaV0sICZlcnJvcik7Ci0gICAgICAgICAgICBpKys7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGVycm9yKSB7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBpZiAob3V0VmFsdWUpCi0gICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUgPSBvdXRWYWx1ZS0+VFlQRV9JTlRfREVDOwotICAgICAgICB3aGlsZSAoaSA8IGxlbikgewotICAgICAgICAgICAgaWYgKHNbaV0gPCAnMCcgfHwgc1tpXSA+ICc5JykgewotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHZhbCA9ICh2YWwqMTApICsgc1tpXS0nMCc7Ci0gICAgICAgICAgICBpKys7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBpZiAobmVnKSB2YWwgPSAtdmFsOwotCi0gICAgd2hpbGUgKGkgPCBsZW4gJiYgaXNzcGFjZTE2KHNbaV0pKSB7Ci0gICAgICAgIGkrKzsKLSAgICB9Ci0KLSAgICBpZiAoaSA9PSBsZW4pIHsKLSAgICAgICAgaWYgKG91dFZhbHVlKQotICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEgPSB2YWw7Ci0gICAgICAgIHJldHVybiB0cnVlOwotICAgIH0KLQotICAgIHJldHVybiBmYWxzZTsKLX0KLQotYm9vbCBSZXNUYWJsZTo6c3RyaW5nVG9GbG9hdChjb25zdCBjaGFyMTZfdCogcywgc2l6ZV90IGxlbiwgUmVzX3ZhbHVlKiBvdXRWYWx1ZSkKLXsKLSAgICB3aGlsZSAobGVuID4gMCAmJiBpc3NwYWNlMTYoKnMpKSB7Ci0gICAgICAgIHMrKzsKLSAgICAgICAgbGVuLS07Ci0gICAgfQotCi0gICAgaWYgKGxlbiA8PSAwKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBjaGFyIGJ1ZlsxMjhdOwotICAgIGludCBpPTA7Ci0gICAgd2hpbGUgKGxlbiA+IDAgJiYgKnMgIT0gMCAmJiBpIDwgMTI2KSB7Ci0gICAgICAgIGlmICgqcyA+IDI1NSkgewotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIGJ1ZltpKytdID0gKnMrKzsKLSAgICAgICAgbGVuLS07Ci0gICAgfQotCi0gICAgaWYgKGxlbiA+IDApIHsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSAgICBpZiAoYnVmWzBdIDwgJzAnICYmIGJ1ZlswXSA+ICc5JyAmJiBidWZbMF0gIT0gJy4nKSB7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBidWZbaV0gPSAwOwotICAgIGNvbnN0IGNoYXIqIGVuZDsKLSAgICBmbG9hdCBmID0gc3RydG9mKGJ1ZiwgKGNoYXIqKikmZW5kKTsKLQotICAgIGlmICgqZW5kICE9IDAgJiYgIWlzc3BhY2UoKHVuc2lnbmVkIGNoYXIpKmVuZCkpIHsKLSAgICAgICAgLy8gTWlnaHQgYmUgYSB1bml0Li4uCi0gICAgICAgIGZsb2F0IHNjYWxlOwotICAgICAgICBpZiAocGFyc2VfdW5pdChlbmQsIG91dFZhbHVlLCAmc2NhbGUsICZlbmQpKSB7Ci0gICAgICAgICAgICBmICo9IHNjYWxlOwotICAgICAgICAgICAgY29uc3QgYm9vbCBuZWcgPSBmIDwgMDsKLSAgICAgICAgICAgIGlmIChuZWcpIGYgPSAtZjsKLSAgICAgICAgICAgIHVpbnQ2NF90IGJpdHMgPSAodWludDY0X3QpKGYqKDE8PDIzKSsuNWYpOwotICAgICAgICAgICAgdWludDMyX3QgcmFkaXg7Ci0gICAgICAgICAgICB1aW50MzJfdCBzaGlmdDsKLSAgICAgICAgICAgIGlmICgoYml0cyYweDdmZmZmZikgPT0gMCkgewotICAgICAgICAgICAgICAgIC8vIEFsd2F5cyB1c2UgMjNwMCBpZiB0aGVyZSBpcyBubyBmcmFjdGlvbiwganVzdCB0byBtYWtlCi0gICAgICAgICAgICAgICAgLy8gdGhpbmdzIGVhc2llciB0byByZWFkLgotICAgICAgICAgICAgICAgIHJhZGl4ID0gUmVzX3ZhbHVlOjpDT01QTEVYX1JBRElYXzIzcDA7Ci0gICAgICAgICAgICAgICAgc2hpZnQgPSAyMzsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoKGJpdHMmMHhmZmZmZmZmZmZmODAwMDAwTEwpID09IDApIHsKLSAgICAgICAgICAgICAgICAvLyBNYWduaXR1ZGUgaXMgemVybyAtLSBjYW4gZml0IGluIDAgYml0cyBvZiBwcmVjaXNpb24uCi0gICAgICAgICAgICAgICAgcmFkaXggPSBSZXNfdmFsdWU6OkNPTVBMRVhfUkFESVhfMHAyMzsKLSAgICAgICAgICAgICAgICBzaGlmdCA9IDA7Ci0gICAgICAgICAgICB9IGVsc2UgaWYgKChiaXRzJjB4ZmZmZmZmZmY4MDAwMDAwMExMKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgLy8gTWFnbml0dWRlIGNhbiBmaXQgaW4gOCBiaXRzIG9mIHByZWNpc2lvbi4KLSAgICAgICAgICAgICAgICByYWRpeCA9IFJlc192YWx1ZTo6Q09NUExFWF9SQURJWF84cDE1OwotICAgICAgICAgICAgICAgIHNoaWZ0ID0gODsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoKGJpdHMmMHhmZmZmZmY4MDAwMDAwMDAwTEwpID09IDApIHsKLSAgICAgICAgICAgICAgICAvLyBNYWduaXR1ZGUgY2FuIGZpdCBpbiAxNiBiaXRzIG9mIHByZWNpc2lvbi4KLSAgICAgICAgICAgICAgICByYWRpeCA9IFJlc192YWx1ZTo6Q09NUExFWF9SQURJWF8xNnA3OwotICAgICAgICAgICAgICAgIHNoaWZ0ID0gMTY7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIC8vIE1hZ25pdHVkZSBuZWVkcyBlbnRpcmUgcmFuZ2UsIHNvIG5vIGZyYWN0aW9uYWwgcGFydC4KLSAgICAgICAgICAgICAgICByYWRpeCA9IFJlc192YWx1ZTo6Q09NUExFWF9SQURJWF8yM3AwOwotICAgICAgICAgICAgICAgIHNoaWZ0ID0gMjM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpbnQzMl90IG1hbnRpc3NhID0gKGludDMyX3QpKAotICAgICAgICAgICAgICAgIChiaXRzPj5zaGlmdCkgJiBSZXNfdmFsdWU6OkNPTVBMRVhfTUFOVElTU0FfTUFTSyk7Ci0gICAgICAgICAgICBpZiAobmVnKSB7Ci0gICAgICAgICAgICAgICAgbWFudGlzc2EgPSAoLW1hbnRpc3NhKSAmIFJlc192YWx1ZTo6Q09NUExFWF9NQU5USVNTQV9NQVNLOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEgfD0gCi0gICAgICAgICAgICAgICAgKHJhZGl4PDxSZXNfdmFsdWU6OkNPTVBMRVhfUkFESVhfU0hJRlQpCi0gICAgICAgICAgICAgICAgfCAobWFudGlzc2E8PFJlc192YWx1ZTo6Q09NUExFWF9NQU5USVNTQV9TSElGVCk7Ci0gICAgICAgICAgICAvL3ByaW50ZigiSW5wdXQgdmFsdWU6ICVmIDB4JTAxNkx4LCBtdWx0OiAlZiwgcmFkaXg6ICVkLCBzaGlmdDogJWQsIGZpbmFsOiAweCUwOHhcbiIsCi0gICAgICAgICAgICAvLyAgICAgICBmICogKG5lZyA/IC0xIDogMSksIGJpdHMsIGYqKDE8PDIzKSwKLSAgICAgICAgICAgIC8vICAgICAgIHJhZGl4LCBzaGlmdCwgb3V0VmFsdWUtPmRhdGEpOwotICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIHdoaWxlICgqZW5kICE9IDAgJiYgaXNzcGFjZSgodW5zaWduZWQgY2hhcikqZW5kKSkgewotICAgICAgICBlbmQrKzsKLSAgICB9Ci0KLSAgICBpZiAoKmVuZCA9PSAwKSB7Ci0gICAgICAgIGlmIChvdXRWYWx1ZSkgewotICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGFUeXBlID0gb3V0VmFsdWUtPlRZUEVfRkxPQVQ7Ci0gICAgICAgICAgICAqKGZsb2F0KikoJm91dFZhbHVlLT5kYXRhKSA9IGY7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBmYWxzZTsKLX0KLQotYm9vbCBSZXNUYWJsZTo6c3RyaW5nVG9WYWx1ZShSZXNfdmFsdWUqIG91dFZhbHVlLCBTdHJpbmcxNiogb3V0U3RyaW5nLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogcywgc2l6ZV90IGxlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYm9vbCBwcmVzZXJ2ZVNwYWNlcywgYm9vbCBjb2VyY2VUeXBlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBhdHRySUQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFN0cmluZzE2KiBkZWZUeXBlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBTdHJpbmcxNiogZGVmUGFja2FnZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQWNjZXNzb3IqIGFjY2Vzc29yLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkKiBhY2Nlc3NvckNvb2tpZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgYXR0clR5cGUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2wgZW5mb3JjZVByaXZhdGUpIGNvbnN0Ci17Ci0gICAgYm9vbCBsb2NhbGl6YXRpb25TZXR0aW5nID0gYWNjZXNzb3IgIT0gTlVMTCAmJiBhY2Nlc3Nvci0+Z2V0TG9jYWxpemF0aW9uU2V0dGluZygpOwotICAgIGNvbnN0IGNoYXIqIGVycm9yTXNnID0gTlVMTDsKLQotICAgIG91dFZhbHVlLT5zaXplID0gc2l6ZW9mKFJlc192YWx1ZSk7Ci0gICAgb3V0VmFsdWUtPnJlczAgPSAwOwotCi0gICAgLy8gRmlyc3Qgc3RyaXAgbGVhZGluZy90cmFpbGluZyB3aGl0ZXNwYWNlLiAgRG8gdGhpcyBiZWZvcmUgaGFuZGxpbmcKLSAgICAvLyBlc2NhcGVzLCBzbyB0aGV5IGNhbiBiZSB1c2VkIHRvIGZvcmNlIHdoaXRlc3BhY2UgaW50byB0aGUgc3RyaW5nLgotICAgIGlmICghcHJlc2VydmVTcGFjZXMpIHsKLSAgICAgICAgd2hpbGUgKGxlbiA+IDAgJiYgaXNzcGFjZTE2KCpzKSkgewotICAgICAgICAgICAgcysrOwotICAgICAgICAgICAgbGVuLS07Ci0gICAgICAgIH0KLSAgICAgICAgd2hpbGUgKGxlbiA+IDAgJiYgaXNzcGFjZTE2KHNbbGVuLTFdKSkgewotICAgICAgICAgICAgbGVuLS07Ci0gICAgICAgIH0KLSAgICAgICAgLy8gSWYgdGhlIHN0cmluZyBlbmRzIHdpdGggJ1wnLCB0aGVuIHdlIGtlZXAgdGhlIHNwYWNlIGFmdGVyIGl0LgotICAgICAgICBpZiAobGVuID4gMCAmJiBzW2xlbi0xXSA9PSAnXFwnICYmIHNbbGVuXSAhPSAwKSB7Ci0gICAgICAgICAgICBsZW4rKzsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vcHJpbnRmKCJWYWx1ZSBmb3I6ICVzXG4iLCBTdHJpbmc4KHMsIGxlbikuc3RyaW5nKCkpOwotCi0gICAgdWludDMyX3QgbDEwblJlcSA9IFJlc1RhYmxlX21hcDo6TDEwTl9OT1RfUkVRVUlSRUQ7Ci0gICAgdWludDMyX3QgYXR0ck1pbiA9IDB4ODAwMDAwMDAsIGF0dHJNYXggPSAweDdmZmZmZmZmOwotICAgIGJvb2wgZnJvbUFjY2Vzc29yID0gZmFsc2U7Ci0gICAgaWYgKGF0dHJJRCAhPSAwICYmICFSZXNfSU5URVJOQUxJRChhdHRySUQpKSB7Ci0gICAgICAgIGNvbnN0IHNzaXplX3QgcCA9IGdldFJlc291cmNlUGFja2FnZUluZGV4KGF0dHJJRCk7Ci0gICAgICAgIGNvbnN0IGJhZ19lbnRyeSogYmFnOwotICAgICAgICBzc2l6ZV90IGNudCA9IHAgPj0gMCA/IGxvY2tCYWcoYXR0cklELCAmYmFnKSA6IC0xOwotICAgICAgICAvL3ByaW50ZigiRm9yIGF0dHIgMHglMDh4IGdvdCBiYWcgb2YgJWRcbiIsIGF0dHJJRCwgY250KTsKLSAgICAgICAgaWYgKGNudCA+PSAwKSB7Ci0gICAgICAgICAgICB3aGlsZSAoY250ID4gMCkgewotICAgICAgICAgICAgICAgIC8vcHJpbnRmKCJFbnRyeSAweCUwOHggPSAweCUwOHhcbiIsIGJhZy0+bWFwLm5hbWUuaWRlbnQsIGJhZy0+bWFwLnZhbHVlLmRhdGEpOwotICAgICAgICAgICAgICAgIHN3aXRjaCAoYmFnLT5tYXAubmFtZS5pZGVudCkgewotICAgICAgICAgICAgICAgIGNhc2UgUmVzVGFibGVfbWFwOjpBVFRSX1RZUEU6Ci0gICAgICAgICAgICAgICAgICAgIGF0dHJUeXBlID0gYmFnLT5tYXAudmFsdWUuZGF0YTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBSZXNUYWJsZV9tYXA6OkFUVFJfTUlOOgotICAgICAgICAgICAgICAgICAgICBhdHRyTWluID0gYmFnLT5tYXAudmFsdWUuZGF0YTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBSZXNUYWJsZV9tYXA6OkFUVFJfTUFYOgotICAgICAgICAgICAgICAgICAgICBhdHRyTWF4ID0gYmFnLT5tYXAudmFsdWUuZGF0YTsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgY2FzZSBSZXNUYWJsZV9tYXA6OkFUVFJfTDEwTjoKLSAgICAgICAgICAgICAgICAgICAgbDEwblJlcSA9IGJhZy0+bWFwLnZhbHVlLmRhdGE7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBiYWcrKzsKLSAgICAgICAgICAgICAgICBjbnQtLTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHVubG9ja0JhZyhiYWcpOwotICAgICAgICB9IGVsc2UgaWYgKGFjY2Vzc29yICYmIGFjY2Vzc29yLT5nZXRBdHRyaWJ1dGVUeXBlKGF0dHJJRCwgJmF0dHJUeXBlKSkgewotICAgICAgICAgICAgZnJvbUFjY2Vzc29yID0gdHJ1ZTsKLSAgICAgICAgICAgIGlmIChhdHRyVHlwZSA9PSBSZXNUYWJsZV9tYXA6OlRZUEVfRU5VTQotICAgICAgICAgICAgICAgICAgICB8fCBhdHRyVHlwZSA9PSBSZXNUYWJsZV9tYXA6OlRZUEVfRkxBR1MKLSAgICAgICAgICAgICAgICAgICAgfHwgYXR0clR5cGUgPT0gUmVzVGFibGVfbWFwOjpUWVBFX0lOVEVHRVIpIHsKLSAgICAgICAgICAgICAgICBhY2Nlc3Nvci0+Z2V0QXR0cmlidXRlTWluKGF0dHJJRCwgJmF0dHJNaW4pOwotICAgICAgICAgICAgICAgIGFjY2Vzc29yLT5nZXRBdHRyaWJ1dGVNYXgoYXR0cklELCAmYXR0ck1heCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAobG9jYWxpemF0aW9uU2V0dGluZykgewotICAgICAgICAgICAgICAgIGwxMG5SZXEgPSBhY2Nlc3Nvci0+Z2V0QXR0cmlidXRlTDEwTihhdHRySUQpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgY29uc3QgYm9vbCBjYW5TdHJpbmdDb2VyY2UgPQotICAgICAgICBjb2VyY2VUeXBlICYmIChhdHRyVHlwZSZSZXNUYWJsZV9tYXA6OlRZUEVfU1RSSU5HKSAhPSAwOwotCi0gICAgaWYgKCpzID09ICdAJykgewotICAgICAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUgPSBvdXRWYWx1ZS0+VFlQRV9SRUZFUkVOQ0U7Ci0KLSAgICAgICAgLy8gTm90ZTogd2UgZG9uJ3QgY2hlY2sgYXR0clR5cGUgaGVyZSBiZWNhdXNlIHRoZSByZWZlcmVuY2UgY2FuCi0gICAgICAgIC8vIGJlIHRvIGFueSBvdGhlciB0eXBlOyB3ZSBqdXN0IG5lZWQgdG8gY291bnQgb24gdGhlIGNsaWVudCBtYWtpbmcKLSAgICAgICAgLy8gc3VyZSB0aGUgcmVmZXJlbmNlZCB0eXBlIGlzIGNvcnJlY3QuCi0gICAgICAgIAotICAgICAgICAvL3ByaW50ZigiTG9va2luZyB1cCByZWY6ICVzXG4iLCBTdHJpbmc4KHMsIGxlbikuc3RyaW5nKCkpOwotCi0gICAgICAgIC8vIEl0J3MgYSByZWZlcmVuY2UhCi0gICAgICAgIGlmIChsZW4gPT0gNSAmJiBzWzFdPT0nbicgJiYgc1syXT09J3UnICYmIHNbM109PSdsJyAmJiBzWzRdPT0nbCcpIHsKLSAgICAgICAgICAgIG91dFZhbHVlLT5kYXRhID0gMDsKLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgYm9vbCBjcmVhdGVJZk5vdEZvdW5kID0gZmFsc2U7Ci0gICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogcmVzb3VyY2VSZWZOYW1lOwotICAgICAgICAgICAgaW50IHJlc291cmNlTmFtZUxlbjsKLSAgICAgICAgICAgIGlmIChsZW4gPiAyICYmIHNbMV0gPT0gJysnKSB7Ci0gICAgICAgICAgICAgICAgY3JlYXRlSWZOb3RGb3VuZCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgcmVzb3VyY2VSZWZOYW1lID0gcyArIDI7Ci0gICAgICAgICAgICAgICAgcmVzb3VyY2VOYW1lTGVuID0gbGVuIC0gMjsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAobGVuID4gMiAmJiBzWzFdID09ICcqJykgewotICAgICAgICAgICAgICAgIGVuZm9yY2VQcml2YXRlID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgcmVzb3VyY2VSZWZOYW1lID0gcyArIDI7Ci0gICAgICAgICAgICAgICAgcmVzb3VyY2VOYW1lTGVuID0gbGVuIC0gMjsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgY3JlYXRlSWZOb3RGb3VuZCA9IGZhbHNlOwotICAgICAgICAgICAgICAgIHJlc291cmNlUmVmTmFtZSA9IHMgKyAxOwotICAgICAgICAgICAgICAgIHJlc291cmNlTmFtZUxlbiA9IGxlbiAtIDE7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBTdHJpbmcxNiBwYWNrYWdlLCB0eXBlLCBuYW1lOwotICAgICAgICAgICAgaWYgKCFleHBhbmRSZXNvdXJjZVJlZihyZXNvdXJjZVJlZk5hbWUscmVzb3VyY2VOYW1lTGVuLCAmcGFja2FnZSwgJnR5cGUsICZuYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkZWZUeXBlLCBkZWZQYWNrYWdlLCAmZXJyb3JNc2cpKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGFjY2Vzc29yICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgYWNjZXNzb3ItPnJlcG9ydEVycm9yKGFjY2Vzc29yQ29va2llLCBlcnJvck1zZyk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgdWludDMyX3Qgc3BlY0ZsYWdzID0gMDsKLSAgICAgICAgICAgIHVpbnQzMl90IHJpZCA9IGlkZW50aWZpZXJGb3JOYW1lKG5hbWUuc3RyaW5nKCksIG5hbWUuc2l6ZSgpLCB0eXBlLnN0cmluZygpLAotICAgICAgICAgICAgICAgICAgICB0eXBlLnNpemUoKSwgcGFja2FnZS5zdHJpbmcoKSwgcGFja2FnZS5zaXplKCksICZzcGVjRmxhZ3MpOwotICAgICAgICAgICAgaWYgKHJpZCAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGVuZm9yY2VQcml2YXRlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmICgoc3BlY0ZsYWdzJlJlc1RhYmxlX3R5cGVTcGVjOjpTUEVDX1BVQkxJQykgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGFjY2Vzc29yICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBhY2Nlc3Nvci0+cmVwb3J0RXJyb3IoYWNjZXNzb3JDb29raWUsICJSZXNvdXJjZSBpcyBub3QgcHVibGljLiIpOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmICghYWNjZXNzb3IpIHsKLSAgICAgICAgICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEgPSByaWQ7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByaWQgPSBSZXNfTUFLRUlEKAotICAgICAgICAgICAgICAgICAgICBhY2Nlc3Nvci0+Z2V0UmVtYXBwZWRQYWNrYWdlKFJlc19HRVRQQUNLQUdFKHJpZCkpLAotICAgICAgICAgICAgICAgICAgICBSZXNfR0VUVFlQRShyaWQpLCBSZXNfR0VURU5UUlkocmlkKSk7Ci0gICAgICAgICAgICAgICAgVEFCTEVfTk9JU1kocHJpbnRmKCJJbmNsICVzOiVzLyVzOiAweCUwOHhcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgocGFja2FnZSkuc3RyaW5nKCksIFN0cmluZzgodHlwZSkuc3RyaW5nKCksCi0gICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgobmFtZSkuc3RyaW5nKCksIHJpZCkpOwotICAgICAgICAgICAgICAgIG91dFZhbHVlLT5kYXRhID0gcmlkOwotICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoYWNjZXNzb3IpIHsKLSAgICAgICAgICAgICAgICB1aW50MzJfdCByaWQgPSBhY2Nlc3Nvci0+Z2V0Q3VzdG9tUmVzb3VyY2VXaXRoQ3JlYXRpb24ocGFja2FnZSwgdHlwZSwgbmFtZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY3JlYXRlSWZOb3RGb3VuZCk7Ci0gICAgICAgICAgICAgICAgaWYgKHJpZCAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIFRBQkxFX05PSVNZKHByaW50ZigiUGNrZyAlczolcy8lczogMHglMDh4XG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nOChwYWNrYWdlKS5zdHJpbmcoKSwgU3RyaW5nOCh0eXBlKS5zdHJpbmcoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgobmFtZSkuc3RyaW5nKCksIHJpZCkpOwotICAgICAgICAgICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YSA9IHJpZDsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGFjY2Vzc29yICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGFjY2Vzc29yLT5yZXBvcnRFcnJvcihhY2Nlc3NvckNvb2tpZSwgIk5vIHJlc291cmNlIGZvdW5kIHRoYXQgbWF0Y2hlcyB0aGUgZ2l2ZW4gbmFtZSIpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICAvLyBpZiB3ZSBnb3QgdG8gaGVyZSwgYW5kIGxvY2FsaXphdGlvbiBpcyByZXF1aXJlZCBhbmQgaXQncyBub3QgYSByZWZlcmVuY2UsCi0gICAgLy8gY29tcGxhaW4gYW5kIGJhaWwuCi0gICAgaWYgKGwxMG5SZXEgPT0gUmVzVGFibGVfbWFwOjpMMTBOX1NVR0dFU1RFRCkgewotICAgICAgICBpZiAobG9jYWxpemF0aW9uU2V0dGluZykgewotICAgICAgICAgICAgaWYgKGFjY2Vzc29yICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICBhY2Nlc3Nvci0+cmVwb3J0RXJyb3IoYWNjZXNzb3JDb29raWUsICJUaGlzIGF0dHJpYnV0ZSBtdXN0IGJlIGxvY2FsaXplZC4iKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBpZiAoKnMgPT0gJyMnKSB7Ci0gICAgICAgIC8vIEl0J3MgYSBjb2xvciEgIENvbnZlcnQgdG8gYW4gaW50ZWdlciBvZiB0aGUgZm9ybSAweGFhcnJnZ2JiLgotICAgICAgICB1aW50MzJfdCBjb2xvciA9IDA7Ci0gICAgICAgIGJvb2wgZXJyb3IgPSBmYWxzZTsKLSAgICAgICAgaWYgKGxlbiA9PSA0KSB7Ci0gICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUgPSBvdXRWYWx1ZS0+VFlQRV9JTlRfQ09MT1JfUkdCNDsKLSAgICAgICAgICAgIGNvbG9yIHw9IDB4RkYwMDAwMDA7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMV0sICZlcnJvcikgPDwgMjA7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMV0sICZlcnJvcikgPDwgMTY7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMl0sICZlcnJvcikgPDwgMTI7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMl0sICZlcnJvcikgPDwgODsKLSAgICAgICAgICAgIGNvbG9yIHw9IGdldF9oZXgoc1szXSwgJmVycm9yKSA8PCA0OwotICAgICAgICAgICAgY29sb3IgfD0gZ2V0X2hleChzWzNdLCAmZXJyb3IpOwotICAgICAgICB9IGVsc2UgaWYgKGxlbiA9PSA1KSB7Ci0gICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUgPSBvdXRWYWx1ZS0+VFlQRV9JTlRfQ09MT1JfQVJHQjQ7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMV0sICZlcnJvcikgPDwgMjg7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMV0sICZlcnJvcikgPDwgMjQ7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMl0sICZlcnJvcikgPDwgMjA7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMl0sICZlcnJvcikgPDwgMTY7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbM10sICZlcnJvcikgPDwgMTI7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbM10sICZlcnJvcikgPDwgODsKLSAgICAgICAgICAgIGNvbG9yIHw9IGdldF9oZXgoc1s0XSwgJmVycm9yKSA8PCA0OwotICAgICAgICAgICAgY29sb3IgfD0gZ2V0X2hleChzWzRdLCAmZXJyb3IpOwotICAgICAgICB9IGVsc2UgaWYgKGxlbiA9PSA3KSB7Ci0gICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUgPSBvdXRWYWx1ZS0+VFlQRV9JTlRfQ09MT1JfUkdCODsKLSAgICAgICAgICAgIGNvbG9yIHw9IDB4RkYwMDAwMDA7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMV0sICZlcnJvcikgPDwgMjA7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMl0sICZlcnJvcikgPDwgMTY7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbM10sICZlcnJvcikgPDwgMTI7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbNF0sICZlcnJvcikgPDwgODsKLSAgICAgICAgICAgIGNvbG9yIHw9IGdldF9oZXgoc1s1XSwgJmVycm9yKSA8PCA0OwotICAgICAgICAgICAgY29sb3IgfD0gZ2V0X2hleChzWzZdLCAmZXJyb3IpOwotICAgICAgICB9IGVsc2UgaWYgKGxlbiA9PSA5KSB7Ci0gICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUgPSBvdXRWYWx1ZS0+VFlQRV9JTlRfQ09MT1JfQVJHQjg7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMV0sICZlcnJvcikgPDwgMjg7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbMl0sICZlcnJvcikgPDwgMjQ7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbM10sICZlcnJvcikgPDwgMjA7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbNF0sICZlcnJvcikgPDwgMTY7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbNV0sICZlcnJvcikgPDwgMTI7Ci0gICAgICAgICAgICBjb2xvciB8PSBnZXRfaGV4KHNbNl0sICZlcnJvcikgPDwgODsKLSAgICAgICAgICAgIGNvbG9yIHw9IGdldF9oZXgoc1s3XSwgJmVycm9yKSA8PCA0OwotICAgICAgICAgICAgY29sb3IgfD0gZ2V0X2hleChzWzhdLCAmZXJyb3IpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZXJyb3IgPSB0cnVlOwotICAgICAgICB9Ci0gICAgICAgIGlmICghZXJyb3IpIHsKLSAgICAgICAgICAgIGlmICgoYXR0clR5cGUmUmVzVGFibGVfbWFwOjpUWVBFX0NPTE9SKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgaWYgKCFjYW5TdHJpbmdDb2VyY2UpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGFjY2Vzc29yICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGFjY2Vzc29yLT5yZXBvcnRFcnJvcihhY2Nlc3NvckNvb2tpZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIkNvbG9yIHR5cGVzIG5vdCBhbGxvd2VkIik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEgPSBjb2xvcjsKLSAgICAgICAgICAgICAgICAvL3ByaW50ZigiQ29sb3IgaW5wdXQ9JXMsIG91dHB1dD0weCV4XG4iLCBTdHJpbmc4KHMsIGxlbikuc3RyaW5nKCksIGNvbG9yKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGlmICgoYXR0clR5cGUmUmVzVGFibGVfbWFwOjpUWVBFX0NPTE9SKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGFjY2Vzc29yICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgYWNjZXNzb3ItPnJlcG9ydEVycm9yKGFjY2Vzc29yQ29va2llLCAiQ29sb3IgdmFsdWUgbm90IHZhbGlkIC0tIgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICIgbXVzdCBiZSAjcmdiLCAjYXJnYiwgI3JyZ2diYiwgb3IgI2FhcnJnZ2JiIik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICNpZiAwCi0gICAgICAgICAgICAgICAgZnByaW50ZihzdGRlcnIsICIlczogQ29sb3IgSUQgJXMgdmFsdWUgJXMgaXMgbm90IHZhbGlkXG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgIlJlc291cmNlIEZpbGUiLCAvLyhjb25zdCBjaGFyKilpbi0+Z2V0UHJpbnRhYmxlU291cmNlKCksCi0gICAgICAgICAgICAgICAgICAgICAgICBTdHJpbmc4KCpjdXJUYWcpLnN0cmluZygpLAotICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nOChzLCBsZW4pLnN0cmluZygpKTsKLSAgICAgICAgICAgICAgICAjZW5kaWYKLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBpZiAoKnMgPT0gJz8nKSB7Ci0gICAgICAgIG91dFZhbHVlLT5kYXRhVHlwZSA9IG91dFZhbHVlLT5UWVBFX0FUVFJJQlVURTsKLQotICAgICAgICAvLyBOb3RlOiB3ZSBkb24ndCBjaGVjayBhdHRyVHlwZSBoZXJlIGJlY2F1c2UgdGhlIHJlZmVyZW5jZSBjYW4KLSAgICAgICAgLy8gYmUgdG8gYW55IG90aGVyIHR5cGU7IHdlIGp1c3QgbmVlZCB0byBjb3VudCBvbiB0aGUgY2xpZW50IG1ha2luZwotICAgICAgICAvLyBzdXJlIHRoZSByZWZlcmVuY2VkIHR5cGUgaXMgY29ycmVjdC4KLQotICAgICAgICAvL3ByaW50ZigiTG9va2luZyB1cCBhdHRyOiAlc1xuIiwgU3RyaW5nOChzLCBsZW4pLnN0cmluZygpKTsKLQotICAgICAgICBzdGF0aWMgY29uc3QgU3RyaW5nMTYgYXR0cjE2KCJhdHRyIik7Ci0gICAgICAgIFN0cmluZzE2IHBhY2thZ2UsIHR5cGUsIG5hbWU7Ci0gICAgICAgIGlmICghZXhwYW5kUmVzb3VyY2VSZWYocysxLCBsZW4tMSwgJnBhY2thZ2UsICZ0eXBlLCAmbmFtZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAmYXR0cjE2LCBkZWZQYWNrYWdlLCAmZXJyb3JNc2cpKSB7Ci0gICAgICAgICAgICBpZiAoYWNjZXNzb3IgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgIGFjY2Vzc29yLT5yZXBvcnRFcnJvcihhY2Nlc3NvckNvb2tpZSwgZXJyb3JNc2cpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0KLSAgICAgICAgLy9wcmludGYoIlBrZzogJXMsIFR5cGU6ICVzLCBOYW1lOiAlc1xuIiwKLSAgICAgICAgLy8gICAgICAgU3RyaW5nOChwYWNrYWdlKS5zdHJpbmcoKSwgU3RyaW5nOCh0eXBlKS5zdHJpbmcoKSwKLSAgICAgICAgLy8gICAgICAgU3RyaW5nOChuYW1lKS5zdHJpbmcoKSk7Ci0gICAgICAgIHVpbnQzMl90IHNwZWNGbGFncyA9IDA7Ci0gICAgICAgIHVpbnQzMl90IHJpZCA9IAotICAgICAgICAgICAgaWRlbnRpZmllckZvck5hbWUobmFtZS5zdHJpbmcoKSwgbmFtZS5zaXplKCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0eXBlLnN0cmluZygpLCB0eXBlLnNpemUoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBhY2thZ2Uuc3RyaW5nKCksIHBhY2thZ2Uuc2l6ZSgpLCAmc3BlY0ZsYWdzKTsKLSAgICAgICAgaWYgKHJpZCAhPSAwKSB7Ci0gICAgICAgICAgICBpZiAoZW5mb3JjZVByaXZhdGUpIHsKLSAgICAgICAgICAgICAgICBpZiAoKHNwZWNGbGFncyZSZXNUYWJsZV90eXBlU3BlYzo6U1BFQ19QVUJMSUMpID09IDApIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGFjY2Vzc29yICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGFjY2Vzc29yLT5yZXBvcnRFcnJvcihhY2Nlc3NvckNvb2tpZSwgIkF0dHJpYnV0ZSBpcyBub3QgcHVibGljLiIpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoIWFjY2Vzc29yKSB7Ci0gICAgICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEgPSByaWQ7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByaWQgPSBSZXNfTUFLRUlEKAotICAgICAgICAgICAgICAgIGFjY2Vzc29yLT5nZXRSZW1hcHBlZFBhY2thZ2UoUmVzX0dFVFBBQ0tBR0UocmlkKSksCi0gICAgICAgICAgICAgICAgUmVzX0dFVFRZUEUocmlkKSwgUmVzX0dFVEVOVFJZKHJpZCkpOwotICAgICAgICAgICAgLy9wcmludGYoIkluY2wgJXM6JXMvJXM6IDB4JTA4eFxuIiwKLSAgICAgICAgICAgIC8vICAgICAgIFN0cmluZzgocGFja2FnZSkuc3RyaW5nKCksIFN0cmluZzgodHlwZSkuc3RyaW5nKCksCi0gICAgICAgICAgICAvLyAgICAgICBTdHJpbmc4KG5hbWUpLnN0cmluZygpLCByaWQpOwotICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEgPSByaWQ7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChhY2Nlc3NvcikgewotICAgICAgICAgICAgdWludDMyX3QgcmlkID0gYWNjZXNzb3ItPmdldEN1c3RvbVJlc291cmNlKHBhY2thZ2UsIHR5cGUsIG5hbWUpOwotICAgICAgICAgICAgaWYgKHJpZCAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgLy9wcmludGYoIk1pbmUgJXM6JXMvJXM6IDB4JTA4eFxuIiwKLSAgICAgICAgICAgICAgICAvLyAgICAgICBTdHJpbmc4KHBhY2thZ2UpLnN0cmluZygpLCBTdHJpbmc4KHR5cGUpLnN0cmluZygpLAotICAgICAgICAgICAgICAgIC8vICAgICAgIFN0cmluZzgobmFtZSkuc3RyaW5nKCksIHJpZCk7Ci0gICAgICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEgPSByaWQ7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoYWNjZXNzb3IgIT0gTlVMTCkgewotICAgICAgICAgICAgYWNjZXNzb3ItPnJlcG9ydEVycm9yKGFjY2Vzc29yQ29va2llLCAiTm8gcmVzb3VyY2UgZm91bmQgdGhhdCBtYXRjaGVzIHRoZSBnaXZlbiBuYW1lIik7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotICAgIGlmIChzdHJpbmdUb0ludChzLCBsZW4sIG91dFZhbHVlKSkgewotICAgICAgICBpZiAoKGF0dHJUeXBlJlJlc1RhYmxlX21hcDo6VFlQRV9JTlRFR0VSKSA9PSAwKSB7Ci0gICAgICAgICAgICAvLyBJZiB0aGlzIHR5cGUgZG9lcyBub3QgYWxsb3cgaW50ZWdlcnMsIGJ1dCBkb2VzIGFsbG93IGZsb2F0cywKLSAgICAgICAgICAgIC8vIGZhbGwgdGhyb3VnaCBvbiB0aGlzIGVycm9yIGNhc2UgYmVjYXVzZSB0aGUgZmxvYXQgdHlwZSBzaG91bGQKLSAgICAgICAgICAgIC8vIGJlIGFibGUgdG8gYWNjZXB0IGFueSBpbnRlZ2VyIHZhbHVlLgotICAgICAgICAgICAgaWYgKCFjYW5TdHJpbmdDb2VyY2UgJiYgKGF0dHJUeXBlJlJlc1RhYmxlX21hcDo6VFlQRV9GTE9BVCkgPT0gMCkgewotICAgICAgICAgICAgICAgIGlmIChhY2Nlc3NvciAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgICAgIGFjY2Vzc29yLT5yZXBvcnRFcnJvcihhY2Nlc3NvckNvb2tpZSwgIkludGVnZXIgdHlwZXMgbm90IGFsbG93ZWQiKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaWYgKCgoaW50MzJfdClvdXRWYWx1ZS0+ZGF0YSkgPCAoKGludDMyX3QpYXR0ck1pbikKLSAgICAgICAgICAgICAgICAgICAgfHwgKChpbnQzMl90KW91dFZhbHVlLT5kYXRhKSA+ICgoaW50MzJfdClhdHRyTWF4KSkgewotICAgICAgICAgICAgICAgIGlmIChhY2Nlc3NvciAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgICAgIGFjY2Vzc29yLT5yZXBvcnRFcnJvcihhY2Nlc3NvckNvb2tpZSwgIkludGVnZXIgdmFsdWUgb3V0IG9mIHJhbmdlIik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgaWYgKHN0cmluZ1RvRmxvYXQocywgbGVuLCBvdXRWYWx1ZSkpIHsKLSAgICAgICAgaWYgKG91dFZhbHVlLT5kYXRhVHlwZSA9PSBSZXNfdmFsdWU6OlRZUEVfRElNRU5TSU9OKSB7Ci0gICAgICAgICAgICBpZiAoKGF0dHJUeXBlJlJlc1RhYmxlX21hcDo6VFlQRV9ESU1FTlNJT04pICE9IDApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICghY2FuU3RyaW5nQ29lcmNlKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGFjY2Vzc29yICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgYWNjZXNzb3ItPnJlcG9ydEVycm9yKGFjY2Vzc29yQ29va2llLCAiRGltZW5zaW9uIHR5cGVzIG5vdCBhbGxvd2VkIik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIGlmIChvdXRWYWx1ZS0+ZGF0YVR5cGUgPT0gUmVzX3ZhbHVlOjpUWVBFX0ZSQUNUSU9OKSB7Ci0gICAgICAgICAgICBpZiAoKGF0dHJUeXBlJlJlc1RhYmxlX21hcDo6VFlQRV9GUkFDVElPTikgIT0gMCkgewotICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKCFjYW5TdHJpbmdDb2VyY2UpIHsKLSAgICAgICAgICAgICAgICBpZiAoYWNjZXNzb3IgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICBhY2Nlc3Nvci0+cmVwb3J0RXJyb3IoYWNjZXNzb3JDb29raWUsICJGcmFjdGlvbiB0eXBlcyBub3QgYWxsb3dlZCIpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSBpZiAoKGF0dHJUeXBlJlJlc1RhYmxlX21hcDo6VFlQRV9GTE9BVCkgPT0gMCkgewotICAgICAgICAgICAgaWYgKCFjYW5TdHJpbmdDb2VyY2UpIHsKLSAgICAgICAgICAgICAgICBpZiAoYWNjZXNzb3IgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICBhY2Nlc3Nvci0+cmVwb3J0RXJyb3IoYWNjZXNzb3JDb29raWUsICJGbG9hdCB0eXBlcyBub3QgYWxsb3dlZCIpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIGlmIChsZW4gPT0gNCkgewotICAgICAgICBpZiAoKHNbMF0gPT0gJ3QnIHx8IHNbMF0gPT0gJ1QnKSAmJgotICAgICAgICAgICAgKHNbMV0gPT0gJ3InIHx8IHNbMV0gPT0gJ1InKSAmJgotICAgICAgICAgICAgKHNbMl0gPT0gJ3UnIHx8IHNbMl0gPT0gJ1UnKSAmJgotICAgICAgICAgICAgKHNbM10gPT0gJ2UnIHx8IHNbM10gPT0gJ0UnKSkgewotICAgICAgICAgICAgaWYgKChhdHRyVHlwZSZSZXNUYWJsZV9tYXA6OlRZUEVfQk9PTEVBTikgPT0gMCkgewotICAgICAgICAgICAgICAgIGlmICghY2FuU3RyaW5nQ29lcmNlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChhY2Nlc3NvciAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBhY2Nlc3Nvci0+cmVwb3J0RXJyb3IoYWNjZXNzb3JDb29raWUsICJCb29sZWFuIHR5cGVzIG5vdCBhbGxvd2VkIik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGFUeXBlID0gb3V0VmFsdWUtPlRZUEVfSU5UX0JPT0xFQU47Ci0gICAgICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEgPSAodWludDMyX3QpLTE7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBpZiAobGVuID09IDUpIHsKLSAgICAgICAgaWYgKChzWzBdID09ICdmJyB8fCBzWzBdID09ICdGJykgJiYKLSAgICAgICAgICAgIChzWzFdID09ICdhJyB8fCBzWzFdID09ICdBJykgJiYKLSAgICAgICAgICAgIChzWzJdID09ICdsJyB8fCBzWzJdID09ICdMJykgJiYKLSAgICAgICAgICAgIChzWzNdID09ICdzJyB8fCBzWzNdID09ICdTJykgJiYKLSAgICAgICAgICAgIChzWzRdID09ICdlJyB8fCBzWzRdID09ICdFJykpIHsKLSAgICAgICAgICAgIGlmICgoYXR0clR5cGUmUmVzVGFibGVfbWFwOjpUWVBFX0JPT0xFQU4pID09IDApIHsKLSAgICAgICAgICAgICAgICBpZiAoIWNhblN0cmluZ0NvZXJjZSkgewotICAgICAgICAgICAgICAgICAgICBpZiAoYWNjZXNzb3IgIT0gTlVMTCkgewotICAgICAgICAgICAgICAgICAgICAgICAgYWNjZXNzb3ItPnJlcG9ydEVycm9yKGFjY2Vzc29yQ29va2llLCAiQm9vbGVhbiB0eXBlcyBub3QgYWxsb3dlZCIpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIG91dFZhbHVlLT5kYXRhVHlwZSA9IG91dFZhbHVlLT5UWVBFX0lOVF9CT09MRUFOOwotICAgICAgICAgICAgICAgIG91dFZhbHVlLT5kYXRhID0gMDsKLSAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLQotICAgIGlmICgoYXR0clR5cGUmUmVzVGFibGVfbWFwOjpUWVBFX0VOVU0pICE9IDApIHsKLSAgICAgICAgY29uc3Qgc3NpemVfdCBwID0gZ2V0UmVzb3VyY2VQYWNrYWdlSW5kZXgoYXR0cklEKTsKLSAgICAgICAgY29uc3QgYmFnX2VudHJ5KiBiYWc7Ci0gICAgICAgIHNzaXplX3QgY250ID0gcCA+PSAwID8gbG9ja0JhZyhhdHRySUQsICZiYWcpIDogLTE7Ci0gICAgICAgIC8vcHJpbnRmKCJHb3QgJWQgZm9yIGVudW1cbiIsIGNudCk7Ci0gICAgICAgIGlmIChjbnQgPj0gMCkgewotICAgICAgICAgICAgcmVzb3VyY2VfbmFtZSBybmFtZTsKLSAgICAgICAgICAgIHdoaWxlIChjbnQgPiAwKSB7Ci0gICAgICAgICAgICAgICAgaWYgKCFSZXNfSU5URVJOQUxJRChiYWctPm1hcC5uYW1lLmlkZW50KSkgewotICAgICAgICAgICAgICAgICAgICAvL3ByaW50ZigiVHJ5aW5nIGF0dHIgIyUwOHhcbiIsIGJhZy0+bWFwLm5hbWUuaWRlbnQpOwotICAgICAgICAgICAgICAgICAgICBpZiAoZ2V0UmVzb3VyY2VOYW1lKGJhZy0+bWFwLm5hbWUuaWRlbnQsICZybmFtZSkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICNpZiAwCi0gICAgICAgICAgICAgICAgICAgICAgICBwcmludGYoIk1hdGNoaW5nICVzIGFnYWluc3QgJXMgKDB4JTA4eClcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nOChzLCBsZW4pLnN0cmluZygpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgocm5hbWUubmFtZSwgcm5hbWUubmFtZUxlbikuc3RyaW5nKCksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYmFnLT5tYXAubmFtZS5pZGVudCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAjZW5kaWYKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChzdHJ6Y21wMTYocywgbGVuLCBybmFtZS5uYW1lLCBybmFtZS5uYW1lTGVuKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGFUeXBlID0gYmFnLT5tYXAudmFsdWUuZGF0YVR5cGU7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGEgPSBiYWctPm1hcC52YWx1ZS5kYXRhOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVubG9ja0JhZyhiYWcpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgCi0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGJhZysrOwotICAgICAgICAgICAgICAgIGNudC0tOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgdW5sb2NrQmFnKGJhZyk7Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoZnJvbUFjY2Vzc29yKSB7Ci0gICAgICAgICAgICBpZiAoYWNjZXNzb3ItPmdldEF0dHJpYnV0ZUVudW0oYXR0cklELCBzLCBsZW4sIG91dFZhbHVlKSkgewotICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgaWYgKChhdHRyVHlwZSZSZXNUYWJsZV9tYXA6OlRZUEVfRkxBR1MpICE9IDApIHsKLSAgICAgICAgY29uc3Qgc3NpemVfdCBwID0gZ2V0UmVzb3VyY2VQYWNrYWdlSW5kZXgoYXR0cklEKTsKLSAgICAgICAgY29uc3QgYmFnX2VudHJ5KiBiYWc7Ci0gICAgICAgIHNzaXplX3QgY250ID0gcCA+PSAwID8gbG9ja0JhZyhhdHRySUQsICZiYWcpIDogLTE7Ci0gICAgICAgIC8vcHJpbnRmKCJHb3QgJWQgZm9yIGZsYWdzXG4iLCBjbnQpOwotICAgICAgICBpZiAoY250ID49IDApIHsKLSAgICAgICAgICAgIGJvb2wgZmFpbGVkID0gZmFsc2U7Ci0gICAgICAgICAgICByZXNvdXJjZV9uYW1lIHJuYW1lOwotICAgICAgICAgICAgb3V0VmFsdWUtPmRhdGFUeXBlID0gUmVzX3ZhbHVlOjpUWVBFX0lOVF9IRVg7Ci0gICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YSA9IDA7Ci0gICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogZW5kID0gcyArIGxlbjsKLSAgICAgICAgICAgIGNvbnN0IGNoYXIxNl90KiBwb3MgPSBzOwotICAgICAgICAgICAgd2hpbGUgKHBvcyA8IGVuZCAmJiAhZmFpbGVkKSB7Ci0gICAgICAgICAgICAgICAgY29uc3QgY2hhcjE2X3QqIHN0YXJ0ID0gcG9zOwotICAgICAgICAgICAgICAgIGVuZCsrOwotICAgICAgICAgICAgICAgIHdoaWxlIChwb3MgPCBlbmQgJiYgKnBvcyAhPSAnfCcpIHsKLSAgICAgICAgICAgICAgICAgICAgcG9zKys7Ci0gICAgICAgICAgICAgICAgfQotCQkJCS8vcHJpbnRmKCJMb29raW5nIGZvcjogJXNcbiIsIFN0cmluZzgoc3RhcnQsIHBvcy1zdGFydCkuc3RyaW5nKCkpOwotICAgICAgICAgICAgICAgIGNvbnN0IGJhZ19lbnRyeSogYmFnaSA9IGJhZzsKLQkJCQlzc2l6ZV90IGk7Ci0gICAgICAgICAgICAgICAgZm9yIChpPTA7IGk8Y250OyBpKyssIGJhZ2krKykgewotICAgICAgICAgICAgICAgICAgICBpZiAoIVJlc19JTlRFUk5BTElEKGJhZ2ktPm1hcC5uYW1lLmlkZW50KSkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy9wcmludGYoIlRyeWluZyBhdHRyICMlMDh4XG4iLCBiYWdpLT5tYXAubmFtZS5pZGVudCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoZ2V0UmVzb3VyY2VOYW1lKGJhZ2ktPm1hcC5uYW1lLmlkZW50LCAmcm5hbWUpKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgI2lmIDAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBwcmludGYoIk1hdGNoaW5nICVzIGFnYWluc3QgJXMgKDB4JTA4eClcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZzgoc3RhcnQscG9zLXN0YXJ0KS5zdHJpbmcoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nOChybmFtZS5uYW1lLCBybmFtZS5uYW1lTGVuKS5zdHJpbmcoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYmFnaS0+bWFwLm5hbWUuaWRlbnQpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICNlbmRpZgotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChzdHJ6Y21wMTYoc3RhcnQsIHBvcy1zdGFydCwgcm5hbWUubmFtZSwgcm5hbWUubmFtZUxlbikgPT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvdXRWYWx1ZS0+ZGF0YSB8PSBiYWdpLT5tYXAudmFsdWUuZGF0YTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChpID49IGNudCkgewotICAgICAgICAgICAgICAgICAgICAvLyBEaWRuJ3QgZmluZCB0aGlzIGZsYWcgaWRlbnRpZmllci4KLSAgICAgICAgICAgICAgICAgICAgZmFpbGVkID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgaWYgKHBvcyA8IGVuZCkgewotICAgICAgICAgICAgICAgICAgICBwb3MrKzsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB1bmxvY2tCYWcoYmFnKTsKLSAgICAgICAgICAgIGlmICghZmFpbGVkKSB7Ci0JCQkJLy9wcmludGYoIkZpbmFsIGZsYWcgdmFsdWU6IDB4JWx4XG4iLCBvdXRWYWx1ZS0+ZGF0YSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotCi0gICAgICAgIGlmIChmcm9tQWNjZXNzb3IpIHsKLSAgICAgICAgICAgIGlmIChhY2Nlc3Nvci0+Z2V0QXR0cmlidXRlRmxhZ3MoYXR0cklELCBzLCBsZW4sIG91dFZhbHVlKSkgewotCQkJCS8vcHJpbnRmKCJGaW5hbCBmbGFnIHZhbHVlOiAweCVseFxuIiwgb3V0VmFsdWUtPmRhdGEpOwotICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgaWYgKChhdHRyVHlwZSZSZXNUYWJsZV9tYXA6OlRZUEVfU1RSSU5HKSA9PSAwKSB7Ci0gICAgICAgIGlmIChhY2Nlc3NvciAhPSBOVUxMKSB7Ci0gICAgICAgICAgICBhY2Nlc3Nvci0+cmVwb3J0RXJyb3IoYWNjZXNzb3JDb29raWUsICJTdHJpbmcgdHlwZXMgbm90IGFsbG93ZWQiKTsKLSAgICAgICAgfQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotCi0gICAgLy8gR2VuZXJpYyBzdHJpbmcgaGFuZGxpbmcuLi4KLSAgICBvdXRWYWx1ZS0+ZGF0YVR5cGUgPSBvdXRWYWx1ZS0+VFlQRV9TVFJJTkc7Ci0gICAgaWYgKG91dFN0cmluZykgewotICAgICAgICBib29sIGZhaWxlZCA9IGNvbGxlY3RTdHJpbmcob3V0U3RyaW5nLCBzLCBsZW4sIHByZXNlcnZlU3BhY2VzLCAmZXJyb3JNc2cpOwotICAgICAgICBpZiAoYWNjZXNzb3IgIT0gTlVMTCkgewotICAgICAgICAgICAgYWNjZXNzb3ItPnJlcG9ydEVycm9yKGFjY2Vzc29yQ29va2llLCBlcnJvck1zZyk7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGZhaWxlZDsKLSAgICB9Ci0KLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotYm9vbCBSZXNUYWJsZTo6Y29sbGVjdFN0cmluZyhTdHJpbmcxNiogb3V0U3RyaW5nLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyMTZfdCogcywgc2l6ZV90IGxlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYm9vbCBwcmVzZXJ2ZVNwYWNlcywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY2hhcioqIG91dEVycm9yTXNnLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICBib29sIGFwcGVuZCkKLXsKLSAgICBTdHJpbmcxNiB0bXA7Ci0KLSAgICBjaGFyIHF1b3RlZCA9IDA7Ci0gICAgY29uc3QgY2hhcjE2X3QqIHAgPSBzOwotICAgIHdoaWxlIChwIDwgKHMrbGVuKSkgewotICAgICAgICB3aGlsZSAocCA8IChzK2xlbikpIHsKLSAgICAgICAgICAgIGNvbnN0IGNoYXIxNl90IGMgPSAqcDsKLSAgICAgICAgICAgIGlmIChjID09ICdcXCcpIHsKLSAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmICghcHJlc2VydmVTcGFjZXMpIHsKLSAgICAgICAgICAgICAgICBpZiAocXVvdGVkID09IDAgJiYgaXNzcGFjZTE2KGMpCi0gICAgICAgICAgICAgICAgICAgICYmIChjICE9ICcgJyB8fCBpc3NwYWNlMTYoKihwKzEpKSkpIHsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChjID09ICciJyAmJiAocXVvdGVkID09IDAgfHwgcXVvdGVkID09ICciJykpIHsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChjID09ICdcJycgJiYgKHF1b3RlZCA9PSAwIHx8IHF1b3RlZCA9PSAnXCcnKSkgewotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwKys7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHAgPCAocytsZW4pKSB7Ci0gICAgICAgICAgICBpZiAocCA+IHMpIHsKLSAgICAgICAgICAgICAgICB0bXAuYXBwZW5kKFN0cmluZzE2KHMsIHAtcykpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKCFwcmVzZXJ2ZVNwYWNlcyAmJiAoKnAgPT0gJyInIHx8ICpwID09ICdcJycpKSB7Ci0gICAgICAgICAgICAgICAgaWYgKHF1b3RlZCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIHF1b3RlZCA9ICpwOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHF1b3RlZCA9IDA7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHArKzsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoIXByZXNlcnZlU3BhY2VzICYmIGlzc3BhY2UxNigqcCkpIHsKLSAgICAgICAgICAgICAgICAvLyBTcGFjZSBvdXRzaWRlIG9mIGEgcXVvdGUgLS0gY29uc3VtZSBhbGwgc3BhY2VzIGFuZAotICAgICAgICAgICAgICAgIC8vIGxlYXZlIGEgc2luZ2xlIHBsYWluIHNwYWNlIGNoYXIuCi0gICAgICAgICAgICAgICAgdG1wLmFwcGVuZChTdHJpbmcxNigiICIpKTsKLSAgICAgICAgICAgICAgICBwKys7Ci0gICAgICAgICAgICAgICAgd2hpbGUgKHAgPCAocytsZW4pICYmIGlzc3BhY2UxNigqcCkpIHsKLSAgICAgICAgICAgICAgICAgICAgcCsrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSBpZiAoKnAgPT0gJ1xcJykgewotICAgICAgICAgICAgICAgIHArKzsKLSAgICAgICAgICAgICAgICBpZiAocCA8IChzK2xlbikpIHsKLSAgICAgICAgICAgICAgICAgICAgc3dpdGNoICgqcCkgewotICAgICAgICAgICAgICAgICAgICBjYXNlICd0JzoKLSAgICAgICAgICAgICAgICAgICAgICAgIHRtcC5hcHBlbmQoU3RyaW5nMTYoIlx0IikpOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgJ24nOgotICAgICAgICAgICAgICAgICAgICAgICAgdG1wLmFwcGVuZChTdHJpbmcxNigiXG4iKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSAnIyc6Ci0gICAgICAgICAgICAgICAgICAgICAgICB0bXAuYXBwZW5kKFN0cmluZzE2KCIjIikpOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgJ0AnOgotICAgICAgICAgICAgICAgICAgICAgICAgdG1wLmFwcGVuZChTdHJpbmcxNigiQCIpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlICc/JzoKLSAgICAgICAgICAgICAgICAgICAgICAgIHRtcC5hcHBlbmQoU3RyaW5nMTYoIj8iKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgY2FzZSAnIic6Ci0gICAgICAgICAgICAgICAgICAgICAgICB0bXAuYXBwZW5kKFN0cmluZzE2KCJcIiIpKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgICAgICBjYXNlICdcJyc6Ci0gICAgICAgICAgICAgICAgICAgICAgICB0bXAuYXBwZW5kKFN0cmluZzE2KCInIikpOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgJ1xcJzoKLSAgICAgICAgICAgICAgICAgICAgICAgIHRtcC5hcHBlbmQoU3RyaW5nMTYoIlxcIikpOwotICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGNhc2UgJ3UnOgotICAgICAgICAgICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjaGFyMTZfdCBjaHIgPSAwOwotICAgICAgICAgICAgICAgICAgICAgICAgaW50IGkgPSAwOwotICAgICAgICAgICAgICAgICAgICAgICAgd2hpbGUgKGkgPCA0ICYmIHBbMV0gIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHArKzsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpKys7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGM7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCpwID49ICcwJyAmJiAqcCA8PSAnOScpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYyA9ICpwIC0gJzAnOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoKnAgPj0gJ2EnICYmICpwIDw9ICdmJykgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjID0gKnAgLSAnYScgKyAxMDsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKCpwID49ICdBJyAmJiAqcCA8PSAnRicpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYyA9ICpwIC0gJ0EnICsgMTA7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKG91dEVycm9yTXNnKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqb3V0RXJyb3JNc2cgPSAiQmFkIGNoYXJhY3RlciBpbiBcXHUgdW5pY29kZSBlc2NhcGUgc2VxdWVuY2UiOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2hyID0gKGNocjw8NCkgfCBjOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgdG1wLmFwcGVuZChTdHJpbmcxNigmY2hyLCAxKSk7Ci0gICAgICAgICAgICAgICAgICAgIH0gYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyBpZ25vcmUgdW5rbm93biBlc2NhcGUgY2hhcnMuCi0gICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBwKys7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgbGVuIC09IChwLXMpOwotICAgICAgICAgICAgcyA9IHA7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBpZiAodG1wLnNpemUoKSAhPSAwKSB7Ci0gICAgICAgIGlmIChsZW4gPiAwKSB7Ci0gICAgICAgICAgICB0bXAuYXBwZW5kKFN0cmluZzE2KHMsIGxlbikpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChhcHBlbmQpIHsKLSAgICAgICAgICAgIG91dFN0cmluZy0+YXBwZW5kKHRtcCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBvdXRTdHJpbmctPnNldFRvKHRtcCk7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBpZiAoYXBwZW5kKSB7Ci0gICAgICAgICAgICBvdXRTdHJpbmctPmFwcGVuZChTdHJpbmcxNihzLCBsZW4pKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIG91dFN0cmluZy0+c2V0VG8ocywgbGVuKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiB0cnVlOwotfQotCi1zaXplX3QgUmVzVGFibGU6OmdldEJhc2VQYWNrYWdlQ291bnQoKSBjb25zdAotewotICAgIGlmIChtRXJyb3IgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotICAgIHJldHVybiBtUGFja2FnZUdyb3Vwcy5zaXplKCk7Ci19Ci0KLWNvbnN0IGNoYXIxNl90KiBSZXNUYWJsZTo6Z2V0QmFzZVBhY2thZ2VOYW1lKHNpemVfdCBpZHgpIGNvbnN0Ci17Ci0gICAgaWYgKG1FcnJvciAhPSBOT19FUlJPUikgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0gICAgTE9HX0ZBVEFMX0lGKGlkeCA+PSBtUGFja2FnZUdyb3Vwcy5zaXplKCksCi0gICAgICAgICAgICAgICAgICJSZXF1ZXN0ZWQgcGFja2FnZSBpbmRleCAlZCBwYXN0IHBhY2thZ2UgY291bnQgJWQiLAotICAgICAgICAgICAgICAgICAoaW50KWlkeCwgKGludCltUGFja2FnZUdyb3Vwcy5zaXplKCkpOwotICAgIHJldHVybiBtUGFja2FnZUdyb3Vwc1tpZHhdLT5uYW1lLnN0cmluZygpOwotfQotCi11aW50MzJfdCBSZXNUYWJsZTo6Z2V0QmFzZVBhY2thZ2VJZChzaXplX3QgaWR4KSBjb25zdAotewotICAgIGlmIChtRXJyb3IgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotICAgIExPR19GQVRBTF9JRihpZHggPj0gbVBhY2thZ2VHcm91cHMuc2l6ZSgpLAotICAgICAgICAgICAgICAgICAiUmVxdWVzdGVkIHBhY2thZ2UgaW5kZXggJWQgcGFzdCBwYWNrYWdlIGNvdW50ICVkIiwKLSAgICAgICAgICAgICAgICAgKGludClpZHgsIChpbnQpbVBhY2thZ2VHcm91cHMuc2l6ZSgpKTsKLSAgICByZXR1cm4gbVBhY2thZ2VHcm91cHNbaWR4XS0+aWQ7Ci19Ci0KLXNpemVfdCBSZXNUYWJsZTo6Z2V0VGFibGVDb3VudCgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1IZWFkZXJzLnNpemUoKTsKLX0KLQotY29uc3QgUmVzU3RyaW5nUG9vbCogUmVzVGFibGU6OmdldFRhYmxlU3RyaW5nQmxvY2soc2l6ZV90IGluZGV4KSBjb25zdAotewotICAgIHJldHVybiAmbUhlYWRlcnNbaW5kZXhdLT52YWx1ZXM7Ci19Ci0KLXZvaWQqIFJlc1RhYmxlOjpnZXRUYWJsZUNvb2tpZShzaXplX3QgaW5kZXgpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIG1IZWFkZXJzW2luZGV4XS0+Y29va2llOwotfQotCi12b2lkIFJlc1RhYmxlOjpnZXRDb25maWd1cmF0aW9ucyhWZWN0b3I8UmVzVGFibGVfY29uZmlnPiogY29uZmlncykgY29uc3QKLXsKLSAgICBjb25zdCBzaXplX3QgSSA9IG1QYWNrYWdlR3JvdXBzLnNpemUoKTsKLSAgICBmb3IgKHNpemVfdCBpPTA7IGk8STsgaSsrKSB7Ci0gICAgICAgIGNvbnN0IFBhY2thZ2VHcm91cCogcGFja2FnZUdyb3VwID0gbVBhY2thZ2VHcm91cHNbaV07Ci0gICAgICAgIGNvbnN0IHNpemVfdCBKID0gcGFja2FnZUdyb3VwLT5wYWNrYWdlcy5zaXplKCk7Ci0gICAgICAgIGZvciAoc2l6ZV90IGo9MDsgajxKOyBqKyspIHsKLSAgICAgICAgICAgIGNvbnN0IFBhY2thZ2UqIHBhY2thZ2UgPSBwYWNrYWdlR3JvdXAtPnBhY2thZ2VzW2pdOwotICAgICAgICAgICAgY29uc3Qgc2l6ZV90IEsgPSBwYWNrYWdlLT50eXBlcy5zaXplKCk7Ci0gICAgICAgICAgICBmb3IgKHNpemVfdCBrPTA7IGs8SzsgaysrKSB7Ci0gICAgICAgICAgICAgICAgY29uc3QgVHlwZSogdHlwZSA9IHBhY2thZ2UtPnR5cGVzW2tdOwotICAgICAgICAgICAgICAgIGlmICh0eXBlID09IE5VTEwpIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgIGNvbnN0IHNpemVfdCBMID0gdHlwZS0+Y29uZmlncy5zaXplKCk7Ci0gICAgICAgICAgICAgICAgZm9yIChzaXplX3QgbD0wOyBsPEw7IGwrKykgewotICAgICAgICAgICAgICAgICAgICBjb25zdCBSZXNUYWJsZV90eXBlKiBjb25maWcgPSB0eXBlLT5jb25maWdzW2xdOwotICAgICAgICAgICAgICAgICAgICBjb25zdCBSZXNUYWJsZV9jb25maWcqIGNmZyA9ICZjb25maWctPmNvbmZpZzsKLSAgICAgICAgICAgICAgICAgICAgLy8gb25seSBpbnNlcnQgdW5pcXVlCi0gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNpemVfdCBNID0gY29uZmlncy0+c2l6ZSgpOwotICAgICAgICAgICAgICAgICAgICBzaXplX3QgbTsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChtPTA7IG08TTsgbSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoMCA9PSAoKmNvbmZpZ3MpW21dLmNvbXBhcmUoKmNmZykpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAvLyBpZiB3ZSBkaWRuJ3QgZmluZCBpdAotICAgICAgICAgICAgICAgICAgICBpZiAobSA9PSBNKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb25maWdzLT5hZGQoKmNmZyk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLXZvaWQgUmVzVGFibGU6OmdldExvY2FsZXMoVmVjdG9yPFN0cmluZzg+KiBsb2NhbGVzKSBjb25zdAotewotICAgIFZlY3RvcjxSZXNUYWJsZV9jb25maWc+IGNvbmZpZ3M7Ci0gICAgTE9HRCgiY2FsbGluZyBnZXRDb25maWd1cmF0aW9ucyIpOwotICAgIGdldENvbmZpZ3VyYXRpb25zKCZjb25maWdzKTsKLSAgICBMT0dEKCJjYWxsZWQgZ2V0Q29uZmlndXJhdGlvbnMgc2l6ZT0lZCIsIChpbnQpY29uZmlncy5zaXplKCkpOwotICAgIGNvbnN0IHNpemVfdCBJID0gY29uZmlncy5zaXplKCk7Ci0gICAgZm9yIChzaXplX3QgaT0wOyBpPEk7IGkrKykgewotICAgICAgICBjaGFyIGxvY2FsZVs2XTsKLSAgICAgICAgY29uZmlnc1tpXS5nZXRMb2NhbGUobG9jYWxlKTsKLSAgICAgICAgY29uc3Qgc2l6ZV90IEogPSBsb2NhbGVzLT5zaXplKCk7Ci0gICAgICAgIHNpemVfdCBqOwotICAgICAgICBmb3IgKGo9MDsgajxKOyBqKyspIHsKLSAgICAgICAgICAgIGlmICgwID09IHN0cmNtcChsb2NhbGUsICgqbG9jYWxlcylbal0uc3RyaW5nKCkpKSB7Ci0gICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGogPT0gSikgewotICAgICAgICAgICAgbG9jYWxlcy0+YWRkKFN0cmluZzgobG9jYWxlKSk7Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLXNzaXplX3QgUmVzVGFibGU6OmdldEVudHJ5KAotICAgIGNvbnN0IFBhY2thZ2UqIHBhY2thZ2UsIGludCB0eXBlSW5kZXgsIGludCBlbnRyeUluZGV4LAotICAgIGNvbnN0IFJlc1RhYmxlX2NvbmZpZyogY29uZmlnLAotICAgIGNvbnN0IFJlc1RhYmxlX3R5cGUqKiBvdXRUeXBlLCBjb25zdCBSZXNUYWJsZV9lbnRyeSoqIG91dEVudHJ5LAotICAgIGNvbnN0IFR5cGUqKiBvdXRUeXBlQ2xhc3MpIGNvbnN0Ci17Ci0gICAgTE9HVigiR2V0dGluZyBlbnRyeSBmcm9tIHBhY2thZ2UgJXBcbiIsIHBhY2thZ2UpOwotICAgIGNvbnN0IFJlc1RhYmxlX3BhY2thZ2UqIGNvbnN0IHBrZyA9IHBhY2thZ2UtPnBhY2thZ2U7Ci0KLSAgICBjb25zdCBUeXBlKiBhbGxUeXBlcyA9IHBhY2thZ2UtPmdldFR5cGUodHlwZUluZGV4KTsKLSAgICBMT0dWKCJhbGxUeXBlcz0lcFxuIiwgYWxsVHlwZXMpOwotICAgIGlmIChhbGxUeXBlcyA9PSBOVUxMKSB7Ci0gICAgICAgIExPR1YoIlNraXBwaW5nIGVudHJ5IHR5cGUgaW5kZXggMHglMDJ4IGJlY2F1c2UgdHlwZSBpcyBOVUxMIVxuIiwgdHlwZUluZGV4KTsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgaWYgKChzaXplX3QpZW50cnlJbmRleCA+PSBhbGxUeXBlcy0+ZW50cnlDb3VudCkgewotICAgICAgICBMT0dXKCJnZXRFbnRyeSBmYWlsaW5nIGJlY2F1c2UgZW50cnlJbmRleCAlZCBpcyBiZXlvbmQgdHlwZSBlbnRyeUNvdW50ICVkIiwKLSAgICAgICAgICAgIGVudHJ5SW5kZXgsIChpbnQpYWxsVHlwZXMtPmVudHJ5Q291bnQpOwotICAgICAgICByZXR1cm4gQkFEX1RZUEU7Ci0gICAgfQotICAgICAgICAKLSAgICBjb25zdCBSZXNUYWJsZV90eXBlKiB0eXBlID0gTlVMTDsKLSAgICB1aW50MzJfdCBvZmZzZXQgPSBSZXNUYWJsZV90eXBlOjpOT19FTlRSWTsKLSAgICBSZXNUYWJsZV9jb25maWcgYmVzdENvbmZpZzsKLSAgICBtZW1zZXQoJmJlc3RDb25maWcsIDAsIHNpemVvZihiZXN0Q29uZmlnKSk7IC8vIG1ha2UgdGhlIGNvbXBpbGVyIHNodXQgdXAKLSAgICAKLSAgICBjb25zdCBzaXplX3QgTlQgPSBhbGxUeXBlcy0+Y29uZmlncy5zaXplKCk7Ci0gICAgZm9yIChzaXplX3QgaT0wOyBpPE5UOyBpKyspIHsKLSAgICAgICAgY29uc3QgUmVzVGFibGVfdHlwZSogY29uc3QgdGhpc1R5cGUgPSBhbGxUeXBlcy0+Y29uZmlnc1tpXTsKLSAgICAgICAgaWYgKHRoaXNUeXBlID09IE5VTEwpIGNvbnRpbnVlOwotICAgICAgICAKLSAgICAgICAgUmVzVGFibGVfY29uZmlnIHRoaXNDb25maWc7Ci0gICAgICAgIHRoaXNDb25maWcuY29weUZyb21EdG9IKHRoaXNUeXBlLT5jb25maWcpOwotCi0gICAgICAgIFRBQkxFX0dFVEVOVFJZKExPR0koIk1hdGNoIGVudHJ5IDB4JXggaW4gdHlwZSAweCV4IChzeiAweCV4KTogaW1zaTolZC8lZD0lZC8lZCBsYW5nOiVjJWM9JWMlYyBjbnQ6JWMlYz0lYyVjICIKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAib3JpZW46JWQ9JWQgdG91Y2g6JWQ9JWQgZGVuc2l0eTolZD0lZCBrZXk6JWQ9JWQgaW5wOiVkPSVkIG5hdjolZD0lZCB3OiVkPSVkIGg6JWQ9JWRcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBlbnRyeUluZGV4LCB0eXBlSW5kZXgrMSwgZHRvaGwodGhpc1R5cGUtPmNvbmZpZy5zaXplKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXNDb25maWcubWNjLCB0aGlzQ29uZmlnLm1uYywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmZpZyA/IGNvbmZpZy0+bWNjIDogMCwgY29uZmlnID8gY29uZmlnLT5tbmMgOiAwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy5sYW5ndWFnZVswXSA/IHRoaXNDb25maWcubGFuZ3VhZ2VbMF0gOiAnLScsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzQ29uZmlnLmxhbmd1YWdlWzFdID8gdGhpc0NvbmZpZy5sYW5ndWFnZVsxXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmZpZyAmJiBjb25maWctPmxhbmd1YWdlWzBdID8gY29uZmlnLT5sYW5ndWFnZVswXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmZpZyAmJiBjb25maWctPmxhbmd1YWdlWzFdID8gY29uZmlnLT5sYW5ndWFnZVsxXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXNDb25maWcuY291bnRyeVswXSA/IHRoaXNDb25maWcuY291bnRyeVswXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXNDb25maWcuY291bnRyeVsxXSA/IHRoaXNDb25maWcuY291bnRyeVsxXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmZpZyAmJiBjb25maWctPmNvdW50cnlbMF0gPyBjb25maWctPmNvdW50cnlbMF0gOiAnLScsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBjb25maWcgJiYgY29uZmlnLT5jb3VudHJ5WzFdID8gY29uZmlnLT5jb3VudHJ5WzFdIDogJy0nLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy5vcmllbnRhdGlvbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmZpZyA/IGNvbmZpZy0+b3JpZW50YXRpb24gOiAwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy50b3VjaHNjcmVlbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmZpZyA/IGNvbmZpZy0+dG91Y2hzY3JlZW4gOiAwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy5kZW5zaXR5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uZmlnID8gY29uZmlnLT5kZW5zaXR5IDogMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXNDb25maWcua2V5Ym9hcmQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBjb25maWcgPyBjb25maWctPmtleWJvYXJkIDogMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXNDb25maWcuaW5wdXRGbGFncywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmZpZyA/IGNvbmZpZy0+aW5wdXRGbGFncyA6IDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzQ29uZmlnLm5hdmlnYXRpb24sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBjb25maWcgPyBjb25maWctPm5hdmlnYXRpb24gOiAwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy5zY3JlZW5XaWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmZpZyA/IGNvbmZpZy0+c2NyZWVuV2lkdGggOiAwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy5zY3JlZW5IZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBjb25maWcgPyBjb25maWctPnNjcmVlbkhlaWdodCA6IDApKTsKLSAgICAgICAgCi0gICAgICAgIC8vIENoZWNrIHRvIG1ha2Ugc3VyZSB0aGlzIG9uZSBpcyB2YWxpZCBmb3IgdGhlIGN1cnJlbnQgcGFyYW1ldGVycy4KLSAgICAgICAgaWYgKGNvbmZpZyAmJiAhdGhpc0NvbmZpZy5tYXRjaCgqY29uZmlnKSkgewotICAgICAgICAgICAgVEFCTEVfR0VURU5UUlkoTE9HSSgiRG9lcyBub3QgbWF0Y2ggY29uZmlnIVxuIikpOwotICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIC8vIENoZWNrIGlmIHRoZXJlIGlzIHRoZSBkZXNpcmVkIGVudHJ5IGluIHRoaXMgdHlwZS4KLSAgICAgICAgCi0gICAgICAgIGNvbnN0IHVpbnQ4X3QqIGNvbnN0IGVuZCA9ICgoY29uc3QgdWludDhfdCopdGhpc1R5cGUpCi0gICAgICAgICAgICArIGR0b2hsKHRoaXNUeXBlLT5oZWFkZXIuc2l6ZSk7Ci0gICAgICAgIGNvbnN0IHVpbnQzMl90KiBjb25zdCBlaW5kZXggPSAoY29uc3QgdWludDMyX3QqKQotICAgICAgICAgICAgKCgoY29uc3QgdWludDhfdCopdGhpc1R5cGUpICsgZHRvaHModGhpc1R5cGUtPmhlYWRlci5oZWFkZXJTaXplKSk7Ci0gICAgICAgIAotICAgICAgICB1aW50MzJfdCB0aGlzT2Zmc2V0ID0gZHRvaGwoZWluZGV4W2VudHJ5SW5kZXhdKTsKLSAgICAgICAgaWYgKHRoaXNPZmZzZXQgPT0gUmVzVGFibGVfdHlwZTo6Tk9fRU5UUlkpIHsKLSAgICAgICAgICAgIFRBQkxFX0dFVEVOVFJZKExPR0koIlNraXBwaW5nIGJlY2F1c2UgaXQgaXMgbm90IGRlZmluZWQhXG4iKSk7Ci0gICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgaWYgKHR5cGUgIT0gTlVMTCkgewotICAgICAgICAgICAgLy8gQ2hlY2sgaWYgdGhpcyBvbmUgaXMgbGVzcyBzcGVjaWZpYyB0aGFuIHRoZSBsYXN0IGZvdW5kLiAgSWYgc28sCi0gICAgICAgICAgICAvLyB3ZSB3aWxsIHNraXAgaXQuICBXZSBjaGVjayBzdGFydGluZyB3aXRoIHRoaW5ncyB3ZSBtb3N0IGNhcmUKLSAgICAgICAgICAgIC8vIGFib3V0IHRvIHRob3NlIHdlIGxlYXN0IGNhcmUgYWJvdXQuCi0gICAgICAgICAgICBpZiAoIXRoaXNDb25maWcuaXNCZXR0ZXJUaGFuKGJlc3RDb25maWcsIGNvbmZpZykpIHsKLSAgICAgICAgICAgICAgICBUQUJMRV9HRVRFTlRSWShMT0dJKCJUaGlzIGNvbmZpZyBpcyB3b3JzZSB0aGFuIGxhc3QhXG4iKSk7Ci0gICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIHR5cGUgPSB0aGlzVHlwZTsKLSAgICAgICAgb2Zmc2V0ID0gdGhpc09mZnNldDsKLSAgICAgICAgYmVzdENvbmZpZyA9IHRoaXNDb25maWc7Ci0gICAgICAgIFRBQkxFX0dFVEVOVFJZKExPR0koIkJlc3QgZW50cnkgc28gZmFyIC0tIHVzaW5nIGl0IVxuIikpOwotICAgICAgICBpZiAoIWNvbmZpZykgYnJlYWs7Ci0gICAgfQotICAgIAotICAgIGlmICh0eXBlID09IE5VTEwpIHsKLSAgICAgICAgVEFCTEVfR0VURU5UUlkoTE9HSSgiTm8gdmFsdWUgZm91bmQgZm9yIHJlcXVlc3RlZCBlbnRyeSFcbiIpKTsKLSAgICAgICAgcmV0dXJuIEJBRF9JTkRFWDsKLSAgICB9Ci0gICAgCi0gICAgb2Zmc2V0ICs9IGR0b2hsKHR5cGUtPmVudHJpZXNTdGFydCk7Ci0gICAgVEFCTEVfTk9JU1koYW91dCA8PCAiTG9va2luZyBpbiByZXNvdXJjZSB0YWJsZSAiIDw8IHBhY2thZ2UtPmhlYWRlci0+aGVhZGVyCi0gICAgICAgICAgPDwgIiwgdHlwZU9mZj0iCi0gICAgICAgICAgPDwgKHZvaWQqKSgoKGNvbnN0IGNoYXIqKXR5cGUpLSgoY29uc3QgY2hhciopcGFja2FnZS0+aGVhZGVyLT5oZWFkZXIpKQotICAgICAgICAgIDw8ICIsIG9mZnNldD0iIDw8ICh2b2lkKilvZmZzZXQgPDwgZW5kbCk7Ci0KLSAgICBpZiAob2Zmc2V0ID4gKGR0b2hsKHR5cGUtPmhlYWRlci5zaXplKS1zaXplb2YoUmVzVGFibGVfZW50cnkpKSkgewotICAgICAgICBMT0dXKCJSZXNUYWJsZV9lbnRyeSBhdCAweCV4IGlzIGJleW9uZCB0eXBlIGNodW5rIGRhdGEgMHgleCIsCi0gICAgICAgICAgICAgb2Zmc2V0LCBkdG9obCh0eXBlLT5oZWFkZXIuc2l6ZSkpOwotICAgICAgICByZXR1cm4gQkFEX1RZUEU7Ci0gICAgfQotICAgIGlmICgob2Zmc2V0JjB4MykgIT0gMCkgewotICAgICAgICBMT0dXKCJSZXNUYWJsZV9lbnRyeSBhdCAweCV4IGlzIG5vdCBvbiBhbiBpbnRlZ2VyIGJvdW5kYXJ5IiwKLSAgICAgICAgICAgICBvZmZzZXQpOwotICAgICAgICByZXR1cm4gQkFEX1RZUEU7Ci0gICAgfQotCi0gICAgY29uc3QgUmVzVGFibGVfZW50cnkqIGNvbnN0IGVudHJ5ID0gKGNvbnN0IFJlc1RhYmxlX2VudHJ5KikKLSAgICAgICAgKCgoY29uc3QgdWludDhfdCopdHlwZSkgKyBvZmZzZXQpOwotICAgIGlmIChkdG9ocyhlbnRyeS0+c2l6ZSkgPCBzaXplb2YoKmVudHJ5KSkgewotICAgICAgICBMT0dXKCJSZXNUYWJsZV9lbnRyeSBzaXplIDB4JXggaXMgdG9vIHNtYWxsIiwgZHRvaHMoZW50cnktPnNpemUpKTsKLSAgICAgICAgcmV0dXJuIEJBRF9UWVBFOwotICAgIH0KLQotICAgICpvdXRUeXBlID0gdHlwZTsKLSAgICAqb3V0RW50cnkgPSBlbnRyeTsKLSAgICBpZiAob3V0VHlwZUNsYXNzICE9IE5VTEwpIHsKLSAgICAgICAgKm91dFR5cGVDbGFzcyA9IGFsbFR5cGVzOwotICAgIH0KLSAgICByZXR1cm4gb2Zmc2V0ICsgZHRvaHMoZW50cnktPnNpemUpOwotfQotCi1zdGF0dXNfdCBSZXNUYWJsZTo6cGFyc2VQYWNrYWdlKGNvbnN0IFJlc1RhYmxlX3BhY2thZ2UqIGNvbnN0IHBrZywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgSGVhZGVyKiBjb25zdCBoZWFkZXIpCi17Ci0gICAgY29uc3QgdWludDhfdCogYmFzZSA9IChjb25zdCB1aW50OF90Kilwa2c7Ci0gICAgc3RhdHVzX3QgZXJyID0gdmFsaWRhdGVfY2h1bmsoJnBrZy0+aGVhZGVyLCBzaXplb2YoKnBrZyksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaGVhZGVyLT5kYXRhRW5kLCAiUmVzVGFibGVfcGFja2FnZSIpOwotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgcmV0dXJuIChtRXJyb3I9ZXJyKTsKLSAgICB9Ci0KLSAgICBjb25zdCBzaXplX3QgcGtnU2l6ZSA9IGR0b2hsKHBrZy0+aGVhZGVyLnNpemUpOwotCi0gICAgaWYgKGR0b2hsKHBrZy0+dHlwZVN0cmluZ3MpID49IHBrZ1NpemUpIHsKLSAgICAgICAgTE9HVygiUmVzVGFibGVfcGFja2FnZSB0eXBlIHN0cmluZ3MgYXQgJXAgYXJlIHBhc3QgY2h1bmsgc2l6ZSAlcC4iLAotICAgICAgICAgICAgICh2b2lkKilkdG9obChwa2ctPnR5cGVTdHJpbmdzKSwgKHZvaWQqKXBrZ1NpemUpOwotICAgICAgICByZXR1cm4gKG1FcnJvcj1CQURfVFlQRSk7Ci0gICAgfQotICAgIGlmICgoZHRvaGwocGtnLT50eXBlU3RyaW5ncykmMHgzKSAhPSAwKSB7Ci0gICAgICAgIExPR1coIlJlc1RhYmxlX3BhY2thZ2UgdHlwZSBzdHJpbmdzIGF0ICVwIGlzIG5vdCBvbiBhbiBpbnRlZ2VyIGJvdW5kYXJ5LiIsCi0gICAgICAgICAgICAgKHZvaWQqKWR0b2hsKHBrZy0+dHlwZVN0cmluZ3MpKTsKLSAgICAgICAgcmV0dXJuIChtRXJyb3I9QkFEX1RZUEUpOwotICAgIH0KLSAgICBpZiAoZHRvaGwocGtnLT5rZXlTdHJpbmdzKSA+PSBwa2dTaXplKSB7Ci0gICAgICAgIExPR1coIlJlc1RhYmxlX3BhY2thZ2Uga2V5IHN0cmluZ3MgYXQgJXAgYXJlIHBhc3QgY2h1bmsgc2l6ZSAlcC4iLAotICAgICAgICAgICAgICh2b2lkKilkdG9obChwa2ctPmtleVN0cmluZ3MpLCAodm9pZCopcGtnU2l6ZSk7Ci0gICAgICAgIHJldHVybiAobUVycm9yPUJBRF9UWVBFKTsKLSAgICB9Ci0gICAgaWYgKChkdG9obChwa2ctPmtleVN0cmluZ3MpJjB4MykgIT0gMCkgewotICAgICAgICBMT0dXKCJSZXNUYWJsZV9wYWNrYWdlIGtleSBzdHJpbmdzIGF0ICVwIGlzIG5vdCBvbiBhbiBpbnRlZ2VyIGJvdW5kYXJ5LiIsCi0gICAgICAgICAgICAgKHZvaWQqKWR0b2hsKHBrZy0+a2V5U3RyaW5ncykpOwotICAgICAgICByZXR1cm4gKG1FcnJvcj1CQURfVFlQRSk7Ci0gICAgfQotICAgIAotICAgIFBhY2thZ2UqIHBhY2thZ2UgPSBOVUxMOwotICAgIFBhY2thZ2VHcm91cCogZ3JvdXAgPSBOVUxMOwotICAgIHVpbnQzMl90IGlkID0gZHRvaGwocGtnLT5pZCk7Ci0gICAgaWYgKGlkICE9IDAgJiYgaWQgPCAyNTYpIHsKLSAgICAgICAgc2l6ZV90IGlkeCA9IG1QYWNrYWdlTWFwW2lkXTsKLSAgICAgICAgaWYgKGlkeCA9PSAwKSB7Ci0gICAgICAgICAgICBpZHggPSBtUGFja2FnZUdyb3Vwcy5zaXplKCkrMTsKLQotICAgICAgICAgICAgY2hhcjE2X3QgdG1wTmFtZVtzaXplb2YocGtnLT5uYW1lKS9zaXplb2YoY2hhcjE2X3QpXTsKLSAgICAgICAgICAgIHN0cmNweTE2X2R0b2godG1wTmFtZSwgcGtnLT5uYW1lLCBzaXplb2YocGtnLT5uYW1lKS9zaXplb2YoY2hhcjE2X3QpKTsKLSAgICAgICAgICAgIGdyb3VwID0gbmV3IFBhY2thZ2VHcm91cChTdHJpbmcxNih0bXBOYW1lKSwgaWQpOwotICAgICAgICAgICAgaWYgKGdyb3VwID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1OT19NRU1PUlkpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBlcnIgPSBncm91cC0+dHlwZVN0cmluZ3Muc2V0VG8oYmFzZStkdG9obChwa2ctPnR5cGVTdHJpbmdzKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBoZWFkZXItPmRhdGFFbmQtKGJhc2UrZHRvaGwocGtnLT50eXBlU3RyaW5ncykpKTsKLSAgICAgICAgICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1lcnIpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZXJyID0gZ3JvdXAtPmtleVN0cmluZ3Muc2V0VG8oYmFzZStkdG9obChwa2ctPmtleVN0cmluZ3MpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaGVhZGVyLT5kYXRhRW5kLShiYXNlK2R0b2hsKHBrZy0+a2V5U3RyaW5ncykpKTsKLSAgICAgICAgICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1lcnIpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvL3ByaW50ZigiQWRkaW5nIG5ldyBwYWNrYWdlIGlkICVkIGF0IGluZGV4ICVkXG4iLCBpZCwgaWR4KTsKLSAgICAgICAgICAgIGVyciA9IG1QYWNrYWdlR3JvdXBzLmFkZChncm91cCk7Ci0gICAgICAgICAgICBpZiAoZXJyIDwgTk9fRVJST1IpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1lcnIpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgbVBhY2thZ2VNYXBbaWRdID0gKHVpbnQ4X3QpaWR4OwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZ3JvdXAgPSBtUGFja2FnZUdyb3Vwcy5pdGVtQXQoaWR4LTEpOwotICAgICAgICAgICAgaWYgKGdyb3VwID09IE5VTEwpIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1VTktOT1dOX0VSUk9SKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBwYWNrYWdlID0gbmV3IFBhY2thZ2UoaGVhZGVyLCBwa2cpOwotICAgICAgICBpZiAocGFja2FnZSA9PSBOVUxMKSB7Ci0gICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1OT19NRU1PUlkpOwotICAgICAgICB9Ci0gICAgICAgIGVyciA9IGdyb3VwLT5wYWNrYWdlcy5hZGQocGFja2FnZSk7Ci0gICAgICAgIGlmIChlcnIgPCBOT19FUlJPUikgewotICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9ZXJyKTsKLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIExPR19BTFdBWVNfRkFUQUwoIlNraW5zIG5vdCBzdXBwb3J0ZWQhIik7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0KLSAgICAKLSAgICAvLyBJdGVyYXRlIHRocm91Z2ggYWxsIGNodW5rcy4KLSAgICBzaXplX3QgY3VyUGFja2FnZSA9IDA7Ci0gICAgCi0gICAgY29uc3QgUmVzQ2h1bmtfaGVhZGVyKiBjaHVuayA9Ci0gICAgICAgIChjb25zdCBSZXNDaHVua19oZWFkZXIqKSgoKGNvbnN0IHVpbnQ4X3QqKXBrZykKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICsgZHRvaHMocGtnLT5oZWFkZXIuaGVhZGVyU2l6ZSkpOwotICAgIGNvbnN0IHVpbnQ4X3QqIGVuZFBvcyA9ICgoY29uc3QgdWludDhfdCopcGtnKSArIGR0b2hzKHBrZy0+aGVhZGVyLnNpemUpOwotICAgIHdoaWxlICgoKGNvbnN0IHVpbnQ4X3QqKWNodW5rKSA8PSAoZW5kUG9zLXNpemVvZihSZXNDaHVua19oZWFkZXIpKSAmJgotICAgICAgICAgICAoKGNvbnN0IHVpbnQ4X3QqKWNodW5rKSA8PSAoZW5kUG9zLWR0b2hsKGNodW5rLT5zaXplKSkpIHsKLSAgICAgICAgVEFCTEVfTk9JU1koTE9HVigiUGFja2FnZUNodW5rOiB0eXBlPTB4JXgsIGhlYWRlclNpemU9MHgleCwgc2l6ZT0weCV4LCBwb3M9JXBcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgZHRvaHMoY2h1bmstPnR5cGUpLCBkdG9ocyhjaHVuay0+aGVhZGVyU2l6ZSksIGR0b2hsKGNodW5rLT5zaXplKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAodm9pZCopKCgoY29uc3QgdWludDhfdCopY2h1bmspIC0gKChjb25zdCB1aW50OF90KiloZWFkZXItPmhlYWRlcikpKSk7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBjc2l6ZSA9IGR0b2hsKGNodW5rLT5zaXplKTsKLSAgICAgICAgY29uc3QgdWludDE2X3QgY3R5cGUgPSBkdG9ocyhjaHVuay0+dHlwZSk7Ci0gICAgICAgIGlmIChjdHlwZSA9PSBSRVNfVEFCTEVfVFlQRV9TUEVDX1RZUEUpIHsKLSAgICAgICAgICAgIGNvbnN0IFJlc1RhYmxlX3R5cGVTcGVjKiB0eXBlU3BlYyA9IChjb25zdCBSZXNUYWJsZV90eXBlU3BlYyopKGNodW5rKTsKLSAgICAgICAgICAgIGVyciA9IHZhbGlkYXRlX2NodW5rKCZ0eXBlU3BlYy0+aGVhZGVyLCBzaXplb2YoKnR5cGVTcGVjKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGVuZFBvcywgIlJlc1RhYmxlX3R5cGVTcGVjIik7Ci0gICAgICAgICAgICBpZiAoZXJyICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9ZXJyKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIAotICAgICAgICAgICAgY29uc3Qgc2l6ZV90IHR5cGVTcGVjU2l6ZSA9IGR0b2hsKHR5cGVTcGVjLT5oZWFkZXIuc2l6ZSk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIExPQURfVEFCTEVfTk9JU1kocHJpbnRmKCJUeXBlU3BlYyBvZmYgJXA6IHR5cGU9MHgleCwgaGVhZGVyU2l6ZT0weCV4LCBzaXplPSVwXG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKHZvaWQqKShiYXNlLShjb25zdCB1aW50OF90KiljaHVuayksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkdG9ocyh0eXBlU3BlYy0+aGVhZGVyLnR5cGUpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHRvaHModHlwZVNwZWMtPmhlYWRlci5oZWFkZXJTaXplKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICh2b2lkKil0eXBlU2l6ZSkpOwotICAgICAgICAgICAgLy8gbG9vayBmb3IgYmxvY2sgb3ZlcnJ1biBvciBpbnQgb3ZlcmZsb3cgd2hlbiBtdWx0aXBseWluZyBieSA0Ci0gICAgICAgICAgICBpZiAoKGR0b2hsKHR5cGVTcGVjLT5lbnRyeUNvdW50KSA+IChJTlQzMl9NQVgvc2l6ZW9mKHVpbnQzMl90KSkKLSAgICAgICAgICAgICAgICAgICAgfHwgZHRvaHModHlwZVNwZWMtPmhlYWRlci5oZWFkZXJTaXplKSsoc2l6ZW9mKHVpbnQzMl90KSpkdG9obCh0eXBlU3BlYy0+ZW50cnlDb3VudCkpCi0gICAgICAgICAgICAgICAgICAgID4gdHlwZVNwZWNTaXplKSkgewotICAgICAgICAgICAgICAgIExPR1coIlJlc1RhYmxlX3R5cGVTcGVjIGVudHJ5IGluZGV4IHRvICVwIGV4dGVuZHMgYmV5b25kIGNodW5rIGVuZCAlcC4iLAotICAgICAgICAgICAgICAgICAgICAgKHZvaWQqKShkdG9ocyh0eXBlU3BlYy0+aGVhZGVyLmhlYWRlclNpemUpCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICsoc2l6ZW9mKHVpbnQzMl90KSpkdG9obCh0eXBlU3BlYy0+ZW50cnlDb3VudCkpKSwKLSAgICAgICAgICAgICAgICAgICAgICh2b2lkKil0eXBlU3BlY1NpemUpOwotICAgICAgICAgICAgICAgIHJldHVybiAobUVycm9yPUJBRF9UWVBFKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIAotICAgICAgICAgICAgaWYgKHR5cGVTcGVjLT5pZCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgTE9HVygiUmVzVGFibGVfdHlwZSBoYXMgYW4gaWQgb2YgMC4iKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1CQURfVFlQRSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIHdoaWxlIChwYWNrYWdlLT50eXBlcy5zaXplKCkgPCB0eXBlU3BlYy0+aWQpIHsKLSAgICAgICAgICAgICAgICBwYWNrYWdlLT50eXBlcy5hZGQoTlVMTCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBUeXBlKiB0ID0gcGFja2FnZS0+dHlwZXNbdHlwZVNwZWMtPmlkLTFdOwotICAgICAgICAgICAgaWYgKHQgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgIHQgPSBuZXcgVHlwZShoZWFkZXIsIHBhY2thZ2UsIGR0b2hsKHR5cGVTcGVjLT5lbnRyeUNvdW50KSk7Ci0gICAgICAgICAgICAgICAgcGFja2FnZS0+dHlwZXMuZWRpdEl0ZW1BdCh0eXBlU3BlYy0+aWQtMSkgPSB0OwotICAgICAgICAgICAgfSBlbHNlIGlmIChkdG9obCh0eXBlU3BlYy0+ZW50cnlDb3VudCkgIT0gdC0+ZW50cnlDb3VudCkgewotICAgICAgICAgICAgICAgIExPR1coIlJlc1RhYmxlX3R5cGVTcGVjIGVudHJ5IGNvdW50IGluY29uc2lzdGVudDogZ2l2ZW4gJWQsIHByZXZpb3VzbHkgJWQiLAotICAgICAgICAgICAgICAgICAgICAoaW50KWR0b2hsKHR5cGVTcGVjLT5lbnRyeUNvdW50KSwgKGludCl0LT5lbnRyeUNvdW50KTsKLSAgICAgICAgICAgICAgICByZXR1cm4gKG1FcnJvcj1CQURfVFlQRSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICB0LT50eXBlU3BlY0ZsYWdzID0gKGNvbnN0IHVpbnQzMl90KikoCi0gICAgICAgICAgICAgICAgICAgICgoY29uc3QgdWludDhfdCopdHlwZVNwZWMpICsgZHRvaHModHlwZVNwZWMtPmhlYWRlci5oZWFkZXJTaXplKSk7Ci0gICAgICAgICAgICB0LT50eXBlU3BlYyA9IHR5cGVTcGVjOwotICAgICAgICAgICAgCi0gICAgICAgIH0gZWxzZSBpZiAoY3R5cGUgPT0gUkVTX1RBQkxFX1RZUEVfVFlQRSkgewotICAgICAgICAgICAgY29uc3QgUmVzVGFibGVfdHlwZSogdHlwZSA9IChjb25zdCBSZXNUYWJsZV90eXBlKikoY2h1bmspOwotICAgICAgICAgICAgZXJyID0gdmFsaWRhdGVfY2h1bmsoJnR5cGUtPmhlYWRlciwgc2l6ZW9mKCp0eXBlKS1zaXplb2YoUmVzVGFibGVfY29uZmlnKSs0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZW5kUG9zLCAiUmVzVGFibGVfdHlwZSIpOwotICAgICAgICAgICAgaWYgKGVyciAhPSBOT19FUlJPUikgewotICAgICAgICAgICAgICAgIHJldHVybiAobUVycm9yPWVycik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIGNvbnN0IHNpemVfdCB0eXBlU2l6ZSA9IGR0b2hsKHR5cGUtPmhlYWRlci5zaXplKTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgTE9BRF9UQUJMRV9OT0lTWShwcmludGYoIlR5cGUgb2ZmICVwOiB0eXBlPTB4JXgsIGhlYWRlclNpemU9MHgleCwgc2l6ZT0lcFxuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICh2b2lkKikoYmFzZS0oY29uc3QgdWludDhfdCopY2h1bmspLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHRvaHModHlwZS0+aGVhZGVyLnR5cGUpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHRvaHModHlwZS0+aGVhZGVyLmhlYWRlclNpemUpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKHZvaWQqKXR5cGVTaXplKSk7Ci0gICAgICAgICAgICBpZiAoZHRvaHModHlwZS0+aGVhZGVyLmhlYWRlclNpemUpKyhzaXplb2YodWludDMyX3QpKmR0b2hsKHR5cGUtPmVudHJ5Q291bnQpKQotICAgICAgICAgICAgICAgID4gdHlwZVNpemUpIHsKLSAgICAgICAgICAgICAgICBMT0dXKCJSZXNUYWJsZV90eXBlIGVudHJ5IGluZGV4IHRvICVwIGV4dGVuZHMgYmV5b25kIGNodW5rIGVuZCAlcC4iLAotICAgICAgICAgICAgICAgICAgICAgKHZvaWQqKShkdG9ocyh0eXBlLT5oZWFkZXIuaGVhZGVyU2l6ZSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKyhzaXplb2YodWludDMyX3QpKmR0b2hsKHR5cGUtPmVudHJ5Q291bnQpKSksCi0gICAgICAgICAgICAgICAgICAgICAodm9pZCopdHlwZVNpemUpOwotICAgICAgICAgICAgICAgIHJldHVybiAobUVycm9yPUJBRF9UWVBFKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChkdG9obCh0eXBlLT5lbnRyeUNvdW50KSAhPSAwCi0gICAgICAgICAgICAgICAgJiYgZHRvaGwodHlwZS0+ZW50cmllc1N0YXJ0KSA+ICh0eXBlU2l6ZS1zaXplb2YoUmVzVGFibGVfZW50cnkpKSkgewotICAgICAgICAgICAgICAgIExPR1coIlJlc1RhYmxlX3R5cGUgZW50cmllc1N0YXJ0IGF0ICVwIGV4dGVuZHMgYmV5b25kIGNodW5rIGVuZCAlcC4iLAotICAgICAgICAgICAgICAgICAgICAgKHZvaWQqKWR0b2hsKHR5cGUtPmVudHJpZXNTdGFydCksICh2b2lkKil0eXBlU2l6ZSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9QkFEX1RZUEUpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHR5cGUtPmlkID09IDApIHsKLSAgICAgICAgICAgICAgICBMT0dXKCJSZXNUYWJsZV90eXBlIGhhcyBhbiBpZCBvZiAwLiIpOwotICAgICAgICAgICAgICAgIHJldHVybiAobUVycm9yPUJBRF9UWVBFKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIAotICAgICAgICAgICAgd2hpbGUgKHBhY2thZ2UtPnR5cGVzLnNpemUoKSA8IHR5cGUtPmlkKSB7Ci0gICAgICAgICAgICAgICAgcGFja2FnZS0+dHlwZXMuYWRkKE5VTEwpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgVHlwZSogdCA9IHBhY2thZ2UtPnR5cGVzW3R5cGUtPmlkLTFdOwotICAgICAgICAgICAgaWYgKHQgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgIHQgPSBuZXcgVHlwZShoZWFkZXIsIHBhY2thZ2UsIGR0b2hsKHR5cGUtPmVudHJ5Q291bnQpKTsKLSAgICAgICAgICAgICAgICBwYWNrYWdlLT50eXBlcy5lZGl0SXRlbUF0KHR5cGUtPmlkLTEpID0gdDsKLSAgICAgICAgICAgIH0gZWxzZSBpZiAoZHRvaGwodHlwZS0+ZW50cnlDb3VudCkgIT0gdC0+ZW50cnlDb3VudCkgewotICAgICAgICAgICAgICAgIExPR1coIlJlc1RhYmxlX3R5cGUgZW50cnkgY291bnQgaW5jb25zaXN0ZW50OiBnaXZlbiAlZCwgcHJldmlvdXNseSAlZCIsCi0gICAgICAgICAgICAgICAgICAgIChpbnQpZHRvaGwodHlwZS0+ZW50cnlDb3VudCksIChpbnQpdC0+ZW50cnlDb3VudCk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9QkFEX1RZUEUpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgCi0gICAgICAgICAgICBUQUJMRV9HRVRFTlRSWSgKLSAgICAgICAgICAgICAgICBSZXNUYWJsZV9jb25maWcgdGhpc0NvbmZpZzsKLSAgICAgICAgICAgICAgICB0aGlzQ29uZmlnLmNvcHlGcm9tRHRvSCh0eXBlLT5jb25maWcpOwotICAgICAgICAgICAgICAgIExPR0koIkFkZGluZyBjb25maWcgdG8gdHlwZSAlZDogaW1zaTolZC8lZCBsYW5nOiVjJWMgY250OiVjJWMgIgotICAgICAgICAgICAgICAgICAgICAgIm9yaWVuOiVkIHRvdWNoOiVkIGRlbnNpdHk6JWQga2V5OiVkIGlucDolZCBuYXY6JWQgdzolZCBoOiVkXG4iLAotICAgICAgICAgICAgICAgICAgICAgIHR5cGUtPmlkLAotICAgICAgICAgICAgICAgICAgICAgIHRoaXNDb25maWcubWNjLCB0aGlzQ29uZmlnLm1uYywKLSAgICAgICAgICAgICAgICAgICAgICB0aGlzQ29uZmlnLmxhbmd1YWdlWzBdID8gdGhpc0NvbmZpZy5sYW5ndWFnZVswXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICB0aGlzQ29uZmlnLmxhbmd1YWdlWzFdID8gdGhpc0NvbmZpZy5sYW5ndWFnZVsxXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICB0aGlzQ29uZmlnLmNvdW50cnlbMF0gPyB0aGlzQ29uZmlnLmNvdW50cnlbMF0gOiAnLScsCi0gICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy5jb3VudHJ5WzFdID8gdGhpc0NvbmZpZy5jb3VudHJ5WzFdIDogJy0nLAotICAgICAgICAgICAgICAgICAgICAgIHRoaXNDb25maWcub3JpZW50YXRpb24sCi0gICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy50b3VjaHNjcmVlbiwKLSAgICAgICAgICAgICAgICAgICAgICB0aGlzQ29uZmlnLmRlbnNpdHksCi0gICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy5rZXlib2FyZCwKLSAgICAgICAgICAgICAgICAgICAgICB0aGlzQ29uZmlnLmlucHV0RmxhZ3MsCi0gICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy5uYXZpZ2F0aW9uLAotICAgICAgICAgICAgICAgICAgICAgIHRoaXNDb25maWcuc2NyZWVuV2lkdGgsCi0gICAgICAgICAgICAgICAgICAgICAgdGhpc0NvbmZpZy5zY3JlZW5IZWlnaHQpKTsKLSAgICAgICAgICAgIHQtPmNvbmZpZ3MuYWRkKHR5cGUpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgc3RhdHVzX3QgZXJyID0gdmFsaWRhdGVfY2h1bmsoY2h1bmssIHNpemVvZihSZXNDaHVua19oZWFkZXIpLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZW5kUG9zLCAiUmVzVGFibGVfcGFja2FnZTp1bmtub3duIik7Ci0gICAgICAgICAgICBpZiAoZXJyICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIChtRXJyb3I9ZXJyKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBjaHVuayA9IChjb25zdCBSZXNDaHVua19oZWFkZXIqKQotICAgICAgICAgICAgKCgoY29uc3QgdWludDhfdCopY2h1bmspICsgY3NpemUpOwotICAgIH0KLQotICAgIGlmIChncm91cC0+dHlwZUNvdW50ID09IDApIHsKLSAgICAgICAgZ3JvdXAtPnR5cGVDb3VudCA9IHBhY2thZ2UtPnR5cGVzLnNpemUoKTsKLSAgICB9Ci0gICAgCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0jaWZuZGVmIEhBVkVfQU5EUk9JRF9PUwotI2RlZmluZSBDSEFSMTZfVE9fQ1NUUihjMTYsIGxlbikgKFN0cmluZzgoU3RyaW5nMTYoYzE2LGxlbikpLnN0cmluZygpKQotCi0jZGVmaW5lIENIQVIxNl9BUlJBWV9FUShjb25zdGFudCwgdmFyLCBsZW4pIFwKLSAgICAgICAgKChsZW4gPT0gKHNpemVvZihjb25zdGFudCkvc2l6ZW9mKGNvbnN0YW50WzBdKSkpICYmICgwID09IG1lbWNtcCgodmFyKSwgKGNvbnN0YW50KSwgKGxlbikpKSkKLQotdm9pZCBSZXNUYWJsZTo6cHJpbnQoKSBjb25zdAotewotICAgIHByaW50ZigibUVycm9yPTB4JXggKCVzKVxuIiwgbUVycm9yLCBzdHJlcnJvcihtRXJyb3IpKTsKLSNpZiAwCi0gICAgcHJpbnRmKCJtUGFyYW1zPSVjJWMtJWMlYyxcbiIsCi0gICAgICAgICAgICBtUGFyYW1zLmxhbmd1YWdlWzBdLCBtUGFyYW1zLmxhbmd1YWdlWzFdLAotICAgICAgICAgICAgbVBhcmFtcy5jb3VudHJ5WzBdLCBtUGFyYW1zLmNvdW50cnlbMV0pOwotI2VuZGlmCi0gICAgc2l6ZV90IHBnQ291bnQgPSBtUGFja2FnZUdyb3Vwcy5zaXplKCk7Ci0gICAgcHJpbnRmKCJQYWNrYWdlIEdyb3VwcyAoJWQpXG4iLCAoaW50KXBnQ291bnQpOwotICAgIGZvciAoc2l6ZV90IHBnSW5kZXg9MDsgcGdJbmRleDxwZ0NvdW50OyBwZ0luZGV4KyspIHsKLSAgICAgICAgY29uc3QgUGFja2FnZUdyb3VwKiBwZyA9IG1QYWNrYWdlR3JvdXBzW3BnSW5kZXhdOwotICAgICAgICBwcmludGYoIlBhY2thZ2UgR3JvdXAgJWQgaWQ9JWQgcGFja2FnZUNvdW50PSVkIG5hbWU9JXNcbiIsCi0gICAgICAgICAgICAgICAgKGludClwZ0luZGV4LCBwZy0+aWQsIChpbnQpcGctPnBhY2thZ2VzLnNpemUoKSwKLSAgICAgICAgICAgICAgICBTdHJpbmc4KHBnLT5uYW1lKS5zdHJpbmcoKSk7Ci0gICAgICAgIAotICAgICAgICBzaXplX3QgcGtnQ291bnQgPSBwZy0+cGFja2FnZXMuc2l6ZSgpOwotICAgICAgICBmb3IgKHNpemVfdCBwa2dJbmRleD0wOyBwa2dJbmRleDxwa2dDb3VudDsgcGtnSW5kZXgrKykgewotICAgICAgICAgICAgY29uc3QgUGFja2FnZSogcGtnID0gcGctPnBhY2thZ2VzW3BrZ0luZGV4XTsKLSAgICAgICAgICAgIHNpemVfdCB0eXBlQ291bnQgPSBwa2ctPnR5cGVzLnNpemUoKTsKLSAgICAgICAgICAgIHByaW50ZigiICBQYWNrYWdlICVkIGlkPSVkIG5hbWU9JXMgdHlwZUNvdW50PSVkXG4iLCAoaW50KXBrZ0luZGV4LAotICAgICAgICAgICAgICAgICAgICBwa2ctPnBhY2thZ2UtPmlkLCBTdHJpbmc4KFN0cmluZzE2KHBrZy0+cGFja2FnZS0+bmFtZSkpLnN0cmluZygpLAotICAgICAgICAgICAgICAgICAgICAoaW50KXR5cGVDb3VudCk7Ci0gICAgICAgICAgICBmb3IgKHNpemVfdCB0eXBlSW5kZXg9MDsgdHlwZUluZGV4PHR5cGVDb3VudDsgdHlwZUluZGV4KyspIHsKLSAgICAgICAgICAgICAgICBjb25zdCBUeXBlKiB0eXBlQ29uZmlncyA9IHBrZy0+Z2V0VHlwZSh0eXBlSW5kZXgpOwotICAgICAgICAgICAgICAgIGlmICh0eXBlQ29uZmlncyA9PSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgICAgIHByaW50ZigiICAgIHR5cGUgJWQgTlVMTFxuIiwgKGludCl0eXBlSW5kZXgpOwotICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgY29uc3Qgc2l6ZV90IE5UQyA9IHR5cGVDb25maWdzLT5jb25maWdzLnNpemUoKTsKLSAgICAgICAgICAgICAgICBwcmludGYoIiAgICB0eXBlICVkIGNvbmZpZ0NvdW50PSVkIGVudHJ5Q291bnQ9JWRcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgIChpbnQpdHlwZUluZGV4LCAoaW50KU5UQywgKGludCl0eXBlQ29uZmlncy0+ZW50cnlDb3VudCk7Ci0gICAgICAgICAgICAgICAgaWYgKHR5cGVDb25maWdzLT50eXBlU3BlY0ZsYWdzICE9IE5VTEwpIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChzaXplX3QgZW50cnlJbmRleD0wOyBlbnRyeUluZGV4PHR5cGVDb25maWdzLT5lbnRyeUNvdW50OyBlbnRyeUluZGV4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHJlc0lEID0gKDB4ZmYwMDAwMDAgJiAoKHBrZy0+cGFja2FnZS0+aWQpPDwyNCkpCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8ICgweDAwZmYwMDAwICYgKCh0eXBlSW5kZXgrMSk8PDE2KSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgKDB4MDAwMGZmZmYgJiAoZW50cnlJbmRleCkpOwotICAgICAgICAgICAgICAgICAgICAgICAgcmVzb3VyY2VfbmFtZSByZXNOYW1lOwotICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy0+Z2V0UmVzb3VyY2VOYW1lKHJlc0lELCAmcmVzTmFtZSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwcmludGYoIiAgICAgIHNwZWMgcmVzb3VyY2UgMHglMDh4ICVzOiVzLyVzOiBmbGFncz0weCUwOHhcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVzSUQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgQ0hBUjE2X1RPX0NTVFIocmVzTmFtZS5wYWNrYWdlLCByZXNOYW1lLnBhY2thZ2VMZW4pLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIENIQVIxNl9UT19DU1RSKHJlc05hbWUudHlwZSwgcmVzTmFtZS50eXBlTGVuKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBDSEFSMTZfVE9fQ1NUUihyZXNOYW1lLm5hbWUsIHJlc05hbWUubmFtZUxlbiksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgZHRvaGwodHlwZUNvbmZpZ3MtPnR5cGVTcGVjRmxhZ3NbZW50cnlJbmRleF0pKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBmb3IgKHNpemVfdCBjb25maWdJbmRleD0wOyBjb25maWdJbmRleDxOVEM7IGNvbmZpZ0luZGV4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgY29uc3QgUmVzVGFibGVfdHlwZSogdHlwZSA9IHR5cGVDb25maWdzLT5jb25maWdzW2NvbmZpZ0luZGV4XTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKCgoKHVpbnQ2NF90KXR5cGUpJjB4MykgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgcHJpbnRmKCIgICAgICBOT04tSU5URUdFUiBSZXNUYWJsZV90eXBlIEFERFJFU1M6ICVwXG4iLCB0eXBlKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIHByaW50ZigiICAgICAgY29uZmlnICVkIGxhbmc9JWMlYyBjbnQ9JWMlYyBvcmllbj0lZCB0b3VjaD0lZCBkZW5zaXR5PSVkIGtleT0lZCBpbmZsPSVkIG5hdj0lZCB3PSVkIGg9JWRcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAoaW50KWNvbmZpZ0luZGV4LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgdHlwZS0+Y29uZmlnLmxhbmd1YWdlWzBdID8gdHlwZS0+Y29uZmlnLmxhbmd1YWdlWzBdIDogJy0nLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgdHlwZS0+Y29uZmlnLmxhbmd1YWdlWzFdID8gdHlwZS0+Y29uZmlnLmxhbmd1YWdlWzFdIDogJy0nLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgdHlwZS0+Y29uZmlnLmNvdW50cnlbMF0gPyB0eXBlLT5jb25maWcuY291bnRyeVswXSA6ICctJywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIHR5cGUtPmNvbmZpZy5jb3VudHJ5WzFdID8gdHlwZS0+Y29uZmlnLmNvdW50cnlbMV0gOiAnLScsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICB0eXBlLT5jb25maWcub3JpZW50YXRpb24sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICB0eXBlLT5jb25maWcudG91Y2hzY3JlZW4sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBkdG9ocyh0eXBlLT5jb25maWcuZGVuc2l0eSksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICB0eXBlLT5jb25maWcua2V5Ym9hcmQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICB0eXBlLT5jb25maWcuaW5wdXRGbGFncywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIHR5cGUtPmNvbmZpZy5uYXZpZ2F0aW9uLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgZHRvaHModHlwZS0+Y29uZmlnLnNjcmVlbldpZHRoKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIGR0b2hzKHR5cGUtPmNvbmZpZy5zY3JlZW5IZWlnaHQpKTsKLSAgICAgICAgICAgICAgICAgICAgc2l6ZV90IGVudHJ5Q291bnQgPSBkdG9obCh0eXBlLT5lbnRyeUNvdW50KTsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZW50cmllc1N0YXJ0ID0gZHRvaGwodHlwZS0+ZW50cmllc1N0YXJ0KTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKChlbnRyaWVzU3RhcnQmMHgzKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwcmludGYoIiAgICAgIE5PTi1JTlRFR0VSIFJlc1RhYmxlX3R5cGUgZW50cmllc1N0YXJ0IE9GRlNFVDogJXBcbiIsICh2b2lkKillbnRyaWVzU3RhcnQpOwotICAgICAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgdHlwZVNpemUgPSBkdG9obCh0eXBlLT5oZWFkZXIuc2l6ZSk7Ci0gICAgICAgICAgICAgICAgICAgIGlmICgodHlwZVNpemUmMHgzKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwcmludGYoIiAgICAgIE5PTi1JTlRFR0VSIFJlc1RhYmxlX3R5cGUgaGVhZGVyLnNpemU6ICVwXG4iLCAodm9pZCopdHlwZVNpemUpOwotICAgICAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgZm9yIChzaXplX3QgZW50cnlJbmRleD0wOyBlbnRyeUluZGV4PGVudHJ5Q291bnQ7IGVudHJ5SW5kZXgrKykgewotICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB1aW50OF90KiBjb25zdCBlbmQgPSAoKGNvbnN0IHVpbnQ4X3QqKXR5cGUpCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKyBkdG9obCh0eXBlLT5oZWFkZXIuc2l6ZSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB1aW50MzJfdCogY29uc3QgZWluZGV4ID0gKGNvbnN0IHVpbnQzMl90KikKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAoKChjb25zdCB1aW50OF90Kil0eXBlKSArIGR0b2hzKHR5cGUtPmhlYWRlci5oZWFkZXJTaXplKSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHRoaXNPZmZzZXQgPSBkdG9obChlaW5kZXhbZW50cnlJbmRleF0pOwotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHRoaXNPZmZzZXQgPT0gUmVzVGFibGVfdHlwZTo6Tk9fRU5UUlkpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcmVzSUQgPSAoMHhmZjAwMDAwMCAmICgocGtnLT5wYWNrYWdlLT5pZCk8PDI0KSkKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgKDB4MDBmZjAwMDAgJiAoKHR5cGVJbmRleCsxKTw8MTYpKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCAoMHgwMDAwZmZmZiAmIChlbnRyeUluZGV4KSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICByZXNvdXJjZV9uYW1lIHJlc05hbWU7Ci0gICAgICAgICAgICAgICAgICAgICAgICB0aGlzLT5nZXRSZXNvdXJjZU5hbWUocmVzSUQsICZyZXNOYW1lKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHByaW50ZigiICAgICAgICByZXNvdXJjZSAweCUwOHggJXM6JXMvJXM6ICIsIHJlc0lELAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBDSEFSMTZfVE9fQ1NUUihyZXNOYW1lLnBhY2thZ2UsIHJlc05hbWUucGFja2FnZUxlbiksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIENIQVIxNl9UT19DU1RSKHJlc05hbWUudHlwZSwgcmVzTmFtZS50eXBlTGVuKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ0hBUjE2X1RPX0NTVFIocmVzTmFtZS5uYW1lLCByZXNOYW1lLm5hbWVMZW4pKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmICgodGhpc09mZnNldCYweDMpICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBwcmludGYoIk5PTi1JTlRFR0VSIE9GRlNFVDogJXBcbiIsICh2b2lkKil0aGlzT2Zmc2V0KTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIGlmICgodGhpc09mZnNldCtzaXplb2YoUmVzVGFibGVfZW50cnkpKSA+IHR5cGVTaXplKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcHJpbnRmKCJPRkZTRVQgT1VUIE9GIEJPVU5EUzogJXArJXAgKHNpemUgaXMgJXApXG4iLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAodm9pZCopZW50cmllc1N0YXJ0LCAodm9pZCopdGhpc09mZnNldCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKHZvaWQqKXR5cGVTaXplKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgUmVzVGFibGVfZW50cnkqIGVudCA9IChjb25zdCBSZXNUYWJsZV9lbnRyeSopCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKCgoY29uc3QgdWludDhfdCopdHlwZSkgKyBlbnRyaWVzU3RhcnQgKyB0aGlzT2Zmc2V0KTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmICgoKGVudHJpZXNTdGFydCArIHRoaXNPZmZzZXQpJjB4MykgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHByaW50ZigiTk9OLUlOVEVHRVIgUmVzVGFibGVfZW50cnkgT0ZGU0VUOiAlcFxuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICh2b2lkKikoZW50cmllc1N0YXJ0ICsgdGhpc09mZnNldCkpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgaWYgKChkdG9ocyhlbnQtPmZsYWdzKSZSZXNUYWJsZV9lbnRyeTo6RkxBR19DT01QTEVYKSAhPSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgcHJpbnRmKCI8YmFnPiIpOwotICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MTZfdCBlc2l6ZSA9IGR0b2hzKGVudC0+c2l6ZSk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKChlc2l6ZSYweDMpICE9IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcHJpbnRmKCJOT04tSU5URUdFUiBSZXNUYWJsZV9lbnRyeSBTSVpFOiAlcFxuIiwgKHZvaWQqKWVzaXplKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmICgodGhpc09mZnNldCtlc2l6ZSkgPiB0eXBlU2l6ZSkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwcmludGYoIlJlc1RhYmxlX2VudHJ5IE9VVCBPRiBCT1VORFM6ICVwKyVwKyVwIChzaXplIGlzICVwKVxuIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICh2b2lkKillbnRyaWVzU3RhcnQsICh2b2lkKil0aGlzT2Zmc2V0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKHZvaWQqKWVzaXplLCAodm9pZCopdHlwZVNpemUpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgUmVzX3ZhbHVlKiB2YWx1ZSA9IChjb25zdCBSZXNfdmFsdWUqKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoKChjb25zdCB1aW50OF90KillbnQpICsgZXNpemUpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHByaW50ZigidD0weCUwMnggZD0weCUwOHggKHM9MHglMDR4IHI9MHglMDJ4KSIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChpbnQpdmFsdWUtPmRhdGFUeXBlLCAoaW50KWR0b2hsKHZhbHVlLT5kYXRhKSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGludClkdG9ocyh2YWx1ZS0+c2l6ZSksIChpbnQpdmFsdWUtPnJlczApOwotICAgICAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgICAgICAgICBpZiAoKGR0b2hzKGVudC0+ZmxhZ3MpJlJlc1RhYmxlX2VudHJ5OjpGTEFHX1BVQkxJQykgIT0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHByaW50ZigiIChQVUJMSUMpIik7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgICAgICBwcmludGYoIlxuIik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLSNlbmRpZiAvLyBIQVZFX0FORFJPSURfT1MKLQotfSAgIC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1NoYXJlZEJ1ZmZlci5jcHAgYi9saWJzL3V0aWxzL1NoYXJlZEJ1ZmZlci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDM1NTVmYjcuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9TaGFyZWRCdWZmZXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTEzICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9TaGFyZWRCdWZmZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotU2hhcmVkQnVmZmVyKiBTaGFyZWRCdWZmZXI6OmFsbG9jKHNpemVfdCBzaXplKQotewotICAgIFNoYXJlZEJ1ZmZlciogc2IgPSBzdGF0aWNfY2FzdDxTaGFyZWRCdWZmZXIgKj4obWFsbG9jKHNpemVvZihTaGFyZWRCdWZmZXIpICsgc2l6ZSkpOwotICAgIGlmIChzYikgewotICAgICAgICBzYi0+bVJlZnMgPSAxOwotICAgICAgICBzYi0+bVNpemUgPSBzaXplOwotICAgIH0KLSAgICByZXR1cm4gc2I7Ci19Ci0KLQotc3NpemVfdCBTaGFyZWRCdWZmZXI6OmRlYWxsb2MoY29uc3QgU2hhcmVkQnVmZmVyKiByZWxlYXNlZCkKLXsKLSAgICBpZiAocmVsZWFzZWQtPm1SZWZzICE9IDApIHJldHVybiAtMTsgLy8gWFhYOiBpbnZhbGlkIG9wZXJhdGlvbgotICAgIGZyZWUoY29uc3RfY2FzdDxTaGFyZWRCdWZmZXIqPihyZWxlYXNlZCkpOwotICAgIHJldHVybiAwOwotfQotCi1TaGFyZWRCdWZmZXIqIFNoYXJlZEJ1ZmZlcjo6ZWRpdCgpIGNvbnN0Ci17Ci0gICAgaWYgKG9ubHlPd25lcigpKSB7Ci0gICAgICAgIHJldHVybiBjb25zdF9jYXN0PFNoYXJlZEJ1ZmZlcio+KHRoaXMpOwotICAgIH0KLSAgICBTaGFyZWRCdWZmZXIqIHNiID0gYWxsb2MobVNpemUpOwotICAgIGlmIChzYikgewotICAgICAgICBtZW1jcHkoc2ItPmRhdGEoKSwgZGF0YSgpLCBzaXplKCkpOwotICAgICAgICByZWxlYXNlKCk7Ci0gICAgfQotICAgIHJldHVybiBzYjsgICAgCi19Ci0KLVNoYXJlZEJ1ZmZlciogU2hhcmVkQnVmZmVyOjplZGl0UmVzaXplKHNpemVfdCBuZXdTaXplKSBjb25zdAotewotICAgIGlmIChvbmx5T3duZXIoKSkgewotICAgICAgICBTaGFyZWRCdWZmZXIqIGJ1ZiA9IGNvbnN0X2Nhc3Q8U2hhcmVkQnVmZmVyKj4odGhpcyk7Ci0gICAgICAgIGlmIChidWYtPm1TaXplID09IG5ld1NpemUpIHJldHVybiBidWY7Ci0gICAgICAgIGJ1ZiA9IChTaGFyZWRCdWZmZXIqKXJlYWxsb2MoYnVmLCBzaXplb2YoU2hhcmVkQnVmZmVyKSArIG5ld1NpemUpOwotICAgICAgICBpZiAoYnVmICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGJ1Zi0+bVNpemUgPSBuZXdTaXplOwotICAgICAgICAgICAgcmV0dXJuIGJ1ZjsKLSAgICAgICAgfQotICAgIH0KLSAgICBTaGFyZWRCdWZmZXIqIHNiID0gYWxsb2MobmV3U2l6ZSk7Ci0gICAgaWYgKHNiKSB7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBteVNpemUgPSBtU2l6ZTsKLSAgICAgICAgbWVtY3B5KHNiLT5kYXRhKCksIGRhdGEoKSwgbmV3U2l6ZSA8IG15U2l6ZSA/IG5ld1NpemUgOiBteVNpemUpOwotICAgICAgICByZWxlYXNlKCk7Ci0gICAgfQotICAgIHJldHVybiBzYjsgICAgCi19Ci0KLVNoYXJlZEJ1ZmZlciogU2hhcmVkQnVmZmVyOjphdHRlbXB0RWRpdCgpIGNvbnN0Ci17Ci0gICAgaWYgKG9ubHlPd25lcigpKSB7Ci0gICAgICAgIHJldHVybiBjb25zdF9jYXN0PFNoYXJlZEJ1ZmZlcio+KHRoaXMpOwotICAgIH0KLSAgICByZXR1cm4gMDsKLX0KLQotU2hhcmVkQnVmZmVyKiBTaGFyZWRCdWZmZXI6OnJlc2V0KHNpemVfdCBuZXdfc2l6ZSkgY29uc3QKLXsKLSAgICAvLyBjaGVhcC1vLXJlc2V0LgotICAgIFNoYXJlZEJ1ZmZlciogc2IgPSBhbGxvYyhuZXdfc2l6ZSk7Ci0gICAgaWYgKHNiKSB7Ci0gICAgICAgIHJlbGVhc2UoKTsKLSAgICB9Ci0gICAgcmV0dXJuIHNiOwotfQotCi12b2lkIFNoYXJlZEJ1ZmZlcjo6YWNxdWlyZSgpIGNvbnN0IHsKLSAgICBhbmRyb2lkX2F0b21pY19pbmMoJm1SZWZzKTsKLX0KLQotaW50MzJfdCBTaGFyZWRCdWZmZXI6OnJlbGVhc2UodWludDMyX3QgZmxhZ3MpIGNvbnN0Ci17Ci0gICAgaW50MzJfdCBwcmV2ID0gMTsKLSAgICBpZiAob25seU93bmVyKCkgfHwgKChwcmV2ID0gYW5kcm9pZF9hdG9taWNfZGVjKCZtUmVmcykpID09IDEpKSB7Ci0gICAgICAgIG1SZWZzID0gMDsKLSAgICAgICAgaWYgKChmbGFncyAmIGVLZWVwU3RvcmFnZSkgPT0gMCkgewotICAgICAgICAgICAgZnJlZShjb25zdF9jYXN0PFNoYXJlZEJ1ZmZlcio+KHRoaXMpKTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gcHJldjsKLX0KLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9Tb2NrZXQuY3BwIGIvbGlicy91dGlscy9Tb2NrZXQuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1MTUwOWEzLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvU29ja2V0LmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDM4OCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIEludGVybmV0IGFkZHJlc3MgY2xhc3MuCi0vLwotCi0jaWZkZWYgSEFWRV9XSU5TT0NLCi0vLyBUaGlzIG5lZWRzIHRvIGNvbWUgZmlyc3QsIG9yIEN5Z3dpbiBnZXRzIGNvbmNlcm5lZCBhYm91dCBhIHBvdGVudGlhbAotLy8gY2xhc2ggYmV0d2VlbiBXaW5Tb2NrIGFuZCA8c3lzL3R5cGVzLmg+LgotIyBpbmNsdWRlIDx3aW5zb2NrMi5oPgotI2VuZGlmCi0KLSNpbmNsdWRlIDx1dGlscy9Tb2NrZXQuaD4KLSNpbmNsdWRlIDx1dGlscy9pbmV0X2FkZHJlc3MuaD4KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLSNpbmNsdWRlIDx1dGlscy9UaW1lcnMuaD4KLQotI2lmbmRlZiBIQVZFX1dJTlNPQ0sKLSMgaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jIGluY2x1ZGUgPHN5cy9zb2NrZXQuaD4KLSMgaW5jbHVkZSA8bmV0aW5ldC9pbi5oPgotIyBpbmNsdWRlIDxhcnBhL2luZXQuaD4KLSNlbmRpZgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDx1bmlzdGQuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotI2luY2x1ZGUgPGFzc2VydC5oPgotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotCi0vKgotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKiAgICAgIFNvY2tldAotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKi8KLQotI2lmbmRlZiBJTlZBTElEX1NPQ0tFVAotIyBkZWZpbmUgSU5WQUxJRF9TT0NLRVQgKC0xKQotI2VuZGlmCi0jZGVmaW5lIFVOREVGX1NPQ0tFVCAgICgodW5zaWduZWQgbG9uZykgSU5WQUxJRF9TT0NLRVQpCi0KLS8qc3RhdGljKi8gYm9vbCBTb2NrZXQ6Om1Cb290SW5pdGlhbGl6ZWQgPSBmYWxzZTsKLQotLyoKLSAqIEV4dHJhY3Qgc3lzdGVtLWRlcGVuZGVudCBlcnJvciBjb2RlLgotICovCi1zdGF0aWMgaW5saW5lIGludCBnZXRTb2NrZXRFcnJvcih2b2lkKSB7Ci0jaWZkZWYgSEFWRV9XSU5TT0NLCi0gICAgcmV0dXJuIFdTQUdldExhc3RFcnJvcigpOwotI2Vsc2UKLSAgICByZXR1cm4gZXJybm87Ci0jZW5kaWYKLX0KLQotLyoKLSAqIE9uZS10aW1lIGluaXRpYWxpemF0aW9uIGZvciBzb2NrZXQgY29kZS4KLSAqLwotLypzdGF0aWMqLyBib29sIFNvY2tldDo6Ym9vdEluaXQodm9pZCkKLXsKLSNpZmRlZiBIQVZFX1dJTlNPQ0sKLSAgICBXU0FEQVRBIHdzYURhdGE7Ci0gICAgaW50IGVycjsKLQotICAgIGVyciA9IFdTQVN0YXJ0dXAoTUFLRVdPUkQoMiwgMCksICZ3c2FEYXRhKTsKLSAgICBpZiAoZXJyICE9IDApIHsKLSAgICAgICAgTE9HKExPR19FUlJPUiwgInNvY2tldCIsICJVbmFibGUgdG8gc3RhcnQgV2luU29ja1xuIik7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBMT0coTE9HX0lORk8sICJzb2NrZXQiLCAiVXNpbmcgV2luU29jayB2JWQuJWRcbiIsCi0gICAgICAgIExPQllURSh3c2FEYXRhLndWZXJzaW9uKSwgSElCWVRFKHdzYURhdGEud1ZlcnNpb24pKTsKLSNlbmRpZgotCi0gICAgbUJvb3RJbml0aWFsaXplZCA9IHRydWU7Ci0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLS8qCi0gKiBPbmUtdGltZSBzaHV0ZG93biBmb3Igc29ja2V0IGNvZGUuCi0gKi8KLS8qc3RhdGljKi8gdm9pZCBTb2NrZXQ6OmZpbmFsU2h1dGRvd24odm9pZCkKLXsKLSNpZmRlZiBIQVZFX1dJTlNPQ0sKLSAgICBXU0FDbGVhbnVwKCk7Ci0jZW5kaWYKLSAgICBtQm9vdEluaXRpYWxpemVkID0gZmFsc2U7Ci19Ci0KLQotLyoKLSAqIFNpbXBsZSBjb25zdHJ1Y3Rvci4gIEFsbG93IHRoZSBhcHBsaWNhdGlvbiB0byBjcmVhdGUgdXMgYW5kIHRoZW4gbWFrZQotICogYmluZC9jb25uZWN0IGNhbGxzLgotICovCi1Tb2NrZXQ6OlNvY2tldCh2b2lkKQotICAgIDogbVNvY2soVU5ERUZfU09DS0VUKQotewotICAgIGlmICghbUJvb3RJbml0aWFsaXplZCkKLSAgICAgICAgTE9HKExPR19XQVJOLCAic29ja2V0IiwgIldBUk5JTkc6IHNvY2tldHMgbm90IGluaXRpYWxpemVkXG4iKTsKLX0KLQotLyoKLSAqIERlc3RydWN0b3IuICBDbG9zZXMgdGhlIHNvY2tldCBhbmQgcmVzZXRzIG91ciBzdG9yYWdlLgotICovCi1Tb2NrZXQ6On5Tb2NrZXQodm9pZCkKLXsKLSAgICBjbG9zZSgpOwotfQotCi0KLS8qCi0gKiBDcmVhdGUgYSBzb2NrZXQgYW5kIGNvbm5lY3QgdG8gdGhlIHNwZWNpZmllZCBob3N0IGFuZCBwb3J0LgotICovCi1pbnQgU29ja2V0Ojpjb25uZWN0KGNvbnN0IGNoYXIqIGhvc3QsIGludCBwb3J0KQotewotICAgIGlmIChtU29jayAhPSBVTkRFRl9TT0NLRVQpIHsKLSAgICAgICAgTE9HKExPR19XQVJOLCAic29ja2V0IiwgIlNvY2tldCBhbHJlYWR5IGNvbm5lY3RlZFxuIik7Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0KLSAgICBJbmV0U29ja2V0QWRkcmVzcyBzb2NrQWRkcjsKLSAgICBpZiAoIXNvY2tBZGRyLmNyZWF0ZShob3N0LCBwb3J0KSkKLSAgICAgICAgcmV0dXJuIC0xOwotCi0gICAgLy9yZXR1cm4gZG9Db25uZWN0KHNvY2tBZGRyKTsKLSAgICBpbnQgZm9vOwotICAgIGZvbyA9IGRvQ29ubmVjdChzb2NrQWRkcik7Ci0gICAgcmV0dXJuIGZvbzsKLX0KLQotLyoKLSAqIENyZWF0ZSBhIHNvY2tldCBhbmQgY29ubmVjdCB0byB0aGUgc3BlY2lmaWVkIGhvc3QgYW5kIHBvcnQuCi0gKi8KLWludCBTb2NrZXQ6OmNvbm5lY3QoY29uc3QgSW5ldEFkZHJlc3MqIGFkZHIsIGludCBwb3J0KQotewotICAgIGlmIChtU29jayAhPSBVTkRFRl9TT0NLRVQpIHsKLSAgICAgICAgTE9HKExPR19XQVJOLCAic29ja2V0IiwgIlNvY2tldCBhbHJlYWR5IGNvbm5lY3RlZFxuIik7Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0KLSAgICBJbmV0U29ja2V0QWRkcmVzcyBzb2NrQWRkcjsKLSAgICBpZiAoIXNvY2tBZGRyLmNyZWF0ZShhZGRyLCBwb3J0KSkKLSAgICAgICAgcmV0dXJuIC0xOwotCi0gICAgcmV0dXJuIGRvQ29ubmVjdChzb2NrQWRkcik7Ci19Ci0KLS8qCi0gKiBGaW5pc2ggY3JlYXRpbmcgYSBzb2NrZXQgYnkgY29ubmVjdGluZyB0byB0aGUgcmVtb3RlIGhvc3QuCi0gKgotICogUmV0dXJucyAwIG9uIHN1Y2Nlc3MuCi0gKi8KLWludCBTb2NrZXQ6OmRvQ29ubmVjdChjb25zdCBJbmV0U29ja2V0QWRkcmVzcyYgc29ja0FkZHIpCi17Ci0jaWZkZWYgSEFWRV9XSU5TT0NLCi0gICAgU09DS0VUIHNvY2s7Ci0jZWxzZQotICAgIGludCBzb2NrOwotI2VuZGlmCi0gICAgY29uc3QgSW5ldEFkZHJlc3MqIGFkZHIgPSBzb2NrQWRkci5nZXRBZGRyZXNzKCk7Ci0gICAgaW50IHBvcnQgPSBzb2NrQWRkci5nZXRQb3J0KCk7Ci0gICAgc3RydWN0IHNvY2thZGRyX2luIGluYWRkcjsKLSAgICBEdXJhdGlvblRpbWVyIGNvbm5lY3RUaW1lcjsKLQotICAgIGFzc2VydChzaXplb2Yoc3RydWN0IHNvY2thZGRyX2luKSA9PSBhZGRyLT5nZXRBZGRyZXNzTGVuZ3RoKCkpOwotICAgIG1lbWNweSgmaW5hZGRyLCBhZGRyLT5nZXRBZGRyZXNzKCksIGFkZHItPmdldEFkZHJlc3NMZW5ndGgoKSk7Ci0gICAgaW5hZGRyLnNpbl9wb3J0ID0gaHRvbnMocG9ydCk7Ci0KLSAgICAvL2ZwcmludGYoc3RkZXJyLCAiLS0tIGNvbm5lY3RpbmcgdG8gJXM6JWRcbiIsCi0gICAgLy8gICAgc29ja0FkZHIuZ2V0SG9zdE5hbWUoKSwgcG9ydCk7Ci0KLSAgICBzb2NrID0gOjpzb2NrZXQoUEZfSU5FVCwgU09DS19TVFJFQU0sIElQUFJPVE9fVENQKTsKLSAgICBpZiAoc29jayA9PSBJTlZBTElEX1NPQ0tFVCkgewotICAgICAgICBpbnQgZXJyID0gZ2V0U29ja2V0RXJyb3IoKTsKLSAgICAgICAgTE9HKExPR19FUlJPUiwgInNvY2tldCIsICJVbmFibGUgdG8gY3JlYXRlIHNvY2tldCAoZXJyPSVkKVxuIiwgZXJyKTsKLSAgICAgICAgcmV0dXJuIChlcnIgIT0gMCkgPyBlcnIgOiAtMTsKLSAgICB9Ci0KLSAgICBjb25uZWN0VGltZXIuc3RhcnQoKTsKLQotICAgIGlmICg6OmNvbm5lY3Qoc29jaywgKHN0cnVjdCBzb2NrYWRkciopICZpbmFkZHIsIHNpemVvZihpbmFkZHIpKSAhPSAwKSB7Ci0gICAgICAgIGludCBlcnIgPSBnZXRTb2NrZXRFcnJvcigpOwotICAgICAgICBMT0coTE9HX1dBUk4sICJzb2NrZXQiLCAiQ29ubmVjdCB0byAlczolZCBmYWlsZWQ6ICVkXG4iLAotICAgICAgICAgICAgc29ja0FkZHIuZ2V0SG9zdE5hbWUoKSwgcG9ydCwgZXJyKTsKLSAgICAgICAgcmV0dXJuIChlcnIgIT0gMCkgPyBlcnIgOiAtMTsKLSAgICB9Ci0KLSAgICBjb25uZWN0VGltZXIuc3RvcCgpOwotICAgIGlmICgobG9uZykgY29ubmVjdFRpbWVyLmR1cmF0aW9uVXNlY3MoKSA+IDEwMDAwMCkgewotICAgICAgICBMT0coTE9HX0lORk8sICJzb2NrZXQiLAotICAgICAgICAgICAgIkNvbm5lY3QgdG8gJXM6JWQgdG9vayAlLjNmc1xuIiwgc29ja0FkZHIuZ2V0SG9zdE5hbWUoKSwKLSAgICAgICAgICAgIHBvcnQsICgobG9uZykgY29ubmVjdFRpbWVyLmR1cmF0aW9uVXNlY3MoKSkgLyAxMDAwMDAwLjApOwotICAgIH0KLQotICAgIG1Tb2NrID0gKHVuc2lnbmVkIGxvbmcpIHNvY2s7Ci0gICAgTE9HKExPR19WRVJCT1NFLCAic29ja2V0IiwKLSAgICAgICAgIi0tLSBjb25uZWN0ZWQgdG8gJXM6JWRcbiIsIHNvY2tBZGRyLmdldEhvc3ROYW1lKCksIHBvcnQpOwotICAgIHJldHVybiAwOwotfQotCi0KLS8qCi0gKiBDbG9zZSB0aGUgc29ja2V0IGlmIGl0IG5lZWRzIGNsb3NpbmcuCi0gKi8KLWJvb2wgU29ja2V0OjpjbG9zZSh2b2lkKQotewotICAgIGlmIChtU29jayAhPSBVTkRFRl9TT0NLRVQpIHsKLSAgICAgICAgLy9mcHJpbnRmKHN0ZGVyciwgIi0tLSBjbG9zaW5nIHNvY2tldCAlbHVcbiIsIG1Tb2NrKTsKLSNpZmRlZiBIQVZFX1dJTlNPQ0sKLSAgICAgICAgaWYgKDo6Y2xvc2Vzb2NrZXQoKFNPQ0tFVCkgbVNvY2spICE9IDApCi0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0jZWxzZQotICAgICAgICBpZiAoOjpjbG9zZSgoaW50KSBtU29jaykgIT0gMCkKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSNlbmRpZgotICAgIH0KLQotICAgIG1Tb2NrID0gVU5ERUZfU09DS0VUOwotCi0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLS8qCi0gKiBSZWFkIGRhdGEgZnJvbSBzb2NrZXQuCi0gKgotICogU3RhbmRhcmQgc2VtYW50aWNzOiByZWFkIHVwIHRvICJsZW4iIGJ5dGVzIGludG8gImJ1ZiIuICBSZXR1cm5zIHRoZQotICogbnVtYmVyIG9mIGJ5dGVzIHJlYWQsIG9yIGxlc3MgdGhhbiB6ZXJvIG9uIGVycm9yLgotICovCi1pbnQgU29ja2V0OjpyZWFkKHZvaWQqIGJ1Ziwgc3NpemVfdCBsZW4pIGNvbnN0Ci17Ci0gICAgaWYgKG1Tb2NrID09IFVOREVGX1NPQ0tFVCkgewotICAgICAgICBMT0coTE9HX0VSUk9SLCAic29ja2V0IiwgIkVSUk9SOiByZWFkIG9uIGludmFsaWQgc29ja2V0XG4iKTsKLSAgICAgICAgcmV0dXJuIC01MDA7Ci0gICAgfQotCi0jaWZkZWYgSEFWRV9XSU5TT0NLCi0gICAgU09DS0VUIHNvY2sgPSAoU09DS0VUKSBtU29jazsKLSNlbHNlCi0gICAgaW50IHNvY2sgPSAoaW50KSBtU29jazsKLSNlbmRpZgotICAgIGludCBjYzsKLQotICAgIGNjID0gcmVjdihzb2NrLCAoY2hhciopYnVmLCBsZW4sIDApOwotICAgIGlmIChjYyA8IDApIHsKLSAgICAgICAgaW50IGVyciA9IGdldFNvY2tldEVycm9yKCk7Ci0gICAgICAgIHJldHVybiAoZXJyID4gMCkgPyAtZXJyIDogLTE7Ci0gICAgfQotCi0gICAgcmV0dXJuIGNjOwotfQotCi0vKgotICogV3JpdGUgZGF0YSB0byBhIHNvY2tldC4KLSAqCi0gKiBTdGFuZGFyZCBzZW1hbnRpY3M6IHdyaXRlIHVwIHRvICJsZW4iIGJ5dGVzIGludG8gImJ1ZiIuICBSZXR1cm5zIHRoZQotICogbnVtYmVyIG9mIGJ5dGVzIHdyaXR0ZW4sIG9yIGxlc3MgdGhhbiB6ZXJvIG9uIGVycm9yLgotICovCi1pbnQgU29ja2V0Ojp3cml0ZShjb25zdCB2b2lkKiBidWYsIHNzaXplX3QgbGVuKSBjb25zdAotewotICAgIGlmIChtU29jayA9PSBVTkRFRl9TT0NLRVQpIHsKLSAgICAgICAgTE9HKExPR19FUlJPUiwgInNvY2tldCIsICJFUlJPUjogd3JpdGUgb24gaW52YWxpZCBzb2NrZXRcbiIpOwotICAgICAgICByZXR1cm4gLTUwMDsKLSAgICB9Ci0KLSNpZmRlZiBIQVZFX1dJTlNPQ0sKLSAgICBTT0NLRVQgc29jayA9IChTT0NLRVQpIG1Tb2NrOwotI2Vsc2UKLSAgICBpbnQgc29jayA9IChpbnQpIG1Tb2NrOwotI2VuZGlmCi0gICAgaW50IGNjOwotCi0gICAgY2MgPSBzZW5kKHNvY2ssIChjb25zdCBjaGFyKilidWYsIGxlbiwgMCk7Ci0gICAgaWYgKGNjIDwgMCkgewotICAgICAgICBpbnQgZXJyID0gZ2V0U29ja2V0RXJyb3IoKTsKLSAgICAgICAgcmV0dXJuIChlcnIgPiAwKSA/IC1lcnIgOiAtMTsKLSAgICB9Ci0KLSAgICByZXR1cm4gY2M7Ci19Ci0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBTb2NrZXQgdGVzdHMKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICovCi0KLS8qCi0gKiBSZWFkIGFsbCBkYXRhIGZyb20gdGhlIHNvY2tldC4gIFRoZSBkYXRhIGlzIHJlYWQgaW50byBhIGJ1ZmZlciB0aGF0Ci0gKiBleHBhbmRzIGFzIG5lZWRlZC4KLSAqCi0gKiBPbiBleGl0LCB0aGUgYnVmZmVyIGlzIHJldHVybmVkLCBhbmQgdGhlIGxlbmd0aCBvZiB0aGUgZGF0YSBpcyBzdG9yZWQKLSAqIGluICIqc3oiLiAgQSBudWxsIGJ5dGUgaXMgYWRkZWQgdG8gdGhlIGVuZCwgYnV0IGlzIG5vdCBpbmNsdWRlZCBpbgotICogdGhlIGxlbmd0aC4KLSAqLwotc3RhdGljIGNoYXIqIHNvY2tldFJlYWRBbGwoY29uc3QgU29ja2V0JiBzLCBpbnQgKnN6KQotewotICAgIGludCBtYXgsIHI7Ci0gICAgY2hhciAqZGF0YSwgKnB0ciwgKnRtcDsKLQotICAgIGRhdGEgPSAoY2hhciopIG1hbGxvYyhtYXggPSAzMjc2OCk7Ci0gICAgaWYgKGRhdGEgPT0gTlVMTCkKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0KLSAgICBwdHIgPSBkYXRhOwotICAgIAotICAgIGZvciAoOzspIHsKLSAgICAgICAgaWYgKChwdHIgLSBkYXRhKSA9PSBtYXgpIHsKLSAgICAgICAgICAgIHRtcCA9IChjaGFyKikgcmVhbGxvYyhkYXRhLCBtYXggKj0gMik7Ci0gICAgICAgICAgICBpZih0bXAgPT0gMCkgewotICAgICAgICAgICAgICAgIGZyZWUoZGF0YSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgciA9IHMucmVhZChwdHIsIG1heCAtIChwdHIgLSBkYXRhKSk7Ci0gICAgICAgIGlmIChyID09IDApCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgaWYgKHIgPCAwKSB7Ci0gICAgICAgICAgICBMT0coTE9HX1dBUk4sICJzb2NrZXQiLCAiV0FSTklORzogc29ja2V0IHJlYWQgZmFpbGVkIChyZXM9JWQpXG4iLHIpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgcHRyICs9IHI7Ci0gICAgfQotCi0gICAgaWYgKChwdHIgLSBkYXRhKSA9PSBtYXgpIHsKLSAgICAgICAgdG1wID0gKGNoYXIqKSByZWFsbG9jKGRhdGEsIG1heCArIDEpOwotICAgICAgICBpZiAodG1wID09IE5VTEwpIHsKLSAgICAgICAgICAgIGZyZWUoZGF0YSk7Ci0gICAgICAgICAgICByZXR1cm4gTlVMTDsKLSAgICAgICAgfQotICAgIH0KLSAgICAqcHRyID0gJ1wwJzsKLSAgICAqc3ogPSAocHRyIC0gZGF0YSk7Ci0gICAgcmV0dXJuIGRhdGE7Ci19Ci0KLS8qCi0gKiBFeGVyY2lzZSB0aGUgU29ja2V0IGNsYXNzLgotICovCi12b2lkIGFuZHJvaWQ6OlRlc3RTb2NrZXRzKHZvaWQpCi17Ci0gICAgcHJpbnRmKCItLS0tLSBTT0NLRVQgVEVTVCAtLS0tLS1cbiIpOwotICAgIFNvY2tldDo6Ym9vdEluaXQoKTsKLQotICAgIGNoYXIqIGJ1ZiA9IE5VTEw7Ci0gICAgaW50IGxlbiwgY2M7Ci0gICAgY29uc3QgY2hhcioga1Rlc3RTdHIgPQotICAgICAgICAiR0VUIC8gSFRUUC8xLjBcbiIKLSAgICAgICAgIkNvbm5lY3Rpb246IGNsb3NlXG4iCi0gICAgICAgICJcbiI7Ci0KLSAgICBTb2NrZXQgc29jazsKLSAgICBpZiAoc29jay5jb25uZWN0KCJ3d3cuZ29vZ2xlLmNvbSIsIDgwKSAhPSAwKSB7Ci0gICAgICAgIGZwcmludGYoc3RkZXJyLCAic29ja2V0IGNvbm5lY3RlZCBmYWlsZWRcbiIpOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgY2MgPSBzb2NrLndyaXRlKGtUZXN0U3RyLCBzdHJsZW4oa1Rlc3RTdHIpKTsKLSAgICBpZiAoY2MgIT0gKGludCkgc3RybGVuKGtUZXN0U3RyKSkgewotICAgICAgICBmcHJpbnRmKHN0ZGVyciwgIndyaXRlIGZhaWxlZCwgcmVzPSVkXG4iLCBjYyk7Ci0gICAgICAgIGdvdG8gYmFpbDsKLSAgICB9Ci0gICAgYnVmID0gc29ja2V0UmVhZEFsbChzb2NrLCAmbGVuKTsKLQotICAgIHByaW50ZigiR09UICclcydcbiIsIGJ1Zik7Ci0KLWJhaWw6Ci0gICAgc29jay5jbG9zZSgpOwotICAgIGZyZWUoYnVmKTsKLX0KLQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9TdGF0aWMuY3BwIGIvbGlicy91dGlscy9TdGF0aWMuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5M2Y3ZTRmLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvU3RhdGljLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDEyMCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8gQWxsIHN0YXRpYyB2YXJpYWJsZXMgZ28gaGVyZSwgdG8gY29udHJvbCBpbml0aWFsaXphdGlvbiBhbmQKLS8vIGRlc3RydWN0aW9uIG9yZGVyIGluIHRoZSBsaWJyYXJ5LgotCi0jaW5jbHVkZSA8cHJpdmF0ZS91dGlscy9TdGF0aWMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0J1ZmZlcmVkVGV4dE91dHB1dC5oPgotI2luY2x1ZGUgPHV0aWxzL0lQQ1RocmVhZFN0YXRlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotY2xhc3MgTGliVXRpbHNGaXJzdFN0YXRpY3MKLXsKLXB1YmxpYzoKLSAgICBMaWJVdGlsc0ZpcnN0U3RhdGljcygpCi0gICAgewotICAgICAgICBpbml0aWFsaXplX3N0cmluZzgoKTsKLSAgICAgICAgaW5pdGlhbGl6ZV9zdHJpbmcxNigpOwotICAgIH0KLSAgICAKLSAgICB+TGliVXRpbHNGaXJzdFN0YXRpY3MoKQotICAgIHsKLSAgICAgICAgdGVybWluYXRlX3N0cmluZzE2KCk7Ci0gICAgICAgIHRlcm1pbmF0ZV9zdHJpbmc4KCk7Ci0gICAgfQotfTsKLQotc3RhdGljIExpYlV0aWxzRmlyc3RTdGF0aWNzIGdGaXJzdFN0YXRpY3M7Ci1pbnQgZ0RhcndpbkNhbnRMb2FkQWxsT2JqZWN0cyA9IDE7Ci0KLS8vIC0tLS0tLS0tLS0tLSBUZXh0IG91dHB1dCBzdHJlYW1zCi0KLVZlY3RvcjxpbnQzMl90PiBnVGV4dEJ1ZmZlcnM7Ci0KLWNsYXNzIExvZ1RleHRPdXRwdXQgOiBwdWJsaWMgQnVmZmVyZWRUZXh0T3V0cHV0Ci17Ci1wdWJsaWM6Ci0gICAgTG9nVGV4dE91dHB1dCgpIDogQnVmZmVyZWRUZXh0T3V0cHV0KE1VTFRJVEhSRUFERUQpIHsgfQotICAgIHZpcnR1YWwgfkxvZ1RleHRPdXRwdXQoKSB7IH07Ci0KLXByb3RlY3RlZDoKLSAgICB2aXJ0dWFsIHN0YXR1c190IHdyaXRlTGluZXMoY29uc3Qgc3RydWN0IGlvdmVjJiB2ZWMsIHNpemVfdCBOKQotICAgIHsKLSAgICAgICAgYW5kcm9pZF93cml0ZXZMb2coJnZlYywgTik7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci19OwotCi1jbGFzcyBGZFRleHRPdXRwdXQgOiBwdWJsaWMgQnVmZmVyZWRUZXh0T3V0cHV0Ci17Ci1wdWJsaWM6Ci0gICAgRmRUZXh0T3V0cHV0KGludCBmZCkgOiBCdWZmZXJlZFRleHRPdXRwdXQoTVVMVElUSFJFQURFRCksIG1GRChmZCkgeyB9Ci0gICAgdmlydHVhbCB+RmRUZXh0T3V0cHV0KCkgeyB9OwotCi1wcm90ZWN0ZWQ6Ci0gICAgdmlydHVhbCBzdGF0dXNfdCB3cml0ZUxpbmVzKGNvbnN0IHN0cnVjdCBpb3ZlYyYgdmVjLCBzaXplX3QgTikKLSAgICB7Ci0gICAgICAgIHdyaXRldihtRkQsICZ2ZWMsIE4pOwotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotCi1wcml2YXRlOgotICAgIGludCBtRkQ7Ci19OwotCi1zdGF0aWMgTG9nVGV4dE91dHB1dCBnTG9nVGV4dE91dHB1dDsKLXN0YXRpYyBGZFRleHRPdXRwdXQgZ1N0ZG91dFRleHRPdXRwdXQoU1RET1VUX0ZJTEVOTyk7Ci1zdGF0aWMgRmRUZXh0T3V0cHV0IGdTdGRlcnJUZXh0T3V0cHV0KFNUREVSUl9GSUxFTk8pOwotCi1UZXh0T3V0cHV0JiBhbG9nKGdMb2dUZXh0T3V0cHV0KTsKLVRleHRPdXRwdXQmIGFvdXQoZ1N0ZG91dFRleHRPdXRwdXQpOwotVGV4dE91dHB1dCYgYWVycihnU3RkZXJyVGV4dE91dHB1dCk7Ci0KLSNpZm5kZWYgTElCVVRJTFNfTkFUSVZFCi0KLS8vIC0tLS0tLS0tLS0tLSBQcm9jZXNzU3RhdGUuY3BwCi0KLU11dGV4IGdQcm9jZXNzTXV0ZXg7Ci1zcDxQcm9jZXNzU3RhdGU+IGdQcm9jZXNzOwotCi1jbGFzcyBMaWJVdGlsc0lQQ3RTdGF0aWNzCi17Ci1wdWJsaWM6Ci0gICAgTGliVXRpbHNJUEN0U3RhdGljcygpCi0gICAgewotICAgIH0KLSAgICAKLSAgICB+TGliVXRpbHNJUEN0U3RhdGljcygpCi0gICAgewotICAgICAgICBJUENUaHJlYWRTdGF0ZTo6c2h1dGRvd24oKTsKLSAgICB9Ci19OwotCi1zdGF0aWMgTGliVXRpbHNJUEN0U3RhdGljcyBnSVBDU3RhdGljczsKLQotLy8gLS0tLS0tLS0tLS0tIFNlcnZpY2VNYW5hZ2VyLmNwcAotCi1NdXRleCBnRGVmYXVsdFNlcnZpY2VNYW5hZ2VyTG9jazsKLXNwPElTZXJ2aWNlTWFuYWdlcj4gZ0RlZmF1bHRTZXJ2aWNlTWFuYWdlcjsKLXNwPElQZXJtaXNzaW9uQ29udHJvbGxlcj4gZ1Blcm1pc3Npb25Db250cm9sbGVyOwotCi0jZW5kaWYKLQotfSAgIC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1N0b3BXYXRjaC5jcHAgYi9saWJzL3V0aWxzL1N0b3BXYXRjaC5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDY4YTFjNTIuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9TdG9wV2F0Y2guY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNzkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwNSBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNkZWZpbmUgTE9HX1RBRyAiU3RvcFdhdGNoIgotCi0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLQotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL0Vycm9ycy5oPgotI2luY2x1ZGUgPHV0aWxzL1N0b3BXYXRjaC5oPgotCi0vKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotCi1TdG9wV2F0Y2g6OlN0b3BXYXRjaChjb25zdCBjaGFyICpuYW1lLCBpbnQgY2xvY2ssIHVpbnQzMl90IGZsYWdzKQotICAgIDogICBtTmFtZShuYW1lKSwgbUNsb2NrKGNsb2NrKSwgbUZsYWdzKGZsYWdzKSwKLSAgICAgICAgbVN0YXJ0VGltZSgwKSwgbU51bUxhcHMoMCkKLXsKLSAgICBtU3RhcnRUaW1lID0gc3lzdGVtVGltZShtQ2xvY2spOwotfQotCi1TdG9wV2F0Y2g6On5TdG9wV2F0Y2goKQotewotICAgIG5zZWNzX3QgZWxhcHNlZCA9IGVsYXBzZWRUaW1lKCk7Ci0gICAgY29uc3QgaW50IG4gPSBtTnVtTGFwczsKLSAgICBMT0dEKCJTdG9wV2F0Y2ggJXMgKHVzKTogJWxsZCAiLCBtTmFtZSwgbnMydXMoZWxhcHNlZCkpOwotICAgIGZvciAoaW50IGk9MCA7IGk8biA7IGkrKykgewotICAgICAgICBjb25zdCBuc2Vjc190IHNvRmFyID0gbUxhcHNbaV0uc29GYXI7Ci0gICAgICAgIGNvbnN0IG5zZWNzX3QgdGhpc0xhcCA9IG1MYXBzW2ldLnRoaXNMYXA7Ci0gICAgICAgIExPR0QoIiBbJWQ6ICVsbGQsICVsbGRdIiwgaSwgbnMydXMoc29GYXIpLCBuczJ1cyh0aGlzTGFwKSk7Ci0gICAgfQotfQotCi1jb25zdCBjaGFyKiBTdG9wV2F0Y2g6Om5hbWUoKSBjb25zdAotewotICAgIHJldHVybiBtTmFtZTsKLX0KLQotbnNlY3NfdCBTdG9wV2F0Y2g6OmxhcCgpCi17Ci0gICAgbnNlY3NfdCBlbGFwc2VkID0gZWxhcHNlZFRpbWUoKTsKLSAgICBpZiAobU51bUxhcHMgPj0gOCkgewotICAgICAgICBlbGFwc2VkID0gMDsKLSAgICB9IGVsc2UgewotICAgICAgICBjb25zdCBpbnQgbiA9IG1OdW1MYXBzOwotICAgICAgICBtTGFwc1tuXS5zb0ZhciAgID0gZWxhcHNlZDsKLSAgICAgICAgbUxhcHNbbl0udGhpc0xhcCA9IG4gPyAoZWxhcHNlZCAtIG1MYXBzW24tMV0uc29GYXIpIDogZWxhcHNlZDsKLSAgICAgICAgbU51bUxhcHMgPSBuKzE7Ci0gICAgfQotICAgIHJldHVybiBlbGFwc2VkOwotfQotCi1uc2Vjc190IFN0b3BXYXRjaDo6ZWxhcHNlZFRpbWUoKSBjb25zdAotewotICAgIHJldHVybiBzeXN0ZW1UaW1lKG1DbG9jaykgLSBtU3RhcnRUaW1lOwotfQotCi0KLS8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9TdHJpbmcxNi5jcHAgYi9saWJzL3V0aWxzL1N0cmluZzE2LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMWY4MWNhZC4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1N0cmluZzE2LmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDYwOSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzE2Lmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9EZWJ1Zy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLSNpbmNsdWRlIDx1dGlscy9UZXh0T3V0cHV0Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS91dGlscy9TdGF0aWMuaD4KLQotI2lmZGVmIEhBVkVfV0lOU09DSwotIyB1bmRlZiAgbmh0b2wKLSMgdW5kZWYgIGh0b25sCi0jIHVuZGVmICBuaHRvcwotIyB1bmRlZiAgaHRvbnMKLQotIyBpZmRlZiBIQVZFX0xJVFRMRV9FTkRJQU4KLSMgIGRlZmluZSBudG9obCh4KSAgICAoICgoeCkgPDwgMjQpIHwgKCgoeCkgPj4gMjQpICYgMjU1KSB8ICgoKHgpIDw8IDgpICYgMHhmZjAwMDApIHwgKCgoeCkgPj4gOCkgJiAweGZmMDApICkKLSMgIGRlZmluZSBodG9ubCh4KSAgICBudG9obCh4KQotIyAgZGVmaW5lIG50b2hzKHgpICAgICggKCgoeCkgPDwgOCkgJiAweGZmMDApIHwgKCgoeCkgPj4gOCkgJiAyNTUpICkKLSMgIGRlZmluZSBodG9ucyh4KSAgICBudG9ocyh4KQotIyBlbHNlCi0jICBkZWZpbmUgbnRvaGwoeCkgICAgKHgpCi0jICBkZWZpbmUgaHRvbmwoeCkgICAgKHgpCi0jICBkZWZpbmUgbnRvaHMoeCkgICAgKHgpCi0jICBkZWZpbmUgaHRvbnMoeCkgICAgKHgpCi0jIGVuZGlmCi0jZWxzZQotIyBpbmNsdWRlIDxuZXRpbmV0L2luLmg+Ci0jZW5kaWYKLQotI2luY2x1ZGUgPG1lbW9yeS5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8Y3R5cGUuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWludCBzdHJjbXAxNihjb25zdCBjaGFyMTZfdCAqczEsIGNvbnN0IGNoYXIxNl90ICpzMikKLXsKLSAgY2hhcjE2X3QgY2g7Ci0gIGludCBkID0gMDsKLQotICB3aGlsZSAoIDEgKSB7Ci0gICAgZCA9IChpbnQpKGNoID0gKnMxKyspIC0gKGludCkqczIrKzsKLSAgICBpZiAoIGQgfHwgIWNoICkKLSAgICAgIGJyZWFrOwotICB9Ci0KLSAgcmV0dXJuIGQ7Ci19Ci0KLWludCBzdHJuY21wMTYoY29uc3QgY2hhcjE2X3QgKnMxLCBjb25zdCBjaGFyMTZfdCAqczIsIHNpemVfdCBuKQotewotICBjaGFyMTZfdCBjaDsKLSAgaW50IGQgPSAwOwotCi0gIHdoaWxlICggbi0tICkgewotICAgIGQgPSAoaW50KShjaCA9ICpzMSsrKSAtIChpbnQpKnMyKys7Ci0gICAgaWYgKCBkIHx8ICFjaCApCi0gICAgICBicmVhazsKLSAgfQotCi0gIHJldHVybiBkOwotfQotCi1jaGFyMTZfdCAqc3RyY3B5MTYoY2hhcjE2X3QgKmRzdCwgY29uc3QgY2hhcjE2X3QgKnNyYykKLXsKLSAgY2hhcjE2X3QgKnEgPSBkc3Q7Ci0gIGNvbnN0IGNoYXIxNl90ICpwID0gc3JjOwotICBjaGFyMTZfdCBjaDsKLQotICBkbyB7Ci0gICAgKnErKyA9IGNoID0gKnArKzsKLSAgfSB3aGlsZSAoIGNoICk7Ci0KLSAgcmV0dXJuIGRzdDsKLX0KLQotc2l6ZV90IHN0cmxlbjE2KGNvbnN0IGNoYXIxNl90ICpzKQotewotICBjb25zdCBjaGFyMTZfdCAqc3MgPSBzOwotICB3aGlsZSAoICpzcyApCi0gICAgc3MrKzsKLSAgcmV0dXJuIHNzLXM7Ci19Ci0KLQotY2hhcjE2X3QgKnN0cm5jcHkxNihjaGFyMTZfdCAqZHN0LCBjb25zdCBjaGFyMTZfdCAqc3JjLCBzaXplX3QgbikKLXsKLSAgY2hhcjE2X3QgKnEgPSBkc3Q7Ci0gIGNvbnN0IGNoYXIxNl90ICpwID0gc3JjOwotICBjaGFyIGNoOwotCi0gIHdoaWxlIChuKSB7Ci0gICAgbi0tOwotICAgICpxKysgPSBjaCA9ICpwKys7Ci0gICAgaWYgKCAhY2ggKQotICAgICAgYnJlYWs7Ci0gIH0KLQotICAqcSA9IDA7Ci0KLSAgcmV0dXJuIGRzdDsKLX0KLQotc2l6ZV90IHN0cm5sZW4xNihjb25zdCBjaGFyMTZfdCAqcywgc2l6ZV90IG1heGxlbikKLXsKLSAgY29uc3QgY2hhcjE2X3QgKnNzID0gczsKLQotICAvKiBJbXBvcnRhbnQ6IHRoZSBtYXhsZW4gdGVzdCBtdXN0IHByZWNlZGUgdGhlIHJlZmVyZW5jZSB0aHJvdWdoIHNzOwotICAgICBzaW5jZSB0aGUgYnl0ZSBiZXlvbmQgdGhlIG1heGltdW0gbWF5IHNlZ2ZhdWx0ICovCi0gIHdoaWxlICgobWF4bGVuID4gMCkgJiYgKnNzKSB7Ci0gICAgc3MrKzsKLSAgICBtYXhsZW4tLTsKLSAgfQotICByZXR1cm4gc3MtczsKLX0KLQotaW50IHN0cnpjbXAxNihjb25zdCBjaGFyMTZfdCAqczEsIHNpemVfdCBuMSwgY29uc3QgY2hhcjE2X3QgKnMyLCBzaXplX3QgbjIpCi17Ci0gICAgY29uc3QgY2hhcjE2X3QqIGUxID0gczErbjE7Ci0gICAgY29uc3QgY2hhcjE2X3QqIGUyID0gczIrbjI7Ci0KLSAgICB3aGlsZSAoczEgPCBlMSAmJiBzMiA8IGUyKSB7Ci0gICAgICAgIGNvbnN0IGludCBkID0gKGludCkqczErKyAtIChpbnQpKnMyKys7Ci0gICAgICAgIGlmIChkKSB7Ci0gICAgICAgICAgICByZXR1cm4gZDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBuMSA8IG4yCi0gICAgICAgID8gKDAgLSAoaW50KSpzMikKLSAgICAgICAgOiAobjEgPiBuMgotICAgICAgICAgICA/ICgoaW50KSpzMSAtIDApCi0gICAgICAgICAgIDogMCk7Ci19Ci0KLWludCBzdHJ6Y21wMTZfaF9uKGNvbnN0IGNoYXIxNl90ICpzMUgsIHNpemVfdCBuMSwgY29uc3QgY2hhcjE2X3QgKnMyTiwgc2l6ZV90IG4yKQotewotICAgIGNvbnN0IGNoYXIxNl90KiBlMSA9IHMxSCtuMTsKLSAgICBjb25zdCBjaGFyMTZfdCogZTIgPSBzMk4rbjI7Ci0KLSAgICB3aGlsZSAoczFIIDwgZTEgJiYgczJOIDwgZTIpIHsKLSAgICAgICAgY29uc3QgY2hhcjE2X3QgYzIgPSBudG9ocygqczJOKTsKLSAgICAgICAgY29uc3QgaW50IGQgPSAoaW50KSpzMUgrKyAtIChpbnQpYzI7Ci0gICAgICAgIHMyTisrOwotICAgICAgICBpZiAoZCkgewotICAgICAgICAgICAgcmV0dXJuIGQ7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gbjEgPCBuMgotICAgICAgICA/ICgwIC0gKGludCludG9ocygqczJOKSkKLSAgICAgICAgOiAobjEgPiBuMgotICAgICAgICAgICA/ICgoaW50KSpzMUggLSAwKQotICAgICAgICAgICA6IDApOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1zdGF0aWMgaW5saW5lIHNpemVfdAotdXRmOF9jaGFyX2xlbih1aW50OF90IGNoKQotewotICAgIHJldHVybiAoKDB4ZTUwMDAwMDAgPj4gKChjaCA+PiAzKSAmIDB4MWUpKSAmIDMpICsgMTsKLX0KLQotI2RlZmluZSBVVEY4X1NISUZUX0FORF9NQVNLKHVuaWNvZGUsIGJ5dGUpICAodW5pY29kZSk8PD02OyAodW5pY29kZSkgfD0gKDB4M2YgJiAoYnl0ZSkpOwotCi1zdGF0aWMgaW5saW5lIHVpbnQzMl90Ci11dGY4X3RvX3V0ZjMyKGNvbnN0IHVpbnQ4X3QgKnNyYywgc2l6ZV90IGxlbmd0aCkKLXsKLSAgICB1aW50MzJfdCB1bmljb2RlOwotCi0gICAgc3dpdGNoIChsZW5ndGgpCi0gICAgewotICAgICAgICBjYXNlIDE6Ci0gICAgICAgICAgICByZXR1cm4gc3JjWzBdOwotICAgICAgICBjYXNlIDI6Ci0gICAgICAgICAgICB1bmljb2RlID0gc3JjWzBdICYgMHgxZjsKLSAgICAgICAgICAgIFVURjhfU0hJRlRfQU5EX01BU0sodW5pY29kZSwgc3JjWzFdKQotICAgICAgICAgICAgcmV0dXJuIHVuaWNvZGU7Ci0gICAgICAgIGNhc2UgMzoKLSAgICAgICAgICAgIHVuaWNvZGUgPSBzcmNbMF0gJiAweDBmOwotICAgICAgICAgICAgVVRGOF9TSElGVF9BTkRfTUFTSyh1bmljb2RlLCBzcmNbMV0pCi0gICAgICAgICAgICBVVEY4X1NISUZUX0FORF9NQVNLKHVuaWNvZGUsIHNyY1syXSkKLSAgICAgICAgICAgIHJldHVybiB1bmljb2RlOwotICAgICAgICBjYXNlIDQ6Ci0gICAgICAgICAgICB1bmljb2RlID0gc3JjWzBdICYgMHgwNzsKLSAgICAgICAgICAgIFVURjhfU0hJRlRfQU5EX01BU0sodW5pY29kZSwgc3JjWzFdKQotICAgICAgICAgICAgVVRGOF9TSElGVF9BTkRfTUFTSyh1bmljb2RlLCBzcmNbMl0pCi0gICAgICAgICAgICBVVEY4X1NISUZUX0FORF9NQVNLKHVuaWNvZGUsIHNyY1szXSkKLSAgICAgICAgICAgIHJldHVybiB1bmljb2RlOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgcmV0dXJuIDB4ZmZmZjsKLSAgICB9Ci0gICAgCi0gICAgLy9wcmludGYoIkNoYXIgYXQgJXA6IGxlbj0lZCwgdXRmLTE2PSVwXG4iLCBzcmMsIGxlbmd0aCwgKHZvaWQqKXJlc3VsdCk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgU2hhcmVkQnVmZmVyKiBnRW1wdHlTdHJpbmdCdWYgPSBOVUxMOwotc3RhdGljIGNoYXIxNl90KiBnRW1wdHlTdHJpbmcgPSBOVUxMOwotCi1zdGF0aWMgaW5saW5lIGNoYXIxNl90KiBnZXRFbXB0eVN0cmluZygpCi17Ci0gICAgZ0VtcHR5U3RyaW5nQnVmLT5hY3F1aXJlKCk7Ci0gICByZXR1cm4gZ0VtcHR5U3RyaW5nOwotfQotCi12b2lkIGluaXRpYWxpemVfc3RyaW5nMTYoKQotewotICAgIFNoYXJlZEJ1ZmZlciogYnVmID0gU2hhcmVkQnVmZmVyOjphbGxvYyhzaXplb2YoY2hhcjE2X3QpKTsKLSAgICBjaGFyMTZfdCogc3RyID0gKGNoYXIxNl90KilidWYtPmRhdGEoKTsKLSAgICAqc3RyID0gMDsKLSAgICBnRW1wdHlTdHJpbmdCdWYgPSBidWY7Ci0gICAgZ0VtcHR5U3RyaW5nID0gc3RyOwotfQotCi12b2lkIHRlcm1pbmF0ZV9zdHJpbmcxNigpCi17Ci0gICAgU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShnRW1wdHlTdHJpbmcpLT5yZWxlYXNlKCk7Ci0gICAgZ0VtcHR5U3RyaW5nQnVmID0gTlVMTDsKLSAgICBnRW1wdHlTdHJpbmcgPSBOVUxMOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLy8gTm90ZTogbm90IGRlYWxpbmcgd2l0aCBnZW5lcmF0aW5nIHN1cnJvZ2F0ZSBwYWlycy4KLXN0YXRpYyBjaGFyMTZfdCogYWxsb2NGcm9tVVRGOChjb25zdCBjaGFyKiBpbiwgc2l6ZV90IGxlbikKLXsKLSAgICBpZiAobGVuID09IDApIHJldHVybiBnZXRFbXB0eVN0cmluZygpOwotICAgIAotICAgIHNpemVfdCBjaGFycyA9IDA7Ci0gICAgY29uc3QgY2hhciogZW5kID0gaW4rbGVuOwotICAgIGNvbnN0IGNoYXIqIHAgPSBpbjsKLSAgICAKLSAgICB3aGlsZSAocCA8IGVuZCkgewotICAgICAgICBjaGFycysrOwotICAgICAgICBwICs9IHV0ZjhfY2hhcl9sZW4oKnApOwotICAgIH0KLSAgICAKLSAgICBTaGFyZWRCdWZmZXIqIGJ1ZiA9IFNoYXJlZEJ1ZmZlcjo6YWxsb2MoKGNoYXJzKzEpKnNpemVvZihjaGFyMTZfdCkpOwotICAgIGlmIChidWYpIHsKLSAgICAgICAgcCA9IGluOwotICAgICAgICBjaGFyMTZfdCogc3RyID0gKGNoYXIxNl90KilidWYtPmRhdGEoKTsKLSAgICAgICAgY2hhcjE2X3QqIGQgPSBzdHI7Ci0gICAgICAgIHdoaWxlIChwIDwgZW5kKSB7Ci0gICAgICAgICAgICBzaXplX3QgbGVuID0gdXRmOF9jaGFyX2xlbigqcCk7Ci0gICAgICAgICAgICAqZCsrID0gKGNoYXIxNl90KXV0ZjhfdG9fdXRmMzIoKGNvbnN0IHVpbnQ4X3QqKXAsIGxlbik7Ci0gICAgICAgICAgICBwICs9IGxlbjsKLSAgICAgICAgfQotICAgICAgICAqZCA9IDA7Ci0gICAgICAgIAotICAgICAgICAvL3ByaW50ZigiQ3JlYXRlZCBVVEYtMTYgc3RyaW5nIGZyb20gVVRGLTggXCIlc1wiOiIsIGluKTsKLSAgICAgICAgLy9wcmludEhleERhdGEoMSwgc3RyLCBidWYtPnNpemUoKSwgMTYsIDEpOwotICAgICAgICAvL3ByaW50ZigiXG4iKTsKLSAgICAgICAgCi0gICAgICAgIHJldHVybiBzdHI7Ci0gICAgfQotICAgIAotICAgIHJldHVybiBnZXRFbXB0eVN0cmluZygpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotU3RyaW5nMTY6OlN0cmluZzE2KCkKLSAgICA6IG1TdHJpbmcoZ2V0RW1wdHlTdHJpbmcoKSkKLXsKLX0KLQotU3RyaW5nMTY6OlN0cmluZzE2KGNvbnN0IFN0cmluZzE2JiBvKQotICAgIDogbVN0cmluZyhvLm1TdHJpbmcpCi17Ci0gICAgU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKS0+YWNxdWlyZSgpOwotfQotCi1TdHJpbmcxNjo6U3RyaW5nMTYoY29uc3QgU3RyaW5nMTYmIG8sIHNpemVfdCBsZW4sIHNpemVfdCBiZWdpbikKLSAgICA6IG1TdHJpbmcoZ2V0RW1wdHlTdHJpbmcoKSkKLXsKLSAgICBzZXRUbyhvLCBsZW4sIGJlZ2luKTsKLX0KLQotU3RyaW5nMTY6OlN0cmluZzE2KGNvbnN0IGNoYXIxNl90KiBvKQotewotICAgIHNpemVfdCBsZW4gPSBzdHJsZW4xNihvKTsKLSAgICBTaGFyZWRCdWZmZXIqIGJ1ZiA9IFNoYXJlZEJ1ZmZlcjo6YWxsb2MoKGxlbisxKSpzaXplb2YoY2hhcjE2X3QpKTsKLSAgICBMT0dfQVNTRVJUKGJ1ZiwgIlVuYWJsZSB0byBhbGxvY2F0ZSBzaGFyZWQgYnVmZmVyIik7Ci0gICAgaWYgKGJ1ZikgewotICAgICAgICBjaGFyMTZfdCogc3RyID0gKGNoYXIxNl90KilidWYtPmRhdGEoKTsKLSAgICAgICAgc3RyY3B5MTYoc3RyLCBvKTsKLSAgICAgICAgbVN0cmluZyA9IHN0cjsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICAKLSAgICBtU3RyaW5nID0gZ2V0RW1wdHlTdHJpbmcoKTsKLX0KLQotU3RyaW5nMTY6OlN0cmluZzE2KGNvbnN0IGNoYXIxNl90KiBvLCBzaXplX3QgbGVuKQotewotICAgIFNoYXJlZEJ1ZmZlciogYnVmID0gU2hhcmVkQnVmZmVyOjphbGxvYygobGVuKzEpKnNpemVvZihjaGFyMTZfdCkpOwotICAgIExPR19BU1NFUlQoYnVmLCAiVW5hYmxlIHRvIGFsbG9jYXRlIHNoYXJlZCBidWZmZXIiKTsKLSAgICBpZiAoYnVmKSB7Ci0gICAgICAgIGNoYXIxNl90KiBzdHIgPSAoY2hhcjE2X3QqKWJ1Zi0+ZGF0YSgpOwotICAgICAgICBtZW1jcHkoc3RyLCBvLCBsZW4qc2l6ZW9mKGNoYXIxNl90KSk7Ci0gICAgICAgIHN0cltsZW5dID0gMDsKLSAgICAgICAgbVN0cmluZyA9IHN0cjsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICAKLSAgICBtU3RyaW5nID0gZ2V0RW1wdHlTdHJpbmcoKTsKLX0KLQotU3RyaW5nMTY6OlN0cmluZzE2KGNvbnN0IFN0cmluZzgmIG8pCi0gICAgOiBtU3RyaW5nKGFsbG9jRnJvbVVURjgoby5zdHJpbmcoKSwgby5zaXplKCkpKQotewotfQotCi1TdHJpbmcxNjo6U3RyaW5nMTYoY29uc3QgY2hhciogbykKLSAgICA6IG1TdHJpbmcoYWxsb2NGcm9tVVRGOChvLCBzdHJsZW4obykpKQotewotfQotCi1TdHJpbmcxNjo6U3RyaW5nMTYoY29uc3QgY2hhciogbywgc2l6ZV90IGxlbikKLSAgICA6IG1TdHJpbmcoYWxsb2NGcm9tVVRGOChvLCBsZW4pKQotewotfQotCi1TdHJpbmcxNjo6flN0cmluZzE2KCkKLXsKLSAgICBTaGFyZWRCdWZmZXI6OmJ1ZmZlckZyb21EYXRhKG1TdHJpbmcpLT5yZWxlYXNlKCk7Ci19Ci0KLXZvaWQgU3RyaW5nMTY6OnNldFRvKGNvbnN0IFN0cmluZzE2JiBvdGhlcikKLXsKLSAgICBTaGFyZWRCdWZmZXI6OmJ1ZmZlckZyb21EYXRhKG90aGVyLm1TdHJpbmcpLT5hY3F1aXJlKCk7Ci0gICAgU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKS0+cmVsZWFzZSgpOwotICAgIG1TdHJpbmcgPSBvdGhlci5tU3RyaW5nOwotfQotCi1zdGF0dXNfdCBTdHJpbmcxNjo6c2V0VG8oY29uc3QgU3RyaW5nMTYmIG90aGVyLCBzaXplX3QgbGVuLCBzaXplX3QgYmVnaW4pCi17Ci0gICAgY29uc3Qgc2l6ZV90IE4gPSBvdGhlci5zaXplKCk7Ci0gICAgaWYgKGJlZ2luID49IE4pIHsKLSAgICAgICAgU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKS0+cmVsZWFzZSgpOwotICAgICAgICBtU3RyaW5nID0gZ2V0RW1wdHlTdHJpbmcoKTsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLSAgICBpZiAoKGJlZ2luK2xlbikgPiBOKSBsZW4gPSBOLWJlZ2luOwotICAgIGlmIChiZWdpbiA9PSAwICYmIGxlbiA9PSBOKSB7Ci0gICAgICAgIHNldFRvKG90aGVyKTsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLQotICAgIGlmICgmb3RoZXIgPT0gdGhpcykgewotICAgICAgICBMT0dfQUxXQVlTX0ZBVEFMKCJOb3QgaW1wbGVtZW50ZWQiKTsKLSAgICB9Ci0KLSAgICByZXR1cm4gc2V0VG8ob3RoZXIuc3RyaW5nKCkrYmVnaW4sIGxlbik7Ci19Ci0KLXN0YXR1c190IFN0cmluZzE2OjpzZXRUbyhjb25zdCBjaGFyMTZfdCogb3RoZXIpCi17Ci0gICAgcmV0dXJuIHNldFRvKG90aGVyLCBzdHJsZW4xNihvdGhlcikpOwotfQotCi1zdGF0dXNfdCBTdHJpbmcxNjo6c2V0VG8oY29uc3QgY2hhcjE2X3QqIG90aGVyLCBzaXplX3QgbGVuKQotewotICAgIFNoYXJlZEJ1ZmZlciogYnVmID0gU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKQotICAgICAgICAtPmVkaXRSZXNpemUoKGxlbisxKSpzaXplb2YoY2hhcjE2X3QpKTsKLSAgICBpZiAoYnVmKSB7Ci0gICAgICAgIGNoYXIxNl90KiBzdHIgPSAoY2hhcjE2X3QqKWJ1Zi0+ZGF0YSgpOwotICAgICAgICBtZW1jcHkoc3RyLCBvdGhlciwgbGVuKnNpemVvZihjaGFyMTZfdCkpOwotICAgICAgICBzdHJbbGVuXSA9IDA7Ci0gICAgICAgIG1TdHJpbmcgPSBzdHI7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX01FTU9SWTsKLX0KLQotc3RhdHVzX3QgU3RyaW5nMTY6OmFwcGVuZChjb25zdCBTdHJpbmcxNiYgb3RoZXIpCi17Ci0gICAgY29uc3Qgc2l6ZV90IG15TGVuID0gc2l6ZSgpOwotICAgIGNvbnN0IHNpemVfdCBvdGhlckxlbiA9IG90aGVyLnNpemUoKTsKLSAgICBpZiAobXlMZW4gPT0gMCkgewotICAgICAgICBzZXRUbyhvdGhlcik7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9IGVsc2UgaWYgKG90aGVyTGVuID09IDApIHsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLSAgICAKLSAgICBTaGFyZWRCdWZmZXIqIGJ1ZiA9IFNoYXJlZEJ1ZmZlcjo6YnVmZmVyRnJvbURhdGEobVN0cmluZykKLSAgICAgICAgLT5lZGl0UmVzaXplKChteUxlbitvdGhlckxlbisxKSpzaXplb2YoY2hhcjE2X3QpKTsKLSAgICBpZiAoYnVmKSB7Ci0gICAgICAgIGNoYXIxNl90KiBzdHIgPSAoY2hhcjE2X3QqKWJ1Zi0+ZGF0YSgpOwotICAgICAgICBtZW1jcHkoc3RyK215TGVuLCBvdGhlciwgKG90aGVyTGVuKzEpKnNpemVvZihjaGFyMTZfdCkpOwotICAgICAgICBtU3RyaW5nID0gc3RyOwotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotICAgIHJldHVybiBOT19NRU1PUlk7Ci19Ci0KLXN0YXR1c190IFN0cmluZzE2OjphcHBlbmQoY29uc3QgY2hhcjE2X3QqIGNocnMsIHNpemVfdCBvdGhlckxlbikKLXsKLSAgICBjb25zdCBzaXplX3QgbXlMZW4gPSBzaXplKCk7Ci0gICAgaWYgKG15TGVuID09IDApIHsKLSAgICAgICAgc2V0VG8oY2hycywgb3RoZXJMZW4pOwotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfSBlbHNlIGlmIChvdGhlckxlbiA9PSAwKSB7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgCi0gICAgU2hhcmVkQnVmZmVyKiBidWYgPSBTaGFyZWRCdWZmZXI6OmJ1ZmZlckZyb21EYXRhKG1TdHJpbmcpCi0gICAgICAgIC0+ZWRpdFJlc2l6ZSgobXlMZW4rb3RoZXJMZW4rMSkqc2l6ZW9mKGNoYXIxNl90KSk7Ci0gICAgaWYgKGJ1ZikgewotICAgICAgICBjaGFyMTZfdCogc3RyID0gKGNoYXIxNl90KilidWYtPmRhdGEoKTsKLSAgICAgICAgbWVtY3B5KHN0citteUxlbiwgY2hycywgb3RoZXJMZW4qc2l6ZW9mKGNoYXIxNl90KSk7Ci0gICAgICAgIHN0cltteUxlbitvdGhlckxlbl0gPSAwOwotICAgICAgICBtU3RyaW5nID0gc3RyOwotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgfQotICAgIHJldHVybiBOT19NRU1PUlk7Ci19Ci0KLXN0YXR1c190IFN0cmluZzE2OjppbnNlcnQoc2l6ZV90IHBvcywgY29uc3QgY2hhcjE2X3QqIGNocnMpCi17Ci0gICAgcmV0dXJuIGluc2VydChwb3MsIGNocnMsIHN0cmxlbjE2KGNocnMpKTsKLX0KLQotc3RhdHVzX3QgU3RyaW5nMTY6Omluc2VydChzaXplX3QgcG9zLCBjb25zdCBjaGFyMTZfdCogY2hycywgc2l6ZV90IGxlbikKLXsKLSAgICBjb25zdCBzaXplX3QgbXlMZW4gPSBzaXplKCk7Ci0gICAgaWYgKG15TGVuID09IDApIHsKLSAgICAgICAgcmV0dXJuIHNldFRvKGNocnMsIGxlbik7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9IGVsc2UgaWYgKGxlbiA9PSAwKSB7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0KLSAgICBpZiAocG9zID4gbXlMZW4pIHBvcyA9IG15TGVuOwotCi0gICAgI2lmIDAKLSAgICBwcmludGYoIkluc2VydCBpbiB0byAlczogcG9zPSVkLCBsZW49JWQsIG15TGVuPSVkLCBjaHJzPSVzXG4iLAotICAgICAgICAgICBTdHJpbmc4KCp0aGlzKS5zdHJpbmcoKSwgcG9zLAotICAgICAgICAgICBsZW4sIG15TGVuLCBTdHJpbmc4KGNocnMsIGxlbikuc3RyaW5nKCkpOwotICAgICNlbmRpZgotCi0gICAgU2hhcmVkQnVmZmVyKiBidWYgPSBTaGFyZWRCdWZmZXI6OmJ1ZmZlckZyb21EYXRhKG1TdHJpbmcpCi0gICAgICAgIC0+ZWRpdFJlc2l6ZSgobXlMZW4rbGVuKzEpKnNpemVvZihjaGFyMTZfdCkpOwotICAgIGlmIChidWYpIHsKLSAgICAgICAgY2hhcjE2X3QqIHN0ciA9IChjaGFyMTZfdCopYnVmLT5kYXRhKCk7Ci0gICAgICAgIGlmIChwb3MgPCBteUxlbikgewotICAgICAgICAgICAgbWVtbW92ZShzdHIrcG9zK2xlbiwgc3RyK3BvcywgKG15TGVuLXBvcykqc2l6ZW9mKGNoYXIxNl90KSk7Ci0gICAgICAgIH0KLSAgICAgICAgbWVtY3B5KHN0citwb3MsIGNocnMsIGxlbipzaXplb2YoY2hhcjE2X3QpKTsKLSAgICAgICAgc3RyW215TGVuK2xlbl0gPSAwOwotICAgICAgICBtU3RyaW5nID0gc3RyOwotICAgICAgICAjaWYgMAotICAgICAgICBwcmludGYoIlJlc3VsdCAoJWQgY2hycyk6ICVzXG4iLCBzaXplKCksIFN0cmluZzgoKnRoaXMpLnN0cmluZygpKTsKLSAgICAgICAgI2VuZGlmCi0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX01FTU9SWTsKLX0KLQotc3NpemVfdCBTdHJpbmcxNjo6ZmluZEZpcnN0KGNoYXIxNl90IGMpIGNvbnN0Ci17Ci0gICAgY29uc3QgY2hhcjE2X3QqIHN0ciA9IHN0cmluZygpOwotICAgIGNvbnN0IGNoYXIxNl90KiBwID0gc3RyOwotICAgIGNvbnN0IGNoYXIxNl90KiBlID0gcCArIHNpemUoKTsKLSAgICB3aGlsZSAocCA8IGUpIHsKLSAgICAgICAgaWYgKCpwID09IGMpIHsKLSAgICAgICAgICAgIHJldHVybiBwLXN0cjsKLSAgICAgICAgfQotICAgICAgICBwKys7Ci0gICAgfQotICAgIHJldHVybiAtMTsKLX0KLQotc3NpemVfdCBTdHJpbmcxNjo6ZmluZExhc3QoY2hhcjE2X3QgYykgY29uc3QKLXsKLSAgICBjb25zdCBjaGFyMTZfdCogc3RyID0gc3RyaW5nKCk7Ci0gICAgY29uc3QgY2hhcjE2X3QqIHAgPSBzdHI7Ci0gICAgY29uc3QgY2hhcjE2X3QqIGUgPSBwICsgc2l6ZSgpOwotICAgIHdoaWxlIChwIDwgZSkgewotICAgICAgICBlLS07Ci0gICAgICAgIGlmICgqZSA9PSBjKSB7Ci0gICAgICAgICAgICByZXR1cm4gZS1zdHI7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIC0xOwotfQotCi1ib29sIFN0cmluZzE2OjpzdGFydHNXaXRoKGNvbnN0IFN0cmluZzE2JiBwcmVmaXgpIGNvbnN0Ci17Ci0gICAgY29uc3Qgc2l6ZV90IHBzID0gcHJlZml4LnNpemUoKTsKLSAgICBpZiAocHMgPiBzaXplKCkpIHJldHVybiBmYWxzZTsKLSAgICByZXR1cm4gc3RyemNtcDE2KG1TdHJpbmcsIHBzLCBwcmVmaXguc3RyaW5nKCksIHBzKSA9PSAwOwotfQotCi1ib29sIFN0cmluZzE2OjpzdGFydHNXaXRoKGNvbnN0IGNoYXIxNl90KiBwcmVmaXgpIGNvbnN0Ci17Ci0gICAgY29uc3Qgc2l6ZV90IHBzID0gc3RybGVuMTYocHJlZml4KTsKLSAgICBpZiAocHMgPiBzaXplKCkpIHJldHVybiBmYWxzZTsKLSAgICByZXR1cm4gc3RybmNtcDE2KG1TdHJpbmcsIHByZWZpeCwgcHMpID09IDA7Ci19Ci0KLXN0YXR1c190IFN0cmluZzE2OjptYWtlTG93ZXIoKQotewotICAgIGNvbnN0IHNpemVfdCBOID0gc2l6ZSgpOwotICAgIGNvbnN0IGNoYXIxNl90KiBzdHIgPSBzdHJpbmcoKTsKLSAgICBjaGFyMTZfdCogZWRpdCA9IE5VTEw7Ci0gICAgZm9yIChzaXplX3QgaT0wOyBpPE47IGkrKykgewotICAgICAgICBjb25zdCBjaGFyMTZfdCB2ID0gc3RyW2ldOwotICAgICAgICBpZiAodiA+PSAnQScgJiYgdiA8PSAnWicpIHsKLSAgICAgICAgICAgIGlmICghZWRpdCkgewotICAgICAgICAgICAgICAgIFNoYXJlZEJ1ZmZlciogYnVmID0gU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKS0+ZWRpdCgpOwotICAgICAgICAgICAgICAgIGlmICghYnVmKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGVkaXQgPSAoY2hhcjE2X3QqKWJ1Zi0+ZGF0YSgpOwotICAgICAgICAgICAgICAgIG1TdHJpbmcgPSBzdHIgPSBlZGl0OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZWRpdFtpXSA9IHRvbG93ZXIoKGNoYXIpdik7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi1zdGF0dXNfdCBTdHJpbmcxNjo6cmVwbGFjZUFsbChjaGFyMTZfdCByZXBsYWNlVGhpcywgY2hhcjE2X3Qgd2l0aFRoaXMpCi17Ci0gICAgY29uc3Qgc2l6ZV90IE4gPSBzaXplKCk7Ci0gICAgY29uc3QgY2hhcjE2X3QqIHN0ciA9IHN0cmluZygpOwotICAgIGNoYXIxNl90KiBlZGl0ID0gTlVMTDsKLSAgICBmb3IgKHNpemVfdCBpPTA7IGk8TjsgaSsrKSB7Ci0gICAgICAgIGlmIChzdHJbaV0gPT0gcmVwbGFjZVRoaXMpIHsKLSAgICAgICAgICAgIGlmICghZWRpdCkgewotICAgICAgICAgICAgICAgIFNoYXJlZEJ1ZmZlciogYnVmID0gU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKS0+ZWRpdCgpOwotICAgICAgICAgICAgICAgIGlmICghYnVmKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGVkaXQgPSAoY2hhcjE2X3QqKWJ1Zi0+ZGF0YSgpOwotICAgICAgICAgICAgICAgIG1TdHJpbmcgPSBzdHIgPSBlZGl0OwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZWRpdFtpXSA9IHdpdGhUaGlzOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgU3RyaW5nMTY6OnJlbW92ZShzaXplX3QgbGVuLCBzaXplX3QgYmVnaW4pCi17Ci0gICAgY29uc3Qgc2l6ZV90IE4gPSBzaXplKCk7Ci0gICAgaWYgKGJlZ2luID49IE4pIHsKLSAgICAgICAgU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKS0+cmVsZWFzZSgpOwotICAgICAgICBtU3RyaW5nID0gZ2V0RW1wdHlTdHJpbmcoKTsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLSAgICBpZiAoKGJlZ2luK2xlbikgPiBOKSBsZW4gPSBOLWJlZ2luOwotICAgIGlmIChiZWdpbiA9PSAwICYmIGxlbiA9PSBOKSB7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0KLSAgICBpZiAoYmVnaW4gPiAwKSB7Ci0gICAgICAgIFNoYXJlZEJ1ZmZlciogYnVmID0gU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKQotICAgICAgICAgICAgLT5lZGl0UmVzaXplKChOKzEpKnNpemVvZihjaGFyMTZfdCkpOwotICAgICAgICBpZiAoIWJ1ZikgewotICAgICAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLSAgICAgICAgfQotICAgICAgICBjaGFyMTZfdCogc3RyID0gKGNoYXIxNl90KilidWYtPmRhdGEoKTsKLSAgICAgICAgbWVtbW92ZShzdHIsIHN0citiZWdpbiwgKE4tYmVnaW4rMSkqc2l6ZW9mKGNoYXIxNl90KSk7Ci0gICAgICAgIG1TdHJpbmcgPSBzdHI7Ci0gICAgfQotICAgIFNoYXJlZEJ1ZmZlciogYnVmID0gU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKQotICAgICAgICAtPmVkaXRSZXNpemUoKGxlbisxKSpzaXplb2YoY2hhcjE2X3QpKTsKLSAgICBpZiAoYnVmKSB7Ci0gICAgICAgIGNoYXIxNl90KiBzdHIgPSAoY2hhcjE2X3QqKWJ1Zi0+ZGF0YSgpOwotICAgICAgICBzdHJbbGVuXSA9IDA7Ci0gICAgICAgIG1TdHJpbmcgPSBzdHI7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX01FTU9SWTsKLX0KLQotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgY29uc3QgU3RyaW5nMTYmIHZhbCkKLXsKLSAgICB0byA8PCBTdHJpbmc4KHZhbCkuc3RyaW5nKCk7Ci0gICAgcmV0dXJuIHRvOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9TdHJpbmc4LmNwcCBiL2xpYnMvdXRpbHMvU3RyaW5nOC5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM1MGQzNDMuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9TdHJpbmc4LmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDYwNCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzguaD4KLQotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotI2luY2x1ZGUgPHV0aWxzL1N0cmluZzE2Lmg+Ci0jaW5jbHVkZSA8dXRpbHMvVGV4dE91dHB1dC5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLQotI2luY2x1ZGUgPHByaXZhdGUvdXRpbHMvU3RhdGljLmg+Ci0KLSNpbmNsdWRlIDxjdHlwZS5oPgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgY29uc3QgdWludDMyX3Qga0J5dGVNYXNrID0gMHgwMDAwMDBCRjsKLXN0YXRpYyBjb25zdCB1aW50MzJfdCBrQnl0ZU1hcmsgPSAweDAwMDAwMDgwOwotCi0vLyBTdXJyb2dhdGVzIGFyZW4ndCB2YWxpZCBmb3IgVVRGLTMyIGNoYXJhY3RlcnMsIHNvIGRlZmluZSBzb21lCi0vLyBjb25zdGFudHMgdGhhdCB3aWxsIGxldCB1cyBzY3JlZW4gdGhlbSBvdXQuCi1zdGF0aWMgY29uc3QgdWludDMyX3Qga1VuaWNvZGVTdXJyb2dhdGVIaWdoU3RhcnQgID0gMHgwMDAwRDgwMDsKLXN0YXRpYyBjb25zdCB1aW50MzJfdCBrVW5pY29kZVN1cnJvZ2F0ZUhpZ2hFbmQgICAgPSAweDAwMDBEQkZGOwotc3RhdGljIGNvbnN0IHVpbnQzMl90IGtVbmljb2RlU3Vycm9nYXRlTG93U3RhcnQgICA9IDB4MDAwMERDMDA7Ci1zdGF0aWMgY29uc3QgdWludDMyX3Qga1VuaWNvZGVTdXJyb2dhdGVMb3dFbmQgICAgID0gMHgwMDAwREZGRjsKLXN0YXRpYyBjb25zdCB1aW50MzJfdCBrVW5pY29kZVN1cnJvZ2F0ZVN0YXJ0ICAgICAgPSBrVW5pY29kZVN1cnJvZ2F0ZUhpZ2hTdGFydDsKLXN0YXRpYyBjb25zdCB1aW50MzJfdCBrVW5pY29kZVN1cnJvZ2F0ZUVuZCAgICAgICAgPSBrVW5pY29kZVN1cnJvZ2F0ZUxvd0VuZDsKLQotLy8gTWFzayB1c2VkIHRvIHNldCBhcHByb3ByaWF0ZSBiaXRzIGluIGZpcnN0IGJ5dGUgb2YgVVRGLTggc2VxdWVuY2UsCi0vLyBpbmRleGVkIGJ5IG51bWJlciBvZiBieXRlcyBpbiB0aGUgc2VxdWVuY2UuCi1zdGF0aWMgY29uc3QgdWludDMyX3Qga0ZpcnN0Qnl0ZU1hcmtbXSA9IHsKLSAgICAweDAwMDAwMDAwLCAweDAwMDAwMDAwLCAweDAwMDAwMEMwLCAweDAwMDAwMEUwLCAweDAwMDAwMEYwCi19OwotCi0vLyBTZXBhcmF0b3IgdXNlZCBieSByZXNvdXJjZSBwYXRocy4gVGhpcyBpcyBub3QgcGxhdGZvcm0gZGVwZW5kZW50IGNvbnRyYXJ5Ci0vLyB0byBPU19QQVRIX1NFUEFSQVRPUi4KLSNkZWZpbmUgUkVTX1BBVEhfU0VQQVJBVE9SICcvJwotCi0vLyBSZXR1cm4gbnVtYmVyIG9mIHV0ZjggYnl0ZXMgcmVxdWlyZWQgZm9yIHRoZSBjaGFyYWN0ZXIuCi1zdGF0aWMgc2l6ZV90IHV0ZjMyX3RvX3V0ZjhfYnl0ZXModWludDMyX3Qgc3JjQ2hhcikKLXsKLSAgICBzaXplX3QgYnl0ZXNUb1dyaXRlOwotCi0gICAgLy8gRmlndXJlIG91dCBob3cgbWFueSBieXRlcyB0aGUgcmVzdWx0IHdpbGwgcmVxdWlyZS4KLSAgICBpZiAoc3JjQ2hhciA8IDB4MDAwMDAwODApCi0gICAgewotICAgICAgICBieXRlc1RvV3JpdGUgPSAxOwotICAgIH0KLSAgICBlbHNlIGlmIChzcmNDaGFyIDwgMHgwMDAwMDgwMCkKLSAgICB7Ci0gICAgICAgIGJ5dGVzVG9Xcml0ZSA9IDI7Ci0gICAgfQotICAgIGVsc2UgaWYgKHNyY0NoYXIgPCAweDAwMDEwMDAwKQotICAgIHsKLSAgICAgICAgaWYgKChzcmNDaGFyIDwga1VuaWNvZGVTdXJyb2dhdGVTdGFydCkKLSAgICAgICAgIHx8IChzcmNDaGFyID4ga1VuaWNvZGVTdXJyb2dhdGVFbmQpKQotICAgICAgICB7Ci0gICAgICAgICAgICBieXRlc1RvV3JpdGUgPSAzOwotICAgICAgICB9Ci0gICAgICAgIGVsc2UKLSAgICAgICAgewotICAgICAgICAgICAgLy8gU3Vycm9nYXRlcyBhcmUgaW52YWxpZCBVVEYtMzIgY2hhcmFjdGVycy4KLSAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICB9Ci0gICAgfQotICAgIC8vIE1heCBjb2RlIHBvaW50IGZvciBVbmljb2RlIGlzIDB4MDAxMEZGRkYuCi0gICAgZWxzZSBpZiAoc3JjQ2hhciA8IDB4MDAxMTAwMDApCi0gICAgewotICAgICAgICBieXRlc1RvV3JpdGUgPSA0OwotICAgIH0KLSAgICBlbHNlCi0gICAgewotICAgICAgICAvLyBJbnZhbGlkIFVURi0zMiBjaGFyYWN0ZXIuCi0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIHJldHVybiBieXRlc1RvV3JpdGU7Ci19Ci0KLS8vIFdyaXRlIG91dCB0aGUgc291cmNlIGNoYXJhY3RlciB0byA8ZHN0UD4uCi0KLXN0YXRpYyB2b2lkIHV0ZjMyX3RvX3V0ZjgodWludDhfdCogZHN0UCwgdWludDMyX3Qgc3JjQ2hhciwgc2l6ZV90IGJ5dGVzKQotewotICAgIGRzdFAgKz0gYnl0ZXM7Ci0gICAgc3dpdGNoIChieXRlcykKLSAgICB7ICAgLyogbm90ZTogZXZlcnl0aGluZyBmYWxscyB0aHJvdWdoLiAqLwotICAgICAgICBjYXNlIDQ6ICotLWRzdFAgPSAodWludDhfdCkoKHNyY0NoYXIgfCBrQnl0ZU1hcmspICYga0J5dGVNYXNrKTsgc3JjQ2hhciA+Pj0gNjsKLSAgICAgICAgY2FzZSAzOiAqLS1kc3RQID0gKHVpbnQ4X3QpKChzcmNDaGFyIHwga0J5dGVNYXJrKSAmIGtCeXRlTWFzayk7IHNyY0NoYXIgPj49IDY7Ci0gICAgICAgIGNhc2UgMjogKi0tZHN0UCA9ICh1aW50OF90KSgoc3JjQ2hhciB8IGtCeXRlTWFyaykgJiBrQnl0ZU1hc2spOyBzcmNDaGFyID4+PSA2OwotICAgICAgICBjYXNlIDE6ICotLWRzdFAgPSAodWludDhfdCkoc3JjQ2hhciB8IGtGaXJzdEJ5dGVNYXJrW2J5dGVzXSk7Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RhdGljIFNoYXJlZEJ1ZmZlciogZ0VtcHR5U3RyaW5nQnVmID0gTlVMTDsKLXN0YXRpYyBjaGFyKiBnRW1wdHlTdHJpbmcgPSBOVUxMOwotCi1leHRlcm4gaW50IGdEYXJ3aW5DYW50TG9hZEFsbE9iamVjdHM7Ci1pbnQgZ0RhcndpbklzUmVhbGx5QW5ub3lpbmc7Ci0KLXN0YXRpYyBpbmxpbmUgY2hhciogZ2V0RW1wdHlTdHJpbmcoKQotewotICAgIGdFbXB0eVN0cmluZ0J1Zi0+YWNxdWlyZSgpOwotICAgIHJldHVybiBnRW1wdHlTdHJpbmc7Ci19Ci0KLXZvaWQgaW5pdGlhbGl6ZV9zdHJpbmc4KCkKLXsKLSNpZmRlZiBMSUJVVElMU19OQVRJVkUKLQkgIC8vIEJpdGUgbWUsIERhcndpbiEKLQkJZ0RhcndpbklzUmVhbGx5QW5ub3lpbmcgPSBnRGFyd2luQ2FudExvYWRBbGxPYmplY3RzOwotI2VuZGlmCi0JCQkKLSAgICBTaGFyZWRCdWZmZXIqIGJ1ZiA9IFNoYXJlZEJ1ZmZlcjo6YWxsb2MoMSk7Ci0gICAgY2hhciogc3RyID0gKGNoYXIqKWJ1Zi0+ZGF0YSgpOwotICAgICpzdHIgPSAwOwotICAgIGdFbXB0eVN0cmluZ0J1ZiA9IGJ1ZjsKLSAgICBnRW1wdHlTdHJpbmcgPSBzdHI7Ci19Ci0KLXZvaWQgdGVybWluYXRlX3N0cmluZzgoKQotewotICAgIFNoYXJlZEJ1ZmZlcjo6YnVmZmVyRnJvbURhdGEoZ0VtcHR5U3RyaW5nKS0+cmVsZWFzZSgpOwotICAgIGdFbXB0eVN0cmluZ0J1ZiA9IE5VTEw7Ci0gICAgZ0VtcHR5U3RyaW5nID0gTlVMTDsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyBjaGFyKiBhbGxvY0Zyb21VVEY4KGNvbnN0IGNoYXIqIGluLCBzaXplX3QgbGVuKQotewotICAgIGlmIChsZW4gPiAwKSB7Ci0gICAgICAgIFNoYXJlZEJ1ZmZlciogYnVmID0gU2hhcmVkQnVmZmVyOjphbGxvYyhsZW4rMSk7Ci0gICAgICAgIExPR19BU1NFUlQoYnVmLCAiVW5hYmxlIHRvIGFsbG9jYXRlIHNoYXJlZCBidWZmZXIiKTsKLSAgICAgICAgaWYgKGJ1ZikgewotICAgICAgICAgICAgY2hhciogc3RyID0gKGNoYXIqKWJ1Zi0+ZGF0YSgpOwotICAgICAgICAgICAgbWVtY3B5KHN0ciwgaW4sIGxlbik7Ci0gICAgICAgICAgICBzdHJbbGVuXSA9IDA7Ci0gICAgICAgICAgICByZXR1cm4gc3RyOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBOVUxMOwotICAgIH0KLQotICAgIHJldHVybiBnZXRFbXB0eVN0cmluZygpOwotfQotCi0vLyBOb3RlOiBub3QgZGVhbGluZyB3aXRoIGV4cGFuZGluZyBzdXJyb2dhdGUgcGFpcnMuCi1zdGF0aWMgY2hhciogYWxsb2NGcm9tVVRGMTYoY29uc3QgY2hhcjE2X3QqIGluLCBzaXplX3QgbGVuKQotewotICAgIGlmIChsZW4gPT0gMCkgcmV0dXJuIGdldEVtcHR5U3RyaW5nKCk7Ci0gICAgCi0gICAgc2l6ZV90IGJ5dGVzID0gMDsKLSAgICBjb25zdCBjaGFyMTZfdCogZW5kID0gaW4rbGVuOwotICAgIGNvbnN0IGNoYXIxNl90KiBwID0gaW47Ci0gICAgCi0gICAgd2hpbGUgKHAgPCBlbmQpIHsKLSAgICAgICAgYnl0ZXMgKz0gdXRmMzJfdG9fdXRmOF9ieXRlcygqcCk7Ci0gICAgICAgIHArKzsKLSAgICB9Ci0gICAgCi0gICAgU2hhcmVkQnVmZmVyKiBidWYgPSBTaGFyZWRCdWZmZXI6OmFsbG9jKGJ5dGVzKzEpOwotICAgIExPR19BU1NFUlQoYnVmLCAiVW5hYmxlIHRvIGFsbG9jYXRlIHNoYXJlZCBidWZmZXIiKTsKLSAgICBpZiAoYnVmKSB7Ci0gICAgICAgIHAgPSBpbjsKLSAgICAgICAgY2hhciogc3RyID0gKGNoYXIqKWJ1Zi0+ZGF0YSgpOwotICAgICAgICBjaGFyKiBkID0gc3RyOwotICAgICAgICB3aGlsZSAocCA8IGVuZCkgewotICAgICAgICAgICAgdWludDMyX3QgYyA9ICpwKys7Ci0gICAgICAgICAgICBzaXplX3QgbGVuID0gdXRmMzJfdG9fdXRmOF9ieXRlcyhjKTsKLSAgICAgICAgICAgIHV0ZjMyX3RvX3V0ZjgoKHVpbnQ4X3QqKWQsIGMsIGxlbik7Ci0gICAgICAgICAgICBkICs9IGxlbjsKLSAgICAgICAgfQotICAgICAgICAqZCA9IDA7Ci0gICAgICAgIAotICAgICAgICByZXR1cm4gc3RyOwotICAgIH0KLSAgICAKLSAgICByZXR1cm4gZ2V0RW1wdHlTdHJpbmcoKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLVN0cmluZzg6OlN0cmluZzgoKQotICAgIDogbVN0cmluZyhnZXRFbXB0eVN0cmluZygpKQotewotfQotCi1TdHJpbmc4OjpTdHJpbmc4KGNvbnN0IFN0cmluZzgmIG8pCi0gICAgOiBtU3RyaW5nKG8ubVN0cmluZykKLXsKLSAgICBTaGFyZWRCdWZmZXI6OmJ1ZmZlckZyb21EYXRhKG1TdHJpbmcpLT5hY3F1aXJlKCk7Ci19Ci0KLVN0cmluZzg6OlN0cmluZzgoY29uc3QgY2hhciogbykKLSAgICA6IG1TdHJpbmcoYWxsb2NGcm9tVVRGOChvLCBzdHJsZW4obykpKQotewotICAgIGlmIChtU3RyaW5nID09IE5VTEwpIHsKLSAgICAgICAgbVN0cmluZyA9IGdldEVtcHR5U3RyaW5nKCk7Ci0gICAgfQotfQotCi1TdHJpbmc4OjpTdHJpbmc4KGNvbnN0IGNoYXIqIG8sIHNpemVfdCBsZW4pCi0gICAgOiBtU3RyaW5nKGFsbG9jRnJvbVVURjgobywgbGVuKSkKLXsKLSAgICBpZiAobVN0cmluZyA9PSBOVUxMKSB7Ci0gICAgICAgIG1TdHJpbmcgPSBnZXRFbXB0eVN0cmluZygpOwotICAgIH0KLX0KLQotU3RyaW5nODo6U3RyaW5nOChjb25zdCBTdHJpbmcxNiYgbykKLSAgICA6IG1TdHJpbmcoYWxsb2NGcm9tVVRGMTYoby5zdHJpbmcoKSwgby5zaXplKCkpKQotewotfQotCi1TdHJpbmc4OjpTdHJpbmc4KGNvbnN0IGNoYXIxNl90KiBvKQotICAgIDogbVN0cmluZyhhbGxvY0Zyb21VVEYxNihvLCBzdHJsZW4xNihvKSkpCi17Ci19Ci0KLVN0cmluZzg6OlN0cmluZzgoY29uc3QgY2hhcjE2X3QqIG8sIHNpemVfdCBsZW4pCi0gICAgOiBtU3RyaW5nKGFsbG9jRnJvbVVURjE2KG8sIGxlbikpCi17Ci19Ci0KLVN0cmluZzg6On5TdHJpbmc4KCkKLXsKLSAgICBTaGFyZWRCdWZmZXI6OmJ1ZmZlckZyb21EYXRhKG1TdHJpbmcpLT5yZWxlYXNlKCk7Ci19Ci0KLXZvaWQgU3RyaW5nODo6c2V0VG8oY29uc3QgU3RyaW5nOCYgb3RoZXIpCi17Ci0gICAgU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShvdGhlci5tU3RyaW5nKS0+YWNxdWlyZSgpOwotICAgIFNoYXJlZEJ1ZmZlcjo6YnVmZmVyRnJvbURhdGEobVN0cmluZyktPnJlbGVhc2UoKTsKLSAgICBtU3RyaW5nID0gb3RoZXIubVN0cmluZzsKLX0KLQotc3RhdHVzX3QgU3RyaW5nODo6c2V0VG8oY29uc3QgY2hhciogb3RoZXIpCi17Ci0gICAgU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKS0+cmVsZWFzZSgpOwotICAgIG1TdHJpbmcgPSBhbGxvY0Zyb21VVEY4KG90aGVyLCBzdHJsZW4ob3RoZXIpKTsKLSAgICBpZiAobVN0cmluZykgcmV0dXJuIE5PX0VSUk9SOwotCi0gICAgbVN0cmluZyA9IGdldEVtcHR5U3RyaW5nKCk7Ci0gICAgcmV0dXJuIE5PX01FTU9SWTsKLX0KLQotc3RhdHVzX3QgU3RyaW5nODo6c2V0VG8oY29uc3QgY2hhciogb3RoZXIsIHNpemVfdCBsZW4pCi17Ci0gICAgU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKS0+cmVsZWFzZSgpOwotICAgIG1TdHJpbmcgPSBhbGxvY0Zyb21VVEY4KG90aGVyLCBsZW4pOwotICAgIGlmIChtU3RyaW5nKSByZXR1cm4gTk9fRVJST1I7Ci0KLSAgICBtU3RyaW5nID0gZ2V0RW1wdHlTdHJpbmcoKTsKLSAgICByZXR1cm4gTk9fTUVNT1JZOwotfQotCi1zdGF0dXNfdCBTdHJpbmc4OjpzZXRUbyhjb25zdCBjaGFyMTZfdCogb3RoZXIsIHNpemVfdCBsZW4pCi17Ci0gICAgU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKS0+cmVsZWFzZSgpOwotICAgIG1TdHJpbmcgPSBhbGxvY0Zyb21VVEYxNihvdGhlciwgbGVuKTsKLSAgICBpZiAobVN0cmluZykgcmV0dXJuIE5PX0VSUk9SOwotCi0gICAgbVN0cmluZyA9IGdldEVtcHR5U3RyaW5nKCk7Ci0gICAgcmV0dXJuIE5PX01FTU9SWTsKLX0KLQotc3RhdHVzX3QgU3RyaW5nODo6YXBwZW5kKGNvbnN0IFN0cmluZzgmIG90aGVyKQotewotICAgIGNvbnN0IHNpemVfdCBvdGhlckxlbiA9IG90aGVyLmJ5dGVzKCk7Ci0gICAgaWYgKGJ5dGVzKCkgPT0gMCkgewotICAgICAgICBzZXRUbyhvdGhlcik7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9IGVsc2UgaWYgKG90aGVyTGVuID09IDApIHsKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotICAgIH0KLQotICAgIHJldHVybiByZWFsX2FwcGVuZChvdGhlci5zdHJpbmcoKSwgb3RoZXJMZW4pOwotfQotCi1zdGF0dXNfdCBTdHJpbmc4OjphcHBlbmQoY29uc3QgY2hhciogb3RoZXIpCi17Ci0gICAgcmV0dXJuIGFwcGVuZChvdGhlciwgc3RybGVuKG90aGVyKSk7Ci19Ci0KLXN0YXR1c190IFN0cmluZzg6OmFwcGVuZChjb25zdCBjaGFyKiBvdGhlciwgc2l6ZV90IG90aGVyTGVuKQotewotICAgIGlmIChieXRlcygpID09IDApIHsKLSAgICAgICAgcmV0dXJuIHNldFRvKG90aGVyLCBvdGhlckxlbik7Ci0gICAgfSBlbHNlIGlmIChvdGhlckxlbiA9PSAwKSB7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0KLSAgICByZXR1cm4gcmVhbF9hcHBlbmQob3RoZXIsIG90aGVyTGVuKTsKLX0KLQotc3RhdHVzX3QgU3RyaW5nODo6cmVhbF9hcHBlbmQoY29uc3QgY2hhciogb3RoZXIsIHNpemVfdCBvdGhlckxlbikKLXsKLSAgICBjb25zdCBzaXplX3QgbXlMZW4gPSBieXRlcygpOwotICAgIAotICAgIFNoYXJlZEJ1ZmZlciogYnVmID0gU2hhcmVkQnVmZmVyOjpidWZmZXJGcm9tRGF0YShtU3RyaW5nKQotICAgICAgICAtPmVkaXRSZXNpemUobXlMZW4rb3RoZXJMZW4rMSk7Ci0gICAgaWYgKGJ1ZikgewotICAgICAgICBjaGFyKiBzdHIgPSAoY2hhciopYnVmLT5kYXRhKCk7Ci0gICAgICAgIG1TdHJpbmcgPSBzdHI7Ci0gICAgICAgIHN0ciArPSBteUxlbjsKLSAgICAgICAgbWVtY3B5KHN0ciwgb3RoZXIsIG90aGVyTGVuKTsKLSAgICAgICAgc3RyW290aGVyTGVuXSA9ICdcMCc7Ci0gICAgICAgIHJldHVybiBOT19FUlJPUjsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX01FTU9SWTsKLX0KLQotY2hhciogU3RyaW5nODo6bG9ja0J1ZmZlcihzaXplX3Qgc2l6ZSkKLXsKLSAgICBTaGFyZWRCdWZmZXIqIGJ1ZiA9IFNoYXJlZEJ1ZmZlcjo6YnVmZmVyRnJvbURhdGEobVN0cmluZykKLSAgICAgICAgLT5lZGl0UmVzaXplKHNpemUrMSk7Ci0gICAgaWYgKGJ1ZikgewotICAgICAgICBjaGFyKiBzdHIgPSAoY2hhciopYnVmLT5kYXRhKCk7Ci0gICAgICAgIG1TdHJpbmcgPSBzdHI7Ci0gICAgICAgIHJldHVybiBzdHI7Ci0gICAgfQotICAgIHJldHVybiBOVUxMOwotfQotCi12b2lkIFN0cmluZzg6OnVubG9ja0J1ZmZlcigpCi17Ci0gICAgdW5sb2NrQnVmZmVyKHN0cmxlbihtU3RyaW5nKSk7Ci19Ci0KLXN0YXR1c190IFN0cmluZzg6OnVubG9ja0J1ZmZlcihzaXplX3Qgc2l6ZSkKLXsKLSAgICBpZiAoc2l6ZSAhPSB0aGlzLT5zaXplKCkpIHsKLSAgICAgICAgU2hhcmVkQnVmZmVyKiBidWYgPSBTaGFyZWRCdWZmZXI6OmJ1ZmZlckZyb21EYXRhKG1TdHJpbmcpCi0gICAgICAgICAgICAtPmVkaXRSZXNpemUoc2l6ZSsxKTsKLSAgICAgICAgaWYgKGJ1ZikgewotICAgICAgICAgICAgY2hhciogc3RyID0gKGNoYXIqKWJ1Zi0+ZGF0YSgpOwotICAgICAgICAgICAgc3RyW3NpemVdID0gMDsKLSAgICAgICAgICAgIG1TdHJpbmcgPSBzdHI7Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgcmV0dXJuIE5PX01FTU9SWTsKLX0KLQotc3NpemVfdCBTdHJpbmc4OjpmaW5kKGNvbnN0IGNoYXIqIG90aGVyLCBzaXplX3Qgc3RhcnQpIGNvbnN0Ci17Ci0gICAgc2l6ZV90IGxlbiA9IHNpemUoKTsKLSAgICBpZiAoc3RhcnQgPj0gbGVuKSB7Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0gICAgY29uc3QgY2hhciogcyA9IG1TdHJpbmcrc3RhcnQ7Ci0gICAgY29uc3QgY2hhciogcCA9IHN0cnN0cihzLCBvdGhlcik7Ci0gICAgcmV0dXJuIHAgPyBwLW1TdHJpbmcgOiAtMTsKLX0KLQotdm9pZCBTdHJpbmc4Ojp0b0xvd2VyKCkKLXsKLSAgICB0b0xvd2VyKDAsIHNpemUoKSk7Ci19Ci0KLXZvaWQgU3RyaW5nODo6dG9Mb3dlcihzaXplX3Qgc3RhcnQsIHNpemVfdCBsZW5ndGgpCi17Ci0gICAgY29uc3Qgc2l6ZV90IGxlbiA9IHNpemUoKTsKLSAgICBpZiAoc3RhcnQgPj0gbGVuKSB7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKHN0YXJ0K2xlbmd0aCA+IGxlbikgewotICAgICAgICBsZW5ndGggPSBsZW4tc3RhcnQ7Ci0gICAgfQotICAgIGNoYXIqIGJ1ZiA9IGxvY2tCdWZmZXIobGVuKTsKLSAgICBidWYgKz0gc3RhcnQ7Ci0gICAgd2hpbGUgKGxlbmd0aCA+IDApIHsKLSAgICAgICAgKmJ1ZiA9IHRvbG93ZXIoKmJ1Zik7Ci0gICAgICAgIGJ1ZisrOwotICAgICAgICBsZW5ndGgtLTsKLSAgICB9Ci0gICAgdW5sb2NrQnVmZmVyKGxlbik7Ci19Ci0KLXZvaWQgU3RyaW5nODo6dG9VcHBlcigpCi17Ci0gICAgdG9VcHBlcigwLCBzaXplKCkpOwotfQotCi12b2lkIFN0cmluZzg6OnRvVXBwZXIoc2l6ZV90IHN0YXJ0LCBzaXplX3QgbGVuZ3RoKQotewotICAgIGNvbnN0IHNpemVfdCBsZW4gPSBzaXplKCk7Ci0gICAgaWYgKHN0YXJ0ID49IGxlbikgewotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmIChzdGFydCtsZW5ndGggPiBsZW4pIHsKLSAgICAgICAgbGVuZ3RoID0gbGVuLXN0YXJ0OwotICAgIH0KLSAgICBjaGFyKiBidWYgPSBsb2NrQnVmZmVyKGxlbik7Ci0gICAgYnVmICs9IHN0YXJ0OwotICAgIHdoaWxlIChsZW5ndGggPiAwKSB7Ci0gICAgICAgICpidWYgPSB0b3VwcGVyKCpidWYpOwotICAgICAgICBidWYrKzsKLSAgICAgICAgbGVuZ3RoLS07Ci0gICAgfQotICAgIHVubG9ja0J1ZmZlcihsZW4pOwotfQotCi1UZXh0T3V0cHV0JiBvcGVyYXRvcjw8KFRleHRPdXRwdXQmIHRvLCBjb25zdCBTdHJpbmc4JiB2YWwpCi17Ci0gICAgdG8gPDwgdmFsLnN0cmluZygpOwotICAgIHJldHVybiB0bzsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBQYXRoIGZ1bmN0aW9ucwotCi0KLXZvaWQgU3RyaW5nODo6c2V0UGF0aE5hbWUoY29uc3QgY2hhciogbmFtZSkKLXsKLSAgICBzZXRQYXRoTmFtZShuYW1lLCBzdHJsZW4obmFtZSkpOwotfQotCi12b2lkIFN0cmluZzg6OnNldFBhdGhOYW1lKGNvbnN0IGNoYXIqIG5hbWUsIHNpemVfdCBsZW4pCi17Ci0gICAgY2hhciogYnVmID0gbG9ja0J1ZmZlcihsZW4pOwotCi0gICAgbWVtY3B5KGJ1ZiwgbmFtZSwgbGVuKTsKLQotICAgIC8vIHJlbW92ZSB0cmFpbGluZyBwYXRoIHNlcGFyYXRvciwgaWYgcHJlc2VudAotICAgIGlmIChsZW4gPiAwICYmIGJ1ZltsZW4tMV0gPT0gT1NfUEFUSF9TRVBBUkFUT1IpCi0gICAgICAgIGxlbi0tOwotCi0gICAgYnVmW2xlbl0gPSAnXDAnOwotCi0gICAgdW5sb2NrQnVmZmVyKGxlbik7Ci19Ci0KLVN0cmluZzggU3RyaW5nODo6Z2V0UGF0aExlYWYodm9pZCkgY29uc3QKLXsKLSAgICBjb25zdCBjaGFyKiBjcDsKLSAgICBjb25zdCBjaGFyKmNvbnN0IGJ1ZiA9IG1TdHJpbmc7Ci0KLSAgICBjcCA9IHN0cnJjaHIoYnVmLCBPU19QQVRIX1NFUEFSQVRPUik7Ci0gICAgaWYgKGNwID09IE5VTEwpCi0gICAgICAgIHJldHVybiBTdHJpbmc4KCp0aGlzKTsKLSAgICBlbHNlCi0gICAgICAgIHJldHVybiBTdHJpbmc4KGNwKzEpOwotfQotCi1TdHJpbmc4IFN0cmluZzg6OmdldFBhdGhEaXIodm9pZCkgY29uc3QKLXsKLSAgICBjb25zdCBjaGFyKiBjcDsKLSAgICBjb25zdCBjaGFyKmNvbnN0IHN0ciA9IG1TdHJpbmc7Ci0KLSAgICBjcCA9IHN0cnJjaHIoc3RyLCBPU19QQVRIX1NFUEFSQVRPUik7Ci0gICAgaWYgKGNwID09IE5VTEwpCi0gICAgICAgIHJldHVybiBTdHJpbmc4KCIiKTsKLSAgICBlbHNlCi0gICAgICAgIHJldHVybiBTdHJpbmc4KHN0ciwgY3AgLSBzdHIpOwotfQotCi1TdHJpbmc4IFN0cmluZzg6OndhbGtQYXRoKFN0cmluZzgqIG91dFJlbWFpbnMpIGNvbnN0Ci17Ci0gICAgY29uc3QgY2hhciogY3A7Ci0gICAgY29uc3QgY2hhcipjb25zdCBzdHIgPSBtU3RyaW5nOwotICAgIGNvbnN0IGNoYXIqIGJ1ZiA9IHN0cjsKLQotICAgIGNwID0gc3RyY2hyKGJ1ZiwgT1NfUEFUSF9TRVBBUkFUT1IpOwotICAgIGlmIChjcCA9PSBidWYpIHsKLSAgICAgICAgLy8gZG9uJ3QgaW5jbHVkZSBhIGxlYWRpbmcgJy8nLgotICAgICAgICBidWYgPSBidWYrMTsKLSAgICAgICAgY3AgPSBzdHJjaHIoYnVmLCBPU19QQVRIX1NFUEFSQVRPUik7Ci0gICAgfQotCi0gICAgaWYgKGNwID09IE5VTEwpIHsKLSAgICAgICAgU3RyaW5nOCByZXMgPSBidWYgIT0gc3RyID8gU3RyaW5nOChidWYpIDogKnRoaXM7Ci0gICAgICAgIGlmIChvdXRSZW1haW5zKSAqb3V0UmVtYWlucyA9IFN0cmluZzgoIiIpOwotICAgICAgICByZXR1cm4gcmVzOwotICAgIH0KLQotICAgIFN0cmluZzggcmVzKGJ1ZiwgY3AtYnVmKTsKLSAgICBpZiAob3V0UmVtYWlucykgKm91dFJlbWFpbnMgPSBTdHJpbmc4KGNwKzEpOwotICAgIHJldHVybiByZXM7Ci19Ci0KLS8qCi0gKiBIZWxwZXIgZnVuY3Rpb24gZm9yIGZpbmRpbmcgdGhlIHN0YXJ0IG9mIGFuIGV4dGVuc2lvbiBpbiBhIHBhdGhuYW1lLgotICoKLSAqIFJldHVybnMgYSBwb2ludGVyIGluc2lkZSBtU3RyaW5nLCBvciBOVUxMIGlmIG5vIGV4dGVuc2lvbiB3YXMgZm91bmQuCi0gKi8KLWNoYXIqIFN0cmluZzg6OmZpbmRfZXh0ZW5zaW9uKHZvaWQpIGNvbnN0Ci17Ci0gICAgY29uc3QgY2hhciogbGFzdFNsYXNoOwotICAgIGNvbnN0IGNoYXIqIGxhc3REb3Q7Ci0gICAgaW50IGV4dExlbjsKLSAgICBjb25zdCBjaGFyKiBjb25zdCBzdHIgPSBtU3RyaW5nOwotCi0gICAgLy8gb25seSBsb29rIGF0IHRoZSBmaWxlbmFtZQotICAgIGxhc3RTbGFzaCA9IHN0cnJjaHIoc3RyLCBPU19QQVRIX1NFUEFSQVRPUik7Ci0gICAgaWYgKGxhc3RTbGFzaCA9PSBOVUxMKQotICAgICAgICBsYXN0U2xhc2ggPSBzdHI7Ci0gICAgZWxzZQotICAgICAgICBsYXN0U2xhc2grKzsKLQotICAgIC8vIGZpbmQgdGhlIGxhc3QgZG90Ci0gICAgbGFzdERvdCA9IHN0cnJjaHIobGFzdFNsYXNoLCAnLicpOwotICAgIGlmIChsYXN0RG90ID09IE5VTEwpCi0gICAgICAgIHJldHVybiBOVUxMOwotCi0gICAgLy8gbG9va3MgZ29vZCwgc2hpcCBpdAotICAgIHJldHVybiBjb25zdF9jYXN0PGNoYXIqPihsYXN0RG90KTsKLX0KLQotU3RyaW5nOCBTdHJpbmc4OjpnZXRQYXRoRXh0ZW5zaW9uKHZvaWQpIGNvbnN0Ci17Ci0gICAgY2hhciogZXh0OwotCi0gICAgZXh0ID0gZmluZF9leHRlbnNpb24oKTsKLSAgICBpZiAoZXh0ICE9IE5VTEwpCi0gICAgICAgIHJldHVybiBTdHJpbmc4KGV4dCk7Ci0gICAgZWxzZQotICAgICAgICByZXR1cm4gU3RyaW5nOCgiIik7Ci19Ci0KLVN0cmluZzggU3RyaW5nODo6Z2V0QmFzZVBhdGgodm9pZCkgY29uc3QKLXsKLSAgICBjaGFyKiBleHQ7Ci0gICAgY29uc3QgY2hhciogY29uc3Qgc3RyID0gbVN0cmluZzsKLQotICAgIGV4dCA9IGZpbmRfZXh0ZW5zaW9uKCk7Ci0gICAgaWYgKGV4dCA9PSBOVUxMKQotICAgICAgICByZXR1cm4gU3RyaW5nOCgqdGhpcyk7Ci0gICAgZWxzZQotICAgICAgICByZXR1cm4gU3RyaW5nOChzdHIsIGV4dCAtIHN0cik7Ci19Ci0KLVN0cmluZzgmIFN0cmluZzg6OmFwcGVuZFBhdGgoY29uc3QgY2hhciogbmFtZSkKLXsKLSAgICAvLyBUT0RPOiBUaGUgdGVzdCBiZWxvdyB3aWxsIGZhaWwgZm9yIFdpbjMyIHBhdGhzLiBGaXggbGF0ZXIgb3IgaWdub3JlLgotICAgIGlmIChuYW1lWzBdICE9IE9TX1BBVEhfU0VQQVJBVE9SKSB7Ci0gICAgICAgIGlmICgqbmFtZSA9PSAnXDAnKSB7Ci0gICAgICAgICAgICAvLyBub3RoaW5nIHRvIGRvCi0gICAgICAgICAgICByZXR1cm4gKnRoaXM7Ci0gICAgICAgIH0KLQotICAgICAgICBzaXplX3QgbGVuID0gbGVuZ3RoKCk7Ci0gICAgICAgIGlmIChsZW4gPT0gMCkgewotICAgICAgICAgICAgLy8gbm8gZXhpc3RpbmcgZmlsZW5hbWUsIGp1c3QgdXNlIHRoZSBuZXcgb25lCi0gICAgICAgICAgICBzZXRQYXRoTmFtZShuYW1lKTsKLSAgICAgICAgICAgIHJldHVybiAqdGhpczsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIG1ha2Ugcm9vbSBmb3Igb2xkUGF0aCArICcvJyArIG5ld1BhdGgKLSAgICAgICAgaW50IG5ld2xlbiA9IHN0cmxlbihuYW1lKTsKLQotICAgICAgICBjaGFyKiBidWYgPSBsb2NrQnVmZmVyKGxlbisxK25ld2xlbik7Ci0KLSAgICAgICAgLy8gaW5zZXJ0IGEgJy8nIGlmIG5lZWRlZAotICAgICAgICBpZiAoYnVmW2xlbi0xXSAhPSBPU19QQVRIX1NFUEFSQVRPUikKLSAgICAgICAgICAgIGJ1ZltsZW4rK10gPSBPU19QQVRIX1NFUEFSQVRPUjsKLQotICAgICAgICBtZW1jcHkoYnVmK2xlbiwgbmFtZSwgbmV3bGVuKzEpOwotICAgICAgICBsZW4gKz0gbmV3bGVuOwotCi0gICAgICAgIHVubG9ja0J1ZmZlcihsZW4pOwotCi0gICAgICAgIHJldHVybiAqdGhpczsKLSAgICB9IGVsc2UgewotICAgICAgICBzZXRQYXRoTmFtZShuYW1lKTsKLSAgICAgICAgcmV0dXJuICp0aGlzOwotICAgIH0KLX0KLQotU3RyaW5nOCYgU3RyaW5nODo6Y29udmVydFRvUmVzUGF0aCgpCi17Ci0jaWYgT1NfUEFUSF9TRVBBUkFUT1IgIT0gUkVTX1BBVEhfU0VQQVJBVE9SCi0gICAgc2l6ZV90IGxlbiA9IGxlbmd0aCgpOwotICAgIGlmIChsZW4gPiAwKSB7Ci0gICAgICAgIGNoYXIgKiBidWYgPSBsb2NrQnVmZmVyKGxlbik7Ci0gICAgICAgIGZvciAoY2hhciAqIGVuZCA9IGJ1ZiArIGxlbjsgYnVmIDwgZW5kOyArK2J1ZikgewotICAgICAgICAgICAgaWYgKCpidWYgPT0gT1NfUEFUSF9TRVBBUkFUT1IpCi0gICAgICAgICAgICAgICAgKmJ1ZiA9IFJFU19QQVRIX1NFUEFSQVRPUjsKLSAgICAgICAgfQotICAgICAgICB1bmxvY2tCdWZmZXIobGVuKTsKLSAgICB9Ci0jZW5kaWYKLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvU3lzdGVtQ2xvY2suY3BwIGIvbGlicy91dGlscy9TeXN0ZW1DbG9jay5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJiZGMwY2UuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9TeXN0ZW1DbG9jay5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxMzkgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLQotLyoKLSAqIFN5c3RlbSBjbG9jayBmdW5jdGlvbnMuCi0gKi8KLQotI2lmIEhBVkVfQU5EUk9JRF9PUwotI2luY2x1ZGUgPGxpbnV4L2lvY3RsLmg+Ci0jaW5jbHVkZSA8bGludXgvcnRjLmg+Ci0jaW5jbHVkZSA8dXRpbHMvQXRvbWljLmg+Ci0jaW5jbHVkZSA8bGludXgvYW5kcm9pZF9hbGFybS5oPgotI2VuZGlmCi0KLSNpbmNsdWRlIDxzeXMvdGltZS5oPgotI2luY2x1ZGUgPGxpbWl0cy5oPgotI2luY2x1ZGUgPGZjbnRsLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLQotI2luY2x1ZGUgPHV0aWxzL1N5c3RlbUNsb2NrLmg+Ci0jaW5jbHVkZSA8dXRpbHMvVGltZXJzLmg+Ci0KLSNkZWZpbmUgTE9HX1RBRyAiU3lzdGVtQ2xvY2siCi0jaW5jbHVkZSAidXRpbHMvTG9nLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLyoKLSAqIFNldCB0aGUgY3VycmVudCB0aW1lLiAgVGhpcyBvbmx5IHdvcmtzIHdoZW4gcnVubmluZyBhcyByb290LgotICovCi1pbnQgc2V0Q3VycmVudFRpbWVNaWxsaXMoaW50NjRfdCBtaWxsaXMpCi17Ci0jaWYgV0lOMzIKLSAgICAvLyBub3QgaW1wbGVtZW50ZWQKLSAgICByZXR1cm4gLTE7Ci0jZWxzZQotICAgIHN0cnVjdCB0aW1ldmFsIHR2OwotI2lmIEhBVkVfQU5EUk9JRF9PUwotICAgIHN0cnVjdCB0aW1lc3BlYyB0czsKLSAgICBpbnQgZmQ7Ci0gICAgaW50IHJlczsKLSNlbmRpZgotICAgIGludCByZXQgPSAwOwotCi0gICAgaWYgKG1pbGxpcyA8PSAwIHx8IG1pbGxpcyAvIDEwMDBMTCA+PSBJTlRfTUFYKSB7Ci0gICAgICAgIHJldHVybiAtMTsKLSAgICB9Ci0KLSAgICB0di50dl9zZWMgPSAodGltZV90KSAobWlsbGlzIC8gMTAwMExMKTsKLSAgICB0di50dl91c2VjID0gKHN1c2Vjb25kc190KSAoKG1pbGxpcyAlIDEwMDBMTCkgKiAxMDAwTEwpOwotCi0gICAgTE9HRCgiU2V0dGluZyB0aW1lIG9mIGRheSB0byBzZWM9JWRcbiIsIChpbnQpIHR2LnR2X3NlYyk7Ci0KLSNpZiBIQVZFX0FORFJPSURfT1MKLSAgICBmZCA9IG9wZW4oIi9kZXYvYWxhcm0iLCBPX1JEV1IpOwotICAgIGlmKGZkIDwgMCkgewotICAgICAgICBMT0dXKCJVbmFibGUgdG8gb3BlbiBhbGFybSBkcml2ZXI6ICVzXG4iLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotICAgIHRzLnR2X3NlYyA9IHR2LnR2X3NlYzsKLSAgICB0cy50dl9uc2VjID0gdHYudHZfdXNlYyAqIDEwMDA7Ci0gICAgcmVzID0gaW9jdGwoZmQsIEFORFJPSURfQUxBUk1fU0VUX1JUQywgJnRzKTsKLSAgICBpZihyZXMgPCAwKSB7Ci0gICAgICAgIExPR1coIlVuYWJsZSB0byBzZXQgcnRjIHRvICVsZDogJXNcbiIsIHR2LnR2X3NlYywgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgcmV0ID0gLTE7Ci0gICAgfQotICAgIGNsb3NlKGZkKTsKLSNlbHNlCi0gICAgaWYgKHNldHRpbWVvZmRheSgmdHYsIE5VTEwpICE9IDApIHsKLSAgICAgICAgTE9HVygiVW5hYmxlIHRvIHNldCBjbG9jayB0byAlZC4lZDogJXNcbiIsCi0gICAgICAgICAgICAoaW50KSB0di50dl9zZWMsIChpbnQpIHR2LnR2X3VzZWMsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgICAgIHJldCA9IC0xOwotICAgIH0KLSNlbmRpZgotCi0gICAgcmV0dXJuIHJldDsKLSNlbmRpZiAvLyBXSU4zMgotfQotCi0vKgotICogbmF0aXZlIHB1YmxpYyBzdGF0aWMgbG9uZyB1cHRpbWVNaWxsaXMoKTsKLSAqLwotaW50NjRfdCB1cHRpbWVNaWxsaXMoKQotewotICAgIGludDY0X3Qgd2hlbiA9IHN5c3RlbVRpbWUoU1lTVEVNX1RJTUVfTU9OT1RPTklDKTsKLSAgICByZXR1cm4gKGludDY0X3QpIG5hbm9zZWNvbmRzX3RvX21pbGxpc2Vjb25kcyh3aGVuKTsKLX0KLQotLyoKLSAqIG5hdGl2ZSBwdWJsaWMgc3RhdGljIGxvbmcgZWxhcHNlZFJlYWx0aW1lKCk7Ci0gKi8KLWludDY0X3QgZWxhcHNlZFJlYWx0aW1lKCkKLXsKLSNpZiBIQVZFX0FORFJPSURfT1MKLSAgICBzdGF0aWMgaW50IHNfZmQgPSAtMTsKLQotICAgIGlmIChzX2ZkID09IC0xKSB7Ci0gICAgICAgIGludCBmZCA9IG9wZW4oIi9kZXYvYWxhcm0iLCBPX1JET05MWSk7Ci0gICAgICAgIGlmIChhbmRyb2lkX2F0b21pY19jbXB4Y2hnKC0xLCBmZCwgJnNfZmQpKSB7Ci0gICAgICAgICAgICBjbG9zZShmZCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBzdHJ1Y3QgdGltZXNwZWMgdHM7Ci0gICAgaW50IHJlc3VsdCA9IGlvY3RsKHNfZmQsCi0gICAgICAgICAgICBBTkRST0lEX0FMQVJNX0dFVF9USU1FKEFORFJPSURfQUxBUk1fRUxBUFNFRF9SRUFMVElNRSksICZ0cyk7Ci0KLSAgICBpZiAocmVzdWx0ID09IDApIHsKLSAgICAgICAgaW50NjRfdCB3aGVuID0gc2Vjb25kc190b19uYW5vc2Vjb25kcyh0cy50dl9zZWMpICsgdHMudHZfbnNlYzsKLSAgICAgICAgcmV0dXJuIChpbnQ2NF90KSBuYW5vc2Vjb25kc190b19taWxsaXNlY29uZHMod2hlbik7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgLy8gWFhYOiB0aGVyZSB3YXMgYW4gZXJyb3IsIHByb2JhYmx5IGJlY2F1c2UgdGhlIGRyaXZlciBkaWRuJ3QKLSAgICAgICAgLy8gZXhpc3QgLi4uIHRoaXMgc2hvdWxkIHJldHVybgotICAgICAgICAvLyBhIHJlYWwgZXJyb3IsIGxpa2UgYW4gZXhjZXB0aW9uIQotICAgICAgICBpbnQ2NF90IHdoZW4gPSBzeXN0ZW1UaW1lKFNZU1RFTV9USU1FX01PTk9UT05JQyk7Ci0gICAgICAgIHJldHVybiAoaW50NjRfdCkgbmFub3NlY29uZHNfdG9fbWlsbGlzZWNvbmRzKHdoZW4pOwotICAgIH0KLSNlbHNlCi0gICAgaW50NjRfdCB3aGVuID0gc3lzdGVtVGltZShTWVNURU1fVElNRV9NT05PVE9OSUMpOwotICAgIHJldHVybiAoaW50NjRfdCkgbmFub3NlY29uZHNfdG9fbWlsbGlzZWNvbmRzKHdoZW4pOwotI2VuZGlmCi19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1RleHRPdXRwdXQuY3BwIGIvbGlicy91dGlscy9UZXh0T3V0cHV0LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggY2ViZWU5OS4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1RleHRPdXRwdXQuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTQ2ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaW5jbHVkZSA8dXRpbHMvVGV4dE91dHB1dC5oPgotCi0jaW5jbHVkZSA8dXRpbHMvRGVidWcuaD4KLQotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGJvb2wgdmFsKQotewotICAgIGlmICh2YWwpIHRvLnByaW50KCJ0cnVlIiwgNCk7Ci0gICAgZWxzZSB0by5wcmludCgiZmFsc2UiLCA1KTsKLSAgICByZXR1cm4gdG87Ci19Ci0KLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGludCB2YWwpCi17Ci0gICAgY2hhciBidWZbMTZdOwotICAgIHNwcmludGYoYnVmLCAiJWQiLCB2YWwpOwotICAgIHRvLnByaW50KGJ1Ziwgc3RybGVuKGJ1ZikpOwotICAgIHJldHVybiB0bzsKLX0KLQotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgbG9uZyB2YWwpCi17Ci0gICAgY2hhciBidWZbMTZdOwotICAgIHNwcmludGYoYnVmLCAiJWxkIiwgdmFsKTsKLSAgICB0by5wcmludChidWYsIHN0cmxlbihidWYpKTsKLSAgICByZXR1cm4gdG87Ci19Ci0KLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIHVuc2lnbmVkIGludCB2YWwpCi17Ci0gICAgY2hhciBidWZbMTZdOwotICAgIHNwcmludGYoYnVmLCAiJXUiLCB2YWwpOwotICAgIHRvLnByaW50KGJ1Ziwgc3RybGVuKGJ1ZikpOwotICAgIHJldHVybiB0bzsKLX0KLQotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgdW5zaWduZWQgbG9uZyB2YWwpCi17Ci0gICAgY2hhciBidWZbMTZdOwotICAgIHNwcmludGYoYnVmLCAiJWx1IiwgdmFsKTsKLSAgICB0by5wcmludChidWYsIHN0cmxlbihidWYpKTsKLSAgICByZXR1cm4gdG87Ci19Ci0KLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGxvbmcgbG9uZyB2YWwpCi17Ci0gICAgY2hhciBidWZbMzJdOwotICAgIHNwcmludGYoYnVmLCAiJUxkIiwgdmFsKTsKLSAgICB0by5wcmludChidWYsIHN0cmxlbihidWYpKTsKLSAgICByZXR1cm4gdG87Ci19Ci0KLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIHVuc2lnbmVkIGxvbmcgbG9uZyB2YWwpCi17Ci0gICAgY2hhciBidWZbMzJdOwotICAgIHNwcmludGYoYnVmLCAiJUx1IiwgdmFsKTsKLSAgICB0by5wcmludChidWYsIHN0cmxlbihidWYpKTsKLSAgICByZXR1cm4gdG87Ci19Ci0KLXN0YXRpYyBUZXh0T3V0cHV0JiBwcmludF9mbG9hdChUZXh0T3V0cHV0JiB0bywgZG91YmxlIHZhbHVlKQotewotICAgIGNoYXIgYnVmWzY0XTsKLSAgICBzcHJpbnRmKGJ1ZiwgIiVnIiwgdmFsdWUpOwotICAgIGlmKCAhc3RyY2hyKGJ1ZiwgJy4nKSAmJiAhc3RyY2hyKGJ1ZiwgJ2UnKSAmJgotICAgICAgICAhc3RyY2hyKGJ1ZiwgJ0UnKSApIHsKLSAgICAgICAgc3RybmNhdChidWYsICIuMCIsIHNpemVvZihidWYpLTEpOwotICAgIH0KLSAgICB0by5wcmludChidWYsIHN0cmxlbihidWYpKTsKLSAgICByZXR1cm4gdG87Ci19Ci0KLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGZsb2F0IHZhbCkKLXsKLSAgICByZXR1cm4gcHJpbnRfZmxvYXQodG8sdmFsKTsKLX0KLQotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgZG91YmxlIHZhbCkKLXsKLSAgICByZXR1cm4gcHJpbnRfZmxvYXQodG8sdmFsKTsKLX0KLQotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgY29uc3Qgdm9pZCogdmFsKQotewotICAgIGNoYXIgYnVmWzE2XTsKLSAgICBzcHJpbnRmKGJ1ZiwgIiVwIiwgdmFsKTsKLSAgICB0by5wcmludChidWYsIHN0cmxlbihidWYpKTsKLSAgICByZXR1cm4gdG87Ci19Ci0KLXN0YXRpYyB2b2lkIHRleHRPdXRwdXRQcmludGVyKHZvaWQqIGNvb2tpZSwgY29uc3QgY2hhciogdHh0KQotewotICAgICgoVGV4dE91dHB1dCopY29va2llKS0+cHJpbnQodHh0LCBzdHJsZW4odHh0KSk7Ci19Ci0KLVRleHRPdXRwdXQmIG9wZXJhdG9yPDwoVGV4dE91dHB1dCYgdG8sIGNvbnN0IFR5cGVDb2RlJiB2YWwpCi17Ci0gICAgcHJpbnRUeXBlQ29kZSh2YWwudHlwZUNvZGUoKSwgdGV4dE91dHB1dFByaW50ZXIsICh2b2lkKikmdG8pOwotICAgIHJldHVybiB0bzsKLX0KLQotSGV4RHVtcDo6SGV4RHVtcChjb25zdCB2b2lkICpidWYsIHNpemVfdCBzaXplLCBzaXplX3QgYnl0ZXNQZXJMaW5lKQotICAgIDogbUJ1ZmZlcihidWYpCi0gICAgLCBtU2l6ZShzaXplKQotICAgICwgbUJ5dGVzUGVyTGluZShieXRlc1BlckxpbmUpCi0gICAgLCBtU2luZ2xlTGluZUN1dG9mZigxNikKLSAgICAsIG1BbGlnbm1lbnQoNCkKLSAgICAsIG1DQXJyYXlTdHlsZShmYWxzZSkKLXsKLSAgICBpZiAoYnl0ZXNQZXJMaW5lID49IDE2KSBtQWxpZ25tZW50ID0gNDsKLSAgICBlbHNlIGlmIChieXRlc1BlckxpbmUgPj0gOCkgbUFsaWdubWVudCA9IDI7Ci0gICAgZWxzZSBtQWxpZ25tZW50ID0gMTsKLX0KLQotVGV4dE91dHB1dCYgb3BlcmF0b3I8PChUZXh0T3V0cHV0JiB0bywgY29uc3QgSGV4RHVtcCYgdmFsKQotewotICAgIHByaW50SGV4RGF0YSgwLCB2YWwuYnVmZmVyKCksIHZhbC5zaXplKCksIHZhbC5ieXRlc1BlckxpbmUoKSwKLSAgICAgICAgdmFsLnNpbmdsZUxpbmVDdXRvZmYoKSwgdmFsLmFsaWdubWVudCgpLCB2YWwuY2FycmF5U3R5bGUoKSwKLSAgICAgICAgdGV4dE91dHB1dFByaW50ZXIsICh2b2lkKikmdG8pOwotICAgIHJldHVybiB0bzsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvVGhyZWFkcy5jcHAgYi9saWJzL3V0aWxzL1RocmVhZHMuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3NDI3MWJhLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvVGhyZWFkcy5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxMTI2ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgImxpYnV0aWxzLnRocmVhZHMiCi0KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvTG9nLmg+Ci0KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPG1lbW9yeS5oPgotI2luY2x1ZGUgPGVycm5vLmg+Ci0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0KLSNpZiBkZWZpbmVkKEhBVkVfUFRIUkVBRFMpCi0jIGluY2x1ZGUgPHB0aHJlYWQuaD4KLSMgaW5jbHVkZSA8c2NoZWQuaD4KLSMgaW5jbHVkZSA8c3lzL3Jlc291cmNlLmg+Ci0jZWxpZiBkZWZpbmVkKEhBVkVfV0lOMzJfVEhSRUFEUykKLSMgaW5jbHVkZSA8d2luZG93cy5oPgotIyBpbmNsdWRlIDxzdGRpbnQuaD4KLSMgaW5jbHVkZSA8cHJvY2Vzcy5oPgotIyBkZWZpbmUgSEFWRV9DUkVBVEVUSFJFQUQgIC8vIEN5Z3dpbiwgdnMuIEhBVkVfX0JFR0lOVEhSRUFERVggZm9yIE1pbkdXCi0jZW5kaWYKLQotI2lmIGRlZmluZWQoSEFWRV9GVVRFWCkKLSNpbmNsdWRlIDxwcml2YXRlL3V0aWxzL2Z1dGV4X3N5bmNocm8uaD4KLSNlbmRpZgotCi0jaWYgZGVmaW5lZChIQVZFX1BSQ1RMKQotI2luY2x1ZGUgPHN5cy9wcmN0bC5oPgotI2VuZGlmCi0KLS8qCi0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqICAgICAgVGhyZWFkIHdyYXBwZXJzCi0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqLwotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIGRlZmluZWQoSEFWRV9QVEhSRUFEUykKLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIFBUSFJFQUQKLSNlbmRpZgotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0vKgotICogQ3JlYXRlIGFuZCBydW4gYSBuZXcgdGhlYWQuCi0gKgotICogV2UgY3JlYXRlIGl0ICJkZXRhY2hlZCIsIHNvIGl0IGNsZWFucyB1cCBhZnRlciBpdHNlbGYuCi0gKi8KLQotdHlwZWRlZiB2b2lkKiAoKmFuZHJvaWRfcHRocmVhZF9lbnRyeSkodm9pZCopOwotCi1zdHJ1Y3QgdGhyZWFkX2RhdGFfdCB7Ci0gICAgdGhyZWFkX2Z1bmNfdCAgIGVudHJ5RnVuY3Rpb247Ci0gICAgdm9pZCogICAgICAgICAgIHVzZXJEYXRhOwotICAgIGludCAgICAgICAgICAgICBwcmlvcml0eTsKLSAgICBjaGFyICogICAgICAgICAgdGhyZWFkTmFtZTsKLQotICAgIC8vIHdlIHVzZSB0aGlzIHRyYW1wb2xpbmUgd2hlbiB3ZSBuZWVkIHRvIHNldCB0aGUgcHJpb3JpdHkgd2l0aAotICAgIC8vIG5pY2Uvc2V0cHJpb3JpdHkuCi0gICAgc3RhdGljIGludCB0cmFtcG9saW5lKGNvbnN0IHRocmVhZF9kYXRhX3QqIHQpIHsKLSAgICAgICAgdGhyZWFkX2Z1bmNfdCBmID0gdC0+ZW50cnlGdW5jdGlvbjsKLSAgICAgICAgdm9pZCogdSA9IHQtPnVzZXJEYXRhOwotICAgICAgICBpbnQgcHJpbyA9IHQtPnByaW9yaXR5OwotICAgICAgICBjaGFyICogbmFtZSA9IHQtPnRocmVhZE5hbWU7Ci0gICAgICAgIGRlbGV0ZSB0OwotICAgICAgICBzZXRwcmlvcml0eShQUklPX1BST0NFU1MsIDAsIHByaW8pOwotICAgICAgICBpZiAobmFtZSkgewotI2lmIGRlZmluZWQoSEFWRV9QUkNUTCkKLSAgICAgICAgICAgIC8vIE1hYyBPUyBkb2Vzbid0IGhhdmUgdGhpcywgYW5kIHdlIGJ1aWxkIGxpYnV0aWwgZm9yIHRoZSBob3N0IHRvbwotICAgICAgICAgICAgaW50IGhhc0F0ID0gMDsKLSAgICAgICAgICAgIGludCBoYXNEb3QgPSAwOwotICAgICAgICAgICAgY2hhciAqcyA9IG5hbWU7Ci0gICAgICAgICAgICB3aGlsZSAoKnMpIHsKLSAgICAgICAgICAgICAgICBpZiAoKnMgPT0gJy4nKSBoYXNEb3QgPSAxOwotICAgICAgICAgICAgICAgIGVsc2UgaWYgKCpzID09ICdAJykgaGFzQXQgPSAxOwotICAgICAgICAgICAgICAgIHMrKzsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGludCBsZW4gPSBzIC0gbmFtZTsKLSAgICAgICAgICAgIGlmIChsZW4gPCAxNSB8fCBoYXNBdCB8fCAhaGFzRG90KSB7Ci0gICAgICAgICAgICAgICAgcyA9IG5hbWU7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHMgPSBuYW1lICsgbGVuIC0gMTU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBwcmN0bChQUl9TRVRfTkFNRSwgKHVuc2lnbmVkIGxvbmcpIHMsIDAsIDAsIDApOwotI2VuZGlmCi0gICAgICAgICAgICBmcmVlKG5hbWUpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmKHUpOwotICAgIH0KLX07Ci0KLWludCBhbmRyb2lkQ3JlYXRlUmF3VGhyZWFkRXRjKGFuZHJvaWRfdGhyZWFkX2Z1bmNfdCBlbnRyeUZ1bmN0aW9uLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQgKnVzZXJEYXRhLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNoYXIqIHRocmVhZE5hbWUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCB0aHJlYWRQcmlvcml0eSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplX3QgdGhyZWFkU3RhY2tTaXplLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFuZHJvaWRfdGhyZWFkX2lkX3QgKnRocmVhZElkKQotewotICAgIHB0aHJlYWRfYXR0cl90IGF0dHI7IAotICAgIHB0aHJlYWRfYXR0cl9pbml0KCZhdHRyKTsKLSAgICBwdGhyZWFkX2F0dHJfc2V0ZGV0YWNoc3RhdGUoJmF0dHIsIFBUSFJFQURfQ1JFQVRFX0RFVEFDSEVEKTsKLQotI2lmZGVmIEhBVkVfQU5EUk9JRF9PUyAgLyogdmFsZ3JpbmQgaXMgcmVqZWN0aW5nIFJULXByaW9yaXR5IGNyZWF0ZSByZXFzICovCi0gICAgaWYgKHRocmVhZFByaW9yaXR5ICE9IFBSSU9SSVRZX0RFRkFVTFQgfHwgdGhyZWFkTmFtZSAhPSBOVUxMKSB7Ci0gICAgICAgIC8vIFdlIGNvdWxkIGF2b2lkIHRoZSB0cmFtcG9saW5lIGlmIHRoZXJlIHdhcyBhIHdheSB0byBnZXQgdG8gdGhlCi0gICAgICAgIC8vIGFuZHJvaWRfdGhyZWFkX2lkX3QgKHBpZCkgZnJvbSBwdGhyZWFkX3QKLSAgICAgICAgdGhyZWFkX2RhdGFfdCogdCA9IG5ldyB0aHJlYWRfZGF0YV90OwotICAgICAgICB0LT5wcmlvcml0eSA9IHRocmVhZFByaW9yaXR5OwotICAgICAgICB0LT50aHJlYWROYW1lID0gdGhyZWFkTmFtZSA/IHN0cmR1cCh0aHJlYWROYW1lKSA6IE5VTEw7Ci0gICAgICAgIHQtPmVudHJ5RnVuY3Rpb24gPSBlbnRyeUZ1bmN0aW9uOwotICAgICAgICB0LT51c2VyRGF0YSA9IHVzZXJEYXRhOwotICAgICAgICBlbnRyeUZ1bmN0aW9uID0gKGFuZHJvaWRfdGhyZWFkX2Z1bmNfdCkmdGhyZWFkX2RhdGFfdDo6dHJhbXBvbGluZTsKLSAgICAgICAgdXNlckRhdGEgPSB0OyAgICAgICAgICAgIAotICAgIH0KLSNlbmRpZgotCi0gICAgaWYgKHRocmVhZFN0YWNrU2l6ZSkgewotICAgICAgICBwdGhyZWFkX2F0dHJfc2V0c3RhY2tzaXplKCZhdHRyLCB0aHJlYWRTdGFja1NpemUpOwotICAgIH0KLSAgICAKLSAgICBlcnJubyA9IDA7Ci0gICAgcHRocmVhZF90IHRocmVhZDsKLSAgICBpbnQgcmVzdWx0ID0gcHRocmVhZF9jcmVhdGUoJnRocmVhZCwgJmF0dHIsCi0gICAgICAgICAgICAgICAgICAgIChhbmRyb2lkX3B0aHJlYWRfZW50cnkpZW50cnlGdW5jdGlvbiwgdXNlckRhdGEpOwotICAgIGlmIChyZXN1bHQgIT0gMCkgewotICAgICAgICBMT0dFKCJhbmRyb2lkQ3JlYXRlUmF3VGhyZWFkRXRjIGZhaWxlZCAoZW50cnk9JXAsIHJlcz0lZCwgZXJybm89JWQpXG4iCi0gICAgICAgICAgICAgIihhbmRyb2lkIHRocmVhZFByaW9yaXR5PSVkKSIsCi0gICAgICAgICAgICBlbnRyeUZ1bmN0aW9uLCByZXN1bHQsIGVycm5vLCB0aHJlYWRQcmlvcml0eSk7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIGlmICh0aHJlYWRJZCAhPSBOVUxMKSB7Ci0gICAgICAgICp0aHJlYWRJZCA9IChhbmRyb2lkX3RocmVhZF9pZF90KXRocmVhZDsgLy8gWFhYOiB0aGlzIGlzIG5vdCBwb3J0YWJsZQotICAgIH0KLSAgICByZXR1cm4gMTsKLX0KLQotYW5kcm9pZF90aHJlYWRfaWRfdCBhbmRyb2lkR2V0VGhyZWFkSWQoKQotewotICAgIHJldHVybiAoYW5kcm9pZF90aHJlYWRfaWRfdClwdGhyZWFkX3NlbGYoKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2VsaWYgZGVmaW5lZChIQVZFX1dJTjMyX1RIUkVBRFMpCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBXSU4zMl9USFJFQURTCi0jZW5kaWYKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLyoKLSAqIFRyYW1wb2xpbmUgdG8gbWFrZSB1cyBfX3N0ZGNhbGwtY29tcGxpYW50LgotICoKLSAqIFdlJ3JlIGV4cGVjdGVkIHRvIGRlbGV0ZSAidkRldGFpbHMiIHdoZW4gd2UncmUgZG9uZS4KLSAqLwotc3RydWN0IHRocmVhZERldGFpbHMgewotICAgIGludCAoKmZ1bmMpKHZvaWQqKTsKLSAgICB2b2lkKiBhcmc7Ci19Owotc3RhdGljIF9fc3RkY2FsbCB1bnNpZ25lZCBpbnQgdGhyZWFkSW50ZXJtZWRpYXJ5KHZvaWQqIHZEZXRhaWxzKQotewotICAgIHN0cnVjdCB0aHJlYWREZXRhaWxzKiBwRGV0YWlscyA9IChzdHJ1Y3QgdGhyZWFkRGV0YWlscyopIHZEZXRhaWxzOwotICAgIGludCByZXN1bHQ7Ci0KLSAgICByZXN1bHQgPSAoKihwRGV0YWlscy0+ZnVuYykpKHBEZXRhaWxzLT5hcmcpOwotCi0gICAgZGVsZXRlIHBEZXRhaWxzOwotCi0gICAgTE9HKExPR19WRVJCT1NFLCAidGhyZWFkIiwgInRocmVhZCBleGl0aW5nXG4iKTsKLSAgICByZXR1cm4gKHVuc2lnbmVkIGludCkgcmVzdWx0OwotfQotCi0vKgotICogQ3JlYXRlIGFuZCBydW4gYSBuZXcgdGhyZWFkLgotICovCi1zdGF0aWMgYm9vbCBkb0NyZWF0ZVRocmVhZChhbmRyb2lkX3RocmVhZF9mdW5jX3QgZm4sIHZvaWQqIGFyZywgYW5kcm9pZF90aHJlYWRfaWRfdCAqaWQpCi17Ci0gICAgSEFORExFIGhUaHJlYWQ7Ci0gICAgc3RydWN0IHRocmVhZERldGFpbHMqIHBEZXRhaWxzID0gbmV3IHRocmVhZERldGFpbHM7IC8vIG11c3QgYmUgb24gaGVhcAotICAgIHVuc2lnbmVkIGludCB0aHJkYWRkcjsKLQotICAgIHBEZXRhaWxzLT5mdW5jID0gZm47Ci0gICAgcERldGFpbHMtPmFyZyA9IGFyZzsKLQotI2lmIGRlZmluZWQoSEFWRV9fQkVHSU5USFJFQURFWCkKLSAgICBoVGhyZWFkID0gKEhBTkRMRSkgX2JlZ2ludGhyZWFkZXgoTlVMTCwgMCwgdGhyZWFkSW50ZXJtZWRpYXJ5LCBwRGV0YWlscywgMCwKLSAgICAgICAgICAgICAgICAgICAgJnRocmRhZGRyKTsKLSAgICBpZiAoaFRocmVhZCA9PSAwKQotI2VsaWYgZGVmaW5lZChIQVZFX0NSRUFURVRIUkVBRCkKLSAgICBoVGhyZWFkID0gQ3JlYXRlVGhyZWFkKE5VTEwsIDAsCi0gICAgICAgICAgICAgICAgICAgIChMUFRIUkVBRF9TVEFSVF9ST1VUSU5FKSB0aHJlYWRJbnRlcm1lZGlhcnksCi0gICAgICAgICAgICAgICAgICAgICh2b2lkKikgcERldGFpbHMsIDAsIChEV09SRCopICZ0aHJkYWRkcik7Ci0gICAgaWYgKGhUaHJlYWQgPT0gTlVMTCkKLSNlbmRpZgotICAgIHsKLSAgICAgICAgTE9HKExPR19XQVJOLCAidGhyZWFkIiwgIldBUk5JTkc6IHRocmVhZCBjcmVhdGUgZmFpbGVkXG4iKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLQotI2lmIGRlZmluZWQoSEFWRV9DUkVBVEVUSFJFQUQpCi0gICAgLyogY2xvc2UgdGhlIG1hbmFnZW1lbnQgaGFuZGxlICovCi0gICAgQ2xvc2VIYW5kbGUoaFRocmVhZCk7Ci0jZW5kaWYKLQotICAgIGlmIChpZCAhPSBOVUxMKSB7Ci0gICAgICAJKmlkID0gKGFuZHJvaWRfdGhyZWFkX2lkX3QpdGhyZGFkZHI7Ci0gICAgfQotCi0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLWludCBhbmRyb2lkQ3JlYXRlUmF3VGhyZWFkRXRjKGFuZHJvaWRfdGhyZWFkX2Z1bmNfdCBmbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkICp1c2VyRGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyKiB0aHJlYWROYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgdGhyZWFkUHJpb3JpdHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IHRocmVhZFN0YWNrU2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhbmRyb2lkX3RocmVhZF9pZF90ICp0aHJlYWRJZCkKLXsKLSAgICByZXR1cm4gZG9DcmVhdGVUaHJlYWQoICBmbiwgdXNlckRhdGEsIHRocmVhZElkKTsKLX0KLQotYW5kcm9pZF90aHJlYWRfaWRfdCBhbmRyb2lkR2V0VGhyZWFkSWQoKQotewotICAgIHJldHVybiAoYW5kcm9pZF90aHJlYWRfaWRfdClHZXRDdXJyZW50VGhyZWFkSWQoKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2Vsc2UKLSNlcnJvciAiVGhyZWFkcyBub3Qgc3VwcG9ydGVkIgotI2VuZGlmCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jcHJhZ21hIG1hcmsgQ29tbW9uIFRocmVhZCBmdW5jdGlvbnMKLSNlbmRpZgotCi1pbnQgYW5kcm9pZENyZWF0ZVRocmVhZChhbmRyb2lkX3RocmVhZF9mdW5jX3QgZm4sIHZvaWQqIGFyZykKLXsKLSAgICByZXR1cm4gY3JlYXRlVGhyZWFkRXRjKGZuLCBhcmcpOwotfQotCi1pbnQgYW5kcm9pZENyZWF0ZVRocmVhZEdldElEKGFuZHJvaWRfdGhyZWFkX2Z1bmNfdCBmbiwgdm9pZCAqYXJnLCBhbmRyb2lkX3RocmVhZF9pZF90ICppZCkKLXsKLSAgICByZXR1cm4gY3JlYXRlVGhyZWFkRXRjKGZuLCBhcmcsICJhbmRyb2lkOnVubmFtZWRfdGhyZWFkIiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIFBSSU9SSVRZX0RFRkFVTFQsIDAsIGlkKTsKLX0KLQotc3RhdGljIGFuZHJvaWRfY3JlYXRlX3RocmVhZF9mbiBnQ3JlYXRlVGhyZWFkRm4gPSBhbmRyb2lkQ3JlYXRlUmF3VGhyZWFkRXRjOwotCi1pbnQgYW5kcm9pZENyZWF0ZVRocmVhZEV0YyhhbmRyb2lkX3RocmVhZF9mdW5jX3QgZW50cnlGdW5jdGlvbiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkICp1c2VyRGF0YSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjaGFyKiB0aHJlYWROYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgdGhyZWFkUHJpb3JpdHksCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZV90IHRocmVhZFN0YWNrU2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBhbmRyb2lkX3RocmVhZF9pZF90ICp0aHJlYWRJZCkKLXsKLSAgICByZXR1cm4gZ0NyZWF0ZVRocmVhZEZuKGVudHJ5RnVuY3Rpb24sIHVzZXJEYXRhLCB0aHJlYWROYW1lLAotICAgICAgICB0aHJlYWRQcmlvcml0eSwgdGhyZWFkU3RhY2tTaXplLCB0aHJlYWRJZCk7Ci19Ci0KLXZvaWQgYW5kcm9pZFNldENyZWF0ZVRocmVhZEZ1bmMoYW5kcm9pZF9jcmVhdGVfdGhyZWFkX2ZuIGZ1bmMpCi17Ci0gICAgZ0NyZWF0ZVRocmVhZEZuID0gZnVuYzsKLX0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vKgotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKiAgICAgIE11dGV4IGNsYXNzCi0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqLwotCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBNdXRleAotI2VuZGlmCi0KLSNpZiBkZWZpbmVkKEhBVkVfUFRIUkVBRFMpICYmICFkZWZpbmVkKEhBVkVfRlVURVgpCi0vKgotICogU2ltcGxlIHB0aHJlYWQgd3JhcHBlci4KLSAqLwotCi1NdXRleDo6TXV0ZXgoKQotewotICAgIF9pbml0KCk7Ci19Ci0KLU11dGV4OjpNdXRleChjb25zdCBjaGFyKiBuYW1lKQotewotICAgIC8vIFhYWDogbmFtZSBub3QgdXNlZCBmb3Igbm93Ci0gICAgX2luaXQoKTsKLX0KLQotdm9pZCBNdXRleDo6X2luaXQoKQotewotICAgIHB0aHJlYWRfbXV0ZXhfdCogcE11dGV4ID0gbmV3IHB0aHJlYWRfbXV0ZXhfdDsKLSAgICBwdGhyZWFkX211dGV4X2luaXQocE11dGV4LCBOVUxMKTsKLSAgICBtU3RhdGUgPSBwTXV0ZXg7Ci19Ci0KLU11dGV4Ojp+TXV0ZXgoKQotewotICAgIGRlbGV0ZSAocHRocmVhZF9tdXRleF90KikgbVN0YXRlOwotfQotCi1zdGF0dXNfdCBNdXRleDo6bG9jaygpCi17Ci0gICAgaW50IHJlczsKLSAgICB3aGlsZSAoKHJlcz1wdGhyZWFkX211dGV4X2xvY2soKHB0aHJlYWRfbXV0ZXhfdCopIG1TdGF0ZSkpID09IEVJTlRSKSA7Ci0gICAgcmV0dXJuIC1yZXM7Ci19Ci0KLXZvaWQgTXV0ZXg6OnVubG9jaygpCi17Ci0gICAgcHRocmVhZF9tdXRleF91bmxvY2soKHB0aHJlYWRfbXV0ZXhfdCopIG1TdGF0ZSk7Ci19Ci0KLXN0YXR1c190IE11dGV4Ojp0cnlMb2NrKCkKLXsKLSAgICBpbnQgcmVzOwotICAgIHdoaWxlICgocmVzPXB0aHJlYWRfbXV0ZXhfdHJ5bG9jaygocHRocmVhZF9tdXRleF90KikgbVN0YXRlKSkgPT0gRUlOVFIpIDsKLSAgICByZXR1cm4gLXJlczsKLX0KLQotI2VsaWYgZGVmaW5lZChIQVZFX0ZVVEVYKQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jZW5kaWYKLQotI2RlZmluZSBTVEFURSAoKGZ1dGV4X211dGV4X3QqKSAoJm1TdGF0ZSkpCi0KLU11dGV4OjpNdXRleCgpCi17Ci0gICAgX2luaXQoKTsKLX0KLQotTXV0ZXg6Ok11dGV4KGNvbnN0IGNoYXIqIG5hbWUpCi17Ci0gICAgX2luaXQoKTsKLX0KLQotdm9pZAotTXV0ZXg6Ol9pbml0KCkKLXsKLSAgICBmdXRleF9tdXRleF9pbml0KFNUQVRFKTsKLX0KLQotTXV0ZXg6On5NdXRleCgpCi17Ci19Ci0KLXN0YXR1c190IE11dGV4Ojpsb2NrKCkKLXsKLSAgICBpbnQgcmVzOwotICAgIHdoaWxlICgocmVzPWZ1dGV4X211dGV4X2xvY2soU1RBVEUsIEZVVEVYX1dBSVRfSU5GSU5JVEUpKSA9PSBFSU5UUikgOwotICAgIHJldHVybiAtcmVzOwotfQotCi12b2lkIE11dGV4Ojp1bmxvY2soKQotewotICAgIGZ1dGV4X211dGV4X3VubG9jayhTVEFURSk7Ci19Ci0KLXN0YXR1c190IE11dGV4Ojp0cnlMb2NrKCkKLXsKLSAgICBpbnQgcmVzOwotICAgIHdoaWxlICgocmVzPWZ1dGV4X211dGV4X3RyeWxvY2soU1RBVEUpKSA9PSBFSU5UUikgOwotICAgIHJldHVybiAtcmVzOwotfQotI3VuZGVmIFNUQVRFCi0KLSNlbGlmIGRlZmluZWQoSEFWRV9XSU4zMl9USFJFQURTKQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jZW5kaWYKLQotTXV0ZXg6Ok11dGV4KCkKLXsKLSAgICBIQU5ETEUgaE11dGV4OwotCi0gICAgYXNzZXJ0KHNpemVvZihoTXV0ZXgpID09IHNpemVvZihtU3RhdGUpKTsKLQotICAgIGhNdXRleCA9IENyZWF0ZU11dGV4KE5VTEwsIEZBTFNFLCBOVUxMKTsKLSAgICBtU3RhdGUgPSAodm9pZCopIGhNdXRleDsKLX0KLQotTXV0ZXg6Ok11dGV4KGNvbnN0IGNoYXIqIG5hbWUpCi17Ci0gICAgLy8gWFhYOiBuYW1lIG5vdCB1c2VkIGZvciBub3cKLSAgICBIQU5ETEUgaE11dGV4OwotCi0gICAgaE11dGV4ID0gQ3JlYXRlTXV0ZXgoTlVMTCwgRkFMU0UsIE5VTEwpOwotICAgIG1TdGF0ZSA9ICh2b2lkKikgaE11dGV4OwotfQotCi1NdXRleDo6fk11dGV4KCkKLXsKLSAgICBDbG9zZUhhbmRsZSgoSEFORExFKSBtU3RhdGUpOwotfQotCi1zdGF0dXNfdCBNdXRleDo6bG9jaygpCi17Ci0gICAgRFdPUkQgZHdXYWl0UmVzdWx0OwotICAgIGR3V2FpdFJlc3VsdCA9IFdhaXRGb3JTaW5nbGVPYmplY3QoKEhBTkRMRSkgbVN0YXRlLCBJTkZJTklURSk7Ci0gICAgcmV0dXJuIGR3V2FpdFJlc3VsdCAhPSBXQUlUX09CSkVDVF8wID8gLTEgOiBOT19FUlJPUjsKLX0KLQotdm9pZCBNdXRleDo6dW5sb2NrKCkKLXsKLSAgICBpZiAoIVJlbGVhc2VNdXRleCgoSEFORExFKSBtU3RhdGUpKQotICAgICAgICBMT0coTE9HX1dBUk4sICJ0aHJlYWQiLCAiV0FSTklORzogYmFkIHJlc3VsdCBmcm9tIHVubG9ja2luZyBtdXRleFxuIik7Ci19Ci0KLXN0YXR1c190IE11dGV4Ojp0cnlMb2NrKCkKLXsKLSAgICBEV09SRCBkd1dhaXRSZXN1bHQ7Ci0KLSAgICBkd1dhaXRSZXN1bHQgPSBXYWl0Rm9yU2luZ2xlT2JqZWN0KChIQU5ETEUpIG1TdGF0ZSwgMCk7Ci0gICAgaWYgKGR3V2FpdFJlc3VsdCAhPSBXQUlUX09CSkVDVF8wICYmIGR3V2FpdFJlc3VsdCAhPSBXQUlUX1RJTUVPVVQpCi0gICAgICAgIExPRyhMT0dfV0FSTiwgInRocmVhZCIsICJXQVJOSU5HOiBiYWQgcmVzdWx0IGZyb20gdHJ5LWxvY2tpbmcgbXV0ZXhcbiIpOwotICAgIHJldHVybiAoZHdXYWl0UmVzdWx0ID09IFdBSVRfT0JKRUNUXzApID8gMCA6IC0xOwotfQotCi0jZWxzZQotI2Vycm9yICJTb21lYm9keSBmb3Jnb3QgdG8gaW1wbGVtZW50IHRocmVhZHMgZm9yIHRoaXMgcGxhdGZvcm0uIgotI2VuZGlmCi0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBDb25kaXRpb24gY2xhc3MKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICovCi0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIENvbmRpdGlvbgotI2VuZGlmCi0KLSNpZiBkZWZpbmVkKEhBVkVfUFRIUkVBRFMpICYmICFkZWZpbmVkKEhBVkVfRlVURVgpCi0KLS8qCi0gKiBDb25zdHJ1Y3Rvci4gIFRoaXMgaXMgYSBzaW1wbGUgcHRocmVhZCB3cmFwcGVyLgotICovCi1Db25kaXRpb246OkNvbmRpdGlvbigpCi17Ci0gICAgcHRocmVhZF9jb25kX3QqIHBDb25kID0gbmV3IHB0aHJlYWRfY29uZF90OwotCi0gICAgcHRocmVhZF9jb25kX2luaXQocENvbmQsIE5VTEwpOwotICAgIG1TdGF0ZSA9IHBDb25kOwotfQotCi0vKgotICogRGVzdHJ1Y3Rvci4KLSAqLwotQ29uZGl0aW9uOjp+Q29uZGl0aW9uKCkKLXsKLSAgICBwdGhyZWFkX2NvbmRfZGVzdHJveSgocHRocmVhZF9jb25kX3QqKSBtU3RhdGUpOwotICAgIGRlbGV0ZSAocHRocmVhZF9jb25kX3QqKSBtU3RhdGU7Ci19Ci0KLS8qCi0gKiBXYWl0IG9uIGEgY29uZGl0aW9uIHZhcmlhYmxlLiAgTG9jayB0aGUgbXV0ZXggYmVmb3JlIGNhbGxpbmcuCi0gKi8KLQotc3RhdHVzX3QgQ29uZGl0aW9uOjp3YWl0KE11dGV4JiBtdXRleCkKLXsKLSAgICBhc3NlcnQobXV0ZXgubVN0YXRlICE9IE5VTEwpOwotCi0gICAgaW50IGNjOwotICAgIHdoaWxlICgoY2MgPSBwdGhyZWFkX2NvbmRfd2FpdCgocHRocmVhZF9jb25kX3QqKW1TdGF0ZSwKLSAgICAgICAgICAgICAgICAocHRocmVhZF9tdXRleF90KikgbXV0ZXgubVN0YXRlKSkgPT0gRUlOVFIpIDsKLSAgICByZXR1cm4gLWNjOwotfQotCi1zdGF0dXNfdCBDb25kaXRpb246OndhaXQoTXV0ZXgmIG11dGV4LCBuc2Vjc190IGFic3RpbWUpCi17Ci0gICAgYXNzZXJ0KG11dGV4Lm1TdGF0ZSAhPSBOVUxMKTsKLQotICAgIHN0cnVjdCB0aW1lc3BlYyB0czsKLSAgICB0cy50dl9zZWMgPSBhYnN0aW1lLzEwMDAwMDAwMDA7Ci0gICAgdHMudHZfbnNlYyA9IGFic3RpbWUtKHRzLnR2X3NlYyoxMDAwMDAwMDAwKTsKLSAgICAKLSAgICBpbnQgY2M7Ci0gICAgd2hpbGUgKChjYyA9IHB0aHJlYWRfY29uZF90aW1lZHdhaXQoKHB0aHJlYWRfY29uZF90KiltU3RhdGUsCi0gICAgICAgICAgICAocHRocmVhZF9tdXRleF90KikgbXV0ZXgubVN0YXRlLCAmdHMpKSA9PSBFSU5UUikgOwotICAgIHJldHVybiAtY2M7Ci19Ci0KLXN0YXR1c190IENvbmRpdGlvbjo6d2FpdFJlbGF0aXZlKE11dGV4JiBtdXRleCwgbnNlY3NfdCByZWx0aW1lKQotewotICAgIHJldHVybiB3YWl0KG11dGV4LCBzeXN0ZW1UaW1lKCkrcmVsdGltZSk7Ci19Ci0KLS8qCi0gKiBTaWduYWwgdGhlIGNvbmRpdGlvbiB2YXJpYWJsZSwgYWxsb3dpbmcgb25lIHRocmVhZCB0byBjb250aW51ZS4KLSAqLwotdm9pZCBDb25kaXRpb246OnNpZ25hbCgpCi17Ci0gICAgcHRocmVhZF9jb25kX3NpZ25hbCgocHRocmVhZF9jb25kX3QqKSBtU3RhdGUpOwotfQotCi0vKgotICogU2lnbmFsIHRoZSBjb25kaXRpb24gdmFyaWFibGUsIGFsbG93aW5nIGFsbCB0aHJlYWRzIHRvIGNvbnRpbnVlLgotICovCi12b2lkIENvbmRpdGlvbjo6YnJvYWRjYXN0KCkKLXsKLSAgICBwdGhyZWFkX2NvbmRfYnJvYWRjYXN0KChwdGhyZWFkX2NvbmRfdCopIG1TdGF0ZSk7Ci19Ci0KLSNlbGlmIGRlZmluZWQoSEFWRV9GVVRFWCkKLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI2VuZGlmCi0KLSNkZWZpbmUgU1RBVEUgKChmdXRleF9jb25kX3QqKSAoJm1TdGF0ZSkpCi0KLS8qCi0gKiBDb25zdHJ1Y3Rvci4gIFRoaXMgaXMgYSBzaW1wbGUgcHRocmVhZCB3cmFwcGVyLgotICovCi1Db25kaXRpb246OkNvbmRpdGlvbigpCi17Ci0gICAgZnV0ZXhfY29uZF9pbml0KFNUQVRFKTsKLX0KLQotLyoKLSAqIERlc3RydWN0b3IuCi0gKi8KLUNvbmRpdGlvbjo6fkNvbmRpdGlvbigpCi17Ci19Ci0KLS8qCi0gKiBXYWl0IG9uIGEgY29uZGl0aW9uIHZhcmlhYmxlLiAgTG9jayB0aGUgbXV0ZXggYmVmb3JlIGNhbGxpbmcuCi0gKi8KLQotc3RhdHVzX3QgQ29uZGl0aW9uOjp3YWl0KE11dGV4JiBtdXRleCkKLXsKLSAgICBhc3NlcnQobXV0ZXgubVN0YXRlICE9IE5VTEwpOwotCi0gICAgaW50IHJlczsKLSAgICB3aGlsZSAoKHJlcyA9IGZ1dGV4X2NvbmRfd2FpdChTVEFURSwKLSAgICAgICAgKGZ1dGV4X211dGV4X3QqKSgmbXV0ZXgubVN0YXRlKSwgRlVURVhfV0FJVF9JTkZJTklURSkpID09IC1FSU5UUikgOwotCi0gICAgcmV0dXJuIC1yZXM7Ci19Ci0KLXN0YXR1c190IENvbmRpdGlvbjo6d2FpdChNdXRleCYgbXV0ZXgsIG5zZWNzX3QgYWJzdGltZSkKLXsKLSAgICBuc2Vjc190IHJlbHRpbWUgPSBhYnN0aW1lIC0gc3lzdGVtVGltZSgpOwotICAgIGlmIChyZWx0aW1lIDw9IDApIHJldHVybiB0cnVlOwotICAgIHJldHVybiB3YWl0UmVsYXRpdmUobXV0ZXgsIHJlbHRpbWUpOwotfQotCi1zdGF0dXNfdCBDb25kaXRpb246OndhaXRSZWxhdGl2ZShNdXRleCYgbXV0ZXgsIG5zZWNzX3QgcmVsdGltZSkKLXsKLSAgICBhc3NlcnQobXV0ZXgubVN0YXRlICE9IE5VTEwpOwotICAgIGludCByZXM7Ci0gICAgdW5zaWduZWQgbXNlYyA9IG5zMm1zKHJlbHRpbWUpOwotICAgIGlmKG1zZWMgPT0gMCkKLSAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgLy8gVGhpcyBjb2RlIHdpbGwgbm90IHRpbWUgb3V0IGF0IHRoZSBjb3JyZWN0IHRpbWUgaWYgaW50ZXJydXB0ZWQgYnkgc2lnbmFscwotICAgIHdoaWxlICgocmVzID0gZnV0ZXhfY29uZF93YWl0KFNUQVRFLAotICAgICAgICAoZnV0ZXhfbXV0ZXhfdCopKCZtdXRleC5tU3RhdGUpLCBtc2VjKSkgPT0gLUVJTlRSKSA7Ci0gICAgcmV0dXJuIHJlczsKLX0KLQotLyoKLSAqIFNpZ25hbCB0aGUgY29uZGl0aW9uIHZhcmlhYmxlLCBhbGxvd2luZyBvbmUgdGhyZWFkIHRvIGNvbnRpbnVlLgotICovCi12b2lkIENvbmRpdGlvbjo6c2lnbmFsKCkKLXsKLSAgICBmdXRleF9jb25kX3NpZ25hbChTVEFURSk7Ci19Ci0KLS8qCi0gKiBTaWduYWwgdGhlIGNvbmRpdGlvbiB2YXJpYWJsZSwgYWxsb3dpbmcgYWxsIHRocmVhZHMgdG8gY29udGludWUuCi0gKi8KLXZvaWQgQ29uZGl0aW9uOjpicm9hZGNhc3QoKQotewotICAgIGZ1dGV4X2NvbmRfYnJvYWRjYXN0KFNUQVRFKTsKLX0KLQotI3VuZGVmIFNUQVRFCi0KLSNlbGlmIGRlZmluZWQoSEFWRV9XSU4zMl9USFJFQURTKQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jZW5kaWYKLQotLyoKLSAqIFdpbmRvd3MgZG9lc24ndCBoYXZlIGEgY29uZGl0aW9uIHZhcmlhYmxlIHNvbHV0aW9uLiAgSXQncyBwb3NzaWJsZQotICogdG8gY3JlYXRlIG9uZSwgYnV0IGl0J3MgZWFzeSB0byBnZXQgaXQgd3JvbmcuICBGb3IgYSBkaXNjdXNzaW9uLCBhbmQKLSAqIHRoZSBvcmlnaW4gb2YgdGhpcyBpbXBsZW1lbnRhdGlvbiwgc2VlOgotICoKLSAqICBodHRwOi8vd3d3LmNzLnd1c3RsLmVkdS9+c2NobWlkdC93aW4zMi1jdi0xLmh0bWwKLSAqCi0gKiBUaGUgaW1wbGVtZW50YXRpb24gc2hvd24gb24gdGhlIHBhZ2UgZG9lcyBOT1QgZm9sbG93IFBPU0lYIHNlbWFudGljcy4KLSAqIEFzIGFuIG9wdGltaXphdGlvbiB0aGV5IHJlcXVpcmUgYWNxdWlyaW5nIHRoZSBleHRlcm5hbCBtdXRleCBiZWZvcmUKLSAqIGNhbGxpbmcgc2lnbmFsKCkgYW5kIGJyb2FkY2FzdCgpLCB3aGVyZWFzIFBPU0lYIG9ubHkgcmVxdWlyZXMgZ3JhYmJpbmcKLSAqIGl0IGJlZm9yZSBjYWxsaW5nIHdhaXQoKS4gIFRoZSBpbXBsZW1lbnRhdGlvbiBoZXJlIGhhcyBiZWVuIHVuLW9wdGltaXplZAotICogdG8gaGF2ZSB0aGUgY29ycmVjdCBiZWhhdmlvci4KLSAqLwotdHlwZWRlZiBzdHJ1Y3QgV2luQ29uZGl0aW9uIHsKLSAgICAvLyBOdW1iZXIgb2Ygd2FpdGluZyB0aHJlYWRzLgotICAgIGludCAgICAgICAgICAgICAgICAgd2FpdGVyc0NvdW50OwotCi0gICAgLy8gU2VyaWFsaXplIGFjY2VzcyB0byB3YWl0ZXJzQ291bnQuCi0gICAgQ1JJVElDQUxfU0VDVElPTiAgICB3YWl0ZXJzQ291bnRMb2NrOwotCi0gICAgLy8gU2VtYXBob3JlIHVzZWQgdG8gcXVldWUgdXAgdGhyZWFkcyB3YWl0aW5nIGZvciB0aGUgY29uZGl0aW9uIHRvCi0gICAgLy8gYmVjb21lIHNpZ25hbGVkLgotICAgIEhBTkRMRSAgICAgICAgICAgICAgc2VtYTsKLQotICAgIC8vIEFuIGF1dG8tcmVzZXQgZXZlbnQgdXNlZCBieSB0aGUgYnJvYWRjYXN0L3NpZ25hbCB0aHJlYWQgdG8gd2FpdAotICAgIC8vIGZvciBhbGwgdGhlIHdhaXRpbmcgdGhyZWFkKHMpIHRvIHdha2UgdXAgYW5kIGJlIHJlbGVhc2VkIGZyb20KLSAgICAvLyB0aGUgc2VtYXBob3JlLgotICAgIEhBTkRMRSAgICAgICAgICAgICAgd2FpdGVyc0RvbmU7Ci0KLSAgICAvLyBUaGlzIG11dGV4IHdvdWxkbid0IGJlIG5lY2Vzc2FyeSBpZiB3ZSByZXF1aXJlZCB0aGF0IHRoZSBjYWxsZXIKLSAgICAvLyBsb2NrIHRoZSBleHRlcm5hbCBtdXRleCBiZWZvcmUgY2FsbGluZyBzaWduYWwoKSBhbmQgYnJvYWRjYXN0KCkuCi0gICAgLy8gSSdtIHRyeWluZyB0byBtaW1pYyBwdGhyZWFkIHNlbWFudGljcyB0aG91Z2guCi0gICAgSEFORExFICAgICAgICAgICAgICBpbnRlcm5hbE11dGV4OwotCi0gICAgLy8gS2VlcHMgdHJhY2sgb2Ygd2hldGhlciB3ZSB3ZXJlIGJyb2FkY2FzdGluZyBvciBzaWduYWxpbmcuICBUaGlzCi0gICAgLy8gYWxsb3dzIHVzIHRvIG9wdGltaXplIHRoZSBjb2RlIGlmIHdlJ3JlIGp1c3Qgc2lnbmFsaW5nLgotICAgIGJvb2wgICAgICAgICAgICAgICAgd2FzQnJvYWRjYXN0OwotCi0gICAgc3RhdHVzX3Qgd2FpdChXaW5Db25kaXRpb24qIGNvbmRTdGF0ZSwgSEFORExFIGhNdXRleCwgbnNlY3NfdCogYWJzdGltZSkKLSAgICB7Ci0gICAgICAgIC8vIEluY3JlbWVudCB0aGUgd2FpdCBjb3VudCwgYXZvaWRpbmcgcmFjZSBjb25kaXRpb25zLgotICAgICAgICBFbnRlckNyaXRpY2FsU2VjdGlvbigmY29uZFN0YXRlLT53YWl0ZXJzQ291bnRMb2NrKTsKLSAgICAgICAgY29uZFN0YXRlLT53YWl0ZXJzQ291bnQrKzsKLSAgICAgICAgLy9wcmludGYoIisrKyB3YWl0OiBpbmNyIHdhaXRlcnNDb3VudCB0byAlZCAodGlkPSVsZClcbiIsCi0gICAgICAgIC8vICAgIGNvbmRTdGF0ZS0+d2FpdGVyc0NvdW50LCBnZXRUaHJlYWRJZCgpKTsKLSAgICAgICAgTGVhdmVDcml0aWNhbFNlY3Rpb24oJmNvbmRTdGF0ZS0+d2FpdGVyc0NvdW50TG9jayk7Ci0gICAgCi0gICAgICAgIERXT1JEIHRpbWVvdXQgPSBJTkZJTklURTsKLSAgICAgICAgaWYgKGFic3RpbWUpIHsKLSAgICAgICAgICAgIG5zZWNzX3QgcmVsdGltZSA9ICphYnN0aW1lIC0gc3lzdGVtVGltZSgpOwotICAgICAgICAgICAgaWYgKHJlbHRpbWUgPCAwKQotICAgICAgICAgICAgICAgIHJlbHRpbWUgPSAwOwotICAgICAgICAgICAgdGltZW91dCA9IHJlbHRpbWUvMTAwMDAwMDsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgLy8gQXRvbWljYWxseSByZWxlYXNlIHRoZSBleHRlcm5hbCBtdXRleCBhbmQgd2FpdCBvbiB0aGUgc2VtYXBob3JlLgotICAgICAgICBEV09SRCByZXMgPQotICAgICAgICAgICAgU2lnbmFsT2JqZWN0QW5kV2FpdChoTXV0ZXgsIGNvbmRTdGF0ZS0+c2VtYSwgdGltZW91dCwgRkFMU0UpOwotICAgIAotICAgICAgICAvL3ByaW50ZigiKysrIHdhaXQ6IGF3YWtlICh0aWQ9JWxkKVxuIiwgZ2V0VGhyZWFkSWQoKSk7Ci0gICAgCi0gICAgICAgIC8vIFJlYWNxdWlyZSBsb2NrIHRvIGF2b2lkIHJhY2UgY29uZGl0aW9ucy4KLSAgICAgICAgRW50ZXJDcml0aWNhbFNlY3Rpb24oJmNvbmRTdGF0ZS0+d2FpdGVyc0NvdW50TG9jayk7Ci0gICAgCi0gICAgICAgIC8vIE5vIGxvbmdlciB3YWl0aW5nLgotICAgICAgICBjb25kU3RhdGUtPndhaXRlcnNDb3VudC0tOwotICAgIAotICAgICAgICAvLyBDaGVjayB0byBzZWUgaWYgd2UncmUgdGhlIGxhc3Qgd2FpdGVyIGFmdGVyIGEgYnJvYWRjYXN0LgotICAgICAgICBib29sIGxhc3RXYWl0ZXIgPSAoY29uZFN0YXRlLT53YXNCcm9hZGNhc3QgJiYgY29uZFN0YXRlLT53YWl0ZXJzQ291bnQgPT0gMCk7Ci0gICAgCi0gICAgICAgIC8vcHJpbnRmKCIrKysgd2FpdDogbGFzdFdhaXRlcj0lZCAod2FzQmM9JWQgd2M9JWQpXG4iLAotICAgICAgICAvLyAgICBsYXN0V2FpdGVyLCBjb25kU3RhdGUtPndhc0Jyb2FkY2FzdCwgY29uZFN0YXRlLT53YWl0ZXJzQ291bnQpOwotICAgIAotICAgICAgICBMZWF2ZUNyaXRpY2FsU2VjdGlvbigmY29uZFN0YXRlLT53YWl0ZXJzQ291bnRMb2NrKTsKLSAgICAKLSAgICAgICAgLy8gSWYgd2UncmUgdGhlIGxhc3Qgd2FpdGVyIHRocmVhZCBkdXJpbmcgdGhpcyBwYXJ0aWN1bGFyIGJyb2FkY2FzdAotICAgICAgICAvLyB0aGVuIHNpZ25hbCBicm9hZGNhc3QoKSB0aGF0IHdlJ3JlIGFsbCBhd2FrZS4gIEl0J2xsIGRyb3AgdGhlCi0gICAgICAgIC8vIGludGVybmFsIG11dGV4LgotICAgICAgICBpZiAobGFzdFdhaXRlcikgewotICAgICAgICAgICAgLy8gQXRvbWljYWxseSBzaWduYWwgdGhlICJ3YWl0ZXJzRG9uZSIgZXZlbnQgYW5kIHdhaXQgdW50aWwgd2UKLSAgICAgICAgICAgIC8vIGNhbiBhY3F1aXJlIHRoZSBpbnRlcm5hbCBtdXRleC4gIFdlIHdhbnQgdG8gZG8gdGhpcyBpbiBvbmUgc3RlcAotICAgICAgICAgICAgLy8gYmVjYXVzZSBpdCBlbnN1cmVzIHRoYXQgZXZlcnlib2R5IGlzIGluIHRoZSBtdXRleCBGSUZPIGJlZm9yZQotICAgICAgICAgICAgLy8gYW55IHRocmVhZCBoYXMgYSBjaGFuY2UgdG8gcnVuLiAgV2l0aG91dCBpdCwgYW5vdGhlciB0aHJlYWQKLSAgICAgICAgICAgIC8vIGNvdWxkIHdha2UgdXAsIGRvIHdvcmssIGFuZCBob3AgYmFjayBpbiBhaGVhZCBvZiB1cy4KLSAgICAgICAgICAgIFNpZ25hbE9iamVjdEFuZFdhaXQoY29uZFN0YXRlLT53YWl0ZXJzRG9uZSwgY29uZFN0YXRlLT5pbnRlcm5hbE11dGV4LAotICAgICAgICAgICAgICAgIElORklOSVRFLCBGQUxTRSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBHcmFiIHRoZSBpbnRlcm5hbCBtdXRleC4KLSAgICAgICAgICAgIFdhaXRGb3JTaW5nbGVPYmplY3QoY29uZFN0YXRlLT5pbnRlcm5hbE11dGV4LCBJTkZJTklURSk7Ci0gICAgICAgIH0KLSAgICAKLSAgICAgICAgLy8gUmVsZWFzZSB0aGUgaW50ZXJuYWwgYW5kIGdyYWIgdGhlIGV4dGVybmFsLgotICAgICAgICBSZWxlYXNlTXV0ZXgoY29uZFN0YXRlLT5pbnRlcm5hbE11dGV4KTsKLSAgICAgICAgV2FpdEZvclNpbmdsZU9iamVjdChoTXV0ZXgsIElORklOSVRFKTsKLSAgICAKLSAgICAgICAgcmV0dXJuIHJlcyA9PSBXQUlUX09CSkVDVF8wID8gTk9fRVJST1IgOiAtMTsKLSAgICB9Ci19IFdpbkNvbmRpdGlvbjsKLQotLyoKLSAqIENvbnN0cnVjdG9yLiAgU2V0IHVwIHRoZSBXaW5Db25kaXRpb24gc3R1ZmYuCi0gKi8KLUNvbmRpdGlvbjo6Q29uZGl0aW9uKCkKLXsKLSAgICBXaW5Db25kaXRpb24qIGNvbmRTdGF0ZSA9IG5ldyBXaW5Db25kaXRpb247Ci0KLSAgICBjb25kU3RhdGUtPndhaXRlcnNDb3VudCA9IDA7Ci0gICAgY29uZFN0YXRlLT53YXNCcm9hZGNhc3QgPSBmYWxzZTsKLSAgICAvLyBzZW1hcGhvcmU6IG5vIHNlY3VyaXR5LCBpbml0aWFsIHZhbHVlIG9mIDAKLSAgICBjb25kU3RhdGUtPnNlbWEgPSBDcmVhdGVTZW1hcGhvcmUoTlVMTCwgMCwgMHg3ZmZmZmZmZiwgTlVMTCk7Ci0gICAgSW5pdGlhbGl6ZUNyaXRpY2FsU2VjdGlvbigmY29uZFN0YXRlLT53YWl0ZXJzQ291bnRMb2NrKTsKLSAgICAvLyBhdXRvLXJlc2V0IGV2ZW50LCBub3Qgc2lnbmFsZWQgaW5pdGlhbGx5Ci0gICAgY29uZFN0YXRlLT53YWl0ZXJzRG9uZSA9IENyZWF0ZUV2ZW50KE5VTEwsIEZBTFNFLCBGQUxTRSwgTlVMTCk7Ci0gICAgLy8gdXNlZCBzbyB3ZSBkb24ndCBoYXZlIHRvIGxvY2sgZXh0ZXJuYWwgbXV0ZXggb24gc2lnbmFsL2Jyb2FkY2FzdAotICAgIGNvbmRTdGF0ZS0+aW50ZXJuYWxNdXRleCA9IENyZWF0ZU11dGV4KE5VTEwsIEZBTFNFLCBOVUxMKTsKLQotICAgIG1TdGF0ZSA9IGNvbmRTdGF0ZTsKLX0KLQotLyoKLSAqIERlc3RydWN0b3IuICBGcmVlIFdpbmRvd3MgcmVzb3VyY2VzIGFzIHdlbGwgYXMgb3VyIGFsbG9jYXRlZCBzdG9yYWdlLgotICovCi1Db25kaXRpb246On5Db25kaXRpb24oKQotewotICAgIFdpbkNvbmRpdGlvbiogY29uZFN0YXRlID0gKFdpbkNvbmRpdGlvbiopIG1TdGF0ZTsKLSAgICBpZiAoY29uZFN0YXRlICE9IE5VTEwpIHsKLSAgICAgICAgQ2xvc2VIYW5kbGUoY29uZFN0YXRlLT5zZW1hKTsKLSAgICAgICAgQ2xvc2VIYW5kbGUoY29uZFN0YXRlLT53YWl0ZXJzRG9uZSk7Ci0gICAgICAgIGRlbGV0ZSBjb25kU3RhdGU7Ci0gICAgfQotfQotCi0KLXN0YXR1c190IENvbmRpdGlvbjo6d2FpdChNdXRleCYgbXV0ZXgpCi17Ci0gICAgV2luQ29uZGl0aW9uKiBjb25kU3RhdGUgPSAoV2luQ29uZGl0aW9uKikgbVN0YXRlOwotICAgIEhBTkRMRSBoTXV0ZXggPSAoSEFORExFKSBtdXRleC5tU3RhdGU7Ci0gICAgCi0gICAgcmV0dXJuICgoV2luQ29uZGl0aW9uKiltU3RhdGUpLT53YWl0KGNvbmRTdGF0ZSwgaE11dGV4LCBOVUxMKTsKLX0KLQotc3RhdHVzX3QgQ29uZGl0aW9uOjp3YWl0KE11dGV4JiBtdXRleCwgbnNlY3NfdCBhYnN0aW1lKQotewotICAgIFdpbkNvbmRpdGlvbiogY29uZFN0YXRlID0gKFdpbkNvbmRpdGlvbiopIG1TdGF0ZTsKLSAgICBIQU5ETEUgaE11dGV4ID0gKEhBTkRMRSkgbXV0ZXgubVN0YXRlOwotCi0gICAgcmV0dXJuICgoV2luQ29uZGl0aW9uKiltU3RhdGUpLT53YWl0KGNvbmRTdGF0ZSwgaE11dGV4LCAmYWJzdGltZSk7Ci19Ci0KLXN0YXR1c190IENvbmRpdGlvbjo6d2FpdFJlbGF0aXZlKE11dGV4JiBtdXRleCwgbnNlY3NfdCByZWx0aW1lKQotewotICAgIHJldHVybiB3YWl0KG11dGV4LCBzeXN0ZW1UaW1lKCkrcmVsdGltZSk7Ci19Ci0KLS8qCi0gKiBTaWduYWwgdGhlIGNvbmRpdGlvbiB2YXJpYWJsZSwgYWxsb3dpbmcgb25lIHRocmVhZCB0byBjb250aW51ZS4KLSAqLwotdm9pZCBDb25kaXRpb246OnNpZ25hbCgpCi17Ci0gICAgV2luQ29uZGl0aW9uKiBjb25kU3RhdGUgPSAoV2luQ29uZGl0aW9uKikgbVN0YXRlOwotCi0gICAgLy8gTG9jayB0aGUgaW50ZXJuYWwgbXV0ZXguICBUaGlzIGVuc3VyZXMgdGhhdCB3ZSBkb24ndCBjbGFzaCB3aXRoCi0gICAgLy8gYnJvYWRjYXN0KCkuCi0gICAgV2FpdEZvclNpbmdsZU9iamVjdChjb25kU3RhdGUtPmludGVybmFsTXV0ZXgsIElORklOSVRFKTsKLQotICAgIEVudGVyQ3JpdGljYWxTZWN0aW9uKCZjb25kU3RhdGUtPndhaXRlcnNDb3VudExvY2spOwotICAgIGJvb2wgaGF2ZVdhaXRlcnMgPSAoY29uZFN0YXRlLT53YWl0ZXJzQ291bnQgPiAwKTsKLSAgICBMZWF2ZUNyaXRpY2FsU2VjdGlvbigmY29uZFN0YXRlLT53YWl0ZXJzQ291bnRMb2NrKTsKLQotICAgIC8vIElmIG5vIHdhaXRlcnMsIHRoZW4gdGhpcyBpcyBhIG5vLW9wLiAgT3RoZXJ3aXNlLCBrbm9jayB0aGUgc2VtYXBob3JlCi0gICAgLy8gZG93biBhIG5vdGNoLgotICAgIGlmIChoYXZlV2FpdGVycykKLSAgICAgICAgUmVsZWFzZVNlbWFwaG9yZShjb25kU3RhdGUtPnNlbWEsIDEsIDApOwotCi0gICAgLy8gUmVsZWFzZSBpbnRlcm5hbCBtdXRleC4KLSAgICBSZWxlYXNlTXV0ZXgoY29uZFN0YXRlLT5pbnRlcm5hbE11dGV4KTsKLX0KLQotLyoKLSAqIFNpZ25hbCB0aGUgY29uZGl0aW9uIHZhcmlhYmxlLCBhbGxvd2luZyBhbGwgdGhyZWFkcyB0byBjb250aW51ZS4KLSAqCi0gKiBGaXJzdCB3ZSBoYXZlIHRvIHdha2UgdXAgYWxsIHRocmVhZHMgd2FpdGluZyBvbiB0aGUgc2VtYXBob3JlLCB0aGVuCi0gKiB3ZSB3YWl0IHVudGlsIGFsbCBvZiB0aGUgdGhyZWFkcyBoYXZlIGFjdHVhbGx5IGJlZW4gd29rZW4gYmVmb3JlCi0gKiByZWxlYXNpbmcgdGhlIGludGVybmFsIG11dGV4LiAgVGhpcyBlbnN1cmVzIHRoYXQgYWxsIHRocmVhZHMgYXJlIHdva2VuLgotICovCi12b2lkIENvbmRpdGlvbjo6YnJvYWRjYXN0KCkKLXsKLSAgICBXaW5Db25kaXRpb24qIGNvbmRTdGF0ZSA9IChXaW5Db25kaXRpb24qKSBtU3RhdGU7Ci0KLSAgICAvLyBMb2NrIHRoZSBpbnRlcm5hbCBtdXRleC4gIFRoaXMga2VlcHMgdGhlIGd1eXMgd2UncmUgd2FraW5nIHVwCi0gICAgLy8gZnJvbSBnZXR0aW5nIHRvbyBmYXIuCi0gICAgV2FpdEZvclNpbmdsZU9iamVjdChjb25kU3RhdGUtPmludGVybmFsTXV0ZXgsIElORklOSVRFKTsKLQotICAgIEVudGVyQ3JpdGljYWxTZWN0aW9uKCZjb25kU3RhdGUtPndhaXRlcnNDb3VudExvY2spOwotICAgIGJvb2wgaGF2ZVdhaXRlcnMgPSBmYWxzZTsKLQotICAgIGlmIChjb25kU3RhdGUtPndhaXRlcnNDb3VudCA+IDApIHsKLSAgICAgICAgaGF2ZVdhaXRlcnMgPSB0cnVlOwotICAgICAgICBjb25kU3RhdGUtPndhc0Jyb2FkY2FzdCA9IHRydWU7Ci0gICAgfQotCi0gICAgaWYgKGhhdmVXYWl0ZXJzKSB7Ci0gICAgICAgIC8vIFdha2UgdXAgYWxsIHRoZSB3YWl0ZXJzLgotICAgICAgICBSZWxlYXNlU2VtYXBob3JlKGNvbmRTdGF0ZS0+c2VtYSwgY29uZFN0YXRlLT53YWl0ZXJzQ291bnQsIDApOwotCi0gICAgICAgIExlYXZlQ3JpdGljYWxTZWN0aW9uKCZjb25kU3RhdGUtPndhaXRlcnNDb3VudExvY2spOwotCi0gICAgICAgIC8vIFdhaXQgZm9yIGFsbCBhd2FrZW5lZCB0aHJlYWRzIHRvIGFjcXVpcmUgdGhlIGNvdW50aW5nIHNlbWFwaG9yZS4KLSAgICAgICAgLy8gVGhlIGxhc3QgZ3V5IHdobyB3YXMgd2FpdGluZyBzZXRzIHRoaXMuCi0gICAgICAgIFdhaXRGb3JTaW5nbGVPYmplY3QoY29uZFN0YXRlLT53YWl0ZXJzRG9uZSwgSU5GSU5JVEUpOwotCi0gICAgICAgIC8vIFJlc2V0IHdhc0Jyb2FkY2FzdC4gIChObyBjcml0IHNlY3Rpb24gbmVlZGVkIGJlY2F1c2Ugbm9ib2R5Ci0gICAgICAgIC8vIGVsc2UgY2FuIHdha2UgdXAgdG8gcG9rZSBhdCBpdC4pCi0gICAgICAgIGNvbmRTdGF0ZS0+d2FzQnJvYWRjYXN0ID0gMDsKLSAgICB9IGVsc2UgewotICAgICAgICAvLyBub3RoaW5nIHRvIGRvCi0gICAgICAgIExlYXZlQ3JpdGljYWxTZWN0aW9uKCZjb25kU3RhdGUtPndhaXRlcnNDb3VudExvY2spOwotICAgIH0KLQotICAgIC8vIFJlbGVhc2UgaW50ZXJuYWwgbXV0ZXguCi0gICAgUmVsZWFzZU11dGV4KGNvbmRTdGF0ZS0+aW50ZXJuYWxNdXRleCk7Ci19Ci0KLSNlbHNlCi0jZXJyb3IgImNvbmRpdGlvbiB2YXJpYWJsZXMgbm90IHN1cHBvcnRlZCBvbiB0aGlzIHBsYXRmb3JtIgotI2VuZGlmCi0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBSZWFkV3JpdGVMb2NrIGNsYXNzCi0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqLwotCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBSZWFkV3JpdGVMb2NrCi0jZW5kaWYKLQotLyoKLSAqIEFkZCBhIHJlYWRlci4gIFJlYWRlcnMgYXJlIG5pY2UuICBUaGV5IHNoYXJlLgotICovCi12b2lkIFJlYWRXcml0ZUxvY2s6OmxvY2tGb3JSZWFkKCkKLXsKLSAgICBtTG9jay5sb2NrKCk7Ci0gICAgd2hpbGUgKG1OdW1Xcml0ZXJzID4gMCkgewotICAgICAgICBMT0coTE9HX0RFQlVHLCAidGhyZWFkIiwgIisrKyBsb2NrRm9yUmVhZDogd2FpdGluZ1xuIik7Ci0gICAgICAgIG1SZWFkV2FpdGVyLndhaXQobUxvY2spOwotICAgIH0KLSAgICBhc3NlcnQobU51bVdyaXRlcnMgPT0gMCk7Ci0gICAgbU51bVJlYWRlcnMrKzsKLSNpZiBkZWZpbmVkKFBSSU5UX1JFTkRFUl9USU1FUykKLSAgICBpZiAobU51bVJlYWRlcnMgPT0gMSkKLSAgICAgICAgbURlYnVnVGltZXIuc3RhcnQoKTsKLSNlbmRpZgotICAgIG1Mb2NrLnVubG9jaygpOwotfQotCi0vKgotICogVHJ5IHRvIGFkZCBhIHJlYWRlci4gIElmIGl0IGRvZXNuJ3Qgd29yayByaWdodCBhd2F5LCByZXR1cm4gImZhbHNlIi4KLSAqLwotYm9vbCBSZWFkV3JpdGVMb2NrOjp0cnlMb2NrRm9yUmVhZCgpCi17Ci0gICAgbUxvY2subG9jaygpOwotICAgIGlmIChtTnVtV3JpdGVycyA+IDApIHsKLSAgICAgICAgbUxvY2sudW5sb2NrKCk7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgYXNzZXJ0KG1OdW1Xcml0ZXJzID09IDApOwotICAgIG1OdW1SZWFkZXJzKys7Ci0jaWYgZGVmaW5lZChQUklOVF9SRU5ERVJfVElNRVMpCi0gICAgaWYgKG1OdW1SZWFkZXJzID09IDEpCi0gICAgICAgIG1EZWJ1Z1RpbWVyLnN0YXJ0KCk7Ci0jZW5kaWYKLSAgICBtTG9jay51bmxvY2soKTsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotLyoKLSAqIFJlbW92ZSBhIHJlYWRlci4KLSAqLwotdm9pZCBSZWFkV3JpdGVMb2NrOjp1bmxvY2tGb3JSZWFkKCkKLXsKLSAgICBtTG9jay5sb2NrKCk7Ci0gICAgaWYgKG1OdW1SZWFkZXJzID09IDApIHsKLSAgICAgICAgTE9HKExPR19XQVJOLCAidGhyZWFkIiwKLSAgICAgICAgICAgICJXQVJOSU5HOiB1bmxvY2tGb3JSZWFkIHJlcXVlc3RlZCwgYnV0IG5vdCBsb2NrZWRcbiIpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGFzc2VydChtTnVtUmVhZGVycyA+IDApOwotICAgIGFzc2VydChtTnVtV3JpdGVycyA9PSAwKTsKLSAgICBtTnVtUmVhZGVycy0tOwotICAgIGlmIChtTnVtUmVhZGVycyA9PSAwKSB7ICAgICAgICAgICAvLyBsYXN0IHJlYWRlcj8KLSNpZiBkZWZpbmVkKFBSSU5UX1JFTkRFUl9USU1FUykKLSAgICAgICAgbURlYnVnVGltZXIuc3RvcCgpOwotICAgICAgICBwcmludGYoIiByZGxrIGhlbGQgJS4zZiBtc2VjXG4iLAotICAgICAgICAgICAgKGRvdWJsZSkgbURlYnVnVGltZXIuZHVyYXRpb25Vc2VjcygpIC8gMTAwMC4wKTsKLSNlbmRpZgotICAgICAgICAvL3ByaW50ZigiKysrIHNpZ25hbGluZyB3cml0ZXJzIChpZiBhbnkpXG4iKTsKLSAgICAgICAgbVdyaXRlV2FpdGVyLnNpZ25hbCgpOyAgICAgIC8vIHdha2Ugb25lIHdyaXRlciAoaWYgYW55KQotICAgIH0KLSAgICBtTG9jay51bmxvY2soKTsKLX0KLQotLyoKLSAqIEFkZCBhIHdyaXRlci4gIFRoaXMgcmVxdWlyZXMgZXhjbHVzaXZlIGFjY2VzcyB0byB0aGUgb2JqZWN0LgotICovCi12b2lkIFJlYWRXcml0ZUxvY2s6OmxvY2tGb3JXcml0ZSgpCi17Ci0gICAgbUxvY2subG9jaygpOwotICAgIHdoaWxlIChtTnVtUmVhZGVycyA+IDAgfHwgbU51bVdyaXRlcnMgPiAwKSB7Ci0gICAgICAgIExPRyhMT0dfREVCVUcsICJ0aHJlYWQiLCAiKysrIGxvY2tGb3JXcml0ZTogd2FpdGluZ1xuIik7Ci0gICAgICAgIG1Xcml0ZVdhaXRlci53YWl0KG1Mb2NrKTsKLSAgICB9Ci0gICAgYXNzZXJ0KG1OdW1SZWFkZXJzID09IDApOwotICAgIGFzc2VydChtTnVtV3JpdGVycyA9PSAwKTsKLSAgICBtTnVtV3JpdGVycysrOwotI2lmIGRlZmluZWQoUFJJTlRfUkVOREVSX1RJTUVTKQotICAgIG1EZWJ1Z1RpbWVyLnN0YXJ0KCk7Ci0jZW5kaWYKLSAgICBtTG9jay51bmxvY2soKTsKLX0KLQotLyoKLSAqIFRyeSB0byBhZGQgYSB3cml0ZXIuICBJZiBpdCBkb2Vzbid0IHdvcmsgcmlnaHQgYXdheSwgcmV0dXJuICJmYWxzZSIuCi0gKi8KLWJvb2wgUmVhZFdyaXRlTG9jazo6dHJ5TG9ja0ZvcldyaXRlKCkKLXsKLSAgICBtTG9jay5sb2NrKCk7Ci0gICAgaWYgKG1OdW1SZWFkZXJzID4gMCB8fCBtTnVtV3JpdGVycyA+IDApIHsKLSAgICAgICAgbUxvY2sudW5sb2NrKCk7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgYXNzZXJ0KG1OdW1SZWFkZXJzID09IDApOwotICAgIGFzc2VydChtTnVtV3JpdGVycyA9PSAwKTsKLSAgICBtTnVtV3JpdGVycysrOwotI2lmIGRlZmluZWQoUFJJTlRfUkVOREVSX1RJTUVTKQotICAgIG1EZWJ1Z1RpbWVyLnN0YXJ0KCk7Ci0jZW5kaWYKLSAgICBtTG9jay51bmxvY2soKTsKLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotLyoKLSAqIFJlbW92ZSBhIHdyaXRlci4KLSAqLwotdm9pZCBSZWFkV3JpdGVMb2NrOjp1bmxvY2tGb3JXcml0ZSgpCi17Ci0gICAgbUxvY2subG9jaygpOwotICAgIGlmIChtTnVtV3JpdGVycyA9PSAwKSB7Ci0gICAgICAgIExPRyhMT0dfV0FSTiwgInRocmVhZCIsCi0gICAgICAgICAgICAiV0FSTklORzogdW5sb2NrRm9yV3JpdGUgcmVxdWVzdGVkLCBidXQgbm90IGxvY2tlZFxuIik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgYXNzZXJ0KG1OdW1Xcml0ZXJzID09IDEpOwotICAgIG1OdW1Xcml0ZXJzLS07Ci0jaWYgZGVmaW5lZChQUklOVF9SRU5ERVJfVElNRVMpCi0gICAgbURlYnVnVGltZXIuc3RvcCgpOwotICAgIC8vcHJpbnRmKCIgd3JsayBoZWxkICUuM2YgbXNlY1xuIiwKLSAgICAvLyAgICAoZG91YmxlKSBtRGVidWdUaW1lci5kdXJhdGlvblVzZWNzKCkgLyAxMDAwLjApOwotI2VuZGlmCi0gICAgLy8gbVdyaXRlV2FpdGVyLnNpZ25hbCgpOyAgICAgICAvLyBzaG91bGQgb3RoZXIgd3JpdGVycyBnZXQgZmlyc3QgZGlicz8KLSAgICAvL3ByaW50ZigiKysrIHNpZ25hbGluZyByZWFkZXJzIChpZiBhbnkpXG4iKTsKLSAgICBtUmVhZFdhaXRlci5icm9hZGNhc3QoKTsgICAgICAgIC8vIHdha2UgYWxsIHJlYWRlcnMgKGlmIGFueSkKLSAgICBtTG9jay51bmxvY2soKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBUaHJlYWQ6OlRocmVhZAotI2VuZGlmCi0KLS8qCi0gKiBUaGlzIGlzIG91ciB0aHJlYWQgb2JqZWN0IQotICovCi0KLVRocmVhZDo6VGhyZWFkKGJvb2wgY2FuQ2FsbEphdmEpCi0gICAgOiAgIG1DYW5DYWxsSmF2YShjYW5DYWxsSmF2YSksCi0gICAgICAgIG1UaHJlYWQodGhyZWFkX2lkX3QoLTEpKSwKLSAgICAgICAgbUxvY2soIlRocmVhZDo6bUxvY2siKSwKLSAgICAgICAgbVN0YXR1cyhOT19FUlJPUiksCi0gICAgICAgIG1FeGl0UGVuZGluZyhmYWxzZSksIG1SdW5uaW5nKGZhbHNlKQotewotfQotCi1UaHJlYWQ6On5UaHJlYWQoKQotewotfQotCi1zdGF0dXNfdCBUaHJlYWQ6OnJlYWR5VG9SdW4oKQotewotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgVGhyZWFkOjpydW4oY29uc3QgY2hhciogbmFtZSwgaW50MzJfdCBwcmlvcml0eSwgc2l6ZV90IHN0YWNrKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0KLSAgICBpZiAobVJ1bm5pbmcpIHsKLSAgICAgICAgLy8gdGhyZWFkIGFscmVhZHkgc3RhcnRlZAotICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgfQotCi0gICAgLy8gcmVzZXQgc3RhdHVzIGFuZCBleGl0UGVuZGluZyB0byB0aGVpciBkZWZhdWx0IHZhbHVlLCBzbyB3ZSBjYW4KLSAgICAvLyB0cnkgYWdhaW4gYWZ0ZXIgYW4gZXJyb3IgaGFwcGVuZWQgKGVpdGhlciBiZWxvdywgb3IgaW4gcmVhZHlUb1J1bigpKQotICAgIG1TdGF0dXMgPSBOT19FUlJPUjsKLSAgICBtRXhpdFBlbmRpbmcgPSBmYWxzZTsKLSAgICBtVGhyZWFkID0gdGhyZWFkX2lkX3QoLTEpOwotICAgIAotICAgIC8vIGhvbGQgYSBzdHJvbmcgcmVmZXJlbmNlIG9uIG91cnNlbGYKLSAgICBtSG9sZFNlbGYgPSB0aGlzOwotCi0gICAgYm9vbCByZXM7Ci0gICAgaWYgKG1DYW5DYWxsSmF2YSkgewotICAgICAgICByZXMgPSBjcmVhdGVUaHJlYWRFdGMoX3RocmVhZExvb3AsCi0gICAgICAgICAgICAgICAgdGhpcywgbmFtZSwgcHJpb3JpdHksIHN0YWNrLCAmbVRocmVhZCk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgcmVzID0gYW5kcm9pZENyZWF0ZVJhd1RocmVhZEV0YyhfdGhyZWFkTG9vcCwKLSAgICAgICAgICAgICAgICB0aGlzLCBuYW1lLCBwcmlvcml0eSwgc3RhY2ssICZtVGhyZWFkKTsKLSAgICB9Ci0gICAgCi0gICAgaWYgKHJlcyA9PSBmYWxzZSkgewotICAgICAgICBtU3RhdHVzID0gVU5LTk9XTl9FUlJPUjsgICAvLyBzb21ldGhpbmcgaGFwcGVuZWQhCi0gICAgICAgIG1SdW5uaW5nID0gZmFsc2U7Ci0gICAgICAgIG1UaHJlYWQgPSB0aHJlYWRfaWRfdCgtMSk7Ci0gICAgfQotICAgIAotICAgIGlmIChtU3RhdHVzIDwgMCkgewotICAgICAgICAvLyBzb21ldGhpbmcgaGFwcGVuZWQsIGRvbid0IGxlYWsKLSAgICAgICAgbUhvbGRTZWxmLmNsZWFyKCk7Ci0gICAgfQotICAgIAotICAgIHJldHVybiBtU3RhdHVzOwotfQotCi1pbnQgVGhyZWFkOjpfdGhyZWFkTG9vcCh2b2lkKiB1c2VyKQotewotICAgIFRocmVhZCogY29uc3Qgc2VsZiA9IHN0YXRpY19jYXN0PFRocmVhZCo+KHVzZXIpOwotICAgIHNwPFRocmVhZD4gc3Ryb25nKHNlbGYtPm1Ib2xkU2VsZik7Ci0gICAgd3A8VGhyZWFkPiB3ZWFrKHN0cm9uZyk7Ci0gICAgc2VsZi0+bUhvbGRTZWxmLmNsZWFyKCk7Ci0KLSAgICAvLyB3ZSdyZSBhYm91dCB0byBydW4uLi4KLSAgICBzZWxmLT5tU3RhdHVzID0gc2VsZi0+cmVhZHlUb1J1bigpOwotICAgIGlmIChzZWxmLT5tU3RhdHVzIT1OT19FUlJPUiB8fCBzZWxmLT5tRXhpdFBlbmRpbmcpIHsKLSAgICAgICAgLy8gcHJldGVuZCB0aGUgdGhyZWFkIG5ldmVyIHN0YXJ0ZWQuLi4KLSAgICAgICAgc2VsZi0+bUV4aXRQZW5kaW5nID0gZmFsc2U7Ci0gICAgICAgIHNlbGYtPm1SdW5uaW5nID0gZmFsc2U7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLSAgICAKLSAgICAvLyB0aHJlYWQgaXMgcnVubmluZyBub3cKLSAgICBzZWxmLT5tUnVubmluZyA9IHRydWU7Ci0KLSAgICBkbyB7Ci0gICAgICAgIGJvb2wgcmVzdWx0ID0gc2VsZi0+dGhyZWFkTG9vcCgpOwotICAgICAgICBpZiAocmVzdWx0ID09IGZhbHNlIHx8IHNlbGYtPm1FeGl0UGVuZGluZykgewotICAgICAgICAgICAgc2VsZi0+bUV4aXRQZW5kaW5nID0gdHJ1ZTsKLSAgICAgICAgICAgIHNlbGYtPm1Mb2NrLmxvY2soKTsKLSAgICAgICAgICAgIHNlbGYtPm1SdW5uaW5nID0gZmFsc2U7Ci0gICAgICAgICAgICBzZWxmLT5tVGhyZWFkRXhpdGVkQ29uZGl0aW9uLnNpZ25hbCgpOwotICAgICAgICAgICAgc2VsZi0+bUxvY2sudW5sb2NrKCk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgLy8gUmVsZWFzZSBvdXIgc3Ryb25nIHJlZmVyZW5jZSwgdG8gbGV0IGEgY2hhbmNlIHRvIHRoZSB0aHJlYWQKLSAgICAgICAgLy8gdG8gZGllIGEgcGVhY2VmdWwgZGVhdGguCi0gICAgICAgIHN0cm9uZy5jbGVhcigpOwotICAgICAgICAvLyBBbmQgaW1tZWRpYXRlbHksIHJlYWNxdWlyZSBhIHN0cm9uZyByZWZlcmVuY2UgZm9yIHRoZSBuZXh0IGxvb3AKLSAgICAgICAgc3Ryb25nID0gd2Vhay5wcm9tb3RlKCk7Ci0gICAgfSB3aGlsZShzdHJvbmcgIT0gMCk7Ci0gICAgCi0gICAgcmV0dXJuIDA7Ci19Ci0KLXZvaWQgVGhyZWFkOjpyZXF1ZXN0RXhpdCgpCi17Ci0gICAgbUV4aXRQZW5kaW5nID0gdHJ1ZTsKLX0KLQotc3RhdHVzX3QgVGhyZWFkOjpyZXF1ZXN0RXhpdEFuZFdhaXQoKQotewotICAgIGlmIChtU3RhdHVzID09IE9LKSB7Ci0KLSAgICAgICAgaWYgKG1UaHJlYWQgPT0gZ2V0VGhyZWFkSWQoKSkgewotICAgICAgICAgICAgTE9HVygKLSAgICAgICAgICAgICJUaHJlYWQgKHRoaXM9JXApOiBkb24ndCBjYWxsIHdhaXRGb3JFeGl0KCkgZnJvbSB0aGlzICIKLSAgICAgICAgICAgICJUaHJlYWQgb2JqZWN0J3MgdGhyZWFkLiBJdCdzIGEgZ3VhcmFudGVlZCBkZWFkbG9jayEiLAotICAgICAgICAgICAgdGhpcyk7Ci0gICAgICAgICAgICByZXR1cm4gV09VTERfQkxPQ0s7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIHJlcXVlc3RFeGl0KCk7Ci0KLSAgICAgICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICAgICAgd2hpbGUgKG1SdW5uaW5nID09IHRydWUpIHsKLSAgICAgICAgICAgIG1UaHJlYWRFeGl0ZWRDb25kaXRpb24ud2FpdChtTG9jayk7Ci0gICAgICAgIH0KLSAgICAgICAgbUV4aXRQZW5kaW5nID0gZmFsc2U7Ci0gICAgfQotICAgIHJldHVybiBtU3RhdHVzOwotfQotCi1ib29sIFRocmVhZDo6ZXhpdFBlbmRpbmcoKSBjb25zdAotewotICAgIHJldHVybiBtRXhpdFBlbmRpbmc7Ci19Ci0KLQotCi19OyAgLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvVGltZXJQcm9iZS5jcHAgYi9saWJzL3V0aWxzL1RpbWVyUHJvYmUuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4MzU0ODBkLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvVGltZXJQcm9iZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxMzEgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDx1dGlscy9UaW1lclByb2JlLmg+Ci0gCi0jaWYgRU5BQkxFX1RJTUVSX1BST0JFCi0KLSNpZmRlZiBMT0dfVEFHCi0jdW5kZWYgTE9HX1RBRwotI2VuZGlmCi0jZGVmaW5lIExPR19UQUcgInRpbWUiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotVmVjdG9yPFRpbWVyUHJvYmU6OkJ1Y2tldD4gVGltZXJQcm9iZTo6Z0J1Y2tldHM7Ci1UaW1lclByb2JlKiBUaW1lclByb2JlOjpnRXhlY3V0ZUNoYWluOwotaW50IFRpbWVyUHJvYmU6OmdJbmRlbnQ7Ci10aW1lc3BlYyBUaW1lclByb2JlOjpnUmVhbEJhc2U7Ci0KLVRpbWVyUHJvYmU6OlRpbWVyUHJvYmUoY29uc3QgY2hhciB0YWdbXSwgaW50KiBzbG90KSA6IG1UYWcodGFnKQotewotICAgIG1OZXh0ID0gZ0V4ZWN1dGVDaGFpbjsKLSAgICBnRXhlY3V0ZUNoYWluID0gdGhpczsKLSAgICBtSW5kZW50ID0gZ0luZGVudDsKLSAgICBnSW5kZW50ICs9IDE7Ci0gICAgaWYgKG1JbmRlbnQgPiAwKSB7Ci0gICAgICAgIGlmICgqc2xvdCA9PSAwKSB7Ci0gICAgICAgICAgICBpbnQgY291bnQgPSBnQnVja2V0cy5hZGQoKTsKLSAgICAgICAgICAgICpzbG90ID0gY291bnQ7Ci0gICAgICAgICAgICBCdWNrZXQmIGJ1Y2tldCA9IGdCdWNrZXRzLmVkaXRJdGVtQXQoY291bnQpOwotICAgICAgICAgICAgbWVtc2V0KCZidWNrZXQsIDAsIHNpemVvZihCdWNrZXQpKTsKLSAgICAgICAgICAgIGJ1Y2tldC5tVGFnID0gdGFnOwotICAgICAgICAgICAgYnVja2V0Lm1TbG90UHRyID0gc2xvdDsKLSAgICAgICAgICAgIGJ1Y2tldC5tSW5kZW50ID0gbUluZGVudDsKLSAgICAgICAgfQotICAgICAgICBtQnVja2V0ID0gKnNsb3Q7Ci0gICAgfQotICAgIGNsb2NrX2dldHRpbWUoQ0xPQ0tfUkVBTFRJTUUsICZtUmVhbFN0YXJ0KTsKLSAgICBpZiAoZ1JlYWxCYXNlLnR2X3NlYyA9PSAwKQotICAgICAgICBnUmVhbEJhc2UgPSBtUmVhbFN0YXJ0OwotICAgIGNsb2NrX2dldHRpbWUoQ0xPQ0tfUFJPQ0VTU19DUFVUSU1FX0lELCAmbVBTdGFydCk7Ci0gICAgY2xvY2tfZ2V0dGltZShDTE9DS19USFJFQURfQ1BVVElNRV9JRCwgJm1UU3RhcnQpOwotfQotCi12b2lkIFRpbWVyUHJvYmU6OmVuZCgpCi17Ci0gICAgdGltZXNwZWMgcmVhbEVuZCwgcEVuZCwgdEVuZDsKLSAgICBjbG9ja19nZXR0aW1lKENMT0NLX1JFQUxUSU1FLCAmcmVhbEVuZCk7Ci0gICAgY2xvY2tfZ2V0dGltZShDTE9DS19QUk9DRVNTX0NQVVRJTUVfSUQsICZwRW5kKTsKLSAgICBjbG9ja19nZXR0aW1lKENMT0NLX1RIUkVBRF9DUFVUSU1FX0lELCAmdEVuZCk7Ci0gICAgcHJpbnQocmVhbEVuZCwgcEVuZCwgdEVuZCk7Ci0gICAgbVRhZyA9IE5VTEw7Ci19Ci0KLVRpbWVyUHJvYmU6On5UaW1lclByb2JlKCkKLXsKLSAgICBpZiAobVRhZyAhPSBOVUxMKQotICAgICAgICBlbmQoKTsKLSAgICBnRXhlY3V0ZUNoYWluID0gbU5leHQ7Ci0gICAgZ0luZGVudC0tOwotfQotCi0KLXVpbnQzMl90IFRpbWVyUHJvYmU6OkVsYXBzZWRUaW1lKGNvbnN0IHRpbWVzcGVjJiBzdGFydCwgY29uc3QgdGltZXNwZWMmIGVuZCkKLXsKLSAgICBpbnQgc2VjID0gZW5kLnR2X3NlYyAtIHN0YXJ0LnR2X3NlYzsKLSAgICBpbnQgbnNlYyA9IGVuZC50dl9uc2VjIC0gc3RhcnQudHZfbnNlYzsKLSAgICBpZiAobnNlYyA8IDApIHsKLSAgICAgICAgc2VjLS07Ci0gICAgICAgIG5zZWMgKz0gMTAwMDAwMDAwMDsKLSAgICB9Ci0gICAgcmV0dXJuIHNlYyAqIDEwMDAwMDAgKyBuc2VjIC8gMTAwMDsKLX0KLQotdm9pZCBUaW1lclByb2JlOjpwcmludChjb25zdCB0aW1lc3BlYyYgciwgY29uc3QgdGltZXNwZWMmIHAsCi0gICAgICAgIGNvbnN0IHRpbWVzcGVjJiB0KSBjb25zdAotewotICAgIHVpbnQzMl90IGVzID0gRWxhcHNlZFRpbWUoZ1JlYWxCYXNlLCBtUmVhbFN0YXJ0KTsKLSAgICB1aW50MzJfdCBlciA9IEVsYXBzZWRUaW1lKG1SZWFsU3RhcnQsIHIpOwotICAgIHVpbnQzMl90IGVwID0gRWxhcHNlZFRpbWUobVBTdGFydCwgcCk7Ci0gICAgdWludDMyX3QgZXQgPSBFbGFwc2VkVGltZShtVFN0YXJ0LCB0KTsKLSAgICBpZiAobUluZGVudCA+IDApIHsKLSAgICAgICAgQnVja2V0JiBidWNrZXQgPSBnQnVja2V0cy5lZGl0SXRlbUF0KG1CdWNrZXQpOwotICAgICAgICBpZiAoYnVja2V0Lm1TdGFydCA9PSAwKQotICAgICAgICAgICAgYnVja2V0Lm1TdGFydCA9IGVzOwotICAgICAgICBidWNrZXQubVJlYWwgKz0gZXI7Ci0gICAgICAgIGJ1Y2tldC5tUHJvY2VzcyArPSBlcDsKLSAgICAgICAgYnVja2V0Lm1UaHJlYWQgKz0gZXQ7Ci0gICAgICAgIGJ1Y2tldC5tQ291bnQrKzsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBpbnQgaW5kZXggPSAwOwotICAgIGludCBidWNrZXRzID0gZ0J1Y2tldHMuc2l6ZSgpOwotICAgIGludCBjb3VudCA9IDE7Ci0gICAgY29uc3QgY2hhciogdGFnID0gbVRhZzsKLSAgICBpbnQgaW5kZW50ID0gbUluZGVudDsKLSAgICBkbyB7Ci0gICAgICAgIExPR0QoIiUtMzAuMzBzOiAoJTNkKSAlLTUuKnMgdGltZT0lLTEwLjNmIHJlYWw9JTdkdXMgcHJvY2Vzcz0lN2R1cyAoJTNkJSUpIHRocmVhZD0lN2R1cyAoJTNkJSUpXG4iLCAKLSAgICAgICAgICAgIHRhZywgY291bnQsIGluZGVudCA+IDUgPyA1IDogaW5kZW50LCAiKysrKysiLCBlcyAvIDEwMDAwMDAuMCwKLSAgICAgICAgICAgIGVyLCBlcCwgZXAgKiAxMDAgLyBlciwgZXQsIGV0ICogMTAwIC8gZXIpOwotICAgICAgICBpZiAoaW5kZXggPj0gYnVja2V0cykKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBCdWNrZXQmIGJ1Y2tldCA9IGdCdWNrZXRzLmVkaXRJdGVtQXQoaW5kZXgpOwotICAgICAgICBjb3VudCA9IGJ1Y2tldC5tQ291bnQ7Ci0gICAgICAgIGVzID0gYnVja2V0Lm1TdGFydDsKLSAgICAgICAgZXIgPSBidWNrZXQubVJlYWw7Ci0gICAgICAgIGVwID0gYnVja2V0Lm1Qcm9jZXNzOwotICAgICAgICBldCA9IGJ1Y2tldC5tVGhyZWFkOwotICAgICAgICB0YWcgPSBidWNrZXQubVRhZzsKLSAgICAgICAgaW5kZW50ID0gYnVja2V0Lm1JbmRlbnQ7Ci0gICAgICAgICpidWNrZXQubVNsb3RQdHIgPSAwOwotICAgIH0gd2hpbGUgKCsraW5kZXgpOyAvLyBhbHdheXMgdHJ1ZQotICAgIGdCdWNrZXRzLmNsZWFyKCk7Ci19Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZgpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9UaW1lcnMuY3BwIGIvbGlicy91dGlscy9UaW1lcnMuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyYWJjODExLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvVGltZXJzLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDI0MCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIFRpbWVyIGZ1bmN0aW9ucy4KLS8vCi0jaW5jbHVkZSA8dXRpbHMvVGltZXJzLmg+Ci0jaW5jbHVkZSA8dXRpbHMvcG9ydGVkLmg+ICAgICAvLyBtYXkgbmVlZCB1c2xlZXAKLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8c3lzL3RpbWUuaD4KLSNpbmNsdWRlIDx0aW1lLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLQotI2lmZGVmIEhBVkVfV0lOMzJfVEhSRUFEUwotI2luY2x1ZGUgPHdpbmRvd3MuaD4KLSNlbmRpZgotCi1uc2Vjc190IHN5c3RlbVRpbWUoaW50IGNsb2NrKQotewotI2lmIGRlZmluZWQoSEFWRV9QT1NJWF9DTE9DS1MpCi0gICAgc3RhdGljIGNvbnN0IGNsb2NraWRfdCBjbG9ja3NbXSA9IHsKLSAgICAgICAgICAgIENMT0NLX1JFQUxUSU1FLAotICAgICAgICAgICAgQ0xPQ0tfTU9OT1RPTklDLAotICAgICAgICAgICAgQ0xPQ0tfUFJPQ0VTU19DUFVUSU1FX0lELAotICAgICAgICAgICAgQ0xPQ0tfVEhSRUFEX0NQVVRJTUVfSUQKLSAgICB9OwotICAgIHN0cnVjdCB0aW1lc3BlYyB0OwotICAgIHQudHZfc2VjID0gdC50dl9uc2VjID0gMDsKLSAgICBjbG9ja19nZXR0aW1lKGNsb2Nrc1tjbG9ja10sICZ0KTsKLSAgICByZXR1cm4gbnNlY3NfdCh0LnR2X3NlYykqMTAwMDAwMDAwMExMICsgdC50dl9uc2VjOwotI2Vsc2UKLSAgICAvLyB3ZSBkb24ndCBzdXBwb3J0IHRoZSBjbG9ja3MgaGVyZS4KLSAgICBzdHJ1Y3QgdGltZXZhbCB0OwotICAgIHQudHZfc2VjID0gdC50dl91c2VjID0gMDsKLSAgICBnZXR0aW1lb2ZkYXkoJnQsIE5VTEwpOwotICAgIHJldHVybiBuc2Vjc190KHQudHZfc2VjKSoxMDAwMDAwMDAwTEwgKyBuc2Vjc190KHQudHZfdXNlYykqMTAwMExMOwotI2VuZGlmCi19Ci0KLS8vI2RlZmluZSBNT05JVE9SX1VTTEVFUAotCi0vKgotICogU2xlZXAgbG9uZyBlbm91Z2ggdGhhdCB3ZSdsbCB3YWtlIHVwICJpbnRlcnZhbCIgbWlsbGlzZWNvbmRzIGFmdGVyCi0gKiB0aGUgcHJldmlvdXMgc25vb3plLgotICoKLSAqIFRoZSAibmV4dFRpY2siIGFyZ3VtZW50IGlzIHVwZGF0ZWQgb24gZWFjaCBjYWxsLCBhbmQgc2hvdWxkIGJlIHBhc3NlZAotICogaW4gZXZlcnkgdGltZS4gIFNldCBpdHMgZmllbGRzIHRvIHplcm8gb24gdGhlIGZpcnN0IGNhbGwuCi0gKgotICogUmV0dXJucyB0aGUgI29mIGludGVydmFscyB3ZSBoYXZlIG92ZXJzbGVwdCwgd2hpY2ggd2lsbCBiZSB6ZXJvIGlmIHdlJ3JlCi0gKiBvbiB0aW1lLiAgW0N1cnJlbnRseSBqdXN0IHJldHVybnMgMCBvciAxLl0KLSAqLwotaW50IHNsZWVwRm9ySW50ZXJ2YWwobG9uZyBpbnRlcnZhbCwgc3RydWN0IHRpbWV2YWwqIHBOZXh0VGljaykKLXsKLSAgICBzdHJ1Y3QgdGltZXZhbCBub3c7Ci0gICAgbG9uZyBsb25nIHRpbWVCZWZvcmVOZXh0OwotICAgIGxvbmcgc2xlZXBUaW1lID0gMDsKLSAgICBib29sIG92ZXJTbGVwdCA9IGZhbHNlOwotICAgIC8vaW50IHVzbGVlcEJpYXMgPSAwOwotCi0jaWZkZWYgVVNMRUVQX0JJQVMKLSAgICAvKgotICAgICAqIExpbnV4IGxpa2VzIHRvIGFkZCA5MDAwbXMgb3Igc28uCi0gICAgICogW25vdCB1c2luZyB0aGlzIGZvciBub3ddCi0gICAgICovCi0gICAgLy91c2xlZXBCaWFzID0gVVNMRUVQX0JJQVM7Ci0jZW5kaWYKLQotICAgIGdldHRpbWVvZmRheSgmbm93LCBOVUxMKTsKLQotICAgIGlmIChwTmV4dFRpY2stPnR2X3NlYyA9PSAwKSB7Ci0gICAgICAgIC8qIHNwZWNpYWwtY2FzZSBmb3IgZmlyc3QgdGltZSB0aHJvdWdoICovCi0gICAgICAgICpwTmV4dFRpY2sgPSBub3c7Ci0gICAgICAgIHNsZWVwVGltZSA9IGludGVydmFsOwotICAgICAgICBhbmRyb2lkOjpEdXJhdGlvblRpbWVyOjphZGRUb1RpbWV2YWwocE5leHRUaWNrLCBpbnRlcnZhbCk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgLyoKLSAgICAgICAgICogQ29tcHV0ZSBob3cgbXVjaCB0aW1lIHRoZXJlIGlzIGJlZm9yZSB0aGUgbmV4dCB0aWNrLiAgSWYgdGhpcwotICAgICAgICAgKiB2YWx1ZSBpcyBuZWdhdGl2ZSwgd2UndmUgcnVuIG92ZXIuICBJZiB3ZSd2ZSBydW4gb3ZlciBhIGxpdHRsZQotICAgICAgICAgKiBiaXQgd2UgY2FuIHNob3J0ZW4gdGhlIG5leHQgZnJhbWUgdG8ga2VlcCB0aGUgcGFjZSBzdGVhZHksIGJ1dAotICAgICAgICAgKiBpZiB3ZSd2ZSBkcmFtYXRpY2FsbHkgb3ZlcnNob3Qgd2UgbmVlZCB0byByZS1zeW5jLgotICAgICAgICAgKi8KLSAgICAgICAgdGltZUJlZm9yZU5leHQgPSBhbmRyb2lkOjpEdXJhdGlvblRpbWVyOjpzdWJ0cmFjdFRpbWV2YWxzKHBOZXh0VGljaywgJm5vdyk7Ci0gICAgICAgIC8vcHJpbnRmKCJUT1A6IG5vdz0lbGQuJWxkIG5leHQ9JWxkLiVsZCBkaWZmPSVsZFxuIiwKLSAgICAgICAgLy8gICAgbm93LnR2X3NlYywgbm93LnR2X3VzZWMsIHBOZXh0VGljay0+dHZfc2VjLCBwTmV4dFRpY2stPnR2X3VzZWMsCi0gICAgICAgIC8vICAgIChsb25nKSB0aW1lQmVmb3JlTmV4dCk7Ci0gICAgICAgIGlmICh0aW1lQmVmb3JlTmV4dCA8IC1pbnRlcnZhbCkgewotICAgICAgICAgICAgLyogd2F5IG92ZXIgKi8KLSAgICAgICAgICAgIG92ZXJTbGVwdCA9IHRydWU7Ci0gICAgICAgICAgICBzbGVlcFRpbWUgPSAwOwotICAgICAgICAgICAgKnBOZXh0VGljayA9IG5vdzsKLSAgICAgICAgfSBlbHNlIGlmICh0aW1lQmVmb3JlTmV4dCA8PSAwKSB7Ci0gICAgICAgICAgICAvKiBzbGlnaHRseSBvdmVyLCBrZWVwIHRoZSBwYWNlIHN0ZWFkeSAqLwotICAgICAgICAgICAgb3ZlclNsZXB0ID0gdHJ1ZTsKLSAgICAgICAgICAgIHNsZWVwVGltZSA9IDA7Ci0gICAgICAgIH0gZWxzZSBpZiAodGltZUJlZm9yZU5leHQgPD0gaW50ZXJ2YWwpIHsKLSAgICAgICAgICAgIC8qIHJpZ2h0IG9uIHNjaGVkdWxlICovCi0gICAgICAgICAgICBzbGVlcFRpbWUgPSB0aW1lQmVmb3JlTmV4dDsKLSAgICAgICAgfSBlbHNlIGlmICh0aW1lQmVmb3JlTmV4dCA+IGludGVydmFsICYmIHRpbWVCZWZvcmVOZXh0IDw9IDIqaW50ZXJ2YWwpIHsKLSAgICAgICAgICAgIC8qIHNsZWVwIGNhbGwgcmV0dXJuZWQgZWFybHk7IGRvIGEgbG9uZ2VyIHNsZWVwIHRoaXMgdGltZSAqLwotICAgICAgICAgICAgc2xlZXBUaW1lID0gdGltZUJlZm9yZU5leHQ7Ci0gICAgICAgIH0gZWxzZSBpZiAodGltZUJlZm9yZU5leHQgPiBpbnRlcnZhbCkgewotICAgICAgICAgICAgLyogd2Ugd2VudCBiYWNrIGluIHRpbWUgLS0gc29tZWJvZHkgdXBkYXRlZCBzeXN0ZW0gY2xvY2s/ICovCi0gICAgICAgICAgICAvKiAoY291bGQgYWxzbyBiZSBhICpzZXJpb3VzbHkqIGJyb2tlbiB1c2xlZXAoKSkgKi8KLSAgICAgICAgICAgIExPRyhMT0dfREVCVUcsICIiLAotICAgICAgICAgICAgICAgICIgSW1wb3NzaWJsZTogdGltZUJlZm9yZU5leHQgPSAlbGRcbiIsIChsb25nKXRpbWVCZWZvcmVOZXh0KTsKLSAgICAgICAgICAgIHNsZWVwVGltZSA9IDA7Ci0gICAgICAgICAgICAqcE5leHRUaWNrID0gbm93OwotICAgICAgICB9Ci0gICAgICAgIGFuZHJvaWQ6OkR1cmF0aW9uVGltZXI6OmFkZFRvVGltZXZhbChwTmV4dFRpY2ssIGludGVydmFsKTsKLSAgICB9Ci0gICAgLy9wcmludGYoIiBCZWZvcmUgc2xlZXA6IG5vdz0lbGQuJWxkIG5leHQ9JWxkLiVsZCBzbGVlcFRpbWU9JWxkXG4iLAotICAgIC8vICAgIG5vdy50dl9zZWMsIG5vdy50dl91c2VjLCBwTmV4dFRpY2stPnR2X3NlYywgcE5leHRUaWNrLT50dl91c2VjLAotICAgIC8vICAgIHNsZWVwVGltZSk7Ci0KLSAgICAvKgotICAgICAqIFNsZWVwIGZvciB0aGUgZGVzaWduYXRlZCBwZXJpb2Qgb2YgdGltZS4KLSAgICAgKgotICAgICAqIExpbnV4IHRlbmRzIHRvIHNsZWVwIGZvciBsb25nZXIgdGhhbiByZXF1ZXN0ZWQsIG9mdGVuIGJ5IDE3LTE4bXMuCi0gICAgICogTWluR1cgdGVuZHMgdG8gc2xlZXAgZm9yIGxlc3MgdGhhbiByZXF1ZXN0ZWQsIGJ5IGFzIG11Y2ggYXMgMTRtcywKLSAgICAgKiBidXQgb2NjYXNpb25hbGx5IG92ZXJzbGVlcHMgZm9yIDQwK21zIChsb29rcyBsaWtlIHNvbWUgZXh0ZXJuYWwKLSAgICAgKiBmYWN0b3JzIHBsdXMgcm91bmQtb2ZmIG9uIGEgNjRIeiBjbG9jaykuICBDeWd3aW4gaXMgcHJldHR5IHN0ZWFkeS4KLSAgICAgKgotICAgICAqIElmIHlvdSBzdGFydCB0aGUgTWluR1cgdmVyc2lvbiwgYW5kIHRoZW4gbGF1bmNoIHRoZSBDeWd3aW4gdmVyc2lvbiwKLSAgICAgKiB0aGUgTWluR1cgY2xvY2sgYmVjb21lcyBtb3JlIGVycmF0aWMuICBOb3QgZW50aXJlbHkgc3VyZSB3aHkuCi0gICAgICoKLSAgICAgKiAoVGhlcmUncyBhIGxvdCBvZiBzdHVmZiBoZXJlOyBpdCdzIHJlYWxseSBqdXN0IGEgdXNsZWVwKCkgY2FsbCB3aXRoCi0gICAgICogYSBidW5jaCBvZiBpbnN0cnVtZW50YXRpb24uKQotICAgICAqLwotICAgIGlmIChzbGVlcFRpbWUgPiAwKSB7Ci0jaWYgZGVmaW5lZChNT05JVE9SX1VTTEVFUCkKLSAgICAgICAgc3RydWN0IHRpbWV2YWwgYmVmb3JlLCBhZnRlcjsKLSAgICAgICAgbG9uZyBsb25nIGFjdHVhbDsKLQotICAgICAgICBnZXR0aW1lb2ZkYXkoJmJlZm9yZSwgTlVMTCk7Ci0gICAgICAgIHVzbGVlcCgobG9uZykgc2xlZXBUaW1lKTsKLSAgICAgICAgZ2V0dGltZW9mZGF5KCZhZnRlciwgTlVMTCk7Ci0KLSAgICAgICAgLyogY2hlY2sgdXNsZWVwKCkgYWNjdXJhY3k7IGRlZmF1bHQgTGludXggdGhyZWFkcyBhcmUgcHJldHR5IHNsb3BweSAqLwotICAgICAgICBhY3R1YWwgPSBhbmRyb2lkOjpEdXJhdGlvblRpbWVyOjpzdWJ0cmFjdFRpbWV2YWxzKCZhZnRlciwgJmJlZm9yZSk7Ci0gICAgICAgIGlmICgobG9uZykgYWN0dWFsIDwgc2xlZXBUaW1lIC0gMTQwMDAgLyooc2xlZXBUaW1lLzEwKSovIHx8Ci0gICAgICAgICAgICAobG9uZykgYWN0dWFsID4gc2xlZXBUaW1lICsgMjAwMDAgLyooc2xlZXBUaW1lLzEwKSovKQotICAgICAgICB7Ci0gICAgICAgICAgICBMT0coTE9HX0RFQlVHLCAiIiwgIiBPZGQgdXNsZWVwOiByZXE9JWxkLCBhY3R1YWw9JWxkXG4iLCBzbGVlcFRpbWUsCi0gICAgICAgICAgICAgICAgKGxvbmcpIGFjdHVhbCk7Ci0gICAgICAgIH0KLSNlbHNlCi0jaWZkZWYgSEFWRV9XSU4zMl9USFJFQURTCi0gICAgICAgIFNsZWVwKCBzbGVlcFRpbWUvMTAwMCApOwotI2Vsc2UgICAgICAgIAotICAgICAgICB1c2xlZXAoKGxvbmcpIHNsZWVwVGltZSk7Ci0jZW5kaWYgICAgICAgIAotI2VuZGlmCi0gICAgfQotCi0gICAgLy9wcmludGYoInNsZXB0ICVkXG4iLCBzbGVlcFRpbWUpOwotCi0gICAgaWYgKG92ZXJTbGVwdCkKLSAgICAgICAgcmV0dXJuIDE7ICAgICAgIC8vIGNsb3NlIGVub3VnaAotICAgIGVsc2UKLSAgICAgICAgcmV0dXJuIDA7Ci19Ci0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBEdXJhdGlvblRpbWVyCi0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqLwotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotLy8gU3RhcnQgdGhlIHRpbWVyLgotdm9pZCBEdXJhdGlvblRpbWVyOjpzdGFydCh2b2lkKQotewotICAgIGdldHRpbWVvZmRheSgmbVN0YXJ0V2hlbiwgTlVMTCk7Ci19Ci0KLS8vIFN0b3AgdGhlIHRpbWVyLgotdm9pZCBEdXJhdGlvblRpbWVyOjpzdG9wKHZvaWQpCi17Ci0gICAgZ2V0dGltZW9mZGF5KCZtU3RvcFdoZW4sIE5VTEwpOwotfQotCi0vLyBHZXQgdGhlIGR1cmF0aW9uIGluIG1pY3Jvc2Vjb25kcy4KLWxvbmcgbG9uZyBEdXJhdGlvblRpbWVyOjpkdXJhdGlvblVzZWNzKHZvaWQpIGNvbnN0Ci17Ci0gICAgcmV0dXJuIChsb25nKSBzdWJ0cmFjdFRpbWV2YWxzKCZtU3RvcFdoZW4sICZtU3RhcnRXaGVuKTsKLX0KLQotLy8gU3VidHJhY3QgdHdvIHRpbWV2YWxzLiAgUmV0dXJucyB0aGUgZGlmZmVyZW5jZSAocHR2MS1wdHYyKSBpbgotLy8gbWljcm9zZWNvbmRzLgotLypzdGF0aWMqLyBsb25nIGxvbmcgRHVyYXRpb25UaW1lcjo6c3VidHJhY3RUaW1ldmFscyhjb25zdCBzdHJ1Y3QgdGltZXZhbCogcHR2MSwKLSAgICBjb25zdCBzdHJ1Y3QgdGltZXZhbCogcHR2MikKLXsKLSAgICBsb25nIGxvbmcgc3RvcCAgPSAoKGxvbmcgbG9uZykgcHR2MS0+dHZfc2VjKSAqIDEwMDAwMDBMTCArCi0gICAgICAgICAgICAgICAgICAgICAgKChsb25nIGxvbmcpIHB0djEtPnR2X3VzZWMpOwotICAgIGxvbmcgbG9uZyBzdGFydCA9ICgobG9uZyBsb25nKSBwdHYyLT50dl9zZWMpICogMTAwMDAwMExMICsKLSAgICAgICAgICAgICAgICAgICAgICAoKGxvbmcgbG9uZykgcHR2Mi0+dHZfdXNlYyk7Ci0gICAgcmV0dXJuIHN0b3AgLSBzdGFydDsKLX0KLQotLy8gQWRkIHRoZSBzcGVjaWZpZWQgYW1vdW50IG9mIHRpbWUgdG8gdGhlIHRpbWV2YWwuCi0vKnN0YXRpYyovIHZvaWQgRHVyYXRpb25UaW1lcjo6YWRkVG9UaW1ldmFsKHN0cnVjdCB0aW1ldmFsKiBwdHYsIGxvbmcgdXNlYykKLXsKLSAgICBpZiAodXNlYyA8IDApIHsKLSAgICAgICAgTE9HKExPR19XQVJOLCAiIiwgIk5lZ2F0aXZlIHZhbHVlcyBub3Qgc3VwcG9ydGVkIGluIGFkZFRvVGltZXZhbFxuIik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICAvLyBub3JtYWxpemUgdHZfdXNlYyBpZiBuZWNlc3NhcnkKLSAgICBpZiAocHR2LT50dl91c2VjID49IDEwMDAwMDApIHsKLSAgICAgICAgcHR2LT50dl9zZWMgKz0gcHR2LT50dl91c2VjIC8gMTAwMDAwMDsKLSAgICAgICAgcHR2LT50dl91c2VjICU9IDEwMDAwMDA7Ci0gICAgfQotCi0gICAgcHR2LT50dl91c2VjICs9IHVzZWMgJSAxMDAwMDAwOwotICAgIGlmIChwdHYtPnR2X3VzZWMgPj0gMTAwMDAwMCkgewotICAgICAgICBwdHYtPnR2X3VzZWMgLT0gMTAwMDAwMDsKLSAgICAgICAgcHR2LT50dl9zZWMrKzsKLSAgICB9Ci0gICAgcHR2LT50dl9zZWMgKz0gdXNlYyAvIDEwMDAwMDA7Ci19Ci0KZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvVW5pY29kZS5jcHAgYi9saWJzL3V0aWxzL1VuaWNvZGUuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzM2Y1MzVmLi4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvVW5pY29kZS5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxOTMgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlICJ1dGlscy9BbmRyb2lkVW5pY29kZS5oIgotI2luY2x1ZGUgImNoYXJhY3RlckRhdGEuaCIKLQotI2RlZmluZSBMT0dfVEFHICJVbmljb2RlIgotI2luY2x1ZGUgInV0aWxzL0xvZy5oIgotCi0vLyBJQ1UgaGVhZGVycyBmb3IgdXNpbmcgbWFjcm9zCi0jaW5jbHVkZSA8dW5pY29kZS91dGYxNi5oPgotCi0jZGVmaW5lIE1JTl9SQURJWCAyCi0jZGVmaW5lIE1BWF9SQURJWCAzNgotCi0jZGVmaW5lIFRZUEVfU0hJRlQgMAotI2RlZmluZSBUWVBFX01BU0sgKCgxPDw1KS0xKQotCi0jZGVmaW5lIERJUkVDVElPTl9TSElGVCAoVFlQRV9TSElGVCs1KQotI2RlZmluZSBESVJFQ1RJT05fTUFTSyAoKDE8PDUpLTEpCi0KLSNkZWZpbmUgTUlSUk9SRURfU0hJRlQgKERJUkVDVElPTl9TSElGVCs1KQotI2RlZmluZSBNSVJST1JFRF9NQVNLICgoMTw8MSktMSkKLQotI2RlZmluZSBUT1VQUEVSX1NISUZUIChNSVJST1JFRF9TSElGVCsxKQotI2RlZmluZSBUT1VQUEVSX01BU0sgKCgxPDw2KS0xKQotCi0jZGVmaW5lIFRPTE9XRVJfU0hJRlQgKFRPVVBQRVJfU0hJRlQrNikKLSNkZWZpbmUgVE9MT1dFUl9NQVNLICgoMTw8NiktMSkKLQotI2RlZmluZSBUT1RJVExFX1NISUZUIChUT0xPV0VSX1NISUZUKzYpCi0jZGVmaW5lIFRPVElUTEVfTUFTSyAoKDE8PDIpLTEpCi0KLSNkZWZpbmUgTUlSUk9SX1NISUZUIChUT1RJVExFX1NISUZUKzIpCi0jZGVmaW5lIE1JUlJPUl9NQVNLICgoMTw8NSktMSkKLQotI2RlZmluZSBOVU1FUklDX1NISUZUIChUT1RJVExFX1NISUZUKzIpCi0jZGVmaW5lIE5VTUVSSUNfTUFTSyAoKDE8PDcpLTEpCi0KLSNkZWZpbmUgREVDT01QT1NJVElPTl9TSElGVCAoMTEpCi0jZGVmaW5lIERFQ09NUE9TSVRJT05fTUFTSyAoKDE8PDUpLTEpCi0KLS8qCi0gKiBSZXR1cm5zIHRoZSB2YWx1ZSBzdG9yZWQgaW4gdGhlIENoYXJhY3RlckRhdGEgdGFibGVzIHRoYXQgY29udGFpbnMKLSAqIGFuIGluZGV4IGludG8gdGhlIHBhY2tlZCBkYXRhIHRhYmxlIGFuZCB0aGUgZGVjb21wb3NpdGlvbiB0eXBlLgotICovCi1zdGF0aWMgdWludDE2X3QgZmluZENoYXJhY3RlclZhbHVlKFVDaGFyMzIgYykKLXsKLSAgICBMT0dfQVNTRVJUKGMgPj0gMCAmJiBjIDw9IDB4MTBGRkZGLCAiZmluZENoYXJhY3RlclZhbHVlIHJlY2VpdmVkIGFuIGludmFsaWQgY29kZXBvaW50Iik7Ci0gICAgaWYgKGMgPCAyNTYpCi0gICAgICAgIHJldHVybiBDaGFyYWN0ZXJEYXRhOjpMQVRJTjFfREFUQVtjXTsKLQotICAgIC8vIFJvdGF0ZSB0aGUgYml0cyBiZWNhdXNlIHRoZSB0YWJsZXMgYXJlIHNlcGFyYXRlZCBpbnRvIGV2ZW4gYW5kIG9kZCBjb2RlcG9pbnRzCi0gICAgYyA9IChjID4+IDEpIHwgKChjICYgMSkgPDwgMjApOwotCi0gICAgQ2hhcmFjdGVyRGF0YTo6UmFuZ2Ugc2VhcmNoID0gQ2hhcmFjdGVyRGF0YTo6RlVMTF9EQVRBW2MgPj4gMTZdOwotICAgIGNvbnN0IHVpbnQzMl90KiBhcnJheSA9IHNlYXJjaC5hcnJheTsKLSAKLSAgICAvLyBUaGlzIHRyaWNrIGlzIHNvIHRoYXQgdGhhdCBjb21wYXJlIGluIHRoZSB3aGlsZSBsb29wIGRvZXMgbm90Ci0gICAgLy8gbmVlZCB0byBzaGlmdCB0aGUgYXJyYXkgZW50cnkgZG93biBieSAxNgotICAgIGMgPDw9IDE2OwotICAgIGMgfD0gMHhGRkZGOwotCi0gICAgaW50IGhpZ2ggPSAoaW50KXNlYXJjaC5sZW5ndGggLSAxOwotICAgIGludCBsb3cgPSAwOwotCi0gICAgaWYgKGhpZ2ggPCAwKQotICAgICAgICByZXR1cm4gMDsKLSAgICAKLSAgICB3aGlsZSAobG93IDwgaGlnaCAtIDEpCi0gICAgewotICAgICAgICBpbnQgcHJvYmUgPSAoaGlnaCArIGxvdykgPj4gMTsKLQotICAgICAgICAvLyBUaGUgZW50cmllcyBjb250YWluIHRoZSBjb2RlcG9pbnQgaW4gdGhlIGhpZ2ggMTYgYml0cyBhbmQgdGhlIGluZGV4Ci0gICAgICAgIC8vIGludG8gUEFDS0VEX0RBVEEgaW4gdGhlIGxvdyAxNi4KLSAgICAgICAgaWYgKGFycmF5W3Byb2JlXSA+ICh1bnNpZ25lZCljKQotICAgICAgICAgICAgaGlnaCA9IHByb2JlOwotICAgICAgICBlbHNlCi0gICAgICAgICAgICBsb3cgPSBwcm9iZTsKLSAgICB9Ci0KLSAgICBMT0dfQVNTRVJUKChhcnJheVtsb3ddIDw9ICh1bnNpZ25lZCljKSwgIkEgc3VpdGFibGUgcmFuZ2Ugd2FzIG5vdCBmb3VuZCIpOwotICAgIHJldHVybiBhcnJheVtsb3ddICYgMHhGRkZGOwotfQotCi11aW50MzJfdCBhbmRyb2lkOjpVbmljb2RlOjpnZXRQYWNrZWREYXRhKFVDaGFyMzIgYykKLXsKLSAgICAvLyBmaW5kQ2hhcmFjdGVyVmFsdWUgcmV0dXJucyBhIDE2LWJpdCB2YWx1ZSB3aXRoIHRoZSB0b3AgNSBiaXRzIGNvbnRhaW5pbmcgYSBkZWNvbXBvc2l0aW9uIHR5cGUKLSAgICAvLyBhbmQgdGhlIHJlbWFpbmluZyBiaXRzIGNvbnRhaW5pbmcgYW4gaW5kZXguCi0gICAgcmV0dXJuIENoYXJhY3RlckRhdGE6OlBBQ0tFRF9EQVRBW2ZpbmRDaGFyYWN0ZXJWYWx1ZShjKSAmIDB4N0ZGXTsKLX0KLQotYW5kcm9pZDo6VW5pY29kZTo6Q2hhclR5cGUgYW5kcm9pZDo6VW5pY29kZTo6Z2V0VHlwZShVQ2hhcjMyIGMpCi17Ci0gICAgaWYgKGMgPCAwIHx8IGMgPj0gMHgxMEZGRkYpCi0gICAgICAgIHJldHVybiBDSEFSVFlQRV9VTkFTU0lHTkVEOwotICAgIHJldHVybiAoQ2hhclR5cGUpKChnZXRQYWNrZWREYXRhKGMpID4+IFRZUEVfU0hJRlQpICYgVFlQRV9NQVNLKTsKLX0KLQotYW5kcm9pZDo6VW5pY29kZTo6RGVjb21wb3NpdGlvblR5cGUgYW5kcm9pZDo6VW5pY29kZTo6Z2V0RGVjb21wb3NpdGlvblR5cGUoVUNoYXIzMiBjKQotewotICAgIC8vIGZpbmRDaGFyYWN0ZXJWYWx1ZSByZXR1cm5zIGEgMTYtYml0IHZhbHVlIHdpdGggdGhlIHRvcCA1IGJpdHMgY29udGFpbmluZyBhIGRlY29tcG9zaXRpb24gdHlwZQotICAgIC8vIGFuZCB0aGUgcmVtYWluaW5nIGJpdHMgY29udGFpbmluZyBhbiBpbmRleC4KLSAgICByZXR1cm4gKERlY29tcG9zaXRpb25UeXBlKSgoZmluZENoYXJhY3RlclZhbHVlKGMpID4+IERFQ09NUE9TSVRJT05fU0hJRlQpICYgREVDT01QT1NJVElPTl9NQVNLKTsKLX0KLQotaW50IGFuZHJvaWQ6OlVuaWNvZGU6OmdldERpZ2l0VmFsdWUoVUNoYXIzMiBjLCBpbnQgcmFkaXgpCi17Ci0gICAgaWYgKHJhZGl4IDwgTUlOX1JBRElYIHx8IHJhZGl4ID4gTUFYX1JBRElYKQotICAgICAgICByZXR1cm4gLTE7Ci0KLSAgICBpbnQgdGVtcFZhbHVlID0gcmFkaXg7Ci0gICAgCi0gICAgaWYgKGMgPj0gJzAnICYmIGMgPD0gJzknKQotICAgICAgICB0ZW1wVmFsdWUgPSBjIC0gJzAnOwotICAgIGVsc2UgaWYgKGMgPj0gJ2EnICYmIGMgPD0gJ3onKQotICAgICAgICB0ZW1wVmFsdWUgPSBjIC0gJ2EnICsgMTA7Ci0gICAgZWxzZSBpZiAoYyA+PSAnQScgJiYgYyA8PSAnWicpCi0gICAgICAgIHRlbXBWYWx1ZSA9IGMgLSAnQScgKyAxMDsKLSAgICAKLSAgICByZXR1cm4gdGVtcFZhbHVlIDwgcmFkaXggPyB0ZW1wVmFsdWUgOiAtMTsKLX0KLQotaW50IGFuZHJvaWQ6OlVuaWNvZGU6OmdldE51bWVyaWNWYWx1ZShVQ2hhcjMyIGMpCi17Ci0gICAgaWYgKGlzTWlycm9yZWQoYykpCi0gICAgICAgIHJldHVybiAtMTsKLSAgICAKLSAgICByZXR1cm4gKGludCkgQ2hhcmFjdGVyRGF0YTo6TlVNRVJJQ1NbKChnZXRQYWNrZWREYXRhKGMpID4+IE5VTUVSSUNfU0hJRlQpICYgTlVNRVJJQ19NQVNLKV07Ci19Ci0KLVVDaGFyMzIgYW5kcm9pZDo6VW5pY29kZTo6dG9Mb3dlcihVQ2hhcjMyIGMpCi17Ci0gICAgcmV0dXJuIGMgKyBDaGFyYWN0ZXJEYXRhOjpMQ0RJRkZbKGdldFBhY2tlZERhdGEoYykgPj4gVE9MT1dFUl9TSElGVCkgJiBUT0xPV0VSX01BU0tdOwotfQotCi1VQ2hhcjMyIGFuZHJvaWQ6OlVuaWNvZGU6OnRvVXBwZXIoVUNoYXIzMiBjKQotewotICAgIHJldHVybiBjICsgQ2hhcmFjdGVyRGF0YTo6VUNESUZGWyhnZXRQYWNrZWREYXRhKGMpID4+IFRPVVBQRVJfU0hJRlQpICYgVE9VUFBFUl9NQVNLXTsKLX0KLQotYW5kcm9pZDo6VW5pY29kZTo6RGlyZWN0aW9uIGFuZHJvaWQ6OlVuaWNvZGU6OmdldERpcmVjdGlvbmFsaXR5KFVDaGFyMzIgYykKLXsKLSAgICB1aW50MzJfdCBkYXRhID0gZ2V0UGFja2VkRGF0YShjKTsKLQotICAgIGlmICgwID09IGRhdGEpCi0gICAgICAgIHJldHVybiBESVJFQ1RJT05BTElUWV9VTkRFRklORUQ7Ci0KLSAgICBEaXJlY3Rpb24gZCA9IChEaXJlY3Rpb24pICgoZGF0YSA+PiBESVJFQ1RJT05fU0hJRlQpICYgRElSRUNUSU9OX01BU0spOwotCi0gICAgaWYgKERJUkVDVElPTl9NQVNLID09IGQpCi0gICAgICAgIHJldHVybiBESVJFQ1RJT05BTElUWV9VTkRFRklORUQ7Ci0gICAgCi0gICAgcmV0dXJuIGQ7Ci19Ci0KLWJvb2wgYW5kcm9pZDo6VW5pY29kZTo6aXNNaXJyb3JlZChVQ2hhcjMyIGMpCi17Ci0gICAgcmV0dXJuICgoZ2V0UGFja2VkRGF0YShjKSA+PiBNSVJST1JFRF9TSElGVCkgJiBNSVJST1JFRF9NQVNLKSAhPSAwOwotfQotCi1VQ2hhcjMyIGFuZHJvaWQ6OlVuaWNvZGU6OnRvTWlycm9yKFVDaGFyMzIgYykKLXsKLSAgICBpZiAoIWlzTWlycm9yZWQoYykpCi0gICAgICAgIHJldHVybiBjOwotCi0gICAgcmV0dXJuIGMgKyBDaGFyYWN0ZXJEYXRhOjpNSVJST1JfRElGRlsoZ2V0UGFja2VkRGF0YShjKSA+PiBNSVJST1JfU0hJRlQpICYgTUlSUk9SX01BU0tdOwotfQotCi1VQ2hhcjMyIGFuZHJvaWQ6OlVuaWNvZGU6OnRvVGl0bGUoVUNoYXIzMiBjKQotewotICAgIGludDMyX3QgZGlmZiA9IENoYXJhY3RlckRhdGE6OlRDRElGRlsoZ2V0UGFja2VkRGF0YShjKSA+PiBUT1RJVExFX1NISUZUKSAmIFRPVElUTEVfTUFTS107Ci0KLSAgICBpZiAoVE9USVRMRV9NQVNLID09IGRpZmYpCi0gICAgICAgIHJldHVybiB0b1VwcGVyKGMpOwotICAgIAotICAgIHJldHVybiBjICsgZGlmZjsKLX0KLQotCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1ZlY3RvckltcGwuY3BwIGIvbGlicy91dGlscy9WZWN0b3JJbXBsLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMmMyZDY2Ny4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1ZlY3RvckltcGwuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNjExICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIlZlY3RvciIKLQotI2luY2x1ZGUgPHN0cmluZy5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9Mb2cuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLSNpbmNsdWRlIDx1dGlscy9TaGFyZWRCdWZmZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9WZWN0b3JJbXBsLmg+Ci0KLS8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY29uc3Qgc2l6ZV90IGtNaW5WZWN0b3JDYXBhY2l0eSA9IDQ7Ci0KLXN0YXRpYyBpbmxpbmUgc2l6ZV90IG1heChzaXplX3QgYSwgc2l6ZV90IGIpIHsKLSAgICByZXR1cm4gYT5iID8gYSA6IGI7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotVmVjdG9ySW1wbDo6VmVjdG9ySW1wbChzaXplX3QgaXRlbVNpemUsIHVpbnQzMl90IGZsYWdzKQotICAgIDogbVN0b3JhZ2UoMCksIG1Db3VudCgwKSwgbUZsYWdzKGZsYWdzKSwgbUl0ZW1TaXplKGl0ZW1TaXplKQotewotfQotCi1WZWN0b3JJbXBsOjpWZWN0b3JJbXBsKGNvbnN0IFZlY3RvckltcGwmIHJocykKLSAgICA6ICAgbVN0b3JhZ2UocmhzLm1TdG9yYWdlKSwgbUNvdW50KHJocy5tQ291bnQpLAotICAgICAgICBtRmxhZ3MocmhzLm1GbGFncyksIG1JdGVtU2l6ZShyaHMubUl0ZW1TaXplKQotewotICAgIGlmIChtU3RvcmFnZSkgewotICAgICAgICBTaGFyZWRCdWZmZXI6OnNoYXJlZEJ1ZmZlcihtU3RvcmFnZSktPmFjcXVpcmUoKTsKLSAgICB9Ci19Ci0KLVZlY3RvckltcGw6On5WZWN0b3JJbXBsKCkKLXsKLSAgICBMT0dfQVNTRVJUKCFtQ291bnQsCi0gICAgICAgICJbJXBdICIKLSAgICAgICAgInN1YmNsYXNzZXMgb2YgVmVjdG9ySW1wbCBtdXN0IGNhbGwgZmluaXNoX3ZlY3RvcigpIgotICAgICAgICAiIGluIHRoZWlyIGRlc3RydWN0b3IuIExlYWtpbmcgJWQgYnl0ZXMuIiwKLSAgICAgICAgdGhpcywgKGludCkobUNvdW50Km1JdGVtU2l6ZSkpOwotICAgIC8vIFdlIGNhbid0IGNhbGwgX2RvX2Rlc3Ryb3koKSBoZXJlIGJlY2F1c2UgdGhlIHZ0YWJsZSBpcyBhbHJlYWR5IGdvbmUuIAotfQotCi1WZWN0b3JJbXBsJiBWZWN0b3JJbXBsOjpvcGVyYXRvciA9IChjb25zdCBWZWN0b3JJbXBsJiByaHMpCi17Ci0gICAgTE9HX0FTU0VSVChtSXRlbVNpemUgPT0gcmhzLm1JdGVtU2l6ZSwKLSAgICAgICAgIlZlY3Rvcjw+IGhhdmUgZGlmZmVyZW50IHR5cGVzICh0aGlzPSVwLCByaHM9JXApIiwgdGhpcywgJnJocyk7Ci0gICAgaWYgKHRoaXMgIT0gJnJocykgewotICAgICAgICByZWxlYXNlX3N0b3JhZ2UoKTsKLSAgICAgICAgaWYgKHJocy5tQ291bnQpIHsKLSAgICAgICAgICAgIG1TdG9yYWdlID0gcmhzLm1TdG9yYWdlOwotICAgICAgICAgICAgbUNvdW50ID0gcmhzLm1Db3VudDsKLSAgICAgICAgICAgIFNoYXJlZEJ1ZmZlcjo6c2hhcmVkQnVmZmVyKG1TdG9yYWdlKS0+YWNxdWlyZSgpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbVN0b3JhZ2UgPSAwOwotICAgICAgICAgICAgbUNvdW50ID0gMDsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gKnRoaXM7Ci19Ci0KLXZvaWQqIFZlY3RvckltcGw6OmVkaXRBcnJheUltcGwoKQotewotICAgIGlmIChtU3RvcmFnZSkgewotICAgICAgICBTaGFyZWRCdWZmZXIqIHNiID0gU2hhcmVkQnVmZmVyOjpzaGFyZWRCdWZmZXIobVN0b3JhZ2UpLT5hdHRlbXB0RWRpdCgpOwotICAgICAgICBpZiAoc2IgPT0gMCkgewotICAgICAgICAgICAgc2IgPSBTaGFyZWRCdWZmZXI6OmFsbG9jKGNhcGFjaXR5KCkgKiBtSXRlbVNpemUpOwotICAgICAgICAgICAgaWYgKHNiKSB7Ci0gICAgICAgICAgICAgICAgX2RvX2NvcHkoc2ItPmRhdGEoKSwgbVN0b3JhZ2UsIG1Db3VudCk7Ci0gICAgICAgICAgICAgICAgcmVsZWFzZV9zdG9yYWdlKCk7Ci0gICAgICAgICAgICAgICAgbVN0b3JhZ2UgPSBzYi0+ZGF0YSgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBtU3RvcmFnZTsKLX0KLQotc2l6ZV90IFZlY3RvckltcGw6OmNhcGFjaXR5KCkgY29uc3QKLXsKLSAgICBpZiAobVN0b3JhZ2UpIHsKLSAgICAgICAgcmV0dXJuIFNoYXJlZEJ1ZmZlcjo6c2hhcmVkQnVmZmVyKG1TdG9yYWdlKS0+c2l6ZSgpIC8gbUl0ZW1TaXplOwotICAgIH0KLSAgICByZXR1cm4gMDsKLX0KLQotc3NpemVfdCBWZWN0b3JJbXBsOjppbnNlcnRWZWN0b3JBdChjb25zdCBWZWN0b3JJbXBsJiB2ZWN0b3IsIHNpemVfdCBpbmRleCkKLXsKLSAgICBpZiAoaW5kZXggPiBzaXplKCkpCi0gICAgICAgIHJldHVybiBCQURfSU5ERVg7Ci0gICAgdm9pZCogd2hlcmUgPSBfZ3JvdyhpbmRleCwgdmVjdG9yLnNpemUoKSk7Ci0gICAgaWYgKHdoZXJlKSB7Ci0gICAgICAgIF9kb19jb3B5KHdoZXJlLCB2ZWN0b3IuYXJyYXlJbXBsKCksIHZlY3Rvci5zaXplKCkpOwotICAgIH0KLSAgICByZXR1cm4gd2hlcmUgPyBpbmRleCA6IChzc2l6ZV90KU5PX01FTU9SWTsKLX0KLQotc3NpemVfdCBWZWN0b3JJbXBsOjphcHBlbmRWZWN0b3IoY29uc3QgVmVjdG9ySW1wbCYgdmVjdG9yKQotewotICAgIHJldHVybiBpbnNlcnRWZWN0b3JBdCh2ZWN0b3IsIHNpemUoKSk7Ci19Ci0KLXNzaXplX3QgVmVjdG9ySW1wbDo6aW5zZXJ0QXQoc2l6ZV90IGluZGV4LCBzaXplX3QgbnVtSXRlbXMpCi17Ci0gICAgcmV0dXJuIGluc2VydEF0KDAsIGluZGV4LCBudW1JdGVtcyk7Ci19Ci0KLXNzaXplX3QgVmVjdG9ySW1wbDo6aW5zZXJ0QXQoY29uc3Qgdm9pZCogaXRlbSwgc2l6ZV90IGluZGV4LCBzaXplX3QgbnVtSXRlbXMpCi17Ci0gICAgaWYgKGluZGV4ID4gc2l6ZSgpKQotICAgICAgICByZXR1cm4gQkFEX0lOREVYOwotICAgIHZvaWQqIHdoZXJlID0gX2dyb3coaW5kZXgsIG51bUl0ZW1zKTsKLSAgICBpZiAod2hlcmUpIHsKLSAgICAgICAgaWYgKGl0ZW0pIHsKLSAgICAgICAgICAgIF9kb19zcGxhdCh3aGVyZSwgaXRlbSwgbnVtSXRlbXMpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgX2RvX2NvbnN0cnVjdCh3aGVyZSwgbnVtSXRlbXMpOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiB3aGVyZSA/IGluZGV4IDogKHNzaXplX3QpTk9fTUVNT1JZOwotfQotCi1zdGF0aWMgaW50IHNvcnRQcm94eShjb25zdCB2b2lkKiBsaHMsIGNvbnN0IHZvaWQqIHJocywgdm9pZCogZnVuYykKLXsKLSAgICByZXR1cm4gKCooVmVjdG9ySW1wbDo6Y29tcGFyX3QpZnVuYykobGhzLCByaHMpOwotfQotCi1zdGF0dXNfdCBWZWN0b3JJbXBsOjpzb3J0KFZlY3RvckltcGw6OmNvbXBhcl90IGNtcCkKLXsKLSAgICByZXR1cm4gc29ydChzb3J0UHJveHksICh2b2lkKiljbXApOwotfQotCi1zdGF0dXNfdCBWZWN0b3JJbXBsOjpzb3J0KFZlY3RvckltcGw6OmNvbXBhcl9yX3QgY21wLCB2b2lkKiBzdGF0ZSkKLXsKLSAgICAvLyB0aGUgc29ydCBtdXN0IGJlIHN0YWJsZS4gd2UncmUgdXNpbmcgaW5zZXJ0aW9uIHNvcnQgd2hpY2gKLSAgICAvLyBpcyB3ZWxsIHN1aXRlZCBmb3Igc21hbGwgYW5kIGFscmVhZHkgc29ydGVkIGFycmF5cwotICAgIC8vIGZvciBiaWcgYXJyYXlzLCBpdCBjb3VsZCBiZSBiZXR0ZXIgdG8gdXNlIG1lcmdlc29ydAotICAgIGNvbnN0IHNzaXplX3QgY291bnQgPSBzaXplKCk7Ci0gICAgaWYgKGNvdW50ID4gMSkgewotICAgICAgICB2b2lkKiBhcnJheSA9IGNvbnN0X2Nhc3Q8dm9pZCo+KGFycmF5SW1wbCgpKTsKLSAgICAgICAgdm9pZCogdGVtcCA9IDA7Ci0gICAgICAgIHNzaXplX3QgaSA9IDE7Ci0gICAgICAgIHdoaWxlIChpIDwgY291bnQpIHsKLSAgICAgICAgICAgIHZvaWQqIGl0ZW0gPSByZWludGVycHJldF9jYXN0PGNoYXIqPihhcnJheSkgKyBtSXRlbVNpemUqKGkpOwotICAgICAgICAgICAgdm9pZCogY3VyciA9IHJlaW50ZXJwcmV0X2Nhc3Q8Y2hhcio+KGFycmF5KSArIG1JdGVtU2l6ZSooaS0xKTsKLSAgICAgICAgICAgIGlmIChjbXAoY3VyciwgaXRlbSwgc3RhdGUpID4gMCkgewotCi0gICAgICAgICAgICAgICAgaWYgKCF0ZW1wKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIHdlJ3JlIGdvaW5nIHRvIGhhdmUgdG8gbW9kaWZ5IHRoZSBhcnJheS4uLgotICAgICAgICAgICAgICAgICAgICBhcnJheSA9IGVkaXRBcnJheUltcGwoKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKCFhcnJheSkgcmV0dXJuIE5PX01FTU9SWTsKLSAgICAgICAgICAgICAgICAgICAgdGVtcCA9IG1hbGxvYyhtSXRlbVNpemUpOwotICAgICAgICAgICAgICAgICAgICBpZiAoIXRlbXApIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgICAgICAgICAgICAgIF9kb19jb25zdHJ1Y3QodGVtcCwgMSk7Ci0gICAgICAgICAgICAgICAgICAgIGl0ZW0gPSByZWludGVycHJldF9jYXN0PGNoYXIqPihhcnJheSkgKyBtSXRlbVNpemUqKGkpOwotICAgICAgICAgICAgICAgICAgICBjdXJyID0gcmVpbnRlcnByZXRfY2FzdDxjaGFyKj4oYXJyYXkpICsgbUl0ZW1TaXplKihpLTEpOwotICAgICAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgICAgIF9kb19jb3B5KHRlbXAsIGl0ZW0sIDEpOwotCi0gICAgICAgICAgICAgICAgc3NpemVfdCBqID0gaS0xOwotICAgICAgICAgICAgICAgIHZvaWQqIG5leHQgPSByZWludGVycHJldF9jYXN0PGNoYXIqPihhcnJheSkgKyBtSXRlbVNpemUqKGkpOyAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgZG8gewotICAgICAgICAgICAgICAgICAgICBfZG9fY29weShuZXh0LCBjdXJyLCAxKTsKLSAgICAgICAgICAgICAgICAgICAgbmV4dCA9IGN1cnI7Ci0gICAgICAgICAgICAgICAgICAgIC0tajsKLSAgICAgICAgICAgICAgICAgICAgY3VyciA9IHJlaW50ZXJwcmV0X2Nhc3Q8Y2hhcio+KGFycmF5KSArIG1JdGVtU2l6ZSooaik7ICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICB9IHdoaWxlIChqPj0wICYmIChjbXAoY3VyciwgdGVtcCwgc3RhdGUpID4gMCkpOwotCi0gICAgICAgICAgICAgICAgX2RvX2NvcHkobmV4dCwgdGVtcCwgMSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpKys7Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIGlmICh0ZW1wKSB7Ci0gICAgICAgICAgICBfZG9fZGVzdHJveSh0ZW1wLCAxKTsKLSAgICAgICAgICAgIGZyZWUodGVtcCk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi12b2lkIFZlY3RvckltcGw6OnBvcCgpCi17Ci0gICAgaWYgKHNpemUoKSkKLSAgICAgICAgcmVtb3ZlSXRlbXNBdChzaXplKCktMSwgMSk7Ci19Ci0KLXZvaWQgVmVjdG9ySW1wbDo6cHVzaCgpCi17Ci0gICAgcHVzaCgwKTsKLX0KLQotdm9pZCBWZWN0b3JJbXBsOjpwdXNoKGNvbnN0IHZvaWQqIGl0ZW0pCi17Ci0gICAgaW5zZXJ0QXQoaXRlbSwgc2l6ZSgpKTsKLX0KLQotc3NpemVfdCBWZWN0b3JJbXBsOjphZGQoKQotewotICAgIHJldHVybiBhZGQoMCk7Ci19Ci0KLXNzaXplX3QgVmVjdG9ySW1wbDo6YWRkKGNvbnN0IHZvaWQqIGl0ZW0pCi17Ci0gICAgcmV0dXJuIGluc2VydEF0KGl0ZW0sIHNpemUoKSk7Ci19Ci0KLXNzaXplX3QgVmVjdG9ySW1wbDo6cmVwbGFjZUF0KHNpemVfdCBpbmRleCkKLXsKLSAgICByZXR1cm4gcmVwbGFjZUF0KDAsIGluZGV4KTsKLX0KLQotc3NpemVfdCBWZWN0b3JJbXBsOjpyZXBsYWNlQXQoY29uc3Qgdm9pZCogcHJvdG90eXBlLCBzaXplX3QgaW5kZXgpCi17Ci0gICAgTE9HX0FTU0VSVChpbmRleDxzaXplKCksCi0gICAgICAgICJbJXBdIHJlcGxhY2U6IGluZGV4PSVkLCBzaXplPSVkIiwgdGhpcywgKGludClpbmRleCwgKGludClzaXplKCkpOwotCi0gICAgdm9pZCogaXRlbSA9IGVkaXRJdGVtTG9jYXRpb24oaW5kZXgpOwotICAgIGlmIChpdGVtID09IDApCi0gICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgX2RvX2Rlc3Ryb3koaXRlbSwgMSk7Ci0gICAgaWYgKHByb3RvdHlwZSA9PSAwKSB7Ci0gICAgICAgIF9kb19jb25zdHJ1Y3QoaXRlbSwgMSk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgX2RvX2NvcHkoaXRlbSwgcHJvdG90eXBlLCAxKTsKLSAgICB9Ci0gICAgcmV0dXJuIHNzaXplX3QoaW5kZXgpOwotfQotCi1zc2l6ZV90IFZlY3RvckltcGw6OnJlbW92ZUl0ZW1zQXQoc2l6ZV90IGluZGV4LCBzaXplX3QgY291bnQpCi17Ci0gICAgTE9HX0FTU0VSVCgoaW5kZXgrY291bnQpPD1zaXplKCksCi0gICAgICAgICJbJXBdIHJlbW92ZTogaW5kZXg9JWQsIGNvdW50PSVkLCBzaXplPSVkIiwKLSAgICAgICAgICAgICAgIHRoaXMsIChpbnQpaW5kZXgsIChpbnQpY291bnQsIChpbnQpc2l6ZSgpKTsKLQotICAgIGlmICgoaW5kZXgrY291bnQpID4gc2l6ZSgpKQotICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOwotICAgX3NocmluayhpbmRleCwgY291bnQpOwotICAgcmV0dXJuIGluZGV4OwotfQotCi12b2lkIFZlY3RvckltcGw6OmZpbmlzaF92ZWN0b3IoKQotewotICAgIHJlbGVhc2Vfc3RvcmFnZSgpOwotICAgIG1TdG9yYWdlID0gMDsKLSAgICBtQ291bnQgPSAwOwotfQotCi12b2lkIFZlY3RvckltcGw6OmNsZWFyKCkKLXsKLSAgICBfc2hyaW5rKDAsIG1Db3VudCk7Ci19Ci0KLXZvaWQqIFZlY3RvckltcGw6OmVkaXRJdGVtTG9jYXRpb24oc2l6ZV90IGluZGV4KQotewotICAgIExPR19BU1NFUlQoaW5kZXg8Y2FwYWNpdHkoKSwKLSAgICAgICAgIlslcF0gaXRlbUxvY2F0aW9uOiBpbmRleD0lZCwgY2FwYWNpdHk9JWQsIGNvdW50PSVkIiwKLSAgICAgICAgdGhpcywgKGludClpbmRleCwgKGludCljYXBhY2l0eSgpLCAoaW50KW1Db3VudCk7Ci0gICAgICAgICAgICAKLSAgICB2b2lkKiBidWZmZXIgPSBlZGl0QXJyYXlJbXBsKCk7Ci0gICAgaWYgKGJ1ZmZlcikKLSAgICAgICAgcmV0dXJuIHJlaW50ZXJwcmV0X2Nhc3Q8Y2hhcio+KGJ1ZmZlcikgKyBpbmRleCptSXRlbVNpemU7Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLWNvbnN0IHZvaWQqIFZlY3RvckltcGw6Oml0ZW1Mb2NhdGlvbihzaXplX3QgaW5kZXgpIGNvbnN0Ci17Ci0gICAgTE9HX0FTU0VSVChpbmRleDxjYXBhY2l0eSgpLAotICAgICAgICAiWyVwXSBlZGl0SXRlbUxvY2F0aW9uOiBpbmRleD0lZCwgY2FwYWNpdHk9JWQsIGNvdW50PSVkIiwKLSAgICAgICAgdGhpcywgKGludClpbmRleCwgKGludCljYXBhY2l0eSgpLCAoaW50KW1Db3VudCk7Ci0KLSAgICBjb25zdCAgdm9pZCogYnVmZmVyID0gYXJyYXlJbXBsKCk7Ci0gICAgaWYgKGJ1ZmZlcikKLSAgICAgICAgcmV0dXJuIHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgY2hhcio+KGJ1ZmZlcikgKyBpbmRleCptSXRlbVNpemU7Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXNzaXplX3QgVmVjdG9ySW1wbDo6c2V0Q2FwYWNpdHkoc2l6ZV90IG5ld19jYXBhY2l0eSkKLXsKLSAgICBzaXplX3QgY3VycmVudF9jYXBhY2l0eSA9IGNhcGFjaXR5KCk7Ci0gICAgc3NpemVfdCBhbW91bnQgPSBuZXdfY2FwYWNpdHkgLSBzaXplKCk7Ci0gICAgaWYgKGFtb3VudCA8PSAwKSB7Ci0gICAgICAgIC8vIHdlIGNhbid0IHJlZHVjZSB0aGUgY2FwYWNpdHkKLSAgICAgICAgcmV0dXJuIGN1cnJlbnRfY2FwYWNpdHk7Ci0gICAgfSAKLSAgICBTaGFyZWRCdWZmZXIqIHNiID0gU2hhcmVkQnVmZmVyOjphbGxvYyhuZXdfY2FwYWNpdHkgKiBtSXRlbVNpemUpOwotICAgIGlmIChzYikgewotICAgICAgICB2b2lkKiBhcnJheSA9IHNiLT5kYXRhKCk7Ci0gICAgICAgIF9kb19jb3B5KGFycmF5LCBtU3RvcmFnZSwgc2l6ZSgpKTsKLSAgICAgICAgcmVsZWFzZV9zdG9yYWdlKCk7Ci0gICAgICAgIG1TdG9yYWdlID0gY29uc3RfY2FzdDx2b2lkKj4oYXJyYXkpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgfQotICAgIHJldHVybiBuZXdfY2FwYWNpdHk7Ci19Ci0KLXZvaWQgVmVjdG9ySW1wbDo6cmVsZWFzZV9zdG9yYWdlKCkKLXsKLSAgICBpZiAobVN0b3JhZ2UpIHsKLSAgICAgICAgY29uc3QgU2hhcmVkQnVmZmVyKiBzYiA9IFNoYXJlZEJ1ZmZlcjo6c2hhcmVkQnVmZmVyKG1TdG9yYWdlKTsKLSAgICAgICAgaWYgKHNiLT5yZWxlYXNlKFNoYXJlZEJ1ZmZlcjo6ZUtlZXBTdG9yYWdlKSA9PSAxKSB7Ci0gICAgICAgICAgICBfZG9fZGVzdHJveShtU3RvcmFnZSwgbUNvdW50KTsKLSAgICAgICAgICAgIFNoYXJlZEJ1ZmZlcjo6ZGVhbGxvYyhzYik7Ci0gICAgICAgIH0gCi0gICAgfQotfQotCi12b2lkKiBWZWN0b3JJbXBsOjpfZ3JvdyhzaXplX3Qgd2hlcmUsIHNpemVfdCBhbW91bnQpCi17Ci0vLyAgICBMT0dWKCJfZ3Jvdyh0aGlzPSVwLCB3aGVyZT0lZCwgYW1vdW50PSVkKSBjb3VudD0lZCwgY2FwYWNpdHk9JWQiLAotLy8gICAgICAgIHRoaXMsIChpbnQpd2hlcmUsIChpbnQpYW1vdW50LCAoaW50KW1Db3VudCwgKGludCljYXBhY2l0eSgpKTsKLQotICAgIGlmICh3aGVyZSA+IG1Db3VudCkKLSAgICAgICAgd2hlcmUgPSBtQ291bnQ7Ci0gICAgICAKLSAgICBjb25zdCBzaXplX3QgbmV3X3NpemUgPSBtQ291bnQgKyBhbW91bnQ7Ci0gICAgaWYgKGNhcGFjaXR5KCkgPCBuZXdfc2l6ZSkgewotICAgICAgICBjb25zdCBzaXplX3QgbmV3X2NhcGFjaXR5ID0gbWF4KGtNaW5WZWN0b3JDYXBhY2l0eSwgKChuZXdfc2l6ZSozKSsxKS8yKTsKLS8vICAgICAgICBMT0dWKCJncm93IHZlY3RvciAlcCwgbmV3X2NhcGFjaXR5PSVkIiwgdGhpcywgKGludCluZXdfY2FwYWNpdHkpOwotICAgICAgICBpZiAoKG1TdG9yYWdlKSAmJgotICAgICAgICAgICAgKG1Db3VudD09d2hlcmUpICYmCi0gICAgICAgICAgICAobUZsYWdzICYgSEFTX1RSSVZJQUxfQ09QWSkgJiYKLSAgICAgICAgICAgIChtRmxhZ3MgJiBIQVNfVFJJVklBTF9EVE9SKSkKLSAgICAgICAgewotICAgICAgICAgICAgY29uc3QgU2hhcmVkQnVmZmVyKiBjdXJfc2IgPSBTaGFyZWRCdWZmZXI6OnNoYXJlZEJ1ZmZlcihtU3RvcmFnZSk7Ci0gICAgICAgICAgICBTaGFyZWRCdWZmZXIqIHNiID0gY3VyX3NiLT5lZGl0UmVzaXplKG5ld19jYXBhY2l0eSAqIG1JdGVtU2l6ZSk7Ci0gICAgICAgICAgICBtU3RvcmFnZSA9IHNiLT5kYXRhKCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBTaGFyZWRCdWZmZXIqIHNiID0gU2hhcmVkQnVmZmVyOjphbGxvYyhuZXdfY2FwYWNpdHkgKiBtSXRlbVNpemUpOwotICAgICAgICAgICAgaWYgKHNiKSB7Ci0gICAgICAgICAgICAgICAgdm9pZCogYXJyYXkgPSBzYi0+ZGF0YSgpOwotICAgICAgICAgICAgICAgIGlmICh3aGVyZT4wKSB7Ci0gICAgICAgICAgICAgICAgICAgIF9kb19jb3B5KGFycmF5LCBtU3RvcmFnZSwgd2hlcmUpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBpZiAobUNvdW50PndoZXJlKSB7Ci0gICAgICAgICAgICAgICAgICAgIGNvbnN0IHZvaWQqIGZyb20gPSByZWludGVycHJldF9jYXN0PGNvbnN0IHVpbnQ4X3QgKj4obVN0b3JhZ2UpICsgd2hlcmUqbUl0ZW1TaXplOwotICAgICAgICAgICAgICAgICAgICB2b2lkKiBkZXN0ID0gcmVpbnRlcnByZXRfY2FzdDx1aW50OF90ICo+KGFycmF5KSArICh3aGVyZSthbW91bnQpKm1JdGVtU2l6ZTsKLSAgICAgICAgICAgICAgICAgICAgX2RvX2NvcHkoZGVzdCwgZnJvbSwgbUNvdW50LXdoZXJlKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmVsZWFzZV9zdG9yYWdlKCk7Ci0gICAgICAgICAgICAgICAgbVN0b3JhZ2UgPSBjb25zdF9jYXN0PHZvaWQqPihhcnJheSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBzc2l6ZV90IHMgPSBtQ291bnQtd2hlcmU7Ci0gICAgICAgIGlmIChzPjApIHsKLSAgICAgICAgICAgIHZvaWQqIGFycmF5ID0gZWRpdEFycmF5SW1wbCgpOyAgICAKLSAgICAgICAgICAgIHZvaWQqIHRvID0gcmVpbnRlcnByZXRfY2FzdDx1aW50OF90ICo+KGFycmF5KSArICh3aGVyZSthbW91bnQpKm1JdGVtU2l6ZTsKLSAgICAgICAgICAgIGNvbnN0IHZvaWQqIGZyb20gPSByZWludGVycHJldF9jYXN0PGNvbnN0IHVpbnQ4X3QgKj4oYXJyYXkpICsgd2hlcmUqbUl0ZW1TaXplOwotICAgICAgICAgICAgX2RvX21vdmVfZm9yd2FyZCh0bywgZnJvbSwgcyk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgbUNvdW50ICs9IGFtb3VudDsKLSAgICB2b2lkKiBmcmVlX3NwYWNlID0gY29uc3RfY2FzdDx2b2lkKj4oaXRlbUxvY2F0aW9uKHdoZXJlKSk7Ci0gICAgcmV0dXJuIGZyZWVfc3BhY2U7Ci19Ci0KLXZvaWQgVmVjdG9ySW1wbDo6X3NocmluayhzaXplX3Qgd2hlcmUsIHNpemVfdCBhbW91bnQpCi17Ci0gICAgaWYgKCFtU3RvcmFnZSkKLSAgICAgICAgcmV0dXJuOwotCi0vLyAgICBMT0dWKCJfc2hyaW5rKHRoaXM9JXAsIHdoZXJlPSVkLCBhbW91bnQ9JWQpIGNvdW50PSVkLCBjYXBhY2l0eT0lZCIsCi0vLyAgICAgICAgdGhpcywgKGludCl3aGVyZSwgKGludClhbW91bnQsIChpbnQpbUNvdW50LCAoaW50KWNhcGFjaXR5KCkpOwotCi0gICAgaWYgKHdoZXJlID49IG1Db3VudCkKLSAgICAgICAgd2hlcmUgPSBtQ291bnQgLSBhbW91bnQ7Ci0KLSAgICBjb25zdCBzaXplX3QgbmV3X3NpemUgPSBtQ291bnQgLSBhbW91bnQ7Ci0gICAgaWYgKG5ld19zaXplKjMgPCBjYXBhY2l0eSgpKSB7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBuZXdfY2FwYWNpdHkgPSBtYXgoa01pblZlY3RvckNhcGFjaXR5LCBuZXdfc2l6ZSoyKTsKLS8vICAgICAgICBMT0dWKCJzaHJpbmsgdmVjdG9yICVwLCBuZXdfY2FwYWNpdHk9JWQiLCB0aGlzLCAoaW50KW5ld19jYXBhY2l0eSk7Ci0gICAgICAgIGlmICgod2hlcmUgPT0gbUNvdW50LWFtb3VudCkgJiYKLSAgICAgICAgICAgIChtRmxhZ3MgJiBIQVNfVFJJVklBTF9DT1BZKSAmJgotICAgICAgICAgICAgKG1GbGFncyAmIEhBU19UUklWSUFMX0RUT1IpKQotICAgICAgICB7Ci0gICAgICAgICAgICBjb25zdCBTaGFyZWRCdWZmZXIqIGN1cl9zYiA9IFNoYXJlZEJ1ZmZlcjo6c2hhcmVkQnVmZmVyKG1TdG9yYWdlKTsKLSAgICAgICAgICAgIFNoYXJlZEJ1ZmZlciogc2IgPSBjdXJfc2ItPmVkaXRSZXNpemUobmV3X2NhcGFjaXR5ICogbUl0ZW1TaXplKTsKLSAgICAgICAgICAgIG1TdG9yYWdlID0gc2ItPmRhdGEoKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIFNoYXJlZEJ1ZmZlciogc2IgPSBTaGFyZWRCdWZmZXI6OmFsbG9jKG5ld19jYXBhY2l0eSAqIG1JdGVtU2l6ZSk7Ci0gICAgICAgICAgICBpZiAoc2IpIHsKLSAgICAgICAgICAgICAgICB2b2lkKiBhcnJheSA9IHNiLT5kYXRhKCk7Ci0gICAgICAgICAgICAgICAgaWYgKHdoZXJlPjApIHsKLSAgICAgICAgICAgICAgICAgICAgX2RvX2NvcHkoYXJyYXksIG1TdG9yYWdlLCB3aGVyZSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGlmIChtQ291bnQgPiB3aGVyZSthbW91bnQpIHsKLSAgICAgICAgICAgICAgICAgICAgY29uc3Qgdm9pZCogZnJvbSA9IHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgdWludDhfdCAqPihtU3RvcmFnZSkgKyAod2hlcmUrYW1vdW50KSptSXRlbVNpemU7Ci0gICAgICAgICAgICAgICAgICAgIHZvaWQqIGRlc3QgPSByZWludGVycHJldF9jYXN0PHVpbnQ4X3QgKj4oYXJyYXkpICsgd2hlcmUqbUl0ZW1TaXplOwotICAgICAgICAgICAgICAgICAgICBfZG9fY29weShkZXN0LCBmcm9tLCBtQ291bnQtKHdoZXJlK2Ftb3VudCkpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICByZWxlYXNlX3N0b3JhZ2UoKTsKLSAgICAgICAgICAgICAgICBtU3RvcmFnZSA9IGNvbnN0X2Nhc3Q8dm9pZCo+KGFycmF5KTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIHZvaWQqIGFycmF5ID0gZWRpdEFycmF5SW1wbCgpOyAgICAKLSAgICAgICAgdm9pZCogdG8gPSByZWludGVycHJldF9jYXN0PHVpbnQ4X3QgKj4oYXJyYXkpICsgd2hlcmUqbUl0ZW1TaXplOwotICAgICAgICBfZG9fZGVzdHJveSh0bywgYW1vdW50KTsKLSAgICAgICAgc3NpemVfdCBzID0gbUNvdW50LSh3aGVyZSthbW91bnQpOwotICAgICAgICBpZiAocz4wKSB7Ci0gICAgICAgICAgICBjb25zdCB2b2lkKiBmcm9tID0gcmVpbnRlcnByZXRfY2FzdDx1aW50OF90ICo+KGFycmF5KSArICh3aGVyZSthbW91bnQpKm1JdGVtU2l6ZTsKLSAgICAgICAgICAgIF9kb19tb3ZlX2JhY2t3YXJkKHRvLCBmcm9tLCBzKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vIGFkanVzdCB0aGUgbnVtYmVyIG9mIGl0ZW1zLi4uCi0gICAgbUNvdW50IC09IGFtb3VudDsKLX0KLQotc2l6ZV90IFZlY3RvckltcGw6Oml0ZW1TaXplKCkgY29uc3QgewotICAgIHJldHVybiBtSXRlbVNpemU7Ci19Ci0KLXZvaWQgVmVjdG9ySW1wbDo6X2RvX2NvbnN0cnVjdCh2b2lkKiBzdG9yYWdlLCBzaXplX3QgbnVtKSBjb25zdAotewotICAgIGlmICghKG1GbGFncyAmIEhBU19UUklWSUFMX0NUT1IpKSB7Ci0gICAgICAgIGRvX2NvbnN0cnVjdChzdG9yYWdlLCBudW0pOwotICAgIH0KLX0KLQotdm9pZCBWZWN0b3JJbXBsOjpfZG9fZGVzdHJveSh2b2lkKiBzdG9yYWdlLCBzaXplX3QgbnVtKSBjb25zdAotewotICAgIGlmICghKG1GbGFncyAmIEhBU19UUklWSUFMX0RUT1IpKSB7Ci0gICAgICAgIGRvX2Rlc3Ryb3koc3RvcmFnZSwgbnVtKTsKLSAgICB9Ci19Ci0KLXZvaWQgVmVjdG9ySW1wbDo6X2RvX2NvcHkodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3QKLXsKLSAgICBpZiAoIShtRmxhZ3MgJiBIQVNfVFJJVklBTF9DT1BZKSkgewotICAgICAgICBkb19jb3B5KGRlc3QsIGZyb20sIG51bSk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgbWVtY3B5KGRlc3QsIGZyb20sIG51bSppdGVtU2l6ZSgpKTsKLSAgICB9Ci19Ci0KLXZvaWQgVmVjdG9ySW1wbDo6X2RvX3NwbGF0KHZvaWQqIGRlc3QsIGNvbnN0IHZvaWQqIGl0ZW0sIHNpemVfdCBudW0pIGNvbnN0IHsKLSAgICBkb19zcGxhdChkZXN0LCBpdGVtLCBudW0pOwotfQotCi12b2lkIFZlY3RvckltcGw6Ol9kb19tb3ZlX2ZvcndhcmQodm9pZCogZGVzdCwgY29uc3Qgdm9pZCogZnJvbSwgc2l6ZV90IG51bSkgY29uc3QgewotICAgIGRvX21vdmVfZm9yd2FyZChkZXN0LCBmcm9tLCBudW0pOwotfQotCi12b2lkIFZlY3RvckltcGw6Ol9kb19tb3ZlX2JhY2t3YXJkKHZvaWQqIGRlc3QsIGNvbnN0IHZvaWQqIGZyb20sIHNpemVfdCBudW0pIGNvbnN0IHsKLSAgICBkb19tb3ZlX2JhY2t3YXJkKGRlc3QsIGZyb20sIG51bSk7Ci19Ci0KLXZvaWQgVmVjdG9ySW1wbDo6cmVzZXJ2ZWRWZWN0b3JJbXBsMSgpIHsgfQotdm9pZCBWZWN0b3JJbXBsOjpyZXNlcnZlZFZlY3RvckltcGwyKCkgeyB9Ci12b2lkIFZlY3RvckltcGw6OnJlc2VydmVkVmVjdG9ySW1wbDMoKSB7IH0KLXZvaWQgVmVjdG9ySW1wbDo6cmVzZXJ2ZWRWZWN0b3JJbXBsNCgpIHsgfQotdm9pZCBWZWN0b3JJbXBsOjpyZXNlcnZlZFZlY3RvckltcGw1KCkgeyB9Ci12b2lkIFZlY3RvckltcGw6OnJlc2VydmVkVmVjdG9ySW1wbDYoKSB7IH0KLXZvaWQgVmVjdG9ySW1wbDo6cmVzZXJ2ZWRWZWN0b3JJbXBsNygpIHsgfQotdm9pZCBWZWN0b3JJbXBsOjpyZXNlcnZlZFZlY3RvckltcGw4KCkgeyB9Ci0KLS8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotU29ydGVkVmVjdG9ySW1wbDo6U29ydGVkVmVjdG9ySW1wbChzaXplX3QgaXRlbVNpemUsIHVpbnQzMl90IGZsYWdzKQotICAgIDogVmVjdG9ySW1wbChpdGVtU2l6ZSwgZmxhZ3MpCi17Ci19Ci0KLVNvcnRlZFZlY3RvckltcGw6OlNvcnRlZFZlY3RvckltcGwoY29uc3QgVmVjdG9ySW1wbCYgcmhzKQotOiBWZWN0b3JJbXBsKHJocykKLXsKLX0KLQotU29ydGVkVmVjdG9ySW1wbDo6flNvcnRlZFZlY3RvckltcGwoKQotewotfQotCi1Tb3J0ZWRWZWN0b3JJbXBsJiBTb3J0ZWRWZWN0b3JJbXBsOjpvcGVyYXRvciA9IChjb25zdCBTb3J0ZWRWZWN0b3JJbXBsJiByaHMpCi17Ci0gICAgcmV0dXJuIHN0YXRpY19jYXN0PFNvcnRlZFZlY3RvckltcGwmPiggVmVjdG9ySW1wbDo6b3BlcmF0b3IgPSAoc3RhdGljX2Nhc3Q8Y29uc3QgVmVjdG9ySW1wbCY+KHJocykpICk7Ci19Ci0KLXNzaXplX3QgU29ydGVkVmVjdG9ySW1wbDo6aW5kZXhPZihjb25zdCB2b2lkKiBpdGVtKSBjb25zdAotewotICAgIHJldHVybiBfaW5kZXhPcmRlck9mKGl0ZW0pOwotfQotCi1zaXplX3QgU29ydGVkVmVjdG9ySW1wbDo6b3JkZXJPZihjb25zdCB2b2lkKiBpdGVtKSBjb25zdAotewotICAgIHNpemVfdCBvOwotICAgIF9pbmRleE9yZGVyT2YoaXRlbSwgJm8pOwotICAgIHJldHVybiBvOwotfQotCi1zc2l6ZV90IFNvcnRlZFZlY3RvckltcGw6Ol9pbmRleE9yZGVyT2YoY29uc3Qgdm9pZCogaXRlbSwgc2l6ZV90KiBvcmRlcikgY29uc3QKLXsKLSAgICAvLyBiaW5hcnkgc2VhcmNoCi0gICAgc3NpemVfdCBlcnIgPSBOQU1FX05PVF9GT1VORDsKLSAgICBzc2l6ZV90IGwgPSAwOwotICAgIHNzaXplX3QgaCA9IHNpemUoKS0xOwotICAgIHNzaXplX3QgbWlkOwotICAgIGNvbnN0IHZvaWQqIGEgPSBhcnJheUltcGwoKTsKLSAgICBjb25zdCBzaXplX3QgcyA9IGl0ZW1TaXplKCk7Ci0gICAgd2hpbGUgKGwgPD0gaCkgewotICAgICAgICBtaWQgPSBsICsgKGggLSBsKS8yOwotICAgICAgICBjb25zdCB2b2lkKiBjb25zdCBjdXJyID0gcmVpbnRlcnByZXRfY2FzdDxjb25zdCBjaGFyICo+KGEpICsgKG1pZCpzKTsKLSAgICAgICAgY29uc3QgaW50IGMgPSBkb19jb21wYXJlKGN1cnIsIGl0ZW0pOwotICAgICAgICBpZiAoYyA9PSAwKSB7Ci0gICAgICAgICAgICBlcnIgPSBsID0gbWlkOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0gZWxzZSBpZiAoYyA8IDApIHsKLSAgICAgICAgICAgIGwgPSBtaWQgKyAxOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaCA9IG1pZCAtIDE7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgaWYgKG9yZGVyKSAqb3JkZXIgPSBsOwotICAgIHJldHVybiBlcnI7Ci19Ci0KLXNzaXplX3QgU29ydGVkVmVjdG9ySW1wbDo6YWRkKGNvbnN0IHZvaWQqIGl0ZW0pCi17Ci0gICAgc2l6ZV90IG9yZGVyOwotICAgIHNzaXplX3QgaW5kZXggPSBfaW5kZXhPcmRlck9mKGl0ZW0sICZvcmRlcik7Ci0gICAgaWYgKGluZGV4IDwgMCkgewotICAgICAgICBpbmRleCA9IFZlY3RvckltcGw6Omluc2VydEF0KGl0ZW0sIG9yZGVyLCAxKTsKLSAgICB9IGVsc2UgewotICAgICAgICBpbmRleCA9IFZlY3RvckltcGw6OnJlcGxhY2VBdChpdGVtLCBpbmRleCk7Ci0gICAgfQotICAgIHJldHVybiBpbmRleDsKLX0KLQotc3NpemVfdCBTb3J0ZWRWZWN0b3JJbXBsOjptZXJnZShjb25zdCBWZWN0b3JJbXBsJiB2ZWN0b3IpCi17Ci0gICAgLy8gbmFpdmUgbWVyZ2UuLi4KLSAgICBpZiAoIXZlY3Rvci5pc0VtcHR5KCkpIHsKLSAgICAgICAgY29uc3Qgdm9pZCogYnVmZmVyID0gdmVjdG9yLmFycmF5SW1wbCgpOwotICAgICAgICBjb25zdCBzaXplX3QgaXMgPSBpdGVtU2l6ZSgpOwotICAgICAgICBzaXplX3QgcyA9IHZlY3Rvci5zaXplKCk7Ci0gICAgICAgIGZvciAoc2l6ZV90IGk9MCA7IGk8cyA7IGkrKykgewotICAgICAgICAgICAgc3NpemVfdCBlcnIgPSBhZGQoIHJlaW50ZXJwcmV0X2Nhc3Q8Y29uc3QgY2hhcio+KGJ1ZmZlcikgKyBpKmlzICk7Ci0gICAgICAgICAgICBpZiAoZXJyPDApIHsKLSAgICAgICAgICAgICAgICByZXR1cm4gZXJyOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3NpemVfdCBTb3J0ZWRWZWN0b3JJbXBsOjptZXJnZShjb25zdCBTb3J0ZWRWZWN0b3JJbXBsJiB2ZWN0b3IpCi17Ci0gICAgLy8gd2UndmUgbWVyZ2luZyBhIHNvcnRlZCB2ZWN0b3IuLi4gbmljZSEKLSAgICBzc2l6ZV90IGVyciA9IE5PX0VSUk9SOwotICAgIGlmICghdmVjdG9yLmlzRW1wdHkoKSkgewotICAgICAgICAvLyBmaXJzdCB0YWtlIGNhcmUgb2YgdGhlIGNhc2Ugd2hlcmUgdGhlIHZlY3RvcnMgYXJlIHNvcnRlZCB0b2dldGhlcgotICAgICAgICBpZiAoZG9fY29tcGFyZSh2ZWN0b3IuaXRlbUxvY2F0aW9uKHZlY3Rvci5zaXplKCktMSksIGFycmF5SW1wbCgpKSA8PSAwKSB7Ci0gICAgICAgICAgICBlcnIgPSBWZWN0b3JJbXBsOjppbnNlcnRWZWN0b3JBdChzdGF0aWNfY2FzdDxjb25zdCBWZWN0b3JJbXBsJj4odmVjdG9yKSwgMCk7Ci0gICAgICAgIH0gZWxzZSBpZiAoZG9fY29tcGFyZSh2ZWN0b3IuYXJyYXlJbXBsKCksIGl0ZW1Mb2NhdGlvbihzaXplKCktMSkpID49IDApIHsKLSAgICAgICAgICAgIGVyciA9IFZlY3RvckltcGw6OmFwcGVuZFZlY3RvcihzdGF0aWNfY2FzdDxjb25zdCBWZWN0b3JJbXBsJj4odmVjdG9yKSk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyB0aGlzIGNvdWxkIGJlIG1hZGUgYSBsaXR0bGUgYmV0dGVyCi0gICAgICAgICAgICBlcnIgPSBtZXJnZShzdGF0aWNfY2FzdDxjb25zdCBWZWN0b3JJbXBsJj4odmVjdG9yKSk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIGVycjsKLX0KLQotc3NpemVfdCBTb3J0ZWRWZWN0b3JJbXBsOjpyZW1vdmUoY29uc3Qgdm9pZCogaXRlbSkKLXsKLSAgICBzc2l6ZV90IGkgPSBpbmRleE9mKGl0ZW0pOwotICAgIGlmIChpPj0wKSB7Ci0gICAgICAgIFZlY3RvckltcGw6OnJlbW92ZUl0ZW1zQXQoaSwgMSk7Ci0gICAgfQotICAgIHJldHVybiBpOwotfQotCi12b2lkIFNvcnRlZFZlY3RvckltcGw6OnJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDEoKSB7IH07Ci12b2lkIFNvcnRlZFZlY3RvckltcGw6OnJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDIoKSB7IH07Ci12b2lkIFNvcnRlZFZlY3RvckltcGw6OnJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDMoKSB7IH07Ci12b2lkIFNvcnRlZFZlY3RvckltcGw6OnJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDQoKSB7IH07Ci12b2lkIFNvcnRlZFZlY3RvckltcGw6OnJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDUoKSB7IH07Ci12b2lkIFNvcnRlZFZlY3RvckltcGw6OnJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDYoKSB7IH07Ci12b2lkIFNvcnRlZFZlY3RvckltcGw6OnJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDcoKSB7IH07Ci12b2lkIFNvcnRlZFZlY3RvckltcGw6OnJlc2VydmVkU29ydGVkVmVjdG9ySW1wbDgoKSB7IH07Ci0KLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1ppcEVudHJ5LmNwcCBiL2xpYnMvdXRpbHMvWmlwRW50cnkuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmYmM5ZTY3Li4wMDAwMDAwCi0tLSBhL2xpYnMvdXRpbHMvWmlwRW50cnkuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNjk2ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDYgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gQWNjZXNzIHRvIGVudHJpZXMgaW4gYSBaaXAgYXJjaGl2ZS4KLS8vCi0KLSNkZWZpbmUgTE9HX1RBRyAiemlwIgotCi0jaW5jbHVkZSAidXRpbHMvWmlwRW50cnkuaCIKLSNpbmNsdWRlICJ1dGlscy9Mb2cuaCIKLQotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi0vKgotICogSW5pdGlhbGl6ZSBhIG5ldyBaaXBFbnRyeSBzdHJ1Y3R1cmUgZnJvbSBhIEZJTEUqIHBvc2l0aW9uZWQgYXQgYQotICogQ2VudHJhbERpcmVjdG9yeUVudHJ5LgotICoKLSAqIE9uIGV4aXQsIHRoZSBmaWxlIHBvaW50ZXIgd2lsbCBiZSBhdCB0aGUgc3RhcnQgb2YgdGhlIG5leHQgQ0RFIG9yCi0gKiBhdCB0aGUgRU9DRC4KLSAqLwotc3RhdHVzX3QgWmlwRW50cnk6OmluaXRGcm9tQ0RFKEZJTEUqIGZwKQotewotICAgIHN0YXR1c190IHJlc3VsdDsKLSAgICBsb25nIHBvc247Ci0gICAgYm9vbCBoYXNERDsKLQotICAgIC8vTE9HVigiaW5pdEZyb21DREUgLS0tXG4iKTsKLQotICAgIC8qIHJlYWQgdGhlIENERSAqLwotICAgIHJlc3VsdCA9IG1DREUucmVhZChmcCk7Ci0gICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikgewotICAgICAgICBMT0dEKCJtQ0RFLnJlYWQgZmFpbGVkXG4iKTsKLSAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICB9Ci0KLSAgICAvL21DREUuZHVtcCgpOwotCi0gICAgLyogdXNpbmcgdGhlIGluZm8gaW4gdGhlIENERSwgZ28gbG9hZCB1cCB0aGUgTEZIICovCi0gICAgcG9zbiA9IGZ0ZWxsKGZwKTsKLSAgICBpZiAoZnNlZWsoZnAsIG1DREUubUxvY2FsSGVhZGVyUmVsT2Zmc2V0LCBTRUVLX1NFVCkgIT0gMCkgewotICAgICAgICBMT0dEKCJsb2NhbCBoZWFkZXIgc2VlayBmYWlsZWQgKCVsZClcbiIsCi0gICAgICAgICAgICBtQ0RFLm1Mb2NhbEhlYWRlclJlbE9mZnNldCk7Ci0gICAgICAgIHJldHVybiBVTktOT1dOX0VSUk9SOwotICAgIH0KLQotICAgIHJlc3VsdCA9IG1MRkgucmVhZChmcCk7Ci0gICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikgewotICAgICAgICBMT0dEKCJtTEZILnJlYWQgZmFpbGVkXG4iKTsKLSAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICB9Ci0KLSAgICBpZiAoZnNlZWsoZnAsIHBvc24sIFNFRUtfU0VUKSAhPSAwKQotICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLQotICAgIC8vbUxGSC5kdW1wKCk7Ci0KLSAgICAvKgotICAgICAqIFdlICptaWdodCogbmVlZCB0byByZWFkIHRoZSBEYXRhIERlc2NyaXB0b3IgYXQgdGhpcyBwb2ludCBhbmQKLSAgICAgKiBpbnRlZ3JhdGUgaXQgaW50byB0aGUgTEZILiAgSWYgdGhpcyBiaXQgaXMgc2V0LCB0aGUgQ1JDLTMyLAotICAgICAqIGNvbXByZXNzZWQgc2l6ZSwgYW5kIHVuY29tcHJlc3NlZCBzaXplIHdpbGwgYmUgemVyby4gIEluIHByYWN0aWNlCi0gICAgICogdGhlc2Ugc2VlbSB0byBiZSByYXJlLgotICAgICAqLwotICAgIGhhc0REID0gKG1MRkgubUdQQml0RmxhZyAmIGtVc2VzRGF0YURlc2NyKSAhPSAwOwotICAgIGlmIChoYXNERCkgewotICAgICAgICAvLyBkbyBzb21ldGhpbmcgY2xldmVyCi0gICAgICAgIC8vTE9HRCgiKysrIGhhcyBkYXRhIGRlc2NyaXB0b3JcbiIpOwotICAgIH0KLQotICAgIC8qCi0gICAgICogU2FuaXR5LWNoZWNrIHRoZSBMRkguICBOb3RlIHRoYXQgdGhpcyB3aWxsIGZhaWwgaWYgdGhlICJrVXNlc0RhdGFEZXNjciIKLSAgICAgKiBmbGFnIGlzIHNldCwgYmVjYXVzZSB0aGUgTEZIIGlzIGluY29tcGxldGUuICAoTm90IGEgcHJvYmxlbSwgc2luY2Ugd2UKLSAgICAgKiBwcmVmZXIgdGhlIENERSB2YWx1ZXMuKQotICAgICAqLwotICAgIGlmICghaGFzREQgJiYgIWNvbXBhcmVIZWFkZXJzKCkpIHsKLSAgICAgICAgTE9HVygiV0FSTklORzogaGVhZGVyIG1pc21hdGNoXG4iKTsKLSAgICAgICAgLy8ga2VlcCBnb2luZz8KLSAgICB9Ci0KLSAgICAvKgotICAgICAqIElmIHRoZSBtVmVyc2lvblRvRXh0cmFjdCBpcyBncmVhdGVyIHRoYW4gMjAsIHdlIG1heSBoYXZlIGFuCi0gICAgICogaXNzdWUgdW5wYWNraW5nIHRoZSByZWNvcmQgLS0gY291bGQgYmUgZW5jcnlwdGVkLCBjb21wcmVzc2VkCi0gICAgICogd2l0aCBzb21ldGhpbmcgd2UgZG9uJ3Qgc3VwcG9ydCwgb3IgdXNlIFppcDY0IGV4dGVuc2lvbnMuICBXZQotICAgICAqIGNhbiBkZWZlciB3b3JyeWluZyBhYm91dCB0aGF0IHRvIHdoZW4gd2UncmUgZXh0cmFjdGluZyBkYXRhLgotICAgICAqLwotCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vKgotICogSW5pdGlhbGl6ZSBhIG5ldyBlbnRyeS4gIFBhc3MgaW4gdGhlIGZpbGUgbmFtZSBhbmQgYW4gb3B0aW9uYWwgY29tbWVudC4KLSAqCi0gKiBJbml0aWFsaXplcyB0aGUgQ0RFIGFuZCB0aGUgTEZILgotICovCi12b2lkIFppcEVudHJ5Ojppbml0TmV3KGNvbnN0IGNoYXIqIGZpbGVOYW1lLCBjb25zdCBjaGFyKiBjb21tZW50KQotewotICAgIGFzc2VydChmaWxlTmFtZSAhPSBOVUxMICYmICpmaWxlTmFtZSAhPSAnXDAnKTsgIC8vIG5hbWUgcmVxdWlyZWQKLQotICAgIC8qIG1vc3QgZmllbGRzIGFyZSBwcm9wZXJseSBpbml0aWFsaXplZCBieSBjb25zdHJ1Y3RvciAqLwotICAgIG1DREUubVZlcnNpb25NYWRlQnkgPSBrRGVmYXVsdE1hZGVCeTsKLSAgICBtQ0RFLm1WZXJzaW9uVG9FeHRyYWN0ID0ga0RlZmF1bHRWZXJzaW9uOwotICAgIG1DREUubUNvbXByZXNzaW9uTWV0aG9kID0ga0NvbXByZXNzU3RvcmVkOwotICAgIG1DREUubUZpbGVOYW1lTGVuZ3RoID0gc3RybGVuKGZpbGVOYW1lKTsKLSAgICBpZiAoY29tbWVudCAhPSBOVUxMKQotICAgICAgICBtQ0RFLm1GaWxlQ29tbWVudExlbmd0aCA9IHN0cmxlbihjb21tZW50KTsKLSAgICBtQ0RFLm1FeHRlcm5hbEF0dHJzID0gMHg4MWI2MDAyMDsgICAvLyBtYXRjaGVzIHdoYXQgV2luWmlwIGRvZXMKLQotICAgIGlmIChtQ0RFLm1GaWxlTmFtZUxlbmd0aCA+IDApIHsKLSAgICAgICAgbUNERS5tRmlsZU5hbWUgPSBuZXcgdW5zaWduZWQgY2hhclttQ0RFLm1GaWxlTmFtZUxlbmd0aCsxXTsKLSAgICAgICAgc3RyY3B5KChjaGFyKikgbUNERS5tRmlsZU5hbWUsIGZpbGVOYW1lKTsKLSAgICB9Ci0gICAgaWYgKG1DREUubUZpbGVDb21tZW50TGVuZ3RoID4gMCkgewotICAgICAgICAvKiBUT0RPOiBzdG9wIGFzc3VtaW5nIG51bGwtdGVybWluYXRlZCBBU0NJSSBoZXJlPyAqLwotICAgICAgICBtQ0RFLm1GaWxlQ29tbWVudCA9IG5ldyB1bnNpZ25lZCBjaGFyW21DREUubUZpbGVDb21tZW50TGVuZ3RoKzFdOwotICAgICAgICBzdHJjcHkoKGNoYXIqKSBtQ0RFLm1GaWxlQ29tbWVudCwgY29tbWVudCk7Ci0gICAgfQotCi0gICAgY29weUNERXRvTEZIKCk7Ci19Ci0KLS8qCi0gKiBJbml0aWFsaXplIGEgbmV3IGVudHJ5LCBzdGFydGluZyB3aXRoIHRoZSBaaXBFbnRyeSBmcm9tIGEgZGlmZmVyZW50Ci0gKiBhcmNoaXZlLgotICoKLSAqIEluaXRpYWxpemVzIHRoZSBDREUgYW5kIHRoZSBMRkguCi0gKi8KLXN0YXR1c190IFppcEVudHJ5Ojppbml0RnJvbUV4dGVybmFsKGNvbnN0IFppcEZpbGUqIHBaaXBGaWxlLAotICAgIGNvbnN0IFppcEVudHJ5KiBwRW50cnkpCi17Ci0gICAgLyoKLSAgICAgKiBDb3B5IGV2ZXJ5dGhpbmcgaW4gdGhlIENERSBvdmVyLCB0aGVuIGZpeCB1cCB0aGUgaGFpcnkgYml0cy4KLSAgICAgKi8KLSAgICBtZW1jcHkoJm1DREUsICZwRW50cnktPm1DREUsIHNpemVvZihtQ0RFKSk7Ci0KLSAgICBpZiAobUNERS5tRmlsZU5hbWVMZW5ndGggPiAwKSB7Ci0gICAgICAgIG1DREUubUZpbGVOYW1lID0gbmV3IHVuc2lnbmVkIGNoYXJbbUNERS5tRmlsZU5hbWVMZW5ndGgrMV07Ci0gICAgICAgIGlmIChtQ0RFLm1GaWxlTmFtZSA9PSBOVUxMKQotICAgICAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLSAgICAgICAgc3RyY3B5KChjaGFyKikgbUNERS5tRmlsZU5hbWUsIChjaGFyKilwRW50cnktPm1DREUubUZpbGVOYW1lKTsKLSAgICB9Ci0gICAgaWYgKG1DREUubUZpbGVDb21tZW50TGVuZ3RoID4gMCkgewotICAgICAgICBtQ0RFLm1GaWxlQ29tbWVudCA9IG5ldyB1bnNpZ25lZCBjaGFyW21DREUubUZpbGVDb21tZW50TGVuZ3RoKzFdOwotICAgICAgICBpZiAobUNERS5tRmlsZUNvbW1lbnQgPT0gTlVMTCkKLSAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgIHN0cmNweSgoY2hhciopIG1DREUubUZpbGVDb21tZW50LCAoY2hhciopcEVudHJ5LT5tQ0RFLm1GaWxlQ29tbWVudCk7Ci0gICAgfQotICAgIGlmIChtQ0RFLm1FeHRyYUZpZWxkTGVuZ3RoID4gMCkgewotICAgICAgICAvKiB3ZSBudWxsLXRlcm1pbmF0ZSB0aGlzLCB0aG91Z2ggaXQgbWF5IG5vdCBiZSBhIHN0cmluZyAqLwotICAgICAgICBtQ0RFLm1FeHRyYUZpZWxkID0gbmV3IHVuc2lnbmVkIGNoYXJbbUNERS5tRXh0cmFGaWVsZExlbmd0aCsxXTsKLSAgICAgICAgaWYgKG1DREUubUV4dHJhRmllbGQgPT0gTlVMTCkKLSAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgIG1lbWNweShtQ0RFLm1FeHRyYUZpZWxkLCBwRW50cnktPm1DREUubUV4dHJhRmllbGQsCi0gICAgICAgICAgICBtQ0RFLm1FeHRyYUZpZWxkTGVuZ3RoKzEpOwotICAgIH0KLQotICAgIC8qIGNvbnN0cnVjdCB0aGUgTEZIIGZyb20gdGhlIENERSAqLwotICAgIGNvcHlDREV0b0xGSCgpOwotCi0gICAgLyoKLSAgICAgKiBUaGUgTEZIICJleHRyYSIgZmllbGQgaXMgaW5kZXBlbmRlbnQgb2YgdGhlIENERSAiZXh0cmEiLCBzbyB3ZQotICAgICAqIGhhbmRsZSBpdCBoZXJlLgotICAgICAqLwotICAgIGFzc2VydChtTEZILm1FeHRyYUZpZWxkID09IE5VTEwpOwotICAgIG1MRkgubUV4dHJhRmllbGRMZW5ndGggPSBwRW50cnktPm1MRkgubUV4dHJhRmllbGRMZW5ndGg7Ci0gICAgaWYgKG1MRkgubUV4dHJhRmllbGRMZW5ndGggPiAwKSB7Ci0gICAgICAgIG1MRkgubUV4dHJhRmllbGQgPSBuZXcgdW5zaWduZWQgY2hhclttTEZILm1FeHRyYUZpZWxkTGVuZ3RoKzFdOwotICAgICAgICBpZiAobUxGSC5tRXh0cmFGaWVsZCA9PSBOVUxMKQotICAgICAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLSAgICAgICAgbWVtY3B5KG1MRkgubUV4dHJhRmllbGQsIHBFbnRyeS0+bUxGSC5tRXh0cmFGaWVsZCwKLSAgICAgICAgICAgIG1MRkgubUV4dHJhRmllbGRMZW5ndGgrMSk7Ci0gICAgfQotCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vKgotICogSW5zZXJ0IHBhZCBieXRlcyBpbiB0aGUgTEZIIGJ5IHR3ZWFraW5nIHRoZSAiZXh0cmEiIGZpZWxkLiAgVGhpcyB3aWxsCi0gKiBwb3RlbnRpYWxseSBjb25mdXNlIHNvbWV0aGluZyB0aGF0IHB1dCAiZXh0cmEiIGRhdGEgaW4gaGVyZSBlYXJsaWVyLAotICogYnV0IEkgY2FuJ3QgZmluZCBhbiBhY3R1YWwgcHJvYmxlbS4KLSAqLwotc3RhdHVzX3QgWmlwRW50cnk6OmFkZFBhZGRpbmcoaW50IHBhZGRpbmcpCi17Ci0gICAgaWYgKHBhZGRpbmcgPD0gMCkKLSAgICAgICAgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOwotCi0gICAgLy9MT0dJKCJIRVk6IGFkZGluZyAlZCBwYWQgYnl0ZXMgdG8gZXhpc3RpbmcgJWQgaW4gJXNcbiIsCi0gICAgLy8gICAgcGFkZGluZywgbUxGSC5tRXh0cmFGaWVsZExlbmd0aCwgbUNERS5tRmlsZU5hbWUpOwotCi0gICAgaWYgKG1MRkgubUV4dHJhRmllbGRMZW5ndGggPiAwKSB7Ci0gICAgICAgIC8qIGV4dGVuZCBleGlzdGluZyBmaWVsZCAqLwotICAgICAgICB1bnNpZ25lZCBjaGFyKiBuZXdFeHRyYTsKLQotICAgICAgICBuZXdFeHRyYSA9IG5ldyB1bnNpZ25lZCBjaGFyW21MRkgubUV4dHJhRmllbGRMZW5ndGggKyBwYWRkaW5nXTsKLSAgICAgICAgaWYgKG5ld0V4dHJhID09IE5VTEwpCi0gICAgICAgICAgICByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICBtZW1zZXQobmV3RXh0cmEgKyBtTEZILm1FeHRyYUZpZWxkTGVuZ3RoLCAwLCBwYWRkaW5nKTsKLSAgICAgICAgbWVtY3B5KG5ld0V4dHJhLCBtTEZILm1FeHRyYUZpZWxkLCBtTEZILm1FeHRyYUZpZWxkTGVuZ3RoKTsKLQotICAgICAgICBkZWxldGVbXSBtTEZILm1FeHRyYUZpZWxkOwotICAgICAgICBtTEZILm1FeHRyYUZpZWxkID0gbmV3RXh0cmE7Ci0gICAgICAgIG1MRkgubUV4dHJhRmllbGRMZW5ndGggKz0gcGFkZGluZzsKLSAgICB9IGVsc2UgewotICAgICAgICAvKiBjcmVhdGUgbmV3IGZpZWxkICovCi0gICAgICAgIG1MRkgubUV4dHJhRmllbGQgPSBuZXcgdW5zaWduZWQgY2hhcltwYWRkaW5nXTsKLSAgICAgICAgbWVtc2V0KG1MRkgubUV4dHJhRmllbGQsIDAsIHBhZGRpbmcpOwotICAgICAgICBtTEZILm1FeHRyYUZpZWxkTGVuZ3RoID0gcGFkZGluZzsKLSAgICB9Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLS8qCi0gKiBTZXQgdGhlIGZpZWxkcyBpbiB0aGUgTEZIIGVxdWFsIHRvIHRoZSBjb3JyZXNwb25kaW5nIGZpZWxkcyBpbiB0aGUgQ0RFLgotICoKLSAqIFRoaXMgZG9lcyBub3QgdG91Y2ggdGhlIExGSCAiZXh0cmEiIGZpZWxkLgotICovCi12b2lkIFppcEVudHJ5Ojpjb3B5Q0RFdG9MRkgodm9pZCkKLXsKLSAgICBtTEZILm1WZXJzaW9uVG9FeHRyYWN0ICA9IG1DREUubVZlcnNpb25Ub0V4dHJhY3Q7Ci0gICAgbUxGSC5tR1BCaXRGbGFnICAgICAgICAgPSBtQ0RFLm1HUEJpdEZsYWc7Ci0gICAgbUxGSC5tQ29tcHJlc3Npb25NZXRob2QgPSBtQ0RFLm1Db21wcmVzc2lvbk1ldGhvZDsKLSAgICBtTEZILm1MYXN0TW9kRmlsZVRpbWUgICA9IG1DREUubUxhc3RNb2RGaWxlVGltZTsKLSAgICBtTEZILm1MYXN0TW9kRmlsZURhdGUgICA9IG1DREUubUxhc3RNb2RGaWxlRGF0ZTsKLSAgICBtTEZILm1DUkMzMiAgICAgICAgICAgICA9IG1DREUubUNSQzMyOwotICAgIG1MRkgubUNvbXByZXNzZWRTaXplICAgID0gbUNERS5tQ29tcHJlc3NlZFNpemU7Ci0gICAgbUxGSC5tVW5jb21wcmVzc2VkU2l6ZSAgPSBtQ0RFLm1VbmNvbXByZXNzZWRTaXplOwotICAgIG1MRkgubUZpbGVOYW1lTGVuZ3RoICAgID0gbUNERS5tRmlsZU5hbWVMZW5ndGg7Ci0gICAgLy8gdGhlICJleHRyYSBmaWVsZCIgaXMgaW5kZXBlbmRlbnQKLQotICAgIGRlbGV0ZVtdIG1MRkgubUZpbGVOYW1lOwotICAgIGlmIChtTEZILm1GaWxlTmFtZUxlbmd0aCA+IDApIHsKLSAgICAgICAgbUxGSC5tRmlsZU5hbWUgPSBuZXcgdW5zaWduZWQgY2hhclttTEZILm1GaWxlTmFtZUxlbmd0aCsxXTsKLSAgICAgICAgc3RyY3B5KChjaGFyKikgbUxGSC5tRmlsZU5hbWUsIChjb25zdCBjaGFyKikgbUNERS5tRmlsZU5hbWUpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIG1MRkgubUZpbGVOYW1lID0gTlVMTDsKLSAgICB9Ci19Ci0KLS8qCi0gKiBTZXQgc29tZSBpbmZvcm1hdGlvbiBhYm91dCBhIGZpbGUgYWZ0ZXIgd2UgYWRkIGl0LgotICovCi12b2lkIFppcEVudHJ5OjpzZXREYXRhSW5mbyhsb25nIHVuY29tcExlbiwgbG9uZyBjb21wTGVuLCB1bnNpZ25lZCBsb25nIGNyYzMyLAotICAgIGludCBjb21wcmVzc2lvbk1ldGhvZCkKLXsKLSAgICBtQ0RFLm1Db21wcmVzc2lvbk1ldGhvZCA9IGNvbXByZXNzaW9uTWV0aG9kOwotICAgIG1DREUubUNSQzMyID0gY3JjMzI7Ci0gICAgbUNERS5tQ29tcHJlc3NlZFNpemUgPSBjb21wTGVuOwotICAgIG1DREUubVVuY29tcHJlc3NlZFNpemUgPSB1bmNvbXBMZW47Ci0gICAgbUNERS5tQ29tcHJlc3Npb25NZXRob2QgPSBjb21wcmVzc2lvbk1ldGhvZDsKLSAgICBpZiAoY29tcHJlc3Npb25NZXRob2QgPT0ga0NvbXByZXNzRGVmbGF0ZWQpIHsKLSAgICAgICAgbUNERS5tR1BCaXRGbGFnIHw9IDB4MDAwMjsgICAgICAvLyBpbmRpY2F0ZXMgbWF4aW11bSBjb21wcmVzc2lvbiB1c2VkCi0gICAgfQotICAgIGNvcHlDREV0b0xGSCgpOwotfQotCi0vKgotICogU2VlIGlmIHRoZSBkYXRhIGluIG1DREUgYW5kIG1MRkggbWF0Y2ggdXAuICBUaGlzIGlzIG1vc3RseSB1c2VmdWwgZm9yCi0gKiBkZWJ1Z2dpbmcgdGhlc2UgY2xhc3NlcywgYnV0IGl0IGNhbiBiZSB1c2VkIHRvIGlkZW50aWZ5IGRhbWFnZWQKLSAqIGFyY2hpdmVzLgotICoKLSAqIFJldHVybnMgImZhbHNlIiBpZiB0aGV5IGRpZmZlci4KLSAqLwotYm9vbCBaaXBFbnRyeTo6Y29tcGFyZUhlYWRlcnModm9pZCkgY29uc3QKLXsKLSAgICBpZiAobUNERS5tVmVyc2lvblRvRXh0cmFjdCAhPSBtTEZILm1WZXJzaW9uVG9FeHRyYWN0KSB7Ci0gICAgICAgIExPR1YoImNtcDogVmVyc2lvblRvRXh0cmFjdFxuIik7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgaWYgKG1DREUubUdQQml0RmxhZyAhPSBtTEZILm1HUEJpdEZsYWcpIHsKLSAgICAgICAgTE9HVigiY21wOiBHUEJpdEZsYWdcbiIpOwotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotICAgIGlmIChtQ0RFLm1Db21wcmVzc2lvbk1ldGhvZCAhPSBtTEZILm1Db21wcmVzc2lvbk1ldGhvZCkgewotICAgICAgICBMT0dWKCJjbXA6IENvbXByZXNzaW9uTWV0aG9kXG4iKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSAgICBpZiAobUNERS5tTGFzdE1vZEZpbGVUaW1lICE9IG1MRkgubUxhc3RNb2RGaWxlVGltZSkgewotICAgICAgICBMT0dWKCJjbXA6IExhc3RNb2RGaWxlVGltZVxuIik7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgaWYgKG1DREUubUxhc3RNb2RGaWxlRGF0ZSAhPSBtTEZILm1MYXN0TW9kRmlsZURhdGUpIHsKLSAgICAgICAgTE9HVigiY21wOiBMYXN0TW9kRmlsZURhdGVcbiIpOwotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotICAgIGlmIChtQ0RFLm1DUkMzMiAhPSBtTEZILm1DUkMzMikgewotICAgICAgICBMT0dWKCJjbXA6IENSQzMyXG4iKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSAgICBpZiAobUNERS5tQ29tcHJlc3NlZFNpemUgIT0gbUxGSC5tQ29tcHJlc3NlZFNpemUpIHsKLSAgICAgICAgTE9HVigiY21wOiBDb21wcmVzc2VkU2l6ZVxuIik7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgaWYgKG1DREUubVVuY29tcHJlc3NlZFNpemUgIT0gbUxGSC5tVW5jb21wcmVzc2VkU2l6ZSkgewotICAgICAgICBMT0dWKCJjbXA6IFVuY29tcHJlc3NlZFNpemVcbiIpOwotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotICAgIGlmIChtQ0RFLm1GaWxlTmFtZUxlbmd0aCAhPSBtTEZILm1GaWxlTmFtZUxlbmd0aCkgewotICAgICAgICBMT0dWKCJjbXA6IEZpbGVOYW1lTGVuZ3RoXG4iKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSNpZiAwICAgICAgIC8vIHRoaXMgc2VlbXMgdG8gYmUgdXNlZCBmb3IgcGFkZGluZywgbm90IHJlYWwgZGF0YQotICAgIGlmIChtQ0RFLm1FeHRyYUZpZWxkTGVuZ3RoICE9IG1MRkgubUV4dHJhRmllbGRMZW5ndGgpIHsKLSAgICAgICAgTE9HVigiY21wOiBFeHRyYUZpZWxkTGVuZ3RoXG4iKTsKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIH0KLSNlbmRpZgotICAgIGlmIChtQ0RFLm1GaWxlTmFtZSAhPSBOVUxMKSB7Ci0gICAgICAgIGlmIChzdHJjbXAoKGNoYXIqKSBtQ0RFLm1GaWxlTmFtZSwgKGNoYXIqKSBtTEZILm1GaWxlTmFtZSkgIT0gMCkgewotICAgICAgICAgICAgTE9HVigiY21wOiBGaWxlTmFtZVxuIik7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQotCi0vKgotICogQ29udmVydCB0aGUgRE9TIGRhdGUvdGltZSBzdGFtcCBpbnRvIGEgVU5JWCB0aW1lIHN0YW1wLgotICovCi10aW1lX3QgWmlwRW50cnk6OmdldE1vZFdoZW4odm9pZCkgY29uc3QKLXsKLSAgICBzdHJ1Y3QgdG0gcGFydHM7Ci0KLSAgICBwYXJ0cy50bV9zZWMgPSAobUNERS5tTGFzdE1vZEZpbGVUaW1lICYgMHgwMDFmKSA8PCAxOwotICAgIHBhcnRzLnRtX21pbiA9IChtQ0RFLm1MYXN0TW9kRmlsZVRpbWUgJiAweDA3ZTApID4+IDU7Ci0gICAgcGFydHMudG1faG91ciA9IChtQ0RFLm1MYXN0TW9kRmlsZVRpbWUgJiAweGY4MDApID4+IDExOwotICAgIHBhcnRzLnRtX21kYXkgPSAobUNERS5tTGFzdE1vZEZpbGVEYXRlICYgMHgwMDFmKTsKLSAgICBwYXJ0cy50bV9tb24gPSAoKG1DREUubUxhc3RNb2RGaWxlRGF0ZSAmIDB4MDFlMCkgPj4gNSkgLTE7Ci0gICAgcGFydHMudG1feWVhciA9ICgobUNERS5tTGFzdE1vZEZpbGVEYXRlICYgMHhmZTAwKSA+PiA5KSArIDgwOwotICAgIHBhcnRzLnRtX3dkYXkgPSBwYXJ0cy50bV95ZGF5ID0gMDsKLSAgICBwYXJ0cy50bV9pc2RzdCA9IC0xOyAgICAgICAgLy8gRFNUIGluZm8gIm5vdCBhdmFpbGFibGUiCi0KLSAgICByZXR1cm4gbWt0aW1lKCZwYXJ0cyk7Ci19Ci0KLS8qCi0gKiBTZXQgdGhlIENERS9MRkggdGltZXN0YW1wIGZyb20gVU5JWCB0aW1lLgotICovCi12b2lkIFppcEVudHJ5OjpzZXRNb2RXaGVuKHRpbWVfdCB3aGVuKQotewotI2lmZGVmIEhBVkVfTE9DQUxUSU1FX1IKLSAgICBzdHJ1Y3QgdG0gdG1SZXN1bHQ7Ci0jZW5kaWYKLSAgICB0aW1lX3QgZXZlbjsKLSAgICB1bnNpZ25lZCBzaG9ydCB6ZGF0ZSwgenRpbWU7Ci0KLSAgICBzdHJ1Y3QgdG0qIHB0bTsKLQotICAgIC8qIHJvdW5kIHVwIHRvIGFuIGV2ZW4gbnVtYmVyIG9mIHNlY29uZHMgKi8KLSAgICBldmVuID0gKHRpbWVfdCkoKCh1bnNpZ25lZCBsb25nKSh3aGVuKSArIDEpICYgKH4xKSk7Ci0KLSAgICAvKiBleHBhbmQgKi8KLSNpZmRlZiBIQVZFX0xPQ0FMVElNRV9SCi0gICAgcHRtID0gbG9jYWx0aW1lX3IoJmV2ZW4sICZ0bVJlc3VsdCk7Ci0jZWxzZQotICAgIHB0bSA9IGxvY2FsdGltZSgmZXZlbik7Ci0jZW5kaWYKLQotICAgIGludCB5ZWFyOwotICAgIHllYXIgPSBwdG0tPnRtX3llYXI7Ci0gICAgaWYgKHllYXIgPCA4MCkKLSAgICAgICAgeWVhciA9IDgwOwotCi0gICAgemRhdGUgPSAoeWVhciAtIDgwKSA8PCA5IHwgKHB0bS0+dG1fbW9uKzEpIDw8IDUgfCBwdG0tPnRtX21kYXk7Ci0gICAgenRpbWUgPSBwdG0tPnRtX2hvdXIgPDwgMTEgfCBwdG0tPnRtX21pbiA8PCA1IHwgcHRtLT50bV9zZWMgPj4gMTsKLQotICAgIG1DREUubUxhc3RNb2RGaWxlVGltZSA9IG1MRkgubUxhc3RNb2RGaWxlVGltZSA9IHp0aW1lOwotICAgIG1DREUubUxhc3RNb2RGaWxlRGF0ZSA9IG1MRkgubUxhc3RNb2RGaWxlRGF0ZSA9IHpkYXRlOwotfQotCi0KLS8qCi0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqICAgICAgWmlwRW50cnk6OkxvY2FsRmlsZUhlYWRlcgotICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci0gKi8KLQotLyoKLSAqIFJlYWQgYSBsb2NhbCBmaWxlIGhlYWRlci4KLSAqCi0gKiBPbiBlbnRyeSwgImZwIiBwb2ludHMgdG8gdGhlIHNpZ25hdHVyZSBhdCB0aGUgc3RhcnQgb2YgdGhlIGhlYWRlci4KLSAqIE9uIGV4aXQsICJmcCIgcG9pbnRzIHRvIHRoZSBzdGFydCBvZiBkYXRhLgotICovCi1zdGF0dXNfdCBaaXBFbnRyeTo6TG9jYWxGaWxlSGVhZGVyOjpyZWFkKEZJTEUqIGZwKQotewotICAgIHN0YXR1c190IHJlc3VsdCA9IE5PX0VSUk9SOwotICAgIHVuc2lnbmVkIGNoYXIgYnVmW2tMRkhMZW5dOwotCi0gICAgYXNzZXJ0KG1GaWxlTmFtZSA9PSBOVUxMKTsKLSAgICBhc3NlcnQobUV4dHJhRmllbGQgPT0gTlVMTCk7Ci0KLSAgICBpZiAoZnJlYWQoYnVmLCAxLCBrTEZITGVuLCBmcCkgIT0ga0xGSExlbikgewotICAgICAgICByZXN1bHQgPSBVTktOT1dOX0VSUk9SOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgaWYgKFppcEVudHJ5OjpnZXRMb25nTEUoJmJ1ZlsweDAwXSkgIT0ga1NpZ25hdHVyZSkgewotICAgICAgICBMT0dEKCJ3aG9vcHM6IGRpZG4ndCBmaW5kIGV4cGVjdGVkIHNpZ25hdHVyZVxuIik7Ci0gICAgICAgIHJlc3VsdCA9IFVOS05PV05fRVJST1I7Ci0gICAgICAgIGdvdG8gYmFpbDsKLSAgICB9Ci0KLSAgICBtVmVyc2lvblRvRXh0cmFjdCA9IFppcEVudHJ5OjpnZXRTaG9ydExFKCZidWZbMHgwNF0pOwotICAgIG1HUEJpdEZsYWcgPSBaaXBFbnRyeTo6Z2V0U2hvcnRMRSgmYnVmWzB4MDZdKTsKLSAgICBtQ29tcHJlc3Npb25NZXRob2QgPSBaaXBFbnRyeTo6Z2V0U2hvcnRMRSgmYnVmWzB4MDhdKTsKLSAgICBtTGFzdE1vZEZpbGVUaW1lID0gWmlwRW50cnk6OmdldFNob3J0TEUoJmJ1ZlsweDBhXSk7Ci0gICAgbUxhc3RNb2RGaWxlRGF0ZSA9IFppcEVudHJ5OjpnZXRTaG9ydExFKCZidWZbMHgwY10pOwotICAgIG1DUkMzMiA9IFppcEVudHJ5OjpnZXRMb25nTEUoJmJ1ZlsweDBlXSk7Ci0gICAgbUNvbXByZXNzZWRTaXplID0gWmlwRW50cnk6OmdldExvbmdMRSgmYnVmWzB4MTJdKTsKLSAgICBtVW5jb21wcmVzc2VkU2l6ZSA9IFppcEVudHJ5OjpnZXRMb25nTEUoJmJ1ZlsweDE2XSk7Ci0gICAgbUZpbGVOYW1lTGVuZ3RoID0gWmlwRW50cnk6OmdldFNob3J0TEUoJmJ1ZlsweDFhXSk7Ci0gICAgbUV4dHJhRmllbGRMZW5ndGggPSBaaXBFbnRyeTo6Z2V0U2hvcnRMRSgmYnVmWzB4MWNdKTsKLQotICAgIC8vIFRPRE86IHZhbGlkYXRlIHNpemVzCi0KLSAgICAvKiBncmFiIGZpbGVuYW1lICovCi0gICAgaWYgKG1GaWxlTmFtZUxlbmd0aCAhPSAwKSB7Ci0gICAgICAgIG1GaWxlTmFtZSA9IG5ldyB1bnNpZ25lZCBjaGFyW21GaWxlTmFtZUxlbmd0aCsxXTsKLSAgICAgICAgaWYgKG1GaWxlTmFtZSA9PSBOVUxMKSB7Ci0gICAgICAgICAgICByZXN1bHQgPSBOT19NRU1PUlk7Ci0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGZyZWFkKG1GaWxlTmFtZSwgMSwgbUZpbGVOYW1lTGVuZ3RoLCBmcCkgIT0gbUZpbGVOYW1lTGVuZ3RoKSB7Ci0gICAgICAgICAgICByZXN1bHQgPSBVTktOT1dOX0VSUk9SOwotICAgICAgICAgICAgZ290byBiYWlsOwotICAgICAgICB9Ci0gICAgICAgIG1GaWxlTmFtZVttRmlsZU5hbWVMZW5ndGhdID0gJ1wwJzsKLSAgICB9Ci0KLSAgICAvKiBncmFiIGV4dHJhIGZpZWxkICovCi0gICAgaWYgKG1FeHRyYUZpZWxkTGVuZ3RoICE9IDApIHsKLSAgICAgICAgbUV4dHJhRmllbGQgPSBuZXcgdW5zaWduZWQgY2hhclttRXh0cmFGaWVsZExlbmd0aCsxXTsKLSAgICAgICAgaWYgKG1FeHRyYUZpZWxkID09IE5VTEwpIHsKLSAgICAgICAgICAgIHJlc3VsdCA9IE5PX01FTU9SWTsKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgfQotICAgICAgICBpZiAoZnJlYWQobUV4dHJhRmllbGQsIDEsIG1FeHRyYUZpZWxkTGVuZ3RoLCBmcCkgIT0gbUV4dHJhRmllbGRMZW5ndGgpIHsKLSAgICAgICAgICAgIHJlc3VsdCA9IFVOS05PV05fRVJST1I7Ci0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgICAgIH0KLSAgICAgICAgbUV4dHJhRmllbGRbbUV4dHJhRmllbGRMZW5ndGhdID0gJ1wwJzsKLSAgICB9Ci0KLWJhaWw6Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotLyoKLSAqIFdyaXRlIGEgbG9jYWwgZmlsZSBoZWFkZXIuCi0gKi8KLXN0YXR1c190IFppcEVudHJ5OjpMb2NhbEZpbGVIZWFkZXI6OndyaXRlKEZJTEUqIGZwKQotewotICAgIHVuc2lnbmVkIGNoYXIgYnVmW2tMRkhMZW5dOwotCi0gICAgWmlwRW50cnk6OnB1dExvbmdMRSgmYnVmWzB4MDBdLCBrU2lnbmF0dXJlKTsKLSAgICBaaXBFbnRyeTo6cHV0U2hvcnRMRSgmYnVmWzB4MDRdLCBtVmVyc2lvblRvRXh0cmFjdCk7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDA2XSwgbUdQQml0RmxhZyk7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDA4XSwgbUNvbXByZXNzaW9uTWV0aG9kKTsKLSAgICBaaXBFbnRyeTo6cHV0U2hvcnRMRSgmYnVmWzB4MGFdLCBtTGFzdE1vZEZpbGVUaW1lKTsKLSAgICBaaXBFbnRyeTo6cHV0U2hvcnRMRSgmYnVmWzB4MGNdLCBtTGFzdE1vZEZpbGVEYXRlKTsKLSAgICBaaXBFbnRyeTo6cHV0TG9uZ0xFKCZidWZbMHgwZV0sIG1DUkMzMik7Ci0gICAgWmlwRW50cnk6OnB1dExvbmdMRSgmYnVmWzB4MTJdLCBtQ29tcHJlc3NlZFNpemUpOwotICAgIFppcEVudHJ5OjpwdXRMb25nTEUoJmJ1ZlsweDE2XSwgbVVuY29tcHJlc3NlZFNpemUpOwotICAgIFppcEVudHJ5OjpwdXRTaG9ydExFKCZidWZbMHgxYV0sIG1GaWxlTmFtZUxlbmd0aCk7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDFjXSwgbUV4dHJhRmllbGRMZW5ndGgpOwotCi0gICAgaWYgKGZ3cml0ZShidWYsIDEsIGtMRkhMZW4sIGZwKSAhPSBrTEZITGVuKQotICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLQotICAgIC8qIHdyaXRlIGZpbGVuYW1lICovCi0gICAgaWYgKG1GaWxlTmFtZUxlbmd0aCAhPSAwKSB7Ci0gICAgICAgIGlmIChmd3JpdGUobUZpbGVOYW1lLCAxLCBtRmlsZU5hbWVMZW5ndGgsIGZwKSAhPSBtRmlsZU5hbWVMZW5ndGgpCi0gICAgICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICB9Ci0KLSAgICAvKiB3cml0ZSAiZXh0cmEgZmllbGQiICovCi0gICAgaWYgKG1FeHRyYUZpZWxkTGVuZ3RoICE9IDApIHsKLSAgICAgICAgaWYgKGZ3cml0ZShtRXh0cmFGaWVsZCwgMSwgbUV4dHJhRmllbGRMZW5ndGgsIGZwKSAhPSBtRXh0cmFGaWVsZExlbmd0aCkKLSAgICAgICAgICAgIHJldHVybiBVTktOT1dOX0VSUk9SOwotICAgIH0KLQotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotCi0vKgotICogRHVtcCB0aGUgY29udGVudHMgb2YgYSBMb2NhbEZpbGVIZWFkZXIgb2JqZWN0LgotICovCi12b2lkIFppcEVudHJ5OjpMb2NhbEZpbGVIZWFkZXI6OmR1bXAodm9pZCkgY29uc3QKLXsKLSAgICBMT0dEKCIgTG9jYWxGaWxlSGVhZGVyIGNvbnRlbnRzOlxuIik7Ci0gICAgTE9HRCgiICB2ZXJzVG9FeHQ9JXUgZ3BCaXRzPTB4JTA0eCBjb21wcmVzc2lvbj0ldVxuIiwKLSAgICAgICAgbVZlcnNpb25Ub0V4dHJhY3QsIG1HUEJpdEZsYWcsIG1Db21wcmVzc2lvbk1ldGhvZCk7Ci0gICAgTE9HRCgiICBtb2RUaW1lPTB4JTA0eCBtb2REYXRlPTB4JTA0eCBjcmMzMj0weCUwOGx4XG4iLAotICAgICAgICBtTGFzdE1vZEZpbGVUaW1lLCBtTGFzdE1vZEZpbGVEYXRlLCBtQ1JDMzIpOwotICAgIExPR0QoIiAgY29tcHJlc3NlZFNpemU9JWx1IHVuY29tcHJlc3NlZFNpemU9JWx1XG4iLAotICAgICAgICBtQ29tcHJlc3NlZFNpemUsIG1VbmNvbXByZXNzZWRTaXplKTsKLSAgICBMT0dEKCIgIGZpbGVuYW1lTGVuPSV1IGV4dHJhTGVuPSV1XG4iLAotICAgICAgICBtRmlsZU5hbWVMZW5ndGgsIG1FeHRyYUZpZWxkTGVuZ3RoKTsKLSAgICBpZiAobUZpbGVOYW1lICE9IE5VTEwpCi0gICAgICAgIExPR0QoIiAgZmlsZW5hbWU6ICclcydcbiIsIG1GaWxlTmFtZSk7Ci19Ci0KLQotLyoKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICogICAgICBaaXBFbnRyeTo6Q2VudHJhbERpckVudHJ5Ci0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqLwotCi0vKgotICogUmVhZCB0aGUgY2VudHJhbCBkaXIgZW50cnkgdGhhdCBhcHBlYXJzIG5leHQgaW4gdGhlIGZpbGUuCi0gKgotICogT24gZW50cnksICJmcCIgc2hvdWxkIGJlIHBvc2l0aW9uZWQgb24gdGhlIHNpZ25hdHVyZSBieXRlcyBmb3IgdGhlCi0gKiBlbnRyeS4gIE9uIGV4aXQsICJmcCIgd2lsbCBwb2ludCBhdCB0aGUgc2lnbmF0dXJlIHdvcmQgZm9yIHRoZSBuZXh0Ci0gKiBlbnRyeSBvciBmb3IgdGhlIEVPQ0QuCi0gKi8KLXN0YXR1c190IFppcEVudHJ5OjpDZW50cmFsRGlyRW50cnk6OnJlYWQoRklMRSogZnApCi17Ci0gICAgc3RhdHVzX3QgcmVzdWx0ID0gTk9fRVJST1I7Ci0gICAgdW5zaWduZWQgY2hhciBidWZba0NERUxlbl07Ci0KLSAgICAvKiBubyByZS11c2UgKi8KLSAgICBhc3NlcnQobUZpbGVOYW1lID09IE5VTEwpOwotICAgIGFzc2VydChtRXh0cmFGaWVsZCA9PSBOVUxMKTsKLSAgICBhc3NlcnQobUZpbGVDb21tZW50ID09IE5VTEwpOwotCi0gICAgaWYgKGZyZWFkKGJ1ZiwgMSwga0NERUxlbiwgZnApICE9IGtDREVMZW4pIHsKLSAgICAgICAgcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIGlmIChaaXBFbnRyeTo6Z2V0TG9uZ0xFKCZidWZbMHgwMF0pICE9IGtTaWduYXR1cmUpIHsKLSAgICAgICAgTE9HRCgiV2hvb3BzOiBkaWRuJ3QgZmluZCBleHBlY3RlZCBzaWduYXR1cmVcbiIpOwotICAgICAgICByZXN1bHQgPSBVTktOT1dOX0VSUk9SOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgbVZlcnNpb25NYWRlQnkgPSBaaXBFbnRyeTo6Z2V0U2hvcnRMRSgmYnVmWzB4MDRdKTsKLSAgICBtVmVyc2lvblRvRXh0cmFjdCA9IFppcEVudHJ5OjpnZXRTaG9ydExFKCZidWZbMHgwNl0pOwotICAgIG1HUEJpdEZsYWcgPSBaaXBFbnRyeTo6Z2V0U2hvcnRMRSgmYnVmWzB4MDhdKTsKLSAgICBtQ29tcHJlc3Npb25NZXRob2QgPSBaaXBFbnRyeTo6Z2V0U2hvcnRMRSgmYnVmWzB4MGFdKTsKLSAgICBtTGFzdE1vZEZpbGVUaW1lID0gWmlwRW50cnk6OmdldFNob3J0TEUoJmJ1ZlsweDBjXSk7Ci0gICAgbUxhc3RNb2RGaWxlRGF0ZSA9IFppcEVudHJ5OjpnZXRTaG9ydExFKCZidWZbMHgwZV0pOwotICAgIG1DUkMzMiA9IFppcEVudHJ5OjpnZXRMb25nTEUoJmJ1ZlsweDEwXSk7Ci0gICAgbUNvbXByZXNzZWRTaXplID0gWmlwRW50cnk6OmdldExvbmdMRSgmYnVmWzB4MTRdKTsKLSAgICBtVW5jb21wcmVzc2VkU2l6ZSA9IFppcEVudHJ5OjpnZXRMb25nTEUoJmJ1ZlsweDE4XSk7Ci0gICAgbUZpbGVOYW1lTGVuZ3RoID0gWmlwRW50cnk6OmdldFNob3J0TEUoJmJ1ZlsweDFjXSk7Ci0gICAgbUV4dHJhRmllbGRMZW5ndGggPSBaaXBFbnRyeTo6Z2V0U2hvcnRMRSgmYnVmWzB4MWVdKTsKLSAgICBtRmlsZUNvbW1lbnRMZW5ndGggPSBaaXBFbnRyeTo6Z2V0U2hvcnRMRSgmYnVmWzB4MjBdKTsKLSAgICBtRGlza051bWJlclN0YXJ0ID0gWmlwRW50cnk6OmdldFNob3J0TEUoJmJ1ZlsweDIyXSk7Ci0gICAgbUludGVybmFsQXR0cnMgPSBaaXBFbnRyeTo6Z2V0U2hvcnRMRSgmYnVmWzB4MjRdKTsKLSAgICBtRXh0ZXJuYWxBdHRycyA9IFppcEVudHJ5OjpnZXRMb25nTEUoJmJ1ZlsweDI2XSk7Ci0gICAgbUxvY2FsSGVhZGVyUmVsT2Zmc2V0ID0gWmlwRW50cnk6OmdldExvbmdMRSgmYnVmWzB4MmFdKTsKLQotICAgIC8vIFRPRE86IHZhbGlkYXRlIHNpemVzIGFuZCBvZmZzZXRzCi0KLSAgICAvKiBncmFiIGZpbGVuYW1lICovCi0gICAgaWYgKG1GaWxlTmFtZUxlbmd0aCAhPSAwKSB7Ci0gICAgICAgIG1GaWxlTmFtZSA9IG5ldyB1bnNpZ25lZCBjaGFyW21GaWxlTmFtZUxlbmd0aCsxXTsKLSAgICAgICAgaWYgKG1GaWxlTmFtZSA9PSBOVUxMKSB7Ci0gICAgICAgICAgICByZXN1bHQgPSBOT19NRU1PUlk7Ci0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKGZyZWFkKG1GaWxlTmFtZSwgMSwgbUZpbGVOYW1lTGVuZ3RoLCBmcCkgIT0gbUZpbGVOYW1lTGVuZ3RoKSB7Ci0gICAgICAgICAgICByZXN1bHQgPSBVTktOT1dOX0VSUk9SOwotICAgICAgICAgICAgZ290byBiYWlsOwotICAgICAgICB9Ci0gICAgICAgIG1GaWxlTmFtZVttRmlsZU5hbWVMZW5ndGhdID0gJ1wwJzsKLSAgICB9Ci0KLSAgICAvKiByZWFkICJleHRyYSBmaWVsZCIgKi8KLSAgICBpZiAobUV4dHJhRmllbGRMZW5ndGggIT0gMCkgewotICAgICAgICBtRXh0cmFGaWVsZCA9IG5ldyB1bnNpZ25lZCBjaGFyW21FeHRyYUZpZWxkTGVuZ3RoKzFdOwotICAgICAgICBpZiAobUV4dHJhRmllbGQgPT0gTlVMTCkgewotICAgICAgICAgICAgcmVzdWx0ID0gTk9fTUVNT1JZOwotICAgICAgICAgICAgZ290byBiYWlsOwotICAgICAgICB9Ci0gICAgICAgIGlmIChmcmVhZChtRXh0cmFGaWVsZCwgMSwgbUV4dHJhRmllbGRMZW5ndGgsIGZwKSAhPSBtRXh0cmFGaWVsZExlbmd0aCkgewotICAgICAgICAgICAgcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgfQotICAgICAgICBtRXh0cmFGaWVsZFttRXh0cmFGaWVsZExlbmd0aF0gPSAnXDAnOwotICAgIH0KLQotCi0gICAgLyogZ3JhYiBjb21tZW50LCBpZiBhbnkgKi8KLSAgICBpZiAobUZpbGVDb21tZW50TGVuZ3RoICE9IDApIHsKLSAgICAgICAgbUZpbGVDb21tZW50ID0gbmV3IHVuc2lnbmVkIGNoYXJbbUZpbGVDb21tZW50TGVuZ3RoKzFdOwotICAgICAgICBpZiAobUZpbGVDb21tZW50ID09IE5VTEwpIHsKLSAgICAgICAgICAgIHJlc3VsdCA9IE5PX01FTU9SWTsKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgfQotICAgICAgICBpZiAoZnJlYWQobUZpbGVDb21tZW50LCAxLCBtRmlsZUNvbW1lbnRMZW5ndGgsIGZwKSAhPSBtRmlsZUNvbW1lbnRMZW5ndGgpCi0gICAgICAgIHsKLSAgICAgICAgICAgIHJlc3VsdCA9IFVOS05PV05fRVJST1I7Ci0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgICAgIH0KLSAgICAgICAgbUZpbGVDb21tZW50W21GaWxlQ29tbWVudExlbmd0aF0gPSAnXDAnOwotICAgIH0KLQotYmFpbDoKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi0vKgotICogV3JpdGUgYSBjZW50cmFsIGRpciBlbnRyeS4KLSAqLwotc3RhdHVzX3QgWmlwRW50cnk6OkNlbnRyYWxEaXJFbnRyeTo6d3JpdGUoRklMRSogZnApCi17Ci0gICAgdW5zaWduZWQgY2hhciBidWZba0NERUxlbl07Ci0KLSAgICBaaXBFbnRyeTo6cHV0TG9uZ0xFKCZidWZbMHgwMF0sIGtTaWduYXR1cmUpOwotICAgIFppcEVudHJ5OjpwdXRTaG9ydExFKCZidWZbMHgwNF0sIG1WZXJzaW9uTWFkZUJ5KTsKLSAgICBaaXBFbnRyeTo6cHV0U2hvcnRMRSgmYnVmWzB4MDZdLCBtVmVyc2lvblRvRXh0cmFjdCk7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDA4XSwgbUdQQml0RmxhZyk7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDBhXSwgbUNvbXByZXNzaW9uTWV0aG9kKTsKLSAgICBaaXBFbnRyeTo6cHV0U2hvcnRMRSgmYnVmWzB4MGNdLCBtTGFzdE1vZEZpbGVUaW1lKTsKLSAgICBaaXBFbnRyeTo6cHV0U2hvcnRMRSgmYnVmWzB4MGVdLCBtTGFzdE1vZEZpbGVEYXRlKTsKLSAgICBaaXBFbnRyeTo6cHV0TG9uZ0xFKCZidWZbMHgxMF0sIG1DUkMzMik7Ci0gICAgWmlwRW50cnk6OnB1dExvbmdMRSgmYnVmWzB4MTRdLCBtQ29tcHJlc3NlZFNpemUpOwotICAgIFppcEVudHJ5OjpwdXRMb25nTEUoJmJ1ZlsweDE4XSwgbVVuY29tcHJlc3NlZFNpemUpOwotICAgIFppcEVudHJ5OjpwdXRTaG9ydExFKCZidWZbMHgxY10sIG1GaWxlTmFtZUxlbmd0aCk7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDFlXSwgbUV4dHJhRmllbGRMZW5ndGgpOwotICAgIFppcEVudHJ5OjpwdXRTaG9ydExFKCZidWZbMHgyMF0sIG1GaWxlQ29tbWVudExlbmd0aCk7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDIyXSwgbURpc2tOdW1iZXJTdGFydCk7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDI0XSwgbUludGVybmFsQXR0cnMpOwotICAgIFppcEVudHJ5OjpwdXRMb25nTEUoJmJ1ZlsweDI2XSwgbUV4dGVybmFsQXR0cnMpOwotICAgIFppcEVudHJ5OjpwdXRMb25nTEUoJmJ1ZlsweDJhXSwgbUxvY2FsSGVhZGVyUmVsT2Zmc2V0KTsKLQotICAgIGlmIChmd3JpdGUoYnVmLCAxLCBrQ0RFTGVuLCBmcCkgIT0ga0NERUxlbikKLSAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0KLSAgICAvKiB3cml0ZSBmaWxlbmFtZSAqLwotICAgIGlmIChtRmlsZU5hbWVMZW5ndGggIT0gMCkgewotICAgICAgICBpZiAoZndyaXRlKG1GaWxlTmFtZSwgMSwgbUZpbGVOYW1lTGVuZ3RoLCBmcCkgIT0gbUZpbGVOYW1lTGVuZ3RoKQotICAgICAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0gICAgfQotCi0gICAgLyogd3JpdGUgImV4dHJhIGZpZWxkIiAqLwotICAgIGlmIChtRXh0cmFGaWVsZExlbmd0aCAhPSAwKSB7Ci0gICAgICAgIGlmIChmd3JpdGUobUV4dHJhRmllbGQsIDEsIG1FeHRyYUZpZWxkTGVuZ3RoLCBmcCkgIT0gbUV4dHJhRmllbGRMZW5ndGgpCi0gICAgICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICB9Ci0KLSAgICAvKiB3cml0ZSBjb21tZW50ICovCi0gICAgaWYgKG1GaWxlQ29tbWVudExlbmd0aCAhPSAwKSB7Ci0gICAgICAgIGlmIChmd3JpdGUobUZpbGVDb21tZW50LCAxLCBtRmlsZUNvbW1lbnRMZW5ndGgsIGZwKSAhPSBtRmlsZUNvbW1lbnRMZW5ndGgpCi0gICAgICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICB9Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLS8qCi0gKiBEdW1wIHRoZSBjb250ZW50cyBvZiBhIENlbnRyYWxEaXJFbnRyeSBvYmplY3QuCi0gKi8KLXZvaWQgWmlwRW50cnk6OkNlbnRyYWxEaXJFbnRyeTo6ZHVtcCh2b2lkKSBjb25zdAotewotICAgIExPR0QoIiBDZW50cmFsRGlyRW50cnkgY29udGVudHM6XG4iKTsKLSAgICBMT0dEKCIgIHZlcnNNYWRlQnk9JXUgdmVyc1RvRXh0PSV1IGdwQml0cz0weCUwNHggY29tcHJlc3Npb249JXVcbiIsCi0gICAgICAgIG1WZXJzaW9uTWFkZUJ5LCBtVmVyc2lvblRvRXh0cmFjdCwgbUdQQml0RmxhZywgbUNvbXByZXNzaW9uTWV0aG9kKTsKLSAgICBMT0dEKCIgIG1vZFRpbWU9MHglMDR4IG1vZERhdGU9MHglMDR4IGNyYzMyPTB4JTA4bHhcbiIsCi0gICAgICAgIG1MYXN0TW9kRmlsZVRpbWUsIG1MYXN0TW9kRmlsZURhdGUsIG1DUkMzMik7Ci0gICAgTE9HRCgiICBjb21wcmVzc2VkU2l6ZT0lbHUgdW5jb21wcmVzc2VkU2l6ZT0lbHVcbiIsCi0gICAgICAgIG1Db21wcmVzc2VkU2l6ZSwgbVVuY29tcHJlc3NlZFNpemUpOwotICAgIExPR0QoIiAgZmlsZW5hbWVMZW49JXUgZXh0cmFMZW49JXUgY29tbWVudExlbj0ldVxuIiwKLSAgICAgICAgbUZpbGVOYW1lTGVuZ3RoLCBtRXh0cmFGaWVsZExlbmd0aCwgbUZpbGVDb21tZW50TGVuZ3RoKTsKLSAgICBMT0dEKCIgIGRpc2tOdW1TdGFydD0ldSBpbnRBdHRyPTB4JTA0eCBleHRBdHRyPTB4JTA4bHggcmVsT2Zmc2V0PSVsdVxuIiwKLSAgICAgICAgbURpc2tOdW1iZXJTdGFydCwgbUludGVybmFsQXR0cnMsIG1FeHRlcm5hbEF0dHJzLAotICAgICAgICBtTG9jYWxIZWFkZXJSZWxPZmZzZXQpOwotCi0gICAgaWYgKG1GaWxlTmFtZSAhPSBOVUxMKQotICAgICAgICBMT0dEKCIgIGZpbGVuYW1lOiAnJXMnXG4iLCBtRmlsZU5hbWUpOwotICAgIGlmIChtRmlsZUNvbW1lbnQgIT0gTlVMTCkKLSAgICAgICAgTE9HRCgiICBjb21tZW50OiAnJXMnXG4iLCBtRmlsZUNvbW1lbnQpOwotfQotCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1ppcEZpbGUuY3BwIGIvbGlicy91dGlscy9aaXBGaWxlLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggODlhYTg3NC4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1ppcEZpbGUuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTI5NiArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA2IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIEFjY2VzcyB0byBaaXAgYXJjaGl2ZXMuCi0vLwotCi0jZGVmaW5lIExPR19UQUcgInppcCIKLQotI2luY2x1ZGUgInV0aWxzL1ppcEZpbGUuaCIKLSNpbmNsdWRlICJ1dGlscy9aaXBVdGlscy5oIgotI2luY2x1ZGUgInV0aWxzL0xvZy5oIgotCi0jaW5jbHVkZSA8emxpYi5oPgotI2RlZmluZSBERUZfTUVNX0xFVkVMIDggICAgICAgICAgICAgICAgLy8gbm9ybWFsbHkgaW4genV0aWwuaD8KLQotI2luY2x1ZGUgPG1lbW9yeS5oPgotI2luY2x1ZGUgPHN5cy9zdGF0Lmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxhc3NlcnQuaD4KLQotdXNpbmcgbmFtZXNwYWNlIGFuZHJvaWQ7Ci0KLS8qCi0gKiBTb21lIGVudmlyb25tZW50cyByZXF1aXJlIHRoZSAiYiIsIHNvbWUgY2hva2Ugb24gaXQuCi0gKi8KLSNkZWZpbmUgRklMRV9PUEVOX1JPICAgICAgICAicmIiCi0jZGVmaW5lIEZJTEVfT1BFTl9SVyAgICAgICAgInIrYiIKLSNkZWZpbmUgRklMRV9PUEVOX1JXX0NSRUFURSAidytiIgotCi0vKiBzaG91bGQgbGl2ZSBzb21ld2hlcmUgZWxzZT8gKi8KLXN0YXRpYyBzdGF0dXNfdCBlcnJub1RvU3RhdHVzKGludCBlcnIpCi17Ci0gICAgaWYgKGVyciA9PSBFTk9FTlQpCi0gICAgICAgIHJldHVybiBOQU1FX05PVF9GT1VORDsKLSAgICBlbHNlIGlmIChlcnIgPT0gRUFDQ0VTKQotICAgICAgICByZXR1cm4gUEVSTUlTU0lPTl9ERU5JRUQ7Ci0gICAgZWxzZQotICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLX0KLQotLyoKLSAqIE9wZW4gYSBmaWxlIGFuZCBwYXJzZSBpdHMgZ3V0cy4KLSAqLwotc3RhdHVzX3QgWmlwRmlsZTo6b3Blbihjb25zdCBjaGFyKiB6aXBGaWxlTmFtZSwgaW50IGZsYWdzKQotewotICAgIGJvb2wgbmV3QXJjaGl2ZSA9IGZhbHNlOwotCi0gICAgYXNzZXJ0KG1aaXBGcCA9PSBOVUxMKTsgICAgIC8vIG5vIHJlb3BlbgotCi0gICAgaWYgKChmbGFncyAmIGtPcGVuVHJ1bmNhdGUpKQotICAgICAgICBmbGFncyB8PSBrT3BlbkNyZWF0ZTsgICAgICAgICAgIC8vIHRydW5jIGltcGxpZXMgY3JlYXRlCi0KLSAgICBpZiAoKGZsYWdzICYga09wZW5SZWFkT25seSkgJiYgKGZsYWdzICYga09wZW5SZWFkV3JpdGUpKQotICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047ICAgICAgIC8vIG5vdCBib3RoCi0gICAgaWYgKCEoKGZsYWdzICYga09wZW5SZWFkT25seSkgfHwgKGZsYWdzICYga09wZW5SZWFkV3JpdGUpKSkKLSAgICAgICAgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOyAgICAgICAvLyBub3QgbmVpdGhlcgotICAgIGlmICgoZmxhZ3MgJiBrT3BlbkNyZWF0ZSkgJiYgIShmbGFncyAmIGtPcGVuUmVhZFdyaXRlKSkKLSAgICAgICAgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOyAgICAgICAvLyBjcmVhdGUgcmVxdWlyZXMgd3JpdGUKLQotICAgIGlmIChmbGFncyAmIGtPcGVuVHJ1bmNhdGUpIHsKLSAgICAgICAgbmV3QXJjaGl2ZSA9IHRydWU7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgbmV3QXJjaGl2ZSA9IChhY2Nlc3MoemlwRmlsZU5hbWUsIEZfT0spICE9IDApOwotICAgICAgICBpZiAoIShmbGFncyAmIGtPcGVuQ3JlYXRlKSAmJiBuZXdBcmNoaXZlKSB7Ci0gICAgICAgICAgICAvKiBub3QgY3JlYXRpbmcsIG11c3QgYWxyZWFkeSBleGlzdCAqLwotICAgICAgICAgICAgTE9HRCgiRmlsZSAlcyBkb2VzIG5vdCBleGlzdCIsIHppcEZpbGVOYW1lKTsKLSAgICAgICAgICAgIHJldHVybiBOQU1FX05PVF9GT1VORDsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8qIG9wZW4gdGhlIGZpbGUgKi8KLSAgICBjb25zdCBjaGFyKiBvcGVuZmxhZ3M7Ci0gICAgaWYgKGZsYWdzICYga09wZW5SZWFkV3JpdGUpIHsKLSAgICAgICAgaWYgKG5ld0FyY2hpdmUpCi0gICAgICAgICAgICBvcGVuZmxhZ3MgPSBGSUxFX09QRU5fUldfQ1JFQVRFOwotICAgICAgICBlbHNlCi0gICAgICAgICAgICBvcGVuZmxhZ3MgPSBGSUxFX09QRU5fUlc7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgb3BlbmZsYWdzID0gRklMRV9PUEVOX1JPOwotICAgIH0KLSAgICBtWmlwRnAgPSBmb3Blbih6aXBGaWxlTmFtZSwgb3BlbmZsYWdzKTsKLSAgICBpZiAobVppcEZwID09IE5VTEwpIHsKLQkJaW50IGVyciA9IGVycm5vOwotCQlMT0dEKCJmb3BlbiBmYWlsZWQ6ICVkXG4iLCBlcnIpOwotICAgICAgICByZXR1cm4gZXJybm9Ub1N0YXR1cyhlcnIpOwotCX0KLQotICAgIHN0YXR1c190IHJlc3VsdDsKLSAgICBpZiAoIW5ld0FyY2hpdmUpIHsKLSAgICAgICAgLyoKLSAgICAgICAgICogTG9hZCB0aGUgY2VudHJhbCBkaXJlY3RvcnkuICBJZiB0aGF0IGZhaWxzLCB0aGVuIHRoaXMgcHJvYmFibHkKLSAgICAgICAgICogaXNuJ3QgYSBaaXAgYXJjaGl2ZS4KLSAgICAgICAgICovCi0gICAgICAgIHJlc3VsdCA9IHJlYWRDZW50cmFsRGlyKCk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgLyoKLSAgICAgICAgICogTmV3bHktY3JlYXRlZC4gIFRoZSBFbmRPZkNlbnRyYWxEaXIgY29uc3RydWN0b3IgYWN0dWFsbHkKLSAgICAgICAgICogc2V0cyBldmVyeXRoaW5nIHRvIGJlIHRoZSB3YXkgd2Ugd2FudCBpdCAoYWxsIHplcm9lcykuICBXZQotICAgICAgICAgKiBzZXQgbU5lZWRDRFJld3JpdGUgc28gdGhhdCB3ZSBjcmVhdGUgKnNvbWV0aGluZyogaWYgdGhlCi0gICAgICAgICAqIGNhbGxlciBkb2Vzbid0IGFkZCBhbnkgZmlsZXMuICAoV2UgY291bGQgYWxzbyBqdXN0IHVubGluawotICAgICAgICAgKiB0aGUgZmlsZSBpZiBpdCdzIGJyYW5kIG5ldyBhbmQgbm90aGluZyB3YXMgYWRkZWQsIGJ1dCB0aGF0J3MKLSAgICAgICAgICogcHJvYmFibHkgZG9pbmcgbW9yZSB0aGFuIHdlIHJlYWxseSBzaG91bGQgLS0gdGhlIHVzZXIgbWlnaHQKLSAgICAgICAgICogaGF2ZSBhIG5lZWQgZm9yIGVtcHR5IHppcCBmaWxlcy4pCi0gICAgICAgICAqLwotICAgICAgICBtTmVlZENEUmV3cml0ZSA9IHRydWU7Ci0gICAgICAgIHJlc3VsdCA9IE5PX0VSUk9SOwotICAgIH0KLQotICAgIGlmIChmbGFncyAmIGtPcGVuUmVhZE9ubHkpCi0gICAgICAgIG1SZWFkT25seSA9IHRydWU7Ci0gICAgZWxzZQotICAgICAgICBhc3NlcnQoIW1SZWFkT25seSk7Ci0KLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi0vKgotICogUmV0dXJuIHRoZSBOdGggZW50cnkgaW4gdGhlIGFyY2hpdmUuCi0gKi8KLVppcEVudHJ5KiBaaXBGaWxlOjpnZXRFbnRyeUJ5SW5kZXgoaW50IGlkeCkgY29uc3QKLXsKLSAgICBpZiAoaWR4IDwgMCB8fCBpZHggPj0gKGludCkgbUVudHJpZXMuc2l6ZSgpKQotICAgICAgICByZXR1cm4gTlVMTDsKLQotICAgIHJldHVybiBtRW50cmllc1tpZHhdOwotfQotCi0vKgotICogRmluZCBhbiBlbnRyeSBieSBuYW1lLgotICovCi1aaXBFbnRyeSogWmlwRmlsZTo6Z2V0RW50cnlCeU5hbWUoY29uc3QgY2hhciogZmlsZU5hbWUpIGNvbnN0Ci17Ci0gICAgLyoKLSAgICAgKiBEbyBhIHN0dXBpZCBsaW5lYXIgc3RyaW5nLWNvbXBhcmUgc2VhcmNoLgotICAgICAqCi0gICAgICogVGhlcmUgYXJlIHZhcmlvdXMgd2F5cyB0byBzcGVlZCB0aGlzIHVwLCBlc3BlY2lhbGx5IHNpbmNlIGl0J3MgcmFyZQotICAgICAqIHRvIGludGVybWluZ2xlIGNoYW5nZXMgdG8gdGhlIGFyY2hpdmUgd2l0aCAiZ2V0IGJ5IG5hbWUiIGNhbGxzLiAgV2UKLSAgICAgKiBkb24ndCB3YW50IHRvIHNvcnQgdGhlIG1FbnRyaWVzIHZlY3RvciBpdHNlbGYsIGhvd2V2ZXIsIGJlY2F1c2UKLSAgICAgKiBpdCdzIHVzZWQgdG8gcmVjcmVhdGUgdGhlIENlbnRyYWwgRGlyZWN0b3J5LgotICAgICAqCi0gICAgICogKEhhc2ggdGFibGUgd29ya3MsIHBhcmFsbGVsIGxpc3Qgb2YgcG9pbnRlcnMgaW4gc29ydGVkIG9yZGVyIGlzIGdvb2QuKQotICAgICAqLwotICAgIGludCBpZHg7Ci0KLSAgICBmb3IgKGlkeCA9IG1FbnRyaWVzLnNpemUoKS0xOyBpZHggPj0gMDsgaWR4LS0pIHsKLSAgICAgICAgWmlwRW50cnkqIHBFbnRyeSA9IG1FbnRyaWVzW2lkeF07Ci0gICAgICAgIGlmICghcEVudHJ5LT5nZXREZWxldGVkKCkgJiYKLSAgICAgICAgICAgIHN0cmNtcChmaWxlTmFtZSwgcEVudHJ5LT5nZXRGaWxlTmFtZSgpKSA9PSAwKQotICAgICAgICB7Ci0gICAgICAgICAgICByZXR1cm4gcEVudHJ5OwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLS8qCi0gKiBFbXB0eSB0aGUgbUVudHJpZXMgdmVjdG9yLgotICovCi12b2lkIFppcEZpbGU6OmRpc2NhcmRFbnRyaWVzKHZvaWQpCi17Ci0gICAgaW50IGNvdW50ID0gbUVudHJpZXMuc2l6ZSgpOwotCi0gICAgd2hpbGUgKC0tY291bnQgPj0gMCkKLSAgICAgICAgZGVsZXRlIG1FbnRyaWVzW2NvdW50XTsKLQotICAgIG1FbnRyaWVzLmNsZWFyKCk7Ci19Ci0KLQotLyoKLSAqIEZpbmQgdGhlIGNlbnRyYWwgZGlyZWN0b3J5IGFuZCByZWFkIHRoZSBjb250ZW50cy4KLSAqCi0gKiBUaGUgZnVuIHRoaW5nIGFib3V0IFpJUCBhcmNoaXZlcyBpcyB0aGF0IHRoZXkgbWF5IG9yIG1heSBub3QgYmUKLSAqIHJlYWRhYmxlIGZyb20gc3RhcnQgdG8gZW5kLiAgSW4gc29tZSBjYXNlcywgbm90YWJseSBmb3IgYXJjaGl2ZXMKLSAqIHRoYXQgd2VyZSB3cml0dGVuIHRvIHN0ZG91dCwgdGhlIG9ubHkgbGVuZ3RoIGluZm9ybWF0aW9uIGlzIGluIHRoZQotICogY2VudHJhbCBkaXJlY3RvcnkgYXQgdGhlIGVuZCBvZiB0aGUgZmlsZS4KLSAqCi0gKiBPZiBjb3Vyc2UsIHRoZSBjZW50cmFsIGRpcmVjdG9yeSBjYW4gYmUgZm9sbG93ZWQgYnkgYSB2YXJpYWJsZS1sZW5ndGgKLSAqIGNvbW1lbnQgZmllbGQsIHNvIHdlIGhhdmUgdG8gc2NhbiB0aHJvdWdoIGl0IGJhY2t3YXJkcy4gIFRoZSBjb21tZW50Ci0gKiBpcyBhdCBtb3N0IDY0SywgcGx1cyB3ZSBoYXZlIDE4IGJ5dGVzIGZvciB0aGUgZW5kLW9mLWNlbnRyYWwtZGlyIHN0dWZmCi0gKiBpdHNlbGYsIHBsdXMgYXBwYXJlbnRseSBzb21ldGltZXMgcGVvcGxlIHRocm93IHJhbmRvbSBqdW5rIG9uIHRoZSBlbmQKLSAqIGp1c3QgZm9yIHRoZSBmdW4gb2YgaXQuCi0gKgotICogVGhpcyBpcyBhbGwgYSBsaXR0bGUgd29iYmx5LiAgSWYgdGhlIHdyb25nIHZhbHVlIGVuZHMgdXAgaW4gdGhlIEVPQ0QKLSAqIGFyZWEsIHdlJ3JlIGhvc2VkLiAgVGhpcyBhcHBlYXJzIHRvIGJlIHRoZSB3YXkgdGhhdCBldmVyYm9keSBoYW5kbGVzCi0gKiBpdCB0aG91Z2gsIHNvIHdlJ3JlIGluIHByZXR0eSBnb29kIGNvbXBhbnkgaWYgdGhpcyBmYWlscy4KLSAqLwotc3RhdHVzX3QgWmlwRmlsZTo6cmVhZENlbnRyYWxEaXIodm9pZCkKLXsKLSAgICBzdGF0dXNfdCByZXN1bHQgPSBOT19FUlJPUjsKLSAgICB1bnNpZ25lZCBjaGFyKiBidWYgPSBOVUxMOwotICAgIG9mZl90IGZpbGVMZW5ndGgsIHNlZWtTdGFydDsKLSAgICBsb25nIHJlYWRBbW91bnQ7Ci0gICAgaW50IGk7Ci0KLSAgICBmc2VlayhtWmlwRnAsIDAsIFNFRUtfRU5EKTsKLSAgICBmaWxlTGVuZ3RoID0gZnRlbGwobVppcEZwKTsKLSAgICByZXdpbmQobVppcEZwKTsKLQotICAgIC8qIHRvbyBzbWFsbCB0byBiZSBhIFpJUCBhcmNoaXZlPyAqLwotICAgIGlmIChmaWxlTGVuZ3RoIDwgRW5kT2ZDZW50cmFsRGlyOjprRU9DRExlbikgewotICAgICAgICBMT0dEKCJMZW5ndGggaXMgJWxkIC0tIHRvbyBzbWFsbFxuIiwgKGxvbmcpZmlsZUxlbmd0aCk7Ci0gICAgICAgIHJlc3VsdCA9IElOVkFMSURfT1BFUkFUSU9OOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgYnVmID0gbmV3IHVuc2lnbmVkIGNoYXJbRW5kT2ZDZW50cmFsRGlyOjprTWF4RU9DRFNlYXJjaF07Ci0gICAgaWYgKGJ1ZiA9PSBOVUxMKSB7Ci0JCUxPR0QoIkZhaWx1cmUgYWxsb2NhdGluZyAlZCBieXRlcyBmb3IgRU9DRCBzZWFyY2giLAotCQkJIEVuZE9mQ2VudHJhbERpcjo6a01heEVPQ0RTZWFyY2gpOwotICAgICAgICByZXN1bHQgPSBOT19NRU1PUlk7Ci0gICAgICAgIGdvdG8gYmFpbDsKLSAgICB9Ci0KLSAgICBpZiAoZmlsZUxlbmd0aCA+IEVuZE9mQ2VudHJhbERpcjo6a01heEVPQ0RTZWFyY2gpIHsKLSAgICAgICAgc2Vla1N0YXJ0ID0gZmlsZUxlbmd0aCAtIEVuZE9mQ2VudHJhbERpcjo6a01heEVPQ0RTZWFyY2g7Ci0gICAgICAgIHJlYWRBbW91bnQgPSBFbmRPZkNlbnRyYWxEaXI6OmtNYXhFT0NEU2VhcmNoOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHNlZWtTdGFydCA9IDA7Ci0gICAgICAgIHJlYWRBbW91bnQgPSAobG9uZykgZmlsZUxlbmd0aDsKLSAgICB9Ci0gICAgaWYgKGZzZWVrKG1aaXBGcCwgc2Vla1N0YXJ0LCBTRUVLX1NFVCkgIT0gMCkgewotCQlMT0dEKCJGYWlsdXJlIHNlZWtpbmcgdG8gZW5kIG9mIHppcCBhdCAlbGQiLCAobG9uZykgc2Vla1N0YXJ0KTsKLSAgICAgICAgcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIC8qIHJlYWQgdGhlIGxhc3QgcGFydCBvZiB0aGUgZmlsZSBpbnRvIHRoZSBidWZmZXIgKi8KLSAgICBpZiAoZnJlYWQoYnVmLCAxLCByZWFkQW1vdW50LCBtWmlwRnApICE9IChzaXplX3QpIHJlYWRBbW91bnQpIHsKLSAgICAgICAgTE9HRCgic2hvcnQgZmlsZT8gd2FudGVkICVsZFxuIiwgcmVhZEFtb3VudCk7Ci0gICAgICAgIHJlc3VsdCA9IFVOS05PV05fRVJST1I7Ci0gICAgICAgIGdvdG8gYmFpbDsKLSAgICB9Ci0KLSAgICAvKiBmaW5kIHRoZSBlbmQtb2YtY2VudHJhbC1kaXIgbWFnaWMgKi8KLSAgICBmb3IgKGkgPSByZWFkQW1vdW50IC0gNDsgaSA+PSAwOyBpLS0pIHsKLSAgICAgICAgaWYgKGJ1ZltpXSA9PSAweDUwICYmCi0gICAgICAgICAgICBaaXBFbnRyeTo6Z2V0TG9uZ0xFKCZidWZbaV0pID09IEVuZE9mQ2VudHJhbERpcjo6a1NpZ25hdHVyZSkKLSAgICAgICAgewotICAgICAgICAgICAgTE9HVigiKysrIEZvdW5kIEVPQ0QgYXQgYnVmKyVkXG4iLCBpKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgfQotICAgIGlmIChpIDwgMCkgewotICAgICAgICBMT0dEKCJFT0NEIG5vdCBmb3VuZCwgbm90IFppcFxuIik7Ci0gICAgICAgIHJlc3VsdCA9IElOVkFMSURfT1BFUkFUSU9OOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgLyogZXh0cmFjdCBlb2NkIHZhbHVlcyAqLwotICAgIHJlc3VsdCA9IG1FT0NELnJlYWRCdWYoYnVmICsgaSwgcmVhZEFtb3VudCAtIGkpOwotICAgIGlmIChyZXN1bHQgIT0gTk9fRVJST1IpIHsKLQkJTE9HRCgiRmFpbHVyZSByZWFkaW5nICVsZCBieXRlcyBvZiBFT0NEIHZhbHVlcyIsIHJlYWRBbW91bnQgLSBpKTsKLSAgICAgICAgZ290byBiYWlsOwotCX0KLSAgICAvL21FT0NELmR1bXAoKTsKLQotICAgIGlmIChtRU9DRC5tRGlza051bWJlciAhPSAwIHx8IG1FT0NELm1EaXNrV2l0aENlbnRyYWxEaXIgIT0gMCB8fAotICAgICAgICBtRU9DRC5tTnVtRW50cmllcyAhPSBtRU9DRC5tVG90YWxOdW1FbnRyaWVzKQotICAgIHsKLSAgICAgICAgTE9HRCgiQXJjaGl2ZSBzcGFubmluZyBub3Qgc3VwcG9ydGVkXG4iKTsKLSAgICAgICAgcmVzdWx0ID0gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgICAgIGdvdG8gYmFpbDsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIFNvIGZhciBzbyBnb29kLiAgIm1DZW50cmFsRGlyU2l6ZSIgaXMgdGhlIHNpemUgaW4gYnl0ZXMgb2YgdGhlCi0gICAgICogY2VudHJhbCBkaXJlY3RvcnksIHNvIHdlIGNhbiBqdXN0IHNlZWsgYmFjayB0aGF0IGZhciB0byBmaW5kIGl0LgotICAgICAqIFdlIGNhbiBhbHNvIHNlZWsgZm9yd2FyZCBtQ2VudHJhbERpck9mZnNldCBieXRlcyBmcm9tIHRoZQotICAgICAqIHN0YXJ0IG9mIHRoZSBmaWxlLgotICAgICAqCi0gICAgICogV2UncmUgbm90IGd1YXJhbnRlZWQgdG8gaGF2ZSB0aGUgcmVzdCBvZiB0aGUgY2VudHJhbCBkaXIgaW4gdGhlCi0gICAgICogYnVmZmVyLCBub3IgYXJlIHdlIGd1YXJhbnRlZWQgdGhhdCB0aGUgY2VudHJhbCBkaXIgd2lsbCBoYXZlIGFueQotICAgICAqIHNvcnQgb2YgY29udmVuaWVudCBzaXplLiAgV2UgbmVlZCB0byBza2lwIHRvIHRoZSBzdGFydCBvZiBpdCBhbmQKLSAgICAgKiByZWFkIHRoZSBoZWFkZXIsIHRoZW4gdGhlIG90aGVyIGdvb2RpZXMuCi0gICAgICoKLSAgICAgKiBUaGUgb25seSB0aGluZyB3ZSByZWFsbHkgbmVlZCByaWdodCBub3cgaXMgdGhlIGZpbGUgY29tbWVudCwgd2hpY2gKLSAgICAgKiB3ZSdyZSBob3BpbmcgdG8gcHJlc2VydmUuCi0gICAgICovCi0gICAgaWYgKGZzZWVrKG1aaXBGcCwgbUVPQ0QubUNlbnRyYWxEaXJPZmZzZXQsIFNFRUtfU0VUKSAhPSAwKSB7Ci0JCUxPR0QoIkZhaWx1cmUgc2Vla2luZyB0byBjZW50cmFsIGRpciBvZmZzZXQgJWxkXG4iLAotCQkJIG1FT0NELm1DZW50cmFsRGlyT2Zmc2V0KTsKLSAgICAgICAgcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIC8qCi0gICAgICogTG9vcCB0aHJvdWdoIGFuZCByZWFkIHRoZSBjZW50cmFsIGRpciBlbnRyaWVzLgotICAgICAqLwotICAgIExPR1YoIlNjYW5uaW5nICVkIGVudHJpZXMuLi5cbiIsIG1FT0NELm1Ub3RhbE51bUVudHJpZXMpOwotICAgIGludCBlbnRyeTsKLSAgICBmb3IgKGVudHJ5ID0gMDsgZW50cnkgPCBtRU9DRC5tVG90YWxOdW1FbnRyaWVzOyBlbnRyeSsrKSB7Ci0gICAgICAgIFppcEVudHJ5KiBwRW50cnkgPSBuZXcgWmlwRW50cnk7Ci0KLSAgICAgICAgcmVzdWx0ID0gcEVudHJ5LT5pbml0RnJvbUNERShtWmlwRnApOwotICAgICAgICBpZiAocmVzdWx0ICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgICAgICBMT0dEKCJpbml0RnJvbUNERSBmYWlsZWRcbiIpOwotICAgICAgICAgICAgZGVsZXRlIHBFbnRyeTsKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgfQotCi0gICAgICAgIG1FbnRyaWVzLmFkZChwRW50cnkpOwotICAgIH0KLQotCi0gICAgLyoKLSAgICAgKiBJZiBhbGwgd2VudCB3ZWxsLCB3ZSBzaG91bGQgbm93IGJlIGJhY2sgYXQgdGhlIEVPQ0QuCi0gICAgICovCi0gICAgewotICAgICAgICB1bnNpZ25lZCBjaGFyIGNoZWNrQnVmWzRdOwotICAgICAgICBpZiAoZnJlYWQoY2hlY2tCdWYsIDEsIDQsIG1aaXBGcCkgIT0gNCkgewotICAgICAgICAgICAgTE9HRCgiRU9DRCBjaGVjayByZWFkIGZhaWxlZFxuIik7Ci0gICAgICAgICAgICByZXN1bHQgPSBJTlZBTElEX09QRVJBVElPTjsKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgfQotICAgICAgICBpZiAoWmlwRW50cnk6OmdldExvbmdMRShjaGVja0J1ZikgIT0gRW5kT2ZDZW50cmFsRGlyOjprU2lnbmF0dXJlKSB7Ci0gICAgICAgICAgICBMT0dEKCJFT0NEIHJlYWQgY2hlY2sgZmFpbGVkXG4iKTsKLSAgICAgICAgICAgIHJlc3VsdCA9IFVOS05PV05fRVJST1I7Ci0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgICAgIH0KLSAgICAgICAgTE9HVigiKysrIEVPQ0QgcmVhZCBjaGVjayBwYXNzZWRcbiIpOwotICAgIH0KLQotYmFpbDoKLSAgICBkZWxldGVbXSBidWY7Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotCi0vKgotICogQWRkIGEgbmV3IGZpbGUgdG8gdGhlIGFyY2hpdmUuCi0gKgotICogVGhpcyByZXF1aXJlcyBjcmVhdGluZyBhbmQgcG9wdWxhdGluZyBhIFppcEVudHJ5IHN0cnVjdHVyZSwgYW5kIGNvcHlpbmcKLSAqIHRoZSBkYXRhIGludG8gdGhlIGZpbGUgYXQgdGhlIGFwcHJvcHJpYXRlIHBvc2l0aW9uLiAgVGhlICJhcHByb3ByaWF0ZQotICogcG9zaXRpb24iIGlzIHRoZSBjdXJyZW50IGxvY2F0aW9uIG9mIHRoZSBjZW50cmFsIGRpcmVjdG9yeSwgd2hpY2ggd2UKLSAqIGNhc3VhbGx5IG92ZXJ3cml0ZSAod2UgY2FuIHB1dCBpdCBiYWNrIGxhdGVyKS4KLSAqCi0gKiBJZiB3ZSB3ZXJlIGNvbmNlcm5lZCBhYm91dCBzYWZldHksIHdlIHdvdWxkIHdhbnQgdG8gbWFrZSBhbGwgY2hhbmdlcwotICogaW4gYSB0ZW1wIGZpbGUgYW5kIHRoZW4gb3ZlcndyaXRlIHRoZSBvcmlnaW5hbCBhZnRlciBldmVyeXRoaW5nIHdhcwotICogc2FmZWx5IHdyaXR0ZW4uICBOb3QgcmVhbGx5IGEgY29uY2VybiBmb3IgdXMuCi0gKi8KLXN0YXR1c190IFppcEZpbGU6OmFkZENvbW1vbihjb25zdCBjaGFyKiBmaWxlTmFtZSwgY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsCi0gICAgY29uc3QgY2hhciogc3RvcmFnZU5hbWUsIGludCBzb3VyY2VUeXBlLCBpbnQgY29tcHJlc3Npb25NZXRob2QsCi0gICAgWmlwRW50cnkqKiBwcEVudHJ5KQotewotICAgIFppcEVudHJ5KiBwRW50cnkgPSBOVUxMOwotICAgIHN0YXR1c190IHJlc3VsdCA9IE5PX0VSUk9SOwotICAgIGxvbmcgbGZoUG9zbiwgc3RhcnRQb3NuLCBlbmRQb3NuLCB1bmNvbXByZXNzZWRMZW47Ci0gICAgRklMRSogaW5wdXRGcCA9IE5VTEw7Ci0gICAgdW5zaWduZWQgbG9uZyBjcmM7Ci0gICAgdGltZV90IG1vZFdoZW47Ci0KLSAgICBpZiAobVJlYWRPbmx5KQotICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0KLSAgICBhc3NlcnQoY29tcHJlc3Npb25NZXRob2QgPT0gWmlwRW50cnk6OmtDb21wcmVzc0RlZmxhdGVkIHx8Ci0gICAgICAgICAgIGNvbXByZXNzaW9uTWV0aG9kID09IFppcEVudHJ5OjprQ29tcHJlc3NTdG9yZWQpOwotCi0gICAgLyogbWFrZSBzdXJlIHdlJ3JlIGluIGEgcmVhc29uYWJsZSBzdGF0ZSAqLwotICAgIGFzc2VydChtWmlwRnAgIT0gTlVMTCk7Ci0gICAgYXNzZXJ0KG1FbnRyaWVzLnNpemUoKSA9PSBtRU9DRC5tVG90YWxOdW1FbnRyaWVzKTsKLQotICAgIC8qIG1ha2Ugc3VyZSBpdCBkb2Vzbid0IGFscmVhZHkgZXhpc3QgKi8KLSAgICBpZiAoZ2V0RW50cnlCeU5hbWUoc3RvcmFnZU5hbWUpICE9IE5VTEwpCi0gICAgICAgIHJldHVybiBBTFJFQURZX0VYSVNUUzsKLQotICAgIGlmICghZGF0YSkgewotICAgICAgICBpbnB1dEZwID0gZm9wZW4oZmlsZU5hbWUsIEZJTEVfT1BFTl9STyk7Ci0gICAgICAgIGlmIChpbnB1dEZwID09IE5VTEwpCi0gICAgICAgICAgICByZXR1cm4gZXJybm9Ub1N0YXR1cyhlcnJubyk7Ci0gICAgfQotCi0gICAgaWYgKGZzZWVrKG1aaXBGcCwgbUVPQ0QubUNlbnRyYWxEaXJPZmZzZXQsIFNFRUtfU0VUKSAhPSAwKSB7Ci0gICAgICAgIHJlc3VsdCA9IFVOS05PV05fRVJST1I7Ci0gICAgICAgIGdvdG8gYmFpbDsKLSAgICB9Ci0KLSAgICBwRW50cnkgPSBuZXcgWmlwRW50cnk7Ci0gICAgcEVudHJ5LT5pbml0TmV3KHN0b3JhZ2VOYW1lLCBOVUxMKTsKLQotICAgIC8qCi0gICAgICogRnJvbSBoZXJlIG9uIG91dCwgZmFpbHVyZXMgYXJlIG1vcmUgaW50ZXJlc3RpbmcuCi0gICAgICovCi0gICAgbU5lZWRDRFJld3JpdGUgPSB0cnVlOwotCi0gICAgLyoKLSAgICAgKiBXcml0ZSB0aGUgTEZILCBldmVuIHRob3VnaCBpdCdzIHN0aWxsIG1vc3RseSBibGFuay4gIFdlIG5lZWQgaXQKLSAgICAgKiBhcyBhIHBsYWNlLWhvbGRlci4gIEluIHRoZW9yeSB0aGUgTEZIIGlzbid0IG5lY2Vzc2FyeSwgYnV0IGluCi0gICAgICogcHJhY3RpY2Ugc29tZSB1dGlsaXRpZXMgZGVtYW5kIGl0LgotICAgICAqLwotICAgIGxmaFBvc24gPSBmdGVsbChtWmlwRnApOwotICAgIHBFbnRyeS0+bUxGSC53cml0ZShtWmlwRnApOwotICAgIHN0YXJ0UG9zbiA9IGZ0ZWxsKG1aaXBGcCk7Ci0KLSAgICAvKgotICAgICAqIENvcHkgdGhlIGRhdGEgaW4sIHBvc3NpYmx5IGNvbXByZXNzaW5nIGl0IGFzIHdlIGdvLgotICAgICAqLwotICAgIGlmIChzb3VyY2VUeXBlID09IFppcEVudHJ5OjprQ29tcHJlc3NTdG9yZWQpIHsKLSAgICAgICAgaWYgKGNvbXByZXNzaW9uTWV0aG9kID09IFppcEVudHJ5OjprQ29tcHJlc3NEZWZsYXRlZCkgewotICAgICAgICAgICAgYm9vbCBmYWlsZWQgPSBmYWxzZTsKLSAgICAgICAgICAgIHJlc3VsdCA9IGNvbXByZXNzRnBUb0ZwKG1aaXBGcCwgaW5wdXRGcCwgZGF0YSwgc2l6ZSwgJmNyYyk7Ci0gICAgICAgICAgICBpZiAocmVzdWx0ICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgICAgICAgICAgTE9HRCgiY29tcHJlc3Npb24gZmFpbGVkLCBzdG9yaW5nXG4iKTsKLSAgICAgICAgICAgICAgICBmYWlsZWQgPSB0cnVlOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAvKgotICAgICAgICAgICAgICAgICAqIE1ha2Ugc3VyZSBpdCBoYXMgY29tcHJlc3NlZCAiZW5vdWdoIi4gIFRoaXMgcHJvYmFibHkgb3VnaHQKLSAgICAgICAgICAgICAgICAgKiB0byBiZSBzZXQgdGhyb3VnaCBhbiBBUEkgY2FsbCwgYnV0IEkgZG9uJ3QgZXhwZWN0IG91cgotICAgICAgICAgICAgICAgICAqIGNyaXRlcmlhIHRvIGNoYW5nZSBvdmVyIHRpbWUuCi0gICAgICAgICAgICAgICAgICovCi0gICAgICAgICAgICAgICAgbG9uZyBzcmMgPSBpbnB1dEZwID8gZnRlbGwoaW5wdXRGcCkgOiBzaXplOwotICAgICAgICAgICAgICAgIGxvbmcgZHN0ID0gZnRlbGwobVppcEZwKSAtIHN0YXJ0UG9zbjsKLSAgICAgICAgICAgICAgICBpZiAoZHN0ICsgKGRzdCAvIDEwKSA+IHNyYykgewotICAgICAgICAgICAgICAgICAgICBMT0dEKCJpbnN1ZmZpY2llbnQgY29tcHJlc3Npb24gKHNyYz0lbGQgZHN0PSVsZCksIHN0b3JpbmdcbiIsCi0gICAgICAgICAgICAgICAgICAgICAgICBzcmMsIGRzdCk7Ci0gICAgICAgICAgICAgICAgICAgIGZhaWxlZCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoZmFpbGVkKSB7Ci0gICAgICAgICAgICAgICAgY29tcHJlc3Npb25NZXRob2QgPSBaaXBFbnRyeTo6a0NvbXByZXNzU3RvcmVkOwotICAgICAgICAgICAgICAgIGlmIChpbnB1dEZwKSByZXdpbmQoaW5wdXRGcCk7Ci0gICAgICAgICAgICAgICAgZnNlZWsobVppcEZwLCBzdGFydFBvc24sIFNFRUtfU0VUKTsKLSAgICAgICAgICAgICAgICAvKiBmYWxsIHRocm91Z2ggdG8ga0NvbXByZXNzU3RvcmVkIGNhc2UgKi8KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICAvKiBoYW5kbGUgIm5vIGNvbXByZXNzaW9uIiByZXF1ZXN0LCBvciBmYWlsZWQgY29tcHJlc3Npb24gZnJvbSBhYm92ZSAqLwotICAgICAgICBpZiAoY29tcHJlc3Npb25NZXRob2QgPT0gWmlwRW50cnk6OmtDb21wcmVzc1N0b3JlZCkgewotICAgICAgICAgICAgaWYgKGlucHV0RnApIHsKLSAgICAgICAgICAgICAgICByZXN1bHQgPSBjb3B5RnBUb0ZwKG1aaXBGcCwgaW5wdXRGcCwgJmNyYyk7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIHJlc3VsdCA9IGNvcHlEYXRhVG9GcChtWmlwRnAsIGRhdGEsIHNpemUsICZjcmMpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikgewotICAgICAgICAgICAgICAgIC8vIGRvbid0IG5lZWQgdG8gdHJ1bmNhdGU7IGhhcHBlbnMgaW4gQ0RFIHJld3JpdGUKLSAgICAgICAgICAgICAgICBMT0dEKCJmYWlsZWQgY29weWluZyBkYXRhIGluXG4iKTsKLSAgICAgICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBjdXJyZW50bHkgc2Vla2VkIHRvIGVuZCBvZiBmaWxlCi0gICAgICAgIHVuY29tcHJlc3NlZExlbiA9IGlucHV0RnAgPyBmdGVsbChpbnB1dEZwKSA6IHNpemU7Ci0gICAgfSBlbHNlIGlmIChzb3VyY2VUeXBlID09IFppcEVudHJ5OjprQ29tcHJlc3NEZWZsYXRlZCkgewotICAgICAgICAvKiB3ZSBzaG91bGQgc3VwcG9ydCB1bmNvbXByZXNzZWQtZnJvbS1jb21wcmVzc2VkLCBidXQgaXQncyBub3QKLSAgICAgICAgICogaW1wb3J0YW50IHJpZ2h0IG5vdyAqLwotICAgICAgICBhc3NlcnQoY29tcHJlc3Npb25NZXRob2QgPT0gWmlwRW50cnk6OmtDb21wcmVzc0RlZmxhdGVkKTsKLQotICAgICAgICBib29sIHNjYW5SZXN1bHQ7Ci0gICAgICAgIGludCBtZXRob2Q7Ci0gICAgICAgIGxvbmcgY29tcHJlc3NlZExlbjsKLQotICAgICAgICBzY2FuUmVzdWx0ID0gWmlwVXRpbHM6OmV4YW1pbmVHemlwKGlucHV0RnAsICZtZXRob2QsICZ1bmNvbXByZXNzZWRMZW4sCi0gICAgICAgICAgICAgICAgICAgICAgICAmY29tcHJlc3NlZExlbiwgJmNyYyk7Ci0gICAgICAgIGlmICghc2NhblJlc3VsdCB8fCBtZXRob2QgIT0gWmlwRW50cnk6OmtDb21wcmVzc0RlZmxhdGVkKSB7Ci0gICAgICAgICAgICBMT0dEKCJ0aGlzIGlzbid0IGEgZGVmbGF0ZWQgZ3ppcCBmaWxlPyIpOwotICAgICAgICAgICAgcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgfQotCi0gICAgICAgIHJlc3VsdCA9IGNvcHlQYXJ0aWFsRnBUb0ZwKG1aaXBGcCwgaW5wdXRGcCwgY29tcHJlc3NlZExlbiwgTlVMTCk7Ci0gICAgICAgIGlmIChyZXN1bHQgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgIExPR0QoImZhaWxlZCBjb3B5aW5nIGd6aXAgZGF0YSBpblxuIik7Ci0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBhc3NlcnQoZmFsc2UpOwotICAgICAgICByZXN1bHQgPSBVTktOT1dOX0VSUk9SOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBXZSBjb3VsZCB3cml0ZSB0aGUgIkRhdGEgRGVzY3JpcHRvciIsIGJ1dCB0aGVyZSBkb2Vzbid0IHNlZW0gdG8KLSAgICAgKiBiZSBhbnkgcG9pbnQgc2luY2Ugd2UncmUgZ29pbmcgdG8gZ28gYmFjayBhbmQgd3JpdGUgdGhlIExGSC4KLSAgICAgKgotICAgICAqIFVwZGF0ZSBmaWxlIG9mZnNldHMuCi0gICAgICovCi0gICAgZW5kUG9zbiA9IGZ0ZWxsKG1aaXBGcCk7ICAgICAgICAgICAgLy8gc2Vla2VkIHRvIGVuZCBvZiBjb21wcmVzc2VkIGRhdGEKLQotICAgIC8qCi0gICAgICogU3VjY2VzcyEgIEZpbGwgb3V0IG5ldyB2YWx1ZXMuCi0gICAgICovCi0gICAgcEVudHJ5LT5zZXREYXRhSW5mbyh1bmNvbXByZXNzZWRMZW4sIGVuZFBvc24gLSBzdGFydFBvc24sIGNyYywKLSAgICAgICAgY29tcHJlc3Npb25NZXRob2QpOwotICAgIG1vZFdoZW4gPSBnZXRNb2RUaW1lKGlucHV0RnAgPyBmaWxlbm8oaW5wdXRGcCkgOiBmaWxlbm8obVppcEZwKSk7Ci0gICAgcEVudHJ5LT5zZXRNb2RXaGVuKG1vZFdoZW4pOwotICAgIHBFbnRyeS0+c2V0TEZIT2Zmc2V0KGxmaFBvc24pOwotICAgIG1FT0NELm1OdW1FbnRyaWVzKys7Ci0gICAgbUVPQ0QubVRvdGFsTnVtRW50cmllcysrOwotICAgIG1FT0NELm1DZW50cmFsRGlyU2l6ZSA9IDA7ICAgICAgLy8gbWFyayBpbnZhbGlkOyBzZXQgYnkgZmx1c2goKQotICAgIG1FT0NELm1DZW50cmFsRGlyT2Zmc2V0ID0gZW5kUG9zbjsKLQotICAgIC8qCi0gICAgICogR28gYmFjayBhbmQgd3JpdGUgdGhlIExGSC4KLSAgICAgKi8KLSAgICBpZiAoZnNlZWsobVppcEZwLCBsZmhQb3NuLCBTRUVLX1NFVCkgIT0gMCkgewotICAgICAgICByZXN1bHQgPSBVTktOT1dOX0VSUk9SOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotICAgIHBFbnRyeS0+bUxGSC53cml0ZShtWmlwRnApOwotCi0gICAgLyoKLSAgICAgKiBBZGQgcEVudHJ5IHRvIHRoZSBsaXN0LgotICAgICAqLwotICAgIG1FbnRyaWVzLmFkZChwRW50cnkpOwotICAgIGlmIChwcEVudHJ5ICE9IE5VTEwpCi0gICAgICAgICpwcEVudHJ5ID0gcEVudHJ5OwotICAgIHBFbnRyeSA9IE5VTEw7Ci0KLWJhaWw6Ci0gICAgaWYgKGlucHV0RnAgIT0gTlVMTCkKLSAgICAgICAgZmNsb3NlKGlucHV0RnApOwotICAgIGRlbGV0ZSBwRW50cnk7Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotLyoKLSAqIEFkZCBhbiBlbnRyeSBieSBjb3B5aW5nIGl0IGZyb20gYW5vdGhlciB6aXAgZmlsZS4gIElmICJwYWRkaW5nIiBpcwotICogbm9uemVybywgdGhlIHNwZWNpZmllZCBudW1iZXIgb2YgYnl0ZXMgd2lsbCBiZSBhZGRlZCB0byB0aGUgImV4dHJhIgotICogZmllbGQgaW4gdGhlIGhlYWRlci4KLSAqCi0gKiBJZiAicHBFbnRyeSIgaXMgbm9uLU5VTEwsIGEgcG9pbnRlciB0byB0aGUgbmV3IGVudHJ5IHdpbGwgYmUgcmV0dXJuZWQuCi0gKi8KLXN0YXR1c190IFppcEZpbGU6OmFkZChjb25zdCBaaXBGaWxlKiBwU291cmNlWmlwLCBjb25zdCBaaXBFbnRyeSogcFNvdXJjZUVudHJ5LAotICAgIGludCBwYWRkaW5nLCBaaXBFbnRyeSoqIHBwRW50cnkpCi17Ci0gICAgWmlwRW50cnkqIHBFbnRyeSA9IE5VTEw7Ci0gICAgc3RhdHVzX3QgcmVzdWx0OwotICAgIGxvbmcgbGZoUG9zbiwgZW5kUG9zbjsKLQotICAgIGlmIChtUmVhZE9ubHkpCi0gICAgICAgIHJldHVybiBJTlZBTElEX09QRVJBVElPTjsKLQotICAgIC8qIG1ha2Ugc3VyZSB3ZSdyZSBpbiBhIHJlYXNvbmFibGUgc3RhdGUgKi8KLSAgICBhc3NlcnQobVppcEZwICE9IE5VTEwpOwotICAgIGFzc2VydChtRW50cmllcy5zaXplKCkgPT0gbUVPQ0QubVRvdGFsTnVtRW50cmllcyk7Ci0KLSAgICBpZiAoZnNlZWsobVppcEZwLCBtRU9DRC5tQ2VudHJhbERpck9mZnNldCwgU0VFS19TRVQpICE9IDApIHsKLSAgICAgICAgcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIHBFbnRyeSA9IG5ldyBaaXBFbnRyeTsKLSAgICBpZiAocEVudHJ5ID09IE5VTEwpIHsKLSAgICAgICAgcmVzdWx0ID0gTk9fTUVNT1JZOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgcmVzdWx0ID0gcEVudHJ5LT5pbml0RnJvbUV4dGVybmFsKHBTb3VyY2VaaXAsIHBTb3VyY2VFbnRyeSk7Ci0gICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikKLSAgICAgICAgZ290byBiYWlsOwotICAgIGlmIChwYWRkaW5nICE9IDApIHsKLSAgICAgICAgcmVzdWx0ID0gcEVudHJ5LT5hZGRQYWRkaW5nKHBhZGRpbmcpOwotICAgICAgICBpZiAocmVzdWx0ICE9IE5PX0VSUk9SKQotICAgICAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIC8qCi0gICAgICogRnJvbSBoZXJlIG9uIG91dCwgZmFpbHVyZXMgYXJlIG1vcmUgaW50ZXJlc3RpbmcuCi0gICAgICovCi0gICAgbU5lZWRDRFJld3JpdGUgPSB0cnVlOwotCi0gICAgLyoKLSAgICAgKiBXcml0ZSB0aGUgTEZILiAgU2luY2Ugd2UncmUgbm90IHJlY29tcHJlc3NpbmcgdGhlIGRhdGEsIHdlIGFscmVhZHkKLSAgICAgKiBoYXZlIGFsbCBvZiB0aGUgZmllbGRzIGZpbGxlZCBvdXQuCi0gICAgICovCi0gICAgbGZoUG9zbiA9IGZ0ZWxsKG1aaXBGcCk7Ci0gICAgcEVudHJ5LT5tTEZILndyaXRlKG1aaXBGcCk7Ci0KLSAgICAvKgotICAgICAqIENvcHkgdGhlIGRhdGEgb3Zlci4KLSAgICAgKgotICAgICAqIElmIHRoZSAiaGFzIGRhdGEgZGVzY3JpcHRvciIgZmxhZyBpcyBzZXQsIHdlIHdhbnQgdG8gY29weSB0aGUgREQKLSAgICAgKiBmaWVsZHMgYXMgd2VsbC4gIFRoaXMgaXMgYSBmaXhlZC1zaXplIGFyZWEgaW1tZWRpYXRlbHkgZm9sbG93aW5nCi0gICAgICogdGhlIGRhdGEuCi0gICAgICovCi0gICAgaWYgKGZzZWVrKHBTb3VyY2VaaXAtPm1aaXBGcCwgcFNvdXJjZUVudHJ5LT5nZXRGaWxlT2Zmc2V0KCksIFNFRUtfU0VUKSAhPSAwKQotICAgIHsKLSAgICAgICAgcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIG9mZl90IGNvcHlMZW47Ci0gICAgY29weUxlbiA9IHBTb3VyY2VFbnRyeS0+Z2V0Q29tcHJlc3NlZExlbigpOwotICAgIGlmICgocFNvdXJjZUVudHJ5LT5tTEZILm1HUEJpdEZsYWcgJiBaaXBFbnRyeTo6a1VzZXNEYXRhRGVzY3IpICE9IDApCi0gICAgICAgIGNvcHlMZW4gKz0gWmlwRW50cnk6OmtEYXRhRGVzY3JpcHRvckxlbjsKLQotICAgIGlmIChjb3B5UGFydGlhbEZwVG9GcChtWmlwRnAsIHBTb3VyY2VaaXAtPm1aaXBGcCwgY29weUxlbiwgTlVMTCkKLSAgICAgICAgIT0gTk9fRVJST1IpCi0gICAgewotICAgICAgICBMT0dXKCJjb3B5IG9mICclcycgZmFpbGVkXG4iLCBwRW50cnktPm1DREUubUZpbGVOYW1lKTsKLSAgICAgICAgcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIC8qCi0gICAgICogVXBkYXRlIGZpbGUgb2Zmc2V0cy4KLSAgICAgKi8KLSAgICBlbmRQb3NuID0gZnRlbGwobVppcEZwKTsKLQotICAgIC8qCi0gICAgICogU3VjY2VzcyEgIEZpbGwgb3V0IG5ldyB2YWx1ZXMuCi0gICAgICovCi0gICAgcEVudHJ5LT5zZXRMRkhPZmZzZXQobGZoUG9zbik7ICAgICAgLy8gc2V0cyBtQ0RFLm1Mb2NhbEhlYWRlclJlbE9mZnNldAotICAgIG1FT0NELm1OdW1FbnRyaWVzKys7Ci0gICAgbUVPQ0QubVRvdGFsTnVtRW50cmllcysrOwotICAgIG1FT0NELm1DZW50cmFsRGlyU2l6ZSA9IDA7ICAgICAgLy8gbWFyayBpbnZhbGlkOyBzZXQgYnkgZmx1c2goKQotICAgIG1FT0NELm1DZW50cmFsRGlyT2Zmc2V0ID0gZW5kUG9zbjsKLQotICAgIC8qCi0gICAgICogQWRkIHBFbnRyeSB0byB0aGUgbGlzdC4KLSAgICAgKi8KLSAgICBtRW50cmllcy5hZGQocEVudHJ5KTsKLSAgICBpZiAocHBFbnRyeSAhPSBOVUxMKQotICAgICAgICAqcHBFbnRyeSA9IHBFbnRyeTsKLSAgICBwRW50cnkgPSBOVUxMOwotCi0gICAgcmVzdWx0ID0gTk9fRVJST1I7Ci0KLWJhaWw6Ci0gICAgZGVsZXRlIHBFbnRyeTsKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi0vKgotICogQ29weSBhbGwgb2YgdGhlIGJ5dGVzIGluICJzcmMiIHRvICJkc3QiLgotICoKLSAqIE9uIGV4aXQsICJzcmNGcCIgd2lsbCBiZSBzZWVrZWQgdG8gdGhlIGVuZCBvZiB0aGUgZmlsZSwgYW5kICJkc3RGcCIKLSAqIHdpbGwgYmUgc2Vla2VkIGltbWVkaWF0ZWx5IHBhc3QgdGhlIGRhdGEuCi0gKi8KLXN0YXR1c190IFppcEZpbGU6OmNvcHlGcFRvRnAoRklMRSogZHN0RnAsIEZJTEUqIHNyY0ZwLCB1bnNpZ25lZCBsb25nKiBwQ1JDMzIpCi17Ci0gICAgdW5zaWduZWQgY2hhciB0bXBCdWZbMzI3NjhdOwotICAgIHNpemVfdCBjb3VudDsKLQotICAgICpwQ1JDMzIgPSBjcmMzMigwTCwgWl9OVUxMLCAwKTsKLQotICAgIHdoaWxlICgxKSB7Ci0gICAgICAgIGNvdW50ID0gZnJlYWQodG1wQnVmLCAxLCBzaXplb2YodG1wQnVmKSwgc3JjRnApOwotICAgICAgICBpZiAoZmVycm9yKHNyY0ZwKSB8fCBmZXJyb3IoZHN0RnApKQotICAgICAgICAgICAgcmV0dXJuIGVycm5vVG9TdGF0dXMoZXJybm8pOwotICAgICAgICBpZiAoY291bnQgPT0gMCkKLSAgICAgICAgICAgIGJyZWFrOwotCi0gICAgICAgICpwQ1JDMzIgPSBjcmMzMigqcENSQzMyLCB0bXBCdWYsIGNvdW50KTsKLQotICAgICAgICBpZiAoZndyaXRlKHRtcEJ1ZiwgMSwgY291bnQsIGRzdEZwKSAhPSBjb3VudCkgewotICAgICAgICAgICAgTE9HRCgiZndyaXRlICVkIGJ5dGVzIGZhaWxlZFxuIiwgKGludCkgY291bnQpOwotICAgICAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLS8qCi0gKiBDb3B5IGFsbCBvZiB0aGUgYnl0ZXMgaW4gInNyYyIgdG8gImRzdCIuCi0gKgotICogT24gZXhpdCwgImRzdEZwIiB3aWxsIGJlIHNlZWtlZCBpbW1lZGlhdGVseSBwYXN0IHRoZSBkYXRhLgotICovCi1zdGF0dXNfdCBaaXBGaWxlOjpjb3B5RGF0YVRvRnAoRklMRSogZHN0RnAsCi0gICAgY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsIHVuc2lnbmVkIGxvbmcqIHBDUkMzMikKLXsKLSAgICBzaXplX3QgY291bnQ7Ci0KLSAgICAqcENSQzMyID0gY3JjMzIoMEwsIFpfTlVMTCwgMCk7Ci0gICAgaWYgKHNpemUgPiAwKSB7Ci0gICAgICAgICpwQ1JDMzIgPSBjcmMzMigqcENSQzMyLCAoY29uc3QgdW5zaWduZWQgY2hhciopZGF0YSwgc2l6ZSk7Ci0gICAgICAgIGlmIChmd3JpdGUoZGF0YSwgMSwgc2l6ZSwgZHN0RnApICE9IHNpemUpIHsKLSAgICAgICAgICAgIExPR0QoImZ3cml0ZSAlZCBieXRlcyBmYWlsZWRcbiIsIChpbnQpIHNpemUpOwotICAgICAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLS8qCi0gKiBDb3B5IHNvbWUgb2YgdGhlIGJ5dGVzIGluICJzcmMiIHRvICJkc3QiLgotICoKLSAqIElmICJwQ1JDMzIiIGlzIE5VTEwsIHRoZSBDUkMgd2lsbCBub3QgYmUgY29tcHV0ZWQuCi0gKgotICogT24gZXhpdCwgInNyY0ZwIiB3aWxsIGJlIHNlZWtlZCB0byB0aGUgZW5kIG9mIHRoZSBmaWxlLCBhbmQgImRzdEZwIgotICogd2lsbCBiZSBzZWVrZWQgaW1tZWRpYXRlbHkgcGFzdCB0aGUgZGF0YSBqdXN0IHdyaXR0ZW4uCi0gKi8KLXN0YXR1c190IFppcEZpbGU6OmNvcHlQYXJ0aWFsRnBUb0ZwKEZJTEUqIGRzdEZwLCBGSUxFKiBzcmNGcCwgbG9uZyBsZW5ndGgsCi0gICAgdW5zaWduZWQgbG9uZyogcENSQzMyKQotewotICAgIHVuc2lnbmVkIGNoYXIgdG1wQnVmWzMyNzY4XTsKLSAgICBzaXplX3QgY291bnQ7Ci0KLSAgICBpZiAocENSQzMyICE9IE5VTEwpCi0gICAgICAgICpwQ1JDMzIgPSBjcmMzMigwTCwgWl9OVUxMLCAwKTsKLQotICAgIHdoaWxlIChsZW5ndGgpIHsKLSAgICAgICAgbG9uZyByZWFkU2l6ZTsKLSAgICAgICAgCi0gICAgICAgIHJlYWRTaXplID0gc2l6ZW9mKHRtcEJ1Zik7Ci0gICAgICAgIGlmIChyZWFkU2l6ZSA+IGxlbmd0aCkKLSAgICAgICAgICAgIHJlYWRTaXplID0gbGVuZ3RoOwotCi0gICAgICAgIGNvdW50ID0gZnJlYWQodG1wQnVmLCAxLCByZWFkU2l6ZSwgc3JjRnApOwotICAgICAgICBpZiAoKGxvbmcpIGNvdW50ICE9IHJlYWRTaXplKSB7ICAgICAvLyBlcnJvciBvciB1bmV4cGVjdGVkIEVPRgotICAgICAgICAgICAgTE9HRCgiZnJlYWQgJWQgYnl0ZXMgZmFpbGVkXG4iLCAoaW50KSByZWFkU2l6ZSk7Ci0gICAgICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgfQotCi0gICAgICAgIGlmIChwQ1JDMzIgIT0gTlVMTCkKLSAgICAgICAgICAgICpwQ1JDMzIgPSBjcmMzMigqcENSQzMyLCB0bXBCdWYsIGNvdW50KTsKLQotICAgICAgICBpZiAoZndyaXRlKHRtcEJ1ZiwgMSwgY291bnQsIGRzdEZwKSAhPSBjb3VudCkgewotICAgICAgICAgICAgTE9HRCgiZndyaXRlICVkIGJ5dGVzIGZhaWxlZFxuIiwgKGludCkgY291bnQpOwotICAgICAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0gICAgICAgIH0KLQotICAgICAgICBsZW5ndGggLT0gcmVhZFNpemU7Ci0gICAgfQotCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vKgotICogQ29tcHJlc3MgYWxsIG9mIHRoZSBkYXRhIGluICJzcmNGcCIgYW5kIHdyaXRlIGl0IHRvICJkc3RGcCIuCi0gKgotICogT24gZXhpdCwgInNyY0ZwIiB3aWxsIGJlIHNlZWtlZCB0byB0aGUgZW5kIG9mIHRoZSBmaWxlLCBhbmQgImRzdEZwIgotICogd2lsbCBiZSBzZWVrZWQgaW1tZWRpYXRlbHkgcGFzdCB0aGUgY29tcHJlc3NlZCBkYXRhLgotICovCi1zdGF0dXNfdCBaaXBGaWxlOjpjb21wcmVzc0ZwVG9GcChGSUxFKiBkc3RGcCwgRklMRSogc3JjRnAsCi0gICAgY29uc3Qgdm9pZCogZGF0YSwgc2l6ZV90IHNpemUsIHVuc2lnbmVkIGxvbmcqIHBDUkMzMikKLXsKLSAgICBzdGF0dXNfdCByZXN1bHQgPSBOT19FUlJPUjsKLQljb25zdCBzaXplX3Qga0J1ZlNpemUgPSAzMjc2ODsKLQl1bnNpZ25lZCBjaGFyKiBpbkJ1ZiA9IE5VTEw7Ci0JdW5zaWduZWQgY2hhciogb3V0QnVmID0gTlVMTDsKLQl6X3N0cmVhbSB6c3RyZWFtOwotICAgIGJvb2wgYXRFb2YgPSBmYWxzZTsgICAgIC8vIG5vIGZlb2YoKSBhdmlhaWxhYmxlIHlldAotCXVuc2lnbmVkIGxvbmcgY3JjOwotCWludCB6ZXJyOwotCi0JLyoKLQkgKiBDcmVhdGUgYW4gaW5wdXQgYnVmZmVyIGFuZCBhbiBvdXRwdXQgYnVmZmVyLgotCSAqLwotCWluQnVmID0gbmV3IHVuc2lnbmVkIGNoYXJba0J1ZlNpemVdOwotCW91dEJ1ZiA9IG5ldyB1bnNpZ25lZCBjaGFyW2tCdWZTaXplXTsKLQlpZiAoaW5CdWYgPT0gTlVMTCB8fCBvdXRCdWYgPT0gTlVMTCkgewotCQlyZXN1bHQgPSBOT19NRU1PUlk7Ci0JCWdvdG8gYmFpbDsKLQl9Ci0KLQkvKgotCSAqIEluaXRpYWxpemUgdGhlIHpsaWIgc3RyZWFtLgotCSAqLwotCW1lbXNldCgmenN0cmVhbSwgMCwgc2l6ZW9mKHpzdHJlYW0pKTsKLQl6c3RyZWFtLnphbGxvYyA9IFpfTlVMTDsKLQl6c3RyZWFtLnpmcmVlID0gWl9OVUxMOwotCXpzdHJlYW0ub3BhcXVlID0gWl9OVUxMOwotCXpzdHJlYW0ubmV4dF9pbiA9IE5VTEw7Ci0JenN0cmVhbS5hdmFpbF9pbiA9IDA7Ci0JenN0cmVhbS5uZXh0X291dCA9IG91dEJ1ZjsKLQl6c3RyZWFtLmF2YWlsX291dCA9IGtCdWZTaXplOwotCXpzdHJlYW0uZGF0YV90eXBlID0gWl9VTktOT1dOOwotCi0JemVyciA9IGRlZmxhdGVJbml0MigmenN0cmVhbSwgWl9CRVNUX0NPTVBSRVNTSU9OLAotCQlaX0RFRkxBVEVELCAtTUFYX1dCSVRTLCBERUZfTUVNX0xFVkVMLCBaX0RFRkFVTFRfU1RSQVRFR1kpOwotCWlmICh6ZXJyICE9IFpfT0spIHsKLQkJcmVzdWx0ID0gVU5LTk9XTl9FUlJPUjsKLQkJaWYgKHplcnIgPT0gWl9WRVJTSU9OX0VSUk9SKSB7Ci0JCQlMT0dFKCJJbnN0YWxsZWQgemxpYiBpcyBub3QgY29tcGF0aWJsZSB3aXRoIGxpbmtlZCB2ZXJzaW9uICglcylcbiIsCi0JCQkJWkxJQl9WRVJTSU9OKTsKLQkJfSBlbHNlIHsKLQkJCUxPR0QoIkNhbGwgdG8gZGVmbGF0ZUluaXQyIGZhaWxlZCAoemVycj0lZClcbiIsIHplcnIpOwotCQl9Ci0JCWdvdG8gYmFpbDsKLQl9Ci0KLSAJY3JjID0gY3JjMzIoMEwsIFpfTlVMTCwgMCk7Ci0KLQkvKgotCSAqIExvb3Agd2hpbGUgd2UgaGF2ZSBkYXRhLgotCSAqLwotCWRvIHsKLQkJc2l6ZV90IGdldFNpemU7Ci0JCWludCBmbHVzaDsKLQotCQkvKiBvbmx5IHJlYWQgaWYgdGhlIGlucHV0IGJ1ZmZlciBpcyBlbXB0eSAqLwotCQlpZiAoenN0cmVhbS5hdmFpbF9pbiA9PSAwICYmICFhdEVvZikgewotICAgICAgICAgICAgTE9HVigiKysrIHJlYWRpbmcgJWQgYnl0ZXNcbiIsIChpbnQpa0J1ZlNpemUpOwotICAgICAgICAgICAgaWYgKGRhdGEpIHsKLSAgICAgICAgICAgICAgICBnZXRTaXplID0gc2l6ZSA+IGtCdWZTaXplID8ga0J1ZlNpemUgOiBzaXplOwotICAgICAgICAgICAgICAgIG1lbWNweShpbkJ1ZiwgZGF0YSwgZ2V0U2l6ZSk7Ci0gICAgICAgICAgICAgICAgZGF0YSA9ICgoY29uc3QgY2hhciopZGF0YSkgKyBnZXRTaXplOwotICAgICAgICAgICAgICAgIHNpemUgLT0gZ2V0U2l6ZTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgZ2V0U2l6ZSA9IGZyZWFkKGluQnVmLCAxLCBrQnVmU2l6ZSwgc3JjRnApOwotICAgICAgICAgICAgICAgIGlmIChmZXJyb3Ioc3JjRnApKSB7Ci0gICAgICAgICAgICAgICAgICAgIExPR0QoImRlZmxhdGUgcmVhZCBmYWlsZWQgKGVycm5vPSVkKVxuIiwgZXJybm8pOwotICAgICAgICAgICAgICAgICAgICBnb3RvIHpfYmFpbDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoZ2V0U2l6ZSA8IGtCdWZTaXplKSB7Ci0gICAgICAgICAgICAgICAgTE9HVigiKysrICBnb3QgJWQgYnl0ZXMsIEVPRiByZWFjaGVkXG4iLAotICAgICAgICAgICAgICAgICAgICAoaW50KWdldFNpemUpOwotICAgICAgICAgICAgICAgIGF0RW9mID0gdHJ1ZTsKLSAgICAgICAgICAgIH0KLQotCQkJY3JjID0gY3JjMzIoY3JjLCBpbkJ1ZiwgZ2V0U2l6ZSk7Ci0KLQkJCXpzdHJlYW0ubmV4dF9pbiA9IGluQnVmOwotCQkJenN0cmVhbS5hdmFpbF9pbiA9IGdldFNpemU7Ci0JCX0KLQotCQlpZiAoYXRFb2YpCi0JCQlmbHVzaCA9IFpfRklOSVNIOyAgICAgICAvKiB0ZWxsIHpsaWIgdGhhdCB3ZSdyZSBkb25lICovCi0JCWVsc2UKLQkJCWZsdXNoID0gWl9OT19GTFVTSDsgICAgIC8qIG1vcmUgdG8gY29tZSEgKi8KLQotCQl6ZXJyID0gZGVmbGF0ZSgmenN0cmVhbSwgZmx1c2gpOwotCQlpZiAoemVyciAhPSBaX09LICYmIHplcnIgIT0gWl9TVFJFQU1fRU5EKSB7Ci0JCQlMT0dEKCJ6bGliIGRlZmxhdGUgY2FsbCBmYWlsZWQgKHplcnI9JWQpXG4iLCB6ZXJyKTsKLQkJCXJlc3VsdCA9IFVOS05PV05fRVJST1I7Ci0JCQlnb3RvIHpfYmFpbDsKLQkJfQotCi0JCS8qIHdyaXRlIHdoZW4gd2UncmUgZnVsbCBvciB3aGVuIHdlJ3JlIGRvbmUgKi8KLQkJaWYgKHpzdHJlYW0uYXZhaWxfb3V0ID09IDAgfHwKLQkJCSh6ZXJyID09IFpfU1RSRUFNX0VORCAmJiB6c3RyZWFtLmF2YWlsX291dCAhPSAodUludCkga0J1ZlNpemUpKQotCQl7Ci0JCQlMT0dWKCIrKysgd3JpdGluZyAlZCBieXRlc1xuIiwgKGludCkgKHpzdHJlYW0ubmV4dF9vdXQgLSBvdXRCdWYpKTsKLSAgICAgICAgICAgIGlmIChmd3JpdGUob3V0QnVmLCAxLCB6c3RyZWFtLm5leHRfb3V0IC0gb3V0QnVmLCBkc3RGcCkgIT0KLSAgICAgICAgICAgICAgICAoc2l6ZV90KSh6c3RyZWFtLm5leHRfb3V0IC0gb3V0QnVmKSkKLSAgICAgICAgICAgIHsKLQkJCQlMT0dEKCJ3cml0ZSAlZCBmYWlsZWQgaW4gZGVmbGF0ZVxuIiwKLSAgICAgICAgICAgICAgICAgICAgKGludCkgKHpzdHJlYW0ubmV4dF9vdXQgLSBvdXRCdWYpKTsKLQkJCQlnb3RvIHpfYmFpbDsKLQkJCX0KLQotCQkJenN0cmVhbS5uZXh0X291dCA9IG91dEJ1ZjsKLQkJCXpzdHJlYW0uYXZhaWxfb3V0ID0ga0J1ZlNpemU7Ci0JCX0KLQl9IHdoaWxlICh6ZXJyID09IFpfT0spOwotCi0JYXNzZXJ0KHplcnIgPT0gWl9TVFJFQU1fRU5EKTsgICAgICAgLyogb3RoZXIgZXJyb3JzIHNob3VsZCd2ZSBiZWVuIGNhdWdodCAqLwotCi0JKnBDUkMzMiA9IGNyYzsKLQotel9iYWlsOgotCWRlZmxhdGVFbmQoJnpzdHJlYW0pOyAgICAgICAgLyogZnJlZSB1cCBhbnkgYWxsb2NhdGVkIHN0cnVjdHVyZXMgKi8KLQotYmFpbDoKLQlkZWxldGVbXSBpbkJ1ZjsKLQlkZWxldGVbXSBvdXRCdWY7Ci0KLQlyZXR1cm4gcmVzdWx0OwotfQotCi0vKgotICogTWFyayBhbiBlbnRyeSBhcyBkZWxldGVkLgotICoKLSAqIFdlIHdpbGwgZXZlbnR1YWxseSBuZWVkIHRvIGNydW5jaCB0aGUgZmlsZSBkb3duLCBidXQgaWYgc2V2ZXJhbCBmaWxlcwotICogYXJlIGJlaW5nIHJlbW92ZWQgKHBlcmhhcHMgYXMgcGFydCBvZiBhbiAidXBkYXRlIiBwcm9jZXNzKSB3ZSBjYW4gbWFrZQotICogdGhpbmdzIGNvbnNpZGVyYWJseSBmYXN0ZXIgYnkgZGVmZXJyaW5nIHRoZSByZW1vdmFsIHRvICJmbHVzaCIgdGltZS4KLSAqLwotc3RhdHVzX3QgWmlwRmlsZTo6cmVtb3ZlKFppcEVudHJ5KiBwRW50cnkpCi17Ci0gICAgLyoKLSAgICAgKiBTaG91bGQgdmVyaWZ5IHRoYXQgcEVudHJ5IGlzIGFjdHVhbGx5IHBhcnQgb2YgdGhpcyBhcmNoaXZlLCBhbmQKLSAgICAgKiBub3Qgc29tZSBzdHJheSBaaXBFbnRyeSBmcm9tIGEgZGlmZmVyZW50IGZpbGUuCi0gICAgICovCi0KLSAgICAvKiBtYXJrIGVudHJ5IGFzIGRlbGV0ZWQsIGFuZCBtYXJrIGFyY2hpdmUgYXMgZGlydHkgKi8KLSAgICBwRW50cnktPnNldERlbGV0ZWQoKTsKLSAgICBtTmVlZENEUmV3cml0ZSA9IHRydWU7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vKgotICogRmx1c2ggYW55IHBlbmRpbmcgd3JpdGVzLgotICoKLSAqIEluIHBhcnRpY3VsYXIsIHRoaXMgd2lsbCBjcnVuY2ggb3V0IGRlbGV0ZWQgZW50cmllcywgYW5kIHdyaXRlIHRoZQotICogQ2VudHJhbCBEaXJlY3RvcnkgYW5kIEVPQ0QgaWYgd2UgaGF2ZSBzdG9tcGVkIG9uIHRoZW0uCi0gKi8KLXN0YXR1c190IFppcEZpbGU6OmZsdXNoKHZvaWQpCi17Ci0gICAgc3RhdHVzX3QgcmVzdWx0ID0gTk9fRVJST1I7Ci0gICAgbG9uZyBlb2NkUG9zbjsKLSAgICBpbnQgaSwgY291bnQ7Ci0KLSAgICBpZiAobVJlYWRPbmx5KQotICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgaWYgKCFtTmVlZENEUmV3cml0ZSkKLSAgICAgICAgcmV0dXJuIE5PX0VSUk9SOwotCi0gICAgYXNzZXJ0KG1aaXBGcCAhPSBOVUxMKTsKLQotICAgIHJlc3VsdCA9IGNydW5jaEFyY2hpdmUoKTsKLSAgICBpZiAocmVzdWx0ICE9IE5PX0VSUk9SKQotICAgICAgICByZXR1cm4gcmVzdWx0OwotCi0gICAgaWYgKGZzZWVrKG1aaXBGcCwgbUVPQ0QubUNlbnRyYWxEaXJPZmZzZXQsIFNFRUtfU0VUKSAhPSAwKQotICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLQotICAgIGNvdW50ID0gbUVudHJpZXMuc2l6ZSgpOwotICAgIGZvciAoaSA9IDA7IGkgPCBjb3VudDsgaSsrKSB7Ci0gICAgICAgIFppcEVudHJ5KiBwRW50cnkgPSBtRW50cmllc1tpXTsKLSAgICAgICAgcEVudHJ5LT5tQ0RFLndyaXRlKG1aaXBGcCk7Ci0gICAgfQotCi0gICAgZW9jZFBvc24gPSBmdGVsbChtWmlwRnApOwotICAgIG1FT0NELm1DZW50cmFsRGlyU2l6ZSA9IGVvY2RQb3NuIC0gbUVPQ0QubUNlbnRyYWxEaXJPZmZzZXQ7Ci0KLSAgICBtRU9DRC53cml0ZShtWmlwRnApOwotCi0gICAgLyoKLSAgICAgKiBJZiB3ZSBoYWQgc29tZSBzdHVmZiBibG9hdCB1cCBkdXJpbmcgY29tcHJlc3Npb24gYW5kIGdldCByZXBsYWNlZAotICAgICAqIHdpdGggcGxhaW4gZmlsZXMsIG9yIGlmIHdlIGRlbGV0ZWQgc29tZSBlbnRyaWVzLCB0aGVyZSdzIGEgbG90Ci0gICAgICogb2Ygd2FzdGVkIHNwYWNlIGF0IHRoZSBlbmQgb2YgdGhlIGZpbGUuICBSZW1vdmUgaXQgbm93LgotICAgICAqLwotICAgIGlmIChmdHJ1bmNhdGUoZmlsZW5vKG1aaXBGcCksIGZ0ZWxsKG1aaXBGcCkpICE9IDApIHsKLSAgICAgICAgTE9HVygiZnRydW5jYXRlIGZhaWxlZCAlbGQ6ICVzXG4iLCBmdGVsbChtWmlwRnApLCBzdHJlcnJvcihlcnJubykpOwotICAgICAgICAvLyBub3QgZmF0YWwKLSAgICB9Ci0KLSAgICAvKiBzaG91bGQgd2UgY2xlYXIgdGhlICJuZXdseSBhZGRlZCIgZmxhZyBpbiBhbGwgZW50cmllcyBub3c/ICovCi0KLSAgICBtTmVlZENEUmV3cml0ZSA9IGZhbHNlOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotLyoKLSAqIENydW5jaCBkZWxldGVkIGZpbGVzIG91dCBvZiBhbiBhcmNoaXZlIGJ5IHNoaWZ0aW5nIHRoZSBsYXRlciBmaWxlcyBkb3duLgotICoKLSAqIEJlY2F1c2Ugd2UncmUgbm90IHVzaW5nIGEgdGVtcCBmaWxlLCB3ZSBkbyB0aGUgb3BlcmF0aW9uIGluc2lkZSB0aGUKLSAqIGN1cnJlbnQgZmlsZS4KLSAqLwotc3RhdHVzX3QgWmlwRmlsZTo6Y3J1bmNoQXJjaGl2ZSh2b2lkKQotewotICAgIHN0YXR1c190IHJlc3VsdCA9IE5PX0VSUk9SOwotICAgIGludCBpLCBjb3VudDsKLSAgICBsb25nIGRlbENvdW50LCBhZGp1c3Q7Ci0KLSNpZiAwCi0gICAgcHJpbnRmKCJDT05URU5UUzpcbiIpOwotICAgIGZvciAoaSA9IDA7IGkgPCAoaW50KSBtRW50cmllcy5zaXplKCk7IGkrKykgewotICAgICAgICBwcmludGYoIiAlZDogbGZoT2ZmPSVsZCBkZWw9JWRcbiIsCi0gICAgICAgICAgICBpLCBtRW50cmllc1tpXS0+Z2V0TEZIT2Zmc2V0KCksIG1FbnRyaWVzW2ldLT5nZXREZWxldGVkKCkpOwotICAgIH0KLSAgICBwcmludGYoIiAgRU5EIGlzICVsZFxuIiwgKGxvbmcpIG1FT0NELm1DZW50cmFsRGlyT2Zmc2V0KTsKLSNlbmRpZgotCi0gICAgLyoKLSAgICAgKiBSb2xsIHRocm91Z2ggdGhlIHNldCBvZiBmaWxlcywgc2hpZnRpbmcgdGhlbSBhcyBhcHByb3ByaWF0ZS4gIFdlCi0gICAgICogY291bGQgcHJvYmFibHkgZ2V0IGEgc2xpZ2h0IHBlcmZvcm1hbmNlIGltcHJvdmVtZW50IGJ5IHNsaWRpbmcKLSAgICAgKiBtdWx0aXBsZSBmaWxlcyBkb3duIGF0IG9uY2UgKGJlY2F1c2Ugd2UgY291bGQgdXNlIGxhcmdlciByZWFkcwotICAgICAqIHdoZW4gb3BlcmF0aW5nIG9uIGJhdGNoZXMgb2Ygc21hbGwgZmlsZXMpLCBidXQgaXQncyBub3QgdGhhdCB1c2VmdWwuCi0gICAgICovCi0gICAgY291bnQgPSBtRW50cmllcy5zaXplKCk7Ci0gICAgZGVsQ291bnQgPSBhZGp1c3QgPSAwOwotICAgIGZvciAoaSA9IDA7IGkgPCBjb3VudDsgaSsrKSB7Ci0gICAgICAgIFppcEVudHJ5KiBwRW50cnkgPSBtRW50cmllc1tpXTsKLSAgICAgICAgbG9uZyBzcGFuOwotCi0gICAgICAgIGlmIChwRW50cnktPmdldExGSE9mZnNldCgpICE9IDApIHsKLSAgICAgICAgICAgIGxvbmcgbmV4dE9mZnNldDsKLQotICAgICAgICAgICAgLyogR2V0IHRoZSBsZW5ndGggb2YgdGhpcyBlbnRyeSBieSBmaW5kaW5nIHRoZSBvZmZzZXQKLSAgICAgICAgICAgICAqIG9mIHRoZSBuZXh0IGVudHJ5LiAgRGlyZWN0b3J5IGVudHJpZXMgZG9uJ3QgaGF2ZQotICAgICAgICAgICAgICogZmlsZSBvZmZzZXRzLCBzbyB3ZSBuZWVkIHRvIGZpbmQgdGhlIG5leHQgbm9uLWRpcmVjdG9yeQotICAgICAgICAgICAgICogZW50cnkuCi0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIG5leHRPZmZzZXQgPSAwOwotICAgICAgICAgICAgZm9yIChpbnQgaWkgPSBpKzE7IG5leHRPZmZzZXQgPT0gMCAmJiBpaSA8IGNvdW50OyBpaSsrKQotICAgICAgICAgICAgICAgIG5leHRPZmZzZXQgPSBtRW50cmllc1tpaV0tPmdldExGSE9mZnNldCgpOwotICAgICAgICAgICAgaWYgKG5leHRPZmZzZXQgPT0gMCkKLSAgICAgICAgICAgICAgICBuZXh0T2Zmc2V0ID0gbUVPQ0QubUNlbnRyYWxEaXJPZmZzZXQ7Ci0gICAgICAgICAgICBzcGFuID0gbmV4dE9mZnNldCAtIHBFbnRyeS0+Z2V0TEZIT2Zmc2V0KCk7Ci0KLSAgICAgICAgICAgIGFzc2VydChzcGFuID49IFppcEVudHJ5OjpMb2NhbEZpbGVIZWFkZXI6OmtMRkhMZW4pOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLyogVGhpcyBpcyBhIGRpcmVjdG9yeSBlbnRyeS4gIEl0IGRvZXNuJ3QgaGF2ZQotICAgICAgICAgICAgICogYW55IGFjdHVhbCBmaWxlIGNvbnRlbnRzLCBzbyB0aGVyZSdzIG5vIG5lZWQgdG8KLSAgICAgICAgICAgICAqIG1vdmUgYW55dGhpbmcuCi0gICAgICAgICAgICAgKi8KLSAgICAgICAgICAgIHNwYW4gPSAwOwotICAgICAgICB9Ci0KLSAgICAgICAgLy9wcmludGYoIisrKyAlZDogb2ZmPSVsZCBzcGFuPSVsZCBkZWw9JWQgW2NvdW50PSVkXVxuIiwKLSAgICAgICAgLy8gICAgaSwgcEVudHJ5LT5nZXRMRkhPZmZzZXQoKSwgc3BhbiwgcEVudHJ5LT5nZXREZWxldGVkKCksIGNvdW50KTsKLQotICAgICAgICBpZiAocEVudHJ5LT5nZXREZWxldGVkKCkpIHsKLSAgICAgICAgICAgIGFkanVzdCArPSBzcGFuOwotICAgICAgICAgICAgZGVsQ291bnQrKzsKLQotICAgICAgICAgICAgZGVsZXRlIHBFbnRyeTsKLSAgICAgICAgICAgIG1FbnRyaWVzLnJlbW92ZUF0KGkpOwotCi0gICAgICAgICAgICAvKiBhZGp1c3QgbG9vcCBjb250cm9sICovCi0gICAgICAgICAgICBjb3VudC0tOwotICAgICAgICAgICAgaS0tOwotICAgICAgICB9IGVsc2UgaWYgKHNwYW4gIT0gMCAmJiBhZGp1c3QgPiAwKSB7Ci0gICAgICAgICAgICAvKiBzaHVmZmxlIHRoaXMgZW50cnkgYmFjayAqLwotICAgICAgICAgICAgLy9wcmludGYoIisrKyBTaHVmZmxpbmcgJyVzJyBiYWNrICVsZFxuIiwKLSAgICAgICAgICAgIC8vICAgIHBFbnRyeS0+Z2V0RmlsZU5hbWUoKSwgYWRqdXN0KTsKLSAgICAgICAgICAgIHJlc3VsdCA9IGZpbGVtb3ZlKG1aaXBGcCwgcEVudHJ5LT5nZXRMRkhPZmZzZXQoKSAtIGFkanVzdCwKLSAgICAgICAgICAgICAgICAgICAgICAgIHBFbnRyeS0+Z2V0TEZIT2Zmc2V0KCksIHNwYW4pOwotICAgICAgICAgICAgaWYgKHJlc3VsdCAhPSBOT19FUlJPUikgewotICAgICAgICAgICAgICAgIC8qIHRoaXMgaXMgd2h5IHlvdSB1c2UgYSB0ZW1wIGZpbGUgKi8KLSAgICAgICAgICAgICAgICBMT0dFKCJlcnJvciBkdXJpbmcgY3J1bmNoIC0gYXJjaGl2ZSBpcyB0b2FzdFxuIik7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIHJlc3VsdDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgcEVudHJ5LT5zZXRMRkhPZmZzZXQocEVudHJ5LT5nZXRMRkhPZmZzZXQoKSAtIGFkanVzdCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvKgotICAgICAqIEZpeCBFT0NEIGluZm8uICBXZSBoYXZlIHRvIHdhaXQgdW50aWwgdGhlIGVuZCB0byBkbyBzb21lIG9mIHRoaXMKLSAgICAgKiBiZWNhdXNlIHdlIHVzZSBtQ2VudHJhbERpck9mZnNldCB0byBkZXRlcm1pbmUgInNwYW4iIGZvciB0aGUKLSAgICAgKiBsYXN0IGVudHJ5LgotICAgICAqLwotICAgIG1FT0NELm1DZW50cmFsRGlyT2Zmc2V0IC09IGFkanVzdDsKLSAgICBtRU9DRC5tTnVtRW50cmllcyAtPSBkZWxDb3VudDsKLSAgICBtRU9DRC5tVG90YWxOdW1FbnRyaWVzIC09IGRlbENvdW50OwotICAgIG1FT0NELm1DZW50cmFsRGlyU2l6ZSA9IDA7ICAvLyBtYXJrIGludmFsaWQ7IHNldCBieSBmbHVzaCgpCi0KLSAgICBhc3NlcnQobUVPQ0QubU51bUVudHJpZXMgPT0gbUVPQ0QubVRvdGFsTnVtRW50cmllcyk7Ci0gICAgYXNzZXJ0KG1FT0NELm1OdW1FbnRyaWVzID09IGNvdW50KTsKLQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLS8qCi0gKiBXb3JrcyBsaWtlIG1lbW1vdmUoKSwgYnV0IG9uIHBpZWNlcyBvZiBhIGZpbGUuCi0gKi8KLXN0YXR1c190IFppcEZpbGU6OmZpbGVtb3ZlKEZJTEUqIGZwLCBvZmZfdCBkc3QsIG9mZl90IHNyYywgc2l6ZV90IG4pCi17Ci0gICAgaWYgKGRzdCA9PSBzcmMgfHwgbiA8PSAwKQotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0KLSAgICB1bnNpZ25lZCBjaGFyIHJlYWRCdWZbMzI3NjhdOwotCi0gICAgaWYgKGRzdCA8IHNyYykgewotICAgICAgICAvKiBzaGlmdCBzdHVmZiB0b3dhcmQgc3RhcnQgb2YgZmlsZTsgbXVzdCByZWFkIGZyb20gc3RhcnQgKi8KLSAgICAgICAgd2hpbGUgKG4gIT0gMCkgewotICAgICAgICAgICAgc2l6ZV90IGdldFNpemUgPSBzaXplb2YocmVhZEJ1Zik7Ci0gICAgICAgICAgICBpZiAoZ2V0U2l6ZSA+IG4pCi0gICAgICAgICAgICAgICAgZ2V0U2l6ZSA9IG47Ci0KLSAgICAgICAgICAgIGlmIChmc2VlayhmcCwgKGxvbmcpIHNyYywgU0VFS19TRVQpICE9IDApIHsKLSAgICAgICAgICAgICAgICBMT0dEKCJmaWxlbW92ZSBzcmMgc2VlayAlbGQgZmFpbGVkXG4iLCAobG9uZykgc3JjKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKGZyZWFkKHJlYWRCdWYsIDEsIGdldFNpemUsIGZwKSAhPSBnZXRTaXplKSB7Ci0gICAgICAgICAgICAgICAgTE9HRCgiZmlsZW1vdmUgcmVhZCAlbGQgb2ZmPSVsZCBmYWlsZWRcbiIsCi0gICAgICAgICAgICAgICAgICAgIChsb25nKSBnZXRTaXplLCAobG9uZykgc3JjKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKGZzZWVrKGZwLCAobG9uZykgZHN0LCBTRUVLX1NFVCkgIT0gMCkgewotICAgICAgICAgICAgICAgIExPR0QoImZpbGVtb3ZlIGRzdCBzZWVrICVsZCBmYWlsZWRcbiIsIChsb25nKSBkc3QpOwotICAgICAgICAgICAgICAgIHJldHVybiBVTktOT1dOX0VSUk9SOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICBpZiAoZndyaXRlKHJlYWRCdWYsIDEsIGdldFNpemUsIGZwKSAhPSBnZXRTaXplKSB7Ci0gICAgICAgICAgICAgICAgTE9HRCgiZmlsZW1vdmUgd3JpdGUgJWxkIG9mZj0lbGQgZmFpbGVkXG4iLAotICAgICAgICAgICAgICAgICAgICAobG9uZykgZ2V0U2l6ZSwgKGxvbmcpIGRzdCk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHNyYyArPSBnZXRTaXplOwotICAgICAgICAgICAgZHN0ICs9IGdldFNpemU7Ci0gICAgICAgICAgICBuIC09IGdldFNpemU7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICAvKiBzaGlmdCBzdHVmZiB0b3dhcmQgZW5kIG9mIGZpbGU7IG11c3QgcmVhZCBmcm9tIGVuZCAqLwotICAgICAgICBhc3NlcnQoZmFsc2UpOyAgICAgIC8vIHdyaXRlIHRoaXMgc29tZWRheSwgbWF5YmUKLSAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0gICAgfQotCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0KLS8qCi0gKiBHZXQgdGhlIG1vZGlmaWNhdGlvbiB0aW1lIGZyb20gYSBmaWxlIGRlc2NyaXB0b3IuCi0gKi8KLXRpbWVfdCBaaXBGaWxlOjpnZXRNb2RUaW1lKGludCBmZCkKLXsKLSAgICBzdHJ1Y3Qgc3RhdCBzYjsKLQotICAgIGlmIChmc3RhdChmZCwgJnNiKSA8IDApIHsKLSAgICAgICAgTE9HRCgiSEVZOiBmc3RhdCBvbiBmZCAlZCBmYWlsZWRcbiIsIGZkKTsKLSAgICAgICAgcmV0dXJuICh0aW1lX3QpIC0xOwotICAgIH0KLQotICAgIHJldHVybiBzYi5zdF9tdGltZTsKLX0KLQotCi0jaWYgMCAgICAgICAvKiB0aGlzIGlzIGEgYmFkIGlkZWEgKi8KLS8qCi0gKiBHZXQgYSBjb3B5IG9mIHRoZSBaaXAgZmlsZSBkZXNjcmlwdG9yLgotICoKLSAqIFdlIGRvbid0IGFsbG93IHRoaXMgaWYgdGhlIGZpbGUgd2FzIG9wZW5lZCByZWFkLXdyaXRlIGJlY2F1c2Ugd2UgdGVuZAotICogdG8gbGVhdmUgdGhlIGZpbGUgY29udGVudHMgaW4gYW4gdW5jZXJ0YWluIHN0YXRlIGJldHdlZW4gY2FsbHMgdG8KLSAqIGZsdXNoKCkuICBUaGUgZHVwbGljYXRlZCBmaWxlIGRlc2NyaXB0b3Igc2hvdWxkIG9ubHkgYmUgdmFsaWQgZm9yIHJlYWRzLgotICovCi1pbnQgWmlwRmlsZTo6Z2V0WmlwRmQodm9pZCkgY29uc3QKLXsKLSAgICBpZiAoIW1SZWFkT25seSkKLSAgICAgICAgcmV0dXJuIElOVkFMSURfT1BFUkFUSU9OOwotICAgIGFzc2VydChtWmlwRnAgIT0gTlVMTCk7Ci0KLSAgICBpbnQgZmQ7Ci0gICAgZmQgPSBkdXAoZmlsZW5vKG1aaXBGcCkpOwotICAgIGlmIChmZCA8IDApIHsKLSAgICAgICAgTE9HRCgiZGlkbid0IHdvcmssIGVycm5vPSVkXG4iLCBlcnJubyk7Ci0gICAgfQotCi0gICAgcmV0dXJuIGZkOwotfQotI2VuZGlmCi0KLQotI2lmIDAKLS8qCi0gKiBFeHBhbmQgZGF0YS4KLSAqLwotYm9vbCBaaXBGaWxlOjp1bmNvbXByZXNzKGNvbnN0IFppcEVudHJ5KiBwRW50cnksIHZvaWQqIGJ1ZikgY29uc3QKLXsKLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0jZW5kaWYKLQotLy8gZnJlZSB0aGUgbWVtb3J5IHdoZW4geW91J3JlIGRvbmUKLXZvaWQqIFppcEZpbGU6OnVuY29tcHJlc3MoY29uc3QgWmlwRW50cnkqIGVudHJ5KQotewotICAgIHNpemVfdCB1bmxlbiA9IGVudHJ5LT5nZXRVbmNvbXByZXNzZWRMZW4oKTsKLSAgICBzaXplX3QgY2xlbiA9IGVudHJ5LT5nZXRDb21wcmVzc2VkTGVuKCk7Ci0KLSAgICB2b2lkKiBidWYgPSBtYWxsb2ModW5sZW4pOwotICAgIGlmIChidWYgPT0gTlVMTCkgewotICAgICAgICByZXR1cm4gTlVMTDsKLSAgICB9Ci0KLSAgICBmc2VlayhtWmlwRnAsIDAsIFNFRUtfU0VUKTsKLQotICAgIG9mZl90IG9mZnNldCA9IGVudHJ5LT5nZXRGaWxlT2Zmc2V0KCk7Ci0gICAgaWYgKGZzZWVrKG1aaXBGcCwgb2Zmc2V0LCBTRUVLX1NFVCkgIT0gMCkgewotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgc3dpdGNoIChlbnRyeS0+Z2V0Q29tcHJlc3Npb25NZXRob2QoKSkKLSAgICB7Ci0gICAgICAgIGNhc2UgWmlwRW50cnk6OmtDb21wcmVzc1N0b3JlZDogewotICAgICAgICAgICAgc3NpemVfdCBhbXQgPSBmcmVhZChidWYsIDEsIHVubGVuLCBtWmlwRnApOwotICAgICAgICAgICAgaWYgKGFtdCAhPSAoc3NpemVfdCl1bmxlbikgewotICAgICAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgICAgIH0KLSNpZiAwCi0gICAgICAgICAgICBwcmludGYoImRhdGEuLi5cbiIpOwotICAgICAgICAgICAgY29uc3QgdW5zaWduZWQgY2hhciogcCA9ICh1bnNpZ25lZCBjaGFyKilidWY7Ci0gICAgICAgICAgICBjb25zdCB1bnNpZ25lZCBjaGFyKiBlbmQgPSBwK3VubGVuOwotICAgICAgICAgICAgZm9yIChpbnQgaT0wOyBpPDMyICYmIHAgPCBlbmQ7IGkrKykgewotICAgICAgICAgICAgICAgIHByaW50ZigiMHglMDh4ICIsIChpbnQpKG9mZnNldCsoaSoweDEwKSkpOwotICAgICAgICAgICAgICAgIGZvciAoaW50IGo9MDsgajwweDEwICYmIHAgPCBlbmQ7IGorKykgewotICAgICAgICAgICAgICAgICAgICBwcmludGYoIiAlMDJ4IiwgKnApOwotICAgICAgICAgICAgICAgICAgICBwKys7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHByaW50ZigiXG4iKTsKLSAgICAgICAgICAgIH0KLSNlbmRpZgotCi0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBaaXBFbnRyeTo6a0NvbXByZXNzRGVmbGF0ZWQ6IHsKLSAgICAgICAgICAgIGlmICghWmlwVXRpbHM6OmluZmxhdGVUb0J1ZmZlcihtWmlwRnAsIGJ1ZiwgdW5sZW4sIGNsZW4pKSB7Ci0gICAgICAgICAgICAgICAgZ290byBiYWlsOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGRlZmF1bHQ6Ci0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotICAgIHJldHVybiBidWY7Ci0KLWJhaWw6Ci0gICAgZnJlZShidWYpOwotICAgIHJldHVybiBOVUxMOwotfQotCi0KLS8qCi0gKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLSAqCQlaaXBGaWxlOjpFbmRPZkNlbnRyYWxEaXIKLSAqID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotICovCi0KLS8qCi0gKiBSZWFkIHRoZSBlbmQtb2YtY2VudHJhbC1kaXIgZmllbGRzLgotICoKLSAqICJidWYiIHNob3VsZCBiZSBwb3NpdGlvbmVkIGF0IHRoZSBFT0NEIHNpZ25hdHVyZSwgYW5kIHNob3VsZCBjb250YWluCi0gKiB0aGUgZW50aXJlIEVPQ0QgYXJlYSBpbmNsdWRpbmcgdGhlIGNvbW1lbnQuCi0gKi8KLXN0YXR1c190IFppcEZpbGU6OkVuZE9mQ2VudHJhbERpcjo6cmVhZEJ1Zihjb25zdCB1bnNpZ25lZCBjaGFyKiBidWYsIGludCBsZW4pCi17Ci0gICAgLyogZG9uJ3QgYWxsb3cgcmUtdXNlICovCi0gICAgYXNzZXJ0KG1Db21tZW50ID09IE5VTEwpOwotCi0gICAgaWYgKGxlbiA8IGtFT0NETGVuKSB7Ci0gICAgICAgIC8qIGxvb2tzIGxpa2UgWklQIGZpbGUgZ290IHRydW5jYXRlZCAqLwotICAgICAgICBMT0dEKCIgWmlwIEVPQ0Q6IGV4cGVjdGVkID49ICVkIGJ5dGVzLCBmb3VuZCAlZFxuIiwKLSAgICAgICAgICAgIGtFT0NETGVuLCBsZW4pOwotICAgICAgICByZXR1cm4gSU5WQUxJRF9PUEVSQVRJT047Ci0gICAgfQotCi0gICAgLyogdGhpcyBzaG91bGQgcHJvYmFibHkgYmUgYW4gYXNzZXJ0KCkgKi8KLSAgICBpZiAoWmlwRW50cnk6OmdldExvbmdMRSgmYnVmWzB4MDBdKSAhPSBrU2lnbmF0dXJlKQotICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLQotICAgIG1EaXNrTnVtYmVyID0gWmlwRW50cnk6OmdldFNob3J0TEUoJmJ1ZlsweDA0XSk7Ci0gICAgbURpc2tXaXRoQ2VudHJhbERpciA9IFppcEVudHJ5OjpnZXRTaG9ydExFKCZidWZbMHgwNl0pOwotICAgIG1OdW1FbnRyaWVzID0gWmlwRW50cnk6OmdldFNob3J0TEUoJmJ1ZlsweDA4XSk7Ci0gICAgbVRvdGFsTnVtRW50cmllcyA9IFppcEVudHJ5OjpnZXRTaG9ydExFKCZidWZbMHgwYV0pOwotICAgIG1DZW50cmFsRGlyU2l6ZSA9IFppcEVudHJ5OjpnZXRMb25nTEUoJmJ1ZlsweDBjXSk7Ci0gICAgbUNlbnRyYWxEaXJPZmZzZXQgPSBaaXBFbnRyeTo6Z2V0TG9uZ0xFKCZidWZbMHgxMF0pOwotICAgIG1Db21tZW50TGVuID0gWmlwRW50cnk6OmdldFNob3J0TEUoJmJ1ZlsweDE0XSk7Ci0KLSAgICAvLyBUT0RPOiB2YWxpZGF0ZSBtQ2VudHJhbERpck9mZnNldAotCi0gICAgaWYgKG1Db21tZW50TGVuID4gMCkgewotICAgICAgICBpZiAoa0VPQ0RMZW4gKyBtQ29tbWVudExlbiA+IGxlbikgewotICAgICAgICAgICAgTE9HRCgiRU9DRCglZCkgKyBjb21tZW50KCVkKSBleGNlZWRzIGxlbiAoJWQpXG4iLAotICAgICAgICAgICAgICAgIGtFT0NETGVuLCBtQ29tbWVudExlbiwgbGVuKTsKLSAgICAgICAgICAgIHJldHVybiBVTktOT1dOX0VSUk9SOwotICAgICAgICB9Ci0gICAgICAgIG1Db21tZW50ID0gbmV3IHVuc2lnbmVkIGNoYXJbbUNvbW1lbnRMZW5dOwotICAgICAgICBtZW1jcHkobUNvbW1lbnQsIGJ1ZiArIGtFT0NETGVuLCBtQ29tbWVudExlbik7Ci0gICAgfQotCi0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vKgotICogV3JpdGUgYW4gZW5kLW9mLWNlbnRyYWwtZGlyZWN0b3J5IHNlY3Rpb24uCi0gKi8KLXN0YXR1c190IFppcEZpbGU6OkVuZE9mQ2VudHJhbERpcjo6d3JpdGUoRklMRSogZnApCi17Ci0gICAgdW5zaWduZWQgY2hhciBidWZba0VPQ0RMZW5dOwotCi0gICAgWmlwRW50cnk6OnB1dExvbmdMRSgmYnVmWzB4MDBdLCBrU2lnbmF0dXJlKTsKLSAgICBaaXBFbnRyeTo6cHV0U2hvcnRMRSgmYnVmWzB4MDRdLCBtRGlza051bWJlcik7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDA2XSwgbURpc2tXaXRoQ2VudHJhbERpcik7Ci0gICAgWmlwRW50cnk6OnB1dFNob3J0TEUoJmJ1ZlsweDA4XSwgbU51bUVudHJpZXMpOwotICAgIFppcEVudHJ5OjpwdXRTaG9ydExFKCZidWZbMHgwYV0sIG1Ub3RhbE51bUVudHJpZXMpOwotICAgIFppcEVudHJ5OjpwdXRMb25nTEUoJmJ1ZlsweDBjXSwgbUNlbnRyYWxEaXJTaXplKTsKLSAgICBaaXBFbnRyeTo6cHV0TG9uZ0xFKCZidWZbMHgxMF0sIG1DZW50cmFsRGlyT2Zmc2V0KTsKLSAgICBaaXBFbnRyeTo6cHV0U2hvcnRMRSgmYnVmWzB4MTRdLCBtQ29tbWVudExlbik7Ci0KLSAgICBpZiAoZndyaXRlKGJ1ZiwgMSwga0VPQ0RMZW4sIGZwKSAhPSBrRU9DRExlbikKLSAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0gICAgaWYgKG1Db21tZW50TGVuID4gMCkgewotICAgICAgICBhc3NlcnQobUNvbW1lbnQgIT0gTlVMTCk7Ci0gICAgICAgIGlmIChmd3JpdGUobUNvbW1lbnQsIG1Db21tZW50TGVuLCAxLCBmcCkgIT0gbUNvbW1lbnRMZW4pCi0gICAgICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICB9Ci0KLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLS8qCi0gKiBEdW1wIHRoZSBjb250ZW50cyBvZiBhbiBFbmRPZkNlbnRyYWxEaXIgb2JqZWN0LgotICovCi12b2lkIFppcEZpbGU6OkVuZE9mQ2VudHJhbERpcjo6ZHVtcCh2b2lkKSBjb25zdAotewotICAgIExPR0QoIiBFbmRPZkNlbnRyYWxEaXIgY29udGVudHM6XG4iKTsKLSAgICBMT0dEKCIgIGRpc2tOdW09JXUgZGlza1dDRD0ldSBudW1FbnQ9JXUgdG90YWxOdW1FbnQ9JXVcbiIsCi0gICAgICAgIG1EaXNrTnVtYmVyLCBtRGlza1dpdGhDZW50cmFsRGlyLCBtTnVtRW50cmllcywgbVRvdGFsTnVtRW50cmllcyk7Ci0gICAgTE9HRCgiICBjZW50RGlyU2l6ZT0lbHUgY2VudERpck9mZj0lbHUgY29tbWVudExlbj0ldVxuIiwKLSAgICAgICAgbUNlbnRyYWxEaXJTaXplLCBtQ2VudHJhbERpck9mZnNldCwgbUNvbW1lbnRMZW4pOwotfQotCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1ppcEZpbGVDUk8uY3BwIGIvbGlicy91dGlscy9aaXBGaWxlQ1JPLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZDMxMmRhZi4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1ppcEZpbGVDUk8uY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNTQgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlICJ1dGlscy9aaXBGaWxlQ1JPLmgiCi0jaW5jbHVkZSAidXRpbHMvWmlwRmlsZVJPLmgiCi0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi1aaXBGaWxlQ1JPIFppcEZpbGVYUk9fb3Blbihjb25zdCBjaGFyKiBwYXRoKSB7Ci0gICAgWmlwRmlsZVJPKiB6aXAgPSBuZXcgWmlwRmlsZVJPKCk7Ci0gICAgaWYgKHppcC0+b3BlbihwYXRoKSA9PSBOT19FUlJPUikgewotICAgICAgICByZXR1cm4gKFppcEZpbGVDUk8pemlwOwotICAgIH0KLSAgICByZXR1cm4gTlVMTDsKLX0KLQotdm9pZCBaaXBGaWxlQ1JPX2Rlc3Ryb3koWmlwRmlsZUNSTyB6aXBUb2tlbikgewotICAgIFppcEZpbGVSTyogemlwID0gKFppcEZpbGVSTyopemlwVG9rZW47Ci0gICAgZGVsZXRlIHppcDsKLX0KLQotWmlwRW50cnlDUk8gWmlwRmlsZUNST19maW5kRW50cnlCeU5hbWUoWmlwRmlsZUNSTyB6aXBUb2tlbiwKLSAgICAgICAgY29uc3QgY2hhciogZmlsZU5hbWUpIHsKLSAgICBaaXBGaWxlUk8qIHppcCA9IChaaXBGaWxlUk8qKXppcFRva2VuOwotICAgIHJldHVybiAoWmlwRW50cnlDUk8pemlwLT5maW5kRW50cnlCeU5hbWUoZmlsZU5hbWUpOwotfQotCi1ib29sIFppcEZpbGVDUk9fZ2V0RW50cnlJbmZvKFppcEZpbGVDUk8gemlwVG9rZW4sIFppcEVudHJ5Uk8gZW50cnlUb2tlbiwKLSAgICAgICAgaW50KiBwTWV0aG9kLCBsb25nKiBwVW5jb21wTGVuLAotICAgICAgICBsb25nKiBwQ29tcExlbiwgb2ZmX3QqIHBPZmZzZXQsIGxvbmcqIHBNb2RXaGVuLCBsb25nKiBwQ3JjMzIpIHsKLSAgICBaaXBGaWxlUk8qIHppcCA9IChaaXBGaWxlUk8qKXppcFRva2VuOwotICAgIFppcEVudHJ5Uk8gZW50cnkgPSAoWmlwRW50cnlSTyllbnRyeVRva2VuOwotICAgIHJldHVybiB6aXAtPmdldEVudHJ5SW5mbyhlbnRyeSwgcE1ldGhvZCwgcFVuY29tcExlbiwgcENvbXBMZW4sIHBPZmZzZXQsCi0gICAgICAgICAgICBwTW9kV2hlbiwgcENyYzMyKTsKLX0KLQotYm9vbCBaaXBGaWxlQ1JPX3VuY29tcHJlc3NFbnRyeShaaXBGaWxlQ1JPIHppcFRva2VuLCBaaXBFbnRyeVJPIGVudHJ5VG9rZW4sIGludCBmZCkgewotICAgIFppcEZpbGVSTyogemlwID0gKFppcEZpbGVSTyopemlwVG9rZW47Ci0gICAgWmlwRW50cnlSTyBlbnRyeSA9IChaaXBFbnRyeVJPKWVudHJ5VG9rZW47Ci0gICAgcmV0dXJuIHppcC0+dW5jb21wcmVzc0VudHJ5KGVudHJ5LCBmZCk7Ci19CmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL1ppcEZpbGVSTy5jcHAgYi9saWJzL3V0aWxzL1ppcEZpbGVSTy5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGFlOGM3MTkuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9aaXBGaWxlUk8uY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNzI0ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gUmVhZC1vbmx5IGFjY2VzcyB0byBaaXAgYXJjaGl2ZXMsIHdpdGggbWluaW1hbCBoZWFwIGFsbG9jYXRpb24uCi0vLwotI2RlZmluZSBMT0dfVEFHICJ6aXBybyIKLS8vI2RlZmluZSBMT0dfTkRFQlVHIDAKLSNpbmNsdWRlICJ1dGlscy9aaXBGaWxlUk8uaCIKLSNpbmNsdWRlICJ1dGlscy9Mb2cuaCIKLSNpbmNsdWRlICJ1dGlscy9taXNjLmgiCi0KLSNpbmNsdWRlIDx6bGliLmg+Ci0KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxmY250bC5oPgotI2luY2x1ZGUgPGVycm5vLmg+Ci0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi0vKgotICogWmlwIGZpbGUgY29uc3RhbnRzLgotICovCi0jZGVmaW5lIGtFT0NEU2lnbmF0dXJlICAgICAgMHgwNjA1NGI1MAotI2RlZmluZSBrRU9DRExlbiAgICAgICAgICAgIDIyCi0jZGVmaW5lIGtFT0NETnVtRW50cmllcyAgICAgOCAgICAgICAgICAgICAgIC8vIG9mZnNldCB0byAjb2YgZW50cmllcyBpbiBmaWxlCi0jZGVmaW5lIGtFT0NERmlsZU9mZnNldCAgICAgMTYgICAgICAgICAgICAgIC8vIG9mZnNldCB0byBjZW50cmFsIGRpcmVjdG9yeQotCi0jZGVmaW5lIGtNYXhDb21tZW50TGVuICAgICAgNjU1MzUgICAgICAgICAgIC8vIGxvbmdlc3QgcG9zc2libGUgaW4gdXNob3J0Ci0jZGVmaW5lIGtNYXhFT0NEU2VhcmNoICAgICAgKGtNYXhDb21tZW50TGVuICsga0VPQ0RMZW4pCi0KLSNkZWZpbmUga0xGSFNpZ25hdHVyZSAgICAgICAweDA0MDM0YjUwCi0jZGVmaW5lIGtMRkhMZW4gICAgICAgICAgICAgMzAgICAgICAgICAgICAgIC8vIGV4Y2x1ZGluZyB2YXJpYWJsZS1sZW4gZmllbGRzCi0jZGVmaW5lIGtMRkhOYW1lTGVuICAgICAgICAgMjYgICAgICAgICAgICAgIC8vIG9mZnNldCB0byBmaWxlbmFtZSBsZW5ndGgKLSNkZWZpbmUga0xGSEV4dHJhTGVuICAgICAgICAyOCAgICAgICAgICAgICAgLy8gb2Zmc2V0IHRvIGV4dHJhIGxlbmd0aAotCi0jZGVmaW5lIGtDREVTaWduYXR1cmUgICAgICAgMHgwMjAxNGI1MAotI2RlZmluZSBrQ0RFTGVuICAgICAgICAgICAgIDQ2ICAgICAgICAgICAgICAvLyBleGNsdWRpbmcgdmFyaWFibGUtbGVuIGZpZWxkcwotI2RlZmluZSBrQ0RFTWV0aG9kICAgICAgICAgIDEwICAgICAgICAgICAgICAvLyBvZmZzZXQgdG8gY29tcHJlc3Npb24gbWV0aG9kCi0jZGVmaW5lIGtDREVNb2RXaGVuICAgICAgICAgMTIgICAgICAgICAgICAgIC8vIG9mZnNldCB0byBtb2RpZmljYXRpb24gdGltZXN0YW1wCi0jZGVmaW5lIGtDREVDUkMgICAgICAgICAgICAgMTYgICAgICAgICAgICAgIC8vIG9mZnNldCB0byBlbnRyeSBDUkMKLSNkZWZpbmUga0NERUNvbXBMZW4gICAgICAgICAyMCAgICAgICAgICAgICAgLy8gb2Zmc2V0IHRvIGNvbXByZXNzZWQgbGVuZ3RoCi0jZGVmaW5lIGtDREVVbmNvbXBMZW4gICAgICAgMjQgICAgICAgICAgICAgIC8vIG9mZnNldCB0byB1bmNvbXByZXNzZWQgbGVuZ3RoCi0jZGVmaW5lIGtDREVOYW1lTGVuICAgICAgICAgMjggICAgICAgICAgICAgIC8vIG9mZnNldCB0byBmaWxlbmFtZSBsZW5ndGgKLSNkZWZpbmUga0NERUV4dHJhTGVuICAgICAgICAzMCAgICAgICAgICAgICAgLy8gb2Zmc2V0IHRvIGV4dHJhIGxlbmd0aAotI2RlZmluZSBrQ0RFQ29tbWVudExlbiAgICAgIDMyICAgICAgICAgICAgICAvLyBvZmZzZXQgdG8gY29tbWVudCBsZW5ndGgKLSNkZWZpbmUga0NERUxvY2FsT2Zmc2V0ICAgICA0MiAgICAgICAgICAgICAgLy8gb2Zmc2V0IHRvIGxvY2FsIGhkcgotCi0vKgotICogVGhlIHZhbHVlcyB3ZSByZXR1cm4gZm9yIFppcEVudHJ5Uk8gdXNlIDAgYXMgYW4gaW52YWxpZCB2YWx1ZSwgc28gd2UKLSAqIHdhbnQgdG8gYWRqdXN0IHRoZSBoYXNoIHRhYmxlIGluZGV4IGJ5IGEgZml4ZWQgYW1vdW50LiAgVXNpbmcgYSBsYXJnZQotICogdmFsdWUgaGVscHMgaW5zdXJlIHRoYXQgcGVvcGxlIGRvbid0IG1peCAmIG1hdGNoIGFyZ3VtZW50cywgZS5nLiB0bwotICogZmluZEVudHJ5QnlJbmRleCgpLgotICovCi0jZGVmaW5lIGtaaXBFbnRyeUFkaiAgICAgICAgMTAwMDAKLQotLyoKLSAqIENvbnZlcnQgYSBaaXBFbnRyeVJPIHRvIGEgaGFzaCB0YWJsZSBpbmRleCwgdmVyaWZ5aW5nIHRoYXQgaXQncyBpbiBhCi0gKiB2YWxpZCByYW5nZS4KLSAqLwotaW50IFppcEZpbGVSTzo6ZW50cnlUb0luZGV4KGNvbnN0IFppcEVudHJ5Uk8gZW50cnkpIGNvbnN0Ci17Ci0gICAgbG9uZyBlbnQgPSAoKGxvbmcpIGVudHJ5KSAtIGtaaXBFbnRyeUFkajsKLSAgICBpZiAoZW50IDwgMCB8fCBlbnQgPj0gbUhhc2hUYWJsZVNpemUgfHwgbUhhc2hUYWJsZVtlbnRdLm5hbWUgPT0gTlVMTCkgewotICAgICAgICBMT0dXKCJJbnZhbGlkIFppcEVudHJ5Uk8gJXAgKCVsZClcbiIsIGVudHJ5LCBlbnQpOwotICAgICAgICByZXR1cm4gLTE7Ci0gICAgfQotICAgIHJldHVybiBlbnQ7Ci19Ci0KLQotLyoKLSAqIE9wZW4gdGhlIHNwZWNpZmllZCBmaWxlIHJlYWQtb25seS4gIFdlIG1lbW9yeS1tYXAgdGhlIGVudGlyZSB0aGluZyBhbmQKLSAqIGNsb3NlIHRoZSBmaWxlIGJlZm9yZSByZXR1cm5pbmcuCi0gKi8KLXN0YXR1c190IFppcEZpbGVSTzo6b3Blbihjb25zdCBjaGFyKiB6aXBGaWxlTmFtZSkKLXsKLSAgICBpbnQgZmQgPSAtMTsKLSAgICBvZmZfdCBsZW5ndGg7Ci0KLSAgICBhc3NlcnQobUZpbGVNYXAgPT0gTlVMTCk7Ci0KLSAgICAvKgotICAgICAqIE9wZW4gYW5kIG1hcCB0aGUgc3BlY2lmaWVkIGZpbGUuCi0gICAgICovCi0gICAgZmQgPSA6Om9wZW4oemlwRmlsZU5hbWUsIE9fUkRPTkxZKTsKLSAgICBpZiAoZmQgPCAwKSB7Ci0gICAgICAgIExPR1coIlVuYWJsZSB0byBvcGVuIHppcCAnJXMnOiAlc1xuIiwgemlwRmlsZU5hbWUsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgICAgIHJldHVybiBOQU1FX05PVF9GT1VORDsKLSAgICB9Ci0KLSAgICBsZW5ndGggPSBsc2VlayhmZCwgMCwgU0VFS19FTkQpOwotICAgIGlmIChsZW5ndGggPCAwKSB7Ci0gICAgICAgIGNsb3NlKGZkKTsKLSAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0gICAgfQotCi0gICAgbUZpbGVNYXAgPSBuZXcgRmlsZU1hcCgpOwotICAgIGlmIChtRmlsZU1hcCA9PSBOVUxMKSB7Ci0gICAgICAgIGNsb3NlKGZkKTsKLSAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLSAgICB9Ci0gICAgaWYgKCFtRmlsZU1hcC0+Y3JlYXRlKHppcEZpbGVOYW1lLCBmZCwgMCwgbGVuZ3RoLCB0cnVlKSkgewotICAgICAgICBMT0dXKCJVbmFibGUgdG8gbWFwICclcyc6ICVzXG4iLCB6aXBGaWxlTmFtZSwgc3RyZXJyb3IoZXJybm8pKTsKLSAgICAgICAgY2xvc2UoZmQpOwotICAgICAgICByZXR1cm4gVU5LTk9XTl9FUlJPUjsKLSAgICB9Ci0KLSAgICBtRmQgPSBmZDsKLQotICAgIC8qCi0gICAgICogR290IGl0IG1hcHBlZCwgdmVyaWZ5IGl0IGFuZCBjcmVhdGUgZGF0YSBzdHJ1Y3R1cmVzIGZvciBmYXN0IGFjY2Vzcy4KLSAgICAgKi8KLSAgICBpZiAoIXBhcnNlWmlwQXJjaGl2ZSgpKSB7Ci0gICAgICAgIG1GaWxlTWFwLT5yZWxlYXNlKCk7Ci0gICAgICAgIG1GaWxlTWFwID0gTlVMTDsKLSAgICAgICAgcmV0dXJuIFVOS05PV05fRVJST1I7Ci0gICAgfQotCi0gICAgcmV0dXJuIE9LOwotfQotCi0vKgotICogUGFyc2UgdGhlIFppcCBhcmNoaXZlLCB2ZXJpZnlpbmcgaXRzIGNvbnRlbnRzIGFuZCBpbml0aWFsaXppbmcgaW50ZXJuYWwKLSAqIGRhdGEgc3RydWN0dXJlcy4KLSAqLwotYm9vbCBaaXBGaWxlUk86OnBhcnNlWmlwQXJjaGl2ZSh2b2lkKQotewotI2RlZmluZSBDSEVDS19PRkZTRVQoX29mZikgeyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLSAgICAgICAgaWYgKCh1bnNpZ25lZCBpbnQpIChfb2ZmKSA+PSBtYXhPZmZzZXQpIHsgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi0gICAgICAgICAgICBMT0dFKCJFUlJPUjogYmFkIG9mZnNldCAldSAobWF4ICVkKTogJXNcbiIsICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgICAgICh1bnNpZ25lZCBpbnQpIChfb2ZmKSwgbWF4T2Zmc2V0LCAjX29mZik7ICAgICAgICAgICAgICAgICAgIFwKLSAgICAgICAgICAgIGdvdG8gYmFpbDsgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi0gICAgICAgIH0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgIH0KLSAgICBjb25zdCB1bnNpZ25lZCBjaGFyKiBiYXNlUHRyID0gKGNvbnN0IHVuc2lnbmVkIGNoYXIqKW1GaWxlTWFwLT5nZXREYXRhUHRyKCk7Ci0gICAgY29uc3QgdW5zaWduZWQgY2hhciogcHRyOwotICAgIHNpemVfdCBsZW5ndGggPSBtRmlsZU1hcC0+Z2V0RGF0YUxlbmd0aCgpOwotICAgIGJvb2wgcmVzdWx0ID0gZmFsc2U7Ci0gICAgdW5zaWduZWQgaW50IGksIG51bUVudHJpZXMsIGNkT2Zmc2V0OwotICAgIHVuc2lnbmVkIGludCB2YWw7Ci0KLSAgICAvKgotICAgICAqIFRoZSBmaXJzdCA0IGJ5dGVzIG9mIHRoZSBmaWxlIHdpbGwgZWl0aGVyIGJlIHRoZSBsb2NhbCBoZWFkZXIKLSAgICAgKiBzaWduYXR1cmUgZm9yIHRoZSBmaXJzdCBmaWxlIChrTEZIU2lnbmF0dXJlKSBvciwgaWYgdGhlIGFyY2hpdmUgZG9lc24ndAotICAgICAqIGhhdmUgYW55IGZpbGVzIGluIGl0LCB0aGUgZW5kLW9mLWNlbnRyYWwtZGlyZWN0b3J5IHNpZ25hdHVyZQotICAgICAqIChrRU9DRFNpZ25hdHVyZSkuCi0gICAgICovCi0gICAgdmFsID0gZ2V0NExFKGJhc2VQdHIpOwotICAgIGlmICh2YWwgPT0ga0VPQ0RTaWduYXR1cmUpIHsKLSAgICAgICAgTE9HSSgiRm91bmQgWmlwIGFyY2hpdmUsIGJ1dCBpdCBsb29rcyBlbXB0eVxuIik7Ci0gICAgICAgIGdvdG8gYmFpbDsKLSAgICB9IGVsc2UgaWYgKHZhbCAhPSBrTEZIU2lnbmF0dXJlKSB7Ci0gICAgICAgIExPR1YoIk5vdCBhIFppcCBhcmNoaXZlIChmb3VuZCAweCUwOHgpXG4iLCB2YWwpOwotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBGaW5kIHRoZSBFT0NELiAgV2UnbGwgZmluZCBpdCBpbW1lZGlhdGVseSB1bmxlc3MgdGhleSBoYXZlIGEgZmlsZQotICAgICAqIGNvbW1lbnQuCi0gICAgICovCi0gICAgcHRyID0gYmFzZVB0ciArIGxlbmd0aCAtIGtFT0NETGVuOwotCi0gICAgd2hpbGUgKHB0ciA+PSBiYXNlUHRyKSB7Ci0gICAgICAgIGlmICgqcHRyID09IChrRU9DRFNpZ25hdHVyZSAmIDB4ZmYpICYmIGdldDRMRShwdHIpID09IGtFT0NEU2lnbmF0dXJlKQotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIHB0ci0tOwotICAgIH0KLSAgICBpZiAocHRyIDwgYmFzZVB0cikgewotICAgICAgICBMT0dJKCJDb3VsZCBub3QgZmluZCBlbmQtb2YtY2VudHJhbC1kaXJlY3RvcnkgaW4gWmlwXG4iKTsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIC8qCi0gICAgICogVGhlcmUgYXJlIHR3byBpbnRlcmVzdGluZyBpdGVtcyBpbiB0aGUgRU9DRCBibG9jazogdGhlIG51bWJlciBvZgotICAgICAqIGVudHJpZXMgaW4gdGhlIGZpbGUsIGFuZCB0aGUgZmlsZSBvZmZzZXQgb2YgdGhlIHN0YXJ0IG9mIHRoZQotICAgICAqIGNlbnRyYWwgZGlyZWN0b3J5LgotICAgICAqCi0gICAgICogKFRoZXJlJ3MgYWN0dWFsbHkgYSBjb3VudCBvZiB0aGUgI29mIGVudHJpZXMgaW4gdGhpcyBmaWxlLCBhbmQgZm9yCi0gICAgICogYWxsIGZpbGVzIHdoaWNoIGNvbXByaXNlIGEgc3Bhbm5lZCBhcmNoaXZlLCBidXQgZm9yIG91ciBwdXJwb3NlcwotICAgICAqIHdlJ3JlIG9ubHkgaW50ZXJlc3RlZCBpbiB0aGUgY3VycmVudCBmaWxlLiAgQmVzaWRlcywgd2UgZXhwZWN0IHRoZQotICAgICAqIHR3byB0byBiZSBlcXVpdmFsZW50IGZvciBvdXIgc3R1ZmYuKQotICAgICAqLwotICAgIG51bUVudHJpZXMgPSBnZXQyTEUocHRyICsga0VPQ0ROdW1FbnRyaWVzKTsKLSAgICBjZE9mZnNldCA9IGdldDRMRShwdHIgKyBrRU9DREZpbGVPZmZzZXQpOwotCi0gICAgLyogdmFsaWQgb2Zmc2V0cyBhcmUgWzAsRU9DRF0gKi8KLSAgICB1bnNpZ25lZCBpbnQgbWF4T2Zmc2V0OwotICAgIG1heE9mZnNldCA9IChwdHIgLSBiYXNlUHRyKSArMTsKLQotICAgIExPR1YoIisrKyBudW1FbnRyaWVzPSVkIGNkT2Zmc2V0PSVkXG4iLCBudW1FbnRyaWVzLCBjZE9mZnNldCk7Ci0gICAgaWYgKG51bUVudHJpZXMgPT0gMCB8fCBjZE9mZnNldCA+PSBsZW5ndGgpIHsKLSAgICAgICAgTE9HVygiSW52YWxpZCBlbnRyaWVzPSVkIG9mZnNldD0lZCAobGVuPSV6ZClcbiIsCi0gICAgICAgICAgICBudW1FbnRyaWVzLCBjZE9mZnNldCwgbGVuZ3RoKTsKLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIC8qCi0gICAgICogQ3JlYXRlIGhhc2ggdGFibGUuICBXZSBoYXZlIGEgbWluaW11bSA3NSUgbG9hZCBmYWN0b3IsIHBvc3NpYmx5IGFzCi0gICAgICogbG93IGFzIDUwJSBhZnRlciB3ZSByb3VuZCBvZmYgdG8gYSBwb3dlciBvZiAyLgotICAgICAqLwotICAgIG1OdW1FbnRyaWVzID0gbnVtRW50cmllczsKLSAgICBtSGFzaFRhYmxlU2l6ZSA9IHJvdW5kVXBQb3dlcjIoMSArICgobnVtRW50cmllcyAqIDQpIC8gMykpOwotICAgIG1IYXNoVGFibGUgPSAoSGFzaEVudHJ5KikgY2FsbG9jKDEsIHNpemVvZihIYXNoRW50cnkpICogbUhhc2hUYWJsZVNpemUpOwotCi0gICAgLyoKLSAgICAgKiBXYWxrIHRocm91Z2ggdGhlIGNlbnRyYWwgZGlyZWN0b3J5LCBhZGRpbmcgZW50cmllcyB0byB0aGUgaGFzaAotICAgICAqIHRhYmxlLgotICAgICAqLwotICAgIHB0ciA9IGJhc2VQdHIgKyBjZE9mZnNldDsKLSAgICBmb3IgKGkgPSAwOyBpIDwgbnVtRW50cmllczsgaSsrKSB7Ci0gICAgICAgIHVuc2lnbmVkIGludCBmaWxlTmFtZUxlbiwgZXh0cmFMZW4sIGNvbW1lbnRMZW4sIGxvY2FsSGRyT2Zmc2V0OwotICAgICAgICBjb25zdCB1bnNpZ25lZCBjaGFyKiBsb2NhbEhkcjsKLSAgICAgICAgdW5zaWduZWQgaW50IGhhc2g7Ci0KLSAgICAgICAgaWYgKGdldDRMRShwdHIpICE9IGtDREVTaWduYXR1cmUpIHsKLSAgICAgICAgICAgIExPR1coIk1pc3NlZCBhIGNlbnRyYWwgZGlyIHNpZyAoYXQgJWQpXG4iLCBpKTsKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgfQotICAgICAgICBpZiAocHRyICsga0NERUxlbiA+IGJhc2VQdHIgKyBsZW5ndGgpIHsKLSAgICAgICAgICAgIExPR1coIlJhbiBvZmYgdGhlIGVuZCAoYXQgJWQpXG4iLCBpKTsKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgfQotCi0gICAgICAgIGxvY2FsSGRyT2Zmc2V0ID0gZ2V0NExFKHB0ciArIGtDREVMb2NhbE9mZnNldCk7Ci0gICAgICAgIENIRUNLX09GRlNFVChsb2NhbEhkck9mZnNldCk7Ci0gICAgICAgIGZpbGVOYW1lTGVuID0gZ2V0MkxFKHB0ciArIGtDREVOYW1lTGVuKTsKLSAgICAgICAgZXh0cmFMZW4gPSBnZXQyTEUocHRyICsga0NERUV4dHJhTGVuKTsKLSAgICAgICAgY29tbWVudExlbiA9IGdldDJMRShwdHIgKyBrQ0RFQ29tbWVudExlbik7Ci0KLSAgICAgICAgLy9MT0dWKCIrKysgJWQ6IGxvY2FsSGRyPSVkIGZubD0lZCBlbD0lZCBjbD0lZFxuIiwKLSAgICAgICAgLy8gICAgaSwgbG9jYWxIZHJPZmZzZXQsIGZpbGVOYW1lTGVuLCBleHRyYUxlbiwgY29tbWVudExlbik7Ci0gICAgICAgIC8vTE9HVigiICclLipzJ1xuIiwgZmlsZU5hbWVMZW4sIHB0ciArIGtDREVMZW4pOwotCi0gICAgICAgIC8qIGFkZCB0aGUgQ0RFIGZpbGVuYW1lIHRvIHRoZSBoYXNoIHRhYmxlICovCi0gICAgICAgIGhhc2ggPSBjb21wdXRlSGFzaCgoY29uc3QgY2hhciopcHRyICsga0NERUxlbiwgZmlsZU5hbWVMZW4pOwotICAgICAgICBhZGRUb0hhc2goKGNvbnN0IGNoYXIqKXB0ciArIGtDREVMZW4sIGZpbGVOYW1lTGVuLCBoYXNoKTsKLQotICAgICAgICBsb2NhbEhkciA9IGJhc2VQdHIgKyBsb2NhbEhkck9mZnNldDsKLSAgICAgICAgaWYgKGdldDRMRShsb2NhbEhkcikgIT0ga0xGSFNpZ25hdHVyZSkgewotICAgICAgICAgICAgTE9HVygiQmFkIG9mZnNldCB0byBsb2NhbCBoZWFkZXI6ICVkIChhdCAlZClcbiIsCi0gICAgICAgICAgICAgICAgbG9jYWxIZHJPZmZzZXQsIGkpOwotICAgICAgICAgICAgZ290byBiYWlsOwotICAgICAgICB9Ci0KLSAgICAgICAgcHRyICs9IGtDREVMZW4gKyBmaWxlTmFtZUxlbiArIGV4dHJhTGVuICsgY29tbWVudExlbjsKLSAgICAgICAgQ0hFQ0tfT0ZGU0VUKHB0ciAtIGJhc2VQdHIpOwotICAgIH0KLQotICAgIHJlc3VsdCA9IHRydWU7Ci0KLWJhaWw6Ci0gICAgcmV0dXJuIHJlc3VsdDsKLSN1bmRlZiBDSEVDS19PRkZTRVQKLX0KLQotCi0vKgotICogU2ltcGxlIHN0cmluZyBoYXNoIGZ1bmN0aW9uIGZvciBub24tbnVsbC10ZXJtaW5hdGVkIHN0cmluZ3MuCi0gKi8KLS8qc3RhdGljKi8gdW5zaWduZWQgaW50IFppcEZpbGVSTzo6Y29tcHV0ZUhhc2goY29uc3QgY2hhciogc3RyLCBpbnQgbGVuKQotewotICAgIHVuc2lnbmVkIGludCBoYXNoID0gMDsKLQotICAgIHdoaWxlIChsZW4tLSkKLSAgICAgICAgaGFzaCA9IGhhc2ggKiAzMSArICpzdHIrKzsKLQotICAgIHJldHVybiBoYXNoOwotfQotCi0vKgotICogQWRkIGEgbmV3IGVudHJ5IHRvIHRoZSBoYXNoIHRhYmxlLgotICovCi12b2lkIFppcEZpbGVSTzo6YWRkVG9IYXNoKGNvbnN0IGNoYXIqIHN0ciwgaW50IHN0ckxlbiwgdW5zaWduZWQgaW50IGhhc2gpCi17Ci0gICAgaW50IGVudCA9IGhhc2ggJiAobUhhc2hUYWJsZVNpemUtMSk7Ci0KLSAgICAvKgotICAgICAqIFdlIG92ZXItYWxsb2NhdGUgdGhlIHRhYmxlLCBzbyB3ZSdyZSBndWFyYW50ZWVkIHRvIGZpbmQgYW4gZW1wdHkgc2xvdC4KLSAgICAgKi8KLSAgICB3aGlsZSAobUhhc2hUYWJsZVtlbnRdLm5hbWUgIT0gTlVMTCkKLSAgICAgICAgZW50ID0gKGVudCArIDEpICYgKG1IYXNoVGFibGVTaXplLTEpOwotCi0gICAgbUhhc2hUYWJsZVtlbnRdLm5hbWUgPSBzdHI7Ci0gICAgbUhhc2hUYWJsZVtlbnRdLm5hbWVMZW4gPSBzdHJMZW47Ci19Ci0KLS8qCi0gKiBGaW5kIGEgbWF0Y2hpbmcgZW50cnkuCi0gKgotICogUmV0dXJucyAwIGlmIG5vdCBmb3VuZC4KLSAqLwotWmlwRW50cnlSTyBaaXBGaWxlUk86OmZpbmRFbnRyeUJ5TmFtZShjb25zdCBjaGFyKiBmaWxlTmFtZSkgY29uc3QKLXsKLSAgICBpbnQgbmFtZUxlbiA9IHN0cmxlbihmaWxlTmFtZSk7Ci0gICAgdW5zaWduZWQgaW50IGhhc2ggPSBjb21wdXRlSGFzaChmaWxlTmFtZSwgbmFtZUxlbik7Ci0gICAgaW50IGVudCA9IGhhc2ggJiAobUhhc2hUYWJsZVNpemUtMSk7Ci0KLSAgICB3aGlsZSAobUhhc2hUYWJsZVtlbnRdLm5hbWUgIT0gTlVMTCkgewotICAgICAgICBpZiAobUhhc2hUYWJsZVtlbnRdLm5hbWVMZW4gPT0gbmFtZUxlbiAmJgotICAgICAgICAgICAgbWVtY21wKG1IYXNoVGFibGVbZW50XS5uYW1lLCBmaWxlTmFtZSwgbmFtZUxlbikgPT0gMCkKLSAgICAgICAgewotICAgICAgICAgICAgLyogbWF0Y2ggKi8KLSAgICAgICAgICAgIHJldHVybiAoWmlwRW50cnlSTykgKGVudCArIGtaaXBFbnRyeUFkaik7Ci0gICAgICAgIH0KLQotICAgICAgICBlbnQgPSAoZW50ICsgMSkgJiAobUhhc2hUYWJsZVNpemUtMSk7Ci0gICAgfQotCi0gICAgcmV0dXJuIE5VTEw7Ci19Ci0KLS8qCi0gKiBGaW5kIHRoZSBOdGggZW50cnkuCi0gKgotICogVGhpcyBjdXJyZW50bHkgaW52b2x2ZXMgd2Fsa2luZyB0aHJvdWdoIHRoZSBzcGFyc2UgaGFzaCB0YWJsZSwgY291bnRpbmcKLSAqIG5vbi1lbXB0eSBlbnRyaWVzLiAgSWYgd2UgbmVlZCB0byBzcGVlZCB0aGlzIHVwIHdlIGNhbiBlaXRoZXIgYWxsb2NhdGUKLSAqIGEgcGFyYWxsZWwgbG9va3VwIHRhYmxlIG9yIChwZXJoYXBzIGJldHRlcikgcHJvdmlkZSBhbiBpdGVyYXRvciBpbnRlcmZhY2UuCi0gKi8KLVppcEVudHJ5Uk8gWmlwRmlsZVJPOjpmaW5kRW50cnlCeUluZGV4KGludCBpZHgpIGNvbnN0Ci17Ci0gICAgaWYgKGlkeCA8IDAgfHwgaWR4ID49IG1OdW1FbnRyaWVzKSB7Ci0gICAgICAgIExPR1coIkludmFsaWQgaW5kZXggJWRcbiIsIGlkeCk7Ci0gICAgICAgIHJldHVybiBOVUxMOwotICAgIH0KLQotICAgIGZvciAoaW50IGVudCA9IDA7IGVudCA8IG1IYXNoVGFibGVTaXplOyBlbnQrKykgewotICAgICAgICBpZiAobUhhc2hUYWJsZVtlbnRdLm5hbWUgIT0gTlVMTCkgewotICAgICAgICAgICAgaWYgKGlkeC0tID09IDApCi0gICAgICAgICAgICAgICAgcmV0dXJuIChaaXBFbnRyeVJPKSAoZW50ICsga1ppcEVudHJ5QWRqKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBOVUxMOwotfQotCi0vKgotICogR2V0IHRoZSB1c2VmdWwgZmllbGRzIGZyb20gdGhlIHppcCBlbnRyeS4KLSAqCi0gKiBSZXR1cm5zICJmYWxzZSIgaWYgdGhlIG9mZnNldHMgdG8gdGhlIGZpZWxkcyBvciB0aGUgY29udGVudHMgb2YgdGhlIGZpZWxkcwotICogYXBwZWFyIHRvIGJlIGJvZ3VzLgotICovCi1ib29sIFppcEZpbGVSTzo6Z2V0RW50cnlJbmZvKFppcEVudHJ5Uk8gZW50cnksIGludCogcE1ldGhvZCwgbG9uZyogcFVuY29tcExlbiwKLSAgICBsb25nKiBwQ29tcExlbiwgb2ZmX3QqIHBPZmZzZXQsIGxvbmcqIHBNb2RXaGVuLCBsb25nKiBwQ3JjMzIpIGNvbnN0Ci17Ci0gICAgaW50IGVudCA9IGVudHJ5VG9JbmRleChlbnRyeSk7Ci0gICAgaWYgKGVudCA8IDApCi0gICAgICAgIHJldHVybiBmYWxzZTsKLQotICAgIC8qCi0gICAgICogUmVjb3ZlciB0aGUgc3RhcnQgb2YgdGhlIGNlbnRyYWwgZGlyZWN0b3J5IGVudHJ5IGZyb20gdGhlIGZpbGVuYW1lCi0gICAgICogcG9pbnRlci4KLSAgICAgKi8KLSAgICBjb25zdCB1bnNpZ25lZCBjaGFyKiBiYXNlUHRyID0gKGNvbnN0IHVuc2lnbmVkIGNoYXIqKW1GaWxlTWFwLT5nZXREYXRhUHRyKCk7Ci0gICAgY29uc3QgdW5zaWduZWQgY2hhciogcHRyID0gKGNvbnN0IHVuc2lnbmVkIGNoYXIqKSBtSGFzaFRhYmxlW2VudF0ubmFtZTsKLSAgICBzaXplX3QgemlwTGVuZ3RoID0gbUZpbGVNYXAtPmdldERhdGFMZW5ndGgoKTsKLQotICAgIHB0ciAtPSBrQ0RFTGVuOwotCi0gICAgaW50IG1ldGhvZCA9IGdldDJMRShwdHIgKyBrQ0RFTWV0aG9kKTsKLSAgICBpZiAocE1ldGhvZCAhPSBOVUxMKQotICAgICAgICAqcE1ldGhvZCA9IG1ldGhvZDsKLQotICAgIGlmIChwTW9kV2hlbiAhPSBOVUxMKQotICAgICAgICAqcE1vZFdoZW4gPSBnZXQ0TEUocHRyICsga0NERU1vZFdoZW4pOwotICAgIGlmIChwQ3JjMzIgIT0gTlVMTCkKLSAgICAgICAgKnBDcmMzMiA9IGdldDRMRShwdHIgKyBrQ0RFQ1JDKTsKLQotICAgIC8qCi0gICAgICogV2UgbmVlZCB0byBtYWtlIHN1cmUgdGhhdCB0aGUgbGVuZ3RocyBhcmUgbm90IHNvIGxhcmdlIHRoYXQgc29tZWJvZHkKLSAgICAgKiB0cnlpbmcgdG8gbWFwIHRoZSBjb21wcmVzc2VkIG9yIHVuY29tcHJlc3NlZCBkYXRhIHJ1bnMgb2ZmIHRoZSBlbmQKLSAgICAgKiBvZiB0aGUgbWFwcGVkIHJlZ2lvbi4KLSAgICAgKi8KLSAgICB1bnNpZ25lZCBsb25nIGxvY2FsSGRyT2Zmc2V0ID0gZ2V0NExFKHB0ciArIGtDREVMb2NhbE9mZnNldCk7Ci0gICAgaWYgKGxvY2FsSGRyT2Zmc2V0ICsga0xGSExlbiA+PSB6aXBMZW5ndGgpIHsKLSAgICAgICAgTE9HRSgiRVJST1I6IGJhZCBsb2NhbCBoZHIgb2Zmc2V0IGluIHppcFxuIik7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0gICAgY29uc3QgdW5zaWduZWQgY2hhciogbG9jYWxIZHIgPSBiYXNlUHRyICsgbG9jYWxIZHJPZmZzZXQ7Ci0gICAgb2ZmX3QgZGF0YU9mZnNldCA9IGxvY2FsSGRyT2Zmc2V0ICsga0xGSExlbgotICAgICAgICArIGdldDJMRShsb2NhbEhkciArIGtMRkhOYW1lTGVuKSArIGdldDJMRShsb2NhbEhkciArIGtMRkhFeHRyYUxlbik7Ci0gICAgaWYgKCh1bnNpZ25lZCBsb25nKSBkYXRhT2Zmc2V0ID49IHppcExlbmd0aCkgewotICAgICAgICBMT0dFKCJFUlJPUjogYmFkIGRhdGEgb2Zmc2V0IGluIHppcFxuIik7Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBpZiAocENvbXBMZW4gIT0gTlVMTCkgewotICAgICAgICAqcENvbXBMZW4gPSBnZXQ0TEUocHRyICsga0NERUNvbXBMZW4pOwotICAgICAgICBpZiAoKnBDb21wTGVuIDwgMCB8fCAoc2l6ZV90KShkYXRhT2Zmc2V0ICsgKnBDb21wTGVuKSA+PSB6aXBMZW5ndGgpIHsKLSAgICAgICAgICAgIExPR0UoIkVSUk9SOiBiYWQgY29tcHJlc3NlZCBsZW5ndGggaW4gemlwXG4iKTsKLSAgICAgICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAgICAgfQotICAgIH0KLSAgICBpZiAocFVuY29tcExlbiAhPSBOVUxMKSB7Ci0gICAgICAgICpwVW5jb21wTGVuID0gZ2V0NExFKHB0ciArIGtDREVVbmNvbXBMZW4pOwotICAgICAgICBpZiAoKnBVbmNvbXBMZW4gPCAwKSB7Ci0gICAgICAgICAgICBMT0dFKCJFUlJPUjogbmVnYXRpdmUgdW5jb21wcmVzc2VkIGxlbmd0aCBpbiB6aXBcbiIpOwotICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgICAgICB9Ci0gICAgICAgIGlmIChtZXRob2QgPT0ga0NvbXByZXNzU3RvcmVkICYmCi0gICAgICAgICAgICAoc2l6ZV90KShkYXRhT2Zmc2V0ICsgKnBVbmNvbXBMZW4pID49IHppcExlbmd0aCkKLSAgICAgICAgewotICAgICAgICAgICAgTE9HRSgiRVJST1I6IGJhZCB1bmNvbXByZXNzZWQgbGVuZ3RoIGluIHppcFxuIik7Ci0gICAgICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBpZiAocE9mZnNldCAhPSBOVUxMKSB7Ci0gICAgICAgICpwT2Zmc2V0ID0gZGF0YU9mZnNldDsKLSAgICB9Ci0gICAgcmV0dXJuIHRydWU7Ci19Ci0KLS8qCi0gKiBDb3B5IHRoZSBlbnRyeSdzIGZpbGVuYW1lIHRvIHRoZSBidWZmZXIuCi0gKi8KLWludCBaaXBGaWxlUk86OmdldEVudHJ5RmlsZU5hbWUoWmlwRW50cnlSTyBlbnRyeSwgY2hhciogYnVmZmVyLCBpbnQgYnVmTGVuKQotICAgIGNvbnN0Ci17Ci0gICAgaW50IGVudCA9IGVudHJ5VG9JbmRleChlbnRyeSk7Ci0gICAgaWYgKGVudCA8IDApCi0gICAgICAgIHJldHVybiAtMTsKLQotICAgIGludCBuYW1lTGVuID0gbUhhc2hUYWJsZVtlbnRdLm5hbWVMZW47Ci0gICAgaWYgKGJ1ZkxlbiA8IG5hbWVMZW4rMSkKLSAgICAgICAgcmV0dXJuIG5hbWVMZW4rMTsKLQotICAgIG1lbWNweShidWZmZXIsIG1IYXNoVGFibGVbZW50XS5uYW1lLCBuYW1lTGVuKTsKLSAgICBidWZmZXJbbmFtZUxlbl0gPSAnXDAnOwotICAgIHJldHVybiAwOwotfQotCi0vKgotICogQ3JlYXRlIGEgbmV3IEZpbGVNYXAgb2JqZWN0IHRoYXQgc3BhbnMgdGhlIGRhdGEgaW4gImVudHJ5Ii4KLSAqLwotRmlsZU1hcCogWmlwRmlsZVJPOjpjcmVhdGVFbnRyeUZpbGVNYXAoWmlwRW50cnlSTyBlbnRyeSkgY29uc3QKLXsKLSAgICAvKgotICAgICAqIFRPRE86IHRoZSBlZmZpY2llbnQgd2F5IHRvIGRvIHRoaXMgaXMgdG8gbW9kaWZ5IEZpbGVNYXAgdG8gYWxsb3cKLSAgICAgKiBzdWItcmVnaW9ucyBvZiBhIGZpbGUgdG8gYmUgbWFwcGVkLiAgQSByZWZlcmVuY2UtY291bnRpbmcgc2NoZW1lCi0gICAgICogY2FuIG1hbmFnZSB0aGUgYmFzZSBtZW1vcnkgbWFwcGluZy4gIEZvciBub3csIHdlIGp1c3QgY3JlYXRlIGEgYnJhbmQKLSAgICAgKiBuZXcgbWFwcGluZyBvZmYgb2YgdGhlIFppcCBhcmNoaXZlIGZpbGUgZGVzY3JpcHRvci4KLSAgICAgKi8KLQotICAgIEZpbGVNYXAqIG5ld01hcDsKLSAgICBsb25nIGNvbXBMZW47Ci0gICAgb2ZmX3Qgb2Zmc2V0OwotCi0gICAgaWYgKCFnZXRFbnRyeUluZm8oZW50cnksIE5VTEwsIE5VTEwsICZjb21wTGVuLCAmb2Zmc2V0LCBOVUxMLCBOVUxMKSkKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0KLSAgICBuZXdNYXAgPSBuZXcgRmlsZU1hcCgpOwotICAgIGlmICghbmV3TWFwLT5jcmVhdGUobUZpbGVNYXAtPmdldEZpbGVOYW1lKCksIG1GZCwgb2Zmc2V0LCBjb21wTGVuLCB0cnVlKSkgewotICAgICAgICBuZXdNYXAtPnJlbGVhc2UoKTsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotCi0gICAgcmV0dXJuIG5ld01hcDsKLX0KLQotLyoKLSAqIFVuY29tcHJlc3MgYW4gZW50cnksIGluIGl0cyBlbnRpcmV0eSwgaW50byB0aGUgcHJvdmlkZWQgb3V0cHV0IGJ1ZmZlci4KLSAqCi0gKiBUaGlzIGRvZXNuJ3QgdmVyaWZ5IHRoZSBkYXRhJ3MgQ1JDLCB3aGljaCBtaWdodCBiZSB1c2VmdWwgZm9yCi0gKiB1bmNvbXByZXNzZWQgZGF0YS4gIFRoZSBjYWxsZXIgc2hvdWxkIGJlIGFibGUgdG8gbWFuYWdlIGl0LgotICovCi1ib29sIFppcEZpbGVSTzo6dW5jb21wcmVzc0VudHJ5KFppcEVudHJ5Uk8gZW50cnksIHZvaWQqIGJ1ZmZlcikgY29uc3QKLXsKLSAgICBjb25zdCBpbnQga1NlcXVlbnRpYWxNaW4gPSAzMjc2ODsKLSAgICBib29sIHJlc3VsdCA9IGZhbHNlOwotICAgIGludCBlbnQgPSBlbnRyeVRvSW5kZXgoZW50cnkpOwotICAgIGlmIChlbnQgPCAwKQotICAgICAgICByZXR1cm4gLTE7Ci0KLSAgICBjb25zdCB1bnNpZ25lZCBjaGFyKiBiYXNlUHRyID0gKGNvbnN0IHVuc2lnbmVkIGNoYXIqKW1GaWxlTWFwLT5nZXREYXRhUHRyKCk7Ci0gICAgaW50IG1ldGhvZDsKLSAgICBsb25nIHVuY29tcExlbiwgY29tcExlbjsKLSAgICBvZmZfdCBvZmZzZXQ7Ci0KLSAgICBnZXRFbnRyeUluZm8oZW50cnksICZtZXRob2QsICZ1bmNvbXBMZW4sICZjb21wTGVuLCAmb2Zmc2V0LCBOVUxMLCBOVUxMKTsKLQotICAgIC8qCi0gICAgICogRXhwZXJpbWVudCB3aXRoIG1hZHZpc2UgaGludC4gIFdoZW4gd2Ugd2FudCB0byB1bmNvbXByZXNzIGEgZmlsZSwKLSAgICAgKiB3ZSBwdWxsIHNvbWUgc3R1ZmYgb3V0IG9mIHRoZSBjZW50cmFsIGRpciBlbnRyeSBhbmQgdGhlbiBoaXQgYQotICAgICAqIGJ1bmNoIG9mIGNvbXByZXNzZWQgb3IgdW5jb21wcmVzc2VkIGRhdGEgc2VxdWVudGlhbGx5LiAgVGhlIENERQotICAgICAqIHZpc2l0IHdpbGwgY2F1c2UgYSBsaW1pdGVkIGFtb3VudCBvZiByZWFkLWFoZWFkIGJlY2F1c2UgaXQncyBhdAotICAgICAqIHRoZSBlbmQgb2YgdGhlIGZpbGUuICBXZSBjb3VsZCBlbmQgdXAgZG9pbmcgbG90cyBvZiBleHRyYSBkaXNrCi0gICAgICogYWNjZXNzIGlmIHRoZSBmaWxlIHdlJ3JlIHByeWluZyBvcGVuIGlzIHNtYWxsLiAgQm90dG9tIGxpbmUgaXMgd2UKLSAgICAgKiBwcm9iYWJseSBkb24ndCB3YW50IHRvIHR1cm4gTUFEVl9TRVFVRU5USUFMIG9uIGFuZCBsZWF2ZSBpdCBvbi4KLSAgICAgKgotICAgICAqIFNvLCBpZiB0aGUgY29tcHJlc3NlZCBzaXplIG9mIHRoZSBmaWxlIGlzIGFib3ZlIGEgY2VydGFpbiBtaW5pbXVtCi0gICAgICogc2l6ZSwgdGVtcG9yYXJpbHkgYm9vc3QgdGhlIHJlYWQtYWhlYWQgaW4gdGhlIGhvcGUgdGhhdCB0aGUgZXh0cmEKLSAgICAgKiBwYWlyIG9mIHN5c3RlbSBjYWxscyBhcmUgbmVnYXRlZCBieSBhIHJlZHVjdGlvbiBpbiBwYWdlIGZhdWx0cy4KLSAgICAgKi8KLSAgICBpZiAoY29tcExlbiA+IGtTZXF1ZW50aWFsTWluKQotICAgICAgICBtRmlsZU1hcC0+YWR2aXNlKEZpbGVNYXA6OlNFUVVFTlRJQUwpOwotCi0gICAgaWYgKG1ldGhvZCA9PSBrQ29tcHJlc3NTdG9yZWQpIHsKLSAgICAgICAgbWVtY3B5KGJ1ZmZlciwgYmFzZVB0ciArIG9mZnNldCwgdW5jb21wTGVuKTsKLSAgICB9IGVsc2UgewotICAgICAgICBpZiAoIWluZmxhdGVCdWZmZXIoYnVmZmVyLCBiYXNlUHRyICsgb2Zmc2V0LCB1bmNvbXBMZW4sIGNvbXBMZW4pKQotICAgICAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIGlmIChjb21wTGVuID4ga1NlcXVlbnRpYWxNaW4pCi0gICAgICAgIG1GaWxlTWFwLT5hZHZpc2UoRmlsZU1hcDo6Tk9STUFMKTsKLQotICAgIHJlc3VsdCA9IHRydWU7Ci0KLWJhaWw6Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotLyoKLSAqIFVuY29tcHJlc3MgYW4gZW50cnksIGluIGl0cyBlbnRpcmV0eSwgdG8gYW4gb3BlbiBmaWxlIGRlc2NyaXB0b3IuCi0gKgotICogVGhpcyBkb2Vzbid0IHZlcmlmeSB0aGUgZGF0YSdzIENSQywgYnV0IHByb2JhYmx5IHNob3VsZC4KLSAqLwotYm9vbCBaaXBGaWxlUk86OnVuY29tcHJlc3NFbnRyeShaaXBFbnRyeVJPIGVudHJ5LCBpbnQgZmQpIGNvbnN0Ci17Ci0gICAgYm9vbCByZXN1bHQgPSBmYWxzZTsKLSAgICBpbnQgZW50ID0gZW50cnlUb0luZGV4KGVudHJ5KTsKLSAgICBpZiAoZW50IDwgMCkKLSAgICAgICAgcmV0dXJuIC0xOwotCi0gICAgY29uc3QgdW5zaWduZWQgY2hhciogYmFzZVB0ciA9IChjb25zdCB1bnNpZ25lZCBjaGFyKiltRmlsZU1hcC0+Z2V0RGF0YVB0cigpOwotICAgIGludCBtZXRob2Q7Ci0gICAgbG9uZyB1bmNvbXBMZW4sIGNvbXBMZW47Ci0gICAgb2ZmX3Qgb2Zmc2V0OwotCi0gICAgZ2V0RW50cnlJbmZvKGVudHJ5LCAmbWV0aG9kLCAmdW5jb21wTGVuLCAmY29tcExlbiwgJm9mZnNldCwgTlVMTCwgTlVMTCk7Ci0KLSAgICBpZiAobWV0aG9kID09IGtDb21wcmVzc1N0b3JlZCkgewotICAgICAgICBzc2l6ZV90IGFjdHVhbDsKLQotICAgICAgICBhY3R1YWwgPSB3cml0ZShmZCwgYmFzZVB0ciArIG9mZnNldCwgdW5jb21wTGVuKTsKLSAgICAgICAgaWYgKGFjdHVhbCA8IDApIHsKLSAgICAgICAgICAgIExPR0UoIldyaXRlIGZhaWxlZDogJXNcbiIsIHN0cmVycm9yKGVycm5vKSk7Ci0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgICAgIH0gZWxzZSBpZiAoYWN0dWFsICE9IHVuY29tcExlbikgewotICAgICAgICAgICAgTE9HRSgiUGFydGlhbCB3cml0ZSBkdXJpbmcgdW5jb21wcmVzcyAoJWQgb2YgJWxkKVxuIiwKLSAgICAgICAgICAgICAgICAoaW50KWFjdHVhbCwgdW5jb21wTGVuKTsKLSAgICAgICAgICAgIGdvdG8gYmFpbDsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIExPR0koIisrKyBzdWNjZXNzZnVsIHdyaXRlXG4iKTsKLSAgICAgICAgfQotICAgIH0gZWxzZSB7Ci0gICAgICAgIGlmICghaW5mbGF0ZUJ1ZmZlcihmZCwgYmFzZVB0citvZmZzZXQsIHVuY29tcExlbiwgY29tcExlbikpCi0gICAgICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgcmVzdWx0ID0gdHJ1ZTsKLQotYmFpbDoKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi0vKgotICogVW5jb21wcmVzcyAiZGVmbGF0ZSIgZGF0YSBmcm9tIG9uZSBidWZmZXIgdG8gYW5vdGhlci4KLSAqLwotLypzdGF0aWMqLyBib29sIFppcEZpbGVSTzo6aW5mbGF0ZUJ1ZmZlcih2b2lkKiBvdXRCdWYsIGNvbnN0IHZvaWQqIGluQnVmLAotICAgIGxvbmcgdW5jb21wTGVuLCBsb25nIGNvbXBMZW4pCi17Ci0gICAgYm9vbCByZXN1bHQgPSBmYWxzZTsKLSAgICB6X3N0cmVhbSB6c3RyZWFtOwotICAgIGludCB6ZXJyOwotCi0gICAgLyoKLSAgICAgKiBJbml0aWFsaXplIHRoZSB6bGliIHN0cmVhbSBzdHJ1Y3QuCi0gICAgICovCi0JbWVtc2V0KCZ6c3RyZWFtLCAwLCBzaXplb2YoenN0cmVhbSkpOwotICAgIHpzdHJlYW0uemFsbG9jID0gWl9OVUxMOwotICAgIHpzdHJlYW0uemZyZWUgPSBaX05VTEw7Ci0gICAgenN0cmVhbS5vcGFxdWUgPSBaX05VTEw7Ci0gICAgenN0cmVhbS5uZXh0X2luID0gKEJ5dGVmKilpbkJ1ZjsKLSAgICB6c3RyZWFtLmF2YWlsX2luID0gY29tcExlbjsKLSAgICB6c3RyZWFtLm5leHRfb3V0ID0gKEJ5dGVmKikgb3V0QnVmOwotICAgIHpzdHJlYW0uYXZhaWxfb3V0ID0gdW5jb21wTGVuOwotICAgIHpzdHJlYW0uZGF0YV90eXBlID0gWl9VTktOT1dOOwotCi0JLyoKLQkgKiBVc2UgdGhlIHVuZG9jdW1lbnRlZCAibmVnYXRpdmUgd2luZG93IGJpdHMiIGZlYXR1cmUgdG8gdGVsbCB6bGliCi0JICogdGhhdCB0aGVyZSdzIG5vIHpsaWIgaGVhZGVyIHdhaXRpbmcgZm9yIGl0LgotCSAqLwotICAgIHplcnIgPSBpbmZsYXRlSW5pdDIoJnpzdHJlYW0sIC1NQVhfV0JJVFMpOwotICAgIGlmICh6ZXJyICE9IFpfT0spIHsKLSAgICAgICAgaWYgKHplcnIgPT0gWl9WRVJTSU9OX0VSUk9SKSB7Ci0gICAgICAgICAgICBMT0dFKCJJbnN0YWxsZWQgemxpYiBpcyBub3QgY29tcGF0aWJsZSB3aXRoIGxpbmtlZCB2ZXJzaW9uICglcylcbiIsCi0gICAgICAgICAgICAgICAgWkxJQl9WRVJTSU9OKTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIExPR0UoIkNhbGwgdG8gaW5mbGF0ZUluaXQyIGZhaWxlZCAoemVycj0lZClcbiIsIHplcnIpOwotICAgICAgICB9Ci0gICAgICAgIGdvdG8gYmFpbDsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIEV4cGFuZCBkYXRhLgotICAgICAqLwotICAgIHplcnIgPSBpbmZsYXRlKCZ6c3RyZWFtLCBaX0ZJTklTSCk7Ci0gICAgaWYgKHplcnIgIT0gWl9TVFJFQU1fRU5EKSB7Ci0gICAgICAgIExPR1coIlppcCBpbmZsYXRlIGZhaWxlZCwgemVycj0lZCAobkluPSVwIGFJbj0ldSBuT3V0PSVwIGFPdXQ9JXUpXG4iLAotICAgICAgICAgICAgemVyciwgenN0cmVhbS5uZXh0X2luLCB6c3RyZWFtLmF2YWlsX2luLAotICAgICAgICAgICAgenN0cmVhbS5uZXh0X291dCwgenN0cmVhbS5hdmFpbF9vdXQpOwotICAgICAgICBnb3RvIHpfYmFpbDsKLSAgICB9Ci0KLSAgICAvKiBwYXJhbm9pYSAqLwotICAgIGlmICgobG9uZykgenN0cmVhbS50b3RhbF9vdXQgIT0gdW5jb21wTGVuKSB7Ci0gICAgICAgIExPR1coIlNpemUgbWlzbWF0Y2ggb24gaW5mbGF0ZWQgZmlsZSAoJWxkIHZzICVsZClcbiIsCi0gICAgICAgICAgICB6c3RyZWFtLnRvdGFsX291dCwgdW5jb21wTGVuKTsKLSAgICAgICAgZ290byB6X2JhaWw7Ci0gICAgfQotCi0gICAgcmVzdWx0ID0gdHJ1ZTsKLQotel9iYWlsOgotICAgIGluZmxhdGVFbmQoJnpzdHJlYW0pOyAgICAgICAgLyogZnJlZSB1cCBhbnkgYWxsb2NhdGVkIHN0cnVjdHVyZXMgKi8KLQotYmFpbDoKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi0vKgotICogVW5jb21wcmVzcyAiZGVmbGF0ZSIgZGF0YSBmcm9tIG9uZSBidWZmZXIgdG8gYW4gb3BlbiBmaWxlIGRlc2NyaXB0b3IuCi0gKi8KLS8qc3RhdGljKi8gYm9vbCBaaXBGaWxlUk86OmluZmxhdGVCdWZmZXIoaW50IGZkLCBjb25zdCB2b2lkKiBpbkJ1ZiwKLSAgICBsb25nIHVuY29tcExlbiwgbG9uZyBjb21wTGVuKQotewotICAgIGJvb2wgcmVzdWx0ID0gZmFsc2U7Ci0gICAgY29uc3QgaW50IGtXcml0ZUJ1ZlNpemUgPSAzMjc2ODsKLSAgICB1bnNpZ25lZCBjaGFyIHdyaXRlQnVmW2tXcml0ZUJ1ZlNpemVdOwotICAgIHpfc3RyZWFtIHpzdHJlYW07Ci0gICAgaW50IHplcnI7Ci0KLSAgICAvKgotICAgICAqIEluaXRpYWxpemUgdGhlIHpsaWIgc3RyZWFtIHN0cnVjdC4KLSAgICAgKi8KLQltZW1zZXQoJnpzdHJlYW0sIDAsIHNpemVvZih6c3RyZWFtKSk7Ci0gICAgenN0cmVhbS56YWxsb2MgPSBaX05VTEw7Ci0gICAgenN0cmVhbS56ZnJlZSA9IFpfTlVMTDsKLSAgICB6c3RyZWFtLm9wYXF1ZSA9IFpfTlVMTDsKLSAgICB6c3RyZWFtLm5leHRfaW4gPSAoQnl0ZWYqKWluQnVmOwotICAgIHpzdHJlYW0uYXZhaWxfaW4gPSBjb21wTGVuOwotICAgIHpzdHJlYW0ubmV4dF9vdXQgPSAoQnl0ZWYqKSB3cml0ZUJ1ZjsKLSAgICB6c3RyZWFtLmF2YWlsX291dCA9IHNpemVvZih3cml0ZUJ1Zik7Ci0gICAgenN0cmVhbS5kYXRhX3R5cGUgPSBaX1VOS05PV047Ci0KLQkvKgotCSAqIFVzZSB0aGUgdW5kb2N1bWVudGVkICJuZWdhdGl2ZSB3aW5kb3cgYml0cyIgZmVhdHVyZSB0byB0ZWxsIHpsaWIKLQkgKiB0aGF0IHRoZXJlJ3Mgbm8gemxpYiBoZWFkZXIgd2FpdGluZyBmb3IgaXQuCi0JICovCi0gICAgemVyciA9IGluZmxhdGVJbml0MigmenN0cmVhbSwgLU1BWF9XQklUUyk7Ci0gICAgaWYgKHplcnIgIT0gWl9PSykgewotICAgICAgICBpZiAoemVyciA9PSBaX1ZFUlNJT05fRVJST1IpIHsKLSAgICAgICAgICAgIExPR0UoIkluc3RhbGxlZCB6bGliIGlzIG5vdCBjb21wYXRpYmxlIHdpdGggbGlua2VkIHZlcnNpb24gKCVzKVxuIiwKLSAgICAgICAgICAgICAgICBaTElCX1ZFUlNJT04pOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgTE9HRSgiQ2FsbCB0byBpbmZsYXRlSW5pdDIgZmFpbGVkICh6ZXJyPSVkKVxuIiwgemVycik7Ci0gICAgICAgIH0KLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIC8qCi0gICAgICogTG9vcCB3aGlsZSB3ZSBoYXZlIG1vcmUgdG8gZG8uCi0gICAgICovCi0gICAgZG8gewotICAgICAgICAvKgotICAgICAgICAgKiBFeHBhbmQgZGF0YS4KLSAgICAgICAgICovCi0gICAgICAgIHplcnIgPSBpbmZsYXRlKCZ6c3RyZWFtLCBaX05PX0ZMVVNIKTsKLSAgICAgICAgaWYgKHplcnIgIT0gWl9PSyAmJiB6ZXJyICE9IFpfU1RSRUFNX0VORCkgewotICAgICAgICAgICAgTE9HVygiemxpYiBpbmZsYXRlOiB6ZXJyPSVkIChuSW49JXAgYUluPSV1IG5PdXQ9JXAgYU91dD0ldSlcbiIsCi0gICAgICAgICAgICAgICAgemVyciwgenN0cmVhbS5uZXh0X2luLCB6c3RyZWFtLmF2YWlsX2luLAotICAgICAgICAgICAgICAgIHpzdHJlYW0ubmV4dF9vdXQsIHpzdHJlYW0uYXZhaWxfb3V0KTsKLSAgICAgICAgICAgIGdvdG8gel9iYWlsOwotICAgICAgICB9Ci0KLSAgICAgICAgLyogd3JpdGUgd2hlbiB3ZSdyZSBmdWxsIG9yIHdoZW4gd2UncmUgZG9uZSAqLwotICAgICAgICBpZiAoenN0cmVhbS5hdmFpbF9vdXQgPT0gMCB8fAotICAgICAgICAgICAgKHplcnIgPT0gWl9TVFJFQU1fRU5EICYmIHpzdHJlYW0uYXZhaWxfb3V0ICE9IHNpemVvZih3cml0ZUJ1ZikpKQotICAgICAgICB7Ci0gICAgICAgICAgICBsb25nIHdyaXRlU2l6ZSA9IHpzdHJlYW0ubmV4dF9vdXQgLSB3cml0ZUJ1ZjsKLSAgICAgICAgICAgIGludCBjYyA9IHdyaXRlKGZkLCB3cml0ZUJ1Ziwgd3JpdGVTaXplKTsKLSAgICAgICAgICAgIGlmIChjYyAhPSAoaW50KSB3cml0ZVNpemUpIHsKLSAgICAgICAgICAgICAgICBMT0dXKCJ3cml0ZSBmYWlsZWQgaW4gaW5mbGF0ZSAoJWQgdnMgJWxkKVxuIiwgY2MsIHdyaXRlU2l6ZSk7Ci0gICAgICAgICAgICAgICAgZ290byB6X2JhaWw7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHpzdHJlYW0ubmV4dF9vdXQgPSB3cml0ZUJ1ZjsKLSAgICAgICAgICAgIHpzdHJlYW0uYXZhaWxfb3V0ID0gc2l6ZW9mKHdyaXRlQnVmKTsKLSAgICAgICAgfQotICAgIH0gd2hpbGUgKHplcnIgPT0gWl9PSyk7Ci0KLSAgICBhc3NlcnQoemVyciA9PSBaX1NUUkVBTV9FTkQpOyAgICAgICAvKiBvdGhlciBlcnJvcnMgc2hvdWxkJ3ZlIGJlZW4gY2F1Z2h0ICovCi0KLSAgICAvKiBwYXJhbm9pYSAqLwotICAgIGlmICgobG9uZykgenN0cmVhbS50b3RhbF9vdXQgIT0gdW5jb21wTGVuKSB7Ci0gICAgICAgIExPR1coIlNpemUgbWlzbWF0Y2ggb24gaW5mbGF0ZWQgZmlsZSAoJWxkIHZzICVsZClcbiIsCi0gICAgICAgICAgICB6c3RyZWFtLnRvdGFsX291dCwgdW5jb21wTGVuKTsKLSAgICAgICAgZ290byB6X2JhaWw7Ci0gICAgfQotCi0gICAgcmVzdWx0ID0gdHJ1ZTsKLQotel9iYWlsOgotICAgIGluZmxhdGVFbmQoJnpzdHJlYW0pOyAgICAgICAgLyogZnJlZSB1cCBhbnkgYWxsb2NhdGVkIHN0cnVjdHVyZXMgKi8KLQotYmFpbDoKLSAgICByZXR1cm4gcmVzdWx0OwotfQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9aaXBVdGlscy5jcHAgYi9saWJzL3V0aWxzL1ppcFV0aWxzLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYmZiYWNmZS4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL1ppcFV0aWxzLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDM0NCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIE1pc2MgemlwL2d6aXAgdXRpbGl0eSBmdW5jdGlvbnMuCi0vLwotCi0jZGVmaW5lIExPR19UQUcgInppcHV0aWwiCi0KLSNpbmNsdWRlICJ1dGlscy9aaXBVdGlscy5oIgotI2luY2x1ZGUgInV0aWxzL1ppcEZpbGVSTy5oIgotI2luY2x1ZGUgInV0aWxzL0xvZy5oIgotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0KLSNpbmNsdWRlIDx6bGliLmg+Ci0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi0vKgotICogVXRpbGl0eSBmdW5jdGlvbiB0aGF0IGV4cGFuZHMgemlwL2d6aXAgImRlZmxhdGUiIGNvbXByZXNzZWQgZGF0YQotICogaW50byBhIGJ1ZmZlci4KLSAqCi0gKiAiZmQiIGlzIGFuIG9wZW4gZmlsZSBwb3NpdGlvbmVkIGF0IHRoZSBzdGFydCBvZiB0aGUgImRlZmxhdGUiIGRhdGEKLSAqICJidWYiIG11c3QgaG9sZCBhdCBsZWFzdCAidW5jb21wcmVzc2VkTGVuIiBieXRlcy4KLSAqLwotLypzdGF0aWMqLyBib29sIFppcFV0aWxzOjppbmZsYXRlVG9CdWZmZXIoaW50IGZkLCB2b2lkKiBidWYsCi0gICAgbG9uZyB1bmNvbXByZXNzZWRMZW4sIGxvbmcgY29tcHJlc3NlZExlbikKLXsKLSAgICBib29sIHJlc3VsdCA9IGZhbHNlOwotCWNvbnN0IHVuc2lnbmVkIGxvbmcga1JlYWRCdWZTaXplID0gMzI3Njg7Ci0JdW5zaWduZWQgY2hhciogcmVhZEJ1ZiA9IE5VTEw7Ci0gICAgel9zdHJlYW0genN0cmVhbTsKLSAgICBpbnQgemVycjsKLSAgICB1bnNpZ25lZCBsb25nIGNvbXBSZW1haW5pbmc7Ci0KLSAgICBhc3NlcnQodW5jb21wcmVzc2VkTGVuID49IDApOwotICAgIGFzc2VydChjb21wcmVzc2VkTGVuID49IDApOwotCi0JcmVhZEJ1ZiA9IG5ldyB1bnNpZ25lZCBjaGFyW2tSZWFkQnVmU2l6ZV07Ci0JaWYgKHJlYWRCdWYgPT0gTlVMTCkKLSAgICAgICAgZ290byBiYWlsOwotICAgIGNvbXBSZW1haW5pbmcgPSBjb21wcmVzc2VkTGVuOwotCi0gICAgLyoKLSAgICAgKiBJbml0aWFsaXplIHRoZSB6bGliIHN0cmVhbS4KLSAgICAgKi8KLQltZW1zZXQoJnpzdHJlYW0sIDAsIHNpemVvZih6c3RyZWFtKSk7Ci0gICAgenN0cmVhbS56YWxsb2MgPSBaX05VTEw7Ci0gICAgenN0cmVhbS56ZnJlZSA9IFpfTlVMTDsKLSAgICB6c3RyZWFtLm9wYXF1ZSA9IFpfTlVMTDsKLSAgICB6c3RyZWFtLm5leHRfaW4gPSBOVUxMOwotICAgIHpzdHJlYW0uYXZhaWxfaW4gPSAwOwotICAgIHpzdHJlYW0ubmV4dF9vdXQgPSAoQnl0ZWYqKSBidWY7Ci0gICAgenN0cmVhbS5hdmFpbF9vdXQgPSB1bmNvbXByZXNzZWRMZW47Ci0gICAgenN0cmVhbS5kYXRhX3R5cGUgPSBaX1VOS05PV047Ci0KLQkvKgotCSAqIFVzZSB0aGUgdW5kb2N1bWVudGVkICJuZWdhdGl2ZSB3aW5kb3cgYml0cyIgZmVhdHVyZSB0byB0ZWxsIHpsaWIKLQkgKiB0aGF0IHRoZXJlJ3Mgbm8gemxpYiBoZWFkZXIgd2FpdGluZyBmb3IgaXQuCi0JICovCi0gICAgemVyciA9IGluZmxhdGVJbml0MigmenN0cmVhbSwgLU1BWF9XQklUUyk7Ci0gICAgaWYgKHplcnIgIT0gWl9PSykgewotICAgICAgICBpZiAoemVyciA9PSBaX1ZFUlNJT05fRVJST1IpIHsKLSAgICAgICAgICAgIExPR0UoIkluc3RhbGxlZCB6bGliIGlzIG5vdCBjb21wYXRpYmxlIHdpdGggbGlua2VkIHZlcnNpb24gKCVzKVxuIiwKLSAgICAgICAgICAgICAgICBaTElCX1ZFUlNJT04pOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgTE9HRSgiQ2FsbCB0byBpbmZsYXRlSW5pdDIgZmFpbGVkICh6ZXJyPSVkKVxuIiwgemVycik7Ci0gICAgICAgIH0KLSAgICAgICAgZ290byBiYWlsOwotICAgIH0KLQotICAgIC8qCi0gICAgICogTG9vcCB3aGlsZSB3ZSBoYXZlIGRhdGEuCi0gICAgICovCi0gICAgZG8gewotICAgICAgICB1bnNpZ25lZCBsb25nIGdldFNpemU7Ci0KLSAgICAgICAgLyogcmVhZCBhcyBtdWNoIGFzIHdlIGNhbiAqLwotICAgICAgICBpZiAoenN0cmVhbS5hdmFpbF9pbiA9PSAwKSB7Ci0gICAgICAgICAgICBnZXRTaXplID0gKGNvbXBSZW1haW5pbmcgPiBrUmVhZEJ1ZlNpemUpID8KLSAgICAgICAgICAgICAgICAgICAgICAgIGtSZWFkQnVmU2l6ZSA6IGNvbXBSZW1haW5pbmc7Ci0gICAgICAgICAgICBMT0dWKCIrKysgcmVhZGluZyAlbGQgYnl0ZXMgKCVsZCBsZWZ0KVxuIiwKLSAgICAgICAgICAgICAgICBnZXRTaXplLCBjb21wUmVtYWluaW5nKTsKLQotICAgICAgICAgICAgaW50IGNjID0gcmVhZChmZCwgcmVhZEJ1ZiwgZ2V0U2l6ZSk7Ci0gICAgICAgICAgICBpZiAoY2MgIT0gKGludCkgZ2V0U2l6ZSkgewotICAgICAgICAgICAgICAgIExPR0QoImluZmxhdGUgcmVhZCBmYWlsZWQgKCVkIHZzICVsZClcbiIsCi0gICAgICAgICAgICAgICAgICAgIGNjLCBnZXRTaXplKTsKLSAgICAgICAgICAgICAgICBnb3RvIHpfYmFpbDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY29tcFJlbWFpbmluZyAtPSBnZXRTaXplOwotCi0gICAgICAgICAgICB6c3RyZWFtLm5leHRfaW4gPSByZWFkQnVmOwotICAgICAgICAgICAgenN0cmVhbS5hdmFpbF9pbiA9IGdldFNpemU7Ci0gICAgICAgIH0KLQotICAgICAgICAvKiB1bmNvbXByZXNzIHRoZSBkYXRhICovCi0gICAgICAgIHplcnIgPSBpbmZsYXRlKCZ6c3RyZWFtLCBaX05PX0ZMVVNIKTsKLSAgICAgICAgaWYgKHplcnIgIT0gWl9PSyAmJiB6ZXJyICE9IFpfU1RSRUFNX0VORCkgewotICAgICAgICAgICAgTE9HRCgiemxpYiBpbmZsYXRlIGNhbGwgZmFpbGVkICh6ZXJyPSVkKVxuIiwgemVycik7Ci0gICAgICAgICAgICBnb3RvIHpfYmFpbDsKLSAgICAgICAgfQotCi0JCS8qIG91dHB1dCBidWZmZXIgaG9sZHMgYWxsLCBzbyBubyBuZWVkIHRvIHdyaXRlIHRoZSBvdXRwdXQgKi8KLSAgICB9IHdoaWxlICh6ZXJyID09IFpfT0spOwotCi0gICAgYXNzZXJ0KHplcnIgPT0gWl9TVFJFQU1fRU5EKTsgICAgICAgLyogb3RoZXIgZXJyb3JzIHNob3VsZCd2ZSBiZWVuIGNhdWdodCAqLwotCi0gICAgaWYgKChsb25nKSB6c3RyZWFtLnRvdGFsX291dCAhPSB1bmNvbXByZXNzZWRMZW4pIHsKLSAgICAgICAgTE9HVygiU2l6ZSBtaXNtYXRjaCBvbiBpbmZsYXRlZCBmaWxlICglbGQgdnMgJWxkKVxuIiwKLSAgICAgICAgICAgIHpzdHJlYW0udG90YWxfb3V0LCB1bmNvbXByZXNzZWRMZW4pOwotICAgICAgICBnb3RvIHpfYmFpbDsKLSAgICB9Ci0KLSAgICAvLyBzdWNjZXNzIQotICAgIHJlc3VsdCA9IHRydWU7Ci0KLXpfYmFpbDoKLSAgICBpbmZsYXRlRW5kKCZ6c3RyZWFtKTsgICAgICAgIC8qIGZyZWUgdXAgYW55IGFsbG9jYXRlZCBzdHJ1Y3R1cmVzICovCi0KLWJhaWw6Ci0JZGVsZXRlW10gcmVhZEJ1ZjsKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi0vKgotICogVXRpbGl0eSBmdW5jdGlvbiB0aGF0IGV4cGFuZHMgemlwL2d6aXAgImRlZmxhdGUiIGNvbXByZXNzZWQgZGF0YQotICogaW50byBhIGJ1ZmZlci4KLSAqCi0gKiAoVGhpcyBpcyBhIGNsb25lIG9mIHRoZSBwcmV2aW91cyBmdW5jdGlvbiwgYnV0IGl0IHRha2VzIGEgRklMRSogaW5zdGVhZAotICogb2YgYW4gZmQuICBXZSBjb3VsZCBwYXNzIGZpbGVubyhmZCkgdG8gdGhlIGFib3ZlLCBidXQgd2UgY2FuIHJ1biBpbnRvCi0gKiB0cm91YmxlIHdoZW4gImZwIiBoYXMgYSBkaWZmZXJlbnQgbm90aW9uIG9mIHdoYXQgZmQncyBmaWxlIHBvc2l0aW9uIGlzLikKLSAqCi0gKiAiZnAiIGlzIGFuIG9wZW4gZmlsZSBwb3NpdGlvbmVkIGF0IHRoZSBzdGFydCBvZiB0aGUgImRlZmxhdGUiIGRhdGEKLSAqICJidWYiIG11c3QgaG9sZCBhdCBsZWFzdCAidW5jb21wcmVzc2VkTGVuIiBieXRlcy4KLSAqLwotLypzdGF0aWMqLyBib29sIFppcFV0aWxzOjppbmZsYXRlVG9CdWZmZXIoRklMRSogZnAsIHZvaWQqIGJ1ZiwKLSAgICBsb25nIHVuY29tcHJlc3NlZExlbiwgbG9uZyBjb21wcmVzc2VkTGVuKQotewotICAgIGJvb2wgcmVzdWx0ID0gZmFsc2U7Ci0JY29uc3QgdW5zaWduZWQgbG9uZyBrUmVhZEJ1ZlNpemUgPSAzMjc2ODsKLQl1bnNpZ25lZCBjaGFyKiByZWFkQnVmID0gTlVMTDsKLSAgICB6X3N0cmVhbSB6c3RyZWFtOwotICAgIGludCB6ZXJyOwotICAgIHVuc2lnbmVkIGxvbmcgY29tcFJlbWFpbmluZzsKLQotICAgIGFzc2VydCh1bmNvbXByZXNzZWRMZW4gPj0gMCk7Ci0gICAgYXNzZXJ0KGNvbXByZXNzZWRMZW4gPj0gMCk7Ci0KLQlyZWFkQnVmID0gbmV3IHVuc2lnbmVkIGNoYXJba1JlYWRCdWZTaXplXTsKLQlpZiAocmVhZEJ1ZiA9PSBOVUxMKQotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgY29tcFJlbWFpbmluZyA9IGNvbXByZXNzZWRMZW47Ci0KLSAgICAvKgotICAgICAqIEluaXRpYWxpemUgdGhlIHpsaWIgc3RyZWFtLgotICAgICAqLwotCW1lbXNldCgmenN0cmVhbSwgMCwgc2l6ZW9mKHpzdHJlYW0pKTsKLSAgICB6c3RyZWFtLnphbGxvYyA9IFpfTlVMTDsKLSAgICB6c3RyZWFtLnpmcmVlID0gWl9OVUxMOwotICAgIHpzdHJlYW0ub3BhcXVlID0gWl9OVUxMOwotICAgIHpzdHJlYW0ubmV4dF9pbiA9IE5VTEw7Ci0gICAgenN0cmVhbS5hdmFpbF9pbiA9IDA7Ci0gICAgenN0cmVhbS5uZXh0X291dCA9IChCeXRlZiopIGJ1ZjsKLSAgICB6c3RyZWFtLmF2YWlsX291dCA9IHVuY29tcHJlc3NlZExlbjsKLSAgICB6c3RyZWFtLmRhdGFfdHlwZSA9IFpfVU5LTk9XTjsKLQotCS8qCi0JICogVXNlIHRoZSB1bmRvY3VtZW50ZWQgIm5lZ2F0aXZlIHdpbmRvdyBiaXRzIiBmZWF0dXJlIHRvIHRlbGwgemxpYgotCSAqIHRoYXQgdGhlcmUncyBubyB6bGliIGhlYWRlciB3YWl0aW5nIGZvciBpdC4KLQkgKi8KLSAgICB6ZXJyID0gaW5mbGF0ZUluaXQyKCZ6c3RyZWFtLCAtTUFYX1dCSVRTKTsKLSAgICBpZiAoemVyciAhPSBaX09LKSB7Ci0gICAgICAgIGlmICh6ZXJyID09IFpfVkVSU0lPTl9FUlJPUikgewotICAgICAgICAgICAgTE9HRSgiSW5zdGFsbGVkIHpsaWIgaXMgbm90IGNvbXBhdGlibGUgd2l0aCBsaW5rZWQgdmVyc2lvbiAoJXMpXG4iLAotICAgICAgICAgICAgICAgIFpMSUJfVkVSU0lPTik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBMT0dFKCJDYWxsIHRvIGluZmxhdGVJbml0MiBmYWlsZWQgKHplcnI9JWQpXG4iLCB6ZXJyKTsKLSAgICAgICAgfQotICAgICAgICBnb3RvIGJhaWw7Ci0gICAgfQotCi0gICAgLyoKLSAgICAgKiBMb29wIHdoaWxlIHdlIGhhdmUgZGF0YS4KLSAgICAgKi8KLSAgICBkbyB7Ci0gICAgICAgIHVuc2lnbmVkIGxvbmcgZ2V0U2l6ZTsKLQotICAgICAgICAvKiByZWFkIGFzIG11Y2ggYXMgd2UgY2FuICovCi0gICAgICAgIGlmICh6c3RyZWFtLmF2YWlsX2luID09IDApIHsKLSAgICAgICAgICAgIGdldFNpemUgPSAoY29tcFJlbWFpbmluZyA+IGtSZWFkQnVmU2l6ZSkgPwotICAgICAgICAgICAgICAgICAgICAgICAga1JlYWRCdWZTaXplIDogY29tcFJlbWFpbmluZzsKLSAgICAgICAgICAgIExPR1YoIisrKyByZWFkaW5nICVsZCBieXRlcyAoJWxkIGxlZnQpXG4iLAotICAgICAgICAgICAgICAgIGdldFNpemUsIGNvbXBSZW1haW5pbmcpOwotCi0gICAgICAgICAgICBpbnQgY2MgPSBmcmVhZChyZWFkQnVmLCBnZXRTaXplLCAxLCBmcCk7Ci0gICAgICAgICAgICBpZiAoY2MgIT0gKGludCkgZ2V0U2l6ZSkgewotICAgICAgICAgICAgICAgIExPR0QoImluZmxhdGUgcmVhZCBmYWlsZWQgKCVkIHZzICVsZClcbiIsCi0gICAgICAgICAgICAgICAgICAgIGNjLCBnZXRTaXplKTsKLSAgICAgICAgICAgICAgICBnb3RvIHpfYmFpbDsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgY29tcFJlbWFpbmluZyAtPSBnZXRTaXplOwotCi0gICAgICAgICAgICB6c3RyZWFtLm5leHRfaW4gPSByZWFkQnVmOwotICAgICAgICAgICAgenN0cmVhbS5hdmFpbF9pbiA9IGdldFNpemU7Ci0gICAgICAgIH0KLQotICAgICAgICAvKiB1bmNvbXByZXNzIHRoZSBkYXRhICovCi0gICAgICAgIHplcnIgPSBpbmZsYXRlKCZ6c3RyZWFtLCBaX05PX0ZMVVNIKTsKLSAgICAgICAgaWYgKHplcnIgIT0gWl9PSyAmJiB6ZXJyICE9IFpfU1RSRUFNX0VORCkgewotICAgICAgICAgICAgTE9HRCgiemxpYiBpbmZsYXRlIGNhbGwgZmFpbGVkICh6ZXJyPSVkKVxuIiwgemVycik7Ci0gICAgICAgICAgICBnb3RvIHpfYmFpbDsKLSAgICAgICAgfQotCi0JCS8qIG91dHB1dCBidWZmZXIgaG9sZHMgYWxsLCBzbyBubyBuZWVkIHRvIHdyaXRlIHRoZSBvdXRwdXQgKi8KLSAgICB9IHdoaWxlICh6ZXJyID09IFpfT0spOwotCi0gICAgYXNzZXJ0KHplcnIgPT0gWl9TVFJFQU1fRU5EKTsgICAgICAgLyogb3RoZXIgZXJyb3JzIHNob3VsZCd2ZSBiZWVuIGNhdWdodCAqLwotCi0gICAgaWYgKChsb25nKSB6c3RyZWFtLnRvdGFsX291dCAhPSB1bmNvbXByZXNzZWRMZW4pIHsKLSAgICAgICAgTE9HVygiU2l6ZSBtaXNtYXRjaCBvbiBpbmZsYXRlZCBmaWxlICglbGQgdnMgJWxkKVxuIiwKLSAgICAgICAgICAgIHpzdHJlYW0udG90YWxfb3V0LCB1bmNvbXByZXNzZWRMZW4pOwotICAgICAgICBnb3RvIHpfYmFpbDsKLSAgICB9Ci0KLSAgICAvLyBzdWNjZXNzIQotICAgIHJlc3VsdCA9IHRydWU7Ci0KLXpfYmFpbDoKLSAgICBpbmZsYXRlRW5kKCZ6c3RyZWFtKTsgICAgICAgIC8qIGZyZWUgdXAgYW55IGFsbG9jYXRlZCBzdHJ1Y3R1cmVzICovCi0KLWJhaWw6Ci0JZGVsZXRlW10gcmVhZEJ1ZjsKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi0vKgotICogTG9vayBhdCB0aGUgY29udGVudHMgb2YgYSBnemlwIGFyY2hpdmUuICBXZSB3YW50IHRvIGtub3cgd2hlcmUgdGhlCi0gKiBkYXRhIHN0YXJ0cywgYW5kIGhvdyBsb25nIGl0IHdpbGwgYmUgYWZ0ZXIgaXQgaXMgdW5jb21wcmVzc2VkLgotICoKLSAqIFdlIGV4cGVjdCB0byBmaW5kIHRoZSBDUkMgYW5kIGxlbmd0aCBhcyB0aGUgbGFzdCA4IGJ5dGVzIG9uIHRoZSBmaWxlLgotICogVGhpcyBpcyBhIHByZXR0eSByZWFzb25hYmxlIHRoaW5nIHRvIGV4cGVjdCBmb3IgbG9jYWxseS1jb21wcmVzc2VkCi0gKiBmaWxlcywgYnV0IHRoZXJlJ3MgYSBzbWFsbCBjaGFuY2UgdGhhdCBzb21lIGV4dHJhIHBhZGRpbmcgZ290IHRocm93bgotICogb24gKHRoZSBtYW4gcGFnZSB0YWxrcyBhYm91dCBjb21wcmVzc2VkIGRhdGEgd3JpdHRlbiB0byB0YXBlKS4gIFdlCi0gKiBkb24ndCBjdXJyZW50bHkgZGVhbCB3aXRoIHRoYXQgaGVyZS4gIElmICJnemlwIC1sIiB3aGluZXMsIHdlJ3JlIGdvaW5nCi0gKiB0byBmYWlsIHRvby4KLSAqCi0gKiBPbiBleGl0LCAiZnAiIGlzIHBvaW50aW5nIGF0IHRoZSBzdGFydCBvZiB0aGUgY29tcHJlc3NlZCBkYXRhLgotICovCi0vKnN0YXRpYyovIGJvb2wgWmlwVXRpbHM6OmV4YW1pbmVHemlwKEZJTEUqIGZwLCBpbnQqIHBDb21wcmVzc2lvbk1ldGhvZCwKLSAgICBsb25nKiBwVW5jb21wcmVzc2VkTGVuLCBsb25nKiBwQ29tcHJlc3NlZExlbiwgdW5zaWduZWQgbG9uZyogcENSQzMyKQotewotICAgIGVudW0geyAgLy8gZmxhZ3MKLSAgICAgICAgRlRFWFQgICAgICAgPSAweDAxLAotICAgICAgICBGSENSQyAgICAgICA9IDB4MDIsCi0gICAgICAgIEZFWFRSQSAgICAgID0gMHgwNCwKLSAgICAgICAgRk5BTUUgICAgICAgPSAweDA4LAotICAgICAgICBGQ09NTUVOVCAgICA9IDB4MTAsCi0gICAgfTsKLSAgICBpbnQgaWM7Ci0gICAgaW50IG1ldGhvZCwgZmxhZ3M7Ci0gICAgaW50IGk7Ci0KLSAgICBpYyA9IGdldGMoZnApOwotICAgIGlmIChpYyAhPSAweDFmIHx8IGdldGMoZnApICE9IDB4OGIpCi0gICAgICAgIHJldHVybiBmYWxzZTsgICAgICAgLy8gbm90IGd6aXAKLSAgICBtZXRob2QgPSBnZXRjKGZwKTsKLSAgICBmbGFncyA9IGdldGMoZnApOwotCi0gICAgLyogcXVpY2sgc2FuaXR5IGNoZWNrcyAqLwotICAgIGlmIChtZXRob2QgPT0gRU9GIHx8IGZsYWdzID09IEVPRikKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotICAgIGlmIChtZXRob2QgIT0gWmlwRmlsZVJPOjprQ29tcHJlc3NEZWZsYXRlZCkKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotCi0gICAgLyogc2tpcCBvdmVyIDQgYnl0ZXMgb2YgbW9kIHRpbWUsIDEgYnl0ZSBYRkwsIDEgYnl0ZSBPUyAqLwotICAgIGZvciAoaSA9IDA7IGkgPCA2OyBpKyspCi0gICAgICAgICh2b2lkKSBnZXRjKGZwKTsKLSAgICAvKiBjb25zdW1lICJleHRyYSIgZmllbGQsIGlmIHByZXNlbnQgKi8KLSAgICBpZiAoKGZsYWdzICYgRkVYVFJBKSAhPSAwKSB7Ci0gICAgICAgIGludCBsZW47Ci0KLSAgICAgICAgbGVuID0gZ2V0YyhmcCk7Ci0gICAgICAgIGxlbiB8PSBnZXRjKGZwKSA8PCA4OwotICAgICAgICB3aGlsZSAobGVuLS0gJiYgZ2V0YyhmcCkgIT0gRU9GKQotICAgICAgICAgICAgOwotICAgIH0KLSAgICAvKiBjb25zdW1lIGZpbGVuYW1lLCBpZiBwcmVzZW50ICovCi0gICAgaWYgKChmbGFncyAmIEZOQU1FKSAhPSAwKSB7Ci0gICAgICAgIGRvIHsKLSAgICAgICAgICAgIGljID0gZ2V0YyhmcCk7Ci0gICAgICAgIH0gd2hpbGUgKGljICE9IDAgJiYgaWMgIT0gRU9GKTsKLSAgICB9Ci0gICAgLyogY29uc3VtZSBjb21tZW50LCBpZiBwcmVzZW50ICovCi0gICAgaWYgKChmbGFncyAmIEZDT01NRU5UKSAhPSAwKSB7Ci0gICAgICAgIGRvIHsKLSAgICAgICAgICAgIGljID0gZ2V0YyhmcCk7Ci0gICAgICAgIH0gd2hpbGUgKGljICE9IDAgJiYgaWMgIT0gRU9GKTsKLSAgICB9Ci0gICAgLyogY29uc3VtZSAxNi1iaXQgaGVhZGVyIENSQywgaWYgcHJlc2VudCAqLwotICAgIGlmICgoZmxhZ3MgJiBGSENSQykgIT0gMCkgewotICAgICAgICAodm9pZCkgZ2V0YyhmcCk7Ci0gICAgICAgICh2b2lkKSBnZXRjKGZwKTsKLSAgICB9Ci0KLSAgICBpZiAoZmVvZihmcCkgfHwgZmVycm9yKGZwKSkKLSAgICAgICAgcmV0dXJuIGZhbHNlOwotCi0gICAgLyogc2VlayB0byB0aGUgZW5kOyBDUkMgYW5kIGxlbmd0aCBhcmUgaW4gdGhlIGxhc3QgOCBieXRlcyAqLwotICAgIGxvbmcgY3VyUG9zbiA9IGZ0ZWxsKGZwKTsKLSAgICB1bnNpZ25lZCBjaGFyIGJ1Zls4XTsKLSAgICBmc2VlayhmcCwgLTgsIFNFRUtfRU5EKTsKLSAgICAqcENvbXByZXNzZWRMZW4gPSBmdGVsbChmcCkgLSBjdXJQb3NuOwotCi0gICAgaWYgKGZyZWFkKGJ1ZiwgMSwgOCwgZnApICE9IDgpCi0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICAvKiBzZWVrIGJhY2sgdG8gc3RhcnQgb2YgY29tcHJlc3NlZCBkYXRhICovCi0gICAgZnNlZWsoZnAsIGN1clBvc24sIFNFRUtfU0VUKTsKLQotICAgICpwQ29tcHJlc3Npb25NZXRob2QgPSBtZXRob2Q7Ci0gICAgKnBDUkMzMiA9IFppcEZpbGVSTzo6Z2V0NExFKCZidWZbMF0pOwotICAgICpwVW5jb21wcmVzc2VkTGVuID0gWmlwRmlsZVJPOjpnZXQ0TEUoJmJ1Zls0XSk7Ci0KLSAgICByZXR1cm4gdHJ1ZTsKLX0KLQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9jaGFyYWN0ZXJEYXRhLmggYi9saWJzL3V0aWxzL2NoYXJhY3RlckRhdGEuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZTkzMWQ5OS4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL2NoYXJhY3RlckRhdGEuaAorKysgL2Rldi9udWxsCkBAIC0xLDczMCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8gQXV0b21hdGljYWxseSBnZW5lcmF0ZWQgb24gMDctMTEtMjAwNiBieSBtYWtlLUNoYXJhY3RlckRhdGFDCi0vLyBETyBOT1QgRURJVCBESVJFQ1RMWQotbmFtZXNwYWNlIENoYXJhY3RlckRhdGEgewotCi0gICAgLy8gU3RydWN0dXJlIGNvbnRhaW5pbmcgYW4gYXJyYXkgb2YgcmFuZ2VzCi0gICAgc3RydWN0IFJhbmdlIHsKLSAgICAgICAgaW50IGxlbmd0aDsKLSAgICAgICAgY29uc3QgdWludDMyX3QqIGFycmF5OwotICAgIH07Ci0KLSAgICAvLyBGb3IgTGF0aW4xIGNoYXJhY3RlcnMganVzdCBpbmRleCBpbnRvIHRoaXMgYXJyYXkgdG8gZ2V0IHRoZSBpbmRleCBhbmQgZGVjb21wb3NpdGlvbgotICAgIHN0YXRpYyBjb25zdCB1aW50MTZfdCBMQVRJTjFfREFUQVtdID0gewotICAgICAgICAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgCi0gICAgICAgIDB4MDAwMSwgMHgwMDAyLCAweDAwMDMsIDB4MDAwMiwgMHgwMDA0LCAweDAwMDMsIDB4MDAwMSwgMHgwMDAxLCAKLSAgICAgICAgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIAotICAgICAgICAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMywgMHgwMDAzLCAweDAwMDMsIDB4MDAwMiwgCi0gICAgICAgIDB4MDAwNSwgMHgwMDA2LCAweDAwMDYsIDB4MDAwNywgMHgwMDA4LCAweDAwMDcsIDB4MDAwNiwgMHgwMDA2LCAKLSAgICAgICAgMHgwMDA5LCAweDAwMEEsIDB4MDAwNiwgMHgwMDBCLCAweDAwMEMsIDB4MDAwRCwgMHgwMDBDLCAweDAwMEMsIAotICAgICAgICAweDAwMEUsIDB4MDAwRiwgMHgwMDEwLCAweDAwMTEsIDB4MDAxMiwgMHgwMDEzLCAweDAwMTQsIDB4MDAxNSwgCi0gICAgICAgIDB4MDAxNiwgMHgwMDE3LCAweDAwMEMsIDB4MDAwNiwgMHgwMDE4LCAweDAwMTksIDB4MDAxQSwgMHgwMDA2LCAKLSAgICAgICAgMHgwMDA2LCAweDAwMUIsIDB4MDAxQywgMHgwMDFELCAweDAwMUUsIDB4MDAxRiwgMHgwMDIwLCAweDAwMjEsIAotICAgICAgICAweDAwMjIsIDB4MDAyMywgMHgwMDI0LCAweDAwMjUsIDB4MDAyNiwgMHgwMDI3LCAweDAwMjgsIDB4MDAyOSwgCi0gICAgICAgIDB4MDAyQSwgMHgwMDJCLCAweDAwMkMsIDB4MDAyRCwgMHgwMDJFLCAweDAwMkYsIDB4MDAzMCwgMHgwMDMxLCAKLSAgICAgICAgMHgwMDMyLCAweDAwMzMsIDB4MDAzNCwgMHgwMDM1LCAweDAwMDYsIDB4MDAzNiwgMHgwMDM3LCAweDAwMzgsIAotICAgICAgICAweDAwMzcsIDB4MDAzOSwgMHgwMDNBLCAweDAwM0IsIDB4MDAzQywgMHgwMDNELCAweDAwM0UsIDB4MDAzRiwgCi0gICAgICAgIDB4MDA0MCwgMHgwMDQxLCAweDAwNDIsIDB4MDA0MywgMHgwMDQ0LCAweDAwNDUsIDB4MDA0NiwgMHgwMDQ3LCAKLSAgICAgICAgMHgwMDQ4LCAweDAwNDksIDB4MDA0QSwgMHgwMDRCLCAweDAwNEMsIDB4MDA0RCwgMHgwMDRFLCAweDAwNEYsIAotICAgICAgICAweDAwNTAsIDB4MDA1MSwgMHgwMDUyLCAweDAwMzUsIDB4MDAxOSwgMHgwMDM2LCAweDAwMTksIDB4MDAwMSwgCi0gICAgICAgIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDMsIDB4MDAwMSwgMHgwMDAxLCAKLSAgICAgICAgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIAotICAgICAgICAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgCi0gICAgICAgIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAweDAwMDEsIDB4MDAwMSwgMHgwMDAxLCAKLSAgICAgICAgMHg1ODUzLCAweDAwMDYsIDB4MDAwOCwgMHgwMDA4LCAweDAwMDgsIDB4MDAwOCwgMHgwMDU0LCAweDAwNTQsIAotICAgICAgICAweDEwMzcsIDB4MDA1NCwgMHg3ODU1LCAweDAwNTYsIDB4MDAxOSwgMHgwMDU3LCAweDAwNTQsIDB4MTAzNywgCi0gICAgICAgIDB4MDA1OCwgMHgwMDU5LCAweDc4NUEsIDB4Nzg1QiwgMHgxMDM3LCAweDEwNUMsIDB4MDA1NCwgMHgwMDA2LCAKLSAgICAgICAgMHgxMDM3LCAweDc4NUQsIDB4Nzg1NSwgMHgwMDVFLCAweDMwNUYsIDB4MzA1RiwgMHgzMDVGLCAweDAwMDYsIAotICAgICAgICAweDA4NjAsIDB4MDg2MCwgMHgwODYwLCAweDA4NjAsIDB4MDg2MCwgMHgwODYwLCAweDAwNjAsIDB4MDg2MCwgCi0gICAgICAgIDB4MDg2MCwgMHgwODYwLCAweDA4NjAsIDB4MDg2MCwgMHgwODYwLCAweDA4NjAsIDB4MDg2MCwgMHgwODYwLCAKLSAgICAgICAgMHgwMDYwLCAweDA4NjAsIDB4MDg2MCwgMHgwODYwLCAweDA4NjAsIDB4MDg2MCwgMHgwODYwLCAweDAwMTksIAotICAgICAgICAweDAwNjAsIDB4MDg2MCwgMHgwODYwLCAweDA4NjAsIDB4MDg2MCwgMHgwODYwLCAweDAwNjAsIDB4MDA1NSwgCi0gICAgICAgIDB4MDg2MSwgMHgwODYxLCAweDA4NjEsIDB4MDg2MSwgMHgwODYxLCAweDA4NjEsIDB4MDA2MSwgMHgwODYxLCAKLSAgICAgICAgMHgwODYxLCAweDA4NjEsIDB4MDg2MSwgMHgwODYxLCAweDA4NjEsIDB4MDg2MSwgMHgwODYxLCAweDA4NjEsIAotICAgICAgICAweDAwNjEsIDB4MDg2MSwgMHgwODYxLCAweDA4NjEsIDB4MDg2MSwgMHgwODYxLCAweDA4NjEsIDB4MDAxOSwgCi0gICAgICAgIDB4MDA2MSwgMHgwODYxLCAweDA4NjEsIDB4MDg2MSwgMHgwODYxLCAweDA4NjEsIDB4MDA2MSwgMHgwODYyCi0gICAgfTsKLQotICAgIC8vIEVhY2ggb2YgdGhlc2UgYXJyYXlzIGlzIHN0cmlwcGVkIGludG8gcmFuZ2VzLiBJbiBvcmRlciB0byBidWlsZCB0aGUgYXJyYXlzLCBlYWNoCi0gICAgLy8gY29kZXBvaW50IHdhcyBiaXQtc2hpZnRlZCBzbyB0aGF0IGV2ZW4gYW5kIG9kZCBjaGFyYWN0ZXJzIHdlcmUgc2VwYXJhdGVkIGludG8gZGlmZmVyZW50Ci0gICAgLy8gYXJyYXlzLiBUaGUgaWRlbnRpZmllciBvZiBlYWNoIGFycmF5IGlzIHRoZSB0b3AgYnl0ZSBhZnRlciBiaXQtc2hpZnRpbmcuCi0gICAgLy8gVGhlIG51bWJlcnMgc3RvcmVkIGluIHRoZSBhcnJheSBhcmUgdGhlIGJpdC1zaGlmdGVkIGNvZGVwb2ludCwgdGhlIGRlY29tcG9zaXRpb24sIGFuZCBhbgotICAgIC8vIGluZGV4IGludG8gYW5vdGhlciBhcnJheSBvZiBhbGwgcG9zc2libGUgcGFja2VkIGRhdGEgdmFsdWVzLiBUaGUgdG9wIDE2IGJpdHMgYXJlIHRoZQotICAgIC8vIGNvZGVwb2ludCBhbmQgdGhlIGJvdHRvbSAxNiBhcmUgdGhlIGRlY29tcG9zaXRpb24gYW5kIGluZGV4LiBUaGUgdG9wIDUgYml0cyBmb3IgdGhlIGRlY29tcG9zaXRpb24KLSAgICAvLyBhbmQgdGhlIHJlc3QgZm9yIHRoZSBpbmRleC4KLSAgICBzdGF0aWMgY29uc3QgdWludDMyX3QgYTBbXSA9IHsKLSAgICAgICAgMHgwMDgwMDg2MywgMHgwMDg4MDA2MywgMHgwMDg5MDg2MywgMHgwMDkzMDA2MywgMHgwMDk0MDg2MywgMHgwMDk4MDg2NCwgMHgwMDk5MTA2MywgMHgwMDlBMDg2MywgCi0gICAgICAgIDB4MDA5QzAwNTUsIDB4MDA5RDA4NjUsIDB4MDBBMDEwNjUsIDB4MDBBMTAwNjUsIDB4MDBBMjA4NjUsIDB4MDBBNTAwNjMsIDB4MDBBNjA4NjMsIDB4MDBBOTAwNjMsIAotICAgICAgICAweDAwQUEwODYzLCAweDAwQjMwMDYzLCAweDAwQjQwODYzLCAweDAwQkMwODY2LCAweDAwQkQwODY1LCAweDAwQzAwMDU1LCAweDAwQzEwMDYzLCAweDAwQzMwMDY3LCAKLSAgICAgICAgMHgwMEM0MDA2NSwgMHgwMEM1MDA2OCwgMHgwMEM2MDA2NSwgMHgwMEM3MDA2OSwgMHgwMEM4MDA2QSwgMHgwMEM5MDA2NSwgMHgwMENBMDA2QiwgMHgwMENCMDA2QywgCi0gICAgICAgIDB4MDBDQzAwNjMsIDB4MDBDRDAwNkQsIDB4MDBDRTAwNkMsIDB4MDBDRjAwNkUsIDB4MDBEMDA4NjMsIDB4MDBEMTAwNjMsIDB4MDBEMzAwNkYsIDB4MDBENDAwNjUsIAotICAgICAgICAweDAwRDUwMDU1LCAweDAwRDYwMDYzLCAweDAwRDcwMDZGLCAweDAwRDgwODY1LCAweDAwRDkwMDcwLCAweDAwREEwMDY1LCAweDAwREMwMDYzLCAweDAwREQwMDU1LCAKLSAgICAgICAgMHgwMERFMDA2MywgMHgwMERGMDA1NSwgMHgwMEUwMDA3MSwgMHgwMEUyMTA3MiwgMHgwMEUzMTA3MywgMHgwMEU0MTA3NCwgMHgwMEU1MTA3MiwgMHgwMEU2MTA3MywgCi0gICAgICAgIDB4MDBFNzA4NjUsIDB4MDBFRjA4NjMsIDB4MDBGMjAwNjMsIDB4MDBGMzA4NjMsIDB4MDBGODA4NTUsIDB4MDBGOTEwNzQsIDB4MDBGQTA4NjMsIDB4MDBGQjAwNzUsIAotICAgICAgICAweDAwRkMwODYzLCAweDAxMEUwMDYzLCAweDAxMEYwODYzLCAweDAxMTAwMDc2LCAweDAxMTEwMDYzLCAweDAxMTMwODYzLCAweDAxMUEwMDU1LCAweDAxMUQwMDc3LCAKLSAgICAgICAgMHgwMTFFMDA2NSwgMHgwMTFGMDA3NywgMHgwMTIwMDA1NSwgMHgwMTIxMDA3OCwgMHgwMTI4MDA1NSwgMHgwMTJBMDA3OSwgMHgwMTJCMDA3QSwgMHgwMTJDMDA1NSwgCi0gICAgICAgIDB4MDEzMDAwN0EsIDB4MDEzMTAwNTUsIDB4MDEzNDAwN0IsIDB4MDEzNTAwNTUsIDB4MDEzOTAwN0MsIDB4MDEzQTAwNTUsIDB4MDE0MDAwN0QsIDB4MDE0MTAwNTUsIAotICAgICAgICAweDAxNDQwMDdELCAweDAxNDUwMDdFLCAweDAxNDYwMDU1LCAweDAxNDkwMDdGLCAweDAxNEEwMDgwLCAweDAxNEIwMDU1LCAweDAxNTg3ODgxLCAweDAxNUQwMDgyLCAKLSAgICAgICAgMHgwMTVFMDA4MSwgMHgwMTYxMDAzNywgMHgwMTYzMDA4MiwgMHgwMTY4MDA4MSwgMHgwMTY5MDAzNywgMHgwMTZDMTAzNywgMHgwMTZGMDAzNywgMHgwMTcwNzg4MSwgCi0gICAgICAgIDB4MDE3MzAwMzcsIDB4MDE3NzAwODEsIDB4MDE3ODAwMzcsIDB4MDE4MDAwODMsIDB4MDFBMDA4ODMsIDB4MDFBMTAwODMsIDB4MDFBMjA4ODMsIDB4MDFBMzAwODMsIAotICAgICAgICAweDAxQjgwMDc4LCAweDAxQkEwODM3LCAweDAxQkIwMDc4LCAweDAxQkQxMDgxLCAweDAxQkUwMDc4LCAweDAxQkYwODA2LCAweDAxQzAwMDc4LCAweDAxQzIxMDM3LCAKLSAgICAgICAgMHgwMUMzMDg4NCwgMHgwMUM0MDg4NSwgMHgwMUM2MDg4NiwgMHgwMUM3MDg4NywgMHgwMUM4MDg1NSwgMHgwMUM5MDA2MCwgMHgwMUQxMDA3OCwgMHgwMUQyMDA2MCwgCi0gICAgICAgIDB4MDFENTA4NjAsIDB4MDFENjA4ODgsIDB4MDFENzA4ODksIDB4MDFEODA4NTUsIDB4MDFEOTAwNjEsIDB4MDFFMTAwOEEsIDB4MDFFMjAwNjEsIDB4MDFFNTA4NjEsIAotICAgICAgICAweDAxRTYwODhCLCAweDAxRTcwODhDLCAweDAxRTgxMDhELCAweDAxRTkxMDc3LCAweDAxRUEwODc3LCAweDAxRUIxMDhFLCAweDAxRUMwMDYzLCAweDAxRjgxMDhGLCAKLSAgICAgICAgMHgwMUY5MTA5MCwgMHgwMUZBMTA5MSwgMHgwMUZCMDAxOSwgMHgwMUZDMDA2NSwgMHgwMUZEMDA2MywgMHgwMUZFMDA1NSwgMHgwMUZGMDA3NywgMHgwMjAwMDg5MiwgCi0gICAgICAgIDB4MDIwMTAwOTIsIDB4MDIwNjA4OTIsIDB4MDIwODAwNjAsIDB4MDIxODAwNjEsIDB4MDIyODA4OTMsIDB4MDIyOTAwOTMsIDB4MDIyRTA4OTMsIDB4MDIzMDAwNjMsIAotICAgICAgICAweDAyM0IwODYzLCAweDAyM0MwMDYzLCAweDAyNDEwMDk0LCAweDAyNDIwMDgzLCAweDAyNDQwMDk1LCAweDAyNDUwMDYzLCAweDAyNjAwMDc3LCAweDAyNjEwODY1LCAKLSAgICAgICAgMHgwMjYyMDA2NSwgMHgwMjY4MDg2MywgMHgwMjZBMDA2MywgMHgwMjZCMDg2MywgMHgwMjZDMDA2MywgMHgwMjZEMDg2MywgMHgwMjcwMDA2MywgMHgwMjcxMDg2MywgCi0gICAgICAgIDB4MDI3NDAwNjMsIDB4MDI3NTA4NjMsIDB4MDI3QjAwNjMsIDB4MDI3QzA4NjMsIDB4MDI3RDAwNzgsIDB4MDI4MDAwNjMsIDB4MDI4ODAwNzgsIDB4MDI5OTAwOTYsIAotICAgICAgICAweDAyQUMwMDc4LCAweDAyQUQwMDk3LCAweDAyQjAwMDc4LCAweDAyQjEwMDk4LCAweDAyQzQwMDc4LCAweDAyQzUwMDk5LCAweDAyQzYwMDc4LCAweDAyQzkwMDgzLCAKLSAgICAgICAgMHgwMkREMDA3OCwgMHgwMkRFMDA4MywgMHgwMkRGMDA5QSwgMHgwMkUxMDA4MywgMHgwMkUzMDA5QSwgMHgwMkU0MDA3OCwgMHgwMkU4MDA5QiwgMHgwMkY2MDA3OCwgCi0gICAgICAgIDB4MDJGODAwOUIsIDB4MDJGQTAwOUEsIDB4MDJGQjAwNzgsIDB4MDMwMDAwOUMsIDB4MDMwMjAwNzgsIDB4MDMwNjAwMEMsIDB4MDMwNzAwNTQsIDB4MDMwODAwODMsIAotICAgICAgICAweDAzMEIwMDc4LCAweDAzMEYwMDlELCAweDAzMTAwMDc4LCAweDAzMTEwODlFLCAweDAzMTQwMDlFLCAweDAzMUUwMDc4LCAweDAzMjAwMDlGLCAweDAzMjEwMDlFLCAKLSAgICAgICAgMHgwMzI2MDA4MywgMHgwMzMwMDBBMCwgMHgwMzMxMDBBMSwgMHgwMzMyMDBBMiwgMHgwMzMzMDBBMywgMHgwMzM0MDBBNCwgMHgwMzM1MDAwNywgMHgwMzM2MDBBNSwgCi0gICAgICAgIDB4MDMzNzAwOUUsIDB4MDMzODAwODMsIDB4MDMzOTAwOUUsIDB4MDMzQjEwOUUsIDB4MDMzRDAwOUUsIDB4MDM2MDA4OUUsIDB4MDM2MjAwOUUsIDB4MDM2QTAwOUQsIAotICAgICAgICAweDAzNkIwMDgzLCAweDAzNkYwMDk1LCAweDAzNzAwMDgzLCAweDAzNzMwMDlGLCAweDAzNzQwMDgzLCAweDAzNzcwMDlFLCAweDAzNzgwMDBFLCAweDAzNzkwMDEwLCAKLSAgICAgICAgMHgwMzdBMDAxMiwgMHgwMzdCMDAxNCwgMHgwMzdDMDAxNiwgMHgwMzdEMDA5RSwgMHgwMzdGMDBBNiwgMHgwMzgwMDA5RCwgMHgwMzg3MDA3OCwgMHgwMzg4MDA5RSwgCi0gICAgICAgIDB4MDM5ODAwODMsIDB4MDNBNjAwNzgsIDB4MDNBNzAwOUUsIDB4MDNCNzAwNzgsIDB4MDNDMDAwOUUsIDB4MDNEMzAwODMsIDB4MDNEOTAwNzgsIDB4MDQ4MTAwODMsIAotICAgICAgICAweDA0ODIwMDcxLCAweDA0OUEwODcxLCAweDA0OUIwMDcxLCAweDA0OUQwMDc4LCAweDA0OUUwMDgzLCAweDA0OUYwMEE3LCAweDA0QTEwMDgzLCAweDA0QTUwMEE3LCAKLSAgICAgICAgMHgwNEE3MDA3OCwgMHgwNEE4MDA3MSwgMHgwNEE5MDA4MywgMHgwNEFCMDA3OCwgMHgwNEFDMDg3MSwgMHgwNEIwMDA3MSwgMHgwNEIxMDA4MywgMHgwNEIyMDA5NywgCi0gICAgICAgIDB4MDRCMzAwQTgsIDB4MDRCNDAwQTksIDB4MDRCNTAwQUEsIDB4MDRCNjAwQUIsIDB4MDRCNzAwQUMsIDB4MDRCODAwOTcsIDB4MDRCOTAwNzgsIDB4MDRDMTAwQTcsIAotICAgICAgICAweDA0QzIwMDc4LCAweDA0QzMwMDcxLCAweDA0QzcwMDc4LCAweDA0QzgwMDcxLCAweDA0QzkwMDc4LCAweDA0Q0EwMDcxLCAweDA0REEwMDc4LCAweDA0REIwMDcxLCAKLSAgICAgICAgMHgwNEREMDA3OCwgMHgwNERFMDA4MywgMHgwNERGMDBBNywgMHgwNEUxMDA4MywgMHgwNEUzMDA3OCwgMHgwNEU0MDBBNywgMHgwNEU1MDA3OCwgMHgwNEU2MDhBNywgCi0gICAgICAgIDB4MDRFNzAwNzEsIDB4MDRFODAwNzgsIDB4MDRFRTA4NzEsIDB4MDRFRjAwNzgsIDB4MDRGMDAwNzEsIDB4MDRGMTAwODMsIDB4MDRGMjAwNzgsIDB4MDRGMzAwQTgsIAotICAgICAgICAweDA0RjQwMEE5LCAweDA0RjUwMEFBLCAweDA0RjYwMEFCLCAweDA0RjcwMEFDLCAweDA0RjgwMDcxLCAweDA0RjkwMDA4LCAweDA0RkEwMEFELCAweDA0RkIwMEFFLCAKLSAgICAgICAgMHgwNEZDMDBBRiwgMHgwNEZEMDA5NCwgMHgwNEZFMDA3OCwgMHgwNTAxMDA4MywgMHgwNTAyMDA3OCwgMHgwNTAzMDA3MSwgMHgwNTA2MDA3OCwgMHgwNTA4MDA3MSwgCi0gICAgICAgIDB4MDUwOTAwNzgsIDB4MDUwQTAwNzEsIDB4MDUxQTAwNzgsIDB4MDUxQjA4NzEsIDB4MDUxQzAwNzEsIDB4MDUxRDAwNzgsIDB4MDUxRTAwODMsIDB4MDUxRjAwQTcsIAotICAgICAgICAweDA1MjEwMDgzLCAweDA1MjIwMDc4LCAweDA1MjQwMDgzLCAweDA1MjUwMDc4LCAweDA1MjYwMDgzLCAweDA1MjcwMDc4LCAweDA1MkQwODcxLCAweDA1MkUwMDcxLCAKLSAgICAgICAgMHgwNTJGMDg3MSwgMHgwNTMwMDA3OCwgMHgwNTMzMDBBOCwgMHgwNTM0MDBBOSwgMHgwNTM1MDBBQSwgMHgwNTM2MDBBQiwgMHgwNTM3MDBBQywgMHgwNTM4MDA4MywgCi0gICAgICAgIDB4MDUzOTAwNzEsIDB4MDUzQjAwNzgsIDB4MDU0MTAwODMsIDB4MDU0MjAwNzgsIDB4MDU0MzAwNzEsIDB4MDU0NzAwNzgsIDB4MDU0ODAwNzEsIDB4MDU0OTAwNzgsIAotICAgICAgICAweDA1NEEwMDcxLCAweDA1NUEwMDc4LCAweDA1NUIwMDcxLCAweDA1NUQwMDc4LCAweDA1NUUwMDgzLCAweDA1NUYwMEE3LCAweDA1NjEwMDgzLCAweDA1NjMwMDc4LCAKLSAgICAgICAgMHgwNTY0MDA4MywgMHgwNTY1MDA3OCwgMHgwNTY2MDBBNywgMHgwNTY3MDA3OCwgMHgwNTY4MDA3MSwgMHgwNTY5MDA3OCwgMHgwNTcwMDA3MSwgMHgwNTcxMDA4MywgCi0gICAgICAgIDB4MDU3MjAwNzgsIDB4MDU3MzAwQTgsIDB4MDU3NDAwQTksIDB4MDU3NTAwQUEsIDB4MDU3NjAwQUIsIDB4MDU3NzAwQUMsIDB4MDU3ODAwNzgsIDB4MDU4MTAwQTcsIAotICAgICAgICAweDA1ODIwMDc4LCAweDA1ODMwMDcxLCAweDA1ODcwMDc4LCAweDA1ODgwMDcxLCAweDA1ODkwMDc4LCAweDA1OEEwMDcxLCAweDA1OUEwMDc4LCAweDA1OUIwMDcxLCAKLSAgICAgICAgMHgwNTlEMDA3OCwgMHgwNTlFMDA4MywgMHgwNTlGMDBBNywgMHgwNUExMDA4MywgMHgwNUEyMDA3OCwgMHgwNUE0MDhBNywgMHgwNUE1MDA3OCwgMHgwNUE2MDhBNywgCi0gICAgICAgIDB4MDVBNzAwNzgsIDB4MDVBQjAwODMsIDB4MDVBQzAwNzgsIDB4MDVBRTA4NzEsIDB4MDVBRjAwNzgsIDB4MDVCMDAwNzEsIDB4MDVCMTAwNzgsIDB4MDVCMzAwQTgsIAotICAgICAgICAweDA1QjQwMEE5LCAweDA1QjUwMEFBLCAweDA1QjYwMEFCLCAweDA1QjcwMEFDLCAweDA1QjgwMDk0LCAweDA1QjkwMDc4LCAweDA1QzEwMDgzLCAweDA1QzIwMDc4LCAKLSAgICAgICAgMHgwNUMzMDA3MSwgMHgwNUM2MDA3OCwgMHgwNUM3MDA3MSwgMHgwNUNBMDg3MSwgMHgwNUNCMDA3OCwgMHgwNUNEMDA3MSwgMHgwNUQwMDA3OCwgMHgwNUQyMDA3MSwgCi0gICAgICAgIDB4MDVEMzAwNzgsIDB4MDVENDAwNzEsIDB4MDVENjAwNzgsIDB4MDVENzAwNzEsIDB4MDVERDAwNzgsIDB4MDVERjAwQTcsIDB4MDVFMDAwODMsIDB4MDVFMTAwQTcsIAotICAgICAgICAweDA1RTIwMDc4LCAweDA1RTMwMEE3LCAweDA1RTUwOEE3LCAweDA1RTcwMDc4LCAweDA1RjMwMEE4LCAweDA1RjQwMEE5LCAweDA1RjUwMEFBLCAweDA1RjYwMEFCLCAKLSAgICAgICAgMHgwNUY3MDBBQywgMHgwNUY4MDBCMCwgMHgwNUY5MDBCMSwgMHgwNUZBMDA1NCwgMHgwNUZFMDA3OCwgMHgwNjAxMDBBNywgMHgwNjAyMDA3OCwgMHgwNjAzMDA3MSwgCi0gICAgICAgIDB4MDYxQTAwNzgsIDB4MDYxQjAwNzEsIDB4MDYxRDAwNzgsIDB4MDYxRjAwODMsIDB4MDYyMTAwQTcsIDB4MDYyMzAwODMsIDB4MDYyNDA4ODMsIDB4MDYyNTAwODMsIAotICAgICAgICAweDA2MjcwMDc4LCAweDA2MkIwMDgzLCAweDA2MkMwMDc4LCAweDA2MzAwMDcxLCAweDA2MzEwMDc4LCAweDA2MzMwMEE4LCAweDA2MzQwMEE5LCAweDA2MzUwMEFBLCAKLSAgICAgICAgMHgwNjM2MDBBQiwgMHgwNjM3MDBBQywgMHgwNjM4MDA3OCwgMHgwNjQxMDBBNywgMHgwNjQyMDA3OCwgMHgwNjQzMDA3MSwgMHgwNjVBMDA3OCwgMHgwNjVCMDA3MSwgCi0gICAgICAgIDB4MDY1RDAwNzgsIDB4MDY1RTAwODMsIDB4MDY1RjAwQTcsIDB4MDY2MDA4QTcsIDB4MDY2MTAwQTcsIDB4MDY2MzAwQjIsIDB4MDY2NDA4QTcsIDB4MDY2NjAwODMsIAotICAgICAgICAweDA2NjcwMDc4LCAweDA2NkIwMEE3LCAweDA2NkMwMDc4LCAweDA2NkYwMDcxLCAweDA2NzEwMDc4LCAweDA2NzMwMEE4LCAweDA2NzQwMEE5LCAweDA2NzUwMEFBLCAKLSAgICAgICAgMHgwNjc2MDBBQiwgMHgwNjc3MDBBQywgMHgwNjc4MDA3OCwgMHgwNjgxMDBBNywgMHgwNjgyMDA3OCwgMHgwNjgzMDA3MSwgMHgwNjlEMDA3OCwgMHgwNjlGMDBBNywgCi0gICAgICAgIDB4MDZBMTAwODMsIDB4MDZBMjAwNzgsIDB4MDZBMzAwQTcsIDB4MDZBNTA4QTcsIDB4MDZBNzAwNzgsIDB4MDZCMDAwNzEsIDB4MDZCMTAwNzgsIDB4MDZCMzAwQTgsIAotICAgICAgICAweDA2QjQwMEE5LCAweDA2QjUwMEFBLCAweDA2QjYwMEFCLCAweDA2QjcwMEFDLCAweDA2QjgwMDc4LCAweDA2QzEwMEE3LCAweDA2QzIwMDc4LCAweDA2QzMwMDcxLCAKLSAgICAgICAgMHgwNkNDMDA3OCwgMHgwNkNEMDA3MSwgMHgwNkQ5MDA3OCwgMHgwNkRBMDA3MSwgMHgwNkRFMDA3OCwgMHgwNkUwMDA3MSwgMHgwNkU0MDA3OCwgMHgwNkU1MDA4MywgCi0gICAgICAgIDB4MDZFNjAwNzgsIDB4MDZFODAwQTcsIDB4MDZFOTAwODMsIDB4MDZFQzAwQTcsIDB4MDZFRDA4QTcsIDB4MDZGMDAwNzgsIDB4MDZGOTAwQTcsIDB4MDZGQTAwOTcsIAotICAgICAgICAweDA2RkIwMDc4LCAweDA3MDEwMDcxLCAweDA3MUEwMDgzLCAweDA3MUUwMDc4LCAweDA3MjAwMDcxLCAweDA3MjMwMDgxLCAweDA3MjQwMDgzLCAweDA3MjgwMEE4LCAKLSAgICAgICAgMHgwNzI5MDBBOSwgMHgwNzJBMDBBQSwgMHgwNzJCMDBBQiwgMHgwNzJDMDBBQywgMHgwNzJEMDA5NywgMHgwNzJFMDA3OCwgMHgwNzQxMDA3MSwgMHgwNzQzMDA3OCwgCi0gICAgICAgIDB4MDc0NDAwNzEsIDB4MDc0NjAwNzgsIDB4MDc0QTAwNzEsIDB4MDc0QzAwNzgsIDB4MDc0RDAwNzEsIDB4MDc1MDAwNzgsIDB4MDc1MTAwNzEsIDB4MDc1MjAwNzgsIAotICAgICAgICAweDA3NTUwMDcxLCAweDA3NTYwMDc4LCAweDA3NTcwMDcxLCAweDA3NUEwMDgzLCAweDA3NUQwMDc4LCAweDA3NUUwMDgzLCAweDA3NUYwMDc4LCAweDA3NjAwMDcxLCAKLSAgICAgICAgMHgwNzYzMDA4MSwgMHgwNzY0MDA4MywgMHgwNzY3MDA3OCwgMHgwNzY4MDBBOCwgMHgwNzY5MDBBOSwgMHgwNzZBMDBBQSwgMHgwNzZCMDBBQiwgMHgwNzZDMDBBQywgCi0gICAgICAgIDB4MDc2RDAwNzgsIDB4MDc2RTEwNzEsIDB4MDc2RjAwNzgsIDB4MDc4MDAwNzEsIDB4MDc4MTAwOTQsIDB4MDc4MjAwOTcsIDB4MDc4NjU4OTcsIDB4MDc4NzAwOTcsIAotICAgICAgICAweDA3OEEwMDk0LCAweDA3OEMwMDgzLCAweDA3OEQwMDk0LCAweDA3OTAwMEE4LCAweDA3OTEwMEE5LCAweDA3OTIwMEFBLCAweDA3OTMwMEFCLCAweDA3OTQwMEFDLCAKLSAgICAgICAgMHgwNzk1MDBCMywgMHgwNzlBMDA5NCwgMHgwNzlEMDBCNCwgMHgwNzlGMDBBNywgMHgwN0EwMDA3MSwgMHgwN0E0MDA3OCwgMHgwN0E1MDA3MSwgMHgwN0E5MDg3MSwgCi0gICAgICAgIDB4MDdBQTAwNzEsIDB4MDdBRTA4NzEsIDB4MDdBRjAwNzEsIDB4MDdCNjAwNzgsIDB4MDdCOTAwODMsIDB4MDdCQjA4ODMsIDB4MDdCRDAwODMsIDB4MDdDNDAwNzEsIAotICAgICAgICAweDA3QzYwMDc4LCAweDA3QzgwMDgzLCAweDA3Q0MwMDc4LCAweDA3Q0QwMDgzLCAweDA3RDEwODgzLCAweDA3RDIwMDgzLCAweDA3RDYwODgzLCAweDA3RDcwMDgzLCAKLSAgICAgICAgMHgwN0RGMDA5NCwgMHgwN0UzMDA4MywgMHgwN0U0MDA5NCwgMHgwN0U3MDA3OCwgMHgwN0U4MDA5NywgMHgwN0U5MDA3OCwgMHgwODAwMDA3MSwgMHgwODExMDA3OCwgCi0gICAgICAgIDB4MDgxMjAwNzEsIDB4MDgxMzA4NzEsIDB4MDgxNDAwNzgsIDB4MDgxNTAwNzEsIDB4MDgxNjAwQTcsIDB4MDgxNzAwODMsIDB4MDgxQTAwNzgsIDB4MDgxQjAwODMsIAotICAgICAgICAweDA4MUMwMEE3LCAweDA4MUQwMDc4LCAweDA4MjAwMEE4LCAweDA4MjEwMEE5LCAweDA4MjIwMEFBLCAweDA4MjMwMEFCLCAweDA4MjQwMEFDLCAweDA4MjUwMDk3LCAKLSAgICAgICAgMHgwODI4MDA3MSwgMHgwODJCMDBBNywgMHgwODJDMDA4MywgMHgwODJEMDA3OCwgMHgwODUwMDBCNSwgMHgwODYzMDA3OCwgMHgwODY4MDA3MSwgMHgwODdFNzg4MSwgCi0gICAgICAgIDB4MDg3RjAwNzgsIDB4MDg4MDAwNzEsIDB4MDhBRDAwNzgsIDB4MDhCMDAwNzEsIDB4MDhEMjAwNzgsIDB4MDhENDAwNzEsIDB4MDhGRDAwNzgsIDB4MDkwMDAwNzEsIAotICAgICAgICAweDA5MjcwMDc4LCAweDA5MjgwMDcxLCAweDA5MkYwMDc4LCAweDA5MzAwMDcxLCAweDA5NDcwMDc4LCAweDA5NDgwMDcxLCAweDA5NUIwMDc4LCAweDA5NUMwMDcxLCAKLSAgICAgICAgMHgwOTYzMDA3OCwgMHgwOTY0MDA3MSwgMHgwOThCMDA3OCwgMHgwOThDMDA3MSwgMHgwOUFFMDA3OCwgMHgwOUIwMDA5NCwgMHgwOUIxMDA5NywgMHgwOUI1MDBCNiwgCi0gICAgICAgIDB4MDlCNjAwQjcsIDB4MDlCNzAwQjgsIDB4MDlCODAwQjksIDB4MDlCOTAwQjAsIDB4MDlCQTAwQkEsIDB4MDlCQjAwQkIsIDB4MDlCQzAwQkMsIDB4MDlCRDAwQkQsIAotICAgICAgICAweDA5QkUwMEJFLCAweDA5QkYwMDc4LCAweDA5QzAwMDcxLCAweDA5QzgwMDU0LCAweDA5Q0QwMDc4LCAweDA5RDAwMDcxLCAweDA5RkIwMDc4LCAweDBBMDEwMDcxLCAKLSAgICAgICAgMHgwQjM3MDA5NywgMHgwQjM4MDA3MSwgMHgwQjNDMDA3OCwgMHgwQjQwMDAwNSwgMHgwQjQxMDA3MSwgMHgwQjRFMDBCRiwgMHgwQjRGMDA3OCwgMHgwQjUwMDA3MSwgCi0gICAgICAgIDB4MEI3NjAwOTcsIDB4MEI3NzAwQzAsIDB4MEI3ODAwQzEsIDB4MEI3OTAwNzgsIDB4MEI4MDAwNzEsIDB4MEI4OTAwODMsIDB4MEI4QjAwNzgsIDB4MEI5MDAwNzEsIAotICAgICAgICAweDBCOTkwMDgzLCAweDBCOUIwMDk3LCAweDBCOUMwMDc4LCAweDBCQTAwMDcxLCAweDBCQTkwMDgzLCAweDBCQUEwMDc4LCAweDBCQjAwMDcxLCAweDBCQjkwMDgzLCAKLSAgICAgICAgMHgwQkJBMDA3OCwgMHgwQkMwMDA3MSwgMHgwQkRBMDBDMiwgMHgwQkRCMDBBNywgMHgwQkRDMDA4MywgMHgwQkRGMDBBNywgMHgwQkUzMDA4MywgMHgwQkU0MDBBNywgCi0gICAgICAgIDB4MEJFNTAwODMsIDB4MEJFQTAwOTcsIDB4MEJFRTAwNzEsIDB4MEJFRjAwNzgsIDB4MEJGMDAwQTgsIDB4MEJGMTAwQTksIDB4MEJGMjAwQUEsIDB4MEJGMzAwQUIsIAotICAgICAgICAweDBCRjQwMEFDLCAweDBCRjUwMDc4LCAweDBCRjgwMEMzLCAweDBCRjkwMEM0LCAweDBCRkEwMEM1LCAweDBCRkIwMEM2LCAweDBCRkMwMEM3LCAweDBCRkQwMDc4LCAKLSAgICAgICAgMHgwQzAwMDAwNiwgMHgwQzAzMDA5OSwgMHgwQzA0MDAwNiwgMHgwQzA2MDA4MywgMHgwQzA3MDAwNSwgMHgwQzA4MDBBOCwgMHgwQzA5MDBBOSwgMHgwQzBBMDBBQSwgCi0gICAgICAgIDB4MEMwQjAwQUIsIDB4MEMwQzAwQUMsIDB4MEMwRDAwNzgsIDB4MEMxMDAwNzEsIDB4MEMzQzAwNzgsIDB4MEM0MDAwNzEsIDB4MEM1NTAwNzgsIDB4MEM4MDAwNzEsIAotICAgICAgICAweDBDOEYwMDc4LCAweDBDOTAwMDgzLCAweDBDOTIwMEE3LCAweDBDOTQwMDgzLCAweDBDOTUwMEM4LCAweDBDOTYwMDc4LCAweDBDOTgwMEE3LCAweDBDOTkwMDgzLCAKLSAgICAgICAgMHgwQzlBMDBBNywgMHgwQzlEMDA4MywgMHgwQzlFMDA3OCwgMHgwQ0EwMDA1NCwgMHgwQ0ExMDA3OCwgMHgwQ0EyMDAwNiwgMHgwQ0EzMDBBOCwgMHgwQ0E0MDBBOSwgCi0gICAgICAgIDB4MENBNTAwQUEsIDB4MENBNjAwQUIsIDB4MENBNzAwQUMsIDB4MENBODAwNzEsIDB4MENCNzAwNzgsIDB4MENCODAwNzEsIDB4MENCQjAwNzgsIDB4MENDMDAwNzEsIAotICAgICAgICAweDBDRDUwMDc4LCAweDBDRDgwMEE3LCAweDBDRTEwMDcxLCAweDBDRTQwMEE3LCAweDBDRTUwMDc4LCAweDBDRTgwMEE4LCAweDBDRTkwMEE5LCAweDBDRUEwMEFBLCAKLSAgICAgICAgMHgwQ0VCMDBBQiwgMHgwQ0VDMDBBQywgMHgwQ0VEMDA3OCwgMHgwQ0VGMDAwNiwgMHgwQ0YwMDA1NCwgMHgwRDAwMDA3MSwgMHgwRDBDMDA4MywgMHgwRDBEMDBBNywgCi0gICAgICAgIDB4MEQwRTAwNzgsIDB4MEQwRjAwOTcsIDB4MEQxMDAwNzgsIDB4MEU4MDAwNTUsIDB4MEU5Njc4ODEsIDB4MEVBNzAwODEsIDB4MEVBODc4ODEsIDB4MEVCMTcwNTUsIAotICAgICAgICAweDBFQjYwMDU1LCAweDBFQkM3ODgxLCAweDBFQkQwMDU1LCAweDBFQ0U3ODgxLCAweDBFRTAwMDgzLCAweDBFRTIwMDc4LCAweDBGMDAwODYzLCAweDBGNEIwODU1LCAKLSAgICAgICAgMHgwRjREMTA1NSwgMHgwRjRFMDA3OCwgMHgwRjUwMDg2MywgMHgwRjdEMDA3OCwgMHgwRjgwMDhDOSwgMHgwRjg0MDhDQSwgMHgwRjg4MDhDOSwgMHgwRjhCMDA3OCwgCi0gICAgICAgIDB4MEY4QzA4Q0EsIDB4MEY4RjAwNzgsIDB4MEY5MDA4QzksIDB4MEY5NDA4Q0EsIDB4MEY5ODA4QzksIDB4MEY5QzA4Q0EsIDB4MEZBMDA4QzksIDB4MEZBMzAwNzgsIAotICAgICAgICAweDBGQTQwOENBLCAweDBGQTcwMDc4LCAweDBGQTgwODU1LCAweDBGQUMwMDc4LCAweDBGQjAwOEM5LCAweDBGQjQwOENBLCAweDBGQjgwOENCLCAweDBGQjkwOENDLCAKLSAgICAgICAgMHgwRkJCMDhDRCwgMHgwRkJDMDhDRSwgMHgwRkJEMDhDRiwgMHgwRkJFMDhEMCwgMHgwRkJGMDA3OCwgMHgwRkMwMDhDOSwgMHgwRkM0MDhEMSwgMHgwRkM4MDhDOSwgCi0gICAgICAgIDB4MEZDQzA4RDEsIDB4MEZEMDA4QzksIDB4MEZENDA4RDEsIDB4MEZEODA4QzksIDB4MEZEOTA4NTUsIDB4MEZEQzA4Q0EsIDB4MEZERDA4RDIsIDB4MEZERTA4RDMsIAotICAgICAgICAweDBGREYwOEQ0LCAweDBGRTAxMDM3LCAweDBGRTEwODU1LCAweDBGRTQwOEQ1LCAweDBGRTYwOEQzLCAweDBGRTcwODM3LCAweDBGRTgwOEM5LCAweDBGRTkwODU1LCAKLSAgICAgICAgMHgwRkVBMDA3OCwgMHgwRkVCMDg1NSwgMHgwRkVDMDhDQSwgMHgwRkVEMDhENiwgMHgwRkVFMDA3OCwgMHgwRkVGMDgzNywgMHgwRkYwMDhDOSwgMHgwRkYxMDg1NSwgCi0gICAgICAgIDB4MEZGNDA4Q0EsIDB4MEZGNTA4RDcsIDB4MEZGNjA4RDgsIDB4MEZGNzA4MzcsIDB4MEZGODAwNzgsIDB4MEZGOTA4NTUsIDB4MEZGQzA4RDksIDB4MEZGRDA4REEsIAotICAgICAgICAweDBGRkUwOEQzLCAweDBGRkYxMDM3LCAweDEwMDAwODA1LCAweDEwMDExMDA1LCAweDEwMDYwMDU3LCAweDEwMDcwMEMyLCAweDEwMDgwMDk5LCAweDEwMEIwMDA2LCAKLSAgICAgICAgMHgxMDBDMDBEQiwgMHgxMDBEMDBCNCwgMHgxMDBFMDBEQiwgMHgxMDBGMDBCNCwgMHgxMDEwMDAwNiwgMHgxMDEyMTAwNiwgMHgxMDE0MDBEQywgMHgxMDE1MDBERCwgCi0gICAgICAgIDB4MTAxNjAwREUsIDB4MTAxNzAwREYsIDB4MTAxODAwMDcsIDB4MTAxQTEwMDcsIDB4MTAxQjEwMDYsIDB4MTAxQzAwMDYsIDB4MTAxRDAwRTAsIDB4MTAxRTEwMDYsIAotICAgICAgICAweDEwMjAwMDM4LCAweDEwMjEwMDA2LCAweDEwMjIwMEUxLCAweDEwMjMwMDBBLCAweDEwMjQxMDA2LCAweDEwMjUwMDA2LCAweDEwMjkwMDE5LCAweDEwMkEwMDM4LCAKLSAgICAgICAgMHgxMDJCMDAwNiwgMHgxMDMwMDA1NywgMHgxMDMyMDA3OCwgMHgxMDM1MDA1NywgMHgxMDM4NzhFMiwgMHgxMDM5MDA3OCwgMHgxMDNBNzhFMywgMHgxMDNCNzhFNCwgCi0gICAgICAgIDB4MTAzQzc4RTUsIDB4MTAzRDc4MEIsIDB4MTAzRTc4MTksIDB4MTAzRjc4MEEsIDB4MTA0MDcwRTIsIDB4MTA0MTcwNUEsIDB4MTA0MjcwRTMsIDB4MTA0MzcwRTQsIAotICAgICAgICAweDEwNDQ3MEU1LCAweDEwNDU3MDBCLCAweDEwNDY3MDE5LCAweDEwNDc3MDBBLCAweDEwNDg3MDgxLCAweDEwNEIwMDc4LCAweDEwNTAwMDA4LCAweDEwNTQxMDA4LCAKLSAgICAgICAgMHgxMDU1MDAwOCwgMHgxMDVCMDA3OCwgMHgxMDY4MDA4MywgMHgxMDZGMDA5NSwgMHgxMDczMDA4MywgMHgxMDc2MDA3OCwgMHgxMDgwMTA1NCwgMHgxMDgxMjg3NywgCi0gICAgICAgIDB4MTA4MjAwNTQsIDB4MTA4MzEwNTQsIDB4MTA4NDAwNTQsIDB4MTA4NTI4NTUsIDB4MTA4NjI4NzcsIDB4MTA4NzI4NTUsIDB4MTA4ODI4NzcsIDB4MTA4QTAwNTQsIAotICAgICAgICAweDEwOEIxMDU0LCAweDEwOEMwMDU0LCAweDEwOEQyODc3LCAweDEwOEYwMDU0LCAweDEwOTA3ODU0LCAweDEwOTIyODc3LCAweDEwOTMwOEU2LCAweDEwOTQyODc3LCAKLSAgICAgICAgMHgxMDk1MDhFNywgMHgxMDk2Mjg3NywgMHgxMDk3MDA1OCwgMHgxMDk4Mjg3NywgMHgxMDk5MDA1NCwgMHgxMDlBMjg1NSwgMHgxMDlCMTA3MSwgMHgxMDlEMDA1NCwgCi0gICAgICAgIDB4MTA5RTI4NTUsIDB4MTA5RjI4NzcsIDB4MTBBMDI4RTgsIDB4MTBBMTAwMTksIDB4MTBBMzI4NTUsIDB4MTBBNTAwNTQsIDB4MTBBNzAwNzgsIDB4MTBBQTMwNUYsIAotICAgICAgICAweDEwQjAxMEU5LCAweDEwQjExMEVBLCAweDEwQjIxMEVCLCAweDEwQjMxMEVDLCAweDEwQjQxMEVELCAweDEwQjUxMEVFLCAweDEwQjYxMEVGLCAweDEwQjcxMEYwLCAKLSAgICAgICAgMHgxMEI4MTBGMSwgMHgxMEI5MTBGMiwgMHgxMEJBMTBGMywgMHgxMEJCMTBGNCwgMHgxMEJDMTBGNSwgMHgxMEJEMTBGNiwgMHgxMEJFMTBGNywgMHgxMEJGMTBGOCwgCi0gICAgICAgIDB4MTBDMDAwRjksIDB4MTBDMTAwRkEsIDB4MTBDMjAwNzgsIDB4MTBDODAwMTksIDB4MTBDQjAwNTQsIDB4MTBDRDA4MTksIDB4MTBDRTAwNTQsIDB4MTBEMDAwMTksIAotICAgICAgICAweDEwRDEwMDU0LCAweDEwRDMwMDE5LCAweDEwRDQwMDU0LCAweDEwRDcwODE5LCAweDEwRDgwMDU0LCAweDEwRTcwODE5LCAweDEwRTgwMDU0LCAweDEwRTkwMDE5LCAKLSAgICAgICAgMHgxMEVCMDA1NCwgMHgxMEZBMDAxOSwgMHgxMTAxMDBFOCwgMHgxMTAyMDhFOCwgMHgxMTAzMDAxOSwgMHgxMTA0MDBGQiwgMHgxMTA2MDhGQywgMHgxMTA3MDAxOSwgCi0gICAgICAgIDB4MTEwOTAwMEIsIDB4MTEwQTAwMTksIDB4MTEwQjAwRTgsIDB4MTEwQzAwMTksIDB4MTEwRDAwRTgsIDB4MTEwRjAwMTksIDB4MTExMDAwRTgsIDB4MTExMjA4RTgsIAotICAgICAgICAweDExMTQwMDE5LCAweDExMTYxMEU4LCAweDExMTcwMEU4LCAweDExMTgxMEU4LCAweDExMTkwMEU4LCAweDExMUEwMDE5LCAweDExMUUwMEZELCAweDExMUYwMEU4LCAKLSAgICAgICAgMHgxMTIyMDhFOCwgMHgxMTIzMDBFOCwgMHgxMTI3MDAxOSwgMHgxMTI5MDBGRCwgMHgxMTJCMDAxOSwgMHgxMTMwMDhFOCwgMHgxMTMyMDBGRCwgMHgxMTM2MDAxOSwgCi0gICAgICAgIDB4MTEzNzA4RkQsIDB4MTEzOTAwRkQsIDB4MTEzQTA4RkQsIDB4MTEzQjAwRkQsIDB4MTEzQzA4RkQsIDB4MTEzRDAwRkQsIDB4MTE0MDA4RkQsIDB4MTE0MTAwRkQsIAotICAgICAgICAweDExNDIwOEZELCAweDExNDMwMEZELCAweDExNDQwOEZELCAweDExNDUwMEZELCAweDExNDYwMEU4LCAweDExNDcwMDE5LCAweDExNDgwMEZFLCAweDExNEEwMDE5LCAKLSAgICAgICAgMHgxMTRDMDBGRiwgMHgxMTREMDAxOSwgMHgxMTUxMDBGRCwgMHgxMTUyMDAxOSwgMHgxMTUzMDEwMCwgMHgxMTU0MDEwMSwgMHgxMTU1MDBFOCwgMHgxMTU2MDhFOCwgCi0gICAgICAgIDB4MTE1ODAwRkQsIDB4MTE1QzAwRTgsIDB4MTE1RDAwMTksIDB4MTE1RjAwRTgsIDB4MTE2MDAwMTksIDB4MTE2NTAwRkUsIDB4MTE2NzAwMTksIDB4MTE2ODAwRkQsIAotICAgICAgICAweDExNjkwMDE5LCAweDExNkIwMEZELCAweDExNzAwOEZELCAweDExNzIwMEZELCAweDExNzUwOEZELCAweDExNzcwMDE5LCAweDExNzgwMEZELCAweDExNzkwMTAyLCAKLSAgICAgICAgMHgxMTdCMDEwMywgMHgxMTdDMDBFOCwgMHgxMTdEMDEwNCwgMHgxMTdGMDEwNSwgMHgxMTgwMDA1NCwgMHgxMTg0MDBGRCwgMHgxMTg2MDA1NCwgMHgxMTkwMDBFOCwgCi0gICAgICAgIDB4MTE5MTAwNTQsIDB4MTE5NTA4MEEsIDB4MTE5NjAwNTQsIDB4MTE5QjAwOTQsIDB4MTFCRTAwMTksIDB4MTFCRjAwNTQsIDB4MTFDRTAwMTksIDB4MTFEQTAwQjQsIAotICAgICAgICAweDExREIwMDA2LCAweDExREMwMDU0LCAweDExRUUwMDc4LCAweDEyMDAwMDU0LCAweDEyMTQwMDc4LCAweDEyMjAwMDU0LCAweDEyMjYwMDc4LCAweDEyMzAxOTA2LCAKLSAgICAgICAgMHgxMjMxMTkwNywgMHgxMjMyMTkwOCwgMHgxMjMzMTkwOSwgMHgxMjM0MTkwQSwgMHgxMjM1MTkwQiwgMHgxMjM2MTkwQywgMHgxMjM3MTkwRCwgMHgxMjM4MTkwRSwgCi0gICAgICAgIDB4MTIzOTE5MEYsIDB4MTIzQTExMDYsIDB4MTIzQjExMDcsIDB4MTIzQzExMDgsIDB4MTIzRDExMDksIDB4MTIzRTExMEEsIDB4MTIzRjExMEIsIDB4MTI0MDExMEMsIAotICAgICAgICAweDEyNDExMTBELCAweDEyNDIxMTBFLCAweDEyNDMxMTBGLCAweDEyNDQxMDVELCAweDEyNDUxMDVCLCAweDEyNDYxMTEwLCAweDEyNDcxMTExLCAweDEyNDgxMTEyLCAKLSAgICAgICAgMHgxMjQ5MTExMywgMHgxMjRBMTExNCwgMHgxMjRCMTExNSwgMHgxMjRDMTExNiwgMHgxMjREMTExNywgMHgxMjRFMTA5NCwgMHgxMjVCMTkxOCwgMHgxMjY4MTkxOSwgCi0gICAgICAgIDB4MTI3NTE4QzMsIDB4MTI3NjAxMUEsIDB4MTI3NzAxMUIsIDB4MTI3ODAxMUMsIDB4MTI3OTAxMUQsIDB4MTI3QTAxMUUsIDB4MTI3QjAwQzQsIDB4MTI3QzAwQzUsIAotICAgICAgICAweDEyN0QwMEM2LCAweDEyN0UwMEM3LCAweDEyN0YwMTFGLCAweDEyODAwMDU0LCAweDEyRkMwMDE5LCAweDEzMDAwMDU0LCAweDEzNEYwMDc4LCAweDEzNTAwMDU0LCAKLSAgICAgICAgMHgxMzU2MDA5NCwgMHgxMzU3MDA1NCwgMHgxMzU5MDA3OCwgMHgxMzgxMDA1NCwgMHgxMzg1MDA3OCwgMHgxMzg2MDA1NCwgMHgxMzk0MDA3OCwgMHgxMzk1MDA1NCwgCi0gICAgICAgIDB4MTNBNjAwNzgsIDB4MTNBODAwNTQsIDB4MTNBQTAwNzgsIDB4MTNBQjAwNTQsIDB4MTNCMDAwNzgsIDB4MTNCMTAwNTQsIDB4MTNCNDAwMDksIDB4MTNCQjAxMDYsIAotICAgICAgICAweDEzQkMwMTA3LCAweDEzQkQwMTA4LCAweDEzQkUwMTA5LCAweDEzQkYwMTBBLCAweDEzQzAwMTA2LCAweDEzQzEwMTA3LCAweDEzQzIwMTA4LCAweDEzQzMwMTA5LCAKLSAgICAgICAgMHgxM0M0MDEwQSwgMHgxM0M1MDEwNiwgMHgxM0M2MDEwNywgMHgxM0M3MDEwOCwgMHgxM0M4MDEwOSwgMHgxM0M5MDEwQSwgMHgxM0NBMDA1NCwgMHgxM0NCMDA3OCwgCi0gICAgICAgIDB4MTNDQzAwNTQsIDB4MTNEODAwNzgsIDB4MTNEOTAwNTQsIDB4MTNFMDAwRTgsIDB4MTNFMTAwMTksIDB4MTNFMjAwRkUsIDB4MTNFMzAwMEEsIDB4MTNFNDAwNzgsIAotICAgICAgICAweDEzRTgwMDE5LCAweDEzRUEwMEU4LCAweDEzRUIwMEZFLCAweDEzRUMwMDE5LCAweDEzRUUwMEU4LCAweDEzRUYwMEZFLCAweDEzRjAwMDE5LCAweDEzRjEwMEZELCAKLSAgICAgICAgMHgxM0YzMDAwOSwgMHgxM0Y2MDA3OCwgMHgxM0Y4MDAxOSwgMHgxNDAwMDA5NCwgMHgxNDgwMDAxOSwgMHgxNEMyMDAwQSwgMHgxNEM3MDEyMCwgMHgxNEM4MDEyMSwgCi0gICAgICAgIDB4MTRDOTAwMEEsIDB4MTRDRDAwMTksIDB4MTRDRTAwRTgsIDB4MTREODAwMTksIDB4MTREQzAxMjIsIDB4MTRERDAwMTksIDB4MTRFMDAwRkQsIDB4MTRFMTAwRTgsIAotICAgICAgICAweDE0RTIwMEZELCAweDE0RTMwMDE5LCAweDE0RTcwMEU4LCAweDE0RTgwMEZFLCAweDE0RUEwMEZELCAweDE0RUIwMDE5LCAweDE0RUMwMDA5LCAweDE0RUUwMEU4LCAKLSAgICAgICAgMHgxNEVGMDAxOSwgMHgxNEYyMDBFOCwgMHgxNEYzMDAxOSwgMHgxNEY0MDBFOCwgMHgxNEY1MDAxOSwgMHgxNEZBMDBFOCwgMHgxNEZDMDBGRCwgMHgxNEZEMDAxOSwgCi0gICAgICAgIDB4MTRGRTAwMDksIDB4MTRGRjAwMTksIDB4MTUwNTAwRTgsIDB4MTUwNjEwRTgsIDB4MTUwNzAwRTgsIDB4MTUxMTAwMTksIDB4MTUxMjAwRTgsIDB4MTUxNDAwMTksIAotICAgICAgICAweDE1MTYwMEZFLCAweDE1MTgwMDE5LCAweDE1MUEwMEZELCAweDE1MUIwMDE5LCAweDE1MUUwMEZELCAweDE1MUYwMEU4LCAweDE1MjAwMDE5LCAweDE1MkMwMEU4LCAKLSAgICAgICAgMHgxNTJEMDAxOSwgMHgxNTMyMDBGRCwgMHgxNTMzMDAxOSwgMHgxNTM1MDBFOCwgMHgxNTM3MDAxOSwgMHgxNTM4MDBFOCwgMHgxNTM5MDAxOSwgMHgxNTNBMTBFOCwgCi0gICAgICAgIDB4MTUzQjEwMTksIDB4MTUzQzAwMTksIDB4MTUzRDAwRkUsIDB4MTUzRTAwRTgsIDB4MTUzRjAwRkUsIDB4MTU0MzAwRTgsIDB4MTU0NjAwRkUsIDB4MTU0NzAwRTgsIAotICAgICAgICAweDE1NDkwMEZFLCAweDE1NEYwMEU4LCAweDE1NTEwMEZFLCAweDE1NTIwMDE5LCAweDE1NTMwMEZELCAweDE1NTcwMDE5LCAweDE1NTgwMEZFLCAweDE1NTkwMEU4LCAKLSAgICAgICAgMHgxNTVBMDBGRSwgMHgxNTVCMDBFOCwgMHgxNTVFMDBGRSwgMHgxNTY0MDBFOCwgMHgxNTY3MDBGRSwgMHgxNTZDMDAxOSwgMHgxNTZFMDhFOCwgMHgxNTZGMDEyMywgCi0gICAgICAgIDB4MTU3MDAwMTksIDB4MTU3MTAwRTgsIDB4MTU3MjAxMjQsIDB4MTU3MzAwRTgsIDB4MTU3NDAwMTksIDB4MTU3NjAwRkQsIDB4MTU3NzAwRTgsIDB4MTU3ODAwMTksIAotICAgICAgICAweDE1N0MwMEZFLCAweDE1N0UwMDE5LCAweDE1ODAwMDU0LCAweDE1OEEwMDc4LCAweDE2MDAwMDk2LCAweDE2MTgwMDk4LCAweDE2MzAwMDc4LCAweDE2NDAwMDYzLCAKLSAgICAgICAgMHgxNjcyMDA1NSwgMHgxNjczMDA1NCwgMHgxNjc2MDA3OCwgMHgxNjdEMDAwNiwgMHgxNjgwMDEyNSwgMHgxNjkzMDA3OCwgMHgxNjk4MDA3MSwgMHgxNkIzMDA3OCwgCi0gICAgICAgIDB4MTZDMDAwNzEsIDB4MTZDQzAwNzgsIDB4MTZEMDAwNzEsIDB4MTZGMDAwNzgsIDB4MTcwMDAwMDYsIDB4MTcwMTAxMjYsIDB4MTcwMzAwMDYsIDB4MTcwNTAwRTAsIAotICAgICAgICAweDE3MDYwMTI2LCAweDE3MDcwMDA2LCAweDE3MEMwMDc4LCAweDE3MEUwMTI2LCAweDE3MEYwMDc4LCAweDE3NDAwMDU0LCAweDE3NEQwMDc4LCAweDE3NEUwMDU0LCAKLSAgICAgICAgMHgxNzdBMDA3OCwgMHgxNzgwMTA1NCwgMHgxN0VCMDA3OCwgMHgxN0Y4MDA1NCwgMHgxN0ZFMDA3OCwgMHgxODAwODgwNSwgMHgxODAxMDAwNiwgMHgxODAyMDA1NCwgCi0gICAgICAgIDB4MTgwMzAwNzEsIDB4MTgwNDAwMDksIDB4MTgwOTAwNTQsIDB4MTgwQTAwMDksIDB4MTgwRTAwOTksIDB4MTgwRjAwQkYsIDB4MTgxMDAwNTQsIDB4MTgxMTAxMjcsIAotICAgICAgICAweDE4MTIwMTI4LCAweDE4MTMwMTI5LCAweDE4MTQwMTJBLCAweDE4MTUwMDgzLCAweDE4MTgwMDk5LCAweDE4MTkwMDgxLCAweDE4MUIxMDU0LCAweDE4MUMxMTJCLCAKLSAgICAgICAgMHgxODFEMTEyQywgMHgxODFFMDA3MSwgMHgxODFGMDA1NCwgMHgxODIwMDA3OCwgMHgxODIxMDA3MSwgMHgxODI2MDg3MSwgMHgxODMyMDA3MSwgMHgxODM4MDg3MSwgCi0gICAgICAgIDB4MTgzOTAwNzEsIDB4MTgzQTA4NzEsIDB4MTgzQzAwNzEsIDB4MTgzRDA4NzEsIDB4MTgzRjAwNzEsIDB4MTg0QTA4NzEsIDB4MTg0QjAwNzEsIDB4MTg0QzAwNzgsIAotICAgICAgICAweDE4NEQwMDgzLCAweDE4NEUxMDM3LCAweDE4NEYwODgxLCAweDE4NTAwMDk5LCAweDE4NTEwMDcxLCAweDE4NTYwODcxLCAweDE4NjIwMDcxLCAweDE4NjgwODcxLCAKLSAgICAgICAgMHgxODY5MDA3MSwgMHgxODZBMDg3MSwgMHgxODZDMDA3MSwgMHgxODZEMDg3MSwgMHgxODZGMDA3MSwgMHgxODdBMDg3MSwgMHgxODdCMDA3MSwgMHgxODdDMDg3MSwgCi0gICAgICAgIDB4MTg3RTAwODEsIDB4MTg3RjA4ODEsIDB4MTg4MDAwNzgsIDB4MTg4MzAwNzEsIDB4MTg5NzAwNzgsIDB4MTg5OTEwNzEsIDB4MThDODAwOTQsIDB4MThDOTc4QUQsIAotICAgICAgICAweDE4Q0E3OEFFLCAweDE4Q0I3ODk0LCAweDE4RDAwMDcxLCAweDE4REMwMDc4LCAweDE4RTAwMDU0LCAweDE4RTgwMDc4LCAweDE4RjgwMDcxLCAweDE5MDAxMDk0LCAKLSAgICAgICAgMHgxOTBGMTA1NCwgMHgxOTEwMTBBRCwgMHgxOTExMTBBRSwgMHgxOTEyMTEyRCwgMHgxOTEzMTEyRSwgMHgxOTE0MTEyRiwgMHgxOTE1MTA5NCwgMHgxOTIyMDA3OCwgCi0gICAgICAgIDB4MTkyODY4NTQsIDB4MTkyOTE5MzAsIDB4MTkyQTE5MzEsIDB4MTkyQjE5MzIsIDB4MTkyQzE5MzMsIDB4MTkyRDE5MzQsIDB4MTkyRTE5MzUsIDB4MTkyRjE5MzYsIAotICAgICAgICAweDE5MzAxODk0LCAweDE5M0UxODU0LCAweDE5NDAxOEFELCAweDE5NDExOEFFLCAweDE5NDIxOTJELCAweDE5NDMxOTJFLCAweDE5NDQxOTJGLCAweDE5NDUxODk0LCAKLSAgICAgICAgMHgxOTU5MTkzNywgMHgxOTVBMTkzOCwgMHgxOTVCMTkzOSwgMHgxOTVDMTkzQSwgMHgxOTVEMTkzQiwgMHgxOTVFMTkzQywgMHgxOTVGMTkzRCwgMHgxOTYwMTA5NCwgCi0gICAgICAgIDB4MTk2NjY4NTQsIDB4MTk2ODE4OTQsIDB4MTk4MDY4OTQsIDB4MTlBQzEwOTQsIDB4MTlCOTY4OTQsIDB4MTlCQzY4NTQsIDB4MTlCRTY4OTQsIDB4MTlFRjY4NTQsIAotICAgICAgICAweDE5RjAxMDk0LCAweDFBMDAwMDcxLCAweDI2REIwMDc4LCAweDI2RTAwMDU0LCAweDI3MDAwMDcxLCAweDRGREUwMDc4LCAweDUwMDAwMDcxLCAweDUyNDcwMDc4LCAKLSAgICAgICAgMHg1MjQ4MDA1NCwgMHg1MjY0MDA3OCwgMHg1MzgwMDAzNywgMHg1MzhDMDA3OCwgMHg1NDAwMDA3MSwgMHg1NDAxMDBDOCwgMHg1NDAyMDA3MSwgMHg1NDAzMDA4MywgCi0gICAgICAgIDB4NTQwNDAwNzEsIDB4NTQxMjAwQTcsIDB4NTQxMzAwODMsIDB4NTQxNDAwNTQsIDB4NTQxNjAwNzgsIDB4NTYwMDAwNzEsIDB4NkJEMjAwNzgsIDB4NkMwMDAxM0UsIAotICAgICAgICAweDcwMDAwMTNGLCAweDdDODAwODcxLCAweDdEMDcwMDcxLCAweDdEMDgwODcxLCAweDdEMEEwMDcxLCAweDdEMEIwODcxLCAweDdEMTIwMDcxLCAweDdEMTMwODcxLCAKLSAgICAgICAgMHg3RDE0MDA3MSwgMHg3RDE1MDg3MSwgMHg3RDE3MDA3OCwgMHg3RDE4MDg3MSwgMHg3RDM2MDA3OCwgMHg3RDM4MDg3MSwgMHg3RDZEMDA3OCwgMHg3RDgwMTA1NSwgCi0gICAgICAgIDB4N0Q4NDAwNzgsIDB4N0Q4QTEwNTUsIDB4N0Q4QzAwNzgsIDB4N0Q4RjAwODMsIDB4N0Q5MDI4OUIsIDB4N0Q5NTA4OUIsIDB4N0RBMTAwNzgsIDB4N0RBMjA4OUIsIAotICAgICAgICAweDdEQTg0MDlFLCAweDdEQUEzODlFLCAweDdEQUI0MDlFLCAweDdEQUMzODlFLCAweDdEQUQ0MDlFLCAweDdEQUUzODlFLCAweDdEQUY0MDlFLCAweDdEQjAzODlFLCAKLSAgICAgICAgMHg3REIxNDA5RSwgMHg3REIyMzg5RSwgMHg3REIzNDA5RSwgMHg3REI0Mzg5RSwgMHg3REI1NDA5RSwgMHg3REI2Mzg5RSwgMHg3REI3NDA5RSwgMHg3REI4Mzg5RSwgCi0gICAgICAgIDB4N0RCOTQwOUUsIDB4N0RCQTM4OUUsIDB4N0RCQjQwOUUsIDB4N0RCQzM4OUUsIDB4N0RCRDQwOUUsIDB4N0RCRTM4OUUsIDB4N0RCRjQwOUUsIDB4N0RDMDM4OUUsIAotICAgICAgICAweDdEQzE0MDlFLCAweDdEQzgzODlFLCAweDdEQzk0MDlFLCAweDdEQ0EzODlFLCAweDdEQ0I0MDlFLCAweDdEQ0MzODlFLCAweDdEQ0Q0MDlFLCAweDdEQ0UzODlFLCAKLSAgICAgICAgMHg3RENGNDA5RSwgMHg3REQxMzg5RSwgMHg3REQyNDA5RSwgMHg3REQ0Mzg5RSwgMHg3REQ1NDA5RSwgMHg3REQ2Mzg5RSwgMHg3REQ3NDA5RSwgMHg3REQ5MDA3OCwgCi0gICAgICAgIDB4N0RFQTIwOUUsIDB4N0RFQjQ4OUUsIDB4N0RFQzIwOUUsIDB4N0RFRjQwOUUsIDB4N0RGMzM4OUUsIDB4N0RGNTQwOUUsIDB4N0RGQzM4OUUsIDB4N0RGRDIwOUUsIAotICAgICAgICAweDdERkU0MDlFLCAweDdERkYzODlFLCAweDdFMDA0MDlFLCAweDdFMzIyMDlFLCAweDdFNEMzODlFLCAweDdFNzA0ODlFLCAweDdFN0I0MDlFLCAweDdFODkyMDlFLCAKLSAgICAgICAgMHg3RTk3Mzg5RSwgMHg3RTlBNDg5RSwgMHg3RTlFMjA5RSwgMHg3RTlGMDBCNCwgMHg3RUEwMDA3OCwgMHg3RUE4Mzg5RSwgMHg3RUFDMjA5RSwgMHg3RUFFMzg5RSwgCi0gICAgICAgIDB4N0VBRjIwOUUsIDB4N0VCMDM4OUUsIDB4N0VCMTIwOUUsIDB4N0VCNDM4OUUsIDB4N0VCNTIwOUUsIDB4N0VCODM4OUUsIDB4N0VCQTIwOUUsIDB4N0VDMzM4OUUsIAotICAgICAgICAweDdFQzgwMDc4LCAweDdFQzkzODlFLCAweDdFQ0IyMDlFLCAweDdFQ0MzODlFLCAweDdFQ0QyMDlFLCAweDdFREEzODlFLCAweDdFREIyMDlFLCAweDdFREMzODlFLCAKLSAgICAgICAgMHg3RURFMjA5RSwgMHg3RUUyMzg5RSwgMHg3RUUzMjA5RSwgMHg3RUU0MDA3OCwgMHg3RUY4NDA5RSwgMHg3RUZFNDE0MCwgMHg3RUZGMDA3OCwgMHg3RjAwMDA4MywgCi0gICAgICAgIDB4N0YwODgwMDYsIDB4N0YwQzgwQkYsIDB4N0YwRDAwNzgsIDB4N0YxMDAwODMsIDB4N0YxMjAwNzgsIDB4N0YxODgwMDYsIDB4N0YxOTgwOTksIDB4N0YxQTgwMzgsIAotICAgICAgICAweDdGMUI4MEJGLCAweDdGMjMwMDA2LCAweDdGMjQ4MEJGLCAweDdGMjUxMDA2LCAweDdGMjcxMDM4LCAweDdGMjg2MDBDLCAweDdGMkE2MDA2LCAweDdGMkM2MDk5LCAKLSAgICAgICAgMHg3RjJENjBCRiwgMHg3RjMwNjAwNiwgMHg3RjMxNjAwQiwgMHg3RjMyNjAxOSwgMHg3RjM0NjAwNiwgMHg3RjM1NjAwNywgMHg3RjM2MDA3OCwgMHg3RjM4NDA5RSwgCi0gICAgICAgIDB4N0Y0MTIwOUUsIDB4N0Y0NjQ4OUUsIDB4N0Y0NzIwOUUsIDB4N0Y0OTQ4OUUsIDB4N0Y0QTIwOUUsIDB4N0Y0QzQ4OUUsIDB4N0Y0RDIwOUUsIDB4N0Y0RTQ4OUUsIAotICAgICAgICAweDdGNEYyMDlFLCAweDdGNTA0ODlFLCAweDdGNTEyMDlFLCAweDdGNTI0ODlFLCAweDdGNTMyMDlFLCAweDdGNTQ0ODlFLCAweDdGNTUyMDlFLCAweDdGNUE0ODlFLCAKLSAgICAgICAgMHg3RjVCMjA5RSwgMHg3RjVDNDg5RSwgMHg3RjVEMjA5RSwgMHg3RjVFNDg5RSwgMHg3RjVGMjA5RSwgMHg3RjYwNDg5RSwgMHg3RjYxMjA5RSwgMHg3RjYyNDg5RSwgCi0gICAgICAgIDB4N0Y2MzIwOUUsIDB4N0Y2NDQ4OUUsIDB4N0Y2NTIwOUUsIDB4N0Y2NjQ4OUUsIDB4N0Y2NzIwOUUsIDB4N0Y2ODQ4OUUsIDB4N0Y2OTIwOUUsIDB4N0Y2QTQ4OUUsIAotICAgICAgICAweDdGNkIyMDlFLCAweDdGNkM0ODlFLCAweDdGNkQyMDlFLCAweDdGNkU0ODlFLCAweDdGNkYyMDlFLCAweDdGNzA0ODlFLCAweDdGNzEyMDlFLCAweDdGNzI0ODlFLCAKLSAgICAgICAgMHg3RjczMjA5RSwgMHg3Rjc0NDg5RSwgMHg3Rjc1MjA5RSwgMHg3Rjc2NDg5RSwgMHg3Rjc3MjA5RSwgMHg3RjdBNDg5RSwgMHg3RjdCMjA5RSwgMHg3RjdGMDA3OCwgCi0gICAgICAgIDB4N0Y4MTg4MDYsIDB4N0Y4Mjg4MDgsIDB4N0Y4Mzg4MDYsIDB4N0Y4NDg4MDksIDB4N0Y4NTg4MDYsIDB4N0Y4Njg4MEMsIDB4N0Y4ODg4MEUsIDB4N0Y4OTg4MTAsIAotICAgICAgICAweDdGOEE4ODEyLCAweDdGOEI4ODE0LCAweDdGOEM4ODE2LCAweDdGOEQ4ODBDLCAweDdGOEU4ODE4LCAweDdGOEY4ODFBLCAweDdGOTA4ODA2LCAweDdGOTE4ODYwLCAKLSAgICAgICAgMHg3RjlFODgwNiwgMHg3RjlGODgzNywgMHg3RkExODg2MSwgMHg3RkFFODgxOSwgMHg3RkIwODgwQSwgMHg3RkIxNTAwOSwgMHg3RkIyNTAwNiwgMHg3RkIzNTA3MSwgCi0gICAgICAgIDB4N0ZCODUwODEsIDB4N0ZCOTUwNzEsIDB4N0ZDRjUwODEsIDB4N0ZEMDUwNzEsIDB4N0ZFMDAwNzgsIDB4N0ZFMTUwNzEsIDB4N0ZFNDAwNzgsIDB4N0ZFNTUwNzEsIAotICAgICAgICAweDdGRTgwMDc4LCAweDdGRTk1MDcxLCAweDdGRUMwMDc4LCAweDdGRUQ1MDcxLCAweDdGRUYwMDc4LCAweDdGRjA4ODA4LCAweDdGRjE4ODE5LCAweDdGRjI4ODU0LCAKLSAgICAgICAgMHg3RkYzODgwOCwgMHg3RkY0NTA1NCwgMHg3RkY1NTAxOSwgMHg3RkY3NTA1NCwgMHg3RkY4MDA3OCwgMHg3RkZEMDE0MSwgMHg3RkZFMDA1NCwgMHg3RkZGMDA3OCwgCi0gICAgICAgIDB4ODAwMDAwNzEsIDB4ODAwNjAwNzgsIDB4ODAwNzAwNzEsIDB4ODAxRjAwNzgsIDB4ODAyMDAwNzEsIDB4ODAyNzAwNzgsIDB4ODAyODAwNzEsIDB4ODAyRjAwNzgsIAotICAgICAgICAweDgwNDAwMDcxLCAweDgwN0UwMDc4LCAweDgwODAwMDk3LCAweDgwODEwMDk0LCAweDgwODIwMDc4LCAweDgwODQwMEI2LCAweDgwODUwMEI3LCAweDgwODYwMEI4LCAKLSAgICAgICAgMHg4MDg3MDBCOSwgMHg4MDg4MDBCMCwgMHg4MDg5MDBCQSwgMHg4MDhBMDBCQiwgMHg4MDhCMDBCQywgMHg4MDhDMDBCRCwgMHg4MDhEMDE0MiwgMHg4MDhFMDE0MywgCi0gICAgICAgIDB4ODA4RjAxNDQsIDB4ODA5MDAxNDUsIDB4ODA5MTAwQjEsIDB4ODA5MjAxNDYsIDB4ODA5MzAxNDcsIDB4ODA5NDAxNDgsIDB4ODA5NTAxNDksIDB4ODA5NjAxNEEsIAotICAgICAgICAweDgwOTcwMTRCLCAweDgwOTgwMTRDLCAweDgwOTkwMTRELCAweDgwOUEwMDc4LCAweDgwOUMwMDk0LCAweDgwQTAwMTRFLCAweDgwQTEwMTRGLCAweDgwQTIwMTUwLCAKLSAgICAgICAgMHg4MEEzMDE1MSwgMHg4MEE0MDE1MiwgMHg4MEE1MDE1MCwgMHg4MEE2MDE1MywgMHg4MEE3MDE1MSwgMHg4MEE4MDE1NCwgMHg4MEE5MDE1NSwgMHg4MEFBMDE1NiwgCi0gICAgICAgIDB4ODBBQjAxNTcsIDB4ODBBQzAxNEYsIDB4ODBBRTAxNTgsIDB4ODBCMDAxNTQsIDB4ODBCMzAxNTAsIDB4ODBCNTAxNTUsIDB4ODBCNjAxNTMsIDB4ODBCOTAxNTEsIAotICAgICAgICAweDgwQkEwMTUwLCAweDgwQkIwMDVGLCAweDgwQkQwMDU0LCAweDgwQzUwMEMzLCAweDgwQzYwMDc4LCAweDgxODAwMDcxLCAweDgxOTAwMEFELCAweDgxOTEwMEIwLCAKLSAgICAgICAgMHg4MTkyMDA3OCwgMHg4MTk4MDA3MSwgMHg4MUE1MDE1OSwgMHg4MUE2MDA3OCwgMHg4MUMwMDA3MSwgMHg4MUNGMDA3OCwgMHg4MUQwMDA3MSwgMHg4MUUyMDA3OCwgCi0gICAgICAgIDB4ODFFNDAwNzEsIDB4ODFFODAwOTQsIDB4ODFFOTAxNTgsIDB4ODFFQTAxNUEsIDB4ODFFQjAwNzgsIDB4ODIwMDAxNUIsIDB4ODIxNDAxNUMsIDB4ODIyODAwNzEsIAotICAgICAgICAweDgyNEYwMDc4LCAweDgyNTAwMEE4LCAweDgyNTEwMEE5LCAweDgyNTIwMEFBLCAweDgyNTMwMEFCLCAweDgyNTQwMEFDLCAweDgyNTUwMDc4LCAweDg0MDAwMDlCLCAKLSAgICAgICAgMHg4NDAzMDA3OCwgMHg4NDA0MDA5QiwgMHg4NDFCMDA3OCwgMHg4NDFDMDA5QiwgMHg4NDFEMDA3OCwgMHg4NDFFMDA5QiwgMHg4NDFGMDA3OCwgMHg4NTAwMDA5QiwgCi0gICAgICAgIDB4ODUwMTAwODMsIDB4ODUwMjAwNzgsIDB4ODUwMzAwODMsIDB4ODUwNDAwNzgsIDB4ODUwNjAwODMsIDB4ODUwODAwOUIsIDB4ODUwQTAwNzgsIDB4ODUwQjAwOUIsIAotICAgICAgICAweDg1MEMwMDc4LCAweDg1MEQwMDlCLCAweDg1MUEwMDc4LCAweDg1MUMwMDgzLCAweDg1MUUwMDc4LCAweDg1MjAwMTVELCAweDg1MjEwMTVFLCAweDg1MjIwMTVGLCAKLSAgICAgICAgMHg4NTIzMDE2MCwgMHg4NTI0MDA3OCwgMHg4NTI4MDA5QSwgMHg4NTJEMDA3OCwgMHhFODAwMDA5NCwgMHhFODdCMDA3OCwgMHhFODgwMDA5NCwgMHhFODk0MDA3OCwgCi0gICAgICAgIDB4RTg5NTAwOTQsIDB4RThBRjA4OTQsIDB4RThCMzAwQTcsIDB4RThCNDAwODMsIDB4RThCNTAwOTQsIDB4RThCNzAwQTcsIDB4RThCQTAwNTcsIDB4RThCRTAwODMsIAotICAgICAgICAweEU4QzIwMDk0LCAweEU4QzMwMDgzLCAweEU4QzYwMDk0LCAweEU4RDUwMDgzLCAweEU4RDcwMDk0LCAweEU4REUwODk0LCAweEU4RTEwMDk0LCAweEU4RUYwMDc4LCAKLSAgICAgICAgMHhFOTAwMDA1NCwgMHhFOTIxMDA4MywgMHhFOTIzMDA3OCwgMHhFOTgwMDA1NCwgMHhFOUFDMDA3OCwgMHhFQTAwMjg3NywgMHhFQTBEMjg1NSwgMHhFQTFBMjg3NywgCi0gICAgICAgIDB4RUEyNzI4NTUsIDB4RUEzNDI4NzcsIDB4RUE0MTI4NTUsIDB4RUE0RTI4NzcsIDB4RUE1MDAwNzgsIDB4RUE1MTI4NzcsIDB4RUE1MjAwNzgsIDB4RUE1MzI4NzcsIAotICAgICAgICAweEVBNTQwMDc4LCAweEVBNTUyODc3LCAweEVBNUIyODU1LCAweEVBNUQwMDc4LCAweEVBNUYyODU1LCAweEVBNjIwMDc4LCAweEVBNjMyODU1LCAweEVBNjgyODc3LCAKLSAgICAgICAgMHhFQTc1Mjg1NSwgMHhFQTgyMjg3NywgMHhFQTgzMDA3OCwgMHhFQTg0Mjg3NywgMHhFQTg2MDA3OCwgMHhFQTg3Mjg3NywgMHhFQThGMjg1NSwgMHhFQTlDMjg3NywgCi0gICAgICAgIDB4RUE5RDAwNzgsIDB4RUE5RTI4NzcsIDB4RUFBNDAwNzgsIDB4RUFBNTI4NzcsIDB4RUFBOTI4NTUsIDB4RUFCNjI4NzcsIDB4RUFDMzI4NTUsIDB4RUFEMDI4NzcsIAotICAgICAgICAweEVBREQyODU1LCAweEVBRUEyODc3LCAweEVBRjcyODU1LCAweEVCMDQyODc3LCAweEVCMTEyODU1LCAweEVCMUUyODc3LCAweEVCMkIyODU1LCAweEVCMzgyODc3LCAKLSAgICAgICAgMHhFQjQ1Mjg1NSwgMHhFQjUzMDA3OCwgMHhFQjU0Mjg3NywgMHhFQjYxMjg1NSwgMHhFQjcxMjg3NywgMHhFQjdFMjg1NSwgMHhFQjhFMjg3NywgMHhFQjlCMjg1NSwgCi0gICAgICAgIDB4RUJBQjI4NzcsIDB4RUJCODI4NTUsIDB4RUJDODI4NzcsIDB4RUJENTI4NTUsIDB4RUJFNTAwNzgsIDB4RUJFNzI4MEUsIDB4RUJFODI4MTAsIDB4RUJFOTI4MTIsIAotICAgICAgICAweEVCRUEyODE0LCAweEVCRUIyODE2LCAweEVCRUMyODBFLCAweEVCRUQyODEwLCAweEVCRUUyODEyLCAweEVCRUYyODE0LCAweEVCRjAyODE2LCAweEVCRjEyODBFLCAKLSAgICAgICAgMHhFQkYyMjgxMCwgMHhFQkYzMjgxMiwgMHhFQkY0MjgxNCwgMHhFQkY1MjgxNiwgMHhFQkY2MjgwRSwgMHhFQkY3MjgxMCwgMHhFQkY4MjgxMiwgMHhFQkY5MjgxNCwgCi0gICAgICAgIDB4RUJGQTI4MTYsIDB4RUJGQjI4MEUsIDB4RUJGQzI4MTAsIDB4RUJGRDI4MTIsIDB4RUJGRTI4MTQsIDB4RUJGRjI4MTYsIDB4RUMwMDAwNzgKLSAgICB9OwotCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IGExW10gPSB7Ci0gICAgICAgIDB4MDAwMDAwNzEsIDB4NTM2QzAwNzgsIDB4N0MwMDA4NzEsIDB4N0QwRjAwNzgKLSAgICB9OwotCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IGE3W10gPSB7Ci0gICAgICAgIDB4MDAxMDAwNTcsIDB4MDA0MDAwNzgsIDB4MDA4MDAwODMsIDB4MDBGODAwNzgsIDB4ODAwMDAxM0YsIDB4RkZGRjAwNzgKLSAgICB9OwotCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IGE4W10gPSB7Ci0gICAgICAgIDB4MDAwMDAxM0YsIDB4N0ZGRjAwNzgKLSAgICB9OwotCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IGExNltdID0gewotICAgICAgICAweDAwODAwODY1LCAweDAwODgwMDY1LCAweDAwODkwODY1LCAweDAwOTMwMDY1LCAweDAwOTQwODY1LCAweDAwOTgwMTYxLCAweDAwOTkxMDY1LCAweDAwOUEwODY1LCAKLSAgICAgICAgMHgwMDlDMDg2MywgMHgwMDlGMTA2MywgMHgwMEEwMDA2MywgMHgwMEExMDg2MywgMHgwMEE0MTA1NSwgMHgwMEE1MDA2NSwgMHgwMEE2MDg2NSwgMHgwMEE5MDA2NSwgCi0gICAgICAgIDB4MDBBQTA4NjUsIDB4MDBCMzAwNjUsIDB4MDBCNDA4NjUsIDB4MDBCQzA4NjMsIDB4MDBCRjExNjIsIDB4MDBDMDAxNjMsIDB4MDBDMTAwNjUsIDB4MDBDMzAwNjMsIAotICAgICAgICAweDAwQzQwMDY4LCAweDAwQzUwMDYzLCAweDAwQzYwMDU1LCAweDAwQzcwMTY0LCAweDAwQzgwMDYzLCAweDAwQzkwMDY4LCAweDAwQ0EwMTY1LCAweDAwQ0IwMTY2LCAKLSAgICAgICAgMHgwMENDMDA2NSwgMHgwMENEMDA1NSwgMHgwMENFMDE2NywgMHgwMENGMDE2OCwgMHgwMEQwMDg2NSwgMHgwMEQxMDA2NSwgMHgwMEQzMDA2MywgMHgwMEQ0MDA2RiwgCi0gICAgICAgIDB4MDBENTAwNTUsIDB4MDBENjAwNjUsIDB4MDBENzA4NjMsIDB4MDBEODAwNzAsIDB4MDBEOTAwNjMsIDB4MDBEQjAxNjksIDB4MDBEQzAwNjUsIDB4MDBERDAwNzEsIAotICAgICAgICAweDAwREUwMDY1LCAweDAwREYwMTZBLCAweDAwRTAwMDcxLCAweDAwRTIxMDc0LCAweDAwRTMxMDcyLCAweDAwRTQxMDczLCAweDAwRTUxMDc0LCAweDAwRTYwODYzLCAKLSAgICAgICAgMHgwMEVFMDE2QiwgMHgwMEVGMDg2NSwgMHgwMEYyMDA2NSwgMHgwMEYzMDg2NSwgMHgwMEY4MTA3MiwgMHgwMEY5MTA3MywgMHgwMEZBMDg2NSwgMHgwMEZCMDE2QywgCi0gICAgICAgIDB4MDBGQzA4NjUsIDB4MDEwRTAwNjUsIDB4MDEwRjA4NjUsIDB4MDExMDAwNTUsIDB4MDExMTAwNjUsIDB4MDExMzA4NjUsIDB4MDExQTAwNTUsIDB4MDExRDAwNjMsIAotICAgICAgICAweDAxMUUwMTZELCAweDAxMUYwMDU1LCAweDAxMjAwMTZFLCAweDAxMjEwMDc4LCAweDAxMjgwMDU1LCAweDAxMjkwMTZGLCAweDAxMkEwMDU1LCAweDAxMkIwMDdBLCAKLSAgICAgICAgMHgwMTJDMDE3MCwgMHgwMTJEMDE3MSwgMHgwMTJFMDA1NSwgMHgwMTMxMDE3MiwgMHgwMTMyMDA1NSwgMHgwMTM0MDE3MywgMHgwMTM1MDA1NSwgMHgwMTM3MDE3MywgCi0gICAgICAgIDB4MDEzODAwNTUsIDB4MDEzQTAxNzQsIDB4MDEzQjAwNTUsIDB4MDE0MTAwN0QsIDB4MDE0MjAwNTUsIDB4MDE0NTAwN0UsIDB4MDE0NjAwNTUsIDB4MDE1ODc4ODEsIAotICAgICAgICAweDAxNUMwMDgyLCAweDAxNUQwMDgxLCAweDAxNjEwMDM3LCAweDAxNjMwMDgyLCAweDAxNjgwMDgxLCAweDAxNjkwMDM3LCAweDAxNkMxMDM3LCAweDAxNkYwMDM3LCAKLSAgICAgICAgMHgwMTcwNzg4MSwgMHgwMTcyMDAzNywgMHgwMTgwMDA4MywgMHgwMUEwMDg4MywgMHgwMUEyMDE3NSwgMHgwMUEzMDA4MywgMHgwMUI4MDA3OCwgMHgwMUJBMDAzNywgCi0gICAgICAgIDB4MDFCQjAwNzgsIDB4MDFDMjA4MzcsIDB4MDFDMzA4MDYsIDB4MDFDNDA4ODUsIDB4MDFDNTAwNzgsIDB4MDFDNzA4ODcsIDB4MDFDODAwNjAsIDB4MDFENTA4NjAsIAotICAgICAgICAweDAxRDYwODg5LCAweDAxRDgwMDYxLCAweDAxRTUwODYxLCAweDAxRTYwODhDLCAweDAxRTcwMDc4LCAweDAxRTgxMTc2LCAweDAxRTkwODc3LCAweDAxRUExMTc3LCAKLSAgICAgICAgMHgwMUVCMDA1NSwgMHgwMUVDMDA2NSwgMHgwMUY4MTA5MywgMHgwMUY5MDA1NSwgMHgwMUZBMTE3OCwgMHgwMUZCMDA2MywgMHgwMUZDMTBEOCwgMHgwMUZEMDA2NSwgCi0gICAgICAgIDB4MDFGRTAwNzcsIDB4MDIwMDA4OTIsIDB4MDIwMjAwOTIsIDB4MDIwMzA4OTIsIDB4MDIwNDAwOTIsIDB4MDIwNjA4OTIsIDB4MDIwNzAwOTIsIDB4MDIwODAwNjAsIAotICAgICAgICAweDAyMEMwODYwLCAweDAyMEQwMDYwLCAweDAyMTgwMDYxLCAweDAyMUMwODYxLCAweDAyMUQwMDYxLCAweDAyMjgwODkzLCAweDAyMkEwMDkzLCAweDAyMkIwODkzLCAKLSAgICAgICAgMHgwMjJDMDA5MywgMHgwMjJFMDg5MywgMHgwMjJGMDA5MywgMHgwMjMwMDA2NSwgMHgwMjNCMDg2NSwgMHgwMjNDMDA2NSwgMHgwMjQxMDA4MywgMHgwMjQzMDA3OCwgCi0gICAgICAgIDB4MDI0NDAwOTUsIDB4MDI0NTAwNjUsIDB4MDI2MDA4NjMsIDB4MDI2MTAwNjMsIDB4MDI2NzAwNzgsIDB4MDI2ODA4NjUsIDB4MDI2QTAwNjUsIDB4MDI2QjA4NjUsIAotICAgICAgICAweDAyNkMwMDY1LCAweDAyNkQwODY1LCAweDAyNzAwMDY1LCAweDAyNzEwODY1LCAweDAyNzQwMDY1LCAweDAyNzUwODY1LCAweDAyN0IwMDY1LCAweDAyN0MwODY1LCAKLSAgICAgICAgMHgwMjdEMDA3OCwgMHgwMjgwMDA2NSwgMHgwMjg4MDA3OCwgMHgwMjk4MDA5NiwgMHgwMkFCMDA3OCwgMHgwMkFDMDA4MSwgMHgwMkFEMDA5NywgMHgwMkIwMDA5OCwgCi0gICAgICAgIDB4MDJDMzEwNTUsIDB4MDJDNDAwOTcsIDB4MDJDNTAwNzgsIDB4MDJDODAwODMsIDB4MDJFMTAwOUEsIDB4MDJFMjAwODMsIDB4MDJFNDAwNzgsIDB4MDJFODAwOUIsIAotICAgICAgICAweDAyRjUwMDc4LCAweDAyRjgwMDlCLCAweDAyRjkwMDlBLCAweDAyRkEwMDc4LCAweDAzMDAwMDlDLCAweDAzMDIwMDc4LCAweDAzMDUwMTQwLCAweDAzMDYwMDlELCAKLSAgICAgICAgMHgwMzA3MDA1NCwgMHgwMzA4MDA4MywgMHgwMzBCMDA3OCwgMHgwMzBEMDA5RCwgMHgwMzBFMDA3OCwgMHgwMzBGMDA5RCwgMHgwMzEwMDA5RSwgMHgwMzExMDg5RSwgCi0gICAgICAgIDB4MDMxMzAwOUUsIDB4MDMxRDAwNzgsIDB4MDMyMDAwOUUsIDB4MDMyNTAwODMsIDB4MDMyRjAwNzgsIDB4MDMzMDAxNzksIDB4MDMzMTAxN0EsIDB4MDMzMjAxN0IsIAotICAgICAgICAweDAzMzMwMTdDLCAweDAzMzQwMTdELCAweDAzMzUwMEE1LCAweDAzMzYwMDlELCAweDAzMzcwMDlFLCAweDAzM0ExMDlFLCAweDAzM0MwMDlFLCAweDAzNjkwODlFLCAKLSAgICAgICAgMHgwMzZBMDA5RSwgMHgwMzZCMDA4MywgMHgwMzZFMDA5QywgMHgwMzZGMDA4MywgMHgwMzcyMDA5RiwgMHgwMzczMDA4MywgMHgwMzc0MDA1NCwgMHgwMzc1MDA4MywgCi0gICAgICAgIDB4MDM3NzAwOUUsIDB4MDM3ODAwMEYsIDB4MDM3OTAwMTEsIDB4MDM3QTAwMTMsIDB4MDM3QjAwMTUsIDB4MDM3QzAwMTcsIDB4MDM3RDAwOUUsIDB4MDM3RTAwQTYsIAotICAgICAgICAweDAzN0YwMDlFLCAweDAzODAwMDlELCAweDAzODcwMDU3LCAweDAzODgwMDgzLCAweDAzODkwMDlFLCAweDAzOTgwMDgzLCAweDAzQTUwMDc4LCAweDAzQTYwMDlFLCAKLSAgICAgICAgMHgwM0I3MDA3OCwgMHgwM0MwMDA5RSwgMHgwM0QzMDA4MywgMHgwM0Q4MDA5RSwgMHgwM0Q5MDA3OCwgMHgwNDgwMDA4MywgMHgwNDgxMDBBNywgMHgwNDgyMDA3MSwgCi0gICAgICAgIDB4MDQ5NDA4NzEsIDB4MDQ5NTAwNzEsIDB4MDQ5ODA4NzEsIDB4MDQ5OTAwNzEsIDB4MDQ5RDAwNzgsIDB4MDQ5RTAwNzEsIDB4MDQ5RjAwQTcsIDB4MDRBMDAwODMsIAotICAgICAgICAweDA0QTQwMEE3LCAweDA0QTYwMDgzLCAweDA0QTcwMDc4LCAweDA0QTgwMDgzLCAweDA0QUEwMDc4LCAweDA0QUMwODcxLCAweDA0QjAwMDcxLCAweDA0QjEwMDgzLCAKLSAgICAgICAgMHgwNEIyMDA5NywgMHgwNEIzMDE3RSwgMHgwNEI0MDE3RiwgMHgwNEI1MDE4MCwgMHgwNEI2MDE4MSwgMHgwNEI3MDE4MiwgMHgwNEI4MDA3OCwgMHgwNEJFMDA3MSwgCi0gICAgICAgIDB4MDRCRjAwNzgsIDB4MDRDMDAwODMsIDB4MDRDMTAwQTcsIDB4MDRDMjAwNzEsIDB4MDRDNjAwNzgsIDB4MDRDNzAwNzEsIDB4MDRDODAwNzgsIDB4MDRDOTAwNzEsIAotICAgICAgICAweDA0RDQwMDc4LCAweDA0RDUwMDcxLCAweDA0RDgwMDc4LCAweDA0REIwMDcxLCAweDA0REQwMDc4LCAweDA0REUwMDcxLCAweDA0REYwMEE3LCAweDA0RTAwMDgzLCAKLSAgICAgICAgMHgwNEUyMDA3OCwgMHgwNEUzMDBBNywgMHgwNEU0MDA3OCwgMHgwNEU1MDhBNywgMHgwNEU2MDA4MywgMHgwNEU3MDA3OCwgMHgwNEVCMDBBNywgMHgwNEVDMDA3OCwgCi0gICAgICAgIDB4MDRFRTA4NzEsIDB4MDRGMDAwNzEsIDB4MDRGMTAwODMsIDB4MDRGMjAwNzgsIDB4MDRGMzAxN0UsIDB4MDRGNDAxN0YsIDB4MDRGNTAxODAsIDB4MDRGNjAxODEsIAotICAgICAgICAweDA0RjcwMTgyLCAweDA0RjgwMDcxLCAweDA0RjkwMDA4LCAweDA0RkEwMEI2LCAweDA0RkIwMEI3LCAweDA0RkMwMTgzLCAweDA0RkQwMDc4LCAweDA1MDAwMDgzLCAKLSAgICAgICAgMHgwNTAxMDBBNywgMHgwNTAyMDA3MSwgMHgwNTA1MDA3OCwgMHgwNTA3MDA3MSwgMHgwNTA4MDA3OCwgMHgwNTA5MDA3MSwgMHgwNTE0MDA3OCwgMHgwNTE1MDA3MSwgCi0gICAgICAgIDB4MDUxODAwNzgsIDB4MDUxOTA4NzEsIDB4MDUxQTAwNzEsIDB4MDUxQjAwNzgsIDB4MDUxQzAwNzEsIDB4MDUxRDAwNzgsIDB4MDUxRjAwQTcsIDB4MDUyMDAwODMsIAotICAgICAgICAweDA1MjEwMDc4LCAweDA1MjMwMDgzLCAweDA1MjQwMDc4LCAweDA1MjUwMDgzLCAweDA1MjcwMDc4LCAweDA1MkMwODcxLCAweDA1MkUwMDc4LCAweDA1MzMwMTdFLCAKLSAgICAgICAgMHgwNTM0MDE3RiwgMHgwNTM1MDE4MCwgMHgwNTM2MDE4MSwgMHgwNTM3MDE4MiwgMHgwNTM4MDA4MywgMHgwNTM5MDA3MSwgMHgwNTNBMDA3OCwgMHgwNTQwMDA4MywgCi0gICAgICAgIDB4MDU0MTAwQTcsIDB4MDU0MjAwNzEsIDB4MDU1NDAwNzgsIDB4MDU1NTAwNzEsIDB4MDU1ODAwNzgsIDB4MDU1OTAwNzEsIDB4MDU1RDAwNzgsIDB4MDU1RTAwNzEsIAotICAgICAgICAweDA1NUYwMEE3LCAweDA1NjAwMDgzLCAweDA1NjQwMEE3LCAweDA1NjYwMDgzLCAweDA1NjcwMDc4LCAweDA1NzAwMDcxLCAweDA1NzEwMDgzLCAweDA1NzIwMDc4LCAKLSAgICAgICAgMHgwNTczMDE3RSwgMHgwNTc0MDE3RiwgMHgwNTc1MDE4MCwgMHgwNTc2MDE4MSwgMHgwNTc3MDE4MiwgMHgwNTc4MDAwOCwgMHgwNTc5MDA3OCwgMHgwNTgwMDA4MywgCi0gICAgICAgIDB4MDU4MTAwQTcsIDB4MDU4MjAwNzEsIDB4MDU4NjAwNzgsIDB4MDU4NzAwNzEsIDB4MDU4ODAwNzgsIDB4MDU4OTAwNzEsIDB4MDU5NDAwNzgsIDB4MDU5NTAwNzEsIAotICAgICAgICAweDA1OTgwMDc4LCAweDA1OTkwMDcxLCAweDA1OUQwMDc4LCAweDA1OUUwMDcxLCAweDA1OUYwMDgzLCAweDA1QTIwMDc4LCAweDA1QTMwMEE3LCAweDA1QTQwMDc4LCAKLSAgICAgICAgMHgwNUE1MDhBNywgMHgwNUE2MDA4MywgMHgwNUE3MDA3OCwgMHgwNUFCMDBBNywgMHgwNUFDMDA3OCwgMHgwNUFFMDg3MSwgMHgwNUFGMDA3MSwgMHgwNUIxMDA3OCwgCi0gICAgICAgIDB4MDVCMzAxN0UsIDB4MDVCNDAxN0YsIDB4MDVCNTAxODAsIDB4MDVCNjAxODEsIDB4MDVCNzAxODIsIDB4MDVCODAwNzEsIDB4MDVCOTAwNzgsIDB4MDVDMTAwNzEsIAotICAgICAgICAweDA1QzUwMDc4LCAweDA1QzcwMDcxLCAweDA1QzgwMDc4LCAweDA1QzkwMDcxLCAweDA1Q0IwMDc4LCAweDA1Q0MwMDcxLCAweDA1Q0QwMDc4LCAweDA1Q0YwMDcxLCAKLSAgICAgICAgMHgwNUQwMDA3OCwgMHgwNUQxMDA3MSwgMHgwNUQyMDA3OCwgMHgwNUQ0MDA3MSwgMHgwNUQ1MDA3OCwgMHgwNUQ3MDA3MSwgMHgwNUREMDA3OCwgMHgwNURGMDBBNywgCi0gICAgICAgIDB4MDVFMTAwNzgsIDB4MDVFMzAwQTcsIDB4MDVFNDAwNzgsIDB4MDVFNTA4QTcsIDB4MDVFNjAwODMsIDB4MDVFNzAwNzgsIDB4MDVFQjAwQTcsIDB4MDVFQzAwNzgsIAotICAgICAgICAweDA1RjMwMTdFLCAweDA1RjQwMTdGLCAweDA1RjUwMTgwLCAweDA1RjYwMTgxLCAweDA1RjcwMTgyLCAweDA1RjgwMTg0LCAweDA1RjkwMDU0LCAweDA1RkMwMDA4LCAKLSAgICAgICAgMHgwNUZEMDA3OCwgMHgwNjAwMDBBNywgMHgwNjAyMDA3MSwgMHgwNjA2MDA3OCwgMHgwNjA3MDA3MSwgMHgwNjA4MDA3OCwgMHgwNjA5MDA3MSwgMHgwNjE0MDA3OCwgCi0gICAgICAgIDB4MDYxNTAwNzEsIDB4MDYxRDAwNzgsIDB4MDYxRjAwODMsIDB4MDYyMDAwQTcsIDB4MDYyMjAwNzgsIDB4MDYyMzAwODMsIDB4MDYyNDAwNzgsIDB4MDYyNTAwODMsIAotICAgICAgICAweDA2MjcwMDc4LCAweDA2MkEwMDgzLCAweDA2MkIwMDc4LCAweDA2MzAwMDcxLCAweDA2MzEwMDc4LCAweDA2MzMwMTdFLCAweDA2MzQwMTdGLCAweDA2MzUwMTgwLCAKLSAgICAgICAgMHgwNjM2MDE4MSwgMHgwNjM3MDE4MiwgMHgwNjM4MDA3OCwgMHgwNjQxMDBBNywgMHgwNjQyMDA3MSwgMHgwNjQ2MDA3OCwgMHgwNjQ3MDA3MSwgMHgwNjQ4MDA3OCwgCi0gICAgICAgIDB4MDY0OTAwNzEsIDB4MDY1NDAwNzgsIDB4MDY1NTAwNzEsIDB4MDY1RDAwNzgsIDB4MDY1RTAwNzEsIDB4MDY1RjAwQjIsIDB4MDY2MDAwQTcsIDB4MDY2MjAwNzgsIAotICAgICAgICAweDA2NjMwOEE3LCAweDA2NjQwMDc4LCAweDA2NjUwOEE3LCAweDA2NjYwMDgzLCAweDA2NjcwMDc4LCAweDA2NkEwMEE3LCAweDA2NkIwMDc4LCAweDA2NzAwMDcxLCAKLSAgICAgICAgMHgwNjcxMDA3OCwgMHgwNjczMDE3RSwgMHgwNjc0MDE3RiwgMHgwNjc1MDE4MCwgMHgwNjc2MDE4MSwgMHgwNjc3MDE4MiwgMHgwNjc4MDA3OCwgMHgwNjgxMDBBNywgCi0gICAgICAgIDB4MDY4MjAwNzEsIDB4MDY4NjAwNzgsIDB4MDY4NzAwNzEsIDB4MDY4ODAwNzgsIDB4MDY4OTAwNzEsIDB4MDY5NDAwNzgsIDB4MDY5NTAwNzEsIDB4MDY5RDAwNzgsIAotICAgICAgICAweDA2OUYwMEE3LCAweDA2QTAwMDgzLCAweDA2QTIwMDc4LCAweDA2QTMwMEE3LCAweDA2QTQwMDc4LCAweDA2QTUwOEE3LCAweDA2QTYwMDgzLCAweDA2QTcwMDc4LCAKLSAgICAgICAgMHgwNkFCMDBBNywgMHgwNkFDMDA3OCwgMHgwNkIwMDA3MSwgMHgwNkIxMDA3OCwgMHgwNkIzMDE3RSwgMHgwNkI0MDE3RiwgMHgwNkI1MDE4MCwgMHgwNkI2MDE4MSwgCi0gICAgICAgIDB4MDZCNzAxODIsIDB4MDZCODAwNzgsIDB4MDZDMTAwQTcsIDB4MDZDMjAwNzEsIDB4MDZDQjAwNzgsIDB4MDZDRDAwNzEsIDB4MDZERjAwNzgsIDB4MDZFMDAwNzEsIAotICAgICAgICAweDA2RTMwMDc4LCAweDA2RTcwMEE3LCAweDA2RTkwMDgzLCAweDA2RUEwMDc4LCAweDA2RUMwMEE3LCAweDA2RUUwOEE3LCAweDA2RUYwMEE3LCAweDA2RjAwMDc4LCAKLSAgICAgICAgMHgwNkY5MDBBNywgMHgwNkZBMDA3OCwgMHgwNzAwMDA3MSwgMHgwNzE4MDA4MywgMHgwNzE5MTA3MSwgMHgwNzFBMDA4MywgMHgwNzFEMDA3OCwgMHgwNzFGMDAwOCwgCi0gICAgICAgIDB4MDcyMDAwNzEsIDB4MDcyMzAwODMsIDB4MDcyNzAwOTcsIDB4MDcyODAxN0UsIDB4MDcyOTAxN0YsIDB4MDcyQTAxODAsIDB4MDcyQjAxODEsIDB4MDcyQzAxODIsIAotICAgICAgICAweDA3MkQwMDk3LCAweDA3MkUwMDc4LCAweDA3NDAwMDcxLCAweDA3NDEwMDc4LCAweDA3NDMwMDcxLCAweDA3NDQwMDc4LCAweDA3NDYwMDcxLCAweDA3NDcwMDc4LCAKLSAgICAgICAgMHgwNzRBMDA3MSwgMHgwNzU0MDA3OCwgMHgwNzU1MDA3MSwgMHgwNzU4MDA4MywgMHgwNzU5MTA3MSwgMHgwNzVBMDA4MywgMHgwNzVFMDA3MSwgMHgwNzVGMDA3OCwgCi0gICAgICAgIDB4MDc2MDAwNzEsIDB4MDc2MjAwNzgsIDB4MDc2NDAwODMsIDB4MDc2NzAwNzgsIDB4MDc2ODAxN0UsIDB4MDc2OTAxN0YsIDB4MDc2QTAxODAsIDB4MDc2QjAxODEsIAotICAgICAgICAweDA3NkMwMTgyLCAweDA3NkQwMDc4LCAweDA3NkUxMDcxLCAweDA3NkYwMDc4LCAweDA3ODAwMDk0LCAweDA3ODIwMDk3LCAweDA3ODkwMDk0LCAweDA3OEMwMDgzLCAKLSAgICAgICAgMHgwNzhEMDA5NCwgMHgwNzkwMDE3RSwgMHgwNzkxMDE3RiwgMHgwNzkyMDE4MCwgMHgwNzkzMDE4MSwgMHgwNzk0MDE4MiwgMHgwNzk1MDBCMywgMHgwNzlBMDA4MywgCi0gICAgICAgIDB4MDc5RDAwQkYsIDB4MDc5RjAwQTcsIDB4MDdBMDAwNzEsIDB4MDdBMTA4NzEsIDB4MDdBMjAwNzEsIDB4MDdBNjA4NzEsIDB4MDdBNzAwNzEsIDB4MDdBQjA4NzEsIAotICAgICAgICAweDA3QUMwMDcxLCAweDA3QjQwODcxLCAweDA3QjUwMDc4LCAweDA3QjgwMDgzLCAweDA3QjkwODgzLCAweDA3QkIxMDgzLCAweDA3QkQwMDgzLCAweDA3QkYwMEE3LCAKLSAgICAgICAgMHgwN0MwMDg4MywgMHgwN0MxMDA4MywgMHgwN0MyMDA5NywgMHgwN0MzMDA4MywgMHgwN0M0MDA3MSwgMHgwN0M2MDA3OCwgMHgwN0M4MDA4MywgMHgwN0M5MDg4MywgCi0gICAgICAgIDB4MDdDQTAwODMsIDB4MDdDRTA4ODMsIDB4MDdDRjAwODMsIDB4MDdEMzA4ODMsIDB4MDdENDAwODMsIDB4MDdEQzA4ODMsIDB4MDdERDAwODMsIDB4MDdERTAwNzgsIAotICAgICAgICAweDA3REYwMDk0LCAweDA3RTYwMDc4LCAweDA3RTcwMDk0LCAweDA3RTgwMDk3LCAweDA3RTkwMDc4LCAweDA4MDAwMDcxLCAweDA4MTUwMDc4LCAweDA4MTYwMDgzLCAKLSAgICAgICAgMHgwODE4MDBBNywgMHgwODE5MDA3OCwgMHgwODFCMDA4MywgMHgwODFEMDA3OCwgMHgwODIwMDE3RSwgMHgwODIxMDE3RiwgMHgwODIyMDE4MCwgMHgwODIzMDE4MSwgCi0gICAgICAgIDB4MDgyNDAxODIsIDB4MDgyNTAwOTcsIDB4MDgyODAwNzEsIDB4MDgyQjAwQTcsIDB4MDgyQzAwODMsIDB4MDgyRDAwNzgsIDB4MDg1MDAwQjUsIDB4MDg2MzAwNzgsIAotICAgICAgICAweDA4NjgwMDcxLCAweDA4N0QwMDk3LCAweDA4N0UwMDc4LCAweDA4ODAwMDcxLCAweDA4QUQwMDc4LCAweDA4QUYwMDcxLCAweDA4RDEwMDc4LCAweDA4RDQwMDcxLCAKLSAgICAgICAgMHgwOEZEMDA3OCwgMHgwOTAwMDA3MSwgMHgwOTI0MDA3OCwgMHgwOTI1MDA3MSwgMHgwOTI3MDA3OCwgMHgwOTI4MDA3MSwgMHgwOTJCMDA3OCwgMHgwOTJEMDA3MSwgCi0gICAgICAgIDB4MDkyRjAwNzgsIDB4MDkzMDAwNzEsIDB4MDk0NDAwNzgsIDB4MDk0NTAwNzEsIDB4MDk0NzAwNzgsIDB4MDk0ODAwNzEsIDB4MDk1ODAwNzgsIDB4MDk1OTAwNzEsIAotICAgICAgICAweDA5NUIwMDc4LCAweDA5NUMwMDcxLCAweDA5NUYwMDc4LCAweDA5NjEwMDcxLCAweDA5NjMwMDc4LCAweDA5NjQwMDcxLCAweDA5NkIwMDc4LCAweDA5NkMwMDcxLCAKLSAgICAgICAgMHgwOTg4MDA3OCwgMHgwOTg5MDA3MSwgMHgwOThCMDA3OCwgMHgwOThDMDA3MSwgMHgwOUFEMDA3OCwgMHgwOUFGMDA4MywgMHgwOUIwMDA5NywgMHgwOUI0MDBBRCwgCi0gICAgICAgIDB4MDlCNTAwQUUsIDB4MDlCNjAxMkQsIDB4MDlCNzAxMkUsIDB4MDlCODAxMkYsIDB4MDlCOTAxODUsIDB4MDlCQTAxODYsIDB4MDlCQjAxODcsIDB4MDlCQzAxODgsIAotICAgICAgICAweDA5QkQwMTg0LCAweDA5QkUwMDc4LCAweDA5QzAwMDcxLCAweDA5QzgwMDU0LCAweDA5Q0QwMDc4LCAweDA5RDAwMDcxLCAweDA5RkEwMDc4LCAweDBBMDAwMDcxLCAKLSAgICAgICAgMHgwQjM2MDA5NywgMHgwQjM3MDA3MSwgMHgwQjNCMDA3OCwgMHgwQjQwMDA3MSwgMHgwQjREMDBCNCwgMHgwQjRFMDA3OCwgMHgwQjUwMDA3MSwgMHgwQjc1MDA5NywgCi0gICAgICAgIDB4MEI3NzAxODksIDB4MEI3ODAwNzgsIDB4MEI4MDAwNzEsIDB4MEI4NjAwNzgsIDB4MEI4NzAwNzEsIDB4MEI4OTAwODMsIDB4MEI4QTAwNzgsIDB4MEI5MDAwNzEsIAotICAgICAgICAweDBCOTkwMDgzLCAweDBCOUEwMDk3LCAweDBCOUIwMDc4LCAweDBCQTAwMDcxLCAweDBCQTkwMDgzLCAweDBCQUEwMDc4LCAweDBCQjAwMDcxLCAweDBCQjYwMDc4LCAKLSAgICAgICAgMHgwQkI3MDA3MSwgMHgwQkI4MDA3OCwgMHgwQkI5MDA4MywgMHgwQkJBMDA3OCwgMHgwQkMwMDA3MSwgMHgwQkRBMDBDMiwgMHgwQkRCMDA4MywgMHgwQkRGMDBBNywgCi0gICAgICAgIDB4MEJFNDAwODMsIDB4MEJFQTAwOTcsIDB4MEJFQjAwODEsIDB4MEJFQzAwOTcsIDB4MEJFRDAwMDgsIDB4MEJFRTAwODMsIDB4MEJFRjAwNzgsIDB4MEJGMDAxN0UsIAotICAgICAgICAweDBCRjEwMTdGLCAweDBCRjIwMTgwLCAweDBCRjMwMTgxLCAweDBCRjQwMTgyLCAweDBCRjUwMDc4LCAweDBCRjgwMTA2LCAweDBCRjkwMTA3LCAweDBCRkEwMTA4LCAKLSAgICAgICAgMHgwQkZCMDEwOSwgMHgwQkZDMDEwQSwgMHgwQkZEMDA3OCwgMHgwQzAwMDAwNiwgMHgwQzA1MDA4MywgMHgwQzA3MDA3OCwgMHgwQzA4MDE3RSwgMHgwQzA5MDE3RiwgCi0gICAgICAgIDB4MEMwQTAxODAsIDB4MEMwQjAxODEsIDB4MEMwQzAxODIsIDB4MEMwRDAwNzgsIDB4MEMxMDAwNzEsIDB4MEMyMTAwODEsIDB4MEMyMjAwNzEsIDB4MEMzQzAwNzgsIAotICAgICAgICAweDBDNDAwMDcxLCAweDBDNTQwMDgzLCAweDBDNTUwMDc4LCAweDBDODAwMDcxLCAweDBDOEUwMDc4LCAweDBDOTAwMDgzLCAweDBDOTEwMEE3LCAweDBDOTMwMDgzLCAKLSAgICAgICAgMHgwQzk0MDBDOCwgMHgwQzk2MDA3OCwgMHgwQzk4MDBBNywgMHgwQzlDMDA4MywgMHgwQzlFMDA3OCwgMHgwQ0EyMDAwNiwgMHgwQ0EzMDE3RSwgMHgwQ0E0MDE3RiwgCi0gICAgICAgIDB4MENBNTAxODAsIDB4MENBNjAxODEsIDB4MENBNzAxODIsIDB4MENBODAwNzEsIDB4MENCNzAwNzgsIDB4MENCODAwNzEsIDB4MENCQTAwNzgsIDB4MENDMDAwNzEsIAotICAgICAgICAweDBDRDUwMDc4LCAweDBDRDgwMEE3LCAweDBDRTAwMDcxLCAweDBDRTQwMEE3LCAweDBDRTUwMDc4LCAweDBDRTgwMTdFLCAweDBDRTkwMTdGLCAweDBDRUEwMTgwLCAKLSAgICAgICAgMHgwQ0VCMDE4MSwgMHgwQ0VDMDE4MiwgMHgwQ0VEMDA3OCwgMHgwQ0VGMDAwNiwgMHgwQ0YwMDA1NCwgMHgwRDAwMDA3MSwgMHgwRDBCMDA4MywgMHgwRDBDMDBBNywgCi0gICAgICAgIDB4MEQwRTAwNzgsIDB4MEQwRjAwOTcsIDB4MEQxMDAwNzgsIDB4MEU4MDAwNTUsIDB4MEU5Njc4ODEsIDB4MEU5NzAwODEsIDB4MEU5ODc4ODEsIDB4MEU5RDAwODEsIAotICAgICAgICAweDBFOUU3ODgxLCAweDBFQjE3MDU1LCAweDBFQjUwMDU1LCAweDBFQ0Q3ODgxLCAweDBFRTAwMDgzLCAweDBFRTIwMDc4LCAweDBGMDAwODY1LCAweDBGNEIwODU1LCAKLSAgICAgICAgMHgwRjREMDk4QSwgMHgwRjRFMDA3OCwgMHgwRjUwMDg2NSwgMHgwRjdEMDA3OCwgMHgwRjgwMDhDOSwgMHgwRjg0MDhDQSwgMHgwRjg4MDhDOSwgMHgwRjhCMDA3OCwgCi0gICAgICAgIDB4MEY4QzA4Q0EsIDB4MEY4RjAwNzgsIDB4MEY5MDA4QzksIDB4MEY5NDA4Q0EsIDB4MEY5ODA4QzksIDB4MEY5QzA4Q0EsIDB4MEZBMDA4QzksIDB4MEZBMzAwNzgsIAotICAgICAgICAweDBGQTQwOENBLCAweDBGQTcwMDc4LCAweDBGQTgwOEM5LCAweDBGQUMwOENBLCAweDBGQjAwOEM5LCAweDBGQjQwOENBLCAweDBGQjgwOENCLCAweDBGQjkwOENDLCAKLSAgICAgICAgMHgwRkJCMDhDRCwgMHgwRkJDMDhDRSwgMHgwRkJEMDhDRiwgMHgwRkJFMDhEMCwgMHgwRkJGMDA3OCwgMHgwRkMwMDhDOSwgMHgwRkM0MDhEMSwgMHgwRkM4MDhDOSwgCi0gICAgICAgIDB4MEZDQzA4RDEsIDB4MEZEMDA4QzksIDB4MEZENDA4RDEsIDB4MEZEODA4QzksIDB4MEZEOTA5OEIsIDB4MEZEQTAwNzgsIDB4MEZEQjA4NTUsIDB4MEZEQzA4Q0EsIAotICAgICAgICAweDBGREQwOEQyLCAweDBGREUxMDM3LCAweDBGRTAwODM3LCAweDBGRTEwOThCLCAweDBGRTIwMDc4LCAweDBGRTMwODU1LCAweDBGRTQwOEQ1LCAweDBGRTYwODM3LCAKLSAgICAgICAgMHgwRkU4MDhDOSwgMHgwRkU5MDg1NSwgMHgwRkVBMDA3OCwgMHgwRkVCMDg1NSwgMHgwRkVDMDhDQSwgMHgwRkVEMDhENiwgMHgwRkVFMDgzNywgMHgwRkYwMDhDOSwgCi0gICAgICAgIDB4MEZGMTA4NTUsIDB4MEZGMjA4OTAsIDB4MEZGMzA4NTUsIDB4MEZGNDA4Q0EsIDB4MEZGNTA4RDcsIDB4MEZGNjA4MzcsIDB4MEZGODAwNzgsIDB4MEZGOTA5OEIsIAotICAgICAgICAweDBGRkEwMDc4LCAweDBGRkIwODU1LCAweDBGRkMwOEQ5LCAweDBGRkQwOERBLCAweDBGRkUwODM3LCAweDBGRkYwMDc4LCAweDEwMDAwODA1LCAweDEwMDExMDA1LCAKLSAgICAgICAgMHgxMDAzNTgwNSwgMHgxMDA0MTAwNSwgMHgxMDA1MDA1NywgMHgxMDA3MDE4QywgMHgxMDA4NTg5OSwgMHgxMDA5MDA5OSwgMHgxMDBCMTAwNiwgMHgxMDBDMDE4RCwgCi0gICAgICAgIDB4MTAwRDAwREIsIDB4MTAwRTAxOEQsIDB4MTAwRjAwREIsIDB4MTAxMDAwMDYsIDB4MTAxMjEwMDYsIDB4MTAxMzAwMDYsIDB4MTAxNDAxOEUsIDB4MTAxNTAxOEYsIAotICAgICAgICAweDEwMTYwMTkwLCAweDEwMTc1ODUzLCAweDEwMTgwMDA3LCAweDEwMTkxMDA3LCAweDEwMUEwMDA2LCAweDEwMUIxMDA2LCAweDEwMUMwMTI2LCAweDEwMUQwMDA2LCAKLSAgICAgICAgMHgxMDFGMDAzOCwgMHgxMDIwMDAwNiwgMHgxMDIyMDAwOSwgMHgxMDIzMTAwNiwgMHgxMDI1MDAwNiwgMHgxMDJCMTAwNiwgMHgxMDJDMDAwNiwgMHgxMDJGMTAwNSwgCi0gICAgICAgIDB4MTAzMDAwNTcsIDB4MTAzMjAwNzgsIDB4MTAzNTAwNTcsIDB4MTAzODc4NTUsIDB4MTAzOTAwNzgsIDB4MTAzQTc5MTAsIDB4MTAzQjc5MTEsIDB4MTAzQzc5MTIsIAotICAgICAgICAweDEwM0Q3ODBCLCAweDEwM0U3ODA5LCAweDEwM0Y3ODU1LCAweDEwNDA3MDVELCAweDEwNDE3MDVCLCAweDEwNDI3MTEwLCAweDEwNDM3MTExLCAweDEwNDQ3MTEyLCAKLSAgICAgICAgMHgxMDQ1NzAwQiwgMHgxMDQ2NzAwOSwgMHgxMDQ3MDA3OCwgMHgxMDQ4NzA4MSwgMHgxMDRBMDA3OCwgMHgxMDUwMDAwOCwgMHgxMDVCMDA3OCwgMHgxMDY4MDA4MywgCi0gICAgICAgIDB4MTA2RTAwOTUsIDB4MTA3MDAwODMsIDB4MTA3MTAwOTUsIDB4MTA3MjAwODMsIDB4MTA3NjAwNzgsIDB4MTA4MDEwNTQsIDB4MTA4MzEwNzcsIDB4MTA4NDEwNTQsIAotICAgICAgICAweDEwODUyODc3LCAweDEwODcyODU1LCAweDEwODgyODc3LCAweDEwODkyODU1LCAweDEwOEEyODc3LCAweDEwOEIwMDU0LCAweDEwOEMyODc3LCAweDEwOEYwMDU0LCAKLSAgICAgICAgMHgxMDkwMTA1NCwgMHgxMDkxMDA1NCwgMHgxMDk1MDk5MSwgMHgxMDk2Mjg3NywgMHgxMDk3Mjg1NSwgMHgxMDk4Mjg3NywgMHgxMDlBMTA3MSwgMHgxMDlDMjg1NSwgCi0gICAgICAgIDB4MTA5RDEwNTQsIDB4MTA5RTI4NTUsIDB4MTA5RjI4NzcsIDB4MTBBMDAwMTksIDB4MTBBMjI4NzcsIDB4MTBBMzI4NTUsIDB4MTBBNTAwMTksIDB4MTBBNjAwNzgsIAotICAgICAgICAweDEwQTkzMDVGLCAweDEwQUYzMTA2LCAweDEwQjAxMTkyLCAweDEwQjExMTkzLCAweDEwQjIxMTk0LCAweDEwQjMxMTk1LCAweDEwQjQxMTk2LCAweDEwQjUxMTk3LCAKLSAgICAgICAgMHgxMEI2MTE5OCwgMHgxMEI3MTE5OSwgMHgxMEI4MTE5QSwgMHgxMEI5MTE5QiwgMHgxMEJBMTE5QywgMHgxMEJCMTE5RCwgMHgxMEJDMTE5RSwgMHgxMEJEMTE5RiwgCi0gICAgICAgIDB4MTBCRTExQTAsIDB4MTBCRjExQTEsIDB4MTBDMDAxQTIsIDB4MTBDMTAxQTMsIDB4MTBDMjAwNzgsIDB4MTBDODAwMTksIDB4MTBDQTAwNTQsIDB4MTBDRDA4MTksIAotICAgICAgICAweDEwQ0UwMDU0LCAweDEwRDEwMDE5LCAweDEwRDIwMDU0LCAweDEwRTYwODU0LCAweDEwRTcwODE5LCAweDEwRTgwMDU0LCAweDEwRkEwMDE5LCAweDExMDAwMEU4LCAKLSAgICAgICAgMHgxMTAyMDAxOSwgMHgxMTA0MDhGQiwgMHgxMTA1MDBGQywgMHgxMTA3MDAxOSwgMHgxMTA4MDBFOCwgMHgxMTA5MDA1OSwgMHgxMTBBMDFBNCwgMHgxMTBCMDAxOSwgCi0gICAgICAgIDB4MTEwRDAwRTgsIDB4MTExMTAwMTksIDB4MTExNTAwRTgsIDB4MTExNjEwRTgsIDB4MTExODAwRTgsIDB4MTExQTAwMTksIDB4MTExQzAwRTgsIDB4MTExRTAwRkUsIAotICAgICAgICAweDExMUYwMEU4LCAweDExMjAwOEU4LCAweDExMjEwMUE1LCAweDExMjIwMEU4LCAweDExMjMwOEU4LCAweDExMjUwMEU4LCAweDExMjYwMDE5LCAweDExMjkwMEZFLCAKLSAgICAgICAgMHgxMTJCMDAxOSwgMHgxMTJGMDBFOCwgMHgxMTMwMDAxOSwgMHgxMTMyMDBGRSwgMHgxMTM2MDgxOSwgMHgxMTM3MDhGRSwgMHgxMTM5MDBGRSwgMHgxMTNBMDhGRSwgCi0gICAgICAgIDB4MTEzQjAwRkUsIDB4MTEzQzA4RkUsIDB4MTEzRDAwRkUsIDB4MTE0MDA4RkUsIDB4MTE0MTAwRkUsIDB4MTE0MjA4RkUsIDB4MTE0MzAwRkUsIDB4MTE0NDA4RkUsIAotICAgICAgICAweDExNDUwMEZFLCAweDExNDYwMDE5LCAweDExNDcwMEZELCAweDExNDkwMDE5LCAweDExNTEwMEZFLCAweDExNTIwMDE5LCAweDExNTMwMEU4LCAweDExNTQwMUE2LCAKLSAgICAgICAgMHgxMTU2MDhFOCwgMHgxMTU4MDBGRSwgMHgxMTVDMDAxOSwgMHgxMTVGMDBFOCwgMHgxMTYwMDAxOSwgMHgxMTY0MDBGRCwgMHgxMTY2MDFBNywgMHgxMTY3MDAxOSwgCi0gICAgICAgIDB4MTE2ODAwRkUsIDB4MTE2OTAwMTksIDB4MTE2QjAwRkUsIDB4MTE3MDA4RkUsIDB4MTE3MjAwRkUsIDB4MTE3NTA4RkUsIDB4MTE3NzAwMTksIDB4MTE3ODAwRkUsIAotICAgICAgICAweDExNzkwMTAyLCAweDExN0EwMEU4LCAweDExN0IwMTAzLCAweDExN0MwMEU4LCAweDExN0QwMTA0LCAweDExN0UwMTA1LCAweDExN0YwMEU4LCAweDExODAwMDU0LCAKLSAgICAgICAgMHgxMTg0MDBGRSwgMHgxMTg2MDA1NCwgMHgxMTkwMDBFOCwgMHgxMTkxMDA1NCwgMHgxMTk0MDgwOSwgMHgxMTk1MDA1NCwgMHgxMTlCMDA5NCwgMHgxMUJEMDA1NCwgCi0gICAgICAgIDB4MTFDQTAwOTQsIDB4MTFDQjAwNTQsIDB4MTFDRDAwMTksIDB4MTFEQTAwQkYsIDB4MTFEQjAwNTQsIDB4MTFFRTAwNzgsIDB4MTIwMDAwNTQsIDB4MTIxMzAwNzgsIAotICAgICAgICAweDEyMjAwMDU0LCAweDEyMjUwMDc4LCAweDEyMzAxOEM0LCAweDEyMzExOEM1LCAweDEyMzIxOEM2LCAweDEyMzMxOEM3LCAweDEyMzQxOTFGLCAweDEyMzUxOTFBLCAKLSAgICAgICAgMHgxMjM2MTkxQiwgMHgxMjM3MTkxQywgMHgxMjM4MTkxRCwgMHgxMjM5MTkxRSwgMHgxMjNBMTBDNCwgMHgxMjNCMTBDNSwgMHgxMjNDMTBDNiwgMHgxMjNEMTBDNywgCi0gICAgICAgIDB4MTIzRTExMUYsIDB4MTIzRjExMUEsIDB4MTI0MDExMUIsIDB4MTI0MTExMUMsIDB4MTI0MjExMUQsIDB4MTI0MzExMUUsIDB4MTI0NDEwNUEsIDB4MTI0NTEwRTMsIAotICAgICAgICAweDEyNDYxMEU0LCAweDEyNDcxMEU1LCAweDEyNDgxMUE4LCAweDEyNDkxMUE5LCAweDEyNEExMUFBLCAweDEyNEIxMUFCLCAweDEyNEMxMUFDLCAweDEyNEQxMUFELCAKLSAgICAgICAgMHgxMjRFMTA5NCwgMHgxMjVCMTkxOCwgMHgxMjY4MTkxOSwgMHgxMjc1MDEwQiwgMHgxMjc2MDEwQywgMHgxMjc3MDEwRCwgMHgxMjc4MDEwRSwgMHgxMjc5MDEwRiwgCi0gICAgICAgIDB4MTI3QTAxMDYsIDB4MTI3QjAxMDcsIDB4MTI3QzAxMDgsIDB4MTI3RDAxMDksIDB4MTI3RTAxMEEsIDB4MTI3RjAwQzMsIDB4MTI4MDAwNTQsIDB4MTJEQjAwMTksIAotICAgICAgICAweDEyREMwMDU0LCAweDEyRTAwMDE5LCAweDEyRTEwMDU0LCAweDEyRkMwMDE5LCAweDEzMDAwMDU0LCAweDEzMzcwMDE5LCAweDEzMzgwMDU0LCAweDEzNEUwMDc4LCAKLSAgICAgICAgMHgxMzUwMDA1NCwgMHgxMzU5MDA3OCwgMHgxMzgwMDA1NCwgMHgxMzgyMDA3OCwgMHgxMzgzMDA1NCwgMHgxMzg1MDA3OCwgMHgxMzg2MDA1NCwgMHgxM0E5MDA3OCwgCi0gICAgICAgIDB4MTNBQzAwNTQsIDB4MTNBRjAwNzgsIDB4MTNCMDAwNTQsIDB4MTNCNDAwMEEsIDB4MTNCQjAwQzQsIDB4MTNCQzAwQzUsIDB4MTNCRDAwQzYsIDB4MTNCRTAwQzcsIAotICAgICAgICAweDEzQkYwMTFGLCAweDEzQzAwMEM0LCAweDEzQzEwMEM1LCAweDEzQzIwMEM2LCAweDEzQzMwMEM3LCAweDEzQzQwMTFGLCAweDEzQzUwMEM0LCAweDEzQzYwMEM1LCAKLSAgICAgICAgMHgxM0M3MDBDNiwgMHgxM0M4MDBDNywgMHgxM0M5MDExRiwgMHgxM0NBMDA3OCwgMHgxM0NDMDA1NCwgMHgxM0RGMDA3OCwgMHgxM0UwMDAxOSwgMHgxM0UxMDBGRCwgCi0gICAgICAgIDB4MTNFMjAwMDksIDB4MTNFMzAwNzgsIDB4MTNFODAwMTksIDB4MTNFOTAwRTgsIDB4MTNFQTAwRkQsIDB4MTNFQjAwMTksIDB4MTNFRTAwRkQsIDB4MTNFRjAwMTksIAotICAgICAgICAweDEzRjEwMEZFLCAweDEzRjMwMDBBLCAweDEzRjYwMDc4LCAweDEzRjgwMDE5LCAweDE0MDAwMDk0LCAweDE0ODAwMDE5LCAweDE0QzEwMDA5LCAweDE0QzYwMUFFLCAKLSAgICAgICAgMHgxNEM3MDFBRiwgMHgxNEM4MDAwOSwgMHgxNENDMDAxOSwgMHgxNENEMDBFOCwgMHgxNEQ4MDAxOSwgMHgxNEUwMDBGRSwgMHgxNEUxMDBFOCwgMHgxNEUyMDBGRSwgCi0gICAgICAgIDB4MTRFMzAwMTksIDB4MTRFNDAwRTgsIDB4MTRFNTAwMTksIDB4MTRFNzAwRkQsIDB4MTRFOTAwMTksIDB4MTRFQTAwRkUsIDB4MTRFQjAwMTksIDB4MTRFQzAwMEEsIAotICAgICAgICAweDE0RUUwMDE5LCAweDE0RjAwMEU4LCAweDE0RjMwMDE5LCAweDE0RjQwMEU4LCAweDE0RjUwMDE5LCAweDE0RkEwMUIwLCAweDE0RkIwMEU4LCAweDE0RkMwMEZFLCAKLSAgICAgICAgMHgxNEZEMDAxOSwgMHgxNEZFMDAwQSwgMHgxNEZGMDAxOSwgMHgxNTA1MDBFOCwgMHgxNTBFMDAxOSwgMHgxNTBGMDBFOCwgMHgxNTExMDAxOSwgMHgxNTE0MDBFOCwgCi0gICAgICAgIDB4MTUxNTAwRkQsIDB4MTUxNzAwMTksIDB4MTUxQTAwRkUsIDB4MTUxQjAwMTksIDB4MTUxRTAwRkUsIDB4MTUxRjAwMTksIDB4MTUyQjAwRTgsIDB4MTUyQzAwMTksIAotICAgICAgICAweDE1MzIwMEZFLCAweDE1MzMwMDE5LCAweDE1MzUwMEU4LCAweDE1MzgwMDE5LCAweDE1MzkwMEU4LCAweDE1M0ExMDE5LCAweDE1M0IwMDE5LCAweDE1M0MwMEZELCAKLSAgICAgICAgMHgxNTNEMDBFOCwgMHgxNTNFMDBGRCwgMHgxNTQyMDBFOCwgMHgxNTQ1MDBGRCwgMHgxNTQ2MDBFOCwgMHgxNTQ4MDBGRCwgMHgxNTRFMDBFOCwgMHgxNTUwMDBGRCwgCi0gICAgICAgIDB4MTU1MTAwRTgsIDB4MTU1MjAwMTksIDB4MTU1MzAwRkUsIDB4MTU1NzAwRkQsIDB4MTU1ODAwRTgsIDB4MTU1OTAwRkQsIDB4MTU1QTAwRTgsIDB4MTU1RDAwRkQsIAotICAgICAgICAweDE1NjMwMEU4LCAweDE1NjYwMEZELCAweDE1NkIwMDE5LCAweDE1NzEwMUIxLCAweDE1NzMwMDE5LCAweDE1NzYwMEZFLCAweDE1NzcwMDE5LCAweDE1NzkwMEU4LCAKLSAgICAgICAgMHgxNTdBMDAxOSwgMHgxNTdCMDBGRCwgMHgxNTdEMDBFOCwgMHgxNTdGMDAxOSwgMHgxNTgwMDA1NCwgMHgxNThBMDA3OCwgMHgxNjAwMDA5NiwgMHgxNjE3MDA3OCwgCi0gICAgICAgIDB4MTYxODAwOTgsIDB4MTYyRjAwNzgsIDB4MTY0MDAwNjUsIDB4MTY3MjAwNTQsIDB4MTY3NTAwNzgsIDB4MTY3QzAwMDYsIDB4MTY3RTAwNUYsIDB4MTY3RjAwMDYsIAotICAgICAgICAweDE2ODAwMTI1LCAweDE2OTMwMDc4LCAweDE2OTgwMDcxLCAweDE2QjMwMDc4LCAweDE2Qjc3ODgxLCAweDE2QjgwMDc4LCAweDE2QzAwMDcxLCAweDE2Q0IwMDc4LCAKLSAgICAgICAgMHgxNkQwMDA3MSwgMHgxNkQzMDA3OCwgMHgxNkQ0MDA3MSwgMHgxNkQ3MDA3OCwgMHgxNkQ4MDA3MSwgMHgxNkRCMDA3OCwgMHgxNkRDMDA3MSwgMHgxNkRGMDA3OCwgCi0gICAgICAgIDB4MTZFMDAwNzEsIDB4MTZFMzAwNzgsIDB4MTZFNDAwNzEsIDB4MTZFNzAwNzgsIDB4MTZFODAwNzEsIDB4MTZFQjAwNzgsIDB4MTZFQzAwNzEsIDB4MTZFRjAwNzgsIAotICAgICAgICAweDE3MDAwMDA2LCAweDE3MDEwMEUwLCAweDE3MDMwMDA2LCAweDE3MDQwMTI2LCAweDE3MDUwMDA2LCAweDE3MDYwMEUwLCAweDE3MDcwMDA2LCAweDE3MEIwMDk5LCAKLSAgICAgICAgMHgxNzBDMDA3OCwgMHgxNzBFMDBFMCwgMHgxNzBGMDA3OCwgMHgxNzQwMDA1NCwgMHgxNzRGMTA1NCwgMHgxNzUwMDA1NCwgMHgxNzc5MTA1NCwgMHgxNzdBMDA3OCwgCi0gICAgICAgIDB4MTc4MDEwNTQsIDB4MTdFQjAwNzgsIDB4MTdGODAwNTQsIDB4MTdGRTAwNzgsIDB4MTgwMDAwMDYsIDB4MTgwMjAwODEsIDB4MTgwMzAxQjIsIDB4MTgwNDAwMEEsIAotICAgICAgICAweDE4MDkwMDU0LCAweDE4MEEwMDBBLCAweDE4MEUwMEI0LCAweDE4MEYwMEJGLCAweDE4MTAwMUIzLCAweDE4MTEwMUI0LCAweDE4MTIwMUI1LCAweDE4MTMwMUI2LCAKLSAgICAgICAgMHgxODE0MDFCNywgMHgxODE1MDA4MywgMHgxODE4MDA4MSwgMHgxODFCMDA1NCwgMHgxODFDMTFCOCwgMHgxODFEMDA4MSwgMHgxODFFMDAwNiwgMHgxODFGMDA1NCwgCi0gICAgICAgIDB4MTgyMDAwNzEsIDB4MTgzMjA4NzEsIDB4MTgzNTAwNzEsIDB4MTgzODA4NzEsIDB4MTgzQTAwNzEsIDB4MTgzQjA4NzEsIDB4MTgzRDAwNzEsIDB4MTgzRTA4NzEsIAotICAgICAgICAweDE4M0YwMDcxLCAweDE4NEIwMDc4LCAweDE4NEMwMDgzLCAweDE4NEQxMDM3LCAweDE4NEUwMDgxLCAweDE4NEY4MDcxLCAweDE4NTAwMDcxLCAweDE4NjIwODcxLCAKLSAgICAgICAgMHgxODY1MDA3MSwgMHgxODY4MDg3MSwgMHgxODZBMDA3MSwgMHgxODZCMDg3MSwgMHgxODZEMDA3MSwgMHgxODZFMDg3MSwgMHgxODZGMDA3MSwgMHgxODdCMDg3MSwgCi0gICAgICAgIDB4MTg3RDAwMDYsIDB4MTg3RTAwODEsIDB4MTg3RjgwNzEsIDB4MTg4MDAwNzgsIDB4MTg4MjAwNzEsIDB4MTg5NjAwNzgsIDB4MTg5ODEwNzEsIDB4MThDNzAwNzgsIAotICAgICAgICAweDE4QzgwMDk0LCAweDE4Qzk3OEI2LCAweDE4Q0E3OEI3LCAweDE4Q0I3ODk0LCAweDE4RDAwMDcxLCAweDE4REMwMDc4LCAweDE4RTAwMDU0LCAweDE4RTgwMDc4LCAKLSAgICAgICAgMHgxOEY4MDA3MSwgMHgxOTAwMTA5NCwgMHgxOTBFMTA1NCwgMHgxOTBGMDA3OCwgMHgxOTEwMTBCNiwgMHgxOTExMTBCNywgMHgxOTEyMTBCOCwgMHgxOTEzMTBCOSwgCi0gICAgICAgIDB4MTkxNDEwQjAsIDB4MTkxNTEwOTQsIDB4MTkyMjAwNzgsIDB4MTkyODE5QjksIDB4MTkyOTE5QkEsIDB4MTkyQTE5QkIsIDB4MTkyQjE5QkMsIDB4MTkyQzE5QkQsIAotICAgICAgICAweDE5MkQxOUJFLCAweDE5MkUxOUJGLCAweDE5MkYxOUMwLCAweDE5MzAxODk0LCAweDE5M0UxODU0LCAweDE5M0YwMDk0LCAweDE5NDAxOEI2LCAweDE5NDExOEI3LCAKLSAgICAgICAgMHgxOTQyMThCOCwgMHgxOTQzMThCOSwgMHgxOTQ0MThCMCwgMHgxOTQ1MTg5NCwgMHgxOTU4MTlDMSwgMHgxOTU5MTlDMiwgMHgxOTVBMTlDMywgMHgxOTVCMTlDNCwgCi0gICAgICAgIDB4MTk1QzE5QzUsIDB4MTk1RDE5QzYsIDB4MTk1RTE5QzcsIDB4MTk1RjE5QzgsIDB4MTk2MDEwOTQsIDB4MTk2NjY4NTQsIDB4MTk2ODE4OTQsIDB4MTk3RjAwNzgsIAotICAgICAgICAweDE5ODA2ODk0LCAweDE5QUMxMDk0LCAweDE5Qjg2ODk0LCAweDE5QkI2ODU0LCAweDE5QkQ2ODk0LCAweDE5RUY2ODU0LCAweDE5RjAxMDk0LCAweDE5RkY2ODU0LCAKLSAgICAgICAgMHgxQTAwMDA3MSwgMHgyNkRCMDA3OCwgMHgyNkUwMDA1NCwgMHgyNzAwMDA3MSwgMHg0RkRFMDA3OCwgMHg1MDAwMDA3MSwgMHg1MDBBMDA4MSwgMHg1MDBCMDA3MSwgCi0gICAgICAgIDB4NTI0NjAwNzgsIDB4NTI0ODAwNTQsIDB4NTI2MzAwNzgsIDB4NTM4MDAwMzcsIDB4NTM4QjAwNzgsIDB4NTQwMDAwNzEsIDB4NTQwNTAwODMsIDB4NTQwNjAwNzEsIAotICAgICAgICAweDU0MTEwMEE3LCAweDU0MTIwMDgzLCAweDU0MTMwMEE3LCAweDU0MTQwMDU0LCAweDU0MTYwMDc4LCAweDU2MDAwMDcxLCAweDZCRDIwMDc4LCAweDZDMDAwMTNFLCAKLSAgICAgICAgMHg3MDAwMDEzRiwgMHg3QzgwMDg3MSwgMHg3RDA3MDA3MSwgMHg3RDBBMDg3MSwgMHg3RDBGMDA3MSwgMHg3RDEyMDg3MSwgMHg3RDEzMDA3MSwgMHg3RDE1MDg3MSwgCi0gICAgICAgIDB4N0QxNzAwNzgsIDB4N0QxODA4NzEsIDB4N0QzNTAwNzgsIDB4N0QzODA4NzEsIDB4N0Q2RDAwNzgsIDB4N0Q4MDEwNTUsIDB4N0Q4MzAwNzgsIDB4N0Q4OTEwNTUsIAotICAgICAgICAweDdEOEMwMDc4LCAweDdEOEUwODlCLCAweDdEOTAyODlCLCAweDdEOTQyODBCLCAweDdEOTUwODlCLCAweDdEOUIwMDc4LCAweDdEOUMwODlCLCAweDdEOUUwMDc4LCAKLSAgICAgICAgMHg3REEwMDg5QiwgMHg3REEyMDA3OCwgMHg3REEzMDg5QiwgMHg3REE3MTA5QiwgMHg3REE4MjA5RSwgMHg3REFBNDg5RSwgMHg3REFCMjA5RSwgMHg3REFDNDg5RSwgCi0gICAgICAgIDB4N0RBRDIwOUUsIDB4N0RBRTQ4OUUsIDB4N0RBRjIwOUUsIDB4N0RCMDQ4OUUsIDB4N0RCMTIwOUUsIDB4N0RCMjQ4OUUsIDB4N0RCMzIwOUUsIDB4N0RCNDQ4OUUsIAotICAgICAgICAweDdEQjUyMDlFLCAweDdEQjY0ODlFLCAweDdEQjcyMDlFLCAweDdEQjg0ODlFLCAweDdEQjkyMDlFLCAweDdEQkE0ODlFLCAweDdEQkIyMDlFLCAweDdEQkM0ODlFLCAKLSAgICAgICAgMHg3REJEMjA5RSwgMHg3REJFNDg5RSwgMHg3REJGMjA5RSwgMHg3REMwNDg5RSwgMHg3REMxMjA5RSwgMHg3REM4NDg5RSwgMHg3REM5MjA5RSwgMHg3RENBNDg5RSwgCi0gICAgICAgIDB4N0RDQjIwOUUsIDB4N0RDQzQ4OUUsIDB4N0RDRDIwOUUsIDB4N0RDRTQ4OUUsIDB4N0RDRjIwOUUsIDB4N0REMTQ4OUUsIDB4N0REMjIwOUUsIDB4N0RENDQ4OUUsIAotICAgICAgICAweDdERDUyMDlFLCAweDdERDY0ODlFLCAweDdERDcyMDlFLCAweDdERDkwMDc4LCAweDdERTk0MDlFLCAweDdERUEzODlFLCAweDdERUI0MDlFLCAweDdERUYyMDlFLCAKLSAgICAgICAgMHg3REYzNDg5RSwgMHg3REY1MjA5RSwgMHg3REZDNDA5RSwgMHg3REZEMzg5RSwgMHg3REZFMjA5RSwgMHg3REZGNDg5RSwgMHg3RTAwNDA5RSwgMHg3RTMyMjA5RSwgCi0gICAgICAgIDB4N0U0QjM4OUUsIDB4N0U2RjQ4OUUsIDB4N0U3QTQwOUUsIDB4N0U4ODIwOUUsIDB4N0U5NjM4OUUsIDB4N0U5QTQ4OUUsIDB4N0U5RTQwOUUsIDB4N0U5RjAwQkYsIAotICAgICAgICAweDdFQTAwMDc4LCAweDdFQTgyMDlFLCAweDdFQTkzODlFLCAweDdFQUQyMDlFLCAweDdFQUUzODlFLCAweDdFQUYyMDlFLCAweDdFQjAzODlFLCAweDdFQjMyMDlFLCAKLSAgICAgICAgMHg3RUI1Mzg5RSwgMHg3RUI3MjA5RSwgMHg3RUI5Mzg5RSwgMHg3RUJBMjA5RSwgMHg3RUJCMzg5RSwgMHg3RUJDMjA5RSwgMHg3RUJFMzg5RSwgMHg3RUJGMjA5RSwgCi0gICAgICAgIDB4N0VDMTM4OUUsIDB4N0VDMjIwOUUsIDB4N0VDNDM4OUUsIDB4N0VDNTIwOUUsIDB4N0VDNjM4OUUsIDB4N0VDODAwNzgsIDB4N0VDOTM4OUUsIDB4N0VDQjIwOUUsIAotICAgICAgICAweDdFQ0UzODlFLCAweDdFQ0YyMDlFLCAweDdFREEzODlFLCAweDdFREIyMDlFLCAweDdFRTEzODlFLCAweDdFRTMyMDlFLCAweDdFRTQwMDc4LCAweDdFRjg0MDlFLCAKLSAgICAgICAgMHg3RUZFMDA1NCwgMHg3RUZGMDA3OCwgMHg3RjAwMDA4MywgMHg3RjA4ODAwNiwgMHg3RjBCODBCNCwgMHg3RjBDODAwNiwgMHg3RjBEMDA3OCwgMHg3RjEwMDA4MywgCi0gICAgICAgIDB4N0YxMjAwNzgsIDB4N0YxODgwOTksIDB4N0YxOTgwMzgsIDB4N0YxQTgwQjQsIDB4N0YyMjAwMDYsIDB4N0YyMzgwQjQsIDB4N0YyNDEwMDYsIDB4N0YyNjEwMzgsIAotICAgICAgICAweDdGMjg2MDA2LCAweDdGMjkwMDc4LCAweDdGMkE2MDBDLCAweDdGMkI2MDA2LCAweDdGMkM2MEI0LCAweDdGMkY2MDA3LCAweDdGMzA2MDA2LCAweDdGMzE2MDBELCAKLSAgICAgICAgMHg3RjMyNjAxOSwgMHg3RjMzMDA3OCwgMHg3RjM0NjAwOCwgMHg3RjM1NjAwNiwgMHg3RjM2MDA3OCwgMHg3RjM4NDg5RSwgMHg3RjM5MDA5RSwgMHg3RjNBMDA3OCwgCi0gICAgICAgIDB4N0YzQjQ4OUUsIDB4N0Y0MDQwOUUsIDB4N0Y0NTM4OUUsIDB4N0Y0NjQwOUUsIDB4N0Y0ODM4OUUsIDB4N0Y0OTQwOUUsIDB4N0Y0QjM4OUUsIDB4N0Y0QzQwOUUsIAotICAgICAgICAweDdGNEQzODlFLCAweDdGNEU0MDlFLCAweDdGNEYzODlFLCAweDdGNTA0MDlFLCAweDdGNTEzODlFLCAweDdGNTI0MDlFLCAweDdGNTMzODlFLCAweDdGNTQ0MDlFLCAKLSAgICAgICAgMHg3RjU5Mzg5RSwgMHg3RjVBNDA5RSwgMHg3RjVCMzg5RSwgMHg3RjVDNDA5RSwgMHg3RjVEMzg5RSwgMHg3RjVFNDA5RSwgMHg3RjVGMzg5RSwgMHg3RjYwNDA5RSwgCi0gICAgICAgIDB4N0Y2MTM4OUUsIDB4N0Y2MjQwOUUsIDB4N0Y2MzM4OUUsIDB4N0Y2NDQwOUUsIDB4N0Y2NTM4OUUsIDB4N0Y2NjQwOUUsIDB4N0Y2NzM4OUUsIDB4N0Y2ODQwOUUsIAotICAgICAgICAweDdGNjkzODlFLCAweDdGNkE0MDlFLCAweDdGNkIzODlFLCAweDdGNkM0MDlFLCAweDdGNkQzODlFLCAweDdGNkU0MDlFLCAweDdGNkYzODlFLCAweDdGNzA0MDlFLCAKLSAgICAgICAgMHg3RjcxMzg5RSwgMHg3RjcyNDA5RSwgMHg3RjczMzg5RSwgMHg3Rjc0NDA5RSwgMHg3Rjc1Mzg5RSwgMHg3Rjc2NDA5RSwgMHg3Rjc5Mzg5RSwgMHg3RjdBNDA5RSwgCi0gICAgICAgIDB4N0Y3RTAwNzgsIDB4N0Y3RjAwNTcsIDB4N0Y4MDg4MDYsIDB4N0Y4MTg4MDcsIDB4N0Y4Mzg4MDYsIDB4N0Y4NDg4MEEsIDB4N0Y4NTg4MEIsIDB4N0Y4Njg4MEQsIAotICAgICAgICAweDdGODc4ODBDLCAweDdGODg4ODBGLCAweDdGODk4ODExLCAweDdGOEE4ODEzLCAweDdGOEI4ODE1LCAweDdGOEM4ODE3LCAweDdGOEQ4ODA2LCAweDdGOEU4ODE5LCAKLSAgICAgICAgMHg3RjhGODgwNiwgMHg3RjkwODg2MCwgMHg3RjlEODgzNSwgMHg3RjlFODgzNiwgMHg3RjlGODgzOCwgMHg3RkEwODg2MSwgMHg3RkFEODgzNSwgMHg3RkFFODgzNiwgCi0gICAgICAgIDB4N0ZBRjg4MDksIDB4N0ZCMDUwMDYsIDB4N0ZCMTUwMEEsIDB4N0ZCMjUwMDYsIDB4N0ZCMzUwNzEsIDB4N0ZDRjUwODEsIDB4N0ZEMDUwNzEsIDB4N0ZERjAwNzgsIAotICAgICAgICAweDdGRTE1MDcxLCAweDdGRTQwMDc4LCAweDdGRTU1MDcxLCAweDdGRTgwMDc4LCAweDdGRTk1MDcxLCAweDdGRUMwMDc4LCAweDdGRUQ1MDcxLCAweDdGRUUwMDc4LCAKLSAgICAgICAgMHg3RkYwODgwOCwgMHg3RkYxODgzNywgMHg3RkYyODgwOCwgMHg3RkYzMDA3OCwgMHg3RkY0NTAxOSwgMHg3RkY2NTA1NCwgMHg3RkY3MDA3OCwgMHg3RkZDMDE0MSwgCi0gICAgICAgIDB4N0ZGRTAwNTQsIDB4N0ZGRjAwNzgsIDB4ODAwMDAwNzEsIDB4ODAxMzAwNzgsIDB4ODAxNDAwNzEsIDB4ODAxRDAwNzgsIDB4ODAxRTAwNzEsIDB4ODAyNzAwNzgsIAotICAgICAgICAweDgwMjgwMDcxLCAweDgwMkYwMDc4LCAweDgwNDAwMDcxLCAweDgwN0QwMDc4LCAweDgwODAwMDA2LCAweDgwODEwMDc4LCAweDgwODMwMEFELCAweDgwODQwMEFFLCAKLSAgICAgICAgMHg4MDg1MDEyRCwgMHg4MDg2MDEyRSwgMHg4MDg3MDEyRiwgMHg4MDg4MDE4NSwgMHg4MDg5MDE4NiwgMHg4MDhBMDE4NywgMHg4MDhCMDE4OCwgMHg4MDhDMDE4NCwgCi0gICAgICAgIDB4ODA4RDAxQzksIDB4ODA4RTAxQ0EsIDB4ODA4RjAxQ0IsIDB4ODA5MDAxQ0MsIDB4ODA5MTAxQ0QsIDB4ODA5MjAxQ0UsIDB4ODA5MzAxQ0YsIDB4ODA5NDAxRDAsIAotICAgICAgICAweDgwOTUwMEJFLCAweDgwOTYwMUQxLCAweDgwOTcwMUQyLCAweDgwOTgwMUQzLCAweDgwOTkwMUQ0LCAweDgwOUEwMDc4LCAweDgwOUIwMDk0LCAweDgwQTAwMTRFLCAKLSAgICAgICAgMHg4MEExMDE1MiwgMHg4MEEyMDE1MywgMHg4MEEzMDE1NywgMHg4MEE0MDE1NCwgMHg4MEE1MDE1NSwgMHg4MEE2MDE1NiwgMHg4MEE3MDE1MiwgMHg4MEE4MDE1MCwgCi0gICAgICAgIDB4ODBBOTAxNTMsIDB4ODBBQTAxRDUsIDB4ODBBQjAxNTQsIDB4ODBBQzAxNEYsIDB4ODBBRDAxNTgsIDB4ODBBRjAxNTIsIDB4ODBCMDAxNTQsIDB4ODBCMjAxRDYsIAotICAgICAgICAweDgwQjMwMTUwLCAweDgwQjUwMUQ3LCAweDgwQjYwMTUzLCAweDgwQjgwMTU2LCAweDgwQjkwMTUyLCAweDgwQkEwMDVGLCAweDgwQkMwMDU0LCAweDgwQzUwMDc4LCAKLSAgICAgICAgMHg4MTgwMDA3MSwgMHg4MThGMDA3OCwgMHg4MTkwMDEyRCwgMHg4MTkxMDBCQiwgMHg4MTkyMDA3OCwgMHg4MTk4MDA3MSwgMHg4MUE1MDA3OCwgMHg4MUMwMDA3MSwgCi0gICAgICAgIDB4ODFDRjAwOTcsIDB4ODFEMDAwNzEsIDB4ODFFMjAwNzgsIDB4ODFFNDAwNzEsIDB4ODFFODAxNEYsIDB4ODFFOTAxNTQsIDB4ODFFQTAxNTUsIDB4ODFFQjAwNzgsIAotICAgICAgICAweDgyMDAwMTVCLCAweDgyMTQwMTVDLCAweDgyMjgwMDcxLCAweDgyNEYwMDc4LCAweDgyNTAwMTdFLCAweDgyNTEwMTdGLCAweDgyNTIwMTgwLCAweDgyNTMwMTgxLCAKLSAgICAgICAgMHg4MjU0MDE4MiwgMHg4MjU1MDA3OCwgMHg4NDAwMDA5QiwgMHg4NDAzMDA3OCwgMHg4NDA1MDA5QiwgMHg4NDFDMDA3OCwgMHg4NDFGMDA5QiwgMHg4NDIwMDA3OCwgCi0gICAgICAgIDB4ODUwMDAwODMsIDB4ODUwMzAwNzgsIDB4ODUwNjAwODMsIDB4ODUwODAwOUIsIDB4ODUxQTAwNzgsIDB4ODUxQzAwODMsIDB4ODUxRDAwNzgsIDB4ODUxRjAwODMsIAotICAgICAgICAweDg1MjAwMUQ4LCAweDg1MjEwMUQ5LCAweDg1MjIwMURBLCAweDg1MjMwMURCLCAweDg1MjQwMDc4LCAweDg1MjgwMDlBLCAweDg1MkMwMDc4LCAweEU4MDAwMDk0LCAKLSAgICAgICAgMHhFODdCMDA3OCwgMHhFODgwMDA5NCwgMHhFODkzMDA3OCwgMHhFODk1MDA5NCwgMHhFOEFGMDg5NCwgMHhFOEIyMDBBNywgMHhFOEIzMDA4MywgMHhFOEI1MDA5NCwgCi0gICAgICAgIDB4RThCNjAwQTcsIDB4RThCOTAwNTcsIDB4RThCRDAwODMsIDB4RThDMTAwOTQsIDB4RThDMjAwODMsIDB4RThDNjAwOTQsIDB4RThENTAwODMsIDB4RThENzAwOTQsIAotICAgICAgICAweEU4REQwODk0LCAweEU4RTAwMDk0LCAweEU4RUYwMDc4LCAweEU5MDAwMDU0LCAweEU5MjEwMDgzLCAweEU5MjIwMDU0LCAweEU5MjMwMDc4LCAweEU5ODAwMDU0LCAKLSAgICAgICAgMHhFOUFCMDA3OCwgMHhFQTAwMjg3NywgMHhFQTBEMjg1NSwgMHhFQTFBMjg3NywgMHhFQTI3Mjg1NSwgMHhFQTJBMDA3OCwgMHhFQTJCMjg1NSwgMHhFQTM0Mjg3NywgCi0gICAgICAgIDB4RUE0MTI4NTUsIDB4RUE0RTAwNzgsIDB4RUE0RjI4NzcsIDB4RUE1MDAwNzgsIDB4RUE1MjI4NzcsIDB4RUE1MzAwNzgsIDB4RUE1NDI4NzcsIDB4RUE1NjAwNzgsIAotICAgICAgICAweEVBNTcyODc3LCAweEVBNUIyODU1LCAweEVBNjgyODc3LCAweEVBNzUyODU1LCAweEVBODIyODc3LCAweEVBODUwMDc4LCAweEVBODYyODc3LCAweEVBOEEwMDc4LCAKLSAgICAgICAgMHhFQThCMjg3NywgMHhFQThFMDA3OCwgMHhFQThGMjg1NSwgMHhFQTlDMjg3NywgMHhFQTlGMDA3OCwgMHhFQUEwMjg3NywgMHhFQUEyMDA3OCwgMHhFQUE1Mjg3NywgCi0gICAgICAgIDB4RUFBODAwNzgsIDB4RUFBOTI4NTUsIDB4RUFCNjI4NzcsIDB4RUFDMzI4NTUsIDB4RUFEMDI4NzcsIDB4RUFERDI4NTUsIDB4RUFFQTI4NzcsIDB4RUFGNzI4NTUsIAotICAgICAgICAweEVCMDQyODc3LCAweEVCMTEyODU1LCAweEVCMUUyODc3LCAweEVCMkIyODU1LCAweEVCMzgyODc3LCAweEVCNDUyODU1LCAweEVCNTMwMDc4LCAweEVCNTQyODc3LCAKLSAgICAgICAgMHhFQjYwMjlEQywgMHhFQjYxMjg1NSwgMHhFQjZEMjlEQywgMHhFQjZFMjg1NSwgMHhFQjcxMjg3NywgMHhFQjdEMjlEQywgMHhFQjdFMjg1NSwgMHhFQjhBMjlEQywgCi0gICAgICAgIDB4RUI4QjI4NTUsIDB4RUI4RTI4NzcsIDB4RUI5QTI5REMsIDB4RUI5QjI4NTUsIDB4RUJBNzI5REMsIDB4RUJBODI4NTUsIDB4RUJBQjI4NzcsIDB4RUJCNzI5REMsIAotICAgICAgICAweEVCQjgyODU1LCAweEVCQzQyOURDLCAweEVCQzUyODU1LCAweEVCQzgyODc3LCAweEVCRDQyOURDLCAweEVCRDUyODU1LCAweEVCRTEyOURDLCAweEVCRTIyODU1LCAKLSAgICAgICAgMHhFQkU1MDA3OCwgMHhFQkU3MjgwRiwgMHhFQkU4MjgxMSwgMHhFQkU5MjgxMywgMHhFQkVBMjgxNSwgMHhFQkVCMjgxNywgMHhFQkVDMjgwRiwgMHhFQkVEMjgxMSwgCi0gICAgICAgIDB4RUJFRTI4MTMsIDB4RUJFRjI4MTUsIDB4RUJGMDI4MTcsIDB4RUJGMTI4MEYsIDB4RUJGMjI4MTEsIDB4RUJGMzI4MTMsIDB4RUJGNDI4MTUsIDB4RUJGNTI4MTcsIAotICAgICAgICAweEVCRjYyODBGLCAweEVCRjcyODExLCAweEVCRjgyODEzLCAweEVCRjkyODE1LCAweEVCRkEyODE3LCAweEVCRkIyODBGLCAweEVCRkMyODExLCAweEVCRkQyODEzLCAKLSAgICAgICAgMHhFQkZFMjgxNSwgMHhFQkZGMjgxNywgMHhFQzAwMDA3OAotICAgIH07Ci0KLSAgICBzdGF0aWMgY29uc3QgdWludDMyX3QgYTE3W10gPSB7Ci0gICAgICAgIDB4MDAwMDAwNzEsIDB4NTM2QjAwNzgsIDB4N0MwMDA4NzEsIDB4N0QwRjAwNzgKLSAgICB9OwotCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IGEyM1tdID0gewotICAgICAgICAweDAwMDAwMDU3LCAweDAwMDEwMDc4LCAweDAwMTAwMDU3LCAweDAwNDAwMDc4LCAweDAwODAwMDgzLCAweDAwRjgwMDc4LCAweDgwMDAwMTNGLCAweEZGRkYwMDc4Ci0gICAgfTsKLQotICAgIHN0YXRpYyBjb25zdCB1aW50MzJfdCBhMjRbXSA9IHsKLSAgICAgICAgMHgwMDAwMDEzRiwgMHg3RkZGMDA3OAotICAgIH07Ci0KLQotICAgIC8vIFRoZSBmdWxsIHNldCBvZiBhbGwgYXJyYXlzIHRvIGJlIHNlYXJjaGVkLgotICAgIHN0YXRpYyBjb25zdCBSYW5nZSBGVUxMX0RBVEFbXSA9IHsKLSAgICAgICAge3NpemVvZihhMCkvc2l6ZW9mKHVpbnQzMl90KSwgYTB9LAotICAgICAgICB7c2l6ZW9mKGExKS9zaXplb2YodWludDMyX3QpLCBhMX0sCi0gICAgICAgIHswLCAwfSwKLSAgICAgICAgezAsIDB9LAotICAgICAgICB7MCwgMH0sCi0gICAgICAgIHswLCAwfSwKLSAgICAgICAgezAsIDB9LAotICAgICAgICB7c2l6ZW9mKGE3KS9zaXplb2YodWludDMyX3QpLCBhN30sCi0gICAgICAgIHtzaXplb2YoYTgpL3NpemVvZih1aW50MzJfdCksIGE4fSwKLSAgICAgICAgezAsIDB9LAotICAgICAgICB7MCwgMH0sCi0gICAgICAgIHswLCAwfSwKLSAgICAgICAgezAsIDB9LAotICAgICAgICB7MCwgMH0sCi0gICAgICAgIHswLCAwfSwKLSAgICAgICAgezAsIDB9LAotICAgICAgICB7c2l6ZW9mKGExNikvc2l6ZW9mKHVpbnQzMl90KSwgYTE2fSwKLSAgICAgICAge3NpemVvZihhMTcpL3NpemVvZih1aW50MzJfdCksIGExN30sCi0gICAgICAgIHswLCAwfSwKLSAgICAgICAgezAsIDB9LAotICAgICAgICB7MCwgMH0sCi0gICAgICAgIHswLCAwfSwKLSAgICAgICAgezAsIDB9LAotICAgICAgICB7c2l6ZW9mKGEyMykvc2l6ZW9mKHVpbnQzMl90KSwgYTIzfSwKLSAgICAgICAge3NpemVvZihhMjQpL3NpemVvZih1aW50MzJfdCksIGEyNH0sCi0gICAgICAgIHswLCAwfSwKLSAgICAgICAgezAsIDB9LAotICAgICAgICB7MCwgMH0sCi0gICAgICAgIHswLCAwfSwKLSAgICAgICAgezAsIDB9LAotICAgICAgICB7MCwgMH0sCi0gICAgICAgIHswLCAwfQotICAgIH07Ci0KLSAgICAvLyBBcnJheSBvZiB1cHBlcmNhc2UgZGlmZmVyZW5jZXMKLSAgICBzdGF0aWMgY29uc3Qgc2hvcnQgVUNESUZGW10gPSB7Ci0gICAgICAgICAgICAwLCAgIC0zMiwgICA3NDMsICAgMTIxLCAgICAtMSwgIC0yMzIsICAtMzAwLCAgICA5NywgCi0gICAgICAgICAgMTYzLCAgIDEzMCwgICAgNTYsICAgIC0yLCAgIC03OSwgIC0yMTAsICAtMjA2LCAgLTIwNSwgCi0gICAgICAgICAtMjAyLCAgLTIwMywgIC0yMDcsICAtMjA5LCAgLTIxMSwgIC0yMTMsICAtMjE0LCAgLTIxOCwgCi0gICAgICAgICAtMjE3LCAgLTIxOSwgICAtODMsICAgIDg0LCAgIC0zOCwgICAtMzcsICAgLTMxLCAgIC02NCwgCi0gICAgICAgICAgLTYzLCAgIC02MiwgICAtNTcsICAgLTQ3LCAgIC01NCwgICAtODYsICAgLTgwLCAgICAgNywgCi0gICAgICAgICAgLTk2LCAgIC00OCwgICAtNTksICAgICA4LCAgICA3NCwgICAgODYsICAgMTAwLCAgIDEyOCwgCi0gICAgICAgICAgMTEyLCAgIDEyNiwgICAgIDksIC03MjA1LCAgIC0xNiwgICAtMjYsIC03MjY0LCAgIC00MAotICAgIH07Ci0KLSAgICAvLyBBcnJheSBvZiBsb3dlcmNhc2UgZGlmZmVyZW5jZXMKLSAgICBzdGF0aWMgY29uc3Qgc2hvcnQgTENESUZGW10gPSB7Ci0gICAgICAgICAgICAwLCAgICAzMiwgICAgIDEsICAtMTk5LCAgLTEyMSwgICAyMTAsICAgMjA2LCAgIDIwNSwgCi0gICAgICAgICAgIDc5LCAgIDIwMiwgICAyMDMsICAgMjA3LCAgIDIxMSwgICAyMDksICAgMjEzLCAgIDIxNCwgCi0gICAgICAgICAgMjE4LCAgIDIxNywgICAyMTksICAgICAyLCAgIC05NywgICAtNTYsICAtMTMwLCAgLTE2MywgCi0gICAgICAgICAgIDgzLCAgICAzOCwgICAgMzcsICAgIDY0LCAgICA2MywgICAtNjAsICAgIC03LCAgICA4MCwgCi0gICAgICAgICAgIDQ4LCAgNzI2NCwgICAgLTgsICAgLTc0LCAgICAtOSwgICAtODYsICAtMTAwLCAgLTExMiwgCi0gICAgICAgICAtMTI4LCAgLTEyNiwgLTc1MTcsIC04MzgzLCAtODI2MiwgICAgMTYsICAgIDI2LCAgICA0MAotICAgIH07Ci0KLSAgICAvLyBBcnJheSBvZiB0aXRsZWNhc2UgZGlmZmVyZW5jZXMKLSAgICBzdGF0aWMgY29uc3Qgc2hvcnQgVENESUZGW10gPSB7Ci0gICAgICAgICAgICAzLCAgICAgMSwgICAgIDAsICAgIC0xCi0gICAgfTsKLQotICAgIC8vIEFycmF5IG9mIG1pcnJvcmVkIGNoYXJhY3RlciBkaWZmZXJlbmNlcwotICAgIHN0YXRpYyBjb25zdCBzaG9ydCBNSVJST1JfRElGRltdID0gewotICAgICAgICAgICAgMCwgICAgIDEsICAgIC0xLCAgICAgMiwgICAgLTIsICAgIDE2LCAgIC0xNiwgICAgIDMsIAotICAgICAgICAgICAtMywgIDIwMTYsICAgMTM4LCAgMTgyNCwgIDIxMDQsICAyMTA4LCAgMjEwNiwgIC0xMzgsIAotICAgICAgICAgICAgOCwgICAgIDcsICAgIC04LCAgICAtNywgLTE4MjQsIC0yMDE2LCAtMjEwNCwgLTIxMDYsIAotICAgICAgICAtMjEwOAotICAgIH07Ci0KLSAgIC8vIEFycmF5IG9mIGFsbCBwb3NzaWJsZSBudW1lcmljIHZhbHVlcwotICAgc3RhdGljIGNvbnN0IGludCBOVU1FUklDU1tdID0gewotICAgICAgICAgICAgLTEsICAgICAgMCwgICAgICAxLCAgICAgIDIsICAgICAgMywgICAgICA0LCAgICAgIDUsICAgICAgNiwgCi0gICAgICAgICAgICAgNywgICAgICA4LCAgICAgIDksICAgICAxMCwgICAgIDExLCAgICAgMTIsICAgICAxMywgICAgIDE0LCAKLSAgICAgICAgICAgIDE1LCAgICAgMTYsICAgICAxNywgICAgIDE4LCAgICAgMTksICAgICAyMCwgICAgIDIxLCAgICAgMjIsIAotICAgICAgICAgICAgMjMsICAgICAyNCwgICAgIDI1LCAgICAgMjYsICAgICAyNywgICAgIDI4LCAgICAgMjksICAgICAzMCwgCi0gICAgICAgICAgICAzMSwgICAgIDMyLCAgICAgMzMsICAgICAzNCwgICAgIDM1LCAgICAgLTIsICAgIDEwMCwgICAxMDAwLCAKLSAgICAgICAgICAgIDQwLCAgICAgNTAsICAgICA2MCwgICAgIDcwLCAgICAgODAsICAgICA5MCwgIDEwMDAwLCAgICA1MDAsIAotICAgICAgICAgIDUwMDAsICAgICAzNiwgICAgIDM3LCAgICAgMzgsICAgICAzOSwgICAgIDQxLCAgICAgNDIsICAgICA0MywgCi0gICAgICAgICAgICA0NCwgICAgIDQ1LCAgICAgNDYsICAgICA0NywgICAgIDQ4LCAgICAgNDksICAgIDIwMCwgICAgMzAwLCAKLSAgICAgICAgICAgNDAwLCAgICA2MDAsICAgIDcwMCwgICAgODAwLCAgICA5MDAsICAgMjAwMCwgICAzMDAwLCAgIDQwMDAsIAotICAgICAgICAgIDYwMDAsICAgNzAwMCwgICA4MDAwLCAgIDkwMDAsICAyMDAwMCwgIDMwMDAwLCAgNDAwMDAsICA1MDAwMCwgCi0gICAgICAgICA2MDAwMCwgIDcwMDAwLCAgODAwMDAsICA5MDAwMAotICAgIH07Ci0KLSAgICAvLyBBbGwgcG9zc2libGUgcGFja2VkIGRhdGEgdmFsdWVzLCBubyBkdXBsaWNhdGVzCi0gICAgc3RhdGljIGNvbnN0IHVpbnQzMl90IFBBQ0tFRF9EQVRBW10gPSB7Ci0gICAgICAgIDB4MDAwMDAwMDAsIDB4MDAwMDAxMkYsIDB4MDAwMDAxNkYsIDB4MDAwMDAxNEYsIDB4MDAwMDAxOEYsIDB4MDAwMDAxOEMsIDB4MDAwMDAxQjgsIDB4MDAwMDAwQjgsIAotICAgICAgICAweDAwMDAwMEJBLCAweDAyMDAwNUI1LCAweDA0MDAwNUI2LCAweDAwMDAwMDk5LCAweDAwMDAwMEY4LCAweDAwMDAwMDk0LCAweDAyMDAwMDY5LCAweDA0MDAwMDY5LCAKLSAgICAgICAgMHgwNjAwMDA2OSwgMHgwODAwMDA2OSwgMHgwQTAwMDA2OSwgMHgwQzAwMDA2OSwgMHgwRTAwMDA2OSwgMHgxMDAwMDA2OSwgMHgxMjAwMDA2OSwgMHgxNDAwMDA2OSwgCi0gICAgICAgIDB4MDYwMDA1QjksIDB4MDAwMDAxQjksIDB4MDgwMDA1QjksIDB4MTYwMjAwMDEsIDB4MTgwMjAwMDEsIDB4MUEwMjAwMDEsIDB4MUMwMjAwMDEsIDB4MUUwMjAwMDEsIAotICAgICAgICAweDIwMDIwMDAxLCAweDIyMDIwMDAxLCAweDI0MDIwMDAxLCAweDI2MDIwMDAxLCAweDI4MDIwMDAxLCAweDJBMDIwMDAxLCAweDJDMDIwMDAxLCAweDJFMDIwMDAxLCAKLSAgICAgICAgMHgzMDAyMDAwMSwgMHgzMjAyMDAwMSwgMHgzNDAyMDAwMSwgMHgzNjAyMDAwMSwgMHgzODAyMDAwMSwgMHgzQTAyMDAwMSwgMHgzQzAyMDAwMSwgMHgzRTAyMDAwMSwgCi0gICAgICAgIDB4NDAwMjAwMDEsIDB4NDIwMjAwMDEsIDB4NDQwMjAwMDEsIDB4NDYwMjAwMDEsIDB4NDgwMjAwMDEsIDB4MDYwMDA1QjUsIDB4MDgwMDA1QjYsIDB4MDAwMDAxQkIsIAotICAgICAgICAweDAwMDAwMUI3LCAweDE2MDAwODAyLCAweDE4MDAwODAyLCAweDFBMDAwODAyLCAweDFDMDAwODAyLCAweDFFMDAwODAyLCAweDIwMDAwODAyLCAweDIyMDAwODAyLCAKLSAgICAgICAgMHgyNDAwMDgwMiwgMHgyNjAwMDgwMiwgMHgyODAwMDgwMiwgMHgyQTAwMDgwMiwgMHgyQzAwMDgwMiwgMHgyRTAwMDgwMiwgMHgzMDAwMDgwMiwgMHgzMjAwMDgwMiwgCi0gICAgICAgIDB4MzQwMDA4MDIsIDB4MzYwMDA4MDIsIDB4MzgwMDA4MDIsIDB4M0EwMDA4MDIsIDB4M0MwMDA4MDIsIDB4M0UwMDA4MDIsIDB4NDAwMDA4MDIsIDB4NDIwMDA4MDIsIAotICAgICAgICAweDQ0MDAwODAyLCAweDQ2MDAwODAyLCAweDQ4MDAwODAyLCAweDAwMDAwMEVDLCAweDAwMDAwMUJDLCAweDAwMDAwMDAyLCAweDBBMDAwNUJELCAweDAwMDAwMTMwLCAKLSAgICAgICAgMHgwMDAwMDBCQywgMHgwMDAwMDBCOSwgMHgwNjAwMDA2QiwgMHgwODAwMDA2QiwgMHgwMDAwMTAwMiwgMHgwNDAwMDA2QiwgMHgwQzAwMDVCRSwgMHg0QTAwMDFBQiwgCi0gICAgICAgIDB4MDAwMjAwMDEsIDB4MDAwMDA4MDIsIDB4MDAwMDE4MDIsIDB4MDAwNDAwMDEsIDB4MDAwNjAwMDEsIDB4MDAwMDIwMDIsIDB4MDAwODAwMDEsIDB4MDAwQzAwMDEsIAotICAgICAgICAweDAwMEUwMDAxLCAweDAwMTAwMDAxLCAweDAwMTQwMDAxLCAweDAwMTYwMDAxLCAweDAwMTgwMDAxLCAweDAwMDA0MDAyLCAweDAwMDA0ODAyLCAweDAwMjAwMDAxLCAKLSAgICAgICAgMHgwMDIyMDAwMSwgMHgwMDAwMDAwNSwgMHgwMEE2MDAwMSwgMHgwMTgwNTgwMiwgMHgwMTA0MjAwMywgMHgwMDI4MDAwMSwgMHgwMDJDMDAwMSwgMHgwMDAwMDAwMSwgCi0gICAgICAgIDB4MDAwMDAwMDAsIDB4MDAwMDcwMDIsIDB4MDAwMDc4MDIsIDB4MDAwMDk4MDIsIDB4MDAwMEE4MDIsIDB4MDAwMEI4MDIsIDB4MDAwMEMwMDIsIDB4MDAwMEM4MDIsIAotICAgICAgICAweDAwMDBEMDAyLCAweDAwMDAwMDA0LCAweDAwMDAwMUE0LCAweDAwMDAwMTA2LCAweDAwMzIwMDAxLCAweDAwMzQwMDAxLCAweDAwMzYwMDAxLCAweDAwMzgwMDAxLCAKLSAgICAgICAgMHgwMDAwRTAwMiwgMHgwMDAwRTgwMiwgMHgwMDAwRjAwMiwgMHgwMDAwRjgwMiwgMHgwMDAxMDAwMiwgMHgwMDAxMDgwMiwgMHgwMDAxMjAwMiwgMHgwMDAxMjgwMiwgCi0gICAgICAgIDB4MDAwMTM4MDIsIDB4MDAzQTAwMDEsIDB4MDAzRTAwMDEsIDB4MDAwMTMwMDIsIDB4MDAwMDAwMUMsIDB4MDAwMDAxMDcsIDB4MDA0MDAwMDEsIDB4MDAwMDAwMTgsIAotICAgICAgICAweDAwMDE0ODAyLCAweDAwMDAwMUI0LCAweDAwMDAwMDM4LCAweDAwMDAwMDI1LCAweDAwMDAwMDUwLCAweDAwMDAwMDU4LCAweDAwMDAwMDQ1LCAweDAwMDAwMDQ0LCAKLSAgICAgICAgMHgwMjAwMDBDOSwgMHgwNjAwMDBDOSwgMHgwQTAwMDBDOSwgMHgwRTAwMDBDOSwgMHgxMjAwMDBDOSwgMHgwMDAwMDBEOCwgMHgwMDAwMDA1QywgMHgwMDAwMDAwOCwgCi0gICAgICAgIDB4MDIwMDAwMDksIDB4MDYwMDAwMDksIDB4MEEwMDAwMDksIDB4MEUwMDAwMDksIDB4MTIwMDAwMDksIDB4MDQwMDAwMEIsIDB4MDgwMDAwMEIsIDB4MDAwMDAwMEIsIAotICAgICAgICAweDE2MDAwMDBCLCAweDRFMDAwMDBCLCAweDAwMDAwMDA2LCAweDRBMDAwMDBCLCAweDAwMDAwMUI1LCAweDAwNDIwMDAxLCAweDA2MDAwMDBCLCAweDBBMDAwMDBCLCAKLSAgICAgICAgMHgwRTAwMDAwQiwgMHgxMjAwMDAwQiwgMHgzRTAwMDAwQiwgMHg1MjAwMDAwQiwgMHg1NjAwMDAwQiwgMHg1QTAwMDAwQiwgMHg1QzAwMDAwQiwgMHgwMDAwMDFCNiwgCi0gICAgICAgIDB4MjQwMDAwMEEsIDB4MjgwMDAwMEEsIDB4MDAwMDAwMTAsIDB4MDIwMDAxQUIsIDB4MDYwMDAxQUIsIDB4MEEwMDAxQUIsIDB4MEUwMDAxQUIsIDB4MTIwMDAxQUIsIAotICAgICAgICAweDAwMDAwMTA4LCAweDAwMDE1ODAyLCAweDAwNDQwMDAxLCAweDAwMDE2MDAyLCAweDAwMDE2ODAyLCAweDAwMDE3MDAyLCAweDAwMDE3ODAyLCAweDAwMDE4MDAyLCAKLSAgICAgICAgMHgwMDAxODgwMiwgMHgwMDQ0MDAwMywgMHgwMDQ2MDAwMSwgMHgwMDQ4MDAwMywgMHgwMDAxOTgwMiwgMHgwMDRBMDAwMSwgMHgwMDRDMDAwMSwgMHgwMDRFMDAwMSwgCi0gICAgICAgIDB4MDAzQzAwMDEsIDB4MDA1MDAwMDEsIDB4MDA1MjAwMDEsIDB4MDAwMDAxQkQsIDB4MDAwMDAxOEQsIDB4MDAwMDAxRDAsIDB4MDAwMDAyNTAsIDB4MDAwMDAyMzAsIAotICAgICAgICAweDA0MDAwNUJFLCAweDAwMDAwMEY5LCAweDAyMDAwMDZCLCAweDBBMDAwMDZCLCAweDBFMDAwMDZCLCAweDEyMDAwMDZCLCAweDAwNTQwMDAxLCAweDAwNTYwMDAxLCAKLSAgICAgICAgMHgwMDAwMDVCOSwgMHgwNDVBMDAwQSwgMHgwODVBMDAwQSwgMHgwQzVBMDAwQSwgMHgxMDVBMDAwQSwgMHgxNDVBMDAwQSwgMHgxODVBMDAwQSwgMHg1MjVBMDAwQSwgCi0gICAgICAgIDB4NUU1QTAwMEEsIDB4MDQwMUEwMEEsIDB4MDgwMUEwMEEsIDB4MEMwMUEwMEEsIDB4MTAwMUEwMEEsIDB4MTQwMUEwMEEsIDB4MTgwMUEwMEEsIDB4NTIwMUEwMEEsIAotICAgICAgICAweDVFMDFBMDBBLCAweDRFMDAwMDBBLCAweDVDMDAwMDBBLCAweDBFMDAwNUI5LCAweDEwMDAwNUI5LCAweDAyMDAwNUI5LCAweDA0MDAwNUI5LCAweDE2MDAwNUI5LCAKLSAgICAgICAgMHgxODAwMDVCOSwgMHgxQTAwMDVCOSwgMHgyMDAwMDVCOSwgMHgyMjAwMDVCOSwgMHgyNDAwMDVCOSwgMHgyNjAwMDVCOSwgMHgwNDAwMDFBQiwgMHgwODAwMDFBQiwgCi0gICAgICAgIDB4MEMwMDAxQUIsIDB4MTAwMDAxQUIsIDB4MTQwMDAxQUIsIDB4MTgwMDAxQUIsIDB4MUMwMDAxQUIsIDB4MjAwMDAxQUIsIDB4MjQwMDAxQUIsIDB4MjgwMDAxQUIsIAotICAgICAgICAweDBDMDAwMDZCLCAweDEwMDAwMDZCLCAweDE0MDAwMDZCLCAweDE4MDAwMDZCLCAweDFDMDAwMDZCLCAweDIwMDAwMDZCLCAweDI0MDAwMDZCLCAweDI4MDAwMDZCLCAKLSAgICAgICAgMHgwMDVDMDAxQywgMHgwMDAxQTgxQywgMHgxQTAwMDFBQiwgMHgxRTAwMDFBQiwgMHgyMjAwMDFBQiwgMHgyNjAwMDFBQiwgMHgyQTAwMDFBQiwgMHgxNjAwMDFBQiwgCi0gICAgICAgIDB4MDIwMDA1QjYsIDB4MTAwMDA1QjYsIDB4MjgwMDA1QjksIDB4MkMwMDA1QjksIDB4MzAwMDA1QjksIDB4MDAwMUIwMDIsIDB4MDIwMDA1QkQsIDB4MDYwMDAwMEEsIAotICAgICAgICAweDBBMDAwMDBBLCAweDBFMDAwMDBBLCAweDEyMDAwMDBBLCAweDE2MDAwMDBBLCAweDNFMDAwMDBBLCAweDBDMDAwMDBCLCAweDEwMDAwMDBCLCAweDE0MDAwMDBCLCAKLSAgICAgICAgMHgyRTAwMDFBQiwgMHgzMjAwMDFBQiwgMHgzNjAwMDFBQiwgMHgzQTAwMDFBQiwgMHgzRTAwMDFBQiwgMHg0MjAwMDFBQiwgMHg0NjAwMDFBQiwgMHg2NDAwMDFBQiwgCi0gICAgICAgIDB4NjgwMDAxQUIsIDB4NkEwMDAxQUIsIDB4NkUwMDAxQUIsIDB4NzIwMDAxQUIsIDB4NzYwMDAxQUIsIDB4N0EwMDAxQUIsIDB4MDAwMDAwMTMsIDB4MDAwMDAwMTIsIAotICAgICAgICAweDAwMDAwMDVBLCAweDAwMDAwMUIwLCAweDdDMDAwMDBCLCAweDgwMDAwMDBCLCAweDgyMDAwMDBCLCAweDg2MDAwMDBCLCAweDhDMDAwMDBCLCAweDYwMDAwMDBCLCAKLSAgICAgICAgMHg5MjAwMDAwQiwgMHg5NjAwMDAwQiwgMHg5ODAwMDAwQiwgMHg5QzAwMDAwQiwgMHhBMDAwMDAwQiwgMHhBNDAwMDAwQiwgMHg0QTAwMDFBQSwgMHgwNDAwMDFBQSwgCi0gICAgICAgIDB4NTIwMDAxQUEsIDB4NjAwMDAxQUEsIDB4MEMwMDAxQUEsIDB4NUUwMDAxQUEsIDB4MTYwMDAxQUEsIDB4NEMwMDAxQUEsIDB4NEUwMDAxQUEsIDB4OUUwMDAxQUEsIAotICAgICAgICAweDA2MDAwMUFBLCAweDg4MDAwMDBBLCAweDJBMDAwMUFBLCAweDAwNUUwMDAxLCAweDAwMDFCODAyLCAweDA0MDAwMDJCLCAweDA4MDAwMDJCLCAweDE2MDAwMDJCLCAKLSAgICAgICAgMHg0QzAwMDAyQiwgMHgwMDAwMjgwMiwgMHgwMDAwMzAwMiwgMHgwMDBBMDAwMSwgMHgwMDEyMDAwMSwgMHgwMDAwMzgwMiwgMHgwMDFBMDAwMSwgMHgwMDFDMDAwMSwgCi0gICAgICAgIDB4MDAxRTAwMDEsIDB4MDAyNDAwMDEsIDB4MDAwMDUwMDIsIDB4MDAwMDYwMDIsIDB4MDAyQTAwMDEsIDB4MDAyRTAwMDEsIDB4MDAzMDAwMDEsIDB4MDAwMDY4MDIsIAotICAgICAgICAweDAwMDA4MDAyLCAweDAwMDA4ODAyLCAweDAwMDA5MDAyLCAweDAwMDBBMDAyLCAweDAwMDBCMDAyLCAweDAwMDBEOTA2LCAweDAwMDExMDAyLCAweDAwMDExODAyLCAKLSAgICAgICAgMHgwMDAxNDAwMiwgMHgwNDAwMDBDOSwgMHgwODAwMDBDOSwgMHgwQzAwMDBDOSwgMHgxMDAwMDBDOSwgMHgxNDAwMDBDOSwgMHgwNDAwMDAwOSwgMHgwODAwMDAwOSwgCi0gICAgICAgIDB4MEMwMDAwMDksIDB4MTAwMDAwMDksIDB4MTQwMDAwMDksIDB4MjIwMDAwMEIsIDB4NEMwMDAwMEIsIDB4MkEwMDAwMEIsIDB4NTAwMDAwMEIsIDB4NTQwMDAwMEIsIAotICAgICAgICAweDU4MDAwMDBCLCAweDI2MDAwMDBBLCAweDAwMDE1MDAyLCAweDAwMDE5MDAyLCAweDAwMDAwMDMwLCAweDAwMDAwMUJFLCAweDAwMDAwMTRFLCAweDAwMDAwMjEwLCAKLSAgICAgICAgMHgwMDAwMDFGMCwgMHgwMDU4MDAwMSwgMHgwNjVBMDAwQSwgMHgwQTVBMDAwQSwgMHgwRTVBMDAwQSwgMHgxMjVBMDAwQSwgMHgxNjVBMDAwQSwgMHgxQTVBMDAwQSwgCi0gICAgICAgIDB4NEM1QTAwMEEsIDB4NEU1QTAwMEEsIDB4MDYwMUEwMEEsIDB4MEEwMUEwMEEsIDB4MEUwMUEwMEEsIDB4MTIwMUEwMEEsIDB4MTYwMUEwMEEsIDB4MUEwMUEwMEEsIAotICAgICAgICAweDRDMDFBMDBBLCAweDRFMDFBMDBBLCAweDYwMDAwMDBBLCAweDAwMDAwMDBBLCAweDEyMDAwNUI5LCAweDE0MDAwNUI5LCAweDFDMDAwNUI5LCAweDFFMDAwNUI5LCAKLSAgICAgICAgMHgxNjAwMDA2QiwgMHgxQTAwMDA2QiwgMHgxRTAwMDA2QiwgMHgyMjAwMDA2QiwgMHgyNjAwMDA2QiwgMHgyQTAwMDA2QiwgMHgwRTAwMDVCNSwgMHgwNDAwMDVCNSwgCi0gICAgICAgIDB4MkEwMDA1QjksIDB4MkUwMDA1QjksIDB4MDIwMDAwMEEsIDB4MDQwMDAwMEEsIDB4MDgwMDAwMEEsIDB4MEMwMDAwMEEsIDB4MTAwMDAwMEEsIDB4MTQwMDAwMEEsIAotICAgICAgICAweDJBMDAwMDBBLCAweDJDMDAwMUFCLCAweDMwMDAwMUFCLCAweDM0MDAwMUFCLCAweDM4MDAwMUFCLCAweDNDMDAwMUFCLCAweDQwMDAwMUFCLCAweDQ0MDAwMUFCLCAKLSAgICAgICAgMHg0ODAwMDFBQiwgMHg2MjAwMDFBQiwgMHg2NjAwMDFBQiwgMHg1MDAwMDFBQiwgMHg2QzAwMDFBQiwgMHg3MDAwMDFBQiwgMHg3NDAwMDFBQiwgMHg3ODAwMDFBQiwgCi0gICAgICAgIDB4NTIwMDAxQUIsIDB4N0UwMDAwMEIsIDB4NUUwMDAwMEIsIDB4ODQwMDAwMEIsIDB4ODgwMDAwMEIsIDB4OEEwMDAwMEIsIDB4OEUwMDAwMEIsIDB4OTAwMDAwMEIsIAotICAgICAgICAweDk0MDAwMDBCLCAweDlBMDAwMDBCLCAweDlFMDAwMDBCLCAweEEyMDAwMDBCLCAweEE2MDAwMDBCLCAweDVDMDAwMUFBLCAweDNFMDAwMUFBLCAweDdFMDAwMUFBLCAKLSAgICAgICAgMHgwNjAwMDAyQiwgMHgwQTAwMDAyQiwgMHgyQTAwMDAyQiwgMHg0RTAwMDAyQiwgMHgwMDAwMDAxOQotICAgIH07Ci19CmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL2V4ZWN1dGFibGVwYXRoX2Rhcndpbi5jcHAgYi9saWJzL3V0aWxzL2V4ZWN1dGFibGVwYXRoX2Rhcndpbi5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJlM2MzYTAuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9leGVjdXRhYmxlcGF0aF9kYXJ3aW4uY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMzEgKzAsMCBAQAotLyoKLSAqIENvcHlyaWdodCAoQykgMjAwOCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKgotICogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7Ci0gKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKLSAqCi0gKiAgICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAotICoKLSAqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKLSAqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCi0gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKLSAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDx1dGlscy9leGVjdXRhYmxlcGF0aC5oPgotI2ltcG9ydCA8Q2FyYm9uL0NhcmJvbi5oPgotI2luY2x1ZGUgPHVuaXN0ZC5oPgotCi12b2lkIGV4ZWN1dGFibGVwYXRoKGNoYXIgc1tQQVRIX01BWF0pCi17Ci0gICAgUHJvY2Vzc1NlcmlhbE51bWJlciBwc247Ci0gICAgR2V0Q3VycmVudFByb2Nlc3MoJnBzbik7Ci0gICAgQ0ZEaWN0aW9uYXJ5UmVmIGRpY3Q7Ci0gICAgZGljdCA9IFByb2Nlc3NJbmZvcm1hdGlvbkNvcHlEaWN0aW9uYXJ5KCZwc24sIDB4ZmZmZmZmZmYpOwotICAgIENGU3RyaW5nUmVmIHZhbHVlID0gKENGU3RyaW5nUmVmKUNGRGljdGlvbmFyeUdldFZhbHVlKGRpY3QsCi0gICAgICAgICAgICAgICAgQ0ZTVFIoIkNGQnVuZGxlRXhlY3V0YWJsZSIpKTsKLSAgICBDRlN0cmluZ0dldENTdHJpbmcodmFsdWUsIHMsIFBBVEhfTUFYKzEsIGtDRlN0cmluZ0VuY29kaW5nVVRGOCk7Ci19Ci0KZGlmZiAtLWdpdCBhL2xpYnMvdXRpbHMvZXhlY3V0YWJsZXBhdGhfbGludXguY3BwIGIvbGlicy91dGlscy9leGVjdXRhYmxlcGF0aF9saW51eC5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGI4ZDJhM2QuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9leGVjdXRhYmxlcGF0aF9saW51eC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwzMCArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2luY2x1ZGUgPHV0aWxzL2V4ZWN1dGFibGVwYXRoLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8bGltaXRzLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLQotdm9pZCBleGVjdXRhYmxlcGF0aChjaGFyIGV4ZVtQQVRIX01BWF0pCi17Ci0gICAgY2hhciBwcm9jWzEwMF07Ci0gICAgc3ByaW50Zihwcm9jLCAiL3Byb2MvJWQvZXhlIiwgZ2V0cGlkKCkpOwotICAgIAotICAgIGludCBlcnIgPSByZWFkbGluayhwcm9jLCBleGUsIFBBVEhfTUFYKTsKLX0KLQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9mdXRleF9zeW5jaHJvLmMgYi9saWJzL3V0aWxzL2Z1dGV4X3N5bmNocm8uYwpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYmExOTUyMC4uMDAwMDAwMAotLS0gYS9saWJzL3V0aWxzL2Z1dGV4X3N5bmNocm8uYworKysgL2Rldi9udWxsCkBAIC0xLDE3NSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA4IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8bGltaXRzLmg+Ci0KLSNpbmNsdWRlIDxzeXMvdGltZS5oPgotI2luY2x1ZGUgPHNjaGVkLmg+Ci0KLSNpbmNsdWRlIDxlcnJuby5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS91dGlscy9mdXRleF9zeW5jaHJvLmg+Ci0KLQotLy8gVGhpcyBmdXRleCBnbHVlIGNvZGUgaXMgbmVlZCBvbiBkZXNrdG9wIGxpbnV4LCBidXQgaXMgYWxyZWFkeSBwYXJ0IG9mIGJpb25pYy4KLSNpZiAhZGVmaW5lZChIQVZFX0ZVVEVYX1dSQVBQRVJTKQotCi0jaW5jbHVkZSA8c3lzL3N5c2NhbGwuaD4KLXR5cGVkZWYgdW5zaWduZWQgaW50IHUzMjsKLSNkZWZpbmUgYXNtbGlua2FnZQotI2RlZmluZSBfX3VzZXIKLSNpbmNsdWRlIDxsaW51eC9mdXRleC5oPgotI2luY2x1ZGUgPHV0aWxzL0F0b21pYy5oPgotCi0KLWludCBmdXRleCAoaW50ICp1YWRkciwgaW50IG9wLCBpbnQgdmFsLCBjb25zdCBzdHJ1Y3QgdGltZXNwZWMgKnRpbWVvdXQsIGludCAqdWFkZHIyLCBpbnQgdmFsMykKLXsKLSAgICBpbnQgZXJyID0gc3lzY2FsbChTWVNfZnV0ZXgsIHVhZGRyLCBvcCwgdmFsLCB0aW1lb3V0LCB1YWRkcjIsIHZhbDMpOwotICAgIHJldHVybiBlcnIgPT0gMCA/IDAgOiAtZXJybm87Ci19Ci0KLWludCBfX2Z1dGV4X3dhaXQodm9sYXRpbGUgdm9pZCAqZnR4LCBpbnQgdmFsLCBjb25zdCBzdHJ1Y3QgdGltZXNwZWMgKnRpbWVvdXQpCi17Ci0gICAgcmV0dXJuIGZ1dGV4KChpbnQqKWZ0eCwgRlVURVhfV0FJVCwgdmFsLCB0aW1lb3V0LCBOVUxMLCAwKTsKLX0KLQotaW50IF9fZnV0ZXhfd2FrZSh2b2xhdGlsZSB2b2lkICpmdHgsIGludCBjb3VudCkKLXsKLSAgICByZXR1cm4gZnV0ZXgoKGludCopZnR4LCBGVVRFWF9XQUtFLCBjb3VudCwgTlVMTCwgTlVMTCwgMCk7Ci19Ci0KLWludCBfX2F0b21pY19jbXB4Y2hnKGludCBvbGQsIGludCBfbmV3LCB2b2xhdGlsZSBpbnQgKnB0cikKLXsKLSAgICByZXR1cm4gYW5kcm9pZF9hdG9taWNfY21weGNoZyhvbGQsIF9uZXcsIHB0cik7Ci19Ci0KLWludCBfX2F0b21pY19zd2FwKGludCBfbmV3LCB2b2xhdGlsZSBpbnQgKnB0cikKLXsKLSAgICByZXR1cm4gYW5kcm9pZF9hdG9taWNfc3dhcChfbmV3LCBwdHIpOwotfQotCi1pbnQgX19hdG9taWNfZGVjKHZvbGF0aWxlIGludCAqcHRyKQotewotICAgIHJldHVybiBhbmRyb2lkX2F0b21pY19kZWMocHRyKTsKLX0KLQotI2Vsc2UgLy8gIWRlZmluZWQoX19hcm1fXykKLQotaW50IF9fZnV0ZXhfd2FpdCh2b2xhdGlsZSB2b2lkICpmdHgsIGludCB2YWwsIGNvbnN0IHN0cnVjdCB0aW1lc3BlYyAqdGltZW91dCk7Ci1pbnQgX19mdXRleF93YWtlKHZvbGF0aWxlIHZvaWQgKmZ0eCwgaW50IGNvdW50KTsKLQotaW50IF9fYXRvbWljX2NtcHhjaGcoaW50IG9sZCwgaW50IF9uZXcsIHZvbGF0aWxlIGludCAqcHRyKTsKLWludCBfX2F0b21pY19zd2FwKGludCBfbmV3LCB2b2xhdGlsZSBpbnQgKnB0cik7Ci1pbnQgX19hdG9taWNfZGVjKHZvbGF0aWxlIGludCAqcHRyKTsKLQotI2VuZGlmIC8vICFkZWZpbmVkKEhBVkVfRlVURVhfV1JBUFBFUlMpCi0KLQotLy8gbG9jayBzdGF0ZXMKLS8vCi0vLyAwOiB1bmxvY2tlZAotLy8gMTogbG9ja2VkLCBubyB3YWl0ZXJzCi0vLyAyOiBsb2NrZWQsIG1heWJlIHdhaXRlcnMKLQotdm9pZCBmdXRleF9tdXRleF9pbml0KGZ1dGV4X211dGV4X3QgKm0pCi17Ci0gICAgbS0+dmFsdWUgPSAwOwotfQotCi1pbnQgZnV0ZXhfbXV0ZXhfbG9jayhmdXRleF9tdXRleF90ICptLCB1bnNpZ25lZCBtc2VjKQotewotICAgIGlmKF9fYXRvbWljX2NtcHhjaGcoMCwgMSwgJm0tPnZhbHVlKSA9PSAwKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLSAgICBpZihtc2VjID09IEZVVEVYX1dBSVRfSU5GSU5JVEUpIHsKLSAgICAgICAgd2hpbGUoX19hdG9taWNfc3dhcCgyLCAmbS0+dmFsdWUpICE9IDApIHsKLSAgICAgICAgICAgIF9fZnV0ZXhfd2FpdCgmbS0+dmFsdWUsIDIsIDApOwotICAgICAgICB9Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgc3RydWN0IHRpbWVzcGVjIHRzOwotICAgICAgICB0cy50dl9zZWMgPSBtc2VjIC8gMTAwMDsKLSAgICAgICAgdHMudHZfbnNlYyA9IChtc2VjICUgMTAwMCkgKiAxMDAwMDAwOwotICAgICAgICB3aGlsZShfX2F0b21pY19zd2FwKDIsICZtLT52YWx1ZSkgIT0gMCkgewotICAgICAgICAgICAgaWYoX19mdXRleF93YWl0KCZtLT52YWx1ZSwgMiwgJnRzKSA9PSAtRVRJTUVET1VUKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuIC0xOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiAwOwotfQotCi1pbnQgZnV0ZXhfbXV0ZXhfdHJ5bG9jayhmdXRleF9tdXRleF90ICptKQotewotICAgIGlmKF9fYXRvbWljX2NtcHhjaGcoMCwgMSwgJm0tPnZhbHVlKSA9PSAwKSB7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLSAgICByZXR1cm4gLTE7Ci19Ci0KLXZvaWQgZnV0ZXhfbXV0ZXhfdW5sb2NrKGZ1dGV4X211dGV4X3QgKm0pCi17Ci0gICAgaWYoX19hdG9taWNfZGVjKCZtLT52YWx1ZSkgIT0gMSkgewotICAgICAgICBtLT52YWx1ZSA9IDA7Ci0gICAgICAgIF9fZnV0ZXhfd2FrZSgmbS0+dmFsdWUsIDEpOwotICAgIH0KLX0KLQotLyogWFhYICp0ZWNobmljYWxseSogdGhlcmUgaXMgYSByYWNlIGNvbmRpdGlvbiB0aGF0IGNvdWxkIGFsbG93Ci0gKiBYWFggYSBzaWduYWwgdG8gYmUgbWlzc2VkLiAgSWYgdGhyZWFkIEEgaXMgcHJlZW1wdGVkIGluIF93YWl0KCkKLSAqIFhYWCBhZnRlciB1bmxvY2tpbmcgdGhlIG11dGV4IGFuZCBiZWZvcmUgd2FpdGluZywgYW5kIGlmIG90aGVyCi0gKiBYWFggdGhyZWFkcyBjYWxsIHNpZ25hbCBvciBicm9hZGNhc3QgVUlOVF9NQVggdGltZXMgKGV4YWN0bHkpLAotICogWFhYIGJlZm9yZSB0aHJlYWQgQSBpcyBzY2hlZHVsZWQgYWdhaW4gYW5kIGNhbGxzIGZ1dGV4X3dhaXQoKSwKLSAqIFhYWCB0aGVuIHRoZSBzaWduYWwgd2lsbCBiZSBsb3N0LgotICovCi0KLXZvaWQgZnV0ZXhfY29uZF9pbml0KGZ1dGV4X2NvbmRfdCAqYykKLXsKLSAgICBjLT52YWx1ZSA9IDA7Ci19Ci0KLWludCBmdXRleF9jb25kX3dhaXQoZnV0ZXhfY29uZF90ICpjLCBmdXRleF9tdXRleF90ICptLCB1bnNpZ25lZCBtc2VjKQotewotICAgIGlmKG1zZWMgPT0gRlVURVhfV0FJVF9JTkZJTklURSl7Ci0gICAgICAgIGludCBvbGR2YWx1ZSA9IGMtPnZhbHVlOwotICAgICAgICBmdXRleF9tdXRleF91bmxvY2sobSk7Ci0gICAgICAgIF9fZnV0ZXhfd2FpdCgmYy0+dmFsdWUsIG9sZHZhbHVlLCAwKTsKLSAgICAgICAgZnV0ZXhfbXV0ZXhfbG9jayhtLCBGVVRFWF9XQUlUX0lORklOSVRFKTsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgaW50IG9sZHZhbHVlID0gYy0+dmFsdWU7Ci0gICAgICAgIHN0cnVjdCB0aW1lc3BlYyB0czsgICAgICAgIAotICAgICAgICB0cy50dl9zZWMgPSBtc2VjIC8gMTAwMDsKLSAgICAgICAgdHMudHZfbnNlYyA9IChtc2VjICUgMTAwMCkgKiAxMDAwMDAwOwotICAgICAgICBmdXRleF9tdXRleF91bmxvY2sobSk7Ci0gICAgICAgIGNvbnN0IGludCBlcnIgPSBfX2Z1dGV4X3dhaXQoJmMtPnZhbHVlLCBvbGR2YWx1ZSwgJnRzKTsKLSAgICAgICAgZnV0ZXhfbXV0ZXhfbG9jayhtLCBGVVRFWF9XQUlUX0lORklOSVRFKTsKLSAgICAgICAgcmV0dXJuIGVycjsKLSAgICB9Ci19Ci0KLXZvaWQgZnV0ZXhfY29uZF9zaWduYWwoZnV0ZXhfY29uZF90ICpjKQotewotICAgIF9fYXRvbWljX2RlYygmYy0+dmFsdWUpOwotICAgIF9fZnV0ZXhfd2FrZSgmYy0+dmFsdWUsIDEpOwotfQotCi12b2lkIGZ1dGV4X2NvbmRfYnJvYWRjYXN0KGZ1dGV4X2NvbmRfdCAqYykKLXsKLSAgICBfX2F0b21pY19kZWMoJmMtPnZhbHVlKTsKLSAgICBfX2Z1dGV4X3dha2UoJmMtPnZhbHVlLCBJTlRfTUFYKTsKLX0KLQpkaWZmIC0tZ2l0IGEvbGlicy91dGlscy9taXNjLmNwcCBiL2xpYnMvdXRpbHMvbWlzYy5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGRjODlkMTUuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9taXNjLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDE4NSArMCwwIEBACi0vKgotICogQ29weXJpZ2h0IChDKSAyMDA1IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqCi0gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KLSAqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAotICoKLSAqICAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKgotICogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQotICogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgotICogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZAotICogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotLy8KLS8vIE1pc2NlbGxhbmVvdXMgdXRpbGl0eSBmdW5jdGlvbnMuCi0vLwotI2luY2x1ZGUgPHV0aWxzL21pc2MuaD4KLQotI2luY2x1ZGUgPHN5cy9zdGF0Lmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxhc3NlcnQuaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vKgotICogTGlrZSBzdHJkdXAoKSwgYnV0IHVzZXMgQysrICJuZXciIG9wZXJhdG9yIGluc3RlYWQgb2YgbWFsbG9jLgotICovCi1jaGFyKiBzdHJkdXBOZXcoY29uc3QgY2hhciogc3RyKQotewotICAgIGNoYXIqIG5ld1N0cjsKLSAgICBpbnQgbGVuOwotCi0gICAgaWYgKHN0ciA9PSBOVUxMKQotICAgICAgICByZXR1cm4gTlVMTDsKLQotICAgIGxlbiA9IHN0cmxlbihzdHIpOwotICAgIG5ld1N0ciA9IG5ldyBjaGFyW2xlbisxXTsKLSAgICBtZW1jcHkobmV3U3RyLCBzdHIsIGxlbisxKTsKLQotICAgIHJldHVybiBuZXdTdHI7Ci19Ci0KLS8qCi0gKiBDb25jYXRlbmF0ZSBhbiBhcmd1bWVudCB2ZWN0b3IuCi0gKi8KLWNoYXIqIGNvbmNhdEFyZ3YoaW50IGFyZ2MsIGNvbnN0IGNoYXIqIGNvbnN0IGFyZ3ZbXSkKLXsKLSAgICBjaGFyKiBuZXdTdHIgPSBOVUxMOwotICAgIGludCBsZW4sIHRvdGFsTGVuLCBwb3NuLCBpZHg7Ci0KLSAgICAvKgotICAgICAqIEZpcnN0LCBmaWd1cmUgb3V0IHRoZSB0b3RhbCBsZW5ndGguCi0gICAgICovCi0gICAgdG90YWxMZW4gPSBpZHggPSAwOwotICAgIHdoaWxlICgxKSB7Ci0gICAgICAgIGlmIChpZHggPT0gYXJnYyB8fCBhcmd2W2lkeF0gPT0gTlVMTCkKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBpZiAoaWR4KQotICAgICAgICAgICAgdG90YWxMZW4rKzsgIC8vIGxlYXZlIGEgc3BhY2UgYmV0d2VlbiBhcmdzCi0gICAgICAgIHRvdGFsTGVuICs9IHN0cmxlbihhcmd2W2lkeF0pOwotICAgICAgICBpZHgrKzsKLSAgICB9Ci0KLSAgICAvKgotICAgICAqIEFsbG9jIHRoZSBzdHJpbmcuCi0gICAgICovCi0gICAgbmV3U3RyID0gbmV3IGNoYXJbdG90YWxMZW4gKzFdOwotICAgIGlmIChuZXdTdHIgPT0gTlVMTCkKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0KLSAgICAvKgotICAgICAqIEZpbmFsbHksIGFsbG9jYXRlIHRoZSBzdHJpbmcgYW5kIGNvcHkgZGF0YSBvdmVyLgotICAgICAqLwotICAgIGlkeCA9IHBvc24gPSAwOwotICAgIHdoaWxlICgxKSB7Ci0gICAgICAgIGlmIChpZHggPT0gYXJnYyB8fCBhcmd2W2lkeF0gPT0gTlVMTCkKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBpZiAoaWR4KQotICAgICAgICAgICAgbmV3U3RyW3Bvc24rK10gPSAnICc7Ci0KLSAgICAgICAgbGVuID0gc3RybGVuKGFyZ3ZbaWR4XSk7Ci0gICAgICAgIG1lbWNweSgmbmV3U3RyW3Bvc25dLCBhcmd2W2lkeF0sIGxlbik7Ci0gICAgICAgIHBvc24gKz0gbGVuOwotCi0gICAgICAgIGlkeCsrOwotICAgIH0KLQotICAgIGFzc2VydChwb3NuID09IHRvdGFsTGVuKTsKLSAgICBuZXdTdHJbcG9zbl0gPSAnXDAnOwotCi0gICAgcmV0dXJuIG5ld1N0cjsKLX0KLQotLyoKLSAqIENvdW50IHRoZSAjb2YgYXJncyBpbiBhbiBhcmd1bWVudCB2ZWN0b3IuICBEb24ndCBjb3VudCB0aGUgZmluYWwgTlVMTC4KLSAqLwotaW50IGNvdW50QXJndihjb25zdCBjaGFyKiBjb25zdCBhcmd2W10pCi17Ci0gICAgaW50IGNvdW50ID0gMDsKLQotICAgIHdoaWxlIChhcmd2W2NvdW50XSAhPSBOVUxMKQotICAgICAgICBjb3VudCsrOwotCi0gICAgcmV0dXJuIGNvdW50OwotfQotCi0KLSNpbmNsdWRlIDxzdGRpby5oPgotLyoKLSAqIEdldCBhIGZpbGUncyB0eXBlLgotICovCi1GaWxlVHlwZSBnZXRGaWxlVHlwZShjb25zdCBjaGFyKiBmaWxlTmFtZSkKLXsKLSAgICBzdHJ1Y3Qgc3RhdCBzYjsKLQotICAgIGlmIChzdGF0KGZpbGVOYW1lLCAmc2IpIDwgMCkgewotICAgICAgICBpZiAoZXJybm8gPT0gRU5PRU5UIHx8IGVycm5vID09IEVOT1RESVIpCi0gICAgICAgICAgICByZXR1cm4ga0ZpbGVUeXBlTm9uZXhpc3RlbnQ7Ci0gICAgICAgIGVsc2UgewotICAgICAgICAgICAgZnByaW50ZihzdGRlcnIsICJnZXRGaWxlVHlwZSBnb3QgZXJybm89JWQgb24gJyVzJ1xuIiwKLSAgICAgICAgICAgICAgICBlcnJubywgZmlsZU5hbWUpOwotICAgICAgICAgICAgcmV0dXJuIGtGaWxlVHlwZVVua25vd247Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBpZiAoU19JU1JFRyhzYi5zdF9tb2RlKSkKLSAgICAgICAgICAgIHJldHVybiBrRmlsZVR5cGVSZWd1bGFyOwotICAgICAgICBlbHNlIGlmIChTX0lTRElSKHNiLnN0X21vZGUpKQotICAgICAgICAgICAgcmV0dXJuIGtGaWxlVHlwZURpcmVjdG9yeTsKLSAgICAgICAgZWxzZSBpZiAoU19JU0NIUihzYi5zdF9tb2RlKSkKLSAgICAgICAgICAgIHJldHVybiBrRmlsZVR5cGVDaGFyRGV2OwotICAgICAgICBlbHNlIGlmIChTX0lTQkxLKHNiLnN0X21vZGUpKQotICAgICAgICAgICAgcmV0dXJuIGtGaWxlVHlwZUJsb2NrRGV2OwotICAgICAgICBlbHNlIGlmIChTX0lTRklGTyhzYi5zdF9tb2RlKSkKLSAgICAgICAgICAgIHJldHVybiBrRmlsZVR5cGVGaWZvOwotI2lmZGVmIEhBVkVfU1lNTElOS1MgICAgICAgICAgICAKLSAgICAgICAgZWxzZSBpZiAoU19JU0xOSyhzYi5zdF9tb2RlKSkKLSAgICAgICAgICAgIHJldHVybiBrRmlsZVR5cGVTeW1saW5rOwotICAgICAgICBlbHNlIGlmIChTX0lTU09DSyhzYi5zdF9tb2RlKSkKLSAgICAgICAgICAgIHJldHVybiBrRmlsZVR5cGVTb2NrZXQ7Ci0jZW5kaWYgICAgICAgICAgICAKLSAgICAgICAgZWxzZQotICAgICAgICAgICAgcmV0dXJuIGtGaWxlVHlwZVVua25vd247Ci0gICAgfQotfQotCi0vKgotICogR2V0IGEgZmlsZSdzIG1vZGlmaWNhdGlvbiBkYXRlLgotICovCi10aW1lX3QgZ2V0RmlsZU1vZERhdGUoY29uc3QgY2hhciogZmlsZU5hbWUpCi17Ci0gICAgc3RydWN0IHN0YXQgc2I7Ci0KLSAgICBpZiAoc3RhdChmaWxlTmFtZSwgJnNiKSA8IDApCi0gICAgICAgIHJldHVybiAodGltZV90KSAtMTsKLQotICAgIHJldHVybiBzYi5zdF9tdGltZTsKLX0KLQotLyoKLSAqIFJvdW5kIHVwIHRvIHRoZSBuZXh0IGhpZ2hlc3QgcG93ZXIgb2YgMi4KLSAqCi0gKiBGb3VuZCBvbiBodHRwOi8vZ3JhcGhpY3Muc3RhbmZvcmQuZWR1L35zZWFuZGVyL2JpdGhhY2tzLmh0bWwuCi0gKi8KLXVuc2lnbmVkIGludCByb3VuZFVwUG93ZXIyKHVuc2lnbmVkIGludCB2YWwpCi17Ci0gICAgdmFsLS07Ci0gICAgdmFsIHw9IHZhbCA+PiAxOwotICAgIHZhbCB8PSB2YWwgPj4gMjsKLSAgICB2YWwgfD0gdmFsID4+IDQ7Ci0gICAgdmFsIHw9IHZhbCA+PiA4OwotICAgIHZhbCB8PSB2YWwgPj4gMTY7Ci0gICAgdmFsKys7Ci0KLSAgICByZXR1cm4gdmFsOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCmRpZmYgLS1naXQgYS9saWJzL3V0aWxzL3BvcnRlZC5jcHAgYi9saWJzL3V0aWxzL3BvcnRlZC5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDY1NmU0NmYuLjAwMDAwMDAKLS0tIGEvbGlicy91dGlscy9wb3J0ZWQuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTA2ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDUgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0vLwotLy8gUG9ydHMgb2Ygc3RhbmRhcmQgZnVuY3Rpb25zIHRoYXQgZG9uJ3QgZXhpc3Qgb24gYSBzcGVjaWZpYyBwbGF0Zm9ybS4KLS8vCi0vLyBOb3RlIHRoZXNlIGFyZSBOT1QgaW4gdGhlICJhbmRyb2lkIiBuYW1lc3BhY2UuCi0vLwotI2luY2x1ZGUgPHV0aWxzL3BvcnRlZC5oPgotCi0jaWYgZGVmaW5lZChORUVEX0dFVFRJTUVPRkRBWSkgfHwgZGVmaW5lZChORUVEX1VTTEVFUCkKLSMgaW5jbHVkZSA8c3lzL3RpbWUuaD4KLSMgaW5jbHVkZSA8d2luZG93cy5oPgotI2VuZGlmCi0KLQotI2lmIGRlZmluZWQoTkVFRF9HRVRUSU1FT0ZEQVkpCi0vKgotICogUmVwbGFjZW1lbnQgZ2V0dGltZW9mZGF5KCkgZm9yIFdpbmRvd3MgZW52aXJvbm1lbnRzIChwcmltYXJpbHkgTWluR1cpLgotICoKLSAqIElnbm9yZXMgInR6Ii4KLSAqLwotaW50IGdldHRpbWVvZmRheShzdHJ1Y3QgdGltZXZhbCogcHR2LCBzdHJ1Y3QgdGltZXpvbmUqIHR6KQotewotICAgIGxvbmcgbG9uZyBuc1RpbWU7ICAgLy8gdGltZSBpbiAxMDBucyB1bml0cyBzaW5jZSBKYW4gMSAxNjAxCi0gICAgRklMRVRJTUUgZnQ7Ci0KLSAgICBpZiAodHogIT0gTlVMTCkgewotICAgICAgICAvLyBvaCB3ZWxsCi0gICAgfQotCi0gICAgOjpHZXRTeXN0ZW1UaW1lQXNGaWxlVGltZSgmZnQpOwotICAgIG5zVGltZSA9IChsb25nIGxvbmcpIGZ0LmR3SGlnaERhdGVUaW1lIDw8IDMyIHwKLSAgICAgICAgICAgICAobG9uZyBsb25nKSBmdC5kd0xvd0RhdGVUaW1lOwotICAgIC8vIGNvbnZlcnQgdG8gdGltZSBpbiB1c2VjIHNpbmNlIEphbiAxIDE5NzAKLSAgICBwdHYtPnR2X3VzZWMgPSAobG9uZykgKChuc1RpbWUgLyAxMExMKSAlIDEwMDAwMDBMTCk7Ci0gICAgcHR2LT50dl9zZWMgPSAobG9uZykgKChuc1RpbWUgLSAxMTY0NDQ3MzYwMDAwMDAwMDBMTCkgLyAxMDAwMDAwMExMKTsKLQotICAgIHJldHVybiAwOwotfQotI2VuZGlmCi0KLSNpZiBkZWZpbmVkKE5FRURfVVNMRUVQKQotLy8KLS8vIFJlcGxhY2VtZW50IHVzbGVlcCBmb3IgV2luZG93cyBlbnZpcm9ubWVudHMgKHByaW1hcmlseSBNaW5HVykuCi0vLwotdm9pZCB1c2xlZXAodW5zaWduZWQgbG9uZyB1c2VjKQotewotICAgIC8vIFdpbjMyIEFQSSBmdW5jdGlvbiBTbGVlcCgpIHRha2VzIG1pbGxpc2Vjb25kcwotICAgIDo6U2xlZXAoKHVzZWMgKyA1MDApIC8gMTAwMCk7Ci19Ci0jZW5kaWYKLQotI2lmIDAgLy9kZWZpbmVkKE5FRURfUElQRSkKLS8vCi0vLyBSZXBsYWNlbWVudCBwaXBlKCkgY29tbWFuZCBmb3IgTWluR1cKLS8vCi0vLyBUaGUgX09fTk9JTkhFUklUIGZsYWcgc2V0cyBiSW5oZXJpdEhhbmRsZSB0byBGQUxTRSBpbiB0aGUKLS8vIFNlY3VyaXR5QXR0cmlidXRlcyBhcmd1bWVudCB0byBDcmVhdGVQaXBlKCkuICBUaGlzIG1lYW5zIHRoZSBoYW5kbGVzCi0vLyBhcmVuJ3QgaW5oZXJpdGVkIHdoZW4gYSBuZXcgcHJvY2VzcyBpcyBjcmVhdGVkLiAgVGhlIGV4YW1wbGVzIEkndmUgc2VlbgotLy8gdXNlIGl0LCBwb3NzaWJseSBiZWNhdXNlIHRoZXJlJ3MgYSBsb3Qgb2YganVuayBnb2luZyBvbiBiZWhpbmQgdGhlCi0vLyBzY2VuZXMuICAoSSdtIGFzc3VtaW5nICJwcm9jZXNzIiBhbmQgInRocmVhZCIgYXJlIGRpZmZlcmVudCBoZXJlLCBzbwotLy8gd2Ugc2hvdWxkIGJlIG9rYXkgc3Bpbm5pbmcgdXAgYSB0aHJlYWQuKSAgVGhlIHJlY29tbWVuZGVkIHByYWN0aWNlIGlzCi0vLyB0byBkdXAoKSB0aGUgZGVzY3JpcHRvciB5b3Ugd2FudCB0aGUgY2hpbGQgdG8gaGF2ZS4KLS8vCi0vLyBJdCBhcHBlYXJzIHRoYXQgdW5uYW1lZCBwaXBlcyBjYW4ndCBkbyBub24tYmxvY2tpbmcgKCJvdmVybGFwcGVkIikgSS9PLgotLy8gWW91IGNhbid0IHVzZSBzZWxlY3QoKSBlaXRoZXIsIHNpbmNlIHRoYXQgb25seSB3b3JrcyBvbiBzb2NrZXRzLiAgVGhlCi0vLyBXaW5kb3dzIEFQSSBjYWxscyB0aGF0IGFyZSB1c2VmdWwgaGVyZSBhbGwgb3BlcmF0ZSBvbiBhIEhBTkRMRSwgbm90Ci0vLyBhbiBpbnRlZ2VyIGZpbGUgZGVzY3JpcHRvciwgYW5kIEkgZG9uJ3QgdGhpbmsgeW91IGNhbiBnZXQgdGhlcmUgZnJvbQotLy8gaGVyZS4gIFRoZSAibmFtZWQgcGlwZSIgc3R1ZmYgaXMgaW5zYW5lLgotLy8KLWludCBwaXBlKGludCBmaWxlZGVzWzJdKQotewotICAgIHJldHVybiBfcGlwZShmaWxlZGVzLCAwLCBfT19CSU5BUlkgfCBfT19OT0lOSEVSSVQpOwotfQotI2VuZGlmCi0KLSNpZiBkZWZpbmVkKE5FRURfU0VURU5WKQotLyoKLSAqIE1pbkdXIGxhY2tzIHRoZXNlLiAgRm9yIG5vdywganVzdCBzdHViIHRoZW0gb3V0IHNvIHRoZSBjb2RlIGNvbXBpbGVzLgotICovCi1pbnQgc2V0ZW52KGNvbnN0IGNoYXIqIG5hbWUsIGNvbnN0IGNoYXIqIHZhbHVlLCBpbnQgb3ZlcndyaXRlKQotewotICAgIHJldHVybiAwOwotfQotdm9pZCB1bnNldGVudihjb25zdCBjaGFyKiBuYW1lKQotewotfQotY2hhciogZ2V0ZW52KGNvbnN0IGNoYXIqIG5hbWUpCi17Ci0gICAgcmV0dXJuIE5VTEw7Ci19Ci0jZW5kaWYKZGlmZiAtLWdpdCBhL29wZW5nbC9pbmNsdWRlL0VHTC9lZ2wuaCBiL29wZW5nbC9pbmNsdWRlL0VHTC9lZ2wuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYzI2OTk3Ni4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvaW5jbHVkZS9FR0wvZWdsLmgKKysrIC9kZXYvbnVsbApAQCAtMSwzMzAgKzAsMCBAQAotLyogLSotIG1vZGU6IGM7IHRhYi13aWR0aDogODsgLSotICovCi0vKiB2aTogc2V0IHN3PTQgdHM9ODogKi8KLS8qIFJlZmVyZW5jZSB2ZXJzaW9uIG9mIGVnbC5oIGZvciBFR0wgMS40LgotICogJFJldmlzaW9uOiA3MjQ0ICQgb24gJERhdGU6IDIwMDktMDEtMjAgMTc6MDY6NTkgLTA4MDAgKFR1ZSwgMjAgSmFuIDIwMDkpICQKLSAqLwotCi0vKgotKiogQ29weXJpZ2h0IChjKSAyMDA3LTIwMDkgVGhlIEtocm9ub3MgR3JvdXAgSW5jLgotKioKLSoqIFBlcm1pc3Npb24gaXMgaGVyZWJ5IGdyYW50ZWQsIGZyZWUgb2YgY2hhcmdlLCB0byBhbnkgcGVyc29uIG9idGFpbmluZyBhCi0qKiBjb3B5IG9mIHRoaXMgc29mdHdhcmUgYW5kL29yIGFzc29jaWF0ZWQgZG9jdW1lbnRhdGlvbiBmaWxlcyAodGhlCi0qKiAiTWF0ZXJpYWxzIiksIHRvIGRlYWwgaW4gdGhlIE1hdGVyaWFscyB3aXRob3V0IHJlc3RyaWN0aW9uLCBpbmNsdWRpbmcKLSoqIHdpdGhvdXQgbGltaXRhdGlvbiB0aGUgcmlnaHRzIHRvIHVzZSwgY29weSwgbW9kaWZ5LCBtZXJnZSwgcHVibGlzaCwKLSoqIGRpc3RyaWJ1dGUsIHN1YmxpY2Vuc2UsIGFuZC9vciBzZWxsIGNvcGllcyBvZiB0aGUgTWF0ZXJpYWxzLCBhbmQgdG8KLSoqIHBlcm1pdCBwZXJzb25zIHRvIHdob20gdGhlIE1hdGVyaWFscyBhcmUgZnVybmlzaGVkIHRvIGRvIHNvLCBzdWJqZWN0IHRvCi0qKiB0aGUgZm9sbG93aW5nIGNvbmRpdGlvbnM6Ci0qKgotKiogVGhlIGFib3ZlIGNvcHlyaWdodCBub3RpY2UgYW5kIHRoaXMgcGVybWlzc2lvbiBub3RpY2Ugc2hhbGwgYmUgaW5jbHVkZWQKLSoqIGluIGFsbCBjb3BpZXMgb3Igc3Vic3RhbnRpYWwgcG9ydGlvbnMgb2YgdGhlIE1hdGVyaWFscy4KLSoqCi0qKiBUSEUgTUFURVJJQUxTIEFSRSBQUk9WSURFRCAiQVMgSVMiLCBXSVRIT1VUIFdBUlJBTlRZIE9GIEFOWSBLSU5ELAotKiogRVhQUkVTUyBPUiBJTVBMSUVELCBJTkNMVURJTkcgQlVUIE5PVCBMSU1JVEVEIFRPIFRIRSBXQVJSQU5USUVTIE9GCi0qKiBNRVJDSEFOVEFCSUxJVFksIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFIEFORCBOT05JTkZSSU5HRU1FTlQuCi0qKiBJTiBOTyBFVkVOVCBTSEFMTCBUSEUgQVVUSE9SUyBPUiBDT1BZUklHSFQgSE9MREVSUyBCRSBMSUFCTEUgRk9SIEFOWQotKiogQ0xBSU0sIERBTUFHRVMgT1IgT1RIRVIgTElBQklMSVRZLCBXSEVUSEVSIElOIEFOIEFDVElPTiBPRiBDT05UUkFDVCwKLSoqIFRPUlQgT1IgT1RIRVJXSVNFLCBBUklTSU5HIEZST00sIE9VVCBPRiBPUiBJTiBDT05ORUNUSU9OIFdJVEggVEhFCi0qKiBNQVRFUklBTFMgT1IgVEhFIFVTRSBPUiBPVEhFUiBERUFMSU5HUyBJTiBUSEUgTUFURVJJQUxTLgotKi8KLQotI2lmbmRlZiBfX2VnbF9oXwotI2RlZmluZSBfX2VnbF9oXwotCi0vKiBBbGwgcGxhdGZvcm0tZGVwZW5kZW50IHR5cGVzIGFuZCBtYWNybyBib2lsZXJwbGF0ZSAoc3VjaCBhcyBFR0xBUEkKLSAqIGFuZCBFR0xBUElFTlRSWSkgc2hvdWxkIGdvIGluIGVnbHBsYXRmb3JtLmguCi0gKi8KLSNpbmNsdWRlIDxFR0wvZWdscGxhdGZvcm0uaD4KLQotI2lmZGVmIF9fY3BsdXNwbHVzCi1leHRlcm4gIkMiIHsKLSNlbmRpZgotCi0vKiBFR0wgVHlwZXMgKi8KLS8qIEVHTGludCBpcyBkZWZpbmVkIGluIGVnbHBsYXRmb3JtLmggKi8KLXR5cGVkZWYgdW5zaWduZWQgaW50IEVHTEJvb2xlYW47Ci10eXBlZGVmIHVuc2lnbmVkIGludCBFR0xlbnVtOwotdHlwZWRlZiB2b2lkICpFR0xDb25maWc7Ci10eXBlZGVmIHZvaWQgKkVHTENvbnRleHQ7Ci10eXBlZGVmIHZvaWQgKkVHTERpc3BsYXk7Ci10eXBlZGVmIHZvaWQgKkVHTFN1cmZhY2U7Ci10eXBlZGVmIHZvaWQgKkVHTENsaWVudEJ1ZmZlcjsKLQotLyogRUdMIFZlcnNpb25pbmcgKi8KLSNkZWZpbmUgRUdMX1ZFUlNJT05fMV8wCQkJMQotI2RlZmluZSBFR0xfVkVSU0lPTl8xXzEJCQkxCi0jZGVmaW5lIEVHTF9WRVJTSU9OXzFfMgkJCTEKLSNkZWZpbmUgRUdMX1ZFUlNJT05fMV8zCQkJMQotI2RlZmluZSBFR0xfVkVSU0lPTl8xXzQJCQkxCi0KLS8qIEVHTCBFbnVtZXJhbnRzLiBCaXRtYXNrcyBhbmQgb3RoZXIgZXhjZXB0aW9uYWwgY2FzZXMgYXNpZGUsIG1vc3QKLSAqIGVudW1zIGFyZSBhc3NpZ25lZCB1bmlxdWUgdmFsdWVzIHN0YXJ0aW5nIGF0IDB4MzAwMC4KLSAqLwotCi0vKiBFR0wgYWxpYXNlcyAqLwotI2RlZmluZSBFR0xfRkFMU0UJCQkwCi0jZGVmaW5lIEVHTF9UUlVFCQkJMQotCi0vKiBPdXQtb2YtYmFuZCBoYW5kbGUgdmFsdWVzICovCi0jZGVmaW5lIEVHTF9ERUZBVUxUX0RJU1BMQVkJCSgoRUdMTmF0aXZlRGlzcGxheVR5cGUpMCkKLSNkZWZpbmUgRUdMX05PX0NPTlRFWFQJCQkoKEVHTENvbnRleHQpMCkKLSNkZWZpbmUgRUdMX05PX0RJU1BMQVkJCQkoKEVHTERpc3BsYXkpMCkKLSNkZWZpbmUgRUdMX05PX1NVUkZBQ0UJCQkoKEVHTFN1cmZhY2UpMCkKLQotLyogT3V0LW9mLWJhbmQgYXR0cmlidXRlIHZhbHVlICovCi0jZGVmaW5lIEVHTF9ET05UX0NBUkUJCQkoKEVHTGludCktMSkKLQotLyogRXJyb3JzIC8gR2V0RXJyb3IgcmV0dXJuIHZhbHVlcyAqLwotI2RlZmluZSBFR0xfU1VDQ0VTUwkJCTB4MzAwMAotI2RlZmluZSBFR0xfTk9UX0lOSVRJQUxJWkVECQkweDMwMDEKLSNkZWZpbmUgRUdMX0JBRF9BQ0NFU1MJCQkweDMwMDIKLSNkZWZpbmUgRUdMX0JBRF9BTExPQwkJCTB4MzAwMwotI2RlZmluZSBFR0xfQkFEX0FUVFJJQlVURQkJMHgzMDA0Ci0jZGVmaW5lIEVHTF9CQURfQ09ORklHCQkJMHgzMDA1Ci0jZGVmaW5lIEVHTF9CQURfQ09OVEVYVAkJCTB4MzAwNgotI2RlZmluZSBFR0xfQkFEX0NVUlJFTlRfU1VSRkFDRQkJMHgzMDA3Ci0jZGVmaW5lIEVHTF9CQURfRElTUExBWQkJCTB4MzAwOAotI2RlZmluZSBFR0xfQkFEX01BVENICQkJMHgzMDA5Ci0jZGVmaW5lIEVHTF9CQURfTkFUSVZFX1BJWE1BUAkJMHgzMDBBCi0jZGVmaW5lIEVHTF9CQURfTkFUSVZFX1dJTkRPVwkJMHgzMDBCCi0jZGVmaW5lIEVHTF9CQURfUEFSQU1FVEVSCQkweDMwMEMKLSNkZWZpbmUgRUdMX0JBRF9TVVJGQUNFCQkJMHgzMDBECi0jZGVmaW5lIEVHTF9DT05URVhUX0xPU1QJCTB4MzAwRQkvKiBFR0wgMS4xIC0gSU1HX3Bvd2VyX21hbmFnZW1lbnQgKi8KLQotLyogUmVzZXJ2ZWQgMHgzMDBGLTB4MzAxRiBmb3IgYWRkaXRpb25hbCBlcnJvcnMgKi8KLQotLyogQ29uZmlnIGF0dHJpYnV0ZXMgKi8KLSNkZWZpbmUgRUdMX0JVRkZFUl9TSVpFCQkJMHgzMDIwCi0jZGVmaW5lIEVHTF9BTFBIQV9TSVpFCQkJMHgzMDIxCi0jZGVmaW5lIEVHTF9CTFVFX1NJWkUJCQkweDMwMjIKLSNkZWZpbmUgRUdMX0dSRUVOX1NJWkUJCQkweDMwMjMKLSNkZWZpbmUgRUdMX1JFRF9TSVpFCQkJMHgzMDI0Ci0jZGVmaW5lIEVHTF9ERVBUSF9TSVpFCQkJMHgzMDI1Ci0jZGVmaW5lIEVHTF9TVEVOQ0lMX1NJWkUJCTB4MzAyNgotI2RlZmluZSBFR0xfQ09ORklHX0NBVkVBVAkJMHgzMDI3Ci0jZGVmaW5lIEVHTF9DT05GSUdfSUQJCQkweDMwMjgKLSNkZWZpbmUgRUdMX0xFVkVMCQkJMHgzMDI5Ci0jZGVmaW5lIEVHTF9NQVhfUEJVRkZFUl9IRUlHSFQJCTB4MzAyQQotI2RlZmluZSBFR0xfTUFYX1BCVUZGRVJfUElYRUxTCQkweDMwMkIKLSNkZWZpbmUgRUdMX01BWF9QQlVGRkVSX1dJRFRICQkweDMwMkMKLSNkZWZpbmUgRUdMX05BVElWRV9SRU5ERVJBQkxFCQkweDMwMkQKLSNkZWZpbmUgRUdMX05BVElWRV9WSVNVQUxfSUQJCTB4MzAyRQotI2RlZmluZSBFR0xfTkFUSVZFX1ZJU1VBTF9UWVBFCQkweDMwMkYKLSNkZWZpbmUgRUdMX1BSRVNFUlZFRF9SRVNPVVJDRVMJCTB4MzAzMAotI2RlZmluZSBFR0xfU0FNUExFUwkJCTB4MzAzMQotI2RlZmluZSBFR0xfU0FNUExFX0JVRkZFUlMJCTB4MzAzMgotI2RlZmluZSBFR0xfU1VSRkFDRV9UWVBFCQkweDMwMzMKLSNkZWZpbmUgRUdMX1RSQU5TUEFSRU5UX1RZUEUJCTB4MzAzNAotI2RlZmluZSBFR0xfVFJBTlNQQVJFTlRfQkxVRV9WQUxVRQkweDMwMzUKLSNkZWZpbmUgRUdMX1RSQU5TUEFSRU5UX0dSRUVOX1ZBTFVFCTB4MzAzNgotI2RlZmluZSBFR0xfVFJBTlNQQVJFTlRfUkVEX1ZBTFVFCTB4MzAzNwotI2RlZmluZSBFR0xfTk9ORQkJCTB4MzAzOAkvKiBBdHRyaWIgbGlzdCB0ZXJtaW5hdG9yICovCi0jZGVmaW5lIEVHTF9CSU5EX1RPX1RFWFRVUkVfUkdCCQkweDMwMzkKLSNkZWZpbmUgRUdMX0JJTkRfVE9fVEVYVFVSRV9SR0JBCTB4MzAzQQotI2RlZmluZSBFR0xfTUlOX1NXQVBfSU5URVJWQUwJCTB4MzAzQgotI2RlZmluZSBFR0xfTUFYX1NXQVBfSU5URVJWQUwJCTB4MzAzQwotI2RlZmluZSBFR0xfTFVNSU5BTkNFX1NJWkUJCTB4MzAzRAotI2RlZmluZSBFR0xfQUxQSEFfTUFTS19TSVpFCQkweDMwM0UKLSNkZWZpbmUgRUdMX0NPTE9SX0JVRkZFUl9UWVBFCQkweDMwM0YKLSNkZWZpbmUgRUdMX1JFTkRFUkFCTEVfVFlQRQkJMHgzMDQwCi0jZGVmaW5lIEVHTF9NQVRDSF9OQVRJVkVfUElYTUFQCQkweDMwNDEJLyogUHNldWRvLWF0dHJpYnV0ZSAobm90IHF1ZXJ5YWJsZSkgKi8KLSNkZWZpbmUgRUdMX0NPTkZPUk1BTlQJCQkweDMwNDIKLQotLyogUmVzZXJ2ZWQgMHgzMDQxLTB4MzA0RiBmb3IgYWRkaXRpb25hbCBjb25maWcgYXR0cmlidXRlcyAqLwotCi0vKiBDb25maWcgYXR0cmlidXRlIHZhbHVlcyAqLwotI2RlZmluZSBFR0xfU0xPV19DT05GSUcJCQkweDMwNTAJLyogRUdMX0NPTkZJR19DQVZFQVQgdmFsdWUgKi8KLSNkZWZpbmUgRUdMX05PTl9DT05GT1JNQU5UX0NPTkZJRwkweDMwNTEJLyogRUdMX0NPTkZJR19DQVZFQVQgdmFsdWUgKi8KLSNkZWZpbmUgRUdMX1RSQU5TUEFSRU5UX1JHQgkJMHgzMDUyCS8qIEVHTF9UUkFOU1BBUkVOVF9UWVBFIHZhbHVlICovCi0jZGVmaW5lIEVHTF9SR0JfQlVGRkVSCQkJMHgzMDhFCS8qIEVHTF9DT0xPUl9CVUZGRVJfVFlQRSB2YWx1ZSAqLwotI2RlZmluZSBFR0xfTFVNSU5BTkNFX0JVRkZFUgkJMHgzMDhGCS8qIEVHTF9DT0xPUl9CVUZGRVJfVFlQRSB2YWx1ZSAqLwotCi0vKiBNb3JlIGNvbmZpZyBhdHRyaWJ1dGUgdmFsdWVzLCBmb3IgRUdMX1RFWFRVUkVfRk9STUFUICovCi0jZGVmaW5lIEVHTF9OT19URVhUVVJFCQkJMHgzMDVDCi0jZGVmaW5lIEVHTF9URVhUVVJFX1JHQgkJCTB4MzA1RAotI2RlZmluZSBFR0xfVEVYVFVSRV9SR0JBCQkweDMwNUUKLSNkZWZpbmUgRUdMX1RFWFRVUkVfMkQJCQkweDMwNUYKLQotLyogQ29uZmlnIGF0dHJpYnV0ZSBtYXNrIGJpdHMgKi8KLSNkZWZpbmUgRUdMX1BCVUZGRVJfQklUCQkJMHgwMDAxCS8qIEVHTF9TVVJGQUNFX1RZUEUgbWFzayBiaXRzICovCi0jZGVmaW5lIEVHTF9QSVhNQVBfQklUCQkJMHgwMDAyCS8qIEVHTF9TVVJGQUNFX1RZUEUgbWFzayBiaXRzICovCi0jZGVmaW5lIEVHTF9XSU5ET1dfQklUCQkJMHgwMDA0CS8qIEVHTF9TVVJGQUNFX1RZUEUgbWFzayBiaXRzICovCi0jZGVmaW5lIEVHTF9WR19DT0xPUlNQQUNFX0xJTkVBUl9CSVQJMHgwMDIwCS8qIEVHTF9TVVJGQUNFX1RZUEUgbWFzayBiaXRzICovCi0jZGVmaW5lIEVHTF9WR19BTFBIQV9GT1JNQVRfUFJFX0JJVAkweDAwNDAJLyogRUdMX1NVUkZBQ0VfVFlQRSBtYXNrIGJpdHMgKi8KLSNkZWZpbmUgRUdMX01VTFRJU0FNUExFX1JFU09MVkVfQk9YX0JJVCAweDAyMDAJLyogRUdMX1NVUkZBQ0VfVFlQRSBtYXNrIGJpdHMgKi8KLSNkZWZpbmUgRUdMX1NXQVBfQkVIQVZJT1JfUFJFU0VSVkVEX0JJVCAweDA0MDAJLyogRUdMX1NVUkZBQ0VfVFlQRSBtYXNrIGJpdHMgKi8KLQotI2RlZmluZSBFR0xfT1BFTkdMX0VTX0JJVAkJMHgwMDAxCS8qIEVHTF9SRU5ERVJBQkxFX1RZUEUgbWFzayBiaXRzICovCi0jZGVmaW5lIEVHTF9PUEVOVkdfQklUCQkJMHgwMDAyCS8qIEVHTF9SRU5ERVJBQkxFX1RZUEUgbWFzayBiaXRzICovCi0jZGVmaW5lIEVHTF9PUEVOR0xfRVMyX0JJVAkJMHgwMDA0CS8qIEVHTF9SRU5ERVJBQkxFX1RZUEUgbWFzayBiaXRzICovCi0jZGVmaW5lIEVHTF9PUEVOR0xfQklUCQkJMHgwMDA4CS8qIEVHTF9SRU5ERVJBQkxFX1RZUEUgbWFzayBiaXRzICovCi0KLS8qIFF1ZXJ5U3RyaW5nIHRhcmdldHMgKi8KLSNkZWZpbmUgRUdMX1ZFTkRPUgkJCTB4MzA1MwotI2RlZmluZSBFR0xfVkVSU0lPTgkJCTB4MzA1NAotI2RlZmluZSBFR0xfRVhURU5TSU9OUwkJCTB4MzA1NQotI2RlZmluZSBFR0xfQ0xJRU5UX0FQSVMJCQkweDMwOEQKLQotLyogUXVlcnlTdXJmYWNlIC8gU3VyZmFjZUF0dHJpYiAvIENyZWF0ZVBidWZmZXJTdXJmYWNlIHRhcmdldHMgKi8KLSNkZWZpbmUgRUdMX0hFSUdIVAkJCTB4MzA1NgotI2RlZmluZSBFR0xfV0lEVEgJCQkweDMwNTcKLSNkZWZpbmUgRUdMX0xBUkdFU1RfUEJVRkZFUgkJMHgzMDU4Ci0jZGVmaW5lIEVHTF9URVhUVVJFX0ZPUk1BVAkJMHgzMDgwCi0jZGVmaW5lIEVHTF9URVhUVVJFX1RBUkdFVAkJMHgzMDgxCi0jZGVmaW5lIEVHTF9NSVBNQVBfVEVYVFVSRQkJMHgzMDgyCi0jZGVmaW5lIEVHTF9NSVBNQVBfTEVWRUwJCTB4MzA4MwotI2RlZmluZSBFR0xfUkVOREVSX0JVRkZFUgkJMHgzMDg2Ci0jZGVmaW5lIEVHTF9WR19DT0xPUlNQQUNFCQkweDMwODcKLSNkZWZpbmUgRUdMX1ZHX0FMUEhBX0ZPUk1BVAkJMHgzMDg4Ci0jZGVmaW5lIEVHTF9IT1JJWk9OVEFMX1JFU09MVVRJT04JMHgzMDkwCi0jZGVmaW5lIEVHTF9WRVJUSUNBTF9SRVNPTFVUSU9OCQkweDMwOTEKLSNkZWZpbmUgRUdMX1BJWEVMX0FTUEVDVF9SQVRJTwkJMHgzMDkyCi0jZGVmaW5lIEVHTF9TV0FQX0JFSEFWSU9SCQkweDMwOTMKLSNkZWZpbmUgRUdMX01VTFRJU0FNUExFX1JFU09MVkUJCTB4MzA5OQotCi0vKiBFR0xfUkVOREVSX0JVRkZFUiB2YWx1ZXMgLyBCaW5kVGV4SW1hZ2UgLyBSZWxlYXNlVGV4SW1hZ2UgYnVmZmVyIHRhcmdldHMgKi8KLSNkZWZpbmUgRUdMX0JBQ0tfQlVGRkVSCQkJMHgzMDg0Ci0jZGVmaW5lIEVHTF9TSU5HTEVfQlVGRkVSCQkweDMwODUKLQotLyogT3BlblZHIGNvbG9yIHNwYWNlcyAqLwotI2RlZmluZSBFR0xfVkdfQ09MT1JTUEFDRV9zUkdCCQkweDMwODkJLyogRUdMX1ZHX0NPTE9SU1BBQ0UgdmFsdWUgKi8KLSNkZWZpbmUgRUdMX1ZHX0NPTE9SU1BBQ0VfTElORUFSCTB4MzA4QQkvKiBFR0xfVkdfQ09MT1JTUEFDRSB2YWx1ZSAqLwotCi0vKiBPcGVuVkcgYWxwaGEgZm9ybWF0cyAqLwotI2RlZmluZSBFR0xfVkdfQUxQSEFfRk9STUFUX05PTlBSRQkweDMwOEIJLyogRUdMX0FMUEhBX0ZPUk1BVCB2YWx1ZSAqLwotI2RlZmluZSBFR0xfVkdfQUxQSEFfRk9STUFUX1BSRQkJMHgzMDhDCS8qIEVHTF9BTFBIQV9GT1JNQVQgdmFsdWUgKi8KLQotLyogQ29uc3RhbnQgc2NhbGUgZmFjdG9yIGJ5IHdoaWNoIGZyYWN0aW9uYWwgZGlzcGxheSByZXNvbHV0aW9ucyAmCi0gKiBhc3BlY3QgcmF0aW8gYXJlIHNjYWxlZCB3aGVuIHF1ZXJpZWQgYXMgaW50ZWdlciB2YWx1ZXMuCi0gKi8KLSNkZWZpbmUgRUdMX0RJU1BMQVlfU0NBTElORwkJMTAwMDAKLQotLyogVW5rbm93biBkaXNwbGF5IHJlc29sdXRpb24vYXNwZWN0IHJhdGlvICovCi0jZGVmaW5lIEVHTF9VTktOT1dOCQkJKChFR0xpbnQpLTEpCi0KLS8qIEJhY2sgYnVmZmVyIHN3YXAgYmVoYXZpb3JzICovCi0jZGVmaW5lIEVHTF9CVUZGRVJfUFJFU0VSVkVECQkweDMwOTQJLyogRUdMX1NXQVBfQkVIQVZJT1IgdmFsdWUgKi8KLSNkZWZpbmUgRUdMX0JVRkZFUl9ERVNUUk9ZRUQJCTB4MzA5NQkvKiBFR0xfU1dBUF9CRUhBVklPUiB2YWx1ZSAqLwotCi0vKiBDcmVhdGVQYnVmZmVyRnJvbUNsaWVudEJ1ZmZlciBidWZmZXIgdHlwZXMgKi8KLSNkZWZpbmUgRUdMX09QRU5WR19JTUFHRQkJMHgzMDk2Ci0KLS8qIFF1ZXJ5Q29udGV4dCB0YXJnZXRzICovCi0jZGVmaW5lIEVHTF9DT05URVhUX0NMSUVOVF9UWVBFCQkweDMwOTcKLQotLyogQ3JlYXRlQ29udGV4dCBhdHRyaWJ1dGVzICovCi0jZGVmaW5lIEVHTF9DT05URVhUX0NMSUVOVF9WRVJTSU9OCTB4MzA5OAotCi0vKiBNdWx0aXNhbXBsZSByZXNvbHV0aW9uIGJlaGF2aW9ycyAqLwotI2RlZmluZSBFR0xfTVVMVElTQU1QTEVfUkVTT0xWRV9ERUZBVUxUIDB4MzA5QQkvKiBFR0xfTVVMVElTQU1QTEVfUkVTT0xWRSB2YWx1ZSAqLwotI2RlZmluZSBFR0xfTVVMVElTQU1QTEVfUkVTT0xWRV9CT1gJMHgzMDlCCS8qIEVHTF9NVUxUSVNBTVBMRV9SRVNPTFZFIHZhbHVlICovCi0KLS8qIEJpbmRBUEkvUXVlcnlBUEkgdGFyZ2V0cyAqLwotI2RlZmluZSBFR0xfT1BFTkdMX0VTX0FQSQkJMHgzMEEwCi0jZGVmaW5lIEVHTF9PUEVOVkdfQVBJCQkJMHgzMEExCi0jZGVmaW5lIEVHTF9PUEVOR0xfQVBJCQkJMHgzMEEyCi0KLS8qIEdldEN1cnJlbnRTdXJmYWNlIHRhcmdldHMgKi8KLSNkZWZpbmUgRUdMX0RSQVcJCQkweDMwNTkKLSNkZWZpbmUgRUdMX1JFQUQJCQkweDMwNUEKLQotLyogV2FpdE5hdGl2ZSBlbmdpbmVzICovCi0jZGVmaW5lIEVHTF9DT1JFX05BVElWRV9FTkdJTkUJCTB4MzA1QgotCi0vKiBFR0wgMS4yIHRva2VucyByZW5hbWVkIGZvciBjb25zaXN0ZW5jeSBpbiBFR0wgMS4zICovCi0jZGVmaW5lIEVHTF9DT0xPUlNQQUNFCQkJRUdMX1ZHX0NPTE9SU1BBQ0UKLSNkZWZpbmUgRUdMX0FMUEhBX0ZPUk1BVAkJRUdMX1ZHX0FMUEhBX0ZPUk1BVAotI2RlZmluZSBFR0xfQ09MT1JTUEFDRV9zUkdCCQlFR0xfVkdfQ09MT1JTUEFDRV9zUkdCCi0jZGVmaW5lIEVHTF9DT0xPUlNQQUNFX0xJTkVBUgkJRUdMX1ZHX0NPTE9SU1BBQ0VfTElORUFSCi0jZGVmaW5lIEVHTF9BTFBIQV9GT1JNQVRfTk9OUFJFCQlFR0xfVkdfQUxQSEFfRk9STUFUX05PTlBSRQotI2RlZmluZSBFR0xfQUxQSEFfRk9STUFUX1BSRQkJRUdMX1ZHX0FMUEhBX0ZPUk1BVF9QUkUKLQotLyogRUdMIGV4dGVuc2lvbnMgbXVzdCByZXF1ZXN0IGVudW0gYmxvY2tzIGZyb20gdGhlIEtocm9ub3MKLSAqIEFQSSBSZWdpc3RyYXIsIHdobyBtYWludGFpbnMgdGhlIGVudW1lcmFudCByZWdpc3RyeS4gU3VibWl0Ci0gKiBhIGJ1ZyBpbiBLaHJvbm9zIEJ1Z3ppbGxhIGFnYWluc3QgdGFzayAiUmVnaXN0cnkiLgotICovCi0KLQotCi0vKiBFR0wgRnVuY3Rpb25zICovCi0KLUVHTEFQSSBFR0xpbnQgRUdMQVBJRU5UUlkgZWdsR2V0RXJyb3Iodm9pZCk7Ci0KLUVHTEFQSSBFR0xEaXNwbGF5IEVHTEFQSUVOVFJZIGVnbEdldERpc3BsYXkoRUdMTmF0aXZlRGlzcGxheVR5cGUgZGlzcGxheV9pZCk7Ci1FR0xBUEkgRUdMQm9vbGVhbiBFR0xBUElFTlRSWSBlZ2xJbml0aWFsaXplKEVHTERpc3BsYXkgZHB5LCBFR0xpbnQgKm1ham9yLCBFR0xpbnQgKm1pbm9yKTsKLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbFRlcm1pbmF0ZShFR0xEaXNwbGF5IGRweSk7Ci0KLUVHTEFQSSBjb25zdCBjaGFyICogRUdMQVBJRU5UUlkgZWdsUXVlcnlTdHJpbmcoRUdMRGlzcGxheSBkcHksIEVHTGludCBuYW1lKTsKLQotRUdMQVBJIEVHTEJvb2xlYW4gRUdMQVBJRU5UUlkgZWdsR2V0Q29uZmlncyhFR0xEaXNwbGF5IGRweSwgRUdMQ29uZmlnICpjb25maWdzLAotCQkJIEVHTGludCBjb25maWdfc2l6ZSwgRUdMaW50ICpudW1fY29uZmlnKTsKLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbENob29zZUNvbmZpZyhFR0xEaXNwbGF5IGRweSwgY29uc3QgRUdMaW50ICphdHRyaWJfbGlzdCwKLQkJCSAgIEVHTENvbmZpZyAqY29uZmlncywgRUdMaW50IGNvbmZpZ19zaXplLAotCQkJICAgRUdMaW50ICpudW1fY29uZmlnKTsKLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbEdldENvbmZpZ0F0dHJpYihFR0xEaXNwbGF5IGRweSwgRUdMQ29uZmlnIGNvbmZpZywKLQkJCSAgICAgIEVHTGludCBhdHRyaWJ1dGUsIEVHTGludCAqdmFsdWUpOwotCi1FR0xBUEkgRUdMU3VyZmFjZSBFR0xBUElFTlRSWSBlZ2xDcmVhdGVXaW5kb3dTdXJmYWNlKEVHTERpc3BsYXkgZHB5LCBFR0xDb25maWcgY29uZmlnLAotCQkJCSAgRUdMTmF0aXZlV2luZG93VHlwZSB3aW4sCi0JCQkJICBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KTsKLUVHTEFQSSBFR0xTdXJmYWNlIEVHTEFQSUVOVFJZIGVnbENyZWF0ZVBidWZmZXJTdXJmYWNlKEVHTERpc3BsYXkgZHB5LCBFR0xDb25maWcgY29uZmlnLAotCQkJCSAgIGNvbnN0IEVHTGludCAqYXR0cmliX2xpc3QpOwotRUdMQVBJIEVHTFN1cmZhY2UgRUdMQVBJRU5UUlkgZWdsQ3JlYXRlUGl4bWFwU3VyZmFjZShFR0xEaXNwbGF5IGRweSwgRUdMQ29uZmlnIGNvbmZpZywKLQkJCQkgIEVHTE5hdGl2ZVBpeG1hcFR5cGUgcGl4bWFwLAotCQkJCSAgY29uc3QgRUdMaW50ICphdHRyaWJfbGlzdCk7Ci1FR0xBUEkgRUdMQm9vbGVhbiBFR0xBUElFTlRSWSBlZ2xEZXN0cm95U3VyZmFjZShFR0xEaXNwbGF5IGRweSwgRUdMU3VyZmFjZSBzdXJmYWNlKTsKLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbFF1ZXJ5U3VyZmFjZShFR0xEaXNwbGF5IGRweSwgRUdMU3VyZmFjZSBzdXJmYWNlLAotCQkJICAgRUdMaW50IGF0dHJpYnV0ZSwgRUdMaW50ICp2YWx1ZSk7Ci0KLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbEJpbmRBUEkoRUdMZW51bSBhcGkpOwotRUdMQVBJIEVHTGVudW0gRUdMQVBJRU5UUlkgZWdsUXVlcnlBUEkodm9pZCk7Ci0KLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbFdhaXRDbGllbnQodm9pZCk7Ci0KLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbFJlbGVhc2VUaHJlYWQodm9pZCk7Ci0KLUVHTEFQSSBFR0xTdXJmYWNlIEVHTEFQSUVOVFJZIGVnbENyZWF0ZVBidWZmZXJGcm9tQ2xpZW50QnVmZmVyKAotCSAgICAgIEVHTERpc3BsYXkgZHB5LCBFR0xlbnVtIGJ1ZnR5cGUsIEVHTENsaWVudEJ1ZmZlciBidWZmZXIsCi0JICAgICAgRUdMQ29uZmlnIGNvbmZpZywgY29uc3QgRUdMaW50ICphdHRyaWJfbGlzdCk7Ci0KLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbFN1cmZhY2VBdHRyaWIoRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2Ugc3VyZmFjZSwKLQkJCSAgICBFR0xpbnQgYXR0cmlidXRlLCBFR0xpbnQgdmFsdWUpOwotRUdMQVBJIEVHTEJvb2xlYW4gRUdMQVBJRU5UUlkgZWdsQmluZFRleEltYWdlKEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIHN1cmZhY2UsIEVHTGludCBidWZmZXIpOwotRUdMQVBJIEVHTEJvb2xlYW4gRUdMQVBJRU5UUlkgZWdsUmVsZWFzZVRleEltYWdlKEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIHN1cmZhY2UsIEVHTGludCBidWZmZXIpOwotCi0KLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbFN3YXBJbnRlcnZhbChFR0xEaXNwbGF5IGRweSwgRUdMaW50IGludGVydmFsKTsKLQotCi1FR0xBUEkgRUdMQ29udGV4dCBFR0xBUElFTlRSWSBlZ2xDcmVhdGVDb250ZXh0KEVHTERpc3BsYXkgZHB5LCBFR0xDb25maWcgY29uZmlnLAotCQkJICAgIEVHTENvbnRleHQgc2hhcmVfY29udGV4dCwKLQkJCSAgICBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KTsKLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbERlc3Ryb3lDb250ZXh0KEVHTERpc3BsYXkgZHB5LCBFR0xDb250ZXh0IGN0eCk7Ci1FR0xBUEkgRUdMQm9vbGVhbiBFR0xBUElFTlRSWSBlZ2xNYWtlQ3VycmVudChFR0xEaXNwbGF5IGRweSwgRUdMU3VyZmFjZSBkcmF3LAotCQkJICBFR0xTdXJmYWNlIHJlYWQsIEVHTENvbnRleHQgY3R4KTsKLQotRUdMQVBJIEVHTENvbnRleHQgRUdMQVBJRU5UUlkgZWdsR2V0Q3VycmVudENvbnRleHQodm9pZCk7Ci1FR0xBUEkgRUdMU3VyZmFjZSBFR0xBUElFTlRSWSBlZ2xHZXRDdXJyZW50U3VyZmFjZShFR0xpbnQgcmVhZGRyYXcpOwotRUdMQVBJIEVHTERpc3BsYXkgRUdMQVBJRU5UUlkgZWdsR2V0Q3VycmVudERpc3BsYXkodm9pZCk7Ci1FR0xBUEkgRUdMQm9vbGVhbiBFR0xBUElFTlRSWSBlZ2xRdWVyeUNvbnRleHQoRUdMRGlzcGxheSBkcHksIEVHTENvbnRleHQgY3R4LAotCQkJICAgRUdMaW50IGF0dHJpYnV0ZSwgRUdMaW50ICp2YWx1ZSk7Ci0KLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbFdhaXRHTCh2b2lkKTsKLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbFdhaXROYXRpdmUoRUdMaW50IGVuZ2luZSk7Ci1FR0xBUEkgRUdMQm9vbGVhbiBFR0xBUElFTlRSWSBlZ2xTd2FwQnVmZmVycyhFR0xEaXNwbGF5IGRweSwgRUdMU3VyZmFjZSBzdXJmYWNlKTsKLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbENvcHlCdWZmZXJzKEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIHN1cmZhY2UsCi0JCQkgIEVHTE5hdGl2ZVBpeG1hcFR5cGUgdGFyZ2V0KTsKLQotLyogVGhpcyBpcyBhIGdlbmVyaWMgZnVuY3Rpb24gcG9pbnRlciB0eXBlLCB3aG9zZSBuYW1lIGluZGljYXRlcyBpdCBtdXN0Ci0gKiBiZSBjYXN0IHRvIHRoZSBwcm9wZXIgdHlwZSAqYW5kIGNhbGxpbmcgY29udmVudGlvbiogYmVmb3JlIHVzZS4KLSAqLwotdHlwZWRlZiB2b2lkICgqX19lZ2xNdXN0Q2FzdFRvUHJvcGVyRnVuY3Rpb25Qb2ludGVyVHlwZSkodm9pZCk7Ci0KLS8qIE5vdywgZGVmaW5lIGVnbEdldFByb2NBZGRyZXNzIHVzaW5nIHRoZSBnZW5lcmljIGZ1bmN0aW9uIHB0ci4gdHlwZSAqLwotRUdMQVBJIF9fZWdsTXVzdENhc3RUb1Byb3BlckZ1bmN0aW9uUG9pbnRlclR5cGUgRUdMQVBJRU5UUlkKLSAgICAgICBlZ2xHZXRQcm9jQWRkcmVzcyhjb25zdCBjaGFyICpwcm9jbmFtZSk7Ci0KLSNpZmRlZiBfX2NwbHVzcGx1cwotfQotI2VuZGlmCi0KLSNlbmRpZiAvKiBfX2VnbF9oXyAqLwpkaWZmIC0tZ2l0IGEvb3BlbmdsL2luY2x1ZGUvRUdML2VnbGV4dC5oIGIvb3BlbmdsL2luY2x1ZGUvRUdML2VnbGV4dC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyNWNmY2I4Li4wMDAwMDAwCi0tLSBhL29wZW5nbC9pbmNsdWRlL0VHTC9lZ2xleHQuaAorKysgL2Rldi9udWxsCkBAIC0xLDEzOCArMCwwIEBACi0jaWZuZGVmIF9fZWdsZXh0X2hfCi0jZGVmaW5lIF9fZWdsZXh0X2hfCi0KLSNpZmRlZiBfX2NwbHVzcGx1cwotZXh0ZXJuICJDIiB7Ci0jZW5kaWYKLQotLyoKLSoqIENvcHlyaWdodCAoYykgMjAwNy0yMDA5IFRoZSBLaHJvbm9zIEdyb3VwIEluYy4KLSoqCi0qKiBQZXJtaXNzaW9uIGlzIGhlcmVieSBncmFudGVkLCBmcmVlIG9mIGNoYXJnZSwgdG8gYW55IHBlcnNvbiBvYnRhaW5pbmcgYQotKiogY29weSBvZiB0aGlzIHNvZnR3YXJlIGFuZC9vciBhc3NvY2lhdGVkIGRvY3VtZW50YXRpb24gZmlsZXMgKHRoZQotKiogIk1hdGVyaWFscyIpLCB0byBkZWFsIGluIHRoZSBNYXRlcmlhbHMgd2l0aG91dCByZXN0cmljdGlvbiwgaW5jbHVkaW5nCi0qKiB3aXRob3V0IGxpbWl0YXRpb24gdGhlIHJpZ2h0cyB0byB1c2UsIGNvcHksIG1vZGlmeSwgbWVyZ2UsIHB1Ymxpc2gsCi0qKiBkaXN0cmlidXRlLCBzdWJsaWNlbnNlLCBhbmQvb3Igc2VsbCBjb3BpZXMgb2YgdGhlIE1hdGVyaWFscywgYW5kIHRvCi0qKiBwZXJtaXQgcGVyc29ucyB0byB3aG9tIHRoZSBNYXRlcmlhbHMgYXJlIGZ1cm5pc2hlZCB0byBkbyBzbywgc3ViamVjdCB0bwotKiogdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOgotKioKLSoqIFRoZSBhYm92ZSBjb3B5cmlnaHQgbm90aWNlIGFuZCB0aGlzIHBlcm1pc3Npb24gbm90aWNlIHNoYWxsIGJlIGluY2x1ZGVkCi0qKiBpbiBhbGwgY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBNYXRlcmlhbHMuCi0qKgotKiogVEhFIE1BVEVSSUFMUyBBUkUgUFJPVklERUQgIkFTIElTIiwgV0lUSE9VVCBXQVJSQU5UWSBPRiBBTlkgS0lORCwKLSoqIEVYUFJFU1MgT1IgSU1QTElFRCwgSU5DTFVESU5HIEJVVCBOT1QgTElNSVRFRCBUTyBUSEUgV0FSUkFOVElFUyBPRgotKiogTUVSQ0hBTlRBQklMSVRZLCBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRSBBTkQgTk9OSU5GUklOR0VNRU5ULgotKiogSU4gTk8gRVZFTlQgU0hBTEwgVEhFIEFVVEhPUlMgT1IgQ09QWVJJR0hUIEhPTERFUlMgQkUgTElBQkxFIEZPUiBBTlkKLSoqIENMQUlNLCBEQU1BR0VTIE9SIE9USEVSIExJQUJJTElUWSwgV0hFVEhFUiBJTiBBTiBBQ1RJT04gT0YgQ09OVFJBQ1QsCi0qKiBUT1JUIE9SIE9USEVSV0lTRSwgQVJJU0lORyBGUk9NLCBPVVQgT0YgT1IgSU4gQ09OTkVDVElPTiBXSVRIIFRIRQotKiogTUFURVJJQUxTIE9SIFRIRSBVU0UgT1IgT1RIRVIgREVBTElOR1MgSU4gVEhFIE1BVEVSSUFMUy4KLSovCi0KLSNpbmNsdWRlIDxFR0wvZWdscGxhdGZvcm0uaD4KLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi0KLS8qIEhlYWRlciBmaWxlIHZlcnNpb24gbnVtYmVyICovCi0vKiBDdXJyZW50IHZlcnNpb24gYXQgaHR0cDovL3d3dy5raHJvbm9zLm9yZy9yZWdpc3RyeS9lZ2wvICovCi0vKiAkUmV2aXNpb246IDcyNDQgJCBvbiAkRGF0ZTogMjAwOS0wMS0yMCAxNzowNjo1OSAtMDgwMCAoVHVlLCAyMCBKYW4gMjAwOSkgJCAqLwotI2RlZmluZSBFR0xfRUdMRVhUX1ZFUlNJT04gMwotCi0jaWZuZGVmIEVHTF9LSFJfY29uZmlnX2F0dHJpYnMKLSNkZWZpbmUgRUdMX0tIUl9jb25maWdfYXR0cmlicyAxCi0jZGVmaW5lIEVHTF9DT05GT1JNQU5UX0tIUgkJCTB4MzA0MgkvKiBFR0xDb25maWcgYXR0cmlidXRlICovCi0jZGVmaW5lIEVHTF9WR19DT0xPUlNQQUNFX0xJTkVBUl9CSVRfS0hSCTB4MDAyMAkvKiBFR0xfU1VSRkFDRV9UWVBFIGJpdGZpZWxkICovCi0jZGVmaW5lIEVHTF9WR19BTFBIQV9GT1JNQVRfUFJFX0JJVF9LSFIJCTB4MDA0MAkvKiBFR0xfU1VSRkFDRV9UWVBFIGJpdGZpZWxkICovCi0jZW5kaWYKLQotI2lmbmRlZiBFR0xfS0hSX2xvY2tfc3VyZmFjZQotI2RlZmluZSBFR0xfS0hSX2xvY2tfc3VyZmFjZSAxCi0jZGVmaW5lIEVHTF9SRUFEX1NVUkZBQ0VfQklUX0tIUgkJMHgwMDAxCS8qIEVHTF9MT0NLX1VTQUdFX0hJTlRfS0hSIGJpdGZpZWxkICovCi0jZGVmaW5lIEVHTF9XUklURV9TVVJGQUNFX0JJVF9LSFIJCTB4MDAwMgkvKiBFR0xfTE9DS19VU0FHRV9ISU5UX0tIUiBiaXRmaWVsZCAqLwotI2RlZmluZSBFR0xfTE9DS19TVVJGQUNFX0JJVF9LSFIJCTB4MDA4MAkvKiBFR0xfU1VSRkFDRV9UWVBFIGJpdGZpZWxkICovCi0jZGVmaW5lIEVHTF9PUFRJTUFMX0ZPUk1BVF9CSVRfS0hSCQkweDAxMDAJLyogRUdMX1NVUkZBQ0VfVFlQRSBiaXRmaWVsZCAqLwotI2RlZmluZSBFR0xfTUFUQ0hfRk9STUFUX0tIUgkJCTB4MzA0MwkvKiBFR0xDb25maWcgYXR0cmlidXRlICovCi0jZGVmaW5lIEVHTF9GT1JNQVRfUkdCXzU2NV9FWEFDVF9LSFIJCTB4MzBDMAkvKiBFR0xfTUFUQ0hfRk9STUFUX0tIUiB2YWx1ZSAqLwotI2RlZmluZSBFR0xfRk9STUFUX1JHQl81NjVfS0hSCQkJMHgzMEMxCS8qIEVHTF9NQVRDSF9GT1JNQVRfS0hSIHZhbHVlICovCi0jZGVmaW5lIEVHTF9GT1JNQVRfUkdCQV84ODg4X0VYQUNUX0tIUgkJMHgzMEMyCS8qIEVHTF9NQVRDSF9GT1JNQVRfS0hSIHZhbHVlICovCi0jZGVmaW5lIEVHTF9GT1JNQVRfUkdCQV84ODg4X0tIUgkJMHgzMEMzCS8qIEVHTF9NQVRDSF9GT1JNQVRfS0hSIHZhbHVlICovCi0jZGVmaW5lIEVHTF9NQVBfUFJFU0VSVkVfUElYRUxTX0tIUgkJMHgzMEM0CS8qIGVnbExvY2tTdXJmYWNlS0hSIGF0dHJpYnV0ZSAqLwotI2RlZmluZSBFR0xfTE9DS19VU0FHRV9ISU5UX0tIUgkJCTB4MzBDNQkvKiBlZ2xMb2NrU3VyZmFjZUtIUiBhdHRyaWJ1dGUgKi8KLSNkZWZpbmUgRUdMX0JJVE1BUF9QT0lOVEVSX0tIUgkJCTB4MzBDNgkvKiBlZ2xRdWVyeVN1cmZhY2UgYXR0cmlidXRlICovCi0jZGVmaW5lIEVHTF9CSVRNQVBfUElUQ0hfS0hSCQkJMHgzMEM3CS8qIGVnbFF1ZXJ5U3VyZmFjZSBhdHRyaWJ1dGUgKi8KLSNkZWZpbmUgRUdMX0JJVE1BUF9PUklHSU5fS0hSCQkJMHgzMEM4CS8qIGVnbFF1ZXJ5U3VyZmFjZSBhdHRyaWJ1dGUgKi8KLSNkZWZpbmUgRUdMX0JJVE1BUF9QSVhFTF9SRURfT0ZGU0VUX0tIUgkJMHgzMEM5CS8qIGVnbFF1ZXJ5U3VyZmFjZSBhdHRyaWJ1dGUgKi8KLSNkZWZpbmUgRUdMX0JJVE1BUF9QSVhFTF9HUkVFTl9PRkZTRVRfS0hSCTB4MzBDQQkvKiBlZ2xRdWVyeVN1cmZhY2UgYXR0cmlidXRlICovCi0jZGVmaW5lIEVHTF9CSVRNQVBfUElYRUxfQkxVRV9PRkZTRVRfS0hSCTB4MzBDQgkvKiBlZ2xRdWVyeVN1cmZhY2UgYXR0cmlidXRlICovCi0jZGVmaW5lIEVHTF9CSVRNQVBfUElYRUxfQUxQSEFfT0ZGU0VUX0tIUgkweDMwQ0MJLyogZWdsUXVlcnlTdXJmYWNlIGF0dHJpYnV0ZSAqLwotI2RlZmluZSBFR0xfQklUTUFQX1BJWEVMX0xVTUlOQU5DRV9PRkZTRVRfS0hSCTB4MzBDRAkvKiBlZ2xRdWVyeVN1cmZhY2UgYXR0cmlidXRlICovCi0jZGVmaW5lIEVHTF9MT1dFUl9MRUZUX0tIUgkJCTB4MzBDRQkvKiBFR0xfQklUTUFQX09SSUdJTl9LSFIgdmFsdWUgKi8KLSNkZWZpbmUgRUdMX1VQUEVSX0xFRlRfS0hSCQkJMHgzMENGCS8qIEVHTF9CSVRNQVBfT1JJR0lOX0tIUiB2YWx1ZSAqLwotI2lmZGVmIEVHTF9FR0xFWFRfUFJPVE9UWVBFUwotRUdMQVBJIEVHTEJvb2xlYW4gRUdMQVBJRU5UUlkgZWdsTG9ja1N1cmZhY2VLSFIgKEVHTERpc3BsYXkgZGlzcGxheSwgRUdMU3VyZmFjZSBzdXJmYWNlLCBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KTsKLUVHTEFQSSBFR0xCb29sZWFuIEVHTEFQSUVOVFJZIGVnbFVubG9ja1N1cmZhY2VLSFIgKEVHTERpc3BsYXkgZGlzcGxheSwgRUdMU3VyZmFjZSBzdXJmYWNlKTsKLSNlbmRpZiAvKiBFR0xfRUdMRVhUX1BST1RPVFlQRVMgKi8KLXR5cGVkZWYgRUdMQm9vbGVhbiAoRUdMQVBJRU5UUllQIFBGTkVHTExPQ0tTVVJGQUNFS0hSUFJPQykgKEVHTERpc3BsYXkgZGlzcGxheSwgRUdMU3VyZmFjZSBzdXJmYWNlLCBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KTsKLXR5cGVkZWYgRUdMQm9vbGVhbiAoRUdMQVBJRU5UUllQIFBGTkVHTFVOTE9DS1NVUkZBQ0VLSFJQUk9DKSAoRUdMRGlzcGxheSBkaXNwbGF5LCBFR0xTdXJmYWNlIHN1cmZhY2UpOwotI2VuZGlmCi0KLSNpZm5kZWYgRUdMX0tIUl9pbWFnZQotI2RlZmluZSBFR0xfS0hSX2ltYWdlIDEKLSNkZWZpbmUgRUdMX05BVElWRV9QSVhNQVBfS0hSCQkJMHgzMEIwCS8qIGVnbENyZWF0ZUltYWdlS0hSIHRhcmdldCAqLwotdHlwZWRlZiB2b2lkICpFR0xJbWFnZUtIUjsKLSNkZWZpbmUgRUdMX05PX0lNQUdFX0tIUgkJCSgoRUdMSW1hZ2VLSFIpMCkKLSNpZmRlZiBFR0xfRUdMRVhUX1BST1RPVFlQRVMKLUVHTEFQSSBFR0xJbWFnZUtIUiBFR0xBUElFTlRSWSBlZ2xDcmVhdGVJbWFnZUtIUiAoRUdMRGlzcGxheSBkcHksIEVHTENvbnRleHQgY3R4LCBFR0xlbnVtIHRhcmdldCwgRUdMQ2xpZW50QnVmZmVyIGJ1ZmZlciwgY29uc3QgRUdMaW50ICphdHRyaWJfbGlzdCk7Ci1FR0xBUEkgRUdMQm9vbGVhbiBFR0xBUElFTlRSWSBlZ2xEZXN0cm95SW1hZ2VLSFIgKEVHTERpc3BsYXkgZHB5LCBFR0xJbWFnZUtIUiBpbWFnZSk7Ci0jZW5kaWYgLyogRUdMX0VHTEVYVF9QUk9UT1RZUEVTICovCi10eXBlZGVmIEVHTEltYWdlS0hSIChFR0xBUElFTlRSWVAgUEZORUdMQ1JFQVRFSU1BR0VLSFJQUk9DKSAoRUdMRGlzcGxheSBkcHksIEVHTENvbnRleHQgY3R4LCBFR0xlbnVtIHRhcmdldCwgRUdMQ2xpZW50QnVmZmVyIGJ1ZmZlciwgY29uc3QgRUdMaW50ICphdHRyaWJfbGlzdCk7Ci10eXBlZGVmIEVHTEJvb2xlYW4gKEVHTEFQSUVOVFJZUCBQRk5FR0xERVNUUk9ZSU1BR0VLSFJQUk9DKSAoRUdMRGlzcGxheSBkcHksIEVHTEltYWdlS0hSIGltYWdlKTsKLSNlbmRpZgotCi0jaWZuZGVmIEVHTF9LSFJfdmdfcGFyZW50X2ltYWdlCi0jZGVmaW5lIEVHTF9LSFJfdmdfcGFyZW50X2ltYWdlIDEKLSNkZWZpbmUgRUdMX1ZHX1BBUkVOVF9JTUFHRV9LSFIJCQkweDMwQkEJLyogZWdsQ3JlYXRlSW1hZ2VLSFIgdGFyZ2V0ICovCi0jZW5kaWYKLQotI2lmbmRlZiBFR0xfS0hSX2dsX3RleHR1cmVfMkRfaW1hZ2UKLSNkZWZpbmUgRUdMX0tIUl9nbF90ZXh0dXJlXzJEX2ltYWdlIDEKLSNkZWZpbmUgRUdMX0dMX1RFWFRVUkVfMkRfS0hSCQkJMHgzMEIxCS8qIGVnbENyZWF0ZUltYWdlS0hSIHRhcmdldCAqLwotI2RlZmluZSBFR0xfR0xfVEVYVFVSRV9MRVZFTF9LSFIJCTB4MzBCQwkvKiBlZ2xDcmVhdGVJbWFnZUtIUiBhdHRyaWJ1dGUgKi8KLSNlbmRpZgotCi0jaWZuZGVmIEVHTF9LSFJfZ2xfdGV4dHVyZV9jdWJlbWFwX2ltYWdlCi0jZGVmaW5lIEVHTF9LSFJfZ2xfdGV4dHVyZV9jdWJlbWFwX2ltYWdlIDEKLSNkZWZpbmUgRUdMX0dMX1RFWFRVUkVfQ1VCRV9NQVBfUE9TSVRJVkVfWF9LSFIJMHgzMEIzCS8qIGVnbENyZWF0ZUltYWdlS0hSIHRhcmdldCAqLwotI2RlZmluZSBFR0xfR0xfVEVYVFVSRV9DVUJFX01BUF9ORUdBVElWRV9YX0tIUgkweDMwQjQJLyogZWdsQ3JlYXRlSW1hZ2VLSFIgdGFyZ2V0ICovCi0jZGVmaW5lIEVHTF9HTF9URVhUVVJFX0NVQkVfTUFQX1BPU0lUSVZFX1lfS0hSCTB4MzBCNQkvKiBlZ2xDcmVhdGVJbWFnZUtIUiB0YXJnZXQgKi8KLSNkZWZpbmUgRUdMX0dMX1RFWFRVUkVfQ1VCRV9NQVBfTkVHQVRJVkVfWV9LSFIJMHgzMEI2CS8qIGVnbENyZWF0ZUltYWdlS0hSIHRhcmdldCAqLwotI2RlZmluZSBFR0xfR0xfVEVYVFVSRV9DVUJFX01BUF9QT1NJVElWRV9aX0tIUgkweDMwQjcJLyogZWdsQ3JlYXRlSW1hZ2VLSFIgdGFyZ2V0ICovCi0jZGVmaW5lIEVHTF9HTF9URVhUVVJFX0NVQkVfTUFQX05FR0FUSVZFX1pfS0hSCTB4MzBCOAkvKiBlZ2xDcmVhdGVJbWFnZUtIUiB0YXJnZXQgKi8KLSNlbmRpZgotCi0jaWZuZGVmIEVHTF9LSFJfZ2xfdGV4dHVyZV8zRF9pbWFnZQotI2RlZmluZSBFR0xfS0hSX2dsX3RleHR1cmVfM0RfaW1hZ2UgMQotI2RlZmluZSBFR0xfR0xfVEVYVFVSRV8zRF9LSFIJCQkweDMwQjIJLyogZWdsQ3JlYXRlSW1hZ2VLSFIgdGFyZ2V0ICovCi0jZGVmaW5lIEVHTF9HTF9URVhUVVJFX1pPRkZTRVRfS0hSCQkweDMwQkQJLyogZWdsQ3JlYXRlSW1hZ2VLSFIgYXR0cmlidXRlICovCi0jZW5kaWYKLQotI2lmbmRlZiBFR0xfS0hSX2dsX3JlbmRlcmJ1ZmZlcl9pbWFnZQotI2RlZmluZSBFR0xfS0hSX2dsX3JlbmRlcmJ1ZmZlcl9pbWFnZSAxCi0jZGVmaW5lIEVHTF9HTF9SRU5ERVJCVUZGRVJfS0hSCQkJMHgzMEI5CS8qIGVnbENyZWF0ZUltYWdlS0hSIHRhcmdldCAqLwotI2VuZGlmCi0KLSNpZm5kZWYgRUdMX0tIUl9pbWFnZV9iYXNlCi0jZGVmaW5lIEVHTF9LSFJfaW1hZ2VfYmFzZSAxCi0vKiBNb3N0IGludGVyZmFjZXMgZGVmaW5lZCBieSBFR0xfS0hSX2ltYWdlX3BpeG1hcCBhYm92ZSAqLwotI2RlZmluZSBFR0xfSU1BR0VfUFJFU0VSVkVEX0tIUgkJCTB4MzBEMgkvKiBlZ2xDcmVhdGVJbWFnZUtIUiBhdHRyaWJ1dGUgKi8KLSNlbmRpZgotCi0jaWZuZGVmIEVHTF9LSFJfaW1hZ2VfcGl4bWFwCi0jZGVmaW5lIEVHTF9LSFJfaW1hZ2VfcGl4bWFwIDEKLS8qIEludGVyZmFjZXMgZGVmaW5lZCBieSBFR0xfS0hSX2ltYWdlIGFib3ZlICovCi0jZW5kaWYKLQotI2lmZGVmIF9fY3BsdXNwbHVzCi19Ci0jZW5kaWYKLQotI2VuZGlmCmRpZmYgLS1naXQgYS9vcGVuZ2wvaW5jbHVkZS9FR0wvZWdsbmF0aXZlcy5oIGIvb3BlbmdsL2luY2x1ZGUvRUdML2VnbG5hdGl2ZXMuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjE2MjJkYy4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvaW5jbHVkZS9FR0wvZWdsbmF0aXZlcy5oCisrKyAvZGV2L251bGwKQEAgLTEsMjcxICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfRUdMTkFUSVZFU19ICi0jZGVmaW5lIEFORFJPSURfRUdMTkFUSVZFU19ICi0KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2lmZGVmIF9fY3BsdXNwbHVzCi1leHRlcm4gIkMiIHsKLSNlbmRpZgotCi0vKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi0KLS8qIGZsYWdzIHJldHVybmVkIGZyb20gc3dhcEJ1ZmZlciAqLwotI2RlZmluZSBFR0xfTkFUSVZFU19GTEFHX1NJWkVfQ0hBTkdFRCAgICAgICAweDAwMDAwMDAxCi0KLS8qIHN1cmZhY2UgZmxhZ3MgKi8KLSNkZWZpbmUgRUdMX05BVElWRVNfRkxBR19ERVNUUk9ZX0JBQ0tCVUZGRVIgMHgwMDAwMDAwMQotCi1lbnVtIG5hdGl2ZV9waXhlbF9mb3JtYXRfdAotewotICAgIE5BVElWRV9QSVhFTF9GT1JNQVRfUkdCQV84ODg4ICAgPSAxLAotICAgIE5BVElWRV9QSVhFTF9GT1JNQVRfUkdCXzU2NSAgICAgPSA0LAotICAgIE5BVElWRV9QSVhFTF9GT1JNQVRfQkdSQV84ODg4ICAgPSA1LAotICAgIE5BVElWRV9QSVhFTF9GT1JNQVRfUkdCQV81NTUxICAgPSA2LAotICAgIE5BVElWRV9QSVhFTF9GT1JNQVRfUkdCQV80NDQ0ICAgPSA3LAotICAgIE5BVElWRV9QSVhFTF9GT1JNQVRfWUNiQ3JfNDIyX1NQPSAweDEwLAotICAgIE5BVElWRV9QSVhFTF9GT1JNQVRfWUNiQ3JfNDIwX1NQPSAweDExLAotfTsKLQotZW51bSBuYXRpdmVfbWVtb3J5X3R5cGVfdAotewotICAgIE5BVElWRV9NRU1PUllfVFlQRV9QTUVNICAgICAgICAgPSAwLAotICAgIE5BVElWRV9NRU1PUllfVFlQRV9HUFUgICAgICAgICAgPSAxLAotICAgIE5BVElWRV9NRU1PUllfVFlQRV9GQiAgICAgICAgICAgPSAyLAotICAgIE5BVElWRV9NRU1PUllfVFlQRV9IRUFQICAgICAgICAgPSAxMjgKLX07Ci0KLQotc3RydWN0IGVnbF9uYXRpdmVfd2luZG93X3QKLXsKLSAgICAvKgotICAgICAqIG1hZ2ljIG11c3QgYmUgc2V0IHRvIDB4NjAwOTEzCi0gICAgICovCi0gICAgdWludDMyX3QgICAgbWFnaWM7Ci0gICAgCi0gICAgLyoKLSAgICAgKiBtdXN0IGJlIHNpemVvZihlZ2xfbmF0aXZlX3dpbmRvd190KQotICAgICAqLwotICAgIHVpbnQzMl90ICAgIHZlcnNpb247Ci0KLSAgICAvKgotICAgICAqIGlkZW50IGlzIHJlc2VydmVkIGZvciB0aGUgQW5kcm9pZCBwbGF0Zm9ybQotICAgICAqLwotICAgIHVpbnQzMl90ICAgIGlkZW50OwotICAgIAotICAgIC8qCi0gICAgICogd2lkdGgsIGhlaWdodCBhbmQgc3RyaWRlIG9mIHRoZSB3aW5kb3cgaW4gcGl4ZWxzCi0gICAgICogQW55IG9mIHRoZXNlIHZhbHVlIGNhbiBiZSBudWwgaW4gd2hpY2ggY2FzZSBHTCBjb21tYW5kcyBhcmUKLSAgICAgKiBhY2NlcHRlZCBhbmQgcHJvY2Vzc2VkIGFzIHVzdWFsLCBidXQgbm90IHJlbmRlcmluZyBvY2N1cnMuCi0gICAgICovCi0gICAgaW50ICAgICAgICAgd2lkdGg7ICAgICAgLy8gdz1oPTAgaXMgbGVnYWwKLSAgICBpbnQgICAgICAgICBoZWlnaHQ7Ci0gICAgaW50ICAgICAgICAgc3RyaWRlOwotCi0gICAgLyoKLSAgICAgKiBmb3JtYXQgb2YgdGhlIG5hdGl2ZSB3aW5kb3cgKHNlZSB1aS9QaXhlbEZvcm1hdC5oKQotICAgICAqLwotICAgIGludCAgICAgICAgIGZvcm1hdDsKLSAgICAKLSAgICAvKgotICAgICAqIE9mZnNldCBvZiB0aGUgYml0cyBpbiB0aGUgVlJBTQotICAgICAqLwotICAgIGludHB0cl90ICAgIG9mZnNldDsKLSAgICAKLSAgICAvKgotICAgICAqIGZsYWdzIGRlc2NyaWJpbmcgc29tZSBhdHRyaWJ1dGVzIG9mIHRoaXMgc3VyZmFjZQotICAgICAqIEVHTF9OQVRJVkVTX0ZMQUdfREVTVFJPWV9CQUNLQlVGRkVSOiBiYWNrYnVmZmVyIG5vdCBwcmVzZXJ2ZWQgYWZ0ZXIgCi0gICAgICogZWdsU3dhcEJ1ZmZlcnMKLSAgICAgKi8KLSAgICB1aW50MzJfdCAgICBmbGFnczsKLSAgICAKLSAgICAvKgotICAgICAqIGhvcml6b250YWwgYW5kIHZlcnRpY2FsIHJlc29sdXRpb24gaW4gRFBJCi0gICAgICovCi0gICAgZmxvYXQgICAgICAgeGRwaTsKLSAgICBmbG9hdCAgICAgICB5ZHBpOwotICAgIAotICAgIC8qCi0gICAgICogcmVmcmVzaCByYXRlIGluIGZyYW1lcyBwZXIgc2Vjb25kIChIeikKLSAgICAgKi8KLSAgICBmbG9hdCAgICAgICBmcHM7Ci0gICAgCi0gICAgCi0gICAgLyoKLSAgICAgKiAgQmFzZSBtZW1vcnkgdmlydHVhbCBhZGRyZXNzIG9mIHRoZSBzdXJmYWNlIGluIHRoZSBDUFUgc2lkZQotICAgICAqLwotICAgIGludHB0cl90ICAgIGJhc2U7Ci0gICAgCi0gICAgLyoKLSAgICAgKiAgSGVhcCB0aGUgb2Zmc2V0IGFib3ZlIGlzIGJhc2VkIGZyb20KLSAgICAgKi8KLSAgICBpbnQgICAgICAgICBmZDsKLSAgICAKLSAgICAvKgotICAgICAqICBNZW1vcnkgdHlwZSB0aGUgc3VyZmFjZSByZXNpZGVzIGludG8KLSAgICAgKi8KLSAgICB1aW50OF90ICAgICBtZW1vcnlfdHlwZTsKLSAgICAKLSAgICAvKgotICAgICAqIFJlc2VydmVkIGZvciBmdXR1cmUgdXNlLiBNVVNUIEJFIFpFUk8uCi0gICAgICovCi0gICAgdWludDhfdCAgICAgcmVzZXJ2ZWRfcGFkWzNdOwotICAgIGludCAgICAgICAgIHJlc2VydmVkWzhdOwotICAgIAotICAgIC8qCi0gICAgICogVmVydGljYWwgc3RyaWRlIChvbmx5IHJlbGV2YW50IHdpdGggcGxhbmFyIGZvcm1hdHMpIAotICAgICAqLwotICAgIAotICAgIGludCAgICAgICAgIHZzdHJpZGU7Ci0KLSAgICAvKgotICAgICAqIEhvb2sgY2FsbGVkIGJ5IEVHTCB0byBob2xkIGEgcmVmZXJlbmNlIG9uIHRoaXMgc3RydWN0dXJlCi0gICAgICovCi0gICAgdm9pZCAgICAgICAgKCppbmNSZWYpKHN0cnVjdCBlZ2xfbmF0aXZlX3dpbmRvd190KiB3aW5kb3cpOwotCi0gICAgLyoKLSAgICAgKiBIb29rIGNhbGxlZCBieSBFR0wgdG8gcmVsZWFzZSBhIHJlZmVyZW5jZSBvbiB0aGlzIHN0cnVjdHVyZQotICAgICAqLwotICAgIHZvaWQgICAgICAgICgqZGVjUmVmKShzdHJ1Y3QgZWdsX25hdGl2ZV93aW5kb3dfdCogd2luZG93KTsKLQotICAgIC8qCi0gICAgICogSG9vayBjYWxsZWQgYnkgRUdMIHRvIHBlcmZvcm0gYSBwYWdlIGZsaXAuIFRoaXMgZnVuY3Rpb24KLSAgICAgKiBtYXkgdXBkYXRlIHRoZSBzaXplIGF0dHJpYnV0ZXMgYWJvdmUsIGluIHdoaWNoIGNhc2UgaXQgcmV0dXJucwotICAgICAqIHRoZSBFR0xfTkFUSVZFU19GTEFHX1NJWkVfQ0hBTkdFRCBiaXQgc2V0LgotICAgICAqLwotICAgIHVpbnQzMl90ICAgICgqc3dhcEJ1ZmZlcnMpKHN0cnVjdCBlZ2xfbmF0aXZlX3dpbmRvd190KiB3aW5kb3cpOwotICAgIAotICAgIC8qCi0gICAgICogUmVzZXJ2ZWQgZm9yIGZ1dHVyZSB1c2UuIE1VU1QgQkUgWkVSTy4KLSAgICAgKi8KLSAgICB2b2lkICAgICAgICAoKnJlc2VydmVkX3Byb2NfMCkodm9pZCk7Ci0KLSAgICAvKgotICAgICAqIFJlc2VydmVkIGZvciBmdXR1cmUgdXNlLiBNVVNUIEJFIFpFUk8uCi0gICAgICovCi0gICAgdm9pZCAgICAgICAgKCpyZXNlcnZlZF9wcm9jXzEpKHZvaWQpOwotICAgIAotICAgIC8qCi0gICAgICogUmVzZXJ2ZWQgZm9yIGZ1dHVyZSB1c2UuIE1VU1QgQkUgWkVSTy4KLSAgICAgKi8KLSAgICB2b2lkICAgICAgICAoKnJlc2VydmVkX3Byb2NfMikodm9pZCk7Ci0KLSAgICAKLSAgICAvKgotICAgICAqIEhvb2sgY2FsbGVkIGJ5IEVHTCB3aGVuIHRoZSBuYXRpdmUgc3VyZmFjZSBpcyBhc3NvY2lhdGVkIHRvIEVHTAotICAgICAqIChlZ2xDcmVhdGVXaW5kb3dTdXJmYWNlKS4gQ2FuIGJlIE5VTEwuCi0gICAgICovCi0gICAgdm9pZCAgICAgICAgKCpjb25uZWN0KShzdHJ1Y3QgZWdsX25hdGl2ZV93aW5kb3dfdCogd2luZG93KTsKLQotICAgIC8qCi0gICAgICogSG9vayBjYWxsZWQgYnkgRUdMIHdoZW4gZWdsRGVzdHJveVN1cmZhY2UgaXMgY2FsbGVkLiAgQ2FuIGJlIE5VTEwuCi0gICAgICovCi0gICAgdm9pZCAgICAgICAgKCpkaXNjb25uZWN0KShzdHJ1Y3QgZWdsX25hdGl2ZV93aW5kb3dfdCogd2luZG93KTsKLSAgICAKLSAgICAvKgotICAgICAqIFJlc2VydmVkIGZvciBmdXR1cmUgdXNlLiBNVVNUIEJFIFpFUk8uCi0gICAgICovCi0gICAgdm9pZCAgICAgICAgKCpyZXNlcnZlZF9wcm9jWzExXSkodm9pZCk7Ci0gICAgCi0gICAgLyoKLSAgICAgKiAgU29tZSBzdG9yYWdlIHJlc2VydmVkIGZvciB0aGUgb2VtIGRyaXZlci4KLSAgICAgKi8KLSAgICBpbnRwdHJfdCAgICBvZW1bNF07Ci19OwotCi0KLXN0cnVjdCBlZ2xfbmF0aXZlX3BpeG1hcF90Ci17Ci0gICAgaW50MzJfdCAgICAgdmVyc2lvbjsgICAgLyogbXVzdCBiZSAzMiAqLwotICAgIGludDMyX3QgICAgIHdpZHRoOwotICAgIGludDMyX3QgICAgIGhlaWdodDsKLSAgICBpbnQzMl90ICAgICBzdHJpZGU7Ci0gICAgdWludDhfdCogICAgZGF0YTsKLSAgICB1aW50OF90ICAgICBmb3JtYXQ7Ci0gICAgdWludDhfdCAgICAgcmZ1WzNdOwotICAgIHVuaW9uIHsKLSAgICAgICAgdWludDMyX3QgICAgY29tcHJlc3NlZEZvcm1hdDsKLSAgICAgICAgaW50MzJfdCAgICAgdnN0cmlkZTsKLSAgICB9OwotICAgIGludDMyX3QgICAgIHJlc2VydmVkOwotfTsKLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi0vKiAKLSAqIFRoaXMgYSBjb252ZW5pZW5jZSBmdW5jdGlvbiB0byBjcmVhdGUgYSBOYXRpdmVXaW5kb3dUeXBlIHN1cmZhY2UKLSAqIHRoYXQgbWFwcyB0byB0aGUgd2hvbGUgc2NyZWVuCi0gKiBUaGlzIGZ1bmN0aW9uIGlzIGFjdHVhbGx5IGltcGxlbWVudGVkIGluIGxpYnVpLnNvCi0gKi8KLQotc3RydWN0IGVnbF9uYXRpdmVfd2luZG93X3QqIGFuZHJvaWRfY3JlYXRlRGlzcGxheVN1cmZhY2UoKTsKLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi0KLS8qCi0gKiBPRU0ncyBlZ2wncyBsaWJyYXJ5IChsaWJoZ2wuc28pIG11c3QgaW1sZW1lbnQgdGhlc2UgaG9va3MgdG8gYWxsb2NhdGUKLSAqIHRoZSBHUFUgbWVtb3J5IHRoZXkgbmVlZCAgCi0gKi8KLQotCi10eXBlZGVmIHN0cnVjdAotewotICAgIC8vIGZvciBpbnRlcm5hbCB1c2UKLSAgICB2b2lkKiAgIHVzZXI7Ci0gICAgLy8gdmlydHVhbCBhZGRyZXNzIG9mIHRoaXMgYXJlYQotICAgIHZvaWQqICAgYmFzZTsKLSAgICAvLyBzaXplIG9mIHRoaXMgYXJlYSBpbiBieXRlcwotICAgIHNpemVfdCAgc2l6ZTsKLSAgICAvLyBwaHlzaWNhbCBhZGRyZXNzIG9mIHRoaXMgYXJlYQotICAgIHZvaWQqICAgcGh5czsKLSAgICAvLyBvZmZzZXQgaW4gdGhpcyBhcmVhIGF2YWlsYWJsZSB0byB0aGUgR1BVCi0gICAgc2l6ZV90ICBvZmZzZXQ7Ci0gICAgLy8gZmQgb2YgdGhpcyBhcmVhCi0gICAgaW50ICAgICBmZDsKLX0gZ3B1X2FyZWFfdDsKLQotdHlwZWRlZiBzdHJ1Y3QKLXsKLSAgICAvLyBhcmVhIHdoZXJlIEdQVSByZWdpc3RlcnMgYXJlIG1hcHBlZAotICAgIGdwdV9hcmVhX3QgcmVnczsKLSAgICAvLyBudW1iZXIgb2YgZXh0cmEgYXJlYXMgKGN1cnJlbnRseSBsaW1pdGVkIHRvIDIpCi0gICAgaW50MzJfdCBjb3VudDsKLSAgICAvLyBleHRyYSBHUFUgYXJlYXMgKGN1cnJlbnRseSBsaW1pdGVkIHRvIDIpCi0gICAgZ3B1X2FyZWFfdCBncHVbMl07Ci19IHJlcXVlc3RfZ3B1X3Q7Ci0KLQotdHlwZWRlZiByZXF1ZXN0X2dwdV90KiAoKk9FTV9FR0xfYWNxdWlyZV9ncHVfdCkodm9pZCogdXNlcik7Ci10eXBlZGVmIGludCAoKk9FTV9FR0xfcmVsZWFzZV9ncHVfdCkodm9pZCogdXNlciwgcmVxdWVzdF9ncHVfdCogaGFuZGxlKTsKLXR5cGVkZWYgdm9pZCAoKnJlZ2lzdGVyX2dwdV90KQotICAgICAgICAodm9pZCogdXNlciwgT0VNX0VHTF9hY3F1aXJlX2dwdV90LCBPRU1fRUdMX3JlbGVhc2VfZ3B1X3QpOwotCi12b2lkIG9lbV9yZWdpc3Rlcl9ncHUoCi0gICAgICAgIHZvaWQqIHVzZXIsCi0gICAgICAgIE9FTV9FR0xfYWNxdWlyZV9ncHVfdCBhY3F1aXJlLAotICAgICAgICBPRU1fRUdMX3JlbGVhc2VfZ3B1X3QgcmVsZWFzZSk7Ci0KLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi0jaWZkZWYgX19jcGx1c3BsdXMKLX0KLSNlbmRpZgotCi0jZW5kaWYgLyogQU5EUk9JRF9FR0xOQVRJVkVTX0ggKi8KZGlmZiAtLWdpdCBhL29wZW5nbC9pbmNsdWRlL0VHTC9lZ2xwbGF0Zm9ybS5oIGIvb3BlbmdsL2luY2x1ZGUvRUdML2VnbHBsYXRmb3JtLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGFjMDA5MDEuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2luY2x1ZGUvRUdML2VnbHBsYXRmb3JtLmgKKysrIC9kZXYvbnVsbApAQCAtMSwxMTcgKzAsMCBAQAotI2lmbmRlZiBfX2VnbHBsYXRmb3JtX2hfCi0jZGVmaW5lIF9fZWdscGxhdGZvcm1faF8KLQotLyoKLSoqIENvcHlyaWdodCAoYykgMjAwNy0yMDA5IFRoZSBLaHJvbm9zIEdyb3VwIEluYy4KLSoqCi0qKiBQZXJtaXNzaW9uIGlzIGhlcmVieSBncmFudGVkLCBmcmVlIG9mIGNoYXJnZSwgdG8gYW55IHBlcnNvbiBvYnRhaW5pbmcgYQotKiogY29weSBvZiB0aGlzIHNvZnR3YXJlIGFuZC9vciBhc3NvY2lhdGVkIGRvY3VtZW50YXRpb24gZmlsZXMgKHRoZQotKiogIk1hdGVyaWFscyIpLCB0byBkZWFsIGluIHRoZSBNYXRlcmlhbHMgd2l0aG91dCByZXN0cmljdGlvbiwgaW5jbHVkaW5nCi0qKiB3aXRob3V0IGxpbWl0YXRpb24gdGhlIHJpZ2h0cyB0byB1c2UsIGNvcHksIG1vZGlmeSwgbWVyZ2UsIHB1Ymxpc2gsCi0qKiBkaXN0cmlidXRlLCBzdWJsaWNlbnNlLCBhbmQvb3Igc2VsbCBjb3BpZXMgb2YgdGhlIE1hdGVyaWFscywgYW5kIHRvCi0qKiBwZXJtaXQgcGVyc29ucyB0byB3aG9tIHRoZSBNYXRlcmlhbHMgYXJlIGZ1cm5pc2hlZCB0byBkbyBzbywgc3ViamVjdCB0bwotKiogdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOgotKioKLSoqIFRoZSBhYm92ZSBjb3B5cmlnaHQgbm90aWNlIGFuZCB0aGlzIHBlcm1pc3Npb24gbm90aWNlIHNoYWxsIGJlIGluY2x1ZGVkCi0qKiBpbiBhbGwgY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBNYXRlcmlhbHMuCi0qKgotKiogVEhFIE1BVEVSSUFMUyBBUkUgUFJPVklERUQgIkFTIElTIiwgV0lUSE9VVCBXQVJSQU5UWSBPRiBBTlkgS0lORCwKLSoqIEVYUFJFU1MgT1IgSU1QTElFRCwgSU5DTFVESU5HIEJVVCBOT1QgTElNSVRFRCBUTyBUSEUgV0FSUkFOVElFUyBPRgotKiogTUVSQ0hBTlRBQklMSVRZLCBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRSBBTkQgTk9OSU5GUklOR0VNRU5ULgotKiogSU4gTk8gRVZFTlQgU0hBTEwgVEhFIEFVVEhPUlMgT1IgQ09QWVJJR0hUIEhPTERFUlMgQkUgTElBQkxFIEZPUiBBTlkKLSoqIENMQUlNLCBEQU1BR0VTIE9SIE9USEVSIExJQUJJTElUWSwgV0hFVEhFUiBJTiBBTiBBQ1RJT04gT0YgQ09OVFJBQ1QsCi0qKiBUT1JUIE9SIE9USEVSV0lTRSwgQVJJU0lORyBGUk9NLCBPVVQgT0YgT1IgSU4gQ09OTkVDVElPTiBXSVRIIFRIRQotKiogTUFURVJJQUxTIE9SIFRIRSBVU0UgT1IgT1RIRVIgREVBTElOR1MgSU4gVEhFIE1BVEVSSUFMUy4KLSovCi0KLS8qIFBsYXRmb3JtLXNwZWNpZmljIHR5cGVzIGFuZCBkZWZpbml0aW9ucyBmb3IgZWdsLmgKLSAqICRSZXZpc2lvbjogNzI0NCAkIG9uICREYXRlOiAyMDA5LTAxLTIwIDE3OjA2OjU5IC0wODAwIChUdWUsIDIwIEphbiAyMDA5KSAkCi0gKgotICogQWRvcHRlcnMgbWF5IG1vZGlmeSBraHJwbGF0Zm9ybS5oIGFuZCB0aGlzIGZpbGUgdG8gc3VpdCB0aGVpciBwbGF0Zm9ybS4KLSAqIFlvdSBhcmUgZW5jb3VyYWdlZCB0byBzdWJtaXQgYWxsIG1vZGlmaWNhdGlvbnMgdG8gdGhlIEtocm9ub3MgZ3JvdXAgc28gdGhhdAotICogdGhleSBjYW4gYmUgaW5jbHVkZWQgaW4gZnV0dXJlIHZlcnNpb25zIG9mIHRoaXMgZmlsZS4gIFBsZWFzZSBzdWJtaXQgY2hhbmdlcwotICogYnkgc2VuZGluZyB0aGVtIHRvIHRoZSBwdWJsaWMgS2hyb25vcyBCdWd6aWxsYSAoaHR0cDovL2tocm9ub3Mub3JnL2J1Z3ppbGxhKQotICogYnkgZmlsaW5nIGEgYnVnIGFnYWluc3QgcHJvZHVjdCAiRUdMIiBjb21wb25lbnQgIlJlZ2lzdHJ5Ii4KLSAqLwotCi0jaW5jbHVkZSA8S0hSL2tocnBsYXRmb3JtLmg+Ci0KLS8qIE1hY3JvcyB1c2VkIGluIEVHTCBmdW5jdGlvbiBwcm90b3R5cGUgZGVjbGFyYXRpb25zLgotICoKLSAqIEVHTCBmdW5jdGlvbnMgc2hvdWxkIGJlIHByb3RvdHlwZWQgYXM6Ci0gKgotICogRUdMQVBJIHJldHVybi10eXBlIEVHTEFQSUVOVFJZIGVnbEZ1bmN0aW9uKGFyZ3VtZW50cyk7Ci0gKiB0eXBlZGVmIHJldHVybi10eXBlIChFWFBBUElFTlRSWVAgUEZORUdMRlVOQ1RJT05QUk9DKSAoYXJndW1lbnRzKTsKLSAqCi0gKiBLSFJPTk9TX0FQSUNBTEwgYW5kIEtIUk9OT1NfQVBJRU5UUlkgYXJlIGRlZmluZWQgaW4gS0hSL2tocnBsYXRmb3JtLmgKLSAqLwotCi0jaWZuZGVmIEVHTEFQSQotI2RlZmluZSBFR0xBUEkgS0hST05PU19BUElDQUxMCi0jZW5kaWYKLQotI2RlZmluZSBFR0xBUElFTlRSWSAgS0hST05PU19BUElFTlRSWQotI2RlZmluZSBFR0xBUElFTlRSWVAgS0hST05PU19BUElFTlRSWSoKLQotLyogVGhlIHR5cGVzIE5hdGl2ZURpc3BsYXlUeXBlLCBOYXRpdmVXaW5kb3dUeXBlLCBhbmQgTmF0aXZlUGl4bWFwVHlwZQotICogYXJlIGFsaWFzZXMgb2Ygd2luZG93LXN5c3RlbS1kZXBlbmRlbnQgdHlwZXMsIHN1Y2ggYXMgWCBEaXNwbGF5ICogb3IKLSAqIFdpbmRvd3MgRGV2aWNlIENvbnRleHQuIFRoZXkgbXVzdCBiZSBkZWZpbmVkIGluIHBsYXRmb3JtLXNwZWNpZmljCi0gKiBjb2RlIGJlbG93LiBUaGUgRUdMLXByZWZpeGVkIHZlcnNpb25zIG9mIE5hdGl2ZSpUeXBlIGFyZSB0aGUgc2FtZQotICogdHlwZXMsIHJlbmFtZWQgaW4gRUdMIDEuMyBzbyBhbGwgdHlwZXMgaW4gdGhlIEFQSSBzdGFydCB3aXRoICJFR0wiLgotICovCi0KLSNpZiBkZWZpbmVkKF9XSU4zMikgfHwgZGVmaW5lZChfX1ZDMzJfXykgJiYgIWRlZmluZWQoX19DWUdXSU5fXykgJiYgIWRlZmluZWQoX19TQ0lURUNIX1NOQVBfXykgLyogV2luMzIgYW5kIFdpbkNFICovCi0jaWZuZGVmIFdJTjMyX0xFQU5fQU5EX01FQU4KLSNkZWZpbmUgV0lOMzJfTEVBTl9BTkRfTUVBTiAxCi0jZW5kaWYKLSNpbmNsdWRlIDx3aW5kb3dzLmg+Ci0KLXR5cGVkZWYgSERDICAgICBFR0xOYXRpdmVEaXNwbGF5VHlwZTsKLXR5cGVkZWYgSEJJVE1BUCBFR0xOYXRpdmVQaXhtYXBUeXBlOwotdHlwZWRlZiBIV05EICAgIEVHTE5hdGl2ZVdpbmRvd1R5cGU7Ci0KLSNlbGlmIGRlZmluZWQoX19XSU5TQ1dfXykgfHwgZGVmaW5lZChfX1NZTUJJQU4zMl9fKSAgLyogU3ltYmlhbiAqLwotCi10eXBlZGVmIGludCAgIEVHTE5hdGl2ZURpc3BsYXlUeXBlOwotdHlwZWRlZiB2b2lkICpFR0xOYXRpdmVXaW5kb3dUeXBlOwotdHlwZWRlZiB2b2lkICpFR0xOYXRpdmVQaXhtYXBUeXBlOwotCi0jZWxpZiBkZWZpbmVkKF9fdW5peF9fKSAmJiAhZGVmaW5lZChBTkRST0lEKQotCi0vKiBYMTEgKHRlbnRhdGl2ZSkgICovCi0jaW5jbHVkZSA8WDExL1hsaWIuaD4KLSNpbmNsdWRlIDxYMTEvWHV0aWwuaD4KLQotdHlwZWRlZiBEaXNwbGF5ICpFR0xOYXRpdmVEaXNwbGF5VHlwZTsKLXR5cGVkZWYgUGl4bWFwICAgRUdMTmF0aXZlUGl4bWFwVHlwZTsKLXR5cGVkZWYgV2luZG93ICAgRUdMTmF0aXZlV2luZG93VHlwZTsKLQotCi0jZWxpZiBkZWZpbmVkKEFORFJPSUQpCi0KLSNpbmNsdWRlIDxFR0wvZWdsbmF0aXZlcy5oPgotCi10eXBlZGVmIHN0cnVjdCBlZ2xfbmF0aXZlX3dpbmRvd190KiAgICAgRUdMTmF0aXZlV2luZG93VHlwZTsKLXR5cGVkZWYgc3RydWN0IGVnbF9uYXRpdmVfcGl4bWFwX3QqICAgICBFR0xOYXRpdmVQaXhtYXBUeXBlOwotdHlwZWRlZiB2b2lkKiAgICAgICAgICAgICAgICAgICAgICAgICAgIEVHTE5hdGl2ZURpc3BsYXlUeXBlOwotCi0jZWxzZQotI2Vycm9yICJQbGF0Zm9ybSBub3QgcmVjb2duaXplZCIKLSNlbmRpZgotCi0vKiBFR0wgMS4yIHR5cGVzLCByZW5hbWVkIGZvciBjb25zaXN0ZW5jeSBpbiBFR0wgMS4zICovCi10eXBlZGVmIEVHTE5hdGl2ZURpc3BsYXlUeXBlIE5hdGl2ZURpc3BsYXlUeXBlOwotdHlwZWRlZiBFR0xOYXRpdmVQaXhtYXBUeXBlICBOYXRpdmVQaXhtYXBUeXBlOwotdHlwZWRlZiBFR0xOYXRpdmVXaW5kb3dUeXBlICBOYXRpdmVXaW5kb3dUeXBlOwotCi0KLS8qIERlZmluZSBFR0xpbnQuIFRoaXMgbXVzdCBiZSBhIHNpZ25lZCBpbnRlZ3JhbCB0eXBlIGxhcmdlIGVub3VnaCB0byBjb250YWluCi0gKiBhbGwgbGVnYWwgYXR0cmlidXRlIG5hbWVzIGFuZCB2YWx1ZXMgcGFzc2VkIGludG8gYW5kIG91dCBvZiBFR0wsIHdoZXRoZXIKLSAqIHRoZWlyIHR5cGUgaXMgYm9vbGVhbiwgYml0bWFzaywgZW51bWVyYW50IChzeW1ib2xpYyBjb25zdGFudCksIGludGVnZXIsCi0gKiBoYW5kbGUsIG9yIG90aGVyLiAgV2hpbGUgaW4gZ2VuZXJhbCBhIDMyLWJpdCBpbnRlZ2VyIHdpbGwgc3VmZmljZSwgaWYKLSAqIGhhbmRsZXMgYXJlIDY0IGJpdCB0eXBlcywgdGhlbiBFR0xpbnQgc2hvdWxkIGJlIGRlZmluZWQgYXMgYSBzaWduZWQgNjQtYml0Ci0gKiBpbnRlZ2VyIHR5cGUuCi0gKi8KLXR5cGVkZWYga2hyb25vc19pbnQzMl90IEVHTGludDsKLQotI2VuZGlmIC8qIF9fZWdscGxhdGZvcm1faCAqLwpkaWZmIC0tZ2l0IGEvb3BlbmdsL2luY2x1ZGUvR0xFUy9lZ2wuaCBiL29wZW5nbC9pbmNsdWRlL0dMRVMvZWdsLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDU3NzhlMDAuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2luY2x1ZGUvR0xFUy9lZ2wuaAorKysgL2Rldi9udWxsCkBAIC0xLDE1ICswLDAgQEAKLS8qCi0gKiBTa2VsZXRvbiBlZ2wuaCB0byBwcm92aWRlIGNvbXBhdGliaWxpdHkgZm9yIGVhcmx5IEdMRVMgMS4wCi0gKiBhcHBsaWNhdGlvbnMuIFNldmVyYWwgZWFybHkgaW1wbGVtZW50YXRpb25zIGluY2x1ZGVkIGdsLmgKLSAqIGluIGVnbC5oIGxlYWRpbmcgYXBwbGljYXRpb25zIHRvIGluY2x1ZGUgb25seSBlZ2wuaAotICoKLSAqICRSZXZpc2lvbjogNjI1MiAkIG9uICREYXRlOjogMjAwOC0wOC0wNiAxNjozNTowOCAtMDcwMCAjJAotICovCi0KLSNpZm5kZWYgX19sZWdhY3lfZWdsX2hfCi0jZGVmaW5lIF9fbGVnYWN5X2VnbF9oXwotCi0jaW5jbHVkZSA8RUdML2VnbC5oPgotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLQotI2VuZGlmIC8qIF9fbGVnYWN5X2VnbF9oXyAqLwpkaWZmIC0tZ2l0IGEvb3BlbmdsL2luY2x1ZGUvR0xFUy9nbC5oIGIvb3BlbmdsL2luY2x1ZGUvR0xFUy9nbC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyZThiOTcxLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9pbmNsdWRlL0dMRVMvZ2wuaAorKysgL2Rldi9udWxsCkBAIC0xLDc2OSArMCwwIEBACi0jaWZuZGVmIF9fZ2xfaF8KLSNkZWZpbmUgX19nbF9oXwotCi0vKiAkUmV2aXNpb246IDcxNzIgJCBvbiAkRGF0ZTo6IDIwMDktMDEtMDkgMTE6MTc6NDEgLTA4MDAgIyQgKi8KLQotI2luY2x1ZGUgPEdMRVMvZ2xwbGF0Zm9ybS5oPgotCi0jaWZkZWYgX19jcGx1c3BsdXMKLWV4dGVybiAiQyIgewotI2VuZGlmCi0KLS8qCi0gKiBUaGlzIGRvY3VtZW50IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBTR0kgRnJlZSBTb2Z0d2FyZSBCIExpY2Vuc2UgVmVyc2lvbgotICogMi4wLiBGb3IgZGV0YWlscywgc2VlIGh0dHA6Ly9vc3Muc2dpLmNvbS9wcm9qZWN0cy9GcmVlQi8gLgotICovCi0KLXR5cGVkZWYgdm9pZCAgICAgICAgICAgICBHTHZvaWQ7Ci10eXBlZGVmIHVuc2lnbmVkIGludCAgICAgR0xlbnVtOwotdHlwZWRlZiB1bnNpZ25lZCBjaGFyICAgIEdMYm9vbGVhbjsKLXR5cGVkZWYgdW5zaWduZWQgaW50ICAgICBHTGJpdGZpZWxkOwotdHlwZWRlZiBraHJvbm9zX2ludDhfdCAgIEdMYnl0ZTsKLXR5cGVkZWYgc2hvcnQgICAgICAgICAgICBHTHNob3J0OwotdHlwZWRlZiBpbnQgICAgICAgICAgICAgIEdMaW50OwotdHlwZWRlZiBpbnQgICAgICAgICAgICAgIEdMc2l6ZWk7Ci10eXBlZGVmIGtocm9ub3NfdWludDhfdCAgR0x1Ynl0ZTsKLXR5cGVkZWYgdW5zaWduZWQgc2hvcnQgICBHTHVzaG9ydDsKLXR5cGVkZWYgdW5zaWduZWQgaW50ICAgICBHTHVpbnQ7Ci10eXBlZGVmIGtocm9ub3NfZmxvYXRfdCAgR0xmbG9hdDsKLXR5cGVkZWYga2hyb25vc19mbG9hdF90ICBHTGNsYW1wZjsKLXR5cGVkZWYga2hyb25vc19pbnQzMl90ICBHTGZpeGVkOwotdHlwZWRlZiBraHJvbm9zX2ludDMyX3QgIEdMY2xhbXB4OwotCi10eXBlZGVmIGtocm9ub3NfaW50cHRyX3QgR0xpbnRwdHI7Ci10eXBlZGVmIGtocm9ub3Nfc3NpemVfdCAgR0xzaXplaXB0cjsKLQotCi0vKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotLyogT3BlbkdMIEVTIGNvcmUgdmVyc2lvbnMgKi8KLSNkZWZpbmUgR0xfVkVSU0lPTl9FU19DTV8xXzAgICAgICAgICAgMQotI2RlZmluZSBHTF9WRVJTSU9OX0VTX0NMXzFfMCAgICAgICAgICAxCi0jZGVmaW5lIEdMX1ZFUlNJT05fRVNfQ01fMV8xICAgICAgICAgIDEKLSNkZWZpbmUgR0xfVkVSU0lPTl9FU19DTF8xXzEgICAgICAgICAgMQotCi0vKiBDbGVhckJ1ZmZlck1hc2sgKi8KLSNkZWZpbmUgR0xfREVQVEhfQlVGRkVSX0JJVCAgICAgICAgICAgICAgIDB4MDAwMDAxMDAKLSNkZWZpbmUgR0xfU1RFTkNJTF9CVUZGRVJfQklUICAgICAgICAgICAgIDB4MDAwMDA0MDAKLSNkZWZpbmUgR0xfQ09MT1JfQlVGRkVSX0JJVCAgICAgICAgICAgICAgIDB4MDAwMDQwMDAKLQotLyogQm9vbGVhbiAqLwotI2RlZmluZSBHTF9GQUxTRSAgICAgICAgICAgICAgICAgICAgICAgICAgMAotI2RlZmluZSBHTF9UUlVFICAgICAgICAgICAgICAgICAgICAgICAgICAgMQotCi0vKiBCZWdpbk1vZGUgKi8KLSNkZWZpbmUgR0xfUE9JTlRTICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDAwMAotI2RlZmluZSBHTF9MSU5FUyAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwMDAxCi0jZGVmaW5lIEdMX0xJTkVfTE9PUCAgICAgICAgICAgICAgICAgICAgICAweDAwMDIKLSNkZWZpbmUgR0xfTElORV9TVFJJUCAgICAgICAgICAgICAgICAgICAgIDB4MDAwMwotI2RlZmluZSBHTF9UUklBTkdMRVMgICAgICAgICAgICAgICAgICAgICAgMHgwMDA0Ci0jZGVmaW5lIEdMX1RSSUFOR0xFX1NUUklQICAgICAgICAgICAgICAgICAweDAwMDUKLSNkZWZpbmUgR0xfVFJJQU5HTEVfRkFOICAgICAgICAgICAgICAgICAgIDB4MDAwNgotCi0vKiBBbHBoYUZ1bmN0aW9uICovCi0jZGVmaW5lIEdMX05FVkVSICAgICAgICAgICAgICAgICAgICAgICAgICAweDAyMDAKLSNkZWZpbmUgR0xfTEVTUyAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDIwMQotI2RlZmluZSBHTF9FUVVBTCAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwMjAyCi0jZGVmaW5lIEdMX0xFUVVBTCAgICAgICAgICAgICAgICAgICAgICAgICAweDAyMDMKLSNkZWZpbmUgR0xfR1JFQVRFUiAgICAgICAgICAgICAgICAgICAgICAgIDB4MDIwNAotI2RlZmluZSBHTF9OT1RFUVVBTCAgICAgICAgICAgICAgICAgICAgICAgMHgwMjA1Ci0jZGVmaW5lIEdMX0dFUVVBTCAgICAgICAgICAgICAgICAgICAgICAgICAweDAyMDYKLSNkZWZpbmUgR0xfQUxXQVlTICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDIwNwotCi0vKiBCbGVuZGluZ0ZhY3RvckRlc3QgKi8KLSNkZWZpbmUgR0xfWkVSTyAgICAgICAgICAgICAgICAgICAgICAgICAgIDAKLSNkZWZpbmUgR0xfT05FICAgICAgICAgICAgICAgICAgICAgICAgICAgIDEKLSNkZWZpbmUgR0xfU1JDX0NPTE9SICAgICAgICAgICAgICAgICAgICAgIDB4MDMwMAotI2RlZmluZSBHTF9PTkVfTUlOVVNfU1JDX0NPTE9SICAgICAgICAgICAgMHgwMzAxCi0jZGVmaW5lIEdMX1NSQ19BTFBIQSAgICAgICAgICAgICAgICAgICAgICAweDAzMDIKLSNkZWZpbmUgR0xfT05FX01JTlVTX1NSQ19BTFBIQSAgICAgICAgICAgIDB4MDMwMwotI2RlZmluZSBHTF9EU1RfQUxQSEEgICAgICAgICAgICAgICAgICAgICAgMHgwMzA0Ci0jZGVmaW5lIEdMX09ORV9NSU5VU19EU1RfQUxQSEEgICAgICAgICAgICAweDAzMDUKLQotLyogQmxlbmRpbmdGYWN0b3JTcmMgKi8KLS8qICAgICAgR0xfWkVSTyAqLwotLyogICAgICBHTF9PTkUgKi8KLSNkZWZpbmUgR0xfRFNUX0NPTE9SICAgICAgICAgICAgICAgICAgICAgIDB4MDMwNgotI2RlZmluZSBHTF9PTkVfTUlOVVNfRFNUX0NPTE9SICAgICAgICAgICAgMHgwMzA3Ci0jZGVmaW5lIEdMX1NSQ19BTFBIQV9TQVRVUkFURSAgICAgICAgICAgICAweDAzMDgKLS8qICAgICAgR0xfU1JDX0FMUEhBICovCi0vKiAgICAgIEdMX09ORV9NSU5VU19TUkNfQUxQSEEgKi8KLS8qICAgICAgR0xfRFNUX0FMUEhBICovCi0vKiAgICAgIEdMX09ORV9NSU5VU19EU1RfQUxQSEEgKi8KLQotLyogQ2xpcFBsYW5lTmFtZSAqLwotI2RlZmluZSBHTF9DTElQX1BMQU5FMCAgICAgICAgICAgICAgICAgICAgMHgzMDAwCi0jZGVmaW5lIEdMX0NMSVBfUExBTkUxICAgICAgICAgICAgICAgICAgICAweDMwMDEKLSNkZWZpbmUgR0xfQ0xJUF9QTEFORTIgICAgICAgICAgICAgICAgICAgIDB4MzAwMgotI2RlZmluZSBHTF9DTElQX1BMQU5FMyAgICAgICAgICAgICAgICAgICAgMHgzMDAzCi0jZGVmaW5lIEdMX0NMSVBfUExBTkU0ICAgICAgICAgICAgICAgICAgICAweDMwMDQKLSNkZWZpbmUgR0xfQ0xJUF9QTEFORTUgICAgICAgICAgICAgICAgICAgIDB4MzAwNQotCi0vKiBDb2xvck1hdGVyaWFsRmFjZSAqLwotLyogICAgICBHTF9GUk9OVF9BTkRfQkFDSyAqLwotCi0vKiBDb2xvck1hdGVyaWFsUGFyYW1ldGVyICovCi0vKiAgICAgIEdMX0FNQklFTlRfQU5EX0RJRkZVU0UgKi8KLQotLyogQ29sb3JQb2ludGVyVHlwZSAqLwotLyogICAgICBHTF9VTlNJR05FRF9CWVRFICovCi0vKiAgICAgIEdMX0ZMT0FUICovCi0vKiAgICAgIEdMX0ZJWEVEICovCi0KLS8qIEN1bGxGYWNlTW9kZSAqLwotI2RlZmluZSBHTF9GUk9OVCAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwNDA0Ci0jZGVmaW5lIEdMX0JBQ0sgICAgICAgICAgICAgICAgICAgICAgICAgICAweDA0MDUKLSNkZWZpbmUgR0xfRlJPTlRfQU5EX0JBQ0sgICAgICAgICAgICAgICAgIDB4MDQwOAotCi0vKiBEZXB0aEZ1bmN0aW9uICovCi0vKiAgICAgIEdMX05FVkVSICovCi0vKiAgICAgIEdMX0xFU1MgKi8KLS8qICAgICAgR0xfRVFVQUwgKi8KLS8qICAgICAgR0xfTEVRVUFMICovCi0vKiAgICAgIEdMX0dSRUFURVIgKi8KLS8qICAgICAgR0xfTk9URVFVQUwgKi8KLS8qICAgICAgR0xfR0VRVUFMICovCi0vKiAgICAgIEdMX0FMV0FZUyAqLwotCi0vKiBFbmFibGVDYXAgKi8KLSNkZWZpbmUgR0xfRk9HICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MEI2MAotI2RlZmluZSBHTF9MSUdIVElORyAgICAgICAgICAgICAgICAgICAgICAgMHgwQjUwCi0jZGVmaW5lIEdMX1RFWFRVUkVfMkQgICAgICAgICAgICAgICAgICAgICAweDBERTEKLSNkZWZpbmUgR0xfQ1VMTF9GQUNFICAgICAgICAgICAgICAgICAgICAgIDB4MEI0NAotI2RlZmluZSBHTF9BTFBIQV9URVNUICAgICAgICAgICAgICAgICAgICAgMHgwQkMwCi0jZGVmaW5lIEdMX0JMRU5EICAgICAgICAgICAgICAgICAgICAgICAgICAweDBCRTIKLSNkZWZpbmUgR0xfQ09MT1JfTE9HSUNfT1AgICAgICAgICAgICAgICAgIDB4MEJGMgotI2RlZmluZSBHTF9ESVRIRVIgICAgICAgICAgICAgICAgICAgICAgICAgMHgwQkQwCi0jZGVmaW5lIEdMX1NURU5DSUxfVEVTVCAgICAgICAgICAgICAgICAgICAweDBCOTAKLSNkZWZpbmUgR0xfREVQVEhfVEVTVCAgICAgICAgICAgICAgICAgICAgIDB4MEI3MQotLyogICAgICBHTF9MSUdIVDAgKi8KLS8qICAgICAgR0xfTElHSFQxICovCi0vKiAgICAgIEdMX0xJR0hUMiAqLwotLyogICAgICBHTF9MSUdIVDMgKi8KLS8qICAgICAgR0xfTElHSFQ0ICovCi0vKiAgICAgIEdMX0xJR0hUNSAqLwotLyogICAgICBHTF9MSUdIVDYgKi8KLS8qICAgICAgR0xfTElHSFQ3ICovCi0jZGVmaW5lIEdMX1BPSU5UX1NNT09USCAgICAgICAgICAgICAgICAgICAweDBCMTAKLSNkZWZpbmUgR0xfTElORV9TTU9PVEggICAgICAgICAgICAgICAgICAgIDB4MEIyMAotI2RlZmluZSBHTF9TQ0lTU09SX1RFU1QgICAgICAgICAgICAgICAgICAgMHgwQzExCi0jZGVmaW5lIEdMX0NPTE9SX01BVEVSSUFMICAgICAgICAgICAgICAgICAweDBCNTcKLSNkZWZpbmUgR0xfTk9STUFMSVpFICAgICAgICAgICAgICAgICAgICAgIDB4MEJBMQotI2RlZmluZSBHTF9SRVNDQUxFX05PUk1BTCAgICAgICAgICAgICAgICAgMHg4MDNBCi0jZGVmaW5lIEdMX1BPTFlHT05fT0ZGU0VUX0ZJTEwgICAgICAgICAgICAweDgwMzcKLSNkZWZpbmUgR0xfVkVSVEVYX0FSUkFZICAgICAgICAgICAgICAgICAgIDB4ODA3NAotI2RlZmluZSBHTF9OT1JNQUxfQVJSQVkgICAgICAgICAgICAgICAgICAgMHg4MDc1Ci0jZGVmaW5lIEdMX0NPTE9SX0FSUkFZICAgICAgICAgICAgICAgICAgICAweDgwNzYKLSNkZWZpbmUgR0xfVEVYVFVSRV9DT09SRF9BUlJBWSAgICAgICAgICAgIDB4ODA3OAotI2RlZmluZSBHTF9NVUxUSVNBTVBMRSAgICAgICAgICAgICAgICAgICAgMHg4MDlECi0jZGVmaW5lIEdMX1NBTVBMRV9BTFBIQV9UT19DT1ZFUkFHRSAgICAgICAweDgwOUUKLSNkZWZpbmUgR0xfU0FNUExFX0FMUEhBX1RPX09ORSAgICAgICAgICAgIDB4ODA5RgotI2RlZmluZSBHTF9TQU1QTEVfQ09WRVJBR0UgICAgICAgICAgICAgICAgMHg4MEEwCi0KLS8qIEVycm9yQ29kZSAqLwotI2RlZmluZSBHTF9OT19FUlJPUiAgICAgICAgICAgICAgICAgICAgICAgMAotI2RlZmluZSBHTF9JTlZBTElEX0VOVU0gICAgICAgICAgICAgICAgICAgMHgwNTAwCi0jZGVmaW5lIEdMX0lOVkFMSURfVkFMVUUgICAgICAgICAgICAgICAgICAweDA1MDEKLSNkZWZpbmUgR0xfSU5WQUxJRF9PUEVSQVRJT04gICAgICAgICAgICAgIDB4MDUwMgotI2RlZmluZSBHTF9TVEFDS19PVkVSRkxPVyAgICAgICAgICAgICAgICAgMHgwNTAzCi0jZGVmaW5lIEdMX1NUQUNLX1VOREVSRkxPVyAgICAgICAgICAgICAgICAweDA1MDQKLSNkZWZpbmUgR0xfT1VUX09GX01FTU9SWSAgICAgICAgICAgICAgICAgIDB4MDUwNQotCi0vKiBGb2dNb2RlICovCi0vKiAgICAgIEdMX0xJTkVBUiAqLwotI2RlZmluZSBHTF9FWFAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwODAwCi0jZGVmaW5lIEdMX0VYUDIgICAgICAgICAgICAgICAgICAgICAgICAgICAweDA4MDEKLQotLyogRm9nUGFyYW1ldGVyICovCi0jZGVmaW5lIEdMX0ZPR19ERU5TSVRZICAgICAgICAgICAgICAgICAgICAweDBCNjIKLSNkZWZpbmUgR0xfRk9HX1NUQVJUICAgICAgICAgICAgICAgICAgICAgIDB4MEI2MwotI2RlZmluZSBHTF9GT0dfRU5EICAgICAgICAgICAgICAgICAgICAgICAgMHgwQjY0Ci0jZGVmaW5lIEdMX0ZPR19NT0RFICAgICAgICAgICAgICAgICAgICAgICAweDBCNjUKLSNkZWZpbmUgR0xfRk9HX0NPTE9SICAgICAgICAgICAgICAgICAgICAgIDB4MEI2NgotCi0vKiBGcm9udEZhY2VEaXJlY3Rpb24gKi8KLSNkZWZpbmUgR0xfQ1cgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDkwMAotI2RlZmluZSBHTF9DQ1cgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwOTAxCi0KLS8qIEdldFBOYW1lICovCi0jZGVmaW5lIEdMX0NVUlJFTlRfQ09MT1IgICAgICAgICAgICAgICAgICAweDBCMDAKLSNkZWZpbmUgR0xfQ1VSUkVOVF9OT1JNQUwgICAgICAgICAgICAgICAgIDB4MEIwMgotI2RlZmluZSBHTF9DVVJSRU5UX1RFWFRVUkVfQ09PUkRTICAgICAgICAgMHgwQjAzCi0jZGVmaW5lIEdMX1BPSU5UX1NJWkUgICAgICAgICAgICAgICAgICAgICAweDBCMTEKLSNkZWZpbmUgR0xfUE9JTlRfU0laRV9NSU4gICAgICAgICAgICAgICAgIDB4ODEyNgotI2RlZmluZSBHTF9QT0lOVF9TSVpFX01BWCAgICAgICAgICAgICAgICAgMHg4MTI3Ci0jZGVmaW5lIEdMX1BPSU5UX0ZBREVfVEhSRVNIT0xEX1NJWkUgICAgICAweDgxMjgKLSNkZWZpbmUgR0xfUE9JTlRfRElTVEFOQ0VfQVRURU5VQVRJT04gICAgIDB4ODEyOQotI2RlZmluZSBHTF9TTU9PVEhfUE9JTlRfU0laRV9SQU5HRSAgICAgICAgMHgwQjEyCi0jZGVmaW5lIEdMX0xJTkVfV0lEVEggICAgICAgICAgICAgICAgICAgICAweDBCMjEKLSNkZWZpbmUgR0xfU01PT1RIX0xJTkVfV0lEVEhfUkFOR0UgICAgICAgIDB4MEIyMgotI2RlZmluZSBHTF9BTElBU0VEX1BPSU5UX1NJWkVfUkFOR0UgICAgICAgMHg4NDZECi0jZGVmaW5lIEdMX0FMSUFTRURfTElORV9XSURUSF9SQU5HRSAgICAgICAweDg0NkUKLSNkZWZpbmUgR0xfQ1VMTF9GQUNFX01PREUgICAgICAgICAgICAgICAgIDB4MEI0NQotI2RlZmluZSBHTF9GUk9OVF9GQUNFICAgICAgICAgICAgICAgICAgICAgMHgwQjQ2Ci0jZGVmaW5lIEdMX1NIQURFX01PREVMICAgICAgICAgICAgICAgICAgICAweDBCNTQKLSNkZWZpbmUgR0xfREVQVEhfUkFOR0UgICAgICAgICAgICAgICAgICAgIDB4MEI3MAotI2RlZmluZSBHTF9ERVBUSF9XUklURU1BU0sgICAgICAgICAgICAgICAgMHgwQjcyCi0jZGVmaW5lIEdMX0RFUFRIX0NMRUFSX1ZBTFVFICAgICAgICAgICAgICAweDBCNzMKLSNkZWZpbmUgR0xfREVQVEhfRlVOQyAgICAgICAgICAgICAgICAgICAgIDB4MEI3NAotI2RlZmluZSBHTF9TVEVOQ0lMX0NMRUFSX1ZBTFVFICAgICAgICAgICAgMHgwQjkxCi0jZGVmaW5lIEdMX1NURU5DSUxfRlVOQyAgICAgICAgICAgICAgICAgICAweDBCOTIKLSNkZWZpbmUgR0xfU1RFTkNJTF9WQUxVRV9NQVNLICAgICAgICAgICAgIDB4MEI5MwotI2RlZmluZSBHTF9TVEVOQ0lMX0ZBSUwgICAgICAgICAgICAgICAgICAgMHgwQjk0Ci0jZGVmaW5lIEdMX1NURU5DSUxfUEFTU19ERVBUSF9GQUlMICAgICAgICAweDBCOTUKLSNkZWZpbmUgR0xfU1RFTkNJTF9QQVNTX0RFUFRIX1BBU1MgICAgICAgIDB4MEI5NgotI2RlZmluZSBHTF9TVEVOQ0lMX1JFRiAgICAgICAgICAgICAgICAgICAgMHgwQjk3Ci0jZGVmaW5lIEdMX1NURU5DSUxfV1JJVEVNQVNLICAgICAgICAgICAgICAweDBCOTgKLSNkZWZpbmUgR0xfTUFUUklYX01PREUgICAgICAgICAgICAgICAgICAgIDB4MEJBMAotI2RlZmluZSBHTF9WSUVXUE9SVCAgICAgICAgICAgICAgICAgICAgICAgMHgwQkEyCi0jZGVmaW5lIEdMX01PREVMVklFV19TVEFDS19ERVBUSCAgICAgICAgICAweDBCQTMKLSNkZWZpbmUgR0xfUFJPSkVDVElPTl9TVEFDS19ERVBUSCAgICAgICAgIDB4MEJBNAotI2RlZmluZSBHTF9URVhUVVJFX1NUQUNLX0RFUFRIICAgICAgICAgICAgMHgwQkE1Ci0jZGVmaW5lIEdMX01PREVMVklFV19NQVRSSVggICAgICAgICAgICAgICAweDBCQTYKLSNkZWZpbmUgR0xfUFJPSkVDVElPTl9NQVRSSVggICAgICAgICAgICAgIDB4MEJBNwotI2RlZmluZSBHTF9URVhUVVJFX01BVFJJWCAgICAgICAgICAgICAgICAgMHgwQkE4Ci0jZGVmaW5lIEdMX0FMUEhBX1RFU1RfRlVOQyAgICAgICAgICAgICAgICAweDBCQzEKLSNkZWZpbmUgR0xfQUxQSEFfVEVTVF9SRUYgICAgICAgICAgICAgICAgIDB4MEJDMgotI2RlZmluZSBHTF9CTEVORF9EU1QgICAgICAgICAgICAgICAgICAgICAgMHgwQkUwCi0jZGVmaW5lIEdMX0JMRU5EX1NSQyAgICAgICAgICAgICAgICAgICAgICAweDBCRTEKLSNkZWZpbmUgR0xfTE9HSUNfT1BfTU9ERSAgICAgICAgICAgICAgICAgIDB4MEJGMAotI2RlZmluZSBHTF9TQ0lTU09SX0JPWCAgICAgICAgICAgICAgICAgICAgMHgwQzEwCi0jZGVmaW5lIEdMX1NDSVNTT1JfVEVTVCAgICAgICAgICAgICAgICAgICAweDBDMTEKLSNkZWZpbmUgR0xfQ09MT1JfQ0xFQVJfVkFMVUUgICAgICAgICAgICAgIDB4MEMyMgotI2RlZmluZSBHTF9DT0xPUl9XUklURU1BU0sgICAgICAgICAgICAgICAgMHgwQzIzCi0jZGVmaW5lIEdMX1VOUEFDS19BTElHTk1FTlQgICAgICAgICAgICAgICAweDBDRjUKLSNkZWZpbmUgR0xfUEFDS19BTElHTk1FTlQgICAgICAgICAgICAgICAgIDB4MEQwNQotI2RlZmluZSBHTF9NQVhfTElHSFRTICAgICAgICAgICAgICAgICAgICAgMHgwRDMxCi0jZGVmaW5lIEdMX01BWF9DTElQX1BMQU5FUyAgICAgICAgICAgICAgICAweDBEMzIKLSNkZWZpbmUgR0xfTUFYX1RFWFRVUkVfU0laRSAgICAgICAgICAgICAgIDB4MEQzMwotI2RlZmluZSBHTF9NQVhfTU9ERUxWSUVXX1NUQUNLX0RFUFRIICAgICAgMHgwRDM2Ci0jZGVmaW5lIEdMX01BWF9QUk9KRUNUSU9OX1NUQUNLX0RFUFRIICAgICAweDBEMzgKLSNkZWZpbmUgR0xfTUFYX1RFWFRVUkVfU1RBQ0tfREVQVEggICAgICAgIDB4MEQzOQotI2RlZmluZSBHTF9NQVhfVklFV1BPUlRfRElNUyAgICAgICAgICAgICAgMHgwRDNBCi0jZGVmaW5lIEdMX01BWF9URVhUVVJFX1VOSVRTICAgICAgICAgICAgICAweDg0RTIKLSNkZWZpbmUgR0xfU1VCUElYRUxfQklUUyAgICAgICAgICAgICAgICAgIDB4MEQ1MAotI2RlZmluZSBHTF9SRURfQklUUyAgICAgICAgICAgICAgICAgICAgICAgMHgwRDUyCi0jZGVmaW5lIEdMX0dSRUVOX0JJVFMgICAgICAgICAgICAgICAgICAgICAweDBENTMKLSNkZWZpbmUgR0xfQkxVRV9CSVRTICAgICAgICAgICAgICAgICAgICAgIDB4MEQ1NAotI2RlZmluZSBHTF9BTFBIQV9CSVRTICAgICAgICAgICAgICAgICAgICAgMHgwRDU1Ci0jZGVmaW5lIEdMX0RFUFRIX0JJVFMgICAgICAgICAgICAgICAgICAgICAweDBENTYKLSNkZWZpbmUgR0xfU1RFTkNJTF9CSVRTICAgICAgICAgICAgICAgICAgIDB4MEQ1NwotI2RlZmluZSBHTF9QT0xZR09OX09GRlNFVF9VTklUUyAgICAgICAgICAgMHgyQTAwCi0jZGVmaW5lIEdMX1BPTFlHT05fT0ZGU0VUX0ZJTEwgICAgICAgICAgICAweDgwMzcKLSNkZWZpbmUgR0xfUE9MWUdPTl9PRkZTRVRfRkFDVE9SICAgICAgICAgIDB4ODAzOAotI2RlZmluZSBHTF9URVhUVVJFX0JJTkRJTkdfMkQgICAgICAgICAgICAgMHg4MDY5Ci0jZGVmaW5lIEdMX1ZFUlRFWF9BUlJBWV9TSVpFICAgICAgICAgICAgICAweDgwN0EKLSNkZWZpbmUgR0xfVkVSVEVYX0FSUkFZX1RZUEUgICAgICAgICAgICAgIDB4ODA3QgotI2RlZmluZSBHTF9WRVJURVhfQVJSQVlfU1RSSURFICAgICAgICAgICAgMHg4MDdDCi0jZGVmaW5lIEdMX05PUk1BTF9BUlJBWV9UWVBFICAgICAgICAgICAgICAweDgwN0UKLSNkZWZpbmUgR0xfTk9STUFMX0FSUkFZX1NUUklERSAgICAgICAgICAgIDB4ODA3RgotI2RlZmluZSBHTF9DT0xPUl9BUlJBWV9TSVpFICAgICAgICAgICAgICAgMHg4MDgxCi0jZGVmaW5lIEdMX0NPTE9SX0FSUkFZX1RZUEUgICAgICAgICAgICAgICAweDgwODIKLSNkZWZpbmUgR0xfQ09MT1JfQVJSQVlfU1RSSURFICAgICAgICAgICAgIDB4ODA4MwotI2RlZmluZSBHTF9URVhUVVJFX0NPT1JEX0FSUkFZX1NJWkUgICAgICAgMHg4MDg4Ci0jZGVmaW5lIEdMX1RFWFRVUkVfQ09PUkRfQVJSQVlfVFlQRSAgICAgICAweDgwODkKLSNkZWZpbmUgR0xfVEVYVFVSRV9DT09SRF9BUlJBWV9TVFJJREUgICAgIDB4ODA4QQotI2RlZmluZSBHTF9WRVJURVhfQVJSQVlfUE9JTlRFUiAgICAgICAgICAgMHg4MDhFCi0jZGVmaW5lIEdMX05PUk1BTF9BUlJBWV9QT0lOVEVSICAgICAgICAgICAweDgwOEYKLSNkZWZpbmUgR0xfQ09MT1JfQVJSQVlfUE9JTlRFUiAgICAgICAgICAgIDB4ODA5MAotI2RlZmluZSBHTF9URVhUVVJFX0NPT1JEX0FSUkFZX1BPSU5URVIgICAgMHg4MDkyCi0jZGVmaW5lIEdMX1NBTVBMRV9CVUZGRVJTICAgICAgICAgICAgICAgICAweDgwQTgKLSNkZWZpbmUgR0xfU0FNUExFUyAgICAgICAgICAgICAgICAgICAgICAgIDB4ODBBOQotI2RlZmluZSBHTF9TQU1QTEVfQ09WRVJBR0VfVkFMVUUgICAgICAgICAgMHg4MEFBCi0jZGVmaW5lIEdMX1NBTVBMRV9DT1ZFUkFHRV9JTlZFUlQgICAgICAgICAweDgwQUIKLQotLyogR2V0VGV4dHVyZVBhcmFtZXRlciAqLwotLyogICAgICBHTF9URVhUVVJFX01BR19GSUxURVIgKi8KLS8qICAgICAgR0xfVEVYVFVSRV9NSU5fRklMVEVSICovCi0vKiAgICAgIEdMX1RFWFRVUkVfV1JBUF9TICovCi0vKiAgICAgIEdMX1RFWFRVUkVfV1JBUF9UICovCi0KLSNkZWZpbmUgR0xfTlVNX0NPTVBSRVNTRURfVEVYVFVSRV9GT1JNQVRTIDB4ODZBMgotI2RlZmluZSBHTF9DT01QUkVTU0VEX1RFWFRVUkVfRk9STUFUUyAgICAgMHg4NkEzCi0KLS8qIEhpbnRNb2RlICovCi0jZGVmaW5lIEdMX0RPTlRfQ0FSRSAgICAgICAgICAgICAgICAgICAgICAweDExMDAKLSNkZWZpbmUgR0xfRkFTVEVTVCAgICAgICAgICAgICAgICAgICAgICAgIDB4MTEwMQotI2RlZmluZSBHTF9OSUNFU1QgICAgICAgICAgICAgICAgICAgICAgICAgMHgxMTAyCi0KLS8qIEhpbnRUYXJnZXQgKi8KLSNkZWZpbmUgR0xfUEVSU1BFQ1RJVkVfQ09SUkVDVElPTl9ISU5UICAgIDB4MEM1MAotI2RlZmluZSBHTF9QT0lOVF9TTU9PVEhfSElOVCAgICAgICAgICAgICAgMHgwQzUxCi0jZGVmaW5lIEdMX0xJTkVfU01PT1RIX0hJTlQgICAgICAgICAgICAgICAweDBDNTIKLSNkZWZpbmUgR0xfRk9HX0hJTlQgICAgICAgICAgICAgICAgICAgICAgIDB4MEM1NAotI2RlZmluZSBHTF9HRU5FUkFURV9NSVBNQVBfSElOVCAgICAgICAgICAgMHg4MTkyCi0KLS8qIExpZ2h0TW9kZWxQYXJhbWV0ZXIgKi8KLSNkZWZpbmUgR0xfTElHSFRfTU9ERUxfQU1CSUVOVCAgICAgICAgICAgIDB4MEI1MwotI2RlZmluZSBHTF9MSUdIVF9NT0RFTF9UV09fU0lERSAgICAgICAgICAgMHgwQjUyCi0KLS8qIExpZ2h0UGFyYW1ldGVyICovCi0jZGVmaW5lIEdMX0FNQklFTlQgICAgICAgICAgICAgICAgICAgICAgICAweDEyMDAKLSNkZWZpbmUgR0xfRElGRlVTRSAgICAgICAgICAgICAgICAgICAgICAgIDB4MTIwMQotI2RlZmluZSBHTF9TUEVDVUxBUiAgICAgICAgICAgICAgICAgICAgICAgMHgxMjAyCi0jZGVmaW5lIEdMX1BPU0lUSU9OICAgICAgICAgICAgICAgICAgICAgICAweDEyMDMKLSNkZWZpbmUgR0xfU1BPVF9ESVJFQ1RJT04gICAgICAgICAgICAgICAgIDB4MTIwNAotI2RlZmluZSBHTF9TUE9UX0VYUE9ORU5UICAgICAgICAgICAgICAgICAgMHgxMjA1Ci0jZGVmaW5lIEdMX1NQT1RfQ1VUT0ZGICAgICAgICAgICAgICAgICAgICAweDEyMDYKLSNkZWZpbmUgR0xfQ09OU1RBTlRfQVRURU5VQVRJT04gICAgICAgICAgIDB4MTIwNwotI2RlZmluZSBHTF9MSU5FQVJfQVRURU5VQVRJT04gICAgICAgICAgICAgMHgxMjA4Ci0jZGVmaW5lIEdMX1FVQURSQVRJQ19BVFRFTlVBVElPTiAgICAgICAgICAweDEyMDkKLQotLyogRGF0YVR5cGUgKi8KLSNkZWZpbmUgR0xfQllURSAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTQwMAotI2RlZmluZSBHTF9VTlNJR05FRF9CWVRFICAgICAgICAgICAgICAgICAgMHgxNDAxCi0jZGVmaW5lIEdMX1NIT1JUICAgICAgICAgICAgICAgICAgICAgICAgICAweDE0MDIKLSNkZWZpbmUgR0xfVU5TSUdORURfU0hPUlQgICAgICAgICAgICAgICAgIDB4MTQwMwotI2RlZmluZSBHTF9GTE9BVCAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNDA2Ci0jZGVmaW5lIEdMX0ZJWEVEICAgICAgICAgICAgICAgICAgICAgICAgICAweDE0MEMKLQotLyogTG9naWNPcCAqLwotI2RlZmluZSBHTF9DTEVBUiAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNTAwCi0jZGVmaW5lIEdMX0FORCAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDE1MDEKLSNkZWZpbmUgR0xfQU5EX1JFVkVSU0UgICAgICAgICAgICAgICAgICAgIDB4MTUwMgotI2RlZmluZSBHTF9DT1BZICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNTAzCi0jZGVmaW5lIEdMX0FORF9JTlZFUlRFRCAgICAgICAgICAgICAgICAgICAweDE1MDQKLSNkZWZpbmUgR0xfTk9PUCAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTUwNQotI2RlZmluZSBHTF9YT1IgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNTA2Ci0jZGVmaW5lIEdMX09SICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDE1MDcKLSNkZWZpbmUgR0xfTk9SICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTUwOAotI2RlZmluZSBHTF9FUVVJViAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNTA5Ci0jZGVmaW5lIEdMX0lOVkVSVCAgICAgICAgICAgICAgICAgICAgICAgICAweDE1MEEKLSNkZWZpbmUgR0xfT1JfUkVWRVJTRSAgICAgICAgICAgICAgICAgICAgIDB4MTUwQgotI2RlZmluZSBHTF9DT1BZX0lOVkVSVEVEICAgICAgICAgICAgICAgICAgMHgxNTBDCi0jZGVmaW5lIEdMX09SX0lOVkVSVEVEICAgICAgICAgICAgICAgICAgICAweDE1MEQKLSNkZWZpbmUgR0xfTkFORCAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTUwRQotI2RlZmluZSBHTF9TRVQgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNTBGCi0KLS8qIE1hdGVyaWFsRmFjZSAqLwotLyogICAgICBHTF9GUk9OVF9BTkRfQkFDSyAqLwotCi0vKiBNYXRlcmlhbFBhcmFtZXRlciAqLwotI2RlZmluZSBHTF9FTUlTU0lPTiAgICAgICAgICAgICAgICAgICAgICAgMHgxNjAwCi0jZGVmaW5lIEdMX1NISU5JTkVTUyAgICAgICAgICAgICAgICAgICAgICAweDE2MDEKLSNkZWZpbmUgR0xfQU1CSUVOVF9BTkRfRElGRlVTRSAgICAgICAgICAgIDB4MTYwMgotLyogICAgICBHTF9BTUJJRU5UICovCi0vKiAgICAgIEdMX0RJRkZVU0UgKi8KLS8qICAgICAgR0xfU1BFQ1VMQVIgKi8KLQotLyogTWF0cml4TW9kZSAqLwotI2RlZmluZSBHTF9NT0RFTFZJRVcgICAgICAgICAgICAgICAgICAgICAgMHgxNzAwCi0jZGVmaW5lIEdMX1BST0pFQ1RJT04gICAgICAgICAgICAgICAgICAgICAweDE3MDEKLSNkZWZpbmUgR0xfVEVYVFVSRSAgICAgICAgICAgICAgICAgICAgICAgIDB4MTcwMgotCi0vKiBOb3JtYWxQb2ludGVyVHlwZSAqLwotLyogICAgICBHTF9CWVRFICovCi0vKiAgICAgIEdMX1NIT1JUICovCi0vKiAgICAgIEdMX0ZMT0FUICovCi0vKiAgICAgIEdMX0ZJWEVEICovCi0KLS8qIFBpeGVsRm9ybWF0ICovCi0jZGVmaW5lIEdMX0FMUEhBICAgICAgICAgICAgICAgICAgICAgICAgICAweDE5MDYKLSNkZWZpbmUgR0xfUkdCICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTkwNwotI2RlZmluZSBHTF9SR0JBICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxOTA4Ci0jZGVmaW5lIEdMX0xVTUlOQU5DRSAgICAgICAgICAgICAgICAgICAgICAweDE5MDkKLSNkZWZpbmUgR0xfTFVNSU5BTkNFX0FMUEhBICAgICAgICAgICAgICAgIDB4MTkwQQotCi0vKiBQaXhlbFN0b3JlUGFyYW1ldGVyICovCi0jZGVmaW5lIEdMX1VOUEFDS19BTElHTk1FTlQgICAgICAgICAgICAgICAweDBDRjUKLSNkZWZpbmUgR0xfUEFDS19BTElHTk1FTlQgICAgICAgICAgICAgICAgIDB4MEQwNQotCi0vKiBQaXhlbFR5cGUgKi8KLS8qICAgICAgR0xfVU5TSUdORURfQllURSAqLwotI2RlZmluZSBHTF9VTlNJR05FRF9TSE9SVF80XzRfNF80ICAgICAgICAgMHg4MDMzCi0jZGVmaW5lIEdMX1VOU0lHTkVEX1NIT1JUXzVfNV81XzEgICAgICAgICAweDgwMzQKLSNkZWZpbmUgR0xfVU5TSUdORURfU0hPUlRfNV82XzUgICAgICAgICAgIDB4ODM2MwotCi0vKiBTaGFkaW5nTW9kZWwgKi8KLSNkZWZpbmUgR0xfRkxBVCAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MUQwMAotI2RlZmluZSBHTF9TTU9PVEggICAgICAgICAgICAgICAgICAgICAgICAgMHgxRDAxCi0KLS8qIFN0ZW5jaWxGdW5jdGlvbiAqLwotLyogICAgICBHTF9ORVZFUiAqLwotLyogICAgICBHTF9MRVNTICovCi0vKiAgICAgIEdMX0VRVUFMICovCi0vKiAgICAgIEdMX0xFUVVBTCAqLwotLyogICAgICBHTF9HUkVBVEVSICovCi0vKiAgICAgIEdMX05PVEVRVUFMICovCi0vKiAgICAgIEdMX0dFUVVBTCAqLwotLyogICAgICBHTF9BTFdBWVMgKi8KLQotLyogU3RlbmNpbE9wICovCi0vKiAgICAgIEdMX1pFUk8gKi8KLSNkZWZpbmUgR0xfS0VFUCAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MUUwMAotI2RlZmluZSBHTF9SRVBMQUNFICAgICAgICAgICAgICAgICAgICAgICAgMHgxRTAxCi0jZGVmaW5lIEdMX0lOQ1IgICAgICAgICAgICAgICAgICAgICAgICAgICAweDFFMDIKLSNkZWZpbmUgR0xfREVDUiAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MUUwMwotLyogICAgICBHTF9JTlZFUlQgKi8KLQotLyogU3RyaW5nTmFtZSAqLwotI2RlZmluZSBHTF9WRU5ET1IgICAgICAgICAgICAgICAgICAgICAgICAgMHgxRjAwCi0jZGVmaW5lIEdMX1JFTkRFUkVSICAgICAgICAgICAgICAgICAgICAgICAweDFGMDEKLSNkZWZpbmUgR0xfVkVSU0lPTiAgICAgICAgICAgICAgICAgICAgICAgIDB4MUYwMgotI2RlZmluZSBHTF9FWFRFTlNJT05TICAgICAgICAgICAgICAgICAgICAgMHgxRjAzCi0KLS8qIFRleENvb3JkUG9pbnRlclR5cGUgKi8KLS8qICAgICAgR0xfU0hPUlQgKi8KLS8qICAgICAgR0xfRkxPQVQgKi8KLS8qICAgICAgR0xfRklYRUQgKi8KLS8qICAgICAgR0xfQllURSAqLwotCi0vKiBUZXh0dXJlRW52TW9kZSAqLwotI2RlZmluZSBHTF9NT0RVTEFURSAgICAgICAgICAgICAgICAgICAgICAgMHgyMTAwCi0jZGVmaW5lIEdMX0RFQ0FMICAgICAgICAgICAgICAgICAgICAgICAgICAweDIxMDEKLS8qICAgICAgR0xfQkxFTkQgKi8KLSNkZWZpbmUgR0xfQUREICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDEwNAotLyogICAgICBHTF9SRVBMQUNFICovCi0KLS8qIFRleHR1cmVFbnZQYXJhbWV0ZXIgKi8KLSNkZWZpbmUgR0xfVEVYVFVSRV9FTlZfTU9ERSAgICAgICAgICAgICAgIDB4MjIwMAotI2RlZmluZSBHTF9URVhUVVJFX0VOVl9DT0xPUiAgICAgICAgICAgICAgMHgyMjAxCi0KLS8qIFRleHR1cmVFbnZUYXJnZXQgKi8KLSNkZWZpbmUgR0xfVEVYVFVSRV9FTlYgICAgICAgICAgICAgICAgICAgIDB4MjMwMAotCi0vKiBUZXh0dXJlTWFnRmlsdGVyICovCi0jZGVmaW5lIEdMX05FQVJFU1QgICAgICAgICAgICAgICAgICAgICAgICAweDI2MDAKLSNkZWZpbmUgR0xfTElORUFSICAgICAgICAgICAgICAgICAgICAgICAgIDB4MjYwMQotCi0vKiBUZXh0dXJlTWluRmlsdGVyICovCi0vKiAgICAgIEdMX05FQVJFU1QgKi8KLS8qICAgICAgR0xfTElORUFSICovCi0jZGVmaW5lIEdMX05FQVJFU1RfTUlQTUFQX05FQVJFU1QgICAgICAgICAweDI3MDAKLSNkZWZpbmUgR0xfTElORUFSX01JUE1BUF9ORUFSRVNUICAgICAgICAgIDB4MjcwMQotI2RlZmluZSBHTF9ORUFSRVNUX01JUE1BUF9MSU5FQVIgICAgICAgICAgMHgyNzAyCi0jZGVmaW5lIEdMX0xJTkVBUl9NSVBNQVBfTElORUFSICAgICAgICAgICAweDI3MDMKLQotLyogVGV4dHVyZVBhcmFtZXRlck5hbWUgKi8KLSNkZWZpbmUgR0xfVEVYVFVSRV9NQUdfRklMVEVSICAgICAgICAgICAgIDB4MjgwMAotI2RlZmluZSBHTF9URVhUVVJFX01JTl9GSUxURVIgICAgICAgICAgICAgMHgyODAxCi0jZGVmaW5lIEdMX1RFWFRVUkVfV1JBUF9TICAgICAgICAgICAgICAgICAweDI4MDIKLSNkZWZpbmUgR0xfVEVYVFVSRV9XUkFQX1QgICAgICAgICAgICAgICAgIDB4MjgwMwotI2RlZmluZSBHTF9HRU5FUkFURV9NSVBNQVAgICAgICAgICAgICAgICAgMHg4MTkxCi0KLS8qIFRleHR1cmVUYXJnZXQgKi8KLS8qICAgICAgR0xfVEVYVFVSRV8yRCAqLwotCi0vKiBUZXh0dXJlVW5pdCAqLwotI2RlZmluZSBHTF9URVhUVVJFMCAgICAgICAgICAgICAgICAgICAgICAgMHg4NEMwCi0jZGVmaW5lIEdMX1RFWFRVUkUxICAgICAgICAgICAgICAgICAgICAgICAweDg0QzEKLSNkZWZpbmUgR0xfVEVYVFVSRTIgICAgICAgICAgICAgICAgICAgICAgIDB4ODRDMgotI2RlZmluZSBHTF9URVhUVVJFMyAgICAgICAgICAgICAgICAgICAgICAgMHg4NEMzCi0jZGVmaW5lIEdMX1RFWFRVUkU0ICAgICAgICAgICAgICAgICAgICAgICAweDg0QzQKLSNkZWZpbmUgR0xfVEVYVFVSRTUgICAgICAgICAgICAgICAgICAgICAgIDB4ODRDNQotI2RlZmluZSBHTF9URVhUVVJFNiAgICAgICAgICAgICAgICAgICAgICAgMHg4NEM2Ci0jZGVmaW5lIEdMX1RFWFRVUkU3ICAgICAgICAgICAgICAgICAgICAgICAweDg0QzcKLSNkZWZpbmUgR0xfVEVYVFVSRTggICAgICAgICAgICAgICAgICAgICAgIDB4ODRDOAotI2RlZmluZSBHTF9URVhUVVJFOSAgICAgICAgICAgICAgICAgICAgICAgMHg4NEM5Ci0jZGVmaW5lIEdMX1RFWFRVUkUxMCAgICAgICAgICAgICAgICAgICAgICAweDg0Q0EKLSNkZWZpbmUgR0xfVEVYVFVSRTExICAgICAgICAgICAgICAgICAgICAgIDB4ODRDQgotI2RlZmluZSBHTF9URVhUVVJFMTIgICAgICAgICAgICAgICAgICAgICAgMHg4NENDCi0jZGVmaW5lIEdMX1RFWFRVUkUxMyAgICAgICAgICAgICAgICAgICAgICAweDg0Q0QKLSNkZWZpbmUgR0xfVEVYVFVSRTE0ICAgICAgICAgICAgICAgICAgICAgIDB4ODRDRQotI2RlZmluZSBHTF9URVhUVVJFMTUgICAgICAgICAgICAgICAgICAgICAgMHg4NENGCi0jZGVmaW5lIEdMX1RFWFRVUkUxNiAgICAgICAgICAgICAgICAgICAgICAweDg0RDAKLSNkZWZpbmUgR0xfVEVYVFVSRTE3ICAgICAgICAgICAgICAgICAgICAgIDB4ODREMQotI2RlZmluZSBHTF9URVhUVVJFMTggICAgICAgICAgICAgICAgICAgICAgMHg4NEQyCi0jZGVmaW5lIEdMX1RFWFRVUkUxOSAgICAgICAgICAgICAgICAgICAgICAweDg0RDMKLSNkZWZpbmUgR0xfVEVYVFVSRTIwICAgICAgICAgICAgICAgICAgICAgIDB4ODRENAotI2RlZmluZSBHTF9URVhUVVJFMjEgICAgICAgICAgICAgICAgICAgICAgMHg4NEQ1Ci0jZGVmaW5lIEdMX1RFWFRVUkUyMiAgICAgICAgICAgICAgICAgICAgICAweDg0RDYKLSNkZWZpbmUgR0xfVEVYVFVSRTIzICAgICAgICAgICAgICAgICAgICAgIDB4ODRENwotI2RlZmluZSBHTF9URVhUVVJFMjQgICAgICAgICAgICAgICAgICAgICAgMHg4NEQ4Ci0jZGVmaW5lIEdMX1RFWFRVUkUyNSAgICAgICAgICAgICAgICAgICAgICAweDg0RDkKLSNkZWZpbmUgR0xfVEVYVFVSRTI2ICAgICAgICAgICAgICAgICAgICAgIDB4ODREQQotI2RlZmluZSBHTF9URVhUVVJFMjcgICAgICAgICAgICAgICAgICAgICAgMHg4NERCCi0jZGVmaW5lIEdMX1RFWFRVUkUyOCAgICAgICAgICAgICAgICAgICAgICAweDg0REMKLSNkZWZpbmUgR0xfVEVYVFVSRTI5ICAgICAgICAgICAgICAgICAgICAgIDB4ODRERAotI2RlZmluZSBHTF9URVhUVVJFMzAgICAgICAgICAgICAgICAgICAgICAgMHg4NERFCi0jZGVmaW5lIEdMX1RFWFRVUkUzMSAgICAgICAgICAgICAgICAgICAgICAweDg0REYKLSNkZWZpbmUgR0xfQUNUSVZFX1RFWFRVUkUgICAgICAgICAgICAgICAgIDB4ODRFMAotI2RlZmluZSBHTF9DTElFTlRfQUNUSVZFX1RFWFRVUkUgICAgICAgICAgMHg4NEUxCi0KLS8qIFRleHR1cmVXcmFwTW9kZSAqLwotI2RlZmluZSBHTF9SRVBFQVQgICAgICAgICAgICAgICAgICAgICAgICAgMHgyOTAxCi0jZGVmaW5lIEdMX0NMQU1QX1RPX0VER0UgICAgICAgICAgICAgICAgICAweDgxMkYKLQotLyogVmVydGV4UG9pbnRlclR5cGUgKi8KLS8qICAgICAgR0xfU0hPUlQgKi8KLS8qICAgICAgR0xfRkxPQVQgKi8KLS8qICAgICAgR0xfRklYRUQgKi8KLS8qICAgICAgR0xfQllURSAqLwotCi0vKiBMaWdodE5hbWUgKi8KLSNkZWZpbmUgR0xfTElHSFQwICAgICAgICAgICAgICAgICAgICAgICAgIDB4NDAwMAotI2RlZmluZSBHTF9MSUdIVDEgICAgICAgICAgICAgICAgICAgICAgICAgMHg0MDAxCi0jZGVmaW5lIEdMX0xJR0hUMiAgICAgICAgICAgICAgICAgICAgICAgICAweDQwMDIKLSNkZWZpbmUgR0xfTElHSFQzICAgICAgICAgICAgICAgICAgICAgICAgIDB4NDAwMwotI2RlZmluZSBHTF9MSUdIVDQgICAgICAgICAgICAgICAgICAgICAgICAgMHg0MDA0Ci0jZGVmaW5lIEdMX0xJR0hUNSAgICAgICAgICAgICAgICAgICAgICAgICAweDQwMDUKLSNkZWZpbmUgR0xfTElHSFQ2ICAgICAgICAgICAgICAgICAgICAgICAgIDB4NDAwNgotI2RlZmluZSBHTF9MSUdIVDcgICAgICAgICAgICAgICAgICAgICAgICAgMHg0MDA3Ci0KLS8qIEJ1ZmZlciBPYmplY3RzICovCi0jZGVmaW5lIEdMX0FSUkFZX0JVRkZFUiAgICAgICAgICAgICAgICAgICAweDg4OTIKLSNkZWZpbmUgR0xfRUxFTUVOVF9BUlJBWV9CVUZGRVIgICAgICAgICAgIDB4ODg5MwotCi0jZGVmaW5lIEdMX0FSUkFZX0JVRkZFUl9CSU5ESU5HICAgICAgICAgICAgICAgMHg4ODk0Ci0jZGVmaW5lIEdMX0VMRU1FTlRfQVJSQVlfQlVGRkVSX0JJTkRJTkcgICAgICAgMHg4ODk1Ci0jZGVmaW5lIEdMX1ZFUlRFWF9BUlJBWV9CVUZGRVJfQklORElORyAgICAgICAgMHg4ODk2Ci0jZGVmaW5lIEdMX05PUk1BTF9BUlJBWV9CVUZGRVJfQklORElORyAgICAgICAgMHg4ODk3Ci0jZGVmaW5lIEdMX0NPTE9SX0FSUkFZX0JVRkZFUl9CSU5ESU5HICAgICAgICAgMHg4ODk4Ci0jZGVmaW5lIEdMX1RFWFRVUkVfQ09PUkRfQVJSQVlfQlVGRkVSX0JJTkRJTkcgMHg4ODlBCi0KLSNkZWZpbmUgR0xfU1RBVElDX0RSQVcgICAgICAgICAgICAgICAgICAgIDB4ODhFNAotI2RlZmluZSBHTF9EWU5BTUlDX0RSQVcgICAgICAgICAgICAgICAgICAgMHg4OEU4Ci0KLSNkZWZpbmUgR0xfQlVGRkVSX1NJWkUgICAgICAgICAgICAgICAgICAgIDB4ODc2NAotI2RlZmluZSBHTF9CVUZGRVJfVVNBR0UgICAgICAgICAgICAgICAgICAgMHg4NzY1Ci0KLS8qIFRleHR1cmUgY29tYmluZSArIGRvdDMgKi8KLSNkZWZpbmUgR0xfU1VCVFJBQ1QgICAgICAgICAgICAgICAgICAgICAgIDB4ODRFNwotI2RlZmluZSBHTF9DT01CSU5FICAgICAgICAgICAgICAgICAgICAgICAgMHg4NTcwCi0jZGVmaW5lIEdMX0NPTUJJTkVfUkdCICAgICAgICAgICAgICAgICAgICAweDg1NzEKLSNkZWZpbmUgR0xfQ09NQklORV9BTFBIQSAgICAgICAgICAgICAgICAgIDB4ODU3MgotI2RlZmluZSBHTF9SR0JfU0NBTEUgICAgICAgICAgICAgICAgICAgICAgMHg4NTczCi0jZGVmaW5lIEdMX0FERF9TSUdORUQgICAgICAgICAgICAgICAgICAgICAweDg1NzQKLSNkZWZpbmUgR0xfSU5URVJQT0xBVEUgICAgICAgICAgICAgICAgICAgIDB4ODU3NQotI2RlZmluZSBHTF9DT05TVEFOVCAgICAgICAgICAgICAgICAgICAgICAgMHg4NTc2Ci0jZGVmaW5lIEdMX1BSSU1BUllfQ09MT1IgICAgICAgICAgICAgICAgICAweDg1NzcKLSNkZWZpbmUgR0xfUFJFVklPVVMgICAgICAgICAgICAgICAgICAgICAgIDB4ODU3OAotI2RlZmluZSBHTF9PUEVSQU5EMF9SR0IgICAgICAgICAgICAgICAgICAgMHg4NTkwCi0jZGVmaW5lIEdMX09QRVJBTkQxX1JHQiAgICAgICAgICAgICAgICAgICAweDg1OTEKLSNkZWZpbmUgR0xfT1BFUkFORDJfUkdCICAgICAgICAgICAgICAgICAgIDB4ODU5MgotI2RlZmluZSBHTF9PUEVSQU5EMF9BTFBIQSAgICAgICAgICAgICAgICAgMHg4NTk4Ci0jZGVmaW5lIEdMX09QRVJBTkQxX0FMUEhBICAgICAgICAgICAgICAgICAweDg1OTkKLSNkZWZpbmUgR0xfT1BFUkFORDJfQUxQSEEgICAgICAgICAgICAgICAgIDB4ODU5QQotCi0jZGVmaW5lIEdMX0FMUEhBX1NDQUxFICAgICAgICAgICAgICAgICAgICAweDBEMUMKLQotI2RlZmluZSBHTF9TUkMwX1JHQiAgICAgICAgICAgICAgICAgICAgICAgMHg4NTgwCi0jZGVmaW5lIEdMX1NSQzFfUkdCICAgICAgICAgICAgICAgICAgICAgICAweDg1ODEKLSNkZWZpbmUgR0xfU1JDMl9SR0IgICAgICAgICAgICAgICAgICAgICAgIDB4ODU4MgotI2RlZmluZSBHTF9TUkMwX0FMUEhBICAgICAgICAgICAgICAgICAgICAgMHg4NTg4Ci0jZGVmaW5lIEdMX1NSQzFfQUxQSEEgICAgICAgICAgICAgICAgICAgICAweDg1ODkKLSNkZWZpbmUgR0xfU1JDMl9BTFBIQSAgICAgICAgICAgICAgICAgICAgIDB4ODU4QQotCi0jZGVmaW5lIEdMX0RPVDNfUkdCICAgICAgICAgICAgICAgICAgICAgICAweDg2QUUKLSNkZWZpbmUgR0xfRE9UM19SR0JBICAgICAgICAgICAgICAgICAgICAgIDB4ODZBRgotCi0vKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSoKLSAqIHJlcXVpcmVkIE9FUyBleHRlbnNpb24gdG9rZW5zCi0gKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi0KLS8qIE9FU19yZWFkX2Zvcm1hdCAqLwotI2lmbmRlZiBHTF9PRVNfcmVhZF9mb3JtYXQKLSNkZWZpbmUgR0xfSU1QTEVNRU5UQVRJT05fQ09MT1JfUkVBRF9UWVBFX09FUyAgICAgICAgICAgICAgICAgICAweDhCOUEKLSNkZWZpbmUgR0xfSU1QTEVNRU5UQVRJT05fQ09MT1JfUkVBRF9GT1JNQVRfT0VTICAgICAgICAgICAgICAgICAweDhCOUIKLSNlbmRpZgotCi0vKiBHTF9PRVNfY29tcHJlc3NlZF9wYWxldHRlZF90ZXh0dXJlICovCi0jaWZuZGVmIEdMX09FU19jb21wcmVzc2VkX3BhbGV0dGVkX3RleHR1cmUKLSNkZWZpbmUgR0xfUEFMRVRURTRfUkdCOF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTAKLSNkZWZpbmUgR0xfUEFMRVRURTRfUkdCQThfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTEKLSNkZWZpbmUgR0xfUEFMRVRURTRfUjVfRzZfQjVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTIKLSNkZWZpbmUgR0xfUEFMRVRURTRfUkdCQTRfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTMKLSNkZWZpbmUgR0xfUEFMRVRURTRfUkdCNV9BMV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTQKLSNkZWZpbmUgR0xfUEFMRVRURThfUkdCOF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTUKLSNkZWZpbmUgR0xfUEFMRVRURThfUkdCQThfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTYKLSNkZWZpbmUgR0xfUEFMRVRURThfUjVfRzZfQjVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTcKLSNkZWZpbmUgR0xfUEFMRVRURThfUkdCQTRfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTgKLSNkZWZpbmUgR0xfUEFMRVRURThfUkdCNV9BMV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhCOTkKLSNlbmRpZgotCi0vKiBPRVNfcG9pbnRfc2l6ZV9hcnJheSAqLwotI2lmbmRlZiBHTF9PRVNfcG9pbnRfc2l6ZV9hcnJheQotI2RlZmluZSBHTF9QT0lOVF9TSVpFX0FSUkFZX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4OEI5QwotI2RlZmluZSBHTF9QT0lOVF9TSVpFX0FSUkFZX1RZUEVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODk4QQotI2RlZmluZSBHTF9QT0lOVF9TSVpFX0FSUkFZX1NUUklERV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODk4QgotI2RlZmluZSBHTF9QT0lOVF9TSVpFX0FSUkFZX1BPSU5URVJfT0VTICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODk4QwotI2RlZmluZSBHTF9QT0lOVF9TSVpFX0FSUkFZX0JVRkZFUl9CSU5ESU5HX09FUyAgICAgICAgICAgICAgICAgIDB4OEI5RgotI2VuZGlmCi0KLS8qIEdMX09FU19wb2ludF9zcHJpdGUgKi8KLSNpZm5kZWYgR0xfT0VTX3BvaW50X3Nwcml0ZQotI2RlZmluZSBHTF9QT0lOVF9TUFJJVEVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODg2MQotI2RlZmluZSBHTF9DT09SRF9SRVBMQUNFX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODg2MgotI2VuZGlmCi0KLS8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwotCi0vKiBBdmFpbGFibGUgb25seSBpbiBDb21tb24gcHJvZmlsZSAqLwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xBbHBoYUZ1bmMgKEdMZW51bSBmdW5jLCBHTGNsYW1wZiByZWYpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDbGVhckNvbG9yIChHTGNsYW1wZiByZWQsIEdMY2xhbXBmIGdyZWVuLCBHTGNsYW1wZiBibHVlLCBHTGNsYW1wZiBhbHBoYSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENsZWFyRGVwdGhmIChHTGNsYW1wZiBkZXB0aCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENsaXBQbGFuZWYgKEdMZW51bSBwbGFuZSwgY29uc3QgR0xmbG9hdCAqZXF1YXRpb24pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDb2xvcjRmIChHTGZsb2F0IHJlZCwgR0xmbG9hdCBncmVlbiwgR0xmbG9hdCBibHVlLCBHTGZsb2F0IGFscGhhKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRGVwdGhSYW5nZWYgKEdMY2xhbXBmIHpOZWFyLCBHTGNsYW1wZiB6RmFyKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRm9nZiAoR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRm9nZnYgKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRnJ1c3R1bWYgKEdMZmxvYXQgbGVmdCwgR0xmbG9hdCByaWdodCwgR0xmbG9hdCBib3R0b20sIEdMZmxvYXQgdG9wLCBHTGZsb2F0IHpOZWFyLCBHTGZsb2F0IHpGYXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRDbGlwUGxhbmVmIChHTGVudW0gcG5hbWUsIEdMZmxvYXQgZXFuWzRdKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsR2V0RmxvYXR2IChHTGVudW0gcG5hbWUsIEdMZmxvYXQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldExpZ2h0ZnYgKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRNYXRlcmlhbGZ2IChHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRUZXhFbnZmdiAoR0xlbnVtIGVudiwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRUZXhQYXJhbWV0ZXJmdiAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xMaWdodE1vZGVsZiAoR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTGlnaHRNb2RlbGZ2IChHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbExpZ2h0ZiAoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xMaWdodGZ2IChHTGVudW0gbGlnaHQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTGluZVdpZHRoIChHTGZsb2F0IHdpZHRoKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTG9hZE1hdHJpeGYgKGNvbnN0IEdMZmxvYXQgKm0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xNYXRlcmlhbGYgKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xNYXRlcmlhbGZ2IChHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xNdWx0TWF0cml4ZiAoY29uc3QgR0xmbG9hdCAqbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbE11bHRpVGV4Q29vcmQ0ZiAoR0xlbnVtIHRhcmdldCwgR0xmbG9hdCBzLCBHTGZsb2F0IHQsIEdMZmxvYXQgciwgR0xmbG9hdCBxKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTm9ybWFsM2YgKEdMZmxvYXQgbngsIEdMZmxvYXQgbnksIEdMZmxvYXQgbnopOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xPcnRob2YgKEdMZmxvYXQgbGVmdCwgR0xmbG9hdCByaWdodCwgR0xmbG9hdCBib3R0b20sIEdMZmxvYXQgdG9wLCBHTGZsb2F0IHpOZWFyLCBHTGZsb2F0IHpGYXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xQb2ludFBhcmFtZXRlcmYgKEdMZW51bSBwbmFtZSwgR0xmbG9hdCBwYXJhbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFBvaW50UGFyYW1ldGVyZnYgKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsUG9pbnRTaXplIChHTGZsb2F0IHNpemUpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xQb2x5Z29uT2Zmc2V0IChHTGZsb2F0IGZhY3RvciwgR0xmbG9hdCB1bml0cyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFJvdGF0ZWYgKEdMZmxvYXQgYW5nbGUsIEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xTY2FsZWYgKEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhFbnZmIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhFbnZmdiAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhQYXJhbWV0ZXJmIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhQYXJhbWV0ZXJmdiAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUcmFuc2xhdGVmIChHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KTsKLQotLyogQXZhaWxhYmxlIGluIGJvdGggQ29tbW9uIGFuZCBDb21tb24tTGl0ZSBwcm9maWxlcyAqLwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xBY3RpdmVUZXh0dXJlIChHTGVudW0gdGV4dHVyZSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEFscGhhRnVuY3ggKEdMZW51bSBmdW5jLCBHTGNsYW1weCByZWYpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xCaW5kQnVmZmVyIChHTGVudW0gdGFyZ2V0LCBHTHVpbnQgYnVmZmVyKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQmluZFRleHR1cmUgKEdMZW51bSB0YXJnZXQsIEdMdWludCB0ZXh0dXJlKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQmxlbmRGdW5jIChHTGVudW0gc2ZhY3RvciwgR0xlbnVtIGRmYWN0b3IpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xCdWZmZXJEYXRhIChHTGVudW0gdGFyZ2V0LCBHTHNpemVpcHRyIHNpemUsIGNvbnN0IEdMdm9pZCAqZGF0YSwgR0xlbnVtIHVzYWdlKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQnVmZmVyU3ViRGF0YSAoR0xlbnVtIHRhcmdldCwgR0xpbnRwdHIgb2Zmc2V0LCBHTHNpemVpcHRyIHNpemUsIGNvbnN0IEdMdm9pZCAqZGF0YSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENsZWFyIChHTGJpdGZpZWxkIG1hc2spOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDbGVhckNvbG9yeCAoR0xjbGFtcHggcmVkLCBHTGNsYW1weCBncmVlbiwgR0xjbGFtcHggYmx1ZSwgR0xjbGFtcHggYWxwaGEpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDbGVhckRlcHRoeCAoR0xjbGFtcHggZGVwdGgpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDbGVhclN0ZW5jaWwgKEdMaW50IHMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDbGllbnRBY3RpdmVUZXh0dXJlIChHTGVudW0gdGV4dHVyZSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENsaXBQbGFuZXggKEdMZW51bSBwbGFuZSwgY29uc3QgR0xmaXhlZCAqZXF1YXRpb24pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDb2xvcjR1YiAoR0x1Ynl0ZSByZWQsIEdMdWJ5dGUgZ3JlZW4sIEdMdWJ5dGUgYmx1ZSwgR0x1Ynl0ZSBhbHBoYSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENvbG9yNHggKEdMZml4ZWQgcmVkLCBHTGZpeGVkIGdyZWVuLCBHTGZpeGVkIGJsdWUsIEdMZml4ZWQgYWxwaGEpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDb2xvck1hc2sgKEdMYm9vbGVhbiByZWQsIEdMYm9vbGVhbiBncmVlbiwgR0xib29sZWFuIGJsdWUsIEdMYm9vbGVhbiBhbHBoYSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENvbG9yUG9pbnRlciAoR0xpbnQgc2l6ZSwgR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDb21wcmVzc2VkVGV4SW1hZ2UyRCAoR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMZW51bSBpbnRlcm5hbGZvcm1hdCwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsIEdMaW50IGJvcmRlciwgR0xzaXplaSBpbWFnZVNpemUsIGNvbnN0IEdMdm9pZCAqZGF0YSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENvbXByZXNzZWRUZXhTdWJJbWFnZTJEIChHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgeG9mZnNldCwgR0xpbnQgeW9mZnNldCwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsIEdMZW51bSBmb3JtYXQsIEdMc2l6ZWkgaW1hZ2VTaXplLCBjb25zdCBHTHZvaWQgKmRhdGEpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDb3B5VGV4SW1hZ2UyRCAoR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMZW51bSBpbnRlcm5hbGZvcm1hdCwgR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsIEdMaW50IGJvcmRlcik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENvcHlUZXhTdWJJbWFnZTJEIChHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgeG9mZnNldCwgR0xpbnQgeW9mZnNldCwgR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDdWxsRmFjZSAoR0xlbnVtIG1vZGUpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEZWxldGVCdWZmZXJzIChHTHNpemVpIG4sIGNvbnN0IEdMdWludCAqYnVmZmVycyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbERlbGV0ZVRleHR1cmVzIChHTHNpemVpIG4sIGNvbnN0IEdMdWludCAqdGV4dHVyZXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEZXB0aEZ1bmMgKEdMZW51bSBmdW5jKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRGVwdGhNYXNrIChHTGJvb2xlYW4gZmxhZyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbERlcHRoUmFuZ2V4IChHTGNsYW1weCB6TmVhciwgR0xjbGFtcHggekZhcik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbERpc2FibGUgKEdMZW51bSBjYXApOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEaXNhYmxlQ2xpZW50U3RhdGUgKEdMZW51bSBhcnJheSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbERyYXdBcnJheXMgKEdMZW51bSBtb2RlLCBHTGludCBmaXJzdCwgR0xzaXplaSBjb3VudCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbERyYXdFbGVtZW50cyAoR0xlbnVtIG1vZGUsIEdMc2l6ZWkgY291bnQsIEdMZW51bSB0eXBlLCBjb25zdCBHTHZvaWQgKmluZGljZXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xFbmFibGUgKEdMZW51bSBjYXApOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xFbmFibGVDbGllbnRTdGF0ZSAoR0xlbnVtIGFycmF5KTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRmluaXNoICh2b2lkKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRmx1c2ggKHZvaWQpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xGb2d4IChHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xGb2d4diAoR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xGcm9udEZhY2UgKEdMZW51bSBtb2RlKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRnJ1c3R1bXggKEdMZml4ZWQgbGVmdCwgR0xmaXhlZCByaWdodCwgR0xmaXhlZCBib3R0b20sIEdMZml4ZWQgdG9wLCBHTGZpeGVkIHpOZWFyLCBHTGZpeGVkIHpGYXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRCb29sZWFudiAoR0xlbnVtIHBuYW1lLCBHTGJvb2xlYW4gKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldEJ1ZmZlclBhcmFtZXRlcml2IChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMaW50ICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRDbGlwUGxhbmV4IChHTGVudW0gcG5hbWUsIEdMZml4ZWQgZXFuWzRdKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsR2VuQnVmZmVycyAoR0xzaXplaSBuLCBHTHVpbnQgKmJ1ZmZlcnMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZW5UZXh0dXJlcyAoR0xzaXplaSBuLCBHTHVpbnQgKnRleHR1cmVzKTsKLUdMX0FQSSBHTGVudW0gR0xfQVBJRU5UUlkgZ2xHZXRFcnJvciAodm9pZCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldEZpeGVkdiAoR0xlbnVtIHBuYW1lLCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRJbnRlZ2VydiAoR0xlbnVtIHBuYW1lLCBHTGludCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsR2V0TGlnaHR4diAoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldE1hdGVyaWFseHYgKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldFBvaW50ZXJ2IChHTGVudW0gcG5hbWUsIHZvaWQgKipwYXJhbXMpOwotR0xfQVBJIGNvbnN0IEdMdWJ5dGUgKiBHTF9BUElFTlRSWSBnbEdldFN0cmluZyAoR0xlbnVtIG5hbWUpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRUZXhFbnZpdiAoR0xlbnVtIGVudiwgR0xlbnVtIHBuYW1lLCBHTGludCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsR2V0VGV4RW52eHYgKEdMZW51bSBlbnYsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsR2V0VGV4UGFyYW1ldGVyaXYgKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xpbnQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldFRleFBhcmFtZXRlcnh2IChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEhpbnQgKEdMZW51bSB0YXJnZXQsIEdMZW51bSBtb2RlKTsKLUdMX0FQSSBHTGJvb2xlYW4gR0xfQVBJRU5UUlkgZ2xJc0J1ZmZlciAoR0x1aW50IGJ1ZmZlcik7Ci1HTF9BUEkgR0xib29sZWFuIEdMX0FQSUVOVFJZIGdsSXNFbmFibGVkIChHTGVudW0gY2FwKTsKLUdMX0FQSSBHTGJvb2xlYW4gR0xfQVBJRU5UUlkgZ2xJc1RleHR1cmUgKEdMdWludCB0ZXh0dXJlKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTGlnaHRNb2RlbHggKEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbExpZ2h0TW9kZWx4diAoR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xMaWdodHggKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTGlnaHR4diAoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbExpbmVXaWR0aHggKEdMZml4ZWQgd2lkdGgpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xMb2FkSWRlbnRpdHkgKHZvaWQpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xMb2FkTWF0cml4eCAoY29uc3QgR0xmaXhlZCAqbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbExvZ2ljT3AgKEdMZW51bSBvcGNvZGUpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xNYXRlcmlhbHggKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xNYXRlcmlhbHh2IChHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xNYXRyaXhNb2RlIChHTGVudW0gbW9kZSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbE11bHRNYXRyaXh4IChjb25zdCBHTGZpeGVkICptKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTXVsdGlUZXhDb29yZDR4IChHTGVudW0gdGFyZ2V0LCBHTGZpeGVkIHMsIEdMZml4ZWQgdCwgR0xmaXhlZCByLCBHTGZpeGVkIHEpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xOb3JtYWwzeCAoR0xmaXhlZCBueCwgR0xmaXhlZCBueSwgR0xmaXhlZCBueik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbE5vcm1hbFBvaW50ZXIgKEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsT3J0aG94IChHTGZpeGVkIGxlZnQsIEdMZml4ZWQgcmlnaHQsIEdMZml4ZWQgYm90dG9tLCBHTGZpeGVkIHRvcCwgR0xmaXhlZCB6TmVhciwgR0xmaXhlZCB6RmFyKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsUGl4ZWxTdG9yZWkgKEdMZW51bSBwbmFtZSwgR0xpbnQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xQb2ludFBhcmFtZXRlcnggKEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFBvaW50UGFyYW1ldGVyeHYgKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsUG9pbnRTaXpleCAoR0xmaXhlZCBzaXplKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsUG9seWdvbk9mZnNldHggKEdMZml4ZWQgZmFjdG9yLCBHTGZpeGVkIHVuaXRzKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsUG9wTWF0cml4ICh2b2lkKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsUHVzaE1hdHJpeCAodm9pZCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFJlYWRQaXhlbHMgKEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LCBHTGVudW0gZm9ybWF0LCBHTGVudW0gdHlwZSwgR0x2b2lkICpwaXhlbHMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xSb3RhdGV4IChHTGZpeGVkIGFuZ2xlLCBHTGZpeGVkIHgsIEdMZml4ZWQgeSwgR0xmaXhlZCB6KTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsU2FtcGxlQ292ZXJhZ2UgKEdMY2xhbXBmIHZhbHVlLCBHTGJvb2xlYW4gaW52ZXJ0KTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsU2FtcGxlQ292ZXJhZ2V4IChHTGNsYW1weCB2YWx1ZSwgR0xib29sZWFuIGludmVydCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFNjYWxleCAoR0xmaXhlZCB4LCBHTGZpeGVkIHksIEdMZml4ZWQgeik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFNjaXNzb3IgKEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0KTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsU2hhZGVNb2RlbCAoR0xlbnVtIG1vZGUpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xTdGVuY2lsRnVuYyAoR0xlbnVtIGZ1bmMsIEdMaW50IHJlZiwgR0x1aW50IG1hc2spOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xTdGVuY2lsTWFzayAoR0x1aW50IG1hc2spOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xTdGVuY2lsT3AgKEdMZW51bSBmYWlsLCBHTGVudW0gemZhaWwsIEdMZW51bSB6cGFzcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFRleENvb3JkUG9pbnRlciAoR0xpbnQgc2l6ZSwgR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhFbnZpIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMaW50IHBhcmFtKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsVGV4RW52eCAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsVGV4RW52aXYgKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xpbnQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFRleEVudnh2IChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFRleEltYWdlMkQgKEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGludCBpbnRlcm5hbGZvcm1hdCwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsIEdMaW50IGJvcmRlciwgR0xlbnVtIGZvcm1hdCwgR0xlbnVtIHR5cGUsIGNvbnN0IEdMdm9pZCAqcGl4ZWxzKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsVGV4UGFyYW1ldGVyaSAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGludCBwYXJhbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFRleFBhcmFtZXRlcnggKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFRleFBhcmFtZXRlcml2IChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMaW50ICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhQYXJhbWV0ZXJ4diAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhTdWJJbWFnZTJEIChHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgeG9mZnNldCwgR0xpbnQgeW9mZnNldCwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsIEdMZW51bSBmb3JtYXQsIEdMZW51bSB0eXBlLCBjb25zdCBHTHZvaWQgKnBpeGVscyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFRyYW5zbGF0ZXggKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xWZXJ0ZXhQb2ludGVyIChHTGludCBzaXplLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlcik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFZpZXdwb3J0IChHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCk7Ci0KLS8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKgotICogUmVxdWlyZWQgT0VTIGV4dGVuc2lvbiBmdW5jdGlvbnMKLSAqLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLQotLyogR0xfT0VTX3JlYWRfZm9ybWF0ICovCi0jaWZuZGVmIEdMX09FU19yZWFkX2Zvcm1hdAotI2RlZmluZSBHTF9PRVNfcmVhZF9mb3JtYXQgMQotI2VuZGlmCi0KLS8qIEdMX09FU19jb21wcmVzc2VkX3BhbGV0dGVkX3RleHR1cmUgKi8KLSNpZm5kZWYgR0xfT0VTX2NvbXByZXNzZWRfcGFsZXR0ZWRfdGV4dHVyZQotI2RlZmluZSBHTF9PRVNfY29tcHJlc3NlZF9wYWxldHRlZF90ZXh0dXJlIDEKLSNlbmRpZgotCi0vKiBHTF9PRVNfcG9pbnRfc2l6ZV9hcnJheSAqLwotI2lmbmRlZiBHTF9PRVNfcG9pbnRfc2l6ZV9hcnJheQotI2RlZmluZSBHTF9PRVNfcG9pbnRfc2l6ZV9hcnJheSAxCi1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFBvaW50U2l6ZVBvaW50ZXJPRVMgKEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKTsKLSNlbmRpZgotCi0vKiBHTF9PRVNfcG9pbnRfc3ByaXRlICovCi0jaWZuZGVmIEdMX09FU19wb2ludF9zcHJpdGUKLSNkZWZpbmUgR0xfT0VTX3BvaW50X3Nwcml0ZSAxCi0jZW5kaWYKLQotI2lmZGVmIF9fY3BsdXNwbHVzCi19Ci0jZW5kaWYKLQotI2VuZGlmIC8qIF9fZ2xfaF8gKi8KLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2luY2x1ZGUvR0xFUy9nbGV4dC5oIGIvb3BlbmdsL2luY2x1ZGUvR0xFUy9nbGV4dC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0YzAxODcxLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9pbmNsdWRlL0dMRVMvZ2xleHQuaAorKysgL2Rldi9udWxsCkBAIC0xLDYyMiArMCwwIEBACi0jaWZuZGVmIF9fZ2xleHRfaF8KLSNkZWZpbmUgX19nbGV4dF9oXwotCi0vKiAkUmV2aXNpb246IDcxNzIgJCBvbiAkRGF0ZTo6IDIwMDktMDEtMDkgMTE6MTc6NDEgLTA4MDAgIyQgKi8KLQotI2lmZGVmIF9fY3BsdXNwbHVzCi1leHRlcm4gIkMiIHsKLSNlbmRpZgotCi0vKgotICogVGhpcyBkb2N1bWVudCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgU0dJIEZyZWUgU29mdHdhcmUgQiBMaWNlbnNlIFZlcnNpb24KLSAqIDIuMC4gRm9yIGRldGFpbHMsIHNlZSBodHRwOi8vb3NzLnNnaS5jb20vcHJvamVjdHMvRnJlZUIvIC4KLSAqLwotCi0jaWZuZGVmIEdMX0FQSUVOVFJZUAotIyAgIGRlZmluZSBHTF9BUElFTlRSWVAgR0xfQVBJRU5UUlkqCi0jZW5kaWYKLQotLyotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qCi0gKiBPRVMgZXh0ZW5zaW9uIHRva2VucwotICotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwotCi0vKiBHTF9PRVNfYmxlbmRfZXF1YXRpb25fc2VwYXJhdGUgKi8KLSNpZm5kZWYgR0xfT0VTX2JsZW5kX2VxdWF0aW9uX3NlcGFyYXRlCi0vKiBCTEVORF9FUVVBVElPTl9SR0JfT0VTIHNhbWUgYXMgQkxFTkRfRVFVQVRJT05fT0VTICovCi0jZGVmaW5lIEdMX0JMRU5EX0VRVUFUSU9OX1JHQl9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4MDA5Ci0jZGVmaW5lIEdMX0JMRU5EX0VRVUFUSU9OX0FMUEhBX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4ODNECi0jZW5kaWYKLQotLyogR0xfT0VTX2JsZW5kX2Z1bmNfc2VwYXJhdGUgKi8KLSNpZm5kZWYgR0xfT0VTX2JsZW5kX2Z1bmNfc2VwYXJhdGUKLSNkZWZpbmUgR0xfQkxFTkRfRFNUX1JHQl9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgwQzgKLSNkZWZpbmUgR0xfQkxFTkRfU1JDX1JHQl9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgwQzkKLSNkZWZpbmUgR0xfQkxFTkRfRFNUX0FMUEhBX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgwQ0EKLSNkZWZpbmUgR0xfQkxFTkRfU1JDX0FMUEhBX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgwQ0IKLSNlbmRpZgotCi0vKiBHTF9PRVNfYmxlbmRfc3VidHJhY3QgKi8KLSNpZm5kZWYgR0xfT0VTX2JsZW5kX3N1YnRyYWN0Ci0jZGVmaW5lIEdMX0JMRU5EX0VRVUFUSU9OX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4MDA5Ci0jZGVmaW5lIEdMX0ZVTkNfQUREX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4MDA2Ci0jZGVmaW5lIEdMX0ZVTkNfU1VCVFJBQ1RfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4MDBBCi0jZGVmaW5lIEdMX0ZVTkNfUkVWRVJTRV9TVUJUUkFDVF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4MDBCCi0jZW5kaWYKLQotLyogR0xfT0VTX2NvbXByZXNzZWRfRVRDMV9SR0I4X3RleHR1cmUgKi8KLSNpZm5kZWYgR0xfT0VTX2NvbXByZXNzZWRfRVRDMV9SR0I4X3RleHR1cmUKLSNkZWZpbmUgR0xfRVRDMV9SR0I4X09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENjQKLSNlbmRpZgotCi0vKiBHTF9PRVNfZGVwdGgyNCAqLwotI2lmbmRlZiBHTF9PRVNfZGVwdGgyNAotI2RlZmluZSBHTF9ERVBUSF9DT01QT05FTlQyNF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODFBNgotI2VuZGlmCi0KLS8qIEdMX09FU19kZXB0aDMyICovCi0jaWZuZGVmIEdMX09FU19kZXB0aDMyCi0jZGVmaW5lIEdMX0RFUFRIX0NPTVBPTkVOVDMyX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4MUE3Ci0jZW5kaWYKLQotLyogR0xfT0VTX2RyYXdfdGV4dHVyZSAqLwotI2lmbmRlZiBHTF9PRVNfZHJhd190ZXh0dXJlCi0jZGVmaW5lIEdMX1RFWFRVUkVfQ1JPUF9SRUNUX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4QjlECi0jZW5kaWYKLQotLyogR0xfT0VTX0VHTF9pbWFnZSAqLwotI2lmbmRlZiBHTF9PRVNfRUdMX2ltYWdlCi10eXBlZGVmIHZvaWQqIEdMZWdsSW1hZ2VPRVM7Ci0jZW5kaWYKLQotLyogR0xfT0VTX2ZpeGVkX3BvaW50ICovCi0jaWZuZGVmIEdMX09FU19maXhlZF9wb2ludAotI2RlZmluZSBHTF9GSVhFRF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTQwQwotI2VuZGlmCi0KLS8qIEdMX09FU19mcmFtZWJ1ZmZlcl9vYmplY3QgKi8KLSNpZm5kZWYgR0xfT0VTX2ZyYW1lYnVmZmVyX29iamVjdAotI2RlZmluZSBHTF9OT05FX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENDAKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENDEKLSNkZWZpbmUgR0xfUkdCQTRfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgwNTYKLSNkZWZpbmUgR0xfUkdCNV9BMV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgwNTcKLSNkZWZpbmUgR0xfUkdCNTY1X09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENjIKLSNkZWZpbmUgR0xfREVQVEhfQ09NUE9ORU5UMTZfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgxQTUKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX1dJRFRIX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENDIKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX0hFSUdIVF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENDMKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX0lOVEVSTkFMX0ZPUk1BVF9PRVMgICAgICAgICAgICAgICAgICAgICAweDhENDQKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX1JFRF9TSVpFX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENTAKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX0dSRUVOX1NJWkVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENTEKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX0JMVUVfU0laRV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENTIKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX0FMUEhBX1NJWkVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENTMKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX0RFUFRIX1NJWkVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENTQKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX1NURU5DSUxfU0laRV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAweDhENTUKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfQVRUQUNITUVOVF9PQkpFQ1RfVFlQRV9PRVMgICAgICAgICAgICAgICAweDhDRDAKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfQVRUQUNITUVOVF9PQkpFQ1RfTkFNRV9PRVMgICAgICAgICAgICAgICAweDhDRDEKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfQVRUQUNITUVOVF9URVhUVVJFX0xFVkVMX09FUyAgICAgICAgICAgICAweDhDRDIKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfQVRUQUNITUVOVF9URVhUVVJFX0NVQkVfTUFQX0ZBQ0VfT0VTICAgICAweDhDRDMKLSNkZWZpbmUgR0xfQ09MT1JfQVRUQUNITUVOVDBfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhDRTAKLSNkZWZpbmUgR0xfREVQVEhfQVRUQUNITUVOVF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhEMDAKLSNkZWZpbmUgR0xfU1RFTkNJTF9BVFRBQ0hNRU5UX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhEMjAKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfQ09NUExFVEVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhDRDUKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfSU5DT01QTEVURV9BVFRBQ0hNRU5UX09FUyAgICAgICAgICAgICAgICAweDhDRDYKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfSU5DT01QTEVURV9NSVNTSU5HX0FUVEFDSE1FTlRfT0VTICAgICAgICAweDhDRDcKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfSU5DT01QTEVURV9ESU1FTlNJT05TX09FUyAgICAgICAgICAgICAgICAweDhDRDkKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfSU5DT01QTEVURV9GT1JNQVRTX09FUyAgICAgICAgICAgICAgICAgICAweDhDREEKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfVU5TVVBQT1JURURfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAweDhDREQKLSNkZWZpbmUgR0xfRlJBTUVCVUZGRVJfQklORElOR19PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhDQTYKLSNkZWZpbmUgR0xfUkVOREVSQlVGRkVSX0JJTkRJTkdfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhDQTcKLSNkZWZpbmUgR0xfTUFYX1JFTkRFUkJVRkZFUl9TSVpFX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg0RTgKLSNkZWZpbmUgR0xfSU5WQUxJRF9GUkFNRUJVRkZFUl9PUEVSQVRJT05fT0VTICAgICAgICAgICAgICAgICAgICAweDA1MDYKLSNlbmRpZgotCi0vKiBHTF9PRVNfbWFwYnVmZmVyICovCi0jaWZuZGVmIEdMX09FU19tYXBidWZmZXIKLSNkZWZpbmUgR0xfV1JJVEVfT05MWV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4QjkKLSNkZWZpbmUgR0xfQlVGRkVSX0FDQ0VTU19PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4QkIKLSNkZWZpbmUgR0xfQlVGRkVSX01BUFBFRF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4QkMKLSNkZWZpbmUgR0xfQlVGRkVSX01BUF9QT0lOVEVSX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4QkQKLSNlbmRpZgotCi0vKiBHTF9PRVNfbWF0cml4X2dldCAqLwotI2lmbmRlZiBHTF9PRVNfbWF0cml4X2dldAotI2RlZmluZSBHTF9NT0RFTFZJRVdfTUFUUklYX0ZMT0FUX0FTX0lOVF9CSVRTX09FUyAgICAgICAgICAgICAgIDB4ODk4RAotI2RlZmluZSBHTF9QUk9KRUNUSU9OX01BVFJJWF9GTE9BVF9BU19JTlRfQklUU19PRVMgICAgICAgICAgICAgIDB4ODk4RQotI2RlZmluZSBHTF9URVhUVVJFX01BVFJJWF9GTE9BVF9BU19JTlRfQklUU19PRVMgICAgICAgICAgICAgICAgIDB4ODk4RgotI2VuZGlmCi0KLS8qIEdMX09FU19tYXRyaXhfcGFsZXR0ZSAqLwotI2lmbmRlZiBHTF9PRVNfbWF0cml4X3BhbGV0dGUKLSNkZWZpbmUgR0xfTUFYX1ZFUlRFWF9VTklUU19PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg2QTQKLSNkZWZpbmUgR0xfTUFYX1BBTEVUVEVfTUFUUklDRVNfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4NDIKLSNkZWZpbmUgR0xfTUFUUklYX1BBTEVUVEVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4NDAKLSNkZWZpbmUgR0xfTUFUUklYX0lOREVYX0FSUkFZX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4NDQKLSNkZWZpbmUgR0xfV0VJR0hUX0FSUkFZX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg2QUQKLSNkZWZpbmUgR0xfQ1VSUkVOVF9QQUxFVFRFX01BVFJJWF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4NDMKLSNkZWZpbmUgR0xfTUFUUklYX0lOREVYX0FSUkFZX1NJWkVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4NDYKLSNkZWZpbmUgR0xfTUFUUklYX0lOREVYX0FSUkFZX1RZUEVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAweDg4NDcKLSNkZWZpbmUgR0xfTUFUUklYX0lOREVYX0FSUkFZX1NUUklERV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAweDg4NDgKLSNkZWZpbmUgR0xfTUFUUklYX0lOREVYX0FSUkFZX1BPSU5URVJfT0VTICAgICAgICAgICAgICAgICAgICAgICAweDg4NDkKLSNkZWZpbmUgR0xfTUFUUklYX0lOREVYX0FSUkFZX0JVRkZFUl9CSU5ESU5HX09FUyAgICAgICAgICAgICAgICAweDhCOUUKLSNkZWZpbmUgR0xfV0VJR0hUX0FSUkFZX1NJWkVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg2QUIKLSNkZWZpbmUgR0xfV0VJR0hUX0FSUkFZX1RZUEVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg2QTkKLSNkZWZpbmUgR0xfV0VJR0hUX0FSUkFZX1NUUklERV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg2QUEKLSNkZWZpbmUgR0xfV0VJR0hUX0FSUkFZX1BPSU5URVJfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg2QUMKLSNkZWZpbmUgR0xfV0VJR0hUX0FSUkFZX0JVRkZFUl9CSU5ESU5HX09FUyAgICAgICAgICAgICAgICAgICAgICAweDg4OUUKLSNlbmRpZgotCi0vKiBHTF9PRVNfcGFja2VkX2RlcHRoX3N0ZW5jaWwgKi8KLSNpZm5kZWYgR0xfT0VTX3BhY2tlZF9kZXB0aF9zdGVuY2lsCi0jZGVmaW5lIEdMX0RFUFRIX1NURU5DSUxfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4NEY5Ci0jZGVmaW5lIEdMX1VOU0lHTkVEX0lOVF8yNF84X09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4NEZBCi0jZGVmaW5lIEdMX0RFUFRIMjRfU1RFTkNJTDhfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4OEYwCi0jZW5kaWYKLQotLyogR0xfT0VTX3JnYjhfcmdiYTggKi8KLSNpZm5kZWYgR0xfT0VTX3JnYjhfcmdiYTgKLSNkZWZpbmUgR0xfUkdCOF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgwNTEKLSNkZWZpbmUgR0xfUkdCQThfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgwNTgKLSNlbmRpZgotCi0vKiBHTF9PRVNfc3RlbmNpbDEgKi8KLSNpZm5kZWYgR0xfT0VTX3N0ZW5jaWwxCi0jZGVmaW5lIEdMX1NURU5DSUxfSU5ERVgxX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4RDQ2Ci0jZW5kaWYKLQotLyogR0xfT0VTX3N0ZW5jaWw0ICovCi0jaWZuZGVmIEdMX09FU19zdGVuY2lsNAotI2RlZmluZSBHTF9TVEVOQ0lMX0lOREVYNF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4OEQ0NwotI2VuZGlmCi0KLS8qIEdMX09FU19zdGVuY2lsOCAqLwotI2lmbmRlZiBHTF9PRVNfc3RlbmNpbDgKLSNkZWZpbmUgR0xfU1RFTkNJTF9JTkRFWDhfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhENDgKLSNlbmRpZgotCi0vKiBHTF9PRVNfc3RlbmNpbF93cmFwICovCi0jaWZuZGVmIEdMX09FU19zdGVuY2lsX3dyYXAKLSNkZWZpbmUgR0xfSU5DUl9XUkFQX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg1MDcKLSNkZWZpbmUgR0xfREVDUl9XUkFQX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg1MDgKLSNlbmRpZgotCi0vKiBHTF9PRVNfdGV4dHVyZV9jdWJlX21hcCAqLwotI2lmbmRlZiBHTF9PRVNfdGV4dHVyZV9jdWJlX21hcAotI2RlZmluZSBHTF9OT1JNQUxfTUFQX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODUxMQotI2RlZmluZSBHTF9SRUZMRUNUSU9OX01BUF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODUxMgotI2RlZmluZSBHTF9URVhUVVJFX0NVQkVfTUFQX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODUxMwotI2RlZmluZSBHTF9URVhUVVJFX0JJTkRJTkdfQ1VCRV9NQVBfT0VTICAgICAgICAgICAgICAgICAgICAgICAgIDB4ODUxNAotI2RlZmluZSBHTF9URVhUVVJFX0NVQkVfTUFQX1BPU0lUSVZFX1hfT0VTICAgICAgICAgICAgICAgICAgICAgIDB4ODUxNQotI2RlZmluZSBHTF9URVhUVVJFX0NVQkVfTUFQX05FR0FUSVZFX1hfT0VTICAgICAgICAgICAgICAgICAgICAgIDB4ODUxNgotI2RlZmluZSBHTF9URVhUVVJFX0NVQkVfTUFQX1BPU0lUSVZFX1lfT0VTICAgICAgICAgICAgICAgICAgICAgIDB4ODUxNwotI2RlZmluZSBHTF9URVhUVVJFX0NVQkVfTUFQX05FR0FUSVZFX1lfT0VTICAgICAgICAgICAgICAgICAgICAgIDB4ODUxOAotI2RlZmluZSBHTF9URVhUVVJFX0NVQkVfTUFQX1BPU0lUSVZFX1pfT0VTICAgICAgICAgICAgICAgICAgICAgIDB4ODUxOQotI2RlZmluZSBHTF9URVhUVVJFX0NVQkVfTUFQX05FR0FUSVZFX1pfT0VTICAgICAgICAgICAgICAgICAgICAgIDB4ODUxQQotI2RlZmluZSBHTF9NQVhfQ1VCRV9NQVBfVEVYVFVSRV9TSVpFX09FUyAgICAgICAgICAgICAgICAgICAgICAgIDB4ODUxQwotI2RlZmluZSBHTF9URVhUVVJFX0dFTl9NT0RFX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MjUwMAotI2RlZmluZSBHTF9URVhUVVJFX0dFTl9TVFJfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4OEQ2MAotI2VuZGlmCi0KLS8qIEdMX09FU190ZXh0dXJlX21pcnJvcmVkX3JlcGVhdCAqLwotI2lmbmRlZiBHTF9PRVNfdGV4dHVyZV9taXJyb3JlZF9yZXBlYXQKLSNkZWZpbmUgR0xfTUlSUk9SRURfUkVQRUFUX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDgzNzAKLSNlbmRpZgotCi0vKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSoKLSAqIEFNRCBleHRlbnNpb24gdG9rZW5zCi0gKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi0KLS8qIEdMX0FNRF9jb21wcmVzc2VkXzNEQ190ZXh0dXJlICovCi0jaWZuZGVmIEdMX0FNRF9jb21wcmVzc2VkXzNEQ190ZXh0dXJlCi0jZGVmaW5lIEdMXzNEQ19YX0FNRCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4N0Y5Ci0jZGVmaW5lIEdMXzNEQ19YWV9BTUQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHg4N0ZBCi0jZW5kaWYKLQotLyogR0xfQU1EX2NvbXByZXNzZWRfQVRDX3RleHR1cmUgKi8KLSNpZm5kZWYgR0xfQU1EX2NvbXByZXNzZWRfQVRDX3RleHR1cmUKLSNkZWZpbmUgR0xfQVRDX1JHQl9BTUQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDhDOTIKLSNkZWZpbmUgR0xfQVRDX1JHQkFfRVhQTElDSVRfQUxQSEFfQU1EICAgICAgICAgICAgICAgICAgICAgICAgICAweDhDOTMKLSNkZWZpbmUgR0xfQVRDX1JHQkFfSU5URVJQT0xBVEVEX0FMUEhBX0FNRCAgICAgICAgICAgICAgICAgICAgICAweDg3RUUKLSNlbmRpZgotCi0vKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSoKLSAqIEVYVCBleHRlbnNpb24gdG9rZW5zCi0gKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi0KLS8qIEdMX0VYVF90ZXh0dXJlX2ZpbHRlcl9hbmlzb3Ryb3BpYyAqLwotI2lmbmRlZiBHTF9FWFRfdGV4dHVyZV9maWx0ZXJfYW5pc290cm9waWMKLSNkZWZpbmUgR0xfVEVYVFVSRV9NQVhfQU5JU09UUk9QWV9FWFQgICAgICAgICAgICAgICAgICAgICAgICAgICAweDg0RkUKLSNkZWZpbmUgR0xfTUFYX1RFWFRVUkVfTUFYX0FOSVNPVFJPUFlfRVhUICAgICAgICAgICAgICAgICAgICAgICAweDg0RkYKLSNlbmRpZgotCi0vKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSoKLSAqIE9FUyBleHRlbnNpb24gZnVuY3Rpb25zCi0gKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi0KLS8qIEdMX09FU19ibGVuZF9lcXVhdGlvbl9zZXBhcmF0ZSAqLwotI2lmbmRlZiBHTF9PRVNfYmxlbmRfZXF1YXRpb25fc2VwYXJhdGUKLSNkZWZpbmUgR0xfT0VTX2JsZW5kX2VxdWF0aW9uX3NlcGFyYXRlIDEKLSNpZmRlZiBHTF9HTEVYVF9QUk9UT1RZUEVTCi1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEJsZW5kRXF1YXRpb25TZXBhcmF0ZU9FUyAoR0xlbnVtIG1vZGVSR0IsIEdMZW51bSBtb2RlQWxwaGEpOwotI2VuZGlmCi10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEJMRU5ERVFVQVRJT05TRVBBUkFURU9FU1BST0MpIChHTGVudW0gbW9kZVJHQiwgR0xlbnVtIG1vZGVBbHBoYSk7Ci0jZW5kaWYKLQotLyogR0xfT0VTX2JsZW5kX2Z1bmNfc2VwYXJhdGUgKi8KLSNpZm5kZWYgR0xfT0VTX2JsZW5kX2Z1bmNfc2VwYXJhdGUKLSNkZWZpbmUgR0xfT0VTX2JsZW5kX2Z1bmNfc2VwYXJhdGUgMQotI2lmZGVmIEdMX0dMRVhUX1BST1RPVFlQRVMKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQmxlbmRGdW5jU2VwYXJhdGVPRVMgKEdMZW51bSBzcmNSR0IsIEdMZW51bSBkc3RSR0IsIEdMZW51bSBzcmNBbHBoYSwgR0xlbnVtIGRzdEFscGhhKTsKLSNlbmRpZgotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xCTEVOREZVTkNTRVBBUkFURU9FU1BST0MpIChHTGVudW0gc3JjUkdCLCBHTGVudW0gZHN0UkdCLCBHTGVudW0gc3JjQWxwaGEsIEdMZW51bSBkc3RBbHBoYSk7Ci0jZW5kaWYKLQotLyogR0xfT0VTX2JsZW5kX3N1YnRyYWN0ICovCi0jaWZuZGVmIEdMX09FU19ibGVuZF9zdWJ0cmFjdAotI2RlZmluZSBHTF9PRVNfYmxlbmRfc3VidHJhY3QgMQotI2lmZGVmIEdMX0dMRVhUX1BST1RPVFlQRVMKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQmxlbmRFcXVhdGlvbk9FUyAoR0xlbnVtIG1vZGUpOwotI2VuZGlmCi10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEJMRU5ERVFVQVRJT05PRVNQUk9DKSAoR0xlbnVtIG1vZGUpOwotI2VuZGlmCi0KLS8qIEdMX09FU19ieXRlX2Nvb3JkaW5hdGVzICovCi0jaWZuZGVmIEdMX09FU19ieXRlX2Nvb3JkaW5hdGVzCi0jZGVmaW5lIEdMX09FU19ieXRlX2Nvb3JkaW5hdGVzIDEKLSNlbmRpZgotCi0vKiBHTF9PRVNfY29tcHJlc3NlZF9FVEMxX1JHQjhfdGV4dHVyZSAqLwotI2lmbmRlZiBHTF9PRVNfY29tcHJlc3NlZF9FVEMxX1JHQjhfdGV4dHVyZQotI2RlZmluZSBHTF9PRVNfY29tcHJlc3NlZF9FVEMxX1JHQjhfdGV4dHVyZSAxCi0jZW5kaWYKLQotLyogR0xfT0VTX2RlcHRoMjQgKi8KLSNpZm5kZWYgR0xfT0VTX2RlcHRoMjQKLSNkZWZpbmUgR0xfT0VTX2RlcHRoMjQgMQotI2VuZGlmCi0KLS8qIEdMX09FU19kZXB0aDMyICovCi0jaWZuZGVmIEdMX09FU19kZXB0aDMyCi0jZGVmaW5lIEdMX09FU19kZXB0aDMyIDEKLSNlbmRpZgotCi0vKiBHTF9PRVNfZHJhd190ZXh0dXJlICovCi0jaWZuZGVmIEdMX09FU19kcmF3X3RleHR1cmUKLSNkZWZpbmUgR0xfT0VTX2RyYXdfdGV4dHVyZSAxCi0jaWZkZWYgR0xfR0xFWFRfUFJPVE9UWVBFUwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEcmF3VGV4c09FUyAoR0xzaG9ydCB4LCBHTHNob3J0IHksIEdMc2hvcnQgeiwgR0xzaG9ydCB3aWR0aCwgR0xzaG9ydCBoZWlnaHQpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEcmF3VGV4aU9FUyAoR0xpbnQgeCwgR0xpbnQgeSwgR0xpbnQgeiwgR0xpbnQgd2lkdGgsIEdMaW50IGhlaWdodCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbERyYXdUZXh4T0VTIChHTGZpeGVkIHgsIEdMZml4ZWQgeSwgR0xmaXhlZCB6LCBHTGZpeGVkIHdpZHRoLCBHTGZpeGVkIGhlaWdodCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbERyYXdUZXhzdk9FUyAoY29uc3QgR0xzaG9ydCAqY29vcmRzKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRHJhd1RleGl2T0VTIChjb25zdCBHTGludCAqY29vcmRzKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsRHJhd1RleHh2T0VTIChjb25zdCBHTGZpeGVkICpjb29yZHMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEcmF3VGV4Zk9FUyAoR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeiwgR0xmbG9hdCB3aWR0aCwgR0xmbG9hdCBoZWlnaHQpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEcmF3VGV4ZnZPRVMgKGNvbnN0IEdMZmxvYXQgKmNvb3Jkcyk7Ci0jZW5kaWYKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMRFJBV1RFWFNPRVNQUk9DKSAoR0xzaG9ydCB4LCBHTHNob3J0IHksIEdMc2hvcnQgeiwgR0xzaG9ydCB3aWR0aCwgR0xzaG9ydCBoZWlnaHQpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xEUkFXVEVYSU9FU1BST0MpIChHTGludCB4LCBHTGludCB5LCBHTGludCB6LCBHTGludCB3aWR0aCwgR0xpbnQgaGVpZ2h0KTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMRFJBV1RFWFhPRVNQUk9DKSAoR0xmaXhlZCB4LCBHTGZpeGVkIHksIEdMZml4ZWQgeiwgR0xmaXhlZCB3aWR0aCwgR0xmaXhlZCBoZWlnaHQpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xEUkFXVEVYU1ZPRVNQUk9DKSAoY29uc3QgR0xzaG9ydCAqY29vcmRzKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMRFJBV1RFWElWT0VTUFJPQykgKGNvbnN0IEdMaW50ICpjb29yZHMpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xEUkFXVEVYWFZPRVNQUk9DKSAoY29uc3QgR0xmaXhlZCAqY29vcmRzKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMRFJBV1RFWEZPRVNQUk9DKSAoR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeiwgR0xmbG9hdCB3aWR0aCwgR0xmbG9hdCBoZWlnaHQpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xEUkFXVEVYRlZPRVNQUk9DKSAoY29uc3QgR0xmbG9hdCAqY29vcmRzKTsKLSNlbmRpZgotCi0vKiBHTF9PRVNfRUdMX2ltYWdlICovCi0jaWZuZGVmIEdMX09FU19FR0xfaW1hZ2UKLSNkZWZpbmUgR0xfT0VTX0VHTF9pbWFnZSAxCi0jaWZkZWYgR0xfR0xFWFRfUFJPVE9UWVBFUwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xFR0xJbWFnZVRhcmdldFRleHR1cmUyRE9FUyAoR0xlbnVtIHRhcmdldCwgR0xlZ2xJbWFnZU9FUyBpbWFnZSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEVHTEltYWdlVGFyZ2V0UmVuZGVyYnVmZmVyU3RvcmFnZU9FUyAoR0xlbnVtIHRhcmdldCwgR0xlZ2xJbWFnZU9FUyBpbWFnZSk7Ci0jZW5kaWYKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMRUdMSU1BR0VUQVJHRVRURVhUVVJFMkRPRVNQUk9DKSAoR0xlbnVtIHRhcmdldCwgR0xlZ2xJbWFnZU9FUyBpbWFnZSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEVHTElNQUdFVEFSR0VUUkVOREVSQlVGRkVSU1RPUkFHRU9FU1BST0MpIChHTGVudW0gdGFyZ2V0LCBHTGVnbEltYWdlT0VTIGltYWdlKTsKLSNlbmRpZgotCi0vKiBHTF9PRVNfZWxlbWVudF9pbmRleF91aW50ICovCi0jaWZuZGVmIEdMX09FU19lbGVtZW50X2luZGV4X3VpbnQKLSNkZWZpbmUgR0xfT0VTX2VsZW1lbnRfaW5kZXhfdWludCAxCi0jZW5kaWYKLQotLyogR0xfT0VTX2V4dGVuZGVkX21hdHJpeF9wYWxldHRlICovCi0jaWZuZGVmIEdMX09FU19leHRlbmRlZF9tYXRyaXhfcGFsZXR0ZQotI2RlZmluZSBHTF9PRVNfZXh0ZW5kZWRfbWF0cml4X3BhbGV0dGUgMQotI2VuZGlmCi0KLS8qIEdMX09FU19mYm9fcmVuZGVyX21pcG1hcCAqLwotI2lmbmRlZiBHTF9PRVNfZmJvX3JlbmRlcl9taXBtYXAKLSNkZWZpbmUgR0xfT0VTX2Zib19yZW5kZXJfbWlwbWFwIDEKLSNlbmRpZgotCi0vKiBHTF9PRVNfZml4ZWRfcG9pbnQgKi8KLSNpZm5kZWYgR0xfT0VTX2ZpeGVkX3BvaW50Ci0jZGVmaW5lIEdMX09FU19maXhlZF9wb2ludCAxCi0jaWZkZWYgR0xfR0xFWFRfUFJPVE9UWVBFUwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xBbHBoYUZ1bmN4T0VTIChHTGVudW0gZnVuYywgR0xjbGFtcHggcmVmKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQ2xlYXJDb2xvcnhPRVMgKEdMY2xhbXB4IHJlZCwgR0xjbGFtcHggZ3JlZW4sIEdMY2xhbXB4IGJsdWUsIEdMY2xhbXB4IGFscGhhKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQ2xlYXJEZXB0aHhPRVMgKEdMY2xhbXB4IGRlcHRoKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQ2xpcFBsYW5leE9FUyAoR0xlbnVtIHBsYW5lLCBjb25zdCBHTGZpeGVkICplcXVhdGlvbik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENvbG9yNHhPRVMgKEdMZml4ZWQgcmVkLCBHTGZpeGVkIGdyZWVuLCBHTGZpeGVkIGJsdWUsIEdMZml4ZWQgYWxwaGEpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEZXB0aFJhbmdleE9FUyAoR0xjbGFtcHggek5lYXIsIEdMY2xhbXB4IHpGYXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xGb2d4T0VTIChHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xGb2d4dk9FUyAoR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xGcnVzdHVteE9FUyAoR0xmaXhlZCBsZWZ0LCBHTGZpeGVkIHJpZ2h0LCBHTGZpeGVkIGJvdHRvbSwgR0xmaXhlZCB0b3AsIEdMZml4ZWQgek5lYXIsIEdMZml4ZWQgekZhcik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldENsaXBQbGFuZXhPRVMgKEdMZW51bSBwbmFtZSwgR0xmaXhlZCBlcW5bNF0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRGaXhlZHZPRVMgKEdMZW51bSBwbmFtZSwgR0xmaXhlZCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsR2V0TGlnaHR4dk9FUyAoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldE1hdGVyaWFseHZPRVMgKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldFRleEVudnh2T0VTIChHTGVudW0gZW52LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldFRleFBhcmFtZXRlcnh2T0VTIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbExpZ2h0TW9kZWx4T0VTIChHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xMaWdodE1vZGVseHZPRVMgKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTGlnaHR4T0VTIChHTGVudW0gbGlnaHQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbExpZ2h0eHZPRVMgKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xMaW5lV2lkdGh4T0VTIChHTGZpeGVkIHdpZHRoKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTG9hZE1hdHJpeHhPRVMgKGNvbnN0IEdMZml4ZWQgKm0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xNYXRlcmlhbHhPRVMgKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xNYXRlcmlhbHh2T0VTIChHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xNdWx0TWF0cml4eE9FUyAoY29uc3QgR0xmaXhlZCAqbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbE11bHRpVGV4Q29vcmQ0eE9FUyAoR0xlbnVtIHRhcmdldCwgR0xmaXhlZCBzLCBHTGZpeGVkIHQsIEdMZml4ZWQgciwgR0xmaXhlZCBxKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsTm9ybWFsM3hPRVMgKEdMZml4ZWQgbngsIEdMZml4ZWQgbnksIEdMZml4ZWQgbnopOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xPcnRob3hPRVMgKEdMZml4ZWQgbGVmdCwgR0xmaXhlZCByaWdodCwgR0xmaXhlZCBib3R0b20sIEdMZml4ZWQgdG9wLCBHTGZpeGVkIHpOZWFyLCBHTGZpeGVkIHpGYXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xQb2ludFBhcmFtZXRlcnhPRVMgKEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFBvaW50UGFyYW1ldGVyeHZPRVMgKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsUG9pbnRTaXpleE9FUyAoR0xmaXhlZCBzaXplKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsUG9seWdvbk9mZnNldHhPRVMgKEdMZml4ZWQgZmFjdG9yLCBHTGZpeGVkIHVuaXRzKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsUm90YXRleE9FUyAoR0xmaXhlZCBhbmdsZSwgR0xmaXhlZCB4LCBHTGZpeGVkIHksIEdMZml4ZWQgeik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFNhbXBsZUNvdmVyYWdleE9FUyAoR0xjbGFtcHggdmFsdWUsIEdMYm9vbGVhbiBpbnZlcnQpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xTY2FsZXhPRVMgKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhFbnZ4T0VTIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhFbnZ4dk9FUyAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhQYXJhbWV0ZXJ4T0VTIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhQYXJhbWV0ZXJ4dk9FUyAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUcmFuc2xhdGV4T0VTIChHTGZpeGVkIHgsIEdMZml4ZWQgeSwgR0xmaXhlZCB6KTsKLSNlbmRpZgotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xBTFBIQUZVTkNYT0VTUFJPQykgKEdMZW51bSBmdW5jLCBHTGNsYW1weCByZWYpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xDTEVBUkNPTE9SWE9FU1BST0MpIChHTGNsYW1weCByZWQsIEdMY2xhbXB4IGdyZWVuLCBHTGNsYW1weCBibHVlLCBHTGNsYW1weCBhbHBoYSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTENMRUFSREVQVEhYT0VTUFJPQykgKEdMY2xhbXB4IGRlcHRoKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMQ0xJUFBMQU5FWE9FU1BST0MpIChHTGVudW0gcGxhbmUsIGNvbnN0IEdMZml4ZWQgKmVxdWF0aW9uKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMQ09MT1I0WE9FU1BST0MpIChHTGZpeGVkIHJlZCwgR0xmaXhlZCBncmVlbiwgR0xmaXhlZCBibHVlLCBHTGZpeGVkIGFscGhhKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMREVQVEhSQU5HRVhPRVNQUk9DKSAoR0xjbGFtcHggek5lYXIsIEdMY2xhbXB4IHpGYXIpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xGT0dYT0VTUFJPQykgKEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEZPR1hWT0VTUFJPQykgKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMRlJVU1RVTVhPRVNQUk9DKSAoR0xmaXhlZCBsZWZ0LCBHTGZpeGVkIHJpZ2h0LCBHTGZpeGVkIGJvdHRvbSwgR0xmaXhlZCB0b3AsIEdMZml4ZWQgek5lYXIsIEdMZml4ZWQgekZhcik7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEdFVENMSVBQTEFORVhPRVNQUk9DKSAoR0xlbnVtIHBuYW1lLCBHTGZpeGVkIGVxbls0XSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEdFVEZJWEVEVk9FU1BST0MpIChHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEdFVExJR0hUWFZPRVNQUk9DKSAoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEdFVE1BVEVSSUFMWFZPRVNQUk9DKSAoR0xlbnVtIGZhY2UsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCAqcGFyYW1zKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMR0VUVEVYRU5WWFZPRVNQUk9DKSAoR0xlbnVtIGVudiwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkICpwYXJhbXMpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xHRVRURVhQQVJBTUVURVJYVk9FU1BST0MpIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTExJR0hUTU9ERUxYT0VTUFJPQykgKEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTExJR0hUTU9ERUxYVk9FU1BST0MpIChHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTExJR0hUWE9FU1BST0MpIChHTGVudW0gbGlnaHQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTExJR0hUWFZPRVNQUk9DKSAoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTExJTkVXSURUSFhPRVNQUk9DKSAoR0xmaXhlZCB3aWR0aCk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTExPQURNQVRSSVhYT0VTUFJPQykgKGNvbnN0IEdMZml4ZWQgKm0pOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xNQVRFUklBTFhPRVNQUk9DKSAoR0xlbnVtIGZhY2UsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTE1BVEVSSUFMWFZPRVNQUk9DKSAoR0xlbnVtIGZhY2UsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMTVVMVE1BVFJJWFhPRVNQUk9DKSAoY29uc3QgR0xmaXhlZCAqbSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTE1VTFRJVEVYQ09PUkQ0WE9FU1BST0MpIChHTGVudW0gdGFyZ2V0LCBHTGZpeGVkIHMsIEdMZml4ZWQgdCwgR0xmaXhlZCByLCBHTGZpeGVkIHEpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xOT1JNQUwzWE9FU1BST0MpIChHTGZpeGVkIG54LCBHTGZpeGVkIG55LCBHTGZpeGVkIG56KTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMT1JUSE9YT0VTUFJPQykgKEdMZml4ZWQgbGVmdCwgR0xmaXhlZCByaWdodCwgR0xmaXhlZCBib3R0b20sIEdMZml4ZWQgdG9wLCBHTGZpeGVkIHpOZWFyLCBHTGZpeGVkIHpGYXIpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xQT0lOVFBBUkFNRVRFUlhPRVNQUk9DKSAoR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMUE9JTlRQQVJBTUVURVJYVk9FU1BST0MpIChHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTFBPSU5UU0laRVhPRVNQUk9DKSAoR0xmaXhlZCBzaXplKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMUE9MWUdPTk9GRlNFVFhPRVNQUk9DKSAoR0xmaXhlZCBmYWN0b3IsIEdMZml4ZWQgdW5pdHMpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xST1RBVEVYT0VTUFJPQykgKEdMZml4ZWQgYW5nbGUsIEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xTQU1QTEVDT1ZFUkFHRVhPRVNQUk9DKSAoR0xjbGFtcHggdmFsdWUsIEdMYm9vbGVhbiBpbnZlcnQpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xTQ0FMRVhPRVNQUk9DKSAoR0xmaXhlZCB4LCBHTGZpeGVkIHksIEdMZml4ZWQgeik7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTFRFWEVOVlhPRVNQUk9DKSAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMVEVYRU5WWFZPRVNQUk9DKSAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xURVhQQVJBTUVURVJYT0VTUFJPQykgKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTFRFWFBBUkFNRVRFUlhWT0VTUFJPQykgKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMVFJBTlNMQVRFWE9FU1BST0MpIChHTGZpeGVkIHgsIEdMZml4ZWQgeSwgR0xmaXhlZCB6KTsKLSNlbmRpZgotCi0vKiBHTF9PRVNfZnJhbWVidWZmZXJfb2JqZWN0ICovCi0jaWZuZGVmIEdMX09FU19mcmFtZWJ1ZmZlcl9vYmplY3QKLSNkZWZpbmUgR0xfT0VTX2ZyYW1lYnVmZmVyX29iamVjdCAxCi0jaWZkZWYgR0xfR0xFWFRfUFJPVE9UWVBFUwotR0xfQVBJIEdMYm9vbGVhbiBHTF9BUElFTlRSWSBnbElzUmVuZGVyYnVmZmVyT0VTIChHTHVpbnQgcmVuZGVyYnVmZmVyKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQmluZFJlbmRlcmJ1ZmZlck9FUyAoR0xlbnVtIHRhcmdldCwgR0x1aW50IHJlbmRlcmJ1ZmZlcik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbERlbGV0ZVJlbmRlcmJ1ZmZlcnNPRVMgKEdMc2l6ZWkgbiwgY29uc3QgR0x1aW50KiByZW5kZXJidWZmZXJzKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsR2VuUmVuZGVyYnVmZmVyc09FUyAoR0xzaXplaSBuLCBHTHVpbnQqIHJlbmRlcmJ1ZmZlcnMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xSZW5kZXJidWZmZXJTdG9yYWdlT0VTIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gaW50ZXJuYWxmb3JtYXQsIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0KTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsR2V0UmVuZGVyYnVmZmVyUGFyYW1ldGVyaXZPRVMgKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xpbnQqIHBhcmFtcyk7Ci1HTF9BUEkgR0xib29sZWFuIEdMX0FQSUVOVFJZIGdsSXNGcmFtZWJ1ZmZlck9FUyAoR0x1aW50IGZyYW1lYnVmZmVyKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQmluZEZyYW1lYnVmZmVyT0VTIChHTGVudW0gdGFyZ2V0LCBHTHVpbnQgZnJhbWVidWZmZXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEZWxldGVGcmFtZWJ1ZmZlcnNPRVMgKEdMc2l6ZWkgbiwgY29uc3QgR0x1aW50KiBmcmFtZWJ1ZmZlcnMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZW5GcmFtZWJ1ZmZlcnNPRVMgKEdMc2l6ZWkgbiwgR0x1aW50KiBmcmFtZWJ1ZmZlcnMpOwotR0xfQVBJIEdMZW51bSBHTF9BUElFTlRSWSBnbENoZWNrRnJhbWVidWZmZXJTdGF0dXNPRVMgKEdMZW51bSB0YXJnZXQpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xGcmFtZWJ1ZmZlclJlbmRlcmJ1ZmZlck9FUyAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIGF0dGFjaG1lbnQsIEdMZW51bSByZW5kZXJidWZmZXJ0YXJnZXQsIEdMdWludCByZW5kZXJidWZmZXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xGcmFtZWJ1ZmZlclRleHR1cmUyRE9FUyAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIGF0dGFjaG1lbnQsIEdMZW51bSB0ZXh0YXJnZXQsIEdMdWludCB0ZXh0dXJlLCBHTGludCBsZXZlbCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldEZyYW1lYnVmZmVyQXR0YWNobWVudFBhcmFtZXRlcml2T0VTIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gYXR0YWNobWVudCwgR0xlbnVtIHBuYW1lLCBHTGludCogcGFyYW1zKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsR2VuZXJhdGVNaXBtYXBPRVMgKEdMZW51bSB0YXJnZXQpOwotI2VuZGlmCi10eXBlZGVmIEdMYm9vbGVhbiAoR0xfQVBJRU5UUllQIFBGTkdMSVNSRU5ERVJCVUZGRVJPRVNQUk9DKSAoR0x1aW50IHJlbmRlcmJ1ZmZlcik7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEJJTkRSRU5ERVJCVUZGRVJPRVNQUk9DKSAoR0xlbnVtIHRhcmdldCwgR0x1aW50IHJlbmRlcmJ1ZmZlcik7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTERFTEVURVJFTkRFUkJVRkZFUlNPRVNQUk9DKSAoR0xzaXplaSBuLCBjb25zdCBHTHVpbnQqIHJlbmRlcmJ1ZmZlcnMpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xHRU5SRU5ERVJCVUZGRVJTT0VTUFJPQykgKEdMc2l6ZWkgbiwgR0x1aW50KiByZW5kZXJidWZmZXJzKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMUkVOREVSQlVGRkVSU1RPUkFHRU9FU1BST0MpIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gaW50ZXJuYWxmb3JtYXQsIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0KTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMR0VUUkVOREVSQlVGRkVSUEFSQU1FVEVSSVZPRVNQUk9DKSAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGludCogcGFyYW1zKTsKLXR5cGVkZWYgR0xib29sZWFuIChHTF9BUElFTlRSWVAgUEZOR0xJU0ZSQU1FQlVGRkVST0VTUFJPQykgKEdMdWludCBmcmFtZWJ1ZmZlcik7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEJJTkRGUkFNRUJVRkZFUk9FU1BST0MpIChHTGVudW0gdGFyZ2V0LCBHTHVpbnQgZnJhbWVidWZmZXIpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xERUxFVEVGUkFNRUJVRkZFUlNPRVNQUk9DKSAoR0xzaXplaSBuLCBjb25zdCBHTHVpbnQqIGZyYW1lYnVmZmVycyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEdFTkZSQU1FQlVGRkVSU09FU1BST0MpIChHTHNpemVpIG4sIEdMdWludCogZnJhbWVidWZmZXJzKTsKLXR5cGVkZWYgR0xlbnVtIChHTF9BUElFTlRSWVAgUEZOR0xDSEVDS0ZSQU1FQlVGRkVSU1RBVFVTT0VTUFJPQykgKEdMZW51bSB0YXJnZXQpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xGUkFNRUJVRkZFUlJFTkRFUkJVRkZFUk9FU1BST0MpIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gYXR0YWNobWVudCwgR0xlbnVtIHJlbmRlcmJ1ZmZlcnRhcmdldCwgR0x1aW50IHJlbmRlcmJ1ZmZlcik7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEZSQU1FQlVGRkVSVEVYVFVSRTJET0VTUFJPQykgKEdMZW51bSB0YXJnZXQsIEdMZW51bSBhdHRhY2htZW50LCBHTGVudW0gdGV4dGFyZ2V0LCBHTHVpbnQgdGV4dHVyZSwgR0xpbnQgbGV2ZWwpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xHRVRGUkFNRUJVRkZFUkFUVEFDSE1FTlRQQVJBTUVURVJJVk9FU1BST0MpIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gYXR0YWNobWVudCwgR0xlbnVtIHBuYW1lLCBHTGludCogcGFyYW1zKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMR0VORVJBVEVNSVBNQVBPRVNQUk9DKSAoR0xlbnVtIHRhcmdldCk7Ci0jZW5kaWYKLQotLyogR0xfT0VTX21hcGJ1ZmZlciAqLwotI2lmbmRlZiBHTF9PRVNfbWFwYnVmZmVyCi0jZGVmaW5lIEdMX09FU19tYXBidWZmZXIgMQotI2lmZGVmIEdMX0dMRVhUX1BST1RPVFlQRVMKLUdMX0FQSSB2b2lkKiBHTF9BUElFTlRSWSBnbE1hcEJ1ZmZlck9FUyAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIGFjY2Vzcyk7Ci1HTF9BUEkgR0xib29sZWFuIEdMX0FQSUVOVFJZIGdsVW5tYXBCdWZmZXJPRVMgKEdMZW51bSB0YXJnZXQpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRCdWZmZXJQb2ludGVydk9FUyAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCB2b2lkKiogcGFyYW1zKTsKLSNlbmRpZgotdHlwZWRlZiB2b2lkKiAoR0xfQVBJRU5UUllQIFBGTkdMTUFQQlVGRkVST0VTUFJPQykgKEdMZW51bSB0YXJnZXQsIEdMZW51bSBhY2Nlc3MpOwotdHlwZWRlZiBHTGJvb2xlYW4gKEdMX0FQSUVOVFJZUCBQRk5HTFVOTUFQQlVGRkVST0VTUFJPQykgKEdMZW51bSB0YXJnZXQpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xHRVRCVUZGRVJQT0lOVEVSVk9FU1BST0MpIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIHZvaWQqKiBwYXJhbXMpOwotI2VuZGlmCi0KLS8qIEdMX09FU19tYXRyaXhfZ2V0ICovCi0jaWZuZGVmIEdMX09FU19tYXRyaXhfZ2V0Ci0jZGVmaW5lIEdMX09FU19tYXRyaXhfZ2V0IDEKLSNlbmRpZgotCi0vKiBHTF9PRVNfbWF0cml4X3BhbGV0dGUgKi8KLSNpZm5kZWYgR0xfT0VTX21hdHJpeF9wYWxldHRlCi0jZGVmaW5lIEdMX09FU19tYXRyaXhfcGFsZXR0ZSAxCi0jaWZkZWYgR0xfR0xFWFRfUFJPVE9UWVBFUwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xDdXJyZW50UGFsZXR0ZU1hdHJpeE9FUyAoR0x1aW50IG1hdHJpeHBhbGV0dGVpbmRleCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbExvYWRQYWxldHRlRnJvbU1vZGVsVmlld01hdHJpeE9FUyAodm9pZCk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbE1hdHJpeEluZGV4UG9pbnRlck9FUyAoR0xpbnQgc2l6ZSwgR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xXZWlnaHRQb2ludGVyT0VTIChHTGludCBzaXplLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlcik7Ci0jZW5kaWYKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMQ1VSUkVOVFBBTEVUVEVNQVRSSVhPRVNQUk9DKSAoR0x1aW50IG1hdHJpeHBhbGV0dGVpbmRleCk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTExPQURQQUxFVFRFRlJPTU1PREVMVklFV01BVFJJWE9FU1BST0MpICh2b2lkKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMTUFUUklYSU5ERVhQT0lOVEVST0VTUFJPQykgKEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMV0VJR0hUUE9JTlRFUk9FU1BST0MpIChHTGludCBzaXplLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlcik7Ci0jZW5kaWYKLQotLyogR0xfT0VTX3BhY2tlZF9kZXB0aF9zdGVuY2lsICovCi0jaWZuZGVmIEdMX09FU19wYWNrZWRfZGVwdGhfc3RlbmNpbAotI2RlZmluZSBHTF9PRVNfcGFja2VkX2RlcHRoX3N0ZW5jaWwgMQotI2VuZGlmCi0KLS8qIEdMX09FU19xdWVyeV9tYXRyaXggKi8KLSNpZm5kZWYgR0xfT0VTX3F1ZXJ5X21hdHJpeAotI2RlZmluZSBHTF9PRVNfcXVlcnlfbWF0cml4IDEKLSNpZmRlZiBHTF9HTEVYVF9QUk9UT1RZUEVTCi1HTF9BUEkgR0xiaXRmaWVsZCBHTF9BUElFTlRSWSBnbFF1ZXJ5TWF0cml4eE9FUyAoR0xmaXhlZCBtYW50aXNzYVsxNl0sIEdMaW50IGV4cG9uZW50WzE2XSk7Ci0jZW5kaWYKLXR5cGVkZWYgR0xiaXRmaWVsZCAoR0xfQVBJRU5UUllQIFBGTkdMUVVFUllNQVRSSVhYT0VTUFJPQykgKEdMZml4ZWQgbWFudGlzc2FbMTZdLCBHTGludCBleHBvbmVudFsxNl0pOwotI2VuZGlmCi0KLS8qIEdMX09FU19yZ2I4X3JnYmE4ICovCi0jaWZuZGVmIEdMX09FU19yZ2I4X3JnYmE4Ci0jZGVmaW5lIEdMX09FU19yZ2I4X3JnYmE4IDEKLSNlbmRpZgotCi0vKiBHTF9PRVNfc2luZ2xlX3ByZWNpc2lvbiAqLwotI2lmbmRlZiBHTF9PRVNfc2luZ2xlX3ByZWNpc2lvbgotI2RlZmluZSBHTF9PRVNfc2luZ2xlX3ByZWNpc2lvbiAxCi0jaWZkZWYgR0xfR0xFWFRfUFJPVE9UWVBFUwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xEZXB0aFJhbmdlZk9FUyAoR0xjbGFtcGYgek5lYXIsIEdMY2xhbXBmIHpGYXIpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xGcnVzdHVtZk9FUyAoR0xmbG9hdCBsZWZ0LCBHTGZsb2F0IHJpZ2h0LCBHTGZsb2F0IGJvdHRvbSwgR0xmbG9hdCB0b3AsIEdMZmxvYXQgek5lYXIsIEdMZmxvYXQgekZhcik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbE9ydGhvZk9FUyAoR0xmbG9hdCBsZWZ0LCBHTGZsb2F0IHJpZ2h0LCBHTGZsb2F0IGJvdHRvbSwgR0xmbG9hdCB0b3AsIEdMZmxvYXQgek5lYXIsIEdMZmxvYXQgekZhcik7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbENsaXBQbGFuZWZPRVMgKEdMZW51bSBwbGFuZSwgY29uc3QgR0xmbG9hdCAqZXF1YXRpb24pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRDbGlwUGxhbmVmT0VTIChHTGVudW0gcG5hbWUsIEdMZmxvYXQgZXFuWzRdKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsQ2xlYXJEZXB0aGZPRVMgKEdMY2xhbXBmIGRlcHRoKTsKLSNlbmRpZgotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xERVBUSFJBTkdFRk9FU1BST0MpIChHTGNsYW1wZiB6TmVhciwgR0xjbGFtcGYgekZhcik7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEZSVVNUVU1GT0VTUFJPQykgKEdMZmxvYXQgbGVmdCwgR0xmbG9hdCByaWdodCwgR0xmbG9hdCBib3R0b20sIEdMZmxvYXQgdG9wLCBHTGZsb2F0IHpOZWFyLCBHTGZsb2F0IHpGYXIpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xPUlRIT0ZPRVNQUk9DKSAoR0xmbG9hdCBsZWZ0LCBHTGZsb2F0IHJpZ2h0LCBHTGZsb2F0IGJvdHRvbSwgR0xmbG9hdCB0b3AsIEdMZmxvYXQgek5lYXIsIEdMZmxvYXQgekZhcik7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTENMSVBQTEFORUZPRVNQUk9DKSAoR0xlbnVtIHBsYW5lLCBjb25zdCBHTGZsb2F0ICplcXVhdGlvbik7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEdFVENMSVBQTEFORUZPRVNQUk9DKSAoR0xlbnVtIHBuYW1lLCBHTGZsb2F0IGVxbls0XSk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTENMRUFSREVQVEhGT0VTUFJPQykgKEdMY2xhbXBmIGRlcHRoKTsKLSNlbmRpZgotCi0vKiBHTF9PRVNfc3RlbmNpbDEgKi8KLSNpZm5kZWYgR0xfT0VTX3N0ZW5jaWwxCi0jZGVmaW5lIEdMX09FU19zdGVuY2lsMSAxCi0jZW5kaWYKLQotLyogR0xfT0VTX3N0ZW5jaWw0ICovCi0jaWZuZGVmIEdMX09FU19zdGVuY2lsNAotI2RlZmluZSBHTF9PRVNfc3RlbmNpbDQgMQotI2VuZGlmCi0KLS8qIEdMX09FU19zdGVuY2lsOCAqLwotI2lmbmRlZiBHTF9PRVNfc3RlbmNpbDgKLSNkZWZpbmUgR0xfT0VTX3N0ZW5jaWw4IDEKLSNlbmRpZgotCi0vKiBHTF9PRVNfc3RlbmNpbF93cmFwICovCi0jaWZuZGVmIEdMX09FU19zdGVuY2lsX3dyYXAKLSNkZWZpbmUgR0xfT0VTX3N0ZW5jaWxfd3JhcCAxCi0jZW5kaWYKLQotLyogR0xfT0VTX3RleHR1cmVfY3ViZV9tYXAgKi8KLSNpZm5kZWYgR0xfT0VTX3RleHR1cmVfY3ViZV9tYXAKLSNkZWZpbmUgR0xfT0VTX3RleHR1cmVfY3ViZV9tYXAgMQotI2lmZGVmIEdMX0dMRVhUX1BST1RPVFlQRVMKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsVGV4R2VuZk9FUyAoR0xlbnVtIGNvb3JkLCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xUZXhHZW5mdk9FUyAoR0xlbnVtIGNvb3JkLCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFRleEdlbmlPRVMgKEdMZW51bSBjb29yZCwgR0xlbnVtIHBuYW1lLCBHTGludCBwYXJhbSk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFRleEdlbml2T0VTIChHTGVudW0gY29vcmQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xpbnQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbFRleEdlbnhPRVMgKEdMZW51bSBjb29yZCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLUdMX0FQSSB2b2lkIEdMX0FQSUVOVFJZIGdsVGV4R2VueHZPRVMgKEdMZW51bSBjb29yZCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xfQVBJIHZvaWQgR0xfQVBJRU5UUlkgZ2xHZXRUZXhHZW5mdk9FUyAoR0xlbnVtIGNvb3JkLCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldFRleEdlbml2T0VTIChHTGVudW0gY29vcmQsIEdMZW51bSBwbmFtZSwgR0xpbnQgKnBhcmFtcyk7Ci1HTF9BUEkgdm9pZCBHTF9BUElFTlRSWSBnbEdldFRleEdlbnh2T0VTIChHTGVudW0gY29vcmQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCAqcGFyYW1zKTsKLSNlbmRpZgotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xURVhHRU5GT0VTUFJPQykgKEdMZW51bSBjb29yZCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMVEVYR0VORlZPRVNQUk9DKSAoR0xlbnVtIGNvb3JkLCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTFRFWEdFTklPRVNQUk9DKSAoR0xlbnVtIGNvb3JkLCBHTGVudW0gcG5hbWUsIEdMaW50IHBhcmFtKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMVEVYR0VOSVZPRVNQUk9DKSAoR0xlbnVtIGNvb3JkLCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMaW50ICpwYXJhbXMpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xURVhHRU5YT0VTUFJPQykgKEdMZW51bSBjb29yZCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLXR5cGVkZWYgdm9pZCAoR0xfQVBJRU5UUllQIFBGTkdMVEVYR0VOWFZPRVNQUk9DKSAoR0xlbnVtIGNvb3JkLCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEdFVFRFWEdFTkZWT0VTUFJPQykgKEdMZW51bSBjb29yZCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMpOwotdHlwZWRlZiB2b2lkIChHTF9BUElFTlRSWVAgUEZOR0xHRVRURVhHRU5JVk9FU1BST0MpIChHTGVudW0gY29vcmQsIEdMZW51bSBwbmFtZSwgR0xpbnQgKnBhcmFtcyk7Ci10eXBlZGVmIHZvaWQgKEdMX0FQSUVOVFJZUCBQRk5HTEdFVFRFWEdFTlhWT0VTUFJPQykgKEdMZW51bSBjb29yZCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkICpwYXJhbXMpOwotI2VuZGlmCi0KLS8qIEdMX09FU190ZXh0dXJlX2Vudl9jcm9zc2JhciAqLwotI2lmbmRlZiBHTF9PRVNfdGV4dHVyZV9lbnZfY3Jvc3NiYXIKLSNkZWZpbmUgR0xfT0VTX3RleHR1cmVfZW52X2Nyb3NzYmFyIDEKLSNlbmRpZgotCi0vKiBHTF9PRVNfdGV4dHVyZV9taXJyb3JlZF9yZXBlYXQgKi8KLSNpZm5kZWYgR0xfT0VTX3RleHR1cmVfbWlycm9yZWRfcmVwZWF0Ci0jZGVmaW5lIEdMX09FU190ZXh0dXJlX21pcnJvcmVkX3JlcGVhdCAxCi0jZW5kaWYKLQotLyotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qCi0gKiBBTUQgZXh0ZW5zaW9uIGZ1bmN0aW9ucwotICotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwotCi0vKiBHTF9BTURfY29tcHJlc3NlZF8zRENfdGV4dHVyZSAqLwotI2lmbmRlZiBHTF9BTURfY29tcHJlc3NlZF8zRENfdGV4dHVyZQotI2RlZmluZSBHTF9BTURfY29tcHJlc3NlZF8zRENfdGV4dHVyZSAxCi0jZW5kaWYKLQotLyogR0xfQU1EX2NvbXByZXNzZWRfQVRDX3RleHR1cmUgKi8KLSNpZm5kZWYgR0xfQU1EX2NvbXByZXNzZWRfQVRDX3RleHR1cmUKLSNkZWZpbmUgR0xfQU1EX2NvbXByZXNzZWRfQVRDX3RleHR1cmUgMQotI2VuZGlmCi0KLS8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKgotICogRVhUIGV4dGVuc2lvbiBmdW5jdGlvbnMKLSAqLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLQotLyogR0xfRVhUX3RleHR1cmVfZmlsdGVyX2FuaXNvdHJvcGljICovCi0jaWZuZGVmIEdMX0VYVF90ZXh0dXJlX2ZpbHRlcl9hbmlzb3Ryb3BpYwotI2RlZmluZSBHTF9FWFRfdGV4dHVyZV9maWx0ZXJfYW5pc290cm9waWMgMQotI2VuZGlmCi0KLS8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKgotICogZGFsdmlrIGV4dGVuc2lvbiBmdW5jdGlvbnMKLSAqLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLSNpZmRlZiBBTkRST0lECi12b2lkIGdsQ29sb3JQb2ludGVyQm91bmRzKEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwKLSAgICAgICAgY29uc3QgR0x2b2lkICpwdHIsIEdMc2l6ZWkgY291bnQpOwotdm9pZCBnbE5vcm1hbFBvaW50ZXJCb3VuZHMoR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLAotICAgICAgICBjb25zdCBHTHZvaWQgKnBvaW50ZXIsIEdMc2l6ZWkgY291bnQpOwotdm9pZCBnbFRleENvb3JkUG9pbnRlckJvdW5kcyhHTGludCBzaXplLCBHTGVudW0gdHlwZSwKLSAgICAgICAgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlciwgR0xzaXplaSBjb3VudCk7Ci12b2lkIGdsVmVydGV4UG9pbnRlckJvdW5kcyhHTGludCBzaXplLCBHTGVudW0gdHlwZSwKLSAgICAgICAgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlciwgR0xzaXplaSBjb3VudCk7Ci0jZW5kaWYKLQotCi0jaWZkZWYgX19jcGx1c3BsdXMKLX0KLSNlbmRpZgotCi0jZW5kaWYgLyogX19nbGV4dF9oXyAqLwotCmRpZmYgLS1naXQgYS9vcGVuZ2wvaW5jbHVkZS9HTEVTL2dscGxhdGZvcm0uaCBiL29wZW5nbC9pbmNsdWRlL0dMRVMvZ2xwbGF0Zm9ybS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAwOTI0Y2FlLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9pbmNsdWRlL0dMRVMvZ2xwbGF0Zm9ybS5oCisrKyAvZGV2L251bGwKQEAgLTEsMzkgKzAsMCBAQAotI2lmbmRlZiBfX2dscGxhdGZvcm1faF8KLSNkZWZpbmUgX19nbHBsYXRmb3JtX2hfCi0KLS8qICRSZXZpc2lvbjogNzE3MiAkIG9uICREYXRlOjogMjAwOS0wMS0wOSAxMToxNzo0MSAtMDgwMCAjJCAqLwotCi0vKgotICogVGhpcyBkb2N1bWVudCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgU0dJIEZyZWUgU29mdHdhcmUgQiBMaWNlbnNlIFZlcnNpb24KLSAqIDIuMC4gRm9yIGRldGFpbHMsIHNlZSBodHRwOi8vb3NzLnNnaS5jb20vcHJvamVjdHMvRnJlZUIvIC4KLSAqLwotCi0vKiBQbGF0Zm9ybS1zcGVjaWZpYyB0eXBlcyBhbmQgZGVmaW5pdGlvbnMgZm9yIE9wZW5HTCBFUyAxLlggIGdsLmgKLSAqIExhc3QgbW9kaWZpZWQgb24gMjAwOC8xMi8xOQotICoKLSAqIEFkb3B0ZXJzIG1heSBtb2RpZnkga2hycGxhdGZvcm0uaCBhbmQgdGhpcyBmaWxlIHRvIHN1aXQgdGhlaXIgcGxhdGZvcm0uCi0gKiBZb3UgYXJlIGVuY291cmFnZWQgdG8gc3VibWl0IGFsbCBtb2RpZmljYXRpb25zIHRvIHRoZSBLaHJvbm9zIGdyb3VwIHNvIHRoYXQKLSAqIHRoZXkgY2FuIGJlIGluY2x1ZGVkIGluIGZ1dHVyZSB2ZXJzaW9ucyBvZiB0aGlzIGZpbGUuICBQbGVhc2Ugc3VibWl0IGNoYW5nZXMKLSAqIGJ5IHNlbmRpbmcgdGhlbSB0byB0aGUgcHVibGljIEtocm9ub3MgQnVnemlsbGEgKGh0dHA6Ly9raHJvbm9zLm9yZy9idWd6aWxsYSkKLSAqIGJ5IGZpbGluZyBhIGJ1ZyBhZ2FpbnN0IHByb2R1Y3QgIk9wZW5HTC1FUyIgY29tcG9uZW50ICJSZWdpc3RyeSIuCi0gKi8KLQotI2luY2x1ZGUgPEtIUi9raHJwbGF0Zm9ybS5oPgotCi0jaWZuZGVmIEdMX0FQSQotI2RlZmluZSBHTF9BUEkgICAgICBLSFJPTk9TX0FQSUNBTEwKLSNlbmRpZgotCi0jaWYgZGVmaW5lZChBTkRST0lEKQotCi0jZGVmaW5lIEdMX0FQSUVOVFJZIEtIUk9OT1NfQVBJRU5UUlkKLQotLy8gWFhYOiB0aGlzIHNob3VsZCBwcm9iYWJseSBub3QgYmUgaGVyZQotI2RlZmluZSBHTF9ESVJFQ1RfVEVYVFVSRV8yRF9RVUFMQ09NTSAgICAgICAgICAgICAgIDB4N0U4MAotCi0vLyBYWFg6IG5vdCBzdXJlIGhvdyB0aGlzIGlzIGludGVuZGVkIHRvIGJlIHVzZWQKLSNkZWZpbmUgR0xfR0xFWFRfUFJPVE9UWVBFUwotCi0jZW5kaWYKLQotI2VuZGlmIC8qIF9fZ2xwbGF0Zm9ybV9oXyAqLwpkaWZmIC0tZ2l0IGEvb3BlbmdsL2luY2x1ZGUvS0hSL2tocnBsYXRmb3JtLmggYi9vcGVuZ2wvaW5jbHVkZS9LSFIva2hycGxhdGZvcm0uaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNGNjMjdjNS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvaW5jbHVkZS9LSFIva2hycGxhdGZvcm0uaAorKysgL2Rldi9udWxsCkBAIC0xLDI0MSArMCwwIEBACi0jaWZuZGVmIF9fa2hycGxhdGZvcm1faF8KLSNkZWZpbmUgX19raHJwbGF0Zm9ybV9oXwotCi0vKgotKiogQ29weXJpZ2h0IChjKSAyMDA4LTIwMDkgVGhlIEtocm9ub3MgR3JvdXAgSW5jLgotKioKLSoqIFBlcm1pc3Npb24gaXMgaGVyZWJ5IGdyYW50ZWQsIGZyZWUgb2YgY2hhcmdlLCB0byBhbnkgcGVyc29uIG9idGFpbmluZyBhCi0qKiBjb3B5IG9mIHRoaXMgc29mdHdhcmUgYW5kL29yIGFzc29jaWF0ZWQgZG9jdW1lbnRhdGlvbiBmaWxlcyAodGhlCi0qKiAiTWF0ZXJpYWxzIiksIHRvIGRlYWwgaW4gdGhlIE1hdGVyaWFscyB3aXRob3V0IHJlc3RyaWN0aW9uLCBpbmNsdWRpbmcKLSoqIHdpdGhvdXQgbGltaXRhdGlvbiB0aGUgcmlnaHRzIHRvIHVzZSwgY29weSwgbW9kaWZ5LCBtZXJnZSwgcHVibGlzaCwKLSoqIGRpc3RyaWJ1dGUsIHN1YmxpY2Vuc2UsIGFuZC9vciBzZWxsIGNvcGllcyBvZiB0aGUgTWF0ZXJpYWxzLCBhbmQgdG8KLSoqIHBlcm1pdCBwZXJzb25zIHRvIHdob20gdGhlIE1hdGVyaWFscyBhcmUgZnVybmlzaGVkIHRvIGRvIHNvLCBzdWJqZWN0IHRvCi0qKiB0aGUgZm9sbG93aW5nIGNvbmRpdGlvbnM6Ci0qKgotKiogVGhlIGFib3ZlIGNvcHlyaWdodCBub3RpY2UgYW5kIHRoaXMgcGVybWlzc2lvbiBub3RpY2Ugc2hhbGwgYmUgaW5jbHVkZWQKLSoqIGluIGFsbCBjb3BpZXMgb3Igc3Vic3RhbnRpYWwgcG9ydGlvbnMgb2YgdGhlIE1hdGVyaWFscy4KLSoqCi0qKiBUSEUgTUFURVJJQUxTIEFSRSBQUk9WSURFRCAiQVMgSVMiLCBXSVRIT1VUIFdBUlJBTlRZIE9GIEFOWSBLSU5ELAotKiogRVhQUkVTUyBPUiBJTVBMSUVELCBJTkNMVURJTkcgQlVUIE5PVCBMSU1JVEVEIFRPIFRIRSBXQVJSQU5USUVTIE9GCi0qKiBNRVJDSEFOVEFCSUxJVFksIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFIEFORCBOT05JTkZSSU5HRU1FTlQuCi0qKiBJTiBOTyBFVkVOVCBTSEFMTCBUSEUgQVVUSE9SUyBPUiBDT1BZUklHSFQgSE9MREVSUyBCRSBMSUFCTEUgRk9SIEFOWQotKiogQ0xBSU0sIERBTUFHRVMgT1IgT1RIRVIgTElBQklMSVRZLCBXSEVUSEVSIElOIEFOIEFDVElPTiBPRiBDT05UUkFDVCwKLSoqIFRPUlQgT1IgT1RIRVJXSVNFLCBBUklTSU5HIEZST00sIE9VVCBPRiBPUiBJTiBDT05ORUNUSU9OIFdJVEggVEhFCi0qKiBNQVRFUklBTFMgT1IgVEhFIFVTRSBPUiBPVEhFUiBERUFMSU5HUyBJTiBUSEUgTUFURVJJQUxTLgotKi8KLQotLyogUGxhdGZvcm0tc3BlY2lmaWMgdHlwZXMgYW5kIGRlZmluaXRpb25zLgotICogJFJldmlzaW9uOiA3MjQ0ICQgb24gJERhdGU6IDIwMDktMDEtMjAgMTc6MDY6NTkgLTA4MDAgKFR1ZSwgMjAgSmFuIDIwMDkpICQKLSAqIAotICogQWRvcHRlcnMgbWF5IG1vZGlmeSB0aGlzIGZpbGUgdG8gc3VpdCB0aGVpciBwbGF0Zm9ybS4gQWRvcHRlcnMgYXJlCi0gKiBlbmNvdXJhZ2VkIHRvIHN1Ym1pdCBwbGF0Zm9ybSBzcGVjaWZpYyBtb2RpZmljYXRpb25zIHRvIHRoZSBLaHJvbm9zCi0gKiBncm91cCBzbyB0aGF0IHRoZXkgY2FuIGJlIGluY2x1ZGVkIGluIGZ1dHVyZSB2ZXJzaW9ucyBvZiB0aGlzIGZpbGUuCi0gKiBQbGVhc2Ugc3VibWl0IGNoYW5nZXMgYnkgc2VuZGluZyB0aGVtIHRvIHRoZSBwdWJsaWMgS2hyb25vcyBCdWd6aWxsYQotICogKGh0dHA6Ly9raHJvbm9zLm9yZy9idWd6aWxsYSkgYnkgZmlsaW5nIGEgYnVnIGFnYWluc3QgcHJvZHVjdAotICogIktocm9ub3MgKGdlbmVyYWwpIiBjb21wb25lbnQgIlJlZ2lzdHJ5Ii4KLSAqCi0gKiBBIHByZWRlZmluZWQgdGVtcGxhdGUgd2hpY2ggZmlsbHMgaW4gc29tZSBvZiB0aGUgYnVnIGZpZWxkcyBjYW4gYmUKLSAqIHJlYWNoZWQgdXNpbmcgaHR0cDovL3Rpbnl1cmwuY29tL2tocnBsYXRmb3JtLWgtYnVncmVwb3J0LCBidXQgeW91Ci0gKiBtdXN0IGNyZWF0ZSBhIEJ1Z3ppbGxhIGxvZ2luIGZpcnN0LgotICogCi0gKgotICogU2VlIHRoZSBJbXBsZW1lbnRlcidzIEd1aWRlbGluZXMgZm9yIGluZm9ybWF0aW9uIGFib3V0IHdoZXJlIHRoaXMgZmlsZQotICogc2hvdWxkIGJlIGxvY2F0ZWQgb24geW91ciBzeXN0ZW0uCi0gKiAgICBodHRwOi8vd3d3Lmtocm9ub3Mub3JnL3JlZ2lzdHJ5L2ltcGxlbWVudGVyc19ndWlkZS5wZGYKLSAqCi0gKiAKLSAqIFRoaXMgZmlsZSBzaG91bGQgYmUgaW5jbHVkZWQgYXMKLSAqICAgICAgICAjaW5jbHVkZSA8S0hSL2tocnBsYXRmb3JtLmg+Ci0gKiBieSB0aGUgS2hyb25vcyBBUEkgaGVhZGVyIGZpbGUgdGhhdCB1c2VzIGl0cyB0eXBlcyBhbmQgZGVmaW5lcy4KLSAqCi0gKiBUaGUgdHlwZXMgaW4gdGhpcyBmaWxlIHNob3VsZCBvbmx5IGJlIHVzZWQgdG8gZGVmaW5lIEFQSS1zcGVjaWZpYyB0eXBlcy4KLSAqIFR5cGVzIGRlZmluZWQgaW4gdGhpcyBmaWxlOgotICogICAga2hyb25vc19pbnQ4X3QgICAgICAgICAgICAgIHNpZ25lZCAgIDggIGJpdAotICogICAga2hyb25vc191aW50OF90ICAgICAgICAgICAgIHVuc2lnbmVkIDggIGJpdAotICogICAga2hyb25vc19pbnQxNl90ICAgICAgICAgICAgIHNpZ25lZCAgIDE2IGJpdAotICogICAga2hyb25vc191aW50MTZfdCAgICAgICAgICAgIHVuc2lnbmVkIDE2IGJpdAotICogICAga2hyb25vc19pbnQzMl90ICAgICAgICAgICAgIHNpZ25lZCAgIDMyIGJpdAotICogICAga2hyb25vc191aW50MzJfdCAgICAgICAgICAgIHVuc2lnbmVkIDMyIGJpdAotICogICAga2hyb25vc19pbnQ2NF90ICAgICAgICAgICAgIHNpZ25lZCAgIDY0IGJpdAotICogICAga2hyb25vc191aW50NjRfdCAgICAgICAgICAgIHVuc2lnbmVkIDY0IGJpdAotICogICAga2hyb25vc19pbnRwdHJfdCAgICAgICAgICAgIHNpZ25lZCAgIHNhbWUgbnVtYmVyIG9mIGJpdHMgYXMgYSBwb2ludGVyCi0gKiAgICBraHJvbm9zX3VpbnRwdHJfdCAgICAgICAgICAgdW5zaWduZWQgc2FtZSBudW1iZXIgb2YgYml0cyBhcyBhIHBvaW50ZXIKLSAqICAgIGtocm9ub3Nfc3NpemVfdCAgICAgICAgICAgICBzaWduZWQgICBzaXplCi0gKiAgICBraHJvbm9zX3VzaXplX3QgICAgICAgICAgICAgdW5zaWduZWQgc2l6ZQotICogICAga2hyb25vc19mbG9hdF90ICAgICAgICAgICAgIHNpZ25lZCAgIDMyIGJpdCBmbG9hdGluZyBwb2ludAotICogICAga2hyb25vc190aW1lX25zX3QgICAgICAgICAgIHVuc2lnbmVkIDY0IGJpdCB0aW1lIGluIG5hbm9zZWNvbmRzCi0gKiAgICBraHJvbm9zX3V0aW1lX25hbm9zZWNvbmRzX3QgdW5zaWduZWQgdGltZSBpbnRlcnZhbCBvciBhYnNvbHV0ZSB0aW1lIGluCi0gKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbmFub3NlY29uZHMKLSAqICAgIGtocm9ub3Nfc3RpbWVfbmFub3NlY29uZHNfdCBzaWduZWQgdGltZSBpbnRlcnZhbCBpbiBuYW5vc2Vjb25kcwotICoKLSAqIEtIUk9OT1NfU1VQUE9SVF9JTlQ2NCBpcyAxIGlmIDY0IGJpdCBpbnRlZ2VycyBhcmUgc3VwcG9ydGVkOyBvdGhlcndpc2UgMC4KLSAqIEtIUk9OT1NfU1VQUE9SVF9GTE9BVCBpcyAxIGlmIGZsb2F0cyBhcmUgc3VwcG9ydGVkOyBvdGhlcndpc2UgMC4KLSAqIAotICoKLSAqIE1hY3JvcyBkZWZpbmVkIGluIHRoaXMgZmlsZToKLSAqICAgIEtIUk9OT1NfQVBJQ0FMTAotICogICAgS0hST05PU19BUElFTlRSWQotICogICAgS0hST05PU19BUElBVFRSSUJVVEVTCi0gKiBUaGVzZSBtYXkgYmUgdXNlZCBpbiBmdW5jdGlvbiBwcm90b3R5cGVzIGFzOgotICogICAgICBLSFJPTk9TX0FQSUNBTEwgdm9pZCBLSFJPTk9TX0FQSUVOVFJZIGZ1bmNuYW1lKAotICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGFyZzEsCi0gKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQgYXJnMikgS0hST05PU19BUElBVFRSSUJVVEVTOwotICovCi0KLS8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICogRGVmaW5pdGlvbiBvZiBLSFJPTk9TX0FQSUNBTEwKLSAqLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICogVGhpcyBwcmVjZWRlcyB0aGUgcmV0dXJuIHR5cGUgb2YgdGhlIGZ1bmN0aW9uIGluIHRoZSBmdW5jdGlvbiBwcm90b3R5cGUuCi0gKi8KLSNpZiBkZWZpbmVkKF9XSU4zMikgJiYgIWRlZmluZWQoX19TQ0lURUNIX1NOQVBfXykKLSMgICBkZWZpbmUgS0hST05PU19BUElDQUxMIF9fZGVjbHNwZWMoZGxsaW1wb3J0KQotI2VsaWYgZGVmaW5lZCAoX19TWU1CSUFOMzJfXykKLSMgICBkZWZpbmUgS0hST05PU19BUElDQUxMIElNUE9SVF9DCi0jZWxzZQotIyAgIGRlZmluZSBLSFJPTk9TX0FQSUNBTEwKLSNlbmRpZgotCi0vKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSAqIERlZmluaXRpb24gb2YgS0hST05PU19BUElFTlRSWQotICotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0gKiBUaGlzIGZvbGxvd3MgdGhlIHJldHVybiB0eXBlIG9mIHRoZSBmdW5jdGlvbiAgYW5kIHByZWNlZGVzIHRoZSBmdW5jdGlvbgotICogbmFtZSBpbiB0aGUgZnVuY3Rpb24gcHJvdG90eXBlLgotICovCi0jaWYgZGVmaW5lZChfV0lOMzIpICYmICFkZWZpbmVkKF9XSU4zMl9XQ0UpICYmICFkZWZpbmVkKF9fU0NJVEVDSF9TTkFQX18pCi0gICAgLyogV2luMzIgYnV0IG5vdCBXaW5DRSAqLwotIyAgIGRlZmluZSBLSFJPTk9TX0FQSUVOVFJZIF9fc3RkY2FsbAotI2Vsc2UKLSMgICBkZWZpbmUgS0hST05PU19BUElFTlRSWQotI2VuZGlmCi0KLS8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICogRGVmaW5pdGlvbiBvZiBLSFJPTk9TX0FQSUFUVFJJQlVURVMKLSAqLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotICogVGhpcyBmb2xsb3dzIHRoZSBjbG9zaW5nIHBhcmVudGhlc2lzIG9mIHRoZSBmdW5jdGlvbiBwcm90b3R5cGUgYXJndW1lbnRzLgotICovCi0jaWYgZGVmaW5lZCAoX19BUk1DQ18yX18pCi0jZGVmaW5lIEtIUk9OT1NfQVBJQVRUUklCVVRFUyBfX3NvZnRmcAotI2Vsc2UKLSNkZWZpbmUgS0hST05PU19BUElBVFRSSUJVVEVTCi0jZW5kaWYKLQotLyotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0gKiBiYXNpYyB0eXBlIGRlZmluaXRpb25zCi0gKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLSNpZiAoZGVmaW5lZChfX1NURENfVkVSU0lPTl9fKSAmJiBfX1NURENfVkVSU0lPTl9fID49IDE5OTkwMUwpIHx8IGRlZmluZWQoX19HTlVDX18pIHx8IGRlZmluZWQoX19TQ09fXykgfHwgZGVmaW5lZChfX1VTTENfXykKLQotCi0vKgotICogVXNpbmcgPHN0ZGludC5oPgotICovCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci10eXBlZGVmIGludDMyX3QgICAgICAgICAgICAgICAgIGtocm9ub3NfaW50MzJfdDsKLXR5cGVkZWYgdWludDMyX3QgICAgICAgICAgICAgICAga2hyb25vc191aW50MzJfdDsKLXR5cGVkZWYgaW50NjRfdCAgICAgICAgICAgICAgICAga2hyb25vc19pbnQ2NF90OwotdHlwZWRlZiB1aW50NjRfdCAgICAgICAgICAgICAgICBraHJvbm9zX3VpbnQ2NF90OwotI2RlZmluZSBLSFJPTk9TX1NVUFBPUlRfSU5UNjQgICAxCi0jZGVmaW5lIEtIUk9OT1NfU1VQUE9SVF9GTE9BVCAgIDEKLQotI2VsaWYgZGVmaW5lZChfX1ZNUyApIHx8IGRlZmluZWQoX19zZ2kpCi0KLS8qCi0gKiBVc2luZyA8aW50dHlwZXMuaD4KLSAqLwotI2luY2x1ZGUgPGludHR5cGVzLmg+Ci10eXBlZGVmIGludDMyX3QgICAgICAgICAgICAgICAgIGtocm9ub3NfaW50MzJfdDsKLXR5cGVkZWYgdWludDMyX3QgICAgICAgICAgICAgICAga2hyb25vc191aW50MzJfdDsKLXR5cGVkZWYgaW50NjRfdCAgICAgICAgICAgICAgICAga2hyb25vc19pbnQ2NF90OwotdHlwZWRlZiB1aW50NjRfdCAgICAgICAgICAgICAgICBraHJvbm9zX3VpbnQ2NF90OwotI2RlZmluZSBLSFJPTk9TX1NVUFBPUlRfSU5UNjQgICAxCi0jZGVmaW5lIEtIUk9OT1NfU1VQUE9SVF9GTE9BVCAgIDEKLQotI2VsaWYgZGVmaW5lZChfV0lOMzIpICYmICFkZWZpbmVkKF9fU0NJVEVDSF9TTkFQX18pCi0KLS8qCi0gKiBXaW4zMgotICovCi10eXBlZGVmIF9faW50MzIgICAgICAgICAgICAgICAgIGtocm9ub3NfaW50MzJfdDsKLXR5cGVkZWYgdW5zaWduZWQgX19pbnQzMiAgICAgICAga2hyb25vc191aW50MzJfdDsKLXR5cGVkZWYgX19pbnQ2NCAgICAgICAgICAgICAgICAga2hyb25vc19pbnQ2NF90OwotdHlwZWRlZiB1bnNpZ25lZCBfX2ludDY0ICAgICAgICBraHJvbm9zX3VpbnQ2NF90OwotI2RlZmluZSBLSFJPTk9TX1NVUFBPUlRfSU5UNjQgICAxCi0jZGVmaW5lIEtIUk9OT1NfU1VQUE9SVF9GTE9BVCAgIDEKLQotI2VsaWYgZGVmaW5lZChfX3N1bl9fKSB8fCBkZWZpbmVkKF9fZGlnaXRhbF9fKQotCi0vKgotICogU3VuIG9yIERpZ2l0YWwKLSAqLwotdHlwZWRlZiBpbnQgICAgICAgICAgICAgICAgICAgICBraHJvbm9zX2ludDMyX3Q7Ci10eXBlZGVmIHVuc2lnbmVkIGludCAgICAgICAgICAgIGtocm9ub3NfdWludDMyX3Q7Ci0jaWYgZGVmaW5lZChfX2FyY2g2NF9fKSB8fCBkZWZpbmVkKF9MUDY0KQotdHlwZWRlZiBsb25nIGludCAgICAgICAgICAgICAgICBraHJvbm9zX2ludDY0X3Q7Ci10eXBlZGVmIHVuc2lnbmVkIGxvbmcgaW50ICAgICAgIGtocm9ub3NfdWludDY0X3Q7Ci0jZWxzZQotdHlwZWRlZiBsb25nIGxvbmcgaW50ICAgICAgICAgICBraHJvbm9zX2ludDY0X3Q7Ci10eXBlZGVmIHVuc2lnbmVkIGxvbmcgbG9uZyBpbnQgIGtocm9ub3NfdWludDY0X3Q7Ci0jZW5kaWYgLyogX19hcmNoNjRfXyAqLwotI2RlZmluZSBLSFJPTk9TX1NVUFBPUlRfSU5UNjQgICAxCi0jZGVmaW5lIEtIUk9OT1NfU1VQUE9SVF9GTE9BVCAgIDEKLQotI2VsaWYgMAotCi0vKgotICogSHlwb3RoZXRpY2FsIHBsYXRmb3JtIHdpdGggbm8gZmxvYXQgb3IgaW50NjQgc3VwcG9ydAotICovCi10eXBlZGVmIGludCAgICAgICAgICAgICAgICAgICAgIGtocm9ub3NfaW50MzJfdDsKLXR5cGVkZWYgdW5zaWduZWQgaW50ICAgICAgICAgICAga2hyb25vc191aW50MzJfdDsKLSNkZWZpbmUgS0hST05PU19TVVBQT1JUX0lOVDY0ICAgMAotI2RlZmluZSBLSFJPTk9TX1NVUFBPUlRfRkxPQVQgICAwCi0KLSNlbHNlCi0KLS8qCi0gKiBHZW5lcmljIGZhbGxiYWNrCi0gKi8KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLXR5cGVkZWYgaW50MzJfdCAgICAgICAgICAgICAgICAga2hyb25vc19pbnQzMl90OwotdHlwZWRlZiB1aW50MzJfdCAgICAgICAgICAgICAgICBraHJvbm9zX3VpbnQzMl90OwotdHlwZWRlZiBpbnQ2NF90ICAgICAgICAgICAgICAgICBraHJvbm9zX2ludDY0X3Q7Ci10eXBlZGVmIHVpbnQ2NF90ICAgICAgICAgICAgICAgIGtocm9ub3NfdWludDY0X3Q7Ci0jZGVmaW5lIEtIUk9OT1NfU1VQUE9SVF9JTlQ2NCAgIDEKLSNkZWZpbmUgS0hST05PU19TVVBQT1JUX0ZMT0FUICAgMQotCi0jZW5kaWYKLQotCi0vKgotICogVHlwZXMgdGhhdCBhcmUgKHNvIGZhcikgdGhlIHNhbWUgb24gYWxsIHBsYXRmb3JtcwotICovCi10eXBlZGVmIHNpZ25lZCAgIGNoYXIgICAgICAgICAga2hyb25vc19pbnQ4X3Q7Ci10eXBlZGVmIHVuc2lnbmVkIGNoYXIgICAgICAgICAga2hyb25vc191aW50OF90OwotdHlwZWRlZiBzaWduZWQgICBzaG9ydCBpbnQgICAgIGtocm9ub3NfaW50MTZfdDsKLXR5cGVkZWYgdW5zaWduZWQgc2hvcnQgaW50ICAgICBraHJvbm9zX3VpbnQxNl90OwotdHlwZWRlZiBzaWduZWQgICBsb25nICBpbnQgICAgIGtocm9ub3NfaW50cHRyX3Q7Ci10eXBlZGVmIHVuc2lnbmVkIGxvbmcgIGludCAgICAga2hyb25vc191aW50cHRyX3Q7Ci10eXBlZGVmIHNpZ25lZCAgIGxvbmcgIGludCAgICAga2hyb25vc19zc2l6ZV90OwotdHlwZWRlZiB1bnNpZ25lZCBsb25nICBpbnQgICAgIGtocm9ub3NfdXNpemVfdDsKLQotI2lmIEtIUk9OT1NfU1VQUE9SVF9GTE9BVAotLyoKLSAqIEZsb2F0IHR5cGUKLSAqLwotdHlwZWRlZiAgICAgICAgICBmbG9hdCAgICAgICAgIGtocm9ub3NfZmxvYXRfdDsKLSNlbmRpZgotCi0jaWYgS0hST05PU19TVVBQT1JUX0lOVDY0Ci0vKiBUaW1lIHR5cGVzCi0gKgotICogVGhlc2UgdHlwZXMgY2FuIGJlIHVzZWQgdG8gcmVwcmVzZW50IGEgdGltZSBpbnRlcnZhbCBpbiBuYW5vc2Vjb25kcyBvciAKLSAqIGFuIGFic29sdXRlIFVuYWRqdXN0ZWQgU3lzdGVtIFRpbWUuICBVbmFkanVzdGVkIFN5c3RlbSBUaW1lIGlzIHRoZSBudW1iZXIgCi0gKiBvZiBuYW5vc2Vjb25kcyBzaW5jZSBzb21lIGFyYml0cmFyeSBzeXN0ZW0gZXZlbnQgKGUuZy4gc2luY2UgdGhlIGxhc3QgCi0gKiB0aW1lIHRoZSBzeXN0ZW0gYm9vdGVkKS4gIFRoZSBVbmFkanVzdGVkIFN5c3RlbSBUaW1lIGlzIGFuIHVuc2lnbmVkIAotICogNjQgYml0IHZhbHVlIHRoYXQgd3JhcHMgYmFjayB0byAwIGV2ZXJ5IDU4NCB5ZWFycy4gIFRpbWUgaW50ZXJ2YWxzIAotICogbWF5IGJlIGVpdGhlciBzaWduZWQgb3IgdW5zaWduZWQuCi0gKi8KLXR5cGVkZWYga2hyb25vc191aW50NjRfdCAgICAgICBraHJvbm9zX3V0aW1lX25hbm9zZWNvbmRzX3Q7Ci10eXBlZGVmIGtocm9ub3NfaW50NjRfdCAgICAgICAga2hyb25vc19zdGltZV9uYW5vc2Vjb25kc190OwotI2VuZGlmCi0KLQotI2VuZGlmIC8qIF9fa2hycGxhdGZvcm1faF8gKi8KZGlmZiAtLWdpdCBhL29wZW5nbC9saWJhZ2wvQW5kcm9pZC5tayBiL29wZW5nbC9saWJhZ2wvQW5kcm9pZC5tawpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOTllZmU0Yy4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGliYWdsL0FuZHJvaWQubWsKKysrIC9kZXYvbnVsbApAQCAtMSwzOSArMCwwIEBACi1MT0NBTF9QQVRIOj0gJChjYWxsIG15LWRpcikKLQotIwotIyBCdWlsZCB0aGUgc29mdHdhcmUgT3BlbkdMIEVTIGxpYnJhcnkKLSMKLQotaW5jbHVkZSAkKENMRUFSX1ZBUlMpCi0KLUxPQ0FMX1NSQ19GSUxFUzo9IFwKLQllZ2wuY3BwICAgICAgICAgICAgICAgICAgICAgXAotCXN0YXRlLmNwcAkJICAgICAgICAgICAgXAotCXRleHR1cmUuY3BwCQkgICAgICAgICAgICBcCi0gICAgVG9rZW5pemVyLmNwcCAgICAgICAgICAgICAgIFwKLSAgICBUb2tlbk1hbmFnZXIuY3BwICAgICAgICAgICAgXAotICAgIFRleHR1cmVPYmplY3RNYW5hZ2VyLmNwcCAgICBcCi0gICAgQnVmZmVyT2JqZWN0TWFuYWdlci5jcHAgICAgIFwKLQlhcnJheS5jcHAuYXJtCQkgICAgICAgIFwKLQlmcC5jcHAuYXJtCQkgICAgICAgICAgICBcCi0JbGlnaHQuY3BwLmFybQkJICAgICAgICBcCi0JbWF0cml4LmNwcC5hcm0JCSAgICAgICAgXAotCW1pcG1hcC5jcHAuYXJtCQkgICAgICAgIFwKLQlwcmltaXRpdmVzLmNwcC5hcm0JICAgICAgICBcCi0JdmVydGV4LmNwcC5hcm0KLQotaWZlcSAoJChUQVJHRVRfQVJDSCksYXJtKQotCUxPQ0FMX1NSQ19GSUxFUyArPSBmaXhlZF9hc20uUyBpdGVyYXRvcnMuUwotCUxPQ0FMX0NGTEFHUyArPSAtZnN0cmljdC1hbGlhc2luZwotZW5kaWYKLQotaWZuZXEgKCQoVEFSR0VUX1NJTVVMQVRPUiksdHJ1ZSkKLSAgICAjIHdlIG5lZWQgdG8gYWNjZXNzIHRoZSBwcml2YXRlIEJpb25pYyBoZWFkZXIgPGJpb25pY190bHMuaD4KLSAgICBMT0NBTF9DRkxBR1MgKz0gLUkkKExPQ0FMX1BBVEgpLy4uLy4uLy4uLy4uL2Jpb25pYy9saWJjL3ByaXZhdGUKLWVuZGlmCi0KLUxPQ0FMX1NIQVJFRF9MSUJSQVJJRVMgOj0gbGliY3V0aWxzIGxpYnV0aWxzIGxpYnBpeGVsZmxpbmdlcgotTE9DQUxfTERMSUJTIDo9IC1scHRocmVhZCAtbGRsCi1MT0NBTF9NT0RVTEU6PSBsaWJhZ2wKLQotaW5jbHVkZSAkKEJVSUxEX1NIQVJFRF9MSUJSQVJZKQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC9CdWZmZXJPYmplY3RNYW5hZ2VyLmNwcCBiL29wZW5nbC9saWJhZ2wvQnVmZmVyT2JqZWN0TWFuYWdlci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDZiZjI4ZWUuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9CdWZmZXJPYmplY3RNYW5hZ2VyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDEwMyArMCwwIEBACi0vKgotICoqIENvcHlyaWdodCAyMDA4LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKioKLSAqKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0gKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSAqKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0gKioKLSAqKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotICoqCi0gKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSAqKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSAqKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0gKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSAqKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3RkZGVmLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9SZWZCYXNlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvS2V5ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLQotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLQotI2luY2x1ZGUgIkJ1ZmZlck9iamVjdE1hbmFnZXIuaCIKLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLXVzaW5nIG5hbWVzcGFjZSBnbDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1FR0xCdWZmZXJPYmplY3RNYW5hZ2VyOjpFR0xCdWZmZXJPYmplY3RNYW5hZ2VyKCkgCi06IFRva2VuTWFuYWdlcigpLCBtQ291bnQoMCkKLXsKLX0KLQotRUdMQnVmZmVyT2JqZWN0TWFuYWdlcjo6fkVHTEJ1ZmZlck9iamVjdE1hbmFnZXIoKQotewotICAgIC8vIGRlc3Ryb3kgYWxsIHRoZSBidWZmZXIgb2JqZWN0cyBhbmQgdGhlaXIgc3RvcmFnZQotICAgIEdMc2l6ZWkgbiA9IG1CdWZmZXJzLnNpemUoKTsKLSAgICBmb3IgKEdMc2l6ZWkgaT0wIDsgaTxuIDsgaSsrKSB7Ci0gICAgICAgIGJ1ZmZlcl90KiBibyA9IG1CdWZmZXJzLnZhbHVlQXQoaSk7Ci0gICAgICAgIGZyZWUoYm8tPmRhdGEpOwotICAgICAgICBkZWxldGUgYm87Ci0gICAgfQotfQotCi1idWZmZXJfdCBjb25zdCogRUdMQnVmZmVyT2JqZWN0TWFuYWdlcjo6YmluZChHTHVpbnQgYnVmZmVyKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgaW50MzJfdCBpID0gbUJ1ZmZlcnMuaW5kZXhPZktleShidWZmZXIpOwotICAgIGlmIChpID49IDApIHsKLSAgICAgICAgcmV0dXJuIG1CdWZmZXJzLnZhbHVlQXQoaSk7Ci0gICAgfQotICAgIGJ1ZmZlcl90KiBibyA9IG5ldyBidWZmZXJfdDsKLSAgICBiby0+ZGF0YSA9IDA7Ci0gICAgYm8tPnVzYWdlID0gR0xfU1RBVElDX0RSQVc7Ci0gICAgYm8tPnNpemUgPSAwOwotICAgIGJvLT5uYW1lID0gYnVmZmVyOwotICAgIG1CdWZmZXJzLmFkZChidWZmZXIsIGJvKTsKLSAgICByZXR1cm4gYm87Ci19Ci0KLWludCBFR0xCdWZmZXJPYmplY3RNYW5hZ2VyOjphbGxvY2F0ZVN0b3JlKGJ1ZmZlcl90KiBibywKLSAgICAgICAgR0xzaXplaXB0ciBzaXplLCBHTGVudW0gdXNhZ2UpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICBpZiAoc2l6ZSAhPSBiby0+c2l6ZSkgewotICAgICAgIHVpbnQ4X3QqIGRhdGEgPSAodWludDhfdCopbWFsbG9jKHNpemUpOwotICAgICAgICBpZiAoZGF0YSA9PSAwKQotICAgICAgICAgICAgcmV0dXJuIC0xOwotICAgICAgICBmcmVlKGJvLT5kYXRhKTsKLSAgICAgICAgYm8tPmRhdGEgPSBkYXRhOwotICAgICAgICBiby0+c2l6ZSA9IHNpemU7Ci0gICAgfQotICAgIGJvLT51c2FnZSA9IHVzYWdlOwotICAgIHJldHVybiAwOwotfQotCi12b2lkIEVHTEJ1ZmZlck9iamVjdE1hbmFnZXI6OmRlbGV0ZUJ1ZmZlcnMoR0xzaXplaSBuLCBjb25zdCBHTHVpbnQqIGJ1ZmZlcnMpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICB3aGlsZSAobi0tKSB7Ci0gICAgICAgIGNvbnN0IEdMdWludCB0ID0gKmJ1ZmZlcnMrKzsKLSAgICAgICAgaWYgKHQpIHsKLSAgICAgICAgICAgIGludDMyX3QgaW5kZXggPSBtQnVmZmVycy5pbmRleE9mS2V5KHQpOwotICAgICAgICAgICAgaWYgKGluZGV4ID49IDApIHsKLSAgICAgICAgICAgICAgICBidWZmZXJfdCogYm8gPSBtQnVmZmVycy52YWx1ZUF0KGluZGV4KTsKLSAgICAgICAgICAgICAgICBmcmVlKGJvLT5kYXRhKTsKLSAgICAgICAgICAgICAgICBtQnVmZmVycy5yZW1vdmVJdGVtc0F0KGluZGV4KTsKLSAgICAgICAgICAgICAgICBkZWxldGUgYm87Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL0J1ZmZlck9iamVjdE1hbmFnZXIuaCBiL29wZW5nbC9saWJhZ2wvQnVmZmVyT2JqZWN0TWFuYWdlci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5ZTkzNDBhLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJhZ2wvQnVmZmVyT2JqZWN0TWFuYWdlci5oCisrKyAvZGV2L251bGwKQEAgLTEsODUgKzAsMCBAQAotLyoKLSAqKgotICoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKioKLSAqKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0gKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSAqKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0gKioKLSAqKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotICoqCi0gKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSAqKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSAqKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0gKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSAqKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaWZuZGVmIEFORFJPSURfT1BFTkdMRVNfQlVGRkVSX09CSkVDVF9NQU5BR0VSX0gKLSNkZWZpbmUgQU5EUk9JRF9PUEVOR0xFU19CVUZGRVJfT0JKRUNUX01BTkFHRVJfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3RkZGVmLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9BdG9taWMuaD4KLSNpbmNsdWRlIDx1dGlscy9SZWZCYXNlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvS2V5ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLQotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLQotI2luY2x1ZGUgIlRva2VuaXplci5oIgotI2luY2x1ZGUgIlRva2VuTWFuYWdlci5oIgotCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgZ2wgewotCi1zdHJ1Y3QgYnVmZmVyX3QgewotICAgIEdMc2l6ZWlwdHIgICAgICBzaXplOwotICAgIEdMZW51bSAgICAgICAgICB1c2FnZTsKLSAgICB1aW50OF90KiAgICAgICAgZGF0YTsKLSAgICB1aW50MzJfdCAgICAgICAgbmFtZTsKLX07Ci0KLX07Ci0KLWNsYXNzIEVHTEJ1ZmZlck9iamVjdE1hbmFnZXIgOiBwdWJsaWMgVG9rZW5NYW5hZ2VyCi17Ci1wdWJsaWM6Ci0gICAgRUdMQnVmZmVyT2JqZWN0TWFuYWdlcigpOwotICAgIH5FR0xCdWZmZXJPYmplY3RNYW5hZ2VyKCk7Ci0KLSAgICAvLyBwcm90b2NvbCBmb3Igc3A8PgotICAgIGlubGluZSAgdm9pZCAgICBpbmNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0OwotICAgIGlubGluZSAgdm9pZCAgICBkZWNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0OwotICAgIHR5cGVkZWYgdm9pZCAgICB3ZWFrcmVmX3R5cGU7Ci0KLSAgICBnbDo6YnVmZmVyX3QgY29uc3QqIGJpbmQoR0x1aW50IGJ1ZmZlcik7Ci0gICAgaW50ICAgICAgICAgICAgICAgICBhbGxvY2F0ZVN0b3JlKGdsOjpidWZmZXJfdCogYm8sIEdMc2l6ZWlwdHIgc2l6ZSwgR0xlbnVtIHVzYWdlKTsKLSAgICB2b2lkICAgICAgICAgICAgICAgIGRlbGV0ZUJ1ZmZlcnMoR0xzaXplaSBuLCBjb25zdCBHTHVpbnQqIGJ1ZmZlcnMpOwotCi1wcml2YXRlOgotICAgIG11dGFibGUgdm9sYXRpbGUgaW50MzJfdCAgICAgICAgICAgIG1Db3VudDsKLSAgICBtdXRhYmxlIE11dGV4ICAgICAgICAgICAgICAgICAgICAgICBtTG9jazsKLSAgICBLZXllZFZlY3RvcjxHTHVpbnQsIGdsOjpidWZmZXJfdCo+ICBtQnVmZmVyczsKLX07Ci0KLXZvaWQgRUdMQnVmZmVyT2JqZWN0TWFuYWdlcjo6aW5jU3Ryb25nKGNvbnN0IHZvaWQqIGlkKSBjb25zdCB7Ci0gICAgYW5kcm9pZF9hdG9taWNfaW5jKCZtQ291bnQpOwotfQotdm9pZCBFR0xCdWZmZXJPYmplY3RNYW5hZ2VyOjpkZWNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0IHsKLSAgICBpZiAoYW5kcm9pZF9hdG9taWNfZGVjKCZtQ291bnQpID09IDEpIHsKLSAgICAgICAgZGVsZXRlIHRoaXM7Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9PUEVOR0xFU19CVUZGRVJfT0JKRUNUX01BTkFHRVJfSAotCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL1RleHR1cmVPYmplY3RNYW5hZ2VyLmNwcCBiL29wZW5nbC9saWJhZ2wvVGV4dHVyZU9iamVjdE1hbmFnZXIuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjZTMxODU0Li4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJhZ2wvVGV4dHVyZU9iamVjdE1hbmFnZXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMzA5ICswLDAgQEAKLS8qCi0gKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqKgotICoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSAqKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotICoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSAqKgotICoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0gKioKLSAqKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotICoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotICoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSAqKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotICoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgImNvbnRleHQuaCIKLSNpbmNsdWRlICJUZXh0dXJlT2JqZWN0TWFuYWdlci5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUVHTFRleHR1cmVPYmplY3Q6OkVHTFRleHR1cmVPYmplY3QoKQotICAgIDogbUNvdW50KDApLCBtU2l6ZSgwKQotewotICAgIGluaXQoKTsKLX0KLQotRUdMVGV4dHVyZU9iamVjdDo6fkVHTFRleHR1cmVPYmplY3QoKQotewotICAgIGlmICghZGlyZWN0KSB7Ci0gICAgICAgIGlmIChtU2l6ZSAmJiBzdXJmYWNlLmRhdGEpCi0gICAgICAgICAgICBmcmVlKHN1cmZhY2UuZGF0YSk7Ci0gICAgICAgIGlmIChtTWlwbWFwcykKLSAgICAgICAgICAgIGZyZWVNaXBtYXBzKCk7Ci0gICAgfQotfQotCi12b2lkIEVHTFRleHR1cmVPYmplY3Q6OmluaXQoKQotewotICAgIG1lbXNldCgmc3VyZmFjZSwgMCwgc2l6ZW9mKHN1cmZhY2UpKTsKLSAgICBzdXJmYWNlLnZlcnNpb24gPSBzaXplb2Yoc3VyZmFjZSk7Ci0gICAgbU1pcG1hcHMgPSAwOwotICAgIG1OdW1FeHRyYUxvZCA9IDA7Ci0gICAgbUlzQ29tcGxldGUgPSBmYWxzZTsKLSAgICB3cmFwcyA9IEdMX1JFUEVBVDsKLSAgICB3cmFwdCA9IEdMX1JFUEVBVDsKLSAgICBtaW5fZmlsdGVyID0gR0xfTElORUFSOwotICAgIG1hZ19maWx0ZXIgPSBHTF9MSU5FQVI7Ci0gICAgaW50ZXJuYWxmb3JtYXQgPSAwOwotICAgIG1lbXNldChjcm9wX3JlY3QsIDAsIHNpemVvZihjcm9wX3JlY3QpKTsKLSAgICBnZW5lcmF0ZV9taXBtYXAgPSBHTF9GQUxTRTsKLSAgICBkaXJlY3QgPSBHTF9GQUxTRTsKLX0KLQotdm9pZCBFR0xUZXh0dXJlT2JqZWN0Ojpjb3B5UGFyYW1ldGVycyhjb25zdCBzcDxFR0xUZXh0dXJlT2JqZWN0PiYgb2xkKQotewotICAgIHdyYXBzID0gb2xkLT53cmFwczsKLSAgICB3cmFwdCA9IG9sZC0+d3JhcHQ7Ci0gICAgbWluX2ZpbHRlciA9IG9sZC0+bWluX2ZpbHRlcjsKLSAgICBtYWdfZmlsdGVyID0gb2xkLT5tYWdfZmlsdGVyOwotICAgIG1lbWNweShjcm9wX3JlY3QsIG9sZC0+Y3JvcF9yZWN0LCBzaXplb2YoY3JvcF9yZWN0KSk7Ci0gICAgZ2VuZXJhdGVfbWlwbWFwID0gb2xkLT5nZW5lcmF0ZV9taXBtYXA7Ci0gICAgZGlyZWN0ID0gb2xkLT5kaXJlY3Q7Ci19Ci0KLXN0YXR1c190IEVHTFRleHR1cmVPYmplY3Q6OmFsbG9jYXRlTWlwbWFwcygpCi17Ci0gICAgLy8gaGVyZSwgYnkgY29uc3RydWN0aW9uLCBtTWlwbWFwcz0wICYmIG1OdW1FeHRyYUxvZD0wCi0KLSAgICBpZiAoIXN1cmZhY2UuZGF0YSkKLSAgICAgICAgcmV0dXJuIE5PX0lOSVQ7Ci0KLSAgICBpbnQgdyA9IHN1cmZhY2Uud2lkdGg7Ci0gICAgaW50IGggPSBzdXJmYWNlLmhlaWdodDsKLSAgICBjb25zdCBpbnQgbnVtTG9kcyA9IDMxIC0gZ2dsQ2x6KG1heCh3LGgpKTsKLSAgICBpZiAobnVtTG9kcyA8PSAwKQotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0KLSAgICBtTWlwbWFwcyA9IChHR0xTdXJmYWNlKiltYWxsb2MobnVtTG9kcyAqIHNpemVvZihHR0xTdXJmYWNlKSk7Ci0gICAgaWYgKCFtTWlwbWFwcykKLSAgICAgICAgcmV0dXJuIE5PX01FTU9SWTsKLQotICAgIG1lbXNldChtTWlwbWFwcywgMCwgbnVtTG9kcyAqIHNpemVvZihHR0xTdXJmYWNlKSk7Ci0gICAgbU51bUV4dHJhTG9kID0gbnVtTG9kczsKLSAgICByZXR1cm4gTk9fRVJST1I7Ci19Ci0KLXZvaWQgRUdMVGV4dHVyZU9iamVjdDo6ZnJlZU1pcG1hcHMoKQotewotICAgIGlmIChtTWlwbWFwcykgewotICAgICAgICBmb3IgKGludCBpPTAgOyBpPG1OdW1FeHRyYUxvZCA7IGkrKykgewotICAgICAgICAgICAgaWYgKG1NaXBtYXBzW2ldLmRhdGEpIHsKLSAgICAgICAgICAgICAgICBmcmVlKG1NaXBtYXBzW2ldLmRhdGEpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGZyZWUobU1pcG1hcHMpOwotICAgICAgICBtTWlwbWFwcyA9IDA7Ci0gICAgICAgIG1OdW1FeHRyYUxvZCA9IDA7Ci0gICAgfQotfQotCi1jb25zdCBHR0xTdXJmYWNlJiBFR0xUZXh0dXJlT2JqZWN0OjptaXAoaW50IGxvZCkgY29uc3QKLXsKLSAgICBpZiAobG9kPD0wIHx8ICFtTWlwbWFwcykKLSAgICAgICAgcmV0dXJuIHN1cmZhY2U7Ci0gICAgbG9kID0gbWluKGxvZC0xLCBtTnVtRXh0cmFMb2QtMSk7Ci0gICAgcmV0dXJuIG1NaXBtYXBzW2xvZF07Ci19Ci0KLUdHTFN1cmZhY2UmIEVHTFRleHR1cmVPYmplY3Q6OmVkaXRNaXAoaW50IGxvZCkKLXsKLSAgICByZXR1cm4gY29uc3RfY2FzdDxHR0xTdXJmYWNlJj4obWlwKGxvZCkpOwotfQotCi1zdGF0dXNfdCBFR0xUZXh0dXJlT2JqZWN0OjpzZXRTdXJmYWNlKEdHTFN1cmZhY2UgY29uc3QqIHMpCi17Ci0gICAgLy8gWFhYOiBnbEZsdXNoKCkgb24gJ3MnCi0gICAgaWYgKG1TaXplICYmIHN1cmZhY2UuZGF0YSkgewotICAgICAgICBmcmVlKHN1cmZhY2UuZGF0YSk7Ci0gICAgfQotICAgIHN1cmZhY2UgPSAqczsKLSAgICBpbnRlcm5hbGZvcm1hdCA9IDA7Ci0KLSAgICAvLyB3ZSBzaG91bGQga2VlcCB0aGUgY3JvcF9yZWN0LCBidXQgaXQncyBkZWxpY2F0ZSBiZWNhdXNlCi0gICAgLy8gdGhlIG5ldyBzaXplIG9mIHRoZSBzdXJmYWNlIGNvdWxkIG1ha2UgaXQgaW52YWxpZC4KLSAgICAvLyBzbyBmb3Igbm93LCB3ZSBqdXN0IGxvb3NlIGl0LgotICAgIG1lbXNldChjcm9wX3JlY3QsIDAsIHNpemVvZihjcm9wX3JlY3QpKTsKLQotICAgIC8vIGl0IHdvdWxkIGJlIG5pY2UgaWYgd2UgY291bGQga2VlcCB0aGUgZ2VuZXJhdGVfbWlwbWFwIGZsYWcsCi0gICAgLy8gd2Ugd291bGQgaGF2ZSB0byBnZW5lcmF0ZSB0aGVtIHJpZ2h0IG5vdyB0aG91Z2guCi0gICAgZ2VuZXJhdGVfbWlwbWFwID0gR0xfRkFMU0U7Ci0KLSAgICBkaXJlY3QgPSBHTF9UUlVFOwotICAgIG1TaXplID0gMDsgIC8vIHdlIGRvbid0IG93biB0aGlzIHN1cmZhY2UKLSAgICBpZiAobU1pcG1hcHMpCi0gICAgICAgIGZyZWVNaXBtYXBzKCk7Ci0gICAgbUlzQ29tcGxldGUgPSB0cnVlOwotICAgIHJldHVybiBOT19FUlJPUjsKLX0KLQotc3RhdHVzX3QgRUdMVGV4dHVyZU9iamVjdDo6cmVhbGxvY2F0ZSgKLSAgICAgICAgR0xpbnQgbGV2ZWwsIGludCB3LCBpbnQgaCwgaW50IHMsCi0gICAgICAgIGludCBmb3JtYXQsIGludCBjb21wcmVzc2VkRm9ybWF0LCBpbnQgYnByKQotewotICAgIGNvbnN0IHNpemVfdCBzaXplID0gaCAqIGJwcjsKLSAgICBpZiAobGV2ZWwgPT0gMCkgCi0gICAgewotICAgICAgICBpZiAoc2l6ZSE9bVNpemUgfHwgIXN1cmZhY2UuZGF0YSkgewotICAgICAgICAgICAgaWYgKG1TaXplICYmIHN1cmZhY2UuZGF0YSkgewotICAgICAgICAgICAgICAgIGZyZWUoc3VyZmFjZS5kYXRhKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHN1cmZhY2UuZGF0YSA9IChHR0x1Ynl0ZSopbWFsbG9jKHNpemUpOwotICAgICAgICAgICAgaWYgKCFzdXJmYWNlLmRhdGEpIHsKLSAgICAgICAgICAgICAgICBtU2l6ZSA9IDA7Ci0gICAgICAgICAgICAgICAgbUlzQ29tcGxldGUgPSBmYWxzZTsKLSAgICAgICAgICAgICAgICByZXR1cm4gTk9fTUVNT1JZOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgbVNpemUgPSBzaXplOwotICAgICAgICB9Ci0gICAgICAgIHN1cmZhY2UudmVyc2lvbiA9IHNpemVvZihHR0xTdXJmYWNlKTsKLSAgICAgICAgc3VyZmFjZS53aWR0aCAgPSB3OwotICAgICAgICBzdXJmYWNlLmhlaWdodCA9IGg7Ci0gICAgICAgIHN1cmZhY2Uuc3RyaWRlID0gczsKLSAgICAgICAgc3VyZmFjZS5mb3JtYXQgPSBmb3JtYXQ7Ci0gICAgICAgIHN1cmZhY2UuY29tcHJlc3NlZEZvcm1hdCA9IGNvbXByZXNzZWRGb3JtYXQ7Ci0gICAgICAgIGlmIChtTWlwbWFwcykKLSAgICAgICAgICAgIGZyZWVNaXBtYXBzKCk7Ci0gICAgICAgIG1Jc0NvbXBsZXRlID0gdHJ1ZTsKLSAgICB9Ci0gICAgZWxzZQotICAgIHsKLSAgICAgICAgaWYgKCFtTWlwbWFwcykgewotICAgICAgICAgICAgaWYgKGFsbG9jYXRlTWlwbWFwcygpICE9IE5PX0VSUk9SKQotICAgICAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgIH0KLQotICAgICAgICBMT0dXX0lGKGxldmVsLTEgPj0gbU51bUV4dHJhTG9kLCAKLSAgICAgICAgICAgICAgICAic3BlY2lmeWluZyBtaXBtYXAgbGV2ZWwgJWQsIGJ1dCAjIG9mIGxldmVsIGlzICVkIiwKLSAgICAgICAgICAgICAgICBsZXZlbCwgbU51bUV4dHJhTG9kKzEpOyAgICAgICAgCi0KLSAgICAgICAgR0dMU3VyZmFjZSYgbWlwbWFwID0gZWRpdE1pcChsZXZlbCk7Ci0gICAgICAgIGlmIChtaXBtYXAuZGF0YSkKLSAgICAgICAgICAgIGZyZWUobWlwbWFwLmRhdGEpOwotCi0gICAgICAgIG1pcG1hcC5kYXRhID0gKEdHTHVieXRlKiltYWxsb2Moc2l6ZSk7Ci0gICAgICAgIGlmICghbWlwbWFwLmRhdGEpIHsKLSAgICAgICAgICAgIG1lbXNldCgmbWlwbWFwLCAwLCBzaXplb2YoR0dMU3VyZmFjZSkpOwotICAgICAgICAgICAgbUlzQ29tcGxldGUgPSBmYWxzZTsKLSAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgIH0KLQotICAgICAgICBtaXBtYXAudmVyc2lvbiA9IHNpemVvZihHR0xTdXJmYWNlKTsKLSAgICAgICAgbWlwbWFwLndpZHRoICA9IHc7Ci0gICAgICAgIG1pcG1hcC5oZWlnaHQgPSBoOwotICAgICAgICBtaXBtYXAuc3RyaWRlID0gczsKLSAgICAgICAgbWlwbWFwLmZvcm1hdCA9IGZvcm1hdDsKLSAgICAgICAgbWlwbWFwLmNvbXByZXNzZWRGb3JtYXQgPSBjb21wcmVzc2VkRm9ybWF0OwotCi0gICAgICAgIC8vIGNoZWNrIGlmIHRoZSB0ZXh0dXJlIGlzIGNvbXBsZXRlCi0gICAgICAgIG1Jc0NvbXBsZXRlID0gdHJ1ZTsKLSAgICAgICAgY29uc3QgR0dMU3VyZmFjZSogcHJldiA9ICZzdXJmYWNlOwotICAgICAgICBmb3IgKGludCBpPTAgOyBpPG1OdW1FeHRyYUxvZCA7IGkrKykgewotICAgICAgICAgICAgY29uc3QgR0dMU3VyZmFjZSogY3VyciA9IG1NaXBtYXBzICsgaTsKLSAgICAgICAgICAgIGlmIChjdXJyLT5mb3JtYXQgIT0gc3VyZmFjZS5mb3JtYXQpIHsKLSAgICAgICAgICAgICAgICBtSXNDb21wbGV0ZSA9IGZhbHNlOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICB1aW50MzJfdCB3ID0gKHByZXYtPndpZHRoICA+PiAxKSA/IDogMTsKLSAgICAgICAgICAgIHVpbnQzMl90IGggPSAocHJldi0+aGVpZ2h0ID4+IDEpID8gOiAxOwotICAgICAgICAgICAgaWYgKHcgIT0gY3Vyci0+d2lkdGggfHwgaCAhPSBjdXJyLT5oZWlnaHQpIHsKLSAgICAgICAgICAgICAgICBtSXNDb21wbGV0ZSA9IGZhbHNlOwotICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgcHJldiA9IGN1cnI7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUVHTFN1cmZhY2VNYW5hZ2VyOjpFR0xTdXJmYWNlTWFuYWdlcigpCi0gICAgOiBUb2tlbk1hbmFnZXIoKSwgbUNvdW50KDApCi17Ci19Ci0KLUVHTFN1cmZhY2VNYW5hZ2VyOjp+RUdMU3VyZmFjZU1hbmFnZXIoKQotewotICAgIC8vIGV2ZXJ5dGhpbmcgZ2V0cyBmcmVlZCBhdXRvbWF0aWNhbGx5IGhlcmUuLi4KLX0KLQotc3A8RUdMVGV4dHVyZU9iamVjdD4gRUdMU3VyZmFjZU1hbmFnZXI6OmNyZWF0ZVRleHR1cmUoR0x1aW50IG5hbWUpCi17Ci0gICAgc3A8RUdMVGV4dHVyZU9iamVjdD4gcmVzdWx0OwotCi0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICBpZiAobVRleHR1cmVzLmluZGV4T2ZLZXkobmFtZSkgPj0gMCkKLSAgICAgICAgcmV0dXJuIHJlc3VsdDsgLy8gYWxyZWFkeSBleGlzdHMhCi0KLSAgICByZXN1bHQgPSBuZXcgRUdMVGV4dHVyZU9iamVjdCgpOwotCi0gICAgc3RhdHVzX3QgZXJyID0gbVRleHR1cmVzLmFkZChuYW1lLCByZXN1bHQpOwotICAgIGlmIChlcnIgPCAwKQotICAgICAgICByZXN1bHQuY2xlYXIoKTsKLQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLXNwPEVHTFRleHR1cmVPYmplY3Q+IEVHTFN1cmZhY2VNYW5hZ2VyOjpyZW1vdmVUZXh0dXJlKEdMdWludCBuYW1lKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgY29uc3Qgc3NpemVfdCBpbmRleCA9IG1UZXh0dXJlcy5pbmRleE9mS2V5KG5hbWUpOwotICAgIGlmIChpbmRleCA+PSAwKSB7Ci0gICAgICAgIHNwPEVHTFRleHR1cmVPYmplY3Q+IHJlc3VsdChtVGV4dHVyZXMudmFsdWVBdChpbmRleCkpOwotICAgICAgICBtVGV4dHVyZXMucmVtb3ZlSXRlbXNBdChpbmRleCk7Ci0gICAgICAgIHJldHVybiByZXN1bHQ7Ci0gICAgfQotICAgIHJldHVybiAwOwotfQotCi1zcDxFR0xUZXh0dXJlT2JqZWN0PiBFR0xTdXJmYWNlTWFuYWdlcjo6cmVwbGFjZVRleHR1cmUoR0x1aW50IG5hbWUpCi17Ci0gICAgc3A8RUdMVGV4dHVyZU9iamVjdD4gdGV4OwotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgY29uc3Qgc3NpemVfdCBpbmRleCA9IG1UZXh0dXJlcy5pbmRleE9mS2V5KG5hbWUpOwotICAgIGlmIChpbmRleCA+PSAwKSB7Ci0gICAgICAgIGNvbnN0IHNwPEVHTFRleHR1cmVPYmplY3Q+JiBvbGQgPSBtVGV4dHVyZXMudmFsdWVBdChpbmRleCk7Ci0gICAgICAgIGNvbnN0IHVpbnQzMl90IHJlZnMgPSBvbGQtPmdldFN0cm9uZ0NvdW50KCk7Ci0gICAgICAgIGlmIChnZ2xfbGlrZWx5KHJlZnMgPT0gMSkpIHsKLSAgICAgICAgICAgIC8vIHdlJ3JlIHRoZSBvbmx5IG93bmVyCi0gICAgICAgICAgICB0ZXggPSBvbGQ7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBrZWVwIHRoZSB0ZXh0dXJlJ3MgcGFyYW1ldGVycwotICAgICAgICAgICAgdGV4ID0gbmV3IEVHTFRleHR1cmVPYmplY3QoKTsKLSAgICAgICAgICAgIHRleC0+Y29weVBhcmFtZXRlcnMob2xkKTsKLSAgICAgICAgICAgIG1UZXh0dXJlcy5yZW1vdmVJdGVtc0F0KGluZGV4KTsKLSAgICAgICAgICAgIG1UZXh0dXJlcy5hZGQobmFtZSwgdGV4KTsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gdGV4OwotfQotCi12b2lkIEVHTFN1cmZhY2VNYW5hZ2VyOjpkZWxldGVUZXh0dXJlcyhHTHNpemVpIG4sIGNvbnN0IEdMdWludCAqdG9rZW5zKQotewotICAgIC8vIGZyZWUgYWxsIHRleHR1cmVzCi0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICBmb3IgKEdMc2l6ZWkgaT0wIDsgaTxuIDsgaSsrKSB7Ci0gICAgICAgIGNvbnN0IEdMdWludCB0KCp0b2tlbnMrKyk7Ci0gICAgICAgIGlmICh0KSB7Ci0gICAgICAgICAgICBtVGV4dHVyZXMucmVtb3ZlSXRlbSh0KTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotc3A8RUdMVGV4dHVyZU9iamVjdD4gRUdMU3VyZmFjZU1hbmFnZXI6OnRleHR1cmUoR0x1aW50IG5hbWUpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKG1Mb2NrKTsKLSAgICBjb25zdCBzc2l6ZV90IGluZGV4ID0gbVRleHR1cmVzLmluZGV4T2ZLZXkobmFtZSk7Ci0gICAgaWYgKGluZGV4ID49IDApCi0gICAgICAgIHJldHVybiBtVGV4dHVyZXMudmFsdWVBdChpbmRleCk7Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL1RleHR1cmVPYmplY3RNYW5hZ2VyLmggYi9vcGVuZ2wvbGliYWdsL1RleHR1cmVPYmplY3RNYW5hZ2VyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDc0ZWQxYTQuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9UZXh0dXJlT2JqZWN0TWFuYWdlci5oCisrKyAvZGV2L251bGwKQEAgLTEsMTQwICswLDAgQEAKLS8qCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jaWZuZGVmIEFORFJPSURfT1BFTkdMRVNfU1VSRkFDRV9ICi0jZGVmaW5lIEFORFJPSURfT1BFTkdMRVNfU1VSRkFDRV9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdGRkZWYuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotI2luY2x1ZGUgPHV0aWxzL0F0b21pYy5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLSNpbmNsdWRlIDx1dGlscy9SZWZCYXNlLmg+Ci0jaW5jbHVkZSA8dXRpbHMvS2V5ZWRWZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLQotI2luY2x1ZGUgPHByaXZhdGUvcGl4ZWxmbGluZ2VyL2dnbF9jb250ZXh0Lmg+Ci0KLSNpbmNsdWRlIDxHTEVTL2dsLmg+Ci0KLSNpbmNsdWRlICJUb2tlbml6ZXIuaCIKLSNpbmNsdWRlICJUb2tlbk1hbmFnZXIuaCIKLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY2xhc3MgRUdMVGV4dHVyZU9iamVjdAotewotcHVibGljOgotICAgICAgICAgICAgICAgICAgICBFR0xUZXh0dXJlT2JqZWN0KCk7Ci0gICAgICAgICAgICAgICAgICAgfkVHTFRleHR1cmVPYmplY3QoKTsKLQotICAgIC8vIHByb3RvY29sIGZvciBzcDw+Ci0gICAgaW5saW5lICB2b2lkICAgICAgICBpbmNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0OwotICAgIGlubGluZSAgdm9pZCAgICAgICAgZGVjU3Ryb25nKGNvbnN0IHZvaWQqIGlkKSBjb25zdDsKLSAgICBpbmxpbmUgIHVpbnQzMl90ICAgIGdldFN0cm9uZ0NvdW50KCkgY29uc3Q7Ci0KLSAgICBzdGF0dXNfdCAgICAgICAgICAgIHNldFN1cmZhY2UoR0dMU3VyZmFjZSBjb25zdCogcyk7Ci0gICAgc3RhdHVzX3QgICAgICAgICAgICByZWFsbG9jYXRlKEdMaW50IGxldmVsLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCB3LCBpbnQgaCwgaW50IHMsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IGZvcm1hdCwgaW50IGNvbXByZXNzZWRGb3JtYXQsIGludCBicHIpOwotICAgIGlubGluZSAgc2l6ZV90ICAgICAgc2l6ZSgpIGNvbnN0OwotICAgIGNvbnN0IEdHTFN1cmZhY2UmICAgbWlwKGludCBsb2QpIGNvbnN0OwotICAgIEdHTFN1cmZhY2UmICAgICAgICAgZWRpdE1pcChpbnQgbG9kKTsKLSAgICBib29sICAgICAgICAgICAgICAgIGhhc01pcG1hcHMoKSBjb25zdCB7IHJldHVybiBtTWlwbWFwcyE9MDsgfQotICAgIGJvb2wgICAgICAgICAgICAgICAgaXNDb21wbGV0ZSgpIGNvbnN0IHsgcmV0dXJuIG1Jc0NvbXBsZXRlOyB9Ci0gICAgdm9pZCAgICAgICAgICAgICAgICBjb3B5UGFyYW1ldGVycyhjb25zdCBzcDxFR0xUZXh0dXJlT2JqZWN0PiYgb2xkKTsKLQotcHJpdmF0ZToKLSAgICAgICAgc3RhdHVzX3QgICAgICAgIGFsbG9jYXRlTWlwbWFwcygpOwotICAgICAgICAgICAgdm9pZCAgICAgICAgZnJlZU1pcG1hcHMoKTsKLSAgICAgICAgICAgIHZvaWQgICAgICAgIGluaXQoKTsKLSAgICBtdXRhYmxlIGludDMyX3QgICAgIG1Db3VudDsKLSAgICBzaXplX3QgICAgICAgICAgICAgIG1TaXplOwotICAgIEdHTFN1cmZhY2UgICAgICAgICAgKm1NaXBtYXBzOwotICAgIGludCAgICAgICAgICAgICAgICAgbU51bUV4dHJhTG9kOwotICAgIGJvb2wgICAgICAgICAgICAgICAgbUlzQ29tcGxldGU7Ci0KLXB1YmxpYzoKLSAgICBHR0xTdXJmYWNlICAgICAgICAgIHN1cmZhY2U7Ci0gICAgR0xlbnVtICAgICAgICAgICAgICB3cmFwczsKLSAgICBHTGVudW0gICAgICAgICAgICAgIHdyYXB0OwotICAgIEdMZW51bSAgICAgICAgICAgICAgbWluX2ZpbHRlcjsKLSAgICBHTGVudW0gICAgICAgICAgICAgIG1hZ19maWx0ZXI7Ci0gICAgR0xlbnVtICAgICAgICAgICAgICBpbnRlcm5hbGZvcm1hdDsKLSAgICBHTGludCAgICAgICAgICAgICAgIGNyb3BfcmVjdFs0XTsKLSAgICBHTGludCAgICAgICAgICAgICAgIGdlbmVyYXRlX21pcG1hcDsKLSAgICBHTGludCAgICAgICAgICAgICAgIGRpcmVjdDsKLX07Ci0KLXZvaWQgRUdMVGV4dHVyZU9iamVjdDo6aW5jU3Ryb25nKGNvbnN0IHZvaWQqIGlkKSBjb25zdCB7Ci0gICAgYW5kcm9pZF9hdG9taWNfaW5jKCZtQ291bnQpOwotfQotdm9pZCBFR0xUZXh0dXJlT2JqZWN0OjpkZWNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0IHsKLSAgICBpZiAoYW5kcm9pZF9hdG9taWNfZGVjKCZtQ291bnQpID09IDEpIHsKLSAgICAgICAgZGVsZXRlIHRoaXM7Ci0gICAgfQotfQotdWludDMyX3QgRUdMVGV4dHVyZU9iamVjdDo6Z2V0U3Ryb25nQ291bnQoKSBjb25zdCB7Ci0gICAgcmV0dXJuIG1Db3VudDsKLX0KLXNpemVfdCBFR0xUZXh0dXJlT2JqZWN0OjpzaXplKCkgY29uc3QgewotICAgIHJldHVybiBtU2l6ZTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBFR0xTdXJmYWNlTWFuYWdlciA6IHB1YmxpYyBUb2tlbk1hbmFnZXIKLXsKLXB1YmxpYzoKLSAgICAgICAgICAgICAgICBFR0xTdXJmYWNlTWFuYWdlcigpOwotICAgICAgICAgICAgICAgIH5FR0xTdXJmYWNlTWFuYWdlcigpOwotCi0gICAgLy8gcHJvdG9jb2wgZm9yIHNwPD4KLSAgICBpbmxpbmUgIHZvaWQgICAgaW5jU3Ryb25nKGNvbnN0IHZvaWQqIGlkKSBjb25zdDsKLSAgICBpbmxpbmUgIHZvaWQgICAgZGVjU3Ryb25nKGNvbnN0IHZvaWQqIGlkKSBjb25zdDsKLSAgICB0eXBlZGVmIHZvaWQgICAgd2Vha3JlZl90eXBlOwotCi0gICAgc3A8RUdMVGV4dHVyZU9iamVjdD4gICAgY3JlYXRlVGV4dHVyZShHTHVpbnQgbmFtZSk7Ci0gICAgc3A8RUdMVGV4dHVyZU9iamVjdD4gICAgcmVtb3ZlVGV4dHVyZShHTHVpbnQgbmFtZSk7Ci0gICAgc3A8RUdMVGV4dHVyZU9iamVjdD4gICAgcmVwbGFjZVRleHR1cmUoR0x1aW50IG5hbWUpOwotICAgIHZvaWQgICAgICAgICAgICAgICAgICAgIGRlbGV0ZVRleHR1cmVzKEdMc2l6ZWkgbiwgY29uc3QgR0x1aW50ICp0b2tlbnMpOwotICAgIHNwPEVHTFRleHR1cmVPYmplY3Q+ICAgIHRleHR1cmUoR0x1aW50IG5hbWUpOwotCi1wcml2YXRlOgotICAgIG11dGFibGUgaW50MzJfdCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbUNvdW50OwotICAgIG11dGFibGUgTXV0ZXggICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbUxvY2s7Ci0gICAgS2V5ZWRWZWN0b3I8IEdMdWludCwgc3A8RUdMVGV4dHVyZU9iamVjdD4gPiBtVGV4dHVyZXM7Ci19OwotCi12b2lkIEVHTFN1cmZhY2VNYW5hZ2VyOjppbmNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0IHsKLSAgICBhbmRyb2lkX2F0b21pY19pbmMoJm1Db3VudCk7Ci19Ci12b2lkIEVHTFN1cmZhY2VNYW5hZ2VyOjpkZWNTdHJvbmcoY29uc3Qgdm9pZCogaWQpIGNvbnN0IHsKLSAgICBpZiAoYW5kcm9pZF9hdG9taWNfZGVjKCZtQ291bnQpID09IDEpIHsKLSAgICAgICAgZGVsZXRlIHRoaXM7Ci0gICAgfQotfQotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX09QRU5HTEVTX1NVUkZBQ0VfSAotCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL1Rva2VuTWFuYWdlci5jcHAgYi9vcGVuZ2wvbGliYWdsL1Rva2VuTWFuYWdlci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGVlYTYwMjUuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9Ub2tlbk1hbmFnZXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNjIgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9zdXJmYWNlLmNwcAotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgIlRva2VuTWFuYWdlci5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLVRva2VuTWFuYWdlcjo6VG9rZW5NYW5hZ2VyKCkKLXsKLSAgICAvLyB0b2tlbiAwIGlzIGFsd2F5cyByZXNlcnZlZAotICAgIG1Ub2tlbml6ZXIucmVzZXJ2ZSgwKTsKLX0KLQotVG9rZW5NYW5hZ2VyOjp+VG9rZW5NYW5hZ2VyKCkKLXsKLX0KLQotc3RhdHVzX3QgVG9rZW5NYW5hZ2VyOjpnZXRUb2tlbihHTHNpemVpIG4sIEdMdWludCAqdG9rZW5zKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgZm9yIChHTHNpemVpIGk9MCA7IGk8biA7IGkrKykKLSAgICAgICAgKnRva2VucysrID0gbVRva2VuaXplci5hY3F1aXJlKCk7Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi12b2lkIFRva2VuTWFuYWdlcjo6cmVjeWNsZVRva2VucyhHTHNpemVpIG4sIGNvbnN0IEdMdWludCAqdG9rZW5zKQotewotICAgIE11dGV4OjpBdXRvbG9jayBfbChtTG9jayk7Ci0gICAgZm9yIChpbnQgaT0wIDsgaTxuIDsgaSsrKSB7Ci0gICAgICAgIGNvbnN0IEdMdWludCB0b2tlbiA9ICp0b2tlbnMrKzsKLSAgICAgICAgaWYgKHRva2VuKSB7Ci0gICAgICAgICAgICBtVG9rZW5pemVyLnJlbGVhc2UodG9rZW4pOwotICAgICAgICB9Ci0gICAgfQotfQotCi1ib29sIFRva2VuTWFuYWdlcjo6aXNUb2tlblZhbGlkKEdMdWludCB0b2tlbikgY29uc3QKLXsKLSAgICBNdXRleDo6QXV0b2xvY2sgX2wobUxvY2spOwotICAgIHJldHVybiBtVG9rZW5pemVyLmlzQWNxdWlyZWQodG9rZW4pOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL1Rva2VuTWFuYWdlci5oIGIvb3BlbmdsL2xpYmFnbC9Ub2tlbk1hbmFnZXIuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDljMTQ2OS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGliYWdsL1Rva2VuTWFuYWdlci5oCisrKyAvZGV2L251bGwKQEAgLTEsNTMgKzAsMCBAQAotLyoKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9PUEVOR0xFU19UT0tFTl9NQU5BR0VSX0gKLSNkZWZpbmUgQU5EUk9JRF9PUEVOR0xFU19UT0tFTl9NQU5BR0VSX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN0ZGRlZi5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvdGhyZWFkcy5oPgotCi0jaW5jbHVkZSA8R0xFUy9nbC5oPgotCi0jaW5jbHVkZSAiVG9rZW5pemVyLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1jbGFzcyBUb2tlbk1hbmFnZXIKLXsKLXB1YmxpYzoKLSAgICAgICAgICAgICAgICBUb2tlbk1hbmFnZXIoKTsKLSAgICAgICAgICAgICAgICB+VG9rZW5NYW5hZ2VyKCk7Ci0KLSAgICBzdGF0dXNfdCAgICBnZXRUb2tlbihHTHNpemVpIG4sIEdMdWludCAqdG9rZW5zKTsKLSAgICB2b2lkICAgICAgICByZWN5Y2xlVG9rZW5zKEdMc2l6ZWkgbiwgY29uc3QgR0x1aW50ICp0b2tlbnMpOwotICAgIGJvb2wgICAgICAgIGlzVG9rZW5WYWxpZChHTHVpbnQgdG9rZW4pIGNvbnN0OwotCi1wcml2YXRlOgotICAgIG11dGFibGUgTXV0ZXggICBtTG9jazsKLSAgICBUb2tlbml6ZXIgICAgICAgbVRva2VuaXplcjsKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX09QRU5HTEVTX1RPS0VOX01BTkFHRVJfSAotCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL1Rva2VuaXplci5jcHAgYi9vcGVuZ2wvbGliYWdsL1Rva2VuaXplci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDliM2VhMWEuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9Ub2tlbml6ZXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTczICswLDAgQEAKLS8qIGxpYnMvb3BlbmdsZXMvVG9rZW5pemVyLmNwcAotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpbmNsdWRlIDxzdGRpby5oPgotCi0jaW5jbHVkZSAiVG9rZW5pemVyLmgiCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1BTkRST0lEX0JBU0lDX1RZUEVTX1RSQUlUUyhUb2tlbml6ZXI6OnJ1bl90KQotCi1Ub2tlbml6ZXI6OlRva2VuaXplcigpCi17Ci19Ci0KLVRva2VuaXplcjo6VG9rZW5pemVyKGNvbnN0IFRva2VuaXplciYgb3RoZXIpCi0gICAgOiBtUmFuZ2VzKG90aGVyLm1SYW5nZXMpCi17Ci19Ci0KLVRva2VuaXplcjo6flRva2VuaXplcigpCi17Ci19Ci0KLXVpbnQzMl90IFRva2VuaXplcjo6YWNxdWlyZSgpCi17Ci0gICAgaWYgKCFtUmFuZ2VzLnNpemUoKSB8fCBtUmFuZ2VzWzBdLmZpcnN0KSB7Ci0gICAgICAgIF9pbnNlcnRUb2tlbkF0KDAsMCk7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLSAgICAKLSAgICAvLyBqdXN0IGV4dGVuZCB0aGUgZmlyc3QgcnVuCi0gICAgY29uc3QgcnVuX3QmIHJ1biA9IG1SYW5nZXNbMF07Ci0gICAgdWludDMyX3QgdG9rZW4gPSBydW4uZmlyc3QgKyBydW4ubGVuZ3RoOwotICAgIF9pbnNlcnRUb2tlbkF0KHRva2VuLCAxKTsKLSAgICByZXR1cm4gdG9rZW47Ci19Ci0KLWJvb2wgVG9rZW5pemVyOjppc0FjcXVpcmVkKHVpbnQzMl90IHRva2VuKSBjb25zdAotewotICAgIHJldHVybiAoX2luZGV4T3JkZXJPZih0b2tlbikgPj0gMCk7Ci19Ci0KLXN0YXR1c190IFRva2VuaXplcjo6cmVzZXJ2ZSh1aW50MzJfdCB0b2tlbikKLXsKLSAgICBzaXplX3QgbzsKLSAgICBjb25zdCBzc2l6ZV90IGkgPSBfaW5kZXhPcmRlck9mKHRva2VuLCAmbyk7Ci0gICAgaWYgKGkgPj0gMCkgewotICAgICAgICByZXR1cm4gQkFEX1ZBTFVFOyAvLyB0aGlzIHRva2VuIGlzIGFscmVhZHkgdGFrZW4KLSAgICB9Ci0gICAgc3NpemVfdCBlcnIgPSBfaW5zZXJ0VG9rZW5BdCh0b2tlbiwgbyk7Ci0gICAgcmV0dXJuIChlcnI8MCkgPyBlcnIgOiBzdGF0dXNfdChOT19FUlJPUik7Ci19Ci0KLXN0YXR1c190IFRva2VuaXplcjo6cmVsZWFzZSh1aW50MzJfdCB0b2tlbikKLXsKLSAgICBjb25zdCBzc2l6ZV90IGkgPSBfaW5kZXhPcmRlck9mKHRva2VuKTsKLSAgICBpZiAoaSA+PSAwKSB7Ci0gICAgICAgIGNvbnN0IHJ1bl90JiBydW4gPSBtUmFuZ2VzW2ldOwotICAgICAgICBpZiAoKHRva2VuID49IHJ1bi5maXJzdCkgJiYgKHRva2VuIDwgcnVuLmZpcnN0K3J1bi5sZW5ndGgpKSB7Ci0gICAgICAgICAgICAvLyB0b2tlbiBpbiB0aGlzIHJhbmdlLCB3ZSBuZWVkIHRvIHNwbGl0Ci0gICAgICAgICAgICBydW5fdCYgcnVuID0gbVJhbmdlcy5lZGl0SXRlbUF0KGkpOwotICAgICAgICAgICAgaWYgKCh0b2tlbiA9PSBydW4uZmlyc3QpIHx8ICh0b2tlbiA9PSBydW4uZmlyc3QrcnVuLmxlbmd0aC0xKSkgewotICAgICAgICAgICAgICAgIGlmICh0b2tlbiA9PSBydW4uZmlyc3QpIHsKLSAgICAgICAgICAgICAgICAgICAgcnVuLmZpcnN0ICs9IDE7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHJ1bi5sZW5ndGggLT0gMTsKLSAgICAgICAgICAgICAgICBpZiAocnVuLmxlbmd0aCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIFhYWDogc2hvdWxkIHdlIHN5c3RlbWF0aWNhbGx5IHJlbW92ZSBhIHJ1biB0aGF0J3MgZW1wdHk/Ci0gICAgICAgICAgICAgICAgICAgIG1SYW5nZXMucmVtb3ZlSXRlbXNBdChpKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIC8vIHNwbGl0IHRoZSBydW4KLSAgICAgICAgICAgICAgICBydW5fdCBuZXdfcnVuOwotICAgICAgICAgICAgICAgIG5ld19ydW4uZmlyc3QgPSB0b2tlbisxOwotICAgICAgICAgICAgICAgIG5ld19ydW4ubGVuZ3RoID0gcnVuLmZpcnN0K3J1bi5sZW5ndGggLSBuZXdfcnVuLmZpcnN0OwotICAgICAgICAgICAgICAgIHJ1bi5sZW5ndGggPSB0b2tlbiAtIHJ1bi5maXJzdDsKLSAgICAgICAgICAgICAgICBtUmFuZ2VzLmluc2VydEF0KG5ld19ydW4sIGkrMSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIE5BTUVfTk9UX0ZPVU5EOwotfQotCi1zc2l6ZV90IFRva2VuaXplcjo6X2luZGV4T3JkZXJPZih1aW50MzJfdCB0b2tlbiwgc2l6ZV90KiBvcmRlcikgY29uc3QKLXsKLSAgICAvLyBiaW5hcnkgc2VhcmNoCi0gICAgc3NpemVfdCBlcnIgPSBOQU1FX05PVF9GT1VORDsKLSAgICBzc2l6ZV90IGwgPSAwOwotICAgIHNzaXplX3QgaCA9IG1SYW5nZXMuc2l6ZSgpLTE7Ci0gICAgc3NpemVfdCBtaWQ7Ci0gICAgY29uc3QgcnVuX3QqIGEgPSBtUmFuZ2VzLmFycmF5KCk7Ci0gICAgd2hpbGUgKGwgPD0gaCkgewotICAgICAgICBtaWQgPSBsICsgKGggLSBsKS8yOwotICAgICAgICBjb25zdCBydW5fdCogY29uc3QgY3VyciA9IGEgKyBtaWQ7Ci0gICAgICAgIGludCBjID0gMDsKLSAgICAgICAgaWYgKHRva2VuIDwgY3Vyci0+Zmlyc3QpICAgICAgICAgICAgICAgICAgICAgICAgYyA9IDE7Ci0gICAgICAgIGVsc2UgaWYgKHRva2VuID49IGN1cnItPmZpcnN0K2N1cnItPmxlbmd0aCkgICAgIGMgPSAtMTsKLSAgICAgICAgaWYgKGMgPT0gMCkgewotICAgICAgICAgICAgZXJyID0gbCA9IG1pZDsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9IGVsc2UgaWYgKGMgPCAwKSB7Ci0gICAgICAgICAgICBsID0gbWlkICsgMTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGggPSBtaWQgLSAxOwotICAgICAgICB9Ci0gICAgfQotICAgIGlmIChvcmRlcikgKm9yZGVyID0gbDsKLSAgICByZXR1cm4gZXJyOwotfQotCi1zc2l6ZV90IFRva2VuaXplcjo6X2luc2VydFRva2VuQXQodWludDMyX3QgdG9rZW4sIHNpemVfdCBpbmRleCkKLXsKLSAgICBjb25zdCBzaXplX3QgYyA9IG1SYW5nZXMuc2l6ZSgpOwotCi0gICAgaWYgKGluZGV4ID49IDEpIHsKLSAgICAgICAgLy8gZG8gd2UgbmVlZCB0byBtZXJnZSB3aXRoIHRoZSBwcmV2aW91cyBydW4/Ci0gICAgICAgIHJ1bl90JiBwID0gbVJhbmdlcy5lZGl0SXRlbUF0KGluZGV4LTEpOwotICAgICAgICBpZiAocC5maXJzdCtwLmxlbmd0aCA9PSB0b2tlbikgewotICAgICAgICAgICAgcC5sZW5ndGggKz0gMTsKLSAgICAgICAgICAgIGlmIChpbmRleCA8IGMpIHsKLSAgICAgICAgICAgICAgICBjb25zdCBydW5fdCYgbiA9IG1SYW5nZXNbaW5kZXhdOwotICAgICAgICAgICAgICAgIGlmICh0b2tlbisxID09IG4uZmlyc3QpIHsKLSAgICAgICAgICAgICAgICAgICAgcC5sZW5ndGggKz0gbi5sZW5ndGg7Ci0gICAgICAgICAgICAgICAgICAgIG1SYW5nZXMucmVtb3ZlSXRlbXNBdChpbmRleCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgcmV0dXJuIGluZGV4OwotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIGlmIChpbmRleCA8IGMpIHsKLSAgICAgICAgLy8gZG8gd2UgbmVlZCB0byBtZXJnZSB3aXRoIHRoZSBuZXh0IHJ1bj8KLSAgICAgICAgcnVuX3QmIG4gPSBtUmFuZ2VzLmVkaXRJdGVtQXQoaW5kZXgpOwotICAgICAgICBpZiAodG9rZW4rMSA9PSBuLmZpcnN0KSB7Ci0gICAgICAgICAgICBuLmZpcnN0IC09IDE7Ci0gICAgICAgICAgICBuLmxlbmd0aCArPSAxOwotICAgICAgICAgICAgcmV0dXJuIGluZGV4OwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcmV0dXJuIG1SYW5nZXMuaW5zZXJ0QXQocnVuX3QodG9rZW4sMSksIGluZGV4KTsKLX0KLQotdm9pZCBUb2tlbml6ZXI6OmR1bXAoKSBjb25zdAotewotICAgIGNvbnN0IHJ1bl90KiByYW5nZXMgPSBtUmFuZ2VzLmFycmF5KCk7Ci0gICAgY29uc3Qgc2l6ZV90IGMgPSBtUmFuZ2VzLnNpemUoKTsKLSAgICBMT0dEKCJUb2tlbml6ZXIgKCVwLCBzaXplID0gJXUpXG4iLCB0aGlzLCBjKTsKLSAgICBmb3IgKHNpemVfdCBpPTAgOyBpPGMgOyBpKyspIHsKLSAgICAgICAgTE9HRCgiJXU6ICgldSwgJXUpXG4iLCBpLCByYW5nZXNbaV0uZmlyc3QsIHJhbmdlc1tpXS5sZW5ndGgpOwotICAgIH0KLX0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC9Ub2tlbml6ZXIuaCBiL29wZW5nbC9saWJhZ2wvVG9rZW5pemVyLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGFjNTU1Y2IuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9Ub2tlbml6ZXIuaAorKysgL2Rldi9udWxsCkBAIC0xLDU5ICswLDAgQEAKLS8qIGxpYnMvb3BlbmdsZXMvVG9rZW5pemVyLmgKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0KLSNpZm5kZWYgQU5EUk9JRF9PUEVOR0xFU19UT0tFTklaRVJfSAotI2RlZmluZSBBTkRST0lEX09QRU5HTEVTX1RPS0VOSVpFUl9ICi0KLSNpbmNsdWRlIDx1dGlscy9WZWN0b3IuaD4KLSNpbmNsdWRlIDx1dGlscy9FcnJvcnMuaD4KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLWNsYXNzIFRva2VuaXplcgotewotcHVibGljOgotICAgICAgICAgICAgICAgIFRva2VuaXplcigpOwotICAgICAgICAgICAgICAgIFRva2VuaXplcihjb25zdCBUb2tlbml6ZXImIG90aGVyKTsKLSAgICAgICAgICAgICAgICB+VG9rZW5pemVyKCk7Ci0KLSAgICB1aW50MzJfdCAgICBhY3F1aXJlKCk7Ci0gICAgc3RhdHVzX3QgICAgcmVzZXJ2ZSh1aW50MzJfdCB0b2tlbik7Ci0gICAgc3RhdHVzX3QgICAgcmVsZWFzZSh1aW50MzJfdCB0b2tlbik7Ci0gICAgYm9vbCAgICAgICAgaXNBY3F1aXJlZCh1aW50MzJfdCB0b2tlbikgY29uc3Q7Ci0KLSAgICB2b2lkIGR1bXAoKSBjb25zdDsKLQotICAgIHN0cnVjdCBydW5fdCB7Ci0gICAgICAgIHJ1bl90KCkge307Ci0gICAgICAgIHJ1bl90KHVpbnQzMl90IGYsIHVpbnQzMl90IGwpIDogZmlyc3QoZiksIGxlbmd0aChsKSB7fQotICAgICAgICB1aW50MzJfdCAgICBmaXJzdDsKLSAgICAgICAgdWludDMyX3QgICAgbGVuZ3RoOwotICAgIH07Ci1wcml2YXRlOgotICAgIHNzaXplX3QgX2luZGV4T3JkZXJPZih1aW50MzJfdCB0b2tlbiwgc2l6ZV90KiBvcmRlcj0wKSBjb25zdDsKLSAgICBzc2l6ZV90IF9pbnNlcnRUb2tlbkF0KHVpbnQzMl90IHRva2VuLCBzaXplX3QgaW5kZXgpOwotICAgIFZlY3RvcjxydW5fdD4gICBtUmFuZ2VzOwotfTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLy8gQU5EUk9JRF9PUEVOR0xFU19UT0tFTklaRVJfSApkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC9hcnJheS5jcHAgYi9vcGVuZ2wvbGliYWdsL2FycmF5LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggOGZhNzU2Ni4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGliYWdsL2FycmF5LmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDE1NTcgKzAsMCBAQAotLyogCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLQotI2luY2x1ZGUgImNvbnRleHQuaCIKLSNpbmNsdWRlICJmcC5oIgotI2luY2x1ZGUgInN0YXRlLmgiCi0jaW5jbHVkZSAibWF0cml4LmgiCi0jaW5jbHVkZSAidmVydGV4LmgiCi0jaW5jbHVkZSAibGlnaHQuaCIKLSNpbmNsdWRlICJwcmltaXRpdmVzLmgiCi0jaW5jbHVkZSAidGV4dHVyZS5oIgotI2luY2x1ZGUgIkJ1ZmZlck9iamVjdE1hbmFnZXIuaCIKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZGVmaW5lIFZDX0NBQ0hFX1NUQVRJU1RJQ1MgICAgIDAKLSNkZWZpbmUgVkNfQ0FDSEVfVFlQRV9OT05FICAgICAgMAotI2RlZmluZSBWQ19DQUNIRV9UWVBFX0lOREVYRUQgICAxCi0jZGVmaW5lIFZDX0NBQ0hFX1RZUEVfTFJVICAgICAgIDIKLSNkZWZpbmUgVkNfQ0FDSEVfVFlQRSAgICAgICAgICAgVkNfQ0FDSEVfVFlQRV9JTkRFWEVECi0KLSNpZiBWQ19DQUNIRV9TVEFUSVNUSUNTCi0jaW5jbHVkZSA8dXRpbHMvVGltZXJzLmg+Ci0jZW5kaWYKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLXN0YXRpYyB2b2lkIHZhbGlkYXRlX2FycmF5cyhvZ2xlc19jb250ZXh0X3QqIGMsIEdMZW51bSBtb2RlKTsKLQotc3RhdGljIHZvaWQgY29tcGlsZUVsZW1lbnRzX19nZW5lcmljKG9nbGVzX2NvbnRleHRfdCosCi0gICAgICAgIHZlcnRleF90KiwgR0xpbnQsIEdMc2l6ZWkpOwotc3RhdGljIHZvaWQgY29tcGlsZUVsZW1lbnRfX2dlbmVyaWMob2dsZXNfY29udGV4dF90KiwKLSAgICAgICAgdmVydGV4X3QqLCBHTGludCk7Ci0KLXN0YXRpYyB2b2lkIGRyYXdQcmltaXRpdmVzUG9pbnRzKG9nbGVzX2NvbnRleHRfdCosIEdMaW50LCBHTHNpemVpKTsKLXN0YXRpYyB2b2lkIGRyYXdQcmltaXRpdmVzTGluZVN0cmlwKG9nbGVzX2NvbnRleHRfdCosIEdMaW50LCBHTHNpemVpKTsKLXN0YXRpYyB2b2lkIGRyYXdQcmltaXRpdmVzTGluZUxvb3Aob2dsZXNfY29udGV4dF90KiwgR0xpbnQsIEdMc2l6ZWkpOwotc3RhdGljIHZvaWQgZHJhd1ByaW1pdGl2ZXNMaW5lcyhvZ2xlc19jb250ZXh0X3QqLCBHTGludCwgR0xzaXplaSk7Ci1zdGF0aWMgdm9pZCBkcmF3UHJpbWl0aXZlc1RyaWFuZ2xlU3RyaXAob2dsZXNfY29udGV4dF90KiwgR0xpbnQsIEdMc2l6ZWkpOwotc3RhdGljIHZvaWQgZHJhd1ByaW1pdGl2ZXNUcmlhbmdsZUZhbihvZ2xlc19jb250ZXh0X3QqLCBHTGludCwgR0xzaXplaSk7Ci1zdGF0aWMgdm9pZCBkcmF3UHJpbWl0aXZlc1RyaWFuZ2xlcyhvZ2xlc19jb250ZXh0X3QqLCBHTGludCwgR0xzaXplaSk7Ci0KLXN0YXRpYyB2b2lkIGRyYXdJbmRleGVkUHJpbWl0aXZlc1BvaW50cyhvZ2xlc19jb250ZXh0X3QqLAotICAgICAgICBHTHNpemVpLCBjb25zdCBHTHZvaWQqKTsKLXN0YXRpYyB2b2lkIGRyYXdJbmRleGVkUHJpbWl0aXZlc0xpbmVTdHJpcChvZ2xlc19jb250ZXh0X3QqLAotICAgICAgICBHTHNpemVpLCBjb25zdCBHTHZvaWQqKTsKLXN0YXRpYyB2b2lkIGRyYXdJbmRleGVkUHJpbWl0aXZlc0xpbmVMb29wKG9nbGVzX2NvbnRleHRfdCosCi0gICAgICAgIEdMc2l6ZWksIGNvbnN0IEdMdm9pZCopOwotc3RhdGljIHZvaWQgZHJhd0luZGV4ZWRQcmltaXRpdmVzTGluZXMob2dsZXNfY29udGV4dF90KiwKLSAgICAgICAgR0xzaXplaSwgY29uc3QgR0x2b2lkKik7Ci1zdGF0aWMgdm9pZCBkcmF3SW5kZXhlZFByaW1pdGl2ZXNUcmlhbmdsZVN0cmlwKG9nbGVzX2NvbnRleHRfdCosCi0gICAgICAgIEdMc2l6ZWksIGNvbnN0IEdMdm9pZCopOwotc3RhdGljIHZvaWQgZHJhd0luZGV4ZWRQcmltaXRpdmVzVHJpYW5nbGVGYW4ob2dsZXNfY29udGV4dF90KiwKLSAgICAgICAgR0xzaXplaSwgY29uc3QgR0x2b2lkKik7Ci1zdGF0aWMgdm9pZCBkcmF3SW5kZXhlZFByaW1pdGl2ZXNUcmlhbmdsZXMob2dsZXNfY29udGV4dF90KiwKLSAgICAgICAgR0xzaXplaSwgY29uc3QgR0x2b2lkKik7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdHlwZWRlZiB2b2lkICgqYXJyYXlzX3ByaW1zX2ZjdF90KShvZ2xlc19jb250ZXh0X3QqLCBHTGludCwgR0xzaXplaSk7Ci1zdGF0aWMgY29uc3QgYXJyYXlzX3ByaW1zX2ZjdF90IGRyYXdBcnJheXNQcmltc1tdID0gewotICAgIGRyYXdQcmltaXRpdmVzUG9pbnRzLAotICAgIGRyYXdQcmltaXRpdmVzTGluZXMsCi0gICAgZHJhd1ByaW1pdGl2ZXNMaW5lTG9vcCwKLSAgICBkcmF3UHJpbWl0aXZlc0xpbmVTdHJpcCwKLSAgICBkcmF3UHJpbWl0aXZlc1RyaWFuZ2xlcywKLSAgICBkcmF3UHJpbWl0aXZlc1RyaWFuZ2xlU3RyaXAsCi0gICAgZHJhd1ByaW1pdGl2ZXNUcmlhbmdsZUZhbgotfTsKLQotdHlwZWRlZiB2b2lkICgqZWxlbWVudHNfcHJpbXNfZmN0X3QpKG9nbGVzX2NvbnRleHRfdCosIEdMc2l6ZWksIGNvbnN0IEdMdm9pZCopOwotc3RhdGljIGNvbnN0IGVsZW1lbnRzX3ByaW1zX2ZjdF90IGRyYXdFbGVtZW50c1ByaW1zW10gPSB7Ci0gICAgZHJhd0luZGV4ZWRQcmltaXRpdmVzUG9pbnRzLAotICAgIGRyYXdJbmRleGVkUHJpbWl0aXZlc0xpbmVzLAotICAgIGRyYXdJbmRleGVkUHJpbWl0aXZlc0xpbmVMb29wLAotICAgIGRyYXdJbmRleGVkUHJpbWl0aXZlc0xpbmVTdHJpcCwKLSAgICBkcmF3SW5kZXhlZFByaW1pdGl2ZXNUcmlhbmdsZXMsCi0gICAgZHJhd0luZGV4ZWRQcmltaXRpdmVzVHJpYW5nbGVTdHJpcCwKLSAgICBkcmF3SW5kZXhlZFByaW1pdGl2ZXNUcmlhbmdsZUZhbgotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jZW5kaWYKLQotdm9pZCBvZ2xlc19pbml0X2FycmF5KG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBjLT5hcnJheXMudmVydGV4LnNpemUgPSA0OwotICAgIGMtPmFycmF5cy52ZXJ0ZXgudHlwZSA9IEdMX0ZMT0FUOwotICAgIGMtPmFycmF5cy5jb2xvci5zaXplID0gNDsKLSAgICBjLT5hcnJheXMuY29sb3IudHlwZSA9IEdMX0ZMT0FUOwotICAgIGMtPmFycmF5cy5ub3JtYWwuc2l6ZSA9IDQ7Ci0gICAgYy0+YXJyYXlzLm5vcm1hbC50eXBlID0gR0xfRkxPQVQ7Ci0gICAgZm9yIChpbnQgaT0wIDsgaTxHR0xfVEVYVFVSRV9VTklUX0NPVU5UIDsgaSsrKSB7Ci0gICAgICAgIGMtPmFycmF5cy50ZXh0dXJlW2ldLnNpemUgPSA0OwotICAgICAgICBjLT5hcnJheXMudGV4dHVyZVtpXS50eXBlID0gR0xfRkxPQVQ7Ci0gICAgfQotICAgIGMtPnZjLmluaXQoKTsKLQotICAgIGlmICghYy0+dmMudkJ1ZmZlcikgewotICAgICAgICAvLyB0aGlzIGNvdWxkIGhhdmUgZmFpbGVkCi0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX09VVF9PRl9NRU1PUlkpOwotICAgIH0KLX0KLQotdm9pZCBvZ2xlc191bmluaXRfYXJyYXkob2dsZXNfY29udGV4dF90KiBjKQotewotICAgIGMtPnZjLnVuaW5pdCgpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBBcnJheSBmZXRjaGVycwotI2VuZGlmCi0KLXN0YXRpYyB2b2lkIGN1cnJlbnRDb2xvcihvZ2xlc19jb250ZXh0X3QqIGMsIEdMZml4ZWQqIHYsIGNvbnN0IEdMdm9pZCopIHsKLSAgICBtZW1jcHkodiwgYy0+Y3VycmVudC5jb2xvci52LCBzaXplb2YodmVjNF90KSk7Ci19Ci1zdGF0aWMgdm9pZCBjdXJyZW50Q29sb3JfY2xhbXAob2dsZXNfY29udGV4dF90KiBjLCBHTGZpeGVkKiB2LCBjb25zdCBHTHZvaWQqKSB7Ci0gICAgbWVtY3B5KHYsIGMtPmN1cnJlbnRDb2xvckNsYW1wZWQudiwgc2l6ZW9mKHZlYzRfdCkpOwotfQotc3RhdGljIHZvaWQgY3VycmVudE5vcm1hbChvZ2xlc19jb250ZXh0X3QqIGMsIEdMZml4ZWQqIHYsIGNvbnN0IEdMdm9pZCopIHsKLSAgICBtZW1jcHkodiwgYy0+Y3VycmVudE5vcm1hbC52LCBzaXplb2YodmVjM190KSk7Ci19Ci1zdGF0aWMgdm9pZCBjdXJyZW50VGV4Q29vcmQob2dsZXNfY29udGV4dF90KiBjLCBHTGZpeGVkKiB2LCBjb25zdCBHTHZvaWQqKSB7Ci0gICAgbWVtY3B5KHYsIGMtPmN1cnJlbnQudGV4dHVyZVtjLT5hcnJheXMudG11XS52LCBzaXplb2YodmVjNF90KSk7Ci19Ci0KLQotc3RhdGljIHZvaWQgZmV0Y2hOb3Aob2dsZXNfY29udGV4dF90KiwgR0xmaXhlZCosIGNvbnN0IEdMdm9pZCopIHsKLX0KLXN0YXRpYyB2b2lkIGZldGNoMmIob2dsZXNfY29udGV4dF90KiwgR0xmaXhlZCogdiwgY29uc3QgR0xieXRlKiBwKSB7Ci0gICAgdlswXSA9IGdnbEludFRvRml4ZWQocFswXSk7Ci0gICAgdlsxXSA9IGdnbEludFRvRml4ZWQocFsxXSk7Ci19Ci1zdGF0aWMgdm9pZCBmZXRjaDJzKG9nbGVzX2NvbnRleHRfdCosIEdMZml4ZWQqIHYsIGNvbnN0IEdMc2hvcnQqIHApIHsKLSAgICB2WzBdID0gZ2dsSW50VG9GaXhlZChwWzBdKTsKLSAgICB2WzFdID0gZ2dsSW50VG9GaXhlZChwWzFdKTsKLX0KLXN0YXRpYyB2b2lkIGZldGNoMngob2dsZXNfY29udGV4dF90KiwgR0xmaXhlZCogdiwgY29uc3QgR0xmaXhlZCogcCkgewotICAgIG1lbWNweSh2LCBwLCAyKnNpemVvZihHTGZpeGVkKSk7Ci19Ci1zdGF0aWMgdm9pZCBmZXRjaDJmKG9nbGVzX2NvbnRleHRfdCosIEdMZml4ZWQqIHYsIGNvbnN0IEdMZmxvYXQqIHApIHsKLSAgICB2WzBdID0gZ2dsRmxvYXRUb0ZpeGVkKHBbMF0pOwotICAgIHZbMV0gPSBnZ2xGbG9hdFRvRml4ZWQocFsxXSk7Ci19Ci1zdGF0aWMgdm9pZCBmZXRjaDNiKG9nbGVzX2NvbnRleHRfdCosIEdMZml4ZWQqIHYsIGNvbnN0IEdMYnl0ZSogcCkgewotICAgIHZbMF0gPSBnZ2xJbnRUb0ZpeGVkKHBbMF0pOwotICAgIHZbMV0gPSBnZ2xJbnRUb0ZpeGVkKHBbMV0pOwotICAgIHZbMl0gPSBnZ2xJbnRUb0ZpeGVkKHBbMl0pOwotfQotc3RhdGljIHZvaWQgZmV0Y2gzcyhvZ2xlc19jb250ZXh0X3QqLCBHTGZpeGVkKiB2LCBjb25zdCBHTHNob3J0KiBwKSB7Ci0gICAgdlswXSA9IGdnbEludFRvRml4ZWQocFswXSk7Ci0gICAgdlsxXSA9IGdnbEludFRvRml4ZWQocFsxXSk7Ci0gICAgdlsyXSA9IGdnbEludFRvRml4ZWQocFsyXSk7Ci19Ci1zdGF0aWMgdm9pZCBmZXRjaDN4KG9nbGVzX2NvbnRleHRfdCosIEdMZml4ZWQqIHYsIGNvbnN0IEdMZml4ZWQqIHApIHsKLSAgICBtZW1jcHkodiwgcCwgMypzaXplb2YoR0xmaXhlZCkpOwotfQotc3RhdGljIHZvaWQgZmV0Y2gzZihvZ2xlc19jb250ZXh0X3QqLCBHTGZpeGVkKiB2LCBjb25zdCBHTGZsb2F0KiBwKSB7Ci0gICAgdlswXSA9IGdnbEZsb2F0VG9GaXhlZChwWzBdKTsKLSAgICB2WzFdID0gZ2dsRmxvYXRUb0ZpeGVkKHBbMV0pOwotICAgIHZbMl0gPSBnZ2xGbG9hdFRvRml4ZWQocFsyXSk7Ci19Ci1zdGF0aWMgdm9pZCBmZXRjaDRiKG9nbGVzX2NvbnRleHRfdCosIEdMZml4ZWQqIHYsIGNvbnN0IEdMYnl0ZSogcCkgewotICAgIHZbMF0gPSBnZ2xJbnRUb0ZpeGVkKHBbMF0pOwotICAgIHZbMV0gPSBnZ2xJbnRUb0ZpeGVkKHBbMV0pOwotICAgIHZbMl0gPSBnZ2xJbnRUb0ZpeGVkKHBbMl0pOwotICAgIHZbM10gPSBnZ2xJbnRUb0ZpeGVkKHBbM10pOwotfQotc3RhdGljIHZvaWQgZmV0Y2g0cyhvZ2xlc19jb250ZXh0X3QqLCBHTGZpeGVkKiB2LCBjb25zdCBHTHNob3J0KiBwKSB7Ci0gICAgdlswXSA9IGdnbEludFRvRml4ZWQocFswXSk7Ci0gICAgdlsxXSA9IGdnbEludFRvRml4ZWQocFsxXSk7Ci0gICAgdlsyXSA9IGdnbEludFRvRml4ZWQocFsyXSk7Ci0gICAgdlszXSA9IGdnbEludFRvRml4ZWQocFszXSk7Ci19Ci1zdGF0aWMgdm9pZCBmZXRjaDR4KG9nbGVzX2NvbnRleHRfdCosIEdMZml4ZWQqIHYsIGNvbnN0IEdMZml4ZWQqIHApIHsKLSAgICBtZW1jcHkodiwgcCwgNCpzaXplb2YoR0xmaXhlZCkpOwotfQotc3RhdGljIHZvaWQgZmV0Y2g0ZihvZ2xlc19jb250ZXh0X3QqLCBHTGZpeGVkKiB2LCBjb25zdCBHTGZsb2F0KiBwKSB7Ci0gICAgdlswXSA9IGdnbEZsb2F0VG9GaXhlZChwWzBdKTsKLSAgICB2WzFdID0gZ2dsRmxvYXRUb0ZpeGVkKHBbMV0pOwotICAgIHZbMl0gPSBnZ2xGbG9hdFRvRml4ZWQocFsyXSk7Ci0gICAgdlszXSA9IGdnbEZsb2F0VG9GaXhlZChwWzNdKTsKLX0KLXN0YXRpYyB2b2lkIGZldGNoRXhwYW5kNHViKG9nbGVzX2NvbnRleHRfdCosIEdMZml4ZWQqIHYsIGNvbnN0IEdMdWJ5dGUqIHApIHsKLSAgICB2WzBdID0gR0dMX1VCX1RPX1gocFswXSk7Ci0gICAgdlsxXSA9IEdHTF9VQl9UT19YKHBbMV0pOwotICAgIHZbMl0gPSBHR0xfVUJfVE9fWChwWzJdKTsKLSAgICB2WzNdID0gR0dMX1VCX1RPX1gocFszXSk7Ci19Ci1zdGF0aWMgdm9pZCBmZXRjaENsYW1wNHgob2dsZXNfY29udGV4dF90KiwgR0xmaXhlZCogdiwgY29uc3QgR0xmaXhlZCogcCkgewotICAgIHZbMF0gPSBnZ2xDbGFtcHgocFswXSk7Ci0gICAgdlsxXSA9IGdnbENsYW1weChwWzFdKTsKLSAgICB2WzJdID0gZ2dsQ2xhbXB4KHBbMl0pOwotICAgIHZbM10gPSBnZ2xDbGFtcHgocFszXSk7Ci19Ci1zdGF0aWMgdm9pZCBmZXRjaENsYW1wNGYob2dsZXNfY29udGV4dF90KiwgR0xmaXhlZCogdiwgY29uc3QgR0xmbG9hdCogcCkgewotICAgIHZbMF0gPSBnZ2xDbGFtcHgoZ2dsRmxvYXRUb0ZpeGVkKHBbMF0pKTsKLSAgICB2WzFdID0gZ2dsQ2xhbXB4KGdnbEZsb2F0VG9GaXhlZChwWzFdKSk7Ci0gICAgdlsyXSA9IGdnbENsYW1weChnZ2xGbG9hdFRvRml4ZWQocFsyXSkpOwotICAgIHZbM10gPSBnZ2xDbGFtcHgoZ2dsRmxvYXRUb0ZpeGVkKHBbM10pKTsKLX0KLXN0YXRpYyB2b2lkIGZldGNoRXhwYW5kM3ViKG9nbGVzX2NvbnRleHRfdCosIEdMZml4ZWQqIHYsIGNvbnN0IEdMdWJ5dGUqIHApIHsKLSAgICB2WzBdID0gR0dMX1VCX1RPX1gocFswXSk7Ci0gICAgdlsxXSA9IEdHTF9VQl9UT19YKHBbMV0pOwotICAgIHZbMl0gPSBHR0xfVUJfVE9fWChwWzJdKTsKLSAgICB2WzNdID0gMHgxMDAwMDsKLX0KLXN0YXRpYyB2b2lkIGZldGNoQ2xhbXAzeChvZ2xlc19jb250ZXh0X3QqLCBHTGZpeGVkKiB2LCBjb25zdCBHTGZpeGVkKiBwKSB7Ci0gICAgdlswXSA9IGdnbENsYW1weChwWzBdKTsKLSAgICB2WzFdID0gZ2dsQ2xhbXB4KHBbMV0pOwotICAgIHZbMl0gPSBnZ2xDbGFtcHgocFsyXSk7Ci0gICAgdlszXSA9IDB4MTAwMDA7Ci19Ci1zdGF0aWMgdm9pZCBmZXRjaENsYW1wM2Yob2dsZXNfY29udGV4dF90KiwgR0xmaXhlZCogdiwgY29uc3QgR0xmbG9hdCogcCkgewotICAgIHZbMF0gPSBnZ2xDbGFtcHgoZ2dsRmxvYXRUb0ZpeGVkKHBbMF0pKTsKLSAgICB2WzFdID0gZ2dsQ2xhbXB4KGdnbEZsb2F0VG9GaXhlZChwWzFdKSk7Ci0gICAgdlsyXSA9IGdnbENsYW1weChnZ2xGbG9hdFRvRml4ZWQocFsyXSkpOwotICAgIHZbM10gPSAweDEwMDAwOwotfQotc3RhdGljIHZvaWQgZmV0Y2hFeHBhbmQzYihvZ2xlc19jb250ZXh0X3QqLCBHTGZpeGVkKiB2LCBjb25zdCBHTGJ5dGUqIHApIHsKLSAgICB2WzBdID0gR0dMX0JfVE9fWChwWzBdKTsKLSAgICB2WzFdID0gR0dMX0JfVE9fWChwWzFdKTsKLSAgICB2WzJdID0gR0dMX0JfVE9fWChwWzJdKTsKLX0KLXN0YXRpYyB2b2lkIGZldGNoRXhwYW5kM3Mob2dsZXNfY29udGV4dF90KiwgR0xmaXhlZCogdiwgY29uc3QgR0xzaG9ydCogcCkgewotICAgIHZbMF0gPSBHR0xfU19UT19YKHBbMF0pOwotICAgIHZbMV0gPSBHR0xfU19UT19YKHBbMV0pOwotICAgIHZbMl0gPSBHR0xfU19UT19YKHBbMl0pOwotfQotCi10eXBlZGVmIGFycmF5X3Q6OmZldGNoZXJfdCBmbl90OyAKLQotc3RhdGljIGNvbnN0IGZuX3QgY29sb3JfZmN0WzJdWzE2XSA9IHsgLy8gc2l6ZT17Myw0fSwgdHlwZT17dWIsZix4fQotICAgIHsgMCwgKGZuX3QpZmV0Y2hFeHBhbmQzdWIsIDAsIDAsIDAsIDAsCi0gICAgICAgICAoZm5fdClmZXRjaDNmLCAwLCAwLCAwLCAwLCAwLAotICAgICAgICAgKGZuX3QpZmV0Y2gzeCB9LAotICAgIHsgMCwgKGZuX3QpZmV0Y2hFeHBhbmQ0dWIsIDAsIDAsIDAsIDAsCi0gICAgICAgICAoZm5fdClmZXRjaDRmLCAwLCAwLCAwLCAwLCAwLAotICAgICAgICAgKGZuX3QpZmV0Y2g0eCB9LAotfTsKLXN0YXRpYyBjb25zdCBmbl90IGNvbG9yX2NsYW1wX2ZjdFsyXVsxNl0gPSB7IC8vIHNpemU9ezMsNH0sIHR5cGU9e3ViLGYseH0KLSAgICB7IDAsIChmbl90KWZldGNoRXhwYW5kM3ViLCAwLCAwLCAwLCAwLAotICAgICAgICAgKGZuX3QpZmV0Y2hDbGFtcDNmLCAwLCAwLCAwLCAwLCAwLAotICAgICAgICAgKGZuX3QpZmV0Y2hDbGFtcDN4IH0sCi0gICAgeyAwLCAoZm5fdClmZXRjaEV4cGFuZDR1YiwgMCwgMCwgMCwgMCwKLSAgICAgICAgIChmbl90KWZldGNoQ2xhbXA0ZiwgMCwgMCwgMCwgMCwgMCwKLSAgICAgICAgIChmbl90KWZldGNoQ2xhbXA0eCB9LAotfTsKLXN0YXRpYyBjb25zdCBmbl90IG5vcm1hbF9mY3RbMV1bMTZdID0geyAvLyBzaXplPXszfSwgdHlwZT17YixzLGYseH0KLSAgICB7IChmbl90KWZldGNoRXhwYW5kM2IsIDAsCi0gICAgICAoZm5fdClmZXRjaEV4cGFuZDNzLCAwLCAwLCAwLAotICAgICAgKGZuX3QpZmV0Y2gzZiwgMCwgMCwgMCwgMCwgMCwKLSAgICAgIChmbl90KWZldGNoM3ggfSwKLX07Ci1zdGF0aWMgY29uc3QgZm5fdCB2ZXJ0ZXhfZmN0WzNdWzE2XSA9IHsgLy8gc2l6ZT17MiwzLDR9LCB0eXBlPXtiLHMsZix4fQotICAgIHsgKGZuX3QpZmV0Y2gyYiwgMCwKLSAgICAgIChmbl90KWZldGNoMnMsIDAsIDAsIDAsCi0gICAgICAoZm5fdClmZXRjaDJmLCAwLCAwLCAwLCAwLCAwLAotICAgICAgKGZuX3QpZmV0Y2gzeCB9LAotICAgIHsgKGZuX3QpZmV0Y2gzYiwgMCwKLSAgICAgIChmbl90KWZldGNoM3MsIDAsIDAsIDAsCi0gICAgICAoZm5fdClmZXRjaDNmLCAwLCAwLCAwLCAwLCAwLAotICAgICAgKGZuX3QpZmV0Y2gzeCB9LAotICAgIHsgKGZuX3QpZmV0Y2g0YiwgMCwKLSAgICAgIChmbl90KWZldGNoNHMsIDAsIDAsIDAsCi0gICAgICAoZm5fdClmZXRjaDRmLCAwLCAwLCAwLCAwLCAwLAotICAgICAgKGZuX3QpZmV0Y2g0eCB9Ci19Owotc3RhdGljIGNvbnN0IGZuX3QgdGV4dHVyZV9mY3RbM11bMTZdID0geyAvLyBzaXplPXsyLDMsNH0sIHR5cGU9e2IscyxmLHh9Ci0gICAgeyAoZm5fdClmZXRjaDJiLCAwLAotICAgICAgKGZuX3QpZmV0Y2gycywgMCwgMCwgMCwKLSAgICAgIChmbl90KWZldGNoMmYsIDAsIDAsIDAsIDAsIDAsCi0gICAgICAoZm5fdClmZXRjaDJ4IH0sCi0gICAgeyAoZm5fdClmZXRjaDNiLCAwLAotICAgICAgKGZuX3QpZmV0Y2gzcywgMCwgMCwgMCwKLSAgICAgIChmbl90KWZldGNoM2YsIDAsIDAsIDAsIDAsIDAsCi0gICAgICAoZm5fdClmZXRjaDN4IH0sCi0gICAgeyAoZm5fdClmZXRjaDRiLCAwLAotICAgICAgKGZuX3QpZmV0Y2g0cywgMCwgMCwgMCwKLSAgICAgIChmbl90KWZldGNoNGYsIDAsIDAsIDAsIDAsIDAsCi0gICAgICAoZm5fdClmZXRjaDR4IH0KLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIGFycmF5X3QKLSNlbmRpZgotCi12b2lkIGFycmF5X3Q6OmluaXQoCi0gICAgICAgIEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwKLSAgICAgICAgY29uc3QgR0x2b2lkICpwb2ludGVyLCBjb25zdCBidWZmZXJfdCogYm8sIEdMc2l6ZWkgY291bnQpCi17Ci0gICAgaWYgKCFzdHJpZGUpIHsKLSAgICAgICAgc3RyaWRlID0gc2l6ZTsKLSAgICAgICAgc3dpdGNoICh0eXBlKSB7Ci0gICAgICAgIGNhc2UgR0xfU0hPUlQ6Ci0gICAgICAgIGNhc2UgR0xfVU5TSUdORURfU0hPUlQ6Ci0gICAgICAgICAgICBzdHJpZGUgKj0gMjsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEdMX0ZMT0FUOgotICAgICAgICBjYXNlIEdMX0ZJWEVEOgotICAgICAgICAgICAgc3RyaWRlICo9IDQ7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgfQotICAgIH0KLSAgICB0aGlzLT5zaXplID0gc2l6ZTsKLSAgICB0aGlzLT50eXBlID0gdHlwZTsKLSAgICB0aGlzLT5zdHJpZGUgPSBzdHJpZGU7Ci0gICAgdGhpcy0+cG9pbnRlciA9IHBvaW50ZXI7Ci0gICAgdGhpcy0+Ym8gPSBibzsKLSAgICB0aGlzLT5ib3VuZHMgPSBjb3VudDsKLX0KLQotaW5saW5lIHZvaWQgYXJyYXlfdDo6cmVzb2x2ZSgpIAotewotICAgIHBoeXNpY2FsX3BvaW50ZXIgPSAoYm8pID8gKGJvLT5kYXRhICsgdWludHB0cl90KHBvaW50ZXIpKSA6IHBvaW50ZXI7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIHZlcnRleF9jYWNoZV90Ci0jZW5kaWYKLQotdm9pZCB2ZXJ0ZXhfY2FjaGVfdDo6aW5pdCgpCi17Ci0gICAgLy8gbWFrZSBzdXJlIHRoZSBzaXplIG9mIHZlcnRleF90IGFsbG93cyBjYWNoZS1saW5lIGFsaWdubWVudAotICAgIENUQTwoc2l6ZW9mKHZlcnRleF90KSAmIDB4MUYpID09IDA+IGFzc2VydEFsaWduZWRTaXplOwotCi0gICAgY29uc3QgaW50IGFsaWduID0gMzI7Ci0gICAgY29uc3Qgc2l6ZV90IHMgPSBWRVJURVhfQlVGRkVSX1NJWkUgKyBWRVJURVhfQ0FDSEVfU0laRTsKLSAgICBjb25zdCBzaXplX3Qgc2l6ZSA9IHMqc2l6ZW9mKHZlcnRleF90KSArIGFsaWduOwotICAgIGJhc2UgPSBtYWxsb2Moc2l6ZSk7Ci0gICAgaWYgKGJhc2UpIHsKLSAgICAgICAgbWVtc2V0KGJhc2UsIDAsIHNpemUpOwotICAgICAgICB2QnVmZmVyID0gKHZlcnRleF90KikoKHNpemVfdChiYXNlKSArIGFsaWduIC0gMSkgJiB+KGFsaWduLTEpKTsKLSAgICAgICAgdkNhY2hlID0gdkJ1ZmZlciArIFZFUlRFWF9CVUZGRVJfU0laRTsKLSAgICAgICAgc2VxdWVuY2UgPSAwOwotICAgIH0KLX0KLQotdm9pZCB2ZXJ0ZXhfY2FjaGVfdDo6dW5pbml0KCkKLXsKLSAgICBmcmVlKGJhc2UpOwotICAgIGJhc2UgPSB2QnVmZmVyID0gdkNhY2hlID0gMDsKLX0KLQotdm9pZCB2ZXJ0ZXhfY2FjaGVfdDo6Y2xlYXIoKQotewotI2lmIFZDX0NBQ0hFX1NUQVRJU1RJQ1MKLSAgICBzdGFydFRpbWUgPSBzeXN0ZW1UaW1lKFNZU1RFTV9USU1FX1RIUkVBRCk7Ci0gICAgdG90YWwgPSAwOwotICAgIG1pc3NlcyA9IDA7Ci0jZW5kaWYKLQotI2lmIFZDX0NBQ0hFX1RZUEUgPT0gVkNfQ0FDSEVfVFlQRV9MUlUKLSAgICB2ZXJ0ZXhfdCogdiA9IHZCdWZmZXI7Ci0gICAgc2l6ZV90IGNvdW50ID0gVkVSVEVYX0JVRkZFUl9TSVpFICsgVkVSVEVYX0NBQ0hFX1NJWkU7Ci0gICAgZG8gewotICAgICAgICB2LT5tcnUgPSAwOwotICAgICAgICB2Kys7Ci0gICAgfSB3aGlsZSAoLS1jb3VudCk7Ci0jZW5kaWYKLQotICAgIHNlcXVlbmNlICs9IElOREVYX1NFUTsKLSAgICBpZiAoc2VxdWVuY2UgPj0gMHg4MDAwMDAwMExVKSB7Ci0gICAgICAgIHNlcXVlbmNlID0gSU5ERVhfU0VROwotICAgICAgICB2ZXJ0ZXhfdCogdiA9IHZCdWZmZXI7Ci0gICAgICAgIHNpemVfdCBjb3VudCA9IFZFUlRFWF9CVUZGRVJfU0laRSArIFZFUlRFWF9DQUNIRV9TSVpFOwotICAgICAgICBkbyB7Ci0gICAgICAgICAgICB2LT5pbmRleCA9IDA7Ci0gICAgICAgICAgICB2Kys7Ci0gICAgICAgIH0gd2hpbGUgKC0tY291bnQpOwotICAgIH0KLX0KLQotdm9pZCB2ZXJ0ZXhfY2FjaGVfdDo6ZHVtcF9zdGF0cyhHTGVudW0gbW9kZSkKLXsKLSNpZiBWQ19DQUNIRV9TVEFUSVNUSUNTCi0gICAgbnNlY3NfdCB0aW1lID0gc3lzdGVtVGltZShTWVNURU1fVElNRV9USFJFQUQpIC0gc3RhcnRUaW1lOwotICAgIHVpbnQzMl90IGhpdHMgPSB0b3RhbCAtIG1pc3NlczsKLSAgICB1aW50MzJfdCBwcmltX2NvdW50OwotICAgIHN3aXRjaCAobW9kZSkgewotICAgIGNhc2UgR0xfUE9JTlRTOiAgICAgICAgICAgICBwcmltX2NvdW50ID0gdG90YWw7ICAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9MSU5FX1NUUklQOiAgICAgICAgIHByaW1fY291bnQgPSB0b3RhbCAtIDE7ICAgICBicmVhazsKLSAgICBjYXNlIEdMX0xJTkVfTE9PUDogICAgICAgICAgcHJpbV9jb3VudCA9IHRvdGFsIC0gMTsgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfTElORVM6ICAgICAgICAgICAgICBwcmltX2NvdW50ID0gdG90YWwgLyAyOyAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9UUklBTkdMRV9TVFJJUDogICAgIHByaW1fY291bnQgPSB0b3RhbCAtIDI7ICAgICBicmVhazsKLSAgICBjYXNlIEdMX1RSSUFOR0xFX0ZBTjogICAgICAgcHJpbV9jb3VudCA9IHRvdGFsIC0gMjsgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfVFJJQU5HTEVTOiAgICAgICAgICBwcmltX2NvdW50ID0gdG90YWwgLyAzOyAgICAgYnJlYWs7Ci0gICAgZGVmYXVsdDogICAgcmV0dXJuOwotICAgIH0KLSAgICBwcmludGYoICJ0b3RhbD0lNXUsIGhpdHM9JTV1LCBtaXNzPSU1dSwgaGl0cmF0ZT0lM3UlJSwiCi0gICAgICAgICAgICAiIHByaW1zPSU1dSwgdGltZT0lNnUgdXMsIHByaW1zL3M9JWQsIHYvdD0lZlxuIiwKLSAgICAgICAgICAgIHRvdGFsLCBoaXRzLCBtaXNzZXMsIChoaXRzKjEwMCkvdG90YWwsCi0gICAgICAgICAgICBwcmltX2NvdW50LCBpbnQobnMydXModGltZSkpLCBpbnQocHJpbV9jb3VudCpmbG9hdChzZWNvbmRzKDEpKS90aW1lKSwKLSAgICAgICAgICAgIGZsb2F0KG1pc3NlcykgLyBwcmltX2NvdW50KTsKLSNlbmRpZgotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNlbmRpZgotCi1zdGF0aWMgX19hdHRyaWJ1dGVfXygobm9pbmxpbmUpKQotdm9pZCBlbmFibGVEaXNhYmxlQ2xpZW50U3RhdGUob2dsZXNfY29udGV4dF90KiBjLCBHTGVudW0gYXJyYXksIGJvb2wgZW5hYmxlKQotewotICAgIGNvbnN0IGludCB0bXUgPSBjLT5hcnJheXMuYWN0aXZlVGV4dHVyZTsKLSAgICBhcnJheV90KiBhOwotICAgIHN3aXRjaCAoYXJyYXkpIHsKLSAgICBjYXNlIEdMX0NPTE9SX0FSUkFZOiAgICAgICAgICAgIGEgPSAmYy0+YXJyYXlzLmNvbG9yOyAgICAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9OT1JNQUxfQVJSQVk6ICAgICAgICAgICBhID0gJmMtPmFycmF5cy5ub3JtYWw7ICAgICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfVEVYVFVSRV9DT09SRF9BUlJBWTogICAgYSA9ICZjLT5hcnJheXMudGV4dHVyZVt0bXVdOyAgICBicmVhazsKLSAgICBjYXNlIEdMX1ZFUlRFWF9BUlJBWTogICAgICAgICAgIGEgPSAmYy0+YXJyYXlzLnZlcnRleDsgICAgICAgICAgYnJlYWs7Ci0gICAgZGVmYXVsdDoKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBhLT5lbmFibGUgPSBlbmFibGUgPyBHTF9UUlVFIDogR0xfRkFMU0U7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIFZlcnRleCBDYWNoZQotI2VuZGlmCi0KLXN0YXRpYyBfX2F0dHJpYnV0ZV9fKChub2lubGluZSkpCi12ZXJ0ZXhfdCogY2FjaGVfdmVydGV4KG9nbGVzX2NvbnRleHRfdCogYywgdmVydGV4X3QqIHYsIHVpbnQzMl90IGluZGV4KQotewotICAgICNpZiBWQ19DQUNIRV9TVEFUSVNUSUNTCi0gICAgICAgIGMtPnZjLm1pc3NlcysrOwotICAgICNlbmRpZgotICAgIGlmIChnZ2xfdW5saWtlbHkodi0+bG9ja2VkKSkgewotICAgICAgICAvLyB3ZSdyZSBqdXN0IGxvb2tpbmcgZm9yIGFuIGVudHJ5IGluIHRoZSBjYWNoZSB0aGF0IGlzIG5vdCBsb2NrZWQuCi0gICAgICAgIC8vIGFuZCB3ZSBrbm93IHRoYXQgdGhlcmUgY2Fubm90IGJlIG1vcmUgdGhhbiAyIGxvY2tlZCBlbnRyaWVzCi0gICAgICAgIC8vIGJlY2F1c2UgYSB0cmlhbmdsZSBuZWVkcyBhdCBtb3N0IDMgdmVydGljZXMuCi0gICAgICAgIC8vIFdlIG5ldmVyIHVzZSB0aGUgZmlyc3QgYW5kIHNlY29uZCBlbnRyaWVzIGJlY2F1c2UgdGhleSBtaWdodCBiZSBpbgotICAgICAgICAvLyB1c2UgYnkgdGhlIHN0cmlwZXIgb3IgZmFuZXIuIEFueSBvdGhlciBlbnRyeSB3aWxsIGRvIGFzIGxvbmcgYXMKLSAgICAgICAgLy8gaXQncyBub3QgbG9ja2VkLgotICAgICAgICAvLyBXZSBjb21wdXRlIGRpcmVjdGx5IHRoZSBpbmRleCBvZiBhICJmcmVlIiBlbnRyeSBmcm9tIHRoZSBsb2NrZWQKLSAgICAgICAgLy8gc3RhdGUgb2YgdlsyXSBhbmQgdlszXS4KLSAgICAgICAgdiA9IGMtPnZjLnZCdWZmZXIgKyAyOwotICAgICAgICB2ICs9IHZbMF0ubG9ja2VkIHwgKHZbMV0ubG9ja2VkPDwxKTsgICAgICAgCi0gICAgfQotICAgIC8vIG5vdGU6IGNvbXBpbGVFbGVtZW50IGNsZWFycyB2LT5mbGFncwotICAgIGMtPmFycmF5cy5jb21waWxlRWxlbWVudChjLCB2LCBpbmRleCk7Ci0gICAgdi0+bG9ja2VkID0gMTsKLSAgICByZXR1cm4gdjsKLX0KLQotc3RhdGljIF9fYXR0cmlidXRlX18oKG5vaW5saW5lKSkKLXZlcnRleF90KiBmZXRjaF92ZXJ0ZXgob2dsZXNfY29udGV4dF90KiBjLCBzaXplX3QgaW5kZXgpCi17Ci0gICAgaW5kZXggfD0gYy0+dmMuc2VxdWVuY2U7Ci0KLSNpZiBWQ19DQUNIRV9UWVBFID09IFZDX0NBQ0hFX1RZUEVfSU5ERVhFRAotCi0gICAgdmVydGV4X3QqIGNvbnN0IHYgPSBjLT52Yy52Q2FjaGUgKyAKLSAgICAgICAgICAgIChpbmRleCAmICh2ZXJ0ZXhfY2FjaGVfdDo6VkVSVEVYX0NBQ0hFX1NJWkUtMSkpOwotCi0gICAgaWYgKGdnbF9saWtlbHkodi0+aW5kZXggPT0gaW5kZXgpKSB7Ci0gICAgICAgIHYtPmxvY2tlZCA9IDE7Ci0gICAgICAgIHJldHVybiB2OwotICAgIH0KLSAgICByZXR1cm4gY2FjaGVfdmVydGV4KGMsIHYsIGluZGV4KTsKLQotI2VsaWYgVkNfQ0FDSEVfVFlQRSA9PSBWQ19DQUNIRV9UWVBFX0xSVQotCi0gICAgdmVydGV4X3QqIHYgPSBjLT52Yy52Q2FjaGUgKyAKLSAgICAgICAgICAgIChpbmRleCAmICgodmVydGV4X2NhY2hlX3Q6OlZFUlRFWF9DQUNIRV9TSVpFLTEpPj4xKSkqMjsKLQotICAgIC8vIGFsd2F5cyByZWNvcmQgTFJVIGluIHZbMF0KLSAgICBpZiAoZ2dsX2xpa2VseSh2WzBdLmluZGV4ID09IGluZGV4KSkgewotICAgICAgICB2WzBdLmxvY2tlZCA9IDE7Ci0gICAgICAgIHZbMF0ubXJ1ID0gMDsKLSAgICAgICAgcmV0dXJuICZ2WzBdOwotICAgIH0KLQotICAgIGlmIChnZ2xfbGlrZWx5KHZbMV0uaW5kZXggPT0gaW5kZXgpKSB7Ci0gICAgICAgIHZbMV0ubG9ja2VkID0gMTsKLSAgICAgICAgdlswXS5tcnUgPSAxOwotICAgICAgICByZXR1cm4gJnZbMV07Ci0gICAgfQotCi0gICAgY29uc3QgaW50IGxydSA9IDEgLSB2WzBdLm1ydTsKLSAgICB2WzBdLm1ydSA9IGxydTsKLSAgICByZXR1cm4gY2FjaGVfdmVydGV4KGMsICZ2W2xydV0sIGluZGV4KTsKLQotI2VsaWYgVkNfQ0FDSEVfVFlQRSA9PSBWQ19DQUNIRV9UWVBFX05PTkUKLQotICAgIC8vIGp1c3QgZm9yIGRlYnVnZ2luZy4uLgotICAgIHZlcnRleF90KiB2ID0gYy0+dmMudkJ1ZmZlciArIDI7Ci0gICAgcmV0dXJuIGNhY2hlX3ZlcnRleChjLCB2LCBpbmRleCk7Ci0KLSNlbmRpZgotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBQcmltaXRpdmUgQXNzZW1ibHkKLSNlbmRpZgotCi12b2lkIGRyYXdQcmltaXRpdmVzUG9pbnRzKG9nbGVzX2NvbnRleHRfdCogYywgR0xpbnQgZmlyc3QsIEdMc2l6ZWkgY291bnQpCi17Ci0gICAgaWYgKGdnbF91bmxpa2VseShjb3VudCA8IDEpKQotICAgICAgICByZXR1cm47Ci0KLSAgICAvLyB2ZXJ0ZXggY2FjaGUgc2l6ZSBtdXN0IGJlIG11bHRpcGxlIG9mIDEKLSAgICBjb25zdCBHTHNpemVpIHZjcyA9IAotICAgICAgICAgICAgKHZlcnRleF9jYWNoZV90OjpWRVJURVhfQlVGRkVSX1NJWkUgKwotICAgICAgICAgICAgIHZlcnRleF9jYWNoZV90OjpWRVJURVhfQ0FDSEVfU0laRSk7Ci0gICAgZG8gewotICAgICAgICB2ZXJ0ZXhfdCogdiA9IGMtPnZjLnZCdWZmZXI7Ci0gICAgICAgIEdMc2l6ZWkgbnVtID0gY291bnQgPiB2Y3MgPyB2Y3MgOiBjb3VudDsgCi0gICAgICAgIGMtPmFycmF5cy5jdWxsID0gdmVydGV4X3Q6OkNMSVBfQUxMOwotICAgICAgICBjLT5hcnJheXMuY29tcGlsZUVsZW1lbnRzKGMsIHYsIGZpcnN0LCBudW0pOwotICAgICAgICBmaXJzdCArPSBudW07Ci0gICAgICAgIGNvdW50IC09IG51bTsKLSAgICAgICAgaWYgKCFjLT5hcnJheXMuY3VsbCkgewotICAgICAgICAgICAgLy8gcXVpY2svdHJpdmlhbCByZWplY3Qgb2YgdGhlIHdob2xlIGJhdGNoCi0gICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgY29uc3QgdWludDMyX3QgY2MgPSB2WzBdLmZsYWdzOwotICAgICAgICAgICAgICAgIGlmIChnZ2xfbGlrZWx5KCEoY2MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEwpKSkKLSAgICAgICAgICAgICAgICAgICAgYy0+cHJpbXMucmVuZGVyUG9pbnQoYywgdik7Ci0gICAgICAgICAgICAgICAgdisrOwotICAgICAgICAgICAgICAgIG51bS0tOwotICAgICAgICAgICAgfSB3aGlsZSAobnVtKTsKLSAgICAgICAgfQotICAgIH0gd2hpbGUgKGNvdW50KTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIGRyYXdQcmltaXRpdmVzTGluZVN0cmlwKG9nbGVzX2NvbnRleHRfdCogYywgR0xpbnQgZmlyc3QsIEdMc2l6ZWkgY291bnQpCi17Ci0gICAgaWYgKGdnbF91bmxpa2VseShjb3VudCA8IDIpKQotICAgICAgICByZXR1cm47Ci0KLSAgICB2ZXJ0ZXhfdCAqdiwgKnYwLCAqdjE7Ci0gICAgYy0+YXJyYXlzLmN1bGwgPSB2ZXJ0ZXhfdDo6Q0xJUF9BTEw7Ci0gICAgYy0+YXJyYXlzLmNvbXBpbGVFbGVtZW50KGMsIGMtPnZjLnZCdWZmZXIsIGZpcnN0KTsKLSAgICBmaXJzdCArPSAxOwotICAgIGNvdW50IC09IDE7Ci0KLSAgICAvLyB2ZXJ0ZXggY2FjaGUgc2l6ZSBtdXN0IGJlIG11bHRpcGxlIG9mIDEKLSAgICBjb25zdCBHTHNpemVpIHZjcyA9IAotICAgICAgICAodmVydGV4X2NhY2hlX3Q6OlZFUlRFWF9CVUZGRVJfU0laRSArCi0gICAgICAgICB2ZXJ0ZXhfY2FjaGVfdDo6VkVSVEVYX0NBQ0hFX1NJWkUgLSAxKTsKLSAgICBkbyB7Ci0gICAgICAgIHYwID0gYy0+dmMudkJ1ZmZlciArIDA7IAotICAgICAgICB2ICA9IGMtPnZjLnZCdWZmZXIgKyAxOwotICAgICAgICBHTHNpemVpIG51bSA9IGNvdW50ID4gdmNzID8gdmNzIDogY291bnQ7IAotICAgICAgICBjLT5hcnJheXMuY29tcGlsZUVsZW1lbnRzKGMsIHYsIGZpcnN0LCBudW0pOwotICAgICAgICBmaXJzdCArPSBudW07Ci0gICAgICAgIGNvdW50IC09IG51bTsKLSAgICAgICAgaWYgKCFjLT5hcnJheXMuY3VsbCkgewotICAgICAgICAgICAgLy8gcXVpY2svdHJpdmlhbCByZWplY3Qgb2YgdGhlIHdob2xlIGJhdGNoCi0gICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgdjEgPSB2Kys7Ci0gICAgICAgICAgICAgICAgY29uc3QgdWludDMyX3QgY2MgPSB2MC0+ZmxhZ3MgJiB2MS0+ZmxhZ3M7Ci0gICAgICAgICAgICAgICAgaWYgKGdnbF9saWtlbHkoIShjYyAmIHZlcnRleF90OjpDTElQX0FMTCkpKQotICAgICAgICAgICAgICAgICAgICBjLT5wcmltcy5yZW5kZXJMaW5lKGMsIHYwLCB2MSk7Ci0gICAgICAgICAgICAgICAgdjAgPSB2MTsKLSAgICAgICAgICAgICAgICBudW0tLTsKLSAgICAgICAgICAgIH0gd2hpbGUgKG51bSk7Ci0gICAgICAgIH0KLSAgICAgICAgLy8gY29weSBiYWNrIHRoZSBsYXN0IHByb2Nlc3NlZCB2ZXJ0ZXgKLSAgICAgICAgYy0+dmMudkJ1ZmZlclswXSA9ICp2MDsKLSAgICAgICAgYy0+YXJyYXlzLmN1bGwgPSB2MC0+ZmxhZ3MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEw7Ci0gICAgfSB3aGlsZSAoY291bnQpOwotfQotCi12b2lkIGRyYXdQcmltaXRpdmVzTGluZUxvb3Aob2dsZXNfY29udGV4dF90KiBjLCBHTGludCBmaXJzdCwgR0xzaXplaSBjb3VudCkKLXsKLSAgICBpZiAoZ2dsX3VubGlrZWx5KGNvdW50IDwgMikpCi0gICAgICAgIHJldHVybjsKLSAgICBkcmF3UHJpbWl0aXZlc0xpbmVTdHJpcChjLCBmaXJzdCwgY291bnQpOwotICAgIGlmIChnZ2xfbGlrZWx5KGNvdW50ID49IDMpKSB7Ci0gICAgICAgIHZlcnRleF90KiB2MCA9IGMtPnZjLnZCdWZmZXI7IAotICAgICAgICB2ZXJ0ZXhfdCogdjEgPSBjLT52Yy52QnVmZmVyICsgMTsKLSAgICAgICAgYy0+YXJyYXlzLmNvbXBpbGVFbGVtZW50KGMsIHYxLCBmaXJzdCk7Ci0gICAgICAgIGNvbnN0IHVpbnQzMl90IGNjID0gdjAtPmZsYWdzICYgdjEtPmZsYWdzOwotICAgICAgICBpZiAoZ2dsX2xpa2VseSghKGNjICYgdmVydGV4X3Q6OkNMSVBfQUxMKSkpCi0gICAgICAgICAgICBjLT5wcmltcy5yZW5kZXJMaW5lKGMsIHYwLCB2MSk7Ci0gICAgfQotfQotCi12b2lkIGRyYXdQcmltaXRpdmVzTGluZXMob2dsZXNfY29udGV4dF90KiBjLCBHTGludCBmaXJzdCwgR0xzaXplaSBjb3VudCkKLXsKLSAgICBpZiAoZ2dsX3VubGlrZWx5KGNvdW50IDwgMikpCi0gICAgICAgIHJldHVybjsKLQotICAgIC8vIHZlcnRleCBjYWNoZSBzaXplIG11c3QgYmUgbXVsdGlwbGUgb2YgMgotICAgIGNvbnN0IEdMc2l6ZWkgdmNzID0gCi0gICAgICAgICgodmVydGV4X2NhY2hlX3Q6OlZFUlRFWF9CVUZGRVJfU0laRSArCi0gICAgICAgIHZlcnRleF9jYWNoZV90OjpWRVJURVhfQ0FDSEVfU0laRSkgLyAyKSAqIDI7Ci0gICAgZG8gewotICAgICAgICB2ZXJ0ZXhfdCogdiA9IGMtPnZjLnZCdWZmZXI7Ci0gICAgICAgIEdMc2l6ZWkgbnVtID0gY291bnQgPiB2Y3MgPyB2Y3MgOiBjb3VudDsgCi0gICAgICAgIGMtPmFycmF5cy5jdWxsID0gdmVydGV4X3Q6OkNMSVBfQUxMOwotICAgICAgICBjLT5hcnJheXMuY29tcGlsZUVsZW1lbnRzKGMsIHYsIGZpcnN0LCBudW0pOwotICAgICAgICBmaXJzdCArPSBudW07Ci0gICAgICAgIGNvdW50IC09IG51bTsKLSAgICAgICAgaWYgKCFjLT5hcnJheXMuY3VsbCkgewotICAgICAgICAgICAgLy8gcXVpY2svdHJpdmlhbCByZWplY3Qgb2YgdGhlIHdob2xlIGJhdGNoCi0gICAgICAgICAgICBudW0gLT0gMjsKLSAgICAgICAgICAgIGRvIHsKLSAgICAgICAgICAgICAgICBjb25zdCB1aW50MzJfdCBjYyA9IHZbMF0uZmxhZ3MgJiB2WzFdLmZsYWdzOwotICAgICAgICAgICAgICAgIGlmIChnZ2xfbGlrZWx5KCEoY2MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEwpKSkKLSAgICAgICAgICAgICAgICAgICAgYy0+cHJpbXMucmVuZGVyTGluZShjLCB2LCB2KzEpOwotICAgICAgICAgICAgICAgIHYgKz0gMjsKLSAgICAgICAgICAgICAgICBudW0gLT0gMjsKLSAgICAgICAgICAgIH0gd2hpbGUgKG51bSA+PSAwKTsKLSAgICAgICAgfQotICAgIH0gd2hpbGUgKGNvdW50ID49IDIpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyB2b2lkIGRyYXdQcmltaXRpdmVzVHJpYW5nbGVGYW5PclN0cmlwKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgR0xpbnQgZmlyc3QsIEdMc2l6ZWkgY291bnQsIGludCB3aW5kaW5nKQotewotICAgIC8vIHdpbmRpbmcgPT0gMiA6IGZhbgotICAgIC8vIHdpbmRpbmcgPT0gMSA6IHN0cmlwCi0KLSAgICBpZiAoZ2dsX3VubGlrZWx5KGNvdW50IDwgMykpCi0gICAgICAgIHJldHVybjsKLQotICAgIHZlcnRleF90ICp2LCAqdjAsICp2MSwgKnYyOwotICAgIGMtPmFycmF5cy5jdWxsID0gdmVydGV4X3Q6OkNMSVBfQUxMOwotICAgIGMtPmFycmF5cy5jb21waWxlRWxlbWVudHMoYywgYy0+dmMudkJ1ZmZlciwgZmlyc3QsIDIpOwotICAgIGZpcnN0ICs9IDI7Ci0gICAgY291bnQgLT0gMjsKLQotICAgIC8vIHZlcnRleCBjYWNoZSBzaXplIG11c3QgYmUgbXVsdGlwbGUgb2YgMi4gVGhpcyBpcyBleHRyZW1lbHkgaW1wb3J0YW50Ci0gICAgLy8gYmVjYXVzZSBpdCBhbGxvd3MgdXMgdG8gcHJlc2VydmUgdGhlIHNhbWUgd2luZGluZyB3aGVuIHRoZSB3aG9sZQotICAgIC8vIGJhdGNoIGlzIGN1bGxlZC4gV2UgYWxzbyBuZWVkIDIgZXh0cmEgdmVydGljZXMgaW4gdGhlIGFycmF5LCBiZWNhdXNlCi0gICAgLy8gd2UgYWx3YXlzIGtlZXAgdGhlIHR3byBmaXJzdCBvbmVzLgotICAgIGNvbnN0IEdMc2l6ZWkgdmNzID0gCi0gICAgICAgICgodmVydGV4X2NhY2hlX3Q6OlZFUlRFWF9CVUZGRVJfU0laRSArCi0gICAgICAgICAgdmVydGV4X2NhY2hlX3Q6OlZFUlRFWF9DQUNIRV9TSVpFIC0gMikgLyAyKSAqIDI7Ci0gICAgZG8gewotICAgICAgICB2MCA9IGMtPnZjLnZCdWZmZXIgKyAwOyAKLSAgICAgICAgdjEgPSBjLT52Yy52QnVmZmVyICsgMTsgCi0gICAgICAgIHYgID0gYy0+dmMudkJ1ZmZlciArIDI7Ci0gICAgICAgIEdMc2l6ZWkgbnVtID0gY291bnQgPiB2Y3MgPyB2Y3MgOiBjb3VudDsgCi0gICAgICAgIGMtPmFycmF5cy5jb21waWxlRWxlbWVudHMoYywgdiwgZmlyc3QsIG51bSk7Ci0gICAgICAgIGZpcnN0ICs9IG51bTsKLSAgICAgICAgY291bnQgLT0gbnVtOwotICAgICAgICBpZiAoIWMtPmFycmF5cy5jdWxsKSB7Ci0gICAgICAgICAgICAvLyBxdWljay90cml2aWFsIHJlamVjdCBvZiB0aGUgd2hvbGUgYmF0Y2gKLSAgICAgICAgICAgIGRvIHsKLSAgICAgICAgICAgICAgICB2MiA9IHYrKzsKLSAgICAgICAgICAgICAgICBjb25zdCB1aW50MzJfdCBjYyA9IHYwLT5mbGFncyAmIHYxLT5mbGFncyAmIHYyLT5mbGFnczsKLSAgICAgICAgICAgICAgICBpZiAoZ2dsX2xpa2VseSghKGNjICYgdmVydGV4X3Q6OkNMSVBfQUxMKSkpCi0gICAgICAgICAgICAgICAgICAgIGMtPnByaW1zLnJlbmRlclRyaWFuZ2xlKGMsIHYwLCB2MSwgdjIpOwotICAgICAgICAgICAgICAgIHN3YXAoKCh3aW5kaW5nXj0xKSA/IHYxIDogdjApLCB2Mik7Ci0gICAgICAgICAgICAgICAgbnVtLS07Ci0gICAgICAgICAgICB9IHdoaWxlIChudW0pOwotICAgICAgICB9Ci0gICAgICAgIGlmIChjb3VudCkgewotICAgICAgICAgICAgdjAgPSBjLT52Yy52QnVmZmVyICsgMiArIG51bSAtIDI7Ci0gICAgICAgICAgICB2MSA9IGMtPnZjLnZCdWZmZXIgKyAyICsgbnVtIC0gMTsKLSAgICAgICAgICAgIGlmICgod2luZGluZyYyKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgLy8gZm9yIHN0cmlwcyBjb3B5IGJhY2sgdGhlIHR3byBsYXN0IGNvbXBpbGVkIHZlcnRpY2VzCi0gICAgICAgICAgICAgICAgYy0+dmMudkJ1ZmZlclswXSA9ICp2MDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGMtPnZjLnZCdWZmZXJbMV0gPSAqdjE7Ci0gICAgICAgICAgICBjLT5hcnJheXMuY3VsbCA9IHYwLT5mbGFncyAmIHYxLT5mbGFncyAmIHZlcnRleF90OjpDTElQX0FMTDsKLSAgICAgICAgfQotICAgIH0gd2hpbGUgKGNvdW50ID4gMCk7Ci19Ci0KLXZvaWQgZHJhd1ByaW1pdGl2ZXNUcmlhbmdsZVN0cmlwKG9nbGVzX2NvbnRleHRfdCogYywgCi0gICAgICAgIEdMaW50IGZpcnN0LCBHTHNpemVpIGNvdW50KSB7Ci0gICAgZHJhd1ByaW1pdGl2ZXNUcmlhbmdsZUZhbk9yU3RyaXAoYywgZmlyc3QsIGNvdW50LCAxKTsKLX0KLQotdm9pZCBkcmF3UHJpbWl0aXZlc1RyaWFuZ2xlRmFuKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgR0xpbnQgZmlyc3QsIEdMc2l6ZWkgY291bnQpIHsKLSAgICBkcmF3UHJpbWl0aXZlc1RyaWFuZ2xlRmFuT3JTdHJpcChjLCBmaXJzdCwgY291bnQsIDIpOwotfQotCi12b2lkIGRyYXdQcmltaXRpdmVzVHJpYW5nbGVzKG9nbGVzX2NvbnRleHRfdCogYywgR0xpbnQgZmlyc3QsIEdMc2l6ZWkgY291bnQpCi17Ci0gICAgaWYgKGdnbF91bmxpa2VseShjb3VudCA8IDMpKQotICAgICAgICByZXR1cm47Ci0KLSAgICAvLyB2ZXJ0ZXggY2FjaGUgc2l6ZSBtdXN0IGJlIG11bHRpcGxlIG9mIDMKLSAgICBjb25zdCBHTHNpemVpIHZjcyA9IAotICAgICAgICAoKHZlcnRleF9jYWNoZV90OjpWRVJURVhfQlVGRkVSX1NJWkUgKwotICAgICAgICB2ZXJ0ZXhfY2FjaGVfdDo6VkVSVEVYX0NBQ0hFX1NJWkUpIC8gMykgKiAzOwotICAgIGRvIHsKLSAgICAgICAgdmVydGV4X3QqIHYgPSBjLT52Yy52QnVmZmVyOwotICAgICAgICBHTHNpemVpIG51bSA9IGNvdW50ID4gdmNzID8gdmNzIDogY291bnQ7IAotICAgICAgICBjLT5hcnJheXMuY3VsbCA9IHZlcnRleF90OjpDTElQX0FMTDsKLSAgICAgICAgYy0+YXJyYXlzLmNvbXBpbGVFbGVtZW50cyhjLCB2LCBmaXJzdCwgbnVtKTsKLSAgICAgICAgZmlyc3QgKz0gbnVtOwotICAgICAgICBjb3VudCAtPSBudW07Ci0gICAgICAgIGlmICghYy0+YXJyYXlzLmN1bGwpIHsKLSAgICAgICAgICAgIC8vIHF1aWNrL3RyaXZpYWwgcmVqZWN0IG9mIHRoZSB3aG9sZSBiYXRjaAotICAgICAgICAgICAgbnVtIC09IDM7Ci0gICAgICAgICAgICBkbyB7Ci0gICAgICAgICAgICAgICAgY29uc3QgdWludDMyX3QgY2MgPSB2WzBdLmZsYWdzICYgdlsxXS5mbGFncyAmIHZbMl0uZmxhZ3M7Ci0gICAgICAgICAgICAgICAgaWYgKGdnbF9saWtlbHkoIShjYyAmIHZlcnRleF90OjpDTElQX0FMTCkpKQotICAgICAgICAgICAgICAgICAgICBjLT5wcmltcy5yZW5kZXJUcmlhbmdsZShjLCB2LCB2KzEsIHYrMik7Ci0gICAgICAgICAgICAgICAgdiArPSAzOwotICAgICAgICAgICAgICAgIG51bSAtPSAzOwotICAgICAgICAgICAgfSB3aGlsZSAobnVtID49IDApOwotICAgICAgICB9Ci0gICAgfSB3aGlsZSAoY291bnQgPj0gMyk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI2VuZGlmCi0KLS8vIHRoaXMgbG9va3MgZ29vZnksIGJ1dCBnY2MgZG9lcyBhIGdyZWF0IGpvYiB3aXRoIHRoaXMuLi4KLXN0YXRpYyBpbmxpbmUgdW5zaWduZWQgaW50IHJlYWRfaW5kZXgoaW50IHR5cGUsIGNvbnN0IEdMdm9pZComIHApIHsKLSAgICB1bnNpZ25lZCBpbnQgcjsKLSAgICBpZiAodHlwZSkgewotICAgICAgICByID0gKihjb25zdCBHTHVieXRlKilwOwotICAgICAgICBwID0gKGNvbnN0IEdMdWJ5dGUqKXAgKyAxOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHIgPSAqKGNvbnN0IEdMdXNob3J0KilwOwotICAgICAgICBwID0gKGNvbnN0IEdMdXNob3J0KilwICsgMTsKLSAgICB9Ci0gICAgcmV0dXJuIHI7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdm9pZCBkcmF3SW5kZXhlZFByaW1pdGl2ZXNQb2ludHMob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICBHTHNpemVpIGNvdW50LCBjb25zdCBHTHZvaWQgKmluZGljZXMpCi17Ci0gICAgaWYgKGdnbF91bmxpa2VseShjb3VudCA8IDEpKQotICAgICAgICByZXR1cm47Ci0gICAgY29uc3QgaW50IHR5cGUgPSAoYy0+YXJyYXlzLmluZGljZXNUeXBlID09IEdMX1VOU0lHTkVEX0JZVEUpOwotICAgIGRvIHsKLSAgICAgICAgdmVydGV4X3QgKiB2ID0gZmV0Y2hfdmVydGV4KGMsIHJlYWRfaW5kZXgodHlwZSwgaW5kaWNlcykpOwotICAgICAgICBpZiAoZ2dsX2xpa2VseSghKHYtPmZsYWdzICYgdmVydGV4X3Q6OkNMSVBfQUxMKSkpCi0gICAgICAgICAgICBjLT5wcmltcy5yZW5kZXJQb2ludChjLCB2KTsKLSAgICAgICAgdi0+bG9ja2VkID0gMDsKLSAgICAgICAgY291bnQtLTsKLSAgICB9IHdoaWxlKGNvdW50KTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIGRyYXdJbmRleGVkUHJpbWl0aXZlc0xpbmVTdHJpcChvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIEdMc2l6ZWkgY291bnQsIGNvbnN0IEdMdm9pZCAqaW5kaWNlcykKLXsKLSAgICBpZiAoZ2dsX3VubGlrZWx5KGNvdW50IDwgMikpCi0gICAgICAgIHJldHVybjsKLSAgICAKLSAgICB2ZXJ0ZXhfdCAqIGNvbnN0IHYgPSBjLT52Yy52QnVmZmVyOwotICAgIHZlcnRleF90KiB2MCA9IHY7Ci0gICAgdmVydGV4X3QqIHYxOwotICAgIAotICAgIGNvbnN0IGludCB0eXBlID0gKGMtPmFycmF5cy5pbmRpY2VzVHlwZSA9PSBHTF9VTlNJR05FRF9CWVRFKTsKLSAgICBjLT5hcnJheXMuY29tcGlsZUVsZW1lbnQoYywgdjAsIHJlYWRfaW5kZXgodHlwZSwgaW5kaWNlcykpOwotICAgIGNvdW50IC09IDE7Ci0gICAgZG8gewotICAgICAgICB2MSA9IGZldGNoX3ZlcnRleChjLCByZWFkX2luZGV4KHR5cGUsIGluZGljZXMpKTsKLSAgICAgICAgY29uc3QgdWludDMyX3QgY2MgPSB2MC0+ZmxhZ3MgJiB2MS0+ZmxhZ3M7Ci0gICAgICAgIGlmIChnZ2xfbGlrZWx5KCEoY2MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEwpKSkKLSAgICAgICAgICAgIGMtPnByaW1zLnJlbmRlckxpbmUoYywgdjAsIHYxKTsKLSAgICAgICAgdjAtPmxvY2tlZCA9IDA7Ci0gICAgICAgIHYwID0gdjE7Ci0gICAgICAgIGNvdW50LS07Ci0gICAgfSB3aGlsZSAoY291bnQpOwotICAgIHYxLT5sb2NrZWQgPSAwOwotfQotCi12b2lkIGRyYXdJbmRleGVkUHJpbWl0aXZlc0xpbmVMb29wKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgR0xzaXplaSBjb3VudCwgY29uc3QgR0x2b2lkICppbmRpY2VzKQotewotICAgIGlmIChnZ2xfdW5saWtlbHkoY291bnQgPD0gMikpIHsKLSAgICAgICAgZHJhd0luZGV4ZWRQcmltaXRpdmVzTGluZXMoYywgY291bnQsIGluZGljZXMpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotIAotICAgIHZlcnRleF90ICogY29uc3QgdiA9IGMtPnZjLnZCdWZmZXI7Ci0gICAgdmVydGV4X3QqIHYwID0gdjsKLSAgICB2ZXJ0ZXhfdCogdjE7Ci0gICAgCi0gICAgY29uc3QgaW50IHR5cGUgPSAoYy0+YXJyYXlzLmluZGljZXNUeXBlID09IEdMX1VOU0lHTkVEX0JZVEUpOwotICAgIGMtPmFycmF5cy5jb21waWxlRWxlbWVudChjLCB2MCwgcmVhZF9pbmRleCh0eXBlLCBpbmRpY2VzKSk7Ci0gICAgY291bnQgLT0gMTsKLSAgICBkbyB7Ci0gICAgICAgIHYxID0gZmV0Y2hfdmVydGV4KGMsIHJlYWRfaW5kZXgodHlwZSwgaW5kaWNlcykpOwotICAgICAgICBjb25zdCB1aW50MzJfdCBjYyA9IHYwLT5mbGFncyAmIHYxLT5mbGFnczsKLSAgICAgICAgaWYgKGdnbF9saWtlbHkoIShjYyAmIHZlcnRleF90OjpDTElQX0FMTCkpKQotICAgICAgICAgICAgYy0+cHJpbXMucmVuZGVyTGluZShjLCB2MCwgdjEpOwotICAgICAgICB2MC0+bG9ja2VkID0gMDsKLSAgICAgICAgdjAgPSB2MTsKLSAgICAgICAgY291bnQtLTsKLSAgICB9IHdoaWxlIChjb3VudCk7Ci0gICAgdjEtPmxvY2tlZCA9IDA7Ci0KLSAgICB2MSA9IGMtPnZjLnZCdWZmZXI7IAotICAgIGNvbnN0IHVpbnQzMl90IGNjID0gdjAtPmZsYWdzICYgdjEtPmZsYWdzOwotICAgIGlmIChnZ2xfbGlrZWx5KCEoY2MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEwpKSkKLSAgICAgICAgYy0+cHJpbXMucmVuZGVyTGluZShjLCB2MCwgdjEpOwotfQotCi12b2lkIGRyYXdJbmRleGVkUHJpbWl0aXZlc0xpbmVzKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgR0xzaXplaSBjb3VudCwgY29uc3QgR0x2b2lkICppbmRpY2VzKQotewotICAgIGlmIChnZ2xfdW5saWtlbHkoY291bnQgPCAyKSkKLSAgICAgICAgcmV0dXJuOwotCi0gICAgY291bnQgLT0gMjsKLSAgICBjb25zdCBpbnQgdHlwZSA9IChjLT5hcnJheXMuaW5kaWNlc1R5cGUgPT0gR0xfVU5TSUdORURfQllURSk7Ci0gICAgZG8gewotICAgICAgICB2ZXJ0ZXhfdCogY29uc3QgdjAgPSBmZXRjaF92ZXJ0ZXgoYywgcmVhZF9pbmRleCh0eXBlLCBpbmRpY2VzKSk7Ci0gICAgICAgIHZlcnRleF90KiBjb25zdCB2MSA9IGZldGNoX3ZlcnRleChjLCByZWFkX2luZGV4KHR5cGUsIGluZGljZXMpKTsKLSAgICAgICAgY29uc3QgdWludDMyX3QgY2MgPSB2MC0+ZmxhZ3MgJiB2MS0+ZmxhZ3M7Ci0gICAgICAgIGlmIChnZ2xfbGlrZWx5KCEoY2MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEwpKSkKLSAgICAgICAgICAgIGMtPnByaW1zLnJlbmRlckxpbmUoYywgdjAsIHYxKTsKLSAgICAgICAgdjAtPmxvY2tlZCA9IDA7Ci0gICAgICAgIHYxLT5sb2NrZWQgPSAwOwotICAgICAgICBjb3VudCAtPSAyOwotICAgIH0gd2hpbGUgKGNvdW50ID49IDApOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyB2b2lkIGRyYXdJbmRleGVkUHJpbWl0aXZlc1RyaWFuZ2xlRmFuT3JTdHJpcChvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIEdMc2l6ZWkgY291bnQsIGNvbnN0IEdMdm9pZCAqaW5kaWNlcywgaW50IHdpbmRpbmcpCi17Ci0gICAgLy8gd2luZGluZyA9PSAyIDogZmFuCi0gICAgLy8gd2luZGluZyA9PSAxIDogc3RyaXAKLQotICAgIGlmIChnZ2xfdW5saWtlbHkoY291bnQgPCAzKSkKLSAgICAgICAgcmV0dXJuOwotICAgIAotICAgIHZlcnRleF90ICogY29uc3QgdiA9IGMtPnZjLnZCdWZmZXI7Ci0gICAgdmVydGV4X3QqIHYwID0gdjsKLSAgICB2ZXJ0ZXhfdCogdjEgPSB2KzE7Ci0gICAgdmVydGV4X3QqIHYyOwotCi0gICAgY29uc3QgaW50IHR5cGUgPSAoYy0+YXJyYXlzLmluZGljZXNUeXBlID09IEdMX1VOU0lHTkVEX0JZVEUpOwotICAgIGMtPmFycmF5cy5jb21waWxlRWxlbWVudChjLCB2MCwgcmVhZF9pbmRleCh0eXBlLCBpbmRpY2VzKSk7Ci0gICAgYy0+YXJyYXlzLmNvbXBpbGVFbGVtZW50KGMsIHYxLCByZWFkX2luZGV4KHR5cGUsIGluZGljZXMpKTsKLSAgICBjb3VudCAtPSAyOwotCi0gICAgLy8gbm90ZTogR0NDIDQuMS4xIGhlcmUgbWFrZXMgYSBwcmV0eSBpbnRlcmVzdGluZyBvcHRpbWl6YXRpb24KLSAgICAvLyB3aGVyZSBpdCBkdXBsaWNhdGVzIHRoZSBsb29wIGJlbG93IGJhc2VkIG9uIGMtPmFycmF5cy5pbmRpY2VzVHlwZQotCi0gICAgZG8gewotICAgICAgICB2MiA9IGZldGNoX3ZlcnRleChjLCByZWFkX2luZGV4KHR5cGUsIGluZGljZXMpKTsKLSAgICAgICAgY29uc3QgdWludDMyX3QgY2MgPSB2MC0+ZmxhZ3MgJiB2MS0+ZmxhZ3MgJiB2Mi0+ZmxhZ3M7Ci0gICAgICAgIGlmIChnZ2xfbGlrZWx5KCEoY2MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEwpKSkKLSAgICAgICAgICAgIGMtPnByaW1zLnJlbmRlclRyaWFuZ2xlKGMsIHYwLCB2MSwgdjIpOwotICAgICAgICB2ZXJ0ZXhfdCogJiBjb25zdW1lZCA9ICgod2luZGluZ149MSkgPyB2MSA6IHYwKTsKLSAgICAgICAgY29uc3VtZWQtPmxvY2tlZCA9IDA7Ci0gICAgICAgIGNvbnN1bWVkID0gdjI7Ci0gICAgICAgIGNvdW50LS07Ci0gICAgfSB3aGlsZSAoY291bnQpOwotICAgIHYwLT5sb2NrZWQgPSB2MS0+bG9ja2VkID0gMDsKLSAgICB2Mi0+bG9ja2VkID0gMDsKLX0KLQotdm9pZCBkcmF3SW5kZXhlZFByaW1pdGl2ZXNUcmlhbmdsZVN0cmlwKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgR0xzaXplaSBjb3VudCwgY29uc3QgR0x2b2lkICppbmRpY2VzKSB7Ci0gICAgZHJhd0luZGV4ZWRQcmltaXRpdmVzVHJpYW5nbGVGYW5PclN0cmlwKGMsIGNvdW50LCBpbmRpY2VzLCAxKTsKLX0KLQotdm9pZCBkcmF3SW5kZXhlZFByaW1pdGl2ZXNUcmlhbmdsZUZhbihvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIEdMc2l6ZWkgY291bnQsIGNvbnN0IEdMdm9pZCAqaW5kaWNlcykgewotICAgIGRyYXdJbmRleGVkUHJpbWl0aXZlc1RyaWFuZ2xlRmFuT3JTdHJpcChjLCBjb3VudCwgaW5kaWNlcywgMik7Ci19Ci0KLXZvaWQgZHJhd0luZGV4ZWRQcmltaXRpdmVzVHJpYW5nbGVzKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgR0xzaXplaSBjb3VudCwgY29uc3QgR0x2b2lkICppbmRpY2VzKQotewotICAgIGlmIChnZ2xfdW5saWtlbHkoY291bnQgPCAzKSkKLSAgICAgICAgcmV0dXJuOwotCi0gICAgY291bnQgLT0gMzsKLSAgICBpZiAoZ2dsX2xpa2VseShjLT5hcnJheXMuaW5kaWNlc1R5cGUgPT0gR0xfVU5TSUdORURfU0hPUlQpKSB7Ci0gICAgICAgIC8vIFRoaXMgY2FzZSBpcyBwcm9iYWJseSBvdXIgbW9zdCBjb21tb24gY2FzZS4uLgotICAgICAgICB1aW50MTZfdCBjb25zdCAqIHAgPSAodWludDE2X3QgY29uc3QgKilpbmRpY2VzOwotICAgICAgICBkbyB7Ci0gICAgICAgICAgICB2ZXJ0ZXhfdCogY29uc3QgdjAgPSBmZXRjaF92ZXJ0ZXgoYywgKnArKyk7Ci0gICAgICAgICAgICB2ZXJ0ZXhfdCogY29uc3QgdjEgPSBmZXRjaF92ZXJ0ZXgoYywgKnArKyk7Ci0gICAgICAgICAgICB2ZXJ0ZXhfdCogY29uc3QgdjIgPSBmZXRjaF92ZXJ0ZXgoYywgKnArKyk7Ci0gICAgICAgICAgICBjb25zdCB1aW50MzJfdCBjYyA9IHYwLT5mbGFncyAmIHYxLT5mbGFncyAmIHYyLT5mbGFnczsKLSAgICAgICAgICAgIGlmIChnZ2xfbGlrZWx5KCEoY2MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEwpKSkKLSAgICAgICAgICAgICAgICBjLT5wcmltcy5yZW5kZXJUcmlhbmdsZShjLCB2MCwgdjEsIHYyKTsKLSAgICAgICAgICAgIHYwLT5sb2NrZWQgPSAwOwotICAgICAgICAgICAgdjEtPmxvY2tlZCA9IDA7Ci0gICAgICAgICAgICB2Mi0+bG9ja2VkID0gMDsKLSAgICAgICAgICAgIGNvdW50IC09IDM7Ci0gICAgICAgIH0gd2hpbGUgKGNvdW50ID49IDApOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIHVpbnQ4X3QgY29uc3QgKiBwID0gKHVpbnQ4X3QgY29uc3QgKilpbmRpY2VzOwotICAgICAgICBkbyB7Ci0gICAgICAgICAgICB2ZXJ0ZXhfdCogY29uc3QgdjAgPSBmZXRjaF92ZXJ0ZXgoYywgKnArKyk7Ci0gICAgICAgICAgICB2ZXJ0ZXhfdCogY29uc3QgdjEgPSBmZXRjaF92ZXJ0ZXgoYywgKnArKyk7Ci0gICAgICAgICAgICB2ZXJ0ZXhfdCogY29uc3QgdjIgPSBmZXRjaF92ZXJ0ZXgoYywgKnArKyk7Ci0gICAgICAgICAgICBjb25zdCB1aW50MzJfdCBjYyA9IHYwLT5mbGFncyAmIHYxLT5mbGFncyAmIHYyLT5mbGFnczsKLSAgICAgICAgICAgIGlmIChnZ2xfbGlrZWx5KCEoY2MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEwpKSkKLSAgICAgICAgICAgICAgICBjLT5wcmltcy5yZW5kZXJUcmlhbmdsZShjLCB2MCwgdjEsIHYyKTsKLSAgICAgICAgICAgIHYwLT5sb2NrZWQgPSAwOwotICAgICAgICAgICAgdjEtPmxvY2tlZCA9IDA7Ci0gICAgICAgICAgICB2Mi0+bG9ja2VkID0gMDsKLSAgICAgICAgICAgIGNvdW50IC09IDM7Ci0gICAgICAgIH0gd2hpbGUgKGNvdW50ID49IDApOwotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jcHJhZ21hIG1hcmsgQXJyYXkgY29tcGlsZXJzCi0jZW5kaWYKLQotdm9pZCBjb21waWxlRWxlbWVudF9fZ2VuZXJpYyhvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2LCBHTGludCBmaXJzdCkKLXsKLSAgICB2LT5mbGFncyA9IDA7Ci0gICAgdi0+aW5kZXggPSBmaXJzdDsKLSAgICBmaXJzdCAmPSB2ZXJ0ZXhfY2FjaGVfdDo6SU5ERVhfTUFTSzsKLSAgICBjb25zdCBHTHVieXRlKiB2cCA9IGMtPmFycmF5cy52ZXJ0ZXguZWxlbWVudChmaXJzdCk7Ci0gICAgYy0+YXJyYXlzLnZlcnRleC5mZXRjaChjLCB2LT5vYmoudiwgdnApOwotICAgIGMtPmFycmF5cy5tdnBfdHJhbnNmb3JtKCZjLT50cmFuc2Zvcm1zLm12cCwgJnYtPmNsaXAsICZ2LT5vYmopOwotICAgIGMtPmFycmF5cy5wZXJzcGVjdGl2ZShjLCB2KTsKLX0KLQotdm9pZCBjb21waWxlRWxlbWVudHNfX2dlbmVyaWMob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICB2ZXJ0ZXhfdCogdiwgR0xpbnQgZmlyc3QsIEdMc2l6ZWkgY291bnQpCi17Ci0gICAgY29uc3QgR0x1Ynl0ZSogdnAgPSBjLT5hcnJheXMudmVydGV4LmVsZW1lbnQoCi0gICAgICAgICAgICBmaXJzdCAmIHZlcnRleF9jYWNoZV90OjpJTkRFWF9NQVNLKTsKLSAgICBjb25zdCBzaXplX3Qgc3RyaWRlID0gYy0+YXJyYXlzLnZlcnRleC5zdHJpZGU7Ci0gICAgdHJhbnNmb3JtX3QgY29uc3QqIGNvbnN0IG12cCA9ICZjLT50cmFuc2Zvcm1zLm12cDsKLSAgICBkbyB7Ci0gICAgICAgIHYtPmZsYWdzID0gMDsKLSAgICAgICAgdi0+aW5kZXggPSBmaXJzdCsrOwotICAgICAgICBjLT5hcnJheXMudmVydGV4LmZldGNoKGMsIHYtPm9iai52LCB2cCk7Ci0gICAgICAgIGMtPmFycmF5cy5tdnBfdHJhbnNmb3JtKG12cCwgJnYtPmNsaXAsICZ2LT5vYmopOwotICAgICAgICBjLT5hcnJheXMucGVyc3BlY3RpdmUoYywgdik7Ci0gICAgICAgIHZwICs9IHN0cmlkZTsKLSAgICAgICAgdisrOwotICAgIH0gd2hpbGUgKC0tY291bnQpOwotfQotCi0vKgotdm9pZCBjb21waWxlRWxlbWVudHNfXzN4X2Z1bGwob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICB2ZXJ0ZXhfdCogdiwgR0xpbnQgZmlyc3QsIEdMc2l6ZWkgY291bnQpCi17Ci0gICAgY29uc3QgR0xmaXhlZCogdnAgPSAoY29uc3QgR0xmaXhlZCopYy0+YXJyYXlzLnZlcnRleC5lbGVtZW50KGZpcnN0KTsKLSAgICBjb25zdCBzaXplX3Qgc3RyaWRlID0gYy0+YXJyYXlzLnZlcnRleC5zdHJpZGUgLyA0OwotLy8gICAgY29uc3QgR0xmaXhlZCogY29uc3QmIG0gPSBjLT50cmFuc2Zvcm1zLm12cC5tYXRyaXgubTsKLSAgICAKLSAgICBHTGZpeGVkIG1bMTZdOwotICAgIG1lbWNweSgmbSwgYy0+dHJhbnNmb3Jtcy5tdnAubWF0cml4Lm0sIHNpemVvZihtKSk7Ci0gICAgCi0gICAgZG8gewotICAgICAgICBjb25zdCBHTGZpeGVkIHJ4ID0gdnBbMF07Ci0gICAgICAgIGNvbnN0IEdMZml4ZWQgcnkgPSB2cFsxXTsKLSAgICAgICAgY29uc3QgR0xmaXhlZCByeiA9IHZwWzJdOwotICAgICAgICB2cCArPSBzdHJpZGU7Ci0gICAgICAgIHYtPmluZGV4ID0gZmlyc3QrKzsKLSAgICAgICAgdi0+Y2xpcC54ID0gbWxhM2EocngsIG1bIDBdLCByeSwgbVsgNF0sIHJ6LCBtWyA4XSwgbVsxMl0pOyAKLSAgICAgICAgdi0+Y2xpcC55ID0gbWxhM2EocngsIG1bIDFdLCByeSwgbVsgNV0sIHJ6LCBtWyA5XSwgbVsxM10pOwotICAgICAgICB2LT5jbGlwLnogPSBtbGEzYShyeCwgbVsgMl0sIHJ5LCBtWyA2XSwgcnosIG1bMTBdLCBtWzE0XSk7Ci0gICAgICAgIHYtPmNsaXAudyA9IG1sYTNhKHJ4LCBtWyAzXSwgcnksIG1bIDddLCByeiwgbVsxMV0sIG1bMTVdKTsKLQotICAgICAgICBjb25zdCBHTGZpeGVkIHcgPSB2LT5jbGlwLnc7Ci0gICAgICAgIHVpbnQzMl90IGNsaXAgPSAwOwotICAgICAgICBpZiAodi0+Y2xpcC54IDwgLXcpICAgY2xpcCB8PSB2ZXJ0ZXhfdDo6Q0xJUF9MOwotICAgICAgICBpZiAodi0+Y2xpcC54ID4gIHcpICAgY2xpcCB8PSB2ZXJ0ZXhfdDo6Q0xJUF9SOwotICAgICAgICBpZiAodi0+Y2xpcC55IDwgLXcpICAgY2xpcCB8PSB2ZXJ0ZXhfdDo6Q0xJUF9COwotICAgICAgICBpZiAodi0+Y2xpcC55ID4gIHcpICAgY2xpcCB8PSB2ZXJ0ZXhfdDo6Q0xJUF9UOwotICAgICAgICBpZiAodi0+Y2xpcC56IDwgLXcpICAgY2xpcCB8PSB2ZXJ0ZXhfdDo6Q0xJUF9OOwotICAgICAgICBpZiAodi0+Y2xpcC56ID4gIHcpICAgY2xpcCB8PSB2ZXJ0ZXhfdDo6Q0xJUF9GOwotICAgICAgICB2LT5mbGFncyA9IGNsaXA7Ci0gICAgICAgIGMtPmFycmF5cy5jdWxsICY9IGNsaXA7Ci0KLSAgICAgICAgLy9jLT5hcnJheXMucGVyc3BlY3RpdmUoYywgdik7Ci0gICAgICAgIHYrKzsKLSAgICB9IHdoaWxlICgtLWNvdW50KTsKLX0KLSovCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIGNsaXBwZXJzCi0jZW5kaWYKLQotc3RhdGljIHZvaWQgY2xpcFZlYzQodmVjNF90JiBudiwgCi0gICAgICAgIEdMZml4ZWQgdCwgY29uc3QgdmVjNF90JiBzLCBjb25zdCB2ZWM0X3QmIHApCi17Ci0gICAgZm9yIChpbnQgaT0wOyBpPDQgOyBpKyspCi0gICAgICAgIG52LnZbaV0gPSBnZ2xNdWxBZGR4KHQsIHMudltpXSAtIHAudltpXSwgcC52W2ldLCAyOCk7Ci19Ci0KLXN0YXRpYyB2b2lkIGNsaXBWZXJ0ZXgob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogbnYsCi0gICAgICAgIEdMZml4ZWQgdCwgY29uc3QgdmVydGV4X3QqIHMsIGNvbnN0IHZlcnRleF90KiBwKQotewotICAgIGNsaXBWZWM0KG52LT5jbGlwLCB0LCBzLT5jbGlwLCBwLT5jbGlwKTsKLSAgICBudi0+Zm9nID0gZ2dsTXVsQWRkeCh0LCBzLT5mb2cgLSBwLT5mb2csIHAtPmZvZywgMjgpOwotICAgIG9nbGVzX3ZlcnRleF9wcm9qZWN0KGMsIG52KTsKLSAgICBudi0+ZmxhZ3MgfD0gIHZlcnRleF90OjpMSVQgfCB2ZXJ0ZXhfdDo6RVlFIHwgdmVydGV4X3Q6OlRUOwotICAgIG52LT5mbGFncyAmPSB+dmVydGV4X3Q6OkNMSVBfQUxMOwotfQotCi1zdGF0aWMgdm9pZCBjbGlwVmVydGV4QyhvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiBudiwKLSAgICAgICAgR0xmaXhlZCB0LCBjb25zdCB2ZXJ0ZXhfdCogcywgY29uc3QgdmVydGV4X3QqIHApCi17Ci0gICAgY2xpcFZlYzQobnYtPmNvbG9yLCB0LCBzLT5jb2xvciwgcC0+Y29sb3IpOwotICAgIGNsaXBWZXJ0ZXgoYywgbnYsIHQsIHMsIHApOwotfQotCi1zdGF0aWMgdm9pZCBjbGlwVmVydGV4VChvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiBudiwKLSAgICAgICAgR0xmaXhlZCB0LCBjb25zdCB2ZXJ0ZXhfdCogcywgY29uc3QgdmVydGV4X3QqIHApCi17Ci0gICAgZm9yIChpbnQgaT0wIDsgaTxHR0xfVEVYVFVSRV9VTklUX0NPVU5UIDsgaSsrKSB7Ci0gICAgICAgIGlmIChjLT5yYXN0ZXJpemVyLnN0YXRlLnRleHR1cmVbaV0uZW5hYmxlKQotICAgICAgICAgICAgY2xpcFZlYzQobnYtPnRleHR1cmVbaV0sIHQsIHMtPnRleHR1cmVbaV0sIHAtPnRleHR1cmVbaV0pOwotICAgIH0KLSAgICBjbGlwVmVydGV4KGMsIG52LCB0LCBzLCBwKTsKLX0KLQotc3RhdGljIHZvaWQgY2xpcFZlcnRleEFsbChvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiBudiwKLSAgICAgICAgR0xmaXhlZCB0LCBjb25zdCB2ZXJ0ZXhfdCogcywgY29uc3QgdmVydGV4X3QqIHApCi17Ci0gICAgY2xpcFZlYzQobnYtPmNvbG9yLCB0LCBzLT5jb2xvciwgcC0+Y29sb3IpOwotICAgIGNsaXBWZXJ0ZXhUKGMsIG52LCB0LCBzLCBwKTsKLX0KLQotc3RhdGljIHZvaWQgY2xpcEV5ZShvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiBudiwKLSAgICAgICAgR0xmaXhlZCB0LCBjb25zdCB2ZXJ0ZXhfdCogcywgY29uc3QgdmVydGV4X3QqIHApCi17Ci0gICAgbnYtPmNsZWFyKCk7Ci0gICAgYy0+YXJyYXlzLmNsaXBWZXJ0ZXgoYywgbnYsIHQsIHAsIHMpOwotICAgIGNsaXBWZWM0KG52LT5leWUsIHQsIHMtPmV5ZSwgcC0+ZXllKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jZW5kaWYKLQotdm9pZCB2YWxpZGF0ZV9hcnJheXMob2dsZXNfY29udGV4dF90KiBjLCBHTGVudW0gbW9kZSkKLXsKLSAgICB1aW50MzJfdCBlbmFibGVzID0gYy0+cmFzdGVyaXplci5zdGF0ZS5lbmFibGVzOwotCi0gICAgLy8gUGVyc3BlY3RpdmUgY29ycmVjdGlvbiBpcyBub3QgbmVlZCBpZiBPcnRobyB0cmFuc2Zvcm0sIGJ1dAotICAgIC8vIHRoZSB1c2VyIGNhbiBzdGlsbCBwcm92aWRlIHRoZSB3IGNvb3JkaW5hdGUgbWFudWFsbHksIHNvIHdlIGNhbid0Ci0gICAgLy8gYXV0b21hdGljYWxseSB0dXJuIGl0IG9mZiAoaW4gZmFjdCB3ZSBjb3VsZCB3aGVuIHRoZSA0dGggY29vcmRpbmF0ZQotICAgIC8vIGlzIG5vdCBzcGNpZmllZCBpbiB0aGUgdmVydGV4IGFycmF5KS4KLSAgICAvLyBXIGludGVycG9sYXRpb24gaXMgbmV2ZXIgbmVlZGVkIGZvciBwb2ludHMuCi0gICAgR0xib29sZWFuIHBlcnNwZWN0aXZlID0gCi0gICAgICAgIGMtPnBlcnNwZWN0aXZlICYmIG1vZGUhPUdMX1BPSU5UUyAmJiAoZW5hYmxlcyAmIEdHTF9FTkFCTEVfVE1VUyk7Ci0gICAgYy0+cmFzdGVyaXplci5wcm9jcy5lbmFibGVEaXNhYmxlKGMsIEdHTF9XX0xFUlAsIHBlcnNwZWN0aXZlKTsKLSAgICAKLSAgICAvLyBzZXQgYW50aS1hbGlhc2luZwotICAgIEdMYm9vbGVhbiBzbW9vdGggPSBHTF9GQUxTRTsKLSAgICBzd2l0Y2ggKG1vZGUpIHsKLSAgICBjYXNlIEdMX1BPSU5UUzoKLSAgICAgICAgc21vb3RoID0gYy0+cG9pbnQuc21vb3RoOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX0xJTkVTOgotICAgIGNhc2UgR0xfTElORV9MT09QOgotICAgIGNhc2UgR0xfTElORV9TVFJJUDoKLSAgICAgICAgc21vb3RoID0gYy0+bGluZS5zbW9vdGg7Ci0gICAgICAgIGJyZWFrOwotICAgIH0KLSAgICBpZiAoKChlbmFibGVzICYgR0dMX0VOQUJMRV9BQSk/MTowKSAhPSBzbW9vdGgpCi0gICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuZW5hYmxlRGlzYWJsZShjLCBHR0xfQUEsIHNtb290aCk7Ci0KLSAgICAvLyBzZXQgdGhlIHNoYWRlIG1vZGVsIGZvciB0aGlzIHByaW1pdGl2ZQotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3Muc2hhZGVNb2RlbChjLAotICAgICAgICAgICAgKG1vZGUgPT0gR0xfUE9JTlRTKSA/IEdMX0ZMQVQgOiBjLT5saWdodGluZy5zaGFkZU1vZGVsKTsKLQotICAgIC8vIGNvbXB1dGUgYWxsIHRoZSBtYXRyaWNlcyB3ZSdsbCBuZWVkLi4uCi0gICAgdWludDMyX3Qgd2FudCA9Ci0gICAgICAgICAgICB0cmFuc2Zvcm1fc3RhdGVfdDo6TVZQIHwKLSAgICAgICAgICAgIHRyYW5zZm9ybV9zdGF0ZV90OjpWSUVXUE9SVDsKLSAgICBpZiAoYy0+bGlnaHRpbmcuZW5hYmxlKSB7IC8vIG5lZWRzIG5vcm1hbCB0cmFuc2Zvcm1zIGFuZCBleWUgY29vcmRzCi0gICAgICAgIHdhbnQgfD0gdHJhbnNmb3JtX3N0YXRlX3Q6Ok1WVUk7Ci0gICAgICAgIHdhbnQgfD0gdHJhbnNmb3JtX3N0YXRlX3Q6Ok1PREVMVklFVzsKLSAgICB9Ci0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX1RNVVMpIHsgLy8gbmVlZHMgdGV4dHVyZSB0cmFuc2Zvcm1zCi0gICAgICAgIHdhbnQgfD0gdHJhbnNmb3JtX3N0YXRlX3Q6OlRFWFRVUkU7Ci0gICAgfQotICAgIGlmIChjLT5jbGlwUGxhbmVzLmVuYWJsZSB8fCAoZW5hYmxlcyAmIEdHTF9FTkFCTEVfRk9HKSkgeyAKLSAgICAgICAgd2FudCB8PSB0cmFuc2Zvcm1fc3RhdGVfdDo6TU9ERUxWSUVXOyAvLyBuZWVkcyBleWUgY29vcmRzCi0gICAgfQotICAgIG9nbGVzX3ZhbGlkYXRlX3RyYW5zZm9ybShjLCB3YW50KTsKLQotICAgIC8vIHRleHR1cmVzLi4uCi0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX1RNVVMpCi0gICAgICAgIG9nbGVzX3ZhbGlkYXRlX3RleHR1cmUoYyk7Ci0KLSAgICAvLyB2ZXJ0ZXggY29tcGlsZXJzCi0gICAgYy0+YXJyYXlzLmNvbXBpbGVFbGVtZW50ID0gY29tcGlsZUVsZW1lbnRfX2dlbmVyaWM7Ci0gICAgYy0+YXJyYXlzLmNvbXBpbGVFbGVtZW50cyA9IGNvbXBpbGVFbGVtZW50c19fZ2VuZXJpYzsKLQotICAgIC8vIHZlcnRleCB0cmFuc2Zvcm0KLSAgICBjLT5hcnJheXMubXZwX3RyYW5zZm9ybSA9Ci0gICAgICAgIGMtPnRyYW5zZm9ybXMubXZwLnBvaW50dltjLT5hcnJheXMudmVydGV4LnNpemUgLSAyXTsKLQotICAgIGMtPmFycmF5cy5tdl90cmFuc2Zvcm0gPQotICAgICAgICBjLT50cmFuc2Zvcm1zLm1vZGVsdmlldy50cmFuc2Zvcm0ucG9pbnR2W2MtPmFycmF5cy52ZXJ0ZXguc2l6ZSAtIDJdOwotICAgIAotICAgIC8qCi0gICAgICogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKLSAgICAgKiAgcGljayBmZXRjaGVycwotICAgICAqICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCi0gICAgICovCi0gICAgCi0gICAgYXJyYXlfbWFjaGluZV90JiBhbSA9IGMtPmFycmF5czsKLSAgICBhbS52ZXJ0ZXguZmV0Y2ggPSBmZXRjaE5vcDsKLSAgICBhbS5ub3JtYWwuZmV0Y2ggPSBjdXJyZW50Tm9ybWFsOwotICAgIGFtLmNvbG9yLmZldGNoID0gY3VycmVudENvbG9yOwotICAgIAotICAgIGlmIChhbS52ZXJ0ZXguZW5hYmxlKSB7Ci0gICAgICAgIGFtLnZlcnRleC5yZXNvbHZlKCk7Ci0gICAgICAgIGlmIChhbS52ZXJ0ZXguYm8gfHwgYW0udmVydGV4LnBvaW50ZXIpIHsKLSAgICAgICAgICAgIGFtLnZlcnRleC5mZXRjaCA9IHZlcnRleF9mY3RbYW0udmVydGV4LnNpemUtMl1bYW0udmVydGV4LnR5cGUgJiAweEZdOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgaWYgKGFtLm5vcm1hbC5lbmFibGUpIHsKLSAgICAgICAgYW0ubm9ybWFsLnJlc29sdmUoKTsKLSAgICAgICAgaWYgKGFtLm5vcm1hbC5ibyB8fCBhbS5ub3JtYWwucG9pbnRlcikgewotICAgICAgICAgICAgYW0ubm9ybWFsLmZldGNoID0gbm9ybWFsX2ZjdFthbS5ub3JtYWwuc2l6ZS0zXVthbS5ub3JtYWwudHlwZSAmIDB4Rl07Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBpZiAoYW0uY29sb3IuZW5hYmxlKSB7Ci0gICAgICAgIGFtLmNvbG9yLnJlc29sdmUoKTsKLSAgICAgICAgaWYgKGMtPmxpZ2h0aW5nLmVuYWJsZSkgewotICAgICAgICAgICAgaWYgKGFtLmNvbG9yLmJvIHx8IGFtLmNvbG9yLnBvaW50ZXIpIHsKLSAgICAgICAgICAgICAgICBhbS5jb2xvci5mZXRjaCA9IGNvbG9yX2ZjdFthbS5jb2xvci5zaXplLTNdW2FtLmNvbG9yLnR5cGUgJiAweEZdOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaWYgKGFtLmNvbG9yLmJvIHx8IGFtLmNvbG9yLnBvaW50ZXIpIHsKLSAgICAgICAgICAgICAgICBhbS5jb2xvci5mZXRjaCA9IGNvbG9yX2NsYW1wX2ZjdFthbS5jb2xvci5zaXplLTNdW2FtLmNvbG9yLnR5cGUgJiAweEZdOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgaW50IGFjdGl2ZVRtdUNvdW50ID0gMDsKLSAgICBmb3IgKGludCBpPTAgOyBpPEdHTF9URVhUVVJFX1VOSVRfQ09VTlQgOyBpKyspIHsKLSAgICAgICAgYW0udGV4dHVyZVtpXS5mZXRjaCA9IGN1cnJlbnRUZXhDb29yZDsKLSAgICAgICAgaWYgKGMtPnJhc3Rlcml6ZXIuc3RhdGUudGV4dHVyZVtpXS5lbmFibGUpIHsKLQotICAgICAgICAgICAgLy8gdGV4dHVyZSBmZXRjaGVycy4uLgotICAgICAgICAgICAgaWYgKGFtLnRleHR1cmVbaV0uZW5hYmxlKSB7Ci0gICAgICAgICAgICAgICAgYW0udGV4dHVyZVtpXS5yZXNvbHZlKCk7Ci0gICAgICAgICAgICAgICAgaWYgKGFtLnRleHR1cmVbaV0uYm8gfHwgYW0udGV4dHVyZVtpXS5wb2ludGVyKSB7Ci0gICAgICAgICAgICAgICAgICAgIGFtLnRleHR1cmVbaV0uZmV0Y2ggPSB0ZXh0dXJlX2ZjdFthbS50ZXh0dXJlW2ldLnNpemUtMl1bYW0udGV4dHVyZVtpXS50eXBlICYgMHhGXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIHRleHR1cmUgdHJhbnNmb3JtLi4uCi0gICAgICAgICAgICBjb25zdCBpbnQgaW5kZXggPSBjLT5hcnJheXMudGV4dHVyZVtpXS5zaXplIC0gMjsKLSAgICAgICAgICAgIGMtPmFycmF5cy50ZXhfdHJhbnNmb3JtW2ldID0KLSAgICAgICAgICAgICAgICBjLT50cmFuc2Zvcm1zLnRleHR1cmVbaV0udHJhbnNmb3JtLnBvaW50dltpbmRleF07Ci0KLSAgICAgICAgICAgIGFtLnRtdSA9IGk7Ci0gICAgICAgICAgICBhY3RpdmVUbXVDb3VudCsrOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gcGljayB0aGUgdmVydGV4LWNsaXBwZXIKLSAgICB1aW50MzJfdCBjbGlwcGVyID0gMDsKLSAgICAvLyB3ZSBtdXN0IHJlbG9hZCAnZW5hYmxlcycgaGVyZQotICAgIGVuYWJsZXMgPSBjLT5yYXN0ZXJpemVyLnN0YXRlLmVuYWJsZXM7Ci0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX1NNT09USCkKLSAgICAgICAgY2xpcHBlciB8PSAxOyAgIC8vIHdlIG5lZWQgdG8gaW50ZXJwb2xhdGUgY29sb3JzCi0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX1RNVVMpCi0gICAgICAgIGNsaXBwZXIgfD0gMjsgICAvLyB3ZSBuZWVkIHRvIGludGVycG9sYXRlIHRleHR1cmVzCi0gICAgc3dpdGNoIChjbGlwcGVyKSB7Ci0gICAgY2FzZSAwOiBjLT5hcnJheXMuY2xpcFZlcnRleCA9IGNsaXBWZXJ0ZXg7ICAgICAgYnJlYWs7Ci0gICAgY2FzZSAxOiBjLT5hcnJheXMuY2xpcFZlcnRleCA9IGNsaXBWZXJ0ZXhDOyAgICAgYnJlYWs7Ci0gICAgY2FzZSAyOiBjLT5hcnJheXMuY2xpcFZlcnRleCA9IGNsaXBWZXJ0ZXhUOyAgICAgYnJlYWs7Ci0gICAgY2FzZSAzOiBjLT5hcnJheXMuY2xpcFZlcnRleCA9IGNsaXBWZXJ0ZXhBbGw7ICAgYnJlYWs7Ci0gICAgfQotICAgIGMtPmFycmF5cy5jbGlwRXllID0gY2xpcEV5ZTsKLQotICAgIC8vIHBpY2sgdGhlIHByaW1pdGl2ZSByYXN0ZXJpemVyCi0gICAgb2dsZXNfdmFsaWRhdGVfcHJpbWl0aXZlcyhjKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdXNpbmcgbmFtZXNwYWNlIGFuZHJvaWQ7Ci0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIGFycmF5IEFQSQotI2VuZGlmCi0KLXZvaWQgZ2xWZXJ0ZXhQb2ludGVyKAotICAgIEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKHNpemU8MiB8fCBzaXplPjQgfHwgc3RyaWRlPDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgc3dpdGNoICh0eXBlKSB7Ci0gICAgY2FzZSBHTF9CWVRFOgotICAgIGNhc2UgR0xfU0hPUlQ6Ci0gICAgY2FzZSBHTF9GSVhFRDoKLSAgICBjYXNlIEdMX0ZMT0FUOgotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGMtPmFycmF5cy52ZXJ0ZXguaW5pdChzaXplLCB0eXBlLCBzdHJpZGUsIHBvaW50ZXIsIGMtPmFycmF5cy5hcnJheV9idWZmZXIsIDApOwotfQotCi12b2lkIGdsQ29sb3JQb2ludGVyKAotICAgIEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgLy8gaW4gdGhlb3J5IG9nbGVzIGRvZXNuJ3QgYWxsb3cgY29sb3IgYXJyYXlzIG9mIHNpemUgMwotICAgIC8vIGJ1dCBpdCBpcyB2ZXJ5IHVzZWZ1bCB0byAndmlzdWFsaXplJyB0aGUgbm9ybWFsIGFycmF5LgotICAgIGlmIChzaXplPDMgfHwgc2l6ZT40IHx8IHN0cmlkZTwwKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIHN3aXRjaCAodHlwZSkgewotICAgIGNhc2UgR0xfVU5TSUdORURfQllURToKLSAgICBjYXNlIEdMX0ZJWEVEOgotICAgIGNhc2UgR0xfRkxPQVQ6Ci0gICAgICAgIGJyZWFrOwotICAgIGRlZmF1bHQ6Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgYy0+YXJyYXlzLmNvbG9yLmluaXQoc2l6ZSwgdHlwZSwgc3RyaWRlLCBwb2ludGVyLCBjLT5hcnJheXMuYXJyYXlfYnVmZmVyLCAwKTsKLX0KLQotdm9pZCBnbE5vcm1hbFBvaW50ZXIoCi0gICAgR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBpZiAoc3RyaWRlPDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgc3dpdGNoICh0eXBlKSB7Ci0gICAgY2FzZSBHTF9CWVRFOgotICAgIGNhc2UgR0xfU0hPUlQ6Ci0gICAgY2FzZSBHTF9GSVhFRDoKLSAgICBjYXNlIEdMX0ZMT0FUOgotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGMtPmFycmF5cy5ub3JtYWwuaW5pdCgzLCB0eXBlLCBzdHJpZGUsIHBvaW50ZXIsIGMtPmFycmF5cy5hcnJheV9idWZmZXIsIDApOwotfQotCi12b2lkIGdsVGV4Q29vcmRQb2ludGVyKAotICAgIEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKHNpemU8MiB8fCBzaXplPjQgfHwgc3RyaWRlPDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgc3dpdGNoICh0eXBlKSB7Ci0gICAgY2FzZSBHTF9CWVRFOgotICAgIGNhc2UgR0xfU0hPUlQ6Ci0gICAgY2FzZSBHTF9GSVhFRDoKLSAgICBjYXNlIEdMX0ZMT0FUOgotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGNvbnN0IGludCB0bXUgPSBjLT5hcnJheXMuYWN0aXZlVGV4dHVyZTsKLSAgICBjLT5hcnJheXMudGV4dHVyZVt0bXVdLmluaXQoc2l6ZSwgdHlwZSwgc3RyaWRlLCBwb2ludGVyLAotICAgICAgICAgICAgYy0+YXJyYXlzLmFycmF5X2J1ZmZlciwgMCk7Ci19Ci0KLQotdm9pZCBnbEVuYWJsZUNsaWVudFN0YXRlKEdMZW51bSBhcnJheSkgewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgZW5hYmxlRGlzYWJsZUNsaWVudFN0YXRlKGMsIGFycmF5LCB0cnVlKTsKLX0KLQotdm9pZCBnbERpc2FibGVDbGllbnRTdGF0ZShHTGVudW0gYXJyYXkpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGVuYWJsZURpc2FibGVDbGllbnRTdGF0ZShjLCBhcnJheSwgZmFsc2UpOwotfQotCi12b2lkIGdsQ2xpZW50QWN0aXZlVGV4dHVyZShHTGVudW0gdGV4dHVyZSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICh0ZXh0dXJlPEdMX1RFWFRVUkUwIHx8IHRleHR1cmU+PUdMX1RFWFRVUkUwK0dHTF9URVhUVVJFX1VOSVRfQ09VTlQpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBjLT5hcnJheXMuYWN0aXZlVGV4dHVyZSA9IHRleHR1cmUgLSBHTF9URVhUVVJFMDsKLX0KLQotdm9pZCBnbERyYXdBcnJheXMoR0xlbnVtIG1vZGUsIEdMaW50IGZpcnN0LCBHTHNpemVpIGNvdW50KQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKGNvdW50PDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgc3dpdGNoIChtb2RlKSB7Ci0gICAgY2FzZSBHTF9QT0lOVFM6Ci0gICAgY2FzZSBHTF9MSU5FX1NUUklQOgotICAgIGNhc2UgR0xfTElORV9MT09QOgotICAgIGNhc2UgR0xfTElORVM6Ci0gICAgY2FzZSBHTF9UUklBTkdMRV9TVFJJUDoKLSAgICBjYXNlIEdMX1RSSUFOR0xFX0ZBTjoKLSAgICBjYXNlIEdMX1RSSUFOR0xFUzoKLSAgICAgICAgYnJlYWs7Ci0gICAgZGVmYXVsdDoKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIGlmIChjb3VudCA9PSAwIHx8ICFjLT5hcnJheXMudmVydGV4LmVuYWJsZSkKLSAgICAgICAgcmV0dXJuOwotICAgIGlmICgoYy0+Y3VsbC5lbmFibGUpICYmIChjLT5jdWxsLmN1bGxGYWNlID09IEdMX0ZST05UX0FORF9CQUNLKSkKLSAgICAgICAgcmV0dXJuOyAvLyBhbGwgdHJpYW5nbGVzIGFyZSBjdWxsZWQKLQotICAgIHZhbGlkYXRlX2FycmF5cyhjLCBtb2RlKTsKLSAgICBkcmF3QXJyYXlzUHJpbXNbbW9kZV0oYywgZmlyc3QsIGNvdW50KTsKLQotI2lmIFZDX0NBQ0hFX1NUQVRJU1RJQ1MKLSAgICBjLT52Yy50b3RhbCA9IGNvdW50OwotICAgIGMtPnZjLmR1bXBfc3RhdHMobW9kZSk7Ci0jZW5kaWYKLX0KLQotdm9pZCBnbERyYXdFbGVtZW50cygKLSAgICBHTGVudW0gbW9kZSwgR0xzaXplaSBjb3VudCwgR0xlbnVtIHR5cGUsIGNvbnN0IEdMdm9pZCAqaW5kaWNlcykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmIChjb3VudDwwKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIHN3aXRjaCAobW9kZSkgewotICAgIGNhc2UgR0xfUE9JTlRTOgotICAgIGNhc2UgR0xfTElORV9TVFJJUDoKLSAgICBjYXNlIEdMX0xJTkVfTE9PUDoKLSAgICBjYXNlIEdMX0xJTkVTOgotICAgIGNhc2UgR0xfVFJJQU5HTEVfU1RSSVA6Ci0gICAgY2FzZSBHTF9UUklBTkdMRV9GQU46Ci0gICAgY2FzZSBHTF9UUklBTkdMRVM6Ci0gICAgICAgIGJyZWFrOwotICAgIGRlZmF1bHQ6Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgc3dpdGNoICh0eXBlKSB7Ci0gICAgY2FzZSBHTF9VTlNJR05FRF9CWVRFOgotICAgIGNhc2UgR0xfVU5TSUdORURfU0hPUlQ6Ci0gICAgICAgIGMtPmFycmF5cy5pbmRpY2VzVHlwZSA9IHR5cGU7Ci0gICAgICAgIGJyZWFrOwotICAgIGRlZmF1bHQ6Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKGNvdW50ID09IDAgfHwgIWMtPmFycmF5cy52ZXJ0ZXguZW5hYmxlKQotICAgICAgICByZXR1cm47Ci0gICAgaWYgKChjLT5jdWxsLmVuYWJsZSkgJiYgKGMtPmN1bGwuY3VsbEZhY2UgPT0gR0xfRlJPTlRfQU5EX0JBQ0spKQotICAgICAgICByZXR1cm47IC8vIGFsbCB0cmlhbmdsZXMgYXJlIGN1bGxlZAotCi0gICAgLy8gY2xlYXIgdGhlIHZlcnRleC1jYWNoZQotICAgIGMtPnZjLmNsZWFyKCk7Ci0gICAgdmFsaWRhdGVfYXJyYXlzKGMsIG1vZGUpOwotICAgIAotICAgIC8vIGlmIGluZGljZXMgYXJlIGluIGEgYnVmZmVyIG9iamVjdCwgdGhlIHBvaW50ZXIgaXMgdHJlYXRlZCBhcyBhbgotICAgIC8vIG9mZnNldCBpbiB0aGF0IGJ1ZmZlci4KLSAgICBpZiAoYy0+YXJyYXlzLmVsZW1lbnRfYXJyYXlfYnVmZmVyKSB7Ci0gICAgICAgIGluZGljZXMgPSBjLT5hcnJheXMuZWxlbWVudF9hcnJheV9idWZmZXItPmRhdGEgKyB1aW50cHRyX3QoaW5kaWNlcyk7Ci0gICAgfQotCi0gICAgZHJhd0VsZW1lbnRzUHJpbXNbbW9kZV0oYywgY291bnQsIGluZGljZXMpOwotCi0jaWYgVkNfQ0FDSEVfU1RBVElTVElDUwotICAgIGMtPnZjLnRvdGFsID0gY291bnQ7Ci0gICAgYy0+dmMuZHVtcF9zdGF0cyhtb2RlKTsKLSNlbmRpZgotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBidWZmZXJzCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXZvaWQgZ2xCaW5kQnVmZmVyKEdMZW51bSB0YXJnZXQsIEdMdWludCBidWZmZXIpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBpZiAoKHRhcmdldCE9R0xfQVJSQVlfQlVGRkVSKSAmJiAodGFyZ2V0IT1HTF9FTEVNRU5UX0FSUkFZX0JVRkZFUikpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICAvLyBjcmVhdGUgYSBidWZmZXIgb2JqZWN0LCBvciBiaW5kIGFuIGV4aXN0aW5nIG9uZQotICAgIGJ1ZmZlcl90IGNvbnN0KiBibyA9IDA7Ci0gICAgaWYgKGJ1ZmZlcikgewotICAgICAgICBibyA9IGMtPmJ1ZmZlck9iamVjdE1hbmFnZXItPmJpbmQoYnVmZmVyKTsKLSAgICAgICAgaWYgKCFibykgewotICAgICAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfT1VUX09GX01FTU9SWSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgKCh0YXJnZXQgPT0gR0xfQVJSQVlfQlVGRkVSKSA/IAotICAgICAgICAgICAgYy0+YXJyYXlzLmFycmF5X2J1ZmZlciA6IGMtPmFycmF5cy5lbGVtZW50X2FycmF5X2J1ZmZlcikgPSBibzsKLX0KLQotdm9pZCBnbEJ1ZmZlckRhdGEoR0xlbnVtIHRhcmdldCwgR0xzaXplaXB0ciBzaXplLCBjb25zdCBHTHZvaWQqIGRhdGEsIEdMZW51bSB1c2FnZSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICgodGFyZ2V0IT1HTF9BUlJBWV9CVUZGRVIpICYmICh0YXJnZXQhPUdMX0VMRU1FTlRfQVJSQVlfQlVGRkVSKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmIChzaXplPDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKCh1c2FnZSE9R0xfU1RBVElDX0RSQVcpICYmICh1c2FnZSE9R0xfRFlOQU1JQ19EUkFXKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGJ1ZmZlcl90IGNvbnN0KiBibyA9ICgodGFyZ2V0ID09IEdMX0FSUkFZX0JVRkZFUikgPyAKLSAgICAgICAgICAgIGMtPmFycmF5cy5hcnJheV9idWZmZXIgOiBjLT5hcnJheXMuZWxlbWVudF9hcnJheV9idWZmZXIpOwotCi0gICAgaWYgKGJvID09IDApIHsKLSAgICAgICAgLy8gY2FuJ3QgbW9kaWZ5IGJ1ZmZlciAwCi0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfT1BFUkFUSU9OKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIGJ1ZmZlcl90KiBlZGl0X2JvID0gY29uc3RfY2FzdDxidWZmZXJfdCo+KGJvKTsKLSAgICBpZiAoYy0+YnVmZmVyT2JqZWN0TWFuYWdlci0+YWxsb2NhdGVTdG9yZShlZGl0X2JvLCBzaXplLCB1c2FnZSkgIT0gMCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9PVVRfT0ZfTUVNT1JZKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBpZiAoZGF0YSkgewotICAgICAgICBtZW1jcHkoYm8tPmRhdGEsIGRhdGEsIHNpemUpOwotICAgIH0KLX0KLQotdm9pZCBnbEJ1ZmZlclN1YkRhdGEoR0xlbnVtIHRhcmdldCwgR0xpbnRwdHIgb2Zmc2V0LCBHTHNpemVpcHRyIHNpemUsIGNvbnN0IEdMdm9pZCogZGF0YSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICgodGFyZ2V0IT1HTF9BUlJBWV9CVUZGRVIpICYmICh0YXJnZXQhPUdMX0VMRU1FTlRfQVJSQVlfQlVGRkVSKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmIChvZmZzZXQ8MCB8fCBzaXplPDAgfHwgZGF0YT09MCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX1ZBTFVFKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBidWZmZXJfdCBjb25zdCogYm8gPSAoKHRhcmdldCA9PSBHTF9BUlJBWV9CVUZGRVIpID8gCi0gICAgICAgICAgICBjLT5hcnJheXMuYXJyYXlfYnVmZmVyIDogYy0+YXJyYXlzLmVsZW1lbnRfYXJyYXlfYnVmZmVyKTsKLQotICAgIGlmIChibyA9PSAwKSB7Ci0gICAgICAgIC8vIGNhbid0IG1vZGlmeSBidWZmZXIgMAotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX09QRVJBVElPTik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKG9mZnNldCtzaXplID4gYm8tPnNpemUpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgbWVtY3B5KGJvLT5kYXRhICsgb2Zmc2V0LCBkYXRhLCBzaXplKTsKLX0KLQotdm9pZCBnbERlbGV0ZUJ1ZmZlcnMoR0xzaXplaSBuLCBjb25zdCBHTHVpbnQqIGJ1ZmZlcnMpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBpZiAobjwwKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgZm9yIChpbnQgaT0wIDsgaTxuIDsgaSsrKSB7Ci0gICAgICAgIEdMdWludCBuYW1lID0gYnVmZmVyc1tpXTsKLSAgICAgICAgaWYgKG5hbWUpIHsKLSAgICAgICAgICAgIC8vIHVuYmluZCBib3VuZCBkZWxldGVkIGJ1ZmZlcnMuLi4KLSAgICAgICAgICAgIGlmIChjLT5hcnJheXMuZWxlbWVudF9hcnJheV9idWZmZXItPm5hbWUgPT0gbmFtZSkgewotICAgICAgICAgICAgICAgIGMtPmFycmF5cy5lbGVtZW50X2FycmF5X2J1ZmZlciA9IDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoYy0+YXJyYXlzLmFycmF5X2J1ZmZlci0+bmFtZSA9PSBuYW1lKSB7Ci0gICAgICAgICAgICAgICAgYy0+YXJyYXlzLmFycmF5X2J1ZmZlciA9IDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoYy0+YXJyYXlzLnZlcnRleC5iby0+bmFtZSA9PSBuYW1lKSB7Ci0gICAgICAgICAgICAgICAgYy0+YXJyYXlzLnZlcnRleC5ibyA9IDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoYy0+YXJyYXlzLm5vcm1hbC5iby0+bmFtZSA9PSBuYW1lKSB7Ci0gICAgICAgICAgICAgICAgYy0+YXJyYXlzLm5vcm1hbC5ibyA9IDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBpZiAoYy0+YXJyYXlzLmNvbG9yLmJvLT5uYW1lID09IG5hbWUpIHsKLSAgICAgICAgICAgICAgICBjLT5hcnJheXMuY29sb3IuYm8gPSAwOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZm9yIChpbnQgdD0wIDsgdDxHR0xfVEVYVFVSRV9VTklUX0NPVU5UIDsgdCsrKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGMtPmFycmF5cy50ZXh0dXJlW3RdLmJvLT5uYW1lID09IG5hbWUpIHsKLSAgICAgICAgICAgICAgICAgICAgYy0+YXJyYXlzLnRleHR1cmVbdF0uYm8gPSAwOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0gICAgCi0gICAgYy0+YnVmZmVyT2JqZWN0TWFuYWdlci0+ZGVsZXRlQnVmZmVycyhuLCBidWZmZXJzKTsKLSAgICBjLT5idWZmZXJPYmplY3RNYW5hZ2VyLT5yZWN5Y2xlVG9rZW5zKG4sIGJ1ZmZlcnMpOwotfQotCi12b2lkIGdsR2VuQnVmZmVycyhHTHNpemVpIG4sIEdMdWludCogYnVmZmVycykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmIChuPDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgYy0+YnVmZmVyT2JqZWN0TWFuYWdlci0+Z2V0VG9rZW4obiwgYnVmZmVycyk7Ci19CmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL2FycmF5LmggYi9vcGVuZ2wvbGliYWdsL2FycmF5LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGUxNTY5NzguLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9hcnJheS5oCisrKyAvZGV2L251bGwKQEAgLTEsMzcgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9hcnJheS5oCi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2lmbmRlZiBBTkRST0lEX09QRU5HTEVTX0FSUkFZX0gKLSNkZWZpbmUgQU5EUk9JRF9PUEVOR0xFU19BUlJBWV9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdGRkZWYuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1uYW1lc3BhY2UgZ2wgewotc3RydWN0IG9nbGVzX2NvbnRleHRfdDsKLX07Ci0KLXZvaWQgb2dsZXNfaW5pdF9hcnJheShvZ2xlc19jb250ZXh0X3QqIGMpOwotdm9pZCBvZ2xlc191bmluaXRfYXJyYXkob2dsZXNfY29udGV4dF90KiBjKTsKLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotI2VuZGlmIC8vIEFORFJPSURfT1BFTkdMRVNfQVJSQVlfSAotCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL2NvbnRleHQuaCBiL29wZW5nbC9saWJhZ2wvY29udGV4dC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlZjM2YjU2Li4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJhZ2wvY29udGV4dC5oCisrKyAvZGV2L251bGwKQEAgLTEsMjAgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9jb250ZXh0LmgKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jaW5jbHVkZSA8cHJpdmF0ZS9vcGVuZ2xlcy9nbF9jb250ZXh0Lmg+Ci0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOjpnbDsKZGlmZiAtLWdpdCBhL29wZW5nbC9saWJhZ2wvZHh0LmNwcCBiL29wZW5nbC9saWJhZ2wvZHh0LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjM4YzgxZi4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGliYWdsL2R4dC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw2MzYgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9keHQuY3BwCi0qKgotKiogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2RlZmluZSBUSU1JTkcgMAotCi0jaWYgVElNSU5HCi0jaW5jbHVkZSA8c3lzL3RpbWUuaD4gLy8gZm9yIG9wdGltaXphdGlvbiB0aW1pbmcKLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2VuZGlmCi0KLSNpbmNsdWRlIDxHTEVTL2dsLmg+Ci0jaW5jbHVkZSA8dXRpbHMvRW5kaWFuLmg+Ci0KLSNpbmNsdWRlICJjb250ZXh0LmgiCi0KLSNkZWZpbmUgVElNSU5HIDAKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1zdGF0aWMgdWludDhfdCBhdmcyM3RhYls2NCo2NF07Ci1zdGF0aWMgdm9sYXRpbGUgaW50IHRhYmxlc19pbml0aWFsaXplZCA9IDA7Ci0KLS8vIERlZmluaXRpb25zIGJlbG93IGFyZSBlcXVpdmFsZW50IHRvIHRoZXNlIG92ZXIgdGhlIHZhbGlkIHJhbmdlIG9mIGFyZ3VtZW50cwotLy8gICNkZWZpbmUgZGl2NSh4KSAoKHgpLzUpCi0vLyAgI2RlZmluZSBkaXY3KHgpICgoeCkvNykKLQotLy8gVXNlIGZpeGVkLXBvaW50IHRvIGRpdmlkZSBieSA1IGFuZCA3Ci0vLyAzMjc3ID0gMl4xNC81ICsgMQotLy8gMjM0MSA9IDJeMTQvNyArIDEKLSNkZWZpbmUgZGl2NSh4KSAoKCh4KSozMjc3KSA+PiAxNCkKLSNkZWZpbmUgZGl2Nyh4KSAoKCh4KSoyMzQxKSA+PiAxNCkKLQotLy8gVGFibGUgd2l0aCBlbnRyeSBbYSA8PCA2IHwgYl0gPSAoMiphICsgYikvMyBmb3IgMCA8PSBhLGIgPCA2NAotI2RlZmluZSBhdmcyMyh4MCx4MSkgYXZnMjN0YWJbKCh4MCkgPDwgNikgfCAoeDEpXQotCi0vLyBFeHRyYWN0IDUvNi81IFJHQgotI2RlZmluZSByZWQoeCkgICAoKCh4KSA+PiAxMSkgJiAweDFmKQotI2RlZmluZSBncmVlbih4KSAoKCh4KSA+PiAgNSkgJiAweDNmKQotI2RlZmluZSBibHVlKHgpICAoICh4KSAgICAgICAgJiAweDFmKQotCi0vKgotICogQ29udmVydCA1LzYvNSBSR0IgKGFzIDMgaW50cykgdG8gOC84LzgKLSAqCi0gKiBPcGVyYXRpb24gY291bnQ6IDggPDwsIDAgJiwgNSB8Ci0gKi8KLWlubGluZSBzdGF0aWMgaW50IHJnYjU2NVNlcFRvODg4KGludCByLCBpbnQgZywgaW50IGIpCi0KLXsKLSAgICByZXR1cm4gKCgoKHIgPDwgMykgfCAociA+PiAyKSkgPDwgMTYpIHwKLSAgICAgICAgICAgICgoKGcgPDwgMikgfCAoZyA+PiA0KSkgPDwgIDgpIHwKLSAgICAgICAgICAgICAoKGIgPDwgMykgfCAoYiA+PiAyKSkpOwotfQotCi0vKgotICogQ29udmVydCA1LzYvNSBSR0IgKGFzIGEgc2luZ2xlIDE2LWJpdCB3b3JkKSB0byA4LzgvOAotICoKLSAqICAgICAgICAgICAgICAgICAgIHI0cjNyMnIxIHIwZzVnNGczIGcyZzFnMGI0IGIzYjJiMWIwICAgcmdiCi0gKiAgICAgICAgICAgIHI0cjNyMiByMXIwZzVnNCBnM2cyZzFnMCBiNGIzYjJiMSBiMCAwIDAgMCAgIHJnYiA8PCAzCi0gKiByNHIzcjJyMSByMHI0cjNyMiBnNWc0ZzNnMiBnMWcwZzVnNCBiNGIzYjJiMSBiMGI0YjNiMiAgIGRlc2lyZWQgcmVzdWx0Ci0gKgotICogQ29uc3RydWN0IHRoZSAyNC1iaXQgUkdCIHdvcmQgYXM6Ci0gKgotICogcjRyM3IycjEgcjAtLS0tLS0gLS0tLS0tLS0gLS0tLS0tLS0gLS0tLS0tLS0gLS0tLS0tLS0gIChyZ2IgPDwgOCkgJiAweGY4MDAwMAotICogICAgICAgICAgICByNHIzcjIgLS0tLS0tLS0gLS0tLS0tLS0gLS0tLS0tLS0gLS0tLS0tLS0gIChyZ2IgPDwgMykgJiAweDA3MDAwMAotICogICAgICAgICAgICAgICAgICAgZzVnNGczZzIgZzFnMC0tLS0gLS0tLS0tLS0gLS0tLS0tLS0gIChyZ2IgPDwgNSkgJiAweDAwZmMwMAotICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGc1ZzQgLS0tLS0tLS0gLS0tLS0tLS0gIChyZ2IgPj4gMSkgJiAweDAwMDMwMAotICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYjRiM2IyYjEgYjAtLS0tLS0gIChyZ2IgPDwgMykgJiAweDAwMDBmOAotICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBiNGIzYjIgIChyZ2IgPj4gMikgJiAweDAwMDAwNwotICoKLSAqIE9wZXJhdGlvbiBjb3VudDogNSA8PCwgNiAmLCA1IHwgKG4uYi4gcmdiID4+IDMgaXMgdXNlZCB0d2ljZSkKLSAqLwotaW5saW5lIHN0YXRpYyBpbnQgcmdiNTY1VG84ODgoaW50IHJnYikKLQotewotICAgIGludCByZ2IzID0gcmdiID4+IDM7Ci0gICAgcmV0dXJuICgoKHJnYiA8PCA4KSAmIDB4ZjgwMDAwKSB8Ci0gICAgICAgICAgICAoIHJnYjMgICAgICAmIDB4MDcwMDAwKSB8Ci0gICAgICAgICAgICAoKHJnYiA8PCA1KSAmIDB4MDBmYzAwKSB8Ci0gICAgICAgICAgICAoKHJnYiA+PiAxKSAmIDB4MDAwMzAwKSB8Ci0gICAgICAgICAgICAoIHJnYjMgICAgICAmIDB4MDAwMGY4KSB8Ci0gICAgICAgICAgICAoKHJnYiA+PiAyKSAmIDB4MDAwMDA3KSk7Ci19Ci0KLSNpZiBfX0JZVEVfT1JERVIgPT0gX19CSUdfRU5ESUFOCi1zdGF0aWMgdWludDMyX3Qgc3dhcCh1aW50MzJfdCB4KSB7Ci0gICAgaW50IGIwID0gKHggPj4gMjQpICYgMHhmZjsKLSAgICBpbnQgYjEgPSAoeCA+PiAxNikgJiAweGZmOwotICAgIGludCBiMiA9ICh4ID4+ICA4KSAmIDB4ZmY7Ci0gICAgaW50IGIzID0gKHggICAgICApICYgMHhmZjsKLSAgICAKLSAgICByZXR1cm4gKHVpbnQzMl90KSgoYjMgPDwgMjQpIHwgKGIyIDw8IDE2KSB8IChiMSA8PCA4KSB8IGIwKTsKLX0KLSNlbmRpZgotCi1zdGF0aWMgdm9pZAotaW5pdF90YWJsZXMoKQotewotICAgIGlmICh0YWJsZXNfaW5pdGlhbGl6ZWQpIHsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIGZvciAoaW50IGkgPSAwOyBpIDwgNjQ7IGkrKykgewotICAgICAgICBmb3IgKGludCBqID0gMDsgaiA8IDY0OyBqKyspIHsKLSAgICAgICAgICAgIGludCBhdmcgPSAoMippICsgaikvMzsKLSAgICAgICAgICAgIGF2ZzIzdGFiWyhpIDw8IDYpIHwgal0gPSBhdmc7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBhc20gdm9sYXRpbGUgKCIiIDogOiA6ICJtZW1vcnkiKTsKLSAgICB0YWJsZXNfaW5pdGlhbGl6ZWQgPSAxOwotfQotCi0vKgotICogVXRpbGl0eSB0byBzY2FuIGEgRFhUMSBjb21wcmVzc2VkIHRleHR1cmUgdG8gZGV0ZXJtaW5lIHdoZXRoZXIgaXQKLSAqIGNvbnRhaW5zIGEgdHJhbnNwYXJlbnQgcGl4ZWwgKGNvbG9yMCA8IGNvbG9yMSwgY29kZSA9PSAzKS4gIFRoaXMKLSAqIG1heSBiZSB1c2VmdWwgaWYgdGhlIGFwcGxpY2F0aW9uIGxhY2tzIGluZm9ybWF0aW9uIGFzIHRvIHdoZXRoZXIKLSAqIHRoZSB0cnVlIGZvcm1hdCBpcyBHTF9DT01QUkVTU0VEX1JHQl9TM1RDX0RYVDFfRVhUIG9yCi0gKiBHTF9DT01QUkVTU0VEX1JHQkFfUzNUQ19EWFQxX0VYVC4KLSAqLwotYm9vbAotRFhUMUhhc0FscGhhKGNvbnN0IEdMdm9pZCAqZGF0YSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KSB7ICAgIAotI2lmIFRJTUlORwotICAgIHN0cnVjdCB0aW1ldmFsIHN0YXJ0X3QsIGVuZF90OwotICAgIHN0cnVjdCB0aW1lem9uZSB0ejsKLSAgICAKLSAgICBnZXR0aW1lb2ZkYXkoJnN0YXJ0X3QsICZ0eik7Ci0jZW5kaWYKLQotICAgIGJvb2wgaGFzQWxwaGEgPSBmYWxzZTsKLQotICAgIGludCB4YmxvY2tzID0gKHdpZHRoICsgMykvNDsKLSAgICBpbnQgeWJsb2NrcyA9IChoZWlnaHQgKyAzKS80OwotICAgIGludCBudW1ibG9ja3MgPSB4YmxvY2tzKnlibG9ja3M7Ci0KLSAgICB1aW50MzJfdCBjb25zdCAqZDMyID0gKHVpbnQzMl90ICopZGF0YTsKLSAgICBmb3IgKGludCBiID0gMDsgYiA8IG51bWJsb2NrczsgYisrKSB7Ci0gICAgICAgIHVpbnQzMl90IGNvbG9ycyA9ICpkMzIrKzsKLSAgICAgICAgCi0jaWYgX19CWVRFX09SREVSID09IF9fQklHX0VORElBTgotICAgICAgICBjb2xvcnMgPSBzd2FwKGNvbG9ycyk7Ci0jZW5kaWYKLSAgICAgICAgCi0gICAgICAgIHVpbnQxNl90IGNvbG9yMCA9IGNvbG9ycyAmIDB4ZmZmZjsKLSAgICAgICAgdWludDE2X3QgY29sb3IxID0gY29sb3JzID4+IDE2OwotICAgICAgICAKLSAgICAgICAgaWYgKGNvbG9yMCA8IGNvbG9yMSkgewotICAgICAgICAgICAgLy8gVGhlcmUncyBubyBuZWVkIHRvIGVuZGlhbi1zd2FwIHdpdGhpbiAnYml0cycKLSAgICAgICAgICAgIC8vIHNpbmNlIHdlIGRvbid0IGNhcmUgd2hpY2ggcGl4ZWwgaXMgdGhlIHRyYW5zcGFyZW50IG9uZQotICAgICAgICAgICAgdWludDMyX3QgYml0cyA9ICpkMzIrKzsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgLy8gRGV0ZWN0IGlmIGFueSAob2RkLCBldmVuKSBwYWlyIG9mIGJpdHMgYXJlICcxMScKLSAgICAgICAgICAgIC8vICAgICAgYml0czogYjMxIGIzMCBiMjkgLi4uIGIzIGIyIGIxIGIwCi0gICAgICAgICAgICAvLyBiaXRzID4+IDE6IGIzMSBiMzEgYjMwIC4uLiBiNCBiMyBiMiBiMQotICAgICAgICAgICAgLy8gICAgICAgICAmOiBiMzEgKGIzMSAmIGIzMCkgKGIyOSAmIGIyOCkgLi4uIChiMiAmIGIxKSAoYjEgJiBiMCkKLSAgICAgICAgICAgIC8vICAmIDB4NTUuLjogICAwIChiMzEgJiBiMzApICAgICAgIDAgICAgIC4uLiAgICAgMCAgICAgKGIxICYgYjApCi0gICAgICAgICAgICBpZiAoKChiaXRzICYgKGJpdHMgPj4gMSkpICYgMHg1NTU1NTU1NSkgIT0gMCkgewotICAgICAgICAgICAgICAgIGhhc0FscGhhID0gdHJ1ZTsKLSAgICAgICAgICAgICAgICBnb3RvIGRvbmU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBTa2lwIDQgYnl0ZXMKLSAgICAgICAgICAgICsrZDMyOwotICAgICAgICB9Ci0gICAgfQotICAgIAotIGRvbmU6Ci0jaWYgVElNSU5HCi0gICAgZ2V0dGltZW9mZGF5KCZlbmRfdCwgJnR6KTsKLSAgICBsb25nIHVzZWMgPSAoZW5kX3QudHZfc2VjIC0gc3RhcnRfdC50dl9zZWMpKjEwMDAwMDAgKwotICAgICAgICAoZW5kX3QudHZfdXNlYyAtIHN0YXJ0X3QudHZfdXNlYyk7Ci0gICAgCi0gICAgcHJpbnRmKCJTY2FubmVkIHc9JWQgaD0lZCBpbiAlbGQgdXNlY1xuIiwgd2lkdGgsIGhlaWdodCwgdXNlYyk7Ci0jZW5kaWYKLSAgICAKLSAgICByZXR1cm4gaGFzQWxwaGE7Ci19Ci0KLXN0YXRpYyB2b2lkCi1kZWNvZGVEWFQxKGNvbnN0IEdMdm9pZCAqZGF0YSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LAotICAgICAgICAgICB2b2lkICpzdXJmYWNlLCBpbnQgc3RyaWRlLAotICAgICAgICAgICBib29sIGhhc0FscGhhKQotICAgIAotewotICAgIGluaXRfdGFibGVzKCk7Ci0gICAgCi0gICAgdWludDMyX3QgY29uc3QgKmQzMiA9ICh1aW50MzJfdCAqKWRhdGE7Ci0gICAgCi0gICAgLy8gQ29sb3IgdGFibGUgZm9yIHRoZSBjdXJyZW50IGJsb2NrCi0gICAgdWludDE2X3QgY1s0XTsKLSAgICBjWzBdID0gY1sxXSA9IGNbMl0gPSBjWzNdID0gMDsKLSAgICAKLSAgICAvLyBTcGVjaWZpZWQgY29sb3JzIGZyb20gdGhlIHByZXZpb3VzIGJsb2NrCi0gICAgdWludDE2X3QgcHJldl9jb2xvcjAgPSAweDAwMDA7Ci0gICAgdWludDE2X3QgcHJldl9jb2xvcjEgPSAweDAwMDA7Ci0gICAgCi0gICAgdWludDE2X3QqIHJvd1B0ciA9ICh1aW50MTZfdCopc3VyZmFjZTsKLSAgICBmb3IgKGludCBiYXNlX3kgPSAwOyBiYXNlX3kgPCBoZWlnaHQ7IGJhc2VfeSArPSA0LCByb3dQdHIgKz0gNCpzdHJpZGUpIHsKLSAgICAgICAgdWludDE2X3QgKmJsb2NrUHRyID0gcm93UHRyOwotICAgICAgICBmb3IgKGludCBiYXNlX3ggPSAwOyBiYXNlX3ggPCB3aWR0aDsgYmFzZV94ICs9IDQsIGJsb2NrUHRyICs9IDQpIHsKLSAgICAgICAgICAgIHVpbnQzMl90IGNvbG9ycyA9ICpkMzIrKzsKLSAgICAgICAgICAgIHVpbnQzMl90IGJpdHMgPSAqZDMyKys7Ci0gICAgICAgICAgICAKLSNpZiBfX0JZVEVfT1JERVIgPT0gX19CSUdfRU5ESUFOCi0gICAgICAgICAgICBjb2xvcnMgPSBzd2FwKGNvbG9ycyk7Ci0gICAgICAgICAgICBiaXRzID0gc3dhcChiaXRzKTsKLSNlbmRpZgotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyBSYXcgY29sb3JzCi0gICAgICAgICAgICB1aW50MTZfdCBjb2xvcjAgPSBjb2xvcnMgJiAweGZmZmY7Ci0gICAgICAgICAgICB1aW50MTZfdCBjb2xvcjEgPSBjb2xvcnMgPj4gMTY7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8vIElmIHRoZSBuZXcgYmxvY2sgaGFzIHRoZSBzYW1lIGJhc2UgY29sb3JzIGFzIHRoZQotICAgICAgICAgICAgLy8gcHJldmlvdXMgb25lLCB3ZSBkb24ndCBuZWVkIHRvIHJlY29tcHV0ZSB0aGUgY29sb3IKLSAgICAgICAgICAgIC8vIHRhYmxlIGNbXQotICAgICAgICAgICAgaWYgKGNvbG9yMCAhPSBwcmV2X2NvbG9yMCB8fCBjb2xvcjEgIT0gcHJldl9jb2xvcjEpIHsKLSAgICAgICAgICAgICAgICAvLyBTdG9yZSByYXcgY29sb3JzIGZvciBjb21wYXJpc29uIHdpdGggbmV4dCBibG9jawotICAgICAgICAgICAgICAgIHByZXZfY29sb3IwID0gY29sb3IwOwotICAgICAgICAgICAgICAgIHByZXZfY29sb3IxID0gY29sb3IxOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIGludCByMCA9ICAgcmVkKGNvbG9yMCk7Ci0gICAgICAgICAgICAgICAgaW50IGcwID0gZ3JlZW4oY29sb3IwKTsKLSAgICAgICAgICAgICAgICBpbnQgYjAgPSAgYmx1ZShjb2xvcjApOwotCi0gICAgICAgICAgICAgICAgaW50IHIxID0gICByZWQoY29sb3IxKTsKLSAgICAgICAgICAgICAgICBpbnQgZzEgPSBncmVlbihjb2xvcjEpOwotICAgICAgICAgICAgICAgIGludCBiMSA9ICBibHVlKGNvbG9yMSk7ICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIGlmIChoYXNBbHBoYSkgewotICAgICAgICAgICAgICAgICAgICBjWzBdID0gKHIwIDw8IDExKSB8ICgoZzAgPj4gMSkgPDwgNikgfCAoYjAgPDwgMSkgfCAweDE7Ci0gICAgICAgICAgICAgICAgICAgIGNbMV0gPSAocjEgPDwgMTEpIHwgKChnMSA+PiAxKSA8PCA2KSB8IChiMSA8PCAxKSB8IDB4MTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBjWzBdID0gY29sb3IwOwotICAgICAgICAgICAgICAgICAgICBjWzFdID0gY29sb3IxOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICBpbnQgcjIsIGcyLCBiMiwgcjMsIGczLCBiMywgYTM7Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgaW50IGJiaXRzID0gYml0cyA+PiAxOwotICAgICAgICAgICAgICAgIGJvb2wgaGFzMiA9ICgoYmJpdHMgJiB+Yml0cykgJiAweDU1NTU1NTU1KSAhPSAwOwotICAgICAgICAgICAgICAgIGJvb2wgaGFzMyA9ICgoYmJpdHMgJiAgYml0cykgJiAweDU1NTU1NTU1KSAhPSAwOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIGlmIChoYXMyIHx8IGhhczMpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGNvbG9yMCA+IGNvbG9yMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgcjIgPSBhdmcyMyhyMCwgcjEpOwotICAgICAgICAgICAgICAgICAgICAgICAgZzIgPSBhdmcyMyhnMCwgZzEpOwotICAgICAgICAgICAgICAgICAgICAgICAgYjIgPSBhdmcyMyhiMCwgYjEpOwotICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgICAgICAgICByMyA9IGF2ZzIzKHIxLCByMCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBnMyA9IGF2ZzIzKGcxLCBnMCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBiMyA9IGF2ZzIzKGIxLCBiMCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICBhMyA9IDE7Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICByMiA9IChyMCArIHIxKSA+PiAxOwotICAgICAgICAgICAgICAgICAgICAgICAgZzIgPSAoZzAgKyBnMSkgPj4gMTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGIyID0gKGIwICsgYjEpID4+IDE7Ci0gICAgICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgICAgIHIzID0gZzMgPSBiMyA9IGEzID0gMDsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBpZiAoaGFzQWxwaGEpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNbMl0gPSAocjIgPDwgMTEpIHwgKChnMiA+PiAxKSA8PCA2KSB8Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKGIyIDw8IDEpIHwgMHgxOwotICAgICAgICAgICAgICAgICAgICAgICAgY1szXSA9IChyMyA8PCAxMSkgfCAoKGczID4+IDEpIDw8IDYpIHwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAoYjMgPDwgMSkgfCBhMzsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGNbMl0gPSAocjIgPDwgMTEpIHwgKGcyIDw8IDUpIHwgYjI7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjWzNdID0gKHIzIDw8IDExKSB8IChnMyA8PCA1KSB8IGIzOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgCi0gICAgICAgICAgICB1aW50MTZfdCogYmxvY2tSb3dQdHIgPSBibG9ja1B0cjsKLSAgICAgICAgICAgIGZvciAoaW50IHkgPSAwOyB5IDwgNDsgeSsrLCBibG9ja1Jvd1B0ciArPSBzdHJpZGUpIHsKLSAgICAgICAgICAgICAgICAvLyBEb24ndCBwcm9jZXNzIHJvd3MgcGFzdCB0aGUgYm90b20KLSAgICAgICAgICAgICAgICBpZiAoYmFzZV95ICsgeSA+PSBoZWlnaHQpIHsKLSAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIGludCB3ID0gbWluKHdpZHRoIC0gYmFzZV94LCA0KTsKLSAgICAgICAgICAgICAgICBmb3IgKGludCB4ID0gMDsgeCA8IHc7IHgrKykgewotICAgICAgICAgICAgICAgICAgICBpbnQgY29kZSA9IGJpdHMgJiAweDM7Ci0gICAgICAgICAgICAgICAgICAgIGJpdHMgPj49IDI7Ci0gICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICBibG9ja1Jvd1B0clt4XSA9IGNbY29kZV07Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQotICAgIAotLy8gT3V0cHV0IGRhdGEgYXMgaW50ZXJuYWxmb3JtYXQ9R0xfUkdCQSwgdHlwZT1HTF9VTlNJR05FRF9CWVRFCi1zdGF0aWMgdm9pZAotZGVjb2RlRFhUMyhjb25zdCBHTHZvaWQgKmRhdGEsIGludCB3aWR0aCwgaW50IGhlaWdodCwKLSAgICAgICAgICAgdm9pZCAqc3VyZmFjZSwgaW50IHN0cmlkZSkKLQotewotICAgIGluaXRfdGFibGVzKCk7Ci0gICAgCi0gICAgdWludDMyX3QgY29uc3QgKmQzMiA9ICh1aW50MzJfdCAqKWRhdGE7Ci0gICAgCi0gICAgLy8gU3BlY2lmaWVkIGNvbG9ycyBmcm9tIHRoZSBwcmV2aW91cyBibG9jawotICAgIHVpbnQxNl90IHByZXZfY29sb3IwID0gMHgwMDAwOwotICAgIHVpbnQxNl90IHByZXZfY29sb3IxID0gMHgwMDAwOwotCi0gICAgLy8gQ29sb3IgdGFibGUgZm9yIHRoZSBjdXJyZW50IGJsb2NrCi0gICAgdWludDMyX3QgY1s0XTsKLSAgICBjWzBdID0gY1sxXSA9IGNbMl0gPSBjWzNdID0gMDsKLQotICAgIHVpbnQzMl90KiByb3dQdHIgPSAodWludDMyX3QqKXN1cmZhY2U7Ci0gICAgZm9yIChpbnQgYmFzZV95ID0gMDsgYmFzZV95IDwgaGVpZ2h0OyBiYXNlX3kgKz0gNCwgcm93UHRyICs9IDQqc3RyaWRlKSB7Ci0gICAgICAgIHVpbnQzMl90ICpibG9ja1B0ciA9IHJvd1B0cjsKLSAgICAgICAgZm9yIChpbnQgYmFzZV94ID0gMDsgYmFzZV94IDwgd2lkdGg7IGJhc2VfeCArPSA0LCBibG9ja1B0ciArPSA0KSB7Ci0gICAgICAgICAgICAKLSNpZiBfX0JZVEVfT1JERVIgPT0gX19CSUdfRU5ESUFOCi0gICAgICAgICAgICB1aW50MzJfdCBhbHBoYWhpID0gKmQzMisrOwotICAgICAgICAgICAgdWludDMyX3QgYWxwaGFsbyA9ICpkMzIrKzsKLSAgICAgICAgICAgIGFscGhhaGkgPSBzd2FwKGFscGhhaGkpOwotICAgICAgICAgICAgYWxwaGFsbyA9IHN3YXAoYWxwaGFsbyk7Ci0jZWxzZQotICAgICAgICAgICAgdWludDMyX3QgYWxwaGFsbyA9ICpkMzIrKzsKLSAgICAgICAgICAgIHVpbnQzMl90IGFscGhhaGkgPSAqZDMyKys7Ci0jZW5kaWYKLQotICAgICAgICAgICAgdWludDMyX3QgY29sb3JzID0gKmQzMisrOwotICAgICAgICAgICAgdWludDMyX3QgYml0cyA9ICpkMzIrKzsKLSAgICAgICAgICAgIAotI2lmIF9fQllURV9PUkRFUiA9PSBfX0JJR19FTkRJQU4KLSAgICAgICAgICAgIGNvbG9ycyA9IHN3YXAoY29sb3JzKTsKLSAgICAgICAgICAgIGJpdHMgPSBzd2FwKGJpdHMpOwotI2VuZGlmCi0gICAgICAgICAgICAKLSAgICAgICAgICAgIHVpbnQ2NF90IGFscGhhID0gKCh1aW50NjRfdClhbHBoYWhpIDw8IDMyKSB8IGFscGhhbG87Ci0KLSAgICAgICAgICAgIC8vIFJhdyBjb2xvcnMKLSAgICAgICAgICAgIHVpbnQxNl90IGNvbG9yMCA9IGNvbG9ycyAmIDB4ZmZmZjsKLSAgICAgICAgICAgIHVpbnQxNl90IGNvbG9yMSA9IGNvbG9ycyA+PiAxNjsKLQotICAgICAgICAgICAgLy8gSWYgdGhlIG5ldyBibG9jayBoYXMgdGhlIHNhbWUgYmFzZSBjb2xvcnMgYXMgdGhlCi0gICAgICAgICAgICAvLyBwcmV2aW91cyBvbmUsIHdlIGRvbid0IG5lZWQgdG8gcmVjb21wdXRlIHRoZSBjb2xvcgotICAgICAgICAgICAgLy8gdGFibGUgY1tdCi0gICAgICAgICAgICBpZiAoY29sb3IwICE9IHByZXZfY29sb3IwIHx8IGNvbG9yMSAhPSBwcmV2X2NvbG9yMSkgewotICAgICAgICAgICAgICAgIC8vIFN0b3JlIHJhdyBjb2xvcnMgZm9yIGNvbXBhcmlzb24gd2l0aCBuZXh0IGJsb2NrCi0gICAgICAgICAgICAgICAgcHJldl9jb2xvcjAgPSBjb2xvcjA7Ci0gICAgICAgICAgICAgICAgcHJldl9jb2xvcjEgPSBjb2xvcjE7Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgaW50IGJiaXRzID0gYml0cyA+PiAxOwotICAgICAgICAgICAgICAgIGJvb2wgaGFzMiA9ICgoYmJpdHMgJiB+Yml0cykgJiAweDU1NTU1NTU1KSAhPSAwOwotICAgICAgICAgICAgICAgIGJvb2wgaGFzMyA9ICgoYmJpdHMgJiAgYml0cykgJiAweDU1NTU1NTU1KSAhPSAwOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIGlmIChoYXMyIHx8IGhhczMpIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IHIwID0gICByZWQoY29sb3IwKTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGcwID0gZ3JlZW4oY29sb3IwKTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGIwID0gIGJsdWUoY29sb3IwKTsKLSAgICAgICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgICAgIGludCByMSA9ICAgcmVkKGNvbG9yMSk7Ci0gICAgICAgICAgICAgICAgICAgIGludCBnMSA9IGdyZWVuKGNvbG9yMSk7Ci0gICAgICAgICAgICAgICAgICAgIGludCBiMSA9ICBibHVlKGNvbG9yMSk7Ci0gICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICBpbnQgcjIgPSBhdmcyMyhyMCwgcjEpOwotICAgICAgICAgICAgICAgICAgICBpbnQgZzIgPSBhdmcyMyhnMCwgZzEpOwotICAgICAgICAgICAgICAgICAgICBpbnQgYjIgPSBhdmcyMyhiMCwgYjEpOwotICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgaW50IHIzID0gYXZnMjMocjEsIHIwKTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGczID0gYXZnMjMoZzEsIGcwKTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGIzID0gYXZnMjMoYjEsIGIwKTsKLQotICAgICAgICAgICAgICAgICAgICBjWzBdID0gcmdiNTY1U2VwVG84ODgocjAsIGcwLCBiMCk7Ci0gICAgICAgICAgICAgICAgICAgIGNbMV0gPSByZ2I1NjVTZXBUbzg4OChyMSwgZzEsIGIxKTsKLSAgICAgICAgICAgICAgICAgICAgY1syXSA9IHJnYjU2NVNlcFRvODg4KHIyLCBnMiwgYjIpOwotICAgICAgICAgICAgICAgICAgICBjWzNdID0gcmdiNTY1U2VwVG84ODgocjMsIGczLCBiMyk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gQ29udmVydCB0byA4IGJpdHMKLSAgICAgICAgICAgICAgICAgICAgY1swXSA9IHJnYjU2NVRvODg4KGNvbG9yMCk7Ci0gICAgICAgICAgICAgICAgICAgIGNbMV0gPSByZ2I1NjVUbzg4OChjb2xvcjEpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgdWludDMyX3QqIGJsb2NrUm93UHRyID0gYmxvY2tQdHI7Ci0gICAgICAgICAgICBmb3IgKGludCB5ID0gMDsgeSA8IDQ7IHkrKywgYmxvY2tSb3dQdHIgKz0gc3RyaWRlKSB7Ci0gICAgICAgICAgICAgICAgLy8gRG9uJ3QgcHJvY2VzcyByb3dzIHBhc3QgdGhlIGJvdG9tCi0gICAgICAgICAgICAgICAgaWYgKGJhc2VfeSArIHkgPj0gaGVpZ2h0KSB7Ci0gICAgICAgICAgICAgICAgICAgIGJyZWFrOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICBpbnQgdyA9IG1pbih3aWR0aCAtIGJhc2VfeCwgNCk7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgeCA9IDA7IHggPCB3OyB4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IGEgPSBhbHBoYSAmIDB4ZjsKLSAgICAgICAgICAgICAgICAgICAgYWxwaGEgPj49IDQ7Ci0KLSAgICAgICAgICAgICAgICAgICAgaW50IGNvZGUgPSBiaXRzICYgMHgzOwotICAgICAgICAgICAgICAgICAgICBiaXRzID4+PSAyOwotCi0gICAgICAgICAgICAgICAgICAgIGJsb2NrUm93UHRyW3hdID0gY1tjb2RlXSB8IChhIDw8IDI4KSB8IChhIDw8IDI0KTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLS8vIE91dHB1dCBkYXRhIGFzIGludGVybmFsZm9ybWF0PUdMX1JHQkEsIHR5cGU9R0xfVU5TSUdORURfQllURQotc3RhdGljIHZvaWQKLWRlY29kZURYVDUoY29uc3QgR0x2b2lkICpkYXRhLCBpbnQgd2lkdGgsIGludCBoZWlnaHQsCi0gICAgICAgICAgIHZvaWQgKnN1cmZhY2UsIGludCBzdHJpZGUpCi0KLXsKLSAgICBpbml0X3RhYmxlcygpOwotICAgIAotICAgIHVpbnQzMl90IGNvbnN0ICpkMzIgPSAodWludDMyX3QgKilkYXRhOwotICAgIAotICAgIC8vIFNwZWNpZmllZCBhbHBoYXMgZnJvbSB0aGUgcHJldmlvdXMgYmxvY2sKLSAgICB1aW50OF90IHByZXZfYWxwaGEwID0gMHgwMDsKLSAgICB1aW50OF90IHByZXZfYWxwaGExID0gMHgwMDsKLQotICAgIC8vIFNwZWNpZmllZCBjb2xvcnMgZnJvbSB0aGUgcHJldmlvdXMgYmxvY2sKLSAgICB1aW50MTZfdCBwcmV2X2NvbG9yMCA9IDB4MDAwMDsKLSAgICAgdWludDE2X3QgcHJldl9jb2xvcjEgPSAweDAwMDA7Ci0KLSAgICAvLyBBbHBoYSB0YWJsZSBmb3IgdGhlIGN1cnJlbnQgYmxvY2sKLSAgICB1aW50OF90IGFbOF07Ci0gICAgYVswXSA9IGFbMV0gPSBhWzJdID0gYVszXSA9IGFbNF0gPSBhWzVdID0gYVs2XSA9IGFbN10gPSAwOwotCi0gICAgLy8gQ29sb3IgdGFibGUgZm9yIHRoZSBjdXJyZW50IGJsb2NrCi0gICAgdWludDMyX3QgY1s0XTsKLSAgICBjWzBdID0gY1sxXSA9IGNbMl0gPSBjWzNdID0gMDsKLQotICAgIGludCBnb29kX2E1ID0gMDsKLSAgICBpbnQgYmFkX2E1ID0gMDsKLSAgICBpbnQgZ29vZF9hNiA9IDA7Ci0gICAgaW50IGJhZF9hNiA9IDA7Ci0gICAgaW50IGdvb2RfYTcgPSAwOwotICAgIGludCBiYWRfYTcgPSAwOwotCi0gICAgdWludDMyX3QqIHJvd1B0ciA9ICh1aW50MzJfdCopc3VyZmFjZTsKLSAgICBmb3IgKGludCBiYXNlX3kgPSAwOyBiYXNlX3kgPCBoZWlnaHQ7IGJhc2VfeSArPSA0LCByb3dQdHIgKz0gNCpzdHJpZGUpIHsKLSAgICAgICAgdWludDMyX3QgKmJsb2NrUHRyID0gcm93UHRyOwotICAgICAgICBmb3IgKGludCBiYXNlX3ggPSAwOyBiYXNlX3ggPCB3aWR0aDsgYmFzZV94ICs9IDQsIGJsb2NrUHRyICs9IDQpIHsKLSAgICAgICAgICAgIAotI2lmIF9fQllURV9PUkRFUiA9PSBfX0JJR19FTkRJQU4KLSAgICAgICAgICAgIHVpbnQzMl90IGFscGhhaGkgPSAqZDMyKys7Ci0gICAgICAgICAgICB1aW50MzJfdCBhbHBoYWxvID0gKmQzMisrOwotICAgICAgICAgICAgYWxwaGFoaSA9IHN3YXAoYWxwaGFoaSk7Ci0gICAgICAgICAgICBhbHBoYWxvID0gc3dhcChhbHBoYWxvKTsKLSNlbHNlCi0gICAgICAgICAgICAgdWludDMyX3QgYWxwaGFsbyA9ICpkMzIrKzsKLSAgICAgICAgICAgICB1aW50MzJfdCBhbHBoYWhpID0gKmQzMisrOwotI2VuZGlmCi0KLSAgICAgICAgICAgIHVpbnQzMl90IGNvbG9ycyA9ICpkMzIrKzsKLSAgICAgICAgICAgIHVpbnQzMl90IGJpdHMgPSAqZDMyKys7Ci0gICAgICAgICAgICAKLSNpZiBfX0JZVEVfT1JERVIgPT0gX19CSUdfRU5ESUFOeAotICAgICAgICAgICAgY29sb3JzID0gc3dhcChjb2xvcnMpOwotICAgICAgICAgICAgYml0cyA9IHN3YXAoYml0cyk7Ci0jZW5kaWYKLSAgICAgICAgICAgIAotICAgICAgICAgICAgdWludDY0X3QgYWxwaGEgPSAoKHVpbnQ2NF90KWFscGhhaGkgPDwgMzIpIHwgYWxwaGFsbzsKLSAgICAgICAgICAgIHVpbnQ2NF90IGFscGhhMCA9IGFscGhhICYgMHhmZjsKLSAgICAgICAgICAgIGFscGhhID4+PSA4OwotICAgICAgICAgICAgdWludDY0X3QgYWxwaGExID0gYWxwaGEgJiAweGZmOwotICAgICAgICAgICAgYWxwaGEgPj49IDg7Ci0KLSAgICAgICAgICAgIGlmIChhbHBoYTAgIT0gcHJldl9hbHBoYTAgfHwgYWxwaGExICE9IHByZXZfYWxwaGExKSB7Ci0gICAgICAgICAgICAgICAgcHJldl9hbHBoYTAgPSBhbHBoYTA7Ci0gICAgICAgICAgICAgICAgcHJldl9hbHBoYTEgPSBhbHBoYTE7Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgYVswXSA9IGFscGhhMDsKLSAgICAgICAgICAgICAgICBhWzFdID0gYWxwaGExOwotICAgICAgICAgICAgICAgIGludCBhMDEgPSBhbHBoYTAgKyBhbHBoYTEgLSAxOwotICAgICAgICAgICAgICAgIGlmIChhbHBoYTAgPiBhbHBoYTEpIHsKLSAgICAgICAgICAgICAgICAgICAgYVsyXSA9IGRpdjcoNiphbHBoYTAgKyAgIGFscGhhMSk7Ci0gICAgICAgICAgICAgICAgICAgIGFbNF0gPSBkaXY3KDQqYWxwaGEwICsgMyphbHBoYTEpOwotICAgICAgICAgICAgICAgICAgICBhWzZdID0gZGl2NygyKmFscGhhMCArIDUqYWxwaGExKTsKLQotICAgICAgICAgICAgICAgICAgICAvLyBVc2Ugc3ltbWV0cnkgdG8gZGVyaXZlIGhhbGYgb2YgdGhlIHZhbHVlcwotICAgICAgICAgICAgICAgICAgICAvLyBBIGZldyB2YWx1ZXMgd2lsbCBiZSBvZmYgYnkgMSAofi41JSkKLSAgICAgICAgICAgICAgICAgICAgLy8gQWx0ZXJuYXRlIHdoaWNoIHZhbHVlcyBhcmUgY29tcHV0ZWQgZGlyZWN0bHkKLSAgICAgICAgICAgICAgICAgICAgLy8gYW5kIHdoaWNoIGFyZSBkZXJpdmVkIHRvIHRyeSB0byByZWR1Y2UgYmlhcwotICAgICAgICAgICAgICAgICAgICBhWzNdID0gYTAxIC0gYVs2XTsKLSAgICAgICAgICAgICAgICAgICAgYVs1XSA9IGEwMSAtIGFbNF07Ci0gICAgICAgICAgICAgICAgICAgIGFbN10gPSBhMDEgLSBhWzJdOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIGFbMl0gPSBkaXY1KDQqYWxwaGEwICsgICBhbHBoYTEpOwotICAgICAgICAgICAgICAgICAgICBhWzRdID0gZGl2NSgyKmFscGhhMCArIDMqYWxwaGExKTsKLSAgICAgICAgICAgICAgICAgICAgYVszXSA9IGEwMSAtIGFbNF07Ci0gICAgICAgICAgICAgICAgICAgIGFbNV0gPSBhMDEgLSBhWzJdOwotICAgICAgICAgICAgICAgICAgICBhWzZdID0gMHgwMDsKLSAgICAgICAgICAgICAgICAgICAgYVs3XSA9IDB4ZmY7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBSYXcgY29sb3JzCi0gICAgICAgICAgICB1aW50MTZfdCBjb2xvcjAgPSBjb2xvcnMgJiAweGZmZmY7Ci0gICAgICAgICAgICB1aW50MTZfdCBjb2xvcjEgPSBjb2xvcnMgPj4gMTY7Ci0KLSAgICAgICAgICAgIC8vIElmIHRoZSBuZXcgYmxvY2sgaGFzIHRoZSBzYW1lIGJhc2UgY29sb3JzIGFzIHRoZQotICAgICAgICAgICAgLy8gcHJldmlvdXMgb25lLCB3ZSBkb24ndCBuZWVkIHRvIHJlY29tcHV0ZSB0aGUgY29sb3IKLSAgICAgICAgICAgIC8vIHRhYmxlIGNbXQotICAgICAgICAgICAgaWYgKGNvbG9yMCAhPSBwcmV2X2NvbG9yMCB8fCBjb2xvcjEgIT0gcHJldl9jb2xvcjEpIHsKLSAgICAgICAgICAgICAgICAvLyBTdG9yZSByYXcgY29sb3JzIGZvciBjb21wYXJpc29uIHdpdGggbmV4dCBibG9jawotICAgICAgICAgICAgICAgIHByZXZfY29sb3IwID0gY29sb3IwOwotICAgICAgICAgICAgICAgIHByZXZfY29sb3IxID0gY29sb3IxOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIGludCBiYml0cyA9IGJpdHMgPj4gMTsKLSAgICAgICAgICAgICAgICBib29sIGhhczIgPSAoKGJiaXRzICYgfmJpdHMpICYgMHg1NTU1NTU1NSkgIT0gMDsKLSAgICAgICAgICAgICAgICBib29sIGhhczMgPSAoKGJiaXRzICYgIGJpdHMpICYgMHg1NTU1NTU1NSkgIT0gMDsKLSAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICBpZiAoaGFzMiB8fCBoYXMzKSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCByMCA9ICAgcmVkKGNvbG9yMCk7Ci0gICAgICAgICAgICAgICAgICAgIGludCBnMCA9IGdyZWVuKGNvbG9yMCk7Ci0gICAgICAgICAgICAgICAgICAgIGludCBiMCA9ICBibHVlKGNvbG9yMCk7Ci0gICAgICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICBpbnQgcjEgPSAgIHJlZChjb2xvcjEpOwotICAgICAgICAgICAgICAgICAgICBpbnQgZzEgPSBncmVlbihjb2xvcjEpOwotICAgICAgICAgICAgICAgICAgICBpbnQgYjEgPSAgYmx1ZShjb2xvcjEpOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICBpbnQgcjIgPSBhdmcyMyhyMCwgcjEpOwotICAgICAgICAgICAgICAgICAgICBpbnQgZzIgPSBhdmcyMyhnMCwgZzEpOwotICAgICAgICAgICAgICAgICAgICBpbnQgYjIgPSBhdmcyMyhiMCwgYjEpOwotICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgaW50IHIzID0gYXZnMjMocjEsIHIwKTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGczID0gYXZnMjMoZzEsIGcwKTsKLSAgICAgICAgICAgICAgICAgICAgaW50IGIzID0gYXZnMjMoYjEsIGIwKTsKLQotICAgICAgICAgICAgICAgICAgICBjWzBdID0gcmdiNTY1U2VwVG84ODgocjAsIGcwLCBiMCk7Ci0gICAgICAgICAgICAgICAgICAgIGNbMV0gPSByZ2I1NjVTZXBUbzg4OChyMSwgZzEsIGIxKTsKLSAgICAgICAgICAgICAgICAgICAgY1syXSA9IHJnYjU2NVNlcFRvODg4KHIyLCBnMiwgYjIpOwotICAgICAgICAgICAgICAgICAgICBjWzNdID0gcmdiNTY1U2VwVG84ODgocjMsIGczLCBiMyk7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gQ29udmVydCB0byA4IGJpdHMKLSAgICAgICAgICAgICAgICAgICAgY1swXSA9IHJnYjU2NVRvODg4KGNvbG9yMCk7Ci0gICAgICAgICAgICAgICAgICAgIGNbMV0gPSByZ2I1NjVUbzg4OChjb2xvcjEpOwotICAgICAgICAgICAgICAgIH0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIHVpbnQzMl90KiBibG9ja1Jvd1B0ciA9IGJsb2NrUHRyOwotICAgICAgICAgICAgZm9yIChpbnQgeSA9IDA7IHkgPCA0OyB5KyssIGJsb2NrUm93UHRyICs9IHN0cmlkZSkgewotICAgICAgICAgICAgICAgIC8vIERvbid0IHByb2Nlc3Mgcm93cyBwYXN0IHRoZSBib3RvbQotICAgICAgICAgICAgICAgIGlmIChiYXNlX3kgKyB5ID49IGhlaWdodCkgewotICAgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgaW50IHcgPSBtaW4od2lkdGggLSBiYXNlX3gsIDQpOwotICAgICAgICAgICAgICAgIGZvciAoaW50IHggPSAwOyB4IDwgdzsgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCBhY29kZSA9IGFscGhhICYgMHg3OwotICAgICAgICAgICAgICAgICAgICBhbHBoYSA+Pj0gMzsKLQotICAgICAgICAgICAgICAgICAgICBpbnQgY29kZSA9IGJpdHMgJiAweDM7Ci0gICAgICAgICAgICAgICAgICAgIGJpdHMgPj49IDI7Ci0KLSAgICAgICAgICAgICAgICAgICAgYmxvY2tSb3dQdHJbeF0gPSBjW2NvZGVdIHwgKGFbYWNvZGVdIDw8IDI0KTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0gICAKLS8qCi0gKiBEZWNvZGUgYSBEWFQtY29tcHJlc3NlZCB0ZXh0dXJlIGludG8gbWVtb3J5LiAgRFhUIHRleHR1cmVzIGNvbnNpc3Qgb2YKLSAqIGEgc2VyaWVzIG9mIDR4NCBwaXhlbCBibG9ja3MgaW4gbGVmdC10by1yaWdodCwgdG9wLWRvd24gb3JkZXIuCi0gKiBUaGUgbnVtYmVyIG9mIGJsb2NrcyBpcyBnaXZlbiBieSBjZWlsKHdpZHRoLzQpKmNlaWwoaGVpZ2h0LzQpLgotICoKLSAqICdkYXRhJyBwb2ludHMgdG8gdGhlIHRleHR1cmUgZGF0YS4gJ3dpZHRoJyBhbmQgJ2hlaWdodCcgaW5kaWNhdGUgdGhlCi0gKiBkaW1lbnNpb25zIG9mIHRoZSB0ZXh0dXJlLiAgV2UgYXNzdW1lIHdpZHRoIGFuZCBoZWlnaHQgYXJlID49IDAgYnV0Ci0gKiBkbyBub3QgcmVxdWlyZSB0aGVtIHRvIGJlIHBvd2VycyBvZiAyIG9yIGRpdmlzaWJsZSBieSBhbnkgZmFjdG9yLgotICoKLSAqIFRoZSBvdXRwdXQgaXMgd3JpdHRlbiB0byAnc3VyZmFjZScgd2l0aCBlYWNoIHNjYW5saW5lIHNlcGFyYXRlZCBieQotICogJ3N0cmlkZScgMi0gb3IgNC1ieXRlIHdvcmRzLgotICoKLSAqICdmb3JtYXQnIGluZGljYXRlcyB0aGUgdHlwZSBvZiBjb21wcmVzc2lvbiBhbmQgbXVzdCBiZSBvbmUgb2YgdGhlIGZvbGxvd2luZzoKLSAqCi0gKiAgIEdMX0NPTVBSRVNTRURfUkdCX1MzVENfRFhUMV9FWFQ6Ci0gKiAgICAgIFRoZSBvdXRwdXQgaXMgd3JpdHRlbiBhcyA1LzYvNSBvcGFxdWUgUkdCICgxNiBiaXQgd29yZHMpLgotICogICAgICA4IGJ5dGVzIGFyZSByZWFkIGZyb20gJ2RhdGEnIGZvciBlYWNoIGJsb2NrLgotICoKLSAqICAgR0xfQ09NUFJFU1NFRF9SR0JBX1MzVENfRFhUMV9FWFQKLSAqICAgICAgVGhlIG91dHB1dCBpcyB3cml0dGVuIGFzIDUvNS81LzEgUkdCQSAoMTYgYml0IHdvcmRzKQotICogICAgICA4IGJ5dGVzIGFyZSByZWFkIGZyb20gJ2RhdGEnIGZvciBlYWNoIGJsb2NrLgotICoKLSAqICAgR0xfQ09NUFJFU1NFRF9SR0JBX1MzVENfRFhUM19FWFQKLSAqICAgR0xfQ09NUFJFU1NFRF9SR0JBX1MzVENfRFhUNV9FWFQKLSAqICAgICAgVGhlIG91dHB1dCBpcyB3cml0dGVuIGFzIDgvOC84LzggQVJHQiAoMzIgYml0IHdvcmRzKQotICogICAgICAxNiBieXRlcyBhcmUgcmVhZCBmcm9tICdkYXRhJyBmb3IgZWFjaCBibG9jay4KLSAqLwotdm9pZAotZGVjb2RlRFhUKGNvbnN0IEdMdm9pZCAqZGF0YSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LAotICAgICAgICAgIHZvaWQgKnN1cmZhY2UsIGludCBzdHJpZGUsIGludCBmb3JtYXQpCi17Ci0jaWYgVElNSU5HCi0gICAgc3RydWN0IHRpbWV2YWwgc3RhcnRfdCwgZW5kX3Q7Ci0gICAgc3RydWN0IHRpbWV6b25lIHR6OwotICAgIAotICAgIGdldHRpbWVvZmRheSgmc3RhcnRfdCwgJnR6KTsKLSNlbmRpZgotCi0gICAgc3dpdGNoIChmb3JtYXQpIHsKLSAgICBjYXNlIEdMX0NPTVBSRVNTRURfUkdCX1MzVENfRFhUMV9FWFQ6Ci0gICAgICAgIGRlY29kZURYVDEoZGF0YSwgd2lkdGgsIGhlaWdodCwgc3VyZmFjZSwgc3RyaWRlLCBmYWxzZSk7Ci0gICAgICAgIGJyZWFrOwotICAgICAgICAKLSAgICBjYXNlIEdMX0NPTVBSRVNTRURfUkdCQV9TM1RDX0RYVDFfRVhUOgotICAgICAgICBkZWNvZGVEWFQxKGRhdGEsIHdpZHRoLCBoZWlnaHQsIHN1cmZhY2UsIHN0cmlkZSwgdHJ1ZSk7Ci0gICAgICAgIGJyZWFrOwotICAgICAgICAKLSAgICBjYXNlIEdMX0NPTVBSRVNTRURfUkdCQV9TM1RDX0RYVDNfRVhUOgotICAgICAgICBkZWNvZGVEWFQzKGRhdGEsIHdpZHRoLCBoZWlnaHQsIHN1cmZhY2UsIHN0cmlkZSk7Ci0gICAgICAgIGJyZWFrOwotICAgICAgICAKLSAgICBjYXNlIEdMX0NPTVBSRVNTRURfUkdCQV9TM1RDX0RYVDVfRVhUOgotICAgICAgICBkZWNvZGVEWFQ1KGRhdGEsIHdpZHRoLCBoZWlnaHQsIHN1cmZhY2UsIHN0cmlkZSk7Ci0gICAgICAgIGJyZWFrOwotICAgIH0KLSAgICAKLSNpZiBUSU1JTkcKLSAgICBnZXR0aW1lb2ZkYXkoJmVuZF90LCAmdHopOwotICAgIGxvbmcgdXNlYyA9IChlbmRfdC50dl9zZWMgLSBzdGFydF90LnR2X3NlYykqMTAwMDAwMCArCi0gICAgICAgIChlbmRfdC50dl91c2VjIC0gc3RhcnRfdC50dl91c2VjKTsKLSAgICAKLSAgICBwcmludGYoIkxvYWRlZCB3PSVkIGg9JWQgaW4gJWxkIHVzZWNcbiIsIHdpZHRoLCBoZWlnaHQsIHVzZWMpOwotI2VuZGlmCi19Ci0KLX0gLy8gbmFtZXNwYWNlIGFuZHJvaWQKZGlmZiAtLWdpdCBhL29wZW5nbC9saWJhZ2wvZHh0LmggYi9vcGVuZ2wvbGliYWdsL2R4dC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBkOTVhMzZjLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJhZ2wvZHh0LmgKKysrIC9kZXYvbnVsbApAQCAtMSwzMyArMCwwIEBACi0vKiBsaWJzL29wZW5nbGVzL2R4dC5oCi0qKgotKiogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2lmbmRlZiBBTkRST0lEX09QRU5HTEVTX1RFWFRVUkVfSAotI2RlZmluZSBBTkRST0lEX09QRU5HTEVTX1RFWFRVUkVfSAotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0KLSNpbmNsdWRlIDxHTEVTL2dsLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotICBib29sIERYVDFIYXNBbHBoYShjb25zdCBHTHZvaWQgKmRhdGEsIGludCB3aWR0aCwgaW50IGhlaWdodCk7Ci0gIHZvaWQgZGVjb2RlRFhUKGNvbnN0IEdMdm9pZCAqZGF0YSwgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LAotICAgICAgICAgICAgICAgICB2b2lkICpzdXJmYWNlLCBpbnQgc3RyaWRlLCBpbnQgZm9ybWF0KTsKLQotfSAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9PUEVOR0xFU19URVhUVVJFX0gKZGlmZiAtLWdpdCBhL29wZW5nbC9saWJhZ2wvZWdsLmNwcCBiL29wZW5nbC9saWJhZ2wvZWdsLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMTQ0NmZiMi4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGliYWdsL2VnbC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxNTQxICswLDAgQEAKLS8qIAotKioKLSoqIENvcHlyaWdodCAyMDA3IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UgVmVyc2lvbiAyLjAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZyBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNkZWZpbmUgTE9HX1RBRyAiRUdMIgotCi0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0cmluZy5oPgotI2luY2x1ZGUgPHVuaXN0ZC5oPgotI2luY2x1ZGUgPGZjbnRsLmg+Ci0jaW5jbHVkZSA8c3lzL2lvY3RsLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8c3lzL21tYW4uaD4KLQotI2luY2x1ZGUgPGN1dGlscy9sb2cuaD4KLSNpbmNsdWRlIDxjdXRpbHMvYXRvbWljLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy90aHJlYWRzLmg+Ci0KLSNpbmNsdWRlIDxFR0wvZWdsLmg+Ci0jaW5jbHVkZSA8RUdML2VnbGV4dC5oPgotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLSNpbmNsdWRlIDxHTEVTL2dsZXh0Lmg+Ci0KLSNpbmNsdWRlIDxwaXhlbGZsaW5nZXIvZm9ybWF0Lmg+Ci0jaW5jbHVkZSA8cGl4ZWxmbGluZ2VyL3BpeGVsZmxpbmdlci5oPgotCi0jaW5jbHVkZSAiY29udGV4dC5oIgotI2luY2x1ZGUgInN0YXRlLmgiCi0jaW5jbHVkZSAidGV4dHVyZS5oIgotI2luY2x1ZGUgIm1hdHJpeC5oIgotCi0jdW5kZWYgTkVMRU0KLSNkZWZpbmUgTkVMRU0oeCkgKHNpemVvZih4KS9zaXplb2YoKih4KSkpCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotY29uc3QgdW5zaWduZWQgaW50IE5VTV9ESVNQTEFZUyA9IDE7Ci0KLXN0YXRpYyBwdGhyZWFkX211dGV4X3QgZ0luaXRNdXRleCA9IFBUSFJFQURfTVVURVhfSU5JVElBTElaRVI7Ci1zdGF0aWMgcHRocmVhZF9tdXRleF90IGdFcnJvcktleU11dGV4ID0gUFRIUkVBRF9NVVRFWF9JTklUSUFMSVpFUjsKLXN0YXRpYyBwdGhyZWFkX2tleV90IGdFR0xFcnJvcktleSA9IC0xOwotI2lmbmRlZiBIQVZFX0FORFJPSURfT1MKLW5hbWVzcGFjZSBnbCB7Ci1wdGhyZWFkX2tleV90IGdHTEtleSA9IC0xOwotfTsgLy8gbmFtZXNwYWNlIGdsCi0jZW5kaWYKLQotdGVtcGxhdGU8dHlwZW5hbWUgVD4KLXN0YXRpYyBUIHNldEVycm9yKEdMaW50IGVycm9yLCBUIHJldHVyblZhbHVlKSB7Ci0gICAgaWYgKGdnbF91bmxpa2VseShnRUdMRXJyb3JLZXkgPT0gLTEpKSB7Ci0gICAgICAgIHB0aHJlYWRfbXV0ZXhfbG9jaygmZ0Vycm9yS2V5TXV0ZXgpOwotICAgICAgICBpZiAoZ0VHTEVycm9yS2V5ID09IC0xKQotICAgICAgICAgICAgcHRocmVhZF9rZXlfY3JlYXRlKCZnRUdMRXJyb3JLZXksIE5VTEwpOwotICAgICAgICBwdGhyZWFkX211dGV4X3VubG9jaygmZ0Vycm9yS2V5TXV0ZXgpOwotICAgIH0KLSAgICBwdGhyZWFkX3NldHNwZWNpZmljKGdFR0xFcnJvcktleSwgKHZvaWQqKWVycm9yKTsKLSAgICByZXR1cm4gcmV0dXJuVmFsdWU7Ci19Ci0KLXN0YXRpYyBHTGludCBnZXRFcnJvcigpIHsKLSAgICBpZiAoZ2dsX3VubGlrZWx5KGdFR0xFcnJvcktleSA9PSAtMSkpCi0gICAgICAgIHJldHVybiBFR0xfU1VDQ0VTUzsKLSAgICBHTGludCBlcnJvciA9IChHTGludClwdGhyZWFkX2dldHNwZWNpZmljKGdFR0xFcnJvcktleSk7Ci0gICAgcHRocmVhZF9zZXRzcGVjaWZpYyhnRUdMRXJyb3JLZXksICh2b2lkKilFR0xfU1VDQ0VTUyk7Ci0gICAgcmV0dXJuIGVycm9yOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0cnVjdCBlZ2xfZGlzcGxheV90Ci17Ci0gICAgZWdsX2Rpc3BsYXlfdCgpIDogdHlwZSgwKSwgaW5pdGlhbGl6ZWQoMCkgeyB9Ci0gICAgCi0gICAgc3RhdGljIGVnbF9kaXNwbGF5X3QmIGdldF9kaXNwbGF5KEVHTERpc3BsYXkgZHB5KTsKLSAgICAKLSAgICBzdGF0aWMgRUdMQm9vbGVhbiBpc192YWxpZChFR0xEaXNwbGF5IGRweSkgewotICAgICAgICByZXR1cm4gKCh1aW50cHRyX3QoZHB5KS0xVSkgPj0gTlVNX0RJU1BMQVlTKSA/IEVHTF9GQUxTRSA6IEVHTF9UUlVFOwotICAgIH0KLQotICAgIE5hdGl2ZURpc3BsYXlUeXBlICAgdHlwZTsKLSAgICB2b2xhdGlsZSBpbnQzMl90ICAgIGluaXRpYWxpemVkOwotfTsKLQotc3RhdGljIGVnbF9kaXNwbGF5X3QgZ0Rpc3BsYXlzW05VTV9ESVNQTEFZU107Ci0KLWVnbF9kaXNwbGF5X3QmIGVnbF9kaXNwbGF5X3Q6OmdldF9kaXNwbGF5KEVHTERpc3BsYXkgZHB5KSB7Ci0gICAgcmV0dXJuIGdEaXNwbGF5c1t1aW50cHRyX3QoZHB5KS0xVV07Ci19Ci0KLXN0cnVjdCBlZ2xfY29udGV4dF90IHsKLSAgICBlbnVtIHsKLSAgICAgICAgSVNfQ1VSUkVOVCAgICAgID0gICAweDAwMDEwMDAwLAotICAgICAgICBORVZFUl9DVVJSRU5UICAgPSAgIDB4MDAwMjAwMDAKLSAgICB9OwotICAgIHVpbnQzMl90ICAgICAgICAgICAgZmxhZ3M7Ci0gICAgRUdMRGlzcGxheSAgICAgICAgICBkcHk7Ci0gICAgRUdMQ29uZmlnICAgICAgICAgICBjb25maWc7Ci0gICAgRUdMU3VyZmFjZSAgICAgICAgICByZWFkOwotICAgIEVHTFN1cmZhY2UgICAgICAgICAgZHJhdzsKLQotICAgIHN0YXRpYyBpbmxpbmUgZWdsX2NvbnRleHRfdCogY29udGV4dChFR0xDb250ZXh0IGN0eCkgewotICAgICAgICBvZ2xlc19jb250ZXh0X3QqIGNvbnN0IGdsID0gc3RhdGljX2Nhc3Q8b2dsZXNfY29udGV4dF90Kj4oY3R4KTsKLSAgICAgICAgcmV0dXJuIHN0YXRpY19jYXN0PGVnbF9jb250ZXh0X3QqPihnbC0+cmFzdGVyaXplci5iYXNlKTsKLSAgICB9Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0cnVjdCBlZ2xfc3VyZmFjZV90Ci17Ci0gICAgZW51bSB7Ci0gICAgICAgIFBBR0VfRkxJUCA9IDB4MDAwMDAwMDEsCi0gICAgICAgIE1BR0lDICAgICA9IDB4MzE0MTUyNjUKLSAgICB9OwotCi0gICAgdWludDMyX3QgICAgICAgICAgICBtYWdpYzsKLSAgICBFR0xEaXNwbGF5ICAgICAgICAgIGRweTsKLSAgICBFR0xDb25maWcgICAgICAgICAgIGNvbmZpZzsKLSAgICBFR0xDb250ZXh0ICAgICAgICAgIGN0eDsKLQotICAgICAgICAgICAgICAgIGVnbF9zdXJmYWNlX3QoRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsIGludDMyX3QgZGVwdGhGb3JtYXQpOwotICAgIHZpcnR1YWwgICAgIH5lZ2xfc3VyZmFjZV90KCk7Ci0gICAgdmlydHVhbCAgICAgYm9vbCAgICBpc1ZhbGlkKCkgY29uc3QgPSAwOwotICAgIAotICAgIHZpcnR1YWwgICAgIEVHTEJvb2xlYW4gIGJpbmREcmF3U3VyZmFjZShvZ2xlc19jb250ZXh0X3QqIGdsKSA9IDA7Ci0gICAgdmlydHVhbCAgICAgRUdMQm9vbGVhbiAgYmluZFJlYWRTdXJmYWNlKG9nbGVzX2NvbnRleHRfdCogZ2wpID0gMDsKLSAgICB2aXJ0dWFsICAgICBFR0xpbnQgICAgICBnZXRXaWR0aCgpIGNvbnN0ID0gMDsKLSAgICB2aXJ0dWFsICAgICBFR0xpbnQgICAgICBnZXRIZWlnaHQoKSBjb25zdCA9IDA7Ci0gICAgdmlydHVhbCAgICAgdm9pZCogICAgICAgZ2V0Qml0cygpIGNvbnN0ID0gMDsKLQotICAgIHZpcnR1YWwgICAgIEVHTGludCAgICAgIGdldEhvcml6b250YWxSZXNvbHV0aW9uKCkgY29uc3Q7Ci0gICAgdmlydHVhbCAgICAgRUdMaW50ICAgICAgZ2V0VmVydGljYWxSZXNvbHV0aW9uKCkgY29uc3Q7Ci0gICAgdmlydHVhbCAgICAgRUdMaW50ICAgICAgZ2V0UmVmcmVzaFJhdGUoKSBjb25zdDsKLSAgICB2aXJ0dWFsICAgICBFR0xpbnQgICAgICBnZXRTd2FwQmVoYXZpb3IoKSBjb25zdDsKLSAgICB2aXJ0dWFsICAgICBFR0xCb29sZWFuICBzd2FwQnVmZmVycygpOwotcHJvdGVjdGVkOgotICAgIEdHTFN1cmZhY2UgICAgICAgICAgICAgIGRlcHRoOwotfTsKLQotZWdsX3N1cmZhY2VfdDo6ZWdsX3N1cmZhY2VfdChFR0xEaXNwbGF5IGRweSwKLSAgICAgICAgRUdMQ29uZmlnIGNvbmZpZywKLSAgICAgICAgaW50MzJfdCBkZXB0aEZvcm1hdCkKLSAgICA6IG1hZ2ljKE1BR0lDKSwgZHB5KGRweSksIGNvbmZpZyhjb25maWcpLCBjdHgoMCkKLXsKLSAgICBkZXB0aC52ZXJzaW9uID0gc2l6ZW9mKEdHTFN1cmZhY2UpOwotICAgIGRlcHRoLmRhdGEgPSAwOwotICAgIGRlcHRoLmZvcm1hdCA9IGRlcHRoRm9ybWF0OwotfQotZWdsX3N1cmZhY2VfdDo6fmVnbF9zdXJmYWNlX3QoKQotewotICAgIG1hZ2ljID0gMDsKLSAgICBmcmVlKGRlcHRoLmRhdGEpOwotfQotRUdMQm9vbGVhbiBlZ2xfc3VyZmFjZV90Ojpzd2FwQnVmZmVycygpIHsKLSAgICByZXR1cm4gRUdMX0ZBTFNFOwotfQotRUdMaW50IGVnbF9zdXJmYWNlX3Q6OmdldEhvcml6b250YWxSZXNvbHV0aW9uKCkgY29uc3QgewotICAgIHJldHVybiAoMCAqIEVHTF9ESVNQTEFZX1NDQUxJTkcpICogKDEuMGYgLyAyNS40Zik7Ci19Ci1FR0xpbnQgZWdsX3N1cmZhY2VfdDo6Z2V0VmVydGljYWxSZXNvbHV0aW9uKCkgY29uc3QgewotICAgIHJldHVybiAoMCAqIEVHTF9ESVNQTEFZX1NDQUxJTkcpICogKDEuMGYgLyAyNS40Zik7Ci19Ci1FR0xpbnQgZWdsX3N1cmZhY2VfdDo6Z2V0UmVmcmVzaFJhdGUoKSBjb25zdCB7Ci0gICAgcmV0dXJuICg2MCAqIEVHTF9ESVNQTEFZX1NDQUxJTkcpOwotfQotRUdMaW50IGVnbF9zdXJmYWNlX3Q6OmdldFN3YXBCZWhhdmlvcigpIGNvbnN0IHsKLSAgICByZXR1cm4gRUdMX0JVRkZFUl9QUkVTRVJWRUQ7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RydWN0IGVnbF93aW5kb3dfc3VyZmFjZV90IDogcHVibGljIGVnbF9zdXJmYWNlX3QKLXsKLSAgICBlZ2xfd2luZG93X3N1cmZhY2VfdCgKLSAgICAgICAgICAgIEVHTERpc3BsYXkgZHB5LCBFR0xDb25maWcgY29uZmlnLAotICAgICAgICAgICAgaW50MzJfdCBkZXB0aEZvcm1hdCwKLSAgICAgICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3QqIHdpbmRvdyk7Ci0KLSAgICAgfmVnbF93aW5kb3dfc3VyZmFjZV90KCk7Ci0KLSAgICB2aXJ0dWFsICAgICBib29sICAgICAgICBpc1ZhbGlkKCkgY29uc3QgeyByZXR1cm4gbmF0aXZlV2luZG93LT5tYWdpYyA9PSAweDYwMDkxMzsgfSAgICAKLSAgICB2aXJ0dWFsICAgICBFR0xCb29sZWFuICBzd2FwQnVmZmVycygpOwotICAgIHZpcnR1YWwgICAgIEVHTEJvb2xlYW4gIGJpbmREcmF3U3VyZmFjZShvZ2xlc19jb250ZXh0X3QqIGdsKTsKLSAgICB2aXJ0dWFsICAgICBFR0xCb29sZWFuICBiaW5kUmVhZFN1cmZhY2Uob2dsZXNfY29udGV4dF90KiBnbCk7Ci0gICAgdmlydHVhbCAgICAgRUdMaW50ICAgICAgZ2V0V2lkdGgoKSBjb25zdCAgICB7IHJldHVybiBuYXRpdmVXaW5kb3ctPndpZHRoOyAgfQotICAgIHZpcnR1YWwgICAgIEVHTGludCAgICAgIGdldEhlaWdodCgpIGNvbnN0ICAgeyByZXR1cm4gbmF0aXZlV2luZG93LT5oZWlnaHQ7IH0KLSAgICB2aXJ0dWFsICAgICB2b2lkKiAgICAgICBnZXRCaXRzKCkgY29uc3Q7Ci0gICAgdmlydHVhbCAgICAgRUdMaW50ICAgICAgZ2V0SG9yaXpvbnRhbFJlc29sdXRpb24oKSBjb25zdDsKLSAgICB2aXJ0dWFsICAgICBFR0xpbnQgICAgICBnZXRWZXJ0aWNhbFJlc29sdXRpb24oKSBjb25zdDsKLSAgICB2aXJ0dWFsICAgICBFR0xpbnQgICAgICBnZXRSZWZyZXNoUmF0ZSgpIGNvbnN0OwotICAgIHZpcnR1YWwgICAgIEVHTGludCAgICAgIGdldFN3YXBCZWhhdmlvcigpIGNvbnN0OwotcHJpdmF0ZToKLSAgICBlZ2xfbmF0aXZlX3dpbmRvd190KiAgICBuYXRpdmVXaW5kb3c7Ci19OwotCi1lZ2xfd2luZG93X3N1cmZhY2VfdDo6ZWdsX3dpbmRvd19zdXJmYWNlX3QoRUdMRGlzcGxheSBkcHksCi0gICAgICAgIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgIGludDMyX3QgZGVwdGhGb3JtYXQsCi0gICAgICAgIGVnbF9uYXRpdmVfd2luZG93X3QqIHdpbmRvdykKLSAgICA6IGVnbF9zdXJmYWNlX3QoZHB5LCBjb25maWcsIGRlcHRoRm9ybWF0KSwgbmF0aXZlV2luZG93KHdpbmRvdykKLXsKLSAgICBpZiAoZGVwdGhGb3JtYXQpIHsKLSAgICAgICAgZGVwdGgud2lkdGggICA9IHdpbmRvdy0+d2lkdGg7Ci0gICAgICAgIGRlcHRoLmhlaWdodCAgPSB3aW5kb3ctPmhlaWdodDsKLSAgICAgICAgZGVwdGguc3RyaWRlICA9IGRlcHRoLndpZHRoOyAvLyB1c2UgdGhlIHdpZHRoIGhlcmUKLSAgICAgICAgZGVwdGguZGF0YSAgICA9IChHR0x1Ynl0ZSopbWFsbG9jKGRlcHRoLnN0cmlkZSpkZXB0aC5oZWlnaHQqMik7Ci0gICAgICAgIGlmIChkZXB0aC5kYXRhID09IDApIHsKLSAgICAgICAgICAgIHNldEVycm9yKEVHTF9CQURfQUxMT0MsIEVHTF9OT19TVVJGQUNFKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgIH0KLSAgICBuYXRpdmVXaW5kb3ctPmluY1JlZihuYXRpdmVXaW5kb3cpOwotfQotZWdsX3dpbmRvd19zdXJmYWNlX3Q6On5lZ2xfd2luZG93X3N1cmZhY2VfdCgpIHsKLSAgICBuYXRpdmVXaW5kb3ctPmRlY1JlZihuYXRpdmVXaW5kb3cpOwotfQotCi1FR0xCb29sZWFuIGVnbF93aW5kb3dfc3VyZmFjZV90Ojpzd2FwQnVmZmVycygpCi17Ci0gICAgdWludDMyX3QgZmxhZ3MgPSBuYXRpdmVXaW5kb3ctPnN3YXBCdWZmZXJzKG5hdGl2ZVdpbmRvdyk7Ci0gICAgaWYgKGZsYWdzICYgRUdMX05BVElWRVNfRkxBR19TSVpFX0NIQU5HRUQpIHsKLSAgICAgICAgLy8gVE9ETzogd2UgcHJvYmFibHkgc2hvdWxkIHJlc2V0IHRoZSBzd2FwIHJlY3QgaGVyZQotICAgICAgICAvLyBpZiB0aGUgd2luZG93IHNpemUgaGFzIGNoYW5nZWQKLSAgICAgICAgaWYgKGRlcHRoLmRhdGEpIHsKLSAgICAgICAgICAgIGZyZWUoZGVwdGguZGF0YSk7Ci0gICAgICAgICAgICBkZXB0aC53aWR0aCAgID0gbmF0aXZlV2luZG93LT53aWR0aDsKLSAgICAgICAgICAgIGRlcHRoLmhlaWdodCAgPSBuYXRpdmVXaW5kb3ctPmhlaWdodDsKLSAgICAgICAgICAgIGRlcHRoLnN0cmlkZSAgPSBuYXRpdmVXaW5kb3ctPnN0cmlkZTsKLSAgICAgICAgICAgIGRlcHRoLmRhdGEgICAgPSAoR0dMdWJ5dGUqKW1hbGxvYyhkZXB0aC5zdHJpZGUqZGVwdGguaGVpZ2h0KjIpOwotICAgICAgICAgICAgaWYgKGRlcHRoLmRhdGEgPT0gMCkgewotICAgICAgICAgICAgICAgIHNldEVycm9yKEVHTF9CQURfQUxMT0MsIEVHTF9OT19TVVJGQUNFKTsKLSAgICAgICAgICAgICAgICByZXR1cm4gRUdMX0ZBTFNFOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBFR0xfVFJVRTsKLX0KLQotRUdMQm9vbGVhbiBlZ2xfd2luZG93X3N1cmZhY2VfdDo6YmluZERyYXdTdXJmYWNlKG9nbGVzX2NvbnRleHRfdCogZ2wpCi17Ci0gICAgR0dMU3VyZmFjZSBidWZmZXI7Ci0gICAgYnVmZmVyLnZlcnNpb24gPSBzaXplb2YoR0dMU3VyZmFjZSk7Ci0gICAgYnVmZmVyLndpZHRoICAgPSBuYXRpdmVXaW5kb3ctPndpZHRoOwotICAgIGJ1ZmZlci5oZWlnaHQgID0gbmF0aXZlV2luZG93LT5oZWlnaHQ7Ci0gICAgYnVmZmVyLnN0cmlkZSAgPSBuYXRpdmVXaW5kb3ctPnN0cmlkZTsKLSAgICBidWZmZXIuZGF0YSAgICA9IChHR0x1Ynl0ZSopbmF0aXZlV2luZG93LT5iYXNlICsgbmF0aXZlV2luZG93LT5vZmZzZXQ7Ci0gICAgYnVmZmVyLmZvcm1hdCAgPSBuYXRpdmVXaW5kb3ctPmZvcm1hdDsKLSAgICBnbC0+cmFzdGVyaXplci5wcm9jcy5jb2xvckJ1ZmZlcihnbCwgJmJ1ZmZlcik7Ci0gICAgaWYgKGRlcHRoLmRhdGEgIT0gZ2wtPnJhc3Rlcml6ZXIuc3RhdGUuYnVmZmVycy5kZXB0aC5kYXRhKQotICAgICAgICBnbC0+cmFzdGVyaXplci5wcm9jcy5kZXB0aEJ1ZmZlcihnbCwgJmRlcHRoKTsKLSAgICByZXR1cm4gRUdMX1RSVUU7Ci19Ci1FR0xCb29sZWFuIGVnbF93aW5kb3dfc3VyZmFjZV90OjpiaW5kUmVhZFN1cmZhY2Uob2dsZXNfY29udGV4dF90KiBnbCkKLXsKLSAgICBHR0xTdXJmYWNlIGJ1ZmZlcjsKLSAgICBidWZmZXIudmVyc2lvbiA9IHNpemVvZihHR0xTdXJmYWNlKTsKLSAgICBidWZmZXIud2lkdGggICA9IG5hdGl2ZVdpbmRvdy0+d2lkdGg7Ci0gICAgYnVmZmVyLmhlaWdodCAgPSBuYXRpdmVXaW5kb3ctPmhlaWdodDsKLSAgICBidWZmZXIuc3RyaWRlICA9IG5hdGl2ZVdpbmRvdy0+c3RyaWRlOwotICAgIGJ1ZmZlci5kYXRhICAgID0gKEdHTHVieXRlKiluYXRpdmVXaW5kb3ctPmJhc2UgKyBuYXRpdmVXaW5kb3ctPm9mZnNldDsKLSAgICBidWZmZXIuZm9ybWF0ICA9IG5hdGl2ZVdpbmRvdy0+Zm9ybWF0OwotICAgIGdsLT5yYXN0ZXJpemVyLnByb2NzLnJlYWRCdWZmZXIoZ2wsICZidWZmZXIpOwotICAgIHJldHVybiBFR0xfVFJVRTsKLX0KLXZvaWQqIGVnbF93aW5kb3dfc3VyZmFjZV90OjpnZXRCaXRzKCkgY29uc3QgewotICAgIHJldHVybiAoR0dMdWJ5dGUqKW5hdGl2ZVdpbmRvdy0+YmFzZSArIG5hdGl2ZVdpbmRvdy0+b2Zmc2V0OwotfQotRUdMaW50IGVnbF93aW5kb3dfc3VyZmFjZV90OjpnZXRIb3Jpem9udGFsUmVzb2x1dGlvbigpIGNvbnN0IHsKLSAgICByZXR1cm4gKG5hdGl2ZVdpbmRvdy0+eGRwaSAqIEVHTF9ESVNQTEFZX1NDQUxJTkcpICogKDEuMGYgLyAyNS40Zik7Ci19Ci1FR0xpbnQgZWdsX3dpbmRvd19zdXJmYWNlX3Q6OmdldFZlcnRpY2FsUmVzb2x1dGlvbigpIGNvbnN0IHsKLSAgICByZXR1cm4gKG5hdGl2ZVdpbmRvdy0+eWRwaSAqIEVHTF9ESVNQTEFZX1NDQUxJTkcpICogKDEuMGYgLyAyNS40Zik7Ci19Ci1FR0xpbnQgZWdsX3dpbmRvd19zdXJmYWNlX3Q6OmdldFJlZnJlc2hSYXRlKCkgY29uc3QgewotICAgIHJldHVybiAobmF0aXZlV2luZG93LT5mcHMgKiBFR0xfRElTUExBWV9TQ0FMSU5HKTsKLX0KLUVHTGludCBlZ2xfd2luZG93X3N1cmZhY2VfdDo6Z2V0U3dhcEJlaGF2aW9yKCkgY29uc3QgewotICAgIHVpbnQzMl90IGZsYWdzID0gbmF0aXZlV2luZG93LT5mbGFnczsKLSAgICBpZiAoZmxhZ3MgJiBFR0xfTkFUSVZFU19GTEFHX0RFU1RST1lfQkFDS0JVRkZFUikKLSAgICAgICAgcmV0dXJuIEVHTF9CVUZGRVJfREVTVFJPWUVEOwotICAgIHJldHVybiBFR0xfQlVGRkVSX1BSRVNFUlZFRDsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3QgZWdsX3BpeG1hcF9zdXJmYWNlX3QgOiBwdWJsaWMgZWdsX3N1cmZhY2VfdAotewotICAgIGVnbF9waXhtYXBfc3VyZmFjZV90KAotICAgICAgICAgICAgRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgICAgICBpbnQzMl90IGRlcHRoRm9ybWF0LAotICAgICAgICAgICAgZWdsX25hdGl2ZV9waXhtYXBfdCBjb25zdCAqIHBpeG1hcCk7Ci0KLSAgICB2aXJ0dWFsIH5lZ2xfcGl4bWFwX3N1cmZhY2VfdCgpIHsgfQotCi0gICAgdmlydHVhbCAgICAgYm9vbCAgICAgICAgaXNWYWxpZCgpIGNvbnN0IHsgcmV0dXJuIG5hdGl2ZVBpeG1hcC52ZXJzaW9uID09IHNpemVvZihlZ2xfbmF0aXZlX3BpeG1hcF90KTsgfSAgICAKLSAgICB2aXJ0dWFsICAgICBFR0xCb29sZWFuICBiaW5kRHJhd1N1cmZhY2Uob2dsZXNfY29udGV4dF90KiBnbCk7Ci0gICAgdmlydHVhbCAgICAgRUdMQm9vbGVhbiAgYmluZFJlYWRTdXJmYWNlKG9nbGVzX2NvbnRleHRfdCogZ2wpOwotICAgIHZpcnR1YWwgICAgIEVHTGludCAgICAgIGdldFdpZHRoKCkgY29uc3QgICAgeyByZXR1cm4gbmF0aXZlUGl4bWFwLndpZHRoOyAgfQotICAgIHZpcnR1YWwgICAgIEVHTGludCAgICAgIGdldEhlaWdodCgpIGNvbnN0ICAgeyByZXR1cm4gbmF0aXZlUGl4bWFwLmhlaWdodDsgfQotICAgIHZpcnR1YWwgICAgIHZvaWQqICAgICAgIGdldEJpdHMoKSBjb25zdCAgICAgeyByZXR1cm4gbmF0aXZlUGl4bWFwLmRhdGE7IH0KLXByaXZhdGU6Ci0gICAgZWdsX25hdGl2ZV9waXhtYXBfdCAgICAgbmF0aXZlUGl4bWFwOwotfTsKLQotZWdsX3BpeG1hcF9zdXJmYWNlX3Q6OmVnbF9waXhtYXBfc3VyZmFjZV90KEVHTERpc3BsYXkgZHB5LAotICAgICAgICBFR0xDb25maWcgY29uZmlnLAotICAgICAgICBpbnQzMl90IGRlcHRoRm9ybWF0LAotICAgICAgICBlZ2xfbmF0aXZlX3BpeG1hcF90IGNvbnN0ICogcGl4bWFwKQotICAgIDogZWdsX3N1cmZhY2VfdChkcHksIGNvbmZpZywgZGVwdGhGb3JtYXQpLCBuYXRpdmVQaXhtYXAoKnBpeG1hcCkKLXsKLSAgICBpZiAoZGVwdGhGb3JtYXQpIHsKLSAgICAgICAgZGVwdGgud2lkdGggICA9IHBpeG1hcC0+d2lkdGg7Ci0gICAgICAgIGRlcHRoLmhlaWdodCAgPSBwaXhtYXAtPmhlaWdodDsKLSAgICAgICAgZGVwdGguc3RyaWRlICA9IGRlcHRoLndpZHRoOyAvLyB1c2UgdGhlIHdpZHRoIGhlcmUKLSAgICAgICAgZGVwdGguZGF0YSAgICA9IChHR0x1Ynl0ZSopbWFsbG9jKGRlcHRoLnN0cmlkZSpkZXB0aC5oZWlnaHQqMik7Ci0gICAgICAgIGlmIChkZXB0aC5kYXRhID09IDApIHsKLSAgICAgICAgICAgIHNldEVycm9yKEVHTF9CQURfQUxMT0MsIEVHTF9OT19TVVJGQUNFKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgIH0KLX0KLUVHTEJvb2xlYW4gZWdsX3BpeG1hcF9zdXJmYWNlX3Q6OmJpbmREcmF3U3VyZmFjZShvZ2xlc19jb250ZXh0X3QqIGdsKQotewotICAgIEdHTFN1cmZhY2UgYnVmZmVyOwotICAgIGJ1ZmZlci52ZXJzaW9uID0gc2l6ZW9mKEdHTFN1cmZhY2UpOwotICAgIGJ1ZmZlci53aWR0aCAgID0gbmF0aXZlUGl4bWFwLndpZHRoOwotICAgIGJ1ZmZlci5oZWlnaHQgID0gbmF0aXZlUGl4bWFwLmhlaWdodDsKLSAgICBidWZmZXIuc3RyaWRlICA9IG5hdGl2ZVBpeG1hcC5zdHJpZGU7Ci0gICAgYnVmZmVyLmRhdGEgICAgPSBuYXRpdmVQaXhtYXAuZGF0YTsKLSAgICBidWZmZXIuZm9ybWF0ICA9IG5hdGl2ZVBpeG1hcC5mb3JtYXQ7Ci0gICAgCi0gICAgZ2wtPnJhc3Rlcml6ZXIucHJvY3MuY29sb3JCdWZmZXIoZ2wsICZidWZmZXIpOwotICAgIGlmIChkZXB0aC5kYXRhICE9IGdsLT5yYXN0ZXJpemVyLnN0YXRlLmJ1ZmZlcnMuZGVwdGguZGF0YSkKLSAgICAgICAgZ2wtPnJhc3Rlcml6ZXIucHJvY3MuZGVwdGhCdWZmZXIoZ2wsICZkZXB0aCk7Ci0gICAgcmV0dXJuIEVHTF9UUlVFOwotfQotRUdMQm9vbGVhbiBlZ2xfcGl4bWFwX3N1cmZhY2VfdDo6YmluZFJlYWRTdXJmYWNlKG9nbGVzX2NvbnRleHRfdCogZ2wpCi17Ci0gICAgR0dMU3VyZmFjZSBidWZmZXI7Ci0gICAgYnVmZmVyLnZlcnNpb24gPSBzaXplb2YoR0dMU3VyZmFjZSk7Ci0gICAgYnVmZmVyLndpZHRoICAgPSBuYXRpdmVQaXhtYXAud2lkdGg7Ci0gICAgYnVmZmVyLmhlaWdodCAgPSBuYXRpdmVQaXhtYXAuaGVpZ2h0OwotICAgIGJ1ZmZlci5zdHJpZGUgID0gbmF0aXZlUGl4bWFwLnN0cmlkZTsKLSAgICBidWZmZXIuZGF0YSAgICA9IG5hdGl2ZVBpeG1hcC5kYXRhOwotICAgIGJ1ZmZlci5mb3JtYXQgID0gbmF0aXZlUGl4bWFwLmZvcm1hdDsKLSAgICBnbC0+cmFzdGVyaXplci5wcm9jcy5yZWFkQnVmZmVyKGdsLCAmYnVmZmVyKTsKLSAgICByZXR1cm4gRUdMX1RSVUU7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RydWN0IGVnbF9wYnVmZmVyX3N1cmZhY2VfdCA6IHB1YmxpYyBlZ2xfc3VyZmFjZV90Ci17Ci0gICAgZWdsX3BidWZmZXJfc3VyZmFjZV90KAotICAgICAgICAgICAgRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsIGludDMyX3QgZGVwdGhGb3JtYXQsCi0gICAgICAgICAgICBpbnQzMl90IHcsIGludDMyX3QgaCwgaW50MzJfdCBmKTsKLQotICAgIHZpcnR1YWwgfmVnbF9wYnVmZmVyX3N1cmZhY2VfdCgpOwotCi0gICAgdmlydHVhbCAgICAgYm9vbCAgICAgICAgaXNWYWxpZCgpIGNvbnN0IHsgcmV0dXJuIHBidWZmZXIuZGF0YSAhPSAwOyB9ICAgIAotICAgIHZpcnR1YWwgICAgIEVHTEJvb2xlYW4gIGJpbmREcmF3U3VyZmFjZShvZ2xlc19jb250ZXh0X3QqIGdsKTsKLSAgICB2aXJ0dWFsICAgICBFR0xCb29sZWFuICBiaW5kUmVhZFN1cmZhY2Uob2dsZXNfY29udGV4dF90KiBnbCk7Ci0gICAgdmlydHVhbCAgICAgRUdMaW50ICAgICAgZ2V0V2lkdGgoKSBjb25zdCAgICB7IHJldHVybiBwYnVmZmVyLndpZHRoOyAgfQotICAgIHZpcnR1YWwgICAgIEVHTGludCAgICAgIGdldEhlaWdodCgpIGNvbnN0ICAgeyByZXR1cm4gcGJ1ZmZlci5oZWlnaHQ7IH0KLSAgICB2aXJ0dWFsICAgICB2b2lkKiAgICAgICBnZXRCaXRzKCkgY29uc3QgICAgIHsgcmV0dXJuIHBidWZmZXIuZGF0YTsgfQotcHJpdmF0ZToKLSAgICBHR0xTdXJmYWNlICBwYnVmZmVyOwotfTsKLQotZWdsX3BidWZmZXJfc3VyZmFjZV90OjplZ2xfcGJ1ZmZlcl9zdXJmYWNlX3QoRUdMRGlzcGxheSBkcHksCi0gICAgICAgIEVHTENvbmZpZyBjb25maWcsIGludDMyX3QgZGVwdGhGb3JtYXQsCi0gICAgICAgIGludDMyX3QgdywgaW50MzJfdCBoLCBpbnQzMl90IGYpCi0gICAgOiBlZ2xfc3VyZmFjZV90KGRweSwgY29uZmlnLCBkZXB0aEZvcm1hdCkKLXsKLSAgICBzaXplX3Qgc2l6ZSA9IHcqaDsKLSAgICBzd2l0Y2ggKGYpIHsKLSAgICAgICAgY2FzZSBHR0xfUElYRUxfRk9STUFUX0FfODogICAgICAgICAgc2l6ZSAqPSAxOyBicmVhazsKLSAgICAgICAgY2FzZSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjU6ICAgICAgc2l6ZSAqPSAyOyBicmVhazsKLSAgICAgICAgY2FzZSBHR0xfUElYRUxfRk9STUFUX1JHQkFfODg4ODogICAgc2l6ZSAqPSA0OyBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIExPR0UoImluY29tcGF0aWJsZSBwaXhlbCBmb3JtYXQgZm9yIHBidWZmZXIgKGZvcm1hdD0lZCkiLCBmKTsKLSAgICAgICAgICAgIHBidWZmZXIuZGF0YSA9IDA7Ci0gICAgICAgICAgICBicmVhazsKLSAgICB9Ci0gICAgcGJ1ZmZlci52ZXJzaW9uID0gc2l6ZW9mKEdHTFN1cmZhY2UpOwotICAgIHBidWZmZXIud2lkdGggICA9IHc7Ci0gICAgcGJ1ZmZlci5oZWlnaHQgID0gaDsKLSAgICBwYnVmZmVyLnN0cmlkZSAgPSB3OwotICAgIHBidWZmZXIuZGF0YSAgICA9IChHR0x1Ynl0ZSopbWFsbG9jKHNpemUpOwotICAgIHBidWZmZXIuZm9ybWF0ICA9IGY7Ci0gICAgCi0gICAgaWYgKGRlcHRoRm9ybWF0KSB7Ci0gICAgICAgIGRlcHRoLndpZHRoICAgPSBwYnVmZmVyLndpZHRoOwotICAgICAgICBkZXB0aC5oZWlnaHQgID0gcGJ1ZmZlci5oZWlnaHQ7Ci0gICAgICAgIGRlcHRoLnN0cmlkZSAgPSBkZXB0aC53aWR0aDsgLy8gdXNlIHRoZSB3aWR0aCBoZXJlCi0gICAgICAgIGRlcHRoLmRhdGEgICAgPSAoR0dMdWJ5dGUqKW1hbGxvYyhkZXB0aC5zdHJpZGUqZGVwdGguaGVpZ2h0KjIpOwotICAgICAgICBpZiAoZGVwdGguZGF0YSA9PSAwKSB7Ci0gICAgICAgICAgICBzZXRFcnJvcihFR0xfQkFEX0FMTE9DLCBFR0xfTk9fU1VSRkFDRSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICB9Ci19Ci1lZ2xfcGJ1ZmZlcl9zdXJmYWNlX3Q6On5lZ2xfcGJ1ZmZlcl9zdXJmYWNlX3QoKSB7Ci0gICAgZnJlZShwYnVmZmVyLmRhdGEpOwotfQotRUdMQm9vbGVhbiBlZ2xfcGJ1ZmZlcl9zdXJmYWNlX3Q6OmJpbmREcmF3U3VyZmFjZShvZ2xlc19jb250ZXh0X3QqIGdsKQotewotICAgIGdsLT5yYXN0ZXJpemVyLnByb2NzLmNvbG9yQnVmZmVyKGdsLCAmcGJ1ZmZlcik7Ci0gICAgaWYgKGRlcHRoLmRhdGEgIT0gZ2wtPnJhc3Rlcml6ZXIuc3RhdGUuYnVmZmVycy5kZXB0aC5kYXRhKQotICAgICAgICBnbC0+cmFzdGVyaXplci5wcm9jcy5kZXB0aEJ1ZmZlcihnbCwgJmRlcHRoKTsKLSAgICByZXR1cm4gRUdMX1RSVUU7Ci19Ci1FR0xCb29sZWFuIGVnbF9wYnVmZmVyX3N1cmZhY2VfdDo6YmluZFJlYWRTdXJmYWNlKG9nbGVzX2NvbnRleHRfdCogZ2wpCi17Ci0gICAgZ2wtPnJhc3Rlcml6ZXIucHJvY3MucmVhZEJ1ZmZlcihnbCwgJnBidWZmZXIpOwotICAgIHJldHVybiBFR0xfVFJVRTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3QgY29uZmlnX3BhaXJfdCB7Ci0gICAgR0xpbnQga2V5OwotICAgIEdMaW50IHZhbHVlOwotfTsKLQotc3RydWN0IGNvbmZpZ3NfdCB7Ci0gICAgY29uc3QgY29uZmlnX3BhaXJfdCogYXJyYXk7Ci0gICAgaW50ICAgICAgICAgICAgICAgICAgc2l6ZTsKLX07Ci0KLXN0cnVjdCBjb25maWdfbWFuYWdlbWVudF90IHsKLSAgICBHTGludCBrZXk7Ci0gICAgYm9vbCAoKm1hdGNoKShHTGludCByZXFWYWx1ZSwgR0xpbnQgY29uZlZhbHVlKTsKLSAgICBzdGF0aWMgYm9vbCBhdExlYXN0KEdMaW50IHJlcVZhbHVlLCBHTGludCBjb25mVmFsdWUpIHsKLSAgICAgICAgcmV0dXJuIChyZXFWYWx1ZSA9PSBFR0xfRE9OVF9DQVJFKSB8fCAoY29uZlZhbHVlID49IHJlcVZhbHVlKTsKLSAgICB9Ci0gICAgc3RhdGljIGJvb2wgZXhhY3QoR0xpbnQgcmVxVmFsdWUsIEdMaW50IGNvbmZWYWx1ZSkgewotICAgICAgICByZXR1cm4gKHJlcVZhbHVlID09IEVHTF9ET05UX0NBUkUpIHx8IChjb25mVmFsdWUgPT0gcmVxVmFsdWUpOwotICAgIH0KLSAgICBzdGF0aWMgYm9vbCBtYXNrKEdMaW50IHJlcVZhbHVlLCBHTGludCBjb25mVmFsdWUpIHsKLSAgICAgICAgcmV0dXJuIChjb25mVmFsdWUgJiByZXFWYWx1ZSkgPT0gcmVxVmFsdWU7Ci0gICAgfQotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgY2hhciBjb25zdCAqIGNvbnN0IGdWZW5kb3JTdHJpbmcgICAgID0gIkdvb2dsZSBJbmMuIjsKLXN0YXRpYyBjaGFyIGNvbnN0ICogY29uc3QgZ1ZlcnNpb25TdHJpbmcgICAgPSAiMS4yIEFuZHJvaWQgRHJpdmVyIjsKLXN0YXRpYyBjaGFyIGNvbnN0ICogY29uc3QgZ0NsaWVudEFwaVN0cmluZyAgPSAiT3BlbkdMIEVTIjsKLXN0YXRpYyBjaGFyIGNvbnN0ICogY29uc3QgZ0V4dGVuc2lvbnNTdHJpbmcgPSAiIjsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdHJ1Y3QgZXh0ZW50aW9uX21hcF90IHsKLSAgICBjb25zdCBjaGFyICogY29uc3QgbmFtZTsKLSAgICBfX2VnbE11c3RDYXN0VG9Qcm9wZXJGdW5jdGlvblBvaW50ZXJUeXBlIGFkZHJlc3M7Ci19OwotCi1zdGF0aWMgY29uc3QgZXh0ZW50aW9uX21hcF90IGdFeHRlbnRpb25NYXBbXSA9IHsKLSAgICB7ICJnbERyYXdUZXhzT0VTIiwgICAgICAgICAgICAgICh2b2lkKCopKCkpJmdsRHJhd1RleHNPRVMgfSwKLSAgICB7ICJnbERyYXdUZXhpT0VTIiwgICAgICAgICAgICAgICh2b2lkKCopKCkpJmdsRHJhd1RleGlPRVMgfSwKLSAgICB7ICJnbERyYXdUZXhmT0VTIiwgICAgICAgICAgICAgICh2b2lkKCopKCkpJmdsRHJhd1RleGZPRVMgfSwKLSAgICB7ICJnbERyYXdUZXh4T0VTIiwgICAgICAgICAgICAgICh2b2lkKCopKCkpJmdsRHJhd1RleHhPRVMgfSwKLSAgICB7ICJnbERyYXdUZXhzdk9FUyIsICAgICAgICAgICAgICh2b2lkKCopKCkpJmdsRHJhd1RleHN2T0VTIH0sCi0gICAgeyAiZ2xEcmF3VGV4aXZPRVMiLCAgICAgICAgICAgICAodm9pZCgqKSgpKSZnbERyYXdUZXhpdk9FUyB9LAotICAgIHsgImdsRHJhd1RleGZ2T0VTIiwgICAgICAgICAgICAgKHZvaWQoKikoKSkmZ2xEcmF3VGV4ZnZPRVMgfSwKLSAgICB7ICJnbERyYXdUZXh4dk9FUyIsICAgICAgICAgICAgICh2b2lkKCopKCkpJmdsRHJhd1RleHh2T0VTIH0sCi0gICAgeyAiZ2xRdWVyeU1hdHJpeHhPRVMiLCAgICAgICAgICAodm9pZCgqKSgpKSZnbFF1ZXJ5TWF0cml4eE9FUyB9LAotICAgIHsgImdsQ2xpcFBsYW5lZiIsICAgICAgICAgICAgICAgKHZvaWQoKikoKSkmZ2xDbGlwUGxhbmVmIH0sCi0gICAgeyAiZ2xDbGlwUGxhbmV4IiwgICAgICAgICAgICAgICAodm9pZCgqKSgpKSZnbENsaXBQbGFuZXggfSwKLSAgICB7ICJnbEJpbmRCdWZmZXIiLCAgICAgICAgICAgICAgICh2b2lkKCopKCkpJmdsQmluZEJ1ZmZlciB9LAotICAgIHsgImdsQnVmZmVyRGF0YSIsICAgICAgICAgICAgICAgKHZvaWQoKikoKSkmZ2xCdWZmZXJEYXRhIH0sCi0gICAgeyAiZ2xCdWZmZXJTdWJEYXRhIiwgICAgICAgICAgICAodm9pZCgqKSgpKSZnbEJ1ZmZlclN1YkRhdGEgfSwKLSAgICB7ICJnbERlbGV0ZUJ1ZmZlcnMiLCAgICAgICAgICAgICh2b2lkKCopKCkpJmdsRGVsZXRlQnVmZmVycyB9LAotICAgIHsgImdsR2VuQnVmZmVycyIsICAgICAgICAgICAgICAgKHZvaWQoKikoKSkmZ2xHZW5CdWZmZXJzIH0sCi19OwotCi0vKiAKLSAqIEluIHRoZSBsaXN0cyBiZWxvdywgYXR0cmlidXRlcyBuYW1lcyBNVVNUIGJlIHNvcnRlZC4KLSAqIEFkZGl0aW9uYWxseSwgYWxsIGNvbmZpZ3MgbXVzdCBiZSBzb3J0ZWQgYWNjb3JkaW5nIHRvCi0gKiB0aGUgRUdMIHNwZWNpZmljYXRpb24uCi0gKi8KLQotc3RhdGljIGNvbmZpZ19wYWlyX3QgY29uc3QgY29uZmlnX2Jhc2VfYXR0cmlidXRlX2xpc3RbXSA9IHsKLSAgICAgICAgeyBFR0xfU1RFTkNJTF9TSVpFLCAgICAgICAgICAgICAgIDAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LAotICAgICAgICB7IEVHTF9DT05GSUdfQ0FWRUFULCAgICAgICAgICAgICAgRUdMX1NMT1dfQ09ORklHICAgICAgICAgICAgICAgICAgIH0sCi0gICAgICAgIHsgRUdMX0xFVkVMLCAgICAgICAgICAgICAgICAgICAgICAwICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSwKLSAgICAgICAgeyBFR0xfTUFYX1BCVUZGRVJfSEVJR0hULCAgICAgICAgIEdHTF9NQVhfVklFV1BPUlRfRElNUyAgICAgICAgICAgICB9LAotICAgICAgICB7IEVHTF9NQVhfUEJVRkZFUl9QSVhFTFMsICAgICAgICAgCi0gICAgICAgICAgICAgICAgR0dMX01BWF9WSUVXUE9SVF9ESU1TKkdHTF9NQVhfVklFV1BPUlRfRElNUyAgICAgICAgICAgICAgICAgfSwKLSAgICAgICAgeyBFR0xfTUFYX1BCVUZGRVJfV0lEVEgsICAgICAgICAgIEdHTF9NQVhfVklFV1BPUlRfRElNUyAgICAgICAgICAgICB9LAotICAgICAgICB7IEVHTF9OQVRJVkVfUkVOREVSQUJMRSwgICAgICAgICAgRUdMX1RSVUUgICAgICAgICAgICAgICAgICAgICAgICAgIH0sCi0gICAgICAgIHsgRUdMX05BVElWRV9WSVNVQUxfSUQsICAgICAgICAgICAwICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSwKLSAgICAgICAgeyBFR0xfTkFUSVZFX1ZJU1VBTF9UWVBFLCAgICAgICAgIEdHTF9QSVhFTF9GT1JNQVRfUkdCXzU2NSAgICAgICAgICB9LAotICAgICAgICB7IEVHTF9TQU1QTEVTLCAgICAgICAgICAgICAgICAgICAgMCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sCi0gICAgICAgIHsgRUdMX1NBTVBMRV9CVUZGRVJTLCAgICAgICAgICAgICAwICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSwKLSAgICAgICAgeyBFR0xfVFJBTlNQQVJFTlRfVFlQRSwgICAgICAgICAgIEVHTF9OT05FICAgICAgICAgICAgICAgICAgICAgICAgICB9LAotICAgICAgICB7IEVHTF9UUkFOU1BBUkVOVF9CTFVFX1ZBTFVFLCAgICAgMCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sCi0gICAgICAgIHsgRUdMX1RSQU5TUEFSRU5UX0dSRUVOX1ZBTFVFLCAgICAwICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSwKLSAgICAgICAgeyBFR0xfVFJBTlNQQVJFTlRfUkVEX1ZBTFVFLCAgICAgIDAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LAotICAgICAgICB7IEVHTF9CSU5EX1RPX1RFWFRVUkVfUkdCQSwgICAgICAgRUdMX0ZBTFNFICAgICAgICAgICAgICAgICAgICAgICAgIH0sCi0gICAgICAgIHsgRUdMX0JJTkRfVE9fVEVYVFVSRV9SR0IsICAgICAgICBFR0xfRkFMU0UgICAgICAgICAgICAgICAgICAgICAgICAgfSwKLSAgICAgICAgeyBFR0xfTUlOX1NXQVBfSU5URVJWQUwsICAgICAgICAgIDEgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LAotICAgICAgICB7IEVHTF9NQVhfU1dBUF9JTlRFUlZBTCwgICAgICAgICAgNCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sCi19OwotCi0vLyBUaGVzZSBjb25maWdzIGNhbiBvdmVycmlkZSB0aGUgYmFzZSBhdHRyaWJ1dGUgbGlzdAotLy8gTk9URTogd2hlbiBhZGRpbmcgYSBjb25maWcgaGVyZSwgZG9uJ3QgZm9yZ2V0IHRvIHVwZGF0ZSBlZ2xDcmVhdGUqU3VyZmFjZSgpCi0KLXN0YXRpYyBjb25maWdfcGFpcl90IGNvbnN0IGNvbmZpZ18wX2F0dHJpYnV0ZV9saXN0W10gPSB7Ci0gICAgICAgIHsgRUdMX0JVRkZFUl9TSVpFLCAgICAgMTYgfSwKLSAgICAgICAgeyBFR0xfQUxQSEFfU0laRSwgICAgICAgMCB9LAotICAgICAgICB7IEVHTF9CTFVFX1NJWkUsICAgICAgICA1IH0sCi0gICAgICAgIHsgRUdMX0dSRUVOX1NJWkUsICAgICAgIDYgfSwKLSAgICAgICAgeyBFR0xfUkVEX1NJWkUsICAgICAgICAgNSB9LAotICAgICAgICB7IEVHTF9ERVBUSF9TSVpFLCAgICAgICAwIH0sCi0gICAgICAgIHsgRUdMX0NPTkZJR19JRCwgICAgICAgIDAgfSwKLSAgICAgICAgeyBFR0xfU1VSRkFDRV9UWVBFLCAgICAgRUdMX1dJTkRPV19CSVR8RUdMX1BCVUZGRVJfQklUfEVHTF9QSVhNQVBfQklUIH0sCi19OwotCi1zdGF0aWMgY29uZmlnX3BhaXJfdCBjb25zdCBjb25maWdfMV9hdHRyaWJ1dGVfbGlzdFtdID0gewotICAgICAgICB7IEVHTF9CVUZGRVJfU0laRSwgICAgIDE2IH0sCi0gICAgICAgIHsgRUdMX0FMUEhBX1NJWkUsICAgICAgIDAgfSwKLSAgICAgICAgeyBFR0xfQkxVRV9TSVpFLCAgICAgICAgNSB9LAotICAgICAgICB7IEVHTF9HUkVFTl9TSVpFLCAgICAgICA2IH0sCi0gICAgICAgIHsgRUdMX1JFRF9TSVpFLCAgICAgICAgIDUgfSwKLSAgICAgICAgeyBFR0xfREVQVEhfU0laRSwgICAgICAxNiB9LAotICAgICAgICB7IEVHTF9DT05GSUdfSUQsICAgICAgICAxIH0sCi0gICAgICAgIHsgRUdMX1NVUkZBQ0VfVFlQRSwgICAgIEVHTF9XSU5ET1dfQklUfEVHTF9QQlVGRkVSX0JJVHxFR0xfUElYTUFQX0JJVCB9LAotfTsKLQotc3RhdGljIGNvbmZpZ19wYWlyX3QgY29uc3QgY29uZmlnXzJfYXR0cmlidXRlX2xpc3RbXSA9IHsKLSAgICAgICAgeyBFR0xfQlVGRkVSX1NJWkUsICAgICAzMiB9LAotICAgICAgICB7IEVHTF9BTFBIQV9TSVpFLCAgICAgICA4IH0sCi0gICAgICAgIHsgRUdMX0JMVUVfU0laRSwgICAgICAgIDggfSwKLSAgICAgICAgeyBFR0xfR1JFRU5fU0laRSwgICAgICAgOCB9LAotICAgICAgICB7IEVHTF9SRURfU0laRSwgICAgICAgICA4IH0sCi0gICAgICAgIHsgRUdMX0RFUFRIX1NJWkUsICAgICAgIDAgfSwKLSAgICAgICAgeyBFR0xfQ09ORklHX0lELCAgICAgICAgMiB9LAotICAgICAgICB7IEVHTF9TVVJGQUNFX1RZUEUsICAgICBFR0xfV0lORE9XX0JJVHxFR0xfUEJVRkZFUl9CSVR8RUdMX1BJWE1BUF9CSVQgfSwKLX07Ci0KLXN0YXRpYyBjb25maWdfcGFpcl90IGNvbnN0IGNvbmZpZ18zX2F0dHJpYnV0ZV9saXN0W10gPSB7Ci0gICAgICAgIHsgRUdMX0JVRkZFUl9TSVpFLCAgICAgMzIgfSwKLSAgICAgICAgeyBFR0xfQUxQSEFfU0laRSwgICAgICAgOCB9LAotICAgICAgICB7IEVHTF9CTFVFX1NJWkUsICAgICAgICA4IH0sCi0gICAgICAgIHsgRUdMX0dSRUVOX1NJWkUsICAgICAgIDggfSwKLSAgICAgICAgeyBFR0xfUkVEX1NJWkUsICAgICAgICAgOCB9LAotICAgICAgICB7IEVHTF9ERVBUSF9TSVpFLCAgICAgIDE2IH0sCi0gICAgICAgIHsgRUdMX0NPTkZJR19JRCwgICAgICAgIDMgfSwKLSAgICAgICAgeyBFR0xfU1VSRkFDRV9UWVBFLCAgICAgRUdMX1dJTkRPV19CSVR8RUdMX1BCVUZGRVJfQklUfEVHTF9QSVhNQVBfQklUIH0sCi19OwotCi1zdGF0aWMgY29uZmlnX3BhaXJfdCBjb25zdCBjb25maWdfNF9hdHRyaWJ1dGVfbGlzdFtdID0gewotICAgICAgICB7IEVHTF9CVUZGRVJfU0laRSwgICAgICA4IH0sCi0gICAgICAgIHsgRUdMX0FMUEhBX1NJWkUsICAgICAgIDggfSwKLSAgICAgICAgeyBFR0xfQkxVRV9TSVpFLCAgICAgICAgMCB9LAotICAgICAgICB7IEVHTF9HUkVFTl9TSVpFLCAgICAgICAwIH0sCi0gICAgICAgIHsgRUdMX1JFRF9TSVpFLCAgICAgICAgIDAgfSwKLSAgICAgICAgeyBFR0xfREVQVEhfU0laRSwgICAgICAgMCB9LAotICAgICAgICB7IEVHTF9DT05GSUdfSUQsICAgICAgICA0IH0sCi0gICAgICAgIHsgRUdMX1NVUkZBQ0VfVFlQRSwgICAgIEVHTF9XSU5ET1dfQklUfEVHTF9QQlVGRkVSX0JJVHxFR0xfUElYTUFQX0JJVCB9LAotfTsKLQotc3RhdGljIGNvbmZpZ19wYWlyX3QgY29uc3QgY29uZmlnXzVfYXR0cmlidXRlX2xpc3RbXSA9IHsKLSAgICAgICAgeyBFR0xfQlVGRkVSX1NJWkUsICAgICAgOCB9LAotICAgICAgICB7IEVHTF9BTFBIQV9TSVpFLCAgICAgICA4IH0sCi0gICAgICAgIHsgRUdMX0JMVUVfU0laRSwgICAgICAgIDAgfSwKLSAgICAgICAgeyBFR0xfR1JFRU5fU0laRSwgICAgICAgMCB9LAotICAgICAgICB7IEVHTF9SRURfU0laRSwgICAgICAgICAwIH0sCi0gICAgICAgIHsgRUdMX0RFUFRIX1NJWkUsICAgICAgMTYgfSwKLSAgICAgICAgeyBFR0xfQ09ORklHX0lELCAgICAgICAgNSB9LAotICAgICAgICB7IEVHTF9TVVJGQUNFX1RZUEUsICAgICBFR0xfV0lORE9XX0JJVHxFR0xfUEJVRkZFUl9CSVR8RUdMX1BJWE1BUF9CSVQgfSwKLX07Ci0KLXN0YXRpYyBjb25maWdzX3QgY29uc3QgZ0NvbmZpZ3NbXSA9IHsKLSAgICAgICAgeyBjb25maWdfMF9hdHRyaWJ1dGVfbGlzdCwgTkVMRU0oY29uZmlnXzBfYXR0cmlidXRlX2xpc3QpIH0sCi0gICAgICAgIHsgY29uZmlnXzFfYXR0cmlidXRlX2xpc3QsIE5FTEVNKGNvbmZpZ18xX2F0dHJpYnV0ZV9saXN0KSB9LAotICAgICAgICB7IGNvbmZpZ18yX2F0dHJpYnV0ZV9saXN0LCBORUxFTShjb25maWdfMl9hdHRyaWJ1dGVfbGlzdCkgfSwKLSAgICAgICAgeyBjb25maWdfM19hdHRyaWJ1dGVfbGlzdCwgTkVMRU0oY29uZmlnXzNfYXR0cmlidXRlX2xpc3QpIH0sCi0gICAgICAgIHsgY29uZmlnXzRfYXR0cmlidXRlX2xpc3QsIE5FTEVNKGNvbmZpZ180X2F0dHJpYnV0ZV9saXN0KSB9LAotICAgICAgICB7IGNvbmZpZ181X2F0dHJpYnV0ZV9saXN0LCBORUxFTShjb25maWdfNV9hdHRyaWJ1dGVfbGlzdCkgfSwKLX07Ci0KLXN0YXRpYyBjb25maWdfbWFuYWdlbWVudF90IGNvbnN0IGdDb25maWdNYW5hZ2VtZW50W10gPSB7Ci0gICAgICAgIHsgRUdMX0JVRkZFUl9TSVpFLCAgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjphdExlYXN0IH0sCi0gICAgICAgIHsgRUdMX0FMUEhBX1NJWkUsICAgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjphdExlYXN0IH0sCi0gICAgICAgIHsgRUdMX0JMVUVfU0laRSwgICAgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjphdExlYXN0IH0sCi0gICAgICAgIHsgRUdMX0dSRUVOX1NJWkUsICAgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjphdExlYXN0IH0sCi0gICAgICAgIHsgRUdMX1JFRF9TSVpFLCAgICAgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjphdExlYXN0IH0sCi0gICAgICAgIHsgRUdMX0RFUFRIX1NJWkUsICAgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjphdExlYXN0IH0sCi0gICAgICAgIHsgRUdMX1NURU5DSUxfU0laRSwgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjphdExlYXN0IH0sCi0gICAgICAgIHsgRUdMX0NPTkZJR19DQVZFQVQsICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX0NPTkZJR19JRCwgICAgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX0xFVkVMLCAgICAgICAgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX01BWF9QQlVGRkVSX0hFSUdIVCwgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX01BWF9QQlVGRkVSX1BJWEVMUywgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX01BWF9QQlVGRkVSX1dJRFRILCAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX05BVElWRV9SRU5ERVJBQkxFLCAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX05BVElWRV9WSVNVQUxfSUQsICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX05BVElWRV9WSVNVQUxfVFlQRSwgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX1NBTVBMRVMsICAgICAgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX1NBTVBMRV9CVUZGRVJTLCAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX1NVUkZBQ0VfVFlQRSwgICAgICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjptYXNrICAgIH0sCi0gICAgICAgIHsgRUdMX1RSQU5TUEFSRU5UX1RZUEUsICAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX1RSQU5TUEFSRU5UX0JMVUVfVkFMVUUsICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX1RSQU5TUEFSRU5UX0dSRUVOX1ZBTFVFLCAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX1RSQU5TUEFSRU5UX1JFRF9WQUxVRSwgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX0JJTkRfVE9fVEVYVFVSRV9SR0JBLCAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX0JJTkRfVE9fVEVYVFVSRV9SR0IsICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX01JTl9TV0FQX0lOVEVSVkFMLCAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi0gICAgICAgIHsgRUdMX01BWF9TV0FQX0lOVEVSVkFMLCAgICAgICAgICBjb25maWdfbWFuYWdlbWVudF90OjpleGFjdCAgIH0sCi19OwotCi1zdGF0aWMgY29uZmlnX3BhaXJfdCBjb25zdCBjb25maWdfZGVmYXVsdHNbXSA9IHsKLSAgICAgICAgeyBFR0xfU1VSRkFDRV9UWVBFLCAgICAgICAgRUdMX1dJTkRPV19CSVQgfSwKLX07Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdGVtcGxhdGU8dHlwZW5hbWUgVD4KLXN0YXRpYyBpbnQgYmluYXJ5U2VhcmNoKFQgY29uc3Qgc29ydGVkQXJyYXlbXSwgaW50IGZpcnN0LCBpbnQgbGFzdCwgRUdMaW50IGtleSkKLXsKLSAgIHdoaWxlIChmaXJzdCA8PSBsYXN0KSB7Ci0gICAgICAgaW50IG1pZCA9IChmaXJzdCArIGxhc3QpIC8gMjsKLSAgICAgICBpZiAoa2V5ID4gc29ydGVkQXJyYXlbbWlkXS5rZXkpIHsgCi0gICAgICAgICAgIGZpcnN0ID0gbWlkICsgMTsKLSAgICAgICB9IGVsc2UgaWYgKGtleSA8IHNvcnRlZEFycmF5W21pZF0ua2V5KSB7IAotICAgICAgICAgICBsYXN0ID0gbWlkIC0gMTsKLSAgICAgICB9IGVsc2UgewotICAgICAgICAgICByZXR1cm4gbWlkOwotICAgICAgIH0KLSAgIH0KLSAgIHJldHVybiAtMTsKLX0KLQotc3RhdGljIGludCBpc0F0dHJpYnV0ZU1hdGNoaW5nKGludCBpLCBFR0xpbnQgYXR0ciwgRUdMaW50IHZhbCkKLXsKLSAgICAvLyBsb29rIGZvciB0aGUgYXR0cmlidXRlIGluIGFsbCBvZiBvdXIgY29uZmlncwotICAgIGNvbmZpZ19wYWlyX3QgY29uc3QqIGNvbmZpZ0ZvdW5kID0gZ0NvbmZpZ3NbaV0uYXJyYXk7IAotICAgIGludCBpbmRleCA9IGJpbmFyeVNlYXJjaDxjb25maWdfcGFpcl90PigKLSAgICAgICAgICAgIGdDb25maWdzW2ldLmFycmF5LAotICAgICAgICAgICAgMCwgZ0NvbmZpZ3NbaV0uc2l6ZS0xLAotICAgICAgICAgICAgYXR0cik7Ci0gICAgaWYgKGluZGV4IDwgMCkgewotICAgICAgICBjb25maWdGb3VuZCA9IGNvbmZpZ19iYXNlX2F0dHJpYnV0ZV9saXN0OyAKLSAgICAgICAgaW5kZXggPSBiaW5hcnlTZWFyY2g8Y29uZmlnX3BhaXJfdD4oCi0gICAgICAgICAgICAgICAgY29uZmlnX2Jhc2VfYXR0cmlidXRlX2xpc3QsCi0gICAgICAgICAgICAgICAgMCwgTkVMRU0oY29uZmlnX2Jhc2VfYXR0cmlidXRlX2xpc3QpLTEsCi0gICAgICAgICAgICAgICAgYXR0cik7Ci0gICAgfQotICAgIGlmIChpbmRleCA+PSAwKSB7Ci0gICAgICAgIC8vIGF0dHJpYnV0ZSBmb3VuZCwgY2hlY2sgaWYgdGhpcyBjb25maWcgY291bGQgbWF0Y2gKLSAgICAgICAgaW50IGNmZ01ndEluZGV4ID0gYmluYXJ5U2VhcmNoPGNvbmZpZ19tYW5hZ2VtZW50X3Q+KAotICAgICAgICAgICAgICAgIGdDb25maWdNYW5hZ2VtZW50LAotICAgICAgICAgICAgICAgIDAsIE5FTEVNKGdDb25maWdNYW5hZ2VtZW50KS0xLAotICAgICAgICAgICAgICAgIGF0dHIpOwotICAgICAgICBpZiAoaW5kZXggPj0gMCkgewotICAgICAgICAgICAgYm9vbCBtYXRjaCA9IGdDb25maWdNYW5hZ2VtZW50W2NmZ01ndEluZGV4XS5tYXRjaCgKLSAgICAgICAgICAgICAgICAgICAgdmFsLCBjb25maWdGb3VuZFtpbmRleF0udmFsdWUpOwotICAgICAgICAgICAgaWYgKG1hdGNoKSB7Ci0gICAgICAgICAgICAgICAgLy8gdGhpcyBjb25maWcgbWF0Y2hlcwotICAgICAgICAgICAgICAgIHJldHVybiAxOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLy8gYXR0cmlidXRlIG5vdCBmb3VuZC4gdGhpcyBzaG91bGQgTkVWRVIgaGFwcGVuLgotICAgICAgICB9Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgLy8gZXJyb3IsIHRoaXMgYXR0cmlidXRlIGRvZXNuJ3QgZXhpc3QKLSAgICB9Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLXN0YXRpYyBpbnQgbWFrZUN1cnJlbnQob2dsZXNfY29udGV4dF90KiBnbCkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGN1cnJlbnQgPSAob2dsZXNfY29udGV4dF90KilnZXRHbFRocmVhZFNwZWNpZmljKCk7Ci0gICAgaWYgKGdsKSB7Ci0gICAgICAgIGVnbF9jb250ZXh0X3QqIGMgPSBlZ2xfY29udGV4dF90Ojpjb250ZXh0KGdsKTsKLSAgICAgICAgaWYgKGMtPmZsYWdzICYgZWdsX2NvbnRleHRfdDo6SVNfQ1VSUkVOVCkgewotICAgICAgICAgICAgaWYgKGN1cnJlbnQgIT0gZ2wpIHsKLSAgICAgICAgICAgICAgICAvLyBpdCBpcyBhbiBlcnJvciB0byBzZXQgYSBjb250ZXh0IGN1cnJlbnQsIGlmIGl0J3MgYWxyZWFkeQotICAgICAgICAgICAgICAgIC8vIGN1cnJlbnQgdG8gYW5vdGhlciB0aHJlYWQKLSAgICAgICAgICAgICAgICByZXR1cm4gLTE7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBpZiAoY3VycmVudCkgewotICAgICAgICAgICAgICAgIC8vIG1hcmsgdGhlIGN1cnJlbnQgY29udGV4dCBhcyBub3QgY3VycmVudCwgYW5kIGZsdXNoCi0gICAgICAgICAgICAgICAgZ2xGbHVzaCgpOwotICAgICAgICAgICAgICAgIGVnbF9jb250ZXh0X3Q6OmNvbnRleHQoY3VycmVudCktPmZsYWdzICY9IH5lZ2xfY29udGV4dF90OjpJU19DVVJSRU5UOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGlmICghKGMtPmZsYWdzICYgZWdsX2NvbnRleHRfdDo6SVNfQ1VSUkVOVCkpIHsKLSAgICAgICAgICAgIC8vIFRoZSBjb250ZXh0IGlzIG5vdCBjdXJyZW50LCBtYWtlIGl0IGN1cnJlbnQhCi0gICAgICAgICAgICBzZXRHbFRocmVhZFNwZWNpZmljKGdsKTsKLSAgICAgICAgICAgIGMtPmZsYWdzIHw9IGVnbF9jb250ZXh0X3Q6OklTX0NVUlJFTlQ7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICBpZiAoY3VycmVudCkgewotICAgICAgICAgICAgLy8gbWFyayB0aGUgY3VycmVudCBjb250ZXh0IGFzIG5vdCBjdXJyZW50LCBhbmQgZmx1c2gKLSAgICAgICAgICAgIGdsRmx1c2goKTsKLSAgICAgICAgICAgIGVnbF9jb250ZXh0X3Q6OmNvbnRleHQoY3VycmVudCktPmZsYWdzICY9IH5lZ2xfY29udGV4dF90OjpJU19DVVJSRU5UOwotICAgICAgICB9Ci0gICAgICAgIC8vIHRoaXMgdGhyZWFkIGhhcyBubyBjb250ZXh0IGF0dGFjaGVkIHRvIGl0Ci0gICAgICAgIHNldEdsVGhyZWFkU3BlY2lmaWMoMCk7Ci0gICAgfQotICAgIHJldHVybiAwOwotfQotCi1zdGF0aWMgRUdMQm9vbGVhbiBnZXRDb25maWdBdHRyaWIoRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgIEVHTGludCBhdHRyaWJ1dGUsIEVHTGludCAqdmFsdWUpCi17Ci0gICAgc2l6ZV90IG51bUNvbmZpZ3MgPSAgTkVMRU0oZ0NvbmZpZ3MpOwotICAgIGludCBpbmRleCA9IChpbnQpY29uZmlnOwotICAgIGlmICh1aW50MzJfdChpbmRleCkgPj0gbnVtQ29uZmlncykKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfQ09ORklHLCBFR0xfRkFMU0UpOwotCi0gICAgaW50IGF0dHJJbmRleDsKLSAgICBhdHRySW5kZXggPSBiaW5hcnlTZWFyY2g8Y29uZmlnX3BhaXJfdD4oCi0gICAgICAgICAgICBnQ29uZmlnc1tpbmRleF0uYXJyYXksCi0gICAgICAgICAgICAwLCBnQ29uZmlnc1tpbmRleF0uc2l6ZS0xLAotICAgICAgICAgICAgYXR0cmlidXRlKTsKLSAgICBpZiAoYXR0ckluZGV4Pj0wKSB7Ci0gICAgICAgICp2YWx1ZSA9IGdDb25maWdzW2luZGV4XS5hcnJheVthdHRySW5kZXhdLnZhbHVlOwotICAgICAgICByZXR1cm4gRUdMX1RSVUU7Ci0gICAgfQotCi0gICAgYXR0ckluZGV4ID0gYmluYXJ5U2VhcmNoPGNvbmZpZ19wYWlyX3Q+KAotICAgICAgICAgICAgY29uZmlnX2Jhc2VfYXR0cmlidXRlX2xpc3QsCi0gICAgICAgICAgICAwLCBORUxFTShjb25maWdfYmFzZV9hdHRyaWJ1dGVfbGlzdCktMSwKLSAgICAgICAgICAgIGF0dHJpYnV0ZSk7Ci0gICAgaWYgKGF0dHJJbmRleD49MCkgewotICAgICAgICAqdmFsdWUgPSBjb25maWdfYmFzZV9hdHRyaWJ1dGVfbGlzdFthdHRySW5kZXhdLnZhbHVlOwotICAgICAgICByZXR1cm4gRUdMX1RSVUU7Ci0gICAgfQotICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0FUVFJJQlVURSwgRUdMX0ZBTFNFKTsKLX0KLQotc3RhdGljIEVHTFN1cmZhY2UgY3JlYXRlV2luZG93U3VyZmFjZShFR0xEaXNwbGF5IGRweSwgRUdMQ29uZmlnIGNvbmZpZywKLSAgICAgICAgTmF0aXZlV2luZG93VHlwZSB3aW5kb3csIGNvbnN0IEVHTGludCAqYXR0cmliX2xpc3QpCi17Ci0gICAgaWYgKGVnbF9kaXNwbGF5X3Q6OmlzX3ZhbGlkKGRweSkgPT0gRUdMX0ZBTFNFKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9ESVNQTEFZLCBFR0xfTk9fU1VSRkFDRSk7Ci0gICAgaWYgKHdpbmRvdyA9PSAwKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9NQVRDSCwgRUdMX05PX1NVUkZBQ0UpOwotCi0gICAgRUdMaW50IHN1cmZhY2VUeXBlOwotICAgIGlmIChnZXRDb25maWdBdHRyaWIoZHB5LCBjb25maWcsIEVHTF9TVVJGQUNFX1RZUEUsICZzdXJmYWNlVHlwZSkgPT0gRUdMX0ZBTFNFKQotICAgICAgICByZXR1cm4gRUdMX0ZBTFNFOwotCi0gICAgaWYgKCEoc3VyZmFjZVR5cGUgJiBFR0xfV0lORE9XX0JJVCkpCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX01BVENILCBFR0xfTk9fU1VSRkFDRSk7Ci0KLSAgICBFR0xpbnQgY29uZmlnSUQ7Ci0gICAgaWYgKGdldENvbmZpZ0F0dHJpYihkcHksIGNvbmZpZywgRUdMX0NPTkZJR19JRCwgJmNvbmZpZ0lEKSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBFR0xfRkFMU0U7Ci0KLSAgICBpbnQzMl90IGRlcHRoRm9ybWF0OwotICAgIGludDMyX3QgcGl4ZWxGb3JtYXQ7Ci0gICAgc3dpdGNoKGNvbmZpZ0lEKSB7Ci0gICAgY2FzZSAwOiAKLSAgICAgICAgcGl4ZWxGb3JtYXQgPSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjU7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IDA7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgMToKLSAgICAgICAgcGl4ZWxGb3JtYXQgPSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjU7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfWl8xNjsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSAyOgotICAgICAgICBwaXhlbEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfUkdCQV84ODg4OyAKLSAgICAgICAgZGVwdGhGb3JtYXQgPSAwOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIDM6Ci0gICAgICAgIHBpeGVsRm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzg4ODg7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfWl8xNjsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSA0OgotICAgICAgICBwaXhlbEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfQV84OyAKLSAgICAgICAgZGVwdGhGb3JtYXQgPSAwOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIDU6Ci0gICAgICAgIHBpeGVsRm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9BXzg7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfWl8xNjsKLSAgICAgICAgYnJlYWs7Ci0gICAgZGVmYXVsdDoKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfTUFUQ0gsIEVHTF9OT19TVVJGQUNFKTsKLSAgICB9Ci0KLSAgICAvLyBGSVhNRTogd2UgZG9uJ3QgaGF2ZSBhY2Nlc3MgdG8gdGhlIHBpeGVsRm9ybWF0IGhlcmUganVzdCB5ZXQuCi0gICAgLy8gKGl0J3MgcG9zc2libGUgdGhhdCB0aGUgc3VyZmFjZSBpcyBub3QgZnVsbHkgaW5pdGlhbGl6ZWQpCi0gICAgLy8gbWF5YmUgdGhpcyBzaG91bGQgYmUgZG9uZSBhZnRlciB0aGUgcGFnZS1mbGlwCi0gICAgLy9pZiAoRUdMaW50KGluZm8uZm9ybWF0KSAhPSBwaXhlbEZvcm1hdCkKLSAgICAvLyAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9NQVRDSCwgRUdMX05PX1NVUkZBQ0UpOwotCi0gICAgZWdsX3N1cmZhY2VfdCogc3VyZmFjZSA9Ci0gICAgICAgIG5ldyBlZ2xfd2luZG93X3N1cmZhY2VfdChkcHksIGNvbmZpZywgZGVwdGhGb3JtYXQsCi0gICAgICAgICAgICAgICAgc3RhdGljX2Nhc3Q8ZWdsX25hdGl2ZV93aW5kb3dfdCo+KHdpbmRvdykpOwotCi0gICAgaWYgKCFzdXJmYWNlLT5pc1ZhbGlkKCkpIHsKLSAgICAgICAgLy8gdGhlcmUgd2FzIGEgcHJvYmxlbSBpbiB0aGUgY3RvciwgdGhlIGVycm9yCi0gICAgICAgIC8vIGZsYWcgaGFzIGJlZW4gc2V0LgotICAgICAgICBkZWxldGUgc3VyZmFjZTsKLSAgICAgICAgc3VyZmFjZSA9IDA7Ci0gICAgfQotICAgIHJldHVybiBzdXJmYWNlOwotfQotCi1zdGF0aWMgRUdMU3VyZmFjZSBjcmVhdGVQaXhtYXBTdXJmYWNlKEVHTERpc3BsYXkgZHB5LCBFR0xDb25maWcgY29uZmlnLAotICAgICAgICBOYXRpdmVQaXhtYXBUeXBlIHBpeG1hcCwgY29uc3QgRUdMaW50ICphdHRyaWJfbGlzdCkKLXsKLSAgICBpZiAoZWdsX2Rpc3BsYXlfdDo6aXNfdmFsaWQoZHB5KSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9OT19TVVJGQUNFKTsKLSAgICBpZiAocGl4bWFwID09IDApCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX01BVENILCBFR0xfTk9fU1VSRkFDRSk7Ci0KLSAgICBFR0xpbnQgc3VyZmFjZVR5cGU7Ci0gICAgaWYgKGdldENvbmZpZ0F0dHJpYihkcHksIGNvbmZpZywgRUdMX1NVUkZBQ0VfVFlQRSwgJnN1cmZhY2VUeXBlKSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBFR0xfRkFMU0U7Ci0KLSAgICBpZiAoIShzdXJmYWNlVHlwZSAmIEVHTF9QSVhNQVBfQklUKSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfTUFUQ0gsIEVHTF9OT19TVVJGQUNFKTsKLQotICAgIEVHTGludCBjb25maWdJRDsKLSAgICBpZiAoZ2V0Q29uZmlnQXR0cmliKGRweSwgY29uZmlnLCBFR0xfQ09ORklHX0lELCAmY29uZmlnSUQpID09IEVHTF9GQUxTRSkKLSAgICAgICAgcmV0dXJuIEVHTF9GQUxTRTsKLQotICAgIGludDMyX3QgZGVwdGhGb3JtYXQ7Ci0gICAgaW50MzJfdCBwaXhlbEZvcm1hdDsKLSAgICBzd2l0Y2goY29uZmlnSUQpIHsKLSAgICBjYXNlIDA6IAotICAgICAgICBwaXhlbEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfUkdCXzU2NTsgCi0gICAgICAgIGRlcHRoRm9ybWF0ID0gMDsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSAxOgotICAgICAgICBwaXhlbEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfUkdCXzU2NTsgCi0gICAgICAgIGRlcHRoRm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9aXzE2OwotICAgICAgICBicmVhazsKLSAgICBjYXNlIDI6Ci0gICAgICAgIHBpeGVsRm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzg4ODg7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IDA7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgMzoKLSAgICAgICAgcGl4ZWxGb3JtYXQgPSBHR0xfUElYRUxfRk9STUFUX1JHQkFfODg4ODsgCi0gICAgICAgIGRlcHRoRm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9aXzE2OwotICAgICAgICBicmVhazsKLSAgICBjYXNlIDQ6Ci0gICAgICAgIHBpeGVsRm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9BXzg7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IDA7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgNToKLSAgICAgICAgcGl4ZWxGb3JtYXQgPSBHR0xfUElYRUxfRk9STUFUX0FfODsgCi0gICAgICAgIGRlcHRoRm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9aXzE2OwotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9NQVRDSCwgRUdMX05PX1NVUkZBQ0UpOwotICAgIH0KLQotICAgIGlmIChwaXhtYXAtPmZvcm1hdCAhPSBwaXhlbEZvcm1hdCkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfTUFUQ0gsIEVHTF9OT19TVVJGQUNFKTsKLQotICAgIGVnbF9zdXJmYWNlX3QqIHN1cmZhY2UgPQotICAgICAgICBuZXcgZWdsX3BpeG1hcF9zdXJmYWNlX3QoZHB5LCBjb25maWcsIGRlcHRoRm9ybWF0LAotICAgICAgICAgICAgICAgIHN0YXRpY19jYXN0PGVnbF9uYXRpdmVfcGl4bWFwX3QqPihwaXhtYXApKTsKLQotICAgIGlmICghc3VyZmFjZS0+aXNWYWxpZCgpKSB7Ci0gICAgICAgIC8vIHRoZXJlIHdhcyBhIHByb2JsZW0gaW4gdGhlIGN0b3IsIHRoZSBlcnJvcgotICAgICAgICAvLyBmbGFnIGhhcyBiZWVuIHNldC4KLSAgICAgICAgZGVsZXRlIHN1cmZhY2U7Ci0gICAgICAgIHN1cmZhY2UgPSAwOwotICAgIH0KLSAgICByZXR1cm4gc3VyZmFjZTsKLX0KLQotc3RhdGljIEVHTFN1cmZhY2UgY3JlYXRlUGJ1ZmZlclN1cmZhY2UoRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgIGNvbnN0IEVHTGludCAqYXR0cmliX2xpc3QpCi17Ci0gICAgaWYgKGVnbF9kaXNwbGF5X3Q6OmlzX3ZhbGlkKGRweSkgPT0gRUdMX0ZBTFNFKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9ESVNQTEFZLCBFR0xfTk9fU1VSRkFDRSk7Ci0KLSAgICBFR0xpbnQgc3VyZmFjZVR5cGU7Ci0gICAgaWYgKGdldENvbmZpZ0F0dHJpYihkcHksIGNvbmZpZywgRUdMX1NVUkZBQ0VfVFlQRSwgJnN1cmZhY2VUeXBlKSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBFR0xfRkFMU0U7Ci0gICAgCi0gICAgaWYgKCEoc3VyZmFjZVR5cGUgJiBFR0xfUEJVRkZFUl9CSVQpKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9NQVRDSCwgRUdMX05PX1NVUkZBQ0UpOwotICAgICAgICAKLSAgICBFR0xpbnQgY29uZmlnSUQ7Ci0gICAgaWYgKGdldENvbmZpZ0F0dHJpYihkcHksIGNvbmZpZywgRUdMX0NPTkZJR19JRCwgJmNvbmZpZ0lEKSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBFR0xfRkFMU0U7Ci0KLSAgICBpbnQzMl90IGRlcHRoRm9ybWF0OwotICAgIGludDMyX3QgcGl4ZWxGb3JtYXQ7Ci0gICAgc3dpdGNoKGNvbmZpZ0lEKSB7Ci0gICAgY2FzZSAwOiAKLSAgICAgICAgcGl4ZWxGb3JtYXQgPSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjU7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IDA7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgMToKLSAgICAgICAgcGl4ZWxGb3JtYXQgPSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjU7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfWl8xNjsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSAyOgotICAgICAgICBwaXhlbEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfUkdCQV84ODg4OyAKLSAgICAgICAgZGVwdGhGb3JtYXQgPSAwOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIDM6Ci0gICAgICAgIHBpeGVsRm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzg4ODg7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfWl8xNjsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSA0OgotICAgICAgICBwaXhlbEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfQV84OyAKLSAgICAgICAgZGVwdGhGb3JtYXQgPSAwOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIDU6Ci0gICAgICAgIHBpeGVsRm9ybWF0ID0gR0dMX1BJWEVMX0ZPUk1BVF9BXzg7IAotICAgICAgICBkZXB0aEZvcm1hdCA9IEdHTF9QSVhFTF9GT1JNQVRfWl8xNjsKLSAgICAgICAgYnJlYWs7Ci0gICAgZGVmYXVsdDoKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfTUFUQ0gsIEVHTF9OT19TVVJGQUNFKTsKLSAgICB9Ci0KLSAgICBpbnQzMl90IHcgPSAwOwotICAgIGludDMyX3QgaCA9IDA7Ci0gICAgd2hpbGUgKGF0dHJpYl9saXN0WzBdKSB7Ci0gICAgICAgIGlmIChhdHRyaWJfbGlzdFswXSA9PSBFR0xfV0lEVEgpICB3ID0gYXR0cmliX2xpc3RbMV07Ci0gICAgICAgIGlmIChhdHRyaWJfbGlzdFswXSA9PSBFR0xfSEVJR0hUKSBoID0gYXR0cmliX2xpc3RbMV07Ci0gICAgICAgIGF0dHJpYl9saXN0Kz0yOwotICAgIH0KLQotICAgIGVnbF9zdXJmYWNlX3QqIHN1cmZhY2UgPQotICAgICAgICBuZXcgZWdsX3BidWZmZXJfc3VyZmFjZV90KGRweSwgY29uZmlnLCBkZXB0aEZvcm1hdCwgdywgaCwgcGl4ZWxGb3JtYXQpOwotCi0gICAgaWYgKCFzdXJmYWNlLT5pc1ZhbGlkKCkpIHsKLSAgICAgICAgLy8gdGhlcmUgd2FzIGEgcHJvYmxlbSBpbiB0aGUgY3RvciwgdGhlIGVycm9yCi0gICAgICAgIC8vIGZsYWcgaGFzIGJlZW4gc2V0LgotICAgICAgICBkZWxldGUgc3VyZmFjZTsKLSAgICAgICAgc3VyZmFjZSA9IDA7Ci0gICAgfQotICAgIHJldHVybiBzdXJmYWNlOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gSW5pdGlhbGl6YXRpb24KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotRUdMRGlzcGxheSBlZ2xHZXREaXNwbGF5KE5hdGl2ZURpc3BsYXlUeXBlIGRpc3BsYXkpCi17Ci0jaWZuZGVmIEhBVkVfQU5EUk9JRF9PUwotICAgIC8vIHRoaXMganVzdCBuZWVkcyB0byBiZSBkb25lIG9uY2UKLSAgICBpZiAoZ0dMS2V5ID09IC0xKSB7Ci0gICAgICAgIHB0aHJlYWRfbXV0ZXhfbG9jaygmZ0luaXRNdXRleCk7Ci0gICAgICAgIGlmIChnR0xLZXkgPT0gLTEpCi0gICAgICAgICAgICBwdGhyZWFkX2tleV9jcmVhdGUoJmdHTEtleSwgTlVMTCk7Ci0gICAgICAgIHB0aHJlYWRfbXV0ZXhfdW5sb2NrKCZnSW5pdE11dGV4KTsKLSAgICB9Ci0jZW5kaWYKLSAgICBpZiAoZGlzcGxheSA9PSBFR0xfREVGQVVMVF9ESVNQTEFZKSB7Ci0gICAgICAgIEVHTERpc3BsYXkgZHB5ID0gKEVHTERpc3BsYXkpMTsKLSAgICAgICAgZWdsX2Rpc3BsYXlfdCYgZCA9IGVnbF9kaXNwbGF5X3Q6OmdldF9kaXNwbGF5KGRweSk7Ci0gICAgICAgIGQudHlwZSA9IGRpc3BsYXk7Ci0gICAgICAgIHJldHVybiBkcHk7Ci0gICAgfSAgICAKLSAgICByZXR1cm4gRUdMX05PX0RJU1BMQVk7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsSW5pdGlhbGl6ZShFR0xEaXNwbGF5IGRweSwgRUdMaW50ICptYWpvciwgRUdMaW50ICptaW5vcikKLXsKLSAgICBpZiAoZWdsX2Rpc3BsYXlfdDo6aXNfdmFsaWQoZHB5KSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9GQUxTRSk7Ci0gICAgCi0gICAgRUdMQm9vbGVhbiByZXMgPSBFR0xfVFJVRTsKLSAgICBlZ2xfZGlzcGxheV90JiBkID0gZWdsX2Rpc3BsYXlfdDo6Z2V0X2Rpc3BsYXkoZHB5KTsKLSAgICAKLSAgICBpZiAoYW5kcm9pZF9hdG9taWNfaW5jKCZkLmluaXRpYWxpemVkKSA9PSAwKSB7Ci0gICAgICAgIC8vIGluaXRpYWxpemUgc3R1ZmYgaGVyZSBpZiBuZWVkZWQKLSAgICAgICAgLy9wdGhyZWFkX211dGV4X2xvY2soJmdJbml0TXV0ZXgpOwotICAgICAgICAvL3B0aHJlYWRfbXV0ZXhfdW5sb2NrKCZnSW5pdE11dGV4KTsKLSAgICB9Ci0KLSAgICBpZiAocmVzID09IEVHTF9UUlVFKSB7Ci0gICAgICAgIGlmIChtYWpvciAhPSBOVUxMKSAqbWFqb3IgPSAxOwotICAgICAgICBpZiAobWlub3IgIT0gTlVMTCkgKm1pbm9yID0gMjsKLSAgICB9Ci0gICAgcmV0dXJuIHJlczsKLX0KLQotRUdMQm9vbGVhbiBlZ2xUZXJtaW5hdGUoRUdMRGlzcGxheSBkcHkpCi17Ci0gICAgaWYgKGVnbF9kaXNwbGF5X3Q6OmlzX3ZhbGlkKGRweSkgPT0gRUdMX0ZBTFNFKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9ESVNQTEFZLCBFR0xfRkFMU0UpOwotCi0gICAgRUdMQm9vbGVhbiByZXMgPSBFR0xfVFJVRTsKLSAgICBlZ2xfZGlzcGxheV90JiBkID0gZWdsX2Rpc3BsYXlfdDo6Z2V0X2Rpc3BsYXkoZHB5KTsKLSAgICBpZiAoYW5kcm9pZF9hdG9taWNfZGVjKCZkLmluaXRpYWxpemVkKSA9PSAxKSB7Ci0gICAgICAgIC8vIFRPRE86IGRlc3Ryb3kgYWxsIHJlc291cmNlcyAoc3VyZmFjZXMsIGNvbnRleHRzLCBldGMuLi4pCi0gICAgICAgIC8vcHRocmVhZF9tdXRleF9sb2NrKCZnSW5pdE11dGV4KTsKLSAgICAgICAgLy9wdGhyZWFkX211dGV4X3VubG9jaygmZ0luaXRNdXRleCk7Ci0gICAgfQotICAgIHJldHVybiByZXM7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIGNvbmZpZ3VyYXRpb24KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotRUdMQm9vbGVhbiBlZ2xHZXRDb25maWdzKCAgIEVHTERpc3BsYXkgZHB5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEVHTENvbmZpZyAqY29uZmlncywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBFR0xpbnQgY29uZmlnX3NpemUsIEVHTGludCAqbnVtX2NvbmZpZykKLXsKLSAgICBpZiAoZWdsX2Rpc3BsYXlfdDo6aXNfdmFsaWQoZHB5KSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9GQUxTRSk7Ci0KLSAgICBHTGludCBudW1Db25maWdzID0gTkVMRU0oZ0NvbmZpZ3MpOwotICAgIGlmICghY29uZmlncykgewotICAgICAgICAqbnVtX2NvbmZpZyA9IG51bUNvbmZpZ3M7Ci0gICAgICAgIHJldHVybiBFR0xfVFJVRTsKLSAgICB9Ci0gICAgR0xpbnQgaTsKLSAgICBmb3IgKGk9MCA7IGk8bnVtQ29uZmlncyAmJiBpPGNvbmZpZ19zaXplIDsgaSsrKSB7Ci0gICAgICAgICpjb25maWdzKysgPSAoRUdMQ29uZmlnKWk7Ci0gICAgfQotICAgICpudW1fY29uZmlnID0gaTsKLSAgICByZXR1cm4gRUdMX1RSVUU7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsQ2hvb3NlQ29uZmlnKCBFR0xEaXNwbGF5IGRweSwgY29uc3QgRUdMaW50ICphdHRyaWJfbGlzdCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBFR0xDb25maWcgKmNvbmZpZ3MsIEVHTGludCBjb25maWdfc2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBFR0xpbnQgKm51bV9jb25maWcpCi17Ci0gICAgaWYgKGVnbF9kaXNwbGF5X3Q6OmlzX3ZhbGlkKGRweSkgPT0gRUdMX0ZBTFNFKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9ESVNQTEFZLCBFR0xfRkFMU0UpOwotCi0gICAgaWYgKGdnbF91bmxpa2VseShjb25maWdzPT0wIHx8IGF0dHJpYl9saXN0PT0wKSkgewotICAgICAgICAqbnVtX2NvbmZpZyA9IDA7Ci0gICAgICAgIHJldHVybiBFR0xfVFJVRTsKLSAgICB9Ci0gICAgCi0gICAgaW50IG51bUF0dHJpYnV0ZXMgPSAwOwotICAgIGludCBudW1Db25maWdzID0gIE5FTEVNKGdDb25maWdzKTsKLSAgICB1aW50MzJfdCBwb3NzaWJsZU1hdGNoID0gKDE8PG51bUNvbmZpZ3MpLTE7Ci0gICAgd2hpbGUocG9zc2libGVNYXRjaCAmJiAqYXR0cmliX2xpc3QgIT0gRUdMX05PTkUpIHsKLSAgICAgICAgbnVtQXR0cmlidXRlcysrOwotICAgICAgICBFR0xpbnQgYXR0ciA9ICphdHRyaWJfbGlzdCsrOwotICAgICAgICBFR0xpbnQgdmFsICA9ICphdHRyaWJfbGlzdCsrOwotICAgICAgICBmb3IgKGludCBpPTAgOyBpPG51bUNvbmZpZ3MgOyBpKyspIHsKLSAgICAgICAgICAgIGlmICghKHBvc3NpYmxlTWF0Y2ggJiAoMTw8aSkpKQotICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgaWYgKGlzQXR0cmlidXRlTWF0Y2hpbmcoaSwgYXR0ciwgdmFsKSA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgcG9zc2libGVNYXRjaCAmPSB+KDE8PGkpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgLy8gbm93LCBoYW5kbGUgdGhlIGF0dHJpYnV0ZXMgd2hpY2ggaGF2ZSBhIHVzZWZ1bCBkZWZhdWx0IHZhbHVlCi0gICAgZm9yIChzaXplX3Qgaj0wIDsgajxORUxFTShjb25maWdfZGVmYXVsdHMpIDsgaisrKSB7Ci0gICAgICAgIC8vIHNlZSBpZiB0aGlzIGF0dHJpYnV0ZSB3YXMgc3BlY2lmaWVkLCBpZiBub3QgYXBwbHkgaXRzCi0gICAgICAgIC8vIGRlZmF1bHQgdmFsdWUKLSAgICAgICAgaWYgKGJpbmFyeVNlYXJjaDxjb25maWdfcGFpcl90PigKLSAgICAgICAgICAgICAgICAoY29uZmlnX3BhaXJfdCBjb25zdCopYXR0cmliX2xpc3QsCi0gICAgICAgICAgICAgICAgMCwgbnVtQXR0cmlidXRlcywKLSAgICAgICAgICAgICAgICBjb25maWdfZGVmYXVsdHNbal0ua2V5KSA8IDApCi0gICAgICAgIHsKLSAgICAgICAgICAgIGZvciAoaW50IGk9MCA7IGk8bnVtQ29uZmlncyA7IGkrKykgewotICAgICAgICAgICAgICAgIGlmICghKHBvc3NpYmxlTWF0Y2ggJiAoMTw8aSkpKQotICAgICAgICAgICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgICAgICAgICBpZiAoaXNBdHRyaWJ1dGVNYXRjaGluZyhpLAotICAgICAgICAgICAgICAgICAgICAgICAgY29uZmlnX2RlZmF1bHRzW2pdLmtleSwKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbmZpZ19kZWZhdWx0c1tqXS52YWx1ZSkgPT0gMCkKLSAgICAgICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgICAgIHBvc3NpYmxlTWF0Y2ggJj0gfigxPDxpKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICAvLyByZXR1cm4gdGhlIGNvbmZpZ3VyYXRpb25zIGZvdW5kCi0gICAgaW50IG49MDsKLSAgICBpZiAocG9zc2libGVNYXRjaCkgewotICAgICAgICBmb3IgKGludCBpPTAgOyBjb25maWdfc2l6ZSAmJiBpPG51bUNvbmZpZ3MgOyBpKyspIHsKLSAgICAgICAgICAgIGlmIChwb3NzaWJsZU1hdGNoICYgKDE8PGkpKSB7Ci0gICAgICAgICAgICAgICAqY29uZmlncysrID0gKEVHTENvbmZpZylpOwotICAgICAgICAgICAgICAgIGNvbmZpZ19zaXplLS07Ci0gICAgICAgICAgICAgICAgbisrOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgICpudW1fY29uZmlnID0gbjsKLSAgICAgcmV0dXJuIEVHTF9UUlVFOwotfQotCi1FR0xCb29sZWFuIGVnbEdldENvbmZpZ0F0dHJpYihFR0xEaXNwbGF5IGRweSwgRUdMQ29uZmlnIGNvbmZpZywKLSAgICAgICAgRUdMaW50IGF0dHJpYnV0ZSwgRUdMaW50ICp2YWx1ZSkKLXsKLSAgICBpZiAoZWdsX2Rpc3BsYXlfdDo6aXNfdmFsaWQoZHB5KSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9GQUxTRSk7Ci0KLSAgICByZXR1cm4gZ2V0Q29uZmlnQXR0cmliKGRweSwgY29uZmlnLCBhdHRyaWJ1dGUsIHZhbHVlKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gc3VyZmFjZXMKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotRUdMU3VyZmFjZSBlZ2xDcmVhdGVXaW5kb3dTdXJmYWNlKCAgRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBOYXRpdmVXaW5kb3dUeXBlIHdpbmRvdywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IEVHTGludCAqYXR0cmliX2xpc3QpCi17Ci0gICAgcmV0dXJuIGNyZWF0ZVdpbmRvd1N1cmZhY2UoZHB5LCBjb25maWcsIHdpbmRvdywgYXR0cmliX2xpc3QpOwotfQotICAgIAotRUdMU3VyZmFjZSBlZ2xDcmVhdGVQaXhtYXBTdXJmYWNlKCAgRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBOYXRpdmVQaXhtYXBUeXBlIHBpeG1hcCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IEVHTGludCAqYXR0cmliX2xpc3QpCi17Ci0gICAgcmV0dXJuIGNyZWF0ZVBpeG1hcFN1cmZhY2UoZHB5LCBjb25maWcsIHBpeG1hcCwgYXR0cmliX2xpc3QpOwotfQotCi1FR0xTdXJmYWNlIGVnbENyZWF0ZVBidWZmZXJTdXJmYWNlKCBFR0xEaXNwbGF5IGRweSwgRUdMQ29uZmlnIGNvbmZpZywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IEVHTGludCAqYXR0cmliX2xpc3QpCi17Ci0gICAgcmV0dXJuIGNyZWF0ZVBidWZmZXJTdXJmYWNlKGRweSwgY29uZmlnLCBhdHRyaWJfbGlzdCk7Ci19Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKLUVHTEJvb2xlYW4gZWdsRGVzdHJveVN1cmZhY2UoRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2UgZWdsU3VyZmFjZSkKLXsKLSAgICBpZiAoZWdsX2Rpc3BsYXlfdDo6aXNfdmFsaWQoZHB5KSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9GQUxTRSk7Ci0gICAgaWYgKGVnbFN1cmZhY2UgIT0gRUdMX05PX1NVUkZBQ0UpIHsKLSAgICAgICAgZWdsX3N1cmZhY2VfdCogc3VyZmFjZSggc3RhdGljX2Nhc3Q8ZWdsX3N1cmZhY2VfdCo+KGVnbFN1cmZhY2UpICk7Ci0gICAgICAgIGlmIChzdXJmYWNlLT5tYWdpYyAhPSBlZ2xfc3VyZmFjZV90OjpNQUdJQykKLSAgICAgICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX1NVUkZBQ0UsIEVHTF9GQUxTRSk7Ci0gICAgICAgIGlmIChzdXJmYWNlLT5kcHkgIT0gZHB5KQotICAgICAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICAgICAgZGVsZXRlIHN1cmZhY2U7Ci0gICAgfQotICAgIHJldHVybiBFR0xfVFJVRTsKLX0KLQotRUdMQm9vbGVhbiBlZ2xRdWVyeVN1cmZhY2UoIEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIGVnbFN1cmZhY2UsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgRUdMaW50IGF0dHJpYnV0ZSwgRUdMaW50ICp2YWx1ZSkKLXsKLSAgICBpZiAoZWdsX2Rpc3BsYXlfdDo6aXNfdmFsaWQoZHB5KSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9GQUxTRSk7Ci0gICAgZWdsX3N1cmZhY2VfdCogc3VyZmFjZSA9IHN0YXRpY19jYXN0PGVnbF9zdXJmYWNlX3QqPihlZ2xTdXJmYWNlKTsKLSAgICBpZiAoc3VyZmFjZS0+ZHB5ICE9IGRweSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLQotICAgIEVHTEJvb2xlYW4gcmV0ID0gRUdMX1RSVUU7Ci0gICAgc3dpdGNoIChhdHRyaWJ1dGUpIHsKLSAgICAgICAgY2FzZSBFR0xfQ09ORklHX0lEOgotICAgICAgICAgICAgcmV0ID0gZ2V0Q29uZmlnQXR0cmliKGRweSwgc3VyZmFjZS0+Y29uZmlnLCBFR0xfQ09ORklHX0lELCB2YWx1ZSk7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBFR0xfV0lEVEg6Ci0gICAgICAgICAgICAqdmFsdWUgPSBzdXJmYWNlLT5nZXRXaWR0aCgpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgRUdMX0hFSUdIVDoKLSAgICAgICAgICAgICp2YWx1ZSA9IHN1cmZhY2UtPmdldEhlaWdodCgpOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgRUdMX0xBUkdFU1RfUEJVRkZFUjoKLSAgICAgICAgICAgIC8vIG5vdCBtb2RpZmllZCBmb3IgYSB3aW5kb3cgb3IgcGl4bWFwIHN1cmZhY2UKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEVHTF9URVhUVVJFX0ZPUk1BVDoKLSAgICAgICAgICAgICp2YWx1ZSA9IEVHTF9OT19URVhUVVJFOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgRUdMX1RFWFRVUkVfVEFSR0VUOgotICAgICAgICAgICAgKnZhbHVlID0gRUdMX05PX1RFWFRVUkU7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBFR0xfTUlQTUFQX1RFWFRVUkU6Ci0gICAgICAgICAgICAqdmFsdWUgPSBFR0xfRkFMU0U7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBFR0xfTUlQTUFQX0xFVkVMOgotICAgICAgICAgICAgKnZhbHVlID0gMDsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEVHTF9SRU5ERVJfQlVGRkVSOgotICAgICAgICAgICAgLy8gVE9ETzogcmV0dXJuIHRoZSByZWFsIFJFTkRFUl9CVUZGRVIgaGVyZQotICAgICAgICAgICAgKnZhbHVlID0gRUdMX0JBQ0tfQlVGRkVSOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIGNhc2UgRUdMX0hPUklaT05UQUxfUkVTT0xVVElPTjoKLSAgICAgICAgICAgIC8vIHBpeGVsL21tICogRUdMX0RJU1BMQVlfU0NBTElORwotICAgICAgICAgICAgKnZhbHVlID0gc3VyZmFjZS0+Z2V0SG9yaXpvbnRhbFJlc29sdXRpb24oKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEVHTF9WRVJUSUNBTF9SRVNPTFVUSU9OOgotICAgICAgICAgICAgLy8gcGl4ZWwvbW0gKiBFR0xfRElTUExBWV9TQ0FMSU5HCi0gICAgICAgICAgICAqdmFsdWUgPSBzdXJmYWNlLT5nZXRWZXJ0aWNhbFJlc29sdXRpb24oKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBjYXNlIEVHTF9QSVhFTF9BU1BFQ1RfUkFUSU86IHsKLSAgICAgICAgICAgIC8vIHcvaCAqIEVHTF9ESVNQTEFZX1NDQUxJTkcKLSAgICAgICAgICAgIGludCB3ciA9IHN1cmZhY2UtPmdldEhvcml6b250YWxSZXNvbHV0aW9uKCk7Ci0gICAgICAgICAgICBpbnQgaHIgPSBzdXJmYWNlLT5nZXRWZXJ0aWNhbFJlc29sdXRpb24oKTsKLSAgICAgICAgICAgICp2YWx1ZSA9ICh3ciAqIEVHTF9ESVNQTEFZX1NDQUxJTkcpIC8gaHI7Ci0gICAgICAgIH0gYnJlYWs7Ci0gICAgICAgIGNhc2UgRUdMX1NXQVBfQkVIQVZJT1I6Ci0gICAgICAgICAgICAqdmFsdWUgPSBzdXJmYWNlLT5nZXRTd2FwQmVoYXZpb3IoKTsgCi0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgZGVmYXVsdDoKLSAgICAgICAgICAgIHJldCA9IHNldEVycm9yKEVHTF9CQURfQVRUUklCVVRFLCBFR0xfRkFMU0UpOwotICAgIH0KLSAgICByZXR1cm4gcmV0OwotfQotCi1FR0xDb250ZXh0IGVnbENyZWF0ZUNvbnRleHQoRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgRUdMQ29udGV4dCBzaGFyZV9saXN0LCBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KQotewotICAgIGlmIChlZ2xfZGlzcGxheV90Ojppc192YWxpZChkcHkpID09IEVHTF9GQUxTRSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX05PX1NVUkZBQ0UpOwotCi0gICAgb2dsZXNfY29udGV4dF90KiBnbCA9IG9nbGVzX2luaXQoc2l6ZW9mKGVnbF9jb250ZXh0X3QpKTsKLSAgICBpZiAoIWdsKSByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9BTExPQywgRUdMX05PX0NPTlRFWFQpOwotCi0gICAgZWdsX2NvbnRleHRfdCogYyA9IHN0YXRpY19jYXN0PGVnbF9jb250ZXh0X3QqPihnbC0+cmFzdGVyaXplci5iYXNlKTsKLSAgICBjLT5mbGFncyA9IGVnbF9jb250ZXh0X3Q6Ok5FVkVSX0NVUlJFTlQ7Ci0gICAgYy0+ZHB5ID0gZHB5OwotICAgIGMtPmNvbmZpZyA9IGNvbmZpZzsKLSAgICBjLT5yZWFkID0gMDsKLSAgICBjLT5kcmF3ID0gMDsKLSAgICByZXR1cm4gKEVHTENvbnRleHQpZ2w7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsRGVzdHJveUNvbnRleHQoRUdMRGlzcGxheSBkcHksIEVHTENvbnRleHQgY3R4KQotewotICAgIGlmIChlZ2xfZGlzcGxheV90Ojppc192YWxpZChkcHkpID09IEVHTF9GQUxTRSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICBlZ2xfY29udGV4dF90KiBjID0gZWdsX2NvbnRleHRfdDo6Y29udGV4dChjdHgpOwotICAgIGlmIChjLT5mbGFncyAmIGVnbF9jb250ZXh0X3Q6OklTX0NVUlJFTlQpCi0gICAgICAgIHNldEdsVGhyZWFkU3BlY2lmaWMoMCk7Ci0gICAgb2dsZXNfdW5pbml0KChvZ2xlc19jb250ZXh0X3QqKWN0eCk7Ci0gICAgcmV0dXJuIEVHTF9UUlVFOwotfQotCi1FR0xCb29sZWFuIGVnbE1ha2VDdXJyZW50KCAgRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2UgZHJhdywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBFR0xTdXJmYWNlIHJlYWQsIEVHTENvbnRleHQgY3R4KQotewotICAgIGlmIChlZ2xfZGlzcGxheV90Ojppc192YWxpZChkcHkpID09IEVHTF9GQUxTRSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICBpZiAoZHJhdykgewotICAgICAgICBlZ2xfc3VyZmFjZV90KiBzID0gKGVnbF9zdXJmYWNlX3QqKWRyYXc7Ci0gICAgICAgIGlmIChzLT5kcHkgIT0gZHB5KQotICAgICAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICAgICAgLy8gVE9ETzogY2hlY2sgdGhhdCBkcmF3IGFuZCByZWFkIGFyZSBjb21wYXRpYmxlIHdpdGggdGhlIGNvbnRleHQKLSAgICB9Ci0KLSAgICBFR0xDb250ZXh0IGN1cnJlbnRfY3R4ID0gRUdMX05PX0NPTlRFWFQ7Ci0gICAgCi0gICAgaWYgKChyZWFkID09IEVHTF9OT19TVVJGQUNFICYmIGRyYXcgPT0gRUdMX05PX1NVUkZBQ0UpICYmIChjdHggIT0gRUdMX05PX0NPTlRFWFQpKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9NQVRDSCwgRUdMX0ZBTFNFKTsKLQotICAgIGlmICgocmVhZCAhPSBFR0xfTk9fU1VSRkFDRSB8fCBkcmF3ICE9IEVHTF9OT19TVVJGQUNFKSAmJiAoY3R4ID09IEVHTF9OT19DT05URVhUKSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfTUFUQ0gsIEVHTF9GQUxTRSk7Ci0KLSAgICBpZiAoY3R4ID09IEVHTF9OT19DT05URVhUKSB7Ci0gICAgICAgIC8vIGlmIHdlJ3JlIGRldGFjaGluZywgd2UgbmVlZCB0aGUgY3VycmVudCBjb250ZXh0Ci0gICAgICAgIGN1cnJlbnRfY3R4ID0gKEVHTENvbnRleHQpZ2V0R2xUaHJlYWRTcGVjaWZpYygpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIGVnbF9jb250ZXh0X3QqIGMgPSBlZ2xfY29udGV4dF90Ojpjb250ZXh0KGN0eCk7Ci0gICAgICAgIGVnbF9zdXJmYWNlX3QqIGQgPSAoZWdsX3N1cmZhY2VfdCopZHJhdzsKLSAgICAgICAgZWdsX3N1cmZhY2VfdCogciA9IChlZ2xfc3VyZmFjZV90KilyZWFkOwotICAgICAgICBpZiAoKGQgJiYgZC0+Y3R4ICYmIGQtPmN0eCAhPSBjdHgpIHx8Ci0gICAgICAgICAgICAociAmJiByLT5jdHggJiYgci0+Y3R4ICE9IGN0eCkpIHsKLSAgICAgICAgICAgIC8vIG9uY2Ugb2YgdGhlIHN1cmZhY2UgaXMgYm91bmQgdG8gYSBjb250ZXh0IGluIGFub3RoZXIgdGhyZWFkCi0gICAgICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9BQ0NFU1MsIEVHTF9GQUxTRSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBvZ2xlc19jb250ZXh0X3QqIGdsID0gKG9nbGVzX2NvbnRleHRfdCopY3R4OwotICAgIGlmIChtYWtlQ3VycmVudChnbCkgPT0gMCkgewotICAgICAgICBpZiAoY3R4KSB7Ci0gICAgICAgICAgICBlZ2xfY29udGV4dF90KiBjID0gZWdsX2NvbnRleHRfdDo6Y29udGV4dChjdHgpOwotICAgICAgICAgICAgZWdsX3N1cmZhY2VfdCogZCA9IChlZ2xfc3VyZmFjZV90KilkcmF3OwotICAgICAgICAgICAgZWdsX3N1cmZhY2VfdCogciA9IChlZ2xfc3VyZmFjZV90KilyZWFkOwotICAgICAgICAgICAgYy0+cmVhZCA9IHJlYWQ7Ci0gICAgICAgICAgICBjLT5kcmF3ID0gZHJhdzsKLSAgICAgICAgICAgIGlmIChjLT5mbGFncyAmIGVnbF9jb250ZXh0X3Q6Ok5FVkVSX0NVUlJFTlQpIHsKLSAgICAgICAgICAgICAgICBjLT5mbGFncyAmPSB+ZWdsX2NvbnRleHRfdDo6TkVWRVJfQ1VSUkVOVDsKLSAgICAgICAgICAgICAgICBHTGludCB3ID0gMDsKLSAgICAgICAgICAgICAgICBHTGludCBoID0gMDsKLSAgICAgICAgICAgICAgICBpZiAoZHJhdykgewotICAgICAgICAgICAgICAgICAgICB3ID0gZC0+Z2V0V2lkdGgoKTsKLSAgICAgICAgICAgICAgICAgICAgaCA9IGQtPmdldEhlaWdodCgpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICBvZ2xlc19zdXJmYWNlcG9ydChnbCwgMCwgMCk7Ci0gICAgICAgICAgICAgICAgb2dsZXNfdmlld3BvcnQoZ2wsIDAsIDAsIHcsIGgpOwotICAgICAgICAgICAgICAgIG9nbGVzX3NjaXNzb3IoZ2wsIDAsIDAsIHcsIGgpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgaWYgKGQpIHsKLSAgICAgICAgICAgICAgICBkLT5jdHggPSBjdHg7Ci0gICAgICAgICAgICAgICAgZC0+YmluZERyYXdTdXJmYWNlKGdsKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChyKSB7Ci0gICAgICAgICAgICAgICAgci0+Y3R4ID0gY3R4OwotICAgICAgICAgICAgICAgIHItPmJpbmRSZWFkU3VyZmFjZShnbCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAvLyBpZiBzdXJmYWNlcyB3ZXJlIGJvdW5kIHRvIHRoZSBjb250ZXh0IGJvdW5kIHRvIHRoaXMgdGhyZWFkCi0gICAgICAgICAgICAvLyBtYXJrIHRoZW4gYXMgdW5ib3VuZC4KLSAgICAgICAgICAgIGlmIChjdXJyZW50X2N0eCkgewotICAgICAgICAgICAgICAgIGVnbF9jb250ZXh0X3QqIGMgPSBlZ2xfY29udGV4dF90Ojpjb250ZXh0KGN1cnJlbnRfY3R4KTsKLSAgICAgICAgICAgICAgICBlZ2xfc3VyZmFjZV90KiBkID0gKGVnbF9zdXJmYWNlX3QqKWMtPmRyYXc7Ci0gICAgICAgICAgICAgICAgZWdsX3N1cmZhY2VfdCogciA9IChlZ2xfc3VyZmFjZV90KiljLT5yZWFkOwotICAgICAgICAgICAgICAgIGlmIChkKSBkLT5jdHggPSBFR0xfTk9fQ09OVEVYVDsKLSAgICAgICAgICAgICAgICBpZiAocikgci0+Y3R4ID0gRUdMX05PX0NPTlRFWFQ7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIEVHTF9UUlVFOwotICAgIH0KLSAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9BQ0NFU1MsIEVHTF9GQUxTRSk7Ci19Ci0KLUVHTENvbnRleHQgZWdsR2V0Q3VycmVudENvbnRleHQodm9pZCkKLXsKLSAgICAvLyBlZ2xHZXRDdXJyZW50Q29udGV4dCByZXR1cm5zIHRoZSBjdXJyZW50IEVHTCByZW5kZXJpbmcgY29udGV4dCwKLSAgICAvLyBhcyBzcGVjaWZpZWQgYnkgZWdsTWFrZUN1cnJlbnQuIElmIG5vIGNvbnRleHQgaXMgY3VycmVudCwKLSAgICAvLyBFR0xfTk9fQ09OVEVYVCBpcyByZXR1cm5lZC4KLSAgICByZXR1cm4gKEVHTENvbnRleHQpZ2V0R2xUaHJlYWRTcGVjaWZpYygpOwotfQotCi1FR0xTdXJmYWNlIGVnbEdldEN1cnJlbnRTdXJmYWNlKEVHTGludCByZWFkZHJhdykKLXsKLSAgICAvLyBlZ2xHZXRDdXJyZW50U3VyZmFjZSByZXR1cm5zIHRoZSByZWFkIG9yIGRyYXcgc3VyZmFjZSBhdHRhY2hlZAotICAgIC8vIHRvIHRoZSBjdXJyZW50IEVHTCByZW5kZXJpbmcgY29udGV4dCwgYXMgc3BlY2lmaWVkIGJ5IGVnbE1ha2VDdXJyZW50LgotICAgIC8vIElmIG5vIGNvbnRleHQgaXMgY3VycmVudCwgRUdMX05PX1NVUkZBQ0UgaXMgcmV0dXJuZWQuCi0gICAgRUdMQ29udGV4dCBjdHggPSAoRUdMQ29udGV4dClnZXRHbFRocmVhZFNwZWNpZmljKCk7Ci0gICAgaWYgKGN0eCA9PSBFR0xfTk9fQ09OVEVYVCkgcmV0dXJuIEVHTF9OT19TVVJGQUNFOwotICAgIGVnbF9jb250ZXh0X3QqIGMgPSBlZ2xfY29udGV4dF90Ojpjb250ZXh0KGN0eCk7Ci0gICAgaWYgKHJlYWRkcmF3ID09IEVHTF9SRUFEKSB7Ci0gICAgICAgIHJldHVybiBjLT5yZWFkOwotICAgIH0gZWxzZSBpZiAocmVhZGRyYXcgPT0gRUdMX0RSQVcpIHsKLSAgICAgICAgcmV0dXJuIGMtPmRyYXc7Ci0gICAgfQotICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0FUVFJJQlVURSwgRUdMX05PX1NVUkZBQ0UpOwotfQotCi1FR0xEaXNwbGF5IGVnbEdldEN1cnJlbnREaXNwbGF5KHZvaWQpCi17Ci0gICAgLy8gZWdsR2V0Q3VycmVudERpc3BsYXkgcmV0dXJucyB0aGUgY3VycmVudCBFR0wgZGlzcGxheSBjb25uZWN0aW9uCi0gICAgLy8gZm9yIHRoZSBjdXJyZW50IEVHTCByZW5kZXJpbmcgY29udGV4dCwgYXMgc3BlY2lmaWVkIGJ5IGVnbE1ha2VDdXJyZW50LgotICAgIC8vIElmIG5vIGNvbnRleHQgaXMgY3VycmVudCwgRUdMX05PX0RJU1BMQVkgaXMgcmV0dXJuZWQuCi0gICAgRUdMQ29udGV4dCBjdHggPSAoRUdMQ29udGV4dClnZXRHbFRocmVhZFNwZWNpZmljKCk7Ci0gICAgaWYgKGN0eCA9PSBFR0xfTk9fQ09OVEVYVCkgcmV0dXJuIEVHTF9OT19ESVNQTEFZOwotICAgIGVnbF9jb250ZXh0X3QqIGMgPSBlZ2xfY29udGV4dF90Ojpjb250ZXh0KGN0eCk7Ci0gICAgcmV0dXJuIGMtPmRweTsKLX0KLQotRUdMQm9vbGVhbiBlZ2xRdWVyeUNvbnRleHQoIEVHTERpc3BsYXkgZHB5LCBFR0xDb250ZXh0IGN0eCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBFR0xpbnQgYXR0cmlidXRlLCBFR0xpbnQgKnZhbHVlKQotewotICAgIGlmIChlZ2xfZGlzcGxheV90Ojppc192YWxpZChkcHkpID09IEVHTF9GQUxTRSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICBlZ2xfY29udGV4dF90KiBjID0gZWdsX2NvbnRleHRfdDo6Y29udGV4dChjdHgpOwotICAgIHN3aXRjaCAoYXR0cmlidXRlKSB7Ci0gICAgICAgIGNhc2UgRUdMX0NPTkZJR19JRDoKLSAgICAgICAgICAgIC8vIFJldHVybnMgdGhlIElEIG9mIHRoZSBFR0wgZnJhbWUgYnVmZmVyIGNvbmZpZ3VyYXRpb24gd2l0aAotICAgICAgICAgICAgLy8gcmVzcGVjdCB0byB3aGljaCB0aGUgY29udGV4dCB3YXMgY3JlYXRlZAotICAgICAgICAgICAgcmV0dXJuIGdldENvbmZpZ0F0dHJpYihkcHksIGMtPmNvbmZpZywgRUdMX0NPTkZJR19JRCwgdmFsdWUpOwotICAgIH0KLSAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9BVFRSSUJVVEUsIEVHTF9GQUxTRSk7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsV2FpdEdMKHZvaWQpCi17Ci0gICAgcmV0dXJuIEVHTF9UUlVFOwotfQotCi1FR0xCb29sZWFuIGVnbFdhaXROYXRpdmUoRUdMaW50IGVuZ2luZSkKLXsKLSAgICByZXR1cm4gRUdMX1RSVUU7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsU3dhcEJ1ZmZlcnMoRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2UgZHJhdykKLXsKLSAgICBpZiAoZWdsX2Rpc3BsYXlfdDo6aXNfdmFsaWQoZHB5KSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9GQUxTRSk7Ci0gICAgCi0gICAgZWdsX3N1cmZhY2VfdCogZCA9IHN0YXRpY19jYXN0PGVnbF9zdXJmYWNlX3QqPihkcmF3KTsKLSAgICBpZiAoZC0+ZHB5ICE9IGRweSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLQotICAgIC8vIHBvc3QgdGhlIHN1cmZhY2UKLSAgICBkLT5zd2FwQnVmZmVycygpOwotCi0gICAgLy8gaWYgaXQncyBib3VuZCB0byBhIGNvbnRleHQsIHVwZGF0ZSB0aGUgYnVmZmVyCi0gICAgaWYgKGQtPmN0eCAhPSBFR0xfTk9fQ09OVEVYVCkgewotICAgICAgICBkLT5iaW5kRHJhd1N1cmZhY2UoKG9nbGVzX2NvbnRleHRfdCopZC0+Y3R4KTsKLSAgICAgICAgLy8gaWYgdGhpcyBzdXJmYWNlIGlzIGFsc28gdGhlIHJlYWQgc3VyZmFjZSBvZiB0aGUgY29udGV4dAotICAgICAgICAvLyBpdCBpcyBib3VuZCB0bywgbWFrZSBzdXJlIHRvIHVwZGF0ZSB0aGUgcmVhZCBidWZmZXIgYXMgd2VsbC4KLSAgICAgICAgLy8gVGhlIEVHTCBzcGVjIGlzIGEgbGl0dGxlIHVuY2xlYXIgYWJvdXQgdGhpcy4KLSAgICAgICAgZWdsX2NvbnRleHRfdCogYyA9IGVnbF9jb250ZXh0X3Q6OmNvbnRleHQoZC0+Y3R4KTsKLSAgICAgICAgaWYgKGMtPnJlYWQgPT0gZHJhdykgewotICAgICAgICAgICAgZC0+YmluZFJlYWRTdXJmYWNlKChvZ2xlc19jb250ZXh0X3QqKWQtPmN0eCk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICByZXR1cm4gRUdMX1RSVUU7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsQ29weUJ1ZmZlcnMoICBFR0xEaXNwbGF5IGRweSwgRUdMU3VyZmFjZSBzdXJmYWNlLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIE5hdGl2ZVBpeG1hcFR5cGUgdGFyZ2V0KQotewotICAgIGlmIChlZ2xfZGlzcGxheV90Ojppc192YWxpZChkcHkpID09IEVHTF9GQUxTRSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICAvLyBUT0RPOiBlZ2xDb3B5QnVmZmVycygpCi0gICAgcmV0dXJuIEVHTF9GQUxTRTsKLX0KLQotRUdMaW50IGVnbEdldEVycm9yKHZvaWQpCi17Ci0gICAgcmV0dXJuIGdldEVycm9yKCk7Ci19Ci0KLWNvbnN0IGNoYXIqIGVnbFF1ZXJ5U3RyaW5nKEVHTERpc3BsYXkgZHB5LCBFR0xpbnQgbmFtZSkKLXsKLSAgICBpZiAoZWdsX2Rpc3BsYXlfdDo6aXNfdmFsaWQoZHB5KSA9PSBFR0xfRkFMU0UpCi0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIChjb25zdCBjaGFyKikwKTsKLQotICAgIHN3aXRjaCAobmFtZSkgewotICAgICAgICBjYXNlIEVHTF9WRU5ET1I6Ci0gICAgICAgICAgICByZXR1cm4gZ1ZlbmRvclN0cmluZzsKLSAgICAgICAgY2FzZSBFR0xfVkVSU0lPTjoKLSAgICAgICAgICAgIHJldHVybiBnVmVyc2lvblN0cmluZzsKLSAgICAgICAgY2FzZSBFR0xfRVhURU5TSU9OUzoKLSAgICAgICAgICAgIHJldHVybiBnRXh0ZW5zaW9uc1N0cmluZzsKLSAgICAgICAgY2FzZSBFR0xfQ0xJRU5UX0FQSVM6Ci0gICAgICAgICAgICByZXR1cm4gZ0NsaWVudEFwaVN0cmluZzsKLSAgICB9Ci0gICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfUEFSQU1FVEVSLCAoY29uc3QgY2hhciAqKTApOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBFR0wgMS4xCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUVHTEJvb2xlYW4gZWdsU3VyZmFjZUF0dHJpYigKLSAgICAgICAgRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2Ugc3VyZmFjZSwgRUdMaW50IGF0dHJpYnV0ZSwgRUdMaW50IHZhbHVlKQotewotICAgIGlmIChlZ2xfZGlzcGxheV90Ojppc192YWxpZChkcHkpID09IEVHTF9GQUxTRSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICAvLyBUT0RPOiBlZ2xTdXJmYWNlQXR0cmliKCkKLSAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9QQVJBTUVURVIsIEVHTF9GQUxTRSk7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsQmluZFRleEltYWdlKAotICAgICAgICBFR0xEaXNwbGF5IGRweSwgRUdMU3VyZmFjZSBzdXJmYWNlLCBFR0xpbnQgYnVmZmVyKQotewotICAgIGlmIChlZ2xfZGlzcGxheV90Ojppc192YWxpZChkcHkpID09IEVHTF9GQUxTRSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICAvLyBUT0RPOiBlZ2xCaW5kVGV4SW1hZ2UoKQotICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX1BBUkFNRVRFUiwgRUdMX0ZBTFNFKTsKLX0KLQotRUdMQm9vbGVhbiBlZ2xSZWxlYXNlVGV4SW1hZ2UoCi0gICAgICAgIEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIHN1cmZhY2UsIEVHTGludCBidWZmZXIpCi17Ci0gICAgaWYgKGVnbF9kaXNwbGF5X3Q6OmlzX3ZhbGlkKGRweSkgPT0gRUdMX0ZBTFNFKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9ESVNQTEFZLCBFR0xfRkFMU0UpOwotICAgIC8vIFRPRE86IGVnbFJlbGVhc2VUZXhJbWFnZSgpCi0gICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfUEFSQU1FVEVSLCBFR0xfRkFMU0UpOwotfQotCi1FR0xCb29sZWFuIGVnbFN3YXBJbnRlcnZhbChFR0xEaXNwbGF5IGRweSwgRUdMaW50IGludGVydmFsKQotewotICAgIGlmIChlZ2xfZGlzcGxheV90Ojppc192YWxpZChkcHkpID09IEVHTF9GQUxTRSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICAvLyBUT0RPOiBlZ2xTd2FwSW50ZXJ2YWwoKQotICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX1BBUkFNRVRFUiwgRUdMX0ZBTFNFKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gRUdMIDEuMgotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1FR0xCb29sZWFuIGVnbEJpbmRBUEkoRUdMZW51bSBhcGkpCi17Ci0gICAgaWYgKGFwaSAhPSBFR0xfT1BFTkdMX0VTX0FQSSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfUEFSQU1FVEVSLCBFR0xfRkFMU0UpOwotICAgIHJldHVybiBFR0xfVFJVRTsKLX0KLQotRUdMZW51bSBlZ2xRdWVyeUFQSSh2b2lkKQotewotICAgIHJldHVybiBFR0xfT1BFTkdMX0VTX0FQSTsKLX0KLQotRUdMQm9vbGVhbiBlZ2xXYWl0Q2xpZW50KHZvaWQpCi17Ci0gICAgZ2xGaW5pc2goKTsKLSAgICByZXR1cm4gRUdMX1RSVUU7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsUmVsZWFzZVRocmVhZCh2b2lkKQotewotICAgIC8vIFRPRE86IGVnbFJlbGVhc2VUaHJlYWQoKQotICAgIHJldHVybiBFR0xfVFJVRTsKLX0KLQotRUdMU3VyZmFjZSBlZ2xDcmVhdGVQYnVmZmVyRnJvbUNsaWVudEJ1ZmZlcigKLSAgICAgICAgICBFR0xEaXNwbGF5IGRweSwgRUdMZW51bSBidWZ0eXBlLCBFR0xDbGllbnRCdWZmZXIgYnVmZmVyLAotICAgICAgICAgIEVHTENvbmZpZyBjb25maWcsIGNvbnN0IEVHTGludCAqYXR0cmliX2xpc3QpCi17Ci0gICAgaWYgKGVnbF9kaXNwbGF5X3Q6OmlzX3ZhbGlkKGRweSkgPT0gRUdMX0ZBTFNFKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9ESVNQTEFZLCBFR0xfTk9fU1VSRkFDRSk7Ci0gICAgLy8gVE9ETzogZWdsQ3JlYXRlUGJ1ZmZlckZyb21DbGllbnRCdWZmZXIoKQotICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX1BBUkFNRVRFUiwgRUdMX05PX1NVUkZBQ0UpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBBbmRyb2lkIGV4dGVuc2lvbnMKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdm9pZCAoKmVnbEdldFByb2NBZGRyZXNzIChjb25zdCBjaGFyICpwcm9jbmFtZSkpKCkKLXsKLSAgICBleHRlbnRpb25fbWFwX3QgY29uc3QgKiBjb25zdCBtYXAgPSBnRXh0ZW50aW9uTWFwOwotICAgIGZvciAodWludDMyX3QgaT0wIDsgaTxORUxFTShnRXh0ZW50aW9uTWFwKSA7IGkrKykgewotICAgICAgICBpZiAoIXN0cmNtcChwcm9jbmFtZSwgbWFwW2ldLm5hbWUpKSB7Ci0gICAgICAgICAgICByZXR1cm4gbWFwW2ldLmFkZHJlc3M7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIE5VTEw7Ci19CmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL2ZpeGVkX2FzbS5TIGIvb3BlbmdsL2xpYmFnbC9maXhlZF9hc20uUwpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNmNiYzU2Zi4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGliYWdsL2ZpeGVkX2FzbS5TCisrKyAvZGV2L251bGwKQEAgLTEsNjUgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9maXhlZF9hc20uUwotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLQotICAgIC50ZXh0Ci0gICAgLmFsaWduCi0gICAgCi0gICAgLmdsb2JhbCBnZ2xGbG9hdFRvRml4ZWQKLSAgICAuZ2xvYmFsIGdnbEZsb2F0VG9GaXhlZEZhc3QKLQotCi0vKgotICogQ29udmVydHMgYSBmbG9hdCB0byBhIHMxNS4xNiBmaXhlZC1wb2ludCBudW1iZXIuCi0gKiB0aGlzIGRvZXNuJ3QgaGFuZGxlIGZsb2F0cyBvdXQgb2YgdGhlIFstMzI3NjgsICszMjc2OFsgcmFuZ2UKLSAqIGFuZCBkb2Vzbid0IHBlcmZvcm1zIHJvdW5kLXRvLW5lYXJlc3QuCi0gKiBob3dldmVyLCBpdCdzIHZlcnkgZmFzdCA6LSkKLSAqLwotCi1nZ2xGbG9hdFRvRml4ZWRGYXN0OgotICAgICAgICBtb3ZzICAgIHIxLCByMCwgbHNsICMxICAgICAgICAgIC8qIHJlbW92ZSBiaXQgc2lnbiAqLwotICAgICAgICBtb3YgICAgIHIyLCAjMHg4RSAgICAgICAgICAgICAgIC8qIDEyNyArIDE1ICovCi0gICAgICAgIHN1YiAgICAgcjEsIHIyLCByMSwgbHNyICMyNCAgICAgLyogY29tcHV0ZSBzaGlmdCAqLwotICAgICAgICBtb3YgICAgIHIyLCByMCwgbHNsICM4ICAgICAgICAgIC8qIG1hbnRpc3NhPDw4ICovCi0gICAgICAgIG9yciAgICAgcjIsIHIyLCAjMHg4MDAwMDAwMCAgICAgLyogYWRkIHRoZSBtaXNzaW5nIDEgKi8KLSAgICAgICAgbW92ICAgICByMCwgcjIsIGxzciByMSAgICAgICAgICAvKiBzY2FsZSB0byAxNi4xNiAqLwotICAgICAgICByc2JjcyAgIHIwLCByMCwgIzAgICAgICAgICAgICAgIC8qIG5lZ2F0ZSBpZiBuZWVkZWQgKi8KLSAgICAgICAgYnggICAgICBscgotCi0vKgotICogdGhpcyB2ZXJzaW9uIHJvdW5kcy10by1uZWFyZXN0IGFuZCBzYXR1cmF0ZXMgbnVtYmVycwotICogb3V0c2lkZSB0aGUgcmFuZ2UgKGJ1dCBub3QgTmFOcykuCi0gKi8KLQotZ2dsRmxvYXRUb0ZpeGVkOgotICAgICAgICBtb3YgICAgIHIxLCByMCwgbHNsICMxICAgICAgICAgIC8qIHJlbW92ZSBiaXQgc2lnbiAqLwotICAgICAgICBtb3YgICAgIHIyLCAjMHg4RSAgICAgICAgICAgICAgIC8qIDEyNyArIDE1ICovCi0gICAgICAgIHN1YnMgICAgcjEsIHIyLCByMSwgbHNyICMyNCAgICAgLyogY29tcHV0ZSBzaGlmdCAqLwotICAgICAgICBibHMgICAgIDBmICAgICAgICAgICAgICAgICAgICAgIC8qIHRvbyBiaWcgKi8KLSAgICAgICAgbW92ICAgICByMiwgcjAsIGxzbCAjOCAgICAgICAgICAvKiBtYW50aXNzYTw8OCAqLwotICAgICAgICBvcnIgICAgIHIyLCByMiwgIzB4ODAwMDAwMDAgICAgIC8qIGFkZCB0aGUgbWlzc2luZyAxICovCi0gICAgICAgIG1vdiAgICAgcjMsIHIwCi0gICAgICAgIG1vdnMgICAgcjAsIHIyLCBsc3IgcjEgICAgICAgICAgLyogc2NhbGUgdG8gMTYuMTYgKi8KLSAgICAgICAgYWRkY3MgICByMCwgcjAsICMxICAgICAgICAgICAgICAvKiByb3VuZC10by1uZWFyZXN0ICovCi0gICAgICAgIHRzdCAgICAgcjMsICMweDgwMDAwMDAwICAgICAgICAgLyogbmVnYXRpdmU/ICovCi0gICAgICAgIHJzYm5lICAgcjAsIHIwLCAjMCAgICAgICAgICAgICAgLyogbmVnYXRlIGlmIG5lZWRlZCAqLwotICAgICAgICBieCAgICAgIGxyIAotIAotMDogICAgICBhbmRzICAgIHIwLCByMCwgIzB4ODAwMDAwMDAgICAgIC8qIGtlZXAgb25seSB0aGUgc2lnbiBiaXQgKi8KLSAgICAgICAgbW92ZXEgICByMCwgIzB4N2ZmZmZmZmYgICAgICAgICAvKiBwb3NpdGl2ZSwgbWF4aW11bSB2YWx1ZSAqLwotICAgICAgICBieCAgICAgIGxyCi0KZGlmZiAtLWdpdCBhL29wZW5nbC9saWJhZ2wvZnAuY3BwIGIvb3BlbmdsL2xpYmFnbC9mcC5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGFlNWYxZmUuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9mcC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw4NyArMCwwIEBACi0vKiBsaWJzL29wZW5nbGVzL2ZwLmNwcAotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpbmNsdWRlICJmcC5oIgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNpZiAhZGVmaW5lZChfX2FybV9fKQotR0dMZml4ZWQgZ2dsRmxvYXRUb0ZpeGVkKGZsb2F0IHYpIHsgICAKLSAgICByZXR1cm4gR0dMZml4ZWQoZmxvb3JmKHYgKiA2NTUzNi4wZiArIDAuNWYpKTsKLX0KLSNlbmRpZgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotbmFtZXNwYWNlIGdsIHsKLQotR0xmbG9hdCBmaXhlZFRvRmxvYXQoR0xmaXhlZCB4KQotewotI2lmIERFQlVHX1VTRV9GTE9BVFMKLSAgICByZXR1cm4geCAvIDY1NTM2LjBmOwotI2Vsc2UKLSAgICBpZiAoIXgpIHJldHVybiAwOwotICAgIGNvbnN0IHVpbnQzMl90IHMgPSB4ICYgMHg4MDAwMDAwMDsKLSAgICB1bmlvbiB7Ci0gICAgICAgIHVpbnQzMl90IGk7Ci0gICAgICAgIGZsb2F0IGY7Ci0gICAgfTsKLSAgICBpID0gcyA/IC14IDogeDsKLSAgICBjb25zdCBpbnQgYyA9IGdnbENseihpKSAtIDg7Ci0gICAgaSA9IChjPj0wKSA/IChpPDxjKSA6IChpPj4tYyk7Ci0gICAgY29uc3QgdWludDMyX3QgZSA9IDEzNCAtIGM7Ci0gICAgaSAmPSB+MHg4MDAwMDA7Ci0gICAgaSB8PSBlPDwyMzsKLSAgICBpIHw9IHM7Ci0gICAgcmV0dXJuIGY7Ci0jZW5kaWYKLX0KLQotZmxvYXQgc2luZWYoZmxvYXQgeCkKLXsKLSAgICBjb25zdCBmbG9hdCBBID0gICAxLjBmIC8gKDIuMGYqTV9QSSk7Ci0gICAgY29uc3QgZmxvYXQgQiA9IC0xNi4wZjsKLSAgICBjb25zdCBmbG9hdCBDID0gICA4LjBmOwotCi0gICAgLy8gc2NhbGUgYW5nbGUgZm9yIGVhc3kgYXJndW1lbnQgcmVkdWN0aW9uCi0gICAgeCAqPSBBOwotICAgIAotICAgIGlmIChmYWJzZih4KSA+PSAwLjVmKSB7Ci0gICAgICAgIC8vIEFyZ3VtZW50IHJlZHVjdGlvbgotICAgICAgICB4ID0geCAtIGNlaWxmKHggKyAwLjVmKSArIDEuMGY7IAotICAgIH0KLQotICAgIGNvbnN0IGZsb2F0IHkgPSBCKngqZmFic2YoeCkgKyBDKng7Ci0gICAgcmV0dXJuIDAuMjIxNWYgKiAoeSpmYWJzZih5KSAtIHkpICsgeTsKLX0KLQotZmxvYXQgY29zaW5lZihmbG9hdCB4KQotewotICAgIHJldHVybiBzaW5lZih4ICsgZmxvYXQoTV9QSS8yKSk7Ci19Ci0KLXZvaWQgc2luY29zZihHTGZsb2F0IGFuZ2xlLCBHTGZsb2F0KiBzLCBHTGZsb2F0KiBjKSB7Ci0gICAgKnMgPSBzaW5lZihhbmdsZSk7Ci0gICAgKmMgPSBjb3NpbmVmKGFuZ2xlKTsKLX0KLQotfTsgLy8gbmFtZXNwYWNlIGZwX3V0aWxzCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL2ZwLmggYi9vcGVuZ2wvbGliYWdsL2ZwLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDZkMGMxODMuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9mcC5oCisrKyAvZGV2L251bGwKQEAgLTEsMjQzICswLDAgQEAKLS8qIGxpYnMvb3BlbmdsZXMvZnAuaAotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9PUEVOR0xFU19GUF9ICi0jZGVmaW5lIEFORFJPSURfT1BFTkdMRVNfRlBfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3RkZGVmLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8bWF0aC5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS9waXhlbGZsaW5nZXIvZ2dsX2NvbnRleHQuaD4KLQotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLQotI2RlZmluZSBERUJVR19VU0VfRkxPQVRTICAgICAgMAotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLWV4dGVybiAiQyIgR0xmaXhlZCBnZ2xGbG9hdFRvRml4ZWQoZmxvYXQgZikgX19hdHRyaWJ1dGVfXygoY29uc3QpKTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1uYW1lc3BhY2UgZ2wgewotCi0gICAgICAgIEdMZmxvYXQgZml4ZWRUb0Zsb2F0KEdMZml4ZWQpIENPTlNUOwotCi0gICAgICAgIHZvaWQgICAgc2luY29zZihHTGZsb2F0IGFuZ2xlLCBHTGZsb2F0KiBzLCBHTGZsb2F0KiBjKTsKLSAgICAgICAgZmxvYXQgICBzaW5lZihHTGZsb2F0IHgpIENPTlNUOwotICAgICAgICBmbG9hdCAgIGNvc2luZWYoR0xmbG9hdCB4KSBDT05TVDsKLQotaW5saW5lIGJvb2wgICAgIGNtcGYoR0xmbG9hdCBhLCBHTGZsb2F0IGIpIENPTlNUOwotaW5saW5lIGJvb2wgICAgIGlzWmVyb2YoR0xmbG9hdCkgQ09OU1Q7Ci1pbmxpbmUgYm9vbCAgICAgaXNPbmVmKEdMZmxvYXQpIENPTlNUOwotCi1pbmxpbmUgaW50ICAgICAgaXNaZXJvT3JOZWdhdGl2ZWYoR0xmbG9hdCkgQ09OU1Q7Ci0KLWlubGluZSBpbnQgICAgICBleHBvbmVudChHTGZsb2F0KSBDT05TVDsKLWlubGluZSBpbnQzMl90ICBtYW50aXNzYShHTGZsb2F0KSBDT05TVDsKLWlubGluZSBHTGZsb2F0ICBjbGFtcFRvWmVyb2YoR0xmbG9hdCkgQ09OU1Q7Ci1pbmxpbmUgR0xmbG9hdCAgcmVjaXByb2NhbGYoR0xmbG9hdCkgQ09OU1Q7Ci1pbmxpbmUgR0xmbG9hdCAgcnNxcnRmKEdMZmxvYXQpIENPTlNUOwotaW5saW5lIEdMZmxvYXQgIHNxcmYoR0xmbG9hdCkgQ09OU1Q7Ci1pbmxpbmUgR0xmbG9hdCAgYWRkRXhwZihHTGZsb2F0IHYsIGludCBlKSBDT05TVDsKLWlubGluZSBHTGZsb2F0ICBtdWwyZihHTGZsb2F0IHYpIENPTlNUOwotaW5saW5lIEdMZmxvYXQgIGRpdjJmKEdMZmxvYXQgdikgQ09OU1Q7Ci1pbmxpbmUgR0xmbG9hdCAgYWJzZihHTGZsb2F0IHYpIENPTlNUOwotCi0KLS8qIAotICogZmxvYXQgZmFzdGV4cGYoZmxvYXQpIDogYSBmYXN0IGFwcHJveGltYXRpb24gb2YgZXhwZih4KQotICoJCWdpdmUgc29tZXdoYXQgYWNjdXJhdGUgcmVzdWx0cyBmb3IgLTg4IDw9IHggPD0gODgKLSAqCi0gKiBleHAoeCkgPSAyXih4L2xuKDIpKQotICogd2UgdXNlIHRoZSBwcm9wZXJ0aWVzIG9mIGZsb2F0IGVuY29kaW5nCi0gKiB0byBnZXQgYSBmYXN0IDJeIGFuZCBsaW5lYXIgaW50ZXJwb2xhdGlvbgotICoKLSAqLwotCi1pbmxpbmUgZmxvYXQgZmFzdGV4cGYoZmxvYXQgeSkgX19hdHRyaWJ1dGVfXygoY29uc3QpKTsKLQotaW5saW5lIGZsb2F0IGZhc3RleHBmKGZsb2F0IHkpCi17Ci0JdW5pb24gewotCQlmbG9hdAlyOwotCQlpbnQzMl90CWk7Ci0JfSB1OwkKLQotCS8vIDEyNypsbigyKSA9IDg4Ci0JaWYgKHkgPCAtODguMGYpIHsKLQkJdS5yID0gMC4wZjsKLQl9IGVsc2UgaWYgKHkgPiA4OC4wZikgewotCQl1LnIgPSBJTkZJTklUWTsKLQl9IGVsc2UgewotCQljb25zdCBmbG9hdCBrT25lT3ZlckxvZ1R3byA9ICgxTDw8MjMpIC8gTV9MTjI7Ci0JCWNvbnN0IGludDMyX3Qga0V4cG9uZW50QmlhcyA9IDEyN0w8PDIzOwotCQljb25zdCBpbnQzMl90IGUgPSBpbnQzMl90KHkqa09uZU92ZXJMb2dUd28pOwotCQl1LmkgPSBlICsga0V4cG9uZW50QmlhczsKLQl9Ci0JCi0JcmV0dXJuIHUucjsKLX0KLQotCi1ib29sIGNtcGYoR0xmbG9hdCBhLCBHTGZsb2F0IGIpIHsKLSNpZiBERUJVR19VU0VfRkxPQVRTCi0gICAgcmV0dXJuIGEgPT0gYjsKLSNlbHNlCi0gICAgdW5pb24gewotICAgICAgICBmbG9hdCAgICAgICBmOwotICAgICAgICB1aW50MzJfdCAgICBpOwotICAgIH0gdWEsIHViOwotICAgIHVhLmYgPSBhOwotICAgIHViLmYgPSBiOwotICAgIHJldHVybiB1YS5pID09IHViLmk7Ci0jZW5kaWYKLX0gCi0KLWJvb2wgaXNaZXJvZihHTGZsb2F0IHYpIHsKLSNpZiBERUJVR19VU0VfRkxPQVRTCi0gICAgcmV0dXJuIHYgPT0gMDsKLSNlbHNlCi0gICAgdW5pb24gewotICAgICAgICBmbG9hdCAgICAgICBmOwotICAgICAgICBpbnQzMl90ICAgICBpOwotICAgIH07Ci0gICAgZiA9IHY7Ci0gICAgcmV0dXJuIChpPDwxKSA9PSAwOwotI2VuZGlmCi19Ci0KLWJvb2wgaXNPbmVmKEdMZmxvYXQgdikgewotICAgIHJldHVybiBjbXBmKHYsIDEuMGYpOwotfQotCi1pbnQgaXNaZXJvT3JOZWdhdGl2ZWYoR0xmbG9hdCB2KSB7Ci0jaWYgREVCVUdfVVNFX0ZMT0FUUwotICAgIHJldHVybiB2IDw9IDA7Ci0jZWxzZQotICAgIHVuaW9uIHsKLSAgICAgICAgZmxvYXQgICAgICAgZjsKLSAgICAgICAgaW50MzJfdCAgICAgaTsKLSAgICB9OwotICAgIGYgPSB2OwotICAgIHJldHVybiBpc1plcm9mKHYpIHwgKGk+PjMxKTsKLSNlbmRpZgotfQotCi1pbnQgZXhwb25lbnQoR0xmbG9hdCB2KSB7Ci0gICAgdW5pb24gewotICAgICAgICBmbG9hdCAgICBmOwotICAgICAgICB1aW50MzJfdCBpOwotICAgIH07Ci0gICAgZiA9IHY7Ci0gICAgcmV0dXJuICgoaSA8PCAxKSA+PiAyNCkgLSAxMjc7Ci19Ci0KLWludDMyX3QgbWFudGlzc2EoR0xmbG9hdCB2KSB7Ci0gICAgdW5pb24gewotICAgICAgICBmbG9hdCAgICBmOwotICAgICAgICB1aW50MzJfdCBpOwotICAgIH07Ci0gICAgZiA9IHY7Ci0gICAgaWYgKCEoaSYweDdGODAwMDAwKSkgcmV0dXJuIDA7Ci0gICAgY29uc3QgaW50IHMgPSBpID4+IDMxOwotICAgIGkgfD0gKDFMPDwyMyk7Ci0gICAgaSAmPSB+MHhGRjAwMDAwMDsKLSAgICByZXR1cm4gcyA/IC1pIDogaTsKLX0KLQotR0xmbG9hdCBjbGFtcFRvWmVyb2YoR0xmbG9hdCB2KSB7Ci0jaWYgREVCVUdfVVNFX0ZMT0FUUwotICAgIHJldHVybiB2PDAgPyAwIDogKHY+MSA/IDEgOiB2KTsKLSNlbHNlCi0gICAgdW5pb24gewotICAgICAgICBmbG9hdCAgICAgICBmOwotICAgICAgICBpbnQzMl90ICAgICBpOwotICAgIH07Ci0gICAgZiA9IHY7Ci0gICAgaSAmPSB+KGk+PjMxKTsKLSAgICByZXR1cm4gZjsKLSNlbmRpZgotfQotCi1HTGZsb2F0IHJlY2lwcm9jYWxmKEdMZmxvYXQgdikgewotICAgIC8vIFhYWDogZG8gYmV0dGVyCi0gICAgcmV0dXJuIDEuMGYgLyB2OwotfQotCi1HTGZsb2F0IHJzcXJ0ZihHTGZsb2F0IHYpIHsKLSAgICAvLyBYWFg6IGRvIGJldHRlcgotICAgIHJldHVybiAxLjBmIC8gc3FydGYodik7Ci19Ci0KLUdMZmxvYXQgc3FyZihHTGZsb2F0IHYpIHsKLSAgICAvLyBYWFg6IGRvIGJldHRlcgotICAgIHJldHVybiB2KnY7Ci19Ci0KLUdMZmxvYXQgYWRkRXhwZihHTGZsb2F0IHYsIGludCBlKSB7Ci0gICAgdW5pb24gewotICAgICAgICBmbG9hdCAgICAgICBmOwotICAgICAgICBpbnQzMl90ICAgICBpOwotICAgIH07Ci0gICAgZiA9IHY7Ci0gICAgaWYgKGk8PDEpIHsgLy8gWFhYOiBkZWFsIHdpdGggb3Zlci91bmRlcmZsb3cJCi0gICAgICAgIGkgKz0gaW50MzJfdChlKTw8MjM7Ci0gICAgfQotICAgIHJldHVybiBmOwotfQotCi1HTGZsb2F0IG11bDJmKEdMZmxvYXQgdikgewotI2lmIERFQlVHX1VTRV9GTE9BVFMKLSAgICByZXR1cm4gdioyOwotI2Vsc2UKLSAgICByZXR1cm4gYWRkRXhwZih2LCAxKTsKLSNlbmRpZgotfQotCi1HTGZsb2F0IGRpdjJmKEdMZmxvYXQgdikgewotI2lmIERFQlVHX1VTRV9GTE9BVFMKLSAgICByZXR1cm4gdiowLjVmOwotI2Vsc2UKLSAgICByZXR1cm4gYWRkRXhwZih2LCAtMSk7Ci0jZW5kaWYKLX0KLQotR0xmbG9hdCAgYWJzZihHTGZsb2F0IHYpIHsKLSNpZiBERUJVR19VU0VfRkxPQVRTCi0gICAgcmV0dXJuIHY8MCA/IC12IDogdjsKLSNlbHNlCi0gICAgdW5pb24gewotICAgICAgICBmbG9hdCAgICAgICBmOwotICAgICAgICBpbnQzMl90ICAgICBpOwotICAgIH07Ci0gICAgZiA9IHY7Ci0gICAgaSAmPSB+MHg4MDAwMDAwMDsKLSAgICByZXR1cm4gZjsKLSNlbmRpZgotfQotCi19OyAgLy8gbmFtZXNwYWNlIGdsCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX09QRU5HTEVTX0ZQX0gKLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC9pdGVyYXRvcnMuUyBiL29wZW5nbC9saWJhZ2wvaXRlcmF0b3JzLlMKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGRhZjI5MzcuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9pdGVyYXRvcnMuUworKysgL2Rldi9udWxsCkBAIC0xLDg4ICswLDAgQEAKLS8qIGxpYnMvb3BlbmdsZXMvaXRlcmF0b3JzLlMKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0KLSAgICAudGV4dAotICAgIC5hbGlnbgotICAgIC5hcm0KLSAgICAKLSAgICAuZ2xvYmFsIGl0ZXJhdG9yczAwMzIKLQotLyoKLSAqIGl0ZXJhdG9yczAwMzIKLSAqCi0gKiBNVVNUIEJFIENBTExFRCBGUk9NIEFSTSBDT0RFCi0gKgotICogcjA6IGNvbnN0IGNvbXB1dGVfaXRlcmF0b3JzX3QqICh0aGlzKQotICogICAgICByMCArIDA6IG1fZHgwMSAKLSAqICAgICAgcjAgKyA0OiBtX2R5MTAKLSAqICAgICAgcjAgKyA4OiBtX2R4MjAKLSAqICAgICAgcjAgKzEyOiBtX2R5MDIKLSAqICAgICAgcjAgKzE2OiBtX3gwCi0gKiAgICAgIHIwICsyMDogbV95MAotICogICAgICByMCArMjQ6IG1fYXJlYQotICoJCXIwICsyODogbV9zY2FsZQotICoJCXIwICsyOTogbV9hcmVhX3NjYWxlOwotICogcjE6IGludDMyX3QqIChvdXQpCi0gKiAgICAgIHIxICsgMDogYwotICogICAgICByMSArIDQ6IGRjZHgKLSAqICAgICAgcjEgKyA4OiBkY2R5Ci0gKiAgIHIyOiBjMAotICogICByMzogYzEKLSAqIFtzcF06IGMyCi0gKi8KLSAKLWl0ZXJhdG9yczAwMzI6Ci0gICAgICAgIHN0bWZkCXNwISwge3I0LCByNSwgcjYsIHI3LCByOCwgbHJ9Ci0gICAgICAgIGxkciAgICAgcjQsIFtzcCwgIzQqNl0KLQotICAgICAgICBsZHJiICAgIHIxMiwgW3IwLCAjMjldCi0gICAgICAgIHN1YiAgICAgcjMsIHIzLCByMgotICAgICAgICBzdWIgICAgIHI0LCByNCwgcjIKLSAgICAgICAgc3ViICAgICByMTIsIHIxMiwgIzE2Ci0gICAgICAgIG1vdiAgICAgcjMsIHIzLCBhc3IgcjEyCi0gICAgICAgIG1vdiAgICAgcjQsIHI0LCBhc3IgcjEyCi0gICAgICAgIAotICAgICAgICBsZHIgICAgIHI1LCBbcjAsICMwXQotICAgICAgICBsZHIgICAgIHIxMiwgW3IwLCAjNF0KLSAgICAgICAgc211bGwgICByOCwgbHIsIHI0LCByNQotICAgICAgICBsZHIgICAgIHI1LCBbcjAsICM4XQotICAgICAgICBzbXVsbCAgIHI2LCByNywgcjQsIHIxMgotICAgICAgICBsZHIgICAgIHIxMiwgW3IwLCAjMTJdCi0gICAgICAgIHNtbGFsICAgcjgsIGxyLCByMywgcjUKLSAgICAgICAgc21sYWwgICByNiwgcjcsIHIzLCByMTIKLQotICAgICAgICBsZHIgICAgIHIzLCBbcjAsICMxNl0gICAgICAgIC8vIG1feDAKLSAgICAgICAgbGRyICAgICByNCwgW3IwLCAjMjBdICAgICAgICAvLyBtX3gxCi0gICAgICAgIAotICAgICAgICBzdHIgICAgIHI2LCBbcjEsICM0XQotICAgICAgICBzdHIgICAgIHI4LCBbcjEsICM4XQotCi0gICAgICAgIHVtdWxsICAgcjYsIHI1LCByMywgcjYKLSAgICAgICAgdW11bGwgICByOCwgcjAsIHI0LCByOAotICAgICAgICBtbGEgICAgIHI3LCByMywgcjcsIHI1Ci0gICAgICAgIG1sYSAgICAgbHIsIHI0LCBsciwgcjAKLSAgICAgICAgYWRkcyAgICByNiwgcjYsIHI4Ci0gICAgICAgIGFkYyAgICAgcjcsIHI3LCBscgotCi0gICAgICAgIG1vdnMgICAgcjYsIHI2LCBsc3IgIzQKLSAgICAgICAgYWRjICAgICByNiwgcjYsIHI3LCBsc2wgIzI4Ci0gICAgICAgIHJzYiAgICAgcjYsIHI2LCByMiwgbHNsICMxNgotICAgICAgICBzdHIgICAgIHI2LCBbcjEsICMwXQotCi0gICAgICAgIGxkbWZkCXNwISwge3I0LCByNSwgcjYsIHI3LCByOCwgcGN9Ci0KZGlmZiAtLWdpdCBhL29wZW5nbC9saWJhZ2wvbGlnaHQuY3BwIGIvb3BlbmdsL2xpYmFnbC9saWdodC5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI1YzQxZDAuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9saWdodC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSw4NTIgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9saWdodC5jcHAKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlICJjb250ZXh0LmgiCi0jaW5jbHVkZSAiZnAuaCIKLSNpbmNsdWRlICJsaWdodC5oIgotI2luY2x1ZGUgInN0YXRlLmgiCi0jaW5jbHVkZSAibWF0cml4LmgiCi0KLQotI2lmIGRlZmluZWQoX19hcm1fXykgJiYgZGVmaW5lZChfX3RodW1iX18pCi0jd2FybmluZyAibGlnaHQuY3BwIHNob3VsZCBub3QgYmUgY29tcGlsZWQgaW4gdGh1bWIgb24gQVJNLiIKLSNlbmRpZgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RhdGljIHZvaWQgaW52YWxpZGF0ZV9saWdodGluZyhvZ2xlc19jb250ZXh0X3QqIGMpOwotc3RhdGljIHZvaWQgbGlnaHRWZXJ0ZXhWYWxpZGF0ZShvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiB2KTsKLXN0YXRpYyB2b2lkIGxpZ2h0VmVydGV4Tm9wKG9nbGVzX2NvbnRleHRfdCogYywgdmVydGV4X3QqIHYpOwotc3RhdGljIHZvaWQgbGlnaHRWZXJ0ZXgob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdik7Ci1zdGF0aWMgdm9pZCBsaWdodFZlcnRleE1hdGVyaWFsKG9nbGVzX2NvbnRleHRfdCogYywgdmVydGV4X3QqIHYpOwotCi1zdGF0aWMgaW5saW5lIHZvaWQgdnNjYWxlMyhHTGZpeGVkKiBkLCBjb25zdCBHTGZpeGVkKiBtLCBHTGZpeGVkIHMpOwotc3RhdGljIGlubGluZSB2b2lkIHZzdWIzdyhHTGZpeGVkKiBkLCBjb25zdCBHTGZpeGVkKiBhLCBjb25zdCBHTGZpeGVkKiBiKTsKLQotc3RhdGljIF9fYXR0cmlidXRlX18oKG5vaW5saW5lKSkKLXZvaWQgdm5vcm0zKEdMZml4ZWQqIGQsIGNvbnN0IEdMZml4ZWQqIGEpOwotCi1zdGF0aWMgaW5saW5lIHZvaWQgdnNhMyhHTGZpeGVkKiBkLAotICAgIGNvbnN0IEdMZml4ZWQqIG0sIEdMZml4ZWQgcywgY29uc3QgR0xmaXhlZCogYSk7Ci1zdGF0aWMgaW5saW5lIHZvaWQgdm1sYTMoR0xmaXhlZCogZCwKLSAgICBjb25zdCBHTGZpeGVkKiBtMCwgY29uc3QgR0xmaXhlZCogbTEsIGNvbnN0IEdMZml4ZWQqIGEpOwotc3RhdGljIGlubGluZSB2b2lkIHZtdWwzKEdMZml4ZWQqIGQsCi0gICAgY29uc3QgR0xmaXhlZCogbTAsIGNvbnN0IEdMZml4ZWQqIG0xKTsKLQotc3RhdGljIEdMZml4ZWQgZm9nX2xpbmVhcihvZ2xlc19jb250ZXh0X3QqIGMsIEdMZml4ZWQgeik7Ci1zdGF0aWMgR0xmaXhlZCBmb2dfZXhwKG9nbGVzX2NvbnRleHRfdCogYywgR0xmaXhlZCB6KTsKLXN0YXRpYyBHTGZpeGVkIGZvZ19leHAyKG9nbGVzX2NvbnRleHRfdCogYywgR0xmaXhlZCB6KTsKLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyB2b2lkIGluaXRfd2hpdGUodmVjNF90JiBjKSB7Ci0gICAgYy5yID0gYy5nID0gYy5iID0gYy5hID0gMHgxMDAwMDsKLX0KLQotdm9pZCBvZ2xlc19pbml0X2xpZ2h0KG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBmb3IgKHVuc2lnbmVkIGludCBpPTAgOyBpPE9HTEVTX01BWF9MSUdIVFMgOyBpKyspIHsKLSAgICAgICAgYy0+bGlnaHRpbmcubGlnaHRzW2ldLmFtYmllbnQuYSA9IDB4MTAwMDA7Ci0gICAgICAgIGMtPmxpZ2h0aW5nLmxpZ2h0c1tpXS5wb3NpdGlvbi56ID0gMHgxMDAwMDsKLSAgICAgICAgYy0+bGlnaHRpbmcubGlnaHRzW2ldLnNwb3REaXIueiA9IC0weDEwMDAwOwotICAgICAgICBjLT5saWdodGluZy5saWdodHNbaV0uc3BvdEN1dG9mZiA9IGdnbEludFRvRml4ZWQoMTgwKTsKLSAgICAgICAgYy0+bGlnaHRpbmcubGlnaHRzW2ldLmF0dGVudWF0aW9uWzBdID0gMHgxMDAwMDsKLSAgICB9Ci0gICAgaW5pdF93aGl0ZShjLT5saWdodGluZy5saWdodHNbMF0uZGlmZnVzZSk7Ci0gICAgaW5pdF93aGl0ZShjLT5saWdodGluZy5saWdodHNbMF0uc3BlY3VsYXIpOwotCi0gICAgYy0+bGlnaHRpbmcuZnJvbnQuYW1iaWVudC5yID0KLSAgICBjLT5saWdodGluZy5mcm9udC5hbWJpZW50LmcgPQotICAgIGMtPmxpZ2h0aW5nLmZyb250LmFtYmllbnQuYiA9IGdnbEZsb2F0VG9GaXhlZCgwLjJmKTsKLSAgICBjLT5saWdodGluZy5mcm9udC5hbWJpZW50LmEgPSAweDEwMDAwOwotICAgIGMtPmxpZ2h0aW5nLmZyb250LmRpZmZ1c2UuciA9Ci0gICAgYy0+bGlnaHRpbmcuZnJvbnQuZGlmZnVzZS5nID0KLSAgICBjLT5saWdodGluZy5mcm9udC5kaWZmdXNlLmIgPSBnZ2xGbG9hdFRvRml4ZWQoMC44Zik7Ci0gICAgYy0+bGlnaHRpbmcuZnJvbnQuZGlmZnVzZS5hID0gMHgxMDAwMDsKLSAgICBjLT5saWdodGluZy5mcm9udC5zcGVjdWxhci5hID0gMHgxMDAwMDsKLSAgICBjLT5saWdodGluZy5mcm9udC5lbWlzc2lvbi5hID0gMHgxMDAwMDsKLQotICAgIGMtPmxpZ2h0aW5nLmxpZ2h0TW9kZWwuYW1iaWVudC5yID0KLSAgICBjLT5saWdodGluZy5saWdodE1vZGVsLmFtYmllbnQuZyA9Ci0gICAgYy0+bGlnaHRpbmcubGlnaHRNb2RlbC5hbWJpZW50LmIgPSBnZ2xGbG9hdFRvRml4ZWQoMC4yZik7Ci0gICAgYy0+bGlnaHRpbmcubGlnaHRNb2RlbC5hbWJpZW50LmEgPSAweDEwMDAwOwotCi0gICAgYy0+bGlnaHRpbmcuY29sb3JNYXRlcmlhbC5mYWNlID0gR0xfRlJPTlRfQU5EX0JBQ0s7Ci0gICAgYy0+bGlnaHRpbmcuY29sb3JNYXRlcmlhbC5tb2RlID0gR0xfQU1CSUVOVF9BTkRfRElGRlVTRTsKLQotICAgIGMtPmZvZy5tb2RlID0gR0xfRVhQOwotICAgIGMtPmZvZy5mb2cgPSBmb2dfZXhwOwotICAgIGMtPmZvZy5kZW5zaXR5ID0gMHgxMDAwMDsKLSAgICBjLT5mb2cuZW5kID0gMHgxMDAwMDsKLSAgICBjLT5mb2cuaW52RW5kTWludXNTdGFydCA9IDB4MTAwMDA7Ci0KLSAgICBpbnZhbGlkYXRlX2xpZ2h0aW5nKGMpOwotICAgICAgIAotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3Muc2hhZGVNb2RlbChjLCBHTF9TTU9PVEgpOwotICAgIGMtPmxpZ2h0aW5nLnNoYWRlTW9kZWwgPSBHTF9TTU9PVEg7Ci19Ci0KLXZvaWQgb2dsZXNfdW5pbml0X2xpZ2h0KG9nbGVzX2NvbnRleHRfdCogYykKLXsKLX0KLQotc3RhdGljIGlubGluZSBpbnQzMl90IGNsYW1wRihHTGZpeGVkIGYpIENPTlNUOwotaW50MzJfdCBjbGFtcEYoR0xmaXhlZCBmKSB7Ci0gICAgZiA9IChmICYgfihmPj4zMSkpOwotICAgIGlmIChmID49IDB4MTAwMDApCi0gICAgICAgIGYgPSAweDEwMDAwOwotICAgIHJldHVybiBmOwotfQotCi1zdGF0aWMgR0xmaXhlZCBmb2dfbGluZWFyKG9nbGVzX2NvbnRleHRfdCogYywgR0xmaXhlZCB6KSB7Ci0gICAgcmV0dXJuIGNsYW1wRihnZ2xNdWx4KChjLT5mb2cuZW5kIC0gKCh6PDApPy16OnopKSwgYy0+Zm9nLmludkVuZE1pbnVzU3RhcnQpKTsKLX0KLQotc3RhdGljIEdMZml4ZWQgZm9nX2V4cChvZ2xlc19jb250ZXh0X3QqIGMsIEdMZml4ZWQgeikgewotICAgIGNvbnN0IGZsb2F0IGUgPSBmaXhlZFRvRmxvYXQoZ2dsTXVseChjLT5mb2cuZGVuc2l0eSwgKCh6PDApPy16OnopKSk7Ci0gICAgcmV0dXJuIGNsYW1wRihnZ2xGbG9hdFRvRml4ZWQoZmFzdGV4cGYoLWUpKSk7Ci19Ci0KLXN0YXRpYyBHTGZpeGVkIGZvZ19leHAyKG9nbGVzX2NvbnRleHRfdCogYywgR0xmaXhlZCB6KSB7Ci0gICAgY29uc3QgZmxvYXQgZSA9IGZpeGVkVG9GbG9hdChnZ2xNdWx4KGMtPmZvZy5kZW5zaXR5LCB6KSk7Ci0gICAgcmV0dXJuIGNsYW1wRihnZ2xGbG9hdFRvRml4ZWQoZmFzdGV4cGYoLWUqZSkpKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jcHJhZ21hIG1hcmsgbWF0aCBoZWxwZXJzCi0jZW5kaWYKLQotc3RhdGljIGlubGluZQotdm9pZCB2c2NhbGUzKEdMZml4ZWQqIGQsIGNvbnN0IEdMZml4ZWQqIG0sIEdMZml4ZWQgcykgewotICAgIGRbMF0gPSBnZ2xNdWx4KG1bMF0sIHMpOwotICAgIGRbMV0gPSBnZ2xNdWx4KG1bMV0sIHMpOwotICAgIGRbMl0gPSBnZ2xNdWx4KG1bMl0sIHMpOwotfQotCi1zdGF0aWMgaW5saW5lCi12b2lkIHZzYTMoR0xmaXhlZCogZCwgY29uc3QgR0xmaXhlZCogbSwgR0xmaXhlZCBzLCBjb25zdCBHTGZpeGVkKiBhKSB7Ci0gICAgZFswXSA9IGdnbE11bEFkZHgobVswXSwgcywgYVswXSk7Ci0gICAgZFsxXSA9IGdnbE11bEFkZHgobVsxXSwgcywgYVsxXSk7Ci0gICAgZFsyXSA9IGdnbE11bEFkZHgobVsyXSwgcywgYVsyXSk7Ci19Ci0KLXN0YXRpYyBpbmxpbmUKLXZvaWQgdnN1YjN3KEdMZml4ZWQqIGQsIGNvbnN0IEdMZml4ZWQqIGEsIGNvbnN0IEdMZml4ZWQqIGIpIHsKLSAgICBjb25zdCBHTGZpeGVkIHdhID0gYVszXTsKLSAgICBjb25zdCBHTGZpeGVkIHdiID0gYlszXTsKLSAgICBpZiAoZ2dsX2xpa2VseSh3YSA9PSB3YikpIHsKLSAgICAgICAgZFswXSA9IGFbMF0gLSBiWzBdOwotICAgICAgICBkWzFdID0gYVsxXSAtIGJbMV07Ci0gICAgICAgIGRbMl0gPSBhWzJdIC0gYlsyXTsKLSAgICB9IGVsc2UgewotICAgICAgICBkWzBdID0gZ2dsTXVsU3VieChhWzBdLCB3YiwgZ2dsTXVseChiWzBdLCB3YSkpOwotICAgICAgICBkWzFdID0gZ2dsTXVsU3VieChhWzFdLCB3YiwgZ2dsTXVseChiWzFdLCB3YSkpOwotICAgICAgICBkWzJdID0gZ2dsTXVsU3VieChhWzJdLCB3YiwgZ2dsTXVseChiWzJdLCB3YSkpOwotICAgIH0KLX0KLQotc3RhdGljIGlubGluZQotdm9pZCB2bWxhMyhHTGZpeGVkKiBkLAotICAgICAgICBjb25zdCBHTGZpeGVkKiBtMCwgY29uc3QgR0xmaXhlZCogbTEsIGNvbnN0IEdMZml4ZWQqIGEpCi17Ci0gICAgZFswXSA9IGdnbE11bEFkZHgobTBbMF0sIG0xWzBdLCBhWzBdKTsKLSAgICBkWzFdID0gZ2dsTXVsQWRkeChtMFsxXSwgbTFbMV0sIGFbMV0pOwotICAgIGRbMl0gPSBnZ2xNdWxBZGR4KG0wWzJdLCBtMVsyXSwgYVsyXSk7Ci19Ci0KLXN0YXRpYyBpbmxpbmUKLXZvaWQgdm11bDMoR0xmaXhlZCogZCwgY29uc3QgR0xmaXhlZCogbTAsIGNvbnN0IEdMZml4ZWQqIG0xKSB7Ci0gICAgZFswXSA9IGdnbE11bHgobTBbMF0sIG0xWzBdKTsKLSAgICBkWzFdID0gZ2dsTXVseChtMFsxXSwgbTFbMV0pOwotICAgIGRbMl0gPSBnZ2xNdWx4KG0wWzJdLCBtMVsyXSk7Ci19Ci0KLXZvaWQgdm5vcm0zKEdMZml4ZWQqIGQsIGNvbnN0IEdMZml4ZWQqIGEpCi17Ci0gICAgLy8gd2UgbXVzdCB0YWtlIGNhcmUgb2Ygb3ZlcmZsb3dzIHdoZW4gbm9ybWFsaXppbmcgYSB2ZWN0b3IKLSAgICBHTGZpeGVkIG47Ci0gICAgaW50MzJfdCB4ID0gYVswXTsgICB4ID0geD49MCA/IHggOiAteDsKLSAgICBpbnQzMl90IHkgPSBhWzFdOyAgIHkgPSB5Pj0wID8geSA6IC15OwotICAgIGludDMyX3QgeiA9IGFbMl07ICAgeiA9IHo+PTAgPyB6IDogLXo7Ci0gICAgaWYgKGdnbF9saWtlbHkoeDw9MHg2ODAwICYmIHk8PTB4NjgwMCAmJiB6PD0gMHg2ODAwKSkgewotICAgICAgICAvLyBpbiB0aGlzIGNhc2UgdGhpcyB3aWxsIGFsbCBmaXQgb24gMzIgYml0cwotICAgICAgICBuID0geCp4ICsgeSp5ICsgeip6OwotICAgICAgICBuID0gZ2dsU3FydFJlY2lweChuKTsKLSAgICAgICAgbiA8PD0gODsKLSAgICB9IGVsc2UgewotICAgICAgICAvLyBoZXJlIG5vcm1eMiBpcyBhdCBsZWFzdCAweDdFQzAwMDAwICg+PjMyID09IDAuNDk1MTE3KQotICAgICAgICBuID0gdnNxdWFyZTMoeCwgeSwgeik7Ci0gICAgICAgIG4gPSBnZ2xTcXJ0UmVjaXB4KG4pOwotICAgIH0KLSAgICB2c2NhbGUzKGQsIGEsIG4pOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBsaWdodGluZyBlcXVhdGlvbnMKLSNlbmRpZgotCi1zdGF0aWMgaW5saW5lIHZvaWQgbGlnaHRfcGlja2VyKG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBpZiAoZ2dsX2xpa2VseSghYy0+bGlnaHRpbmcuZW5hYmxlKSkgewotICAgICAgICBjLT5saWdodGluZy5saWdodFZlcnRleCA9IGxpZ2h0VmVydGV4Tm9wOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmIChjLT5saWdodGluZy5jb2xvck1hdGVyaWFsLmVuYWJsZSkgewotICAgICAgICBjLT5saWdodGluZy5saWdodFZlcnRleCA9IGxpZ2h0VmVydGV4TWF0ZXJpYWw7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgYy0+bGlnaHRpbmcubGlnaHRWZXJ0ZXggPSBsaWdodFZlcnRleDsKLSAgICB9Ci19Ci0KLXN0YXRpYyBpbmxpbmUgdm9pZCB2YWxpZGF0ZV9saWdodF9tdmkob2dsZXNfY29udGV4dF90KiBjKQotewotICAgIHVpbnQzMl90IGVuID0gYy0+bGlnaHRpbmcuZW5hYmxlZExpZ2h0czsKLSAgICB3aGlsZSAoZW4pIHsKLSAgICAgICAgY29uc3QgaW50IGkgPSAzMSAtIGdnbENseihlbik7Ci0gICAgICAgIGVuICY9IH4oMTw8aSk7Ci0gICAgICAgIGxpZ2h0X3QmIGwgPSBjLT5saWdodGluZy5saWdodHNbaV07Ci0gICAgICAgIGMtPnRyYW5zZm9ybXMubXZ1aS5wb2ludDMoJmMtPnRyYW5zZm9ybXMubXZ1aSwKLSAgICAgICAgICAgICAgICAmbC5vYmpQb3NpdGlvbiwgJmwucG9zaXRpb24pOwotICAgICAgICB2bm9ybTMobC5ub3JtYWxpemVkT2JqUG9zaXRpb24udiwgbC5vYmpQb3NpdGlvbi52KTsKLSAgICB9Ci19Ci0KLXN0YXRpYyBpbmxpbmUgdm9pZCB2YWxpZGF0ZV9saWdodChvZ2xlc19jb250ZXh0X3QqIGMpCi17Ci0gICAgLy8gaWYgY29sb3JNYXRlcmlhbCBpcyBlbmFibGVkLCB3ZSBnZXQgdGhlIGNvbG9yIGZyb20gdGhlIHZlcnRleAotICAgIGlmICghYy0+bGlnaHRpbmcuY29sb3JNYXRlcmlhbC5lbmFibGUpIHsKLSAgICAgICAgbWF0ZXJpYWxfdCYgbWF0ZXJpYWwgPSBjLT5saWdodGluZy5mcm9udDsKLSAgICAgICAgdWludDMyX3QgZW4gPSBjLT5saWdodGluZy5lbmFibGVkTGlnaHRzOwotICAgICAgICB3aGlsZSAoZW4pIHsKLSAgICAgICAgICAgIGNvbnN0IGludCBpID0gMzEgLSBnZ2xDbHooZW4pOwotICAgICAgICAgICAgZW4gJj0gfigxPDxpKTsKLSAgICAgICAgICAgIGxpZ2h0X3QmIGwgPSBjLT5saWdodGluZy5saWdodHNbaV07Ci0gICAgICAgICAgICB2bXVsMyhsLmltcGxpY2l0QW1iaWVudC52LCAgbWF0ZXJpYWwuYW1iaWVudC52LCAgbC5hbWJpZW50LnYpOwotICAgICAgICAgICAgdm11bDMobC5pbXBsaWNpdERpZmZ1c2UudiwgIG1hdGVyaWFsLmRpZmZ1c2UudiwgIGwuZGlmZnVzZS52KTsKLSAgICAgICAgICAgIHZtdWwzKGwuaW1wbGljaXRTcGVjdWxhci52LCBtYXRlcmlhbC5zcGVjdWxhci52LCBsLnNwZWN1bGFyLnYpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICAvLyB0aGlzIGlzIGp1c3QgYSBmbGFnIHRvIHRlbGwgaWYgd2UgaGF2ZSBhIHNwZWN1bGFyIGNvbXBvbmVudAotICAgICAgICAgICAgbC5pbXBsaWNpdFNwZWN1bGFyLnZbM10gPQotICAgICAgICAgICAgICAgICAgICBsLmltcGxpY2l0U3BlY3VsYXIuciB8Ci0gICAgICAgICAgICAgICAgICAgIGwuaW1wbGljaXRTcGVjdWxhci5nIHwKLSAgICAgICAgICAgICAgICAgICAgbC5pbXBsaWNpdFNwZWN1bGFyLmI7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIGwuckNvbnN0QXR0ZW51YXRpb24gPSAobC5hdHRlbnVhdGlvblsxXSB8IGwuYXR0ZW51YXRpb25bMl0pPT0wOwotICAgICAgICAgICAgaWYgKGwuckNvbnN0QXR0ZW51YXRpb24pCi0gICAgICAgICAgICAgICAgbC5yQ29uc3RBdHRlbnVhdGlvbiA9IGdnbFJlY2lwRmFzdChsLmF0dGVudWF0aW9uWzBdKTsKLSAgICAgICAgfQotICAgICAgICAvLyBlbWlzc2lvbiBhbmQgYW1iaWVudCBmb3IgdGhlIHdob2xlIHNjZW5lCi0gICAgICAgIHZtbGEzKCAgYy0+bGlnaHRpbmcuaW1wbGljaXRTY2VuZUVtaXNzaW9uQW5kQW1iaWVudC52LAotICAgICAgICAgICAgICAgIGMtPmxpZ2h0aW5nLmxpZ2h0TW9kZWwuYW1iaWVudC52LAotICAgICAgICAgICAgICAgIG1hdGVyaWFsLmFtYmllbnQudiwgCi0gICAgICAgICAgICAgICAgbWF0ZXJpYWwuZW1pc3Npb24udik7Ci0gICAgICAgIGMtPmxpZ2h0aW5nLmltcGxpY2l0U2NlbmVFbWlzc2lvbkFuZEFtYmllbnQuYSA9IG1hdGVyaWFsLmRpZmZ1c2UuYTsKLSAgICB9Ci0gICAgdmFsaWRhdGVfbGlnaHRfbXZpKGMpOwotfQotCi12b2lkIGludmFsaWRhdGVfbGlnaHRpbmcob2dsZXNfY29udGV4dF90KiBjKQotewotICAgIC8vIFRPRE86IHBpY2sgbGlnaHRWZXJ0ZXhWYWxpZGF0ZSBvciBsaWdodFZlcnRleFZhbGlkYXRlTVZJCi0gICAgLy8gaW5zdGVhZCBvZiBzeXN0ZW1hdGljYWxseSB0aGUgaGVhdmllciBsaWdodFZlcnRleFZhbGlkYXRlKCkKLSAgICBjLT5saWdodGluZy5saWdodFZlcnRleCA9IGxpZ2h0VmVydGV4VmFsaWRhdGU7Ci19Ci0KLXZvaWQgb2dsZXNfaW52YWxpZGF0ZV9saWdodGluZ19tdnVpKG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBpbnZhbGlkYXRlX2xpZ2h0aW5nKGMpOwotfQotCi12b2lkIGxpZ2h0VmVydGV4Tm9wKG9nbGVzX2NvbnRleHRfdCosIHZlcnRleF90KiB2KQotewotICAgIC8vIHdlIHNob3VsZCBuZXZlciBlbmQtdXAgaGVyZQotfQotCi12b2lkIGxpZ2h0VmVydGV4VmFsaWRhdGVNVkkob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikKLXsKLSAgICB2YWxpZGF0ZV9saWdodF9tdmkoYyk7Ci0gICAgbGlnaHRfcGlja2VyKGMpOwotICAgIGMtPmxpZ2h0aW5nLmxpZ2h0VmVydGV4KGMsIHYpOwotfQotCi12b2lkIGxpZ2h0VmVydGV4VmFsaWRhdGUob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikKLXsKLSAgICB2YWxpZGF0ZV9saWdodChjKTsKLSAgICBsaWdodF9waWNrZXIoYyk7Ci0gICAgYy0+bGlnaHRpbmcubGlnaHRWZXJ0ZXgoYywgdik7Ci19Ci0KLXZvaWQgbGlnaHRWZXJ0ZXhNYXRlcmlhbChvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiB2KQotewotICAgIC8vIGZldGNoIHRoZSBtYXRlcmlhbCBjb2xvcgotICAgIGNvbnN0IEdMdm9pZCogY3AgPSBjLT5hcnJheXMuY29sb3IuZWxlbWVudCgKLSAgICAgICAgICAgIHYtPmluZGV4ICYgdmVydGV4X2NhY2hlX3Q6OklOREVYX01BU0spOwotICAgIGMtPmFycmF5cy5jb2xvci5mZXRjaChjLCB2LT5jb2xvci52LCBjcCk7Ci0KLSAgICAvLyBhY3F1aXJlIHRoZSBjb2xvci1tYXRlcmlhbCBmcm9tIHRoZSB2ZXJ0ZXgKLSAgICBtYXRlcmlhbF90JiBtYXRlcmlhbCA9IGMtPmxpZ2h0aW5nLmZyb250OwotICAgIG1hdGVyaWFsLmFtYmllbnQgPQotICAgIG1hdGVyaWFsLmRpZmZ1c2UgPSB2LT5jb2xvcjsKLSAgICAvLyBpbXBsaWNpdCBhcmd1bWVudHMgbmVlZCB0byBiZSBjb21wdXRlZCBwZXIvdmVydGV4Ci0gICAgdWludDMyX3QgZW4gPSBjLT5saWdodGluZy5lbmFibGVkTGlnaHRzOwotICAgIHdoaWxlIChlbikgewotICAgICAgICBjb25zdCBpbnQgaSA9IDMxIC0gZ2dsQ2x6KGVuKTsKLSAgICAgICAgZW4gJj0gfigxPDxpKTsKLSAgICAgICAgbGlnaHRfdCYgbCA9IGMtPmxpZ2h0aW5nLmxpZ2h0c1tpXTsKLSAgICAgICAgdm11bDMobC5pbXBsaWNpdEFtYmllbnQudiwgIG1hdGVyaWFsLmFtYmllbnQudiwgIGwuYW1iaWVudC52KTsKLSAgICAgICAgdm11bDMobC5pbXBsaWNpdERpZmZ1c2UudiwgIG1hdGVyaWFsLmRpZmZ1c2UudiwgIGwuZGlmZnVzZS52KTsKLSAgICAgICAgdm11bDMobC5pbXBsaWNpdFNwZWN1bGFyLnYsIG1hdGVyaWFsLnNwZWN1bGFyLnYsIGwuc3BlY3VsYXIudik7Ci0gICAgfQotICAgIC8vIGVtaXNzaW9uIGFuZCBhbWJpZW50IGZvciB0aGUgd2hvbGUgc2NlbmUKLSAgICB2bWxhMyggIGMtPmxpZ2h0aW5nLmltcGxpY2l0U2NlbmVFbWlzc2lvbkFuZEFtYmllbnQudiwKLSAgICAgICAgICAgIGMtPmxpZ2h0aW5nLmxpZ2h0TW9kZWwuYW1iaWVudC52LAotICAgICAgICAgICAgbWF0ZXJpYWwuYW1iaWVudC52LCAKLSAgICAgICAgICAgIG1hdGVyaWFsLmVtaXNzaW9uLnYpOwotICAgIGMtPmxpZ2h0aW5nLmltcGxpY2l0U2NlbmVFbWlzc2lvbkFuZEFtYmllbnQuYSA9IG1hdGVyaWFsLmRpZmZ1c2UuYTsKLQotICAgIC8vIG5vdyB3ZSBjYW4gbGlnaHQgb3VyIHZlcnRleCBhcyB1c3VhbAotICAgIGxpZ2h0VmVydGV4KGMsIHYpOwotfQotCi12b2lkIGxpZ2h0VmVydGV4KG9nbGVzX2NvbnRleHRfdCogYywgdmVydGV4X3QqIHYpCi17Ci0gICAgLy8gZW1pc3Npb24gYW5kIGFtYmllbnQgZm9yIHRoZSB3aG9sZSBzY2VuZQotICAgIHZlYzRfdCByID0gYy0+bGlnaHRpbmcuaW1wbGljaXRTY2VuZUVtaXNzaW9uQW5kQW1iaWVudDsKLQotICAgIHVpbnQzMl90IGVuID0gYy0+bGlnaHRpbmcuZW5hYmxlZExpZ2h0czsKLSAgICBpZiAoZ2dsX2xpa2VseShlbikpIHsKLSAgICAgICAgLy8gc2luY2Ugd2UgZG8gdGhlIGxpZ2h0aW5nIGluIG9iamVjdC1zcGFjZSwgd2UgZG9uJ3QgbmVlZCB0bwotICAgICAgICAvLyB0cmFuc2Zvcm0gZWFjaCBub3JtYWwuIEhvd2V2ZXIsIHdlIG1pZ2h0IHN0aWxsIGhhdmUgdG8gbm9ybWFsaXplCi0gICAgICAgIC8vIGl0IGlmIEdMX05PUk1BTElaRSBpcyBlbmFibGVkLgotICAgICAgICB2ZWM0X3QgbjsKLSAgICAgICAgYy0+YXJyYXlzLm5vcm1hbC5mZXRjaChjLCBuLnYsCi0gICAgICAgICAgICBjLT5hcnJheXMubm9ybWFsLmVsZW1lbnQodi0+aW5kZXggJiB2ZXJ0ZXhfY2FjaGVfdDo6SU5ERVhfTUFTSykpOwotICAgICAgICBpZiAoYy0+dHJhbnNmb3Jtcy5yZXNjYWxlTm9ybWFscyA9PSBHTF9OT1JNQUxJWkUpCi0gICAgICAgICAgICB2bm9ybTMobi52LCBuLnYpOwotCi0gICAgICAgIGNvbnN0IG1hdGVyaWFsX3QmIG1hdGVyaWFsID0gYy0+bGlnaHRpbmcuZnJvbnQ7Ci0gICAgICAgIGNvbnN0IGludCB0d29TaWRlID0gYy0+bGlnaHRpbmcubGlnaHRNb2RlbC50d29TaWRlOwotCi0gICAgICAgIHdoaWxlIChlbikgewotICAgICAgICAgICAgY29uc3QgaW50IGkgPSAzMSAtIGdnbENseihlbik7Ci0gICAgICAgICAgICBlbiAmPSB+KDE8PGkpOwotICAgICAgICAgICAgY29uc3QgbGlnaHRfdCYgbCA9IGMtPmxpZ2h0aW5nLmxpZ2h0c1tpXTsKLSAgICAgICAgICAgIAotICAgICAgICAgICAgdmVjNF90IGQsIHQ7Ci0gICAgICAgICAgICBHTGZpeGVkIHM7Ci0gICAgICAgICAgICBHTGZpeGVkIHNxRGlzdCA9IDB4MTAwMDA7Ci0KLSAgICAgICAgICAgIC8vIGNvbXB1dGUgdmVydGV4LXRvLWxpZ2h0IHZlY3RvcgotICAgICAgICAgICAgaWYgKGdnbF91bmxpa2VseShsLnBvc2l0aW9uLncpKSB7Ci0gICAgICAgICAgICAgICAgdnN1YjN3KGQudiwgbC5vYmpQb3NpdGlvbi52LCB2LT5vYmoudik7Ci0gICAgICAgICAgICAgICAgc3FEaXN0ID0gZG90MyhkLnYsIGQudik7Ci0gICAgICAgICAgICAgICAgdnNjYWxlMyhkLnYsIGQudiwgZ2dsU3FydFJlY2lweChzcURpc3QpKTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgLy8gVE9ETzogYXZvaWQgY29weSBoZXJlCi0gICAgICAgICAgICAgICAgZCA9IGwubm9ybWFsaXplZE9ialBvc2l0aW9uOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBhbWJpZW50ICYgZGlmZnVzZQotICAgICAgICAgICAgcyA9IGRvdDMobi52LCBkLnYpOwotICAgICAgICAgICAgcyA9IChzPDApID8gKHR3b1NpZGU/KC1zKTowKSA6IHM7Ci0gICAgICAgICAgICB2c2EzKHQudiwgbC5pbXBsaWNpdERpZmZ1c2UudiwgcywgbC5pbXBsaWNpdEFtYmllbnQudik7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8vIHNwZWN1bGFyCi0gICAgICAgICAgICBpZiAoZ2dsX3VubGlrZWx5KHMgJiYgbC5pbXBsaWNpdFNwZWN1bGFyLnZbM10pKSB7Ci0gICAgICAgICAgICAgICAgdmVjNF90IGg7Ci0gICAgICAgICAgICAgICAgaC54ID0gZC54OwotICAgICAgICAgICAgICAgIGgueSA9IGQueTsKLSAgICAgICAgICAgICAgICBoLnogPSBkLnogKyAweDEwMDAwOwotICAgICAgICAgICAgICAgIHZub3JtMyhoLnYsIGgudik7Ci0gICAgICAgICAgICAgICAgcyA9IGRvdDMobi52LCBoLnYpOwotICAgICAgICAgICAgICAgIHMgPSAoczwwKSA/ICh0d29TaWRlPygtcyk6MCkgOiBzOwotICAgICAgICAgICAgICAgIGlmIChzID4gMCkgewotICAgICAgICAgICAgICAgICAgICBzID0gZ2dsUG93eChzLCBtYXRlcmlhbC5zaGluaW5lc3MpOwotICAgICAgICAgICAgICAgICAgICB2c2EzKHQudiwgbC5pbXBsaWNpdFNwZWN1bGFyLnYsIHMsIHQudik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBzcG90Ci0gICAgICAgICAgICBpZiAoZ2dsX3VubGlrZWx5KGwuc3BvdEN1dG9mZiAhPSBnZ2xJbnRUb0ZpeGVkKDE4MCkpKSB7Ci0gICAgICAgICAgICAgICAgR0xmaXhlZCBzcG90QXR0ID0gLWRvdDMobC5ub3JtYWxpemVkU3BvdERpci52LCBkLnYpOwotICAgICAgICAgICAgICAgIGlmIChzcG90QXR0ID49IGwuc3BvdEN1dG9mZkNvc2luZSkgewotICAgICAgICAgICAgICAgICAgICB2c2NhbGUzKHQudiwgdC52LCBnZ2xQb3d4KHNwb3RBdHQsIGwuc3BvdEV4cCkpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgLy8gYXR0ZW51YXRpb24KLSAgICAgICAgICAgIGlmIChnZ2xfdW5saWtlbHkobC5wb3NpdGlvbi53KSkgewotICAgICAgICAgICAgICAgIGlmIChsLnJDb25zdEF0dGVudWF0aW9uKSB7Ci0gICAgICAgICAgICAgICAgICAgIHMgPSBsLnJDb25zdEF0dGVudWF0aW9uOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHMgPSBnZ2xNdWxBZGR4KHNxRGlzdCwgbC5hdHRlbnVhdGlvblsyXSwgbC5hdHRlbnVhdGlvblswXSk7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChsLmF0dGVudWF0aW9uWzFdKQotICAgICAgICAgICAgICAgICAgICAgICAgcyA9IGdnbE11bEFkZHgoZ2dsU3FydHgoc3FEaXN0KSwgbC5hdHRlbnVhdGlvblsxXSwgcyk7Ci0gICAgICAgICAgICAgICAgICAgIHMgPSBnZ2xSZWNpcEZhc3Qocyk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIHZzY2FsZTModC52LCB0LnYsIHMpOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICByLnIgKz0gdC5yOwotICAgICAgICAgICAgci5nICs9IHQuZzsKLSAgICAgICAgICAgIHIuYiArPSB0LmI7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgdi0+Y29sb3IuciA9IGdnbENsYW1weChyLnIpOwotICAgIHYtPmNvbG9yLmcgPSBnZ2xDbGFtcHgoci5nKTsKLSAgICB2LT5jb2xvci5iID0gZ2dsQ2xhbXB4KHIuYik7Ci0gICAgdi0+Y29sb3IuYSA9IGdnbENsYW1weChyLmEpOwotICAgIHYtPmZsYWdzIHw9IHZlcnRleF90OjpMSVQ7Ci19Ci0KLXN0YXRpYyB2b2lkIGxpZ2h0TW9kZWx4KEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSwgb2dsZXNfY29udGV4dF90KiBjKQotewotICAgIGlmIChnZ2xfdW5saWtlbHkocG5hbWUgIT0gR0xfTElHSFRfTU9ERUxfVFdPX1NJREUpKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgYy0+bGlnaHRpbmcubGlnaHRNb2RlbC50d29TaWRlID0gcGFyYW0gPyBHTF9UUlVFIDogR0xfRkFMU0U7Ci0gICAgaW52YWxpZGF0ZV9saWdodGluZyhjKTsKLX0KLQotc3RhdGljIHZvaWQgbGlnaHR4KEdMZW51bSBpLCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0sIG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBpZiAoZ2dsX3VubGlrZWx5KHVpbnQzMl90KGktR0xfTElHSFQwKSA+PSBPR0xFU19NQVhfTElHSFRTKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgbGlnaHRfdCYgbGlnaHQgPSBjLT5saWdodGluZy5saWdodHNbaS1HTF9MSUdIVDBdOwotICAgIGNvbnN0IEdMZml4ZWQga0RlZ1RvUmFkID0gR0xmaXhlZCgoTV9QSSAqIGdnbEludFRvRml4ZWQoMSkpIC8gMTgwLjBmKTsKLSAgICBzd2l0Y2ggKHBuYW1lKSB7Ci0gICAgY2FzZSBHTF9TUE9UX0VYUE9ORU5UOgotICAgICAgICBpZiAoR0dMZml4ZWQocGFyYW0pID49IGdnbEludFRvRml4ZWQoMTI4KSkgewotICAgICAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgbGlnaHQuc3BvdEV4cCA9IHBhcmFtOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1NQT1RfQ1VUT0ZGOgotICAgICAgICBpZiAocGFyYW0hPWdnbEludFRvRml4ZWQoMTgwKSAmJiBHR0xmaXhlZChwYXJhbSk+PWdnbEludFRvRml4ZWQoOTApKSB7Ci0gICAgICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX1ZBTFVFKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBsaWdodC5zcG90Q3V0b2ZmID0gcGFyYW07Ci0gICAgICAgIGxpZ2h0LnNwb3RDdXRvZmZDb3NpbmUgPSAKLSAgICAgICAgICAgICAgICBnZ2xGbG9hdFRvRml4ZWQoY29zaW5lZigoTV9QSS8oMTgwLjBmKjY1NTM2LjBmKSkqcGFyYW0pKTsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9DT05TVEFOVF9BVFRFTlVBVElPTjoKLSAgICAgICAgaWYgKHBhcmFtIDwgMCkgewotICAgICAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICAgICAgbGlnaHQuYXR0ZW51YXRpb25bMF0gPSBwYXJhbTsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9MSU5FQVJfQVRURU5VQVRJT046Ci0gICAgICAgIGlmIChwYXJhbSA8IDApIHsKLSAgICAgICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIGxpZ2h0LmF0dGVudWF0aW9uWzFdID0gcGFyYW07Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfUVVBRFJBVElDX0FUVEVOVUFUSU9OOgotICAgICAgICBpZiAocGFyYW0gPCAwKSB7Ci0gICAgICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX1ZBTFVFKTsKLSAgICAgICAgICAgIHJldHVybjsKLSAgICAgICAgfQotICAgICAgICBsaWdodC5hdHRlbnVhdGlvblsyXSA9IHBhcmFtOwotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGludmFsaWRhdGVfbGlnaHRpbmcoYyk7Ci19Ci0KLXN0YXRpYyB2b2lkIGxpZ2h0eHYoR0xlbnVtIGksIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zLCBvZ2xlc19jb250ZXh0X3QqIGMpCi17Ci0gICAgaWYgKGdnbF91bmxpa2VseSh1aW50MzJfdChpLUdMX0xJR0hUMCkgPj0gT0dMRVNfTUFYX0xJR0hUUykpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIEdMZml4ZWQqIHdoYXQ7Ci0gICAgbGlnaHRfdCYgbGlnaHQgPSBjLT5saWdodGluZy5saWdodHNbaS1HTF9MSUdIVDBdOwotICAgIHN3aXRjaCAocG5hbWUpIHsKLSAgICBjYXNlIEdMX0FNQklFTlQ6Ci0gICAgICAgIHdoYXQgPSBsaWdodC5hbWJpZW50LnY7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfRElGRlVTRToKLSAgICAgICAgd2hhdCA9IGxpZ2h0LmRpZmZ1c2UudjsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9TUEVDVUxBUjoKLSAgICAgICAgd2hhdCA9IGxpZ2h0LnNwZWN1bGFyLnY7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfUE9TSVRJT046IHsKLSAgICAgICAgb2dsZXNfdmFsaWRhdGVfdHJhbnNmb3JtKGMsIHRyYW5zZm9ybV9zdGF0ZV90OjpNT0RFTFZJRVcpOwotICAgICAgICB0cmFuc2Zvcm1fdCYgbXYgPSBjLT50cmFuc2Zvcm1zLm1vZGVsdmlldy50cmFuc2Zvcm07Ci0gICAgICAgIG1lbWNweShsaWdodC5wb3NpdGlvbi52LCBwYXJhbXMsIHNpemVvZihsaWdodC5wb3NpdGlvbi52KSk7Ci0gICAgICAgIG12LnBvaW50NCgmbXYsICZsaWdodC5wb3NpdGlvbiwgJmxpZ2h0LnBvc2l0aW9uKTsKLSAgICAgICAgaW52YWxpZGF0ZV9saWdodGluZyhjKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBjYXNlIEdMX1NQT1RfRElSRUNUSU9OOiB7Ci0gICAgICAgIG9nbGVzX3ZhbGlkYXRlX3RyYW5zZm9ybShjLCB0cmFuc2Zvcm1fc3RhdGVfdDo6TVZVSSk7Ci0gICAgICAgIHRyYW5zZm9ybV90JiBtdnVpID0gYy0+dHJhbnNmb3Jtcy5tdnVpOwotICAgICAgICBtdnVpLnBvaW50MygmbXZ1aSwgJmxpZ2h0LnNwb3REaXIsICh2ZWM0X3QqKXBhcmFtcyk7Ci0gICAgICAgIHZub3JtMyhsaWdodC5ub3JtYWxpemVkU3BvdERpci52LCBsaWdodC5zcG90RGlyLnYpOwotICAgICAgICBpbnZhbGlkYXRlX2xpZ2h0aW5nKGMpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGRlZmF1bHQ6Ci0gICAgICAgIGxpZ2h0eChpLCBwbmFtZSwgcGFyYW1zWzBdLCBjKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICB3aGF0WzBdID0gcGFyYW1zWzBdOwotICAgIHdoYXRbMV0gPSBwYXJhbXNbMV07Ci0gICAgd2hhdFsyXSA9IHBhcmFtc1syXTsKLSAgICB3aGF0WzNdID0gcGFyYW1zWzNdOwotICAgIGludmFsaWRhdGVfbGlnaHRpbmcoYyk7Ci19Ci0KLXN0YXRpYyB2b2lkIG1hdGVyaWFseChHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtLCBvZ2xlc19jb250ZXh0X3QqIGMpCi17Ci0gICAgaWYgKGdnbF91bmxpa2VseShmYWNlICE9IEdMX0ZST05UX0FORF9CQUNLKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmIChnZ2xfdW5saWtlbHkocG5hbWUgIT0gR0xfU0hJTklORVNTKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGMtPmxpZ2h0aW5nLmZyb250LnNoaW5pbmVzcyA9IHBhcmFtOwotICAgIGludmFsaWRhdGVfbGlnaHRpbmcoYyk7Ci19Ci0KLXN0YXRpYyB2b2lkIGZvZ3goR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtLCBvZ2xlc19jb250ZXh0X3QqIGMpCi17Ci0gICAgc3dpdGNoIChwbmFtZSkgewotICAgIGNhc2UgR0xfRk9HX0RFTlNJVFk6Ci0gICAgICAgIGlmIChwYXJhbSA+PSAwKSB7Ci0gICAgICAgICAgICBjLT5mb2cuZGVuc2l0eSA9IHBhcmFtOwotICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgIH0KLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfRk9HX1NUQVJUOgotICAgICAgICBjLT5mb2cuc3RhcnQgPSBwYXJhbTsKLSAgICAgICAgYy0+Zm9nLmludkVuZE1pbnVzU3RhcnQgPSBnZ2xSZWNpcChjLT5mb2cuZW5kIC0gYy0+Zm9nLnN0YXJ0KTsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9GT0dfRU5EOgotICAgICAgICBjLT5mb2cuZW5kID0gcGFyYW07Ci0gICAgICAgIGMtPmZvZy5pbnZFbmRNaW51c1N0YXJ0ID0gZ2dsUmVjaXAoYy0+Zm9nLmVuZCAtIGMtPmZvZy5zdGFydCk7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfRk9HX01PREU6Ci0gICAgICAgIHN3aXRjaCAocGFyYW0pIHsKLSAgICAgICAgY2FzZSBHTF9MSU5FQVI6Ci0gICAgICAgICAgICBjLT5mb2cubW9kZSA9IHBhcmFtOwotICAgICAgICAgICAgYy0+Zm9nLmZvZyA9IGZvZ19saW5lYXI7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBHTF9FWFA6Ci0gICAgICAgICAgICBjLT5mb2cubW9kZSA9IHBhcmFtOwotICAgICAgICAgICAgYy0+Zm9nLmZvZyA9IGZvZ19leHA7Ci0gICAgICAgICAgICBicmVhazsKLSAgICAgICAgY2FzZSBHTF9FWFAyOgotICAgICAgICAgICAgYy0+Zm9nLm1vZGUgPSBwYXJhbTsKLSAgICAgICAgICAgIGMtPmZvZy5mb2cgPSBmb2dfZXhwMjsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICBkZWZhdWx0OgotICAgICAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgICAgIGJyZWFrOwotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgIGRlZmF1bHQ6Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIGJyZWFrOwotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdXNpbmcgbmFtZXNwYWNlIGFuZHJvaWQ7Ci0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIGxpZ2h0aW5nIEFQSXMKLSNlbmRpZgotCi12b2lkIGdsU2hhZGVNb2RlbChHTGVudW0gbW9kZSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmIChnZ2xfdW5saWtlbHkobW9kZSAhPSBHTF9TTU9PVEggJiYgbW9kZSAhPSBHTF9GTEFUKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGMtPmxpZ2h0aW5nLnNoYWRlTW9kZWwgPSBtb2RlOwotfQotCi12b2lkIGdsTGlnaHRNb2RlbGYoR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgbGlnaHRNb2RlbHgocG5hbWUsIGdnbEZsb2F0VG9GaXhlZChwYXJhbSksIGMpOwotfQotCi12b2lkIGdsTGlnaHRNb2RlbHgoR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgbGlnaHRNb2RlbHgocG5hbWUsIHBhcmFtLCBjKTsKLX0KLQotdm9pZCBnbExpZ2h0TW9kZWxmdihHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmIChwbmFtZSA9PSBHTF9MSUdIVF9NT0RFTF9UV09fU0lERSkgewotICAgICAgICBsaWdodE1vZGVseChwbmFtZSwgZ2dsRmxvYXRUb0ZpeGVkKHBhcmFtc1swXSksIGMpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgaWYgKGdnbF91bmxpa2VseShwbmFtZSAhPSBHTF9MSUdIVF9NT0RFTF9BTUJJRU5UKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgYy0+bGlnaHRpbmcubGlnaHRNb2RlbC5hbWJpZW50LnIgPSBnZ2xGbG9hdFRvRml4ZWQocGFyYW1zWzBdKTsKLSAgICBjLT5saWdodGluZy5saWdodE1vZGVsLmFtYmllbnQuZyA9IGdnbEZsb2F0VG9GaXhlZChwYXJhbXNbMV0pOwotICAgIGMtPmxpZ2h0aW5nLmxpZ2h0TW9kZWwuYW1iaWVudC5iID0gZ2dsRmxvYXRUb0ZpeGVkKHBhcmFtc1syXSk7Ci0gICAgYy0+bGlnaHRpbmcubGlnaHRNb2RlbC5hbWJpZW50LmEgPSBnZ2xGbG9hdFRvRml4ZWQocGFyYW1zWzNdKTsKLSAgICBpbnZhbGlkYXRlX2xpZ2h0aW5nKGMpOwotfQotCi12b2lkIGdsTGlnaHRNb2RlbHh2KEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKHBuYW1lID09IEdMX0xJR0hUX01PREVMX1RXT19TSURFKSB7Ci0gICAgICAgIGxpZ2h0TW9kZWx4KHBuYW1lLCBwYXJhbXNbMF0sIGMpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgaWYgKGdnbF91bmxpa2VseShwbmFtZSAhPSBHTF9MSUdIVF9NT0RFTF9BTUJJRU5UKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgYy0+bGlnaHRpbmcubGlnaHRNb2RlbC5hbWJpZW50LnIgPSBwYXJhbXNbMF07Ci0gICAgYy0+bGlnaHRpbmcubGlnaHRNb2RlbC5hbWJpZW50LmcgPSBwYXJhbXNbMV07Ci0gICAgYy0+bGlnaHRpbmcubGlnaHRNb2RlbC5hbWJpZW50LmIgPSBwYXJhbXNbMl07Ci0gICAgYy0+bGlnaHRpbmcubGlnaHRNb2RlbC5hbWJpZW50LmEgPSBwYXJhbXNbM107Ci0gICAgaW52YWxpZGF0ZV9saWdodGluZyhjKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jZW5kaWYKLQotdm9pZCBnbExpZ2h0ZihHTGVudW0gaSwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgbGlnaHR4KGksIHBuYW1lLCBnZ2xGbG9hdFRvRml4ZWQocGFyYW0pLCBjKTsKLX0KLQotdm9pZCBnbExpZ2h0eChHTGVudW0gaSwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgbGlnaHR4KGksIHBuYW1lLCBwYXJhbSwgYyk7Ci19Ci0KLXZvaWQgZ2xMaWdodGZ2KEdMZW51bSBpLCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIHN3aXRjaCAocG5hbWUpIHsKLSAgICBjYXNlIEdMX1NQT1RfRVhQT05FTlQ6Ci0gICAgY2FzZSBHTF9TUE9UX0NVVE9GRjoKLSAgICBjYXNlIEdMX0NPTlNUQU5UX0FUVEVOVUFUSU9OOgotICAgIGNhc2UgR0xfTElORUFSX0FUVEVOVUFUSU9OOgotICAgIGNhc2UgR0xfUVVBRFJBVElDX0FUVEVOVUFUSU9OOgotICAgICAgICBsaWdodHgoaSwgcG5hbWUsIGdnbEZsb2F0VG9GaXhlZChwYXJhbXNbMF0pLCBjKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIEdMZml4ZWQgcGFyYW1zeFs0XTsKLSAgICBwYXJhbXN4WzBdID0gZ2dsRmxvYXRUb0ZpeGVkKHBhcmFtc1swXSk7Ci0gICAgcGFyYW1zeFsxXSA9IGdnbEZsb2F0VG9GaXhlZChwYXJhbXNbMV0pOwotICAgIHBhcmFtc3hbMl0gPSBnZ2xGbG9hdFRvRml4ZWQocGFyYW1zWzJdKTsKLSAgICBpZiAocG5hbWUgIT0gR0xfU1BPVF9ESVJFQ1RJT04pCi0gICAgICAgIHBhcmFtc3hbM10gPSBnZ2xGbG9hdFRvRml4ZWQocGFyYW1zWzNdKTsKLQotICAgIGxpZ2h0eHYoaSwgcG5hbWUsIHBhcmFtc3gsIGMpOwotfQotCi12b2lkIGdsTGlnaHR4dihHTGVudW0gaSwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBsaWdodHh2KGksIHBuYW1lLCBwYXJhbXMsIGMpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNlbmRpZgotCi12b2lkIGdsTWF0ZXJpYWxmKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBtYXRlcmlhbHgoZmFjZSwgcG5hbWUsIGdnbEZsb2F0VG9GaXhlZChwYXJhbSksIGMpOwotfQotCi12b2lkIGdsTWF0ZXJpYWx4KEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBtYXRlcmlhbHgoZmFjZSwgcG5hbWUsIHBhcmFtLCBjKTsKLX0KLQotdm9pZCBnbE1hdGVyaWFsZnYoCi0gICAgR0xlbnVtIGZhY2UsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKGdnbF91bmxpa2VseShmYWNlICE9IEdMX0ZST05UX0FORF9CQUNLKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIEdMZml4ZWQqIHdoYXQ9MDsKLSAgICBHTGZpeGVkKiBvdGhlcj0wOwotICAgIHN3aXRjaCAocG5hbWUpIHsKLSAgICBjYXNlIEdMX0FNQklFTlQ6ICAgIHdoYXQgPSBjLT5saWdodGluZy5mcm9udC5hbWJpZW50LnY7IGJyZWFrOwotICAgIGNhc2UgR0xfRElGRlVTRTogICAgd2hhdCA9IGMtPmxpZ2h0aW5nLmZyb250LmRpZmZ1c2UudjsgYnJlYWs7Ci0gICAgY2FzZSBHTF9TUEVDVUxBUjogICB3aGF0ID0gYy0+bGlnaHRpbmcuZnJvbnQuc3BlY3VsYXIudjsgYnJlYWs7Ci0gICAgY2FzZSBHTF9FTUlTU0lPTjogICB3aGF0ID0gYy0+bGlnaHRpbmcuZnJvbnQuZW1pc3Npb24udjsgYnJlYWs7Ci0gICAgY2FzZSBHTF9BTUJJRU5UX0FORF9ESUZGVVNFOgotICAgICAgICB3aGF0ICA9IGMtPmxpZ2h0aW5nLmZyb250LmFtYmllbnQudjsgYnJlYWs7Ci0gICAgICAgIG90aGVyID0gYy0+bGlnaHRpbmcuZnJvbnQuZGlmZnVzZS52OyBicmVhazsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9TSElOSU5FU1M6Ci0gICAgICAgIGMtPmxpZ2h0aW5nLmZyb250LnNoaW5pbmVzcyA9IGdnbEZsb2F0VG9GaXhlZChwYXJhbXNbMF0pOwotICAgICAgICBpbnZhbGlkYXRlX2xpZ2h0aW5nKGMpOwotICAgICAgICByZXR1cm47Ci0gICAgZGVmYXVsdDoKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICB3aGF0WzBdID0gZ2dsRmxvYXRUb0ZpeGVkKHBhcmFtc1swXSk7Ci0gICAgd2hhdFsxXSA9IGdnbEZsb2F0VG9GaXhlZChwYXJhbXNbMV0pOwotICAgIHdoYXRbMl0gPSBnZ2xGbG9hdFRvRml4ZWQocGFyYW1zWzJdKTsKLSAgICB3aGF0WzNdID0gZ2dsRmxvYXRUb0ZpeGVkKHBhcmFtc1szXSk7Ci0gICAgaWYgKG90aGVyKSB7Ci0gICAgICAgIG90aGVyWzBdID0gd2hhdFswXTsKLSAgICAgICAgb3RoZXJbMV0gPSB3aGF0WzFdOwotICAgICAgICBvdGhlclsyXSA9IHdoYXRbMl07Ci0gICAgICAgIG90aGVyWzNdID0gd2hhdFszXTsKLSAgICB9Ci0gICAgaW52YWxpZGF0ZV9saWdodGluZyhjKTsKLX0KLQotdm9pZCBnbE1hdGVyaWFseHYoCi0gICAgR0xlbnVtIGZhY2UsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKGdnbF91bmxpa2VseShmYWNlICE9IEdMX0ZST05UX0FORF9CQUNLKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIEdMZml4ZWQqIHdoYXQ9MDsKLSAgICBHTGZpeGVkKiBvdGhlcj0wOwotICAgIHN3aXRjaCAocG5hbWUpIHsKLSAgICBjYXNlIEdMX0FNQklFTlQ6ICAgIHdoYXQgPSBjLT5saWdodGluZy5mcm9udC5hbWJpZW50LnY7IGJyZWFrOwotICAgIGNhc2UgR0xfRElGRlVTRTogICAgd2hhdCA9IGMtPmxpZ2h0aW5nLmZyb250LmRpZmZ1c2UudjsgYnJlYWs7Ci0gICAgY2FzZSBHTF9TUEVDVUxBUjogICB3aGF0ID0gYy0+bGlnaHRpbmcuZnJvbnQuc3BlY3VsYXIudjsgYnJlYWs7Ci0gICAgY2FzZSBHTF9FTUlTU0lPTjogICB3aGF0ID0gYy0+bGlnaHRpbmcuZnJvbnQuZW1pc3Npb24udjsgYnJlYWs7Ci0gICAgY2FzZSBHTF9BTUJJRU5UX0FORF9ESUZGVVNFOgotICAgICAgICB3aGF0ID0gYy0+bGlnaHRpbmcuZnJvbnQuYW1iaWVudC52OyBicmVhazsKLSAgICAgICAgb3RoZXI9IGMtPmxpZ2h0aW5nLmZyb250LmRpZmZ1c2UudjsgYnJlYWs7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfU0hJTklORVNTOgotICAgICAgICBjLT5saWdodGluZy5mcm9udC5zaGluaW5lc3MgPSBnZ2xGbG9hdFRvRml4ZWQocGFyYW1zWzBdKTsKLSAgICAgICAgaW52YWxpZGF0ZV9saWdodGluZyhjKTsKLSAgICAgICAgcmV0dXJuOwotICAgIGRlZmF1bHQ6Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgd2hhdFswXSA9IHBhcmFtc1swXTsKLSAgICB3aGF0WzFdID0gcGFyYW1zWzFdOwotICAgIHdoYXRbMl0gPSBwYXJhbXNbMl07Ci0gICAgd2hhdFszXSA9IHBhcmFtc1szXTsKLSAgICBpZiAob3RoZXIpIHsKLSAgICAgICAgb3RoZXJbMF0gPSB3aGF0WzBdOwotICAgICAgICBvdGhlclsxXSA9IHdoYXRbMV07Ci0gICAgICAgIG90aGVyWzJdID0gd2hhdFsyXTsKLSAgICAgICAgb3RoZXJbM10gPSB3aGF0WzNdOwotICAgIH0KLSAgICBpbnZhbGlkYXRlX2xpZ2h0aW5nKGMpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBmb2cKLSNlbmRpZgotCi12b2lkIGdsRm9nZihHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIEdMZml4ZWQgcGFyYW14ID0gKEdMZml4ZWQpcGFyYW07Ci0gICAgaWYgKHBuYW1lICE9IEdMX0ZPR19NT0RFKQotICAgICAgICBwYXJhbXggPSBnZ2xGbG9hdFRvRml4ZWQocGFyYW0pOwotICAgIGZvZ3gocG5hbWUsIHBhcmFteCwgYyk7Ci19Ci0KLXZvaWQgZ2xGb2d4KEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSkgewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgZm9neChwbmFtZSwgcGFyYW0sIGMpOwotfQotCi12b2lkIGdsRm9nZnYoR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBpZiAocG5hbWUgIT0gR0xfRk9HX0NPTE9SKSB7Ci0gICAgICAgIEdMZml4ZWQgcGFyYW14ID0gKEdMZml4ZWQpcGFyYW1zWzBdOwotICAgICAgICBpZiAocG5hbWUgIT0gR0xfRk9HX01PREUpCi0gICAgICAgICAgICBwYXJhbXggPSBnZ2xGbG9hdFRvRml4ZWQocGFyYW1zWzBdKTsKLSAgICAgICAgZm9neChwbmFtZSwgcGFyYW14LCBjKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBHTGZpeGVkIHBhcmFtc3hbNF07Ci0gICAgcGFyYW1zeFswXSA9IGdnbEZsb2F0VG9GaXhlZChwYXJhbXNbMF0pOwotICAgIHBhcmFtc3hbMV0gPSBnZ2xGbG9hdFRvRml4ZWQocGFyYW1zWzFdKTsKLSAgICBwYXJhbXN4WzJdID0gZ2dsRmxvYXRUb0ZpeGVkKHBhcmFtc1syXSk7Ci0gICAgcGFyYW1zeFszXSA9IGdnbEZsb2F0VG9GaXhlZChwYXJhbXNbM10pOwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuZm9nQ29sb3IzeHYoYywgcGFyYW1zeCk7Ci19Ci0KLXZvaWQgZ2xGb2d4dihHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmIChwbmFtZSAhPSBHTF9GT0dfQ09MT1IpIHsKLSAgICAgICAgZm9neChwbmFtZSwgcGFyYW1zWzBdLCBjKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmZvZ0NvbG9yM3h2KGMsIHBhcmFtcyk7Ci19CmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL2xpZ2h0LmggYi9vcGVuZ2wvbGliYWdsL2xpZ2h0LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDZkYWUyNWYuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9saWdodC5oCisrKyAvZGV2L251bGwKQEAgLTEsMzggKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9saWdodC5oCi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2lmbmRlZiBBTkRST0lEX09QRU5HTEVTX0xJR0hUX0gKLSNkZWZpbmUgQU5EUk9JRF9PUEVOR0xFU19MSUdIVF9ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdGRkZWYuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1uYW1lc3BhY2UgZ2wgewotc3RydWN0IG9nbGVzX2NvbnRleHRfdDsKLX07Ci0KLXZvaWQgb2dsZXNfaW5pdF9saWdodChvZ2xlc19jb250ZXh0X3QqIGMpOwotdm9pZCBvZ2xlc191bmluaXRfbGlnaHQob2dsZXNfY29udGV4dF90KiBjKTsKLXZvaWQgb2dsZXNfaW52YWxpZGF0ZV9saWdodGluZ19tdnVpKG9nbGVzX2NvbnRleHRfdCogYyk7Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX09QRU5HTEVTX0xJR0hUX0gKLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC9tYXRyaXguY3BwIGIvb3BlbmdsL2xpYmFnbC9tYXRyaXguY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBmMTc1Y2RhLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJhZ2wvbWF0cml4LmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDExNDUgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9tYXRyaXguY3BwCi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0KLSNpbmNsdWRlICJjb250ZXh0LmgiCi0jaW5jbHVkZSAiZnAuaCIKLSNpbmNsdWRlICJzdGF0ZS5oIgotI2luY2x1ZGUgIm1hdHJpeC5oIgotI2luY2x1ZGUgInZlcnRleC5oIgotI2luY2x1ZGUgImxpZ2h0LmgiCi0KLSNpZiBkZWZpbmVkKF9fYXJtX18pICYmIGRlZmluZWQoX190aHVtYl9fKQotI3dhcm5pbmcgIm1hdHJpeC5jcHAgc2hvdWxkIG5vdCBiZSBjb21waWxlZCBpbiB0aHVtYiBvbiBBUk0uIgotI2VuZGlmCi0KLSNkZWZpbmUgSShfaSwgX2opICgoX2opKyA0KihfaSkpCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgY29uc3QgR0xmbG9hdCBnSWRlbnRpdHlmWzE2XSA9IHsgMSwwLDAsMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAwLDEsMCwwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsMCwxLDAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwwLDAsMSB9OwotCi1zdGF0aWMgY29uc3QgbWF0cml4eF90IGdJZGVudGl0eXggPSB7IAotICAgICAgICAgICAgeyAgIDB4MTAwMDAsMCwwLDAsCi0gICAgICAgICAgICAgICAgMCwweDEwMDAwLDAsMCwKLSAgICAgICAgICAgICAgICAwLDAsMHgxMDAwMCwwLAotICAgICAgICAgICAgICAgIDAsMCwwLDB4MTAwMDAKLSAgICAgICAgICAgIH0KLSAgICAgICAgfTsKLQotc3RhdGljIHZvaWQgcG9pbnQyX19ub3AodHJhbnNmb3JtX3QgY29uc3QqLCB2ZWM0X3QqIGMsIHZlYzRfdCBjb25zdCogbyk7Ci1zdGF0aWMgdm9pZCBwb2ludDNfX25vcCh0cmFuc2Zvcm1fdCBjb25zdCosIHZlYzRfdCogYywgdmVjNF90IGNvbnN0KiBvKTsKLXN0YXRpYyB2b2lkIHBvaW50NF9fbm9wKHRyYW5zZm9ybV90IGNvbnN0KiwgdmVjNF90KiBjLCB2ZWM0X3QgY29uc3QqIG8pOwotc3RhdGljIHZvaWQgbm9ybWFsX19ub3AodHJhbnNmb3JtX3QgY29uc3QqLCB2ZWM0X3QqIGMsIHZlYzRfdCBjb25zdCogbyk7Ci1zdGF0aWMgdm9pZCBwb2ludDJfX2dlbmVyaWModHJhbnNmb3JtX3QgY29uc3QqLCB2ZWM0X3QqIGMsIHZlYzRfdCBjb25zdCogbyk7Ci1zdGF0aWMgdm9pZCBwb2ludDNfX2dlbmVyaWModHJhbnNmb3JtX3QgY29uc3QqLCB2ZWM0X3QqIGMsIHZlYzRfdCBjb25zdCogbyk7Ci1zdGF0aWMgdm9pZCBwb2ludDRfX2dlbmVyaWModHJhbnNmb3JtX3QgY29uc3QqLCB2ZWM0X3QqIGMsIHZlYzRfdCBjb25zdCogbyk7Ci1zdGF0aWMgdm9pZCBub3JtYWxfX2dlbmVyaWModHJhbnNmb3JtX3QgY29uc3QqLCB2ZWM0X3QqIGMsIHZlYzRfdCBjb25zdCogbyk7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI2VuZGlmCi0KLXZvaWQgb2dsZXNfaW5pdF9tYXRyaXgob2dsZXNfY29udGV4dF90KiBjKQotewotICAgIGMtPnRyYW5zZm9ybXMubW9kZWx2aWV3LmluaXQoT0dMRVNfTU9ERUxWSUVXX1NUQUNLX0RFUFRIKTsKLSAgICBjLT50cmFuc2Zvcm1zLnByb2plY3Rpb24uaW5pdChPR0xFU19QUk9KRUNUSU9OX1NUQUNLX0RFUFRIKTsKLSAgICBmb3IgKGludCBpPTA7IGk8R0dMX1RFWFRVUkVfVU5JVF9DT1VOVCA7IGkrKykKLSAgICAgICAgYy0+dHJhbnNmb3Jtcy50ZXh0dXJlW2ldLmluaXQoT0dMRVNfVEVYVFVSRV9TVEFDS19ERVBUSCk7Ci0KLSAgICBjLT50cmFuc2Zvcm1zLmN1cnJlbnQgPSAmYy0+dHJhbnNmb3Jtcy5tb2RlbHZpZXc7Ci0gICAgYy0+dHJhbnNmb3Jtcy5tYXRyaXhNb2RlID0gR0xfTU9ERUxWSUVXOwotICAgIGMtPnRyYW5zZm9ybXMuZGlydHkgPSAgIHRyYW5zZm9ybV9zdGF0ZV90OjpWSUVXUE9SVCB8IAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRyYW5zZm9ybV9zdGF0ZV90OjpNVlVJIHwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICB0cmFuc2Zvcm1fc3RhdGVfdDo6TVZJVCB8Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgdHJhbnNmb3JtX3N0YXRlX3Q6Ok1WUDsKLSAgICBjLT50cmFuc2Zvcm1zLm12cC5sb2FkSWRlbnRpdHkoKTsKLSAgICBjLT50cmFuc2Zvcm1zLm12cDQubG9hZElkZW50aXR5KCk7Ci0gICAgYy0+dHJhbnNmb3Jtcy5tdml0NC5sb2FkSWRlbnRpdHkoKTsKLSAgICBjLT50cmFuc2Zvcm1zLm12dWkubG9hZElkZW50aXR5KCk7Ci0gICAgYy0+dHJhbnNmb3Jtcy52cHQubG9hZElkZW50aXR5KCk7Ci0gICAgYy0+dHJhbnNmb3Jtcy52cHQuek5lYXIgPSAwLjBmOwotICAgIGMtPnRyYW5zZm9ybXMudnB0LnpGYXIgID0gMS4wZjsKLX0KLQotdm9pZCBvZ2xlc191bmluaXRfbWF0cml4KG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBjLT50cmFuc2Zvcm1zLm1vZGVsdmlldy51bmluaXQoKTsKLSAgICBjLT50cmFuc2Zvcm1zLnByb2plY3Rpb24udW5pbml0KCk7Ci0gICAgZm9yIChpbnQgaT0wOyBpPEdHTF9URVhUVVJFX1VOSVRfQ09VTlQgOyBpKyspCi0gICAgICAgIGMtPnRyYW5zZm9ybXMudGV4dHVyZVtpXS51bmluaXQoKTsKLX0KLQotc3RhdGljIHZvaWQgdmFsaWRhdGVfcGVyc3BlY3RpdmUob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikKLXsKLSAgICBjb25zdCB1aW50MzJfdCBlbmFibGVzID0gYy0+cmFzdGVyaXplci5zdGF0ZS5lbmFibGVzOwotICAgIGMtPmFycmF5cy5wZXJzcGVjdGl2ZSA9IChjLT5jbGlwUGxhbmVzLmVuYWJsZSkgPwotICAgICAgICBvZ2xlc192ZXJ0ZXhfY2xpcEFsbFBlcnNwZWN0aXZlM0QgOiBvZ2xlc192ZXJ0ZXhfcGVyc3BlY3RpdmUzRDsKLSAgICBpZiAoZW5hYmxlcyAmIChHR0xfRU5BQkxFX0RFUFRIX1RFU1R8R0dMX0VOQUJMRV9GT0cpKSB7Ci0gICAgICAgIGMtPmFycmF5cy5wZXJzcGVjdGl2ZSA9IG9nbGVzX3ZlcnRleF9wZXJzcGVjdGl2ZTNEWjsKLSAgICAgICAgaWYgKGMtPmNsaXBQbGFuZXMuZW5hYmxlIHx8IChlbmFibGVzJkdHTF9FTkFCTEVfRk9HKSkKLSAgICAgICAgICAgIGMtPmFycmF5cy5wZXJzcGVjdGl2ZSA9IG9nbGVzX3ZlcnRleF9jbGlwQWxsUGVyc3BlY3RpdmUzRFo7Ci0gICAgfQotICAgIGlmICgoYy0+YXJyYXlzLnZlcnRleC5zaXplICE9IDQpICYmCi0gICAgICAgIChjLT50cmFuc2Zvcm1zLm12cDQuZmxhZ3MgJiB0cmFuc2Zvcm1fdDo6RkxBR1NfMkRfUFJPSkVDVElPTikpIHsKLSAgICAgICAgYy0+YXJyYXlzLnBlcnNwZWN0aXZlID0gb2dsZXNfdmVydGV4X3BlcnNwZWN0aXZlMkQ7Ci0gICAgfQotICAgIGMtPmFycmF5cy5wZXJzcGVjdGl2ZShjLCB2KTsKLX0KLQotdm9pZCBvZ2xlc19pbnZhbGlkYXRlX3BlcnNwZWN0aXZlKG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBjLT5hcnJheXMucGVyc3BlY3RpdmUgPSB2YWxpZGF0ZV9wZXJzcGVjdGl2ZTsKLX0KLQotdm9pZCBvZ2xlc192YWxpZGF0ZV90cmFuc2Zvcm1faW1wbChvZ2xlc19jb250ZXh0X3QqIGMsIHVpbnQzMl90IHdhbnQpCi17Ci0gICAgaW50IGRpcnR5ID0gYy0+dHJhbnNmb3Jtcy5kaXJ0eSAmIHdhbnQ7Ci0KLSAgICAvLyBWYWxpZGF0ZSB0aGUgbW9kZWx2aWV3Ci0gICAgaWYgKGRpcnR5ICYgdHJhbnNmb3JtX3N0YXRlX3Q6Ok1PREVMVklFVykgewotICAgICAgICBjLT50cmFuc2Zvcm1zLm1vZGVsdmlldy52YWxpZGF0ZSgpOwotICAgIH0KLQotICAgIC8vIFZhbGlkYXRlIHRoZSBwcm9qZWN0aW9uIHN0YWNrIChpbiBmYWN0LCBpdCdzIG5ldmVyIG5lZWRlZCkKLSAgICBpZiAoZGlydHkgJiB0cmFuc2Zvcm1fc3RhdGVfdDo6UFJPSkVDVElPTikgewotICAgICAgICBjLT50cmFuc2Zvcm1zLnByb2plY3Rpb24udmFsaWRhdGUoKTsKLSAgICB9Ci0KLSAgICAvLyBWYWxpZGF0ZSB0aGUgdmlld3BvcnQgdHJhbnNmb3JtYXRpb24KLSAgICBpZiAoZGlydHkgJiB0cmFuc2Zvcm1fc3RhdGVfdDo6VklFV1BPUlQpIHsKLSAgICAgICAgdnBfdHJhbnNmb3JtX3QmIHZwdCA9IGMtPnRyYW5zZm9ybXMudnB0OwotICAgICAgICB2cHQudHJhbnNmb3JtLm1hdHJpeC5sb2FkKHZwdC5tYXRyaXgpOwotICAgICAgICB2cHQudHJhbnNmb3JtLnBpY2tlcigpOwotICAgIH0KLQotICAgIC8vIFdlIG5lZWQgdG8gdXBkYXRlIHRoZSBtdnAgKHVzZWQgdG8gdHJhbnNmb3JtIGVhY2ggdmVydGV4KQotICAgIGlmIChkaXJ0eSAmIHRyYW5zZm9ybV9zdGF0ZV90OjpNVlApIHsKLSAgICAgICAgYy0+dHJhbnNmb3Jtcy51cGRhdGVfbXZwKCk7Ci0gICAgICAgIC8vIGludmFsaWRhdGUgcGVyc3BlY3RpdmUgKGRpdmlkZSBieSBXKSBhbmQgdmlldyB2b2x1bWUgY2xpcHBpbmcKLSAgICAgICAgb2dsZXNfaW52YWxpZGF0ZV9wZXJzcGVjdGl2ZShjKTsKLSAgICB9Ci0KLSAgICAvLyBWYWxpZGF0ZSB0aGUgbXZ1aSAoZm9yIG5vcm1hbCB0cmFuc2Zvcm1hdGlvbikKLSAgICBpZiAoZGlydHkgJiB0cmFuc2Zvcm1fc3RhdGVfdDo6TVZVSSkgewotICAgICAgICBjLT50cmFuc2Zvcm1zLnVwZGF0ZV9tdnVpKCk7Ci0gICAgICAgIG9nbGVzX2ludmFsaWRhdGVfbGlnaHRpbmdfbXZ1aShjKTsKLSAgICB9Ci0KLSAgICAvLyBWYWxpZGF0ZSB0aGUgdGV4dHVyZSBzdGFjawotICAgIGlmIChkaXJ0eSAmIHRyYW5zZm9ybV9zdGF0ZV90OjpURVhUVVJFKSB7Ci0gICAgICAgIGZvciAoaW50IGk9MDsgaTxHR0xfVEVYVFVSRV9VTklUX0NPVU5UIDsgaSsrKQotICAgICAgICAgICAgYy0+dHJhbnNmb3Jtcy50ZXh0dXJlW2ldLnZhbGlkYXRlKCk7Ci0gICAgfQotCi0gICAgLy8gVmFsaWRhdGUgdGhlIG12aXQ0ICh1c2VyLWNsaXAgcGxhbmVzKQotICAgIGlmIChkaXJ0eSAmIHRyYW5zZm9ybV9zdGF0ZV90OjpNVklUKSB7Ci0gICAgICAgIGMtPnRyYW5zZm9ybXMudXBkYXRlX212aXQoKTsKLSAgICB9Ci0KLSAgICBjLT50cmFuc2Zvcm1zLmRpcnR5ICY9IH53YW50OwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayB0cmFuc2Zvcm1fdAotI2VuZGlmCi0KLXZvaWQgdHJhbnNmb3JtX3Q6OmxvYWRJZGVudGl0eSgpIHsKLSAgICBtYXRyaXggPSBnSWRlbnRpdHl4OwotICAgIGZsYWdzID0gMDsKLSAgICBvcHMgPSBPUF9JREVOVElUWTsKLSAgICBwb2ludDIgPSBwb2ludDJfX25vcDsKLSAgICBwb2ludDMgPSBwb2ludDNfX25vcDsKLSAgICBwb2ludDQgPSBwb2ludDRfX25vcDsKLX0KLQotCi1zdGF0aWMgaW5saW5lCi1pbnQgbm90WmVybyhHTGZpeGVkIHYpIHsKLSAgICByZXR1cm4gYWJzKHYpICYgfjB4MzsKLX0KLQotc3RhdGljIGlubGluZQotaW50IG5vdE9uZShHTGZpeGVkIHYpIHsKLSAgICByZXR1cm4gbm90WmVybyh2IC0gMHgxMDAwMCk7Ci19Ci0KLXZvaWQgdHJhbnNmb3JtX3Q6OnBpY2tlcigpCi17Ci0gICAgY29uc3QgR0xmaXhlZCogY29uc3QgbSA9IG1hdHJpeC5tOwotCi0gICAgLy8gWFhYOiBwaWNrZXIgbmVlZHMgdG8gYmUgc21hcnRlcgotICAgIGZsYWdzID0gMDsKLSAgICBvcHMgPSBPUF9BTEw7Ci0gICAgcG9pbnQyID0gcG9pbnQyX19nZW5lcmljOwotICAgIHBvaW50MyA9IHBvaW50M19fZ2VuZXJpYzsKLSAgICBwb2ludDQgPSBwb2ludDRfX2dlbmVyaWM7Ci0gICAgCi0gICAgLy8gZmluZCBvdXQgaWYgdGhpcyBpcyBhIDJEIHByb2plY3Rpb24KLSAgICBpZiAoIShub3RaZXJvKG1bM10pIHwgbm90WmVybyhtWzddKSB8IG5vdFplcm8obVsxMV0pIHwgbm90T25lKG1bMTVdKSkpIHsKLSAgICAgICAgZmxhZ3MgfD0gRkxBR1NfMkRfUFJPSkVDVElPTjsKLSAgICB9Ci19Ci0KLXZvaWQgbXZ1aV90cmFuc2Zvcm1fdDo6cGlja2VyKCkKLXsKLSAgICBmbGFncyA9IDA7Ci0gICAgb3BzID0gT1BfQUxMOwotICAgIHBvaW50MyA9IG5vcm1hbF9fZ2VuZXJpYzsKLX0KLQotdm9pZCB0cmFuc2Zvcm1fdDo6ZHVtcChjb25zdCBjaGFyKiB3aGF0KQotewotICAgIEdMZml4ZWQgY29uc3QgKiBjb25zdCBtID0gbWF0cml4Lm07Ci0gICAgTE9HRCgiJXM6Iiwgd2hhdCk7Ci0gICAgZm9yIChpbnQgaT0wIDsgaTw0IDsgaSsrKQotICAgICAgICBMT0dEKCJbJTA4eCAlMDh4ICUwOHggJTA4eF0gWyVmICVmICVmICVmXVxuIiwKLSAgICAgICAgICAgIG1bSSgwLGkpXSwgbVtJKDEsaSldLCBtW0koMixpKV0sIG1bSSgzLGkpXSwKLSAgICAgICAgICAgIGZpeGVkVG9GbG9hdChtW0koMCxpKV0pLAotICAgICAgICAgICAgZml4ZWRUb0Zsb2F0KG1bSSgxLGkpXSksIAotICAgICAgICAgICAgZml4ZWRUb0Zsb2F0KG1bSSgyLGkpXSksCi0gICAgICAgICAgICBmaXhlZFRvRmxvYXQobVtJKDMsaSldKSk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIG1hdHJpeHhfdAotI2VuZGlmCi0KLXZvaWQgbWF0cml4eF90Ojpsb2FkKGNvbnN0IG1hdHJpeGZfdCYgcmhzKSB7Ci0gICAgR0xmaXhlZCogeHAgPSBtOwotICAgIEdMZmxvYXQgY29uc3QqIGZwID0gcmhzLmVsZW1lbnRzKCk7Ci0gICAgdW5zaWduZWQgaW50IGkgPSAxNjsKLSAgICBkbyB7Ci0gICAgICAgIGNvbnN0IEdMZmxvYXQgZiA9ICpmcCsrOwotICAgICAgICAqeHArKyA9IGlzWmVyb2YoZikgPyAwIDogZ2dsRmxvYXRUb0ZpeGVkKGYpOwotICAgIH0gd2hpbGUgKC0taSk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIG1hdHJpeGZfdAotI2VuZGlmCi0KLXZvaWQgbWF0cml4Zl90OjptdWx0aXBseShtYXRyaXhmX3QmIHIsIGNvbnN0IG1hdHJpeGZfdCYgbGhzLCBjb25zdCBtYXRyaXhmX3QmIHJocykKLXsKLSAgICBHTGZsb2F0IGNvbnN0KiBjb25zdCBtID0gbGhzLm07Ci0gICAgZm9yIChpbnQgaT0wIDsgaTw0IDsgaSsrKSB7Ci0gICAgICAgIHJlZ2lzdGVyIGNvbnN0IGZsb2F0IHJoc19pMCA9IHJocy5tWyBJKGksMCkgXTsKLSAgICAgICAgcmVnaXN0ZXIgZmxvYXQgcmkwID0gbVsgSSgwLDApIF0gKiByaHNfaTA7Ci0gICAgICAgIHJlZ2lzdGVyIGZsb2F0IHJpMSA9IG1bIEkoMCwxKSBdICogcmhzX2kwOwotICAgICAgICByZWdpc3RlciBmbG9hdCByaTIgPSBtWyBJKDAsMikgXSAqIHJoc19pMDsKLSAgICAgICAgcmVnaXN0ZXIgZmxvYXQgcmkzID0gbVsgSSgwLDMpIF0gKiByaHNfaTA7Ci0gICAgICAgIGZvciAoaW50IGo9MSA7IGo8NCA7IGorKykgewotICAgICAgICAgICAgcmVnaXN0ZXIgY29uc3QgZmxvYXQgcmhzX2lqID0gcmhzLm1bIEkoaSxqKSBdOwotICAgICAgICAgICAgcmkwICs9IG1bIEkoaiwwKSBdICogcmhzX2lqOwotICAgICAgICAgICAgcmkxICs9IG1bIEkoaiwxKSBdICogcmhzX2lqOwotICAgICAgICAgICAgcmkyICs9IG1bIEkoaiwyKSBdICogcmhzX2lqOwotICAgICAgICAgICAgcmkzICs9IG1bIEkoaiwzKSBdICogcmhzX2lqOwotICAgICAgICB9Ci0gICAgICAgIHIubVsgSShpLDApIF0gPSByaTA7Ci0gICAgICAgIHIubVsgSShpLDEpIF0gPSByaTE7Ci0gICAgICAgIHIubVsgSShpLDIpIF0gPSByaTI7Ci0gICAgICAgIHIubVsgSShpLDMpIF0gPSByaTM7Ci0gICAgfQotfQotCi12b2lkIG1hdHJpeGZfdDo6ZHVtcChjb25zdCBjaGFyKiB3aGF0KSB7Ci0gICAgTE9HRCgiJXMiLCB3aGF0KTsKLSAgICBMT0dEKCJbICU5ZiAlOWYgJTlmICU5ZiBdIiwgbVtJKDAsMCldLCBtW0koMSwwKV0sIG1bSSgyLDApXSwgbVtJKDMsMCldKTsKLSAgICBMT0dEKCJbICU5ZiAlOWYgJTlmICU5ZiBdIiwgbVtJKDAsMSldLCBtW0koMSwxKV0sIG1bSSgyLDEpXSwgbVtJKDMsMSldKTsKLSAgICBMT0dEKCJbICU5ZiAlOWYgJTlmICU5ZiBdIiwgbVtJKDAsMildLCBtW0koMSwyKV0sIG1bSSgyLDIpXSwgbVtJKDMsMildKTsKLSAgICBMT0dEKCJbICU5ZiAlOWYgJTlmICU5ZiBdIiwgbVtJKDAsMyldLCBtW0koMSwzKV0sIG1bSSgyLDMpXSwgbVtJKDMsMyldKTsKLX0KLQotdm9pZCBtYXRyaXhmX3Q6OmxvYWRJZGVudGl0eSgpIHsKLSAgICBtZW1jcHkobSwgZ0lkZW50aXR5Ziwgc2l6ZW9mKG0pKTsKLX0KLQotdm9pZCBtYXRyaXhmX3Q6OnNldChjb25zdCBHTGZpeGVkKiByaHMpIHsKLSAgICBsb2FkKHJocyk7Ci19Ci0KLXZvaWQgbWF0cml4Zl90OjpzZXQoY29uc3QgR0xmbG9hdCogcmhzKSB7Ci0gICAgbG9hZChyaHMpOwotfQotCi12b2lkIG1hdHJpeGZfdDo6bG9hZChjb25zdCBHTGZpeGVkKiByaHMpIHsKLSAgICBHTGZsb2F0KiBmcCA9IG07Ci0gICAgdW5zaWduZWQgaW50IGkgPSAxNjsKLSAgICBkbyB7Ci0gICAgICAgICpmcCsrID0gZml4ZWRUb0Zsb2F0KCpyaHMrKyk7Ci0gICAgfSB3aGlsZSAoLS1pKTsKLX0KLQotdm9pZCBtYXRyaXhmX3Q6OmxvYWQoY29uc3QgR0xmbG9hdCogcmhzKSB7Ci0gICAgbWVtY3B5KG0sIHJocywgc2l6ZW9mKG0pKTsKLX0KLQotdm9pZCBtYXRyaXhmX3Q6OmxvYWQoY29uc3QgbWF0cml4Zl90JiByaHMpIHsKLSAgICBvcGVyYXRvciA9IChyaHMpOwotfQotCi12b2lkIG1hdHJpeGZfdDo6bXVsdGlwbHkoY29uc3QgbWF0cml4Zl90JiByaHMpIHsKLSAgICBtYXRyaXhmX3QgcjsKLSAgICBtdWx0aXBseShyLCAqdGhpcywgcmhzKTsKLSAgICBvcGVyYXRvciA9IChyKTsKLX0KLQotdm9pZCBtYXRyaXhmX3Q6OnRyYW5zbGF0ZShHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KSB7Ci0gICAgZm9yIChpbnQgaT0wIDsgaTw0IDsgaSsrKSB7Ci0gICAgICAgIG1bMTIraV0gKz0gbVtpXSp4ICsgbVs0K2ldKnkgKyBtWzgraV0qejsKLSAgICB9Ci19Ci0KLXZvaWQgbWF0cml4Zl90OjpzY2FsZShHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KSB7Ci0gICAgZm9yIChpbnQgaT0wIDsgaTw0IDsgaSsrKSB7Ci0gICAgICAgIG1bICBpXSAqPSB4OwotICAgICAgICBtWzQraV0gKj0geTsKLSAgICAgICAgbVs4K2ldICo9IHo7Ci0gICAgfQotfQotCi12b2lkIG1hdHJpeGZfdDo6cm90YXRlKEdMZmxvYXQgYSwgR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeikKLXsKLSAgICBtYXRyaXhmX3Qgcm90YXRpb247Ci0gICAgR0xmbG9hdCogciA9IHJvdGF0aW9uLm07Ci0gICAgR0xmbG9hdCBjLCBzOwotICAgIHJbM10gPSAwOyAgIHJbN10gPSAwOyAgIHJbMTFdPSAwOwotICAgIHJbMTJdPSAwOyAgIHJbMTNdPSAwOyAgIHJbMTRdPSAwOyAgIHJbMTVdPSAxOwotICAgIGEgKj0gR0xmbG9hdChNX1BJIC8gMTgwLjBmKTsKLSAgICBzaW5jb3NmKGEsICZzLCAmYyk7Ci0gICAgaWYgKGlzT25lZih4KSAmJiBpc1plcm9mKHkpICYmIGlzWmVyb2YoeikpIHsKLSAgICAgICAgcls1XSA9IGM7ICAgclsxMF09IGM7Ci0gICAgICAgIHJbNl0gPSBzOyAgIHJbOV0gPSAtczsKLSAgICAgICAgclsxXSA9IDA7ICAgclsyXSA9IDA7Ci0gICAgICAgIHJbNF0gPSAwOyAgIHJbOF0gPSAwOwotICAgICAgICByWzBdID0gMTsKLSAgICB9IGVsc2UgaWYgKGlzWmVyb2YoeCkgJiYgaXNPbmVmKHkpICYmIGlzWmVyb2YoeikpIHsKLSAgICAgICAgclswXSA9IGM7ICAgclsxMF09IGM7Ci0gICAgICAgIHJbOF0gPSBzOyAgIHJbMl0gPSAtczsKLSAgICAgICAgclsxXSA9IDA7ICAgcls0XSA9IDA7Ci0gICAgICAgIHJbNl0gPSAwOyAgIHJbOV0gPSAwOwotICAgICAgICByWzVdID0gMTsKLSAgICB9IGVsc2UgaWYgKGlzWmVyb2YoeCkgJiYgaXNaZXJvZih5KSAmJiBpc09uZWYoeikpIHsKLSAgICAgICAgclswXSA9IGM7ICAgcls1XSA9IGM7Ci0gICAgICAgIHJbMV0gPSBzOyAgIHJbNF0gPSAtczsKLSAgICAgICAgclsyXSA9IDA7ICAgcls2XSA9IDA7Ci0gICAgICAgIHJbOF0gPSAwOyAgIHJbOV0gPSAwOwotICAgICAgICByWzEwXT0gMTsKLSAgICB9IGVsc2UgewotICAgICAgICBjb25zdCBHTGZsb2F0IGxlbiA9IHNxcnRmKHgqeCArIHkqeSArIHoqeik7Ci0gICAgICAgIGlmICghaXNPbmVmKGxlbikpIHsKLSAgICAgICAgICAgIGNvbnN0IEdMZmxvYXQgcmVjaXBMZW4gPSByZWNpcHJvY2FsZihsZW4pOwotICAgICAgICAgICAgeCAqPSByZWNpcExlbjsKLSAgICAgICAgICAgIHkgKj0gcmVjaXBMZW47Ci0gICAgICAgICAgICB6ICo9IHJlY2lwTGVuOwotICAgICAgICB9Ci0gICAgICAgIGNvbnN0IEdMZmxvYXQgbmMgPSAxLjBmIC0gYzsKLSAgICAgICAgY29uc3QgR0xmbG9hdCB4eSA9IHggKiB5OwotICAgICAgICBjb25zdCBHTGZsb2F0IHl6ID0geSAqIHo7Ci0gICAgICAgIGNvbnN0IEdMZmxvYXQgenggPSB6ICogeDsKLSAgICAgICAgY29uc3QgR0xmbG9hdCB4cyA9IHggKiBzOwotICAgICAgICBjb25zdCBHTGZsb2F0IHlzID0geSAqIHM7Ci0gICAgICAgIGNvbnN0IEdMZmxvYXQgenMgPSB6ICogczsJCQotICAgICAgICByWyAwXSA9IHgqeCpuYyArICBjOyAgICByWyA0XSA9ICB4eSpuYyAtIHpzOyAgICByWyA4XSA9ICB6eCpuYyArIHlzOwotICAgICAgICByWyAxXSA9ICB4eSpuYyArIHpzOyAgICByWyA1XSA9IHkqeSpuYyArICBjOyAgICByWyA5XSA9ICB5eipuYyAtIHhzOwotICAgICAgICByWyAyXSA9ICB6eCpuYyAtIHlzOyAgICByWyA2XSA9ICB5eipuYyArIHhzOyAgICByWzEwXSA9IHoqeipuYyArICBjOwotICAgIH0KLSAgICBtdWx0aXBseShyb3RhdGlvbik7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIG1hdHJpeF9zdGFja190Ci0jZW5kaWYKLQotdm9pZCBtYXRyaXhfc3RhY2tfdDo6aW5pdChpbnQgZGVwdGgpIHsKLSAgICBzdGFjayA9IG5ldyBtYXRyaXhmX3RbZGVwdGhdOwotICAgIG9wcyA9IG5ldyB1aW50OF90W2RlcHRoXTsKLSAgICBtYXhEZXB0aCA9IGRlcHRoOwotICAgIGRlcHRoID0gMDsKLSAgICBkaXJ0eSA9IDA7Ci0gICAgbG9hZElkZW50aXR5KCk7Ci19Ci0KLXZvaWQgbWF0cml4X3N0YWNrX3Q6OnVuaW5pdCgpIHsKLSAgICBkZWxldGUgW10gc3RhY2s7Ci0gICAgZGVsZXRlIFtdIG9wczsKLX0KLQotdm9pZCBtYXRyaXhfc3RhY2tfdDo6bG9hZElkZW50aXR5KCkgewotICAgIHRyYW5zZm9ybS5sb2FkSWRlbnRpdHkoKTsKLSAgICBzdGFja1tkZXB0aF0ubG9hZElkZW50aXR5KCk7Ci0gICAgb3BzW2RlcHRoXSA9IE9QX0lERU5USVRZOwotfQotCi12b2lkIG1hdHJpeF9zdGFja190Ojpsb2FkKGNvbnN0IEdMZml4ZWQqIHJocykKLXsgICAKLSAgICBtZW1jcHkodHJhbnNmb3JtLm1hdHJpeC5tLCByaHMsIHNpemVvZih0cmFuc2Zvcm0ubWF0cml4Lm0pKTsKLSAgICBzdGFja1tkZXB0aF0ubG9hZChyaHMpOwotICAgIG9wc1tkZXB0aF0gPSBPUF9BTEw7ICAgIC8vIFRPRE86IHdlIHNob3VsZCBsb29rIGF0IHRoZSBtYXRyaXgKLX0KLQotdm9pZCBtYXRyaXhfc3RhY2tfdDo6bG9hZChjb25zdCBHTGZsb2F0KiByaHMpCi17Ci0gICAgc3RhY2tbZGVwdGhdLmxvYWQocmhzKTsKLSAgICBvcHNbZGVwdGhdID0gT1BfQUxMOyAgICAvLyBUT0RPOiB3ZSBzaG91bGQgbG9vayBhdCB0aGUgbWF0cml4Ci19Ci0KLXZvaWQgbWF0cml4X3N0YWNrX3Q6Om11bHRpcGx5KGNvbnN0IG1hdHJpeGZfdCYgcmhzKQoteyAgICAKLSAgICBzdGFja1tkZXB0aF0ubXVsdGlwbHkocmhzKTsKLSAgICBvcHNbZGVwdGhdID0gT1BfQUxMOyAgICAvLyBUT0RPOiB3ZSBzaG91bGQgbG9vayBhdCB0aGUgbWF0cml4Ci19Ci0KLXZvaWQgbWF0cml4X3N0YWNrX3Q6OnRyYW5zbGF0ZShHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KQotewotICAgIHN0YWNrW2RlcHRoXS50cmFuc2xhdGUoeCx5LHopOwotICAgIG9wc1tkZXB0aF0gfD0gT1BfVFJBTlNMQVRFOwotfQotCi12b2lkIG1hdHJpeF9zdGFja190OjpzY2FsZShHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KQotewotICAgIHN0YWNrW2RlcHRoXS5zY2FsZSh4LHkseik7Ci0gICAgaWYgKHg9PXkgJiYgeT09eikgewotICAgICAgICBvcHNbZGVwdGhdIHw9IE9QX1VOSUZPUk1fU0NBTEU7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgb3BzW2RlcHRoXSB8PSBPUF9TQ0FMRTsKLSAgICB9Ci19Ci0KLXZvaWQgbWF0cml4X3N0YWNrX3Q6OnJvdGF0ZShHTGZsb2F0IGEsIEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopCi17Ci0gICAgc3RhY2tbZGVwdGhdLnJvdGF0ZShhLHgseSx6KTsKLSAgICBvcHNbZGVwdGhdIHw9IE9QX1JPVEFURTsKLX0KLQotdm9pZCBtYXRyaXhfc3RhY2tfdDo6dmFsaWRhdGUoKQotewotICAgIGlmIChkaXJ0eSAmIERPX0ZMT0FUX1RPX0ZJWEVEKSB7Ci0gICAgICAgIHRyYW5zZm9ybS5tYXRyaXgubG9hZCh0b3AoKSk7Ci0gICAgfQotICAgIGlmIChkaXJ0eSAmIERPX1BJQ0tFUikgewotICAgICAgICB0cmFuc2Zvcm0ucGlja2VyKCk7Ci0gICAgfQotICAgIGRpcnR5ID0gMDsKLX0KLQotR0xpbnQgbWF0cml4X3N0YWNrX3Q6OnB1c2goKQotewotICAgIGlmIChkZXB0aCA+PSAobWF4RGVwdGgtMSkpIHsKLSAgICAgICAgcmV0dXJuIEdMX1NUQUNLX09WRVJGTE9XOwotICAgIH0KLSAgICBzdGFja1tkZXB0aCsxXSA9IHN0YWNrW2RlcHRoXTsKLSAgICBvcHNbZGVwdGgrMV0gPSBvcHNbZGVwdGhdOwotICAgIGRlcHRoKys7Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLUdMaW50IG1hdHJpeF9zdGFja190Ojpwb3AoKQotewotICAgIGlmIChkZXB0aCA9PSAwKSB7Ci0gICAgICAgIHJldHVybiBHTF9TVEFDS19VTkRFUkZMT1c7Ci0gICAgfQotICAgIGRlcHRoLS07Ci0gICAgcmV0dXJuIDA7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIHZwX3RyYW5zZm9ybV90Ci0jZW5kaWYKLQotdm9pZCB2cF90cmFuc2Zvcm1fdDo6bG9hZElkZW50aXR5KCkgewotICAgIHRyYW5zZm9ybS5sb2FkSWRlbnRpdHkoKTsKLSAgICBtYXRyaXgubG9hZElkZW50aXR5KCk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIHRyYW5zZm9ybV9zdGF0ZV90Ci0jZW5kaWYKLQotdm9pZCB0cmFuc2Zvcm1fc3RhdGVfdDo6aW52YWxpZGF0ZSgpCi17Ci0gICAgc3dpdGNoIChtYXRyaXhNb2RlKSB7Ci0gICAgY2FzZSBHTF9NT0RFTFZJRVc6ICBkaXJ0eSB8PSBNT0RFTFZJRVcgIHwgTVZQIHwgTVZVSSB8IE1WSVQ7ICAgIGJyZWFrOwotICAgIGNhc2UgR0xfUFJPSkVDVElPTjogZGlydHkgfD0gUFJPSkVDVElPTiB8IE1WUDsgICAgICAgICAgICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1RFWFRVUkU6ICAgIGRpcnR5IHw9IFRFWFRVUkUgICAgfCBNVlA7ICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgfQotICAgIGN1cnJlbnQtPmRpcnR5ID0gICAgbWF0cml4X3N0YWNrX3Q6OkRPX1BJQ0tFUiB8Ci0gICAgICAgICAgICAgICAgICAgICAgICBtYXRyaXhfc3RhY2tfdDo6RE9fRkxPQVRfVE9fRklYRUQ7Ci19Ci0KLXZvaWQgdHJhbnNmb3JtX3N0YXRlX3Q6OnVwZGF0ZV9tdnAoKQotewotICAgIG1hdHJpeGZfdCB0ZW1wX212cDsKLSAgICBtYXRyaXhmX3Q6Om11bHRpcGx5KHRlbXBfbXZwLCBwcm9qZWN0aW9uLnRvcCgpLCBtb2RlbHZpZXcudG9wKCkpOwotICAgIG12cDQubWF0cml4LmxvYWQodGVtcF9tdnApOwotICAgIG12cDQucGlja2VyKCk7Ci0KLSAgICBpZiAobXZwNC5mbGFncyAmIHRyYW5zZm9ybV90OjpGTEFHU18yRF9QUk9KRUNUSU9OKSB7Ci0gICAgICAgIC8vIHRoZSBtdnAgbWF0cml4IGRvZXNuJ3QgdHJhbnNmb3JtIFcsIGluIHRoaXMgY2FzZSB3ZSBjYW4KLSAgICAgICAgLy8gcHJlbXVsdGlwbHkgaXQgd2l0aCB0aGUgdmlld3BvcnQgdHJhbnNmb3JtYXRpb24uIEluIGFkZGl0aW9uIHRvCi0gICAgICAgIC8vIGJlaW5nIG1vcmUgZWZmaWNpZW50LCB0aGlzIGlzIGFsc28gbXVjaCBtb3JlIGFjY3VyYXRlIGFuZCBpbiBmYWN0Ci0gICAgICAgIC8vIGlzIG5lZWRlZCBmb3IgMkQgZHJhd2luZyB3aXRoIGEgcmVzdWx0aW5nIDE6MSBtYXBwaW5nLgotICAgICAgICBtYXRyaXhmX3QgbXZwdjsKLSAgICAgICAgbWF0cml4Zl90OjptdWx0aXBseShtdnB2LCB2cHQubWF0cml4LCB0ZW1wX212cCk7Ci0gICAgICAgIG12cC5tYXRyaXgubG9hZChtdnB2KTsKLSAgICAgICAgbXZwLnBpY2tlcigpOwotICAgIH0gZWxzZSB7Ci0gICAgICAgIG12cCA9IG12cDQ7Ci0gICAgfQotfQotCi1zdGF0aWMgaW5saW5lIAotR0xmbG9hdCBkZXQyMihHTGZsb2F0IGEsIEdMZmxvYXQgYiwgR0xmbG9hdCBjLCBHTGZsb2F0IGQpIHsKLSAgICByZXR1cm4gYSpkIC0gYipjOwotfQotCi1zdGF0aWMgaW5saW5lCi1HTGZsb2F0IG5kZXQyMihHTGZsb2F0IGEsIEdMZmxvYXQgYiwgR0xmbG9hdCBjLCBHTGZsb2F0IGQpIHsKLSAgICByZXR1cm4gYipjIC0gYSpkOwotfQotCi1zdGF0aWMgX19hdHRyaWJ1dGVfXygobm9pbmxpbmUpKQotdm9pZCBpbnZlcnQoR0xmbG9hdCogaW52ZXJzZSwgY29uc3QgR0xmbG9hdCogc3JjKQotewotICAgIGRvdWJsZSB0OwotICAgIGludCBpLCBqLCBrLCBzd2FwOwotICAgIEdMZmxvYXQgdG1wWzRdWzRdOwotICAgIAotICAgIG1lbWNweShpbnZlcnNlLCBnSWRlbnRpdHlmLCBzaXplb2YoZ0lkZW50aXR5ZikpOwotICAgIG1lbWNweSh0bXAsIHNyYywgc2l6ZW9mKEdMZmxvYXQpKjE2KTsKLSAgICAKLSAgICBmb3IgKGkgPSAwOyBpIDwgNDsgaSsrKSB7Ci0gICAgICAgIC8vIGxvb2sgZm9yIGxhcmdlc3QgZWxlbWVudCBpbiBjb2x1bW4KLSAgICAgICAgc3dhcCA9IGk7Ci0gICAgICAgIGZvciAoaiA9IGkgKyAxOyBqIDwgNDsgaisrKSB7Ci0gICAgICAgICAgICBpZiAoZmFicyh0bXBbal1baV0pID4gZmFicyh0bXBbaV1baV0pKSB7Ci0gICAgICAgICAgICAgICAgc3dhcCA9IGo7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgCi0gICAgICAgIGlmIChzd2FwICE9IGkpIHsKLSAgICAgICAgICAgIC8qIHN3YXAgcm93cy4gKi8KLSAgICAgICAgICAgIGZvciAoayA9IDA7IGsgPCA0OyBrKyspIHsKLSAgICAgICAgICAgICAgICB0ID0gdG1wW2ldW2tdOwotICAgICAgICAgICAgICAgIHRtcFtpXVtrXSA9IHRtcFtzd2FwXVtrXTsKLSAgICAgICAgICAgICAgICB0bXBbc3dhcF1ba10gPSB0OwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgIHQgPSBpbnZlcnNlW2kqNCtrXTsKLSAgICAgICAgICAgICAgICBpbnZlcnNlW2kqNCtrXSA9IGludmVyc2Vbc3dhcCo0K2tdOwotICAgICAgICAgICAgICAgIGludmVyc2Vbc3dhcCo0K2tdID0gdDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgdCA9IDEuMGYgLyB0bXBbaV1baV07Ci0gICAgICAgIGZvciAoayA9IDA7IGsgPCA0OyBrKyspIHsKLSAgICAgICAgICAgIHRtcFtpXVtrXSAqPSB0OwotICAgICAgICAgICAgaW52ZXJzZVtpKjQra10gKj0gdDsKLSAgICAgICAgfQotICAgICAgICBmb3IgKGogPSAwOyBqIDwgNDsgaisrKSB7Ci0gICAgICAgICAgICBpZiAoaiAhPSBpKSB7Ci0gICAgICAgICAgICAgICAgdCA9IHRtcFtqXVtpXTsKLSAgICAgICAgICAgICAgICBmb3IgKGsgPSAwOyBrIDwgNDsgaysrKSB7Ci0gICAgICAgICAgICAgICAgICAgIHRtcFtqXVtrXSAtPSB0bXBbaV1ba10qdDsKLSAgICAgICAgICAgICAgICAgICAgaW52ZXJzZVtqKjQra10gLT0gaW52ZXJzZVtpKjQra10qdDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLXZvaWQgdHJhbnNmb3JtX3N0YXRlX3Q6OnVwZGF0ZV9tdml0KCkKLXsKLSAgICBHTGZsb2F0IHJbMTZdOwotICAgIGNvbnN0IEdMZmxvYXQqIGNvbnN0IG12ID0gbW9kZWx2aWV3LnRvcCgpLmVsZW1lbnRzKCk7Ci0gICAgaW52ZXJ0KHIsIG12KTsKLSAgICAvLyBjb252ZXJ0IHRvIGZpeGVkLXBvaW50IGFuZCB0cmFuc3Bvc2UKLSAgICBHTGZpeGVkKiBjb25zdCB4ID0gbXZpdDQubWF0cml4Lm07Ci0gICAgZm9yIChpbnQgaT0wIDsgaTw0IDsgaSsrKQotICAgICAgICBmb3IgKGludCBqPTAgOyBqPDQgOyBqKyspCi0gICAgICAgICAgICB4W0koaSxqKV0gPSBnZ2xGbG9hdFRvRml4ZWQocltJKGosaSldKTsKLSAgICBtdml0NC5waWNrZXIoKTsKLX0KLQotdm9pZCB0cmFuc2Zvcm1fc3RhdGVfdDo6dXBkYXRlX212dWkoKQotewotICAgIGNvbnN0IEdMZmxvYXQqIGNvbnN0IG12ID0gbW9kZWx2aWV3LnRvcCgpLmVsZW1lbnRzKCk7Ci0KLSAgICAvKgotICAgIFdoZW4gdHJhbnNmb3JtaW5nIG5vcm1hbHMsIHdlIGNhbiB1c2UgdGhlIHVwcGVyIDN4MyBtYXRyaXgsIHNlZToKLSAgICBodHRwOi8vd3d3Lm9wZW5nbC5vcmcvZG9jdW1lbnRhdGlvbi9zcGVjcy92ZXJzaW9uMS4xL2dsc3BlYzEuMS9ub2RlMjYuaHRtbAotICAgICovCi0gICAgCi0gICAgLy8gQWxzbyBub3RlIHRoYXQ6Ci0gICAgLy8gICAgICBsKG9iaikgPSAgdHIoTSkubChleWUpIGZvciBpbmZpbml0ZSBsaWdodAotICAgIC8vICAgICAgbChvYmopID0gaW52KE0pLmwoZXllKSBmb3IgbG9jYWwgbGlnaHQKLQotICAgIGNvbnN0IHVpbnQzMl90IG9wcyA9IG1vZGVsdmlldy50b3Bfb3BzKCkgJiB+T1BfVFJBTlNMQVRFOwotICAgIGlmIChnZ2xfbGlrZWx5KCghKG9wcyAmIH5PUF9ST1RBVEUpKSB8fAotICAgICAgICAocmVzY2FsZU5vcm1hbHMgJiYgbW9kZWx2aWV3LmlzUmlnaWRCb2R5KCkpKSkgewotICAgICAgICAvLyBpZiB0aGUgbW9kZWx2aWV3IG1hdHJpeCBpcyBhIHJpZ2lkIGJvZHkgdHJhbnNmb3JtYXRpb24KLSAgICAgICAgLy8gKHRyYW5zbGF0aW9uLCByb3RhdGlvbiwgdW5pZm9ybSBzY2FsaW5nKSwgdGhlbiB3ZSBjYW4gYnlwYXNzCi0gICAgICAgIC8vIHRoZSBpbnZlcnNlIGJ5IHRyYW5zcG9zaW5nIHRoZSBtYXRyaXguCi0gICAgICAgIEdMZmxvYXQgcmVzY2FsZSA9IDEuMGY7Ci0gICAgICAgIGlmIChyZXNjYWxlTm9ybWFscyA9PSBHTF9SRVNDQUxFX05PUk1BTCkgewotICAgICAgICAgICAgaWYgKCEob3BzICYgfk9QX1VOSUZPUk1fU0NBTEUpKSB7Ci0gICAgICAgICAgICAgICAgcmVzY2FsZSA9IHJlY2lwcm9jYWxmKG12W0koMCwwKV0pOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICByZXNjYWxlID0gcnNxcnRmKAotICAgICAgICAgICAgICAgICAgICAgICAgc3FyZihtdltJKDIsMCldKSArIHNxcmYobXZbSSgyLDEpXSkgKyBzcXJmKG12W0koMiwyKV0pKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICBHTGZpeGVkKiBjb25zdCB4ID0gbXZ1aS5tYXRyaXgubTsKLSAgICAgICAgZm9yIChpbnQgaT0wIDsgaTwzIDsgaSsrKSB7Ci0gICAgICAgICAgICB4W0koaSwwKV0gPSBnZ2xGbG9hdFRvRml4ZWQobXZbSSgwLGkpXSAqIHJlc2NhbGUpOwotICAgICAgICAgICAgeFtJKGksMSldID0gZ2dsRmxvYXRUb0ZpeGVkKG12W0koMSxpKV0gKiByZXNjYWxlKTsKLSAgICAgICAgICAgIHhbSShpLDIpXSA9IGdnbEZsb2F0VG9GaXhlZChtdltJKDIsaSldICogcmVzY2FsZSk7Ci0gICAgICAgIH0KLSAgICAgICAgbXZ1aS5waWNrZXIoKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIEdMZmxvYXQgclszXVszXTsKLSAgICByWzBdWzBdID0gZGV0MjIobXZbSSgxLDEpXSwgbXZbSSgyLDEpXSwgbXZbSSgxLDIpXSwgbXZbSSgyLDIpXSk7Ci0gICAgclswXVsxXSA9bmRldDIyKG12W0koMCwxKV0sIG12W0koMiwxKV0sIG12W0koMCwyKV0sIG12W0koMiwyKV0pOwotICAgIHJbMF1bMl0gPSBkZXQyMihtdltJKDAsMSldLCBtdltJKDEsMSldLCBtdltJKDAsMildLCBtdltJKDEsMildKTsKLSAgICByWzFdWzBdID1uZGV0MjIobXZbSSgxLDApXSwgbXZbSSgyLDApXSwgbXZbSSgxLDIpXSwgbXZbSSgyLDIpXSk7Ci0gICAgclsxXVsxXSA9IGRldDIyKG12W0koMCwwKV0sIG12W0koMiwwKV0sIG12W0koMCwyKV0sIG12W0koMiwyKV0pOwotICAgIHJbMV1bMl0gPW5kZXQyMihtdltJKDAsMCldLCBtdltJKDEsMCldLCBtdltJKDAsMildLCBtdltJKDEsMildKTsKLSAgICByWzJdWzBdID0gZGV0MjIobXZbSSgxLDApXSwgbXZbSSgyLDApXSwgbXZbSSgxLDEpXSwgbXZbSSgyLDEpXSk7Ci0gICAgclsyXVsxXSA9bmRldDIyKG12W0koMCwwKV0sIG12W0koMiwwKV0sIG12W0koMCwxKV0sIG12W0koMiwxKV0pOwotICAgIHJbMl1bMl0gPSBkZXQyMihtdltJKDAsMCldLCBtdltJKDEsMCldLCBtdltJKDAsMSldLCBtdltJKDEsMSldKTsgICAgICAgIAotCi0gICAgR0xmbG9hdCByZGV0OwotICAgIGlmIChyZXNjYWxlTm9ybWFscyA9PSBHTF9SRVNDQUxFX05PUk1BTCkgewotICAgICAgICByZGV0ID0gcnNxcnRmKHNxcmYoclswXVsyXSkgKyBzcXJmKHJbMV1bMl0pICsgc3FyZihyWzJdWzJdKSk7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgcmRldCA9IHJlY2lwcm9jYWxmKCAKLSAgICAgICAgICAgIHJbMF1bMF0qbXZbSSgwLDApXSArIHJbMF1bMV0qbXZbSSgxLDApXSArIHJbMF1bMl0qbXZbSSgyLDApXSk7Ci0gICAgfQotCi0gICAgR0xmaXhlZCogY29uc3QgeCA9IG12dWkubWF0cml4Lm07Ci0gICAgZm9yIChpbnQgaT0wIDsgaTwzIDsgaSsrKSB7Ci0gICAgICAgIHhbSShpLDApXSA9IGdnbEZsb2F0VG9GaXhlZChyW2ldWzBdICogcmRldCk7Ci0gICAgICAgIHhbSShpLDEpXSA9IGdnbEZsb2F0VG9GaXhlZChyW2ldWzFdICogcmRldCk7Ci0gICAgICAgIHhbSShpLDIpXSA9IGdnbEZsb2F0VG9GaXhlZChyW2ldWzJdICogcmRldCk7Ci0gICAgfQotICAgIG12dWkucGlja2VyKCk7Ci19Ci0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gdHJhbnNmb3JtYXRpb24gYW5kIG1hdHJpY2VzIEFQSQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jcHJhZ21hIG1hcmsgdHJhbnNmb3JtYXRpb24gYW5kIG1hdHJpY2VzIEFQSQotI2VuZGlmCi0KLWludCBvZ2xlc19zdXJmYWNlcG9ydChvZ2xlc19jb250ZXh0X3QqIGMsIEdMaW50IHgsIEdMaW50IHkpCi17Ci0gICAgYy0+dmlld3BvcnQuc3VyZmFjZXBvcnQueCA9IHg7Ci0gICAgYy0+dmlld3BvcnQuc3VyZmFjZXBvcnQueSA9IHk7Ci0KLSAgICBvZ2xlc192aWV3cG9ydChjLCAKLSAgICAgICAgICAgIGMtPnZpZXdwb3J0LngsCi0gICAgICAgICAgICBjLT52aWV3cG9ydC55LAotICAgICAgICAgICAgYy0+dmlld3BvcnQudywKLSAgICAgICAgICAgIGMtPnZpZXdwb3J0LmgpOwotCi0gICAgb2dsZXNfc2Npc3NvcihjLAotICAgICAgICAgICAgYy0+dmlld3BvcnQuc2Npc3Nvci54LAotICAgICAgICAgICAgYy0+dmlld3BvcnQuc2Npc3Nvci55LAotICAgICAgICAgICAgYy0+dmlld3BvcnQuc2Npc3Nvci53LAotICAgICAgICAgICAgYy0+dmlld3BvcnQuc2Npc3Nvci5oKTsKLQotICAgIHJldHVybiAwOwotfQotCi12b2lkIG9nbGVzX3NjaXNzb3Iob2dsZXNfY29udGV4dF90KiBjLCAKLSAgICAgICAgR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3LCBHTHNpemVpIGgpCi17Ci0gICAgaWYgKCh3fGgpIDwgMCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX1ZBTFVFKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBjLT52aWV3cG9ydC5zY2lzc29yLnggPSB4OwotICAgIGMtPnZpZXdwb3J0LnNjaXNzb3IueSA9IHk7Ci0gICAgYy0+dmlld3BvcnQuc2Npc3Nvci53ID0gdzsKLSAgICBjLT52aWV3cG9ydC5zY2lzc29yLmggPSBoOwotICAgIAotICAgIHggKz0gYy0+dmlld3BvcnQuc3VyZmFjZXBvcnQueDsKLSAgICB5ICs9IGMtPnZpZXdwb3J0LnN1cmZhY2Vwb3J0Lnk7Ci0KLSAgICB5ID0gYy0+cmFzdGVyaXplci5zdGF0ZS5idWZmZXJzLmNvbG9yLmhlaWdodCAtICh5ICsgaCk7Ci0gICAgYy0+cmFzdGVyaXplci5wcm9jcy5zY2lzc29yKGMsIHgsIHksIHcsIGgpOwotfQotCi12b2lkIG9nbGVzX3ZpZXdwb3J0KG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3LCBHTHNpemVpIGgpCi17Ci0gICAgaWYgKCh3fGgpPDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBjLT52aWV3cG9ydC54ID0geDsKLSAgICBjLT52aWV3cG9ydC55ID0geTsKLSAgICBjLT52aWV3cG9ydC53ID0gdzsKLSAgICBjLT52aWV3cG9ydC5oID0gaDsKLQotICAgIHggKz0gYy0+dmlld3BvcnQuc3VyZmFjZXBvcnQueDsKLSAgICB5ICs9IGMtPnZpZXdwb3J0LnN1cmZhY2Vwb3J0Lnk7Ci0KLSAgICBHTGludCBIID0gYy0+cmFzdGVyaXplci5zdGF0ZS5idWZmZXJzLmNvbG9yLmhlaWdodDsKLSAgICBHTGZsb2F0IHN4ID0gZGl2MmYodyk7Ci0gICAgR0xmbG9hdCBveCA9IHN4ICsgeDsKLSAgICBHTGZsb2F0IHN5ID0gZGl2MmYoaCk7Ci0gICAgR0xmbG9hdCBveSA9IHN5IC0geSArIChIIC0gaCk7Ci0KLSAgICBHTGZsb2F0IG5lYXIgPSBjLT50cmFuc2Zvcm1zLnZwdC56TmVhcjsKLSAgICBHTGZsb2F0IGZhciAgPSBjLT50cmFuc2Zvcm1zLnZwdC56RmFyOwotICAgIEdMZmxvYXQgQSA9IGRpdjJmKGZhciAtIG5lYXIpOwotICAgIEdMZmxvYXQgQiA9IGRpdjJmKGZhciArIG5lYXIpOwotCi0gICAgLy8gY29tcHV0ZSB2aWV3cG9ydCBtYXRyaXgKLSAgICBHTGZsb2F0KiBjb25zdCBmID0gYy0+dHJhbnNmb3Jtcy52cHQubWF0cml4LmVkaXRFbGVtZW50cygpOwotICAgIGZbMF0gPSBzeDsgIGZbNF0gPSAwOyAgIGZbIDhdID0gMDsgIGZbMTJdID0gb3g7Ci0gICAgZlsxXSA9IDA7ICAgZls1XSA9LXN5OyAgZlsgOV0gPSAwOyAgZlsxM10gPSBveTsKLSAgICBmWzJdID0gMDsgICBmWzZdID0gMDsgICBmWzEwXSA9IEE7ICBmWzE0XSA9IEI7Ci0gICAgZlszXSA9IDA7ICAgZls3XSA9IDA7ICAgZlsxMV0gPSAwOyAgZlsxNV0gPSAxOwotICAgIGMtPnRyYW5zZm9ybXMuZGlydHkgfD0gdHJhbnNmb3JtX3N0YXRlX3Q6OlZJRVdQT1JUOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBtYXRyaXggKiB2ZXJ0ZXgKLSNlbmRpZgotCi12b2lkIHBvaW50Ml9fZ2VuZXJpYyh0cmFuc2Zvcm1fdCBjb25zdCogbXgsIHZlYzRfdCogbGhzLCB2ZWM0X3QgY29uc3QqIHJocykgewotICAgIGNvbnN0IEdMZml4ZWQqIGNvbnN0IG0gPSBteC0+bWF0cml4Lm07Ci0gICAgY29uc3QgR0xmaXhlZCByeCA9IHJocy0+eDsKLSAgICBjb25zdCBHTGZpeGVkIHJ5ID0gcmhzLT55OwotICAgIGxocy0+eCA9IG1sYTJhKHJ4LCBtWyAwXSwgcnksIG1bIDRdLCBtWzEyXSk7IAotICAgIGxocy0+eSA9IG1sYTJhKHJ4LCBtWyAxXSwgcnksIG1bIDVdLCBtWzEzXSk7Ci0gICAgbGhzLT56ID0gbWxhMmEocngsIG1bIDJdLCByeSwgbVsgNl0sIG1bMTRdKTsKLSAgICBsaHMtPncgPSBtbGEyYShyeCwgbVsgM10sIHJ5LCBtWyA3XSwgbVsxNV0pOwotfQotCi12b2lkIHBvaW50M19fZ2VuZXJpYyh0cmFuc2Zvcm1fdCBjb25zdCogbXgsIHZlYzRfdCogbGhzLCB2ZWM0X3QgY29uc3QqIHJocykgewotICAgIGNvbnN0IEdMZml4ZWQqIGNvbnN0IG0gPSBteC0+bWF0cml4Lm07Ci0gICAgY29uc3QgR0xmaXhlZCByeCA9IHJocy0+eDsKLSAgICBjb25zdCBHTGZpeGVkIHJ5ID0gcmhzLT55OwotICAgIGNvbnN0IEdMZml4ZWQgcnogPSByaHMtPno7Ci0gICAgbGhzLT54ID0gbWxhM2EocngsIG1bIDBdLCByeSwgbVsgNF0sIHJ6LCBtWyA4XSwgbVsxMl0pOyAKLSAgICBsaHMtPnkgPSBtbGEzYShyeCwgbVsgMV0sIHJ5LCBtWyA1XSwgcnosIG1bIDldLCBtWzEzXSk7Ci0gICAgbGhzLT56ID0gbWxhM2EocngsIG1bIDJdLCByeSwgbVsgNl0sIHJ6LCBtWzEwXSwgbVsxNF0pOwotICAgIGxocy0+dyA9IG1sYTNhKHJ4LCBtWyAzXSwgcnksIG1bIDddLCByeiwgbVsxMV0sIG1bMTVdKTsKLX0KLQotdm9pZCBwb2ludDRfX2dlbmVyaWModHJhbnNmb3JtX3QgY29uc3QqIG14LCB2ZWM0X3QqIGxocywgdmVjNF90IGNvbnN0KiByaHMpIHsKLSAgICBjb25zdCBHTGZpeGVkKiBjb25zdCBtID0gbXgtPm1hdHJpeC5tOwotICAgIGNvbnN0IEdMZml4ZWQgcnggPSByaHMtPng7Ci0gICAgY29uc3QgR0xmaXhlZCByeSA9IHJocy0+eTsKLSAgICBjb25zdCBHTGZpeGVkIHJ6ID0gcmhzLT56OwotICAgIGNvbnN0IEdMZml4ZWQgcncgPSByaHMtPnc7Ci0gICAgbGhzLT54ID0gbWxhNChyeCwgbVsgMF0sIHJ5LCBtWyA0XSwgcnosIG1bIDhdLCBydywgbVsxMl0pOyAKLSAgICBsaHMtPnkgPSBtbGE0KHJ4LCBtWyAxXSwgcnksIG1bIDVdLCByeiwgbVsgOV0sIHJ3LCBtWzEzXSk7Ci0gICAgbGhzLT56ID0gbWxhNChyeCwgbVsgMl0sIHJ5LCBtWyA2XSwgcnosIG1bMTBdLCBydywgbVsxNF0pOwotICAgIGxocy0+dyA9IG1sYTQocngsIG1bIDNdLCByeSwgbVsgN10sIHJ6LCBtWzExXSwgcncsIG1bMTVdKTsKLX0KLQotdm9pZCBub3JtYWxfX2dlbmVyaWModHJhbnNmb3JtX3QgY29uc3QqIG14LCB2ZWM0X3QqIGxocywgdmVjNF90IGNvbnN0KiByaHMpIHsKLSAgICBjb25zdCBHTGZpeGVkKiBjb25zdCBtID0gbXgtPm1hdHJpeC5tOwotICAgIGNvbnN0IEdMZml4ZWQgcnggPSByaHMtPng7Ci0gICAgY29uc3QgR0xmaXhlZCByeSA9IHJocy0+eTsKLSAgICBjb25zdCBHTGZpeGVkIHJ6ID0gcmhzLT56OwotICAgIGxocy0+eCA9IG1sYTMocngsIG1bIDBdLCByeSwgbVsgNF0sIHJ6LCBtWyA4XSk7IAotICAgIGxocy0+eSA9IG1sYTMocngsIG1bIDFdLCByeSwgbVsgNV0sIHJ6LCBtWyA5XSk7Ci0gICAgbGhzLT56ID0gbWxhMyhyeCwgbVsgMl0sIHJ5LCBtWyA2XSwgcnosIG1bMTBdKTsKLX0KLQotCi12b2lkIHBvaW50Ml9fbm9wKHRyYW5zZm9ybV90IGNvbnN0KiwgdmVjNF90KiBsaHMsIHZlYzRfdCBjb25zdCogcmhzKSB7Ci0gICAgbGhzLT56ID0gMDsKLSAgICBsaHMtPncgPSAweDEwMDAwOwotICAgIGlmIChsaHMgIT0gcmhzKSB7Ci0gICAgICAgIGxocy0+eCA9IHJocy0+eDsKLSAgICAgICAgbGhzLT55ID0gcmhzLT55OwotICAgIH0KLX0KLQotdm9pZCBwb2ludDNfX25vcCh0cmFuc2Zvcm1fdCBjb25zdCosIHZlYzRfdCogbGhzLCB2ZWM0X3QgY29uc3QqIHJocykgewotICAgIGxocy0+dyA9IDB4MTAwMDA7Ci0gICAgaWYgKGxocyAhPSByaHMpIHsKLSAgICAgICAgbGhzLT54ID0gcmhzLT54OwotICAgICAgICBsaHMtPnkgPSByaHMtPnk7Ci0gICAgICAgIGxocy0+eiA9IHJocy0+ejsKLSAgICB9Ci19Ci0KLXZvaWQgcG9pbnQ0X19ub3AodHJhbnNmb3JtX3QgY29uc3QqLCB2ZWM0X3QqIGxocywgdmVjNF90IGNvbnN0KiByaHMpIHsKLSAgICBpZiAobGhzICE9IHJocykKLSAgICAgICAgKmxocyA9ICpyaHM7Ci19Ci0KLQotc3RhdGljIHZvaWQgZnJ1c3R1bWYoCi0gICAgICAgICAgICBHTGZsb2F0IGxlZnQsIEdMZmxvYXQgcmlnaHQsIAotICAgICAgICAgICAgR0xmbG9hdCBib3R0b20sIEdMZmxvYXQgdG9wLAotICAgICAgICAgICAgR0xmbG9hdCB6TmVhciwgR0xmbG9hdCB6RmFyLAotICAgICAgICAgICAgb2dsZXNfY29udGV4dF90KiBjKQotICAgIHsKLSAgICBpZiAoY21wZihsZWZ0LHJpZ2h0KSB8fAotICAgICAgICBjbXBmKHRvcCwgYm90dG9tKSB8fAotICAgICAgICBjbXBmKHpOZWFyLCB6RmFyKSB8fAotICAgICAgICBpc1plcm9Pck5lZ2F0aXZlZih6TmVhcikgfHwKLSAgICAgICAgaXNaZXJvT3JOZWdhdGl2ZWYoekZhcikpCi0gICAgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX1ZBTFVFKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBjb25zdCBHTGZsb2F0IHJfd2lkdGggID0gcmVjaXByb2NhbGYocmlnaHQgLSBsZWZ0KTsKLSAgICBjb25zdCBHTGZsb2F0IHJfaGVpZ2h0ID0gcmVjaXByb2NhbGYodG9wIC0gYm90dG9tKTsKLSAgICBjb25zdCBHTGZsb2F0IHJfZGVwdGggID0gcmVjaXByb2NhbGYoek5lYXIgLSB6RmFyKTsKLSAgICBjb25zdCBHTGZsb2F0IHggPSBtdWwyZih6TmVhciAqIHJfd2lkdGgpOwotICAgIGNvbnN0IEdMZmxvYXQgeSA9IG11bDJmKHpOZWFyICogcl9oZWlnaHQpOwotICAgIGNvbnN0IEdMZmxvYXQgQSA9IG11bDJmKChyaWdodCArIGxlZnQpICogcl93aWR0aCk7Ci0gICAgY29uc3QgR0xmbG9hdCBCID0gKHRvcCArIGJvdHRvbSkgKiByX2hlaWdodDsKLSAgICBjb25zdCBHTGZsb2F0IEMgPSAoekZhciArIHpOZWFyKSAqIHJfZGVwdGg7Ci0gICAgY29uc3QgR0xmbG9hdCBEID0gbXVsMmYoekZhciAqIHpOZWFyICogcl9kZXB0aCk7Ci0gICAgR0xmbG9hdCBmWzE2XTsKLSAgICBmWyAwXSA9IHg7Ci0gICAgZlsgNV0gPSB5OwotICAgIGZbIDhdID0gQTsKLSAgICBmWyA5XSA9IEI7Ci0gICAgZlsxMF0gPSBDOwotICAgIGZbMTRdID0gRDsKLSAgICBmWzExXSA9IC0xLjBmOwotICAgIGZbIDFdID0gZlsgMl0gPSBmWyAzXSA9Ci0gICAgZlsgNF0gPSBmWyA2XSA9IGZbIDddID0KLSAgICBmWzEyXSA9IGZbMTNdID0gZlsxNV0gPSAwLjBmOwotCi0gICAgbWF0cml4Zl90IHJoczsKLSAgICByaHMuc2V0KGYpOwotICAgIGMtPnRyYW5zZm9ybXMuY3VycmVudC0+bXVsdGlwbHkocmhzKTsKLSAgICBjLT50cmFuc2Zvcm1zLmludmFsaWRhdGUoKTsKLX0KLQotc3RhdGljIHZvaWQgb3J0aG9mKCAKLSAgICAgICAgR0xmbG9hdCBsZWZ0LCBHTGZsb2F0IHJpZ2h0LCAKLSAgICAgICAgR0xmbG9hdCBib3R0b20sIEdMZmxvYXQgdG9wLAotICAgICAgICBHTGZsb2F0IHpOZWFyLCBHTGZsb2F0IHpGYXIsCi0gICAgICAgIG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBpZiAoY21wZihsZWZ0LHJpZ2h0KSB8fAotICAgICAgICBjbXBmKHRvcCwgYm90dG9tKSB8fAotICAgICAgICBjbXBmKHpOZWFyLCB6RmFyKSkKLSAgICB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGNvbnN0IEdMZmxvYXQgcl93aWR0aCAgPSByZWNpcHJvY2FsZihyaWdodCAtIGxlZnQpOwotICAgIGNvbnN0IEdMZmxvYXQgcl9oZWlnaHQgPSByZWNpcHJvY2FsZih0b3AgLSBib3R0b20pOwotICAgIGNvbnN0IEdMZmxvYXQgcl9kZXB0aCAgPSByZWNpcHJvY2FsZih6RmFyIC0gek5lYXIpOwotICAgIGNvbnN0IEdMZmxvYXQgeCA9ICBtdWwyZihyX3dpZHRoKTsKLSAgICBjb25zdCBHTGZsb2F0IHkgPSAgbXVsMmYocl9oZWlnaHQpOwotICAgIGNvbnN0IEdMZmxvYXQgeiA9IC1tdWwyZihyX2RlcHRoKTsKLSAgICBjb25zdCBHTGZsb2F0IHR4ID0gLShyaWdodCArIGxlZnQpICogcl93aWR0aDsKLSAgICBjb25zdCBHTGZsb2F0IHR5ID0gLSh0b3AgKyBib3R0b20pICogcl9oZWlnaHQ7Ci0gICAgY29uc3QgR0xmbG9hdCB0eiA9IC0oekZhciArIHpOZWFyKSAqIHJfZGVwdGg7Ci0gICAgR0xmbG9hdCBmWzE2XTsKLSAgICBmWyAwXSA9IHg7Ci0gICAgZlsgNV0gPSB5OwotICAgIGZbMTBdID0gejsKLSAgICBmWzEyXSA9IHR4OwotICAgIGZbMTNdID0gdHk7Ci0gICAgZlsxNF0gPSB0ejsKLSAgICBmWzE1XSA9IDEuMGY7Ci0gICAgZlsgMV0gPSBmWyAyXSA9IGZbIDNdID0KLSAgICBmWyA0XSA9IGZbIDZdID0gZlsgN10gPQotICAgIGZbIDhdID0gZlsgOV0gPSBmWzExXSA9IDAuMGY7Ci0gICAgbWF0cml4Zl90IHJoczsKLSAgICByaHMuc2V0KGYpOwotICAgIGMtPnRyYW5zZm9ybXMuY3VycmVudC0+bXVsdGlwbHkocmhzKTsKLSAgICBjLT50cmFuc2Zvcm1zLmludmFsaWRhdGUoKTsKLX0KLQotc3RhdGljIHZvaWQgZGVwdGhSYW5nZWYoR0xjbGFtcGYgek5lYXIsIEdMY2xhbXBmIHpGYXIsIG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICB6TmVhciA9IGNsYW1wVG9aZXJvZih6TmVhciA+IDEgPyAxIDogek5lYXIpOwotICAgIHpGYXIgID0gY2xhbXBUb1plcm9mKHpGYXIgID4gMSA/IDEgOiB6RmFyKTsKLSAgICBHTGZsb2F0KiBjb25zdCBmID0gYy0+dHJhbnNmb3Jtcy52cHQubWF0cml4LmVkaXRFbGVtZW50cygpOwotICAgIGZbMTBdID0gZGl2MmYoekZhciAtIHpOZWFyKTsKLSAgICBmWzE0XSA9IGRpdjJmKHpGYXIgKyB6TmVhcik7Ci0gICAgYy0+dHJhbnNmb3Jtcy5kaXJ0eSB8PSB0cmFuc2Zvcm1fc3RhdGVfdDo6VklFV1BPUlQ7Ci0gICAgYy0+dHJhbnNmb3Jtcy52cHQuek5lYXIgPSB6TmVhcjsKLSAgICBjLT50cmFuc2Zvcm1zLnZwdC56RmFyICA9IHpGYXI7Ci19Ci0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLQotdXNpbmcgbmFtZXNwYWNlIGFuZHJvaWQ7Ci0KLXZvaWQgZ2xNYXRyaXhNb2RlKEdMZW51bSBtb2RlKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgbWF0cml4X3N0YWNrX3QqIHN0YWNrID0gMDsKLSAgICBzd2l0Y2ggKG1vZGUpIHsKLSAgICBjYXNlIEdMX01PREVMVklFVzoKLSAgICAgICAgc3RhY2sgPSAmYy0+dHJhbnNmb3Jtcy5tb2RlbHZpZXc7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfUFJPSkVDVElPTjoKLSAgICAgICAgc3RhY2sgPSAmYy0+dHJhbnNmb3Jtcy5wcm9qZWN0aW9uOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1RFWFRVUkU6Ci0gICAgICAgIHN0YWNrID0gJmMtPnRyYW5zZm9ybXMudGV4dHVyZVtjLT50ZXh0dXJlcy5hY3RpdmVdOwotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGMtPnRyYW5zZm9ybXMubWF0cml4TW9kZSA9IG1vZGU7Ci0gICAgYy0+dHJhbnNmb3Jtcy5jdXJyZW50ID0gc3RhY2s7Ci19Ci0KLXZvaWQgZ2xMb2FkSWRlbnRpdHkoKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+dHJhbnNmb3Jtcy5jdXJyZW50LT5sb2FkSWRlbnRpdHkoKTsgLy8gYWxzbyBsb2FkcyB0aGUgR0xmaXhlZCB0cmFuc2Zvcm0KLSAgICBjLT50cmFuc2Zvcm1zLmludmFsaWRhdGUoKTsKLSAgICBjLT50cmFuc2Zvcm1zLmN1cnJlbnQtPmRpcnR5ID0gMDsKLX0KLQotdm9pZCBnbExvYWRNYXRyaXhmKGNvbnN0IEdMZmxvYXQqIG0pCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT50cmFuc2Zvcm1zLmN1cnJlbnQtPmxvYWQobSk7Ci0gICAgYy0+dHJhbnNmb3Jtcy5pbnZhbGlkYXRlKCk7Ci19Ci0KLXZvaWQgZ2xMb2FkTWF0cml4eChjb25zdCBHTGZpeGVkKiBtKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+dHJhbnNmb3Jtcy5jdXJyZW50LT5sb2FkKG0pOyAvLyBhbHNvIGxvYWRzIHRoZSBHTGZpeGVkIHRyYW5zZm9ybQotICAgIGMtPnRyYW5zZm9ybXMuaW52YWxpZGF0ZSgpOwotICAgIGMtPnRyYW5zZm9ybXMuY3VycmVudC0+ZGlydHkgJj0gfm1hdHJpeF9zdGFja190OjpET19GTE9BVF9UT19GSVhFRDsKLX0KLQotdm9pZCBnbE11bHRNYXRyaXhmKGNvbnN0IEdMZmxvYXQqIG0pCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBtYXRyaXhmX3QgcmhzOwotICAgIHJocy5zZXQobSk7Ci0gICAgYy0+dHJhbnNmb3Jtcy5jdXJyZW50LT5tdWx0aXBseShyaHMpOwotICAgIGMtPnRyYW5zZm9ybXMuaW52YWxpZGF0ZSgpOwotfQotCi12b2lkIGdsTXVsdE1hdHJpeHgoY29uc3QgR0xmaXhlZCogbSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIG1hdHJpeGZfdCByaHM7Ci0gICAgcmhzLnNldChtKTsKLSAgICBjLT50cmFuc2Zvcm1zLmN1cnJlbnQtPm11bHRpcGx5KHJocyk7Ci0gICAgYy0+dHJhbnNmb3Jtcy5pbnZhbGlkYXRlKCk7Ci19Ci0KLXZvaWQgZ2xQb3BNYXRyaXgoKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgR0xpbnQgZXJyID0gYy0+dHJhbnNmb3Jtcy5jdXJyZW50LT5wb3AoKTsKLSAgICBpZiAoZ2dsX3VubGlrZWx5KGVycikpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgZXJyKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBjLT50cmFuc2Zvcm1zLmludmFsaWRhdGUoKTsKLX0KLQotdm9pZCBnbFB1c2hNYXRyaXgoKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgR0xpbnQgZXJyID0gYy0+dHJhbnNmb3Jtcy5jdXJyZW50LT5wdXNoKCk7Ci0gICAgaWYgKGdnbF91bmxpa2VseShlcnIpKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIGVycik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgYy0+dHJhbnNmb3Jtcy5pbnZhbGlkYXRlKCk7Ci19Ci0KLXZvaWQgZ2xGcnVzdHVtZigKLSAgICAgICAgR0xmbG9hdCBsZWZ0LCBHTGZsb2F0IHJpZ2h0LCAKLSAgICAgICAgR0xmbG9hdCBib3R0b20sIEdMZmxvYXQgdG9wLAotICAgICAgICBHTGZsb2F0IHpOZWFyLCBHTGZsb2F0IHpGYXIpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBmcnVzdHVtZihsZWZ0LCByaWdodCwgYm90dG9tLCB0b3AsIHpOZWFyLCB6RmFyLCBjKTsKLX0KLQotdm9pZCBnbEZydXN0dW14KCAKLSAgICAgICAgR0xmaXhlZCBsZWZ0LCBHTGZpeGVkIHJpZ2h0LAotICAgICAgICBHTGZpeGVkIGJvdHRvbSwgR0xmaXhlZCB0b3AsCi0gICAgICAgIEdMZml4ZWQgek5lYXIsIEdMZml4ZWQgekZhcikKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGZydXN0dW1mKCBmaXhlZFRvRmxvYXQobGVmdCksIGZpeGVkVG9GbG9hdChyaWdodCksCi0gICAgICAgICAgICAgIGZpeGVkVG9GbG9hdChib3R0b20pLCBmaXhlZFRvRmxvYXQodG9wKSwKLSAgICAgICAgICAgICAgZml4ZWRUb0Zsb2F0KHpOZWFyKSwgZml4ZWRUb0Zsb2F0KHpGYXIpLAotICAgICAgICAgICAgICBjKTsKLX0KLQotdm9pZCBnbE9ydGhvZiggCi0gICAgICAgIEdMZmxvYXQgbGVmdCwgR0xmbG9hdCByaWdodCwgCi0gICAgICAgIEdMZmxvYXQgYm90dG9tLCBHTGZsb2F0IHRvcCwKLSAgICAgICAgR0xmbG9hdCB6TmVhciwgR0xmbG9hdCB6RmFyKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgb3J0aG9mKGxlZnQsIHJpZ2h0LCBib3R0b20sIHRvcCwgek5lYXIsIHpGYXIsIGMpOwotfQotCi12b2lkIGdsT3J0aG94KAotICAgICAgICBHTGZpeGVkIGxlZnQsIEdMZml4ZWQgcmlnaHQsCi0gICAgICAgIEdMZml4ZWQgYm90dG9tLCBHTGZpeGVkIHRvcCwKLSAgICAgICAgR0xmaXhlZCB6TmVhciwgR0xmaXhlZCB6RmFyKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgb3J0aG9mKCBmaXhlZFRvRmxvYXQobGVmdCksIGZpeGVkVG9GbG9hdChyaWdodCksCi0gICAgICAgICAgICBmaXhlZFRvRmxvYXQoYm90dG9tKSwgZml4ZWRUb0Zsb2F0KHRvcCksCi0gICAgICAgICAgICBmaXhlZFRvRmxvYXQoek5lYXIpLCBmaXhlZFRvRmxvYXQoekZhciksCi0gICAgICAgICAgICBjKTsKLX0KLQotdm9pZCBnbFJvdGF0ZWYoR0xmbG9hdCBhLCBHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+dHJhbnNmb3Jtcy5jdXJyZW50LT5yb3RhdGUoYSwgeCwgeSwgeik7Ci0gICAgYy0+dHJhbnNmb3Jtcy5pbnZhbGlkYXRlKCk7Ci19Ci0KLXZvaWQgZ2xSb3RhdGV4KEdMZml4ZWQgYSwgR0xmaXhlZCB4LCBHTGZpeGVkIHksIEdMZml4ZWQgeikKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGMtPnRyYW5zZm9ybXMuY3VycmVudC0+cm90YXRlKCAKLSAgICAgICAgICAgIGZpeGVkVG9GbG9hdChhKSwgZml4ZWRUb0Zsb2F0KHgpLAotICAgICAgICAgICAgZml4ZWRUb0Zsb2F0KHkpLCBmaXhlZFRvRmxvYXQoeikpOwotICAgIGMtPnRyYW5zZm9ybXMuaW52YWxpZGF0ZSgpOwotfQotCi12b2lkIGdsU2NhbGVmKEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT50cmFuc2Zvcm1zLmN1cnJlbnQtPnNjYWxlKHgsIHksIHopOwotICAgIGMtPnRyYW5zZm9ybXMuaW52YWxpZGF0ZSgpOwotfQotCi12b2lkIGdsU2NhbGV4KEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT50cmFuc2Zvcm1zLmN1cnJlbnQtPnNjYWxlKAotICAgICAgICAgICAgZml4ZWRUb0Zsb2F0KHgpLCBmaXhlZFRvRmxvYXQoeSksIGZpeGVkVG9GbG9hdCh6KSk7Ci0gICAgYy0+dHJhbnNmb3Jtcy5pbnZhbGlkYXRlKCk7Ci19Ci0KLXZvaWQgZ2xUcmFuc2xhdGVmKEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT50cmFuc2Zvcm1zLmN1cnJlbnQtPnRyYW5zbGF0ZSh4LCB5LCB6KTsKLSAgICBjLT50cmFuc2Zvcm1zLmludmFsaWRhdGUoKTsKLX0KLQotdm9pZCBnbFRyYW5zbGF0ZXgoR0xmaXhlZCB4LCBHTGZpeGVkIHksIEdMZml4ZWQgeikKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGMtPnRyYW5zZm9ybXMuY3VycmVudC0+dHJhbnNsYXRlKAotICAgICAgICAgICAgZml4ZWRUb0Zsb2F0KHgpLCBmaXhlZFRvRmxvYXQoeSksIGZpeGVkVG9GbG9hdCh6KSk7Ci0gICAgYy0+dHJhbnNmb3Jtcy5pbnZhbGlkYXRlKCk7Ci19Ci0KLXZvaWQgZ2xTY2lzc29yKEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgdywgR0xzaXplaSBoKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgb2dsZXNfc2Npc3NvcihjLCB4LCB5LCB3LCBoKTsKLX0KLQotdm9pZCBnbFZpZXdwb3J0KEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgdywgR0xzaXplaSBoKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgb2dsZXNfdmlld3BvcnQoYywgeCwgeSwgdywgaCk7Ci19Ci0KLXZvaWQgZ2xEZXB0aFJhbmdlZihHTGNsYW1wZiB6TmVhciwgR0xjbGFtcGYgekZhcikKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGRlcHRoUmFuZ2VmKHpOZWFyLCB6RmFyLCBjKTsKLX0KLQotdm9pZCBnbERlcHRoUmFuZ2V4KEdMY2xhbXB4IHpOZWFyLCBHTGNsYW1weCB6RmFyKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgZGVwdGhSYW5nZWYoZml4ZWRUb0Zsb2F0KHpOZWFyKSwgZml4ZWRUb0Zsb2F0KHpGYXIpLCBjKTsKLX0KLQotdm9pZCBnbFBvbHlnb25PZmZzZXR4KEdMZml4ZWQgZmFjdG9yLCBHTGZpeGVkIHVuaXRzKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+cG9seWdvbk9mZnNldC5mYWN0b3IgPSBmYWN0b3I7Ci0gICAgYy0+cG9seWdvbk9mZnNldC51bml0cyA9IHVuaXRzOwotfQotCi12b2lkIGdsUG9seWdvbk9mZnNldChHTGZsb2F0IGZhY3RvciwgR0xmbG9hdCB1bml0cykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGMtPnBvbHlnb25PZmZzZXQuZmFjdG9yID0gZ2dsRmxvYXRUb0ZpeGVkKGZhY3Rvcik7Ci0gICAgYy0+cG9seWdvbk9mZnNldC51bml0cyA9IGdnbEZsb2F0VG9GaXhlZCh1bml0cyk7Ci19Ci0KLUdMYml0ZmllbGQgZ2xRdWVyeU1hdHJpeHhPRVMoR0xmaXhlZCogbSwgR0xpbnQqIGUpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBHTGJpdGZpZWxkIHN0YXR1cyA9IDA7Ci0gICAgR0xmbG9hdCBjb25zdCogZiA9IGMtPnRyYW5zZm9ybXMuY3VycmVudC0+dG9wKCkuZWxlbWVudHMoKTsKLSAgICBmb3IgIChpbnQgaT0wIDsgaTwxNiA7IGkrKykgewotICAgICAgICBpZiAoaXNuYW4oZltpXSkgfHwgaXNpbmYoZltpXSkpIHsKLSAgICAgICAgICAgIHN0YXR1cyB8PSAxPDxpOwotICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgIH0KLSAgICAgICAgZVtpXSA9IGV4cG9uZW50KGZbaV0pIC0gNzsKLSAgICAgICAgbVtpXSA9IG1hbnRpc3NhKGZbaV0pOwotICAgIH0KLSAgICByZXR1cm4gc3RhdHVzOwotfQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC9tYXRyaXguaCBiL29wZW5nbC9saWJhZ2wvbWF0cml4LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM5YTM4YTkuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9tYXRyaXguaAorKysgL2Rldi9udWxsCkBAIC0xLDM1NSArMCwwIEBACi0vKiBsaWJzL29wZW5nbGVzL21hdHJpeC5oCi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2lmbmRlZiBBTkRST0lEX09QRU5HTEVTX01BVFJJWF9ICi0jZGVmaW5lIEFORFJPSURfT1BFTkdMRVNfTUFUUklYX0gKLQotI2luY2x1ZGUgPHN0ZGludC5oPgotI2luY2x1ZGUgPHN0ZGRlZi5oPgotI2luY2x1ZGUgPHN5cy90eXBlcy5oPgotI2luY2x1ZGUgPHV0aWxzL0xvZy5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS9waXhlbGZsaW5nZXIvZ2dsX2NvbnRleHQuaD4KLQotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1jb25zdCBpbnQgT0dMRVNfTU9ERUxWSUVXX1NUQUNLX0RFUFRIICAgPSAxNjsKLWNvbnN0IGludCBPR0xFU19QUk9KRUNUSU9OX1NUQUNLX0RFUFRIICA9ICAyOwotY29uc3QgaW50IE9HTEVTX1RFWFRVUkVfU1RBQ0tfREVQVEggICAgID0gIDI7Ci0KLXZvaWQgb2dsZXNfaW5pdF9tYXRyaXgob2dsZXNfY29udGV4dF90Kik7Ci12b2lkIG9nbGVzX3VuaW5pdF9tYXRyaXgob2dsZXNfY29udGV4dF90Kik7Ci12b2lkIG9nbGVzX2ludmFsaWRhdGVfcGVyc3BlY3RpdmUob2dsZXNfY29udGV4dF90KiBjKTsKLXZvaWQgb2dsZXNfdmFsaWRhdGVfdHJhbnNmb3JtX2ltcGwob2dsZXNfY29udGV4dF90KiBjLCB1aW50MzJfdCB3YW50KTsKLQotaW50IG9nbGVzX3N1cmZhY2Vwb3J0KG9nbGVzX2NvbnRleHRfdCogYywgR0xpbnQgeCwgR0xpbnQgeSk7Ci0KLXZvaWQgb2dsZXNfc2Npc3NvcihvZ2xlc19jb250ZXh0X3QqIGMsIAotICAgICAgICBHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHcsIEdMc2l6ZWkgaCk7Ci0KLXZvaWQgb2dsZXNfdmlld3BvcnQob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICBHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHcsIEdMc2l6ZWkgaCk7Ci0KLWlubGluZSB2b2lkIG9nbGVzX3ZhbGlkYXRlX3RyYW5zZm9ybSgKLSAgICAgICAgb2dsZXNfY29udGV4dF90KiBjLCB1aW50MzJfdCB3YW50KQotewotICAgIGlmIChjLT50cmFuc2Zvcm1zLmRpcnR5ICYgd2FudCkKLSAgICAgICAgb2dsZXNfdmFsaWRhdGVfdHJhbnNmb3JtX2ltcGwoYywgd2FudCk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotaW5saW5lCi1HTGZpeGVkIHZzcXVhcmUzKEdMZml4ZWQgYSwgR0xmaXhlZCBiLCBHTGZpeGVkIGMpIAotewotI2lmIGRlZmluZWQoX19hcm1fXykgJiYgIWRlZmluZWQoX190aHVtYl9fKQotCi0gICAgR0xmaXhlZCByOwotICAgIGludDMyX3QgdDsKLSAgICBhc20oCi0gICAgICAgICJzbXVsbCAlMCwgJTEsICUyLCAlMiAgICAgICBcbiIKLSAgICAgICAgInNtbGFsICUwLCAlMSwgJTMsICUzICAgICAgIFxuIgotICAgICAgICAic21sYWwgJTAsICUxLCAlNCwgJTQgICAgICAgXG4iCi0gICAgICAgICJtb3ZzICAlMCwgJTAsIGxzciAjMTYgICAgICBcbiIKLSAgICAgICAgImFkYyAgICUwLCAlMCwgJTEsIGxzbCAjMTYgIFxuIgotICAgICAgICA6ICAgIj0mciIociksICI9JnIiKHQpIAotICAgICAgICA6ICAgIiVyIihhKSwgInIiKGIpLCAiciIoYykKLSAgICAgICAgOiAgICJjYyIKLSAgICAgICAgKTsgCi0gICAgcmV0dXJuIHI7Ci0KLSNlbHNlCi0KLSAgICByZXR1cm4gKCggICBpbnQ2NF90KGEpKmEgKwotICAgICAgICAgICAgICAgIGludDY0X3QoYikqYiArCi0gICAgICAgICAgICAgICAgaW50NjRfdChjKSpjICsgMHg4MDAwKT4+MTYpOwotCi0jZW5kaWYKLX0KLQotc3RhdGljIGlubGluZSBHTGZpeGVkIG1sYTJhKCBHTGZpeGVkIGEwLCBHTGZpeGVkIGIwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMZml4ZWQgYTEsIEdMZml4ZWQgYjEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xmaXhlZCBjKQotewotI2lmIGRlZmluZWQoX19hcm1fXykgJiYgIWRlZmluZWQoX190aHVtYl9fKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgIEdMZml4ZWQgcjsKLSAgICBpbnQzMl90IHQ7Ci0gICAgYXNtKAotICAgICAgICAic211bGwgJTAsICUxLCAlMiwgJTMgICAgICAgXG4iCi0gICAgICAgICJzbWxhbCAlMCwgJTEsICU0LCAlNSAgICAgICBcbiIKLSAgICAgICAgImFkZCAgICUwLCAlNiwgJTAsIGxzciAjMTYgIFxuIgotICAgICAgICAiYWRkICAgJTAsICUwLCAlMSwgbHNsICMxNiAgXG4iCi0gICAgICAgIDogICAiPSZyIihyKSwgIj0mciIodCkgCi0gICAgICAgIDogICAiJXIiKGEwKSwgInIiKGIwKSwgCi0gICAgICAgICAgICAiJXIiKGExKSwgInIiKGIxKSwKLSAgICAgICAgICAgICJyIihjKQotICAgICAgICA6Ci0gICAgICAgICk7IAotICAgIHJldHVybiByOwotICAgIAotI2Vsc2UKLQotICAgIHJldHVybiAoKCAgIGludDY0X3QoYTApKmIwICsKLSAgICAgICAgICAgICAgICBpbnQ2NF90KGExKSpiMSk+PjE2KSArIGM7Ci0KLSNlbmRpZgotfQotCi1zdGF0aWMgaW5saW5lIEdMZml4ZWQgbWxhM2EoIEdMZml4ZWQgYTAsIEdMZml4ZWQgYjAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMZml4ZWQgYTEsIEdMZml4ZWQgYjEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMZml4ZWQgYTIsIEdMZml4ZWQgYjIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMZml4ZWQgYykKLXsKLSNpZiBkZWZpbmVkKF9fYXJtX18pICYmICFkZWZpbmVkKF9fdGh1bWJfXykKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAKLSAgICBHTGZpeGVkIHI7Ci0gICAgaW50MzJfdCB0OwotICAgIGFzbSgKLSAgICAgICAgInNtdWxsICUwLCAlMSwgJTIsICUzICAgICAgIFxuIgotICAgICAgICAic21sYWwgJTAsICUxLCAlNCwgJTUgICAgICAgXG4iCi0gICAgICAgICJzbWxhbCAlMCwgJTEsICU2LCAlNyAgICAgICBcbiIKLSAgICAgICAgImFkZCAgICUwLCAlOCwgJTAsIGxzciAjMTYgIFxuIgotICAgICAgICAiYWRkICAgJTAsICUwLCAlMSwgbHNsICMxNiAgXG4iCi0gICAgICAgIDogICAiPSZyIihyKSwgIj0mciIodCkgCi0gICAgICAgIDogICAiJXIiKGEwKSwgInIiKGIwKSwKLSAgICAgICAgICAgICIlciIoYTEpLCAiciIoYjEpLAotICAgICAgICAgICAgIiVyIihhMiksICJyIihiMiksCi0gICAgICAgICAgICAiciIoYykKLSAgICAgICAgOgotICAgICAgICApOyAKLSAgICByZXR1cm4gcjsKLSAgICAKLSNlbHNlCi0KLSAgICByZXR1cm4gKCggICBpbnQ2NF90KGEwKSpiMCArCi0gICAgICAgICAgICAgICAgaW50NjRfdChhMSkqYjEgKwotICAgICAgICAgICAgICAgIGludDY0X3QoYTIpKmIyKT4+MTYpICsgYzsKLQotI2VuZGlmCi19Ci0KLS8vIGIwLCBiMSwgYjIgYXJlIHNpZ25lZCAxNi1iaXQgcXVhbml0aWVzCi0vLyB0aGF0IGhhdmUgYmVlbiBzaGlmdGVkIHJpZ2h0IGJ5ICdzaGlmdCcgYml0cyByZWxhdGl2ZSB0byBub3JtYWwKLS8vIFMxNi4xNiBmaXhlZCBwb2ludAotc3RhdGljIGlubGluZSBHTGZpeGVkIG1sYTNhMTYoIEdMZml4ZWQgYTAsIGludDMyX3QgYjFiMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBHTGZpeGVkIGExLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMZml4ZWQgYTIsIGludDMyX3QgYjIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xpbnQgc2hpZnQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xmaXhlZCBjKQotewotI2lmIGRlZmluZWQoX19hcm1fXykgJiYgIWRlZmluZWQoX190aHVtYl9fKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgIEdMZml4ZWQgcjsKLSAgICBhc20oCi0gICAgICAgICJzbXVsd2IgJTAsICUxLCAlMiAgICAgICAgICBcbiIKLSAgICAgICAgInNtbGF3dCAlMCwgJTMsICUyLCAlMCAgICAgIFxuIiAKLSAgICAgICAgInNtbGF3YiAlMCwgJTQsICU1LCAlMCAgICAgIFxuIgotICAgICAgICAiYWRkICAgICUwLCAlNywgJTAsIGxzbCAlNiAgXG4iCi0gICAgICAgIDogICAiPSZyIihyKQotICAgICAgICA6ICAgInIiKGEwKSwgInIiKGIxYjApLAotICAgICAgICAgICAgInIiKGExKSwKLSAgICAgICAgICAgICJyIihhMiksICJyIihiMiksCi0gICAgICAgICAgICAiciIoc2hpZnQpLAotICAgICAgICAgICAgInIiKGMpCi0gICAgICAgIDoKLSAgICAgICAgKTsgCi0gICAgcmV0dXJuIHI7Ci0gICAgCi0jZWxzZQotCi0gICAgaW50MzJfdCBhY2N1bTsKLSAgICBpbnQxNl90IGIwID0gYjFiMCAmIDB4ZmZmZjsKLSAgICBpbnQxNl90IGIxID0gKGIxYjAgPj4gMTYpICYgMHhmZmZmOwotICAgIGFjY3VtICA9IGludDY0X3QoYTApKmludDE2X3QoYjApID4+IDE2OwotICAgIGFjY3VtICs9IGludDY0X3QoYTEpKmludDE2X3QoYjEpID4+IDE2OwotICAgIGFjY3VtICs9IGludDY0X3QoYTIpKmludDE2X3QoYjIpID4+IDE2OwotICAgIGFjY3VtID0gKGFjY3VtIDw8IHNoaWZ0KSArIGM7Ci0gICAgcmV0dXJuIGFjY3VtOwotCi0jZW5kaWYKLX0KLQotCi1zdGF0aWMgaW5saW5lIEdMZml4ZWQgbWxhM2ExNl9idGIoIEdMZml4ZWQgYTAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMZml4ZWQgYTEsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMZml4ZWQgYTIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgYjFiMCwgaW50MzJfdCB4eGIyLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBHTGludCBzaGlmdCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xmaXhlZCBjKQotewotI2lmIGRlZmluZWQoX19hcm1fXykgJiYgIWRlZmluZWQoX190aHVtYl9fKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgIEdMZml4ZWQgcjsKLSAgICBhc20oCi0gICAgICAgICJzbXVsd2IgJTAsICUxLCAlNCAgICAgICAgICBcbiIKLSAgICAgICAgInNtbGF3dCAlMCwgJTIsICU0LCAlMCAgICAgIFxuIiAKLSAgICAgICAgInNtbGF3YiAlMCwgJTMsICU1LCAlMCAgICAgIFxuIgotICAgICAgICAiYWRkICAgICUwLCAlNywgJTAsIGxzbCAlNiAgXG4iCi0gICAgICAgIDogICAiPSZyIihyKQotICAgICAgICA6ICAgInIiKGEwKSwKLSAgICAgICAgICAgICJyIihhMSksCi0gICAgICAgICAgICAiciIoYTIpLAotICAgICAgICAgICAgInIiKGIxYjApLCAiciIoeHhiMiksCi0gICAgICAgICAgICAiciIoc2hpZnQpLAotICAgICAgICAgICAgInIiKGMpCi0gICAgICAgIDoKLSAgICAgICAgKTsgCi0gICAgcmV0dXJuIHI7Ci0gICAgCi0jZWxzZQotCi0gICAgaW50MzJfdCBhY2N1bTsKLSAgICBpbnQxNl90IGIwID0gIGIxYjAgICAgICAgICYgMHhmZmZmOwotICAgIGludDE2X3QgYjEgPSAoYjFiMCA+PiAxNikgJiAweGZmZmY7Ci0gICAgaW50MTZfdCBiMiA9ICB4eGIyICAgICAgICAmIDB4ZmZmZjsKLSAgICBhY2N1bSAgPSBpbnQ2NF90KGEwKSppbnQxNl90KGIwKSA+PiAxNjsKLSAgICBhY2N1bSArPSBpbnQ2NF90KGExKSppbnQxNl90KGIxKSA+PiAxNjsKLSAgICBhY2N1bSArPSBpbnQ2NF90KGEyKSppbnQxNl90KGIyKSA+PiAxNjsKLSAgICBhY2N1bSA9IChhY2N1bSA8PCBzaGlmdCkgKyBjOwotICAgIHJldHVybiBhY2N1bTsKLQotI2VuZGlmCi19Ci0KLXN0YXRpYyBpbmxpbmUgR0xmaXhlZCBtbGEzYTE2X2J0dCggR0xmaXhlZCBhMCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xmaXhlZCBhMSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xmaXhlZCBhMiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCBiMWIwLCBpbnQzMl90IGIyeHgsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMaW50IHNoaWZ0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBHTGZpeGVkIGMpCi17Ci0jaWYgZGVmaW5lZChfX2FybV9fKSAmJiAhZGVmaW5lZChfX3RodW1iX18pCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgR0xmaXhlZCByOwotICAgIGFzbSgKLSAgICAgICAgInNtdWx3YiAlMCwgJTEsICU0ICAgICAgICAgIFxuIgotICAgICAgICAic21sYXd0ICUwLCAlMiwgJTQsICUwICAgICAgXG4iIAotICAgICAgICAic21sYXd0ICUwLCAlMywgJTUsICUwICAgICAgXG4iCi0gICAgICAgICJhZGQgICAgJTAsICU3LCAlMCwgbHNsICU2ICBcbiIKLSAgICAgICAgOiAgICI9JnIiKHIpCi0gICAgICAgIDogICAiciIoYTApLAotICAgICAgICAgICAgInIiKGExKSwKLSAgICAgICAgICAgICJyIihhMiksCi0gICAgICAgICAgICAiciIoYjFiMCksICJyIihiMnh4KSwKLSAgICAgICAgICAgICJyIihzaGlmdCksCi0gICAgICAgICAgICAiciIoYykKLSAgICAgICAgOgotICAgICAgICApOyAKLSAgICByZXR1cm4gcjsKLSAgICAKLSNlbHNlCi0KLSAgICBpbnQzMl90IGFjY3VtOwotICAgIGludDE2X3QgYjAgPSAgYjFiMCAgICAgICAgJiAweGZmZmY7Ci0gICAgaW50MTZfdCBiMSA9IChiMWIwID4+IDE2KSAmIDB4ZmZmZjsKLSAgICBpbnQxNl90IGIyID0gKGIyeHggPj4gMTYpICYgMHhmZmZmOwotICAgIGFjY3VtICA9IGludDY0X3QoYTApKmludDE2X3QoYjApID4+IDE2OwotICAgIGFjY3VtICs9IGludDY0X3QoYTEpKmludDE2X3QoYjEpID4+IDE2OwotICAgIGFjY3VtICs9IGludDY0X3QoYTIpKmludDE2X3QoYjIpID4+IDE2OwotICAgIGFjY3VtID0gKGFjY3VtIDw8IHNoaWZ0KSArIGM7Ci0gICAgcmV0dXJuIGFjY3VtOwotCi0jZW5kaWYKLX0KLQotc3RhdGljIGlubGluZSBHTGZpeGVkIG1sYTMoIEdMZml4ZWQgYTAsIEdMZml4ZWQgYjAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xmaXhlZCBhMSwgR0xmaXhlZCBiMSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBHTGZpeGVkIGEyLCBHTGZpeGVkIGIyKQotewotI2lmIGRlZmluZWQoX19hcm1fXykgJiYgIWRlZmluZWQoX190aHVtYl9fKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotICAgIEdMZml4ZWQgcjsKLSAgICBpbnQzMl90IHQ7Ci0gICAgYXNtKAotICAgICAgICAic211bGwgJTAsICUxLCAlMiwgJTMgICAgICAgXG4iCi0gICAgICAgICJzbWxhbCAlMCwgJTEsICU0LCAlNSAgICAgICBcbiIKLSAgICAgICAgInNtbGFsICUwLCAlMSwgJTYsICU3ICAgICAgIFxuIgotICAgICAgICAibW92cyAgJTAsICUwLCBsc3IgIzE2ICAgICAgXG4iCi0gICAgICAgICJhZGMgICAlMCwgJTAsICUxLCBsc2wgIzE2ICBcbiIKLSAgICAgICAgOiAgICI9JnIiKHIpLCAiPSZyIih0KSAKLSAgICAgICAgOiAgICIlciIoYTApLCAiciIoYjApLAotICAgICAgICAgICAgIiVyIihhMSksICJyIihiMSksCi0gICAgICAgICAgICAiJXIiKGEyKSwgInIiKGIyKQotICAgICAgICA6ICAgImNjIgotICAgICAgICApOyAKLSAgICByZXR1cm4gcjsKLSAgICAKLSNlbHNlCi0KLSAgICByZXR1cm4gKCggICBpbnQ2NF90KGEwKSpiMCArCi0gICAgICAgICAgICAgICAgaW50NjRfdChhMSkqYjEgKwotICAgICAgICAgICAgICAgIGludDY0X3QoYTIpKmIyICsgMHg4MDAwKT4+MTYpOwotCi0jZW5kaWYKLX0KLQotc3RhdGljIGlubGluZSBHTGZpeGVkIG1sYTQoIEdMZml4ZWQgYTAsIEdMZml4ZWQgYjAsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xmaXhlZCBhMSwgR0xmaXhlZCBiMSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBHTGZpeGVkIGEyLCBHTGZpeGVkIGIyLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMZml4ZWQgYTMsIEdMZml4ZWQgYjMpCi17Ci0jaWYgZGVmaW5lZChfX2FybV9fKSAmJiAhZGVmaW5lZChfX3RodW1iX18pCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgCi0gICAgR0xmaXhlZCByOwotICAgIGludDMyX3QgdDsKLSAgICBhc20oCi0gICAgICAgICJzbXVsbCAlMCwgJTEsICUyLCAlMyAgICAgICBcbiIKLSAgICAgICAgInNtbGFsICUwLCAlMSwgJTQsICU1ICAgICAgIFxuIgotICAgICAgICAic21sYWwgJTAsICUxLCAlNiwgJTcgICAgICAgXG4iCi0gICAgICAgICJzbWxhbCAlMCwgJTEsICU4LCAlOSAgICAgICBcbiIKLSAgICAgICAgIm1vdnMgICUwLCAlMCwgbHNyICMxNiAgICAgIFxuIgotICAgICAgICAiYWRjICAgJTAsICUwLCAlMSwgbHNsICMxNiAgXG4iCi0gICAgICAgIDogICAiPSZyIihyKSwgIj0mciIodCkgCi0gICAgICAgIDogICAiJXIiKGEwKSwgInIiKGIwKSwKLSAgICAgICAgICAgICIlciIoYTEpLCAiciIoYjEpLAotICAgICAgICAgICAgIiVyIihhMiksICJyIihiMiksCi0gICAgICAgICAgICAiJXIiKGEzKSwgInIiKGIzKQotICAgICAgICA6ICAgImNjIgotICAgICAgICApOyAKLSAgICByZXR1cm4gcjsKLSAgICAKLSNlbHNlCi0KLSAgICByZXR1cm4gKCggICBpbnQ2NF90KGEwKSpiMCArCi0gICAgICAgICAgICAgICAgaW50NjRfdChhMSkqYjEgKwotICAgICAgICAgICAgICAgIGludDY0X3QoYTIpKmIyICsKLSAgICAgICAgICAgICAgICBpbnQ2NF90KGEzKSpiMyArIDB4ODAwMCk+PjE2KTsKLQotI2VuZGlmCi19Ci0KLWlubGluZQotR0xmaXhlZCBkb3Q0KGNvbnN0IEdMZml4ZWQqIGEsIGNvbnN0IEdMZml4ZWQqIGIpIAotewotICAgIHJldHVybiBtbGE0KGFbMF0sIGJbMF0sIGFbMV0sIGJbMV0sIGFbMl0sIGJbMl0sIGFbM10sIGJbM10pOwotfQotCi0KLWlubGluZQotR0xmaXhlZCBkb3QzKGNvbnN0IEdMZml4ZWQqIGEsIGNvbnN0IEdMZml4ZWQqIGIpIAotewotICAgIHJldHVybiBtbGEzKGFbMF0sIGJbMF0sIGFbMV0sIGJbMV0sIGFbMl0sIGJbMl0pOwotfQotCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX09QRU5HTEVTX01BVFJJWF9ICi0KZGlmZiAtLWdpdCBhL29wZW5nbC9saWJhZ2wvbWlwbWFwLmNwcCBiL29wZW5nbC9saWJhZ2wvbWlwbWFwLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggY2NkNzdiNy4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGliYWdsL21pcG1hcC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxOTIgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9taXBtYXAuY3BwCi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0KLSNpbmNsdWRlICJjb250ZXh0LmgiCi0jaW5jbHVkZSAic3RhdGUuaCIKLSNpbmNsdWRlICJ0ZXh0dXJlLmgiCi0jaW5jbHVkZSAiVGV4dHVyZU9iamVjdE1hbmFnZXIuaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXR1c190IGJ1aWxkQVB5cmFtaWQob2dsZXNfY29udGV4dF90KiBjLCBFR0xUZXh0dXJlT2JqZWN0KiB0ZXgpCi17Ci0gICAgaW50IGxldmVsID0gMDsKLSAgICBjb25zdCBHR0xTdXJmYWNlKiBiYXNlID0gJnRleC0+c3VyZmFjZTsgICAgCi0gICAgY29uc3QgR0dMRm9ybWF0JiBwaXhlbEZvcm1hdChjLT5yYXN0ZXJpemVyLmZvcm1hdHNbYmFzZS0+Zm9ybWF0XSk7Ci0KLSAgICBpbnQgdyA9IGJhc2UtPndpZHRoOwotICAgIGludCBoID0gYmFzZS0+aGVpZ2h0OwotICAgIGlmICgodyZoKSA9PSAxKQotICAgICAgICByZXR1cm4gTk9fRVJST1I7Ci0KLSAgICB3ID0gKHc+PjEpID8gOiAxOwotICAgIGggPSAoaD4+MSkgPyA6IDE7Ci0KLSAgICB3aGlsZSh0cnVlKSB7Ci0gICAgICAgICsrbGV2ZWw7Ci0gICAgICAgIGNvbnN0IGludCBicHIgPSB3ICogcGl4ZWxGb3JtYXQuc2l6ZTsKLSAgICAgICAgaWYgKHRleC0+cmVhbGxvY2F0ZShsZXZlbCwgdywgaCwgdywKLSAgICAgICAgICAgICAgICBiYXNlLT5mb3JtYXQsIGJhc2UtPmNvbXByZXNzZWRGb3JtYXQsIGJwcikgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgICAgIHJldHVybiBOT19NRU1PUlk7Ci0gICAgICAgIH0KLSAgICAKLSAgICAgICAgaW50IHN0cmlkZSA9IHc7Ci0gICAgICAgIGludCBicyA9IGJhc2UtPnN0cmlkZTsKLSAgICAgICAgR0dMU3VyZmFjZSYgY3VyID0gdGV4LT5lZGl0TWlwKGxldmVsKTsKLQotICAgICAgICBpZiAoYmFzZS0+Zm9ybWF0ID09IEdHTF9QSVhFTF9GT1JNQVRfUkdCXzU2NSkKLSAgICAgICAgewotICAgICAgICAgICAgdWludDE2X3QgY29uc3QgKiBzcmMgPSAodWludDE2X3QgY29uc3QgKiliYXNlLT5kYXRhOwotICAgICAgICAgICAgdWludDE2X3QqIGRzdCA9ICh1aW50MTZfdCopY3VyLmRhdGE7Ci0gICAgICAgICAgICBjb25zdCB1aW50MzJfdCBtYXNrID0gMHgwN0UwRjgxRjsKLSAgICAgICAgICAgIGZvciAoaW50IHk9MCA7IHk8aCA7IHkrKykgewotICAgICAgICAgICAgICAgIHNpemVfdCBvZmZzZXQgPSAoeSoyKSAqIGJzOwotICAgICAgICAgICAgICAgIGZvciAoaW50IHg9MCA7IHg8dyA7IHgrKykgewotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBwMDAgPSBzcmNbb2Zmc2V0XTsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcDEwID0gc3JjW29mZnNldCsxXTsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcDAxID0gc3JjW29mZnNldCtic107Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHAxMSA9IHNyY1tvZmZzZXQrYnMrMV07Ci0gICAgICAgICAgICAgICAgICAgIHAwMCA9IChwMDAgfCAocDAwIDw8IDE2KSkgJiBtYXNrOwotICAgICAgICAgICAgICAgICAgICBwMDEgPSAocDAxIHwgKHAwMSA8PCAxNikpICYgbWFzazsKLSAgICAgICAgICAgICAgICAgICAgcDEwID0gKHAxMCB8IChwMTAgPDwgMTYpKSAmIG1hc2s7Ci0gICAgICAgICAgICAgICAgICAgIHAxMSA9IChwMTEgfCAocDExIDw8IDE2KSkgJiBtYXNrOwotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBncmIgPSAoKHAwMCArIHAxMCArIHAwMSArIHAxMSkgPj4gMikgJiBtYXNrOwotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCByZ2IgPSAoZ3JiICYgMHhGRkZGKSB8IChncmIgPj4gMTYpOwotICAgICAgICAgICAgICAgICAgICBkc3RbeCArIHkqc3RyaWRlXSA9IHJnYjsKLSAgICAgICAgICAgICAgICAgICAgb2Zmc2V0ICs9IDI7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGVsc2UgaWYgKGJhc2UtPmZvcm1hdCA9PSBHR0xfUElYRUxfRk9STUFUX1JHQkFfNTU1MSkKLSAgICAgICAgewotICAgICAgICAgICAgdWludDE2X3QgY29uc3QgKiBzcmMgPSAodWludDE2X3QgY29uc3QgKiliYXNlLT5kYXRhOwotICAgICAgICAgICAgdWludDE2X3QqIGRzdCA9ICh1aW50MTZfdCopY3VyLmRhdGE7Ci0gICAgICAgICAgICBmb3IgKGludCB5PTAgOyB5PGggOyB5KyspIHsKLSAgICAgICAgICAgICAgICBzaXplX3Qgb2Zmc2V0ID0gKHkqMikgKiBiczsKLSAgICAgICAgICAgICAgICBmb3IgKGludCB4PTAgOyB4PHcgOyB4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcDAwID0gc3JjW29mZnNldF07Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHAxMCA9IHNyY1tvZmZzZXQrMV07Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHAwMSA9IHNyY1tvZmZzZXQrYnNdOwotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBwMTEgPSBzcmNbb2Zmc2V0K2JzKzFdOwotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCByID0gKChwMDA+PjExKSsocDEwPj4xMSkrKHAwMT4+MTEpKyhwMTE+PjExKSsyKT4+MjsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZyA9ICgoKHAwMD4+NikrKHAxMD4+NikrKHAwMT4+NikrKHAxMT4+NikrMik+PjIpJjB4M0Y7Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGIgPSAoKHAwMCYweDNFKSsocDEwJjB4M0UpKyhwMDEmMHgzRSkrKHAxMSYweDNFKSs0KT4+MzsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgYSA9ICgocDAwJjEpKyhwMTAmMSkrKHAwMSYxKSsocDExJjEpKzIpPj4yOwotICAgICAgICAgICAgICAgICAgICBkc3RbeCArIHkqc3RyaWRlXSA9IChyPDwxMSl8KGc8PDYpfChiPDwxKXxhOwotICAgICAgICAgICAgICAgICAgICBvZmZzZXQgKz0gMjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgZWxzZSBpZiAoYmFzZS0+Zm9ybWF0ID09IEdHTF9QSVhFTF9GT1JNQVRfUkdCQV84ODg4KQotICAgICAgICB7Ci0gICAgICAgICAgICB1aW50MzJfdCBjb25zdCAqIHNyYyA9ICh1aW50MzJfdCBjb25zdCAqKWJhc2UtPmRhdGE7Ci0gICAgICAgICAgICB1aW50MzJfdCogZHN0ID0gKHVpbnQzMl90KiljdXIuZGF0YTsKLSAgICAgICAgICAgIGZvciAoaW50IHk9MCA7IHk8aCA7IHkrKykgewotICAgICAgICAgICAgICAgIHNpemVfdCBvZmZzZXQgPSAoeSoyKSAqIGJzOwotICAgICAgICAgICAgICAgIGZvciAoaW50IHg9MCA7IHg8dyA7IHgrKykgewotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBwMDAgPSBzcmNbb2Zmc2V0XTsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcDEwID0gc3JjW29mZnNldCsxXTsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcDAxID0gc3JjW29mZnNldCtic107Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHAxMSA9IHNyY1tvZmZzZXQrYnMrMV07Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHJiMDAgPSBwMDAgJiAweDAwRkYwMEZGOwotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCByYjAxID0gcDAxICYgMHgwMEZGMDBGRjsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcmIxMCA9IHAxMCAmIDB4MDBGRjAwRkY7Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHJiMTEgPSBwMTEgJiAweDAwRkYwMEZGOwotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBnYTAwID0gKHAwMCA+PiA4KSAmIDB4MDBGRjAwRkY7Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IGdhMDEgPSAocDAxID4+IDgpICYgMHgwMEZGMDBGRjsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgZ2ExMCA9IChwMTAgPj4gOCkgJiAweDAwRkYwMEZGOwotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBnYTExID0gKHAxMSA+PiA4KSAmIDB4MDBGRjAwRkY7Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHJiID0gKHJiMDAgKyByYjAxICsgcmIxMCArIHJiMTEpPj4yOwotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBnYSA9IChnYTAwICsgZ2EwMSArIGdhMTAgKyBnYTExKT4+MjsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcmdiYSA9IChyYiAmIDB4MDBGRjAwRkYpIHwgKChnYSAmIDB4MDBGRjAwRkYpPDw4KTsKLSAgICAgICAgICAgICAgICAgICAgZHN0W3ggKyB5KnN0cmlkZV0gPSByZ2JhOwotICAgICAgICAgICAgICAgICAgICBvZmZzZXQgKz0gMjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgZWxzZSBpZiAoKGJhc2UtPmZvcm1hdCA9PSBHR0xfUElYRUxfRk9STUFUX1JHQl84ODgpIHx8Ci0gICAgICAgICAgICAgICAgIChiYXNlLT5mb3JtYXQgPT0gR0dMX1BJWEVMX0ZPUk1BVF9MQV84OCkgfHwKLSAgICAgICAgICAgICAgICAgKGJhc2UtPmZvcm1hdCA9PSBHR0xfUElYRUxfRk9STUFUX0FfOCkgfHwKLSAgICAgICAgICAgICAgICAgKGJhc2UtPmZvcm1hdCA9PSBHR0xfUElYRUxfRk9STUFUX0xfOCkpCi0gICAgICAgIHsKLSAgICAgICAgICAgIGludCBza2lwOwotICAgICAgICAgICAgc3dpdGNoIChiYXNlLT5mb3JtYXQpIHsKLSAgICAgICAgICAgIGNhc2UgR0dMX1BJWEVMX0ZPUk1BVF9SR0JfODg4OiAgc2tpcCA9IDM7ICAgYnJlYWs7Ci0gICAgICAgICAgICBjYXNlIEdHTF9QSVhFTF9GT1JNQVRfTEFfODg6ICAgIHNraXAgPSAyOyAgIGJyZWFrOwotICAgICAgICAgICAgZGVmYXVsdDogICAgICAgICAgICAgICAgICAgICAgICBza2lwID0gMTsgICBicmVhazsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIHVpbnQ4X3QgY29uc3QgKiBzcmMgPSAodWludDhfdCBjb25zdCAqKWJhc2UtPmRhdGE7Ci0gICAgICAgICAgICB1aW50OF90KiBkc3QgPSAodWludDhfdCopY3VyLmRhdGE7ICAgICAgICAgICAgCi0gICAgICAgICAgICBicyAqPSBza2lwOwotICAgICAgICAgICAgc3RyaWRlICo9IHNraXA7Ci0gICAgICAgICAgICBmb3IgKGludCB5PTAgOyB5PGggOyB5KyspIHsKLSAgICAgICAgICAgICAgICBzaXplX3Qgb2Zmc2V0ID0gKHkqMikgKiBiczsKLSAgICAgICAgICAgICAgICBmb3IgKGludCB4PTAgOyB4PHcgOyB4KyspIHsKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgYz0wIDsgYzxza2lwIDsgYysrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBwMDAgPSBzcmNbYytvZmZzZXRdOwotICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcDEwID0gc3JjW2Mrb2Zmc2V0K3NraXBdOwotICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcDAxID0gc3JjW2Mrb2Zmc2V0K2JzXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHAxMSA9IHNyY1tjK29mZnNldCticytza2lwXTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGRzdFt4ICsgeSpzdHJpZGUgKyBjXSA9IChwMDAgKyBwMTAgKyBwMDEgKyBwMTEpID4+IDI7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgb2Zmc2V0ICs9IDIqc2tpcDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICAgICAgZWxzZSBpZiAoYmFzZS0+Zm9ybWF0ID09IEdHTF9QSVhFTF9GT1JNQVRfUkdCQV80NDQ0KQotICAgICAgICB7Ci0gICAgICAgICAgICB1aW50MTZfdCBjb25zdCAqIHNyYyA9ICh1aW50MTZfdCBjb25zdCAqKWJhc2UtPmRhdGE7Ci0gICAgICAgICAgICB1aW50MTZfdCogZHN0ID0gKHVpbnQxNl90KiljdXIuZGF0YTsKLSAgICAgICAgICAgIGZvciAoaW50IHk9MCA7IHk8aCA7IHkrKykgewotICAgICAgICAgICAgICAgIHNpemVfdCBvZmZzZXQgPSAoeSoyKSAqIGJzOwotICAgICAgICAgICAgICAgIGZvciAoaW50IHg9MCA7IHg8dyA7IHgrKykgewotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBwMDAgPSBzcmNbb2Zmc2V0XTsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcDEwID0gc3JjW29mZnNldCsxXTsKLSAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgcDAxID0gc3JjW29mZnNldCtic107Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHAxMSA9IHNyY1tvZmZzZXQrYnMrMV07Ci0gICAgICAgICAgICAgICAgICAgIHAwMCA9ICgocDAwIDw8IDEyKSAmIDB4MEYwRjAwMDApIHwgKHAwMCAmIDB4MEYwRik7Ci0gICAgICAgICAgICAgICAgICAgIHAxMCA9ICgocDEwIDw8IDEyKSAmIDB4MEYwRjAwMDApIHwgKHAxMCAmIDB4MEYwRik7Ci0gICAgICAgICAgICAgICAgICAgIHAwMSA9ICgocDAxIDw8IDEyKSAmIDB4MEYwRjAwMDApIHwgKHAwMSAmIDB4MEYwRik7Ci0gICAgICAgICAgICAgICAgICAgIHAxMSA9ICgocDExIDw8IDEyKSAmIDB4MEYwRjAwMDApIHwgKHAxMSAmIDB4MEYwRik7Ci0gICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IHJiZ2EgPSAocDAwICsgcDEwICsgcDAxICsgcDExKSA+PiAyOwotICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCByZ2JhID0gKHJiZ2EgJiAweDBGMEYpIHwgKChyYmdhPj4xMikgJiAweEYwRjApOwotICAgICAgICAgICAgICAgICAgICBkc3RbeCArIHkqc3RyaWRlXSA9IHJnYmE7Ci0gICAgICAgICAgICAgICAgICAgIG9mZnNldCArPSAyOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIExPR0UoIlVuc3VwcG9ydGVkIGZvcm1hdCAoJWQpIiwgYmFzZS0+Zm9ybWF0KTsKLSAgICAgICAgICAgIHJldHVybiBCQURfVFlQRTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIGV4aXQgY29uZGl0aW9uOiB3ZSBqdXN0IHByb2Nlc3NlZCB0aGUgMXgxIExPRHMKLSAgICAgICAgaWYgKCh3JmgpID09IDEpCi0gICAgICAgICAgICBicmVhazsKLQotICAgICAgICBiYXNlID0gJmN1cjsKLSAgICAgICAgdyA9ICh3Pj4xKSA/IDogMTsKLSAgICAgICAgaCA9IChoPj4xKSA/IDogMTsKLSAgICB9Ci0gICAgcmV0dXJuIE5PX0VSUk9SOwotfQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZApkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC9wcmltaXRpdmVzLmNwcCBiL29wZW5nbC9saWJhZ2wvcHJpbWl0aXZlcy5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGYxNjRjMDIuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC9wcmltaXRpdmVzLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDExMTEgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9wcmltaXRpdmVzLmNwcAotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPG1hdGguaD4KLQotI2luY2x1ZGUgImNvbnRleHQuaCIKLSNpbmNsdWRlICJwcmltaXRpdmVzLmgiCi0jaW5jbHVkZSAibGlnaHQuaCIKLSNpbmNsdWRlICJtYXRyaXguaCIKLSNpbmNsdWRlICJ2ZXJ0ZXguaCIKLSNpbmNsdWRlICJmcC5oIgotI2luY2x1ZGUgIlRleHR1cmVPYmplY3RNYW5hZ2VyLmgiCi0KLWV4dGVybiAiQyIgdm9pZCBpdGVyYXRvcnMwMDMyKGNvbnN0IHZvaWQqIHRoYXQsCi0gICAgICAgIGludDMyX3QqIGl0LCBpbnQzMl90IGMwLCBpbnQzMl90IGMxLCBpbnQzMl90IGMyKTsKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyB2b2lkIHByaW1pdGl2ZV9wb2ludChvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiB2KTsKLXN0YXRpYyB2b2lkIHByaW1pdGl2ZV9saW5lKG9nbGVzX2NvbnRleHRfdCogYywgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEpOwotc3RhdGljIHZvaWQgcHJpbWl0aXZlX2NsaXBfdHJpYW5nbGUob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICB2ZXJ0ZXhfdCogdjAsIHZlcnRleF90KiB2MSwgdmVydGV4X3QqIHYyKTsKLQotc3RhdGljIHZvaWQgcHJpbWl0aXZlX25vcF9wb2ludChvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiB2KTsKLXN0YXRpYyB2b2lkIHByaW1pdGl2ZV9ub3BfbGluZShvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxKTsKLXN0YXRpYyB2b2lkIHByaW1pdGl2ZV9ub3BfdHJpYW5nbGUob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICB2ZXJ0ZXhfdCogdjAsIHZlcnRleF90KiB2MSwgdmVydGV4X3QqIHYyKTsKLQotc3RhdGljIGlubGluZSBib29sIGN1bGxfdHJpYW5nbGUob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICB2ZXJ0ZXhfdCogdjAsIHZlcnRleF90KiB2MSwgdmVydGV4X3QqIHYyKTsKLQotc3RhdGljIHZvaWQgbGVycF90cmlhbmdsZShvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpOwotCi1zdGF0aWMgdm9pZCBsZXJwX3RleGNvb3JkcyhvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpOwotCi1zdGF0aWMgdm9pZCBsZXJwX3RleGNvb3Jkc193KG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2Mik7Ci0KLXN0YXRpYyB2b2lkIHRyaWFuZ2xlKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2Mik7Ci0KLXN0YXRpYyB2b2lkIGNsaXBfdHJpYW5nbGUob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICB2ZXJ0ZXhfdCogdjAsIHZlcnRleF90KiB2MSwgdmVydGV4X3QqIHYyKTsKLQotc3RhdGljIHVuc2lnbmVkIGludCBjbGlwX2xpbmUob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICB2ZXJ0ZXhfdCogcywgdmVydGV4X3QqIHApOwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNlbmRpZgotCi1zdGF0aWMgdm9pZCBsaWdodFRyaWFuZ2xlRGFya1Ntb290aChvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpCi17Ci0gICAgaWYgKCEodjAtPmZsYWdzICYgdmVydGV4X3Q6OkxJVCkpIHsKLSAgICAgICAgdjAtPmZsYWdzIHw9IHZlcnRleF90OjpMSVQ7Ci0gICAgICAgIGNvbnN0IEdMdm9pZCogY3AgPSBjLT5hcnJheXMuY29sb3IuZWxlbWVudCgKLSAgICAgICAgICAgICAgICB2MC0+aW5kZXggJiB2ZXJ0ZXhfY2FjaGVfdDo6SU5ERVhfTUFTSyk7Ci0gICAgICAgIGMtPmFycmF5cy5jb2xvci5mZXRjaChjLCB2MC0+Y29sb3IudiwgY3ApOwotICAgIH0KLSAgICBpZiAoISh2MS0+ZmxhZ3MgJiB2ZXJ0ZXhfdDo6TElUKSkgewotICAgICAgICB2MS0+ZmxhZ3MgfD0gdmVydGV4X3Q6OkxJVDsKLSAgICAgICAgY29uc3QgR0x2b2lkKiBjcCA9IGMtPmFycmF5cy5jb2xvci5lbGVtZW50KAotICAgICAgICAgICAgICAgIHYxLT5pbmRleCAmIHZlcnRleF9jYWNoZV90OjpJTkRFWF9NQVNLKTsKLSAgICAgICAgYy0+YXJyYXlzLmNvbG9yLmZldGNoKGMsIHYxLT5jb2xvci52LCBjcCk7Ci0gICAgfQotICAgIGlmKCEodjItPmZsYWdzICYgdmVydGV4X3Q6OkxJVCkpIHsKLSAgICAgICAgdjItPmZsYWdzIHw9IHZlcnRleF90OjpMSVQ7Ci0gICAgICAgIGNvbnN0IEdMdm9pZCogY3AgPSBjLT5hcnJheXMuY29sb3IuZWxlbWVudCgKLSAgICAgICAgICAgICAgICB2Mi0+aW5kZXggJiB2ZXJ0ZXhfY2FjaGVfdDo6SU5ERVhfTUFTSyk7Ci0gICAgICAgIGMtPmFycmF5cy5jb2xvci5mZXRjaChjLCB2Mi0+Y29sb3IudiwgY3ApOwotICAgIH0KLX0KLQotc3RhdGljIHZvaWQgbGlnaHRUcmlhbmdsZURhcmtGbGF0KG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2MikKLXsKLSAgICBpZiAoISh2Mi0+ZmxhZ3MgJiB2ZXJ0ZXhfdDo6TElUKSkgewotICAgICAgICB2Mi0+ZmxhZ3MgfD0gdmVydGV4X3Q6OkxJVDsKLSAgICAgICAgY29uc3QgR0x2b2lkKiBjcCA9IGMtPmFycmF5cy5jb2xvci5lbGVtZW50KAotICAgICAgICAgICAgICAgIHYyLT5pbmRleCAmIHZlcnRleF9jYWNoZV90OjpJTkRFWF9NQVNLKTsKLSAgICAgICAgYy0+YXJyYXlzLmNvbG9yLmZldGNoKGMsIHYyLT5jb2xvci52LCBjcCk7Ci0gICAgfQotICAgIC8vIGNvbmZpZ3VyZSB0aGUgcmFzdGVyaXplciBoZXJlLCBiZWZvcmUgd2UgY2xpcAotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuY29sb3I0eHYoYywgdjItPmNvbG9yLnYpOwotfQotCi1zdGF0aWMgdm9pZCBsaWdodFRyaWFuZ2xlU21vb3RoKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2MikKLXsKLSAgICBpZiAoISh2MC0+ZmxhZ3MgJiB2ZXJ0ZXhfdDo6TElUKSkKLSAgICAgICAgYy0+bGlnaHRpbmcubGlnaHRWZXJ0ZXgoYywgdjApOwotICAgIGlmICghKHYxLT5mbGFncyAmIHZlcnRleF90OjpMSVQpKQotICAgICAgICBjLT5saWdodGluZy5saWdodFZlcnRleChjLCB2MSk7Ci0gICAgaWYoISh2Mi0+ZmxhZ3MgJiB2ZXJ0ZXhfdDo6TElUKSkKLSAgICAgICAgYy0+bGlnaHRpbmcubGlnaHRWZXJ0ZXgoYywgdjIpOwotfQotCi1zdGF0aWMgdm9pZCBsaWdodFRyaWFuZ2xlRmxhdChvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpCi17Ci0gICAgaWYgKCEodjItPmZsYWdzICYgdmVydGV4X3Q6OkxJVCkpCi0gICAgICAgIGMtPmxpZ2h0aW5nLmxpZ2h0VmVydGV4KGMsIHYyKTsKLSAgICAvLyBjb25maWd1cmUgdGhlIHJhc3Rlcml6ZXIgaGVyZSwgYmVmb3JlIHdlIGNsaXAKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmNvbG9yNHh2KGMsIHYyLT5jb2xvci52KTsKLX0KLQotLy8gVGhlIGZvZyB2ZXJzaW9ucy4uLgotCi1zdGF0aWMgaW5saW5lCi12b2lkIGxpZ2h0VmVydGV4RGFya1Ntb290aEZvZyhvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiB2KQotewotICAgIGlmICghKHYtPmZsYWdzICYgdmVydGV4X3Q6OkxJVCkpIHsKLSAgICAgICAgdi0+ZmxhZ3MgfD0gdmVydGV4X3Q6OkxJVDsKLSAgICAgICAgdi0+Zm9nID0gYy0+Zm9nLmZvZyhjLCB2LT5leWUueik7Ci0gICAgICAgIGNvbnN0IEdMdm9pZCogY3AgPSBjLT5hcnJheXMuY29sb3IuZWxlbWVudCgKLSAgICAgICAgICAgICAgICB2LT5pbmRleCAmIHZlcnRleF9jYWNoZV90OjpJTkRFWF9NQVNLKTsKLSAgICAgICAgYy0+YXJyYXlzLmNvbG9yLmZldGNoKGMsIHYtPmNvbG9yLnYsIGNwKTsKLSAgICB9Ci19Ci1zdGF0aWMgaW5saW5lCi12b2lkIGxpZ2h0VmVydGV4RGFya0ZsYXRGb2cob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikKLXsKLSAgICBpZiAoISh2LT5mbGFncyAmIHZlcnRleF90OjpMSVQpKSB7Ci0gICAgICAgIHYtPmZsYWdzIHw9IHZlcnRleF90OjpMSVQ7Ci0gICAgICAgIHYtPmZvZyA9IGMtPmZvZy5mb2coYywgdi0+ZXllLnopOwotICAgIH0KLX0KLXN0YXRpYyBpbmxpbmUKLXZvaWQgbGlnaHRWZXJ0ZXhTbW9vdGhGb2cob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikKLXsKLSAgICBpZiAoISh2LT5mbGFncyAmIHZlcnRleF90OjpMSVQpKSB7Ci0gICAgICAgIHYtPmZvZyA9IGMtPmZvZy5mb2coYywgdi0+ZXllLnopOwotICAgICAgICBjLT5saWdodGluZy5saWdodFZlcnRleChjLCB2KTsKLSAgICB9Ci19Ci0KLXN0YXRpYyB2b2lkIGxpZ2h0VHJpYW5nbGVEYXJrU21vb3RoRm9nKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2MikKLXsKLSAgICBsaWdodFZlcnRleERhcmtTbW9vdGhGb2coYywgdjApOwotICAgIGxpZ2h0VmVydGV4RGFya1Ntb290aEZvZyhjLCB2MSk7Ci0gICAgbGlnaHRWZXJ0ZXhEYXJrU21vb3RoRm9nKGMsIHYyKTsKLX0KLQotc3RhdGljIHZvaWQgbGlnaHRUcmlhbmdsZURhcmtGbGF0Rm9nKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2MikKLXsKLSAgICBsaWdodFZlcnRleERhcmtGbGF0Rm9nKGMsIHYwKTsKLSAgICBsaWdodFZlcnRleERhcmtGbGF0Rm9nKGMsIHYxKTsKLSAgICBsaWdodFZlcnRleERhcmtTbW9vdGhGb2coYywgdjIpOwotICAgIC8vIGNvbmZpZ3VyZSB0aGUgcmFzdGVyaXplciBoZXJlLCBiZWZvcmUgd2UgY2xpcAotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuY29sb3I0eHYoYywgdjItPmNvbG9yLnYpOwotfQotCi1zdGF0aWMgdm9pZCBsaWdodFRyaWFuZ2xlU21vb3RoRm9nKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2MikKLXsKLSAgICBsaWdodFZlcnRleFNtb290aEZvZyhjLCB2MCk7Ci0gICAgbGlnaHRWZXJ0ZXhTbW9vdGhGb2coYywgdjEpOwotICAgIGxpZ2h0VmVydGV4U21vb3RoRm9nKGMsIHYyKTsKLX0KLQotc3RhdGljIHZvaWQgbGlnaHRUcmlhbmdsZUZsYXRGb2cob2dsZXNfY29udGV4dF90KiBjLAotICAgICAgICB2ZXJ0ZXhfdCogdjAsIHZlcnRleF90KiB2MSwgdmVydGV4X3QqIHYyKQotewotICAgIGxpZ2h0VmVydGV4RGFya0ZsYXRGb2coYywgdjApOwotICAgIGxpZ2h0VmVydGV4RGFya0ZsYXRGb2coYywgdjEpOwotICAgIGxpZ2h0VmVydGV4U21vb3RoRm9nKGMsIHYyKTsKLSAgICAvLyBjb25maWd1cmUgdGhlIHJhc3Rlcml6ZXIgaGVyZSwgYmVmb3JlIHdlIGNsaXAKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmNvbG9yNHh2KGMsIHYyLT5jb2xvci52KTsKLX0KLQotCi0KLXR5cGVkZWYgdm9pZCAoKmxpZ2h0X3ByaW1pdGl2ZV90KShvZ2xlc19jb250ZXh0X3QqLAotICAgICAgICB2ZXJ0ZXhfdCosIHZlcnRleF90KiwgdmVydGV4X3QqKTsKLQotLy8gZm9nIDB4NCwgbGlnaHQgMHgyLCBzbW9vdGggMHgxCi1zdGF0aWMgY29uc3QgbGlnaHRfcHJpbWl0aXZlX3QgbGlnaHRQcmltaXRpdmVbOF0gPSB7Ci0gICAgbGlnaHRUcmlhbmdsZURhcmtGbGF0LCAgICAgICAgICAvLyBubyBmb2cgfCBkYXJrICB8IGZsYXQKLSAgICBsaWdodFRyaWFuZ2xlRGFya1Ntb290aCwgICAgICAgIC8vIG5vIGZvZyB8IGRhcmsgIHwgc21vb3RoCi0gICAgbGlnaHRUcmlhbmdsZUZsYXQsICAgICAgICAgICAgICAvLyBubyBmb2cgfCBsaWdodCB8IGZsYXQKLSAgICBsaWdodFRyaWFuZ2xlU21vb3RoLCAgICAgICAgICAgIC8vIG5vIGZvZyB8IGxpZ2h0IHwgc21vb3RoCi0gICAgbGlnaHRUcmlhbmdsZURhcmtGbGF0Rm9nLCAgICAgICAvLyBmb2cgICAgfCBkYXJrICB8IGZsYXQKLSAgICBsaWdodFRyaWFuZ2xlRGFya1Ntb290aEZvZywgICAgIC8vIGZvZyAgICB8IGRhcmsgIHwgc21vb3RoCi0gICAgbGlnaHRUcmlhbmdsZUZsYXRGb2csICAgICAgICAgICAvLyBmb2cgICAgfCBsaWdodCB8IGZsYXQKLSAgICBsaWdodFRyaWFuZ2xlU21vb3RoRm9nICAgICAgICAgIC8vIGZvZyAgICB8IGxpZ2h0IHwgc21vb3RoCi19OwotCi12b2lkIG9nbGVzX3ZhbGlkYXRlX3ByaW1pdGl2ZXMob2dsZXNfY29udGV4dF90KiBjKQotewotICAgIGNvbnN0IHVpbnQzMl90IGVuYWJsZXMgPSBjLT5yYXN0ZXJpemVyLnN0YXRlLmVuYWJsZXM7Ci0KLSAgICAvLyBzZXQgdXAgdGhlIGxpZ2h0aW5nL3NoYWRpbmcvc21vb3RoaW5nL2ZvZ2dpbmcgZnVuY3Rpb24KLSAgICBpbnQgaW5kZXggPSBlbmFibGVzICYgR0dMX0VOQUJMRV9TTU9PVEggPyAweDEgOiAwOwotICAgIGluZGV4IHw9IGMtPmxpZ2h0aW5nLmVuYWJsZSA/IDB4MiA6IDA7Ci0gICAgaW5kZXggfD0gZW5hYmxlcyAmIEdHTF9FTkFCTEVfRk9HID8gMHg0IDogMDsKLSAgICBjLT5saWdodGluZy5saWdodFRyaWFuZ2xlID0gbGlnaHRQcmltaXRpdmVbaW5kZXhdOwotICAgIAotICAgIC8vIHNldCB1cCB0aGUgcHJpbWl0aXZlIHJlbmRlcmVycwotICAgIGlmIChnZ2xfbGlrZWx5KGMtPmFycmF5cy52ZXJ0ZXguZW5hYmxlKSkgewotICAgICAgICBjLT5wcmltcy5yZW5kZXJQb2ludCAgICA9IHByaW1pdGl2ZV9wb2ludDsKLSAgICAgICAgYy0+cHJpbXMucmVuZGVyTGluZSAgICAgPSBwcmltaXRpdmVfbGluZTsKLSAgICAgICAgYy0+cHJpbXMucmVuZGVyVHJpYW5nbGUgPSBwcmltaXRpdmVfY2xpcF90cmlhbmdsZTsKLSAgICB9IGVsc2UgewotICAgICAgICBjLT5wcmltcy5yZW5kZXJQb2ludCAgICA9IHByaW1pdGl2ZV9ub3BfcG9pbnQ7Ci0gICAgICAgIGMtPnByaW1zLnJlbmRlckxpbmUgICAgID0gcHJpbWl0aXZlX25vcF9saW5lOwotICAgICAgICBjLT5wcmltcy5yZW5kZXJUcmlhbmdsZSA9IHByaW1pdGl2ZV9ub3BfdHJpYW5nbGU7Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXZvaWQgY29tcHV0ZV9pdGVyYXRvcnNfdDo6aW5pdFRyaWFuZ2xlKAotICAgICAgICB2ZXJ0ZXhfdCBjb25zdCogdjAsIHZlcnRleF90IGNvbnN0KiB2MSwgdmVydGV4X3QgY29uc3QqIHYyKQotewotICAgIG1fZHgwMSA9IHYxLT53aW5kb3cueCAtIHYwLT53aW5kb3cueDsKLSAgICBtX2R5MTAgPSB2MC0+d2luZG93LnkgLSB2MS0+d2luZG93Lnk7Ci0gICAgbV9keDIwID0gdjAtPndpbmRvdy54IC0gdjItPndpbmRvdy54OwotICAgIG1fZHkwMiA9IHYyLT53aW5kb3cueSAtIHYwLT53aW5kb3cueTsKLSAgICBtX2FyZWEgPSBtX2R4MDEqbV9keTAyICsgKC1tX2R5MTApKm1fZHgyMDsKLX0KLQotdm9pZCBjb21wdXRlX2l0ZXJhdG9yc190Ojppbml0TGluZSgKLSAgICAgICAgdmVydGV4X3QgY29uc3QqIHYwLCB2ZXJ0ZXhfdCBjb25zdCogdjEpCi17Ci0gICAgbV9keDAxID0gbV9keTAyID0gdjEtPndpbmRvdy54IC0gdjAtPndpbmRvdy54OwotICAgIG1fZHkxMCA9IG1fZHgyMCA9IHYwLT53aW5kb3cueSAtIHYxLT53aW5kb3cueTsKLSAgICBtX2FyZWEgPSBtX2R4MDEqbV9keTAyICsgKC1tX2R5MTApKm1fZHgyMDsKLX0KLQotdm9pZCBjb21wdXRlX2l0ZXJhdG9yc190Ojppbml0TGVycCh2ZXJ0ZXhfdCBjb25zdCogdjAsIHVpbnQzMl90IGVuYWJsZXMpCi17Ci0gICAgbV94MCA9IHYwLT53aW5kb3cueDsKLSAgICBtX3kwID0gdjAtPndpbmRvdy55OwotICAgIGNvbnN0IEdHTGNvb3JkIGFyZWEgPSAobV9hcmVhICsgVFJJX0hBTEYpID4+IFRSSV9GUkFDVElPTl9CSVRTOwotICAgIGNvbnN0IEdHTGNvb3JkIG1pbkFyZWEgPSAyOyAvLyBjYW5ub3QgYmUgaW52ZXJ0ZWQKLSAgICAvLyB0cmlhbmdsZXMgd2l0aCBhbiBhcmVhIHNtYWxsZXIgdGhhbiAxLjAgYXJlIG5vdCBzbW9vdGgtc2hhZGVkCi0KLSAgICBpbnQgcT0wLCBzPTAsIGQ9MDsKLSAgICBpZiAoYWJzKGFyZWEpID49IG1pbkFyZWEpIHsKLSAgICAgICAgLy8gSGVyZSB3ZSBkbyBzb21lIHZvb2RvbyBtYWdpYywgdG8gY29tcHV0ZSBhIHN1aXRhYmxlIHNjYWxlCi0gICAgICAgIC8vIGZhY3RvciBmb3IgZGVsdGFzL2FyZWE6Ci0KLSAgICAgICAgLy8gRmlyc3QgY29tcHV0ZSB0aGUgMS9hcmVhIHdpdGggZnVsbCAzMi1iaXRzIHByZWNpc2lvbiwKLSAgICAgICAgLy8gZ2dsUmVjaXBRTm9ybWFsaXplZCByZXR1cm5zIGEgbnVtYmVyIFstMC41LCAwLjVbIGFuZCBhbiBleHBvbmVudC4KLSAgICAgICAgZCA9IGdnbFJlY2lwUU5vcm1hbGl6ZWQoYXJlYSwgJnEpOwotCi0gICAgICAgIC8vIFRoZW4gY29tcHV0ZSB0aGUgbWluaW11bSBsZWZ0LXNoaWZ0IHRvIG5vdCBvdmVyZmxvdyB0aGUgbXVscwotICAgICAgICAvLyBiZWxvdy4gCi0gICAgICAgIHMgPSAzMiAtIGdnbENseihhYnMobV9keTAyKXxhYnMobV9keTEwKXxhYnMobV9keDAxKXxhYnMobV9keDIwKSk7Ci0KLSAgICAgICAgLy8gV2UnbGwga2VlcCAxNi1iaXRzIG9mIHByZWNpc2lvbiBmb3IgZGVsdGFzL2FyZWEuIFNvIHdlIG5lZWQKLSAgICAgICAgLy8gdG8gc2hpZnQgZXZlcnl0aGluZyBsZWZ0IGFuIGV4dHJhIDE1IGJpdHMuCi0gICAgICAgIHMgKz0gMTU7Ci0gICAgICAgIAotICAgICAgICAvLyBtYWtlIHN1cmUgYWxsIGZpbmFsIHNoaWZ0cyBhcmUgbm90ID4gMzIsIGJlY2F1c2UgZ2dsTXVseAotICAgICAgICAvLyBjYW4ndCBoYW5kbGUgaXQuCi0gICAgICAgIGlmIChzIDwgcSkgcyA9IHE7Ci0gICAgICAgIGlmIChzID4gMzIpIHsKLSAgICAgICAgICAgIGQgPj49IDMyLXM7Ci0gICAgICAgICAgICBzID0gMzI7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBtX2R4MDEgPSBnZ2xNdWx4KG1fZHgwMSwgZCwgcyk7Ci0gICAgbV9keTEwID0gZ2dsTXVseChtX2R5MTAsIGQsIHMpOwotICAgIG1fZHgyMCA9IGdnbE11bHgobV9keDIwLCBkLCBzKTsKLSAgICBtX2R5MDIgPSBnZ2xNdWx4KG1fZHkwMiwgZCwgcyk7Ci0gICAgbV9hcmVhX3NjYWxlID0gMzIgKyBxIC0gczsKLSAgICBtX3NjYWxlID0gMDsKLQotICAgIGlmIChlbmFibGVzICYgR0dMX0VOQUJMRV9UTVVTKSB7Ci0gICAgICAgIGNvbnN0IGludCBBID0gZ2dsQ2x6KGFicyhtX2R5MDIpfGFicyhtX2R5MTApfGFicyhtX2R4MDEpfGFicyhtX2R4MjApKTsKLSAgICAgICAgY29uc3QgaW50IEIgPSBnZ2xDbHooYWJzKG1feDApfGFicyhtX3kwKSk7Ci0gICAgICAgIG1fc2NhbGUgPSBtYXgoMCwgMzIgLSAoQSArIDE2KSkgKwotICAgICAgICAgICAgICAgICAgbWF4KDAsIDMyIC0gKEIgKyBUUklfRlJBQ1RJT05fQklUUykpICsgMTsKLSAgICB9Ci19Ci0KLWludCBjb21wdXRlX2l0ZXJhdG9yc190OjppdGVyYXRvcnNTY2FsZShHR0xmaXhlZCogaXQsCi0gICAgICAgIGludDMyX3QgYzAsIGludDMyX3QgYzEsIGludDMyX3QgYzIpIGNvbnN0Ci17Ci0gICAgaW50MzJfdCBkYzAxID0gYzEgLSBjMDsKLSAgICBpbnQzMl90IGRjMDIgPSBjMiAtIGMwOwotICAgIGNvbnN0IGludCBBID0gZ2dsQ2x6KGFicyhjMCkpOwotICAgIGNvbnN0IGludCBCID0gZ2dsQ2x6KGFicyhkYzAxKXxhYnMoZGMwMikpOwotICAgIGNvbnN0IGludCBzY2FsZSA9IG1pbihBLCBCIC0gbV9zY2FsZSkgLSAyOwotICAgIGlmIChzY2FsZSA+PSAwKSB7Ci0gICAgICAgIGMwICAgPDw9IHNjYWxlOwotICAgICAgICBkYzAxIDw8PSBzY2FsZTsKLSAgICAgICAgZGMwMiA8PD0gc2NhbGU7Ci0gICAgfSBlbHNlIHsKLSAgICAgICAgYzAgICA+Pj0gLXNjYWxlOwotICAgICAgICBkYzAxID4+PSAtc2NhbGU7Ci0gICAgICAgIGRjMDIgPj49IC1zY2FsZTsKLSAgICB9Ci0gICAgY29uc3QgaW50IHMgPSBtX2FyZWFfc2NhbGU7Ci0gICAgaW50MzJfdCBkY2R4ID0gZ2dsTXVsQWRkeChkYzAxLCBtX2R5MDIsIGdnbE11bHgoZGMwMiwgbV9keTEwLCBzKSwgcyk7Ci0gICAgaW50MzJfdCBkY2R5ID0gZ2dsTXVsQWRkeChkYzAyLCBtX2R4MDEsIGdnbE11bHgoZGMwMSwgbV9keDIwLCBzKSwgcyk7Ci0gICAgaW50MzJfdCBjID0gYzAgLSAoZ2dsTXVsQWRkeChkY2R4LCBtX3gwLCAKLSAgICAgICAgICAgIGdnbE11bHgoZGNkeSwgbV95MCwgVFJJX0ZSQUNUSU9OX0JJVFMpLCBUUklfRlJBQ1RJT05fQklUUykpOwotICAgIGl0WzBdID0gYzsKLSAgICBpdFsxXSA9IGRjZHg7Ci0gICAgaXRbMl0gPSBkY2R5OwotICAgIHJldHVybiBzY2FsZTsKLX0KLQotdm9pZCBjb21wdXRlX2l0ZXJhdG9yc190OjppdGVyYXRvcnMxNjE2KEdHTGZpeGVkKiBpdCwKLSAgICAgICAgR0dMZml4ZWQgYzAsIEdHTGZpeGVkIGMxLCBHR0xmaXhlZCBjMikgY29uc3QKLXsKLSAgICBjb25zdCBHR0xmaXhlZCBkYzAxID0gYzEgLSBjMDsKLSAgICBjb25zdCBHR0xmaXhlZCBkYzAyID0gYzIgLSBjMDsKLSAgICAvLyAxNi4xNiB4IDE2LjE2ID09IDMyLjMyIC0tPiAxNi4xNgotICAgIGNvbnN0IGludCBzID0gbV9hcmVhX3NjYWxlOwotICAgIGludDMyX3QgZGNkeCA9IGdnbE11bEFkZHgoZGMwMSwgbV9keTAyLCBnZ2xNdWx4KGRjMDIsIG1fZHkxMCwgcyksIHMpOwotICAgIGludDMyX3QgZGNkeSA9IGdnbE11bEFkZHgoZGMwMiwgbV9keDAxLCBnZ2xNdWx4KGRjMDEsIG1fZHgyMCwgcyksIHMpOwotICAgIGludDMyX3QgYyA9IGMwIC0gKGdnbE11bEFkZHgoZGNkeCwgbV94MCwKLSAgICAgICAgICAgIGdnbE11bHgoZGNkeSwgbV95MCwgVFJJX0ZSQUNUSU9OX0JJVFMpLCBUUklfRlJBQ1RJT05fQklUUykpOwotICAgIGl0WzBdID0gYzsKLSAgICBpdFsxXSA9IGRjZHg7Ci0gICAgaXRbMl0gPSBkY2R5OwotfQotCi12b2lkIGNvbXB1dGVfaXRlcmF0b3JzX3Q6Oml0ZXJhdG9yczAwMzIoaW50NjRfdCogaXQsCi0gICAgICAgIGludDMyX3QgYzAsIGludDMyX3QgYzEsIGludDMyX3QgYzIpIGNvbnN0Ci17Ci0gICAgY29uc3QgaW50IHMgPSBtX2FyZWFfc2NhbGUgLSAxNjsKLSAgICBpbnQzMl90IGRjMDEgPSAoYzEgLSBjMCk+PnM7Ci0gICAgaW50MzJfdCBkYzAyID0gKGMyIC0gYzApPj5zOwotICAgIC8vIDE2LjE2IHggMTYuMTYgPT0gMzIuMzIKLSAgICBpbnQ2NF90IGRjZHggPSBnZ2xNdWxpaShkYzAxLCBtX2R5MDIpICsgZ2dsTXVsaWkoZGMwMiwgbV9keTEwKTsKLSAgICBpbnQ2NF90IGRjZHkgPSBnZ2xNdWxpaShkYzAyLCBtX2R4MDEpICsgZ2dsTXVsaWkoZGMwMSwgbV9keDIwKTsKLSAgICBpdFsgMF0gPSAoYzA8PDE2KSAtICgoZGNkeCptX3gwICsgZGNkeSptX3kwKT4+NCk7Ci0gICAgaXRbIDFdID0gZGNkeDsKLSAgICBpdFsgMl0gPSBkY2R5OwotfQotCi0jaWYgZGVmaW5lZChfX2FybV9fKSAmJiAhZGVmaW5lZChfX3RodW1iX18pCi1pbmxpbmUgdm9pZCBjb21wdXRlX2l0ZXJhdG9yc190OjppdGVyYXRvcnMwMDMyKGludDMyX3QqIGl0LAotICAgICAgICBpbnQzMl90IGMwLCBpbnQzMl90IGMxLCBpbnQzMl90IGMyKSBjb25zdAotewotICAgIDo6aXRlcmF0b3JzMDAzMih0aGlzLCBpdCwgYzAsIGMxLCBjMik7Ci19Ci0jZWxzZQotdm9pZCBjb21wdXRlX2l0ZXJhdG9yc190OjppdGVyYXRvcnMwMDMyKGludDMyX3QqIGl0LAotICAgICAgICBpbnQzMl90IGMwLCBpbnQzMl90IGMxLCBpbnQzMl90IGMyKSBjb25zdAotewotICAgIGludDY0X3QgaXQ2NFszXTsKLSAgICBpdGVyYXRvcnMwMDMyKGl0LCBjMCwgYzEsIGMyKTsKLSAgICBpdFswXSA9IGl0NjRbMF07Ci0gICAgaXRbMV0gPSBpdDY0WzFdOwotICAgIGl0WzJdID0gaXQ2NFsyXTsKLX0KLSNlbmRpZgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyBpbmxpbmUgaW50MzJfdCBjbGFtcFooR0xmaXhlZCB6KSBDT05TVDsKLWludDMyX3QgY2xhbXBaKEdMZml4ZWQgeikgewotICAgIHogPSAoeiAmIH4oej4+MzEpKTsKLSAgICBpZiAoeiA+PSAweDEwMDAwKQotICAgICAgICB6ID0gMHhGRkZGOwotICAgIHJldHVybiB6OwotfQotCi1zdGF0aWMgX19hdHRyaWJ1dGVfXygobm9pbmxpbmUpKQotdm9pZCBmZXRjaF90ZXhjb29yZF9pbXBsKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2MikKLXsKLSAgICB2ZXJ0ZXhfdCogY29uc3QgdnR4WzNdID0geyB2MCwgdjEsIHYyIH07Ci0gICAgYXJyYXlfdCBjb25zdCAqIGNvbnN0IHRleGNvb3JkQXJyYXkgPSBjLT5hcnJheXMudGV4dHVyZTsKLSAgICAKLSAgICBmb3IgKGludCBpPTAgOyBpPEdHTF9URVhUVVJFX1VOSVRfQ09VTlQgOyBpKyspIHsKLSAgICAgICAgaWYgKCEoYy0+cmFzdGVyaXplci5zdGF0ZS50ZXh0dXJlW2ldLmVuYWJsZSkpCi0gICAgICAgICAgICBjb250aW51ZTsKLSAgICAgICAgCi0gICAgICAgIGZvciAoaW50IGo9MCA7IGo8MyA7IGorKykgewotICAgICAgICAgICAgdmVydGV4X3QqIGNvbnN0IHYgPSB2dHhbal07Ci0gICAgICAgICAgICBpZiAodi0+ZmxhZ3MgJiB2ZXJ0ZXhfdDo6VFQpCi0gICAgICAgICAgICAgICAgY29udGludWU7Ci0KLSAgICAgICAgICAgIC8vIE5PVEU6IGhlcmUgd2UgY291bGQgY29tcHV0ZSBhdXRvbWF0aWMgdGV4Z2VuCi0gICAgICAgICAgICAvLyBzdWNoIGFzIHNwaGVyZS9jdWJlIG1hcHMsIGluc3RlYWQgb2YgZmV0Y2hpbmcgdGhlbQotICAgICAgICAgICAgLy8gZnJvbSB0aGUgdGV4dGNvb3JkIGFycmF5LgotCi0gICAgICAgICAgICB2ZWM0X3QmIGNvb3JkcyA9IHYtPnRleHR1cmVbaV07Ci0gICAgICAgICAgICBjb25zdCBHTHVieXRlKiB0cCA9IHRleGNvb3JkQXJyYXlbaV0uZWxlbWVudCgKLSAgICAgICAgICAgICAgICAgICAgdi0+aW5kZXggJiB2ZXJ0ZXhfY2FjaGVfdDo6SU5ERVhfTUFTSyk7Ci0gICAgICAgICAgICB0ZXhjb29yZEFycmF5W2ldLmZldGNoKGMsIGNvb3Jkcy52LCB0cCk7Ci0KLSAgICAgICAgICAgIC8vIHRyYW5zZm9ybSB0ZXh0dXJlIGNvb3JkaW5hdGVzLi4uCi0gICAgICAgICAgICBjb29yZHMuUSA9IDB4MTAwMDA7Ci0gICAgICAgICAgICBjb25zdCB0cmFuc2Zvcm1fdCYgdHIgPSBjLT50cmFuc2Zvcm1zLnRleHR1cmVbaV0udHJhbnNmb3JtOyAKLSAgICAgICAgICAgIGlmIChnZ2xfdW5saWtlbHkodHIub3BzKSkgewotICAgICAgICAgICAgICAgIGMtPmFycmF5cy50ZXhfdHJhbnNmb3JtW2ldKCZ0ciwgJmNvb3JkcywgJmNvb3Jkcyk7Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIGRpdmlkZSBieSBRCi0gICAgICAgICAgICBjb25zdCBHR0xmaXhlZCBxID0gY29vcmRzLlE7Ci0gICAgICAgICAgICBpZiAoZ2dsX3VubGlrZWx5KHEgIT0gMHgxMDAwMCkpIHsKLSAgICAgICAgICAgICAgICBjb25zdCBpbnQzMl90IHFpbnYgPSBnZ2xSZWNpcDI4KHEpOwotICAgICAgICAgICAgICAgIGNvb3Jkcy5TID0gZ2dsTXVseChjb29yZHMuUywgcWludiwgMjgpOwotICAgICAgICAgICAgICAgIGNvb3Jkcy5UID0gZ2dsTXVseChjb29yZHMuVCwgcWludiwgMjgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIHYwLT5mbGFncyB8PSB2ZXJ0ZXhfdDo6VFQ7Ci0gICAgdjEtPmZsYWdzIHw9IHZlcnRleF90OjpUVDsKLSAgICB2Mi0+ZmxhZ3MgfD0gdmVydGV4X3Q6OlRUOwotfQotCi1pbmxpbmUgdm9pZCBmZXRjaF90ZXhjb29yZChvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpCi17Ci0gICAgY29uc3QgdWludDMyX3QgZW5hYmxlcyA9IGMtPnJhc3Rlcml6ZXIuc3RhdGUuZW5hYmxlczsKLSAgICBpZiAoIShlbmFibGVzICYgR0dMX0VOQUJMRV9UTVVTKSkKLSAgICAgICAgcmV0dXJuOwotCi0gICAgLy8gRmV0Y2ggJiB0cmFuc2Zvcm0gdGV4dHVyZSBjb29yZGluYXRlcy4uLgotICAgIGlmIChnZ2xfbGlrZWx5KHYwLT5mbGFncyAmIHYxLT5mbGFncyAmIHYyLT5mbGFncyAmIHZlcnRleF90OjpUVCkpIHsKLSAgICAgICAgLy8gYWxyZWFkeSBkb25lIGZvciBhbGwgdGhyZWUgdmVydGljZXMsIGJhaWwuLi4KLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBmZXRjaF90ZXhjb29yZF9pbXBsKGMsIHYwLCB2MSwgdjIpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBQb2ludAotI2VuZGlmCi0KLXZvaWQgcHJpbWl0aXZlX25vcF9wb2ludChvZ2xlc19jb250ZXh0X3QqLCB2ZXJ0ZXhfdCopIHsKLX0KLQotdm9pZCBwcmltaXRpdmVfcG9pbnQob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikKLXsKLSAgICAvLyBsaWdodGluZyAmIGNsYW1waW5nLi4uCi0gICAgY29uc3QgdWludDMyX3QgZW5hYmxlcyA9IGMtPnJhc3Rlcml6ZXIuc3RhdGUuZW5hYmxlczsKLQotICAgIGlmIChnZ2xfdW5saWtlbHkoISh2LT5mbGFncyAmIHZlcnRleF90OjpMSVQpKSkgewotICAgICAgICBpZiAoYy0+bGlnaHRpbmcuZW5hYmxlKSB7Ci0gICAgICAgICAgICBjLT5saWdodGluZy5saWdodFZlcnRleChjLCB2KTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHYtPmZsYWdzIHw9IHZlcnRleF90OjpMSVQ7Ci0gICAgICAgICAgICBjb25zdCBHTHZvaWQqIGNwID0gYy0+YXJyYXlzLmNvbG9yLmVsZW1lbnQoCi0gICAgICAgICAgICAgICAgICAgIHYtPmluZGV4ICYgdmVydGV4X2NhY2hlX3Q6OklOREVYX01BU0spOwotICAgICAgICAgICAgYy0+YXJyYXlzLmNvbG9yLmZldGNoKGMsIHYtPmNvbG9yLnYsIGNwKTsKLSAgICAgICAgfQotICAgICAgICBpZiAoZW5hYmxlcyAmIEdHTF9FTkFCTEVfRk9HKSB7Ci0gICAgICAgICAgICB2LT5mb2cgPSBjLT5mb2cuZm9nKGMsIHYtPmV5ZS56KTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vIFhYWDogd2UgZG9uJ3QgbmVlZCB0byBkbyB0aGF0IGVhY2gtdGltZQotICAgIC8vIGlmIGNvbG9yIGFycmF5IGFuZCBsaWdodGluZyBub3QgZW5hYmxlZCAKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmNvbG9yNHh2KGMsIHYtPmNvbG9yLnYpOwotCi0gICAgLy8gWFhYOiBsb29rIGludG8gRVMgcG9pbnQtc3ByaXRlIGV4dGVuc2lvbgotICAgIGlmIChlbmFibGVzICYgR0dMX0VOQUJMRV9UTVVTKSB7Ci0gICAgICAgIGZldGNoX3RleGNvb3JkKGMsIHYsdix2KTsKLSAgICAgICAgZm9yIChpbnQgaT0wIDsgaTxHR0xfVEVYVFVSRV9VTklUX0NPVU5UIDsgaSsrKSB7Ci0gICAgICAgICAgICBpZiAoIWMtPnJhc3Rlcml6ZXIuc3RhdGUudGV4dHVyZVtpXS5lbmFibGUpIAotICAgICAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICAgICAgaW50MzJfdCBpdHRbOF07Ci0gICAgICAgICAgICBpdHRbMV0gPSBpdHRbMl0gPSBpdHRbNF0gPSBpdHRbNV0gPSAwOwotICAgICAgICAgICAgaXR0WzZdID0gaXR0WzddID0gMTY7IC8vIFhYWDogY2hlY2sgdGhhdAotICAgICAgICAgICAgaWYgKGMtPnJhc3Rlcml6ZXIuc3RhdGUudGV4dHVyZVtpXS5zX3dyYXAgPT0gR0dMX0NMQU1QKSB7Ci0gICAgICAgICAgICAgICAgaW50IHdpZHRoID0gYy0+dGV4dHVyZXMudG11W2ldLnRleHR1cmUtPnN1cmZhY2Uud2lkdGg7Ci0gICAgICAgICAgICAgICAgaXR0WzBdID0gdi0+dGV4dHVyZVtpXS5TICogd2lkdGg7Ci0gICAgICAgICAgICAgICAgaXR0WzZdID0gMDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChjLT5yYXN0ZXJpemVyLnN0YXRlLnRleHR1cmVbaV0udF93cmFwID09IEdHTF9DTEFNUCkgewotICAgICAgICAgICAgICAgIGludCBoZWlnaHQgPSBjLT50ZXh0dXJlcy50bXVbaV0udGV4dHVyZS0+c3VyZmFjZS5oZWlnaHQ7Ci0gICAgICAgICAgICAgICAgaXR0WzNdID0gdi0+dGV4dHVyZVtpXS5UICogaGVpZ2h0OwotICAgICAgICAgICAgICAgIGl0dFs3XSA9IDA7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleENvb3JkR3JhZFNjYWxlOHh2KGMsIGksIGl0dCk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX0RFUFRIX1RFU1QpIHsKLSAgICAgICAgaW50MzJfdCBpdHpbM107Ci0gICAgICAgIGl0elswXSA9IGNsYW1wWih2LT53aW5kb3cueikgKiAweDAwMDEwMDAxOwotICAgICAgICBpdHpbMV0gPSBpdHpbMl0gPSAwOwotICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnpHcmFkM3h2KGMsIGl0eik7Ci0gICAgfQotCi0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX0ZPRykgewotICAgICAgICBHTGZpeGVkIGl0ZlszXTsKLSAgICAgICAgaXRmWzBdID0gdi0+Zm9nOwotICAgICAgICBpdGZbMV0gPSBpdGZbMl0gPSAwOwotICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmZvZ0dyYWQzeHYoYywgaXRmKTsKLSAgICB9Ci0KLSAgICAvLyBSZW5kZXIgb3VyIHBvaW50Li4uCi0gICAgYy0+cmFzdGVyaXplci5wcm9jcy5wb2ludHgoYywgdi0+d2luZG93LnYsIGMtPnBvaW50LnNpemUpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBMaW5lCi0jZW5kaWYKLQotdm9pZCBwcmltaXRpdmVfbm9wX2xpbmUob2dsZXNfY29udGV4dF90KiwgdmVydGV4X3QqLCB2ZXJ0ZXhfdCopIHsKLX0KLQotdm9pZCBwcmltaXRpdmVfbGluZShvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxKQotewotICAgIC8vIGdldCB0ZXh0dXJlIGNvb3JkaW5hdGVzCi0gICAgZmV0Y2hfdGV4Y29vcmQoYywgdjAsIHYxLCB2MSk7Ci0KLSAgICAvLyBsaWdodC9zaGFkZSB0aGUgdmVydGljZXMgZmlyc3QgKHRoZXkncmUgY29waWVkIGJlbG93KQotICAgIGMtPmxpZ2h0aW5nLmxpZ2h0VHJpYW5nbGUoYywgdjAsIHYxLCB2MSk7Ci0KLSAgICAvLyBjbGlwIHRoZSBsaW5lIGlmIG5lZWRlZAotICAgIGlmIChnZ2xfdW5saWtlbHkoKHYwLT5mbGFncyB8IHYxLT5mbGFncykgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEwpKSB7Ci0gICAgICAgIHVuc2lnbmVkIGludCBjb3VudCA9IGNsaXBfbGluZShjLCB2MCwgdjEpOwotICAgICAgICBpZiAoZ2dsX3VubGlrZWx5KGNvdW50ID09IDApKQotICAgICAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIC8vIGNvbXB1dGUgaXRlcmF0b3JzLi4uCi0gICAgY29uc3QgdWludDMyX3QgZW5hYmxlcyA9IGMtPnJhc3Rlcml6ZXIuc3RhdGUuZW5hYmxlczsKLSAgICBjb25zdCB1aW50MzJfdCBtYXNrID0gICBHR0xfRU5BQkxFX1RNVVMgfAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdHTF9FTkFCTEVfU01PT1RIIHwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBHR0xfRU5BQkxFX1cgfCAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBHR0xfRU5BQkxFX0ZPRyB8Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgR0dMX0VOQUJMRV9ERVBUSF9URVNUOwotCi0gICAgaWYgKGdnbF91bmxpa2VseShlbmFibGVzICYgbWFzaykpIHsKLSAgICAgICAgYy0+bGVycC5pbml0TGluZSh2MCwgdjEpOwotICAgICAgICBsZXJwX3RyaWFuZ2xlKGMsIHYwLCB2MSwgdjApOwotICAgIH0KLQotICAgIC8vIHJlbmRlciBvdXIgbGluZQotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MubGluZXgoYywgdjAtPndpbmRvdy52LCB2MS0+d2luZG93LnYsIGMtPmxpbmUud2lkdGgpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNwcmFnbWEgbWFyayBUcmlhbmdsZQotI2VuZGlmCi0KLXZvaWQgcHJpbWl0aXZlX25vcF90cmlhbmdsZShvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpIHsKLX0KLQotdm9pZCBwcmltaXRpdmVfY2xpcF90cmlhbmdsZShvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpCi17Ci0gICAgdWludDMyX3QgY2MgPSAodjAtPmZsYWdzIHwgdjEtPmZsYWdzIHwgdjItPmZsYWdzKSAmIHZlcnRleF90OjpDTElQX0FMTDsKLSAgICBpZiAoZ2dsX2xpa2VseSghY2MpKSB7Ci0gICAgICAgIC8vIGNvZGUgYmVsb3cgbXVzdCBiZSBhcyBvcHRpbWl6ZWQgYXMgcG9zc2libGUsIHRoaXMgaXMgdGhlCi0gICAgICAgIC8vIGNvbW1vbiBjb2RlIHBhdGguCi0KLSAgICAgICAgLy8gVGhpcyB0cmlhbmdsZSBpcyBub3QgY2xpcHBlZCwgdGVzdCBpZiBpdCdzIGN1bGxlZAotICAgICAgICAvLyB1bmNsaXBwZWQgdHJpYW5nbGUuLi4KLSAgICAgICAgYy0+bGVycC5pbml0VHJpYW5nbGUodjAsIHYxLCB2Mik7Ci0gICAgICAgIGlmIChjdWxsX3RyaWFuZ2xlKGMsIHYwLCB2MSwgdjIpKQotICAgICAgICAgICAgcmV0dXJuOyAvLyBjdWxsZWQhCi0KLSAgICAgICAgLy8gRmV0Y2ggYWxsIHRleHR1cmUgY29vcmRpbmF0ZXMgaWYgbmVlZGVkCi0gICAgICAgIGZldGNoX3RleGNvb3JkKGMsIHYwLCB2MSwgdjIpOwotCi0gICAgICAgIC8vIGxpZ2h0IChvciBzaGFkZSkgb3VyIHRyaWFuZ2xlIQotICAgICAgICBjLT5saWdodGluZy5saWdodFRyaWFuZ2xlKGMsIHYwLCB2MSwgdjIpOwotCi0gICAgICAgIHRyaWFuZ2xlKGMsIHYwLCB2MSwgdjIpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgLy8gVGhlIGFzc3VtcHRpb24gaGVyZSBpcyB0aGF0IHdlJ3JlIG5vdCBnb2luZyB0byBjbGlwIHZlcnkgb2Z0ZW4sCi0gICAgLy8gYW5kIGV2ZW4gbW9yZSByYXJlbHkgd2lsbCB3ZSBjbGlwIGEgdHJpYW5nbGUgdGhhdCBlbmRzIHVwCi0gICAgLy8gYmVpbmcgY3VsbGVkIG91dC4gU28gaXQncyBva2F5IHRvIGxpZ2h0IHRoZSB2ZXJ0aWNlcyBoZXJlLCBldmVuIHRob3VnaAotICAgIC8vIGluIGEgZmV3IGNhc2VzIHdlIHdvbid0IHJlbmRlciB0aGUgdHJpYW5nbGUgKGlmIGN1bGxlZCkuCi0KLSAgICAvLyBGZXRjaCB0ZXh0dXJlIGNvb3JkaW5hdGVzLi4uCi0gICAgZmV0Y2hfdGV4Y29vcmQoYywgdjAsIHYxLCB2Mik7Ci0KLSAgICAvLyBsaWdodCAob3Igc2hhZGUpIG91ciB0cmlhbmdsZSEKLSAgICBjLT5saWdodGluZy5saWdodFRyaWFuZ2xlKGMsIHYwLCB2MSwgdjIpOwotCi0gICAgY2xpcF90cmlhbmdsZShjLCB2MCwgdjEsIHYyKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdm9pZCB0cmlhbmdsZShvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpCi17Ci0gICAgLy8gY29tcHV0ZSBpdGVyYXRvcnMuLi4KLSAgICBjb25zdCB1aW50MzJfdCBlbmFibGVzID0gYy0+cmFzdGVyaXplci5zdGF0ZS5lbmFibGVzOwotICAgIGNvbnN0IHVpbnQzMl90IG1hc2sgPSAgIEdHTF9FTkFCTEVfVE1VUyB8Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgR0dMX0VOQUJMRV9TTU9PVEggfAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdHTF9FTkFCTEVfVyB8IAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdHTF9FTkFCTEVfRk9HIHwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBHR0xfRU5BQkxFX0RFUFRIX1RFU1Q7Ci0KLSAgICBpZiAoZ2dsX2xpa2VseShlbmFibGVzICYgbWFzaykpCi0gICAgICAgIGxlcnBfdHJpYW5nbGUoYywgdjAsIHYxLCB2Mik7Ci0KLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRyaWFuZ2xleChjLCB2MC0+d2luZG93LnYsIHYxLT53aW5kb3cudiwgdjItPndpbmRvdy52KTsKLX0KLQotdm9pZCBsZXJwX3RyaWFuZ2xlKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2MikKLXsKLSAgICBjb25zdCB1aW50MzJfdCBlbmFibGVzID0gYy0+cmFzdGVyaXplci5zdGF0ZS5lbmFibGVzOwotICAgIGMtPmxlcnAuaW5pdExlcnAodjAsIGVuYWJsZXMpOwotCi0gICAgLy8gc2V0IHVwIHRleHR1cmUgaXRlcmF0b3JzCi0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX1RNVVMpIHsKLSAgICAgICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX1cpIHsKLSAgICAgICAgICAgIGxlcnBfdGV4Y29vcmRzX3coYywgdjAsIHYxLCB2Mik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBsZXJwX3RleGNvb3JkcyhjLCB2MCwgdjEsIHYyKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIC8vIHNldCB1cCB0aGUgY29sb3IgaXRlcmF0b3JzCi0gICAgY29uc3QgY29tcHV0ZV9pdGVyYXRvcnNfdCYgbGVycCA9IGMtPmxlcnA7Ci0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX1NNT09USCkgewotICAgICAgICBHTGZpeGVkIGl0Y1sxMl07Ci0gICAgICAgIGZvciAoaW50IGk9MCA7IGk8NCA7IGkrKykgewotICAgICAgICAgICAgY29uc3QgR0dMY29sb3IgYzAgPSB2MC0+Y29sb3IudltpXSAqIDI1NTsKLSAgICAgICAgICAgIGNvbnN0IEdHTGNvbG9yIGMxID0gdjEtPmNvbG9yLnZbaV0gKiAyNTU7Ci0gICAgICAgICAgICBjb25zdCBHR0xjb2xvciBjMiA9IHYyLT5jb2xvci52W2ldICogMjU1OwotICAgICAgICAgICAgbGVycC5pdGVyYXRvcnMxNjE2KCZpdGNbaSozXSwgYzAsIGMxLCBjMik7Ci0gICAgICAgIH0KLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy5jb2xvckdyYWQxMnh2KGMsIGl0Yyk7Ci0gICAgfQotCi0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX0RFUFRIX1RFU1QpIHsKLSAgICAgICAgaW50MzJfdCBpdHpbM107Ci0gICAgICAgIGNvbnN0IGludDMyX3QgdjB6ID0gY2xhbXBaKHYwLT53aW5kb3cueik7Ci0gICAgICAgIGNvbnN0IGludDMyX3QgdjF6ID0gY2xhbXBaKHYxLT53aW5kb3cueik7Ci0gICAgICAgIGNvbnN0IGludDMyX3QgdjJ6ID0gY2xhbXBaKHYyLT53aW5kb3cueik7Ci0gICAgICAgIGlmIChnZ2xfdW5saWtlbHkoYy0+cG9seWdvbk9mZnNldC5lbmFibGUpKSB7Ci0gICAgICAgICAgICBjb25zdCBpbnQzMl90IHVuaXRzID0gKGMtPnBvbHlnb25PZmZzZXQudW5pdHMgPDwgMTYpOwotICAgICAgICAgICAgY29uc3QgR0xmaXhlZCBmYWN0b3IgPSBjLT5wb2x5Z29uT2Zmc2V0LmZhY3RvcjsKLSAgICAgICAgICAgIGlmIChmYWN0b3IpIHsKLSAgICAgICAgICAgICAgICBpbnQ2NF90IGl0ejY0WzNdOwotICAgICAgICAgICAgICAgIGxlcnAuaXRlcmF0b3JzMDAzMihpdHo2NCwgdjB6LCB2MXosIHYyeik7Ci0gICAgICAgICAgICAgICAgaW50NjRfdCBtYXhEZXB0aFNsb3BlID0gbWF4KGl0ejY0WzFdLCBpdHo2NFsyXSk7Ci0gICAgICAgICAgICAgICAgaXR6WzBdID0gdWludDMyX3QoaXR6NjRbMF0pIAotICAgICAgICAgICAgICAgICAgICAgICAgKyB1aW50MzJfdCgobWF4RGVwdGhTbG9wZSpmYWN0b3IpPj4xNikgKyB1bml0czsKLSAgICAgICAgICAgICAgICBpdHpbMV0gPSB1aW50MzJfdChpdHo2NFsxXSk7Ci0gICAgICAgICAgICAgICAgaXR6WzJdID0gdWludDMyX3QoaXR6NjRbMl0pOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBsZXJwLml0ZXJhdG9yczAwMzIoaXR6LCB2MHosIHYxeiwgdjJ6KTsKLSAgICAgICAgICAgICAgICBpdHpbMF0gKz0gdW5pdHM7IAotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgbGVycC5pdGVyYXRvcnMwMDMyKGl0eiwgdjB6LCB2MXosIHYyeik7Ci0gICAgICAgIH0KLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy56R3JhZDN4dihjLCBpdHopOwotICAgIH0gICAgCi0KLSAgICBpZiAoZ2dsX3VubGlrZWx5KGVuYWJsZXMgJiBHR0xfRU5BQkxFX0ZPRykpIHsKLSAgICAgICAgR0xmaXhlZCBpdGZbM107Ci0gICAgICAgIGxlcnAuaXRlcmF0b3JzMTYxNihpdGYsIHYwLT5mb2csIHYxLT5mb2csIHYyLT5mb2cpOwotICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmZvZ0dyYWQzeHYoYywgaXRmKTsKLSAgICB9Ci19Ci0KLQotc3RhdGljIGlubGluZQotaW50IGNvbXB1dGVfbG9kKG9nbGVzX2NvbnRleHRfdCogYywgaW50IGksCi0gICAgICAgIGludDMyX3QgczAsIGludDMyX3QgdDAsIGludDMyX3QgczEsIGludDMyX3QgdDEsIGludDMyX3QgczIsIGludDMyX3QgdDIpCi17Ci0gICAgLy8gQ29tcHV0ZSBtaXBtYXAgbGV2ZWwgLyBwcmltaXRpdmUKLSAgICAvLyByaG8gPSBzcXJ0KCB0ZXhlbEFyZWEgLyBhcmVhICkKLSAgICAvLyBsb2QgPSBsb2cyKCByaG8gKQotICAgIC8vIGxvZCA9IGxvZzIoIHRleGVsQXJlYSAvIGFyZWEgKSAvIDIKLSAgICAvLyBsb2QgPSAobG9nMiggdGV4ZWxBcmVhICkgLSBsb2cyKCBhcmVhICkpIC8gMgotICAgIGNvbnN0IGNvbXB1dGVfaXRlcmF0b3JzX3QmIGxlcnAgPSBjLT5sZXJwOwotICAgIGNvbnN0IEdHTGNvb3JkIGFyZWEgPSBhYnMobGVycC5hcmVhKCkpOwotICAgIGNvbnN0IGludCB3ID0gYy0+dGV4dHVyZXMudG11W2ldLnRleHR1cmUtPnN1cmZhY2Uud2lkdGg7Ci0gICAgY29uc3QgaW50IGggPSBjLT50ZXh0dXJlcy50bXVbaV0udGV4dHVyZS0+c3VyZmFjZS5oZWlnaHQ7Ci0gICAgY29uc3QgaW50IHNoaWZ0ID0gMTYgKyAoMTYgLSBUUklfRlJBQ1RJT05fQklUUyk7Ci0gICAgaW50MzJfdCB0ZXhlbEFyZWEgPSBhYnMoIGdnbE11bHgoczEtczAsIHQyLXQwLCBzaGlmdCkgLQotICAgICAgICAgICAgZ2dsTXVseChzMi1zMCwgdDEtdDAsIHNoaWZ0KSApKncqaDsKLSAgICBpbnQgbG9nMlRBcmVhID0gKDMyLVRSSV9GUkFDVElPTl9CSVRTICAtMSkgLSBnZ2xDbHoodGV4ZWxBcmVhKTsKLSAgICBpbnQgbG9nMkFyZWEgID0gKDMyLVRSSV9GUkFDVElPTl9CSVRTKjItMSkgLSBnZ2xDbHooYXJlYSk7Ci0gICAgaW50IGxvZCA9IChsb2cyVEFyZWEgLSBsb2cyQXJlYSArIDEpID4+IDE7Ci0gICAgcmV0dXJuIGxvZDsKLX0KLQotdm9pZCBsZXJwX3RleGNvb3JkcyhvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpCi17Ci0gICAgY29uc3QgY29tcHV0ZV9pdGVyYXRvcnNfdCYgbGVycCA9IGMtPmxlcnA7Ci0gICAgaW50MzJfdCBpdHRbOF0gX19hdHRyaWJ1dGVfXygoYWxpZ25lZCgxNikpKTsKLSAgICBmb3IgKGludCBpPTAgOyBpPEdHTF9URVhUVVJFX1VOSVRfQ09VTlQgOyBpKyspIHsKLSAgICAgICAgY29uc3QgdGV4dHVyZV90JiB0bXUgPSBjLT5yYXN0ZXJpemVyLnN0YXRlLnRleHR1cmVbaV07Ci0gICAgICAgIGlmICghdG11LmVuYWJsZSkgCi0gICAgICAgICAgICBjb250aW51ZTsKLQotICAgICAgICAvLyBjb21wdXRlIHRoZSBqYWNvYmlhbnMgdXNpbmcgYmxvY2sgZmxvYXRpbmctcG9pbnQKLSAgICAgICAgaW50MzJfdCBzMCA9IHYwLT50ZXh0dXJlW2ldLlM7Ci0gICAgICAgIGludDMyX3QgdDAgPSB2MC0+dGV4dHVyZVtpXS5UOwotICAgICAgICBpbnQzMl90IHMxID0gdjEtPnRleHR1cmVbaV0uUzsKLSAgICAgICAgaW50MzJfdCB0MSA9IHYxLT50ZXh0dXJlW2ldLlQ7Ci0gICAgICAgIGludDMyX3QgczIgPSB2Mi0+dGV4dHVyZVtpXS5TOwotICAgICAgICBpbnQzMl90IHQyID0gdjItPnRleHR1cmVbaV0uVDsKLQotICAgICAgICBjb25zdCBHTGVudW0gbWluX2ZpbHRlciA9IGMtPnRleHR1cmVzLnRtdVtpXS50ZXh0dXJlLT5taW5fZmlsdGVyOwotICAgICAgICBpZiAoZ2dsX3VubGlrZWx5KG1pbl9maWx0ZXIgPj0gR0xfTkVBUkVTVF9NSVBNQVBfTkVBUkVTVCkpIHsKLSAgICAgICAgICAgIGludCBsb2QgPSBjb21wdXRlX2xvZChjLCBpLCBzMCwgdDAsIHMxLCB0MSwgczIsIHQyKTsKLSAgICAgICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuYmluZFRleHR1cmVMb2QoYywgaSwKLSAgICAgICAgICAgICAgICAgICAgJmMtPnRleHR1cmVzLnRtdVtpXS50ZXh0dXJlLT5taXAobG9kKSk7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBwcmVtdWx0aXBseSAocyx0KSB3aGVuIGNsYW1wbGluZwotICAgICAgICBpZiAodG11LnNfd3JhcCA9PSBHR0xfQ0xBTVApIHsKLSAgICAgICAgICAgIGNvbnN0IGludCB3aWR0aCA9IHRtdS5zdXJmYWNlLndpZHRoOwotICAgICAgICAgICAgczAgKj0gd2lkdGg7Ci0gICAgICAgICAgICBzMSAqPSB3aWR0aDsKLSAgICAgICAgICAgIHMyICo9IHdpZHRoOwotICAgICAgICB9Ci0gICAgICAgIGlmICh0bXUudF93cmFwID09IEdHTF9DTEFNUCkgewotICAgICAgICAgICAgY29uc3QgaW50IGhlaWdodCA9IHRtdS5zdXJmYWNlLmhlaWdodDsKLSAgICAgICAgICAgIHQwICo9IGhlaWdodDsKLSAgICAgICAgICAgIHQxICo9IGhlaWdodDsKLSAgICAgICAgICAgIHQyICo9IGhlaWdodDsKLSAgICAgICAgfQotICAgICAgICBpdHRbNl0gPSAtbGVycC5pdGVyYXRvcnNTY2FsZShpdHQrMCwgczAsIHMxLCBzMik7Ci0gICAgICAgIGl0dFs3XSA9IC1sZXJwLml0ZXJhdG9yc1NjYWxlKGl0dCszLCB0MCwgdDEsIHQyKTsKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy50ZXhDb29yZEdyYWRTY2FsZTh4dihjLCBpLCBpdHQpOwotICAgIH0KLX0KLQotdm9pZCBsZXJwX3RleGNvb3Jkc193KG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2MikKLXsKLSAgICBjb25zdCBjb21wdXRlX2l0ZXJhdG9yc190JiBsZXJwID0gYy0+bGVycDsKLSAgICBpbnQzMl90IGl0dFs4XSBfX2F0dHJpYnV0ZV9fKChhbGlnbmVkKDE2KSkpOwotICAgIGludDMyX3QgaXR3WzNdOwotCi0gICAgLy8gY29tcHV0ZSBXJ3Mgc2NhbGUgdG8gMi4zMAotICAgIGludDMyX3QgdzAgPSB2MC0+d2luZG93Lnc7Ci0gICAgaW50MzJfdCB3MSA9IHYxLT53aW5kb3cudzsKLSAgICBpbnQzMl90IHcyID0gdjItPndpbmRvdy53OwotICAgIGludCB3c2NhbGUgPSAzMiAtIGdnbENseih3MHx3MXx3Mik7Ci0KLSAgICAvLyBjb21wdXRlIHRoZSBqYWNvYmlhbiB1c2luZyBibG9jayBmbG9hdGluZy1wb2ludCAgICAKLSAgICBpbnQgc2MgPSBsZXJwLml0ZXJhdG9yc1NjYWxlKGl0dywgdzAsIHcxLCB3Mik7Ci0gICAgc2MgKz0gIHdzY2FsZSAtIDE2OwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3Mud0dyYWQzeHYoYywgaXR3KTsKLQotICAgIGZvciAoaW50IGk9MCA7IGk8R0dMX1RFWFRVUkVfVU5JVF9DT1VOVCA7IGkrKykgewotICAgICAgICBjb25zdCB0ZXh0dXJlX3QmIHRtdSA9IGMtPnJhc3Rlcml6ZXIuc3RhdGUudGV4dHVyZVtpXTsKLSAgICAgICAgaWYgKCF0bXUuZW5hYmxlKSAKLSAgICAgICAgICAgIGNvbnRpbnVlOwotCi0gICAgICAgIC8vIGNvbXB1dGUgdGhlIGphY29iaWFucyB1c2luZyBibG9jayBmbG9hdGluZy1wb2ludAotICAgICAgICBpbnQzMl90IHMwID0gdjAtPnRleHR1cmVbaV0uUzsKLSAgICAgICAgaW50MzJfdCB0MCA9IHYwLT50ZXh0dXJlW2ldLlQ7Ci0gICAgICAgIGludDMyX3QgczEgPSB2MS0+dGV4dHVyZVtpXS5TOwotICAgICAgICBpbnQzMl90IHQxID0gdjEtPnRleHR1cmVbaV0uVDsKLSAgICAgICAgaW50MzJfdCBzMiA9IHYyLT50ZXh0dXJlW2ldLlM7Ci0gICAgICAgIGludDMyX3QgdDIgPSB2Mi0+dGV4dHVyZVtpXS5UOwotCi0gICAgICAgIGNvbnN0IEdMZW51bSBtaW5fZmlsdGVyID0gYy0+dGV4dHVyZXMudG11W2ldLnRleHR1cmUtPm1pbl9maWx0ZXI7Ci0gICAgICAgIGlmIChnZ2xfdW5saWtlbHkobWluX2ZpbHRlciA+PSBHTF9ORUFSRVNUX01JUE1BUF9ORUFSRVNUKSkgewotICAgICAgICAgICAgaW50IGxvZCA9IGNvbXB1dGVfbG9kKGMsIGksIHMwLCB0MCwgczEsIHQxLCBzMiwgdDIpOwotICAgICAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy5iaW5kVGV4dHVyZUxvZChjLCBpLAotICAgICAgICAgICAgICAgICAgICAmYy0+dGV4dHVyZXMudG11W2ldLnRleHR1cmUtPm1pcChsb2QpKTsKLSAgICAgICAgfQotCi0gICAgICAgIC8vIHByZW11bHRpcGx5IChzLHQpIHdoZW4gY2xhbXBsaW5nCi0gICAgICAgIGlmICh0bXUuc193cmFwID09IEdHTF9DTEFNUCkgewotICAgICAgICAgICAgY29uc3QgaW50IHdpZHRoID0gdG11LnN1cmZhY2Uud2lkdGg7Ci0gICAgICAgICAgICBzMCAqPSB3aWR0aDsKLSAgICAgICAgICAgIHMxICo9IHdpZHRoOwotICAgICAgICAgICAgczIgKj0gd2lkdGg7Ci0gICAgICAgIH0KLSAgICAgICAgaWYgKHRtdS50X3dyYXAgPT0gR0dMX0NMQU1QKSB7Ci0gICAgICAgICAgICBjb25zdCBpbnQgaGVpZ2h0ID0gdG11LnN1cmZhY2UuaGVpZ2h0OwotICAgICAgICAgICAgdDAgKj0gaGVpZ2h0OwotICAgICAgICAgICAgdDEgKj0gaGVpZ2h0OwotICAgICAgICAgICAgdDIgKj0gaGVpZ2h0OwotICAgICAgICB9Ci0KLSAgICAgICAgczAgPSBnZ2xNdWx4KHMwLCB3MCwgd3NjYWxlKTsKLSAgICAgICAgdDAgPSBnZ2xNdWx4KHQwLCB3MCwgd3NjYWxlKTsKLSAgICAgICAgczEgPSBnZ2xNdWx4KHMxLCB3MSwgd3NjYWxlKTsKLSAgICAgICAgdDEgPSBnZ2xNdWx4KHQxLCB3MSwgd3NjYWxlKTsKLSAgICAgICAgczIgPSBnZ2xNdWx4KHMyLCB3Miwgd3NjYWxlKTsKLSAgICAgICAgdDIgPSBnZ2xNdWx4KHQyLCB3Miwgd3NjYWxlKTsKLQotICAgICAgICBpdHRbNl0gPSBzYyAtIGxlcnAuaXRlcmF0b3JzU2NhbGUoaXR0KzAsIHMwLCBzMSwgczIpOwotICAgICAgICBpdHRbN10gPSBzYyAtIGxlcnAuaXRlcmF0b3JzU2NhbGUoaXR0KzMsIHQwLCB0MSwgdDIpOwotICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleENvb3JkR3JhZFNjYWxlOHh2KGMsIGksIGl0dCk7Ci0gICAgfQotfQotCi0KLXN0YXRpYyBpbmxpbmUKLWJvb2wgY3VsbF90cmlhbmdsZShvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiB2MCwgdmVydGV4X3QqIHYxLCB2ZXJ0ZXhfdCogdjIpCi17Ci0gICAgaWYgKGdnbF9saWtlbHkoYy0+Y3VsbC5lbmFibGUpKSB7Ci0gICAgICAgIGNvbnN0IEdMZW51bSB3aW5kaW5nID0gKGMtPmxlcnAuYXJlYSgpID4gMCkgPyBHTF9DVyA6IEdMX0NDVzsKLSAgICAgICAgY29uc3QgR0xlbnVtIGZhY2UgPSAod2luZGluZyA9PSBjLT5jdWxsLmZyb250RmFjZSkgPyBHTF9GUk9OVCA6IEdMX0JBQ0s7Ci0gICAgICAgIGlmIChmYWNlID09IGMtPmN1bGwuY3VsbEZhY2UpCi0gICAgICAgICAgICByZXR1cm4gdHJ1ZTsgLy8gY3VsbGVkIQotICAgIH0KLSAgICByZXR1cm4gZmFsc2U7Ci19Ci0KLXN0YXRpYyBpbmxpbmUKLUdMZml4ZWQgZnJ1c3R1bVBsYW5lRGlzdChpbnQgcGxhbmUsIGNvbnN0IHZlYzRfdCYgcykKLXsKLSAgICBjb25zdCBHTGZpeGVkIGQgPSBzLnZbIHBsYW5lID4+IDEgXTsKLSAgICByZXR1cm4gICgocGxhbmUgJiAxKSA/IChzLncgLSBkKSA6IChzLncgKyBkKSk7IAotfQotCi1zdGF0aWMgaW5saW5lCi1pbnQzMl90IGNsaXBEaXZpZGUoR0xmaXhlZCBhLCBHTGZpeGVkIGIpIHsKLSAgICAvLyByZXR1cm5zIGEgNC4yOCBmaXhlZC1wb2ludAotICAgIHJldHVybiBnZ2xNdWxEaXZpKDFMVTw8MjgsIGEsIGIpOwotfSAKLQotdm9pZCBjbGlwX3RyaWFuZ2xlKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgdmVydGV4X3QqIHYwLCB2ZXJ0ZXhfdCogdjEsIHZlcnRleF90KiB2MikKLXsKLSAgICB1aW50MzJfdCBhbGxfY2MgPSAodjAtPmZsYWdzIHwgdjEtPmZsYWdzIHwgdjItPmZsYWdzKSAmIHZlcnRleF90OjpDTElQX0FMTDsKLQotICAgIHZlcnRleF90ICpwMCwgKnAxLCAqcDI7Ci0gICAgY29uc3QgaW50IE1BWF9DTElQUElOR19QTEFORVMgPSA2ICsgT0dMRVNfTUFYX0NMSVBfUExBTkVTOwotICAgIGNvbnN0IGludCBNQVhfVkVSVElDRVMgPSAzOwotCi0gICAgLy8gVGVtcG9yYXJ5IGJ1ZmZlciB0byBob2xkIHRoZSBuZXcgdmVydGljZXMuIEVhY2ggcGxhbmUgY2FuIGFkZCB1cCB0byAKLSAgICAvLyB0d28gbmV3IHZlcnRpY2VzIChiZWNhdXNlIHRoZSBwb2x5Z29uIGlzIGNvbnZleCkuCi0gICAgLy8gV2UgbmVlZCBvbmUgZXh0cmEgZWxlbWVudCwgdG8gaGFuZGxlIGFuIG92ZXJmbG93IGNhc2Ugd2hlbgotICAgIC8vIHRoZSBwb2x5Z29uIGRlZ2VuZXJhdGVzIGludG8gc29tZXRoaW5nIG5vbiBjb252ZXguCi0gICAgdmVydGV4X3QgYnVmZmVyW01BWF9DTElQUElOR19QTEFORVMgKiAyICsgMV07ICAgLy8gfjNLQgotICAgIHZlcnRleF90KiBidWYgPSBidWZmZXI7Ci0KLSAgICAvLyBvcmlnaW5hbCBsaXN0IG9mIHZlcnRpY2VzIChwb2x5Z29uIHRvIGNsaXAsIGluIGZhY3QgdGhpcwotICAgIC8vIGZ1bmN0aW9uIHdvcmtzIHdpdGggYW4gYXJiaXRyYXJ5IHBvbHlnb24pLgotICAgIHZlcnRleF90KiBpblszXSA9IHsgdjAsIHYxLCB2MiB9OwotICAgIAotICAgIC8vIG91dHB1dCBsaXN0cyAod2UgbmVlZCAyLCB3aGljaCB3ZSB1c2UgYmFjayBhbmQgZm9ydGgpCi0gICAgLy8gKG1heGltdW0gb3V0cG91dCBsaXN0J3Mgc2l6ZSBpcyBNQVhfQ0xJUFBJTkdfUExBTkVTICsgTUFYX1ZFUlRJQ0VTKQotICAgIC8vIDIgbW9yZSBlbGVtZW50cyBmb3Igb3ZlcmZsb3cgd2hlbiBub24gY29udmV4IHBvbHlnb25zLgotICAgIHZlcnRleF90KiBvdXRbMl1bTUFYX0NMSVBQSU5HX1BMQU5FUyArIE1BWF9WRVJUSUNFUyArIDJdOwotICAgIHVuc2lnbmVkIGludCBvdXRpID0gMDsKLSAgICAKLSAgICAvLyBjdXJyZW50IGlucHV0IGxpc3QKLSAgICB2ZXJ0ZXhfdCoqIGl2bCA9IGluOwotCi0gICAgLy8gMyBpbnB1dCB2ZXJ0aWNlcywgMCBpbiB0aGUgb3V0cHV0IGxpc3QsIGZpcnN0IHBsYW5lCi0gICAgdW5zaWduZWQgaW50IGljID0gMzsKLQotICAgIC8vIFVzZXIgY2xpcC1wbGFuZXMgZmlyc3QsIHRoZSBjbGlwcGluZyBpcyBhbHdheXMgZG9uZSBpbiBleWUtY29vcmRpbmF0ZQotICAgIC8vIHRoaXMgaXMgYmFzaWNhbGx5IHRoZSBzYW1lIGFsZ29yaXRobSB0aGFuIGZvciB0aGUgdmlldy12b2x1bWUKLSAgICAvLyBjbGlwcGluZywgZXhjZXB0IGZvciB0aGUgY29tcHV0YXRpb24gb2YgdGhlIGRpc3RhbmNlICh2ZXJ0ZXgsIHBsYW5lKQotICAgIC8vIGFuZCB0aGUgZmFjdCB0aGF0IHdlIG5lZWQgdG8gY29tcHV0ZSB0aGUgZXllLWNvb3JkaW5hdGVzIG9mIGVhY2gKLSAgICAvLyBuZXcgdmVydGV4IHdlIGNyZWF0ZS4KLSAgICAKLSAgICBpZiAoZ2dsX3VubGlrZWx5KGFsbF9jYyAmIHZlcnRleF90OjpVU0VSX0NMSVBfQUxMKSkKLSAgICB7Ci0gICAgICAgIHVuc2lnbmVkIGludCBwbGFuZSA9IDA7Ci0gICAgICAgIHVpbnQzMl90IGNjID0gKGFsbF9jYyAmIHZlcnRleF90OjpVU0VSX0NMSVBfQUxMKSA+PiA4OwotICAgICAgICBkbyB7Ci0gICAgICAgICAgICBpZiAoY2MgJiAxKSB7ICAgICAgICAKLSAgICAgICAgICAgICAgICAvLyBwb2ludGVycyB0byBvdXIgb3V0cHV0IGxpc3QgKGhlYWQgYW5kIGN1cnJlbnQpCi0gICAgICAgICAgICAgICAgdmVydGV4X3QqKiBjb25zdCBvdmwgPSAmb3V0W291dGldWzBdOwotICAgICAgICAgICAgICAgIHZlcnRleF90Kiogb3V0cHV0ID0gb3ZsOwotICAgICAgICAgICAgICAgIHVuc2lnbmVkIGludCBvYyA9IDA7Ci0gICAgICAgICAgICAgICAgdW5zaWduZWQgaW50IHNlbnRpbmVsID0gMDsKLSAgICAgICAgICAgICAgICAvLyBwcmV2aW91cyB2ZXJ0ZXgsIGNvbXB1dGUgZGlzdGFuY2UgdG8gdGhlIHBsYW5lCi0gICAgICAgICAgICAgICAgdmVydGV4X3QqIHMgPSBpdmxbaWMtMV07Ci0gICAgICAgICAgICAgICAgY29uc3QgdmVjNF90JiBlcXVhdGlvbiA9IGMtPmNsaXBQbGFuZXMucGxhbmVbcGxhbmVdLmVxdWF0aW9uOwotICAgICAgICAgICAgICAgIEdMZml4ZWQgc2QgPSBkb3Q0KGVxdWF0aW9uLnYsIHMtPmV5ZS52KTsKLSAgICAgICAgICAgICAgICAvLyBjbGlwIGVhY2ggdmVydGV4IGFnYWluc3QgdGhpcyBwbGFuZS4uLgotICAgICAgICAgICAgICAgIGZvciAodW5zaWduZWQgaW50IGk9MCA7IGk8aWMgOyBpKyspIHsgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgdmVydGV4X3QqIHAgPSBpdmxbaV07Ci0gICAgICAgICAgICAgICAgICAgIGNvbnN0IEdMZml4ZWQgcGQgPSBkb3Q0KGVxdWF0aW9uLnYsIHAtPmV5ZS52KTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHNkID49IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChwZCA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gYm90aCBpbnNpZGUKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAqb3V0cHV0KysgPSBwOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9jKys7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIHMgaW5zaWRlLCBwIG91dHNpZGUgKGV4aXRpbmcpCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgR0xmaXhlZCB0ID0gY2xpcERpdmlkZShzZCwgc2QtcGQpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGMtPmFycmF5cy5jbGlwRXllKGMsIGJ1ZiwgdCwgcCwgcyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKm91dHB1dCsrID0gYnVmKys7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgb2MrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoKytzZW50aW5lbCA+PSAzKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm47IC8vIG5vbi1jb252ZXggcG9seWdvbiEKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChwZCA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gcyBvdXRzaWRlIChlbnRlcmluZykKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAocGQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgR0xmaXhlZCB0ID0gY2xpcERpdmlkZShwZCwgcGQtc2QpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjLT5hcnJheXMuY2xpcEV5ZShjLCBidWYsIHQsIHMsIHApOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqb3V0cHV0KysgPSBidWYrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgb2MrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCsrc2VudGluZWwgPj0gMykKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybjsgLy8gbm9uLWNvbnZleCBwb2x5Z29uIQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAqb3V0cHV0KysgPSBwOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9jKys7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gYm90aCBvdXRzaWRlCi0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcyA9IHA7Ci0gICAgICAgICAgICAgICAgICAgIHNkID0gcGQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIC8vIG91dHB1dCBsaXN0IGJlY29tZSB0aGUgbmV3IGlucHV0IGxpc3QKLSAgICAgICAgICAgICAgICBpZiAob2M8MykKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuOyAvLyBsZXNzIHRoYW4gMyB2ZXJ0aWNlcyBsZWZ0PyB3ZSdyZSBkb25lIQotICAgICAgICAgICAgICAgIGl2bCA9IG92bDsKLSAgICAgICAgICAgICAgICBpYyA9IG9jOwotICAgICAgICAgICAgICAgIG91dGkgPSAxLW91dGk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjYyA+Pj0gMTsKLSAgICAgICAgICAgIHBsYW5lKys7Ci0gICAgICAgIH0gd2hpbGUgKGNjKTsKLSAgICB9Ci0KLSAgICAvLyBmcnVzdHVtIGNsaXAtcGxhbmVzCi0gICAgaWYgKGFsbF9jYyAmIHZlcnRleF90OjpGUlVTVFVNX0NMSVBfQUxMKQotICAgIHsKLSAgICAgICAgdW5zaWduZWQgaW50IHBsYW5lID0gMDsKLSAgICAgICAgdWludDMyX3QgY2MgPSBhbGxfY2MgJiB2ZXJ0ZXhfdDo6RlJVU1RVTV9DTElQX0FMTDsKLSAgICAgICAgZG8gewotICAgICAgICAgICAgaWYgKGNjICYgMSkgeyAgICAgICAgCi0gICAgICAgICAgICAgICAgLy8gcG9pbnRlcnMgdG8gb3VyIG91dHB1dCBsaXN0IChoZWFkIGFuZCBjdXJyZW50KQotICAgICAgICAgICAgICAgIHZlcnRleF90KiogY29uc3Qgb3ZsID0gJm91dFtvdXRpXVswXTsKLSAgICAgICAgICAgICAgICB2ZXJ0ZXhfdCoqIG91dHB1dCA9IG92bDsKLSAgICAgICAgICAgICAgICB1bnNpZ25lZCBpbnQgb2MgPSAwOwotICAgICAgICAgICAgICAgIHVuc2lnbmVkIGludCBzZW50aW5lbCA9IDA7Ci0gICAgICAgICAgICAgICAgLy8gcHJldmlvdXMgdmVydGV4LCBjb21wdXRlIGRpc3RhbmNlIHRvIHRoZSBwbGFuZQotICAgICAgICAgICAgICAgIHZlcnRleF90KiBzID0gaXZsW2ljLTFdOwotICAgICAgICAgICAgICAgIEdMZml4ZWQgc2QgPSBmcnVzdHVtUGxhbmVEaXN0KHBsYW5lLCBzLT5jbGlwKTsKLSAgICAgICAgICAgICAgICAvLyBjbGlwIGVhY2ggdmVydGV4IGFnYWluc3QgdGhpcyBwbGFuZS4uLgotICAgICAgICAgICAgICAgIGZvciAodW5zaWduZWQgaW50IGk9MCA7IGk8aWMgOyBpKyspIHsgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgdmVydGV4X3QqIHAgPSBpdmxbaV07Ci0gICAgICAgICAgICAgICAgICAgIGNvbnN0IEdMZml4ZWQgcGQgPSBmcnVzdHVtUGxhbmVEaXN0KHBsYW5lLCBwLT5jbGlwKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHNkID49IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChwZCA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gYm90aCBpbnNpZGUKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAqb3V0cHV0KysgPSBwOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9jKys7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIHMgaW5zaWRlLCBwIG91dHNpZGUgKGV4aXRpbmcpCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgR0xmaXhlZCB0ID0gY2xpcERpdmlkZShzZCwgc2QtcGQpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGMtPmFycmF5cy5jbGlwVmVydGV4KGMsIGJ1ZiwgdCwgcCwgcyk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgKm91dHB1dCsrID0gYnVmKys7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgb2MrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoKytzZW50aW5lbCA+PSAzKQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm47IC8vIG5vbi1jb252ZXggcG9seWdvbiEKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChwZCA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gcyBvdXRzaWRlIChlbnRlcmluZykKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAocGQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgR0xmaXhlZCB0ID0gY2xpcERpdmlkZShwZCwgcGQtc2QpOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjLT5hcnJheXMuY2xpcFZlcnRleChjLCBidWYsIHQsIHMsIHApOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqb3V0cHV0KysgPSBidWYrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgb2MrKzsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCsrc2VudGluZWwgPj0gMykKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybjsgLy8gbm9uLWNvbnZleCBwb2x5Z29uIQotICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAqb3V0cHV0KysgPSBwOwotICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9jKys7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gYm90aCBvdXRzaWRlCi0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcyA9IHA7Ci0gICAgICAgICAgICAgICAgICAgIHNkID0gcGQ7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIC8vIG91dHB1dCBsaXN0IGJlY29tZSB0aGUgbmV3IGlucHV0IGxpc3QKLSAgICAgICAgICAgICAgICBpZiAob2M8MykKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuOyAvLyBsZXNzIHRoYW4gMyB2ZXJ0aWNlcyBsZWZ0PyB3ZSdyZSBkb25lIQotICAgICAgICAgICAgICAgIGl2bCA9IG92bDsKLSAgICAgICAgICAgICAgICBpYyA9IG9jOwotICAgICAgICAgICAgICAgIG91dGkgPSAxLW91dGk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjYyA+Pj0gMTsKLSAgICAgICAgICAgIHBsYW5lKys7Ci0gICAgICAgIH0gd2hpbGUgKGNjKTsKLSAgICB9Ci0gICAgCi0gICAgLy8gZmluYWxseSB3ZSBjYW4gcmVuZGVyIG91ciB0cmlhbmdsZXMuLi4KLSAgICBwMCA9IGl2bFswXTsKLSAgICBwMSA9IGl2bFsxXTsKLSAgICBmb3IgKHVuc2lnbmVkIGludCBpPTIgOyBpPGljIDsgaSsrKSB7Ci0gICAgICAgIHAyID0gaXZsW2ldOwotICAgICAgICBjLT5sZXJwLmluaXRUcmlhbmdsZShwMCwgcDEsIHAyKTsKLSAgICAgICAgaWYgKGN1bGxfdHJpYW5nbGUoYywgcDAsIHAxLCBwMikpIHsKLSAgICAgICAgICAgIHAxID0gcDI7Ci0gICAgICAgICAgICBjb250aW51ZTsgLy8gY3VsbGVkIQotICAgICAgICB9Ci0gICAgICAgIHRyaWFuZ2xlKGMsIHAwLCBwMSwgcDIpOwotICAgICAgICBwMSA9IHAyOwotICAgIH0KLX0KLQotdW5zaWduZWQgaW50IGNsaXBfbGluZShvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiBzLCB2ZXJ0ZXhfdCogcCkKLXsKLSAgICBjb25zdCB1aW50MzJfdCBhbGxfY2MgPSAocy0+ZmxhZ3MgfCBwLT5mbGFncykgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEw7Ci0KLSAgICBpZiAoZ2dsX3VubGlrZWx5KGFsbF9jYyAmIHZlcnRleF90OjpVU0VSX0NMSVBfQUxMKSkKLSAgICB7Ci0gICAgICAgIHVuc2lnbmVkIGludCBwbGFuZSA9IDA7Ci0gICAgICAgIHVpbnQzMl90IGNjID0gKGFsbF9jYyAmIHZlcnRleF90OjpVU0VSX0NMSVBfQUxMKSA+PiA4OwotICAgICAgICBkbyB7Ci0gICAgICAgICAgICBpZiAoY2MgJiAxKSB7Ci0gICAgICAgICAgICAgICAgY29uc3QgdmVjNF90JiBlcXVhdGlvbiA9IGMtPmNsaXBQbGFuZXMucGxhbmVbcGxhbmVdLmVxdWF0aW9uOwotICAgICAgICAgICAgICAgIGNvbnN0IEdMZml4ZWQgc2QgPSBkb3Q0KGVxdWF0aW9uLnYsIHMtPmV5ZS52KTsKLSAgICAgICAgICAgICAgICBjb25zdCBHTGZpeGVkIHBkID0gZG90NChlcXVhdGlvbi52LCBwLT5leWUudik7Ci0gICAgICAgICAgICAgICAgaWYgKHNkID49IDApIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHBkID49IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIGJvdGggaW5zaWRlCi0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyBzIGluc2lkZSwgcCBvdXRzaWRlIChleGl0aW5nKQotICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgR0xmaXhlZCB0ID0gY2xpcERpdmlkZShzZCwgc2QtcGQpOwotICAgICAgICAgICAgICAgICAgICAgICAgYy0+YXJyYXlzLmNsaXBFeWUoYywgcCwgdCwgcCwgcyk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBpZiAocGQgPj0gMCkgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gcyBvdXRzaWRlIChlbnRlcmluZykKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChwZCkgewotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IEdMZml4ZWQgdCA9IGNsaXBEaXZpZGUocGQsIHBkLXNkKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjLT5hcnJheXMuY2xpcEV5ZShjLCBzLCB0LCBzLCBwKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgLy8gYm90aCBvdXRzaWRlCi0gICAgICAgICAgICAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICAgICAgY2MgPj49IDE7Ci0gICAgICAgICAgICBwbGFuZSsrOwotICAgICAgICB9IHdoaWxlIChjYyk7Ci0gICAgfQotCi0gICAgLy8gZnJ1c3R1bSBjbGlwLXBsYW5lcwotICAgIGlmIChhbGxfY2MgJiB2ZXJ0ZXhfdDo6RlJVU1RVTV9DTElQX0FMTCkKLSAgICB7Ci0gICAgICAgIHVuc2lnbmVkIGludCBwbGFuZSA9IDA7Ci0gICAgICAgIHVpbnQzMl90IGNjID0gYWxsX2NjICYgdmVydGV4X3Q6OkZSVVNUVU1fQ0xJUF9BTEw7Ci0gICAgICAgIGRvIHsKLSAgICAgICAgICAgIGlmIChjYyAmIDEpIHsKLSAgICAgICAgICAgICAgICBjb25zdCBHTGZpeGVkIHNkID0gZnJ1c3R1bVBsYW5lRGlzdChwbGFuZSwgcy0+Y2xpcCk7Ci0gICAgICAgICAgICAgICAgY29uc3QgR0xmaXhlZCBwZCA9IGZydXN0dW1QbGFuZURpc3QocGxhbmUsIHAtPmNsaXApOwotICAgICAgICAgICAgICAgIGlmIChzZCA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChwZCA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAvLyBib3RoIGluc2lkZQotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gcyBpbnNpZGUsIHAgb3V0c2lkZSAoZXhpdGluZykKLSAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IEdMZml4ZWQgdCA9IGNsaXBEaXZpZGUoc2QsIHNkLXBkKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGMtPmFycmF5cy5jbGlwVmVydGV4KGMsIHAsIHQsIHAsIHMpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKHBkID49IDApIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIC8vIHMgb3V0c2lkZSAoZW50ZXJpbmcpCi0gICAgICAgICAgICAgICAgICAgICAgICBpZiAocGQpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBHTGZpeGVkIHQgPSBjbGlwRGl2aWRlKHBkLCBwZC1zZCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYy0+YXJyYXlzLmNsaXBWZXJ0ZXgoYywgcywgdCwgcywgcCk7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgICAgIC8vIGJvdGggb3V0c2lkZQotICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gMDsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGNjID4+PSAxOwotICAgICAgICAgICAgcGxhbmUrKzsKLSAgICAgICAgfSB3aGlsZSAoY2MpOwotICAgIH0KLQotICAgIHJldHVybiAyOwotfQotCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL3ByaW1pdGl2ZXMuaCBiL29wZW5nbC9saWJhZ2wvcHJpbWl0aXZlcy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAxYmVmNjA0Li4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJhZ2wvcHJpbWl0aXZlcy5oCisrKyAvZGV2L251bGwKQEAgLTEsMzcgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy9wcmltaXRpdmVzLmgKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jaWZuZGVmIEFORFJPSURfT1BFTkdMRVNfUFJJTUlUSVZFU19ICi0jZGVmaW5lIEFORFJPSURfT1BFTkdMRVNfUFJJTUlUSVZFU19ICi0KLSNpbmNsdWRlIDxzdGRpbnQuaD4KLSNpbmNsdWRlIDxzdGRkZWYuaD4KLSNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KLQotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLW5hbWVzcGFjZSBnbCB7Ci1zdHJ1Y3Qgb2dsZXNfY29udGV4dF90OwotfTsKLQotdm9pZCBvZ2xlc192YWxpZGF0ZV9wcmltaXRpdmVzKG9nbGVzX2NvbnRleHRfdCogYyk7Ci0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX09QRU5HTEVTX1BSSU1JVElWRVNfSAotCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGliYWdsL3N0YXRlLmNwcCBiL29wZW5nbC9saWJhZ2wvc3RhdGUuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1Y2JhYmVhLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJhZ2wvc3RhdGUuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsNTg2ICswLDAgQEAKLS8qIGxpYnMvb3BlbmdsZXMvc3RhdGUuY3BwCi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotCi0jaW5jbHVkZSAiY29udGV4dC5oIgotI2luY2x1ZGUgImZwLmgiCi0jaW5jbHVkZSAic3RhdGUuaCIKLSNpbmNsdWRlICJhcnJheS5oIgotI2luY2x1ZGUgIm1hdHJpeC5oIgotI2luY2x1ZGUgInZlcnRleC5oIgotI2luY2x1ZGUgImxpZ2h0LmgiCi0jaW5jbHVkZSAidGV4dHVyZS5oIgotI2luY2x1ZGUgIkJ1ZmZlck9iamVjdE1hbmFnZXIuaCIKLSNpbmNsdWRlICJUZXh0dXJlT2JqZWN0TWFuYWdlci5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RhdGljIGNoYXIgY29uc3QgKiBjb25zdCBnVmVuZG9yU3RyaW5nICAgICA9ICJBbmRyb2lkIjsKLXN0YXRpYyBjaGFyIGNvbnN0ICogY29uc3QgZ1JlbmRlcmVyU3RyaW5nICAgPSAiQW5kcm9pZCBQaXhlbEZsaW5nZXIgMS4wIjsKLXN0YXRpYyBjaGFyIGNvbnN0ICogY29uc3QgZ1ZlcnNpb25TdHJpbmcgICAgPSAiT3BlbkdMIEVTLUNNIDEuMCI7Ci1zdGF0aWMgY2hhciBjb25zdCAqIGNvbnN0IGdFeHRlbnNpb25zU3RyaW5nID0KLSAgICAiR0xfT0VTX2J5dGVfY29vcmRpbmF0ZXMgIiAgICAgICAgICAgICAgLy8gT0sKLSAgICAiR0xfT0VTX2ZpeGVkX3BvaW50ICIgICAgICAgICAgICAgICAgICAgLy8gT0sKLSAgICAiR0xfT0VTX3NpbmdsZV9wcmVjaXNpb24gIiAgICAgICAgICAgICAgLy8gT0sKLSAgICAiR0xfT0VTX3JlYWRfZm9ybWF0ICIgICAgICAgICAgICAgICAgICAgLy8gT0sKLSAgICAiR0xfT0VTX2NvbXByZXNzZWRfcGFsZXR0ZWRfdGV4dHVyZSAiICAgLy8gT0sKLSAgICAiR0xfT0VTX2RyYXdfdGV4dHVyZSAiICAgICAgICAgICAgICAgICAgLy8gT0sKLSAgICAiR0xfT0VTX21hdHJpeF9nZXQgIiAgICAgICAgICAgICAgICAgICAgLy8gT0sKLSAgICAiR0xfT0VTX3F1ZXJ5X21hdHJpeCAiICAgICAgICAgICAgICAgICAgLy8gT0sKLSAgICAvLyAgICAgICAgIkdMX09FU19wb2ludF9zaXplX2FycmF5ICIgICAgICAgICAgICAgIC8vIFRPRE8KLSAgICAvLyAgICAgICAgIkdMX09FU19wb2ludF9zcHJpdGUgIiAgICAgICAgICAgICAgICAgIC8vIFRPRE8KLSAgICAiR0xfQVJCX3RleHR1cmVfY29tcHJlc3Npb24gIiAgICAgICAgICAgLy8gT0sKLSAgICAiR0xfQVJCX3RleHR1cmVfbm9uX3Bvd2VyX29mX3R3byAiICAgICAgLy8gT0sKLSAgICAiR0xfQU5EUk9JRF9kaXJlY3RfdGV4dHVyZSAiICAgICAgICAgICAgLy8gT0sKLSAgICAiR0xfQU5EUk9JRF91c2VyX2NsaXBfcGxhbmUgIiAgICAgICAgICAgLy8gT0sKLSAgICAiR0xfQU5EUk9JRF92ZXJ0ZXhfYnVmZmVyX29iamVjdCAiICAgICAgLy8gT0sKLSAgICAiR0xfQU5EUk9JRF9nZW5lcmF0ZV9taXBtYXAgIiAgICAgICAgICAgLy8gT0sKLSAgICA7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI2VuZGlmCi0KLW9nbGVzX2NvbnRleHRfdCAqb2dsZXNfaW5pdChzaXplX3QgZXh0cmEpCi17Ci0gICAgdm9pZCogY29uc3QgYmFzZSA9IG1hbGxvYyhleHRyYSArIHNpemVvZihvZ2xlc19jb250ZXh0X3QpICsgMzIpOwotCWlmICghYmFzZSkgcmV0dXJuIDA7Ci0KLSAgICBvZ2xlc19jb250ZXh0X3QgKmMgPQotICAgICAgICAgICAgKG9nbGVzX2NvbnRleHRfdCAqKSgocHRyZGlmZl90KGJhc2UpICsgZXh0cmEgKyAzMSkgJiB+MHgxRkwpOwotICAgIG1lbXNldChjLCAwLCBzaXplb2Yob2dsZXNfY29udGV4dF90KSk7Ci0gICAgZ2dsX2luaXRfY29udGV4dCgmKGMtPnJhc3Rlcml6ZXIpKTsKLSAgICAKLSAgICAvLyBYWFg6IHRoaXMgc2hvdWxkIGJlIHBhc3NlZCBhcyBhbiBhcmd1bWVudAotICAgIHNwPEVHTFN1cmZhY2VNYW5hZ2VyPiBzbWdyKG5ldyBFR0xTdXJmYWNlTWFuYWdlcigpKTsKLSAgICBjLT5zdXJmYWNlTWFuYWdlciA9IHNtZ3IuZ2V0KCk7Ci0gICAgYy0+c3VyZmFjZU1hbmFnZXItPmluY1N0cm9uZyhjKTsKLQotICAgIHNwPEVHTEJ1ZmZlck9iamVjdE1hbmFnZXI+IGJvbWdyKG5ldyBFR0xCdWZmZXJPYmplY3RNYW5hZ2VyKCkpOwotICAgIGMtPmJ1ZmZlck9iamVjdE1hbmFnZXIgPSBib21nci5nZXQoKTsKLSAgICBjLT5idWZmZXJPYmplY3RNYW5hZ2VyLT5pbmNTdHJvbmcoYyk7Ci0KLSAgICBvZ2xlc19pbml0X2FycmF5KGMpOwotICAgIG9nbGVzX2luaXRfbWF0cml4KGMpOwotICAgIG9nbGVzX2luaXRfdmVydGV4KGMpOwotICAgIG9nbGVzX2luaXRfbGlnaHQoYyk7Ci0gICAgb2dsZXNfaW5pdF90ZXh0dXJlKGMpOwotCi0gICAgYy0+cmFzdGVyaXplci5iYXNlID0gYmFzZTsKLSAgICBjLT5wb2ludC5zaXplID0gVFJJX09ORTsKLSAgICBjLT5saW5lLndpZHRoID0gVFJJX09ORTsKLSAgICAgICAgICAgIAotICAgIC8vIGluIE9wZW5HTCwgd3JpdGluZyB0byB0aGUgZGVwdGggYnVmZmVyIGlzIGVuYWJsZWQgYnkgZGVmYXVsdC4KLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmRlcHRoTWFzayhjLCAxKTsKLSAgICAKLSAgICAvLyBPcGVuR0wgZW5hYmxlcyBkaXRoZXJpbmcgYnkgZGVmYXVsdAotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuZW5hYmxlKGMsIEdMX0RJVEhFUik7Ci0KLSAgICByZXR1cm4gYzsKLX0KLQotdm9pZCBvZ2xlc191bmluaXQob2dsZXNfY29udGV4dF90KiBjKQotewotICAgIG9nbGVzX3VuaW5pdF9hcnJheShjKTsKLSAgICBvZ2xlc191bmluaXRfbWF0cml4KGMpOwotICAgIG9nbGVzX3VuaW5pdF92ZXJ0ZXgoYyk7Ci0gICAgb2dsZXNfdW5pbml0X2xpZ2h0KGMpOwotICAgIG9nbGVzX3VuaW5pdF90ZXh0dXJlKGMpOwotICAgIGMtPnN1cmZhY2VNYW5hZ2VyLT5kZWNTdHJvbmcoYyk7Ci0gICAgYy0+YnVmZmVyT2JqZWN0TWFuYWdlci0+ZGVjU3Ryb25nKGMpOwotICAgIGdnbF91bmluaXRfY29udGV4dCgmKGMtPnJhc3Rlcml6ZXIpKTsKLQlmcmVlKGMtPnJhc3Rlcml6ZXIuYmFzZSk7Ci19Ci0KLXZvaWQgX29nbGVzX2Vycm9yKG9nbGVzX2NvbnRleHRfdCogYywgR0xlbnVtIGVycm9yKQotewotICAgIGlmIChjLT5lcnJvciA9PSBHTF9OT19FUlJPUikKLSAgICAgICAgYy0+ZXJyb3IgPSBlcnJvcjsKLX0KLQotc3RhdGljIGJvb2wgc3RlbmNpbG9wX3ZhbGlkKEdMZW51bSBvcCkgewotICAgIHN3aXRjaCAob3ApIHsKLSAgICBjYXNlIEdMX0tFRVA6Ci0gICAgY2FzZSBHTF9aRVJPOgotICAgIGNhc2UgR0xfUkVQTEFDRToKLSAgICBjYXNlIEdMX0lOQ1I6Ci0gICAgY2FzZSBHTF9ERUNSOgotICAgIGNhc2UgR0xfSU5WRVJUOgotICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICB9Ci0gICAgcmV0dXJuIGZhbHNlOwotfQotCi1zdGF0aWMgdm9pZCBlbmFibGVfZGlzYWJsZShvZ2xlc19jb250ZXh0X3QqIGMsIEdMZW51bSBjYXAsIGludCBlbmFibGVkKQotewotICAgIGlmICgoY2FwID49IEdMX0xJR0hUMCkgJiYgKGNhcDxHTF9MSUdIVDArT0dMRVNfTUFYX0xJR0hUUykpIHsKLSAgICAgICAgYy0+bGlnaHRpbmcubGlnaHRzW2NhcC1HTF9MSUdIVDBdLmVuYWJsZSA9IGVuYWJsZWQ7Ci0gICAgICAgIGMtPmxpZ2h0aW5nLmVuYWJsZWRMaWdodHMgJj0gfigxPDwoY2FwLUdMX0xJR0hUMCkpOwotICAgICAgICBjLT5saWdodGluZy5lbmFibGVkTGlnaHRzIHw9IChlbmFibGVkPDwoY2FwLUdMX0xJR0hUMCkpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgc3dpdGNoIChjYXApIHsKLSAgICBjYXNlIEdMX1BPSU5UX1NNT09USDoKLSAgICAgICAgYy0+cG9pbnQuc21vb3RoID0gZW5hYmxlZDsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9MSU5FX1NNT09USDoKLSAgICAgICAgYy0+bGluZS5zbW9vdGggPSBlbmFibGVkOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1BPTFlHT05fT0ZGU0VUX0ZJTEw6Ci0gICAgICAgIGMtPnBvbHlnb25PZmZzZXQuZW5hYmxlID0gZW5hYmxlZDsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9DVUxMX0ZBQ0U6Ci0gICAgICAgIGMtPmN1bGwuZW5hYmxlID0gZW5hYmxlZDsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9MSUdIVElORzoKLSAgICAgICAgYy0+bGlnaHRpbmcuZW5hYmxlID0gZW5hYmxlZDsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9DT0xPUl9NQVRFUklBTDoKLSAgICAgICAgYy0+bGlnaHRpbmcuY29sb3JNYXRlcmlhbC5lbmFibGUgPSBlbmFibGVkOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX05PUk1BTElaRToKLSAgICBjYXNlIEdMX1JFU0NBTEVfTk9STUFMOgotICAgICAgICBjLT50cmFuc2Zvcm1zLnJlc2NhbGVOb3JtYWxzID0gZW5hYmxlZCA/IGNhcCA6IDA7Ci0gICAgICAgIC8vIFhYWDogaW52YWxpZGF0ZSBtdml0Ci0gICAgICAgIGJyZWFrOwotCi0gICAgY2FzZSBHTF9DTElQX1BMQU5FMDoKLSAgICBjYXNlIEdMX0NMSVBfUExBTkUxOgotICAgIGNhc2UgR0xfQ0xJUF9QTEFORTI6Ci0gICAgY2FzZSBHTF9DTElQX1BMQU5FMzoKLSAgICBjYXNlIEdMX0NMSVBfUExBTkU0OgotICAgIGNhc2UgR0xfQ0xJUF9QTEFORTU6Ci0gICAgICAgIGMtPmNsaXBQbGFuZXMuZW5hYmxlICY9IH4oMTw8KGNhcC1HTF9DTElQX1BMQU5FMCkpOwotICAgICAgICBjLT5jbGlwUGxhbmVzLmVuYWJsZSB8PSAoZW5hYmxlZDw8KGNhcC1HTF9DTElQX1BMQU5FMCkpOwotICAgICAgICBvZ2xlc19pbnZhbGlkYXRlX3BlcnNwZWN0aXZlKGMpOwotICAgICAgICBicmVhazsKLQotICAgIGNhc2UgR0xfRk9HOgotICAgIGNhc2UgR0xfREVQVEhfVEVTVDoKLSAgICAgICAgb2dsZXNfaW52YWxpZGF0ZV9wZXJzcGVjdGl2ZShjKTsKLSAgICAgICAgLy8gZmFsbC10aHJvdWdoLi4uCi0gICAgY2FzZSBHTF9CTEVORDoKLSAgICBjYXNlIEdMX1NDSVNTT1JfVEVTVDoKLSAgICBjYXNlIEdMX0FMUEhBX1RFU1Q6Ci0gICAgY2FzZSBHTF9DT0xPUl9MT0dJQ19PUDoKLSAgICBjYXNlIEdMX0RJVEhFUjoKLSAgICBjYXNlIEdMX1NURU5DSUxfVEVTVDoKLSAgICBjYXNlIEdMX1RFWFRVUkVfMkQ6Ci0gICAgICAgIC8vIHRoZXNlIG5lZWQgdG8gZmFsbCB0aHJvdWdoIGludG8gdGhlIHJhc3Rlcml6ZXIKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy5lbmFibGVEaXNhYmxlKGMsIGNhcCwgZW5hYmxlZCk7Ci0gICAgICAgIGJyZWFrOwotICAgICAgICAKLSAgICBjYXNlIEdMX01VTFRJU0FNUExFOgotICAgIGNhc2UgR0xfU0FNUExFX0FMUEhBX1RPX0NPVkVSQUdFOgotICAgIGNhc2UgR0xfU0FNUExFX0FMUEhBX1RPX09ORToKLSAgICBjYXNlIEdMX1NBTVBMRV9DT1ZFUkFHRToKLSAgICAgICAgLy8gbm90IHN1cHBvcnRlZCBpbiB0aGlzIGltcGxlbWVudGF0aW9uCi0gICAgICAgIGJyZWFrOwotCi0gICAgZGVmYXVsdDoKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLXVzaW5nIG5hbWVzcGFjZSBhbmRyb2lkOwotCi0jaWYgMAotI3ByYWdtYSBtYXJrIC0KLSNlbmRpZgotCi0vLyBUaGVzZSBvbmVzIGFyZSBzdXBlci1lYXN5LCB3ZSdyZSBub3Qgc3VwcG9ydGluZyB0aG9zZSBmZWF0dXJlcyEKLXZvaWQgZ2xTYW1wbGVDb3ZlcmFnZShHTGNsYW1wZiB2YWx1ZSwgR0xib29sZWFuIGludmVydCkgewotfQotdm9pZCBnbFNhbXBsZUNvdmVyYWdleChHTGNsYW1weCB2YWx1ZSwgR0xib29sZWFuIGludmVydCkgewotfQotdm9pZCBnbFN0ZW5jaWxGdW5jKEdMZW51bSBmdW5jLCBHTGludCByZWYsIEdMdWludCBtYXNrKSB7Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBpZiAoZnVuYyA8IEdMX05FVkVSIHx8IGZ1bmMgPiBHTF9BTFdBWVMpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICAvLyBmcm9tIE9wZW5HTHxFUyAxLjAgc2VwY2lmaWNhdGlvbjoKLSAgICAvLyBJZiB0aGVyZSBpcyBubyBzdGVuY2lsIGJ1ZmZlciwgbm8gc3RlbmNpbCBtb2RpZmljYXRpb24gY2FuIG9jY3VyCi0gICAgLy8gYW5kIGl0IGlzIGFzIGlmIHRoZSBzdGVuY2lsIHRlc3QgYWx3YXlzIHBhc3Nlcy4KLX0KLQotdm9pZCBnbFN0ZW5jaWxPcChHTGVudW0gZmFpbCwgR0xlbnVtIHpmYWlsLCBHTGVudW0genBhc3MpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICgoc3RlbmNpbG9wX3ZhbGlkKGZhaWwpICYKLSAgICAgICAgIHN0ZW5jaWxvcF92YWxpZCh6ZmFpbCkgJgotICAgICAgICAgc3RlbmNpbG9wX3ZhbGlkKHpwYXNzKSkgPT0gMCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXZvaWQgZ2xBbHBoYUZ1bmMoR0xlbnVtIGZ1bmMsIEdMY2xhbXBmIHJlZikKLXsKLSAgICBnbEFscGhhRnVuY3goZnVuYywgZ2dsRmxvYXRUb0ZpeGVkKHJlZikpOwotfQotCi12b2lkIGdsQ3VsbEZhY2UoR0xlbnVtIG1vZGUpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBzd2l0Y2ggKG1vZGUpIHsKLSAgICBjYXNlIEdMX0ZST05UOgotICAgIGNhc2UgR0xfQkFDSzoKLSAgICBjYXNlIEdMX0ZST05UX0FORF9CQUNLOgotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgIH0KLSAgICBjLT5jdWxsLmN1bGxGYWNlID0gbW9kZTsKLX0KLQotdm9pZCBnbEZyb250RmFjZShHTGVudW0gbW9kZSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIHN3aXRjaCAobW9kZSkgewotICAgIGNhc2UgR0xfQ1c6Ci0gICAgY2FzZSBHTF9DQ1c6Ci0gICAgICAgIGJyZWFrOwotICAgIGRlZmF1bHQ6Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgYy0+Y3VsbC5mcm9udEZhY2UgPSBtb2RlOwotfQotCi12b2lkIGdsSGludChHTGVudW0gdGFyZ2V0LCBHTGVudW0gbW9kZSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIHN3aXRjaCAodGFyZ2V0KSB7Ci0gICAgY2FzZSBHTF9GT0dfSElOVDoKLSAgICBjYXNlIEdMX0dFTkVSQVRFX01JUE1BUF9ISU5UOgotICAgIGNhc2UgR0xfTElORV9TTU9PVEhfSElOVDoKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9QT0lOVF9TTU9PVEhfSElOVDoKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy5lbmFibGVEaXNhYmxlKGMsIAotICAgICAgICAgICAgICAgIEdHTF9QT0lOVF9TTU9PVEhfTklDRSwgbW9kZT09R0xfTklDRVNUKTsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9QRVJTUEVDVElWRV9DT1JSRUNUSU9OX0hJTlQ6Ci0gICAgICAgIGMtPnBlcnNwZWN0aXZlID0gKG1vZGUgPT0gR0xfTklDRVNUKSA/IDEgOiAwOwotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgIH0KLX0KLQotdm9pZCBnbEVuYWJsZShHTGVudW0gY2FwKSB7Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBlbmFibGVfZGlzYWJsZShjLCBjYXAsIDEpOwotfQotdm9pZCBnbERpc2FibGUoR0xlbnVtIGNhcCkgewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgZW5hYmxlX2Rpc2FibGUoYywgY2FwLCAwKTsKLX0KLQotdm9pZCBnbEZpbmlzaCgpCi17IC8vIG5vdGhpbmcgdG8gZG8gZm9yIG91ciBzb2Z0d2FyZSBpbXBsZW1lbnRhdGlvbgotfQotCi12b2lkIGdsRmx1c2goKQoteyAvLyBub3RoaW5nIHRvIGRvIGZvciBvdXIgc29mdHdhcmUgaW1wbGVtZW50YXRpb24KLX0KLQotR0xlbnVtIGdsR2V0RXJyb3IoKQotewotICAgIC8vIEZyb20gT3BlbkdMfEVTIDEuMCBzcGVjaWZpY2F0aW9uOgotICAgIC8vIElmIG1vcmUgdGhhbiBvbmUgZmxhZyBoYXMgcmVjb3JkZWQgYW4gZXJyb3IsIGdsR2V0RXJyb3IgcmV0dXJucwotICAgIC8vIGFuZCBjbGVhcnMgYW4gYXJiaXRyYXJ5IGVycm9yIGZsYWcgdmFsdWUuIFRodXMsIGdsR2V0RXJyb3Igc2hvdWxkCi0gICAgLy8gYWx3YXlzIGJlIGNhbGxlZCBpbiBhIGxvb3AsIHVudGlsIGl0IHJldHVybnMgR0xfTk9fRVJST1IsCi0gICAgLy8gaWYgYWxsIGVycm9yIGZsYWdzIGFyZSB0byBiZSByZXNldC4KLQotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKGMtPmVycm9yKSB7Ci0gICAgICAgIGNvbnN0IEdMZW51bSByZXQoYy0+ZXJyb3IpOwotICAgICAgICBjLT5lcnJvciA9IDA7Ci0gICAgICAgIHJldHVybiByZXQ7Ci0gICAgfQotICAgIAotICAgIGlmIChjLT5yYXN0ZXJpemVyLmVycm9yKSB7Ci0gICAgICAgIGNvbnN0IEdMZW51bSByZXQoYy0+cmFzdGVyaXplci5lcnJvcik7Ci0gICAgICAgIGMtPnJhc3Rlcml6ZXIuZXJyb3IgPSAwOwotICAgICAgICByZXR1cm4gcmV0OwotICAgIH0KLQotICAgIHJldHVybiBHTF9OT19FUlJPUjsKLX0KLQotY29uc3QgR0x1Ynl0ZSogZ2xHZXRTdHJpbmcoR0xlbnVtIHN0cmluZykKLXsKLSAgICBzd2l0Y2ggKHN0cmluZykgewotICAgIGNhc2UgR0xfVkVORE9SOiAgICAgcmV0dXJuIChjb25zdCBHTHVieXRlKilnVmVuZG9yU3RyaW5nOwotICAgIGNhc2UgR0xfUkVOREVSRVI6ICAgcmV0dXJuIChjb25zdCBHTHVieXRlKilnUmVuZGVyZXJTdHJpbmc7Ci0gICAgY2FzZSBHTF9WRVJTSU9OOiAgICByZXR1cm4gKGNvbnN0IEdMdWJ5dGUqKWdWZXJzaW9uU3RyaW5nOwotICAgIGNhc2UgR0xfRVhURU5TSU9OUzogcmV0dXJuIChjb25zdCBHTHVieXRlKilnRXh0ZW5zaW9uc1N0cmluZzsKLSAgICB9Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgIHJldHVybiAwOwotfQotCi12b2lkIGdsR2V0SW50ZWdlcnYoR0xlbnVtIHBuYW1lLCBHTGludCAqcGFyYW1zKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgc3dpdGNoIChwbmFtZSkgewotICAgIGNhc2UgR0xfQUxJQVNFRF9QT0lOVF9TSVpFX1JBTkdFOgotICAgICAgICBwYXJhbXNbMF0gPSAwOwotICAgICAgICBwYXJhbXNbMV0gPSBHR0xfTUFYX0FMSUFTRURfUE9JTlRfU0laRTsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9BTElBU0VEX0xJTkVfV0lEVEhfUkFOR0U6Ci0gICAgICAgIHBhcmFtc1swXSA9IDA7Ci0gICAgICAgIHBhcmFtc1sxXSA9IEdHTF9NQVhfQUxJQVNFRF9QT0lOVF9TSVpFOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX0FMUEhBX0JJVFM6IHsKLSAgICAgICAgaW50IGluZGV4ID0gYy0+cmFzdGVyaXplci5zdGF0ZS5idWZmZXJzLmNvbG9yLmZvcm1hdDsKLSAgICAgICAgR0dMRm9ybWF0IGNvbnN0ICogZm9ybWF0cyA9IGdnbEdldFBpeGVsRm9ybWF0VGFibGUoKTsKLSAgICAgICAgcGFyYW1zWzBdID0gZm9ybWF0c1tpbmRleF0uYWggLSBmb3JtYXRzW2luZGV4XS5hbDsKLSAgICAgICAgYnJlYWs7IAotICAgICAgICB9Ci0gICAgY2FzZSBHTF9SRURfQklUUzogewotICAgICAgICBpbnQgaW5kZXggPSBjLT5yYXN0ZXJpemVyLnN0YXRlLmJ1ZmZlcnMuY29sb3IuZm9ybWF0OwotICAgICAgICBHR0xGb3JtYXQgY29uc3QgKiBmb3JtYXRzID0gZ2dsR2V0UGl4ZWxGb3JtYXRUYWJsZSgpOwotICAgICAgICBwYXJhbXNbMF0gPSBmb3JtYXRzW2luZGV4XS5yaCAtIGZvcm1hdHNbaW5kZXhdLnJsOwotICAgICAgICBicmVhazsgCi0gICAgICAgIH0KLSAgICBjYXNlIEdMX0dSRUVOX0JJVFM6IHsKLSAgICAgICAgaW50IGluZGV4ID0gYy0+cmFzdGVyaXplci5zdGF0ZS5idWZmZXJzLmNvbG9yLmZvcm1hdDsKLSAgICAgICAgR0dMRm9ybWF0IGNvbnN0ICogZm9ybWF0cyA9IGdnbEdldFBpeGVsRm9ybWF0VGFibGUoKTsKLSAgICAgICAgcGFyYW1zWzBdID0gZm9ybWF0c1tpbmRleF0uZ2ggLSBmb3JtYXRzW2luZGV4XS5nbDsKLSAgICAgICAgYnJlYWs7IAotICAgICAgICB9Ci0gICAgY2FzZSBHTF9CTFVFX0JJVFM6IHsKLSAgICAgICAgaW50IGluZGV4ID0gYy0+cmFzdGVyaXplci5zdGF0ZS5idWZmZXJzLmNvbG9yLmZvcm1hdDsKLSAgICAgICAgR0dMRm9ybWF0IGNvbnN0ICogZm9ybWF0cyA9IGdnbEdldFBpeGVsRm9ybWF0VGFibGUoKTsKLSAgICAgICAgcGFyYW1zWzBdID0gZm9ybWF0c1tpbmRleF0uYmggLSBmb3JtYXRzW2luZGV4XS5ibDsKLSAgICAgICAgYnJlYWs7IAotICAgICAgICB9Ci0gICAgY2FzZSBHTF9DT01QUkVTU0VEX1RFWFRVUkVfRk9STUFUUzoKLSAgICAgICAgcGFyYW1zWyAwXSA9IEdMX1BBTEVUVEU0X1JHQjhfT0VTOwotICAgICAgICBwYXJhbXNbIDFdID0gR0xfUEFMRVRURTRfUkdCQThfT0VTOwotICAgICAgICBwYXJhbXNbIDJdID0gR0xfUEFMRVRURTRfUjVfRzZfQjVfT0VTOwotICAgICAgICBwYXJhbXNbIDNdID0gR0xfUEFMRVRURTRfUkdCQTRfT0VTOwotICAgICAgICBwYXJhbXNbIDRdID0gR0xfUEFMRVRURTRfUkdCNV9BMV9PRVM7Ci0gICAgICAgIHBhcmFtc1sgNV0gPSBHTF9QQUxFVFRFOF9SR0I4X09FUzsKLSAgICAgICAgcGFyYW1zWyA2XSA9IEdMX1BBTEVUVEU4X1JHQkE4X09FUzsKLSAgICAgICAgcGFyYW1zWyA3XSA9IEdMX1BBTEVUVEU4X1I1X0c2X0I1X09FUzsKLSAgICAgICAgcGFyYW1zWyA4XSA9IEdMX1BBTEVUVEU4X1JHQkE0X09FUzsKLSAgICAgICAgcGFyYW1zWyA5XSA9IEdMX1BBTEVUVEU4X1JHQjVfQTFfT0VTOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX0RFUFRIX0JJVFM6Ci0gICAgICAgIHBhcmFtc1swXSA9IGMtPnJhc3Rlcml6ZXIuc3RhdGUuYnVmZmVycy5kZXB0aC5mb3JtYXQgPyAwIDogMTY7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfSU1QTEVNRU5UQVRJT05fQ09MT1JfUkVBRF9GT1JNQVRfT0VTOgotICAgICAgICBwYXJhbXNbMF0gPSBHTF9SR0I7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfSU1QTEVNRU5UQVRJT05fQ09MT1JfUkVBRF9UWVBFX09FUzoKLSAgICAgICAgcGFyYW1zWzBdID0gR0xfVU5TSUdORURfU0hPUlRfNV82XzU7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfTUFYX0xJR0hUUzoKLSAgICAgICAgcGFyYW1zWzBdID0gT0dMRVNfTUFYX0xJR0hUUzsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9NQVhfQ0xJUF9QTEFORVM6Ci0gICAgICAgIHBhcmFtc1swXSA9IE9HTEVTX01BWF9DTElQX1BMQU5FUzsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9NQVhfTU9ERUxWSUVXX1NUQUNLX0RFUFRIOgotICAgICAgICBwYXJhbXNbMF0gPSBPR0xFU19NT0RFTFZJRVdfU1RBQ0tfREVQVEg7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfTUFYX1BST0pFQ1RJT05fU1RBQ0tfREVQVEg6Ci0gICAgICAgIHBhcmFtc1swXSA9IE9HTEVTX1BST0pFQ1RJT05fU1RBQ0tfREVQVEg7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfTUFYX1RFWFRVUkVfU1RBQ0tfREVQVEg6Ci0gICAgICAgIHBhcmFtc1swXSA9IE9HTEVTX1RFWFRVUkVfU1RBQ0tfREVQVEg7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfTUFYX1RFWFRVUkVfU0laRToKLSAgICAgICAgcGFyYW1zWzBdID0gR0dMX01BWF9URVhUVVJFX1NJWkU7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfTUFYX1RFWFRVUkVfVU5JVFM6Ci0gICAgICAgIHBhcmFtc1swXSA9IEdHTF9URVhUVVJFX1VOSVRfQ09VTlQ7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfTUFYX1ZJRVdQT1JUX0RJTVM6Ci0gICAgICAgIHBhcmFtc1swXSA9IEdHTF9NQVhfVklFV1BPUlRfRElNUzsKLSAgICAgICAgcGFyYW1zWzFdID0gR0dMX01BWF9WSUVXUE9SVF9ESU1TOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX05VTV9DT01QUkVTU0VEX1RFWFRVUkVfRk9STUFUUzoKLSAgICAgICAgcGFyYW1zWzBdID0gT0dMRVNfTlVNX0NPTVBSRVNTRURfVEVYVFVSRV9GT1JNQVRTOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1NNT09USF9MSU5FX1dJRFRIX1JBTkdFOgotICAgICAgICBwYXJhbXNbMF0gPSAwOwotICAgICAgICBwYXJhbXNbMV0gPSBHR0xfTUFYX1NNT09USF9MSU5FX1dJRFRIOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1NNT09USF9QT0lOVF9TSVpFX1JBTkdFOgotICAgICAgICBwYXJhbXNbMF0gPSAwOwotICAgICAgICBwYXJhbXNbMV0gPSBHR0xfTUFYX1NNT09USF9QT0lOVF9TSVpFOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1NURU5DSUxfQklUUzoKLSAgICAgICAgcGFyYW1zWzBdID0gMDsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9TVUJQSVhFTF9CSVRTOgotICAgICAgICBwYXJhbXNbMF0gPSBHR0xfU1VCUElYRUxfQklUUzsKLSAgICAgICAgYnJlYWs7Ci0KLSAgICBjYXNlIEdMX01PREVMVklFV19NQVRSSVhfRkxPQVRfQVNfSU5UX0JJVFNfT0VTOgotICAgICAgICBtZW1jcHkoIHBhcmFtcywKLSAgICAgICAgICAgICAgICBjLT50cmFuc2Zvcm1zLm1vZGVsdmlldy50b3AoKS5lbGVtZW50cygpLAotICAgICAgICAgICAgICAgIDE2KnNpemVvZihHTGludCkpOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1BST0pFQ1RJT05fTUFUUklYX0ZMT0FUX0FTX0lOVF9CSVRTX09FUzoKLSAgICAgICAgbWVtY3B5KCBwYXJhbXMsCi0gICAgICAgICAgICAgICAgYy0+dHJhbnNmb3Jtcy5wcm9qZWN0aW9uLnRvcCgpLmVsZW1lbnRzKCksCi0gICAgICAgICAgICAgICAgMTYqc2l6ZW9mKEdMaW50KSk7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfVEVYVFVSRV9NQVRSSVhfRkxPQVRfQVNfSU5UX0JJVFNfT0VTOgotICAgICAgICBtZW1jcHkoIHBhcmFtcywKLSAgICAgICAgICAgICAgICBjLT50cmFuc2Zvcm1zLnRleHR1cmVbYy0+dGV4dHVyZXMuYWN0aXZlXS50b3AoKS5lbGVtZW50cygpLAotICAgICAgICAgICAgICAgIDE2KnNpemVvZihHTGludCkpOwotICAgICAgICBicmVhazsKLQotICAgIGRlZmF1bHQ6Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIGJyZWFrOwotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIGdsUG9pbnRTaXplKEdMZmxvYXQgc2l6ZSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmIChzaXplIDw9IDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBjLT5wb2ludC5zaXplID0gVFJJX0ZST01fRklYRUQoZ2dsRmxvYXRUb0ZpeGVkKHNpemUpKTsKLX0KLQotdm9pZCBnbFBvaW50U2l6ZXgoR0xmaXhlZCBzaXplKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKHNpemUgPD0gMCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGMtPnBvaW50LnNpemUgPSBUUklfRlJPTV9GSVhFRChzaXplKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIGdsTGluZVdpZHRoKEdMZmxvYXQgd2lkdGgpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBpZiAod2lkdGggPD0gMCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGMtPmxpbmUud2lkdGggPSBUUklfRlJPTV9GSVhFRChnZ2xGbG9hdFRvRml4ZWQod2lkdGgpKTsKLX0KLQotdm9pZCBnbExpbmVXaWR0aHgoR0xmaXhlZCB3aWR0aCkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICh3aWR0aCA8PSAwKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgYy0+bGluZS53aWR0aCA9IFRSSV9GUk9NX0ZJWEVEKHdpZHRoKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIGdsQ29sb3JNYXNrKEdMYm9vbGVhbiByLCBHTGJvb2xlYW4gZywgR0xib29sZWFuIGIsIEdMYm9vbGVhbiBhKSB7Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmNvbG9yTWFzayhjLCByLCBnLCBiLCBhKTsKLX0KLQotdm9pZCBnbERlcHRoTWFzayhHTGJvb2xlYW4gZmxhZykgewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+cmFzdGVyaXplci5wcm9jcy5kZXB0aE1hc2soYywgZmxhZyk7Ci19Ci0KLXZvaWQgZ2xTdGVuY2lsTWFzayhHTHVpbnQgbWFzaykgewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+cmFzdGVyaXplci5wcm9jcy5zdGVuY2lsTWFzayhjLCBtYXNrKTsKLX0KLQotdm9pZCBnbERlcHRoRnVuYyhHTGVudW0gZnVuYykgewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+cmFzdGVyaXplci5wcm9jcy5kZXB0aEZ1bmMoYywgZnVuYyk7Ci19Ci0KLXZvaWQgZ2xMb2dpY09wKEdMZW51bSBvcGNvZGUpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MubG9naWNPcChjLCBvcGNvZGUpOwotfQotCi12b2lkIGdsQWxwaGFGdW5jeChHTGVudW0gZnVuYywgR0xjbGFtcHggcmVmKSB7Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmFscGhhRnVuY3goYywgZnVuYywgcmVmKTsKLX0KLQotdm9pZCBnbEJsZW5kRnVuYyhHTGVudW0gc2ZhY3RvciwgR0xlbnVtIGRmYWN0b3IpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuYmxlbmRGdW5jKGMsIHNmYWN0b3IsIGRmYWN0b3IpOwotfQotCi12b2lkIGdsQ2xlYXIoR0xiaXRmaWVsZCBtYXNrKSB7Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmNsZWFyKGMsIG1hc2spOwotfQotCi12b2lkIGdsQ2xlYXJDb2xvcngoR0xjbGFtcHggcmVkLCBHTGNsYW1weCBncmVlbiwgR0xjbGFtcHggYmx1ZSwgR0xjbGFtcHggYWxwaGEpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuY2xlYXJDb2xvcngoYywgcmVkLCBncmVlbiwgYmx1ZSwgYWxwaGEpOwotfQotCi12b2lkIGdsQ2xlYXJDb2xvcihHTGNsYW1wZiByLCBHTGNsYW1wZiBnLCBHTGNsYW1wZiBiLCBHTGNsYW1wZiBhKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+cmFzdGVyaXplci5wcm9jcy5jbGVhckNvbG9yeChjLAotICAgICAgICAgICAgICAgICAgICBnZ2xGbG9hdFRvRml4ZWQociksCi0gICAgICAgICAgICAgICAgICAgIGdnbEZsb2F0VG9GaXhlZChnKSwKLSAgICAgICAgICAgICAgICAgICAgZ2dsRmxvYXRUb0ZpeGVkKGIpLAotICAgICAgICAgICAgICAgICAgICBnZ2xGbG9hdFRvRml4ZWQoYSkpOwotfQotCi12b2lkIGdsQ2xlYXJEZXB0aHgoR0xjbGFtcHggZGVwdGgpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuY2xlYXJEZXB0aHgoYywgZGVwdGgpOwotfQotCi12b2lkIGdsQ2xlYXJEZXB0aGYoR0xjbGFtcGYgZGVwdGgpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmNsZWFyRGVwdGh4KGMsIGdnbEZsb2F0VG9GaXhlZChkZXB0aCkpOwotfQotCi12b2lkIGdsQ2xlYXJTdGVuY2lsKEdMaW50IHMpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuY2xlYXJTdGVuY2lsKGMsIHMpOwotfQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC9zdGF0ZS5oIGIvb3BlbmdsL2xpYmFnbC9zdGF0ZS5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1NWE1Y2NiLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJhZ2wvc3RhdGUuaAorKysgL2Rldi9udWxsCkBAIC0xLDU0ICswLDAgQEAKLS8qIGxpYnMvb3BlbmdsZXMvc3RhdGUuaAotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpZm5kZWYgQU5EUk9JRF9PUEVOR0xFU19TVEFURV9ICi0jZGVmaW5lIEFORFJPSURfT1BFTkdMRVNfU1RBVEVfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3RkZGVmLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDxwcml2YXRlL3BpeGVsZmxpbmdlci9nZ2xfY29udGV4dC5oPgotCi0jaW5jbHVkZSA8R0xFUy9nbC5oPgotCi0jaW5jbHVkZSA8c3RkaW8uaD4KLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi1vZ2xlc19jb250ZXh0X3QgKm9nbGVzX2luaXQoc2l6ZV90IGV4dHJhKTsKLXZvaWQgb2dsZXNfdW5pbml0KG9nbGVzX2NvbnRleHRfdCogYyk7Ci12b2lkIF9vZ2xlc19lcnJvcihvZ2xlc19jb250ZXh0X3QqIGMsIEdMZW51bSBlcnJvcik7Ci0KLSNpZm5kZWYgVFJBQ0VfR0xfRVJST1JTCi0jZGVmaW5lIFRSQUNFX0dMX0VSUk9SUyAwCi0jZW5kaWYKLQotI2lmIFRSQUNFX0dMX0VSUk9SUwotI2RlZmluZSBvZ2xlc19lcnJvcihjLCBlcnJvcikgXAotZG8geyBcCi0gIHByaW50Zigib2dsZXNfZXJyb3IgYXQgZmlsZSAlcyBsaW5lICVkXG4iLCBfX0ZJTEVfXywgX19MSU5FX18pOyBcCi0gIF9vZ2xlc19lcnJvcihjLCBlcnJvcik7IFwKLX0gd2hpbGUgKDApCi0jZWxzZSAvKiAhVFJBQ0VfR0xfRVJST1JTICovCi0jZGVmaW5lIG9nbGVzX2Vycm9yKGMsIGVycm9yKSBfb2dsZXNfZXJyb3IoKGMpLCAoZXJyb3IpKQotI2VuZGlmCi0KLX07IC8vIG5hbWVzcGFjZSBhbmRyb2lkCi0KLSNlbmRpZiAvLyBBTkRST0lEX09QRU5HTEVTX1NUQVRFX0gKLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC90ZXh0dXJlLmNwcCBiL29wZW5nbC9saWJhZ2wvdGV4dHVyZS5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGI2ZjUzNGIuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC90ZXh0dXJlLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDE0MjEgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy90ZXh0dXJlLmNwcAotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgImNvbnRleHQuaCIKLSNpbmNsdWRlICJmcC5oIgotI2luY2x1ZGUgInN0YXRlLmgiCi0jaW5jbHVkZSAidGV4dHVyZS5oIgotI2luY2x1ZGUgIlRleHR1cmVPYmplY3RNYW5hZ2VyLmgiCi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgdm9pZCBiaW5kVGV4dHVyZVRtdSgKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMsIGludCB0bXUsIEdMdWludCB0ZXh0dXJlLCBjb25zdCBzcDxFR0xUZXh0dXJlT2JqZWN0PiYgdGV4KTsKLQotc3RhdGljIF9fYXR0cmlidXRlX18oKG5vaW5saW5lKSkKLXZvaWQgZ2VuZXJhdGVNaXBtYXAob2dsZXNfY29udGV4dF90KiBjLCBHTGludCBsZXZlbCk7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jcHJhZ21hIG1hcmsgSW5pdAotI2VuZGlmCi0KLXZvaWQgb2dsZXNfaW5pdF90ZXh0dXJlKG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBjLT50ZXh0dXJlcy5wYWNrQWxpZ25tZW50ICAgPSA0OwotICAgIGMtPnRleHR1cmVzLnVucGFja0FsaWdubWVudCA9IDQ7Ci0KLSAgICAvLyBlYWNoIGNvbnRleHQgaGFzIGEgZGVmYXVsdCBuYW1lZCAoMCkgdGV4dHVyZSAobm90IHNoYXJlZCkKLSAgICBjLT50ZXh0dXJlcy5kZWZhdWx0VGV4dHVyZSA9IG5ldyBFR0xUZXh0dXJlT2JqZWN0KCk7Ci0gICAgYy0+dGV4dHVyZXMuZGVmYXVsdFRleHR1cmUtPmluY1N0cm9uZyhjKTsKLSAgICAKLSAgICAvLyBiaW5kIHRoZSBkZWZhdWx0IHRleHR1cmUgdG8gZWFjaCB0ZXh0dXJlIHVuaXQKLSAgICBmb3IgKGludCBpPTA7IGk8R0dMX1RFWFRVUkVfVU5JVF9DT1VOVCA7IGkrKykgewotICAgICAgICBiaW5kVGV4dHVyZVRtdShjLCBpLCAwLCBjLT50ZXh0dXJlcy5kZWZhdWx0VGV4dHVyZSk7Ci0gICAgICAgIG1lbXNldChjLT5jdXJyZW50LnRleHR1cmVbaV0udiwgMCwgc2l6ZW9mKHZlYzRfdCkpOwotICAgICAgICBjLT5jdXJyZW50LnRleHR1cmVbaV0uUSA9IDB4MTAwMDA7Ci0gICAgfQotfQotCi12b2lkIG9nbGVzX3VuaW5pdF90ZXh0dXJlKG9nbGVzX2NvbnRleHRfdCogYykKLXsKLSAgICBpZiAoYy0+dGV4dHVyZXMuZ2dsKQotICAgICAgICBnZ2xVbmluaXQoYy0+dGV4dHVyZXMuZ2dsKTsKLSAgICBjLT50ZXh0dXJlcy5kZWZhdWx0VGV4dHVyZS0+ZGVjU3Ryb25nKGMpOwotICAgIGZvciAoaW50IGk9MDsgaTxHR0xfVEVYVFVSRV9VTklUX0NPVU5UIDsgaSsrKSB7Ci0gICAgICAgIGlmIChjLT50ZXh0dXJlcy50bXVbaV0udGV4dHVyZSkKLSAgICAgICAgICAgIGMtPnRleHR1cmVzLnRtdVtpXS50ZXh0dXJlLT5kZWNTdHJvbmcoYyk7Ci0gICAgfQotfQotCi1zdGF0aWMgX19hdHRyaWJ1dGVfXygobm9pbmxpbmUpKQotdm9pZCB2YWxpZGF0ZV90bXUob2dsZXNfY29udGV4dF90KiBjLCBpbnQgaSkKLXsKLSAgICB0ZXh0dXJlX3VuaXRfdCYgdShjLT50ZXh0dXJlcy50bXVbaV0pOwotICAgIGlmICh1LmRpcnR5KSB7Ci0gICAgICAgIHUuZGlydHkgPSAwOwotICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmFjdGl2ZVRleHR1cmUoYywgaSk7Ci0gICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuYmluZFRleHR1cmUoYywgJih1LnRleHR1cmUtPnN1cmZhY2UpKTsKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy50ZXhHZW5pKGMsIEdHTF9TLAotICAgICAgICAgICAgICAgIEdHTF9URVhUVVJFX0dFTl9NT0RFLCBHR0xfQVVUT01BVElDKTsKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy50ZXhHZW5pKGMsIEdHTF9ULAotICAgICAgICAgICAgICAgIEdHTF9URVhUVVJFX0dFTl9NT0RFLCBHR0xfQVVUT01BVElDKTsKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy50ZXhQYXJhbWV0ZXJpKGMsIEdHTF9URVhUVVJFXzJELAotICAgICAgICAgICAgICAgIEdHTF9URVhUVVJFX1dSQVBfUywgdS50ZXh0dXJlLT53cmFwcyk7Ci0gICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MudGV4UGFyYW1ldGVyaShjLCBHR0xfVEVYVFVSRV8yRCwKLSAgICAgICAgICAgICAgICBHR0xfVEVYVFVSRV9XUkFQX1QsIHUudGV4dHVyZS0+d3JhcHQpOwotICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleFBhcmFtZXRlcmkoYywgR0dMX1RFWFRVUkVfMkQsCi0gICAgICAgICAgICAgICAgR0dMX1RFWFRVUkVfTUlOX0ZJTFRFUiwgdS50ZXh0dXJlLT5taW5fZmlsdGVyKTsKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy50ZXhQYXJhbWV0ZXJpKGMsIEdHTF9URVhUVVJFXzJELAotICAgICAgICAgICAgICAgIEdHTF9URVhUVVJFX01BR19GSUxURVIsIHUudGV4dHVyZS0+bWFnX2ZpbHRlcik7Ci0KLSAgICAgICAgLy8gZGlzYWJsZSB0aGlzIHRleHR1cmUgdW5pdCBpZiBpdCdzIG5vdCBjb21wbGV0ZQotICAgICAgICBpZiAoIXUudGV4dHVyZS0+aXNDb21wbGV0ZSgpKSB7Ci0gICAgICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmRpc2FibGUoYywgR0dMX1RFWFRVUkVfMkQpOwotICAgICAgICB9Ci0gICAgfQotfQotCi12b2lkIG9nbGVzX3ZhbGlkYXRlX3RleHR1cmVfaW1wbChvZ2xlc19jb250ZXh0X3QqIGMpCi17Ci0gICAgZm9yIChpbnQgaT0wIDsgaTxHR0xfVEVYVFVSRV9VTklUX0NPVU5UIDsgaSsrKSB7Ci0gICAgICAgIGlmIChjLT5yYXN0ZXJpemVyLnN0YXRlLnRleHR1cmVbaV0uZW5hYmxlKQotICAgICAgICAgICAgdmFsaWRhdGVfdG11KGMsIGkpOwotICAgIH0KLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmFjdGl2ZVRleHR1cmUoYywgYy0+dGV4dHVyZXMuYWN0aXZlKTsKLX0KLQotc3RhdGljCi12b2lkIGludmFsaWRhdGVfdGV4dHVyZShvZ2xlc19jb250ZXh0X3QqIGMsIGludCB0bXUsIHVpbnQ4X3QgZmxhZ3MgPSAweEZGKSB7Ci0gICAgYy0+dGV4dHVyZXMudG11W3RtdV0uZGlydHkgPSBmbGFnczsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jcHJhZ21hIG1hcmsgRm9ybWF0IGNvbnZlcnNpb24KLSNlbmRpZgotCi1zdGF0aWMgdWludDMyX3QgZ2wyZm9ybWF0X3RhYmxlWzZdWzRdID0gewotICAgIC8vIEJZVEUsIDU2NSwgNDQ0NCwgNTU1MQotICAgIHsgR0dMX1BJWEVMX0ZPUk1BVF9BXzgsCi0gICAgICAwLCAwLCAwIH0sICAgICAgICAgICAgICAgICAgICAgICAgLy8gR0xfQUxQSEEKLSAgICB7IEdHTF9QSVhFTF9GT1JNQVRfUkdCXzg4OCwKLSAgICAgIEdHTF9QSVhFTF9GT1JNQVRfUkdCXzU2NSwKLSAgICAgIDAsIDAgfSwgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBHTF9SR0IKLSAgICB7IEdHTF9QSVhFTF9GT1JNQVRfUkdCQV84ODg4LAotICAgICAgMCwKLSAgICAgIEdHTF9QSVhFTF9GT1JNQVRfUkdCQV80NDQ0LAotICAgICAgR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzU1NTEgfSwgICAgIC8vIEdMX1JHQkEKLSAgICB7IEdHTF9QSVhFTF9GT1JNQVRfTF84LAotICAgICAgMCwgMCwgMCB9LCAgICAgICAgICAgICAgICAgICAgICAgIC8vIEdMX0xVTUlOQU5DRQotICAgIHsgR0dMX1BJWEVMX0ZPUk1BVF9MQV84OCwKLSAgICAgIDAsIDAsIDAgfSwgICAgICAgICAgICAgICAgICAgICAgICAvLyBHTF9MVU1JTkFOQ0VfQUxQSEEKLX07Ci0KLXN0YXRpYyBpbnQzMl90IGNvbnZlcnRHTFBpeGVsRm9ybWF0KEdMaW50IGZvcm1hdCwgR0xlbnVtIHR5cGUpCi17Ci0gICAgaW50MzJfdCBmaSA9IC0xOwotICAgIGludDMyX3QgdGkgPSAtMTsKLSAgICBzd2l0Y2ggKGZvcm1hdCkgewotICAgIGNhc2UgR0xfQUxQSEE6ICAgICAgICAgICAgICBmaSA9IDA7ICAgICBicmVhazsKLSAgICBjYXNlIEdMX1JHQjogICAgICAgICAgICAgICAgZmkgPSAxOyAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9SR0JBOiAgICAgICAgICAgICAgIGZpID0gMjsgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfTFVNSU5BTkNFOiAgICAgICAgICBmaSA9IDM7ICAgICBicmVhazsKLSAgICBjYXNlIEdMX0xVTUlOQU5DRV9BTFBIQTogICAgZmkgPSA0OyAgICAgYnJlYWs7Ci0gICAgfQotICAgIHN3aXRjaCAodHlwZSkgewotICAgIGNhc2UgR0xfVU5TSUdORURfQllURTogICAgICAgICAgdGkgPSAwOyBicmVhazsKLSAgICBjYXNlIEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81OiAgIHRpID0gMTsgYnJlYWs7Ci0gICAgY2FzZSBHTF9VTlNJR05FRF9TSE9SVF80XzRfNF80OiB0aSA9IDI7IGJyZWFrOwotICAgIGNhc2UgR0xfVU5TSUdORURfU0hPUlRfNV81XzVfMTogdGkgPSAzOyBicmVhazsKLSAgICB9Ci0gICAgaWYgKGZpPT0tMSB8fCB0aT09LTEpCi0gICAgICAgIHJldHVybiAwOwotICAgIHJldHVybiBnbDJmb3JtYXRfdGFibGVbZmldW3RpXTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1zdGF0aWMgR0xlbnVtIHZhbGlkRm9ybWF0VHlwZShvZ2xlc19jb250ZXh0X3QqIGMsIEdMZW51bSBmb3JtYXQsIEdMZW51bSB0eXBlKQotewotICAgIEdMZW51bSBlcnJvciA9IDA7Ci0gICAgaWYgKGZvcm1hdDxHTF9BTFBIQSB8fCBmb3JtYXQ+R0xfTFVNSU5BTkNFX0FMUEhBKSB7Ci0gICAgICAgIGVycm9yID0gR0xfSU5WQUxJRF9FTlVNOwotICAgIH0KLSAgICBpZiAodHlwZSAhPSBHTF9VTlNJR05FRF9CWVRFICYmIHR5cGUgIT0gR0xfVU5TSUdORURfU0hPUlRfNF80XzRfNCAmJgotICAgICAgICB0eXBlICE9IEdMX1VOU0lHTkVEX1NIT1JUXzVfNV81XzEgJiYgdHlwZSAhPSBHTF9VTlNJR05FRF9TSE9SVF81XzZfNSkgewotICAgICAgICBlcnJvciA9IEdMX0lOVkFMSURfRU5VTTsKLSAgICB9Ci0gICAgaWYgKHR5cGUgPT0gR0xfVU5TSUdORURfU0hPUlRfNV82XzUgJiYgZm9ybWF0ICE9IEdMX1JHQikgewotICAgICAgICBlcnJvciA9IEdMX0lOVkFMSURfT1BFUkFUSU9OOwotICAgIH0KLSAgICBpZiAoKHR5cGUgPT0gR0xfVU5TSUdORURfU0hPUlRfNF80XzRfNCB8fAotICAgICAgICAgdHlwZSA9PSBHTF9VTlNJR05FRF9TSE9SVF81XzVfNV8xKSAgJiYgZm9ybWF0ICE9IEdMX1JHQkEpIHsKLSAgICAgICAgZXJyb3IgPSBHTF9JTlZBTElEX09QRVJBVElPTjsKLSAgICB9Ci0gICAgaWYgKGVycm9yKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIGVycm9yKTsKLSAgICB9Ci0gICAgcmV0dXJuIGVycm9yOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUdHTENvbnRleHQqIGdldFJhc3Rlcml6ZXIob2dsZXNfY29udGV4dF90KiBjKQotewotICAgIEdHTENvbnRleHQqIGdnbCA9IGMtPnRleHR1cmVzLmdnbDsKLSAgICBpZiAoZ2dsX3VubGlrZWx5KCFnZ2wpKSB7Ci0gICAgICAgIC8vIHRoaXMgaXMgcXVpdGUgaGVhdnkgdGhlIGZpcnN0IHRpbWUuLi4KLSAgICAgICAgZ2dsSW5pdCgmZ2dsKTsKLSAgICAgICAgaWYgKCFnZ2wpIHsKLSAgICAgICAgICAgIHJldHVybiAwOwotICAgICAgICB9Ci0gICAgICAgIEdHTGZpeGVkIGNvbG9yc1s0XSA9IHsgMCwgMCwgMCwgMHgxMDAwMCB9OwotICAgICAgICBjLT50ZXh0dXJlcy5nZ2wgPSBnZ2w7Ci0gICAgICAgIGdnbC0+YWN0aXZlVGV4dHVyZShnZ2wsIDApOwotICAgICAgICBnZ2wtPmVuYWJsZShnZ2wsIEdHTF9URVhUVVJFXzJEKTsKLSAgICAgICAgZ2dsLT50ZXhFbnZpKGdnbCwgR0dMX1RFWFRVUkVfRU5WLCBHR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0dMX1JFUExBQ0UpOwotICAgICAgICBnZ2wtPmRpc2FibGUoZ2dsLCBHR0xfRElUSEVSKTsKLSAgICAgICAgZ2dsLT5zaGFkZU1vZGVsKGdnbCwgR0dMX0ZMQVQpOwotICAgICAgICBnZ2wtPmNvbG9yNHh2KGdnbCwgY29sb3JzKTsKLSAgICB9Ci0gICAgcmV0dXJuIGdnbDsKLX0KLQotc3RhdGljIF9fYXR0cmlidXRlX18oKG5vaW5saW5lKSkKLWludCBjb3B5UGl4ZWxzKAotICAgICAgICBvZ2xlc19jb250ZXh0X3QqIGMsCi0gICAgICAgIGNvbnN0IEdHTFN1cmZhY2UmIGRzdCwKLSAgICAgICAgR0xpbnQgeG9mZnNldCwgR0xpbnQgeW9mZnNldCwKLSAgICAgICAgY29uc3QgR0dMU3VyZmFjZSYgc3JjLAotICAgICAgICBHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHcsIEdMc2l6ZWkgaCkKLXsKLSAgICBpZiAoKGRzdC5mb3JtYXQgPT0gc3JjLmZvcm1hdCkgJiYKLSAgICAgICAgKGRzdC5zdHJpZGUgPT0gc3JjLnN0cmlkZSkgJiYKLSAgICAgICAgKGRzdC53aWR0aCA9PSBzcmMud2lkdGgpICYmCi0gICAgICAgIChkc3QuaGVpZ2h0ID09IHNyYy5oZWlnaHQpICYmCi0gICAgICAgIChkc3Quc3RyaWRlID4gMCkgJiYKLSAgICAgICAgKCh4fHkpID09IDApICYmCi0gICAgICAgICgoeG9mZnNldHx5b2Zmc2V0KSA9PSAwKSkKLSAgICB7Ci0gICAgICAgIC8vIHRoaXMgaXMgYSBjb21tb24gY2FzZS4uLgotICAgICAgICBjb25zdCBHR0xGb3JtYXQmIHBpeGVsRm9ybWF0KGMtPnJhc3Rlcml6ZXIuZm9ybWF0c1tzcmMuZm9ybWF0XSk7Ci0gICAgICAgIGNvbnN0IHNpemVfdCBzaXplID0gc3JjLmhlaWdodCAqIHNyYy5zdHJpZGUgKiBwaXhlbEZvcm1hdC5zaXplOwotICAgICAgICBtZW1jcHkoZHN0LmRhdGEsIHNyYy5kYXRhLCBzaXplKTsKLSAgICAgICAgcmV0dXJuIDA7Ci0gICAgfQotCi0gICAgLy8gdXNlIHBpeGVsLWZsaW5nZXIgdG8gaGFuZGxlIGFsbCB0aGUgY29udmVyc2lvbnMKLSAgICBHR0xDb250ZXh0KiBnZ2wgPSBnZXRSYXN0ZXJpemVyKGMpOwotICAgIGlmICghZ2dsKSB7Ci0gICAgICAgIC8vIHRoZSBvbmx5IHJlYXNvbiB0aGlzIHdvdWxkIGZhaWwgaXMgYmVjYXVzZSB3ZSByYW4gb3V0IG9mIG1lbW9yeQotICAgICAgICByZXR1cm4gR0xfT1VUX09GX01FTU9SWTsKLSAgICB9Ci0KLSAgICBnZ2wtPmNvbG9yQnVmZmVyKGdnbCwgJmRzdCk7Ci0gICAgZ2dsLT5iaW5kVGV4dHVyZShnZ2wsICZzcmMpOwotICAgIGdnbC0+dGV4Q29vcmQyaShnZ2wsIHgteG9mZnNldCwgeS15b2Zmc2V0KTsKLSAgICBnZ2wtPnJlY3RpKGdnbCwgeG9mZnNldCwgeW9mZnNldCwgeG9mZnNldCt3LCB5b2Zmc2V0K2gpOwotICAgIHJldHVybiAwOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyBfX2F0dHJpYnV0ZV9fKChub2lubGluZSkpCi1zcDxFR0xUZXh0dXJlT2JqZWN0PiBnZXRBbmRCaW5kQWN0aXZlVGV4dHVyZU9iamVjdChvZ2xlc19jb250ZXh0X3QqIGMpCi17Ci0gICAgc3A8RUdMVGV4dHVyZU9iamVjdD4gdGV4OwotICAgIGNvbnN0IGludCBhY3RpdmUgPSBjLT50ZXh0dXJlcy5hY3RpdmU7Ci0gICAgY29uc3QgR0x1aW50IG5hbWUgPSBjLT50ZXh0dXJlcy50bXVbYWN0aXZlXS5uYW1lOwotCi0gICAgLy8gZnJlZSB0aGUgcmVmZXJlbmNlIHRvIHRoZSBwcmV2aW91c2x5IGJvdW5kIG9iamVjdAotICAgIHRleHR1cmVfdW5pdF90JiB1KGMtPnRleHR1cmVzLnRtdVthY3RpdmVdKTsKLSAgICBpZiAodS50ZXh0dXJlKQotICAgICAgICB1LnRleHR1cmUtPmRlY1N0cm9uZyhjKTsKLQotICAgIGlmIChuYW1lID09IDApIHsKLSAgICAgICAgLy8gMCBpcyBvdXIgbG9jYWwgdGV4dHVyZSBvYmplY3QsIG5vdCBzaGFyZWQgd2l0aCBhbnlvbmUuIAotICAgICAgICAvLyBCdXQgaXQgYWZmZWN0cyBhbGwgYm91bmQgVE1VcyBpbW1lZGlhdGVseS4KLSAgICAgICAgLy8gKHdlIG5lZWQgdG8gaW52YWxpZGF0ZSBhbGwgdW5pdHMgYm91bmQgdG8gdGhpcyB0ZXh0dXJlIG9iamVjdCkKLSAgICAgICAgdGV4ID0gYy0+dGV4dHVyZXMuZGVmYXVsdFRleHR1cmU7Ci0gICAgICAgIGZvciAoaW50IGk9MCA7IGk8R0dMX1RFWFRVUkVfVU5JVF9DT1VOVCA7IGkrKykgewotICAgICAgICAgICAgaWYgKGMtPnRleHR1cmVzLnRtdVtpXS50ZXh0dXJlID09IHRleC5nZXQoKSkKLSAgICAgICAgICAgICAgICBpbnZhbGlkYXRlX3RleHR1cmUoYywgaSk7Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgewotICAgICAgICAvLyBnZXQgYSBuZXcgdGV4dHVyZSBvYmplY3QgZm9yIHRoYXQgbmFtZQotICAgICAgICB0ZXggPSBjLT5zdXJmYWNlTWFuYWdlci0+cmVwbGFjZVRleHR1cmUobmFtZSk7Ci0gICAgfQotCi0gICAgLy8gYmluZCB0aGlzIHRleHR1cmUgdG8gdGhlIGN1cnJlbnQgYWN0aXZlIHRleHR1cmUgdW5pdAotICAgIC8vIGFuZCBhZGQgYSByZWZlcmVuY2UgdG8gdGhpcyB0ZXh0dXJlIG9iamVjdAotICAgIHUudGV4dHVyZSA9IHRleC5nZXQoKTsKLSAgICB1LnRleHR1cmUtPmluY1N0cm9uZyhjKTsKLSAgICB1Lm5hbWUgPSBuYW1lOwotICAgIGludmFsaWRhdGVfdGV4dHVyZShjLCBhY3RpdmUpOyAgICAKLSAgICByZXR1cm4gdGV4OwotfQotCi12b2lkIGJpbmRUZXh0dXJlVG11KAotICAgIG9nbGVzX2NvbnRleHRfdCogYywgaW50IHRtdSwgR0x1aW50IHRleHR1cmUsIGNvbnN0IHNwPEVHTFRleHR1cmVPYmplY3Q+JiB0ZXgpCi17Ci0gICAgaWYgKHRleC5nZXQoKSA9PSBjLT50ZXh0dXJlcy50bXVbdG11XS50ZXh0dXJlKQotICAgICAgICByZXR1cm47Ci0gICAgCi0gICAgLy8gZnJlZSB0aGUgcmVmZXJlbmNlIHRvIHRoZSBwcmV2aW91c2x5IGJvdW5kIG9iamVjdAotICAgIHRleHR1cmVfdW5pdF90JiB1KGMtPnRleHR1cmVzLnRtdVt0bXVdKTsKLSAgICBpZiAodS50ZXh0dXJlKQotICAgICAgICB1LnRleHR1cmUtPmRlY1N0cm9uZyhjKTsKLQotICAgIC8vIGJpbmQgdGhpcyB0ZXh0dXJlIHRvIHRoZSBjdXJyZW50IGFjdGl2ZSB0ZXh0dXJlIHVuaXQKLSAgICAvLyBhbmQgYWRkIGEgcmVmZXJlbmNlIHRvIHRoaXMgdGV4dHVyZSBvYmplY3QKLSAgICB1LnRleHR1cmUgPSB0ZXguZ2V0KCk7Ci0gICAgdS50ZXh0dXJlLT5pbmNTdHJvbmcoYyk7Ci0gICAgdS5uYW1lID0gdGV4dHVyZTsKLSAgICBpbnZhbGlkYXRlX3RleHR1cmUoYywgdG11KTsKLX0KLQotaW50IGNyZWF0ZVRleHR1cmVTdXJmYWNlKG9nbGVzX2NvbnRleHRfdCogYywKLSAgICAgICAgR0dMU3VyZmFjZSoqIG91dFN1cmZhY2UsIGludDMyX3QqIG91dFNpemUsIEdMaW50IGxldmVsLAotICAgICAgICBHTGVudW0gZm9ybWF0LCBHTGVudW0gdHlwZSwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsCi0gICAgICAgIEdMZW51bSBjb21wcmVzc2VkRm9ybWF0ID0gMCkKLXsKLSAgICAvLyBmaW5kIG91dCB3aGljaCB0ZXh0dXJlIGlzIGJvdW5kIHRvIHRoZSBjdXJyZW50IHVuaXQKLSAgICBjb25zdCBpbnQgYWN0aXZlID0gYy0+dGV4dHVyZXMuYWN0aXZlOwotICAgIGNvbnN0IEdMdWludCBuYW1lID0gYy0+dGV4dHVyZXMudG11W2FjdGl2ZV0ubmFtZTsKLQotICAgIC8vIGNvbnZlcnQgdGhlIHBpeGVsZm9ybWF0IHRvIG9uZSB3ZSBjYW4gaGFuZGxlCi0gICAgY29uc3QgaW50MzJfdCBmb3JtYXRJZHggPSBjb252ZXJ0R0xQaXhlbEZvcm1hdChmb3JtYXQsIHR5cGUpOwotICAgIGlmIChmb3JtYXRJZHggPT0gMCkgeyAvLyB3ZSBkb24ndCBrbm93IHdoYXQgdG8gZG8gd2l0aCB0aGlzCi0gICAgICAgIHJldHVybiBHTF9JTlZBTElEX09QRVJBVElPTjsKLSAgICB9Ci0gICAgCi0gICAgLy8gZmlndXJlIG91dCB0aGUgc2l6ZSB3ZSBuZWVkIGFzIHdlbGwgYXMgdGhlIHN0cmlkZQotICAgIGNvbnN0IEdHTEZvcm1hdCYgcGl4ZWxGb3JtYXQoYy0+cmFzdGVyaXplci5mb3JtYXRzW2Zvcm1hdElkeF0pOwotICAgIGNvbnN0IGludDMyX3QgYWxpZ24gPSBjLT50ZXh0dXJlcy51bnBhY2tBbGlnbm1lbnQtMTsKLSAgICBjb25zdCBpbnQzMl90IGJwciA9ICgod2lkdGggKiBwaXhlbEZvcm1hdC5zaXplKSArIGFsaWduKSAmIH5hbGlnbjsKLSAgICBjb25zdCBzaXplX3Qgc2l6ZSA9IGJwciAqIGhlaWdodDsKLSAgICBjb25zdCBpbnQzMl90IHN0cmlkZSA9IGJwciAvIHBpeGVsRm9ybWF0LnNpemU7Ci0KLSAgICBpZiAobGV2ZWwgPiAwKSB7Ci0gICAgICAgIGNvbnN0IGludCBhY3RpdmUgPSBjLT50ZXh0dXJlcy5hY3RpdmU7Ci0gICAgICAgIEVHTFRleHR1cmVPYmplY3QqIHRleCA9IGMtPnRleHR1cmVzLnRtdVthY3RpdmVdLnRleHR1cmU7Ci0gICAgICAgIHN0YXR1c190IGVyciA9IHRleC0+cmVhbGxvY2F0ZShsZXZlbCwKLSAgICAgICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCBzdHJpZGUsIGZvcm1hdElkeCwgY29tcHJlc3NlZEZvcm1hdCwgYnByKTsKLSAgICAgICAgaWYgKGVyciAhPSBOT19FUlJPUikKLSAgICAgICAgICAgIHJldHVybiBHTF9PVVRfT0ZfTUVNT1JZOwotICAgICAgICBHR0xTdXJmYWNlJiBzdXJmYWNlID0gdGV4LT5lZGl0TWlwKGxldmVsKTsKLSAgICAgICAgKm91dFN1cmZhY2UgPSAmc3VyZmFjZTsKLSAgICAgICAgKm91dFNpemUgPSBzaXplOwotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0KLSAgICBzcDxFR0xUZXh0dXJlT2JqZWN0PiB0ZXggPSBnZXRBbmRCaW5kQWN0aXZlVGV4dHVyZU9iamVjdChjKTsKLSAgICBzdGF0dXNfdCBlcnIgPSB0ZXgtPnJlYWxsb2NhdGUobGV2ZWwsCi0gICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCBzdHJpZGUsIGZvcm1hdElkeCwgY29tcHJlc3NlZEZvcm1hdCwgYnByKTsKLSAgICBpZiAoZXJyICE9IE5PX0VSUk9SKQotICAgICAgICByZXR1cm4gR0xfT1VUX09GX01FTU9SWTsKLQotICAgIHRleC0+aW50ZXJuYWxmb3JtYXQgPSBmb3JtYXQ7Ci0gICAgKm91dFN1cmZhY2UgPSAmdGV4LT5zdXJmYWNlOwotICAgICpvdXRTaXplID0gc2l6ZTsKLSAgICByZXR1cm4gMDsKLX0KLQotc3RhdGljIHZvaWQgZGVjb2RlUGFsZXR0ZTQoY29uc3QgR0x2b2lkICpkYXRhLCBpbnQgbGV2ZWwsIGludCB3aWR0aCwgaW50IGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQgKnN1cmZhY2UsIGludCBzdHJpZGUsIGludCBmb3JtYXQpCi0KLXsKLSAgICBpbnQgaW5kZXhCaXRzID0gODsKLSAgICBpbnQgZW50cnlTaXplID0gMDsKLSAgICBzd2l0Y2ggKGZvcm1hdCkgewotICAgIGNhc2UgR0xfUEFMRVRURTRfUkdCOF9PRVM6Ci0gICAgICAgIGluZGV4Qml0cyA9IDQ7Ci0gICAgICAgIC8qIEZBTExUSFJPVUdIICovCi0gICAgY2FzZSBHTF9QQUxFVFRFOF9SR0I4X09FUzoKLSAgICAgICAgZW50cnlTaXplID0gMzsKLSAgICAgICAgYnJlYWs7Ci0KLSAgICBjYXNlIEdMX1BBTEVUVEU0X1JHQkE4X09FUzoKLSAgICAgICAgaW5kZXhCaXRzID0gNDsKLSAgICAgICAgLyogRkFMTFRIUk9VR0ggKi8KLSAgICBjYXNlIEdMX1BBTEVUVEU4X1JHQkE4X09FUzoKLSAgICAgICAgZW50cnlTaXplID0gNDsKLSAgICAgICAgYnJlYWs7Ci0KLSAgICBjYXNlIEdMX1BBTEVUVEU0X1I1X0c2X0I1X09FUzoKLSAgICBjYXNlIEdMX1BBTEVUVEU0X1JHQkE0X09FUzoKLSAgICBjYXNlIEdMX1BBTEVUVEU0X1JHQjVfQTFfT0VTOgotICAgICAgICBpbmRleEJpdHMgPSA0OwotICAgICAgICAvKiBGQUxMVEhST1VHSCAqLwotICAgIGNhc2UgR0xfUEFMRVRURThfUjVfRzZfQjVfT0VTOgotICAgIGNhc2UgR0xfUEFMRVRURThfUkdCQTRfT0VTOgotICAgIGNhc2UgR0xfUEFMRVRURThfUkdCNV9BMV9PRVM6Ci0gICAgICAgIGVudHJ5U2l6ZSA9IDI7Ci0gICAgICAgIGJyZWFrOwotICAgIH0KLQotICAgIGNvbnN0IGludCBwYWxldHRlU2l6ZSA9ICgxIDw8IGluZGV4Qml0cykgKiBlbnRyeVNpemU7Ci0gICAgdWludDhfdCBjb25zdCogcGl4ZWxzID0gKHVpbnQ4X3QgKilkYXRhICsgcGFsZXR0ZVNpemU7Ci0gICAgZm9yIChpbnQgaT0wIDsgaTxsZXZlbCA7IGkrKykgewotICAgICAgICBpbnQgdyA9ICh3aWR0aCAgPj4gaSkgPyA6IDE7Ci0gICAgICAgIGludCBoID0gKGhlaWdodCA+PiBpKSA/IDogMTsKLSAgICAgICAgcGl4ZWxzICs9IGggKiAoKHcgKiBpbmRleEJpdHMpIC8gOCk7Ci0gICAgfQotICAgIHdpZHRoICA9ICh3aWR0aCAgPj4gbGV2ZWwpID8gOiAxOwotICAgIGhlaWdodCA9IChoZWlnaHQgPj4gbGV2ZWwpID8gOiAxOwotCi0gICAgaWYgKGVudHJ5U2l6ZSA9PSAyKSB7Ci0gICAgICAgIHVpbnQ4X3QgY29uc3QqIGNvbnN0IHBhbGV0dGUgPSAodWludDhfdCopZGF0YTsKLSAgICAgICAgZm9yIChpbnQgeT0wIDsgeTxoZWlnaHQgOyB5KyspIHsKLSAgICAgICAgICAgIHVpbnQ4X3QqIHAgPSAodWludDhfdCopc3VyZmFjZSArIHkqc3RyaWRlKjI7Ci0gICAgICAgICAgICBpZiAoaW5kZXhCaXRzID09IDgpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCB4PTAgOyB4PHdpZHRoIDsgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCBpbmRleCA9IDIgKiAoKnBpeGVscysrKTsKLSAgICAgICAgICAgICAgICAgICAgKnArKyA9IHBhbGV0dGVbaW5kZXggKyAwXTsKLSAgICAgICAgICAgICAgICAgICAgKnArKyA9IHBhbGV0dGVbaW5kZXggKyAxXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGZvciAoaW50IHg9MCA7IHg8d2lkdGggOyB4Kz0yKSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCB2ID0gKnBpeGVscysrOwotICAgICAgICAgICAgICAgICAgICBpbnQgaW5kZXggPSAyICogKHYgPj4gNCk7Ci0gICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMF07Ci0gICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMV07Ci0gICAgICAgICAgICAgICAgICAgIGlmICh4KzEgPCB3aWR0aCkgewotICAgICAgICAgICAgICAgICAgICAgICAgaW5kZXggPSAyICogKHYgJiAweEYpOwotICAgICAgICAgICAgICAgICAgICAgICAgKnArKyA9IHBhbGV0dGVbaW5kZXggKyAwXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMV07Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgaWYgKGVudHJ5U2l6ZSA9PSAzKSB7Ci0gICAgICAgIHVpbnQ4X3QgY29uc3QqIGNvbnN0IHBhbGV0dGUgPSAodWludDhfdCopZGF0YTsKLSAgICAgICAgZm9yIChpbnQgeT0wIDsgeTxoZWlnaHQgOyB5KyspIHsKLSAgICAgICAgICAgIHVpbnQ4X3QqIHAgPSAodWludDhfdCopc3VyZmFjZSArIHkqc3RyaWRlKjM7Ci0gICAgICAgICAgICBpZiAoaW5kZXhCaXRzID09IDgpIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCB4PTAgOyB4PHdpZHRoIDsgeCsrKSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCBpbmRleCA9IDMgKiAoKnBpeGVscysrKTsKLSAgICAgICAgICAgICAgICAgICAgKnArKyA9IHBhbGV0dGVbaW5kZXggKyAwXTsKLSAgICAgICAgICAgICAgICAgICAgKnArKyA9IHBhbGV0dGVbaW5kZXggKyAxXTsKLSAgICAgICAgICAgICAgICAgICAgKnArKyA9IHBhbGV0dGVbaW5kZXggKyAyXTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIGZvciAoaW50IHg9MCA7IHg8d2lkdGggOyB4Kz0yKSB7Ci0gICAgICAgICAgICAgICAgICAgIGludCB2ID0gKnBpeGVscysrOwotICAgICAgICAgICAgICAgICAgICBpbnQgaW5kZXggPSAzICogKHYgPj4gNCk7Ci0gICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMF07Ci0gICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMV07Ci0gICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMl07Ci0gICAgICAgICAgICAgICAgICAgIGlmICh4KzEgPCB3aWR0aCkgewotICAgICAgICAgICAgICAgICAgICAgICAgaW5kZXggPSAzICogKHYgJiAweEYpOwotICAgICAgICAgICAgICAgICAgICAgICAgKnArKyA9IHBhbGV0dGVbaW5kZXggKyAwXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMV07Ci0gICAgICAgICAgICAgICAgICAgICAgICAqcCsrID0gcGFsZXR0ZVtpbmRleCArIDJdOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfSBlbHNlIGlmIChlbnRyeVNpemUgPT0gNCkgewotICAgICAgICB1aW50OF90IGNvbnN0KiBjb25zdCBwYWxldHRlID0gKHVpbnQ4X3QqKWRhdGE7Ci0gICAgICAgIGZvciAoaW50IHk9MCA7IHk8aGVpZ2h0IDsgeSsrKSB7Ci0gICAgICAgICAgICB1aW50OF90KiBwID0gKHVpbnQ4X3QqKXN1cmZhY2UgKyB5KnN0cmlkZSo0OwotICAgICAgICAgICAgaWYgKGluZGV4Qml0cyA9PSA4KSB7Ci0gICAgICAgICAgICAgICAgZm9yIChpbnQgeD0wIDsgeDx3aWR0aCA7IHgrKykgewotICAgICAgICAgICAgICAgICAgICBpbnQgaW5kZXggPSA0ICogKCpwaXhlbHMrKyk7Ci0gICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMF07Ci0gICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMV07Ci0gICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMl07Ci0gICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgM107Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBmb3IgKGludCB4PTAgOyB4PHdpZHRoIDsgeCs9MikgewotICAgICAgICAgICAgICAgICAgICBpbnQgdiA9ICpwaXhlbHMrKzsKLSAgICAgICAgICAgICAgICAgICAgaW50IGluZGV4ID0gNCAqICh2ID4+IDQpOwotICAgICAgICAgICAgICAgICAgICAqcCsrID0gcGFsZXR0ZVtpbmRleCArIDBdOwotICAgICAgICAgICAgICAgICAgICAqcCsrID0gcGFsZXR0ZVtpbmRleCArIDFdOwotICAgICAgICAgICAgICAgICAgICAqcCsrID0gcGFsZXR0ZVtpbmRleCArIDJdOwotICAgICAgICAgICAgICAgICAgICAqcCsrID0gcGFsZXR0ZVtpbmRleCArIDNdOwotICAgICAgICAgICAgICAgICAgICBpZiAoeCsxIDwgd2lkdGgpIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIGluZGV4ID0gNCAqICh2ICYgMHhGKTsKLSAgICAgICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgMF07Ci0gICAgICAgICAgICAgICAgICAgICAgICAqcCsrID0gcGFsZXR0ZVtpbmRleCArIDFdOwotICAgICAgICAgICAgICAgICAgICAgICAgKnArKyA9IHBhbGV0dGVbaW5kZXggKyAyXTsKLSAgICAgICAgICAgICAgICAgICAgICAgICpwKysgPSBwYWxldHRlW2luZGV4ICsgM107Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci19Ci0KLQotCi1zdGF0aWMgX19hdHRyaWJ1dGVfXygobm9pbmxpbmUpKQotdm9pZCBzZXRfZGVwdGhfYW5kX2ZvZyhvZ2xlc19jb250ZXh0X3QqIGMsIEdMaW50IHopCi17Ci0gICAgY29uc3QgdWludDMyX3QgZW5hYmxlcyA9IGMtPnJhc3Rlcml6ZXIuc3RhdGUuZW5hYmxlczsKLSAgICAvLyB3ZSBuZWVkIHRvIGNvbXB1dGUgWncKLSAgICBpbnQzMl90IGl0ZXJhdG9yc1szXTsKLSAgICBpdGVyYXRvcnNbMV0gPSBpdGVyYXRvcnNbMl0gPSAwOwotICAgIEdHTGZpeGVkIFp3OwotICAgIEdHTGZpeGVkIG4gPSBnZ2xGbG9hdFRvRml4ZWQoYy0+dHJhbnNmb3Jtcy52cHQuek5lYXIpOwotICAgIEdHTGZpeGVkIGYgPSBnZ2xGbG9hdFRvRml4ZWQoYy0+dHJhbnNmb3Jtcy52cHQuekZhcik7Ci0gICAgaWYgKHo8PTApICAgICAgIFp3ID0gbjsKLSAgICBlbHNlIGlmICh6Pj0xKSAgWncgPSBmOwotICAgIGVsc2UgICAgICAgICAgICBadyA9IGdnbE11bEFkZHgoeiwgKGYtbiksIG4pOwotICAgIGlmIChlbmFibGVzICYgR0dMX0VOQUJMRV9GT0cpIHsKLSAgICAgICAgLy8gc2V0IHVwIGZvZyBpZiBuZWVkZWQuLi4KLSAgICAgICAgaXRlcmF0b3JzWzBdID0gYy0+Zm9nLmZvZyhjLCBadyk7Ci0gICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuZm9nR3JhZDN4dihjLCBpdGVyYXRvcnMpOwotICAgIH0KLSAgICBpZiAoZW5hYmxlcyAmIEdHTF9FTkFCTEVfREVQVEhfVEVTVCkgewotICAgICAgICAvLyBzZXQgdXAgei10ZXN0IGlmIG5lZWRlZC4uLgotICAgICAgICBpbnQzMl90IHogPSAoWncgJiB+KFp3Pj4zMSkpOwotICAgICAgICBpZiAoeiA+PSAweDEwMDAwKQotICAgICAgICAgICAgeiA9IDB4RkZGRjsKLSAgICAgICAgaXRlcmF0b3JzWzBdID0gKHogPDwgMTYpIHwgejsKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy56R3JhZDN4dihjLCBpdGVyYXRvcnMpOwotICAgIH0KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jcHJhZ21hIG1hcmsgR2VuZXJhdGUgbWltYXBzCi0jZW5kaWYKLQotZXh0ZXJuIHN0YXR1c190IGJ1aWxkQVB5cmFtaWQob2dsZXNfY29udGV4dF90KiBjLCBFR0xUZXh0dXJlT2JqZWN0KiB0ZXgpOwotCi12b2lkIGdlbmVyYXRlTWlwbWFwKG9nbGVzX2NvbnRleHRfdCogYywgR0xpbnQgbGV2ZWwpCi17Ci0gICAgaWYgKGxldmVsID09IDApIHsKLSAgICAgICAgY29uc3QgaW50IGFjdGl2ZSA9IGMtPnRleHR1cmVzLmFjdGl2ZTsKLSAgICAgICAgRUdMVGV4dHVyZU9iamVjdCogdGV4ID0gYy0+dGV4dHVyZXMudG11W2FjdGl2ZV0udGV4dHVyZTsKLSAgICAgICAgaWYgKHRleC0+Z2VuZXJhdGVfbWlwbWFwKSB7Ci0gICAgICAgICAgICBpZiAoYnVpbGRBUHlyYW1pZChjLCB0ZXgpICE9IE5PX0VSUk9SKSB7Ci0gICAgICAgICAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfT1VUX09GX01FTU9SWSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotfQotCi0KLXN0YXRpYyB2b2lkIHRleFBhcmFtZXRlcngoCi0gICAgICAgIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSwgb2dsZXNfY29udGV4dF90KiBjKQotewotICAgIGlmICh0YXJnZXQgIT0gR0xfVEVYVFVSRV8yRCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIAotICAgIEVHTFRleHR1cmVPYmplY3QqIHRleHR1cmVPYmplY3QgPSBjLT50ZXh0dXJlcy50bXVbYy0+dGV4dHVyZXMuYWN0aXZlXS50ZXh0dXJlOyAgICAKLSAgICBzd2l0Y2ggKHBuYW1lKSB7Ci0gICAgY2FzZSBHTF9URVhUVVJFX1dSQVBfUzoKLSAgICAgICAgaWYgKChwYXJhbSA9PSBHTF9SRVBFQVQpIHx8Ci0gICAgICAgICAgICAocGFyYW0gPT0gR0xfQ0xBTVBfVE9fRURHRSkpIHsKLSAgICAgICAgICAgIHRleHR1cmVPYmplY3QtPndyYXBzID0gcGFyYW07Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBnb3RvIGludmFsaWRfZW51bTsKLSAgICAgICAgfQotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1RFWFRVUkVfV1JBUF9UOgotICAgICAgICBpZiAoKHBhcmFtID09IEdMX1JFUEVBVCkgfHwKLSAgICAgICAgICAgIChwYXJhbSA9PSBHTF9DTEFNUF9UT19FREdFKSkgewotICAgICAgICAgICAgdGV4dHVyZU9iamVjdC0+d3JhcHQgPSBwYXJhbTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGdvdG8gaW52YWxpZF9lbnVtOwotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfVEVYVFVSRV9NSU5fRklMVEVSOgotICAgICAgICBpZiAoKHBhcmFtID09IEdMX05FQVJFU1QpIHx8Ci0gICAgICAgICAgICAocGFyYW0gPT0gR0xfTElORUFSKSB8fAotICAgICAgICAgICAgKHBhcmFtID09IEdMX05FQVJFU1RfTUlQTUFQX05FQVJFU1QpIHx8Ci0gICAgICAgICAgICAocGFyYW0gPT0gR0xfTElORUFSX01JUE1BUF9ORUFSRVNUKSB8fAotICAgICAgICAgICAgKHBhcmFtID09IEdMX05FQVJFU1RfTUlQTUFQX0xJTkVBUikgfHwKLSAgICAgICAgICAgIChwYXJhbSA9PSBHTF9MSU5FQVJfTUlQTUFQX0xJTkVBUikpIHsKLSAgICAgICAgICAgIHRleHR1cmVPYmplY3QtPm1pbl9maWx0ZXIgPSBwYXJhbTsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIGdvdG8gaW52YWxpZF9lbnVtOwotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfVEVYVFVSRV9NQUdfRklMVEVSOgotICAgICAgICBpZiAoKHBhcmFtID09IEdMX05FQVJFU1QpIHx8Ci0gICAgICAgICAgICAocGFyYW0gPT0gR0xfTElORUFSKSkgewotICAgICAgICAgICAgdGV4dHVyZU9iamVjdC0+bWFnX2ZpbHRlciA9IHBhcmFtOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZ290byBpbnZhbGlkX2VudW07Ci0gICAgICAgIH0KLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9HRU5FUkFURV9NSVBNQVA6Ci0gICAgICAgIHRleHR1cmVPYmplY3QtPmdlbmVyYXRlX21pcG1hcCA9IHBhcmFtOwotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotaW52YWxpZF9lbnVtOgotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGludmFsaWRhdGVfdGV4dHVyZShjLCBjLT50ZXh0dXJlcy5hY3RpdmUpOwotfQotCi0KLXN0YXRpYyB2b2lkIGRyYXdUZXh4T0VTKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHosIEdMZml4ZWQgdywgR0xmaXhlZCBoLAotICAgICAgICBvZ2xlc19jb250ZXh0X3QqIGMpCi17Ci0gICAgLy8gcXVpY2tseSByZWplY3QgZW1wdHkgcmVjdHMKLSAgICBpZiAoKHd8aCkgPD0gMCkKLSAgICAgICAgcmV0dXJuOyAgICAgICAgICAgICAgICAKLQotICAgIGNvbnN0IEdHTFN1cmZhY2UmIGNiU3VyZmFjZSA9IGMtPnJhc3Rlcml6ZXIuc3RhdGUuYnVmZmVycy5jb2xvci5zOwotICAgIHkgPSBnZ2xJbnRUb0ZpeGVkKGNiU3VyZmFjZS5oZWlnaHQpIC0gKHkgKyBoKTsKLSAgICB3ID4+PSBGSVhFRF9CSVRTOwotICAgIGggPj49IEZJWEVEX0JJVFM7Ci0KLSAgICAvLyBzZXQgdXAgYWxsIHRleHR1cmUgdW5pdHMKLSAgICBmb3IgKGludCBpPTAgOyBpPEdHTF9URVhUVVJFX1VOSVRfQ09VTlQgOyBpKyspIHsKLSAgICAgICAgaWYgKCFjLT5yYXN0ZXJpemVyLnN0YXRlLnRleHR1cmVbaV0uZW5hYmxlKQotICAgICAgICAgICAgY29udGludWU7Ci0KLSAgICAgICAgaW50MzJfdCB0ZXhjb29yZHNbOF07Ci0gICAgICAgIHRleHR1cmVfdW5pdF90JiB1KGMtPnRleHR1cmVzLnRtdVtpXSk7Ci0KLSAgICAgICAgLy8gdmFsaWRhdGUgdGhpcyB0bXUgKGJpbmQsIHdyYXAsIGZpbHRlcikKLSAgICAgICAgdmFsaWRhdGVfdG11KGMsIGkpOwotICAgICAgICAvLyB3ZSBDTEFNUCBoZXJlLCB3aGljaCB3b3JrcyB3aXRoIHByZW11bHRpcGxpZWQgKHMsdCkKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy50ZXhQYXJhbWV0ZXJpKGMsCi0gICAgICAgICAgICAgICAgR0dMX1RFWFRVUkVfMkQsIEdHTF9URVhUVVJFX1dSQVBfUywgR0dMX0NMQU1QKTsKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy50ZXhQYXJhbWV0ZXJpKGMsCi0gICAgICAgICAgICAgICAgR0dMX1RFWFRVUkVfMkQsIEdHTF9URVhUVVJFX1dSQVBfVCwgR0dMX0NMQU1QKTsKLSAgICAgICAgdS5kaXJ0eSA9IDB4RkY7IC8vIFhYWDogc2hvdWxkIGJlIG1vcmUgc3VidGxlCi0KLSAgICAgICAgRUdMVGV4dHVyZU9iamVjdCogdGV4dHVyZU9iamVjdCA9IHUudGV4dHVyZTsgIAotICAgICAgICBjb25zdCBHTGludCBVY3IgPSB0ZXh0dXJlT2JqZWN0LT5jcm9wX3JlY3RbMF0gPDwgMTY7Ci0gICAgICAgIGNvbnN0IEdMaW50IFZjciA9IHRleHR1cmVPYmplY3QtPmNyb3BfcmVjdFsxXSA8PCAxNjsKLSAgICAgICAgY29uc3QgR0xpbnQgV2NyID0gdGV4dHVyZU9iamVjdC0+Y3JvcF9yZWN0WzJdIDw8IDE2OwotICAgICAgICBjb25zdCBHTGludCBIY3IgPSB0ZXh0dXJlT2JqZWN0LT5jcm9wX3JlY3RbM10gPDwgMTY7Ci0KLSAgICAgICAgLy8gY29tcHV0ZXMgdGV4dHVyZSBjb29yZGluYXRlcyAocHJlLW11bHRpcGxpZWQpCi0gICAgICAgIGludDMyX3QgZHNkeCA9IFdjciAvIHc7ICAgLy8gZHNkeCA9ICAoKFdjci93KS9XdCkqV3QKLSAgICAgICAgaW50MzJfdCBkdGR5ID0tSGNyIC8gaDsgICAvLyBkdGR5ID0gLSgoSGNyL2gpL0h0KSpIdAotICAgICAgICBpbnQzMl90IHMwICAgPSBVY3IgICAgICAgLSBnZ2xNdWx4KGRzZHgsIHgpOyAvLyBzMCA9IFVjciAtIHggKiBkc2R4Ci0gICAgICAgIGludDMyX3QgdDAgICA9IChWY3IrSGNyKSAtIGdnbE11bHgoZHRkeSwgeSk7IC8vIHQwID0gKFZjcitIY3IpIC0geSpkdGR5Ci0gICAgICAgIHRleGNvb3Jkc1swXSA9IHMwOwotICAgICAgICB0ZXhjb29yZHNbMV0gPSBkc2R4OwotICAgICAgICB0ZXhjb29yZHNbMl0gPSAwOwotICAgICAgICB0ZXhjb29yZHNbM10gPSB0MDsKLSAgICAgICAgdGV4Y29vcmRzWzRdID0gMDsKLSAgICAgICAgdGV4Y29vcmRzWzVdID0gZHRkeTsKLSAgICAgICAgdGV4Y29vcmRzWzZdID0gMDsKLSAgICAgICAgdGV4Y29vcmRzWzddID0gMDsKLSAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy50ZXhDb29yZEdyYWRTY2FsZTh4dihjLCBpLCB0ZXhjb29yZHMpOwotICAgIH0KLQotICAgIGNvbnN0IHVpbnQzMl90IGVuYWJsZXMgPSBjLT5yYXN0ZXJpemVyLnN0YXRlLmVuYWJsZXM7Ci0gICAgaWYgKGdnbF91bmxpa2VseShlbmFibGVzICYgKEdHTF9FTkFCTEVfREVQVEhfVEVTVHxHR0xfRU5BQkxFX0ZPRykpKQotICAgICAgICBzZXRfZGVwdGhfYW5kX2ZvZyhjLCB6KTsKLQotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuYWN0aXZlVGV4dHVyZShjLCBjLT50ZXh0dXJlcy5hY3RpdmUpOwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuY29sb3I0eHYoYywgYy0+Y3VycmVudENvbG9yQ2xhbXBlZC52KTsKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLmRpc2FibGUoYywgR0dMX1dfTEVSUCk7Ci0gICAgYy0+cmFzdGVyaXplci5wcm9jcy5kaXNhYmxlKGMsIEdHTF9BQSk7Ci0gICAgYy0+cmFzdGVyaXplci5wcm9jcy5zaGFkZU1vZGVsKGMsIEdMX0ZMQVQpOwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MucmVjdGkoYywgCi0gICAgICAgICAgICBnZ2xGaXhlZFRvSW50Um91bmQoeCksCi0gICAgICAgICAgICBnZ2xGaXhlZFRvSW50Um91bmQoeSksCi0gICAgICAgICAgICBnZ2xGaXhlZFRvSW50Um91bmQoeCkrdywKLSAgICAgICAgICAgIGdnbEZpeGVkVG9JbnRSb3VuZCh5KStoKTsKLX0KLQotc3RhdGljIHZvaWQgZHJhd1RleGlPRVMoR0xpbnQgeCwgR0xpbnQgeSwgR0xpbnQgeiwgR0xpbnQgdywgR0xpbnQgaCwgb2dsZXNfY29udGV4dF90KiBjKQotewotICAgIC8vIEFsbCBjb29yZGluYXRlcyBhcmUgaW50ZWdlciwgc28gaWYgd2UgaGF2ZSBvbmx5IG9uZQotICAgIC8vIHRleHR1cmUgdW5pdCBhY3RpdmUgYW5kIG5vIHNjYWxpbmcgaXMgcmVxdWlyZWQKLSAgICAvLyBUSEVOLCB3ZSBjYW4gdXNlIG91ciBzcGVjaWFsIDE6MSBtYXBwaW5nCi0gICAgLy8gd2hpY2ggaXMgYSBsb3QgZmFzdGVyLgotCi0gICAgaWYgKGdnbF9saWtlbHkoYy0+cmFzdGVyaXplci5zdGF0ZS5lbmFibGVkX3RtdSA9PSAxKSkgewotICAgICAgICBjb25zdCBpbnQgdG11ID0gMDsKLSAgICAgICAgdGV4dHVyZV91bml0X3QmIHUoYy0+dGV4dHVyZXMudG11W3RtdV0pOwotICAgICAgICBFR0xUZXh0dXJlT2JqZWN0KiB0ZXh0dXJlT2JqZWN0ID0gdS50ZXh0dXJlOyAgCi0gICAgICAgIGNvbnN0IEdMaW50IFdjciA9IHRleHR1cmVPYmplY3QtPmNyb3BfcmVjdFsyXTsKLSAgICAgICAgY29uc3QgR0xpbnQgSGNyID0gdGV4dHVyZU9iamVjdC0+Y3JvcF9yZWN0WzNdOwotCi0gICAgICAgIGlmICgodyA9PSBXY3IpICYmIChoID09IC1IY3IpKSB7Ci0gICAgICAgICAgICBpZiAoKHd8aCkgPD0gMCkgcmV0dXJuOyAvLyBxdWlja2x5IHJlamVjdCBlbXB0eSByZWN0cwotCi0gICAgICAgICAgICBpZiAodS5kaXJ0eSkgewotICAgICAgICAgICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuYWN0aXZlVGV4dHVyZShjLCB0bXUpOwotICAgICAgICAgICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuYmluZFRleHR1cmUoYywgJih1LnRleHR1cmUtPnN1cmZhY2UpKTsKLSAgICAgICAgICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleFBhcmFtZXRlcmkoYywgR0dMX1RFWFRVUkVfMkQsCi0gICAgICAgICAgICAgICAgICAgICAgICBHR0xfVEVYVFVSRV9NSU5fRklMVEVSLCB1LnRleHR1cmUtPm1pbl9maWx0ZXIpOwotICAgICAgICAgICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MudGV4UGFyYW1ldGVyaShjLCBHR0xfVEVYVFVSRV8yRCwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdHTF9URVhUVVJFX01BR19GSUxURVIsIHUudGV4dHVyZS0+bWFnX2ZpbHRlcik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleEdlbmkoYywgR0dMX1MsCi0gICAgICAgICAgICAgICAgICAgIEdHTF9URVhUVVJFX0dFTl9NT0RFLCBHR0xfT05FX1RPX09ORSk7Ci0gICAgICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleEdlbmkoYywgR0dMX1QsCi0gICAgICAgICAgICAgICAgICAgIEdHTF9URVhUVVJFX0dFTl9NT0RFLCBHR0xfT05FX1RPX09ORSk7Ci0gICAgICAgICAgICB1LmRpcnR5ID0gMHhGRjsgLy8gWFhYOiBzaG91bGQgYmUgbW9yZSBzdWJ0bGUKLSAgICAgICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuYWN0aXZlVGV4dHVyZShjLCBjLT50ZXh0dXJlcy5hY3RpdmUpOwotICAgICAgICAgICAgCi0gICAgICAgICAgICBjb25zdCBHR0xTdXJmYWNlJiBjYlN1cmZhY2UgPSBjLT5yYXN0ZXJpemVyLnN0YXRlLmJ1ZmZlcnMuY29sb3IuczsKLSAgICAgICAgICAgIHkgPSBjYlN1cmZhY2UuaGVpZ2h0IC0gKHkgKyBoKTsKLSAgICAgICAgICAgIGNvbnN0IEdMaW50IFVjciA9IHRleHR1cmVPYmplY3QtPmNyb3BfcmVjdFswXTsKLSAgICAgICAgICAgIGNvbnN0IEdMaW50IFZjciA9IHRleHR1cmVPYmplY3QtPmNyb3BfcmVjdFsxXTsKLSAgICAgICAgICAgIGNvbnN0IEdMaW50IHMwICA9IFVjciAtIHg7Ci0gICAgICAgICAgICBjb25zdCBHTGludCB0MCAgPSAoVmNyICsgSGNyKSAtIHk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIGNvbnN0IEdMdWludCB0dyA9IHRleHR1cmVPYmplY3QtPnN1cmZhY2Uud2lkdGg7Ci0gICAgICAgICAgICBjb25zdCBHTHVpbnQgdGggPSB0ZXh0dXJlT2JqZWN0LT5zdXJmYWNlLmhlaWdodDsKLSAgICAgICAgICAgIGlmICgodWludDMyX3QoczAreCt3KSA+IHR3KSB8fCAodWludDMyX3QodDAreStoKSA+IHRoKSkgewotICAgICAgICAgICAgICAgIC8vIFRoZSBHTCBzcGVjIGlzIHVuY2xlYXIgYWJvdXQgd2hhdCBzaG91bGQgaGFwcGVuCi0gICAgICAgICAgICAgICAgLy8gaW4gdGhpcyBjYXNlLCBzbyB3ZSBqdXN0IHVzZSB0aGUgc2xvdyBjYXNlLCB3aGljaAotICAgICAgICAgICAgICAgIC8vIGF0IGxlYXN0IHdvbid0IGNyYXNoCi0gICAgICAgICAgICAgICAgZ290byBzbG93X2Nhc2U7Ci0gICAgICAgICAgICB9IAotCi0gICAgICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleENvb3JkMmkoYywgczAsIHQwKTsKLSAgICAgICAgICAgIGNvbnN0IHVpbnQzMl90IGVuYWJsZXMgPSBjLT5yYXN0ZXJpemVyLnN0YXRlLmVuYWJsZXM7Ci0gICAgICAgICAgICBpZiAoZ2dsX3VubGlrZWx5KGVuYWJsZXMgJiAoR0dMX0VOQUJMRV9ERVBUSF9URVNUfEdHTF9FTkFCTEVfRk9HKSkpCi0gICAgICAgICAgICAgICAgc2V0X2RlcHRoX2FuZF9mb2coYywgeik7Ci0KLSAgICAgICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuY29sb3I0eHYoYywgYy0+Y3VycmVudENvbG9yQ2xhbXBlZC52KTsKLSAgICAgICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuZGlzYWJsZShjLCBHR0xfV19MRVJQKTsKLSAgICAgICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuZGlzYWJsZShjLCBHR0xfQUEpOwotICAgICAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy5zaGFkZU1vZGVsKGMsIEdMX0ZMQVQpOwotICAgICAgICAgICAgYy0+cmFzdGVyaXplci5wcm9jcy5yZWN0aShjLCB4LCB5LCB4K3csIHkraCk7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICB9Ci0KLXNsb3dfY2FzZToKLSAgICBkcmF3VGV4eE9FUygKLSAgICAgICAgICAgIGdnbEludFRvRml4ZWQoeCksIGdnbEludFRvRml4ZWQoeSksIGdnbEludFRvRml4ZWQoeiksCi0gICAgICAgICAgICBnZ2xJbnRUb0ZpeGVkKHcpLCBnZ2xJbnRUb0ZpeGVkKGgpLAotICAgICAgICAgICAgYyk7Ci19Ci0KLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdXNpbmcgbmFtZXNwYWNlIGFuZHJvaWQ7Ci0KLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jcHJhZ21hIG1hcmsgVGV4dHVyZSBBUEkKLSNlbmRpZgotCi12b2lkIGdsQWN0aXZlVGV4dHVyZShHTGVudW0gdGV4dHVyZSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICh1aW50MzJfdCh0ZXh0dXJlLUdMX1RFWFRVUkUwKSA+IHVpbnQzMl90KEdHTF9URVhUVVJFX1VOSVRfQ09VTlQpKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgYy0+dGV4dHVyZXMuYWN0aXZlID0gdGV4dHVyZSAtIEdMX1RFWFRVUkUwOwotICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MuYWN0aXZlVGV4dHVyZShjLCBjLT50ZXh0dXJlcy5hY3RpdmUpOwotfQotCi12b2lkIGdsQmluZFRleHR1cmUoR0xlbnVtIHRhcmdldCwgR0x1aW50IHRleHR1cmUpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBpZiAodGFyZ2V0ICE9IEdMX1RFWFRVUkVfMkQpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIC8vIEJpbmQgb3IgY3JlYXRlIGEgdGV4dHVyZQotICAgIHNwPEVHTFRleHR1cmVPYmplY3Q+IHRleDsgICAgCi0gICAgaWYgKHRleHR1cmUgPT0gMCkgewotICAgICAgICAvLyAwIGlzIG91ciBsb2NhbCB0ZXh0dXJlIG9iamVjdAotICAgICAgICB0ZXggPSBjLT50ZXh0dXJlcy5kZWZhdWx0VGV4dHVyZTsKLSAgICB9IGVsc2UgewotICAgICAgICB0ZXggPSBjLT5zdXJmYWNlTWFuYWdlci0+dGV4dHVyZSh0ZXh0dXJlKTsKLSAgICAgICAgaWYgKGdnbF91bmxpa2VseSh0ZXggPT0gMCkpIHsKLSAgICAgICAgICAgIHRleCA9IGMtPnN1cmZhY2VNYW5hZ2VyLT5jcmVhdGVUZXh0dXJlKHRleHR1cmUpOwotICAgICAgICAgICAgaWYgKHRleCA9PSAwKSB7Ci0gICAgICAgICAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfT1VUX09GX01FTU9SWSk7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIGJpbmRUZXh0dXJlVG11KGMsIGMtPnRleHR1cmVzLmFjdGl2ZSwgdGV4dHVyZSwgdGV4KTsKLX0KLQotdm9pZCBnbEdlblRleHR1cmVzKEdMc2l6ZWkgbiwgR0x1aW50ICp0ZXh0dXJlcykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmIChuPDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgLy8gZ2VuZXJhdGUgdW5pcXVlIChzaGFyZWQpIHRleHR1cmUgbmFtZXMKLSAgICBjLT5zdXJmYWNlTWFuYWdlci0+Z2V0VG9rZW4obiwgdGV4dHVyZXMpOwotfQotCi12b2lkIGdsRGVsZXRlVGV4dHVyZXMoR0xzaXplaSBuLCBjb25zdCBHTHVpbnQgKnRleHR1cmVzKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKG48MCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX1ZBTFVFKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIC8vIElmIGRlbGV0aW5nIGEgYm91bmQgdGV4dHVyZSwgYmluZCB0aGlzIHVuaXQgdG8gMAotICAgIGZvciAoaW50IHQ9MCA7IHQ8R0dMX1RFWFRVUkVfVU5JVF9DT1VOVCA7IHQrKykgewotICAgICAgICBpZiAoYy0+dGV4dHVyZXMudG11W3RdLm5hbWUgPT0gMCkKLSAgICAgICAgICAgIGNvbnRpbnVlOwotICAgICAgICBmb3IgKGludCBpPTAgOyBpPG4gOyBpKyspIHsKLSAgICAgICAgICAgIGlmICh0ZXh0dXJlc1tpXSAmJiAodGV4dHVyZXNbaV0gPT0gYy0+dGV4dHVyZXMudG11W3RdLm5hbWUpKSB7Ci0gICAgICAgICAgICAgICAgLy8gYmluZCB0aGlzIHRtdSB0byB0ZXh0dXJlIDAKLSAgICAgICAgICAgICAgICBzcDxFR0xUZXh0dXJlT2JqZWN0PiB0ZXgoYy0+dGV4dHVyZXMuZGVmYXVsdFRleHR1cmUpOwotICAgICAgICAgICAgICAgIGJpbmRUZXh0dXJlVG11KGMsIHQsIDAsIHRleCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgYy0+c3VyZmFjZU1hbmFnZXItPmRlbGV0ZVRleHR1cmVzKG4sIHRleHR1cmVzKTsKLSAgICBjLT5zdXJmYWNlTWFuYWdlci0+cmVjeWNsZVRva2VucyhuLCB0ZXh0dXJlcyk7Ci19Ci0KLXZvaWQgZ2xNdWx0aVRleENvb3JkNGYoCi0gICAgICAgIEdMZW51bSB0YXJnZXQsIEdMZmxvYXQgcywgR0xmbG9hdCB0LCBHTGZsb2F0IHIsIEdMZmxvYXQgcSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICh1aW50MzJfdCh0YXJnZXQtR0xfVEVYVFVSRTApID4gdWludDMyX3QoR0dMX1RFWFRVUkVfVU5JVF9DT1VOVCkpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBjb25zdCBpbnQgdG11ID0gdGFyZ2V0LUdMX1RFWFRVUkUwOwotICAgIGMtPmN1cnJlbnQudGV4dHVyZVt0bXVdLlMgPSBnZ2xGbG9hdFRvRml4ZWQocyk7Ci0gICAgYy0+Y3VycmVudC50ZXh0dXJlW3RtdV0uVCA9IGdnbEZsb2F0VG9GaXhlZCh0KTsKLSAgICBjLT5jdXJyZW50LnRleHR1cmVbdG11XS5SID0gZ2dsRmxvYXRUb0ZpeGVkKHIpOwotICAgIGMtPmN1cnJlbnQudGV4dHVyZVt0bXVdLlEgPSBnZ2xGbG9hdFRvRml4ZWQocSk7Ci19Ci0KLXZvaWQgZ2xNdWx0aVRleENvb3JkNHgoCi0gICAgICAgIEdMZW51bSB0YXJnZXQsIEdMZml4ZWQgcywgR0xmaXhlZCB0LCBHTGZpeGVkIHIsIEdMZml4ZWQgcSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICh1aW50MzJfdCh0YXJnZXQtR0xfVEVYVFVSRTApID4gdWludDMyX3QoR0dMX1RFWFRVUkVfVU5JVF9DT1VOVCkpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBjb25zdCBpbnQgdG11ID0gdGFyZ2V0LUdMX1RFWFRVUkUwOwotICAgIGMtPmN1cnJlbnQudGV4dHVyZVt0bXVdLlMgPSBzOwotICAgIGMtPmN1cnJlbnQudGV4dHVyZVt0bXVdLlQgPSB0OwotICAgIGMtPmN1cnJlbnQudGV4dHVyZVt0bXVdLlIgPSByOwotICAgIGMtPmN1cnJlbnQudGV4dHVyZVt0bXVdLlEgPSBxOwotfQotCi12b2lkIGdsUGl4ZWxTdG9yZWkoR0xlbnVtIHBuYW1lLCBHTGludCBwYXJhbSkKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICgocG5hbWUgIT0gR0xfUEFDS19BTElHTk1FTlQpICYmIChwbmFtZSAhPSBHTF9VTlBBQ0tfQUxJR05NRU5UKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfSAgICAKLSAgICBpZiAoKHBhcmFtPD0wIHx8IHBhcmFtPjgpIHx8IChwYXJhbSAmIChwYXJhbS0xKSkpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKHBuYW1lID09IEdMX1BBQ0tfQUxJR05NRU5UKQotICAgICAgICBjLT50ZXh0dXJlcy5wYWNrQWxpZ25tZW50ID0gcGFyYW07Ci0gICAgaWYgKHBuYW1lID09IEdMX1VOUEFDS19BTElHTk1FTlQpCi0gICAgICAgIGMtPnRleHR1cmVzLnVucGFja0FsaWdubWVudCA9IHBhcmFtOwotfQotCi12b2lkIGdsVGV4RW52ZihHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleEVudmkoYywgdGFyZ2V0LCBwbmFtZSwgR0xpbnQocGFyYW0pKTsKLX0KLQotdm9pZCBnbFRleEVudmZ2KAotICAgICAgICBHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmIChwbmFtZSA9PSBHTF9URVhUVVJFX0VOVl9NT0RFKSB7Ci0gICAgICAgIGMtPnJhc3Rlcml6ZXIucHJvY3MudGV4RW52aShjLCB0YXJnZXQsIHBuYW1lLCBHTGludCgqcGFyYW1zKSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKHBuYW1lID09IEdMX1RFWFRVUkVfRU5WX0NPTE9SKSB7Ci0gICAgICAgIEdHTGZpeGVkIGZpeGVkWzRdOwotICAgICAgICBmb3IgKGludCBpPTAgOyBpPDQgOyBpKyspCi0gICAgICAgICAgICBmaXhlZFtpXSA9IGdnbEZsb2F0VG9GaXhlZChwYXJhbXNbaV0pOwotICAgICAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleEVudnh2KGMsIHRhcmdldCwgcG5hbWUsIGZpeGVkKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotfQotCi12b2lkIGdsVGV4RW52eChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT5yYXN0ZXJpemVyLnByb2NzLnRleEVudmkoYywgdGFyZ2V0LCBwbmFtZSwgcGFyYW0pOwotfQotCi12b2lkIGdsVGV4RW52eHYoCi0gICAgICAgIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+cmFzdGVyaXplci5wcm9jcy50ZXhFbnZ4dihjLCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotfQotCi12b2lkIGdsVGV4UGFyYW1ldGVyaXYoCi0gICAgICAgIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xpbnQqIHBhcmFtcykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICh0YXJnZXQgIT0gR0dMX1RFWFRVUkVfMkQpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIEVHTFRleHR1cmVPYmplY3QqIHRleHR1cmVPYmplY3QgPSBjLT50ZXh0dXJlcy50bXVbYy0+dGV4dHVyZXMuYWN0aXZlXS50ZXh0dXJlOwotICAgIHN3aXRjaCAocG5hbWUpIHsKLSAgICBjYXNlIEdMX1RFWFRVUkVfQ1JPUF9SRUNUX09FUzoKLSAgICAgICAgbWVtY3B5KHRleHR1cmVPYmplY3QtPmNyb3BfcmVjdCwgcGFyYW1zLCA0KnNpemVvZihHTGludCkpOwotICAgICAgICBicmVhazsKLSAgICBkZWZhdWx0OgotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotfQotCi12b2lkIGdsVGV4UGFyYW1ldGVyZigKLSAgICAgICAgR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgdGV4UGFyYW1ldGVyeCh0YXJnZXQsIHBuYW1lLCBHTGZpeGVkKHBhcmFtKSwgYyk7Ci19Ci0KLXZvaWQgZ2xUZXhQYXJhbWV0ZXJ4KAotICAgICAgICBHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICB0ZXhQYXJhbWV0ZXJ4KHRhcmdldCwgcG5hbWUsIHBhcmFtLCBjKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotI2lmIDAKLSNwcmFnbWEgbWFyayAtCi0jZW5kaWYKLQotdm9pZCBnbENvbXByZXNzZWRUZXhJbWFnZTJEKAotICAgICAgICBHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xlbnVtIGludGVybmFsZm9ybWF0LAotICAgICAgICBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwgR0xpbnQgYm9yZGVyLAotICAgICAgICBHTHNpemVpIGltYWdlU2l6ZSwgY29uc3QgR0x2b2lkICpkYXRhKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKHRhcmdldCAhPSBHTF9URVhUVVJFXzJEKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKChpbnRlcm5hbGZvcm1hdCA8IEdMX1BBTEVUVEU0X1JHQjhfT0VTIHx8Ci0gICAgICAgICBpbnRlcm5hbGZvcm1hdCA+IEdMX1BBTEVUVEU4X1JHQjVfQTFfT0VTKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmICh3aWR0aDwwIHx8IGhlaWdodDwwIHx8IGJvcmRlciE9MCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX1ZBTFVFKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIC8vICJ1bmNvbXByZXNzIiB0aGUgdGV4dHVyZSBzaW5jZSBwaXhlbGZsaW5nZXIgZG9lc24ndCBzdXBwb3J0Ci0gICAgLy8gYW55IGNvbXByZXNzZWQgdGV4dHVyZSBmb3JtYXQgbmF0aXZlbHkuIAotICAgIEdMZW51bSBmb3JtYXQ7Ci0gICAgR0xlbnVtIHR5cGU7Ci0gICAgc3dpdGNoIChpbnRlcm5hbGZvcm1hdCkgewotICAgIGNhc2UgR0xfUEFMRVRURThfUkdCOF9PRVM6Ci0gICAgY2FzZSBHTF9QQUxFVFRFNF9SR0I4X09FUzoKLSAgICAgICAgZm9ybWF0ICAgICAgPSBHTF9SR0I7Ci0gICAgICAgIHR5cGUgICAgICAgID0gR0xfVU5TSUdORURfQllURTsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9QQUxFVFRFOF9SR0JBOF9PRVM6Ci0gICAgY2FzZSBHTF9QQUxFVFRFNF9SR0JBOF9PRVM6Ci0gICAgICAgIGZvcm1hdCAgICAgID0gR0xfUkdCQTsKLSAgICAgICAgdHlwZSAgICAgICAgPSBHTF9VTlNJR05FRF9CWVRFOwotICAgICAgICBicmVhazsKLSAgICBjYXNlIEdMX1BBTEVUVEU4X1I1X0c2X0I1X09FUzoKLSAgICBjYXNlIEdMX1BBTEVUVEU0X1I1X0c2X0I1X09FUzoKLSAgICAgICAgZm9ybWF0ICAgICAgPSBHTF9SR0I7Ci0gICAgICAgIHR5cGUgICAgICAgID0gR0xfVU5TSUdORURfU0hPUlRfNV82XzU7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0xfUEFMRVRURThfUkdCQTRfT0VTOgotICAgIGNhc2UgR0xfUEFMRVRURTRfUkdCQTRfT0VTOgotICAgICAgICBmb3JtYXQgICAgICA9IEdMX1JHQkE7Ci0gICAgICAgIHR5cGUgICAgICAgID0gR0xfVU5TSUdORURfU0hPUlRfNF80XzRfNDsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHTF9QQUxFVFRFOF9SR0I1X0ExX09FUzoKLSAgICBjYXNlIEdMX1BBTEVUVEU0X1JHQjVfQTFfT0VTOgotICAgICAgICBmb3JtYXQgICAgICA9IEdMX1JHQkE7Ci0gICAgICAgIHR5cGUgICAgICAgID0gR0xfVU5TSUdORURfU0hPUlRfNV81XzVfMTsKLSAgICAgICAgYnJlYWs7Ci0gICAgZGVmYXVsdDoKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIGlmICghZGF0YSB8fCAhd2lkdGggfHwgIWhlaWdodCkgewotICAgICAgICAvLyB1bmNsZWFyIGlmIHRoaXMgaXMgYW4gZXJyb3Igb3Igbm90Li4uCi0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBpbnQzMl90IHNpemU7Ci0gICAgR0dMU3VyZmFjZSogc3VyZmFjZTsKLSAgICAvLyBhbGwgbWlwbWFwIGxldmVscyBhcmUgc3BlY2lmaWVkIGF0IG9uY2UuCi0gICAgY29uc3QgaW50IG51bUxldmVscyA9IGxldmVsPDAgPyAtbGV2ZWwgOiAxOwotICAgIGZvciAoaW50IGk9MCA7IGk8bnVtTGV2ZWxzIDsgaSsrKSB7Ci0gICAgICAgIGludCBsb2RfdyA9ICh3aWR0aCAgPj4gaSkgPyA6IDE7Ci0gICAgICAgIGludCBsb2RfaCA9IChoZWlnaHQgPj4gaSkgPyA6IDE7Ci0gICAgICAgIGludCBlcnJvciA9IGNyZWF0ZVRleHR1cmVTdXJmYWNlKGMsICZzdXJmYWNlLCAmc2l6ZSwKLSAgICAgICAgICAgICAgICBpLCBmb3JtYXQsIHR5cGUsIGxvZF93LCBsb2RfaCk7Ci0gICAgICAgIGlmIChlcnJvcikgewotICAgICAgICAgICAgb2dsZXNfZXJyb3IoYywgZXJyb3IpOwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0gICAgICAgIGRlY29kZVBhbGV0dGU0KGRhdGEsIGksIHdpZHRoLCBoZWlnaHQsCi0gICAgICAgICAgICAgICAgc3VyZmFjZS0+ZGF0YSwgc3VyZmFjZS0+c3RyaWRlLCBpbnRlcm5hbGZvcm1hdCk7Ci0gICAgfQotfQotCi0KLXZvaWQgZ2xUZXhJbWFnZTJEKAotICAgICAgICBHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgaW50ZXJuYWxmb3JtYXQsCi0gICAgICAgIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LCBHTGludCBib3JkZXIsCi0gICAgICAgIEdMZW51bSBmb3JtYXQsIEdMZW51bSB0eXBlLCBjb25zdCBHTHZvaWQgKnBpeGVscykKLXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGlmICh0YXJnZXQgIT0gR0xfVEVYVFVSRV8yRCAmJiB0YXJnZXQgIT0gR0xfRElSRUNUX1RFWFRVUkVfMkRfUVVBTENPTU0pIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBpZiAod2lkdGg8MCB8fCBoZWlnaHQ8MCB8fCBib3JkZXIhPTAgfHwgbGV2ZWwgPCAwKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmIChmb3JtYXQgIT0gaW50ZXJuYWxmb3JtYXQpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9PUEVSQVRJT04pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmICh2YWxpZEZvcm1hdFR5cGUoYywgZm9ybWF0LCB0eXBlKSkgewotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgaW50MzJfdCBzaXplID0gMDsKLSAgICBHR0xTdXJmYWNlKiBzdXJmYWNlID0gMDsKLSAgICBpZiAodGFyZ2V0ICE9IEdMX0RJUkVDVF9URVhUVVJFXzJEX1FVQUxDT01NKSB7Ci0gICAgICAgIGludCBlcnJvciA9IGNyZWF0ZVRleHR1cmVTdXJmYWNlKGMsICZzdXJmYWNlLCAmc2l6ZSwKLSAgICAgICAgICAgICAgICBsZXZlbCwgZm9ybWF0LCB0eXBlLCB3aWR0aCwgaGVpZ2h0KTsKLSAgICAgICAgaWYgKGVycm9yKSB7Ci0gICAgICAgICAgICBvZ2xlc19lcnJvcihjLCBlcnJvcik7Ci0gICAgICAgICAgICByZXR1cm47Ci0gICAgICAgIH0KLSAgICB9IGVsc2UgaWYgKHBpeGVscyA9PSAwIHx8IGxldmVsICE9IDApIHsKLSAgICAgICAgLy8gcGl4ZWwgY2FuJ3QgYmUgbnVsbCBmb3IgZGlyZWN0IHRleHR1cmUKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9PUEVSQVRJT04pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgaWYgKHBpeGVscykgewotICAgICAgICBjb25zdCBpbnQzMl90IGZvcm1hdElkeCA9IGNvbnZlcnRHTFBpeGVsRm9ybWF0KGZvcm1hdCwgdHlwZSk7Ci0gICAgICAgIGNvbnN0IEdHTEZvcm1hdCYgcGl4ZWxGb3JtYXQoYy0+cmFzdGVyaXplci5mb3JtYXRzW2Zvcm1hdElkeF0pOwotICAgICAgICBjb25zdCBpbnQzMl90IGFsaWduID0gYy0+dGV4dHVyZXMudW5wYWNrQWxpZ25tZW50LTE7Ci0gICAgICAgIGNvbnN0IGludDMyX3QgYnByID0gKCh3aWR0aCAqIHBpeGVsRm9ybWF0LnNpemUpICsgYWxpZ24pICYgfmFsaWduOwotICAgICAgICBjb25zdCBzaXplX3Qgc2l6ZSA9IGJwciAqIGhlaWdodDsKLSAgICAgICAgY29uc3QgaW50MzJfdCBzdHJpZGUgPSBicHIgLyBwaXhlbEZvcm1hdC5zaXplOwotCi0gICAgICAgIEdHTFN1cmZhY2UgdXNlclN1cmZhY2U7Ci0gICAgICAgIHVzZXJTdXJmYWNlLnZlcnNpb24gPSBzaXplb2YodXNlclN1cmZhY2UpOwotICAgICAgICB1c2VyU3VyZmFjZS53aWR0aCAgPSB3aWR0aDsKLSAgICAgICAgdXNlclN1cmZhY2UuaGVpZ2h0ID0gaGVpZ2h0OwotICAgICAgICB1c2VyU3VyZmFjZS5zdHJpZGUgPSBzdHJpZGU7Ci0gICAgICAgIHVzZXJTdXJmYWNlLmZvcm1hdCA9IGZvcm1hdElkeDsKLSAgICAgICAgdXNlclN1cmZhY2UuY29tcHJlc3NlZEZvcm1hdCA9IDA7Ci0gICAgICAgIHVzZXJTdXJmYWNlLmRhdGEgPSAoR0x1Ynl0ZSopcGl4ZWxzOwotCi0gICAgICAgIGlmICh0YXJnZXQgIT0gR0xfRElSRUNUX1RFWFRVUkVfMkRfUVVBTENPTU0pIHsKLSAgICAgICAgICAgIGludCBlcnIgPSBjb3B5UGl4ZWxzKGMsICpzdXJmYWNlLCAwLCAwLCB1c2VyU3VyZmFjZSwgMCwgMCwgd2lkdGgsIGhlaWdodCk7IAotICAgICAgICAgICAgaWYgKGVycikgewotICAgICAgICAgICAgICAgIG9nbGVzX2Vycm9yKGMsIGVycik7Ci0gICAgICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZ2VuZXJhdGVNaXBtYXAoYywgbGV2ZWwpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgLy8gYmluZCBpdCB0byB0aGUgdGV4dHVyZSB1bml0Ci0gICAgICAgICAgICBzcDxFR0xUZXh0dXJlT2JqZWN0PiB0ZXggPSBnZXRBbmRCaW5kQWN0aXZlVGV4dHVyZU9iamVjdChjKTsKLSAgICAgICAgICAgIHRleC0+c2V0U3VyZmFjZSgmdXNlclN1cmZhY2UpOwotICAgICAgICB9Ci0gICAgfQotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXZvaWQgZ2xDb21wcmVzc2VkVGV4U3ViSW1hZ2UyRCgKLSAgICAgICAgR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMaW50IHhvZmZzZXQsCi0gICAgICAgIEdMaW50IHlvZmZzZXQsIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LAotICAgICAgICBHTGVudW0gZm9ybWF0LCBHTHNpemVpIGltYWdlU2l6ZSwKLSAgICAgICAgY29uc3QgR0x2b2lkICpkYXRhKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLX0KLQotdm9pZCBnbFRleFN1YkltYWdlMkQoCi0gICAgICAgIEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGludCB4b2Zmc2V0LAotICAgICAgICBHTGludCB5b2Zmc2V0LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwKLSAgICAgICAgR0xlbnVtIGZvcm1hdCwgR0xlbnVtIHR5cGUsIGNvbnN0IEdMdm9pZCAqcGl4ZWxzKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKHRhcmdldCAhPSBHTF9URVhUVVJFXzJEKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKHhvZmZzZXQ8MCB8fCB5b2Zmc2V0PDAgfHwgd2lkdGg8MCB8fCBoZWlnaHQ8MCB8fCBsZXZlbDwwKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmICh2YWxpZEZvcm1hdFR5cGUoYywgZm9ybWF0LCB0eXBlKSkgewotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgLy8gZmluZCBvdXQgd2hpY2ggdGV4dHVyZSBpcyBib3VuZCB0byB0aGUgY3VycmVudCB1bml0Ci0gICAgY29uc3QgaW50IGFjdGl2ZSA9IGMtPnRleHR1cmVzLmFjdGl2ZTsKLSAgICBFR0xUZXh0dXJlT2JqZWN0KiB0ZXggPSBjLT50ZXh0dXJlcy50bXVbYWN0aXZlXS50ZXh0dXJlOwotICAgIGNvbnN0IEdHTFN1cmZhY2UmIHN1cmZhY2UodGV4LT5taXAobGV2ZWwpKTsKLQotICAgIGlmICghdGV4LT5pbnRlcm5hbGZvcm1hdCB8fCB0ZXgtPmRpcmVjdCkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX09QRVJBVElPTik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKCh4b2Zmc2V0ICsgd2lkdGggID4gR0xzaXplaShzdXJmYWNlLndpZHRoKSkgfHwKLSAgICAgICAgKHlvZmZzZXQgKyBoZWlnaHQgPiBHTHNpemVpKHN1cmZhY2UuaGVpZ2h0KSkpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKCF3aWR0aCB8fCAhaGVpZ2h0KSB7Ci0gICAgICAgIHJldHVybjsgLy8gb2theSwgYnV0IG5vLW9wLgotICAgIH0KLQotICAgIC8vIGZpZ3VyZSBvdXQgdGhlIHNpemUgd2UgbmVlZCBhcyB3ZWxsIGFzIHRoZSBzdHJpZGUKLSAgICBjb25zdCBpbnQzMl90IGZvcm1hdElkeCA9IGNvbnZlcnRHTFBpeGVsRm9ybWF0KGZvcm1hdCwgdHlwZSk7Ci0gICAgaWYgKGZvcm1hdElkeCA9PSAwKSB7IC8vIHdlIGRvbid0IGtub3cgd2hhdCB0byBkbyB3aXRoIHRoaXMKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9PUEVSQVRJT04pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgY29uc3QgR0dMRm9ybWF0JiBwaXhlbEZvcm1hdChjLT5yYXN0ZXJpemVyLmZvcm1hdHNbZm9ybWF0SWR4XSk7Ci0gICAgY29uc3QgaW50MzJfdCBhbGlnbiA9IGMtPnRleHR1cmVzLnVucGFja0FsaWdubWVudC0xOwotICAgIGNvbnN0IGludDMyX3QgYnByID0gKCh3aWR0aCAqIHBpeGVsRm9ybWF0LnNpemUpICsgYWxpZ24pICYgfmFsaWduOwotICAgIGNvbnN0IHNpemVfdCBzaXplID0gYnByICogaGVpZ2h0OwotICAgIGNvbnN0IGludDMyX3Qgc3RyaWRlID0gYnByIC8gcGl4ZWxGb3JtYXQuc2l6ZTsKLSAgICBHR0xTdXJmYWNlIHVzZXJTdXJmYWNlOwotICAgIHVzZXJTdXJmYWNlLnZlcnNpb24gPSBzaXplb2YodXNlclN1cmZhY2UpOwotICAgIHVzZXJTdXJmYWNlLndpZHRoICA9IHdpZHRoOwotICAgIHVzZXJTdXJmYWNlLmhlaWdodCA9IGhlaWdodDsKLSAgICB1c2VyU3VyZmFjZS5zdHJpZGUgPSBzdHJpZGU7Ci0gICAgdXNlclN1cmZhY2UuZm9ybWF0ID0gZm9ybWF0SWR4OwotICAgIHVzZXJTdXJmYWNlLmNvbXByZXNzZWRGb3JtYXQgPSAwOwotICAgIHVzZXJTdXJmYWNlLmRhdGEgPSAoR0x1Ynl0ZSopcGl4ZWxzOwotCi0gICAgaW50IGVyciA9IGNvcHlQaXhlbHMoYywKLSAgICAgICAgICAgIHN1cmZhY2UsIHhvZmZzZXQsIHlvZmZzZXQsCi0gICAgICAgICAgICB1c2VyU3VyZmFjZSwgMCwgMCwgd2lkdGgsIGhlaWdodCk7IAotICAgIGlmIChlcnIpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgZXJyKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLQotICAgIGdlbmVyYXRlTWlwbWFwKGMsIGxldmVsKTsKLQotICAgIC8vIHNpbmNlIHdlIG9ubHkgY2hhbmdlZCB0aGUgY29udGVudCBvZiB0aGUgdGV4dHVyZSwgd2UgZG9uJ3QgbmVlZAotICAgIC8vIHRvIGNhbGwgYmluZFRleHR1cmUgb24gdGhlIG1haW4gcmFzdGVyaXplci4KLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIGdsQ29weVRleEltYWdlMkQoCi0gICAgICAgIEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGVudW0gaW50ZXJuYWxmb3JtYXQsCi0gICAgICAgIEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LAotICAgICAgICBHTGludCBib3JkZXIpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBpZiAodGFyZ2V0ICE9IEdMX1RFWFRVUkVfMkQpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBpZiAoaW50ZXJuYWxmb3JtYXQ8R0xfQUxQSEEgfHwgaW50ZXJuYWxmb3JtYXQ+R0xfTFVNSU5BTkNFX0FMUEhBKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKHdpZHRoPDAgfHwgaGVpZ2h0PDAgfHwgYm9yZGVyIT0wIHx8IGxldmVsPDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBHTGVudW0gZm9ybWF0ID0gMDsKLSAgICBHTGVudW0gdHlwZSA9IEdMX1VOU0lHTkVEX0JZVEU7Ci0gICAgY29uc3QgR0dMU3VyZmFjZSYgY2JTdXJmYWNlID0gYy0+cmFzdGVyaXplci5zdGF0ZS5idWZmZXJzLmNvbG9yLnM7Ci0gICAgY29uc3QgaW50IGNiRm9ybWF0SWR4ID0gY2JTdXJmYWNlLmZvcm1hdDsKLSAgICBzd2l0Y2ggKGNiRm9ybWF0SWR4KSB7Ci0gICAgY2FzZSBHR0xfUElYRUxfRk9STUFUX1JHQl81NjU6Ci0gICAgICAgIHR5cGUgPSBHTF9VTlNJR05FRF9TSE9SVF81XzZfNTsKLSAgICAgICAgYnJlYWs7Ci0gICAgY2FzZSBHR0xfUElYRUxfRk9STUFUX1JHQkFfNTU1MToKLSAgICAgICAgdHlwZSA9IEdMX1VOU0lHTkVEX1NIT1JUXzVfNV81XzE7Ci0gICAgICAgIGJyZWFrOwotICAgIGNhc2UgR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzQ0NDQ6Ci0gICAgICAgIHR5cGUgPSBHTF9VTlNJR05FRF9TSE9SVF80XzRfNF80OwotICAgICAgICBicmVhazsKLSAgICB9Ci0gICAgc3dpdGNoIChpbnRlcm5hbGZvcm1hdCkgewotICAgIGNhc2UgR0xfQUxQSEE6Ci0gICAgY2FzZSBHTF9MVU1JTkFOQ0VfQUxQSEE6Ci0gICAgY2FzZSBHTF9MVU1JTkFOQ0U6Ci0gICAgICAgIHR5cGUgPSBHTF9VTlNJR05FRF9CWVRFOwotICAgICAgICBicmVhazsgICAgCi0gICAgfQotCi0gICAgLy8gZmlndXJlIG91dCB0aGUgZm9ybWF0IHRvIHVzZSBmb3IgdGhlIG5ldyB0ZXh0dXJlCi0gICAgc3dpdGNoIChjYkZvcm1hdElkeCkgewotICAgIGNhc2UgR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzg4ODg6Ci0gICAgY2FzZSBHR0xfUElYRUxfRk9STUFUX0FfODoKLSAgICBjYXNlIEdHTF9QSVhFTF9GT1JNQVRfUkdCQV81NTUxOgotICAgIGNhc2UgR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzQ0NDQ6Ci0gICAgICAgIGZvcm1hdCA9IGludGVybmFsZm9ybWF0OwotICAgICAgICBicmVhazsgICAgCi0gICAgY2FzZSBHR0xfUElYRUxfRk9STUFUX1JHQlhfODg4ODoKLSAgICBjYXNlIEdHTF9QSVhFTF9GT1JNQVRfUkdCXzg4ODoKLSAgICBjYXNlIEdHTF9QSVhFTF9GT1JNQVRfUkdCXzU2NToKLSAgICBjYXNlIEdHTF9QSVhFTF9GT1JNQVRfTF84OgotICAgICAgICBzd2l0Y2ggKGludGVybmFsZm9ybWF0KSB7Ci0gICAgICAgIGNhc2UgR0xfTFVNSU5BTkNFOgotICAgICAgICBjYXNlIEdMX1JHQjoKLSAgICAgICAgICAgIGZvcm1hdCA9IGludGVybmFsZm9ybWF0OwotICAgICAgICAgICAgYnJlYWs7ICAgIAotICAgICAgICB9Ci0gICAgICAgIGJyZWFrOwotICAgIH0KLQotICAgIGlmIChmb3JtYXQgPT0gMCkgewotICAgICAgICAvLyBpbnZhbGlkIGNvbWJpbmF0aW9uCi0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICAvLyBjcmVhdGUgdGhlIG5ldyB0ZXh0dXJlLi4uCi0gICAgaW50MzJfdCBzaXplOwotICAgIEdHTFN1cmZhY2UqIHN1cmZhY2U7Ci0gICAgaW50IGVycm9yID0gY3JlYXRlVGV4dHVyZVN1cmZhY2UoYywgJnN1cmZhY2UsICZzaXplLAotICAgICAgICAgICAgbGV2ZWwsIGZvcm1hdCwgdHlwZSwgd2lkdGgsIGhlaWdodCk7Ci0gICAgaWYgKGVycm9yKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIGVycm9yKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICAKLSAgICAvLyBUaGUgYm90dG9tIHJvdyBpcyBzdG9yZWQgZmlyc3QgaW4gdGV4dHVyZXMKLSAgICBHR0xTdXJmYWNlIHR4U3VyZmFjZSgqc3VyZmFjZSk7Ci0gICAgdHhTdXJmYWNlLnN0cmlkZSA9IC10eFN1cmZhY2Uuc3RyaWRlOwotCi0gICAgLy8gKHgseSkgaXMgdGhlIGxvd2VyLWxlZnQgY29ybmVyIG9mIGNvbG9yQnVmZmVyCi0gICAgeSA9IGNiU3VyZmFjZS5oZWlnaHQgLSAoeSArIGhlaWdodCk7Ci0KLSAgICBpbnQgZXJyID0gY29weVBpeGVscyhjLAotICAgICAgICAgICAgdHhTdXJmYWNlLCAwLCAwLAotICAgICAgICAgICAgY2JTdXJmYWNlLCB4LCB5LCBjYlN1cmZhY2Uud2lkdGgsIGNiU3VyZmFjZS5oZWlnaHQpOyAgCi0gICAgaWYgKGVycikgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBlcnIpOwotICAgIH0KLQotICAgIGdlbmVyYXRlTWlwbWFwKGMsIGxldmVsKTsKLX0KLQotdm9pZCBnbENvcHlUZXhTdWJJbWFnZTJEKAotICAgICAgICBHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgeG9mZnNldCwgR0xpbnQgeW9mZnNldCwKLSAgICAgICAgR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBpZiAodGFyZ2V0ICE9IEdMX1RFWFRVUkVfMkQpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9FTlVNKTsKLSAgICAgICAgcmV0dXJuOwotICAgIH0KLSAgICBpZiAoeG9mZnNldDwwIHx8IHlvZmZzZXQ8MCB8fCB3aWR0aDwwIHx8IGhlaWdodDwwIHx8IGxldmVsPDApIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKCF3aWR0aCB8fCAhaGVpZ2h0KSB7Ci0gICAgICAgIHJldHVybjsgLy8gb2theSwgYnV0IG5vLW9wLgotICAgIH0KLQotICAgIC8vIGZpbmQgb3V0IHdoaWNoIHRleHR1cmUgaXMgYm91bmQgdG8gdGhlIGN1cnJlbnQgdW5pdAotICAgIGNvbnN0IGludCBhY3RpdmUgPSBjLT50ZXh0dXJlcy5hY3RpdmU7Ci0gICAgRUdMVGV4dHVyZU9iamVjdCogdGV4ID0gYy0+dGV4dHVyZXMudG11W2FjdGl2ZV0udGV4dHVyZTsKLSAgICBjb25zdCBHR0xTdXJmYWNlJiBzdXJmYWNlKHRleC0+bWlwKGxldmVsKSk7Ci0KLSAgICBpZiAoIXRleC0+aW50ZXJuYWxmb3JtYXQpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9PUEVSQVRJT04pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmICgoeG9mZnNldCArIHdpZHRoICA+IEdMc2l6ZWkoc3VyZmFjZS53aWR0aCkpIHx8Ci0gICAgICAgICh5b2Zmc2V0ICsgaGVpZ2h0ID4gR0xzaXplaShzdXJmYWNlLmhlaWdodCkpKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgLy8gVGhlIGJvdHRvbSByb3cgaXMgc3RvcmVkIGZpcnN0IGluIHRleHR1cmVzCi0gICAgR0dMU3VyZmFjZSB0eFN1cmZhY2Uoc3VyZmFjZSk7Ci0gICAgdHhTdXJmYWNlLnN0cmlkZSA9IC10eFN1cmZhY2Uuc3RyaWRlOwotCi0gICAgLy8gKHgseSkgaXMgdGhlIGxvd2VyLWxlZnQgY29ybmVyIG9mIGNvbG9yQnVmZmVyCi0gICAgY29uc3QgR0dMU3VyZmFjZSYgY2JTdXJmYWNlID0gYy0+cmFzdGVyaXplci5zdGF0ZS5idWZmZXJzLmNvbG9yLnM7Ci0gICAgeSA9IGNiU3VyZmFjZS5oZWlnaHQgLSAoeSArIGhlaWdodCk7Ci0KLSAgICBpbnQgZXJyID0gY29weVBpeGVscyhjLAotICAgICAgICAgICAgc3VyZmFjZSwgeG9mZnNldCwgeW9mZnNldCwKLSAgICAgICAgICAgIGNiU3VyZmFjZSwgeCwgeSwgd2lkdGgsIGhlaWdodCk7ICAKLSAgICBpZiAoZXJyKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIGVycik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBnZW5lcmF0ZU1pcG1hcChjLCBsZXZlbCk7Ci19Ci0KLXZvaWQgZ2xSZWFkUGl4ZWxzKAotICAgICAgICBHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwKLSAgICAgICAgR0xlbnVtIGZvcm1hdCwgR0xlbnVtIHR5cGUsIEdMdm9pZCAqcGl4ZWxzKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgaWYgKChmb3JtYXQgIT0gR0xfUkdCQSkgJiYgKGZvcm1hdCAhPSBHTF9SR0IpKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfRU5VTSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0gICAgaWYgKCh0eXBlICE9IEdMX1VOU0lHTkVEX0JZVEUpICYmICh0eXBlICE9IEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81KSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmICh3aWR0aDwwIHx8IGhlaWdodDwwKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotICAgIGlmICh4PDAgfHwgeDwwKSB7Ci0gICAgICAgIG9nbGVzX2Vycm9yKGMsIEdMX0lOVkFMSURfVkFMVUUpOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgaW50MzJfdCBmb3JtYXRJZHggPSBHR0xfUElYRUxfRk9STUFUX05PTkU7Ci0gICAgaWYgKChmb3JtYXQgPT0gR0xfUkdCQSkgJiYgKHR5cGUgPT0gR0xfVU5TSUdORURfQllURSkpIHsKLSAgICAgICAgZm9ybWF0SWR4ID0gR0dMX1BJWEVMX0ZPUk1BVF9SR0JBXzg4ODg7Ci0gICAgfSBlbHNlIGlmICgoZm9ybWF0ID09IEdMX1JHQikgJiYgKHR5cGUgPT0gR0xfVU5TSUdORURfU0hPUlRfNV82XzUpKSB7Ci0gICAgICAgIGZvcm1hdElkeCA9IEdHTF9QSVhFTF9GT1JNQVRfUkdCXzU2NTsKLSAgICB9IGVsc2UgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX09QRVJBVElPTik7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBjb25zdCBHR0xTdXJmYWNlJiByZWFkU3VyZmFjZSA9IGMtPnJhc3Rlcml6ZXIuc3RhdGUuYnVmZmVycy5yZWFkLnM7Ci0gICAgaWYgKCh4K3dpZHRoID4gR0xpbnQocmVhZFN1cmZhY2Uud2lkdGgpKSB8fAotICAgICAgICAgICAgKHkraGVpZ2h0ID4gR0xpbnQocmVhZFN1cmZhY2UuaGVpZ2h0KSkpIHsKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfSU5WQUxJRF9WQUxVRSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBjb25zdCBHR0xGb3JtYXQmIHBpeGVsRm9ybWF0KGMtPnJhc3Rlcml6ZXIuZm9ybWF0c1tmb3JtYXRJZHhdKTsKLSAgICBjb25zdCBpbnQzMl90IGFsaWduID0gYy0+dGV4dHVyZXMucGFja0FsaWdubWVudC0xOwotICAgIGNvbnN0IGludDMyX3QgYnByID0gKCh3aWR0aCAqIHBpeGVsRm9ybWF0LnNpemUpICsgYWxpZ24pICYgfmFsaWduOwotICAgIGNvbnN0IGludDMyX3Qgc3RyaWRlID0gYnByIC8gcGl4ZWxGb3JtYXQuc2l6ZTsKLQotICAgIEdHTFN1cmZhY2UgdXNlclN1cmZhY2U7Ci0gICAgdXNlclN1cmZhY2UudmVyc2lvbiA9IHNpemVvZih1c2VyU3VyZmFjZSk7Ci0gICAgdXNlclN1cmZhY2Uud2lkdGggID0gd2lkdGg7Ci0gICAgdXNlclN1cmZhY2UuaGVpZ2h0ID0gaGVpZ2h0OwotICAgIHVzZXJTdXJmYWNlLnN0cmlkZSA9IC1zdHJpZGU7IC8vIGJvdHRvbSByb3cgaXMgdHJhbnNmZXJlZCBmaXJzdAotICAgIHVzZXJTdXJmYWNlLmZvcm1hdCA9IGZvcm1hdElkeDsKLSAgICB1c2VyU3VyZmFjZS5jb21wcmVzc2VkRm9ybWF0ID0gMDsKLSAgICB1c2VyU3VyZmFjZS5kYXRhID0gKEdMdWJ5dGUqKXBpeGVsczsKLQotICAgIC8vIHVzZSBwaXhlbC1mbGluZ2VyIHRvIGhhbmRsZSBhbGwgdGhlIGNvbnZlcnNpb25zCi0gICAgR0dMQ29udGV4dCogZ2dsID0gZ2V0UmFzdGVyaXplcihjKTsKLSAgICBpZiAoIWdnbCkgewotICAgICAgICAvLyB0aGUgb25seSByZWFzb24gdGhpcyB3b3VsZCBmYWlsIGlzIGJlY2F1c2Ugd2UgcmFuIG91dCBvZiBtZW1vcnkKLSAgICAgICAgb2dsZXNfZXJyb3IoYywgR0xfT1VUX09GX01FTU9SWSk7Ci0gICAgICAgIHJldHVybjsKLSAgICB9Ci0KLSAgICBnZ2wtPmNvbG9yQnVmZmVyKGdnbCwgJnVzZXJTdXJmYWNlKTsgIC8vIGRlc3RpbmF0aW9uIGlzIHVzZXIgYnVmZmVyIAotICAgIGdnbC0+YmluZFRleHR1cmUoZ2dsLCAmcmVhZFN1cmZhY2UpOyAgLy8gc291cmNlIGlzIHJlYWQtYnVmZmVyCi0gICAgZ2dsLT50ZXhDb29yZDJpKGdnbCwgeCwgcmVhZFN1cmZhY2UuaGVpZ2h0IC0gKHkgKyBoZWlnaHQpKTsKLSAgICBnZ2wtPnJlY3RpKGdnbCwgMCwgMCwgd2lkdGgsIGhlaWdodCk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLSNpZiAwCi0jcHJhZ21hIG1hcmsgLQotI3ByYWdtYSBtYXJrIERyYXdUZXh0dXJlIEV4dGVuc2lvbgotI2VuZGlmCi0KLXZvaWQgZ2xEcmF3VGV4c3ZPRVMoY29uc3QgR0xzaG9ydCogY29vcmRzKSB7Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBkcmF3VGV4aU9FUyhjb29yZHNbMF0sIGNvb3Jkc1sxXSwgY29vcmRzWzJdLCBjb29yZHNbM10sIGNvb3Jkc1s0XSwgYyk7Ci19Ci12b2lkIGdsRHJhd1RleGl2T0VTKGNvbnN0IEdMaW50KiBjb29yZHMpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGRyYXdUZXhpT0VTKGNvb3Jkc1swXSwgY29vcmRzWzFdLCBjb29yZHNbMl0sIGNvb3Jkc1szXSwgY29vcmRzWzRdLCBjKTsKLX0KLXZvaWQgZ2xEcmF3VGV4c09FUyhHTHNob3J0IHggLCBHTHNob3J0IHksIEdMc2hvcnQgeiwgR0xzaG9ydCB3LCBHTHNob3J0IGgpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGRyYXdUZXhpT0VTKHgsIHksIHosIHcsIGgsIGMpOwotfQotdm9pZCBnbERyYXdUZXhpT0VTKEdMaW50IHgsIEdMaW50IHksIEdMaW50IHosIEdMaW50IHcsIEdMaW50IGgpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGRyYXdUZXhpT0VTKHgsIHksIHosIHcsIGgsIGMpOwotfQotCi12b2lkIGdsRHJhd1RleGZ2T0VTKGNvbnN0IEdMZmxvYXQqIGNvb3JkcykgewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgZHJhd1RleHhPRVMoCi0gICAgICAgICAgICBnZ2xGbG9hdFRvRml4ZWQoY29vcmRzWzBdKSwKLSAgICAgICAgICAgIGdnbEZsb2F0VG9GaXhlZChjb29yZHNbMV0pLAotICAgICAgICAgICAgZ2dsRmxvYXRUb0ZpeGVkKGNvb3Jkc1syXSksCi0gICAgICAgICAgICBnZ2xGbG9hdFRvRml4ZWQoY29vcmRzWzNdKSwKLSAgICAgICAgICAgIGdnbEZsb2F0VG9GaXhlZChjb29yZHNbNF0pLAotICAgICAgICAgICAgYyk7Ci19Ci12b2lkIGdsRHJhd1RleHh2T0VTKGNvbnN0IEdMZml4ZWQqIGNvb3JkcykgewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgZHJhd1RleHhPRVMoY29vcmRzWzBdLCBjb29yZHNbMV0sIGNvb3Jkc1syXSwgY29vcmRzWzNdLCBjb29yZHNbNF0sIGMpOwotfQotdm9pZCBnbERyYXdUZXhmT0VTKEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHosIEdMZmxvYXQgdywgR0xmbG9hdCBoKXsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGRyYXdUZXh4T0VTKAotICAgICAgICAgICAgZ2dsRmxvYXRUb0ZpeGVkKHgpLCBnZ2xGbG9hdFRvRml4ZWQoeSksIGdnbEZsb2F0VG9GaXhlZCh6KSwKLSAgICAgICAgICAgIGdnbEZsb2F0VG9GaXhlZCh3KSwgZ2dsRmxvYXRUb0ZpeGVkKGgpLAotICAgICAgICAgICAgYyk7Ci19Ci12b2lkIGdsRHJhd1RleHhPRVMoR0xmaXhlZCB4LCBHTGZpeGVkIHksIEdMZml4ZWQgeiwgR0xmaXhlZCB3LCBHTGZpeGVkIGgpIHsKLSAgICBvZ2xlc19jb250ZXh0X3QqIGMgPSBvZ2xlc19jb250ZXh0X3Q6OmdldCgpOwotICAgIGRyYXdUZXh4T0VTKHgsIHksIHosIHcsIGgsIGMpOwotfQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC90ZXh0dXJlLmggYi9vcGVuZ2wvbGliYWdsL3RleHR1cmUuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNWM1Nzk0OC4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGliYWdsL3RleHR1cmUuaAorKysgL2Rldi9udWxsCkBAIC0xLDQ1ICswLDAgQEAKLS8qIGxpYnMvb3BlbmdsZXMvdGV4dHVyZS5oCi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2lmbmRlZiBBTkRST0lEX09QRU5HTEVTX1RFWFRVUkVfSAotI2RlZmluZSBBTkRST0lEX09QRU5HTEVTX1RFWFRVUkVfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3RkZGVmLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLSNpbmNsdWRlIDxwcml2YXRlL3BpeGVsZmxpbmdlci9nZ2xfY29udGV4dC5oPgotCi0jaW5jbHVkZSA8R0xFUy9nbC5oPgotCi0jaW5jbHVkZSAiY29udGV4dC5oIgotCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0KLXZvaWQgb2dsZXNfaW5pdF90ZXh0dXJlKG9nbGVzX2NvbnRleHRfdCogYyk7Ci12b2lkIG9nbGVzX3VuaW5pdF90ZXh0dXJlKG9nbGVzX2NvbnRleHRfdCogYyk7Ci12b2lkIG9nbGVzX3ZhbGlkYXRlX3RleHR1cmVfaW1wbChvZ2xlc19jb250ZXh0X3QqIGMpOwotCi1pbmxpbmUgdm9pZCBvZ2xlc192YWxpZGF0ZV90ZXh0dXJlKG9nbGVzX2NvbnRleHRfdCogYykgewotICAgIGlmIChjLT5yYXN0ZXJpemVyLnN0YXRlLmVuYWJsZXMgJiBHR0xfRU5BQkxFX1RNVVMpCi0gICAgICAgIG9nbGVzX3ZhbGlkYXRlX3RleHR1cmVfaW1wbChjKTsKLX0KLQotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9PUEVOR0xFU19URVhUVVJFX0gKZGlmZiAtLWdpdCBhL29wZW5nbC9saWJhZ2wvdmVydGV4LmNwcCBiL29wZW5nbC9saWJhZ2wvdmVydGV4LmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGFkMDRkNi4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGliYWdsL3ZlcnRleC5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwyNDcgKzAsMCBAQAotLyogbGlicy9vcGVuZ2xlcy92ZXJ0ZXguY3BwCi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSAiY29udGV4dC5oIgotI2luY2x1ZGUgImZwLmgiCi0jaW5jbHVkZSAidmVydGV4LmgiCi0jaW5jbHVkZSAic3RhdGUuaCIKLSNpbmNsdWRlICJtYXRyaXguaCIKLQotbmFtZXNwYWNlIGFuZHJvaWQgewotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXZvaWQgb2dsZXNfaW5pdF92ZXJ0ZXgob2dsZXNfY29udGV4dF90KiBjKQotewotICAgIGMtPmN1bGwuZW5hYmxlID0gR0xfRkFMU0U7Ci0gICAgYy0+Y3VsbC5jdWxsRmFjZSA9IEdMX0JBQ0s7Ci0gICAgYy0+Y3VsbC5mcm9udEZhY2UgPSBHTF9DQ1c7Ci0KLSAgICBjLT5jdXJyZW50LmNvbG9yLnIgPSAweDEwMDAwOwotICAgIGMtPmN1cnJlbnQuY29sb3IuZyA9IDB4MTAwMDA7Ci0gICAgYy0+Y3VycmVudC5jb2xvci5iID0gMHgxMDAwMDsKLSAgICBjLT5jdXJyZW50LmNvbG9yLmEgPSAweDEwMDAwOwotCi0gICAgYy0+Y3VycmVudE5vcm1hbC56ID0gMHgxMDAwMDsKLX0KLQotdm9pZCBvZ2xlc191bmluaXRfdmVydGV4KG9nbGVzX2NvbnRleHRfdCogYykKLXsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gdmVydGV4IHByb2Nlc3NpbmcKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLy8gRGl2aWRlcyBhIHZlcnRleCBjbGlwIGNvb3JkaW5hdGVzIGJ5IFcKLXN0YXRpYyBpbmxpbmUKLXZvaWQgcGVyc3BlY3RpdmUob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdiwgdWludDMyX3QgZW5hYmxlcykKLXsKLSAgICAvLyBbeCx5LHpdd2luZG93ID0gdnB0ICogKFt4LHksel1jbGlwIC8gY2xpcC53KQotICAgIC8vIFt3XXdpbmRvdyA9IDEvdwotCi0gICAgLy8gV2l0aCBhIHJlZ3VsYXIgcHJvamVjdGlvbiBnZW5lcmF0ZWQgYnkgZ2xGcnVzdHVtKCksCi0gICAgLy8gd2UgaGF2ZSB3PS16LCB0aGVyZWZvcmUsIHcgaXMgaW4gW3pOZWFyLCB6RmFyXS4KLSAgICAvLyBBbHNvLCB6TmVhciBhbmQgekZhciBhcmUgc3RyaWNseSBwb3NpdGl2ZSwKLSAgICAvLyBhbmQgMS93ICh3aW5kb3cudykgaXMgaW4gWzEvekZhciwgMS96TmVhcl0sIHVzdWFsbHkgdGhpcwotICAgIC8vIG1lYW5zIF0wLCAraW5mWyAtLSBob3dldmVyLCBpdCBpcyBhbHdheXMgcmVjb21tZW5kZWQKLSAgICAvLyB0byB1c2UgYXMgbGFyZ2UgdmFsdWVzIGFzIHBvc3NpYmxlIGZvciB6TmVhci4KLSAgICAvLyBBbGwgaW4gYWxsLCB3IGlzIHVzdWFsbHkgc21hbGxlciB0aGFuIDEuMCAoYXNzdW1pbmcKLSAgICAvLyB6TmVhciBpcyBhdCBsZWFzdCAxLjApOyBhbmQgZXZlbiBpZiB6TmVhciBpcyBzbWFsbGVyIHRoYW4gMS4wCi0gICAgLy8gdmFsdWVzIG9mIHcgd29uJ3QgYmUgdG9vIGJpZy4KLQotICAgIGNvbnN0IGludDMyX3QgcncgPSBnZ2xSZWNpcDI4KHYtPmNsaXAudyk7Ci0gICAgY29uc3QgR0xmaXhlZCogY29uc3QgbSA9IGMtPnRyYW5zZm9ybXMudnB0LnRyYW5zZm9ybS5tYXRyaXgubTsKLSAgICB2LT53aW5kb3cudyA9IHJ3OwotICAgIHYtPndpbmRvdy54ID0gZ2dsTXVsQWRkeChnZ2xNdWx4KHYtPmNsaXAueCwgcncsIDE2KSwgbVsgMF0sIG1bMTJdLCAyOCk7IAotICAgIHYtPndpbmRvdy55ID0gZ2dsTXVsQWRkeChnZ2xNdWx4KHYtPmNsaXAueSwgcncsIDE2KSwgbVsgNV0sIG1bMTNdLCAyOCk7Ci0gICAgdi0+d2luZG93LnggPSBUUklfRlJPTV9GSVhFRCh2LT53aW5kb3cueCk7Ci0gICAgdi0+d2luZG93LnkgPSBUUklfRlJPTV9GSVhFRCh2LT53aW5kb3cueSk7Ci0gICAgaWYgKGVuYWJsZXMgJiBHR0xfRU5BQkxFX0RFUFRIX1RFU1QpIHsKLSAgICAgICAgdi0+d2luZG93LnogPSBnZ2xNdWxBZGR4KGdnbE11bHgodi0+Y2xpcC56LCBydywgMTYpLCBtWzEwXSwgbVsxNF0sIDI4KTsKLSAgICB9Ci19Ci0KLS8vIGZydXN0dW0gY2xpcHBpbmcgYW5kIFctZGl2aWRlCi1zdGF0aWMgaW5saW5lCi12b2lkIGNsaXBGcnVzdHVtUGVyc3BlY3RpdmUob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdiwgdWludDMyX3QgZW5hYmxlcykKLXsKLSAgICAvLyBuZGMgPSBjbGlwIC8gVwotICAgIC8vIHdpbmRvdyA9IG5jZCAqIHZpZXdwb3J0Ci0gICAgCi0gICAgLy8gY2xpcCB0byB0aGUgdmlldy12b2x1bWUKLSAgICB1aW50MzJfdCBjbGlwID0gdi0+ZmxhZ3MgJiB2ZXJ0ZXhfdDo6Q0xJUF9BTEw7Ci0gICAgY29uc3QgR0xmaXhlZCB3ID0gdi0+Y2xpcC53OwotICAgIGlmICh2LT5jbGlwLnggPCAtdykgICBjbGlwIHw9IHZlcnRleF90OjpDTElQX0w7Ci0gICAgaWYgKHYtPmNsaXAueCA+ICB3KSAgIGNsaXAgfD0gdmVydGV4X3Q6OkNMSVBfUjsKLSAgICBpZiAodi0+Y2xpcC55IDwgLXcpICAgY2xpcCB8PSB2ZXJ0ZXhfdDo6Q0xJUF9COwotICAgIGlmICh2LT5jbGlwLnkgPiAgdykgICBjbGlwIHw9IHZlcnRleF90OjpDTElQX1Q7Ci0gICAgaWYgKHYtPmNsaXAueiA8IC13KSAgIGNsaXAgfD0gdmVydGV4X3Q6OkNMSVBfTjsKLSAgICBpZiAodi0+Y2xpcC56ID4gIHcpICAgY2xpcCB8PSB2ZXJ0ZXhfdDo6Q0xJUF9GOwotCi0gICAgdi0+ZmxhZ3MgfD0gY2xpcDsKLSAgICBjLT5hcnJheXMuY3VsbCAmPSBjbGlwOwotCi0gICAgaWYgKGdnbF9saWtlbHkoIWNsaXApKSB7Ci0gICAgICAgIC8vIGlmIHRoZSB2ZXJ0ZXggaXMgY2xpcHBlZCwgd2UgZG9uJ3QgZG8gdGhlIHBlcnNwZWN0aXZlCi0gICAgICAgIC8vIGRpdmlkZSwgc2luY2Ugd2UgZG9uJ3QgbmVlZCBpdHMgd2luZG93IGNvb3JkaW5hdGVzLgotICAgICAgICBwZXJzcGVjdGl2ZShjLCB2LCBlbmFibGVzKTsKLSAgICB9Ci19Ci0KLS8vIGZydXN0dW0gY2xpcHBpbmcsIHVzZXIgY2xpcHBpbmcgYW5kIFctZGl2aWRlCi1zdGF0aWMgaW5saW5lCi12b2lkIGNsaXBBbGxQZXJzcGVjdGl2ZShvZ2xlc19jb250ZXh0X3QqIGMsIHZlcnRleF90KiB2LCB1aW50MzJfdCBlbmFibGVzKQotewotICAgIC8vIGNvbXB1dGUgZXllIGNvb3JkaW5hdGVzCi0gICAgYy0+YXJyYXlzLm12X3RyYW5zZm9ybSgKLSAgICAgICAgICAgICZjLT50cmFuc2Zvcm1zLm1vZGVsdmlldy50cmFuc2Zvcm0sICZ2LT5leWUsICZ2LT5vYmopOwotICAgIHYtPmZsYWdzIHw9IHZlcnRleF90OjpFWUU7Ci0KLSAgICAvLyBjbGlwIHRoaXMgdmVydGV4IGFnYWluc3QgZWFjaCB1c2VyIGNsaXAgcGxhbmUKLSAgICB1aW50MzJfdCBjbGlwID0gMDsKLSAgICBpbnQgcGxhbmVzID0gYy0+Y2xpcFBsYW5lcy5lbmFibGU7Ci0gICAgd2hpbGUgKHBsYW5lcykgewotICAgICAgICBjb25zdCBpbnQgaSA9IDMxIC0gZ2dsQ2x6KHBsYW5lcyk7Ci0gICAgICAgIHBsYW5lcyAmPSB+KDE8PGkpOwotICAgICAgICAvLyBYWFg6IHdlIHNob3VsZCBoYXZlIGEgc3BlY2lhbCBkb3QoKSBmb3IgMiwzLDQgY29vcmRzIHZlcnRpY2VzCi0gICAgICAgIEdMZml4ZWQgZCA9IGRvdDQoYy0+Y2xpcFBsYW5lcy5wbGFuZVtpXS5lcXVhdGlvbi52LCB2LT5leWUudik7Ci0gICAgICAgIGlmIChkIDwgMCkgewotICAgICAgICAgICAgY2xpcCB8PSAweDEwMDw8aTsKLSAgICAgICAgfQotICAgIH0KLSAgICB2LT5mbGFncyB8PSBjbGlwOwotCi0gICAgY2xpcEZydXN0dW1QZXJzcGVjdGl2ZShjLCB2LCBlbmFibGVzKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIG9nbGVzX3ZlcnRleF9wcm9qZWN0KG9nbGVzX2NvbnRleHRfdCogYywgdmVydGV4X3QqIHYpIHsKLSAgICBwZXJzcGVjdGl2ZShjLCB2LCBjLT5yYXN0ZXJpemVyLnN0YXRlLmVuYWJsZXMpOwotfQotCi12b2lkIG9nbGVzX3ZlcnRleF9wZXJzcGVjdGl2ZTJEKG9nbGVzX2NvbnRleHRfdCogYywgdmVydGV4X3QqIHYpCi17Ci0gICAgLy8gaGVyZSB3ZSBhc3N1bWUgdz0xLjAgYW5kIHRoZSB2aWV3cG9ydCB0cmFuc2Zvcm1hdGlvbgotICAgIC8vIGhhcyBiZWVuIGFwcGxpZWQgYWxyZWFkeS4KLSAgICBjLT5hcnJheXMuY3VsbCA9IDA7Ci0gICAgdi0+d2luZG93LnggPSBUUklfRlJPTV9GSVhFRCh2LT5jbGlwLngpOwotICAgIHYtPndpbmRvdy55ID0gVFJJX0ZST01fRklYRUQodi0+Y2xpcC55KTsKLSAgICB2LT53aW5kb3cueiA9IHYtPmNsaXAuejsKLSAgICB2LT53aW5kb3cudyA9IHYtPmNsaXAudyA8PCAxMjsKLX0KLQotdm9pZCBvZ2xlc192ZXJ0ZXhfcGVyc3BlY3RpdmUzRFoob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikgewotICAgIGNsaXBGcnVzdHVtUGVyc3BlY3RpdmUoYywgdiwgR0dMX0VOQUJMRV9ERVBUSF9URVNUKTsKLX0KLXZvaWQgb2dsZXNfdmVydGV4X3BlcnNwZWN0aXZlM0Qob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikgewotICAgIGNsaXBGcnVzdHVtUGVyc3BlY3RpdmUoYywgdiwgMCk7Ci19Ci12b2lkIG9nbGVzX3ZlcnRleF9jbGlwQWxsUGVyc3BlY3RpdmUzRFoob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikgewotICAgIGNsaXBBbGxQZXJzcGVjdGl2ZShjLCB2LCBHR0xfRU5BQkxFX0RFUFRIX1RFU1QpOwotfQotdm9pZCBvZ2xlc192ZXJ0ZXhfY2xpcEFsbFBlcnNwZWN0aXZlM0Qob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCogdikgewotICAgIGNsaXBBbGxQZXJzcGVjdGl2ZShjLCB2LCAwKTsKLX0KLQotc3RhdGljIHZvaWQgY2xpcFBsYW5leChHTGVudW0gcGxhbmUsIGNvbnN0IEdMZml4ZWQqIGVxdSwgb2dsZXNfY29udGV4dF90KiBjKQotewotICAgIGNvbnN0IGludCBwID0gcGxhbmUgLSBHTF9DTElQX1BMQU5FMDsKLSAgICBpZiAoZ2dsX3VubGlrZWx5KHVpbnQzMl90KHApID4gKEdMX0NMSVBfUExBTkU1IC0gR0xfQ0xJUF9QTEFORTApKSkgewotICAgICAgICBvZ2xlc19lcnJvcihjLCBHTF9JTlZBTElEX0VOVU0pOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgdmVjNF90JiBlcXVhdGlvbiA9IGMtPmNsaXBQbGFuZXMucGxhbmVbcF0uZXF1YXRpb247Ci0gICAgbWVtY3B5KGVxdWF0aW9uLnYsIGVxdSwgc2l6ZW9mKHZlYzRfdCkpOwotCi0gICAgb2dsZXNfdmFsaWRhdGVfdHJhbnNmb3JtKGMsIHRyYW5zZm9ybV9zdGF0ZV90OjpNVklUKTsKLSAgICB0cmFuc2Zvcm1fdCYgbXZpdCA9IGMtPnRyYW5zZm9ybXMubXZpdDQ7Ci0gICAgbXZpdC5wb2ludDQoJm12aXQsICZlcXVhdGlvbiwgJmVxdWF0aW9uKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdXNpbmcgbmFtZXNwYWNlIGFuZHJvaWQ7Ci0KLQotdm9pZCBnbENvbG9yNGYoR0xmbG9hdCByLCBHTGZsb2F0IGcsIEdMZmxvYXQgYiwgR0xmbG9hdCBhKQotewotICAgIG9nbGVzX2NvbnRleHRfdCogYyA9IG9nbGVzX2NvbnRleHRfdDo6Z2V0KCk7Ci0gICAgYy0+Y3VycmVudC5jb2xvci5yICAgICAgID0gZ2dsRmxvYXRUb0ZpeGVkKHIpOwotICAgIGMtPmN1cnJlbnRDb2xvckNsYW1wZWQuciA9IGdnbENsYW1weChjLT5jdXJyZW50LmNvbG9yLnIpOwotICAgIGMtPmN1cnJlbnQuY29sb3IuZyAgICAgICA9IGdnbEZsb2F0VG9GaXhlZChnKTsKLSAgICBjLT5jdXJyZW50Q29sb3JDbGFtcGVkLmcgPSBnZ2xDbGFtcHgoYy0+Y3VycmVudC5jb2xvci5nKTsKLSAgICBjLT5jdXJyZW50LmNvbG9yLmIgICAgICAgPSBnZ2xGbG9hdFRvRml4ZWQoYik7Ci0gICAgYy0+Y3VycmVudENvbG9yQ2xhbXBlZC5iID0gZ2dsQ2xhbXB4KGMtPmN1cnJlbnQuY29sb3IuYik7Ci0gICAgYy0+Y3VycmVudC5jb2xvci5hICAgICAgID0gZ2dsRmxvYXRUb0ZpeGVkKGEpOwotICAgIGMtPmN1cnJlbnRDb2xvckNsYW1wZWQuYSA9IGdnbENsYW1weChjLT5jdXJyZW50LmNvbG9yLmEpOwotfQotCi12b2lkIGdsQ29sb3I0eChHTGZpeGVkIHIsIEdMZml4ZWQgZywgR0xmaXhlZCBiLCBHTGZpeGVkIGEpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT5jdXJyZW50LmNvbG9yLnIgPSByOwotICAgIGMtPmN1cnJlbnQuY29sb3IuZyA9IGc7Ci0gICAgYy0+Y3VycmVudC5jb2xvci5iID0gYjsKLSAgICBjLT5jdXJyZW50LmNvbG9yLmEgPSBhOwotICAgIGMtPmN1cnJlbnRDb2xvckNsYW1wZWQuciA9IGdnbENsYW1weChyKTsKLSAgICBjLT5jdXJyZW50Q29sb3JDbGFtcGVkLmcgPSBnZ2xDbGFtcHgoZyk7Ci0gICAgYy0+Y3VycmVudENvbG9yQ2xhbXBlZC5iID0gZ2dsQ2xhbXB4KGIpOwotICAgIGMtPmN1cnJlbnRDb2xvckNsYW1wZWQuYSA9IGdnbENsYW1weChhKTsKLX0KLQotdm9pZCBnbE5vcm1hbDNmKEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT5jdXJyZW50Tm9ybWFsLnggPSBnZ2xGbG9hdFRvRml4ZWQoeCk7Ci0gICAgYy0+Y3VycmVudE5vcm1hbC55ID0gZ2dsRmxvYXRUb0ZpeGVkKHkpOwotICAgIGMtPmN1cnJlbnROb3JtYWwueiA9IGdnbEZsb2F0VG9GaXhlZCh6KTsKLX0KLQotdm9pZCBnbE5vcm1hbDN4KEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjLT5jdXJyZW50Tm9ybWFsLnggPSB4OwotICAgIGMtPmN1cnJlbnROb3JtYWwueSA9IHk7Ci0gICAgYy0+Y3VycmVudE5vcm1hbC56ID0gejsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIGdsQ2xpcFBsYW5lZihHTGVudW0gcGxhbmUsIGNvbnN0IEdMZmxvYXQqIGVxdSkKLXsKLSAgICBjb25zdCBHTGZpeGVkIGVxdXhbNF0gPSB7Ci0gICAgICAgICAgICBnZ2xGbG9hdFRvRml4ZWQoZXF1WzBdKSwKLSAgICAgICAgICAgIGdnbEZsb2F0VG9GaXhlZChlcXVbMV0pLAotICAgICAgICAgICAgZ2dsRmxvYXRUb0ZpeGVkKGVxdVsyXSksCi0gICAgICAgICAgICBnZ2xGbG9hdFRvRml4ZWQoZXF1WzNdKQotICAgIH07Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjbGlwUGxhbmV4KHBsYW5lLCBlcXV4LCBjKTsKLX0KLQotdm9pZCBnbENsaXBQbGFuZXgoR0xlbnVtIHBsYW5lLCBjb25zdCBHTGZpeGVkKiBlcXUpCi17Ci0gICAgb2dsZXNfY29udGV4dF90KiBjID0gb2dsZXNfY29udGV4dF90OjpnZXQoKTsKLSAgICBjbGlwUGxhbmV4KHBsYW5lLCBlcXUsIGMpOwotfQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYmFnbC92ZXJ0ZXguaCBiL29wZW5nbC9saWJhZ2wvdmVydGV4LmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDU1ZTYyMTMuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYmFnbC92ZXJ0ZXguaAorKysgL2Rldi9udWxsCkBAIC0xLDQ4ICswLDAgQEAKLS8qIGxpYnMvb3BlbmdsZXMvdmVydGV4LmgKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0jaWZuZGVmIEFORFJPSURfT1BFTkdMRVNfVkVSVEVYX0gKLSNkZWZpbmUgQU5EUk9JRF9PUEVOR0xFU19WRVJURVhfSAotCi0jaW5jbHVkZSA8c3RkaW50Lmg+Ci0jaW5jbHVkZSA8c3RkZGVmLmg+Ci0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLQotbmFtZXNwYWNlIGdsIHsKLXN0cnVjdCB2ZXJ0ZXhfdDsKLXN0cnVjdCBvZ2xlc19jb250ZXh0X3Q7Ci19OwotCi12b2lkIG9nbGVzX2luaXRfdmVydGV4KG9nbGVzX2NvbnRleHRfdCogYyk7Ci12b2lkIG9nbGVzX3VuaW5pdF92ZXJ0ZXgob2dsZXNfY29udGV4dF90KiBjKTsKLQotdm9pZCBvZ2xlc192ZXJ0ZXhfcGVyc3BlY3RpdmUyRChvZ2xlc19jb250ZXh0X3QqLCB2ZXJ0ZXhfdCopOwotCi12b2lkIG9nbGVzX3ZlcnRleF9wZXJzcGVjdGl2ZTNEKG9nbGVzX2NvbnRleHRfdCosIHZlcnRleF90Kik7Ci12b2lkIG9nbGVzX3ZlcnRleF9wZXJzcGVjdGl2ZTNEWihvZ2xlc19jb250ZXh0X3QqLCB2ZXJ0ZXhfdCopOwotdm9pZCBvZ2xlc192ZXJ0ZXhfY2xpcEFsbFBlcnNwZWN0aXZlM0Qob2dsZXNfY29udGV4dF90KiwgdmVydGV4X3QqKTsKLXZvaWQgb2dsZXNfdmVydGV4X2NsaXBBbGxQZXJzcGVjdGl2ZTNEWihvZ2xlc19jb250ZXh0X3QqLCB2ZXJ0ZXhfdCopOwotCi0KLXZvaWQgb2dsZXNfdmVydGV4X3Byb2plY3Qob2dsZXNfY29udGV4dF90KiBjLCB2ZXJ0ZXhfdCopOwotCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLy8gQU5EUk9JRF9PUEVOR0xFU19WRVJURVhfSAotCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGlicy9BbmRyb2lkLm1rIGIvb3BlbmdsL2xpYnMvQW5kcm9pZC5tawpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMmVjYzc3Ni4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGlicy9BbmRyb2lkLm1rCisrKyAvZGV2L251bGwKQEAgLTEsNTMgKzAsMCBAQAotTE9DQUxfUEFUSDo9ICQoY2FsbCBteS1kaXIpCi0KLSMKLSMgQnVpbGQgTUVUQSBFR0wgbGlicmFyeQotIwotCi1pbmNsdWRlICQoQ0xFQVJfVkFSUykKLQotTE9DQUxfU1JDX0ZJTEVTOj0gCVwKLQlFR0wvZWdsLmNwcCAJCVwKLQlFR0wvZ3B1LmNwcAkJCVwKLSMKLQotTE9DQUxfU0hBUkVEX0xJQlJBUklFUyArPSBsaWJjdXRpbHMgbGlidXRpbHMgbGlidWkKLUxPQ0FMX0xETElCUyA6PSAtbHB0aHJlYWQgLWxkbAotTE9DQUxfTU9EVUxFOj0gbGliRUdMCi0KLSMgbmVlZGVkIG9uIHNpbSBidWlsZCBiZWNhdXNlIG9mIHdlaXJkIGxvZ2dpbmcgaXNzdWVzCi1pZmVxICgkKFRBUkdFVF9TSU1VTEFUT1IpLHRydWUpCi1lbHNlCi0gICAgTE9DQUxfU0hBUkVEX0xJQlJBUklFUyArPSBsaWJkbAotICAgICMgd2UgbmVlZCB0byBhY2Nlc3MgdGhlIEJpb25pYyBwcml2YXRlIGhlYWRlciA8YmlvbmljX3Rscy5oPgotICAgIExPQ0FMX0NGTEFHUyArPSAtSSQoTE9DQUxfUEFUSCkvLi4vLi4vLi4vLi4vYmlvbmljL2xpYmMvcHJpdmF0ZQotZW5kaWYKLQotaW5jbHVkZSAkKEJVSUxEX1NIQVJFRF9MSUJSQVJZKQotCi0KLQotIwotIyBCdWlsZCB0aGUgd3JhcHBlciBPcGVuR0wgRVMgbGlicmFyeQotIwotCi1pbmNsdWRlICQoQ0xFQVJfVkFSUykKLQotTE9DQUxfU1JDX0ZJTEVTOj0gCVwKLQlHTEVTX0NNL2dsLmNwcC5hcm0gCQlcCi0JR0xFU19DTS9nbF9sb2dnZXIuY3BwIAlcCi0jCi0KLUxPQ0FMX1NIQVJFRF9MSUJSQVJJRVMgKz0gbGliY3V0aWxzIGxpYnV0aWxzIGxpYnVpIGxpYkVHTAotTE9DQUxfTERMSUJTIDo9IC1scHRocmVhZCAtbGRsCi1MT0NBTF9NT0RVTEU6PSBsaWJHTEVTdjFfQ00KLQotIyBuZWVkZWQgb24gc2ltIGJ1aWxkIGJlY2F1c2Ugb2Ygd2VpcmQgbG9nZ2luZyBpc3N1ZXMKLWlmZXEgKCQoVEFSR0VUX1NJTVVMQVRPUiksdHJ1ZSkKLWVsc2UKLSAgICBMT0NBTF9TSEFSRURfTElCUkFSSUVTICs9IGxpYmRsCi0gICAgIyB3ZSBuZWVkIHRvIGFjY2VzcyB0aGUgQmlvbmljIHByaXZhdGUgaGVhZGVyIDxiaW9uaWNfdGxzLmg+Ci0gICAgTE9DQUxfQ0ZMQUdTICs9IC1JJChMT0NBTF9QQVRIKS8uLi8uLi8uLi8uLi9iaW9uaWMvbGliYy9wcml2YXRlCi1lbmRpZgotCi1pbmNsdWRlICQoQlVJTERfU0hBUkVEX0xJQlJBUlkpCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGlicy9FR0wvZWdsLmNwcCBiL29wZW5nbC9saWJzL0VHTC9lZ2wuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlMzU3NzNlLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJzL0VHTC9lZ2wuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTM2MyArMCwwIEBACi0vKiAKLSAqKiBDb3B5cmlnaHQgMjAwNywgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoqCi0gKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotICoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0gKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotICoqCi0gKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSAqKgotICoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0gKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0gKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotICoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0gKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJHTExvZ2dlciIKLQotI2luY2x1ZGUgPGN0eXBlLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxkbGZjbi5oPgotCi0jaW5jbHVkZSA8c3lzL2lvY3RsLmg+Ci0KLSNpZiBIQVZFX0FORFJPSURfT1MKLSNpbmNsdWRlIDxsaW51eC9hbmRyb2lkX3BtZW0uaD4KLSNlbmRpZgotCi0jaW5jbHVkZSA8RUdML2VnbC5oPgotI2luY2x1ZGUgPEVHTC9lZ2xleHQuaD4KLSNpbmNsdWRlIDxHTEVTL2dsLmg+Ci0jaW5jbHVkZSA8R0xFUy9nbGV4dC5oPgotCi0jaW5jbHVkZSA8Y3V0aWxzL2xvZy5oPgotI2luY2x1ZGUgPGN1dGlscy9hdG9taWMuaD4KLSNpbmNsdWRlIDxjdXRpbHMvcHJvcGVydGllcy5oPgotI2luY2x1ZGUgPGN1dGlscy9tZW1vcnkuaD4KLQotI2luY2x1ZGUgPHV0aWxzL1JlZkJhc2UuaD4KLQotI2luY2x1ZGUgImhvb2tzLmgiCi0jaW5jbHVkZSAiZWdsX2ltcGwuaCIKLQotCi0jZGVmaW5lIE1BS0VfQ09ORklHKF9pbXBsLCBfaW5kZXgpICAoKEVHTENvbmZpZykoKChfaW1wbCk8PDI0KSB8IChfaW5kZXgpKSkKLSNkZWZpbmUgc2V0RXJyb3IoX2UsIF9yKSBzZXRFcnJvckV0YyhfX0ZVTkNUSU9OX18sIF9fTElORV9fLCBfZSwgX3IpCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI2RlZmluZSBWRVJTSU9OX01JTk9SIDEKLSNkZWZpbmUgVkVSU0lPTl9NQUpPUiA0Ci1zdGF0aWMgY2hhciBjb25zdCAqIGNvbnN0IGdWZW5kb3JTdHJpbmcgICAgID0gIkFuZHJvaWQiOwotc3RhdGljIGNoYXIgY29uc3QgKiBjb25zdCBnVmVyc2lvblN0cmluZyAgICA9ICIxLjMxIEFuZHJvaWQgTUVUQS1FR0wiOwotc3RhdGljIGNoYXIgY29uc3QgKiBjb25zdCBnQ2xpZW50QXBpU3RyaW5nICA9ICJPcGVuR0wgRVMiOwotc3RhdGljIGNoYXIgY29uc3QgKiBjb25zdCBnRXh0ZW5zaW9uU3RyaW5nICA9ICIiOwotCi10ZW1wbGF0ZSA8aW50IE1BR0lDPgotc3RydWN0IGVnbF9vYmplY3RfdAotewotICAgIGVnbF9vYmplY3RfdCgpIDogbWFnaWMoTUFHSUMpIHsgfQotICAgIH5lZ2xfb2JqZWN0X3QoKSB7IG1hZ2ljID0gMDsgfQotICAgIGJvb2wgaXNWYWxpZCgpIGNvbnN0IHsgcmV0dXJuIG1hZ2ljID09IE1BR0lDOyB9Ci1wcml2YXRlOgotICAgIHVpbnQzMl90ICAgIG1hZ2ljOwotfTsKLQotc3RydWN0IGVnbF9kaXNwbGF5X3QgOiBwdWJsaWMgZWdsX29iamVjdF90PCdfZHB5Jz4KLXsKLSAgICBFR0xEaXNwbGF5ICBkcHlzWzJdOwotICAgIEVHTENvbmZpZyogIGNvbmZpZ3NbMl07Ci0gICAgRUdMaW50ICAgICAgbnVtQ29uZmlnc1syXTsKLSAgICBFR0xpbnQgICAgICBudW1Ub3RhbENvbmZpZ3M7Ci0gICAgY2hhciBjb25zdCogZXh0ZW5zaW9uc1N0cmluZzsKLSAgICB2b2xhdGlsZSBpbnQzMl90IHJlZnM7Ci0gICAgc3RydWN0IHN0cmluZ3NfdCB7Ci0gICAgICAgIGNoYXIgY29uc3QgKiB2ZW5kb3I7Ci0gICAgICAgIGNoYXIgY29uc3QgKiB2ZXJzaW9uOwotICAgICAgICBjaGFyIGNvbnN0ICogY2xpZW50QXBpOwotICAgICAgICBjaGFyIGNvbnN0ICogZXh0ZW5zaW9uczsKLSAgICB9OwotICAgIHN0cmluZ3NfdCAgIHF1ZXJ5U3RyaW5nWzJdOwotfTsKLQotc3RydWN0IGVnbF9zdXJmYWNlX3QgOiBwdWJsaWMgZWdsX29iamVjdF90PCdfc3JmJz4KLXsKLSAgICBlZ2xfc3VyZmFjZV90KEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIHN1cmZhY2UsCi0gICAgICAgICAgICBOYXRpdmVXaW5kb3dUeXBlIHdpbmRvdywgaW50IGltcGwsIGVnbF9jb25uZWN0aW9uX3QgY29uc3QqIGNueCkgCi0gICAgOiBkcHkoZHB5KSwgc3VyZmFjZShzdXJmYWNlKSwgd2luZG93KHdpbmRvdyksIGltcGwoaW1wbCksIGNueChjbngpCi0gICAgewotICAgICAgICAvLyBOT1RFOiB3aW5kb3cgbXVzdCBiZSBpbmNSZWYnZWQgYW5kIGNvbm5lY3RlZCBhbHJlYWR5Ci0gICAgfQotICAgIH5lZ2xfc3VyZmFjZV90KCkgewotICAgICAgICBpZiAod2luZG93KSB7Ci0gICAgICAgICAgICBpZiAod2luZG93LT5kaXNjb25uZWN0KQotICAgICAgICAgICAgICAgIHdpbmRvdy0+ZGlzY29ubmVjdCh3aW5kb3cpOwotICAgICAgICAgICAgd2luZG93LT5kZWNSZWYod2luZG93KTsKLSAgICAgICAgfQotICAgIH0KLSAgICBFR0xEaXNwbGF5ICAgICAgICAgICAgICAgICAgZHB5OwotICAgIEVHTFN1cmZhY2UgICAgICAgICAgICAgICAgICBzdXJmYWNlOwotICAgIE5hdGl2ZVdpbmRvd1R5cGUgICAgICAgICAgICB3aW5kb3c7Ci0gICAgaW50ICAgICAgICAgICAgICAgICAgICAgICAgIGltcGw7Ci0gICAgZWdsX2Nvbm5lY3Rpb25fdCBjb25zdCogICAgIGNueDsKLX07Ci0KLXN0cnVjdCBlZ2xfY29udGV4dF90IDogcHVibGljIGVnbF9vYmplY3RfdDwnX2N0eCc+Ci17Ci0gICAgZWdsX2NvbnRleHRfdChFR0xEaXNwbGF5IGRweSwgRUdMQ29udGV4dCBjb250ZXh0LAotICAgICAgICAgICAgaW50IGltcGwsIGVnbF9jb25uZWN0aW9uX3QgY29uc3QqIGNueCkgCi0gICAgOiBkcHkoZHB5KSwgY29udGV4dChjb250ZXh0KSwgcmVhZCgwKSwgZHJhdygwKSwgaW1wbChpbXBsKSwgY254KGNueCkKLSAgICB7Ci0gICAgfQotICAgIEVHTERpc3BsYXkgICAgICAgICAgICAgICAgICBkcHk7Ci0gICAgRUdMQ29udGV4dCAgICAgICAgICAgICAgICAgIGNvbnRleHQ7Ci0gICAgRUdMU3VyZmFjZSAgICAgICAgICAgICAgICAgIHJlYWQ7Ci0gICAgRUdMU3VyZmFjZSAgICAgICAgICAgICAgICAgIGRyYXc7Ci0gICAgaW50ICAgICAgICAgICAgICAgICAgICAgICAgIGltcGw7Ci0gICAgZWdsX2Nvbm5lY3Rpb25fdCBjb25zdCogICAgIGNueDsKLX07Ci0KLXN0cnVjdCB0bHNfdAotewotICAgIHRsc190KCkgOiBlcnJvcihFR0xfU1VDQ0VTUyksIGN0eCgwKSB7IH0KLSAgICBFR0xpbnQgICAgICBlcnJvcjsKLSAgICBFR0xDb250ZXh0ICBjdHg7Ci19OwotCi1zdGF0aWMgdm9pZCBnbF91bmltcGxlbWVudGVkKCkgewotICAgIExPR0UoImNhbGxlZCB1bmltcGxlbWVudGVkIE9wZW5HTCBFUyBBUEkiKTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gR0wgLyBFR0wgaG9va3MKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotI3VuZGVmIEdMX0VOVFJZCi0jdW5kZWYgRUdMX0VOVFJZCi0jZGVmaW5lIEdMX0VOVFJZKF9yLCBfYXBpLCAuLi4pICNfYXBpLAotI2RlZmluZSBFR0xfRU5UUlkoX3IsIF9hcGksIC4uLikgI19hcGksCi0KLXN0YXRpYyBjaGFyIGNvbnN0ICogY29uc3QgZ2xfbmFtZXNbXSA9IHsKLSAgICAjaW5jbHVkZSAiZ2xfZW50cmllcy5pbiIKLSAgICBOVUxMCi19OwotCi1zdGF0aWMgY2hhciBjb25zdCAqIGNvbnN0IGVnbF9uYW1lc1tdID0gewotICAgICNpbmNsdWRlICJlZ2xfZW50cmllcy5pbiIKLSAgICBOVUxMCi19OwotCi0jdW5kZWYgR0xfRU5UUlkKLSN1bmRlZiBFR0xfRU5UUlkKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1lZ2xfY29ubmVjdGlvbl90IGdFR0xJbXBsWzJdOwotc3RhdGljIGVnbF9kaXNwbGF5X3QgZ0Rpc3BsYXlbTlVNX0RJU1BMQVlTXTsKLXN0YXRpYyBwdGhyZWFkX211dGV4X3QgZ1RocmVhZExvY2FsU3RvcmFnZUtleU11dGV4ID0gUFRIUkVBRF9NVVRFWF9JTklUSUFMSVpFUjsKLXN0YXRpYyBwdGhyZWFkX2tleV90IGdFR0xUaHJlYWRMb2NhbFN0b3JhZ2VLZXkgPSAtMTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1nbF9ob29rc190IGdIb29rc1tJTVBMX05VTV9JTVBMRU1FTlRBVElPTlNdOwotcHRocmVhZF9rZXlfdCBnR0xXcmFwcGVyS2V5ID0gLTE7Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotc3RhdGljIF9fYXR0cmlidXRlX18oKG5vaW5saW5lKSkKLWNvbnN0IGNoYXIgKmVnbF9zdHJlcnJvcihFR0xpbnQgZXJyKQotewotICAgIHN3aXRjaCAoZXJyKXsKLSAgICAgICAgY2FzZSBFR0xfU1VDQ0VTUzogICAgICAgICAgICAgICByZXR1cm4gIkVHTF9TVUNDRVNTIjsKLSAgICAgICAgY2FzZSBFR0xfTk9UX0lOSVRJQUxJWkVEOiAgICAgICByZXR1cm4gIkVHTF9OT1RfSU5JVElBTElaRUQiOwotICAgICAgICBjYXNlIEVHTF9CQURfQUNDRVNTOiAgICAgICAgICAgIHJldHVybiAiRUdMX0JBRF9BQ0NFU1MiOwotICAgICAgICBjYXNlIEVHTF9CQURfQUxMT0M6ICAgICAgICAgICAgIHJldHVybiAiRUdMX0JBRF9BTExPQyI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9BVFRSSUJVVEU6ICAgICAgICAgcmV0dXJuICJFR0xfQkFEX0FUVFJJQlVURSI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9DT05GSUc6ICAgICAgICAgICAgcmV0dXJuICJFR0xfQkFEX0NPTkZJRyI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9DT05URVhUOiAgICAgICAgICAgcmV0dXJuICJFR0xfQkFEX0NPTlRFWFQiOwotICAgICAgICBjYXNlIEVHTF9CQURfQ1VSUkVOVF9TVVJGQUNFOiAgIHJldHVybiAiRUdMX0JBRF9DVVJSRU5UX1NVUkZBQ0UiOwotICAgICAgICBjYXNlIEVHTF9CQURfRElTUExBWTogICAgICAgICAgIHJldHVybiAiRUdMX0JBRF9ESVNQTEFZIjsKLSAgICAgICAgY2FzZSBFR0xfQkFEX01BVENIOiAgICAgICAgICAgICByZXR1cm4gIkVHTF9CQURfTUFUQ0giOwotICAgICAgICBjYXNlIEVHTF9CQURfTkFUSVZFX1BJWE1BUDogICAgIHJldHVybiAiRUdMX0JBRF9OQVRJVkVfUElYTUFQIjsKLSAgICAgICAgY2FzZSBFR0xfQkFEX05BVElWRV9XSU5ET1c6ICAgICByZXR1cm4gIkVHTF9CQURfTkFUSVZFX1dJTkRPVyI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9QQVJBTUVURVI6ICAgICAgICAgcmV0dXJuICJFR0xfQkFEX1BBUkFNRVRFUiI7Ci0gICAgICAgIGNhc2UgRUdMX0JBRF9TVVJGQUNFOiAgICAgICAgICAgcmV0dXJuICJFR0xfQkFEX1NVUkZBQ0UiOwotICAgICAgICBjYXNlIEVHTF9DT05URVhUX0xPU1Q6ICAgICAgICAgIHJldHVybiAiRUdMX0NPTlRFWFRfTE9TVCI7Ci0gICAgICAgIGRlZmF1bHQ6IHJldHVybiAiVU5LTk9XTiI7Ci0gICAgfQotfQotCi1zdGF0aWMgX19hdHRyaWJ1dGVfXygobm9pbmxpbmUpKQotdm9pZCBjbGVhclRMUygpIHsKLSAgICBpZiAoZ0VHTFRocmVhZExvY2FsU3RvcmFnZUtleSAhPSAtMSkgewotICAgICAgICB0bHNfdCogdGxzID0gKHRsc190KilwdGhyZWFkX2dldHNwZWNpZmljKGdFR0xUaHJlYWRMb2NhbFN0b3JhZ2VLZXkpOwotICAgICAgICBpZiAodGxzKSB7Ci0gICAgICAgICAgICBkZWxldGUgdGxzOwotICAgICAgICAgICAgcHRocmVhZF9zZXRzcGVjaWZpYyhnRUdMVGhyZWFkTG9jYWxTdG9yYWdlS2V5LCAwKTsKLSAgICAgICAgfQotICAgIH0KLX0KLQotc3RhdGljIHRsc190KiBnZXRUTFMoKQotewotICAgIHRsc190KiB0bHMgPSAodGxzX3QqKXB0aHJlYWRfZ2V0c3BlY2lmaWMoZ0VHTFRocmVhZExvY2FsU3RvcmFnZUtleSk7Ci0gICAgaWYgKHRscyA9PSAwKSB7Ci0gICAgICAgIHRscyA9IG5ldyB0bHNfdDsKLSAgICAgICAgcHRocmVhZF9zZXRzcGVjaWZpYyhnRUdMVGhyZWFkTG9jYWxTdG9yYWdlS2V5LCB0bHMpOwotICAgIH0KLSAgICByZXR1cm4gdGxzOwotfQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUPgotc3RhdGljIF9fYXR0cmlidXRlX18oKG5vaW5saW5lKSkKLVQgc2V0RXJyb3JFdGMoY29uc3QgY2hhciogY2FsbGVyLCBpbnQgbGluZSwgRUdMaW50IGVycm9yLCBUIHJldHVyblZhbHVlKSB7Ci0gICAgaWYgKGdFR0xUaHJlYWRMb2NhbFN0b3JhZ2VLZXkgPT0gLTEpIHsKLSAgICAgICAgcHRocmVhZF9tdXRleF9sb2NrKCZnVGhyZWFkTG9jYWxTdG9yYWdlS2V5TXV0ZXgpOwotICAgICAgICBpZiAoZ0VHTFRocmVhZExvY2FsU3RvcmFnZUtleSA9PSAtMSkKLSAgICAgICAgICAgIHB0aHJlYWRfa2V5X2NyZWF0ZSgmZ0VHTFRocmVhZExvY2FsU3RvcmFnZUtleSwgTlVMTCk7Ci0gICAgICAgIHB0aHJlYWRfbXV0ZXhfdW5sb2NrKCZnVGhyZWFkTG9jYWxTdG9yYWdlS2V5TXV0ZXgpOwotICAgIH0KLSAgICB0bHNfdCogdGxzID0gZ2V0VExTKCk7Ci0gICAgaWYgKHRscy0+ZXJyb3IgIT0gZXJyb3IpIHsKLSAgICAgICAgTE9HRSgiJXM6JWQgZXJyb3IgJXggKCVzKSIsIGNhbGxlciwgbGluZSwgZXJyb3IsIGVnbF9zdHJlcnJvcihlcnJvcikpOwotICAgICAgICB0bHMtPmVycm9yID0gZXJyb3I7Ci0gICAgfQotICAgIHJldHVybiByZXR1cm5WYWx1ZTsKLX0KLQotc3RhdGljIF9fYXR0cmlidXRlX18oKG5vaW5saW5lKSkKLUdMaW50IGdldEVycm9yKCkgewotICAgIGlmIChnRUdMVGhyZWFkTG9jYWxTdG9yYWdlS2V5ID09IC0xKQotICAgICAgICByZXR1cm4gRUdMX1NVQ0NFU1M7Ci0gICAgdGxzX3QqIHRscyA9ICh0bHNfdCopcHRocmVhZF9nZXRzcGVjaWZpYyhnRUdMVGhyZWFkTG9jYWxTdG9yYWdlS2V5KTsKLSAgICBpZiAoIXRscykgcmV0dXJuIEVHTF9TVUNDRVNTOwotICAgIEdMaW50IGVycm9yID0gdGxzLT5lcnJvcjsKLSAgICB0bHMtPmVycm9yID0gRUdMX1NVQ0NFU1M7Ci0gICAgcmV0dXJuIGVycm9yOwotfQotCi1zdGF0aWMgX19hdHRyaWJ1dGVfXygobm9pbmxpbmUpKQotdm9pZCBzZXRDb250ZXh0KEVHTENvbnRleHQgY3R4KSB7Ci0gICAgaWYgKGdFR0xUaHJlYWRMb2NhbFN0b3JhZ2VLZXkgPT0gLTEpIHsKLSAgICAgICAgcHRocmVhZF9tdXRleF9sb2NrKCZnVGhyZWFkTG9jYWxTdG9yYWdlS2V5TXV0ZXgpOwotICAgICAgICBpZiAoZ0VHTFRocmVhZExvY2FsU3RvcmFnZUtleSA9PSAtMSkKLSAgICAgICAgICAgIHB0aHJlYWRfa2V5X2NyZWF0ZSgmZ0VHTFRocmVhZExvY2FsU3RvcmFnZUtleSwgTlVMTCk7Ci0gICAgICAgIHB0aHJlYWRfbXV0ZXhfdW5sb2NrKCZnVGhyZWFkTG9jYWxTdG9yYWdlS2V5TXV0ZXgpOwotICAgIH0KLSAgICB0bHNfdCogdGxzID0gZ2V0VExTKCk7Ci0gICAgdGxzLT5jdHggPSBjdHg7Ci19Ci0KLXN0YXRpYyBfX2F0dHJpYnV0ZV9fKChub2lubGluZSkpCi1FR0xDb250ZXh0IGdldENvbnRleHQoKSB7Ci0gICAgaWYgKGdFR0xUaHJlYWRMb2NhbFN0b3JhZ2VLZXkgPT0gLTEpCi0gICAgICAgIHJldHVybiBFR0xfTk9fQ09OVEVYVDsKLSAgICB0bHNfdCogdGxzID0gKHRsc190KilwdGhyZWFkX2dldHNwZWNpZmljKGdFR0xUaHJlYWRMb2NhbFN0b3JhZ2VLZXkpOwotICAgIGlmICghdGxzKSByZXR1cm4gRUdMX05PX0NPTlRFWFQ7Ci0gICAgcmV0dXJuIHRscy0+Y3R4OwotfQotCi0KLS8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotY2xhc3MgSVN1cmZhY2VDb21wb3NlcjsKLWNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBnZXRTdXJmYWNlRmxpbmdlcigpOwotcmVxdWVzdF9ncHVfdCogZ3B1X2FjcXVpcmUodm9pZCogdXNlcik7Ci1pbnQgZ3B1X3JlbGVhc2Uodm9pZCosIHJlcXVlc3RfZ3B1X3QqIGdwdSk7Ci0KLXN0YXRpYyBfX2F0dHJpYnV0ZV9fKChub2lubGluZSkpCi12b2lkICpsb2FkX2RyaXZlcihjb25zdCBjaGFyKiBkcml2ZXIsIGdsX2hvb2tzX3QqIGhvb2tzKQotewotICAgIHZvaWQqIGRzbyA9IGRsb3Blbihkcml2ZXIsIFJUTERfTk9XIHwgUlRMRF9MT0NBTCk7Ci0gICAgTE9HRV9JRighZHNvLAotICAgICAgICAgICAgImNvdWxkbid0IGxvYWQgPCVzPiBsaWJyYXJ5ICglcykiLAotICAgICAgICAgICAgZHJpdmVyLCBkbGVycm9yKCkpOwotCi0gICAgaWYgKGRzbykgewotICAgICAgICB2b2lkKiogY3VycjsKLSAgICAgICAgY2hhciBjb25zdCAqIGNvbnN0ICogYXBpOwotICAgICAgICBnbF9ob29rc190OjpnbF90KiBnbCA9ICZob29rcy0+Z2w7Ci0gICAgICAgIGN1cnIgPSAodm9pZCoqKWdsOwotICAgICAgICBhcGkgPSBnbF9uYW1lczsKLSAgICAgICAgd2hpbGUgKCphcGkpIHsKLSAgICAgICAgICAgIHZvaWQqIGYgPSBkbHN5bShkc28sICphcGkpOwotICAgICAgICAgICAgLy9MT0dEKCI8JXM+IEAgMHglcCIsICphcGksIGYpOwotICAgICAgICAgICAgaWYgKGYgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgIC8vTE9HVygiPCVzPiBub3QgZm91bmQgaW4gJXMiLCAqYXBpLCBkcml2ZXIpOwotICAgICAgICAgICAgICAgIGYgPSAodm9pZCopZ2xfdW5pbXBsZW1lbnRlZDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgICpjdXJyKysgPSBmOwotICAgICAgICAgICAgYXBpKys7Ci0gICAgICAgIH0KLSAgICAgICAgZ2xfaG9va3NfdDo6ZWdsX3QqIGVnbCA9ICZob29rcy0+ZWdsOwotICAgICAgICBjdXJyID0gKHZvaWQqKillZ2w7Ci0gICAgICAgIGFwaSA9IGVnbF9uYW1lczsKLSAgICAgICAgd2hpbGUgKCphcGkpIHsKLSAgICAgICAgICAgIHZvaWQqIGYgPSBkbHN5bShkc28sICphcGkpOwotICAgICAgICAgICAgaWYgKGYgPT0gTlVMTCkgewotICAgICAgICAgICAgICAgIC8vTE9HVygiPCVzPiBub3QgZm91bmQgaW4gJXMiLCAqYXBpLCBkcml2ZXIpOwotICAgICAgICAgICAgICAgIGYgPSAodm9pZCopMDsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgICpjdXJyKysgPSBmOwotICAgICAgICAgICAgYXBpKys7Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBob29rIHRoaXMgZHJpdmVyIHVwIHdpdGggc3VyZmFjZWZsaW5nZXIgaWYgbmVlZGVkCi0gICAgICAgIHJlZ2lzdGVyX2dwdV90IHJlZ2lzdGVyX2dwdSA9IAotICAgICAgICAgICAgKHJlZ2lzdGVyX2dwdV90KWRsc3ltKGRzbywgIm9lbV9yZWdpc3Rlcl9ncHUiKTsKLQotICAgICAgICBpZiAocmVnaXN0ZXJfZ3B1ICE9IE5VTEwpIHsKLSAgICAgICAgICAgIGlmIChnZXRTdXJmYWNlRmxpbmdlcigpICE9IDApIHsKLSAgICAgICAgICAgICAgICByZWdpc3Rlcl9ncHUoZHNvLCBncHVfYWNxdWlyZSwgZ3B1X3JlbGVhc2UpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBkc287Ci19Ci0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+Ci1zdGF0aWMgX19hdHRyaWJ1dGVfXygobm9pbmxpbmUpKQotaW50IGJpbmFyeVNlYXJjaCgKLSAgICAgICAgVCBjb25zdCBzb3J0ZWRBcnJheVtdLCBpbnQgZmlyc3QsIGludCBsYXN0LCBUIGtleSkKLXsKLSAgICB3aGlsZSAoZmlyc3QgPD0gbGFzdCkgewotICAgICAgICBpbnQgbWlkID0gKGZpcnN0ICsgbGFzdCkgLyAyOwotICAgICAgICBpZiAoa2V5ID4gc29ydGVkQXJyYXlbbWlkXSkgeyAKLSAgICAgICAgICAgIGZpcnN0ID0gbWlkICsgMTsKLSAgICAgICAgfSBlbHNlIGlmIChrZXkgPCBzb3J0ZWRBcnJheVttaWRdKSB7IAotICAgICAgICAgICAgbGFzdCA9IG1pZCAtIDE7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICByZXR1cm4gbWlkOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiAtMTsKLX0KLQotc3RhdGljIEVHTGludCBjb25maWdUb1VuaXF1ZUlkKGVnbF9kaXNwbGF5X3QgY29uc3QqIGRwLCBpbnQgaSwgaW50IGluZGV4KSAKLXsKLSAgICAvLyBOT1RFOiB0aGlzIG1hcHBpbmcgd29ya3Mgb25seSBpZiB3ZSBoYXZlIG5vIG1vcmUgdGhhbiB0d28gRUdMaW1wbAotICAgIHJldHVybiAoaT4wID8gZHAtPm51bUNvbmZpZ3NbMF0gOiAwKSArIGluZGV4OwotfQotCi1zdGF0aWMgdm9pZCB1bmlxdWVJZFRvQ29uZmlnKGVnbF9kaXNwbGF5X3QgY29uc3QqIGRwLCBFR0xpbnQgY29uZmlnSWQsCi0gICAgICAgIGludCYgaSwgaW50JiBpbmRleCkgCi17Ci0gICAgLy8gTk9URTogdGhpcyBtYXBwaW5nIHdvcmtzIG9ubHkgaWYgd2UgaGF2ZSBubyBtb3JlIHRoYW4gdHdvIEVHTGltcGwKLSAgICBzaXplX3QgbnVtQ29uZmlncyA9IGRwLT5udW1Db25maWdzWzBdOwotICAgIGkgPSBjb25maWdJZCAvIG51bUNvbmZpZ3M7Ci0gICAgaW5kZXggPSBjb25maWdJZCAlIG51bUNvbmZpZ3M7Ci19Ci0KLXN0YXRpYyBpbnQgY21wX2NvbmZpZ3MoY29uc3Qgdm9pZCogYSwgY29uc3Qgdm9pZCAqYikKLXsKLSAgICBFR0xDb25maWcgYzAgPSAqKEVHTENvbmZpZyBjb25zdCAqKWE7Ci0gICAgRUdMQ29uZmlnIGMxID0gKihFR0xDb25maWcgY29uc3QgKiliOwotICAgIHJldHVybiBjMDxjMSA/IC0xIDogKGMwPmMxID8gMSA6IDApOwotfQotCi1zdHJ1Y3QgZXh0ZW50aW9uX21hcF90IHsKLSAgICBjb25zdCBjaGFyKiBuYW1lOwotICAgIF9fZWdsTXVzdENhc3RUb1Byb3BlckZ1bmN0aW9uUG9pbnRlclR5cGUgYWRkcmVzczsKLX07Ci0KLXN0YXRpYyBjb25zdCBleHRlbnRpb25fbWFwX3QgZ0V4dGVudGlvbk1hcFtdID0gewotfTsKLQotc3RhdGljIGV4dGVudGlvbl9tYXBfdCBnR0xFeHRlbnRpb25NYXBbTUFYX05VTUJFUl9PRl9HTF9FWFRFTlNJT05TXTsKLQotc3RhdGljIHZvaWQoKmZpbmRQcm9jQWRkcmVzcyhjb25zdCBjaGFyKiBuYW1lLAotICAgICAgICBjb25zdCBleHRlbnRpb25fbWFwX3QqIG1hcCwgc2l6ZV90IG4pKSgpIAotewotICAgIGZvciAodWludDMyX3QgaT0wIDsgaTxuIDsgaSsrKSB7Ci0gICAgICAgIGlmICghc3RyY21wKG5hbWUsIG1hcFtpXS5uYW1lKSkgewotICAgICAgICAgICAgcmV0dXJuIG1hcFtpXS5hZGRyZXNzOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBOVUxMOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0YXRpYyBpbnQgZ2xfY29udGV4dF9sb3N0KCkgewotICAgIHNldEdsVGhyZWFkU3BlY2lmaWMoJmdIb29rc1tJTVBMX0NPTlRFWFRfTE9TVF0pOwotICAgIHJldHVybiAwOwotfQotc3RhdGljIGludCBlZ2xfY29udGV4dF9sb3N0KCkgewotICAgIHNldEdsVGhyZWFkU3BlY2lmaWMoJmdIb29rc1tJTVBMX0NPTlRFWFRfTE9TVF0pOwotICAgIHJldHVybiBFR0xfRkFMU0U7Ci19Ci1zdGF0aWMgRUdMQm9vbGVhbiBlZ2xfY29udGV4dF9sb3N0X3N3YXBfYnVmZmVycyh2b2lkKiwgdm9pZCopIHsKLSAgICB1c2xlZXAoMTAwMDAwKTsgLy8gZG9uJ3QgdXNlIGFsbCB0aGUgQ1BVCi0gICAgc2V0R2xUaHJlYWRTcGVjaWZpYygmZ0hvb2tzW0lNUExfQ09OVEVYVF9MT1NUXSk7Ci0gICAgcmV0dXJuIEVHTF9GQUxTRTsKLX0KLXN0YXRpYyBHTGludCBlZ2xfY29udGV4dF9sb3N0X2dldF9lcnJvcigpIHsKLSAgICByZXR1cm4gRUdMX0NPTlRFWFRfTE9TVDsKLX0KLXN0YXRpYyBpbnQgZXh0X2NvbnRleHRfbG9zdCgpIHsKLSAgICByZXR1cm4gMDsKLX0KLQotc3RhdGljIHZvaWQgZ2xfbm9fY29udGV4dCgpIHsKLSAgICBMT0dFKCJjYWxsIHRvIE9wZW5HTCBFUyBBUEkgd2l0aCBubyBjdXJyZW50IGNvbnRleHQiKTsKLX0KLXN0YXRpYyB2b2lkIGVhcmx5X2VnbF9pbml0KHZvaWQpIAotewotI2lmICFVU0VfRkFTVF9UTFNfS0VZCi0gICAgcHRocmVhZF9rZXlfY3JlYXRlKCZnR0xXcmFwcGVyS2V5LCBOVUxMKTsKLSNlbmRpZgotICAgIHVpbnQzMl90IGFkZHIgPSAodWludDMyX3QpKCh2b2lkKilnbF9ub19jb250ZXh0KTsKLSAgICBhbmRyb2lkX21lbXNldDMyKAotICAgICAgICAgICAgKHVpbnQzMl90Kikodm9pZCopJmdIb29rc1tJTVBMX05PX0NPTlRFWFRdLCAKLSAgICAgICAgICAgIGFkZHIsIAotICAgICAgICAgICAgc2l6ZW9mKGdIb29rc1tJTVBMX05PX0NPTlRFWFRdKSk7Ci0gICAgc2V0R2xUaHJlYWRTcGVjaWZpYygmZ0hvb2tzW0lNUExfTk9fQ09OVEVYVF0pOwotfQotCi1zdGF0aWMgcHRocmVhZF9vbmNlX3Qgb25jZV9jb250cm9sID0gUFRIUkVBRF9PTkNFX0lOSVQ7Ci1zdGF0aWMgaW50IHNFYXJseUluaXRTdGF0ZSA9IHB0aHJlYWRfb25jZSgmb25jZV9jb250cm9sLCAmZWFybHlfZWdsX2luaXQpOwotCi0KLXN0YXRpYyBpbmxpbmUKLWVnbF9kaXNwbGF5X3QqIGdldF9kaXNwbGF5KEVHTERpc3BsYXkgZHB5KQotewotICAgIHVpbnRwdHJfdCBpbmRleCA9IHVpbnRwdHJfdChkcHkpLTFVOwotICAgIHJldHVybiAoaW5kZXggPj0gTlVNX0RJU1BMQVlTKSA/IE5VTEwgOiAmZ0Rpc3BsYXlbaW5kZXhdOwotfQotCi1zdGF0aWMgaW5saW5lCi1lZ2xfc3VyZmFjZV90KiBnZXRfc3VyZmFjZShFR0xTdXJmYWNlIHN1cmZhY2UpCi17Ci0gICAgZWdsX3N1cmZhY2VfdCogcyA9IChlZ2xfc3VyZmFjZV90ICopc3VyZmFjZTsKLSAgICByZXR1cm4gczsKLX0KLQotc3RhdGljIGlubGluZQotZWdsX2NvbnRleHRfdCogZ2V0X2NvbnRleHQoRUdMQ29udGV4dCBjb250ZXh0KQotewotICAgIGVnbF9jb250ZXh0X3QqIGMgPSAoZWdsX2NvbnRleHRfdCAqKWNvbnRleHQ7Ci0gICAgcmV0dXJuIGM7Ci19Ci0KLXN0YXRpYyBlZ2xfY29ubmVjdGlvbl90KiB2YWxpZGF0ZV9kaXNwbGF5X2NvbmZpZygKLSAgICAgICAgRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgIGVnbF9kaXNwbGF5X3QgY29uc3QqJiBkcCwgaW50JiBpbXBsLCBpbnQmIGluZGV4KQotewotICAgIGRwID0gZ2V0X2Rpc3BsYXkoZHB5KTsKLSAgICBpZiAoIWRwKSByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9ESVNQTEFZLCAoZWdsX2Nvbm5lY3Rpb25fdCopTlVMTCk7Ci0KLSAgICBpbXBsID0gdWludHB0cl90KGNvbmZpZyk+PjI0OwotICAgIGlmICh1aW50MzJfdChpbXBsKSA+PSAyKSB7Ci0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0NPTkZJRywgKGVnbF9jb25uZWN0aW9uX3QqKU5VTEwpOwotICAgIH0gCi0gICAgaW5kZXggPSB1aW50cHRyX3QoY29uZmlnKSAmIDB4RkZGRkZGOwotICAgIGlmIChpbmRleCA+PSBkcC0+bnVtQ29uZmlnc1tpbXBsXSkgewotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9DT05GSUcsIChlZ2xfY29ubmVjdGlvbl90KilOVUxMKTsKLSAgICB9Ci0gICAgZWdsX2Nvbm5lY3Rpb25fdCogY29uc3QgY254ID0gJmdFR0xJbXBsW2ltcGxdOwotICAgIGlmIChjbngtPmRzbyA9PSAwKSB7Ci0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0NPTkZJRywgKGVnbF9jb25uZWN0aW9uX3QqKU5VTEwpOwotICAgIH0KLSAgICByZXR1cm4gY254OwotfQotCi1zdGF0aWMgRUdMQm9vbGVhbiB2YWxpZGF0ZV9kaXNwbGF5X2NvbnRleHQoRUdMRGlzcGxheSBkcHksIEVHTENvbnRleHQgY3R4KQotewotICAgIGlmICgodWludHB0cl90KGRweSktMVUpID49IE5VTV9ESVNQTEFZUykKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICBpZiAoIWdldF9kaXNwbGF5KGRweSktPmlzVmFsaWQoKSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICBpZiAoIWN0eCkgLy8gVE9ETzogbWFrZSBzdXJlIGNvbnRleHQgaXMgYSB2YWxpZCBvYmplY3QKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfQ09OVEVYVCwgRUdMX0ZBTFNFKTsKLSAgICBpZiAoIWdldF9jb250ZXh0KGN0eCktPmlzVmFsaWQoKSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfQ09OVEVYVCwgRUdMX0ZBTFNFKTsKLSAgICByZXR1cm4gRUdMX1RSVUU7Ci19Ci0KLXN0YXRpYyBFR0xCb29sZWFuIHZhbGlkYXRlX2Rpc3BsYXlfc3VyZmFjZShFR0xEaXNwbGF5IGRweSwgRUdMU3VyZmFjZSBzdXJmYWNlKQotewotICAgIGlmICgodWludHB0cl90KGRweSktMVUpID49IE5VTV9ESVNQTEFZUykKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICBpZiAoIWdldF9kaXNwbGF5KGRweSktPmlzVmFsaWQoKSkKLSAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLSAgICBpZiAoIXN1cmZhY2UpIC8vIFRPRE86IG1ha2Ugc3VyZSBzdXJmYWNlIGlzIGEgdmFsaWQgb2JqZWN0Ci0gICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX1NVUkZBQ0UsIEVHTF9GQUxTRSk7Ci0gICAgaWYgKCFnZXRfc3VyZmFjZShzdXJmYWNlKS0+aXNWYWxpZCgpKQotICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9TVVJGQUNFLCBFR0xfRkFMU0UpOwotICAgIHJldHVybiBFR0xfVFJVRTsKLX0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotfTsgLy8gbmFtZXNwYWNlIGFuZHJvaWQKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotdXNpbmcgbmFtZXNwYWNlIGFuZHJvaWQ7Ci0KLUVHTERpc3BsYXkgZWdsR2V0RGlzcGxheShOYXRpdmVEaXNwbGF5VHlwZSBkaXNwbGF5KQotewotICAgIGlmIChzRWFybHlJbml0U3RhdGUpIHsKLSAgICAgICAgcmV0dXJuIEVHTF9OT19ESVNQTEFZOwotICAgIH0KLQotICAgIHVpbnQzMl90IGluZGV4ID0gdWludDMyX3QoZGlzcGxheSk7Ci0gICAgaWYgKGluZGV4ID49IE5VTV9ESVNQTEFZUykgewotICAgICAgICByZXR1cm4gRUdMX05PX0RJU1BMQVk7Ci0gICAgfQotICAgIAotICAgIEVHTERpc3BsYXkgZHB5ID0gRUdMRGlzcGxheSh1aW50cHRyX3QoZGlzcGxheSkgKyAxTFUpOwotICAgIGVnbF9kaXNwbGF5X3QqIGQgPSAmZ0Rpc3BsYXlbaW5kZXhdOwotICAgICAgICAKLSAgICAvLyBkeW5hbWljYWxseSBsb2FkIGFsbCBvdXIgRUdMIGltcGxlbWVudGF0aW9ucyBmb3IgdGhhdCBkaXNwbGF5Ci0gICAgLy8gYW5kIGNhbGwgaW50byB0aGUgcmVhbCBlZ2xHZXRHaXNwbGF5KCkKLSAgICBlZ2xfY29ubmVjdGlvbl90KiBjbnggPSAmZ0VHTEltcGxbSU1QTF9TT0ZUV0FSRV07Ci0gICAgaWYgKGNueC0+ZHNvID09IDApIHsKLSAgICAgICAgY254LT5ob29rcyA9ICZnSG9va3NbSU1QTF9TT0ZUV0FSRV07Ci0gICAgICAgIGNueC0+ZHNvID0gbG9hZF9kcml2ZXIoImxpYmFnbC5zbyIsIGNueC0+aG9va3MpOwotICAgIH0KLSAgICBpZiAoY254LT5kc28gJiYgZC0+ZHB5c1tJTVBMX1NPRlRXQVJFXT09RUdMX05PX0RJU1BMQVkpIHsKLSAgICAgICAgZC0+ZHB5c1tJTVBMX1NPRlRXQVJFXSA9IGNueC0+aG9va3MtPmVnbC5lZ2xHZXREaXNwbGF5KGRpc3BsYXkpOwotICAgICAgICBMT0dFX0lGKGQtPmRweXNbSU1QTF9TT0ZUV0FSRV09PUVHTF9OT19ESVNQTEFZLAotICAgICAgICAgICAgICAgICJObyBFR0xEaXNwbGF5IGZvciBzb2Z0d2FyZSBFR0whIik7Ci0gICAgfQotCi0gICAgY254ID0gJmdFR0xJbXBsW0lNUExfSEFSRFdBUkVdOwotICAgIGlmIChjbngtPmRzbyA9PSAwICYmIGNueC0+dW5hdmFpbGFibGUgPT0gMCkgewotICAgICAgICBjaGFyIHZhbHVlW1BST1BFUlRZX1ZBTFVFX01BWF07Ci0gICAgICAgIHByb3BlcnR5X2dldCgiZGVidWcuZWdsLmh3IiwgdmFsdWUsICIxIik7Ci0gICAgICAgIGlmIChhdG9pKHZhbHVlKSAhPSAwKSB7Ci0gICAgICAgICAgICBjbngtPmhvb2tzID0gJmdIb29rc1tJTVBMX0hBUkRXQVJFXTsKLSAgICAgICAgICAgIGNueC0+ZHNvID0gbG9hZF9kcml2ZXIoImxpYmhnbC5zbyIsIGNueC0+aG9va3MpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgTE9HRCgiM0QgaGFyZHdhcmUgYWNjZWxlcmF0aW9uIGlzIGRpc2FibGVkIik7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgaWYgKGNueC0+ZHNvICYmIGQtPmRweXNbSU1QTF9IQVJEV0FSRV09PUVHTF9OT19ESVNQTEFZKSB7Ci0gICAgICAgIGFuZHJvaWRfbWVtc2V0MzIoCi0gICAgICAgICAgICAgICAgKHVpbnQzMl90Kikodm9pZCopJmdIb29rc1tJTVBMX0NPTlRFWFRfTE9TVF0uZ2wsCi0gICAgICAgICAgICAgICAgKHVpbnQzMl90KSgodm9pZCopZ2xfY29udGV4dF9sb3N0KSwKLSAgICAgICAgICAgICAgICBzaXplb2YoZ0hvb2tzW0lNUExfQ09OVEVYVF9MT1NUXS5nbCkpOwotICAgICAgICBhbmRyb2lkX21lbXNldDMyKAotICAgICAgICAgICAgICAgICh1aW50MzJfdCopKHZvaWQqKSZnSG9va3NbSU1QTF9DT05URVhUX0xPU1RdLmVnbCwKLSAgICAgICAgICAgICAgICAodWludDMyX3QpKCh2b2lkKillZ2xfY29udGV4dF9sb3N0KSwKLSAgICAgICAgICAgICAgICBzaXplb2YoZ0hvb2tzW0lNUExfQ09OVEVYVF9MT1NUXS5lZ2wpKTsKLSAgICAgICAgYW5kcm9pZF9tZW1zZXQzMigKLSAgICAgICAgICAgICAgICAodWludDMyX3QqKSh2b2lkKikmZ0hvb2tzW0lNUExfQ09OVEVYVF9MT1NUXS5leHQsCi0gICAgICAgICAgICAgICAgKHVpbnQzMl90KSgodm9pZCopZXh0X2NvbnRleHRfbG9zdCksCi0gICAgICAgICAgICAgICAgc2l6ZW9mKGdIb29rc1tJTVBMX0NPTlRFWFRfTE9TVF0uZXh0KSk7Ci0KLSAgICAgICAgZ0hvb2tzW0lNUExfQ09OVEVYVF9MT1NUXS5lZ2wuZWdsU3dhcEJ1ZmZlcnMgPQotICAgICAgICAgICAgICAgIGVnbF9jb250ZXh0X2xvc3Rfc3dhcF9idWZmZXJzOwotICAgICAgICAKLSAgICAgICAgZ0hvb2tzW0lNUExfQ09OVEVYVF9MT1NUXS5lZ2wuZWdsR2V0RXJyb3IgPQotICAgICAgICAgICAgICAgIGVnbF9jb250ZXh0X2xvc3RfZ2V0X2Vycm9yOwotCi0gICAgICAgIGdIb29rc1tJTVBMX0NPTlRFWFRfTE9TVF0uZWdsLmVnbFRlcm1pbmF0ZSA9Ci0gICAgICAgICAgICAgICAgZ0hvb2tzW0lNUExfSEFSRFdBUkVdLmVnbC5lZ2xUZXJtaW5hdGU7Ci0gICAgICAgIAotICAgICAgICBkLT5kcHlzW0lNUExfSEFSRFdBUkVdID0gY254LT5ob29rcy0+ZWdsLmVnbEdldERpc3BsYXkoZGlzcGxheSk7Ci0gICAgICAgIGlmIChkLT5kcHlzW0lNUExfSEFSRFdBUkVdID09IEVHTF9OT19ESVNQTEFZKSB7Ci0gICAgICAgICAgICBMT0dFKCJoL3cgYWNjZWxlcmF0ZWQgZWdsR2V0RGlzcGxheSgpIGZhaWxlZCAoJXMpIiwKLSAgICAgICAgICAgICAgICAgICAgZWdsX3N0cmVycm9yKGNueC0+aG9va3MtPmVnbC5lZ2xHZXRFcnJvcigpKSk7Ci0gICAgICAgICAgICBkbGNsb3NlKCh2b2lkKiljbngtPmRzbyk7Ci0gICAgICAgICAgICBjbngtPmRzbyA9IDA7Ci0gICAgICAgICAgICAvLyBpbiBjYXNlIG9mIGZhaWx1cmUsIHdlIHdhbnQgdG8gbWFrZSBzdXJlIHdlIGRvbid0IHRyeSBhZ2FpbgotICAgICAgICAgICAgLy8gYXMgaXQncyBleHBlbnNpdmUuCi0gICAgICAgICAgICBjbngtPnVuYXZhaWxhYmxlID0gMTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIHJldHVybiBkcHk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIEluaXRpYWxpemF0aW9uCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUVHTEJvb2xlYW4gZWdsSW5pdGlhbGl6ZShFR0xEaXNwbGF5IGRweSwgRUdMaW50ICptYWpvciwgRUdMaW50ICptaW5vcikKLXsKLSAgICBlZ2xfZGlzcGxheV90ICogY29uc3QgZHAgPSBnZXRfZGlzcGxheShkcHkpOwotICAgIGlmICghZHApIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9GQUxTRSk7Ci0KLSAgICBpZiAoYW5kcm9pZF9hdG9taWNfaW5jKCZkcC0+cmVmcykgPiAwKSB7Ci0gICAgICAgIGlmIChtYWpvciAhPSBOVUxMKSAqbWFqb3IgPSBWRVJTSU9OX01BSk9SOwotICAgICAgICBpZiAobWlub3IgIT0gTlVMTCkgKm1pbm9yID0gVkVSU0lPTl9NSU5PUjsKLSAgICAgICAgcmV0dXJuIEVHTF9UUlVFOwotICAgIH0KLSAgICAKLSAgICBzZXRHbFRocmVhZFNwZWNpZmljKCZnSG9va3NbSU1QTF9OT19DT05URVhUXSk7Ci0gICAgCi0gICAgLy8gaW5pdGlhbGl6ZSBlYWNoIEVHTCBhbmQKLSAgICAvLyBidWlsZCBvdXIgb3duIGV4dGVuc2lvbiBzdHJpbmcgZmlyc3QsIGJhc2VkIG9uIHRoZSBleHRlbnNpb24gd2Uga25vdwotICAgIC8vIGFuZCB0aGUgZXh0ZW5zaW9uIHN1cHBvcnRlZCBieSBvdXIgY2xpZW50IGltcGxlbWVudGF0aW9uCi0gICAgZHAtPmV4dGVuc2lvbnNTdHJpbmcgPSBzdHJkdXAoZ0V4dGVuc2lvblN0cmluZyk7Ci0gICAgZm9yIChpbnQgaT0wIDsgaTwyIDsgaSsrKSB7Ci0gICAgICAgIGVnbF9jb25uZWN0aW9uX3QqIGNvbnN0IGNueCA9ICZnRUdMSW1wbFtpXTsKLSAgICAgICAgY254LT5tYWpvciA9IC0xOwotICAgICAgICBjbngtPm1pbm9yID0gLTE7Ci0gICAgICAgIGlmICghY254LT5kc28pIAotICAgICAgICAgICAgY29udGludWU7Ci0KLSAgICAgICAgaWYgKGNueC0+aG9va3MtPmVnbC5lZ2xJbml0aWFsaXplKAotICAgICAgICAgICAgICAgIGRwLT5kcHlzW2ldLCAmY254LT5tYWpvciwgJmNueC0+bWlub3IpKSB7Ci0KLSAgICAgICAgICAgIC8vTE9HRCgiaW5pdGlhbGl6ZWQgJWQgZHB5PSVwLCB2ZXI9JWQuJWQsIGNueD0lcCIsCi0gICAgICAgICAgICAvLyAgICAgICAgaSwgZHAtPmRweXNbaV0sIGNueC0+bWFqb3IsIGNueC0+bWlub3IsIGNueCk7Ci0KLSAgICAgICAgICAgIC8vIGdldCB0aGUgcXVlcnktc3RyaW5ncyBmb3IgdGhpcyBkaXNwbGF5IGZvciBlYWNoIGltcGxlbWVudGF0aW9uCi0gICAgICAgICAgICBkcC0+cXVlcnlTdHJpbmdbaV0udmVuZG9yID0KLSAgICAgICAgICAgICAgICBjbngtPmhvb2tzLT5lZ2wuZWdsUXVlcnlTdHJpbmcoZHAtPmRweXNbaV0sIEVHTF9WRU5ET1IpOwotICAgICAgICAgICAgZHAtPnF1ZXJ5U3RyaW5nW2ldLnZlcnNpb24gPQotICAgICAgICAgICAgICAgIGNueC0+aG9va3MtPmVnbC5lZ2xRdWVyeVN0cmluZyhkcC0+ZHB5c1tpXSwgRUdMX1ZFUlNJT04pOwotICAgICAgICAgICAgZHAtPnF1ZXJ5U3RyaW5nW2ldLmV4dGVuc2lvbnMgPSBzdHJkdXAoCi0gICAgICAgICAgICAgICAgICAgIGNueC0+aG9va3MtPmVnbC5lZ2xRdWVyeVN0cmluZyhkcC0+ZHB5c1tpXSwgRUdMX0VYVEVOU0lPTlMpKTsKLSAgICAgICAgICAgIGRwLT5xdWVyeVN0cmluZ1tpXS5jbGllbnRBcGkgPQotICAgICAgICAgICAgICAgIGNueC0+aG9va3MtPmVnbC5lZ2xRdWVyeVN0cmluZyhkcC0+ZHB5c1tpXSwgRUdMX0NMSUVOVF9BUElTKTsKLQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgTE9HRCgiJWQ6IGVnbEluaXRpYWxpemUoKSBmYWlsZWQgKCVzKSIsIAotICAgICAgICAgICAgICAgICAgICBpLCBlZ2xfc3RyZXJyb3IoY254LT5ob29rcy0+ZWdsLmVnbEdldEVycm9yKCkpKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIEVHTEJvb2xlYW4gcmVzID0gRUdMX0ZBTFNFOwotICAgIGZvciAoaW50IGk9MCA7IGk8MiA7IGkrKykgewotICAgICAgICBlZ2xfY29ubmVjdGlvbl90KiBjb25zdCBjbnggPSAmZ0VHTEltcGxbaV07Ci0gICAgICAgIGlmIChjbngtPmRzbyAmJiBjbngtPm1ham9yPj0wICYmIGNueC0+bWlub3I+PTApIHsKLSAgICAgICAgICAgIEVHTGludCBuOwotICAgICAgICAgICAgaWYgKGNueC0+aG9va3MtPmVnbC5lZ2xHZXRDb25maWdzKGRwLT5kcHlzW2ldLCAwLCAwLCAmbikpIHsKLSAgICAgICAgICAgICAgICBkcC0+Y29uZmlnc1tpXSA9IChFR0xDb25maWcqKW1hbGxvYyhzaXplb2YoRUdMQ29uZmlnKSpuKTsKLSAgICAgICAgICAgICAgICBpZiAoZHAtPmNvbmZpZ3NbaV0pIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGNueC0+aG9va3MtPmVnbC5lZ2xHZXRDb25maWdzKAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRwLT5kcHlzW2ldLCBkcC0+Y29uZmlnc1tpXSwgbiwgJmRwLT5udW1Db25maWdzW2ldKSkKLSAgICAgICAgICAgICAgICAgICAgewotICAgICAgICAgICAgICAgICAgICAgICAgLy8gc29ydCB0aGUgY29uZmlndXJhdGlvbnMgc28gd2UgY2FuIGRvIGJpbmFyeSBzZWFyY2hlcwotICAgICAgICAgICAgICAgICAgICAgICAgcXNvcnQoICBkcC0+Y29uZmlnc1tpXSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZHAtPm51bUNvbmZpZ3NbaV0sCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpemVvZihFR0xDb25maWcpLCBjbXBfY29uZmlncyk7Ci0KLSAgICAgICAgICAgICAgICAgICAgICAgIGRwLT5udW1Ub3RhbENvbmZpZ3MgKz0gbjsKLSAgICAgICAgICAgICAgICAgICAgICAgIHJlcyA9IEVHTF9UUlVFOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotCi0gICAgaWYgKHJlcyA9PSBFR0xfVFJVRSkgewotICAgICAgICBpZiAobWFqb3IgIT0gTlVMTCkgKm1ham9yID0gMTsKLSAgICAgICAgaWYgKG1pbm9yICE9IE5VTEwpICptaW5vciA9IDI7Ci0gICAgICAgIHJldHVybiBFR0xfVFJVRTsKLSAgICB9Ci0gICAgcmV0dXJuIHNldEVycm9yKEVHTF9OT1RfSU5JVElBTElaRUQsIEVHTF9GQUxTRSk7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsVGVybWluYXRlKEVHTERpc3BsYXkgZHB5KQotewotICAgIGVnbF9kaXNwbGF5X3QqIGNvbnN0IGRwID0gZ2V0X2Rpc3BsYXkoZHB5KTsKLSAgICBpZiAoIWRwKSByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9ESVNQTEFZLCBFR0xfRkFMU0UpOwotICAgIGlmIChhbmRyb2lkX2F0b21pY19kZWMoJmRwLT5yZWZzKSAhPSAxKQotICAgICAgICByZXR1cm4gRUdMX1RSVUU7Ci0gICAgICAgIAotICAgIEVHTEJvb2xlYW4gcmVzID0gRUdMX0ZBTFNFOwotICAgIGZvciAoaW50IGk9MCA7IGk8MiA7IGkrKykgewotICAgICAgICBlZ2xfY29ubmVjdGlvbl90KiBjb25zdCBjbnggPSAmZ0VHTEltcGxbaV07Ci0gICAgICAgIGlmIChjbngtPmRzbykgewotICAgICAgICAgICAgY254LT5ob29rcy0+ZWdsLmVnbFRlcm1pbmF0ZShkcC0+ZHB5c1tpXSk7Ci0gICAgICAgICAgICAKLSAgICAgICAgICAgIC8qIFJFVklTSVQ6IGl0J3MgdW5jbGVhciB3aGF0IHRvIGRvIGlmIGVnbFRlcm1pbmF0ZSgpIGZhaWxzLAotICAgICAgICAgICAgICogb24gb25lIGVuZCB3ZSBzaG91bGRuJ3QgY2FyZSwgb24gdGhlIG90aGVyIGVuZCBpZiBpdCBmYWlscwotICAgICAgICAgICAgICogaXQgbWlnaHQgbm90IGJlIHNhZmUgdG8gY2FsbCBkbGNsb3NlKCkgKHRoZXJlIGNvdWxkIGJlIHNvbWUKLSAgICAgICAgICAgICAqIHRocmVhZHMgYXJvdW5kKS4gKi8KLSAgICAgICAgICAgIAotICAgICAgICAgICAgZnJlZShkcC0+Y29uZmlnc1tpXSk7Ci0gICAgICAgICAgICBmcmVlKCh2b2lkKilkcC0+cXVlcnlTdHJpbmdbaV0uZXh0ZW5zaW9ucyk7Ci0gICAgICAgICAgICBkcC0+bnVtQ29uZmlnc1tpXSA9IDA7Ci0gICAgICAgICAgICBkcC0+ZHB5c1tpXSA9IEVHTF9OT19ESVNQTEFZOwotICAgICAgICAgICAgZGxjbG9zZSgodm9pZCopY254LT5kc28pOwotICAgICAgICAgICAgY254LT5kc28gPSAwOwotICAgICAgICAgICAgcmVzID0gRUdMX1RSVUU7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgZnJlZSgodm9pZCopZHAtPmV4dGVuc2lvbnNTdHJpbmcpOwotICAgIGRwLT5leHRlbnNpb25zU3RyaW5nID0gMDsKLSAgICBkcC0+bnVtVG90YWxDb25maWdzID0gMDsKLSAgICBjbGVhclRMUygpOwotICAgIHJldHVybiByZXM7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIGNvbmZpZ3VyYXRpb24KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotRUdMQm9vbGVhbiBlZ2xHZXRDb25maWdzKCAgIEVHTERpc3BsYXkgZHB5LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEVHTENvbmZpZyAqY29uZmlncywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBFR0xpbnQgY29uZmlnX3NpemUsIEVHTGludCAqbnVtX2NvbmZpZykKLXsKLSAgICBlZ2xfZGlzcGxheV90IGNvbnN0ICogY29uc3QgZHAgPSBnZXRfZGlzcGxheShkcHkpOwotICAgIGlmICghZHApIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9GQUxTRSk7Ci0KLSAgICBHTGludCBudW1Db25maWdzID0gZHAtPm51bVRvdGFsQ29uZmlnczsKLSAgICBpZiAoIWNvbmZpZ3MpIHsKLSAgICAgICAgKm51bV9jb25maWcgPSBudW1Db25maWdzOwotICAgICAgICByZXR1cm4gRUdMX1RSVUU7Ci0gICAgfQotICAgIEdMaW50IG4gPSAwOwotICAgIGZvciAoaW50IGo9MCA7IGo8MiA7IGorKykgewotICAgICAgICBmb3IgKGludCBpPTAgOyBpPGRwLT5udW1Db25maWdzW2pdICYmIGNvbmZpZ19zaXplIDsgaSsrKSB7Ci0gICAgICAgICAgICAqY29uZmlncysrID0gTUFLRV9DT05GSUcoaiwgaSk7Ci0gICAgICAgICAgICBjb25maWdfc2l6ZS0tOwotICAgICAgICAgICAgbisrOwotICAgICAgICB9Ci0gICAgfSAgICAKLSAgICAKLSAgICAqbnVtX2NvbmZpZyA9IG47Ci0gICAgcmV0dXJuIEVHTF9UUlVFOwotfQotCi1FR0xCb29sZWFuIGVnbENob29zZUNvbmZpZyggRUdMRGlzcGxheSBkcHksIGNvbnN0IEVHTGludCAqYXR0cmliX2xpc3QsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgRUdMQ29uZmlnICpjb25maWdzLCBFR0xpbnQgY29uZmlnX3NpemUsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgRUdMaW50ICpudW1fY29uZmlnKQotewotICAgIGVnbF9kaXNwbGF5X3QgY29uc3QgKiBjb25zdCBkcCA9IGdldF9kaXNwbGF5KGRweSk7Ci0gICAgaWYgKCFkcCkgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLQotICAgIGlmIChjb25maWdzID09IDApIHsKLSAgICAgICAgKm51bV9jb25maWcgPSAwOwotICAgICAgICByZXR1cm4gRUdMX1RSVUU7Ci0gICAgfQotCi0gICAgRUdMaW50IG47Ci0gICAgRUdMQm9vbGVhbiByZXMgPSBFR0xfRkFMU0U7Ci0gICAgKm51bV9jb25maWcgPSAwOwotCi0gICAgCi0gICAgLy8gSXQgaXMgdW5mb3J0dW5hdGUsIGJ1dCB3ZSBuZWVkIHRvIHJlbWFwIHRoZSBFR0xfQ09ORklHX0lEcywgCi0gICAgLy8gdG8gZG8gIHRoaXMsIHdlIGhhdmUgdG8gZ28gdGhyb3VnaCB0aGUgYXR0cmliX2xpc3QgYXJyYXkgb25jZQotICAgIC8vIHRvIGZpZ3VyZSBvdXQgYm90aCBpdHMgc2l6ZSBhbmQgaWYgaXQgY29udGFpbnMgYW4gRUdMX0NPTkZJR19JRAotICAgIC8vIGtleS4gSWYgc28sIHRoZSBmdWxsIGFycmF5IGlzIGNvcGllZCBhbmQgcGF0Y2hlZC4KLSAgICAvLyBOT1RFOiB3ZSBhc3N1bWUgdGhhdCB0aGVyZSBjYW4gYmUgb25seSBvbmUgb2NjdXJyZW5jZQotICAgIC8vIG9mIEVHTF9DT05GSUdfSUQuCi0gICAgCi0gICAgRUdMaW50IHBhdGNoX2luZGV4ID0gLTE7Ci0gICAgR0xpbnQgYXR0cjsKLSAgICBzaXplX3Qgc2l6ZSA9IDA7Ci0gICAgd2hpbGUgKChhdHRyPWF0dHJpYl9saXN0W3NpemVdKSkgewotICAgICAgICBpZiAoYXR0ciA9PSBFR0xfQ09ORklHX0lEKQotICAgICAgICAgICAgcGF0Y2hfaW5kZXggPSBzaXplOwotICAgICAgICBzaXplICs9IDI7Ci0gICAgfQotICAgIGlmIChwYXRjaF9pbmRleCA+PSAwKSB7Ci0gICAgICAgIHNpemUgKz0gMjsgLy8gd2UgbmVlZCBjb3B5IHRoZSBzZW50aW5lbCBhcyB3ZWxsCi0gICAgICAgIEVHTGludCogbmV3X2xpc3QgPSAoRUdMaW50KiltYWxsb2Moc2l6ZSpzaXplb2YoRUdMaW50KSk7Ci0gICAgICAgIGlmIChuZXdfbGlzdCA9PSAwKQotICAgICAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfQUxMT0MsIEVHTF9GQUxTRSk7Ci0gICAgICAgIG1lbWNweShuZXdfbGlzdCwgYXR0cmliX2xpc3QsIHNpemUqc2l6ZW9mKEVHTGludCkpOwotCi0gICAgICAgIC8vIHBhdGNoIHRoZSByZXF1ZXN0ZWQgRUdMX0NPTkZJR19JRAotICAgICAgICBpbnQgaSwgaW5kZXg7Ci0gICAgICAgIEVHTGludCYgY29uZmlnSWQobmV3X2xpc3RbcGF0Y2hfaW5kZXgrMV0pOwotICAgICAgICB1bmlxdWVJZFRvQ29uZmlnKGRwLCBjb25maWdJZCwgaSwgaW5kZXgpOwotICAgICAgICAKLSAgICAgICAgZWdsX2Nvbm5lY3Rpb25fdCogY29uc3QgY254ID0gJmdFR0xJbXBsW2ldOwotICAgICAgICBpZiAoY254LT5kc28pIHsKLSAgICAgICAgICAgIGNueC0+aG9va3MtPmVnbC5lZ2xHZXRDb25maWdBdHRyaWIoCi0gICAgICAgICAgICAgICAgICAgIGRwLT5kcHlzW2ldLCBkcC0+Y29uZmlnc1tpXVtpbmRleF0sIAotICAgICAgICAgICAgICAgICAgICBFR0xfQ09ORklHX0lELCAmY29uZmlnSWQpOwotCi0gICAgICAgICAgICAvLyBhbmQgc3dpdGNoIHRvIHRoZSBuZXcgbGlzdAotICAgICAgICAgICAgYXR0cmliX2xpc3QgPSBjb25zdF9jYXN0PGNvbnN0IEVHTGludCAqPihuZXdfbGlzdCk7Ci0KLSAgICAgICAgICAgIC8vIEF0IHRoaXMgcG9pbnQsIHRoZSBvbmx5IGNvbmZpZ3VyYXRpb24gdGhhdCBjYW4gbWF0Y2ggaXMKLSAgICAgICAgICAgIC8vIGRwLT5jb25maWdzW2ldW2luZGV4XSwgaG93ZXZlciwgd2UgZG9uJ3Qga25vdyBpZiBpdCB3b3VsZCBiZQotICAgICAgICAgICAgLy8gcmVqZWN0ZWQgYmVjYXVzZSBvZiB0aGUgb3RoZXIgYXR0cmlidXRlcywgc28gd2UgZG8gaGF2ZSB0byBjYWxsCi0gICAgICAgICAgICAvLyBjbngtPmhvb2tzLT5lZ2wuZWdsQ2hvb3NlQ29uZmlnKCkgLS0gYnV0IHdlIGRvbid0IGhhdmUgdG8gbG9vcAotICAgICAgICAgICAgLy8gdGhyb3VnaCBhbGwgdGhlIEVHTGltcGxbXS4KLSAgICAgICAgICAgIC8vIFdlIGFsc28ga25vdyB3ZSBjYW4gb25seSBnZXQgYSBzaW5nbGUgY29uZmlnIGJhY2ssIGFuZCB3ZSBrbm93Ci0gICAgICAgICAgICAvLyB3aGljaCBvbmUuCi0KLSAgICAgICAgICAgIHJlcyA9IGNueC0+aG9va3MtPmVnbC5lZ2xDaG9vc2VDb25maWcoCi0gICAgICAgICAgICAgICAgICAgIGRwLT5kcHlzW2ldLCBhdHRyaWJfbGlzdCwgY29uZmlncywgY29uZmlnX3NpemUsICZuKTsKLSAgICAgICAgICAgIGlmIChyZXMgJiYgbj4wKSB7Ci0gICAgICAgICAgICAgICAgLy8gbiBoYXMgdG8gYmUgMCBvciAxLCBieSBjb25zdHJ1Y3Rpb24sIGFuZCB3ZSBhbHJlYWR5IGtub3cKLSAgICAgICAgICAgICAgICAvLyB3aGljaCBjb25maWcgaXQgd2lsbCByZXR1cm4gKHNpbmNlIHRoZXJlIGNhbiBiZSBvbmx5IG9uZSkuCi0gICAgICAgICAgICAgICAgY29uZmlnc1swXSA9IE1BS0VfQ09ORklHKGksIGluZGV4KTsKLSAgICAgICAgICAgICAgICAqbnVtX2NvbmZpZyA9IDE7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBmcmVlKGNvbnN0X2Nhc3Q8RUdMaW50ICo+KGF0dHJpYl9saXN0KSk7Ci0gICAgICAgIHJldHVybiByZXM7Ci0gICAgfQotCi0gICAgZm9yIChpbnQgaT0wIDsgaTwyIDsgaSsrKSB7Ci0gICAgICAgIGVnbF9jb25uZWN0aW9uX3QqIGNvbnN0IGNueCA9ICZnRUdMSW1wbFtpXTsKLSAgICAgICAgaWYgKGNueC0+ZHNvKSB7Ci0gICAgICAgICAgICBpZiAoY254LT5ob29rcy0+ZWdsLmVnbENob29zZUNvbmZpZygKLSAgICAgICAgICAgICAgICAgICAgZHAtPmRweXNbaV0sIGF0dHJpYl9saXN0LCBjb25maWdzLCBjb25maWdfc2l6ZSwgJm4pKSB7Ci0gICAgICAgICAgICAgICAgLy8gbm93IHdlIG5lZWQgdG8gY29udmVydCB0aGVzZSBjbGllbnQgRUdMQ29uZmlnIHRvIG91cgotICAgICAgICAgICAgICAgIC8vIGludGVybmFsIEVHTENvbmZpZyBmb3JtYXQuIFRoaXMgaXMgZG9uZSBpbiBPKG4gbG9nIG4pLgotICAgICAgICAgICAgICAgIGZvciAoaW50IGo9MCA7IGo8biA7IGorKykgewotICAgICAgICAgICAgICAgICAgICBpbnQgaW5kZXggPSBiaW5hcnlTZWFyY2g8RUdMQ29uZmlnPigKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkcC0+Y29uZmlnc1tpXSwgMCwgZHAtPm51bUNvbmZpZ3NbaV0tMSwgY29uZmlnc1tqXSk7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChpbmRleCA+PSAwKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb25maWdzW2pdID0gTUFLRV9DT05GSUcoaSwgaW5kZXgpOwotICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfQ09ORklHLCBFR0xfRkFMU0UpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGNvbmZpZ3MgKz0gbjsKLSAgICAgICAgICAgICAgICBjb25maWdfc2l6ZSAtPSBuOwotICAgICAgICAgICAgICAgICpudW1fY29uZmlnICs9IG47Ci0gICAgICAgICAgICAgICAgcmVzID0gRUdMX1RSVUU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIHJlczsKLX0KLQotRUdMQm9vbGVhbiBlZ2xHZXRDb25maWdBdHRyaWIoRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgIEVHTGludCBhdHRyaWJ1dGUsIEVHTGludCAqdmFsdWUpCi17Ci0gICAgZWdsX2Rpc3BsYXlfdCBjb25zdCogZHAgPSAwOwotICAgIGludCBpPTAsIGluZGV4PTA7Ci0gICAgZWdsX2Nvbm5lY3Rpb25fdCogY254ID0gdmFsaWRhdGVfZGlzcGxheV9jb25maWcoZHB5LCBjb25maWcsIGRwLCBpLCBpbmRleCk7Ci0gICAgaWYgKCFjbngpIHJldHVybiBFR0xfRkFMU0U7Ci0gICAgCi0gICAgaWYgKGF0dHJpYnV0ZSA9PSBFR0xfQ09ORklHX0lEKSB7Ci0gICAgICAgIC8vIEVHTF9DT05GSUdfSURzIG11c3QgYmUgdW5pcXVlLCBqdXN0IHVzZSB0aGUgb3JkZXIgb2YgdGhlIHNlbGVjdGVkCi0gICAgICAgIC8vIEVHTENvbmZpZy4KLSAgICAgICAgKnZhbHVlID0gY29uZmlnVG9VbmlxdWVJZChkcCwgaSwgaW5kZXgpOwotICAgICAgICByZXR1cm4gRUdMX1RSVUU7Ci0gICAgfQotICAgIHJldHVybiBjbngtPmhvb2tzLT5lZ2wuZWdsR2V0Q29uZmlnQXR0cmliKAotICAgICAgICAgICAgZHAtPmRweXNbaV0sIGRwLT5jb25maWdzW2ldW2luZGV4XSwgYXR0cmlidXRlLCB2YWx1ZSk7Ci19Ci0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIHN1cmZhY2VzCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLUVHTFN1cmZhY2UgZWdsQ3JlYXRlV2luZG93U3VyZmFjZSggIEVHTERpc3BsYXkgZHB5LCBFR0xDb25maWcgY29uZmlnLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTmF0aXZlV2luZG93VHlwZSB3aW5kb3csCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KQotewotICAgIGVnbF9kaXNwbGF5X3QgY29uc3QqIGRwID0gMDsKLSAgICBpbnQgaT0wLCBpbmRleD0wOwotICAgIGVnbF9jb25uZWN0aW9uX3QqIGNueCA9IHZhbGlkYXRlX2Rpc3BsYXlfY29uZmlnKGRweSwgY29uZmlnLCBkcCwgaSwgaW5kZXgpOwotICAgIGlmIChjbngpIHsKLSAgICAgICAgLy8gd2luZG93IG11c3QgYmUgY29ubmVjdGVkIHVwb24gY2FsbGluZyB1bmRlcmx5aW5nCi0gICAgICAgIC8vIGVnbENyZWF0ZVdpbmRvd1N1cmZhY2UKLSAgICAgICAgaWYgKHdpbmRvdykgewotICAgICAgICAgICAgd2luZG93LT5pbmNSZWYod2luZG93KTsKLSAgICAgICAgICAgIGlmICh3aW5kb3ctPmNvbm5lY3QpCi0gICAgICAgICAgICAgICAgd2luZG93LT5jb25uZWN0KHdpbmRvdyk7Ci0gICAgICAgIH0KLQotICAgICAgICBFR0xTdXJmYWNlIHN1cmZhY2UgPSBjbngtPmhvb2tzLT5lZ2wuZWdsQ3JlYXRlV2luZG93U3VyZmFjZSgKLSAgICAgICAgICAgICAgICBkcC0+ZHB5c1tpXSwgZHAtPmNvbmZpZ3NbaV1baW5kZXhdLCB3aW5kb3csIGF0dHJpYl9saXN0KTsgICAgICAgCi0gICAgICAgIGlmIChzdXJmYWNlICE9IEVHTF9OT19TVVJGQUNFKSB7Ci0gICAgICAgICAgICBlZ2xfc3VyZmFjZV90KiBzID0gbmV3IGVnbF9zdXJmYWNlX3QoZHB5LCBzdXJmYWNlLCB3aW5kb3csIGksIGNueCk7Ci0gICAgICAgICAgICByZXR1cm4gczsKLSAgICAgICAgfQotICAgICAgICAKLSAgICAgICAgLy8gc29tZXRoaW5nIHdlbnQgd3JvbmcsIGRpc2Nvbm5lY3QgYW5kIGZyZWUgd2luZG93Ci0gICAgICAgIC8vICh3aWxsIGRpc2Nvbm5lY3QoKSBhdXRvbWF0aWNhbGx5KQotICAgICAgICBpZiAod2luZG93KSB7Ci0gICAgICAgICAgICB3aW5kb3ctPmRlY1JlZih3aW5kb3cpOwotICAgICAgICB9ICAgICAgICAKLSAgICB9Ci0gICAgcmV0dXJuIEVHTF9OT19TVVJGQUNFOwotfQotCi1FR0xTdXJmYWNlIGVnbENyZWF0ZVBpeG1hcFN1cmZhY2UoICBFR0xEaXNwbGF5IGRweSwgRUdMQ29uZmlnIGNvbmZpZywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIE5hdGl2ZVBpeG1hcFR5cGUgcGl4bWFwLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgRUdMaW50ICphdHRyaWJfbGlzdCkKLXsKLSAgICBlZ2xfZGlzcGxheV90IGNvbnN0KiBkcCA9IDA7Ci0gICAgaW50IGk9MCwgaW5kZXg9MDsKLSAgICBlZ2xfY29ubmVjdGlvbl90KiBjbnggPSB2YWxpZGF0ZV9kaXNwbGF5X2NvbmZpZyhkcHksIGNvbmZpZywgZHAsIGksIGluZGV4KTsKLSAgICBpZiAoY254KSB7Ci0gICAgICAgIEVHTFN1cmZhY2Ugc3VyZmFjZSA9IGNueC0+aG9va3MtPmVnbC5lZ2xDcmVhdGVQaXhtYXBTdXJmYWNlKAotICAgICAgICAgICAgICAgIGRwLT5kcHlzW2ldLCBkcC0+Y29uZmlnc1tpXVtpbmRleF0sIHBpeG1hcCwgYXR0cmliX2xpc3QpOwotICAgICAgICBpZiAoc3VyZmFjZSAhPSBFR0xfTk9fU1VSRkFDRSkgewotICAgICAgICAgICAgZWdsX3N1cmZhY2VfdCogcyA9IG5ldyBlZ2xfc3VyZmFjZV90KGRweSwgc3VyZmFjZSwgTlVMTCwgaSwgY254KTsKLSAgICAgICAgICAgIHJldHVybiBzOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBFR0xfTk9fU1VSRkFDRTsKLX0KLQotRUdMU3VyZmFjZSBlZ2xDcmVhdGVQYnVmZmVyU3VyZmFjZSggRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KQotewotICAgIGVnbF9kaXNwbGF5X3QgY29uc3QqIGRwID0gMDsKLSAgICBpbnQgaT0wLCBpbmRleD0wOwotICAgIGVnbF9jb25uZWN0aW9uX3QqIGNueCA9IHZhbGlkYXRlX2Rpc3BsYXlfY29uZmlnKGRweSwgY29uZmlnLCBkcCwgaSwgaW5kZXgpOwotICAgIGlmIChjbngpIHsKLSAgICAgICAgRUdMU3VyZmFjZSBzdXJmYWNlID0gY254LT5ob29rcy0+ZWdsLmVnbENyZWF0ZVBidWZmZXJTdXJmYWNlKAotICAgICAgICAgICAgICAgIGRwLT5kcHlzW2ldLCBkcC0+Y29uZmlnc1tpXVtpbmRleF0sIGF0dHJpYl9saXN0KTsKLSAgICAgICAgaWYgKHN1cmZhY2UgIT0gRUdMX05PX1NVUkZBQ0UpIHsKLSAgICAgICAgICAgIGVnbF9zdXJmYWNlX3QqIHMgPSBuZXcgZWdsX3N1cmZhY2VfdChkcHksIHN1cmZhY2UsIE5VTEwsIGksIGNueCk7Ci0gICAgICAgICAgICByZXR1cm4gczsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gRUdMX05PX1NVUkZBQ0U7Ci19Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKLUVHTEJvb2xlYW4gZWdsRGVzdHJveVN1cmZhY2UoRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2Ugc3VyZmFjZSkKLXsKLSAgICBpZiAoIXZhbGlkYXRlX2Rpc3BsYXlfc3VyZmFjZShkcHksIHN1cmZhY2UpKQotICAgICAgICByZXR1cm4gRUdMX0ZBTFNFOyAgICAKLSAgICBlZ2xfZGlzcGxheV90IGNvbnN0ICogY29uc3QgZHAgPSBnZXRfZGlzcGxheShkcHkpOwotICAgIGVnbF9zdXJmYWNlX3QgY29uc3QgKiBjb25zdCBzID0gZ2V0X3N1cmZhY2Uoc3VyZmFjZSk7Ci0KLSAgICBFR0xCb29sZWFuIHJlc3VsdCA9IHMtPmNueC0+aG9va3MtPmVnbC5lZ2xEZXN0cm95U3VyZmFjZSgKLSAgICAgICAgICAgIGRwLT5kcHlzW3MtPmltcGxdLCBzLT5zdXJmYWNlKTsKLSAgICAKLSAgICBkZWxldGUgczsKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi1FR0xCb29sZWFuIGVnbFF1ZXJ5U3VyZmFjZSggRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2Ugc3VyZmFjZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBFR0xpbnQgYXR0cmlidXRlLCBFR0xpbnQgKnZhbHVlKQotewotICAgIGlmICghdmFsaWRhdGVfZGlzcGxheV9zdXJmYWNlKGRweSwgc3VyZmFjZSkpCi0gICAgICAgIHJldHVybiBFR0xfRkFMU0U7ICAgIAotICAgIGVnbF9kaXNwbGF5X3QgY29uc3QgKiBjb25zdCBkcCA9IGdldF9kaXNwbGF5KGRweSk7Ci0gICAgZWdsX3N1cmZhY2VfdCBjb25zdCAqIGNvbnN0IHMgPSBnZXRfc3VyZmFjZShzdXJmYWNlKTsKLQotICAgIHJldHVybiBzLT5jbngtPmhvb2tzLT5lZ2wuZWdsUXVlcnlTdXJmYWNlKAotICAgICAgICAgICAgZHAtPmRweXNbcy0+aW1wbF0sIHMtPnN1cmZhY2UsIGF0dHJpYnV0ZSwgdmFsdWUpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBjb250ZXh0ZXMKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotRUdMQ29udGV4dCBlZ2xDcmVhdGVDb250ZXh0KEVHTERpc3BsYXkgZHB5LCBFR0xDb25maWcgY29uZmlnLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEVHTENvbnRleHQgc2hhcmVfbGlzdCwgY29uc3QgRUdMaW50ICphdHRyaWJfbGlzdCkKLXsKLSAgICBlZ2xfZGlzcGxheV90IGNvbnN0KiBkcCA9IDA7Ci0gICAgaW50IGk9MCwgaW5kZXg9MDsKLSAgICBlZ2xfY29ubmVjdGlvbl90KiBjbnggPSB2YWxpZGF0ZV9kaXNwbGF5X2NvbmZpZyhkcHksIGNvbmZpZywgZHAsIGksIGluZGV4KTsKLSAgICBpZiAoY254KSB7Ci0gICAgICAgIEVHTENvbnRleHQgY29udGV4dCA9IGNueC0+aG9va3MtPmVnbC5lZ2xDcmVhdGVDb250ZXh0KAotICAgICAgICAgICAgICAgIGRwLT5kcHlzW2ldLCBkcC0+Y29uZmlnc1tpXVtpbmRleF0sIHNoYXJlX2xpc3QsIGF0dHJpYl9saXN0KTsKLSAgICAgICAgaWYgKGNvbnRleHQgIT0gRUdMX05PX0NPTlRFWFQpIHsKLSAgICAgICAgICAgIGVnbF9jb250ZXh0X3QqIGMgPSBuZXcgZWdsX2NvbnRleHRfdChkcHksIGNvbnRleHQsIGksIGNueCk7Ci0gICAgICAgICAgICByZXR1cm4gYzsKLSAgICAgICAgfQotICAgIH0KLSAgICByZXR1cm4gRUdMX05PX0NPTlRFWFQ7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsRGVzdHJveUNvbnRleHQoRUdMRGlzcGxheSBkcHksIEVHTENvbnRleHQgY3R4KQotewotICAgIGlmICghdmFsaWRhdGVfZGlzcGxheV9jb250ZXh0KGRweSwgY3R4KSkKLSAgICAgICAgcmV0dXJuIEVHTF9GQUxTRTsKLSAgICBlZ2xfZGlzcGxheV90IGNvbnN0ICogY29uc3QgZHAgPSBnZXRfZGlzcGxheShkcHkpOwotICAgIGVnbF9jb250ZXh0X3QgKiBjb25zdCBjID0gZ2V0X2NvbnRleHQoY3R4KTsKLSAgICBFR0xCb29sZWFuIHJlc3VsdCA9IGMtPmNueC0+aG9va3MtPmVnbC5lZ2xEZXN0cm95Q29udGV4dCgKLSAgICAgICAgICAgIGRwLT5kcHlzW2MtPmltcGxdLCBjLT5jb250ZXh0KTsKLSAgICBkZWxldGUgYzsKLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi1FR0xCb29sZWFuIGVnbE1ha2VDdXJyZW50KCAgRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2UgZHJhdywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBFR0xTdXJmYWNlIHJlYWQsIEVHTENvbnRleHQgY3R4KQotewotICAgIGVnbF9kaXNwbGF5X3QgY29uc3QgKiBjb25zdCBkcCA9IGdldF9kaXNwbGF5KGRweSk7Ci0gICAgaWYgKCFkcCkgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfRElTUExBWSwgRUdMX0ZBTFNFKTsKLQotICAgIGlmIChyZWFkID09IEVHTF9OT19TVVJGQUNFICYmIGRyYXcgID09IEVHTF9OT19TVVJGQUNFICYmCi0gICAgICAgICAgICBjdHggPT0gRUdMX05PX0NPTlRFWFQpIAotICAgIHsKLSAgICAgICAgRUdMQm9vbGVhbiByZXN1bHQgPSBFR0xfVFJVRTsKLSAgICAgICAgY3R4ID0gZ2V0Q29udGV4dCgpOwotICAgICAgICBpZiAoY3R4KSB7Ci0gICAgICAgICAgICBlZ2xfY29udGV4dF90ICogY29uc3QgYyA9IGdldF9jb250ZXh0KGN0eCk7Ci0gICAgICAgICAgICByZXN1bHQgPSBjLT5jbngtPmhvb2tzLT5lZ2wuZWdsTWFrZUN1cnJlbnQoZHAtPmRweXNbYy0+aW1wbF0sIDAsIDAsIDApOwotICAgICAgICAgICAgaWYgKHJlc3VsdCA9PSBFR0xfVFJVRSkgewotICAgICAgICAgICAgICAgIHNldEdsVGhyZWFkU3BlY2lmaWMoJmdIb29rc1tJTVBMX05PX0NPTlRFWFRdKTsKLSAgICAgICAgICAgICAgICBzZXRDb250ZXh0KEVHTF9OT19DT05URVhUKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgICAgICByZXR1cm4gcmVzdWx0OwotICAgIH0KLQotICAgIGlmICghdmFsaWRhdGVfZGlzcGxheV9jb250ZXh0KGRweSwgY3R4KSkKLSAgICAgICAgcmV0dXJuIEVHTF9GQUxTRTsgICAgCi0gICAgCi0gICAgZWdsX2NvbnRleHRfdCAqIGNvbnN0IGMgPSBnZXRfY29udGV4dChjdHgpOwotICAgIGlmIChkcmF3ICE9IEVHTF9OT19TVVJGQUNFKSB7Ci0gICAgICAgIGVnbF9zdXJmYWNlX3QgY29uc3QgKiBkID0gZ2V0X3N1cmZhY2UoZHJhdyk7Ci0gICAgICAgIGlmICghZCkgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfU1VSRkFDRSwgRUdMX0ZBTFNFKTsKLSAgICAgICAgaWYgKGQtPmltcGwgIT0gYy0+aW1wbCkKLSAgICAgICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX01BVENILCBFR0xfRkFMU0UpOwotICAgICAgICBkcmF3ID0gZC0+c3VyZmFjZTsKLSAgICB9Ci0gICAgaWYgKHJlYWQgIT0gRUdMX05PX1NVUkZBQ0UpIHsKLSAgICAgICAgZWdsX3N1cmZhY2VfdCBjb25zdCAqIHIgPSBnZXRfc3VyZmFjZShyZWFkKTsKLSAgICAgICAgaWYgKCFyKSByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9TVVJGQUNFLCBFR0xfRkFMU0UpOwotICAgICAgICBpZiAoci0+aW1wbCAhPSBjLT5pbXBsKQotICAgICAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfTUFUQ0gsIEVHTF9GQUxTRSk7Ci0gICAgICAgIHJlYWQgPSByLT5zdXJmYWNlOwotICAgIH0KLSAgICBFR0xCb29sZWFuIHJlc3VsdCA9IGMtPmNueC0+aG9va3MtPmVnbC5lZ2xNYWtlQ3VycmVudCgKLSAgICAgICAgICAgIGRwLT5kcHlzW2MtPmltcGxdLCBkcmF3LCByZWFkLCBjLT5jb250ZXh0KTsKLQotICAgIGlmIChyZXN1bHQgPT0gRUdMX1RSVUUpIHsKLSAgICAgICAgc2V0R2xUaHJlYWRTcGVjaWZpYyhjLT5jbngtPmhvb2tzKTsKLSAgICAgICAgc2V0Q29udGV4dChjdHgpOwotICAgICAgICBjLT5yZWFkID0gcmVhZDsKLSAgICAgICAgYy0+ZHJhdyA9IGRyYXc7Ci0gICAgfQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLQotRUdMQm9vbGVhbiBlZ2xRdWVyeUNvbnRleHQoIEVHTERpc3BsYXkgZHB5LCBFR0xDb250ZXh0IGN0eCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBFR0xpbnQgYXR0cmlidXRlLCBFR0xpbnQgKnZhbHVlKQotewotICAgIGlmICghdmFsaWRhdGVfZGlzcGxheV9jb250ZXh0KGRweSwgY3R4KSkKLSAgICAgICAgcmV0dXJuIEVHTF9GQUxTRTsgICAgCi0gICAgCi0gICAgZWdsX2Rpc3BsYXlfdCBjb25zdCAqIGNvbnN0IGRwID0gZ2V0X2Rpc3BsYXkoZHB5KTsKLSAgICBlZ2xfY29udGV4dF90ICogY29uc3QgYyA9IGdldF9jb250ZXh0KGN0eCk7Ci0KLSAgICByZXR1cm4gYy0+Y254LT5ob29rcy0+ZWdsLmVnbFF1ZXJ5Q29udGV4dCgKLSAgICAgICAgICAgIGRwLT5kcHlzW2MtPmltcGxdLCBjLT5jb250ZXh0LCBhdHRyaWJ1dGUsIHZhbHVlKTsKLX0KLQotRUdMQ29udGV4dCBlZ2xHZXRDdXJyZW50Q29udGV4dCh2b2lkKQotewotICAgIEVHTENvbnRleHQgY3R4ID0gZ2V0Q29udGV4dCgpOwotICAgIHJldHVybiBjdHg7Ci19Ci0KLUVHTFN1cmZhY2UgZWdsR2V0Q3VycmVudFN1cmZhY2UoRUdMaW50IHJlYWRkcmF3KQotewotICAgIEVHTENvbnRleHQgY3R4ID0gZ2V0Q29udGV4dCgpOwotICAgIGlmIChjdHgpIHsKLSAgICAgICAgZWdsX2NvbnRleHRfdCBjb25zdCAqIGNvbnN0IGMgPSBnZXRfY29udGV4dChjdHgpOwotICAgICAgICBpZiAoIWMpIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0NPTlRFWFQsIEVHTF9OT19TVVJGQUNFKTsKLSAgICAgICAgc3dpdGNoIChyZWFkZHJhdykgewotICAgICAgICAgICAgY2FzZSBFR0xfUkVBRDogcmV0dXJuIGMtPnJlYWQ7Ci0gICAgICAgICAgICBjYXNlIEVHTF9EUkFXOiByZXR1cm4gYy0+ZHJhdzsgICAgICAgICAgICAKLSAgICAgICAgICAgIGRlZmF1bHQ6IHJldHVybiBzZXRFcnJvcihFR0xfQkFEX1BBUkFNRVRFUiwgRUdMX05PX1NVUkZBQ0UpOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiBFR0xfTk9fU1VSRkFDRTsKLX0KLQotRUdMRGlzcGxheSBlZ2xHZXRDdXJyZW50RGlzcGxheSh2b2lkKQotewotICAgIEVHTENvbnRleHQgY3R4ID0gZ2V0Q29udGV4dCgpOwotICAgIGlmIChjdHgpIHsKLSAgICAgICAgZWdsX2NvbnRleHRfdCBjb25zdCAqIGNvbnN0IGMgPSBnZXRfY29udGV4dChjdHgpOwotICAgICAgICBpZiAoIWMpIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0NPTlRFWFQsIEVHTF9OT19TVVJGQUNFKTsKLSAgICAgICAgcmV0dXJuIGMtPmRweTsKLSAgICB9Ci0gICAgcmV0dXJuIEVHTF9OT19ESVNQTEFZOwotfQotCi1FR0xCb29sZWFuIGVnbFdhaXRHTCh2b2lkKQotewotICAgIEVHTEJvb2xlYW4gcmVzID0gRUdMX1RSVUU7Ci0gICAgRUdMQ29udGV4dCBjdHggPSBnZXRDb250ZXh0KCk7Ci0gICAgaWYgKGN0eCkgewotICAgICAgICBlZ2xfY29udGV4dF90IGNvbnN0ICogY29uc3QgYyA9IGdldF9jb250ZXh0KGN0eCk7Ci0gICAgICAgIGlmICghYykgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfQ09OVEVYVCwgRUdMX0ZBTFNFKTsKLSAgICAgICAgaWYgKHVpbnQzMl90KGMtPmltcGwpPj0yKQotICAgICAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfQ09OVEVYVCwgRUdMX0ZBTFNFKTsKLSAgICAgICAgZWdsX2Nvbm5lY3Rpb25fdCogY29uc3QgY254ID0gJmdFR0xJbXBsW2MtPmltcGxdOwotICAgICAgICBpZiAoIWNueC0+ZHNvKSAKLSAgICAgICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0NPTlRFWFQsIEVHTF9GQUxTRSk7Ci0gICAgICAgIHJlcyA9IGNueC0+aG9va3MtPmVnbC5lZ2xXYWl0R0woKTsKLSAgICB9Ci0gICAgcmV0dXJuIHJlczsKLX0KLQotRUdMQm9vbGVhbiBlZ2xXYWl0TmF0aXZlKEVHTGludCBlbmdpbmUpCi17Ci0gICAgRUdMQm9vbGVhbiByZXMgPSBFR0xfVFJVRTsKLSAgICBFR0xDb250ZXh0IGN0eCA9IGdldENvbnRleHQoKTsKLSAgICBpZiAoY3R4KSB7Ci0gICAgICAgIGVnbF9jb250ZXh0X3QgY29uc3QgKiBjb25zdCBjID0gZ2V0X2NvbnRleHQoY3R4KTsKLSAgICAgICAgaWYgKCFjKSByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9DT05URVhULCBFR0xfRkFMU0UpOwotICAgICAgICBpZiAodWludDMyX3QoYy0+aW1wbCk+PTIpCi0gICAgICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9DT05URVhULCBFR0xfRkFMU0UpOwotICAgICAgICBlZ2xfY29ubmVjdGlvbl90KiBjb25zdCBjbnggPSAmZ0VHTEltcGxbYy0+aW1wbF07Ci0gICAgICAgIGlmICghY254LT5kc28pIAotICAgICAgICAgICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfQ09OVEVYVCwgRUdMX0ZBTFNFKTsKLSAgICAgICAgcmVzID0gY254LT5ob29rcy0+ZWdsLmVnbFdhaXROYXRpdmUoZW5naW5lKTsKLSAgICB9Ci0gICAgcmV0dXJuIHJlczsKLX0KLQotRUdMaW50IGVnbEdldEVycm9yKHZvaWQpCi17Ci0gICAgRUdMaW50IHJlc3VsdCA9IEVHTF9TVUNDRVNTOwotICAgIGZvciAoaW50IGk9MCA7IGk8MiA7IGkrKykgewotICAgICAgICBFR0xpbnQgZXJyID0gRUdMX1NVQ0NFU1M7Ci0gICAgICAgIGVnbF9jb25uZWN0aW9uX3QqIGNvbnN0IGNueCA9ICZnRUdMSW1wbFtpXTsKLSAgICAgICAgaWYgKGNueC0+ZHNvKQotICAgICAgICAgICAgZXJyID0gY254LT5ob29rcy0+ZWdsLmVnbEdldEVycm9yKCk7Ci0gICAgICAgIGlmIChlcnIhPUVHTF9TVUNDRVNTICYmIHJlc3VsdD09RUdMX1NVQ0NFU1MpCi0gICAgICAgICAgICByZXN1bHQgPSBlcnI7Ci0gICAgfQotICAgIGlmIChyZXN1bHQgPT0gRUdMX1NVQ0NFU1MpCi0gICAgICAgIHJlc3VsdCA9IGdldEVycm9yKCk7Ci0gICAgcmV0dXJuIHJlc3VsdDsKLX0KLQotdm9pZCAoKmVnbEdldFByb2NBZGRyZXNzKGNvbnN0IGNoYXIgKnByb2NuYW1lKSkoKQotewotICAgIF9fZWdsTXVzdENhc3RUb1Byb3BlckZ1bmN0aW9uUG9pbnRlclR5cGUgYWRkcjsKLSAgICBhZGRyID0gZmluZFByb2NBZGRyZXNzKHByb2NuYW1lLCBnRXh0ZW50aW9uTWFwLCBORUxFTShnRXh0ZW50aW9uTWFwKSk7Ci0gICAgaWYgKGFkZHIpIHJldHVybiBhZGRyOwotCi0gICAgcmV0dXJuIE5VTEw7IC8vIFRPRE86IGZpbmlzaCBpbXBsZW1lbnRhdGlvbiBiZWxvdwotCi0gICAgYWRkciA9IGZpbmRQcm9jQWRkcmVzcyhwcm9jbmFtZSwgZ0dMRXh0ZW50aW9uTWFwLCBORUxFTShnR0xFeHRlbnRpb25NYXApKTsKLSAgICBpZiAoYWRkcikgcmV0dXJuIGFkZHI7Ci0gICAgCi0gICAgYWRkciA9IDA7Ci0gICAgaW50IHNsb3QgPSAtMTsKLSAgICBmb3IgKGludCBpPTAgOyBpPDIgOyBpKyspIHsKLSAgICAgICAgZWdsX2Nvbm5lY3Rpb25fdCogY29uc3QgY254ID0gJmdFR0xJbXBsW2ldOwotICAgICAgICBpZiAoY254LT5kc28pIHsKLSAgICAgICAgICAgIGlmIChjbngtPmhvb2tzLT5lZ2wuZWdsR2V0UHJvY0FkZHJlc3MpIHsKLSAgICAgICAgICAgICAgICBhZGRyID0gY254LT5ob29rcy0+ZWdsLmVnbEdldFByb2NBZGRyZXNzKHByb2NuYW1lKTsKLSAgICAgICAgICAgICAgICBpZiAoYWRkcikgewotICAgICAgICAgICAgICAgICAgICBpZiAoc2xvdCA9PSAtMSkgewotICAgICAgICAgICAgICAgICAgICAgICAgc2xvdCA9IDA7IC8vIFhYWDogZmluZCBmcmVlIHNsb3QKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChzbG90ID09IC0xKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYWRkciA9IDA7Ci0gICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7Ci0gICAgICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgY254LT5ob29rcy0+ZXh0LmV4dGVuc2lvbnNbc2xvdF0gPSBhZGRyOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICBpZiAoc2xvdCA+PSAwKSB7Ci0gICAgICAgIGFkZHIgPSAwOyAvLyBYWFg6IGFkZHJlc3Mgb2Ygc3R1YiAnc2xvdCcKLSAgICAgICAgZ0dMRXh0ZW50aW9uTWFwW3Nsb3RdLm5hbWUgPSBzdHJkdXAocHJvY25hbWUpOwotICAgICAgICBnR0xFeHRlbnRpb25NYXBbc2xvdF0uYWRkcmVzcyA9IGFkZHI7Ci0gICAgfQotICAgIAotICAgIHJldHVybiBhZGRyOwotCi0gICAgCi0gICAgLyoKLSAgICAgKiAgVE9ETzogRm9yIE9wZW5HTCBFUyBleHRlbnNpb25zLCB3ZSBtdXN0IGdlbmVyYXRlIGEgc3R1YgotICAgICAqICB0aGF0IGxvb2tzIGxpa2UKLSAgICAgKiAgICAgIG1vdiAgICAgcjEyLCAjMHhGRkZGMEZGRgotICAgICAqICAgICAgbGRyICAgICByMTIsIFtyMTIsICMtMTVdCi0gICAgICogICAgICBsZHIgICAgIHIxMiwgW3IxMiwgI1RMU19TTE9UX09QRU5HTF9BUEkqNF0KLSAgICAgKiAgICAgIG1vdiAgICAgcjEyLCBbcjEyLCAjYXBpX29mZnNldF0KLSAgICAgKiAgICAgIGxkcm5lICAgcGMsIHIxMgotICAgICAqICAgICAgbW92ICAgICBwYywgI3Vuc3VwcG9ydGVkX2V4dGVuc2lvbgotICAgICAqIAotICAgICAqICBhbmQgd3JpdGUgdGhlIGFkZHJlc3Mgb2YgdGhlIGV4dGVuc2lvbiBpbiAqYWxsKgotICAgICAqICBnbF9ob29rc190OjpnbF9leHRfdCBhdCBvZmZzZXQgImFwaV9vZmZzZXQiIGZyb20gZ2xfaG9va3NfdAotICAgICAqIAotICAgICAqLwotfQotCi1FR0xCb29sZWFuIGVnbFN3YXBCdWZmZXJzKEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIGRyYXcpCi17Ci0gICAgaWYgKCF2YWxpZGF0ZV9kaXNwbGF5X3N1cmZhY2UoZHB5LCBkcmF3KSkKLSAgICAgICAgcmV0dXJuIEVHTF9GQUxTRTsgICAgCi0gICAgZWdsX2Rpc3BsYXlfdCBjb25zdCAqIGNvbnN0IGRwID0gZ2V0X2Rpc3BsYXkoZHB5KTsKLSAgICBlZ2xfc3VyZmFjZV90IGNvbnN0ICogY29uc3QgcyA9IGdldF9zdXJmYWNlKGRyYXcpOwotICAgIHJldHVybiBzLT5jbngtPmhvb2tzLT5lZ2wuZWdsU3dhcEJ1ZmZlcnMoZHAtPmRweXNbcy0+aW1wbF0sIHMtPnN1cmZhY2UpOwotfQotCi1FR0xCb29sZWFuIGVnbENvcHlCdWZmZXJzKCAgRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2Ugc3VyZmFjZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBOYXRpdmVQaXhtYXBUeXBlIHRhcmdldCkKLXsKLSAgICBpZiAoIXZhbGlkYXRlX2Rpc3BsYXlfc3VyZmFjZShkcHksIHN1cmZhY2UpKQotICAgICAgICByZXR1cm4gRUdMX0ZBTFNFOyAgICAKLSAgICBlZ2xfZGlzcGxheV90IGNvbnN0ICogY29uc3QgZHAgPSBnZXRfZGlzcGxheShkcHkpOwotICAgIGVnbF9zdXJmYWNlX3QgY29uc3QgKiBjb25zdCBzID0gZ2V0X3N1cmZhY2Uoc3VyZmFjZSk7Ci0gICAgcmV0dXJuIHMtPmNueC0+aG9va3MtPmVnbC5lZ2xDb3B5QnVmZmVycygKLSAgICAgICAgICAgIGRwLT5kcHlzW3MtPmltcGxdLCBzLT5zdXJmYWNlLCB0YXJnZXQpOwotfQotCi1jb25zdCBjaGFyKiBlZ2xRdWVyeVN0cmluZyhFR0xEaXNwbGF5IGRweSwgRUdMaW50IG5hbWUpCi17Ci0gICAgZWdsX2Rpc3BsYXlfdCBjb25zdCAqIGNvbnN0IGRwID0gZ2V0X2Rpc3BsYXkoZHB5KTsKLSAgICBzd2l0Y2ggKG5hbWUpIHsKLSAgICAgICAgY2FzZSBFR0xfVkVORE9SOgotICAgICAgICAgICAgcmV0dXJuIGdWZW5kb3JTdHJpbmc7Ci0gICAgICAgIGNhc2UgRUdMX1ZFUlNJT046Ci0gICAgICAgICAgICByZXR1cm4gZ1ZlcnNpb25TdHJpbmc7Ci0gICAgICAgIGNhc2UgRUdMX0VYVEVOU0lPTlM6Ci0gICAgICAgICAgICByZXR1cm4gZ0V4dGVuc2lvblN0cmluZzsKLSAgICAgICAgY2FzZSBFR0xfQ0xJRU5UX0FQSVM6Ci0gICAgICAgICAgICByZXR1cm4gZ0NsaWVudEFwaVN0cmluZzsKLSAgICB9Ci0gICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfUEFSQU1FVEVSLCAoY29uc3QgY2hhciAqKTApOwotfQotCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS8vIEVHTCAxLjEKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotRUdMQm9vbGVhbiBlZ2xTdXJmYWNlQXR0cmliKAotICAgICAgICBFR0xEaXNwbGF5IGRweSwgRUdMU3VyZmFjZSBzdXJmYWNlLCBFR0xpbnQgYXR0cmlidXRlLCBFR0xpbnQgdmFsdWUpCi17Ci0gICAgaWYgKCF2YWxpZGF0ZV9kaXNwbGF5X3N1cmZhY2UoZHB5LCBzdXJmYWNlKSkKLSAgICAgICAgcmV0dXJuIEVHTF9GQUxTRTsgICAgCi0gICAgZWdsX2Rpc3BsYXlfdCBjb25zdCAqIGNvbnN0IGRwID0gZ2V0X2Rpc3BsYXkoZHB5KTsKLSAgICBlZ2xfc3VyZmFjZV90IGNvbnN0ICogY29uc3QgcyA9IGdldF9zdXJmYWNlKHN1cmZhY2UpOwotICAgIGlmIChzLT5jbngtPmhvb2tzLT5lZ2wuZWdsU3VyZmFjZUF0dHJpYikgewotICAgICAgICByZXR1cm4gcy0+Y254LT5ob29rcy0+ZWdsLmVnbFN1cmZhY2VBdHRyaWIoCi0gICAgICAgICAgICAgICAgZHAtPmRweXNbcy0+aW1wbF0sIHMtPnN1cmZhY2UsIGF0dHJpYnV0ZSwgdmFsdWUpOwotICAgIH0KLSAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9TVVJGQUNFLCBFR0xfRkFMU0UpOwotfQotCi1FR0xCb29sZWFuIGVnbEJpbmRUZXhJbWFnZSgKLSAgICAgICAgRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2Ugc3VyZmFjZSwgRUdMaW50IGJ1ZmZlcikKLXsKLSAgICBpZiAoIXZhbGlkYXRlX2Rpc3BsYXlfc3VyZmFjZShkcHksIHN1cmZhY2UpKQotICAgICAgICByZXR1cm4gRUdMX0ZBTFNFOyAgICAKLSAgICBlZ2xfZGlzcGxheV90IGNvbnN0ICogY29uc3QgZHAgPSBnZXRfZGlzcGxheShkcHkpOwotICAgIGVnbF9zdXJmYWNlX3QgY29uc3QgKiBjb25zdCBzID0gZ2V0X3N1cmZhY2Uoc3VyZmFjZSk7Ci0gICAgaWYgKHMtPmNueC0+aG9va3MtPmVnbC5lZ2xCaW5kVGV4SW1hZ2UpIHsKLSAgICAgICAgcmV0dXJuIHMtPmNueC0+aG9va3MtPmVnbC5lZ2xCaW5kVGV4SW1hZ2UoCi0gICAgICAgICAgICAgICAgZHAtPmRweXNbcy0+aW1wbF0sIHMtPnN1cmZhY2UsIGJ1ZmZlcik7Ci0gICAgfQotICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX1NVUkZBQ0UsIEVHTF9GQUxTRSk7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsUmVsZWFzZVRleEltYWdlKAotICAgICAgICBFR0xEaXNwbGF5IGRweSwgRUdMU3VyZmFjZSBzdXJmYWNlLCBFR0xpbnQgYnVmZmVyKQotewotICAgIGlmICghdmFsaWRhdGVfZGlzcGxheV9zdXJmYWNlKGRweSwgc3VyZmFjZSkpCi0gICAgICAgIHJldHVybiBFR0xfRkFMU0U7ICAgIAotICAgIGVnbF9kaXNwbGF5X3QgY29uc3QgKiBjb25zdCBkcCA9IGdldF9kaXNwbGF5KGRweSk7Ci0gICAgZWdsX3N1cmZhY2VfdCBjb25zdCAqIGNvbnN0IHMgPSBnZXRfc3VyZmFjZShzdXJmYWNlKTsKLSAgICBpZiAocy0+Y254LT5ob29rcy0+ZWdsLmVnbFJlbGVhc2VUZXhJbWFnZSkgewotICAgICAgICByZXR1cm4gcy0+Y254LT5ob29rcy0+ZWdsLmVnbFJlbGVhc2VUZXhJbWFnZSgKLSAgICAgICAgICAgICAgICBkcC0+ZHB5c1tzLT5pbXBsXSwgcy0+c3VyZmFjZSwgYnVmZmVyKTsKLSAgICB9Ci0gICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfU1VSRkFDRSwgRUdMX0ZBTFNFKTsKLX0KLQotRUdMQm9vbGVhbiBlZ2xTd2FwSW50ZXJ2YWwoRUdMRGlzcGxheSBkcHksIEVHTGludCBpbnRlcnZhbCkKLXsKLSAgICBlZ2xfZGlzcGxheV90ICogY29uc3QgZHAgPSBnZXRfZGlzcGxheShkcHkpOwotICAgIGlmICghZHApIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0RJU1BMQVksIEVHTF9GQUxTRSk7Ci0KLSAgICBFR0xCb29sZWFuIHJlcyA9IEVHTF9UUlVFOwotICAgIGZvciAoaW50IGk9MCA7IGk8MiA7IGkrKykgewotICAgICAgICBlZ2xfY29ubmVjdGlvbl90KiBjb25zdCBjbnggPSAmZ0VHTEltcGxbaV07Ci0gICAgICAgIGlmIChjbngtPmRzbykgewotICAgICAgICAgICAgaWYgKGNueC0+aG9va3MtPmVnbC5lZ2xTd2FwSW50ZXJ2YWwpIHsKLSAgICAgICAgICAgICAgICBpZiAoY254LT5ob29rcy0+ZWdsLmVnbFN3YXBJbnRlcnZhbChkcC0+ZHB5c1tpXSwgaW50ZXJ2YWwpID09IEVHTF9GQUxTRSkgewotICAgICAgICAgICAgICAgICAgICByZXMgPSBFR0xfRkFMU0U7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiByZXM7Ci19Ci0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gRUdMIDEuMgotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1FR0xCb29sZWFuIGVnbFdhaXRDbGllbnQodm9pZCkKLXsKLSAgICBFR0xCb29sZWFuIHJlcyA9IEVHTF9UUlVFOwotICAgIEVHTENvbnRleHQgY3R4ID0gZ2V0Q29udGV4dCgpOwotICAgIGlmIChjdHgpIHsKLSAgICAgICAgZWdsX2NvbnRleHRfdCBjb25zdCAqIGNvbnN0IGMgPSBnZXRfY29udGV4dChjdHgpOwotICAgICAgICBpZiAoIWMpIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0NPTlRFWFQsIEVHTF9GQUxTRSk7Ci0gICAgICAgIGlmICh1aW50MzJfdChjLT5pbXBsKT49MikKLSAgICAgICAgICAgIHJldHVybiBzZXRFcnJvcihFR0xfQkFEX0NPTlRFWFQsIEVHTF9GQUxTRSk7Ci0gICAgICAgIGVnbF9jb25uZWN0aW9uX3QqIGNvbnN0IGNueCA9ICZnRUdMSW1wbFtjLT5pbXBsXTsKLSAgICAgICAgaWYgKCFjbngtPmRzbykgCi0gICAgICAgICAgICByZXR1cm4gc2V0RXJyb3IoRUdMX0JBRF9DT05URVhULCBFR0xfRkFMU0UpOwotICAgICAgICBpZiAoY254LT5ob29rcy0+ZWdsLmVnbFdhaXRDbGllbnQpIHsKLSAgICAgICAgICAgIHJlcyA9IGNueC0+aG9va3MtPmVnbC5lZ2xXYWl0Q2xpZW50KCk7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICByZXMgPSBjbngtPmhvb2tzLT5lZ2wuZWdsV2FpdEdMKCk7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIHJlczsKLX0KLQotRUdMQm9vbGVhbiBlZ2xCaW5kQVBJKEVHTGVudW0gYXBpKQotewotICAgIC8vIGJpbmQgdGhpcyBBUEkgb24gYWxsIEVHTHMKLSAgICBFR0xCb29sZWFuIHJlcyA9IEVHTF9UUlVFOwotICAgIGZvciAoaW50IGk9MCA7IGk8MiA7IGkrKykgewotICAgICAgICBlZ2xfY29ubmVjdGlvbl90KiBjb25zdCBjbnggPSAmZ0VHTEltcGxbaV07Ci0gICAgICAgIGlmIChjbngtPmRzbykgewotICAgICAgICAgICAgaWYgKGNueC0+aG9va3MtPmVnbC5lZ2xCaW5kQVBJKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGNueC0+aG9va3MtPmVnbC5lZ2xCaW5kQVBJKGFwaSkgPT0gRUdMX0ZBTFNFKSB7Ci0gICAgICAgICAgICAgICAgICAgIHJlcyA9IEVHTF9GQUxTRTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIHJlczsKLX0KLQotRUdMZW51bSBlZ2xRdWVyeUFQSSh2b2lkKQotewotICAgIGZvciAoaW50IGk9MCA7IGk8MiA7IGkrKykgewotICAgICAgICBlZ2xfY29ubmVjdGlvbl90KiBjb25zdCBjbnggPSAmZ0VHTEltcGxbaV07Ci0gICAgICAgIGlmIChjbngtPmRzbykgewotICAgICAgICAgICAgaWYgKGNueC0+aG9va3MtPmVnbC5lZ2xRdWVyeUFQSSkgewotICAgICAgICAgICAgICAgIC8vIHRoZSBmaXJzdCBvbmUgd2UgZmluZCBpcyBva2F5LCBiZWNhdXNlIHRoZXkgYWxsCi0gICAgICAgICAgICAgICAgLy8gc2hvdWxkIGJlIHRoZSBzYW1lCi0gICAgICAgICAgICAgICAgcmV0dXJuIGNueC0+aG9va3MtPmVnbC5lZ2xRdWVyeUFQSSgpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIC8vIG9yLCBpdCBjYW4gb25seSBiZSBPcGVuR0wgRVMKLSAgICByZXR1cm4gRUdMX09QRU5HTF9FU19BUEk7Ci19Ci0KLUVHTEJvb2xlYW4gZWdsUmVsZWFzZVRocmVhZCh2b2lkKQotewotICAgIGZvciAoaW50IGk9MCA7IGk8MiA7IGkrKykgewotICAgICAgICBlZ2xfY29ubmVjdGlvbl90KiBjb25zdCBjbnggPSAmZ0VHTEltcGxbaV07Ci0gICAgICAgIGlmIChjbngtPmRzbykgewotICAgICAgICAgICAgaWYgKGNueC0+aG9va3MtPmVnbC5lZ2xSZWxlYXNlVGhyZWFkKSB7Ci0gICAgICAgICAgICAgICAgY254LT5ob29rcy0+ZWdsLmVnbFJlbGVhc2VUaHJlYWQoKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotICAgIH0KLSAgICBjbGVhclRMUygpOyAgICAKLSAgICByZXR1cm4gRUdMX1RSVUU7Ci19Ci0KLUVHTFN1cmZhY2UgZWdsQ3JlYXRlUGJ1ZmZlckZyb21DbGllbnRCdWZmZXIoCi0gICAgICAgICAgRUdMRGlzcGxheSBkcHksIEVHTGVudW0gYnVmdHlwZSwgRUdMQ2xpZW50QnVmZmVyIGJ1ZmZlciwKLSAgICAgICAgICBFR0xDb25maWcgY29uZmlnLCBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KQotewotICAgIGVnbF9kaXNwbGF5X3QgY29uc3QqIGRwID0gMDsKLSAgICBpbnQgaT0wLCBpbmRleD0wOwotICAgIGVnbF9jb25uZWN0aW9uX3QqIGNueCA9IHZhbGlkYXRlX2Rpc3BsYXlfY29uZmlnKGRweSwgY29uZmlnLCBkcCwgaSwgaW5kZXgpOwotICAgIGlmICghY254KSByZXR1cm4gRUdMX0ZBTFNFOwotICAgIGlmIChjbngtPmhvb2tzLT5lZ2wuZWdsQ3JlYXRlUGJ1ZmZlckZyb21DbGllbnRCdWZmZXIpIHsKLSAgICAgICAgcmV0dXJuIGNueC0+aG9va3MtPmVnbC5lZ2xDcmVhdGVQYnVmZmVyRnJvbUNsaWVudEJ1ZmZlcigKLSAgICAgICAgICAgICAgICBkcC0+ZHB5c1tpXSwgYnVmdHlwZSwgYnVmZmVyLCBkcC0+Y29uZmlnc1tpXVtpbmRleF0sIGF0dHJpYl9saXN0KTsKLSAgICB9Ci0gICAgcmV0dXJuIHNldEVycm9yKEVHTF9CQURfQ09ORklHLCBFR0xfTk9fU1VSRkFDRSk7Ci19CmRpZmYgLS1naXQgYS9vcGVuZ2wvbGlicy9FR0wvZ3B1LmNwcCBiL29wZW5nbC9saWJzL0VHTC9ncHUuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzZjlmZDYzLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJzL0VHTC9ncHUuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMjEyICswLDAgQEAKLS8qIAotICoqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKioKLSAqKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0gKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSAqKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0gKioKLSAqKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotICoqCi0gKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSAqKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSAqKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0gKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSAqKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jZGVmaW5lIExPR19UQUcgIkVHTCIKLQotI2luY2x1ZGUgPGN0eXBlLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLQotI2luY2x1ZGUgPHN5cy9pb2N0bC5oPgotCi0jaWYgSEFWRV9BTkRST0lEX09TCi0jaW5jbHVkZSA8bGludXgvYW5kcm9pZF9wbWVtLmg+Ci0jZW5kaWYKLQotI2luY2x1ZGUgPGN1dGlscy9sb2cuaD4KLSNpbmNsdWRlIDxjdXRpbHMvcHJvcGVydGllcy5oPgotCi0jaW5jbHVkZSA8dXRpbHMvSU1lbW9yeS5oPgotI2luY2x1ZGUgPHV0aWxzL3RocmVhZHMuaD4KLSNpbmNsdWRlIDx1dGlscy9JU2VydmljZU1hbmFnZXIuaD4KLSNpbmNsdWRlIDx1dGlscy9JUENUaHJlYWRTdGF0ZS5oPgotI2luY2x1ZGUgPHV0aWxzL1BhcmNlbC5oPgotCi0jaW5jbHVkZSA8dWkvRUdMRGlzcGxheVN1cmZhY2UuaD4KLSNpbmNsdWRlIDx1aS9JU3VyZmFjZUNvbXBvc2VyLmg+Ci0KLSNpbmNsdWRlICJob29rcy5oIgotI2luY2x1ZGUgImVnbF9pbXBsLmgiCi0KLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLS8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLQotLyoKLSAqIHdlIHByb3ZpZGUgb3VyIG93biBhbGxvY2F0b3JzIGZvciB0aGUgR1BVIHJlZ2lvbnMsIHRoZXNlCi0gKiBhbGxvY2F0b3JzIGdvIHRocm91Z2ggc3VyZmFjZWZsaW5nZXIgCi0gKi8KLQotc3RhdGljIE11dGV4ICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdSZWdpb25zTG9jazsKLXN0YXRpYyByZXF1ZXN0X2dwdV90ICAgICAgICAgICAgICAgICAgICBnUmVnaW9uczsKLXN0YXRpYyBzcDxJU3VyZmFjZUNvbXBvc2VyPiAgICAgICAgICAgICBnU3VyZmFjZU1hbmFnZXI7Ci1JU3VyZmFjZUNvbXBvc2VyKiAgICAgICAgICAgICAgICAgICAgICAgR0xFU19sb2NhbFN1cmZhY2VNYW5hZ2VyID0gMDsKLQotZXh0ZXJuIGVnbF9jb25uZWN0aW9uX3QgZ0VHTEltcGxbMl07Ci0KLWNvbnN0IHNwPElTdXJmYWNlQ29tcG9zZXI+JiBnZXRTdXJmYWNlRmxpbmdlcigpCi17Ci0gICAgTXV0ZXg6OkF1dG9sb2NrIF9sKGdSZWdpb25zTG9jayk7Ci0KLSAgICAvKgotICAgICAqIFRoZXJlIGlzIGEgbGl0dGxlIGJpdCBvZiB2b29kb28gbWFnaWMgaGVyZS4gV2Ugd2FudCB0byBhY2Nlc3MKLSAgICAgKiBzdXJmYWNlZmxpbmdlciBmb3IgYWxsb2NhdGluZyBHUFUgcmVnaW9ucywgaG93ZXZlciwgd2hlbiB3ZSBhcmUKLSAgICAgKiBydW5uaW5nIGFzIHBhcnQgb2Ygc3VyZmFjZWZsaW5nZXIsIHdlIHdhbnQgdG8gYnlwYXNzIHRoZQotICAgICAqIHNlcnZpY2UgbWFuYWdlciBiZWNhdXNlIHN1cmZhY2VmbGluZ2VyIG1pZ2h0IG5vdCBiZSByZWdpc3RlcmVkIHlldC4KLSAgICAgKiBTdXJmYWNlRmxpbmdlciB3aWxsIHBvcHVsYXRlICJHTEVTX2xvY2FsU3VyZmFjZU1hbmFnZXIiIHdpdGggaXRzCi0gICAgICogb3duIGFkZHJlc3MsIHNvIHdlIGNhbiBqdXN0IHVzZSB0aGF0LgotICAgICAqLwotICAgIGlmIChnU3VyZmFjZU1hbmFnZXIgPT0gMCkgewotICAgICAgICBpZiAoR0xFU19sb2NhbFN1cmZhY2VNYW5hZ2VyKSB7Ci0gICAgICAgICAgICAvLyB3ZSdyZSBydW5uaW5nIGluIFN1cmZhY2VGbGluZ2VyJ3MgY29udGV4dAotICAgICAgICAgICAgZ1N1cmZhY2VNYW5hZ2VyID0gIEdMRVNfbG9jYWxTdXJmYWNlTWFuYWdlcjsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIC8vIHdlJ3JlIGEgcmVtb3RlIHByb2Nlc3Mgb3Igbm90IHBhcnQgb2Ygc3VyZmFjZWZsaW5nZXIsCi0gICAgICAgICAgICAvLyBnbyB0aHJvdWdoIHRoZSBzZXJ2aWNlIG1hbmFnZXIKLSAgICAgICAgICAgIHNwPElTZXJ2aWNlTWFuYWdlcj4gc20gPSBkZWZhdWx0U2VydmljZU1hbmFnZXIoKTsKLSAgICAgICAgICAgIGlmIChzbSAhPSBOVUxMKSB7Ci0gICAgICAgICAgICAgICAgc3A8SUJpbmRlcj4gYmluZGVyID0gc20tPmdldFNlcnZpY2UoU3RyaW5nMTYoIlN1cmZhY2VGbGluZ2VyIikpOwotICAgICAgICAgICAgICAgIGdTdXJmYWNlTWFuYWdlciA9IGludGVyZmFjZV9jYXN0PElTdXJmYWNlQ29tcG9zZXI+KGJpbmRlcik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgcmV0dXJuIGdTdXJmYWNlTWFuYWdlcjsKLX0KLQotY2xhc3MgR1BVUmV2b2tlUmVxdWVzdGVyIDogcHVibGljIEJuR1BVQ2FsbGJhY2sKLXsKLXB1YmxpYzoKLSAgICB2aXJ0dWFsIHZvaWQgZ3B1TG9zdCgpIHsKLSAgICAgICAgTE9HRCgiQ09OVEVYVF9MT1NUOiBSZWxlYXNpbmcgR1BVIHVwb24gcmVxdWVzdCBmcm9tIFN1cmZhY2VGbGluZ2VyLiIpOwotICAgICAgICBnRUdMSW1wbFtJTVBMX0hBUkRXQVJFXS5ob29rcyA9ICZnSG9va3NbSU1QTF9DT05URVhUX0xPU1RdOwotICAgIH0KLX07Ci0KLXN0YXRpYyBzcDxHUFVSZXZva2VSZXF1ZXN0ZXI+IGdSZXZva2VyQ2FsbGJhY2s7Ci0KLQotcmVxdWVzdF9ncHVfdCogZ3B1X2FjcXVpcmUodm9pZCogdXNlcikKLXsKLSAgICBzcDxJU3VyZmFjZUNvbXBvc2VyPiBzZXJ2ZXIoIGdldFN1cmZhY2VGbGluZ2VyKCkgKTsKLQotICAgIE11dGV4OjpBdXRvbG9jayBfbChnUmVnaW9uc0xvY2spOwotICAgIGlmIChzZXJ2ZXIgPT0gTlVMTCkgewotICAgICAgICByZXR1cm4gMDsKLSAgICB9Ci0gICAgCi0gICAgSVN1cmZhY2VDb21wb3Nlcjo6Z3B1X2luZm9fdCBpbmZvOwotICAgIAotICAgIGlmIChnUmV2b2tlckNhbGxiYWNrID09IDApCi0gICAgICAgIGdSZXZva2VyQ2FsbGJhY2sgPSBuZXcgR1BVUmV2b2tlUmVxdWVzdGVyKCk7Ci0KLSAgICBzdGF0dXNfdCBlcnIgPSBzZXJ2ZXItPnJlcXVlc3RHUFUoZ1Jldm9rZXJDYWxsYmFjaywgJmluZm8pOwotICAgIGlmIChlcnIgIT0gTk9fRVJST1IpIHsKLSAgICAgICAgTE9HRCgicmVxdWVzdEdQVSByZXR1cm5lZCAlZCIsIGVycik7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLQotICAgIGJvb2wgZmFpbGVkID0gZmFsc2U7Ci0gICAgcmVxdWVzdF9ncHVfdCogZ3B1ID0gJmdSZWdpb25zOwotICAgIG1lbXNldChncHUsIDAsIHNpemVvZigqZ3B1KSk7Ci0gICAgCi0gICAgaWYgKGluZm8ucmVncyAhPSAwKSB7Ci0gICAgICAgIHNwPElNZW1vcnlIZWFwPiBoZWFwKGluZm8ucmVncy0+Z2V0TWVtb3J5KCkpOwotICAgICAgICBpZiAoaGVhcCAhPSAwKSB7Ci0gICAgICAgICAgICBpbnQgZmQgPSBoZWFwLT5oZWFwSUQoKTsKLSAgICAgICAgICAgIGdwdS0+cmVncy5mZCA9IGZkOwotICAgICAgICAgICAgZ3B1LT5yZWdzLmJhc2UgPSBpbmZvLnJlZ3MtPnBvaW50ZXIoKTsgCi0gICAgICAgICAgICBncHUtPnJlZ3Muc2l6ZSA9IGluZm8ucmVncy0+c2l6ZSgpOyAKLSAgICAgICAgICAgIGdwdS0+cmVncy51c2VyID0gaW5mby5yZWdzLmdldCgpOwotI2lmIEhBVkVfQU5EUk9JRF9PUwotICAgICAgICAgICAgc3RydWN0IHBtZW1fcmVnaW9uIHJlZ2lvbjsKLSAgICAgICAgICAgIGlmIChpb2N0bChmZCwgUE1FTV9HRVRfUEhZUywgJnJlZ2lvbikgPj0gMCkKLSAgICAgICAgICAgICAgICBncHUtPnJlZ3MucGh5cyA9ICh2b2lkKilyZWdpb24ub2Zmc2V0OwotI2VuZGlmCi0gICAgICAgICAgICBpbmZvLnJlZ3MtPmluY1N0cm9uZyhncHUpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgTE9HRSgiR1BVIHJlZ2lzdGVyIGhhbmRsZSAlcCBpcyBpbnZhbGlkISIsIGluZm8ucmVncy5nZXQoKSk7Ci0gICAgICAgICAgICBmYWlsZWQgPSB0cnVlOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgZm9yIChzaXplX3QgaT0wIDsgaTxpbmZvLmNvdW50ICYmICFmYWlsZWQgOyBpKyspIHsKLSAgICAgICAgc3A8SU1lbW9yeT4mIHJlZ2lvbihpbmZvLnJlZ2lvbnNbaV0ucmVnaW9uKTsKLSAgICAgICAgaWYgKHJlZ2lvbiAhPSAwKSB7Ci0gICAgICAgICAgICBzcDxJTWVtb3J5SGVhcD4gaGVhcChyZWdpb24tPmdldE1lbW9yeSgpKTsKLSAgICAgICAgICAgIGlmIChoZWFwICE9IDApIHsKLSAgICAgICAgICAgICAgICBjb25zdCBpbnQgZmQgPSBoZWFwLT5oZWFwSUQoKTsKLSAgICAgICAgICAgICAgICBncHUtPmdwdVtpXS5mZCA9IGZkOwotICAgICAgICAgICAgICAgIGdwdS0+Z3B1W2ldLmJhc2UgPSByZWdpb24tPnBvaW50ZXIoKTsgCi0gICAgICAgICAgICAgICAgZ3B1LT5ncHVbaV0uc2l6ZSA9IHJlZ2lvbi0+c2l6ZSgpOyAKLSAgICAgICAgICAgICAgICBncHUtPmdwdVtpXS51c2VyID0gcmVnaW9uLmdldCgpOwotICAgICAgICAgICAgICAgIGdwdS0+Z3B1W2ldLm9mZnNldCA9IGluZm8ucmVnaW9uc1tpXS5yZXNlcnZlZDsKLSNpZiBIQVZFX0FORFJPSURfT1MKLSAgICAgICAgICAgICAgICBzdHJ1Y3QgcG1lbV9yZWdpb24gcmVnOwotICAgICAgICAgICAgICAgIGlmIChpb2N0bChmZCwgUE1FTV9HRVRfUEhZUywgJnJlZykgPj0gMCkKLSAgICAgICAgICAgICAgICAgICAgZ3B1LT5ncHVbaV0ucGh5cyA9ICh2b2lkKilyZWcub2Zmc2V0OwotI2VuZGlmCi0gICAgICAgICAgICAgICAgcmVnaW9uLT5pbmNTdHJvbmcoZ3B1KTsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgTE9HRSgiR1BVIHJlZ2lvbiBoYW5kbGUgWyVkLCAlcF0gaXMgaW52YWxpZCEiLCBpLCByZWdpb24uZ2V0KCkpOwotICAgICAgICAgICAgICAgIGZhaWxlZCA9IHRydWU7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgaWYgKGZhaWxlZCkgewotICAgICAgICAvLyBzb21ldGhpbmcgd2VudCB3cm9uZywgY2xlYW4gdXAgZXZlcnl0aGluZyEKLSAgICAgICAgaWYgKGdwdS0+cmVncy51c2VyKSB7Ci0gICAgICAgICAgICBzdGF0aWNfY2FzdDxJTWVtb3J5Kj4oZ3B1LT5yZWdzLnVzZXIpLT5kZWNTdHJvbmcoZ3B1KTsKLSAgICAgICAgICAgIGZvciAoc2l6ZV90IGk9MCA7IGk8aW5mby5jb3VudCA7IGkrKykgewotICAgICAgICAgICAgICAgIGlmIChncHUtPmdwdVtpXS51c2VyKSB7Ci0gICAgICAgICAgICAgICAgICAgIHN0YXRpY19jYXN0PElNZW1vcnkqPihncHUtPmdwdVtpXS51c2VyKS0+ZGVjU3Ryb25nKGdwdSk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgfQotICAgIAotICAgIGdwdS0+Y291bnQgPSBpbmZvLmNvdW50OwotICAgIHJldHVybiBncHU7Ci19Ci0KLWludCBncHVfcmVsZWFzZSh2b2lkKiwgcmVxdWVzdF9ncHVfdCogZ3B1KQotewotICAgIHNwPElNZW1vcnk+IHJlZ3M7Ci0KLSAgICB7IC8vIHNjb3BlIGZvciBsb2NrCi0gICAgICAgIE11dGV4OjpBdXRvbG9jayBfbChnUmVnaW9uc0xvY2spOwotICAgICAgICByZWdzID0gc3RhdGljX2Nhc3Q8SU1lbW9yeSo+KGdwdS0+cmVncy51c2VyKTsgICAKLSAgICAgICAgZ3B1LT5yZWdzLnVzZXIgPSAwOwotICAgICAgICBpZiAocmVncyAhPSAwKSByZWdzLT5kZWNTdHJvbmcoZ3B1KTsKLSAgICAgICAgCi0gICAgICAgIGZvciAoaW50IGk9MCA7IGk8Z3B1LT5jb3VudCA7IGkrKykgewotICAgICAgICAgICAgc3A8SU1lbW9yeT4gcihzdGF0aWNfY2FzdDxJTWVtb3J5Kj4oZ3B1LT5ncHVbaV0udXNlcikpOwotICAgICAgICAgICAgZ3B1LT5ncHVbaV0udXNlciA9IDA7Ci0gICAgICAgICAgICBpZiAociAhPSAwKSByLT5kZWNTdHJvbmcoZ3B1KTsKLSAgICAgICAgfQotICAgIH0KLSAgICAKLSAgICAvLyB0aGVyZSBpcyBhIHNwZWNpYWwgdHJhbnNhY3Rpb24gdG8gcmVsaW5xdWlzaCB0aGUgR1BVCi0gICAgLy8gKGl0IHdpbGwgaGFwcGVuIGF1dG9tYXRpY2FsbHkgYW55d2F5IGlmIHdlIGRvbid0IGRvIHRoaXMpCi0gICAgUGFyY2VsIGRhdGEsIHJlcGx5OwotICAgIC8vIE5PVEU6IHRoaXMgdHJhbnNhY3Rpb24gZG9lcyBub3QgcmVxdWlyZSBhbiBpbnRlcmZhY2UgdG9rZW4KLSAgICByZWdzLT5hc0JpbmRlcigpLT50cmFuc2FjdCgxMDAwLCBkYXRhLCAmcmVwbHkpOwotICAgIHJldHVybiAxOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYnMvR0xFU19DTS9nbC5jcHAgYi9vcGVuZ2wvbGlicy9HTEVTX0NNL2dsLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggODY1Y2Y0NC4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGlicy9HTEVTX0NNL2dsLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDExNiArMCwwIEBACi0vKiAKLSAqKiBDb3B5cmlnaHQgMjAwNywgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoqCi0gKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotICoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0gKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotICoqCi0gKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSAqKgotICoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0gKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0gKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotICoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0gKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJHTEVTX0NNIgotCi0jaW5jbHVkZSA8Y3R5cGUuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotCi0jaW5jbHVkZSA8c3lzL2lvY3RsLmg+Ci0KLSNpbmNsdWRlIDxHTEVTL2dsLmg+Ci0jaW5jbHVkZSA8R0xFUy9nbGV4dC5oPgotCi0jaW5jbHVkZSA8Y3V0aWxzL2xvZy5oPgotI2luY2x1ZGUgPGN1dGlscy9wcm9wZXJ0aWVzLmg+Ci0KLSNpbmNsdWRlICJob29rcy5oIgotCi11c2luZyBuYW1lc3BhY2UgYW5kcm9pZDsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotLy8gZXh0ZW5zaW9ucyBmb3IgdGhlIGZyYW1ld29yawotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi12b2lkIGdsQ29sb3JQb2ludGVyQm91bmRzKEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwKLSAgICAgICAgY29uc3QgR0x2b2lkICpwdHIsIEdMc2l6ZWkgY291bnQpIHsKLSAgICBnbENvbG9yUG9pbnRlcihzaXplLCB0eXBlLCBzdHJpZGUsIHB0cik7Ci19Ci12b2lkIGdsTm9ybWFsUG9pbnRlckJvdW5kcyhHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsCi0gICAgICAgIGNvbnN0IEdMdm9pZCAqcG9pbnRlciwgR0xzaXplaSBjb3VudCkgewotICAgIGdsTm9ybWFsUG9pbnRlcih0eXBlLCBzdHJpZGUsIHBvaW50ZXIpOwotfQotdm9pZCBnbFRleENvb3JkUG9pbnRlckJvdW5kcyhHTGludCBzaXplLCBHTGVudW0gdHlwZSwKLSAgICAgICAgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlciwgR0xzaXplaSBjb3VudCkgewotICAgIGdsVGV4Q29vcmRQb2ludGVyKHNpemUsIHR5cGUsIHN0cmlkZSwgcG9pbnRlcik7Ci19Ci12b2lkIGdsVmVydGV4UG9pbnRlckJvdW5kcyhHTGludCBzaXplLCBHTGVudW0gdHlwZSwKLSAgICAgICAgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlciwgR0xzaXplaSBjb3VudCkgewotICAgIGdsVmVydGV4UG9pbnRlcihzaXplLCB0eXBlLCBzdHJpZGUsIHBvaW50ZXIpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0vLyBBY3R1YWwgR0wgZW50cnktcG9pbnRzCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLSNpZiBHTF9MT0dHRVIKLSMgICBpbmNsdWRlICJnbF9sb2dnZXIuaCIKLSMgICBkZWZpbmUgR0xfTE9HR0VSX0lNUEwoX3gpIF94Ci0jZWxzZQotIyAgIGRlZmluZSBHTF9MT0dHRVJfSU1QTChfeCkKLSNlbmRpZgotCi0jdW5kZWYgQVBJX0VOVFJZCi0jdW5kZWYgQ0FMTF9HTF9BUEkKLSN1bmRlZiBDQUxMX0dMX0FQSV9SRVRVUk4KLQotI2lmIFVTRV9GQVNUX1RMU19LRVkKLQotICAgICNkZWZpbmUgQVBJX0VOVFJZKF9hcGkpIF9fYXR0cmlidXRlX18oKG5ha2VkKSkgX2FwaQotCi0gICAgI2RlZmluZSBDQUxMX0dMX0FQSShfYXBpLCAuLi4pICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgYXNtIHZvbGF0aWxlKCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLSAgICAgICAgICAgICJtb3YgICByMTIsICMweEZGRkYwRkZGICAgXG4iICAgICAgICAgICAgICAgICAgICAgICBcCi0gICAgICAgICAgICAibGRyICAgcjEyLCBbcjEyLCAjLTE1XSAgIFxuIiAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgImxkciAgIHIxMiwgW3IxMiwgJVt0bHNdXSBcbiIgICAgICAgICAgICAgICAgICAgICAgIFwKLSAgICAgICAgICAgICJjbXAgICByMTIsICMwICAgICAgICAgICAgXG4iICAgICAgICAgICAgICAgICAgICAgICBcCi0gICAgICAgICAgICAibGRybmUgcGMsICBbcjEyLCAlW2FwaV1dIFxuIiAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgImJ4ICAgIGxyICAgICAgICAgICAgICAgICBcbiIgICAgICAgICAgICAgICAgICAgICAgIFwKLSAgICAgICAgICAgIDogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi0gICAgICAgICAgICA6IFt0bHNdICJKIihUTFNfU0xPVF9PUEVOR0xfQVBJKjQpLCAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgICBbYXBpXSAiSiIoX19idWlsdGluX29mZnNldG9mKGdsX2hvb2tzX3QsIGdsLl9hcGkpKSAgICBcCi0gICAgICAgICAgICA6ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAotICAgICAgICAgICAgKTsKLSAgICAKLSAgICAjZGVmaW5lIENBTExfR0xfQVBJX1JFVFVSTihfYXBpLCAuLi4pIFwKLSAgICAgICAgQ0FMTF9HTF9BUEkoX2FwaSwgX19WQV9BUkdTX18pIFwKLSAgICAgICAgcmV0dXJuIDA7IC8vIHBsYWNhdGUgZ2NjJ3Mgd2FybmluZ3MuIG5ldmVyIHJlYWNoZWQuCi0KLSNlbHNlCi0KLSAgICAjZGVmaW5lIEFQSV9FTlRSWShfYXBpKSBfYXBpCi0KLSAgICAjZGVmaW5lIENBTExfR0xfQVBJKF9hcGksIC4uLikgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFwKLSAgICAgICAgZ2xfaG9va3NfdDo6Z2xfdCBjb25zdCAqIGNvbnN0IF9jID0gJmdldEdsVGhyZWFkU3BlY2lmaWMoKS0+Z2w7IFwKLSAgICAgICAgR0xfTE9HR0VSX0lNUEwoIGxvZ18jI19hcGkoX19WQV9BUkdTX18pOyApICAgICAgICAgICAgICAgICAgICAgIFwKLSAgICAgICAgX2MtPl9hcGkoX19WQV9BUkdTX18pCi0gICAgCi0gICAgI2RlZmluZSBDQUxMX0dMX0FQSV9SRVRVUk4oX2FwaSwgLi4uKSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBcCi0gICAgICAgIGdsX2hvb2tzX3Q6OmdsX3QgY29uc3QgKiBjb25zdCBfYyA9ICZnZXRHbFRocmVhZFNwZWNpZmljKCktPmdsOyBcCi0gICAgICAgIEdMX0xPR0dFUl9JTVBMKCBsb2dfIyNfYXBpKF9fVkFfQVJHU19fKTsgKSAgICAgICAgICAgICAgICAgICAgICBcCi0gICAgICAgIHJldHVybiBfYy0+X2FwaShfX1ZBX0FSR1NfXykKLQotI2VuZGlmCi0KLWV4dGVybiAiQyIgewotI2luY2x1ZGUgImdsX2FwaS5pbiIKLX0KLQotI3VuZGVmIEFQSV9FTlRSWQotI3VuZGVmIENBTExfR0xfQVBJCi0jdW5kZWYgQ0FMTF9HTF9BUElfUkVUVVJOCi0KZGlmZiAtLWdpdCBhL29wZW5nbC9saWJzL0dMRVNfQ00vZ2xfYXBpLmluIGIvb3BlbmdsL2xpYnMvR0xFU19DTS9nbF9hcGkuaW4KZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDkyMzRlZjIuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYnMvR0xFU19DTS9nbF9hcGkuaW4KKysrIC9kZXYvbnVsbApAQCAtMSw2MDYgKzAsMCBAQAotdm9pZCBBUElfRU5UUlkoZ2xBY3RpdmVUZXh0dXJlKShHTGVudW0gdGV4dHVyZSkgewotICAgIENBTExfR0xfQVBJKGdsQWN0aXZlVGV4dHVyZSwgdGV4dHVyZSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQWxwaGFGdW5jKShHTGVudW0gZnVuYywgR0xjbGFtcGYgcmVmKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xBbHBoYUZ1bmMsIGZ1bmMsIHJlZik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQWxwaGFGdW5jeCkoR0xlbnVtIGZ1bmMsIEdMY2xhbXB4IHJlZikgewotICAgIENBTExfR0xfQVBJKGdsQWxwaGFGdW5jeCwgZnVuYywgcmVmKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xCaW5kVGV4dHVyZSkoR0xlbnVtIHRhcmdldCwgR0x1aW50IHRleHR1cmUpIHsKLSAgICBDQUxMX0dMX0FQSShnbEJpbmRUZXh0dXJlLCB0YXJnZXQsIHRleHR1cmUpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEJsZW5kRnVuYykoR0xlbnVtIHNmYWN0b3IsIEdMZW51bSBkZmFjdG9yKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xCbGVuZEZ1bmMsIHNmYWN0b3IsIGRmYWN0b3IpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENsZWFyKShHTGJpdGZpZWxkIG1hc2spIHsKLSAgICBDQUxMX0dMX0FQSShnbENsZWFyLCBtYXNrKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDbGVhckNvbG9yKShHTGNsYW1wZiByZWQsIEdMY2xhbXBmIGdyZWVuLCBHTGNsYW1wZiBibHVlLCBHTGNsYW1wZiBhbHBoYSkgewotICAgIENBTExfR0xfQVBJKGdsQ2xlYXJDb2xvciwgcmVkLCBncmVlbiwgYmx1ZSwgYWxwaGEpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENsZWFyQ29sb3J4KShHTGNsYW1weCByZWQsIEdMY2xhbXB4IGdyZWVuLCBHTGNsYW1weCBibHVlLCBHTGNsYW1weCBhbHBoYSkgewotICAgIENBTExfR0xfQVBJKGdsQ2xlYXJDb2xvcngsIHJlZCwgZ3JlZW4sIGJsdWUsIGFscGhhKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDbGVhckRlcHRoZikoR0xjbGFtcGYgZGVwdGgpIHsKLSAgICBDQUxMX0dMX0FQSShnbENsZWFyRGVwdGhmLCBkZXB0aCk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQ2xlYXJEZXB0aHgpKEdMY2xhbXB4IGRlcHRoKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xDbGVhckRlcHRoeCwgZGVwdGgpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENsZWFyU3RlbmNpbCkoR0xpbnQgcykgewotICAgIENBTExfR0xfQVBJKGdsQ2xlYXJTdGVuY2lsLCBzKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDbGllbnRBY3RpdmVUZXh0dXJlKShHTGVudW0gdGV4dHVyZSkgewotICAgIENBTExfR0xfQVBJKGdsQ2xpZW50QWN0aXZlVGV4dHVyZSwgdGV4dHVyZSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQ29sb3I0ZikoR0xmbG9hdCByZWQsIEdMZmxvYXQgZ3JlZW4sIEdMZmxvYXQgYmx1ZSwgR0xmbG9hdCBhbHBoYSkgewotICAgIENBTExfR0xfQVBJKGdsQ29sb3I0ZiwgcmVkLCBncmVlbiwgYmx1ZSwgYWxwaGEpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENvbG9yNHgpKEdMZml4ZWQgcmVkLCBHTGZpeGVkIGdyZWVuLCBHTGZpeGVkIGJsdWUsIEdMZml4ZWQgYWxwaGEpIHsKLSAgICBDQUxMX0dMX0FQSShnbENvbG9yNHgsIHJlZCwgZ3JlZW4sIGJsdWUsIGFscGhhKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDb2xvck1hc2spKEdMYm9vbGVhbiByLCBHTGJvb2xlYW4gZywgR0xib29sZWFuIGIsIEdMYm9vbGVhbiBhKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xDb2xvck1hc2ssIHIsIGcsIGIsIGEpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENvbG9yUG9pbnRlcikoR0xpbnQgc2l6ZSwgR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnB0cikKLXsKLSAgICBDQUxMX0dMX0FQSShnbENvbG9yUG9pbnRlciwgc2l6ZSwgdHlwZSwgc3RyaWRlLCBwdHIpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENvbXByZXNzZWRUZXhJbWFnZTJEKShHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xlbnVtIGludGVybmFsZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LCBHTGludCBib3JkZXIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xzaXplaSBpbWFnZVNpemUsIGNvbnN0IEdMdm9pZCAqZGF0YSkgewotICAgIENBTExfR0xfQVBJKGdsQ29tcHJlc3NlZFRleEltYWdlMkQsIHRhcmdldCwgbGV2ZWwsIGludGVybmFsZm9ybWF0LAotICAgICAgICAgICAgd2lkdGgsIGhlaWdodCwgYm9yZGVyLCBpbWFnZVNpemUsIGRhdGEpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENvbXByZXNzZWRUZXhTdWJJbWFnZTJEKSggR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMaW50IHhvZmZzZXQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMaW50IHlvZmZzZXQsIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBHTGVudW0gZm9ybWF0LCBHTHNpemVpIGltYWdlU2l6ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgR0x2b2lkICpkYXRhKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xDb21wcmVzc2VkVGV4U3ViSW1hZ2UyRCwgdGFyZ2V0LCBsZXZlbCwgeG9mZnNldCwgeW9mZnNldCwKLSAgICAgICAgICAgIHdpZHRoLCBoZWlnaHQsIGZvcm1hdCwgaW1hZ2VTaXplLCBkYXRhKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDb3B5VGV4SW1hZ2UyRCkoICBHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xlbnVtIGludGVybmFsZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICAgICAgR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgICAgICBHTGludCBib3JkZXIpIHsKLSAgICBDQUxMX0dMX0FQSShnbENvcHlUZXhJbWFnZTJELCB0YXJnZXQsIGxldmVsLCBpbnRlcm5hbGZvcm1hdCwgeCwgeSwKLSAgICAgICAgICAgIHdpZHRoLCBoZWlnaHQsIGJvcmRlcik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQ29weVRleFN1YkltYWdlMkQpKCAgIEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGludCB4b2Zmc2V0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMaW50IHlvZmZzZXQsIEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xzaXplaSBoZWlnaHQpIHsKLSAgICBDQUxMX0dMX0FQSShnbENvcHlUZXhTdWJJbWFnZTJELCB0YXJnZXQsIGxldmVsLCB4b2Zmc2V0LCB5b2Zmc2V0LCB4LCB5LAotICAgICAgICAgICAgd2lkdGgsIGhlaWdodCk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQ3VsbEZhY2UpKEdMZW51bSBtb2RlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xDdWxsRmFjZSwgbW9kZSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRGVsZXRlVGV4dHVyZXMpKEdMc2l6ZWkgbiwgY29uc3QgR0x1aW50ICp0ZXh0dXJlcykgewotICAgIENBTExfR0xfQVBJKGdsRGVsZXRlVGV4dHVyZXMsIG4sIHRleHR1cmVzKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xEZXB0aEZ1bmMpKEdMZW51bSBmdW5jKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEZXB0aEZ1bmMsIGZ1bmMpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbERlcHRoTWFzaykoR0xib29sZWFuIGZsYWcpIHsKLSAgICBDQUxMX0dMX0FQSShnbERlcHRoTWFzaywgZmxhZyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRGVwdGhSYW5nZWYpKEdMY2xhbXBmIHpOZWFyLCBHTGNsYW1wZiB6RmFyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEZXB0aFJhbmdlZiwgek5lYXIsIHpGYXIpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbERlcHRoUmFuZ2V4KShHTGNsYW1weCB6TmVhciwgR0xjbGFtcHggekZhcikgewotICAgIENBTExfR0xfQVBJKGdsRGVwdGhSYW5nZXgsIHpOZWFyLCB6RmFyKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xEaXNhYmxlKShHTGVudW0gY2FwKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEaXNhYmxlLCBjYXApOwotfQotCi12b2lkIEFQSV9FTlRSWShnbERpc2FibGVDbGllbnRTdGF0ZSkoR0xlbnVtIGFycmF5KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEaXNhYmxlQ2xpZW50U3RhdGUsIGFycmF5KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xEcmF3QXJyYXlzKShHTGVudW0gbW9kZSwgR0xpbnQgZmlyc3QsIEdMc2l6ZWkgY291bnQpIHsKLSAgICBDQUxMX0dMX0FQSShnbERyYXdBcnJheXMsIG1vZGUsIGZpcnN0LCBjb3VudCk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRHJhd0VsZW1lbnRzKShHTGVudW0gbW9kZSwgR0xzaXplaSBjb3VudCwKLSAgICAgICAgICAgICAgICAgICAgR0xlbnVtIHR5cGUsIGNvbnN0IEdMdm9pZCAqaW5kaWNlcykgewotICAgIENBTExfR0xfQVBJKGdsRHJhd0VsZW1lbnRzLCBtb2RlLCBjb3VudCwgdHlwZSwgaW5kaWNlcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRW5hYmxlKShHTGVudW0gY2FwKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xFbmFibGUsIGNhcCk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRW5hYmxlQ2xpZW50U3RhdGUpKEdMZW51bSBhcnJheSkgewotICAgIENBTExfR0xfQVBJKGdsRW5hYmxlQ2xpZW50U3RhdGUsIGFycmF5KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xGaW5pc2gpKHZvaWQpIHsKLSAgICBDQUxMX0dMX0FQSShnbEZpbmlzaCk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRmx1c2gpKHZvaWQpIHsKLSAgICBDQUxMX0dMX0FQSShnbEZsdXNoKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xGb2dmKShHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbEZvZ2YsIHBuYW1lLCBwYXJhbSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRm9nZnYpKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xGb2dmdiwgcG5hbWUsIHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRm9neCkoR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xGb2d4LCBwbmFtZSwgcGFyYW0pOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEZvZ3h2KShHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsRm9neHYsIHBuYW1lLCBwYXJhbXMpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEZyb250RmFjZSkoR0xlbnVtIG1vZGUpIHsKLSAgICBDQUxMX0dMX0FQSShnbEZyb250RmFjZSwgbW9kZSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRnJ1c3R1bWYpKEdMZmxvYXQgbGVmdCwgR0xmbG9hdCByaWdodCwKLSAgICAgICAgICAgICAgICBHTGZsb2F0IGJvdHRvbSwgR0xmbG9hdCB0b3AsCi0gICAgICAgICAgICAgICAgR0xmbG9hdCB6TmVhciwgR0xmbG9hdCB6RmFyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xGcnVzdHVtZiwgbGVmdCwgcmlnaHQsIGJvdHRvbSwgdG9wLCB6TmVhciwgekZhcik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRnJ1c3R1bXgpKEdMZml4ZWQgbGVmdCwgR0xmaXhlZCByaWdodCwKLSAgICAgICAgICAgICAgICBHTGZpeGVkIGJvdHRvbSwgR0xmaXhlZCB0b3AsCi0gICAgICAgICAgICAgICAgR0xmaXhlZCB6TmVhciwgR0xmaXhlZCB6RmFyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xGcnVzdHVteCwgbGVmdCwgcmlnaHQsIGJvdHRvbSwgdG9wLCB6TmVhciwgekZhcik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsR2VuVGV4dHVyZXMpKEdMc2l6ZWkgbiwgR0x1aW50ICp0ZXh0dXJlcykgewotICAgIENBTExfR0xfQVBJKGdsR2VuVGV4dHVyZXMsIG4sIHRleHR1cmVzKTsKLX0KLQotR0xlbnVtIEFQSV9FTlRSWShnbEdldEVycm9yKSh2b2lkKSB7Ci0gICAgQ0FMTF9HTF9BUElfUkVUVVJOKGdsR2V0RXJyb3IpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEdldEludGVnZXJ2KShHTGVudW0gcG5hbWUsIEdMaW50ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdldEludGVnZXJ2LCBwbmFtZSwgcGFyYW1zKTsKLX0KLQotY29uc3QgR0x1Ynl0ZSAqIEFQSV9FTlRSWShnbEdldFN0cmluZykoR0xlbnVtIG5hbWUpIHsKLSAgICBDQUxMX0dMX0FQSV9SRVRVUk4oZ2xHZXRTdHJpbmcsIG5hbWUpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEhpbnQpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBtb2RlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xIaW50LCB0YXJnZXQsIG1vZGUpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExpZ2h0TW9kZWxmKShHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbExpZ2h0TW9kZWxmLCBwbmFtZSwgcGFyYW0pOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExpZ2h0TW9kZWxmdikoR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbExpZ2h0TW9kZWxmdiwgcG5hbWUsIHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTGlnaHRNb2RlbHgpKEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsTGlnaHRNb2RlbHgsIHBuYW1lLCBwYXJhbSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTGlnaHRNb2RlbHh2KShHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsTGlnaHRNb2RlbHh2LCBwbmFtZSwgcGFyYW1zKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xMaWdodGYpKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xMaWdodGYsIGxpZ2h0LCBwbmFtZSwgcGFyYW0pOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExpZ2h0ZnYpKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbExpZ2h0ZnYsIGxpZ2h0LCBwbmFtZSwgcGFyYW1zKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xMaWdodHgpKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xMaWdodHgsIGxpZ2h0LCBwbmFtZSwgcGFyYW0pOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExpZ2h0eHYpKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbExpZ2h0eHYsIGxpZ2h0LCBwbmFtZSwgcGFyYW1zKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xMaW5lV2lkdGgpKEdMZmxvYXQgd2lkdGgpIHsKLSAgICBDQUxMX0dMX0FQSShnbExpbmVXaWR0aCwgd2lkdGgpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExpbmVXaWR0aHgpKEdMZml4ZWQgd2lkdGgpIHsKLSAgICBDQUxMX0dMX0FQSShnbExpbmVXaWR0aHgsIHdpZHRoKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xMb2FkSWRlbnRpdHkpKHZvaWQpIHsKLSAgICBDQUxMX0dMX0FQSShnbExvYWRJZGVudGl0eSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTG9hZE1hdHJpeGYpKGNvbnN0IEdMZmxvYXQgKm0pIHsKLSAgICBDQUxMX0dMX0FQSShnbExvYWRNYXRyaXhmLCBtKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xMb2FkTWF0cml4eCkoY29uc3QgR0xmaXhlZCAqbSkgewotICAgIENBTExfR0xfQVBJKGdsTG9hZE1hdHJpeHgsIG0pOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExvZ2ljT3ApKEdMZW51bSBvcGNvZGUpIHsKLSAgICBDQUxMX0dMX0FQSShnbExvZ2ljT3AsIG9wY29kZSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTWF0ZXJpYWxmKShHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xNYXRlcmlhbGYsIGZhY2UsIHBuYW1lLCBwYXJhbSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTWF0ZXJpYWxmdikoR0xlbnVtIGZhY2UsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xNYXRlcmlhbGZ2LCBmYWNlLCBwbmFtZSwgcGFyYW1zKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xNYXRlcmlhbHgpKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbE1hdGVyaWFseCwgZmFjZSwgcG5hbWUsIHBhcmFtKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xNYXRlcmlhbHh2KShHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbE1hdGVyaWFseHYsIGZhY2UsIHBuYW1lLCBwYXJhbXMpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbE1hdHJpeE1vZGUpKEdMZW51bSBtb2RlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xNYXRyaXhNb2RlLCBtb2RlKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xNdWx0TWF0cml4ZikoY29uc3QgR0xmbG9hdCAqbSkgewotICAgIENBTExfR0xfQVBJKGdsTXVsdE1hdHJpeGYsIG0pOwotfQotCi12b2lkIEFQSV9FTlRSWShnbE11bHRNYXRyaXh4KShjb25zdCBHTGZpeGVkICptKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xNdWx0TWF0cml4eCwgbSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTXVsdGlUZXhDb29yZDRmKShHTGVudW0gdGFyZ2V0LCBHTGZsb2F0IHMsIEdMZmxvYXQgdCwgR0xmbG9hdCByLCBHTGZsb2F0IHEpIHsKLSAgICBDQUxMX0dMX0FQSShnbE11bHRpVGV4Q29vcmQ0ZiwgdGFyZ2V0LCBzLCB0LCByLCBxKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xNdWx0aVRleENvb3JkNHgpKEdMZW51bSB0YXJnZXQsIEdMZml4ZWQgcywgR0xmaXhlZCB0LCBHTGZpeGVkIHIsIEdMZml4ZWQgcSkgewotICAgIENBTExfR0xfQVBJKGdsTXVsdGlUZXhDb29yZDR4LCB0YXJnZXQsIHMsIHQsIHIsIHEpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbE5vcm1hbDNmKShHTGZsb2F0IG54LCBHTGZsb2F0IG55LCBHTGZsb2F0IG56KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xOb3JtYWwzZiwgbngsIG55LCBueik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTm9ybWFsM3gpKEdMZml4ZWQgbngsIEdMZml4ZWQgbnksIEdMZml4ZWQgbnopIHsKLSAgICBDQUxMX0dMX0FQSShnbE5vcm1hbDN4LCBueCwgbnksIG56KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xOb3JtYWxQb2ludGVyKShHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlcikgewotICAgIENBTExfR0xfQVBJKGdsTm9ybWFsUG9pbnRlciwgdHlwZSwgc3RyaWRlLCBwb2ludGVyKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xPcnRob2YpKCAgR0xmbG9hdCBsZWZ0LCBHTGZsb2F0IHJpZ2h0LAotICAgICAgICAgICAgICAgIEdMZmxvYXQgYm90dG9tLCBHTGZsb2F0IHRvcCwKLSAgICAgICAgICAgICAgICBHTGZsb2F0IHpOZWFyLCBHTGZsb2F0IHpGYXIpIHsKLSAgICBDQUxMX0dMX0FQSShnbE9ydGhvZiwgbGVmdCwgcmlnaHQsIGJvdHRvbSwgdG9wLCB6TmVhciwgekZhcik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsT3J0aG94KSggIEdMZml4ZWQgbGVmdCwgR0xmaXhlZCByaWdodCwKLSAgICAgICAgICAgICAgICBHTGZpeGVkIGJvdHRvbSwgR0xmaXhlZCB0b3AsCi0gICAgICAgICAgICAgICAgR0xmaXhlZCB6TmVhciwgR0xmaXhlZCB6RmFyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xPcnRob3gsIGxlZnQsIHJpZ2h0LCBib3R0b20sIHRvcCwgek5lYXIsIHpGYXIpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFBpeGVsU3RvcmVpKShHTGVudW0gcG5hbWUsIEdMaW50IHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xQaXhlbFN0b3JlaSwgcG5hbWUsIHBhcmFtKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xQb2ludFNpemUpKEdMZmxvYXQgc2l6ZSkgewotICAgIENBTExfR0xfQVBJKGdsUG9pbnRTaXplLCBzaXplKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xQb2ludFNpemV4KShHTGZpeGVkIHNpemUpIHsKLSAgICBDQUxMX0dMX0FQSShnbFBvaW50U2l6ZXgsIHNpemUpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFBvbHlnb25PZmZzZXQpKEdMZmxvYXQgZmFjdG9yLCBHTGZsb2F0IHVuaXRzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xQb2x5Z29uT2Zmc2V0LCBmYWN0b3IsIHVuaXRzKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xQb2x5Z29uT2Zmc2V0eCkoR0xmaXhlZCBmYWN0b3IsIEdMZml4ZWQgdW5pdHMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFBvbHlnb25PZmZzZXR4LCBmYWN0b3IsIHVuaXRzKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xQb3BNYXRyaXgpKHZvaWQpIHsKLSAgICBDQUxMX0dMX0FQSShnbFBvcE1hdHJpeCk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsUHVzaE1hdHJpeCkodm9pZCkgewotICAgIENBTExfR0xfQVBJKGdsUHVzaE1hdHJpeCk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsUmVhZFBpeGVscykoICBHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgR0xlbnVtIGZvcm1hdCwgR0xlbnVtIHR5cGUsIEdMdm9pZCAqcGl4ZWxzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xSZWFkUGl4ZWxzLCB4LCB5LCB3aWR0aCwgaGVpZ2h0LCBmb3JtYXQsIHR5cGUsIHBpeGVscyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsUm90YXRlZikoR0xmbG9hdCBhbmdsZSwgR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeikgewotICAgIENBTExfR0xfQVBJKGdsUm90YXRlZiwgYW5nbGUsIHgsIHksIHopOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFJvdGF0ZXgpKEdMZml4ZWQgYW5nbGUsIEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopIHsKLSAgICBDQUxMX0dMX0FQSShnbFJvdGF0ZXgsIGFuZ2xlLCB4LCB5LCB6KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xTYW1wbGVDb3ZlcmFnZSkoR0xjbGFtcGYgdmFsdWUsIEdMYm9vbGVhbiBpbnZlcnQpIHsKLSAgICBDQUxMX0dMX0FQSShnbFNhbXBsZUNvdmVyYWdlLCB2YWx1ZSwgaW52ZXJ0KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xTYW1wbGVDb3ZlcmFnZXgpKEdMY2xhbXB4IHZhbHVlLCBHTGJvb2xlYW4gaW52ZXJ0KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xTYW1wbGVDb3ZlcmFnZXgsIHZhbHVlLCBpbnZlcnQpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFNjYWxlZikoR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeikgewotICAgIENBTExfR0xfQVBJKGdsU2NhbGVmLCB4LCB5LCB6KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xTY2FsZXgpKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopIHsKLSAgICBDQUxMX0dMX0FQSShnbFNjYWxleCwgeCwgeSwgeik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsU2Npc3NvcikoR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQpIHsKLSAgICBDQUxMX0dMX0FQSShnbFNjaXNzb3IsIHgsIHksIHdpZHRoLCBoZWlnaHQpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFNoYWRlTW9kZWwpKEdMZW51bSBtb2RlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xTaGFkZU1vZGVsLCBtb2RlKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xTdGVuY2lsRnVuYykoR0xlbnVtIGZ1bmMsIEdMaW50IHJlZiwgR0x1aW50IG1hc2spIHsKLSAgICBDQUxMX0dMX0FQSShnbFN0ZW5jaWxGdW5jLCBmdW5jLCByZWYsIG1hc2spOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFN0ZW5jaWxNYXNrKShHTHVpbnQgbWFzaykgewotICAgIENBTExfR0xfQVBJKGdsU3RlbmNpbE1hc2ssIG1hc2spOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFN0ZW5jaWxPcCkoR0xlbnVtIGZhaWwsIEdMZW51bSB6ZmFpbCwgR0xlbnVtIHpwYXNzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xTdGVuY2lsT3AsIGZhaWwsIHpmYWlsLCB6cGFzcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4Q29vcmRQb2ludGVyKSggR0xpbnQgc2l6ZSwgR0xlbnVtIHR5cGUsCi0gICAgICAgICAgICAgICAgICAgICAgICBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUZXhDb29yZFBvaW50ZXIsIHNpemUsIHR5cGUsIHN0cmlkZSwgcG9pbnRlcik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4RW52ZikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUZXhFbnZmLCB0YXJnZXQsIHBuYW1lLCBwYXJhbSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4RW52ZnYpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUZXhFbnZmdiwgdGFyZ2V0LCBwbmFtZSwgcGFyYW1zKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUZXhFbnZ4KShHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleEVudngsIHRhcmdldCwgcG5hbWUsIHBhcmFtKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUZXhFbnZ4dikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleEVudnh2LCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFRleEltYWdlMkQpKCAgR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMaW50IGludGVybmFsZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwgR0xpbnQgYm9yZGVyLCBHTGVudW0gZm9ybWF0LAotICAgICAgICAgICAgICAgICAgICBHTGVudW0gdHlwZSwgY29uc3QgR0x2b2lkICpwaXhlbHMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleEltYWdlMkQsIHRhcmdldCwgbGV2ZWwsIGludGVybmFsZm9ybWF0LCB3aWR0aCwgaGVpZ2h0LAotICAgICAgICAgICAgYm9yZGVyLCBmb3JtYXQsIHR5cGUsIHBpeGVscyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4UGFyYW1ldGVyZikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUZXhQYXJhbWV0ZXJmLCB0YXJnZXQsIHBuYW1lLCBwYXJhbSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4UGFyYW1ldGVyeCkoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUZXhQYXJhbWV0ZXJ4LCB0YXJnZXQsIHBuYW1lLCBwYXJhbSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4U3ViSW1hZ2UyRCkoICAgR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMaW50IHhvZmZzZXQsCi0gICAgICAgICAgICAgICAgICAgICAgICBHTGludCB5b2Zmc2V0LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMZW51bSBmb3JtYXQsIEdMZW51bSB0eXBlLCBjb25zdCBHTHZvaWQgKnBpeGVscykgewotICAgIENBTExfR0xfQVBJKGdsVGV4U3ViSW1hZ2UyRCwgdGFyZ2V0LCBsZXZlbCwgeG9mZnNldCwgeW9mZnNldCwKLSAgICAgICAgICAgIHdpZHRoLCBoZWlnaHQsIGZvcm1hdCwgdHlwZSwgcGl4ZWxzKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUcmFuc2xhdGVmKShHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUcmFuc2xhdGVmLCB4LCB5LCB6KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUcmFuc2xhdGV4KShHTGZpeGVkIHgsIEdMZml4ZWQgeSwgR0xmaXhlZCB6KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUcmFuc2xhdGV4LCB4LCB5LCB6KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xWZXJ0ZXhQb2ludGVyKSggICBHTGludCBzaXplLCBHTGVudW0gdHlwZSwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIpIHsKLSAgICBDQUxMX0dMX0FQSShnbFZlcnRleFBvaW50ZXIsIHNpemUsIHR5cGUsIHN0cmlkZSwgcG9pbnRlcik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVmlld3BvcnQpKEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xWaWV3cG9ydCwgeCwgeSwgd2lkdGgsIGhlaWdodCk7Ci19Ci0KLS8vIEVTIDEuMQotdm9pZCBBUElfRU5UUlkoZ2xDbGlwUGxhbmVmKShHTGVudW0gcGxhbmUsIGNvbnN0IEdMZmxvYXQgKmVxdWF0aW9uKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xDbGlwUGxhbmVmLCBwbGFuZSwgZXF1YXRpb24pOwotfQotdm9pZCBBUElfRU5UUlkoZ2xDbGlwUGxhbmV4KShHTGVudW0gcGxhbmUsIGNvbnN0IEdMZml4ZWQgKmVxdWF0aW9uKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xDbGlwUGxhbmV4LCBwbGFuZSwgZXF1YXRpb24pOwotfQotdm9pZCBBUElfRU5UUlkoZ2xCaW5kQnVmZmVyKShHTGVudW0gdGFyZ2V0LCBHTHVpbnQgYnVmZmVyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xCaW5kQnVmZmVyLCB0YXJnZXQsIGJ1ZmZlcik7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEJ1ZmZlckRhdGEpKEdMZW51bSB0YXJnZXQsIEdMc2l6ZWlwdHIgc2l6ZSwgY29uc3QgR0x2b2lkKiBkYXRhLCBHTGVudW0gdXNhZ2UpIHsKLSAgICBDQUxMX0dMX0FQSShnbEJ1ZmZlckRhdGEsIHRhcmdldCwgc2l6ZSwgZGF0YSwgdXNhZ2UpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xCdWZmZXJTdWJEYXRhKShHTGVudW0gdGFyZ2V0LCBHTGludHB0ciBvZmZzZXQsIEdMc2l6ZWlwdHIgc2l6ZSwgY29uc3QgR0x2b2lkKiBkYXRhKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xCdWZmZXJTdWJEYXRhLCB0YXJnZXQsIG9mZnNldCwgc2l6ZSwgZGF0YSk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbERlbGV0ZUJ1ZmZlcnMpKEdMc2l6ZWkgbiwgY29uc3QgR0x1aW50KiBidWZmZXJzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEZWxldGVCdWZmZXJzLCBuLCBidWZmZXJzKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2VuQnVmZmVycykoR0xzaXplaSBuLCBHTHVpbnQqIGJ1ZmZlcnMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdlbkJ1ZmZlcnMsIG4sIGJ1ZmZlcnMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRCb29sZWFudikoR0xlbnVtIHBuYW1lLCBHTGJvb2xlYW4gKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0Qm9vbGVhbnYsIHBuYW1lLCBwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRGaXhlZHYpKEdMZW51bSBwbmFtZSwgR0xmaXhlZCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZXRGaXhlZHYsIHBuYW1lLCBwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRGbG9hdHYpKEdMZW51bSBwbmFtZSwgR0xmbG9hdCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZXRGbG9hdHYsIHBuYW1lLCBwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRQb2ludGVydikoR0xlbnVtIHBuYW1lLCB2b2lkICoqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZXRQb2ludGVydiwgcG5hbWUsIHBhcmFtcyk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEdldEJ1ZmZlclBhcmFtZXRlcml2KShHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMaW50ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdldEJ1ZmZlclBhcmFtZXRlcml2LCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRDbGlwUGxhbmVmKShHTGVudW0gcG5hbWUsIEdMZmxvYXQgZXFuWzRdKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZXRDbGlwUGxhbmVmLCBwbmFtZSwgZXFuKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0Q2xpcFBsYW5leCkoR0xlbnVtIHBuYW1lLCBHTGZpeGVkIGVxbls0XSkgewotICAgIENBTExfR0xfQVBJKGdsR2V0Q2xpcFBsYW5leCwgcG5hbWUsIGVxbik7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEdldExpZ2h0eHYpKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdldExpZ2h0eHYsIGxpZ2h0LCBwbmFtZSwgcGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0TGlnaHRmdikoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0TGlnaHRmdiwgbGlnaHQsIHBuYW1lLCBwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRNYXRlcmlhbHh2KShHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdldE1hdGVyaWFseHYsIGZhY2UsIHBuYW1lLCBwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRNYXRlcmlhbGZ2KShHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdldE1hdGVyaWFsZnYsIGZhY2UsIHBuYW1lLCBwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRUZXhFbnZmdikoR0xlbnVtIGVudiwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdldFRleEVudmZ2LCBlbnYsIHBuYW1lLCBwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRUZXhFbnZpdikoR0xlbnVtIGVudiwgR0xlbnVtIHBuYW1lLCBHTGludCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZXRUZXhFbnZpdiwgZW52LCBwbmFtZSwgcGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0VGV4RW52eHYpKEdMZW51bSBlbnYsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZXRUZXhFbnZ4diwgZW52LCBwbmFtZSwgcGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0VGV4UGFyYW1ldGVyZnYpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmbG9hdCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZXRUZXhQYXJhbWV0ZXJmdiwgdGFyZ2V0LCBwbmFtZSwgcGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0VGV4UGFyYW1ldGVyaXYpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xpbnQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0VGV4UGFyYW1ldGVyaXYsIHRhcmdldCwgcG5hbWUsIHBhcmFtcyk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEdldFRleFBhcmFtZXRlcnh2KShHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0VGV4UGFyYW1ldGVyeHYsIHRhcmdldCwgcG5hbWUsIHBhcmFtcyk7Ci19Ci1HTGJvb2xlYW4gQVBJX0VOVFJZKGdsSXNCdWZmZXIpKEdMdWludCBidWZmZXIpIHsKLSAgICBDQUxMX0dMX0FQSV9SRVRVUk4oZ2xJc0J1ZmZlciwgYnVmZmVyKTsKLX0KLUdMYm9vbGVhbiBBUElfRU5UUlkoZ2xJc0VuYWJsZWQpKEdMZW51bSBjYXApIHsKLSAgICBDQUxMX0dMX0FQSV9SRVRVUk4oZ2xJc0VuYWJsZWQsIGNhcCk7Ci19Ci1HTGJvb2xlYW4gQVBJX0VOVFJZKGdsSXNUZXh0dXJlKShHTHVpbnQgdGV4dHVyZSkgewotICAgIENBTExfR0xfQVBJX1JFVFVSTihnbElzVGV4dHVyZSwgdGV4dHVyZSk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbFBvaW50UGFyYW1ldGVyZikoR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xQb2ludFBhcmFtZXRlcmYsIHBuYW1lLCBwYXJhbSk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbFBvaW50UGFyYW1ldGVyZnYpKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xQb2ludFBhcmFtZXRlcmZ2LCBwbmFtZSwgcGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsUG9pbnRQYXJhbWV0ZXJ4KShHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbFBvaW50UGFyYW1ldGVyeCwgcG5hbWUsIHBhcmFtKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsUG9pbnRQYXJhbWV0ZXJ4dikoR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFBvaW50UGFyYW1ldGVyeHYsIHBuYW1lLCBwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xDb2xvcjR1YikoR0x1Ynl0ZSByZWQsIEdMdWJ5dGUgZ3JlZW4sIEdMdWJ5dGUgYmx1ZSwgR0x1Ynl0ZSBhbHBoYSkgewotICAgIENBTExfR0xfQVBJKGdsQ29sb3I0dWIsIHJlZCwgZ3JlZW4sIGJsdWUsIGFscGhhKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4RW52aSkoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGludCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsVGV4RW52aSwgdGFyZ2V0LCBwbmFtZSwgcGFyYW0pOwotfQotdm9pZCBBUElfRU5UUlkoZ2xUZXhFbnZpdikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGludCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUZXhFbnZpdiwgdGFyZ2V0LCBwbmFtZSwgcGFyYW1zKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUZXhQYXJhbWV0ZXJmdikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleFBhcmFtZXRlcmZ2LCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFRleFBhcmFtZXRlcml2KShHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMaW50ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleFBhcmFtZXRlcml2LCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFRleFBhcmFtZXRlcmkpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xpbnQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleFBhcmFtZXRlcmksIHRhcmdldCwgcG5hbWUsIHBhcmFtKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4UGFyYW1ldGVyeHYpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUZXhQYXJhbWV0ZXJ4diwgdGFyZ2V0LCBwbmFtZSwgcGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsUG9pbnRTaXplUG9pbnRlck9FUykoR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIpIHsKLSAgICBDQUxMX0dMX0FQSShnbFBvaW50U2l6ZVBvaW50ZXJPRVMsIHR5cGUsIHN0cmlkZSwgcG9pbnRlcik7Ci19Ci0KLS8vIEV4dGVuc2lvbnMKLXZvaWQgQVBJX0VOVFJZKGdsRHJhd1RleHNPRVMpKEdMc2hvcnQgeCAsIEdMc2hvcnQgeSwgR0xzaG9ydCB6LCBHTHNob3J0IHcsIEdMc2hvcnQgaCkgewotICAgIENBTExfR0xfQVBJKGdsRHJhd1RleHNPRVMsIHgsIHksIHosIHcsIGgpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xEcmF3VGV4aU9FUykoR0xpbnQgeCwgR0xpbnQgeSwgR0xpbnQgeiwgR0xpbnQgdywgR0xpbnQgaCkgewotICAgIENBTExfR0xfQVBJKGdsRHJhd1RleGlPRVMsIHgsIHksIHosIHcsIGgpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xEcmF3VGV4Zk9FUykoR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeiwgR0xmbG9hdCB3LCBHTGZsb2F0IGgpIHsKLSAgICBDQUxMX0dMX0FQSShnbERyYXdUZXhmT0VTLCB4LCB5LCB6LCB3LCBoKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsRHJhd1RleHhPRVMpKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHosIEdMZml4ZWQgdywgR0xmaXhlZCBoKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEcmF3VGV4eE9FUywgeCwgeSwgeiwgdywgaCk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbERyYXdUZXhzdk9FUykoY29uc3QgR0xzaG9ydCogY29vcmRzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEcmF3VGV4c3ZPRVMsIGNvb3Jkcyk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbERyYXdUZXhpdk9FUykoY29uc3QgR0xpbnQqIGNvb3JkcykgewotICAgIENBTExfR0xfQVBJKGdsRHJhd1RleGl2T0VTLCBjb29yZHMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xEcmF3VGV4ZnZPRVMpKGNvbnN0IEdMZmxvYXQqIGNvb3JkcykgewotICAgIENBTExfR0xfQVBJKGdsRHJhd1RleGZ2T0VTLCBjb29yZHMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xEcmF3VGV4eHZPRVMpKGNvbnN0IEdMZml4ZWQqIGNvb3JkcykgewotICAgIENBTExfR0xfQVBJKGdsRHJhd1RleHh2T0VTLCBjb29yZHMpOwotfQotR0xiaXRmaWVsZCBBUElfRU5UUlkoZ2xRdWVyeU1hdHJpeHhPRVMpKEdMZml4ZWQqIG1hbnRpc3NhLCBHTGludCogZXhwb25lbnQpIHsKLSAgICBDQUxMX0dMX0FQSV9SRVRVUk4oZ2xRdWVyeU1hdHJpeHhPRVMsIG1hbnRpc3NhLCBleHBvbmVudCk7Ci19CmRpZmYgLS1naXQgYS9vcGVuZ2wvbGlicy9HTEVTX0NNL2dsX2xvZ2dlci5jcHAgYi9vcGVuZ2wvbGlicy9HTEVTX0NNL2dsX2xvZ2dlci5jcHAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDI3YmU1YzkuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYnMvR0xFU19DTS9nbF9sb2dnZXIuY3BwCisrKyAvZGV2L251bGwKQEAgLTEsMTA2MCArMCwwIEBACi0vKgotICoqIENvcHlyaWdodCAyMDA3LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0gKioKLSAqKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKLSAqKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCi0gKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKioKLSAqKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCi0gKioKLSAqKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKLSAqKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KLSAqKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2RlZmluZSBMT0dfVEFHICJHTExvZ2dlciIKLQotI2luY2x1ZGUgPGN0eXBlLmg+Ci0jaW5jbHVkZSA8c3RyaW5nLmg+Ci0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxkbGZjbi5oPgotCi0jaW5jbHVkZSA8c3lzL2lvY3RsLmg+Ci0KLSNpbmNsdWRlIDxFR0wvZWdsLmg+Ci0jaW5jbHVkZSA8RUdML2VnbGV4dC5oPgotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLSNpbmNsdWRlIDxHTEVTL2dsZXh0Lmg+Ci0KLSNpbmNsdWRlIDxjdXRpbHMvbG9nLmg+Ci0jaW5jbHVkZSA8Y3V0aWxzL2F0b21pYy5oPgotI2luY2x1ZGUgPGN1dGlscy9wcm9wZXJ0aWVzLmg+Ci0KLSNpbmNsdWRlIDx1dGlscy9TdHJpbmc4Lmg+Ci0KLSNpbmNsdWRlICJnbF9sb2dnZXIuaCIKLQotI3VuZGVmIE5FTEVNCi0jZGVmaW5lIE5FTEVNKHgpIChzaXplb2YoeCkvc2l6ZW9mKCooeCkpKQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXRlbXBsYXRlPHR5cGVuYW1lIFQ+Ci1zdGF0aWMgaW50IGJpbmFyeVNlYXJjaChUIGNvbnN0IHNvcnRlZEFycmF5W10sIGludCBmaXJzdCwgaW50IGxhc3QsIEVHTGludCBrZXkpCi17Ci0gICB3aGlsZSAoZmlyc3QgPD0gbGFzdCkgewotICAgICAgIGludCBtaWQgPSAoZmlyc3QgKyBsYXN0KSAvIDI7Ci0gICAgICAgaWYgKGtleSA+IHNvcnRlZEFycmF5W21pZF0ua2V5KSB7Ci0gICAgICAgICAgIGZpcnN0ID0gbWlkICsgMTsKLSAgICAgICB9IGVsc2UgaWYgKGtleSA8IHNvcnRlZEFycmF5W21pZF0ua2V5KSB7Ci0gICAgICAgICAgIGxhc3QgPSBtaWQgLSAxOwotICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgIHJldHVybiBtaWQ7Ci0gICAgICAgfQotICAgfQotICAgcmV0dXJuIC0xOwotfQotCi1zdHJ1Y3QgcGFpcl90IHsKLSAgICBjb25zdCBjaGFyKiBuYW1lOwotICAgIGludCAgICAgICAgIGtleTsKLX07Ci0KLXN0YXRpYyBjb25zdCBwYWlyX3QgZ0VudW1NYXBbXSA9IHsKLSAgICAjZGVmaW5lIEdMRU5VTShOQU1FLCBWQUxVRSkgeyAjTkFNRSwgVkFMVUUgfSwKLSAgICAjaW5jbHVkZSAiZ2xfZW51bXMuaW4iCi0gICAgI3VuZGVmIEdMRU5VTQotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi10ZW1wbGF0ZTx0eXBlbmFtZSBUWVBFPgotY2xhc3MgR0xMb2dWYWx1ZSB7Ci1wdWJsaWM6Ci0gICAgR0xMb2dWYWx1ZShUWVBFIHZhbHVlKSA6IG1WYWx1ZSh2YWx1ZSkgeyB9Ci0gICAgY29uc3QgVFlQRSYgZ2V0VmFsdWUoKSBjb25zdCB7IHJldHVybiBtVmFsdWU7IH0KLSAgICBTdHJpbmc4IHRvU3RyaW5nKCkgY29uc3QgewotICAgICAgICByZXR1cm4gY29udmVydFRvU3RyaW5nKG1WYWx1ZSk7Ci0gICAgfQotcHJpdmF0ZToKLSAgICBjb25zdCBUWVBFJiBtVmFsdWU7Ci0gICAgU3RyaW5nOCBjb252ZXJ0VG9TdHJpbmcodW5zaWduZWQgaW50IHYpIGNvbnN0IHsKLSAgICAgICAgY2hhciBidWZbMTZdOwotICAgICAgICBzbnByaW50ZihidWYsIDE2LCAiJXUiLCB2KTsKLSAgICAgICAgcmV0dXJuIFN0cmluZzgoYnVmKTsKLSAgICB9Ci0gICAgU3RyaW5nOCBjb252ZXJ0VG9TdHJpbmcodW5zaWduZWQgbG9uZyB2KSBjb25zdCB7Ci0gICAgICAgIGNoYXIgYnVmWzE2XTsKLSAgICAgICAgc25wcmludGYoYnVmLCAxNiwgIiVsdSIsIHYpOwotICAgICAgICByZXR1cm4gU3RyaW5nOChidWYpOwotICAgIH0KLSAgICBTdHJpbmc4IGNvbnZlcnRUb1N0cmluZyhpbnQgdikgY29uc3QgewotICAgICAgICBjaGFyIGJ1ZlsxNl07Ci0gICAgICAgIHNucHJpbnRmKGJ1ZiwgMTYsICIlZCIsIHYpOwotICAgICAgICByZXR1cm4gU3RyaW5nOChidWYpOwotICAgIH0KLSAgICBTdHJpbmc4IGNvbnZlcnRUb1N0cmluZyhsb25nIHYpIGNvbnN0IHsKLSAgICAgICAgY2hhciBidWZbMTZdOwotICAgICAgICBzbnByaW50ZihidWYsIDE2LCAiJWxkIiwgdik7Ci0gICAgICAgIHJldHVybiBTdHJpbmc4KGJ1Zik7Ci0gICAgfQotICAgIFN0cmluZzggY29udmVydFRvU3RyaW5nKGZsb2F0IHYpIGNvbnN0IHsKLSAgICAgICAgY2hhciBidWZbMTZdOwotICAgICAgICBzbnByaW50ZihidWYsIDE2LCAiJWYiLCB2KTsKLSAgICAgICAgcmV0dXJuIFN0cmluZzgoYnVmKTsKLSAgICB9Ci0gICAgU3RyaW5nOCBjb252ZXJ0VG9TdHJpbmcodm9pZCBjb25zdCogdikgY29uc3QgewotICAgICAgICBjaGFyIGJ1ZlsxNl07Ci0gICAgICAgIHNucHJpbnRmKGJ1ZiwgMTYsICIlcCIsIHYpOwotICAgICAgICByZXR1cm4gU3RyaW5nOChidWYpOwotICAgIH0KLX07Ci0KLWNsYXNzIEdMTG9nRW51bSA6IHB1YmxpYyBHTExvZ1ZhbHVlPEdMZW51bT4gewotcHVibGljOgotICAgIEdMTG9nRW51bShHTGVudW0gdikgOiBHTExvZ1ZhbHVlPEdMZW51bT4odikgeyB9Ci0gICAgU3RyaW5nOCB0b1N0cmluZygpIGNvbnN0IHsKLSAgICAgICAgR0xlbnVtIHYgPSBnZXRWYWx1ZSgpOwotICAgICAgICBpbnQgaSA9IGJpbmFyeVNlYXJjaDxwYWlyX3Q+KGdFbnVtTWFwLCAwLCBORUxFTShnRW51bU1hcCktMSwgdik7Ci0gICAgICAgIGlmIChpID49IDApIHsKLSAgICAgICAgICAgIHJldHVybiBTdHJpbmc4KGdFbnVtTWFwW2ldLm5hbWUpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgY2hhciBidWZbMTZdOwotICAgICAgICAgICAgc25wcmludGYoYnVmLCAxNiwgIjB4JTA0eCIsIHYpOwotICAgICAgICAgICAgcmV0dXJuIFN0cmluZzgoYnVmKTsKLSAgICAgICAgfQotICAgIH0KLX07Ci0KLWNsYXNzIEdMTG9nQ2xlYXJCaXRmaWVsZCA6IHB1YmxpYyBHTExvZ1ZhbHVlPEdMYml0ZmllbGQ+IHsKLXB1YmxpYzoKLSAgICBHTExvZ0NsZWFyQml0ZmllbGQoR0xiaXRmaWVsZCB2KSA6IEdMTG9nVmFsdWU8R0xiaXRmaWVsZD4odikgeyB9Ci0gICAgU3RyaW5nOCB0b1N0cmluZygpIGNvbnN0IHsKLSAgICAgICAgY2hhciBidWZbMTZdOwotICAgICAgICBzbnByaW50ZihidWYsIDE2LCAiMHglMDh4IiwgZ2V0VmFsdWUoKSk7Ci0gICAgICAgIHJldHVybiBTdHJpbmc4KGJ1Zik7Ci0gICAgfQotfTsKLQotY2xhc3MgR0xMb2dCb29sIDogcHVibGljIEdMTG9nVmFsdWU8R0xib29sZWFuPiB7Ci1wdWJsaWM6Ci0gICAgR0xMb2dCb29sKEdMYm9vbGVhbiB2KSA6IEdMTG9nVmFsdWU8R0xib29sZWFuPih2KSB7IH0KLSAgICBTdHJpbmc4IHRvU3RyaW5nKCkgY29uc3QgewotICAgICAgICBHTGJvb2xlYW4gdiA9IGdldFZhbHVlKCk7Ci0gICAgICAgIGlmICh2ID09IEdMX1RSVUUpICAgcmV0dXJuIFN0cmluZzgoIkdMX1RSVUUiKTsKLSAgICAgICAgaWYgKHYgPT0gR0xfRkFMU0UpICByZXR1cm4gU3RyaW5nOCgiR0xfRkFMU0UiKTsKLSAgICAgICAgcmV0dXJuIEdMTG9nVmFsdWU8R0xib29sZWFuPjo6dG9TdHJpbmcoKTsKLSAgICB9Ci19OwotCi1jbGFzcyBHTExvZ0ZpeGVkIDogcHVibGljIEdMTG9nVmFsdWU8R0xmaXhlZD4gewotcHVibGljOgotICAgIEdMTG9nRml4ZWQoR0xmaXhlZCB2KSA6IEdMTG9nVmFsdWU8R0xmaXhlZD4odikgeyB9Ci0gICAgU3RyaW5nOCB0b1N0cmluZygpIGNvbnN0IHsKLSAgICAgICAgY2hhciBidWZbMTZdOwotICAgICAgICBzbnByaW50ZihidWYsIDE2LCAiMHglMDh4IiwgZ2V0VmFsdWUoKSk7Ci0gICAgICAgIHJldHVybiBTdHJpbmc4KGJ1Zik7Ci0gICAgfQotfTsKLQotCi10ZW1wbGF0ZSA8dHlwZW5hbWUgVFlQRT4KLWNsYXNzIEdMTG9nQnVmZmVyIDogcHVibGljIEdMTG9nVmFsdWU8VFlQRSAqPiB7Ci1wdWJsaWM6Ci0gICAgR0xMb2dCdWZmZXIoVFlQRSogYnVmZmVyLCBzaXplX3QgY291bnQgPSAtMSkKLSAgICAgICAgOiBHTExvZ1ZhbHVlPFRZUEUqPihidWZmZXIpCi0gICAgeyAvLyBvdXRwdXQgYnVmZmVyCi0gICAgfQotICAgIEdMTG9nQnVmZmVyKFRZUEUgY29uc3QqIGJ1ZmZlciwgc2l6ZV90IGNvdW50ID0gLTEpCi0gICAgOiBHTExvZ1ZhbHVlPFRZUEUqPihjb25zdF9jYXN0PFRZUEUqPihidWZmZXIpKQotICAgIHsgLy8gaW5wdXQgYnVmZmVyCi0gICAgfQotfTsKLQotY2xhc3MgR0xMb2cKLXsKLXB1YmxpYzoKLSAgICBHTExvZyhjb25zdCBjaGFyKiBuYW1lKSA6IG1OdW1QYXJhbXMoMCkgewotICAgICAgICBtU3RyaW5nLmFwcGVuZChuYW1lKTsKLSAgICAgICAgbVN0cmluZy5hcHBlbmQoIigiKTsKLSAgICB9Ci0KLSAgICB+R0xMb2coKSB7Ci0gICAgICAgIExPR0QoIiVzKTsiLCBtU3RyaW5nLnN0cmluZygpKTsKLSAgICB9Ci0KLSAgICBHTExvZyYgb3BlcmF0b3IgPDwgKHVuc2lnbmVkIGNoYXIgdikgewotICAgICAgICByZXR1cm4gKnRoaXMgPDwgR0xMb2dWYWx1ZTx1bnNpZ25lZCBpbnQ+KHYpOwotICAgIH0KLSAgICBHTExvZyYgb3BlcmF0b3IgPDwgKHNob3J0IHYpIHsKLSAgICAgICAgcmV0dXJuICp0aGlzIDw8IEdMTG9nVmFsdWU8dW5zaWduZWQgaW50Pih2KTsKLSAgICB9Ci0gICAgR0xMb2cmIG9wZXJhdG9yIDw8ICh1bnNpZ25lZCBpbnQgdikgewotICAgICAgICByZXR1cm4gKnRoaXMgPDwgR0xMb2dWYWx1ZTx1bnNpZ25lZCBpbnQ+KHYpOwotICAgIH0KLSAgICBHTExvZyYgb3BlcmF0b3IgPDwgKGludCB2KSB7Ci0gICAgICAgIHJldHVybiAqdGhpcyA8PCBHTExvZ1ZhbHVlPGludD4odik7Ci0gICAgfQotICAgIEdMTG9nJiBvcGVyYXRvciA8PCAobG9uZyB2KSB7Ci0gICAgICAgIHJldHVybiAqdGhpcyA8PCBHTExvZ1ZhbHVlPGxvbmc+KHYpOwotICAgIH0KLSAgICBHTExvZyYgb3BlcmF0b3IgPDwgKHVuc2lnbmVkIGxvbmcgdikgewotICAgICAgICByZXR1cm4gKnRoaXMgPDwgR0xMb2dWYWx1ZTx1bnNpZ25lZCBsb25nPih2KTsKLSAgICB9Ci0gICAgR0xMb2cmIG9wZXJhdG9yIDw8IChmbG9hdCB2KSB7Ci0gICAgICAgIHJldHVybiAqdGhpcyA8PCBHTExvZ1ZhbHVlPGZsb2F0Pih2KTsKLSAgICB9Ci0gICAgR0xMb2cmIG9wZXJhdG9yIDw8IChjb25zdCB2b2lkKiB2KSB7Ci0gICAgICAgIHJldHVybiAqdGhpcyA8PCBHTExvZ1ZhbHVlPGNvbnN0IHZvaWQqID4odik7Ci0gICAgfQotCi0gICAgdGVtcGxhdGUgPHR5cGVuYW1lIFRZUEU+Ci0gICAgR0xMb2cmIG9wZXJhdG9yIDw8IChjb25zdCBUWVBFJiByaHMpIHsKLSAgICAgICAgaWYgKG1OdW1QYXJhbXMgPiAwKQotICAgICAgICAgICAgbVN0cmluZy5hcHBlbmQoIiwgIik7Ci0gICAgICAgIG1TdHJpbmcuYXBwZW5kKHJocy50b1N0cmluZygpKTsKLSAgICAgICAgbU51bVBhcmFtcysrOwotICAgICAgICByZXR1cm4gKnRoaXM7Ci0gICAgfQotCi0gICAgY29uc3QgU3RyaW5nOCYgc3RyaW5nKCkgY29uc3QgeyByZXR1cm4gbVN0cmluZzsgfQotcHJpdmF0ZToKLSAgICBHTExvZyhjb25zdCBHTExvZyYpOwotCi0gICAgU3RyaW5nOCBtU3RyaW5nOwotICAgIGludCBtTnVtUGFyYW1zOwotfTsKLQotI2RlZmluZSBBUElfRU5UUlkoYXBpKSAgICAgICAgICAgICAgICAgICAgICBsb2dfIyNhcGkKLSNkZWZpbmUgQ0FMTF9HTF9BUEkoX3gsIC4uLikKLSNkZWZpbmUgQ0FMTF9HTF9BUElfUkVUVVJOKF94LCAuLi4pICAgICAgICAgcmV0dXJuKDApOwotCi12b2lkIEFQSV9FTlRSWShnbEFjdGl2ZVRleHR1cmUpKEdMZW51bSB0ZXh0dXJlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xBY3RpdmVUZXh0dXJlLCB0ZXh0dXJlKTsKLSAgICBHTExvZygiZ2xBY3RpdmVUZXh0dXJlIikgPDwgR0xMb2dFbnVtKHRleHR1cmUpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEFscGhhRnVuYykoR0xlbnVtIGZ1bmMsIEdMY2xhbXBmIHJlZikgewotICAgIENBTExfR0xfQVBJKGdsQWxwaGFGdW5jLCBmdW5jLCByZWYpOwotICAgIEdMTG9nKCJnbEFscGhhRnVuYyIpIDw8IEdMTG9nRW51bShmdW5jKSA8PCByZWY7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQWxwaGFGdW5jeCkoR0xlbnVtIGZ1bmMsIEdMY2xhbXB4IHJlZikgewotICAgIENBTExfR0xfQVBJKGdsQWxwaGFGdW5jeCwgZnVuYywgcmVmKTsKLSAgICBHTExvZygiZ2xBbHBoYUZ1bmN4IikgPDwgR0xMb2dFbnVtKGZ1bmMpIDw8IEdMTG9nRml4ZWQocmVmKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xCaW5kVGV4dHVyZSkoR0xlbnVtIHRhcmdldCwgR0x1aW50IHRleHR1cmUpIHsKLSAgICBDQUxMX0dMX0FQSShnbEJpbmRUZXh0dXJlLCB0YXJnZXQsIHRleHR1cmUpOwotICAgIEdMTG9nKCJnbEJpbmRUZXh0dXJlIikgPDwgR0xMb2dFbnVtKHRhcmdldCkgPDwgdGV4dHVyZTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xCbGVuZEZ1bmMpKEdMZW51bSBzZmFjdG9yLCBHTGVudW0gZGZhY3RvcikgewotICAgIENBTExfR0xfQVBJKGdsQmxlbmRGdW5jLCBzZmFjdG9yLCBkZmFjdG9yKTsKLSAgICBHTExvZygiZ2xCbGVuZEZ1bmMiKSA8PCBHTExvZ0VudW0oc2ZhY3RvcikgPDwgR0xMb2dFbnVtKGRmYWN0b3IpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENsZWFyKShHTGJpdGZpZWxkIG1hc2spIHsKLSAgICBDQUxMX0dMX0FQSShnbENsZWFyLCBtYXNrKTsKLSAgICBHTExvZygiZ2xDbGVhciIpIDw8IEdMTG9nQ2xlYXJCaXRmaWVsZChtYXNrKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDbGVhckNvbG9yKShHTGNsYW1wZiByZWQsIEdMY2xhbXBmIGdyZWVuLCBHTGNsYW1wZiBibHVlLCBHTGNsYW1wZiBhbHBoYSkgewotICAgIENBTExfR0xfQVBJKGdsQ2xlYXJDb2xvciwgcmVkLCBncmVlbiwgYmx1ZSwgYWxwaGEpOwotICAgIEdMTG9nKCJnbENsZWFyQ29sb3IiKSA8PCByZWQgPDwgZ3JlZW4gPDwgYmx1ZSA8PCBhbHBoYTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDbGVhckNvbG9yeCkoR0xjbGFtcHggcmVkLCBHTGNsYW1weCBncmVlbiwgR0xjbGFtcHggYmx1ZSwgR0xjbGFtcHggYWxwaGEpIHsKLSAgICBDQUxMX0dMX0FQSShnbENsZWFyQ29sb3J4LCByZWQsIGdyZWVuLCBibHVlLCBhbHBoYSk7Ci0gICAgR0xMb2coImdsQ2xlYXJDb2xvcngiKSA8PCBHTExvZ0ZpeGVkKHJlZCkgPDwgR0xMb2dGaXhlZChncmVlbikgPDwgR0xMb2dGaXhlZChibHVlKSA8PCBHTExvZ0ZpeGVkKGFscGhhKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDbGVhckRlcHRoZikoR0xjbGFtcGYgZGVwdGgpIHsKLSAgICBDQUxMX0dMX0FQSShnbENsZWFyRGVwdGhmLCBkZXB0aCk7Ci0gICAgR0xMb2coImdsQ2xlYXJEZXB0aGYiKSA8PCBkZXB0aDsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDbGVhckRlcHRoeCkoR0xjbGFtcHggZGVwdGgpIHsKLSAgICBDQUxMX0dMX0FQSShnbENsZWFyRGVwdGh4LCBkZXB0aCk7Ci0gICAgR0xMb2coImdsQ2xlYXJEZXB0aHgiKSA8PCBHTExvZ0ZpeGVkKGRlcHRoKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDbGVhclN0ZW5jaWwpKEdMaW50IHMpIHsKLSAgICBDQUxMX0dMX0FQSShnbENsZWFyU3RlbmNpbCwgcyk7Ci0gICAgR0xMb2coImdsQ2xlYXJTdGVuY2lsIikgPDwgczsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDbGllbnRBY3RpdmVUZXh0dXJlKShHTGVudW0gdGV4dHVyZSkgewotICAgIENBTExfR0xfQVBJKGdsQ2xpZW50QWN0aXZlVGV4dHVyZSwgdGV4dHVyZSk7Ci0gICAgR0xMb2coImdsQ2xpZW50QWN0aXZlVGV4dHVyZSIpIDw8IEdMTG9nRW51bSh0ZXh0dXJlKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDb2xvcjRmKShHTGZsb2F0IHJlZCwgR0xmbG9hdCBncmVlbiwgR0xmbG9hdCBibHVlLCBHTGZsb2F0IGFscGhhKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xDb2xvcjRmLCByZWQsIGdyZWVuLCBibHVlLCBhbHBoYSk7Ci0gICAgR0xMb2coImdsQ29sb3I0ZiIpIDw8IHJlZCA8PCBncmVlbiA8PCBibHVlIDw8IGFscGhhOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENvbG9yNHgpKEdMZml4ZWQgcmVkLCBHTGZpeGVkIGdyZWVuLCBHTGZpeGVkIGJsdWUsIEdMZml4ZWQgYWxwaGEpIHsKLSAgICBDQUxMX0dMX0FQSShnbENvbG9yNHgsIHJlZCwgZ3JlZW4sIGJsdWUsIGFscGhhKTsKLSAgICBHTExvZygiZ2xDb2xvcjR4IikgPDwgR0xMb2dGaXhlZChyZWQpIDw8IEdMTG9nRml4ZWQoZ3JlZW4pIDw8IEdMTG9nRml4ZWQoYmx1ZSkgPDwgR0xMb2dGaXhlZChhbHBoYSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQ29sb3JNYXNrKShHTGJvb2xlYW4gciwgR0xib29sZWFuIGcsIEdMYm9vbGVhbiBiLCBHTGJvb2xlYW4gYSkgewotICAgIENBTExfR0xfQVBJKGdsQ29sb3JNYXNrLCByLCBnLCBiLCBhKTsKLSAgICBHTExvZygiZ2xDb2xvck1hc2siKSA8PCBHTExvZ0Jvb2wocikgPDwgR0xMb2dCb29sKGcpIDw8IEdMTG9nQm9vbChiKSA8PCBHTExvZ0Jvb2woYSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQ29sb3JQb2ludGVyKShHTGludCBzaXplLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcHRyKQotewotICAgIENBTExfR0xfQVBJKGdsQ29sb3JQb2ludGVyLCBzaXplLCB0eXBlLCBzdHJpZGUsIHB0cik7Ci0gICAgR0xMb2coImdsQ29sb3JQb2ludGVyIikgPDwgc2l6ZSA8PCBHTExvZ0VudW0odHlwZSkgPDwgc3RyaWRlIDw8IHB0cjsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDb21wcmVzc2VkVGV4SW1hZ2UyRCkoR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMZW51bSBpbnRlcm5hbGZvcm1hdCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwgR0xpbnQgYm9yZGVyLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMc2l6ZWkgaW1hZ2VTaXplLCBjb25zdCBHTHZvaWQgKmRhdGEpIHsKLSAgICBDQUxMX0dMX0FQSShnbENvbXByZXNzZWRUZXhJbWFnZTJELCB0YXJnZXQsIGxldmVsLCBpbnRlcm5hbGZvcm1hdCwKLSAgICAgICAgICAgIHdpZHRoLCBoZWlnaHQsIGJvcmRlciwgaW1hZ2VTaXplLCBkYXRhKTsKLSAgICBHTExvZygiZ2xDb21wcmVzc2VkVGV4SW1hZ2UyRCIpCi0gICAgICAgICAgICAgICAgPDwgR0xMb2dFbnVtKHRhcmdldCkgPDwgbGV2ZWwgPDwgR0xMb2dFbnVtKGludGVybmFsZm9ybWF0KQotICAgICAgICAgICAgICAgIDw8IHdpZHRoIDw8IGhlaWdodCA8PCBib3JkZXIgPDwgaW1hZ2VTaXplIDw8IGRhdGE7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQ29tcHJlc3NlZFRleFN1YkltYWdlMkQpKCBHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgeG9mZnNldCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xpbnQgeW9mZnNldCwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdMZW51bSBmb3JtYXQsIEdMc2l6ZWkgaW1hZ2VTaXplLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBHTHZvaWQgKmRhdGEpIHsKLSAgICBDQUxMX0dMX0FQSShnbENvbXByZXNzZWRUZXhTdWJJbWFnZTJELCB0YXJnZXQsIGxldmVsLCB4b2Zmc2V0LCB5b2Zmc2V0LAotICAgICAgICAgICAgd2lkdGgsIGhlaWdodCwgZm9ybWF0LCBpbWFnZVNpemUsIGRhdGEpOwotICAgIEdMTG9nKCJnbENvbXByZXNzZWRUZXhTdWJJbWFnZTJEIikKLSAgICAgICAgICAgIDw8IEdMTG9nRW51bSh0YXJnZXQpIDw8IGxldmVsIDw8IHhvZmZzZXQgPDwgeW9mZnNldAotICAgICAgICAgICAgPDwgd2lkdGggPDwgaGVpZ2h0IDw8IEdMTG9nRW51bShmb3JtYXQpIDw8IGltYWdlU2l6ZSA8PCBkYXRhOwotfQotCi12b2lkIEFQSV9FTlRSWShnbENvcHlUZXhJbWFnZTJEKSggIEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGVudW0gaW50ZXJuYWxmb3JtYXQsCi0gICAgICAgICAgICAgICAgICAgICAgICBHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMaW50IGJvcmRlcikgewotICAgIENBTExfR0xfQVBJKGdsQ29weVRleEltYWdlMkQsIHRhcmdldCwgbGV2ZWwsIGludGVybmFsZm9ybWF0LCB4LCB5LAotICAgICAgICAgICAgd2lkdGgsIGhlaWdodCwgYm9yZGVyKTsKLSAgICBHTExvZygiZ2xDb3B5VGV4SW1hZ2UyRCIpCi0gICAgICAgICAgICA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBsZXZlbCA8PCBHTExvZ0VudW0oaW50ZXJuYWxmb3JtYXQpCi0gICAgICAgICAgICA8PCB4IDw8IHkgPDwgd2lkdGggPDwgaGVpZ2h0IDw8IGJvcmRlcjsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xDb3B5VGV4U3ViSW1hZ2UyRCkoICAgR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMaW50IHhvZmZzZXQsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgR0xpbnQgeW9mZnNldCwgR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3aWR0aCwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBHTHNpemVpIGhlaWdodCkgewotICAgIENBTExfR0xfQVBJKGdsQ29weVRleFN1YkltYWdlMkQsIHRhcmdldCwgbGV2ZWwsIHhvZmZzZXQsIHlvZmZzZXQsIHgsIHksCi0gICAgICAgICAgICB3aWR0aCwgaGVpZ2h0KTsKLSAgICBHTExvZygiZ2xDb3B5VGV4U3ViSW1hZ2UyRCIpCi0gICAgICAgICAgICA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBsZXZlbCA8PCB4b2Zmc2V0IDw8IHlvZmZzZXQKLSAgICAgICAgICAgIDw8IHggPDwgeSA8PCB3aWR0aCA8PCBoZWlnaHQ7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsQ3VsbEZhY2UpKEdMZW51bSBtb2RlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xDdWxsRmFjZSwgbW9kZSk7Ci0gICAgR0xMb2coImdsQ3VsbEZhY2UiKSA8PCBHTExvZ0VudW0obW9kZSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRGVsZXRlVGV4dHVyZXMpKEdMc2l6ZWkgbiwgY29uc3QgR0x1aW50ICp0ZXh0dXJlcykgewotICAgIENBTExfR0xfQVBJKGdsRGVsZXRlVGV4dHVyZXMsIG4sIHRleHR1cmVzKTsKLSAgICBHTExvZygiZ2xEZWxldGVUZXh0dXJlcyIpIDw8IG4gPDwgR0xMb2dCdWZmZXI8R0x1aW50Pih0ZXh0dXJlcywgbik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRGVwdGhGdW5jKShHTGVudW0gZnVuYykgewotICAgIENBTExfR0xfQVBJKGdsRGVwdGhGdW5jLCBmdW5jKTsKLSAgICBHTExvZygiZ2xEZXB0aEZ1bmMiKSA8PCBHTExvZ0VudW0oZnVuYyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRGVwdGhNYXNrKShHTGJvb2xlYW4gZmxhZykgewotICAgIENBTExfR0xfQVBJKGdsRGVwdGhNYXNrLCBmbGFnKTsKLSAgICBHTExvZygiZ2xEZXB0aE1hc2siKSA8PCBHTExvZ0Jvb2woZmxhZyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRGVwdGhSYW5nZWYpKEdMY2xhbXBmIHpOZWFyLCBHTGNsYW1wZiB6RmFyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEZXB0aFJhbmdlZiwgek5lYXIsIHpGYXIpOwotICAgIEdMTG9nKCJnbERlcHRoUmFuZ2VmIikgPDwgek5lYXIgPDwgekZhcjsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xEZXB0aFJhbmdleCkoR0xjbGFtcHggek5lYXIsIEdMY2xhbXB4IHpGYXIpIHsKLSAgICBDQUxMX0dMX0FQSShnbERlcHRoUmFuZ2V4LCB6TmVhciwgekZhcik7Ci0gICAgR0xMb2coImdsRGVwdGhSYW5nZXgiKSA8PCBHTExvZ0ZpeGVkKHpOZWFyKSA8PCBHTExvZ0ZpeGVkKHpGYXIpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbERpc2FibGUpKEdMZW51bSBjYXApIHsKLSAgICBDQUxMX0dMX0FQSShnbERpc2FibGUsIGNhcCk7Ci0gICAgR0xMb2coImdsRGlzYWJsZSIpIDw8IEdMTG9nRW51bShjYXApOwotfQotCi12b2lkIEFQSV9FTlRSWShnbERpc2FibGVDbGllbnRTdGF0ZSkoR0xlbnVtIGFycmF5KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEaXNhYmxlQ2xpZW50U3RhdGUsIGFycmF5KTsKLSAgICBHTExvZygiZ2xEaXNhYmxlQ2xpZW50U3RhdGUiKSA8PCBHTExvZ0VudW0oYXJyYXkpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbERyYXdBcnJheXMpKEdMZW51bSBtb2RlLCBHTGludCBmaXJzdCwgR0xzaXplaSBjb3VudCkgewotICAgIENBTExfR0xfQVBJKGdsRHJhd0FycmF5cywgbW9kZSwgZmlyc3QsIGNvdW50KTsKLSAgICBHTExvZygiZ2xEcmF3QXJyYXlzIikgPDwgR0xMb2dFbnVtKG1vZGUpIDw8IGZpcnN0IDw8IGNvdW50OwotfQotCi12b2lkIEFQSV9FTlRSWShnbERyYXdFbGVtZW50cykoR0xlbnVtIG1vZGUsIEdMc2l6ZWkgY291bnQsCi0gICAgICAgICAgICAgICAgICAgIEdMZW51bSB0eXBlLCBjb25zdCBHTHZvaWQgKmluZGljZXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbERyYXdFbGVtZW50cywgbW9kZSwgY291bnQsIHR5cGUsIGluZGljZXMpOwotICAgIEdMTG9nIGxvZygiZ2xEcmF3RWxlbWVudHMiKTsKLSAgICBsb2cgPDwgR0xMb2dFbnVtKG1vZGUpIDw8IGNvdW50IDw8IEdMTG9nRW51bSh0eXBlKTsKLSAgICBpZiAodHlwZSA9PSBHTF9VTlNJR05FRF9CWVRFKSB7Ci0gICAgICAgIGxvZyA8PCBHTExvZ0J1ZmZlcjxHTHVieXRlPihzdGF0aWNfY2FzdDxjb25zdCBHTHVieXRlKj4oaW5kaWNlcyksIGNvdW50KTsKLSAgICB9IGVsc2UgewotICAgICAgICBsb2cgPDwgR0xMb2dCdWZmZXI8R0x1c2hvcnQ+KHN0YXRpY19jYXN0PGNvbnN0IEdMdXNob3J0Kj4oaW5kaWNlcyksIGNvdW50KTsKLSAgICB9Ci0gICAgbG9nOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEVuYWJsZSkoR0xlbnVtIGNhcCkgewotICAgIENBTExfR0xfQVBJKGdsRW5hYmxlLCBjYXApOwotICAgIEdMTG9nKCJnbEVuYWJsZSIpIDw8IEdMTG9nRW51bShjYXApOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEVuYWJsZUNsaWVudFN0YXRlKShHTGVudW0gYXJyYXkpIHsKLSAgICBDQUxMX0dMX0FQSShnbEVuYWJsZUNsaWVudFN0YXRlLCBhcnJheSk7Ci0gICAgR0xMb2coImdsRW5hYmxlQ2xpZW50U3RhdGUiKSA8PCBHTExvZ0VudW0oYXJyYXkpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEZpbmlzaCkodm9pZCkgewotICAgIENBTExfR0xfQVBJKGdsRmluaXNoKTsKLSAgICBHTExvZygiZ2xGaW5pc2giKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xGbHVzaCkodm9pZCkgewotICAgIENBTExfR0xfQVBJKGdsRmx1c2gpOwotICAgIEdMTG9nKCJnbEZsdXNoIik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRm9nZikoR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xGb2dmLCBwbmFtZSwgcGFyYW0pOwotICAgIEdMTG9nKCJnbEZvZ2YiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IHBhcmFtOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEZvZ2Z2KShHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsRm9nZnYsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsRm9nZnYiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZmxvYXQ+KHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRm9neCkoR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xGb2d4LCBwbmFtZSwgcGFyYW0pOwotICAgIEdMTG9nKCJnbEZvZ3giKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nRml4ZWQocGFyYW0pOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEZvZ3h2KShHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsRm9neHYsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsRm9nZngiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRnJvbnRGYWNlKShHTGVudW0gbW9kZSkgewotICAgIENBTExfR0xfQVBJKGdsRnJvbnRGYWNlLCBtb2RlKTsKLSAgICBHTExvZygiZ2xGcm9udEZhY2UiKSA8PCBHTExvZ0VudW0obW9kZSk7Ci0gfQotCi12b2lkIEFQSV9FTlRSWShnbEZydXN0dW1mKShHTGZsb2F0IGxlZnQsIEdMZmxvYXQgcmlnaHQsCi0gICAgICAgICAgICAgICAgR0xmbG9hdCBib3R0b20sIEdMZmxvYXQgdG9wLAotICAgICAgICAgICAgICAgIEdMZmxvYXQgek5lYXIsIEdMZmxvYXQgekZhcikgewotICAgIENBTExfR0xfQVBJKGdsRnJ1c3R1bWYsIGxlZnQsIHJpZ2h0LCBib3R0b20sIHRvcCwgek5lYXIsIHpGYXIpOwotICAgIEdMTG9nKCJnbEZydXN0dW1mIikgPDwgbGVmdCA8PCByaWdodCA8PCBib3R0b20gPDwgdG9wIDw8IHpOZWFyIDw8IHpGYXI7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsRnJ1c3R1bXgpKEdMZml4ZWQgbGVmdCwgR0xmaXhlZCByaWdodCwKLSAgICAgICAgICAgICAgICBHTGZpeGVkIGJvdHRvbSwgR0xmaXhlZCB0b3AsCi0gICAgICAgICAgICAgICAgR0xmaXhlZCB6TmVhciwgR0xmaXhlZCB6RmFyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xGcnVzdHVteCwgbGVmdCwgcmlnaHQsIGJvdHRvbSwgdG9wLCB6TmVhciwgekZhcik7Ci0gICAgR0xMb2coImdsRnJ1c3R1bXgiKQotICAgICAgICAgICAgPDwgR0xMb2dGaXhlZChsZWZ0KSA8PCBHTExvZ0ZpeGVkKHJpZ2h0KQotICAgICAgICAgICAgPDwgR0xMb2dGaXhlZChib3R0b20pIDw8IEdMTG9nRml4ZWQodG9wKQotICAgICAgICAgICAgPDwgR0xMb2dGaXhlZCh6TmVhcikgPDwgR0xMb2dGaXhlZCh6RmFyKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xHZW5UZXh0dXJlcykoR0xzaXplaSBuLCBHTHVpbnQgKnRleHR1cmVzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZW5UZXh0dXJlcywgbiwgdGV4dHVyZXMpOwotICAgIEdMTG9nKCJnbEdlblRleHR1cmVzIikgPDwgbiA8PCBHTExvZ0J1ZmZlcjxHTHVpbnQ+KHRleHR1cmVzLCBuKTsKLX0KLQotR0xlbnVtIEFQSV9FTlRSWShnbEdldEVycm9yKSh2b2lkKSB7Ci0gICAgR0xMb2coImdsR2V0RXJyb3IiKTsKLSAgICBDQUxMX0dMX0FQSV9SRVRVUk4oZ2xHZXRFcnJvcik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0SW50ZWdlcnYpKEdMZW51bSBwbmFtZSwgR0xpbnQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0SW50ZWdlcnYsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsR2V0SW50ZWdlcnYiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMaW50PihwYXJhbXMpOwotfQotCi1jb25zdCBHTHVieXRlICogQVBJX0VOVFJZKGdsR2V0U3RyaW5nKShHTGVudW0gbmFtZSkgewotICAgIEdMTG9nKCJnbEdldFN0cmluZyIpIDw8IEdMTG9nRW51bShuYW1lKTsKLSAgICBDQUxMX0dMX0FQSV9SRVRVUk4oZ2xHZXRTdHJpbmcsIG5hbWUpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbEhpbnQpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBtb2RlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xIaW50LCB0YXJnZXQsIG1vZGUpOwotICAgIEdMTG9nKCJHTGVudW0iKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBHTExvZ0VudW0obW9kZSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTGlnaHRNb2RlbGYpKEdMZW51bSBwbmFtZSwgR0xmbG9hdCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsTGlnaHRNb2RlbGYsIHBuYW1lLCBwYXJhbSk7Ci0gICAgR0xMb2coImdsTGlnaHRNb2RlbGYiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IHBhcmFtOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExpZ2h0TW9kZWxmdikoR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbExpZ2h0TW9kZWxmdiwgcG5hbWUsIHBhcmFtcyk7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBHTExvZygiZ2xMaWdodE1vZGVsZnYiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZmxvYXQ+KHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTGlnaHRNb2RlbHgpKEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsTGlnaHRNb2RlbHgsIHBuYW1lLCBwYXJhbSk7Ci0gICAgR0xMb2coImdsTGlnaHRNb2RlbHgiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nRml4ZWQocGFyYW0pOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExpZ2h0TW9kZWx4dikoR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbExpZ2h0TW9kZWx4diwgcG5hbWUsIHBhcmFtcyk7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBHTExvZygiZ2xMaWdodE1vZGVseHYiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTGlnaHRmKShHTGVudW0gbGlnaHQsIEdMZW51bSBwbmFtZSwgR0xmbG9hdCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsTGlnaHRmLCBsaWdodCwgcG5hbWUsIHBhcmFtKTsKLSAgICBHTExvZygiZ2xMaWdodGYiKSA8PCBHTExvZ0VudW0obGlnaHQpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgcGFyYW07Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTGlnaHRmdikoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsTGlnaHRmdiwgbGlnaHQsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsTGlnaHRmdiIpIDw8IEdMTG9nRW51bShsaWdodCkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBHTExvZ0J1ZmZlcjxHTGZsb2F0PihwYXJhbXMpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExpZ2h0eCkoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pIHsKLSAgIENBTExfR0xfQVBJKGdsTGlnaHR4LCBsaWdodCwgcG5hbWUsIHBhcmFtKTsKLSAgIEdMTG9nKCJnbExpZ2h0eCIpIDw8IEdMTG9nRW51bShsaWdodCkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBHTExvZ0ZpeGVkKHBhcmFtKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xMaWdodHh2KShHTGVudW0gbGlnaHQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xMaWdodHh2LCBsaWdodCwgcG5hbWUsIHBhcmFtcyk7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBHTExvZygiZ2xMaWdodHh2IikgPDwgR0xMb2dFbnVtKGxpZ2h0KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTGluZVdpZHRoKShHTGZsb2F0IHdpZHRoKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xMaW5lV2lkdGgsIHdpZHRoKTsKLSAgICBHTExvZygiZ2xMaW5lV2lkdGgiKSA8PCB3aWR0aDsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xMaW5lV2lkdGh4KShHTGZpeGVkIHdpZHRoKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xMaW5lV2lkdGh4LCB3aWR0aCk7Ci0gICAgR0xMb2coImdsTGluZVdpZHRoIikgPDwgR0xMb2dGaXhlZCh3aWR0aCk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTG9hZElkZW50aXR5KSh2b2lkKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xMb2FkSWRlbnRpdHkpOwotICAgIEdMTG9nKCJnbExvYWRJZGVudGl0eSIpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbExvYWRNYXRyaXhmKShjb25zdCBHTGZsb2F0ICptKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xMb2FkTWF0cml4ZiwgbSk7Ci0gICAgR0xMb2coImdsTG9hZE1hdHJpeGYiKSA8PCBHTExvZ0J1ZmZlcjxHTGZsb2F0PihtLCAxNik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTG9hZE1hdHJpeHgpKGNvbnN0IEdMZml4ZWQgKm0pIHsKLSAgICBDQUxMX0dMX0FQSShnbExvYWRNYXRyaXh4LCBtKTsKLSAgICBHTExvZygiZ2xMb2FkTWF0cml4eCIpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KG0sIDE2KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xMb2dpY09wKShHTGVudW0gb3Bjb2RlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xMb2dpY09wLCBvcGNvZGUpOwotICAgIEdMTG9nKCJnbExvZ2ljT3AiKSA8PCBHTExvZ0VudW0ob3Bjb2RlKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xNYXRlcmlhbGYpKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbE1hdGVyaWFsZiwgZmFjZSwgcG5hbWUsIHBhcmFtKTsKLSAgICBHTExvZygiZ2xNYXRlcmlhbGYiKSA8PCBHTExvZ0VudW0oZmFjZSkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBwYXJhbTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xNYXRlcmlhbGZ2KShHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbE1hdGVyaWFsZnYsIGZhY2UsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsTWF0ZXJpYWxmdiIpIDw8IEdMTG9nRW51bShmYWNlKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZmxvYXQ+KHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTWF0ZXJpYWx4KShHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xNYXRlcmlhbHgsIGZhY2UsIHBuYW1lLCBwYXJhbSk7Ci0gICAgR0xMb2coImdsTWF0ZXJpYWx4IikgPDwgR0xMb2dFbnVtKGZhY2UpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dGaXhlZChwYXJhbSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTWF0ZXJpYWx4dikoR0xlbnVtIGZhY2UsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xNYXRlcmlhbHh2LCBmYWNlLCBwbmFtZSwgcGFyYW1zKTsKLSAgICAvLyBYWFg6IHdlIG5lZWQgdG8gY29tcHV0ZSB0aGUgc2l6ZSBvZiB0aGlzIGJ1ZmZlcgotICAgIEdMTG9nKCJnbE1hdGVyaWFseHYiKSA8PCBHTExvZ0VudW0oZmFjZSkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBHTExvZ0J1ZmZlcjxHTGZpeGVkPihwYXJhbXMpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbE1hdHJpeE1vZGUpKEdMZW51bSBtb2RlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xNYXRyaXhNb2RlLCBtb2RlKTsKLSAgICBHTExvZygiZ2xNYXRyaXhNb2RlIikgPDwgR0xMb2dFbnVtKG1vZGUpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbE11bHRNYXRyaXhmKShjb25zdCBHTGZsb2F0ICptKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xNdWx0TWF0cml4ZiwgbSk7Ci0gICAgR0xMb2coImdsTXVsdE1hdHJpeGYiKSA8PCBHTExvZ0J1ZmZlcjxHTGZsb2F0PihtLCAxNik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTXVsdE1hdHJpeHgpKGNvbnN0IEdMZml4ZWQgKm0pIHsKLSAgICBDQUxMX0dMX0FQSShnbE11bHRNYXRyaXh4LCBtKTsKLSAgICBHTExvZygiZ2xNdWx0TWF0cml4eCIpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KG0sIDE2KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xNdWx0aVRleENvb3JkNGYpKEdMZW51bSB0YXJnZXQsIEdMZmxvYXQgcywgR0xmbG9hdCB0LCBHTGZsb2F0IHIsIEdMZmxvYXQgcSkgewotICAgIENBTExfR0xfQVBJKGdsTXVsdGlUZXhDb29yZDRmLCB0YXJnZXQsIHMsIHQsIHIsIHEpOwotICAgIEdMTG9nKCJnbE11bHRpVGV4Q29vcmQ0ZiIpIDw8IEdMTG9nRW51bSh0YXJnZXQpIDw8IHMgPDwgdCA8PCByIDw8IHE7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTXVsdGlUZXhDb29yZDR4KShHTGVudW0gdGFyZ2V0LCBHTGZpeGVkIHMsIEdMZml4ZWQgdCwgR0xmaXhlZCByLCBHTGZpeGVkIHEpIHsKLSAgICBDQUxMX0dMX0FQSShnbE11bHRpVGV4Q29vcmQ0eCwgdGFyZ2V0LCBzLCB0LCByLCBxKTsKLSAgICBHTExvZygiZ2xNdWx0aVRleENvb3JkNHgiKSA8PCBHTExvZ0VudW0odGFyZ2V0KQotICAgICAgICA8PCBHTExvZ0ZpeGVkKHMpIDw8IEdMTG9nRml4ZWQodCkgPDwgR0xMb2dGaXhlZChyKSA8PCBHTExvZ0ZpeGVkKHEpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbE5vcm1hbDNmKShHTGZsb2F0IG54LCBHTGZsb2F0IG55LCBHTGZsb2F0IG56KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xOb3JtYWwzZiwgbngsIG55LCBueik7Ci0gICAgR0xMb2coImdsTm9ybWFsM2YiKSA8PCBueCA8PCBueSA8PCBuejsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xOb3JtYWwzeCkoR0xmaXhlZCBueCwgR0xmaXhlZCBueSwgR0xmaXhlZCBueikgewotICAgIENBTExfR0xfQVBJKGdsTm9ybWFsM3gsIG54LCBueSwgbnopOwotICAgIEdMTG9nKCJnbE5vcm1hbDN4IikgPDwgR0xMb2dGaXhlZChueCkgPDwgR0xMb2dGaXhlZChueSkgPDwgR0xMb2dGaXhlZChueik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsTm9ybWFsUG9pbnRlcikoR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIpIHsKLSAgICBDQUxMX0dMX0FQSShnbE5vcm1hbFBvaW50ZXIsIHR5cGUsIHN0cmlkZSwgcG9pbnRlcik7Ci0gICAgR0xMb2coImdsTm9ybWFsUG9pbnRlciIpIDw8IEdMTG9nRW51bSh0eXBlKSA8PCBzdHJpZGUgPDwgcG9pbnRlcjsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xPcnRob2YpKCAgR0xmbG9hdCBsZWZ0LCBHTGZsb2F0IHJpZ2h0LAotICAgICAgICAgICAgICAgIEdMZmxvYXQgYm90dG9tLCBHTGZsb2F0IHRvcCwKLSAgICAgICAgICAgICAgICBHTGZsb2F0IHpOZWFyLCBHTGZsb2F0IHpGYXIpIHsKLSAgICBDQUxMX0dMX0FQSShnbE9ydGhvZiwgbGVmdCwgcmlnaHQsIGJvdHRvbSwgdG9wLCB6TmVhciwgekZhcik7Ci0gICAgR0xMb2coImdsT3J0aG9mIikgPDwgbGVmdCA8PCByaWdodCA8PCBib3R0b20gPDwgdG9wIDw8IHpOZWFyIDw8IHpGYXI7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsT3J0aG94KSggIEdMZml4ZWQgbGVmdCwgR0xmaXhlZCByaWdodCwKLSAgICAgICAgICAgICAgICBHTGZpeGVkIGJvdHRvbSwgR0xmaXhlZCB0b3AsCi0gICAgICAgICAgICAgICAgR0xmaXhlZCB6TmVhciwgR0xmaXhlZCB6RmFyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xPcnRob3gsIGxlZnQsIHJpZ2h0LCBib3R0b20sIHRvcCwgek5lYXIsIHpGYXIpOwotICAgIEdMTG9nKCJnbE9ydGhveCIpIDw8IEdMTG9nRml4ZWQobGVmdCkgPDwgR0xMb2dGaXhlZChyaWdodCkKLSAgICAgICAgICAgIDw8IEdMTG9nRml4ZWQoYm90dG9tKSA8PCBHTExvZ0ZpeGVkKHRvcCkKLSAgICAgICAgICAgIDw8IEdMTG9nRml4ZWQoek5lYXIpIDw8IEdMTG9nRml4ZWQoekZhcik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsUGl4ZWxTdG9yZWkpKEdMZW51bSBwbmFtZSwgR0xpbnQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbFBpeGVsU3RvcmVpLCBwbmFtZSwgcGFyYW0pOwotICAgIEdMTG9nKCJnbFBpeGVsU3RvcmVpIikgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBwYXJhbTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xQb2ludFNpemUpKEdMZmxvYXQgc2l6ZSkgewotICAgIENBTExfR0xfQVBJKGdsUG9pbnRTaXplLCBzaXplKTsKLSAgICBHTExvZygiZ2xQb2ludFNpemUiKSA8PCBzaXplOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFBvaW50U2l6ZXgpKEdMZml4ZWQgc2l6ZSkgewotICAgIENBTExfR0xfQVBJKGdsUG9pbnRTaXpleCwgc2l6ZSk7Ci0gICAgR0xMb2coImdsUG9pbnRTaXpleCIpIDw8IEdMTG9nRml4ZWQoc2l6ZSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsUG9seWdvbk9mZnNldCkoR0xmbG9hdCBmYWN0b3IsIEdMZmxvYXQgdW5pdHMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFBvbHlnb25PZmZzZXQsIGZhY3RvciwgdW5pdHMpOwotICAgIEdMTG9nKCJnbFBvbHlnb25PZmZzZXQiKSA8PCBmYWN0b3IgPDwgdW5pdHM7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsUG9seWdvbk9mZnNldHgpKEdMZml4ZWQgZmFjdG9yLCBHTGZpeGVkIHVuaXRzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xQb2x5Z29uT2Zmc2V0eCwgZmFjdG9yLCB1bml0cyk7Ci0gICAgR0xMb2coImdsUG9seWdvbk9mZnNldHgiKSA8PCBHTExvZ0ZpeGVkKGZhY3RvcikgPDwgR0xMb2dGaXhlZCh1bml0cyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsUG9wTWF0cml4KSh2b2lkKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xQb3BNYXRyaXgpOwotICAgIEdMTG9nKCJnbFBvcE1hdHJpeCIpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFB1c2hNYXRyaXgpKHZvaWQpIHsKLSAgICBDQUxMX0dMX0FQSShnbFB1c2hNYXRyaXgpOwotICAgIEdMTG9nKCJnbFB1c2hNYXRyaXgiKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xSZWFkUGl4ZWxzKSggIEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LAotICAgICAgICAgICAgICAgICAgICBHTGVudW0gZm9ybWF0LCBHTGVudW0gdHlwZSwgR0x2b2lkICpwaXhlbHMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFJlYWRQaXhlbHMsIHgsIHksIHdpZHRoLCBoZWlnaHQsIGZvcm1hdCwgdHlwZSwgcGl4ZWxzKTsKLSAgICAvLyBYWFg6IHdlIG5lZWQgdG8gY29tcHV0ZSB0aGUgc2l6ZSBvZiB0aGlzIGJ1ZmZlcgotICAgIEdMTG9nKCJnbFJlYWRQaXhlbHMiKSA8PCB4IDw8IHkgPDwgd2lkdGggPDwgaGVpZ2h0IDw8IEdMTG9nRW51bShmb3JtYXQpIDw8IEdMTG9nRW51bSh0eXBlKQotICAgICAgICAgICAgPDwgR0xMb2dCdWZmZXI8dW5zaWduZWQgY2hhcj4oc3RhdGljX2Nhc3Q8dW5zaWduZWQgY2hhciAqPihwaXhlbHMpKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xSb3RhdGVmKShHTGZsb2F0IGFuZ2xlLCBHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xSb3RhdGVmLCBhbmdsZSwgeCwgeSwgeik7Ci0gICAgR0xMb2coImdsUm90YXRlZiIpIDw8IGFuZ2xlIDw8IHggPDwgeSA8PCB6OwotfQotCi12b2lkIEFQSV9FTlRSWShnbFJvdGF0ZXgpKEdMZml4ZWQgYW5nbGUsIEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopIHsKLSAgICBDQUxMX0dMX0FQSShnbFJvdGF0ZXgsIGFuZ2xlLCB4LCB5LCB6KTsKLSAgICBHTExvZygiZ2xSb3RhdGV4IikgPDwgR0xMb2dGaXhlZChhbmdsZSkgPDwgR0xMb2dGaXhlZCh4KSA8PCBHTExvZ0ZpeGVkKHkpIDw8IEdMTG9nRml4ZWQoeik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsU2FtcGxlQ292ZXJhZ2UpKEdMY2xhbXBmIHZhbHVlLCBHTGJvb2xlYW4gaW52ZXJ0KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xTYW1wbGVDb3ZlcmFnZSwgdmFsdWUsIGludmVydCk7Ci0gICAgR0xMb2coImdsU2FtcGxlQ292ZXJhZ2UiKSA8PCB2YWx1ZSA8PCBHTExvZ0Jvb2woaW52ZXJ0KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xTYW1wbGVDb3ZlcmFnZXgpKEdMY2xhbXB4IHZhbHVlLCBHTGJvb2xlYW4gaW52ZXJ0KSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xTYW1wbGVDb3ZlcmFnZXgsIHZhbHVlLCBpbnZlcnQpOwotICAgIEdMTG9nKCJnbFNhbXBsZUNvdmVyYWdleCIpIDw8IEdMTG9nRml4ZWQodmFsdWUpIDw8IEdMTG9nQm9vbChpbnZlcnQpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFNjYWxlZikoR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeikgewotICAgIENBTExfR0xfQVBJKGdsU2NhbGVmLCB4LCB5LCB6KTsKLSAgICBHTExvZygiZ2xTY2FsZWYiKSA8PCB4IDw8IHkgPDwgejsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xTY2FsZXgpKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopIHsKLSAgICBDQUxMX0dMX0FQSShnbFNjYWxleCwgeCwgeSwgeik7Ci0gICAgR0xMb2coImdsU2NhbGV4IikgPDwgR0xMb2dGaXhlZCh4KSA8PCBHTExvZ0ZpeGVkKHkpIDw8IEdMTG9nRml4ZWQoeik7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsU2Npc3NvcikoR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQpIHsKLSAgICBDQUxMX0dMX0FQSShnbFNjaXNzb3IsIHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgIEdMTG9nKCJnbFNjaXNzb3IiKSA8PCB4IDw8IHkgPDwgd2lkdGggPDwgaGVpZ2h0OwotfQotCi12b2lkIEFQSV9FTlRSWShnbFNoYWRlTW9kZWwpKEdMZW51bSBtb2RlKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xTaGFkZU1vZGVsLCBtb2RlKTsKLSAgICBHTExvZygiZ2xTaGFkZU1vZGVsIikgPDwgR0xMb2dFbnVtKG1vZGUpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFN0ZW5jaWxGdW5jKShHTGVudW0gZnVuYywgR0xpbnQgcmVmLCBHTHVpbnQgbWFzaykgewotICAgIENBTExfR0xfQVBJKGdsU3RlbmNpbEZ1bmMsIGZ1bmMsIHJlZiwgbWFzayk7Ci0gICAgR0xMb2coImdsU3RlbmNpbEZ1bmMiKSA8PCBHTExvZ0VudW0oZnVuYykgPDwgcmVmIDw8IG1hc2s7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsU3RlbmNpbE1hc2spKEdMdWludCBtYXNrKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xTdGVuY2lsTWFzaywgbWFzayk7Ci0gICAgR0xMb2coImdsU3RlbmNpbE1hc2siKSA8PCBtYXNrOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFN0ZW5jaWxPcCkoR0xlbnVtIGZhaWwsIEdMZW51bSB6ZmFpbCwgR0xlbnVtIHpwYXNzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xTdGVuY2lsT3AsIGZhaWwsIHpmYWlsLCB6cGFzcyk7Ci0gICAgR0xMb2coImdsU3RlbmNpbE9wIikgPDwgR0xMb2dFbnVtKGZhaWwpIDw8IEdMTG9nRW51bSh6ZmFpbCkgPDwgR0xMb2dFbnVtKHpwYXNzKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUZXhDb29yZFBvaW50ZXIpKCBHTGludCBzaXplLCBHTGVudW0gdHlwZSwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleENvb3JkUG9pbnRlciwgc2l6ZSwgdHlwZSwgc3RyaWRlLCBwb2ludGVyKTsKLSAgICBHTExvZygiZ2xUZXhDb29yZFBvaW50ZXIiKSA8PCBzaXplIDw8IEdMTG9nRW51bSh0eXBlKSA8PCBzdHJpZGUgPDwgcG9pbnRlcjsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUZXhFbnZmKShHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleEVudmYsIHRhcmdldCwgcG5hbWUsIHBhcmFtKTsKLSAgICBHTExvZygiZ2xUZXhFbnZmIikgPDwgR0xMb2dFbnVtKHRhcmdldCkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBwYXJhbTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUZXhFbnZmdikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleEVudmZ2LCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsVGV4RW52eCIpIDw8IEdMTG9nRW51bSh0YXJnZXQpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xmbG9hdD4ocGFyYW1zKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUZXhFbnZ4KShHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleEVudngsIHRhcmdldCwgcG5hbWUsIHBhcmFtKTsKLSAgICBHTExvZygiZ2xUZXhFbnZ4IikgPDwgR0xMb2dFbnVtKHRhcmdldCkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBHTExvZ0ZpeGVkKHBhcmFtKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUZXhFbnZ4dikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleEVudnh2LCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsVGV4RW52eHYiKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4SW1hZ2UyRCkoICBHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgaW50ZXJuYWxmb3JtYXQsCi0gICAgICAgICAgICAgICAgICAgIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LCBHTGludCBib3JkZXIsIEdMZW51bSBmb3JtYXQsCi0gICAgICAgICAgICAgICAgICAgIEdMZW51bSB0eXBlLCBjb25zdCBHTHZvaWQgKnBpeGVscykgewotICAgIENBTExfR0xfQVBJKGdsVGV4SW1hZ2UyRCwgdGFyZ2V0LCBsZXZlbCwgaW50ZXJuYWxmb3JtYXQsIHdpZHRoLCBoZWlnaHQsCi0gICAgICAgICAgICBib3JkZXIsIGZvcm1hdCwgdHlwZSwgcGl4ZWxzKTsKLSAgICBHTExvZygiZ2xUZXhJbWFnZTJEIikgPDwgR0xMb2dFbnVtKHRhcmdldCkgPDwgbGV2ZWwgPDwgR0xMb2dFbnVtKGludGVybmFsZm9ybWF0KQotICAgICAgICAgICAgPDwgd2lkdGggPDwgaGVpZ2h0IDw8IGJvcmRlciA8PCBHTExvZ0VudW0oZm9ybWF0KSA8PCBHTExvZ0VudW0odHlwZSkKLSAgICAgICAgICAgIDw8IEdMTG9nQnVmZmVyPHVuc2lnbmVkIGNoYXI+KCBzdGF0aWNfY2FzdDxjb25zdCB1bnNpZ25lZCBjaGFyICo+KHBpeGVscykpOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFRleFBhcmFtZXRlcmYpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmbG9hdCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsVGV4UGFyYW1ldGVyZiwgdGFyZ2V0LCBwbmFtZSwgcGFyYW0pOwotICAgIEdMTG9nKCJnbFRleFBhcmFtZXRlcmYiKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IHBhcmFtOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFRleFBhcmFtZXRlcngpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsVGV4UGFyYW1ldGVyeCwgdGFyZ2V0LCBwbmFtZSwgcGFyYW0pOwotICAgIEdMTG9nKCJnbFRleFBhcmFtZXRlcngiKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nRml4ZWQocGFyYW0pOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFRleFN1YkltYWdlMkQpKCAgIEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGludCB4b2Zmc2V0LAotICAgICAgICAgICAgICAgICAgICAgICAgR0xpbnQgeW9mZnNldCwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsCi0gICAgICAgICAgICAgICAgICAgICAgICBHTGVudW0gZm9ybWF0LCBHTGVudW0gdHlwZSwgY29uc3QgR0x2b2lkICpwaXhlbHMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleFN1YkltYWdlMkQsIHRhcmdldCwgbGV2ZWwsIHhvZmZzZXQsIHlvZmZzZXQsCi0gICAgICAgICAgICB3aWR0aCwgaGVpZ2h0LCBmb3JtYXQsIHR5cGUsIHBpeGVscyk7Ci0gICAgR0xMb2coImdsVGV4U3ViSW1hZ2UyRCIpIDw8IEdMTG9nRW51bSh0YXJnZXQpIDw8IGxldmVsIDw8IHhvZmZzZXQgPDwgeW9mZnNldAotICAgICAgICAgICAgPDwgd2lkdGggPDwgaGVpZ2h0IDw8IEdMTG9nRW51bShmb3JtYXQpIDw8IEdMTG9nRW51bSh0eXBlKQotICAgICAgICAgICAgPDwgR0xMb2dCdWZmZXI8dW5zaWduZWQgY2hhcj4oIHN0YXRpY19jYXN0PGNvbnN0IHVuc2lnbmVkIGNoYXIgKj4ocGl4ZWxzKSk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVHJhbnNsYXRlZikoR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeikgewotICAgIENBTExfR0xfQVBJKGdsVHJhbnNsYXRlZiwgeCwgeSwgeik7Ci0gICAgR0xMb2coImdsVHJhbnNsYXRlZiIpIDw8IHggPDwgeSA8PCB6OwotfQotCi12b2lkIEFQSV9FTlRSWShnbFRyYW5zbGF0ZXgpKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopIHsKLSAgICBDQUxMX0dMX0FQSShnbFRyYW5zbGF0ZXgsIHgsIHksIHopOwotICAgIEdMTG9nKCJnbFRyYW5zbGF0ZXgiKSA8PCBHTExvZ0ZpeGVkKHgpIDw8IEdMTG9nRml4ZWQoeSkgPDwgR0xMb2dGaXhlZCh6KTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xWZXJ0ZXhQb2ludGVyKSggICBHTGludCBzaXplLCBHTGVudW0gdHlwZSwKLSAgICAgICAgICAgICAgICAgICAgICAgIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIpIHsKLSAgICBDQUxMX0dMX0FQSShnbFZlcnRleFBvaW50ZXIsIHNpemUsIHR5cGUsIHN0cmlkZSwgcG9pbnRlcik7Ci0gICAgR0xMb2coImdsVmVydGV4UG9pbnRlciIpIDw8IHNpemUgPDwgR0xMb2dFbnVtKHR5cGUpIDw8IHN0cmlkZSA8PCBwb2ludGVyOwotfQotCi12b2lkIEFQSV9FTlRSWShnbFZpZXdwb3J0KShHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCkgewotICAgIENBTExfR0xfQVBJKGdsVmlld3BvcnQsIHgsIHksIHdpZHRoLCBoZWlnaHQpOwotICAgIEdMTG9nKCJnbFZpZXdwb3J0IikgPDwgeCA8PCB5IDw8IHdpZHRoIDw8IGhlaWdodDsKLX0KLQotLy8gRVMgMS4xCi12b2lkIEFQSV9FTlRSWShnbENsaXBQbGFuZWYpKEdMZW51bSBwbGFuZSwgY29uc3QgR0xmbG9hdCAqZXF1YXRpb24pIHsKLSAgICBDQUxMX0dMX0FQSShnbENsaXBQbGFuZWYsIHBsYW5lLCBlcXVhdGlvbik7Ci0gICAgR0xMb2coImdsQ2xpcFBsYW5lZiIpIDw8IEdMTG9nRW51bShwbGFuZSkgPDwgR0xMb2dCdWZmZXI8R0xmbG9hdD4oZXF1YXRpb24sIDQpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xDbGlwUGxhbmV4KShHTGVudW0gcGxhbmUsIGNvbnN0IEdMZml4ZWQgKmVxdWF0aW9uKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xDbGlwUGxhbmV4LCBwbGFuZSwgZXF1YXRpb24pOwotICAgIEdMTG9nKCJnbENsaXBQbGFuZXgiKSA8PCBHTExvZ0VudW0ocGxhbmUpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KGVxdWF0aW9uLCA0KTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsQmluZEJ1ZmZlcikoR0xlbnVtIHRhcmdldCwgR0x1aW50IGJ1ZmZlcikgewotICAgIENBTExfR0xfQVBJKGdsQmluZEJ1ZmZlciwgdGFyZ2V0LCBidWZmZXIpOwotICAgIEdMTG9nKCJnbEJpbmRCdWZmZXIiKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBidWZmZXI7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEJ1ZmZlckRhdGEpKEdMZW51bSB0YXJnZXQsIEdMc2l6ZWlwdHIgc2l6ZSwgY29uc3QgR0x2b2lkKiBkYXRhLCBHTGVudW0gdXNhZ2UpIHsKLSAgICBDQUxMX0dMX0FQSShnbEJ1ZmZlckRhdGEsIHRhcmdldCwgc2l6ZSwgZGF0YSwgdXNhZ2UpOwotICAgIEdMTG9nKCJnbEJ1ZmZlckRhdGEiKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBzaXplCi0gICAgICAgIDw8IEdMTG9nQnVmZmVyPHVuc2lnbmVkIGNoYXI+KHN0YXRpY19jYXN0PGNvbnN0IHVuc2lnbmVkIGNoYXIqPihkYXRhKSwgc2l6ZSk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEJ1ZmZlclN1YkRhdGEpKEdMZW51bSB0YXJnZXQsIEdMaW50cHRyIG9mZnNldCwgR0xzaXplaXB0ciBzaXplLCBjb25zdCBHTHZvaWQqIGRhdGEpIHsKLSAgICBDQUxMX0dMX0FQSShnbEJ1ZmZlclN1YkRhdGEsIHRhcmdldCwgb2Zmc2V0LCBzaXplLCBkYXRhKTsKLSAgICBHTExvZygiZ2xCdWZmZXJTdWJEYXRhIikgPDwgR0xMb2dFbnVtKHRhcmdldCkgPDwgb2Zmc2V0IDw8IHNpemUKLSAgICAgICAgPDwgR0xMb2dCdWZmZXI8dW5zaWduZWQgY2hhcj4oc3RhdGljX2Nhc3Q8Y29uc3QgdW5zaWduZWQgY2hhcio+KGRhdGEpLCBzaXplKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsRGVsZXRlQnVmZmVycykoR0xzaXplaSBuLCBjb25zdCBHTHVpbnQqIGJ1ZmZlcnMpIHsKLSAgICBDQUxMX0dMX0FQSShnbERlbGV0ZUJ1ZmZlcnMsIG4sIGJ1ZmZlcnMpOwotICAgIEdMTG9nKCJnbERlbGV0ZUJ1ZmZlcnMiKSA8PCBuIDw8IEdMTG9nQnVmZmVyPEdMdWludD4oYnVmZmVycywgbik7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEdlbkJ1ZmZlcnMpKEdMc2l6ZWkgbiwgR0x1aW50KiBidWZmZXJzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZW5CdWZmZXJzLCBuLCBidWZmZXJzKTsKLSAgICBHTExvZygiZ2xHZW5CdWZmZXJzIikgPDwgbiA8PCBHTExvZ0J1ZmZlcjxHTHVpbnQ+KGJ1ZmZlcnMsIG4pOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRCb29sZWFudikoR0xlbnVtIHBuYW1lLCBHTGJvb2xlYW4gKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0Qm9vbGVhbnYsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsR2V0Qm9vbGVhbnYiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMYm9vbGVhbj4ocGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0Rml4ZWR2KShHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0Rml4ZWR2LCBwbmFtZSwgcGFyYW1zKTsKLSAgICAvLyBYWFg6IHdlIG5lZWQgdG8gY29tcHV0ZSB0aGUgc2l6ZSBvZiB0aGlzIGJ1ZmZlcgotICAgIEdMTG9nKCJnbEdldEZpeGVkdiIpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xmaXhlZD4ocGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0RmxvYXR2KShHTGVudW0gcG5hbWUsIEdMZmxvYXQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0RmxvYXR2LCBwbmFtZSwgcGFyYW1zKTsKLSAgICAvLyBYWFg6IHdlIG5lZWQgdG8gY29tcHV0ZSB0aGUgc2l6ZSBvZiB0aGlzIGJ1ZmZlcgotICAgIEdMTG9nKCJnbEdldEZsb2F0diIpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xmbG9hdD4ocGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0UG9pbnRlcnYpKEdMZW51bSBwbmFtZSwgdm9pZCAqKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0UG9pbnRlcnYsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsR2V0UG9pbnRlcnYiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPHZvaWQqPihwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRCdWZmZXJQYXJhbWV0ZXJpdikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGludCAqcGFyYW1zKSB7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBDQUxMX0dMX0FQSShnbEdldEJ1ZmZlclBhcmFtZXRlcml2LCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotICAgIEdMTG9nKCJnbEdldEJ1ZmZlclBhcmFtZXRlcml2IikgPDwgR0xMb2dFbnVtKHRhcmdldCkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBHTExvZ0J1ZmZlcjxHTGludD4ocGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0Q2xpcFBsYW5lZikoR0xlbnVtIHBuYW1lLCBHTGZsb2F0IGVxbls0XSkgewotICAgIENBTExfR0xfQVBJKGdsR2V0Q2xpcFBsYW5lZiwgcG5hbWUsIGVxbik7Ci0gICAgR0xMb2coImdsR2V0Q2xpcFBsYW5lZiIpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xmbG9hdD4oZXFuLCA0KTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0Q2xpcFBsYW5leCkoR0xlbnVtIHBuYW1lLCBHTGZpeGVkIGVxbls0XSkgewotICAgIENBTExfR0xfQVBJKGdsR2V0Q2xpcFBsYW5leCwgcG5hbWUsIGVxbik7Ci0gICAgR0xMb2coImdsR2V0Q2xpcFBsYW5leCIpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xmaXhlZD4oZXFuLCA0KTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0TGlnaHR4dikoR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0TGlnaHR4diwgbGlnaHQsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsR2V0TGlnaHR4diIpIDw8IEdMTG9nRW51bShsaWdodCkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBHTExvZ0J1ZmZlcjxHTGZpeGVkPihwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRMaWdodGZ2KShHTGVudW0gbGlnaHQsIEdMZW51bSBwbmFtZSwgR0xmbG9hdCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZXRMaWdodGZ2LCBsaWdodCwgcG5hbWUsIHBhcmFtcyk7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBHTExvZygiZ2xHZXRMaWdodGZ2IikgPDwgR0xMb2dFbnVtKGxpZ2h0KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZmxvYXQ+KHBhcmFtcyk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEdldE1hdGVyaWFseHYpKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0TWF0ZXJpYWx4diwgZmFjZSwgcG5hbWUsIHBhcmFtcyk7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBHTExvZygiZ2xHZXRNYXRlcmlhbHh2IikgPDwgR0xMb2dFbnVtKGZhY2UpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xmaXhlZD4ocGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0TWF0ZXJpYWxmdikoR0xlbnVtIGZhY2UsIEdMZW51bSBwbmFtZSwgR0xmbG9hdCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xHZXRNYXRlcmlhbGZ2LCBmYWNlLCBwbmFtZSwgcGFyYW1zKTsKLSAgICAvLyBYWFg6IHdlIG5lZWQgdG8gY29tcHV0ZSB0aGUgc2l6ZSBvZiB0aGlzIGJ1ZmZlcgotICAgIEdMTG9nKCJnbEdldE1hdGVyaWFsZnYiKSA8PCBHTExvZ0VudW0oZmFjZSkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBHTExvZ0J1ZmZlcjxHTGZsb2F0PihwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRUZXhFbnZmdikoR0xlbnVtIGVudiwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdldFRleEVudmZ2LCBlbnYsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsR2V0VGV4RW52ZnYiKSA8PCBHTExvZ0VudW0oZW52KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZmxvYXQ+KHBhcmFtcyk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEdldFRleEVudml2KShHTGVudW0gZW52LCBHTGVudW0gcG5hbWUsIEdMaW50ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdldFRleEVudml2LCBlbnYsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsR2V0VGV4RW52aXYiKSA8PCBHTExvZ0VudW0oZW52KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMaW50PihwYXJhbXMpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xHZXRUZXhFbnZ4dikoR0xlbnVtIGVudiwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbEdldFRleEVudnh2LCBlbnYsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsR2V0VGV4RW52eHYiKSA8PCBHTExvZ0VudW0oZW52KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KHBhcmFtcyk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEdldFRleFBhcmFtZXRlcmZ2KShHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0VGV4UGFyYW1ldGVyZnYsIHRhcmdldCwgcG5hbWUsIHBhcmFtcyk7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBHTExvZygiZ2xHZXRUZXhQYXJhbWV0ZXJmdiIpIDw8IEdMTG9nRW51bSh0YXJnZXQpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xmbG9hdD4ocGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsR2V0VGV4UGFyYW1ldGVyaXYpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xpbnQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0VGV4UGFyYW1ldGVyaXYsIHRhcmdldCwgcG5hbWUsIHBhcmFtcyk7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBHTExvZygiZ2xHZXRUZXhQYXJhbWV0ZXJpdiIpIDw8IEdMTG9nRW51bSh0YXJnZXQpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xpbnQ+KHBhcmFtcyk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbEdldFRleFBhcmFtZXRlcnh2KShHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsR2V0VGV4UGFyYW1ldGVyeHYsIHRhcmdldCwgcG5hbWUsIHBhcmFtcyk7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBHTExvZygiZ2xHZXRUZXhQYXJhbWV0ZXJ4diIpIDw8IEdMTG9nRW51bSh0YXJnZXQpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xmaXhlZD4ocGFyYW1zKTsKLX0KLUdMYm9vbGVhbiBBUElfRU5UUlkoZ2xJc0J1ZmZlcikoR0x1aW50IGJ1ZmZlcikgewotICAgIEdMTG9nKCJnbElzQnVmZmVyIikgPDwgYnVmZmVyOwotICAgIENBTExfR0xfQVBJX1JFVFVSTihnbElzQnVmZmVyLCBidWZmZXIpOwotfQotR0xib29sZWFuIEFQSV9FTlRSWShnbElzRW5hYmxlZCkoR0xlbnVtIGNhcCkgewotICAgIEdMTG9nKCJnbElzRW5hYmxlZCIpIDw8IEdMTG9nRW51bShjYXApOwotICAgIENBTExfR0xfQVBJX1JFVFVSTihnbElzRW5hYmxlZCwgY2FwKTsKLX0KLUdMYm9vbGVhbiBBUElfRU5UUlkoZ2xJc1RleHR1cmUpKEdMdWludCB0ZXh0dXJlKSB7Ci0gICAgR0xMb2coImdsSXNUZXh0dXJlIikgPDwgdGV4dHVyZTsKLSAgICBDQUxMX0dMX0FQSV9SRVRVUk4oZ2xJc1RleHR1cmUsIHRleHR1cmUpOwotfQotdm9pZCBBUElfRU5UUlkoZ2xQb2ludFBhcmFtZXRlcmYpKEdMZW51bSBwbmFtZSwgR0xmbG9hdCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsUG9pbnRQYXJhbWV0ZXJmLCBwbmFtZSwgcGFyYW0pOwotICAgIEdMTG9nKCJnbFBvaW50UGFyYW1ldGVyZiIpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgcGFyYW07Ci19Ci12b2lkIEFQSV9FTlRSWShnbFBvaW50UGFyYW1ldGVyZnYpKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xQb2ludFBhcmFtZXRlcmZ2LCBwbmFtZSwgcGFyYW1zKTsKLSAgICAvLyBYWFg6IHdlIG5lZWQgdG8gY29tcHV0ZSB0aGUgc2l6ZSBvZiB0aGlzIGJ1ZmZlcgotICAgIEdMTG9nKCJnbFBvaW50UGFyYW1ldGVyZnYiKSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZmxvYXQ+KHBhcmFtcyk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbFBvaW50UGFyYW1ldGVyeCkoR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xQb2ludFBhcmFtZXRlcngsIHBuYW1lLCBwYXJhbSk7Ci0gICAgR0xMb2coImdsUG9pbnRQYXJhbWV0ZXJ4IikgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBHTExvZ0ZpeGVkKHBhcmFtKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsUG9pbnRQYXJhbWV0ZXJ4dikoR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFBvaW50UGFyYW1ldGVyeHYsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsUG9pbnRQYXJhbWV0ZXJ4diIpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xmaXhlZD4ocGFyYW1zKTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsQ29sb3I0dWIpKEdMdWJ5dGUgcmVkLCBHTHVieXRlIGdyZWVuLCBHTHVieXRlIGJsdWUsIEdMdWJ5dGUgYWxwaGEpIHsKLSAgICBDQUxMX0dMX0FQSShnbENvbG9yNHViLCByZWQsIGdyZWVuLCBibHVlLCBhbHBoYSk7Ci0gICAgR0xMb2coImdsQ29sb3I0dWIiKSA8PCByZWQgPDwgZ3JlZW4gPDwgYmx1ZSA8PCBhbHBoYTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4RW52aSkoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGludCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsVGV4RW52aSwgdGFyZ2V0LCBwbmFtZSwgcGFyYW0pOwotICAgIEdMTG9nKCJnbFRleEVudmkiKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IHBhcmFtOwotfQotdm9pZCBBUElfRU5UUlkoZ2xUZXhFbnZpdikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGludCAqcGFyYW1zKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xUZXhFbnZpdiwgdGFyZ2V0LCBwbmFtZSwgcGFyYW1zKTsKLSAgICAvLyBYWFg6IHdlIG5lZWQgdG8gY29tcHV0ZSB0aGUgc2l6ZSBvZiB0aGlzIGJ1ZmZlcgotICAgIEdMTG9nKCJnbFRleEVudml2IikgPDwgR0xMb2dFbnVtKHRhcmdldCkgPDwgR0xMb2dFbnVtKHBuYW1lKSA8PCBHTExvZ0J1ZmZlcjxHTGludD4ocGFyYW1zKTsKLX0KLQotdm9pZCBBUElfRU5UUlkoZ2xUZXhQYXJhbWV0ZXJmdikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleFBhcmFtZXRlcmZ2LCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsVGV4UGFyYW1ldGVyZnYiKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZmxvYXQ+KHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4UGFyYW1ldGVyaXYpKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xpbnQgKnBhcmFtcykgewotICAgIENBTExfR0xfQVBJKGdsVGV4UGFyYW1ldGVyaXYsIHRhcmdldCwgcG5hbWUsIHBhcmFtcyk7Ci0gICAgLy8gWFhYOiB3ZSBuZWVkIHRvIGNvbXB1dGUgdGhlIHNpemUgb2YgdGhpcyBidWZmZXIKLSAgICBHTExvZygiZ2xUZXhQYXJhbWV0ZXJpdiIpIDw8IEdMTG9nRW51bSh0YXJnZXQpIDw8IEdMTG9nRW51bShwbmFtZSkgPDwgR0xMb2dCdWZmZXI8R0xpbnQ+KHBhcmFtcyk7Ci19Ci0KLXZvaWQgQVBJX0VOVFJZKGdsVGV4UGFyYW1ldGVyaSkoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGludCBwYXJhbSkgewotICAgIENBTExfR0xfQVBJKGdsVGV4UGFyYW1ldGVyaSwgdGFyZ2V0LCBwbmFtZSwgcGFyYW0pOwotICAgIEdMTG9nKCJnbFRleFBhcmFtZXRlcmkiKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IHBhcmFtOwotfQotdm9pZCBBUElfRU5UUlkoZ2xUZXhQYXJhbWV0ZXJ4dikoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpIHsKLSAgICBDQUxMX0dMX0FQSShnbFRleFBhcmFtZXRlcnh2LCB0YXJnZXQsIHBuYW1lLCBwYXJhbXMpOwotICAgIC8vIFhYWDogd2UgbmVlZCB0byBjb21wdXRlIHRoZSBzaXplIG9mIHRoaXMgYnVmZmVyCi0gICAgR0xMb2coImdsVGV4UGFyYW1ldGVyeHYiKSA8PCBHTExvZ0VudW0odGFyZ2V0KSA8PCBHTExvZ0VudW0ocG5hbWUpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KHBhcmFtcyk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbFBvaW50U2l6ZVBvaW50ZXJPRVMpKEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xQb2ludFNpemVQb2ludGVyT0VTLCB0eXBlLCBzdHJpZGUsIHBvaW50ZXIpOwotICAgIEdMTG9nKCJnbFBvaW50U2l6ZVBvaW50ZXJPRVMiKSA8PCBHTExvZ0VudW0odHlwZSkgPDwgc3RyaWRlIDw8IHBvaW50ZXI7Ci19Ci0KLS8vIEV4dGVuc2lvbnMKLXZvaWQgQVBJX0VOVFJZKGdsRHJhd1RleHNPRVMpKEdMc2hvcnQgeCAsIEdMc2hvcnQgeSwgR0xzaG9ydCB6LCBHTHNob3J0IHcsIEdMc2hvcnQgaCkgewotICAgIENBTExfR0xfQVBJKGdsRHJhd1RleHNPRVMsIHgsIHksIHosIHcsIGgpOwotICAgIEdMTG9nKCJnbERyYXdUZXhzT0VTIikgPDwgeCA8PCB5IDw8IHogPDwgdyA8PCBoOwotfQotdm9pZCBBUElfRU5UUlkoZ2xEcmF3VGV4aU9FUykoR0xpbnQgeCwgR0xpbnQgeSwgR0xpbnQgeiwgR0xpbnQgdywgR0xpbnQgaCkgewotICAgIENBTExfR0xfQVBJKGdsRHJhd1RleGlPRVMsIHgsIHksIHosIHcsIGgpOwotICAgIEdMTG9nKCJnbERyYXdUZXhpT0VTIikgPDwgeCA8PCB5IDw8IHogPDwgdyA8PCBoOwotfQotdm9pZCBBUElfRU5UUlkoZ2xEcmF3VGV4Zk9FUykoR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeiwgR0xmbG9hdCB3LCBHTGZsb2F0IGgpIHsKLSAgICBDQUxMX0dMX0FQSShnbERyYXdUZXhmT0VTLCB4LCB5LCB6LCB3LCBoKTsKLSAgICBHTExvZygiZ2xEcmF3VGV4Zk9FUyIpIDw8IHggPDwgeSA8PCB6IDw8IHcgPDwgaDsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsRHJhd1RleHhPRVMpKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHosIEdMZml4ZWQgdywgR0xmaXhlZCBoKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEcmF3VGV4eE9FUywgeCwgeSwgeiwgdywgaCk7Ci0gICAgR0xMb2coImdsRHJhd1RleGZPRVMiKSA8PCBHTExvZ0ZpeGVkKHgpIDw8IEdMTG9nRml4ZWQoeSkgPDwgR0xMb2dGaXhlZCh6KSA8PCBHTExvZ0ZpeGVkKHcpIDw8IEdMTG9nRml4ZWQoaCk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbERyYXdUZXhzdk9FUykoY29uc3QgR0xzaG9ydCogY29vcmRzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEcmF3VGV4c3ZPRVMsIGNvb3Jkcyk7Ci0gICAgR0xMb2coImdsRHJhd1RleHN2T0VTIikgPDwgR0xMb2dCdWZmZXI8R0xzaG9ydD4oY29vcmRzLCA1KTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsRHJhd1RleGl2T0VTKShjb25zdCBHTGludCogY29vcmRzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEcmF3VGV4aXZPRVMsIGNvb3Jkcyk7Ci0gICAgR0xMb2coImdsRHJhd1RleGl2T0VTIikgPDwgR0xMb2dCdWZmZXI8R0xpbnQ+KGNvb3JkcywgNSk7Ci19Ci12b2lkIEFQSV9FTlRSWShnbERyYXdUZXhmdk9FUykoY29uc3QgR0xmbG9hdCogY29vcmRzKSB7Ci0gICAgQ0FMTF9HTF9BUEkoZ2xEcmF3VGV4ZnZPRVMsIGNvb3Jkcyk7Ci0gICAgR0xMb2coImdsRHJhd1RleGZ2T0VTIikgPDwgR0xMb2dCdWZmZXI8R0xmbG9hdD4oY29vcmRzLCA1KTsKLX0KLXZvaWQgQVBJX0VOVFJZKGdsRHJhd1RleHh2T0VTKShjb25zdCBHTGZpeGVkKiBjb29yZHMpIHsKLSAgICBDQUxMX0dMX0FQSShnbERyYXdUZXh4dk9FUywgY29vcmRzKTsKLSAgICBHTExvZygiZ2xEcmF3VGV4eHZPRVMiKSA8PCBHTExvZ0J1ZmZlcjxHTGZpeGVkPihjb29yZHMsIDUpOwotfQotR0xiaXRmaWVsZCBBUElfRU5UUlkoZ2xRdWVyeU1hdHJpeHhPRVMpKEdMZml4ZWQqIG1hbnRpc3NhLCBHTGludCogZXhwb25lbnQpIHsKLSAgICBHTExvZygiZ2xRdWVyeU1hdHJpeHhPRVMiKSA8PCBHTExvZ0J1ZmZlcjxHTGZpeGVkPihtYW50aXNzYSwgMTYpIDw8IEdMTG9nQnVmZmVyPEdMZml4ZWQ+KGV4cG9uZW50LCAxNik7Ci0gICAgQ0FMTF9HTF9BUElfUkVUVVJOKGdsUXVlcnlNYXRyaXh4T0VTLCBtYW50aXNzYSwgZXhwb25lbnQpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYnMvZWdsX2VudHJpZXMuaW4gYi9vcGVuZ2wvbGlicy9lZ2xfZW50cmllcy5pbgpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzNiNGM2NS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGlicy9lZ2xfZW50cmllcy5pbgorKysgL2Rldi9udWxsCkBAIC0xLDQ1ICswLDAgQEAKLUVHTF9FTlRSWShFR0xEaXNwbGF5LCBlZ2xHZXREaXNwbGF5LCBOYXRpdmVEaXNwbGF5VHlwZSkKLUVHTF9FTlRSWShFR0xCb29sZWFuLCBlZ2xJbml0aWFsaXplLCBFR0xEaXNwbGF5LCBFR0xpbnQqLCBFR0xpbnQqKQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbFRlcm1pbmF0ZSwgRUdMRGlzcGxheSkKLUVHTF9FTlRSWShFR0xCb29sZWFuLCBlZ2xHZXRDb25maWdzLCBFR0xEaXNwbGF5LCBFR0xDb25maWcqLCBFR0xpbnQsIEVHTGludCopCi1FR0xfRU5UUlkoRUdMQm9vbGVhbiwgZWdsQ2hvb3NlQ29uZmlnLCBFR0xEaXNwbGF5LCBjb25zdCBFR0xpbnQgKiwgRUdMQ29uZmlnICosIEVHTGludCwgRUdMaW50ICopCi0KLUVHTF9FTlRSWShFR0xCb29sZWFuLCBlZ2xHZXRDb25maWdBdHRyaWIsIEVHTERpc3BsYXksIEVHTENvbmZpZywgRUdMaW50LCBFR0xpbnQgKikKLUVHTF9FTlRSWShFR0xTdXJmYWNlLCBlZ2xDcmVhdGVXaW5kb3dTdXJmYWNlLCBFR0xEaXNwbGF5LCBFR0xDb25maWcsIE5hdGl2ZVdpbmRvd1R5cGUsIGNvbnN0IEVHTGludCAqKQotRUdMX0VOVFJZKEVHTFN1cmZhY2UsIGVnbENyZWF0ZVBpeG1hcFN1cmZhY2UsIEVHTERpc3BsYXksIEVHTENvbmZpZywgTmF0aXZlUGl4bWFwVHlwZSwgY29uc3QgRUdMaW50ICopCi1FR0xfRU5UUlkoRUdMU3VyZmFjZSwgZWdsQ3JlYXRlUGJ1ZmZlclN1cmZhY2UsICBFR0xEaXNwbGF5LCBFR0xDb25maWcsIGNvbnN0IEVHTGludCAqKQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbERlc3Ryb3lTdXJmYWNlLCBFR0xEaXNwbGF5LCBFR0xTdXJmYWNlKQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbFF1ZXJ5U3VyZmFjZSwgIEVHTERpc3BsYXksIEVHTFN1cmZhY2UsIEVHTGludCwgRUdMaW50ICopCi1FR0xfRU5UUlkoRUdMQ29udGV4dCwgZWdsQ3JlYXRlQ29udGV4dCwgRUdMRGlzcGxheSwgRUdMQ29uZmlnLCBFR0xDb250ZXh0LCBjb25zdCBFR0xpbnQgKikKLUVHTF9FTlRSWShFR0xCb29sZWFuLCBlZ2xEZXN0cm95Q29udGV4dCwgRUdMRGlzcGxheSwgRUdMQ29udGV4dCkKLUVHTF9FTlRSWShFR0xCb29sZWFuLCBlZ2xNYWtlQ3VycmVudCwgRUdMRGlzcGxheSwgRUdMU3VyZmFjZSwgRUdMU3VyZmFjZSwgRUdMQ29udGV4dCkKLUVHTF9FTlRSWShFR0xDb250ZXh0LCBlZ2xHZXRDdXJyZW50Q29udGV4dCwgdm9pZCkKLUVHTF9FTlRSWShFR0xTdXJmYWNlLCBlZ2xHZXRDdXJyZW50U3VyZmFjZSwgRUdMaW50KQotRUdMX0VOVFJZKEVHTERpc3BsYXksIGVnbEdldEN1cnJlbnREaXNwbGF5LCB2b2lkKQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbFF1ZXJ5Q29udGV4dCwgIEVHTERpc3BsYXksIEVHTENvbnRleHQsIEVHTGludCwgRUdMaW50ICopCi1FR0xfRU5UUlkoRUdMQm9vbGVhbiwgZWdsV2FpdEdMLCB2b2lkKQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbFdhaXROYXRpdmUsIEVHTGludCkKLUVHTF9FTlRSWShFR0xCb29sZWFuLCBlZ2xTd2FwQnVmZmVycywgRUdMRGlzcGxheSwgRUdMU3VyZmFjZSkKLUVHTF9FTlRSWShFR0xCb29sZWFuLCBlZ2xDb3B5QnVmZmVycywgRUdMRGlzcGxheSwgRUdMU3VyZmFjZSwgTmF0aXZlUGl4bWFwVHlwZSkKLUVHTF9FTlRSWShFR0xpbnQsIGVnbEdldEVycm9yLCB2b2lkKQotRUdMX0VOVFJZKGNvbnN0IGNoYXIqLCBlZ2xRdWVyeVN0cmluZywgRUdMRGlzcGxheSwgRUdMaW50KQotRUdMX0VOVFJZKF9fZWdsTXVzdENhc3RUb1Byb3BlckZ1bmN0aW9uUG9pbnRlclR5cGUsIGVnbEdldFByb2NBZGRyZXNzLCBjb25zdCBjaGFyICopCi0KLS8qIEVHTCAxLjEgKi8KLQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbFN1cmZhY2VBdHRyaWIsIEVHTERpc3BsYXksIEVHTFN1cmZhY2UsIEVHTGludCwgRUdMaW50KQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbEJpbmRUZXhJbWFnZSwgRUdMRGlzcGxheSwgRUdMU3VyZmFjZSwgRUdMaW50KQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbFJlbGVhc2VUZXhJbWFnZSwgRUdMRGlzcGxheSwgRUdMU3VyZmFjZSwgRUdMaW50KQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbFN3YXBJbnRlcnZhbCwgRUdMRGlzcGxheSwgRUdMaW50KQotCi0vKiBFR0wgMS4yICovCi0KLUVHTF9FTlRSWShFR0xCb29sZWFuLCBlZ2xCaW5kQVBJLCBFR0xlbnVtKQotRUdMX0VOVFJZKEVHTGVudW0sIGVnbFF1ZXJ5QVBJLCB2b2lkKQotRUdMX0VOVFJZKEVHTEJvb2xlYW4sIGVnbFdhaXRDbGllbnQsIHZvaWQpCi1FR0xfRU5UUlkoRUdMQm9vbGVhbiwgZWdsUmVsZWFzZVRocmVhZCwgdm9pZCkKLUVHTF9FTlRSWShFR0xTdXJmYWNlLCBlZ2xDcmVhdGVQYnVmZmVyRnJvbUNsaWVudEJ1ZmZlciwgRUdMRGlzcGxheSwgRUdMZW51bSwgRUdMQ2xpZW50QnVmZmVyLCBFR0xDb25maWcsIGNvbnN0IEVHTGludCAqKQotCi0vKiBFR0wgMS4zICovCi0KLS8qIEVHTCAxLjQgKi8KZGlmZiAtLWdpdCBhL29wZW5nbC9saWJzL2VnbF9pbXBsLmggYi9vcGVuZ2wvbGlicy9lZ2xfaW1wbC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2MmNlM2ZjLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJzL2VnbF9pbXBsLmgKKysrIC9kZXYvbnVsbApAQCAtMSw0MyArMCwwIEBACi0vKiAKLSAqKiBDb3B5cmlnaHQgMjAwNywgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoqCi0gKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotICoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0gKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotICoqCi0gKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSAqKgotICoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0gKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0gKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotICoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0gKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0VHTF9JTVBMX0gKLSNkZWZpbmUgQU5EUk9JRF9FR0xfSU1QTF9ICi0KLSNpbmNsdWRlIDxjdHlwZS5oPgotCi0jaW5jbHVkZSA8RUdML2VnbC5oPgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLXN0cnVjdCBnbF9ob29rc190OwotCi1zdHJ1Y3QgZWdsX2Nvbm5lY3Rpb25fdAotewotICAgIHZvaWQgdm9sYXRpbGUgKiAgICAgZHNvOwotICAgIGdsX2hvb2tzX3QgKiAgICAgICAgaG9va3M7Ci0gICAgRUdMaW50ICAgICAgICAgICAgICBtYWpvcjsKLSAgICBFR0xpbnQgICAgICAgICAgICAgIG1pbm9yOwotICAgIGludCAgICAgICAgICAgICAgICAgdW5hdmFpbGFibGU7Ci19OwotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLyogQU5EUk9JRF9FR0xfSU1QTF9IICovCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGlicy9nbF9lbnRyaWVzLmluIGIvb3BlbmdsL2xpYnMvZ2xfZW50cmllcy5pbgpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYjk3ZThmZS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGlicy9nbF9lbnRyaWVzLmluCisrKyAvZGV2L251bGwKQEAgLTEsMTU5ICswLDAgQEAKLUdMX0VOVFJZKHZvaWQsIGdsQ29sb3I0ZiwgR0xmbG9hdCwgR0xmbG9hdCwgR0xmbG9hdCwgR0xmbG9hdCkKLUdMX0VOVFJZKHZvaWQsIGdsQ29sb3I0eCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCkKLUdMX0VOVFJZKHZvaWQsIGdsTm9ybWFsM2YsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQpCi1HTF9FTlRSWSh2b2lkLCBnbE5vcm1hbDN4LCBHTGZpeGVkLCBHTGZpeGVkLCBHTGZpeGVkKQotR0xfRU5UUlkodm9pZCwgZ2xDdWxsRmFjZSwgR0xlbnVtKQotR0xfRU5UUlkodm9pZCwgZ2xGcm9udEZhY2UsIEdMZW51bSkKLUdMX0VOVFJZKHZvaWQsIGdsRGlzYWJsZSwgR0xlbnVtKQotR0xfRU5UUlkodm9pZCwgZ2xFbmFibGUsIEdMZW51bSkKLUdMX0VOVFJZKHZvaWQsIGdsRmluaXNoLCB2b2lkKQotR0xfRU5UUlkodm9pZCwgZ2xGbHVzaCwgdm9pZCkKLUdMX0VOVFJZKEdMZW51bSwgZ2xHZXRFcnJvciwgdm9pZCkKLUdMX0VOVFJZKGNvbnN0IEdMdWJ5dGUqLCBnbEdldFN0cmluZywgR0xlbnVtKQotR0xfRU5UUlkodm9pZCwgZ2xHZXRJbnRlZ2VydiwgR0xlbnVtLCBHTGludCAqKQotR0xfRU5UUlkodm9pZCwgZ2xDb2xvck1hc2ssIEdMYm9vbGVhbiwgR0xib29sZWFuLCBHTGJvb2xlYW4sIEdMYm9vbGVhbikKLUdMX0VOVFJZKHZvaWQsIGdsRGVwdGhNYXNrLCBHTGJvb2xlYW4pCi1HTF9FTlRSWSh2b2lkLCBnbFN0ZW5jaWxNYXNrLCBHTHVpbnQpCi1HTF9FTlRSWSh2b2lkLCBnbERlcHRoRnVuYywgR0xlbnVtKQotR0xfRU5UUlkodm9pZCwgZ2xEZXB0aFJhbmdlZiwgR0xjbGFtcGYgek5lYXIsIEdMY2xhbXBmIHpGYXIpCi1HTF9FTlRSWSh2b2lkLCBnbERlcHRoUmFuZ2V4LCBHTGNsYW1weCB6TmVhciwgR0xjbGFtcHggekZhcikKLUdMX0VOVFJZKHZvaWQsIGdsUG9seWdvbk9mZnNldCwgR0xmbG9hdCBmYWN0b3IsIEdMZmxvYXQgdW5pdHMpCi1HTF9FTlRSWSh2b2lkLCBnbFBvbHlnb25PZmZzZXR4LCBHTGZpeGVkIGZhY3RvciwgR0xmaXhlZCB1bml0cykKLUdMX0VOVFJZKHZvaWQsIGdsTG9naWNPcCwgR0xlbnVtIG9wY29kZSkKLUdMX0VOVFJZKHZvaWQsIGdsQWxwaGFGdW5jeCwgR0xlbnVtIGZ1bmMsIEdMY2xhbXB4IHJlZikKLUdMX0VOVFJZKHZvaWQsIGdsQWxwaGFGdW5jLCBHTGVudW0gZnVuYywgR0xjbGFtcGYgcmVmKQotR0xfRU5UUlkodm9pZCwgZ2xCbGVuZEZ1bmMsIEdMZW51bSBzZmFjdG9yLCBHTGVudW0gZGZhY3RvcikKLUdMX0VOVFJZKHZvaWQsIGdsQ2xlYXIsIEdMYml0ZmllbGQgbWFzaykKLUdMX0VOVFJZKHZvaWQsIGdsQ2xlYXJDb2xvciwgR0xjbGFtcGYgciwgR0xjbGFtcGYgZywgR0xjbGFtcGYgYiwgR0xjbGFtcGYgYSkKLUdMX0VOVFJZKHZvaWQsIGdsQ2xlYXJDb2xvcngsIEdMY2xhbXB4IHIsIEdMY2xhbXB4IGcsIEdMY2xhbXB4IGIsIEdMY2xhbXB4IGEpCi1HTF9FTlRSWSh2b2lkLCBnbENsZWFyRGVwdGhmLCBHTGNsYW1wZiBkZXB0aCkKLUdMX0VOVFJZKHZvaWQsIGdsQ2xlYXJEZXB0aHgsIEdMY2xhbXB4IGRlcHRoKQotR0xfRU5UUlkodm9pZCwgZ2xDbGVhclN0ZW5jaWwsIEdMaW50IHMpCi1HTF9FTlRSWSh2b2lkLCBnbFBvaW50U2l6ZSwgR0xmbG9hdCkKLUdMX0VOVFJZKHZvaWQsIGdsUG9pbnRTaXpleCwgR0xmaXhlZCkKLUdMX0VOVFJZKHZvaWQsIGdsU2FtcGxlQ292ZXJhZ2UsIEdMY2xhbXBmIHZhbHVlLCBHTGJvb2xlYW4gaW52ZXJ0KQotR0xfRU5UUlkodm9pZCwgZ2xTYW1wbGVDb3ZlcmFnZXgsIEdMY2xhbXB4IHZhbHVlLCBHTGJvb2xlYW4gaW52ZXJ0KQotR0xfRU5UUlkodm9pZCwgZ2xTdGVuY2lsRnVuYywgR0xlbnVtIGZ1bmMsIEdMaW50IHJlZiwgR0x1aW50IG1hc2spCi1HTF9FTlRSWSh2b2lkLCBnbFN0ZW5jaWxPcCwgR0xlbnVtIGZhaWwsIEdMZW51bSB6ZmFpbCwgR0xlbnVtIHpwYXNzKQotR0xfRU5UUlkodm9pZCwgZ2xTY2lzc29yLCBHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCkKLUdMX0VOVFJZKHZvaWQsIGdsSGludCwgR0xlbnVtLCBHTGVudW0gbW9kZSkKLUdMX0VOVFJZKHZvaWQsIGdsTGluZVdpZHRoLCBHTGZsb2F0IHdpZHRoKQotR0xfRU5UUlkodm9pZCwgZ2xMaW5lV2lkdGh4LCBHTGZpeGVkIHdpZHRoKQotR0xfRU5UUlkodm9pZCwgZ2xTaGFkZU1vZGVsLCBHTGVudW0pCi1HTF9FTlRSWSh2b2lkLCBnbExpZ2h0TW9kZWxmLCBHTGVudW0sIEdMZmxvYXQpCi1HTF9FTlRSWSh2b2lkLCBnbExpZ2h0TW9kZWxmdiwgR0xlbnVtLCBjb25zdCBHTGZsb2F0ICopCi1HTF9FTlRSWSh2b2lkLCBnbExpZ2h0TW9kZWx4LCBHTGVudW0sIEdMZml4ZWQpCi1HTF9FTlRSWSh2b2lkLCBnbExpZ2h0TW9kZWx4diwgR0xlbnVtLCBjb25zdCBHTGZpeGVkICopCi1HTF9FTlRSWSh2b2lkLCBnbExpZ2h0ZiwgR0xlbnVtLCBHTGVudW0sIEdMZmxvYXQpCi1HTF9FTlRSWSh2b2lkLCBnbExpZ2h0ZnYsIEdMZW51bSwgR0xlbnVtLCBjb25zdCBHTGZsb2F0ICopCi1HTF9FTlRSWSh2b2lkLCBnbExpZ2h0eCwgR0xlbnVtLCBHTGVudW0sIEdMZml4ZWQpCi1HTF9FTlRSWSh2b2lkLCBnbExpZ2h0eHYsIEdMZW51bSwgR0xlbnVtLCBjb25zdCBHTGZpeGVkICopCi1HTF9FTlRSWSh2b2lkLCBnbE1hdGVyaWFsZiwgR0xlbnVtLCBHTGVudW0sIEdMZmxvYXQpCi1HTF9FTlRSWSh2b2lkLCBnbE1hdGVyaWFsZnYsIEdMZW51bSwgR0xlbnVtLCBjb25zdCBHTGZsb2F0ICopCi1HTF9FTlRSWSh2b2lkLCBnbE1hdGVyaWFseCwgR0xlbnVtLCBHTGVudW0sIEdMZml4ZWQpCi1HTF9FTlRSWSh2b2lkLCBnbE1hdGVyaWFseHYsIEdMZW51bSwgR0xlbnVtLCBjb25zdCBHTGZpeGVkICopCi1HTF9FTlRSWSh2b2lkLCBnbEZvZ2YsIEdMZW51bSwgR0xmbG9hdCkKLUdMX0VOVFJZKHZvaWQsIGdsRm9nZnYsIEdMZW51bSwgY29uc3QgR0xmbG9hdCAqKQotR0xfRU5UUlkodm9pZCwgZ2xGb2d4LCBHTGVudW0sIEdMZml4ZWQpCi1HTF9FTlRSWSh2b2lkLCBnbEZvZ3h2LCBHTGVudW0sIGNvbnN0IEdMZml4ZWQgKikKLUdMX0VOVFJZKHZvaWQsIGdsVmVydGV4UG9pbnRlciwgR0xpbnQsIEdMZW51bSwgR0xzaXplaSwgY29uc3QgR0x2b2lkICopCi1HTF9FTlRSWSh2b2lkLCBnbENvbG9yUG9pbnRlciwgR0xpbnQsIEdMZW51bSwgR0xzaXplaSwgY29uc3QgR0x2b2lkICopCi1HTF9FTlRSWSh2b2lkLCBnbE5vcm1hbFBvaW50ZXIsIEdMZW51bSwgR0xzaXplaSwgY29uc3QgR0x2b2lkICopCi1HTF9FTlRSWSh2b2lkLCBnbFRleENvb3JkUG9pbnRlciwgR0xpbnQsIEdMZW51bSwgR0xzaXplaSwgY29uc3QgR0x2b2lkICopCi1HTF9FTlRSWSh2b2lkLCBnbEVuYWJsZUNsaWVudFN0YXRlLCBHTGVudW0pCi1HTF9FTlRSWSh2b2lkLCBnbERpc2FibGVDbGllbnRTdGF0ZSwgR0xlbnVtKQotR0xfRU5UUlkodm9pZCwgZ2xDbGllbnRBY3RpdmVUZXh0dXJlLCBHTGVudW0pCi1HTF9FTlRSWSh2b2lkLCBnbERyYXdBcnJheXMsIEdMZW51bSwgR0xpbnQgZmlyc3QsIEdMc2l6ZWkpCi1HTF9FTlRSWSh2b2lkLCBnbERyYXdFbGVtZW50cywgR0xlbnVtLCBHTHNpemVpLCBHTGVudW0sIGNvbnN0IEdMdm9pZCAqKQotR0xfRU5UUlkodm9pZCwgZ2xMb2FkSWRlbnRpdHksIHZvaWQpCi1HTF9FTlRSWSh2b2lkLCBnbExvYWRNYXRyaXhmLCBjb25zdCBHTGZsb2F0KikKLUdMX0VOVFJZKHZvaWQsIGdsTG9hZE1hdHJpeHgsIGNvbnN0IEdMZml4ZWQqKQotR0xfRU5UUlkodm9pZCwgZ2xNYXRyaXhNb2RlLCBHTGVudW0gbW9kZSkKLUdMX0VOVFJZKHZvaWQsIGdsTXVsdE1hdHJpeGYsIGNvbnN0IEdMZmxvYXQqKQotR0xfRU5UUlkodm9pZCwgZ2xNdWx0TWF0cml4eCwgY29uc3QgR0xmaXhlZCopCi1HTF9FTlRSWSh2b2lkLCBnbFBvcE1hdHJpeCwgdm9pZCkKLUdMX0VOVFJZKHZvaWQsIGdsUHVzaE1hdHJpeCwgdm9pZCkKLUdMX0VOVFJZKHZvaWQsIGdsRnJ1c3R1bWYsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQpCi1HTF9FTlRSWSh2b2lkLCBnbEZydXN0dW14LCBHTGZpeGVkLCBHTGZpeGVkLCBHTGZpeGVkLCBHTGZpeGVkLCBHTGZpeGVkLCBHTGZpeGVkKQotR0xfRU5UUlkodm9pZCwgZ2xPcnRob2YsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQpCi1HTF9FTlRSWSh2b2lkLCBnbE9ydGhveCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCkKLUdMX0VOVFJZKHZvaWQsIGdsUm90YXRlZiwgR0xmbG9hdCwgR0xmbG9hdCwgR0xmbG9hdCwgR0xmbG9hdCkKLUdMX0VOVFJZKHZvaWQsIGdsUm90YXRleCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCkKLUdMX0VOVFJZKHZvaWQsIGdsU2NhbGVmLCBHTGZsb2F0LCBHTGZsb2F0LCBHTGZsb2F0KQotR0xfRU5UUlkodm9pZCwgZ2xTY2FsZXgsIEdMZml4ZWQsIEdMZml4ZWQsIEdMZml4ZWQpCi1HTF9FTlRSWSh2b2lkLCBnbFRyYW5zbGF0ZWYsIEdMZmxvYXQsIEdMZmxvYXQsIEdMZmxvYXQpCi1HTF9FTlRSWSh2b2lkLCBnbFRyYW5zbGF0ZXgsIEdMZml4ZWQsIEdMZml4ZWQsIEdMZml4ZWQpCi1HTF9FTlRSWSh2b2lkLCBnbFZpZXdwb3J0LCBHTGludCwgR0xpbnQsIEdMc2l6ZWksIEdMc2l6ZWkpCi1HTF9FTlRSWSh2b2lkLCBnbEFjdGl2ZVRleHR1cmUsIEdMZW51bSkKLUdMX0VOVFJZKHZvaWQsIGdsQmluZFRleHR1cmUsIEdMZW51bSwgR0x1aW50KQotR0xfRU5UUlkodm9pZCwgZ2xHZW5UZXh0dXJlcywgR0xzaXplaSwgR0x1aW50KikKLUdMX0VOVFJZKHZvaWQsIGdsRGVsZXRlVGV4dHVyZXMsIEdMc2l6ZWkgbiwgY29uc3QgR0x1aW50ICopCi1HTF9FTlRSWSh2b2lkLCBnbE11bHRpVGV4Q29vcmQ0ZiwgR0xlbnVtLCBHTGZsb2F0LCBHTGZsb2F0LCBHTGZsb2F0LCBHTGZsb2F0KQotR0xfRU5UUlkodm9pZCwgZ2xNdWx0aVRleENvb3JkNHgsIEdMZW51bSwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCkKLUdMX0VOVFJZKHZvaWQsIGdsUGl4ZWxTdG9yZWksIEdMZW51bSwgR0xpbnQpCi1HTF9FTlRSWSh2b2lkLCBnbFRleEVudmYsIEdMZW51bSwgR0xlbnVtLCBHTGZsb2F0KQotR0xfRU5UUlkodm9pZCwgZ2xUZXhFbnZmdiwgR0xlbnVtLCBHTGVudW0sIGNvbnN0IEdMZmxvYXQqKQotR0xfRU5UUlkodm9pZCwgZ2xUZXhFbnZ4LCBHTGVudW0sIEdMZW51bSwgR0xmaXhlZCkKLUdMX0VOVFJZKHZvaWQsIGdsVGV4RW52eHYsIEdMZW51bSwgR0xlbnVtLCBjb25zdCBHTGZpeGVkKikKLUdMX0VOVFJZKHZvaWQsIGdsVGV4UGFyYW1ldGVyZiwgR0xlbnVtLCBHTGVudW0sIEdMZmxvYXQpCi1HTF9FTlRSWSh2b2lkLCBnbFRleFBhcmFtZXRlcngsIEdMZW51bSwgR0xlbnVtLCBHTGZpeGVkKQotR0xfRU5UUlkodm9pZCwgZ2xDb21wcmVzc2VkVGV4SW1hZ2UyRCwgICAgR0xlbnVtLCBHTGludCwgR0xlbnVtLCBHTHNpemVpLCBHTHNpemVpLCBHTGludCwgR0xzaXplaSwgY29uc3QgR0x2b2lkKikKLUdMX0VOVFJZKHZvaWQsIGdsQ29tcHJlc3NlZFRleFN1YkltYWdlMkQsIEdMZW51bSwgR0xpbnQsIEdMaW50LCBHTGludCwgR0xzaXplaSwgR0xzaXplaSwgR0xlbnVtLCBHTHNpemVpLCBjb25zdCBHTHZvaWQqKQotR0xfRU5UUlkodm9pZCwgZ2xDb3B5VGV4SW1hZ2UyRCwgR0xlbnVtLCBHTGludCwgR0xlbnVtLCBHTGludCwgR0xpbnQsIEdMc2l6ZWksIEdMc2l6ZWksIEdMaW50KQotR0xfRU5UUlkodm9pZCwgZ2xDb3B5VGV4U3ViSW1hZ2UyRCwgR0xlbnVtLCBHTGludCwgR0xpbnQsIEdMaW50LCBHTGludCwgR0xpbnQsIEdMc2l6ZWksIEdMc2l6ZWkpCi1HTF9FTlRSWSh2b2lkLCBnbFRleEltYWdlMkQsIEdMZW51bSwgR0xpbnQsIEdMaW50LCBHTHNpemVpLCBHTHNpemVpLCBHTGludCwgR0xlbnVtLCBHTGVudW0sIGNvbnN0IEdMdm9pZCopCi1HTF9FTlRSWSh2b2lkLCBnbFRleFN1YkltYWdlMkQsIEdMZW51bSwgR0xpbnQsIEdMaW50LCBHTGludCwgR0xzaXplaSwgR0xzaXplaSwgR0xlbnVtLCBHTGVudW0sIGNvbnN0IEdMdm9pZCopCi1HTF9FTlRSWSh2b2lkLCBnbFJlYWRQaXhlbHMsIEdMaW50LCBHTGludCwgR0xzaXplaSwgR0xzaXplaSwgR0xlbnVtLCBHTGVudW0sIEdMdm9pZCAqKQotCi0vLyAxLjEgYWRkaXRpb25zCi1HTF9FTlRSWSh2b2lkLCBnbENsaXBQbGFuZWYsIEdMZW51bSBwbGFuZSwgY29uc3QgR0xmbG9hdCopCi1HTF9FTlRSWSh2b2lkLCBnbENsaXBQbGFuZXgsIEdMZW51bSBwbGFuZSwgY29uc3QgR0xmaXhlZCopCi1HTF9FTlRSWSh2b2lkLCBnbEJpbmRCdWZmZXIsIEdMZW51bSwgR0x1aW50KQotR0xfRU5UUlkodm9pZCwgZ2xCdWZmZXJEYXRhLCBHTGVudW0sIEdMc2l6ZWlwdHIsIGNvbnN0IEdMdm9pZCosIEdMZW51bSkKLUdMX0VOVFJZKHZvaWQsIGdsQnVmZmVyU3ViRGF0YSwgR0xlbnVtLCBHTGludHB0ciwgR0xzaXplaXB0ciwgY29uc3QgR0x2b2lkKikKLUdMX0VOVFJZKHZvaWQsIGdsRGVsZXRlQnVmZmVycywgR0xzaXplaSwgY29uc3QgR0x1aW50KikKLUdMX0VOVFJZKHZvaWQsIGdsR2VuQnVmZmVycywgR0xzaXplaSwgR0x1aW50KikKLUdMX0VOVFJZKHZvaWQsIGdsR2V0Qm9vbGVhbnYsIEdMZW51bSwgR0xib29sZWFuICopCi1HTF9FTlRSWSh2b2lkLCBnbEdldEZpeGVkdiwgR0xlbnVtLCBHTGZpeGVkICopCi1HTF9FTlRSWSh2b2lkLCBnbEdldEZsb2F0diwgR0xlbnVtLCBHTGZsb2F0ICopCi1HTF9FTlRSWSh2b2lkLCBnbEdldFBvaW50ZXJ2LCBHTGVudW0sIHZvaWQgKiopCi1HTF9FTlRSWSh2b2lkLCBnbEdldEJ1ZmZlclBhcmFtZXRlcml2LCBHTGVudW0sIEdMZW51bSwgR0xpbnQgKikKLUdMX0VOVFJZKHZvaWQsIGdsR2V0Q2xpcFBsYW5lZiwgR0xlbnVtLCBHTGZsb2F0WzRdKQotR0xfRU5UUlkodm9pZCwgZ2xHZXRDbGlwUGxhbmV4LCBHTGVudW0sIEdMZml4ZWRbNF0pCi1HTF9FTlRSWSh2b2lkLCBnbEdldExpZ2h0eHYsIEdMZW51bSwgR0xlbnVtLCBHTGZpeGVkICopCi1HTF9FTlRSWSh2b2lkLCBnbEdldExpZ2h0ZnYsIEdMZW51bSwgR0xlbnVtLCBHTGZsb2F0ICopCi1HTF9FTlRSWSh2b2lkLCBnbEdldE1hdGVyaWFseHYsIEdMZW51bSwgR0xlbnVtLCBHTGZpeGVkICopCi1HTF9FTlRSWSh2b2lkLCBnbEdldE1hdGVyaWFsZnYsIEdMZW51bSwgR0xlbnVtLCBHTGZsb2F0ICopCi1HTF9FTlRSWSh2b2lkLCBnbEdldFRleEVudmZ2LCBHTGVudW0sIEdMZW51bSwgR0xmbG9hdCAqKQotR0xfRU5UUlkodm9pZCwgZ2xHZXRUZXhFbnZpdiwgR0xlbnVtLCBHTGVudW0sIEdMaW50ICopCi1HTF9FTlRSWSh2b2lkLCBnbEdldFRleEVudnh2LCBHTGVudW0sIEdMZW51bSwgR0xmaXhlZCAqKQotR0xfRU5UUlkodm9pZCwgZ2xHZXRUZXhQYXJhbWV0ZXJmdiwgR0xlbnVtLCBHTGVudW0sIEdMZmxvYXQgKikKLUdMX0VOVFJZKHZvaWQsIGdsR2V0VGV4UGFyYW1ldGVyaXYsIEdMZW51bSwgR0xlbnVtLCBHTGludCAqKQotR0xfRU5UUlkodm9pZCwgZ2xHZXRUZXhQYXJhbWV0ZXJ4diwgR0xlbnVtLCBHTGVudW0sIEdMZml4ZWQgKikKLUdMX0VOVFJZKEdMYm9vbGVhbiwgZ2xJc0J1ZmZlciwgR0x1aW50KQotR0xfRU5UUlkoR0xib29sZWFuLCBnbElzRW5hYmxlZCwgR0xlbnVtKQotR0xfRU5UUlkoR0xib29sZWFuLCBnbElzVGV4dHVyZSwgR0x1aW50KQotR0xfRU5UUlkodm9pZCwgZ2xQb2ludFBhcmFtZXRlcmYsIEdMZW51bSwgR0xmbG9hdCkKLUdMX0VOVFJZKHZvaWQsIGdsUG9pbnRQYXJhbWV0ZXJmdiwgR0xlbnVtLCBjb25zdCBHTGZsb2F0ICopCi1HTF9FTlRSWSh2b2lkLCBnbFBvaW50UGFyYW1ldGVyeCwgR0xlbnVtLCBHTGZpeGVkKQotR0xfRU5UUlkodm9pZCwgZ2xQb2ludFBhcmFtZXRlcnh2LCBHTGVudW0sIGNvbnN0IEdMZml4ZWQgKikKLUdMX0VOVFJZKHZvaWQsIGdsQ29sb3I0dWIsIEdMdWJ5dGUsIEdMdWJ5dGUsIEdMdWJ5dGUsIEdMdWJ5dGUpCi1HTF9FTlRSWSh2b2lkLCBnbFRleEVudmksIEdMZW51bSwgR0xlbnVtLCBHTGludCkKLUdMX0VOVFJZKHZvaWQsIGdsVGV4RW52aXYsIEdMZW51bSwgR0xlbnVtLCBjb25zdCBHTGludCAqKQotR0xfRU5UUlkodm9pZCwgZ2xUZXhQYXJhbWV0ZXJmdiwgR0xlbnVtLCBHTGVudW0sIGNvbnN0IEdMZmxvYXQgKikKLUdMX0VOVFJZKHZvaWQsIGdsVGV4UGFyYW1ldGVyaXYsIEdMZW51bSwgR0xlbnVtLCBjb25zdCBHTGludCAqKQotR0xfRU5UUlkodm9pZCwgZ2xUZXhQYXJhbWV0ZXJpLCBHTGVudW0sIEdMZW51bSwgR0xpbnQpCi1HTF9FTlRSWSh2b2lkLCBnbFRleFBhcmFtZXRlcnh2LCBHTGVudW0sIEdMZW51bSwgY29uc3QgR0xmaXhlZCAqKQotR0xfRU5UUlkodm9pZCwgZ2xQb2ludFNpemVQb2ludGVyT0VTLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCopCi0KLS8vIEV4dGVuc2lvbnMKLUdMX0VOVFJZKHZvaWQsIGdsRHJhd1RleHNPRVMsIEdMc2hvcnQsIEdMc2hvcnQsIEdMc2hvcnQsIEdMc2hvcnQsIEdMc2hvcnQpCi1HTF9FTlRSWSh2b2lkLCBnbERyYXdUZXhpT0VTLCBHTGludCwgR0xpbnQsIEdMaW50LCBHTGludCwgR0xpbnQpCi1HTF9FTlRSWSh2b2lkLCBnbERyYXdUZXhmT0VTLCBHTGZsb2F0LCBHTGZsb2F0LCBHTGZsb2F0LCBHTGZsb2F0LCBHTGZsb2F0KQotR0xfRU5UUlkodm9pZCwgZ2xEcmF3VGV4eE9FUywgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCwgR0xmaXhlZCkKLUdMX0VOVFJZKHZvaWQsIGdsRHJhd1RleHN2T0VTLCBjb25zdCBHTHNob3J0KikKLUdMX0VOVFJZKHZvaWQsIGdsRHJhd1RleGl2T0VTLCBjb25zdCBHTGludCopCi1HTF9FTlRSWSh2b2lkLCBnbERyYXdUZXhmdk9FUywgY29uc3QgR0xmbG9hdCopCi1HTF9FTlRSWSh2b2lkLCBnbERyYXdUZXh4dk9FUywgY29uc3QgR0xmaXhlZCopCi1HTF9FTlRSWShHTGJpdGZpZWxkLCBnbFF1ZXJ5TWF0cml4eE9FUywgR0xmaXhlZCogbWFudGlzc2EsIEdMaW50KiBleHBvbmVudCkKLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYnMvZ2xfZW51bXMuaW4gYi9vcGVuZ2wvbGlicy9nbF9lbnVtcy5pbgpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZmZjMmZhZC4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvbGlicy9nbF9lbnVtcy5pbgorKysgL2Rldi9udWxsCkBAIC0xLDI2MSArMCwwIEBACi1HTEVOVU0oR0xfUE9JTlRTLCAweDAwMDApCi1HTEVOVU0oR0xfTElORVMsIDB4MDAwMSkKLUdMRU5VTShHTF9MSU5FX0xPT1AsIDB4MDAwMikKLUdMRU5VTShHTF9MSU5FX1NUUklQLCAweDAwMDMpCi1HTEVOVU0oR0xfVFJJQU5HTEVTLCAweDAwMDQpCi1HTEVOVU0oR0xfVFJJQU5HTEVfU1RSSVAsIDB4MDAwNSkKLUdMRU5VTShHTF9UUklBTkdMRV9GQU4sIDB4MDAwNikKLUdMRU5VTShHTF9BREQsIDB4MDEwNCkKLUdMRU5VTShHTF9ORVZFUiwgMHgwMjAwKQotR0xFTlVNKEdMX0xFU1MsIDB4MDIwMSkKLUdMRU5VTShHTF9FUVVBTCwgMHgwMjAyKQotR0xFTlVNKEdMX0xFUVVBTCwgMHgwMjAzKQotR0xFTlVNKEdMX0dSRUFURVIsIDB4MDIwNCkKLUdMRU5VTShHTF9OT1RFUVVBTCwgMHgwMjA1KQotR0xFTlVNKEdMX0dFUVVBTCwgMHgwMjA2KQotR0xFTlVNKEdMX0FMV0FZUywgMHgwMjA3KQotR0xFTlVNKEdMX1NSQ19DT0xPUiwgMHgwMzAwKQotR0xFTlVNKEdMX09ORV9NSU5VU19TUkNfQ09MT1IsIDB4MDMwMSkKLUdMRU5VTShHTF9TUkNfQUxQSEEsIDB4MDMwMikKLUdMRU5VTShHTF9PTkVfTUlOVVNfU1JDX0FMUEhBLCAweDAzMDMpCi1HTEVOVU0oR0xfRFNUX0FMUEhBLCAweDAzMDQpCi1HTEVOVU0oR0xfT05FX01JTlVTX0RTVF9BTFBIQSwgMHgwMzA1KQotR0xFTlVNKEdMX0RTVF9DT0xPUiwgMHgwMzA2KQotR0xFTlVNKEdMX09ORV9NSU5VU19EU1RfQ09MT1IsIDB4MDMwNykKLUdMRU5VTShHTF9TUkNfQUxQSEFfU0FUVVJBVEUsIDB4MDMwOCkKLUdMRU5VTShHTF9GUk9OVCwgMHgwNDA0KQotR0xFTlVNKEdMX0JBQ0ssIDB4MDQwNSkKLUdMRU5VTShHTF9GUk9OVF9BTkRfQkFDSywgMHgwNDA4KQotR0xFTlVNKEdMX0lOVkFMSURfRU5VTSwgMHgwNTAwKQotR0xFTlVNKEdMX0lOVkFMSURfVkFMVUUsIDB4MDUwMSkKLUdMRU5VTShHTF9JTlZBTElEX09QRVJBVElPTiwgMHgwNTAyKQotR0xFTlVNKEdMX1NUQUNLX09WRVJGTE9XLCAweDA1MDMpCi1HTEVOVU0oR0xfU1RBQ0tfVU5ERVJGTE9XLCAweDA1MDQpCi1HTEVOVU0oR0xfT1VUX09GX01FTU9SWSwgMHgwNTA1KQotR0xFTlVNKEdMX0VYUCwgMHgwODAwKQotR0xFTlVNKEdMX0VYUDIsIDB4MDgwMSkKLUdMRU5VTShHTF9DVywgMHgwOTAwKQotR0xFTlVNKEdMX0NDVywgMHgwOTAxKQotR0xFTlVNKEdMX1BPSU5UX1NNT09USCwgMHgwQjEwKQotR0xFTlVNKEdMX1NNT09USF9QT0lOVF9TSVpFX1JBTkdFLCAweDBCMTIpCi1HTEVOVU0oR0xfTElORV9TTU9PVEgsIDB4MEIyMCkKLUdMRU5VTShHTF9TTU9PVEhfTElORV9XSURUSF9SQU5HRSwgMHgwQjIyKQotR0xFTlVNKEdMX0NVTExfRkFDRSwgMHgwQjQ0KQotR0xFTlVNKEdMX0xJR0hUSU5HLCAweDBCNTApCi1HTEVOVU0oR0xfTElHSFRfTU9ERUxfVFdPX1NJREUsIDB4MEI1MikKLUdMRU5VTShHTF9MSUdIVF9NT0RFTF9BTUJJRU5ULCAweDBCNTMpCi1HTEVOVU0oR0xfQ09MT1JfTUFURVJJQUwsIDB4MEI1NykKLUdMRU5VTShHTF9GT0csIDB4MEI2MCkKLUdMRU5VTShHTF9GT0dfREVOU0lUWSwgMHgwQjYyKQotR0xFTlVNKEdMX0ZPR19TVEFSVCwgMHgwQjYzKQotR0xFTlVNKEdMX0ZPR19FTkQsIDB4MEI2NCkKLUdMRU5VTShHTF9GT0dfTU9ERSwgMHgwQjY1KQotR0xFTlVNKEdMX0ZPR19DT0xPUiwgMHgwQjY2KQotR0xFTlVNKEdMX0RFUFRIX1RFU1QsIDB4MEI3MSkKLUdMRU5VTShHTF9TVEVOQ0lMX1RFU1QsIDB4MEI5MCkKLUdMRU5VTShHTF9OT1JNQUxJWkUsIDB4MEJBMSkKLUdMRU5VTShHTF9BTFBIQV9URVNULCAweDBCQzApCi1HTEVOVU0oR0xfRElUSEVSLCAweDBCRDApCi1HTEVOVU0oR0xfQkxFTkQsIDB4MEJFMikKLUdMRU5VTShHTF9DT0xPUl9MT0dJQ19PUCwgMHgwQkYyKQotR0xFTlVNKEdMX1NDSVNTT1JfVEVTVCwgMHgwQzExKQotR0xFTlVNKEdMX1BFUlNQRUNUSVZFX0NPUlJFQ1RJT05fSElOVCwgMHgwQzUwKQotR0xFTlVNKEdMX1BPSU5UX1NNT09USF9ISU5ULCAweDBDNTEpCi1HTEVOVU0oR0xfTElORV9TTU9PVEhfSElOVCwgMHgwQzUyKQotR0xFTlVNKEdMX1BPTFlHT05fU01PT1RIX0hJTlQsIDB4MEM1MykKLUdMRU5VTShHTF9GT0dfSElOVCwgMHgwQzU0KQotR0xFTlVNKEdMX1VOUEFDS19BTElHTk1FTlQsIDB4MENGNSkKLUdMRU5VTShHTF9QQUNLX0FMSUdOTUVOVCwgMHgwRDA1KQotR0xFTlVNKEdMX01BWF9MSUdIVFMsIDB4MEQzMSkKLUdMRU5VTShHTF9NQVhfQ0xJUF9QTEFORVMsIDB4MEQzMikKLUdMRU5VTShHTF9NQVhfVEVYVFVSRV9TSVpFLCAweDBEMzMpCi1HTEVOVU0oR0xfTUFYX01PREVMVklFV19TVEFDS19ERVBUSCwgMHgwRDM2KQotR0xFTlVNKEdMX01BWF9QUk9KRUNUSU9OX1NUQUNLX0RFUFRILCAweDBEMzgpCi1HTEVOVU0oR0xfTUFYX1RFWFRVUkVfU1RBQ0tfREVQVEgsIDB4MEQzOSkKLUdMRU5VTShHTF9NQVhfVklFV1BPUlRfRElNUywgMHgwRDNBKQotR0xFTlVNKEdMX1JFRF9CSVRTLCAweDBENTIpCi1HTEVOVU0oR0xfR1JFRU5fQklUUywgMHgwRDUzKQotR0xFTlVNKEdMX0JMVUVfQklUUywgMHgwRDU0KQotR0xFTlVNKEdMX0FMUEhBX0JJVFMsIDB4MEQ1NSkKLUdMRU5VTShHTF9ERVBUSF9CSVRTLCAweDBENTYpCi1HTEVOVU0oR0xfU1RFTkNJTF9CSVRTLCAweDBENTcpCi1HTEVOVU0oR0xfVEVYVFVSRV8yRCwgMHgwREUxKQotR0xFTlVNKEdMX0RPTlRfQ0FSRSwgMHgxMTAwKQotR0xFTlVNKEdMX0ZBU1RFU1QsIDB4MTEwMSkKLUdMRU5VTShHTF9OSUNFU1QsIDB4MTEwMikKLUdMRU5VTShHTF9BTUJJRU5ULCAweDEyMDApCi1HTEVOVU0oR0xfRElGRlVTRSwgMHgxMjAxKQotR0xFTlVNKEdMX1NQRUNVTEFSLCAweDEyMDIpCi1HTEVOVU0oR0xfUE9TSVRJT04sIDB4MTIwMykKLUdMRU5VTShHTF9TUE9UX0RJUkVDVElPTiwgMHgxMjA0KQotR0xFTlVNKEdMX1NQT1RfRVhQT05FTlQsIDB4MTIwNSkKLUdMRU5VTShHTF9TUE9UX0NVVE9GRiwgMHgxMjA2KQotR0xFTlVNKEdMX0NPTlNUQU5UX0FUVEVOVUFUSU9OLCAweDEyMDcpCi1HTEVOVU0oR0xfTElORUFSX0FUVEVOVUFUSU9OLCAweDEyMDgpCi1HTEVOVU0oR0xfUVVBRFJBVElDX0FUVEVOVUFUSU9OLCAweDEyMDkpCi1HTEVOVU0oR0xfQllURSwgMHgxNDAwKQotR0xFTlVNKEdMX1VOU0lHTkVEX0JZVEUsIDB4MTQwMSkKLUdMRU5VTShHTF9TSE9SVCwgMHgxNDAyKQotR0xFTlVNKEdMX1VOU0lHTkVEX1NIT1JULCAweDE0MDMpCi1HTEVOVU0oR0xfRkxPQVQsIDB4MTQwNikKLUdMRU5VTShHTF9GSVhFRCwgMHgxNDBDKQotR0xFTlVNKEdMX0NMRUFSLCAweDE1MDApCi1HTEVOVU0oR0xfQU5ELCAweDE1MDEpCi1HTEVOVU0oR0xfQU5EX1JFVkVSU0UsIDB4MTUwMikKLUdMRU5VTShHTF9DT1BZLCAweDE1MDMpCi1HTEVOVU0oR0xfQU5EX0lOVkVSVEVELCAweDE1MDQpCi1HTEVOVU0oR0xfTk9PUCwgMHgxNTA1KQotR0xFTlVNKEdMX1hPUiwgMHgxNTA2KQotR0xFTlVNKEdMX09SLCAweDE1MDcpCi1HTEVOVU0oR0xfTk9SLCAweDE1MDgpCi1HTEVOVU0oR0xfRVFVSVYsIDB4MTUwOSkKLUdMRU5VTShHTF9JTlZFUlQsIDB4MTUwQSkKLUdMRU5VTShHTF9PUl9SRVZFUlNFLCAweDE1MEIpCi1HTEVOVU0oR0xfQ09QWV9JTlZFUlRFRCwgMHgxNTBDKQotR0xFTlVNKEdMX09SX0lOVkVSVEVELCAweDE1MEQpCi1HTEVOVU0oR0xfTkFORCwgMHgxNTBFKQotR0xFTlVNKEdMX1NFVCwgMHgxNTBGKQotR0xFTlVNKEdMX0VNSVNTSU9OLCAweDE2MDApCi1HTEVOVU0oR0xfU0hJTklORVNTLCAweDE2MDEpCi1HTEVOVU0oR0xfQU1CSUVOVF9BTkRfRElGRlVTRSwgMHgxNjAyKQotR0xFTlVNKEdMX01PREVMVklFVywgMHgxNzAwKQotR0xFTlVNKEdMX1BST0pFQ1RJT04sIDB4MTcwMSkKLUdMRU5VTShHTF9URVhUVVJFLCAweDE3MDIpCi1HTEVOVU0oR0xfQUxQSEEsIDB4MTkwNikKLUdMRU5VTShHTF9SR0IsIDB4MTkwNykKLUdMRU5VTShHTF9SR0JBLCAweDE5MDgpCi1HTEVOVU0oR0xfTFVNSU5BTkNFLCAweDE5MDkpCi1HTEVOVU0oR0xfTFVNSU5BTkNFX0FMUEhBLCAweDE5MEEpCi1HTEVOVU0oR0xfRkxBVCwgMHgxRDAwKQotR0xFTlVNKEdMX1NNT09USCwgMHgxRDAxKQotR0xFTlVNKEdMX0tFRVAsIDB4MUUwMCkKLUdMRU5VTShHTF9SRVBMQUNFLCAweDFFMDEpCi1HTEVOVU0oR0xfUkVQTEFDRSwgMHgxRTAxKQotR0xFTlVNKEdMX0lOQ1IsIDB4MUUwMikKLUdMRU5VTShHTF9ERUNSLCAweDFFMDMpCi1HTEVOVU0oR0xfVkVORE9SLCAweDFGMDApCi1HTEVOVU0oR0xfUkVOREVSRVIsIDB4MUYwMSkKLUdMRU5VTShHTF9WRVJTSU9OLCAweDFGMDIpCi1HTEVOVU0oR0xfRVhURU5TSU9OUywgMHgxRjAzKQotR0xFTlVNKEdMX01PRFVMQVRFLCAweDIxMDApCi1HTEVOVU0oR0xfREVDQUwsIDB4MjEwMSkKLUdMRU5VTShHTF9URVhUVVJFX0VOVl9NT0RFLCAweDIyMDApCi1HTEVOVU0oR0xfVEVYVFVSRV9FTlZfQ09MT1IsIDB4MjIwMSkKLUdMRU5VTShHTF9URVhUVVJFX0VOViwgMHgyMzAwKQotR0xFTlVNKEdMX05FQVJFU1QsIDB4MjYwMCkKLUdMRU5VTShHTF9MSU5FQVIsIDB4MjYwMSkKLUdMRU5VTShHTF9ORUFSRVNUX01JUE1BUF9ORUFSRVNULCAweDI3MDApCi1HTEVOVU0oR0xfTElORUFSX01JUE1BUF9ORUFSRVNULCAweDI3MDEpCi1HTEVOVU0oR0xfTkVBUkVTVF9NSVBNQVBfTElORUFSLCAweDI3MDIpCi1HTEVOVU0oR0xfTElORUFSX01JUE1BUF9MSU5FQVIsIDB4MjcwMykKLUdMRU5VTShHTF9URVhUVVJFX01BR19GSUxURVIsIDB4MjgwMCkKLUdMRU5VTShHTF9URVhUVVJFX01JTl9GSUxURVIsIDB4MjgwMSkKLUdMRU5VTShHTF9URVhUVVJFX1dSQVBfUywgMHgyODAyKQotR0xFTlVNKEdMX1RFWFRVUkVfV1JBUF9ULCAweDI4MDMpCi1HTEVOVU0oR0xfQ0xBTVAsIDB4MjkwMCkKLUdMRU5VTShHTF9SRVBFQVQsIDB4MjkwMSkKLUdMRU5VTShHTF9DTElQX1BMQU5FMCwgMHgzMDAwKQotR0xFTlVNKEdMX0NMSVBfUExBTkUxLCAweDMwMDEpCi1HTEVOVU0oR0xfQ0xJUF9QTEFORTIsIDB4MzAwMikKLUdMRU5VTShHTF9DTElQX1BMQU5FMywgMHgzMDAzKQotR0xFTlVNKEdMX0NMSVBfUExBTkU0LCAweDMwMDQpCi1HTEVOVU0oR0xfQ0xJUF9QTEFORTUsIDB4MzAwNSkKLUdMRU5VTShHTF9MSUdIVDAsIDB4NDAwMCkKLUdMRU5VTShHTF9MSUdIVDEsIDB4NDAwMSkKLUdMRU5VTShHTF9MSUdIVDIsIDB4NDAwMikKLUdMRU5VTShHTF9MSUdIVDMsIDB4NDAwMykKLUdMRU5VTShHTF9MSUdIVDQsIDB4NDAwNCkKLUdMRU5VTShHTF9MSUdIVDUsIDB4NDAwNSkKLUdMRU5VTShHTF9MSUdIVDYsIDB4NDAwNikKLUdMRU5VTShHTF9MSUdIVDcsIDB4NDAwNykKLUdMRU5VTShHTF9ESVJFQ1RfVEVYVFVSRV8yRF9RVUFMQ09NTSwgMHg3RTgwKQotR0xFTlVNKEdMX1VOU0lHTkVEX1NIT1JUXzRfNF80XzQsIDB4ODAzMykKLUdMRU5VTShHTF9VTlNJR05FRF9TSE9SVF81XzVfNV8xLCAweDgwMzQpCi1HTEVOVU0oR0xfUE9MWUdPTl9PRkZTRVRfRklMTCwgMHg4MDM3KQotR0xFTlVNKEdMX1JFU0NBTEVfTk9STUFMLCAweDgwM0EpCi1HTEVOVU0oR0xfVkVSVEVYX0FSUkFZLCAweDgwNzQpCi1HTEVOVU0oR0xfTk9STUFMX0FSUkFZLCAweDgwNzUpCi1HTEVOVU0oR0xfQ09MT1JfQVJSQVksIDB4ODA3NikKLUdMRU5VTShHTF9URVhUVVJFX0NPT1JEX0FSUkFZLCAweDgwNzgpCi1HTEVOVU0oR0xfTVVMVElTQU1QTEUsIDB4ODA5RCkKLUdMRU5VTShHTF9TQU1QTEVfQUxQSEFfVE9fQ09WRVJBR0UsIDB4ODA5RSkKLUdMRU5VTShHTF9TQU1QTEVfQUxQSEFfVE9fT05FLCAweDgwOUYpCi1HTEVOVU0oR0xfU0FNUExFX0NPVkVSQUdFLCAweDgwQTApCi1HTEVOVU0oR0xfTUFYX0VMRU1FTlRTX1ZFUlRJQ0VTLCAweDgwRTgpCi1HTEVOVU0oR0xfTUFYX0VMRU1FTlRTX0lORElDRVMsIDB4ODBFOSkKLUdMRU5VTShHTF9DTEFNUF9UT19FREdFLCAweDgxMkYpCi1HTEVOVU0oR0xfR0VORVJBVEVfTUlQTUFQLCAweDgxOTEpCi1HTEVOVU0oR0xfR0VORVJBVEVfTUlQTUFQX0hJTlQsIDB4ODE5MikKLUdMRU5VTShHTF9VTlNJR05FRF9TSE9SVF81XzZfNSwgMHg4MzYzKQotR0xFTlVNKEdMX0FMSUFTRURfUE9JTlRfU0laRV9SQU5HRSwgMHg4NDZEKQotR0xFTlVNKEdMX0FMSUFTRURfTElORV9XSURUSF9SQU5HRSwgMHg4NDZFKQotR0xFTlVNKEdMX1RFWFRVUkUwLCAweDg0QzApCi1HTEVOVU0oR0xfVEVYVFVSRTEsIDB4ODRDMSkKLUdMRU5VTShHTF9URVhUVVJFMiwgMHg4NEMyKQotR0xFTlVNKEdMX1RFWFRVUkUzLCAweDg0QzMpCi1HTEVOVU0oR0xfVEVYVFVSRTQsIDB4ODRDNCkKLUdMRU5VTShHTF9URVhUVVJFNSwgMHg4NEM1KQotR0xFTlVNKEdMX1RFWFRVUkU2LCAweDg0QzYpCi1HTEVOVU0oR0xfVEVYVFVSRTcsIDB4ODRDNykKLUdMRU5VTShHTF9URVhUVVJFOCwgMHg4NEM4KQotR0xFTlVNKEdMX1RFWFRVUkU5LCAweDg0QzkpCi1HTEVOVU0oR0xfVEVYVFVSRTEwLCAweDg0Q0EpCi1HTEVOVU0oR0xfVEVYVFVSRTExLCAweDg0Q0IpCi1HTEVOVU0oR0xfVEVYVFVSRTEyLCAweDg0Q0MpCi1HTEVOVU0oR0xfVEVYVFVSRTEzLCAweDg0Q0QpCi1HTEVOVU0oR0xfVEVYVFVSRTE0LCAweDg0Q0UpCi1HTEVOVU0oR0xfVEVYVFVSRTE1LCAweDg0Q0YpCi1HTEVOVU0oR0xfVEVYVFVSRTE2LCAweDg0RDApCi1HTEVOVU0oR0xfVEVYVFVSRTE3LCAweDg0RDEpCi1HTEVOVU0oR0xfVEVYVFVSRTE4LCAweDg0RDIpCi1HTEVOVU0oR0xfVEVYVFVSRTE5LCAweDg0RDMpCi1HTEVOVU0oR0xfVEVYVFVSRTIwLCAweDg0RDQpCi1HTEVOVU0oR0xfVEVYVFVSRTIxLCAweDg0RDUpCi1HTEVOVU0oR0xfVEVYVFVSRTIyLCAweDg0RDYpCi1HTEVOVU0oR0xfVEVYVFVSRTIzLCAweDg0RDcpCi1HTEVOVU0oR0xfVEVYVFVSRTI0LCAweDg0RDgpCi1HTEVOVU0oR0xfVEVYVFVSRTI1LCAweDg0RDkpCi1HTEVOVU0oR0xfVEVYVFVSRTI2LCAweDg0REEpCi1HTEVOVU0oR0xfVEVYVFVSRTI3LCAweDg0REIpCi1HTEVOVU0oR0xfVEVYVFVSRTI4LCAweDg0REMpCi1HTEVOVU0oR0xfVEVYVFVSRTI5LCAweDg0REQpCi1HTEVOVU0oR0xfVEVYVFVSRTMwLCAweDg0REUpCi1HTEVOVU0oR0xfVEVYVFVSRTMxLCAweDg0REYpCi1HTEVOVU0oR0xfTUFYX1RFWFRVUkVfVU5JVFMsIDB4ODRFMikKLUdMRU5VTShHTF9OVU1fQ09NUFJFU1NFRF9URVhUVVJFX0ZPUk1BVFMsIDB4ODZBMikKLUdMRU5VTShHTF9DT01QUkVTU0VEX1RFWFRVUkVfRk9STUFUUywgMHg4NkEzKQotR0xFTlVNKEdMX0JVRkZFUl9TSVpFLCAweDg3NjQpCi1HTEVOVU0oR0xfQlVGRkVSX1VTQUdFLCAweDg3NjUpCi1HTEVOVU0oR0xfUE9JTlRfU1BSSVRFX09FUywgMHg4ODYxKQotR0xFTlVNKEdMX0NPT1JEX1JFUExBQ0VfT0VTLCAweDg4NjIpCi1HTEVOVU0oR0xfQVJSQVlfQlVGRkVSLCAweDg4OTIpCi1HTEVOVU0oR0xfRUxFTUVOVF9BUlJBWV9CVUZGRVIsIDB4ODg5MykKLUdMRU5VTShHTF9BUlJBWV9CVUZGRVJfQklORElORywgMHg4ODk0KQotR0xFTlVNKEdMX0VMRU1FTlRfQVJSQVlfQlVGRkVSX0JJTkRJTkcsIDB4ODg5NSkKLUdMRU5VTShHTF9WRVJURVhfQVJSQVlfQlVGRkVSX0JJTkRJTkcsIDB4ODg5NikKLUdMRU5VTShHTF9OT1JNQUxfQVJSQVlfQlVGRkVSX0JJTkRJTkcsIDB4ODg5NykKLUdMRU5VTShHTF9DT0xPUl9BUlJBWV9CVUZGRVJfQklORElORywgMHg4ODk4KQotR0xFTlVNKEdMX1RFWFRVUkVfQ09PUkRfQVJSQVlfQlVGRkVSX0JJTkRJTkcsIDB4ODg5QSkKLUdMRU5VTShHTF9TVEFUSUNfRFJBVywgMHg4OEU0KQotR0xFTlVNKEdMX0RZTkFNSUNfRFJBVywgMHg4OEU4KQotR0xFTlVNKEdMX1BPSU5UX1NJWkVfQVJSQVlfVFlQRV9PRVMsIDB4ODk4QSkKLUdMRU5VTShHTF9QT0lOVF9TSVpFX0FSUkFZX1NUUklERV9PRVMsIDB4ODk4QikKLUdMRU5VTShHTF9QT0lOVF9TSVpFX0FSUkFZX1BPSU5URVJfT0VTLCAweDg5OEMpCi1HTEVOVU0oR0xfTU9ERUxWSUVXX01BVFJJWF9GTE9BVF9BU19JTlRfQklUU19PRVMsIDB4ODk4RCkKLUdMRU5VTShHTF9QUk9KRUNUSU9OX01BVFJJWF9GTE9BVF9BU19JTlRfQklUU19PRVMsIDB4ODk4RSkKLUdMRU5VTShHTF9URVhUVVJFX01BVFJJWF9GTE9BVF9BU19JTlRfQklUU19PRVMsIDB4ODk4RikKLUdMRU5VTShHTF9QQUxFVFRFNF9SR0I4X09FUywgMHg4QjkwKQotR0xFTlVNKEdMX1BBTEVUVEU0X1JHQkE4X09FUywgMHg4QjkxKQotR0xFTlVNKEdMX1BBTEVUVEU0X1I1X0c2X0I1X09FUywgMHg4QjkyKQotR0xFTlVNKEdMX1BBTEVUVEU0X1JHQkE0X09FUywgMHg4QjkzKQotR0xFTlVNKEdMX1BBTEVUVEU0X1JHQjVfQTFfT0VTLCAweDhCOTQpCi1HTEVOVU0oR0xfUEFMRVRURThfUkdCOF9PRVMsIDB4OEI5NSkKLUdMRU5VTShHTF9QQUxFVFRFOF9SR0JBOF9PRVMsIDB4OEI5NikKLUdMRU5VTShHTF9QQUxFVFRFOF9SNV9HNl9CNV9PRVMsIDB4OEI5NykKLUdMRU5VTShHTF9QQUxFVFRFOF9SR0JBNF9PRVMsIDB4OEI5OCkKLUdMRU5VTShHTF9QQUxFVFRFOF9SR0I1X0ExX09FUywgMHg4Qjk5KQotR0xFTlVNKEdMX0lNUExFTUVOVEFUSU9OX0NPTE9SX1JFQURfVFlQRV9PRVMsIDB4OEI5QSkKLUdMRU5VTShHTF9JTVBMRU1FTlRBVElPTl9DT0xPUl9SRUFEX0ZPUk1BVF9PRVMsIDB4OEI5QikKLUdMRU5VTShHTF9QT0lOVF9TSVpFX0FSUkFZX09FUywgMHg4QjlDKQotR0xFTlVNKEdMX1RFWFRVUkVfQ1JPUF9SRUNUX09FUywgMHg4QjlEKQotR0xFTlVNKEdMX1BPSU5UX1NJWkVfQVJSQVlfQlVGRkVSX0JJTkRJTkdfT0VTLCAweDhCOUYpCmRpZmYgLS1naXQgYS9vcGVuZ2wvbGlicy9nbF9sb2dnZXIuaCBiL29wZW5nbC9saWJzL2dsX2xvZ2dlci5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjZTg1ZGQxLi4wMDAwMDAwCi0tLSBhL29wZW5nbC9saWJzL2dsX2xvZ2dlci5oCisrKyAvZGV2L251bGwKQEAgLTEsMjYgKzAsMCBAQAotLyogCi0gKiogQ29weXJpZ2h0IDIwMDcsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSAqKgotICoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSAqKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotICoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSAqKgotICoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0gKioKLSAqKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotICoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotICoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSAqKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotICoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotICovCi0KLSNpZm5kZWYgQU5EUk9JRF9HTF9MT0dHRVJfSAotI2RlZmluZSBBTkRST0lEX0dMX0xPR0dFUl9ICi0KLW5hbWVzcGFjZSBhbmRyb2lkIHsKLSNkZWZpbmUgR0xfRU5UUlkociwgYXBpLCAuLi4pIHIgbG9nXyMjYXBpKF9fVkFfQVJHU19fKTsKLSNpbmNsdWRlICJnbF9lbnRyaWVzLmluIgotI3VuZGVmIEdMX0VOVFJZCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotCi0jZW5kaWYgLyogQU5EUk9JRF9HTF9MT0dHRVJfSCAqLwpkaWZmIC0tZ2l0IGEvb3BlbmdsL2xpYnMvaG9va3MuaCBiL29wZW5nbC9saWJzL2hvb2tzLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDYzZmIwMTcuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYnMvaG9va3MuaAorKysgL2Rldi9udWxsCkBAIC0xLDEzNCArMCwwIEBACi0vKiAKLSAqKiBDb3B5cmlnaHQgMjAwNywgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoqCi0gKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotICoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0gKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotICoqCi0gKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSAqKgotICoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0gKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0gKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotICoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0gKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0gKi8KLQotI2lmbmRlZiBBTkRST0lEX0dMRVNfQ01fSE9PS1NfSAotI2RlZmluZSBBTkRST0lEX0dMRVNfQ01fSE9PS1NfSAotCi0jaW5jbHVkZSA8Y3R5cGUuaD4KLSNpbmNsdWRlIDxzdHJpbmcuaD4KLSNpbmNsdWRlIDxlcnJuby5oPgotCi0jaW5jbHVkZSA8RUdML2VnbC5oPgotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLQotI2RlZmluZSBHTF9MT0dHRVIgICAgICAgICAgICAgICAgICAgMAotI2lmICFkZWZpbmVkKF9fYXJtX18pCi0jZGVmaW5lIFVTRV9TTE9XX0JJTkRJTkcgICAgICAgICAgICAxCi0jZWxzZQotI2RlZmluZSBVU0VfU0xPV19CSU5ESU5HICAgICAgICAgICAgMAotI2VuZGlmCi0jdW5kZWYgTkVMRU0KLSNkZWZpbmUgTkVMRU0oeCkgICAgICAgICAgICAgICAgICAgIChzaXplb2YoeCkvc2l6ZW9mKCooeCkpKQotI2RlZmluZSBNQVhfTlVNQkVSX09GX0dMX0VYVEVOU0lPTlMgMzIKLQotCi0jaWYgZGVmaW5lZChIQVZFX0FORFJPSURfT1MpICYmICFVU0VfU0xPV19CSU5ESU5HICYmICFHTF9MT0dHRVIgJiYgX19PUFRJTUlaRV9fCi0jZGVmaW5lIFVTRV9GQVNUX1RMU19LRVkgICAgICAgICAgICAxCi0jZWxzZQotI2RlZmluZSBVU0VfRkFTVF9UTFNfS0VZICAgICAgICAgICAgMAotI2VuZGlmCi0KLSNpZiBVU0VfRkFTVF9UTFNfS0VZCi0jICAgaW5jbHVkZSA8YmlvbmljX3Rscy5oPiAgLyogc3BlY2lhbCBwcml2YXRlIEMgbGlicmFyeSBoZWFkZXIgKi8KLSNlbmRpZgotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi1uYW1lc3BhY2UgYW5kcm9pZCB7Ci0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0KLS8vICBFR0xEaXNwbGF5IGFyZSBnbG9iYWwsIG5vdCBhdHRhY2hlZCB0byBhIGdpdmVuIHRocmVhZAotY29uc3QgdW5zaWduZWQgaW50IE5VTV9ESVNQTEFZUyA9IDE7Ci0KLWVudW0gewotICAgIElNUExfSEFSRFdBUkUgPSAwLAotICAgIElNUExfU09GVFdBUkUsCi0gICAgSU1QTF9DT05URVhUX0xPU1QsCi0gICAgSU1QTF9OT19DT05URVhULAotICAgIAotICAgIElNUExfTlVNX0lNUExFTUVOVEFUSU9OUwotfTsKLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0vLyBHTCAvIEVHTCBob29rcwotCi0jdW5kZWYgR0xfRU5UUlkKLSN1bmRlZiBFR0xfRU5UUlkKLSNkZWZpbmUgR0xfRU5UUlkoX3IsIF9hcGksIC4uLikgX3IgKCpfYXBpKShfX1ZBX0FSR1NfXyk7Ci0jZGVmaW5lIEVHTF9FTlRSWShfciwgX2FwaSwgLi4uKSBfciAoKl9hcGkpKF9fVkFfQVJHU19fKTsKLQotc3RydWN0IGdsX2hvb2tzX3QgewotICAgIHN0cnVjdCBnbF90IHsKLSAgICAgICAgI2luY2x1ZGUgImdsX2VudHJpZXMuaW4iCi0gICAgfSBnbDsKLSAgICBzdHJ1Y3QgZWdsX3QgewotICAgICAgICAjaW5jbHVkZSAiZWdsX2VudHJpZXMuaW4iCi0gICAgfSBlZ2w7Ci0gICAgc3RydWN0IGdsX2V4dF90IHsKLSAgICAgICAgdm9pZCAoKmV4dGVuc2lvbnNbTUFYX05VTUJFUl9PRl9HTF9FWFRFTlNJT05TXSkodm9pZCk7Ci0gICAgfSBleHQ7Ci19OwotI3VuZGVmIEdMX0VOVFJZCi0jdW5kZWYgRUdMX0VOVFJZCi0KLQotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi1leHRlcm4gZ2xfaG9va3NfdCBnSG9va3NbSU1QTF9OVU1fSU1QTEVNRU5UQVRJT05TXTsKLWV4dGVybiBwdGhyZWFkX2tleV90IGdHTFdyYXBwZXJLZXk7Ci0KLSNpZiBVU0VfRkFTVF9UTFNfS0VZCi0KLS8vIFdlIGhhdmUgYSBkZWRpY2F0ZWQgVExTIHNsb3QgaW4gYmlvbmljCi1zdGF0aWMgaW5saW5lIGdsX2hvb2tzX3QgY29uc3QgKiB2b2xhdGlsZSAqIGdldF90bHNfaG9va3MoKSB7Ci0gICAgdm9sYXRpbGUgdm9pZCAqdGxzX2Jhc2UgPSBfX2dldF90bHMoKTsKLSAgICBnbF9ob29rc190IGNvbnN0ICogdm9sYXRpbGUgKiB0bHNfaG9va3MgPSAKLSAgICAgICAgICAgIHJlaW50ZXJwcmV0X2Nhc3Q8Z2xfaG9va3NfdCBjb25zdCAqIHZvbGF0aWxlICo+KHRsc19iYXNlKTsKLSAgICByZXR1cm4gdGxzX2hvb2tzOwotfQotCi1zdGF0aWMgaW5saW5lIHZvaWQgc2V0R2xUaHJlYWRTcGVjaWZpYyhnbF9ob29rc190IGNvbnN0ICp2YWx1ZSkgewotICAgIGdsX2hvb2tzX3QgY29uc3QgKiB2b2xhdGlsZSAqIHRsc19ob29rcyA9IGdldF90bHNfaG9va3MoKTsKLSAgICB0bHNfaG9va3NbVExTX1NMT1RfT1BFTkdMX0FQSV0gPSB2YWx1ZTsKLX0KLQotc3RhdGljIGdsX2hvb2tzX3QgY29uc3QqIGdldEdsVGhyZWFkU3BlY2lmaWMoKSB7Ci0gICAgZ2xfaG9va3NfdCBjb25zdCAqIHZvbGF0aWxlICogdGxzX2hvb2tzID0gZ2V0X3Rsc19ob29rcygpOwotICAgIGdsX2hvb2tzX3QgY29uc3QqIGhvb2tzID0gdGxzX2hvb2tzW1RMU19TTE9UX09QRU5HTF9BUEldOwotICAgIGlmIChob29rcykgcmV0dXJuIGhvb2tzOwotICAgIHJldHVybiAmZ0hvb2tzW0lNUExfTk9fQ09OVEVYVF07Ci19Ci0KLSNlbHNlCi0KLXN0YXRpYyBpbmxpbmUgdm9pZCBzZXRHbFRocmVhZFNwZWNpZmljKGdsX2hvb2tzX3QgY29uc3QgKnZhbHVlKSB7Ci0gICAgcHRocmVhZF9zZXRzcGVjaWZpYyhnR0xXcmFwcGVyS2V5LCB2YWx1ZSk7Ci19Ci0KLXN0YXRpYyBnbF9ob29rc190IGNvbnN0KiBnZXRHbFRocmVhZFNwZWNpZmljKCkgewotICAgIGdsX2hvb2tzX3QgY29uc3QqIGhvb2tzID0gIHN0YXRpY19jYXN0PGdsX2hvb2tzX3QqPihwdGhyZWFkX2dldHNwZWNpZmljKGdHTFdyYXBwZXJLZXkpKTsKLSAgICBpZiAoaG9va3MpIHJldHVybiBob29rczsKLSAgICByZXR1cm4gJmdIb29rc1tJTVBMX05PX0NPTlRFWFRdOwotfQotCi0jZW5kaWYKLQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi19OyAvLyBuYW1lc3BhY2UgYW5kcm9pZAotLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCi0jZW5kaWYgLyogQU5EUk9JRF9HTEVTX0NNX0hPT0tTX0ggKi8KZGlmZiAtLWdpdCBhL29wZW5nbC9saWJzL3Rvb2xzL2VudW1leHRyYWN0LnNoIGIvb3BlbmdsL2xpYnMvdG9vbHMvZW51bWV4dHJhY3Quc2gKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDU3MDczMDIuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL2xpYnMvdG9vbHMvZW51bWV4dHJhY3Quc2gKKysrIC9kZXYvbnVsbApAQCAtMSwzMiArMCwwIEBACi0jIS9iaW4vc2gKLQotYXdrICcKLS9eI2RlZmluZSBHTF8vIHsKLSAgICBuYW1lc1tjb3VudF0gPSAkMjsKLSAgICB2YWx1ZXNbY291bnRdID0gJDM7Ci0gICAgc29ydFtjb3VudF0gPSAkMyArIDA7Ci0gICAgY291bnQrKzsKLX0KLUVORCB7Ci0gICAgZm9yIChpID0gMTsgaSA8IGNvdW50OyBpKyspIHsKLSAgICAgICAgZm9yIChqID0gMDsgaiA8IGk7IGorKykgewotICAgICAgICAgICAgaWYgKHNvcnRbaV0gPCBzb3J0W2pdKSB7Ci0gICAgICAgICAgICAgICAgdG4gPSBuYW1lc1tpXTsKLSAgICAgICAgICAgICAgICB0diA9IHZhbHVlc1tpXTsKLSAgICAgICAgICAgICAgICB0cyA9IHNvcnRbaV07Ci0gICAgICAgICAgICAgICAgbmFtZXNbaV0gPSBuYW1lc1tqXTsKLSAgICAgICAgICAgICAgICB2YWx1ZXNbaV0gPSB2YWx1ZXNbal07Ci0gICAgICAgICAgICAgICAgc29ydFtpXSA9IHNvcnRbal07Ci0gICAgICAgICAgICAgICAgbmFtZXNbal0gPSB0bjsKLSAgICAgICAgICAgICAgICB2YWx1ZXNbal0gPSB0djsKLSAgICAgICAgICAgICAgICBzb3J0W2pdID0gdHM7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLSAgICB9Ci0gCi0gICAgZm9yIChpID0gMDsgaSA8IGNvdW50OyBpKyspIHsKLSAgICAgICAgcHJpbnRmKCJHTEVOVU0oJXMsICVzKVxuIiwgbmFtZXNbaV0sIHZhbHVlc1tpXSk7Ci0gICAgfQotfQotJyA8ICQxCi0KZGlmZiAtLWdpdCBhL29wZW5nbC90ZXN0cy9BbmRyb2lkLm1rIGIvb3BlbmdsL3Rlc3RzL0FuZHJvaWQubWsKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDUwNTNlN2QuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rlc3RzL0FuZHJvaWQubWsKKysrIC9kZXYvbnVsbApAQCAtMSArMCwwIEBACi1pbmNsdWRlICQoY2FsbCBhbGwtc3ViZGlyLW1ha2VmaWxlcykKZGlmZiAtLWdpdCBhL29wZW5nbC90ZXN0cy9hbmdlbGVzL0FuZHJvaWQubWsgYi9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9BbmRyb2lkLm1rCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA0Njk1OGQzLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy9hbmdlbGVzL0FuZHJvaWQubWsKKysrIC9kZXYvbnVsbApAQCAtMSwxNyArMCwwIEBACi0jIENvcHlyaWdodCAyMDA2IFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLQotTE9DQUxfUEFUSDo9ICQoY2FsbCBteS1kaXIpCi1pbmNsdWRlICQoQ0xFQVJfVkFSUykKLUxPQ0FMX1NSQ19GSUxFUzo9IGFwcC1saW51eC5jIGRlbW8uYy5hcm0KLUxPQ0FMX1NIQVJFRF9MSUJSQVJJRVMgOj0gbGliRUdMIGxpYkdMRVN2MV9DTSBsaWJ1aQotTE9DQUxfTU9EVUxFOj0gYW5nZWxlcwotTE9DQUxfTU9EVUxFX1RBR1MgOj0gdGVzdHMKLWluY2x1ZGUgJChCVUlMRF9FWEVDVVRBQkxFKQotCi0KLWluY2x1ZGUgJChDTEVBUl9WQVJTKQotTE9DQUxfU1JDX0ZJTEVTOj0gZ3B1c3RhdGUuYwotTE9DQUxfU0hBUkVEX0xJQlJBUklFUyA6PSBsaWJFR0wgbGliR0xFU3YxX0NNCi1MT0NBTF9NT0RVTEU6PSBncHVzdGF0ZQotTE9DQUxfTU9EVUxFX1RBR1MgOj0gdGVzdHMKLWluY2x1ZGUgJChCVUlMRF9FWEVDVVRBQkxFKQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvTU9EVUxFX0xJQ0VOU0VfQlNEX09SX0xHUEwgYi9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9NT0RVTEVfTElDRU5TRV9CU0RfT1JfTEdQTApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZTY5ZGUyOS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9NT0RVTEVfTElDRU5TRV9CU0RfT1JfTEdQTAorKysgL2Rldi9udWxsCmRpZmYgLS1naXQgYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9SRUFETUUudHh0IGIvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvUkVBRE1FLnR4dApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMzhiOGE0YS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9SRUFETUUudHh0CisrKyAvZGV2L251bGwKQEAgLTEsNzcgKzAsMCBAQAotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tDQotU2FuIEFuZ2VsZXMgT2JzZXJ2YXRpb24gT3BlbkdMIEVTIHZlcnNpb24gZXhhbXBsZQ0KLUNvcHlyaWdodCAyMDA0LTIwMDUgSmV0cm8gTGF1aGENCi1XZWI6IGh0dHA6Ly9pa2kuZmkvamV0cm8vDQotU2VlIGZpbGUgbGljZW5zZS50eHQgZm9yIGxpY2Vuc2luZyBpbmZvcm1hdGlvbi4NCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0NCi0NCi1UaGlzIGlzIGFuIE9wZW5HTCBFUyBwb3J0IG9mIHRoZSBzbWFsbCBzZWxmLXJ1bm5pbmcgZGVtb25zdHJhdGlvbg0KLWNhbGxlZCAiU2FuIEFuZ2VsZXMgT2JzZXJ2YXRpb24iLCB3aGljaCB3YXMgZmlyc3QgcHJlc2VudGVkIGluIHRoZQ0KLUFzc2VtYmx5JzIwMDQgZXZlbnQuIEl0IHdvbiB0aGUgZmlyc3QgcGxhY2UgaW4gdGhlIDQgS0IgaW50cm8NCi1jb21wZXRpdGlvbiBjYXRlZ29yeS4NCi0NCi1UaGUgZGVtb25zdHJhdGlvbiBmZWF0dXJlcyBhIHNpZ2h0c2VlaW5nIG9mIGEgZnV0dXJpc3RpYyBjaXR5DQotaGF2aW5nIG1hbnkgZGlmZmVyZW50IGtpbmQgb2YgYnVpbGRpbmdzIGFuZCBpdGVtcy4gRXZlcnl0aGluZyBpcw0KLWZsYXQgc2hhZGVkIHdpdGggdGhyZWUgZGlmZmVyZW50IGxpZ2h0cy4NCi0NCi1UaGUgb3JpZ2luYWwgdmVyc2lvbiB3YXMgbWFkZSBmb3IgZGVza3RvcCB3aXRoIE9wZW5HTC4gSXQgd2FzDQotbmF0dXJhbGx5IGhlYXZpbHkgc2l6ZSBvcHRpbWl6ZWQgaW4gb3JkZXIgdG8gZml0IGl0IGluIHRoZSBzaXplDQotbGltaXQuIEZvciB0aGlzIE9wZW5HTCBFUyB2ZXJzaW9uIGV4YW1wbGUgbXVjaCBvZiB0aGUgY29kZSBpcw0KLWNsZWFuZWQgdXAgYW5kIHRoZSBzb3VuZCBpcyByZW1vdmVkLiBBbHNvIGRldGFpbCBsZXZlbCBpcyBsb3dlcmVkLA0KLWFsdGhvdWdoIGl0IHN0aWxsIGNvbnRhaW5zIG92ZXIgNjAwMDAgZmFjZXMuDQotDQotVGhlIFdpbjMyICgyMDAwL1hQKSBiaW5hcnkgcGFja2FnZSBvZiBvcmlnaW5hbCB2ZXJzaW9uIGlzDQotYXZhaWxhYmxlIGZyb20gdGhpcyBhZGRyZXNzOiBodHRwOi8vamV0LnJvL2ZpbGVzL2FuZ2VsZXMuemlwDQotDQotRmlyc3QgdmVyc2lvbiBvZiB0aGlzIE9wZW5HTCBFUyBwb3J0IHdhcyBzdWJtaXR0ZWQgdG8gdGhlIEtocm9ub3MNCi1PcGVuR0wgRVMgQ29kaW5nIENoYWxsZW5nZSBoZWxkIGluIDIwMDQtMjAwNS4NCi0NCi1BcyBhIGNvZGUgZXhhbXBsZSwgdGhpcyBzb3VyY2Ugc2hvd3MgdGhlIGZvbGxvd2luZzoNCi0gICogSG93IHRvIGNyZWF0ZSBhIG1pbmltYWwgYW5kIHBvcnRhYmxlIGFkIGhvYyBmcmFtZXdvcmsNCi0gICAgZm9yIHNtYWxsIHRlc3RpbmcvZGVtb25zdHJhdGlvbiBwcm9ncmFtcy4gVGhpcyBmcmFtZXdvcmsNCi0gICAgY29tcGlsZXMgZm9yIGJvdGggZGVza3RvcCBhbmQgUG9ja2V0UEMgV2luMzIgZW52aXJvbm1lbnQsDQotICAgIGFuZCBhIHNlcGFyYXRlIHNvdXJjZSBpcyBpbmNsdWRlZCBmb3IgTGludXggd2l0aCBYMTEuDQotICAqIEhvdyB0byBkeW5hbWljYWxseSBmaW5kIGFuZCB1c2UgdGhlIE9wZW5HTCBFUyBETEwgb3INCi0gICAgc2hhcmVkIG9iamVjdCwgc28gdGhhdCB0aGUgbGlicmFyeSBpcyBub3QgbmVlZGVkIGF0DQotICAgIHRoZSBjb21waWxlL2xpbmsgc3RhZ2UuDQotICAqIEhvdyB0byB1c2UgdGhlIGJhc2ljIGZlYXR1cmVzIG9mIE9wZW5HTCBFUyAxLjAvMS4xDQotICAgIENvbW1vbiBMaXRlLCBzdWNoIGFzIHZlcnRleCBhcnJheXMsIGNvbG9yIGFycmF5cyBhbmQNCi0gICAgbGlnaHRpbmcuDQotICAqIEhvdyB0byBjcmVhdGUgYSBzZWxmIGNvbnRhaW5lZCBzbWFsbCBkZW1vbnN0cmF0aW9uDQotICAgIGFwcGxpY2F0aW9uIHdpdGggb2JqZWN0cyBnZW5lcmF0ZWQgdXNpbmcgcHJvY2VkdXJhbA0KLSAgICBhbGdvcml0aG1zLg0KLQ0KLUFzIHRoZSBvcmlnaW5hbCB2ZXJzaW9uIHdhcyBvcHRpbWl6ZWQgZm9yIHNpemUgaW5zdGVhZCBvZg0KLXBlcmZvcm1hbmNlLCB0aGF0IGhvbGRzIHRydWUgZm9yIHRoaXMgT3BlbkdMIEVTIHZlcnNpb24gYXMNCi13ZWxsLiBUaHVzIHRoZSBwZXJmb3JtYW5jZSBjb3VsZCBiZSBzaWduaWZpY2FudGx5IGluY3JlYXNlZCwNCi1mb3IgZXhhbXBsZSBieSBjaGFuZ2luZyB0aGUgY29kZSB0byB1c2UgZ2xEcmF3RWxlbWVudHMNCi1pbnN0ZWFkIG9mIGdsRHJhd0FycmF5cy4gVGhlIGNvZGUgdXNlcyBvbmx5IE9wZW5HTCBFUyAxLjANCi1Db21tb24gTGl0ZSAtbGV2ZWwgZnVuY3Rpb24gY2FsbHMgd2l0aG91dCBhbnkgZXh0ZW5zaW9ucy4NCi0NCi1UaGUgcmVmZXJlbmNlIE9wZW5HTCBFUyBpbXBsZW1lbnRhdGlvbnMgdXNlZCBmb3IgdGhpcyBhcHBsaWNhdGlvbjoNCi0gICogSHlicmlkJ3MgT3BlbkdMIEVTIEFQSSBJbXBsZW1lbnRhdGlvbiAoR2VyYmVyYSkgdmVyc2lvbiAyLjAuNA0KLSAgICBQcmVidWlsdCBXaW4zMiBQQyBleGVjdXRhYmxlOiBTYW5PR0xFUy1HZXJiZXJhLmV4ZQ0KLSAgKiBQb3dlclZSIE1CWCBTREssIE9wZW5HTCBFUyBXaW5kb3dzIFBDIEVtdWxhdGlvbiB2ZXJzaW9uIDEuMDQuMTQuMDE3MA0KLSAgICBQcmVidWlsdCBXaW4zMiBQQyBleGVjdXRhYmxlOiBTYW5PR0xFUy1QVlJTREsuZXhlDQotDQotTm90ZSB0aGF0IERJU0FCTEVfSU1QT1JUR0wgcHJlcHJvY2Vzc29yIG1hY3JvIGNhbiBiZSB1c2VkDQotdG8gc3BlY2lmeSBub3QgdG8gdXNlIGR5bmFtaWMgcnVudGltZSBiaW5kaW5nIG9mIHRoZSBsaWJyYXJ5Lg0KLVlvdSBhbHNvIG5lZWQgdG8gZGVmaW5lIHByZXByb2Nlc3NvciBtYWNybyBQVlJTREsgdG8gY29tcGlsZQ0KLXRoZSBzb3VyY2Ugd2l0aCBQb3dlclZSIE9wZW5HTCBFUyBTREsuDQotDQotVGhlIGRlbW8gYXBwbGljYXRpb24gaXMgYnJpZWZseSB0ZXN0ZWQgd2l0aCBhIGZldyBvdGhlciBPcGVuR0wgRVMNCi1pbXBsZW1lbnRhdGlvbnMgYXMgd2VsbCAoZS5nLiBWaW5jZW50LCBHTEVTb25HTCBvbiBMaW51eCwgRGVsbA0KLUF4aW0gWDUwdikuIE1vc3Qgb2YgdGhlc2Ugb3RoZXIgaW1wbGVtZW50YXRpb25zIHJlbmRlcmVkIHRoZSBkZW1vDQotZXJyb25lb3VzbHkgaW4gc29tZSBhc3BlY3QuIFRoaXMgbWF5IGluZGljYXRlIHRoYXQgdGhlIGRlbW8gc291cmNlDQotY291bGQgc3RpbGwgaGF2ZSBzb21lIHdvcmsgdG8gZG8gd2l0aCBjb21wYXRpYmlsaXR5IGFuZCBjb3JyZWN0DQotQVBJIHVzYWdlLCBhbHRob3VnaCB0aGUgbm9uLWNvbmZvcm1pbmcgaW1wbGVtZW50YXRpb25zIGFyZSBtb3N0DQotcHJvYmFibHkgdW5maW5pc2hlZCBhcyB3ZWxsLg0KLQ0KLVRoYW5rcyBhbmQgQWNrbm93bGVkZ2VtZW50czoNCi0NCi0qIFRvbmkgTPZubmJlcmcgKCFDdWJlKSBjcmVhdGVkIHRoZSBtdXNpYyBmb3Igb3JpZ2luYWwgdmVyc2lvbiwgd2hpY2gNCi0gIGlzIG5vdCBmZWF0dXJlZCBpbiB0aGlzIE9wZW5HTCBFUyBwb3J0Lg0KLSogU2FyYSBLYXBsaSAoc3QgUmFuYSkgZm9yIGFkZGl0aW9uYWwgY2FtZXJhIHdvcmsuDQotKiBQYXVsIEJvdXJrZSBmb3IgaW5mb3JtYXRpb24gYWJvdXQgdGhlIHN1cGVyc2hhcGVzLg0KLQ0KLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQ0KZGlmZiAtLWdpdCBhL29wZW5nbC90ZXN0cy9hbmdlbGVzL2FwcC1saW51eC5jIGIvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvYXBwLWxpbnV4LmMKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDdkMGQzMjAuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvYXBwLWxpbnV4LmMKKysrIC9kZXYvbnVsbApAQCAtMSwyMjMgKzAsMCBAQAotLyogU2FuIEFuZ2VsZXMgT2JzZXJ2YXRpb24gT3BlbkdMIEVTIHZlcnNpb24gZXhhbXBsZQotICogQ29weXJpZ2h0IDIwMDQtMjAwNSBKZXRybyBMYXVoYQotICogQWxsIHJpZ2h0cyByZXNlcnZlZC4KLSAqIFdlYjogaHR0cDovL2lraS5maS9qZXRyby8KLSAqCi0gKiBUaGlzIHNvdXJjZSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IKLSAqIG1vZGlmeSBpdCB1bmRlciB0aGUgdGVybXMgb2YgRUlUSEVSOgotICogICAoMSkgVGhlIEdOVSBMZXNzZXIgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkgdGhlIEZyZWUKLSAqICAgICAgIFNvZnR3YXJlIEZvdW5kYXRpb247IGVpdGhlciB2ZXJzaW9uIDIuMSBvZiB0aGUgTGljZW5zZSwgb3IgKGF0Ci0gKiAgICAgICB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uIFRoZSB0ZXh0IG9mIHRoZSBHTlUgTGVzc2VyCi0gKiAgICAgICBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGlzIGluY2x1ZGVkIHdpdGggdGhpcyBzb3VyY2UgaW4gdGhlCi0gKiAgICAgICBmaWxlIExJQ0VOU0UtTEdQTC50eHQuCi0gKiAgICgyKSBUaGUgQlNELXN0eWxlIGxpY2Vuc2UgdGhhdCBpcyBpbmNsdWRlZCB3aXRoIHRoaXMgc291cmNlIGluCi0gKiAgICAgICB0aGUgZmlsZSBMSUNFTlNFLUJTRC50eHQuCi0gKgotICogVGhpcyBzb3VyY2UgaXMgZGlzdHJpYnV0ZWQgaW4gdGhlIGhvcGUgdGhhdCBpdCB3aWxsIGJlIHVzZWZ1bCwKLSAqIGJ1dCBXSVRIT1VUIEFOWSBXQVJSQU5UWTsgd2l0aG91dCBldmVuIHRoZSBpbXBsaWVkIHdhcnJhbnR5IG9mCi0gKiBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuIFNlZSB0aGUgZmlsZXMKLSAqIExJQ0VOU0UtTEdQTC50eHQgYW5kIExJQ0VOU0UtQlNELnR4dCBmb3IgbW9yZSBkZXRhaWxzLgotICoKLSAqICRJZDogYXBwLWxpbnV4LmMsdiAxLjQgMjAwNS8wMi8wOCAxODo0Mjo0OCB0b25pYyBFeHAgJAotICogJFJldmlzaW9uOiAxLjQgJAotICoKLSAqIFBhcnRzIG9mIHRoaXMgc291cmNlIGZpbGUgaXMgYmFzZWQgb24gdGVzdC9leGFtcGxlIGNvZGUgZnJvbQotICogR0xFU29uR0wgaW1wbGVtZW50YXRpb24gYnkgRGF2aWQgQmx5dGhlLiBIZXJlIGlzIGNvcHkgb2YgdGhlCi0gKiBsaWNlbnNlIG5vdGljZSBmcm9tIHRoYXQgc291cmNlOgotICoKLSAqIENvcHlyaWdodCAoQykgMjAwMyAgRGF2aWQgQmx5dGhlICAgQWxsIFJpZ2h0cyBSZXNlcnZlZC4KLSAqCi0gKiBQZXJtaXNzaW9uIGlzIGhlcmVieSBncmFudGVkLCBmcmVlIG9mIGNoYXJnZSwgdG8gYW55IHBlcnNvbiBvYnRhaW5pbmcgYQotICogY29weSBvZiB0aGlzIHNvZnR3YXJlIGFuZCBhc3NvY2lhdGVkIGRvY3VtZW50YXRpb24gZmlsZXMgKHRoZSAiU29mdHdhcmUiKSwKLSAqIHRvIGRlYWwgaW4gdGhlIFNvZnR3YXJlIHdpdGhvdXQgcmVzdHJpY3Rpb24sIGluY2x1ZGluZyB3aXRob3V0IGxpbWl0YXRpb24KLSAqIHRoZSByaWdodHMgdG8gdXNlLCBjb3B5LCBtb2RpZnksIG1lcmdlLCBwdWJsaXNoLCBkaXN0cmlidXRlLCBzdWJsaWNlbnNlLAotICogYW5kL29yIHNlbGwgY29waWVzIG9mIHRoZSBTb2Z0d2FyZSwgYW5kIHRvIHBlcm1pdCBwZXJzb25zIHRvIHdob20gdGhlCi0gKiBTb2Z0d2FyZSBpcyBmdXJuaXNoZWQgdG8gZG8gc28sIHN1YmplY3QgdG8gdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOgotICoKLSAqIFRoZSBhYm92ZSBjb3B5cmlnaHQgbm90aWNlIGFuZCB0aGlzIHBlcm1pc3Npb24gbm90aWNlIHNoYWxsIGJlIGluY2x1ZGVkCi0gKiBpbiBhbGwgY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBTb2Z0d2FyZS4KLSAqCi0gKiBUSEUgU09GVFdBUkUgSVMgUFJPVklERUQgIkFTIElTIiwgV0lUSE9VVCBXQVJSQU5UWSBPRiBBTlkgS0lORCwgRVhQUkVTUwotICogT1IgSU1QTElFRCwgSU5DTFVESU5HIEJVVCBOT1QgTElNSVRFRCBUTyBUSEUgV0FSUkFOVElFUyBPRiBNRVJDSEFOVEFCSUxJVFksCi0gKiBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRSBBTkQgTk9OSU5GUklOR0VNRU5ULiAgSU4gTk8gRVZFTlQgU0hBTEwKLSAqIERBVklEIEJMWVRIRSBCRSBMSUFCTEUgRk9SIEFOWSBDTEFJTSwgREFNQUdFUyBPUiBPVEhFUiBMSUFCSUxJVFksIFdIRVRIRVIgSU4KLSAqIEFOIEFDVElPTiBPRiBDT05UUkFDVCwgVE9SVCBPUiBPVEhFUldJU0UsIEFSSVNJTkcgRlJPTSwgT1VUIE9GIE9SIElOCi0gKiBDT05ORUNUSU9OIFdJVEggVEhFIFNPRlRXQVJFIE9SIFRIRSBVU0UgT1IgT1RIRVIgREVBTElOR1MgSU4gVEhFIFNPRlRXQVJFLgotICovCi0KLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN5cy90aW1lLmg+Ci0KLSNpbmNsdWRlIDxFR0wvZWdsLmg+Ci0jaW5jbHVkZSA8R0xFUy9nbC5oPgotCi0jaW5jbHVkZSAiYXBwLmgiCi0KLQotaW50IGdBcHBBbGl2ZSA9IDE7Ci0KLXN0YXRpYyBjb25zdCBjaGFyIHNBcHBOYW1lW10gPQotICAgICJTYW4gQW5nZWxlcyBPYnNlcnZhdGlvbiBPcGVuR0wgRVMgdmVyc2lvbiBleGFtcGxlIChMaW51eCkiOwotCi1zdGF0aWMgaW50IHNXaW5kb3dXaWR0aCA9IFdJTkRPV19ERUZBVUxUX1dJRFRIOwotc3RhdGljIGludCBzV2luZG93SGVpZ2h0ID0gV0lORE9XX0RFRkFVTFRfSEVJR0hUOwotc3RhdGljIEVHTERpc3BsYXkgc0VnbERpc3BsYXkgPSBFR0xfTk9fRElTUExBWTsKLXN0YXRpYyBFR0xDb250ZXh0IHNFZ2xDb250ZXh0ID0gRUdMX05PX0NPTlRFWFQ7Ci1zdGF0aWMgRUdMU3VyZmFjZSBzRWdsU3VyZmFjZSA9IEVHTF9OT19TVVJGQUNFOwotCi1jb25zdCBjaGFyICplZ2xfc3RyZXJyb3IodW5zaWduZWQgZXJyKQotewotICAgIHN3aXRjaChlcnIpewotICAgIGNhc2UgRUdMX1NVQ0NFU1M6IHJldHVybiAiU1VDQ0VTUyI7Ci0gICAgY2FzZSBFR0xfTk9UX0lOSVRJQUxJWkVEOiByZXR1cm4gIk5PVCBJTklUSUFMSVpFRCI7Ci0gICAgY2FzZSBFR0xfQkFEX0FDQ0VTUzogcmV0dXJuICJCQUQgQUNDRVNTIjsKLSAgICBjYXNlIEVHTF9CQURfQUxMT0M6IHJldHVybiAiQkFEIEFMTE9DIjsKLSAgICBjYXNlIEVHTF9CQURfQVRUUklCVVRFOiByZXR1cm4gIkJBRF9BVFRSSUJVVEUiOwotICAgIGNhc2UgRUdMX0JBRF9DT05GSUc6IHJldHVybiAiQkFEIENPTkZJRyI7Ci0gICAgY2FzZSBFR0xfQkFEX0NPTlRFWFQ6IHJldHVybiAiQkFEIENPTlRFWFQiOwotICAgIGNhc2UgRUdMX0JBRF9DVVJSRU5UX1NVUkZBQ0U6IHJldHVybiAiQkFEIENVUlJFTlQgU1VSRkFDRSI7Ci0gICAgY2FzZSBFR0xfQkFEX0RJU1BMQVk6IHJldHVybiAiQkFEIERJU1BMQVkiOwotICAgIGNhc2UgRUdMX0JBRF9NQVRDSDogcmV0dXJuICJCQUQgTUFUQ0giOwotICAgIGNhc2UgRUdMX0JBRF9OQVRJVkVfUElYTUFQOiByZXR1cm4gIkJBRCBOQVRJVkUgUElYTUFQIjsKLSAgICBjYXNlIEVHTF9CQURfTkFUSVZFX1dJTkRPVzogcmV0dXJuICJCQUQgTkFUSVZFIFdJTkRPVyI7Ci0gICAgY2FzZSBFR0xfQkFEX1BBUkFNRVRFUjogcmV0dXJuICJCQUQgUEFSQU1FVEVSIjsKLSAgICBjYXNlIEVHTF9CQURfU1VSRkFDRTogcmV0dXJuICJCQURfU1VSRkFDRSI7Ci0vLyAgICBjYXNlIEVHTF9DT05URVhUX0xPU1Q6IHJldHVybiAiQ09OVEVYVCBMT1NUIjsKLSAgICBkZWZhdWx0OiByZXR1cm4gIlVOS05PV04iOwotICAgIH0KLX0KLQotdm9pZCBlZ2xfZXJyb3IoY29uc3QgY2hhciAqbmFtZSkKLXsKLSAgICB1bnNpZ25lZCBlcnIgPSBlZ2xHZXRFcnJvcigpOwotICAgIGlmKGVyciAhPSBFR0xfU1VDQ0VTUykgewotICAgICAgICBmcHJpbnRmKHN0ZGVyciwiJXMoKTogZWdsIGVycm9yIDB4JXggKCVzKVxuIiwgCi0gICAgICAgICAgICAgICAgbmFtZSwgZXJyLCBlZ2xfc3RyZXJyb3IoZXJyKSk7Ci0gICAgfQotfQotCi1zdGF0aWMgdm9pZCBjaGVja0dMRXJyb3JzKCkKLXsKLSAgICBHTGVudW0gZXJyb3IgPSBnbEdldEVycm9yKCk7Ci0gICAgaWYgKGVycm9yICE9IEdMX05PX0VSUk9SKQotICAgICAgICBmcHJpbnRmKHN0ZGVyciwgIkdMIEVycm9yOiAweCUwNHhcbiIsIChpbnQpZXJyb3IpOwotfQotCi0KLXN0YXRpYyB2b2lkIGNoZWNrRUdMRXJyb3JzKCkKLXsKLSAgICBFR0xpbnQgZXJyb3IgPSBlZ2xHZXRFcnJvcigpOwotICAgIC8vIEdMRVNvbkdMIHNlZW1zIHRvIGJlIHJldHVybmluZyAwIHdoZW4gdGhlcmUgaXMgbm8gZXJyb3JzPwotICAgIGlmIChlcnJvciAmJiBlcnJvciAhPSBFR0xfU1VDQ0VTUykKLSAgICAgICAgZnByaW50ZihzdGRlcnIsICJFR0wgRXJyb3I6IDB4JTA0eFxuIiwgKGludCllcnJvcik7Ci19Ci0KLXN0YXRpYyBpbnQgaW5pdEdyYXBoaWNzKCkKLXsKLSAgICBFR0xpbnQgc19jb25maWdBdHRyaWJzW10gPSB7Ci0gICAgICAgICBFR0xfUkVEX1NJWkUsICAgICAgIDUsCi0gICAgICAgICBFR0xfR1JFRU5fU0laRSwgICAgIDYsCi0gICAgICAgICBFR0xfQkxVRV9TSVpFLCAgICAgIDUsCi0gI2lmIDEKLSAgICAgICAgIEVHTF9ERVBUSF9TSVpFLCAgICAgMTYsCi0gICAgICAgICBFR0xfU1RFTkNJTF9TSVpFLCAgIDAsCi0gI2Vsc2UKLSAgICAgICAgIEVHTF9BTFBIQV9TSVpFLCAgICAgRUdMX0RPTlRfQ0FSRSwKLSAgICAgICAgIEVHTF9ERVBUSF9TSVpFLCAgICAgRUdMX0RPTlRfQ0FSRSwKLSAgICAgICAgIEVHTF9TVEVOQ0lMX1NJWkUsICAgRUdMX0RPTlRfQ0FSRSwKLSAgICAgICAgIEVHTF9TVVJGQUNFX1RZUEUsICAgRUdMX0RPTlRfQ0FSRSwKLSAjZW5kaWYKLSAgICAgICAgIEVHTF9OT05FCi0gICAgIH07Ci0gICAgIAotICAgICBFR0xpbnQgbnVtQ29uZmlncyA9IC0xOwotICAgICBFR0xpbnQgbWFqb3JWZXJzaW9uOwotICAgICBFR0xpbnQgbWlub3JWZXJzaW9uOwotICAgICBFR0xDb25maWcgY29uZmlnOwotICAgICBFR0xDb250ZXh0IGNvbnRleHQ7Ci0gICAgIEVHTFN1cmZhY2Ugc3VyZmFjZTsKLSAgICAgCi0gICAgIEVHTERpc3BsYXkgZHB5OwotCi0gICAgIGRweSA9IGVnbEdldERpc3BsYXkoRUdMX0RFRkFVTFRfRElTUExBWSk7Ci0gICAgIGVnbF9lcnJvcigiZWdsR2V0RGlzcGxheSIpOwotICAgICBmcHJpbnRmKHN0ZGVyciwiZHB5ID0gMHglMDh4XG4iLCAodW5zaWduZWQpIGRweSk7Ci0gICAgIAotICAgICBlZ2xJbml0aWFsaXplKGRweSwgJm1ham9yVmVyc2lvbiwgJm1pbm9yVmVyc2lvbik7Ci0gICAgIGVnbF9lcnJvcigiZWdsSW5pdGlhbGl6ZSIpOwotCi0gICAgIGVnbEdldENvbmZpZ3MoZHB5LCBOVUxMLCAwLCAmbnVtQ29uZmlncyk7Ci0gICAgIGVnbF9lcnJvcigiZWdsR2V0Q29uZmlncyIpOwotICAgICBmcHJpbnRmKHN0ZGVyciwibnVtIGNvbmZpZ3MgJWRcbiIsIG51bUNvbmZpZ3MpOwotICAgICAKLSAgICAgZWdsQ2hvb3NlQ29uZmlnKGRweSwgc19jb25maWdBdHRyaWJzLCAmY29uZmlnLCAxLCAmbnVtQ29uZmlncyk7Ci0gICAgIGVnbF9lcnJvcigiZWdsQ2hvb3NlQ29uZmlnIik7Ci0KLSAgICAgc3VyZmFjZSA9IGVnbENyZWF0ZVdpbmRvd1N1cmZhY2UoZHB5LCBjb25maWcsCi0gICAgICAgICAgICAgYW5kcm9pZF9jcmVhdGVEaXNwbGF5U3VyZmFjZSgpLCBOVUxMKTsKLSAgICAgZWdsX2Vycm9yKCJlZ2xNYXBXaW5kb3dTdXJmYWNlIik7Ci0KLSAgICAgZnByaW50ZihzdGRlcnIsInN1cmZhY2UgPSAlcFxuIiwgc3VyZmFjZSk7Ci0KLSAgICAgY29udGV4dCA9IGVnbENyZWF0ZUNvbnRleHQoZHB5LCBjb25maWcsIE5VTEwsIE5VTEwpOwotICAgICBlZ2xfZXJyb3IoImVnbENyZWF0ZUNvbnRleHQiKTsKLSAgICAgZnByaW50ZihzdGRlcnIsImNvbnRleHQgPSAlcFxuIiwgY29udGV4dCk7Ci0gICAgIAotICAgICBlZ2xNYWtlQ3VycmVudChkcHksIHN1cmZhY2UsIHN1cmZhY2UsIGNvbnRleHQpOyAgIAotICAgICBlZ2xfZXJyb3IoImVnbE1ha2VDdXJyZW50Iik7Ci0KLSAgICAgZWdsUXVlcnlTdXJmYWNlKGRweSwgc3VyZmFjZSwgRUdMX1dJRFRILCAmc1dpbmRvd1dpZHRoKTsKLSAgICAgZWdsUXVlcnlTdXJmYWNlKGRweSwgc3VyZmFjZSwgRUdMX0hFSUdIVCwgJnNXaW5kb3dIZWlnaHQpOwotCi0gICAgc0VnbERpc3BsYXkgPSBkcHk7Ci0gICAgc0VnbFN1cmZhY2UgPSBzdXJmYWNlOwotICAgIHNFZ2xDb250ZXh0ID0gY29udGV4dDsKLQotICAgIHJldHVybiBFR0xfVFJVRTsKLX0KLQotCi1zdGF0aWMgdm9pZCBkZWluaXRHcmFwaGljcygpCi17Ci0gICAgZWdsTWFrZUN1cnJlbnQoc0VnbERpc3BsYXksIE5VTEwsIE5VTEwsIE5VTEwpOwotICAgIGVnbERlc3Ryb3lDb250ZXh0KHNFZ2xEaXNwbGF5LCBzRWdsQ29udGV4dCk7Ci0gICAgZWdsRGVzdHJveVN1cmZhY2Uoc0VnbERpc3BsYXksIHNFZ2xTdXJmYWNlKTsKLSAgICBlZ2xUZXJtaW5hdGUoc0VnbERpc3BsYXkpOwotfQotCi0KLWludCBtYWluKGludCBhcmdjLCBjaGFyICphcmd2W10pCi17Ci0gICAgLy8gbm90IHJlZmVyZW5jZWQ6Ci0gICAgYXJnYyA9IGFyZ2M7Ci0gICAgYXJndiA9IGFyZ3Y7Ci0KLSAgICBpZiAoIWluaXRHcmFwaGljcygpKQotICAgIHsKLSAgICAgICAgZnByaW50ZihzdGRlcnIsICJHcmFwaGljcyBpbml0aWFsaXphdGlvbiBmYWlsZWQuXG4iKTsKLSAgICAgICAgcmV0dXJuIEVYSVRfRkFJTFVSRTsKLSAgICB9Ci0KLSAgICBhcHBJbml0KCk7Ci0gICAgCi0gICAgd2hpbGUgKGdBcHBBbGl2ZSkKLSAgICB7Ci0gICAgICAgIHN0cnVjdCB0aW1ldmFsIHRpbWVOb3c7Ci0KLSAgICAgICAgaWYgKGdBcHBBbGl2ZSkKLSAgICAgICAgewotICAgICAgICAgICAgZ2V0dGltZW9mZGF5KCZ0aW1lTm93LCBOVUxMKTsKLSAgICAgICAgICAgIGFwcFJlbmRlcih0aW1lTm93LnR2X3NlYyAqIDEwMDAgKyB0aW1lTm93LnR2X3VzZWMgLyAxMDAwLAotICAgICAgICAgICAgICAgICAgICAgIHNXaW5kb3dXaWR0aCwgc1dpbmRvd0hlaWdodCk7Ci0gICAgICAgICAgICBjaGVja0dMRXJyb3JzKCk7Ci0gICAgICAgICAgICBlZ2xTd2FwQnVmZmVycyhzRWdsRGlzcGxheSwgc0VnbFN1cmZhY2UpOwotICAgICAgICAgICAgY2hlY2tFR0xFcnJvcnMoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIGFwcERlaW5pdCgpOwotICAgIGRlaW5pdEdyYXBoaWNzKCk7Ci0KLSAgICByZXR1cm4gRVhJVF9TVUNDRVNTOwotfQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvYXBwLmggYi9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9hcHAuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNzBlYmQzNS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9hcHAuaAorKysgL2Rldi9udWxsCkBAIC0xLDU2ICswLDAgQEAKLS8qIFNhbiBBbmdlbGVzIE9ic2VydmF0aW9uIE9wZW5HTCBFUyB2ZXJzaW9uIGV4YW1wbGUKLSAqIENvcHlyaWdodCAyMDA0LTIwMDUgSmV0cm8gTGF1aGEKLSAqIEFsbCByaWdodHMgcmVzZXJ2ZWQuCi0gKiBXZWI6IGh0dHA6Ly9pa2kuZmkvamV0cm8vCi0gKgotICogVGhpcyBzb3VyY2UgaXMgZnJlZSBzb2Z0d2FyZTsgeW91IGNhbiByZWRpc3RyaWJ1dGUgaXQgYW5kL29yCi0gKiBtb2RpZnkgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIEVJVEhFUjoKLSAqICAgKDEpIFRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgYXMgcHVibGlzaGVkIGJ5IHRoZSBGcmVlCi0gKiAgICAgICBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIgdmVyc2lvbiAyLjEgb2YgdGhlIExpY2Vuc2UsIG9yIChhdAotICogICAgICAgeW91ciBvcHRpb24pIGFueSBsYXRlciB2ZXJzaW9uLiBUaGUgdGV4dCBvZiB0aGUgR05VIExlc3NlcgotICogICAgICAgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBpcyBpbmNsdWRlZCB3aXRoIHRoaXMgc291cmNlIGluIHRoZQotICogICAgICAgZmlsZSBMSUNFTlNFLUxHUEwudHh0LgotICogICAoMikgVGhlIEJTRC1zdHlsZSBsaWNlbnNlIHRoYXQgaXMgaW5jbHVkZWQgd2l0aCB0aGlzIHNvdXJjZSBpbgotICogICAgICAgdGhlIGZpbGUgTElDRU5TRS1CU0QudHh0LgotICoKLSAqIFRoaXMgc291cmNlIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsCi0gKiBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgotICogTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiBTZWUgdGhlIGZpbGVzCi0gKiBMSUNFTlNFLUxHUEwudHh0IGFuZCBMSUNFTlNFLUJTRC50eHQgZm9yIG1vcmUgZGV0YWlscy4KLSAqCi0gKiAkSWQ6IGFwcC5oLHYgMS4xNCAyMDA1LzAyLzA2IDIxOjEzOjU0IHRvbmljIEV4cCAkCi0gKiAkUmV2aXNpb246IDEuMTQgJAotICovCi0KLSNpZm5kZWYgQVBQX0hfSU5DTFVERUQKLSNkZWZpbmUgQVBQX0hfSU5DTFVERUQKLQotCi0jaWZkZWYgX19jcGx1c3BsdXMKLWV4dGVybiAiQyIgewotI2VuZGlmCi0KLQotI2RlZmluZSBXSU5ET1dfREVGQVVMVF9XSURUSCAgICA2NDAKLSNkZWZpbmUgV0lORE9XX0RFRkFVTFRfSEVJR0hUICAgNDgwCi0KLSNkZWZpbmUgV0lORE9XX0JQUCAgICAgICAgICAgICAgMTYKLQotCi0vLyBUaGUgc2ltcGxlIGZyYW1ld29yayBleHBlY3RzIHRoZSBhcHBsaWNhdGlvbiBjb2RlIHRvIGRlZmluZSB0aGVzZSBmdW5jdGlvbnMuCi1leHRlcm4gdm9pZCBhcHBJbml0KCk7Ci1leHRlcm4gdm9pZCBhcHBEZWluaXQoKTsKLWV4dGVybiB2b2lkIGFwcFJlbmRlcihsb25nIHRpY2ssIGludCB3aWR0aCwgaW50IGhlaWdodCk7Ci0KLS8qIFZhbHVlIGlzIG5vbi16ZXJvIHdoZW4gYXBwbGljYXRpb24gaXMgYWxpdmUsIGFuZCAwIHdoZW4gaXQgaXMgY2xvc2luZy4KLSAqIERlZmluZWQgYnkgdGhlIGFwcGxpY2F0aW9uIGZyYW1ld29yay4KLSAqLwotZXh0ZXJuIGludCBnQXBwQWxpdmU7Ci0KLQotI2lmZGVmIF9fY3BsdXNwbHVzCi19Ci0jZW5kaWYKLQotCi0jZW5kaWYgLy8gIUFQUF9IX0lOQ0xVREVECmRpZmYgLS1naXQgYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9jYW1zLmggYi9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9jYW1zLmgKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDJiMWFjYjMuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvY2Ftcy5oCisrKyAvZGV2L251bGwKQEAgLTEsNjUgKzAsMCBAQAotLyogU2FuIEFuZ2VsZXMgT2JzZXJ2YXRpb24gT3BlbkdMIEVTIHZlcnNpb24gZXhhbXBsZQotICogQ29weXJpZ2h0IDIwMDQtMjAwNSBKZXRybyBMYXVoYQotICogQWxsIHJpZ2h0cyByZXNlcnZlZC4KLSAqIFdlYjogaHR0cDovL2lraS5maS9qZXRyby8KLSAqCi0gKiBUaGlzIHNvdXJjZSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IKLSAqIG1vZGlmeSBpdCB1bmRlciB0aGUgdGVybXMgb2YgRUlUSEVSOgotICogICAoMSkgVGhlIEdOVSBMZXNzZXIgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkgdGhlIEZyZWUKLSAqICAgICAgIFNvZnR3YXJlIEZvdW5kYXRpb247IGVpdGhlciB2ZXJzaW9uIDIuMSBvZiB0aGUgTGljZW5zZSwgb3IgKGF0Ci0gKiAgICAgICB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uIFRoZSB0ZXh0IG9mIHRoZSBHTlUgTGVzc2VyCi0gKiAgICAgICBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGlzIGluY2x1ZGVkIHdpdGggdGhpcyBzb3VyY2UgaW4gdGhlCi0gKiAgICAgICBmaWxlIExJQ0VOU0UtTEdQTC50eHQuCi0gKiAgICgyKSBUaGUgQlNELXN0eWxlIGxpY2Vuc2UgdGhhdCBpcyBpbmNsdWRlZCB3aXRoIHRoaXMgc291cmNlIGluCi0gKiAgICAgICB0aGUgZmlsZSBMSUNFTlNFLUJTRC50eHQuCi0gKgotICogVGhpcyBzb3VyY2UgaXMgZGlzdHJpYnV0ZWQgaW4gdGhlIGhvcGUgdGhhdCBpdCB3aWxsIGJlIHVzZWZ1bCwKLSAqIGJ1dCBXSVRIT1VUIEFOWSBXQVJSQU5UWTsgd2l0aG91dCBldmVuIHRoZSBpbXBsaWVkIHdhcnJhbnR5IG9mCi0gKiBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuIFNlZSB0aGUgZmlsZXMKLSAqIExJQ0VOU0UtTEdQTC50eHQgYW5kIExJQ0VOU0UtQlNELnR4dCBmb3IgbW9yZSBkZXRhaWxzLgotICoKLSAqICRJZDogY2Ftcy5oLHYgMS43IDIwMDUvMDEvMzEgMjI6MTU6MTUgdG9uaWMgRXhwICQKLSAqICRSZXZpc2lvbjogMS43ICQKLSAqLwotCi0jaWZuZGVmIENBTVNfSF9JTkNMVURFRAotI2RlZmluZSBDQU1TX0hfSU5DTFVERUQKLQotCi0vKiBMZW5ndGggaW4gbWlsbGlzZWNvbmRzIG9mIG9uZSBjYW1lcmEgdHJhY2sgYmFzZSB1bml0LgotICogVGhlIHZhbHVlIG9yaWdpbmF0ZXMgZnJvbSB0aGUgbXVzaWMgc3luY2hyb25pemF0aW9uLgotICovCi0jZGVmaW5lIENBTVRSQUNLX0xFTiAgICA1NDQyCi0KLQotLy8gQ2FtZXJhIHRyYWNrIGRlZmluaXRpb24gZm9yIG9uZSBjYW1lcmEgdHJ1Y2tpbmcgc2hvdC4KLXR5cGVkZWYgc3RydWN0Ci17Ci0gICAgLyogRml2ZSBwYXJhbWV0ZXJzIG9mIHNyY1s1XSBhbmQgZGVzdFs1XToKLSAgICAgKiBleWVYLCBleWVZLCBleWVaLCB2aWV3QW5nbGUsIHZpZXdIZWlnaHRPZmZzCi0gICAgICovCi0gICAgc2hvcnQgc3JjWzVdLCBkZXN0WzVdOwotICAgIHVuc2lnbmVkIGNoYXIgZGlzdDsgICAgIC8vIGlmID4wLCBjYW0gcm90YXRlcyBhcm91bmQgZXllIHh5IG9uIGRpc3QgKiAwLjEKLSAgICB1bnNpZ25lZCBjaGFyIGxlbjsgICAgICAvLyBsZW5ndGggbXVsdGlwbGllcgotfSBDQU1UUkFDSzsKLQotc3RhdGljIENBTVRSQUNLIHNDYW1UcmFja3NbXSA9Ci17Ci0gICAgeyB7IDQ1MDAsIDI3MDAsIDEwMCwgNzAsIC0zMCB9LCB7IDUwLCA1MCwgLTkwLCAtMTAwLCAwIH0sIDIwLCAxIH0sCi0gICAgeyB7IC0xNDQ4LCA0Mjk0LCAyNSwgMzYzLCAwIH0sIHsgLTEzNiwgMjAyLCAxMjUsIC05OCwgMTAwIH0sIDAsIDEgfSwKLSAgICB7IHsgMTQzNywgNDkzMCwgMjAwLCAtMjc1LCAtMjAgfSwgeyAxNjg0LCAwLCAwLCA5LCAwIH0sIDAsIDEgfSwKLSAgICB7IHsgMTgwMCwgMzYwOSwgMjAwLCAwLCA2NzUgfSwgeyAwLCAwLCAwLCAzMDAsIDAgfSwgMCwgMSB9LAotICAgIHsgeyA5MjMsIDk5NiwgNTAsIDIzMzYsIC04MCB9LCB7IDAsIC0yMCwgLTUwLCAwLCAxNzAgfSwgMCwgMSB9LAotICAgIHsgeyAtMTY2MywgLTQzLCA2MDAsIDIxNzAsIDAgfSwgeyAyMCwgMCwgLTYwMCwgMCwgMTAwIH0sIDAsIDEgfSwKLSAgICB7IHsgMTA0OSwgLTE0MjAsIDE3NSwgMjExMSwgLTE3IH0sIHsgMCwgMCwgMCwgLTMzNCwgMCB9LCAwLCAyIH0sCi0gICAgeyB7IDAsIDAsIDUwLCAzMDAsIDI1IH0sIHsgMCwgMCwgMCwgMzAwLCAwIH0sIDcwLCAyIH0sCi0gICAgeyB7IC00NzMsIC05NTMsIDM1MDAsIC0zNTMsIC0zNTAgfSwgeyAwLCAwLCAtMjgwMCwgMCwgMCB9LCAwLCAyIH0sCi0gICAgeyB7IDE5MSwgMTkzOCwgMzUsIDExMzksIC0xNyB9LCB7IDEyMDUsIC0yOTA5LCAwLCAwLCAwIH0sIDAsIDIgfSwKLSAgICB7IHsgLTE0NDksIC0yNzAwLCAxNTAsIDAsIDAgfSwgeyAwLCAyMDAwLCAwLCAwLCAwIH0sIDAsIDIgfSwKLSAgICB7IHsgNTI3MywgNDk5MiwgNjUwLCAzNzMsIC01MCB9LCB7IC00NTk4LCAtMzA3MiwgMCwgMCwgMCB9LCAwLCAyIH0sCi0gICAgeyB7IDMyMjMsIC0zMjgyLCAxMDc1LCAtMzkzLCAtMjUgfSwgeyAxNjQ5LCAtMTY0OSwgMCwgMCwgMCB9LCAwLCAyIH0KLX07Ci0jZGVmaW5lIENBTVRSQUNLX0NPVU5UIChzaXplb2YoY2FtVHJhY2tzKSAvIHNpemVvZihjYW1UcmFja3NbMF0pKQotCi0KLSNlbmRpZiAvLyAhQ0FNU19IX0lOQ0xVREVECmRpZmYgLS1naXQgYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9kZW1vLmMgYi9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9kZW1vLmMKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDgwMmYzOTguLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvZGVtby5jCisrKyAvZGV2L251bGwKQEAgLTEsNzkyICswLDAgQEAKLS8qIFNhbiBBbmdlbGVzIE9ic2VydmF0aW9uIE9wZW5HTCBFUyB2ZXJzaW9uIGV4YW1wbGUKLSAqIENvcHlyaWdodCAyMDA0LTIwMDUgSmV0cm8gTGF1aGEKLSAqIEFsbCByaWdodHMgcmVzZXJ2ZWQuCi0gKiBXZWI6IGh0dHA6Ly9pa2kuZmkvamV0cm8vCi0gKgotICogVGhpcyBzb3VyY2UgaXMgZnJlZSBzb2Z0d2FyZTsgeW91IGNhbiByZWRpc3RyaWJ1dGUgaXQgYW5kL29yCi0gKiBtb2RpZnkgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIEVJVEhFUjoKLSAqICAgKDEpIFRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgYXMgcHVibGlzaGVkIGJ5IHRoZSBGcmVlCi0gKiAgICAgICBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIgdmVyc2lvbiAyLjEgb2YgdGhlIExpY2Vuc2UsIG9yIChhdAotICogICAgICAgeW91ciBvcHRpb24pIGFueSBsYXRlciB2ZXJzaW9uLiBUaGUgdGV4dCBvZiB0aGUgR05VIExlc3NlcgotICogICAgICAgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBpcyBpbmNsdWRlZCB3aXRoIHRoaXMgc291cmNlIGluIHRoZQotICogICAgICAgZmlsZSBMSUNFTlNFLUxHUEwudHh0LgotICogICAoMikgVGhlIEJTRC1zdHlsZSBsaWNlbnNlIHRoYXQgaXMgaW5jbHVkZWQgd2l0aCB0aGlzIHNvdXJjZSBpbgotICogICAgICAgdGhlIGZpbGUgTElDRU5TRS1CU0QudHh0LgotICoKLSAqIFRoaXMgc291cmNlIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsCi0gKiBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgotICogTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiBTZWUgdGhlIGZpbGVzCi0gKiBMSUNFTlNFLUxHUEwudHh0IGFuZCBMSUNFTlNFLUJTRC50eHQgZm9yIG1vcmUgZGV0YWlscy4KLSAqCi0gKiAkSWQ6IGRlbW8uYyx2IDEuMTAgMjAwNS8wMi8wOCAyMDo1NDozOSB0b25pYyBFeHAgJAotICogJFJldmlzaW9uOiAxLjEwICQKLSAqLwotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8bWF0aC5oPgotI2luY2x1ZGUgPGZsb2F0Lmg+Ci0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0KLSNpbmNsdWRlIDxHTEVTL2dsLmg+Ci0KLSNpbmNsdWRlICJhcHAuaCIKLSNpbmNsdWRlICJzaGFwZXMuaCIKLSNpbmNsdWRlICJjYW1zLmgiCi0KLQotLy8gVG90YWwgcnVuIGxlbmd0aCBpcyAyMCAqIGNhbWVyYSB0cmFjayBiYXNlIHVuaXQgbGVuZ3RoIChzZWUgY2Ftcy5oKS4KLSNkZWZpbmUgUlVOX0xFTkdUSCAgKDIwICogQ0FNVFJBQ0tfTEVOKQotI3VuZGVmIFBJCi0jZGVmaW5lIFBJIDMuMTQxNTkyNjUzNTg5NzkzMmYKLSNkZWZpbmUgUkFORE9NX1VJTlRfTUFYIDY1NTM1Ci0KLQotc3RhdGljIHVuc2lnbmVkIGxvbmcgc1JhbmRvbVNlZWQgPSAwOwotCi1zdGF0aWMgdm9pZCBzZWVkUmFuZG9tKHVuc2lnbmVkIGxvbmcgc2VlZCkKLXsKLSAgICBzUmFuZG9tU2VlZCA9IHNlZWQ7Ci19Ci0KLXN0YXRpYyB1bnNpZ25lZCBsb25nIHJhbmRvbVVJbnQoKQotewotICAgIHNSYW5kb21TZWVkID0gc1JhbmRvbVNlZWQgKiAweDM0M2ZkICsgMHgyNjllYzM7Ci0gICAgcmV0dXJuIHNSYW5kb21TZWVkID4+IDE2OwotfQotCi0KLS8vIENhcHBlZCBjb252ZXJzaW9uIGZyb20gZmxvYXQgdG8gZml4ZWQuCi1zdGF0aWMgbG9uZyBmbG9hdFRvRml4ZWQoZmxvYXQgdmFsdWUpCi17Ci0gICAgaWYgKHZhbHVlIDwgLTMyNzY4KSB2YWx1ZSA9IC0zMjc2ODsKLSAgICBpZiAodmFsdWUgPiAzMjc2NykgdmFsdWUgPSAzMjc2NzsKLSAgICByZXR1cm4gKGxvbmcpKHZhbHVlICogNjU1MzYpOwotfQotCi0jZGVmaW5lIEZJWEVEKHZhbHVlKSBmbG9hdFRvRml4ZWQodmFsdWUpCi0KLQotLy8gRGVmaW5pdGlvbiBvZiBvbmUgR0wgb2JqZWN0IGluIHRoaXMgZGVtby4KLXR5cGVkZWYgc3RydWN0IHsKLSAgICAvKiBWZXJ0ZXggYXJyYXkgYW5kIGNvbG9yIGFycmF5IGFyZSBlbmFibGVkIGZvciBhbGwgb2JqZWN0cywgc28gdGhlaXIKLSAgICAgKiBwb2ludGVycyBtdXN0IGFsd2F5cyBiZSB2YWxpZCBhbmQgbm9uLU5VTEwuIE5vcm1hbCBhcnJheSBpcyBub3QKLSAgICAgKiB1c2VkIGJ5IHRoZSBncm91bmQgcGxhbmUsIHNvIHdoZW4gaXRzIHBvaW50ZXIgaXMgTlVMTCB0aGVuIG5vcm1hbAotICAgICAqIGFycmF5IHVzYWdlIGlzIGRpc2FibGVkLgotICAgICAqCi0gICAgICogVmVydGV4IGFycmF5IGlzIHN1cHBvc2VkIHRvIHVzZSBHTF9GSVhFRCBkYXRhdHlwZSBhbmQgc3RyaWRlIDAKLSAgICAgKiAoaS5lLiB0aWdodGx5IHBhY2tlZCBhcnJheSkuIENvbG9yIGFycmF5IGlzIHN1cHBvc2VkIHRvIGhhdmUgNAotICAgICAqIGNvbXBvbmVudHMgcGVyIGNvbG9yIHdpdGggR0xfVU5TSUdORURfQllURSBkYXRhdHlwZSBhbmQgc3RyaWRlIDAuCi0gICAgICogTm9ybWFsIGFycmF5IGlzIHN1cHBvc2VkIHRvIHVzZSBHTF9GSVhFRCBkYXRhdHlwZSBhbmQgc3RyaWRlIDAuCi0gICAgICovCi0gICAgR0xmaXhlZCAqdmVydGV4QXJyYXk7Ci0gICAgR0x1Ynl0ZSAqY29sb3JBcnJheTsKLSAgICBHTGZpeGVkICpub3JtYWxBcnJheTsKLSAgICBHTGludCB2ZXJ0ZXhDb21wb25lbnRzOwotICAgIEdMc2l6ZWkgY291bnQ7Ci19IEdMT0JKRUNUOwotCi0KLXN0YXRpYyBsb25nIHNTdGFydFRpY2sgPSAwOwotc3RhdGljIGxvbmcgc1RpY2sgPSAwOwotCi1zdGF0aWMgaW50IHNDdXJyZW50Q2FtVHJhY2sgPSAwOwotc3RhdGljIGxvbmcgc0N1cnJlbnRDYW1UcmFja1N0YXJ0VGljayA9IDA7Ci1zdGF0aWMgbG9uZyBzTmV4dENhbVRyYWNrU3RhcnRUaWNrID0gMHg3ZmZmZmZmZjsKLQotc3RhdGljIEdMT0JKRUNUICpzU3VwZXJTaGFwZU9iamVjdHNbU1VQRVJTSEFQRV9DT1VOVF0gPSB7IE5VTEwgfTsKLXN0YXRpYyBHTE9CSkVDVCAqc0dyb3VuZFBsYW5lID0gTlVMTDsKLQotCi10eXBlZGVmIHN0cnVjdCB7Ci0gICAgZmxvYXQgeCwgeSwgejsKLX0gVkVDVE9SMzsKLQotCi1zdGF0aWMgdm9pZCBmcmVlR0xPYmplY3QoR0xPQkpFQ1QgKm9iamVjdCkKLXsKLSAgICBpZiAob2JqZWN0ID09IE5VTEwpCi0gICAgICAgIHJldHVybjsKLSAgICBmcmVlKG9iamVjdC0+bm9ybWFsQXJyYXkpOwotICAgIGZyZWUob2JqZWN0LT5jb2xvckFycmF5KTsKLSAgICBmcmVlKG9iamVjdC0+dmVydGV4QXJyYXkpOwotICAgIGZyZWUob2JqZWN0KTsKLX0KLQotCi1zdGF0aWMgR0xPQkpFQ1QgKiBuZXdHTE9iamVjdChsb25nIHZlcnRpY2VzLCBpbnQgdmVydGV4Q29tcG9uZW50cywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCB1c2VOb3JtYWxBcnJheSkKLXsKLSAgICBHTE9CSkVDVCAqcmVzdWx0OwotICAgIHJlc3VsdCA9IChHTE9CSkVDVCAqKW1hbGxvYyhzaXplb2YoR0xPQkpFQ1QpKTsKLSAgICBpZiAocmVzdWx0ID09IE5VTEwpCi0gICAgICAgIHJldHVybiBOVUxMOwotICAgIHJlc3VsdC0+Y291bnQgPSB2ZXJ0aWNlczsKLSAgICByZXN1bHQtPnZlcnRleENvbXBvbmVudHMgPSB2ZXJ0ZXhDb21wb25lbnRzOwotICAgIHJlc3VsdC0+dmVydGV4QXJyYXkgPSAoR0xmaXhlZCAqKW1hbGxvYyh2ZXJ0aWNlcyAqIHZlcnRleENvbXBvbmVudHMgKgotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YoR0xmaXhlZCkpOwotICAgIHJlc3VsdC0+Y29sb3JBcnJheSA9IChHTHVieXRlICopbWFsbG9jKHZlcnRpY2VzICogNCAqIHNpemVvZihHTHVieXRlKSk7Ci0gICAgaWYgKHVzZU5vcm1hbEFycmF5KQotICAgIHsKLSAgICAgICAgcmVzdWx0LT5ub3JtYWxBcnJheSA9IChHTGZpeGVkICopbWFsbG9jKHZlcnRpY2VzICogMyAqCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YoR0xmaXhlZCkpOwotICAgIH0KLSAgICBlbHNlCi0gICAgICAgIHJlc3VsdC0+bm9ybWFsQXJyYXkgPSBOVUxMOwotICAgIGlmIChyZXN1bHQtPnZlcnRleEFycmF5ID09IE5VTEwgfHwKLSAgICAgICAgcmVzdWx0LT5jb2xvckFycmF5ID09IE5VTEwgfHwKLSAgICAgICAgKHVzZU5vcm1hbEFycmF5ICYmIHJlc3VsdC0+bm9ybWFsQXJyYXkgPT0gTlVMTCkpCi0gICAgewotICAgICAgICBmcmVlR0xPYmplY3QocmVzdWx0KTsKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0gICAgfQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLQotc3RhdGljIHZvaWQgZHJhd0dMT2JqZWN0KEdMT0JKRUNUICpvYmplY3QpCi17Ci0gICAgYXNzZXJ0KG9iamVjdCAhPSBOVUxMKTsKLQotICAgIGdsVmVydGV4UG9pbnRlcihvYmplY3QtPnZlcnRleENvbXBvbmVudHMsIEdMX0ZJWEVELAotICAgICAgICAgICAgICAgICAgICAwLCBvYmplY3QtPnZlcnRleEFycmF5KTsKLSAgICBnbENvbG9yUG9pbnRlcig0LCBHTF9VTlNJR05FRF9CWVRFLCAwLCBvYmplY3QtPmNvbG9yQXJyYXkpOwotCi0gICAgLy8gQWxyZWFkeSBkb25lIGluIGluaXRpYWxpemF0aW9uOgotICAgIC8vZ2xFbmFibGVDbGllbnRTdGF0ZShHTF9WRVJURVhfQVJSQVkpOwotICAgIC8vZ2xFbmFibGVDbGllbnRTdGF0ZShHTF9DT0xPUl9BUlJBWSk7Ci0KLSAgICBpZiAob2JqZWN0LT5ub3JtYWxBcnJheSkKLSAgICB7Ci0gICAgICAgIGdsTm9ybWFsUG9pbnRlcihHTF9GSVhFRCwgMCwgb2JqZWN0LT5ub3JtYWxBcnJheSk7Ci0gICAgICAgIGdsRW5hYmxlQ2xpZW50U3RhdGUoR0xfTk9STUFMX0FSUkFZKTsKLSAgICB9Ci0gICAgZWxzZQotICAgICAgICBnbERpc2FibGVDbGllbnRTdGF0ZShHTF9OT1JNQUxfQVJSQVkpOwotICAgIGdsRHJhd0FycmF5cyhHTF9UUklBTkdMRVMsIDAsIG9iamVjdC0+Y291bnQpOwotfQotCi0KLXN0YXRpYyB2b2lkIHZlY3RvcjNTdWIoVkVDVE9SMyAqZGVzdCwgVkVDVE9SMyAqdjEsIFZFQ1RPUjMgKnYyKQotewotICAgIGRlc3QtPnggPSB2MS0+eCAtIHYyLT54OwotICAgIGRlc3QtPnkgPSB2MS0+eSAtIHYyLT55OwotICAgIGRlc3QtPnogPSB2MS0+eiAtIHYyLT56OwotfQotCi0KLXN0YXRpYyB2b2lkIHN1cGVyU2hhcGVNYXAoVkVDVE9SMyAqcG9pbnQsIGZsb2F0IHIxLCBmbG9hdCByMiwgZmxvYXQgdCwgZmxvYXQgcCkKLXsKLSAgICAvLyBzcGhlcmUtbWFwcGluZyBvZiBzdXBlcnNoYXBlIHBhcmFtZXRlcnMKLSAgICBwb2ludC0+eCA9IChmbG9hdCkoY29zKHQpICogY29zKHApIC8gcjEgLyByMik7Ci0gICAgcG9pbnQtPnkgPSAoZmxvYXQpKHNpbih0KSAqIGNvcyhwKSAvIHIxIC8gcjIpOwotICAgIHBvaW50LT56ID0gKGZsb2F0KShzaW4ocCkgLyByMik7Ci19Ci0KLQotc3RhdGljIGZsb2F0IHNzRnVuYyhjb25zdCBmbG9hdCB0LCBjb25zdCBmbG9hdCAqcCkKLXsKLSAgICByZXR1cm4gKGZsb2F0KShwb3cocG93KGZhYnMoY29zKHBbMF0gKiB0IC8gNCkpIC8gcFsxXSwgcFs0XSkgKwotICAgICAgICAgICAgICAgICAgICAgICBwb3coZmFicyhzaW4ocFswXSAqIHQgLyA0KSkgLyBwWzJdLCBwWzVdKSwgMSAvIHBbM10pKTsKLX0KLQotCi0vLyBDcmVhdGVzIGFuZCByZXR1cm5zIGEgc3VwZXJzaGFwZSBvYmplY3QuCi0vLyBCYXNlZCBvbiBQYXVsIEJvdXJrZSdzIFBPVi1SYXkgaW1wbGVtZW50YXRpb24uCi0vLyBodHRwOi8vYXN0cm9ub215LnN3aW4uZWR1LmF1L35wYm91cmtlL3BvdnJheS9zdXBlcnNoYXBlLwotc3RhdGljIEdMT0JKRUNUICogY3JlYXRlU3VwZXJTaGFwZShjb25zdCBmbG9hdCAqcGFyYW1zKQotewotICAgIGNvbnN0IGludCByZXNvbDEgPSAoaW50KXBhcmFtc1tTVVBFUlNIQVBFX1BBUkFNUyAtIDNdOwotICAgIGNvbnN0IGludCByZXNvbDIgPSAoaW50KXBhcmFtc1tTVVBFUlNIQVBFX1BBUkFNUyAtIDJdOwotICAgIC8vIGxhdGl0dWRlIDAgdG8gcGkvMiBmb3Igbm8gbWlycm9yZWQgYm90dG9tCi0gICAgLy8gKGxhdGl0dWRlQmVnaW49PTAgZm9yIC1waS8yIHRvIHBpLzIgb3JpZ2luYWxseSkKLSAgICBjb25zdCBpbnQgbGF0aXR1ZGVCZWdpbiA9IHJlc29sMiAvIDQ7Ci0gICAgY29uc3QgaW50IGxhdGl0dWRlRW5kID0gcmVzb2wyIC8gMjsgICAgLy8gbm9uLWluY2x1c2l2ZQotICAgIGNvbnN0IGludCBsb25naXR1ZGVDb3VudCA9IHJlc29sMTsKLSAgICBjb25zdCBpbnQgbGF0aXR1ZGVDb3VudCA9IGxhdGl0dWRlRW5kIC0gbGF0aXR1ZGVCZWdpbjsKLSAgICBjb25zdCBsb25nIHRyaWFuZ2xlQ291bnQgPSBsb25naXR1ZGVDb3VudCAqIGxhdGl0dWRlQ291bnQgKiAyOwotICAgIGNvbnN0IGxvbmcgdmVydGljZXMgPSB0cmlhbmdsZUNvdW50ICogMzsKLSAgICBHTE9CSkVDVCAqcmVzdWx0OwotICAgIGZsb2F0IGJhc2VDb2xvclszXTsKLSAgICBpbnQgYSwgbG9uZ2l0dWRlLCBsYXRpdHVkZTsKLSAgICBsb25nIGN1cnJlbnRWZXJ0ZXgsIGN1cnJlbnRRdWFkOwotCi0gICAgcmVzdWx0ID0gbmV3R0xPYmplY3QodmVydGljZXMsIDMsIDEpOwotICAgIGlmIChyZXN1bHQgPT0gTlVMTCkKLSAgICAgICAgcmV0dXJuIE5VTEw7Ci0KLSAgICBmb3IgKGEgPSAwOyBhIDwgMzsgKythKQotICAgICAgICBiYXNlQ29sb3JbYV0gPSAoKHJhbmRvbVVJbnQoKSAlIDE1NSkgKyAxMDApIC8gMjU1LmY7Ci0KLSAgICBjdXJyZW50UXVhZCA9IDA7Ci0gICAgY3VycmVudFZlcnRleCA9IDA7Ci0KLSAgICAvLyBsb25naXR1ZGUgLXBpIHRvIHBpCi0gICAgZm9yIChsb25naXR1ZGUgPSAwOyBsb25naXR1ZGUgPCBsb25naXR1ZGVDb3VudDsgKytsb25naXR1ZGUpCi0gICAgewotCi0gICAgICAgIC8vIGxhdGl0dWRlIDAgdG8gcGkvMgotICAgICAgICBmb3IgKGxhdGl0dWRlID0gbGF0aXR1ZGVCZWdpbjsgbGF0aXR1ZGUgPCBsYXRpdHVkZUVuZDsgKytsYXRpdHVkZSkKLSAgICAgICAgewotICAgICAgICAgICAgZmxvYXQgdDEgPSAtUEkgKyBsb25naXR1ZGUgKiAyICogUEkgLyByZXNvbDE7Ci0gICAgICAgICAgICBmbG9hdCB0MiA9IC1QSSArIChsb25naXR1ZGUgKyAxKSAqIDIgKiBQSSAvIHJlc29sMTsKLSAgICAgICAgICAgIGZsb2F0IHAxID0gLVBJIC8gMiArIGxhdGl0dWRlICogMiAqIFBJIC8gcmVzb2wyOwotICAgICAgICAgICAgZmxvYXQgcDIgPSAtUEkgLyAyICsgKGxhdGl0dWRlICsgMSkgKiAyICogUEkgLyByZXNvbDI7Ci0gICAgICAgICAgICBmbG9hdCByMCwgcjEsIHIyLCByMzsKLQotICAgICAgICAgICAgcjAgPSBzc0Z1bmModDEsIHBhcmFtcyk7Ci0gICAgICAgICAgICByMSA9IHNzRnVuYyhwMSwgJnBhcmFtc1s2XSk7Ci0gICAgICAgICAgICByMiA9IHNzRnVuYyh0MiwgcGFyYW1zKTsKLSAgICAgICAgICAgIHIzID0gc3NGdW5jKHAyLCAmcGFyYW1zWzZdKTsKLQotICAgICAgICAgICAgaWYgKHIwICE9IDAgJiYgcjEgIT0gMCAmJiByMiAhPSAwICYmIHIzICE9IDApCi0gICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgVkVDVE9SMyBwYSwgcGIsIHBjLCBwZDsKLSAgICAgICAgICAgICAgICBWRUNUT1IzIHYxLCB2MiwgbjsKLSAgICAgICAgICAgICAgICBmbG9hdCBjYTsKLSAgICAgICAgICAgICAgICBpbnQgaTsKLSAgICAgICAgICAgICAgICAvL2Zsb2F0IGxlblNxLCBpbnZMZW5TcTsKLQotICAgICAgICAgICAgICAgIHN1cGVyU2hhcGVNYXAoJnBhLCByMCwgcjEsIHQxLCBwMSk7Ci0gICAgICAgICAgICAgICAgc3VwZXJTaGFwZU1hcCgmcGIsIHIyLCByMSwgdDIsIHAxKTsKLSAgICAgICAgICAgICAgICBzdXBlclNoYXBlTWFwKCZwYywgcjIsIHIzLCB0MiwgcDIpOwotICAgICAgICAgICAgICAgIHN1cGVyU2hhcGVNYXAoJnBkLCByMCwgcjMsIHQxLCBwMik7Ci0KLSAgICAgICAgICAgICAgICAvLyBrbHVkZ2UgdG8gc2V0IGxvd2VyIGVkZ2Ugb2YgdGhlIG9iamVjdCB0byBmaXhlZCBsZXZlbAotICAgICAgICAgICAgICAgIGlmIChsYXRpdHVkZSA9PSBsYXRpdHVkZUJlZ2luICsgMSkKLSAgICAgICAgICAgICAgICAgICAgcGEueiA9IHBiLnogPSAwOwotCi0gICAgICAgICAgICAgICAgdmVjdG9yM1N1YigmdjEsICZwYiwgJnBhKTsKLSAgICAgICAgICAgICAgICB2ZWN0b3IzU3ViKCZ2MiwgJnBkLCAmcGEpOwotCi0gICAgICAgICAgICAgICAgLy8gQ2FsY3VsYXRlIG5vcm1hbCB3aXRoIGNyb3NzIHByb2R1Y3QuCi0gICAgICAgICAgICAgICAgLyogICBpICAgIGogICAgayAgICAgIGkgICAgagotICAgICAgICAgICAgICAgICAqIHYxLnggdjEueSB2MS56IHwgdjEueCB2MS55Ci0gICAgICAgICAgICAgICAgICogdjIueCB2Mi55IHYyLnogfCB2Mi54IHYyLnkKLSAgICAgICAgICAgICAgICAgKi8KLQotICAgICAgICAgICAgICAgIG4ueCA9IHYxLnkgKiB2Mi56IC0gdjEueiAqIHYyLnk7Ci0gICAgICAgICAgICAgICAgbi55ID0gdjEueiAqIHYyLnggLSB2MS54ICogdjIuejsKLSAgICAgICAgICAgICAgICBuLnogPSB2MS54ICogdjIueSAtIHYxLnkgKiB2Mi54OwotCi0gICAgICAgICAgICAgICAgLyogUHJlLW5vcm1hbGl6YXRpb24gb2YgdGhlIG5vcm1hbHMgaXMgZGlzYWJsZWQgaGVyZSBiZWNhdXNlCi0gICAgICAgICAgICAgICAgICogdGhleSB3aWxsIGJlIG5vcm1hbGl6ZWQgYW55d2F5IGxhdGVyIGR1ZSB0byBhdXRvbWF0aWMKLSAgICAgICAgICAgICAgICAgKiBub3JtYWxpemF0aW9uIChHTF9OT1JNQUxJWkUpLiBJdCBpcyBlbmFibGVkIGJlY2F1c2UgdGhlCi0gICAgICAgICAgICAgICAgICogb2JqZWN0cyBhcmUgc2NhbGVkIHdpdGggZ2xTY2FsZS4KLSAgICAgICAgICAgICAgICAgKi8KLSAgICAgICAgICAgICAgICAvKgotICAgICAgICAgICAgICAgIGxlblNxID0gbi54ICogbi54ICsgbi55ICogbi55ICsgbi56ICogbi56OwotICAgICAgICAgICAgICAgIGludkxlblNxID0gKGZsb2F0KSgxIC8gc3FydChsZW5TcSkpOwotICAgICAgICAgICAgICAgIG4ueCAqPSBpbnZMZW5TcTsKLSAgICAgICAgICAgICAgICBuLnkgKj0gaW52TGVuU3E7Ci0gICAgICAgICAgICAgICAgbi56ICo9IGludkxlblNxOwotICAgICAgICAgICAgICAgICovCi0KLSAgICAgICAgICAgICAgICBjYSA9IHBhLnogKyAwLjVmOwotCi0gICAgICAgICAgICAgICAgZm9yIChpID0gY3VycmVudFZlcnRleCAqIDM7Ci0gICAgICAgICAgICAgICAgICAgICBpIDwgKGN1cnJlbnRWZXJ0ZXggKyA2KSAqIDM7Ci0gICAgICAgICAgICAgICAgICAgICBpICs9IDMpCi0gICAgICAgICAgICAgICAgewotICAgICAgICAgICAgICAgICAgICByZXN1bHQtPm5vcm1hbEFycmF5W2ldID0gRklYRUQobi54KTsKLSAgICAgICAgICAgICAgICAgICAgcmVzdWx0LT5ub3JtYWxBcnJheVtpICsgMV0gPSBGSVhFRChuLnkpOwotICAgICAgICAgICAgICAgICAgICByZXN1bHQtPm5vcm1hbEFycmF5W2kgKyAyXSA9IEZJWEVEKG4ueik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgIGZvciAoaSA9IGN1cnJlbnRWZXJ0ZXggKiA0OwotICAgICAgICAgICAgICAgICAgICAgaSA8IChjdXJyZW50VmVydGV4ICsgNikgKiA0OwotICAgICAgICAgICAgICAgICAgICAgaSArPSA0KQotICAgICAgICAgICAgICAgIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IGEsIGNvbG9yWzNdOwotICAgICAgICAgICAgICAgICAgICBmb3IgKGEgPSAwOyBhIDwgMzsgKythKQotICAgICAgICAgICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBjb2xvclthXSA9IChpbnQpKGNhICogYmFzZUNvbG9yW2FdICogMjU1KTsKLSAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjb2xvclthXSA+IDI1NSkgY29sb3JbYV0gPSAyNTU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgcmVzdWx0LT5jb2xvckFycmF5W2ldID0gKEdMdWJ5dGUpY29sb3JbMF07Ci0gICAgICAgICAgICAgICAgICAgIHJlc3VsdC0+Y29sb3JBcnJheVtpICsgMV0gPSAoR0x1Ynl0ZSljb2xvclsxXTsKLSAgICAgICAgICAgICAgICAgICAgcmVzdWx0LT5jb2xvckFycmF5W2kgKyAyXSA9IChHTHVieXRlKWNvbG9yWzJdOwotICAgICAgICAgICAgICAgICAgICByZXN1bHQtPmNvbG9yQXJyYXlbaSArIDNdID0gMDsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmVzdWx0LT52ZXJ0ZXhBcnJheVtjdXJyZW50VmVydGV4ICogM10gPSBGSVhFRChwYS54KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMV0gPSBGSVhFRChwYS55KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMl0gPSBGSVhFRChwYS56KTsKLSAgICAgICAgICAgICAgICArK2N1cnJlbnRWZXJ0ZXg7Ci0gICAgICAgICAgICAgICAgcmVzdWx0LT52ZXJ0ZXhBcnJheVtjdXJyZW50VmVydGV4ICogM10gPSBGSVhFRChwYi54KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMV0gPSBGSVhFRChwYi55KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMl0gPSBGSVhFRChwYi56KTsKLSAgICAgICAgICAgICAgICArK2N1cnJlbnRWZXJ0ZXg7Ci0gICAgICAgICAgICAgICAgcmVzdWx0LT52ZXJ0ZXhBcnJheVtjdXJyZW50VmVydGV4ICogM10gPSBGSVhFRChwZC54KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMV0gPSBGSVhFRChwZC55KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMl0gPSBGSVhFRChwZC56KTsKLSAgICAgICAgICAgICAgICArK2N1cnJlbnRWZXJ0ZXg7Ci0gICAgICAgICAgICAgICAgcmVzdWx0LT52ZXJ0ZXhBcnJheVtjdXJyZW50VmVydGV4ICogM10gPSBGSVhFRChwYi54KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMV0gPSBGSVhFRChwYi55KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMl0gPSBGSVhFRChwYi56KTsKLSAgICAgICAgICAgICAgICArK2N1cnJlbnRWZXJ0ZXg7Ci0gICAgICAgICAgICAgICAgcmVzdWx0LT52ZXJ0ZXhBcnJheVtjdXJyZW50VmVydGV4ICogM10gPSBGSVhFRChwYy54KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMV0gPSBGSVhFRChwYy55KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMl0gPSBGSVhFRChwYy56KTsKLSAgICAgICAgICAgICAgICArK2N1cnJlbnRWZXJ0ZXg7Ci0gICAgICAgICAgICAgICAgcmVzdWx0LT52ZXJ0ZXhBcnJheVtjdXJyZW50VmVydGV4ICogM10gPSBGSVhFRChwZC54KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMV0gPSBGSVhFRChwZC55KTsKLSAgICAgICAgICAgICAgICByZXN1bHQtPnZlcnRleEFycmF5W2N1cnJlbnRWZXJ0ZXggKiAzICsgMl0gPSBGSVhFRChwZC56KTsKLSAgICAgICAgICAgICAgICArK2N1cnJlbnRWZXJ0ZXg7Ci0gICAgICAgICAgICB9IC8vIHIwICYmIHIxICYmIHIyICYmIHIzCi0gICAgICAgICAgICArK2N1cnJlbnRRdWFkOwotICAgICAgICB9IC8vIGxhdGl0dWRlCi0gICAgfSAvLyBsb25naXR1ZGUKLQotICAgIC8vIFNldCBudW1iZXIgb2YgdmVydGljZXMgaW4gb2JqZWN0IHRvIHRoZSBhY3R1YWwgYW1vdW50IGNyZWF0ZWQuCi0gICAgcmVzdWx0LT5jb3VudCA9IGN1cnJlbnRWZXJ0ZXg7Ci0KLSAgICByZXR1cm4gcmVzdWx0OwotfQotCi0KLXN0YXRpYyBHTE9CSkVDVCAqIGNyZWF0ZUdyb3VuZFBsYW5lKCkKLXsKLSAgICBjb25zdCBpbnQgc2NhbGUgPSA0OwotICAgIGNvbnN0IGludCB5QmVnaW4gPSAtMTUsIHlFbmQgPSAxNTsgICAgLy8gZW5kcyBhcmUgbm9uLWluY2x1c2l2ZQotICAgIGNvbnN0IGludCB4QmVnaW4gPSAtMTUsIHhFbmQgPSAxNTsKLSAgICBjb25zdCBsb25nIHRyaWFuZ2xlQ291bnQgPSAoeUVuZCAtIHlCZWdpbikgKiAoeEVuZCAtIHhCZWdpbikgKiAyOwotICAgIGNvbnN0IGxvbmcgdmVydGljZXMgPSB0cmlhbmdsZUNvdW50ICogMzsKLSAgICBHTE9CSkVDVCAqcmVzdWx0OwotICAgIGludCB4LCB5OwotICAgIGxvbmcgY3VycmVudFZlcnRleCwgY3VycmVudFF1YWQ7Ci0KLSAgICByZXN1bHQgPSBuZXdHTE9iamVjdCh2ZXJ0aWNlcywgMiwgMCk7Ci0gICAgaWYgKHJlc3VsdCA9PSBOVUxMKQotICAgICAgICByZXR1cm4gTlVMTDsKLQotICAgIGN1cnJlbnRRdWFkID0gMDsKLSAgICBjdXJyZW50VmVydGV4ID0gMDsKLQotICAgIGZvciAoeSA9IHlCZWdpbjsgeSA8IHlFbmQ7ICsreSkKLSAgICB7Ci0gICAgICAgIGZvciAoeCA9IHhCZWdpbjsgeCA8IHhFbmQ7ICsreCkKLSAgICAgICAgewotICAgICAgICAgICAgR0x1Ynl0ZSBjb2xvcjsKLSAgICAgICAgICAgIGludCBpLCBhOwotICAgICAgICAgICAgY29sb3IgPSAoR0x1Ynl0ZSkoKHJhbmRvbVVJbnQoKSAmIDB4NWYpICsgODEpOyAgLy8gMTAxIDExMTEKLSAgICAgICAgICAgIGZvciAoaSA9IGN1cnJlbnRWZXJ0ZXggKiA0OyBpIDwgKGN1cnJlbnRWZXJ0ZXggKyA2KSAqIDQ7IGkgKz0gNCkKLSAgICAgICAgICAgIHsKLSAgICAgICAgICAgICAgICByZXN1bHQtPmNvbG9yQXJyYXlbaV0gPSBjb2xvcjsKLSAgICAgICAgICAgICAgICByZXN1bHQtPmNvbG9yQXJyYXlbaSArIDFdID0gY29sb3I7Ci0gICAgICAgICAgICAgICAgcmVzdWx0LT5jb2xvckFycmF5W2kgKyAyXSA9IGNvbG9yOwotICAgICAgICAgICAgICAgIHJlc3VsdC0+Y29sb3JBcnJheVtpICsgM10gPSAwOwotICAgICAgICAgICAgfQotCi0gICAgICAgICAgICAvLyBBeGlzIGJpdHMgZm9yIHF1YWQgdHJpYW5nbGVzOgotICAgICAgICAgICAgLy8geDogMDExMTAwICgweDFjKSwgeTogMTEwMDAxICgweDMxKSAgKGNsb2Nrd2lzZSkKLSAgICAgICAgICAgIC8vIHg6IDAwMTExMCAoMHgwZSksIHk6IDEwMDAxMSAoMHgyMykgIChjb3VudGVyLWNsb2Nrd2lzZSkKLSAgICAgICAgICAgIGZvciAoYSA9IDA7IGEgPCA2OyArK2EpCi0gICAgICAgICAgICB7Ci0gICAgICAgICAgICAgICAgY29uc3QgaW50IHhtID0geCArICgoMHgxYyA+PiBhKSAmIDEpOwotICAgICAgICAgICAgICAgIGNvbnN0IGludCB5bSA9IHkgKyAoKDB4MzEgPj4gYSkgJiAxKTsKLSAgICAgICAgICAgICAgICBjb25zdCBmbG9hdCBtID0gKGZsb2F0KShjb3MoeG0gKiAyKSAqIHNpbih5bSAqIDQpICogMC43NWYpOwotICAgICAgICAgICAgICAgIHJlc3VsdC0+dmVydGV4QXJyYXlbY3VycmVudFZlcnRleCAqIDJdID0KLSAgICAgICAgICAgICAgICAgICAgRklYRUQoeG0gKiBzY2FsZSArIG0pOwotICAgICAgICAgICAgICAgIHJlc3VsdC0+dmVydGV4QXJyYXlbY3VycmVudFZlcnRleCAqIDIgKyAxXSA9Ci0gICAgICAgICAgICAgICAgICAgIEZJWEVEKHltICogc2NhbGUgKyBtKTsKLSAgICAgICAgICAgICAgICArK2N1cnJlbnRWZXJ0ZXg7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICArK2N1cnJlbnRRdWFkOwotICAgICAgICB9Ci0gICAgfQotICAgIHJldHVybiByZXN1bHQ7Ci19Ci0KLQotc3RhdGljIHZvaWQgZHJhd0dyb3VuZFBsYW5lKCkKLXsKLSAgICBnbERpc2FibGUoR0xfQ1VMTF9GQUNFKTsKLSAgICBnbERpc2FibGUoR0xfREVQVEhfVEVTVCk7Ci0gICAgZ2xFbmFibGUoR0xfQkxFTkQpOwotICAgIGdsQmxlbmRGdW5jKEdMX1pFUk8sIEdMX1NSQ19DT0xPUik7Ci0gICAgZ2xEaXNhYmxlKEdMX0xJR0hUSU5HKTsKLQotICAgIGRyYXdHTE9iamVjdChzR3JvdW5kUGxhbmUpOwotCi0gICAgZ2xFbmFibGUoR0xfTElHSFRJTkcpOwotICAgIGdsRGlzYWJsZShHTF9CTEVORCk7Ci0gICAgZ2xFbmFibGUoR0xfREVQVEhfVEVTVCk7Ci19Ci0KLQotc3RhdGljIHZvaWQgZHJhd0ZhZGVRdWFkKCkKLXsKLSAgICBzdGF0aWMgY29uc3QgR0xmaXhlZCBxdWFkVmVydGljZXNbXSA9IHsKLSAgICAgICAgLTB4MTAwMDAsIC0weDEwMDAwLAotICAgICAgICAgMHgxMDAwMCwgLTB4MTAwMDAsCi0gICAgICAgIC0weDEwMDAwLCAgMHgxMDAwMCwKLSAgICAgICAgIDB4MTAwMDAsIC0weDEwMDAwLAotICAgICAgICAgMHgxMDAwMCwgIDB4MTAwMDAsCi0gICAgICAgIC0weDEwMDAwLCAgMHgxMDAwMAotICAgIH07Ci0KLSAgICBjb25zdCBpbnQgYmVnaW5GYWRlID0gc1RpY2sgLSBzQ3VycmVudENhbVRyYWNrU3RhcnRUaWNrOwotICAgIGNvbnN0IGludCBlbmRGYWRlID0gc05leHRDYW1UcmFja1N0YXJ0VGljayAtIHNUaWNrOwotICAgIGNvbnN0IGludCBtaW5GYWRlID0gYmVnaW5GYWRlIDwgZW5kRmFkZSA/IGJlZ2luRmFkZSA6IGVuZEZhZGU7Ci0KLSAgICBpZiAobWluRmFkZSA8IDEwMjQpCi0gICAgewotICAgICAgICBjb25zdCBHTGZpeGVkIGZhZGVDb2xvciA9IG1pbkZhZGUgPDwgNjsKLSAgICAgICAgZ2xDb2xvcjR4KGZhZGVDb2xvciwgZmFkZUNvbG9yLCBmYWRlQ29sb3IsIDApOwotCi0gICAgICAgIGdsRGlzYWJsZShHTF9ERVBUSF9URVNUKTsKLSAgICAgICAgZ2xFbmFibGUoR0xfQkxFTkQpOwotICAgICAgICBnbEJsZW5kRnVuYyhHTF9aRVJPLCBHTF9TUkNfQ09MT1IpOwotICAgICAgICBnbERpc2FibGUoR0xfTElHSFRJTkcpOwotCi0gICAgICAgIGdsTWF0cml4TW9kZShHTF9NT0RFTFZJRVcpOwotICAgICAgICBnbExvYWRJZGVudGl0eSgpOwotCi0gICAgICAgIGdsTWF0cml4TW9kZShHTF9QUk9KRUNUSU9OKTsKLSAgICAgICAgZ2xMb2FkSWRlbnRpdHkoKTsKLQotICAgICAgICBnbERpc2FibGVDbGllbnRTdGF0ZShHTF9DT0xPUl9BUlJBWSk7Ci0gICAgICAgIGdsRGlzYWJsZUNsaWVudFN0YXRlKEdMX05PUk1BTF9BUlJBWSk7Ci0gICAgICAgIGdsVmVydGV4UG9pbnRlcigyLCBHTF9GSVhFRCwgMCwgcXVhZFZlcnRpY2VzKTsKLSAgICAgICAgZ2xEcmF3QXJyYXlzKEdMX1RSSUFOR0xFUywgMCwgNik7Ci0KLSAgICAgICAgZ2xFbmFibGVDbGllbnRTdGF0ZShHTF9DT0xPUl9BUlJBWSk7Ci0KLSAgICAgICAgZ2xNYXRyaXhNb2RlKEdMX01PREVMVklFVyk7Ci0KLSAgICAgICAgZ2xFbmFibGUoR0xfTElHSFRJTkcpOwotICAgICAgICBnbERpc2FibGUoR0xfQkxFTkQpOwotICAgICAgICBnbEVuYWJsZShHTF9ERVBUSF9URVNUKTsKLSAgICB9Ci19Ci0KLQotLy8gQ2FsbGVkIGZyb20gdGhlIGFwcCBmcmFtZXdvcmsuCi12b2lkIGFwcEluaXQoKQotewotICAgIGludCBhOwotCi0gICAgZ2xFbmFibGUoR0xfTk9STUFMSVpFKTsKLSAgICBnbEVuYWJsZShHTF9ERVBUSF9URVNUKTsKLSAgICBnbERpc2FibGUoR0xfQ1VMTF9GQUNFKTsKLSAgICBnbFNoYWRlTW9kZWwoR0xfRkxBVCk7Ci0KLSAgICBnbEVuYWJsZShHTF9MSUdIVElORyk7Ci0gICAgZ2xFbmFibGUoR0xfTElHSFQwKTsKLSAgICBnbEVuYWJsZShHTF9MSUdIVDEpOwotICAgIGdsRW5hYmxlKEdMX0xJR0hUMik7Ci0KLSAgICBnbEVuYWJsZUNsaWVudFN0YXRlKEdMX1ZFUlRFWF9BUlJBWSk7Ci0gICAgZ2xFbmFibGVDbGllbnRTdGF0ZShHTF9DT0xPUl9BUlJBWSk7Ci0KLSAgICBzZWVkUmFuZG9tKDE1KTsKLQotICAgIGZvciAoYSA9IDA7IGEgPCBTVVBFUlNIQVBFX0NPVU5UOyArK2EpCi0gICAgewotICAgICAgICBzU3VwZXJTaGFwZU9iamVjdHNbYV0gPSBjcmVhdGVTdXBlclNoYXBlKHNTdXBlclNoYXBlUGFyYW1zW2FdKTsKLSAgICAgICAgYXNzZXJ0KHNTdXBlclNoYXBlT2JqZWN0c1thXSAhPSBOVUxMKTsKLSAgICB9Ci0gICAgc0dyb3VuZFBsYW5lID0gY3JlYXRlR3JvdW5kUGxhbmUoKTsKLSAgICBhc3NlcnQoc0dyb3VuZFBsYW5lICE9IE5VTEwpOwotfQotCi0KLS8vIENhbGxlZCBmcm9tIHRoZSBhcHAgZnJhbWV3b3JrLgotdm9pZCBhcHBEZWluaXQoKQotewotICAgIGludCBhOwotICAgIGZvciAoYSA9IDA7IGEgPCBTVVBFUlNIQVBFX0NPVU5UOyArK2EpCi0gICAgICAgIGZyZWVHTE9iamVjdChzU3VwZXJTaGFwZU9iamVjdHNbYV0pOwotICAgIGZyZWVHTE9iamVjdChzR3JvdW5kUGxhbmUpOwotfQotCi0KLXN0YXRpYyB2b2lkIGdsdVBlcnNwZWN0aXZlKEdMZmxvYXQgZm92eSwgR0xmbG9hdCBhc3BlY3QsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICBHTGZsb2F0IHpOZWFyLCBHTGZsb2F0IHpGYXIpCi17Ci0gICAgR0xmbG9hdCB4bWluLCB4bWF4LCB5bWluLCB5bWF4OwotCi0gICAgeW1heCA9IHpOZWFyICogKEdMZmxvYXQpdGFuKGZvdnkgKiBQSSAvIDM2MCk7Ci0gICAgeW1pbiA9IC15bWF4OwotICAgIHhtaW4gPSB5bWluICogYXNwZWN0OwotICAgIHhtYXggPSB5bWF4ICogYXNwZWN0OwotCi0gICAgZ2xGcnVzdHVteCgoR0xmaXhlZCkoeG1pbiAqIDY1NTM2KSwgKEdMZml4ZWQpKHhtYXggKiA2NTUzNiksCi0gICAgICAgICAgICAgICAoR0xmaXhlZCkoeW1pbiAqIDY1NTM2KSwgKEdMZml4ZWQpKHltYXggKiA2NTUzNiksCi0gICAgICAgICAgICAgICAoR0xmaXhlZCkoek5lYXIgKiA2NTUzNiksIChHTGZpeGVkKSh6RmFyICogNjU1MzYpKTsKLX0KLQotCi1zdGF0aWMgdm9pZCBwcmVwYXJlRnJhbWUoaW50IHdpZHRoLCBpbnQgaGVpZ2h0KQotewotICAgIGdsVmlld3BvcnQoMCwgMCwgd2lkdGgsIGhlaWdodCk7Ci0KLSAgICBnbENsZWFyQ29sb3J4KChHTGZpeGVkKSgwLjFmICogNjU1MzYpLAotICAgICAgICAgICAgICAgICAgKEdMZml4ZWQpKDAuMmYgKiA2NTUzNiksCi0gICAgICAgICAgICAgICAgICAoR0xmaXhlZCkoMC4zZiAqIDY1NTM2KSwgMHgxMDAwMCk7Ci0gICAgZ2xDbGVhcihHTF9ERVBUSF9CVUZGRVJfQklUIHwgR0xfQ09MT1JfQlVGRkVSX0JJVCk7Ci0KLSAgICBnbE1hdHJpeE1vZGUoR0xfUFJPSkVDVElPTik7Ci0gICAgZ2xMb2FkSWRlbnRpdHkoKTsKLSAgICBnbHVQZXJzcGVjdGl2ZSg0NSwgKGZsb2F0KXdpZHRoIC8gaGVpZ2h0LCAwLjVmLCAxNTApOwotCi0gICAgZ2xNYXRyaXhNb2RlKEdMX01PREVMVklFVyk7Ci0KLSAgICBnbExvYWRJZGVudGl0eSgpOwotfQotCi0KLXN0YXRpYyB2b2lkIGNvbmZpZ3VyZUxpZ2h0QW5kTWF0ZXJpYWwoKQotewotICAgIHN0YXRpYyBHTGZpeGVkIGxpZ2h0MFBvc2l0aW9uW10gPSB7IC0weDQwMDAwLCAweDEwMDAwLCAweDEwMDAwLCAwIH07Ci0gICAgc3RhdGljIEdMZml4ZWQgbGlnaHQwRGlmZnVzZVtdID0geyAweDEwMDAwLCAweDY2NjYsIDAsIDB4MTAwMDAgfTsKLSAgICBzdGF0aWMgR0xmaXhlZCBsaWdodDFQb3NpdGlvbltdID0geyAweDEwMDAwLCAtMHgyMDAwMCwgLTB4MTAwMDAsIDAgfTsKLSAgICBzdGF0aWMgR0xmaXhlZCBsaWdodDFEaWZmdXNlW10gPSB7IDB4MTFlYiwgMHgyM2Q3LCAweDU5OTksIDB4MTAwMDAgfTsKLSAgICBzdGF0aWMgR0xmaXhlZCBsaWdodDJQb3NpdGlvbltdID0geyAtMHgxMDAwMCwgMCwgLTB4NDAwMDAsIDAgfTsKLSAgICBzdGF0aWMgR0xmaXhlZCBsaWdodDJEaWZmdXNlW10gPSB7IDB4MTFlYiwgMHgyYjg1LCAweDIzZDcsIDB4MTAwMDAgfTsKLSAgICBzdGF0aWMgR0xmaXhlZCBtYXRlcmlhbFNwZWN1bGFyW10gPSB7IDB4MTAwMDAsIDB4MTAwMDAsIDB4MTAwMDAsIDB4MTAwMDAgfTsKLQotICAgIGdsTGlnaHR4dihHTF9MSUdIVDAsIEdMX1BPU0lUSU9OLCBsaWdodDBQb3NpdGlvbik7Ci0gICAgZ2xMaWdodHh2KEdMX0xJR0hUMCwgR0xfRElGRlVTRSwgbGlnaHQwRGlmZnVzZSk7Ci0gICAgZ2xMaWdodHh2KEdMX0xJR0hUMSwgR0xfUE9TSVRJT04sIGxpZ2h0MVBvc2l0aW9uKTsKLSAgICBnbExpZ2h0eHYoR0xfTElHSFQxLCBHTF9ESUZGVVNFLCBsaWdodDFEaWZmdXNlKTsKLSAgICBnbExpZ2h0eHYoR0xfTElHSFQyLCBHTF9QT1NJVElPTiwgbGlnaHQyUG9zaXRpb24pOwotICAgIGdsTGlnaHR4dihHTF9MSUdIVDIsIEdMX0RJRkZVU0UsIGxpZ2h0MkRpZmZ1c2UpOwotICAgIGdsTWF0ZXJpYWx4dihHTF9GUk9OVF9BTkRfQkFDSywgR0xfU1BFQ1VMQVIsIG1hdGVyaWFsU3BlY3VsYXIpOwotCi0gICAgZ2xNYXRlcmlhbHgoR0xfRlJPTlRfQU5EX0JBQ0ssIEdMX1NISU5JTkVTUywgNjAgPDwgMTYpOwotICAgIGdsRW5hYmxlKEdMX0NPTE9SX01BVEVSSUFMKTsKLX0KLQotCi1zdGF0aWMgdm9pZCBkcmF3TW9kZWxzKGZsb2F0IHpTY2FsZSkKLXsKLSAgICBjb25zdCBpbnQgdHJhbnNsYXRpb25TY2FsZSA9IDk7Ci0gICAgaW50IHgsIHk7Ci0KLSAgICBzZWVkUmFuZG9tKDkpOwotCi0gICAgZ2xTY2FsZXgoMSA8PCAxNiwgMSA8PCAxNiwgKEdMZml4ZWQpKHpTY2FsZSAqIDY1NTM2KSk7Ci0KLSAgICBmb3IgKHkgPSAtNTsgeSA8PSA1OyArK3kpCi0gICAgewotICAgICAgICBmb3IgKHggPSAtNTsgeCA8PSA1OyArK3gpCi0gICAgICAgIHsKLSAgICAgICAgICAgIGZsb2F0IGJ1aWxkaW5nU2NhbGU7Ci0gICAgICAgICAgICBHTGZpeGVkIGZpeGVkU2NhbGU7Ci0KLSAgICAgICAgICAgIGludCBjdXJTaGFwZSA9IHJhbmRvbVVJbnQoKSAlIFNVUEVSU0hBUEVfQ09VTlQ7Ci0gICAgICAgICAgICBidWlsZGluZ1NjYWxlID0gc1N1cGVyU2hhcGVQYXJhbXNbY3VyU2hhcGVdW1NVUEVSU0hBUEVfUEFSQU1TIC0gMV07Ci0gICAgICAgICAgICBmaXhlZFNjYWxlID0gKEdMZml4ZWQpKGJ1aWxkaW5nU2NhbGUgKiA2NTUzNik7Ci0KLSAgICAgICAgICAgIGdsUHVzaE1hdHJpeCgpOwotICAgICAgICAgICAgZ2xUcmFuc2xhdGV4KCh4ICogdHJhbnNsYXRpb25TY2FsZSkgKiA2NTUzNiwKLSAgICAgICAgICAgICAgICAgICAgICAgICAoeSAqIHRyYW5zbGF0aW9uU2NhbGUpICogNjU1MzYsCi0gICAgICAgICAgICAgICAgICAgICAgICAgMCk7Ci0gICAgICAgICAgICBnbFJvdGF0ZXgoKEdMZml4ZWQpKChyYW5kb21VSW50KCkgJSAzNjApIDw8IDE2KSwgMCwgMCwgMSA8PCAxNik7Ci0gICAgICAgICAgICBnbFNjYWxleChmaXhlZFNjYWxlLCBmaXhlZFNjYWxlLCBmaXhlZFNjYWxlKTsKLQotICAgICAgICAgICAgZHJhd0dMT2JqZWN0KHNTdXBlclNoYXBlT2JqZWN0c1tjdXJTaGFwZV0pOwotICAgICAgICAgICAgZ2xQb3BNYXRyaXgoKTsKLSAgICAgICAgfQotICAgIH0KLQotICAgIGZvciAoeCA9IC0yOyB4IDw9IDI7ICsreCkKLSAgICB7Ci0gICAgICAgIGNvbnN0IGludCBzaGlwU2NhbGUxMDAgPSB0cmFuc2xhdGlvblNjYWxlICogNTAwOwotICAgICAgICBjb25zdCBpbnQgb2ZmczEwMCA9IHggKiBzaGlwU2NhbGUxMDAgKyAoc1RpY2sgJSBzaGlwU2NhbGUxMDApOwotICAgICAgICBmbG9hdCBvZmZzID0gb2ZmczEwMCAqIDAuMDFmOwotICAgICAgICBHTGZpeGVkIGZpeGVkT2ZmcyA9IChHTGZpeGVkKShvZmZzICogNjU1MzYpOwotICAgICAgICBnbFB1c2hNYXRyaXgoKTsKLSAgICAgICAgZ2xUcmFuc2xhdGV4KGZpeGVkT2ZmcywgLTQgKiA2NTUzNiwgMiA8PCAxNik7Ci0gICAgICAgIGRyYXdHTE9iamVjdChzU3VwZXJTaGFwZU9iamVjdHNbU1VQRVJTSEFQRV9DT1VOVCAtIDFdKTsKLSAgICAgICAgZ2xQb3BNYXRyaXgoKTsKLSAgICAgICAgZ2xQdXNoTWF0cml4KCk7Ci0gICAgICAgIGdsVHJhbnNsYXRleCgtNCAqIDY1NTM2LCBmaXhlZE9mZnMsIDQgPDwgMTYpOwotICAgICAgICBnbFJvdGF0ZXgoOTAgPDwgMTYsIDAsIDAsIDEgPDwgMTYpOwotICAgICAgICBkcmF3R0xPYmplY3Qoc1N1cGVyU2hhcGVPYmplY3RzW1NVUEVSU0hBUEVfQ09VTlQgLSAxXSk7Ci0gICAgICAgIGdsUG9wTWF0cml4KCk7Ci0gICAgfQotfQotCi0KLS8qIEZvbGxvd2luZyBnbHVMb29rQXQgaW1wbGVtZW50YXRpb24gaXMgYWRhcHRlZCBmcm9tIHRoZQotICogTWVzYSAzRCBHcmFwaGljcyBsaWJyYXJ5LiBodHRwOi8vd3d3Lm1lc2EzZC5vcmcKLSAqLwotc3RhdGljIHZvaWQgZ2x1TG9va0F0KEdMZmxvYXQgZXlleCwgR0xmbG9hdCBleWV5LCBHTGZsb2F0IGV5ZXosCi0JICAgICAgICAgICAgICBHTGZsb2F0IGNlbnRlcngsIEdMZmxvYXQgY2VudGVyeSwgR0xmbG9hdCBjZW50ZXJ6LAotCSAgICAgICAgICAgICAgR0xmbG9hdCB1cHgsIEdMZmxvYXQgdXB5LCBHTGZsb2F0IHVweikKLXsKLSAgICBHTGZsb2F0IG1bMTZdOwotICAgIEdMZmxvYXQgeFszXSwgeVszXSwgelszXTsKLSAgICBHTGZsb2F0IG1hZzsKLQotICAgIC8qIE1ha2Ugcm90YXRpb24gbWF0cml4ICovCi0KLSAgICAvKiBaIHZlY3RvciAqLwotICAgIHpbMF0gPSBleWV4IC0gY2VudGVyeDsKLSAgICB6WzFdID0gZXlleSAtIGNlbnRlcnk7Ci0gICAgelsyXSA9IGV5ZXogLSBjZW50ZXJ6OwotICAgIG1hZyA9IChmbG9hdClzcXJ0KHpbMF0gKiB6WzBdICsgelsxXSAqIHpbMV0gKyB6WzJdICogelsyXSk7Ci0gICAgaWYgKG1hZykgewkJCS8qIG1waWNobGVyLCAxOTk1MDUxNSAqLwotICAgICAgICB6WzBdIC89IG1hZzsKLSAgICAgICAgelsxXSAvPSBtYWc7Ci0gICAgICAgIHpbMl0gLz0gbWFnOwotICAgIH0KLQotICAgIC8qIFkgdmVjdG9yICovCi0gICAgeVswXSA9IHVweDsKLSAgICB5WzFdID0gdXB5OwotICAgIHlbMl0gPSB1cHo7Ci0KLSAgICAvKiBYIHZlY3RvciA9IFkgY3Jvc3MgWiAqLwotICAgIHhbMF0gPSB5WzFdICogelsyXSAtIHlbMl0gKiB6WzFdOwotICAgIHhbMV0gPSAteVswXSAqIHpbMl0gKyB5WzJdICogelswXTsKLSAgICB4WzJdID0geVswXSAqIHpbMV0gLSB5WzFdICogelswXTsKLQotICAgIC8qIFJlY29tcHV0ZSBZID0gWiBjcm9zcyBYICovCi0gICAgeVswXSA9IHpbMV0gKiB4WzJdIC0gelsyXSAqIHhbMV07Ci0gICAgeVsxXSA9IC16WzBdICogeFsyXSArIHpbMl0gKiB4WzBdOwotICAgIHlbMl0gPSB6WzBdICogeFsxXSAtIHpbMV0gKiB4WzBdOwotCi0gICAgLyogbXBpY2hsZXIsIDE5OTUwNTE1ICovCi0gICAgLyogY3Jvc3MgcHJvZHVjdCBnaXZlcyBhcmVhIG9mIHBhcmFsbGVsb2dyYW0sIHdoaWNoIGlzIDwgMS4wIGZvcgotICAgICAqIG5vbi1wZXJwZW5kaWN1bGFyIHVuaXQtbGVuZ3RoIHZlY3RvcnM7IHNvIG5vcm1hbGl6ZSB4LCB5IGhlcmUKLSAgICAgKi8KLQotICAgIG1hZyA9IChmbG9hdClzcXJ0KHhbMF0gKiB4WzBdICsgeFsxXSAqIHhbMV0gKyB4WzJdICogeFsyXSk7Ci0gICAgaWYgKG1hZykgewotICAgICAgICB4WzBdIC89IG1hZzsKLSAgICAgICAgeFsxXSAvPSBtYWc7Ci0gICAgICAgIHhbMl0gLz0gbWFnOwotICAgIH0KLQotICAgIG1hZyA9IChmbG9hdClzcXJ0KHlbMF0gKiB5WzBdICsgeVsxXSAqIHlbMV0gKyB5WzJdICogeVsyXSk7Ci0gICAgaWYgKG1hZykgewotICAgICAgICB5WzBdIC89IG1hZzsKLSAgICAgICAgeVsxXSAvPSBtYWc7Ci0gICAgICAgIHlbMl0gLz0gbWFnOwotICAgIH0KLQotI2RlZmluZSBNKHJvdyxjb2wpICBtW2NvbCo0K3Jvd10KLSAgICBNKDAsIDApID0geFswXTsKLSAgICBNKDAsIDEpID0geFsxXTsKLSAgICBNKDAsIDIpID0geFsyXTsKLSAgICBNKDAsIDMpID0gMC4wOwotICAgIE0oMSwgMCkgPSB5WzBdOwotICAgIE0oMSwgMSkgPSB5WzFdOwotICAgIE0oMSwgMikgPSB5WzJdOwotICAgIE0oMSwgMykgPSAwLjA7Ci0gICAgTSgyLCAwKSA9IHpbMF07Ci0gICAgTSgyLCAxKSA9IHpbMV07Ci0gICAgTSgyLCAyKSA9IHpbMl07Ci0gICAgTSgyLCAzKSA9IDAuMDsKLSAgICBNKDMsIDApID0gMC4wOwotICAgIE0oMywgMSkgPSAwLjA7Ci0gICAgTSgzLCAyKSA9IDAuMDsKLSAgICBNKDMsIDMpID0gMS4wOwotI3VuZGVmIE0KLSAgICB7Ci0gICAgICAgIGludCBhOwotICAgICAgICBHTGZpeGVkIGZpeGVkTVsxNl07Ci0gICAgICAgIGZvciAoYSA9IDA7IGEgPCAxNjsgKythKQotICAgICAgICAgICAgZml4ZWRNW2FdID0gKEdMZml4ZWQpKG1bYV0gKiA2NTUzNik7Ci0gICAgICAgIGdsTXVsdE1hdHJpeHgoZml4ZWRNKTsKLSAgICB9Ci0KLSAgICAvKiBUcmFuc2xhdGUgRXllIHRvIE9yaWdpbiAqLwotICAgIGdsVHJhbnNsYXRleCgoR0xmaXhlZCkoLWV5ZXggKiA2NTUzNiksCi0gICAgICAgICAgICAgICAgIChHTGZpeGVkKSgtZXlleSAqIDY1NTM2KSwKLSAgICAgICAgICAgICAgICAgKEdMZml4ZWQpKC1leWV6ICogNjU1MzYpKTsKLX0KLQotCi1zdGF0aWMgdm9pZCBjYW1UcmFjaygpCi17Ci0gICAgZmxvYXQgbGVycFs1XTsKLSAgICBmbG9hdCBlWCwgZVksIGVaLCBjWCwgY1ksIGNaOwotICAgIGZsb2F0IHRyYWNrUG9zOwotICAgIENBTVRSQUNLICpjYW07Ci0gICAgbG9uZyBjdXJyZW50Q2FtVGljazsKLSAgICBpbnQgYTsKLQotICAgIGlmIChzTmV4dENhbVRyYWNrU3RhcnRUaWNrIDw9IHNUaWNrKQotICAgIHsKLSAgICAgICAgKytzQ3VycmVudENhbVRyYWNrOwotICAgICAgICBzQ3VycmVudENhbVRyYWNrU3RhcnRUaWNrID0gc05leHRDYW1UcmFja1N0YXJ0VGljazsKLSAgICB9Ci0gICAgc05leHRDYW1UcmFja1N0YXJ0VGljayA9IHNDdXJyZW50Q2FtVHJhY2tTdGFydFRpY2sgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzQ2FtVHJhY2tzW3NDdXJyZW50Q2FtVHJhY2tdLmxlbiAqIENBTVRSQUNLX0xFTjsKLQotICAgIGNhbSA9ICZzQ2FtVHJhY2tzW3NDdXJyZW50Q2FtVHJhY2tdOwotICAgIGN1cnJlbnRDYW1UaWNrID0gc1RpY2sgLSBzQ3VycmVudENhbVRyYWNrU3RhcnRUaWNrOwotICAgIHRyYWNrUG9zID0gKGZsb2F0KWN1cnJlbnRDYW1UaWNrIC8gKENBTVRSQUNLX0xFTiAqIGNhbS0+bGVuKTsKLQotICAgIGZvciAoYSA9IDA7IGEgPCA1OyArK2EpCi0gICAgICAgIGxlcnBbYV0gPSAoY2FtLT5zcmNbYV0gKyBjYW0tPmRlc3RbYV0gKiB0cmFja1BvcykgKiAwLjAxZjsKLQotICAgIGlmIChjYW0tPmRpc3QpCi0gICAgewotICAgICAgICBmbG9hdCBkaXN0ID0gY2FtLT5kaXN0ICogMC4xZjsKLSAgICAgICAgY1ggPSBsZXJwWzBdOwotICAgICAgICBjWSA9IGxlcnBbMV07Ci0gICAgICAgIGNaID0gbGVycFsyXTsKLSAgICAgICAgZVggPSBjWCAtIChmbG9hdCljb3MobGVycFszXSkgKiBkaXN0OwotICAgICAgICBlWSA9IGNZIC0gKGZsb2F0KXNpbihsZXJwWzNdKSAqIGRpc3Q7Ci0gICAgICAgIGVaID0gY1ogLSBsZXJwWzRdOwotICAgIH0KLSAgICBlbHNlCi0gICAgewotICAgICAgICBlWCA9IGxlcnBbMF07Ci0gICAgICAgIGVZID0gbGVycFsxXTsKLSAgICAgICAgZVogPSBsZXJwWzJdOwotICAgICAgICBjWCA9IGVYICsgKGZsb2F0KWNvcyhsZXJwWzNdKTsKLSAgICAgICAgY1kgPSBlWSArIChmbG9hdClzaW4obGVycFszXSk7Ci0gICAgICAgIGNaID0gZVogKyBsZXJwWzRdOwotICAgIH0KLSAgICBnbHVMb29rQXQoZVgsIGVZLCBlWiwgY1gsIGNZLCBjWiwgMCwgMCwgMSk7Ci19Ci0KLQotLy8gQ2FsbGVkIGZyb20gdGhlIGFwcCBmcmFtZXdvcmsuCi0vKiBUaGUgdGljayBpcyBjdXJyZW50IHRpbWUgaW4gbWlsbGlzZWNvbmRzLCB3aWR0aCBhbmQgaGVpZ2h0Ci0gKiBhcmUgdGhlIGltYWdlIGRpbWVuc2lvbnMgdG8gYmUgcmVuZGVyZWQuCi0gKi8KLXZvaWQgYXBwUmVuZGVyKGxvbmcgdGljaywgaW50IHdpZHRoLCBpbnQgaGVpZ2h0KQotewotICAgIGlmIChzU3RhcnRUaWNrID09IDApCi0gICAgICAgIHNTdGFydFRpY2sgPSB0aWNrOwotICAgIGlmICghZ0FwcEFsaXZlKQotICAgICAgICByZXR1cm47Ci0KLSAgICAvLyBBY3R1YWwgdGljayB2YWx1ZSBpcyAiYmx1cnJlZCIgYSBsaXR0bGUgYml0LgotICAgIHNUaWNrID0gKHNUaWNrICsgdGljayAtIHNTdGFydFRpY2spID4+IDE7Ci0KLSAgICAvLyBUZXJtaW5hdGUgYXBwbGljYXRpb24gYWZ0ZXIgcnVubmluZyB0aHJvdWdoIHRoZSBkZW1vbnN0cmF0aW9uIG9uY2UuCi0gICAgaWYgKHNUaWNrID49IFJVTl9MRU5HVEgpCi0gICAgewotICAgICAgICBnQXBwQWxpdmUgPSAwOwotICAgICAgICByZXR1cm47Ci0gICAgfQotCi0gICAgLy8gUHJlcGFyZSBPcGVuR0wgRVMgZm9yIHJlbmRlcmluZyBvZiB0aGUgZnJhbWUuCi0gICAgcHJlcGFyZUZyYW1lKHdpZHRoLCBoZWlnaHQpOwotCi0gICAgLy8gVXBkYXRlIHRoZSBjYW1lcmEgcG9zaXRpb24gYW5kIHNldCB0aGUgbG9va2F0LgotICAgIGNhbVRyYWNrKCk7Ci0KLSAgICAvLyBDb25maWd1cmUgZW52aXJvbm1lbnQuCi0gICAgY29uZmlndXJlTGlnaHRBbmRNYXRlcmlhbCgpOwotCi0gICAgLy8gRHJhdyB0aGUgcmVmbGVjdGlvbiBieSBkcmF3aW5nIG1vZGVscyB3aXRoIG5lZ2F0ZWQgWi1heGlzLgotICAgIGdsUHVzaE1hdHJpeCgpOwotICAgIGRyYXdNb2RlbHMoLTEpOwotICAgIGdsUG9wTWF0cml4KCk7Ci0KLSAgICAvLyBCbGVuZCB0aGUgZ3JvdW5kIHBsYW5lIHRvIHRoZSB3aW5kb3cuCi0gICAgZHJhd0dyb3VuZFBsYW5lKCk7Ci0KLSAgICAvLyBEcmF3IGFsbCB0aGUgbW9kZWxzIG5vcm1hbGx5LgotICAgIGRyYXdNb2RlbHMoMSk7Ci0KLSAgICAvLyBEcmF3IGZhZGUgcXVhZCBvdmVyIHdob2xlIHdpbmRvdyAod2hlbiBjaGFuZ2luZyBjYW1lcmFzKS4KLSAgICBkcmF3RmFkZVF1YWQoKTsKLX0KZGlmZiAtLWdpdCBhL29wZW5nbC90ZXN0cy9hbmdlbGVzL2dwdXN0YXRlLmMgYi9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9ncHVzdGF0ZS5jCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzYzU0MGM5Li4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy9hbmdlbGVzL2dwdXN0YXRlLmMKKysrIC9kZXYvbnVsbApAQCAtMSwzOSArMCwwIEBACi0jaW5jbHVkZSA8ZXJybm8uaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN5cy9tbWFuLmg+Ci0jaW5jbHVkZSA8dW5pc3RkLmg+Ci0jaW5jbHVkZSA8ZmNudGwuaD4KLQotc3RhdGljIHZvaWQgKm1hcF9tZW1vcnkoY29uc3QgY2hhciAqZm4sIHVuc2lnbmVkIGJhc2UsIHVuc2lnbmVkIHNpemUpCi17Ci0gICAgaW50IGZkOwotICAgIHZvaWQgKnB0cjsKLSAgICAKLSAgICBmZCA9IG9wZW4oZm4sIE9fUkRXUiB8IE9fU1lOQyk7Ci0gICAgaWYoZmQgPCAwKSB7Ci0gICAgICAgIHBlcnJvcigiY2Fubm90IG9wZW4gJXMgZm9yIG1hcHBpbmciKTsKLSAgICAgICAgcmV0dXJuIE1BUF9GQUlMRUQ7Ci0gICAgfQotCi0gICAgcHRyID0gbW1hcCgwLCBzaXplLCBQUk9UX1JFQUQgfCBQUk9UX1dSSVRFLAotICAgICAgICAgICAgICAgTUFQX1NIQVJFRCwgZmQsIGJhc2UpOwotICAgIGNsb3NlKGZkKTsKLSAgICAKLSAgICBpZihwdHIgPT0gTUFQX0ZBSUxFRCkgewotICAgICAgICBmcHJpbnRmKHN0ZGVyciwiY2Fubm90IG1hcCAlcyAoQCUwOHgsJTA4eClcbiIsIGZuLCBiYXNlLCBzaXplKTsKLSAgICB9Ci0gICAgcmV0dXJuIHB0cjsgICAgCi19Ci0KLQotaW50IG1haW4oaW50IGFyZ2MsIGNoYXIqKiBhcmd2KQotewotICAgIHZvaWQgKmdycF9yZWdzID0gbWFwX21lbW9yeSgiL2Rldi9odzNkIiwgMCwgMTAyNCAqIDEwMjQpOwotICAgIHByaW50ZigiR1BVIGJhc2UgbWFwcGVkIGF0ICVwXG4iLCBncnBfcmVncyk7Ci0gICAgaW50IHN0YXRlX29mZnNldCA9IDB4MTAxNDA7Ci0gICAgcHJpbnRmKCJHUFUgc3RhdGUgPSAlMDhseFxuIiwKLSAgICAgICAgICAgICooKGxvbmcqKSgoY2hhciopZ3JwX3JlZ3MgKyBzdGF0ZV9vZmZzZXQpKSAgKTsKLQotICAgIHJldHVybiAwOwotfQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvaW5jbHVkZS9HTEVTL2VnbC5oIGIvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvaW5jbHVkZS9HTEVTL2VnbC5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjZGY4NDEwLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy9hbmdlbGVzL2luY2x1ZGUvR0xFUy9lZ2wuaAorKysgL2Rldi9udWxsCkBAIC0xLDIyOSArMCwwIEBACi0jaWZuZGVmIF9fZWdsX2hfCi0jZGVmaW5lIF9fZWdsX2hfCi0KLS8qCi0qKiBMaWNlbnNlIEFwcGxpY2FiaWxpdHkuIEV4Y2VwdCB0byB0aGUgZXh0ZW50IHBvcnRpb25zIG9mIHRoaXMgZmlsZSBhcmUKLSoqIG1hZGUgc3ViamVjdCB0byBhbiBhbHRlcm5hdGl2ZSBsaWNlbnNlIGFzIHBlcm1pdHRlZCBpbiB0aGUgU0dJIEZyZWUKLSoqIFNvZnR3YXJlIExpY2Vuc2UgQiwgVmVyc2lvbiAxLjAgKHRoZSAiTGljZW5zZSIpLCB0aGUgY29udGVudHMgb2YgdGhpcwotKiogZmlsZSBhcmUgc3ViamVjdCBvbmx5IHRvIHRoZSBwcm92aXNpb25zIG9mIHRoZSBMaWNlbnNlLiBZb3UgbWF5IG5vdCB1c2UKLSoqIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiBZb3UgbWF5IG9idGFpbiBhIGNvcHkKLSoqIG9mIHRoZSBMaWNlbnNlIGF0IFNpbGljb24gR3JhcGhpY3MsIEluYy4sIGF0dG46IExlZ2FsIFNlcnZpY2VzLCAxNjAwCi0qKiBBbXBoaXRoZWF0cmUgUGFya3dheSwgTW91bnRhaW4gVmlldywgQ0EgOTQwNDMtMTM1MSwgb3IgYXQ6Ci0qKgotKiogaHR0cDovL29zcy5zZ2kuY29tL3Byb2plY3RzL0ZyZWVCCi0qKgotKiogTm90ZSB0aGF0LCBhcyBwcm92aWRlZCBpbiB0aGUgTGljZW5zZSwgdGhlIFNvZnR3YXJlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuCi0qKiAiQVMgSVMiIGJhc2lzLCB3aXRoIEFMTCBFWFBSRVNTIEFORCBJTVBMSUVEIFdBUlJBTlRJRVMgQU5EIENPTkRJVElPTlMKLSoqIERJU0NMQUlNRUQsIElOQ0xVRElORywgV0lUSE9VVCBMSU1JVEFUSU9OLCBBTlkgSU1QTElFRCBXQVJSQU5USUVTIEFORAotKiogQ09ORElUSU9OUyBPRiBNRVJDSEFOVEFCSUxJVFksIFNBVElTRkFDVE9SWSBRVUFMSVRZLCBGSVRORVNTIEZPUiBBCi0qKiBQQVJUSUNVTEFSIFBVUlBPU0UsIEFORCBOT04tSU5GUklOR0VNRU5ULgotKioKLSoqIE9yaWdpbmFsIENvZGUuIFRoZSBPcmlnaW5hbCBDb2RlIGlzOiBPcGVuR0wgU2FtcGxlIEltcGxlbWVudGF0aW9uLAotKiogVmVyc2lvbiAxLjIuMSwgcmVsZWFzZWQgSmFudWFyeSAyNiwgMjAwMCwgZGV2ZWxvcGVkIGJ5IFNpbGljb24gR3JhcGhpY3MsCi0qKiBJbmMuIFRoZSBPcmlnaW5hbCBDb2RlIGlzIENvcHlyaWdodCAoYykgMTk5MS0yMDA0IFNpbGljb24gR3JhcGhpY3MsIEluYy4KLSoqIENvcHlyaWdodCBpbiBhbnkgcG9ydGlvbnMgY3JlYXRlZCBieSB0aGlyZCBwYXJ0aWVzIGlzIGFzIGluZGljYXRlZAotKiogZWxzZXdoZXJlIGhlcmVpbi4gQWxsIFJpZ2h0cyBSZXNlcnZlZC4KLSoqCi0qKiBBZGRpdGlvbmFsIE5vdGljZSBQcm92aXNpb25zOiBUaGUgYXBwbGljYXRpb24gcHJvZ3JhbW1pbmcgaW50ZXJmYWNlcwotKiogZXN0YWJsaXNoZWQgYnkgU0dJIGluIGNvbmp1bmN0aW9uIHdpdGggdGhlIE9yaWdpbmFsIENvZGUgYXJlIFRoZQotKiogT3BlbkdMKFIpIEdyYXBoaWNzIFN5c3RlbTogQSBTcGVjaWZpY2F0aW9uIChWZXJzaW9uIDEuMi4xKSwgcmVsZWFzZWQKLSoqIEFwcmlsIDEsIDE5OTk7IFRoZSBPcGVuR0woUikgR3JhcGhpY3MgU3lzdGVtIFV0aWxpdHkgTGlicmFyeSAoVmVyc2lvbgotKiogMS4zKSwgcmVsZWFzZWQgTm92ZW1iZXIgNCwgMTk5ODsgYW5kIE9wZW5HTChSKSBHcmFwaGljcyB3aXRoIHRoZSBYCi0qKiBXaW5kb3cgU3lzdGVtKFIpIChWZXJzaW9uIDEuMyksIHJlbGVhc2VkIE9jdG9iZXIgMTksIDE5OTguIFRoaXMgc29mdHdhcmUKLSoqIHdhcyBjcmVhdGVkIHVzaW5nIHRoZSBPcGVuR0woUikgdmVyc2lvbiAxLjIuMSBTYW1wbGUgSW1wbGVtZW50YXRpb24KLSoqIHB1Ymxpc2hlZCBieSBTR0ksIGJ1dCBoYXMgbm90IGJlZW4gaW5kZXBlbmRlbnRseSB2ZXJpZmllZCBhcyBiZWluZwotKiogY29tcGxpYW50IHdpdGggdGhlIE9wZW5HTChSKSB2ZXJzaW9uIDEuMi4xIFNwZWNpZmljYXRpb24uCi0qLwotCi0jaW5jbHVkZSA8R0xFUy9nbC5oPgotI2luY2x1ZGUgPEdMRVMvZWdsdHlwZXMuaD4KLQotLyoKLSoqIGVnbHR5cGVzLmggaXMgcGxhdGZvcm0gZGVwZW5kZW50LiBJdCBkZWZpbmVzOgotKioKLSoqICAgICAtIEVHTCB0eXBlcyBhbmQgcmVzb3VyY2VzCi0qKiAgICAgLSBOYXRpdmUgdHlwZXMKLSoqICAgICAtIEVHTCBhbmQgbmF0aXZlIGhhbmRsZSB2YWx1ZXMKLSoqCi0qKiBFR0wgdHlwZXMgYW5kIHJlc291cmNlcyBhcmUgdG8gYmUgdHlwZWRlZidlZCB3aXRoIGFwcHJvcHJpYXRlIHBsYXRmb3JtCi0qKiBkZXBlbmRlbnQgcmVzb3VyY2UgaGFuZGxlIHR5cGVzLiBFR0xpbnQgbXVzdCBiZSBhbiBpbnRlZ2VyIG9mIGF0IGxlYXN0Ci0qKiAzMi1iaXQuCi0qKgotKiogTmF0aXZlRGlzcGxheVR5cGUsIE5hdGl2ZVdpbmRvd1R5cGUgYW5kIE5hdGl2ZVBpeG1hcFR5cGUgYXJlIHRvIGJlCi0qKiByZXBsYWNlZCB3aXRoIGNvcnJlc3BvbmRpbmcgdHlwZXMgb2YgdGhlIG5hdGl2ZSB3aW5kb3cgc3lzdGVtIGluIGVnbC5oLgotKioKLSoqIEVHTCBhbmQgbmF0aXZlIGhhbmRsZSB2YWx1ZXMgbXVzdCBtYXRjaCB0aGVpciB0eXBlcy4KLSoqCi0qKiBFeGFtcGxlIGVnbHR5cGVzLmg6Ci0qLwotCi0jaWYgMAotCi0jaW5jbHVkZSA8c3lzL3R5cGVzLmg+Ci0jaW5jbHVkZSA8bmF0aXZlX3dpbmRvd19zeXN0ZW0uaD4KLQotLyoKLSoqIFR5cGVzIGFuZCByZXNvdXJjZXMKLSovCi10eXBlZGVmIGludCBFR0xCb29sZWFuOwotdHlwZWRlZiBpbnQzMl90IEVHTGludDsKLXR5cGVkZWYgdm9pZCAqRUdMRGlzcGxheTsKLXR5cGVkZWYgdm9pZCAqRUdMQ29uZmlnOwotdHlwZWRlZiB2b2lkICpFR0xTdXJmYWNlOwotdHlwZWRlZiB2b2lkICpFR0xDb250ZXh0OwotCi0vKgotKiogRUdMIGFuZCBuYXRpdmUgaGFuZGxlIHZhbHVlcwotKi8KLSNkZWZpbmUgRUdMX0RFRkFVTFRfRElTUExBWSAoKE5hdGl2ZURpc3BsYXlUeXBlKTApCi0jZGVmaW5lIEVHTF9OT19DT05URVhUICgoRUdMQ29udGV4dCkwKQotI2RlZmluZSBFR0xfTk9fRElTUExBWSAoKEVHTERpc3BsYXkpMCkKLSNkZWZpbmUgRUdMX05PX1NVUkZBQ0UgKChFR0xTdXJmYWNlKTApCi0KLSNlbmRpZgotCi0vKgotKiogVmVyc2lvbmluZyBhbmQgZXh0ZW5zaW9ucwotKi8KLSNkZWZpbmUgRUdMX1ZFUlNJT05fMV8wCQkgICAgICAgMQotCi0vKgotKiogQm9vbGVhbgotKi8KLSNkZWZpbmUgRUdMX0ZBTFNFCQkgICAgICAgMAotI2RlZmluZSBFR0xfVFJVRQkJICAgICAgIDEKLQotLyoKLSoqIEVycm9ycwotKi8KLSNkZWZpbmUgRUdMX1NVQ0NFU1MJCSAgICAgICAweDMwMDAKLSNkZWZpbmUgRUdMX05PVF9JTklUSUFMSVpFRAkgICAgICAgMHgzMDAxCi0jZGVmaW5lIEVHTF9CQURfQUNDRVNTCQkgICAgICAgMHgzMDAyCi0jZGVmaW5lIEVHTF9CQURfQUxMT0MJCSAgICAgICAweDMwMDMKLSNkZWZpbmUgRUdMX0JBRF9BVFRSSUJVVEUJICAgICAgIDB4MzAwNAotI2RlZmluZSBFR0xfQkFEX0NPTkZJRwkJICAgICAgIDB4MzAwNQotI2RlZmluZSBFR0xfQkFEX0NPTlRFWFQJCSAgICAgICAweDMwMDYKLSNkZWZpbmUgRUdMX0JBRF9DVVJSRU5UX1NVUkZBQ0UgICAgICAgIDB4MzAwNwotI2RlZmluZSBFR0xfQkFEX0RJU1BMQVkJCSAgICAgICAweDMwMDgKLSNkZWZpbmUgRUdMX0JBRF9NQVRDSAkJICAgICAgIDB4MzAwOQotI2RlZmluZSBFR0xfQkFEX05BVElWRV9QSVhNQVAJICAgICAgIDB4MzAwQQotI2RlZmluZSBFR0xfQkFEX05BVElWRV9XSU5ET1cJICAgICAgIDB4MzAwQgotI2RlZmluZSBFR0xfQkFEX1BBUkFNRVRFUgkgICAgICAgMHgzMDBDCi0jZGVmaW5lIEVHTF9CQURfU1VSRkFDRQkJICAgICAgIDB4MzAwRAotLyogMHgzMDBFIC0gMHgzMDFGIHJlc2VydmVkIGZvciBhZGRpdGlvbmFsIGVycm9ycy4gKi8KLQotLyoKLSoqIENvbmZpZyBhdHRyaWJ1dGVzCi0qLwotI2RlZmluZSBFR0xfQlVGRkVSX1NJWkUJCSAgICAgICAweDMwMjAKLSNkZWZpbmUgRUdMX0FMUEhBX1NJWkUJCSAgICAgICAweDMwMjEKLSNkZWZpbmUgRUdMX0JMVUVfU0laRQkJICAgICAgIDB4MzAyMgotI2RlZmluZSBFR0xfR1JFRU5fU0laRQkJICAgICAgIDB4MzAyMwotI2RlZmluZSBFR0xfUkVEX1NJWkUJCSAgICAgICAweDMwMjQKLSNkZWZpbmUgRUdMX0RFUFRIX1NJWkUJCSAgICAgICAweDMwMjUKLSNkZWZpbmUgRUdMX1NURU5DSUxfU0laRQkgICAgICAgMHgzMDI2Ci0jZGVmaW5lIEVHTF9DT05GSUdfQ0FWRUFUCSAgICAgICAweDMwMjcKLSNkZWZpbmUgRUdMX0NPTkZJR19JRAkJICAgICAgIDB4MzAyOAotI2RlZmluZSBFR0xfTEVWRUwJCSAgICAgICAweDMwMjkKLSNkZWZpbmUgRUdMX01BWF9QQlVGRkVSX0hFSUdIVAkgICAgICAgMHgzMDJBCi0jZGVmaW5lIEVHTF9NQVhfUEJVRkZFUl9QSVhFTFMJICAgICAgIDB4MzAyQgotI2RlZmluZSBFR0xfTUFYX1BCVUZGRVJfV0lEVEgJICAgICAgIDB4MzAyQwotI2RlZmluZSBFR0xfTkFUSVZFX1JFTkRFUkFCTEUJICAgICAgIDB4MzAyRAotI2RlZmluZSBFR0xfTkFUSVZFX1ZJU1VBTF9JRAkgICAgICAgMHgzMDJFCi0jZGVmaW5lIEVHTF9OQVRJVkVfVklTVUFMX1RZUEUJICAgICAgIDB4MzAyRgotLyojZGVmaW5lIEVHTF9QUkVTRVJWRURfUkVTT1VSQ0VTCSAweDMwMzAqLwotI2RlZmluZSBFR0xfU0FNUExFUwkJICAgICAgIDB4MzAzMQotI2RlZmluZSBFR0xfU0FNUExFX0JVRkZFUlMJICAgICAgIDB4MzAzMgotI2RlZmluZSBFR0xfU1VSRkFDRV9UWVBFCSAgICAgICAweDMwMzMKLSNkZWZpbmUgRUdMX1RSQU5TUEFSRU5UX1RZUEUJICAgICAgIDB4MzAzNAotI2RlZmluZSBFR0xfVFJBTlNQQVJFTlRfQkxVRV9WQUxVRSAgICAgMHgzMDM1Ci0jZGVmaW5lIEVHTF9UUkFOU1BBUkVOVF9HUkVFTl9WQUxVRSAgICAweDMwMzYKLSNkZWZpbmUgRUdMX1RSQU5TUEFSRU5UX1JFRF9WQUxVRSAgICAgIDB4MzAzNwotCi0vKgotKiogQ29uZmlnIGF0dHJpYnV0ZSBhbmQgdmFsdWUKLSovCi0jZGVmaW5lIEVHTF9OT05FCQkgICAgICAgMHgzMDM4Ci0vKiAweDMwMzkgLSAweDMwNEYgcmVzZXJ2ZWQgZm9yIGFkZGl0aW9uYWwgY29uZmlnIGF0dHJpYnV0ZXMuICovCi0KLS8qCi0qKiBDb25maWcgdmFsdWVzCi0qLwotI2RlZmluZSBFR0xfRE9OVF9DQVJFCQkgICAgICAgKChFR0xpbnQpIC0xKQotI2RlZmluZSBFR0xfUEJVRkZFUl9CSVQJCSAgICAgICAweDAxCi0jZGVmaW5lIEVHTF9QSVhNQVBfQklUCQkgICAgICAgMHgwMgotI2RlZmluZSBFR0xfV0lORE9XX0JJVAkJICAgICAgIDB4MDQKLSNkZWZpbmUgRUdMX1NMT1dfQ09ORklHCQkgICAgICAgMHgzMDUwCi0jZGVmaW5lIEVHTF9OT05fQ09ORk9STUFOVF9DT05GSUcgICAgICAweDMwNTEKLSNkZWZpbmUgRUdMX1RSQU5TUEFSRU5UX1JHQgkgICAgICAgMHgzMDUyCi0KLS8qCi0qKiBTdHJpbmcgbmFtZXMKLSovCi0jZGVmaW5lIEVHTF9WRU5ET1IJCSAgICAgICAweDMwNTMKLSNkZWZpbmUgRUdMX1ZFUlNJT04JCSAgICAgICAweDMwNTQKLSNkZWZpbmUgRUdMX0VYVEVOU0lPTlMJCSAgICAgICAweDMwNTUKLQotLyoKLSoqIFN1cmZhY2UgYXR0cmlidXRlcwotKi8KLSNkZWZpbmUgRUdMX0hFSUdIVAkJICAgICAgIDB4MzA1NgotI2RlZmluZSBFR0xfV0lEVEgJCSAgICAgICAweDMwNTcKLSNkZWZpbmUgRUdMX0xBUkdFU1RfUEJVRkZFUgkgICAgICAgMHgzMDU4Ci0KLS8qCi0qKiBDdXJyZW50IHN1cmZhY2VzCi0qLwotI2RlZmluZSBFR0xfRFJBVwkJICAgICAgIDB4MzA1OQotI2RlZmluZSBFR0xfUkVBRAkJICAgICAgIDB4MzA1QQotCi0vKgotKiogRW5naW5lcwotKi8KLSNkZWZpbmUgRUdMX0NPUkVfTkFUSVZFX0VOR0lORQkgICAgICAgMHgzMDVCCi0KLS8qIDB4MzA1Qy0weDNGRkZGIHJlc2VydmVkIGZvciBmdXR1cmUgdXNlICovCi0KLS8qCi0qKiBGdW5jdGlvbnMKLSovCi0jaWZkZWYgX19jcGx1c3BsdXMKLWV4dGVybiAiQyIgewotI2VuZGlmCi0KLUdMQVBJIEVHTGludCBBUElFTlRSWSBlZ2xHZXRFcnJvciAodm9pZCk7Ci0KLUdMQVBJIEVHTERpc3BsYXkgQVBJRU5UUlkgZWdsR2V0RGlzcGxheSAoTmF0aXZlRGlzcGxheVR5cGUgZGlzcGxheSk7Ci1HTEFQSSBFR0xCb29sZWFuIEFQSUVOVFJZIGVnbEluaXRpYWxpemUgKEVHTERpc3BsYXkgZHB5LCBFR0xpbnQgKm1ham9yLCBFR0xpbnQgKm1pbm9yKTsKLUdMQVBJIEVHTEJvb2xlYW4gQVBJRU5UUlkgZWdsVGVybWluYXRlIChFR0xEaXNwbGF5IGRweSk7Ci1HTEFQSSBjb25zdCBjaGFyICogQVBJRU5UUlkgZWdsUXVlcnlTdHJpbmcgKEVHTERpc3BsYXkgZHB5LCBFR0xpbnQgbmFtZSk7Ci1HTEFQSSB2b2lkICgqIEFQSUVOVFJZIGVnbEdldFByb2NBZGRyZXNzIChjb25zdCBjaGFyICpwcm9jbmFtZSkpKCk7Ci0KLUdMQVBJIEVHTEJvb2xlYW4gQVBJRU5UUlkgZWdsR2V0Q29uZmlncyAoRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyAqY29uZmlncywgRUdMaW50IGNvbmZpZ19zaXplLCBFR0xpbnQgKm51bV9jb25maWcpOwotR0xBUEkgRUdMQm9vbGVhbiBBUElFTlRSWSBlZ2xDaG9vc2VDb25maWcgKEVHTERpc3BsYXkgZHB5LCBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0LCBFR0xDb25maWcgKmNvbmZpZ3MsIEVHTGludCBjb25maWdfc2l6ZSwgRUdMaW50ICpudW1fY29uZmlnKTsKLUdMQVBJIEVHTEJvb2xlYW4gQVBJRU5UUlkgZWdsR2V0Q29uZmlnQXR0cmliIChFR0xEaXNwbGF5IGRweSwgRUdMQ29uZmlnIGNvbmZpZywgRUdMaW50IGF0dHJpYnV0ZSwgRUdMaW50ICp2YWx1ZSk7Ci0KLUdMQVBJIEVHTFN1cmZhY2UgQVBJRU5UUlkgZWdsQ3JlYXRlV2luZG93U3VyZmFjZSAoRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsIE5hdGl2ZVdpbmRvd1R5cGUgd2luZG93LCBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KTsKLUdMQVBJIEVHTFN1cmZhY2UgQVBJRU5UUlkgZWdsQ3JlYXRlUGl4bWFwU3VyZmFjZSAoRUdMRGlzcGxheSBkcHksIEVHTENvbmZpZyBjb25maWcsIE5hdGl2ZVBpeG1hcFR5cGUgcGl4bWFwLCBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KTsKLUdMQVBJIEVHTFN1cmZhY2UgQVBJRU5UUlkgZWdsQ3JlYXRlUGJ1ZmZlclN1cmZhY2UgKEVHTERpc3BsYXkgZHB5LCBFR0xDb25maWcgY29uZmlnLCBjb25zdCBFR0xpbnQgKmF0dHJpYl9saXN0KTsKLUdMQVBJIEVHTEJvb2xlYW4gQVBJRU5UUlkgZWdsRGVzdHJveVN1cmZhY2UgKEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIHN1cmZhY2UpOwotR0xBUEkgRUdMQm9vbGVhbiBBUElFTlRSWSBlZ2xRdWVyeVN1cmZhY2UgKEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIHN1cmZhY2UsIEVHTGludCBhdHRyaWJ1dGUsIEVHTGludCAqdmFsdWUpOwotCi1HTEFQSSBFR0xDb250ZXh0IEFQSUVOVFJZIGVnbENyZWF0ZUNvbnRleHQgKEVHTERpc3BsYXkgZHB5LCBFR0xDb25maWcgY29uZmlnLCBFR0xDb250ZXh0IHNoYXJlX2xpc3QsIGNvbnN0IEVHTGludCAqYXR0cmliX2xpc3QpOwotR0xBUEkgRUdMQm9vbGVhbiBBUElFTlRSWSBlZ2xEZXN0cm95Q29udGV4dCAoRUdMRGlzcGxheSBkcHksIEVHTENvbnRleHQgY3R4KTsKLUdMQVBJIEVHTEJvb2xlYW4gQVBJRU5UUlkgZWdsTWFrZUN1cnJlbnQgKEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIGRyYXcsIEVHTFN1cmZhY2UgcmVhZCwgRUdMQ29udGV4dCBjdHgpOwotR0xBUEkgRUdMQ29udGV4dCBBUElFTlRSWSBlZ2xHZXRDdXJyZW50Q29udGV4dCAodm9pZCk7Ci1HTEFQSSBFR0xTdXJmYWNlIEFQSUVOVFJZIGVnbEdldEN1cnJlbnRTdXJmYWNlIChFR0xpbnQgcmVhZGRyYXcpOwotR0xBUEkgRUdMRGlzcGxheSBBUElFTlRSWSBlZ2xHZXRDdXJyZW50RGlzcGxheSAodm9pZCk7Ci1HTEFQSSBFR0xCb29sZWFuIEFQSUVOVFJZIGVnbFF1ZXJ5Q29udGV4dCAoRUdMRGlzcGxheSBkcHksIEVHTENvbnRleHQgY3R4LCBFR0xpbnQgYXR0cmlidXRlLCBFR0xpbnQgKnZhbHVlKTsKLQotR0xBUEkgRUdMQm9vbGVhbiBBUElFTlRSWSBlZ2xXYWl0R0wgKHZvaWQpOwotR0xBUEkgRUdMQm9vbGVhbiBBUElFTlRSWSBlZ2xXYWl0TmF0aXZlIChFR0xpbnQgZW5naW5lKTsKLUdMQVBJIEVHTEJvb2xlYW4gQVBJRU5UUlkgZWdsU3dhcEJ1ZmZlcnMgKEVHTERpc3BsYXkgZHB5LCBFR0xTdXJmYWNlIGRyYXcpOwotR0xBUEkgRUdMQm9vbGVhbiBBUElFTlRSWSBlZ2xDb3B5QnVmZmVycyAoRUdMRGlzcGxheSBkcHksIEVHTFN1cmZhY2Ugc3VyZmFjZSwgTmF0aXZlUGl4bWFwVHlwZSB0YXJnZXQpOwotCi0jaWZkZWYgX19jcGx1c3BsdXMKLX0KLSNlbmRpZgotCi0jZW5kaWYgLyogX19fZWdsX2hfICovCmRpZmYgLS1naXQgYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9pbmNsdWRlL0dMRVMvZWdsdHlwZXMuaCBiL29wZW5nbC90ZXN0cy9hbmdlbGVzL2luY2x1ZGUvR0xFUy9lZ2x0eXBlcy5oCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA5ZGIzNmM5Li4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy9hbmdlbGVzL2luY2x1ZGUvR0xFUy9lZ2x0eXBlcy5oCisrKyAvZGV2L251bGwKQEAgLTEsMjAgKzAsMCBAQAotLyoKLSoqIFR5cGVzIGFuZCByZXNvdXJjZXMKLSovCi10eXBlZGVmIGludCBFR0xCb29sZWFuOwotdHlwZWRlZiBsb25nIEVHTGludDsKLXR5cGVkZWYgdm9pZCAqRUdMRGlzcGxheTsKLXR5cGVkZWYgdm9pZCAqRUdMQ29uZmlnOwotdHlwZWRlZiB2b2lkICpFR0xTdXJmYWNlOwotdHlwZWRlZiB2b2lkICpFR0xDb250ZXh0OwotdHlwZWRlZiB2b2lkICpOYXRpdmVEaXNwbGF5VHlwZTsKLXR5cGVkZWYgdm9pZCAqTmF0aXZlV2luZG93VHlwZTsKLXR5cGVkZWYgdm9pZCAqTmF0aXZlUGl4bWFwVHlwZTsKLQotLyoKLSoqIEVHTCBhbmQgbmF0aXZlIGhhbmRsZSB2YWx1ZXMKLSovCi0jZGVmaW5lIEVHTF9ERUZBVUxUX0RJU1BMQVkgKChOYXRpdmVEaXNwbGF5VHlwZSkwKQotI2RlZmluZSBFR0xfTk9fQ09OVEVYVCAoKEVHTENvbnRleHQpMCkKLSNkZWZpbmUgRUdMX05PX0RJU1BMQVkgKChFR0xEaXNwbGF5KTApCi0jZGVmaW5lIEVHTF9OT19TVVJGQUNFICgoRUdMU3VyZmFjZSkwKQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvaW5jbHVkZS9HTEVTL2dsLmggYi9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9pbmNsdWRlL0dMRVMvZ2wuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDE1NDgyMi4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9pbmNsdWRlL0dMRVMvZ2wuaAorKysgL2Rldi9udWxsCkBAIC0xLDU4NCArMCwwIEBACi0jaWZuZGVmIF9fZ2xfaF8KLSNkZWZpbmUgX19nbF9oXwotCi0jaWZkZWYgX19jcGx1c3BsdXMKLWV4dGVybiAiQyIgewotI2VuZGlmCi0KLS8qCi0qKiBMaWNlbnNlIEFwcGxpY2FiaWxpdHkuIEV4Y2VwdCB0byB0aGUgZXh0ZW50IHBvcnRpb25zIG9mIHRoaXMgZmlsZSBhcmUKLSoqIG1hZGUgc3ViamVjdCB0byBhbiBhbHRlcm5hdGl2ZSBsaWNlbnNlIGFzIHBlcm1pdHRlZCBpbiB0aGUgU0dJIEZyZWUKLSoqIFNvZnR3YXJlIExpY2Vuc2UgQiwgVmVyc2lvbiAxLjAgKHRoZSAiTGljZW5zZSIpLCB0aGUgY29udGVudHMgb2YgdGhpcwotKiogZmlsZSBhcmUgc3ViamVjdCBvbmx5IHRvIHRoZSBwcm92aXNpb25zIG9mIHRoZSBMaWNlbnNlLiBZb3UgbWF5IG5vdCB1c2UKLSoqIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiBZb3UgbWF5IG9idGFpbiBhIGNvcHkKLSoqIG9mIHRoZSBMaWNlbnNlIGF0IFNpbGljb24gR3JhcGhpY3MsIEluYy4sIGF0dG46IExlZ2FsIFNlcnZpY2VzLCAxNjAwCi0qKiBBbXBoaXRoZWF0cmUgUGFya3dheSwgTW91bnRhaW4gVmlldywgQ0EgOTQwNDMtMTM1MSwgb3IgYXQ6Ci0qKiAKLSoqIGh0dHA6Ly9vc3Muc2dpLmNvbS9wcm9qZWN0cy9GcmVlQgotKiogCi0qKiBOb3RlIHRoYXQsIGFzIHByb3ZpZGVkIGluIHRoZSBMaWNlbnNlLCB0aGUgU29mdHdhcmUgaXMgZGlzdHJpYnV0ZWQgb24gYW4KLSoqICJBUyBJUyIgYmFzaXMsIHdpdGggQUxMIEVYUFJFU1MgQU5EIElNUExJRUQgV0FSUkFOVElFUyBBTkQgQ09ORElUSU9OUwotKiogRElTQ0xBSU1FRCwgSU5DTFVESU5HLCBXSVRIT1VUIExJTUlUQVRJT04sIEFOWSBJTVBMSUVEIFdBUlJBTlRJRVMgQU5ECi0qKiBDT05ESVRJT05TIE9GIE1FUkNIQU5UQUJJTElUWSwgU0FUSVNGQUNUT1JZIFFVQUxJVFksIEZJVE5FU1MgRk9SIEEKLSoqIFBBUlRJQ1VMQVIgUFVSUE9TRSwgQU5EIE5PTi1JTkZSSU5HRU1FTlQuCi0qKiAKLSoqIE9yaWdpbmFsIENvZGUuIFRoZSBPcmlnaW5hbCBDb2RlIGlzOiBPcGVuR0wgU2FtcGxlIEltcGxlbWVudGF0aW9uLAotKiogVmVyc2lvbiAxLjIuMSwgcmVsZWFzZWQgSmFudWFyeSAyNiwgMjAwMCwgZGV2ZWxvcGVkIGJ5IFNpbGljb24gR3JhcGhpY3MsCi0qKiBJbmMuIFRoZSBPcmlnaW5hbCBDb2RlIGlzIENvcHlyaWdodCAoYykgMTk5MS0yMDAwIFNpbGljb24gR3JhcGhpY3MsIEluYy4KLSoqIENvcHlyaWdodCBpbiBhbnkgcG9ydGlvbnMgY3JlYXRlZCBieSB0aGlyZCBwYXJ0aWVzIGlzIGFzIGluZGljYXRlZAotKiogZWxzZXdoZXJlIGhlcmVpbi4gQWxsIFJpZ2h0cyBSZXNlcnZlZC4KLSoqIAotKiogQWRkaXRpb25hbCBOb3RpY2UgUHJvdmlzaW9uczogVGhlIGFwcGxpY2F0aW9uIHByb2dyYW1taW5nIGludGVyZmFjZXMKLSoqIGVzdGFibGlzaGVkIGJ5IFNHSSBpbiBjb25qdW5jdGlvbiB3aXRoIHRoZSBPcmlnaW5hbCBDb2RlIGFyZSBUaGUKLSoqIE9wZW5HTChSKSBHcmFwaGljcyBTeXN0ZW06IEEgU3BlY2lmaWNhdGlvbiAoVmVyc2lvbiAxLjIuMSksIHJlbGVhc2VkCi0qKiBBcHJpbCAxLCAxOTk5OyBUaGUgT3BlbkdMKFIpIEdyYXBoaWNzIFN5c3RlbSBVdGlsaXR5IExpYnJhcnkgKFZlcnNpb24KLSoqIDEuMyksIHJlbGVhc2VkIE5vdmVtYmVyIDQsIDE5OTg7IGFuZCBPcGVuR0woUikgR3JhcGhpY3Mgd2l0aCB0aGUgWAotKiogV2luZG93IFN5c3RlbShSKSAoVmVyc2lvbiAxLjMpLCByZWxlYXNlZCBPY3RvYmVyIDE5LCAxOTk4LiBUaGlzIHNvZnR3YXJlCi0qKiB3YXMgY3JlYXRlZCB1c2luZyB0aGUgT3BlbkdMKFIpIHZlcnNpb24gMS4yLjEgU2FtcGxlIEltcGxlbWVudGF0aW9uCi0qKiBwdWJsaXNoZWQgYnkgU0dJLCBidXQgaGFzIG5vdCBiZWVuIGluZGVwZW5kZW50bHkgdmVyaWZpZWQgYXMgYmVpbmcKLSoqIGNvbXBsaWFudCB3aXRoIHRoZSBPcGVuR0woUikgdmVyc2lvbiAxLjIuMSBTcGVjaWZpY2F0aW9uLgotKi8KLQotI2lmIGRlZmluZWQoX1dJTjMyKSAmJiAhZGVmaW5lZChBUElFTlRSWSkgJiYgIWRlZmluZWQoX19DWUdXSU5fXykKLSNkZWZpbmUgV0lOMzJfTEVBTl9BTkRfTUVBTiAxCi0jaW5jbHVkZSA8d2luZG93cy5oPgotI2VuZGlmCi0KLSNpZm5kZWYgQVBJRU5UUlkKLSNkZWZpbmUgQVBJRU5UUlkKLSNlbmRpZgotI2lmbmRlZiBHTEFQSQotI2RlZmluZSBHTEFQSSBleHRlcm4KLSNlbmRpZgotCi10eXBlZGVmIHVuc2lnbmVkIGludCBHTGVudW07Ci10eXBlZGVmIHVuc2lnbmVkIGNoYXIgR0xib29sZWFuOwotdHlwZWRlZiB1bnNpZ25lZCBpbnQgR0xiaXRmaWVsZDsKLXR5cGVkZWYgc2lnbmVkIGNoYXIgR0xieXRlOwotdHlwZWRlZiBzaG9ydCBHTHNob3J0OwotdHlwZWRlZiBpbnQgR0xpbnQ7Ci10eXBlZGVmIGludCBHTHNpemVpOwotdHlwZWRlZiB1bnNpZ25lZCBjaGFyIEdMdWJ5dGU7Ci10eXBlZGVmIHVuc2lnbmVkIHNob3J0IEdMdXNob3J0OwotdHlwZWRlZiB1bnNpZ25lZCBpbnQgR0x1aW50OwotdHlwZWRlZiBmbG9hdCBHTGZsb2F0OwotdHlwZWRlZiBmbG9hdCBHTGNsYW1wZjsKLXR5cGVkZWYgdm9pZCBHTHZvaWQ7Ci10eXBlZGVmIGludCBHTGludHB0ckFSQjsKLXR5cGVkZWYgaW50IEdMc2l6ZWlwdHJBUkI7Ci10eXBlZGVmIGludCBHTGZpeGVkOwotdHlwZWRlZiBpbnQgR0xjbGFtcHg7Ci0vKiBJbnRlcm5hbCBjb252ZW5pZW5jZSB0eXBlZGVmcyAqLwotdHlwZWRlZiB2b2lkICgqX0dMZnVuY3B0cikoKTsKLQotLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCi0KLS8qIEV4dGVuc2lvbnMgKi8KLSNkZWZpbmUgR0xfT0VTX1ZFUlNJT05fMV8wICAgICAgICAgICAgICAgIDEKLSNkZWZpbmUgR0xfT0VTX3JlYWRfZm9ybWF0ICAgICAgICAgICAgICAgIDEKLSNkZWZpbmUgR0xfT0VTX2NvbXByZXNzZWRfcGFsZXR0ZWRfdGV4dHVyZSAxCi0KLS8qIENsZWFyQnVmZmVyTWFzayAqLwotI2RlZmluZSBHTF9ERVBUSF9CVUZGRVJfQklUICAgICAgICAgICAgICAgMHgwMDAwMDEwMAotI2RlZmluZSBHTF9TVEVOQ0lMX0JVRkZFUl9CSVQgICAgICAgICAgICAgMHgwMDAwMDQwMAotI2RlZmluZSBHTF9DT0xPUl9CVUZGRVJfQklUICAgICAgICAgICAgICAgMHgwMDAwNDAwMAotCi0vKiBCb29sZWFuICovCi0jZGVmaW5lIEdMX0ZBTFNFICAgICAgICAgICAgICAgICAgICAgICAgICAwCi0jZGVmaW5lIEdMX1RSVUUgICAgICAgICAgICAgICAgICAgICAgICAgICAxCi0KLS8qIEJlZ2luTW9kZSAqLwotI2RlZmluZSBHTF9QT0lOVFMgICAgICAgICAgICAgICAgICAgICAgICAgMHgwMDAwCi0jZGVmaW5lIEdMX0xJTkVTICAgICAgICAgICAgICAgICAgICAgICAgICAweDAwMDEKLSNkZWZpbmUgR0xfTElORV9MT09QICAgICAgICAgICAgICAgICAgICAgIDB4MDAwMgotI2RlZmluZSBHTF9MSU5FX1NUUklQICAgICAgICAgICAgICAgICAgICAgMHgwMDAzCi0jZGVmaW5lIEdMX1RSSUFOR0xFUyAgICAgICAgICAgICAgICAgICAgICAweDAwMDQKLSNkZWZpbmUgR0xfVFJJQU5HTEVfU1RSSVAgICAgICAgICAgICAgICAgIDB4MDAwNQotI2RlZmluZSBHTF9UUklBTkdMRV9GQU4gICAgICAgICAgICAgICAgICAgMHgwMDA2Ci0KLS8qIEFscGhhRnVuY3Rpb24gKi8KLSNkZWZpbmUgR0xfTkVWRVIgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDIwMAotI2RlZmluZSBHTF9MRVNTICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwMjAxCi0jZGVmaW5lIEdMX0VRVUFMICAgICAgICAgICAgICAgICAgICAgICAgICAweDAyMDIKLSNkZWZpbmUgR0xfTEVRVUFMICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDIwMwotI2RlZmluZSBHTF9HUkVBVEVSICAgICAgICAgICAgICAgICAgICAgICAgMHgwMjA0Ci0jZGVmaW5lIEdMX05PVEVRVUFMICAgICAgICAgICAgICAgICAgICAgICAweDAyMDUKLSNkZWZpbmUgR0xfR0VRVUFMICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDIwNgotI2RlZmluZSBHTF9BTFdBWVMgICAgICAgICAgICAgICAgICAgICAgICAgMHgwMjA3Ci0KLS8qIEJsZW5kaW5nRmFjdG9yRGVzdCAqLwotI2RlZmluZSBHTF9aRVJPICAgICAgICAgICAgICAgICAgICAgICAgICAgMAotI2RlZmluZSBHTF9PTkUgICAgICAgICAgICAgICAgICAgICAgICAgICAgMQotI2RlZmluZSBHTF9TUkNfQ09MT1IgICAgICAgICAgICAgICAgICAgICAgMHgwMzAwCi0jZGVmaW5lIEdMX09ORV9NSU5VU19TUkNfQ09MT1IgICAgICAgICAgICAweDAzMDEKLSNkZWZpbmUgR0xfU1JDX0FMUEhBICAgICAgICAgICAgICAgICAgICAgIDB4MDMwMgotI2RlZmluZSBHTF9PTkVfTUlOVVNfU1JDX0FMUEhBICAgICAgICAgICAgMHgwMzAzCi0jZGVmaW5lIEdMX0RTVF9BTFBIQSAgICAgICAgICAgICAgICAgICAgICAweDAzMDQKLSNkZWZpbmUgR0xfT05FX01JTlVTX0RTVF9BTFBIQSAgICAgICAgICAgIDB4MDMwNQotCi0vKiBCbGVuZGluZ0ZhY3RvclNyYyAqLwotLyogICAgICBHTF9aRVJPICovCi0vKiAgICAgIEdMX09ORSAqLwotI2RlZmluZSBHTF9EU1RfQ09MT1IgICAgICAgICAgICAgICAgICAgICAgMHgwMzA2Ci0jZGVmaW5lIEdMX09ORV9NSU5VU19EU1RfQ09MT1IgICAgICAgICAgICAweDAzMDcKLSNkZWZpbmUgR0xfU1JDX0FMUEhBX1NBVFVSQVRFICAgICAgICAgICAgIDB4MDMwOAotLyogICAgICBHTF9TUkNfQUxQSEEgKi8KLS8qICAgICAgR0xfT05FX01JTlVTX1NSQ19BTFBIQSAqLwotLyogICAgICBHTF9EU1RfQUxQSEEgKi8KLS8qICAgICAgR0xfT05FX01JTlVTX0RTVF9BTFBIQSAqLwotCi0vKiBDb2xvck1hdGVyaWFsRmFjZSAqLwotLyogICAgICBHTF9GUk9OVF9BTkRfQkFDSyAqLwotCi0vKiBDb2xvck1hdGVyaWFsUGFyYW1ldGVyICovCi0vKiAgICAgIEdMX0FNQklFTlRfQU5EX0RJRkZVU0UgKi8KLQotLyogQ29sb3JQb2ludGVyVHlwZSAqLwotLyogICAgICBHTF9VTlNJR05FRF9CWVRFICovCi0vKiAgICAgIEdMX0ZMT0FUICovCi0vKiAgICAgIEdMX0ZJWEVEICovCi0KLS8qIEN1bGxGYWNlTW9kZSAqLwotI2RlZmluZSBHTF9GUk9OVCAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwNDA0Ci0jZGVmaW5lIEdMX0JBQ0sgICAgICAgICAgICAgICAgICAgICAgICAgICAweDA0MDUKLSNkZWZpbmUgR0xfRlJPTlRfQU5EX0JBQ0sgICAgICAgICAgICAgICAgIDB4MDQwOAotCi0vKiBEZXB0aEZ1bmN0aW9uICovCi0vKiAgICAgIEdMX05FVkVSICovCi0vKiAgICAgIEdMX0xFU1MgKi8KLS8qICAgICAgR0xfRVFVQUwgKi8KLS8qICAgICAgR0xfTEVRVUFMICovCi0vKiAgICAgIEdMX0dSRUFURVIgKi8KLS8qICAgICAgR0xfTk9URVFVQUwgKi8KLS8qICAgICAgR0xfR0VRVUFMICovCi0vKiAgICAgIEdMX0FMV0FZUyAqLwotCi0vKiBFbmFibGVDYXAgKi8KLSNkZWZpbmUgR0xfRk9HICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MEI2MAotI2RlZmluZSBHTF9MSUdIVElORyAgICAgICAgICAgICAgICAgICAgICAgMHgwQjUwCi0jZGVmaW5lIEdMX1RFWFRVUkVfMkQgICAgICAgICAgICAgICAgICAgICAweDBERTEKLSNkZWZpbmUgR0xfQ1VMTF9GQUNFICAgICAgICAgICAgICAgICAgICAgIDB4MEI0NAotI2RlZmluZSBHTF9BTFBIQV9URVNUICAgICAgICAgICAgICAgICAgICAgMHgwQkMwCi0jZGVmaW5lIEdMX0JMRU5EICAgICAgICAgICAgICAgICAgICAgICAgICAweDBCRTIKLSNkZWZpbmUgR0xfQ09MT1JfTE9HSUNfT1AgICAgICAgICAgICAgICAgIDB4MEJGMgotI2RlZmluZSBHTF9ESVRIRVIgICAgICAgICAgICAgICAgICAgICAgICAgMHgwQkQwCi0jZGVmaW5lIEdMX1NURU5DSUxfVEVTVCAgICAgICAgICAgICAgICAgICAweDBCOTAKLSNkZWZpbmUgR0xfREVQVEhfVEVTVCAgICAgICAgICAgICAgICAgICAgIDB4MEI3MQotLyogICAgICBHTF9MSUdIVDAgKi8KLS8qICAgICAgR0xfTElHSFQxICovCi0vKiAgICAgIEdMX0xJR0hUMiAqLwotLyogICAgICBHTF9MSUdIVDMgKi8KLS8qICAgICAgR0xfTElHSFQ0ICovCi0vKiAgICAgIEdMX0xJR0hUNSAqLwotLyogICAgICBHTF9MSUdIVDYgKi8KLS8qICAgICAgR0xfTElHSFQ3ICovCi0jZGVmaW5lIEdMX1BPSU5UX1NNT09USCAgICAgICAgICAgICAgICAgICAweDBCMTAKLSNkZWZpbmUgR0xfTElORV9TTU9PVEggICAgICAgICAgICAgICAgICAgIDB4MEIyMAotI2RlZmluZSBHTF9TQ0lTU09SX1RFU1QgICAgICAgICAgICAgICAgICAgMHgwQzExCi0jZGVmaW5lIEdMX0NPTE9SX01BVEVSSUFMICAgICAgICAgICAgICAgICAweDBCNTcKLSNkZWZpbmUgR0xfTk9STUFMSVpFICAgICAgICAgICAgICAgICAgICAgIDB4MEJBMQotI2RlZmluZSBHTF9SRVNDQUxFX05PUk1BTCAgICAgICAgICAgICAgICAgMHg4MDNBCi0jZGVmaW5lIEdMX1BPTFlHT05fT0ZGU0VUX0ZJTEwgICAgICAgICAgICAweDgwMzcKLSNkZWZpbmUgR0xfVkVSVEVYX0FSUkFZICAgICAgICAgICAgICAgICAgIDB4ODA3NAotI2RlZmluZSBHTF9OT1JNQUxfQVJSQVkgICAgICAgICAgICAgICAgICAgMHg4MDc1Ci0jZGVmaW5lIEdMX0NPTE9SX0FSUkFZICAgICAgICAgICAgICAgICAgICAweDgwNzYKLSNkZWZpbmUgR0xfVEVYVFVSRV9DT09SRF9BUlJBWSAgICAgICAgICAgIDB4ODA3OAotI2RlZmluZSBHTF9NVUxUSVNBTVBMRSAgICAgICAgICAgICAgICAgICAgMHg4MDlECi0jZGVmaW5lIEdMX1NBTVBMRV9BTFBIQV9UT19DT1ZFUkFHRSAgICAgICAweDgwOUUKLSNkZWZpbmUgR0xfU0FNUExFX0FMUEhBX1RPX09ORSAgICAgICAgICAgIDB4ODA5RgotI2RlZmluZSBHTF9TQU1QTEVfQ09WRVJBR0UgICAgICAgICAgICAgICAgMHg4MEEwCi0KLS8qIEVycm9yQ29kZSAqLwotI2RlZmluZSBHTF9OT19FUlJPUiAgICAgICAgICAgICAgICAgICAgICAgMAotI2RlZmluZSBHTF9JTlZBTElEX0VOVU0gICAgICAgICAgICAgICAgICAgMHgwNTAwCi0jZGVmaW5lIEdMX0lOVkFMSURfVkFMVUUgICAgICAgICAgICAgICAgICAweDA1MDEKLSNkZWZpbmUgR0xfSU5WQUxJRF9PUEVSQVRJT04gICAgICAgICAgICAgIDB4MDUwMgotI2RlZmluZSBHTF9TVEFDS19PVkVSRkxPVyAgICAgICAgICAgICAgICAgMHgwNTAzCi0jZGVmaW5lIEdMX1NUQUNLX1VOREVSRkxPVyAgICAgICAgICAgICAgICAweDA1MDQKLSNkZWZpbmUgR0xfT1VUX09GX01FTU9SWSAgICAgICAgICAgICAgICAgIDB4MDUwNQotCi0vKiBGb2dNb2RlICovCi0vKiAgICAgIEdMX0xJTkVBUiAqLwotI2RlZmluZSBHTF9FWFAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwODAwCi0jZGVmaW5lIEdMX0VYUDIgICAgICAgICAgICAgICAgICAgICAgICAgICAweDA4MDEKLQotLyogRm9nUGFyYW1ldGVyICovCi0jZGVmaW5lIEdMX0ZPR19ERU5TSVRZICAgICAgICAgICAgICAgICAgICAweDBCNjIKLSNkZWZpbmUgR0xfRk9HX1NUQVJUICAgICAgICAgICAgICAgICAgICAgIDB4MEI2MwotI2RlZmluZSBHTF9GT0dfRU5EICAgICAgICAgICAgICAgICAgICAgICAgMHgwQjY0Ci0jZGVmaW5lIEdMX0ZPR19NT0RFICAgICAgICAgICAgICAgICAgICAgICAweDBCNjUKLSNkZWZpbmUgR0xfRk9HX0NPTE9SICAgICAgICAgICAgICAgICAgICAgIDB4MEI2NgotCi0vKiBGcm9udEZhY2VEaXJlY3Rpb24gKi8KLSNkZWZpbmUgR0xfQ1cgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDkwMAotI2RlZmluZSBHTF9DQ1cgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwOTAxCi0KLS8qIEdldFBOYW1lICovCi0jZGVmaW5lIEdMX1NNT09USF9QT0lOVF9TSVpFX1JBTkdFICAgICAgICAweDBCMTIKLSNkZWZpbmUgR0xfU01PT1RIX0xJTkVfV0lEVEhfUkFOR0UgICAgICAgIDB4MEIyMgotI2RlZmluZSBHTF9BTElBU0VEX1BPSU5UX1NJWkVfUkFOR0UgICAgICAgMHg4NDZECi0jZGVmaW5lIEdMX0FMSUFTRURfTElORV9XSURUSF9SQU5HRSAgICAgICAweDg0NkUKLSNkZWZpbmUgR0xfSU1QTEVNRU5UQVRJT05fQ09MT1JfUkVBRF9UWVBFX09FUyAweDhCOUEKLSNkZWZpbmUgR0xfSU1QTEVNRU5UQVRJT05fQ09MT1JfUkVBRF9GT1JNQVRfT0VTIDB4OEI5QgotI2RlZmluZSBHTF9NQVhfTElHSFRTICAgICAgICAgICAgICAgICAgICAgMHgwRDMxCi0jZGVmaW5lIEdMX01BWF9URVhUVVJFX1NJWkUgICAgICAgICAgICAgICAweDBEMzMKLSNkZWZpbmUgR0xfTUFYX01PREVMVklFV19TVEFDS19ERVBUSCAgICAgIDB4MEQzNgotI2RlZmluZSBHTF9NQVhfUFJPSkVDVElPTl9TVEFDS19ERVBUSCAgICAgMHgwRDM4Ci0jZGVmaW5lIEdMX01BWF9URVhUVVJFX1NUQUNLX0RFUFRIICAgICAgICAweDBEMzkKLSNkZWZpbmUgR0xfTUFYX1ZJRVdQT1JUX0RJTVMgICAgICAgICAgICAgIDB4MEQzQQotI2RlZmluZSBHTF9NQVhfRUxFTUVOVFNfVkVSVElDRVMgICAgICAgICAgMHg4MEU4Ci0jZGVmaW5lIEdMX01BWF9FTEVNRU5UU19JTkRJQ0VTICAgICAgICAgICAweDgwRTkKLSNkZWZpbmUgR0xfTUFYX1RFWFRVUkVfVU5JVFMgICAgICAgICAgICAgIDB4ODRFMgotI2RlZmluZSBHTF9OVU1fQ09NUFJFU1NFRF9URVhUVVJFX0ZPUk1BVFMgMHg4NkEyCi0jZGVmaW5lIEdMX0NPTVBSRVNTRURfVEVYVFVSRV9GT1JNQVRTICAgICAweDg2QTMKLSNkZWZpbmUgR0xfU1VCUElYRUxfQklUUyAgICAgICAgICAgICAgICAgIDB4MEQ1MAotI2RlZmluZSBHTF9SRURfQklUUyAgICAgICAgICAgICAgICAgICAgICAgMHgwRDUyCi0jZGVmaW5lIEdMX0dSRUVOX0JJVFMgICAgICAgICAgICAgICAgICAgICAweDBENTMKLSNkZWZpbmUgR0xfQkxVRV9CSVRTICAgICAgICAgICAgICAgICAgICAgIDB4MEQ1NAotI2RlZmluZSBHTF9BTFBIQV9CSVRTICAgICAgICAgICAgICAgICAgICAgMHgwRDU1Ci0jZGVmaW5lIEdMX0RFUFRIX0JJVFMgICAgICAgICAgICAgICAgICAgICAweDBENTYKLSNkZWZpbmUgR0xfU1RFTkNJTF9CSVRTICAgICAgICAgICAgICAgICAgIDB4MEQ1NwotCi0vKiBIaW50TW9kZSAqLwotI2RlZmluZSBHTF9ET05UX0NBUkUgICAgICAgICAgICAgICAgICAgICAgMHgxMTAwCi0jZGVmaW5lIEdMX0ZBU1RFU1QgICAgICAgICAgICAgICAgICAgICAgICAweDExMDEKLSNkZWZpbmUgR0xfTklDRVNUICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTEwMgotCi0vKiBIaW50VGFyZ2V0ICovCi0jZGVmaW5lIEdMX1BFUlNQRUNUSVZFX0NPUlJFQ1RJT05fSElOVCAgICAweDBDNTAKLSNkZWZpbmUgR0xfUE9JTlRfU01PT1RIX0hJTlQgICAgICAgICAgICAgIDB4MEM1MQotI2RlZmluZSBHTF9MSU5FX1NNT09USF9ISU5UICAgICAgICAgICAgICAgMHgwQzUyCi0jZGVmaW5lIEdMX1BPTFlHT05fU01PT1RIX0hJTlQgICAgICAgICAgICAweDBDNTMKLSNkZWZpbmUgR0xfRk9HX0hJTlQgICAgICAgICAgICAgICAgICAgICAgIDB4MEM1NAotCi0vKiBMaWdodE1vZGVsUGFyYW1ldGVyICovCi0jZGVmaW5lIEdMX0xJR0hUX01PREVMX0FNQklFTlQgICAgICAgICAgICAweDBCNTMKLSNkZWZpbmUgR0xfTElHSFRfTU9ERUxfVFdPX1NJREUgICAgICAgICAgIDB4MEI1MgotCi0vKiBMaWdodFBhcmFtZXRlciAqLwotI2RlZmluZSBHTF9BTUJJRU5UICAgICAgICAgICAgICAgICAgICAgICAgMHgxMjAwCi0jZGVmaW5lIEdMX0RJRkZVU0UgICAgICAgICAgICAgICAgICAgICAgICAweDEyMDEKLSNkZWZpbmUgR0xfU1BFQ1VMQVIgICAgICAgICAgICAgICAgICAgICAgIDB4MTIwMgotI2RlZmluZSBHTF9QT1NJVElPTiAgICAgICAgICAgICAgICAgICAgICAgMHgxMjAzCi0jZGVmaW5lIEdMX1NQT1RfRElSRUNUSU9OICAgICAgICAgICAgICAgICAweDEyMDQKLSNkZWZpbmUgR0xfU1BPVF9FWFBPTkVOVCAgICAgICAgICAgICAgICAgIDB4MTIwNQotI2RlZmluZSBHTF9TUE9UX0NVVE9GRiAgICAgICAgICAgICAgICAgICAgMHgxMjA2Ci0jZGVmaW5lIEdMX0NPTlNUQU5UX0FUVEVOVUFUSU9OICAgICAgICAgICAweDEyMDcKLSNkZWZpbmUgR0xfTElORUFSX0FUVEVOVUFUSU9OICAgICAgICAgICAgIDB4MTIwOAotI2RlZmluZSBHTF9RVUFEUkFUSUNfQVRURU5VQVRJT04gICAgICAgICAgMHgxMjA5Ci0KLS8qIERhdGFUeXBlICovCi0jZGVmaW5lIEdMX0JZVEUgICAgICAgICAgICAgICAgICAgICAgICAgICAweDE0MDAKLSNkZWZpbmUgR0xfVU5TSUdORURfQllURSAgICAgICAgICAgICAgICAgIDB4MTQwMQotI2RlZmluZSBHTF9TSE9SVCAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNDAyCi0jZGVmaW5lIEdMX1VOU0lHTkVEX1NIT1JUICAgICAgICAgICAgICAgICAweDE0MDMKLSNkZWZpbmUgR0xfRkxPQVQgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTQwNgotI2RlZmluZSBHTF9GSVhFRCAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNDBDCi0KLS8qIExvZ2ljT3AgKi8KLSNkZWZpbmUgR0xfQ0xFQVIgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTUwMAotI2RlZmluZSBHTF9BTkQgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNTAxCi0jZGVmaW5lIEdMX0FORF9SRVZFUlNFICAgICAgICAgICAgICAgICAgICAweDE1MDIKLSNkZWZpbmUgR0xfQ09QWSAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTUwMwotI2RlZmluZSBHTF9BTkRfSU5WRVJURUQgICAgICAgICAgICAgICAgICAgMHgxNTA0Ci0jZGVmaW5lIEdMX05PT1AgICAgICAgICAgICAgICAgICAgICAgICAgICAweDE1MDUKLSNkZWZpbmUgR0xfWE9SICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTUwNgotI2RlZmluZSBHTF9PUiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNTA3Ci0jZGVmaW5lIEdMX05PUiAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDE1MDgKLSNkZWZpbmUgR0xfRVFVSVYgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTUwOQotI2RlZmluZSBHTF9JTlZFUlQgICAgICAgICAgICAgICAgICAgICAgICAgMHgxNTBBCi0jZGVmaW5lIEdMX09SX1JFVkVSU0UgICAgICAgICAgICAgICAgICAgICAweDE1MEIKLSNkZWZpbmUgR0xfQ09QWV9JTlZFUlRFRCAgICAgICAgICAgICAgICAgIDB4MTUwQwotI2RlZmluZSBHTF9PUl9JTlZFUlRFRCAgICAgICAgICAgICAgICAgICAgMHgxNTBECi0jZGVmaW5lIEdMX05BTkQgICAgICAgICAgICAgICAgICAgICAgICAgICAweDE1MEUKLSNkZWZpbmUgR0xfU0VUICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTUwRgotCi0vKiBNYXRlcmlhbEZhY2UgKi8KLS8qICAgICAgR0xfRlJPTlRfQU5EX0JBQ0sgKi8KLQotLyogTWF0ZXJpYWxQYXJhbWV0ZXIgKi8KLSNkZWZpbmUgR0xfRU1JU1NJT04gICAgICAgICAgICAgICAgICAgICAgIDB4MTYwMAotI2RlZmluZSBHTF9TSElOSU5FU1MgICAgICAgICAgICAgICAgICAgICAgMHgxNjAxCi0jZGVmaW5lIEdMX0FNQklFTlRfQU5EX0RJRkZVU0UgICAgICAgICAgICAweDE2MDIKLS8qICAgICAgR0xfQU1CSUVOVCAqLwotLyogICAgICBHTF9ESUZGVVNFICovCi0vKiAgICAgIEdMX1NQRUNVTEFSICovCi0KLS8qIE1hdHJpeE1vZGUgKi8KLSNkZWZpbmUgR0xfTU9ERUxWSUVXICAgICAgICAgICAgICAgICAgICAgIDB4MTcwMAotI2RlZmluZSBHTF9QUk9KRUNUSU9OICAgICAgICAgICAgICAgICAgICAgMHgxNzAxCi0jZGVmaW5lIEdMX1RFWFRVUkUgICAgICAgICAgICAgICAgICAgICAgICAweDE3MDIKLQotLyogTm9ybWFsUG9pbnRlclR5cGUgKi8KLS8qICAgICAgR0xfQllURSAqLwotLyogICAgICBHTF9TSE9SVCAqLwotLyogICAgICBHTF9GTE9BVCAqLwotLyogICAgICBHTF9GSVhFRCAqLwotCi0vKiBQaXhlbEZvcm1hdCAqLwotI2RlZmluZSBHTF9BTFBIQSAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxOTA2Ci0jZGVmaW5lIEdMX1JHQiAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDE5MDcKLSNkZWZpbmUgR0xfUkdCQSAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MTkwOAotI2RlZmluZSBHTF9MVU1JTkFOQ0UgICAgICAgICAgICAgICAgICAgICAgMHgxOTA5Ci0jZGVmaW5lIEdMX0xVTUlOQU5DRV9BTFBIQSAgICAgICAgICAgICAgICAweDE5MEEKLQotLyogUGl4ZWxTdG9yZVBhcmFtZXRlciAqLwotI2RlZmluZSBHTF9VTlBBQ0tfQUxJR05NRU5UICAgICAgICAgICAgICAgMHgwQ0Y1Ci0jZGVmaW5lIEdMX1BBQ0tfQUxJR05NRU5UICAgICAgICAgICAgICAgICAweDBEMDUKLQotLyogUGl4ZWxUeXBlICovCi0vKiAgICAgIEdMX1VOU0lHTkVEX0JZVEUgKi8KLSNkZWZpbmUgR0xfVU5TSUdORURfU0hPUlRfNF80XzRfNCAgICAgICAgIDB4ODAzMwotI2RlZmluZSBHTF9VTlNJR05FRF9TSE9SVF81XzVfNV8xICAgICAgICAgMHg4MDM0Ci0jZGVmaW5lIEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81ICAgICAgICAgICAweDgzNjMKLQotLyogU2hhZGluZ01vZGVsICovCi0jZGVmaW5lIEdMX0ZMQVQgICAgICAgICAgICAgICAgICAgICAgICAgICAweDFEMDAKLSNkZWZpbmUgR0xfU01PT1RIICAgICAgICAgICAgICAgICAgICAgICAgIDB4MUQwMQotCi0vKiBTdGVuY2lsRnVuY3Rpb24gKi8KLS8qICAgICAgR0xfTkVWRVIgKi8KLS8qICAgICAgR0xfTEVTUyAqLwotLyogICAgICBHTF9FUVVBTCAqLwotLyogICAgICBHTF9MRVFVQUwgKi8KLS8qICAgICAgR0xfR1JFQVRFUiAqLwotLyogICAgICBHTF9OT1RFUVVBTCAqLwotLyogICAgICBHTF9HRVFVQUwgKi8KLS8qICAgICAgR0xfQUxXQVlTICovCi0KLS8qIFN0ZW5jaWxPcCAqLwotLyogICAgICBHTF9aRVJPICovCi0jZGVmaW5lIEdMX0tFRVAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDFFMDAKLSNkZWZpbmUgR0xfUkVQTEFDRSAgICAgICAgICAgICAgICAgICAgICAgIDB4MUUwMQotI2RlZmluZSBHTF9JTkNSICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgxRTAyCi0jZGVmaW5lIEdMX0RFQ1IgICAgICAgICAgICAgICAgICAgICAgICAgICAweDFFMDMKLS8qICAgICAgR0xfSU5WRVJUICovCi0KLS8qIFN0cmluZ05hbWUgKi8KLSNkZWZpbmUgR0xfVkVORE9SICAgICAgICAgICAgICAgICAgICAgICAgIDB4MUYwMAotI2RlZmluZSBHTF9SRU5ERVJFUiAgICAgICAgICAgICAgICAgICAgICAgMHgxRjAxCi0jZGVmaW5lIEdMX1ZFUlNJT04gICAgICAgICAgICAgICAgICAgICAgICAweDFGMDIKLSNkZWZpbmUgR0xfRVhURU5TSU9OUyAgICAgICAgICAgICAgICAgICAgIDB4MUYwMwotCi0vKiBUZXhDb29yZFBvaW50ZXJUeXBlICovCi0vKiAgICAgIEdMX1NIT1JUICovCi0vKiAgICAgIEdMX0ZMT0FUICovCi0vKiAgICAgIEdMX0ZJWEVEICovCi0vKiAgICAgIEdMX0JZVEUgKi8KLQotLyogVGV4dHVyZUVudk1vZGUgKi8KLSNkZWZpbmUgR0xfTU9EVUxBVEUgICAgICAgICAgICAgICAgICAgICAgIDB4MjEwMAotI2RlZmluZSBHTF9ERUNBTCAgICAgICAgICAgICAgICAgICAgICAgICAgMHgyMTAxCi0vKiAgICAgIEdMX0JMRU5EICovCi0jZGVmaW5lIEdMX0FERCAgICAgICAgICAgICAgICAgICAgICAgICAgICAweDAxMDQKLS8qICAgICAgR0xfUkVQTEFDRSAqLwotCi0vKiBUZXh0dXJlRW52UGFyYW1ldGVyICovCi0jZGVmaW5lIEdMX1RFWFRVUkVfRU5WX01PREUgICAgICAgICAgICAgICAweDIyMDAKLSNkZWZpbmUgR0xfVEVYVFVSRV9FTlZfQ09MT1IgICAgICAgICAgICAgIDB4MjIwMQotCi0vKiBUZXh0dXJlRW52VGFyZ2V0ICovCi0jZGVmaW5lIEdMX1RFWFRVUkVfRU5WICAgICAgICAgICAgICAgICAgICAweDIzMDAKLQotLyogVGV4dHVyZU1hZ0ZpbHRlciAqLwotI2RlZmluZSBHTF9ORUFSRVNUICAgICAgICAgICAgICAgICAgICAgICAgMHgyNjAwCi0jZGVmaW5lIEdMX0xJTkVBUiAgICAgICAgICAgICAgICAgICAgICAgICAweDI2MDEKLQotLyogVGV4dHVyZU1pbkZpbHRlciAqLwotLyogICAgICBHTF9ORUFSRVNUICovCi0vKiAgICAgIEdMX0xJTkVBUiAqLwotI2RlZmluZSBHTF9ORUFSRVNUX01JUE1BUF9ORUFSRVNUICAgICAgICAgMHgyNzAwCi0jZGVmaW5lIEdMX0xJTkVBUl9NSVBNQVBfTkVBUkVTVCAgICAgICAgICAweDI3MDEKLSNkZWZpbmUgR0xfTkVBUkVTVF9NSVBNQVBfTElORUFSICAgICAgICAgIDB4MjcwMgotI2RlZmluZSBHTF9MSU5FQVJfTUlQTUFQX0xJTkVBUiAgICAgICAgICAgMHgyNzAzCi0KLS8qIFRleHR1cmVQYXJhbWV0ZXJOYW1lICovCi0jZGVmaW5lIEdMX1RFWFRVUkVfTUFHX0ZJTFRFUiAgICAgICAgICAgICAweDI4MDAKLSNkZWZpbmUgR0xfVEVYVFVSRV9NSU5fRklMVEVSICAgICAgICAgICAgIDB4MjgwMQotI2RlZmluZSBHTF9URVhUVVJFX1dSQVBfUyAgICAgICAgICAgICAgICAgMHgyODAyCi0jZGVmaW5lIEdMX1RFWFRVUkVfV1JBUF9UICAgICAgICAgICAgICAgICAweDI4MDMKLQotLyogVGV4dHVyZVRhcmdldCAqLwotLyogICAgICBHTF9URVhUVVJFXzJEICovCi0KLS8qIFRleHR1cmVVbml0ICovCi0jZGVmaW5lIEdMX1RFWFRVUkUwICAgICAgICAgICAgICAgICAgICAgICAweDg0QzAKLSNkZWZpbmUgR0xfVEVYVFVSRTEgICAgICAgICAgICAgICAgICAgICAgIDB4ODRDMQotI2RlZmluZSBHTF9URVhUVVJFMiAgICAgICAgICAgICAgICAgICAgICAgMHg4NEMyCi0jZGVmaW5lIEdMX1RFWFRVUkUzICAgICAgICAgICAgICAgICAgICAgICAweDg0QzMKLSNkZWZpbmUgR0xfVEVYVFVSRTQgICAgICAgICAgICAgICAgICAgICAgIDB4ODRDNAotI2RlZmluZSBHTF9URVhUVVJFNSAgICAgICAgICAgICAgICAgICAgICAgMHg4NEM1Ci0jZGVmaW5lIEdMX1RFWFRVUkU2ICAgICAgICAgICAgICAgICAgICAgICAweDg0QzYKLSNkZWZpbmUgR0xfVEVYVFVSRTcgICAgICAgICAgICAgICAgICAgICAgIDB4ODRDNwotI2RlZmluZSBHTF9URVhUVVJFOCAgICAgICAgICAgICAgICAgICAgICAgMHg4NEM4Ci0jZGVmaW5lIEdMX1RFWFRVUkU5ICAgICAgICAgICAgICAgICAgICAgICAweDg0QzkKLSNkZWZpbmUgR0xfVEVYVFVSRTEwICAgICAgICAgICAgICAgICAgICAgIDB4ODRDQQotI2RlZmluZSBHTF9URVhUVVJFMTEgICAgICAgICAgICAgICAgICAgICAgMHg4NENCCi0jZGVmaW5lIEdMX1RFWFRVUkUxMiAgICAgICAgICAgICAgICAgICAgICAweDg0Q0MKLSNkZWZpbmUgR0xfVEVYVFVSRTEzICAgICAgICAgICAgICAgICAgICAgIDB4ODRDRAotI2RlZmluZSBHTF9URVhUVVJFMTQgICAgICAgICAgICAgICAgICAgICAgMHg4NENFCi0jZGVmaW5lIEdMX1RFWFRVUkUxNSAgICAgICAgICAgICAgICAgICAgICAweDg0Q0YKLSNkZWZpbmUgR0xfVEVYVFVSRTE2ICAgICAgICAgICAgICAgICAgICAgIDB4ODREMAotI2RlZmluZSBHTF9URVhUVVJFMTcgICAgICAgICAgICAgICAgICAgICAgMHg4NEQxCi0jZGVmaW5lIEdMX1RFWFRVUkUxOCAgICAgICAgICAgICAgICAgICAgICAweDg0RDIKLSNkZWZpbmUgR0xfVEVYVFVSRTE5ICAgICAgICAgICAgICAgICAgICAgIDB4ODREMwotI2RlZmluZSBHTF9URVhUVVJFMjAgICAgICAgICAgICAgICAgICAgICAgMHg4NEQ0Ci0jZGVmaW5lIEdMX1RFWFRVUkUyMSAgICAgICAgICAgICAgICAgICAgICAweDg0RDUKLSNkZWZpbmUgR0xfVEVYVFVSRTIyICAgICAgICAgICAgICAgICAgICAgIDB4ODRENgotI2RlZmluZSBHTF9URVhUVVJFMjMgICAgICAgICAgICAgICAgICAgICAgMHg4NEQ3Ci0jZGVmaW5lIEdMX1RFWFRVUkUyNCAgICAgICAgICAgICAgICAgICAgICAweDg0RDgKLSNkZWZpbmUgR0xfVEVYVFVSRTI1ICAgICAgICAgICAgICAgICAgICAgIDB4ODREOQotI2RlZmluZSBHTF9URVhUVVJFMjYgICAgICAgICAgICAgICAgICAgICAgMHg4NERBCi0jZGVmaW5lIEdMX1RFWFRVUkUyNyAgICAgICAgICAgICAgICAgICAgICAweDg0REIKLSNkZWZpbmUgR0xfVEVYVFVSRTI4ICAgICAgICAgICAgICAgICAgICAgIDB4ODREQwotI2RlZmluZSBHTF9URVhUVVJFMjkgICAgICAgICAgICAgICAgICAgICAgMHg4NERECi0jZGVmaW5lIEdMX1RFWFRVUkUzMCAgICAgICAgICAgICAgICAgICAgICAweDg0REUKLSNkZWZpbmUgR0xfVEVYVFVSRTMxICAgICAgICAgICAgICAgICAgICAgIDB4ODRERgotCi0vKiBUZXh0dXJlV3JhcE1vZGUgKi8KLSNkZWZpbmUgR0xfUkVQRUFUICAgICAgICAgICAgICAgICAgICAgICAgIDB4MjkwMQotI2RlZmluZSBHTF9DTEFNUF9UT19FREdFICAgICAgICAgICAgICAgICAgMHg4MTJGCi0KLS8qIFBpeGVsSW50ZXJuYWxGb3JtYXQgKi8KLSNkZWZpbmUgR0xfUEFMRVRURTRfUkdCOF9PRVMgICAgICAgICAgICAgIDB4OEI5MAotI2RlZmluZSBHTF9QQUxFVFRFNF9SR0JBOF9PRVMgICAgICAgICAgICAgMHg4QjkxCi0jZGVmaW5lIEdMX1BBTEVUVEU0X1I1X0c2X0I1X09FUyAgICAgICAgICAweDhCOTIKLSNkZWZpbmUgR0xfUEFMRVRURTRfUkdCQTRfT0VTICAgICAgICAgICAgIDB4OEI5MwotI2RlZmluZSBHTF9QQUxFVFRFNF9SR0I1X0ExX09FUyAgICAgICAgICAgMHg4Qjk0Ci0jZGVmaW5lIEdMX1BBTEVUVEU4X1JHQjhfT0VTICAgICAgICAgICAgICAweDhCOTUKLSNkZWZpbmUgR0xfUEFMRVRURThfUkdCQThfT0VTICAgICAgICAgICAgIDB4OEI5NgotI2RlZmluZSBHTF9QQUxFVFRFOF9SNV9HNl9CNV9PRVMgICAgICAgICAgMHg4Qjk3Ci0jZGVmaW5lIEdMX1BBTEVUVEU4X1JHQkE0X09FUyAgICAgICAgICAgICAweDhCOTgKLSNkZWZpbmUgR0xfUEFMRVRURThfUkdCNV9BMV9PRVMgICAgICAgICAgIDB4OEI5OQotCi0vKiBWZXJ0ZXhQb2ludGVyVHlwZSAqLwotLyogICAgICBHTF9TSE9SVCAqLwotLyogICAgICBHTF9GTE9BVCAqLwotLyogICAgICBHTF9GSVhFRCAqLwotLyogICAgICBHTF9CWVRFICovCi0KLS8qIExpZ2h0TmFtZSAqLwotI2RlZmluZSBHTF9MSUdIVDAgICAgICAgICAgICAgICAgICAgICAgICAgMHg0MDAwCi0jZGVmaW5lIEdMX0xJR0hUMSAgICAgICAgICAgICAgICAgICAgICAgICAweDQwMDEKLSNkZWZpbmUgR0xfTElHSFQyICAgICAgICAgICAgICAgICAgICAgICAgIDB4NDAwMgotI2RlZmluZSBHTF9MSUdIVDMgICAgICAgICAgICAgICAgICAgICAgICAgMHg0MDAzCi0jZGVmaW5lIEdMX0xJR0hUNCAgICAgICAgICAgICAgICAgICAgICAgICAweDQwMDQKLSNkZWZpbmUgR0xfTElHSFQ1ICAgICAgICAgICAgICAgICAgICAgICAgIDB4NDAwNQotI2RlZmluZSBHTF9MSUdIVDYgICAgICAgICAgICAgICAgICAgICAgICAgMHg0MDA2Ci0jZGVmaW5lIEdMX0xJR0hUNyAgICAgICAgICAgICAgICAgICAgICAgICAweDQwMDcKLQotCi0vKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KLQotR0xBUEkgdm9pZCBBUElFTlRSWSBnbEFjdGl2ZVRleHR1cmUgKEdMZW51bSB0ZXh0dXJlKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xBbHBoYUZ1bmMgKEdMZW51bSBmdW5jLCBHTGNsYW1wZiByZWYpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbEFscGhhRnVuY3ggKEdMZW51bSBmdW5jLCBHTGNsYW1weCByZWYpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbEJpbmRUZXh0dXJlIChHTGVudW0gdGFyZ2V0LCBHTHVpbnQgdGV4dHVyZSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsQmxlbmRGdW5jIChHTGVudW0gc2ZhY3RvciwgR0xlbnVtIGRmYWN0b3IpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbENsZWFyIChHTGJpdGZpZWxkIG1hc2spOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbENsZWFyQ29sb3IgKEdMY2xhbXBmIHJlZCwgR0xjbGFtcGYgZ3JlZW4sIEdMY2xhbXBmIGJsdWUsIEdMY2xhbXBmIGFscGhhKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xDbGVhckNvbG9yeCAoR0xjbGFtcHggcmVkLCBHTGNsYW1weCBncmVlbiwgR0xjbGFtcHggYmx1ZSwgR0xjbGFtcHggYWxwaGEpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbENsZWFyRGVwdGhmIChHTGNsYW1wZiBkZXB0aCk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsQ2xlYXJEZXB0aHggKEdMY2xhbXB4IGRlcHRoKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xDbGVhclN0ZW5jaWwgKEdMaW50IHMpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbENsaWVudEFjdGl2ZVRleHR1cmUgKEdMZW51bSB0ZXh0dXJlKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xDb2xvcjRmIChHTGZsb2F0IHJlZCwgR0xmbG9hdCBncmVlbiwgR0xmbG9hdCBibHVlLCBHTGZsb2F0IGFscGhhKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xDb2xvcjR4IChHTGZpeGVkIHJlZCwgR0xmaXhlZCBncmVlbiwgR0xmaXhlZCBibHVlLCBHTGZpeGVkIGFscGhhKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xDb2xvck1hc2sgKEdMYm9vbGVhbiByZWQsIEdMYm9vbGVhbiBncmVlbiwgR0xib29sZWFuIGJsdWUsIEdMYm9vbGVhbiBhbHBoYSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsQ29sb3JQb2ludGVyIChHTGludCBzaXplLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlcik7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsQ29tcHJlc3NlZFRleEltYWdlMkQgKEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGVudW0gaW50ZXJuYWxmb3JtYXQsIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LCBHTGludCBib3JkZXIsIEdMc2l6ZWkgaW1hZ2VTaXplLCBjb25zdCBHTHZvaWQgKmRhdGEpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbENvbXByZXNzZWRUZXhTdWJJbWFnZTJEIChHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgeG9mZnNldCwgR0xpbnQgeW9mZnNldCwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsIEdMZW51bSBmb3JtYXQsIEdMc2l6ZWkgaW1hZ2VTaXplLCBjb25zdCBHTHZvaWQgKmRhdGEpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbENvcHlUZXhJbWFnZTJEIChHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xlbnVtIGludGVybmFsZm9ybWF0LCBHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwgR0xpbnQgYm9yZGVyKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xDb3B5VGV4U3ViSW1hZ2UyRCAoR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMaW50IHhvZmZzZXQsIEdMaW50IHlvZmZzZXQsIEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0KTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xDdWxsRmFjZSAoR0xlbnVtIG1vZGUpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbERlbGV0ZVRleHR1cmVzIChHTHNpemVpIG4sIGNvbnN0IEdMdWludCAqdGV4dHVyZXMpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbERlcHRoRnVuYyAoR0xlbnVtIGZ1bmMpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbERlcHRoTWFzayAoR0xib29sZWFuIGZsYWcpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbERlcHRoUmFuZ2VmIChHTGNsYW1wZiB6TmVhciwgR0xjbGFtcGYgekZhcik7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsRGVwdGhSYW5nZXggKEdMY2xhbXB4IHpOZWFyLCBHTGNsYW1weCB6RmFyKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xEaXNhYmxlIChHTGVudW0gY2FwKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xEaXNhYmxlQ2xpZW50U3RhdGUgKEdMZW51bSBhcnJheSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsRHJhd0FycmF5cyAoR0xlbnVtIG1vZGUsIEdMaW50IGZpcnN0LCBHTHNpemVpIGNvdW50KTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xEcmF3RWxlbWVudHMgKEdMZW51bSBtb2RlLCBHTHNpemVpIGNvdW50LCBHTGVudW0gdHlwZSwgY29uc3QgR0x2b2lkICppbmRpY2VzKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xFbmFibGUgKEdMZW51bSBjYXApOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbEVuYWJsZUNsaWVudFN0YXRlIChHTGVudW0gYXJyYXkpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbEZpbmlzaCAodm9pZCk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsRmx1c2ggKHZvaWQpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbEZvZ2YgKEdMZW51bSBwbmFtZSwgR0xmbG9hdCBwYXJhbSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsRm9nZnYgKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xGb2d4IChHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0pOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbEZvZ3h2IChHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcyk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsRnJvbnRGYWNlIChHTGVudW0gbW9kZSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsRnJ1c3R1bWYgKEdMZmxvYXQgbGVmdCwgR0xmbG9hdCByaWdodCwgR0xmbG9hdCBib3R0b20sIEdMZmxvYXQgdG9wLCBHTGZsb2F0IHpOZWFyLCBHTGZsb2F0IHpGYXIpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbEZydXN0dW14IChHTGZpeGVkIGxlZnQsIEdMZml4ZWQgcmlnaHQsIEdMZml4ZWQgYm90dG9tLCBHTGZpeGVkIHRvcCwgR0xmaXhlZCB6TmVhciwgR0xmaXhlZCB6RmFyKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xHZW5UZXh0dXJlcyAoR0xzaXplaSBuLCBHTHVpbnQgKnRleHR1cmVzKTsKLUdMQVBJIEdMZW51bSBBUElFTlRSWSBnbEdldEVycm9yICh2b2lkKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xHZXRJbnRlZ2VydiAoR0xlbnVtIHBuYW1lLCBHTGludCAqcGFyYW1zKTsKLUdMQVBJIGNvbnN0IEdMdWJ5dGUgKiBBUElFTlRSWSBnbEdldFN0cmluZyAoR0xlbnVtIG5hbWUpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbEhpbnQgKEdMZW51bSB0YXJnZXQsIEdMZW51bSBtb2RlKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMaWdodE1vZGVsZiAoR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMaWdodE1vZGVsZnYgKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMaWdodE1vZGVseCAoR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMaWdodE1vZGVseHYgKEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMaWdodGYgKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMaWdodGZ2IChHTGVudW0gbGlnaHQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMaWdodHggKEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMaWdodHh2IChHTGVudW0gbGlnaHQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMaW5lV2lkdGggKEdMZmxvYXQgd2lkdGgpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbExpbmVXaWR0aHggKEdMZml4ZWQgd2lkdGgpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbExvYWRJZGVudGl0eSAodm9pZCk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsTG9hZE1hdHJpeGYgKGNvbnN0IEdMZmxvYXQgKm0pOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbExvYWRNYXRyaXh4IChjb25zdCBHTGZpeGVkICptKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xMb2dpY09wIChHTGVudW0gb3Bjb2RlKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xNYXRlcmlhbGYgKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbE1hdGVyaWFsZnYgKEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcyk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsTWF0ZXJpYWx4IChHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xNYXRlcmlhbHh2IChHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbE1hdHJpeE1vZGUgKEdMZW51bSBtb2RlKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xNdWx0TWF0cml4ZiAoY29uc3QgR0xmbG9hdCAqbSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsTXVsdE1hdHJpeHggKGNvbnN0IEdMZml4ZWQgKm0pOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbE11bHRpVGV4Q29vcmQ0ZiAoR0xlbnVtIHRhcmdldCwgR0xmbG9hdCBzLCBHTGZsb2F0IHQsIEdMZmxvYXQgciwgR0xmbG9hdCBxKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xNdWx0aVRleENvb3JkNHggKEdMZW51bSB0YXJnZXQsIEdMZml4ZWQgcywgR0xmaXhlZCB0LCBHTGZpeGVkIHIsIEdMZml4ZWQgcSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsTm9ybWFsM2YgKEdMZmxvYXQgbngsIEdMZmxvYXQgbnksIEdMZmxvYXQgbnopOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbE5vcm1hbDN4IChHTGZpeGVkIG54LCBHTGZpeGVkIG55LCBHTGZpeGVkIG56KTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xOb3JtYWxQb2ludGVyIChHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlcik7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsT3J0aG9mIChHTGZsb2F0IGxlZnQsIEdMZmxvYXQgcmlnaHQsIEdMZmxvYXQgYm90dG9tLCBHTGZsb2F0IHRvcCwgR0xmbG9hdCB6TmVhciwgR0xmbG9hdCB6RmFyKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xPcnRob3ggKEdMZml4ZWQgbGVmdCwgR0xmaXhlZCByaWdodCwgR0xmaXhlZCBib3R0b20sIEdMZml4ZWQgdG9wLCBHTGZpeGVkIHpOZWFyLCBHTGZpeGVkIHpGYXIpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFBpeGVsU3RvcmVpIChHTGVudW0gcG5hbWUsIEdMaW50IHBhcmFtKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xQb2ludFNpemUgKEdMZmxvYXQgc2l6ZSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsUG9pbnRTaXpleCAoR0xmaXhlZCBzaXplKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xQb2x5Z29uT2Zmc2V0IChHTGZsb2F0IGZhY3RvciwgR0xmbG9hdCB1bml0cyk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsUG9seWdvbk9mZnNldHggKEdMZml4ZWQgZmFjdG9yLCBHTGZpeGVkIHVuaXRzKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xQb3BNYXRyaXggKHZvaWQpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFB1c2hNYXRyaXggKHZvaWQpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFJlYWRQaXhlbHMgKEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LCBHTGVudW0gZm9ybWF0LCBHTGVudW0gdHlwZSwgR0x2b2lkICpwaXhlbHMpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFJvdGF0ZWYgKEdMZmxvYXQgYW5nbGUsIEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFJvdGF0ZXggKEdMZml4ZWQgYW5nbGUsIEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFNhbXBsZUNvdmVyYWdlIChHTGNsYW1wZiB2YWx1ZSwgR0xib29sZWFuIGludmVydCk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsU2FtcGxlQ292ZXJhZ2V4IChHTGNsYW1weCB2YWx1ZSwgR0xib29sZWFuIGludmVydCk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsU2NhbGVmIChHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6KTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xTY2FsZXggKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFNjaXNzb3IgKEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0KTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xTaGFkZU1vZGVsIChHTGVudW0gbW9kZSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsU3RlbmNpbEZ1bmMgKEdMZW51bSBmdW5jLCBHTGludCByZWYsIEdMdWludCBtYXNrKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xTdGVuY2lsTWFzayAoR0x1aW50IG1hc2spOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFN0ZW5jaWxPcCAoR0xlbnVtIGZhaWwsIEdMZW51bSB6ZmFpbCwgR0xlbnVtIHpwYXNzKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xUZXhDb29yZFBvaW50ZXIgKEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xUZXhFbnZmIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFRleEVudmZ2IChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcyk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsVGV4RW52eCAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xUZXhFbnZ4diAoR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFRleEltYWdlMkQgKEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGludCBpbnRlcm5hbGZvcm1hdCwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsIEdMaW50IGJvcmRlciwgR0xlbnVtIGZvcm1hdCwgR0xlbnVtIHR5cGUsIGNvbnN0IEdMdm9pZCAqcGl4ZWxzKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xUZXhQYXJhbWV0ZXJmIChHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0pOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFRleFBhcmFtZXRlcnggKEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSk7Ci1HTEFQSSB2b2lkIEFQSUVOVFJZIGdsVGV4U3ViSW1hZ2UyRCAoR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMaW50IHhvZmZzZXQsIEdMaW50IHlvZmZzZXQsIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LCBHTGVudW0gZm9ybWF0LCBHTGVudW0gdHlwZSwgY29uc3QgR0x2b2lkICpwaXhlbHMpOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFRyYW5zbGF0ZWYgKEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHopOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFRyYW5zbGF0ZXggKEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHopOwotR0xBUEkgdm9pZCBBUElFTlRSWSBnbFZlcnRleFBvaW50ZXIgKEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyKTsKLUdMQVBJIHZvaWQgQVBJRU5UUlkgZ2xWaWV3cG9ydCAoR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQpOwotCi0jaWZkZWYgX19jcGx1c3BsdXMKLX0KLSNlbmRpZgotCi0jZW5kaWYgLyogX19nbF9oXyAqLwpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvbGljZW5zZS1CU0QudHh0IGIvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvbGljZW5zZS1CU0QudHh0CmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4OTI0ZTNjLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy9hbmdlbGVzL2xpY2Vuc2UtQlNELnR4dAorKysgL2Rldi9udWxsCkBAIC0xLDM0ICswLDAgQEAKLVRoaXMgaXMgdGhlIEJTRC1zdHlsZSBsaWNlbnNlIGZvciB0aGUgIlNhbiBBbmdlbGVzIE9ic2VydmF0aW9uIg0KLU9wZW5HTCBFUyB2ZXJzaW9uIGV4YW1wbGUgc291cmNlIGNvZGUNCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0NCi0NCi1TYW4gQW5nZWxlcyBPYnNlcnZhdGlvbiBPcGVuR0wgRVMgdmVyc2lvbiBleGFtcGxlDQotQ29weXJpZ2h0IChjKSAyMDA0LTIwMDUsIEpldHJvIExhdWhhDQotQWxsIHJpZ2h0cyByZXNlcnZlZC4NCi0NCi1SZWRpc3RyaWJ1dGlvbiBhbmQgdXNlIGluIHNvdXJjZSBhbmQgYmluYXJ5IGZvcm1zLCB3aXRoIG9yIHdpdGhvdXQNCi1tb2RpZmljYXRpb24sIGFyZSBwZXJtaXR0ZWQgcHJvdmlkZWQgdGhhdCB0aGUgZm9sbG93aW5nIGNvbmRpdGlvbnMNCi1hcmUgbWV0Og0KLQ0KLSAgICAqIFJlZGlzdHJpYnV0aW9ucyBvZiBzb3VyY2UgY29kZSBtdXN0IHJldGFpbiB0aGUgYWJvdmUgY29weXJpZ2h0DQotICAgICAgbm90aWNlLCB0aGlzIGxpc3Qgb2YgY29uZGl0aW9ucyBhbmQgdGhlIGZvbGxvd2luZyBkaXNjbGFpbWVyLg0KLSAgICAqIFJlZGlzdHJpYnV0aW9ucyBpbiBiaW5hcnkgZm9ybSBtdXN0IHJlcHJvZHVjZSB0aGUgYWJvdmUgY29weXJpZ2h0DQotICAgICAgbm90aWNlLCB0aGlzIGxpc3Qgb2YgY29uZGl0aW9ucyBhbmQgdGhlIGZvbGxvd2luZyBkaXNjbGFpbWVyIGluDQotICAgICAgdGhlIGRvY3VtZW50YXRpb24gYW5kL29yIG90aGVyIG1hdGVyaWFscyBwcm92aWRlZCB3aXRoIHRoZQ0KLSAgICAgIGRpc3RyaWJ1dGlvbi4NCi0gICAgKiBOZWl0aGVyIHRoZSBuYW1lIG9mIHRoZSBzb2Z0d2FyZSBwcm9kdWN0J3MgY29weXJpZ2h0IG93bmVyIG5vcg0KLSAgICAgIHRoZSBuYW1lcyBvZiBpdHMgY29udHJpYnV0b3JzIG1heSBiZSB1c2VkIHRvIGVuZG9yc2Ugb3IgcHJvbW90ZQ0KLSAgICAgIHByb2R1Y3RzIGRlcml2ZWQgZnJvbSB0aGlzIHNvZnR3YXJlIHdpdGhvdXQgc3BlY2lmaWMgcHJpb3Igd3JpdHRlbg0KLSAgICAgIHBlcm1pc3Npb24uDQotDQotVEhJUyBTT0ZUV0FSRSBJUyBQUk9WSURFRCBCWSBUSEUgQ09QWVJJR0hUIEhPTERFUlMgQU5EIENPTlRSSUJVVE9SUw0KLSJBUyBJUyIgQU5EIEFOWSBFWFBSRVNTIE9SIElNUExJRUQgV0FSUkFOVElFUywgSU5DTFVESU5HLCBCVVQgTk9UDQotTElNSVRFRCBUTywgVEhFIElNUExJRUQgV0FSUkFOVElFUyBPRiBNRVJDSEFOVEFCSUxJVFkgQU5EIEZJVE5FU1MgRk9SDQotQSBQQVJUSUNVTEFSIFBVUlBPU0UgQVJFIERJU0NMQUlNRUQuIElOIE5PIEVWRU5UIFNIQUxMIFRIRSBDT1BZUklHSFQNCi1PV05FUiBPUiBDT05UUklCVVRPUlMgQkUgTElBQkxFIEZPUiBBTlkgRElSRUNULCBJTkRJUkVDVCwgSU5DSURFTlRBTCwNCi1TUEVDSUFMLCBFWEVNUExBUlksIE9SIENPTlNFUVVFTlRJQUwgREFNQUdFUyAoSU5DTFVESU5HLCBCVVQgTk9UIExJTUlURUQNCi1UTywgUFJPQ1VSRU1FTlQgT0YgU1VCU1RJVFVURSBHT09EUyBPUiBTRVJWSUNFUzsgTE9TUyBPRiBVU0UsIERBVEEsIE9SDQotUFJPRklUUzsgT1IgQlVTSU5FU1MgSU5URVJSVVBUSU9OKSBIT1dFVkVSIENBVVNFRCBBTkQgT04gQU5ZIFRIRU9SWSBPRg0KLUxJQUJJTElUWSwgV0hFVEhFUiBJTiBDT05UUkFDVCwgU1RSSUNUIExJQUJJTElUWSwgT1IgVE9SVCAoSU5DTFVESU5HDQotTkVHTElHRU5DRSBPUiBPVEhFUldJU0UpIEFSSVNJTkcgSU4gQU5ZIFdBWSBPVVQgT0YgVEhFIFVTRSBPRiBUSElTDQotU09GVFdBUkUsIEVWRU4gSUYgQURWSVNFRCBPRiBUSEUgUE9TU0lCSUxJVFkgT0YgU1VDSCBEQU1BR0UuDQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvbGljZW5zZS1MR1BMLnR4dCBiL29wZW5nbC90ZXN0cy9hbmdlbGVzL2xpY2Vuc2UtTEdQTC50eHQKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGIxZTNmNWEuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvbGljZW5zZS1MR1BMLnR4dAorKysgL2Rldi9udWxsCkBAIC0xLDUwNCArMCwwIEBACi0JCSAgR05VIExFU1NFUiBHRU5FUkFMIFBVQkxJQyBMSUNFTlNFCi0JCSAgICAgICBWZXJzaW9uIDIuMSwgRmVicnVhcnkgMTk5OQotCi0gQ29weXJpZ2h0IChDKSAxOTkxLCAxOTk5IEZyZWUgU29mdHdhcmUgRm91bmRhdGlvbiwgSW5jLgotICAgICA1OSBUZW1wbGUgUGxhY2UsIFN1aXRlIDMzMCwgQm9zdG9uLCBNQSAgMDIxMTEtMTMwNyAgVVNBCi0gRXZlcnlvbmUgaXMgcGVybWl0dGVkIHRvIGNvcHkgYW5kIGRpc3RyaWJ1dGUgdmVyYmF0aW0gY29waWVzCi0gb2YgdGhpcyBsaWNlbnNlIGRvY3VtZW50LCBidXQgY2hhbmdpbmcgaXQgaXMgbm90IGFsbG93ZWQuCi0KLVtUaGlzIGlzIHRoZSBmaXJzdCByZWxlYXNlZCB2ZXJzaW9uIG9mIHRoZSBMZXNzZXIgR1BMLiAgSXQgYWxzbyBjb3VudHMKLSBhcyB0aGUgc3VjY2Vzc29yIG9mIHRoZSBHTlUgTGlicmFyeSBQdWJsaWMgTGljZW5zZSwgdmVyc2lvbiAyLCBoZW5jZQotIHRoZSB2ZXJzaW9uIG51bWJlciAyLjEuXQotCi0JCQkgICAgUHJlYW1ibGUKLQotICBUaGUgbGljZW5zZXMgZm9yIG1vc3Qgc29mdHdhcmUgYXJlIGRlc2lnbmVkIHRvIHRha2UgYXdheSB5b3VyCi1mcmVlZG9tIHRvIHNoYXJlIGFuZCBjaGFuZ2UgaXQuICBCeSBjb250cmFzdCwgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYwotTGljZW5zZXMgYXJlIGludGVuZGVkIHRvIGd1YXJhbnRlZSB5b3VyIGZyZWVkb20gdG8gc2hhcmUgYW5kIGNoYW5nZQotZnJlZSBzb2Z0d2FyZS0tdG8gbWFrZSBzdXJlIHRoZSBzb2Z0d2FyZSBpcyBmcmVlIGZvciBhbGwgaXRzIHVzZXJzLgotCi0gIFRoaXMgbGljZW5zZSwgdGhlIExlc3NlciBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlLCBhcHBsaWVzIHRvIHNvbWUKLXNwZWNpYWxseSBkZXNpZ25hdGVkIHNvZnR3YXJlIHBhY2thZ2VzLS10eXBpY2FsbHkgbGlicmFyaWVzLS1vZiB0aGUKLUZyZWUgU29mdHdhcmUgRm91bmRhdGlvbiBhbmQgb3RoZXIgYXV0aG9ycyB3aG8gZGVjaWRlIHRvIHVzZSBpdC4gIFlvdQotY2FuIHVzZSBpdCB0b28sIGJ1dCB3ZSBzdWdnZXN0IHlvdSBmaXJzdCB0aGluayBjYXJlZnVsbHkgYWJvdXQgd2hldGhlcgotdGhpcyBsaWNlbnNlIG9yIHRoZSBvcmRpbmFyeSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGlzIHRoZSBiZXR0ZXIKLXN0cmF0ZWd5IHRvIHVzZSBpbiBhbnkgcGFydGljdWxhciBjYXNlLCBiYXNlZCBvbiB0aGUgZXhwbGFuYXRpb25zIGJlbG93LgotCi0gIFdoZW4gd2Ugc3BlYWsgb2YgZnJlZSBzb2Z0d2FyZSwgd2UgYXJlIHJlZmVycmluZyB0byBmcmVlZG9tIG9mIHVzZSwKLW5vdCBwcmljZS4gIE91ciBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlcyBhcmUgZGVzaWduZWQgdG8gbWFrZSBzdXJlIHRoYXQKLXlvdSBoYXZlIHRoZSBmcmVlZG9tIHRvIGRpc3RyaWJ1dGUgY29waWVzIG9mIGZyZWUgc29mdHdhcmUgKGFuZCBjaGFyZ2UKLWZvciB0aGlzIHNlcnZpY2UgaWYgeW91IHdpc2gpOyB0aGF0IHlvdSByZWNlaXZlIHNvdXJjZSBjb2RlIG9yIGNhbiBnZXQKLWl0IGlmIHlvdSB3YW50IGl0OyB0aGF0IHlvdSBjYW4gY2hhbmdlIHRoZSBzb2Z0d2FyZSBhbmQgdXNlIHBpZWNlcyBvZgotaXQgaW4gbmV3IGZyZWUgcHJvZ3JhbXM7IGFuZCB0aGF0IHlvdSBhcmUgaW5mb3JtZWQgdGhhdCB5b3UgY2FuIGRvCi10aGVzZSB0aGluZ3MuCi0KLSAgVG8gcHJvdGVjdCB5b3VyIHJpZ2h0cywgd2UgbmVlZCB0byBtYWtlIHJlc3RyaWN0aW9ucyB0aGF0IGZvcmJpZAotZGlzdHJpYnV0b3JzIHRvIGRlbnkgeW91IHRoZXNlIHJpZ2h0cyBvciB0byBhc2sgeW91IHRvIHN1cnJlbmRlciB0aGVzZQotcmlnaHRzLiAgVGhlc2UgcmVzdHJpY3Rpb25zIHRyYW5zbGF0ZSB0byBjZXJ0YWluIHJlc3BvbnNpYmlsaXRpZXMgZm9yCi15b3UgaWYgeW91IGRpc3RyaWJ1dGUgY29waWVzIG9mIHRoZSBsaWJyYXJ5IG9yIGlmIHlvdSBtb2RpZnkgaXQuCi0KLSAgRm9yIGV4YW1wbGUsIGlmIHlvdSBkaXN0cmlidXRlIGNvcGllcyBvZiB0aGUgbGlicmFyeSwgd2hldGhlciBncmF0aXMKLW9yIGZvciBhIGZlZSwgeW91IG11c3QgZ2l2ZSB0aGUgcmVjaXBpZW50cyBhbGwgdGhlIHJpZ2h0cyB0aGF0IHdlIGdhdmUKLXlvdS4gIFlvdSBtdXN0IG1ha2Ugc3VyZSB0aGF0IHRoZXksIHRvbywgcmVjZWl2ZSBvciBjYW4gZ2V0IHRoZSBzb3VyY2UKLWNvZGUuICBJZiB5b3UgbGluayBvdGhlciBjb2RlIHdpdGggdGhlIGxpYnJhcnksIHlvdSBtdXN0IHByb3ZpZGUKLWNvbXBsZXRlIG9iamVjdCBmaWxlcyB0byB0aGUgcmVjaXBpZW50cywgc28gdGhhdCB0aGV5IGNhbiByZWxpbmsgdGhlbQotd2l0aCB0aGUgbGlicmFyeSBhZnRlciBtYWtpbmcgY2hhbmdlcyB0byB0aGUgbGlicmFyeSBhbmQgcmVjb21waWxpbmcKLWl0LiAgQW5kIHlvdSBtdXN0IHNob3cgdGhlbSB0aGVzZSB0ZXJtcyBzbyB0aGV5IGtub3cgdGhlaXIgcmlnaHRzLgotCi0gIFdlIHByb3RlY3QgeW91ciByaWdodHMgd2l0aCBhIHR3by1zdGVwIG1ldGhvZDogKDEpIHdlIGNvcHlyaWdodCB0aGUKLWxpYnJhcnksIGFuZCAoMikgd2Ugb2ZmZXIgeW91IHRoaXMgbGljZW5zZSwgd2hpY2ggZ2l2ZXMgeW91IGxlZ2FsCi1wZXJtaXNzaW9uIHRvIGNvcHksIGRpc3RyaWJ1dGUgYW5kL29yIG1vZGlmeSB0aGUgbGlicmFyeS4KLQotICBUbyBwcm90ZWN0IGVhY2ggZGlzdHJpYnV0b3IsIHdlIHdhbnQgdG8gbWFrZSBpdCB2ZXJ5IGNsZWFyIHRoYXQKLXRoZXJlIGlzIG5vIHdhcnJhbnR5IGZvciB0aGUgZnJlZSBsaWJyYXJ5LiAgQWxzbywgaWYgdGhlIGxpYnJhcnkgaXMKLW1vZGlmaWVkIGJ5IHNvbWVvbmUgZWxzZSBhbmQgcGFzc2VkIG9uLCB0aGUgcmVjaXBpZW50cyBzaG91bGQga25vdwotdGhhdCB3aGF0IHRoZXkgaGF2ZSBpcyBub3QgdGhlIG9yaWdpbmFsIHZlcnNpb24sIHNvIHRoYXQgdGhlIG9yaWdpbmFsCi1hdXRob3IncyByZXB1dGF0aW9uIHdpbGwgbm90IGJlIGFmZmVjdGVkIGJ5IHByb2JsZW1zIHRoYXQgbWlnaHQgYmUKLWludHJvZHVjZWQgYnkgb3RoZXJzLgotDAotICBGaW5hbGx5LCBzb2Z0d2FyZSBwYXRlbnRzIHBvc2UgYSBjb25zdGFudCB0aHJlYXQgdG8gdGhlIGV4aXN0ZW5jZSBvZgotYW55IGZyZWUgcHJvZ3JhbS4gIFdlIHdpc2ggdG8gbWFrZSBzdXJlIHRoYXQgYSBjb21wYW55IGNhbm5vdAotZWZmZWN0aXZlbHkgcmVzdHJpY3QgdGhlIHVzZXJzIG9mIGEgZnJlZSBwcm9ncmFtIGJ5IG9idGFpbmluZyBhCi1yZXN0cmljdGl2ZSBsaWNlbnNlIGZyb20gYSBwYXRlbnQgaG9sZGVyLiAgVGhlcmVmb3JlLCB3ZSBpbnNpc3QgdGhhdAotYW55IHBhdGVudCBsaWNlbnNlIG9idGFpbmVkIGZvciBhIHZlcnNpb24gb2YgdGhlIGxpYnJhcnkgbXVzdCBiZQotY29uc2lzdGVudCB3aXRoIHRoZSBmdWxsIGZyZWVkb20gb2YgdXNlIHNwZWNpZmllZCBpbiB0aGlzIGxpY2Vuc2UuCi0KLSAgTW9zdCBHTlUgc29mdHdhcmUsIGluY2x1ZGluZyBzb21lIGxpYnJhcmllcywgaXMgY292ZXJlZCBieSB0aGUKLW9yZGluYXJ5IEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlLiAgVGhpcyBsaWNlbnNlLCB0aGUgR05VIExlc3NlcgotR2VuZXJhbCBQdWJsaWMgTGljZW5zZSwgYXBwbGllcyB0byBjZXJ0YWluIGRlc2lnbmF0ZWQgbGlicmFyaWVzLCBhbmQKLWlzIHF1aXRlIGRpZmZlcmVudCBmcm9tIHRoZSBvcmRpbmFyeSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlLiAgV2UgdXNlCi10aGlzIGxpY2Vuc2UgZm9yIGNlcnRhaW4gbGlicmFyaWVzIGluIG9yZGVyIHRvIHBlcm1pdCBsaW5raW5nIHRob3NlCi1saWJyYXJpZXMgaW50byBub24tZnJlZSBwcm9ncmFtcy4KLQotICBXaGVuIGEgcHJvZ3JhbSBpcyBsaW5rZWQgd2l0aCBhIGxpYnJhcnksIHdoZXRoZXIgc3RhdGljYWxseSBvciB1c2luZwotYSBzaGFyZWQgbGlicmFyeSwgdGhlIGNvbWJpbmF0aW9uIG9mIHRoZSB0d28gaXMgbGVnYWxseSBzcGVha2luZyBhCi1jb21iaW5lZCB3b3JrLCBhIGRlcml2YXRpdmUgb2YgdGhlIG9yaWdpbmFsIGxpYnJhcnkuICBUaGUgb3JkaW5hcnkKLUdlbmVyYWwgUHVibGljIExpY2Vuc2UgdGhlcmVmb3JlIHBlcm1pdHMgc3VjaCBsaW5raW5nIG9ubHkgaWYgdGhlCi1lbnRpcmUgY29tYmluYXRpb24gZml0cyBpdHMgY3JpdGVyaWEgb2YgZnJlZWRvbS4gIFRoZSBMZXNzZXIgR2VuZXJhbAotUHVibGljIExpY2Vuc2UgcGVybWl0cyBtb3JlIGxheCBjcml0ZXJpYSBmb3IgbGlua2luZyBvdGhlciBjb2RlIHdpdGgKLXRoZSBsaWJyYXJ5LgotCi0gIFdlIGNhbGwgdGhpcyBsaWNlbnNlIHRoZSAiTGVzc2VyIiBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGJlY2F1c2UgaXQKLWRvZXMgTGVzcyB0byBwcm90ZWN0IHRoZSB1c2VyJ3MgZnJlZWRvbSB0aGFuIHRoZSBvcmRpbmFyeSBHZW5lcmFsCi1QdWJsaWMgTGljZW5zZS4gIEl0IGFsc28gcHJvdmlkZXMgb3RoZXIgZnJlZSBzb2Z0d2FyZSBkZXZlbG9wZXJzIExlc3MKLW9mIGFuIGFkdmFudGFnZSBvdmVyIGNvbXBldGluZyBub24tZnJlZSBwcm9ncmFtcy4gIFRoZXNlIGRpc2FkdmFudGFnZXMKLWFyZSB0aGUgcmVhc29uIHdlIHVzZSB0aGUgb3JkaW5hcnkgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmb3IgbWFueQotbGlicmFyaWVzLiAgSG93ZXZlciwgdGhlIExlc3NlciBsaWNlbnNlIHByb3ZpZGVzIGFkdmFudGFnZXMgaW4gY2VydGFpbgotc3BlY2lhbCBjaXJjdW1zdGFuY2VzLgotCi0gIEZvciBleGFtcGxlLCBvbiByYXJlIG9jY2FzaW9ucywgdGhlcmUgbWF5IGJlIGEgc3BlY2lhbCBuZWVkIHRvCi1lbmNvdXJhZ2UgdGhlIHdpZGVzdCBwb3NzaWJsZSB1c2Ugb2YgYSBjZXJ0YWluIGxpYnJhcnksIHNvIHRoYXQgaXQgYmVjb21lcwotYSBkZS1mYWN0byBzdGFuZGFyZC4gIFRvIGFjaGlldmUgdGhpcywgbm9uLWZyZWUgcHJvZ3JhbXMgbXVzdCBiZQotYWxsb3dlZCB0byB1c2UgdGhlIGxpYnJhcnkuICBBIG1vcmUgZnJlcXVlbnQgY2FzZSBpcyB0aGF0IGEgZnJlZQotbGlicmFyeSBkb2VzIHRoZSBzYW1lIGpvYiBhcyB3aWRlbHkgdXNlZCBub24tZnJlZSBsaWJyYXJpZXMuICBJbiB0aGlzCi1jYXNlLCB0aGVyZSBpcyBsaXR0bGUgdG8gZ2FpbiBieSBsaW1pdGluZyB0aGUgZnJlZSBsaWJyYXJ5IHRvIGZyZWUKLXNvZnR3YXJlIG9ubHksIHNvIHdlIHVzZSB0aGUgTGVzc2VyIEdlbmVyYWwgUHVibGljIExpY2Vuc2UuCi0KLSAgSW4gb3RoZXIgY2FzZXMsIHBlcm1pc3Npb24gdG8gdXNlIGEgcGFydGljdWxhciBsaWJyYXJ5IGluIG5vbi1mcmVlCi1wcm9ncmFtcyBlbmFibGVzIGEgZ3JlYXRlciBudW1iZXIgb2YgcGVvcGxlIHRvIHVzZSBhIGxhcmdlIGJvZHkgb2YKLWZyZWUgc29mdHdhcmUuICBGb3IgZXhhbXBsZSwgcGVybWlzc2lvbiB0byB1c2UgdGhlIEdOVSBDIExpYnJhcnkgaW4KLW5vbi1mcmVlIHByb2dyYW1zIGVuYWJsZXMgbWFueSBtb3JlIHBlb3BsZSB0byB1c2UgdGhlIHdob2xlIEdOVQotb3BlcmF0aW5nIHN5c3RlbSwgYXMgd2VsbCBhcyBpdHMgdmFyaWFudCwgdGhlIEdOVS9MaW51eCBvcGVyYXRpbmcKLXN5c3RlbS4KLQotICBBbHRob3VnaCB0aGUgTGVzc2VyIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgaXMgTGVzcyBwcm90ZWN0aXZlIG9mIHRoZQotdXNlcnMnIGZyZWVkb20sIGl0IGRvZXMgZW5zdXJlIHRoYXQgdGhlIHVzZXIgb2YgYSBwcm9ncmFtIHRoYXQgaXMKLWxpbmtlZCB3aXRoIHRoZSBMaWJyYXJ5IGhhcyB0aGUgZnJlZWRvbSBhbmQgdGhlIHdoZXJld2l0aGFsIHRvIHJ1bgotdGhhdCBwcm9ncmFtIHVzaW5nIGEgbW9kaWZpZWQgdmVyc2lvbiBvZiB0aGUgTGlicmFyeS4KLQotICBUaGUgcHJlY2lzZSB0ZXJtcyBhbmQgY29uZGl0aW9ucyBmb3IgY29weWluZywgZGlzdHJpYnV0aW9uIGFuZAotbW9kaWZpY2F0aW9uIGZvbGxvdy4gIFBheSBjbG9zZSBhdHRlbnRpb24gdG8gdGhlIGRpZmZlcmVuY2UgYmV0d2VlbiBhCi0id29yayBiYXNlZCBvbiB0aGUgbGlicmFyeSIgYW5kIGEgIndvcmsgdGhhdCB1c2VzIHRoZSBsaWJyYXJ5Ii4gIFRoZQotZm9ybWVyIGNvbnRhaW5zIGNvZGUgZGVyaXZlZCBmcm9tIHRoZSBsaWJyYXJ5LCB3aGVyZWFzIHRoZSBsYXR0ZXIgbXVzdAotYmUgY29tYmluZWQgd2l0aCB0aGUgbGlicmFyeSBpbiBvcmRlciB0byBydW4uCi0MCi0JCSAgR05VIExFU1NFUiBHRU5FUkFMIFBVQkxJQyBMSUNFTlNFCi0gICBURVJNUyBBTkQgQ09ORElUSU9OUyBGT1IgQ09QWUlORywgRElTVFJJQlVUSU9OIEFORCBNT0RJRklDQVRJT04KLQotICAwLiBUaGlzIExpY2Vuc2UgQWdyZWVtZW50IGFwcGxpZXMgdG8gYW55IHNvZnR3YXJlIGxpYnJhcnkgb3Igb3RoZXIKLXByb2dyYW0gd2hpY2ggY29udGFpbnMgYSBub3RpY2UgcGxhY2VkIGJ5IHRoZSBjb3B5cmlnaHQgaG9sZGVyIG9yCi1vdGhlciBhdXRob3JpemVkIHBhcnR5IHNheWluZyBpdCBtYXkgYmUgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIHRlcm1zIG9mCi10aGlzIExlc3NlciBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIChhbHNvIGNhbGxlZCAidGhpcyBMaWNlbnNlIikuCi1FYWNoIGxpY2Vuc2VlIGlzIGFkZHJlc3NlZCBhcyAieW91Ii4KLQotICBBICJsaWJyYXJ5IiBtZWFucyBhIGNvbGxlY3Rpb24gb2Ygc29mdHdhcmUgZnVuY3Rpb25zIGFuZC9vciBkYXRhCi1wcmVwYXJlZCBzbyBhcyB0byBiZSBjb252ZW5pZW50bHkgbGlua2VkIHdpdGggYXBwbGljYXRpb24gcHJvZ3JhbXMKLSh3aGljaCB1c2Ugc29tZSBvZiB0aG9zZSBmdW5jdGlvbnMgYW5kIGRhdGEpIHRvIGZvcm0gZXhlY3V0YWJsZXMuCi0KLSAgVGhlICJMaWJyYXJ5IiwgYmVsb3csIHJlZmVycyB0byBhbnkgc3VjaCBzb2Z0d2FyZSBsaWJyYXJ5IG9yIHdvcmsKLXdoaWNoIGhhcyBiZWVuIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZXNlIHRlcm1zLiAgQSAid29yayBiYXNlZCBvbiB0aGUKLUxpYnJhcnkiIG1lYW5zIGVpdGhlciB0aGUgTGlicmFyeSBvciBhbnkgZGVyaXZhdGl2ZSB3b3JrIHVuZGVyCi1jb3B5cmlnaHQgbGF3OiB0aGF0IGlzIHRvIHNheSwgYSB3b3JrIGNvbnRhaW5pbmcgdGhlIExpYnJhcnkgb3IgYQotcG9ydGlvbiBvZiBpdCwgZWl0aGVyIHZlcmJhdGltIG9yIHdpdGggbW9kaWZpY2F0aW9ucyBhbmQvb3IgdHJhbnNsYXRlZAotc3RyYWlnaHRmb3J3YXJkbHkgaW50byBhbm90aGVyIGxhbmd1YWdlLiAgKEhlcmVpbmFmdGVyLCB0cmFuc2xhdGlvbiBpcwotaW5jbHVkZWQgd2l0aG91dCBsaW1pdGF0aW9uIGluIHRoZSB0ZXJtICJtb2RpZmljYXRpb24iLikKLQotICAiU291cmNlIGNvZGUiIGZvciBhIHdvcmsgbWVhbnMgdGhlIHByZWZlcnJlZCBmb3JtIG9mIHRoZSB3b3JrIGZvcgotbWFraW5nIG1vZGlmaWNhdGlvbnMgdG8gaXQuICBGb3IgYSBsaWJyYXJ5LCBjb21wbGV0ZSBzb3VyY2UgY29kZSBtZWFucwotYWxsIHRoZSBzb3VyY2UgY29kZSBmb3IgYWxsIG1vZHVsZXMgaXQgY29udGFpbnMsIHBsdXMgYW55IGFzc29jaWF0ZWQKLWludGVyZmFjZSBkZWZpbml0aW9uIGZpbGVzLCBwbHVzIHRoZSBzY3JpcHRzIHVzZWQgdG8gY29udHJvbCBjb21waWxhdGlvbgotYW5kIGluc3RhbGxhdGlvbiBvZiB0aGUgbGlicmFyeS4KLQotICBBY3Rpdml0aWVzIG90aGVyIHRoYW4gY29weWluZywgZGlzdHJpYnV0aW9uIGFuZCBtb2RpZmljYXRpb24gYXJlIG5vdAotY292ZXJlZCBieSB0aGlzIExpY2Vuc2U7IHRoZXkgYXJlIG91dHNpZGUgaXRzIHNjb3BlLiAgVGhlIGFjdCBvZgotcnVubmluZyBhIHByb2dyYW0gdXNpbmcgdGhlIExpYnJhcnkgaXMgbm90IHJlc3RyaWN0ZWQsIGFuZCBvdXRwdXQgZnJvbQotc3VjaCBhIHByb2dyYW0gaXMgY292ZXJlZCBvbmx5IGlmIGl0cyBjb250ZW50cyBjb25zdGl0dXRlIGEgd29yayBiYXNlZAotb24gdGhlIExpYnJhcnkgKGluZGVwZW5kZW50IG9mIHRoZSB1c2Ugb2YgdGhlIExpYnJhcnkgaW4gYSB0b29sIGZvcgotd3JpdGluZyBpdCkuICBXaGV0aGVyIHRoYXQgaXMgdHJ1ZSBkZXBlbmRzIG9uIHdoYXQgdGhlIExpYnJhcnkgZG9lcwotYW5kIHdoYXQgdGhlIHByb2dyYW0gdGhhdCB1c2VzIHRoZSBMaWJyYXJ5IGRvZXMuCi0gIAotICAxLiBZb3UgbWF5IGNvcHkgYW5kIGRpc3RyaWJ1dGUgdmVyYmF0aW0gY29waWVzIG9mIHRoZSBMaWJyYXJ5J3MKLWNvbXBsZXRlIHNvdXJjZSBjb2RlIGFzIHlvdSByZWNlaXZlIGl0LCBpbiBhbnkgbWVkaXVtLCBwcm92aWRlZCB0aGF0Ci15b3UgY29uc3BpY3VvdXNseSBhbmQgYXBwcm9wcmlhdGVseSBwdWJsaXNoIG9uIGVhY2ggY29weSBhbgotYXBwcm9wcmlhdGUgY29weXJpZ2h0IG5vdGljZSBhbmQgZGlzY2xhaW1lciBvZiB3YXJyYW50eTsga2VlcCBpbnRhY3QKLWFsbCB0aGUgbm90aWNlcyB0aGF0IHJlZmVyIHRvIHRoaXMgTGljZW5zZSBhbmQgdG8gdGhlIGFic2VuY2Ugb2YgYW55Ci13YXJyYW50eTsgYW5kIGRpc3RyaWJ1dGUgYSBjb3B5IG9mIHRoaXMgTGljZW5zZSBhbG9uZyB3aXRoIHRoZQotTGlicmFyeS4KLQotICBZb3UgbWF5IGNoYXJnZSBhIGZlZSBmb3IgdGhlIHBoeXNpY2FsIGFjdCBvZiB0cmFuc2ZlcnJpbmcgYSBjb3B5LAotYW5kIHlvdSBtYXkgYXQgeW91ciBvcHRpb24gb2ZmZXIgd2FycmFudHkgcHJvdGVjdGlvbiBpbiBleGNoYW5nZSBmb3IgYQotZmVlLgotDAotICAyLiBZb3UgbWF5IG1vZGlmeSB5b3VyIGNvcHkgb3IgY29waWVzIG9mIHRoZSBMaWJyYXJ5IG9yIGFueSBwb3J0aW9uCi1vZiBpdCwgdGh1cyBmb3JtaW5nIGEgd29yayBiYXNlZCBvbiB0aGUgTGlicmFyeSwgYW5kIGNvcHkgYW5kCi1kaXN0cmlidXRlIHN1Y2ggbW9kaWZpY2F0aW9ucyBvciB3b3JrIHVuZGVyIHRoZSB0ZXJtcyBvZiBTZWN0aW9uIDEKLWFib3ZlLCBwcm92aWRlZCB0aGF0IHlvdSBhbHNvIG1lZXQgYWxsIG9mIHRoZXNlIGNvbmRpdGlvbnM6Ci0KLSAgICBhKSBUaGUgbW9kaWZpZWQgd29yayBtdXN0IGl0c2VsZiBiZSBhIHNvZnR3YXJlIGxpYnJhcnkuCi0KLSAgICBiKSBZb3UgbXVzdCBjYXVzZSB0aGUgZmlsZXMgbW9kaWZpZWQgdG8gY2FycnkgcHJvbWluZW50IG5vdGljZXMKLSAgICBzdGF0aW5nIHRoYXQgeW91IGNoYW5nZWQgdGhlIGZpbGVzIGFuZCB0aGUgZGF0ZSBvZiBhbnkgY2hhbmdlLgotCi0gICAgYykgWW91IG11c3QgY2F1c2UgdGhlIHdob2xlIG9mIHRoZSB3b3JrIHRvIGJlIGxpY2Vuc2VkIGF0IG5vCi0gICAgY2hhcmdlIHRvIGFsbCB0aGlyZCBwYXJ0aWVzIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGlzIExpY2Vuc2UuCi0KLSAgICBkKSBJZiBhIGZhY2lsaXR5IGluIHRoZSBtb2RpZmllZCBMaWJyYXJ5IHJlZmVycyB0byBhIGZ1bmN0aW9uIG9yIGEKLSAgICB0YWJsZSBvZiBkYXRhIHRvIGJlIHN1cHBsaWVkIGJ5IGFuIGFwcGxpY2F0aW9uIHByb2dyYW0gdGhhdCB1c2VzCi0gICAgdGhlIGZhY2lsaXR5LCBvdGhlciB0aGFuIGFzIGFuIGFyZ3VtZW50IHBhc3NlZCB3aGVuIHRoZSBmYWNpbGl0eQotICAgIGlzIGludm9rZWQsIHRoZW4geW91IG11c3QgbWFrZSBhIGdvb2QgZmFpdGggZWZmb3J0IHRvIGVuc3VyZSB0aGF0LAotICAgIGluIHRoZSBldmVudCBhbiBhcHBsaWNhdGlvbiBkb2VzIG5vdCBzdXBwbHkgc3VjaCBmdW5jdGlvbiBvcgotICAgIHRhYmxlLCB0aGUgZmFjaWxpdHkgc3RpbGwgb3BlcmF0ZXMsIGFuZCBwZXJmb3JtcyB3aGF0ZXZlciBwYXJ0IG9mCi0gICAgaXRzIHB1cnBvc2UgcmVtYWlucyBtZWFuaW5nZnVsLgotCi0gICAgKEZvciBleGFtcGxlLCBhIGZ1bmN0aW9uIGluIGEgbGlicmFyeSB0byBjb21wdXRlIHNxdWFyZSByb290cyBoYXMKLSAgICBhIHB1cnBvc2UgdGhhdCBpcyBlbnRpcmVseSB3ZWxsLWRlZmluZWQgaW5kZXBlbmRlbnQgb2YgdGhlCi0gICAgYXBwbGljYXRpb24uICBUaGVyZWZvcmUsIFN1YnNlY3Rpb24gMmQgcmVxdWlyZXMgdGhhdCBhbnkKLSAgICBhcHBsaWNhdGlvbi1zdXBwbGllZCBmdW5jdGlvbiBvciB0YWJsZSB1c2VkIGJ5IHRoaXMgZnVuY3Rpb24gbXVzdAotICAgIGJlIG9wdGlvbmFsOiBpZiB0aGUgYXBwbGljYXRpb24gZG9lcyBub3Qgc3VwcGx5IGl0LCB0aGUgc3F1YXJlCi0gICAgcm9vdCBmdW5jdGlvbiBtdXN0IHN0aWxsIGNvbXB1dGUgc3F1YXJlIHJvb3RzLikKLQotVGhlc2UgcmVxdWlyZW1lbnRzIGFwcGx5IHRvIHRoZSBtb2RpZmllZCB3b3JrIGFzIGEgd2hvbGUuICBJZgotaWRlbnRpZmlhYmxlIHNlY3Rpb25zIG9mIHRoYXQgd29yayBhcmUgbm90IGRlcml2ZWQgZnJvbSB0aGUgTGlicmFyeSwKLWFuZCBjYW4gYmUgcmVhc29uYWJseSBjb25zaWRlcmVkIGluZGVwZW5kZW50IGFuZCBzZXBhcmF0ZSB3b3JrcyBpbgotdGhlbXNlbHZlcywgdGhlbiB0aGlzIExpY2Vuc2UsIGFuZCBpdHMgdGVybXMsIGRvIG5vdCBhcHBseSB0byB0aG9zZQotc2VjdGlvbnMgd2hlbiB5b3UgZGlzdHJpYnV0ZSB0aGVtIGFzIHNlcGFyYXRlIHdvcmtzLiAgQnV0IHdoZW4geW91Ci1kaXN0cmlidXRlIHRoZSBzYW1lIHNlY3Rpb25zIGFzIHBhcnQgb2YgYSB3aG9sZSB3aGljaCBpcyBhIHdvcmsgYmFzZWQKLW9uIHRoZSBMaWJyYXJ5LCB0aGUgZGlzdHJpYnV0aW9uIG9mIHRoZSB3aG9sZSBtdXN0IGJlIG9uIHRoZSB0ZXJtcyBvZgotdGhpcyBMaWNlbnNlLCB3aG9zZSBwZXJtaXNzaW9ucyBmb3Igb3RoZXIgbGljZW5zZWVzIGV4dGVuZCB0byB0aGUKLWVudGlyZSB3aG9sZSwgYW5kIHRodXMgdG8gZWFjaCBhbmQgZXZlcnkgcGFydCByZWdhcmRsZXNzIG9mIHdobyB3cm90ZQotaXQuCi0KLVRodXMsIGl0IGlzIG5vdCB0aGUgaW50ZW50IG9mIHRoaXMgc2VjdGlvbiB0byBjbGFpbSByaWdodHMgb3IgY29udGVzdAoteW91ciByaWdodHMgdG8gd29yayB3cml0dGVuIGVudGlyZWx5IGJ5IHlvdTsgcmF0aGVyLCB0aGUgaW50ZW50IGlzIHRvCi1leGVyY2lzZSB0aGUgcmlnaHQgdG8gY29udHJvbCB0aGUgZGlzdHJpYnV0aW9uIG9mIGRlcml2YXRpdmUgb3IKLWNvbGxlY3RpdmUgd29ya3MgYmFzZWQgb24gdGhlIExpYnJhcnkuCi0KLUluIGFkZGl0aW9uLCBtZXJlIGFnZ3JlZ2F0aW9uIG9mIGFub3RoZXIgd29yayBub3QgYmFzZWQgb24gdGhlIExpYnJhcnkKLXdpdGggdGhlIExpYnJhcnkgKG9yIHdpdGggYSB3b3JrIGJhc2VkIG9uIHRoZSBMaWJyYXJ5KSBvbiBhIHZvbHVtZSBvZgotYSBzdG9yYWdlIG9yIGRpc3RyaWJ1dGlvbiBtZWRpdW0gZG9lcyBub3QgYnJpbmcgdGhlIG90aGVyIHdvcmsgdW5kZXIKLXRoZSBzY29wZSBvZiB0aGlzIExpY2Vuc2UuCi0KLSAgMy4gWW91IG1heSBvcHQgdG8gYXBwbHkgdGhlIHRlcm1zIG9mIHRoZSBvcmRpbmFyeSBHTlUgR2VuZXJhbCBQdWJsaWMKLUxpY2Vuc2UgaW5zdGVhZCBvZiB0aGlzIExpY2Vuc2UgdG8gYSBnaXZlbiBjb3B5IG9mIHRoZSBMaWJyYXJ5LiAgVG8gZG8KLXRoaXMsIHlvdSBtdXN0IGFsdGVyIGFsbCB0aGUgbm90aWNlcyB0aGF0IHJlZmVyIHRvIHRoaXMgTGljZW5zZSwgc28KLXRoYXQgdGhleSByZWZlciB0byB0aGUgb3JkaW5hcnkgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UsIHZlcnNpb24gMiwKLWluc3RlYWQgb2YgdG8gdGhpcyBMaWNlbnNlLiAgKElmIGEgbmV3ZXIgdmVyc2lvbiB0aGFuIHZlcnNpb24gMiBvZiB0aGUKLW9yZGluYXJ5IEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGhhcyBhcHBlYXJlZCwgdGhlbiB5b3UgY2FuIHNwZWNpZnkKLXRoYXQgdmVyc2lvbiBpbnN0ZWFkIGlmIHlvdSB3aXNoLikgIERvIG5vdCBtYWtlIGFueSBvdGhlciBjaGFuZ2UgaW4KLXRoZXNlIG5vdGljZXMuCi0MCi0gIE9uY2UgdGhpcyBjaGFuZ2UgaXMgbWFkZSBpbiBhIGdpdmVuIGNvcHksIGl0IGlzIGlycmV2ZXJzaWJsZSBmb3IKLXRoYXQgY29weSwgc28gdGhlIG9yZGluYXJ5IEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFwcGxpZXMgdG8gYWxsCi1zdWJzZXF1ZW50IGNvcGllcyBhbmQgZGVyaXZhdGl2ZSB3b3JrcyBtYWRlIGZyb20gdGhhdCBjb3B5LgotCi0gIFRoaXMgb3B0aW9uIGlzIHVzZWZ1bCB3aGVuIHlvdSB3aXNoIHRvIGNvcHkgcGFydCBvZiB0aGUgY29kZSBvZgotdGhlIExpYnJhcnkgaW50byBhIHByb2dyYW0gdGhhdCBpcyBub3QgYSBsaWJyYXJ5LgotCi0gIDQuIFlvdSBtYXkgY29weSBhbmQgZGlzdHJpYnV0ZSB0aGUgTGlicmFyeSAob3IgYSBwb3J0aW9uIG9yCi1kZXJpdmF0aXZlIG9mIGl0LCB1bmRlciBTZWN0aW9uIDIpIGluIG9iamVjdCBjb2RlIG9yIGV4ZWN1dGFibGUgZm9ybQotdW5kZXIgdGhlIHRlcm1zIG9mIFNlY3Rpb25zIDEgYW5kIDIgYWJvdmUgcHJvdmlkZWQgdGhhdCB5b3UgYWNjb21wYW55Ci1pdCB3aXRoIHRoZSBjb21wbGV0ZSBjb3JyZXNwb25kaW5nIG1hY2hpbmUtcmVhZGFibGUgc291cmNlIGNvZGUsIHdoaWNoCi1tdXN0IGJlIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSB0ZXJtcyBvZiBTZWN0aW9ucyAxIGFuZCAyIGFib3ZlIG9uIGEKLW1lZGl1bSBjdXN0b21hcmlseSB1c2VkIGZvciBzb2Z0d2FyZSBpbnRlcmNoYW5nZS4KLQotICBJZiBkaXN0cmlidXRpb24gb2Ygb2JqZWN0IGNvZGUgaXMgbWFkZSBieSBvZmZlcmluZyBhY2Nlc3MgdG8gY29weQotZnJvbSBhIGRlc2lnbmF0ZWQgcGxhY2UsIHRoZW4gb2ZmZXJpbmcgZXF1aXZhbGVudCBhY2Nlc3MgdG8gY29weSB0aGUKLXNvdXJjZSBjb2RlIGZyb20gdGhlIHNhbWUgcGxhY2Ugc2F0aXNmaWVzIHRoZSByZXF1aXJlbWVudCB0bwotZGlzdHJpYnV0ZSB0aGUgc291cmNlIGNvZGUsIGV2ZW4gdGhvdWdoIHRoaXJkIHBhcnRpZXMgYXJlIG5vdAotY29tcGVsbGVkIHRvIGNvcHkgdGhlIHNvdXJjZSBhbG9uZyB3aXRoIHRoZSBvYmplY3QgY29kZS4KLQotICA1LiBBIHByb2dyYW0gdGhhdCBjb250YWlucyBubyBkZXJpdmF0aXZlIG9mIGFueSBwb3J0aW9uIG9mIHRoZQotTGlicmFyeSwgYnV0IGlzIGRlc2lnbmVkIHRvIHdvcmsgd2l0aCB0aGUgTGlicmFyeSBieSBiZWluZyBjb21waWxlZCBvcgotbGlua2VkIHdpdGggaXQsIGlzIGNhbGxlZCBhICJ3b3JrIHRoYXQgdXNlcyB0aGUgTGlicmFyeSIuICBTdWNoIGEKLXdvcmssIGluIGlzb2xhdGlvbiwgaXMgbm90IGEgZGVyaXZhdGl2ZSB3b3JrIG9mIHRoZSBMaWJyYXJ5LCBhbmQKLXRoZXJlZm9yZSBmYWxscyBvdXRzaWRlIHRoZSBzY29wZSBvZiB0aGlzIExpY2Vuc2UuCi0KLSAgSG93ZXZlciwgbGlua2luZyBhICJ3b3JrIHRoYXQgdXNlcyB0aGUgTGlicmFyeSIgd2l0aCB0aGUgTGlicmFyeQotY3JlYXRlcyBhbiBleGVjdXRhYmxlIHRoYXQgaXMgYSBkZXJpdmF0aXZlIG9mIHRoZSBMaWJyYXJ5IChiZWNhdXNlIGl0Ci1jb250YWlucyBwb3J0aW9ucyBvZiB0aGUgTGlicmFyeSksIHJhdGhlciB0aGFuIGEgIndvcmsgdGhhdCB1c2VzIHRoZQotbGlicmFyeSIuICBUaGUgZXhlY3V0YWJsZSBpcyB0aGVyZWZvcmUgY292ZXJlZCBieSB0aGlzIExpY2Vuc2UuCi1TZWN0aW9uIDYgc3RhdGVzIHRlcm1zIGZvciBkaXN0cmlidXRpb24gb2Ygc3VjaCBleGVjdXRhYmxlcy4KLQotICBXaGVuIGEgIndvcmsgdGhhdCB1c2VzIHRoZSBMaWJyYXJ5IiB1c2VzIG1hdGVyaWFsIGZyb20gYSBoZWFkZXIgZmlsZQotdGhhdCBpcyBwYXJ0IG9mIHRoZSBMaWJyYXJ5LCB0aGUgb2JqZWN0IGNvZGUgZm9yIHRoZSB3b3JrIG1heSBiZSBhCi1kZXJpdmF0aXZlIHdvcmsgb2YgdGhlIExpYnJhcnkgZXZlbiB0aG91Z2ggdGhlIHNvdXJjZSBjb2RlIGlzIG5vdC4KLVdoZXRoZXIgdGhpcyBpcyB0cnVlIGlzIGVzcGVjaWFsbHkgc2lnbmlmaWNhbnQgaWYgdGhlIHdvcmsgY2FuIGJlCi1saW5rZWQgd2l0aG91dCB0aGUgTGlicmFyeSwgb3IgaWYgdGhlIHdvcmsgaXMgaXRzZWxmIGEgbGlicmFyeS4gIFRoZQotdGhyZXNob2xkIGZvciB0aGlzIHRvIGJlIHRydWUgaXMgbm90IHByZWNpc2VseSBkZWZpbmVkIGJ5IGxhdy4KLQotICBJZiBzdWNoIGFuIG9iamVjdCBmaWxlIHVzZXMgb25seSBudW1lcmljYWwgcGFyYW1ldGVycywgZGF0YQotc3RydWN0dXJlIGxheW91dHMgYW5kIGFjY2Vzc29ycywgYW5kIHNtYWxsIG1hY3JvcyBhbmQgc21hbGwgaW5saW5lCi1mdW5jdGlvbnMgKHRlbiBsaW5lcyBvciBsZXNzIGluIGxlbmd0aCksIHRoZW4gdGhlIHVzZSBvZiB0aGUgb2JqZWN0Ci1maWxlIGlzIHVucmVzdHJpY3RlZCwgcmVnYXJkbGVzcyBvZiB3aGV0aGVyIGl0IGlzIGxlZ2FsbHkgYSBkZXJpdmF0aXZlCi13b3JrLiAgKEV4ZWN1dGFibGVzIGNvbnRhaW5pbmcgdGhpcyBvYmplY3QgY29kZSBwbHVzIHBvcnRpb25zIG9mIHRoZQotTGlicmFyeSB3aWxsIHN0aWxsIGZhbGwgdW5kZXIgU2VjdGlvbiA2LikKLQotICBPdGhlcndpc2UsIGlmIHRoZSB3b3JrIGlzIGEgZGVyaXZhdGl2ZSBvZiB0aGUgTGlicmFyeSwgeW91IG1heQotZGlzdHJpYnV0ZSB0aGUgb2JqZWN0IGNvZGUgZm9yIHRoZSB3b3JrIHVuZGVyIHRoZSB0ZXJtcyBvZiBTZWN0aW9uIDYuCi1BbnkgZXhlY3V0YWJsZXMgY29udGFpbmluZyB0aGF0IHdvcmsgYWxzbyBmYWxsIHVuZGVyIFNlY3Rpb24gNiwKLXdoZXRoZXIgb3Igbm90IHRoZXkgYXJlIGxpbmtlZCBkaXJlY3RseSB3aXRoIHRoZSBMaWJyYXJ5IGl0c2VsZi4KLQwKLSAgNi4gQXMgYW4gZXhjZXB0aW9uIHRvIHRoZSBTZWN0aW9ucyBhYm92ZSwgeW91IG1heSBhbHNvIGNvbWJpbmUgb3IKLWxpbmsgYSAid29yayB0aGF0IHVzZXMgdGhlIExpYnJhcnkiIHdpdGggdGhlIExpYnJhcnkgdG8gcHJvZHVjZSBhCi13b3JrIGNvbnRhaW5pbmcgcG9ydGlvbnMgb2YgdGhlIExpYnJhcnksIGFuZCBkaXN0cmlidXRlIHRoYXQgd29yawotdW5kZXIgdGVybXMgb2YgeW91ciBjaG9pY2UsIHByb3ZpZGVkIHRoYXQgdGhlIHRlcm1zIHBlcm1pdAotbW9kaWZpY2F0aW9uIG9mIHRoZSB3b3JrIGZvciB0aGUgY3VzdG9tZXIncyBvd24gdXNlIGFuZCByZXZlcnNlCi1lbmdpbmVlcmluZyBmb3IgZGVidWdnaW5nIHN1Y2ggbW9kaWZpY2F0aW9ucy4KLQotICBZb3UgbXVzdCBnaXZlIHByb21pbmVudCBub3RpY2Ugd2l0aCBlYWNoIGNvcHkgb2YgdGhlIHdvcmsgdGhhdCB0aGUKLUxpYnJhcnkgaXMgdXNlZCBpbiBpdCBhbmQgdGhhdCB0aGUgTGlicmFyeSBhbmQgaXRzIHVzZSBhcmUgY292ZXJlZCBieQotdGhpcyBMaWNlbnNlLiAgWW91IG11c3Qgc3VwcGx5IGEgY29weSBvZiB0aGlzIExpY2Vuc2UuICBJZiB0aGUgd29yawotZHVyaW5nIGV4ZWN1dGlvbiBkaXNwbGF5cyBjb3B5cmlnaHQgbm90aWNlcywgeW91IG11c3QgaW5jbHVkZSB0aGUKLWNvcHlyaWdodCBub3RpY2UgZm9yIHRoZSBMaWJyYXJ5IGFtb25nIHRoZW0sIGFzIHdlbGwgYXMgYSByZWZlcmVuY2UKLWRpcmVjdGluZyB0aGUgdXNlciB0byB0aGUgY29weSBvZiB0aGlzIExpY2Vuc2UuICBBbHNvLCB5b3UgbXVzdCBkbyBvbmUKLW9mIHRoZXNlIHRoaW5nczoKLQotICAgIGEpIEFjY29tcGFueSB0aGUgd29yayB3aXRoIHRoZSBjb21wbGV0ZSBjb3JyZXNwb25kaW5nCi0gICAgbWFjaGluZS1yZWFkYWJsZSBzb3VyY2UgY29kZSBmb3IgdGhlIExpYnJhcnkgaW5jbHVkaW5nIHdoYXRldmVyCi0gICAgY2hhbmdlcyB3ZXJlIHVzZWQgaW4gdGhlIHdvcmsgKHdoaWNoIG11c3QgYmUgZGlzdHJpYnV0ZWQgdW5kZXIKLSAgICBTZWN0aW9ucyAxIGFuZCAyIGFib3ZlKTsgYW5kLCBpZiB0aGUgd29yayBpcyBhbiBleGVjdXRhYmxlIGxpbmtlZAotICAgIHdpdGggdGhlIExpYnJhcnksIHdpdGggdGhlIGNvbXBsZXRlIG1hY2hpbmUtcmVhZGFibGUgIndvcmsgdGhhdAotICAgIHVzZXMgdGhlIExpYnJhcnkiLCBhcyBvYmplY3QgY29kZSBhbmQvb3Igc291cmNlIGNvZGUsIHNvIHRoYXQgdGhlCi0gICAgdXNlciBjYW4gbW9kaWZ5IHRoZSBMaWJyYXJ5IGFuZCB0aGVuIHJlbGluayB0byBwcm9kdWNlIGEgbW9kaWZpZWQKLSAgICBleGVjdXRhYmxlIGNvbnRhaW5pbmcgdGhlIG1vZGlmaWVkIExpYnJhcnkuICAoSXQgaXMgdW5kZXJzdG9vZAotICAgIHRoYXQgdGhlIHVzZXIgd2hvIGNoYW5nZXMgdGhlIGNvbnRlbnRzIG9mIGRlZmluaXRpb25zIGZpbGVzIGluIHRoZQotICAgIExpYnJhcnkgd2lsbCBub3QgbmVjZXNzYXJpbHkgYmUgYWJsZSB0byByZWNvbXBpbGUgdGhlIGFwcGxpY2F0aW9uCi0gICAgdG8gdXNlIHRoZSBtb2RpZmllZCBkZWZpbml0aW9ucy4pCi0KLSAgICBiKSBVc2UgYSBzdWl0YWJsZSBzaGFyZWQgbGlicmFyeSBtZWNoYW5pc20gZm9yIGxpbmtpbmcgd2l0aCB0aGUKLSAgICBMaWJyYXJ5LiAgQSBzdWl0YWJsZSBtZWNoYW5pc20gaXMgb25lIHRoYXQgKDEpIHVzZXMgYXQgcnVuIHRpbWUgYQotICAgIGNvcHkgb2YgdGhlIGxpYnJhcnkgYWxyZWFkeSBwcmVzZW50IG9uIHRoZSB1c2VyJ3MgY29tcHV0ZXIgc3lzdGVtLAotICAgIHJhdGhlciB0aGFuIGNvcHlpbmcgbGlicmFyeSBmdW5jdGlvbnMgaW50byB0aGUgZXhlY3V0YWJsZSwgYW5kICgyKQotICAgIHdpbGwgb3BlcmF0ZSBwcm9wZXJseSB3aXRoIGEgbW9kaWZpZWQgdmVyc2lvbiBvZiB0aGUgbGlicmFyeSwgaWYKLSAgICB0aGUgdXNlciBpbnN0YWxscyBvbmUsIGFzIGxvbmcgYXMgdGhlIG1vZGlmaWVkIHZlcnNpb24gaXMKLSAgICBpbnRlcmZhY2UtY29tcGF0aWJsZSB3aXRoIHRoZSB2ZXJzaW9uIHRoYXQgdGhlIHdvcmsgd2FzIG1hZGUgd2l0aC4KLQotICAgIGMpIEFjY29tcGFueSB0aGUgd29yayB3aXRoIGEgd3JpdHRlbiBvZmZlciwgdmFsaWQgZm9yIGF0Ci0gICAgbGVhc3QgdGhyZWUgeWVhcnMsIHRvIGdpdmUgdGhlIHNhbWUgdXNlciB0aGUgbWF0ZXJpYWxzCi0gICAgc3BlY2lmaWVkIGluIFN1YnNlY3Rpb24gNmEsIGFib3ZlLCBmb3IgYSBjaGFyZ2Ugbm8gbW9yZQotICAgIHRoYW4gdGhlIGNvc3Qgb2YgcGVyZm9ybWluZyB0aGlzIGRpc3RyaWJ1dGlvbi4KLQotICAgIGQpIElmIGRpc3RyaWJ1dGlvbiBvZiB0aGUgd29yayBpcyBtYWRlIGJ5IG9mZmVyaW5nIGFjY2VzcyB0byBjb3B5Ci0gICAgZnJvbSBhIGRlc2lnbmF0ZWQgcGxhY2UsIG9mZmVyIGVxdWl2YWxlbnQgYWNjZXNzIHRvIGNvcHkgdGhlIGFib3ZlCi0gICAgc3BlY2lmaWVkIG1hdGVyaWFscyBmcm9tIHRoZSBzYW1lIHBsYWNlLgotCi0gICAgZSkgVmVyaWZ5IHRoYXQgdGhlIHVzZXIgaGFzIGFscmVhZHkgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZXNlCi0gICAgbWF0ZXJpYWxzIG9yIHRoYXQgeW91IGhhdmUgYWxyZWFkeSBzZW50IHRoaXMgdXNlciBhIGNvcHkuCi0KLSAgRm9yIGFuIGV4ZWN1dGFibGUsIHRoZSByZXF1aXJlZCBmb3JtIG9mIHRoZSAid29yayB0aGF0IHVzZXMgdGhlCi1MaWJyYXJ5IiBtdXN0IGluY2x1ZGUgYW55IGRhdGEgYW5kIHV0aWxpdHkgcHJvZ3JhbXMgbmVlZGVkIGZvcgotcmVwcm9kdWNpbmcgdGhlIGV4ZWN1dGFibGUgZnJvbSBpdC4gIEhvd2V2ZXIsIGFzIGEgc3BlY2lhbCBleGNlcHRpb24sCi10aGUgbWF0ZXJpYWxzIHRvIGJlIGRpc3RyaWJ1dGVkIG5lZWQgbm90IGluY2x1ZGUgYW55dGhpbmcgdGhhdCBpcwotbm9ybWFsbHkgZGlzdHJpYnV0ZWQgKGluIGVpdGhlciBzb3VyY2Ugb3IgYmluYXJ5IGZvcm0pIHdpdGggdGhlIG1ham9yCi1jb21wb25lbnRzIChjb21waWxlciwga2VybmVsLCBhbmQgc28gb24pIG9mIHRoZSBvcGVyYXRpbmcgc3lzdGVtIG9uCi13aGljaCB0aGUgZXhlY3V0YWJsZSBydW5zLCB1bmxlc3MgdGhhdCBjb21wb25lbnQgaXRzZWxmIGFjY29tcGFuaWVzCi10aGUgZXhlY3V0YWJsZS4KLQotICBJdCBtYXkgaGFwcGVuIHRoYXQgdGhpcyByZXF1aXJlbWVudCBjb250cmFkaWN0cyB0aGUgbGljZW5zZQotcmVzdHJpY3Rpb25zIG9mIG90aGVyIHByb3ByaWV0YXJ5IGxpYnJhcmllcyB0aGF0IGRvIG5vdCBub3JtYWxseQotYWNjb21wYW55IHRoZSBvcGVyYXRpbmcgc3lzdGVtLiAgU3VjaCBhIGNvbnRyYWRpY3Rpb24gbWVhbnMgeW91IGNhbm5vdAotdXNlIGJvdGggdGhlbSBhbmQgdGhlIExpYnJhcnkgdG9nZXRoZXIgaW4gYW4gZXhlY3V0YWJsZSB0aGF0IHlvdQotZGlzdHJpYnV0ZS4KLQwKLSAgNy4gWW91IG1heSBwbGFjZSBsaWJyYXJ5IGZhY2lsaXRpZXMgdGhhdCBhcmUgYSB3b3JrIGJhc2VkIG9uIHRoZQotTGlicmFyeSBzaWRlLWJ5LXNpZGUgaW4gYSBzaW5nbGUgbGlicmFyeSB0b2dldGhlciB3aXRoIG90aGVyIGxpYnJhcnkKLWZhY2lsaXRpZXMgbm90IGNvdmVyZWQgYnkgdGhpcyBMaWNlbnNlLCBhbmQgZGlzdHJpYnV0ZSBzdWNoIGEgY29tYmluZWQKLWxpYnJhcnksIHByb3ZpZGVkIHRoYXQgdGhlIHNlcGFyYXRlIGRpc3RyaWJ1dGlvbiBvZiB0aGUgd29yayBiYXNlZCBvbgotdGhlIExpYnJhcnkgYW5kIG9mIHRoZSBvdGhlciBsaWJyYXJ5IGZhY2lsaXRpZXMgaXMgb3RoZXJ3aXNlCi1wZXJtaXR0ZWQsIGFuZCBwcm92aWRlZCB0aGF0IHlvdSBkbyB0aGVzZSB0d28gdGhpbmdzOgotCi0gICAgYSkgQWNjb21wYW55IHRoZSBjb21iaW5lZCBsaWJyYXJ5IHdpdGggYSBjb3B5IG9mIHRoZSBzYW1lIHdvcmsKLSAgICBiYXNlZCBvbiB0aGUgTGlicmFyeSwgdW5jb21iaW5lZCB3aXRoIGFueSBvdGhlciBsaWJyYXJ5Ci0gICAgZmFjaWxpdGllcy4gIFRoaXMgbXVzdCBiZSBkaXN0cmlidXRlZCB1bmRlciB0aGUgdGVybXMgb2YgdGhlCi0gICAgU2VjdGlvbnMgYWJvdmUuCi0KLSAgICBiKSBHaXZlIHByb21pbmVudCBub3RpY2Ugd2l0aCB0aGUgY29tYmluZWQgbGlicmFyeSBvZiB0aGUgZmFjdAotICAgIHRoYXQgcGFydCBvZiBpdCBpcyBhIHdvcmsgYmFzZWQgb24gdGhlIExpYnJhcnksIGFuZCBleHBsYWluaW5nCi0gICAgd2hlcmUgdG8gZmluZCB0aGUgYWNjb21wYW55aW5nIHVuY29tYmluZWQgZm9ybSBvZiB0aGUgc2FtZSB3b3JrLgotCi0gIDguIFlvdSBtYXkgbm90IGNvcHksIG1vZGlmeSwgc3VibGljZW5zZSwgbGluayB3aXRoLCBvciBkaXN0cmlidXRlCi10aGUgTGlicmFyeSBleGNlcHQgYXMgZXhwcmVzc2x5IHByb3ZpZGVkIHVuZGVyIHRoaXMgTGljZW5zZS4gIEFueQotYXR0ZW1wdCBvdGhlcndpc2UgdG8gY29weSwgbW9kaWZ5LCBzdWJsaWNlbnNlLCBsaW5rIHdpdGgsIG9yCi1kaXN0cmlidXRlIHRoZSBMaWJyYXJ5IGlzIHZvaWQsIGFuZCB3aWxsIGF1dG9tYXRpY2FsbHkgdGVybWluYXRlIHlvdXIKLXJpZ2h0cyB1bmRlciB0aGlzIExpY2Vuc2UuICBIb3dldmVyLCBwYXJ0aWVzIHdobyBoYXZlIHJlY2VpdmVkIGNvcGllcywKLW9yIHJpZ2h0cywgZnJvbSB5b3UgdW5kZXIgdGhpcyBMaWNlbnNlIHdpbGwgbm90IGhhdmUgdGhlaXIgbGljZW5zZXMKLXRlcm1pbmF0ZWQgc28gbG9uZyBhcyBzdWNoIHBhcnRpZXMgcmVtYWluIGluIGZ1bGwgY29tcGxpYW5jZS4KLQotICA5LiBZb3UgYXJlIG5vdCByZXF1aXJlZCB0byBhY2NlcHQgdGhpcyBMaWNlbnNlLCBzaW5jZSB5b3UgaGF2ZSBub3QKLXNpZ25lZCBpdC4gIEhvd2V2ZXIsIG5vdGhpbmcgZWxzZSBncmFudHMgeW91IHBlcm1pc3Npb24gdG8gbW9kaWZ5IG9yCi1kaXN0cmlidXRlIHRoZSBMaWJyYXJ5IG9yIGl0cyBkZXJpdmF0aXZlIHdvcmtzLiAgVGhlc2UgYWN0aW9ucyBhcmUKLXByb2hpYml0ZWQgYnkgbGF3IGlmIHlvdSBkbyBub3QgYWNjZXB0IHRoaXMgTGljZW5zZS4gIFRoZXJlZm9yZSwgYnkKLW1vZGlmeWluZyBvciBkaXN0cmlidXRpbmcgdGhlIExpYnJhcnkgKG9yIGFueSB3b3JrIGJhc2VkIG9uIHRoZQotTGlicmFyeSksIHlvdSBpbmRpY2F0ZSB5b3VyIGFjY2VwdGFuY2Ugb2YgdGhpcyBMaWNlbnNlIHRvIGRvIHNvLCBhbmQKLWFsbCBpdHMgdGVybXMgYW5kIGNvbmRpdGlvbnMgZm9yIGNvcHlpbmcsIGRpc3RyaWJ1dGluZyBvciBtb2RpZnlpbmcKLXRoZSBMaWJyYXJ5IG9yIHdvcmtzIGJhc2VkIG9uIGl0LgotCi0gIDEwLiBFYWNoIHRpbWUgeW91IHJlZGlzdHJpYnV0ZSB0aGUgTGlicmFyeSAob3IgYW55IHdvcmsgYmFzZWQgb24gdGhlCi1MaWJyYXJ5KSwgdGhlIHJlY2lwaWVudCBhdXRvbWF0aWNhbGx5IHJlY2VpdmVzIGEgbGljZW5zZSBmcm9tIHRoZQotb3JpZ2luYWwgbGljZW5zb3IgdG8gY29weSwgZGlzdHJpYnV0ZSwgbGluayB3aXRoIG9yIG1vZGlmeSB0aGUgTGlicmFyeQotc3ViamVjdCB0byB0aGVzZSB0ZXJtcyBhbmQgY29uZGl0aW9ucy4gIFlvdSBtYXkgbm90IGltcG9zZSBhbnkgZnVydGhlcgotcmVzdHJpY3Rpb25zIG9uIHRoZSByZWNpcGllbnRzJyBleGVyY2lzZSBvZiB0aGUgcmlnaHRzIGdyYW50ZWQgaGVyZWluLgotWW91IGFyZSBub3QgcmVzcG9uc2libGUgZm9yIGVuZm9yY2luZyBjb21wbGlhbmNlIGJ5IHRoaXJkIHBhcnRpZXMgd2l0aAotdGhpcyBMaWNlbnNlLgotDAotICAxMS4gSWYsIGFzIGEgY29uc2VxdWVuY2Ugb2YgYSBjb3VydCBqdWRnbWVudCBvciBhbGxlZ2F0aW9uIG9mIHBhdGVudAotaW5mcmluZ2VtZW50IG9yIGZvciBhbnkgb3RoZXIgcmVhc29uIChub3QgbGltaXRlZCB0byBwYXRlbnQgaXNzdWVzKSwKLWNvbmRpdGlvbnMgYXJlIGltcG9zZWQgb24geW91ICh3aGV0aGVyIGJ5IGNvdXJ0IG9yZGVyLCBhZ3JlZW1lbnQgb3IKLW90aGVyd2lzZSkgdGhhdCBjb250cmFkaWN0IHRoZSBjb25kaXRpb25zIG9mIHRoaXMgTGljZW5zZSwgdGhleSBkbyBub3QKLWV4Y3VzZSB5b3UgZnJvbSB0aGUgY29uZGl0aW9ucyBvZiB0aGlzIExpY2Vuc2UuICBJZiB5b3UgY2Fubm90Ci1kaXN0cmlidXRlIHNvIGFzIHRvIHNhdGlzZnkgc2ltdWx0YW5lb3VzbHkgeW91ciBvYmxpZ2F0aW9ucyB1bmRlciB0aGlzCi1MaWNlbnNlIGFuZCBhbnkgb3RoZXIgcGVydGluZW50IG9ibGlnYXRpb25zLCB0aGVuIGFzIGEgY29uc2VxdWVuY2UgeW91Ci1tYXkgbm90IGRpc3RyaWJ1dGUgdGhlIExpYnJhcnkgYXQgYWxsLiAgRm9yIGV4YW1wbGUsIGlmIGEgcGF0ZW50Ci1saWNlbnNlIHdvdWxkIG5vdCBwZXJtaXQgcm95YWx0eS1mcmVlIHJlZGlzdHJpYnV0aW9uIG9mIHRoZSBMaWJyYXJ5IGJ5Ci1hbGwgdGhvc2Ugd2hvIHJlY2VpdmUgY29waWVzIGRpcmVjdGx5IG9yIGluZGlyZWN0bHkgdGhyb3VnaCB5b3UsIHRoZW4KLXRoZSBvbmx5IHdheSB5b3UgY291bGQgc2F0aXNmeSBib3RoIGl0IGFuZCB0aGlzIExpY2Vuc2Ugd291bGQgYmUgdG8KLXJlZnJhaW4gZW50aXJlbHkgZnJvbSBkaXN0cmlidXRpb24gb2YgdGhlIExpYnJhcnkuCi0KLUlmIGFueSBwb3J0aW9uIG9mIHRoaXMgc2VjdGlvbiBpcyBoZWxkIGludmFsaWQgb3IgdW5lbmZvcmNlYWJsZSB1bmRlciBhbnkKLXBhcnRpY3VsYXIgY2lyY3Vtc3RhbmNlLCB0aGUgYmFsYW5jZSBvZiB0aGUgc2VjdGlvbiBpcyBpbnRlbmRlZCB0byBhcHBseSwKLWFuZCB0aGUgc2VjdGlvbiBhcyBhIHdob2xlIGlzIGludGVuZGVkIHRvIGFwcGx5IGluIG90aGVyIGNpcmN1bXN0YW5jZXMuCi0KLUl0IGlzIG5vdCB0aGUgcHVycG9zZSBvZiB0aGlzIHNlY3Rpb24gdG8gaW5kdWNlIHlvdSB0byBpbmZyaW5nZSBhbnkKLXBhdGVudHMgb3Igb3RoZXIgcHJvcGVydHkgcmlnaHQgY2xhaW1zIG9yIHRvIGNvbnRlc3QgdmFsaWRpdHkgb2YgYW55Ci1zdWNoIGNsYWltczsgdGhpcyBzZWN0aW9uIGhhcyB0aGUgc29sZSBwdXJwb3NlIG9mIHByb3RlY3RpbmcgdGhlCi1pbnRlZ3JpdHkgb2YgdGhlIGZyZWUgc29mdHdhcmUgZGlzdHJpYnV0aW9uIHN5c3RlbSB3aGljaCBpcwotaW1wbGVtZW50ZWQgYnkgcHVibGljIGxpY2Vuc2UgcHJhY3RpY2VzLiAgTWFueSBwZW9wbGUgaGF2ZSBtYWRlCi1nZW5lcm91cyBjb250cmlidXRpb25zIHRvIHRoZSB3aWRlIHJhbmdlIG9mIHNvZnR3YXJlIGRpc3RyaWJ1dGVkCi10aHJvdWdoIHRoYXQgc3lzdGVtIGluIHJlbGlhbmNlIG9uIGNvbnNpc3RlbnQgYXBwbGljYXRpb24gb2YgdGhhdAotc3lzdGVtOyBpdCBpcyB1cCB0byB0aGUgYXV0aG9yL2Rvbm9yIHRvIGRlY2lkZSBpZiBoZSBvciBzaGUgaXMgd2lsbGluZwotdG8gZGlzdHJpYnV0ZSBzb2Z0d2FyZSB0aHJvdWdoIGFueSBvdGhlciBzeXN0ZW0gYW5kIGEgbGljZW5zZWUgY2Fubm90Ci1pbXBvc2UgdGhhdCBjaG9pY2UuCi0KLVRoaXMgc2VjdGlvbiBpcyBpbnRlbmRlZCB0byBtYWtlIHRob3JvdWdobHkgY2xlYXIgd2hhdCBpcyBiZWxpZXZlZCB0bwotYmUgYSBjb25zZXF1ZW5jZSBvZiB0aGUgcmVzdCBvZiB0aGlzIExpY2Vuc2UuCi0KLSAgMTIuIElmIHRoZSBkaXN0cmlidXRpb24gYW5kL29yIHVzZSBvZiB0aGUgTGlicmFyeSBpcyByZXN0cmljdGVkIGluCi1jZXJ0YWluIGNvdW50cmllcyBlaXRoZXIgYnkgcGF0ZW50cyBvciBieSBjb3B5cmlnaHRlZCBpbnRlcmZhY2VzLCB0aGUKLW9yaWdpbmFsIGNvcHlyaWdodCBob2xkZXIgd2hvIHBsYWNlcyB0aGUgTGlicmFyeSB1bmRlciB0aGlzIExpY2Vuc2UgbWF5IGFkZAotYW4gZXhwbGljaXQgZ2VvZ3JhcGhpY2FsIGRpc3RyaWJ1dGlvbiBsaW1pdGF0aW9uIGV4Y2x1ZGluZyB0aG9zZSBjb3VudHJpZXMsCi1zbyB0aGF0IGRpc3RyaWJ1dGlvbiBpcyBwZXJtaXR0ZWQgb25seSBpbiBvciBhbW9uZyBjb3VudHJpZXMgbm90IHRodXMKLWV4Y2x1ZGVkLiAgSW4gc3VjaCBjYXNlLCB0aGlzIExpY2Vuc2UgaW5jb3Jwb3JhdGVzIHRoZSBsaW1pdGF0aW9uIGFzIGlmCi13cml0dGVuIGluIHRoZSBib2R5IG9mIHRoaXMgTGljZW5zZS4KLQotICAxMy4gVGhlIEZyZWUgU29mdHdhcmUgRm91bmRhdGlvbiBtYXkgcHVibGlzaCByZXZpc2VkIGFuZC9vciBuZXcKLXZlcnNpb25zIG9mIHRoZSBMZXNzZXIgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmcm9tIHRpbWUgdG8gdGltZS4KLVN1Y2ggbmV3IHZlcnNpb25zIHdpbGwgYmUgc2ltaWxhciBpbiBzcGlyaXQgdG8gdGhlIHByZXNlbnQgdmVyc2lvbiwKLWJ1dCBtYXkgZGlmZmVyIGluIGRldGFpbCB0byBhZGRyZXNzIG5ldyBwcm9ibGVtcyBvciBjb25jZXJucy4KLQotRWFjaCB2ZXJzaW9uIGlzIGdpdmVuIGEgZGlzdGluZ3Vpc2hpbmcgdmVyc2lvbiBudW1iZXIuICBJZiB0aGUgTGlicmFyeQotc3BlY2lmaWVzIGEgdmVyc2lvbiBudW1iZXIgb2YgdGhpcyBMaWNlbnNlIHdoaWNoIGFwcGxpZXMgdG8gaXQgYW5kCi0iYW55IGxhdGVyIHZlcnNpb24iLCB5b3UgaGF2ZSB0aGUgb3B0aW9uIG9mIGZvbGxvd2luZyB0aGUgdGVybXMgYW5kCi1jb25kaXRpb25zIGVpdGhlciBvZiB0aGF0IHZlcnNpb24gb3Igb2YgYW55IGxhdGVyIHZlcnNpb24gcHVibGlzaGVkIGJ5Ci10aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uLiAgSWYgdGhlIExpYnJhcnkgZG9lcyBub3Qgc3BlY2lmeSBhCi1saWNlbnNlIHZlcnNpb24gbnVtYmVyLCB5b3UgbWF5IGNob29zZSBhbnkgdmVyc2lvbiBldmVyIHB1Ymxpc2hlZCBieQotdGhlIEZyZWUgU29mdHdhcmUgRm91bmRhdGlvbi4KLQwKLSAgMTQuIElmIHlvdSB3aXNoIHRvIGluY29ycG9yYXRlIHBhcnRzIG9mIHRoZSBMaWJyYXJ5IGludG8gb3RoZXIgZnJlZQotcHJvZ3JhbXMgd2hvc2UgZGlzdHJpYnV0aW9uIGNvbmRpdGlvbnMgYXJlIGluY29tcGF0aWJsZSB3aXRoIHRoZXNlLAotd3JpdGUgdG8gdGhlIGF1dGhvciB0byBhc2sgZm9yIHBlcm1pc3Npb24uICBGb3Igc29mdHdhcmUgd2hpY2ggaXMKLWNvcHlyaWdodGVkIGJ5IHRoZSBGcmVlIFNvZnR3YXJlIEZvdW5kYXRpb24sIHdyaXRlIHRvIHRoZSBGcmVlCi1Tb2Z0d2FyZSBGb3VuZGF0aW9uOyB3ZSBzb21ldGltZXMgbWFrZSBleGNlcHRpb25zIGZvciB0aGlzLiAgT3VyCi1kZWNpc2lvbiB3aWxsIGJlIGd1aWRlZCBieSB0aGUgdHdvIGdvYWxzIG9mIHByZXNlcnZpbmcgdGhlIGZyZWUgc3RhdHVzCi1vZiBhbGwgZGVyaXZhdGl2ZXMgb2Ygb3VyIGZyZWUgc29mdHdhcmUgYW5kIG9mIHByb21vdGluZyB0aGUgc2hhcmluZwotYW5kIHJldXNlIG9mIHNvZnR3YXJlIGdlbmVyYWxseS4KLQotCQkJICAgIE5PIFdBUlJBTlRZCi0KLSAgMTUuIEJFQ0FVU0UgVEhFIExJQlJBUlkgSVMgTElDRU5TRUQgRlJFRSBPRiBDSEFSR0UsIFRIRVJFIElTIE5PCi1XQVJSQU5UWSBGT1IgVEhFIExJQlJBUlksIFRPIFRIRSBFWFRFTlQgUEVSTUlUVEVEIEJZIEFQUExJQ0FCTEUgTEFXLgotRVhDRVBUIFdIRU4gT1RIRVJXSVNFIFNUQVRFRCBJTiBXUklUSU5HIFRIRSBDT1BZUklHSFQgSE9MREVSUyBBTkQvT1IKLU9USEVSIFBBUlRJRVMgUFJPVklERSBUSEUgTElCUkFSWSAiQVMgSVMiIFdJVEhPVVQgV0FSUkFOVFkgT0YgQU5ZCi1LSU5ELCBFSVRIRVIgRVhQUkVTU0VEIE9SIElNUExJRUQsIElOQ0xVRElORywgQlVUIE5PVCBMSU1JVEVEIFRPLCBUSEUKLUlNUExJRUQgV0FSUkFOVElFUyBPRiBNRVJDSEFOVEFCSUxJVFkgQU5EIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUgotUFVSUE9TRS4gIFRIRSBFTlRJUkUgUklTSyBBUyBUTyBUSEUgUVVBTElUWSBBTkQgUEVSRk9STUFOQ0UgT0YgVEhFCi1MSUJSQVJZIElTIFdJVEggWU9VLiAgU0hPVUxEIFRIRSBMSUJSQVJZIFBST1ZFIERFRkVDVElWRSwgWU9VIEFTU1VNRQotVEhFIENPU1QgT0YgQUxMIE5FQ0VTU0FSWSBTRVJWSUNJTkcsIFJFUEFJUiBPUiBDT1JSRUNUSU9OLgotCi0gIDE2LiBJTiBOTyBFVkVOVCBVTkxFU1MgUkVRVUlSRUQgQlkgQVBQTElDQUJMRSBMQVcgT1IgQUdSRUVEIFRPIElOCi1XUklUSU5HIFdJTEwgQU5ZIENPUFlSSUdIVCBIT0xERVIsIE9SIEFOWSBPVEhFUiBQQVJUWSBXSE8gTUFZIE1PRElGWQotQU5EL09SIFJFRElTVFJJQlVURSBUSEUgTElCUkFSWSBBUyBQRVJNSVRURUQgQUJPVkUsIEJFIExJQUJMRSBUTyBZT1UKLUZPUiBEQU1BR0VTLCBJTkNMVURJTkcgQU5ZIEdFTkVSQUwsIFNQRUNJQUwsIElOQ0lERU5UQUwgT1IKLUNPTlNFUVVFTlRJQUwgREFNQUdFUyBBUklTSU5HIE9VVCBPRiBUSEUgVVNFIE9SIElOQUJJTElUWSBUTyBVU0UgVEhFCi1MSUJSQVJZIChJTkNMVURJTkcgQlVUIE5PVCBMSU1JVEVEIFRPIExPU1MgT0YgREFUQSBPUiBEQVRBIEJFSU5HCi1SRU5ERVJFRCBJTkFDQ1VSQVRFIE9SIExPU1NFUyBTVVNUQUlORUQgQlkgWU9VIE9SIFRISVJEIFBBUlRJRVMgT1IgQQotRkFJTFVSRSBPRiBUSEUgTElCUkFSWSBUTyBPUEVSQVRFIFdJVEggQU5ZIE9USEVSIFNPRlRXQVJFKSwgRVZFTiBJRgotU1VDSCBIT0xERVIgT1IgT1RIRVIgUEFSVFkgSEFTIEJFRU4gQURWSVNFRCBPRiBUSEUgUE9TU0lCSUxJVFkgT0YgU1VDSAotREFNQUdFUy4KLQotCQkgICAgIEVORCBPRiBURVJNUyBBTkQgQ09ORElUSU9OUwotDAotICAgICAgICAgICBIb3cgdG8gQXBwbHkgVGhlc2UgVGVybXMgdG8gWW91ciBOZXcgTGlicmFyaWVzCi0KLSAgSWYgeW91IGRldmVsb3AgYSBuZXcgbGlicmFyeSwgYW5kIHlvdSB3YW50IGl0IHRvIGJlIG9mIHRoZSBncmVhdGVzdAotcG9zc2libGUgdXNlIHRvIHRoZSBwdWJsaWMsIHdlIHJlY29tbWVuZCBtYWtpbmcgaXQgZnJlZSBzb2Z0d2FyZSB0aGF0Ci1ldmVyeW9uZSBjYW4gcmVkaXN0cmlidXRlIGFuZCBjaGFuZ2UuICBZb3UgY2FuIGRvIHNvIGJ5IHBlcm1pdHRpbmcKLXJlZGlzdHJpYnV0aW9uIHVuZGVyIHRoZXNlIHRlcm1zIChvciwgYWx0ZXJuYXRpdmVseSwgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZQotb3JkaW5hcnkgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSkuCi0KLSAgVG8gYXBwbHkgdGhlc2UgdGVybXMsIGF0dGFjaCB0aGUgZm9sbG93aW5nIG5vdGljZXMgdG8gdGhlIGxpYnJhcnkuICBJdCBpcwotc2FmZXN0IHRvIGF0dGFjaCB0aGVtIHRvIHRoZSBzdGFydCBvZiBlYWNoIHNvdXJjZSBmaWxlIHRvIG1vc3QgZWZmZWN0aXZlbHkKLWNvbnZleSB0aGUgZXhjbHVzaW9uIG9mIHdhcnJhbnR5OyBhbmQgZWFjaCBmaWxlIHNob3VsZCBoYXZlIGF0IGxlYXN0IHRoZQotImNvcHlyaWdodCIgbGluZSBhbmQgYSBwb2ludGVyIHRvIHdoZXJlIHRoZSBmdWxsIG5vdGljZSBpcyBmb3VuZC4KLQotICAgIDxvbmUgbGluZSB0byBnaXZlIHRoZSBsaWJyYXJ5J3MgbmFtZSBhbmQgYSBicmllZiBpZGVhIG9mIHdoYXQgaXQgZG9lcy4+Ci0gICAgQ29weXJpZ2h0IChDKSA8eWVhcj4gIDxuYW1lIG9mIGF1dGhvcj4KLQotICAgIFRoaXMgbGlicmFyeSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IKLSAgICBtb2RpZnkgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljCi0gICAgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkgdGhlIEZyZWUgU29mdHdhcmUgRm91bmRhdGlvbjsgZWl0aGVyCi0gICAgdmVyc2lvbiAyLjEgb2YgdGhlIExpY2Vuc2UsIG9yIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCi0KLSAgICBUaGlzIGxpYnJhcnkgaXMgZGlzdHJpYnV0ZWQgaW4gdGhlIGhvcGUgdGhhdCBpdCB3aWxsIGJlIHVzZWZ1bCwKLSAgICBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgotICAgIE1FUkNIQU5UQUJJTElUWSBvciBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRS4gIFNlZSB0aGUgR05VCi0gICAgTGVzc2VyIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KLQotICAgIFlvdSBzaG91bGQgaGF2ZSByZWNlaXZlZCBhIGNvcHkgb2YgdGhlIEdOVSBMZXNzZXIgR2VuZXJhbCBQdWJsaWMKLSAgICBMaWNlbnNlIGFsb25nIHdpdGggdGhpcyBsaWJyYXJ5OyBpZiBub3QsIHdyaXRlIHRvIHRoZSBGcmVlIFNvZnR3YXJlCi0gICAgRm91bmRhdGlvbiwgSW5jLiwgNTkgVGVtcGxlIFBsYWNlLCBTdWl0ZSAzMzAsIEJvc3RvbiwgTUEgIDAyMTExLTEzMDcgIFVTQQotCi1BbHNvIGFkZCBpbmZvcm1hdGlvbiBvbiBob3cgdG8gY29udGFjdCB5b3UgYnkgZWxlY3Ryb25pYyBhbmQgcGFwZXIgbWFpbC4KLQotWW91IHNob3VsZCBhbHNvIGdldCB5b3VyIGVtcGxveWVyIChpZiB5b3Ugd29yayBhcyBhIHByb2dyYW1tZXIpIG9yIHlvdXIKLXNjaG9vbCwgaWYgYW55LCB0byBzaWduIGEgImNvcHlyaWdodCBkaXNjbGFpbWVyIiBmb3IgdGhlIGxpYnJhcnksIGlmCi1uZWNlc3NhcnkuICBIZXJlIGlzIGEgc2FtcGxlOyBhbHRlciB0aGUgbmFtZXM6Ci0KLSAgWW95b2R5bmUsIEluYy4sIGhlcmVieSBkaXNjbGFpbXMgYWxsIGNvcHlyaWdodCBpbnRlcmVzdCBpbiB0aGUKLSAgbGlicmFyeSBgRnJvYicgKGEgbGlicmFyeSBmb3IgdHdlYWtpbmcga25vYnMpIHdyaXR0ZW4gYnkgSmFtZXMgUmFuZG9tIEhhY2tlci4KLQotICA8c2lnbmF0dXJlIG9mIFR5IENvb24+LCAxIEFwcmlsIDE5OTAKLSAgVHkgQ29vbiwgUHJlc2lkZW50IG9mIFZpY2UKLQotVGhhdCdzIGFsbCB0aGVyZSBpcyB0byBpdCEKLQotCmRpZmYgLS1naXQgYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9saWNlbnNlLnR4dCBiL29wZW5nbC90ZXN0cy9hbmdlbGVzL2xpY2Vuc2UudHh0CmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA2MjA4NDFlLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy9hbmdlbGVzL2xpY2Vuc2UudHh0CisrKyAvZGV2L251bGwKQEAgLTEsMTkgKzAsMCBAQAotU2FuIEFuZ2VsZXMgT2JzZXJ2YXRpb24gT3BlbkdMIEVTIHZlcnNpb24gZXhhbXBsZQ0KLUNvcHlyaWdodCAyMDA0LTIwMDUgSmV0cm8gTGF1aGENCi1BbGwgcmlnaHRzIHJlc2VydmVkLg0KLVdlYjogaHR0cDovL2lraS5maS9qZXRyby8NCi0NCi1UaGlzIHNvdXJjZSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3INCi1tb2RpZnkgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIEVJVEhFUjoNCi0gICgxKSBUaGUgR05VIExlc3NlciBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieSB0aGUgRnJlZQ0KLSAgICAgIFNvZnR3YXJlIEZvdW5kYXRpb247IGVpdGhlciB2ZXJzaW9uIDIuMSBvZiB0aGUgTGljZW5zZSwgb3IgKGF0DQotICAgICAgeW91ciBvcHRpb24pIGFueSBsYXRlciB2ZXJzaW9uLiBUaGUgdGV4dCBvZiB0aGUgR05VIExlc3Nlcg0KLSAgICAgIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgaXMgaW5jbHVkZWQgd2l0aCB0aGlzIHNvdXJjZSBpbiB0aGUNCi0gICAgICBmaWxlIExJQ0VOU0UtTEdQTC50eHQuDQotICAoMikgVGhlIEJTRC1zdHlsZSBsaWNlbnNlIHRoYXQgaXMgaW5jbHVkZWQgd2l0aCB0aGlzIHNvdXJjZSBpbg0KLSAgICAgIHRoZSBmaWxlIExJQ0VOU0UtQlNELnR4dC4NCi0NCi1UaGlzIHNvdXJjZSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLA0KLWJ1dCBXSVRIT1VUIEFOWSBXQVJSQU5UWTsgd2l0aG91dCBldmVuIHRoZSBpbXBsaWVkIHdhcnJhbnR5IG9mDQotTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiBTZWUgdGhlIGZpbGVzDQotTElDRU5TRS1MR1BMLnR4dCBhbmQgTElDRU5TRS1CU0QudHh0IGZvciBtb3JlIGRldGFpbHMuDQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL2FuZ2VsZXMvc2hhcGVzLmggYi9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9zaGFwZXMuaApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggMjVmZmFlOC4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdGVzdHMvYW5nZWxlcy9zaGFwZXMuaAorKysgL2Rldi9udWxsCkBAIC0xLDU5ICswLDAgQEAKLS8qIFNhbiBBbmdlbGVzIE9ic2VydmF0aW9uIE9wZW5HTCBFUyB2ZXJzaW9uIGV4YW1wbGUKLSAqIENvcHlyaWdodCAyMDA0LTIwMDUgSmV0cm8gTGF1aGEKLSAqIEFsbCByaWdodHMgcmVzZXJ2ZWQuCi0gKiBXZWI6IGh0dHA6Ly9pa2kuZmkvamV0cm8vCi0gKgotICogVGhpcyBzb3VyY2UgaXMgZnJlZSBzb2Z0d2FyZTsgeW91IGNhbiByZWRpc3RyaWJ1dGUgaXQgYW5kL29yCi0gKiBtb2RpZnkgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIEVJVEhFUjoKLSAqICAgKDEpIFRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgYXMgcHVibGlzaGVkIGJ5IHRoZSBGcmVlCi0gKiAgICAgICBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIgdmVyc2lvbiAyLjEgb2YgdGhlIExpY2Vuc2UsIG9yIChhdAotICogICAgICAgeW91ciBvcHRpb24pIGFueSBsYXRlciB2ZXJzaW9uLiBUaGUgdGV4dCBvZiB0aGUgR05VIExlc3NlcgotICogICAgICAgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBpcyBpbmNsdWRlZCB3aXRoIHRoaXMgc291cmNlIGluIHRoZQotICogICAgICAgZmlsZSBMSUNFTlNFLUxHUEwudHh0LgotICogICAoMikgVGhlIEJTRC1zdHlsZSBsaWNlbnNlIHRoYXQgaXMgaW5jbHVkZWQgd2l0aCB0aGlzIHNvdXJjZSBpbgotICogICAgICAgdGhlIGZpbGUgTElDRU5TRS1CU0QudHh0LgotICoKLSAqIFRoaXMgc291cmNlIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsCi0gKiBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgotICogTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiBTZWUgdGhlIGZpbGVzCi0gKiBMSUNFTlNFLUxHUEwudHh0IGFuZCBMSUNFTlNFLUJTRC50eHQgZm9yIG1vcmUgZGV0YWlscy4KLSAqCi0gKiAkSWQ6IHNoYXBlcy5oLHYgMS42IDIwMDUvMDEvMzEgMjI6MTU6MzAgdG9uaWMgRXhwICQKLSAqICRSZXZpc2lvbjogMS42ICQKLSAqLwotCi0jaWZuZGVmIFNIQVBFU19IX0lOQ0xVREVECi0jZGVmaW5lIFNIQVBFU19IX0lOQ0xVREVECi0KLQotI2RlZmluZSBTVVBFUlNIQVBFX1BBUkFNUyAxNQotCi1zdGF0aWMgY29uc3QgZmxvYXQgc1N1cGVyU2hhcGVQYXJhbXNbXVtTVVBFUlNIQVBFX1BBUkFNU10gPQotewotICAgIC8vIG0gIGEgICAgIGIgICAgIG4xICAgICAgbjIgICAgIG4zICAgICBtICAgICBhICAgICBiICAgICBuMSAgICAgbjIgICAgICBuMyAgIHJlczEgcmVzMiBzY2FsZSAgKG9yZy5yZXMxLHJlczIpCi0gICAgeyAxMCwgMSwgICAgMiwgICAgOTAsICAgICAgMSwgICAtNDUsICAgIDgsICAgIDEsICAgIDEsICAgIC0xLCAgICAgMSwgIC0wLjRmLCAgIDIwLCAgMzAsIDIgfSwgLy8gNDAsIDYwCi0gICAgeyAxMCwgMSwgICAgMiwgICAgOTAsICAgICAgMSwgICAtNDUsICAgIDQsICAgIDEsICAgIDEsICAgIDEwLCAgICAgMSwgIC0wLjRmLCAgIDIwLCAgMjAsIDQgfSwgLy8gNDAsIDQwCi0gICAgeyAxMCwgMSwgICAgMiwgICAgNjAsICAgICAgMSwgICAtMTAsICAgIDQsICAgIDEsICAgIDEsICAgIC0xLCAgICAtMiwgIC0wLjRmLCAgIDQxLCAgNDEsIDEgfSwgLy8gODIsIDgyCi0gICAgeyAgNiwgMSwgICAgMSwgICAgNjAsICAgICAgMSwgICAtNzAsICAgIDgsICAgIDEsICAgIDEsICAwLjRmLCAgICAgMywgIDAuMjVmLCAgIDIwLCAgMjAsIDEgfSwgLy8gNDAsIDQwCi0gICAgeyAgNCwgMSwgICAgMSwgICAgMzAsICAgICAgMSwgICAgMjAsICAgMTIsICAgIDEsICAgIDEsICAwLjRmLCAgICAgMywgIDAuMjVmLCAgIDEwLCAgMzAsIDEgfSwgLy8gMjAsIDYwCi0gICAgeyAgOCwgMSwgICAgMSwgICAgMzAsICAgICAgMSwgICAgLTQsICAgIDgsICAgIDIsICAgIDEsICAgIC0xLCAgICAgNSwgICAwLjVmLCAgIDI1LCAgMjYsIDEgfSwgLy8gNjAsIDYwCi0gICAgeyAxMywgMSwgICAgMSwgICAgMzAsICAgICAgMSwgICAgLTQsICAgMTMsICAgIDEsICAgIDEsICAgICAxLCAgICAgNSwgICAgICAxLCAgIDMwLCAgMzAsIDYgfSwgLy8gNjAsIDYwCi0gICAgeyAxMCwgMSwgMS4xZiwgLTAuNWYsICAgMC4xZiwgICAgNzAsICAgNjAsICAgIDEsICAgIDEsICAgLTkwLCAgICAgMCwgLTAuMjVmLCAgIDIwLCAgNjAsIDggfSwgLy8gNjAsIDE4MAotICAgIHsgIDcsIDEsICAgIDEsICAgIDIwLCAgLTAuM2YsIC0zLjVmLCAgICA2LCAgICAxLCAgICAxLCAgICAtMSwgIDQuNWYsICAgMC41ZiwgICAxMCwgIDIwLCA0IH0sIC8vIDYwLCA4MAotICAgIHsgIDQsIDEsICAgIDEsICAgIDEwLCAgICAgMTAsICAgIDEwLCAgICA0LCAgICAxLCAgICAxLCAgICAxMCwgICAgMTAsICAgICAxMCwgICAxMCwgIDIwLCAxIH0sIC8vIDIwLCA0MAotICAgIHsgIDQsIDEsICAgIDEsICAgICAxLCAgICAgIDEsICAgICAxLCAgICA0LCAgICAxLCAgICAxLCAgICAgMSwgICAgIDEsICAgICAgMSwgICAxMCwgIDEwLCAyIH0sIC8vIDEwLCAxMAotICAgIHsgIDEsIDEsICAgIDEsICAgIDM4LCAtMC4yNWYsICAgIDE5LCAgICA0LCAgICAxLCAgICAxLCAgICAxMCwgICAgMTAsICAgICAxMCwgICAxMCwgIDE1LCAyIH0sIC8vIDIwLCA0MAotICAgIHsgIDIsIDEsICAgIDEsICAwLjdmLCAgIDAuM2YsICAwLjJmLCAgICAzLCAgICAxLCAgICAxLCAgIDEwMCwgICAxMDAsICAgIDEwMCwgICAxMCwgIDI1LCAyIH0sIC8vIDIwLCA1MAotICAgIHsgIDYsIDEsICAgIDEsICAgICAxLCAgICAgIDEsICAgICAxLCAgICAzLCAgICAxLCAgICAxLCAgICAgMSwgICAgIDEsICAgICAgMSwgICAzMCwgIDMwLCAyIH0sIC8vIDYwLCA2MAotICAgIHsgIDMsIDEsICAgIDEsICAgICAxLCAgICAgIDEsICAgICAxLCAgICA2LCAgICAxLCAgICAxLCAgICAgMiwgICAgIDEsICAgICAgMSwgICAxMCwgIDIwLCAyIH0sIC8vIDIwLCA0MAotICAgIHsgIDYsIDEsICAgIDEsICAgICA2LCAgIDUuNWYsICAgMTAwLCAgICA2LCAgICAxLCAgICAxLCAgICAyNSwgICAgMTAsICAgICAxMCwgICAzMCwgIDIwLCAyIH0sIC8vIDYwLCA0MAotICAgIHsgIDMsIDEsICAgIDEsICAwLjVmLCAgIDEuN2YsICAxLjdmLCAgICAyLCAgICAxLCAgICAxLCAgICAxMCwgICAgMTAsICAgICAxMCwgICAyMCwgIDIwLCAyIH0sIC8vIDQwLCA0MAotICAgIHsgIDUsIDEsICAgIDEsICAwLjFmLCAgIDEuN2YsICAxLjdmLCAgICAxLCAgICAxLCAgICAxLCAgMC4zZiwgIDAuNWYsICAgMC41ZiwgICAyMCwgIDIwLCA0IH0sIC8vIDQwLCA0MAotICAgIHsgIDIsIDEsICAgIDEsICAgICA2LCAgIDUuNWYsICAgMTAwLCAgICA2LCAgICAxLCAgICAxLCAgICAgNCwgICAgMTAsICAgICAxMCwgICAxMCwgIDIyLCAxIH0sIC8vIDQwLCA0MAotICAgIHsgIDYsIDEsICAgIDEsICAgIC0xLCAgICAgNzAsICAwLjFmLCAgICA5LCAgICAxLCAwLjVmLCAgIC05OCwgMC4wNWYsICAgIC00NSwgICAyMCwgIDMwLCA0IH0sIC8vIDYwLCA5MQotICAgIHsgIDYsIDEsICAgIDEsICAgIC0xLCAgICAgOTAsIC0wLjFmLCAgICA3LCAgICAxLCAgICAxLCAgICA5MCwgIDEuM2YsICAgICAzNCwgICAxMywgIDE2LCAxIH0sIC8vIDMyLCA2MAotfTsKLSNkZWZpbmUgU1VQRVJTSEFQRV9DT1VOVCAoc2l6ZW9mKHNTdXBlclNoYXBlUGFyYW1zKSAvIHNpemVvZihzU3VwZXJTaGFwZVBhcmFtc1swXSkpCi0KLQotI2VuZGlmIC8vICFTSEFQRVNfSF9JTkNMVURFRApkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL2ZpbHRlci9BbmRyb2lkLm1rIGIvb3BlbmdsL3Rlc3RzL2ZpbHRlci9BbmRyb2lkLm1rCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBhNDQ4ZjBkLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy9maWx0ZXIvQW5kcm9pZC5taworKysgL2Rldi9udWxsCkBAIC0xLDE3ICswLDAgQEAKLUxPQ0FMX1BBVEg6PSAkKGNhbGwgbXktZGlyKQotaW5jbHVkZSAkKENMRUFSX1ZBUlMpCi0KLUxPQ0FMX1NSQ19GSUxFUzo9IFwKLQlmaWx0ZXIuYwotCi1MT0NBTF9TSEFSRURfTElCUkFSSUVTIDo9IFwKLQlsaWJjdXRpbHMgXAotICAgIGxpYkVHTCBcCi0gICAgbGliR0xFU3YxX0NNIFwKLSAgICBsaWJ1aQotCi1MT0NBTF9NT0RVTEU6PSB0ZXN0LW9wZW5nbC1maWx0ZXIKLQotTE9DQUxfTU9EVUxFX1RBR1MgOj0gdGVzdHMKLQotaW5jbHVkZSAkKEJVSUxEX0VYRUNVVEFCTEUpCmRpZmYgLS1naXQgYS9vcGVuZ2wvdGVzdHMvZmlsdGVyL2ZpbHRlci5jIGIvb3BlbmdsL3Rlc3RzL2ZpbHRlci9maWx0ZXIuYwpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGU5NzExOS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdGVzdHMvZmlsdGVyL2ZpbHRlci5jCisrKyAvZGV2L251bGwKQEAgLTEsMTMwICswLDAgQEAKLSNpbmNsdWRlIDxzdGRsaWIuaD4KLSNpbmNsdWRlIDxzdGRpby5oPgotCi0jaW5jbHVkZSA8RUdML2VnbC5oPgotI2luY2x1ZGUgPEdMRVMvZ2wuaD4KLSNpbmNsdWRlIDxHTEVTL2dsZXh0Lmg+Ci0KLWludCBtYWluKGludCBhcmdjLCBjaGFyKiogYXJndikKLXsKLSAgICBpZiAoYXJnYyE9MiAmJiBhcmdjIT0zKSB7Ci0gICAgICAgIHByaW50ZigidXNhZ2U6ICVzIDwwLTY+IFtwYnVmZmVyXVxuIiwgYXJndlswXSk7Ci0gICAgICAgIHJldHVybiAwOwotICAgIH0KLSAgICAKLSAgICBjb25zdCBpbnQgdGVzdCA9IGF0b2koYXJndlsxXSk7Ci0gICAgaW50IHVzZVBidWZmZXIgPSBhcmdjPT0zICYmICFzdHJjbXAoYXJndlsyXSwgInBidWZmZXIiKTsKLSAgICBFR0xpbnQgc19jb25maWdBdHRyaWJzW10gPSB7Ci0gICAgICAgICBFR0xfU1VSRkFDRV9UWVBFLCBFR0xfUEJVRkZFUl9CSVR8RUdMX1dJTkRPV19CSVQsCi0gICAgICAgICBFR0xfUkVEX1NJWkUsICAgICAgIDUsCi0gICAgICAgICBFR0xfR1JFRU5fU0laRSwgICAgIDYsCi0gICAgICAgICBFR0xfQkxVRV9TSVpFLCAgICAgIDUsCi0gICAgICAgICBFR0xfTk9ORQotICAgICB9OwotICAgICAKLSAgICAgRUdMaW50IG51bUNvbmZpZ3MgPSAtMTsKLSAgICAgRUdMaW50IG1ham9yVmVyc2lvbjsKLSAgICAgRUdMaW50IG1pbm9yVmVyc2lvbjsKLSAgICAgRUdMQ29uZmlnIGNvbmZpZzsKLSAgICAgRUdMQ29udGV4dCBjb250ZXh0OwotICAgICBFR0xTdXJmYWNlIHN1cmZhY2U7Ci0gICAgIEVHTGludCB3LCBoOwotICAgICAKLSAgICAgRUdMRGlzcGxheSBkcHk7Ci0KLSAgICAgZHB5ID0gZWdsR2V0RGlzcGxheShFR0xfREVGQVVMVF9ESVNQTEFZKTsKLSAgICAgZWdsSW5pdGlhbGl6ZShkcHksICZtYWpvclZlcnNpb24sICZtaW5vclZlcnNpb24pOwotICAgICBlZ2xDaG9vc2VDb25maWcoZHB5LCBzX2NvbmZpZ0F0dHJpYnMsICZjb25maWcsIDEsICZudW1Db25maWdzKTsKLSAgICAgaWYgKCF1c2VQYnVmZmVyKSB7Ci0gICAgICAgICBzdXJmYWNlID0gZWdsQ3JlYXRlV2luZG93U3VyZmFjZShkcHksIGNvbmZpZywKLSAgICAgICAgICAgICAgICAgYW5kcm9pZF9jcmVhdGVEaXNwbGF5U3VyZmFjZSgpLCBOVUxMKTsKLSAgICAgfSBlbHNlIHsKLSAgICAgICAgIHByaW50ZigidXNpbmcgcGJ1ZmZlclxuIik7Ci0gICAgICAgICBFR0xpbnQgYXR0cmlic1tdID0geyBFR0xfV0lEVEgsIDMyMCwgRUdMX0hFSUdIVCwgNDgwLCBFR0xfTk9ORSB9OwotICAgICAgICAgc3VyZmFjZSA9IGVnbENyZWF0ZVBidWZmZXJTdXJmYWNlKGRweSwgY29uZmlnLCBhdHRyaWJzKTsKLSAgICAgICAgIGlmIChzdXJmYWNlID09IEVHTF9OT19TVVJGQUNFKSB7Ci0gICAgICAgICAgICAgcHJpbnRmKCJlZ2xDcmVhdGVQYnVmZmVyU3VyZmFjZSBlcnJvciAleFxuIiwgZWdsR2V0RXJyb3IoKSk7Ci0gICAgICAgICB9Ci0gICAgIH0KLSAgICAgY29udGV4dCA9IGVnbENyZWF0ZUNvbnRleHQoZHB5LCBjb25maWcsIE5VTEwsIE5VTEwpOwotICAgICBlZ2xNYWtlQ3VycmVudChkcHksIHN1cmZhY2UsIHN1cmZhY2UsIGNvbnRleHQpOyAgIAotICAgICBlZ2xRdWVyeVN1cmZhY2UoZHB5LCBzdXJmYWNlLCBFR0xfV0lEVEgsICZ3KTsKLSAgICAgZWdsUXVlcnlTdXJmYWNlKGRweSwgc3VyZmFjZSwgRUdMX0hFSUdIVCwgJmgpOwotICAgICBHTGludCBkaW0gPSB3PGggPyB3IDogaDsKLQotICAgICBnbENsZWFyKEdMX0NPTE9SX0JVRkZFUl9CSVQpOwotCi0gICAgIEdMaW50IGNyb3BbNF0gPSB7IDAsIDQsIDQsIC00IH07Ci0gICAgIGdsQmluZFRleHR1cmUoR0xfVEVYVFVSRV8yRCwgMCk7Ci0gICAgIGdsVGV4UGFyYW1ldGVyaXYoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9DUk9QX1JFQ1RfT0VTLCBjcm9wKTsKLSAgICAgZ2xUZXhQYXJhbWV0ZXJ4KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfTUFHX0ZJTFRFUiwgR0xfTElORUFSKTsKLSAgICAgZ2xUZXhQYXJhbWV0ZXJ4KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfTUlOX0ZJTFRFUiwgR0xfTElORUFSKTsKLSAgICAgZ2xUZXhFbnZ4KEdMX1RFWFRVUkVfRU5WLCBHTF9URVhUVVJFX0VOVl9NT0RFLCBHTF9SRVBMQUNFKTsKLSAgICAgZ2xFbmFibGUoR0xfVEVYVFVSRV8yRCk7Ci0gICAgIGdsQ29sb3I0ZigxLDEsMSwxKTsKLQotICAgICAvLyBwYWNraW5nIGlzIGFsd2F5cyA0Ci0gICAgIHVpbnQ4X3QgdDhbXSAgPSB7IAotICAgICAgICAgICAgIDB4MDAsIDB4NTUsIDB4MDAsIDB4NTUsIAotICAgICAgICAgICAgIDB4QUEsIDB4RkYsIDB4QUEsIDB4RkYsCi0gICAgICAgICAgICAgMHgwMCwgMHg1NSwgMHgwMCwgMHg1NSwgCi0gICAgICAgICAgICAgMHhBQSwgMHhGRiwgMHhBQSwgMHhGRiAgfTsKLQotICAgICB1aW50MTZfdCB0MTZbXSAgPSB7IAotICAgICAgICAgICAgIDB4MDAwMCwgMHg1NTU1LCAweDAwMDAsIDB4NTU1NSwgCi0gICAgICAgICAgICAgMHhBQUFBLCAweEZGRkYsIDB4QUFBQSwgMHhGRkZGLAotICAgICAgICAgICAgIDB4MDAwMCwgMHg1NTU1LCAweDAwMDAsIDB4NTU1NSwgCi0gICAgICAgICAgICAgMHhBQUFBLCAweEZGRkYsIDB4QUFBQSwgMHhGRkZGICB9OwotCi0gICAgIHVpbnQxNl90IHQ1NTUxW10gID0geyAKLSAgICAgICAgICAgICAweDAwMDAsIDB4RkZGRiwgMHgwMDAwLCAweEZGRkYsIAotICAgICAgICAgICAgIDB4RkZGRiwgMHgwMDAwLCAweEZGRkYsIDB4MDAwMCwKLSAgICAgICAgICAgICAweDAwMDAsIDB4RkZGRiwgMHgwMDAwLCAweEZGRkYsIAotICAgICAgICAgICAgIDB4RkZGRiwgMHgwMDAwLCAweEZGRkYsIDB4MDAwMCAgfTsKLQotICAgICB1aW50MzJfdCB0MzJbXSAgPSB7IAotICAgICAgICAgICAgIDB4RkYwMDAwMDAsIDB4RkYwMDAwRkYsIDB4RkYwMEZGMDAsIDB4RkZGRjAwMDAsIAotICAgICAgICAgICAgIDB4RkYwMEZGMDAsIDB4RkZGRjAwMDAsIDB4RkYwMDAwMDAsIDB4RkYwMDAwRkYsIAotICAgICAgICAgICAgIDB4RkYwMEZGRkYsIDB4RkYwMEZGMDAsIDB4MDBGRjAwRkYsIDB4RkZGRkZGMDAsIAotICAgICAgICAgICAgIDB4RkYwMDAwMDAsIDB4RkZGRjAwRkYsIDB4RkYwMEZGRkYsIDB4RkZGRkZGRkYKLSAgICAgfTsKLQotICAgICBzd2l0Y2godGVzdCkgCi0gICAgIHsKLSAgICAgY2FzZSAxOgotICAgICAgICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsIEdMX0xVTUlOQU5DRSwKLSAgICAgICAgICAgICAgICAgNCwgNCwgMCwgR0xfTFVNSU5BTkNFLCBHTF9VTlNJR05FRF9CWVRFLCB0OCk7Ci0gICAgICAgICBicmVhazsKLSAgICAgY2FzZSAyOgotICAgICAgICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsIEdMX1JHQiwKLSAgICAgICAgICAgICAgICAgNCwgNCwgMCwgR0xfUkdCLCBHTF9VTlNJR05FRF9TSE9SVF81XzZfNSwgdDE2KTsKLSAgICAgICAgIGJyZWFrOwotICAgICBjYXNlIDM6Ci0gICAgICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwgR0xfUkdCQSwKLSAgICAgICAgICAgICAgICAgNCwgNCwgMCwgR0xfUkdCQSwgR0xfVU5TSUdORURfU0hPUlRfNF80XzRfNCwgdDE2KTsKLSAgICAgICAgIGJyZWFrOwotICAgICBjYXNlIDQ6Ci0gICAgICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwgR0xfTFVNSU5BTkNFX0FMUEhBLAotICAgICAgICAgICAgICAgICA0LCA0LCAwLCBHTF9MVU1JTkFOQ0VfQUxQSEEsIEdMX1VOU0lHTkVEX0JZVEUsIHQxNik7Ci0gICAgICAgICBicmVhazsKLSAgICAgY2FzZSA1OgotICAgICAgICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsIEdMX1JHQkEsCi0gICAgICAgICAgICAgICAgIDQsIDQsIDAsIEdMX1JHQkEsIEdMX1VOU0lHTkVEX1NIT1JUXzVfNV81XzEsIHQ1NTUxKTsKLSAgICAgICAgIGJyZWFrOwotICAgICBjYXNlIDY6Ci0gICAgICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwgR0xfUkdCQSwKLSAgICAgICAgICAgICAgICAgNCwgNCwgMCwgR0xfUkdCQSwgR0xfVU5TSUdORURfQllURSwgdDMyKTsKLSAgICAgICAgIGJyZWFrOwotICAgICB9Ci0KLSAgICAgZ2xEcmF3VGV4aU9FUygwLCAwLCAwLCBkaW0sIGRpbSk7Ci0KLSAgICAgaWYgKCF1c2VQYnVmZmVyKSB7Ci0gICAgICAgICBlZ2xTd2FwQnVmZmVycyhkcHksIHN1cmZhY2UpOwotICAgICB9IGVsc2UgewotICAgICAgICAgZ2xGaW5pc2goKTsKLSAgICAgfQotICAgICAKLSAgICAgZWdsVGVybWluYXRlKGRweSk7Ci0gICAgIHJldHVybiAwOwotfQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL2ZpbmlzaC9BbmRyb2lkLm1rIGIvb3BlbmdsL3Rlc3RzL2ZpbmlzaC9BbmRyb2lkLm1rCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyNjgzNmMxLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy9maW5pc2gvQW5kcm9pZC5taworKysgL2Rldi9udWxsCkBAIC0xLDE3ICswLDAgQEAKLUxPQ0FMX1BBVEg6PSAkKGNhbGwgbXktZGlyKQotaW5jbHVkZSAkKENMRUFSX1ZBUlMpCi0KLUxPQ0FMX1NSQ19GSUxFUzo9IFwKLQlmaW5pc2guYwotCi1MT0NBTF9TSEFSRURfTElCUkFSSUVTIDo9IFwKLQlsaWJjdXRpbHMgXAotICAgIGxpYkVHTCBcCi0gICAgbGliR0xFU3YxX0NNIFwKLSAgICBsaWJ1aQotCi1MT0NBTF9NT0RVTEU6PSB0ZXN0LW9wZW5nbC1maW5pc2gKLQotTE9DQUxfTU9EVUxFX1RBR1MgOj0gdGVzdHMKLQotaW5jbHVkZSAkKEJVSUxEX0VYRUNVVEFCTEUpCmRpZmYgLS1naXQgYS9vcGVuZ2wvdGVzdHMvZmluaXNoL2ZpbmlzaC5jIGIvb3BlbmdsL3Rlc3RzL2ZpbmlzaC9maW5pc2guYwpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNDVmYzc1OC4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdGVzdHMvZmluaXNoL2ZpbmlzaC5jCisrKyAvZGV2L251bGwKQEAgLTEsMjI0ICswLDAgQEAKLS8qCi0gKiBDb3B5cmlnaHQgKEMpIDIwMDcgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotICoKLSAqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwotICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgotICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0Ci0gKgotICogICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKLSAqCi0gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCi0gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAotICogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCi0gKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCi0gKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSAqLwotCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8c3RkaW8uaD4KLSNpbmNsdWRlIDx0aW1lLmg+Ci0jaW5jbHVkZSA8c2NoZWQuaD4KLSNpbmNsdWRlIDxzeXMvcmVzb3VyY2UuaD4KLQotI2luY2x1ZGUgPEVHTC9lZ2wuaD4KLSNpbmNsdWRlIDxHTEVTL2dsLmg+Ci0jaW5jbHVkZSA8R0xFUy9nbGV4dC5oPgotCi0KLWxvbmcgbG9uZyBzeXN0ZW1UaW1lKCkKLXsKLSAgICBzdHJ1Y3QgdGltZXNwZWMgdDsKLSAgICB0LnR2X3NlYyA9IHQudHZfbnNlYyA9IDA7Ci0gICAgY2xvY2tfZ2V0dGltZShDTE9DS19NT05PVE9OSUMsICZ0KTsKLSAgICByZXR1cm4gKGxvbmcgbG9uZykodC50dl9zZWMpKjEwMDAwMDAwMDBMTCArIHQudHZfbnNlYzsKLX0KLQotaW50IG1haW4oaW50IGFyZ2MsIGNoYXIqKiBhcmd2KQotewotICAgIEVHTGludCBzX2NvbmZpZ0F0dHJpYnNbXSA9IHsKLSAgICAgICAgIEVHTF9SRURfU0laRSwgICAgICAgNSwKLSAgICAgICAgIEVHTF9HUkVFTl9TSVpFLCAgICAgNiwKLSAgICAgICAgIEVHTF9CTFVFX1NJWkUsICAgICAgNSwKLSAgICAgICAgIEVHTF9OT05FCi0gICAgIH07Ci0gICAgIAotICAgICBFR0xpbnQgbnVtQ29uZmlncyA9IC0xOwotICAgICBFR0xpbnQgbWFqb3JWZXJzaW9uOwotICAgICBFR0xpbnQgbWlub3JWZXJzaW9uOwotICAgICBFR0xDb25maWcgY29uZmlnOwotICAgICBFR0xDb250ZXh0IGNvbnRleHQ7Ci0gICAgIEVHTFN1cmZhY2Ugc3VyZmFjZTsKLSAgICAgRUdMaW50IHcsIGg7Ci0gICAgIAotICAgICBFR0xEaXNwbGF5IGRweTsKLQotICAgICBkcHkgPSBlZ2xHZXREaXNwbGF5KEVHTF9ERUZBVUxUX0RJU1BMQVkpOwotICAgICBlZ2xJbml0aWFsaXplKGRweSwgJm1ham9yVmVyc2lvbiwgJm1pbm9yVmVyc2lvbik7Ci0gICAgIGVnbENob29zZUNvbmZpZyhkcHksIHNfY29uZmlnQXR0cmlicywgJmNvbmZpZywgMSwgJm51bUNvbmZpZ3MpOwotICAgICBzdXJmYWNlID0gZWdsQ3JlYXRlV2luZG93U3VyZmFjZShkcHksIGNvbmZpZywgCi0gICAgICAgICAgICAgYW5kcm9pZF9jcmVhdGVEaXNwbGF5U3VyZmFjZSgpLCBOVUxMKTsKLSAgICAgY29udGV4dCA9IGVnbENyZWF0ZUNvbnRleHQoZHB5LCBjb25maWcsIE5VTEwsIE5VTEwpOwotICAgICBlZ2xNYWtlQ3VycmVudChkcHksIHN1cmZhY2UsIHN1cmZhY2UsIGNvbnRleHQpOyAgIAotICAgICBlZ2xRdWVyeVN1cmZhY2UoZHB5LCBzdXJmYWNlLCBFR0xfV0lEVEgsICZ3KTsKLSAgICAgZWdsUXVlcnlTdXJmYWNlKGRweSwgc3VyZmFjZSwgRUdMX0hFSUdIVCwgJmgpOwotICAgICBHTGludCBkaW0gPSB3PGggPyB3IDogaDsKLQotICAgICBnbEJpbmRUZXh0dXJlKEdMX1RFWFRVUkVfMkQsIDApOwotICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NQUdfRklMVEVSLCBHTF9ORUFSRVNUKTsKLSAgICAgZ2xUZXhQYXJhbWV0ZXJ4KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfTUlOX0ZJTFRFUiwgR0xfTkVBUkVTVCk7Ci0gICAgIGdsVGV4RW52eChHTF9URVhUVVJFX0VOViwgR0xfVEVYVFVSRV9FTlZfTU9ERSwgR0xfUkVQTEFDRSk7Ci0gICAgIGdsRW5hYmxlKEdMX1RFWFRVUkVfMkQpOwotICAgICBnbENvbG9yNGYoMSwxLDEsMSk7Ci0gICAgIGdsRGlzYWJsZShHTF9ESVRIRVIpOwotICAgICBnbFNoYWRlTW9kZWwoR0xfRkxBVCk7Ci0KLSAgICAgbG9uZyBsb25nIG5vdywgdDsKLSAgICAgaW50IGk7Ci0KLSAgICAgY2hhciogdGV4ZWxzID0gbWFsbG9jKDUxMio1MTIqMik7Ci0gICAgIG1lbXNldCh0ZXhlbHMsMHhGRiw1MTIqNTEyKjIpOwotICAgICAKLSAgICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsIEdMX1JHQiwKLSAgICAgICAgICAgICA1MTIsIDUxMiwgMCwgR0xfUkdCLCBHTF9VTlNJR05FRF9TSE9SVF81XzZfNSwgdGV4ZWxzKTsKLQotICAgICBjaGFyKiBkc3QgPSBtYWxsb2MoMzIwKjQ4MCoyKTsKLSAgICAgbWVtc2V0KGRzdCwgMCwgMzIwKjQ4MCoyKTsKLSAgICAgcHJpbnRmKCIzMDcyMDAgYnl0ZXMgbWVtY3B5XG4iKTsKLSAgICAgZm9yIChpPTAgOyBpPDQgOyBpKyspIHsKLSAgICAgICAgIG5vdyA9IHN5c3RlbVRpbWUoKTsKLSAgICAgICAgIG1lbWNweShkc3QsIHRleGVscywgMzIwKjQ4MCoyKTsKLSAgICAgICAgIHQgPSBzeXN0ZW1UaW1lKCk7Ci0gICAgICAgICBwcmludGYoIm1lbWNweSgpIHRpbWUgPSAlbGx1IHVzXG4iLCAodC1ub3cpLzEwMDApOwotICAgICAgICAgZmZsdXNoKHN0ZG91dCk7Ci0gICAgIH0KLSAgICAgZnJlZShkc3QpOwotCi0gICAgIGZyZWUodGV4ZWxzKTsKLQotICAgICBzZXRwcmlvcml0eShQUklPX1BST0NFU1MsIDAsIC0yMCk7Ci0gICAgIAotICAgICBwcmludGYoIjUxMng1MTIgdW5tb2RpZmllZCB0ZXh0dXJlLCA1MTJ4NTEyIGJsaXQ6XG4iKTsKLSAgICAgZ2xDbGVhcihHTF9DT0xPUl9CVUZGRVJfQklUKTsKLSAgICAgZm9yIChpPTAgOyBpPDQgOyBpKyspIHsKLSAgICAgICAgIEdMaW50IGNyb3BbNF0gPSB7IDAsIDUxMiwgNTEyLCAtNTEyIH07Ci0gICAgICAgICBnbFRleFBhcmFtZXRlcml2KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfQ1JPUF9SRUNUX09FUywgY3JvcCk7Ci0gICAgICAgICBub3cgPSBzeXN0ZW1UaW1lKCk7Ci0gICAgICAgICBnbERyYXdUZXhpT0VTKDAsIDAsIDAsIDUxMiwgNTEyKTsKLSAgICAgICAgIGdsRmluaXNoKCk7Ci0gICAgICAgICB0ID0gc3lzdGVtVGltZSgpOwotICAgICAgICAgcHJpbnRmKCJnbEZpbmlzaCgpIHRpbWUgPSAlbGx1IHVzXG4iLCAodC1ub3cpLzEwMDApOwotICAgICAgICAgZmZsdXNoKHN0ZG91dCk7Ci0gICAgICAgICBlZ2xTd2FwQnVmZmVycyhkcHksIHN1cmZhY2UpOwotICAgICB9Ci0gICAgIAotICAgICBwcmludGYoIjUxMng1MTIgdW5tb2RpZmllZCB0ZXh0dXJlLCAxeDEgYmxpdDpcbiIpOwotICAgICBnbENsZWFyKEdMX0NPTE9SX0JVRkZFUl9CSVQpOwotICAgICBmb3IgKGk9MCA7IGk8NCA7IGkrKykgewotICAgICAgICAgR0xpbnQgY3JvcFs0XSA9IHsgMCwgMSwgMSwgLTEgfTsKLSAgICAgICAgIGdsVGV4UGFyYW1ldGVyaXYoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9DUk9QX1JFQ1RfT0VTLCBjcm9wKTsKLSAgICAgICAgIG5vdyA9IHN5c3RlbVRpbWUoKTsKLSAgICAgICAgIGdsRHJhd1RleGlPRVMoMCwgMCwgMCwgMSwgMSk7Ci0gICAgICAgICBnbEZpbmlzaCgpOwotICAgICAgICAgdCA9IHN5c3RlbVRpbWUoKTsKLSAgICAgICAgIHByaW50ZigiZ2xGaW5pc2goKSB0aW1lID0gJWxsdSB1c1xuIiwgKHQtbm93KS8xMDAwKTsKLSAgICAgICAgIGZmbHVzaChzdGRvdXQpOwotICAgICAgICAgZWdsU3dhcEJ1ZmZlcnMoZHB5LCBzdXJmYWNlKTsKLSAgICAgfQotICAgICAKLSAgICAgcHJpbnRmKCI1MTJ4NTEyIHVubW9kaWZpZWQgdGV4dHVyZSwgNTEyeDUxMiBibGl0ICh4Mik6XG4iKTsKLSAgICAgZ2xDbGVhcihHTF9DT0xPUl9CVUZGRVJfQklUKTsKLSAgICAgZm9yIChpPTAgOyBpPDQgOyBpKyspIHsKLSAgICAgICAgIEdMaW50IGNyb3BbNF0gPSB7IDAsIDUxMiwgNTEyLCAtNTEyIH07Ci0gICAgICAgICBnbFRleFBhcmFtZXRlcml2KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfQ1JPUF9SRUNUX09FUywgY3JvcCk7Ci0gICAgICAgICBub3cgPSBzeXN0ZW1UaW1lKCk7Ci0gICAgICAgICBnbERyYXdUZXhpT0VTKDAsIDAsIDAsIDUxMiwgNTEyKTsKLSAgICAgICAgIGdsRHJhd1RleGlPRVMoMCwgMCwgMCwgNTEyLCA1MTIpOwotICAgICAgICAgZ2xGaW5pc2goKTsKLSAgICAgICAgIHQgPSBzeXN0ZW1UaW1lKCk7Ci0gICAgICAgICBwcmludGYoImdsRmluaXNoKCkgdGltZSA9ICVsbHUgdXNcbiIsICh0LW5vdykvMTAwMCk7Ci0gICAgICAgICBmZmx1c2goc3Rkb3V0KTsKLSAgICAgICAgIGVnbFN3YXBCdWZmZXJzKGRweSwgc3VyZmFjZSk7Ci0gICAgIH0KLQotICAgICBwcmludGYoIjUxMng1MTIgdW5tb2RpZmllZCB0ZXh0dXJlLCAxeDEgYmxpdCAoeDIpOlxuIik7Ci0gICAgIGdsQ2xlYXIoR0xfQ09MT1JfQlVGRkVSX0JJVCk7Ci0gICAgIGZvciAoaT0wIDsgaTw0IDsgaSsrKSB7Ci0gICAgICAgICBHTGludCBjcm9wWzRdID0geyAwLCAxLCAxLCAtMSB9OwotICAgICAgICAgZ2xUZXhQYXJhbWV0ZXJpdihHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX0NST1BfUkVDVF9PRVMsIGNyb3ApOwotICAgICAgICAgbm93ID0gc3lzdGVtVGltZSgpOwotICAgICAgICAgZ2xEcmF3VGV4aU9FUygwLCAwLCAwLCAxLCAxKTsKLSAgICAgICAgIGdsRHJhd1RleGlPRVMoMCwgMCwgMCwgMSwgMSk7Ci0gICAgICAgICBnbEZpbmlzaCgpOwotICAgICAgICAgdCA9IHN5c3RlbVRpbWUoKTsKLSAgICAgICAgIHByaW50ZigiZ2xGaW5pc2goKSB0aW1lID0gJWxsdSB1c1xuIiwgKHQtbm93KS8xMDAwKTsKLSAgICAgICAgIGZmbHVzaChzdGRvdXQpOwotICAgICAgICAgZWdsU3dhcEJ1ZmZlcnMoZHB5LCBzdXJmYWNlKTsKLSAgICAgfQotCi0gICAgIAotICAgICBwcmludGYoIjUxMng1MTIgKDF4MSB0ZXhlbCBNT0RJRklFRCB0ZXh0dXJlKSwgNTEyeDUxMiBibGl0OlxuIik7Ci0gICAgIGdsQ2xlYXIoR0xfQ09MT1JfQlVGRkVSX0JJVCk7Ci0gICAgIGZvciAoaT0wIDsgaTw0IDsgaSsrKSB7Ci0gICAgICAgICB1aW50MTZfdCBncmVlbiA9IDB4N0UwOwotICAgICAgICAgR0xpbnQgY3JvcFs0XSA9IHsgMCwgNTEyLCA1MTIsIC01MTIgfTsKLSAgICAgICAgIGdsVGV4UGFyYW1ldGVyaXYoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9DUk9QX1JFQ1RfT0VTLCBjcm9wKTsKLSAgICAgICAgIGdsVGV4U3ViSW1hZ2UyRChHTF9URVhUVVJFXzJELCAwLCAwLCAwLCAxLCAxLCBHTF9SR0IsIEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81LCAmZ3JlZW4pOwotICAgICAgICAgbm93ID0gc3lzdGVtVGltZSgpOwotICAgICAgICAgZ2xEcmF3VGV4aU9FUygwLCAwLCAwLCA1MTIsIDUxMik7Ci0gICAgICAgICBnbEZpbmlzaCgpOwotICAgICAgICAgdCA9IHN5c3RlbVRpbWUoKTsKLSAgICAgICAgIHByaW50ZigiZ2xGaW5pc2goKSB0aW1lID0gJWxsdSB1c1xuIiwgKHQtbm93KS8xMDAwKTsKLSAgICAgICAgIGZmbHVzaChzdGRvdXQpOwotICAgICAgICAgZWdsU3dhcEJ1ZmZlcnMoZHB5LCBzdXJmYWNlKTsKLSAgICAgfQotCi0KLSAgICAgaW50MTZfdCB0ZXhlbCA9IDB4RjgwMDsKLSAgICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsIEdMX1JHQiwKLSAgICAgICAgICAgICAxLCAxLCAwLCBHTF9SR0IsIEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81LCAmdGV4ZWwpOwotCi0gICAgIHByaW50ZigiMXgxIHVubW9kaWZpZWQgdGV4dHVyZSwgMXgxIGJsaXQ6XG4iKTsKLSAgICAgZ2xDbGVhcihHTF9DT0xPUl9CVUZGRVJfQklUKTsKLSAgICAgZm9yIChpPTAgOyBpPDQgOyBpKyspIHsKLSAgICAgICAgIEdMaW50IGNyb3BbNF0gPSB7IDAsIDEsIDEsIC0xIH07Ci0gICAgICAgICBnbFRleFBhcmFtZXRlcml2KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfQ1JPUF9SRUNUX09FUywgY3JvcCk7Ci0gICAgICAgICBub3cgPSBzeXN0ZW1UaW1lKCk7Ci0gICAgICAgICBnbERyYXdUZXhpT0VTKDAsIDAsIDAsIDEsIDEpOwotICAgICAgICAgZ2xGaW5pc2goKTsKLSAgICAgICAgIHQgPSBzeXN0ZW1UaW1lKCk7Ci0gICAgICAgICBwcmludGYoImdsRmluaXNoKCkgdGltZSA9ICVsbHUgdXNcbiIsICh0LW5vdykvMTAwMCk7Ci0gICAgICAgICBlZ2xTd2FwQnVmZmVycyhkcHksIHN1cmZhY2UpOwotICAgICB9Ci0KLSAgICAgcHJpbnRmKCIxeDEgdW5tb2RpZmllZCB0ZXh0dXJlLCA1MTJ4NTEyIGJsaXQ6XG4iKTsKLSAgICAgZ2xDbGVhcihHTF9DT0xPUl9CVUZGRVJfQklUKTsKLSAgICAgZm9yIChpPTAgOyBpPDQgOyBpKyspIHsKLSAgICAgICAgIEdMaW50IGNyb3BbNF0gPSB7IDAsIDEsIDEsIC0xIH07Ci0gICAgICAgICBnbFRleFBhcmFtZXRlcml2KEdMX1RFWFRVUkVfMkQsIEdMX1RFWFRVUkVfQ1JPUF9SRUNUX09FUywgY3JvcCk7Ci0gICAgICAgICBub3cgPSBzeXN0ZW1UaW1lKCk7Ci0gICAgICAgICBnbERyYXdUZXhpT0VTKDAsIDAsIDAsIDUxMiwgNTEyKTsKLSAgICAgICAgIGdsRmluaXNoKCk7Ci0gICAgICAgICB0ID0gc3lzdGVtVGltZSgpOwotICAgICAgICAgcHJpbnRmKCJnbEZpbmlzaCgpIHRpbWUgPSAlbGx1IHVzXG4iLCAodC1ub3cpLzEwMDApOwotICAgICAgICAgZmZsdXNoKHN0ZG91dCk7Ci0gICAgICAgICBlZ2xTd2FwQnVmZmVycyhkcHksIHN1cmZhY2UpOwotICAgICB9Ci0KLSAgICAgcHJpbnRmKCIxeDEgKDF4MSB0ZXhlbCBNT0RJRklFRCB0ZXh0dXJlKSwgNTEyeDUxMiBibGl0OlxuIik7Ci0gICAgIGdsQ2xlYXIoR0xfQ09MT1JfQlVGRkVSX0JJVCk7Ci0gICAgIGZvciAoaT0wIDsgaTw0IDsgaSsrKSB7Ci0gICAgICAgICB1aW50MTZfdCBncmVlbiA9IDB4N0UwOwotICAgICAgICAgR0xpbnQgY3JvcFs0XSA9IHsgMCwgMSwgMSwgLTEgfTsKLSAgICAgICAgIGdsVGV4UGFyYW1ldGVyaXYoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9DUk9QX1JFQ1RfT0VTLCBjcm9wKTsKLSAgICAgICAgIGdsVGV4U3ViSW1hZ2UyRChHTF9URVhUVVJFXzJELCAwLCAwLCAwLCAxLCAxLCBHTF9SR0IsIEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81LCAmZ3JlZW4pOwotICAgICAgICAgbm93ID0gc3lzdGVtVGltZSgpOwotICAgICAgICAgZ2xEcmF3VGV4aU9FUygwLCAwLCAwLCAxLCAxKTsKLSAgICAgICAgIGdsRmluaXNoKCk7Ci0gICAgICAgICB0ID0gc3lzdGVtVGltZSgpOwotICAgICAgICAgcHJpbnRmKCJnbEZpbmlzaCgpIHRpbWUgPSAlbGx1IHVzXG4iLCAodC1ub3cpLzEwMDApOwotICAgICAgICAgZmZsdXNoKHN0ZG91dCk7Ci0gICAgICAgICBlZ2xTd2FwQnVmZmVycyhkcHksIHN1cmZhY2UpOwotICAgICB9Ci0KLSAgICAgcmV0dXJuIDA7Ci19CmRpZmYgLS1naXQgYS9vcGVuZ2wvdGVzdHMvdGV4dHVyZXMvQW5kcm9pZC5tayBiL29wZW5nbC90ZXN0cy90ZXh0dXJlcy9BbmRyb2lkLm1rCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBhOGM2MjIwLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy90ZXh0dXJlcy9BbmRyb2lkLm1rCisrKyAvZGV2L251bGwKQEAgLTEsMTcgKzAsMCBAQAotTE9DQUxfUEFUSDo9ICQoY2FsbCBteS1kaXIpCi1pbmNsdWRlICQoQ0xFQVJfVkFSUykKLQotTE9DQUxfU1JDX0ZJTEVTOj0gXAotCXRleHR1cmVzLmMKLQotTE9DQUxfU0hBUkVEX0xJQlJBUklFUyA6PSBcCi0JbGliY3V0aWxzIFwKLSAgICBsaWJFR0wgXAotICAgIGxpYkdMRVN2MV9DTSBcCi0gICAgbGlidWkKLQotTE9DQUxfTU9EVUxFOj0gdGVzdC1vcGVuZ2wtdGV4dHVyZXMKLQotTE9DQUxfTU9EVUxFX1RBR1MgOj0gdGVzdHMKLQotaW5jbHVkZSAkKEJVSUxEX0VYRUNVVEFCTEUpCmRpZmYgLS1naXQgYS9vcGVuZ2wvdGVzdHMvdGV4dHVyZXMvdGV4dHVyZXMuYyBiL29wZW5nbC90ZXN0cy90ZXh0dXJlcy90ZXh0dXJlcy5jCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyMTQyOTFiLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90ZXN0cy90ZXh0dXJlcy90ZXh0dXJlcy5jCisrKyAvZGV2L251bGwKQEAgLTEsMTA5ICswLDAgQEAKLS8qCi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotI2luY2x1ZGUgPHN0ZGxpYi5oPgotI2luY2x1ZGUgPHN0ZGlvLmg+Ci0KLSNpbmNsdWRlIDxFR0wvZWdsLmg+Ci0jaW5jbHVkZSA8R0xFUy9nbC5oPgotI2luY2x1ZGUgPEdMRVMvZ2xleHQuaD4KLQotaW50IG1haW4oaW50IGFyZ2MsIGNoYXIqKiBhcmd2KQotewotICAgIEVHTGludCBzX2NvbmZpZ0F0dHJpYnNbXSA9IHsKLSAgICAgICAgIEVHTF9SRURfU0laRSwgICAgICAgNSwKLSAgICAgICAgIEVHTF9HUkVFTl9TSVpFLCAgICAgNiwKLSAgICAgICAgIEVHTF9CTFVFX1NJWkUsICAgICAgNSwKLSAgICAgICAgIEVHTF9OT05FCi0gICAgIH07Ci0gICAgIAotICAgICBFR0xpbnQgbnVtQ29uZmlncyA9IC0xOwotICAgICBFR0xpbnQgbWFqb3JWZXJzaW9uOwotICAgICBFR0xpbnQgbWlub3JWZXJzaW9uOwotICAgICBFR0xDb25maWcgY29uZmlnOwotICAgICBFR0xDb250ZXh0IGNvbnRleHQ7Ci0gICAgIEVHTFN1cmZhY2Ugc3VyZmFjZTsKLSAgICAgRUdMaW50IHcsIGg7Ci0gICAgIAotICAgICBFR0xEaXNwbGF5IGRweTsKLQotICAgICBkcHkgPSBlZ2xHZXREaXNwbGF5KEVHTF9ERUZBVUxUX0RJU1BMQVkpOwotICAgICBlZ2xJbml0aWFsaXplKGRweSwgJm1ham9yVmVyc2lvbiwgJm1pbm9yVmVyc2lvbik7Ci0gICAgIGVnbENob29zZUNvbmZpZyhkcHksIHNfY29uZmlnQXR0cmlicywgJmNvbmZpZywgMSwgJm51bUNvbmZpZ3MpOwotICAgICBzdXJmYWNlID0gZWdsQ3JlYXRlV2luZG93U3VyZmFjZShkcHksIGNvbmZpZywKLSAgICAgICAgICAgICBhbmRyb2lkX2NyZWF0ZURpc3BsYXlTdXJmYWNlKCksIE5VTEwpOwotICAgICBjb250ZXh0ID0gZWdsQ3JlYXRlQ29udGV4dChkcHksIGNvbmZpZywgTlVMTCwgTlVMTCk7Ci0gICAgIGVnbE1ha2VDdXJyZW50KGRweSwgc3VyZmFjZSwgc3VyZmFjZSwgY29udGV4dCk7ICAgCi0gICAgIGVnbFF1ZXJ5U3VyZmFjZShkcHksIHN1cmZhY2UsIEVHTF9XSURUSCwgJncpOwotICAgICBlZ2xRdWVyeVN1cmZhY2UoZHB5LCBzdXJmYWNlLCBFR0xfSEVJR0hULCAmaCk7Ci0gICAgIEdMaW50IGRpbSA9IHc8aCA/IHcgOiBoOwotCi0KLSAgICAgR0xpbnQgY3JvcFs0XSA9IHsgMCwgNCwgNCwgLTQgfTsKLSAgICAgZ2xCaW5kVGV4dHVyZShHTF9URVhUVVJFXzJELCAwKTsKLSAgICAgZ2xUZXhQYXJhbWV0ZXJpdihHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX0NST1BfUkVDVF9PRVMsIGNyb3ApOwotICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NQUdfRklMVEVSLCBHTF9MSU5FQVIpOwotICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NSU5fRklMVEVSLCBHTF9MSU5FQVIpOwotICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9XUkFQX1MsIEdMX0NMQU1QX1RPX0VER0UpOwotICAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9XUkFQX1QsIEdMX0NMQU1QX1RPX0VER0UpOwotICAgICBnbFRleEVudngoR0xfVEVYVFVSRV9FTlYsIEdMX1RFWFRVUkVfRU5WX01PREUsIEdMX1JFUExBQ0UpOwotICAgICBnbEVuYWJsZShHTF9URVhUVVJFXzJEKTsKLSAgICAgZ2xDb2xvcjRmKDEsMSwxLDEpOwotCi0gICAgIC8vIHBhY2tpbmcgaXMgYWx3YXlzIDQKLSAgICAgdWludDhfdCB0OFtdICA9IHsgCi0gICAgICAgICAgICAgMHgwMCwgMHg1NSwgMHgwMCwgMHg1NSwgCi0gICAgICAgICAgICAgMHhBQSwgMHhGRiwgMHhBQSwgMHhGRiwKLSAgICAgICAgICAgICAweDAwLCAweDU1LCAweDAwLCAweDU1LCAKLSAgICAgICAgICAgICAweEFBLCAweEZGLCAweEFBLCAweEZGICB9OwotCi0gICAgIHVpbnQxNl90IHQxNltdICA9IHsgCi0gICAgICAgICAgICAgMHgwMDAwLCAweDU1NTUsIDB4MDAwMCwgMHg1NTU1LCAKLSAgICAgICAgICAgICAweEFBQUEsIDB4RkZGRiwgMHhBQUFBLCAweEZGRkYsCi0gICAgICAgICAgICAgMHgwMDAwLCAweDU1NTUsIDB4MDAwMCwgMHg1NTU1LCAKLSAgICAgICAgICAgICAweEFBQUEsIDB4RkZGRiwgMHhBQUFBLCAweEZGRkYgIH07Ci0KLSAgICAgdWludDE2X3QgdDU1NTFbXSAgPSB7IAotICAgICAgICAgICAgIDB4MDAwMCwgMHhGRkZGLCAweDAwMDAsIDB4RkZGRiwgCi0gICAgICAgICAgICAgMHhGRkZGLCAweDAwMDAsIDB4RkZGRiwgMHgwMDAwLAotICAgICAgICAgICAgIDB4MDAwMCwgMHhGRkZGLCAweDAwMDAsIDB4RkZGRiwgCi0gICAgICAgICAgICAgMHhGRkZGLCAweDAwMDAsIDB4RkZGRiwgMHgwMDAwICB9OwotCi0gICAgIHVpbnQzMl90IHQzMltdICA9IHsgCi0gICAgICAgICAgICAgMHhGRjAwMDAwMCwgMHhGRjAwMDBGRiwgMHhGRjAwRkYwMCwgMHhGRkZGMDAwMCwgCi0gICAgICAgICAgICAgMHhGRjAwRkYwMCwgMHhGRkZGMDAwMCwgMHhGRjAwMDAwMCwgMHhGRjAwMDBGRiwgCi0gICAgICAgICAgICAgMHhGRjAwRkZGRiwgMHhGRjAwRkYwMCwgMHgwMEZGMDBGRiwgMHhGRkZGRkYwMCwgCi0gICAgICAgICAgICAgMHhGRjAwMDAwMCwgMHhGRkZGMDBGRiwgMHhGRjAwRkZGRiwgMHhGRkZGRkZGRgotICAgICB9OwotCi0KLSAgICAgZ2xDbGVhcihHTF9DT0xPUl9CVUZGRVJfQklUKTsKLSAgICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsIEdMX0xVTUlOQU5DRSwgNCwgNCwgMCwgR0xfTFVNSU5BTkNFLCBHTF9VTlNJR05FRF9CWVRFLCB0OCk7Ci0gICAgIGdsRHJhd1RleGlPRVMoMCwgMCwgMCwgZGltLzIsIGRpbS8yKTsKLQotICAgICBnbFRleEltYWdlMkQoR0xfVEVYVFVSRV8yRCwgMCwgR0xfUkdCLCA0LCA0LCAwLCBHTF9SR0IsIEdMX1VOU0lHTkVEX1NIT1JUXzVfNl81LCB0MTYpOwotICAgICBnbERyYXdUZXhpT0VTKGRpbS8yLCAwLCAwLCBkaW0vMiwgZGltLzIpOwotCi0gICAgIGdsVGV4SW1hZ2UyRChHTF9URVhUVVJFXzJELCAwLCBHTF9SR0JBLCA0LCA0LCAwLCBHTF9SR0JBLCBHTF9VTlNJR05FRF9TSE9SVF80XzRfNF80LCB0MTYpOwotICAgICBnbERyYXdUZXhpT0VTKDAsIGRpbS8yLCAwLCBkaW0vMiwgZGltLzIpOwotCi0gICAgIGdsVGV4SW1hZ2UyRChHTF9URVhUVVJFXzJELCAwLCBHTF9SR0JBLCA0LCA0LCAwLCBHTF9SR0JBLCBHTF9VTlNJR05FRF9CWVRFLCB0MzIpOwotICAgICBnbERyYXdUZXhpT0VTKGRpbS8yLCBkaW0vMiwgMCwgZGltLzIsIGRpbS8yKTsKLQotICAgICBlZ2xTd2FwQnVmZmVycyhkcHksIHN1cmZhY2UpOwotICAgICByZXR1cm4gMDsKLX0KZGlmZiAtLWdpdCBhL29wZW5nbC90ZXN0cy90cml0ZXgvQW5kcm9pZC5tayBiL29wZW5nbC90ZXN0cy90cml0ZXgvQW5kcm9pZC5tawpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNWNkMWYwNC4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdGVzdHMvdHJpdGV4L0FuZHJvaWQubWsKKysrIC9kZXYvbnVsbApAQCAtMSwxNyArMCwwIEBACi1MT0NBTF9QQVRIOj0gJChjYWxsIG15LWRpcikKLWluY2x1ZGUgJChDTEVBUl9WQVJTKQotCi1MT0NBTF9TUkNfRklMRVM6PSBcCi0JdHJpdGV4LmMKLQotTE9DQUxfU0hBUkVEX0xJQlJBUklFUyA6PSBcCi0JbGliY3V0aWxzIFwKLSAgICBsaWJFR0wgXAotICAgIGxpYkdMRVN2MV9DTSBcCi0gICAgbGlidWkKLQotTE9DQUxfTU9EVUxFOj0gdGVzdC1vcGVuZ2wtdHJpdGV4Ci0KLUxPQ0FMX01PRFVMRV9UQUdTIDo9IHRlc3RzCi0KLWluY2x1ZGUgJChCVUlMRF9FWEVDVVRBQkxFKQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rlc3RzL3RyaXRleC90cml0ZXguYyBiL29wZW5nbC90ZXN0cy90cml0ZXgvdHJpdGV4LmMKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDYwYTdmZWIuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rlc3RzL3RyaXRleC90cml0ZXguYworKysgL2Rldi9udWxsCkBAIC0xLDI3MyArMCwwIEBACi0vLyBDYWxscyBnbERyYXdFbGVtZW50cygpIHRoZSBudW1iZXIgb2YgdGltZXMgc3BlY2lmaWVkIGJ5Ci0vLyBJVEVSQVRJT05TLiBTaG91bGQgZHJhdyBhIGNoZWNrZXJib2FyZCBvbiB0aGUgc2NyZWVuIGFmdGVyCi0vLyBhIGZldyBzZWNvbmRzLgotLy8KLS8vIFBvcnRlZCBmcm9tIGEgSmF2YSB2ZXJzaW9uIGJ5IEdvb2dsZS4KLQotI2luY2x1ZGUgPEVHTC9lZ2wuaD4NCi0jaW5jbHVkZSA8R0xFUy9nbC5oPgotCi0jaW5jbHVkZSA8c3RkaW8uaD4NCi0jaW5jbHVkZSA8c3RkbGliLmg+Ci0jaW5jbHVkZSA8bWF0aC5oPgotDQotRUdMRGlzcGxheSBlZ2xEaXNwbGF5Ow0KLUVHTFN1cmZhY2UgZWdsU3VyZmFjZTsNCi1FR0xDb250ZXh0IGVnbENvbnRleHQ7DQotR0x1aW50IHRleHR1cmU7DQotDQotI2RlZmluZSBGSVhFRF9PTkUgMHgxMDAwMAotI2RlZmluZSBJVEVSQVRJT05TIDUwDQotDQotaW50IGluaXRfZ2xfc3VyZmFjZSh2b2lkKTsNCi12b2lkIGZyZWVfZ2xfc3VyZmFjZSh2b2lkKTsNCi12b2lkIGluaXRfc2NlbmUodm9pZCk7DQotdm9pZCByZW5kZXIoaW50IHF1YWRzKTsNCi12b2lkIGNyZWF0ZV90ZXh0dXJlKHZvaWQpOwotaW50IHJlYWRUaW1lcih2b2lkKTsNCi0KLXN0YXRpYyB2b2lkIGdsdUxvb2tBdChmbG9hdCBleWVYLCBmbG9hdCBleWVZLCBmbG9hdCBleWVaLAotICAgICAgICBmbG9hdCBjZW50ZXJYLCBmbG9hdCBjZW50ZXJZLCBmbG9hdCBjZW50ZXJaLCBmbG9hdCB1cFgsIGZsb2F0IHVwWSwKLSAgICAgICAgZmxvYXQgdXBaKQotewotICAgIC8vIFNlZSB0aGUgT3BlbkdMIEdMVVQgZG9jdW1lbnRhdGlvbiBmb3IgZ2x1TG9va0F0IGZvciBhIGRlc2NyaXB0aW9uCi0gICAgLy8gb2YgdGhlIGFsZ29yaXRobS4gV2UgaW1wbGVtZW50IGl0IGluIGEgc3RyYWlnaHRmb3J3YXJkIHdheToKLQotICAgIGZsb2F0IGZ4ID0gY2VudGVyWCAtIGV5ZVg7Ci0gICAgZmxvYXQgZnkgPSBjZW50ZXJZIC0gZXllWTsKLSAgICBmbG9hdCBmeiA9IGNlbnRlclogLSBleWVaOwotCi0gICAgLy8gTm9ybWFsaXplIGYKLSAgICBmbG9hdCBybGYgPSAxLjBmIC8gc3FydGYoZngqZnggKyBmeSpmeSArIGZ6KmZ6KTsKLSAgICBmeCAqPSBybGY7Ci0gICAgZnkgKj0gcmxmOwotICAgIGZ6ICo9IHJsZjsKLQotICAgIC8vIE5vcm1hbGl6ZSB1cAotICAgIGZsb2F0IHJsdXAgPSAxLjBmIC8gc3FydGYodXBYKnVwWCArIHVwWSp1cFkgKyB1cFoqdXBaKTsKLSAgICB1cFggKj0gcmx1cDsKLSAgICB1cFkgKj0gcmx1cDsKLSAgICB1cFogKj0gcmx1cDsKLQotICAgIC8vIGNvbXB1dGUgcyA9IGYgeCB1cCAoeCBtZWFucyAiY3Jvc3MgcHJvZHVjdCIpCi0KLSAgICBmbG9hdCBzeCA9IGZ5ICogdXBaIC0gZnogKiB1cFk7Ci0gICAgZmxvYXQgc3kgPSBmeiAqIHVwWCAtIGZ4ICogdXBaOwotICAgIGZsb2F0IHN6ID0gZnggKiB1cFkgLSBmeSAqIHVwWDsKLQotICAgIC8vIGNvbXB1dGUgdSA9IHMgeCBmCi0gICAgZmxvYXQgdXggPSBzeSAqIGZ6IC0gc3ogKiBmeTsKLSAgICBmbG9hdCB1eSA9IHN6ICogZnggLSBzeCAqIGZ6OwotICAgIGZsb2F0IHV6ID0gc3ggKiBmeSAtIHN5ICogZng7Ci0KLSAgICBmbG9hdCBtWzE2XSA7Ci0gICAgbVswXSA9IHN4OwotICAgIG1bMV0gPSB1eDsKLSAgICBtWzJdID0gLWZ4OwotICAgIG1bM10gPSAwLjBmOwotCi0gICAgbVs0XSA9IHN5OwotICAgIG1bNV0gPSB1eTsKLSAgICBtWzZdID0gLWZ5OwotICAgIG1bN10gPSAwLjBmOwotCi0gICAgbVs4XSA9IHN6OwotICAgIG1bOV0gPSB1ejsKLSAgICBtWzEwXSA9IC1mejsKLSAgICBtWzExXSA9IDAuMGY7Ci0KLSAgICBtWzEyXSA9IDAuMGY7Ci0gICAgbVsxM10gPSAwLjBmOwotICAgIG1bMTRdID0gMC4wZjsKLSAgICBtWzE1XSA9IDEuMGY7Ci0KLSAgICBnbE11bHRNYXRyaXhmKG0pOwotICAgIGdsVHJhbnNsYXRlZigtZXllWCwgLWV5ZVksIC1leWVaKTsKLX0KLQ0KLWludCBtYWluKGludCBhcmdjLCBjaGFyICoqYXJndikNCi17DQotICAgIGludCBxOwotICAgIGludCBzdGFydCwgZW5kOw0KLQotICAgIHByaW50ZigiSW5pdGlhbGl6aW5nIEVHTC4uLlxuIik7Ci0NCi0gICAgaWYoIWluaXRfZ2xfc3VyZmFjZSgpKQ0KLSAgICB7DQotICAgICAgICBwcmludGYoIkdMIGluaXRpYWxpc2F0aW9uIGZhaWxlZCAtIGV4aXRpbmdcbiIpOw0KLSAgICAgICAgcmV0dXJuIDA7DQotICAgIH0NCi0NCi0gICAgaW5pdF9zY2VuZSgpOw0KLQ0KLSAgICBjcmVhdGVfdGV4dHVyZSgpOw0KLQ0KLSAgICBwcmludGYoIlN0YXJ0IHRlc3QuLi5cbiIpOwotCi0gICAgcmVuZGVyKGFyZ2M9PTIgPyBhdG9pKGFyZ3ZbMV0pIDogSVRFUkFUSU9OUyk7DQotDQotICAgIGZyZWVfZ2xfc3VyZmFjZSgpOw0KLQ0KLSAgICByZXR1cm4gMDsNCi19DQotDQotaW50IGluaXRfZ2xfc3VyZmFjZSh2b2lkKQ0KLXsNCi0gICAgRUdMaW50IG51bUNvbmZpZ3MgPSAxOw0KLSAgICBFR0xDb25maWcgbXlDb25maWcgPSB7MH07DQotICAgIEVHTGludCBhdHRyaWJbXSA9DQotICAgIHsNCi0gICAgICAgICAgICBFR0xfREVQVEhfU0laRSwgICAgIDE2LA0KLSAgICAgICAgICAgIEVHTF9OT05FDQotICAgIH07DQotDQotICAgIGlmICggKGVnbERpc3BsYXkgPSBlZ2xHZXREaXNwbGF5KEVHTF9ERUZBVUxUX0RJU1BMQVkpKSA9PSBFR0xfTk9fRElTUExBWSApDQotICAgIHsNCi0gICAgICAgIHByaW50ZigiZWdsR2V0RGlzcGxheSBmYWlsZWRcbiIpOw0KLSAgICAgICAgcmV0dXJuIDA7DQotICAgIH0NCi0NCi0gICAgaWYgKCBlZ2xJbml0aWFsaXplKGVnbERpc3BsYXksIE5VTEwsIE5VTEwpICE9IEVHTF9UUlVFICkNCi0gICAgew0KLSAgICAgICAgcHJpbnRmKCJlZ2xJbml0aWFsaXplIGZhaWxlZFxuIik7DQotICAgICAgICByZXR1cm4gMDsNCi0gICAgfQ0KLQ0KLSAgICBpZiAoIGVnbENob29zZUNvbmZpZyhlZ2xEaXNwbGF5LCBhdHRyaWIsICZteUNvbmZpZywgMSwgJm51bUNvbmZpZ3MpICE9IEVHTF9UUlVFICkNCi0gICAgew0KLSAgICAgICAgcHJpbnRmKCJlZ2xDaG9vc2VDb25maWcgZmFpbGVkXG4iKTsNCi0gICAgICAgIHJldHVybiAwOw0KLSAgICB9DQotDQotICAgIGlmICggKGVnbFN1cmZhY2UgPSBlZ2xDcmVhdGVXaW5kb3dTdXJmYWNlKGVnbERpc3BsYXksIG15Q29uZmlnLAotICAgICAgICAgICAgYW5kcm9pZF9jcmVhdGVEaXNwbGF5U3VyZmFjZSgpLCAwKSkgPT0gRUdMX05PX1NVUkZBQ0UgKQ0KLSAgICB7DQotICAgICAgICBwcmludGYoImVnbENyZWF0ZVdpbmRvd1N1cmZhY2UgZmFpbGVkXG4iKTsNCi0gICAgICAgIHJldHVybiAwOw0KLSAgICB9DQotDQotICAgIGlmICggKGVnbENvbnRleHQgPSBlZ2xDcmVhdGVDb250ZXh0KGVnbERpc3BsYXksIG15Q29uZmlnLCAwLCAwKSkgPT0gRUdMX05PX0NPTlRFWFQgKQ0KLSAgICB7DQotICAgICAgICBwcmludGYoImVnbENyZWF0ZUNvbnRleHQgZmFpbGVkXG4iKTsNCi0gICAgICAgIHJldHVybiAwOw0KLSAgICB9DQotDQotICAgIGlmICggZWdsTWFrZUN1cnJlbnQoZWdsRGlzcGxheSwgZWdsU3VyZmFjZSwgZWdsU3VyZmFjZSwgZWdsQ29udGV4dCkgIT0gRUdMX1RSVUUgKQ0KLSAgICB7DQotICAgICAgICBwcmludGYoImVnbE1ha2VDdXJyZW50IGZhaWxlZFxuIik7DQotICAgICAgICByZXR1cm4gMDsNCi0gICAgfQ0KLQ0KLSAgICByZXR1cm4gMTsNCi19DQotDQotdm9pZCBmcmVlX2dsX3N1cmZhY2Uodm9pZCkNCi17DQotICAgIGlmIChlZ2xEaXNwbGF5ICE9IEVHTF9OT19ESVNQTEFZKQ0KLSAgICB7DQotICAgICAgICBlZ2xNYWtlQ3VycmVudCggRUdMX05PX0RJU1BMQVksIEVHTF9OT19TVVJGQUNFLA0KLSAgICAgICAgICAgICAgICBFR0xfTk9fU1VSRkFDRSwgRUdMX05PX0NPTlRFWFQgKTsNCi0gICAgICAgIGVnbERlc3Ryb3lDb250ZXh0KCBlZ2xEaXNwbGF5LCBlZ2xDb250ZXh0ICk7DQotICAgICAgICBlZ2xEZXN0cm95U3VyZmFjZSggZWdsRGlzcGxheSwgZWdsU3VyZmFjZSApOw0KLSAgICAgICAgZWdsVGVybWluYXRlKCBlZ2xEaXNwbGF5ICk7DQotICAgICAgICBlZ2xEaXNwbGF5ID0gRUdMX05PX0RJU1BMQVk7DQotICAgIH0NCi19DQotDQotdm9pZCBpbml0X3NjZW5lKHZvaWQpDQotew0KLSAgICBnbERpc2FibGUoR0xfRElUSEVSKTsKLSAgICBnbEVuYWJsZShHTF9DVUxMX0ZBQ0UpOwotCi0gICAgZmxvYXQgcmF0aW8gPSAzMjAuMGYgLyA0ODAuMGY7Ci0gICAgZ2xWaWV3cG9ydCgwLCAwLCAzMjAsIDQ4MCk7Ci0KLSAgICBnbE1hdHJpeE1vZGUoR0xfUFJPSkVDVElPTik7Ci0gICAgZ2xMb2FkSWRlbnRpdHkoKTsKLSAgICBnbEZydXN0dW1mKC1yYXRpbywgcmF0aW8sIC0xLCAxLCAxLCAxMCk7Ci0KLSAgICBnbE1hdHJpeE1vZGUoR0xfTU9ERUxWSUVXKTsNCi0gICAgZ2xMb2FkSWRlbnRpdHkoKTsKLSAgICBnbHVMb29rQXQoCi0gICAgICAgICAgICAwLCAwLCAzLCAgLy8gZXllCi0gICAgICAgICAgICAwLCAwLCAwLCAgLy8gY2VudGVyCi0gICAgICAgICAgICAwLCAxLCAwKTsgLy8gdXAKLQ0KLSAgICBnbEVuYWJsZShHTF9URVhUVVJFXzJEKTsNCi0gICAgZ2xFbmFibGVDbGllbnRTdGF0ZShHTF9WRVJURVhfQVJSQVkpOw0KLSAgICBnbEVuYWJsZUNsaWVudFN0YXRlKEdMX1RFWFRVUkVfQ09PUkRfQVJSQVkpOwotfQ0KLQ0KLXZvaWQgY3JlYXRlX3RleHR1cmUodm9pZCkNCi17DQotICAgIGNvbnN0IHVuc2lnbmVkIGludCBvbiA9IDB4ZmYwMDAwZmY7Ci0gICAgY29uc3QgdW5zaWduZWQgaW50IG9mZiA9IDB4ZmZmZmZmZmY7Ci0gICAgY29uc3QgdW5zaWduZWQgaW50IHBpeGVsc1tdID0KLSAgICB7Ci0gICAgICAgICAgICBvbiwgb2ZmLCBvbiwgb2ZmLCBvbiwgb2ZmLCBvbiwgb2ZmLAotICAgICAgICAgICAgb2ZmLCBvbiwgb2ZmLCBvbiwgb2ZmLCBvbiwgb2ZmLCBvbiwKLSAgICAgICAgICAgIG9uLCBvZmYsIG9uLCBvZmYsIG9uLCBvZmYsIG9uLCBvZmYsCi0gICAgICAgICAgICBvZmYsIG9uLCBvZmYsIG9uLCBvZmYsIG9uLCBvZmYsIG9uLAotICAgICAgICAgICAgb24sIG9mZiwgb24sIG9mZiwgb24sIG9mZiwgb24sIG9mZiwKLSAgICAgICAgICAgIG9mZiwgb24sIG9mZiwgb24sIG9mZiwgb24sIG9mZiwgb24sCi0gICAgICAgICAgICBvbiwgb2ZmLCBvbiwgb2ZmLCBvbiwgb2ZmLCBvbiwgb2ZmLAotICAgICAgICAgICAgb2ZmLCBvbiwgb2ZmLCBvbiwgb2ZmLCBvbiwgb2ZmLCBvbiwKLSAgICB9Ow0KLSAgICBnbEdlblRleHR1cmVzKDEsICZ0ZXh0dXJlKTsNCi0gICAgZ2xCaW5kVGV4dHVyZShHTF9URVhUVVJFXzJELCB0ZXh0dXJlKTsNCi0gICAgZ2xUZXhJbWFnZTJEKEdMX1RFWFRVUkVfMkQsIDAsIEdMX1JHQkEsIDgsIDgsIDAsIEdMX1JHQkEsIEdMX1VOU0lHTkVEX0JZVEUsIHBpeGVscyk7DQotICAgIGdsVGV4UGFyYW1ldGVyeChHTF9URVhUVVJFXzJELCBHTF9URVhUVVJFX01JTl9GSUxURVIsIEdMX05FQVJFU1QpOw0KLSAgICBnbFRleFBhcmFtZXRlcngoR0xfVEVYVFVSRV8yRCwgR0xfVEVYVFVSRV9NQUdfRklMVEVSLCBHTF9ORUFSRVNUKTsNCi0gICAgZ2xUZXhFbnZ4KEdMX1RFWFRVUkVfRU5WLCBHTF9URVhUVVJFX0VOVl9NT0RFLCBHTF9SRVBMQUNFKTsNCi19DQotDQotdm9pZCByZW5kZXIoaW50IHF1YWRzKQ0KLXsNCi0gICAgaW50IGksIGo7DQotDQotICAgIGNvbnN0IEdMZmxvYXQgdmVydGljZXNbXSA9IHsNCi0gICAgICAgICAgICAtMSwgIC0xLCAgMCwNCi0gICAgICAgICAgICAgMSwgIC0xLCAgMCwNCi0gICAgICAgICAgICAgMSwgICAxLCAgMCwNCi0gICAgICAgICAgICAtMSwgICAxLCAgMA0KLSAgICB9Ow0KLQ0KLSAgICBjb25zdCBHTGZpeGVkIHRleENvb3Jkc1tdID0gew0KLSAgICAgICAgICAgIDAsICAgICAgICAgICAgMCwNCi0gICAgICAgICAgICBGSVhFRF9PTkUsICAgIDAsDQotICAgICAgICAgICAgRklYRURfT05FLCAgICBGSVhFRF9PTkUsDQotICAgICAgICAgICAgMCwgICAgICAgICAgICBGSVhFRF9PTkUNCi0gICAgfTsNCi0NCi0gICAgY29uc3QgR0x1c2hvcnQgdGVtcGxhdGVbXSA9IHsgMCwgMSwgMiwgIDAsIDIsIDMgfTsKLQotCi0gICAgR0x1c2hvcnQqIGluZGljZXMgPSAoR0x1c2hvcnQqKW1hbGxvYyhxdWFkcypzaXplb2YodGVtcGxhdGUpKTsKLSAgICBmb3IgKGk9MCA7IGk8cXVhZHMgOyBpKyspCi0gICAgICAgIG1lbWNweShpbmRpY2VzKyhzaXplb2YodGVtcGxhdGUpL3NpemVvZihpbmRpY2VzWzBdKSkqaSwgdGVtcGxhdGUsIHNpemVvZih0ZW1wbGF0ZSkpOwotDQotICAgIGdsVmVydGV4UG9pbnRlcigzLCBHTF9GTE9BVCwgMCwgdmVydGljZXMpOw0KLSAgICBnbFRleENvb3JkUG9pbnRlcigyLCBHTF9GSVhFRCwgMCwgdGV4Q29vcmRzKTsKLQotICAgIC8vIG1ha2Ugc3VyZSB0byBkbyBhIGNvdXBsZSBlZ2xTd2FwQnVmZmVycyB0byBtYWtlIHN1cmUgdGhlcmUgYXJlCi0gICAgLy8gbm8gcHJvYmxlbXMgd2l0aCB0aGUgdmVyeSBmaXJzdCBvbmVzICh3aG8ga25vd3MpCi0gICAgZ2xDbGVhckNvbG9yKDAuNCwgMC40LCAwLjQsIDAuNCk7Ci0gICAgZ2xDbGVhcihHTF9ERVBUSF9CVUZGRVJfQklUIHwgR0xfQ09MT1JfQlVGRkVSX0JJVCk7Ci0gICAgZWdsU3dhcEJ1ZmZlcnMoZWdsRGlzcGxheSwgZWdsU3VyZmFjZSk7Ci0gICAgZ2xDbGVhckNvbG9yKDAuNiwgMC42LCAwLjYsIDAuNik7Ci0gICAgZ2xDbGVhcihHTF9ERVBUSF9CVUZGRVJfQklUIHwgR0xfQ09MT1JfQlVGRkVSX0JJVCk7Ci0gICAgZWdsU3dhcEJ1ZmZlcnMoZWdsRGlzcGxheSwgZWdsU3VyZmFjZSk7Ci0gICAgZ2xDbGVhckNvbG9yKDEuMCwgMS4wLCAxLjAsIDEuMCk7Ci0KLSAgICBmb3IgKGo9MCA7IGo8MTAgOyBqKyspIHsKLSAgICAgICAgcHJpbnRmKCJsb29wICVkIC8gMTAgKCVkIHF1YWRzIC8gbG9vcClcbiIsIGosIHF1YWRzKTsKLQotICAgICAgICBpbnQgbmVsZW0gPSBzaXplb2YodGVtcGxhdGUpL3NpemVvZih0ZW1wbGF0ZVswXSk7Ci0gICAgICAgIGdsQ2xlYXIoR0xfREVQVEhfQlVGRkVSX0JJVCB8IEdMX0NPTE9SX0JVRkZFUl9CSVQpOwotICAgICAgICBnbERyYXdFbGVtZW50cyhHTF9UUklBTkdMRVMsIG5lbGVtKnF1YWRzLCBHTF9VTlNJR05FRF9TSE9SVCwgaW5kaWNlcyk7Ci0gICAgICAgIGVnbFN3YXBCdWZmZXJzKGVnbERpc3BsYXksIGVnbFN1cmZhY2UpOwotICAgIH0KLQotICAgIGZyZWUoaW5kaWNlcyk7DQotfQ0KLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL2dlbiBiL29wZW5nbC90b29scy9nbGdlbi9nZW4KZGVsZXRlZCBmaWxlIG1vZGUgMTAwNzU1CmluZGV4IDFjNDk4NjEuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL2dlbgorKysgL2Rldi9udWxsCkBAIC0xLDk5ICswLDAgQEAKLSMhL2Jpbi9zaAotcm0gLXJmIG91dCBnZW5lcmF0ZWQKLQotbWtkaXIgb3V0Ci1ta2RpciAtcCBvdXQvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMKLW1rZGlyIC1wIG91dC9jb20vZ29vZ2xlL2FuZHJvaWQvZ2xlc19qbmkKLW1rZGlyIC1wIG91dC9hbmRyb2lkL2dyYXBoaWNzCi0KLWVjaG8gInBhY2thZ2UgYW5kcm9pZC5ncmFwaGljczsiID4gb3V0L2FuZHJvaWQvZ3JhcGhpY3MvQ2FudmFzLmphdmEKLWVjaG8gInB1YmxpYyBpbnRlcmZhY2UgQ2FudmFzIHt9IiA+PiBvdXQvYW5kcm9pZC9ncmFwaGljcy9DYW52YXMuamF2YQotCi1HTEZJTEU9b3V0L2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMLmphdmEKLWNwIHN0dWJzL0dMSGVhZGVyLmphdmEtaWYgJEdMRklMRQotCi1HTEdFTl9GSUxFUz0iQ0Z1bmMuamF2YSBDVHlwZS5qYXZhIENvZGVFbWl0dGVyLmphdmEgR2VuZXJhdGVHTC5qYXZhIEpGdW5jLmphdmEgSlR5cGUuamF2YSBKbmlDb2RlRW1pdHRlci5qYXZhIFBhcmFtZXRlckNoZWNrZXIuamF2YSIKLQotcHVzaGQgc3JjID4gL2Rldi9udWxsCi1qYXZhYyAke0dMR0VOX0ZJTEVTfQotcG9wZCA+IC9kZXYvbnVsbAotamF2YSAtY2xhc3NwYXRoIHNyYyBHZW5lcmF0ZUdMIC1jIGdsc3BlYy0xLjAgZ2xzcGVjLTEuMGV4dCBnbHNwZWMtMS4xIGdsc3BlYy0xLjFleHQgZ2xzcGVjLTEuMWV4dHBhY2sgZ2xzcGVjLWNoZWNrcwotCi1wdXNoZCBvdXQgPiAvZGV2L251bGwKLW1rZGlyIGNsYXNzZXMKLWphdmFjIC1kIGNsYXNzZXMgY29tL2dvb2dsZS9hbmRyb2lkL2dsZXNfam5pL0dMSW1wbC5qYXZhIGphdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTAuamF2YSBqYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDEwRXh0LmphdmEgamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMS5qYXZhIGphdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTFFeHQuamF2YSBqYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDExRXh0ZW5zaW9uUGFjay5qYXZhCi1wb3BkID4gL2Rldi9udWxsCi0KLXJtIC1yZiBnZW5lcmF0ZWQKLW1rZGlyIC1wIGdlbmVyYXRlZC9DCi1jcCBvdXQvY29tX2dvb2dsZV9hbmRyb2lkX2dsZXNfam5pX0dMSW1wbC5jcHAgZ2VuZXJhdGVkL0MKLWNwIC1yIG91dC9jb20gZ2VuZXJhdGVkCi1jcCAtciBvdXQvamF2YXggZ2VuZXJhdGVkCi0KLXJtIC1yZiBvdXQKLQotIyBjb21fZ29vZ2xlX2FuZHJvaWRfZ2xlc19qbmlfR0xJbXBsLmNwcAotaWYgY21wIC4uLy4uLy4uL2ZyYW1ld29ya3MvYmFzZS9jb3JlL2puaS9jb21fZ29vZ2xlX2FuZHJvaWRfZ2xlc19qbmlfR0xJbXBsLmNwcCBnZW5lcmF0ZWQvQy9jb21fZ29vZ2xlX2FuZHJvaWRfZ2xlc19qbmlfR0xJbXBsLmNwcCA7IHRoZW4KLWVjaG8gY29tX2dvb2dsZV9hbmRyb2lkX2dsZXNfam5pX0dMSW1wbC5jcHAgdW5jaGFuZ2VkCi1lbHNlCi1lY2hvIFBsZWFzZSBlZGl0IC4uLy4uLy4uL2ZyYW1ld29ya3MvYmFzZS9jb3JlL2puaS9jb21fZ29vZ2xlX2FuZHJvaWRfZ2xlc19qbmlfR0xJbXBsLmNwcAotZWNobyBQbGVhc2UgY3AgZ2VuZXJhdGVkL0MvY29tX2dvb2dsZV9hbmRyb2lkX2dsZXNfam5pX0dMSW1wbC5jcHAgLi4vLi4vLi4vZnJhbWV3b3Jrcy9iYXNlL2NvcmUvam5pCi1maQotCi0jIEdMSW1wbC5qYXZhCi1pZiBjbXAgLi4vLi4vamF2YS9jb20vZ29vZ2xlL2FuZHJvaWQvZ2xlc19qbmkvR0xJbXBsLmphdmEgZ2VuZXJhdGVkL2NvbS9nb29nbGUvYW5kcm9pZC9nbGVzX2puaS9HTEltcGwuamF2YSA7IHRoZW4KLWVjaG8gR0xJbXBsLmphdmEgdW5jaGFuZ2VkCi1lbHNlCi1lY2hvIFBsZWFzZSBlZGl0IC4uLy4uL2phdmEvY29tL2dvb2dsZS9hbmRyb2lkL2dsZXNfam5pL0dMSW1wbC5qYXZhCi1lY2hvIFBsZWFzZSBjcCBnZW5lcmF0ZWQvY29tL2dvb2dsZS9hbmRyb2lkL2dsZXNfam5pL0dMSW1wbC5qYXZhIC4uLy4uL2phdmEvY29tL2dvb2dsZS9hbmRyb2lkL2dsZXNfam5pCi1maQotCi0jIEdMLmphdmEKLWlmIGNtcCAuLi8uLi9qYXZhL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMLmphdmEgZ2VuZXJhdGVkL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMLmphdmEgOyB0aGVuCi1lY2hvIEdMLmphdmEgdW5jaGFuZ2VkCi1lbHNlCi1lY2hvIFBsZWFzZSBlZGl0IC4uLy4uL2phdmEvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wuamF2YQotZWNobyBQbGVhc2UgY3AgZ2VuZXJhdGVkL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMLmphdmEgLi4vLi4vamF2YS9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTC5qYXZhCi1maQotCi0jIEdMMTAuamF2YQotaWYgY21wIC4uLy4uL2phdmEvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMC5qYXZhIGdlbmVyYXRlZC9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDEwLmphdmEgOyB0aGVuCi1lY2hvIEdMMTAuamF2YSB1bmNoYW5nZWQKLWVsc2UKLWVjaG8gUGxlYXNlIGVkaXQgLi4vLi4vamF2YS9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDEwLmphdmEKLWVjaG8gUGxlYXNlIGNwIGdlbmVyYXRlZC9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDEwLmphdmEgLi4vLi4vamF2YS9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDEwLmphdmEKLWZpCi0KLSMgR0wxMEV4dC5qYXZhCi1pZiBjbXAgLi4vLi4vamF2YS9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDEwRXh0LmphdmEgZ2VuZXJhdGVkL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTBFeHQuamF2YSA7IHRoZW4KLWVjaG8gR0wxMEV4dC5qYXZhIHVuY2hhbmdlZAotZWxzZQotZWNobyBQbGVhc2UgZWRpdCAuLi8uLi9qYXZhL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTBFeHQuamF2YQotZWNobyBQbGVhc2UgY3AgZ2VuZXJhdGVkL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTBFeHQuamF2YSAuLi8uLi9qYXZhL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTBFeHQuamF2YQotZmkKLQotIyBHTDExLmphdmEKLWlmIGNtcCAuLi8uLi9qYXZhL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTEuamF2YSBnZW5lcmF0ZWQvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMS5qYXZhIDsgdGhlbgotZWNobyBHTDExLmphdmEgdW5jaGFuZ2VkCi1lbHNlCi1lY2hvIFBsZWFzZSBlZGl0IC4uLy4uL2phdmEvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMS5qYXZhCi1lY2hvIFBsZWFzZSBjcCBnZW5lcmF0ZWQvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMS5qYXZhIC4uLy4uL2phdmEvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMS5qYXZhCi1maQotCi0jIEdMMTFFeHQuamF2YQotaWYgY21wIC4uLy4uL2phdmEvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMUV4dC5qYXZhIGdlbmVyYXRlZC9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDExRXh0LmphdmEgOyB0aGVuCi1lY2hvIEdMMTFFeHQuamF2YSB1bmNoYW5nZWQKLWVsc2UKLWVjaG8gUGxlYXNlIGVkaXQgLi4vLi4vamF2YS9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDExRXh0LmphdmEKLWVjaG8gUGxlYXNlIGNwIGdlbmVyYXRlZC9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDExRXh0LmphdmEgLi4vLi4vamF2YS9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDExRXh0LmphdmEKLWZpCi0KLSMgR0wxMUV4dGVuc2lvblBhY2suamF2YQotaWYgY21wIC4uLy4uL2phdmEvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMUV4dGVuc2lvblBhY2suamF2YSBnZW5lcmF0ZWQvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMUV4dGVuc2lvblBhY2suamF2YSA7IHRoZW4KLWVjaG8gR0wxMUV4dGVuc2lvblBhY2suamF2YSB1bmNoYW5nZWQKLWVsc2UKLWVjaG8gUGxlYXNlIGVkaXQgLi4vLi4vamF2YS9qYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDExRXh0ZW5zaW9uUGFjay5qYXZhCi1lY2hvIFBsZWFzZSBjcCBnZW5lcmF0ZWQvamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMUV4dGVuc2lvblBhY2suamF2YSAuLi8uLi9qYXZhL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTFFeHRlbnNpb25QYWNrLmphdmEKLWZpCi0KLXJtIC1yZiBnZW5lcmF0ZWQKZGlmZiAtLWdpdCBhL29wZW5nbC90b29scy9nbGdlbi9nbHNwZWMtMS4wIGIvb3BlbmdsL3Rvb2xzL2dsZ2VuL2dsc3BlYy0xLjAKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGM0NDIzMjAuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL2dsc3BlYy0xLjAKKysrIC9kZXYvbnVsbApAQCAtMSwxMDYgKzAsMCBAQAotdm9pZCBnbEFjdGl2ZVRleHR1cmUgKCBHTGVudW0gdGV4dHVyZSApDQotdm9pZCBnbEFscGhhRnVuYyAoIEdMZW51bSBmdW5jLCBHTGNsYW1wZiByZWYgKQ0KLXZvaWQgZ2xBbHBoYUZ1bmN4ICggR0xlbnVtIGZ1bmMsIEdMY2xhbXB4IHJlZiApDQotdm9pZCBnbEJpbmRUZXh0dXJlICggR0xlbnVtIHRhcmdldCwgR0x1aW50IHRleHR1cmUgKQ0KLXZvaWQgZ2xCbGVuZEZ1bmMgKCBHTGVudW0gc2ZhY3RvciwgR0xlbnVtIGRmYWN0b3IgKQ0KLXZvaWQgZ2xDbGVhciAoIEdMYml0ZmllbGQgbWFzayApDQotdm9pZCBnbENsZWFyQ29sb3IgKCBHTGNsYW1wZiByZWQsIEdMY2xhbXBmIGdyZWVuLCBHTGNsYW1wZiBibHVlLCBHTGNsYW1wZiBhbHBoYSApDQotdm9pZCBnbENsZWFyQ29sb3J4ICggR0xjbGFtcHggcmVkLCBHTGNsYW1weCBncmVlbiwgR0xjbGFtcHggYmx1ZSwgR0xjbGFtcHggYWxwaGEgKQ0KLXZvaWQgZ2xDbGVhckRlcHRoZiAoIEdMY2xhbXBmIGRlcHRoICkNCi12b2lkIGdsQ2xlYXJEZXB0aHggKCBHTGNsYW1weCBkZXB0aCApDQotdm9pZCBnbENsZWFyU3RlbmNpbCAoIEdMaW50IHMgKQ0KLXZvaWQgZ2xDbGllbnRBY3RpdmVUZXh0dXJlICggR0xlbnVtIHRleHR1cmUgKQ0KLXZvaWQgZ2xDb2xvcjRmICggR0xmbG9hdCByZWQsIEdMZmxvYXQgZ3JlZW4sIEdMZmxvYXQgYmx1ZSwgR0xmbG9hdCBhbHBoYSApDQotdm9pZCBnbENvbG9yNHggKCBHTGZpeGVkIHJlZCwgR0xmaXhlZCBncmVlbiwgR0xmaXhlZCBibHVlLCBHTGZpeGVkIGFscGhhICkNCi12b2lkIGdsQ29sb3JNYXNrICggR0xib29sZWFuIHJlZCwgR0xib29sZWFuIGdyZWVuLCBHTGJvb2xlYW4gYmx1ZSwgR0xib29sZWFuIGFscGhhICkNCi12b2lkIGdsQ29sb3JQb2ludGVyICggR0xpbnQgc2l6ZSwgR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIgKQ0KLXZvaWQgZ2xDb21wcmVzc2VkVGV4SW1hZ2UyRCAoIEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGVudW0gaW50ZXJuYWxmb3JtYXQsIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LCBHTGludCBib3JkZXIsIEdMc2l6ZWkgaW1hZ2VTaXplLCBjb25zdCBHTHZvaWQgKmRhdGEgKQ0KLXZvaWQgZ2xDb21wcmVzc2VkVGV4U3ViSW1hZ2UyRCAoIEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGludCB4b2Zmc2V0LCBHTGludCB5b2Zmc2V0LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwgR0xlbnVtIGZvcm1hdCwgR0xzaXplaSBpbWFnZVNpemUsIGNvbnN0IEdMdm9pZCAqZGF0YSApDQotdm9pZCBnbENvcHlUZXhJbWFnZTJEICggR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMZW51bSBpbnRlcm5hbGZvcm1hdCwgR0xpbnQgeCwgR0xpbnQgeSwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsIEdMaW50IGJvcmRlciApDQotdm9pZCBnbENvcHlUZXhTdWJJbWFnZTJEICggR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMaW50IHhvZmZzZXQsIEdMaW50IHlvZmZzZXQsIEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0ICkNCi12b2lkIGdsQ3VsbEZhY2UgKCBHTGVudW0gbW9kZSApDQotdm9pZCBnbERlbGV0ZVRleHR1cmVzICggR0xzaXplaSBuLCBjb25zdCBHTHVpbnQgKnRleHR1cmVzICkNCi12b2lkIGdsRGVwdGhGdW5jICggR0xlbnVtIGZ1bmMgKQ0KLXZvaWQgZ2xEZXB0aE1hc2sgKCBHTGJvb2xlYW4gZmxhZyApDQotdm9pZCBnbERlcHRoUmFuZ2VmICggR0xjbGFtcGYgek5lYXIsIEdMY2xhbXBmIHpGYXIgKQ0KLXZvaWQgZ2xEZXB0aFJhbmdleCAoIEdMY2xhbXB4IHpOZWFyLCBHTGNsYW1weCB6RmFyICkNCi12b2lkIGdsRGlzYWJsZSAoIEdMZW51bSBjYXAgKQ0KLXZvaWQgZ2xEaXNhYmxlQ2xpZW50U3RhdGUgKCBHTGVudW0gYXJyYXkgKQ0KLXZvaWQgZ2xEcmF3QXJyYXlzICggR0xlbnVtIG1vZGUsIEdMaW50IGZpcnN0LCBHTHNpemVpIGNvdW50ICkNCi12b2lkIGdsRHJhd0VsZW1lbnRzICggR0xlbnVtIG1vZGUsIEdMc2l6ZWkgY291bnQsIEdMZW51bSB0eXBlLCBjb25zdCBHTHZvaWQgKmluZGljZXMgKQ0KLXZvaWQgZ2xFbmFibGUgKCBHTGVudW0gY2FwICkNCi12b2lkIGdsRW5hYmxlQ2xpZW50U3RhdGUgKCBHTGVudW0gYXJyYXkgKQ0KLXZvaWQgZ2xGaW5pc2ggKCB2b2lkICkNCi12b2lkIGdsRmx1c2ggKCB2b2lkICkNCi12b2lkIGdsRm9nZiAoIEdMZW51bSBwbmFtZSwgR0xmbG9hdCBwYXJhbSApDQotdm9pZCBnbEZvZ2Z2ICggR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMgKQ0KLXZvaWQgZ2xGb2d4ICggR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtICkNCi12b2lkIGdsRm9neHYgKCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZml4ZWQgKnBhcmFtcyApDQotdm9pZCBnbEZyb250RmFjZSAoIEdMZW51bSBtb2RlICkNCi12b2lkIGdsRnJ1c3R1bWYgKCBHTGZsb2F0IGxlZnQsIEdMZmxvYXQgcmlnaHQsIEdMZmxvYXQgYm90dG9tLCBHTGZsb2F0IHRvcCwgR0xmbG9hdCB6TmVhciwgR0xmbG9hdCB6RmFyICkNCi12b2lkIGdsRnJ1c3R1bXggKCBHTGZpeGVkIGxlZnQsIEdMZml4ZWQgcmlnaHQsIEdMZml4ZWQgYm90dG9tLCBHTGZpeGVkIHRvcCwgR0xmaXhlZCB6TmVhciwgR0xmaXhlZCB6RmFyICkNCi12b2lkIGdsR2VuVGV4dHVyZXMgKCBHTHNpemVpIG4sIEdMdWludCAqdGV4dHVyZXMgKQ0KLUdMZW51bSBnbEdldEVycm9yICggdm9pZCApDQotdm9pZCBnbEdldEludGVnZXJ2ICggR0xlbnVtIHBuYW1lLCBHTGludCAqcGFyYW1zICkNCi1jb25zdCBHTHVieXRlICogZ2xHZXRTdHJpbmcgKCBHTGVudW0gbmFtZSApDQotdm9pZCBnbEhpbnQgKCBHTGVudW0gdGFyZ2V0LCBHTGVudW0gbW9kZSApDQotdm9pZCBnbExpZ2h0TW9kZWxmICggR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtICkNCi12b2lkIGdsTGlnaHRNb2RlbGZ2ICggR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZsb2F0ICpwYXJhbXMgKQ0KLXZvaWQgZ2xMaWdodE1vZGVseCAoIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSApDQotdm9pZCBnbExpZ2h0TW9kZWx4diAoIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zICkNCi12b2lkIGdsTGlnaHRmICggR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0gKQ0KLXZvaWQgZ2xMaWdodGZ2ICggR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcyApDQotdm9pZCBnbExpZ2h0eCAoIEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtICkNCi12b2lkIGdsTGlnaHR4diAoIEdMZW51bSBsaWdodCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMgKQ0KLXZvaWQgZ2xMaW5lV2lkdGggKCBHTGZsb2F0IHdpZHRoICkNCi12b2lkIGdsTGluZVdpZHRoeCAoIEdMZml4ZWQgd2lkdGggKQ0KLXZvaWQgZ2xMb2FkSWRlbnRpdHkgKCB2b2lkICkNCi12b2lkIGdsTG9hZE1hdHJpeGYgKCBjb25zdCBHTGZsb2F0ICptICkNCi12b2lkIGdsTG9hZE1hdHJpeHggKCBjb25zdCBHTGZpeGVkICptICkNCi12b2lkIGdsTG9naWNPcCAoIEdMZW51bSBvcGNvZGUgKQ0KLXZvaWQgZ2xNYXRlcmlhbGYgKCBHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtICkNCi12b2lkIGdsTWF0ZXJpYWxmdiAoIEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcyApDQotdm9pZCBnbE1hdGVyaWFseCAoIEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgcGFyYW0gKQ0KLXZvaWQgZ2xNYXRlcmlhbHh2ICggR0xlbnVtIGZhY2UsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zICkNCi12b2lkIGdsTWF0cml4TW9kZSAoIEdMZW51bSBtb2RlICkNCi12b2lkIGdsTXVsdE1hdHJpeGYgKCBjb25zdCBHTGZsb2F0ICptICkNCi12b2lkIGdsTXVsdE1hdHJpeHggKCBjb25zdCBHTGZpeGVkICptICkNCi12b2lkIGdsTXVsdGlUZXhDb29yZDRmICggR0xlbnVtIHRhcmdldCwgR0xmbG9hdCBzLCBHTGZsb2F0IHQsIEdMZmxvYXQgciwgR0xmbG9hdCBxICkNCi12b2lkIGdsTXVsdGlUZXhDb29yZDR4ICggR0xlbnVtIHRhcmdldCwgR0xmaXhlZCBzLCBHTGZpeGVkIHQsIEdMZml4ZWQgciwgR0xmaXhlZCBxICkNCi12b2lkIGdsTm9ybWFsM2YgKCBHTGZsb2F0IG54LCBHTGZsb2F0IG55LCBHTGZsb2F0IG56ICkNCi12b2lkIGdsTm9ybWFsM3ggKCBHTGZpeGVkIG54LCBHTGZpeGVkIG55LCBHTGZpeGVkIG56ICkNCi12b2lkIGdsTm9ybWFsUG9pbnRlciAoIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyICkNCi12b2lkIGdsT3J0aG9mICggR0xmbG9hdCBsZWZ0LCBHTGZsb2F0IHJpZ2h0LCBHTGZsb2F0IGJvdHRvbSwgR0xmbG9hdCB0b3AsIEdMZmxvYXQgek5lYXIsIEdMZmxvYXQgekZhciApDQotdm9pZCBnbE9ydGhveCAoIEdMZml4ZWQgbGVmdCwgR0xmaXhlZCByaWdodCwgR0xmaXhlZCBib3R0b20sIEdMZml4ZWQgdG9wLCBHTGZpeGVkIHpOZWFyLCBHTGZpeGVkIHpGYXIgKQ0KLXZvaWQgZ2xQaXhlbFN0b3JlaSAoIEdMZW51bSBwbmFtZSwgR0xpbnQgcGFyYW0gKQ0KLXZvaWQgZ2xQb2ludFNpemUgKCBHTGZsb2F0IHNpemUgKQ0KLXZvaWQgZ2xQb2ludFNpemV4ICggR0xmaXhlZCBzaXplICkNCi12b2lkIGdsUG9seWdvbk9mZnNldCAoIEdMZmxvYXQgZmFjdG9yLCBHTGZsb2F0IHVuaXRzICkNCi12b2lkIGdsUG9seWdvbk9mZnNldHggKCBHTGZpeGVkIGZhY3RvciwgR0xmaXhlZCB1bml0cyApDQotdm9pZCBnbFBvcE1hdHJpeCAoIHZvaWQgKQ0KLXZvaWQgZ2xQdXNoTWF0cml4ICggdm9pZCApDQotdm9pZCBnbFJlYWRQaXhlbHMgKCBHTGludCB4LCBHTGludCB5LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwgR0xlbnVtIGZvcm1hdCwgR0xlbnVtIHR5cGUsIEdMdm9pZCAqcGl4ZWxzICkNCi12b2lkIGdsUm90YXRlZiAoIEdMZmxvYXQgYW5nbGUsIEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHogKQ0KLXZvaWQgZ2xSb3RhdGV4ICggR0xmaXhlZCBhbmdsZSwgR0xmaXhlZCB4LCBHTGZpeGVkIHksIEdMZml4ZWQgeiApDQotdm9pZCBnbFNhbXBsZUNvdmVyYWdlICggR0xjbGFtcGYgdmFsdWUsIEdMYm9vbGVhbiBpbnZlcnQgKQ0KLXZvaWQgZ2xTYW1wbGVDb3ZlcmFnZXggKCBHTGNsYW1weCB2YWx1ZSwgR0xib29sZWFuIGludmVydCApDQotdm9pZCBnbFNjYWxlZiAoIEdMZmxvYXQgeCwgR0xmbG9hdCB5LCBHTGZsb2F0IHogKQ0KLXZvaWQgZ2xTY2FsZXggKCBHTGZpeGVkIHgsIEdMZml4ZWQgeSwgR0xmaXhlZCB6ICkNCi12b2lkIGdsU2Npc3NvciAoIEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0ICkNCi12b2lkIGdsU2hhZGVNb2RlbCAoIEdMZW51bSBtb2RlICkNCi12b2lkIGdsU3RlbmNpbEZ1bmMgKCBHTGVudW0gZnVuYywgR0xpbnQgcmVmLCBHTHVpbnQgbWFzayApDQotdm9pZCBnbFN0ZW5jaWxNYXNrICggR0x1aW50IG1hc2sgKQ0KLXZvaWQgZ2xTdGVuY2lsT3AgKCBHTGVudW0gZmFpbCwgR0xlbnVtIHpmYWlsLCBHTGVudW0genBhc3MgKQ0KLXZvaWQgZ2xUZXhDb29yZFBvaW50ZXIgKCBHTGludCBzaXplLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlciApDQotdm9pZCBnbFRleEVudmYgKCBHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgcGFyYW0gKQ0KLXZvaWQgZ2xUZXhFbnZmdiAoIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zICkNCi12b2lkIGdsVGV4RW52eCAoIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSApDQotdm9pZCBnbFRleEVudnh2ICggR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGZpeGVkICpwYXJhbXMgKQ0KLXZvaWQgZ2xUZXhJbWFnZTJEICggR0xlbnVtIHRhcmdldCwgR0xpbnQgbGV2ZWwsIEdMaW50IGludGVybmFsZm9ybWF0LCBHTHNpemVpIHdpZHRoLCBHTHNpemVpIGhlaWdodCwgR0xpbnQgYm9yZGVyLCBHTGVudW0gZm9ybWF0LCBHTGVudW0gdHlwZSwgY29uc3QgR0x2b2lkICpwaXhlbHMgKQ0KLXZvaWQgZ2xUZXhQYXJhbWV0ZXJmICggR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtICkNCi12b2lkIGdsVGV4UGFyYW1ldGVyeCAoIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCBwYXJhbSApDQotdm9pZCBnbFRleFN1YkltYWdlMkQgKCBHTGVudW0gdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgeG9mZnNldCwgR0xpbnQgeW9mZnNldCwgR0xzaXplaSB3aWR0aCwgR0xzaXplaSBoZWlnaHQsIEdMZW51bSBmb3JtYXQsIEdMZW51bSB0eXBlLCBjb25zdCBHTHZvaWQgKnBpeGVscyApDQotdm9pZCBnbFRyYW5zbGF0ZWYgKCBHTGZsb2F0IHgsIEdMZmxvYXQgeSwgR0xmbG9hdCB6ICkNCi12b2lkIGdsVHJhbnNsYXRleCAoIEdMZml4ZWQgeCwgR0xmaXhlZCB5LCBHTGZpeGVkIHogKQ0KLXZvaWQgZ2xWZXJ0ZXhQb2ludGVyICggR0xpbnQgc2l6ZSwgR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIgKQ0KLXZvaWQgZ2xWaWV3cG9ydCAoIEdMaW50IHgsIEdMaW50IHksIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0ICkNCmRpZmYgLS1naXQgYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vZ2xzcGVjLTEuMGV4dCBiL29wZW5nbC90b29scy9nbGdlbi9nbHNwZWMtMS4wZXh0CmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA3ZDE5NzU4Li4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9nbHNwZWMtMS4wZXh0CisrKyAvZGV2L251bGwKQEAgLTEgKzAsMCBAQAotR0xiaXRmaWVsZCBnbFF1ZXJ5TWF0cml4eE9FUyAoIEdMZml4ZWQgKm1hbnRpc3NhLCBHTGludCAqZXhwb25lbnQgKQ0KZGlmZiAtLWdpdCBhL29wZW5nbC90b29scy9nbGdlbi9nbHNwZWMtMS4xIGIvb3BlbmdsL3Rvb2xzL2dsZ2VuL2dsc3BlYy0xLjEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDkxNDlhN2YuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL2dsc3BlYy0xLjEKKysrIC9kZXYvbnVsbApAQCAtMSw0MiArMCwwIEBACi12b2lkIGdsQmluZEJ1ZmZlciAoIEdMZW51bSB0YXJnZXQsIEdMdWludCBidWZmZXIgKQ0KLXZvaWQgZ2xCdWZmZXJEYXRhICggR0xlbnVtIHRhcmdldCwgR0xzaXplaXB0ciBzaXplLCBjb25zdCBHTHZvaWQgKmRhdGEsIEdMZW51bSB1c2FnZSApDQotdm9pZCBnbEJ1ZmZlclN1YkRhdGEgKCBHTGVudW0gdGFyZ2V0LCBHTGludHB0ciBvZmZzZXQsIEdMc2l6ZWlwdHIgc2l6ZSwgY29uc3QgR0x2b2lkICpkYXRhICkNCi12b2lkIGdsQ2xpcFBsYW5lZiAoIEdMZW51bSBwbGFuZSwgY29uc3QgR0xmbG9hdCAqZXF1YXRpb24gKQ0KLXZvaWQgZ2xDbGlwUGxhbmV4ICggR0xlbnVtIHBsYW5lLCBjb25zdCBHTGZpeGVkICplcXVhdGlvbiApDQotdm9pZCBnbENvbG9yNHViICggR0x1Ynl0ZSByZWQsIEdMdWJ5dGUgZ3JlZW4sIEdMdWJ5dGUgYmx1ZSwgR0x1Ynl0ZSBhbHBoYSApDQotdm9pZCBnbENvbG9yUG9pbnRlciAoIEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgR0xpbnQgb2Zmc2V0ICkNCi12b2lkIGdsRGVsZXRlQnVmZmVycyAoIEdMc2l6ZWkgbiwgY29uc3QgR0x1aW50ICpidWZmZXJzICkNCi12b2lkIGdsRHJhd0VsZW1lbnRzICggR0xlbnVtIG1vZGUsIEdMc2l6ZWkgY291bnQsIEdMZW51bSB0eXBlLCBHTGludCBvZmZzZXQgKQ0KLXZvaWQgZ2xHZW5CdWZmZXJzICggR0xzaXplaSBuLCBHTHVpbnQgKmJ1ZmZlcnMgKQ0KLXZvaWQgZ2xHZXRCb29sZWFudiAoIEdMZW51bSBwbmFtZSwgR0xib29sZWFuICpwYXJhbXMgKQ0KLXZvaWQgZ2xHZXRCdWZmZXJQYXJhbWV0ZXJpdiAoIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xpbnQgKnBhcmFtcyApDQotdm9pZCBnbEdldENsaXBQbGFuZWYgKCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgKmVxbiApDQotdm9pZCBnbEdldENsaXBQbGFuZXggKCBHTGVudW0gcG5hbWUsIEdMZml4ZWQgKmVxbiApDQotdm9pZCBnbEdldEZpeGVkdiAoIEdMZW51bSBwbmFtZSwgR0xmaXhlZCAqcGFyYW1zICkNCi12b2lkIGdsR2V0RmxvYXR2ICggR0xlbnVtIHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMgKQ0KLXZvaWQgZ2xHZXRMaWdodGZ2ICggR0xlbnVtIGxpZ2h0LCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgKnBhcmFtcyApDQotdm9pZCBnbEdldExpZ2h0eHYgKCBHTGVudW0gbGlnaHQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCAqcGFyYW1zICkNCi12b2lkIGdsR2V0TWF0ZXJpYWxmdiAoIEdMZW51bSBmYWNlLCBHTGVudW0gcG5hbWUsIEdMZmxvYXQgKnBhcmFtcyApDQotdm9pZCBnbEdldE1hdGVyaWFseHYgKCBHTGVudW0gZmFjZSwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkICpwYXJhbXMgKQ0KLXZvaWQgZ2xHZXRUZXhFbnZpdiAoIEdMZW51bSBlbnYsIEdMZW51bSBwbmFtZSwgR0xpbnQgKnBhcmFtcyApDQotdm9pZCBnbEdldFRleEVudnh2ICggR0xlbnVtIGVudiwgR0xlbnVtIHBuYW1lLCBHTGZpeGVkICpwYXJhbXMgKQ0KLXZvaWQgZ2xHZXRUZXhQYXJhbWV0ZXJmdiAoIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmbG9hdCAqcGFyYW1zICkNCi12b2lkIGdsR2V0VGV4UGFyYW1ldGVyaXYgKCBHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMaW50ICpwYXJhbXMgKQ0KLXZvaWQgZ2xHZXRUZXhQYXJhbWV0ZXJ4diAoIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgR0xmaXhlZCAqcGFyYW1zICkNCi1HTGJvb2xlYW4gZ2xJc0J1ZmZlciAoIEdMdWludCBidWZmZXIgKQ0KLUdMYm9vbGVhbiBnbElzRW5hYmxlZCAoIEdMZW51bSBjYXAgKQ0KLUdMYm9vbGVhbiBnbElzVGV4dHVyZSAoIEdMdWludCB0ZXh0dXJlICkNCi12b2lkIGdsTm9ybWFsUG9pbnRlciAoIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgR0xpbnQgb2Zmc2V0ICkNCi12b2lkIGdsUG9pbnRQYXJhbWV0ZXJmICggR0xlbnVtIHBuYW1lLCBHTGZsb2F0IHBhcmFtICkNCi12b2lkIGdsUG9pbnRQYXJhbWV0ZXJmdiAoIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmbG9hdCAqcGFyYW1zICkNCi12b2lkIGdsUG9pbnRQYXJhbWV0ZXJ4ICggR0xlbnVtIHBuYW1lLCBHTGZpeGVkIHBhcmFtICkNCi12b2lkIGdsUG9pbnRQYXJhbWV0ZXJ4diAoIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zICkNCi12b2lkIGdsUG9pbnRTaXplUG9pbnRlck9FUyAoIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgY29uc3QgR0x2b2lkICpwb2ludGVyICkNCi12b2lkIGdsVGV4Q29vcmRQb2ludGVyICggR0xpbnQgc2l6ZSwgR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBHTGludCBvZmZzZXQgKQ0KLXZvaWQgZ2xUZXhFbnZpICggR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBHTGludCBwYXJhbSApDQotdm9pZCBnbFRleEVudml2ICggR0xlbnVtIHRhcmdldCwgR0xlbnVtIHBuYW1lLCBjb25zdCBHTGludCAqcGFyYW1zICkNCi12b2lkIGdsVGV4UGFyYW1ldGVyZnYgKCBHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMZmxvYXQgKnBhcmFtcyApDQotdm9pZCBnbFRleFBhcmFtZXRlcmkgKCBHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIEdMaW50IHBhcmFtICkNCi12b2lkIGdsVGV4UGFyYW1ldGVyaXYgKCBHTGVudW0gdGFyZ2V0LCBHTGVudW0gcG5hbWUsIGNvbnN0IEdMaW50ICpwYXJhbXMgKQ0KLXZvaWQgZ2xUZXhQYXJhbWV0ZXJ4diAoIEdMZW51bSB0YXJnZXQsIEdMZW51bSBwbmFtZSwgY29uc3QgR0xmaXhlZCAqcGFyYW1zICkNCi12b2lkIGdsVmVydGV4UG9pbnRlciAoIEdMaW50IHNpemUsIEdMZW51bSB0eXBlLCBHTHNpemVpIHN0cmlkZSwgR0xpbnQgb2Zmc2V0ICkNCmRpZmYgLS1naXQgYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vZ2xzcGVjLTEuMWV4dCBiL29wZW5nbC90b29scy9nbGdlbi9nbHNwZWMtMS4xZXh0CmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjYzA4YzczLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9nbHNwZWMtMS4xZXh0CisrKyAvZGV2L251bGwKQEAgLTEsMTYgKzAsMCBAQAotdm9pZCBnbEN1cnJlbnRQYWxldHRlTWF0cml4T0VTICggR0x1aW50IG1hdHJpeHBhbGV0dGVpbmRleCApDQotdm9pZCBnbERyYXdUZXhmT0VTICggR0xmbG9hdCB4LCBHTGZsb2F0IHksIEdMZmxvYXQgeiwgR0xmbG9hdCB3aWR0aCwgR0xmbG9hdCBoZWlnaHQgKQ0KLXZvaWQgZ2xEcmF3VGV4ZnZPRVMgKCBjb25zdCBHTGZsb2F0ICpjb29yZHMgKQ0KLXZvaWQgZ2xEcmF3VGV4aU9FUyAoIEdMaW50IHgsIEdMaW50IHksIEdMaW50IHosIEdMaW50IHdpZHRoLCBHTGludCBoZWlnaHQgKQ0KLXZvaWQgZ2xEcmF3VGV4aXZPRVMgKCBjb25zdCBHTGludCAqY29vcmRzICkNCi12b2lkIGdsRHJhd1RleHNPRVMgKCBHTHNob3J0IHgsIEdMc2hvcnQgeSwgR0xzaG9ydCB6LCBHTHNob3J0IHdpZHRoLCBHTHNob3J0IGhlaWdodCApDQotdm9pZCBnbERyYXdUZXhzdk9FUyAoIGNvbnN0IEdMc2hvcnQgKmNvb3JkcyApDQotdm9pZCBnbERyYXdUZXh4T0VTICggR0xmaXhlZCB4LCBHTGZpeGVkIHksIEdMZml4ZWQgeiwgR0xmaXhlZCB3aWR0aCwgR0xmaXhlZCBoZWlnaHQgKQ0KLXZvaWQgZ2xEcmF3VGV4eHZPRVMgKCBjb25zdCBHTGZpeGVkICpjb29yZHMgKQ0KLXZvaWQgZ2xFbmFibGUgKCBHTGVudW0gY2FwICkNCi12b2lkIGdsRW5hYmxlQ2xpZW50U3RhdGUgKCBHTGVudW0gYXJyYXkgKQ0KLXZvaWQgZ2xMb2FkUGFsZXR0ZUZyb21Nb2RlbFZpZXdNYXRyaXhPRVMgKCB2b2lkICkNCi12b2lkIGdsTWF0cml4SW5kZXhQb2ludGVyT0VTICggR0xpbnQgc2l6ZSwgR0xlbnVtIHR5cGUsIEdMc2l6ZWkgc3RyaWRlLCBjb25zdCBHTHZvaWQgKnBvaW50ZXIgKQ0KLXZvaWQgZ2xNYXRyaXhJbmRleFBvaW50ZXJPRVMgKCBHTGludCBzaXplLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIEdMaW50IG9mZnNldCApDQotdm9pZCBnbFdlaWdodFBvaW50ZXJPRVMgKCBHTGludCBzaXplLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIGNvbnN0IEdMdm9pZCAqcG9pbnRlciApDQotdm9pZCBnbFdlaWdodFBvaW50ZXJPRVMgKCBHTGludCBzaXplLCBHTGVudW0gdHlwZSwgR0xzaXplaSBzdHJpZGUsIEdMaW50IG9mZnNldCApDQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL2dsc3BlYy0xLjFleHRwYWNrIGIvb3BlbmdsL3Rvb2xzL2dsZ2VuL2dsc3BlYy0xLjFleHRwYWNrCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBjYTllNmQyLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9nbHNwZWMtMS4xZXh0cGFjaworKysgL2Rldi9udWxsCkBAIC0xLDM4ICswLDAgQEAKLXZvaWQgZ2xCaW5kRnJhbWVidWZmZXJPRVMgKCBHTGludCB0YXJnZXQsIEdMaW50IGZyYW1lYnVmZmVyICkKLXZvaWQgZ2xCaW5kUmVuZGVyYnVmZmVyT0VTICggR0xpbnQgdGFyZ2V0LCBHTGludCByZW5kZXJidWZmZXIgKQotdm9pZCBnbEJpbmRUZXh0dXJlICggR0xpbnQgdGFyZ2V0LCBHTGludCB0ZXh0dXJlICkKLXZvaWQgZ2xCbGVuZEVxdWF0aW9uICggR0xpbnQgbW9kZSApCi12b2lkIGdsQmxlbmRFcXVhdGlvblNlcGFyYXRlICggR0xpbnQgbW9kZVJHQiwgR0xpbnQgbW9kZUFscGhhICkKLXZvaWQgZ2xCbGVuZEZ1bmNTZXBhcmF0ZSAoIEdMaW50IHNyY1JHQiwgR0xpbnQgZHN0UkdCLCBHTGludCBzcmNBbHBoYSwgR0xpbnQgZHN0QWxwaGEgKQotR0xpbnQgZ2xDaGVja0ZyYW1lYnVmZmVyU3RhdHVzT0VTICggR0xpbnQgdGFyZ2V0ICkKLXZvaWQgZ2xDb21wcmVzc2VkVGV4SW1hZ2UyRCAoIEdMZW51bSB0YXJnZXQsIEdMaW50IGxldmVsLCBHTGVudW0gaW50ZXJuYWxmb3JtYXQsIEdMc2l6ZWkgd2lkdGgsIEdMc2l6ZWkgaGVpZ2h0LCBHTGludCBib3JkZXIsIEdMc2l6ZWkgaW1hZ2VTaXplLCBjb25zdCBHTHZvaWQgKmRhdGEgKQotdm9pZCBnbENvcHlUZXhJbWFnZTJEICggR0xpbnQgdGFyZ2V0LCBHTGludCBsZXZlbCwgR0xpbnQgaW50ZXJuYWxmb3JtYXQsIEdMaW50IHgsIEdMaW50IHksIEdMaW50IHdpZHRoLCBHTGludCBoZWlnaHQsIEdMaW50IGJvcmRlciApCi12b2lkIGdsRGVsZXRlRnJhbWVidWZmZXJzT0VTICggR0xpbnQgbiwgR0xpbnQgKmZyYW1lYnVmZmVycyApCi12b2lkIGdsRGVsZXRlUmVuZGVyYnVmZmVyc09FUyAoIEdMaW50IG4sIEdMaW50ICpyZW5kZXJidWZmZXJzICkKLXZvaWQgZ2xFbmFibGUgKCBHTGludCBjYXAgKQotdm9pZCBnbEZyYW1lYnVmZmVyUmVuZGVyYnVmZmVyT0VTICggR0xpbnQgdGFyZ2V0LCBHTGludCBhdHRhY2htZW50LCBHTGludCByZW5kZXJidWZmZXJ0YXJnZXQsIEdMaW50IHJlbmRlcmJ1ZmZlciApCi12b2lkIGdsRnJhbWVidWZmZXJUZXh0dXJlMkRPRVMgKCBHTGludCB0YXJnZXQsIEdMaW50IGF0dGFjaG1lbnQsIEdMaW50IHRleHRhcmdldCwgR0xpbnQgdGV4dHVyZSwgR0xpbnQgbGV2ZWwgKQotdm9pZCBnbEdlbmVyYXRlTWlwbWFwT0VTICggR0xpbnQgdGFyZ2V0ICkKLXZvaWQgZ2xHZW5GcmFtZWJ1ZmZlcnNPRVMgKCBHTGludCBuLCBHTGludCAqZnJhbWVidWZmZXJzICkKLXZvaWQgZ2xHZW5SZW5kZXJidWZmZXJzT0VTICggR0xpbnQgbiwgR0xpbnQgKnJlbmRlcmJ1ZmZlcnMgKQotdm9pZCBnbEdldEZyYW1lYnVmZmVyQXR0YWNobWVudFBhcmFtZXRlcml2T0VTICggR0xpbnQgdGFyZ2V0LCBHTGludCBhdHRhY2htZW50LCBHTGludCBwbmFtZSwgR0xpbnQgKnBhcmFtcyApCi12b2lkIGdsR2V0SW50ZWdlcnYgKCBHTGludCBwbmFtZSwgR0xpbnQgKnBhcmFtcyApCi12b2lkIGdsR2V0UmVuZGVyYnVmZmVyUGFyYW1ldGVyaXZPRVMgKCBHTGludCB0YXJnZXQsIEdMaW50IHBuYW1lLCBHTGludCAqcGFyYW1zICkKLXZvaWQgZ2xHZXRUZXhHZW5mdiAoIEdMaW50IGNvb3JkLCBHTGludCBwbmFtZSwgR0xmbG9hdCAqcGFyYW1zICkKLXZvaWQgZ2xHZXRUZXhHZW5pdiAoIEdMaW50IGNvb3JkLCBHTGludCBwbmFtZSwgR0xpbnQgKnBhcmFtcyApCi12b2lkIGdsR2V0VGV4R2VueHYgKCBHTGludCBjb29yZCwgR0xpbnQgcG5hbWUsIEdMaW50ICpwYXJhbXMgKQotR0xib29sZWFuIGdsSXNGcmFtZWJ1ZmZlck9FUyAoIEdMaW50IGZyYW1lYnVmZmVyICkKLUdMYm9vbGVhbiBnbElzUmVuZGVyYnVmZmVyT0VTICggR0xpbnQgcmVuZGVyYnVmZmVyICkKLXZvaWQgZ2xSZW5kZXJidWZmZXJTdG9yYWdlT0VTICggR0xpbnQgdGFyZ2V0LCBHTGludCBpbnRlcm5hbGZvcm1hdCwgR0xpbnQgd2lkdGgsIEdMaW50IGhlaWdodCApCi12b2lkIGdsU3RlbmNpbE9wICggR0xpbnQgZmFpbCwgR0xpbnQgemZhaWwsIEdMaW50IHpwYXNzICkKLXZvaWQgZ2xUZXhFbnZmICggR0xpbnQgdGFyZ2V0LCBHTGludCBwbmFtZSwgR0xmbG9hdCBwYXJhbSApCi12b2lkIGdsVGV4RW52ZnYgKCBHTGludCB0YXJnZXQsIEdMaW50IHBuYW1lLCBHTGZsb2F0ICpwYXJhbXMgKQotdm9pZCBnbFRleEVudnggKCBHTGludCB0YXJnZXQsIEdMaW50IHBuYW1lLCBHTGludCBwYXJhbSApCi12b2lkIGdsVGV4RW52eHYgKCBHTGludCB0YXJnZXQsIEdMaW50IHBuYW1lLCBHTGludCAqcGFyYW1zICkKLXZvaWQgZ2xUZXhHZW5mICggR0xpbnQgY29vcmQsIEdMaW50IHBuYW1lLCBHTGZsb2F0IHBhcmFtICkKLXZvaWQgZ2xUZXhHZW5mdiAoIEdMaW50IGNvb3JkLCBHTGludCBwbmFtZSwgR0xmbG9hdCAqcGFyYW1zICkKLXZvaWQgZ2xUZXhHZW5pICggR0xpbnQgY29vcmQsIEdMaW50IHBuYW1lLCBHTGludCBwYXJhbSApCi12b2lkIGdsVGV4R2VuaXYgKCBHTGludCBjb29yZCwgR0xpbnQgcG5hbWUsIEdMaW50ICpwYXJhbXMgKQotdm9pZCBnbFRleEdlbnggKCBHTGludCBjb29yZCwgR0xpbnQgcG5hbWUsIEdMaW50IHBhcmFtICkKLXZvaWQgZ2xUZXhHZW54diAoIEdMaW50IGNvb3JkLCBHTGludCBwbmFtZSwgR0xpbnQgKnBhcmFtcyApCi12b2lkIGdsVGV4UGFyYW1ldGVyZiAoIEdMaW50IHRhcmdldCwgR0xpbnQgcG5hbWUsIEdMZmxvYXQgcGFyYW0gKQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL2dsc3BlYy1jaGVja3MgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vZ2xzcGVjLWNoZWNrcwpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTg0ZWQ2NS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vZ2xzcGVjLWNoZWNrcworKysgL2Rldi9udWxsCkBAIC0xLDU5ICswLDAgQEAKLWdsQ2xpcFBsYW5lZiBjaGVjayBlcXVhdGlvbiA0DQotZ2xDbGlwUGxhbmV4IGNoZWNrIGVxdWF0aW9uIDQNCi1nbERlbGV0ZUJ1ZmZlcnMgY2hlY2sgYnVmZmVycyBuIA0KLWdsRGVsZXRlVGV4dHVyZXMgY2hlY2sgdGV4dHVyZXMgbg0KLWdsRHJhd0VsZW1lbnRzIGNoZWNrX0FJT09CRSBpbmRpY2VzIGNvdW50DQotZ2xGb2cgaWZjaGVjayBwYXJhbXMgMSBwbmFtZSBHTF9GT0dfTU9ERSxHTF9GT0dfREVOU0lUWSxHTF9GT0dfU1RBUlQsR0xfRk9HX0VORCBpZmNoZWNrIHBhcmFtcyA0IHBuYW1lIEdMX0ZPR19DT0xPUg0KLWdsR2VuQnVmZmVycyBjaGVjayBidWZmZXJzIG4NCi1nbEdlblRleHR1cmVzIGNoZWNrIHRleHR1cmVzIG4NCi1nbEdldENsaXBQbGFuZSBjaGVjayBlcW4gNA0KLWdsR2V0SW50ZWdlcnYgaWZjaGVjayBwYXJhbXMgMSBwbmFtZSBHTF9BTFBIQV9CSVRTLEdMX0FMUEhBX1RFU1RfRlVOQyxHTF9BTFBIQV9URVNUX1JFRixHTF9CTEVORF9EU1QsR0xfQkxVRV9CSVRTLEdMX0NPTE9SX0FSUkFZX0JVRkZFUl9CSU5ESU5HLEdMX0NPTE9SX0FSUkFZX1NJWkUsR0xfQ09MT1JfQVJSQVlfU1RSSURFLEdMX0NPTE9SX0FSUkFZX1RZUEUsR0xfQ1VMTF9GQUNFLEdMX0RFUFRIX0JJVFMsR0xfREVQVEhfQ0xFQVJfVkFMVUUsR0xfREVQVEhfRlVOQyxHTF9ERVBUSF9XUklURU1BU0ssR0xfRk9HX0RFTlNJVFksR0xfRk9HX0VORCxHTF9GT0dfTU9ERSxHTF9GT0dfU1RBUlQsR0xfRlJPTlRfRkFDRSxHTF9HUkVFTl9CSVRTLEdMX0lNUExFTUVOVEFUSU9OX0NPTE9SX1JFQURfRk9STUFUX09FUyxHTF9JTVBMRU1FTlRBVElPTl9DT0xPUl9SRUFEX1RZUEVfT0VTLEdMX0xJR0hUX01PREVMX1RXT19TSURFLEdMX0xJTkVfU01PT1RIX0hJTlQsR0xfTElORV9XSURUSCxHTF9MT0dJQ19PUF9NT0RFLEdMX01BVFJJWF9JTkRFWF9BUlJBWV9CVUZGRVJfQklORElOR19PRVMsR0xfTUFUUklYX0lOREVYX0FSUkFZX1NJWkVfT0VTLEdMX01BVFJJWF9JTkRFWF9BUlJBWV9TVFJJREVfT0VTLEdMX01BVFJJWF9JTkRFWF9BUlJBWV9UWVBFX09FUyxHTF9NQVRSSVhfTU9ERSxHTF9NQVhfQ0xJUF9QTEFORVMsR0xfTUFYX0VMRU1FTlRTX0lORElDRVMsR0xfTUFYX0VMRU1FTlRTX1ZFUlRJQ0VTLEdMX01BWF9MSUdIVFMsR0xfTUFYX01PREVMVklFV19TVEFDS19ERVBUSCxHTF9NQVhfUEFMRVRURV9NQVRSSUNFU19PRVMsR0xfTUFYX1BST0pFQ1RJT05fU1RBQ0tfREVQVEgsR0xfTUFYX1RFWFRVUkVfU0laRSxHTF9NQVhfVEVYVFVSRV9TVEFDS19ERVBUSCxHTF9NQVhfVEVYVFVSRV9VTklUUyxHTF9NQVhfVkVSVEVYX1VOSVRTX09FUyxHTF9NT0RFTFZJRVdfU1RBQ0tfREVQVEgsR0xfTk9STUFMX0FSUkFZX0JVRkZFUl9CSU5ESU5HLEdMX05PUk1BTF9BUlJBWV9TVFJJREUsR0xfTk9STUFMX0FSUkFZX1RZUEUsR0xfTlVNX0NPTVBSRVNTRURfVEVYVFVSRV9GT1JNQVRTLEdMX1BBQ0tfQUxJR05NRU5ULEdMX1BFUlNQRUNUSVZFX0NPUlJFQ1RJT05fSElOVCxHTF9QT0lOVF9TSVpFLEdMX1BPSU5UX1NJWkVfQVJSQVlfQlVGRkVSX0JJTkRJTkdfT0VTLEdMX1BPSU5UX1NJWkVfQVJSQVlfU1RSSURFX09FUyxHTF9QT0lOVF9TSVpFX0FSUkFZX1RZUEVfT0VTLEdMX1BPSU5UX1NNT09USF9ISU5ULEdMX1BPTFlHT05fT0ZGU0VUX0ZBQ1RPUixHTF9QT0xZR09OX09GRlNFVF9VTklUUyxHTF9QUk9KRUNUSU9OX1NUQUNLX0RFUFRILEdMX1JFRF9CSVRTLEdMX1NIQURFX01PREVMLEdMX1NURU5DSUxfQklUUyxHTF9TVEVOQ0lMX0NMRUFSX1ZBTFVFLEdMX1NURU5DSUxfRkFJTCxHTF9TVEVOQ0lMX0ZVTkMsR0xfU1RFTkNJTF9QQVNTX0RFUFRIX0ZBSUwsR0xfU1RFTkNJTF9QQVNTX0RFUFRIX1BBU1MsR0xfU1RFTkNJTF9SRUYsR0xfU1RFTkNJTF9WQUxVRV9NQVNLLEdMX1NURU5DSUxfV1JJVEVNQVNLLEdMX1NVQlBJWEVMX0JJVFMsR0xfVEVYVFVSRV9CSU5ESU5HXzJELEdMX1RFWFRVUkVfQ09PUkRfQVJSQVlfQlVGRkVSX0JJTkRJTkcsR0xfVEVYVFVSRV9DT09SRF9BUlJBWV9TSVpFLEdMX1RFWFRVUkVfQ09PUkRfQVJSQVlfU1RSSURFLEdMX1RFWFRVUkVfQ09PUkRfQVJSQVlfVFlQRSxHTF9URVhUVVJFX1NUQUNLX0RFUFRILEdMX1VOUEFDS19BTElHTk1FTlQsR0xfVkVSVEVYX0FSUkFZX0JVRkZFUl9CSU5ESU5HLEdMX1ZFUlRFWF9BUlJBWV9TSVpFLEdMX1ZFUlRFWF9BUlJBWV9TVFJJREUsR0xfVkVSVEVYX0FSUkFZX1RZUEUsR0xfV0VJR0hUX0FSUkFZX0JVRkZFUl9CSU5ESU5HX09FUyxHTF9XRUlHSFRfQVJSQVlfU0laRV9PRVMsR0xfV0VJR0hUX0FSUkFZX1NUUklERV9PRVMsR0xfV0VJR0hUX0FSUkFZX1RZUEVfT0VTIGlmY2hlY2sgcGFyYW1zIDIgcG5hbWUgR0xfQUxJQVNFRF9QT0lOVF9TSVpFX1JBTkdFLEdMX0FMSUFTRURfTElORV9XSURUSF9SQU5HRSxHTF9ERVBUSF9SQU5HRSxHTF9NQVhfVklFV1BPUlRfRElNUyxHTF9TTU9PVEhfTElORV9XSURUSF9SQU5HRSxHTF9TTU9PVEhfUE9JTlRfU0laRV9SQU5HRSBpZmNoZWNrIHBhcmFtcyA0IHBuYW1lIEdMX0NPTE9SX0NMRUFSX1ZBTFVFLEdMX0NPTE9SX1dSSVRFTUFTSyxHTF9TQ0lTU09SX0JPWCxHTF9WSUVXUE9SVCBpZmNoZWNrIHBhcmFtcyAxNiBwbmFtZSBHTF9NT0RFTFZJRVdfTUFUUklYLEdMX01PREVMVklFV19NQVRSSVhfRkxPQVRfQVNfSU5UX0JJVFNfT0VTLEdMX1BST0pFQ1RJT05fTUFUUklYLEdMX1BST0pFQ1RJT05fTUFUUklYX0ZMT0FUX0FTX0lOVF9CSVRTX09FUyxHTF9URVhUVVJFX01BVFJJWCxHTF9URVhUVVJFX01BVFJJWF9GTE9BVF9BU19JTlRfQklUU19PRVMgaWZjaGVjayBwYXJhbXMgX05VTV9DT01QUkVTU0VEX1RFWFRVUkVfRk9STUFUUyBwbmFtZSBHTF9DT01QUkVTU0VEX1RFWFRVUkVfRk9STUFUUyxHTF9GT0dfQ09MT1IsR0xfTElHSFRfTU9ERUxfQU1CSUVOVA0KLWdsR2V0TGlnaHQgaWZjaGVjayBwYXJhbXMgMSBwbmFtZSBHTF9TUE9UX0VYUE9ORU5ULEdMX1NQT1RfQ1VUT0ZGLEdMX0NPTlNUQU5UX0FUVEVOVUFUSU9OLEdMX0xJTkVBUl9BVFRFTlVBVElPTixHTF9RVUFEUkFUSUNfQVRURU5VQVRJT04gaWZjaGVjayBwYXJhbXMgMyBwbmFtZSBHTF9TUE9UX0RJUkVDVElPTiBpZmNoZWNrIHBhcmFtcyA0IHBuYW1lIEdMX0FNQklFTlQsR0xfRElGRlVTRSxHTF9TUEVDVUxBUixHTF9FTUlTU0lPTg0KLWdsR2V0TWF0ZXJpYWwgaWZjaGVjayBwYXJhbXMgMSBwbmFtZSBHTF9TSElOSU5FU1MgaWZjaGVjayBwYXJhbXMgNCBwbmFtZSBHTF9BTUJJRU5ULEdMX0RJRkZVU0UsR0xfU1BFQ1VMQVIsR0xfRU1JU1NJT04sR0xfQU1CSUVOVF9BTkRfRElGRlVTRQ0KLWdsR2V0VGV4RW52IGlmY2hlY2sgcGFyYW1zIDEgcG5hbWUgR0xfVEVYVFVSRV9FTlZfTU9ERSxHTF9DT01CSU5FX1JHQixHTF9DT01CSU5FX0FMUEhBIGlmY2hlY2sgcGFyYW1zIDQgcG5hbWUgR0xfVEVYVFVSRV9FTlZfQ09MT1INCi1nbEdldFRleFBhcmFtZXRlciBjaGVjayBwYXJhbXMgMQ0KLWdsTGlnaHRNb2RlbCBpZmNoZWNrIHBhcmFtcyAxIHBuYW1lIEdMX0xJR0hUX01PREVMX1RXT19TSURFIGlmY2hlY2sgcGFyYW1zIDQgcG5hbWUgR0xfTElHSFRfTU9ERUxfQU1CSUVOVA0KLWdsTGlnaHQgaWZjaGVjayBwYXJhbXMgMSBwbmFtZSBHTF9TUE9UX0VYUE9ORU5ULEdMX1NQT1RfQ1VUT0ZGLEdMX0NPTlNUQU5UX0FUVEVOVUFUSU9OLEdMX0xJTkVBUl9BVFRFTlVBVElPTixHTF9RVUFEUkFUSUNfQVRURU5VQVRJT04gaWZjaGVjayBwYXJhbXMgMyBwbmFtZSBHTF9TUE9UX0RJUkVDVElPTiBpZmNoZWNrIHBhcmFtcyA0IHBuYW1lIEdMX0FNQklFTlQsR0xfRElGRlVTRSxHTF9TUEVDVUxBUixHTF9FTUlTU0lPTg0KLWdsTG9hZE1hdHJpeCBjaGVjayBtIDE2DQotZ2xNYXRlcmlhbCBpZmNoZWNrIHBhcmFtcyAxIHBuYW1lIEdMX1NISU5JTkVTUyBpZmNoZWNrIHBhcmFtcyA0IHBuYW1lIEdMX0FNQklFTlQsR0xfRElGRlVTRSxHTF9TUEVDVUxBUixHTF9FTUlTU0lPTixHTF9BTUJJRU5UX0FORF9ESUZGVVNFDQotZ2xNdWx0TWF0cml4IGNoZWNrIG0gMTYNCi1nbFBvaW50UGFyYW1ldGVyIGNoZWNrIHBhcmFtcyAxDQotZ2xUZXhFbnYgaWZjaGVjayBwYXJhbXMgMSBwbmFtZSBHTF9URVhUVVJFX0VOVl9NT0RFLEdMX0NPTUJJTkVfUkdCLEdMX0NPTUJJTkVfQUxQSEEgaWZjaGVjayBwYXJhbXMgNCBwbmFtZSBHTF9URVhUVVJFX0VOVl9DT0xPUg0KLWdsVGV4SW1hZ2UyRCBudWxsQWxsb3dlZA0KLWdsVGV4U3ViSW1hZ2UyRCBudWxsQWxsb3dlZA0KLWdsQnVmZmVyRGF0YSBudWxsQWxsb3dlZA0KLWdsVGV4UGFyYW1ldGVyIGNoZWNrIHBhcmFtcyAxDQotZ2xRdWVyeU1hdHJpeHhPRVMgY2hlY2sgbWFudGlzc2EgMTYgY2hlY2sgZXhwb25lbnQgMTYgcmV0dXJuIC0xDQotZ2xEcmF3VGV4ZnZPRVMgY2hlY2sgY29vcmRzIDUNCi1nbERyYXdUZXhpdk9FUyBjaGVjayBjb29yZHMgNQ0KLWdsRHJhd1RleHN2T0VTIGNoZWNrIGNvb3JkcyA1DQotZ2xEcmF3VGV4eHZPRVMgY2hlY2sgY29vcmRzIDUNCi1nbEJpbmRGcmFtZWJ1ZmZlck9FUyB1bnN1cHBvcnRlZA0KLWdsQmluZFJlbmRlcmJ1ZmZlck9FUyB1bnN1cHBvcnRlZA0KLWdsQmxlbmRFcXVhdGlvbiB1bnN1cHBvcnRlZA0KLWdsQmxlbmRFcXVhdGlvblNlcGFyYXRlIHVuc3VwcG9ydGVkDQotZ2xCbGVuZEZ1bmNTZXBhcmF0ZSB1bnN1cHBvcnRlZA0KLWdsQ2hlY2tGcmFtZWJ1ZmZlclN0YXR1c09FUyB1bnN1cHBvcnRlZCByZXR1cm4gMA0KLWdsQ3VycmVudFBhbGV0dGVNYXRyaXhPRVMgdW5zdXBwb3J0ZWQNCi1nbERlbGV0ZUZyYW1lYnVmZmVyc09FUyB1bnN1cHBvcnRlZA0KLWdsRGVsZXRlUmVuZGVyYnVmZmVyc09FUyB1bnN1cHBvcnRlZA0KLWdsRnJhbWVidWZmZXJSZW5kZXJidWZmZXJPRVMgdW5zdXBwb3J0ZWQNCi1nbEZyYW1lYnVmZmVyU3RvcmFnZU9FUyB1bnN1cHBvcnRlZA0KLWdsRnJhbWVidWZmZXJUZXh0dXJlMkRPRVMgdW5zdXBwb3J0ZWQNCi1nbEdlbkZyYW1lYnVmZmVyc09FUyB1bnN1cHBvcnRlZA0KLWdsR2VuUmVuZGVyYnVmZmVyc09FUyB1bnN1cHBvcnRlZA0KLWdsR2VuZXJhdGVNaXBtYXBPRVMgdW5zdXBwb3J0ZWQNCi1nbEdldEJ1ZmZlclBhcmFtZXRlciB1bnN1cHBvcnRlZA0KLWdsR2V0RnJhbWVidWZmZXJBdHRhY2htZW50UGFyYW1ldGVyaXZPRVMgdW5zdXBwb3J0ZWQNCi1nbEdldFJlbmRlcmJ1ZmZlclBhcmFtZXRlcml2T0VTIHVuc3VwcG9ydGVkDQotZ2xHZXRUZXhHZW4gdW5zdXBwb3J0ZWQNCi1nbElzRnJhbWVidWZmZXJPRVMgdW5zdXBwb3J0ZWQgcmV0dXJuIEpOSV9GQUxTRQ0KLWdsSXNSZW5kZXJidWZmZXJPRVMgdW5zdXBwb3J0ZWQgcmV0dXJuIEpOSV9GQUxTRQ0KLWdsTG9hZFBhbGV0dGVGcm9tTW9kZWxWaWV3TWF0cml4T0VTIHVuc3VwcG9ydGVkDQotZ2xNYXRyaXhJbmRleFBvaW50ZXJPRVMgdW5zdXBwb3J0ZWQNCi1nbFJlbmRlcmJ1ZmZlclN0b3JhZ2VPRVMgdW5zdXBwb3J0ZWQgcmV0dXJuIGZhbHNlDQotZ2xUZXhHZW4gdW5zdXBwb3J0ZWQNCi1nbFRleEdlbmYgdW5zdXBwb3J0ZWQNCi1nbFRleEdlbmkgdW5zdXBwb3J0ZWQNCi1nbFRleEdlbnggdW5zdXBwb3J0ZWQNCi1nbFdlaWdodFBvaW50ZXJPRVMgdW5zdXBwb3J0ZWQNCmRpZmYgLS1naXQgYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0NGdW5jLmphdmEgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0NGdW5jLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDA3OTRmNDEuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3NyYy9DRnVuYy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTU1ICswLDAgQEAKLQ0KLWltcG9ydCBqYXZhLnV0aWwuKjsNCi0NCi1wdWJsaWMgY2xhc3MgQ0Z1bmMgew0KLQ0KLSAgICBTdHJpbmcgb3JpZ2luYWw7DQotDQotICAgIENUeXBlIGZ0eXBlOw0KLSAgICBTdHJpbmcgZm5hbWU7DQotDQotICAgIExpc3Q8U3RyaW5nPiBhcmdOYW1lcyA9IG5ldyBBcnJheUxpc3Q8U3RyaW5nPigpOw0KLSAgICBMaXN0PENUeXBlPiBhcmdUeXBlcyA9IG5ldyBBcnJheUxpc3Q8Q1R5cGU+KCk7DQotDQotICAgIGJvb2xlYW4gaGFzUG9pbnRlckFyZyA9IGZhbHNlOw0KLSAgICBib29sZWFuIGhhc1R5cGVkUG9pbnRlckFyZyA9IGZhbHNlOw0KLQ0KLSAgICBwdWJsaWMgQ0Z1bmMoU3RyaW5nIG9yaWdpbmFsKSB7DQotICAgICAgICB0aGlzLm9yaWdpbmFsID0gb3JpZ2luYWw7DQotICAgIH0NCi0NCi0gICAgcHVibGljIFN0cmluZyBnZXRPcmlnaW5hbCgpIHsNCi0gICAgICAgIHJldHVybiBvcmlnaW5hbDsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgdm9pZCBzZXROYW1lKFN0cmluZyBmbmFtZSkgew0KLSAgICAgICAgdGhpcy5mbmFtZSA9IGZuYW1lOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0TmFtZSgpIHsNCi0gICAgICAgIHJldHVybiBmbmFtZTsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgdm9pZCBzZXRUeXBlKENUeXBlIGZ0eXBlKSB7DQotICAgICAgICB0aGlzLmZ0eXBlID0gZnR5cGU7DQotICAgIH0NCi0NCi0gICAgcHVibGljIENUeXBlIGdldFR5cGUoKSB7DQotICAgICAgICByZXR1cm4gZnR5cGU7DQotICAgIH0NCi0NCi0gICAgcHVibGljIHZvaWQgYWRkQXJndW1lbnQoU3RyaW5nIGFyZ05hbWUsIENUeXBlIGFyZ1R5cGUpIHsNCi0gICAgICAgIGFyZ05hbWVzLmFkZChhcmdOYW1lKTsNCi0gICAgICAgIGFyZ1R5cGVzLmFkZChhcmdUeXBlKTsNCi0NCi0gICAgICAgIGlmIChhcmdUeXBlLmlzUG9pbnRlcigpKSB7DQotICAgICAgICAgICAgaGFzUG9pbnRlckFyZyA9IHRydWU7DQotICAgICAgICB9DQotICAgICAgICBpZiAoYXJnVHlwZS5pc1R5cGVkUG9pbnRlcigpKSB7DQotICAgICAgICAgICAgaGFzVHlwZWRQb2ludGVyQXJnID0gdHJ1ZTsNCi0gICAgICAgIH0NCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgaW50IGdldE51bUFyZ3MoKSB7DQotICAgICAgICByZXR1cm4gYXJnTmFtZXMuc2l6ZSgpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBpbnQgZ2V0QXJnSW5kZXgoU3RyaW5nIG5hbWUpIHsNCi0gICAgICAgIGludCBsZW4gPSBhcmdOYW1lcy5zaXplKCk7DQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGxlbjsgaSsrKSB7DQotICAgICAgICAgICAgaWYgKG5hbWUuZXF1YWxzKGFyZ05hbWVzLmdldChpKSkpIHsNCi0gICAgICAgICAgICAgICAgcmV0dXJuIGk7DQotICAgICAgICAgICAgfQ0KLSAgICAgICAgfQ0KLSAgICAgICAgcmV0dXJuIC0xOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0QXJnTmFtZShpbnQgaW5kZXgpIHsNCi0gICAgICAgIHJldHVybiBhcmdOYW1lcy5nZXQoaW5kZXgpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBDVHlwZSBnZXRBcmdUeXBlKGludCBpbmRleCkgew0KLSAgICAgICAgcmV0dXJuIGFyZ1R5cGVzLmdldChpbmRleCk7DQotICAgIH0NCi0NCi0gICAgcHVibGljIGJvb2xlYW4gaGFzUG9pbnRlckFyZygpIHsNCi0gICAgICAgIHJldHVybiBoYXNQb2ludGVyQXJnOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBib29sZWFuIGhhc1R5cGVkUG9pbnRlckFyZygpIHsNCi0gICAgICAgIHJldHVybiBoYXNUeXBlZFBvaW50ZXJBcmc7DQotICAgIH0NCi0NCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsNCi0gICAgICAgIFN0cmluZyBzID0gICJGdW5jdGlvbiAiICsgZm5hbWUgKyAiIHJldHVybnMgIiArIGZ0eXBlICsgIjogIjsNCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgYXJnTmFtZXMuc2l6ZSgpOyBpKyspIHsNCi0gICAgICAgICAgICBpZiAoaSA+IDApIHsNCi0gICAgICAgICAgICAgICAgcyArPSAiLCAiOw0KLSAgICAgICAgICAgIH0NCi0gICAgICAgICAgICBzICs9IGFyZ1R5cGVzLmdldChpKSArICIgIiArIGFyZ05hbWVzLmdldChpKTsNCi0gICAgICAgIH0NCi0gICAgICAgIHJldHVybiBzOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBzdGF0aWMgQ0Z1bmMgcGFyc2VDRnVuYyhTdHJpbmcgcykgew0KLSAgICAgICAgQ0Z1bmMgY2Z1bmMgPSBuZXcgQ0Z1bmMocyk7DQotICAgICAgICBTdHJpbmdbXSB0b2tlbnMgPSBzLnNwbGl0KCJcXHMiKTsNCi0NCi0gICAgICAgIGludCBpID0gMDsNCi0gICAgICAgIENUeXBlIGZ0eXBlID0gbmV3IENUeXBlKCk7DQotICAgICAgICBTdHJpbmcgZnR5cGVOYW1lID0gdG9rZW5zW2krK107DQotICAgICAgICBpZiAoZnR5cGVOYW1lLmVxdWFscygiY29uc3QiKSkgew0KLSAgICAgICAgICAgIGZ0eXBlLnNldElzQ29uc3QodHJ1ZSk7DQotICAgICAgICAgICAgZnR5cGVOYW1lID0gdG9rZW5zW2krK107DQotICAgICAgICB9DQotICAgICAgICBmdHlwZS5zZXRCYXNlVHlwZShmdHlwZU5hbWUpOw0KLQ0KLSAgICAgICAgU3RyaW5nIGZuYW1lID0gdG9rZW5zW2krK107DQotICAgICAgICBpZiAoZm5hbWUuZXF1YWxzKCIqIikpIHsNCi0gICAgICAgICAgICBmdHlwZS5zZXRJc1BvaW50ZXIodHJ1ZSk7DQotICAgICAgICAgICAgZm5hbWUgPSB0b2tlbnNbaSsrXTsNCi0gICAgICAgIH0NCi0JDQotICAgICAgICBjZnVuYy5zZXROYW1lKGZuYW1lKTsNCi0gICAgICAgIGNmdW5jLnNldFR5cGUoZnR5cGUpOw0KLQkNCi0gICAgICAgIHdoaWxlIChpIDwgdG9rZW5zLmxlbmd0aCkgew0KLSAgICAgICAgICAgIFN0cmluZyB0b2sgPSB0b2tlbnNbaSsrXTsNCi0JICAgIA0KLSAgICAgICAgICAgIGlmICh0b2suZXF1YWxzKCIoIikpIHsNCi0gICAgICAgICAgICAgICAgY29udGludWU7DQotICAgICAgICAgICAgfQ0KLSAgICAgICAgICAgIGlmICh0b2suZXF1YWxzKCIpIikpIHsNCi0gICAgICAgICAgICAgICAgYnJlYWs7DQotICAgICAgICAgICAgfQ0KLQ0KLSAgICAgICAgICAgIENUeXBlIGFyZ1R5cGUgPSBuZXcgQ1R5cGUoKTsNCi0JICAgIA0KLSAgICAgICAgICAgIFN0cmluZyBhcmdUeXBlTmFtZSA9IHRvazsNCi0gICAgICAgICAgICBTdHJpbmcgYXJnTmFtZSA9ICIiOw0KLQkgICAgDQotICAgICAgICAgICAgaWYgKGFyZ1R5cGVOYW1lLmVxdWFscygiY29uc3QiKSkgew0KLSAgICAgICAgICAgICAgICBhcmdUeXBlLnNldElzQ29uc3QodHJ1ZSk7DQotICAgICAgICAgICAgICAgIGFyZ1R5cGVOYW1lID0gdG9rZW5zW2krK107DQotICAgICAgICAgICAgfQ0KLSAgICAgICAgICAgIGFyZ1R5cGUuc2V0QmFzZVR5cGUoYXJnVHlwZU5hbWUpOw0KLQ0KLSAgICAgICAgICAgIGlmIChhcmdUeXBlTmFtZS5lcXVhbHMoInZvaWQiKSkgew0KLSAgICAgICAgICAgICAgICBicmVhazsNCi0gICAgICAgICAgICB9DQotCSAgICANCi0gICAgICAgICAgICBhcmdOYW1lID0gdG9rZW5zW2krK107DQotICAgICAgICAgICAgaWYgKGFyZ05hbWUuc3RhcnRzV2l0aCgiKiIpKSB7DQotICAgICAgICAgICAgICAgIGFyZ1R5cGUuc2V0SXNQb2ludGVyKHRydWUpOw0KLSAgICAgICAgICAgICAgICBhcmdOYW1lID0gYXJnTmFtZS5zdWJzdHJpbmcoMSwgYXJnTmFtZS5sZW5ndGgoKSk7DQotICAgICAgICAgICAgfQ0KLSAgICAgICAgICAgIGlmIChhcmdOYW1lLmVuZHNXaXRoKCIsIikpIHsNCi0gICAgICAgICAgICAgICAgYXJnTmFtZSA9IGFyZ05hbWUuc3Vic3RyaW5nKDAsIGFyZ05hbWUubGVuZ3RoKCkgLSAxKTsNCi0gICAgICAgICAgICB9DQotCSAgICANCi0gICAgICAgICAgICBjZnVuYy5hZGRBcmd1bWVudChhcmdOYW1lLCBhcmdUeXBlKTsNCi0gICAgICAgIH0NCi0NCi0gICAgICAgIHJldHVybiBjZnVuYzsNCi0gICAgfQ0KLX0NCmRpZmYgLS1naXQgYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0NUeXBlLmphdmEgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0NUeXBlLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDMzMWVjNjIuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3NyYy9DVHlwZS5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsODUgKzAsMCBAQAotDQotcHVibGljIGNsYXNzIENUeXBlIHsNCi0NCi0gICAgU3RyaW5nIGJhc2VUeXBlOw0KLSAgICBib29sZWFuIGlzQ29uc3Q7DQotICAgIGJvb2xlYW4gaXNQb2ludGVyOw0KLQ0KLSAgICBwdWJsaWMgQ1R5cGUoKSB7DQotICAgIH0NCi0NCi0gICAgcHVibGljIENUeXBlKFN0cmluZyBiYXNlVHlwZSkgew0KLQlzZXRCYXNlVHlwZShiYXNlVHlwZSk7DQotICAgIH0NCi0NCi0gICAgcHVibGljIENUeXBlKFN0cmluZyBiYXNlVHlwZSwgYm9vbGVhbiBpc0NvbnN0LCBib29sZWFuIGlzUG9pbnRlcikgew0KLQlzZXRCYXNlVHlwZShiYXNlVHlwZSk7DQotCXNldElzQ29uc3QoaXNDb25zdCk7DQotCXNldElzUG9pbnRlcihpc1BvaW50ZXIpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0RGVjbGFyYXRpb24oKSB7DQotCXJldHVybiBiYXNlVHlwZSArIChpc1BvaW50ZXIgPyAiICoiIDogIiIpOw0KLSAgICB9DQotICAgIA0KLSAgICBwdWJsaWMgdm9pZCBzZXRJc0NvbnN0KGJvb2xlYW4gaXNDb25zdCkgew0KLQl0aGlzLmlzQ29uc3QgPSBpc0NvbnN0Ow0KLSAgICB9DQotDQotICAgIHB1YmxpYyBib29sZWFuIGlzQ29uc3QoKSB7DQotCXJldHVybiBpc0NvbnN0Ow0KLSAgICB9DQotDQotICAgIHB1YmxpYyB2b2lkIHNldElzUG9pbnRlcihib29sZWFuIGlzUG9pbnRlcikgew0KLQl0aGlzLmlzUG9pbnRlciA9IGlzUG9pbnRlcjsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1BvaW50ZXIoKSB7DQotCXJldHVybiBpc1BvaW50ZXI7DQotICAgIH0NCi0NCi0gICAgYm9vbGVhbiBpc1ZvaWQoKSB7DQotCVN0cmluZyBiYXNlVHlwZSA9IGdldEJhc2VUeXBlKCk7DQotCXJldHVybiBiYXNlVHlwZS5lcXVhbHMoIkdMdm9pZCIpIHx8DQotCSAgICBiYXNlVHlwZS5lcXVhbHMoInZvaWQiKTsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgYm9vbGVhbiBpc1R5cGVkUG9pbnRlcigpIHsNCi0JcmV0dXJuIGlzUG9pbnRlcigpICYmICFpc1ZvaWQoKTsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgdm9pZCBzZXRCYXNlVHlwZShTdHJpbmcgYmFzZVR5cGUpIHsNCi0JdGhpcy5iYXNlVHlwZSA9IGJhc2VUeXBlOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0QmFzZVR5cGUoKSB7DQotCXJldHVybiBiYXNlVHlwZTsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgU3RyaW5nIHRvU3RyaW5nKCkgew0KLQlTdHJpbmcgcyA9ICIiOw0KLQlpZiAoaXNDb25zdCgpKSB7DQotCSAgICBzICs9ICJjb25zdCAiOw0KLQl9DQotCXMgKz0gYmFzZVR5cGU7DQotCWlmIChpc1BvaW50ZXIoKSkgew0KLQkgICAgcyArPSAiKiI7DQotCX0NCi0NCi0JcmV0dXJuIHM7DQotICAgIH0NCi0NCi0gICAgcHVibGljIGludCBoYXNoQ29kZSgpIHsNCi0JcmV0dXJuIGJhc2VUeXBlLmhhc2hDb2RlKCkgXiAoaXNQb2ludGVyID8gMiA6IDApIF4gKGlzQ29uc3QgPyAxIDogMCk7DQotICAgIH0NCi0NCi0gICAgcHVibGljIGJvb2xlYW4gZXF1YWxzKE9iamVjdCBvKSB7DQotCWlmIChvICE9IG51bGwgJiYgbyBpbnN0YW5jZW9mIENUeXBlKSB7DQotCSAgICBDVHlwZSBjID0gKENUeXBlKW87DQotCSAgICByZXR1cm4gYmFzZVR5cGUuZXF1YWxzKGMuYmFzZVR5cGUpICYmDQotCQlpc1BvaW50ZXIoKSA9PSBjLmlzUG9pbnRlcigpICYmDQotCQlpc0NvbnN0KCkgPT0gYy5pc0NvbnN0KCk7DQotCX0NCi0JcmV0dXJuIGZhbHNlOw0KLSAgICB9DQotfQ0KZGlmZiAtLWdpdCBhL29wZW5nbC90b29scy9nbGdlbi9zcmMvQ29kZUVtaXR0ZXIuamF2YSBiL29wZW5nbC90b29scy9nbGdlbi9zcmMvQ29kZUVtaXR0ZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggM2U5YjkwYS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0NvZGVFbWl0dGVyLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSw4ICswLDAgQEAKLQ0KLXB1YmxpYyBpbnRlcmZhY2UgQ29kZUVtaXR0ZXIgew0KLQ0KLSAgICB2b2lkIHNldFZlcnNpb24oaW50IHZlcnNpb24sIGJvb2xlYW4gZXh0LCBib29sZWFuIHBhY2spOw0KLSAgICB2b2lkIGVtaXRDb2RlKENGdW5jIGNmdW5jLCBTdHJpbmcgb3JpZ2luYWwpOw0KLSAgICB2b2lkIGFkZE5hdGl2ZVJlZ2lzdHJhdGlvbihTdHJpbmcgZm5hbWUpOw0KLSAgICB2b2lkIGVtaXROYXRpdmVSZWdpc3RyYXRpb24oKTsNCi19DQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3NyYy9HZW5lcmF0ZUdMLmphdmEgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0dlbmVyYXRlR0wuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjU3ZWU2ZS4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0dlbmVyYXRlR0wuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDE2NCArMCwwIEBACi0NCi1pbXBvcnQgamF2YS5pby4qOw0KLWltcG9ydCBqYXZhLnV0aWwuKjsNCi0NCi1wdWJsaWMgY2xhc3MgR2VuZXJhdGVHTCB7DQotDQotICAgIHN0YXRpYyB2b2lkIGNvcHkoU3RyaW5nIGZpbGVuYW1lLCBQcmludFN0cmVhbSBvdXQpIHRocm93cyBJT0V4Y2VwdGlvbiB7DQotICAgICAgICBCdWZmZXJlZFJlYWRlciBiciA9IG5ldyBCdWZmZXJlZFJlYWRlcihuZXcgRmlsZVJlYWRlcihmaWxlbmFtZSkpOw0KLSAgICAgICAgU3RyaW5nIHM7DQotICAgICAgICB3aGlsZSAoKHMgPSBici5yZWFkTGluZSgpKSAhPSBudWxsKSB7DQotICAgICAgICAgICAgb3V0LnByaW50bG4ocyk7DQotICAgICAgICB9DQotICAgIH0NCi0NCi0gICAgcHJpdmF0ZSBzdGF0aWMgdm9pZCBlbWl0KGludCB2ZXJzaW9uLCBib29sZWFuIGV4dCwgYm9vbGVhbiBwYWNrLA0KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ29kZUVtaXR0ZXIgZW1pdHRlciwNCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIEJ1ZmZlcmVkUmVhZGVyIHNwZWNSZWFkZXIsDQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQcmludFN0cmVhbSBnbFN0cmVhbSwNCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIFByaW50U3RyZWFtIGdsSW1wbFN0cmVhbSwNCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIFByaW50U3RyZWFtIGNTdHJlYW0pIHRocm93cyBFeGNlcHRpb24gew0KLSAgICAgICAgU3RyaW5nIHMgPSBudWxsOw0KLSAgICAgICAgaW50IGNvdW50ZXIgPSAwOw0KLSAgICAgICAgd2hpbGUgKChzID0gc3BlY1JlYWRlci5yZWFkTGluZSgpKSAhPSBudWxsKSB7DQotICAgICAgICAgICAgaWYgKHMudHJpbSgpLnN0YXJ0c1dpdGgoIi8vIikpIHsNCi0gICAgICAgICAgICAgICAgY29udGludWU7DQotICAgICAgICAgICAgfQ0KLQ0KLSAgICAgICAgICAgIENGdW5jIGNmdW5jID0gQ0Z1bmMucGFyc2VDRnVuYyhzKTsNCi0NCi0gICAgICAgICAgICBTdHJpbmcgZm5hbWUgPSBjZnVuYy5nZXROYW1lKCk7DQotICAgICAgICAgICAgRmlsZSBmID0gbmV3IEZpbGUoInN0dWJzLyIgKyBmbmFtZSArDQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIi5qYXZhLTEiICsgdmVyc2lvbiArICItaWYiKTsNCi0gICAgICAgICAgICBpZiAoZi5leGlzdHMoKSkgew0KLSAgICAgICAgICAgICAgICBTeXN0ZW0ub3V0LnByaW50bG4oIlNwZWNpYWwtY2FzaW5nIGZ1bmN0aW9uICIgKyBmbmFtZSk7DQotICAgICAgICAgICAgICAgIGNvcHkoInN0dWJzLyIgKyBmbmFtZSArDQotICAgICAgICAgICAgICAgICAgICAgIi5qYXZhLTEiICsgdmVyc2lvbiArICItaWYiLCBnbFN0cmVhbSk7DQotICAgICAgICAgICAgICAgIGNvcHkoInN0dWJzLyIgKyBmbmFtZSArICIuamF2YS1pbXBsIiwgZ2xJbXBsU3RyZWFtKTsNCi0gICAgICAgICAgICAgICAgY29weSgic3R1YnMvIiArIGZuYW1lICsgIi5jcHAiLCBjU3RyZWFtKTsNCi0NCi0gICAgICAgICAgICAgICAgLy8gUmVnaXN0ZXIgbmF0aXZlIGZ1bmN0aW9uIG5hbWVzDQotICAgICAgICAgICAgICAgIC8vIFRoaXMgc2hvdWxkIGJlIGltcHJvdmVkIHRvIHJlcXVpcmUgZmV3ZXIgZGlzY3JldGUgZmlsZXMNCi0gICAgICAgICAgICAgICAgU3RyaW5nIGZpbGVuYW1lID0gInN0dWJzLyIgKyBmbmFtZSArICIubmF0aXZlUmVnIjsNCi0gICAgICAgICAgICAgICAgQnVmZmVyZWRSZWFkZXIgYnIgPQ0KLSAgICAgICAgICAgICAgICAgICAgbmV3IEJ1ZmZlcmVkUmVhZGVyKG5ldyBGaWxlUmVhZGVyKGZpbGVuYW1lKSk7DQotICAgICAgICAgICAgICAgIFN0cmluZyBuZnVuYzsNCi0gICAgICAgICAgICAgICAgd2hpbGUgKChuZnVuYyA9IGJyLnJlYWRMaW5lKCkpICE9IG51bGwpIHsNCi0gICAgICAgICAgICAgICAgICAgIGVtaXR0ZXIuYWRkTmF0aXZlUmVnaXN0cmF0aW9uKG5mdW5jKTsNCi0gICAgICAgICAgICAgICAgfQ0KLSAgICAgICAgICAgIH0gZWxzZSB7DQotICAgICAgICAgICAgICAgIGVtaXR0ZXIuc2V0VmVyc2lvbih2ZXJzaW9uLCBleHQsIHBhY2spOw0KLSAgICAgICAgICAgICAgICBlbWl0dGVyLmVtaXRDb2RlKGNmdW5jLCBzKTsNCi0gICAgICAgICAgICB9DQotICAgICAgICB9DQotICAgIH0NCi0NCi0gICAgcHVibGljIHN0YXRpYyB2b2lkIG1haW4oU3RyaW5nW10gYXJncykgdGhyb3dzIEV4Y2VwdGlvbiB7DQotICAgICAgICBTdHJpbmcgY2xhc3NQYXRoTmFtZSA9ICJjb20vZ29vZ2xlL2FuZHJvaWQvZ2xlc19qbmkvR0xJbXBsIjsNCi0gICAgICAgIGJvb2xlYW4gdXNlQ29udGV4dFBvaW50ZXIgPSB0cnVlOw0KLQ0KLSAgICAgICAgaW50IGFpZHggPSAwOw0KLSAgICAgICAgd2hpbGUgKGFyZ3NbYWlkeF0uY2hhckF0KDApID09ICctJykgew0KLSAgICAgICAgICAgIHN3aXRjaCAoYXJnc1thaWR4XS5jaGFyQXQoMSkpIHsNCi0gICAgICAgICAgICBjYXNlICdjJzoNCi0gICAgICAgICAgICAgICAgdXNlQ29udGV4dFBvaW50ZXIgPSBmYWxzZTsNCi0gICAgICAgICAgICAgICAgYnJlYWs7DQotDQotICAgICAgICAgICAgZGVmYXVsdDoNCi0gICAgICAgICAgICAgICAgU3lzdGVtLmVyci5wcmludGxuKCJVbmtub3duIGZsYWc6ICIgKyBhcmdzW2FpZHhdKTsNCi0gICAgICAgICAgICAgICAgU3lzdGVtLmV4aXQoMSk7DQotICAgICAgICAgICAgfQ0KLQ0KLSAgICAgICAgICAgIGFpZHgrKzsNCi0gICAgICAgIH0NCi0NCi0gICAgICAgIFN5c3RlbS5vdXQucHJpbnRsbigidXNlQ29udGV4dFBvaW50ZXIgPSAiICsgdXNlQ29udGV4dFBvaW50ZXIpOw0KLQ0KLSAgICAgICAgQnVmZmVyZWRSZWFkZXIgc3BlYzEwUmVhZGVyID0NCi0gICAgICAgICAgICBuZXcgQnVmZmVyZWRSZWFkZXIobmV3IEZpbGVSZWFkZXIoYXJnc1thaWR4KytdKSk7DQotICAgICAgICBCdWZmZXJlZFJlYWRlciBzcGVjMTBFeHRSZWFkZXIgPQ0KLSAgICAgICAgICAgIG5ldyBCdWZmZXJlZFJlYWRlcihuZXcgRmlsZVJlYWRlcihhcmdzW2FpZHgrK10pKTsNCi0gICAgICAgIEJ1ZmZlcmVkUmVhZGVyIHNwZWMxMVJlYWRlciA9DQotICAgICAgICAgICAgbmV3IEJ1ZmZlcmVkUmVhZGVyKG5ldyBGaWxlUmVhZGVyKGFyZ3NbYWlkeCsrXSkpOw0KLSAgICAgICAgQnVmZmVyZWRSZWFkZXIgc3BlYzExRXh0UmVhZGVyID0NCi0gICAgICAgICAgICBuZXcgQnVmZmVyZWRSZWFkZXIobmV3IEZpbGVSZWFkZXIoYXJnc1thaWR4KytdKSk7DQotICAgICAgICBCdWZmZXJlZFJlYWRlciBzcGVjMTFFeHRQYWNrUmVhZGVyID0NCi0gICAgICAgICAgICBuZXcgQnVmZmVyZWRSZWFkZXIobmV3IEZpbGVSZWFkZXIoYXJnc1thaWR4KytdKSk7DQotICAgICAgICBCdWZmZXJlZFJlYWRlciBjaGVja3NSZWFkZXIgPQ0KLSAgICAgICAgICAgIG5ldyBCdWZmZXJlZFJlYWRlcihuZXcgRmlsZVJlYWRlcihhcmdzW2FpZHgrK10pKTsNCi0NCi0gICAgICAgIFN0cmluZyBnbDEwRmlsZW5hbWUgPSAiamF2YXgvbWljcm9lZGl0aW9uL2tocm9ub3Mvb3BlbmdsZXMvR0wxMC5qYXZhIjsNCi0gICAgICAgIFN0cmluZyBnbDEwRXh0RmlsZW5hbWUgPQ0KLSAgICAgICAgICAgICJqYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDEwRXh0LmphdmEiOw0KLSAgICAgICAgU3RyaW5nIGdsMTFGaWxlbmFtZSA9ICJqYXZheC9taWNyb2VkaXRpb24va2hyb25vcy9vcGVuZ2xlcy9HTDExLmphdmEiOw0KLSAgICAgICAgU3RyaW5nIGdsMTFFeHRGaWxlbmFtZSA9DQotICAgICAgICAgICAgImphdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTFFeHQuamF2YSI7DQotICAgICAgICBTdHJpbmcgZ2wxMUV4dFBhY2tGaWxlbmFtZSA9DQotICAgICAgICAgICAgImphdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMMTFFeHRlbnNpb25QYWNrLmphdmEiOw0KLSAgICAgICAgU3RyaW5nIGdsSW1wbEZpbGVuYW1lID0gImNvbS9nb29nbGUvYW5kcm9pZC9nbGVzX2puaS9HTEltcGwuamF2YSI7DQotICAgICAgICBTdHJpbmcgY0ZpbGVuYW1lID0gImNvbV9nb29nbGVfYW5kcm9pZF9nbGVzX2puaV9HTEltcGwuY3BwIjsNCi0NCi0gICAgICAgIFByaW50U3RyZWFtIGdsMTBTdHJlYW0gPQ0KLSAgICAgICAgICAgIG5ldyBQcmludFN0cmVhbShuZXcgRmlsZU91dHB1dFN0cmVhbSgib3V0LyIgKyBnbDEwRmlsZW5hbWUpKTsNCi0gICAgICAgIFByaW50U3RyZWFtIGdsMTBFeHRTdHJlYW0gPQ0KLSAgICAgICAgICAgIG5ldyBQcmludFN0cmVhbShuZXcgRmlsZU91dHB1dFN0cmVhbSgib3V0LyIgKyBnbDEwRXh0RmlsZW5hbWUpKTsNCi0gICAgICAgIFByaW50U3RyZWFtIGdsMTFTdHJlYW0gPQ0KLSAgICAgICAgICAgIG5ldyBQcmludFN0cmVhbShuZXcgRmlsZU91dHB1dFN0cmVhbSgib3V0LyIgKyBnbDExRmlsZW5hbWUpKTsNCi0gICAgICAgIFByaW50U3RyZWFtIGdsMTFFeHRTdHJlYW0gPQ0KLSAgICAgICAgICAgIG5ldyBQcmludFN0cmVhbShuZXcgRmlsZU91dHB1dFN0cmVhbSgib3V0LyIgKyBnbDExRXh0RmlsZW5hbWUpKTsNCi0gICAgICAgIFByaW50U3RyZWFtIGdsMTFFeHRQYWNrU3RyZWFtID0NCi0gICAgICAgICAgICBuZXcgUHJpbnRTdHJlYW0obmV3IEZpbGVPdXRwdXRTdHJlYW0oIm91dC8iICsgZ2wxMUV4dFBhY2tGaWxlbmFtZSkpOw0KLSAgICAgICAgUHJpbnRTdHJlYW0gZ2xJbXBsU3RyZWFtID0NCi0gICAgICAgICAgICBuZXcgUHJpbnRTdHJlYW0obmV3IEZpbGVPdXRwdXRTdHJlYW0oIm91dC8iICsgZ2xJbXBsRmlsZW5hbWUpKTsNCi0gICAgICAgIFByaW50U3RyZWFtIGNTdHJlYW0gPQ0KLSAgICAgICAgICAgIG5ldyBQcmludFN0cmVhbShuZXcgRmlsZU91dHB1dFN0cmVhbSgib3V0LyIgKyBjRmlsZW5hbWUpKTsNCi0NCi0gICAgICAgIFBhcmFtZXRlckNoZWNrZXIgY2hlY2tlciA9IG5ldyBQYXJhbWV0ZXJDaGVja2VyKGNoZWNrc1JlYWRlcik7DQotDQotICAgICAgICBDb2RlRW1pdHRlciBlbWl0dGVyID0NCi0gICAgICAgICAgICBuZXcgSm5pQ29kZUVtaXR0ZXIoY2xhc3NQYXRoTmFtZSwNCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY2hlY2tlciwNCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2wxMFN0cmVhbSwgZ2wxMEV4dFN0cmVhbSwNCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2wxMVN0cmVhbSwgZ2wxMUV4dFN0cmVhbSwgZ2wxMUV4dFBhY2tTdHJlYW0sDQotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdsSW1wbFN0cmVhbSwgY1N0cmVhbSwNCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdXNlQ29udGV4dFBvaW50ZXIpOw0KLQ0KLSAgICAgICAgZ2wxMFN0cmVhbS5wcmludGxuKCIvKiAvL2RldmljZS9qYXZhL2FuZHJvaWQvIiArIGdsMTBGaWxlbmFtZSk7DQotICAgICAgICBnbDEwRXh0U3RyZWFtLnByaW50bG4oIi8qIC8vZGV2aWNlL2phdmEvYW5kcm9pZC8iICsgZ2wxMEV4dEZpbGVuYW1lKTsNCi0gICAgICAgIGdsMTFTdHJlYW0ucHJpbnRsbigiLyogLy9kZXZpY2UvamF2YS9hbmRyb2lkLyIgKyBnbDExRmlsZW5hbWUpOw0KLSAgICAgICAgZ2wxMUV4dFN0cmVhbS5wcmludGxuKCIvKiAvL2RldmljZS9qYXZhL2FuZHJvaWQvIiArIGdsMTFFeHRGaWxlbmFtZSk7DQotICAgICAgICBnbDExRXh0UGFja1N0cmVhbS5wcmludGxuKCIvKiAvL2RldmljZS9qYXZhL2FuZHJvaWQvIiArDQotICAgICAgICAgICAgZ2wxMUV4dFBhY2tGaWxlbmFtZSk7DQotICAgICAgICBnbEltcGxTdHJlYW0ucHJpbnRsbigiLyogLy9kZXZpY2UvamF2YS9hbmRyb2lkLyIgKyBnbEltcGxGaWxlbmFtZSk7DQotICAgICAgICBjU3RyZWFtLnByaW50bG4oIi8qIC8vZGV2aWNlL2xpYnMvYW5kcm9pZF9ydW50aW1lLyIgKyBjRmlsZW5hbWUpOw0KLQ0KLSAgICAgICAgY29weSgic3R1YnMvR0wxMEhlYWRlci5qYXZhLWlmIiwgZ2wxMFN0cmVhbSk7DQotICAgICAgICBjb3B5KCJzdHVicy9HTDEwRXh0SGVhZGVyLmphdmEtaWYiLCBnbDEwRXh0U3RyZWFtKTsNCi0gICAgICAgIGNvcHkoInN0dWJzL0dMMTFIZWFkZXIuamF2YS1pZiIsIGdsMTFTdHJlYW0pOw0KLSAgICAgICAgY29weSgic3R1YnMvR0wxMUV4dEhlYWRlci5qYXZhLWlmIiwgZ2wxMUV4dFN0cmVhbSk7DQotICAgICAgICBjb3B5KCJzdHVicy9HTDExRXh0ZW5zaW9uUGFja0hlYWRlci5qYXZhLWlmIiwgZ2wxMUV4dFBhY2tTdHJlYW0pOw0KLSAgICAgICAgY29weSgic3R1YnMvR0xJbXBsSGVhZGVyLmphdmEtaW1wbCIsIGdsSW1wbFN0cmVhbSk7DQotICAgICAgICBjb3B5KCJzdHVicy9HTENIZWFkZXIuY3BwIiwgY1N0cmVhbSk7DQotDQotICAgICAgICBlbWl0KDAsIGZhbHNlLCBmYWxzZSwNCi0gICAgICAgICAgICAgZW1pdHRlciwgc3BlYzEwUmVhZGVyLCBnbDEwU3RyZWFtLCBnbEltcGxTdHJlYW0sIGNTdHJlYW0pOw0KLSAgICAgICAgZW1pdCgwLCB0cnVlLCBmYWxzZSwNCi0gICAgICAgICAgICAgZW1pdHRlciwgc3BlYzEwRXh0UmVhZGVyLCBnbDEwRXh0U3RyZWFtLCBnbEltcGxTdHJlYW0sIGNTdHJlYW0pOw0KLSAgICAgICAgZW1pdCgxLCBmYWxzZSwgZmFsc2UsDQotICAgICAgICAgICAgIGVtaXR0ZXIsIHNwZWMxMVJlYWRlciwgZ2wxMVN0cmVhbSwgZ2xJbXBsU3RyZWFtLCBjU3RyZWFtKTsNCi0gICAgICAgIGVtaXQoMSwgdHJ1ZSwgZmFsc2UsDQotICAgICAgICAgICAgIGVtaXR0ZXIsIHNwZWMxMUV4dFJlYWRlciwgZ2wxMUV4dFN0cmVhbSwgZ2xJbXBsU3RyZWFtLCBjU3RyZWFtKTsNCi0gICAgICAgIGVtaXQoMSwgdHJ1ZSwgdHJ1ZSwNCi0gICAgICAgICAgICAgZW1pdHRlciwgc3BlYzExRXh0UGFja1JlYWRlciwgZ2wxMUV4dFBhY2tTdHJlYW0sIGdsSW1wbFN0cmVhbSwNCi0gICAgICAgICAgICAgY1N0cmVhbSk7DQotDQotICAgICAgICBlbWl0dGVyLmVtaXROYXRpdmVSZWdpc3RyYXRpb24oKTsNCi0NCi0gICAgICAgIGdsMTBTdHJlYW0ucHJpbnRsbigifSIpOw0KLSAgICAgICAgZ2wxMEV4dFN0cmVhbS5wcmludGxuKCJ9Iik7DQotICAgICAgICBnbDExU3RyZWFtLnByaW50bG4oIn0iKTsNCi0gICAgICAgIGdsMTFFeHRTdHJlYW0ucHJpbnRsbigifSIpOw0KLSAgICAgICAgZ2wxMUV4dFBhY2tTdHJlYW0ucHJpbnRsbigifSIpOw0KLSAgICAgICAgZ2xJbXBsU3RyZWFtLnByaW50bG4oIn0iKTsNCi0gICAgfQ0KLX0NCmRpZmYgLS1naXQgYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0pGdW5jLmphdmEgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0pGdW5jLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDQyZDQ2NmMuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3NyYy9KRnVuYy5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTQ4ICswLDAgQEAKLQ0KLWltcG9ydCBqYXZhLnV0aWwuQXJyYXlMaXN0Ow0KLWltcG9ydCBqYXZhLnV0aWwuTGlzdDsNCi0NCi1wdWJsaWMgY2xhc3MgSkZ1bmMgew0KLQ0KLSAgICBTdHJpbmcgY2xhc3NOYW1lID0gImNvbS5nb29nbGUuYW5kcm9pZC5nbGVzX2puaS5HTDExSW1wbCI7DQotDQotICAgIENGdW5jIGNmdW5jOw0KLSAgICBKVHlwZSBmdHlwZTsNCi0gICAgU3RyaW5nIGZuYW1lOw0KLQ0KLSAgICBMaXN0PFN0cmluZz4gYXJnTmFtZXMgPSBuZXcgQXJyYXlMaXN0PFN0cmluZz4oKTsNCi0gICAgTGlzdDxKVHlwZT4gYXJnVHlwZXMgPSBuZXcgQXJyYXlMaXN0PEpUeXBlPigpOw0KLSAgICBMaXN0PEludGVnZXI+IGFyZ0NJbmRpY2VzID0gbmV3IEFycmF5TGlzdDxJbnRlZ2VyPigpOw0KLQ0KLSAgICBib29sZWFuIGhhc0J1ZmZlckFyZyA9IGZhbHNlOw0KLSAgICBib29sZWFuIGhhc1R5cGVkQnVmZmVyQXJnID0gZmFsc2U7DQotICAgIEFycmF5TGlzdDxTdHJpbmc+IGJ1ZmZlckFyZ05hbWVzID0gbmV3IEFycmF5TGlzdDxTdHJpbmc+KCk7DQotDQotICAgIHB1YmxpYyBKRnVuYyhDRnVuYyBjZnVuYykgew0KLSAgICAgICAgdGhpcy5jZnVuYyA9IGNmdW5jOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBDRnVuYyBnZXRDRnVuYygpIHsNCi0gICAgICAgIHJldHVybiBjZnVuYzsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgdm9pZCBzZXROYW1lKFN0cmluZyBmbmFtZSkgew0KLSAgICAgICAgdGhpcy5mbmFtZSA9IGZuYW1lOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0TmFtZSgpIHsNCi0gICAgICAgIHJldHVybiBmbmFtZTsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgdm9pZCBzZXRUeXBlKEpUeXBlIGZ0eXBlKSB7DQotICAgICAgICB0aGlzLmZ0eXBlID0gZnR5cGU7DQotICAgIH0NCi0NCi0gICAgcHVibGljIEpUeXBlIGdldFR5cGUoKSB7DQotICAgICAgICByZXR1cm4gZnR5cGU7DQotICAgIH0NCi0NCi0gICAgcHVibGljIHZvaWQgc2V0Q2xhc3NOYW1lKFN0cmluZyBjbGFzc05hbWUpIHsNCi0gICAgICAgIHRoaXMuY2xhc3NOYW1lID0gY2xhc3NOYW1lOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0Q2xhc3NOYW1lKCkgew0KLSAgICAgICAgcmV0dXJuIGNsYXNzTmFtZTsNCi0gICAgfQ0KLSAgICANCi0gICAgcHVibGljIGJvb2xlYW4gaGFzQnVmZmVyQXJnKCkgew0KLSAgICAgICAgcmV0dXJuIGhhc0J1ZmZlckFyZzsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgYm9vbGVhbiBoYXNUeXBlZEJ1ZmZlckFyZygpIHsNCi0gICAgICAgIHJldHVybiBoYXNUeXBlZEJ1ZmZlckFyZzsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgU3RyaW5nIGdldEJ1ZmZlckFyZ05hbWUoaW50IGluZGV4KSB7DQotICAgICAgICByZXR1cm4gYnVmZmVyQXJnTmFtZXMuZ2V0KGluZGV4KTsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgdm9pZCBhZGRBcmd1bWVudChTdHJpbmcgYXJnTmFtZSwgSlR5cGUgYXJnVHlwZSwgaW50IGNpbmRleCkgew0KLSAgICAgICAgYXJnTmFtZXMuYWRkKGFyZ05hbWUpOw0KLSAgICAgICAgYXJnVHlwZXMuYWRkKGFyZ1R5cGUpOw0KLSAgICAgICAgYXJnQ0luZGljZXMuYWRkKG5ldyBJbnRlZ2VyKGNpbmRleCkpOw0KLQ0KLSAgICAgICAgaWYgKGFyZ1R5cGUuaXNCdWZmZXIoKSkgew0KLSAgICAgICAgICAgIGhhc0J1ZmZlckFyZyA9IHRydWU7DQotICAgICAgICAgICAgYnVmZmVyQXJnTmFtZXMuYWRkKGFyZ05hbWUpOw0KLSAgICAgICAgfQ0KLSAgICAgICAgaWYgKGFyZ1R5cGUuaXNUeXBlZEJ1ZmZlcigpKSB7DQotICAgICAgICAgICAgaGFzVHlwZWRCdWZmZXJBcmcgPSB0cnVlOw0KLSAgICAgICAgICAgIGJ1ZmZlckFyZ05hbWVzLmFkZChhcmdOYW1lKTsNCi0gICAgICAgIH0NCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgaW50IGdldE51bUFyZ3MoKSB7DQotICAgICAgICByZXR1cm4gYXJnTmFtZXMuc2l6ZSgpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBpbnQgZ2V0QXJnSW5kZXgoU3RyaW5nIG5hbWUpIHsNCi0gICAgICAgIGludCBsZW4gPSBhcmdOYW1lcy5zaXplKCk7DQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IGxlbjsgaSsrKSB7DQotICAgICAgICAgICAgaWYgKG5hbWUuZXF1YWxzKGFyZ05hbWVzLmdldChpKSkpIHsNCi0gICAgICAgICAgICAgICAgcmV0dXJuIGk7DQotICAgICAgICAgICAgfQ0KLSAgICAgICAgfQ0KLSAgICAgICAgcmV0dXJuIC0xOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBTdHJpbmcgZ2V0QXJnTmFtZShpbnQgaW5kZXgpIHsNCi0gICAgICAgIHJldHVybiBhcmdOYW1lcy5nZXQoaW5kZXgpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBKVHlwZSBnZXRBcmdUeXBlKGludCBpbmRleCkgew0KLSAgICAgICAgcmV0dXJuIGFyZ1R5cGVzLmdldChpbmRleCk7DQotICAgIH0NCi0NCi0gICAgcHVibGljIGludCBnZXRBcmdDSW5kZXgoaW50IGluZGV4KSB7DQotICAgICAgICByZXR1cm4gYXJnQ0luZGljZXMuZ2V0KGluZGV4KS5pbnRWYWx1ZSgpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBzdGF0aWMgSkZ1bmMgY29udmVydChDRnVuYyBjZnVuYywgYm9vbGVhbiB1c2VBcnJheSkgew0KLSAgICAgICAgSkZ1bmMgamZ1bmMgPSBuZXcgSkZ1bmMoY2Z1bmMpOw0KLSAgICAgICAgamZ1bmMuc2V0TmFtZShjZnVuYy5nZXROYW1lKCkpOw0KLSAgICAgICAgamZ1bmMuc2V0VHlwZShKVHlwZS5jb252ZXJ0KGNmdW5jLmdldFR5cGUoKSwgZmFsc2UpKTsNCi0JDQotICAgICAgICBpbnQgbnVtQXJncyA9IGNmdW5jLmdldE51bUFyZ3MoKTsNCi0gICAgICAgIGludCBudW1PZmZzZXRzID0gMDsNCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQXJnczsgaSsrKSB7DQotICAgICAgICAgICAgQ1R5cGUgY0FyZ1R5cGUgPSBjZnVuYy5nZXRBcmdUeXBlKGkpOw0KLSAgICAgICAgICAgIGlmIChjQXJnVHlwZS5pc1R5cGVkUG9pbnRlcigpICYmIHVzZUFycmF5KSB7DQotICAgICAgICAgICAgICAgICsrbnVtT2Zmc2V0czsNCi0gICAgICAgICAgICB9DQotICAgICAgICB9DQotDQotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUFyZ3M7IGkrKykgew0KLSAgICAgICAgICAgIFN0cmluZyBjQXJnTmFtZSA9IGNmdW5jLmdldEFyZ05hbWUoaSk7DQotICAgICAgICAgICAgQ1R5cGUgY0FyZ1R5cGUgPSBjZnVuYy5nZXRBcmdUeXBlKGkpOw0KLQ0KLSAgICAgICAgICAgIGpmdW5jLmFkZEFyZ3VtZW50KGNBcmdOYW1lLCBKVHlwZS5jb252ZXJ0KGNBcmdUeXBlLCB1c2VBcnJheSksIGkpOw0KLSAgICAgICAgICAgIGlmIChjQXJnVHlwZS5pc1R5cGVkUG9pbnRlcigpICYmIHVzZUFycmF5KSB7DQotICAgICAgICAgICAgICAgIGlmIChudW1PZmZzZXRzID4gMSkgew0KLSAgICAgICAgICAgICAgICAgICAgamZ1bmMuYWRkQXJndW1lbnQoY0FyZ05hbWUgKyAiT2Zmc2V0IiwgbmV3IEpUeXBlKCJpbnQiKSwgaSk7DQotICAgICAgICAgICAgICAgIH0gZWxzZSB7DQotICAgICAgICAgICAgICAgICAgICBqZnVuYy5hZGRBcmd1bWVudCgib2Zmc2V0IiwgbmV3IEpUeXBlKCJpbnQiKSwgaSk7DQotICAgICAgICAgICAgICAgIH0NCi0gICAgICAgICAgICB9DQotICAgICAgICB9DQotDQotICAgICAgICByZXR1cm4gamZ1bmM7DQotICAgIH0NCi0NCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsNCi0gICAgICAgIFN0cmluZyBzID0gICJGdW5jdGlvbiAiICsgZm5hbWUgKyAiIHJldHVybnMgIiArIGZ0eXBlICsgIjogIjsNCi0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgYXJnTmFtZXMuc2l6ZSgpOyBpKyspIHsNCi0gICAgICAgICAgICBpZiAoaSA+IDApIHsNCi0gICAgICAgICAgICAgICAgcyArPSAiLCAiOw0KLSAgICAgICAgICAgIH0NCi0gICAgICAgICAgICBzICs9IGFyZ1R5cGVzLmdldChpKSArICIgIiArIGFyZ05hbWVzLmdldChpKTsNCi0gICAgICAgIH0NCi0gICAgICAgIHJldHVybiBzOw0KLSAgICB9DQotDQotfQ0KZGlmZiAtLWdpdCBhL29wZW5nbC90b29scy9nbGdlbi9zcmMvSlR5cGUuamF2YSBiL29wZW5nbC90b29scy9nbGdlbi9zcmMvSlR5cGUuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggYTE2ZDQ0MC4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0pUeXBlLmphdmEKKysrIC9kZXYvbnVsbApAQCAtMSwxMzkgKzAsMCBAQAotDQotaW1wb3J0IGphdmEudXRpbC5IYXNoTWFwOw0KLQ0KLXB1YmxpYyBjbGFzcyBKVHlwZSB7DQotICAgIA0KLSAgICBTdHJpbmcgYmFzZVR5cGU7DQotICAgIGJvb2xlYW4gaXNBcnJheTsNCi0gICAgYm9vbGVhbiBpc0NsYXNzOw0KLQ0KLSAgICBzdGF0aWMgSGFzaE1hcDxDVHlwZSxKVHlwZT4gdHlwZU1hcHBpbmcgPSBuZXcgSGFzaE1hcDxDVHlwZSxKVHlwZT4oKTsNCi0gICAgc3RhdGljIEhhc2hNYXA8Q1R5cGUsSlR5cGU+IGFycmF5VHlwZU1hcHBpbmcgPSBuZXcgSGFzaE1hcDxDVHlwZSxKVHlwZT4oKTsNCi0NCi0gICAgc3RhdGljIHsNCi0JLy8gUHJpbWl0aXZlIHR5cGVzDQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMYml0ZmllbGQiKSwgbmV3IEpUeXBlKCJpbnQiKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMYm9vbGVhbiIpLCBuZXcgSlR5cGUoImJvb2xlYW4iKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMY2xhbXBmIiksIG5ldyBKVHlwZSgiZmxvYXQiKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMY2xhbXB4IiksIG5ldyBKVHlwZSgiaW50IikpOw0KLQl0eXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTGVudW0iKSwgbmV3IEpUeXBlKCJpbnQiKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMZmxvYXQiKSwgbmV3IEpUeXBlKCJmbG9hdCIpKTsNCi0JdHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0xmaXhlZCIpLCBuZXcgSlR5cGUoImludCIpKTsNCi0JdHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0xpbnQiKSwgbmV3IEpUeXBlKCJpbnQiKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMaW50cHRyIiksIG5ldyBKVHlwZSgiaW50IikpOw0KLQl0eXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTHNob3J0IiksIG5ldyBKVHlwZSgic2hvcnQiKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMc2l6ZWkiKSwgbmV3IEpUeXBlKCJpbnQiKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMc2l6ZWlwdHIiKSwgbmV3IEpUeXBlKCJpbnQiKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMdWJ5dGUiKSwgbmV3IEpUeXBlKCJieXRlIikpOw0KLQl0eXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTHVpbnQiKSwgbmV3IEpUeXBlKCJpbnQiKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoInZvaWQiKSwgbmV3IEpUeXBlKCJ2b2lkIikpOw0KLQl0eXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTHVieXRlIiwgdHJ1ZSwgdHJ1ZSksIG5ldyBKVHlwZSgiU3RyaW5nIikpOw0KLQ0KLQkvLyBVbnR5cGVkIHBvaW50ZXJzIG1hcCB0byB1bnR5cGVkIEJ1ZmZlcnMNCi0JdHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0x2b2lkIiwgdHJ1ZSwgdHJ1ZSksDQotCQkJbmV3IEpUeXBlKCJqYXZhLm5pby5CdWZmZXIiLCB0cnVlLCBmYWxzZSkpOw0KLQl0eXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTHZvaWQiLCBmYWxzZSwgdHJ1ZSksDQotCQkJbmV3IEpUeXBlKCJqYXZhLm5pby5CdWZmZXIiLCB0cnVlLCBmYWxzZSkpOw0KLQl0eXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJ2b2lkIiwgZmFsc2UsIHRydWUpLA0KLQkJCW5ldyBKVHlwZSgiamF2YS5uaW8uQnVmZmVyIiwgdHJ1ZSwgZmFsc2UpKTsNCi0NCi0JLy8gVHlwZWQgcG9pbnRlcnMgbWFwIHRvIHR5cGVkIEJ1ZmZlcnMNCi0JdHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0xib29sZWFuIiwgZmFsc2UsIHRydWUpLA0KLQkJCW5ldyBKVHlwZSgiamF2YS5uaW8uSW50QnVmZmVyIiwgdHJ1ZSwgZmFsc2UpKTsNCi0JdHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0xmaXhlZCIsIGZhbHNlLCB0cnVlKSwNCi0JCQluZXcgSlR5cGUoImphdmEubmlvLkludEJ1ZmZlciIsIHRydWUsIGZhbHNlKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMZml4ZWQiLCB0cnVlLCB0cnVlKSwNCi0JCQluZXcgSlR5cGUoImphdmEubmlvLkludEJ1ZmZlciIsIHRydWUsIGZhbHNlKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMZmxvYXQiLCBmYWxzZSwgdHJ1ZSksDQotCQkJbmV3IEpUeXBlKCJqYXZhLm5pby5GbG9hdEJ1ZmZlciIsIHRydWUsIGZhbHNlKSk7DQotCXR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMZmxvYXQiLCB0cnVlLCB0cnVlKSwNCi0JCQluZXcgSlR5cGUoImphdmEubmlvLkZsb2F0QnVmZmVyIiwgdHJ1ZSwgZmFsc2UpKTsNCi0JdHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0xpbnQiLCBmYWxzZSwgdHJ1ZSksDQotCQkJbmV3IEpUeXBlKCJqYXZhLm5pby5JbnRCdWZmZXIiLCB0cnVlLCBmYWxzZSkpOw0KLQl0eXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTGludCIsIHRydWUsIHRydWUpLA0KLQkJCW5ldyBKVHlwZSgiamF2YS5uaW8uSW50QnVmZmVyIiwgdHJ1ZSwgZmFsc2UpKTsNCi0JdHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0x1aW50IiwgZmFsc2UsIHRydWUpLA0KLQkJCW5ldyBKVHlwZSgiamF2YS5uaW8uSW50QnVmZmVyIiwgdHJ1ZSwgZmFsc2UpKTsNCi0JdHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0x1aW50IiwgdHJ1ZSwgdHJ1ZSksDQotCQkJbmV3IEpUeXBlKCJqYXZhLm5pby5JbnRCdWZmZXIiLCB0cnVlLCBmYWxzZSkpOw0KLQl0eXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTHNob3J0IiwgdHJ1ZSwgdHJ1ZSksDQotCQkJbmV3IEpUeXBlKCJqYXZhLm5pby5TaG9ydEJ1ZmZlciIsIHRydWUsIGZhbHNlKSk7DQotDQotCS8vIFR5cGVkIHBvaW50ZXJzIG1hcCB0byBhcnJheXMgKyBvZmZzZXRzDQotCWFycmF5VHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0xib29sZWFuIiwgZmFsc2UsIHRydWUpLA0KLQkJCSAgICAgbmV3IEpUeXBlKCJib29sZWFuIiwgZmFsc2UsIHRydWUpKTsNCi0JYXJyYXlUeXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTGZpeGVkIiwgdHJ1ZSwgdHJ1ZSksIG5ldyBKVHlwZSgiaW50IiwgZmFsc2UsIHRydWUpKTsNCi0JYXJyYXlUeXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTGZpeGVkIiwgZmFsc2UsIHRydWUpLCBuZXcgSlR5cGUoImludCIsIGZhbHNlLCB0cnVlKSk7DQotCWFycmF5VHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0xmbG9hdCIsIGZhbHNlLCB0cnVlKSwgbmV3IEpUeXBlKCJmbG9hdCIsIGZhbHNlLCB0cnVlKSk7DQotCWFycmF5VHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0xmbG9hdCIsIHRydWUsIHRydWUpLCBuZXcgSlR5cGUoImZsb2F0IiwgZmFsc2UsIHRydWUpKTsNCi0JYXJyYXlUeXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTGludCIsIGZhbHNlLCB0cnVlKSwgbmV3IEpUeXBlKCJpbnQiLCBmYWxzZSwgdHJ1ZSkpOw0KLQlhcnJheVR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMaW50IiwgdHJ1ZSwgdHJ1ZSksIG5ldyBKVHlwZSgiaW50IiwgZmFsc2UsIHRydWUpKTsNCi0JYXJyYXlUeXBlTWFwcGluZy5wdXQobmV3IENUeXBlKCJHTHNob3J0IiwgdHJ1ZSwgdHJ1ZSksIG5ldyBKVHlwZSgic2hvcnQiLCBmYWxzZSwgdHJ1ZSkpOw0KLQlhcnJheVR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMdWludCIsIGZhbHNlLCB0cnVlKSwgbmV3IEpUeXBlKCJpbnQiLCBmYWxzZSwgdHJ1ZSkpOw0KLQlhcnJheVR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMdWludCIsIHRydWUsIHRydWUpLCBuZXcgSlR5cGUoImludCIsIGZhbHNlLCB0cnVlKSk7DQotCWFycmF5VHlwZU1hcHBpbmcucHV0KG5ldyBDVHlwZSgiR0xpbnRwdHIiKSwgbmV3IEpUeXBlKCJpbnQiLCBmYWxzZSwgdHJ1ZSkpOw0KLQlhcnJheVR5cGVNYXBwaW5nLnB1dChuZXcgQ1R5cGUoIkdMc2l6ZWlwdHIiKSwgbmV3IEpUeXBlKCJpbnQiLCBmYWxzZSwgdHJ1ZSkpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBKVHlwZSgpIHsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgSlR5cGUoU3RyaW5nIHByaW1pdGl2ZVR5cGVOYW1lKSB7DQotCXRoaXMuYmFzZVR5cGUgPSBwcmltaXRpdmVUeXBlTmFtZTsNCi0JdGhpcy5pc0NsYXNzID0gZmFsc2U7DQotCXRoaXMuaXNBcnJheSA9IGZhbHNlOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBKVHlwZShTdHJpbmcgcHJpbWl0aXZlVHlwZU5hbWUsIGJvb2xlYW4gaXNDbGFzcywgYm9vbGVhbiBpc0FycmF5KSB7DQotCXRoaXMuYmFzZVR5cGUgPSBwcmltaXRpdmVUeXBlTmFtZTsNCi0JdGhpcy5pc0NsYXNzID0gaXNDbGFzczsNCi0JdGhpcy5pc0FycmF5ID0gaXNBcnJheTsNCi0gICAgfQ0KLQ0KLSAgICBwdWJsaWMgU3RyaW5nIGdldEJhc2VUeXBlKCkgew0KLQlyZXR1cm4gYmFzZVR5cGU7DQotICAgIH0NCi0NCi0gICAgcHVibGljIFN0cmluZyB0b1N0cmluZygpIHsNCi0JcmV0dXJuIGJhc2VUeXBlICsgKGlzQXJyYXkgPyAiW10iIDogIiIpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBib29sZWFuIGlzQXJyYXkoKSB7DQotCXJldHVybiBpc0FycmF5Ow0KLSAgICB9DQotDQotICAgIHB1YmxpYyBib29sZWFuIGlzQ2xhc3MoKSB7DQotCXJldHVybiBpc0NsYXNzOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBib29sZWFuIGlzUHJpbWl0aXZlKCkgew0KLQlyZXR1cm4gIWlzQ2xhc3MoKSAmJiAhaXNBcnJheSgpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBib29sZWFuIGlzVm9pZCgpIHsNCi0JcmV0dXJuIGJhc2VUeXBlLmVxdWFscygidm9pZCIpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBib29sZWFuIGlzQnVmZmVyKCkgew0KLQlyZXR1cm4gYmFzZVR5cGUuaW5kZXhPZigiQnVmZmVyIikgIT0gLTE7DQotICAgIH0NCi0NCi0gICAgcHVibGljIGJvb2xlYW4gaXNUeXBlZEJ1ZmZlcigpIHsNCi0JcmV0dXJuICFiYXNlVHlwZS5lcXVhbHMoImphdmEubmlvLkJ1ZmZlciIpICYmDQotCSAgICAoYmFzZVR5cGUuaW5kZXhPZigiQnVmZmVyIikgIT0gLTEpOw0KLSAgICB9DQotDQotICAgIHB1YmxpYyBzdGF0aWMgSlR5cGUgY29udmVydChDVHlwZSBjdHlwZSwgYm9vbGVhbiB1c2VBcnJheSkgew0KLSAJSlR5cGUgamF2YVR5cGUgPSBudWxsOw0KLSAJaWYgKHVzZUFycmF5KSB7DQotIAkgICAgamF2YVR5cGUgPSBhcnJheVR5cGVNYXBwaW5nLmdldChjdHlwZSk7DQotIAl9DQotIAlpZiAoamF2YVR5cGUgPT0gbnVsbCkgew0KLSAJICAgIGphdmFUeXBlID0gdHlwZU1hcHBpbmcuZ2V0KGN0eXBlKTsNCi0gCX0NCi0gCWlmIChqYXZhVHlwZSA9PSBudWxsKSB7DQotIAkgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIlVuc3VwcG9ydGVkIEMgdHlwZTogIiArIGN0eXBlKTsNCi0gCX0NCi0gCXJldHVybiBqYXZhVHlwZTsNCi0gICAgfQ0KLX0NCmRpZmYgLS1naXQgYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0puaUNvZGVFbWl0dGVyLmphdmEgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL0puaUNvZGVFbWl0dGVyLmphdmEKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDMzYjlhM2UuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3NyYy9KbmlDb2RlRW1pdHRlci5qYXZhCisrKyAvZGV2L251bGwKQEAgLTEsMTA4NiArMCwwIEBACi1pbXBvcnQgamF2YS5pby5QcmludFN0cmVhbTsKLWltcG9ydCBqYXZhLmlvLkZpbGVOb3RGb3VuZEV4Y2VwdGlvbjsKLWltcG9ydCBqYXZhLmlvLkZpbGVPdXRwdXRTdHJlYW07Ci1pbXBvcnQgamF2YS51dGlsLkFycmF5TGlzdDsKLWltcG9ydCBqYXZhLnV0aWwuSGFzaFNldDsKLWltcG9ydCBqYXZhLnV0aWwuSXRlcmF0b3I7Ci1pbXBvcnQgamF2YS51dGlsLkxpc3Q7Ci0KLS8qKgotICogRW1pdHMgYSBKYXZhIGludGVyZmFjZSBhbmQgSmF2YSAmIEMgaW1wbGVtZW50YXRpb24gZm9yIGEgQyBmdW5jdGlvbi4KLSAqCi0gKiA8cD4gVGhlIEphdmEgaW50ZXJmYWNlIHdpbGwgaGF2ZSBCdWZmZXIgYW5kIGFycmF5IHZhcmlhbnRzIGZvciBmdW5jdGlvbnMgdGhhdAotICogaGF2ZSBhIHR5cGVkIHBvaW50ZXIgYXJndW1lbnQuICBUaGUgYXJyYXkgdmFyaWFudCB3aWxsIGNvbnZlcnQgYSBzaW5nbGUgIjx0eXBlPiAqZGF0YSIKLSAqIGFyZ3VtZW50IHRvIGEgcGFpciBvZiBhcmd1bWVudHMgIjx0eXBlPltdIGRhdGEsIGludCBvZmZzZXQiLgotICovCi1wdWJsaWMgY2xhc3MgSm5pQ29kZUVtaXR0ZXIgaW1wbGVtZW50cyBDb2RlRW1pdHRlciB7Ci0KLSAgICAvLyBJZiB0cnVlLCB1c2UgQysrIHN0eWxlIGZvciBjYWxsaW5nIHRocm91Z2ggYSBKTklFbnYgKjoKLSAgICAvLyBlbnYtPkZ1bmMoLi4uKQotICAgIC8vIElmIGZhbHNlLCB1c2UgQyBzdHlsZToKLSAgICAvLyAoKmVudiktPkZ1bmMoZW52LCAuLi4pCi0gICAgc3RhdGljIGZpbmFsIGJvb2xlYW4gbVVzZUNQbHVzUGx1cyA9IHRydWU7Ci0KLSAgICBib29sZWFuIG1Vc2VDb250ZXh0UG9pbnRlciA9IHRydWU7Ci0KLSAgICBTdHJpbmcgbUNsYXNzUGF0aE5hbWU7Ci0gICAgCi0gICAgUGFyYW1ldGVyQ2hlY2tlciBtQ2hlY2tlcjsKLSAgICBQcmludFN0cmVhbSBtSmF2YTEwSW50ZXJmYWNlU3RyZWFtOwotICAgIFByaW50U3RyZWFtIG1KYXZhMTBFeHRJbnRlcmZhY2VTdHJlYW07Ci0gICAgUHJpbnRTdHJlYW0gbUphdmExMUludGVyZmFjZVN0cmVhbTsKLSAgICBQcmludFN0cmVhbSBtSmF2YTExRXh0SW50ZXJmYWNlU3RyZWFtOwotICAgIFByaW50U3RyZWFtIG1KYXZhMTFFeHRQYWNrSW50ZXJmYWNlU3RyZWFtOwotICAgIFByaW50U3RyZWFtIG1KYXZhSW1wbFN0cmVhbTsKLSAgICBQcmludFN0cmVhbSBtQ1N0cmVhbTsKLQotICAgIFByaW50U3RyZWFtIG1KYXZhSW50ZXJmYWNlU3RyZWFtOwotCi0gICAgTGlzdDxTdHJpbmc+IG5hdGl2ZVJlZ2lzdHJhdGlvbnMgPSBuZXcgQXJyYXlMaXN0PFN0cmluZz4oKTsKLQotICAgIGJvb2xlYW4gbmVlZHNFeGl0OwotCi0gICAgc3RhdGljIFN0cmluZyBpbmRlbnQgPSAiICAgICI7Ci0KLSAgICBIYXNoU2V0PFN0cmluZz4gbUZ1bmN0aW9uc0VtaXR0ZWQgPSBuZXcgSGFzaFNldDxTdHJpbmc+KCk7Ci0KLSAgICAvKioKLSAgICAgKiBAcGFyYW0gamF2YTEwSW50ZXJmYWNlU3RyZWFtIHRoZSBQcmludFN0cmVhbSB0byB3aGljaCB0byBlbWl0IHRoZSBKYXZhIGludGVyZmFjZSBmb3IgR0wgMS4wIGZ1bmN0aW9ucwotICAgICAqIEBwYXJhbSBqYXZhMTBFeHRJbnRlcmZhY2VTdHJlYW0gdGhlIFByaW50U3RyZWFtIHRvIHdoaWNoIHRvIGVtaXQgdGhlIEphdmEgaW50ZXJmYWNlIGZvciBHTCAxLjAgZXh0ZW5zaW9uIGZ1bmN0aW9ucwotICAgICAqIEBwYXJhbSBqYXZhMTFJbnRlcmZhY2VTdHJlYW0gdGhlIFByaW50U3RyZWFtIHRvIHdoaWNoIHRvIGVtaXQgdGhlIEphdmEgaW50ZXJmYWNlIGZvciBHTCAxLjEgZnVuY3Rpb25zIAotICAgICAqIEBwYXJhbSBqYXZhMTFFeHRJbnRlcmZhY2VTdHJlYW0gdGhlIFByaW50U3RyZWFtIHRvIHdoaWNoIHRvIGVtaXQgdGhlIEphdmEgaW50ZXJmYWNlIGZvciBHTCAxLjEgRXh0ZW5zaW9uIGZ1bmN0aW9ucwotICAgICAqIEBwYXJhbSBqYXZhMTFFeHRQYWNrSW50ZXJmYWNlU3RyZWFtIHRoZSBQcmludFN0cmVhbSB0byB3aGljaCB0byBlbWl0IHRoZSBKYXZhIGludGVyZmFjZSBmb3IgR0wgMS4xIEV4dGVuc2lvbiBQYWNrIGZ1bmN0aW9ucwotICAgICAqIEBwYXJhbSBqYXZhSW1wbFN0cmVhbSB0aGUgUHJpbnRTdHJlYW0gdG8gd2hpY2ggdG8gZW1pdCB0aGUgSmF2YSBpbXBsZW1lbnRhdGlvbgotICAgICAqIEBwYXJhbSBjU3RyZWFtIHRoZSBQcmludFN0cmVhbSB0byB3aGljaCB0byBlbWl0IHRoZSBDIGltcGxlbWVudGF0aW9uCi0gICAgICovCi0gICAgcHVibGljIEpuaUNvZGVFbWl0dGVyKFN0cmluZyBjbGFzc1BhdGhOYW1lLAotICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJhbWV0ZXJDaGVja2VyIGNoZWNrZXIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgIFByaW50U3RyZWFtIGphdmExMEludGVyZmFjZVN0cmVhbSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgUHJpbnRTdHJlYW0gamF2YTEwRXh0SW50ZXJmYWNlU3RyZWFtLAotICAgICAgICAgICAgICAgICAgICAgICAgICBQcmludFN0cmVhbSBqYXZhMTFJbnRlcmZhY2VTdHJlYW0sCi0gICAgICAgICAgICAgICAgICAgICAgICAgIFByaW50U3RyZWFtIGphdmExMUV4dEludGVyZmFjZVN0cmVhbSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgUHJpbnRTdHJlYW0gamF2YTExRXh0UGFja0ludGVyZmFjZVN0cmVhbSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgUHJpbnRTdHJlYW0gamF2YUltcGxTdHJlYW0sCi0gICAgICAgICAgICAgICAgICAgICAgICAgIFByaW50U3RyZWFtIGNTdHJlYW0sCi0gICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2xlYW4gdXNlQ29udGV4dFBvaW50ZXIpIHsKLSAgICAgICAgbUNsYXNzUGF0aE5hbWUgPSBjbGFzc1BhdGhOYW1lOwotICAgICAgICBtQ2hlY2tlciA9IGNoZWNrZXI7Ci0gICAgICAgIG1KYXZhMTBJbnRlcmZhY2VTdHJlYW0gPSBqYXZhMTBJbnRlcmZhY2VTdHJlYW07Ci0gICAgICAgIG1KYXZhMTBFeHRJbnRlcmZhY2VTdHJlYW0gPSBqYXZhMTBFeHRJbnRlcmZhY2VTdHJlYW07Ci0gICAgICAgIG1KYXZhMTFJbnRlcmZhY2VTdHJlYW0gPSBqYXZhMTFJbnRlcmZhY2VTdHJlYW07Ci0gICAgICAgIG1KYXZhMTFFeHRJbnRlcmZhY2VTdHJlYW0gPSBqYXZhMTFFeHRJbnRlcmZhY2VTdHJlYW07Ci0gICAgICAgIG1KYXZhMTFFeHRQYWNrSW50ZXJmYWNlU3RyZWFtID0gamF2YTExRXh0UGFja0ludGVyZmFjZVN0cmVhbTsKLSAgICAgICAgbUphdmFJbXBsU3RyZWFtID0gamF2YUltcGxTdHJlYW07Ci0gICAgICAgIG1DU3RyZWFtID0gY1N0cmVhbTsKLSAgICAgICAgbVVzZUNvbnRleHRQb2ludGVyID0gdXNlQ29udGV4dFBvaW50ZXI7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgc2V0VmVyc2lvbihpbnQgdmVyc2lvbiwgYm9vbGVhbiBleHQsIGJvb2xlYW4gcGFjaykgewotICAgICAgICBpZiAodmVyc2lvbiA9PSAwKSB7Ci0gICAgICAgICAgICBtSmF2YUludGVyZmFjZVN0cmVhbSA9IGV4dCA/IG1KYXZhMTBFeHRJbnRlcmZhY2VTdHJlYW0gOgotICAgICAgICAgICAgICAgIG1KYXZhMTBJbnRlcmZhY2VTdHJlYW07Ci0gICAgICAgIH0gZWxzZSBpZiAodmVyc2lvbiA9PSAxKSB7Ci0gICAgICAgICAgICBtSmF2YUludGVyZmFjZVN0cmVhbSA9IGV4dCA/Ci0gICAgICAgICAgICAgICAgKHBhY2sgPyBtSmF2YTExRXh0UGFja0ludGVyZmFjZVN0cmVhbSA6Ci0gICAgICAgICAgICAgICAgIG1KYXZhMTFFeHRJbnRlcmZhY2VTdHJlYW0pIDoKLSAgICAgICAgICAgICAgICBtSmF2YTExSW50ZXJmYWNlU3RyZWFtOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgdGhyb3cgbmV3IFJ1bnRpbWVFeGNlcHRpb24oIkJhZCB2ZXJzaW9uOiAiICsgdmVyc2lvbik7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBlbWl0Q29kZShDRnVuYyBjZnVuYywgU3RyaW5nIG9yaWdpbmFsKSB7Ci0gICAgICAgIEpGdW5jIGpmdW5jOwotICAgICAgICBTdHJpbmcgc2lnbmF0dXJlOwotICAgICAgICBib29sZWFuIGR1cGxpY2F0ZTsKLSAgICAgICAgCi0gICAgICAgIGlmIChjZnVuYy5oYXNUeXBlZFBvaW50ZXJBcmcoKSkgewotICAgICAgICAgICAgamZ1bmMgPSBKRnVuYy5jb252ZXJ0KGNmdW5jLCB0cnVlKTsKLQotICAgICAgICAgICAgLy8gRG9uJ3QgZW1pdCBkdXBsaWNhdGUgZnVuY3Rpb25zCi0gICAgICAgICAgICAvLyBUaGVzZSBtYXkgYXBwZWFyIGJlY2F1c2UgdGhleSBhcmUgZGVmaW5lZCBpbiBtdWx0aXBsZQotICAgICAgICAgICAgLy8gSmF2YSBpbnRlcmZhY2VzIChlLmcuLCBHTDExL0dMMTFFeHRlbnNpb25QYWNrKQotICAgICAgICAgICAgc2lnbmF0dXJlID0gamZ1bmMudG9TdHJpbmcoKTsKLSAgICAgICAgICAgIGR1cGxpY2F0ZSA9IGZhbHNlOwotICAgICAgICAgICAgaWYgKG1GdW5jdGlvbnNFbWl0dGVkLmNvbnRhaW5zKHNpZ25hdHVyZSkpIHsKLSAgICAgICAgICAgICAgICBkdXBsaWNhdGUgPSB0cnVlOwotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBtRnVuY3Rpb25zRW1pdHRlZC5hZGQoc2lnbmF0dXJlKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgaWYgKCFkdXBsaWNhdGUpIHsKLSAgICAgICAgICAgICAgICBlbWl0TmF0aXZlRGVjbGFyYXRpb24oamZ1bmMsIG1KYXZhSW1wbFN0cmVhbSk7Ci0gICAgICAgICAgICAgICAgZW1pdEphdmFDb2RlKGpmdW5jLCBtSmF2YUltcGxTdHJlYW0pOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgZW1pdEphdmFJbnRlcmZhY2VDb2RlKGpmdW5jLCBtSmF2YUludGVyZmFjZVN0cmVhbSk7Ci0gICAgICAgICAgICBpZiAoIWR1cGxpY2F0ZSkgewotICAgICAgICAgICAgICAgIGVtaXRKbmlDb2RlKGpmdW5jLCBtQ1N0cmVhbSk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBqZnVuYyA9IEpGdW5jLmNvbnZlcnQoY2Z1bmMsIGZhbHNlKTsKLQotICAgICAgICBzaWduYXR1cmUgPSBqZnVuYy50b1N0cmluZygpOwotICAgICAgICBkdXBsaWNhdGUgPSBmYWxzZTsKLSAgICAgICAgaWYgKG1GdW5jdGlvbnNFbWl0dGVkLmNvbnRhaW5zKHNpZ25hdHVyZSkpIHsKLSAgICAgICAgICAgIGR1cGxpY2F0ZSA9IHRydWU7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBtRnVuY3Rpb25zRW1pdHRlZC5hZGQoc2lnbmF0dXJlKTsKLSAgICAgICAgfQotCi0gICAgICAgIGlmICghZHVwbGljYXRlKSB7Ci0gICAgICAgICAgICBlbWl0TmF0aXZlRGVjbGFyYXRpb24oamZ1bmMsIG1KYXZhSW1wbFN0cmVhbSk7Ci0gICAgICAgIH0KLSAgICAgICAgZW1pdEphdmFJbnRlcmZhY2VDb2RlKGpmdW5jLCBtSmF2YUludGVyZmFjZVN0cmVhbSk7Ci0gICAgICAgIGlmICghZHVwbGljYXRlKSB7Ci0gICAgICAgICAgICBlbWl0SmF2YUNvZGUoamZ1bmMsIG1KYXZhSW1wbFN0cmVhbSk7Ci0gICAgICAgICAgICBlbWl0Sm5pQ29kZShqZnVuYywgbUNTdHJlYW0pOwotICAgICAgICB9Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZW1pdE5hdGl2ZURlY2xhcmF0aW9uKEpGdW5jIGpmdW5jLCBQcmludFN0cmVhbSBvdXQpIHsKLSAgICAgICAgb3V0LnByaW50bG4oIiAgICAvLyBDIGZ1bmN0aW9uICIgKyBqZnVuYy5nZXRDRnVuYygpLmdldE9yaWdpbmFsKCkpOwotICAgICAgICBvdXQucHJpbnRsbigpOwotCi0gICAgICAgIGVtaXRGdW5jdGlvbihqZnVuYywgb3V0LCB0cnVlLCBmYWxzZSk7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZW1pdEphdmFJbnRlcmZhY2VDb2RlKEpGdW5jIGpmdW5jLCBQcmludFN0cmVhbSBvdXQpIHsKLSAgICAgICAgZW1pdEZ1bmN0aW9uKGpmdW5jLCBvdXQsIGZhbHNlLCB0cnVlKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBlbWl0SmF2YUNvZGUoSkZ1bmMgamZ1bmMsIFByaW50U3RyZWFtIG91dCkgewotICAgICAgICBlbWl0RnVuY3Rpb24oamZ1bmMsIG91dCwgZmFsc2UsIGZhbHNlKTsKLSAgICB9Ci0gICAgCi0gICAgdm9pZCBlbWl0RnVuY3Rpb25DYWxsKEpGdW5jIGpmdW5jLCBQcmludFN0cmVhbSBvdXQsIFN0cmluZyBpaWksIGJvb2xlYW4gZ3JhYkFycmF5ICkgewotICAgICAgICBib29sZWFuIGlzVm9pZCA9IGpmdW5jLmdldFR5cGUoKS5pc1ZvaWQoKTsKLSAgICAgICAgYm9vbGVhbiBpc1BvaW50ZXJGdW5jID0gamZ1bmMuZ2V0TmFtZSgpLmVuZHNXaXRoKCJQb2ludGVyIikgJiYKLSAgICAgICAgICAgIGpmdW5jLmdldENGdW5jKCkuaGFzUG9pbnRlckFyZygpOwotCi0gICAgICAgIGlmICghaXNWb2lkKSB7Ci0gICAgICAgICAgICBvdXQucHJpbnRsbihpaWkgKwotICAgICAgICAgICAgICAgICAgICAgICAgamZ1bmMuZ2V0VHlwZSgpICsgIiBfcmV0dXJuVmFsdWU7Iik7Ci0gICAgICAgIH0KLSAgICAgICAgb3V0LnByaW50bG4oaWlpICsKLSAgICAgICAgICAgICAgICAgICAgKGlzVm9pZCA/ICIiIDogIl9yZXR1cm5WYWx1ZSA9ICIpICsKLSAgICAgICAgICAgICAgICAgICAgamZ1bmMuZ2V0TmFtZSgpICsKLSAgICAgICAgICAgICAgICAgICAgKGlzUG9pbnRlckZ1bmMgPyAiQm91bmRzIiA6ICIiICkgKwotICAgICAgICAgICAgICAgICAgICAiKCIpOwotCQotICAgICAgICBpbnQgbnVtQXJncyA9IGpmdW5jLmdldE51bUFyZ3MoKTsKLSAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1BcmdzOyBpKyspIHsKLSAgICAgICAgICAgIFN0cmluZyBhcmdOYW1lID0gamZ1bmMuZ2V0QXJnTmFtZShpKTsKLSAgICAgICAgICAgIEpUeXBlIGFyZ1R5cGUgPSBqZnVuYy5nZXRBcmdUeXBlKGkpOwotCi0gICAgICAgICAgICBpZiAoZ3JhYkFycmF5ICYmIGFyZ1R5cGUuaXNUeXBlZEJ1ZmZlcigpKSB7Ci0gICAgICAgICAgICAgICAgU3RyaW5nIHR5cGVOYW1lID0gYXJnVHlwZS5nZXRCYXNlVHlwZSgpOwotICAgICAgICAgICAgICAgIHR5cGVOYW1lID0gdHlwZU5hbWUuc3Vic3RyaW5nKDksIHR5cGVOYW1lLmxlbmd0aCgpIC0gNik7Ci0gICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgaW5kZW50ICsgImdldCIgKyB0eXBlTmFtZSArICJBcnJheSgiICsgYXJnTmFtZSArICIpLCIpOwotICAgICAgICAgICAgICAgIG91dC5wcmludChpaWkgKyBpbmRlbnQgKyAiZ2V0T2Zmc2V0KCIgKyBhcmdOYW1lICsgIikiKTsgCi0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIG91dC5wcmludChpaWkgKyBpbmRlbnQgKyBhcmdOYW1lKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChpID09IG51bUFyZ3MgLSAxKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGlzUG9pbnRlckZ1bmMpIHsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oIiwiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgaW5kZW50ICsgYXJnTmFtZSArICIucmVtYWluaW5nKCkiKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbigpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgb3V0LnByaW50bG4oIiwiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCQotICAgICAgICBvdXQucHJpbnRsbihpaWkgKyAiKTsiKTsKLSAgICB9Ci0KLSAgICB2b2lkIHByaW50SWZjaGVja1Bvc3RhbWJsZShQcmludFN0cmVhbSBvdXQsIGJvb2xlYW4gaXNCdWZmZXIsCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYm9vbGVhbiBlbWl0RXhjZXB0aW9uQ2hlY2ssIFN0cmluZyBpaWkpIHsKLSAgICAgICAgcHJpbnRJZmNoZWNrUG9zdGFtYmxlKG91dCwgaXNCdWZmZXIsIGVtaXRFeGNlcHRpb25DaGVjaywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJvZmZzZXQiLCAiX3JlbWFpbmluZyIsIGlpaSk7Ci0gICAgfQotCi0gICAgdm9pZCBwcmludElmY2hlY2tQb3N0YW1ibGUoUHJpbnRTdHJlYW0gb3V0LCBib29sZWFuIGlzQnVmZmVyLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJvb2xlYW4gZW1pdEV4Y2VwdGlvbkNoZWNrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFN0cmluZyBvZmZzZXQsIFN0cmluZyByZW1haW5pbmcsIFN0cmluZyBpaWkpIHsKLSAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICBkZWZhdWx0OiIpOwotICAgICAgICBvdXQucHJpbnRsbihpaWkgKyAiICAgICAgICBfbmVlZGVkID0gMDsiKTsKLSAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICAgICAgYnJlYWs7Iik7Ci0gICAgICAgIG91dC5wcmludGxuKGlpaSArICJ9Iik7Ci0KLSAgICAgICAgb3V0LnByaW50bG4oaWlpICsgImlmICgiICsgcmVtYWluaW5nICsgIiA8IF9uZWVkZWQpIHsiKTsKLSAgICAgICAgaWYgKGVtaXRFeGNlcHRpb25DaGVjaykgewotICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgaW5kZW50ICsgIl9leGNlcHRpb24gPSAxOyIpOwotICAgICAgICB9Ci0gICAgICAgIG91dC5wcmludGxuKGlpaSArIGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgIChtVXNlQ1BsdXNQbHVzID8gIl9lbnYiIDogIigqX2VudikiKSArCi0gICAgICAgICAgICAgICAgICAgICItPlRocm93TmV3KCIgKwotICAgICAgICAgICAgICAgICAgICAobVVzZUNQbHVzUGx1cyA/ICIiIDogIl9lbnYsICIpICsKLSAgICAgICAgICAgICAgICAgICAgIklBRUNsYXNzLCAiICsKLSAgICAgICAgICAgICAgICAgICAgIlwiIiArCi0gICAgICAgICAgICAgICAgICAgIChpc0J1ZmZlciA/IAotICAgICAgICAgICAgICAgICAgICAgInJlbWFpbmluZygpIiA6ICJsZW5ndGggLSAiICsgb2Zmc2V0KSArCi0gICAgICAgICAgICAgICAgICAgICIgPCBuZWVkZWRcIik7Iik7Ci0gICAgICAgIG91dC5wcmludGxuKGlpaSArIGluZGVudCArICJnb3RvIGV4aXQ7Iik7Ci0gICAgICAgIG5lZWRzRXhpdCA9IHRydWU7Ci0gICAgICAgIG91dC5wcmludGxuKGlpaSArICJ9Iik7Ci0gICAgfQotCi0gICAgYm9vbGVhbiBpc051bGxBbGxvd2VkKENGdW5jIGNmdW5jKSB7Ci0gICAgICAgIFN0cmluZ1tdIGNoZWNrcyA9IG1DaGVja2VyLmdldENoZWNrcyhjZnVuYy5nZXROYW1lKCkpOwotICAgICAgICBpbnQgaW5kZXggPSAxOwotICAgICAgICBpZiAoY2hlY2tzICE9IG51bGwpIHsKLSAgICAgICAgICAgIHdoaWxlIChpbmRleCA8IGNoZWNrcy5sZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBpZiAoY2hlY2tzW2luZGV4XS5lcXVhbHMoInJldHVybiIpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGluZGV4ICs9IDI7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChjaGVja3NbaW5kZXhdLnN0YXJ0c1dpdGgoImNoZWNrIikpIHsKLSAgICAgICAgICAgICAgICAgICAgaW5kZXggKz0gMzsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGNoZWNrc1tpbmRleF0uZXF1YWxzKCJpZmNoZWNrIikpIHsKLSAgICAgICAgICAgICAgICAgICAgaW5kZXggKz0gNTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGNoZWNrc1tpbmRleF0uZXF1YWxzKCJ1bnN1cHBvcnRlZCIpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGluZGV4ICs9IDE7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChjaGVja3NbaW5kZXhdLmVxdWFscygibnVsbEFsbG93ZWQiKSkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0ub3V0LnByaW50bG4oIkVycm9yOiB1bmtub3duIGtleXdvcmQgXCIiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNoZWNrc1tpbmRleF0gKyAiXCIiKTsKLSAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmV4aXQoMCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICBTdHJpbmcgZ2V0RXJyb3JSZXR1cm5WYWx1ZShDRnVuYyBjZnVuYykgewotICAgICAgICBDVHlwZSByZXR1cm5UeXBlID0gY2Z1bmMuZ2V0VHlwZSgpOwotICAgICAgICBib29sZWFuIGlzVm9pZCA9IHJldHVyblR5cGUuaXNWb2lkKCk7Ci0gICAgICAgIGlmIChpc1ZvaWQpIHsKLSAgICAgICAgICAgIHJldHVybiBudWxsOwotICAgICAgICB9Ci0KLSAgICAgICAgU3RyaW5nW10gY2hlY2tzID0gbUNoZWNrZXIuZ2V0Q2hlY2tzKGNmdW5jLmdldE5hbWUoKSk7Ci0KLSAgICAgICAgaW50IGluZGV4ID0gMTsKLSAgICAgICAgaWYgKGNoZWNrcyAhPSBudWxsKSB7Ci0gICAgICAgICAgICB3aGlsZSAoaW5kZXggPCBjaGVja3MubGVuZ3RoKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGNoZWNrc1tpbmRleF0uZXF1YWxzKCJyZXR1cm4iKSkgewotICAgICAgICAgICAgICAgICAgICByZXR1cm4gY2hlY2tzW2luZGV4ICsgMV07Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChjaGVja3NbaW5kZXhdLnN0YXJ0c1dpdGgoImNoZWNrIikpIHsKLSAgICAgICAgICAgICAgICAgICAgaW5kZXggKz0gMzsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGNoZWNrc1tpbmRleF0uZXF1YWxzKCJpZmNoZWNrIikpIHsKLSAgICAgICAgICAgICAgICAgICAgaW5kZXggKz0gNTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGNoZWNrc1tpbmRleF0uZXF1YWxzKCJ1bnN1cHBvcnRlZCIpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGluZGV4ICs9IDE7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChjaGVja3NbaW5kZXhdLmVxdWFscygibnVsbEFsbG93ZWQiKSkgewotICAgICAgICAgICAgICAgICAgICBpbmRleCArPSAxOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIFN5c3RlbS5vdXQucHJpbnRsbigiRXJyb3I6IHVua25vd24ga2V5d29yZCBcIiIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY2hlY2tzW2luZGV4XSArICJcIiIpOwotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uZXhpdCgwKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gbnVsbDsKLSAgICB9Ci0KLSAgICBib29sZWFuIGlzVW5zdXBwb3J0ZWRGdW5jKENGdW5jIGNmdW5jKSB7Ci0gICAgICAgIFN0cmluZ1tdIGNoZWNrcyA9IG1DaGVja2VyLmdldENoZWNrcyhjZnVuYy5nZXROYW1lKCkpOwotICAgICAgICBpbnQgaW5kZXggPSAxOwotICAgICAgICBpZiAoY2hlY2tzICE9IG51bGwpIHsKLSAgICAgICAgICAgIHdoaWxlIChpbmRleCA8IGNoZWNrcy5sZW5ndGgpIHsKLSAgICAgICAgICAgICAgICBpZiAoY2hlY2tzW2luZGV4XS5lcXVhbHMoInVuc3VwcG9ydGVkIikpIHsKLSAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChjaGVja3NbaW5kZXhdLmVxdWFscygicmV0dXJuIikpIHsKLSAgICAgICAgICAgICAgICAgICAgaW5kZXggKz0gMjsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGNoZWNrc1tpbmRleF0uc3RhcnRzV2l0aCgiY2hlY2siKSkgewotICAgICAgICAgICAgICAgICAgICBpbmRleCArPSAzOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoY2hlY2tzW2luZGV4XS5lcXVhbHMoImlmY2hlY2siKSkgewotICAgICAgICAgICAgICAgICAgICBpbmRleCArPSA1OwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoY2hlY2tzW2luZGV4XS5lcXVhbHMoIm51bGxBbGxvd2VkIikpIHsKLSAgICAgICAgICAgICAgICAgICAgaW5kZXggKz0gMTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0ub3V0LnByaW50bG4oIkVycm9yOiB1bmtub3duIGtleXdvcmQgXCIiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNoZWNrc1tpbmRleF0gKyAiXCIiKTsKLSAgICAgICAgICAgICAgICAgICAgU3lzdGVtLmV4aXQoMCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBmYWxzZTsKLSAgICB9Ci0KLSAgICB2b2lkIGVtaXROYXRpdmVCb3VuZHNDaGVja3MoQ0Z1bmMgY2Z1bmMsIFN0cmluZyBjbmFtZSwgUHJpbnRTdHJlYW0gb3V0LAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBib29sZWFuIGlzQnVmZmVyLCBib29sZWFuIGVtaXRFeGNlcHRpb25DaGVjaywKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU3RyaW5nIG9mZnNldCwgU3RyaW5nIHJlbWFpbmluZywgU3RyaW5nIGlpaSkgewotICAgICAgICBDVHlwZSByZXR1cm5UeXBlID0gY2Z1bmMuZ2V0VHlwZSgpOwotICAgICAgICBib29sZWFuIGlzVm9pZCA9IHJldHVyblR5cGUuaXNWb2lkKCk7Ci0KLSAgICAgICAgU3RyaW5nW10gY2hlY2tzID0gbUNoZWNrZXIuZ2V0Q2hlY2tzKGNmdW5jLmdldE5hbWUoKSk7Ci0gICAgICAgIFN0cmluZyBjaGVja1ZhcjsKLSAgICAgICAgU3RyaW5nIHJldHZhbCA9IGdldEVycm9yUmV0dXJuVmFsdWUoY2Z1bmMpOwotCi0gICAgICAgIGJvb2xlYW4gbGFzdFdhc0lmY2hlY2sgPSBmYWxzZTsKLQotICAgICAgICBpbnQgaW5kZXggPSAxOwotICAgICAgICBpZiAoY2hlY2tzICE9IG51bGwpIHsKLSAgICAgICAgICAgIGJvb2xlYW4gcmVtYWluaW5nRGVjbGFyZWQgPSBmYWxzZTsKLSAgICAgICAgICAgIGJvb2xlYW4gbnVsbENoZWNrRGVjbGFyZWQgPSBmYWxzZTsKLSAgICAgICAgICAgIGJvb2xlYW4gb2Zmc2V0Q2hlY2tlZCA9IGZhbHNlOwotICAgICAgICAgICAgd2hpbGUgKGluZGV4IDwgY2hlY2tzLmxlbmd0aCkgewotICAgICAgICAgICAgICAgIGlmIChjaGVja3NbaW5kZXhdLnN0YXJ0c1dpdGgoImNoZWNrIikpIHsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGxhc3RXYXNJZmNoZWNrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBwcmludElmY2hlY2tQb3N0YW1ibGUob3V0LCBpc0J1ZmZlciwgZW1pdEV4Y2VwdGlvbkNoZWNrLAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9mZnNldCwgcmVtYWluaW5nLCBpaWkpOwotICAgICAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgICAgIGxhc3RXYXNJZmNoZWNrID0gZmFsc2U7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChjbmFtZSAhPSBudWxsICYmICFjbmFtZS5lcXVhbHMoY2hlY2tzW2luZGV4ICsgMV0pKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBpbmRleCArPSAzOwotICAgICAgICAgICAgICAgICAgICAgICAgY29udGludWU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgImlmICgiICsgcmVtYWluaW5nICsgIiA8ICIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjaGVja3NbaW5kZXggKyAyXSArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICIpIHsiKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKGVtaXRFeGNlcHRpb25DaGVjaykgewotICAgICAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgaW5kZW50ICsgIl9leGNlcHRpb24gPSAxOyIpOwotICAgICAgICAgICAgICAgICAgICB9Ci0JCSAgICBTdHJpbmcgZXhjZXB0aW9uQ2xhc3NOYW1lID0gIklBRUNsYXNzIjsKLQkJICAgIC8vIElmIHRoZSAiY2hlY2siIGtleXdvcmQgd2FzIG9mIHRoZSBmb3JtCi0JCSAgICAvLyAiY2hlY2tfPGNsYXNzIG5hbWU+IiwgdXNlIHRoZSBjbGFzcyBuYW1lIGluIHRoZQotCQkgICAgLy8gZXhjZXB0aW9uIHRvIGJlIHRocm93bgotCQkgICAgaW50IHVuZGVyc2NvcmUgPSBjaGVja3NbaW5kZXhdLmluZGV4T2YoJ18nKTsKLQkJICAgIGlmICh1bmRlcnNjb3JlID49IDApIHsKLQkJCWV4Y2VwdGlvbkNsYXNzTmFtZSA9IGNoZWNrc1tpbmRleF0uc3Vic3RyaW5nKHVuZGVyc2NvcmUgKyAxKSArICJDbGFzcyI7Ci0JCSAgICB9Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArIGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChtVXNlQ1BsdXNQbHVzID8gIl9lbnYiIDogIigqX2VudikiKSArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICItPlRocm93TmV3KCIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAobVVzZUNQbHVzUGx1cyA/ICIiIDogIl9lbnYsICIpICsKLQkJCQlleGNlcHRpb25DbGFzc05hbWUgKyAiLCAiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIlwiIiArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChpc0J1ZmZlciA/IAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgInJlbWFpbmluZygpIiA6ICJsZW5ndGggLSAiICsgb2Zmc2V0KSArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICIgPCAiICsgY2hlY2tzW2luZGV4ICsgMl0gKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiXCIpOyIpOwotCi0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArIGluZGVudCArICJnb3RvIGV4aXQ7Iik7Ci0gICAgICAgICAgICAgICAgICAgIG5lZWRzRXhpdCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICJ9Iik7Ci0gICAgICAgICAgICAgICAgCi0gICAgICAgICAgICAgICAgICAgIGluZGV4ICs9IDM7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChjaGVja3NbaW5kZXhdLmVxdWFscygiaWZjaGVjayIpKSB7Ci0gICAgICAgICAgICAgICAgICAgIFN0cmluZ1tdIG1hdGNoZXMgPSBjaGVja3NbaW5kZXggKyA0XS5zcGxpdCgiLCIpOwotCi0gICAgICAgICAgICAgICAgICAgIGlmICghbGFzdFdhc0lmY2hlY2spIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICJpbnQgX25lZWRlZDsiKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAic3dpdGNoICgiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNoZWNrc1tpbmRleCArIDNdICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICIpIHsiKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBtYXRjaGVzLmxlbmd0aDsgaSsrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbigiI2lmIGRlZmluZWQoIiArIG1hdGNoZXNbaV0gKyAiKSIpOwotICAgICAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICIgICAgY2FzZSAiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1hdGNoZXNbaV0gKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIjoiKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKCIjZW5kaWYgLy8gZGVmaW5lZCgiICsgbWF0Y2hlc1tpXSArICIpIik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIiAgICAgICAgX25lZWRlZCA9ICIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjaGVja3NbaW5kZXggKyAyXSArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICI7Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICIgICAgICAgIGJyZWFrOyIpOwotICAgICAgICAgICAgICAgIAotICAgICAgICAgICAgICAgICAgICBsYXN0V2FzSWZjaGVjayA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIGluZGV4ICs9IDU7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChjaGVja3NbaW5kZXhdLmVxdWFscygicmV0dXJuIikpIHsKLSAgICAgICAgICAgICAgICAgICAgLy8gaWdub3JlCi0gICAgICAgICAgICAgICAgICAgIGluZGV4ICs9IDI7Ci0gICAgICAgICAgICAgICAgfSBlbHNlIGlmIChjaGVja3NbaW5kZXhdLmVxdWFscygidW5zdXBwb3J0ZWQiKSkgewotICAgICAgICAgICAgICAgICAgICAvLyBpZ25vcmUKLSAgICAgICAgICAgICAgICAgICAgaW5kZXggKz0gMTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGNoZWNrc1tpbmRleF0uZXF1YWxzKCJudWxsQWxsb3dlZCIpKSB7Ci0gICAgICAgICAgICAgICAgICAgIC8vIGlnbm9yZQotICAgICAgICAgICAgICAgICAgICBpbmRleCArPSAxOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIFN5c3RlbS5vdXQucHJpbnRsbigiRXJyb3I6IHVua25vd24ga2V5d29yZCBcIiIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY2hlY2tzW2luZGV4XSArICJcIiIpOwotICAgICAgICAgICAgICAgICAgICBTeXN0ZW0uZXhpdCgwKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAobGFzdFdhc0lmY2hlY2spIHsKLSAgICAgICAgICAgIHByaW50SWZjaGVja1Bvc3RhbWJsZShvdXQsIGlzQnVmZmVyLCBlbWl0RXhjZXB0aW9uQ2hlY2ssIGlpaSk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBib29sZWFuIGhhc05vbkNvbnN0QXJnKEpGdW5jIGpmdW5jLCBDRnVuYyBjZnVuYywKLSAgICAgICAgTGlzdDxJbnRlZ2VyPiBub25QcmltaXRpdmVBcmdzKSB7Ci0gICAgICAgIGlmIChub25QcmltaXRpdmVBcmdzLnNpemUoKSA+IDApIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSBub25QcmltaXRpdmVBcmdzLnNpemUoKSAtIDE7IGkgPj0gMDsgaS0tKSB7Ci0gICAgICAgICAgICAgICAgaW50IGlkeCA9IG5vblByaW1pdGl2ZUFyZ3MuZ2V0KGkpLmludFZhbHVlKCk7Ci0gICAgICAgICAgICAgICAgaW50IGNJbmRleCA9IGpmdW5jLmdldEFyZ0NJbmRleChpZHgpOwotICAgICAgICAgICAgICAgIGlmIChqZnVuYy5nZXRBcmdUeXBlKGlkeCkuaXNBcnJheSgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmICghY2Z1bmMuZ2V0QXJnVHlwZShjSW5kZXgpLmlzQ29uc3QoKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGpmdW5jLmdldEFyZ1R5cGUoaWR4KS5pc0J1ZmZlcigpKSB7Ci0gICAgICAgICAgICAgICAgICAgIGlmICghY2Z1bmMuZ2V0QXJnVHlwZShjSW5kZXgpLmlzQ29uc3QoKSkgewotICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICByZXR1cm4gZmFsc2U7Ci0gICAgfQotICAgIAotICAgIC8qKgotICAgICAqIEVtaXQgYSBmdW5jdGlvbiBpbiBzZXZlcmFsIHZhcmlhbnRzOgotICAgICAqCi0gICAgICogaWYgbmF0aXZlRGVjbDogcHVibGljIG5hdGl2ZSA8cmV0dXJudHlwZT4gZnVuYyhhcmdzKTsKLSAgICAgKgotICAgICAqIGlmICFuYXRpdmVEZWNsOgotICAgICAqICAgaWYgaW50ZXJmYWNlRGVjbDogIHB1YmxpYyA8cmV0dXJudHlwZT4gZnVuYyhhcmdzKTsKLSAgICAgKiAgIGlmICFpbnRlcmZhY2VEZWNsOiBwdWJsaWMgPHJldHVybnR5cGU+IGZ1bmMoYXJncykgeyBib2R5IH0KLSAgICAgKi8KLSAgICB2b2lkIGVtaXRGdW5jdGlvbihKRnVuYyBqZnVuYywKLSAgICAgICAgICAgICAgICAgICAgICBQcmludFN0cmVhbSBvdXQsCi0gICAgICAgICAgICAgICAgICAgICAgYm9vbGVhbiBuYXRpdmVEZWNsLCBib29sZWFuIGludGVyZmFjZURlY2wpIHsKLSAgICAgICAgYm9vbGVhbiBpc1BvaW50ZXJGdW5jID0KLSAgICAgICAgICAgIGpmdW5jLmdldE5hbWUoKS5lbmRzV2l0aCgiUG9pbnRlciIpICYmCi0gICAgICAgICAgICBqZnVuYy5nZXRDRnVuYygpLmhhc1BvaW50ZXJBcmcoKTsKLQotICAgICAgICBpZiAoIW5hdGl2ZURlY2wgJiYgIWludGVyZmFjZURlY2wgJiYgIWlzUG9pbnRlckZ1bmMpIHsKLSAgICAgICAgICAgIC8vIElmIGl0J3Mgbm90IGEgcG9pbnRlciBmdW5jdGlvbiwgd2UndmUgYWxyZWFkeSBlbWl0dGVkIGl0Ci0gICAgICAgICAgICAvLyB3aXRoIG5hdGl2ZURlY2wgPT0gdHJ1ZQotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKGlzUG9pbnRlckZ1bmMpIHsKLSAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgICAgICAobmF0aXZlRGVjbCA/ICJwcml2YXRlIG5hdGl2ZSAiIDoKLSAgICAgICAgICAgICAgICAgICAgICAgICAoaW50ZXJmYWNlRGVjbCA/ICIiIDogInB1YmxpYyAiKSkgKwotICAgICAgICAgICAgICAgICAgICAgICAgamZ1bmMuZ2V0VHlwZSgpICsgIiAiICsKLSAgICAgICAgICAgICAgICAgICAgICAgIGpmdW5jLmdldE5hbWUoKSArCi0gICAgICAgICAgICAgICAgICAgICAgICAobmF0aXZlRGVjbCA/ICJCb3VuZHMiIDogIiIpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICIoIik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKwotICAgICAgICAgICAgICAgICAgICAgICAgKG5hdGl2ZURlY2wgPyAicHVibGljIG5hdGl2ZSAiIDoKLSAgICAgICAgICAgICAgICAgICAgICAgICAoaW50ZXJmYWNlRGVjbCA/ICIiIDogInB1YmxpYyAiKSkgKwotICAgICAgICAgICAgICAgICAgICAgICAgamZ1bmMuZ2V0VHlwZSgpICsgIiAiICsKLSAgICAgICAgICAgICAgICAgICAgICAgIGpmdW5jLmdldE5hbWUoKSArCi0gICAgICAgICAgICAgICAgICAgICAgICAiKCIpOwotICAgICAgICB9Ci0JCi0gICAgICAgIGludCBudW1BcmdzID0gamZ1bmMuZ2V0TnVtQXJncygpOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUFyZ3M7IGkrKykgewotICAgICAgICAgICAgU3RyaW5nIGFyZ05hbWUgPSBqZnVuYy5nZXRBcmdOYW1lKGkpOwotICAgICAgICAgICAgSlR5cGUgYXJnVHlwZSA9IGpmdW5jLmdldEFyZ1R5cGUoaSk7Ci0JICAgIAotICAgICAgICAgICAgb3V0LnByaW50KGluZGVudCArIGluZGVudCArIGFyZ1R5cGUgKyAiICIgKyBhcmdOYW1lKTsKLSAgICAgICAgICAgIGlmIChpID09IG51bUFyZ3MgLSAxKSB7Ci0gICAgICAgICAgICAgICAgaWYgKGlzUG9pbnRlckZ1bmMgJiYgbmF0aXZlRGVjbCkgewotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbigiLCIpOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyBpbmRlbnQgKyAiaW50IHJlbWFpbmluZyIpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKCk7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBvdXQucHJpbnRsbigiLCIpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG5hdGl2ZURlY2wgfHwgaW50ZXJmYWNlRGVjbCkgewotICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgIik7Iik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyAiKSB7Iik7Ci0KLSAgICAgICAgICAgIFN0cmluZyBpaWkgPSBpbmRlbnQgKyBpbmRlbnQ7Ci0KLSAgICAgICAgICAgIFN0cmluZyBmbmFtZSA9IGpmdW5jLmdldE5hbWUoKTsKLSAgICAgICAgICAgIGlmIChpc1BvaW50ZXJGdW5jKSB7Ci0gICAgICAgICAgICAgICAgLy8gVE9ETyAtIGRlYWwgd2l0aCBWQk8gdmFyaWFudHMKLSAgICAgICAgICAgICAgICBpZiAoZm5hbWUuZXF1YWxzKCJnbENvbG9yUG9pbnRlciIpKSB7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICJpZiAoKHNpemUgPT0gNCkgJiYiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICAoKHR5cGUgPT0gR0xfRkxPQVQpIHx8Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICIgICAgICh0eXBlID09IEdMX1VOU0lHTkVEX0JZVEUpIHx8Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICIgICAgICh0eXBlID09IEdMX0ZJWEVEKSkgJiYiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICAoc3RyaWRlID49IDApKSB7Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArIGluZGVudCArICJfY29sb3JQb2ludGVyID0gcG9pbnRlcjsiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIn0iKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGZuYW1lLmVxdWFscygiZ2xOb3JtYWxQb2ludGVyIikpIHsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgImlmICgoKHR5cGUgPT0gR0xfRkxPQVQpIHx8Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICIgICAgICh0eXBlID09IEdMX0JZVEUpIHx8Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICIgICAgICh0eXBlID09IEdMX1NIT1JUKSB8fCIpOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpaWkgKyAiICAgICAodHlwZSA9PSBHTF9GSVhFRCkpICYmIik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICIgICAgKHN0cmlkZSA+PSAwKSkgeyIpOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpaWkgKyBpbmRlbnQgKyAiX25vcm1hbFBvaW50ZXIgPSBwb2ludGVyOyIpOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpaWkgKyAifSIpOwotICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoZm5hbWUuZXF1YWxzKCJnbFRleENvb3JkUG9pbnRlciIpKSB7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICJpZiAoKChzaXplID09IDIpIHx8Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICIgICAgIChzaXplID09IDMpIHx8Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICIgICAgIChzaXplID09IDQpKSAmJiIpOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpaWkgKyAiICAgICgodHlwZSA9PSBHTF9GTE9BVCkgfHwiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICAgKHR5cGUgPT0gR0xfQllURSkgfHwiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICAgKHR5cGUgPT0gR0xfU0hPUlQpIHx8Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICIgICAgICh0eXBlID09IEdMX0ZJWEVEKSkgJiYiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICAoc3RyaWRlID49IDApKSB7Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArIGluZGVudCArICJfdGV4Q29vcmRQb2ludGVyID0gcG9pbnRlcjsiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIn0iKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGZuYW1lLmVxdWFscygiZ2xWZXJ0ZXhQb2ludGVyIikpIHsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgImlmICgoKHNpemUgPT0gMikgfHwiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICAgKHNpemUgPT0gMykgfHwiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICAgKHNpemUgPT0gNCkpICYmIik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGlpaSArICIgICAgKCh0eXBlID09IEdMX0ZMT0FUKSB8fCIpOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpaWkgKyAiICAgICAodHlwZSA9PSBHTF9CWVRFKSB8fCIpOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpaWkgKyAiICAgICAodHlwZSA9PSBHTF9TSE9SVCkgfHwiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIiAgICAgKHR5cGUgPT0gR0xfRklYRUQpKSAmJiIpOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpaWkgKyAiICAgIChzdHJpZGUgPj0gMCkpIHsiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgaW5kZW50ICsgIl92ZXJ0ZXhQb2ludGVyID0gcG9pbnRlcjsiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaWlpICsgIn0iKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgIC8vIGVtaXRCb3VuZHNDaGVja3MoamZ1bmMsIG91dCwgaWlpKTsKLSAgICAgICAgICAgIGVtaXRGdW5jdGlvbkNhbGwoamZ1bmMsIG91dCwgaWlpLCBmYWxzZSk7Ci0KLSAgICAgICAgICAgIGJvb2xlYW4gaXNWb2lkID0gamZ1bmMuZ2V0VHlwZSgpLmlzVm9pZCgpOwotCi0gICAgICAgICAgICBpZiAoIWlzVm9pZCkgewotICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArIGluZGVudCArICJyZXR1cm4gX3JldHVyblZhbHVlOyIpOwotICAgICAgICAgICAgfQotICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgIn0iKTsKLSAgICAgICAgfQotICAgICAgICBvdXQucHJpbnRsbigpOwotICAgIH0KLQotICAgIHB1YmxpYyBzdGF0aWMgU3RyaW5nIGdldEpuaU5hbWUoSlR5cGUgalR5cGUpIHsKLSAgICAgICAgU3RyaW5nIGpuaU5hbWUgPSAiIjsKLSAgICAgICAgaWYgKGpUeXBlLmlzQ2xhc3MoKSkgewotICAgICAgICAgICAgcmV0dXJuICJMIiArIGpUeXBlLmdldEJhc2VUeXBlKCkgKyAiOyI7Ci0gICAgICAgIH0gZWxzZSBpZiAoalR5cGUuaXNBcnJheSgpKSB7Ci0gICAgICAgICAgICBqbmlOYW1lID0gIlsiOwotICAgICAgICB9Ci0JCi0gICAgICAgIFN0cmluZyBiYXNlVHlwZSA9IGpUeXBlLmdldEJhc2VUeXBlKCk7Ci0gICAgICAgIGlmIChiYXNlVHlwZS5lcXVhbHMoImludCIpKSB7Ci0gICAgICAgICAgICBqbmlOYW1lICs9ICJJIjsKLSAgICAgICAgfSBlbHNlIGlmIChiYXNlVHlwZS5lcXVhbHMoImZsb2F0IikpIHsKLSAgICAgICAgICAgIGpuaU5hbWUgKz0gIkYiOwotICAgICAgICB9IGVsc2UgaWYgKGJhc2VUeXBlLmVxdWFscygiYm9vbGVhbiIpKSB7Ci0gICAgICAgICAgICBqbmlOYW1lICs9ICJaIjsKLSAgICAgICAgfSBlbHNlIGlmIChiYXNlVHlwZS5lcXVhbHMoInNob3J0IikpIHsKLSAgICAgICAgICAgIGpuaU5hbWUgKz0gIlMiOwotICAgICAgICB9IGVsc2UgaWYgKGJhc2VUeXBlLmVxdWFscygibG9uZyIpKSB7Ci0gICAgICAgICAgICBqbmlOYW1lICs9ICJMIjsKLSAgICAgICAgfSBlbHNlIGlmIChiYXNlVHlwZS5lcXVhbHMoImJ5dGUiKSkgewotICAgICAgICAgICAgam5pTmFtZSArPSAiQiI7Ci0gICAgICAgIH0KLSAgICAgICAgcmV0dXJuIGpuaU5hbWU7Ci0gICAgfQotCi0gICAgU3RyaW5nIGdldEpuaVR5cGUoSlR5cGUgalR5cGUpIHsKLSAgICAgICAgaWYgKGpUeXBlLmlzVm9pZCgpKSB7Ci0gICAgICAgICAgICByZXR1cm4gInZvaWQiOwotICAgICAgICB9Ci0KLSAgICAgICAgU3RyaW5nIGJhc2VUeXBlID0galR5cGUuZ2V0QmFzZVR5cGUoKTsKLSAgICAgICAgaWYgKGpUeXBlLmlzUHJpbWl0aXZlKCkpIHsKLSAgICAgICAgICAgIGlmIChiYXNlVHlwZS5lcXVhbHMoIlN0cmluZyIpKSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuICJqc3RyaW5nIjsKLSAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgcmV0dXJuICJqIiArIGJhc2VUeXBlOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgaWYgKGpUeXBlLmlzQXJyYXkoKSkgewotICAgICAgICAgICAgcmV0dXJuICJqIiArIGJhc2VUeXBlICsgIkFycmF5IjsKLSAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgIHJldHVybiAiam9iamVjdCI7Ci0gICAgICAgIH0KLSAgICB9Ci0gICAgCi0gICAgU3RyaW5nIGdldEpuaU1hbmdsZWROYW1lKFN0cmluZyBuYW1lKSB7Ci0gICAgICAgIG5hbWUgPSBuYW1lLnJlcGxhY2VBbGwoIl8iLCAiXzEiKTsKLSAgICAgICAgbmFtZSA9IG5hbWUucmVwbGFjZUFsbCgiOyIsICJfMiIpOwotICAgICAgICBuYW1lID0gbmFtZS5yZXBsYWNlQWxsKCJcXFsiLCAiXzMiKTsKLSAgICAgICAgcmV0dXJuIG5hbWU7Ci0gICAgfQotCi0gICAgcHVibGljIHZvaWQgZW1pdEpuaUNvZGUoSkZ1bmMgamZ1bmMsIFByaW50U3RyZWFtIG91dCkgewotICAgICAgICBDRnVuYyBjZnVuYyA9IGpmdW5jLmdldENGdW5jKCk7Ci0JCi0gICAgICAgIC8vIEVtaXQgY29tbWVudCBpZGVudGlmeWluZyBvcmlnaW5hbCBDIGZ1bmN0aW9uCi0gICAgICAgIC8vCi0gICAgICAgIC8vIEV4YW1wbGU6Ci0gICAgICAgIC8vCi0gICAgICAgIC8vIC8qIHZvaWQgZ2xDbGlwUGxhbmVmICggR0xlbnVtIHBsYW5lLCBjb25zdCBHTGZsb2F0ICplcXVhdGlvbiApICovCi0gICAgICAgIC8vCi0gICAgICAgIG91dC5wcmludGxuKCIvKiAiICsgY2Z1bmMuZ2V0T3JpZ2luYWwoKSArICIgKi8iKTsKLQotICAgICAgICAvLyBFbWl0IEpOSSBzaWduYXR1cmUgKG5hbWUpCi0gICAgICAgIC8vCi0gICAgICAgIC8vIEV4YW1wbGU6Ci0gICAgICAgIC8vCi0gICAgICAgIC8vIHZvaWQKLSAgICAgICAgLy8gYW5kcm9pZF9nbENsaXBQbGFuZWZfX0lfM0ZJCi0gICAgICAgIC8vCi0KLSAgICAgICAgU3RyaW5nIG91dE5hbWUgPSAiYW5kcm9pZF8iICsgamZ1bmMuZ2V0TmFtZSgpOwotICAgICAgICBib29sZWFuIGlzUG9pbnRlckZ1bmMgPSBvdXROYW1lLmVuZHNXaXRoKCJQb2ludGVyIikgJiYKLSAgICAgICAgICAgIGpmdW5jLmdldENGdW5jKCkuaGFzUG9pbnRlckFyZygpOwotICAgICAgICBib29sZWFuIGlzVkJPUG9pbnRlckZ1bmMgPSAob3V0TmFtZS5lbmRzV2l0aCgiUG9pbnRlciIpIHx8Ci0gICAgICAgICAgICBvdXROYW1lLmVuZHNXaXRoKCJEcmF3RWxlbWVudHMiKSkgJiYKLSAgICAgICAgICAgICFqZnVuYy5nZXRDRnVuYygpLmhhc1BvaW50ZXJBcmcoKTsKLSAgICAgICAgaWYgKGlzUG9pbnRlckZ1bmMpIHsKLSAgICAgICAgICAgIG91dE5hbWUgKz0gIkJvdW5kcyI7Ci0gICAgICAgIH0KLQotICAgICAgICBvdXQucHJpbnQoInN0YXRpYyAiKTsKLSAgICAgICAgb3V0LnByaW50bG4oZ2V0Sm5pVHlwZShqZnVuYy5nZXRUeXBlKCkpKTsKLSAgICAgICAgb3V0LnByaW50KG91dE5hbWUpOwotCi0gICAgICAgIFN0cmluZyByc2lnbmF0dXJlID0gZ2V0Sm5pTmFtZShqZnVuYy5nZXRUeXBlKCkpOwotCi0gICAgICAgIFN0cmluZyBzaWduYXR1cmUgPSAiIjsKLSAgICAgICAgaW50IG51bUFyZ3MgPSBqZnVuYy5nZXROdW1BcmdzKCk7Ci0gICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbnVtQXJnczsgaSsrKSB7Ci0gICAgICAgICAgICBKVHlwZSBhcmdUeXBlID0gamZ1bmMuZ2V0QXJnVHlwZShpKTsKLSAgICAgICAgICAgIHNpZ25hdHVyZSArPSBnZXRKbmlOYW1lKGFyZ1R5cGUpOwotICAgICAgICB9Ci0gICAgICAgIGlmIChpc1BvaW50ZXJGdW5jKSB7Ci0gICAgICAgICAgICBzaWduYXR1cmUgKz0gIkkiOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gQXBwZW5kIHNpZ25hdHVyZSB0byBmdW5jdGlvbiBuYW1lCi0gICAgICAgIFN0cmluZyBzaWcgPSBnZXRKbmlNYW5nbGVkTmFtZShzaWduYXR1cmUpLnJlcGxhY2UoJy4nLCAnXycpOyAgICAgICAgCi0gICAgICAgIG91dC5wcmludCgiX18iICsgc2lnKTsKLSAgICAgICAgb3V0TmFtZSArPSAiX18iICsgc2lnOwotCQotICAgICAgICBzaWduYXR1cmUgPSBzaWduYXR1cmUucmVwbGFjZSgnLicsICcvJyk7Ci0gICAgICAgIHJzaWduYXR1cmUgPSByc2lnbmF0dXJlLnJlcGxhY2UoJy4nLCAnLycpOwotCQotICAgICAgICBvdXQucHJpbnRsbigpOwotICAgICAgICBpZiAocnNpZ25hdHVyZS5sZW5ndGgoKSA9PSAwKSB7Ci0gICAgICAgICAgICByc2lnbmF0dXJlID0gIlYiOwotICAgICAgICB9Ci0KLSAgICAgICAgU3RyaW5nIHMgPSAie1wiIiArCi0gICAgICAgICAgICBqZnVuYy5nZXROYW1lKCkgKwotICAgICAgICAgICAgKGlzUG9pbnRlckZ1bmMgPyAiQm91bmRzIiA6ICIiKSArCi0gICAgICAgICAgICAiXCIsIFwiKCIgKyBzaWduYXR1cmUgKyIpIiArCi0gICAgICAgICAgICByc2lnbmF0dXJlICsKLSAgICAgICAgICAgICJcIiwgKHZvaWQgKikgIiArCi0gICAgICAgICAgICBvdXROYW1lICsKLSAgICAgICAgICAgICIgfSwiOwotICAgICAgICBuYXRpdmVSZWdpc3RyYXRpb25zLmFkZChzKTsKLQotICAgICAgICBMaXN0PEludGVnZXI+IG5vblByaW1pdGl2ZUFyZ3MgPSBuZXcgQXJyYXlMaXN0PEludGVnZXI+KCk7Ci0gICAgICAgIGludCBudW1CdWZmZXJBcmdzID0gMDsKLSAgICAgICAgTGlzdDxTdHJpbmc+IGJ1ZmZlckFyZ05hbWVzID0gbmV3IEFycmF5TGlzdDxTdHJpbmc+KCk7Ci0KLSAgICAgICAgLy8gRW1pdCBKTkkgc2lnbmF0dXJlIChhcmd1bWVudHMpCi0gICAgICAgIC8vCi0gICAgICAgIC8vIEV4YW1wbGU6Ci0gICAgICAgIC8vCi0gICAgICAgIC8vIChKTklFbnYgKl9lbnYsIGpvYmplY3QgdGhpcywgamludCBwbGFuZSwgamZsb2F0QXJyYXkgZXF1YXRpb25fcmVmLCBqaW50IG9mZnNldCkgewotICAgICAgICAvLwotICAgICAgICBvdXQucHJpbnQoIiAgKEpOSUVudiAqX2Vudiwgam9iamVjdCBfdGhpcyIpOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUFyZ3M7IGkrKykgewotICAgICAgICAgICAgb3V0LnByaW50KCIsICIpOwotICAgICAgICAgICAgSlR5cGUgYXJnVHlwZSA9IGpmdW5jLmdldEFyZ1R5cGUoaSk7Ci0gICAgICAgICAgICBTdHJpbmcgc3VmZml4OwotICAgICAgICAgICAgaWYgKCFhcmdUeXBlLmlzUHJpbWl0aXZlKCkpIHsKLSAgICAgICAgICAgICAgICBpZiAoYXJnVHlwZS5pc0FycmF5KCkpIHsKLSAgICAgICAgICAgICAgICAgICAgc3VmZml4ID0gIl9yZWYiOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIHN1ZmZpeCA9ICJfYnVmIjsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgbm9uUHJpbWl0aXZlQXJncy5hZGQobmV3IEludGVnZXIoaSkpOwotICAgICAgICAgICAgICAgIGlmIChqZnVuYy5nZXRBcmdUeXBlKGkpLmlzQnVmZmVyKCkpIHsKLSAgICAgICAgICAgICAgICAgICAgaW50IGNJbmRleCA9IGpmdW5jLmdldEFyZ0NJbmRleChpKTsKLSAgICAgICAgICAgICAgICAgICAgU3RyaW5nIGNuYW1lID0gY2Z1bmMuZ2V0QXJnTmFtZShjSW5kZXgpOwotICAgICAgICAgICAgICAgICAgICBidWZmZXJBcmdOYW1lcy5hZGQoY25hbWUpOwotICAgICAgICAgICAgICAgICAgICBudW1CdWZmZXJBcmdzKys7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICBzdWZmaXggPSAiIjsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgb3V0LnByaW50KGdldEpuaVR5cGUoYXJnVHlwZSkgKyAiICIgKyBqZnVuYy5nZXRBcmdOYW1lKGkpICsgc3VmZml4KTsKLSAgICAgICAgfQotICAgICAgICBpZiAoaXNQb2ludGVyRnVuYykgewotICAgICAgICAgICAgb3V0LnByaW50KCIsIGppbnQgcmVtYWluaW5nIik7Ci0gICAgICAgIH0KLSAgICAgICAgb3V0LnByaW50bG4oIikgeyIpOwotCQotICAgICAgICBpbnQgbnVtQXJyYXlzID0gMDsKLSAgICAgICAgaW50IG51bUJ1ZmZlcnMgPSAwOwotICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG5vblByaW1pdGl2ZUFyZ3Muc2l6ZSgpOyBpKyspIHsKLSAgICAgICAgICAgIGludCBpZHggPSBub25QcmltaXRpdmVBcmdzLmdldChpKS5pbnRWYWx1ZSgpOwotICAgICAgICAgICAgaW50IGNJbmRleCA9IGpmdW5jLmdldEFyZ0NJbmRleChpZHgpOwotICAgICAgICAgICAgU3RyaW5nIGNuYW1lID0gY2Z1bmMuZ2V0QXJnTmFtZShjSW5kZXgpOwotICAgICAgICAgICAgaWYgKGpmdW5jLmdldEFyZ1R5cGUoaWR4KS5pc0FycmF5KCkpIHsKLSAgICAgICAgICAgICAgICArK251bUFycmF5czsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIGlmIChqZnVuYy5nZXRBcmdUeXBlKGlkeCkuaXNCdWZmZXIoKSkgewotICAgICAgICAgICAgICAgICsrbnVtQnVmZmVyczsKLSAgICAgICAgICAgIH0KLSAgICAgICAgfQotCi0gICAgICAgIC8vIEVtaXQgbWV0aG9kIGJvZHkKLQotICAgICAgICAvLyBFbWl0IGxvY2FsIHZhcmlhYmxlIGRlY2xhcmF0aW9ucyBmb3IgX2V4Y2VwdGlvbiBhbmQgX3JldHVyblZhbHVlCi0gICAgICAgIC8vCi0gICAgICAgIC8vIEV4YW1wbGU6Ci0gICAgICAgIC8vCi0gICAgICAgIC8vIGFuZHJvaWQ6OmdsOjpvZ2xlc19jb250ZXh0X3QgKmN0eDsKLSAgICAgICAgLy8gCi0gICAgICAgIC8vIGppbnQgX2V4Y2VwdGlvbjsKLSAgICAgICAgLy8gR0xlbnVtIF9yZXR1cm5WYWx1ZTsKLSAgICAgICAgLy8KLSAgICAgICAgQ1R5cGUgcmV0dXJuVHlwZSA9IGNmdW5jLmdldFR5cGUoKTsKLSAgICAgICAgYm9vbGVhbiBpc1ZvaWQgPSByZXR1cm5UeXBlLmlzVm9pZCgpOwotCi0gICAgICAgIGJvb2xlYW4gaXNVbnN1cHBvcnRlZCA9IGlzVW5zdXBwb3J0ZWRGdW5jKGNmdW5jKTsKLSAgICAgICAgaWYgKGlzVW5zdXBwb3J0ZWQpIHsKLSAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgICAgICAiX2Vudi0+VGhyb3dOZXcoVU9FQ2xhc3MsIik7Ci0gICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKwotICAgICAgICAgICAgICAgICAgICAgICAgIiAgICBcIiIgKyBjZnVuYy5nZXROYW1lKCkgKyAiXCIpOyIpOwotICAgICAgICAgICAgaWYgKCFpc1ZvaWQpIHsKLSAgICAgICAgICAgICAgICBTdHJpbmcgcmV0dmFsID0gZ2V0RXJyb3JSZXR1cm5WYWx1ZShjZnVuYyk7Ci0gICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgInJldHVybiAiICsgcmV0dmFsICsgIjsiKTsKLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG91dC5wcmludGxuKCJ9Iik7Ci0gICAgICAgICAgICBvdXQucHJpbnRsbigpOwotICAgICAgICAgICAgcmV0dXJuOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG1Vc2VDb250ZXh0UG9pbnRlcikgewotICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsKLSAgICAgICAgICAgICAgICAiYW5kcm9pZDo6Z2w6Om9nbGVzX2NvbnRleHRfdCAqY3R4ID0gZ2V0Q29udGV4dChfZW52LCBfdGhpcyk7Iik7Ci0gICAgICAgIH0KLQotICAgICAgICBib29sZWFuIGVtaXRFeGNlcHRpb25DaGVjayA9IChudW1BcnJheXMgPiAwIHx8IG51bUJ1ZmZlcnMgPiAwKSAmJgotICAgICAgICAgICAgaGFzTm9uQ29uc3RBcmcoamZ1bmMsIGNmdW5jLCBub25QcmltaXRpdmVBcmdzKTsKLSAgICAgICAgLy8gbUNoZWNrZXIuZ2V0Q2hlY2tzKGNmdW5jLmdldE5hbWUoKSkgIT0gbnVsbAotCi0gICAgICAgIC8vIEVtaXQgYW4gX2V4ZXB0aW9uIHZhcmlhYmxlIGlmIHRoZXJlIHdpbGwgYmUgZXJyb3IgY2hlY2tzCi0gICAgICAgIGlmIChlbWl0RXhjZXB0aW9uQ2hlY2spIHsKLSAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArICJqaW50IF9leGNlcHRpb24gPSAwOyIpOwotICAgICAgICB9Ci0KLSAgICAgICAgLy8gRW1pdCBhIHNpbmdsZSBfYXJyYXkgb3IgbXVsdGlwbGUgX1hYWEFycmF5IHZhcmlhYmxlcwotICAgICAgICBpZiAobnVtQnVmZmVyQXJncyA9PSAxKSB7Ci0gICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgImphcnJheSBfYXJyYXkgPSAoamFycmF5KSAwOyIpOwotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCBudW1CdWZmZXJBcmdzOyBpKyspIHsKLSAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyAiamFycmF5IF8iICsgYnVmZmVyQXJnTmFtZXMuZ2V0KGkpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAiQXJyYXkgPSAoamFycmF5KSAwOyIpOwotICAgICAgICAgICAgfQotICAgICAgICB9Ci0gICAgICAgIGlmICghaXNWb2lkKSB7Ci0gICAgICAgICAgICBTdHJpbmcgcmV0dmFsID0gZ2V0RXJyb3JSZXR1cm5WYWx1ZShjZnVuYyk7Ci0gICAgICAgICAgICBpZiAocmV0dmFsICE9IG51bGwpIHsKLSAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyByZXR1cm5UeXBlLmdldERlY2xhcmF0aW9uKCkgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICIgX3JldHVyblZhbHVlID0gIiArIHJldHZhbCArICI7Iik7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArIHJldHVyblR5cGUuZ2V0RGVjbGFyYXRpb24oKSArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgIiBfcmV0dXJuVmFsdWU7Iik7Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICAvLyBFbWl0IGxvY2FsIHZhcmlhYmxlIGRlY2xhcmF0aW9ucyBmb3IgcG9pbnRlciBhcmd1bWVudHMKLSAgICAgICAgLy8KLSAgICAgICAgLy8gRXhhbXBsZToKLSAgICAgICAgLy8KLSAgICAgICAgLy8gR0xmaXhlZCAqZXFuX2Jhc2U7Ci0gICAgICAgIC8vIEdMZml4ZWQgKmVxbjsKLSAgICAgICAgLy8KLSAgICAgICAgU3RyaW5nIG9mZnNldCA9ICJvZmZzZXQiOwotICAgICAgICBTdHJpbmcgcmVtYWluaW5nID0gIl9yZW1haW5pbmciOwotICAgICAgICBpZiAobm9uUHJpbWl0aXZlQXJncy5zaXplKCkgPiAwKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG5vblByaW1pdGl2ZUFyZ3Muc2l6ZSgpOyBpKyspIHsKLSAgICAgICAgICAgICAgICBpbnQgaWR4ID0gbm9uUHJpbWl0aXZlQXJncy5nZXQoaSkuaW50VmFsdWUoKTsKLSAgICAgICAgICAgICAgICBpbnQgY0luZGV4ID0gamZ1bmMuZ2V0QXJnQ0luZGV4KGlkeCk7Ci0gICAgICAgICAgICAgICAgU3RyaW5nIGNuYW1lID0gY2Z1bmMuZ2V0QXJnTmFtZShjSW5kZXgpOwotCi0gICAgICAgICAgICAgICAgQ1R5cGUgdHlwZSA9IGNmdW5jLmdldEFyZ1R5cGUoamZ1bmMuZ2V0QXJnQ0luZGV4KGlkeCkpOwotICAgICAgICAgICAgICAgIFN0cmluZyBkZWNsID0gdHlwZS5nZXREZWNsYXJhdGlvbigpOwotICAgICAgICAgICAgICAgIGlmIChqZnVuYy5nZXRBcmdUeXBlKGlkeCkuaXNBcnJheSgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRlY2wgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGVjbC5lbmRzV2l0aCgiKiIpID8gIiIgOiAiICIpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgamZ1bmMuZ2V0QXJnTmFtZShpZHgpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIl9iYXNlID0gKCIgKyBkZWNsICsgIikgMDsiKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgcmVtYWluaW5nID0gKG51bUFycmF5cyA8PSAxICYmIG51bUJ1ZmZlcnMgPD0gMSkgPyAiX3JlbWFpbmluZyIgOgotICAgICAgICAgICAgICAgICAgICAiXyIgKyBjbmFtZSArICJSZW1haW5pbmciOwotICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgImppbnQgIiArIHJlbWFpbmluZyArICI7Iik7Ci0gICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBkZWNsICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGVjbC5lbmRzV2l0aCgiKiIpID8gIiIgOiAiICIpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICBqZnVuYy5nZXRBcmdOYW1lKGlkeCkgKyAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAiID0gKCIgKyBkZWNsICsgIikgMDsiKTsKLSAgICAgICAgICAgIH0KLQotICAgICAgICAgICAgb3V0LnByaW50bG4oKTsKLSAgICAgICAgfQotCi0gICAgICAgIFN0cmluZyByZXR2YWwgPSBpc1ZvaWQgPyAiIiA6ICIgX3JldHVyblZhbHVlIjsKLQotICAgICAgICAvLyBFbWl0ICdHZXRQcmltaXRpdmVBcnJheUNyaXRpY2FsJyBmb3IgYXJyYXlzCi0gICAgICAgIC8vIEVtaXQgJ0dldFBvaW50ZXInIGNhbGxzIGZvciBCdWZmZXIgcG9pbnRlcnMKLSAgICAgICAgaW50IGJ1ZkFyZ0lkeCA9IDA7Ci0gICAgICAgIGlmIChub25QcmltaXRpdmVBcmdzLnNpemUoKSA+IDApIHsKLSAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgbm9uUHJpbWl0aXZlQXJncy5zaXplKCk7IGkrKykgewotICAgICAgICAgICAgICAgIGludCBpZHggPSBub25QcmltaXRpdmVBcmdzLmdldChpKS5pbnRWYWx1ZSgpOwotICAgICAgICAgICAgICAgIGludCBjSW5kZXggPSBqZnVuYy5nZXRBcmdDSW5kZXgoaWR4KTsKLQkJCi0gICAgICAgICAgICAgICAgU3RyaW5nIGNuYW1lID0gY2Z1bmMuZ2V0QXJnTmFtZShjSW5kZXgpOwotICAgICAgICAgICAgICAgIG9mZnNldCA9IG51bUFycmF5cyA8PSAxID8gIm9mZnNldCIgOgotICAgICAgICAgICAgICAgICAgICBjbmFtZSArICJPZmZzZXQiOwotICAgICAgICAgICAgICAgIHJlbWFpbmluZyA9IChudW1BcnJheXMgPD0gMSAmJiBudW1CdWZmZXJzIDw9IDEpID8gIl9yZW1haW5pbmciIDoKLSAgICAgICAgICAgICAgICAgICAgIl8iICsgY25hbWUgKyAiUmVtYWluaW5nIjsKLQotICAgICAgICAgICAgICAgIGlmIChqZnVuYy5nZXRBcmdUeXBlKGlkeCkuaXNBcnJheSgpKSB7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJpZiAoISIgKyAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY25hbWUgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiX3JlZikgeyIpOwotICAgICAgICAgICAgICAgICAgICBpZiAoZW1pdEV4Y2VwdGlvbkNoZWNrKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyBpbmRlbnQgKyAiX2V4Y2VwdGlvbiA9IDE7Iik7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgIiAgICAiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKG1Vc2VDUGx1c1BsdXMgPyAiX2VudiIgOiAiKCpfZW52KSIpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIi0+VGhyb3dOZXcoIiArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChtVXNlQ1BsdXNQbHVzID8gIiIgOiAiX2VudiwgIikgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiSUFFQ2xhc3MsICIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiXCIiICsgY25hbWUgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiID09IG51bGxcIik7Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArICIgICAgZ290byBleGl0OyIpOwotICAgICAgICAgICAgICAgICAgICBuZWVkc0V4aXQgPSB0cnVlOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyAifSIpOwotCi0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArICJpZiAoIiArIG9mZnNldCArICIgPCAwKSB7Iik7Ci0gICAgICAgICAgICAgICAgICAgIGlmIChlbWl0RXhjZXB0aW9uQ2hlY2spIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArIGluZGVudCArICJfZXhjZXB0aW9uID0gMTsiKTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyAiICAgICIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAobVVzZUNQbHVzUGx1cyA/ICJfZW52IiA6ICIoKl9lbnYpIikgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiLT5UaHJvd05ldygiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKG1Vc2VDUGx1c1BsdXMgPyAiIiA6ICJfZW52LCAiKSArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJJQUVDbGFzcywgIiArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJcIiIgKyBvZmZzZXQgKyAiIDwgMFwiKTsiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgIiAgICBnb3RvIGV4aXQ7Iik7Ci0gICAgICAgICAgICAgICAgICAgIG5lZWRzRXhpdCA9IHRydWU7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArICJ9Iik7Ci0KLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgcmVtYWluaW5nICsgIiA9ICIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKG1Vc2VDUGx1c1BsdXMgPyAiX2VudiIgOiAiKCpfZW52KSIpICsgCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiLT5HZXRBcnJheUxlbmd0aCgiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChtVXNlQ1BsdXNQbHVzID8gIiIgOiAiX2VudiwgIikgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY25hbWUgKyAiX3JlZikgLSAiICsgb2Zmc2V0ICsgIjsiKTsKLQotICAgICAgICAgICAgICAgICAgICBlbWl0TmF0aXZlQm91bmRzQ2hlY2tzKGNmdW5jLCBjbmFtZSwgb3V0LCBmYWxzZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBlbWl0RXhjZXB0aW9uQ2hlY2ssCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0LCByZW1haW5pbmcsICIgICAgIik7Ci0KLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY25hbWUgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiX2Jhc2UgPSAoIiArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNmdW5jLmdldEFyZ1R5cGUoY0luZGV4KS5nZXREZWNsYXJhdGlvbigpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIikiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgIiAgICAiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKG1Vc2VDUGx1c1BsdXMgPyAiX2VudiIgOiAiKCpfZW52KSIpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIi0+R2V0UHJpbWl0aXZlQXJyYXlDcml0aWNhbCgiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKG1Vc2VDUGx1c1BsdXMgPyAiIiA6ICJfZW52LCAiKSArIAotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBqZnVuYy5nZXRBcmdOYW1lKGlkeCkgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiX3JlZiwgKGpib29sZWFuICopMCk7Iik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNuYW1lICsgIiA9ICIgKyBjbmFtZSArICJfYmFzZSArICIgKyBvZmZzZXQgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiOyIpOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbigpOwotICAgICAgICAgICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICAgICAgICAgIFN0cmluZyBhcnJheSA9IG51bUJ1ZmZlckFyZ3MgPD0gMSA/ICJfYXJyYXkiIDoKLSAgICAgICAgICAgICAgICAgICAgICAgICJfIiArIGJ1ZmZlckFyZ05hbWVzLmdldChidWZBcmdJZHgrKykgKyAiQXJyYXkiOwotCi0gICAgICAgICAgICAgICAgICAgIGJvb2xlYW4gbnVsbEFsbG93ZWQgPSBpc051bGxBbGxvd2VkKGNmdW5jKTsKLSAgICAgICAgICAgICAgICAgICAgaWYgKG51bGxBbGxvd2VkKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyAiaWYgKCIgKyBjbmFtZSArICJfYnVmKSB7Iik7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnQoaW5kZW50KTsKLSAgICAgICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgICAgICAgICAKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY25hbWUgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiID0gKCIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjZnVuYy5nZXRBcmdUeXBlKGNJbmRleCkuZ2V0RGVjbGFyYXRpb24oKSArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICIpZ2V0UG9pbnRlcihfZW52LCAiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY25hbWUgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiX2J1ZiwgJiIgKyBhcnJheSArICIsICYiICsgcmVtYWluaW5nICsgIik7Iik7Ci0KLSAgICAgICAgICAgICAgICAgICAgaWYgKG51bGxBbGxvd2VkKSB7Ci0gICAgICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyAifSIpOwotICAgICAgICAgICAgICAgICAgICB9Ci0KLSAgICAgICAgICAgICAgICAgICAgZW1pdE5hdGl2ZUJvdW5kc0NoZWNrcyhjZnVuYywgY25hbWUsIG91dCwgdHJ1ZSwKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBlbWl0RXhjZXB0aW9uQ2hlY2ssCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0LCByZW1haW5pbmcsICIgICAgIik7Ci0gICAgICAgICAgICAgICAgfQotICAgICAgICAgICAgfQotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKCFpc1ZvaWQpIHsKLSAgICAgICAgICAgIG91dC5wcmludChpbmRlbnQgKyAiX3JldHVyblZhbHVlID0gIik7Ci0gICAgICAgIH0gZWxzZSB7Ci0gICAgICAgICAgICBvdXQucHJpbnQoaW5kZW50KTsKLSAgICAgICAgfQotICAgICAgICBTdHJpbmcgbmFtZSA9IGNmdW5jLmdldE5hbWUoKTsKLQotICAgICAgICBpZiAobVVzZUNvbnRleHRQb2ludGVyKSB7Ci0gICAgICAgICAgICBuYW1lID0gbmFtZS5zdWJzdHJpbmcoMiwgbmFtZS5sZW5ndGgoKSk7IC8vIFN0cmlwIG9mZiAnZ2wnIHByZWZpeAotICAgICAgICAgICAgbmFtZSA9IG5hbWUuc3Vic3RyaW5nKDAsIDEpLnRvTG93ZXJDYXNlKCkgKwotICAgICAgICAgICAgICAgIG5hbWUuc3Vic3RyaW5nKDEsIG5hbWUubGVuZ3RoKCkpOwotICAgICAgICAgICAgb3V0LnByaW50KCJjdHgtPnByb2NzLiIpOwotICAgICAgICB9Ci0gICAgICAgIAotICAgICAgICBvdXQucHJpbnQobmFtZSArIChpc1BvaW50ZXJGdW5jID8gIkJvdW5kcyIgOiAiIikgKyAiKCIpOwotCi0gICAgICAgIG51bUFyZ3MgPSBjZnVuYy5nZXROdW1BcmdzKCk7ICAgIAotICAgICAgICBpZiAobnVtQXJncyA9PSAwKSB7Ci0gICAgICAgICAgICBpZiAobVVzZUNvbnRleHRQb2ludGVyKSB7Ci0gICAgICAgICAgICAgICAgb3V0LnByaW50bG4oImN0eCk7Iik7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIG91dC5wcmludGxuKCIpOyIpOwotICAgICAgICAgICAgfQotICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgaWYgKG1Vc2VDb250ZXh0UG9pbnRlcikgewotICAgICAgICAgICAgICAgIG91dC5wcmludGxuKCJjdHgsIik7Ci0gICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgIG91dC5wcmludGxuKCk7Ci0gICAgICAgICAgICB9Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gMDsgaSA8IG51bUFyZ3M7IGkrKykgewotICAgICAgICAgICAgICAgIFN0cmluZyB0eXBlY2FzdDsKLSAgICAgICAgICAgICAgICBpZiAoaSA9PSBudW1BcmdzIC0gMSAmJiBpc1ZCT1BvaW50ZXJGdW5jKSB7Ci0gICAgICAgICAgICAgICAgICAgIHR5cGVjYXN0ID0gImNvbnN0IEdMdm9pZCAqIjsKLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICB0eXBlY2FzdCA9IGNmdW5jLmdldEFyZ1R5cGUoaSkuZ2V0RGVjbGFyYXRpb24oKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICAgICAgb3V0LnByaW50KGluZGVudCArIGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICIoIiArCi0gICAgICAgICAgICAgICAgICAgICAgICAgIHR5cGVjYXN0ICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgIikiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgY2Z1bmMuZ2V0QXJnTmFtZShpKSk7Ci0KLSAgICAgICAgICAgICAgICBpZiAoaSA9PSBudW1BcmdzIC0gMSkgewotICAgICAgICAgICAgICAgICAgICBpZiAoaXNQb2ludGVyRnVuYykgewotICAgICAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oIiwiKTsKLSAgICAgICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArIGluZGVudCArICIoR0xzaXplaSlyZW1haW5pbmciKTsKLSAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKLSAgICAgICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKCk7Ci0gICAgICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgICAgICB9IGVsc2UgewotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbigiLCIpOwotICAgICAgICAgICAgICAgIH0KLSAgICAgICAgICAgIH0KLSAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArICIpOyIpOwotICAgICAgICB9Ci0KLSAgICAgICAgaWYgKG5lZWRzRXhpdCkgewotICAgICAgICAgICAgb3V0LnByaW50bG4oKTsKLSAgICAgICAgICAgIG91dC5wcmludGxuKCJleGl0OiIpOwotICAgICAgICAgICAgbmVlZHNFeGl0ID0gZmFsc2U7Ci0gICAgICAgIH0KLQotICAgICAgICBidWZBcmdJZHggPSAwOwotICAgICAgICBpZiAobm9uUHJpbWl0aXZlQXJncy5zaXplKCkgPiAwKSB7Ci0gICAgICAgICAgICBmb3IgKGludCBpID0gbm9uUHJpbWl0aXZlQXJncy5zaXplKCkgLSAxOyBpID49IDA7IGktLSkgewotICAgICAgICAgICAgICAgIGludCBpZHggPSBub25QcmltaXRpdmVBcmdzLmdldChpKS5pbnRWYWx1ZSgpOwotCi0gICAgICAgICAgICAgICAgaW50IGNJbmRleCA9IGpmdW5jLmdldEFyZ0NJbmRleChpZHgpOwotICAgICAgICAgICAgICAgIGlmIChqZnVuYy5nZXRBcmdUeXBlKGlkeCkuaXNBcnJheSgpKSB7Ci0JCSAgICAKLSAgICAgICAgICAgICAgICAgICAgLy8gSWYgdGhlIGFyZ3VtZW50IGlzICdjb25zdCcsIEdMIHdpbGwgbm90IHdyaXRlIHRvIGl0LgotICAgICAgICAgICAgICAgICAgICAvLyBJbiB0aGlzIGNhc2UsIHdlIGNhbiB1c2UgdGhlICdKTklfQUJPUlQnIGZsYWcgdG8gYXZvaWQKLSAgICAgICAgICAgICAgICAgICAgLy8gdGhlIG5lZWQgdG8gd3JpdGUgYmFjayB0byB0aGUgSmF2YSBhcnJheQotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiaWYgKCIgKyBqZnVuYy5nZXRBcmdOYW1lKGlkeCkgKyAiX2Jhc2UpIHsiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgaW5kZW50ICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKG1Vc2VDUGx1c1BsdXMgPyAiX2VudiIgOiAiKCpfZW52KSIpICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIi0+UmVsZWFzZVByaW1pdGl2ZUFycmF5Q3JpdGljYWwoIiArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChtVXNlQ1BsdXNQbHVzID8gIiIgOiAiX2VudiwgIikgKyAKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgamZ1bmMuZ2V0QXJnTmFtZShpZHgpICsgIl9yZWYsICIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjZnVuYy5nZXRBcmdOYW1lKGNJbmRleCkgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiX2Jhc2UsIik7Ci0gICAgICAgICAgICAgICAgICAgIG91dC5wcmludGxuKGluZGVudCArIGluZGVudCArIGluZGVudCArCi0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChjZnVuYy5nZXRBcmdUeXBlKGNJbmRleCkuaXNDb25zdCgpID8KLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJKTklfQUJPUlQiIDoKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJfZXhjZXB0aW9uID8gSk5JX0FCT1JUOiAwIikgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiKTsiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgIn0iKTsKLSAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGpmdW5jLmdldEFyZ1R5cGUoaWR4KS5pc0J1ZmZlcigpKSB7Ci0gICAgICAgICAgICAgICAgICAgIFN0cmluZyBhcnJheSA9IG51bUJ1ZmZlckFyZ3MgPD0gMSA/ICJfYXJyYXkiIDoKLSAgICAgICAgICAgICAgICAgICAgICAgICJfIiArIGJ1ZmZlckFyZ05hbWVzLmdldChidWZBcmdJZHgrKykgKyAiQXJyYXkiOwotICAgICAgICAgICAgICAgICAgICBvdXQucHJpbnRsbihpbmRlbnQgKyAiaWYgKCIgKyBhcnJheSArICIpIHsiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgaW5kZW50ICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgInJlbGVhc2VQb2ludGVyKF9lbnYsICIgKyBhcnJheSArICIsICIgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjZnVuYy5nZXRBcmdOYW1lKGNJbmRleCkgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiLCAiICsKLSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGNmdW5jLmdldEFyZ1R5cGUoY0luZGV4KS5pc0NvbnN0KCkgPwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIkpOSV9GQUxTRSIgOiAiX2V4Y2VwdGlvbiA/IEpOSV9GQUxTRSA6IEpOSV9UUlVFIikgKwotICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiKTsiKTsKLSAgICAgICAgICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgIn0iKTsKLSAgICAgICAgICAgICAgICB9Ci0gICAgICAgICAgICB9Ci0gICAgICAgIH0KLQotICAgICAgICBpZiAoIWlzVm9pZCkgewotICAgICAgICAgICAgb3V0LnByaW50bG4oaW5kZW50ICsgInJldHVybiBfcmV0dXJuVmFsdWU7Iik7Ci0gICAgICAgIH0KLQotICAgICAgICBvdXQucHJpbnRsbigifSIpOwotICAgICAgICBvdXQucHJpbnRsbigpOwotICAgIH0KLQotICAgIHB1YmxpYyB2b2lkIGFkZE5hdGl2ZVJlZ2lzdHJhdGlvbihTdHJpbmcgcykgewotICAgICAgICBuYXRpdmVSZWdpc3RyYXRpb25zLmFkZChzKTsKLSAgICB9Ci0KLSAgICBwdWJsaWMgdm9pZCBlbWl0TmF0aXZlUmVnaXN0cmF0aW9uKCkgewotICAgICAgICBtQ1N0cmVhbS5wcmludGxuKCJzdGF0aWMgY29uc3QgY2hhciAqY2xhc3NQYXRoTmFtZSA9IFwiIiArCi0gICAgICAgICAgICAgICAgICAgICAgICBtQ2xhc3NQYXRoTmFtZSArCi0gICAgICAgICAgICAgICAgICAgICAgICAiXCI7Iik7Ci0gICAgICAgIG1DU3RyZWFtLnByaW50bG4oKTsKLQotICAgICAgICBtQ1N0cmVhbS5wcmludGxuKCJzdGF0aWMgSk5JTmF0aXZlTWV0aG9kIG1ldGhvZHNbXSA9IHsiKTsKLQotICAgICAgICBtQ1N0cmVhbS5wcmludGxuKCJ7XCJfbmF0aXZlQ2xhc3NJbml0XCIsIFwiKClWXCIsICh2b2lkKiluYXRpdmVDbGFzc0luaXQgfSwiKTsKLQotICAgICAgICBJdGVyYXRvcjxTdHJpbmc+IGkgPSBuYXRpdmVSZWdpc3RyYXRpb25zLml0ZXJhdG9yKCk7Ci0gICAgICAgIHdoaWxlIChpLmhhc05leHQoKSkgewotICAgICAgICAgICAgbUNTdHJlYW0ucHJpbnRsbihpLm5leHQoKSk7Ci0gICAgICAgIH0KLQotICAgICAgICBtQ1N0cmVhbS5wcmludGxuKCJ9OyIpOwotICAgICAgICBtQ1N0cmVhbS5wcmludGxuKCk7Ci0gICAgCi0KLSAgICAgICAgbUNTdHJlYW0ucHJpbnRsbigiaW50IHJlZ2lzdGVyX2NvbV9nb29nbGVfYW5kcm9pZF9nbGVzX2puaV9HTEltcGwoSk5JRW52ICpfZW52KSIpOwotICAgICAgICBtQ1N0cmVhbS5wcmludGxuKCJ7Iik7Ci0gICAgICAgIG1DU3RyZWFtLnByaW50bG4oaW5kZW50ICsKLSAgICAgICAgICAgICAgICAgICAgICAgICJpbnQgZXJyOyIpOwotCi0gICAgICAgIG1DU3RyZWFtLnByaW50bG4oaW5kZW50ICsKLSAgICAgICAgICAgICAgICAgICAgICAgICJlcnIgPSBhbmRyb2lkOjpBbmRyb2lkUnVudGltZTo6cmVnaXN0ZXJOYXRpdmVNZXRob2RzKF9lbnYsIGNsYXNzUGF0aE5hbWUsIG1ldGhvZHMsIE5FTEVNKG1ldGhvZHMpKTsiKTsKLQotICAgICAgICBtQ1N0cmVhbS5wcmludGxuKGluZGVudCArICJyZXR1cm4gZXJyOyIpOwotICAgICAgICBtQ1N0cmVhbS5wcmludGxuKCJ9Iik7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3NyYy9QYXJhbWV0ZXJDaGVja2VyLmphdmEgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL1BhcmFtZXRlckNoZWNrZXIuamF2YQpkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGYyNmFjZC4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3JjL1BhcmFtZXRlckNoZWNrZXIuamF2YQorKysgL2Rldi9udWxsCkBAIC0xLDI4ICswLDAgQEAKLQotaW1wb3J0IGphdmEuaW8uQnVmZmVyZWRSZWFkZXI7Ci1pbXBvcnQgamF2YS51dGlsLkhhc2hNYXA7Ci0KLXB1YmxpYyBjbGFzcyBQYXJhbWV0ZXJDaGVja2VyIHsKLQotICAgIEhhc2hNYXA8U3RyaW5nLFN0cmluZ1tdPiBtYXAgPSBuZXcgSGFzaE1hcDxTdHJpbmcsU3RyaW5nW10+KCk7Ci0KLSAgICBwdWJsaWMgUGFyYW1ldGVyQ2hlY2tlcihCdWZmZXJlZFJlYWRlciByZWFkZXIpIHRocm93cyBFeGNlcHRpb24gewotICAgICAgICBTdHJpbmcgczsKLSAgICAgICAgd2hpbGUgKChzID0gcmVhZGVyLnJlYWRMaW5lKCkpICE9IG51bGwpIHsKLSAgICAgICAgICAgIFN0cmluZ1tdIHRva2VucyA9IHMuc3BsaXQoIlxccyIpOwotICAgICAgICAgICAgbWFwLnB1dCh0b2tlbnNbMF0sIHRva2Vucyk7Ci0gICAgICAgIH0KLSAgICB9Ci0KLSAgICBwdWJsaWMgU3RyaW5nW10gZ2V0Q2hlY2tzKFN0cmluZyBmdW5jdGlvbk5hbWUpIHsKLSAgICAgICAgU3RyaW5nW10gY2hlY2tzID0gbWFwLmdldChmdW5jdGlvbk5hbWUpOwotICAgICAgICBpZiAoY2hlY2tzID09IG51bGwgJiYKLSAgICAgICAgICAgIChmdW5jdGlvbk5hbWUuZW5kc1dpdGgoImZ2IikgfHwKLSAgICAgICAgICAgICBmdW5jdGlvbk5hbWUuZW5kc1dpdGgoInh2IikgfHwKLSAgICAgICAgICAgICBmdW5jdGlvbk5hbWUuZW5kc1dpdGgoIml2IikpKSB7Ci0gICAgICAgICAgICBmdW5jdGlvbk5hbWUgPSBmdW5jdGlvbk5hbWUuc3Vic3RyaW5nKDAsIGZ1bmN0aW9uTmFtZS5sZW5ndGgoKSAtIDIpOwotICAgICAgICAgICAgY2hlY2tzID0gbWFwLmdldChmdW5jdGlvbk5hbWUpOwotICAgICAgICB9Ci0gICAgICAgIHJldHVybiBjaGVja3M7Ci0gICAgfQotfQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL0dMMTBFeHRIZWFkZXIuamF2YS1pZiBiL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTDEwRXh0SGVhZGVyLmphdmEtaWYKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IGIwOTk5YzIuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL0dMMTBFeHRIZWFkZXIuamF2YS1pZgorKysgL2Rldi9udWxsCkBAIC0xLDIyICswLDAgQEAKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNywgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0vLyBUaGlzIHNvdXJjZSBmaWxlIGlzIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGVkCi0KLXBhY2thZ2UgamF2YXgubWljcm9lZGl0aW9uLmtocm9ub3Mub3BlbmdsZXM7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgR0wxMEV4dCBleHRlbmRzIEdMIHsKLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL0dMMTBIZWFkZXIuamF2YS1pZiBiL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTDEwSGVhZGVyLmphdmEtaWYKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDgzOTI4MjEuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL0dMMTBIZWFkZXIuamF2YS1pZgorKysgL2Rldi9udWxsCkBAIC0xLDI1OSArMCwwIEBACi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotLy8gVGhpcyBzb3VyY2UgZmlsZSBpcyBhdXRvbWF0aWNhbGx5IGdlbmVyYXRlZAotCi1wYWNrYWdlIGphdmF4Lm1pY3JvZWRpdGlvbi5raHJvbm9zLm9wZW5nbGVzOwotCi1wdWJsaWMgaW50ZXJmYWNlIEdMMTAgZXh0ZW5kcyBHTCB7Ci0gICAgaW50IEdMX0FERCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAxMDQ7Ci0gICAgaW50IEdMX0FMSUFTRURfTElORV9XSURUSF9SQU5HRSAgICAgICAgICAgICAgPSAweDg0NkU7Ci0gICAgaW50IEdMX0FMSUFTRURfUE9JTlRfU0laRV9SQU5HRSAgICAgICAgICAgICAgPSAweDg0NkQ7Ci0gICAgaW50IEdMX0FMUEhBICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE5MDY7Ci0gICAgaW50IEdMX0FMUEhBX0JJVFMgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBENTU7Ci0gICAgaW50IEdMX0FMUEhBX1RFU1QgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCQzA7Ci0gICAgaW50IEdMX0FMV0FZUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAyMDc7Ci0gICAgaW50IEdMX0FNQklFTlQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDEyMDA7Ci0gICAgaW50IEdMX0FNQklFTlRfQU5EX0RJRkZVU0UgICAgICAgICAgICAgICAgICAgPSAweDE2MDI7Ci0gICAgaW50IEdMX0FORCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MDE7Ci0gICAgaW50IEdMX0FORF9JTlZFUlRFRCAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MDQ7Ci0gICAgaW50IEdMX0FORF9SRVZFUlNFICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MDI7Ci0gICAgaW50IEdMX0JBQ0sgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDA0MDU7Ci0gICAgaW50IEdMX0JMRU5EICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCRTI7Ci0gICAgaW50IEdMX0JMVUVfQklUUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBENTQ7Ci0gICAgaW50IEdMX0JZVEUgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE0MDA7Ci0gICAgaW50IEdMX0NDVyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDA5MDE7Ci0gICAgaW50IEdMX0NMQU1QX1RPX0VER0UgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgxMkY7Ci0gICAgaW50IEdMX0NMRUFSICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MDA7Ci0gICAgaW50IEdMX0NPTE9SX0FSUkFZICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwNzY7Ci0gICAgaW50IEdMX0NPTE9SX0JVRkZFUl9CSVQgICAgICAgICAgICAgICAgICAgICAgPSAweDQwMDA7Ci0gICAgaW50IEdMX0NPTE9SX0xPR0lDX09QICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCRjI7Ci0gICAgaW50IEdMX0NPTE9SX01BVEVSSUFMICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCNTc7Ci0gICAgaW50IEdMX0NPTVBSRVNTRURfVEVYVFVSRV9GT1JNQVRTICAgICAgICAgICAgPSAweDg2QTM7Ci0gICAgaW50IEdMX0NPTlNUQU5UX0FUVEVOVUFUSU9OICAgICAgICAgICAgICAgICAgPSAweDEyMDc7Ci0gICAgaW50IEdMX0NPUFkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MDM7Ci0gICAgaW50IEdMX0NPUFlfSU5WRVJURUQgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MEM7Ci0gICAgaW50IEdMX0NVTExfRkFDRSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCNDQ7Ci0gICAgaW50IEdMX0NXICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDA5MDA7Ci0gICAgaW50IEdMX0RFQ0FMICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDIxMDE7Ci0gICAgaW50IEdMX0RFQ1IgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDFFMDM7Ci0gICAgaW50IEdMX0RFUFRIX0JJVFMgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBENTY7Ci0gICAgaW50IEdMX0RFUFRIX0JVRkZFUl9CSVQgICAgICAgICAgICAgICAgICAgICAgPSAweDAxMDA7Ci0gICAgaW50IEdMX0RFUFRIX1RFU1QgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCNzE7Ci0gICAgaW50IEdMX0RJRkZVU0UgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDEyMDE7Ci0gICAgaW50IEdMX0RJVEhFUiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCRDA7Ci0gICAgaW50IEdMX0RPTlRfQ0FSRSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDExMDA7Ci0gICAgaW50IEdMX0RTVF9BTFBIQSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAzMDQ7Ci0gICAgaW50IEdMX0RTVF9DT0xPUiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAzMDY7Ci0gICAgaW50IEdMX0VNSVNTSU9OICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE2MDA7Ci0gICAgaW50IEdMX0VRVUFMICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAyMDI7Ci0gICAgaW50IEdMX0VRVUlWICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MDk7Ci0gICAgaW50IEdMX0VYUCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDA4MDA7Ci0gICAgaW50IEdMX0VYUDIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDA4MDE7Ci0gICAgaW50IEdMX0VYVEVOU0lPTlMgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDFGMDM7Ci0gICAgaW50IEdMX0ZBTFNFICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAwOwotICAgIGludCBHTF9GQVNURVNUICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxMTAxOwotICAgIGludCBHTF9GSVhFRCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxNDBDOwotICAgIGludCBHTF9GTEFUICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxRDAwOwotICAgIGludCBHTF9GTE9BVCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxNDA2OwotICAgIGludCBHTF9GT0cgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjYwOwotICAgIGludCBHTF9GT0dfQ09MT1IgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjY2OwotICAgIGludCBHTF9GT0dfREVOU0lUWSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjYyOwotICAgIGludCBHTF9GT0dfRU5EICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjY0OwotICAgIGludCBHTF9GT0dfSElOVCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQzU0OwotICAgIGludCBHTF9GT0dfTU9ERSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjY1OwotICAgIGludCBHTF9GT0dfU1RBUlQgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjYzOwotICAgIGludCBHTF9GUk9OVCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwNDA0OwotICAgIGludCBHTF9GUk9OVF9BTkRfQkFDSyAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwNDA4OwotICAgIGludCBHTF9HRVFVQUwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwMjA2OwotICAgIGludCBHTF9HUkVBVEVSICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwMjA0OwotICAgIGludCBHTF9HUkVFTl9CSVRTICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwRDUzOwotICAgIGludCBHTF9JTVBMRU1FTlRBVElPTl9DT0xPUl9SRUFEX0ZPUk1BVF9PRVMgID0gMHg4QjlCOwotICAgIGludCBHTF9JTVBMRU1FTlRBVElPTl9DT0xPUl9SRUFEX1RZUEVfT0VTICAgID0gMHg4QjlBOwotICAgIGludCBHTF9JTkNSICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxRTAyOwotICAgIGludCBHTF9JTlZBTElEX0VOVU0gICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwNTAwOwotICAgIGludCBHTF9JTlZBTElEX09QRVJBVElPTiAgICAgICAgICAgICAgICAgICAgID0gMHgwNTAyOwotICAgIGludCBHTF9JTlZBTElEX1ZBTFVFICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwNTAxOwotICAgIGludCBHTF9JTlZFUlQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxNTBBOwotICAgIGludCBHTF9LRUVQICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxRTAwOwotICAgIGludCBHTF9MRVFVQUwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwMjAzOwotICAgIGludCBHTF9MRVNTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwMjAxOwotICAgIGludCBHTF9MSUdIVF9NT0RFTF9BTUJJRU5UICAgICAgICAgICAgICAgICAgID0gMHgwQjUzOwotICAgIGludCBHTF9MSUdIVF9NT0RFTF9UV09fU0lERSAgICAgICAgICAgICAgICAgID0gMHgwQjUyOwotICAgIGludCBHTF9MSUdIVDAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg0MDAwOwotICAgIGludCBHTF9MSUdIVDEgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg0MDAxOwotICAgIGludCBHTF9MSUdIVDIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg0MDAyOwotICAgIGludCBHTF9MSUdIVDMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg0MDAzOwotICAgIGludCBHTF9MSUdIVDQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg0MDA0OwotICAgIGludCBHTF9MSUdIVDUgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg0MDA1OwotICAgIGludCBHTF9MSUdIVDYgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg0MDA2OwotICAgIGludCBHTF9MSUdIVDcgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg0MDA3OwotICAgIGludCBHTF9MSUdIVElORyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjUwOwotICAgIGludCBHTF9MSU5FX0xPT1AgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwMDAyOwotICAgIGludCBHTF9MSU5FX1NNT09USCAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjIwOwotICAgIGludCBHTF9MSU5FX1NNT09USF9ISU5UICAgICAgICAgICAgICAgICAgICAgID0gMHgwQzUyOwotICAgIGludCBHTF9MSU5FX1NUUklQICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwMDAzOwotICAgIGludCBHTF9MSU5FQVIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgyNjAxOwotICAgIGludCBHTF9MSU5FQVJfQVRURU5VQVRJT04gICAgICAgICAgICAgICAgICAgID0gMHgxMjA4OwotICAgIGludCBHTF9MSU5FQVJfTUlQTUFQX0xJTkVBUiAgICAgICAgICAgICAgICAgID0gMHgyNzAzOwotICAgIGludCBHTF9MSU5FQVJfTUlQTUFQX05FQVJFU1QgICAgICAgICAgICAgICAgID0gMHgyNzAxOwotICAgIGludCBHTF9MSU5FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwMDAxOwotICAgIGludCBHTF9MVU1JTkFOQ0UgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxOTA5OwotICAgIGludCBHTF9MVU1JTkFOQ0VfQUxQSEEgICAgICAgICAgICAgICAgICAgICAgID0gMHgxOTBBOwotICAgIGludCBHTF9NQVhfRUxFTUVOVFNfSU5ESUNFUyAgICAgICAgICAgICAgICAgID0gMHg4MEU5OwotICAgIGludCBHTF9NQVhfRUxFTUVOVFNfVkVSVElDRVMgICAgICAgICAgICAgICAgID0gMHg4MEU4OwotICAgIGludCBHTF9NQVhfTElHSFRTICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwRDMxOwotICAgIGludCBHTF9NQVhfTU9ERUxWSUVXX1NUQUNLX0RFUFRIICAgICAgICAgICAgID0gMHgwRDM2OwotICAgIGludCBHTF9NQVhfUFJPSkVDVElPTl9TVEFDS19ERVBUSCAgICAgICAgICAgID0gMHgwRDM4OwotICAgIGludCBHTF9NQVhfVEVYVFVSRV9TSVpFICAgICAgICAgICAgICAgICAgICAgID0gMHgwRDMzOwotICAgIGludCBHTF9NQVhfVEVYVFVSRV9TVEFDS19ERVBUSCAgICAgICAgICAgICAgID0gMHgwRDM5OwotICAgIGludCBHTF9NQVhfVEVYVFVSRV9VTklUUyAgICAgICAgICAgICAgICAgICAgID0gMHg4NEUyOwotICAgIGludCBHTF9NQVhfVklFV1BPUlRfRElNUyAgICAgICAgICAgICAgICAgICAgID0gMHgwRDNBOwotICAgIGludCBHTF9NT0RFTFZJRVcgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxNzAwOwotICAgIGludCBHTF9NT0RVTEFURSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgyMTAwOwotICAgIGludCBHTF9NVUxUSVNBTVBMRSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4MDlEOwotICAgIGludCBHTF9OQU5EICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxNTBFOwotICAgIGludCBHTF9ORUFSRVNUICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgyNjAwOwotICAgIGludCBHTF9ORUFSRVNUX01JUE1BUF9MSU5FQVIgICAgICAgICAgICAgICAgID0gMHgyNzAyOwotICAgIGludCBHTF9ORUFSRVNUX01JUE1BUF9ORUFSRVNUICAgICAgICAgICAgICAgID0gMHgyNzAwOwotICAgIGludCBHTF9ORVZFUiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwMjAwOwotICAgIGludCBHTF9OSUNFU1QgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxMTAyOwotICAgIGludCBHTF9OT19FUlJPUiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMDsKLSAgICBpbnQgR0xfTk9PUCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MTUwNTsKLSAgICBpbnQgR0xfTk9SICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MTUwODsKLSAgICBpbnQgR0xfTk9STUFMX0FSUkFZICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODA3NTsKLSAgICBpbnQgR0xfTk9STUFMSVpFICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEJBMTsKLSAgICBpbnQgR0xfTk9URVFVQUwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MDIwNTsKLSAgICBpbnQgR0xfTlVNX0NPTVBSRVNTRURfVEVYVFVSRV9GT1JNQVRTICAgICAgICA9IDB4ODZBMjsKLSAgICBpbnQgR0xfT05FICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDE7Ci0gICAgaW50IEdMX09ORV9NSU5VU19EU1RfQUxQSEEgICAgICAgICAgICAgICAgICAgPSAweDAzMDU7Ci0gICAgaW50IEdMX09ORV9NSU5VU19EU1RfQ09MT1IgICAgICAgICAgICAgICAgICAgPSAweDAzMDc7Ci0gICAgaW50IEdMX09ORV9NSU5VU19TUkNfQUxQSEEgICAgICAgICAgICAgICAgICAgPSAweDAzMDM7Ci0gICAgaW50IEdMX09ORV9NSU5VU19TUkNfQ09MT1IgICAgICAgICAgICAgICAgICAgPSAweDAzMDE7Ci0gICAgaW50IEdMX09SICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MDc7Ci0gICAgaW50IEdMX09SX0lOVkVSVEVEICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MEQ7Ci0gICAgaW50IEdMX09SX1JFVkVSU0UgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MEI7Ci0gICAgaW50IEdMX09VVF9PRl9NRU1PUlkgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDA1MDU7Ci0gICAgaW50IEdMX1BBQ0tfQUxJR05NRU5UICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBEMDU7Ci0gICAgaW50IEdMX1BBTEVUVEU0X1I1X0c2X0I1X09FUyAgICAgICAgICAgICAgICAgPSAweDhCOTI7Ci0gICAgaW50IEdMX1BBTEVUVEU0X1JHQjVfQTFfT0VTICAgICAgICAgICAgICAgICAgPSAweDhCOTQ7Ci0gICAgaW50IEdMX1BBTEVUVEU0X1JHQjhfT0VTICAgICAgICAgICAgICAgICAgICAgPSAweDhCOTA7Ci0gICAgaW50IEdMX1BBTEVUVEU0X1JHQkE0X09FUyAgICAgICAgICAgICAgICAgICAgPSAweDhCOTM7Ci0gICAgaW50IEdMX1BBTEVUVEU0X1JHQkE4X09FUyAgICAgICAgICAgICAgICAgICAgPSAweDhCOTE7Ci0gICAgaW50IEdMX1BBTEVUVEU4X1I1X0c2X0I1X09FUyAgICAgICAgICAgICAgICAgPSAweDhCOTc7Ci0gICAgaW50IEdMX1BBTEVUVEU4X1JHQjVfQTFfT0VTICAgICAgICAgICAgICAgICAgPSAweDhCOTk7Ci0gICAgaW50IEdMX1BBTEVUVEU4X1JHQjhfT0VTICAgICAgICAgICAgICAgICAgICAgPSAweDhCOTU7Ci0gICAgaW50IEdMX1BBTEVUVEU4X1JHQkE0X09FUyAgICAgICAgICAgICAgICAgICAgPSAweDhCOTg7Ci0gICAgaW50IEdMX1BBTEVUVEU4X1JHQkE4X09FUyAgICAgICAgICAgICAgICAgICAgPSAweDhCOTY7Ci0gICAgaW50IEdMX1BFUlNQRUNUSVZFX0NPUlJFQ1RJT05fSElOVCAgICAgICAgICAgPSAweDBDNTA7Ci0gICAgaW50IEdMX1BPSU5UX1NNT09USCAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCMTA7Ci0gICAgaW50IEdMX1BPSU5UX1NNT09USF9ISU5UICAgICAgICAgICAgICAgICAgICAgPSAweDBDNTE7Ci0gICAgaW50IEdMX1BPSU5UUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAwMDA7Ci0gICAgaW50IEdMX1BPSU5UX0ZBREVfVEhSRVNIT0xEX1NJWkUgICAgICAgICAgICAgPSAweDgxMjg7Ci0gICAgaW50IEdMX1BPSU5UX1NJWkUgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCMTE7Ci0gICAgaW50IEdMX1BPTFlHT05fT0ZGU0VUX0ZJTEwgICAgICAgICAgICAgICAgICAgPSAweDgwMzc7Ci0gICAgaW50IEdMX1BPTFlHT05fU01PT1RIX0hJTlQgICAgICAgICAgICAgICAgICAgPSAweDBDNTM7Ci0gICAgaW50IEdMX1BPU0lUSU9OICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDEyMDM7Ci0gICAgaW50IEdMX1BST0pFQ1RJT04gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE3MDE7Ci0gICAgaW50IEdMX1FVQURSQVRJQ19BVFRFTlVBVElPTiAgICAgICAgICAgICAgICAgPSAweDEyMDk7Ci0gICAgaW50IEdMX1JFRF9CSVRTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBENTI7Ci0gICAgaW50IEdMX1JFTkRFUkVSICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDFGMDE7Ci0gICAgaW50IEdMX1JFUEVBVCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDI5MDE7Ci0gICAgaW50IEdMX1JFUExBQ0UgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDFFMDE7Ci0gICAgaW50IEdMX1JFU0NBTEVfTk9STUFMICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwM0E7Ci0gICAgaW50IEdMX1JHQiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE5MDc7Ci0gICAgaW50IEdMX1JHQkEgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE5MDg7Ci0gICAgaW50IEdMX1NBTVBMRV9BTFBIQV9UT19DT1ZFUkFHRSAgICAgICAgICAgICAgPSAweDgwOUU7Ci0gICAgaW50IEdMX1NBTVBMRV9BTFBIQV9UT19PTkUgICAgICAgICAgICAgICAgICAgPSAweDgwOUY7Ci0gICAgaW50IEdMX1NBTVBMRV9DT1ZFUkFHRSAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwQTA7Ci0gICAgaW50IEdMX1NDSVNTT1JfVEVTVCAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBDMTE7Ci0gICAgaW50IEdMX1NFVCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE1MEY7Ci0gICAgaW50IEdMX1NISU5JTkVTUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE2MDE7Ci0gICAgaW50IEdMX1NIT1JUICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE0MDI7Ci0gICAgaW50IEdMX1NNT09USCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDFEMDE7Ci0gICAgaW50IEdMX1NNT09USF9MSU5FX1dJRFRIX1JBTkdFICAgICAgICAgICAgICAgPSAweDBCMjI7Ci0gICAgaW50IEdMX1NNT09USF9QT0lOVF9TSVpFX1JBTkdFICAgICAgICAgICAgICAgPSAweDBCMTI7Ci0gICAgaW50IEdMX1NQRUNVTEFSICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDEyMDI7Ci0gICAgaW50IEdMX1NQT1RfQ1VUT0ZGICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDEyMDY7Ci0gICAgaW50IEdMX1NQT1RfRElSRUNUSU9OICAgICAgICAgICAgICAgICAgICAgICAgPSAweDEyMDQ7Ci0gICAgaW50IEdMX1NQT1RfRVhQT05FTlQgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDEyMDU7Ci0gICAgaW50IEdMX1NSQ19BTFBIQSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAzMDI7Ci0gICAgaW50IEdMX1NSQ19BTFBIQV9TQVRVUkFURSAgICAgICAgICAgICAgICAgICAgPSAweDAzMDg7Ci0gICAgaW50IEdMX1NSQ19DT0xPUiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAzMDA7Ci0gICAgaW50IEdMX1NUQUNLX09WRVJGTE9XICAgICAgICAgICAgICAgICAgICAgICAgPSAweDA1MDM7Ci0gICAgaW50IEdMX1NUQUNLX1VOREVSRkxPVyAgICAgICAgICAgICAgICAgICAgICAgPSAweDA1MDQ7Ci0gICAgaW50IEdMX1NURU5DSUxfQklUUyAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBENTc7Ci0gICAgaW50IEdMX1NURU5DSUxfQlVGRkVSX0JJVCAgICAgICAgICAgICAgICAgICAgPSAweDA0MDA7Ci0gICAgaW50IEdMX1NURU5DSUxfVEVTVCAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCOTA7Ci0gICAgaW50IEdMX1NVQlBJWEVMX0JJVFMgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBENTA7Ci0gICAgaW50IEdMX1RFWFRVUkUgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE3MDI7Ci0gICAgaW50IEdMX1RFWFRVUkVfMkQgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBERTE7Ci0gICAgaW50IEdMX1RFWFRVUkVfQ09PUkRfQVJSQVkgICAgICAgICAgICAgICAgICAgPSAweDgwNzg7Ci0gICAgaW50IEdMX1RFWFRVUkVfRU5WICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDIzMDA7Ci0gICAgaW50IEdMX1RFWFRVUkVfRU5WX0NPTE9SICAgICAgICAgICAgICAgICAgICAgPSAweDIyMDE7Ci0gICAgaW50IEdMX1RFWFRVUkVfRU5WX01PREUgICAgICAgICAgICAgICAgICAgICAgPSAweDIyMDA7Ci0gICAgaW50IEdMX1RFWFRVUkVfTUFHX0ZJTFRFUiAgICAgICAgICAgICAgICAgICAgPSAweDI4MDA7Ci0gICAgaW50IEdMX1RFWFRVUkVfTUlOX0ZJTFRFUiAgICAgICAgICAgICAgICAgICAgPSAweDI4MDE7Ci0gICAgaW50IEdMX1RFWFRVUkVfV1JBUF9TICAgICAgICAgICAgICAgICAgICAgICAgPSAweDI4MDI7Ci0gICAgaW50IEdMX1RFWFRVUkVfV1JBUF9UICAgICAgICAgICAgICAgICAgICAgICAgPSAweDI4MDM7Ci0gICAgaW50IEdMX1RFWFRVUkUwICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0QzA7Ci0gICAgaW50IEdMX1RFWFRVUkUxICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0QzE7Ci0gICAgaW50IEdMX1RFWFRVUkUyICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0QzI7Ci0gICAgaW50IEdMX1RFWFRVUkUzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0QzM7Ci0gICAgaW50IEdMX1RFWFRVUkU0ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0QzQ7Ci0gICAgaW50IEdMX1RFWFRVUkU1ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0QzU7Ci0gICAgaW50IEdMX1RFWFRVUkU2ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0QzY7Ci0gICAgaW50IEdMX1RFWFRVUkU3ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0Qzc7Ci0gICAgaW50IEdMX1RFWFRVUkU4ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0Qzg7Ci0gICAgaW50IEdMX1RFWFRVUkU5ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0Qzk7Ci0gICAgaW50IEdMX1RFWFRVUkUxMCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0Q0E7Ci0gICAgaW50IEdMX1RFWFRVUkUxMSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0Q0I7Ci0gICAgaW50IEdMX1RFWFRVUkUxMiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0Q0M7Ci0gICAgaW50IEdMX1RFWFRVUkUxMyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0Q0Q7Ci0gICAgaW50IEdMX1RFWFRVUkUxNCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0Q0U7Ci0gICAgaW50IEdMX1RFWFRVUkUxNSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0Q0Y7Ci0gICAgaW50IEdMX1RFWFRVUkUxNiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDA7Ci0gICAgaW50IEdMX1RFWFRVUkUxNyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDE7Ci0gICAgaW50IEdMX1RFWFRVUkUxOCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDI7Ci0gICAgaW50IEdMX1RFWFRVUkUxOSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDM7Ci0gICAgaW50IEdMX1RFWFRVUkUyMCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDQ7Ci0gICAgaW50IEdMX1RFWFRVUkUyMSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDU7Ci0gICAgaW50IEdMX1RFWFRVUkUyMiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDY7Ci0gICAgaW50IEdMX1RFWFRVUkUyMyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDc7Ci0gICAgaW50IEdMX1RFWFRVUkUyNCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDg7Ci0gICAgaW50IEdMX1RFWFRVUkUyNSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RDk7Ci0gICAgaW50IEdMX1RFWFRVUkUyNiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0REE7Ci0gICAgaW50IEdMX1RFWFRVUkUyNyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0REI7Ci0gICAgaW50IEdMX1RFWFRVUkUyOCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0REM7Ci0gICAgaW50IEdMX1RFWFRVUkUyOSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0REQ7Ci0gICAgaW50IEdMX1RFWFRVUkUzMCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0REU7Ci0gICAgaW50IEdMX1RFWFRVUkUzMSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0REY7Ci0gICAgaW50IEdMX1RSSUFOR0xFX0ZBTiAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAwMDY7Ci0gICAgaW50IEdMX1RSSUFOR0xFX1NUUklQICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAwMDU7Ci0gICAgaW50IEdMX1RSSUFOR0xFUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDAwMDQ7Ci0gICAgaW50IEdMX1RSVUUgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAxOwotICAgIGludCBHTF9VTlBBQ0tfQUxJR05NRU5UICAgICAgICAgICAgICAgICAgICAgID0gMHgwQ0Y1OwotICAgIGludCBHTF9VTlNJR05FRF9CWVRFICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxNDAxOwotICAgIGludCBHTF9VTlNJR05FRF9TSE9SVCAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxNDAzOwotICAgIGludCBHTF9VTlNJR05FRF9TSE9SVF80XzRfNF80ICAgICAgICAgICAgICAgID0gMHg4MDMzOwotICAgIGludCBHTF9VTlNJR05FRF9TSE9SVF81XzVfNV8xICAgICAgICAgICAgICAgID0gMHg4MDM0OwotICAgIGludCBHTF9VTlNJR05FRF9TSE9SVF81XzZfNSAgICAgICAgICAgICAgICAgID0gMHg4MzYzOwotICAgIGludCBHTF9WRU5ET1IgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxRjAwOwotICAgIGludCBHTF9WRVJTSU9OICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxRjAyOwotICAgIGludCBHTF9WRVJURVhfQVJSQVkgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4MDc0OwotICAgIGludCBHTF9YT1IgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxNTA2OwotICAgIGludCBHTF9aRVJPICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMDsKLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL0dMMTFFeHRIZWFkZXIuamF2YS1pZiBiL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTDExRXh0SGVhZGVyLmphdmEtaWYKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDdiZTIxNjQuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL0dMMTFFeHRIZWFkZXIuamF2YS1pZgorKysgL2Rldi9udWxsCkBAIC0xLDQwICswLDAgQEAKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNywgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0vLyBUaGlzIHNvdXJjZSBmaWxlIGlzIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGVkCi0KLXBhY2thZ2UgamF2YXgubWljcm9lZGl0aW9uLmtocm9ub3Mub3BlbmdsZXM7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgR0wxMUV4dCBleHRlbmRzIEdMIHsKLSAgICBpbnQgR0xfTUFUUklYX0lOREVYX0FSUkFZX0JVRkZFUl9CSU5ESU5HX09FUyA9IDB4OEI5RTsKLSAgICBpbnQgR0xfTUFUUklYX0lOREVYX0FSUkFZX09FUyAgICAgICAgICAgICAgICA9IDB4ODg0NDsKLSAgICBpbnQgR0xfTUFUUklYX0lOREVYX0FSUkFZX1BPSU5URVJfT0VTICAgICAgICA9IDB4ODg0OTsKLSAgICBpbnQgR0xfTUFUUklYX0lOREVYX0FSUkFZX1NJWkVfT0VTICAgICAgICAgICA9IDB4ODg0NjsKLSAgICBpbnQgR0xfTUFUUklYX0lOREVYX0FSUkFZX1NUUklERV9PRVMgICAgICAgICA9IDB4ODg0ODsKLSAgICBpbnQgR0xfTUFUUklYX0lOREVYX0FSUkFZX1RZUEVfT0VTICAgICAgICAgICA9IDB4ODg0NzsKLSAgICBpbnQgR0xfTUFUUklYX1BBTEVUVEVfT0VTICAgICAgICAgICAgICAgICAgICA9IDB4ODg0MDsKLSAgICBpbnQgR0xfTUFYX1BBTEVUVEVfTUFUUklDRVNfT0VTICAgICAgICAgICAgICA9IDB4ODg0MjsKLSAgICBpbnQgR0xfTUFYX1ZFUlRFWF9VTklUU19PRVMgICAgICAgICAgICAgICAgICA9IDB4ODZBNDsKLSAgICBpbnQgR0xfVEVYVFVSRV9DUk9QX1JFQ1RfT0VTICAgICAgICAgICAgICAgICA9IDB4OEI5RDsKLSAgICBpbnQgR0xfV0VJR0hUX0FSUkFZX0JVRkZFUl9CSU5ESU5HX09FUyAgICAgICA9IDB4ODg5RTsKLSAgICBpbnQgR0xfV0VJR0hUX0FSUkFZX09FUyAgICAgICAgICAgICAgICAgICAgICA9IDB4ODZBRDsKLSAgICBpbnQgR0xfV0VJR0hUX0FSUkFZX1BPSU5URVJfT0VTICAgICAgICAgICAgICA9IDB4ODZBQzsKLSAgICBpbnQgR0xfV0VJR0hUX0FSUkFZX1NJWkVfT0VTICAgICAgICAgICAgICAgICA9IDB4ODZBQjsKLSAgICBpbnQgR0xfV0VJR0hUX0FSUkFZX1NUUklERV9PRVMgICAgICAgICAgICAgICA9IDB4ODZBQTsKLSAgICBpbnQgR0xfV0VJR0hUX0FSUkFZX1RZUEVfT0VTICAgICAgICAgICAgICAgICA9IDB4ODZBOTsKLQotICAgIHZvaWQgZ2xUZXhQYXJhbWV0ZXJmdihpbnQgdGFyZ2V0LCBpbnQgcG5hbWUsIGZsb2F0W10gcGFyYW0sIGludCBvZmZzZXQpOwotCmRpZmYgLS1naXQgYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvR0wxMUV4dGVuc2lvblBhY2tIZWFkZXIuamF2YS1pZiBiL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTDExRXh0ZW5zaW9uUGFja0hlYWRlci5qYXZhLWlmCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBhODAwMTkxLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTDExRXh0ZW5zaW9uUGFja0hlYWRlci5qYXZhLWlmCisrKyAvZGV2L251bGwKQEAgLTEsMTA4ICswLDAgQEAKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNywgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0vLyBUaGlzIHNvdXJjZSBmaWxlIGlzIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGVkCi0KLXBhY2thZ2UgamF2YXgubWljcm9lZGl0aW9uLmtocm9ub3Mub3BlbmdsZXM7Ci0KLXB1YmxpYyBpbnRlcmZhY2UgR0wxMUV4dGVuc2lvblBhY2sgZXh0ZW5kcyBHTCB7Ci0gICAgaW50IEdMX0JMRU5EX0RTVF9BTFBIQSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODBDQTsKLSAgICBpbnQgR0xfQkxFTkRfRFNUX1JHQiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4MEM4OwotICAgIGludCBHTF9CTEVORF9FUVVBVElPTiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwMDk7Ci0gICAgaW50IEdMX0JMRU5EX0VRVUFUSU9OX0FMUEhBICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODgzRDsKLSAgICBpbnQgR0xfQkxFTkRfRVFVQVRJT05fUkdCICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4MDA5OwotICAgIGludCBHTF9CTEVORF9TUkNfQUxQSEEgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwQ0I7Ci0gICAgaW50IEdMX0JMRU5EX1NSQ19SR0IgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODBDOTsKLSAgICBpbnQgR0xfQ09MT1JfQVRUQUNITUVOVDBfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4Q0UwOwotICAgIGludCBHTF9DT0xPUl9BVFRBQ0hNRU5UMV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhDRTE7Ci0gICAgaW50IEdMX0NPTE9SX0FUVEFDSE1FTlQyX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OENFMjsKLSAgICBpbnQgR0xfQ09MT1JfQVRUQUNITUVOVDNfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4Q0UzOwotICAgIGludCBHTF9DT0xPUl9BVFRBQ0hNRU5UNF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhDRTQ7Ci0gICAgaW50IEdMX0NPTE9SX0FUVEFDSE1FTlQ1X09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OENFNTsKLSAgICBpbnQgR0xfQ09MT1JfQVRUQUNITUVOVDZfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4Q0U2OwotICAgIGludCBHTF9DT0xPUl9BVFRBQ0hNRU5UN19PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhDRTc7Ci0gICAgaW50IEdMX0NPTE9SX0FUVEFDSE1FTlQ4X09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OENFODsKLSAgICBpbnQgR0xfQ09MT1JfQVRUQUNITUVOVDlfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4Q0U5OwotICAgIGludCBHTF9DT0xPUl9BVFRBQ0hNRU5UMTBfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhDRUE7Ci0gICAgaW50IEdMX0NPTE9SX0FUVEFDSE1FTlQxMV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OENFQjsKLSAgICBpbnQgR0xfQ09MT1JfQVRUQUNITUVOVDEyX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4Q0VDOwotICAgIGludCBHTF9DT0xPUl9BVFRBQ0hNRU5UMTNfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhDRUQ7Ci0gICAgaW50IEdMX0NPTE9SX0FUVEFDSE1FTlQxNF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OENFRTsKLSAgICBpbnQgR0xfQ09MT1JfQVRUQUNITUVOVDE1X09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4Q0VGOwotICAgIGludCBHTF9ERUNSX1dSQVAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1MDg7Ci0gICAgaW50IEdMX0RFUFRIX0FUVEFDSE1FTlRfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OEQwMDsKLSAgICBpbnQgR0xfREVQVEhfQ09NUE9ORU5UICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgxOTAyOwotICAgIGludCBHTF9ERVBUSF9DT01QT05FTlQxNiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgxQTU7Ci0gICAgaW50IEdMX0RFUFRIX0NPTVBPTkVOVDI0ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODFBNjsKLSAgICBpbnQgR0xfREVQVEhfQ09NUE9ORU5UMzIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4MUE3OwotICAgIGludCBHTF9GUkFNRUJVRkZFUl9BVFRBQ0hNRU5UX09CSkVDVF9OQU1FX09FUyAgICAgICAgICAgPSAweDhDRDE7Ci0gICAgaW50IEdMX0ZSQU1FQlVGRkVSX0FUVEFDSE1FTlRfT0JKRUNUX1RZUEVfT0VTICAgICAgICAgICA9IDB4OENEMDsKLSAgICBpbnQgR0xfRlJBTUVCVUZGRVJfQVRUQUNITUVOVF9URVhUVVJFX0NVQkVfTUFQX0ZBQ0VfT0VTID0gMHg4Q0QzOwotICAgIGludCBHTF9GUkFNRUJVRkZFUl9BVFRBQ0hNRU5UX1RFWFRVUkVfTEVWRUxfT0VTICAgICAgICAgPSAweDhDRDI7Ci0gICAgaW50IEdMX0ZSQU1FQlVGRkVSX0JJTkRJTkdfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OENBNjsKLSAgICBpbnQgR0xfRlJBTUVCVUZGRVJfQ09NUExFVEVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4Q0Q1OwotICAgIGludCBHTF9GUkFNRUJVRkZFUl9JTkNPTVBMRVRFX0FUVEFDSE1FTlRfT0VTICAgICAgICAgICAgPSAweDhDRDY7Ci0gICAgaW50IEdMX0ZSQU1FQlVGRkVSX0lOQ09NUExFVEVfRElNRU5TSU9OU19PRVMgICAgICAgICAgICA9IDB4OENEOTsKLSAgICBpbnQgR0xfRlJBTUVCVUZGRVJfSU5DT01QTEVURV9EUkFXX0JVRkZFUl9PRVMgICAgICAgICAgID0gMHg4Q0RCOwotICAgIGludCBHTF9GUkFNRUJVRkZFUl9JTkNPTVBMRVRFX0ZPUk1BVFNfT0VTICAgICAgICAgICAgICAgPSAweDhDREE7Ci0gICAgaW50IEdMX0ZSQU1FQlVGRkVSX0lOQ09NUExFVEVfTUlTU0lOR19BVFRBQ0hNRU5UX09FUyAgICA9IDB4OENENzsKLSAgICBpbnQgR0xfRlJBTUVCVUZGRVJfSU5DT01QTEVURV9SRUFEX0JVRkZFUl9PRVMgICAgICAgICAgID0gMHg4Q0RDOwotICAgIGludCBHTF9GUkFNRUJVRkZFUl9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhENDA7Ci0gICAgaW50IEdMX0ZSQU1FQlVGRkVSX1VOU1VQUE9SVEVEX09FUyAgICAgICAgICAgICAgICAgICAgICA9IDB4OENERDsKLSAgICBpbnQgR0xfRlVOQ19BREQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4MDA2OwotICAgIGludCBHTF9GVU5DX1JFVkVSU0VfU1VCVFJBQ1QgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwMEI7Ci0gICAgaW50IEdMX0ZVTkNfU1VCVFJBQ1QgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODAwQTsKLSAgICBpbnQgR0xfSU5DUl9XUkFQICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTA3OwotICAgIGludCBHTF9JTlZBTElEX0ZSQU1FQlVGRkVSX09QRVJBVElPTl9PRVMgICAgICAgICAgICAgICAgPSAweDA1MDY7Ci0gICAgaW50IEdMX01BWF9DT0xPUl9BVFRBQ0hNRU5UU19PRVMgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OENERjsKLSAgICBpbnQgR0xfTUFYX0NVQkVfTUFQX1RFWFRVUkVfU0laRSAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTFDOwotICAgIGludCBHTF9NQVhfUkVOREVSQlVGRkVSX1NJWkVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg0RTg7Ci0gICAgaW50IEdMX01JUlJPUkVEX1JFUEVBVCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODM3MDsKLSAgICBpbnQgR0xfTk9STUFMX01BUCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTExOwotICAgIGludCBHTF9SRUZMRUNUSU9OX01BUCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1MTI7Ci0gICAgaW50IEdMX1JFTkRFUkJVRkZFUl9BTFBIQV9TSVpFX09FUyAgICAgICAgICAgICAgICAgICAgICA9IDB4OEQ1MzsKLSAgICBpbnQgR0xfUkVOREVSQlVGRkVSX0JJTkRJTkdfT0VTICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4Q0E3OwotICAgIGludCBHTF9SRU5ERVJCVUZGRVJfQkxVRV9TSVpFX09FUyAgICAgICAgICAgICAgICAgICAgICAgPSAweDhENTI7Ci0gICAgaW50IEdMX1JFTkRFUkJVRkZFUl9ERVBUSF9TSVpFX09FUyAgICAgICAgICAgICAgICAgICAgICA9IDB4OEQ1NDsKLSAgICBpbnQgR0xfUkVOREVSQlVGRkVSX0dSRUVOX1NJWkVfT0VTICAgICAgICAgICAgICAgICAgICAgID0gMHg4RDUxOwotICAgIGludCBHTF9SRU5ERVJCVUZGRVJfSEVJR0hUX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhENDM7Ci0gICAgaW50IEdMX1JFTkRFUkJVRkZFUl9JTlRFUk5BTF9GT1JNQVRfT0VTICAgICAgICAgICAgICAgICA9IDB4OEQ0NDsKLSAgICBpbnQgR0xfUkVOREVSQlVGRkVSX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4RDQxOwotICAgIGludCBHTF9SRU5ERVJCVUZGRVJfUkVEX1NJWkVfT0VTICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhENTA7Ci0gICAgaW50IEdMX1JFTkRFUkJVRkZFUl9TVEVOQ0lMX1NJWkVfT0VTICAgICAgICAgICAgICAgICAgICA9IDB4OEQ1NTsKLSAgICBpbnQgR0xfUkVOREVSQlVGRkVSX1dJRFRIX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4RDQyOwotICAgIGludCBHTF9SR0I1X0ExICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwNTc7Ci0gICAgaW50IEdMX1JHQjU2NV9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OEQ2MjsKLSAgICBpbnQgR0xfUkdCOCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4MDUxOwotICAgIGludCBHTF9SR0JBNCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwNTY7Ci0gICAgaW50IEdMX1JHQkE4ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODA1ODsKLSAgICBpbnQgR0xfU1RFTkNJTF9BVFRBQ0hNRU5UX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4RDIwOwotICAgIGludCBHTF9TVEVOQ0lMX0lOREVYICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDE5MDE7Ci0gICAgaW50IEdMX1NURU5DSUxfSU5ERVgxX09FUyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4OEQ0NjsKLSAgICBpbnQgR0xfU1RFTkNJTF9JTkRFWDRfT0VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4RDQ3OwotICAgIGludCBHTF9TVEVOQ0lMX0lOREVYOF9PRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhENDg7Ci0gICAgaW50IEdMX1NUUiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IC0xOwotICAgIGludCBHTF9URVhUVVJFX0JJTkRJTkdfQ1VCRV9NQVAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1MTQ7Ci0gICAgaW50IEdMX1RFWFRVUkVfQ1VCRV9NQVAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODUxMzsKLSAgICBpbnQgR0xfVEVYVFVSRV9DVUJFX01BUF9ORUdBVElWRV9YICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTE2OwotICAgIGludCBHTF9URVhUVVJFX0NVQkVfTUFQX05FR0FUSVZFX1kgICAgICAgICAgICAgICAgICAgICAgPSAweDg1MTg7Ci0gICAgaW50IEdMX1RFWFRVUkVfQ1VCRV9NQVBfTkVHQVRJVkVfWiAgICAgICAgICAgICAgICAgICAgICA9IDB4ODUxQTsKLSAgICBpbnQgR0xfVEVYVFVSRV9DVUJFX01BUF9QT1NJVElWRV9YICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTE1OwotICAgIGludCBHTF9URVhUVVJFX0NVQkVfTUFQX1BPU0lUSVZFX1kgICAgICAgICAgICAgICAgICAgICAgPSAweDg1MTc7Ci0gICAgaW50IEdMX1RFWFRVUkVfQ1VCRV9NQVBfUE9TSVRJVkVfWiAgICAgICAgICAgICAgICAgICAgICA9IDB4ODUxOTsKLSAgICBpbnQgR0xfVEVYVFVSRV9HRU5fTU9ERSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgyNTAwOwotICAgIGludCBHTF9URVhUVVJFX0dFTl9TVFIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDhENjA7Ci0KZGlmZiAtLWdpdCBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTDExSGVhZGVyLmphdmEtaWYgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvR0wxMUhlYWRlci5qYXZhLWlmCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBiMGU1YTZiLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTDExSGVhZGVyLmphdmEtaWYKKysrIC9kZXYvbnVsbApAQCAtMSwxNDUgKzAsMCBAQAotKioKLSoqIENvcHlyaWdodCAyMDA2LCBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0qKgotKiogTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IAotKiogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLiAKLSoqIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCAKLSoqCi0qKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIAotKioKLSoqIFVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUgCi0qKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLCAKLSoqIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiAKLSoqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQgCi0qKiBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KLSovCi0KLS8vIFRoaXMgc291cmNlIGZpbGUgaXMgYXV0b21hdGljYWxseSBnZW5lcmF0ZWQKLQotcGFja2FnZSBqYXZheC5taWNyb2VkaXRpb24ua2hyb25vcy5vcGVuZ2xlczsKLQotcHVibGljIGludGVyZmFjZSBHTDExIGV4dGVuZHMgR0wxMCB7Ci0gICAgaW50IEdMX0FDVElWRV9URVhUVVJFICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODRFMDsKLSAgICBpbnQgR0xfQUREX1NJR05FRCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTc0OwotICAgIGludCBHTF9BTFBIQV9TQ0FMRSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBEMUM7Ci0gICAgaW50IEdMX0FMUEhBX1RFU1RfRlVOQyAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEJDMTsKLSAgICBpbnQgR0xfQUxQSEFfVEVTVF9SRUYgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQkMyOwotICAgIGludCBHTF9BUlJBWV9CVUZGRVIgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg4OTI7Ci0gICAgaW50IEdMX0FSUkFZX0JVRkZFUl9CSU5ESU5HICAgICAgICAgICAgICAgICAgICA9IDB4ODg5NDsKLSAgICBpbnQgR0xfQkxFTkRfRFNUICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQkUwOwotICAgIGludCBHTF9CTEVORF9TUkMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCRTE7Ci0gICAgaW50IEdMX0JVRkZFUl9BQ0NFU1MgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODhCQjsKLSAgICBpbnQgR0xfQlVGRkVSX1NJWkUgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NzY0OwotICAgIGludCBHTF9CVUZGRVJfVVNBR0UgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg3NjU7Ci0gICAgaW50IEdMX0NMSUVOVF9BQ1RJVkVfVEVYVFVSRSAgICAgICAgICAgICAgICAgICA9IDB4ODRFMTsKLSAgICBpbnQgR0xfQ0xJUF9QTEFORTAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgzMDAwOwotICAgIGludCBHTF9DTElQX1BMQU5FMSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDMwMDE7Ci0gICAgaW50IEdMX0NMSVBfUExBTkUyICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MzAwMjsKLSAgICBpbnQgR0xfQ0xJUF9QTEFORTMgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgzMDAzOwotICAgIGludCBHTF9DTElQX1BMQU5FNCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDMwMDQ7Ci0gICAgaW50IEdMX0NMSVBfUExBTkU1ICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MzAwNTsKLSAgICBpbnQgR0xfQ09MT1JfQVJSQVlfQlVGRkVSX0JJTkRJTkcgICAgICAgICAgICAgID0gMHg4ODk4OwotICAgIGludCBHTF9DT0xPUl9BUlJBWV9QT0lOVEVSICAgICAgICAgICAgICAgICAgICAgPSAweDgwOTA7Ci0gICAgaW50IEdMX0NPTE9SX0FSUkFZX1NJWkUgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODA4MTsKLSAgICBpbnQgR0xfQ09MT1JfQVJSQVlfU1RSSURFICAgICAgICAgICAgICAgICAgICAgID0gMHg4MDgzOwotICAgIGludCBHTF9DT0xPUl9BUlJBWV9UWVBFICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwODI7Ci0gICAgaW50IEdMX0NPTE9SX0NMRUFSX1ZBTFVFICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEMyMjsKLSAgICBpbnQgR0xfQ09MT1JfV1JJVEVNQVNLICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQzIzOwotICAgIGludCBHTF9DT01CSU5FICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1NzA7Ci0gICAgaW50IEdMX0NPTUJJTkVfQUxQSEEgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODU3MjsKLSAgICBpbnQgR0xfQ09NQklORV9SR0IgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTcxOwotICAgIGludCBHTF9DT05TVEFOVCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1NzY7Ci0gICAgaW50IEdMX0NPT1JEX1JFUExBQ0VfT0VTICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODg2MjsKLSAgICBpbnQgR0xfQ1VMTF9GQUNFX01PREUgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjQ1OwotICAgIGludCBHTF9DVVJSRU5UX0NPTE9SICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCMDA7Ci0gICAgaW50IEdMX0NVUlJFTlRfTk9STUFMICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEIwMjsKLSAgICBpbnQgR0xfQ1VSUkVOVF9URVhUVVJFX0NPT1JEUyAgICAgICAgICAgICAgICAgID0gMHgwQjAzOwotICAgIGludCBHTF9ERVBUSF9DTEVBUl9WQUxVRSAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCNzM7Ci0gICAgaW50IEdMX0RFUFRIX0ZVTkMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEI3NDsKLSAgICBpbnQgR0xfREVQVEhfUkFOR0UgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjcwOwotICAgIGludCBHTF9ERVBUSF9XUklURU1BU0sgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCNzI7Ci0gICAgaW50IEdMX0RPVDNfUkdCICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODZBRTsKLSAgICBpbnQgR0xfRE9UM19SR0JBICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NkFGOwotICAgIGludCBHTF9EWU5BTUlDX0RSQVcgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg4RTg7Ci0gICAgaW50IEdMX0VMRU1FTlRfQVJSQVlfQlVGRkVSICAgICAgICAgICAgICAgICAgICA9IDB4ODg5MzsKLSAgICBpbnQgR0xfRUxFTUVOVF9BUlJBWV9CVUZGRVJfQklORElORyAgICAgICAgICAgID0gMHg4ODk1OwotICAgIGludCBHTF9GUk9OVF9GQUNFICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCNDY7Ci0gICAgaW50IEdMX0dFTkVSQVRFX01JUE1BUCAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODE5MTsKLSAgICBpbnQgR0xfR0VORVJBVEVfTUlQTUFQX0hJTlQgICAgICAgICAgICAgICAgICAgID0gMHg4MTkyOwotICAgIGludCBHTF9JTlRFUlBPTEFURSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1NzU7Ci0gICAgaW50IEdMX0xJTkVfV0lEVEggICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEIyMTsKLSAgICBpbnQgR0xfTE9HSUNfT1BfTU9ERSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQkYwOwotICAgIGludCBHTF9NQVRSSVhfTU9ERSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCQTA7Ci0gICAgaW50IEdMX01BWF9DTElQX1BMQU5FUyAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEQzMjsKLSAgICBpbnQgR0xfTU9ERUxWSUVXX01BVFJJWCAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQkE2OwotICAgIGludCBHTF9NT0RFTFZJRVdfTUFUUklYX0ZMT0FUX0FTX0lOVF9CSVRTX09FUyAgPSAweDg5OEQ7Ci0gICAgaW50IEdMX01PREVMVklFV19TVEFDS19ERVBUSCAgICAgICAgICAgICAgICAgICA9IDB4MEJBMzsKLSAgICBpbnQgR0xfTk9STUFMX0FSUkFZX0JVRkZFUl9CSU5ESU5HICAgICAgICAgICAgID0gMHg4ODk3OwotICAgIGludCBHTF9OT1JNQUxfQVJSQVlfUE9JTlRFUiAgICAgICAgICAgICAgICAgICAgPSAweDgwOEY7Ci0gICAgaW50IEdMX05PUk1BTF9BUlJBWV9TVFJJREUgICAgICAgICAgICAgICAgICAgICA9IDB4ODA3RjsKLSAgICBpbnQgR0xfTk9STUFMX0FSUkFZX1RZUEUgICAgICAgICAgICAgICAgICAgICAgID0gMHg4MDdFOwotICAgIGludCBHTF9PUEVSQU5EMF9BTFBIQSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1OTg7Ci0gICAgaW50IEdMX09QRVJBTkQwX1JHQiAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODU5MDsKLSAgICBpbnQgR0xfT1BFUkFORDFfQUxQSEEgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTk5OwotICAgIGludCBHTF9PUEVSQU5EMV9SR0IgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1OTE7Ci0gICAgaW50IEdMX09QRVJBTkQyX0FMUEhBICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODU5QTsKLSAgICBpbnQgR0xfT1BFUkFORDJfUkdCICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTkyOwotICAgIGludCBHTF9QT0lOVF9ESVNUQU5DRV9BVFRFTlVBVElPTiAgICAgICAgICAgICAgPSAweDgxMjk7Ci0gICAgaW50IEdMX1BPSU5UX0ZBREVfVEhSRVNIT0xEX1NJWkUgICAgICAgICAgICAgICA9IDB4ODEyODsKLSAgICBpbnQgR0xfUE9JTlRfU0laRSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjExOwotICAgIGludCBHTF9QT0lOVF9TSVpFX0FSUkFZX0JVRkZFUl9CSU5ESU5HX09FUyAgICAgPSAweDhCOUY7Ci0gICAgaW50IEdMX1BPSU5UX1NJWkVfQVJSQVlfT0VTICAgICAgICAgICAgICAgICAgICA9IDB4OEI5QzsKLSAgICBpbnQgR0xfUE9JTlRfU0laRV9BUlJBWV9QT0lOVEVSX09FUyAgICAgICAgICAgID0gMHg4OThDOwotICAgIGludCBHTF9QT0lOVF9TSVpFX0FSUkFZX1NUUklERV9PRVMgICAgICAgICAgICAgPSAweDg5OEI7Ci0gICAgaW50IEdMX1BPSU5UX1NJWkVfQVJSQVlfVFlQRV9PRVMgICAgICAgICAgICAgICA9IDB4ODk4QTsKLSAgICBpbnQgR0xfUE9JTlRfU0laRV9NQVggICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4MTI3OwotICAgIGludCBHTF9QT0lOVF9TSVpFX01JTiAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDgxMjY7Ci0gICAgaW50IEdMX1BPSU5UX1NQUklURV9PRVMgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODg2MTsKLSAgICBpbnQgR0xfUE9MWUdPTl9PRkZTRVRfRkFDVE9SICAgICAgICAgICAgICAgICAgID0gMHg4MDM4OwotICAgIGludCBHTF9QT0xZR09OX09GRlNFVF9VTklUUyAgICAgICAgICAgICAgICAgICAgPSAweDJBMDA7Ci0gICAgaW50IEdMX1BSRVZJT1VTICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODU3ODsKLSAgICBpbnQgR0xfUFJJTUFSWV9DT0xPUiAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTc3OwotICAgIGludCBHTF9QUk9KRUNUSU9OX01BVFJJWCAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCQTc7Ci0gICAgaW50IEdMX1BST0pFQ1RJT05fTUFUUklYX0ZMT0FUX0FTX0lOVF9CSVRTX09FUyA9IDB4ODk4RTsKLSAgICBpbnQgR0xfUFJPSkVDVElPTl9TVEFDS19ERVBUSCAgICAgICAgICAgICAgICAgID0gMHgwQkE0OwotICAgIGludCBHTF9SR0JfU0NBTEUgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1NzM7Ci0gICAgaW50IEdMX1NBTVBMRV9CVUZGRVJTICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODBBODsKLSAgICBpbnQgR0xfU0FNUExFX0NPVkVSQUdFX0lOVkVSVCAgICAgICAgICAgICAgICAgID0gMHg4MEFCOwotICAgIGludCBHTF9TQU1QTEVfQ09WRVJBR0VfVkFMVUUgICAgICAgICAgICAgICAgICAgPSAweDgwQUE7Ci0gICAgaW50IEdMX1NBTVBMRVMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODBBOTsKLSAgICBpbnQgR0xfU0NJU1NPUl9CT1ggICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQzEwOwotICAgIGludCBHTF9TSEFERV9NT0RFTCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCNTQ7Ci0gICAgaW50IEdMX1NSQzBfQUxQSEEgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODU4ODsKLSAgICBpbnQgR0xfU1JDMF9SR0IgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NTgwOwotICAgIGludCBHTF9TUkMxX0FMUEhBICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1ODk7Ci0gICAgaW50IEdMX1NSQzFfUkdCICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODU4MTsKLSAgICBpbnQgR0xfU1JDMl9BTFBIQSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4NThBOwotICAgIGludCBHTF9TUkMyX1JHQiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDg1ODI7Ci0gICAgaW50IEdMX1NUQVRJQ19EUkFXICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODhFNDsKLSAgICBpbnQgR0xfU1RFTkNJTF9DTEVBUl9WQUxVRSAgICAgICAgICAgICAgICAgICAgID0gMHgwQjkxOwotICAgIGludCBHTF9TVEVOQ0lMX0ZBSUwgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCOTQ7Ci0gICAgaW50IEdMX1NURU5DSUxfRlVOQyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEI5MjsKLSAgICBpbnQgR0xfU1RFTkNJTF9QQVNTX0RFUFRIX0ZBSUwgICAgICAgICAgICAgICAgID0gMHgwQjk1OwotICAgIGludCBHTF9TVEVOQ0lMX1BBU1NfREVQVEhfUEFTUyAgICAgICAgICAgICAgICAgPSAweDBCOTY7Ci0gICAgaW50IEdMX1NURU5DSUxfUkVGICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEI5NzsKLSAgICBpbnQgR0xfU1RFTkNJTF9WQUxVRV9NQVNLICAgICAgICAgICAgICAgICAgICAgID0gMHgwQjkzOwotICAgIGludCBHTF9TVEVOQ0lMX1dSSVRFTUFTSyAgICAgICAgICAgICAgICAgICAgICAgPSAweDBCOTg7Ci0gICAgaW50IEdMX1NVQlRSQUNUICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODRFNzsKLSAgICBpbnQgR0xfVEVYVFVSRV9CSU5ESU5HXzJEICAgICAgICAgICAgICAgICAgICAgID0gMHg4MDY5OwotICAgIGludCBHTF9URVhUVVJFX0NPT1JEX0FSUkFZX0JVRkZFUl9CSU5ESU5HICAgICAgPSAweDg4OUE7Ci0gICAgaW50IEdMX1RFWFRVUkVfQ09PUkRfQVJSQVlfUE9JTlRFUiAgICAgICAgICAgICA9IDB4ODA5MjsKLSAgICBpbnQgR0xfVEVYVFVSRV9DT09SRF9BUlJBWV9TSVpFICAgICAgICAgICAgICAgID0gMHg4MDg4OwotICAgIGludCBHTF9URVhUVVJFX0NPT1JEX0FSUkFZX1NUUklERSAgICAgICAgICAgICAgPSAweDgwOEE7Ci0gICAgaW50IEdMX1RFWFRVUkVfQ09PUkRfQVJSQVlfVFlQRSAgICAgICAgICAgICAgICA9IDB4ODA4OTsKLSAgICBpbnQgR0xfVEVYVFVSRV9NQVRSSVggICAgICAgICAgICAgICAgICAgICAgICAgID0gMHgwQkE4OwotICAgIGludCBHTF9URVhUVVJFX01BVFJJWF9GTE9BVF9BU19JTlRfQklUU19PRVMgICAgPSAweDg5OEY7Ci0gICAgaW50IEdMX1RFWFRVUkVfU1RBQ0tfREVQVEggICAgICAgICAgICAgICAgICAgICA9IDB4MEJBNTsKLSAgICBpbnQgR0xfVkVSVEVYX0FSUkFZX0JVRkZFUl9CSU5ESU5HICAgICAgICAgICAgID0gMHg4ODk2OwotICAgIGludCBHTF9WRVJURVhfQVJSQVlfUE9JTlRFUiAgICAgICAgICAgICAgICAgICAgPSAweDgwOEU7Ci0gICAgaW50IEdMX1ZFUlRFWF9BUlJBWV9TSVpFICAgICAgICAgICAgICAgICAgICAgICA9IDB4ODA3QTsKLSAgICBpbnQgR0xfVkVSVEVYX0FSUkFZX1NUUklERSAgICAgICAgICAgICAgICAgICAgID0gMHg4MDdDOwotICAgIGludCBHTF9WRVJURVhfQVJSQVlfVFlQRSAgICAgICAgICAgICAgICAgICAgICAgPSAweDgwN0I7Ci0gICAgaW50IEdMX1ZJRVdQT1JUICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9IDB4MEJBMjsKLSAgICBpbnQgR0xfV1JJVEVfT05MWSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gMHg4OEI5OwotCi0gICAgdm9pZCBnbEdldFBvaW50ZXJ2KGludCBwbmFtZSwgamF2YS5uaW8uQnVmZmVyW10gcGFyYW1zKTsKZGlmZiAtLWdpdCBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTDExSW1wbEhlYWRlci5qYXZhLWltcGwgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvR0wxMUltcGxIZWFkZXIuamF2YS1pbXBsCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA1MDFiZTY1Li4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTDExSW1wbEhlYWRlci5qYXZhLWltcGwKKysrIC9kZXYvbnVsbApAQCAtMSwzMCArMCwwIEBACi0vLyBDb3B5cmlnaHQgMjAwNiBUaGUgQW5kcm9pZCBPcGVuIFNvdXJjZSBQcm9qZWN0Ci0KLS8vIEFsbCBSaWdodHMgUmVzZXJ2ZWQuCi0KLS8vIFRoaXMgc291cmNlIGZpbGUgaXMgYXV0b21hdGljYWxseSBnZW5lcmF0ZWQKLQotcGFja2FnZSBjb20uZ29vZ2xlLmFuZHJvaWQuZ2xlc19qbmk7Ci0KLWltcG9ydCBqYXZhLm5pby5CdWZmZXI7Ci1pbXBvcnQgamF2YXgubWljcm9lZGl0aW9uLmtocm9ub3Mub3BlbmdsZXMuR0wxMTsKLWltcG9ydCBhbmRyb2lkLmdyYXBoaWNzLkNhbnZhczsKLQotcHVibGljIGNsYXNzIEdMMTFJbXBsIGltcGxlbWVudHMgR0wxMSB7Ci0KLSAgICAvLyBQcml2YXRlIGFjY2Vzc29ycyBmb3IgbmF0aXZlIGNvZGUKLQotICAgIG5hdGl2ZSBwcml2YXRlIHN0YXRpYyB2b2lkIF9uYXRpdmVDbGFzc0luaXQoKTsKLSAgICBzdGF0aWMgewotCV9uYXRpdmVDbGFzc0luaXQoKTsKLSAgICB9Ci0KLSAgICBCdWZmZXIgX2NvbG9yUG9pbnRlciA9IG51bGw7Ci0gICAgQnVmZmVyIF9ub3JtYWxQb2ludGVyID0gbnVsbDsKLSAgICBCdWZmZXIgX3RleENvb3JkUG9pbnRlciA9IG51bGw7Ci0gICAgQnVmZmVyIF92ZXJ0ZXhQb2ludGVyID0gbnVsbDsKLQotICAgIHB1YmxpYyBHTDExSW1wbCgpIHsKLSAgICB9Ci0KLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL0dMQ0hlYWRlci5jcHAgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvR0xDSGVhZGVyLmNwcApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggNjQ5NTY4Ni4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvR0xDSGVhZGVyLmNwcAorKysgL2Rldi9udWxsCkBAIC0xLDEyOSArMCwwIEBACi0qKgotKiogQ29weXJpZ2h0IDIwMDYsIFRoZSBBbmRyb2lkIE9wZW4gU291cmNlIFByb2plY3QKLSoqCi0qKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsgCi0qKiB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIAotKiogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IAotKioKLSoqICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAgCi0qKgotKiogVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZSAKLSoqIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIAotKiogV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuIAotKiogU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZCAKLSoqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgotKi8KLQotLy8gVGhpcyBzb3VyY2UgZmlsZSBpcyBhdXRvbWF0aWNhbGx5IGdlbmVyYXRlZAotCi0jaW5jbHVkZSA8YW5kcm9pZF9ydW50aW1lL0FuZHJvaWRSdW50aW1lLmg+Ci0jaW5jbHVkZSA8dXRpbHMvbWlzYy5oPgotCi0jaW5jbHVkZSA8YXNzZXJ0Lmg+Ci0jaW5jbHVkZSA8R0xFUy9nbC5oPgotCi0jaW5jbHVkZSA8cHJpdmF0ZS9vcGVuZ2xlcy9nbF9jb250ZXh0Lmg+Ci0KLSNkZWZpbmUgX05VTV9DT01QUkVTU0VEX1RFWFRVUkVfRk9STUFUUyBcCi0gICAgICAgICg6OmFuZHJvaWQ6Ok9HTEVTX05VTV9DT01QUkVTU0VEX1RFWFRVUkVfRk9STUFUUykKLQotc3RhdGljIGludCBpbml0aWFsaXplZCA9IDA7Ci0KLXN0YXRpYyBqY2xhc3MgbmlvQWNjZXNzQ2xhc3M7Ci1zdGF0aWMgamNsYXNzIGJ1ZmZlckNsYXNzOwotc3RhdGljIGpjbGFzcyBPT01FQ2xhc3M7Ci1zdGF0aWMgamNsYXNzIFVPRUNsYXNzOwotc3RhdGljIGpjbGFzcyBJQUVDbGFzczsKLXN0YXRpYyBqY2xhc3MgQUlPT0JFQ2xhc3M7Ci1zdGF0aWMgam1ldGhvZElEIGdldEJhc2VQb2ludGVySUQ7Ci1zdGF0aWMgam1ldGhvZElEIGdldEJhc2VBcnJheUlEOwotc3RhdGljIGptZXRob2RJRCBnZXRCYXNlQXJyYXlPZmZzZXRJRDsKLXN0YXRpYyBqZmllbGRJRCBwb3NpdGlvbklEOwotc3RhdGljIGpmaWVsZElEIGxpbWl0SUQ7Ci1zdGF0aWMgamZpZWxkSUQgZWxlbWVudFNpemVTaGlmdElEOwotCi0vKiBDYWNoZSBtZXRob2QgSURzIGVhY2ggdGltZSB0aGUgY2xhc3MgaXMgbG9hZGVkLiAqLwotCi12b2lkCi1uYXRpdmVDbGFzc0luaXRCdWZmZXIoSk5JRW52ICpfZW52KQotewotICAgIGpjbGFzcyBuaW9BY2Nlc3NDbGFzc0xvY2FsID0gX2Vudi0+RmluZENsYXNzKCJqYXZhL25pby9OSU9BY2Nlc3MiKTsKLSAgICBuaW9BY2Nlc3NDbGFzcyA9IChqY2xhc3MpIF9lbnYtPk5ld0dsb2JhbFJlZihuaW9BY2Nlc3NDbGFzc0xvY2FsKTsKLQotICAgIGpjbGFzcyBidWZmZXJDbGFzc0xvY2FsID0gX2Vudi0+RmluZENsYXNzKCJqYXZhL25pby9CdWZmZXIiKTsKLSAgICBidWZmZXJDbGFzcyA9IChqY2xhc3MpIF9lbnYtPk5ld0dsb2JhbFJlZihidWZmZXJDbGFzc0xvY2FsKTsKLQotICAgIGdldEJhc2VQb2ludGVySUQgPSBfZW52LT5HZXRTdGF0aWNNZXRob2RJRChuaW9BY2Nlc3NDbGFzcywKLSAgICAgICAgICAgICJnZXRCYXNlUG9pbnRlciIsICIoTGphdmEvbmlvL0J1ZmZlcjspSiIpOwotICAgIGdldEJhc2VBcnJheUlEID0gX2Vudi0+R2V0U3RhdGljTWV0aG9kSUQobmlvQWNjZXNzQ2xhc3MsCi0gICAgICAgICAgICAiZ2V0QmFzZUFycmF5IiwgIihMamF2YS9uaW8vQnVmZmVyOylMamF2YS9sYW5nL09iamVjdDsiKTsKLSAgICBnZXRCYXNlQXJyYXlPZmZzZXRJRCA9IF9lbnYtPkdldFN0YXRpY01ldGhvZElEKG5pb0FjY2Vzc0NsYXNzLAotICAgICAgICAgICAgImdldEJhc2VBcnJheU9mZnNldCIsICIoTGphdmEvbmlvL0J1ZmZlcjspSSIpOwotCi0gICAgcG9zaXRpb25JRCA9IF9lbnYtPkdldEZpZWxkSUQoYnVmZmVyQ2xhc3MsICJwb3NpdGlvbiIsICJJIik7Ci0gICAgbGltaXRJRCA9IF9lbnYtPkdldEZpZWxkSUQoYnVmZmVyQ2xhc3MsICJsaW1pdCIsICJJIik7Ci0gICAgZWxlbWVudFNpemVTaGlmdElEID0KLSAgICAgICAgX2Vudi0+R2V0RmllbGRJRChidWZmZXJDbGFzcywgIl9lbGVtZW50U2l6ZVNoaWZ0IiwgIkkiKTsKLX0KLQotCi1zdGF0aWMgdm9pZAotbmF0aXZlQ2xhc3NJbml0KEpOSUVudiAqX2VudiwgamNsYXNzIGdsSW1wbENsYXNzKQotewotICAgIG5hdGl2ZUNsYXNzSW5pdEJ1ZmZlcihfZW52KTsKLQotICAgIGpjbGFzcyBJQUVDbGFzc0xvY2FsID0KLSAgICAgICAgX2Vudi0+RmluZENsYXNzKCJqYXZhL2xhbmcvSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uIik7Ci0gICAgamNsYXNzIE9PTUVDbGFzc0xvY2FsID0KLSAgICAgICAgIF9lbnYtPkZpbmRDbGFzcygiamF2YS9sYW5nL091dE9mTWVtb3J5RXJyb3IiKTsKLSAgICBqY2xhc3MgVU9FQ2xhc3NMb2NhbCA9Ci0gICAgICAgICBfZW52LT5GaW5kQ2xhc3MoImphdmEvbGFuZy9VbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbiIpOwotICAgIGpjbGFzcyBBSU9PQkVDbGFzc0xvY2FsID0KLSAgICAgICAgIF9lbnYtPkZpbmRDbGFzcygiamF2YS9sYW5nL0FycmF5SW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbiIpOwotCi0gICAgSUFFQ2xhc3MgPSAoamNsYXNzKSBfZW52LT5OZXdHbG9iYWxSZWYoSUFFQ2xhc3NMb2NhbCk7Ci0gICAgT09NRUNsYXNzID0gKGpjbGFzcykgX2Vudi0+TmV3R2xvYmFsUmVmKE9PTUVDbGFzc0xvY2FsKTsKLSAgICBVT0VDbGFzcyA9IChqY2xhc3MpIF9lbnYtPk5ld0dsb2JhbFJlZihVT0VDbGFzc0xvY2FsKTsKLSAgICBBSU9PQkVDbGFzcyA9IChqY2xhc3MpIF9lbnYtPk5ld0dsb2JhbFJlZihBSU9PQkVDbGFzc0xvY2FsKTsKLX0KLQotc3RhdGljIHZvaWQgKgotZ2V0UG9pbnRlcihKTklFbnYgKl9lbnYsIGpvYmplY3QgYnVmZmVyLCBqYXJyYXkgKmFycmF5LCBqaW50ICpyZW1haW5pbmcpCi17Ci0gICAgamludCBwb3NpdGlvbjsKLSAgICBqaW50IGxpbWl0OwotICAgIGppbnQgZWxlbWVudFNpemVTaGlmdDsKLSAgICBqbG9uZyBwb2ludGVyOwotICAgIGppbnQgb2Zmc2V0OwotICAgIHZvaWQgKmRhdGE7Ci0KLSAgICBwb3NpdGlvbiA9IF9lbnYtPkdldEludEZpZWxkKGJ1ZmZlciwgcG9zaXRpb25JRCk7Ci0gICAgbGltaXQgPSBfZW52LT5HZXRJbnRGaWVsZChidWZmZXIsIGxpbWl0SUQpOwotICAgIGVsZW1lbnRTaXplU2hpZnQgPSBfZW52LT5HZXRJbnRGaWVsZChidWZmZXIsIGVsZW1lbnRTaXplU2hpZnRJRCk7Ci0gICAgKnJlbWFpbmluZyA9IChsaW1pdCAtIHBvc2l0aW9uKSA8PCBlbGVtZW50U2l6ZVNoaWZ0OwotICAgIHBvaW50ZXIgPSBfZW52LT5DYWxsU3RhdGljTG9uZ01ldGhvZChuaW9BY2Nlc3NDbGFzcywKLSAgICAgICAgICAgIGdldEJhc2VQb2ludGVySUQsIGJ1ZmZlcik7Ci0gICAgaWYgKHBvaW50ZXIgIT0gMEwpIHsKLSAgICAgICAgKmFycmF5ID0gTlVMTDsKLSAgICAgICAgcmV0dXJuICh2b2lkICopIChqaW50KSBwb2ludGVyOwotICAgIH0KLSAgICAKLSAgICAqYXJyYXkgPSAoamFycmF5KSBfZW52LT5DYWxsU3RhdGljT2JqZWN0TWV0aG9kKG5pb0FjY2Vzc0NsYXNzLAotICAgICAgICAgICAgZ2V0QmFzZUFycmF5SUQsIGJ1ZmZlcik7Ci0gICAgb2Zmc2V0ID0gX2Vudi0+Q2FsbFN0YXRpY0ludE1ldGhvZChuaW9BY2Nlc3NDbGFzcywKLSAgICAgICAgICAgIGdldEJhc2VBcnJheU9mZnNldElELCBidWZmZXIpOwotICAgIGRhdGEgPSBfZW52LT5HZXRQcmltaXRpdmVBcnJheUNyaXRpY2FsKCphcnJheSwgKGpib29sZWFuICopIDApOwotICAgIAotICAgIHJldHVybiAodm9pZCAqKSAoKGNoYXIgKikgZGF0YSArIG9mZnNldCk7Ci19Ci0KLQotc3RhdGljIHZvaWQKLXJlbGVhc2VQb2ludGVyKEpOSUVudiAqX2VudiwgamFycmF5IGFycmF5LCB2b2lkICpkYXRhLCBqYm9vbGVhbiBjb21taXQpCi17Ci0gICAgX2Vudi0+UmVsZWFzZVByaW1pdGl2ZUFycmF5Q3JpdGljYWwoYXJyYXksIGRhdGEsCi0JCQkJCSAgIGNvbW1pdCA/IDAgOiBKTklfQUJPUlQpOwotfQotCi0vLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQotCmRpZmYgLS1naXQgYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvR0xIZWFkZXIuamF2YS1pZiBiL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTEhlYWRlci5qYXZhLWlmCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAzYjc4ZjNkLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9HTEhlYWRlci5qYXZhLWlmCisrKyAvZGV2L251bGwKQEAgLTEsMjIgKzAsMCBAQAotLyogLy9kZXZpY2UvamF2YS9hbmRyb2lkL2phdmF4L21pY3JvZWRpdGlvbi9raHJvbm9zL29wZW5nbGVzL0dMLmphdmEKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi1wYWNrYWdlIGphdmF4Lm1pY3JvZWRpdGlvbi5raHJvbm9zLm9wZW5nbGVzOwotCi1wdWJsaWMgaW50ZXJmYWNlIEdMIHsKLX0KLQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL0dMSW1wbEhlYWRlci5qYXZhLWltcGwgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvR0xJbXBsSGVhZGVyLmphdmEtaW1wbApkZWxldGVkIGZpbGUgbW9kZSAxMDA2NDQKaW5kZXggZGIzYTQxYy4uMDAwMDAwMAotLS0gYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvR0xJbXBsSGVhZGVyLmphdmEtaW1wbAorKysgL2Rldi9udWxsCkBAIC0xLDQ4ICswLDAgQEAKLSoqCi0qKiBDb3B5cmlnaHQgMjAwNiwgVGhlIEFuZHJvaWQgT3BlbiBTb3VyY2UgUHJvamVjdAotKioKLSoqIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOyAKLSoqIHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4gCi0qKiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQgCi0qKgotKiogICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMCAKLSoqCi0qKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIAotKiogZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywgCi0qKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4gCi0qKiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIAotKiogbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCi0qLwotCi0vLyBUaGlzIHNvdXJjZSBmaWxlIGlzIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGVkCi0KLXBhY2thZ2UgY29tLmdvb2dsZS5hbmRyb2lkLmdsZXNfam5pOwotCi1pbXBvcnQgamF2YS5uaW8uQnVmZmVyOwotaW1wb3J0IGphdmF4Lm1pY3JvZWRpdGlvbi5raHJvbm9zLm9wZW5nbGVzLkdMMTA7Ci1pbXBvcnQgamF2YXgubWljcm9lZGl0aW9uLmtocm9ub3Mub3BlbmdsZXMuR0wxMEV4dDsKLWltcG9ydCBqYXZheC5taWNyb2VkaXRpb24ua2hyb25vcy5vcGVuZ2xlcy5HTDExOwotaW1wb3J0IGphdmF4Lm1pY3JvZWRpdGlvbi5raHJvbm9zLm9wZW5nbGVzLkdMMTFFeHQ7Ci1pbXBvcnQgamF2YXgubWljcm9lZGl0aW9uLmtocm9ub3Mub3BlbmdsZXMuR0wxMUV4dGVuc2lvblBhY2s7Ci0KLXB1YmxpYyBjbGFzcyBHTEltcGwgaW1wbGVtZW50cyBHTDEwLCBHTDEwRXh0LCBHTDExLCBHTDExRXh0LCBHTDExRXh0ZW5zaW9uUGFjayB7Ci0KLSAgICAvLyBQcml2YXRlIGFjY2Vzc29ycyBmb3IgbmF0aXZlIGNvZGUKLQotICAgIG5hdGl2ZSBwcml2YXRlIHN0YXRpYyB2b2lkIF9uYXRpdmVDbGFzc0luaXQoKTsKLSAgICBzdGF0aWMgewotCV9uYXRpdmVDbGFzc0luaXQoKTsKLSAgICB9Ci0KLSAgICBCdWZmZXIgX2NvbG9yUG9pbnRlciA9IG51bGw7Ci0gICAgQnVmZmVyIF9ub3JtYWxQb2ludGVyID0gbnVsbDsKLSAgICBCdWZmZXIgX3RleENvb3JkUG9pbnRlciA9IG51bGw7Ci0gICAgQnVmZmVyIF92ZXJ0ZXhQb2ludGVyID0gbnVsbDsKLQotICAgIHB1YmxpYyBHTEltcGwoKSB7Ci0gICAgfQotCi0gICAgIHB1YmxpYyB2b2lkIGdsR2V0UG9pbnRlcnYoaW50IHBuYW1lLCBqYXZhLm5pby5CdWZmZXJbXSBwYXJhbXMpIHsKLSAgICAgICAgIHRocm93IG5ldyBVbnN1cHBvcnRlZE9wZXJhdGlvbkV4Y2VwdGlvbigiZ2xHZXRQb2ludGVydiIpOwotICAgICB9Ci0KZGlmZiAtLWdpdCBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9nbEdldFN0cmluZy5jcHAgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvZ2xHZXRTdHJpbmcuY3BwCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCAyMmUxMjk3Li4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9nbEdldFN0cmluZy5jcHAKKysrIC9kZXYvbnVsbApAQCAtMSwxMCArMCwwIEBACi0jaW5jbHVkZSA8c3RyaW5nLmg+DQotDQotLyogY29uc3QgR0x1Ynl0ZSAqIGdsR2V0U3RyaW5nICggR0xlbnVtIG5hbWUgKSAqLw0KLWpzdHJpbmcNCi1hbmRyb2lkX2dsR2V0U3RyaW5nDQotICAoSk5JRW52ICpfZW52LCBqb2JqZWN0IF90aGlzLCBqaW50IG5hbWUpIHsNCi0gICAgY29uc3QgY2hhciAqIGNoYXJzID0gKGNvbnN0IGNoYXIgKilnbEdldFN0cmluZygoR0xlbnVtKW5hbWUpOw0KLSAgICBqc3RyaW5nIG91dHB1dCA9IF9lbnYtPk5ld1N0cmluZ1VURihjaGFycyk7DQotICAgIHJldHVybiBvdXRwdXQ7DQotfQ0KZGlmZiAtLWdpdCBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9nbEdldFN0cmluZy5qYXZhLTEwLWlmIGIvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL2dsR2V0U3RyaW5nLmphdmEtMTAtaWYKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDg5OGZhYmMuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL2dsR2V0U3RyaW5nLmphdmEtMTAtaWYKKysrIC9kZXYvbnVsbApAQCAtMSw0ICswLDAgQEAKLSAgICBwdWJsaWMgU3RyaW5nIGdsR2V0U3RyaW5nKA0KLSAgICAgICAgaW50IG5hbWUNCi0gICAgKTsNCi0NCmRpZmYgLS1naXQgYS9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvZ2xHZXRTdHJpbmcuamF2YS1pZiBiL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9nbEdldFN0cmluZy5qYXZhLWlmCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCA4OThmYWJjLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9nbEdldFN0cmluZy5qYXZhLWlmCisrKyAvZGV2L251bGwKQEAgLTEsNCArMCwwIEBACi0gICAgcHVibGljIFN0cmluZyBnbEdldFN0cmluZygNCi0gICAgICAgIGludCBuYW1lDQotICAgICk7DQotDQpkaWZmIC0tZ2l0IGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL2dsR2V0U3RyaW5nLmphdmEtaW1wbCBiL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9nbEdldFN0cmluZy5qYXZhLWltcGwKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDhjNzg4MWMuLjAwMDAwMDAKLS0tIGEvb3BlbmdsL3Rvb2xzL2dsZ2VuL3N0dWJzL2dsR2V0U3RyaW5nLmphdmEtaW1wbAorKysgL2Rldi9udWxsCkBAIC0xLDE2ICswLDAgQEAKLSAgICAvLyBDIGZ1bmN0aW9uIGNvbnN0IEdMdWJ5dGUgKiBnbEdldFN0cmluZyAoIEdMZW51bSBuYW1lICkNCi0NCi0gICAgcHVibGljIG5hdGl2ZSBTdHJpbmcgX2dsR2V0U3RyaW5nKA0KLSAgICAgICAgaW50IG5hbWUNCi0gICAgKTsNCi0NCi0gICAgcHVibGljIFN0cmluZyBnbEdldFN0cmluZygNCi0gICAgICAgIGludCBuYW1lDQotICAgICkgew0KLSAgICAgICAgU3RyaW5nIHJldHVyblZhbHVlOw0KLSAgICAgICAgcmV0dXJuVmFsdWUgPSBfZ2xHZXRTdHJpbmcoDQotICAgICAgICAgICAgbmFtZQ0KLSAgICAgICAgKTsNCi0gICAgICAgIHJldHVybiByZXR1cm5WYWx1ZTsNCi0gICAgfQ0KLQ0KZGlmZiAtLWdpdCBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9nbEdldFN0cmluZy5uYXRpdmVSZWcgYi9vcGVuZ2wvdG9vbHMvZ2xnZW4vc3R1YnMvZ2xHZXRTdHJpbmcubmF0aXZlUmVnCmRlbGV0ZWQgZmlsZSBtb2RlIDEwMDY0NAppbmRleCBlNjQxODdjLi4wMDAwMDAwCi0tLSBhL29wZW5nbC90b29scy9nbGdlbi9zdHVicy9nbEdldFN0cmluZy5uYXRpdmVSZWcKKysrIC9kZXYvbnVsbApAQCAtMSArMCwwIEBACi17Il9nbEdldFN0cmluZyIsICIoSSlMamF2YS9sYW5nL1N0cmluZzsiLCAodm9pZCAqKSBhbmRyb2lkX2dsR2V0U3RyaW5nIH0sDQpkaWZmIC0tZ2l0IGEvc2VydmljZXMvQW5kcm9pZC5tayBiL3NlcnZpY2VzL0FuZHJvaWQubWsKZGVsZXRlZCBmaWxlIG1vZGUgMTAwNjQ0CmluZGV4IDVlOTEyZDYuLjAwMDAwMDAKLS0tIGEvc2VydmljZXMvQW5kcm9pZC5taworKysgL2Rldi9udWxsCkBAIC0xLDE3ICswLDAgQEAKLUxPQ0FMX1BBVEg6PSAkKGNhbGwgbXktZGlyKQotCi0jIHRoZSBsaWJyYXJ5Ci0jID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQotaW5jbHVkZSAkKENMRUFSX1ZBUlMpCi0KLUxPQ0FMX1NSQ19GSUxFUyA6PSBcCi0gICAgICAgICAgICAkKGNhbGwgYWxsLXN1YmRpci1qYXZhLWZpbGVzKQotCi1MT0NBTF9NT0RVTEU6PSBzZXJ2aWNlcwotCi1MT0NBTF9KQVZBX0xJQlJBUklFUyA6PSBhbmRyb2lkLnBvbGljeQotCi1pbmNsdWRlICQoQlVJTERfSkFWQV9MSUJSQVJZKQotCi1pbmNsdWRlICQoQlVJTERfRFJPSURET0MpCi0K